summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorunknown <magnus@neptunus.(none)>2004-04-14 10:53:21 +0200
committerunknown <magnus@neptunus.(none)>2004-04-14 10:53:21 +0200
commit6386c55cee50bad6a9979d1fab28e03bb8612ca7 (patch)
tree3fbbacf704304b69228474b9f03549ccd585a017
parent0ba6cb48d84f1ff951d09871a96be6cdef3f2c3c (diff)
downloadmariadb-git-6386c55cee50bad6a9979d1fab28e03bb8612ca7.tar.gz
Initial revision of NDB Cluster files
BitKeeper/etc/logging_ok: Logging to logging@openlogging.org accepted
-rw-r--r--BitKeeper/etc/logging_ok1
-rw-r--r--ndb/BinDist.sh121
-rw-r--r--ndb/Defs.mk84
-rw-r--r--ndb/Epilogue.mk853
-rw-r--r--ndb/Makefile62
-rw-r--r--ndb/README7
-rw-r--r--ndb/bin/.empty0
-rwxr-xr-xndb/bin/check-regression.sh180
-rwxr-xr-xndb/bin/makeTestPrograms_html.sh22
-rwxr-xr-xndb/bin/mysqlcluster11
-rwxr-xr-xndb/bin/mysqlcluster_install_db119
-rwxr-xr-xndb/bin/mysqlclusterd34
-rw-r--r--ndb/bin/regression.sh644
-rw-r--r--ndb/config/Defs.DEBUG.mk4
-rw-r--r--ndb/config/Defs.HPUX.HPPA.GCC.mk50
-rw-r--r--ndb/config/Defs.IBMAIX.POWERPC.GCC.mk49
-rw-r--r--ndb/config/Defs.LINUX.x86.GCC.mk56
-rw-r--r--ndb/config/Defs.LINUX.x86.ICC.mk54
-rw-r--r--ndb/config/Defs.LINUX.x86_64.GCC.mk54
-rw-r--r--ndb/config/Defs.MACOSX.POWERPC.GCC.mk58
-rw-r--r--ndb/config/Defs.OSE.PPC750.DIAB.mk47
-rw-r--r--ndb/config/Defs.RELEASE.mk3
-rw-r--r--ndb/config/Defs.RELEASE_TRACE.mk3
-rw-r--r--ndb/config/Defs.SIMCELLO.SOFTOSE.GCC.mk53
-rw-r--r--ndb/config/Defs.SOFTOSE.SPARC.GCC.mk57
-rw-r--r--ndb/config/Defs.SOLARIS.SPARC.FORTE6.mk54
-rw-r--r--ndb/config/Defs.SOLARIS.SPARC.GCC.mk54
-rw-r--r--ndb/config/Defs.SOLARIS.SPARC_64.GCC.mk53
-rw-r--r--ndb/config/Defs.SOLARIS6.SPARC.GCC.mk53
-rw-r--r--ndb/config/Defs.TRU64X.ALPHA.GCC.mk49
-rw-r--r--ndb/config/Defs.WIN32.x86.VC7.mk61
-rwxr-xr-xndb/config/GuessConfig.sh113
-rw-r--r--ndb/config/Makefile.am31
-rw-r--r--ndb/config/acinclude.m41513
-rw-r--r--ndb/config/configure.in2085
-rw-r--r--ndb/demos/1-node/1-api-3/Ndb.cfg2
-rw-r--r--ndb/demos/1-node/1-db-2/Ndb.cfg2
-rw-r--r--ndb/demos/1-node/1-mgm-1/Ndb.cfg2
-rw-r--r--ndb/demos/1-node/1-mgm-1/template_config.ini70
-rw-r--r--ndb/demos/2-node/2-api-4/Ndb.cfg2
-rw-r--r--ndb/demos/2-node/2-api-5/Ndb.cfg2
-rw-r--r--ndb/demos/2-node/2-api-6/Ndb.cfg2
-rw-r--r--ndb/demos/2-node/2-api-7/Ndb.cfg2
-rw-r--r--ndb/demos/2-node/2-db-2/Ndb.cfg2
-rw-r--r--ndb/demos/2-node/2-db-3/Ndb.cfg2
-rw-r--r--ndb/demos/2-node/2-mgm-1/Ndb.cfg2
-rw-r--r--ndb/demos/2-node/2-mgm-1/template_config.ini157
-rw-r--r--ndb/demos/config-templates/config_template-1-REP.ini87
-rw-r--r--ndb/demos/config-templates/config_template-4.ini336
-rw-r--r--ndb/demos/config-templates/config_template-install.ini64
-rw-r--r--ndb/demos/run_demo1-PS-SS_common.sh50
-rwxr-xr-xndb/demos/run_demo1-PS.sh30
-rwxr-xr-xndb/demos/run_demo1-SS.sh30
-rwxr-xr-xndb/demos/run_demo1.sh41
-rwxr-xr-xndb/demos/run_demo2.sh54
-rw-r--r--ndb/docs/Makefile97
-rw-r--r--ndb/docs/README30
-rw-r--r--ndb/docs/doxygen/Doxyfile.mgmapi877
-rw-r--r--ndb/docs/doxygen/Doxyfile.ndb937
-rw-r--r--ndb/docs/doxygen/Doxyfile.ndbapi877
-rw-r--r--ndb/docs/doxygen/Doxyfile.odbc921
-rw-r--r--ndb/docs/doxygen/Doxyfile.test921
-rw-r--r--ndb/docs/doxygen/header.mgmapi.tex44
-rw-r--r--ndb/docs/doxygen/header.ndbapi.tex44
-rwxr-xr-xndb/docs/doxygen/postdoxy.pl97
-rwxr-xr-xndb/docs/doxygen/predoxy.pl34
-rw-r--r--ndb/env.sh8
-rw-r--r--ndb/examples/Makefile26
-rw-r--r--ndb/examples/configurations/demos.tarbin0 -> 40960 bytes
-rw-r--r--ndb/examples/ndbapi_async_example/Makefile34
-rw-r--r--ndb/examples/ndbapi_async_example/ndbapi_async.cpp505
-rw-r--r--ndb/examples/ndbapi_async_example/readme.txt3
-rw-r--r--ndb/examples/ndbapi_example1/Makefile33
-rw-r--r--ndb/examples/ndbapi_example1/ndbapi_example1.cpp193
-rw-r--r--ndb/examples/ndbapi_example2/Makefile33
-rw-r--r--ndb/examples/ndbapi_example2/ndbapi_example2.cpp110
-rw-r--r--ndb/examples/ndbapi_example3/Makefile33
-rw-r--r--ndb/examples/ndbapi_example3/ndbapi_example3.cpp202
-rw-r--r--ndb/examples/ndbapi_example4/Makefile33
-rw-r--r--ndb/examples/ndbapi_example4/ndbapi_example4.cpp252
-rw-r--r--ndb/examples/ndbapi_example5/Makefile33
-rw-r--r--ndb/examples/ndbapi_example5/ndbapi_example5.cpp230
-rw-r--r--ndb/examples/ndbapi_scan_example/Makefile35
-rw-r--r--ndb/examples/ndbapi_scan_example/ndbapi_scan.cpp824
-rw-r--r--ndb/examples/ndbapi_scan_example/readme.txt3
-rw-r--r--ndb/examples/select_all/Makefile33
-rw-r--r--ndb/examples/select_all/select_all.cpp258
-rwxr-xr-xndb/home/bin/Linuxmkisofsbin0 -> 503146 bytes
-rwxr-xr-xndb/home/bin/Solarismkisofsbin0 -> 634084 bytes
-rwxr-xr-xndb/home/bin/cvs2cl.pl1865
-rwxr-xr-xndb/home/bin/cvschk569
-rwxr-xr-xndb/home/bin/fix-cvs-root17
-rwxr-xr-xndb/home/bin/import-from-bk.sh158
-rwxr-xr-xndb/home/bin/ndb_deploy27
-rwxr-xr-xndb/home/bin/ndbdoxy.pl184
-rwxr-xr-xndb/home/bin/ngcalc78
-rw-r--r--ndb/home/bin/signallog2html.lib/signallog2list.awk102
-rw-r--r--ndb/home/bin/signallog2html.lib/uniq_blocks.awk29
-rwxr-xr-xndb/home/bin/signallog2html.sh349
-rwxr-xr-xndb/home/bin/stripcr90
-rw-r--r--ndb/include/debugger/DebuggerNames.hpp71
-rw-r--r--ndb/include/debugger/EventLogger.hpp234
-rw-r--r--ndb/include/debugger/GrepError.hpp94
-rw-r--r--ndb/include/debugger/SignalLoggerManager.hpp169
-rw-r--r--ndb/include/editline/editline.h38
-rw-r--r--ndb/include/kernel/AttributeDescriptor.hpp250
-rw-r--r--ndb/include/kernel/AttributeHeader.hpp204
-rw-r--r--ndb/include/kernel/AttributeList.hpp32
-rw-r--r--ndb/include/kernel/BlockNumbers.h81
-rw-r--r--ndb/include/kernel/GlobalSignalNumbers.h957
-rw-r--r--ndb/include/kernel/GrepEvent.hpp59
-rw-r--r--ndb/include/kernel/Interpreter.hpp284
-rw-r--r--ndb/include/kernel/LogLevel.hpp173
-rw-r--r--ndb/include/kernel/NodeBitmask.hpp89
-rw-r--r--ndb/include/kernel/NodeInfo.hpp94
-rw-r--r--ndb/include/kernel/NodeState.hpp308
-rw-r--r--ndb/include/kernel/RefConvert.hpp47
-rw-r--r--ndb/include/kernel/kernel_types.h43
-rw-r--r--ndb/include/kernel/ndb_limits.h94
-rw-r--r--ndb/include/kernel/signaldata/AbortAll.hpp88
-rw-r--r--ndb/include/kernel/signaldata/AccFrag.hpp89
-rw-r--r--ndb/include/kernel/signaldata/AccLock.hpp65
-rw-r--r--ndb/include/kernel/signaldata/AccScan.hpp164
-rw-r--r--ndb/include/kernel/signaldata/AccSizeAltReq.hpp53
-rw-r--r--ndb/include/kernel/signaldata/AlterIndx.hpp268
-rw-r--r--ndb/include/kernel/signaldata/AlterTab.hpp125
-rw-r--r--ndb/include/kernel/signaldata/AlterTable.hpp179
-rw-r--r--ndb/include/kernel/signaldata/AlterTrig.hpp288
-rw-r--r--ndb/include/kernel/signaldata/ApiRegSignalData.hpp92
-rw-r--r--ndb/include/kernel/signaldata/ApiVersion.hpp60
-rw-r--r--ndb/include/kernel/signaldata/ArbitSignalData.hpp154
-rw-r--r--ndb/include/kernel/signaldata/AttrInfo.hpp52
-rw-r--r--ndb/include/kernel/signaldata/BackupContinueB.hpp38
-rw-r--r--ndb/include/kernel/signaldata/BackupImpl.hpp366
-rw-r--r--ndb/include/kernel/signaldata/BackupSignalData.hpp252
-rw-r--r--ndb/include/kernel/signaldata/BlockCommitOrd.hpp62
-rw-r--r--ndb/include/kernel/signaldata/BuildIndx.hpp308
-rw-r--r--ndb/include/kernel/signaldata/CheckNodeGroups.hpp63
-rw-r--r--ndb/include/kernel/signaldata/CloseComReqConf.hpp53
-rw-r--r--ndb/include/kernel/signaldata/CmInit.hpp48
-rw-r--r--ndb/include/kernel/signaldata/CmRegSignalData.hpp192
-rw-r--r--ndb/include/kernel/signaldata/CmvmiCfgConf.hpp49
-rw-r--r--ndb/include/kernel/signaldata/CntrMasterConf.hpp47
-rw-r--r--ndb/include/kernel/signaldata/CntrMasterReq.hpp50
-rw-r--r--ndb/include/kernel/signaldata/ConfigParamId.hpp71
-rw-r--r--ndb/include/kernel/signaldata/ContinueFragmented.hpp36
-rw-r--r--ndb/include/kernel/signaldata/CopyActive.hpp84
-rw-r--r--ndb/include/kernel/signaldata/CopyFrag.hpp87
-rw-r--r--ndb/include/kernel/signaldata/CopyGCIReq.hpp63
-rw-r--r--ndb/include/kernel/signaldata/CreateEvnt.hpp488
-rw-r--r--ndb/include/kernel/signaldata/CreateFrag.hpp61
-rw-r--r--ndb/include/kernel/signaldata/CreateFragmentation.hpp101
-rw-r--r--ndb/include/kernel/signaldata/CreateIndx.hpp295
-rw-r--r--ndb/include/kernel/signaldata/CreateTab.hpp108
-rw-r--r--ndb/include/kernel/signaldata/CreateTable.hpp140
-rw-r--r--ndb/include/kernel/signaldata/CreateTrig.hpp414
-rw-r--r--ndb/include/kernel/signaldata/DiAddTab.hpp90
-rw-r--r--ndb/include/kernel/signaldata/DiGetNodes.hpp62
-rw-r--r--ndb/include/kernel/signaldata/DictSchemaInfo.hpp45
-rw-r--r--ndb/include/kernel/signaldata/DictSizeAltReq.hpp51
-rw-r--r--ndb/include/kernel/signaldata/DictStart.hpp54
-rw-r--r--ndb/include/kernel/signaldata/DictTabInfo.hpp483
-rw-r--r--ndb/include/kernel/signaldata/DihAddFrag.hpp62
-rw-r--r--ndb/include/kernel/signaldata/DihContinueB.hpp75
-rw-r--r--ndb/include/kernel/signaldata/DihSizeAltReq.hpp50
-rw-r--r--ndb/include/kernel/signaldata/DihStartTab.hpp65
-rw-r--r--ndb/include/kernel/signaldata/DihSwitchReplica.hpp72
-rw-r--r--ndb/include/kernel/signaldata/DisconnectRep.hpp61
-rw-r--r--ndb/include/kernel/signaldata/DropIndx.hpp253
-rw-r--r--ndb/include/kernel/signaldata/DropTab.hpp114
-rw-r--r--ndb/include/kernel/signaldata/DropTabFile.hpp64
-rw-r--r--ndb/include/kernel/signaldata/DropTable.hpp80
-rw-r--r--ndb/include/kernel/signaldata/DropTrig.hpp300
-rw-r--r--ndb/include/kernel/signaldata/DumpStateOrd.hpp134
-rw-r--r--ndb/include/kernel/signaldata/EmptyLcp.hpp77
-rw-r--r--ndb/include/kernel/signaldata/EndTo.hpp49
-rw-r--r--ndb/include/kernel/signaldata/EventReport.hpp159
-rw-r--r--ndb/include/kernel/signaldata/EventSubscribeReq.hpp60
-rw-r--r--ndb/include/kernel/signaldata/ExecFragReq.hpp43
-rw-r--r--ndb/include/kernel/signaldata/FailRep.hpp56
-rw-r--r--ndb/include/kernel/signaldata/FireTrigOrd.hpp200
-rw-r--r--ndb/include/kernel/signaldata/FsAppendReq.hpp57
-rw-r--r--ndb/include/kernel/signaldata/FsCloseReq.hpp85
-rw-r--r--ndb/include/kernel/signaldata/FsConf.hpp77
-rw-r--r--ndb/include/kernel/signaldata/FsOpenReq.hpp266
-rw-r--r--ndb/include/kernel/signaldata/FsReadWriteReq.hpp152
-rw-r--r--ndb/include/kernel/signaldata/FsRef.hpp116
-rw-r--r--ndb/include/kernel/signaldata/FsRemoveReq.hpp78
-rw-r--r--ndb/include/kernel/signaldata/GCPSave.hpp98
-rw-r--r--ndb/include/kernel/signaldata/GetTabInfo.hpp126
-rw-r--r--ndb/include/kernel/signaldata/GetTableId.hpp93
-rw-r--r--ndb/include/kernel/signaldata/GrepImpl.hpp891
-rw-r--r--ndb/include/kernel/signaldata/HotSpareRep.hpp48
-rwxr-xr-xndb/include/kernel/signaldata/IndxAttrInfo.hpp56
-rwxr-xr-xndb/include/kernel/signaldata/IndxKeyInfo.hpp56
-rw-r--r--ndb/include/kernel/signaldata/InvalidateNodeLCPConf.hpp41
-rw-r--r--ndb/include/kernel/signaldata/InvalidateNodeLCPReq.hpp42
-rw-r--r--ndb/include/kernel/signaldata/KeyInfo.hpp45
-rw-r--r--ndb/include/kernel/signaldata/LCP.hpp154
-rw-r--r--ndb/include/kernel/signaldata/ListTables.hpp166
-rw-r--r--ndb/include/kernel/signaldata/LqhFrag.hpp252
-rw-r--r--ndb/include/kernel/signaldata/LqhKey.hpp534
-rw-r--r--ndb/include/kernel/signaldata/LqhSizeAltReq.hpp53
-rw-r--r--ndb/include/kernel/signaldata/LqhTransConf.hpp218
-rw-r--r--ndb/include/kernel/signaldata/ManagementServer.hpp87
-rw-r--r--ndb/include/kernel/signaldata/MasterGCP.hpp84
-rw-r--r--ndb/include/kernel/signaldata/MasterLCP.hpp86
-rw-r--r--ndb/include/kernel/signaldata/NFCompleteRep.hpp80
-rw-r--r--ndb/include/kernel/signaldata/NdbSttor.hpp85
-rw-r--r--ndb/include/kernel/signaldata/NdbfsContinueB.hpp35
-rw-r--r--ndb/include/kernel/signaldata/NextScan.hpp67
-rw-r--r--ndb/include/kernel/signaldata/NodeFailRep.hpp68
-rw-r--r--ndb/include/kernel/signaldata/NodeStateSignalData.hpp94
-rw-r--r--ndb/include/kernel/signaldata/PackedSignal.hpp43
-rw-r--r--ndb/include/kernel/signaldata/PrepDropTab.hpp170
-rw-r--r--ndb/include/kernel/signaldata/PrepFailReqRef.hpp49
-rw-r--r--ndb/include/kernel/signaldata/ReadNodesConf.hpp109
-rw-r--r--ndb/include/kernel/signaldata/RelTabMem.hpp69
-rw-r--r--ndb/include/kernel/signaldata/RepImpl.hpp500
-rw-r--r--ndb/include/kernel/signaldata/ResumeReq.hpp69
-rw-r--r--ndb/include/kernel/signaldata/ScanFrag.hpp325
-rw-r--r--ndb/include/kernel/signaldata/ScanTab.hpp453
-rw-r--r--ndb/include/kernel/signaldata/SetLogLevelOrd.hpp70
-rw-r--r--ndb/include/kernel/signaldata/SetVarReq.hpp84
-rw-r--r--ndb/include/kernel/signaldata/SignalData.hpp65
-rw-r--r--ndb/include/kernel/signaldata/SignalDataPrint.hpp36
-rw-r--r--ndb/include/kernel/signaldata/SignalDroppedRep.hpp44
-rw-r--r--ndb/include/kernel/signaldata/SrFragidConf.hpp43
-rw-r--r--ndb/include/kernel/signaldata/StartFragReq.hpp47
-rw-r--r--ndb/include/kernel/signaldata/StartInfo.hpp84
-rw-r--r--ndb/include/kernel/signaldata/StartMe.hpp63
-rw-r--r--ndb/include/kernel/signaldata/StartOrd.hpp48
-rw-r--r--ndb/include/kernel/signaldata/StartPerm.hpp68
-rw-r--r--ndb/include/kernel/signaldata/StartRec.hpp61
-rw-r--r--ndb/include/kernel/signaldata/StartTo.hpp50
-rw-r--r--ndb/include/kernel/signaldata/StopMe.hpp70
-rw-r--r--ndb/include/kernel/signaldata/StopPerm.hpp96
-rw-r--r--ndb/include/kernel/signaldata/StopReq.hpp202
-rw-r--r--ndb/include/kernel/signaldata/SumaImpl.hpp619
-rw-r--r--ndb/include/kernel/signaldata/SystemError.hpp60
-rw-r--r--ndb/include/kernel/signaldata/TamperOrd.hpp40
-rw-r--r--ndb/include/kernel/signaldata/TcCommit.hpp74
-rw-r--r--ndb/include/kernel/signaldata/TcContinueB.hpp49
-rw-r--r--ndb/include/kernel/signaldata/TcHbRep.hpp64
-rw-r--r--ndb/include/kernel/signaldata/TcIndx.hpp528
-rw-r--r--ndb/include/kernel/signaldata/TcKeyConf.hpp131
-rw-r--r--ndb/include/kernel/signaldata/TcKeyFailConf.hpp53
-rw-r--r--ndb/include/kernel/signaldata/TcKeyRef.hpp52
-rw-r--r--ndb/include/kernel/signaldata/TcKeyReq.hpp547
-rw-r--r--ndb/include/kernel/signaldata/TcRollbackRep.hpp50
-rw-r--r--ndb/include/kernel/signaldata/TcSizeAltReq.hpp52
-rw-r--r--ndb/include/kernel/signaldata/TestOrd.hpp229
-rwxr-xr-xndb/include/kernel/signaldata/TransIdAI.hpp59
-rw-r--r--ndb/include/kernel/signaldata/TrigAttrInfo.hpp138
-rw-r--r--ndb/include/kernel/signaldata/TupAccess.hpp172
-rw-r--r--ndb/include/kernel/signaldata/TupCommit.hpp51
-rw-r--r--ndb/include/kernel/signaldata/TupFrag.hpp188
-rw-r--r--ndb/include/kernel/signaldata/TupKey.hpp126
-rw-r--r--ndb/include/kernel/signaldata/TupSizeAltReq.hpp58
-rw-r--r--ndb/include/kernel/signaldata/TuxBound.hpp56
-rw-r--r--ndb/include/kernel/signaldata/TuxContinueB.hpp30
-rw-r--r--ndb/include/kernel/signaldata/TuxMaint.hpp66
-rw-r--r--ndb/include/kernel/signaldata/TuxSizeAltReq.hpp48
-rw-r--r--ndb/include/kernel/signaldata/UpdateTo.hpp59
-rw-r--r--ndb/include/kernel/signaldata/UtilDelete.hpp121
-rw-r--r--ndb/include/kernel/signaldata/UtilExecute.hpp136
-rw-r--r--ndb/include/kernel/signaldata/UtilLock.hpp334
-rw-r--r--ndb/include/kernel/signaldata/UtilPrepare.hpp161
-rw-r--r--ndb/include/kernel/signaldata/UtilRelease.hpp83
-rw-r--r--ndb/include/kernel/signaldata/UtilSequence.hpp101
-rw-r--r--ndb/include/kernel/signaldata/WaitGCP.hpp109
-rw-r--r--ndb/include/kernel/trigger_definitions.h66
-rw-r--r--ndb/include/logger/ConsoleLogHandler.hpp57
-rw-r--r--ndb/include/logger/FileLogHandler.hpp110
-rw-r--r--ndb/include/logger/LogHandler.hpp198
-rw-r--r--ndb/include/logger/Logger.hpp294
-rw-r--r--ndb/include/logger/SysLogHandler.hpp97
-rw-r--r--ndb/include/mgmapi/mgmapi.h663
-rw-r--r--ndb/include/mgmapi/mgmapi_debug.h114
-rw-r--r--ndb/include/mgmcommon/ConfigRetriever.hpp116
-rw-r--r--ndb/include/mgmcommon/IPCConfig.hpp79
-rw-r--r--ndb/include/mgmcommon/MgmtErrorReporter.hpp74
-rw-r--r--ndb/include/mgmcommon/NdbConfig.h34
-rw-r--r--ndb/include/ndb_types.h51
-rw-r--r--ndb/include/ndb_version.h53
-rw-r--r--ndb/include/ndbapi/AttrType.hpp329
-rw-r--r--ndb/include/ndbapi/Ndb.hpp1702
-rw-r--r--ndb/include/ndbapi/NdbApi.hpp33
-rw-r--r--ndb/include/ndbapi/NdbConnection.hpp885
-rw-r--r--ndb/include/ndbapi/NdbCursorOperation.hpp93
-rw-r--r--ndb/include/ndbapi/NdbDictionary.hpp1033
-rw-r--r--ndb/include/ndbapi/NdbError.hpp212
-rw-r--r--ndb/include/ndbapi/NdbEventOperation.hpp205
-rw-r--r--ndb/include/ndbapi/NdbIndexOperation.hpp192
-rw-r--r--ndb/include/ndbapi/NdbOperation.hpp1338
-rw-r--r--ndb/include/ndbapi/NdbPool.hpp35
-rw-r--r--ndb/include/ndbapi/NdbRecAttr.hpp512
-rw-r--r--ndb/include/ndbapi/NdbReceiver.hpp72
-rw-r--r--ndb/include/ndbapi/NdbResultSet.hpp114
-rw-r--r--ndb/include/ndbapi/NdbScanFilter.hpp177
-rw-r--r--ndb/include/ndbapi/NdbScanOperation.hpp248
-rw-r--r--ndb/include/ndbapi/NdbSchemaCon.hpp132
-rw-r--r--ndb/include/ndbapi/NdbSchemaOp.hpp458
-rw-r--r--ndb/include/ndbapi/ndbapi_limits.h47
-rw-r--r--ndb/include/newtonapi/dba.h732
-rw-r--r--ndb/include/newtonapi/defs/pcn_types.h41
-rw-r--r--ndb/include/portlib/NdbCondition.h94
-rw-r--r--ndb/include/portlib/NdbConstant.hpp28
-rw-r--r--ndb/include/portlib/NdbDaemon.h72
-rw-r--r--ndb/include/portlib/NdbEnv.h34
-rw-r--r--ndb/include/portlib/NdbHost.h43
-rw-r--r--ndb/include/portlib/NdbMain.h66
-rw-r--r--ndb/include/portlib/NdbMem.h82
-rw-r--r--ndb/include/portlib/NdbMutex.h114
-rw-r--r--ndb/include/portlib/NdbSleep.h38
-rw-r--r--ndb/include/portlib/NdbStdio.h36
-rw-r--r--ndb/include/portlib/NdbTCP.h137
-rw-r--r--ndb/include/portlib/NdbThread.h103
-rw-r--r--ndb/include/portlib/NdbTick.h69
-rw-r--r--ndb/include/portlib/NdbUnistd.h39
-rw-r--r--ndb/include/portlib/PortDefs.h96
-rw-r--r--ndb/include/portlib/prefetch.h69
-rw-r--r--ndb/include/transporter/TransporterCallback.hpp345
-rw-r--r--ndb/include/transporter/TransporterDefinitions.hpp152
-rw-r--r--ndb/include/transporter/TransporterRegistry.hpp281
-rw-r--r--ndb/include/util/Base64.hpp26
-rw-r--r--ndb/include/util/BaseString.hpp260
-rw-r--r--ndb/include/util/Bitmask.hpp755
-rw-r--r--ndb/include/util/File.hpp206
-rw-r--r--ndb/include/util/InputStream.hpp48
-rw-r--r--ndb/include/util/NdbAutoPtr.hpp49
-rw-r--r--ndb/include/util/NdbOut.hpp132
-rw-r--r--ndb/include/util/NdbSqlUtil.hpp357
-rw-r--r--ndb/include/util/NdbString.h48
-rw-r--r--ndb/include/util/OutputStream.hpp67
-rw-r--r--ndb/include/util/Parser.hpp290
-rw-r--r--ndb/include/util/Properties.hpp246
-rw-r--r--ndb/include/util/SimpleProperties.hpp290
-rw-r--r--ndb/include/util/SocketServer.hpp132
-rw-r--r--ndb/include/util/UtilBuffer.hpp90
-rw-r--r--ndb/include/util/Vector.hpp289
-rw-r--r--ndb/include/util/getarg.h115
-rw-r--r--ndb/include/util/md5_hash.hpp25
-rw-r--r--ndb/include/util/random.h84
-rw-r--r--ndb/include/util/socket_io.h40
-rw-r--r--ndb/include/util/uucode.h36
-rw-r--r--ndb/include/util/version.h50
-rw-r--r--ndb/lib/.empty0
-rw-r--r--ndb/mysqlclusterenv.sh51
-rw-r--r--ndb/src/Makefile34
-rw-r--r--ndb/src/client/Makefile9
-rw-r--r--ndb/src/client/odbc/Extra.mk59
-rw-r--r--ndb/src/client/odbc/Makefile75
-rwxr-xr-xndb/src/client/odbc/NdbOdbc.cpp78
-rwxr-xr-xndb/src/client/odbc/NdbOdbc.def85
-rw-r--r--ndb/src/client/odbc/codegen/CodeGen.cpp229
-rw-r--r--ndb/src/client/odbc/codegen/CodeGen.hpp69
-rw-r--r--ndb/src/client/odbc/codegen/Code_base.cpp167
-rw-r--r--ndb/src/client/odbc/codegen/Code_base.hpp237
-rw-r--r--ndb/src/client/odbc/codegen/Code_column.cpp72
-rw-r--r--ndb/src/client/odbc/codegen/Code_column.hpp122
-rw-r--r--ndb/src/client/odbc/codegen/Code_comp_op.cpp485
-rw-r--r--ndb/src/client/odbc/codegen/Code_comp_op.hpp172
-rw-r--r--ndb/src/client/odbc/codegen/Code_create_index.cpp124
-rw-r--r--ndb/src/client/odbc/codegen/Code_create_index.hpp203
-rw-r--r--ndb/src/client/odbc/codegen/Code_create_row.cpp162
-rw-r--r--ndb/src/client/odbc/codegen/Code_create_row.hpp99
-rw-r--r--ndb/src/client/odbc/codegen/Code_create_table.cpp137
-rw-r--r--ndb/src/client/odbc/codegen/Code_create_table.hpp178
-rw-r--r--ndb/src/client/odbc/codegen/Code_data_type.cpp44
-rw-r--r--ndb/src/client/odbc/codegen/Code_data_type.hpp49
-rw-r--r--ndb/src/client/odbc/codegen/Code_ddl.cpp37
-rw-r--r--ndb/src/client/odbc/codegen/Code_ddl.hpp63
-rw-r--r--ndb/src/client/odbc/codegen/Code_ddl_column.cpp104
-rw-r--r--ndb/src/client/odbc/codegen/Code_ddl_column.hpp150
-rw-r--r--ndb/src/client/odbc/codegen/Code_ddl_constr.cpp51
-rw-r--r--ndb/src/client/odbc/codegen/Code_ddl_constr.hpp65
-rw-r--r--ndb/src/client/odbc/codegen/Code_ddl_row.cpp54
-rw-r--r--ndb/src/client/odbc/codegen/Code_ddl_row.hpp72
-rw-r--r--ndb/src/client/odbc/codegen/Code_delete.cpp205
-rw-r--r--ndb/src/client/odbc/codegen/Code_delete.hpp69
-rw-r--r--ndb/src/client/odbc/codegen/Code_delete_index.cpp164
-rw-r--r--ndb/src/client/odbc/codegen/Code_delete_index.hpp156
-rw-r--r--ndb/src/client/odbc/codegen/Code_delete_lookup.cpp162
-rw-r--r--ndb/src/client/odbc/codegen/Code_delete_lookup.hpp152
-rw-r--r--ndb/src/client/odbc/codegen/Code_delete_scan.cpp110
-rw-r--r--ndb/src/client/odbc/codegen/Code_delete_scan.hpp130
-rw-r--r--ndb/src/client/odbc/codegen/Code_dml.cpp51
-rw-r--r--ndb/src/client/odbc/codegen/Code_dml.hpp67
-rw-r--r--ndb/src/client/odbc/codegen/Code_dml_column.cpp47
-rw-r--r--ndb/src/client/odbc/codegen/Code_dml_column.hpp46
-rw-r--r--ndb/src/client/odbc/codegen/Code_dml_row.cpp56
-rw-r--r--ndb/src/client/odbc/codegen/Code_dml_row.hpp76
-rw-r--r--ndb/src/client/odbc/codegen/Code_drop_index.cpp87
-rw-r--r--ndb/src/client/odbc/codegen/Code_drop_index.hpp136
-rw-r--r--ndb/src/client/odbc/codegen/Code_drop_table.cpp87
-rw-r--r--ndb/src/client/odbc/codegen/Code_drop_table.hpp124
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr.cpp79
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr.hpp219
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_column.cpp160
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_column.hpp120
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_const.cpp138
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_const.hpp120
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_conv.cpp273
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_conv.hpp141
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_func.cpp401
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_func.hpp193
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_op.cpp424
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_op.hpp166
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_param.cpp279
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_param.hpp136
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_row.cpp204
-rw-r--r--ndb/src/client/odbc/codegen/Code_expr_row.hpp272
-rw-r--r--ndb/src/client/odbc/codegen/Code_idx_column.cpp49
-rw-r--r--ndb/src/client/odbc/codegen/Code_idx_column.hpp50
-rw-r--r--ndb/src/client/odbc/codegen/Code_insert.cpp253
-rw-r--r--ndb/src/client/odbc/codegen/Code_insert.hpp229
-rw-r--r--ndb/src/client/odbc/codegen/Code_pred.cpp70
-rw-r--r--ndb/src/client/odbc/codegen/Code_pred.hpp172
-rw-r--r--ndb/src/client/odbc/codegen/Code_pred_op.cpp188
-rw-r--r--ndb/src/client/odbc/codegen/Code_pred_op.hpp158
-rw-r--r--ndb/src/client/odbc/codegen/Code_query.cpp299
-rw-r--r--ndb/src/client/odbc/codegen/Code_query.hpp155
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_count.cpp177
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_count.hpp162
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_distinct.cpp204
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_distinct.hpp165
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_filter.cpp161
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_filter.hpp162
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_group.cpp301
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_group.hpp221
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_index.cpp186
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_index.hpp160
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_join.cpp192
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_join.hpp159
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_lookup.cpp184
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_lookup.hpp155
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_project.cpp184
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_project.hpp178
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_range.cpp211
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_range.hpp186
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_repeat.cpp109
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_repeat.hpp133
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_scan.cpp177
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_scan.hpp174
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_sort.cpp239
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_sort.hpp208
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_sys.cpp130
-rw-r--r--ndb/src/client/odbc/codegen/Code_query_sys.hpp148
-rw-r--r--ndb/src/client/odbc/codegen/Code_root.cpp307
-rw-r--r--ndb/src/client/odbc/codegen/Code_root.hpp162
-rw-r--r--ndb/src/client/odbc/codegen/Code_select.cpp406
-rw-r--r--ndb/src/client/odbc/codegen/Code_select.hpp132
-rw-r--r--ndb/src/client/odbc/codegen/Code_set_row.cpp44
-rw-r--r--ndb/src/client/odbc/codegen/Code_set_row.hpp76
-rw-r--r--ndb/src/client/odbc/codegen/Code_stmt.cpp49
-rw-r--r--ndb/src/client/odbc/codegen/Code_stmt.hpp76
-rw-r--r--ndb/src/client/odbc/codegen/Code_table.cpp254
-rw-r--r--ndb/src/client/odbc/codegen/Code_table.hpp202
-rw-r--r--ndb/src/client/odbc/codegen/Code_table_list.cpp53
-rw-r--r--ndb/src/client/odbc/codegen/Code_table_list.hpp73
-rw-r--r--ndb/src/client/odbc/codegen/Code_update.cpp246
-rw-r--r--ndb/src/client/odbc/codegen/Code_update.hpp102
-rw-r--r--ndb/src/client/odbc/codegen/Code_update_index.cpp196
-rw-r--r--ndb/src/client/odbc/codegen/Code_update_index.hpp171
-rw-r--r--ndb/src/client/odbc/codegen/Code_update_lookup.cpp194
-rw-r--r--ndb/src/client/odbc/codegen/Code_update_lookup.hpp167
-rw-r--r--ndb/src/client/odbc/codegen/Code_update_scan.cpp146
-rw-r--r--ndb/src/client/odbc/codegen/Code_update_scan.hpp160
-rw-r--r--ndb/src/client/odbc/codegen/Makefile104
-rw-r--r--ndb/src/client/odbc/codegen/SimpleGram.ypp1629
-rw-r--r--ndb/src/client/odbc/codegen/SimpleParser.cpp95
-rw-r--r--ndb/src/client/odbc/codegen/SimpleParser.hpp161
-rw-r--r--ndb/src/client/odbc/codegen/SimpleScan.lpp241
-rw-r--r--ndb/src/client/odbc/common/AttrArea.cpp91
-rw-r--r--ndb/src/client/odbc/common/AttrArea.hpp135
-rw-r--r--ndb/src/client/odbc/common/CodeTree.cpp37
-rw-r--r--ndb/src/client/odbc/common/CodeTree.hpp49
-rw-r--r--ndb/src/client/odbc/common/ConnArea.cpp109
-rw-r--r--ndb/src/client/odbc/common/ConnArea.hpp135
-rw-r--r--ndb/src/client/odbc/common/Ctx.cpp360
-rw-r--r--ndb/src/client/odbc/common/Ctx.hpp182
-rw-r--r--ndb/src/client/odbc/common/DataField.cpp3030
-rw-r--r--ndb/src/client/odbc/common/DataField.hpp446
-rw-r--r--ndb/src/client/odbc/common/DataRow.cpp140
-rw-r--r--ndb/src/client/odbc/common/DataRow.hpp185
-rw-r--r--ndb/src/client/odbc/common/DataType.cpp546
-rw-r--r--ndb/src/client/odbc/common/DataType.hpp292
-rw-r--r--ndb/src/client/odbc/common/DescArea.cpp167
-rw-r--r--ndb/src/client/odbc/common/DescArea.hpp266
-rw-r--r--ndb/src/client/odbc/common/DiagArea.cpp284
-rw-r--r--ndb/src/client/odbc/common/DiagArea.hpp196
-rw-r--r--ndb/src/client/odbc/common/Makefile29
-rw-r--r--ndb/src/client/odbc/common/OdbcData.cpp563
-rw-r--r--ndb/src/client/odbc/common/OdbcData.hpp283
-rw-r--r--ndb/src/client/odbc/common/ResultArea.cpp29
-rw-r--r--ndb/src/client/odbc/common/ResultArea.hpp162
-rw-r--r--ndb/src/client/odbc/common/Sqlstate.cpp93
-rw-r--r--ndb/src/client/odbc/common/Sqlstate.hpp85
-rw-r--r--ndb/src/client/odbc/common/StmtArea.cpp112
-rw-r--r--ndb/src/client/odbc/common/StmtArea.hpp157
-rw-r--r--ndb/src/client/odbc/common/StmtInfo.cpp78
-rw-r--r--ndb/src/client/odbc/common/StmtInfo.hpp86
-rw-r--r--ndb/src/client/odbc/common/common.cpp17
-rw-r--r--ndb/src/client/odbc/common/common.hpp120
-rw-r--r--ndb/src/client/odbc/dictionary/DictCatalog.cpp42
-rw-r--r--ndb/src/client/odbc/dictionary/DictCatalog.hpp64
-rw-r--r--ndb/src/client/odbc/dictionary/DictColumn.cpp23
-rw-r--r--ndb/src/client/odbc/dictionary/DictColumn.hpp143
-rw-r--r--ndb/src/client/odbc/dictionary/DictIndex.cpp29
-rw-r--r--ndb/src/client/odbc/dictionary/DictIndex.hpp108
-rw-r--r--ndb/src/client/odbc/dictionary/DictSchema.cpp155
-rw-r--r--ndb/src/client/odbc/dictionary/DictSchema.hpp89
-rw-r--r--ndb/src/client/odbc/dictionary/DictSys.cpp433
-rw-r--r--ndb/src/client/odbc/dictionary/DictSys.hpp77
-rw-r--r--ndb/src/client/odbc/dictionary/DictTable.cpp91
-rw-r--r--ndb/src/client/odbc/dictionary/DictTable.hpp192
-rw-r--r--ndb/src/client/odbc/dictionary/Makefile20
-rw-r--r--ndb/src/client/odbc/docs/class.fig332
-rw-r--r--ndb/src/client/odbc/docs/descfield.pl1482
-rw-r--r--ndb/src/client/odbc/docs/diag.txt48
-rw-r--r--ndb/src/client/odbc/docs/getinfo.pl3676
-rw-r--r--ndb/src/client/odbc/docs/gettypeinfo.pl645
-rw-r--r--ndb/src/client/odbc/docs/handleattr.pl2232
-rw-r--r--ndb/src/client/odbc/docs/main.hpp104
-rw-r--r--ndb/src/client/odbc/docs/ndbodbc.html659
-rw-r--r--ndb/src/client/odbc/docs/select.fig94
-rw-r--r--ndb/src/client/odbc/docs/systables.pl2192
-rw-r--r--ndb/src/client/odbc/docs/type.txt333
-rw-r--r--ndb/src/client/odbc/driver/Func.data2822
-rw-r--r--ndb/src/client/odbc/driver/Func.pl352
-rw-r--r--ndb/src/client/odbc/driver/Makefile16
-rw-r--r--ndb/src/client/odbc/driver/SQLAllocConnect.cpp52
-rw-r--r--ndb/src/client/odbc/driver/SQLAllocEnv.cpp46
-rw-r--r--ndb/src/client/odbc/driver/SQLAllocHandle.cpp62
-rw-r--r--ndb/src/client/odbc/driver/SQLAllocHandleStd.cpp56
-rw-r--r--ndb/src/client/odbc/driver/SQLAllocStmt.cpp52
-rw-r--r--ndb/src/client/odbc/driver/SQLBindCol.cpp50
-rw-r--r--ndb/src/client/odbc/driver/SQLBindParam.cpp52
-rw-r--r--ndb/src/client/odbc/driver/SQLBindParameter.cpp54
-rw-r--r--ndb/src/client/odbc/driver/SQLBrowseConnect.cpp61
-rw-r--r--ndb/src/client/odbc/driver/SQLBulkOperations.cpp53
-rw-r--r--ndb/src/client/odbc/driver/SQLCancel.cpp45
-rw-r--r--ndb/src/client/odbc/driver/SQLCloseCursor.cpp45
-rw-r--r--ndb/src/client/odbc/driver/SQLColAttribute.cpp51
-rw-r--r--ndb/src/client/odbc/driver/SQLColAttributes.cpp51
-rw-r--r--ndb/src/client/odbc/driver/SQLColumnPrivileges.cpp67
-rw-r--r--ndb/src/client/odbc/driver/SQLColumns.cpp49
-rw-r--r--ndb/src/client/odbc/driver/SQLConnect.cpp51
-rw-r--r--ndb/src/client/odbc/driver/SQLCopyDesc.cpp58
-rw-r--r--ndb/src/client/odbc/driver/SQLDataSources.cpp65
-rw-r--r--ndb/src/client/odbc/driver/SQLDescribeCol.cpp53
-rw-r--r--ndb/src/client/odbc/driver/SQLDescribeParam.cpp50
-rw-r--r--ndb/src/client/odbc/driver/SQLDisconnect.cpp45
-rw-r--r--ndb/src/client/odbc/driver/SQLDriverConnect.cpp52
-rw-r--r--ndb/src/client/odbc/driver/SQLDrivers.cpp65
-rw-r--r--ndb/src/client/odbc/driver/SQLEndTran.cpp70
-rw-r--r--ndb/src/client/odbc/driver/SQLError.cpp67
-rw-r--r--ndb/src/client/odbc/driver/SQLExecDirect.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLExecute.cpp45
-rw-r--r--ndb/src/client/odbc/driver/SQLExtendedFetch.cpp59
-rw-r--r--ndb/src/client/odbc/driver/SQLFetch.cpp45
-rw-r--r--ndb/src/client/odbc/driver/SQLFetchScroll.cpp55
-rw-r--r--ndb/src/client/odbc/driver/SQLForeignKeys.cpp75
-rw-r--r--ndb/src/client/odbc/driver/SQLFreeConnect.cpp53
-rw-r--r--ndb/src/client/odbc/driver/SQLFreeEnv.cpp52
-rw-r--r--ndb/src/client/odbc/driver/SQLFreeHandle.cpp60
-rw-r--r--ndb/src/client/odbc/driver/SQLFreeStmt.cpp54
-rw-r--r--ndb/src/client/odbc/driver/SQLGetConnectAttr.cpp49
-rw-r--r--ndb/src/client/odbc/driver/SQLGetConnectOption.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLGetCursorName.cpp48
-rw-r--r--ndb/src/client/odbc/driver/SQLGetData.cpp50
-rw-r--r--ndb/src/client/odbc/driver/SQLGetDescField.cpp50
-rw-r--r--ndb/src/client/odbc/driver/SQLGetDescRec.cpp55
-rw-r--r--ndb/src/client/odbc/driver/SQLGetDiagField.cpp50
-rw-r--r--ndb/src/client/odbc/driver/SQLGetDiagRec.cpp51
-rw-r--r--ndb/src/client/odbc/driver/SQLGetEnvAttr.cpp66
-rw-r--r--ndb/src/client/odbc/driver/SQLGetFunctions.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLGetInfo.cpp49
-rw-r--r--ndb/src/client/odbc/driver/SQLGetStmtAttr.cpp49
-rw-r--r--ndb/src/client/odbc/driver/SQLGetStmtOption.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLGetTypeInfo.cpp46
-rw-r--r--ndb/src/client/odbc/driver/SQLMoreResults.cpp42
-rw-r--r--ndb/src/client/odbc/driver/SQLNativeSql.cpp61
-rw-r--r--ndb/src/client/odbc/driver/SQLNumParams.cpp46
-rw-r--r--ndb/src/client/odbc/driver/SQLNumResultCols.cpp46
-rw-r--r--ndb/src/client/odbc/driver/SQLParamData.cpp46
-rw-r--r--ndb/src/client/odbc/driver/SQLParamOptions.cpp55
-rw-r--r--ndb/src/client/odbc/driver/SQLPrepare.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLPrimaryKeys.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLProcedureColumns.cpp67
-rw-r--r--ndb/src/client/odbc/driver/SQLProcedures.cpp63
-rw-r--r--ndb/src/client/odbc/driver/SQLPutData.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLRowCount.cpp46
-rw-r--r--ndb/src/client/odbc/driver/SQLSetConnectAttr.cpp48
-rw-r--r--ndb/src/client/odbc/driver/SQLSetConnectOption.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLSetCursorName.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLSetDescField.cpp49
-rw-r--r--ndb/src/client/odbc/driver/SQLSetDescRec.cpp54
-rw-r--r--ndb/src/client/odbc/driver/SQLSetEnvAttr.cpp65
-rw-r--r--ndb/src/client/odbc/driver/SQLSetParam.cpp52
-rw-r--r--ndb/src/client/odbc/driver/SQLSetPos.cpp57
-rw-r--r--ndb/src/client/odbc/driver/SQLSetScrollOptions.cpp57
-rw-r--r--ndb/src/client/odbc/driver/SQLSetStmtAttr.cpp48
-rw-r--r--ndb/src/client/odbc/driver/SQLSetStmtOption.cpp47
-rw-r--r--ndb/src/client/odbc/driver/SQLSpecialColumns.cpp69
-rw-r--r--ndb/src/client/odbc/driver/SQLStatistics.cpp67
-rw-r--r--ndb/src/client/odbc/driver/SQLTablePrivileges.cpp63
-rw-r--r--ndb/src/client/odbc/driver/SQLTables.cpp49
-rw-r--r--ndb/src/client/odbc/driver/SQLTransact.cpp71
-rw-r--r--ndb/src/client/odbc/driver/driver.cpp150
-rw-r--r--ndb/src/client/odbc/driver/driver.hpp28
-rw-r--r--ndb/src/client/odbc/executor/Exec_comp_op.cpp518
-rw-r--r--ndb/src/client/odbc/executor/Exec_create_index.cpp46
-rw-r--r--ndb/src/client/odbc/executor/Exec_create_table.cpp83
-rw-r--r--ndb/src/client/odbc/executor/Exec_delete_index.cpp82
-rw-r--r--ndb/src/client/odbc/executor/Exec_delete_lookup.cpp82
-rw-r--r--ndb/src/client/odbc/executor/Exec_delete_scan.cpp54
-rw-r--r--ndb/src/client/odbc/executor/Exec_drop_index.cpp38
-rw-r--r--ndb/src/client/odbc/executor/Exec_drop_table.cpp38
-rw-r--r--ndb/src/client/odbc/executor/Exec_expr_conv.cpp54
-rw-r--r--ndb/src/client/odbc/executor/Exec_expr_func.cpp284
-rw-r--r--ndb/src/client/odbc/executor/Exec_expr_op.cpp147
-rw-r--r--ndb/src/client/odbc/executor/Exec_insert.cpp144
-rw-r--r--ndb/src/client/odbc/executor/Exec_pred_op.cpp197
-rw-r--r--ndb/src/client/odbc/executor/Exec_query_index.cpp136
-rw-r--r--ndb/src/client/odbc/executor/Exec_query_lookup.cpp136
-rw-r--r--ndb/src/client/odbc/executor/Exec_query_range.cpp143
-rw-r--r--ndb/src/client/odbc/executor/Exec_query_scan.cpp130
-rw-r--r--ndb/src/client/odbc/executor/Exec_query_sys.cpp761
-rw-r--r--ndb/src/client/odbc/executor/Exec_update_index.cpp96
-rw-r--r--ndb/src/client/odbc/executor/Exec_update_lookup.cpp96
-rw-r--r--ndb/src/client/odbc/executor/Exec_update_scan.cpp61
-rw-r--r--ndb/src/client/odbc/executor/Executor.cpp68
-rw-r--r--ndb/src/client/odbc/executor/Executor.hpp52
-rw-r--r--ndb/src/client/odbc/executor/Makefile36
-rw-r--r--ndb/src/client/odbc/handles/AttrDbc.cpp473
-rw-r--r--ndb/src/client/odbc/handles/AttrEnv.cpp123
-rw-r--r--ndb/src/client/odbc/handles/AttrRoot.cpp92
-rw-r--r--ndb/src/client/odbc/handles/AttrStmt.cpp1005
-rw-r--r--ndb/src/client/odbc/handles/DescSpec.cpp1140
-rw-r--r--ndb/src/client/odbc/handles/FuncTab.cpp100
-rw-r--r--ndb/src/client/odbc/handles/HandleBase.cpp162
-rw-r--r--ndb/src/client/odbc/handles/HandleBase.hpp67
-rw-r--r--ndb/src/client/odbc/handles/HandleDbc.cpp419
-rw-r--r--ndb/src/client/odbc/handles/HandleDbc.hpp111
-rw-r--r--ndb/src/client/odbc/handles/HandleDesc.cpp254
-rw-r--r--ndb/src/client/odbc/handles/HandleDesc.hpp89
-rw-r--r--ndb/src/client/odbc/handles/HandleEnv.cpp144
-rw-r--r--ndb/src/client/odbc/handles/HandleEnv.hpp77
-rw-r--r--ndb/src/client/odbc/handles/HandleRoot.cpp270
-rw-r--r--ndb/src/client/odbc/handles/HandleRoot.hpp103
-rw-r--r--ndb/src/client/odbc/handles/HandleStmt.cpp823
-rw-r--r--ndb/src/client/odbc/handles/HandleStmt.hpp117
-rw-r--r--ndb/src/client/odbc/handles/InfoTab.cpp878
-rw-r--r--ndb/src/client/odbc/handles/Makefile28
-rw-r--r--ndb/src/client/odbc/handles/PoolNdb.cpp80
-rw-r--r--ndb/src/client/odbc/handles/PoolNdb.hpp44
-rw-r--r--ndb/src/client/odbc/handles/handles.hpp28
-rw-r--r--ndb/src/common/Makefile15
-rw-r--r--ndb/src/common/debugger/BlockNames.cpp39
-rw-r--r--ndb/src/common/debugger/DebuggerNames.cpp154
-rw-r--r--ndb/src/common/debugger/EventLogger.cpp1454
-rw-r--r--ndb/src/common/debugger/GrepError.cpp133
-rw-r--r--ndb/src/common/debugger/LogLevel.cpp29
-rw-r--r--ndb/src/common/debugger/Makefile11
-rw-r--r--ndb/src/common/debugger/SignalLoggerManager.cpp513
-rw-r--r--ndb/src/common/debugger/signaldata/AccLock.cpp75
-rw-r--r--ndb/src/common/debugger/signaldata/AlterIndx.cpp35
-rw-r--r--ndb/src/common/debugger/signaldata/AlterTab.cpp38
-rw-r--r--ndb/src/common/debugger/signaldata/AlterTable.cpp38
-rw-r--r--ndb/src/common/debugger/signaldata/AlterTrig.cpp51
-rw-r--r--ndb/src/common/debugger/signaldata/BackupImpl.cpp142
-rw-r--r--ndb/src/common/debugger/signaldata/BackupSignalData.cpp129
-rw-r--r--ndb/src/common/debugger/signaldata/CloseComReqConf.cpp53
-rw-r--r--ndb/src/common/debugger/signaldata/ContinueB.cpp36
-rw-r--r--ndb/src/common/debugger/signaldata/CopyGCI.cpp58
-rw-r--r--ndb/src/common/debugger/signaldata/CreateEvnt.cpp38
-rw-r--r--ndb/src/common/debugger/signaldata/CreateFragmentation.cpp56
-rw-r--r--ndb/src/common/debugger/signaldata/CreateIndx.cpp38
-rw-r--r--ndb/src/common/debugger/signaldata/CreateTrig.cpp120
-rw-r--r--ndb/src/common/debugger/signaldata/DictTabInfo.cpp153
-rw-r--r--ndb/src/common/debugger/signaldata/DihContinueB.cpp217
-rw-r--r--ndb/src/common/debugger/signaldata/DihSwitchReplicaReq.cpp48
-rw-r--r--ndb/src/common/debugger/signaldata/DisconnectRep.cpp30
-rw-r--r--ndb/src/common/debugger/signaldata/DropIndx.cpp38
-rw-r--r--ndb/src/common/debugger/signaldata/DropTab.cpp50
-rw-r--r--ndb/src/common/debugger/signaldata/DropTrig.cpp89
-rw-r--r--ndb/src/common/debugger/signaldata/FailRep.cpp31
-rw-r--r--ndb/src/common/debugger/signaldata/FireTrigOrd.cpp56
-rw-r--r--ndb/src/common/debugger/signaldata/FsAppendReq.cpp38
-rw-r--r--ndb/src/common/debugger/signaldata/FsCloseReq.cpp40
-rw-r--r--ndb/src/common/debugger/signaldata/FsConf.cpp33
-rw-r--r--ndb/src/common/debugger/signaldata/FsOpenReq.cpp59
-rw-r--r--ndb/src/common/debugger/signaldata/FsReadWriteReq.cpp85
-rw-r--r--ndb/src/common/debugger/signaldata/FsRef.cpp75
-rw-r--r--ndb/src/common/debugger/signaldata/GCPSave.cpp78
-rwxr-xr-xndb/src/common/debugger/signaldata/IndxAttrInfo.cpp31
-rwxr-xr-xndb/src/common/debugger/signaldata/IndxKeyInfo.cpp31
-rw-r--r--ndb/src/common/debugger/signaldata/LCP.cpp88
-rw-r--r--ndb/src/common/debugger/signaldata/LqhFrag.cpp61
-rw-r--r--ndb/src/common/debugger/signaldata/LqhKey.cpp161
-rw-r--r--ndb/src/common/debugger/signaldata/LqhTrans.cpp40
-rw-r--r--ndb/src/common/debugger/signaldata/Makefile32
-rw-r--r--ndb/src/common/debugger/signaldata/MasterLCP.cpp87
-rw-r--r--ndb/src/common/debugger/signaldata/NFCompleteRep.cpp44
-rw-r--r--ndb/src/common/debugger/signaldata/NdbSttor.cpp50
-rw-r--r--ndb/src/common/debugger/signaldata/NdbfsContinueB.cpp38
-rw-r--r--ndb/src/common/debugger/signaldata/PackedSignal.cpp104
-rw-r--r--ndb/src/common/debugger/signaldata/PrepDropTab.cpp50
-rw-r--r--ndb/src/common/debugger/signaldata/PrepFailReqRef.cpp53
-rw-r--r--ndb/src/common/debugger/signaldata/ScanTab.cpp163
-rw-r--r--ndb/src/common/debugger/signaldata/SignalDataPrint.cpp254
-rw-r--r--ndb/src/common/debugger/signaldata/SignalDroppedRep.cpp34
-rw-r--r--ndb/src/common/debugger/signaldata/SignalNames.cpp673
-rw-r--r--ndb/src/common/debugger/signaldata/StartRec.cpp52
-rw-r--r--ndb/src/common/debugger/signaldata/SumaImpl.cpp167
-rw-r--r--ndb/src/common/debugger/signaldata/SystemError.cpp41
-rw-r--r--ndb/src/common/debugger/signaldata/TcIndx.cpp159
-rw-r--r--ndb/src/common/debugger/signaldata/TcKeyConf.cpp59
-rw-r--r--ndb/src/common/debugger/signaldata/TcKeyRef.cpp28
-rw-r--r--ndb/src/common/debugger/signaldata/TcKeyReq.cpp114
-rw-r--r--ndb/src/common/debugger/signaldata/TcRollbackRep.cpp28
-rw-r--r--ndb/src/common/debugger/signaldata/TrigAttrInfo.cpp53
-rw-r--r--ndb/src/common/debugger/signaldata/TupAccess.cpp131
-rw-r--r--ndb/src/common/debugger/signaldata/TupCommit.cpp28
-rw-r--r--ndb/src/common/debugger/signaldata/TupKey.cpp50
-rw-r--r--ndb/src/common/debugger/signaldata/TuxMaint.cpp45
-rw-r--r--ndb/src/common/debugger/signaldata/UtilDelete.cpp65
-rw-r--r--ndb/src/common/debugger/signaldata/UtilExecute.cpp59
-rw-r--r--ndb/src/common/debugger/signaldata/UtilLock.cpp158
-rw-r--r--ndb/src/common/debugger/signaldata/UtilPrepare.cpp64
-rw-r--r--ndb/src/common/debugger/signaldata/UtilSequence.cpp67
-rw-r--r--ndb/src/common/debugger/signaldata/print.awk55
-rw-r--r--ndb/src/common/editline/MANIFEST15
-rw-r--r--ndb/src/common/editline/Makefile18
-rw-r--r--ndb/src/common/editline/README53
-rw-r--r--ndb/src/common/editline/complete.c211
-rw-r--r--ndb/src/common/editline/editline.3178
-rw-r--r--ndb/src/common/editline/editline.c1514
-rw-r--r--ndb/src/common/editline/editline_internal.h47
-rw-r--r--ndb/src/common/editline/editline_win32.c33
-rw-r--r--ndb/src/common/editline/sysunix.c143
-rw-r--r--ndb/src/common/editline/test/Makefile10
-rw-r--r--ndb/src/common/editline/test/testit.c59
-rw-r--r--ndb/src/common/editline/unix.h26
-rw-r--r--ndb/src/common/logger/ConsoleLogHandler.cpp68
-rw-r--r--ndb/src/common/logger/FileLogHandler.cpp241
-rw-r--r--ndb/src/common/logger/LogHandler.cpp142
-rw-r--r--ndb/src/common/logger/LogHandlerList.cpp183
-rw-r--r--ndb/src/common/logger/LogHandlerList.hpp93
-rw-r--r--ndb/src/common/logger/Logger.cpp358
-rw-r--r--ndb/src/common/logger/Makefile27
-rw-r--r--ndb/src/common/logger/SysLogHandler.cpp159
-rw-r--r--ndb/src/common/logger/listtest/LogHandlerListUnitTest.cpp165
-rw-r--r--ndb/src/common/logger/listtest/LogHandlerListUnitTest.hpp40
-rw-r--r--ndb/src/common/logger/listtest/Makefile14
-rw-r--r--ndb/src/common/logger/loggertest/LoggerUnitTest.cpp201
-rw-r--r--ndb/src/common/logger/loggertest/LoggerUnitTest.hpp49
-rw-r--r--ndb/src/common/logger/loggertest/Makefile16
-rw-r--r--ndb/src/common/mgmcommon/Config.cpp255
-rw-r--r--ndb/src/common/mgmcommon/Config.hpp86
-rw-r--r--ndb/src/common/mgmcommon/ConfigInfo.cpp2629
-rw-r--r--ndb/src/common/mgmcommon/ConfigInfo.hpp120
-rw-r--r--ndb/src/common/mgmcommon/ConfigRetriever.cpp514
-rw-r--r--ndb/src/common/mgmcommon/IPCConfig.cpp336
-rw-r--r--ndb/src/common/mgmcommon/InitConfigFileParser.cpp544
-rw-r--r--ndb/src/common/mgmcommon/InitConfigFileParser.hpp142
-rw-r--r--ndb/src/common/mgmcommon/LocalConfig.cpp308
-rw-r--r--ndb/src/common/mgmcommon/LocalConfig.hpp83
-rw-r--r--ndb/src/common/mgmcommon/Makefile26
-rw-r--r--ndb/src/common/mgmcommon/NdbConfig.c61
-rw-r--r--ndb/src/common/mgmcommon/printConfig/Makefile14
-rw-r--r--ndb/src/common/mgmcommon/printConfig/printConfig.cpp89
-rw-r--r--ndb/src/common/portlib/Makefile43
-rw-r--r--ndb/src/common/portlib/memtest/Makefile12
-rw-r--r--ndb/src/common/portlib/memtest/memtest.c245
-rw-r--r--ndb/src/common/portlib/memtest/munmaptest/Makefile14
-rw-r--r--ndb/src/common/portlib/memtest/munmaptest/munmaptest.cpp251
-rw-r--r--ndb/src/common/portlib/mmstest/mmslist.cpp103
-rw-r--r--ndb/src/common/portlib/mmstest/mmstest.cpp76
-rw-r--r--ndb/src/common/portlib/ose/Makefile31
-rw-r--r--ndb/src/common/portlib/ose/NdbCondition.c244
-rw-r--r--ndb/src/common/portlib/ose/NdbConditionOSE.h103
-rw-r--r--ndb/src/common/portlib/ose/NdbEnv.c55
-rw-r--r--ndb/src/common/portlib/ose/NdbHost.c55
-rw-r--r--ndb/src/common/portlib/ose/NdbMem.c183
-rw-r--r--ndb/src/common/portlib/ose/NdbMem_SoftOse.cpp53
-rw-r--r--ndb/src/common/portlib/ose/NdbMutex.c86
-rw-r--r--ndb/src/common/portlib/ose/NdbOut.cpp99
-rw-r--r--ndb/src/common/portlib/ose/NdbSleep.c36
-rw-r--r--ndb/src/common/portlib/ose/NdbTCP.c38
-rw-r--r--ndb/src/common/portlib/ose/NdbThread.c184
-rw-r--r--ndb/src/common/portlib/ose/NdbTick.c64
-rw-r--r--ndb/src/common/portlib/test/Makefile15
-rw-r--r--ndb/src/common/portlib/test/NdbPortLibTest.cpp621
-rw-r--r--ndb/src/common/portlib/unix/Makefile27
-rw-r--r--ndb/src/common/portlib/unix/NdbCondition.c179
-rw-r--r--ndb/src/common/portlib/unix/NdbDaemon.c170
-rw-r--r--ndb/src/common/portlib/unix/NdbEnv.c34
-rw-r--r--ndb/src/common/portlib/unix/NdbHost.c34
-rw-r--r--ndb/src/common/portlib/unix/NdbMem.c76
-rw-r--r--ndb/src/common/portlib/unix/NdbMutex.c93
-rw-r--r--ndb/src/common/portlib/unix/NdbSleep.c48
-rw-r--r--ndb/src/common/portlib/unix/NdbTCP.c60
-rw-r--r--ndb/src/common/portlib/unix/NdbThread.c119
-rw-r--r--ndb/src/common/portlib/unix/NdbTick.c110
-rw-r--r--ndb/src/common/portlib/win32/Makefile30
-rw-r--r--ndb/src/common/portlib/win32/NdbCondition.c184
-rw-r--r--ndb/src/common/portlib/win32/NdbDaemon.c44
-rw-r--r--ndb/src/common/portlib/win32/NdbEnv.c33
-rw-r--r--ndb/src/common/portlib/win32/NdbHost.c53
-rw-r--r--ndb/src/common/portlib/win32/NdbMem.c237
-rw-r--r--ndb/src/common/portlib/win32/NdbMutex.c78
-rw-r--r--ndb/src/common/portlib/win32/NdbSleep.c35
-rw-r--r--ndb/src/common/portlib/win32/NdbTCP.c39
-rw-r--r--ndb/src/common/portlib/win32/NdbThread.c118
-rw-r--r--ndb/src/common/portlib/win32/NdbTick.c64
-rw-r--r--ndb/src/common/transporter/Makefile62
-rw-r--r--ndb/src/common/transporter/OSE_Receiver.cpp360
-rw-r--r--ndb/src/common/transporter/OSE_Receiver.hpp119
-rw-r--r--ndb/src/common/transporter/OSE_Signals.hpp144
-rw-r--r--ndb/src/common/transporter/OSE_Transporter.cpp487
-rw-r--r--ndb/src/common/transporter/OSE_Transporter.hpp158
-rw-r--r--ndb/src/common/transporter/Packer.cpp502
-rw-r--r--ndb/src/common/transporter/Packer.hpp85
-rw-r--r--ndb/src/common/transporter/SCI_Transporter.cpp1006
-rw-r--r--ndb/src/common/transporter/SCI_Transporter.hpp390
-rw-r--r--ndb/src/common/transporter/SHM_Buffer.hpp217
-rw-r--r--ndb/src/common/transporter/SHM_Transporter.cpp238
-rw-r--r--ndb/src/common/transporter/SHM_Transporter.hpp156
-rw-r--r--ndb/src/common/transporter/SHM_Transporter.unix.cpp179
-rw-r--r--ndb/src/common/transporter/SHM_Transporter.win32.cpp172
-rw-r--r--ndb/src/common/transporter/SendBuffer.cpp89
-rw-r--r--ndb/src/common/transporter/SendBuffer.hpp191
-rw-r--r--ndb/src/common/transporter/TCP_Transporter.cpp603
-rw-r--r--ndb/src/common/transporter/TCP_Transporter.hpp290
-rw-r--r--ndb/src/common/transporter/Transporter.cpp147
-rw-r--r--ndb/src/common/transporter/Transporter.hpp177
-rw-r--r--ndb/src/common/transporter/TransporterInternalDefinitions.hpp323
-rw-r--r--ndb/src/common/transporter/TransporterRegistry.cpp1188
-rw-r--r--ndb/src/common/transporter/basictest/Makefile15
-rw-r--r--ndb/src/common/transporter/basictest/basicTransporterTest.cpp536
-rw-r--r--ndb/src/common/transporter/buddy.cpp328
-rw-r--r--ndb/src/common/transporter/buddy.hpp173
-rw-r--r--ndb/src/common/transporter/failoverSCI/Makefile18
-rw-r--r--ndb/src/common/transporter/failoverSCI/failoverSCI.cpp866
-rw-r--r--ndb/src/common/transporter/perftest/Makefile15
-rw-r--r--ndb/src/common/transporter/perftest/perfTransporterTest.cpp774
-rw-r--r--ndb/src/common/transporter/priotest/Makefile15
-rw-r--r--ndb/src/common/transporter/priotest/prioOSE/Makefile17
-rw-r--r--ndb/src/common/transporter/priotest/prioSCI/Makefile17
-rw-r--r--ndb/src/common/transporter/priotest/prioSCI/prioSCI.cpp29
-rw-r--r--ndb/src/common/transporter/priotest/prioSHM/Makefile13
-rw-r--r--ndb/src/common/transporter/priotest/prioSHM/prioSHM.cpp26
-rw-r--r--ndb/src/common/transporter/priotest/prioTCP/Makefile13
-rw-r--r--ndb/src/common/transporter/priotest/prioTCP/prioTCP.cpp26
-rw-r--r--ndb/src/common/transporter/priotest/prioTransporterTest.cpp768
-rw-r--r--ndb/src/common/transporter/priotest/prioTransporterTest.hpp35
-rw-r--r--ndb/src/common/util/Base64.cpp111
-rw-r--r--ndb/src/common/util/BaseString.cpp418
-rw-r--r--ndb/src/common/util/File.cpp207
-rw-r--r--ndb/src/common/util/InputStream.cpp61
-rw-r--r--ndb/src/common/util/Makefile36
-rw-r--r--ndb/src/common/util/NdbErrHnd.cpp493
-rw-r--r--ndb/src/common/util/NdbOut.cpp175
-rw-r--r--ndb/src/common/util/NdbSqlUtil.cpp351
-rw-r--r--ndb/src/common/util/OutputStream.cpp98
-rw-r--r--ndb/src/common/util/Parser.cpp349
-rw-r--r--ndb/src/common/util/Properties.cpp1019
-rw-r--r--ndb/src/common/util/SimpleProperties.cpp509
-rw-r--r--ndb/src/common/util/SocketServer.cpp307
-rw-r--r--ndb/src/common/util/filetest/FileUnitTest.cpp238
-rw-r--r--ndb/src/common/util/filetest/FileUnitTest.hpp41
-rw-r--r--ndb/src/common/util/filetest/Makefile14
-rw-r--r--ndb/src/common/util/getarg.3315
-rw-r--r--ndb/src/common/util/getarg.3.ps458
-rw-r--r--ndb/src/common/util/getarg.c599
-rw-r--r--ndb/src/common/util/getarg.cat3237
-rw-r--r--ndb/src/common/util/md5_hash.cpp235
-rw-r--r--ndb/src/common/util/random.c292
-rw-r--r--ndb/src/common/util/socket_io.cpp284
-rw-r--r--ndb/src/common/util/strdup.c28
-rw-r--r--ndb/src/common/util/strlcat.c52
-rw-r--r--ndb/src/common/util/strlcpy.c65
-rw-r--r--ndb/src/common/util/testProperties/Makefile12
-rw-r--r--ndb/src/common/util/testProperties/testProperties.cpp203
-rw-r--r--ndb/src/common/util/testSimpleProperties/Makefile12
-rw-r--r--ndb/src/common/util/testSimpleProperties/sp_test.cpp95
-rw-r--r--ndb/src/common/util/uucode.c235
-rw-r--r--ndb/src/common/util/version.c224
-rw-r--r--ndb/src/cw/Makefile6
-rw-r--r--ndb/src/cw/cpcc-win32/C++/CPC_GUI.cpp215
-rw-r--r--ndb/src/cw/cpcc-win32/C++/CPC_GUI.dsp216
-rw-r--r--ndb/src/cw/cpcc-win32/C++/CPC_GUI.dsw29
-rw-r--r--ndb/src/cw/cpcc-win32/C++/CPC_GUI.h40
-rw-r--r--ndb/src/cw/cpcc-win32/C++/CPC_GUI.icobin0 -> 1078 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/C++/CPC_GUI.rc193
-rw-r--r--ndb/src/cw/cpcc-win32/C++/CPC_GUI.sln21
-rw-r--r--ndb/src/cw/cpcc-win32/C++/CPC_GUI.suobin0 -> 8704 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/C++/CPC_GUI.vcproj240
-rw-r--r--ndb/src/cw/cpcc-win32/C++/Closed.ICObin0 -> 1078 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/C++/NdbControls.cpp436
-rw-r--r--ndb/src/cw/cpcc-win32/C++/Open.ICObin0 -> 1078 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/C++/StdAfx.cpp24
-rw-r--r--ndb/src/cw/cpcc-win32/C++/StdAfx.h72
-rw-r--r--ndb/src/cw/cpcc-win32/C++/TreeView.cpp19
-rw-r--r--ndb/src/cw/cpcc-win32/C++/TreeView.h19
-rw-r--r--ndb/src/cw/cpcc-win32/C++/bmp00001.bmpbin0 -> 622 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/C++/resource.h90
-rw-r--r--ndb/src/cw/cpcc-win32/C++/small.icobin0 -> 318 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/C++/toolbar.bmpbin0 -> 622 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/App.icobin0 -> 1078 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/AssemblyInfo.cs58
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/CPC_Form.cs1400
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/Computer.cs256
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/ComputerAddDialog.cs242
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/ComputerRemoveDialog.cs228
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/DATABASE.ICObin0 -> 1078 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/Database.cs162
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/NDB_CPC.csproj240
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/NDB_CPC.csproj.user48
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/NDB_CPC.ncbbin0 -> 19456 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/NDB_CPC.sln21
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/PanelWizard.cs1883
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/Process.cs144
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/ProcessDefineDialog.cs435
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/fileaccess/FileMgmt.cs41
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/simpleparser/SimpleCPCParser.cs360
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/socketcomm/SocketComm.cs207
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/socketcomm/myTcpClient.cs26
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/startDatabaseDlg.cs251
-rw-r--r--ndb/src/cw/cpcc-win32/csharp/telnetclient/telnetClient.cs408
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Computer.cls20
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Database.cls18
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 110.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 231.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 237.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 241.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 242.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 270.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 271.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 273.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 31.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 337.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 338.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Icon 339.icobin0 -> 766 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/MSSCCPRJ.SCC5
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Module1.bas233
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/NdbCPC.vbp49
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/NdbCPC.vbw13
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/Process.cls22
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/closed folder.icobin0 -> 10134 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/computer.icobin0 -> 10134 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmAbout.frm245
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmLogin.frm119
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmMain.frm1207
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmNewComputer.frm124
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmNewComputer.frxbin0 -> 4 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmNewDatabase.frxbin0 -> 12 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmNewDatabase1.frm187
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmNewDatabase2.frm136
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmNewDatabase2.log1
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmNewDatabase3.frm88
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmOptions.frm231
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmSplash.frm159
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/frmSplash.frxbin0 -> 70450 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/networking.icobin0 -> 10134 bytes
-rw-r--r--ndb/src/cw/cpcc-win32/vb6/open folder.icobin0 -> 10134 bytes
-rw-r--r--ndb/src/cw/cpcd/APIService.cpp385
-rw-r--r--ndb/src/cw/cpcd/APIService.hpp64
-rw-r--r--ndb/src/cw/cpcd/CPCD.cpp435
-rw-r--r--ndb/src/cw/cpcd/CPCD.hpp382
-rw-r--r--ndb/src/cw/cpcd/Makefile11
-rw-r--r--ndb/src/cw/cpcd/Monitor.cpp76
-rw-r--r--ndb/src/cw/cpcd/Process.cpp482
-rw-r--r--ndb/src/cw/cpcd/common.cpp158
-rw-r--r--ndb/src/cw/cpcd/common.hpp35
-rw-r--r--ndb/src/cw/cpcd/main.cpp178
-rw-r--r--ndb/src/cw/test/socketclient/Makefile24
-rw-r--r--ndb/src/cw/test/socketclient/socketClientTest.cpp65
-rw-r--r--ndb/src/cw/util/ClientInterface.cpp185
-rw-r--r--ndb/src/cw/util/ClientInterface.hpp51
-rw-r--r--ndb/src/cw/util/Makefile10
-rw-r--r--ndb/src/cw/util/SocketRegistry.cpp213
-rw-r--r--ndb/src/cw/util/SocketRegistry.hpp290
-rw-r--r--ndb/src/cw/util/SocketService.cpp60
-rw-r--r--ndb/src/cw/util/SocketService.hpp46
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/list.h56
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/os/inttypes.h53
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/rmlib.h212
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/sci_errno.h216
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/sci_types.h300
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/sisci_api.h2170
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/sisci_demolib.h226
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/sisci_error.h89
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/sisci_types.h133
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/sisci_version.h91
-rw-r--r--ndb/src/external/LINUX.x86/sci/include/version.h25
-rw-r--r--ndb/src/external/SOLARIS.SPARC/sci/include/sisci_api.h2148
-rw-r--r--ndb/src/external/SOLARIS.SPARC/sci/include/sisci_error.h89
-rw-r--r--ndb/src/external/SOLARIS.SPARC/sci/include/sisci_types.h133
-rw-r--r--ndb/src/external/SOLARIS.SPARC/sci/include/sisci_version.h91
-rw-r--r--ndb/src/external/WIN32.x86/sci/include/rmlib.h212
-rw-r--r--ndb/src/external/WIN32.x86/sci/include/scilib.h330
-rw-r--r--ndb/src/external/WIN32.x86/sci/include/sisci_api.h2217
-rw-r--r--ndb/src/external/WIN32.x86/sci/include/sisci_demolib.h226
-rw-r--r--ndb/src/external/WIN32.x86/sci/include/sisci_error.h94
-rw-r--r--ndb/src/external/WIN32.x86/sci/include/sisci_types.h133
-rw-r--r--ndb/src/external/WIN32.x86/sci/lib/SISCI_LIBRARY_WIN32.TXT77
-rw-r--r--ndb/src/external/WIN32.x86/sci/lib/scilib.libbin0 -> 17918 bytes
-rw-r--r--ndb/src/external/WIN32.x86/sci/lib/scilib_md.libbin0 -> 18000 bytes
-rw-r--r--ndb/src/external/WIN32.x86/sci/lib/scilib_mt.libbin0 -> 17924 bytes
-rw-r--r--ndb/src/external/WIN32.x86/sci/lib/sisci_api.libbin0 -> 264284 bytes
-rw-r--r--ndb/src/external/WIN32.x86/sci/lib/sisci_api_md.libbin0 -> 265578 bytes
-rw-r--r--ndb/src/external/WIN32.x86/sci/lib/sisci_api_mt.libbin0 -> 264386 bytes
-rw-r--r--ndb/src/kernel/Makefile5
-rw-r--r--ndb/src/kernel/blocks/ERROR_codes.txt425
-rw-r--r--ndb/src/kernel/blocks/Makefile28
-rw-r--r--ndb/src/kernel/blocks/NodeRestart.new.txt82
-rw-r--r--ndb/src/kernel/blocks/NodeRestart.txt80
-rw-r--r--ndb/src/kernel/blocks/Start.txt97
-rw-r--r--ndb/src/kernel/blocks/SystemRestart.new.txt61
-rw-r--r--ndb/src/kernel/blocks/SystemRestart.txt61
-rw-r--r--ndb/src/kernel/blocks/backup/Backup.cpp4691
-rw-r--r--ndb/src/kernel/blocks/backup/Backup.hpp728
-rw-r--r--ndb/src/kernel/blocks/backup/Backup.txt343
-rw-r--r--ndb/src/kernel/blocks/backup/BackupFormat.hpp149
-rw-r--r--ndb/src/kernel/blocks/backup/BackupInit.cpp215
-rw-r--r--ndb/src/kernel/blocks/backup/FsBuffer.hpp346
-rw-r--r--ndb/src/kernel/blocks/backup/Makefile18
-rw-r--r--ndb/src/kernel/blocks/backup/read.cpp479
-rw-r--r--ndb/src/kernel/blocks/backup/restore/Makefile20
-rw-r--r--ndb/src/kernel/blocks/backup/restore/Restore.cpp1112
-rw-r--r--ndb/src/kernel/blocks/backup/restore/Restore.hpp328
-rw-r--r--ndb/src/kernel/blocks/backup/restore/main.cpp1689
-rw-r--r--ndb/src/kernel/blocks/backup/restore/myVector.hpp128
-rw-r--r--ndb/src/kernel/blocks/cmvmi/Cmvmi.cpp1531
-rw-r--r--ndb/src/kernel/blocks/cmvmi/Cmvmi.hpp131
-rw-r--r--ndb/src/kernel/blocks/cmvmi/Makefile9
-rw-r--r--ndb/src/kernel/blocks/dbacc/Dbacc.hpp1568
-rw-r--r--ndb/src/kernel/blocks/dbacc/DbaccInit.cpp248
-rw-r--r--ndb/src/kernel/blocks/dbacc/DbaccMain.cpp13285
-rw-r--r--ndb/src/kernel/blocks/dbacc/Makefile11
-rw-r--r--ndb/src/kernel/blocks/dbdict/CreateIndex.txt152
-rw-r--r--ndb/src/kernel/blocks/dbdict/CreateTable.new.txt29
-rw-r--r--ndb/src/kernel/blocks/dbdict/CreateTable.txt35
-rw-r--r--ndb/src/kernel/blocks/dbdict/Dbdict.cpp11628
-rw-r--r--ndb/src/kernel/blocks/dbdict/Dbdict.hpp1987
-rw-r--r--ndb/src/kernel/blocks/dbdict/Dbdict.txt88
-rw-r--r--ndb/src/kernel/blocks/dbdict/DropTable.txt140
-rw-r--r--ndb/src/kernel/blocks/dbdict/Event.txt102
-rw-r--r--ndb/src/kernel/blocks/dbdict/Makefile12
-rw-r--r--ndb/src/kernel/blocks/dbdict/Master_AddTable.sfl751
-rw-r--r--ndb/src/kernel/blocks/dbdict/SchemaFile.hpp57
-rw-r--r--ndb/src/kernel/blocks/dbdict/Slave_AddTable.sfl416
-rw-r--r--ndb/src/kernel/blocks/dbdict/printSchemafile/Makefile12
-rw-r--r--ndb/src/kernel/blocks/dbdict/printSchemafile/printSchemafile.cpp99
-rw-r--r--ndb/src/kernel/blocks/dbdih/Dbdih.hpp1606
-rw-r--r--ndb/src/kernel/blocks/dbdih/DbdihInit.cpp319
-rw-r--r--ndb/src/kernel/blocks/dbdih/DbdihMain.cpp14104
-rw-r--r--ndb/src/kernel/blocks/dbdih/LCP.txt35
-rw-r--r--ndb/src/kernel/blocks/dbdih/Makefile13
-rw-r--r--ndb/src/kernel/blocks/dbdih/Sysfile.hpp275
-rw-r--r--ndb/src/kernel/blocks/dbdih/printSysfile/Makefile12
-rw-r--r--ndb/src/kernel/blocks/dbdih/printSysfile/printSysfile.cpp160
-rw-r--r--ndb/src/kernel/blocks/dblqh/Dblqh.hpp2899
-rw-r--r--ndb/src/kernel/blocks/dblqh/DblqhInit.cpp416
-rw-r--r--ndb/src/kernel/blocks/dblqh/DblqhMain.cpp18014
-rw-r--r--ndb/src/kernel/blocks/dblqh/Makefile12
-rw-r--r--ndb/src/kernel/blocks/dblqh/redoLogReader/Makefile9
-rw-r--r--ndb/src/kernel/blocks/dblqh/redoLogReader/records.cpp312
-rw-r--r--ndb/src/kernel/blocks/dblqh/redoLogReader/records.hpp235
-rw-r--r--ndb/src/kernel/blocks/dblqh/redoLogReader/redoLogFileReader.cpp465
-rw-r--r--ndb/src/kernel/blocks/dbtc/Dbtc.hpp1947
-rw-r--r--ndb/src/kernel/blocks/dbtc/DbtcInit.cpp357
-rw-r--r--ndb/src/kernel/blocks/dbtc/DbtcMain.cpp13111
-rw-r--r--ndb/src/kernel/blocks/dbtc/Makefile11
-rw-r--r--ndb/src/kernel/blocks/dbtup/AttributeOffset.hpp89
-rw-r--r--ndb/src/kernel/blocks/dbtup/Dbtup.hpp2359
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupAbort.cpp473
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupBuffer.cpp329
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupCommit.cpp589
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupDebug.cpp399
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupExecQuery.cpp2067
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupFixAlloc.cpp384
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupGen.cpp1319
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupIndex.cpp576
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupLCP.cpp593
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupMeta.cpp597
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupPagMan.cpp360
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupPageMap.cpp556
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupRoutines.cpp896
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupStoredProcDef.cpp230
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupSystemRestart.cpp1003
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupTabDesMan.cpp191
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupTrigger.cpp1138
-rw-r--r--ndb/src/kernel/blocks/dbtup/DbtupUndoLog.cpp284
-rw-r--r--ndb/src/kernel/blocks/dbtup/Makefile26
-rw-r--r--ndb/src/kernel/blocks/dbtup/Notes.txt183
-rw-r--r--ndb/src/kernel/blocks/dbtux/Dbtux.hpp1218
-rw-r--r--ndb/src/kernel/blocks/dbtux/DbtuxCmp.cpp202
-rw-r--r--ndb/src/kernel/blocks/dbtux/DbtuxDebug.cpp369
-rw-r--r--ndb/src/kernel/blocks/dbtux/DbtuxGen.cpp221
-rw-r--r--ndb/src/kernel/blocks/dbtux/DbtuxMaint.cpp369
-rw-r--r--ndb/src/kernel/blocks/dbtux/DbtuxMeta.cpp427
-rw-r--r--ndb/src/kernel/blocks/dbtux/DbtuxNode.cpp633
-rw-r--r--ndb/src/kernel/blocks/dbtux/DbtuxScan.cpp1124
-rw-r--r--ndb/src/kernel/blocks/dbtux/DbtuxTree.cpp755
-rw-r--r--ndb/src/kernel/blocks/dbtux/Makefile17
-rw-r--r--ndb/src/kernel/blocks/dbtux/tuxstatus.html120
-rw-r--r--ndb/src/kernel/blocks/dbutil/DbUtil.cpp2628
-rw-r--r--ndb/src/kernel/blocks/dbutil/DbUtil.hpp484
-rw-r--r--ndb/src/kernel/blocks/dbutil/DbUtil.txt68
-rw-r--r--ndb/src/kernel/blocks/dbutil/Makefile8
-rw-r--r--ndb/src/kernel/blocks/grep/Grep.cpp2001
-rw-r--r--ndb/src/kernel/blocks/grep/Grep.hpp548
-rw-r--r--ndb/src/kernel/blocks/grep/GrepInit.cpp165
-rw-r--r--ndb/src/kernel/blocks/grep/Makefile9
-rw-r--r--ndb/src/kernel/blocks/grep/systab_test/Makefile12
-rw-r--r--ndb/src/kernel/blocks/grep/systab_test/grep_systab_test.cpp138
-rw-r--r--ndb/src/kernel/blocks/mutexes.hpp39
-rw-r--r--ndb/src/kernel/blocks/ndbcntr/Makefile12
-rw-r--r--ndb/src/kernel/blocks/ndbcntr/Ndbcntr.hpp513
-rw-r--r--ndb/src/kernel/blocks/ndbcntr/NdbcntrInit.cpp126
-rw-r--r--ndb/src/kernel/blocks/ndbcntr/NdbcntrMain.cpp3385
-rw-r--r--ndb/src/kernel/blocks/ndbcntr/NdbcntrSysTable.cpp94
-rw-r--r--ndb/src/kernel/blocks/ndbfs/AsyncFile.cpp1050
-rw-r--r--ndb/src/kernel/blocks/ndbfs/AsyncFile.hpp234
-rw-r--r--ndb/src/kernel/blocks/ndbfs/AsyncFileTest/AsyncFileTest.cpp697
-rw-r--r--ndb/src/kernel/blocks/ndbfs/AsyncFileTest/Makefile27
-rw-r--r--ndb/src/kernel/blocks/ndbfs/CircularIndex.cpp20
-rw-r--r--ndb/src/kernel/blocks/ndbfs/CircularIndex.hpp116
-rw-r--r--ndb/src/kernel/blocks/ndbfs/Filename.cpp220
-rw-r--r--ndb/src/kernel/blocks/ndbfs/Filename.hpp97
-rw-r--r--ndb/src/kernel/blocks/ndbfs/Makefile14
-rw-r--r--ndb/src/kernel/blocks/ndbfs/MemoryChannel.cpp18
-rw-r--r--ndb/src/kernel/blocks/ndbfs/MemoryChannel.hpp168
-rw-r--r--ndb/src/kernel/blocks/ndbfs/MemoryChannelOSE.hpp205
-rw-r--r--ndb/src/kernel/blocks/ndbfs/MemoryChannelTest/Makefile13
-rw-r--r--ndb/src/kernel/blocks/ndbfs/MemoryChannelTest/MemoryChannelTest.cpp197
-rw-r--r--ndb/src/kernel/blocks/ndbfs/Ndbfs.cpp1008
-rw-r--r--ndb/src/kernel/blocks/ndbfs/Ndbfs.hpp126
-rw-r--r--ndb/src/kernel/blocks/ndbfs/OpenFiles.hpp114
-rw-r--r--ndb/src/kernel/blocks/ndbfs/Pool.hpp262
-rw-r--r--ndb/src/kernel/blocks/ndbfs/VoidFs.cpp200
-rw-r--r--ndb/src/kernel/blocks/new-block.tar.gzbin0 -> 1816 bytes
-rw-r--r--ndb/src/kernel/blocks/qmgr/Makefile11
-rw-r--r--ndb/src/kernel/blocks/qmgr/Qmgr.hpp488
-rw-r--r--ndb/src/kernel/blocks/qmgr/QmgrInit.cpp115
-rw-r--r--ndb/src/kernel/blocks/qmgr/QmgrMain.cpp4533
-rw-r--r--ndb/src/kernel/blocks/qmgr/timer.hpp72
-rw-r--r--ndb/src/kernel/blocks/suma/Makefile10
-rw-r--r--ndb/src/kernel/blocks/suma/Suma.cpp3971
-rw-r--r--ndb/src/kernel/blocks/suma/Suma.hpp597
-rw-r--r--ndb/src/kernel/blocks/suma/Suma.txt192
-rw-r--r--ndb/src/kernel/blocks/suma/SumaInit.cpp186
-rw-r--r--ndb/src/kernel/blocks/trix/Makefile8
-rw-r--r--ndb/src/kernel/blocks/trix/Trix.cpp967
-rw-r--r--ndb/src/kernel/blocks/trix/Trix.hpp203
-rw-r--r--ndb/src/kernel/error/Error.hpp85
-rw-r--r--ndb/src/kernel/error/ErrorHandlingMacros.hpp50
-rw-r--r--ndb/src/kernel/error/ErrorMessages.cpp75
-rw-r--r--ndb/src/kernel/error/ErrorMessages.hpp22
-rw-r--r--ndb/src/kernel/error/ErrorReporter.cpp397
-rw-r--r--ndb/src/kernel/error/ErrorReporter.hpp98
-rw-r--r--ndb/src/kernel/error/Makefile12
-rw-r--r--ndb/src/kernel/error/TimeModule.cpp109
-rw-r--r--ndb/src/kernel/error/TimeModule.hpp46
-rw-r--r--ndb/src/kernel/ndb-main/Main.cpp340
-rw-r--r--ndb/src/kernel/ndb-main/Makefile42
-rw-r--r--ndb/src/kernel/ndb-main/SimBlockList.cpp111
-rw-r--r--ndb/src/kernel/vm/Array.hpp165
-rw-r--r--ndb/src/kernel/vm/ArrayFifoList.hpp30
-rw-r--r--ndb/src/kernel/vm/ArrayList.hpp30
-rw-r--r--ndb/src/kernel/vm/ArrayPool.hpp874
-rw-r--r--ndb/src/kernel/vm/CArray.hpp141
-rw-r--r--ndb/src/kernel/vm/Callback.hpp31
-rw-r--r--ndb/src/kernel/vm/ClusterConfiguration.cpp485
-rw-r--r--ndb/src/kernel/vm/ClusterConfiguration.hpp105
-rw-r--r--ndb/src/kernel/vm/Configuration.cpp338
-rw-r--r--ndb/src/kernel/vm/Configuration.hpp112
-rw-r--r--ndb/src/kernel/vm/DLFifoList.hpp348
-rw-r--r--ndb/src/kernel/vm/DLHashTable.hpp494
-rw-r--r--ndb/src/kernel/vm/DLHashTable2.hpp501
-rw-r--r--ndb/src/kernel/vm/DLList.hpp369
-rw-r--r--ndb/src/kernel/vm/DataBuffer.hpp532
-rw-r--r--ndb/src/kernel/vm/Emulator.cpp336
-rw-r--r--ndb/src/kernel/vm/Emulator.hpp101
-rw-r--r--ndb/src/kernel/vm/FastScheduler.cpp483
-rw-r--r--ndb/src/kernel/vm/FastScheduler.hpp353
-rw-r--r--ndb/src/kernel/vm/GlobalData.hpp118
-rw-r--r--ndb/src/kernel/vm/KeyTable.hpp43
-rw-r--r--ndb/src/kernel/vm/KeyTable2.hpp43
-rw-r--r--ndb/src/kernel/vm/LongSignal.hpp79
-rw-r--r--ndb/src/kernel/vm/Makefile29
-rw-r--r--ndb/src/kernel/vm/MetaData.cpp113
-rw-r--r--ndb/src/kernel/vm/MetaData.hpp247
-rw-r--r--ndb/src/kernel/vm/Mutex.cpp282
-rw-r--r--ndb/src/kernel/vm/Mutex.hpp321
-rw-r--r--ndb/src/kernel/vm/Prio.hpp32
-rw-r--r--ndb/src/kernel/vm/RequestTracker.hpp58
-rw-r--r--ndb/src/kernel/vm/SLList.hpp295
-rw-r--r--ndb/src/kernel/vm/SafeCounter.cpp159
-rw-r--r--ndb/src/kernel/vm/SafeCounter.hpp301
-rw-r--r--ndb/src/kernel/vm/SectionReader.cpp143
-rw-r--r--ndb/src/kernel/vm/SectionReader.hpp49
-rw-r--r--ndb/src/kernel/vm/SignalCounter.hpp164
-rw-r--r--ndb/src/kernel/vm/SimBlockList.hpp51
-rw-r--r--ndb/src/kernel/vm/SimplePropertiesSection.cpp223
-rw-r--r--ndb/src/kernel/vm/SimulatedBlock.cpp1733
-rw-r--r--ndb/src/kernel/vm/SimulatedBlock.hpp665
-rw-r--r--ndb/src/kernel/vm/ThreadConfig.cpp207
-rw-r--r--ndb/src/kernel/vm/ThreadConfig.hpp39
-rw-r--r--ndb/src/kernel/vm/TimeQueue.cpp209
-rw-r--r--ndb/src/kernel/vm/TimeQueue.hpp62
-rw-r--r--ndb/src/kernel/vm/TransporterCallback.cpp430
-rw-r--r--ndb/src/kernel/vm/VMSignal.cpp34
-rw-r--r--ndb/src/kernel/vm/VMSignal.hpp182
-rw-r--r--ndb/src/kernel/vm/WaitQueue.hpp35
-rw-r--r--ndb/src/kernel/vm/WatchDog.cpp143
-rw-r--r--ndb/src/kernel/vm/WatchDog.hpp56
-rw-r--r--ndb/src/kernel/vm/al_test/Makefile12
-rw-r--r--ndb/src/kernel/vm/al_test/arrayListTest.cpp317
-rw-r--r--ndb/src/kernel/vm/al_test/arrayPoolTest.cpp299
-rw-r--r--ndb/src/kernel/vm/al_test/main.cpp69
-rw-r--r--ndb/src/kernel/vm/pc.hpp248
-rw-r--r--ndb/src/kernel/vm/testCopy/Makefile9
-rw-r--r--ndb/src/kernel/vm/testCopy/rr.cpp33
-rw-r--r--ndb/src/kernel/vm/testCopy/testCopy.cpp343
-rw-r--r--ndb/src/kernel/vm/testDataBuffer/Makefile10
-rw-r--r--ndb/src/kernel/vm/testDataBuffer/testDataBuffer.cpp186
-rw-r--r--ndb/src/kernel/vm/testLongSig/Makefile9
-rw-r--r--ndb/src/kernel/vm/testLongSig/testLongSig.cpp300
-rw-r--r--ndb/src/kernel/vm/testSimplePropertiesSection/Makefile10
-rw-r--r--ndb/src/kernel/vm/testSimplePropertiesSection/test.cpp170
-rw-r--r--ndb/src/mgmapi/Makefile27
-rw-r--r--ndb/src/mgmapi/mgmapi.cpp1465
-rw-r--r--ndb/src/mgmapi/test/Makefile13
-rw-r--r--ndb/src/mgmapi/test/keso.c457
-rw-r--r--ndb/src/mgmapi/test/mgmSrvApi.cpp128
-rw-r--r--ndb/src/mgmclient/CommandInterpreter.cpp2013
-rw-r--r--ndb/src/mgmclient/CommandInterpreter.hpp198
-rw-r--r--ndb/src/mgmclient/CpcClient.cpp564
-rw-r--r--ndb/src/mgmclient/CpcClient.hpp104
-rw-r--r--ndb/src/mgmclient/Makefile25
-rw-r--r--ndb/src/mgmclient/main.cpp105
-rw-r--r--ndb/src/mgmclient/test_cpcd/Makefile17
-rw-r--r--ndb/src/mgmclient/test_cpcd/test_cpcd.cpp152
-rw-r--r--ndb/src/mgmsrv/CommandInterpreter.cpp1240
-rw-r--r--ndb/src/mgmsrv/CommandInterpreter.hpp176
-rw-r--r--ndb/src/mgmsrv/Makefile42
-rw-r--r--ndb/src/mgmsrv/MgmtSrvr.cpp2545
-rw-r--r--ndb/src/mgmsrv/MgmtSrvr.hpp780
-rw-r--r--ndb/src/mgmsrv/MgmtSrvrConfig.cpp312
-rw-r--r--ndb/src/mgmsrv/MgmtSrvrGeneralSignalHandling.cpp140
-rw-r--r--ndb/src/mgmsrv/NodeLogLevel.cpp68
-rw-r--r--ndb/src/mgmsrv/NodeLogLevel.hpp53
-rw-r--r--ndb/src/mgmsrv/NodeLogLevelList.cpp182
-rw-r--r--ndb/src/mgmsrv/NodeLogLevelList.hpp93
-rw-r--r--ndb/src/mgmsrv/Services.cpp1082
-rw-r--r--ndb/src/mgmsrv/Services.hpp125
-rw-r--r--ndb/src/mgmsrv/SignalQueue.cpp105
-rw-r--r--ndb/src/mgmsrv/SignalQueue.hpp100
-rw-r--r--ndb/src/mgmsrv/convertStrToInt.cpp45
-rw-r--r--ndb/src/mgmsrv/convertStrToInt.hpp25
-rw-r--r--ndb/src/mgmsrv/main.cpp429
-rw-r--r--ndb/src/mgmsrv/mkconfig/Makefile14
-rw-r--r--ndb/src/mgmsrv/mkconfig/mkconfig.cpp65
-rw-r--r--ndb/src/ndbapi/API.hpp26
-rw-r--r--ndb/src/ndbapi/ClusterMgr.cpp772
-rw-r--r--ndb/src/ndbapi/ClusterMgr.hpp215
-rw-r--r--ndb/src/ndbapi/DictCache.cpp246
-rw-r--r--ndb/src/ndbapi/DictCache.hpp79
-rw-r--r--ndb/src/ndbapi/Makefile61
-rw-r--r--ndb/src/ndbapi/Ndb.cpp1285
-rw-r--r--ndb/src/ndbapi/NdbApiSignal.cpp300
-rw-r--r--ndb/src/ndbapi/NdbApiSignal.hpp212
-rw-r--r--ndb/src/ndbapi/NdbConnection.cpp1787
-rw-r--r--ndb/src/ndbapi/NdbConnectionScan.cpp572
-rw-r--r--ndb/src/ndbapi/NdbCursorOperation.cpp57
-rw-r--r--ndb/src/ndbapi/NdbDictionary.cpp800
-rw-r--r--ndb/src/ndbapi/NdbDictionaryImpl.cpp2733
-rw-r--r--ndb/src/ndbapi/NdbDictionaryImpl.hpp653
-rw-r--r--ndb/src/ndbapi/NdbEventOperation.cpp125
-rw-r--r--ndb/src/ndbapi/NdbEventOperationImpl.cpp1305
-rw-r--r--ndb/src/ndbapi/NdbEventOperationImpl.hpp206
-rw-r--r--ndb/src/ndbapi/NdbImpl.hpp173
-rw-r--r--ndb/src/ndbapi/NdbIndexOperation.cpp719
-rw-r--r--ndb/src/ndbapi/NdbLinHash.hpp446
-rw-r--r--ndb/src/ndbapi/NdbOperation.cpp432
-rw-r--r--ndb/src/ndbapi/NdbOperationDefine.cpp769
-rw-r--r--ndb/src/ndbapi/NdbOperationExec.cpp967
-rw-r--r--ndb/src/ndbapi/NdbOperationInt.cpp1130
-rw-r--r--ndb/src/ndbapi/NdbOperationScan.cpp576
-rw-r--r--ndb/src/ndbapi/NdbOperationSearch.cpp500
-rw-r--r--ndb/src/ndbapi/NdbPool.cpp69
-rw-r--r--ndb/src/ndbapi/NdbPoolImpl.cpp527
-rw-r--r--ndb/src/ndbapi/NdbPoolImpl.hpp162
-rw-r--r--ndb/src/ndbapi/NdbRecAttr.cpp126
-rw-r--r--ndb/src/ndbapi/NdbReceiver.cpp46
-rw-r--r--ndb/src/ndbapi/NdbResultSet.cpp102
-rw-r--r--ndb/src/ndbapi/NdbScanFilter.cpp779
-rw-r--r--ndb/src/ndbapi/NdbScanOperation.cpp647
-rw-r--r--ndb/src/ndbapi/NdbScanReceiver.cpp187
-rw-r--r--ndb/src/ndbapi/NdbScanReceiver.hpp211
-rw-r--r--ndb/src/ndbapi/NdbSchemaCon.cpp163
-rw-r--r--ndb/src/ndbapi/NdbSchemaOp.cpp202
-rw-r--r--ndb/src/ndbapi/NdbUtil.cpp69
-rw-r--r--ndb/src/ndbapi/NdbUtil.hpp99
-rw-r--r--ndb/src/ndbapi/Ndberror.cpp635
-rw-r--r--ndb/src/ndbapi/Ndbif.cpp1356
-rw-r--r--ndb/src/ndbapi/Ndbinit.cpp285
-rw-r--r--ndb/src/ndbapi/Ndblist.cpp798
-rw-r--r--ndb/src/ndbapi/ObjectMap.hpp145
-rw-r--r--ndb/src/ndbapi/ScanOperation.txt10
-rw-r--r--ndb/src/ndbapi/TransporterFacade.cpp989
-rw-r--r--ndb/src/ndbapi/TransporterFacade.hpp316
-rw-r--r--ndb/src/ndbapi/signal-sender/Makefile19
-rw-r--r--ndb/src/ndbapi/signal-sender/SignalSender.cpp238
-rw-r--r--ndb/src/ndbapi/signal-sender/SignalSender.hpp82
-rw-r--r--ndb/src/ndbbaseclient/Makefile29
-rw-r--r--ndb/src/ndbbaseclient/ndbbaseclient_dummy.cpp0
-rw-r--r--ndb/src/ndbclient/Makefile37
-rw-r--r--ndb/src/ndbclient/ndbclient_dummy.cpp0
-rw-r--r--ndb/src/newtonapi/Makefile27
-rw-r--r--ndb/src/newtonapi/dba_binding.cpp433
-rw-r--r--ndb/src/newtonapi/dba_bulkread.cpp267
-rw-r--r--ndb/src/newtonapi/dba_config.cpp115
-rw-r--r--ndb/src/newtonapi/dba_dac.cpp842
-rw-r--r--ndb/src/newtonapi/dba_error.cpp118
-rw-r--r--ndb/src/newtonapi/dba_init.cpp86
-rw-r--r--ndb/src/newtonapi/dba_internal.hpp123
-rw-r--r--ndb/src/newtonapi/dba_process.cpp123
-rw-r--r--ndb/src/newtonapi/dba_process.hpp56
-rw-r--r--ndb/src/newtonapi/dba_schema.cpp149
-rw-r--r--ndb/src/rep/ExtSender.cpp149
-rw-r--r--ndb/src/rep/ExtSender.hpp76
-rw-r--r--ndb/src/rep/Makefile30
-rw-r--r--ndb/src/rep/NodeConnectInfo.hpp29
-rw-r--r--ndb/src/rep/README147
-rw-r--r--ndb/src/rep/RepApiInterpreter.cpp80
-rw-r--r--ndb/src/rep/RepApiInterpreter.hpp54
-rw-r--r--ndb/src/rep/RepApiService.cpp320
-rw-r--r--ndb/src/rep/RepApiService.hpp59
-rw-r--r--ndb/src/rep/RepCommandInterpreter.cpp456
-rw-r--r--ndb/src/rep/RepCommandInterpreter.hpp45
-rw-r--r--ndb/src/rep/RepComponents.cpp138
-rw-r--r--ndb/src/rep/RepComponents.hpp61
-rw-r--r--ndb/src/rep/RepMain.cpp98
-rw-r--r--ndb/src/rep/Requestor.cpp225
-rw-r--r--ndb/src/rep/Requestor.hpp153
-rw-r--r--ndb/src/rep/RequestorSubscriptions.cpp60
-rw-r--r--ndb/src/rep/SignalQueue.cpp106
-rw-r--r--ndb/src/rep/SignalQueue.hpp117
-rw-r--r--ndb/src/rep/TODO119
-rw-r--r--ndb/src/rep/adapters/AppNDB.cpp581
-rw-r--r--ndb/src/rep/adapters/AppNDB.hpp145
-rw-r--r--ndb/src/rep/adapters/ExtAPI.cpp31
-rw-r--r--ndb/src/rep/adapters/ExtAPI.hpp107
-rw-r--r--ndb/src/rep/adapters/ExtNDB.cpp552
-rw-r--r--ndb/src/rep/adapters/ExtNDB.hpp113
-rw-r--r--ndb/src/rep/adapters/Makefile11
-rw-r--r--ndb/src/rep/adapters/TableInfoPs.hpp118
-rw-r--r--ndb/src/rep/dbug_hack.cpp73
-rw-r--r--ndb/src/rep/rep_version.hpp86
-rw-r--r--ndb/src/rep/repapi/Makefile25
-rw-r--r--ndb/src/rep/repapi/repapi.cpp603
-rw-r--r--ndb/src/rep/repapi/repapi.h216
-rw-r--r--ndb/src/rep/state/Channel.cpp487
-rw-r--r--ndb/src/rep/state/Channel.hpp206
-rw-r--r--ndb/src/rep/state/Interval.cpp169
-rw-r--r--ndb/src/rep/state/Interval.hpp107
-rw-r--r--ndb/src/rep/state/Makefile17
-rw-r--r--ndb/src/rep/state/RepState.cpp865
-rw-r--r--ndb/src/rep/state/RepState.hpp275
-rw-r--r--ndb/src/rep/state/RepStateEvent.cpp284
-rw-r--r--ndb/src/rep/state/RepStateRequests.cpp294
-rw-r--r--ndb/src/rep/state/testInterval/Makefile12
-rw-r--r--ndb/src/rep/state/testInterval/testInterval.cpp127
-rw-r--r--ndb/src/rep/state/testRepState/Makefile15
-rw-r--r--ndb/src/rep/state/testRepState/testRequestor.cpp166
-rw-r--r--ndb/src/rep/state/testRepState/testRequestor.hpp24
-rw-r--r--ndb/src/rep/storage/GCIBuffer.cpp174
-rw-r--r--ndb/src/rep/storage/GCIBuffer.hpp112
-rw-r--r--ndb/src/rep/storage/GCIContainer.cpp272
-rw-r--r--ndb/src/rep/storage/GCIContainer.hpp120
-rw-r--r--ndb/src/rep/storage/GCIContainerPS.cpp128
-rw-r--r--ndb/src/rep/storage/GCIContainerPS.hpp90
-rw-r--r--ndb/src/rep/storage/GCIPage.cpp165
-rw-r--r--ndb/src/rep/storage/GCIPage.hpp114
-rw-r--r--ndb/src/rep/storage/LogRecord.hpp82
-rw-r--r--ndb/src/rep/storage/Makefile14
-rw-r--r--ndb/src/rep/storage/NodeConnectInfo.hpp29
-rw-r--r--ndb/src/rep/storage/NodeGroup.cpp149
-rw-r--r--ndb/src/rep/storage/NodeGroup.hpp109
-rw-r--r--ndb/src/rep/storage/NodeGroupInfo.cpp218
-rw-r--r--ndb/src/rep/storage/NodeGroupInfo.hpp147
-rw-r--r--ndb/src/rep/transfer/Makefile11
-rw-r--r--ndb/src/rep/transfer/TransPS.cpp550
-rw-r--r--ndb/src/rep/transfer/TransPS.hpp132
-rw-r--r--ndb/src/rep/transfer/TransSS.cpp650
-rw-r--r--ndb/src/rep/transfer/TransSS.hpp143
-rw-r--r--ndb/src/rep/transfer/TransSSSubscriptions.cpp193
-rw-r--r--ndb/src/scripts/Makefile5
-rw-r--r--ndb/test/Makefile19
-rw-r--r--ndb/test/include/HugoAsynchTransactions.hpp75
-rw-r--r--ndb/test/include/HugoCalculator.hpp59
-rw-r--r--ndb/test/include/HugoOperations.hpp149
-rw-r--r--ndb/test/include/HugoTransactions.hpp129
-rw-r--r--ndb/test/include/NDBT.hpp39
-rw-r--r--ndb/test/include/NDBT_DataSet.hpp140
-rw-r--r--ndb/test/include/NDBT_DataSetTransaction.hpp162
-rw-r--r--ndb/test/include/NDBT_Error.hpp97
-rw-r--r--ndb/test/include/NDBT_Output.hpp30
-rw-r--r--ndb/test/include/NDBT_ResultRow.hpp55
-rw-r--r--ndb/test/include/NDBT_ReturnCodes.h42
-rw-r--r--ndb/test/include/NDBT_Stats.hpp76
-rw-r--r--ndb/test/include/NDBT_Table.hpp135
-rw-r--r--ndb/test/include/NDBT_Tables.hpp47
-rw-r--r--ndb/test/include/NDBT_Test.hpp417
-rw-r--r--ndb/test/include/NdbBackup.hpp54
-rw-r--r--ndb/test/include/NdbConfig.hpp46
-rw-r--r--ndb/test/include/NdbGrep.hpp53
-rw-r--r--ndb/test/include/NdbRestarter.hpp95
-rw-r--r--ndb/test/include/NdbRestarts.hpp120
-rw-r--r--ndb/test/include/NdbTest.hpp35
-rw-r--r--ndb/test/include/NdbTimer.hpp110
-rw-r--r--ndb/test/include/TestNdbEventOperation.hpp24
-rw-r--r--ndb/test/include/UtilTransactions.hpp108
-rw-r--r--ndb/test/ndbapi/Makefile55
-rw-r--r--ndb/test/ndbapi/acid/Makefile10
-rw-r--r--ndb/test/ndbapi/acid/acid.cpp561
-rw-r--r--ndb/test/ndbapi/acid2/Makefile10
-rw-r--r--ndb/test/ndbapi/acid2/TraceNdbApi.cpp543
-rw-r--r--ndb/test/ndbapi/acid2/TraceNdbApi.hpp132
-rw-r--r--ndb/test/ndbapi/acid2/VerifyNdbApi.cpp151
-rw-r--r--ndb/test/ndbapi/acid2/VerifyNdbApi.hpp466
-rw-r--r--ndb/test/ndbapi/acid2/acid2.cpp693
-rw-r--r--ndb/test/ndbapi/bank/Bank.hpp142
-rw-r--r--ndb/test/ndbapi/bank/Makefile12
-rw-r--r--ndb/test/ndbapi/bank/bankCreator/Makefile8
-rw-r--r--ndb/test/ndbapi/bank/bankCreator/bankCreator.cpp55
-rw-r--r--ndb/test/ndbapi/bank/bankMakeGL/Makefile8
-rw-r--r--ndb/test/ndbapi/bank/bankMakeGL/bankMakeGL.cpp56
-rw-r--r--ndb/test/ndbapi/bank/bankSumAccounts/Makefile8
-rw-r--r--ndb/test/ndbapi/bank/bankSumAccounts/bankSumAccounts.cpp56
-rw-r--r--ndb/test/ndbapi/bank/bankTimer/Makefile8
-rw-r--r--ndb/test/ndbapi/bank/bankTimer/bankTimer.cpp58
-rw-r--r--ndb/test/ndbapi/bank/bankTransactionMaker/Makefile8
-rw-r--r--ndb/test/ndbapi/bank/bankTransactionMaker/bankTransactionMaker.cpp58
-rw-r--r--ndb/test/ndbapi/bank/bankValidateAllGLs/Makefile8
-rw-r--r--ndb/test/ndbapi/bank/bankValidateAllGLs/bankValidateAllGLs.cpp56
-rw-r--r--ndb/test/ndbapi/bank/src/Bank.cpp2458
-rw-r--r--ndb/test/ndbapi/bank/src/BankLoad.cpp582
-rw-r--r--ndb/test/ndbapi/bank/src/Makefile7
-rw-r--r--ndb/test/ndbapi/bank/testBank/Makefile9
-rw-r--r--ndb/test/ndbapi/bank/testBank/testBank.cpp150
-rwxr-xr-xndb/test/ndbapi/basicAsynch/Makefile9
-rwxr-xr-xndb/test/ndbapi/basicAsynch/testBasicAsynch.cpp186
-rw-r--r--ndb/test/ndbapi/bulk_copy/Makefile9
-rw-r--r--ndb/test/ndbapi/bulk_copy/bulk_copy.cpp276
-rw-r--r--ndb/test/ndbapi/cello-sessionDb/celloDb.cpp1503
-rw-r--r--ndb/test/ndbapi/create_all_tabs/Makefile11
-rw-r--r--ndb/test/ndbapi/create_all_tabs/create_all_tabs.cpp63
-rw-r--r--ndb/test/ndbapi/create_tab/Makefile11
-rw-r--r--ndb/test/ndbapi/create_tab/create_tab.cpp107
-rw-r--r--ndb/test/ndbapi/drop_all_tabs/Makefile11
-rw-r--r--ndb/test/ndbapi/drop_all_tabs/drop_all_tabs.cpp56
-rw-r--r--ndb/test/ndbapi/flexAsynch/Makefile47
-rw-r--r--ndb/test/ndbapi/flexAsynch/flexAsynch.cpp984
-rw-r--r--ndb/test/ndbapi/flexBench/Makefile47
-rw-r--r--ndb/test/ndbapi/flexBench/flexBench.cpp1157
-rwxr-xr-xndb/test/ndbapi/flexBench/ndbplot.pl305
-rw-r--r--ndb/test/ndbapi/flexHammer/Makefile9
-rw-r--r--ndb/test/ndbapi/flexHammer/README67
-rw-r--r--ndb/test/ndbapi/flexHammer/flexHammer.cpp891
-rw-r--r--ndb/test/ndbapi/flexScan/Makefile9
-rw-r--r--ndb/test/ndbapi/flexScan/README66
-rw-r--r--ndb/test/ndbapi/flexScan/flexScan.cpp1677
-rw-r--r--ndb/test/ndbapi/flexTT/Makefile47
-rw-r--r--ndb/test/ndbapi/flexTT/flexTT.cpp928
-rw-r--r--ndb/test/ndbapi/flexTimedAsynch/Makefile11
-rw-r--r--ndb/test/ndbapi/flexTimedAsynch/flexTimedAsynch.cpp852
-rw-r--r--ndb/test/ndbapi/flex_bench_mysql/Makefile15
-rw-r--r--ndb/test/ndbapi/flex_bench_mysql/flex_bench_mysql.cpp1753
-rw-r--r--ndb/test/ndbapi/indexTest/Makefile9
-rw-r--r--ndb/test/ndbapi/indexTest/index.cpp998
-rw-r--r--ndb/test/ndbapi/indexTest2/Makefile9
-rw-r--r--ndb/test/ndbapi/indexTest2/index2.cpp836
-rw-r--r--ndb/test/ndbapi/interpreterInTup/Makefile10
-rw-r--r--ndb/test/ndbapi/interpreterInTup/interpreterInTup.cpp1527
-rw-r--r--ndb/test/ndbapi/lmc-bench/Makefile6
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/Makefile8
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/generator/Makefile13
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/generator/asyncGenerator.cpp572
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/generator/mainAsyncGenerator.cpp395
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/include/dbGenerator.h63
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/include/testData.h156
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/include/userInterface.h79
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/user/Makefile11
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/user/macros.h52
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/user/ndb_async1.cpp647
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/user/ndb_async2.cpp754
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/user/ndb_error.hpp63
-rw-r--r--ndb/test/ndbapi/lmc-bench/async-src/user/userInterface.cpp120
-rw-r--r--ndb/test/ndbapi/lmc-bench/bin/.empty0
-rw-r--r--ndb/test/ndbapi/lmc-bench/include/ndb_schema.hpp78
-rw-r--r--ndb/test/ndbapi/lmc-bench/include/testDefinitions.h96
-rw-r--r--ndb/test/ndbapi/lmc-bench/lib/.empty0
-rw-r--r--ndb/test/ndbapi/lmc-bench/script/Makefile5
-rwxr-xr-xndb/test/ndbapi/lmc-bench/script/async-lmc-bench-l-p10.sh14
-rwxr-xr-xndb/test/ndbapi/lmc-bench/script/async-lmc-bench-l.sh14
-rwxr-xr-xndb/test/ndbapi/lmc-bench/script/async-lmc-bench-p10.sh14
-rwxr-xr-xndb/test/ndbapi/lmc-bench/script/async-lmc-bench.sh14
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/Makefile8
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/README8
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/generator/Makefile17
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/generator/dbGenerator.c544
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/generator/dbGenerator.h61
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/generator/mainGenerator.c327
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/include/testData.h103
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/include/userInterface.h128
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/makevars.linux6
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/makevars.sparc15
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/populator/Makefile15
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/populator/dbPopulate.c246
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/populator/dbPopulate.h59
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/populator/mainPopulate.c78
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/Makefile11
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/localDbPrepare.c648
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/macros.h52
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/ndb_error.hpp32
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/ndb_user_populate.cpp165
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction.cpp825
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction2.cpp825
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction3.cpp793
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction4.cpp770
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction5.cpp769
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction6.cpp561
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/old/Makefile10
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/old/userHandle.h190
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/old/userInterface.c456
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/old/userTransaction.c474
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/userHandle.h51
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/userInterface.cpp742
-rw-r--r--ndb/test/ndbapi/lmc-bench/src/user/userTransaction.c474
-rw-r--r--ndb/test/ndbapi/restarter/Makefile11
-rw-r--r--ndb/test/ndbapi/restarter/restarter.cpp131
-rw-r--r--ndb/test/ndbapi/restarter2/Makefile11
-rw-r--r--ndb/test/ndbapi/restarter2/restarter2.cpp118
-rw-r--r--ndb/test/ndbapi/restarts/Makefile11
-rw-r--r--ndb/test/ndbapi/restarts/restarts.cpp117
-rw-r--r--ndb/test/ndbapi/ronja/benchronja/Makefile10
-rw-r--r--ndb/test/ndbapi/ronja/benchronja/benchronja.cpp1205
-rw-r--r--ndb/test/ndbapi/ronja/initronja/Makefile9
-rw-r--r--ndb/test/ndbapi/ronja/initronja/initronja.cpp349
-rw-r--r--ndb/test/ndbapi/telco/InsertRecs.cpp571
-rw-r--r--ndb/test/ndbapi/telco/Makefile10
-rw-r--r--ndb/test/ndbapi/telco/adoInsertRecs.cpp363
-rw-r--r--ndb/test/ndbapi/telco/msa.cpp1206
-rw-r--r--ndb/test/ndbapi/telco/readme9
-rw-r--r--ndb/test/ndbapi/testBackup/Makefile9
-rw-r--r--ndb/test/ndbapi/testBackup/testBackup.cpp476
-rw-r--r--ndb/test/ndbapi/testBasic/Makefile9
-rw-r--r--ndb/test/ndbapi/testBasic/testBasic.cpp1265
-rw-r--r--ndb/test/ndbapi/testBlobs/Makefile11
-rw-r--r--ndb/test/ndbapi/testBlobs/testBlobs.cpp200
-rw-r--r--ndb/test/ndbapi/testDataBuffers/Makefile9
-rw-r--r--ndb/test/ndbapi/testDataBuffers/testDataBuffers.cpp618
-rw-r--r--ndb/test/ndbapi/testDict/Makefile11
-rw-r--r--ndb/test/ndbapi/testDict/testDict.cpp1578
-rw-r--r--ndb/test/ndbapi/testGrep/Makefile10
-rw-r--r--ndb/test/ndbapi/testGrep/testGrep.cpp540
-rw-r--r--ndb/test/ndbapi/testGrep/verify/Makefile11
-rw-r--r--ndb/test/ndbapi/testGrep/verify/testGrepVerify.cpp122
-rw-r--r--ndb/test/ndbapi/testIndex/Makefile11
-rw-r--r--ndb/test/ndbapi/testIndex/testIndex.cpp1494
-rw-r--r--ndb/test/ndbapi/testInterpreter/Makefile9
-rw-r--r--ndb/test/ndbapi/testInterpreter/testInterpreter.cpp231
-rw-r--r--ndb/test/ndbapi/testMgm/Makefile9
-rw-r--r--ndb/test/ndbapi/testMgm/testMgm.cpp185
-rw-r--r--ndb/test/ndbapi/testNdbApi/Makefile9
-rw-r--r--ndb/test/ndbapi/testNdbApi/testNdbApi.cpp1013
-rw-r--r--ndb/test/ndbapi/testNodeRestart/Makefile9
-rw-r--r--ndb/test/ndbapi/testNodeRestart/testNodeRestart.cpp449
-rw-r--r--ndb/test/ndbapi/testOIBasic/Makefile13
-rw-r--r--ndb/test/ndbapi/testOIBasic/testOIBasic.cpp2772
-rw-r--r--ndb/test/ndbapi/testOIBasic/times.txt8
-rw-r--r--ndb/test/ndbapi/testOperations/Makefile9
-rw-r--r--ndb/test/ndbapi/testOperations/testOperations.cpp271
-rw-r--r--ndb/test/ndbapi/testOrderedIndex/Makefile9
-rw-r--r--ndb/test/ndbapi/testOrderedIndex/testOrderedIndex.cpp224
-rw-r--r--ndb/test/ndbapi/testRestartGci/Makefile9
-rw-r--r--ndb/test/ndbapi/testRestartGci/testRestartGci.cpp218
-rw-r--r--ndb/test/ndbapi/testScan/Makefile9
-rw-r--r--ndb/test/ndbapi/testScan/ScanFunctions.hpp392
-rw-r--r--ndb/test/ndbapi/testScan/testScan.cpp1293
-rw-r--r--ndb/test/ndbapi/testScanInterpreter/Makefile9
-rw-r--r--ndb/test/ndbapi/testScanInterpreter/ScanFilter.hpp131
-rw-r--r--ndb/test/ndbapi/testScanInterpreter/ScanInterpretTest.hpp528
-rw-r--r--ndb/test/ndbapi/testScanInterpreter/testScanInterpreter.cpp280
-rw-r--r--ndb/test/ndbapi/testSystemRestart/Makefile11
-rw-r--r--ndb/test/ndbapi/testSystemRestart/testSystemRestart.cpp942
-rw-r--r--ndb/test/ndbapi/testTimeout/Makefile9
-rw-r--r--ndb/test/ndbapi/testTimeout/testTimeout.cpp261
-rw-r--r--ndb/test/ndbapi/testTransactions/Makefile10
-rw-r--r--ndb/test/ndbapi/testTransactions/testTransactions.cpp411
-rw-r--r--ndb/test/ndbapi/test_event/Makefile9
-rw-r--r--ndb/test/ndbapi/test_event/test_event.cpp142
-rw-r--r--ndb/test/ndbapi/vw_test/Makefile11
-rw-r--r--ndb/test/ndbapi/vw_test/bcd.h27
-rw-r--r--ndb/test/ndbapi/vw_test/cdrserver.cpp1633
-rw-r--r--ndb/test/ndbapi/vw_test/script/client_start10
-rw-r--r--ndb/test/ndbapi/vw_test/size.cpp27
-rw-r--r--ndb/test/ndbapi/vw_test/utv.h161
-rw-r--r--ndb/test/ndbapi/vw_test/vcdrfunc.h55
-rw-r--r--ndb/test/ndbnet/test.run3
-rw-r--r--ndb/test/ndbnet/testError.run266
-rw-r--r--ndb/test/ndbnet/testMNF.run277
-rw-r--r--ndb/test/ndbnet/testNR.run60
-rw-r--r--ndb/test/ndbnet/testNR1.run61
-rw-r--r--ndb/test/ndbnet/testNR4.run77
-rw-r--r--ndb/test/ndbnet/testSRhang.run50
-rw-r--r--ndb/test/ndbnet/testTR295.run75
-rw-r--r--ndb/test/newtonapi/Makefile8
-rw-r--r--ndb/test/newtonapi/basic_test/Makefile25
-rw-r--r--ndb/test/newtonapi/basic_test/basic/Makefile14
-rw-r--r--ndb/test/newtonapi/basic_test/basic/basic.cpp322
-rw-r--r--ndb/test/newtonapi/basic_test/bulk_read/Makefile14
-rw-r--r--ndb/test/newtonapi/basic_test/bulk_read/br_test.cpp263
-rw-r--r--ndb/test/newtonapi/basic_test/common.cpp137
-rw-r--r--ndb/test/newtonapi/basic_test/common.hpp65
-rw-r--r--ndb/test/newtonapi/basic_test/ptr_binding/Makefile14
-rw-r--r--ndb/test/newtonapi/basic_test/ptr_binding/ptr_binding_test.cpp265
-rw-r--r--ndb/test/newtonapi/basic_test/too_basic.cpp109
-rw-r--r--ndb/test/newtonapi/perf_test/Makefile14
-rw-r--r--ndb/test/newtonapi/perf_test/perf.cpp646
-rw-r--r--ndb/test/odbc/Makefile13
-rw-r--r--ndb/test/odbc/SQL99_test/Makefile26
-rw-r--r--ndb/test/odbc/SQL99_test/SQL99_test.cpp2138
-rw-r--r--ndb/test/odbc/SQL99_test/SQL99_test.h261
-rw-r--r--ndb/test/odbc/client/Makefile95
-rw-r--r--ndb/test/odbc/client/NDBT_ALLOCHANDLE.cpp53
-rw-r--r--ndb/test/odbc/client/NDBT_ALLOCHANDLE_HDBC.cpp59
-rw-r--r--ndb/test/odbc/client/NDBT_SQLConnect.cpp82
-rw-r--r--ndb/test/odbc/client/NDBT_SQLPrepare.cpp109
-rw-r--r--ndb/test/odbc/client/SQLAllocEnvTest.cpp115
-rw-r--r--ndb/test/odbc/client/SQLAllocHandleTest.cpp314
-rw-r--r--ndb/test/odbc/client/SQLAllocHandleTest_bf.cpp259
-rw-r--r--ndb/test/odbc/client/SQLBindColTest.cpp537
-rw-r--r--ndb/test/odbc/client/SQLBindParameterTest.cpp219
-rw-r--r--ndb/test/odbc/client/SQLCancelTest.cpp254
-rw-r--r--ndb/test/odbc/client/SQLCloseCursorTest.cpp92
-rw-r--r--ndb/test/odbc/client/SQLColAttributeTest.cpp328
-rw-r--r--ndb/test/odbc/client/SQLColAttributeTest1.cpp143
-rw-r--r--ndb/test/odbc/client/SQLColAttributeTest2.cpp277
-rw-r--r--ndb/test/odbc/client/SQLColAttributeTest3.cpp275
-rw-r--r--ndb/test/odbc/client/SQLConnectTest.cpp165
-rw-r--r--ndb/test/odbc/client/SQLCopyDescTest.cpp140
-rw-r--r--ndb/test/odbc/client/SQLDescribeColTest.cpp260
-rw-r--r--ndb/test/odbc/client/SQLDisconnectTest.cpp155
-rw-r--r--ndb/test/odbc/client/SQLDriverConnectTest.cpp96
-rw-r--r--ndb/test/odbc/client/SQLEndTranTest.cpp108
-rw-r--r--ndb/test/odbc/client/SQLErrorTest.cpp107
-rw-r--r--ndb/test/odbc/client/SQLExecDirectTest.cpp353
-rw-r--r--ndb/test/odbc/client/SQLExecuteTest.cpp122
-rw-r--r--ndb/test/odbc/client/SQLFetchScrollTest.cpp82
-rw-r--r--ndb/test/odbc/client/SQLFetchTest.cpp438
-rw-r--r--ndb/test/odbc/client/SQLFreeHandleTest.cpp195
-rw-r--r--ndb/test/odbc/client/SQLFreeStmtTest.cpp182
-rw-r--r--ndb/test/odbc/client/SQLGetConnectAttrTest.cpp131
-rw-r--r--ndb/test/odbc/client/SQLGetCursorNameTest.cpp221
-rw-r--r--ndb/test/odbc/client/SQLGetDataTest.cpp358
-rw-r--r--ndb/test/odbc/client/SQLGetDescFieldTest.cpp113
-rw-r--r--ndb/test/odbc/client/SQLGetDescRecTest.cpp95
-rw-r--r--ndb/test/odbc/client/SQLGetDiagFieldTest.cpp236
-rw-r--r--ndb/test/odbc/client/SQLGetDiagRecSimpleTest.cpp167
-rw-r--r--ndb/test/odbc/client/SQLGetDiagRecTest.cpp207
-rw-r--r--ndb/test/odbc/client/SQLGetEnvAttrTest.cpp110
-rw-r--r--ndb/test/odbc/client/SQLGetFunctionsTest.cpp284
-rw-r--r--ndb/test/odbc/client/SQLGetInfoTest.cpp215
-rw-r--r--ndb/test/odbc/client/SQLGetStmtAttrTest.cpp155
-rw-r--r--ndb/test/odbc/client/SQLGetTypeInfoTest.cpp202
-rw-r--r--ndb/test/odbc/client/SQLMoreResultsTest.cpp91
-rw-r--r--ndb/test/odbc/client/SQLNumResultColsTest.cpp202
-rw-r--r--ndb/test/odbc/client/SQLParamDataTest.cpp105
-rw-r--r--ndb/test/odbc/client/SQLPrepareTest.cpp285
-rw-r--r--ndb/test/odbc/client/SQLPutDataTest.cpp108
-rw-r--r--ndb/test/odbc/client/SQLRowCountTest.cpp203
-rw-r--r--ndb/test/odbc/client/SQLSetConnectAttrTest.cpp131
-rw-r--r--ndb/test/odbc/client/SQLSetCursorNameTest.cpp215
-rw-r--r--ndb/test/odbc/client/SQLSetDescFieldTest.cpp100
-rw-r--r--ndb/test/odbc/client/SQLSetDescRecTest.cpp99
-rw-r--r--ndb/test/odbc/client/SQLSetEnvAttrTest.cpp108
-rw-r--r--ndb/test/odbc/client/SQLSetStmtAttrTest.cpp108
-rw-r--r--ndb/test/odbc/client/SQLTablesTest.cpp227
-rw-r--r--ndb/test/odbc/client/SQLTransactTest.cpp305
-rw-r--r--ndb/test/odbc/client/common.hpp81
-rw-r--r--ndb/test/odbc/client/main.cpp158
-rw-r--r--ndb/test/odbc/dm-iodbc/Makefile38
-rw-r--r--ndb/test/odbc/dm-unixodbc/Makefile39
-rw-r--r--ndb/test/odbc/driver/Makefile30
-rw-r--r--ndb/test/odbc/driver/testOdbcDriver.cpp4969
-rw-r--r--ndb/test/odbc/test_compiler/Makefile21
-rw-r--r--ndb/test/odbc/test_compiler/test_compiler.cpp233
-rw-r--r--ndb/test/odbc/tpcb/Makefile30
-rw-r--r--ndb/test/odbc/tpcb/Makefile_mysql33
-rw-r--r--ndb/test/odbc/tpcb/Makefile_ndb30
-rw-r--r--ndb/test/odbc/tpcb/readme.txt15
-rw-r--r--ndb/test/odbc/tpcb/timesten.h188
-rw-r--r--ndb/test/odbc/tpcb/tpcb.cpp1415
-rw-r--r--ndb/test/odbc/tpcb/ttTime.c366
-rw-r--r--ndb/test/odbc/tpcb/ttTime.h125
-rw-r--r--ndb/test/run-test/Makefile22
-rw-r--r--ndb/test/run-test/README.ATRT34
-rwxr-xr-xndb/test/run-test/atrt-analyze-result.sh12
-rwxr-xr-xndb/test/run-test/atrt-clear-result.sh4
-rwxr-xr-xndb/test/run-test/atrt-gather-result.sh16
-rwxr-xr-xndb/test/run-test/atrt-setup.sh6
-rw-r--r--ndb/test/run-test/main.cpp942
-rwxr-xr-xndb/test/run-test/make-config.sh465
-rwxr-xr-xndb/test/run-test/make-html-reports.sh437
-rwxr-xr-xndb/test/run-test/make-index.sh242
-rw-r--r--ndb/test/run-test/run-test.hpp92
-rw-r--r--ndb/test/src/HugoAsynchTransactions.cpp491
-rw-r--r--ndb/test/src/HugoCalculator.cpp243
-rw-r--r--ndb/test/src/HugoOperations.cpp793
-rw-r--r--ndb/test/src/HugoTransactions.cpp2404
-rw-r--r--ndb/test/src/Makefile31
-rw-r--r--ndb/test/src/NDBT_Error.cpp285
-rw-r--r--ndb/test/src/NDBT_Output.cpp36
-rw-r--r--ndb/test/src/NDBT_ResultRow.cpp199
-rw-r--r--ndb/test/src/NDBT_ReturnCodes.cpp50
-rw-r--r--ndb/test/src/NDBT_Table.cpp158
-rw-r--r--ndb/test/src/NDBT_Tables.cpp842
-rw-r--r--ndb/test/src/NDBT_Test.cpp1119
-rw-r--r--ndb/test/src/NdbBackup.cpp452
-rw-r--r--ndb/test/src/NdbConfig.cpp163
-rw-r--r--ndb/test/src/NdbGrep.cpp334
-rw-r--r--ndb/test/src/NdbRestarter.cpp664
-rw-r--r--ndb/test/src/NdbRestarts.cpp875
-rw-r--r--ndb/test/src/UtilTransactions.cpp1393
-rw-r--r--ndb/test/tools/Makefile9
-rw-r--r--ndb/test/tools/hugoCalculator/Makefile11
-rw-r--r--ndb/test/tools/hugoCalculator/hugoCalculator.cpp68
-rw-r--r--ndb/test/tools/hugoFill/Makefile11
-rw-r--r--ndb/test/tools/hugoFill/hugoFill.cpp78
-rw-r--r--ndb/test/tools/hugoLoad/Makefile11
-rw-r--r--ndb/test/tools/hugoLoad/hugoLoad.cpp82
-rw-r--r--ndb/test/tools/hugoLockRecords/Makefile9
-rw-r--r--ndb/test/tools/hugoLockRecords/hugoLockRecords.cpp90
-rw-r--r--ndb/test/tools/hugoPkDelete/Makefile9
-rw-r--r--ndb/test/tools/hugoPkDelete/hugoPkDel.cpp86
-rw-r--r--ndb/test/tools/hugoPkRead/Makefile9
-rw-r--r--ndb/test/tools/hugoPkRead/hugoPkRead.cpp91
-rw-r--r--ndb/test/tools/hugoPkReadRecord/Makefile11
-rw-r--r--ndb/test/tools/hugoPkReadRecord/hugoPkReadRecord.cpp193
-rw-r--r--ndb/test/tools/hugoPkUpdate/Makefile9
-rw-r--r--ndb/test/tools/hugoPkUpdate/hugoPkUpd.cpp88
-rw-r--r--ndb/test/tools/hugoScanRead/Makefile9
-rw-r--r--ndb/test/tools/hugoScanRead/hugoScanRead.cpp90
-rw-r--r--ndb/test/tools/hugoScanUpdate/Makefile9
-rw-r--r--ndb/test/tools/hugoScanUpdate/hugoScanUpd.cpp100
-rw-r--r--ndb/test/tools/restart/Makefile11
-rw-r--r--ndb/test/tools/restart/restart.cpp84
-rw-r--r--ndb/test/tools/waiter/Makefile11
-rw-r--r--ndb/test/tools/waiter/waiter.cpp57
-rw-r--r--ndb/tools/Makefile12
-rwxr-xr-xndb/tools/clean-links.sh21
-rw-r--r--ndb/tools/copy_tab/Makefile9
-rw-r--r--ndb/tools/copy_tab/copy_tab.cpp99
-rw-r--r--ndb/tools/cpcc/Makefile12
-rw-r--r--ndb/tools/cpcc/cpcc.cpp349
-rw-r--r--ndb/tools/create_index/Makefile11
-rw-r--r--ndb/tools/create_index/create_index.cpp95
-rw-r--r--ndb/tools/delete_all/Makefile9
-rw-r--r--ndb/tools/delete_all/delete_all.cpp93
-rw-r--r--ndb/tools/desc/Makefile9
-rw-r--r--ndb/tools/desc/desc.cpp78
-rw-r--r--ndb/tools/drop_index/Makefile11
-rw-r--r--ndb/tools/drop_index/drop_index.cpp76
-rw-r--r--ndb/tools/drop_tab/Makefile11
-rw-r--r--ndb/tools/drop_tab/drop_tab.cpp80
-rw-r--r--ndb/tools/init_rm/init_rm.c69
-rw-r--r--ndb/tools/list_tables/Makefile9
-rw-r--r--ndb/tools/list_tables/listTables.cpp194
-rw-r--r--ndb/tools/make-errors.pl181
-rwxr-xr-xndb/tools/make-links.sh20
-rw-r--r--ndb/tools/ndbnet/Makefile.PL158
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net.pm42
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/Base.pm12
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/Client.pm252
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/Command.pm641
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/Config.pm235
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/Database.pm321
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/Env.pm94
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/Node.pm747
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/NodeApi.pm84
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/NodeDb.pm116
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/NodeMgmt.pm318
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/Server.pm149
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/ServerINET.pm116
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Net/ServerUNIX.pm54
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Run.pm40
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Run/Base.pm12
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Run/Database.pm89
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Run/Env.pm84
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Run/Node.pm114
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util.pm37
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/Base.pm113
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/Dir.pm170
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/Event.pm103
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/File.pm163
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/IO.pm213
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/Lock.pm136
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/Log.pm367
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/Socket.pm158
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/SocketINET.pm86
-rw-r--r--ndb/tools/ndbnet/lib/NDB/Util/SocketUNIX.pm76
-rw-r--r--ndb/tools/ndbnet/ndbnet.pl339
-rw-r--r--ndb/tools/ndbnet/ndbnetd.pl400
-rw-r--r--ndb/tools/ndbnet/ndbrun33
-rw-r--r--ndb/tools/ndbsql/Makefile44
-rw-r--r--ndb/tools/ndbsql/ndbsql.cpp947
-rwxr-xr-xndb/tools/rgrep194
-rw-r--r--ndb/tools/select_all/Makefile9
-rw-r--r--ndb/tools/select_all/select_all.cpp286
-rw-r--r--ndb/tools/select_count/Makefile9
-rw-r--r--ndb/tools/select_count/select_count.cpp90
-rw-r--r--ndb/tools/src/counterviewer/CounterViewer.java725
-rw-r--r--ndb/tools/transproxy/Makefile29
-rw-r--r--ndb/tools/transproxy/transproxy.cpp369
-rw-r--r--ndb/tools/verify_index/Makefile9
-rw-r--r--ndb/tools/verify_index/verify_index.cpp86
1835 files changed, 500032 insertions, 0 deletions
diff --git a/BitKeeper/etc/logging_ok b/BitKeeper/etc/logging_ok
index b3533f2a834..9210c176e6e 100644
--- a/BitKeeper/etc/logging_ok
+++ b/BitKeeper/etc/logging_ok
@@ -76,6 +76,7 @@ konstantin@mysql.com
kostja@oak.local
lenz@kallisto.mysql.com
lenz@mysql.com
+magnus@neptunus.(none)
marko@hundin.mysql.fi
miguel@hegel.(none)
miguel@hegel.br
diff --git a/ndb/BinDist.sh b/ndb/BinDist.sh
new file mode 100644
index 00000000000..ed0b294c912
--- /dev/null
+++ b/ndb/BinDist.sh
@@ -0,0 +1,121 @@
+#
+# Invoked from scripts/make_binary_distribution as "sh BinDist.sh".
+# Prints list of dirs and files to include under mysql/ndb.
+#
+
+# release notes
+
+grep -v '^#' <<__END__
+#ReleaseNotes.html
+mysqlclusterenv.sh
+__END__
+
+# subset of bins, libs, includes
+
+grep -v '^#' <<__END__
+bin/
+bin/ndb
+bin/mgmtsrvr
+bin/mgmtclient
+bin/mysqlcluster
+bin/mysqlcluster_install_db
+bin/mysqlclusterd
+bin/restore
+bin/ndb_rep
+bin/desc
+bin/flexBench
+bin/select_all
+bin/select_count
+bin/delete_all
+bin/ndbsql
+bin/drop_tab
+bin/drop_index
+bin/list_tables
+bin/waiter
+lib/
+lib/libNEWTON_API.a
+lib/libNEWTON_API.so
+lib/libNDB_API.a
+lib/libNDB_API.so
+lib/libMGM_API.a
+lib/libMGM_API.so
+lib/libNDB_ODBC.so
+lib/libMGM_API_pic.a
+lib/libNDB_API_pic.a
+include/
+include/ndb_types.h
+include/ndb_version.h
+include/mgmapi/
+include/mgmapi/mgmapi.h
+include/mgmapi/mgmapi_debug.h
+include/ndbapi/
+include/ndbapi/ndbapi_limits.h
+include/ndbapi/AttrType.hpp
+include/ndbapi/Ndb.hpp
+include/ndbapi/NdbApi.hpp
+include/ndbapi/NdbConnection.hpp
+include/ndbapi/NdbCursorOperation.hpp
+include/ndbapi/NdbDictionary.hpp
+include/ndbapi/NdbError.hpp
+include/ndbapi/NdbEventOperation.hpp
+include/ndbapi/NdbIndexOperation.hpp
+include/ndbapi/NdbOperation.hpp
+include/ndbapi/NdbPool.hpp
+include/ndbapi/NdbRecAttr.hpp
+include/ndbapi/NdbReceiver.hpp
+include/ndbapi/NdbResultSet.hpp
+include/ndbapi/NdbScanFilter.hpp
+include/ndbapi/NdbScanOperation.hpp
+include/ndbapi/NdbSchemaCon.hpp
+include/ndbapi/NdbSchemaOp.hpp
+include/newtonapi/dba.h
+include/newtonapi/defs/pcn_types.h
+__END__
+
+#if [ -f /usr/local/lib/libstdc++.a ]; then
+# cp /usr/local/lib/libstdc++.a lib/.
+# echo lib/libstdc++.a
+#fi
+#if [ -f /usr/local/lib/libstdc++.so.5 ]; then
+# cp /usr/local/lib/libstdc++.so.5 lib/.
+# echo lib/libstdc++.so.5
+#fi
+#if [ -f /usr/local/lib/libgcc_s.so.1 ]; then
+# cp /usr/local/lib/libgcc_s.so.1 lib/.
+# echo lib/libgcc_s.so.1
+#fi
+
+# docs
+
+#find docs/*.html docs/*.pdf -print | sort -t/
+
+# demos
+
+find demos -print | grep -v /SCCS | sort -t/
+
+# examples
+
+grep -v '^#' <<__END__
+examples/
+examples/Makefile
+examples/ndbapi_example1/
+examples/ndbapi_example1/Makefile
+examples/ndbapi_example1/ndbapi_example1.cpp
+examples/ndbapi_example2/
+examples/ndbapi_example2/Makefile
+examples/ndbapi_example2/ndbapi_example2.cpp
+examples/ndbapi_example3/
+examples/ndbapi_example3/Makefile
+examples/ndbapi_example3/ndbapi_example3.cpp
+examples/ndbapi_example4/
+examples/ndbapi_example4/Makefile
+examples/ndbapi_example4/ndbapi_example4.cpp
+examples/ndbapi_example5/
+examples/ndbapi_example5/Makefile
+examples/ndbapi_example5/ndbapi_example5.cpp
+examples/select_all/
+examples/select_all/Makefile
+examples/select_all/select_all.cpp
+__END__
+
+exit 0
diff --git a/ndb/Defs.mk b/ndb/Defs.mk
new file mode 100644
index 00000000000..d5a21c64ca9
--- /dev/null
+++ b/ndb/Defs.mk
@@ -0,0 +1,84 @@
+include $(NDB_TOP)/config/config.mk
+include $(NDB_TOP)/config/Defs.$(NDB_VERSION).mk
+include $(NDB_TOP)/config/Defs.$(NDB_OS).$(NDB_ARCH).$(NDB_COMPILER).mk
+
+ifeq ($(NDB_OS), WIN32)
+# Windows specific definitions
+OBJEXT := obj
+LIBEXT := lib
+LIBPREFIX :=
+fixpath = `cygpath -w $1`
+ar_rcs = lib -out:`cygpath -w $1` $2
+link_so = link -DLL -OUT:`cygpath -w $1` $(WIN_LIBS) $2
+#check-odbc = Y
+USE_EDITLINE := N
+#STRCASECMP is defined in include/portlib/PortDefs.h to _strcmpi
+else
+#Common definitions for almost all non-Windows environments
+OBJEXT := o
+LIBEXT := a
+LIBPREFIX := lib
+fixpath = $1
+ar_rcs = $(AR_RCS) $1 $2
+#check-odbc = $(findstring sqlext.h, $(wildcard /usr/include/sqlext.h) $(wildcard /usr/local/include/sqlext.h))
+CCFLAGS_TOP += -DHAVE_STRCASECMP
+
+endif
+
+ifeq ($(NDB_OS), WIN32)
+CCFLAGS_TOP += -DHAVE_STRDUP
+NDB_STRLCPY := Y
+NDB_STRLCAT := Y
+SHLIBEXT := dll
+endif
+
+ifeq ($(NDB_OS), LINUX)
+CCFLAGS_TOP += -DHAVE_STRDUP
+NDB_STRLCAT := Y
+NDB_STRLCPY := Y
+SHLIBEXT := so
+endif
+
+ifeq ($(NDB_OS), SOLARIS)
+CCFLAGS_TOP += -DHAVE_STRDUP
+NDB_STRLCAT := Y
+NDB_STRLCPY := Y
+SHLIBEXT := so
+endif
+
+ifeq ($(NDB_OS), HPUX)
+CCFLAGS_TOP += -DHAVE_STRDUP
+NDB_STRLCAT := Y
+NDB_STRLCPY := Y
+SHLIBEXT := sl
+endif
+
+ifeq ($(NDB_OS), MACOSX)
+CCFLAGS_TOP += -DHAVE_STRLCAT
+CCFLAGS_TOP += -DHAVE_STRLCAT
+CCFLAGS_TOP += -DHAVE_STRLCPY
+CCFLAGS_TOP += -DNDBOUT_UINTPTR
+SHLIBEXT := dylib
+endif
+
+ifeq ($(NDB_OS), OSE)
+NDB_STRDUP := Y
+NDB_STRLCAT := Y
+NDB_STRLCPY := Y
+SHLIBEXT := so
+endif
+
+ifeq ($(NDB_OS), SOFTOSE)
+NDB_STRDUP := Y
+NDB_STRLCAT := Y
+NDB_STRLCPY := Y
+SHLIBEXT := so
+endif
+
+ifeq ($(NDB_SCI), Y)
+CCFLAGS_TOP += -DHAVE_SCI
+endif
+
+ifneq ($(findstring OSE, $(NDB_OS)),)
+USE_EDITLINE := N
+endif
diff --git a/ndb/Epilogue.mk b/ndb/Epilogue.mk
new file mode 100644
index 00000000000..89cd4034208
--- /dev/null
+++ b/ndb/Epilogue.mk
@@ -0,0 +1,853 @@
+# .KEEP_STATE:
+# bk test !!!
+
+###
+# For building some intermediary targets in /tmp (only useful on solaris)
+ifneq ($(NDB_BUILDROOT),)
+NDB_TOPABS := $(shell cd $(NDB_TOP) && /bin/pwd)
+NDB_BUILDDIR := $(subst $(NDB_TOPABS),$(NDB_BUILDROOT),$(CURDIR))/
+ifeq ($(wildcard $(NDB_BUILDDIR)),)
+dummy := $(shell mkdir -p $(NDB_BUILDDIR))
+endif
+endif
+
+###
+CCFLAGS_TOP += -DNDB_$(NDB_OS) -DNDB_$(NDB_ARCH) -DNDB_$(NDB_COMPILER)
+
+ifdef BIN_TARGET
+BIN_EXE = Y
+endif
+
+###
+#
+# OS specifics
+#
+
+# Disable shared libraries on HP-UX for the time being.
+ifeq ($(NDB_OS), HPUX)
+ SO_LIB := N
+ PIC_LIB := N
+ PIC_ARCHIVE := N
+ NONPIC_ARCHIVE := Y
+endif
+
+ifeq ($(NDB_OS), OSE)
+ SO_LIB := N
+ PIC_LIB := N
+ PIC_ARCHIVE := N
+ NONPIC_ARCHIVE := Y
+
+ifdef BIN_TARGET
+ BIN_LIB_TARGET := lib$(BIN_TARGET).a
+ BIN_TARGET := lib$(BIN_TARGET).a
+endif
+endif
+
+ifeq ($(NDB_OS), SOFTOSE)
+ SO_LIB := N
+ PIC_LIB := N
+ PIC_ARCHIVE := N
+
+ifdef BIN_TARGET
+ BIN_EXE_TARGET := $(BIN_TARGET)
+ BIN_LIB_TARGET := lib$(BIN_TARGET).a
+ EXTRA_MAIN := osemain.o
+endif
+endif
+
+ifeq ($(filter OSE, $(NDB_OS)),)
+ BIN_EXE_TARGET := $(BIN_TARGET)
+endif
+
+
+ifeq ($(NDB_OS), MACOSX)
+.LIBPATTERNS= lib%.dylib lib%.a
+endif
+
+###
+#
+#
+
+###
+# External dependencies definition : the place we store libraries
+# we get from outside the NDB development group.
+EXTERNAL_DEPENDS_TOP=$(NDB_TOP)/src/external/$(NDB_OS).$(NDB_ARCH)
+
+
+###
+#
+# TYPE Handling
+
+#
+# TYPE := kernel
+#
+ifneq ($(filter kernel, $(TYPE)),)
+CCFLAGS_LOC += \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/vm) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/error) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel) \
+ -I$(call fixpath,$(NDB_TOP)/include/kernel) \
+ -I$(call fixpath,$(NDB_TOP)/include/transporter) \
+ -I$(call fixpath,$(NDB_TOP)/include/debugger) \
+ -I$(call fixpath,$(NDB_TOP)/include/mgmcommon) \
+ -I$(call fixpath,$(NDB_TOP)/include/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/include/util) \
+ -I$(call fixpath,$(NDB_TOP)/include/portlib) \
+ -I$(call fixpath,$(NDB_TOP)/include/logger)
+endif
+
+#
+# TYPE := ndbapi
+#
+ifneq ($(filter ndbapi, $(TYPE)),)
+CCFLAGS_LOC += \
+ -I$(call fixpath,$(NDB_TOP)/include/kernel) \
+ -I$(call fixpath,$(NDB_TOP)/include/transporter) \
+ -I$(call fixpath,$(NDB_TOP)/include/debugger) \
+ -I$(call fixpath,$(NDB_TOP)/include/mgmcommon) \
+ -I$(call fixpath,$(NDB_TOP)/include/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/include/util) \
+ -I$(call fixpath,$(NDB_TOP)/include/portlib) \
+ -I$(call fixpath,$(NDB_TOP)/include/logger)
+endif
+
+#
+# TYPE := ndbapiclient
+#
+ifneq ($(filter ndbapiclient, $(TYPE)),)
+CCFLAGS_LOC += \
+ -I$(call fixpath,$(NDB_TOP)/include/ndbapi)
+
+BIN_TARGET_LIBS += NDB_API
+endif
+
+#
+# TYPE := mgmapiclient
+#
+ifneq ($(filter mgmapiclient, $(TYPE)),)
+CCFLAGS_LOC += \
+ -I$(call fixpath,$(NDB_TOP)/include/mgmapi)
+
+BIN_TARGET_LIBS += MGM_API
+endif
+
+#
+# TYPE := ndbapitest
+#
+ifneq ($(filter ndbapitest, $(TYPE)),)
+CCFLAGS_LOC += \
+ -I$(call fixpath,$(NDB_TOP)/include/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/include/util) \
+ -I$(call fixpath,$(NDB_TOP)/include/portlib) \
+ -I$(call fixpath,$(NDB_TOP)/test/include) \
+ -I$(call fixpath,$(NDB_TOP)/include/mgmapi)
+
+BIN_TARGET_LIBS += NDBT
+LDFLAGS_LOC += -lNDB_API -lMGM_API -lm
+
+endif
+
+#
+# TYPE := signalsender
+#
+ifneq ($(filter signalsender, $(TYPE)),)
+CCFLAGS_LOC += \
+ -I$(call fixpath,$(NDB_TOP)/include/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/src/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/src/ndbapi/signal-sender) \
+ -I$(call fixpath,$(NDB_TOP)/include/util) \
+ -I$(call fixpath,$(NDB_TOP)/include/portlib) \
+ -I$(call fixpath,$(NDB_TOP)/include/transporter) \
+ -I$(call fixpath,$(NDB_TOP)/include/mgmcommon) \
+ -I$(call fixpath,$(NDB_TOP)/include/kernel)
+
+BIN_TARGET_LIBS += NDB_API
+BIN_TARGET_ARCHIVES += editline signal-sender
+
+endif
+
+
+#
+# TYPE := repserver
+#
+ifneq ($(filter repserver, $(TYPE)),)
+CCFLAGS_LOC += \
+ -I$(call fixpath,$(NDB_TOP)/include/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/src) \
+ -I$(call fixpath,$(NDB_TOP)/src/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/src/ndbapi/signal-sender) \
+ -I$(call fixpath,$(NDB_TOP)/include/util) \
+ -I$(call fixpath,$(NDB_TOP)/include/portlib) \
+ -I$(call fixpath,$(NDB_TOP)/include/transporter) \
+ -I$(call fixpath,$(NDB_TOP)/include/mgmcommon) \
+ -I$(call fixpath,$(NDB_TOP)/include/kernel)
+endif
+
+#
+# TYPE := odbcclient
+#
+
+ifneq ($(filter odbcclient, $(TYPE)),)
+TYPE += util
+LDFLAGS_LOC += -lm
+#ifneq ($(call check-odbc),)
+ifneq ($(NDB_ODBC),N)
+ifeq ($(NDB_OS), SOLARIS)
+CCFLAGS_LOC += -I/usr/local/include
+BIN_TARGET_LIBS_DIRS += /usr/local/lib
+BIN_TARGET_LIBS += odbc odbcinst NDBT
+endif
+ifeq ($(NDB_OS), LINUX)
+BIN_TARGET_LIBS += odbc odbcinst NDBT
+endif
+ifeq ($(NDB_OS), MACOSX)
+BIN_TARGET_LIBS += odbc odbcinst NDBT
+endif
+ifeq ($(NDB_OS), IBMAIX)
+BIN_TARGET_LIBS += odbc odbcinst NDBT
+endif
+ifeq ($(NDB_OS), TRU64X)
+BIN_TARGET_LIBS += odbc odbcinst NDBT
+endif
+else
+BIN_EXE = N
+endif
+endif
+
+#
+# TYPE := *
+#
+#
+# TYPE := util
+#
+ifneq ($(filter util, $(TYPE)),)
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include/util) \
+ -I$(call fixpath,$(NDB_TOP)/include/portlib) \
+ -I$(call fixpath,$(NDB_TOP)/include/logger)
+BIN_TARGET_LIBS += logger general portlib
+endif
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include)
+
+ifeq ($(NDB_SCI), Y)
+BIN_TARGET_LIBS += sisci
+BIN_TARGET_LIBS_DIRS += $(EXTERNAL_DEPENDS_TOP)/sci/lib
+
+CCFLAGS_LOC += -I$(call fixpath,$(EXTERNAL_DEPENDS_TOP)/sci/include)
+endif
+
+#
+# TYPE Handling
+###
+
+###
+#
+# First rule
+#
+first:
+ $(MAKE) libs
+ $(MAKE) bins
+
+ifeq ($(findstring all,$(replace-targets)),)
+all: first
+endif
+
+###
+#
+# Nice to have rules
+api: libs
+ $(MAKE) -C $(NDB_TOP)/src/ndbapi bins
+
+mgm: libs
+ $(MAKE) -C $(NDB_TOP)/src/mgmsrv bins
+
+ndb: libs
+ $(MAKE) -C $(NDB_TOP)/src/kernel/ndb-main bins
+
+apitest: first
+ $(MAKE) -C $(NDB_TOP)/test/ndbapi all
+
+#-lNDBT:
+# $(MAKE) -C $(NDB_TOP)/test/src all
+#
+#-lNDB_API: libs
+# $(MAKE) -C $(NDB_TOP)/src/ndbapi bins
+
+#
+# Libs/Bins
+#
+ifdef PREREQ_LOC
+_libs:: $(PREREQ_LOC)
+_bins:: $(PREREQ_LOC)
+endif
+
+L_DIRS := $(LIB_DIRS) $(DIRS)
+B_DIRS := $(BIN_DIRS) $(DIRS)
+A_DIRS := $(LIB_DIRS) $(BIN_DIRS) $(DIRS)
+
+_libs::
+
+_bins::
+
+libs: _libs $(patsubst %, _libs_%, $(L_DIRS))
+$(patsubst %, _libs_%, $(L_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _libs_%,%,$@) libs
+
+bins: _bins $(patsubst %, _bins_%, $(B_DIRS))
+$(patsubst %, _bins_%, $(B_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _bins_%,%,$@) bins
+
+###
+#
+# Links
+_links:
+ -$(NDB_TOP)/tools/make-links.sh $(NDB_TOP)/include `pwd`
+
+links: _links $(patsubst %, _links_%, $(A_DIRS))
+$(patsubst %, _links_%, $(A_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _links_%,%,$@) links
+
+
+####
+#
+# OSE build_spec (
+ifdef SOURCES
+BS := Y
+endif
+
+ifdef SOURCES_c
+BS := Y
+endif
+
+_build_spec: Makefile
+ifdef BS
+ @echo "TYPE = SWU" > build.spec
+ @echo "include $(NDB_TOP)/Ndb.mk" >> build.spec
+# @for i in $(CCFLAGS_LOC); do echo "INC += $$i" >> build.spec ; done
+ @for i in $(patsubst -I%, %, $(CCFLAGS_LOC)); do echo "INC += $$i" >> build.spec ; done
+ @echo "INC += /vobs/cello/cls/rtosi_if/include" >> build.spec
+ @echo "INC += /vobs/cello/cls/rtosi_if/include.@@@" >> build.spec
+ @echo "INC += /vobs/cello/cls/rtosi_if/include.<<<" >> build.spec
+endif
+
+build_spec: _build_spec $(patsubst %, _build_spec_%, $(A_DIRS))
+$(patsubst %, _build_spec_%, $(A_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _build_spec_%,%,$@) build_spec
+
+###
+#
+# Phony targets
+
+.PHONY: $(A_DIRS)
+
+###
+#
+# Dummy rule
+
+DUMMY:
+
+###
+#
+# Definitions of...
+
+PIC_DIR := $(NDB_BUILDDIR).pic
+A_TMP_DIR := $(NDB_BUILDDIR).a_tmp
+SO_TMP_DIR := $(NDB_BUILDDIR).so_tmp
+PIC_TMP_DIR := $(NDB_BUILDDIR).pic_tmp
+
+$(PIC_DIR):
+ mkdir -p $(PIC_DIR)
+
+SRC_C := $(filter %.C, $(SOURCES))
+SRC_CPP := $(filter %.cpp, $(SOURCES))
+SRC_CC := $(filter %.cc, $(SOURCES))
+SRC_c := $(filter %.c, $(SOURCES)) $(filter %.c, $(SOURCES.c))
+SRC_YPP := $(filter %.ypp, $(SOURCES))
+SRC_LPP := $(filter %.lpp, $(SOURCES))
+
+OBJECTS := $(SRC_C:%.C=%.$(OBJEXT)) \
+ $(SRC_CPP:%.cpp=%.$(OBJEXT)) \
+ $(SRC_CC:%.cc=%.$(OBJEXT)) \
+ $(SRC_c:%.c=%.$(OBJEXT)) \
+ $(SRC_YPP:%.ypp=%.tab.$(OBJEXT)) \
+ $(SRC_LPP:%.lpp=%.yy.$(OBJEXT)) \
+ $(OBJECTS_LOC)
+
+PIC_OBJS := $(OBJECTS:%=$(PIC_DIR)/%)
+
+LIB_DIR := $(NDB_TOP)/lib
+BIN_DIR := $(NDB_TOP)/bin
+
+###
+#
+# ARCHIVE_TARGET
+#
+ifdef ARCHIVE_TARGET
+
+ifndef NONPIC_ARCHIVE
+NONPIC_ARCHIVE := Y
+endif
+
+ifeq ($(NONPIC_ARCHIVE), Y)
+_libs:: $(LIB_DIR)/$(LIBPREFIX)$(ARCHIVE_TARGET).$(LIBEXT)
+$(LIB_DIR)/$(LIBPREFIX)$(ARCHIVE_TARGET).$(LIBEXT) : $(OBJECTS)
+ $(call ar_rcs,$@,$(OBJECTS))
+
+endif # NONPIC_ARCHIVE := Y
+
+ifeq ($(PIC_ARCHIVE), Y)
+_libs:: $(PIC_DIR) $(LIB_DIR)/$(LIBPREFIX)$(ARCHIVE_TARGET)_pic.$(LIBEXT)
+$(LIB_DIR)/$(LIBPREFIX)$(ARCHIVE_TARGET)_pic.$(LIBEXT) : $(PIC_OBJS)
+ cd $(PIC_DIR) && $(call ar_rcs,../$@,$(OBJECTS))
+
+PIC_DEP := Y
+
+endif # PIC_ARCHIVE := Y
+
+endif # ARCHIVE_TARGET
+
+###
+#
+# LIB_TARGET
+#
+ifdef LIB_TARGET
+
+ifeq ($(A_LIB), Y)
+
+A_LIB_ARCHIVES := $(LIB_TARGET_ARCHIVES:%=$(LIB_DIR)/$(LIBPREFIX)%.$(LIBEXT))
+
+_bins:: $(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET).$(LIBEXT)
+$(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET).$(LIBEXT) : $(A_LIB_ARCHIVES)
+ @rm -rf $(A_TMP_DIR) && mkdir $(A_TMP_DIR)
+ cd $(A_TMP_DIR) && for i in $^; do ar -x ../$$i; done && $(call ar_rcs,../$@,*.$(OBJEXT))
+ $(NDB_TOP)/home/bin/ndb_deploy $@
+endif # A_LIB := Y
+
+ifeq ($(SO_LIB), Y)
+ifneq ($(NDB_OS), WIN32)
+SO_LIB_ARCHIVES := $(LIB_TARGET_ARCHIVES:%=$(LIB_DIR)/$(LIBPREFIX)%_pic.$(LIBEXT))
+
+_bins:: $(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET).$(SHLIBEXT)
+$(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET).$(SHLIBEXT) : $(SO_LIB_ARCHIVES)
+ @rm -rf $(SO_TMP_DIR) && mkdir $(SO_TMP_DIR)
+ cd $(SO_TMP_DIR) && for i in $^; do ar -x ../$$i; done
+ifneq ($(NDB_OS), MACOSX)
+ $(SO) $@.new $(SO_TMP_DIR)/*.$(OBJEXT) -L$(LIB_DIR) $(LIB_TARGET_LIBS) $(LDFLAGS_LAST)
+ rm -f $@; mv $@.new $@
+else
+ $(SO) $@ $(SO_TMP_DIR)/*.$(OBJEXT) -L$(LIB_DIR) $(LIB_TARGET_LIBS) $(LDFLAGS_LAST)
+endif
+ifeq ($(NDB_VERSION), RELEASE)
+ifneq ($(NDB_OS), MACOSX)
+ strip $@
+endif
+endif
+ $(NDB_TOP)/home/bin/ndb_deploy $@
+else # WIN32
+SO_LIB_ARCHIVES := $(LIB_TARGET_ARCHIVES:%=$(LIB_DIR)/$(LIBPREFIX)%_pic.$(LIBEXT))
+
+_bins:: $(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET).$(SHLIBEXT)
+$(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET).$(SHLIBEXT) : $(SO_LIB_ARCHIVES)
+ @rm -rf $(SO_TMP_DIR) && mkdir $(SO_TMP_DIR)
+ cd $(SO_TMP_DIR) && for i in $^; do ar -x ../$$i; done
+ $(call link_so,$@.new,$(SO_TMP_DIR)/*.$(OBJEXT))
+ rm -f $@; mv $@.new $@
+#ifeq ($(NDB_VERSION), RELEASE)
+# strip $@
+#endif
+
+endif
+endif # SO_LIB := Y
+
+ifeq ($(PIC_LIB), Y)
+
+PIC_LIB_ARCHIVES := $(LIB_TARGET_ARCHIVES:%=$(LIB_DIR)/$(LIBPREFIX)%_pic.$(LIBEXT))
+
+_bins:: $(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET)_pic.$(LIBEXT)
+$(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET)_pic.$(LIBEXT) : $(PIC_LIB_ARCHIVES)
+ @rm -rf $(PIC_TMP_DIR) && mkdir $(PIC_TMP_DIR)
+ cd $(PIC_TMP_DIR) && for i in $^; do ar -x ../$$i; done && $(call ar_rcs,../$@,*.$(OBJEXT))
+
+endif # PIC_LIB := Y
+
+endif # LIB_TARGET
+
+###
+#
+# BIN_TARGET
+#
+ifeq ($(BIN_EXE), Y)
+ifneq ($(NDB_OS), WIN32)
+BIN_LIBS := $(BIN_TARGET_ARCHIVES:%=$(LIB_DIR)/$(LIBPREFIX)%.$(LIBEXT))
+BIN_LIBS += $(BIN_TARGET_LIBS:%=-l%)
+
+BIN_DEPS := $(OBJECTS) $(EXTRA_MAIN) $(BIN_LIBS)
+BIN_LIB_DIRS := $(BIN_TARGET_LIBS_DIRS:%=-L%)
+
+BIN_FLAGS := $(BIN_LIB_DIRS) $(BIN_DEPS)
+
+VPATH := $(LIB_DIR) $(BIN_TARGET_LIBS_DIRS)
+_bins:: $(BIN_DIR)/$(BIN_TARGET)
+$(BIN_DIR)/$(BIN_TARGET) : $(BIN_DEPS)
+ $(LINK.cc) $(LDFLAGS) $(LDLIBS) -L$(LIB_DIR) $(BIN_FLAGS) -o $@.new $(LDFLAGS_LAST)
+ rm -f $@; mv $@.new $@
+ifeq ($(NDB_VERSION), RELEASE)
+ifneq ($(NDB_OS), MACOSX)
+ strip $@
+endif
+endif
+ $(NDB_TOP)/home/bin/ndb_deploy $@
+else # WIN32
+BIN_LIBS := $(foreach lib,$(BIN_TARGET_ARCHIVES),$(call fixpath,$(LIB_DIR)/$(LIBPREFIX)$(lib).$(LIBEXT)))
+BIN_LIBS += $(BIN_TARGET_LIBS:%=$(LIBPREFIX)%.$(LIBEXT))
+
+BIN_DEPS := $(OBJECTS) $(BIN_TARGET_ARCHIVES:%=$(LIB_DIR)/$(LIBPREFIX)%.$(LIBEXT))
+BIN_LIB_DIRS := -libpath:$(call fixpath,$(LIB_DIR)) $(BIN_TARGET_LIBS_DIRS:%=-libpath:%)
+
+BIN_FLAGS := $(BIN_LIB_DIRS)
+
+VPATH := $(LIB_DIR) $(BIN_TARGET_LIBS_DIRS)
+_bins:: $(BIN_DIR)/$(BIN_TARGET).exe
+$(BIN_DIR)/$(BIN_TARGET).exe : $(BIN_DEPS)
+ $(LINK.cc) -out:$(call fixpath,$@.new) $(OBJECTS) $(BIN_FLAGS) $(BIN_LIBS)
+ rm -f $@; mv $@.new $@
+ifeq ($(NDB_VERSION), RELEASE)
+ strip $@
+endif
+
+endif
+endif
+
+###
+#
+# SOURCES.sh
+#
+ifdef SOURCES.sh
+
+BIN_SRC := $(SOURCES.sh:%=$(BIN_DIR)/%)
+
+_bins:: $(BIN_SRC)
+
+$(BIN_SRC) : $(SOURCES.sh)
+ rm -f $(^:%=$(BIN_DIR)/%)
+ cp $^ $(BIN_DIR)
+endif
+
+#
+# Compile rules PIC objects
+#
+ifeq ($(NDB_OS), WIN32)
+OUT := -Fo
+else
+OUT := -o
+endif
+
+$(PIC_DIR)/%.$(OBJEXT): %.C
+ $(C++) $(OUT)$@ -c $(CCFLAGS) $(CFLAGS_$<) $(PIC) $<
+
+$(PIC_DIR)/%.$(OBJEXT): %.cpp
+ $(C++) $(OUT)$@ -c $(CCFLAGS) $(CFLAGS_$<) $(PIC) $<
+
+$(PIC_DIR)/%.$(OBJEXT): %.cc
+ $(C++) $(OUT)$@ -c $(CCFLAGS) $(CFLAGS_$<) $(PIC) $<
+
+$(PIC_DIR)/%.$(OBJEXT): %.c
+ $(CC) $(OUT)$@ -c $(CFLAGS) $(CFLAGS_$<) $(PIC) $<
+
+#
+# Compile rules
+#
+%.$(OBJEXT) : %.cpp
+ $(C++) $(OUT)$@ -c $(CCFLAGS) $(CFLAGS_$<) $(NON_PIC) $<
+
+%.$(OBJEXT) : %.C
+ $(C++) $(OUT)$@ -c $(CCFLAGS) $(CFLAGS_$<) $(NON_PIC) $<
+
+%.$(OBJEXT) : %.cc
+ $(C++) $(OUT)$@ -c $(CCFLAGS) $(CFLAGS_$<) $(NON_PIC) $<
+
+%.$(OBJEXT) : %.c
+ $(CC) $(OUT)$@ -c $(CFLAGS) $(CFLAGS_$<) $(NON_PIC) $<
+
+%.s : %.C
+ $(C++) -S $(CCFLAGS) $(CFLAGS_$<) $(NON_PIC) $<
+
+%.s : %.cpp
+ $(C++) -S $(CCFLAGS) $(CFLAGS_$<) $(NON_PIC) $<
+
+%.s : %.cc
+ $(C++) -S $(CCFLAGS) $(CFLAGS_$<) $(NON_PIC) $<
+
+%.s : %.c
+ $(CC) -S $(CCFLAGS) $(CFLAGS_$<) $(NON_PIC) $<
+
+BISON = bison
+BISONHACK = :
+%.tab.cpp %.tab.hpp : %.ypp
+ $(BISON) $<
+ $(BISONHACK) $*.tab.cpp $*.tab.hpp
+
+FLEX = flex
+FLEXHACK = :
+%.yy.cpp : %.lpp
+ $(FLEX) -o$@ $<
+ $(FLEXHACK) $@
+
+###
+#
+# Defines regarding dependencies
+
+DEPMK := $(NDB_BUILDDIR).depend.mk
+
+DEPDIR := $(NDB_BUILDDIR).depend
+
+DEPENDENCIES := $(SRC_C:%.C=$(DEPDIR)/%.d) \
+ $(SRC_CC:%.cc=$(DEPDIR)/%.d) \
+ $(SRC_CPP:%.cpp=$(DEPDIR)/%.d) \
+ $(SRC_c:%.c=$(DEPDIR)/%.d) \
+ $(SRC_YPP:%.ypp=$(DEPDIR)/%.tab.d) \
+ $(SRC_LPP:%.lpp=$(DEPDIR)/%.yy.d)
+
+###
+#
+# Dependency rule
+
+_depend: $(DEPMK)
+
+depend: _depend $(patsubst %, _depend_%, $(A_DIRS))
+
+$(patsubst %, _depend_%, $(A_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _depend_%,%,$@) depend
+
+###
+#
+# Clean dependencies
+
+_clean_dep:
+ -rm -rf $(DEPMK) $(DEPDIR)/*
+
+clean_dep: _clean_dep $(patsubst %, _clean_dep_%, $(A_DIRS))
+
+$(patsubst %, _clean_dep_%, $(A_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _clean_dep_%,%,$@) clean_dep
+
+###
+#
+# Generate dependencies
+
+$(DEPDIR):
+ -@mkdir -p $(DEPDIR)
+
+$(DEPDIR)/%.d: %.C
+ @echo Generating depend for $<
+ @$(MAKEDEPEND) $(CCFLAGS) $(CFLAGS_$<) $< >$@
+
+$(DEPDIR)/%.d: %.c
+ @echo Generating depend for $<
+ @$(MAKEDEPEND) $(CCFLAGS) $(CFLAGS_$<) $< >$@
+
+$(DEPDIR)/%.d: %.cpp
+ @echo Generating depend for $<
+ @$(MAKEDEPEND) $(CCFLAGS) $(CFLAGS_$<) $< >$@
+
+$(DEPDIR)/%.d: %.cc
+ @echo Generating depend for $<
+ @$(MAKEDEPEND) $(CCFLAGS) $(CFLAGS_$<) $< >$@
+
+ifeq ($(NDB_OS), WIN32)
+ifndef PIC_DEP
+DEP_PTN := -e 's/\(.*\)\.o[ :]*/\1.$(OBJEXT) $(DEPDIR)\/\1.d : /g'
+else
+DEP_PTN := -e 's/\(.*\)\.o[ :]*/\1.$(OBJEXT) $(PIC_DIR)\/\1.$(OBJEXT) $(DEPDIR)\/\1.d : /g'
+endif
+else
+ifndef PIC_DEP
+DEP_PTN := -e 's!\(.*\)\.$(OBJEXT)[ :]*!\1.$(OBJEXT) $(DEPDIR)\/\1.d : !g'
+else
+DEP_PTN := -e 's!\(.*\)\.$(OBJEXT)[ :]*!\1.$(OBJEXT) $(PIC_DIR)\/\1.$(OBJEXT) $(DEPDIR)\/\1.d : !g'
+endif
+endif
+#DEP_PTN += -e 's!/usr/include/[-+a-zA-Z0-9_/.]*!!g'
+#DEP_PTN += -e 's!/usr/local/lib/gcc-lib/[-+a-zA-Z0-9_/.]*!!g'
+
+$(DEPMK): $(DEPDIR) $(SRC_YPP:%.ypp=%.tab.hpp) $(SRC_LPP:%.lpp=%.yy.cpp) $(DEPENDENCIES) $(wildcard $(NDB_TOP)/.update.d)
+ @echo "updating .depend.mk"
+ @sed $(DEP_PTN) /dev/null $(DEPENDENCIES) >$(DEPMK)
+
+###
+#
+# clean
+#
+_clean:
+ -rm -rf SunWS_cache $(PIC_DIR)/SunWS_cache
+ifeq ($(NONPIC_ARCHIVE), Y)
+ -rm -f $(OBJECTS) $(LIB_DIR)/$(LIBPREFIX)$(ARCHIVE_TARGET).$(LIBEXT)
+endif
+ifeq ($(PIC_ARCHIVE), Y)
+ -rm -f $(PIC_OBJS) $(LIB_DIR)/$(LIBPREFIX)$(ARCHIVE_TARGET)_pic.$(LIBEXT)
+endif
+ifdef BIN_TARGET
+ -rm -f $(OBJECTS)
+endif
+ifdef LIB_TARGET
+ifeq ($(A_LIB), Y)
+ -rm -f $(A_TMP_DIR)/*
+endif
+ifeq ($(SO_LIB), Y)
+ -rm -f $(SO_TMP_DIR)/*
+endif
+ifeq ($(PIC_LIB), Y)
+ -rm -f $(PIC_TMP_DIR)/*
+endif
+endif
+ifneq ($(SRC_YPP),)
+ -rm -f $(SRC_YPP:%.ypp=%.tab.[hc]pp) $(SRC_YPP:%.ypp=%.output)
+endif
+ifneq ($(SRC_LPP),)
+ -rm -f $(SRC_LPP:%.lpp=%.yy.*)
+endif
+ifdef CLEAN_LOC
+ -rm -f $(CLEAN_LOC)
+endif
+
+###
+#
+# clean all
+#
+clobber: cleanall
+_cleanall: _clean clean_links
+ -rm -f osemain.con osemain.c
+ifdef LIB_TARGET
+ifeq ($(A_LIB), Y)
+ -rm -f $(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET).$(LIBEXT)
+endif
+ifeq ($(SO_LIB), Y)
+ -rm -f $(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET).$(SHLIBEXT)
+endif
+ifeq ($(PIC_LIB), Y)
+ -rm -f $(LIB_DIR)/$(LIBPREFIX)$(LIB_TARGET)_pic.$(LIBEXT)
+endif
+endif
+ifdef BIN_TARGET
+ -rm -f $(BIN_DIR)/$(BIN_TARGET)
+endif
+
+clean_links:
+
+###
+#
+# Dist clean
+#
+_distclean: _tidy
+ rm -rf $(DEPDIR) $(PIC_DIR) $(PIC_TMP_DIR) $(SO_TMP_DIR) $(A_TMP_DIR) Sources build.spec
+
+###
+#
+# tidy
+#
+_tidy: _cleanall _clean_dep
+ -rm -f *~ *.$(OBJEXT) *.$(LIBEXT) *.${SHLIBEXT}
+
+#
+# clean cleanall tidy - recursion
+#
+ifeq ($(findstring clean,$(replace-targets)),)
+clean: _clean $(patsubst %, _clean_%, $(A_DIRS))
+endif
+
+$(patsubst %, _clean_%, $(A_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _clean_%,%,$@) clean
+
+cleanall: _cleanall $(patsubst %, _cleanall_%, $(A_DIRS))
+
+$(patsubst %, _cleanall_%, $(A_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _cleanall_%,%,$@) cleanall
+
+tidy: _tidy $(patsubst %, _tidy_%, $(A_DIRS))
+
+$(patsubst %, _tidy_%, $(A_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _tidy_%,%,$@) tidy
+
+distclean: _distclean $(patsubst %, _distclean_%, $(A_DIRS))
+
+$(patsubst %, _distclean_%, $(A_DIRS)) : DUMMY
+ $(MAKE) -C $(patsubst _distclean_%,%,$@) distclean
+
+###
+#
+# Guess configuration
+
+$(NDB_TOP)/config/config.mk: $(NDB_TOP)/config/GuessConfig.sh
+ $(NDB_TOP)/config/GuessConfig.sh -D
+
+$(NDB_TOP)/config/Defs....mk: $(NDB_TOP)/config/config.mk
+$(NDB_TOP)/config/Defs..mk: $(NDB_TOP)/config/config.mk
+
+###
+# Soft ose envirment stuff
+#
+osemain.con: $(NDB_TOP)/src/env/softose/osemain_con.org
+ cp $< $@
+ echo "PRI_PROC(init_$(BIN_TARGET), init_$(BIN_TARGET), 65535, 3, ndb, 0, NULL)" >> $@
+
+osemain.c: $(OSE_LOC)/sfk-solaris2/krn-solaris2/src/osemain.c
+ ln -s $< $@
+
+osemain.o : osemain.con
+
+$(DEPDIR)/osemain.d : osemain.con
+
+###
+#
+# These target dont want dependencies
+
+NO_DEP=clean clobber cleanall tidy clean_dep $(DEPDIR) build_spec \
+ $(NDB_TOP)/config/config.mk distclean osemain.con osemain.c
+
+ifeq ($(filter $(NO_DEP), $(MAKECMDGOALS)),)
+ifneq ($(strip $(DEPENDENCIES)),)
+ include $(DEPMK)
+endif
+endif
+
+###
+#
+# Auxiliary targets
+
+sources: Sources
+
+Sources: Makefile
+ @rm -f $@
+ @for f in Makefile $(A_DIRS) $(SOURCES) $(SOURCES.c); do echo $$f; done >$@
+
+###
+#
+# TAG generation for emacs and vi folks
+#
+# In emacs "Esc- ." or "M- ." to find a symbol location
+# In vi use the :\tag command
+# by convention:
+# TAGS is used with emacs
+# tags is used with vi
+#
+# Hopefully the make is being done from $(NDB_TOP)/src
+# and your TAGS/tags file then is in the same directory.
+
+TAGS: DUMMY
+ rm -f TAGS
+ find $(NDB_TOP) -name "*.[ch]" | xargs $(ETAGS) --append
+ find $(NDB_TOP) -name "*.[ch]pp" | xargs $(ETAGS) --append
+
+tags: DUMMY
+ rm -f tags
+ find $(NDB_TOP) -name "*.[ch]" | xargs $(CTAGS) --append
+ find $(NDB_TOP) -name "*.[ch]pp" | xargs $(CTAGS) --append
+
+install:
+
+
+ebrowse: DUMMY
+ cd $(NDB_TOP); rm -f EBROWSE
+ cd $(NDB_TOP); find . -name "*.hpp" -or -name "*.cpp" -or -name "*.h" -or -name "*.c" > tmpfile~
+ cd $(NDB_TOP); ebrowse --file tmpfile~
+ cd $(NDB_TOP); rm -f tmpfile~
diff --git a/ndb/Makefile b/ndb/Makefile
new file mode 100644
index 00000000000..586a430bb17
--- /dev/null
+++ b/ndb/Makefile
@@ -0,0 +1,62 @@
+include .defs.mk
+
+DIRS := src test tools examples
+
+# hack before full autoconf
+replace-targets := all clean
+NDB_RELEASE := $(shell ../scripts/mysql_config --version)
+
+include $(NDB_TOP)/Epilogue.mk
+
+_libs_test : _bins_src
+_libs_tools : _libs_test
+_libs_examples : _bins_src
+_bins_src : _libs_src
+_bins_tools : _bins_src
+
+# always release compile except for ndbapi static lib
+all:
+ $(MAKE) -C src/ndbapi libs
+ $(MAKE) libs NDB_VERSION=RELEASE
+ $(MAKE) bins NDB_VERSION=RELEASE
+ifeq ($(NDB_OS),LINUX)
+ NDB_RELEASE=$(NDB_RELEASE) $(MAKE) -j1 -C docs all </dev/null || :
+endif
+
+# old distclean matches clean better
+clean: distclean
+ $(MAKE) -C docs clean
+
+nuke-deps:
+ find . -name '.depend*' | xargs rm -rf
+
+vim-tags:
+ bk sfiles -g | ctags --c-types=+p --extra=+fq -L -
+
+cvs-update:
+ifeq ($(NDB_VERSION),main)
+ -cvs update -d
+else
+ifeq ($(NDB_TAG),HEAD)
+ -cvs -q update
+ -cd include && cvs -q update -d
+ -cd src && cvs -q update -d
+ -cd test && cvs -q update -d
+ -cd tools && cvs -q update -d
+else
+ -cvs -q update -r $(NDB_TAG)
+ -cd include && cvs -q update -d -r $(NDB_TAG)
+ -cd src && cvs -q update -d -r $(NDB_TAG)
+ -cd test && cvs -q update -d -r $(NDB_TAG)
+ -cd tools && cvs -q update -d -r $(NDB_TAG)
+endif
+endif
+ make nuke-deps
+ make vim-tags
+ make TAGS
+
+bk-update:
+ bk pull
+ make nuke-deps
+ make vim-tags
+ make TAGS
diff --git a/ndb/README b/ndb/README
new file mode 100644
index 00000000000..3b21fca1d48
--- /dev/null
+++ b/ndb/README
@@ -0,0 +1,7 @@
+INSTALLATION
+To compile a pentium version of MySQL Cluster from this BK clone do:
+
+shell> cd /home/bk/mysql-4.1-ndb
+shell> BUILD/compile-pentium-debug -c --prefix=/usr/local/mysql-4.1-ndb
+shell> make
+
diff --git a/ndb/bin/.empty b/ndb/bin/.empty
new file mode 100644
index 00000000000..e69de29bb2d
--- /dev/null
+++ b/ndb/bin/.empty
diff --git a/ndb/bin/check-regression.sh b/ndb/bin/check-regression.sh
new file mode 100755
index 00000000000..93a31ccb39c
--- /dev/null
+++ b/ndb/bin/check-regression.sh
@@ -0,0 +1,180 @@
+#!/bin/sh
+# NAME
+# check-regression.sh
+#
+# SYNOPSIS
+# check-regression.sh
+#
+# DESCRIPTION
+#
+# This scrip must be run before any major cvs checkins are done.
+# It will perform a number of regression tests to check that
+# nothing is broken.
+#
+# OPTIONS
+#
+# EXAMPLES
+#
+#
+# ENVIRONMENT
+# NDB_PROJ_HOME Home dir for ndb
+# verbose verbose printouts
+#
+# FILES
+# $NDB_PROJ_HOME/lib/funcs.sh general shell script functions
+#
+#
+# SEE ALSO
+#
+# DIAGNOSTICTS
+#
+#
+# VERSION
+# 1.0
+#
+# AUTHOR
+#
+#
+
+. $NDB_PROJ_HOME/lib/funcs.sh # Load some good stuff
+
+synopsis="check-regression.sh"
+progname=`basename $0`
+
+numOfTestsOK=0
+numOfTestsFailed=0
+
+LOG=check-regression.`date '+%Y-%m-%d'`
+
+executeTest()
+{
+ eval "$@" | tee -a $LOG
+
+ if [ $? -eq 0 ]
+ then
+ echo "SUCCESS: $@"
+ numOfTestsOK=`expr $numOfTestsOK + 1`
+ else
+ echo "FAILED: $@"
+ numOfTestsFailed=`expr $numOfTestsFailed + 1`
+ fi
+}
+
+#
+# INFO
+#
+trace "Starting: `date`"
+trace "NDB_PROJ_HOME = $NDB_PROJ_HOME"
+trace "NDB_TOP = $NDB_TOP"
+
+#
+# THE TESTS TO EXECUTE
+#
+
+# Testsuite: testDataBuffers
+# Number of tests: 1
+executeTest 'drop_tab ' TB00 TB01 TB02 TB03 TB04 TB05 TB06 TB07 TB08 TB09 TB10 TB11 TB12 TB13 TB14 TB15
+executeTest 'testDataBuffers'
+executeTest 'drop_tab ' TB00 TB01 TB02 TB03 TB04 TB05 TB06 TB07 TB08 TB09 TB10 TB11 TB12 TB13 TB14 TB15
+
+TABLES="T9 T13"
+
+# Testsuite: testBasic
+# Number of tests: 16
+executeTest 'testBasic -n PkInsert' $TABLES
+executeTest 'testBasic -n PkRead' $TABLES
+executeTest 'testBasic -n PkUpdate' $TABLES
+executeTest 'testBasic -n PkDelete' $TABLES
+#executeTest 'testBasic -n UpdateAndRead'
+#executeTest 'testBasic -n PkReadAndLocker'
+#executeTest 'testBasic -n PkReadAndLocker2'
+#executeTest 'testBasic -n PkReadUpdateAndLocker'
+#executeTest 'testBasic -n ReadWithLocksAndInserts'
+#executeTest 'testBasic -n ReadConsistency'
+#executeTest 'testBasic -n PkInsertTwice'
+#executeTest 'testBasic -n Fill'
+#executeTest 'testBasic -n FillTwice'
+#executeTest 'testBasic -n NoCommitSleep'
+#executeTest 'testBasic -n NoCommit626'
+#executeTest 'testBasic -n NoCommitAndClose'
+
+# Testsuite: testBasicAsynch
+# Number of tests: 4
+executeTest 'testBasicAsynch -n PkInsertAsynch' $TABLES
+executeTest 'testBasicAsynch -n PkReadAsynch' $TABLES
+executeTest 'testBasicAsynch -n PkUpdateAsynch' $TABLES
+executeTest 'testBasicAsynch -n PkDeleteAsynch' $TABLES
+
+# Testsuite: testDict
+# Number of tests: 6
+#executeTest 'testDict -n CreateAndDrop'
+#executeTest 'testDict -n CreateAndDropWithData'
+#executeTest 'testDict -n CreateAndDropDuring'
+#executeTest 'testDict -n CreateInvalidTables'
+#executeTest 'testDict -n CreateTableWhenDbIsFull'
+#executeTest 'testDict -n CreateMaxTables'
+
+# Testsuite: testScan
+# Number of tests: 34
+#executeTest 'testScan -n ScanRead'
+#executeTest 'testScan -n ScanRead16'
+executeTest 'testScan -n ScanRead240' $TABLES
+executeTest 'testScan -n ScanUpdate' $TABLES
+executeTest 'testScan -n ScanUpdate2' $TABLES
+executeTest 'testScan -n ScanDelete' $TABLES
+executeTest 'testScan -n ScanDelete2' $TABLES
+#executeTest 'testScan -n ScanUpdateAndScanRead'
+#executeTest 'testScan -n ScanReadAndLocker'
+#executeTest 'testScan -n ScanReadAndPkRead'
+#executeTest 'testScan -n ScanRead488'
+#executeTest 'testScan -n ScanWithLocksAndInserts'
+#executeTest 'testScan -n ScanReadAbort'
+#executeTest 'testScan -n ScanReadAbort15'
+#executeTest 'testScan -n ScanReadAbort16'
+#executeTest 'testScan -n ScanUpdateAbort16'
+#executeTest 'testScan -n ScanReadAbort240'
+#executeTest 'testScan -n ScanReadRestart'
+#executeTest 'testScan -n ScanReadRestart16'
+#executeTest 'testScan -n ScanReadRestart32'
+#executeTest 'testScan -n ScanUpdateRestart'
+#executeTest 'testScan -n ScanUpdateRestart16'
+#executeTest 'testScan -n CheckGetValue'
+#executeTest 'testScan -n CloseWithoutStop'
+#executeTest 'testScan -n NextScanWhenNoMore'
+#executeTest 'testScan -n ExecuteScanWithoutOpenScan'
+#executeTest 'testScan -n OnlyOpenScanOnce'
+#executeTest 'testScan -n OnlyOneOpInScanTrans'
+#executeTest 'testScan -n OnlyOneOpBeforeOpenScan'
+#executeTest 'testScan -n OnlyOneScanPerTrans'
+#executeTest 'testScan -n NoCloseTransaction'
+#executeTest 'testScan -n CheckInactivityTimeOut'
+#executeTest 'testScan -n CheckInactivityBeforeClose'
+#executeTest 'testScan -n CheckAfterTerror'
+
+# Testsuite: testScanInterpreter
+# Number of tests: 1
+#executeTest 'testScanInterpreter -n ScanLessThan'
+
+TABLES="T6 T13"
+
+# Testsuite: testSystemRestart
+# Number of tests: 4
+executeTest 'testSystemRestart -l 1 -n SR1' $TABLES
+executeTest 'testSystemRestart -l 1 -n SR2' $TABLES
+#executeTest 'testSystemRestart -n SR_UNDO'
+#executeTest 'testSystemRestart -n SR_FULLDB'
+
+# TESTS FINISHED
+trace "Finished: `date`"
+
+#
+# TEST SUMMARY
+#
+if [ $numOfTestsFailed -eq 0 ]
+then
+ echo "-- REGRESSION TEST SUCCESSFUL --"
+else
+ echo "-- REGRESSION TEST FAILED!! --"
+fi
+echo "Number of successful tests: $numOfTestsOK"
+echo "Number of failed tests : $numOfTestsFailed"
diff --git a/ndb/bin/makeTestPrograms_html.sh b/ndb/bin/makeTestPrograms_html.sh
new file mode 100755
index 00000000000..ac31c8a6267
--- /dev/null
+++ b/ndb/bin/makeTestPrograms_html.sh
@@ -0,0 +1,22 @@
+#!/bin/sh
+rm $1
+touch $1
+echo "<table border="1" width=640>" >> $1
+echo "<tr>" >> $1
+echo "<td><b>Name</b></td><td>&nbsp;</td><td width="70%"><b>Description</b></td>" >> $1
+echo "</tr>" >> $1
+testBasic --print_html >> $1
+testBackup --print_html >> $1
+testBasicAsynch --print_html >> $1
+testDict --print_html >> $1
+testBank --print_html >> $1
+testIndex --print_html >> $1
+testNdbApi --print_html >> $1
+testNodeRestart --print_html >> $1
+testOperations --print_html >> $1
+testRestartGci --print_html >> $1
+testScan --print_html >> $1
+testScanInterpreter --print_html >> $1
+testSystemRestart --print_html >> $1
+echo "</table>" >> $1
+
diff --git a/ndb/bin/mysqlcluster b/ndb/bin/mysqlcluster
new file mode 100755
index 00000000000..81fc7308942
--- /dev/null
+++ b/ndb/bin/mysqlcluster
@@ -0,0 +1,11 @@
+#!/bin/sh
+if [ -z "$MYSQLCLUSTER_TOP" -o ! -d "$MYSQLCLUSTER_TOP" ]; then
+ echo "MYSQLCLUSTER_TOP not set or directory does not exist"
+ exit 1
+fi
+if [ -z "$MYSQLCLUSTER_TOP" -o ! -d "$MYSQLCLUSTER_TOP/ndb" ]; then
+ echo "$MYSQLCLUSTER_TOP/ndb directory does not exist"
+ exit 1
+fi
+
+mysql --socket=$MYSQLCLUSTER_TOP/data/mysqlcluster.sock $*
diff --git a/ndb/bin/mysqlcluster_install_db b/ndb/bin/mysqlcluster_install_db
new file mode 100755
index 00000000000..6fe95ff105d
--- /dev/null
+++ b/ndb/bin/mysqlcluster_install_db
@@ -0,0 +1,119 @@
+#!/bin/sh
+
+NDB_HOME=
+export NDB_CONNECTSTRING
+if [ -z "$MYSQLCLUSTER_TOP" ]; then
+ echo "MYSQLCLUSTER_TOP not set"
+ exit 1
+fi
+if [ -d "$MYSQLCLUSTER_TOP" ]; then :; else
+ echo "$MYSQLCLUSTER_TOP directory does not exist"
+ exit 1
+fi
+if [ -d "$MYSQLCLUSTER_TOP/ndb" ]; then :; else
+ echo "$MYSQLCLUSTER_TOP/ndb directory does not exist"
+ exit 1
+fi
+
+start_default_ndbcluster() {
+
+# configurable parameters, make sure to change in mysqlcluterd as well
+MYSQLCLUSTER_FILESYSTEM=$MYSQLCLUSTER_TOP/data/mysqlclusterfs
+MYSQLCLUSTER_PORT_BASE="22" # using ports MYSQLCLUSTER_PORT_BASE{"00","01", etc}
+# end configurable parameters
+
+# do some checks
+
+NDB_CONNECTSTRING=
+
+[ -d "$MYSQLCLUSTER_FILESYSTEM" ] || mkdir "$MYSQLCLUSTER_FILESYSTEM"
+if [ -d "$MYSQLCLUSTER_FILESYSTEM" ]; then :; else
+ echo "$MYSQLCLUSTER_FILESYSTEM filesystem directory does not exist"
+ exit 1
+fi
+
+
+# set som help variables
+
+NDB_HOST="localhost"
+NDB_PORT=$MYSQLCLUSTER_PORT_BASE"00"
+NDB_CONNECTSTRING_BASE="host=$NDB_HOST:$NDB_PORT;nodeid="
+
+
+# Edit file system path and ports in config file
+
+cd $MYSQLCLUSTER_FILESYSTEM
+sed \
+ -e s,"WRITE_PATH_TO_FILESYSTEM_2_HERE",$MYSQLCLUSTER_FILESYSTEM,g \
+ -e s,"CHOOSE_PORT_BASE",$MYSQLCLUSTER_PORT_BASE,g \
+ < $MYSQLCLUSTER_TOP/ndb/demos/config-templates/config_template-install.ini \
+ > config.ini
+
+
+# Start management server as deamon
+
+NDB_ID="1"
+NDB_CONNECTSTRING=$NDB_CONNECTSTRING_BASE$NDB_ID
+#xterm -e mgmtsrvr -c $MYSQLCLUSTER_FILESYSTEM/config.ini &
+if mgmtsrvr -d -c $MYSQLCLUSTER_FILESYSTEM/config.ini ; then :; else
+ echo "Unable to start mgmtsrvr"
+ exit 1
+fi
+
+
+# Start database node
+
+cd $MYSQLCLUSTER_FILESYSTEM # the output from the database node gets where it starts
+NDB_ID="2"
+NDB_CONNECTSTRING=$NDB_CONNECTSTRING_BASE$NDB_ID
+#xterm -T "NDB Cluster DB Node" -geometry 80x10 -xrm *.hold:true -e ndb -i &
+ndb -d -i &
+
+# Start xterm for application programs
+
+NDB_ID="3"
+NDB_CONNECTSTRING=$NDB_CONNECTSTRING_BASE$NDB_ID
+#xterm -T "NDB Cluster API Node" -geometry 80x10 &
+echo set before running ndbApi programs > export NDB_CONNECTSTRING=$NDB_CONNECTSTRING
+
+# Start management client
+
+#xterm -T "NDB Management Client" -geometry 80x10 -xrm *.hold:true -e mgmtclient $NDB_HOST $NDB_PORT &
+echo "NDB Management Client starts with: mgmtclient $NDB_HOST $NDB_PORT"
+
+# test if Ndb Cluster starts properly
+
+NDB_ID="11"
+NDB_CONNECTSTRING=$NDB_CONNECTSTRING_BASE$NDB_ID
+if list_tables | grep "NDBT_ProgramExit: 0 - OK"; then :; else
+ echo "Ndbcluster startup failed"
+ exit 1
+fi
+}
+
+start_mysql_install_db() {
+ # run install of regular MySQL Server
+
+ cd $MYSQLCLUSTER_TOP
+ scripts/mysql_install_db --basedir=$MYSQLCLUSTER_TOP --datadir=$MYSQLCLUSTER_TOP/data --socket=$MYSQLCLUSTER_TOP/data/mysqlcluster.sock $*
+}
+
+if test "$1" = "ndb_started"
+then
+ shift
+ mgmt_host=$1
+ shift
+ mgmt_port=$1
+ shift
+ if [ -z "$mgmt_host" -o -z "$mgmt_port" ]; then
+ echo "syntax: ndb_started hostname port"
+ exit 1
+ fi
+ NDB_CONNECTSTRING="host=$mgmt_host:$mgmt_port;nodeid=11"
+ echo using NDB_CONNECTSTRING=$NDB_CONNECTSTRING
+ start_mysql_install_db $*
+else
+ start_default_ndbcluster
+ start_mysql_install_db
+fi
+
diff --git a/ndb/bin/mysqlclusterd b/ndb/bin/mysqlclusterd
new file mode 100755
index 00000000000..3b4deb3ed48
--- /dev/null
+++ b/ndb/bin/mysqlclusterd
@@ -0,0 +1,34 @@
+#!/bin/sh
+
+# configurable parameters
+MYSQLCLUSTER_PORT_BASE="22"
+# end configurable parameters
+
+if [ -z "$MYSQLCLUSTER_TOP" -o ! -d "$MYSQLCLUSTER_TOP" ]; then
+ echo "MYSQLCLUSTER_TOP not set or directory does not exist"
+ exit 1
+fi
+if [ -z "$MYSQLCLUSTER_TOP" -o ! -d "$MYSQLCLUSTER_TOP/ndb" ]; then
+ echo "$MYSQLCLUSTER_TOP/ndb directory does not exist"
+ exit 1
+fi
+
+if test "$1" = "ndb_started"
+then
+ shift
+ mgmt_host=$1
+ shift
+ mgmt_port=$1
+ shift
+ if [ -z "$mgmt_host" -o -z "$mgmt_port" ]; then
+ echo "syntax: ndb_started hostname port"
+ exit 1
+ fi
+ NDB_CONNECTSTRING="host=$mgmt_host:$mgmt_port;nodeid=11"
+ echo using NDB_CONNECTSTRING=$NDB_CONNECTSTRING
+else
+ NDB_CONNECTSTRING="host=localhost:"$MYSQLCLUSTER_PORT_BASE"00;nodeid=11"
+fi
+export NDB_CONNECTSTRING
+
+mysqld --default-table-type=ndbcluster --basedir=$MYSQLCLUSTER_TOP --datadir=$MYSQLCLUSTER_TOP/data --socket=$MYSQLCLUSTER_TOP/data/mysqlcluster.sock $*
diff --git a/ndb/bin/regression.sh b/ndb/bin/regression.sh
new file mode 100644
index 00000000000..5e3491af208
--- /dev/null
+++ b/ndb/bin/regression.sh
@@ -0,0 +1,644 @@
+#!/bin/sh
+# NAME
+# regression.sh
+#
+# SYNOPSIS
+# regression.sh
+#
+# DESCRIPTION
+#
+# This script runs a number of regression tests to verify that nothing
+# is broken. Currently it executes the same tests as in the autotest
+# regression suite.
+#
+# OPTIONS
+#
+# EXAMPLES
+#
+#
+# ENVIRONMENT
+# verbose verbose printouts
+#
+# FILES
+#
+#
+# SEE ALSO
+#
+# DIAGNOSTICTS
+#
+#
+# VERSION
+# 1.0
+#
+# AUTHOR
+#
+#
+
+
+# die prints the supplied message to stderr,
+# prefixed with the program name, and exits
+# with the exit code given by "-e num" or
+# 1, if no -e option is present.
+#
+die ()
+{
+ die_code__=1
+ [ "X$1" = X-e ] && { die_code__=$2; shift 2; }
+ [ "X$1" = X-- ] && shift
+ errmsg "$@"
+ exit $die_code__
+}
+
+
+# msg prints the supplied message to stderr,
+# prefixed with the program name.
+#
+errmsg ()
+{
+ echo "${progname:-<no program name set>}:" "$@" >&2
+}
+
+# rawdie prints the supplied message to stderr.
+# It then exits with the exit code given with "-e num"
+# or 1, if no -e option is present.
+#
+rawdie ()
+{
+ rawdie_code__=1
+ [ "X$1" = X-e ] && { rawdie_code__=$2; shift 2; }
+ [ "X$1" = X-- ] && shift
+ rawerrmsg "$@"
+ exit $rawdie_code__
+}
+
+# Syndie prints the supplied message (if present) to stderr,
+# prefixed with the program name, on the first line.
+# On the second line, it prints $synopsis.
+# It then exits with the exit code given with "-e num"
+# or 1, if no -e option is present.
+#
+syndie ()
+{
+ syndie_code__=1
+ [ "X$1" = X-e ] && { syndie_code__=$2; shift 2; }
+ [ "X$1" = X-- ] && shift
+ [ -n "$*" ] && msg "$*"
+ rawdie -e $syndie_code__ "Synopsis: $synopsis"
+}
+
+
+
+
+# msg prints the supplied message to stdout,
+# prefixed with the program name.
+#
+msg ()
+{
+ echo "${progname:-<no program name set>}:" "$@"
+}
+
+rawmsg () { echo "$*"; } # print the supplied message to stdout
+rawerrmsg () { echo "$*" >&2; } # print the supplied message to stderr
+
+# trace prints the supplied message to stdout if verbose is non-null
+#
+trace ()
+{
+ [ -n "$verbose" ] && msg "$@"
+}
+
+
+# errtrace prints the supplied message to stderr if verbose is non-null
+#
+errtrace ()
+{
+ [ -n "$verbose" ] && msg "$@" >&2
+}
+
+
+synopsis="regression.sh"
+progname=`basename $0`
+
+numOfTestsOK=0
+numOfTestsFailed=0
+
+LOG=regression-$1.`date '+%Y-%m-%d'`
+
+executeTest()
+{
+ eval "$@" | tee -a $LOG
+
+ if [ $? -eq 0 ]
+ then
+ echo "SUCCESS: $@"
+ numOfTestsOK=`expr $numOfTestsOK + 1`
+ else
+ echo "FAILED: $@"
+ numOfTestsFailed=`expr $numOfTestsFailed + 1`
+ fi
+}
+
+#
+# INFO
+#
+trace "Starting: `date`"
+trace "NDB_TOP = $NDB_TOP"
+
+#
+# THE TESTS TO EXECUTE
+#
+
+# BASIC FUNCTIONALITY
+if [ $1 = "basic" ]
+then
+executeTest 'testBasic -n PkRead'
+executeTest 'drop_all_tabs'
+
+executeTest 'testBasic -n PkUpdate'
+executeTest 'drop_all_tabs'
+
+executeTest 'testBasic -n PkDelete'
+executeTest 'drop_all_tabs'
+
+executeTest 'testBasic -n PkInsert'
+executeTest 'drop_all_tabs'
+
+executeTest 'testBasic -n UpdateAndRead'
+executeTest 'drop_all_tabs'
+
+executeTest 'testBasic -n PkReadAndLocker' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n PkReadAndLocker2' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n PkReadUpdateAndLocker' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n ReadWithLocksAndInserts' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n PkInsertTwice' T1 T6 T10
+executeTest 'drop_tab' T1 T6 T10
+
+executeTest 'testBasic -n PkDirtyRead'
+executeTest 'drop_all_tabs'
+
+executeTest 'testBasic -n Fill' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n Fill' T1
+executeTest 'drop_tab' T1
+
+executeTest 'testBasic -n NoCommitSleep' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n NoCommit626' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n NoCommitAndClose' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n Commit626' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n CommitTry626' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n CommitAsMuch626' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n NoCommit626' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n NoCommitRollback626' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n Commit630' T1 T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n CommitTry630' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n CommitAsMuch630' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n NoCommit630' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n NoCommitRollback630' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n NoCommitAndClose' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n RollbackUpdate' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n RollbackDeleteMultiple' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n ImplicitRollbackDelete' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n CommitDelete' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n RollbackNothing' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testBasic -n ReadConsistency' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testBasic -n PkRead' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_4092
+executeTest 'drop_tab' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_4092
+
+executeTest 'testBasic -n PkUpdate' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_4092
+executeTest 'drop_tab' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_4092
+
+executeTest 'testBasic -n PkDelete' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_4092
+executeTest 'drop_tab' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_4092
+
+executeTest 'testBasic -n PkInsert' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_409
+executeTest 'drop_tab' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_4092
+
+executeTest 'testBasic -n UpdateAndRead' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_4092
+#executeTest 'drop_tab' TPK_33 TPK_34 TPK_1003 TPK_2003 TPK_4092
+
+executeTest 'testBasicAsynch -n PkInsertAsynch'
+executeTest 'drop_all_tabs'
+
+executeTest 'testBasicAsynch -n PkReadAsynch'
+executeTest 'drop_all_tabs'
+
+executeTest 'testBasicAsynch -n PkUpdateAsynch'
+executeTest 'drop_all_tabs'
+
+executeTest 'testBasicAsynch -n PkDeleteAsynch'
+executeTest 'drop_all_tabs'
+fi
+
+# SCAN TESTS
+if [ $1 = "scan" ]
+then
+executeTest 'testScan -n ScanRead16'
+executeTest 'drop_all_tabs'
+
+executeTest 'testScan -n ScanRead240'
+executeTest 'drop_all_tabs'
+
+executeTest 'testScan -n ScanUpdate'
+executeTest 'drop_all_tabs'
+
+executeTest 'testScan -n ScanUpdate2' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanDelete'
+executeTest 'drop_all_tab'
+
+executeTest 'testScan -n ScanDelete2' T10
+executeTest 'drop_tab' T10
+
+executeTest 'testScan -n ScanUpdateAndScanRead' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanReadAndLocker' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanReadAndPkRead' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanRead488' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanWithLocksAndInserts' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanReadAbort' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanReadAbort15' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanReadAbort240' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanUpdateAbort16' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanReadRestart' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ScanUpdateRestart' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n CheckGetValue' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n CloseWithoutStop' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n NextScanWhenNoMore' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n ExecuteScanWithoutOpenScan' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n OnlyOpenScanOnce' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n OnlyOneOpInScanTrans' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n OnlyOneOpBeforeOpenScan' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n OnlyOneScanPerTrans' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n NoCloseTransaction' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n CheckInactivityTimeOut' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n CheckInactivityBeforeClose' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testScan -n CheckAfterTerror' T6
+executeTest 'drop_tab' T6
+fi
+
+
+# DICT TESTS
+if [ $1 = "dict" ]
+then
+executeTest 'testDict -n CreateAndDrop'
+executeTest 'drop_all_tabs'
+
+executeTest 'testDict -n CreateAndDropWithData'
+executeTest 'drop_all_tabs'
+
+executeTest 'testDict -n CreateAndDropDuring' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testDict -n CreateInvalidTables'
+executeTest 'drop_all_tabs'
+
+executeTest 'testDict -n CreateTableWhenDbIsFull' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testDict -n CreateMaxTables' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testDict -n FragmentTypeAll' T1 T6 T7 T8
+executeTest 'drop_tab' T1 T6 T7 T8
+
+executeTest 'testDict -n FragmentTypeAllLarge' T1 T6 T7 T8
+executeTest 'drop_tab' T1 T6 T7 T8
+
+executeTest 'testDict -n TemporaryTables' T1 T6 T7 T8
+executeTest 'drop_tab' T1 T6 T7 T8
+fi
+
+# TEST NDBAPI
+if [ $1 = "api" ]
+then
+executeTest 'testNdbApi -n MaxNdb' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testNdbApi -n MaxTransactions' T1 T6 T7 T8 T13
+executeTest 'drop_tab' T1 T6 T7 T8 T13
+
+executeTest 'testNdbApi -n MaxOperations' T1 T6 T7 T8 T1
+executeTest 'drop_tab' T1 T6 T7 T8 T13
+
+executeTest 'testNdbApi -n MaxGetValue' T1 T6 T7 T8 T13
+executeTest 'drop_tab' T1 T6 T7 T8 T13
+
+executeTest 'testNdbApi -n MaxEqual'
+executeTest 'drop_all_tabs'
+
+executeTest 'testNdbApi -n DeleteNdb' T1 T6
+executeTest 'drop_tab' T1 T6
+
+executeTest 'testNdbApi -n WaitUntilReady' T1 T6 T7 T8 T13
+executeTest 'drop_tab' T1 T6 T7 T8 T13
+
+executeTest 'testNdbApi -n GetOperationNoTab' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testNdbApi -n NdbErrorOperation' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testNdbApi -n MissingOperation' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testNdbApi -n GetValueInUpdate' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testNdbApi -n UpdateWithoutKeys' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testNdbApi -n UpdateWithoutValues' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadRead' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadReadEx' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadInsert' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadUpdate' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadDelete' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadExRead' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadExReadEx' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadExInsert' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadExUpdate' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n ReadExDelete' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n InsertRead' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n InsertReadEx' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n InsertInsert' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n InsertUpdate' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n InsertDelete' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n UpdateRead' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n UpdateReadEx' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n UpdateInsert' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n UpdateUpdate' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n UpdateDelete' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n DeleteRead' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n DeleteReadEx' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n DeleteInsert' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n DeleteUpdate' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testOperations -n DeleteDelete' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testRestartGci' T6
+executeTest 'drop_tab' T6
+
+executeTest 'testIndex -n CreateAll'
+executeTest 'drop_all_tabs'
+
+executeTest 'testIndex -n InsertDeleteGentle' T1 T6 T8 T10
+executeTest 'drop_tab' T1 T6 T8 T10
+
+executeTest 'testIndex -n InsertDelete' T1 T6 T8 T10
+executeTest 'drop_tab' T1 T6 T8 T10
+
+executeTest 'testIndex -n CreateLoadDropGentle' T1 T6 T8 T10
+executeTest 'drop_tab' T1 T6 T8 T10
+
+executeTest 'testIndex -n CreateLoadDrop' T1 T6 T8 T10
+executeTest 'drop_tab' T1 T6 T8 T10
+
+executeTest 'testBackup' -n BackupOne
+
+executeTest 'testBackup' -n BackupBank T6
+executeTest 'drop_tab' T6
+fi
+
+# TEST SYSTEM RESTARTS
+if [ $1 = "sr" ]
+then
+executeTest 'testSystemRestart -n SR1' T1
+executeTest 'testSystemRestart -n SR1' T6
+executeTest 'testSystemRestart -n SR1' T7
+executeTest 'testSystemRestart -n SR1' T8
+executeTest 'testSystemRestart -n SR1' T10
+executeTest 'testSystemRestart -n SR2' T1
+executeTest 'testSystemRestart -n SR2' T6
+executeTest 'testSystemRestart -n SR2' T7
+executeTest 'testSystemRestart -n SR2' T10
+executeTest 'testSystemRestart -n SR2' T13
+executeTest 'testSystemRestart -n SR3' T6
+executeTest 'testSystemRestart -n SR3' T10
+executeTest 'testSystemRestart -n SR4' T6
+executeTest 'testSystemRestart -n SR_UNDO' T1
+executeTest 'testSystemRestart -n SR_UNDO' T6
+executeTest 'testSystemRestart -n SR_UNDO' T7
+executeTest 'testSystemRestart -n SR_UNDO' T8
+executeTest 'testSystemRestart -n SR_UNDO' T10
+executeTest 'drop_tab' T1 T6 T7 T8 T10
+fi
+
+# TEST NODE RESTARTS
+if [ $1 = "nr" ]
+then
+executeTest 'testNodeRestart -n NoLoad' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n PkRead' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n PkReadPkUpdate' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n ReadUpdateScan' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n Terror' T6 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n FullDb' T6 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n RestartRandomNode' T6 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n RestartRandomNodeError' T6 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n RestartRandomNodeInitial' T6 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n RestartNFDuringNR' T6 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n RestartNodeDuringLCP' T6 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n RestartMasterNodeError' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n TwoNodeFailure' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n TwoMasterNodeFailure' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n FiftyPercentFail' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n RestartAllNodes' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n RestartAllNodesAbort' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n RestartAllNodesError9999' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+executeTest 'testNodeRestart -n FiftyPercentStopAndWait' T6 T8 T13
+executeTest 'drop_tab' T6 T8 T13
+
+fi
+
+# TESTS FINISHED
+trace "Finished: `date`"
+
+#
+# TEST SUMMARY
+#
+if [ $numOfTestsFailed -eq 0 ]
+then
+ echo "-- REGRESSION TEST SUCCESSFUL --"
+else
+ echo "-- REGRESSION TEST FAILED!! --"
+fi
+echo "Number of successful tests: $numOfTestsOK"
+echo "Number of failed tests : $numOfTestsFailed"
diff --git a/ndb/config/Defs.DEBUG.mk b/ndb/config/Defs.DEBUG.mk
new file mode 100644
index 00000000000..309ae90a0ba
--- /dev/null
+++ b/ndb/config/Defs.DEBUG.mk
@@ -0,0 +1,4 @@
+
+VERSION_FLAGS := -DNDB_DEBUG -DUSE_EMULATED_JAM -DVM_TRACE -DERROR_INSERT -DARRAY_GUARD
+#-DDEBUG_TRANSPORTER
+
diff --git a/ndb/config/Defs.HPUX.HPPA.GCC.mk b/ndb/config/Defs.HPUX.HPPA.GCC.mk
new file mode 100644
index 00000000000..895c7672071
--- /dev/null
+++ b/ndb/config/Defs.HPUX.HPPA.GCC.mk
@@ -0,0 +1,50 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := g++
+CC := gcc
+AR_RCS := ar rcs
+SO := ld -b -o
+
+SHLIBEXT := sl
+
+MAKEDEPEND := g++ -M
+PIC := -fPIC
+
+RPCGENFLAGS := -MA -C -N
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+CCFLAGS_WARNINGS = -Wno-long-long -W -Wall -pedantic
+# -Wno-sign-compare Use this flag if you are annoyed with all the warnings
+CCFLAGS_TOP = -DHPUX -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS -DNO_COMMAND_HANDLER
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O3
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O3 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP = -lpthread -lnsl -lrt
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
+
diff --git a/ndb/config/Defs.IBMAIX.POWERPC.GCC.mk b/ndb/config/Defs.IBMAIX.POWERPC.GCC.mk
new file mode 100644
index 00000000000..ae975fb2cb8
--- /dev/null
+++ b/ndb/config/Defs.IBMAIX.POWERPC.GCC.mk
@@ -0,0 +1,49 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := g++
+CC := gcc
+AR_RCS := $(PURE) ar rcs
+SO := g++ -shared -o
+
+MAKEDEPEND := g++ -M
+PIC := -fPIC
+
+RPCGENFLAGS := -M -C -N
+
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+CCFLAGS_WARNINGS = -Wno-long-long -Wall #-pedantic
+# Add these for more warnings -Weffc++ -W
+CCFLAGS_TOP = -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS
+CCFLAGS_TOP += -fno-rtti
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O3
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O3 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP = -lpthread -lrt
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
diff --git a/ndb/config/Defs.LINUX.x86.GCC.mk b/ndb/config/Defs.LINUX.x86.GCC.mk
new file mode 100644
index 00000000000..a1cc3c52a7e
--- /dev/null
+++ b/ndb/config/Defs.LINUX.x86.GCC.mk
@@ -0,0 +1,56 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := g++$(GCC_VERSION)
+CC := gcc$(GCC_VERSION)
+AR_RCS := $(PURE) ar rcs
+SO := gcc$(GCC_VERSION) -shared -lpthread -o
+
+MAKEDEPEND := g++$(GCC_VERSION) -M
+PIC := -fPIC
+
+RPCGENFLAGS := -M -C -N
+
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+# gcc3.3 __THROW problem if -pedantic and -O2
+ifeq ($(NDB_VERSION),DEBUG)
+CCFLAGS_WARNINGS = -Wno-long-long -Wall -pedantic
+else
+CCFLAGS_WARNINGS = -Wno-long-long -Wall
+endif
+# Add these for more warnings -Weffc++ -W
+CCFLAGS_TOP = -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS
+CCFLAGS_TOP += -fno-rtti -fno-exceptions
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O2
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O2 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP =
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(CC) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
+
+LDFLAGS_LAST = -lpthread -lrt -Wl,-Bstatic -lstdc++ -Wl,-Bdynamic
diff --git a/ndb/config/Defs.LINUX.x86.ICC.mk b/ndb/config/Defs.LINUX.x86.ICC.mk
new file mode 100644
index 00000000000..8e8540409da
--- /dev/null
+++ b/ndb/config/Defs.LINUX.x86.ICC.mk
@@ -0,0 +1,54 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := icc
+CC := icc
+AR_RCS := $(PURE) ar rcs
+SO := g++$(GCC_VERSION) -shared -lpthread -o
+
+MAKEDEPEND := g++$(GCC_VERSION) -M
+PIC := -fPIC
+
+RPCGENFLAGS := -M -C -N
+
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+# gcc3.3 __THROW problem if -pedantic and -O2
+ifeq ($(NDB_VERSION),DEBUG)
+CCFLAGS_WARNINGS =
+else
+CCFLAGS_WARNINGS =
+endif
+# Add these for more warnings -Weffc++ -W
+CCFLAGS_TOP = -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS
+CCFLAGS_TOP +=
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O2
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O2 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP = -lpthread -lrt
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
diff --git a/ndb/config/Defs.LINUX.x86_64.GCC.mk b/ndb/config/Defs.LINUX.x86_64.GCC.mk
new file mode 100644
index 00000000000..a238d29ef4c
--- /dev/null
+++ b/ndb/config/Defs.LINUX.x86_64.GCC.mk
@@ -0,0 +1,54 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := g++
+CC := gcc
+AR_RCS := $(PURE) ar rcs
+SO := g++ -shared -lpthread -o
+
+MAKEDEPEND := g++ -M
+PIC := -fPIC
+
+RPCGENFLAGS := -M -C -N
+
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+# gcc3.3 __THROW problem if -pedantic and -O2
+ifeq ($(NDB_VERSION),DEBUG)
+CCFLAGS_WARNINGS = -Wno-long-long -Wall -pedantic
+else
+CCFLAGS_WARNINGS = -Wno-long-long -Wall
+endif
+# Add these for more warnings -Weffc++ -W
+CCFLAGS_TOP = -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS
+CCFLAGS_TOP += -fno-rtti -fno-exceptions -m64
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O2
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O2 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP = -lpthread -lrt
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
diff --git a/ndb/config/Defs.MACOSX.POWERPC.GCC.mk b/ndb/config/Defs.MACOSX.POWERPC.GCC.mk
new file mode 100644
index 00000000000..bb73e9bcc61
--- /dev/null
+++ b/ndb/config/Defs.MACOSX.POWERPC.GCC.mk
@@ -0,0 +1,58 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := gcc
+CC := gcc
+CXX := gcc
+AR_RCS := $(PURE) ar rcs
+#SO := g++ -dynamiclib -Wl,-segprot,__TEXT,rwx,rwx -o
+SO := gcc -dynamiclib -o
+
+SHLIBEXT := dylib
+
+MAKEDEPEND := gcc -M
+PIC := -fPIC
+
+RPCGENFLAGS := -M -C -N
+
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+CCFLAGS_WARNINGS = -Wno-long-long -Wall -Winline #-Werror#-pedantic
+# Add these for more warnings -Weffc++ -W
+CCFLAGS_TOP = -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS -D_BIG_ENDIAN
+CXX_FLAGS_TOP = -fno-rtti -felide-constructors -fno-exceptions -fno-omit-fram-pointer
+C_FLAGS_TOP += -fno-omit-frame-pointer
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O3
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O3 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(CXXFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(C_FLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP =
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
+
+#LDFLAGS_LAST = -Wl,-Bstatic -lstdc++ -Wl,-Bdynamic
+LDFLAGS_LAST = -lstdc++
+
diff --git a/ndb/config/Defs.OSE.PPC750.DIAB.mk b/ndb/config/Defs.OSE.PPC750.DIAB.mk
new file mode 100644
index 00000000000..8773021a152
--- /dev/null
+++ b/ndb/config/Defs.OSE.PPC750.DIAB.mk
@@ -0,0 +1,47 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := dplus
+CC := dcc
+AR_RCS := $(PURE) ar rcs
+SO := dar -r
+
+MAKEDEPEND := g++ -M -nostdinc
+PIC :=
+
+RPCGENFLAGS := -MA -C -N
+
+###
+#
+# Flags
+#
+CCFLAGS_INCLUDE = -I/vobs/cello/cls/rtosi_if/include -I/vobs/cello/cls/rtosi_if/include.mp750 -I/vobs/cello/cls/rtosi_if/include.ppc
+CCFLAGS_TOP = -tPPC750EH -DBIG_ENDIAN -D_BIG_ENDIAN -DPPC -DPPC750 -DOSE_DELTA -DMP -Xlint -Xforce-prototypes -DINLINE=__inline__ -Xansi -Xsmall-data=0 -Xsmall-const=0 -Xstrings-in-text
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -XO
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -XO -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_INCLUDE)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_INCLUDE)
+
+LDFLAGS_TOP =
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
+
+
+
diff --git a/ndb/config/Defs.RELEASE.mk b/ndb/config/Defs.RELEASE.mk
new file mode 100644
index 00000000000..fad72d53a43
--- /dev/null
+++ b/ndb/config/Defs.RELEASE.mk
@@ -0,0 +1,3 @@
+
+VERSION_FLAGS := -DNDB_RELEASE -DUSE_EMULATED_JAM -DNDEBUG
+
diff --git a/ndb/config/Defs.RELEASE_TRACE.mk b/ndb/config/Defs.RELEASE_TRACE.mk
new file mode 100644
index 00000000000..06726f282e4
--- /dev/null
+++ b/ndb/config/Defs.RELEASE_TRACE.mk
@@ -0,0 +1,3 @@
+
+VERSION_FLAGS := -DNDB_RELEASE -DUSE_EMULATED_JAM -DNDEBUG -DVM_TRACE -DERROR_INSERT -DARRAY_GUARD
+
diff --git a/ndb/config/Defs.SIMCELLO.SOFTOSE.GCC.mk b/ndb/config/Defs.SIMCELLO.SOFTOSE.GCC.mk
new file mode 100644
index 00000000000..8d73e7a752b
--- /dev/null
+++ b/ndb/config/Defs.SIMCELLO.SOFTOSE.GCC.mk
@@ -0,0 +1,53 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := g++
+CC := gcc
+AR_RCS := $(PURE) ar rcs
+SO := g++ -shared -o
+
+MAKEDEPEND := g++ -M
+PIC := -fPIC
+
+###
+#
+# Flags
+#
+NDB_STRDUP := Y
+CCFLAGS_WARNINGS = -Wall -pedantic -Wno-sign-compare
+CC_FLAGS_OSE = -DSPARC -DSIM -DOSE_DELTA -DMP
+CCFLAGS_TOP = $(CC_FLAGS_OSE) $(CC_FLAGS_WARNINGS) -DNDB_STRDUP
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O3
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O3 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+
+CCFLAGS_LOC_OSE= -I/vobs/cello/cls/rtosi_if/include.sparc
+
+
+CCFLAGS = $(CCFLAGS_LOC_OSE) $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC_OSE) $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDLIBS_LOC = -L$(NDB_TOP)/lib -L$(OSE_LOC)/sfk-solaris2/lib -L$(OSE_LOC)/sfk-solaris2/krn-solaris2/lib
+
+LDLIBS_TOP =
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(LDFLAGS)
+
+
+
diff --git a/ndb/config/Defs.SOFTOSE.SPARC.GCC.mk b/ndb/config/Defs.SOFTOSE.SPARC.GCC.mk
new file mode 100644
index 00000000000..6788fa956bf
--- /dev/null
+++ b/ndb/config/Defs.SOFTOSE.SPARC.GCC.mk
@@ -0,0 +1,57 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := g++
+CC := gcc
+AR_RCS := $(PURE) ar rcs
+SO := g++ -shared -o
+
+MAKEDEPEND := g++ -M
+PIC := -fPIC
+
+###
+#
+# Flags
+#
+NDB_STRDUP := Y
+CCFLAGS_WARNINGS = -Wno-long-long -Wall -pedantic -Wno-sign-compare -ansi
+CC_FLAGS_OSE = -DUSE_OSEDEF_H -DOSE_DELTA -DOS_DEBUG -DBIG_ENDIAN
+CCFLAGS_TOP = $(CC_FLAGS_OSE) $(CC_FLAGS_WARNINGS) -DNDB_STRDUP
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O3
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O3 -g
+else
+VERSION_FLAGS += -g -DOS_DEBUG
+endif
+endif
+
+OSE_LOC = /opt/as/OSE/OSE4.3.1
+
+CCFLAGS_LOC_OSESTD = -I$(OSE_LOC)/sfk-solaris2/std-include
+CCFLAGS_LOC_OSE = -I$(OSE_LOC)/sfk-solaris2/include -I$(OSE_LOC)/sfk-solaris2/krn-solaris2/include -I$(NDB_TOP)/src/env/softose
+
+
+CCFLAGS = $(CCFLAGS_LOC_OSE) $(CCFLAGS_LOC_OSESTD) $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC_OSE) $(CCFLAGS_LOC_OSESTD) $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDLIBS_LOC = -L$(NDB_TOP)/lib -L$(OSE_LOC)/sfk-solaris2/lib -L$(OSE_LOC)/sfk-solaris2/krn-solaris2/lib
+
+LDLIBS_TOP =
+
+LDLIBS_LAST = -lsoftose_env -lsoftose_krn -llnh -lefs -lshell -lfss -ltosv -lrtc -lheap -linetutil -linetapi -lsoftose -lsoftose_env -lsoftose_krn -losepthread -lrtc -lnsl -lsocket -lpthread -lcrt -lm
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(LDFLAGS)
+
+
+
diff --git a/ndb/config/Defs.SOLARIS.SPARC.FORTE6.mk b/ndb/config/Defs.SOLARIS.SPARC.FORTE6.mk
new file mode 100644
index 00000000000..8a95205703d
--- /dev/null
+++ b/ndb/config/Defs.SOLARIS.SPARC.FORTE6.mk
@@ -0,0 +1,54 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := CC
+CC := /opt/as/forte6/SUNWspro/bin/cc
+AR_RCS := $(PURE) CC -xar -o
+SO := CC -G -z text -o
+
+MAKEDEPEND := CC -xM1
+PIC := -KPIC
+ETAGS := etags
+CTAGS := ctags
+
+RPCGENFLAGS := -MA -C -N
+
+###
+#
+# Flags
+
+CCFLAGS_TOP = -mt -DSOLARIS -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS
+
+ifneq ($(PURE),)
+ CCFLAGS_TOP += -xs
+ CCFLAGS_TOP += -DNDB_PURIFY
+endif
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -xO3
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -xO3 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS)
+
+LDFLAGS_TOP = -L/opt/as/forte6/SUNWspro/WS6/lib -lpthread -lsocket -lnsl -lrt
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) -xildoff $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
+
+
+
+
diff --git a/ndb/config/Defs.SOLARIS.SPARC.GCC.mk b/ndb/config/Defs.SOLARIS.SPARC.GCC.mk
new file mode 100644
index 00000000000..25920515278
--- /dev/null
+++ b/ndb/config/Defs.SOLARIS.SPARC.GCC.mk
@@ -0,0 +1,54 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+CXX := gcc
+C++ := g++
+CC := gcc
+AR_RCS := ar rcs
+SO := gcc -G -o
+
+#GXX_VERSION := $(shell gcc --version | sed -e 's,.*\([0-9][0-9]*\.[0-9][0-9]*\.[0-9][0-9]*\).*,\1,1' -e q)
+
+MAKEDEPEND := g++ -M
+PIC := -fPIC
+
+RPCGENFLAGS := -MA -C -N
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+CCFLAGS_WARNINGS = -Wno-long-long -W -Wall -pedantic
+# -Wno-sign-compare Use this flag if you are annoyed with all the warnings
+CCFLAGS_TOP = -DSOLARIS -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS -DNO_COMMAND_HANDLER
+CCFLAGS_TOP += -fno-rtti
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O2
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O2 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP =
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(CXX) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
+
+LDFLAGS_LAST = -lpthread -lsocket -lnsl -lrt -Wl,-Bstatic -lstdc++ -Wl,-Bdynamic
+
diff --git a/ndb/config/Defs.SOLARIS.SPARC_64.GCC.mk b/ndb/config/Defs.SOLARIS.SPARC_64.GCC.mk
new file mode 100644
index 00000000000..2b8b9d4cc24
--- /dev/null
+++ b/ndb/config/Defs.SOLARIS.SPARC_64.GCC.mk
@@ -0,0 +1,53 @@
+###
+#
+# Note: LD_LIBRARY_PATH must be set for /usr/local/lib/sparcv9 to dynamically link
+# to 64-bit libraries
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := g++ -m64
+CC := gcc -m64
+AR_RCS := ar rcs
+SO := g++ -m64 -shared -o
+
+MAKEDEPEND := g++ -M
+PIC := -fPIC
+
+RPCGENFLAGS := -MA -C -N
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+CCFLAGS_WARNINGS = -Wno-long-long -W -Wall -pedantic
+# -Wno-sign-compare Use this flag if you are annoyed with all the warnings
+CCFLAGS_TOP = -DSOLARIS -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS -DNO_COMMAND_HANDLER
+CCFLAGS_TOP += -fno-rtti
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O2
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O2 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP = -lpthread -lsocket -lnsl -lrt
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
+
+
diff --git a/ndb/config/Defs.SOLARIS6.SPARC.GCC.mk b/ndb/config/Defs.SOLARIS6.SPARC.GCC.mk
new file mode 100644
index 00000000000..f1c570ba101
--- /dev/null
+++ b/ndb/config/Defs.SOLARIS6.SPARC.GCC.mk
@@ -0,0 +1,53 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := g++
+CC := gcc
+AR_RCS := $(PURE) ar rcs
+SO := g++ -shared -o
+
+MAKEDEPEND := g++ -M
+PIC := -fPIC
+
+RPCGENFLAGS := -MA -C -N
+
+###
+#
+# Flags
+#
+CCFLAGS_WARNINGS = -Wno-long-long -Wall -pedantic
+# -Wno-sign-compare Use this flag if you are annoyed with all the warnings
+CCFLAGS_TOP = -DSOLARIS -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS -DNO_COMMAND_HANDLER
+
+# SOLARIS 6 should use the same settings as SOLARIS7
+# if something in the SOLARIS 7 port does not work for SOLARIS 6
+# it can be ifdefed using
+# if ! defined NDB_SOLRIS6
+CCFLAGS_TOP = -DNDB_SOLARIS
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O3
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O3
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP = -lpthread -lsocket -lnsl -lposix4
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
+
+
diff --git a/ndb/config/Defs.TRU64X.ALPHA.GCC.mk b/ndb/config/Defs.TRU64X.ALPHA.GCC.mk
new file mode 100644
index 00000000000..ae975fb2cb8
--- /dev/null
+++ b/ndb/config/Defs.TRU64X.ALPHA.GCC.mk
@@ -0,0 +1,49 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+C++ := g++
+CC := gcc
+AR_RCS := $(PURE) ar rcs
+SO := g++ -shared -o
+
+MAKEDEPEND := g++ -M
+PIC := -fPIC
+
+RPCGENFLAGS := -M -C -N
+
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+CCFLAGS_WARNINGS = -Wno-long-long -Wall #-pedantic
+# Add these for more warnings -Weffc++ -W
+CCFLAGS_TOP = -D_REENTRANT -D_POSIX_PTHREAD_SEMANTICS
+CCFLAGS_TOP += -fno-rtti
+
+ifeq (RELEASE, $(NDB_VERSION))
+VERSION_FLAGS += -O3
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+VERSION_FLAGS += -O3 -g
+else
+VERSION_FLAGS += -g
+endif
+endif
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP = -lpthread -lrt
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+LINK.cc = $(PURE) $(C++) $(CCFLAGS) $(LDFLAGS)
+
+LINK.c = $(PURE) $(CC) $(CFLAGS) $(LDFLAGS)
diff --git a/ndb/config/Defs.WIN32.x86.VC7.mk b/ndb/config/Defs.WIN32.x86.VC7.mk
new file mode 100644
index 00000000000..e66dacb78e7
--- /dev/null
+++ b/ndb/config/Defs.WIN32.x86.VC7.mk
@@ -0,0 +1,61 @@
+###
+#
+# Defines
+SHELL := /bin/sh
+
+
+DEFINES = -D_WIN32 -D_M_IX86=600 -D_MSC_EXTENSIONS=0 -U_cdecl -D_MT
+#
+MAKEDEPEND = g++ -M --nostdinc --nostdinc++ -I"`cygpath -u "$(MSVCDIR)\include"`" -I"`cygpath -u "$(MSVCDIR)\PlatformSDK\include"`" $(DEFINES)
+PIC = -D_LIB
+NON_PIC = -D_LIB
+
+RPCGENFLAGS := -M -C -N
+
+ETAGS := etags
+CTAGS := ctags
+
+###
+#
+# Flags
+#
+CCFLAGS_WARNINGS =
+CCFLAGS_TOP =
+CCFLAGS_LOC =
+CCFLAGS_WIN = -DWIN32 -D_WIN32_WINNT=0x0500 -DWINVER=0x0500 -D_MBCS -DNO_COMMAND_HANDLER
+CCFLAGS_WIN += -W3 -EHsc
+#CCFLAGS_WIN += -clr
+
+ifeq (RELEASE, $(NDB_VERSION))
+CCFLAGS_WIN += -MT -O2 -Ob1 -DNO_DEBUG_MESSAGES
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+CCFLAGS_WIN += -MT -O2 -Ob1 -DNO_DEBUG_MESSAGES
+else
+CCFLAGS_WIN += -MTd -Zi -Od -GS -D_DEBUG
+endif
+endif
+
+C++ = cl -nologo $(CCFLAGS_WIN)
+CC = cl -nologo $(CCFLAGS_WIN)
+
+CCFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+CFLAGS = $(CCFLAGS_LOC) $(CCFLAGS_TOP) $(USER_FLAGS) $(VERSION_FLAGS) $(CCFLAGS_WARNINGS)
+
+LDFLAGS_TOP =
+
+LDFLAGS = $(LDFLAGS_LOC) $(LDFLAGS_TOP)
+
+LDLIBS = $(LDLIBS_LOC) $(LDLIBS_TOP)
+
+WIN_LIBS := Ws2_32.lib Advapi32.lib
+
+ifeq (RELEASE, $(NDB_VERSION))
+LINK.cc = link -INCREMENTAL:NO -NOLOGO -LARGEADDRESSAWARE $(WIN_LIBS)
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+LINK.cc = link -INCREMENTAL:NO -NOLOGO -LARGEADDRESSAWARE $(WIN_LIBS)
+else
+LINK.cc = link -INCREMENTAL -NOLOGO -DEBUG -LARGEADDRESSAWARE $(WIN_LIBS)
+endif
+endif
diff --git a/ndb/config/GuessConfig.sh b/ndb/config/GuessConfig.sh
new file mode 100755
index 00000000000..a1ecdecfa93
--- /dev/null
+++ b/ndb/config/GuessConfig.sh
@@ -0,0 +1,113 @@
+#! /bin/sh
+
+if [ -z "$NDB_TOP" ]
+then
+ echo "You have not set NDB_TOP. Exiting" 1>&2
+ exit 1
+fi
+
+if [ -z "$NDB_SCI" ]
+then
+ NDB_SCI=N
+fi
+
+os=`uname -s`
+case $os in
+Linux)
+ NDB_OS=LINUX
+ NDB_ARCH=x86
+ NDB_COMPILER=GCC
+ ;;
+Darwin)
+ NDB_OS=MACOSX
+ NDB_ARCH=POWERPC
+ NDB_COMPILER=GCC
+ ;;
+HP-UX)
+ NDB_OS=HPUX
+ NDB_ARCH=HPPA
+ NDB_COMPILER=GCC
+ ;;
+CYGWIN_NT-5.0)
+ NDB_OS=WIN32
+ NDB_ARCH=x86
+ NDB_COMPILER=VC7
+ ;;
+*)
+ if [ "$os" = "SunOS" ] && [ `uname -r` = "5.6" ]
+ then
+ NDB_OS=OSE
+ NDB_ARCH=PPC750
+ NDB_COMPILER=DIAB
+ else
+ NDB_OS=SOLARIS
+ NDB_ARCH=SPARC
+ NDB_COMPILER=GCC
+ fi;;
+esac
+
+if [ -z "$NDB_ODBC" ]
+then
+ val=N
+ if [ -f /usr/include/sqlext.h -o -f /usr/local/include/sqlext.h ]
+ then
+ val=Y
+ fi
+ case $NDB_OS in
+ LINUX)
+ NDB_ODBC=$val
+ ;;
+ MACOSX)
+ NDB_ODBC=$val
+ ;;
+ *)
+ NDB_ODBC=N
+ ;;
+ esac
+fi
+
+
+mch=`uname -m`
+case $mch in
+x86_64)
+ NDB_ARCH=x86_64
+ ;;
+*)
+ ;;
+esac
+
+if [ -f $NDB_TOP/config/Makefile ]
+then
+TERMCAP_LIB=`grep TERMCAP_LIB $NDB_TOP/config/Makefile | sed -e s,"TERMCAP_LIB.*=.*-l","",g`
+fi
+if [ "$TERMCAP_LIB" = "" ]
+then
+TERMCAP_LIB=termcap
+fi
+
+# Allow for selecting GCC, but must be 2nd parameter
+if [ $# -gt 1 -a "$2" = "-GCC" ]
+then
+ NDB_COMPILER=GCC
+fi
+
+(
+ echo "# This file was automatically generated `date`"
+ echo "NDB_OS := $NDB_OS"
+ echo "NDB_ARCH := $NDB_ARCH"
+ echo "NDB_COMPILER := $NDB_COMPILER"
+
+ if [ $# -gt 0 -a "$1" = "-R" ]
+ then
+ echo "NDB_VERSION := RELEASE"
+ else
+ echo "NDB_VERSION := DEBUG"
+ fi
+
+ echo "NDB_SCI := $NDB_SCI"
+ echo "NDB_ODBC := $NDB_ODBC"
+ echo "TERMCAP_LIB := $TERMCAP_LIB"
+) > $NDB_TOP/config/config.mk
+
+exit 0
+
diff --git a/ndb/config/Makefile.am b/ndb/config/Makefile.am
new file mode 100644
index 00000000000..b5fd81814a1
--- /dev/null
+++ b/ndb/config/Makefile.am
@@ -0,0 +1,31 @@
+# Copyright (C) 2003 MySQL AB
+#
+# 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
+
+# Process this file with automake to create Makefile.in
+
+AUTOMAKE_OPTIONS = foreign
+
+# These are built from source in the Docs directory
+EXTRA_DIST =
+SUBDIRS =
+
+# Relink after clean
+linked_sources =
+
+CLEANFILES = $(linked_sources)
+
+# This is just so that the linking is done early.
+config.h:
diff --git a/ndb/config/acinclude.m4 b/ndb/config/acinclude.m4
new file mode 100644
index 00000000000..b9edaf801ed
--- /dev/null
+++ b/ndb/config/acinclude.m4
@@ -0,0 +1,1513 @@
+# Local macros for automake & autoconf
+
+AC_DEFUN(MYSQL_CHECK_LIBEDIT_INTERFACE,[
+ AC_CACHE_CHECK([libedit variant of rl_completion_entry_function], mysql_cv_libedit_interface,
+ AC_TRY_COMPILE(
+ [
+ #include "stdio.h"
+ #include "readline/readline.h"
+ ],
+ [
+ char res= *(*rl_completion_entry_function)(0,0);
+ completion_matches(0,0);
+ ],
+ [
+ mysql_cv_libedit_interface=yes
+ AC_DEFINE_UNQUOTED(USE_LIBEDIT_INTERFACE)
+ ],
+ [mysql_cv_libedit_interface=no]
+ )
+ )
+])
+
+AC_DEFUN(MYSQL_CHECK_NEW_RL_INTERFACE,[
+ AC_CACHE_CHECK([defined rl_compentry_func_t and rl_completion_func_t], mysql_cv_new_rl_interface,
+ AC_TRY_COMPILE(
+ [
+ #include "stdio.h"
+ #include "readline/readline.h"
+ ],
+ [
+ rl_completion_func_t *func1= (rl_completion_func_t*)0;
+ rl_compentry_func_t *func2= (rl_compentry_func_t*)0;
+ ],
+ [
+ mysql_cv_new_rl_interface=yes
+ AC_DEFINE_UNQUOTED(USE_NEW_READLINE_INTERFACE)
+ ],
+ [mysql_cv_new_rl_interface=no]
+ )
+ )
+])
+
+# A local version of AC_CHECK_SIZEOF that includes sys/types.h
+dnl MYSQL_CHECK_SIZEOF(TYPE [, CROSS-SIZE])
+AC_DEFUN(MYSQL_CHECK_SIZEOF,
+[changequote(<<, >>)dnl
+dnl The name to #define.
+define(<<AC_TYPE_NAME>>, translit(sizeof_$1, [a-z *], [A-Z_P]))dnl
+dnl The cache variable name.
+define(<<AC_CV_NAME>>, translit(ac_cv_sizeof_$1, [ *], [_p]))dnl
+changequote([, ])dnl
+AC_MSG_CHECKING(size of $1)
+AC_CACHE_VAL(AC_CV_NAME,
+[AC_TRY_RUN([#include <stdio.h>
+#include <sys/types.h>
+#if STDC_HEADERS
+#include <stdlib.h>
+#include <stddef.h>
+#endif
+main()
+{
+ FILE *f=fopen("conftestval", "w");
+ if (!f) exit(1);
+ fprintf(f, "%d\n", sizeof($1));
+ exit(0);
+}], AC_CV_NAME=`cat conftestval`, AC_CV_NAME=0, ifelse([$2], , , AC_CV_NAME=$2))])dnl
+AC_MSG_RESULT($AC_CV_NAME)
+AC_DEFINE_UNQUOTED(AC_TYPE_NAME, $AC_CV_NAME)
+undefine([AC_TYPE_NAME])dnl
+undefine([AC_CV_NAME])dnl
+])
+
+#---START: Used in for client configure
+AC_DEFUN(MYSQL_TYPE_ACCEPT,
+[ac_save_CXXFLAGS="$CXXFLAGS"
+AC_CACHE_CHECK([base type of last arg to accept], mysql_cv_btype_last_arg_accept,
+AC_LANG_SAVE
+AC_LANG_CPLUSPLUS
+if test "$ac_cv_prog_gxx" = "yes"
+then
+ CXXFLAGS=`echo $CXXFLAGS -Werror | sed 's/-fbranch-probabilities//'`
+fi
+mysql_cv_btype_last_arg_accept=none
+[AC_TRY_COMPILE([#if defined(inline)
+#undef inline
+#endif
+#include <stdlib.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+],
+[int a = accept(1, (struct sockaddr *) 0, (socklen_t *) 0); return (a != 0);],
+mysql_cv_btype_last_arg_accept=socklen_t)]
+if test "$mysql_cv_btype_last_arg_accept" = "none"; then
+[AC_TRY_COMPILE([#if defined(inline)
+#undef inline
+#endif
+#include <stdlib.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+],
+[int a = accept(1, (struct sockaddr *) 0, (size_t *) 0); return (a != 0);],
+mysql_cv_btype_last_arg_accept=size_t)]
+fi
+if test "$mysql_cv_btype_last_arg_accept" = "none"; then
+mysql_cv_btype_last_arg_accept=int
+fi)
+AC_LANG_RESTORE
+AC_DEFINE_UNQUOTED(SOCKET_SIZE_TYPE, $mysql_cv_btype_last_arg_accept)
+CXXFLAGS="$ac_save_CXXFLAGS"
+])
+#---END:
+
+dnl Find type of qsort
+AC_DEFUN(MYSQL_TYPE_QSORT,
+[AC_CACHE_CHECK([return type of qsort], mysql_cv_type_qsort,
+[AC_TRY_COMPILE([#include <stdlib.h>
+#ifdef __cplusplus
+extern "C"
+#endif
+void qsort(void *base, size_t nel, size_t width,
+ int (*compar) (const void *, const void *));
+],
+[int i;], mysql_cv_type_qsort=void, mysql_cv_type_qsort=int)])
+AC_DEFINE_UNQUOTED(RETQSORTTYPE, $mysql_cv_type_qsort)
+if test "$mysql_cv_type_qsort" = "void"
+then
+ AC_DEFINE_UNQUOTED(QSORT_TYPE_IS_VOID, 1)
+fi
+])
+
+AC_DEFUN(MYSQL_TIMESPEC_TS,
+[AC_CACHE_CHECK([if struct timespec has a ts_sec member], mysql_cv_timespec_ts,
+[AC_TRY_COMPILE([#include <pthread.h>
+#ifdef __cplusplus
+extern "C"
+#endif
+],
+[struct timespec abstime;
+
+abstime.ts_sec = time(NULL)+1;
+abstime.ts_nsec = 0;
+], mysql_cv_timespec_ts=yes, mysql_cv_timespec_ts=no)])
+if test "$mysql_cv_timespec_ts" = "yes"
+then
+ AC_DEFINE(HAVE_TIMESPEC_TS_SEC)
+fi
+])
+
+AC_DEFUN(MYSQL_TZNAME,
+[AC_CACHE_CHECK([if we have tzname variable], mysql_cv_tzname,
+[AC_TRY_COMPILE([#include <time.h>
+#ifdef __cplusplus
+extern "C"
+#endif
+],
+[ tzset();
+ return tzname[0] != 0;
+], mysql_cv_tzname=yes, mysql_cv_tzname=no)])
+if test "$mysql_cv_tzname" = "yes"
+then
+ AC_DEFINE(HAVE_TZNAME)
+fi
+])
+
+AC_DEFUN(MYSQL_CHECK_ZLIB_WITH_COMPRESS, [
+save_LIBS="$LIBS"
+LIBS="-l$1 $LIBS"
+AC_CACHE_CHECK([if libz with compress], mysql_cv_compress,
+[AC_TRY_RUN([#include <zlib.h>
+#ifdef __cplusplus
+extern "C"
+#endif
+int main(int argv, char **argc)
+{
+ return 0;
+}
+
+int link_test()
+{
+ return compress(0, (unsigned long*) 0, "", 0);
+}
+], mysql_cv_compress=yes, mysql_cv_compress=no)])
+if test "$mysql_cv_compress" = "yes"
+then
+ AC_DEFINE(HAVE_COMPRESS)
+else
+ LIBS="$save_LIBS"
+fi
+])
+
+#---START: Used in for client configure
+AC_DEFUN(MYSQL_CHECK_ULONG,
+[AC_MSG_CHECKING(for type ulong)
+AC_CACHE_VAL(ac_cv_ulong,
+[AC_TRY_RUN([#include <stdio.h>
+#include <sys/types.h>
+main()
+{
+ ulong foo;
+ foo++;
+ exit(0);
+}], ac_cv_ulong=yes, ac_cv_ulong=no, ac_cv_ulong=no)])
+AC_MSG_RESULT($ac_cv_ulong)
+if test "$ac_cv_ulong" = "yes"
+then
+ AC_DEFINE(HAVE_ULONG)
+fi
+])
+
+AC_DEFUN(MYSQL_CHECK_UCHAR,
+[AC_MSG_CHECKING(for type uchar)
+AC_CACHE_VAL(ac_cv_uchar,
+[AC_TRY_RUN([#include <stdio.h>
+#include <sys/types.h>
+main()
+{
+ uchar foo;
+ foo++;
+ exit(0);
+}], ac_cv_uchar=yes, ac_cv_uchar=no, ac_cv_uchar=no)])
+AC_MSG_RESULT($ac_cv_uchar)
+if test "$ac_cv_uchar" = "yes"
+then
+ AC_DEFINE(HAVE_UCHAR)
+fi
+])
+
+AC_DEFUN(MYSQL_CHECK_UINT,
+[AC_MSG_CHECKING(for type uint)
+AC_CACHE_VAL(ac_cv_uint,
+[AC_TRY_RUN([#include <stdio.h>
+#include <sys/types.h>
+main()
+{
+ uint foo;
+ foo++;
+ exit(0);
+}], ac_cv_uint=yes, ac_cv_uint=no, ac_cv_uint=no)])
+AC_MSG_RESULT($ac_cv_uint)
+if test "$ac_cv_uint" = "yes"
+then
+ AC_DEFINE(HAVE_UINT)
+fi
+])
+
+
+AC_DEFUN(MYSQL_CHECK_IN_ADDR_T,
+[AC_MSG_CHECKING(for type in_addr_t)
+AC_CACHE_VAL(ac_cv_in_addr_t,
+[AC_TRY_RUN([#include <stdio.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+
+int main(int argc, char **argv)
+{
+ in_addr_t foo;
+ exit(0);
+}], ac_cv_in_addr_t=yes, ac_cv_in_addr_t=no, ac_cv_in_addr_t=no)])
+AC_MSG_RESULT($ac_cv_in_addr_t)
+if test "$ac_cv_in_addr_t" = "yes"
+then
+ AC_DEFINE(HAVE_IN_ADDR_T)
+fi
+])
+
+
+AC_DEFUN(MYSQL_PTHREAD_YIELD,
+[AC_CACHE_CHECK([if pthread_yield takes zero arguments], ac_cv_pthread_yield_zero_arg,
+[AC_TRY_LINK([#define _GNU_SOURCE
+#include <pthread.h>
+#ifdef __cplusplus
+extern "C"
+#endif
+],
+[
+ pthread_yield();
+], ac_cv_pthread_yield_zero_arg=yes, ac_cv_pthread_yield_zero_arg=yeso)])
+if test "$ac_cv_pthread_yield_zero_arg" = "yes"
+then
+ AC_DEFINE(HAVE_PTHREAD_YIELD_ZERO_ARG)
+fi
+]
+[AC_CACHE_CHECK([if pthread_yield takes 1 argument], ac_cv_pthread_yield_one_arg,
+[AC_TRY_LINK([#define _GNU_SOURCE
+#include <pthread.h>
+#ifdef __cplusplus
+extern "C"
+#endif
+],
+[
+ pthread_yield(0);
+], ac_cv_pthread_yield_one_arg=yes, ac_cv_pthread_yield_one_arg=no)])
+if test "$ac_cv_pthread_yield_one_arg" = "yes"
+then
+ AC_DEFINE(HAVE_PTHREAD_YIELD_ONE_ARG)
+fi
+]
+)
+
+
+
+#---END:
+
+AC_DEFUN(MYSQL_CHECK_FP_EXCEPT,
+[AC_MSG_CHECKING(for type fp_except)
+AC_CACHE_VAL(ac_cv_fp_except,
+[AC_TRY_RUN([#include <stdio.h>
+#include <sys/types.h>
+#include <ieeefp.h>
+main()
+{
+ fp_except foo;
+ foo++;
+ exit(0);
+}], ac_cv_fp_except=yes, ac_cv_fp_except=no, ac_cv_fp_except=no)])
+AC_MSG_RESULT($ac_cv_fp_except)
+if test "$ac_cv_fp_except" = "yes"
+then
+ AC_DEFINE(HAVE_FP_EXCEPT)
+fi
+])
+
+# From fileutils-3.14/aclocal.m4
+
+# @defmac AC_PROG_CC_STDC
+# @maindex PROG_CC_STDC
+# @ovindex CC
+# If the C compiler in not in ANSI C mode by default, try to add an option
+# to output variable @code{CC} to make it so. This macro tries various
+# options that select ANSI C on some system or another. It considers the
+# compiler to be in ANSI C mode if it defines @code{__STDC__} to 1 and
+# handles function prototypes correctly.
+#
+# Patched by monty to only check if __STDC__ is defined. With the original
+# check it's impossible to get things to work with the Sunpro compiler from
+# Workshop 4.2
+#
+# If you use this macro, you should check after calling it whether the C
+# compiler has been set to accept ANSI C; if not, the shell variable
+# @code{am_cv_prog_cc_stdc} is set to @samp{no}. If you wrote your source
+# code in ANSI C, you can make an un-ANSIfied copy of it by using the
+# program @code{ansi2knr}, which comes with Ghostscript.
+# @end defmac
+
+AC_DEFUN(AM_PROG_CC_STDC,
+[AC_REQUIRE([AC_PROG_CC])
+AC_MSG_CHECKING(for ${CC-cc} option to accept ANSI C)
+AC_CACHE_VAL(am_cv_prog_cc_stdc,
+[am_cv_prog_cc_stdc=no
+ac_save_CC="$CC"
+# Don't try gcc -ansi; that turns off useful extensions and
+# breaks some systems' header files.
+# AIX -qlanglvl=ansi
+# Ultrix and OSF/1 -std1
+# HP-UX -Aa -D_HPUX_SOURCE
+# SVR4 -Xc -D__EXTENSIONS__
+# removed "-Xc -D__EXTENSIONS__" beacause sun c++ does not like it.
+for ac_arg in "" -qlanglvl=ansi -std1 "-Aa -D_HPUX_SOURCE"
+do
+ CC="$ac_save_CC $ac_arg"
+ AC_TRY_COMPILE(
+[#if !defined(__STDC__)
+choke me
+#endif
+/* DYNIX/ptx V4.1.3 can't compile sys/stat.h with -Xc -D__EXTENSIONS__. */
+#ifdef _SEQUENT_
+# include <sys/types.h>
+# include <sys/stat.h>
+#endif
+], [
+int test (int i, double x);
+struct s1 {int (*f) (int a);};
+struct s2 {int (*f) (double a);};],
+[am_cv_prog_cc_stdc="$ac_arg"; break])
+done
+CC="$ac_save_CC"
+])
+AC_MSG_RESULT($am_cv_prog_cc_stdc)
+case "x$am_cv_prog_cc_stdc" in
+ x|xno) ;;
+ *) CC="$CC $am_cv_prog_cc_stdc" ;;
+esac
+])
+
+#
+# Check to make sure that the build environment is sane.
+#
+
+AC_DEFUN(AM_SANITY_CHECK,
+[AC_MSG_CHECKING([whether build environment is sane])
+sleep 1
+echo timestamp > conftestfile
+# Do this in a subshell so we don't clobber the current shell's
+# arguments. FIXME: maybe try `-L' hack like GETLOADAVG test?
+if (set X `ls -t $srcdir/configure conftestfile`; test "[$]2" = conftestfile)
+then
+ # Ok.
+ :
+else
+ AC_MSG_ERROR([newly created file is older than distributed files!
+Check your system clock])
+fi
+rm -f conftest*
+AC_MSG_RESULT(yes)])
+
+# Orginal from bash-2.0 aclocal.m4, Changed to use termcap last by monty.
+
+AC_DEFUN(MYSQL_CHECK_LIB_TERMCAP,
+[
+AC_CACHE_VAL(mysql_cv_termcap_lib,
+[AC_CHECK_LIB(ncurses, tgetent, mysql_cv_termcap_lib=libncurses,
+ [AC_CHECK_LIB(curses, tgetent, mysql_cv_termcap_lib=libcurses,
+ [AC_CHECK_LIB(termcap, tgetent, mysql_cv_termcap_lib=libtermcap,
+ mysql_cv_termcap_lib=NOT_FOUND)])])])
+AC_MSG_CHECKING(for termcap functions library)
+if test "$mysql_cv_termcap_lib" = "NOT_FOUND"; then
+AC_MSG_ERROR([No curses/termcap library found])
+elif test "$mysql_cv_termcap_lib" = "libtermcap"; then
+TERMCAP_LIB=-ltermcap
+elif test "$mysql_cv_termcap_lib" = "libncurses"; then
+TERMCAP_LIB=-lncurses
+else
+TERMCAP_LIB=-lcurses
+fi
+AC_MSG_RESULT($TERMCAP_LIB)
+])
+
+dnl Check type of signal routines (posix, 4.2bsd, 4.1bsd or v7)
+AC_DEFUN(MYSQL_SIGNAL_CHECK,
+[AC_REQUIRE([AC_TYPE_SIGNAL])
+AC_MSG_CHECKING(for type of signal functions)
+AC_CACHE_VAL(mysql_cv_signal_vintage,
+[
+ AC_TRY_LINK([#include <signal.h>],[
+ sigset_t ss;
+ struct sigaction sa;
+ sigemptyset(&ss); sigsuspend(&ss);
+ sigaction(SIGINT, &sa, (struct sigaction *) 0);
+ sigprocmask(SIG_BLOCK, &ss, (sigset_t *) 0);
+ ], mysql_cv_signal_vintage=posix,
+ [
+ AC_TRY_LINK([#include <signal.h>], [
+ int mask = sigmask(SIGINT);
+ sigsetmask(mask); sigblock(mask); sigpause(mask);
+ ], mysql_cv_signal_vintage=4.2bsd,
+ [
+ AC_TRY_LINK([
+ #include <signal.h>
+ RETSIGTYPE foo() { }], [
+ int mask = sigmask(SIGINT);
+ sigset(SIGINT, foo); sigrelse(SIGINT);
+ sighold(SIGINT); sigpause(SIGINT);
+ ], mysql_cv_signal_vintage=svr3, mysql_cv_signal_vintage=v7
+ )]
+ )]
+)
+])
+AC_MSG_RESULT($mysql_cv_signal_vintage)
+if test "$mysql_cv_signal_vintage" = posix; then
+AC_DEFINE(HAVE_POSIX_SIGNALS)
+elif test "$mysql_cv_signal_vintage" = "4.2bsd"; then
+AC_DEFINE(HAVE_BSD_SIGNALS)
+elif test "$mysql_cv_signal_vintage" = svr3; then
+AC_DEFINE(HAVE_USG_SIGHOLD)
+fi
+])
+
+AC_DEFUN(MYSQL_CHECK_GETPW_FUNCS,
+[AC_MSG_CHECKING(whether programs are able to redeclare getpw functions)
+AC_CACHE_VAL(mysql_cv_can_redecl_getpw,
+[AC_TRY_COMPILE([#include <sys/types.h>
+#include <pwd.h>
+extern struct passwd *getpwent();], [struct passwd *z; z = getpwent();],
+ mysql_cv_can_redecl_getpw=yes,mysql_cv_can_redecl_getpw=no)])
+AC_MSG_RESULT($mysql_cv_can_redecl_getpw)
+if test "$mysql_cv_can_redecl_getpw" = "no"; then
+AC_DEFINE(HAVE_GETPW_DECLS)
+fi
+])
+
+AC_DEFUN(MYSQL_HAVE_TIOCGWINSZ,
+[AC_MSG_CHECKING(for TIOCGWINSZ in sys/ioctl.h)
+AC_CACHE_VAL(mysql_cv_tiocgwinsz_in_ioctl,
+[AC_TRY_COMPILE([#include <sys/types.h>
+#include <sys/ioctl.h>], [int x = TIOCGWINSZ;],
+ mysql_cv_tiocgwinsz_in_ioctl=yes,mysql_cv_tiocgwinsz_in_ioctl=no)])
+AC_MSG_RESULT($mysql_cv_tiocgwinsz_in_ioctl)
+if test "$mysql_cv_tiocgwinsz_in_ioctl" = "yes"; then
+AC_DEFINE(GWINSZ_IN_SYS_IOCTL)
+fi
+])
+
+AC_DEFUN(MYSQL_HAVE_FIONREAD,
+[AC_MSG_CHECKING(for FIONREAD in sys/ioctl.h)
+AC_CACHE_VAL(mysql_cv_fionread_in_ioctl,
+[AC_TRY_COMPILE([#include <sys/types.h>
+#include <sys/ioctl.h>], [int x = FIONREAD;],
+ mysql_cv_fionread_in_ioctl=yes,mysql_cv_fionread_in_ioctl=no)])
+AC_MSG_RESULT($mysql_cv_fionread_in_ioctl)
+if test "$mysql_cv_fionread_in_ioctl" = "yes"; then
+AC_DEFINE(FIONREAD_IN_SYS_IOCTL)
+fi
+])
+
+AC_DEFUN(MYSQL_HAVE_TIOCSTAT,
+[AC_MSG_CHECKING(for TIOCSTAT in sys/ioctl.h)
+AC_CACHE_VAL(mysql_cv_tiocstat_in_ioctl,
+[AC_TRY_COMPILE([#include <sys/types.h>
+#include <sys/ioctl.h>], [int x = TIOCSTAT;],
+ mysql_cv_tiocstat_in_ioctl=yes,mysql_cv_tiocstat_in_ioctl=no)])
+AC_MSG_RESULT($mysql_cv_tiocstat_in_ioctl)
+if test "$mysql_cv_tiocstat_in_ioctl" = "yes"; then
+AC_DEFINE(TIOCSTAT_IN_SYS_IOCTL)
+fi
+])
+
+AC_DEFUN(MYSQL_STRUCT_DIRENT_D_INO,
+[AC_REQUIRE([AC_HEADER_DIRENT])
+AC_MSG_CHECKING(if struct dirent has a d_ino member)
+AC_CACHE_VAL(mysql_cv_dirent_has_dino,
+[AC_TRY_COMPILE([
+#include <stdio.h>
+#include <sys/types.h>
+#ifdef HAVE_UNISTD_H
+# include <unistd.h>
+#endif /* HAVE_UNISTD_H */
+#if defined(HAVE_DIRENT_H)
+# include <dirent.h>
+#else
+# define dirent direct
+# ifdef HAVE_SYS_NDIR_H
+# include <sys/ndir.h>
+# endif /* SYSNDIR */
+# ifdef HAVE_SYS_DIR_H
+# include <sys/dir.h>
+# endif /* SYSDIR */
+# ifdef HAVE_NDIR_H
+# include <ndir.h>
+# endif
+#endif /* HAVE_DIRENT_H */
+],[
+struct dirent d; int z; z = d.d_ino;
+], mysql_cv_dirent_has_dino=yes, mysql_cv_dirent_has_dino=no)])
+AC_MSG_RESULT($mysql_cv_dirent_has_dino)
+if test "$mysql_cv_dirent_has_dino" = "yes"; then
+AC_DEFINE(STRUCT_DIRENT_HAS_D_INO)
+fi
+])
+
+AC_DEFUN(MYSQL_TYPE_SIGHANDLER,
+[AC_MSG_CHECKING([whether signal handlers are of type void])
+AC_CACHE_VAL(mysql_cv_void_sighandler,
+[AC_TRY_COMPILE([#include <sys/types.h>
+#include <signal.h>
+#ifdef signal
+#undef signal
+#endif
+#ifdef __cplusplus
+extern "C"
+#endif
+void (*signal ()) ();],
+[int i;], mysql_cv_void_sighandler=yes, mysql_cv_void_sighandler=no)])dnl
+AC_MSG_RESULT($mysql_cv_void_sighandler)
+if test "$mysql_cv_void_sighandler" = "yes"; then
+AC_DEFINE(VOID_SIGHANDLER)
+fi
+])
+
+AC_DEFUN(MYSQL_CXX_BOOL,
+[
+AC_REQUIRE([AC_PROG_CXX])
+AC_MSG_CHECKING(if ${CXX} supports bool types)
+AC_CACHE_VAL(mysql_cv_have_bool,
+[
+AC_LANG_SAVE
+AC_LANG_CPLUSPLUS
+AC_TRY_COMPILE(,[bool b = true;],
+mysql_cv_have_bool=yes,
+mysql_cv_have_bool=no)
+AC_LANG_RESTORE
+])
+AC_MSG_RESULT($mysql_cv_have_bool)
+if test "$mysql_cv_have_bool" = yes; then
+AC_DEFINE(HAVE_BOOL)
+fi
+])dnl
+
+AC_DEFUN(MYSQL_STACK_DIRECTION,
+ [AC_CACHE_CHECK(stack direction for C alloca, ac_cv_c_stack_direction,
+ [AC_TRY_RUN([#include <stdlib.h>
+ int find_stack_direction ()
+ {
+ static char *addr = 0;
+ auto char dummy;
+ if (addr == 0)
+ {
+ addr = &dummy;
+ return find_stack_direction ();
+ }
+ else
+ return (&dummy > addr) ? 1 : -1;
+ }
+ int main ()
+ {
+ exit (find_stack_direction() < 0);
+ }], ac_cv_c_stack_direction=1, ac_cv_c_stack_direction=-1,
+ ac_cv_c_stack_direction=0)])
+ AC_DEFINE_UNQUOTED(STACK_DIRECTION, $ac_cv_c_stack_direction)
+])dnl
+
+AC_DEFUN(MYSQL_FUNC_ALLOCA,
+[
+# Since we have heard that alloca fails on IRIX never define it on a
+# SGI machine
+if test ! "$host_vendor" = "sgi"
+then
+ AC_REQUIRE_CPP()dnl Set CPP; we run AC_EGREP_CPP conditionally.
+ # The Ultrix 4.2 mips builtin alloca declared by alloca.h only works
+ # for constant arguments. Useless!
+ AC_CACHE_CHECK([for working alloca.h], ac_cv_header_alloca_h,
+ [AC_TRY_LINK([#include <alloca.h>], [char *p = alloca(2 * sizeof(int));],
+ ac_cv_header_alloca_h=yes, ac_cv_header_alloca_h=no)])
+ if test "$ac_cv_header_alloca_h" = "yes"
+ then
+ AC_DEFINE(HAVE_ALLOCA)
+ fi
+
+ AC_CACHE_CHECK([for alloca], ac_cv_func_alloca_works,
+ [AC_TRY_LINK([
+ #ifdef __GNUC__
+ # define alloca __builtin_alloca
+ #else
+ # if HAVE_ALLOCA_H
+ # include <alloca.h>
+ # else
+ # ifdef _AIX
+ #pragma alloca
+ # else
+ # ifndef alloca /* predefined by HP cc +Olibcalls */
+ char *alloca ();
+ # endif
+ # endif
+ # endif
+ #endif
+ ], [char *p = (char *) alloca(1);],
+ ac_cv_func_alloca_works=yes, ac_cv_func_alloca_works=no)])
+ if test "$ac_cv_func_alloca_works" = "yes"; then
+ AC_DEFINE(HAVE_ALLOCA)
+ fi
+
+ if test "$ac_cv_func_alloca_works" = "no"; then
+ # The SVR3 libPW and SVR4 libucb both contain incompatible functions
+ # that cause trouble. Some versions do not even contain alloca or
+ # contain a buggy version. If you still want to use their alloca,
+ # use ar to extract alloca.o from them instead of compiling alloca.c.
+ ALLOCA=alloca.o
+ AC_DEFINE(C_ALLOCA)
+
+ AC_CACHE_CHECK(whether alloca needs Cray hooks, ac_cv_os_cray,
+ [AC_EGREP_CPP(webecray,
+ [#if defined(CRAY) && ! defined(CRAY2)
+ webecray
+ #else
+ wenotbecray
+ #endif
+ ], ac_cv_os_cray=yes, ac_cv_os_cray=no)])
+ if test "$ac_cv_os_cray" = "yes"; then
+ for ac_func in _getb67 GETB67 getb67; do
+ AC_CHECK_FUNC($ac_func, [AC_DEFINE_UNQUOTED(CRAY_STACKSEG_END, $ac_func)
+ break])
+ done
+ fi
+ fi
+ AC_SUBST(ALLOCA)dnl
+else
+ AC_MSG_RESULT("Skipped alloca tests")
+fi
+])
+
+AC_DEFUN(MYSQL_CHECK_LONGLONG_TO_FLOAT,
+[
+AC_MSG_CHECKING(if conversion of longlong to float works)
+AC_CACHE_VAL(ac_cv_conv_longlong_to_float,
+[AC_TRY_RUN([#include <stdio.h>
+typedef long long longlong;
+main()
+{
+ longlong ll=1;
+ float f;
+ FILE *file=fopen("conftestval", "w");
+ f = (float) ll;
+ fprintf(file,"%g\n",f);
+ fclose(file);
+ exit (0);
+}], ac_cv_conv_longlong_to_float=`cat conftestval`, ac_cv_conv_longlong_to_float=0, ifelse([$2], , , ac_cv_conv_longlong_to_float=$2))])dnl
+if test "$ac_cv_conv_longlong_to_float" = "1" -o "$ac_cv_conv_longlong_to_float" = "yes"
+then
+ ac_cv_conv_longlong_to_float=yes
+else
+ ac_cv_conv_longlong_to_float=no
+fi
+AC_MSG_RESULT($ac_cv_conv_longlong_to_float)
+])
+
+AC_DEFUN(MYSQL_CHECK_CPU,
+[AC_CACHE_CHECK([if compiler supports optimizations for current cpu],
+mysql_cv_cpu,[
+
+ac_save_CFLAGS="$CFLAGS"
+if test -r /proc/cpuinfo ; then
+ cpuinfo="cat /proc/cpuinfo"
+ cpu_family=`$cpuinfo | grep 'cpu family' | cut -d ':' -f 2 | cut -d ' ' -f 2 | head -1`
+ cpu_vendor=`$cpuinfo | grep 'vendor_id' | cut -d ':' -f 2 | cut -d ' ' -f 2 | head -1`
+fi
+if test "$cpu_vendor" = "AuthenticAMD"; then
+ if test $cpu_family -ge 6; then
+ cpu_set="athlon pentiumpro k5 pentium i486 i386";
+ elif test $cpu_family -eq 5; then
+ cpu_set="k5 pentium i486 i386";
+ elif test $cpu_family -eq 4; then
+ cpu_set="i486 i386"
+ else
+ cpu_set="i386"
+ fi
+elif test "$cpu_vendor" = "GenuineIntel"; then
+ if test $cpu_family -ge 6; then
+ cpu_set="pentiumpro pentium i486 i386";
+ elif test $cpu_family -eq 5; then
+ cpu_set="pentium i486 i386";
+ elif test $cpu_family -eq 4; then
+ cpu_set="i486 i386"
+ else
+ cpu_set="i386"
+ fi
+fi
+
+for ac_arg in $cpu_set;
+do
+ CFLAGS="$ac_save_CFLAGS -mcpu=$ac_arg -march=$ac_arg -DCPU=$ac_arg"
+ AC_TRY_COMPILE([],[int i],mysql_cv_cpu=$ac_arg; break;, mysql_cv_cpu="unknown")
+done
+
+if test "$mysql_cv_cpu" = "unknown"
+then
+ CFLAGS="$ac_save_CFLAGS"
+ AC_MSG_RESULT(none)
+else
+ AC_MSG_RESULT($mysql_cv_cpu)
+fi
+]]))
+
+AC_DEFUN(MYSQL_CHECK_VIO, [
+ AC_ARG_WITH([vio],
+ [ --with-vio Include the Virtual IO support],
+ [vio="$withval"],
+ [vio=no])
+
+ if test "$vio" = "yes"
+ then
+ vio_dir="vio"
+ vio_libs="../vio/libvio.la"
+ AC_DEFINE(HAVE_VIO)
+ else
+ vio_dir=""
+ vio_libs=""
+ fi
+ AC_SUBST([vio_dir])
+ AC_SUBST([vio_libs])
+])
+
+AC_DEFUN(MYSQL_FIND_OPENSSL, [
+ incs="$1"
+ libs="$2"
+ case "$incs---$libs" in
+ ---)
+ for d in /usr/ssl/include /usr/local/ssl/include /usr/include \
+/usr/include/ssl /opt/ssl/include /opt/openssl/include \
+/usr/local/ssl/include /usr/local/include ; do
+ if test -f $d/openssl/ssl.h ; then
+ OPENSSL_INCLUDE=-I$d
+ fi
+ done
+
+ for d in /usr/ssl/lib /usr/local/ssl/lib /usr/lib/openssl \
+/usr/lib /usr/lib64 /opt/ssl/lib /opt/openssl/lib /usr/local/lib/ ; do
+ if test -f $d/libssl.a || test -f $d/libssl.so || test -f $d/libssl.dylib ; then
+ OPENSSL_LIB=$d
+ fi
+ done
+ ;;
+ ---* | *---)
+ AC_MSG_ERROR([if either 'includes' or 'libs' is specified, both must be specified])
+ ;;
+ * )
+ if test -f $incs/openssl/ssl.h ; then
+ OPENSSL_INCLUDE=-I$incs
+ fi
+ if test -f $libs/libssl.a || test -f $libs/libssl.so || test -f $libs/libssl.dylib ; then
+ OPENSSL_LIB=$libs
+ fi
+ ;;
+ esac
+
+ # On RedHat 9 we need kerberos to compile openssl
+ for d in /usr/kerberos/include
+ do
+ if test -f $d/krb5.h ; then
+ OPENSSL_KERBEROS_INCLUDE="$d"
+ fi
+ done
+
+
+ if test -z "$OPENSSL_LIB" -o -z "$OPENSSL_INCLUDE" ; then
+ echo "Could not find an installation of OpenSSL"
+ if test -n "$OPENSSL_LIB" ; then
+ if test "$IS_LINUX" = "true"; then
+ echo "Looks like you've forgotten to install OpenSSL development RPM"
+ fi
+ fi
+ exit 1
+ fi
+
+])
+
+AC_DEFUN(MYSQL_CHECK_OPENSSL, [
+AC_MSG_CHECKING(for OpenSSL)
+ AC_ARG_WITH([openssl],
+ [ --with-openssl Include the OpenSSL support],
+ [openssl="$withval"],
+ [openssl=no])
+
+ AC_ARG_WITH([openssl-includes],
+ [
+ --with-openssl-includes=DIR
+ Find OpenSSL headers in DIR],
+ [openssl_includes="$withval"],
+ [openssl_includes=""])
+
+ AC_ARG_WITH([openssl-libs],
+ [
+ --with-openssl-libs=DIR
+ Find OpenSSL libraries in DIR],
+ [openssl_libs="$withval"],
+ [openssl_libs=""])
+
+ if test "$openssl" = "yes"
+ then
+ MYSQL_FIND_OPENSSL([$openssl_includes], [$openssl_libs])
+ #force VIO use
+ vio_dir="vio"
+ vio_libs="../vio/libvio.la"
+ AC_DEFINE(HAVE_VIO)
+ AC_MSG_RESULT(yes)
+ openssl_libs="-L$OPENSSL_LIB -lssl -lcrypto"
+ # Don't set openssl_includes to /usr/include as this gives us a lot of
+ # compiler warnings when using gcc 3.x
+ openssl_includes=""
+ if test "$OPENSSL_INCLUDE" != "-I/usr/include"
+ then
+ openssl_includes="$OPENSSL_INCLUDE"
+ fi
+ if test "$OPENSSL_KERBEROS_INCLUDE"
+ then
+ openssl_includes="$openssl_includes -I$OPENSSL_KERBEROS_INCLUDE"
+ fi
+ AC_DEFINE(HAVE_OPENSSL)
+
+ # openssl-devel-0.9.6 requires dlopen() and we can't link staticly
+ # on many platforms (We should actually test this here, but it's quite
+ # hard) to do as we are doing libtool for linking.
+ using_static=""
+ case "$CLIENT_EXTRA_LDFLAGS $MYSQLD_EXTRA_LDFLAGS" in
+ *-all-static*) using_static="yes" ;;
+ esac
+ if test "$using_static" = "yes"
+ then
+ echo "You can't use the --all-static link option when using openssl."
+ exit 1
+ fi
+ NON_THREADED_CLIENT_LIBS="$NON_THREADED_CLIENT_LIBS $openssl_libs"
+ else
+ AC_MSG_RESULT(no)
+ fi
+ AC_SUBST(openssl_libs)
+ AC_SUBST(openssl_includes)
+])
+
+
+AC_DEFUN(MYSQL_CHECK_MYSQLFS, [
+ AC_ARG_WITH([mysqlfs],
+ [
+ --with-mysqlfs Include the corba-based MySQL file system],
+ [mysqlfs="$withval"],
+ [mysqlfs=no])
+
+dnl Call MYSQL_CHECK_ORBIT even if mysqlfs == no, so that @orbit_*@
+dnl get substituted.
+ MYSQL_CHECK_ORBIT
+
+ AC_MSG_CHECKING(if we should build MySQLFS)
+ fs_dirs=""
+ if test "$mysqlfs" = "yes"
+ then
+ if test -n "$orbit_exec_prefix"
+ then
+ fs_dirs=fs
+ AC_MSG_RESULT([yes])
+ else
+ AC_MSG_RESULT(disabled because ORBIT, the CORBA ORB, was not found)
+ fi
+ else
+ AC_MSG_RESULT([no])
+ fi
+ AC_SUBST([fs_dirs])
+])
+
+AC_DEFUN(MYSQL_CHECK_ORBIT, [
+AC_MSG_CHECKING(for ORBit)
+orbit_config_path=`which orbit-config`
+if test -n "$orbit_config_path" -a $? = 0
+then
+ orbit_exec_prefix=`orbit-config --exec-prefix`
+ orbit_includes=`orbit-config --cflags server`
+ orbit_libs=`orbit-config --libs server`
+ orbit_idl="$orbit_exec_prefix/bin/orbit-idl"
+ AC_MSG_RESULT(found!)
+ AC_DEFINE(HAVE_ORBIT)
+else
+ orbit_exec_prefix=
+ orbit_includes=
+ orbit_libs=
+ orbit_idl=
+ AC_MSG_RESULT(not found)
+fi
+AC_SUBST(orbit_includes)
+AC_SUBST(orbit_libs)
+AC_SUBST(orbit_idl)
+])
+
+AC_DEFUN([MYSQL_CHECK_ISAM], [
+ AC_ARG_WITH([isam], [
+ --with-isam Enable the ISAM table type],
+ [with_isam="$withval"],
+ [with_isam=no])
+
+ isam_libs=
+ if test X"$with_isam" = X"yes"
+ then
+ AC_DEFINE(HAVE_ISAM)
+ isam_libs="\$(top_builddir)/isam/libnisam.a\
+ \$(top_builddir)/merge/libmerge.a"
+ fi
+ AC_SUBST(isam_libs)
+])
+
+
+dnl ---------------------------------------------------------------------------
+dnl Macro: MYSQL_CHECK_BDB
+dnl Sets HAVE_BERKELEY_DB if inst library is found
+dnl Makes sure db version is correct.
+dnl Looks in $srcdir for Berkeley distribution if not told otherwise
+dnl ---------------------------------------------------------------------------
+
+AC_DEFUN([MYSQL_CHECK_BDB], [
+ AC_ARG_WITH([berkeley-db],
+ [
+ --with-berkeley-db[=DIR]
+ Use BerkeleyDB located in DIR],
+ [bdb="$withval"],
+ [bdb=no])
+
+ AC_ARG_WITH([berkeley-db-includes],
+ [
+ --with-berkeley-db-includes=DIR
+ Find Berkeley DB headers in DIR],
+ [bdb_includes="$withval"],
+ [bdb_includes=default])
+
+ AC_ARG_WITH([berkeley-db-libs],
+ [
+ --with-berkeley-db-libs=DIR
+ Find Berkeley DB libraries in DIR],
+ [bdb_libs="$withval"],
+ [bdb_libs=default])
+
+ AC_MSG_CHECKING([for BerkeleyDB])
+
+dnl SORT OUT THE SUPPLIED ARGUMENTS TO DETERMINE WHAT TO DO
+dnl echo "DBG1: bdb='$bdb'; incl='$bdb_includes'; lib='$bdb_libs'"
+ have_berkeley_db=no
+ case "$bdb" in
+ no )
+ mode=no
+ AC_MSG_RESULT([no])
+ ;;
+ yes | default )
+ case "$bdb_includes---$bdb_libs" in
+ default---default )
+ mode=search-$bdb
+ AC_MSG_RESULT([searching...])
+ ;;
+ default---* | *---default | yes---* | *---yes )
+ AC_MSG_ERROR([if either 'includes' or 'libs' is specified, both must be specified])
+ ;;
+ * )
+ mode=supplied-two
+ AC_MSG_RESULT([supplied])
+ ;;
+ esac
+ ;;
+ * )
+ mode=supplied-one
+ AC_MSG_RESULT([supplied])
+ ;;
+ esac
+
+dnl echo "DBG2: [$mode] bdb='$bdb'; incl='$bdb_includes'; lib='$bdb_libs'"
+
+ case $mode in
+ no )
+ bdb_includes=
+ bdb_libs=
+ bdb_libs_with_path=
+ ;;
+ supplied-two )
+ MYSQL_CHECK_INSTALLED_BDB([$bdb_includes], [$bdb_libs])
+ case $bdb_dir_ok in
+ installed ) mode=yes ;;
+ * ) AC_MSG_ERROR([didn't find valid BerkeleyDB: $bdb_dir_ok]) ;;
+ esac
+ ;;
+ supplied-one )
+ MYSQL_CHECK_BDB_DIR([$bdb])
+ case $bdb_dir_ok in
+ source ) mode=compile ;;
+ installed ) mode=yes ;;
+ * ) AC_MSG_ERROR([didn't find valid BerkeleyDB: $bdb_dir_ok]) ;;
+ esac
+ ;;
+ search-* )
+ MYSQL_SEARCH_FOR_BDB
+ case $bdb_dir_ok in
+ source ) mode=compile ;;
+ installed ) mode=yes ;;
+ * )
+ # not found
+ case $mode in
+ *-yes ) AC_MSG_ERROR([no suitable BerkeleyDB found]) ;;
+ * ) mode=no ;;
+ esac
+ bdb_includes=
+ bdb_libs=
+ bdb_libs_with_path=
+ ;;
+ esac
+ ;;
+ *)
+ AC_MSG_ERROR([impossible case condition '$mode': please report this to bugs@lists.mysql.com])
+ ;;
+ esac
+
+dnl echo "DBG3: [$mode] bdb='$bdb'; incl='$bdb_includes'; lib='$bdb_libs'"
+ case $mode in
+ no )
+ AC_MSG_RESULT([Not using Berkeley DB])
+ ;;
+ yes )
+ have_berkeley_db="yes"
+ AC_MSG_RESULT([Using Berkeley DB in '$bdb_includes'])
+ ;;
+ compile )
+ have_berkeley_db="$bdb"
+ AC_MSG_RESULT([Compiling Berekeley DB in '$have_berkeley_db'])
+ ;;
+ * )
+ AC_MSG_ERROR([impossible case condition '$mode': please report this to bugs@lists.mysql.com])
+ ;;
+ esac
+
+ AC_SUBST(bdb_includes)
+ AC_SUBST(bdb_libs)
+ AC_SUBST(bdb_libs_with_path)
+])
+
+AC_DEFUN([MYSQL_CHECK_INSTALLED_BDB], [
+dnl echo ["MYSQL_CHECK_INSTALLED_BDB ($1) ($2)"]
+ inc="$1"
+ lib="$2"
+ if test -f "$inc/db.h"
+ then
+ MYSQL_CHECK_BDB_VERSION([$inc/db.h],
+ [.*#define[ ]*], [[ ][ ]*])
+
+ if test X"$bdb_version_ok" = Xyes; then
+ save_LDFLAGS="$LDFLAGS"
+ LDFLAGS="-L$lib $LDFLAGS"
+ AC_CHECK_LIB(db,db_env_create, [
+ bdb_dir_ok=installed
+ MYSQL_TOP_BUILDDIR([inc])
+ MYSQL_TOP_BUILDDIR([lib])
+ bdb_includes="-I$inc"
+ bdb_libs="-L$lib -ldb"
+ bdb_libs_with_path="$lib/libdb.a"
+ ])
+ LDFLAGS="$save_LDFLAGS"
+ else
+ bdb_dir_ok="$bdb_version_ok"
+ fi
+ else
+ bdb_dir_ok="no db.h file in '$inc'"
+ fi
+])
+
+AC_DEFUN([MYSQL_CHECK_BDB_DIR], [
+dnl ([$bdb])
+dnl echo ["MYSQL_CHECK_BDB_DIR ($1)"]
+ dir="$1"
+
+ MYSQL_CHECK_INSTALLED_BDB([$dir/include], [$dir/lib])
+
+ if test X"$bdb_dir_ok" != Xinstalled; then
+ # test to see if it's a source dir
+ rel="$dir/dist/RELEASE"
+ if test -f "$rel"; then
+ MYSQL_CHECK_BDB_VERSION([$rel], [], [=])
+ if test X"$bdb_version_ok" = Xyes; then
+ bdb_dir_ok=source
+ bdb="$dir"
+ MYSQL_TOP_BUILDDIR([dir])
+ bdb_includes="-I$dir/build_unix"
+ bdb_libs="-L$dir/build_unix -ldb"
+ bdb_libs_with_path="$dir/build_unix/libdb.a"
+ else
+ bdb_dir_ok="$bdb_version_ok"
+ fi
+ else
+ bdb_dir_ok="'$dir' doesn't look like a BDB directory ($bdb_dir_ok)"
+ fi
+ fi
+])
+
+AC_DEFUN([MYSQL_SEARCH_FOR_BDB], [
+dnl echo ["MYSQL_SEARCH_FOR_BDB"]
+ bdb_dir_ok="no BerkeleyDB found"
+
+ for test_dir in $srcdir/bdb $srcdir/db-*.*.* /usr/local/BerkeleyDB*; do
+dnl echo "-----------> Looking at ($test_dir; `cd $test_dir && pwd`)"
+ MYSQL_CHECK_BDB_DIR([$test_dir])
+ if test X"$bdb_dir_ok" = Xsource || test X"$bdb_dir_ok" = Xinstalled; then
+dnl echo "-----------> Found it ($bdb), ($srcdir)"
+dnl This is needed so that 'make distcheck' works properly (VPATH build).
+dnl VPATH build won't work if bdb is not under the source tree; but in
+dnl that case, hopefully people will just make and install inside the
+dnl tree, or install BDB first, and then use the installed version.
+ case "$bdb" in
+ "$srcdir/"* ) bdb=`echo "$bdb" | sed -e "s,^$srcdir/,,"` ;;
+ esac
+ break
+ fi
+ done
+])
+
+dnl MYSQL_CHECK_BDB_VERSION takes 3 arguments:
+dnl 1) the file to look in
+dnl 2) the search pattern before DB_VERSION_XXX
+dnl 3) the search pattern between DB_VERSION_XXX and the number
+dnl It assumes that the number is the last thing on the line
+AC_DEFUN([MYSQL_CHECK_BDB_VERSION], [
+ db_major=`sed -e '/^[$2]DB_VERSION_MAJOR[$3]/ !d' -e 's///' [$1]`
+ db_minor=`sed -e '/^[$2]DB_VERSION_MINOR[$3]/ !d' -e 's///' [$1]`
+ db_patch=`sed -e '/^[$2]DB_VERSION_PATCH[$3]/ !d' -e 's///' [$1]`
+ test -z "$db_major" && db_major=0
+ test -z "$db_minor" && db_minor=0
+ test -z "$db_patch" && db_patch=0
+
+ # This is ugly, but about as good as it can get
+# mysql_bdb=
+# if test $db_major -eq 3 && test $db_minor -eq 2 && test $db_patch -eq 3
+# then
+# mysql_bdb=h
+# elif test $db_major -eq 3 && test $db_minor -eq 2 && test $db_patch -eq 9
+# then
+# want_bdb_version="3.2.9a" # hopefully this will stay up-to-date
+# mysql_bdb=a
+# fi
+
+dnl RAM:
+want_bdb_version="4.1.24"
+bdb_version_ok=yes
+
+# if test -n "$mysql_bdb" && \
+# grep "DB_VERSION_STRING.*:.*$mysql_bdb: " [$1] > /dev/null
+# then
+# bdb_version_ok=yes
+# else
+# bdb_version_ok="invalid version $db_major.$db_minor.$db_patch"
+# bdb_version_ok="$bdb_version_ok (must be version 3.2.3h or $want_bdb_version)"
+# fi
+])
+
+AC_DEFUN([MYSQL_TOP_BUILDDIR], [
+ case "$[$1]" in
+ /* ) ;; # don't do anything with an absolute path
+ "$srcdir"/* )
+ # If BDB is under the source directory, we need to look under the
+ # build directory for bdb/build_unix.
+ # NOTE: I'm being lazy, and assuming the user did not specify
+ # something like --with-berkeley-db=bdb (it would be missing "./").
+ [$1]="\$(top_builddir)/"`echo "$[$1]" | sed -e "s,^$srcdir/,,"`
+ ;;
+ * )
+ AC_MSG_ERROR([The BDB directory must be directly under the MySQL source directory, or be specified using the full path. ('$srcdir'; '$[$1]')])
+ ;;
+ esac
+ if test X"$[$1]" != "/"
+ then
+ [$1]=`echo $[$1] | sed -e 's,/$,,'`
+ fi
+])
+
+dnl ---------------------------------------------------------------------------
+dnl END OF MYSQL_CHECK_BDB SECTION
+dnl ---------------------------------------------------------------------------
+
+dnl ---------------------------------------------------------------------------
+dnl Macro: MYSQL_CHECK_INNODB
+dnl Sets HAVE_INNOBASE_DB if --with-innodb is used
+dnl ---------------------------------------------------------------------------
+
+AC_DEFUN([MYSQL_CHECK_INNODB], [
+ AC_ARG_WITH([innodb],
+ [
+ --without-innodb Do not include the InnoDB table handler],
+ [innodb="$withval"],
+ [innodb=yes])
+
+ AC_MSG_CHECKING([for Innodb])
+
+ have_innodb=no
+ innodb_includes=
+ innodb_libs=
+ case "$innodb" in
+ yes )
+ AC_MSG_RESULT([Using Innodb])
+ AC_DEFINE(HAVE_INNOBASE_DB)
+ have_innodb="yes"
+ innodb_includes="-I../innobase/include"
+ innodb_system_libs=""
+dnl Some libs are listed several times, in order for gcc to sort out
+dnl circular references.
+ innodb_libs="\
+ \$(top_builddir)/innobase/usr/libusr.a\
+ \$(top_builddir)/innobase/srv/libsrv.a\
+ \$(top_builddir)/innobase/dict/libdict.a\
+ \$(top_builddir)/innobase/que/libque.a\
+ \$(top_builddir)/innobase/srv/libsrv.a\
+ \$(top_builddir)/innobase/ibuf/libibuf.a\
+ \$(top_builddir)/innobase/row/librow.a\
+ \$(top_builddir)/innobase/pars/libpars.a\
+ \$(top_builddir)/innobase/btr/libbtr.a\
+ \$(top_builddir)/innobase/trx/libtrx.a\
+ \$(top_builddir)/innobase/read/libread.a\
+ \$(top_builddir)/innobase/usr/libusr.a\
+ \$(top_builddir)/innobase/buf/libbuf.a\
+ \$(top_builddir)/innobase/ibuf/libibuf.a\
+ \$(top_builddir)/innobase/eval/libeval.a\
+ \$(top_builddir)/innobase/log/liblog.a\
+ \$(top_builddir)/innobase/fsp/libfsp.a\
+ \$(top_builddir)/innobase/fut/libfut.a\
+ \$(top_builddir)/innobase/fil/libfil.a\
+ \$(top_builddir)/innobase/lock/liblock.a\
+ \$(top_builddir)/innobase/mtr/libmtr.a\
+ \$(top_builddir)/innobase/page/libpage.a\
+ \$(top_builddir)/innobase/rem/librem.a\
+ \$(top_builddir)/innobase/thr/libthr.a\
+ \$(top_builddir)/innobase/sync/libsync.a\
+ \$(top_builddir)/innobase/data/libdata.a\
+ \$(top_builddir)/innobase/mach/libmach.a\
+ \$(top_builddir)/innobase/ha/libha.a\
+ \$(top_builddir)/innobase/dyn/libdyn.a\
+ \$(top_builddir)/innobase/mem/libmem.a\
+ \$(top_builddir)/innobase/sync/libsync.a\
+ \$(top_builddir)/innobase/ut/libut.a\
+ \$(top_builddir)/innobase/os/libos.a\
+ \$(top_builddir)/innobase/ut/libut.a"
+
+ AC_CHECK_LIB(rt, aio_read, [innodb_system_libs="-lrt"])
+ ;;
+ * )
+ AC_MSG_RESULT([Not using Innodb])
+ ;;
+ esac
+
+ AC_SUBST(innodb_includes)
+ AC_SUBST(innodb_libs)
+ AC_SUBST(innodb_system_libs)
+])
+
+dnl ---------------------------------------------------------------------------
+dnl END OF MYSQL_CHECK_INNODB SECTION
+dnl ---------------------------------------------------------------------------
+
+dnl ---------------------------------------------------------------------------
+dnl Macro: MYSQL_CHECK_NDBCLUSTER
+dnl Sets HAVE_NDBCLUSTER_DB if --with-ndbcluster is used
+dnl ---------------------------------------------------------------------------
+
+AC_DEFUN([MYSQL_CHECK_NDBCLUSTER], [
+ AC_ARG_WITH([ndbcluster],
+ [
+ --without-ndbcluster Do not include the Ndb Cluster table handler],
+ [ndbcluster="$withval"],
+ [ndbcluster=yes])
+
+ AC_MSG_CHECKING([for Ndb Cluster])
+
+ have_ndbcluster=no
+ ndbcluster_includes=
+ ndbcluster_libs=
+ case "$ndbcluster" in
+ yes )
+ AC_MSG_RESULT([Using Ndb Cluster])
+ AC_DEFINE(HAVE_NDBCLUSTER_DB)
+ have_ndbcluster="yes"
+ ndbcluster_includes="-I../ndb/include -I../ndb/include/ndbapi"
+ ndbcluster_libs="\$(top_builddir)/ndb/lib/libNDB_API.a"
+ ndbcluster_system_libs=""
+ esac
+
+ AC_SUBST(ndbcluster_includes)
+ AC_SUBST(ndbcluster_libs)
+ AC_SUBST(ndbcluster_system_libs)
+])
+
+dnl ---------------------------------------------------------------------------
+dnl END OF MYSQL_CHECK_NDBCLUSTER SECTION
+dnl ---------------------------------------------------------------------------
+
+dnl By default, many hosts won't let programs access large files;
+dnl one must use special compiler options to get large-file access to work.
+dnl For more details about this brain damage please see:
+dnl http://www.sas.com/standards/large.file/x_open.20Mar96.html
+
+dnl Written by Paul Eggert <eggert@twinsun.com>.
+
+dnl Internal subroutine of AC_SYS_LARGEFILE.
+dnl AC_SYS_LARGEFILE_FLAGS(FLAGSNAME)
+AC_DEFUN(AC_SYS_LARGEFILE_FLAGS,
+ [AC_CACHE_CHECK([for $1 value to request large file support],
+ ac_cv_sys_largefile_$1,
+ [if ($GETCONF LFS_$1) >conftest.1 2>conftest.2 && test ! -s conftest.2
+ then
+ ac_cv_sys_largefile_$1=`cat conftest.1`
+ else
+ ac_cv_sys_largefile_$1=no
+ ifelse($1, CFLAGS,
+ [case "$host_os" in
+ # HP-UX 10.20 requires -D__STDC_EXT__ with gcc 2.95.1.
+changequote(, )dnl
+ hpux10.[2-9][0-9]* | hpux1[1-9]* | hpux[2-9][0-9]*)
+changequote([, ])dnl
+ if test "$GCC" = yes; then
+ case `$CC --version 2>/dev/null` in
+ 2.95.*) ac_cv_sys_largefile_CFLAGS=-D__STDC_EXT__ ;;
+ esac
+ fi
+ ;;
+ # IRIX 6.2 and later require cc -n32.
+changequote(, )dnl
+ irix6.[2-9]* | irix6.1[0-9]* | irix[7-9].* | irix[1-9][0-9]*)
+changequote([, ])dnl
+ if test "$GCC" != yes; then
+ ac_cv_sys_largefile_CFLAGS=-n32
+ fi
+ esac
+ if test "$ac_cv_sys_largefile_CFLAGS" != no; then
+ ac_save_CC="$CC"
+ CC="$CC $ac_cv_sys_largefile_CFLAGS"
+ AC_TRY_LINK(, , , ac_cv_sys_largefile_CFLAGS=no)
+ CC="$ac_save_CC"
+ fi])
+ fi
+ rm -f conftest*])])
+
+dnl Internal subroutine of AC_SYS_LARGEFILE.
+dnl AC_SYS_LARGEFILE_SPACE_APPEND(VAR, VAL)
+AC_DEFUN(AC_SYS_LARGEFILE_SPACE_APPEND,
+ [case $2 in
+ no) ;;
+ ?*)
+ case "[$]$1" in
+ '') $1=$2 ;;
+ *) $1=[$]$1' '$2 ;;
+ esac ;;
+ esac])
+
+dnl Internal subroutine of AC_SYS_LARGEFILE.
+dnl AC_SYS_LARGEFILE_MACRO_VALUE(C-MACRO, CACHE-VAR, COMMENT, CODE-TO-SET-DEFAULT)
+AC_DEFUN(AC_SYS_LARGEFILE_MACRO_VALUE,
+ [AC_CACHE_CHECK([for $1], $2,
+ [$2=no
+changequote(, )dnl
+ for ac_flag in $ac_cv_sys_largefile_CFLAGS no; do
+ case "$ac_flag" in
+ -D$1)
+ $2=1 ;;
+ -D$1=*)
+ $2=`expr " $ac_flag" : '[^=]*=\(.*\)'` ;;
+ esac
+ done
+ $4
+changequote([, ])dnl
+ ])
+ if test "[$]$2" != no; then
+ AC_DEFINE_UNQUOTED([$1], [$]$2, [$3])
+ fi])
+
+AC_DEFUN(MYSQL_SYS_LARGEFILE,
+ [AC_REQUIRE([AC_CANONICAL_HOST])
+ AC_ARG_ENABLE(largefile,
+ [ --disable-largefile Omit support for large files])
+ if test "$enable_largefile" != no; then
+ AC_CHECK_TOOL(GETCONF, getconf)
+ AC_SYS_LARGEFILE_FLAGS(CFLAGS)
+ AC_SYS_LARGEFILE_FLAGS(LDFLAGS)
+ AC_SYS_LARGEFILE_FLAGS(LIBS)
+
+ for ac_flag in $ac_cv_sys_largefile_CFLAGS no; do
+ case "$ac_flag" in
+ no) ;;
+ -D_FILE_OFFSET_BITS=*) ;;
+ -D_LARGEFILE_SOURCE | -D_LARGEFILE_SOURCE=*) ;;
+ -D_LARGE_FILES | -D_LARGE_FILES=*) ;;
+ -D?* | -I?*)
+ AC_SYS_LARGEFILE_SPACE_APPEND(CPPFLAGS, "$ac_flag") ;;
+ *)
+ AC_SYS_LARGEFILE_SPACE_APPEND(CFLAGS, "$ac_flag") ;;
+ esac
+ done
+ AC_SYS_LARGEFILE_SPACE_APPEND(LDFLAGS, "$ac_cv_sys_largefile_LDFLAGS")
+ AC_SYS_LARGEFILE_SPACE_APPEND(LIBS, "$ac_cv_sys_largefile_LIBS")
+
+ AC_SYS_LARGEFILE_MACRO_VALUE(_FILE_OFFSET_BITS,
+ ac_cv_sys_file_offset_bits,
+ [Number of bits in a file offset, on hosts where this is settable.],
+ [case "$host_os" in
+ # HP-UX 10.20 and later
+ hpux10.[2-9][0-9]* | hpux1[1-9]* | hpux[2-9][0-9]*)
+ ac_cv_sys_file_offset_bits=64 ;;
+ # We can't declare _FILE_OFFSET_BITS here as this will cause
+ # compile errors as AC_PROG_CC adds include files in confdefs.h
+ # We solve this (until autoconf is fixed) by instead declaring it
+ # as define instead
+ solaris2.[8,9])
+ CFLAGS="$CFLAGS -D_FILE_OFFSET_BITS=64"
+ CXXFLAGS="$CXXFLAGS -D_FILE_OFFSET_BITS=64"
+ ac_cv_sys_file_offset_bits=no ;;
+ esac])
+ AC_SYS_LARGEFILE_MACRO_VALUE(_LARGEFILE_SOURCE,
+ ac_cv_sys_largefile_source,
+ [Define to make fseeko etc. visible, on some hosts.],
+ [case "$host_os" in
+ # HP-UX 10.20 and later
+ hpux10.[2-9][0-9]* | hpux1[1-9]* | hpux[2-9][0-9]*)
+ ac_cv_sys_largefile_source=1 ;;
+ esac])
+ AC_SYS_LARGEFILE_MACRO_VALUE(_LARGE_FILES,
+ ac_cv_sys_large_files,
+ [Define for large files, on AIX-style hosts.],
+ [case "$host_os" in
+ # AIX 4.2 and later
+ aix4.[2-9]* | aix4.1[0-9]* | aix[5-9].* | aix[1-9][0-9]*)
+ ac_cv_sys_large_files=1 ;;
+ esac])
+ fi
+ ])
+
+
+# Local version of _AC_PROG_CXX_EXIT_DECLARATION that does not
+# include #stdlib.h as default as this breaks things on Solaris
+# (Conflicts with pthreads and big file handling)
+
+m4_define([_AC_PROG_CXX_EXIT_DECLARATION],
+[for ac_declaration in \
+ ''\
+ 'extern "C" void std::exit (int) throw (); using std::exit;' \
+ 'extern "C" void std::exit (int); using std::exit;' \
+ 'extern "C" void exit (int) throw ();' \
+ 'extern "C" void exit (int);' \
+ 'void exit (int);' \
+ '#include <stdlib.h>'
+do
+ _AC_COMPILE_IFELSE([AC_LANG_PROGRAM([@%:@include <stdlib.h>
+$ac_declaration],
+ [exit (42);])],
+ [],
+ [continue])
+ _AC_COMPILE_IFELSE([AC_LANG_PROGRAM([$ac_declaration],
+ [exit (42);])],
+ [break])
+done
+rm -f conftest*
+if test -n "$ac_declaration"; then
+ echo '#ifdef __cplusplus' >>confdefs.h
+ echo $ac_declaration >>confdefs.h
+ echo '#endif' >>confdefs.h
+fi
+])# _AC_PROG_CXX_EXIT_DECLARATION
+
+dnl ---------------------------------------------------------------------------
+
diff --git a/ndb/config/configure.in b/ndb/config/configure.in
new file mode 100644
index 00000000000..4fa5ccdb672
--- /dev/null
+++ b/ndb/config/configure.in
@@ -0,0 +1,2085 @@
+dnl -*- ksh -*-
+dnl Process this file with autoconf to produce a configure script.
+
+AC_INIT(../../sql/mysqld.cc)
+AC_CANONICAL_SYSTEM
+# The Docs Makefile.am parses this line!
+AM_INIT_AUTOMAKE(mysql, 4.1.2-3.4.3-alpha)
+AM_CONFIG_HEADER(config.h)
+
+PROTOCOL_VERSION=10
+DOT_FRM_VERSION=6
+# See the libtool docs for information on how to do shared lib versions.
+SHARED_LIB_VERSION=14:0:0
+
+# Set all version vars based on $VERSION. How do we do this more elegant ?
+# Remember that regexps needs to quote [ and ] since this is run through m4
+MYSQL_NO_DASH_VERSION=`echo $VERSION | sed -e "s|[[a-z]]*-.*$||"`
+MYSQL_BASE_VERSION=`echo $MYSQL_NO_DASH_VERSION | sed -e "s|\.[[^.]]*$||"`
+MYSQL_VERSION_ID=`echo $MYSQL_NO_DASH_VERSION. | sed -e 's/\./ /g; s/ \([[0-9]]\) / 0\\1 /g; s/ //g'`
+
+# The port should be constant for a LONG time
+MYSQL_TCP_PORT_DEFAULT=3306
+MYSQL_UNIX_ADDR_DEFAULT="/tmp/mysql.sock"
+
+#####
+#####
+
+AC_SUBST(MYSQL_NO_DASH_VERSION)
+AC_SUBST(MYSQL_BASE_VERSION)
+AC_SUBST(MYSQL_VERSION_ID)
+AC_SUBST(PROTOCOL_VERSION)
+AC_DEFINE_UNQUOTED(PROTOCOL_VERSION, $PROTOCOL_VERSION)
+AC_SUBST(DOT_FRM_VERSION)
+AC_DEFINE_UNQUOTED(DOT_FRM_VERSION, $DOT_FRM_VERSION)
+AC_SUBST(SHARED_LIB_VERSION)
+
+# Canonicalize the configuration name.
+SYSTEM_TYPE="$host_vendor-$host_os"
+MACHINE_TYPE="$host_cpu"
+AC_SUBST(SYSTEM_TYPE)
+AC_DEFINE_UNQUOTED(SYSTEM_TYPE, "$SYSTEM_TYPE")
+AC_SUBST(MACHINE_TYPE)
+AC_DEFINE_UNQUOTED(MACHINE_TYPE, "$MACHINE_TYPE")
+
+# Detect intel x86 like processor
+BASE_MACHINE_TYPE=$MACHINE_TYPE
+case $MACHINE_TYPE in
+ i?86) BASE_MACHINE_TYPE=i386 ;;
+esac
+
+# Save some variables and the command line options for mysqlbug
+SAVE_ASFLAGS="$ASFLAGS"
+SAVE_CFLAGS="$CFLAGS"
+SAVE_CXXFLAGS="$CXXFLAGS"
+SAVE_LDFLAGS="$LDFLAGS"
+SAVE_CXXLDFLAGS="$CXXLDFLAGS"
+CONF_COMMAND="$0 $ac_configure_args"
+AC_SUBST(CONF_COMMAND)
+AC_SUBST(SAVE_ASFLAGS)
+AC_SUBST(SAVE_CFLAGS)
+AC_SUBST(SAVE_CXXFLAGS)
+AC_SUBST(SAVE_LDFLAGS)
+AC_SUBST(SAVE_CXXLDFLAGS)
+AC_SUBST(CXXLDFLAGS)
+
+AC_PREREQ(2.12)dnl Minimum Autoconf version required.
+
+AM_MAINTAINER_MODE
+#AC_ARG_PROGRAM # Automaticly invoked by AM_INIT_AUTOMAKE
+AM_SANITY_CHECK
+# This is needed is SUBDIRS is set
+AC_PROG_MAKE_SET
+
+# This is need before AC_PROG_CC
+#
+
+if test "x${CFLAGS-}" = x ; then
+ cflags_is_set=no
+else
+ cflags_is_set=yes
+fi
+
+if test "x${CPPFLAGS-}" = x ; then
+ cppflags_is_set=no
+else
+ cppflags_is_set=yes
+fi
+
+if test "x${LDFLAGS-}" = x ; then
+ ldflags_is_set=no
+else
+ ldflags_is_set=yes
+fi
+
+# The following hack should ensure that configure doesn't add optimizing
+# or debugging flags to CFLAGS or CXXFLAGS
+CFLAGS="$CFLAGS "
+CXXFLAGS="$CXXFLAGS "
+
+dnl Checks for programs.
+AC_PROG_AWK
+AC_PROG_CC
+AC_PROG_CXX
+AC_PROG_CPP
+
+# Print version of CC and CXX compiler (if they support --version)
+case $SYSTEM_TYPE in
+ *netware*)
+CC_VERSION=`$CC -version | grep -i version`
+ ;;
+ *)
+CC_VERSION=`$CC --version | sed 1q`
+ ;;
+esac
+if test $? -eq "0"
+then
+ AC_MSG_CHECKING("C Compiler version");
+ AC_MSG_RESULT("$CC $CC_VERSION")
+else
+CC_VERSION=""
+fi
+case $SYSTEM_TYPE in
+ *netware*)
+CXX_VERSION=`$CXX -version | grep -i version`
+ ;;
+ *)
+CXX_VERSION=`$CXX --version | sed 1q`
+ ;;
+esac
+if test $? -eq "0"
+then
+ AC_MSG_CHECKING("C++ compiler version");
+ AC_MSG_RESULT("$CXX $CXX_VERSION")
+else
+CXX_VERSION=""
+fi
+AC_SUBST(CXX_VERSION)
+AC_SUBST(CC_VERSION)
+
+# Fix for sgi gcc / sgiCC which tries to emulate gcc
+if test "$CC" = "sgicc"
+then
+ ac_cv_prog_gcc="no"
+fi
+if test "$CXX" = "sgi++"
+then
+ GXX="no"
+fi
+
+if test "$ac_cv_prog_gcc" = "yes"
+then
+ AS="$CC -c"
+ AC_SUBST(AS)
+else
+ AC_PATH_PROG(AS, as, as)
+fi
+# Still need ranlib for readline; local static use only so no libtool.
+AC_PROG_RANLIB
+# We use libtool
+#AC_LIBTOOL_WIN32_DLL
+AC_PROG_LIBTOOL
+
+# Ensure that we have --preserve-dup-deps defines, otherwise we get link
+# problems of 'mysql' with CXX=g++
+LIBTOOL="$LIBTOOL --preserve-dup-deps"
+AC_SUBST(LIBTOOL)dnl
+
+#AC_LIBTOOL_DLOPEN AC_LIBTOOL_WIN32_DLL AC_DISABLE_FAST_INSTALL AC_DISABLE_SHARED AC_DISABLE_STATIC
+
+# AC_PROG_INSTALL
+AC_PROG_INSTALL
+test -z "$INSTALL_SCRIPT" && INSTALL_SCRIPT='${INSTALL_PROGRAM}'
+
+# Not critical since the generated file is distributed
+AC_PROG_YACC
+AC_CHECK_PROG(PDFMANUAL, pdftex, manual.pdf)
+AC_CHECK_PROG(DVIS, tex, manual.dvi)
+
+AC_MSG_CHECKING("return type of sprintf")
+
+#check the return type of sprintf
+case $SYSTEM_TYPE in
+ *netware*)
+ AC_DEFINE(SPRINTF_RETURNS_INT) AC_MSG_RESULT("int")
+ ;;
+ *)
+AC_TRY_RUN([
+ int main()
+ {
+ char* s = "hello";
+ char buf[6];
+ if((int)sprintf(buf, s) == strlen(s))
+ return 0;
+
+ return -1;
+ }
+ ],
+AC_DEFINE(SPRINTF_RETURNS_INT) AC_MSG_RESULT("int"),
+ AC_TRY_RUN([
+ int main()
+ {
+ char* s = "hello";
+ char buf[6];
+ if((char*)sprintf(buf,s) == buf + strlen(s))
+ return 0;
+ return -1;
+ }
+], AC_DEFINE(SPRINTF_RETURNS_PTR) AC_MSG_RESULT("ptr"),
+ AC_DEFINE(SPRINTF_RETURNS_GARBAGE) AC_MSG_RESULT("garbage")))
+ ;;
+esac
+
+
+# option, cache_name, variable,
+# code to execute if yes, code to exectute if fail
+AC_DEFUN(AC_SYS_COMPILER_FLAG,
+[
+ AC_MSG_CHECKING($1)
+ OLD_CFLAGS="[$]CFLAGS"
+ AC_CACHE_VAL(mysql_cv_option_$2,
+ [
+ CFLAGS="[$]OLD_CFLAGS $1"
+ AC_TRY_RUN([int main(){exit(0);}],mysql_cv_option_$2=yes,mysql_cv_option_$2=no,mysql_cv_option_$2=no)
+ ])
+
+ CFLAGS="[$]OLD_CFLAGS"
+
+ if test x"[$]mysql_cv_option_$2" = "xyes" ; then
+ $3="[$]$3 $1"
+ AC_MSG_RESULT(yes)
+ $5
+ else
+ AC_MSG_RESULT(no)
+ $4
+ fi
+])
+
+# arch, option, cache_name, variable
+AC_DEFUN(AC_SYS_CPU_COMPILER_FLAG,
+[
+ if test "`uname -m 2>/dev/null`" = "$1" ; then
+ AC_SYS_COMPILER_FLAG($2,$3,$4)
+ fi
+])
+
+# os, option, cache_name, variable
+AC_DEFUN(AC_SYS_OS_COMPILER_FLAG,
+[
+ if test "x$mysql_cv_sys_os" = "x$1" ; then
+ AC_SYS_COMPILER_FLAG($2,$3,$4)
+ fi
+])
+
+# We need some special hacks when running slowaris
+AC_PATH_PROG(uname_prog, uname, no)
+
+# We should go through this and put all the explictly system dependent
+# stuff in one place
+AC_MSG_CHECKING(operating system)
+AC_CACHE_VAL(mysql_cv_sys_os,
+[
+if test "$uname_prog" != "no"; then
+ mysql_cv_sys_os="`uname`"
+else
+ mysql_cv_sys_os="Not Solaris"
+fi
+])
+AC_MSG_RESULT($mysql_cv_sys_os)
+
+# This should be rewritten to use $target_os
+case "$target_os" in
+ sco3.2v5*)
+ CFLAGS="$CFLAGS -DSCO"
+ CXXFLAGS="$CXXFLAGS -DSCO"
+ LD='$(CC) $(CFLAGS)'
+ case "$CFLAGS" in
+ *-belf*)
+ AC_SYS_COMPILER_FLAG(-belf,sco_belf_option,CFLAGS,[],[
+ case "$LDFLAGS" in
+ *-belf*) ;;
+ *) echo "Adding -belf option to ldflags."
+ LDFLAGS="$LDFLAGS -belf"
+ ;;
+ esac
+ ])
+ ;;
+ *)
+ AC_SYS_COMPILER_FLAG(-belf,sco_belf_option,CFLAGS,[],[
+ case "$LDFLAGS" in
+ *-belf*) ;;
+ *)
+ echo "Adding -belf option to ldflags."
+ LDFLAGS="$LDFLAGS -belf"
+ ;;
+ esac
+ ])
+ ;;
+ esac
+ ;;
+ sysv5UnixWare*)
+ if test "$GCC" != "yes"; then
+ # We are using built-in inline function
+ CFLAGS="$CFLAGS -Kalloca"
+ fi
+ CXXFLAGS="$CXXFLAGS -DNO_CPLUSPLUS_ALLOCA"
+ ;;
+ sysv5OpenUNIX8*)
+ if test "$GCC" != "yes"; then
+ # We are using built-in inline function
+ CFLAGS="$CFLAGS -Kalloca"
+ fi
+ CXXFLAGS="$CXXFLAGS -DNO_CPLUSPLUS_ALLOCA"
+ ;;
+esac
+AC_SUBST(CC)
+AC_SUBST(CFLAGS)
+AC_SUBST(CXX)
+AC_SUBST(CXXFLAGS)
+AC_SUBST(LD)
+AC_SUBST(INSTALL_SCRIPT)
+
+export CC CXX CFLAGS LD LDFLAGS AR
+
+if test "$GXX" = "yes"
+then
+ # mysqld requires -fno-implicit-templates.
+ # Disable exceptions as they seams to create problems with gcc and threads.
+ # mysqld doesn't use run-time-type-checking, so we disable it.
+ CXXFLAGS="$CXXFLAGS -fno-implicit-templates -fno-exceptions -fno-rtti"
+
+ # If you are using 'gcc' 3.0 (not g++) to compile C++ programs on Linux,
+ # we will gets some problems when linking static programs.
+ # The following code is used to fix this problem.
+
+ if test "$CXX" = "gcc" -o "$CXX" = "ccache gcc"
+ then
+ if $CXX -v 2>&1 | grep 'version 3' > /dev/null 2>&1
+ then
+ CXXFLAGS="$CXXFLAGS -DUSE_MYSYS_NEW -DDEFINE_CXA_PURE_VIRTUAL"
+ fi
+ fi
+fi
+
+# Avoid bug in fcntl on some versions of linux
+AC_MSG_CHECKING("if we should use 'skip-locking' as default for $target_os")
+# Any wariation of Linux
+if expr "$target_os" : "[[Ll]]inux.*" > /dev/null
+then
+ MYSQLD_DEFAULT_SWITCHES="--skip-locking"
+ IS_LINUX="true"
+ AC_MSG_RESULT("yes");
+else
+ MYSQLD_DEFAULT_SWITCHES=""
+ IS_LINUX="false"
+ AC_MSG_RESULT("no");
+fi
+AC_SUBST(MYSQLD_DEFAULT_SWITCHES)
+AC_SUBST(IS_LINUX)
+
+dnl Find paths to some shell programs
+AC_PATH_PROG(LN, ln, ln)
+# This must be able to take a -f flag like normal unix ln.
+AC_PATH_PROG(LN_CP_F, ln, ln)
+if ! ( expr "$SYSTEM_TYPE" : ".*netware.*" > /dev/null ); then
+# If ln -f does not exists use -s (AFS systems)
+if test -n "$LN_CP_F"; then
+ LN_CP_F="$LN_CP_F -s"
+fi
+fi
+
+AC_PATH_PROG(MV, mv, mv)
+AC_PATH_PROG(RM, rm, rm)
+AC_PATH_PROG(CP, cp, cp)
+AC_PATH_PROG(SED, sed, sed)
+AC_PATH_PROG(CMP, cmp, cmp)
+AC_PATH_PROG(CHMOD, chmod, chmod)
+AC_PATH_PROG(HOSTNAME, hostname, hostname)
+# Check for a GNU tar named 'gtar', or 'gnutar' (MacOS X) and
+# fall back to 'tar' otherwise and hope that it's a GNU tar as well
+AC_CHECK_PROGS(TAR, gnutar gtar tar)
+
+dnl We use a path for perl so the script startup works
+dnl We make sure to use perl, not perl5, in hopes that the RPMs will
+dnl not depend on the perl5 binary being installed (probably a bug in RPM)
+AC_PATH_PROG(PERL, perl, no)
+if test "$PERL" != "no" && $PERL -e 'require 5' > /dev/null 2>&1
+then
+ PERL5=$PERL
+else
+ AC_PATH_PROG(PERL5, perl5, no)
+ if test "$PERL5" != no
+ then
+ PERL=$PERL5
+ ac_cv_path_PERL=$ac_cv_path_PERL5
+ fi
+fi
+
+AC_SUBST(HOSTNAME)
+AC_SUBST(PERL)
+AC_SUBST(PERL5)
+
+# Lock for PS
+AC_PATH_PROG(PS, ps, ps)
+AC_MSG_CHECKING("how to check if pid exists")
+PS=$ac_cv_path_PS
+# Linux style
+if $PS p $$ 2> /dev/null | grep $0 > /dev/null
+then
+ FIND_PROC="$PS p \$\$PID | grep mysqld > /dev/null"
+# Solaris
+elif $PS -p $$ 2> /dev/null | grep $0 > /dev/null
+then
+ FIND_PROC="$PS -p \$\$PID | grep mysqld > /dev/null"
+# BSD style
+elif $PS -uaxww 2> /dev/null | grep $0 > /dev/null
+then
+ FIND_PROC="$PS -uaxww | grep mysqld | grep \" \$\$PID \" > /dev/null"
+# SysV style
+elif $PS -ef 2> /dev/null | grep $0 > /dev/null
+then
+ FIND_PROC="$PS -ef | grep mysqld | grep \" \$\$PID \" > /dev/null"
+# Do anybody use this?
+elif $PS $$ 2> /dev/null | grep $0 > /dev/null
+then
+ FIND_PROC="$PS \$\$PID | grep mysqld > /dev/null"
+else
+ case $SYSTEM_TYPE in
+ *freebsd*)
+ FIND_PROC="$PS p \$\$PID | grep mysqld > /dev/null"
+ ;;
+ *darwin*)
+ FIND_PROC="$PS -uaxww | grep mysqld | grep \" \$\$PID \" > /dev/null"
+ ;;
+ *cygwin*)
+ FIND_PROC="$PS -e | grep mysqld | grep \" \$\$PID \" > /dev/null"
+ ;;
+ *netware* | *modesto*)
+ FIND_PROC=
+ ;;
+ *)
+ AC_MSG_ERROR([Could not find the right ps switches. Which OS is this ?. See the Installation chapter in the Reference Manual.])
+ esac
+fi
+AC_SUBST(FIND_PROC)
+AC_MSG_RESULT("$FIND_PROC")
+
+# Check if a pid is valid
+AC_PATH_PROG(KILL, kill, kill)
+AC_MSG_CHECKING("for kill switches")
+if $ac_cv_path_KILL -0 $$
+then
+ CHECK_PID="$ac_cv_path_KILL -0 \$\$PID > /dev/null 2> /dev/null"
+elif kill -s 0 $$
+then
+ CHECK_PID="$ac_cv_path_KILL -s 0 \$\$PID > /dev/null 2> /dev/null"
+else
+ AC_MSG_WARN([kill -0 to check for pid seems to fail])
+ CHECK_PID="$ac_cv_path_KILL -s SIGCONT \$\$PID > /dev/null 2> /dev/null"
+fi
+AC_SUBST(CHECK_PID)
+AC_MSG_RESULT("$CHECK_PID")
+
+# We need a ANSI C compiler
+AM_PROG_CC_STDC
+
+# We need an assembler, too
+AM_PROG_AS
+
+if test "$am_cv_prog_cc_stdc" = "no"
+then
+ AC_MSG_ERROR([MySQL requires a ANSI C compiler (and a C++ compiler). Try gcc. See the Installation chapter in the Reference Manual.])
+fi
+
+NOINST_LDFLAGS=
+
+static_nss=""
+STATIC_NSS_FLAGS=""
+OTHER_LIBC_LIB=""
+AC_ARG_WITH(other-libc,
+ [ --with-other-libc=DIR Link against libc and other standard libraries
+ installed in the specified non-standard location
+ overriding default. Originally added to be able to
+ link against glibc 2.2 without making the user
+ upgrade the standard libc installation.],
+ [
+ other_libc_include="$withval/include"
+ other_libc_lib="$withval/lib"
+ with_other_libc="yes"
+ enable_shared="no"
+ all_is_static="yes"
+ CFLAGS="$CFLAGS -I$other_libc_include"
+ # There seems to be a feature in gcc that treats system and libc headers
+ # silently when they violatate ANSI C++ standard, but it is strict otherwise
+ # since gcc cannot now recognize that our headers are libc, we work around
+ # by telling it to be permissive. Note that this option only works with
+ # new versions of gcc (2.95.x and above)
+ CXXFLAGS="$CXXFLAGS -fpermissive -I$other_libc_include"
+ if test -f "$other_libc_lib/libnss_files.a"
+ then
+ # libc has been compiled with --enable-static-nss
+ # we need special flags, but we will have to add those later
+ STATIC_NSS_FLAGS="-lc -lnss_files -lnss_dns -lresolv"
+ STATIC_NSS_FLAGS="$STATIC_NSS_FLAGS $STATIC_NSS_FLAGS"
+ OTHER_LIBC_LIB="-static -L$other_libc_lib"
+ static_nss=1
+ else
+ # this is a dirty hack. We if we detect static nss glibc in the special
+ # location, we do not re-direct the linker to get libraries from there
+ # during check. The reason is that if we did, we would have to find a
+ # way to append the special static nss flags to LIBS every time we do
+ # any check - this is definitely feasible, but not worthwhile the risk
+ # of breaking other things. So for our purposes it would be sufficient
+ # to assume that whoever is using static NSS knows what he is doing and
+ # has sensible libraries in the regular location
+ LDFLAGS="$LDFLAGS -static -L$other_libc_lib "
+ fi
+
+ # When linking against custom libc installed separately, we want to force
+ # all binary builds to be static, including the build done by configure
+ # itself to test for system features.
+ with_mysqld_ldflags="-all-static"
+ with_client_ldflags="-all-static"
+ NOINST_LDFLAGS="-all-static"
+ ],
+ [
+ other_libc_include=
+ other_libc_lib=
+ with_other_libc="no"
+ ]
+)
+AC_SUBST(NOINST_LDFLAGS)
+
+#
+# Check if we are using Linux and a glibc compiled with static nss
+# (this is true on the MySQL build machines to avoid NSS problems)
+#
+
+if test "$IS_LINUX" = "true" -a "$static_nss" = ""
+then
+ tmp=`nm /usr/lib/libc.a | grep _nss_files_getaliasent_r`
+ if test -n "$tmp"
+ then
+ STATIC_NSS_FLAGS="-lc -lnss_files -lnss_dns -lresolv"
+ STATIC_NSS_FLAGS="$STATIC_NSS_FLAGS $STATIC_NSS_FLAGS"
+ static_nss=1
+ fi
+fi
+
+
+AC_ARG_WITH(server-suffix,
+ [ --with-server-suffix Append value to the version string.],
+ [ MYSQL_SERVER_SUFFIX=`echo "$withval" | sed -e 's/^\(...................................\)..*$/\1/'` ],
+ [ MYSQL_SERVER_SUFFIX= ]
+ )
+AC_SUBST(MYSQL_SERVER_SUFFIX)
+
+# Set flags if we wants to have MIT threads.
+AC_ARG_WITH(mit-threads,
+ [ --with-mit-threads Always use included thread lib.],
+ [ with_mit_threads=$withval ],
+ [ with_mit_threads=no ]
+ )
+
+if test "$with_mit_threads" = "yes"
+then
+ enable_largefile="no" # Will not work on Linux.
+fi
+
+# Set flags if we want to force to use pthreads
+AC_ARG_WITH(pthread,
+ [ --with-pthread Force use of pthread library.],
+ [ with_pthread=$withval ],
+ [ with_pthread=no ]
+ )
+
+# Force use of thread libs LIBS
+AC_ARG_WITH(named-thread-libs,
+ [ --with-named-thread-libs=ARG
+ Use specified thread libraries instead of
+ those automatically found by configure.],
+ [ with_named_thread=$withval ],
+ [ with_named_thread=no ]
+ )
+
+# Force use of a curses libs
+AC_ARG_WITH(named-curses-libs,
+ [ --with-named-curses-libs=ARG
+ Use specified curses libraries instead of
+ those automatically found by configure.],
+ [ with_named_curses=$withval ],
+ [ with_named_curses=no ]
+ )
+
+# Force use of a zlib (compress)
+AC_ARG_WITH(named-z-libs,
+ [ --with-named-z-libs=ARG
+ Use specified zlib libraries instead of
+ those automatically found by configure.],
+ [ with_named_zlib=$withval ],
+ [ with_named_zlib=z ]
+ )
+
+# Make thread safe client
+AC_ARG_ENABLE(thread-safe-client,
+ [ --enable-thread-safe-client
+ Compile the client with threads.],
+ [ THREAD_SAFE_CLIENT=$enableval ],
+ [ THREAD_SAFE_CLIENT=no ]
+ )
+
+# compile with strings functions in assembler
+AC_ARG_ENABLE(assembler,
+ [ --enable-assembler Use assembler versions of some string
+ functions if available.],
+ [ ENABLE_ASSEMBLER=$enableval ],
+ [ ENABLE_ASSEMBLER=no ]
+ )
+
+AC_MSG_CHECKING(if we should use assembler functions)
+# For now we only support assembler on i386 and sparc systems
+AM_CONDITIONAL(ASSEMBLER_x86, test "$ENABLE_ASSEMBLER" = "yes" -a "$BASE_MACHINE_TYPE" = "i386")
+AM_CONDITIONAL(ASSEMBLER_sparc32, test "$ENABLE_ASSEMBLER" = "yes" -a "$BASE_MACHINE_TYPE" = "sparc")
+AM_CONDITIONAL(ASSEMBLER_sparc64, test "$ENABLE_ASSEMBLER" = "yes" -a "$BASE_MACHINE_TYPE" = "sparcv9")
+AM_CONDITIONAL(ASSEMBLER, test "$ASSEMBLER_x86_TRUE" = "" -o "$ASSEMBLER_sparc32_TRUE" = "")
+
+if test "$ASSEMBLER_TRUE" = ""
+then
+ AC_MSG_RESULT([yes])
+else
+ AC_MSG_RESULT([no])
+fi
+
+
+AC_MSG_CHECKING(if we should use RAID)
+AC_ARG_WITH(raid,
+ [ --with-raid Enable RAID Support],
+ [ USE_RAID=$withval ],
+ [ USE_RAID=no ]
+ )
+if test "$USE_RAID" = "yes"
+then
+ AC_MSG_RESULT([yes])
+ AC_DEFINE([USE_RAID])
+else
+ AC_MSG_RESULT([no])
+fi
+
+# Use this to set the place used for unix socket used to local communication.
+AC_ARG_WITH(unix-socket-path,
+ [ --with-unix-socket-path=SOCKET
+ Where to put the unix-domain socket. SOCKET must be
+ an absolute file name.],
+ [ MYSQL_UNIX_ADDR=$withval ],
+ [ MYSQL_UNIX_ADDR=$MYSQL_UNIX_ADDR_DEFAULT ]
+ )
+AC_SUBST(MYSQL_UNIX_ADDR)
+
+AC_ARG_WITH(tcp-port,
+ [ --with-tcp-port=port-number
+ Which port to use for MySQL services (default 3306)],
+ [ MYSQL_TCP_PORT=$withval ],
+ [ MYSQL_TCP_PORT=$MYSQL_TCP_PORT_DEFAULT ]
+ )
+AC_SUBST(MYSQL_TCP_PORT)
+# We might want to document the assigned port in the manual.
+AC_SUBST(MYSQL_TCP_PORT_DEFAULT)
+
+# Use this to set the place used for unix socket used to local communication.
+AC_ARG_WITH(mysqld-user,
+ [ --with-mysqld-user=username
+ What user the mysqld daemon shall be run as.],
+ [ MYSQLD_USER=$withval ],
+ [ MYSQLD_USER=mysql ]
+ )
+AC_SUBST(MYSQLD_USER)
+
+# If we should allow LOAD DATA LOCAL
+AC_MSG_CHECKING(If we should should enable LOAD DATA LOCAL by default)
+AC_ARG_ENABLE(local-infile,
+ [ --enable-local-infile Enable LOAD DATA LOCAL INFILE (default: disabled)],
+ [ ENABLED_LOCAL_INFILE=$enableval ],
+ [ ENABLED_LOCAL_INFILE=no ]
+ )
+if test "$ENABLED_LOCAL_INFILE" = "yes"
+then
+ AC_MSG_RESULT([yes])
+ AC_DEFINE([ENABLED_LOCAL_INFILE])
+else
+ AC_MSG_RESULT([no])
+fi
+
+MYSQL_SYS_LARGEFILE
+
+# Types that must be checked AFTER large file support is checked
+AC_TYPE_SIZE_T
+
+#--------------------------------------------------------------------
+# Check for system header files
+#--------------------------------------------------------------------
+
+AC_HEADER_DIRENT
+AC_HEADER_STDC
+AC_HEADER_SYS_WAIT
+AC_CHECK_HEADERS(fcntl.h float.h floatingpoint.h ieeefp.h limits.h \
+ memory.h pwd.h select.h \
+ stdlib.h stddef.h \
+ strings.h string.h synch.h sys/mman.h sys/socket.h netinet/in.h arpa/inet.h \
+ sys/timeb.h sys/types.h sys/un.h sys/vadvise.h sys/wait.h term.h \
+ unistd.h utime.h sys/utime.h termio.h termios.h sched.h crypt.h alloca.h \
+ sys/ioctl.h malloc.h sys/malloc.h linux/config.h)
+
+#--------------------------------------------------------------------
+# Check for system libraries. Adds the library to $LIBS
+# and defines HAVE_LIBM etc
+#--------------------------------------------------------------------
+
+AC_CHECK_LIB(m, floor, [], AC_CHECK_LIB(m, __infinity))
+
+AC_CHECK_LIB(nsl_r, gethostbyname_r, [],
+ AC_CHECK_LIB(nsl, gethostbyname_r))
+AC_CHECK_FUNC(gethostbyname_r)
+
+AC_CHECK_FUNC(setsockopt, , AC_CHECK_LIB(socket, setsockopt))
+AC_CHECK_FUNC(yp_get_default_domain, ,
+ AC_CHECK_LIB(nsl, yp_get_default_domain))
+AC_CHECK_FUNC(p2open, , AC_CHECK_LIB(gen, p2open))
+# This may get things to compile even if bind-8 is installed
+AC_CHECK_FUNC(bind, , AC_CHECK_LIB(bind, bind))
+# For crypt() on Linux
+AC_CHECK_LIB(crypt, crypt)
+AC_CHECK_FUNC(crypt, AC_DEFINE(HAVE_CRYPT))
+
+# For sem_xxx functions on Solaris 2.6
+AC_CHECK_FUNC(sem_init, , AC_CHECK_LIB(posix4, sem_init))
+
+# For compress in zlib
+MYSQL_CHECK_ZLIB_WITH_COMPRESS($with_named_zlib)
+
+#--------------------------------------------------------------------
+# Check for TCP wrapper support
+#--------------------------------------------------------------------
+
+AC_ARG_WITH(libwrap,
+[ --with-libwrap[=DIR] Compile in libwrap (tcp_wrappers) support],[
+ case "$with_libwrap" in
+ no) : ;;
+ yes|*)
+ _cppflags=${CPPFLAGS}
+ _ldflags=${LDFLAGS}
+
+ if test "$with_libwrap" != "yes"; then
+ CPPFLAGS="${CPPFLAGS} -I$with_libwrap/include"
+ LDFLAGS="${LDFLAGS} -L$with_libwrap/lib"
+ fi
+
+ _libs=${LIBS}
+ AC_CHECK_HEADER(tcpd.h,
+ LIBS="-lwrap $LIBS"
+ AC_MSG_CHECKING(for TCP wrappers library -lwrap)
+ AC_TRY_LINK([#include <tcpd.h>
+int allow_severity = 0;
+int deny_severity = 0;
+
+struct request_info *req;
+],[hosts_access (req)],
+ AC_MSG_RESULT(yes)
+ AC_DEFINE(LIBWRAP)
+ AC_DEFINE(HAVE_LIBWRAP)
+ if test "$with_libwrap" != "yes"; then
+ WRAPLIBS="-L${with_libwrap}/lib"
+ fi
+ WRAPLIBS="${WRAPLIBS} -lwrap",
+ AC_MSG_RESULT(no)
+ CPPFLAGS=${_cppflags} LDFLAGS=${_ldflags}),
+ CPPFLAGS=${_cppflags} LDFLAGS=${_ldflags})
+ LDFLAGS=${_ldflags} LIBS=${_libs}
+ ;;
+ esac
+])
+AC_SUBST(WRAPLIBS)
+
+if test "$IS_LINUX" = "true"; then
+ AC_MSG_CHECKING([for atomic operations])
+
+ atom_ops=
+ AC_TRY_RUN([
+#include <asm/atomic.h>
+int main()
+{
+ atomic_t v;
+
+ atomic_set(&v, 23);
+ atomic_add(5, &v);
+ return atomic_read(&v) == 28 ? 0 : -1;
+}
+ ], AC_DEFINE(HAVE_ATOMIC_ADD) atom_ops="${atom_ops}atomic_add ",
+ )
+ AC_TRY_RUN([
+#include <asm/atomic.h>
+int main()
+{
+ atomic_t v;
+
+ atomic_set(&v, 23);
+ atomic_sub(5, &v);
+ return atomic_read(&v) == 18 ? 0 : -1;
+}
+ ], AC_DEFINE(HAVE_ATOMIC_SUB) atom_ops="${atom_ops}atomic_sub ",
+ )
+
+ if test -z "$atom_ops"; then atom_ops="no"; fi
+ AC_MSG_RESULT($atom_ops)
+
+ AC_ARG_WITH(pstack,
+ [ --with-pstack Use the pstack backtrace library],
+ [ USE_PSTACK=$withval ],
+ [ USE_PSTACK=no ])
+ pstack_libs=
+ pstack_dirs=
+ if test "$USE_PSTACK" = yes -a "$IS_LINUX" = "true" -a "$BASE_MACHINE_TYPE" = "i386" -a "$with_mit_threads" = "no"
+ then
+ have_libiberty= have_libbfd=
+ my_save_LIBS="$LIBS"
+dnl I have no idea if this is a good test - can not find docs for libiberty
+ AC_CHECK_LIB([iberty], [fdmatch],
+ [have_libiberty=yes
+ AC_CHECK_LIB([bfd], [bfd_openr], [have_libbfd=yes], , [-liberty])])
+ LIBS="$my_save_LIBS"
+
+ if test x"$have_libiberty" = xyes -a x"$have_libbfd" = xyes
+ then
+ pstack_dirs='$(top_srcdir)'/pstack
+ pstack_libs="../pstack/libpstack.a -lbfd -liberty"
+ # We must link staticly when using pstack
+ with_mysqld_ldflags="-all-static"
+ AC_SUBST([pstack_dirs])
+ AC_SUBST([pstack_libs])
+ AC_DEFINE([USE_PSTACK])
+dnl This check isn't needed, but might be nice to give some feedback....
+dnl AC_CHECK_HEADER(libiberty.h,
+dnl have_libiberty_h=yes,
+dnl have_libiberty_h=no)
+ else
+ USE_PSTACK="no"
+ fi
+ else
+ USE_PSTACK="no"
+ fi
+fi
+AM_CONDITIONAL(COMPILE_PSTACK, test "$USE_PSTACK" = "yes")
+AC_MSG_CHECKING([if we should use pstack])
+AC_MSG_RESULT([$USE_PSTACK])
+
+# Check for gtty if termio.h doesn't exists
+if test "$ac_cv_header_termio_h" = "no" -a "$ac_cv_header_termios_h" = "no"
+then
+ AC_CHECK_FUNC(gtty, , AC_CHECK_LIB(compat, gtty))
+fi
+# We make a special variable for client library's to avoid including
+# thread libs in the client.
+NON_THREADED_CLIENT_LIBS="$LIBS"
+
+AC_MSG_CHECKING([for int8])
+case $SYSTEM_TYPE in
+ *netware)
+ AC_MSG_RESULT([no])
+ ;;
+ *)
+AC_TRY_RUN([
+#ifdef HAVE_STDLIB_H
+#include <stdlib.h>
+#endif
+
+#ifdef HAVE_STDDEF_H
+#include <stddef.h>
+#endif
+
+#ifdef HAVE_SYS_TYPES_H
+#include <sys/types.h>
+#endif
+
+int main()
+{
+ int8 i;
+ return 0;
+}
+], AC_DEFINE(HAVE_INT_8_16_32) AC_MSG_RESULT([yes]), AC_MSG_RESULT([no])
+)
+ ;;
+esac
+
+#
+# Some system specific hacks
+#
+
+MAX_C_OPTIMIZE="-O3"
+MAX_CXX_OPTIMIZE="-O3"
+
+case $SYSTEM_TYPE in
+ *solaris2.7*)
+ # Solaris 2.7 has a broken /usr/include/widec.h
+ # Make a fixed copy in ./include
+ echo "Fixing broken include files for $SYSTEM_TYPE"
+ echo " - Creating local copy of widec.h"
+ if test ! -d include
+ then
+ mkdir ./include
+ fi
+ builddir=`pwd`
+ sed -e "s|^#if[ ]*!defined(lint) && !defined(__lint)|#if !defined\(lint\) \&\& !defined\(__lint\) \&\& !defined\(getwc\)|" < /usr/include/widec.h > include/widec.h
+ CFLAGS="$CFLAGS -DHAVE_CURSES_H -I$builddir/include -DHAVE_RWLOCK_T"
+ CXXFLAGS="$CXXFLAGS -DHAVE_CURSES_H -I$builddir/include -DHAVE_RWLOCK_T"
+ ;;
+ *solaris2.8*)
+ # Solaris 2.8 has a broken /usr/include/widec.h
+ # Make a fixed copy in ./include
+ echo "Fixing broken include files for $SYSTEM_TYPE"
+ echo " - Creating local copy of widec.h"
+ if test ! -d include
+ then
+ mkdir ./include
+ fi
+ builddir=`pwd`
+ sed -e "s|^#if[ ]*!defined(__lint)|#if !defined\(__lint\) \&\& !defined\(getwc\)|" < /usr/include/widec.h > include/widec.h
+ CFLAGS="$CFLAGS -DHAVE_CURSES_H -I$builddir/include -DHAVE_RWLOCK_T"
+ CXXFLAGS="$CXXFLAGS -DHAVE_CURSES_H -I$builddir/include -DHAVE_RWLOCK_T"
+ ;;
+ *solaris2.5.1*)
+ echo "Enabling getpass() workaround for Solaris 2.5.1"
+ CFLAGS="$CFLAGS -DHAVE_BROKEN_GETPASS -DSOLARIS -DHAVE_RWLOCK_T";
+ CXXFLAGS="$CXXFLAGS -DHAVE_RWLOCK_T -DSOLARIS"
+ ;;
+ *solaris*)
+ CFLAGS="$CFLAGS -DHAVE_RWLOCK_T"
+ CXXFLAGS="$CXXFLAGS -DHAVE_RWLOCK_T"
+ ;;
+ *SunOS*)
+ echo "Enabling getpass() workaround for SunOS"
+ CFLAGS="$CFLAGS -DHAVE_BROKEN_GETPASS -DSOLARIS";
+ ;;
+ *hpux10.20*)
+ echo "Enabling workarounds for hpux 10.20"
+ CFLAGS="$CFLAGS -DHAVE_BROKEN_SNPRINTF -DSIGNALS_DONT_BREAK_READ -DDO_NOT_REMOVE_THREAD_WRAPPERS -DHPUX10 -DSIGNAL_WITH_VIO_CLOSE -DHAVE_BROKEN_PTHREAD_COND_TIMEDWAIT -DHAVE_POSIX1003_4a_MUTEX"
+ CXXFLAGS="$CXXFLAGS -DHAVE_BROKEN_SNPRINTF -D_INCLUDE_LONGLONG -DSIGNALS_DONT_BREAK_READ -DDO_NOT_REMOVE_THREAD_WRAPPERS -DHPUX10 -DSIGNAL_WITH_VIO_CLOSE -DHAVE_BROKEN_PTHREAD_COND_TIMEDWAIT -DHAVE_POSIX1003_4a_MUTEX"
+ if test "$with_named_thread" = "no"
+ then
+ echo "Using --with-named-thread=-lpthread"
+ with_named_thread="-lcma"
+ fi
+ ;;
+ *hpux11.*)
+ echo "Enabling workarounds for hpux 11"
+ CFLAGS="$CFLAGS -DHPUX11 -DHAVE_BROKEN_PREAD -DDONT_USE_FINITE -DHAVE_BROKEN_GETPASS -DNO_FCNTL_NONBLOCK -DDO_NOT_REMOVE_THREAD_WRAPPERS -DHAVE_BROKEN_PTHREAD_COND_TIMEDWAIT"
+ CXXFLAGS="$CXXFLAGS -DHPUX11 -DHAVE_BROKEN_PREAD -DDONT_USE_FINITE -D_INCLUDE_LONGLONG -DNO_FCNTL_NONBLOCK -DDO_NOT_REMOVE_THREAD_WRAPPERS -DHAVE_BROKEN_PTHREAD_COND_TIMEDWAIT"
+ if test "$with_named_thread" = "no"
+ then
+ echo "Using --with-named-thread=-lpthread"
+ with_named_thread="-lpthread"
+ fi
+ # Fixes for HPUX 11.0 compiler
+ if test "$ac_cv_prog_gcc" = "no"
+ then
+ CFLAGS="$CFLAGS -DHAVE_BROKEN_INLINE"
+ CXXFLAGS="$CXXFLAGS +O2"
+ MAX_C_OPTIMIZE=""
+ MAX_CXX_OPTIMIZE=""
+ fi
+ ;;
+ *rhapsody*)
+ if test "$ac_cv_prog_gcc" = "yes"
+ then
+ CPPFLAGS="$CPPFLAGS -traditional-cpp "
+ CFLAGS="-DHAVE_CTHREADS_WRAPPER -DDO_NOT_REMOVE_THREAD_WRAPPERS"
+ CXXFLAGS="-DHAVE_CTHREADS_WRAPPER"
+ if test $with_named_curses = "no"
+ then
+ with_named_curses=""
+ fi
+ fi
+ ;;
+ *darwin5*)
+ if test "$ac_cv_prog_gcc" = "yes"
+ then
+ FLAGS="-traditional-cpp -DHAVE_DARWIN_THREADS -D_P1003_1B_VISIBLE -DSIGNAL_WITH_VIO_CLOSE -DSIGNALS_DONT_BREAK_READ -DHAVE_BROKEN_REALPATH"
+ CFLAGS="$CFLAGS $FLAGS"
+ CXXFLAGS="$CXXFLAGS $FLAGS"
+ MAX_C_OPTIMIZE="-O"
+ with_named_curses=""
+ fi
+ ;;
+ *darwin6*)
+ if test "$ac_cv_prog_gcc" = "yes"
+ then
+ FLAGS="-DHAVE_DARWIN_THREADS -D_P1003_1B_VISIBLE -DSIGNAL_WITH_VIO_CLOSE -DSIGNALS_DONT_BREAK_READ -DHAVE_BROKEN_REALPATH"
+ CFLAGS="$CFLAGS $FLAGS"
+ CXXFLAGS="$CXXFLAGS $FLAGS"
+ MAX_C_OPTIMIZE="-O"
+ fi
+ ;;
+ *darwin7*)
+ if test "$ac_cv_prog_gcc" = "yes"
+ then
+ FLAGS="-DHAVE_DARWIN_THREADS -D_P1003_1B_VISIBLE -DSIGNAL_WITH_VIO_CLOSE -DSIGNALS_DONT_BREAK_READ"
+ CFLAGS="$CFLAGS $FLAGS"
+ CXXFLAGS="$CXXFLAGS $FLAGS"
+ MAX_C_OPTIMIZE="-O"
+ fi
+ ;;
+ *freebsd*)
+ echo "Adding fix for interrupted reads"
+ OSVERSION=`sysctl -a | grep osreldate | awk '{ print $2 }'`
+ if test "$OSVERSION" -gt "480100" && \
+ test "$OSVERSION" -lt "500000" || \
+ test "$OSVERSION" -gt "500109"
+ then
+ CXXFLAGS="$CXXFLAGS -DMYSQLD_NET_RETRY_COUNT=1000000"
+ else
+ CFLAGS="$CFLAGS -DHAVE_BROKEN_REALPATH"
+ CXXFLAGS="$CXXFLAGS -DMYSQLD_NET_RETRY_COUNT=1000000 -DHAVE_BROKEN_REALPATH"
+ fi
+ ;;
+ *netbsd*)
+ echo "Adding flag -Dunix"
+ CFLAGS="$CFLAGS -Dunix"
+ CXXFLAGS="$CXXFLAGS -Dunix"
+ OVERRIDE_MT_LD_ADD="\$(top_srcdir)/mit-pthreads/obj/libpthread.a"
+ ;;
+ *bsdi*)
+ echo "Adding fix for BSDI"
+ CFLAGS="$CFLAGS -D__BSD__ -DHAVE_BROKEN_REALPATH"
+ AC_DEFINE_UNQUOTED(SOCKOPT_OPTLEN_TYPE, size_t)
+ ;;
+ *sgi-irix6*)
+ if test "$with_named_thread" = "no"
+ then
+ echo "Using --with-named-thread=-lpthread"
+ with_named_thread="-lpthread"
+ fi
+ CXXFLAGS="$CXXFLAGS -D_BOOL"
+ ;;
+ *aix4.3*)
+ echo "Adding defines for AIX"
+ CFLAGS="$CFLAGS -Wa,-many -DUNDEF_HAVE_INITGROUPS -DSIGNALS_DONT_BREAK_READ"
+ CXXFLAGS="$CXXFLAGS -Wa,-many -DUNDEF_HAVE_INITGROUPS -DSIGNALS_DONT_BREAK_READ"
+ ;;
+dnl Is this the right match for DEC OSF on alpha?
+ *dec-osf*)
+ if test "$ac_cv_prog_gcc" = "yes" && test "$host_cpu" = "alpha"
+ then
+ echo "Adding defines for DEC OSF on alpha"
+ CFLAGS="$CFLAGS -mieee"
+ CXXFLAGS="$CXXFLAGS -mieee"
+ fi
+ echo "Adding defines for OSF1"
+ # gethostbyname_r is deprecated and doesn't work ok on OSF1
+ CFLAGS="$CFLAGS -DUNDEF_HAVE_GETHOSTBYNAME_R"
+ CXXFLAGS="$CXXFLAGS -DUNDEF_HAVE_GETHOSTBYNAME_R"
+ ;;
+ *netware*)
+ # No need for curses library so set it to null
+ with_named_curses=""
+
+ # No thread library - in LibC
+ with_named_thread=""
+
+ #
+ # Edit Makefile.in files.
+ #
+ echo -n "configuring Makefile.in files for NetWare... "
+ for file in sql/Makefile.in libmysql/Makefile.in libmysql_r/Makefile.in sql/share/Makefile.in strings/Makefile.in client/Makefile.in
+ do
+ # echo "#### $file ####"
+ filedir="`dirname $file`"
+ filebase="`basename $file`"
+ filesed=$filedir/$filebase.sed
+ #
+ # Backup and always use original file
+ #
+ if test -f $file.bk
+ then
+ cp -fp $file.bk $file
+ else
+ cp -fp $file $file.bk
+ fi
+ case $file in
+ sql/Makefile.in)
+ # Use gen_lex_hash.linux instead of gen_lex_hash
+ # Add library dependencies to mysqld_DEPENDENCIES
+ lib_DEPENDENCIES="\$(bdb_libs_with_path) \$(innodb_libs) \$(pstack_libs) \$(innodb_system_libs) \$(openssl_libs)"
+ cat > $filesed << EOF
+s,\(^.*\$(MAKE) gen_lex_hash\),#\1,
+s,\(\./gen_lex_hash\),\1.linux,
+s%\(mysqld_DEPENDENCIES = \) %\1$lib_DEPENDENCIES %
+EOF
+ ;;
+ sql/share/Makefile.in)
+ cat > $filesed << EOF
+s,\(extra/comp_err\),\1.linux,
+EOF
+ ;;
+ libmysql/Makefile.in)
+ cat > $filesed << EOF
+s,\(\./conf_to_src\)\( \$(top_srcdir)\),\1.linux\2,
+s,\(: conf_to_src\),\1.linux,
+EOF
+ ;;
+ libmysql_r/Makefile.in)
+ cat > $filesed << EOF
+s,\(\./conf_to_src\)\( \$(top_srcdir)\),\1.linux\2,
+s,\(: conf_to_src\),\1.linux,
+EOF
+ ;;
+ strings/Makefile.in)
+ cat > $filesed << EOF
+s,\(\./conf_to_src\)\( \$(top_srcdir)\),\1.linux\2,
+s,\(: conf_to_src\),\1.linux,
+EOF
+ ;;
+ client/Makefile.in)
+ #
+ cat > $filesed << EOF
+s,libmysqlclient.la,.libs/libmysqlclient.a,
+EOF
+ ;;
+ esac
+ if `sed -f $filesed $file > $file.nw`;\
+ then
+ mv -f $file.nw $file
+ rm -f $filesed
+ else
+ exit 1
+ fi
+ # wait for file system changes to complete
+ sleep 1
+ done
+ echo "done"
+
+ #
+ # Make sure the following files are writable.
+ #
+ # When the files are retrieved from some source code control systems they are read-only.
+ #
+ echo -n "making sure specific build files are writable... "
+ for file in \
+ Docs/include.texi \
+ Docs/mysql.info \
+ Docs/manual.txt \
+ Docs/manual_toc.html \
+ Docs/manual.html \
+ Docs/INSTALL-BINARY \
+ INSTALL-SOURCE \
+ COPYING \
+ COPYING.LIB \
+ MIRRORS
+ do
+ if test -e $file; then
+ chmod +w $file
+ fi
+ done
+ echo "done"
+
+ ;;
+esac
+
+
+#---START: Used in for client configure
+# Check if we threads are in libc or if we should use
+# -lpthread, -lpthreads or mit-pthreads
+# We have to check libc last because else it fails on Solaris 2.6
+
+with_posix_threads="no"
+# Hack for DEC-UNIX (OSF1)
+if test "$with_named_thread" = "no" -a "$with_mit_threads" = "no"
+then
+ # Look for LinuxThreads.
+ AC_MSG_CHECKING("LinuxThreads")
+ res=`grep Linuxthreads /usr/include/pthread.h 2>/dev/null | wc -l`
+ if test "$res" -gt 0
+ then
+ AC_MSG_RESULT("Found")
+ AC_DEFINE(HAVE_LINUXTHREADS)
+ # Linux 2.0 sanity check
+ AC_TRY_COMPILE([#include <sched.h>], [int a = sched_get_priority_min(1);], ,
+ AC_MSG_ERROR([Syntax error in sched.h. Change _P to __P in the /usr/include/sched.h file. See the Installation chapter in the Reference Manual]))
+ # RedHat 5.0 does not work with dynamic linking of this. -static also
+ # gives a speed increase in linux so it does not hurt on other systems.
+ with_named_thread="-lpthread"
+ else
+ AC_MSG_RESULT("Not found")
+ # If this is a linux machine we should barf
+ if test "$IS_LINUX" = "true"
+ then
+ AC_MSG_ERROR([This is a linux system and Linuxthreads was not
+found. On linux Linuxthreads should be used. Please install Linuxthreads
+(or a new glibc) and try again. See the Installation chapter in the
+Reference Manual for more information.])
+ else
+ AC_MSG_CHECKING("DEC threads")
+ if test -f /usr/shlib/libpthread.so -a -f /usr/lib/libmach.a -a -f /usr/ccs/lib/cmplrs/cc/libexc.a
+ then
+ with_named_thread="-lpthread -lmach -lexc"
+ CFLAGS="$CFLAGS -D_REENTRANT"
+ CXXFLAGS="$CXXFLAGS -D_REENTRANT"
+ AC_DEFINE(HAVE_DEC_THREADS)
+ AC_MSG_RESULT("yes")
+ else
+ AC_MSG_RESULT("no")
+ AC_MSG_CHECKING("DEC 3.2 threads")
+ if test -f /usr/shlib/libpthreads.so -a -f /usr/lib/libmach.a -a -f /usr/ccs/lib/cmplrs/cc/libexc.a
+ then
+ with_named_thread="-lpthreads -lmach -lc_r"
+ AC_DEFINE(HAVE_DEC_THREADS)
+ AC_DEFINE(HAVE_DEC_3_2_THREADS)
+ with_osf32_threads="yes"
+ MYSQLD_DEFAULT_SWITCHES="--skip-thread-priority"
+ AC_MSG_RESULT("yes")
+ else
+ AC_MSG_RESULT("no")
+ fi
+ fi
+ fi
+ fi
+fi
+
+
+dnl This is needed because -lsocket has to come after the thread
+dnl library on SCO.
+AC_DEFUN([MYSQL_REMOVE_SOCKET_FROM_LIBS_HACK], [
+ LIBS=`echo " $LIBS " | sed -e 's/ -lsocket / /g'`
+])
+# Hack for SCO UNIX
+if test "$with_named_thread" = "no"
+then
+ AC_MSG_CHECKING("SCO threads")
+ if expr "$SYSTEM_TYPE" : ".*sco.*" > /dev/null
+ then
+ if test -f /usr/lib/libgthreads.a -o -f /usr/lib/libgthreads.so
+ then
+ MYSQL_REMOVE_SOCKET_FROM_LIBS_HACK
+ with_named_thread="-lgthreads -lsocket -lgthreads"
+ # sched.h conflicts with fsu-threads
+ touch ./include/sched.h
+
+ # We must have gcc
+ if expr "$CC" : ".*gcc.*"
+ then
+ AC_MSG_RESULT("yes")
+ else
+ AC_MSG_ERROR([On SCO UNIX MySQL must be compiled with gcc. See the Installation chapter in the Reference Manual.]);
+ fi
+ AC_MSG_RESULT("yes")
+ elif test -f /usr/local/lib/libpthread.a -o -f /usr/local/lib/libpthread.so
+ then
+ MYSQL_REMOVE_SOCKET_FROM_LIBS_HACK
+ with_named_thread="-lpthread -lsocket"
+ # sched.h conflicts with fsu-threads
+ # touch ./include/sched.h
+
+ AC_MSG_CHECKING("for gcc")
+ # We must have gcc
+ if expr "$CC" : ".*gcc.*"
+ then
+ AC_MSG_RESULT("yes")
+ else
+ AC_MSG_ERROR([On SCO UNIX MySQL must be compiled with gcc. See the Installation chapter in the Reference Manual.]);
+ fi
+ AC_MSG_RESULT("yes")
+ # Hack for SCO UnixWare 7.1.x
+ #
+ elif test "$with_named_thread" = "no"
+ then
+ AC_MSG_RESULT("no")
+ AC_MSG_CHECKING("SCO UnixWare 7.1.x native threads")
+ if expr "$SYSTEM_TYPE" : ".*sco.*" > /dev/null
+ then
+ if test -f /usr/lib/libthread.so -o -f /usr/lib/libthreadT.so
+ then
+ MYSQL_REMOVE_SOCKET_FROM_LIBS_HACK
+ if expr "$CC" : ".*gcc.*"
+ then
+ with_named_thread="-pthread -lsocket -lnsl"
+ else
+ with_named_thread="-Kthread -lsocket -lnsl"
+ fi
+ if expr "$SYSTEM_TYPE" : ".*unixware7.0.0" > /dev/null
+ then
+ AC_DEFINE(HAVE_UNIXWARE7_THREADS)
+ else
+ AC_DEFINE(HAVE_UNIXWARE7_POSIX)
+ fi
+ AC_MSG_RESULT("yes")
+ # We must have cc
+ AC_MSG_CHECKING("for gcc")
+ if expr "$CC" : ".*gcc.*"
+ then
+ CC="$CC -pthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ CXX="$CXX -pthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ else
+ CC="$CC -Kthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ CXX="$CXX -Kthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ fi
+ else
+ { echo "configure: error: Can't find thread libs on SCO UnixWare7. See the Installation chapter in the Reference Manual." 1>&2; exit 1; };
+ fi
+ else
+ AC_MSG_RESULT("no")
+ fi
+ else
+ AC_MSG_ERROR([On SCO UNIX MySQL requires that the FSUThreads package is installed. See the Installation chapter in the Reference Manual.]);
+ fi
+ else
+ AC_MSG_RESULT("no")
+ fi
+fi
+# Hack for SCO UnixWare7
+#
+if test "$with_named_thread" = "no"
+then
+ AC_MSG_CHECKING("SCO UnixWare7 native threads")
+ if expr "$SYSTEM_TYPE" : ".*UnixWare*" > /dev/null
+ then
+ if test -f /usr/lib/libthread.so -o -f /usr/lib/libthreadT.so
+ then
+ MYSQL_REMOVE_SOCKET_FROM_LIBS_HACK
+ if expr "$CC" : ".*gcc.*"
+ then
+ with_named_thread="-pthread -lsocket -lnsl"
+ else
+ with_named_thread="-Kthread -lsocket -lnsl"
+ fi
+ if expr "$SYSTEM_TYPE" : ".*unixware7.0.0" > /dev/null
+ then
+ AC_DEFINE(HAVE_UNIXWARE7_THREADS)
+ else
+ AC_DEFINE(HAVE_UNIXWARE7_POSIX)
+ fi
+ # We must have cc
+ AC_MSG_CHECKING("for gcc")
+ if expr "$CC" : ".*gcc.*"
+ then
+ CC="$CC -pthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ CXX="$CXX -pthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ else
+ CC="$CC -Kthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ CXX="$CXX -Kthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ fi
+ AC_MSG_RESULT("yes")
+ else
+ { echo "configure: error: Can't find thread libs on SCO UnixWare7. See the Installation chapter in the Reference Manual." 1>&2; exit 1; };
+ fi
+ else
+ AC_MSG_RESULT("no")
+ fi
+fi
+
+# Hack for Caldera OpenUNIX8
+#
+if test "$with_named_thread" = "no"
+then
+ AC_MSG_CHECKING("OpenUNIX8 native threads")
+ if expr "$SYSTEM_TYPE" : ".*OpenUNIX*" > /dev/null
+ then
+ if test -f /usr/lib/libthread.so -o -f /usr/lib/libthreadT.so
+ then
+ MYSQL_REMOVE_SOCKET_FROM_LIBS_HACK
+ if expr "$CC" : ".*gcc.*"
+ then
+ with_named_thread="-pthread -lsocket -lnsl"
+ else
+ with_named_thread="-Kthread -lsocket -lnsl"
+ fi
+ if expr "$SYSTEM_TYPE" : ".*unixware7.0.0" > /dev/null
+ then
+ AC_DEFINE(HAVE_UNIXWARE7_THREADS)
+ else
+ AC_DEFINE(HAVE_UNIXWARE7_POSIX)
+ fi
+ # We must have cc
+ AC_MSG_CHECKING("for gcc")
+ if expr "$CC" : ".*gcc.*"
+ then
+ CC="$CC -pthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ CXX="$CXX -pthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ else
+ CC="$CC -Kthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ CXX="$CXX -Kthread -DUNIXWARE_7 -DHAVE_BROKEN_RWLOCK";
+ fi
+ AC_MSG_RESULT("yes")
+ else
+ { echo "configure: error: Can't find thread libs on Caldera OpenUNIX 8. See the Installation chapter in the Reference Manual." 1>&2; exit 1; };
+ fi
+ else
+ AC_MSG_RESULT("no")
+ fi
+fi
+
+# Hack for Siemens UNIX
+if test "$with_named_thread" = "no" -a "$with_mit_threads" = "no"
+then
+ AC_MSG_CHECKING("Siemens threads")
+ if test -f /usr/lib/libxnet.so -a "$SYSTEM_TYPE" = "sni-sysv4"
+ then
+ LIBS="-lxnet $LIBS"
+ NON_THREADED_CLIENT_LIBS="$NON_THREADED_CLIENT_LIBS -lxnet"
+ with_named_thread="-Kthread $LDFLAGS -lxnet"
+ LD_FLAGS=""
+ CFLAGS="-Kthread $CFLAGS"
+ CXXFLAGS="-Kthread $CXXFLAGS"
+ AC_MSG_RESULT("yes")
+ else
+ AC_MSG_RESULT("no")
+ fi
+fi
+
+# Use library named -lpthread
+if test "$with_named_thread" = "no" -a "$with_pthread" = "yes"
+then
+ with_named_thread="-lpthread"
+fi
+
+#---END:
+
+# Hack for Solaris >= 2.5
+# We want both the new and the old interface
+
+if test "$with_named_thread" = "no" -a "$with_mit_threads" = "no"
+then
+ AC_MSG_CHECKING("Solaris threads")
+ if test -f /usr/lib/libpthread.so -a -f /usr/lib/libthread.so
+ then
+ with_named_thread="-lpthread -lthread"
+ AC_MSG_RESULT("yes")
+ else
+ AC_MSG_RESULT("no")
+ fi
+fi
+
+TOOLS_LIBS="$NON_THREADED_CLIENT_LIBS"
+
+# Should we use named pthread library ?
+AC_MSG_CHECKING("named thread libs:")
+if test "$with_named_thread" != "no"
+then
+ LIBS="$with_named_thread $LIBS $with_named_thread"
+ TOOLS_LIBS="$with_named_thread $TOOLS_LIBS $with_named_thread"
+ with_posix_threads="yes"
+ with_mit_threads="no"
+ AC_MSG_RESULT("$with_named_thread")
+else
+ AC_MSG_RESULT("no")
+ if test "$with_mit_threads" = "no"
+ then
+ # pthread_create is in standard libraries (As in BSDI 3.0)
+ AC_MSG_CHECKING("for pthread_create in -libc");
+ AC_TRY_LINK(
+ [#include <pthread.h>],
+ [ (void) pthread_create((pthread_t*) 0,(pthread_attr_t*) 0, 0, 0); ],
+ with_posix_threads=yes, with_posix_threads=no)
+ AC_MSG_RESULT("$with_posix_threads")
+ if test "$with_posix_threads" = "no"
+ then
+ AC_MSG_CHECKING("for pthread_create in -lpthread");
+ ac_save_LIBS="$LIBS"
+ ac_save_TOOLS_LIBS="$TOOLS_LIBS"
+ LIBS="$LIBS -lpthread"
+ TOOLS_LIBS="$TOOLS_LIBS -lpthread"
+ AC_TRY_LINK(
+ [#include <pthread.h>],
+ [ (void) pthread_create((pthread_t*) 0,(pthread_attr_t*) 0, 0, 0); ],
+ with_posix_threads=yes, with_posix_threads=no)
+ AC_MSG_RESULT("$with_posix_threads")
+ if test "$with_posix_threads" = "no"
+ then
+ LIBS=" $ac_save_LIBS -lpthreads"
+ TOOLS_LIBS=" $ac_save_TOOLS_LIBS -lpthreads"
+ AC_MSG_CHECKING("for pthread_create in -lpthreads");
+ AC_TRY_LINK(
+ [#include <pthread.h>],
+ [ pthread_create((pthread_t*) 0,(pthread_attr_t*) 0, 0, 0); ],
+ with_posix_threads=yes, with_posix_threads=no)
+ AC_MSG_RESULT("$with_posix_threads")
+ if test "$with_posix_threads" = "no"
+ then
+ # This is for FreeBSD
+ LIBS="$ac_save_LIBS -pthread"
+ TOOLS_LIBS="$ac_save_TOOLS_LIBS -pthread"
+ AC_MSG_CHECKING("for pthread_create in -pthread");
+ AC_TRY_LINK(
+ [#include <pthread.h>],
+ [ pthread_create((pthread_t*) 0,(pthread_attr_t*) 0, 0, 0); ],
+ with_posix_threads=yes, with_posix_threads=no)
+ AC_MSG_RESULT("$with_posix_threads")
+ if test "$with_posix_threads" = "no"
+ then
+ with_mit_threads="yes"
+ LIBS="$ac_save_LIBS"
+ TOOLS_LIBS="$ac_save_TOOLS_LIBS"
+ fi
+ fi
+ fi
+ fi
+ fi
+fi
+
+#---START: Used in for client configure
+# Must be checked after, because strtok_r may be in -lpthread
+# On AIX strtok_r is in libc_r
+
+my_save_LIBS="$LIBS"
+AC_CHECK_LIB(pthread,strtok_r)
+LIBS="$my_save_LIBS"
+if test "$ac_cv_lib_pthread_strtok_r" = "no"
+then
+ AC_CHECK_LIB(c_r,strtok_r)
+ case "$with_osf32_threads---$target_os" in
+ # Don't keep -lc_r in LIBS; -pthread handles it magically
+ yes---* | *---freebsd* | *---hpux*) LIBS="$my_save_LIBS" ;;
+
+ esac
+ AC_CHECK_FUNCS(strtok_r pthread_init)
+else
+ AC_CHECK_FUNCS(strtok_r)
+fi
+#---END:
+
+# Check for dlopen, needed for user definable functions
+# This must be checked after threads on AIX
+# We only need this for mysqld, not for the clients.
+
+my_save_LIBS="$LIBS"
+LIBS=""
+AC_CHECK_LIB(dl,dlopen)
+LIBDL=$LIBS
+LIBS="$my_save_LIBS"
+AC_SUBST(LIBDL)
+
+# System characteristics
+case $SYSTEM_TYPE in
+ *netware* | *modesto*) ;;
+ *)
+AC_SYS_RESTARTABLE_SYSCALLS
+ ;;
+esac
+
+# Build optimized or debug version ?
+# First check for gcc and g++
+if test "$ac_cv_prog_gcc" = "yes"
+then
+ DEBUG_CFLAGS="-g"
+ DEBUG_OPTIMIZE_CC="-O"
+ OPTIMIZE_CFLAGS="$MAX_C_OPTIMIZE"
+else
+ DEBUG_CFLAGS="-g"
+ DEBUG_OPTIMIZE_CC=""
+ OPTIMIZE_CFLAGS="-O"
+fi
+if test "$ac_cv_prog_cxx_g" = "yes"
+then
+ DEBUG_CXXFLAGS="-g"
+ DEBUG_OPTIMIZE_CXX="-O"
+ OPTIMIZE_CXXFLAGS="$MAX_CXX_OPTIMIZE"
+else
+ DEBUG_CXXFLAGS="-g"
+ DEBUG_OPTIMIZE_CXX=""
+ OPTIMIZE_CXXFLAGS="-O"
+fi
+
+if expr "$SYSTEM_TYPE" : ".*netware.*" > /dev/null; then
+ DEBUG_CFLAGS="$DEBUG_CFLAGS -DDEBUG -sym internal,codeview4"
+ DEBUG_CXXFLAGS="$DEBUG_CXXFLAGS -DDEBUG -sym internal,codeview4"
+ OPTIMIZE_CFLAGS="$OPTIMIZE_CFLAGS -DNDEBUG"
+ OPTIMIZE_CXXFLAGS="$OPTIMIZE_CXXFLAGS -DNDEBUG"
+fi
+
+AC_ARG_WITH(debug,
+ [ --without-debug Build a production version without debugging code],
+ [with_debug=$withval],
+ [with_debug=no])
+if test "$with_debug" = "yes"
+then
+ # Medium debug.
+ CFLAGS="$DEBUG_CFLAGS $DEBUG_OPTIMIZE_CC -DDBUG_ON -DSAFE_MUTEX $CFLAGS"
+ CXXFLAGS="$DEBUG_CXXFLAGS $DEBUG_OPTIMIZE_CXX -DSAFE_MUTEX $CXXFLAGS"
+elif test "$with_debug" = "full"
+then
+ # Full debug. Very slow in some cases
+ CFLAGS="$DEBUG_CFLAGS -DDBUG_ON -DSAFE_MUTEX -DSAFEMALLOC $CFLAGS"
+ CXXFLAGS="$DEBUG_CXXFLAGS -DSAFE_MUTEX -DSAFEMALLOC $CXXFLAGS"
+else
+ # Optimized version. No debug
+ CFLAGS="$OPTIMIZE_CFLAGS -DDBUG_OFF $CFLAGS"
+ CXXFLAGS="$OPTIMIZE_CXXFLAGS -DDBUG_OFF $CXXFLAGS"
+fi
+
+# Force static compilation to avoid linking problems/get more speed
+AC_ARG_WITH(mysqld-ldflags,
+ [ --with-mysqld-ldflags Extra linking arguments for mysqld],
+ [MYSQLD_EXTRA_LDFLAGS=$withval],
+ [MYSQLD_EXTRA_LDFLAGS=])
+AC_SUBST(MYSQLD_EXTRA_LDFLAGS)
+
+AC_ARG_WITH(client-ldflags,
+ [ --with-client-ldflags Extra linking arguments for clients],
+ [CLIENT_EXTRA_LDFLAGS=$withval],
+ [CLIENT_EXTRA_LDFLAGS=])
+AC_SUBST(CLIENT_EXTRA_LDFLAGS)
+
+AC_ARG_WITH(lib-ccflags,
+ [ --with-lib-ccflags Extra CC options for libraries],
+ [LIB_EXTRA_CCFLAGS=$withval],
+ [LIB_EXTRA_CCFLAGS=])
+AC_SUBST(LIB_EXTRA_CCFLAGS)
+
+# Avoid stupid bug on some OS
+AC_ARG_WITH(low-memory,
+ [ --with-low-memory Try to use less memory to compile to avoid
+ memory limitations.],
+ [with_lowmem=$withval],
+ [with_lowmem=no])
+if test "$with_lowmem" = "yes"
+then
+ if test "$ac_cv_prog_gcc" = "yes"
+ then
+ LM_CFLAGS="-fno-inline"
+ else
+ LM_CFLAGS="-O0"
+ fi
+else
+ LM_CFLAGS=""
+fi
+AC_SUBST(LM_CFLAGS)
+
+AC_ARG_WITH(comment,
+ [ --with-comment Comment about compilation environment.],
+ [with_comment=$withval],
+ [with_comment=no])
+if test "$with_comment" != "no"
+then
+ COMPILATION_COMMENT=$with_comment
+else
+ COMPILATION_COMMENT="Source distribution"
+fi
+AC_SUBST(COMPILATION_COMMENT)
+
+AC_MSG_CHECKING("need of special linking flags")
+if test "$IS_LINUX" = "true" -a "$ac_cv_prog_gcc" = "yes" -a "$all_is_static" != "yes"
+then
+ LDFLAGS="$LDFLAGS -rdynamic"
+ AC_MSG_RESULT("-rdynamic")
+else
+ AC_MSG_RESULT("none")
+fi
+
+dnl Checks for typedefs, structures, and compiler characteristics.
+AC_C_CONST
+AC_C_INLINE
+AC_TYPE_OFF_T
+AC_STRUCT_ST_RDEV
+AC_HEADER_TIME
+AC_STRUCT_TM
+# AC_CHECK_SIZEOF return 0 when it does not find the size of a
+# type. We want a error instead.
+AC_CHECK_SIZEOF(char, 1)
+if test "$ac_cv_sizeof_char" -eq 0
+then
+ AC_MSG_ERROR([No size for char type.
+A likely cause for this could be that there isn't any
+static libraries installed. You can verify this by checking if you have libm.a
+in /lib, /usr/lib or some other standard place. If this is the problem,
+install the static libraries and try again. If this isn't the problem,
+examine config.log for possible errors. If you want to report this, use
+'scripts/mysqlbug' and include at least the last 20 rows from config.log!])
+fi
+AC_CHECK_SIZEOF(char*, 4)
+AC_CHECK_SIZEOF(int, 4)
+if test "$ac_cv_sizeof_int" -eq 0
+then
+ AC_MSG_ERROR("No size for int type.")
+fi
+AC_CHECK_SIZEOF(long, 4)
+if test "$ac_cv_sizeof_long" -eq 0
+then
+ AC_MSG_ERROR("No size for long type.")
+fi
+AC_CHECK_SIZEOF(long long, 8)
+if test "$ac_cv_sizeof_long_long" -eq 0
+then
+ AC_MSG_ERROR("MySQL needs a long long type.")
+fi
+# off_t is not a builtin type
+MYSQL_CHECK_SIZEOF(off_t, 4)
+if test "$ac_cv_sizeof_off_t" -eq 0
+then
+ AC_MSG_ERROR("MySQL needs a off_t type.")
+fi
+# This always gives a warning. Ignore it unless you are cross compiling
+AC_C_BIGENDIAN
+#---START: Used in for client configure
+# Check base type of last arg to accept
+MYSQL_TYPE_ACCEPT
+
+#---END:
+# Find where the stack goes
+MYSQL_STACK_DIRECTION
+# We want to skip alloca on irix unconditionally. It may work on some version..
+MYSQL_FUNC_ALLOCA
+# Do struct timespec have members tv_sec or ts_sec
+MYSQL_TIMESPEC_TS
+# Do we have the tzname variable
+MYSQL_TZNAME
+# Do the system files define ulong
+MYSQL_CHECK_ULONG
+# Do the system files define uchar
+MYSQL_CHECK_UCHAR
+# Do the system files define uint
+MYSQL_CHECK_UINT
+# Check for fp_except in ieeefp.h
+MYSQL_CHECK_FP_EXCEPT
+# Check for IN_ADDR_T
+MYSQL_CHECK_IN_ADDR_T
+# Do the c++ compiler have a bool type
+MYSQL_CXX_BOOL
+# Check some common bugs with gcc 2.8.# on sparc
+if ! ( expr "$SYSTEM_TYPE" : ".*netware.*" > /dev/null ); then
+MYSQL_CHECK_LONGLONG_TO_FLOAT
+if test "$ac_cv_conv_longlong_to_float" != "yes"
+then
+ AC_MSG_ERROR([Your compiler cannot convert a longlong value to a float!
+If you are using gcc 2.8.# you should upgrade to egcs 1.0.3 or newer and try
+again]);
+fi
+fi
+MYSQL_PTHREAD_YIELD
+
+######################################################################
+# For readline/libedit (We simply move the mimimum amount of stuff from
+# the readline/libedit configure.in here)
+
+dnl Checks for header files.
+AC_CHECK_HEADERS(malloc.h sys/cdefs.h)
+
+dnl Checks for library functions.
+AC_FUNC_ALLOCA
+AC_PROG_GCC_TRADITIONAL
+AC_TYPE_SIGNAL
+AC_CHECK_FUNCS(re_comp regcomp strdup)
+
+AC_CHECK_HEADERS(vis.h)
+AC_CHECK_FUNCS(strlcat strlcpy)
+AC_CHECK_FUNCS(issetugid)
+AC_CHECK_FUNCS(fgetln)
+AC_CHECK_FUNCS(getline flockfile)
+
+# from old readline settting:
+
+MAKE_SHELL=/bin/sh
+AC_SUBST(MAKE_SHELL)
+
+# Already-done: stdlib.h string.h unistd.h termios.h
+AC_CHECK_HEADERS(varargs.h stdarg.h dirent.h locale.h ndir.h sys/dir.h \
+ sys/file.h sys/ndir.h sys/ptem.h sys/pte.h sys/select.h sys/stream.h \
+ sys/mman.h curses.h termcap.h termio.h termbits.h asm/termbits.h grp.h \
+paths.h semaphore.h)
+
+# Already-done: strcasecmp
+AC_CHECK_FUNCS(lstat putenv select setenv setlocale strcoll tcgetattr)
+
+AC_STAT_MACROS_BROKEN
+MYSQL_SIGNAL_CHECK
+MYSQL_CHECK_GETPW_FUNCS
+MYSQL_HAVE_TIOCGWINSZ
+MYSQL_HAVE_FIONREAD
+MYSQL_HAVE_TIOCSTAT
+MYSQL_STRUCT_DIRENT_D_INO
+MYSQL_TYPE_SIGHANDLER
+if test "$with_named_curses" = "no"
+then
+ MYSQL_CHECK_LIB_TERMCAP
+else
+ TERMCAP_LIB="$with_named_curses"
+fi
+AC_SUBST(TERMCAP_LIB)
+
+# End of readline/libedit stuff
+#########################################################################
+
+dnl Checks for library functions.
+
+#
+# The following code disables intrinsic function support while we test for
+# library functions. This is to avoid configure problems with Intel ecc
+# compiler
+
+ORG_CFLAGS="$CFLAGS"
+if test "$GCC" != "yes"; then
+ AC_SYS_COMPILER_FLAG(-nolib_inline,nolib_inline,CFLAGS,[],[])
+fi
+
+AC_FUNC_MMAP
+AC_TYPE_SIGNAL
+MYSQL_TYPE_QSORT
+AC_FUNC_UTIME_NULL
+AC_FUNC_VPRINTF
+
+AC_CHECK_FUNCS(alarm bcmp bfill bmove bzero chsize cuserid fchmod fcntl \
+ fconvert fdatasync finite fpresetsticky fpsetmask fsync ftruncate \
+ getcwd gethostbyaddr_r gethostbyname_r getpass getpassphrase getpwnam \
+ getpwuid getrlimit getrusage getwd gmtime_r index initgroups isnan \
+ localtime_r locking longjmp lrand48 madvise mallinfo memcpy memmove \
+ mkstemp mlockall perror poll pread pthread_attr_create clock_gettime \
+ pthread_attr_getstacksize pthread_attr_setprio pthread_attr_setschedparam \
+ pthread_attr_setstacksize pthread_condattr_create pthread_getsequence_np \
+ pthread_key_delete pthread_rwlock_rdlock pthread_setprio \
+ pthread_setprio_np pthread_setschedparam pthread_sigmask readlink \
+ realpath rename rint rwlock_init setupterm sighold sigset sigthreadmask \
+ snprintf socket stpcpy strcasecmp strerror strnlen strpbrk strstr strtol \
+ strtoll strtoul strtoull tell tempnam thr_setconcurrency vidattr)
+
+# isinf() could be a function or a macro (HPUX)
+AC_MSG_CHECKING(for isinf with <math.h>)
+AC_TRY_LINK([#include <math.h>], [float f = 0.0; isinf(f)],
+ AC_MSG_RESULT(yes) AC_DEFINE(HAVE_ISINF,,[isinf() macro or function]),
+ AC_MSG_RESULT(no))
+
+CFLAGS="$ORG_CFLAGS"
+
+# Sanity check: We chould not have any fseeko symbol unless
+# large_file_support=yes
+AC_CHECK_FUNC(fseeko,
+[if test "$large_file_support" = no -a "$IS_LINUX" = "true";
+then
+ AC_MSG_ERROR("Found fseeko symbol but large_file_support is not enabled!");
+fi]
+)
+
+my_save_LIBS="$LIBS"
+LIBS="$LIBS $LIBDL"
+AC_CHECK_FUNCS(dlopen dlerror)
+LIBS="$my_save_LIBS"
+
+# Check definition of gethostbyaddr_r (glibc2 defines this with 8 arguments)
+ac_save_CXXFLAGS="$CXXFLAGS"
+AC_CACHE_CHECK([style of gethost* routines], mysql_cv_gethost_style,
+AC_LANG_SAVE
+AC_LANG_CPLUSPLUS
+
+# Do not treat warnings as errors if we are linking against other libc
+# this is to work around gcc not being permissive on non-system includes
+# with respect to ANSI C++
+# We also remove the -fbranch-probabilities option as this will give warnings
+# about not profiled code, which confuses configure
+if test "$ac_cv_prog_gxx" = "yes" -a "$with_other_libc" = "no"
+then
+ CXXFLAGS=`echo "$CXXFLAGS -Werror" | sed 's/-fbranch-probabilities//'`
+fi
+
+AC_TRY_COMPILE(
+[#undef inline
+#if !defined(SCO) && !defined(__osf__) && !defined(_REENTRANT)
+#define _REENTRANT
+#endif
+#include <pthread.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+#include <netdb.h>],
+[int skr;
+ struct hostent *foo = gethostbyaddr_r((const char *) 0,
+ 0, 0, (struct hostent *) 0, (char *) NULL, 0, &skr); return (foo == 0);],
+mysql_cv_gethost_style=solaris, mysql_cv_gethost_style=other))
+AC_LANG_RESTORE
+CXXFLAGS="$ac_save_CXXFLAGS"
+if test "$mysql_cv_gethost_style" = "solaris"
+then
+ AC_DEFINE(HAVE_SOLARIS_STYLE_GETHOST)
+fi
+
+#---START: Used in for client configure
+
+# Check definition of gethostbyname_r (glibc2.0.100 is different from Solaris)
+ac_save_CXXFLAGS="$CXXFLAGS"
+AC_CACHE_CHECK([style of gethostname_r routines], mysql_cv_gethostname_style,
+AC_LANG_SAVE
+AC_LANG_CPLUSPLUS
+if test "$ac_cv_prog_gxx" = "yes" -a "$with_other_libc" = "no"
+then
+ CXXFLAGS=`echo "$CXXFLAGS -Werror" | sed 's/-fbranch-probabilities//'`
+fi
+AC_TRY_COMPILE(
+[#undef inline
+#if !defined(SCO) && !defined(__osf__) && !defined(_REENTRANT)
+#define _REENTRANT
+#endif
+#include <pthread.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+#include <netdb.h>],
+[int skr;
+
+ skr = gethostbyname_r((const char *) 0,
+ (struct hostent*) 0, (char*) 0, 0, (struct hostent **) 0, &skr);],
+mysql_cv_gethostname_style=glibc2, mysql_cv_gethostname_style=other))
+AC_LANG_RESTORE
+CXXFLAGS="$ac_save_CXXFLAGS"
+if test "$mysql_cv_gethostname_style" = "glibc2"
+then
+ AC_DEFINE(HAVE_GETHOSTBYNAME_R_GLIBC2_STYLE)
+fi
+
+# Check 3rd argument of getthostbyname_r
+ac_save_CXXFLAGS="$CXXFLAGS"
+AC_CACHE_CHECK([3 argument to gethostname_r routines], mysql_cv_gethostname_arg,
+AC_LANG_SAVE
+AC_LANG_CPLUSPLUS
+if test "$ac_cv_prog_gxx" = "yes" -a "$with_other_libc" = "no"
+then
+ CXXFLAGS=`echo "$CXXFLAGS -Werror" | sed 's/-fbranch-probabilities//'`
+fi
+AC_TRY_COMPILE(
+[#undef inline
+#if !defined(SCO) && !defined(__osf__) && !defined(_REENTRANT)
+#define _REENTRANT
+#endif
+#include <pthread.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+#include <netdb.h>],
+[int skr;
+
+ skr = gethostbyname_r((const char *) 0, (struct hostent*) 0, (struct hostent_data*) 0);],
+mysql_cv_gethostname_arg=hostent_data, mysql_cv_gethostname_arg=char))
+AC_LANG_RESTORE
+CXXFLAGS="$ac_save_CXXFLAGS"
+if test "$mysql_cv_gethostname_arg" = "hostent_data"
+then
+ AC_DEFINE(HAVE_GETHOSTBYNAME_R_RETURN_INT)
+fi
+
+
+if test "$with_mit_threads" = "no"
+then
+ # Check definition of pthread_getspecific
+ AC_CACHE_CHECK("args to pthread_getspecific", mysql_cv_getspecific_args,
+ AC_TRY_COMPILE(
+[#if !defined(SCO) && !defined(__osf__) && !defined(_REENTRANT)
+#define _REENTRANT
+#endif
+#define _POSIX_PTHREAD_SEMANTICS
+#include <pthread.h> ],
+[ void *pthread_getspecific(pthread_key_t key);
+pthread_getspecific((pthread_key_t) NULL); ],
+mysql_cv_getspecific_args=POSIX, mysql_cv_getspecific_args=other))
+ if test "$mysql_cv_getspecific_args" = "other"
+ then
+ AC_DEFINE(HAVE_NONPOSIX_PTHREAD_GETSPECIFIC)
+ fi
+
+ # Check definition of pthread_mutex_init
+ AC_CACHE_CHECK("args to pthread_mutex_init", mysql_cv_mutex_init_args,
+ AC_TRY_COMPILE(
+[#if !defined(SCO) && !defined(__osf__)
+#define _REENTRANT
+#endif
+#define _POSIX_PTHREAD_SEMANTICS
+#include <pthread.h> ],
+[
+ pthread_mutexattr_t attr;
+ pthread_mutex_t mp;
+ pthread_mutex_init(&mp,&attr); ],
+mysql_cv_mutex_init_args=POSIX, mysql_cv_mutex_init_args=other))
+ if test "$mysql_cv_mutex_init_args" = "other"
+ then
+ AC_DEFINE(HAVE_NONPOSIX_PTHREAD_MUTEX_INIT)
+ fi
+fi
+#---END:
+
+#---START: Used in for client configure
+# Check definition of readdir_r
+AC_CACHE_CHECK("args to readdir_r", mysql_cv_readdir_r,
+AC_TRY_LINK(
+[#if !defined(SCO) && !defined(__osf__)
+#define _REENTRANT
+#endif
+#define _POSIX_PTHREAD_SEMANTICS
+#include <pthread.h>
+#include <dirent.h>],
+[ int readdir_r(DIR *dirp, struct dirent *entry, struct dirent **result);
+readdir_r((DIR *) NULL, (struct dirent *) NULL, (struct dirent **) NULL); ],
+mysql_cv_readdir_r=POSIX, mysql_cv_readdir_r=other))
+if test "$mysql_cv_readdir_r" = "POSIX"
+then
+ AC_DEFINE(HAVE_READDIR_R)
+fi
+
+# Check definition of posix sigwait()
+AC_CACHE_CHECK("style of sigwait", mysql_cv_sigwait,
+AC_TRY_LINK(
+[#if !defined(SCO) && !defined(__osf__)
+#define _REENTRANT
+#endif
+#define _POSIX_PTHREAD_SEMANTICS
+#include <pthread.h>
+#include <signal.h>],
+[#ifndef _AIX
+sigset_t set;
+int sig;
+sigwait(&set,&sig);
+#endif],
+mysql_cv_sigwait=POSIX, mysql_cv_sigwait=other))
+if test "$mysql_cv_sigwait" = "POSIX"
+then
+ AC_DEFINE(HAVE_SIGWAIT)
+fi
+
+if test "$mysql_cv_sigwait" != "POSIX"
+then
+unset mysql_cv_sigwait
+# Check definition of posix sigwait()
+AC_CACHE_CHECK("style of sigwait", mysql_cv_sigwait,
+AC_TRY_LINK(
+[#if !defined(SCO) && !defined(__osf__)
+#define _REENTRANT
+#endif
+#define _POSIX_PTHREAD_SEMANTICS
+#include <pthread.h>
+#include <signal.h>],
+[sigset_t set;
+int sig;
+sigwait(&set);],
+mysql_cv_sigwait=NONPOSIX, mysql_cv_sigwait=other))
+if test "$mysql_cv_sigwait" = "NONPOSIX"
+then
+ AC_DEFINE(HAVE_NONPOSIX_SIGWAIT)
+fi
+fi
+#---END:
+
+# Check if pthread_attr_setscope() exists
+AC_CACHE_CHECK("for pthread_attr_setscope", mysql_cv_pthread_attr_setscope,
+AC_TRY_LINK(
+[#if !defined(SCO) && !defined(__osf__)
+#define _REENTRANT
+#endif
+#define _POSIX_PTHREAD_SEMANTICS
+#include <pthread.h>],
+[pthread_attr_t thr_attr;
+pthread_attr_setscope(&thr_attr,0);],
+mysql_cv_pthread_attr_setscope=yes, mysql_cv_pthread_attr_setscope=no))
+if test "$mysql_cv_pthread_attr_setscope" = "yes"
+then
+ AC_DEFINE(HAVE_PTHREAD_ATTR_SETSCOPE)
+fi
+
+# Check for bad includes
+AC_MSG_CHECKING("can netinet files be included")
+AC_TRY_COMPILE(
+[#include <sys/types.h>
+#include <sys/socket.h>
+#include <netinet/in_systm.h>
+#include <netinet/in.h>
+#include <netinet/ip.h>
+#include <netinet/tcp.h>],
+[ printf("1\n"); ],
+netinet_inc=yes, netinet_inc=no)
+if test "$netinet_inc" = "no"
+then
+ AC_DEFINE(HAVE_BROKEN_NETINET_INCLUDES)
+fi
+AC_MSG_RESULT("$netinet_inc")
+
+# Some usefull subst
+AC_SUBST(CC)
+AC_SUBST(GXX)
+
+# Output results
+AC_OUTPUT(Makefile dnl
+ , , [
+ test -z "$CONFIG_HEADERS" || echo timestamp > stamp-h
+ ])
diff --git a/ndb/demos/1-node/1-api-3/Ndb.cfg b/ndb/demos/1-node/1-api-3/Ndb.cfg
new file mode 100644
index 00000000000..61309af029e
--- /dev/null
+++ b/ndb/demos/1-node/1-api-3/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 3
+127.0.0.1 10000
diff --git a/ndb/demos/1-node/1-db-2/Ndb.cfg b/ndb/demos/1-node/1-db-2/Ndb.cfg
new file mode 100644
index 00000000000..9315950b67a
--- /dev/null
+++ b/ndb/demos/1-node/1-db-2/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 2
+127.0.0.1 10000
diff --git a/ndb/demos/1-node/1-mgm-1/Ndb.cfg b/ndb/demos/1-node/1-mgm-1/Ndb.cfg
new file mode 100644
index 00000000000..61d4c0ecc17
--- /dev/null
+++ b/ndb/demos/1-node/1-mgm-1/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 1
+127.0.0.1 10000
diff --git a/ndb/demos/1-node/1-mgm-1/template_config.ini b/ndb/demos/1-node/1-mgm-1/template_config.ini
new file mode 100644
index 00000000000..76bb7867e3c
--- /dev/null
+++ b/ndb/demos/1-node/1-mgm-1/template_config.ini
@@ -0,0 +1,70 @@
+###############################################################################
+#
+# Initial system configuration file for MySQL Cluster v3.1.0 (Demo 1)
+#
+###############################################################################
+
+[DB DEFAULT]
+NoOfReplicas: 1
+#LockPagesInMainMemory: Y
+StopOnError: Y
+#MaxNoOfConcurrentOperations: 1024
+#MaxNoOfConcurrentTransactions: 1024
+NoOfIndexPages: 1500
+NoOfDataPages: 5000
+#TimeBetweenLocalCheckpoints: 20
+#TimeBetweenGlobalCheckpoints: 1500
+#NoOfFragmentLogFiles: 8
+BackupMemory: 4M
+BackupDataBufferSize: 2M
+BackupLogBufferSize: 2M
+BackupWriteSize: 32k
+
+[COMPUTER]
+Id: 1
+ByteOrder: Little
+HostName: localhost
+
+[MGM]
+Id: 1
+ExecuteOnComputer: 1
+PortNumber: 10000
+PortNumberStats: 10001
+
+
+[DB]
+Id: 2
+ExecuteOnComputer: 1
+FileSystemPath: WRITE_PATH_TO_FILESYSTEM_2_HERE
+
+[API]
+Id: 3
+ExecuteOnComputer: 1
+
+# Designated MySQL Server API node id
+[API]
+Id: 11
+ExecuteOnComputer: 1
+
+[TCP DEFAULT]
+SendSignalId: N
+Compression: N
+Checksum: N
+SendBufferSize: 2000
+MaxReceiveSize: 2000
+
+[TCP]
+NodeId1: 1
+NodeId2: 2
+PortNumber: 10002
+
+[TCP]
+NodeId1: 2
+NodeId2: 3
+PortNumber: 10003
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 2
+NodeId2: 11
+PortNumber: 10011
diff --git a/ndb/demos/2-node/2-api-4/Ndb.cfg b/ndb/demos/2-node/2-api-4/Ndb.cfg
new file mode 100644
index 00000000000..1713a9b5893
--- /dev/null
+++ b/ndb/demos/2-node/2-api-4/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 4
+127.0.0.1 10000
diff --git a/ndb/demos/2-node/2-api-5/Ndb.cfg b/ndb/demos/2-node/2-api-5/Ndb.cfg
new file mode 100644
index 00000000000..faa2882eeea
--- /dev/null
+++ b/ndb/demos/2-node/2-api-5/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 5
+127.0.0.1 10000
diff --git a/ndb/demos/2-node/2-api-6/Ndb.cfg b/ndb/demos/2-node/2-api-6/Ndb.cfg
new file mode 100644
index 00000000000..bc2c4809453
--- /dev/null
+++ b/ndb/demos/2-node/2-api-6/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 6
+127.0.0.1 10000
diff --git a/ndb/demos/2-node/2-api-7/Ndb.cfg b/ndb/demos/2-node/2-api-7/Ndb.cfg
new file mode 100644
index 00000000000..4107fdb6c5e
--- /dev/null
+++ b/ndb/demos/2-node/2-api-7/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 7
+127.0.0.1 10000
diff --git a/ndb/demos/2-node/2-db-2/Ndb.cfg b/ndb/demos/2-node/2-db-2/Ndb.cfg
new file mode 100644
index 00000000000..9315950b67a
--- /dev/null
+++ b/ndb/demos/2-node/2-db-2/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 2
+127.0.0.1 10000
diff --git a/ndb/demos/2-node/2-db-3/Ndb.cfg b/ndb/demos/2-node/2-db-3/Ndb.cfg
new file mode 100644
index 00000000000..61309af029e
--- /dev/null
+++ b/ndb/demos/2-node/2-db-3/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 3
+127.0.0.1 10000
diff --git a/ndb/demos/2-node/2-mgm-1/Ndb.cfg b/ndb/demos/2-node/2-mgm-1/Ndb.cfg
new file mode 100644
index 00000000000..61d4c0ecc17
--- /dev/null
+++ b/ndb/demos/2-node/2-mgm-1/Ndb.cfg
@@ -0,0 +1,2 @@
+OwnProcessId 1
+127.0.0.1 10000
diff --git a/ndb/demos/2-node/2-mgm-1/template_config.ini b/ndb/demos/2-node/2-mgm-1/template_config.ini
new file mode 100644
index 00000000000..3edb909609a
--- /dev/null
+++ b/ndb/demos/2-node/2-mgm-1/template_config.ini
@@ -0,0 +1,157 @@
+###############################################################################
+#
+# Initial system configuration file for MySQL Cluster v3.1.0 (Demo 2)
+#
+###############################################################################
+
+[COMPUTER]
+Id: 1
+ByteOrder: Little
+HostName: localhost
+
+[COMPUTER]
+Id: 2
+ByteOrder: Little
+HostName: localhost
+
+[MGM]
+Id: 1
+ExecuteOnComputer: 1
+PortNumber: 10000
+PortNumberStats: 10001
+ArbitrationRank: 1
+
+[DB DEFAULT]
+NoOfReplicas: 2
+#LockPagesInMainMemory: N
+StopOnError: N
+#MaxNoOfConcurrentOperations: 1024
+#MaxNoOfConcurrentTransactions: 1024
+NoOfIndexPages: 200
+NoOfDataPages: 600
+#TimeBetweenLocalCheckpoints: 20
+#TimeBetweenGlobalCheckpoints: 1500
+#NoOfFragmentLogFiles: 8
+BackupMemory: 4M
+BackupDataBufferSize: 2M
+BackupLogBufferSize: 2M
+BackupWriteSize: 32k
+
+[DB]
+Id: 2
+ExecuteOnComputer: 1
+FileSystemPath: WRITE_PATH_TO_FILESYSTEM_2_HERE
+
+[DB]
+Id: 3
+ExecuteOnComputer: 2
+FileSystemPath: WRITE_PATH_TO_FILESYSTEM_3_HERE
+
+[API DEFAULT]
+ArbitrationRank: 1
+
+[API]
+Id: 4
+ExecuteOnComputer: 1
+
+[API]
+Id: 5
+ExecuteOnComputer: 1
+
+[API]
+Id: 6
+ExecuteOnComputer: 2
+
+[API]
+Id: 7
+ExecuteOnComputer: 2
+
+# Designated MySQL Server API node id
+[API]
+Id: 11
+ExecuteOnComputer: 1
+
+# Designated MySQL Server API node id
+[API]
+Id: 12
+ExecuteOnComputer: 2
+
+
+[TCP]
+NodeId1: 1
+NodeId2: 2
+PortNumber: 10002
+
+[TCP]
+NodeId1: 1
+NodeId2: 3
+PortNumber: 10003
+
+[TCP]
+NodeId1: 2
+NodeId2: 3
+PortNumber: 10004
+
+[TCP]
+NodeId1: 2
+NodeId2: 4
+PortNumber: 10005
+
+[TCP]
+NodeId1: 2
+NodeId2: 5
+PortNumber: 10006
+
+[TCP]
+NodeId1: 2
+NodeId2: 6
+PortNumber: 10007
+
+[TCP]
+NodeId1: 2
+NodeId2: 7
+PortNumber: 10008
+
+[TCP]
+NodeId1: 3
+NodeId2: 4
+PortNumber: 10009
+
+[TCP]
+NodeId1: 3
+NodeId2: 5
+PortNumber: 10010
+
+[TCP]
+NodeId1: 3
+NodeId2: 6
+PortNumber: 10011
+
+[TCP]
+NodeId1: 3
+NodeId2: 7
+PortNumber: 10012
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 2
+NodeId2: 11
+PortNumber: 10013
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 3
+NodeId2: 11
+PortNumber: 10014
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 2
+NodeId2: 12
+PortNumber: 10015
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 3
+NodeId2: 12
+PortNumber: 10016
diff --git a/ndb/demos/config-templates/config_template-1-REP.ini b/ndb/demos/config-templates/config_template-1-REP.ini
new file mode 100644
index 00000000000..71be3f2f53f
--- /dev/null
+++ b/ndb/demos/config-templates/config_template-1-REP.ini
@@ -0,0 +1,87 @@
+###############################################################################
+#
+# Initial system configuration file for MySQL Cluster v3.1.0 (Demo 1)
+#
+###############################################################################
+
+[DB DEFAULT]
+NoOfReplicas: 1
+StopOnError: Y
+NoOfIndexPages: 1500
+NoOfDataPages: 5000
+BackupMemory: 4M
+BackupDataBufferSize: 2M
+BackupLogBufferSize: 2M
+BackupWriteSize: 32k
+
+[COMPUTER]
+Id: 1
+ByteOrder: Little
+HostName: CHOOSE_HOSTNAME
+
+[EXTERNAL SYSTEM]
+Name: External
+
+[MGM]
+Id: 1
+ExecuteOnComputer: 1
+PortNumber: CHOOSE_PORT_BASE00
+PortNumberStats: CHOOSE_PORT_BASE01
+
+
+[DB]
+Id: 2
+ExecuteOnComputer: 1
+FileSystemPath: WRITE_PATH_TO_FILESYSTEM_2_HERE
+
+[API]
+Id: 3
+ExecuteOnComputer: 1
+
+[REP]
+Id: CHOOSE_REP_ID
+ExecuteOnComputer: 1
+
+[EXTERNAL REP]
+Id: CHOOSE_EXTREP_ID
+System: External
+
+# Designated MySQL Server API node id
+[API]
+Id: 11
+ExecuteOnComputer: 1
+
+[TCP DEFAULT]
+SendSignalId: N
+Compression: N
+Checksum: N
+SendBufferSize: 2000
+MaxReceiveSize: 2000
+
+[TCP]
+NodeId1: 1
+NodeId2: 2
+PortNumber: CHOOSE_PORT_BASE02
+
+[TCP]
+NodeId1: 2
+NodeId2: 3
+PortNumber: CHOOSE_PORT_BASE03
+
+[TCP]
+NodeId1: 2
+NodeId2: CHOOSE_REP_ID
+PortNumber: CHOOSE_PORT_BASE04
+
+[TCP]
+Hostname1: CHOOSE_HOSTNAME
+Hostname2: CHOOSE_EXTHOSTNAME
+NodeId1: CHOOSE_REP_ID
+NodeId2: External.CHOOSE_EXTREP_ID
+PortNumber: 10099
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 2
+NodeId2: 11
+PortNumber: CHOOSE_PORT_BASE11
diff --git a/ndb/demos/config-templates/config_template-4.ini b/ndb/demos/config-templates/config_template-4.ini
new file mode 100644
index 00000000000..e47c9037344
--- /dev/null
+++ b/ndb/demos/config-templates/config_template-4.ini
@@ -0,0 +1,336 @@
+###############################################################################
+#
+# 4-node system configuration file for MySQL Cluster
+#
+###############################################################################
+
+[DB DEFAULT]
+NoOfReplicas: 1
+StopOnError: N
+NoOfIndexPages: 1500
+NoOfDataPages: 5000
+BackupMemory: 4M
+BackupDataBufferSize: 2M
+BackupLogBufferSize: 2M
+BackupWriteSize: 32k
+
+[COMPUTER]
+Id: 1
+ByteOrder: Little
+HostName: CHOOSE_HOSTNAME_1
+
+[COMPUTER]
+Id: 2
+ByteOrder: Little
+HostName: CHOOSE_HOSTNAME_2
+
+[COMPUTER]
+Id: 3
+ByteOrder: Little
+HostName: CHOOSE_HOSTNAME_3
+
+[COMPUTER]
+Id: 4
+ByteOrder: Little
+HostName: CHOOSE_HOSTNAME_4
+
+[MGM]
+Id: 1
+ExecuteOnComputer: 1
+PortNumber: CHOOSE_PORT_BASE00
+PortNumberStats: CHOOSE_PORT_BASE01
+
+[DB]
+Id: 2
+ExecuteOnComputer: 1
+FileSystemPath: WRITE_PATH_TO_FILESYSTEM_1_HERE
+
+[DB]
+Id: 3
+ExecuteOnComputer: 2
+FileSystemPath: WRITE_PATH_TO_FILESYSTEM_2_HERE
+
+[DB]
+Id: 4
+ExecuteOnComputer: 3
+FileSystemPath: WRITE_PATH_TO_FILESYSTEM_3_HERE
+
+[DB]
+Id: 5
+ExecuteOnComputer: 4
+FileSystemPath: WRITE_PATH_TO_FILESYSTEM_4_HERE
+
+[API]
+Id: 6
+ExecuteOnComputer: 1
+
+[API]
+Id: 7
+ExecuteOnComputer: 2
+
+[API]
+Id: 8
+ExecuteOnComputer: 3
+
+[API]
+Id: 9
+ExecuteOnComputer: 4
+
+# Designated MySQL Server API node id
+[API]
+Id: 11
+ExecuteOnComputer: 1
+
+# Designated MySQL Server API node id
+[API]
+Id: 12
+ExecuteOnComputer: 2
+
+# Designated MySQL Server API node id
+[API]
+Id: 13
+ExecuteOnComputer: 3
+
+# Designated MySQL Server API node id
+[API]
+Id: 14
+ExecuteOnComputer: 4
+
+[TCP DEFAULT]
+SendSignalId: N
+Compression: N
+Checksum: N
+SendBufferSize: 2000
+MaxReceiveSize: 2000
+
+# Management server
+[TCP]
+NodeId1: 1
+NodeId2: 2
+PortNumber: CHOOSE_PORT_BASE02
+
+[TCP]
+NodeId1: 1
+NodeId2: 3
+PortNumber: CHOOSE_PORT_BASE03
+
+[TCP]
+NodeId1: 1
+NodeId2: 4
+PortNumber: CHOOSE_PORT_BASE04
+
+[TCP]
+NodeId1: 1
+NodeId2: 5
+PortNumber: CHOOSE_PORT_BASE05
+
+# Database cluster
+[TCP]
+NodeId1: 2
+NodeId2: 3
+PortNumber: CHOOSE_PORT_BASE06
+
+[TCP]
+NodeId1: 2
+NodeId2: 4
+PortNumber: CHOOSE_PORT_BASE07
+
+[TCP]
+NodeId1: 2
+NodeId2: 5
+PortNumber: CHOOSE_PORT_BASE08
+
+[TCP]
+NodeId1: 3
+NodeId2: 4
+PortNumber: CHOOSE_PORT_BASE09
+
+[TCP]
+NodeId1: 3
+NodeId2: 5
+PortNumber: CHOOSE_PORT_BASE10
+
+[TCP]
+NodeId1: 4
+NodeId2: 5
+PortNumber: CHOOSE_PORT_BASE11
+
+# API node 6
+[TCP]
+NodeId1: 6
+NodeId2: 2
+PortNumber: CHOOSE_PORT_BASE12
+
+[TCP]
+NodeId1: 6
+NodeId2: 3
+PortNumber: CHOOSE_PORT_BASE13
+
+[TCP]
+NodeId1: 6
+NodeId2: 4
+PortNumber: CHOOSE_PORT_BASE14
+
+[TCP]
+NodeId1: 6
+NodeId2: 5
+PortNumber: CHOOSE_PORT_BASE15
+
+# API node 7
+[TCP]
+NodeId1: 7
+NodeId2: 2
+PortNumber: CHOOSE_PORT_BASE16
+
+[TCP]
+NodeId1: 7
+NodeId2: 3
+PortNumber: CHOOSE_PORT_BASE17
+
+[TCP]
+NodeId1: 7
+NodeId2: 4
+PortNumber: CHOOSE_PORT_BASE18
+
+[TCP]
+NodeId1: 7
+NodeId2: 5
+PortNumber: CHOOSE_PORT_BASE19
+
+# API node 8
+[TCP]
+NodeId1: 8
+NodeId2: 2
+PortNumber: CHOOSE_PORT_BASE20
+
+[TCP]
+NodeId1: 8
+NodeId2: 3
+PortNumber: CHOOSE_PORT_BASE21
+
+[TCP]
+NodeId1: 8
+NodeId2: 4
+PortNumber: CHOOSE_PORT_BASE22
+
+[TCP]
+NodeId1: 8
+NodeId2: 5
+PortNumber: CHOOSE_PORT_BASE23
+
+# API node 9
+[TCP]
+NodeId1: 9
+NodeId2: 2
+PortNumber: CHOOSE_PORT_BASE24
+
+[TCP]
+NodeId1: 9
+NodeId2: 3
+PortNumber: CHOOSE_PORT_BASE25
+
+[TCP]
+NodeId1: 9
+NodeId2: 4
+PortNumber: CHOOSE_PORT_BASE26
+
+[TCP]
+NodeId1: 9
+NodeId2: 5
+PortNumber: CHOOSE_PORT_BASE27
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 2
+NodeId2: 11
+PortNumber: CHOOSE_PORT_BASE28
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 3
+NodeId2: 11
+PortNumber: CHOOSE_PORT_BASE29
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 4
+NodeId2: 11
+PortNumber: CHOOSE_PORT_BASE30
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 5
+NodeId2: 11
+PortNumber: CHOOSE_PORT_BASE31
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 2
+NodeId2: 12
+PortNumber: CHOOSE_PORT_BASE32
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 3
+NodeId2: 12
+PortNumber: CHOOSE_PORT_BASE33
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 4
+NodeId2: 12
+PortNumber: CHOOSE_PORT_BASE34
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 5
+NodeId2: 12
+PortNumber: CHOOSE_PORT_BASE35
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 2
+NodeId2: 13
+PortNumber: CHOOSE_PORT_BASE36
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 3
+NodeId2: 13
+PortNumber: CHOOSE_PORT_BASE37
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 4
+NodeId2: 13
+PortNumber: CHOOSE_PORT_BASE38
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 5
+NodeId2: 13
+PortNumber: CHOOSE_PORT_BASE39
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 2
+NodeId2: 14
+PortNumber: CHOOSE_PORT_BASE40
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 3
+NodeId2: 14
+PortNumber: CHOOSE_PORT_BASE41
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 4
+NodeId2: 14
+PortNumber: CHOOSE_PORT_BASE42
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 5
+NodeId2: 14
+PortNumber: CHOOSE_PORT_BASE43
diff --git a/ndb/demos/config-templates/config_template-install.ini b/ndb/demos/config-templates/config_template-install.ini
new file mode 100644
index 00000000000..e31906ba609
--- /dev/null
+++ b/ndb/demos/config-templates/config_template-install.ini
@@ -0,0 +1,64 @@
+###############################################################################
+#
+# Initial system configuration file for MySQL Cluster v3.1.0 (Demo 1)
+#
+###############################################################################
+
+[DB DEFAULT]
+NoOfReplicas: 1
+StopOnError: N
+NoOfIndexPages: 1500
+NoOfDataPages: 5000
+BackupMemory: 4M
+BackupDataBufferSize: 2M
+BackupLogBufferSize: 2M
+BackupWriteSize: 32k
+
+[COMPUTER]
+Id: 1
+ByteOrder: Little
+HostName: localhost
+
+[MGM]
+Id: 1
+ExecuteOnComputer: 1
+PortNumber: CHOOSE_PORT_BASE00
+PortNumberStats: CHOOSE_PORT_BASE01
+
+
+[DB]
+Id: 2
+ExecuteOnComputer: 1
+FileSystemPath: WRITE_PATH_TO_FILESYSTEM_2_HERE
+
+[API]
+Id: 3
+ExecuteOnComputer: 1
+
+# Designated MySQL Server API node id
+[API]
+Id: 11
+ExecuteOnComputer: 1
+
+[TCP DEFAULT]
+SendSignalId: N
+Compression: N
+Checksum: N
+SendBufferSize: 2000
+MaxReceiveSize: 2000
+
+[TCP]
+NodeId1: 1
+NodeId2: 2
+PortNumber: CHOOSE_PORT_BASE02
+
+[TCP]
+NodeId1: 2
+NodeId2: 3
+PortNumber: CHOOSE_PORT_BASE03
+
+# Designated MySQL Server API node connection
+[TCP]
+NodeId1: 2
+NodeId2: 11
+PortNumber: CHOOSE_PORT_BASE11
diff --git a/ndb/demos/run_demo1-PS-SS_common.sh b/ndb/demos/run_demo1-PS-SS_common.sh
new file mode 100644
index 00000000000..625e9655087
--- /dev/null
+++ b/ndb/demos/run_demo1-PS-SS_common.sh
@@ -0,0 +1,50 @@
+echo $NDB_HOST $NDB_EXTHOST
+
+NDB_PORT=$NDB_PORT_BASE"00"
+NDB_CONNECTSTRING_BASE="host=$NDB_HOST:$NDB_PORT;nodeid="
+
+# Edit file system path
+
+cd $NDB_DEMO
+sed -e s,"WRITE_PATH_TO_FILESYSTEM_2_HERE",$NDB_DEMO/filesystem,g \
+ -e s,"CHOOSE_HOSTNAME",$NDB_HOST,g\
+ -e s,"CHOOSE_EXTHOSTNAME",$NDB_EXTHOST,g\
+ -e s,"CHOOSE_PORT_BASE",$NDB_PORT_BASE,g\
+ -e s,"CHOOSE_REP_ID",$NDB_REP_ID,g\
+ -e s,"CHOOSE_EXTREP_ID",$NDB_EXTREP_ID,g\
+ < ../config-templates/config_template-1-REP.ini > config.ini
+
+# Start management server as deamon
+
+NDB_ID="1"
+NDB_CONNECTSTRING=$NDB_CONNECTSTRING_BASE$NDB_ID
+export NDB_CONNECTSTRING
+if mgmtsrvr -d -c config.ini ; then :; else
+ echo "Unable to start mgmtsrvr"
+ exit 1
+fi
+
+# Start database node
+
+NDB_ID="2"
+NDB_CONNECTSTRING=$NDB_CONNECTSTRING_BASE$NDB_ID
+export NDB_CONNECTSTRING
+xterm -T "$NDB_DEMO_NAME DB Node $NDB_ID" -geometry 80x10 -xrm *.hold:true -e ndb -i &
+
+# Start xterm for application programs
+
+NDB_ID="3"
+NDB_CONNECTSTRING=$NDB_CONNECTSTRING_BASE$NDB_ID
+export NDB_CONNECTSTRING
+xterm -T "$NDB_DEMO_NAME API Node $NDB_ID" -geometry 80x10 &
+
+# Start xterm for rep node
+
+NDB_ID=$NDB_REP_ID
+NDB_CONNECTSTRING=$NDB_CONNECTSTRING_BASE$NDB_ID
+export NDB_CONNECTSTRING
+xterm -T "$NDB_DEMO_NAME REP Node $NDB_ID" -geometry 80x10 -xrm *.hold:true -e ndb_rep &
+
+# Start management client
+
+xterm -T "$NDB_DEMO_NAME Mgmt Client" -geometry 80x10 -xrm *.hold:true -e mgmtclient $NDB_HOST $NDB_PORT &
diff --git a/ndb/demos/run_demo1-PS.sh b/ndb/demos/run_demo1-PS.sh
new file mode 100755
index 00000000000..82cfdd5e65b
--- /dev/null
+++ b/ndb/demos/run_demo1-PS.sh
@@ -0,0 +1,30 @@
+#!/bin/sh
+if [ -z "$MYSQLCLUSTER_TOP" ]; then
+ echo "MYSQLCLUSTER_TOP not set"
+ exit 1
+fi
+if [ -d "$MYSQLCLUSTER_TOP/ndb" ]; then :; else
+ echo "$MYSQLCLUSTER_TOP/ndb directory does not exist"
+ exit 1
+fi
+NDB_CONNECTSTRING=
+NDB_HOME=
+NDB_DEMO=$MYSQLCLUSTER_TOP/ndb/demos/1-node-PS
+
+NDB_PORT_BASE="102"
+NDB_REP_ID="5"
+NDB_EXTREP_ID="4"
+
+NDB_DEMO_NAME="Demo 1-PS MySQL Cluster"
+NDB_HOST1=$1
+NDB_HOST2=$2
+if [ -z "$NDB_HOST1" ]; then
+ NDB_HOST1=localhost
+fi
+if [ -z "$NDB_HOST2" ]; then
+ NDB_HOST2=localhost
+fi
+NDB_HOST=$NDB_HOST1
+NDB_EXTHOST=$NDB_HOST2
+
+source $MYSQLCLUSTER_TOP/ndb/demos/run_demo1-PS-SS_common.sh
diff --git a/ndb/demos/run_demo1-SS.sh b/ndb/demos/run_demo1-SS.sh
new file mode 100755
index 00000000000..5ede57c44c4
--- /dev/null
+++ b/ndb/demos/run_demo1-SS.sh
@@ -0,0 +1,30 @@
+#!/bin/sh
+if [ -z "$MYSQLCLUSTER_TOP" ]; then
+ echo "MYSQLCLUSTER_TOP not set"
+ exit 1
+fi
+if [ -d "$MYSQLCLUSTER_TOP/ndb" ]; then :; else
+ echo "$MYSQLCLUSTER_TOP/ndb directory does not exist"
+ exit 1
+fi
+NDB_CONNECTSTRING=
+NDB_HOME=
+NDB_DEMO=$MYSQLCLUSTER_TOP/ndb/demos/1-node-SS
+
+NDB_PORT_BASE="101"
+NDB_REP_ID="4"
+NDB_EXTREP_ID="5"
+
+NDB_DEMO_NAME="Demo 1-SS MySQL Cluster"
+NDB_HOST1=$1
+NDB_HOST2=$2
+if [ -z "$NDB_HOST1" ]; then
+ NDB_HOST1=localhost
+fi
+if [ -z "$NDB_HOST2" ]; then
+ NDB_HOST2=localhost
+fi
+NDB_HOST=$NDB_HOST2
+NDB_EXTHOST=$NDB_HOST1
+
+source $MYSQLCLUSTER_TOP/ndb/demos/run_demo1-PS-SS_common.sh
diff --git a/ndb/demos/run_demo1.sh b/ndb/demos/run_demo1.sh
new file mode 100755
index 00000000000..df6e3fc799d
--- /dev/null
+++ b/ndb/demos/run_demo1.sh
@@ -0,0 +1,41 @@
+#!/bin/sh
+if [ -z "$MYSQLCLUSTER_TOP" ]; then
+ echo "MYSQLCLUSTER_TOP not set"
+ exit 1
+fi
+if [ -d "$MYSQLCLUSTER_TOP/ndb" ]; then :; else
+ echo "$MYSQLCLUSTER_TOP/ndb directory does not exist"
+ exit 1
+fi
+NDB_CONNECTSTRING=
+NDB_HOME=
+ndb_demo=$MYSQLCLUSTER_TOP/ndb/demos
+
+# Edit file system path
+
+cd $ndb_demo/1-node/1-mgm-1
+sed -e s,"WRITE_PATH_TO_FILESYSTEM_2_HERE",$ndb_demo/1-node/1-db-2/filesystem,g \
+ < template_config.ini > config.ini
+
+# Start management server as deamon
+
+cd $ndb_demo/1-node/1-mgm-1
+if mgmtsrvr -d -c config.ini ; then :; else
+ echo "Unable to start mgmtsrvr"
+ exit 1
+fi
+
+# Start database node
+
+cd $ndb_demo/1-node/1-db-2
+xterm -T "Demo 1 NDB Cluster DB Node 2" -geometry 80x10 -xrm *.hold:true -e ndb -i &
+
+# Start xterm for application programs
+
+cd $ndb_demo/1-node/1-api-3
+xterm -T "Demo 1 NDB Cluster API Node 3" -geometry 80x10 &
+
+# Start management client
+
+cd $ndb_demo
+xterm -T "Demo 1 NDB Management Client" -geometry 80x10 -xrm *.hold:true -e mgmtclient localhost 10000 &
diff --git a/ndb/demos/run_demo2.sh b/ndb/demos/run_demo2.sh
new file mode 100755
index 00000000000..9bae7517d5f
--- /dev/null
+++ b/ndb/demos/run_demo2.sh
@@ -0,0 +1,54 @@
+#!/bin/sh
+if [ -z "$MYSQLCLUSTER_TOP" ]; then
+ echo "MYSQLCLUSTER_TOP not set"
+ exit 1
+fi
+if [ -d "$MYSQLCLUSTER_TOP/ndb" ]; then :; else
+ echo "$MYSQLCLUSTER_TOP/ndb directory does not exist"
+ exit 1
+fi
+NDB_CONNECTSTRING=
+NDB_HOME=
+ndb_demo=$MYSQLCLUSTER_TOP/ndb/demos
+
+# Edit file system path
+
+cd $ndb_demo/2-node/2-mgm-1
+sed -e s,"WRITE_PATH_TO_FILESYSTEM_2_HERE",$ndb_demo/2-node/2-db-2/filesystem,g \
+ -e s,"WRITE_PATH_TO_FILESYSTEM_3_HERE",$ndb_demo/2-node/2-db-3/filesystem,g \
+ < template_config.ini > config.ini
+
+# Start management server as deamon
+
+cd $ndb_demo/2-node/2-mgm-1
+if mgmtsrvr -d -c config.ini ; then :; else
+ echo "Unable to start mgmtsrvr"
+ exit 1
+fi
+
+#xterm -T "Demo 2 NDB Management Server" -geometry 80x10 -xrm *.hold:true -e mgmtsrvr -c config.ini &
+
+# Start database node
+
+cd $ndb_demo/2-node/2-db-2
+xterm -T "Demo 2 NDB Cluster DB Node 2" -geometry 80x10 -xrm *.hold:true -e ndb -i &
+
+# Start database node
+
+cd $ndb_demo/2-node/2-db-3
+xterm -T "Demo 2 NDB Cluster DB Node 3" -geometry 80x10 -xrm *.hold:true -e ndb -i &
+
+# Start xterm for application programs
+
+cd $ndb_demo/2-node/2-api-4
+xterm -T "Demo 2 NDB Cluster API Node 4" -geometry 80x10 &
+
+# Start xterm for application programs
+
+cd $ndb_demo/2-node/2-api-5
+xterm -T "Demo 2 NDB Cluster API Node 5" -geometry 80x10 &
+
+# Start management client
+
+cd $ndb_demo
+xterm -T "Demo 2 NDB Management Client" -geometry 80x10 -xrm *.hold:true -e mgmtclient localhost 10000 &
diff --git a/ndb/docs/Makefile b/ndb/docs/Makefile
new file mode 100644
index 00000000000..a2139b66044
--- /dev/null
+++ b/ndb/docs/Makefile
@@ -0,0 +1,97 @@
+include .defs.mk
+#
+# hack before full autoconf
+replace-targets := all clean
+first-docs: all
+
+include $(NDB_TOP)/Epilogue.mk
+
+all: ndbapidoc mgmapidoc
+
+DOXYGEN = doxygen
+DOXYTOP = $(shell cd $(NDB_TOP); pwd)/docs
+DOXYDIR = $(DOXYTOP)/doxygen
+DOXYTMP = $(DOXYTOP)/.doxytmp
+DOXYOUT = $(DOXYTOP)/.doxyout
+
+clean:
+ rm -rf ndbapi.pdf ndbapi.html mgmapi.pdf mgmapi.html
+ rm -rf $(DOXYTMP) $(DOXYOUT)
+
+###
+#
+# NDB API Programmer's Guide
+#
+ndbapidoc: ndbapi.pdf
+
+ndbapi.pdf: $(NDB_TOP)/include/ndb_version.h
+ @set -x; \
+ rm -rf ndbapi.pdf ndbapi.html; \
+ rm -rf $(DOXYTMP) $(DOXYOUT); \
+ mkdir -p $(DOXYTMP) $(DOXYOUT); \
+ (cd $(NDB_TOP)/include/ndbapi && \
+ find . -type f -print | \
+ grep -v /SCCS | \
+ cpio -pdm $(DOXYTMP)); \
+ (cd $(NDB_TOP)/examples && \
+ cp -p */*.[ch]pp $(DOXYTMP)); \
+ $(DOXYDIR)/predoxy.pl; \
+ mv footer.html $(DOXYTMP); \
+ (cd $(DOXYTMP) && \
+ $(DOXYGEN) $(DOXYDIR)/Doxyfile.ndbapi); \
+ $(DOXYDIR)/postdoxy.pl $(DOXYOUT)/ndbapi.latex "NDB API Programmer Guide"; \
+ (cd $(DOXYOUT) && \
+ find ndbapi.html -print | cpio -pdm $(DOXYTOP)); \
+ (cd $(DOXYOUT)/ndbapi.latex && \
+ pdflatex refman.tex && makeindex refman && pdflatex refman.tex && \
+ cp -p refman.pdf $(DOXYTOP)/ndbapi.pdf);
+
+###
+#
+# MGM API Guide
+#
+mgmapidoc: mgmapi.pdf
+
+mgmapi.pdf: $(NDB_TOP)/include/ndb_version.h
+ @set -x; \
+ rm -rf mgmapi.pdf mgmapi.html; \
+ rm -rf $(DOXYTMP) $(DOXYOUT); \
+ mkdir -p $(DOXYTMP) $(DOXYOUT); \
+ (cd $(NDB_TOP)/include/mgmapi && \
+ find . -type f -print | \
+ grep -v /SCCS | \
+ cpio -pdm $(DOXYTMP)); \
+ $(DOXYDIR)/predoxy.pl; \
+ mv footer.html $(DOXYTMP); \
+ (cd $(DOXYTMP) && \
+ $(DOXYGEN) $(DOXYDIR)/Doxyfile.mgmapi); \
+ $(DOXYDIR)/postdoxy.pl $(OUTDIR)/mgmapi.latex "NDB Cluster MGM API Guide"; \
+ (cd $(DOXYOUT) && \
+ find mgmapi.html -print | cpio -pdm $(DOXYTOP)); \
+ (cd $(DOXYOUT)/mgmapi.latex && \
+ pdflatex refman.tex && makeindex refman && pdflatex refman.tex && \
+ cp -p refman.pdf $(DOXYTOP)/mgmapi.pdf);
+
+###
+#
+# Complete Source Browser except for
+# ndbapi odbc test tools win32 lib examples docs CVS config bin
+# include/ndbapi
+# include/newtonapi src/newtonapi
+# include/mgmapi src/mgmapi
+# src/client
+ndbdoc: DUMMY
+ mkdir -p $(OUTDIR)
+ cd $(NDB_TOP) ; $(DOXYGEN) $(DOXYDIR)/Doxyfile.ndb
+
+###
+#
+# odbcdoc - Complete Source Browser for NDB ODBC (src/client/odbc)
+
+odbcdoc: DUMMY
+ mkdir -p $(OUTDIR)
+ cd $(NDB_TOP) ; $(DOXYGEN) $(DOXYDIR)/Doxyfile.odbc
+
+testdoc: DUMMY
+ mkdir -p $(OUTDIR)
+ cd $(NDB_TOP) ; $(DOXYGEN) $(DOXYDIR)/Doxyfile.test
diff --git a/ndb/docs/README b/ndb/docs/README
new file mode 100644
index 00000000000..262e9003aca
--- /dev/null
+++ b/ndb/docs/README
@@ -0,0 +1,30 @@
+Create MySQL Cluster user documentation from source code
+--------------------------------------------------------
+(All these require Doxygen.)
+
+* make clean
+ Remove all generated documentation and tmp files
+
+* make ndbapidoc
+ Makes the NDB API Programmer's Guide (in HTML)
+
+* make ndbapipdf
+ Makes the NDB API Programmer Guide (in PDF)
+
+* make mgmapidoc
+ Makes the MGM API Reference Manual (in HTML)
+
+* make mgmapipdf
+ Makes the MGM API Reference Manual (in PDF)
+
+* make ndbdoc
+ Makes source code browser for NDB Cluster (in HTML)
+ (Requires Graphviz.)
+
+Doxygen and Graphviz can be found at:
+ http://www.doxygen.org
+or at (for Red Hat 9.0 RPMs):
+ http://dentrassi.de/download/doxygen/
+
+--
+lars@mysql.com
diff --git a/ndb/docs/doxygen/Doxyfile.mgmapi b/ndb/docs/doxygen/Doxyfile.mgmapi
new file mode 100644
index 00000000000..4287b37fd97
--- /dev/null
+++ b/ndb/docs/doxygen/Doxyfile.mgmapi
@@ -0,0 +1,877 @@
+# Doxyfile 1.2.12
+
+# This file describes the settings to be used by the documentation system
+# doxygen (www.doxygen.org) for a project
+#
+# All text after a hash (#) is considered a comment and will be ignored
+# The format is:
+# TAG = value [value, ...]
+# For lists items can also be appended using:
+# TAG += value [value, ...]
+# Values that contain spaces should be placed between quotes (" ")
+
+#---------------------------------------------------------------------------
+# General configuration options
+#---------------------------------------------------------------------------
+DETAILS_AT_TOP = yes
+HIDE_FRIEND_COMPOUNDS = yes
+
+# The PROJECT_NAME tag is a single word (or a sequence of words surrounded
+# by quotes) that should identify the project.
+
+PROJECT_NAME =
+
+# The PROJECT_NUMBER tag can be used to enter a project or revision number.
+# This could be handy for archiving the generated documentation or
+# if some version control system is used.
+
+PROJECT_NUMBER =
+
+# The OUTPUT_DIRECTORY tag is used to specify the (relative or absolute)
+# base path where the generated documentation will be put.
+# If a relative path is entered, it will be relative to the location
+# where doxygen was started. If left blank the current directory will be used.
+
+OUTPUT_DIRECTORY =
+
+# The OUTPUT_LANGUAGE tag is used to specify the language in which all
+# documentation generated by doxygen is written. Doxygen will use this
+# information to generate all constant output in the proper language.
+# The default language is English, other supported languages are:
+# Brazilian, Chinese, Croatian, Czech, Danish, Dutch, Finnish, French,
+# German, Hungarian, Italian, Japanese, Korean, Norwegian, Polish,
+# Portuguese, Romanian, Russian, Slovak, Slovene, Spanish and Swedish.
+
+OUTPUT_LANGUAGE = English
+
+# If the EXTRACT_ALL tag is set to YES doxygen will assume all entities in
+# documentation are documented, even if no documentation was available.
+# Private class members and static file members will be hidden unless
+# the EXTRACT_PRIVATE and EXTRACT_STATIC tags are set to YES
+
+EXTRACT_ALL = NO
+
+# If the EXTRACT_PRIVATE tag is set to YES all private members of a class
+# will be included in the documentation.
+
+EXTRACT_PRIVATE = NO
+
+# If the EXTRACT_STATIC tag is set to YES all static members of a file
+# will be included in the documentation.
+
+EXTRACT_STATIC = NO
+
+# If the HIDE_UNDOC_MEMBERS tag is set to YES, Doxygen will hide all
+# undocumented members of documented classes, files or namespaces.
+# If set to NO (the default) these members will be included in the
+# various overviews, but no documentation section is generated.
+# This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_MEMBERS = NO
+
+# If the HIDE_UNDOC_CLASSES tag is set to YES, Doxygen will hide all
+# undocumented classes that are normally visible in the class hierarchy.
+# If set to NO (the default) these class will be included in the various
+# overviews. This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_CLASSES = NO
+
+# If the BRIEF_MEMBER_DESC tag is set to YES (the default) Doxygen will
+# include brief member descriptions after the members that are listed in
+# the file and class documentation (similar to JavaDoc).
+# Set to NO to disable this.
+
+BRIEF_MEMBER_DESC = YES
+
+# If the REPEAT_BRIEF tag is set to YES (the default) Doxygen will prepend
+# the brief description of a member or function before the detailed description.
+# Note: if both HIDE_UNDOC_MEMBERS and BRIEF_MEMBER_DESC are set to NO, the
+# brief descriptions will be completely suppressed.
+
+REPEAT_BRIEF = YES
+
+# If the ALWAYS_DETAILED_SEC and REPEAT_BRIEF tags are both set to YES then
+# Doxygen will generate a detailed section even if there is only a brief
+# description.
+
+ALWAYS_DETAILED_SEC = NO
+
+# If the FULL_PATH_NAMES tag is set to YES then Doxygen will prepend the full
+# path before files name in the file list and in the header files. If set
+# to NO the shortest path that makes the file name unique will be used.
+
+FULL_PATH_NAMES = NO
+
+# If the FULL_PATH_NAMES tag is set to YES then the STRIP_FROM_PATH tag
+# can be used to strip a user defined part of the path. Stripping is
+# only done if one of the specified strings matches the left-hand part of
+# the path. It is allowed to use relative paths in the argument list.
+
+STRIP_FROM_PATH =
+
+# The INTERNAL_DOCS tag determines if documentation
+# that is typed after a \internal command is included. If the tag is set
+# to NO (the default) then the documentation will be excluded.
+# Set it to YES to include the internal documentation.
+
+INTERNAL_DOCS = NO
+
+# Setting the STRIP_CODE_COMMENTS tag to YES (the default) will instruct
+# doxygen to hide any special comment blocks from generated source code
+# fragments. Normal C and C++ comments will always remain visible.
+
+STRIP_CODE_COMMENTS = YES
+
+# If the CASE_SENSE_NAMES tag is set to NO then Doxygen will only generate
+# file names in lower case letters. If set to YES upper case letters are also
+# allowed. This is useful if you have classes or files whose names only differ
+# in case and if your file system supports case sensitive file names. Windows
+# users are adviced to set this option to NO.
+
+CASE_SENSE_NAMES = YES
+
+# If the SHORT_NAMES tag is set to YES, doxygen will generate much shorter
+# (but less readable) file names. This can be useful is your file systems
+# doesn't support long names like on DOS, Mac, or CD-ROM.
+
+SHORT_NAMES = NO
+
+# If the HIDE_SCOPE_NAMES tag is set to NO (the default) then Doxygen
+# will show members with their full class and namespace scopes in the
+# documentation. If set to YES the scope will be hidden.
+
+HIDE_SCOPE_NAMES = NO
+
+# If the VERBATIM_HEADERS tag is set to YES (the default) then Doxygen
+# will generate a verbatim copy of the header file for each class for
+# which an include is specified. Set to NO to disable this.
+
+VERBATIM_HEADERS = YES
+
+# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
+# will put list of the files that are included by a file in the documentation
+# of that file.
+
+SHOW_INCLUDE_FILES = YES
+
+# If the JAVADOC_AUTOBRIEF tag is set to YES then Doxygen
+# will interpret the first line (until the first dot) of a JavaDoc-style
+# comment as the brief description. If set to NO, the JavaDoc
+# comments will behave just like the Qt-style comments (thus requiring an
+# explict @brief command for a brief description.
+
+JAVADOC_AUTOBRIEF = NO
+
+# If the INHERIT_DOCS tag is set to YES (the default) then an undocumented
+# member inherits the documentation from any documented member that it
+# reimplements.
+
+INHERIT_DOCS = YES
+
+# If the INLINE_INFO tag is set to YES (the default) then a tag [inline]
+# is inserted in the documentation for inline members.
+
+INLINE_INFO = YES
+
+# If the SORT_MEMBER_DOCS tag is set to YES (the default) then doxygen
+# will sort the (detailed) documentation of file and class members
+# alphabetically by member name. If set to NO the members will appear in
+# declaration order.
+
+SORT_MEMBER_DOCS = YES
+
+# If member grouping is used in the documentation and the DISTRIBUTE_GROUP_DOC
+# tag is set to YES, then doxygen will reuse the documentation of the first
+# member in the group (if any) for the other members of the group. By default
+# all members of a group must be documented explicitly.
+
+DISTRIBUTE_GROUP_DOC = NO
+
+# The TAB_SIZE tag can be used to set the number of spaces in a tab.
+# Doxygen uses this value to replace tabs by spaces in code fragments.
+
+TAB_SIZE = 8
+
+# The GENERATE_TODOLIST tag can be used to enable (YES) or
+# disable (NO) the todo list. This list is created by putting \todo
+# commands in the documentation.
+
+GENERATE_TODOLIST = YES
+
+# The GENERATE_TESTLIST tag can be used to enable (YES) or
+# disable (NO) the test list. This list is created by putting \test
+# commands in the documentation.
+
+GENERATE_TESTLIST = YES
+
+# The GENERATE_BUGLIST tag can be used to enable (YES) or
+# disable (NO) the bug list. This list is created by putting \bug
+# commands in the documentation.
+
+GENERATE_BUGLIST = YES
+
+# This tag can be used to specify a number of aliases that acts
+# as commands in the documentation. An alias has the form "name=value".
+# For example adding "sideeffect=\par Side Effects:\n" will allow you to
+# put the command \sideeffect (or @sideeffect) in the documentation, which
+# will result in a user defined paragraph with heading "Side Effects:".
+# You can put \n's in the value part of an alias to insert newlines.
+
+ALIASES =
+
+# The ENABLED_SECTIONS tag can be used to enable conditional
+# documentation sections, marked by \if sectionname ... \endif.
+
+ENABLED_SECTIONS =
+
+# The MAX_INITIALIZER_LINES tag determines the maximum number of lines
+# the initial value of a variable or define consist of for it to appear in
+# the documentation. If the initializer consists of more lines than specified
+# here it will be hidden. Use a value of 0 to hide initializers completely.
+# The appearance of the initializer of individual variables and defines in the
+# documentation can be controlled using \showinitializer or \hideinitializer
+# command in the documentation regardless of this setting.
+
+MAX_INITIALIZER_LINES = 30
+
+# Set the OPTIMIZE_OUTPUT_FOR_C tag to YES if your project consists of C sources
+# only. Doxygen will then generate output that is more tailored for C.
+# For instance some of the names that are used will be different. The list
+# of all members will be omitted, etc.
+
+OPTIMIZE_OUTPUT_FOR_C = NO
+
+# Set the SHOW_USED_FILES tag to NO to disable the list of files generated
+# at the bottom of the documentation of classes and structs. If set to YES the
+# list will mention the files that were used to generate the documentation.
+
+SHOW_USED_FILES = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to warning and progress messages
+#---------------------------------------------------------------------------
+
+# The QUIET tag can be used to turn on/off the messages that are generated
+# by doxygen. Possible values are YES and NO. If left blank NO is used.
+
+QUIET = NO
+
+# The WARNINGS tag can be used to turn on/off the warning messages that are
+# generated by doxygen. Possible values are YES and NO. If left blank
+# NO is used.
+
+WARNINGS = YES
+
+# If WARN_IF_UNDOCUMENTED is set to YES, then doxygen will generate warnings
+# for undocumented members. If EXTRACT_ALL is set to YES then this flag will
+# automatically be disabled.
+
+WARN_IF_UNDOCUMENTED = YES
+
+# The WARN_FORMAT tag determines the format of the warning messages that
+# doxygen can produce. The string should contain the $file, $line, and $text
+# tags, which will be replaced by the file and line number from which the
+# warning originated and the warning text.
+
+WARN_FORMAT =
+
+# The WARN_LOGFILE tag can be used to specify a file to which warning
+# and error messages should be written. If left blank the output is written
+# to stderr.
+
+WARN_LOGFILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the input files
+#---------------------------------------------------------------------------
+
+# The INPUT tag can be used to specify the files and/or directories that contain
+# documented source files. You may enter file names like "myfile.cpp" or
+# directories like "/usr/src/myproject". Separate the files or directories
+# with spaces.
+
+INPUT = .
+
+# If the value of the INPUT tag contains directories, you can use the
+# FILE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank file matching one of the following patterns are included:
+# *.c *.cc *.cxx *.c++ *.java *.ii *.ixx *.ipp *.i++ *.inl *.h *.hh *.hxx *.hpp
+# *.h++ *.idl
+
+FILE_PATTERNS =
+
+# The RECURSIVE tag can be used to turn specify whether or not subdirectories
+# should be searched for input files as well. Possible values are YES and NO.
+# If left blank NO is used.
+
+RECURSIVE = NO
+
+# The EXCLUDE tag can be used to specify files and/or directories that should
+# excluded from the INPUT source files. This way you can easily exclude a
+# subdirectory from a directory tree whose root is specified with the INPUT tag.
+
+EXCLUDE =
+
+# If the value of the INPUT tag contains directories, you can use the
+# EXCLUDE_PATTERNS tag to specify one or more wildcard patterns to exclude
+# certain files from those directories.
+
+EXCLUDE_PATTERNS =
+
+# The EXAMPLE_PATH tag can be used to specify one or more files or
+# directories that contain example code fragments that are included (see
+# the \include command).
+
+EXAMPLE_PATH = .
+
+# If the value of the EXAMPLE_PATH tag contains directories, you can use the
+# EXAMPLE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank all files are included.
+
+EXAMPLE_PATTERNS =
+
+# If the EXAMPLE_RECURSIVE tag is set to YES then subdirectories will be
+# searched for input files to be used with the \include or \dontinclude
+# commands irrespective of the value of the RECURSIVE tag.
+# Possible values are YES and NO. If left blank NO is used.
+
+EXAMPLE_RECURSIVE = NO
+
+# The IMAGE_PATH tag can be used to specify one or more files or
+# directories that contain image that are included in the documentation (see
+# the \image command).
+
+IMAGE_PATH =
+
+# The INPUT_FILTER tag can be used to specify a program that doxygen should
+# invoke to filter for each input file. Doxygen will invoke the filter program
+# by executing (via popen()) the command <filter> <input-file>, where <filter>
+# is the value of the INPUT_FILTER tag, and <input-file> is the name of an
+# input file. Doxygen will then use the output that the filter program writes
+# to standard output.
+
+INPUT_FILTER =
+
+# If the FILTER_SOURCE_FILES tag is set to YES, the input filter (if set using
+# INPUT_FILTER) will be used to filter the input files when producing source
+# files to browse.
+
+FILTER_SOURCE_FILES = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to source browsing
+#---------------------------------------------------------------------------
+
+# If the SOURCE_BROWSER tag is set to YES then a list of source files will
+# be generated. Documented entities will be cross-referenced with these sources.
+
+SOURCE_BROWSER = NO
+
+# Setting the INLINE_SOURCES tag to YES will include the body
+# of functions and classes directly in the documentation.
+
+INLINE_SOURCES = NO
+
+# If the REFERENCED_BY_RELATION tag is set to YES (the default)
+# then for each documented function all documented
+# functions referencing it will be listed.
+
+REFERENCED_BY_RELATION = YES
+
+# If the REFERENCES_RELATION tag is set to YES (the default)
+# then for each documented function all documented entities
+# called/used by that function will be listed.
+
+REFERENCES_RELATION = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to the alphabetical class index
+#---------------------------------------------------------------------------
+
+# If the ALPHABETICAL_INDEX tag is set to YES, an alphabetical index
+# of all compounds will be generated. Enable this if the project
+# contains a lot of classes, structs, unions or interfaces.
+
+ALPHABETICAL_INDEX = NO
+
+# If the alphabetical index is enabled (see ALPHABETICAL_INDEX) then
+# the COLS_IN_ALPHA_INDEX tag can be used to specify the number of columns
+# in which this list will be split (can be a number in the range [1..20])
+
+COLS_IN_ALPHA_INDEX = 5
+
+# In case all classes in a project start with a common prefix, all
+# classes will be put under the same header in the alphabetical index.
+# The IGNORE_PREFIX tag can be used to specify one or more prefixes that
+# should be ignored while generating the index headers.
+
+IGNORE_PREFIX =
+
+#---------------------------------------------------------------------------
+# configuration options related to the HTML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_HTML tag is set to YES (the default) Doxygen will
+# generate HTML output.
+
+GENERATE_HTML = YES
+
+# The HTML_OUTPUT tag is used to specify where the HTML docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `html' will be used as the default path.
+
+HTML_OUTPUT = ../.doxyout/mgmapi.html
+
+# The HTML_HEADER tag can be used to specify a personal HTML header for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard header.
+
+HTML_HEADER =
+
+# The HTML_FOOTER tag can be used to specify a personal HTML footer for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard footer.
+
+HTML_FOOTER = footer.html
+
+# The HTML_STYLESHEET tag can be used to specify a user defined cascading
+# style sheet that is used by each HTML page. It can be used to
+# fine-tune the look of the HTML output. If the tag is left blank doxygen
+# will generate a default style sheet
+
+HTML_STYLESHEET =
+
+# If the HTML_ALIGN_MEMBERS tag is set to YES, the members of classes,
+# files or namespaces will be aligned in HTML using tables. If set to
+# NO a bullet list will be used.
+
+HTML_ALIGN_MEMBERS = YES
+
+# If the GENERATE_HTMLHELP tag is set to YES, additional index files
+# will be generated that can be used as input for tools like the
+# Microsoft HTML help workshop to generate a compressed HTML help file (.chm)
+# of the generated HTML documentation.
+
+GENERATE_HTMLHELP = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the GENERATE_CHI flag
+# controls if a separate .chi index file is generated (YES) or that
+# it should be included in the master .chm file (NO).
+
+GENERATE_CHI = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the BINARY_TOC flag
+# controls whether a binary table of contents is generated (YES) or a
+# normal table of contents (NO) in the .chm file.
+
+BINARY_TOC = NO
+
+# The TOC_EXPAND flag can be set to YES to add extra items for group members
+# to the contents of the Html help documentation and to the tree view.
+
+TOC_EXPAND = NO
+
+# The DISABLE_INDEX tag can be used to turn on/off the condensed index at
+# top of each HTML page. The value NO (the default) enables the index and
+# the value YES disables it.
+
+DISABLE_INDEX = NO
+
+# This tag can be used to set the number of enum values (range [1..20])
+# that doxygen will group on one line in the generated HTML documentation.
+
+ENUM_VALUES_PER_LINE = 4
+
+# If the GENERATE_TREEVIEW tag is set to YES, a side panel will be
+# generated containing a tree-like index structure (just like the one that
+# is generated for HTML Help). For this to work a browser that supports
+# JavaScript and frames is required (for instance Mozilla, Netscape 4.0+,
+# or Internet explorer 4.0+). Note that for large projects the tree generation
+# can take a very long time. In such cases it is better to disable this feature.
+# Windows users are probably better off using the HTML help feature.
+
+GENERATE_TREEVIEW = NO
+
+# If the treeview is enabled (see GENERATE_TREEVIEW) then this tag can be
+# used to set the initial width (in pixels) of the frame in which the tree
+# is shown.
+
+TREEVIEW_WIDTH = 250
+
+#---------------------------------------------------------------------------
+# configuration options related to the LaTeX output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_LATEX tag is set to YES (the default) Doxygen will
+# generate Latex output.
+
+GENERATE_LATEX = YES
+
+# The LATEX_OUTPUT tag is used to specify where the LaTeX docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `latex' will be used as the default path.
+
+LATEX_OUTPUT = ../.doxyout/mgmapi.latex
+
+# If the COMPACT_LATEX tag is set to YES Doxygen generates more compact
+# LaTeX documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_LATEX = NO
+
+# The PAPER_TYPE tag can be used to set the paper type that is used
+# by the printer. Possible values are: a4, a4wide, letter, legal and
+# executive. If left blank a4wide will be used.
+
+PAPER_TYPE =
+
+# The EXTRA_PACKAGES tag can be to specify one or more names of LaTeX
+# packages that should be included in the LaTeX output.
+
+EXTRA_PACKAGES =
+
+# The LATEX_HEADER tag can be used to specify a personal LaTeX header for
+# the generated latex document. The header should contain everything until
+# the first chapter. If it is left blank doxygen will generate a
+# standard header. Notice: only use this tag if you know what you are doing!
+
+LATEX_HEADER = ../doxygen/header.mgmapi.tex
+
+# If the PDF_HYPERLINKS tag is set to YES, the LaTeX that is generated
+# is prepared for conversion to pdf (using ps2pdf). The pdf file will
+# contain links (just like the HTML output) instead of page references
+# This makes the output suitable for online browsing using a pdf viewer.
+
+PDF_HYPERLINKS = YES
+
+# If the USE_PDFLATEX tag is set to YES, pdflatex will be used instead of
+# plain latex in the generated Makefile. Set this option to YES to get a
+# higher quality PDF documentation.
+
+USE_PDFLATEX = YES
+
+# If the LATEX_BATCHMODE tag is set to YES, doxygen will add the \\batchmode.
+# command to the generated LaTeX files. This will instruct LaTeX to keep
+# running if errors occur, instead of asking the user for help.
+# This option is also used when generating formulas in HTML.
+
+LATEX_BATCHMODE = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the RTF output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_RTF tag is set to YES Doxygen will generate RTF output
+# The RTF output is optimised for Word 97 and may not look very pretty with
+# other RTF readers or editors.
+
+GENERATE_RTF = NO
+
+# The RTF_OUTPUT tag is used to specify where the RTF docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `rtf' will be used as the default path.
+
+RTF_OUTPUT = ../mgmapi.rtf
+
+# If the COMPACT_RTF tag is set to YES Doxygen generates more compact
+# RTF documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_RTF = NO
+
+# If the RTF_HYPERLINKS tag is set to YES, the RTF that is generated
+# will contain hyperlink fields. The RTF file will
+# contain links (just like the HTML output) instead of page references.
+# This makes the output suitable for online browsing using WORD or other
+# programs which support those fields.
+# Note: wordpad (write) and others do not support links.
+
+RTF_HYPERLINKS = NO
+
+# Load stylesheet definitions from file. Syntax is similar to doxygen's
+# config file, i.e. a series of assigments. You only have to provide
+# replacements, missing definitions are set to their default value.
+
+RTF_STYLESHEET_FILE =
+
+# Set optional variables used in the generation of an rtf document.
+# Syntax is similar to doxygen's config file.
+
+RTF_EXTENSIONS_FILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the man page output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_MAN tag is set to YES (the default) Doxygen will
+# generate man pages
+
+GENERATE_MAN = NO
+
+# The MAN_OUTPUT tag is used to specify where the man pages will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `man' will be used as the default path.
+
+MAN_OUTPUT =
+
+# The MAN_EXTENSION tag determines the extension that is added to
+# the generated man pages (default is the subroutine's section .3)
+
+MAN_EXTENSION =
+
+# If the MAN_LINKS tag is set to YES and Doxygen generates man output,
+# then it will generate one additional man file for each entity
+# documented in the real man page(s). These additional files
+# only source the real man page, but without them the man command
+# would be unable to find the correct page. The default is NO.
+
+MAN_LINKS = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the XML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_XML tag is set to YES Doxygen will
+# generate an XML file that captures the structure of
+# the code including all documentation. Note that this
+# feature is still experimental and incomplete at the
+# moment.
+
+GENERATE_XML = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the preprocessor
+#---------------------------------------------------------------------------
+
+# If the ENABLE_PREPROCESSING tag is set to YES (the default) Doxygen will
+# evaluate all C-preprocessor directives found in the sources and include
+# files.
+
+ENABLE_PREPROCESSING = YES
+
+# If the MACRO_EXPANSION tag is set to YES Doxygen will expand all macro
+# names in the source code. If set to NO (the default) only conditional
+# compilation will be performed. Macro expansion can be done in a controlled
+# way by setting EXPAND_ONLY_PREDEF to YES.
+
+MACRO_EXPANSION = YES
+
+# If the EXPAND_ONLY_PREDEF and MACRO_EXPANSION tags are both set to YES
+# then the macro expansion is limited to the macros specified with the
+# PREDEFINED and EXPAND_AS_PREDEFINED tags.
+
+EXPAND_ONLY_PREDEF = YES
+
+# If the SEARCH_INCLUDES tag is set to YES (the default) the includes files
+# in the INCLUDE_PATH (see below) will be search if a #include is found.
+
+SEARCH_INCLUDES = YES
+
+# The INCLUDE_PATH tag can be used to specify one or more directories that
+# contain include files that are not input files but should be processed by
+# the preprocessor.
+
+INCLUDE_PATH =
+
+# You can use the INCLUDE_FILE_PATTERNS tag to specify one or more wildcard
+# patterns (like *.h and *.hpp) to filter out the header-files in the
+# directories. If left blank, the patterns specified with FILE_PATTERNS will
+# be used.
+
+INCLUDE_FILE_PATTERNS =
+
+# The PREDEFINED tag can be used to specify one or more macro names that
+# are defined before the preprocessor is started (similar to the -D option of
+# gcc). The argument of the tag is a list of macros of the form: name
+# or name=definition (no spaces). If the definition and the = are
+# omitted =1 is assumed.
+
+PREDEFINED = DOXYGEN_SHOULD_SKIP_DEPRECATED \
+ DOXYGEN_SHOULD_SKIP_INTERNAL \
+ protected=private
+
+# If the MACRO_EXPANSION and EXPAND_PREDEF_ONLY tags are set to YES then
+# this tag can be used to specify a list of macro names that should be expanded.
+# The macro definition that is found in the sources will be used.
+# Use the PREDEFINED tag if you want to use a different macro definition.
+
+EXPAND_AS_DEFINED =
+
+# If the SKIP_FUNCTION_MACROS tag is set to YES (the default) then
+# doxygen's preprocessor will remove all function-like macros that are alone
+# on a line and do not end with a semicolon. Such function macros are typically
+# used for boiler-plate code, and will confuse the parser if not removed.
+
+SKIP_FUNCTION_MACROS = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to external references
+#---------------------------------------------------------------------------
+
+# The TAGFILES tag can be used to specify one or more tagfiles.
+
+TAGFILES =
+
+# When a file name is specified after GENERATE_TAGFILE, doxygen will create
+# a tag file that is based on the input files it reads.
+
+GENERATE_TAGFILE =
+
+# If the ALLEXTERNALS tag is set to YES all external classes will be listed
+# in the class index. If set to NO only the inherited external classes
+# will be listed.
+
+ALLEXTERNALS = NO
+
+# The PERL_PATH should be the absolute path and name of the perl script
+# interpreter (i.e. the result of `which perl').
+
+PERL_PATH =
+
+#---------------------------------------------------------------------------
+# Configuration options related to the dot tool
+#---------------------------------------------------------------------------
+
+# If the CLASS_DIAGRAMS tag is set to YES (the default) Doxygen will
+# generate a inheritance diagram (in Html, RTF and LaTeX) for classes with base or
+# super classes. Setting the tag to NO turns the diagrams off. Note that this
+# option is superceded by the HAVE_DOT option below. This is only a fallback. It is
+# recommended to install and use dot, since it yield more powerful graphs.
+
+CLASS_DIAGRAMS = YES
+
+# If you set the HAVE_DOT tag to YES then doxygen will assume the dot tool is
+# available from the path. This tool is part of Graphviz, a graph visualization
+# toolkit from AT&T and Lucent Bell Labs. The other options in this section
+# have no effect if this option is set to NO (the default)
+
+HAVE_DOT = NO
+
+# If the CLASS_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect inheritance relations. Setting this tag to YES will force the
+# the CLASS_DIAGRAMS tag to NO.
+
+CLASS_GRAPH = YES
+
+# If the COLLABORATION_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect implementation dependencies (inheritance, containment, and
+# class references variables) of the class with other documented classes.
+
+COLLABORATION_GRAPH = YES
+
+# If set to YES, the inheritance and collaboration graphs will show the
+# relations between templates and their instances.
+
+TEMPLATE_RELATIONS = YES
+
+# If set to YES, the inheritance and collaboration graphs will hide
+# inheritance and usage relations if the target is undocumented
+# or is not a class.
+
+HIDE_UNDOC_RELATIONS = YES
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDE_GRAPH, and HAVE_DOT
+# tags are set to YES then doxygen will generate a graph for each documented
+# file showing the direct and indirect include dependencies of the file with
+# other documented files.
+
+INCLUDE_GRAPH = YES
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDED_BY_GRAPH, and
+# HAVE_DOT tags are set to YES then doxygen will generate a graph for each
+# documented header file showing the documented files that directly or
+# indirectly include this file.
+
+INCLUDED_BY_GRAPH = YES
+
+# If the GRAPHICAL_HIERARCHY and HAVE_DOT tags are set to YES then doxygen
+# will graphical hierarchy of all classes instead of a textual one.
+
+GRAPHICAL_HIERARCHY = YES
+
+# The tag DOT_PATH can be used to specify the path where the dot tool can be
+# found. If left blank, it is assumed the dot tool can be found on the path.
+
+DOT_PATH =
+
+# The DOTFILE_DIRS tag can be used to specify one or more directories that
+# contain dot files that are included in the documentation (see the
+# \dotfile command).
+
+DOTFILE_DIRS =
+
+# The MAX_DOT_GRAPH_WIDTH tag can be used to set the maximum allowed width
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_WIDTH = 1024
+
+# The MAX_DOT_GRAPH_HEIGHT tag can be used to set the maximum allows height
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_HEIGHT = 1024
+
+# If the GENERATE_LEGEND tag is set to YES (the default) Doxygen will
+# generate a legend page explaining the meaning of the various boxes and
+# arrows in the dot generated graphs.
+
+GENERATE_LEGEND = YES
+
+# If the DOT_CLEANUP tag is set to YES (the default) Doxygen will
+# remove the intermedate dot files that are used to generate
+# the various graphs.
+
+DOT_CLEANUP = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to the search engine
+#---------------------------------------------------------------------------
+
+# The SEARCHENGINE tag specifies whether or not a search engine should be
+# used. If set to NO the values of all tags below this one will be ignored.
+
+SEARCHENGINE = NO
+
+# The CGI_NAME tag should be the name of the CGI script that
+# starts the search engine (doxysearch) with the correct parameters.
+# A script with this name will be generated by doxygen.
+
+CGI_NAME =
+
+# The CGI_URL tag should be the absolute URL to the directory where the
+# cgi binaries are located. See the documentation of your http daemon for
+# details.
+
+CGI_URL =
+
+# The DOC_URL tag should be the absolute URL to the directory where the
+# documentation is located. If left blank the absolute path to the
+# documentation, with file:// prepended to it, will be used.
+
+DOC_URL =
+
+# The DOC_ABSPATH tag should be the absolute path to the directory where the
+# documentation is located. If left blank the directory on the local machine
+# will be used.
+
+DOC_ABSPATH =
+
+# The BIN_ABSPATH tag must point to the directory where the doxysearch binary
+# is installed.
+
+BIN_ABSPATH =
+
+# The EXT_DOC_PATHS tag can be used to specify one or more paths to
+# documentation generated for other projects. This allows doxysearch to search
+# the documentation for these projects as well.
+
+EXT_DOC_PATHS =
diff --git a/ndb/docs/doxygen/Doxyfile.ndb b/ndb/docs/doxygen/Doxyfile.ndb
new file mode 100644
index 00000000000..d43a66323f8
--- /dev/null
+++ b/ndb/docs/doxygen/Doxyfile.ndb
@@ -0,0 +1,937 @@
+# Doxyfile 1.2.14
+
+# This file describes the settings to be used by the documentation system
+# doxygen (www.doxygen.org) for a project
+#
+# All text after a hash (#) is considered a comment and will be ignored
+# The format is:
+# TAG = value [value, ...]
+# For lists items can also be appended using:
+# TAG += value [value, ...]
+# Values that contain spaces should be placed between quotes (" ")
+
+#---------------------------------------------------------------------------
+# General configuration options
+#---------------------------------------------------------------------------
+
+# The PROJECT_NAME tag is a single word (or a sequence of words surrounded
+# by quotes) that should identify the project.
+
+PROJECT_NAME = "NDB Cluster"
+
+# The PROJECT_NUMBER tag can be used to enter a project or revision number.
+# This could be handy for archiving the generated documentation or
+# if some version control system is used.
+
+PROJECT_NUMBER =
+
+# The OUTPUT_DIRECTORY tag is used to specify the (relative or absolute)
+# base path where the generated documentation will be put.
+# If a relative path is entered, it will be relative to the location
+# where doxygen was started. If left blank the current directory will be used.
+
+OUTPUT_DIRECTORY =
+
+# The OUTPUT_LANGUAGE tag is used to specify the language in which all
+# documentation generated by doxygen is written. Doxygen will use this
+# information to generate all constant output in the proper language.
+# The default language is English, other supported languages are:
+# Brazilian, Chinese, Croatian, Czech, Danish, Dutch, Finnish, French,
+# German, Greek, Hungarian, Italian, Japanese, Korean, Norwegian, Polish,
+# Portuguese, Romanian, Russian, Slovak, Slovene, Spanish and Swedish.
+
+OUTPUT_LANGUAGE = English
+
+# If the EXTRACT_ALL tag is set to YES doxygen will assume all entities in
+# documentation are documented, even if no documentation was available.
+# Private class members and static file members will be hidden unless
+# the EXTRACT_PRIVATE and EXTRACT_STATIC tags are set to YES
+
+EXTRACT_ALL = YES
+
+# If the EXTRACT_PRIVATE tag is set to YES all private members of a class
+# will be included in the documentation.
+
+EXTRACT_PRIVATE = YES
+
+# If the EXTRACT_STATIC tag is set to YES all static members of a file
+# will be included in the documentation.
+
+EXTRACT_STATIC = YES
+
+# If the EXTRACT_LOCAL_CLASSES tag is set to YES classes (and structs)
+# defined locally in source files will be included in the documentation.
+# If set to NO only classes defined in header files are included.
+
+EXTRACT_LOCAL_CLASSES = YES
+
+# If the HIDE_UNDOC_MEMBERS tag is set to YES, Doxygen will hide all
+# undocumented members of documented classes, files or namespaces.
+# If set to NO (the default) these members will be included in the
+# various overviews, but no documentation section is generated.
+# This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_MEMBERS = NO
+
+# If the HIDE_UNDOC_CLASSES tag is set to YES, Doxygen will hide all
+# undocumented classes that are normally visible in the class hierarchy.
+# If set to NO (the default) these class will be included in the various
+# overviews. This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_CLASSES = NO
+
+# If the BRIEF_MEMBER_DESC tag is set to YES (the default) Doxygen will
+# include brief member descriptions after the members that are listed in
+# the file and class documentation (similar to JavaDoc).
+# Set to NO to disable this.
+
+BRIEF_MEMBER_DESC = YES
+
+# If the REPEAT_BRIEF tag is set to YES (the default) Doxygen will prepend
+# the brief description of a member or function before the detailed description.
+# Note: if both HIDE_UNDOC_MEMBERS and BRIEF_MEMBER_DESC are set to NO, the
+# brief descriptions will be completely suppressed.
+
+REPEAT_BRIEF = YES
+
+# If the ALWAYS_DETAILED_SEC and REPEAT_BRIEF tags are both set to YES then
+# Doxygen will generate a detailed section even if there is only a brief
+# description.
+
+ALWAYS_DETAILED_SEC = NO
+
+# If the INLINE_INHERITED_MEMB tag is set to YES, doxygen will show all inherited
+# members of a class in the documentation of that class as if those members were
+# ordinary class members. Constructors, destructors and assignment operators of
+# the base classes will not be shown.
+
+INLINE_INHERITED_MEMB = NO
+
+# If the FULL_PATH_NAMES tag is set to YES then Doxygen will prepend the full
+# path before files name in the file list and in the header files. If set
+# to NO the shortest path that makes the file name unique will be used.
+
+FULL_PATH_NAMES = YES
+
+# If the FULL_PATH_NAMES tag is set to YES then the STRIP_FROM_PATH tag
+# can be used to strip a user defined part of the path. Stripping is
+# only done if one of the specified strings matches the left-hand part of
+# the path. It is allowed to use relative paths in the argument list.
+
+STRIP_FROM_PATH = .
+
+# The INTERNAL_DOCS tag determines if documentation
+# that is typed after a \internal command is included. If the tag is set
+# to NO (the default) then the documentation will be excluded.
+# Set it to YES to include the internal documentation.
+
+INTERNAL_DOCS = YES
+
+# Setting the STRIP_CODE_COMMENTS tag to YES (the default) will instruct
+# doxygen to hide any special comment blocks from generated source code
+# fragments. Normal C and C++ comments will always remain visible.
+
+STRIP_CODE_COMMENTS = YES
+
+# If the CASE_SENSE_NAMES tag is set to NO then Doxygen will only generate
+# file names in lower case letters. If set to YES upper case letters are also
+# allowed. This is useful if you have classes or files whose names only differ
+# in case and if your file system supports case sensitive file names. Windows
+# users are adviced to set this option to NO.
+
+CASE_SENSE_NAMES = YES
+
+# If the SHORT_NAMES tag is set to YES, doxygen will generate much shorter
+# (but less readable) file names. This can be useful is your file systems
+# doesn't support long names like on DOS, Mac, or CD-ROM.
+
+SHORT_NAMES = NO
+
+# If the HIDE_SCOPE_NAMES tag is set to NO (the default) then Doxygen
+# will show members with their full class and namespace scopes in the
+# documentation. If set to YES the scope will be hidden.
+
+HIDE_SCOPE_NAMES = NO
+
+# If the VERBATIM_HEADERS tag is set to YES (the default) then Doxygen
+# will generate a verbatim copy of the header file for each class for
+# which an include is specified. Set to NO to disable this.
+
+VERBATIM_HEADERS = YES
+
+# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
+# will put list of the files that are included by a file in the documentation
+# of that file.
+
+SHOW_INCLUDE_FILES = YES
+
+# If the JAVADOC_AUTOBRIEF tag is set to YES then Doxygen
+# will interpret the first line (until the first dot) of a JavaDoc-style
+# comment as the brief description. If set to NO, the JavaDoc
+# comments will behave just like the Qt-style comments (thus requiring an
+# explict @brief command for a brief description.
+
+JAVADOC_AUTOBRIEF = NO
+
+# If the INHERIT_DOCS tag is set to YES (the default) then an undocumented
+# member inherits the documentation from any documented member that it
+# reimplements.
+
+INHERIT_DOCS = YES
+
+# If the INLINE_INFO tag is set to YES (the default) then a tag [inline]
+# is inserted in the documentation for inline members.
+
+INLINE_INFO = YES
+
+# If the SORT_MEMBER_DOCS tag is set to YES (the default) then doxygen
+# will sort the (detailed) documentation of file and class members
+# alphabetically by member name. If set to NO the members will appear in
+# declaration order.
+
+SORT_MEMBER_DOCS = YES
+
+# If member grouping is used in the documentation and the DISTRIBUTE_GROUP_DOC
+# tag is set to YES, then doxygen will reuse the documentation of the first
+# member in the group (if any) for the other members of the group. By default
+# all members of a group must be documented explicitly.
+
+DISTRIBUTE_GROUP_DOC = NO
+
+# The TAB_SIZE tag can be used to set the number of spaces in a tab.
+# Doxygen uses this value to replace tabs by spaces in code fragments.
+
+TAB_SIZE = 8
+
+# The GENERATE_TODOLIST tag can be used to enable (YES) or
+# disable (NO) the todo list. This list is created by putting \todo
+# commands in the documentation.
+
+GENERATE_TODOLIST = YES
+
+# The GENERATE_TESTLIST tag can be used to enable (YES) or
+# disable (NO) the test list. This list is created by putting \test
+# commands in the documentation.
+
+GENERATE_TESTLIST = YES
+
+# The GENERATE_BUGLIST tag can be used to enable (YES) or
+# disable (NO) the bug list. This list is created by putting \bug
+# commands in the documentation.
+
+GENERATE_BUGLIST = YES
+
+# This tag can be used to specify a number of aliases that acts
+# as commands in the documentation. An alias has the form "name=value".
+# For example adding "sideeffect=\par Side Effects:\n" will allow you to
+# put the command \sideeffect (or @sideeffect) in the documentation, which
+# will result in a user defined paragraph with heading "Side Effects:".
+# You can put \n's in the value part of an alias to insert newlines.
+
+ALIASES =
+
+# The ENABLED_SECTIONS tag can be used to enable conditional
+# documentation sections, marked by \if sectionname ... \endif.
+
+ENABLED_SECTIONS =
+
+# The MAX_INITIALIZER_LINES tag determines the maximum number of lines
+# the initial value of a variable or define consist of for it to appear in
+# the documentation. If the initializer consists of more lines than specified
+# here it will be hidden. Use a value of 0 to hide initializers completely.
+# The appearance of the initializer of individual variables and defines in the
+# documentation can be controlled using \showinitializer or \hideinitializer
+# command in the documentation regardless of this setting.
+
+MAX_INITIALIZER_LINES = 30
+
+# Set the OPTIMIZE_OUTPUT_FOR_C tag to YES if your project consists of C sources
+# only. Doxygen will then generate output that is more tailored for C.
+# For instance some of the names that are used will be different. The list
+# of all members will be omitted, etc.
+
+OPTIMIZE_OUTPUT_FOR_C = NO
+
+# Set the SHOW_USED_FILES tag to NO to disable the list of files generated
+# at the bottom of the documentation of classes and structs. If set to YES the
+# list will mention the files that were used to generate the documentation.
+
+SHOW_USED_FILES = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to warning and progress messages
+#---------------------------------------------------------------------------
+
+# The QUIET tag can be used to turn on/off the messages that are generated
+# by doxygen. Possible values are YES and NO. If left blank NO is used.
+
+QUIET = NO
+
+# The WARNINGS tag can be used to turn on/off the warning messages that are
+# generated by doxygen. Possible values are YES and NO. If left blank
+# NO is used.
+
+WARNINGS = YES
+
+# If WARN_IF_UNDOCUMENTED is set to YES, then doxygen will generate warnings
+# for undocumented members. If EXTRACT_ALL is set to YES then this flag will
+# automatically be disabled.
+
+WARN_IF_UNDOCUMENTED = YES
+
+# The WARN_FORMAT tag determines the format of the warning messages that
+# doxygen can produce. The string should contain the $file, $line, and $text
+# tags, which will be replaced by the file and line number from which the
+# warning originated and the warning text.
+
+WARN_FORMAT = "$file:$line: $text"
+
+# The WARN_LOGFILE tag can be used to specify a file to which warning
+# and error messages should be written. If left blank the output is written
+# to stderr.
+
+WARN_LOGFILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the input files
+#---------------------------------------------------------------------------
+
+# The INPUT tag can be used to specify the files and/or directories that contain
+# documented source files. You may enter file names like "myfile.cpp" or
+# directories like "/usr/src/myproject". Separate the files or directories
+# with spaces.
+
+INPUT =
+
+# If the value of the INPUT tag contains directories, you can use the
+# FILE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank the following patterns are tested:
+# *.c *.cc *.cxx *.cpp *.c++ *.java *.ii *.ixx *.ipp *.i++ *.inl *.h *.hh *.hxx *.hpp
+# *.h++ *.idl *.odl
+
+FILE_PATTERNS =
+
+# The RECURSIVE tag can be used to turn specify whether or not subdirectories
+# should be searched for input files as well. Possible values are YES and NO.
+# If left blank NO is used.
+
+RECURSIVE = YES
+
+# The EXCLUDE tag can be used to specify files and/or directories that should
+# excluded from the INPUT source files. This way you can easily exclude a
+# subdirectory from a directory tree whose root is specified with the INPUT tag.
+
+EXCLUDE = test \
+ tools \
+ win32 \
+ lib \
+ examples \
+ docs \
+ CVS \
+ SCCS \
+ config \
+ bin \
+ include/ndbapi \
+ include/newtonapi \
+ src/newtonapi \
+ include/mgmapi \
+ src/mgmapi \
+ src/client
+
+# The EXCLUDE_SYMLINKS tag can be used select whether or not files or directories
+# that are symbolic links (a Unix filesystem feature) are excluded from the input.
+
+EXCLUDE_SYMLINKS = NO
+
+# If the value of the INPUT tag contains directories, you can use the
+# EXCLUDE_PATTERNS tag to specify one or more wildcard patterns to exclude
+# certain files from those directories.
+
+EXCLUDE_PATTERNS = *CVS* \
+ *SCCS*
+
+# The EXAMPLE_PATH tag can be used to specify one or more files or
+# directories that contain example code fragments that are included (see
+# the \include command).
+
+EXAMPLE_PATH =
+
+# If the value of the EXAMPLE_PATH tag contains directories, you can use the
+# EXAMPLE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank all files are included.
+
+EXAMPLE_PATTERNS =
+
+# If the EXAMPLE_RECURSIVE tag is set to YES then subdirectories will be
+# searched for input files to be used with the \include or \dontinclude
+# commands irrespective of the value of the RECURSIVE tag.
+# Possible values are YES and NO. If left blank NO is used.
+
+EXAMPLE_RECURSIVE = NO
+
+# The IMAGE_PATH tag can be used to specify one or more files or
+# directories that contain image that are included in the documentation (see
+# the \image command).
+
+IMAGE_PATH =
+
+# The INPUT_FILTER tag can be used to specify a program that doxygen should
+# invoke to filter for each input file. Doxygen will invoke the filter program
+# by executing (via popen()) the command <filter> <input-file>, where <filter>
+# is the value of the INPUT_FILTER tag, and <input-file> is the name of an
+# input file. Doxygen will then use the output that the filter program writes
+# to standard output.
+
+INPUT_FILTER =
+
+# If the FILTER_SOURCE_FILES tag is set to YES, the input filter (if set using
+# INPUT_FILTER) will be used to filter the input files when producing source
+# files to browse.
+
+FILTER_SOURCE_FILES = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to source browsing
+#---------------------------------------------------------------------------
+
+# If the SOURCE_BROWSER tag is set to YES then a list of source files will
+# be generated. Documented entities will be cross-referenced with these sources.
+
+SOURCE_BROWSER = YES
+
+# Setting the INLINE_SOURCES tag to YES will include the body
+# of functions and classes directly in the documentation.
+
+INLINE_SOURCES = YES
+
+# If the REFERENCED_BY_RELATION tag is set to YES (the default)
+# then for each documented function all documented
+# functions referencing it will be listed.
+
+REFERENCED_BY_RELATION = YES
+
+# If the REFERENCES_RELATION tag is set to YES (the default)
+# then for each documented function all documented entities
+# called/used by that function will be listed.
+
+REFERENCES_RELATION = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to the alphabetical class index
+#---------------------------------------------------------------------------
+
+# If the ALPHABETICAL_INDEX tag is set to YES, an alphabetical index
+# of all compounds will be generated. Enable this if the project
+# contains a lot of classes, structs, unions or interfaces.
+
+ALPHABETICAL_INDEX = YES
+
+# If the alphabetical index is enabled (see ALPHABETICAL_INDEX) then
+# the COLS_IN_ALPHA_INDEX tag can be used to specify the number of columns
+# in which this list will be split (can be a number in the range [1..20])
+
+COLS_IN_ALPHA_INDEX = 5
+
+# In case all classes in a project start with a common prefix, all
+# classes will be put under the same header in the alphabetical index.
+# The IGNORE_PREFIX tag can be used to specify one or more prefixes that
+# should be ignored while generating the index headers.
+
+IGNORE_PREFIX =
+
+#---------------------------------------------------------------------------
+# configuration options related to the HTML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_HTML tag is set to YES (the default) Doxygen will
+# generate HTML output.
+
+GENERATE_HTML = YES
+
+# The HTML_OUTPUT tag is used to specify where the HTML docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `html' will be used as the default path.
+
+HTML_OUTPUT = ndb.html
+
+# The HTML_FILE_EXTENSION tag can be used to specify the file extension for
+# each generated HTML page (for example: .htm,.php,.asp). If it is left blank
+# doxygen will generate files with .html extension.
+
+HTML_FILE_EXTENSION = .html
+
+# The HTML_HEADER tag can be used to specify a personal HTML header for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard header.
+
+HTML_HEADER =
+
+# The HTML_FOOTER tag can be used to specify a personal HTML footer for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard footer.
+
+HTML_FOOTER =
+
+# The HTML_STYLESHEET tag can be used to specify a user defined cascading
+# style sheet that is used by each HTML page. It can be used to
+# fine-tune the look of the HTML output. If the tag is left blank doxygen
+# will generate a default style sheet
+
+HTML_STYLESHEET =
+
+# If the HTML_ALIGN_MEMBERS tag is set to YES, the members of classes,
+# files or namespaces will be aligned in HTML using tables. If set to
+# NO a bullet list will be used.
+
+HTML_ALIGN_MEMBERS = YES
+
+# If the GENERATE_HTMLHELP tag is set to YES, additional index files
+# will be generated that can be used as input for tools like the
+# Microsoft HTML help workshop to generate a compressed HTML help file (.chm)
+# of the generated HTML documentation.
+
+GENERATE_HTMLHELP = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the GENERATE_CHI flag
+# controls if a separate .chi index file is generated (YES) or that
+# it should be included in the master .chm file (NO).
+
+GENERATE_CHI = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the BINARY_TOC flag
+# controls whether a binary table of contents is generated (YES) or a
+# normal table of contents (NO) in the .chm file.
+
+BINARY_TOC = NO
+
+# The TOC_EXPAND flag can be set to YES to add extra items for group members
+# to the contents of the Html help documentation and to the tree view.
+
+TOC_EXPAND = NO
+
+# The DISABLE_INDEX tag can be used to turn on/off the condensed index at
+# top of each HTML page. The value NO (the default) enables the index and
+# the value YES disables it.
+
+DISABLE_INDEX = NO
+
+# This tag can be used to set the number of enum values (range [1..20])
+# that doxygen will group on one line in the generated HTML documentation.
+
+ENUM_VALUES_PER_LINE = 4
+
+# If the GENERATE_TREEVIEW tag is set to YES, a side panel will be
+# generated containing a tree-like index structure (just like the one that
+# is generated for HTML Help). For this to work a browser that supports
+# JavaScript and frames is required (for instance Mozilla, Netscape 4.0+,
+# or Internet explorer 4.0+). Note that for large projects the tree generation
+# can take a very long time. In such cases it is better to disable this feature.
+# Windows users are probably better off using the HTML help feature.
+
+GENERATE_TREEVIEW = NO
+
+# If the treeview is enabled (see GENERATE_TREEVIEW) then this tag can be
+# used to set the initial width (in pixels) of the frame in which the tree
+# is shown.
+
+TREEVIEW_WIDTH = 250
+
+#---------------------------------------------------------------------------
+# configuration options related to the LaTeX output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_LATEX tag is set to YES (the default) Doxygen will
+# generate Latex output.
+
+GENERATE_LATEX = NO
+
+# The LATEX_OUTPUT tag is used to specify where the LaTeX docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `latex' will be used as the default path.
+
+LATEX_OUTPUT = ndb.latex
+
+# If the COMPACT_LATEX tag is set to YES Doxygen generates more compact
+# LaTeX documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_LATEX = NO
+
+# The PAPER_TYPE tag can be used to set the paper type that is used
+# by the printer. Possible values are: a4, a4wide, letter, legal and
+# executive. If left blank a4wide will be used.
+
+PAPER_TYPE = a4wide
+
+# The EXTRA_PACKAGES tag can be to specify one or more names of LaTeX
+# packages that should be included in the LaTeX output.
+
+EXTRA_PACKAGES =
+
+# The LATEX_HEADER tag can be used to specify a personal LaTeX header for
+# the generated latex document. The header should contain everything until
+# the first chapter. If it is left blank doxygen will generate a
+# standard header. Notice: only use this tag if you know what you are doing!
+
+LATEX_HEADER =
+
+# If the PDF_HYPERLINKS tag is set to YES, the LaTeX that is generated
+# is prepared for conversion to pdf (using ps2pdf). The pdf file will
+# contain links (just like the HTML output) instead of page references
+# This makes the output suitable for online browsing using a pdf viewer.
+
+PDF_HYPERLINKS = NO
+
+# If the USE_PDFLATEX tag is set to YES, pdflatex will be used instead of
+# plain latex in the generated Makefile. Set this option to YES to get a
+# higher quality PDF documentation.
+
+USE_PDFLATEX = NO
+
+# If the LATEX_BATCHMODE tag is set to YES, doxygen will add the \\batchmode.
+# command to the generated LaTeX files. This will instruct LaTeX to keep
+# running if errors occur, instead of asking the user for help.
+# This option is also used when generating formulas in HTML.
+
+LATEX_BATCHMODE = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the RTF output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_RTF tag is set to YES Doxygen will generate RTF output
+# The RTF output is optimised for Word 97 and may not look very pretty with
+# other RTF readers or editors.
+
+GENERATE_RTF = NO
+
+# The RTF_OUTPUT tag is used to specify where the RTF docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `rtf' will be used as the default path.
+
+RTF_OUTPUT = rtf
+
+# If the COMPACT_RTF tag is set to YES Doxygen generates more compact
+# RTF documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_RTF = NO
+
+# If the RTF_HYPERLINKS tag is set to YES, the RTF that is generated
+# will contain hyperlink fields. The RTF file will
+# contain links (just like the HTML output) instead of page references.
+# This makes the output suitable for online browsing using WORD or other
+# programs which support those fields.
+# Note: wordpad (write) and others do not support links.
+
+RTF_HYPERLINKS = NO
+
+# Load stylesheet definitions from file. Syntax is similar to doxygen's
+# config file, i.e. a series of assigments. You only have to provide
+# replacements, missing definitions are set to their default value.
+
+RTF_STYLESHEET_FILE =
+
+# Set optional variables used in the generation of an rtf document.
+# Syntax is similar to doxygen's config file.
+
+RTF_EXTENSIONS_FILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the man page output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_MAN tag is set to YES (the default) Doxygen will
+# generate man pages
+
+GENERATE_MAN = NO
+
+# The MAN_OUTPUT tag is used to specify where the man pages will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `man' will be used as the default path.
+
+MAN_OUTPUT = man
+
+# The MAN_EXTENSION tag determines the extension that is added to
+# the generated man pages (default is the subroutine's section .3)
+
+MAN_EXTENSION = .3
+
+# If the MAN_LINKS tag is set to YES and Doxygen generates man output,
+# then it will generate one additional man file for each entity
+# documented in the real man page(s). These additional files
+# only source the real man page, but without them the man command
+# would be unable to find the correct page. The default is NO.
+
+MAN_LINKS = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the XML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_XML tag is set to YES Doxygen will
+# generate an XML file that captures the structure of
+# the code including all documentation. Note that this
+# feature is still experimental and incomplete at the
+# moment.
+
+GENERATE_XML = NO
+
+#---------------------------------------------------------------------------
+# configuration options for the AutoGen Definitions output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_AUTOGEN_DEF tag is set to YES Doxygen will
+# generate an AutoGen Definitions (see autogen.sf.net) file
+# that captures the structure of the code including all
+# documentation. Note that this feature is still experimental
+# and incomplete at the moment.
+
+GENERATE_AUTOGEN_DEF = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the preprocessor
+#---------------------------------------------------------------------------
+
+# If the ENABLE_PREPROCESSING tag is set to YES (the default) Doxygen will
+# evaluate all C-preprocessor directives found in the sources and include
+# files.
+
+ENABLE_PREPROCESSING = YES
+
+# If the MACRO_EXPANSION tag is set to YES Doxygen will expand all macro
+# names in the source code. If set to NO (the default) only conditional
+# compilation will be performed. Macro expansion can be done in a controlled
+# way by setting EXPAND_ONLY_PREDEF to YES.
+
+MACRO_EXPANSION = NO
+
+# If the EXPAND_ONLY_PREDEF and MACRO_EXPANSION tags are both set to YES
+# then the macro expansion is limited to the macros specified with the
+# PREDEFINED and EXPAND_AS_PREDEFINED tags.
+
+EXPAND_ONLY_PREDEF = NO
+
+# If the SEARCH_INCLUDES tag is set to YES (the default) the includes files
+# in the INCLUDE_PATH (see below) will be search if a #include is found.
+
+SEARCH_INCLUDES = YES
+
+# The INCLUDE_PATH tag can be used to specify one or more directories that
+# contain include files that are not input files but should be processed by
+# the preprocessor.
+
+INCLUDE_PATH =
+
+# You can use the INCLUDE_FILE_PATTERNS tag to specify one or more wildcard
+# patterns (like *.h and *.hpp) to filter out the header-files in the
+# directories. If left blank, the patterns specified with FILE_PATTERNS will
+# be used.
+
+INCLUDE_FILE_PATTERNS =
+
+# The PREDEFINED tag can be used to specify one or more macro names that
+# are defined before the preprocessor is started (similar to the -D option of
+# gcc). The argument of the tag is a list of macros of the form: name
+# or name=definition (no spaces). If the definition and the = are
+# omitted =1 is assumed.
+
+PREDEFINED =
+
+# If the MACRO_EXPANSION and EXPAND_PREDEF_ONLY tags are set to YES then
+# this tag can be used to specify a list of macro names that should be expanded.
+# The macro definition that is found in the sources will be used.
+# Use the PREDEFINED tag if you want to use a different macro definition.
+
+EXPAND_AS_DEFINED =
+
+# If the SKIP_FUNCTION_MACROS tag is set to YES (the default) then
+# doxygen's preprocessor will remove all function-like macros that are alone
+# on a line and do not end with a semicolon. Such function macros are typically
+# used for boiler-plate code, and will confuse the parser if not removed.
+
+SKIP_FUNCTION_MACROS = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to external references
+#---------------------------------------------------------------------------
+
+# The TAGFILES tag can be used to specify one or more tagfiles.
+
+TAGFILES =
+
+# When a file name is specified after GENERATE_TAGFILE, doxygen will create
+# a tag file that is based on the input files it reads.
+
+GENERATE_TAGFILE =
+
+# If the ALLEXTERNALS tag is set to YES all external classes will be listed
+# in the class index. If set to NO only the inherited external classes
+# will be listed.
+
+ALLEXTERNALS = NO
+
+# If the EXTERNAL_GROUPS tag is set to YES all external groups will be listed
+# in the modules index. If set to NO, only the current project's groups will
+# be listed.
+
+EXTERNAL_GROUPS = YES
+
+# The PERL_PATH should be the absolute path and name of the perl script
+# interpreter (i.e. the result of `which perl').
+
+PERL_PATH = /usr/bin/perl
+
+#---------------------------------------------------------------------------
+# Configuration options related to the dot tool
+#---------------------------------------------------------------------------
+
+# If the CLASS_DIAGRAMS tag is set to YES (the default) Doxygen will
+# generate a inheritance diagram (in Html, RTF and LaTeX) for classes with base or
+# super classes. Setting the tag to NO turns the diagrams off. Note that this
+# option is superceded by the HAVE_DOT option below. This is only a fallback. It is
+# recommended to install and use dot, since it yield more powerful graphs.
+
+CLASS_DIAGRAMS = YES
+
+# If you set the HAVE_DOT tag to YES then doxygen will assume the dot tool is
+# available from the path. This tool is part of Graphviz, a graph visualization
+# toolkit from AT&T and Lucent Bell Labs. The other options in this section
+# have no effect if this option is set to NO (the default)
+
+HAVE_DOT = YES
+
+# If the CLASS_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect inheritance relations. Setting this tag to YES will force the
+# the CLASS_DIAGRAMS tag to NO.
+
+CLASS_GRAPH = YES
+
+# If the COLLABORATION_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect implementation dependencies (inheritance, containment, and
+# class references variables) of the class with other documented classes.
+
+COLLABORATION_GRAPH = YES
+
+# If set to YES, the inheritance and collaboration graphs will show the
+# relations between templates and their instances.
+
+TEMPLATE_RELATIONS = YES
+
+# If set to YES, the inheritance and collaboration graphs will hide
+# inheritance and usage relations if the target is undocumented
+# or is not a class.
+
+HIDE_UNDOC_RELATIONS = NO
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDE_GRAPH, and HAVE_DOT
+# tags are set to YES then doxygen will generate a graph for each documented
+# file showing the direct and indirect include dependencies of the file with
+# other documented files.
+
+INCLUDE_GRAPH = YES
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDED_BY_GRAPH, and
+# HAVE_DOT tags are set to YES then doxygen will generate a graph for each
+# documented header file showing the documented files that directly or
+# indirectly include this file.
+
+INCLUDED_BY_GRAPH = YES
+
+# If the GRAPHICAL_HIERARCHY and HAVE_DOT tags are set to YES then doxygen
+# will graphical hierarchy of all classes instead of a textual one.
+
+GRAPHICAL_HIERARCHY = YES
+
+# The DOT_IMAGE_FORMAT tag can be used to set the image format of the images
+# generated by dot. Possible values are gif, jpg, and png
+# If left blank gif will be used.
+
+DOT_IMAGE_FORMAT = gif
+
+# The tag DOT_PATH can be used to specify the path where the dot tool can be
+# found. If left blank, it is assumed the dot tool can be found on the path.
+
+DOT_PATH =
+
+# The DOTFILE_DIRS tag can be used to specify one or more directories that
+# contain dot files that are included in the documentation (see the
+# \dotfile command).
+
+DOTFILE_DIRS =
+
+# The MAX_DOT_GRAPH_WIDTH tag can be used to set the maximum allowed width
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_WIDTH = 1024
+
+# The MAX_DOT_GRAPH_HEIGHT tag can be used to set the maximum allows height
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_HEIGHT = 1024
+
+# If the GENERATE_LEGEND tag is set to YES (the default) Doxygen will
+# generate a legend page explaining the meaning of the various boxes and
+# arrows in the dot generated graphs.
+
+GENERATE_LEGEND = YES
+
+# If the DOT_CLEANUP tag is set to YES (the default) Doxygen will
+# remove the intermedate dot files that are used to generate
+# the various graphs.
+
+DOT_CLEANUP = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to the search engine
+#---------------------------------------------------------------------------
+
+# The SEARCHENGINE tag specifies whether or not a search engine should be
+# used. If set to NO the values of all tags below this one will be ignored.
+
+SEARCHENGINE = NO
+
+# The CGI_NAME tag should be the name of the CGI script that
+# starts the search engine (doxysearch) with the correct parameters.
+# A script with this name will be generated by doxygen.
+
+CGI_NAME = search.cgi
+
+# The CGI_URL tag should be the absolute URL to the directory where the
+# cgi binaries are located. See the documentation of your http daemon for
+# details.
+
+CGI_URL =
+
+# The DOC_URL tag should be the absolute URL to the directory where the
+# documentation is located. If left blank the absolute path to the
+# documentation, with file:// prepended to it, will be used.
+
+DOC_URL =
+
+# The DOC_ABSPATH tag should be the absolute path to the directory where the
+# documentation is located. If left blank the directory on the local machine
+# will be used.
+
+DOC_ABSPATH =
+
+# The BIN_ABSPATH tag must point to the directory where the doxysearch binary
+# is installed.
+
+BIN_ABSPATH = /usr/local/bin
+
+# The EXT_DOC_PATHS tag can be used to specify one or more paths to
+# documentation generated for other projects. This allows doxysearch to search
+# the documentation for these projects as well.
+
+EXT_DOC_PATHS =
diff --git a/ndb/docs/doxygen/Doxyfile.ndbapi b/ndb/docs/doxygen/Doxyfile.ndbapi
new file mode 100644
index 00000000000..61d58d4fea3
--- /dev/null
+++ b/ndb/docs/doxygen/Doxyfile.ndbapi
@@ -0,0 +1,877 @@
+# Doxyfile 1.2.12
+
+# This file describes the settings to be used by the documentation system
+# doxygen (www.doxygen.org) for a project
+#
+# All text after a hash (#) is considered a comment and will be ignored
+# The format is:
+# TAG = value [value, ...]
+# For lists items can also be appended using:
+# TAG += value [value, ...]
+# Values that contain spaces should be placed between quotes (" ")
+
+#---------------------------------------------------------------------------
+# General configuration options
+#---------------------------------------------------------------------------
+DETAILS_AT_TOP = yes
+HIDE_FRIEND_COMPOUNDS = yes
+
+# The PROJECT_NAME tag is a single word (or a sequence of words surrounded
+# by quotes) that should identify the project.
+
+PROJECT_NAME =
+
+# The PROJECT_NUMBER tag can be used to enter a project or revision number.
+# This could be handy for archiving the generated documentation or
+# if some version control system is used.
+
+PROJECT_NUMBER =
+
+# The OUTPUT_DIRECTORY tag is used to specify the (relative or absolute)
+# base path where the generated documentation will be put.
+# If a relative path is entered, it will be relative to the location
+# where doxygen was started. If left blank the current directory will be used.
+
+OUTPUT_DIRECTORY =
+
+# The OUTPUT_LANGUAGE tag is used to specify the language in which all
+# documentation generated by doxygen is written. Doxygen will use this
+# information to generate all constant output in the proper language.
+# The default language is English, other supported languages are:
+# Brazilian, Chinese, Croatian, Czech, Danish, Dutch, Finnish, French,
+# German, Hungarian, Italian, Japanese, Korean, Norwegian, Polish,
+# Portuguese, Romanian, Russian, Slovak, Slovene, Spanish and Swedish.
+
+OUTPUT_LANGUAGE = English
+
+# If the EXTRACT_ALL tag is set to YES doxygen will assume all entities in
+# documentation are documented, even if no documentation was available.
+# Private class members and static file members will be hidden unless
+# the EXTRACT_PRIVATE and EXTRACT_STATIC tags are set to YES
+
+EXTRACT_ALL = NO
+
+# If the EXTRACT_PRIVATE tag is set to YES all private members of a class
+# will be included in the documentation.
+
+EXTRACT_PRIVATE = NO
+
+# If the EXTRACT_STATIC tag is set to YES all static members of a file
+# will be included in the documentation.
+
+EXTRACT_STATIC = NO
+
+# If the HIDE_UNDOC_MEMBERS tag is set to YES, Doxygen will hide all
+# undocumented members of documented classes, files or namespaces.
+# If set to NO (the default) these members will be included in the
+# various overviews, but no documentation section is generated.
+# This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_MEMBERS = NO
+
+# If the HIDE_UNDOC_CLASSES tag is set to YES, Doxygen will hide all
+# undocumented classes that are normally visible in the class hierarchy.
+# If set to NO (the default) these class will be included in the various
+# overviews. This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_CLASSES = NO
+
+# If the BRIEF_MEMBER_DESC tag is set to YES (the default) Doxygen will
+# include brief member descriptions after the members that are listed in
+# the file and class documentation (similar to JavaDoc).
+# Set to NO to disable this.
+
+BRIEF_MEMBER_DESC = YES
+
+# If the REPEAT_BRIEF tag is set to YES (the default) Doxygen will prepend
+# the brief description of a member or function before the detailed description.
+# Note: if both HIDE_UNDOC_MEMBERS and BRIEF_MEMBER_DESC are set to NO, the
+# brief descriptions will be completely suppressed.
+
+REPEAT_BRIEF = YES
+
+# If the ALWAYS_DETAILED_SEC and REPEAT_BRIEF tags are both set to YES then
+# Doxygen will generate a detailed section even if there is only a brief
+# description.
+
+ALWAYS_DETAILED_SEC = NO
+
+# If the FULL_PATH_NAMES tag is set to YES then Doxygen will prepend the full
+# path before files name in the file list and in the header files. If set
+# to NO the shortest path that makes the file name unique will be used.
+
+FULL_PATH_NAMES = NO
+
+# If the FULL_PATH_NAMES tag is set to YES then the STRIP_FROM_PATH tag
+# can be used to strip a user defined part of the path. Stripping is
+# only done if one of the specified strings matches the left-hand part of
+# the path. It is allowed to use relative paths in the argument list.
+
+STRIP_FROM_PATH =
+
+# The INTERNAL_DOCS tag determines if documentation
+# that is typed after a \internal command is included. If the tag is set
+# to NO (the default) then the documentation will be excluded.
+# Set it to YES to include the internal documentation.
+
+INTERNAL_DOCS = NO
+
+# Setting the STRIP_CODE_COMMENTS tag to YES (the default) will instruct
+# doxygen to hide any special comment blocks from generated source code
+# fragments. Normal C and C++ comments will always remain visible.
+
+STRIP_CODE_COMMENTS = YES
+
+# If the CASE_SENSE_NAMES tag is set to NO then Doxygen will only generate
+# file names in lower case letters. If set to YES upper case letters are also
+# allowed. This is useful if you have classes or files whose names only differ
+# in case and if your file system supports case sensitive file names. Windows
+# users are adviced to set this option to NO.
+
+CASE_SENSE_NAMES = YES
+
+# If the SHORT_NAMES tag is set to YES, doxygen will generate much shorter
+# (but less readable) file names. This can be useful is your file systems
+# doesn't support long names like on DOS, Mac, or CD-ROM.
+
+SHORT_NAMES = NO
+
+# If the HIDE_SCOPE_NAMES tag is set to NO (the default) then Doxygen
+# will show members with their full class and namespace scopes in the
+# documentation. If set to YES the scope will be hidden.
+
+HIDE_SCOPE_NAMES = NO
+
+# If the VERBATIM_HEADERS tag is set to YES (the default) then Doxygen
+# will generate a verbatim copy of the header file for each class for
+# which an include is specified. Set to NO to disable this.
+
+VERBATIM_HEADERS = YES
+
+# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
+# will put list of the files that are included by a file in the documentation
+# of that file.
+
+SHOW_INCLUDE_FILES = YES
+
+# If the JAVADOC_AUTOBRIEF tag is set to YES then Doxygen
+# will interpret the first line (until the first dot) of a JavaDoc-style
+# comment as the brief description. If set to NO, the JavaDoc
+# comments will behave just like the Qt-style comments (thus requiring an
+# explict @brief command for a brief description.
+
+JAVADOC_AUTOBRIEF = NO
+
+# If the INHERIT_DOCS tag is set to YES (the default) then an undocumented
+# member inherits the documentation from any documented member that it
+# reimplements.
+
+INHERIT_DOCS = YES
+
+# If the INLINE_INFO tag is set to YES (the default) then a tag [inline]
+# is inserted in the documentation for inline members.
+
+INLINE_INFO = YES
+
+# If the SORT_MEMBER_DOCS tag is set to YES (the default) then doxygen
+# will sort the (detailed) documentation of file and class members
+# alphabetically by member name. If set to NO the members will appear in
+# declaration order.
+
+SORT_MEMBER_DOCS = YES
+
+# If member grouping is used in the documentation and the DISTRIBUTE_GROUP_DOC
+# tag is set to YES, then doxygen will reuse the documentation of the first
+# member in the group (if any) for the other members of the group. By default
+# all members of a group must be documented explicitly.
+
+DISTRIBUTE_GROUP_DOC = NO
+
+# The TAB_SIZE tag can be used to set the number of spaces in a tab.
+# Doxygen uses this value to replace tabs by spaces in code fragments.
+
+TAB_SIZE = 8
+
+# The GENERATE_TODOLIST tag can be used to enable (YES) or
+# disable (NO) the todo list. This list is created by putting \todo
+# commands in the documentation.
+
+GENERATE_TODOLIST = YES
+
+# The GENERATE_TESTLIST tag can be used to enable (YES) or
+# disable (NO) the test list. This list is created by putting \test
+# commands in the documentation.
+
+GENERATE_TESTLIST = YES
+
+# The GENERATE_BUGLIST tag can be used to enable (YES) or
+# disable (NO) the bug list. This list is created by putting \bug
+# commands in the documentation.
+
+GENERATE_BUGLIST = YES
+
+# This tag can be used to specify a number of aliases that acts
+# as commands in the documentation. An alias has the form "name=value".
+# For example adding "sideeffect=\par Side Effects:\n" will allow you to
+# put the command \sideeffect (or @sideeffect) in the documentation, which
+# will result in a user defined paragraph with heading "Side Effects:".
+# You can put \n's in the value part of an alias to insert newlines.
+
+ALIASES =
+
+# The ENABLED_SECTIONS tag can be used to enable conditional
+# documentation sections, marked by \if sectionname ... \endif.
+
+ENABLED_SECTIONS =
+
+# The MAX_INITIALIZER_LINES tag determines the maximum number of lines
+# the initial value of a variable or define consist of for it to appear in
+# the documentation. If the initializer consists of more lines than specified
+# here it will be hidden. Use a value of 0 to hide initializers completely.
+# The appearance of the initializer of individual variables and defines in the
+# documentation can be controlled using \showinitializer or \hideinitializer
+# command in the documentation regardless of this setting.
+
+MAX_INITIALIZER_LINES = 30
+
+# Set the OPTIMIZE_OUTPUT_FOR_C tag to YES if your project consists of C sources
+# only. Doxygen will then generate output that is more tailored for C.
+# For instance some of the names that are used will be different. The list
+# of all members will be omitted, etc.
+
+OPTIMIZE_OUTPUT_FOR_C = NO
+
+# Set the SHOW_USED_FILES tag to NO to disable the list of files generated
+# at the bottom of the documentation of classes and structs. If set to YES the
+# list will mention the files that were used to generate the documentation.
+
+SHOW_USED_FILES = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to warning and progress messages
+#---------------------------------------------------------------------------
+
+# The QUIET tag can be used to turn on/off the messages that are generated
+# by doxygen. Possible values are YES and NO. If left blank NO is used.
+
+QUIET = NO
+
+# The WARNINGS tag can be used to turn on/off the warning messages that are
+# generated by doxygen. Possible values are YES and NO. If left blank
+# NO is used.
+
+WARNINGS = YES
+
+# If WARN_IF_UNDOCUMENTED is set to YES, then doxygen will generate warnings
+# for undocumented members. If EXTRACT_ALL is set to YES then this flag will
+# automatically be disabled.
+
+WARN_IF_UNDOCUMENTED = YES
+
+# The WARN_FORMAT tag determines the format of the warning messages that
+# doxygen can produce. The string should contain the $file, $line, and $text
+# tags, which will be replaced by the file and line number from which the
+# warning originated and the warning text.
+
+WARN_FORMAT =
+
+# The WARN_LOGFILE tag can be used to specify a file to which warning
+# and error messages should be written. If left blank the output is written
+# to stderr.
+
+WARN_LOGFILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the input files
+#---------------------------------------------------------------------------
+
+# The INPUT tag can be used to specify the files and/or directories that contain
+# documented source files. You may enter file names like "myfile.cpp" or
+# directories like "/usr/src/myproject". Separate the files or directories
+# with spaces.
+
+INPUT = .
+
+# If the value of the INPUT tag contains directories, you can use the
+# FILE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank file matching one of the following patterns are included:
+# *.c *.cc *.cxx *.c++ *.java *.ii *.ixx *.ipp *.i++ *.inl *.h *.hh *.hxx *.hpp
+# *.h++ *.idl
+
+FILE_PATTERNS =
+
+# The RECURSIVE tag can be used to turn specify whether or not subdirectories
+# should be searched for input files as well. Possible values are YES and NO.
+# If left blank NO is used.
+
+RECURSIVE = NO
+
+# The EXCLUDE tag can be used to specify files and/or directories that should
+# excluded from the INPUT source files. This way you can easily exclude a
+# subdirectory from a directory tree whose root is specified with the INPUT tag.
+
+EXCLUDE =
+
+# If the value of the INPUT tag contains directories, you can use the
+# EXCLUDE_PATTERNS tag to specify one or more wildcard patterns to exclude
+# certain files from those directories.
+
+EXCLUDE_PATTERNS =
+
+# The EXAMPLE_PATH tag can be used to specify one or more files or
+# directories that contain example code fragments that are included (see
+# the \include command).
+
+EXAMPLE_PATH = .
+
+# If the value of the EXAMPLE_PATH tag contains directories, you can use the
+# EXAMPLE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank all files are included.
+
+EXAMPLE_PATTERNS =
+
+# If the EXAMPLE_RECURSIVE tag is set to YES then subdirectories will be
+# searched for input files to be used with the \include or \dontinclude
+# commands irrespective of the value of the RECURSIVE tag.
+# Possible values are YES and NO. If left blank NO is used.
+
+EXAMPLE_RECURSIVE = NO
+
+# The IMAGE_PATH tag can be used to specify one or more files or
+# directories that contain image that are included in the documentation (see
+# the \image command).
+
+IMAGE_PATH =
+
+# The INPUT_FILTER tag can be used to specify a program that doxygen should
+# invoke to filter for each input file. Doxygen will invoke the filter program
+# by executing (via popen()) the command <filter> <input-file>, where <filter>
+# is the value of the INPUT_FILTER tag, and <input-file> is the name of an
+# input file. Doxygen will then use the output that the filter program writes
+# to standard output.
+
+INPUT_FILTER =
+
+# If the FILTER_SOURCE_FILES tag is set to YES, the input filter (if set using
+# INPUT_FILTER) will be used to filter the input files when producing source
+# files to browse.
+
+FILTER_SOURCE_FILES = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to source browsing
+#---------------------------------------------------------------------------
+
+# If the SOURCE_BROWSER tag is set to YES then a list of source files will
+# be generated. Documented entities will be cross-referenced with these sources.
+
+SOURCE_BROWSER = NO
+
+# Setting the INLINE_SOURCES tag to YES will include the body
+# of functions and classes directly in the documentation.
+
+INLINE_SOURCES = NO
+
+# If the REFERENCED_BY_RELATION tag is set to YES (the default)
+# then for each documented function all documented
+# functions referencing it will be listed.
+
+REFERENCED_BY_RELATION = YES
+
+# If the REFERENCES_RELATION tag is set to YES (the default)
+# then for each documented function all documented entities
+# called/used by that function will be listed.
+
+REFERENCES_RELATION = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to the alphabetical class index
+#---------------------------------------------------------------------------
+
+# If the ALPHABETICAL_INDEX tag is set to YES, an alphabetical index
+# of all compounds will be generated. Enable this if the project
+# contains a lot of classes, structs, unions or interfaces.
+
+ALPHABETICAL_INDEX = NO
+
+# If the alphabetical index is enabled (see ALPHABETICAL_INDEX) then
+# the COLS_IN_ALPHA_INDEX tag can be used to specify the number of columns
+# in which this list will be split (can be a number in the range [1..20])
+
+COLS_IN_ALPHA_INDEX = 5
+
+# In case all classes in a project start with a common prefix, all
+# classes will be put under the same header in the alphabetical index.
+# The IGNORE_PREFIX tag can be used to specify one or more prefixes that
+# should be ignored while generating the index headers.
+
+IGNORE_PREFIX =
+
+#---------------------------------------------------------------------------
+# configuration options related to the HTML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_HTML tag is set to YES (the default) Doxygen will
+# generate HTML output.
+
+GENERATE_HTML = YES
+
+# The HTML_OUTPUT tag is used to specify where the HTML docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `html' will be used as the default path.
+
+HTML_OUTPUT = ../.doxyout/ndbapi.html
+
+# The HTML_HEADER tag can be used to specify a personal HTML header for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard header.
+
+HTML_HEADER =
+
+# The HTML_FOOTER tag can be used to specify a personal HTML footer for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard footer.
+
+HTML_FOOTER = footer.html
+
+# The HTML_STYLESHEET tag can be used to specify a user defined cascading
+# style sheet that is used by each HTML page. It can be used to
+# fine-tune the look of the HTML output. If the tag is left blank doxygen
+# will generate a default style sheet
+
+HTML_STYLESHEET =
+
+# If the HTML_ALIGN_MEMBERS tag is set to YES, the members of classes,
+# files or namespaces will be aligned in HTML using tables. If set to
+# NO a bullet list will be used.
+
+HTML_ALIGN_MEMBERS = YES
+
+# If the GENERATE_HTMLHELP tag is set to YES, additional index files
+# will be generated that can be used as input for tools like the
+# Microsoft HTML help workshop to generate a compressed HTML help file (.chm)
+# of the generated HTML documentation.
+
+GENERATE_HTMLHELP = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the GENERATE_CHI flag
+# controls if a separate .chi index file is generated (YES) or that
+# it should be included in the master .chm file (NO).
+
+GENERATE_CHI = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the BINARY_TOC flag
+# controls whether a binary table of contents is generated (YES) or a
+# normal table of contents (NO) in the .chm file.
+
+BINARY_TOC = NO
+
+# The TOC_EXPAND flag can be set to YES to add extra items for group members
+# to the contents of the Html help documentation and to the tree view.
+
+TOC_EXPAND = NO
+
+# The DISABLE_INDEX tag can be used to turn on/off the condensed index at
+# top of each HTML page. The value NO (the default) enables the index and
+# the value YES disables it.
+
+DISABLE_INDEX = NO
+
+# This tag can be used to set the number of enum values (range [1..20])
+# that doxygen will group on one line in the generated HTML documentation.
+
+ENUM_VALUES_PER_LINE = 4
+
+# If the GENERATE_TREEVIEW tag is set to YES, a side panel will be
+# generated containing a tree-like index structure (just like the one that
+# is generated for HTML Help). For this to work a browser that supports
+# JavaScript and frames is required (for instance Mozilla, Netscape 4.0+,
+# or Internet explorer 4.0+). Note that for large projects the tree generation
+# can take a very long time. In such cases it is better to disable this feature.
+# Windows users are probably better off using the HTML help feature.
+
+GENERATE_TREEVIEW = NO
+
+# If the treeview is enabled (see GENERATE_TREEVIEW) then this tag can be
+# used to set the initial width (in pixels) of the frame in which the tree
+# is shown.
+
+TREEVIEW_WIDTH = 250
+
+#---------------------------------------------------------------------------
+# configuration options related to the LaTeX output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_LATEX tag is set to YES (the default) Doxygen will
+# generate Latex output.
+
+GENERATE_LATEX = YES
+
+# The LATEX_OUTPUT tag is used to specify where the LaTeX docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `latex' will be used as the default path.
+
+LATEX_OUTPUT = ../.doxyout/ndbapi.latex
+
+# If the COMPACT_LATEX tag is set to YES Doxygen generates more compact
+# LaTeX documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_LATEX = NO
+
+# The PAPER_TYPE tag can be used to set the paper type that is used
+# by the printer. Possible values are: a4, a4wide, letter, legal and
+# executive. If left blank a4wide will be used.
+
+PAPER_TYPE =
+
+# The EXTRA_PACKAGES tag can be to specify one or more names of LaTeX
+# packages that should be included in the LaTeX output.
+
+EXTRA_PACKAGES =
+
+# The LATEX_HEADER tag can be used to specify a personal LaTeX header for
+# the generated latex document. The header should contain everything until
+# the first chapter. If it is left blank doxygen will generate a
+# standard header. Notice: only use this tag if you know what you are doing!
+
+LATEX_HEADER = ../doxygen/header.ndbapi.tex
+
+# If the PDF_HYPERLINKS tag is set to YES, the LaTeX that is generated
+# is prepared for conversion to pdf (using ps2pdf). The pdf file will
+# contain links (just like the HTML output) instead of page references
+# This makes the output suitable for online browsing using a pdf viewer.
+
+PDF_HYPERLINKS = YES
+
+# If the USE_PDFLATEX tag is set to YES, pdflatex will be used instead of
+# plain latex in the generated Makefile. Set this option to YES to get a
+# higher quality PDF documentation.
+
+USE_PDFLATEX = YES
+
+# If the LATEX_BATCHMODE tag is set to YES, doxygen will add the \\batchmode.
+# command to the generated LaTeX files. This will instruct LaTeX to keep
+# running if errors occur, instead of asking the user for help.
+# This option is also used when generating formulas in HTML.
+
+LATEX_BATCHMODE = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the RTF output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_RTF tag is set to YES Doxygen will generate RTF output
+# The RTF output is optimised for Word 97 and may not look very pretty with
+# other RTF readers or editors.
+
+GENERATE_RTF = NO
+
+# The RTF_OUTPUT tag is used to specify where the RTF docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `rtf' will be used as the default path.
+
+RTF_OUTPUT = ../ndbapi.rtf
+
+# If the COMPACT_RTF tag is set to YES Doxygen generates more compact
+# RTF documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_RTF = NO
+
+# If the RTF_HYPERLINKS tag is set to YES, the RTF that is generated
+# will contain hyperlink fields. The RTF file will
+# contain links (just like the HTML output) instead of page references.
+# This makes the output suitable for online browsing using WORD or other
+# programs which support those fields.
+# Note: wordpad (write) and others do not support links.
+
+RTF_HYPERLINKS = NO
+
+# Load stylesheet definitions from file. Syntax is similar to doxygen's
+# config file, i.e. a series of assigments. You only have to provide
+# replacements, missing definitions are set to their default value.
+
+RTF_STYLESHEET_FILE =
+
+# Set optional variables used in the generation of an rtf document.
+# Syntax is similar to doxygen's config file.
+
+RTF_EXTENSIONS_FILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the man page output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_MAN tag is set to YES (the default) Doxygen will
+# generate man pages
+
+GENERATE_MAN = NO
+
+# The MAN_OUTPUT tag is used to specify where the man pages will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `man' will be used as the default path.
+
+MAN_OUTPUT =
+
+# The MAN_EXTENSION tag determines the extension that is added to
+# the generated man pages (default is the subroutine's section .3)
+
+MAN_EXTENSION =
+
+# If the MAN_LINKS tag is set to YES and Doxygen generates man output,
+# then it will generate one additional man file for each entity
+# documented in the real man page(s). These additional files
+# only source the real man page, but without them the man command
+# would be unable to find the correct page. The default is NO.
+
+MAN_LINKS = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the XML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_XML tag is set to YES Doxygen will
+# generate an XML file that captures the structure of
+# the code including all documentation. Note that this
+# feature is still experimental and incomplete at the
+# moment.
+
+GENERATE_XML = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the preprocessor
+#---------------------------------------------------------------------------
+
+# If the ENABLE_PREPROCESSING tag is set to YES (the default) Doxygen will
+# evaluate all C-preprocessor directives found in the sources and include
+# files.
+
+ENABLE_PREPROCESSING = YES
+
+# If the MACRO_EXPANSION tag is set to YES Doxygen will expand all macro
+# names in the source code. If set to NO (the default) only conditional
+# compilation will be performed. Macro expansion can be done in a controlled
+# way by setting EXPAND_ONLY_PREDEF to YES.
+
+MACRO_EXPANSION = YES
+
+# If the EXPAND_ONLY_PREDEF and MACRO_EXPANSION tags are both set to YES
+# then the macro expansion is limited to the macros specified with the
+# PREDEFINED and EXPAND_AS_PREDEFINED tags.
+
+EXPAND_ONLY_PREDEF = YES
+
+# If the SEARCH_INCLUDES tag is set to YES (the default) the includes files
+# in the INCLUDE_PATH (see below) will be search if a #include is found.
+
+SEARCH_INCLUDES = YES
+
+# The INCLUDE_PATH tag can be used to specify one or more directories that
+# contain include files that are not input files but should be processed by
+# the preprocessor.
+
+INCLUDE_PATH =
+
+# You can use the INCLUDE_FILE_PATTERNS tag to specify one or more wildcard
+# patterns (like *.h and *.hpp) to filter out the header-files in the
+# directories. If left blank, the patterns specified with FILE_PATTERNS will
+# be used.
+
+INCLUDE_FILE_PATTERNS =
+
+# The PREDEFINED tag can be used to specify one or more macro names that
+# are defined before the preprocessor is started (similar to the -D option of
+# gcc). The argument of the tag is a list of macros of the form: name
+# or name=definition (no spaces). If the definition and the = are
+# omitted =1 is assumed.
+
+PREDEFINED = DOXYGEN_SHOULD_SKIP_DEPRECATED \
+ DOXYGEN_SHOULD_SKIP_INTERNAL \
+ protected=private
+
+# If the MACRO_EXPANSION and EXPAND_PREDEF_ONLY tags are set to YES then
+# this tag can be used to specify a list of macro names that should be expanded.
+# The macro definition that is found in the sources will be used.
+# Use the PREDEFINED tag if you want to use a different macro definition.
+
+EXPAND_AS_DEFINED =
+
+# If the SKIP_FUNCTION_MACROS tag is set to YES (the default) then
+# doxygen's preprocessor will remove all function-like macros that are alone
+# on a line and do not end with a semicolon. Such function macros are typically
+# used for boiler-plate code, and will confuse the parser if not removed.
+
+SKIP_FUNCTION_MACROS = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to external references
+#---------------------------------------------------------------------------
+
+# The TAGFILES tag can be used to specify one or more tagfiles.
+
+TAGFILES =
+
+# When a file name is specified after GENERATE_TAGFILE, doxygen will create
+# a tag file that is based on the input files it reads.
+
+GENERATE_TAGFILE =
+
+# If the ALLEXTERNALS tag is set to YES all external classes will be listed
+# in the class index. If set to NO only the inherited external classes
+# will be listed.
+
+ALLEXTERNALS = NO
+
+# The PERL_PATH should be the absolute path and name of the perl script
+# interpreter (i.e. the result of `which perl').
+
+PERL_PATH =
+
+#---------------------------------------------------------------------------
+# Configuration options related to the dot tool
+#---------------------------------------------------------------------------
+
+# If the CLASS_DIAGRAMS tag is set to YES (the default) Doxygen will
+# generate a inheritance diagram (in Html, RTF and LaTeX) for classes with base or
+# super classes. Setting the tag to NO turns the diagrams off. Note that this
+# option is superceded by the HAVE_DOT option below. This is only a fallback. It is
+# recommended to install and use dot, since it yield more powerful graphs.
+
+CLASS_DIAGRAMS = YES
+
+# If you set the HAVE_DOT tag to YES then doxygen will assume the dot tool is
+# available from the path. This tool is part of Graphviz, a graph visualization
+# toolkit from AT&T and Lucent Bell Labs. The other options in this section
+# have no effect if this option is set to NO (the default)
+
+HAVE_DOT = NO
+
+# If the CLASS_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect inheritance relations. Setting this tag to YES will force the
+# the CLASS_DIAGRAMS tag to NO.
+
+CLASS_GRAPH = YES
+
+# If the COLLABORATION_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect implementation dependencies (inheritance, containment, and
+# class references variables) of the class with other documented classes.
+
+COLLABORATION_GRAPH = YES
+
+# If set to YES, the inheritance and collaboration graphs will show the
+# relations between templates and their instances.
+
+TEMPLATE_RELATIONS = YES
+
+# If set to YES, the inheritance and collaboration graphs will hide
+# inheritance and usage relations if the target is undocumented
+# or is not a class.
+
+HIDE_UNDOC_RELATIONS = YES
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDE_GRAPH, and HAVE_DOT
+# tags are set to YES then doxygen will generate a graph for each documented
+# file showing the direct and indirect include dependencies of the file with
+# other documented files.
+
+INCLUDE_GRAPH = YES
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDED_BY_GRAPH, and
+# HAVE_DOT tags are set to YES then doxygen will generate a graph for each
+# documented header file showing the documented files that directly or
+# indirectly include this file.
+
+INCLUDED_BY_GRAPH = YES
+
+# If the GRAPHICAL_HIERARCHY and HAVE_DOT tags are set to YES then doxygen
+# will graphical hierarchy of all classes instead of a textual one.
+
+GRAPHICAL_HIERARCHY = YES
+
+# The tag DOT_PATH can be used to specify the path where the dot tool can be
+# found. If left blank, it is assumed the dot tool can be found on the path.
+
+DOT_PATH =
+
+# The DOTFILE_DIRS tag can be used to specify one or more directories that
+# contain dot files that are included in the documentation (see the
+# \dotfile command).
+
+DOTFILE_DIRS =
+
+# The MAX_DOT_GRAPH_WIDTH tag can be used to set the maximum allowed width
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_WIDTH = 1024
+
+# The MAX_DOT_GRAPH_HEIGHT tag can be used to set the maximum allows height
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_HEIGHT = 1024
+
+# If the GENERATE_LEGEND tag is set to YES (the default) Doxygen will
+# generate a legend page explaining the meaning of the various boxes and
+# arrows in the dot generated graphs.
+
+GENERATE_LEGEND = YES
+
+# If the DOT_CLEANUP tag is set to YES (the default) Doxygen will
+# remove the intermedate dot files that are used to generate
+# the various graphs.
+
+DOT_CLEANUP = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to the search engine
+#---------------------------------------------------------------------------
+
+# The SEARCHENGINE tag specifies whether or not a search engine should be
+# used. If set to NO the values of all tags below this one will be ignored.
+
+SEARCHENGINE = NO
+
+# The CGI_NAME tag should be the name of the CGI script that
+# starts the search engine (doxysearch) with the correct parameters.
+# A script with this name will be generated by doxygen.
+
+CGI_NAME =
+
+# The CGI_URL tag should be the absolute URL to the directory where the
+# cgi binaries are located. See the documentation of your http daemon for
+# details.
+
+CGI_URL =
+
+# The DOC_URL tag should be the absolute URL to the directory where the
+# documentation is located. If left blank the absolute path to the
+# documentation, with file:// prepended to it, will be used.
+
+DOC_URL =
+
+# The DOC_ABSPATH tag should be the absolute path to the directory where the
+# documentation is located. If left blank the directory on the local machine
+# will be used.
+
+DOC_ABSPATH =
+
+# The BIN_ABSPATH tag must point to the directory where the doxysearch binary
+# is installed.
+
+BIN_ABSPATH =
+
+# The EXT_DOC_PATHS tag can be used to specify one or more paths to
+# documentation generated for other projects. This allows doxysearch to search
+# the documentation for these projects as well.
+
+EXT_DOC_PATHS =
diff --git a/ndb/docs/doxygen/Doxyfile.odbc b/ndb/docs/doxygen/Doxyfile.odbc
new file mode 100644
index 00000000000..93e052d5b9d
--- /dev/null
+++ b/ndb/docs/doxygen/Doxyfile.odbc
@@ -0,0 +1,921 @@
+# Doxyfile 1.2.14
+
+# This file describes the settings to be used by the documentation system
+# doxygen (www.doxygen.org) for a project
+#
+# All text after a hash (#) is considered a comment and will be ignored
+# The format is:
+# TAG = value [value, ...]
+# For lists items can also be appended using:
+# TAG += value [value, ...]
+# Values that contain spaces should be placed between quotes (" ")
+
+#---------------------------------------------------------------------------
+# General configuration options
+#---------------------------------------------------------------------------
+
+# The PROJECT_NAME tag is a single word (or a sequence of words surrounded
+# by quotes) that should identify the project.
+
+PROJECT_NAME = "NDB ODBC"
+
+# The PROJECT_NUMBER tag can be used to enter a project or revision number.
+# This could be handy for archiving the generated documentation or
+# if some version control system is used.
+
+PROJECT_NUMBER =
+
+# The OUTPUT_DIRECTORY tag is used to specify the (relative or absolute)
+# base path where the generated documentation will be put.
+# If a relative path is entered, it will be relative to the location
+# where doxygen was started. If left blank the current directory will be used.
+
+OUTPUT_DIRECTORY =
+
+# The OUTPUT_LANGUAGE tag is used to specify the language in which all
+# documentation generated by doxygen is written. Doxygen will use this
+# information to generate all constant output in the proper language.
+# The default language is English, other supported languages are:
+# Brazilian, Chinese, Croatian, Czech, Danish, Dutch, Finnish, French,
+# German, Greek, Hungarian, Italian, Japanese, Korean, Norwegian, Polish,
+# Portuguese, Romanian, Russian, Slovak, Slovene, Spanish and Swedish.
+
+OUTPUT_LANGUAGE = English
+
+# If the EXTRACT_ALL tag is set to YES doxygen will assume all entities in
+# documentation are documented, even if no documentation was available.
+# Private class members and static file members will be hidden unless
+# the EXTRACT_PRIVATE and EXTRACT_STATIC tags are set to YES
+
+EXTRACT_ALL = YES
+
+# If the EXTRACT_PRIVATE tag is set to YES all private members of a class
+# will be included in the documentation.
+
+EXTRACT_PRIVATE = YES
+
+# If the EXTRACT_STATIC tag is set to YES all static members of a file
+# will be included in the documentation.
+
+EXTRACT_STATIC = YES
+
+# If the EXTRACT_LOCAL_CLASSES tag is set to YES classes (and structs)
+# defined locally in source files will be included in the documentation.
+# If set to NO only classes defined in header files are included.
+
+EXTRACT_LOCAL_CLASSES = YES
+
+# If the HIDE_UNDOC_MEMBERS tag is set to YES, Doxygen will hide all
+# undocumented members of documented classes, files or namespaces.
+# If set to NO (the default) these members will be included in the
+# various overviews, but no documentation section is generated.
+# This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_MEMBERS = NO
+
+# If the HIDE_UNDOC_CLASSES tag is set to YES, Doxygen will hide all
+# undocumented classes that are normally visible in the class hierarchy.
+# If set to NO (the default) these class will be included in the various
+# overviews. This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_CLASSES = NO
+
+# If the BRIEF_MEMBER_DESC tag is set to YES (the default) Doxygen will
+# include brief member descriptions after the members that are listed in
+# the file and class documentation (similar to JavaDoc).
+# Set to NO to disable this.
+
+BRIEF_MEMBER_DESC = YES
+
+# If the REPEAT_BRIEF tag is set to YES (the default) Doxygen will prepend
+# the brief description of a member or function before the detailed description.
+# Note: if both HIDE_UNDOC_MEMBERS and BRIEF_MEMBER_DESC are set to NO, the
+# brief descriptions will be completely suppressed.
+
+REPEAT_BRIEF = YES
+
+# If the ALWAYS_DETAILED_SEC and REPEAT_BRIEF tags are both set to YES then
+# Doxygen will generate a detailed section even if there is only a brief
+# description.
+
+ALWAYS_DETAILED_SEC = NO
+
+# If the INLINE_INHERITED_MEMB tag is set to YES, doxygen will show all inherited
+# members of a class in the documentation of that class as if those members were
+# ordinary class members. Constructors, destructors and assignment operators of
+# the base classes will not be shown.
+
+INLINE_INHERITED_MEMB = NO
+
+# If the FULL_PATH_NAMES tag is set to YES then Doxygen will prepend the full
+# path before files name in the file list and in the header files. If set
+# to NO the shortest path that makes the file name unique will be used.
+
+FULL_PATH_NAMES = YES
+
+# If the FULL_PATH_NAMES tag is set to YES then the STRIP_FROM_PATH tag
+# can be used to strip a user defined part of the path. Stripping is
+# only done if one of the specified strings matches the left-hand part of
+# the path. It is allowed to use relative paths in the argument list.
+
+STRIP_FROM_PATH = .
+
+# The INTERNAL_DOCS tag determines if documentation
+# that is typed after a \internal command is included. If the tag is set
+# to NO (the default) then the documentation will be excluded.
+# Set it to YES to include the internal documentation.
+
+INTERNAL_DOCS = YES
+
+# Setting the STRIP_CODE_COMMENTS tag to YES (the default) will instruct
+# doxygen to hide any special comment blocks from generated source code
+# fragments. Normal C and C++ comments will always remain visible.
+
+STRIP_CODE_COMMENTS = YES
+
+# If the CASE_SENSE_NAMES tag is set to NO then Doxygen will only generate
+# file names in lower case letters. If set to YES upper case letters are also
+# allowed. This is useful if you have classes or files whose names only differ
+# in case and if your file system supports case sensitive file names. Windows
+# users are adviced to set this option to NO.
+
+CASE_SENSE_NAMES = YES
+
+# If the SHORT_NAMES tag is set to YES, doxygen will generate much shorter
+# (but less readable) file names. This can be useful is your file systems
+# doesn't support long names like on DOS, Mac, or CD-ROM.
+
+SHORT_NAMES = NO
+
+# If the HIDE_SCOPE_NAMES tag is set to NO (the default) then Doxygen
+# will show members with their full class and namespace scopes in the
+# documentation. If set to YES the scope will be hidden.
+
+HIDE_SCOPE_NAMES = NO
+
+# If the VERBATIM_HEADERS tag is set to YES (the default) then Doxygen
+# will generate a verbatim copy of the header file for each class for
+# which an include is specified. Set to NO to disable this.
+
+VERBATIM_HEADERS = YES
+
+# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
+# will put list of the files that are included by a file in the documentation
+# of that file.
+
+SHOW_INCLUDE_FILES = YES
+
+# If the JAVADOC_AUTOBRIEF tag is set to YES then Doxygen
+# will interpret the first line (until the first dot) of a JavaDoc-style
+# comment as the brief description. If set to NO, the JavaDoc
+# comments will behave just like the Qt-style comments (thus requiring an
+# explict @brief command for a brief description.
+
+JAVADOC_AUTOBRIEF = NO
+
+# If the INHERIT_DOCS tag is set to YES (the default) then an undocumented
+# member inherits the documentation from any documented member that it
+# reimplements.
+
+INHERIT_DOCS = YES
+
+# If the INLINE_INFO tag is set to YES (the default) then a tag [inline]
+# is inserted in the documentation for inline members.
+
+INLINE_INFO = YES
+
+# If the SORT_MEMBER_DOCS tag is set to YES (the default) then doxygen
+# will sort the (detailed) documentation of file and class members
+# alphabetically by member name. If set to NO the members will appear in
+# declaration order.
+
+SORT_MEMBER_DOCS = YES
+
+# If member grouping is used in the documentation and the DISTRIBUTE_GROUP_DOC
+# tag is set to YES, then doxygen will reuse the documentation of the first
+# member in the group (if any) for the other members of the group. By default
+# all members of a group must be documented explicitly.
+
+DISTRIBUTE_GROUP_DOC = NO
+
+# The TAB_SIZE tag can be used to set the number of spaces in a tab.
+# Doxygen uses this value to replace tabs by spaces in code fragments.
+
+TAB_SIZE = 8
+
+# The GENERATE_TODOLIST tag can be used to enable (YES) or
+# disable (NO) the todo list. This list is created by putting \todo
+# commands in the documentation.
+
+GENERATE_TODOLIST = YES
+
+# The GENERATE_TESTLIST tag can be used to enable (YES) or
+# disable (NO) the test list. This list is created by putting \test
+# commands in the documentation.
+
+GENERATE_TESTLIST = YES
+
+# The GENERATE_BUGLIST tag can be used to enable (YES) or
+# disable (NO) the bug list. This list is created by putting \bug
+# commands in the documentation.
+
+GENERATE_BUGLIST = YES
+
+# This tag can be used to specify a number of aliases that acts
+# as commands in the documentation. An alias has the form "name=value".
+# For example adding "sideeffect=\par Side Effects:\n" will allow you to
+# put the command \sideeffect (or @sideeffect) in the documentation, which
+# will result in a user defined paragraph with heading "Side Effects:".
+# You can put \n's in the value part of an alias to insert newlines.
+
+ALIASES =
+
+# The ENABLED_SECTIONS tag can be used to enable conditional
+# documentation sections, marked by \if sectionname ... \endif.
+
+ENABLED_SECTIONS =
+
+# The MAX_INITIALIZER_LINES tag determines the maximum number of lines
+# the initial value of a variable or define consist of for it to appear in
+# the documentation. If the initializer consists of more lines than specified
+# here it will be hidden. Use a value of 0 to hide initializers completely.
+# The appearance of the initializer of individual variables and defines in the
+# documentation can be controlled using \showinitializer or \hideinitializer
+# command in the documentation regardless of this setting.
+
+MAX_INITIALIZER_LINES = 30
+
+# Set the OPTIMIZE_OUTPUT_FOR_C tag to YES if your project consists of C sources
+# only. Doxygen will then generate output that is more tailored for C.
+# For instance some of the names that are used will be different. The list
+# of all members will be omitted, etc.
+
+OPTIMIZE_OUTPUT_FOR_C = NO
+
+# Set the SHOW_USED_FILES tag to NO to disable the list of files generated
+# at the bottom of the documentation of classes and structs. If set to YES the
+# list will mention the files that were used to generate the documentation.
+
+SHOW_USED_FILES = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to warning and progress messages
+#---------------------------------------------------------------------------
+
+# The QUIET tag can be used to turn on/off the messages that are generated
+# by doxygen. Possible values are YES and NO. If left blank NO is used.
+
+QUIET = NO
+
+# The WARNINGS tag can be used to turn on/off the warning messages that are
+# generated by doxygen. Possible values are YES and NO. If left blank
+# NO is used.
+
+WARNINGS = YES
+
+# If WARN_IF_UNDOCUMENTED is set to YES, then doxygen will generate warnings
+# for undocumented members. If EXTRACT_ALL is set to YES then this flag will
+# automatically be disabled.
+
+WARN_IF_UNDOCUMENTED = YES
+
+# The WARN_FORMAT tag determines the format of the warning messages that
+# doxygen can produce. The string should contain the $file, $line, and $text
+# tags, which will be replaced by the file and line number from which the
+# warning originated and the warning text.
+
+WARN_FORMAT = "$file:$line: $text"
+
+# The WARN_LOGFILE tag can be used to specify a file to which warning
+# and error messages should be written. If left blank the output is written
+# to stderr.
+
+WARN_LOGFILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the input files
+#---------------------------------------------------------------------------
+
+# The INPUT tag can be used to specify the files and/or directories that contain
+# documented source files. You may enter file names like "myfile.cpp" or
+# directories like "/usr/src/myproject". Separate the files or directories
+# with spaces.
+
+INPUT = src/client/odbc
+
+# If the value of the INPUT tag contains directories, you can use the
+# FILE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank the following patterns are tested:
+# *.c *.cc *.cxx *.cpp *.c++ *.java *.ii *.ixx *.ipp *.i++ *.inl *.h *.hh *.hxx *.hpp
+# *.h++ *.idl *.odl
+
+FILE_PATTERNS =
+
+# The RECURSIVE tag can be used to turn specify whether or not subdirectories
+# should be searched for input files as well. Possible values are YES and NO.
+# If left blank NO is used.
+
+RECURSIVE = YES
+
+# The EXCLUDE tag can be used to specify files and/or directories that should
+# excluded from the INPUT source files. This way you can easily exclude a
+# subdirectory from a directory tree whose root is specified with the INPUT tag.
+
+EXCLUDE =
+
+# The EXCLUDE_SYMLINKS tag can be used select whether or not files or directories
+# that are symbolic links (a Unix filesystem feature) are excluded from the input.
+
+EXCLUDE_SYMLINKS = NO
+
+# If the value of the INPUT tag contains directories, you can use the
+# EXCLUDE_PATTERNS tag to specify one or more wildcard patterns to exclude
+# certain files from those directories.
+
+EXCLUDE_PATTERNS =
+
+# The EXAMPLE_PATH tag can be used to specify one or more files or
+# directories that contain example code fragments that are included (see
+# the \include command).
+
+EXAMPLE_PATH =
+
+# If the value of the EXAMPLE_PATH tag contains directories, you can use the
+# EXAMPLE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank all files are included.
+
+EXAMPLE_PATTERNS =
+
+# If the EXAMPLE_RECURSIVE tag is set to YES then subdirectories will be
+# searched for input files to be used with the \include or \dontinclude
+# commands irrespective of the value of the RECURSIVE tag.
+# Possible values are YES and NO. If left blank NO is used.
+
+EXAMPLE_RECURSIVE = NO
+
+# The IMAGE_PATH tag can be used to specify one or more files or
+# directories that contain image that are included in the documentation (see
+# the \image command).
+
+IMAGE_PATH =
+
+# The INPUT_FILTER tag can be used to specify a program that doxygen should
+# invoke to filter for each input file. Doxygen will invoke the filter program
+# by executing (via popen()) the command <filter> <input-file>, where <filter>
+# is the value of the INPUT_FILTER tag, and <input-file> is the name of an
+# input file. Doxygen will then use the output that the filter program writes
+# to standard output.
+
+INPUT_FILTER =
+
+# If the FILTER_SOURCE_FILES tag is set to YES, the input filter (if set using
+# INPUT_FILTER) will be used to filter the input files when producing source
+# files to browse.
+
+FILTER_SOURCE_FILES = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to source browsing
+#---------------------------------------------------------------------------
+
+# If the SOURCE_BROWSER tag is set to YES then a list of source files will
+# be generated. Documented entities will be cross-referenced with these sources.
+
+SOURCE_BROWSER = YES
+
+# Setting the INLINE_SOURCES tag to YES will include the body
+# of functions and classes directly in the documentation.
+
+INLINE_SOURCES = YES
+
+# If the REFERENCED_BY_RELATION tag is set to YES (the default)
+# then for each documented function all documented
+# functions referencing it will be listed.
+
+REFERENCED_BY_RELATION = YES
+
+# If the REFERENCES_RELATION tag is set to YES (the default)
+# then for each documented function all documented entities
+# called/used by that function will be listed.
+
+REFERENCES_RELATION = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to the alphabetical class index
+#---------------------------------------------------------------------------
+
+# If the ALPHABETICAL_INDEX tag is set to YES, an alphabetical index
+# of all compounds will be generated. Enable this if the project
+# contains a lot of classes, structs, unions or interfaces.
+
+ALPHABETICAL_INDEX = YES
+
+# If the alphabetical index is enabled (see ALPHABETICAL_INDEX) then
+# the COLS_IN_ALPHA_INDEX tag can be used to specify the number of columns
+# in which this list will be split (can be a number in the range [1..20])
+
+COLS_IN_ALPHA_INDEX = 5
+
+# In case all classes in a project start with a common prefix, all
+# classes will be put under the same header in the alphabetical index.
+# The IGNORE_PREFIX tag can be used to specify one or more prefixes that
+# should be ignored while generating the index headers.
+
+IGNORE_PREFIX =
+
+#---------------------------------------------------------------------------
+# configuration options related to the HTML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_HTML tag is set to YES (the default) Doxygen will
+# generate HTML output.
+
+GENERATE_HTML = YES
+
+# The HTML_OUTPUT tag is used to specify where the HTML docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `html' will be used as the default path.
+
+HTML_OUTPUT = ../.doxyout/odbc.html
+
+# The HTML_FILE_EXTENSION tag can be used to specify the file extension for
+# each generated HTML page (for example: .htm,.php,.asp). If it is left blank
+# doxygen will generate files with .html extension.
+
+HTML_FILE_EXTENSION = .html
+
+# The HTML_HEADER tag can be used to specify a personal HTML header for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard header.
+
+HTML_HEADER =
+
+# The HTML_FOOTER tag can be used to specify a personal HTML footer for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard footer.
+
+HTML_FOOTER =
+
+# The HTML_STYLESHEET tag can be used to specify a user defined cascading
+# style sheet that is used by each HTML page. It can be used to
+# fine-tune the look of the HTML output. If the tag is left blank doxygen
+# will generate a default style sheet
+
+HTML_STYLESHEET =
+
+# If the HTML_ALIGN_MEMBERS tag is set to YES, the members of classes,
+# files or namespaces will be aligned in HTML using tables. If set to
+# NO a bullet list will be used.
+
+HTML_ALIGN_MEMBERS = YES
+
+# If the GENERATE_HTMLHELP tag is set to YES, additional index files
+# will be generated that can be used as input for tools like the
+# Microsoft HTML help workshop to generate a compressed HTML help file (.chm)
+# of the generated HTML documentation.
+
+GENERATE_HTMLHELP = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the GENERATE_CHI flag
+# controls if a separate .chi index file is generated (YES) or that
+# it should be included in the master .chm file (NO).
+
+GENERATE_CHI = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the BINARY_TOC flag
+# controls whether a binary table of contents is generated (YES) or a
+# normal table of contents (NO) in the .chm file.
+
+BINARY_TOC = NO
+
+# The TOC_EXPAND flag can be set to YES to add extra items for group members
+# to the contents of the Html help documentation and to the tree view.
+
+TOC_EXPAND = NO
+
+# The DISABLE_INDEX tag can be used to turn on/off the condensed index at
+# top of each HTML page. The value NO (the default) enables the index and
+# the value YES disables it.
+
+DISABLE_INDEX = NO
+
+# This tag can be used to set the number of enum values (range [1..20])
+# that doxygen will group on one line in the generated HTML documentation.
+
+ENUM_VALUES_PER_LINE = 4
+
+# If the GENERATE_TREEVIEW tag is set to YES, a side panel will be
+# generated containing a tree-like index structure (just like the one that
+# is generated for HTML Help). For this to work a browser that supports
+# JavaScript and frames is required (for instance Mozilla, Netscape 4.0+,
+# or Internet explorer 4.0+). Note that for large projects the tree generation
+# can take a very long time. In such cases it is better to disable this feature.
+# Windows users are probably better off using the HTML help feature.
+
+GENERATE_TREEVIEW = NO
+
+# If the treeview is enabled (see GENERATE_TREEVIEW) then this tag can be
+# used to set the initial width (in pixels) of the frame in which the tree
+# is shown.
+
+TREEVIEW_WIDTH = 250
+
+#---------------------------------------------------------------------------
+# configuration options related to the LaTeX output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_LATEX tag is set to YES (the default) Doxygen will
+# generate Latex output.
+
+GENERATE_LATEX = NO
+
+# The LATEX_OUTPUT tag is used to specify where the LaTeX docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `latex' will be used as the default path.
+
+LATEX_OUTPUT = ../.doxyout/odbc.latex
+
+# If the COMPACT_LATEX tag is set to YES Doxygen generates more compact
+# LaTeX documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_LATEX = NO
+
+# The PAPER_TYPE tag can be used to set the paper type that is used
+# by the printer. Possible values are: a4, a4wide, letter, legal and
+# executive. If left blank a4wide will be used.
+
+PAPER_TYPE = a4wide
+
+# The EXTRA_PACKAGES tag can be to specify one or more names of LaTeX
+# packages that should be included in the LaTeX output.
+
+EXTRA_PACKAGES =
+
+# The LATEX_HEADER tag can be used to specify a personal LaTeX header for
+# the generated latex document. The header should contain everything until
+# the first chapter. If it is left blank doxygen will generate a
+# standard header. Notice: only use this tag if you know what you are doing!
+
+LATEX_HEADER =
+
+# If the PDF_HYPERLINKS tag is set to YES, the LaTeX that is generated
+# is prepared for conversion to pdf (using ps2pdf). The pdf file will
+# contain links (just like the HTML output) instead of page references
+# This makes the output suitable for online browsing using a pdf viewer.
+
+PDF_HYPERLINKS = NO
+
+# If the USE_PDFLATEX tag is set to YES, pdflatex will be used instead of
+# plain latex in the generated Makefile. Set this option to YES to get a
+# higher quality PDF documentation.
+
+USE_PDFLATEX = NO
+
+# If the LATEX_BATCHMODE tag is set to YES, doxygen will add the \\batchmode.
+# command to the generated LaTeX files. This will instruct LaTeX to keep
+# running if errors occur, instead of asking the user for help.
+# This option is also used when generating formulas in HTML.
+
+LATEX_BATCHMODE = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the RTF output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_RTF tag is set to YES Doxygen will generate RTF output
+# The RTF output is optimised for Word 97 and may not look very pretty with
+# other RTF readers or editors.
+
+GENERATE_RTF = NO
+
+# The RTF_OUTPUT tag is used to specify where the RTF docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `rtf' will be used as the default path.
+
+RTF_OUTPUT = rtf
+
+# If the COMPACT_RTF tag is set to YES Doxygen generates more compact
+# RTF documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_RTF = NO
+
+# If the RTF_HYPERLINKS tag is set to YES, the RTF that is generated
+# will contain hyperlink fields. The RTF file will
+# contain links (just like the HTML output) instead of page references.
+# This makes the output suitable for online browsing using WORD or other
+# programs which support those fields.
+# Note: wordpad (write) and others do not support links.
+
+RTF_HYPERLINKS = NO
+
+# Load stylesheet definitions from file. Syntax is similar to doxygen's
+# config file, i.e. a series of assigments. You only have to provide
+# replacements, missing definitions are set to their default value.
+
+RTF_STYLESHEET_FILE =
+
+# Set optional variables used in the generation of an rtf document.
+# Syntax is similar to doxygen's config file.
+
+RTF_EXTENSIONS_FILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the man page output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_MAN tag is set to YES (the default) Doxygen will
+# generate man pages
+
+GENERATE_MAN = NO
+
+# The MAN_OUTPUT tag is used to specify where the man pages will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `man' will be used as the default path.
+
+MAN_OUTPUT = man
+
+# The MAN_EXTENSION tag determines the extension that is added to
+# the generated man pages (default is the subroutine's section .3)
+
+MAN_EXTENSION = .3
+
+# If the MAN_LINKS tag is set to YES and Doxygen generates man output,
+# then it will generate one additional man file for each entity
+# documented in the real man page(s). These additional files
+# only source the real man page, but without them the man command
+# would be unable to find the correct page. The default is NO.
+
+MAN_LINKS = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the XML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_XML tag is set to YES Doxygen will
+# generate an XML file that captures the structure of
+# the code including all documentation. Note that this
+# feature is still experimental and incomplete at the
+# moment.
+
+GENERATE_XML = NO
+
+#---------------------------------------------------------------------------
+# configuration options for the AutoGen Definitions output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_AUTOGEN_DEF tag is set to YES Doxygen will
+# generate an AutoGen Definitions (see autogen.sf.net) file
+# that captures the structure of the code including all
+# documentation. Note that this feature is still experimental
+# and incomplete at the moment.
+
+GENERATE_AUTOGEN_DEF = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the preprocessor
+#---------------------------------------------------------------------------
+
+# If the ENABLE_PREPROCESSING tag is set to YES (the default) Doxygen will
+# evaluate all C-preprocessor directives found in the sources and include
+# files.
+
+ENABLE_PREPROCESSING = YES
+
+# If the MACRO_EXPANSION tag is set to YES Doxygen will expand all macro
+# names in the source code. If set to NO (the default) only conditional
+# compilation will be performed. Macro expansion can be done in a controlled
+# way by setting EXPAND_ONLY_PREDEF to YES.
+
+MACRO_EXPANSION = NO
+
+# If the EXPAND_ONLY_PREDEF and MACRO_EXPANSION tags are both set to YES
+# then the macro expansion is limited to the macros specified with the
+# PREDEFINED and EXPAND_AS_PREDEFINED tags.
+
+EXPAND_ONLY_PREDEF = NO
+
+# If the SEARCH_INCLUDES tag is set to YES (the default) the includes files
+# in the INCLUDE_PATH (see below) will be search if a #include is found.
+
+SEARCH_INCLUDES = YES
+
+# The INCLUDE_PATH tag can be used to specify one or more directories that
+# contain include files that are not input files but should be processed by
+# the preprocessor.
+
+INCLUDE_PATH =
+
+# You can use the INCLUDE_FILE_PATTERNS tag to specify one or more wildcard
+# patterns (like *.h and *.hpp) to filter out the header-files in the
+# directories. If left blank, the patterns specified with FILE_PATTERNS will
+# be used.
+
+INCLUDE_FILE_PATTERNS =
+
+# The PREDEFINED tag can be used to specify one or more macro names that
+# are defined before the preprocessor is started (similar to the -D option of
+# gcc). The argument of the tag is a list of macros of the form: name
+# or name=definition (no spaces). If the definition and the = are
+# omitted =1 is assumed.
+
+PREDEFINED =
+
+# If the MACRO_EXPANSION and EXPAND_PREDEF_ONLY tags are set to YES then
+# this tag can be used to specify a list of macro names that should be expanded.
+# The macro definition that is found in the sources will be used.
+# Use the PREDEFINED tag if you want to use a different macro definition.
+
+EXPAND_AS_DEFINED =
+
+# If the SKIP_FUNCTION_MACROS tag is set to YES (the default) then
+# doxygen's preprocessor will remove all function-like macros that are alone
+# on a line and do not end with a semicolon. Such function macros are typically
+# used for boiler-plate code, and will confuse the parser if not removed.
+
+SKIP_FUNCTION_MACROS = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to external references
+#---------------------------------------------------------------------------
+
+# The TAGFILES tag can be used to specify one or more tagfiles.
+
+TAGFILES =
+
+# When a file name is specified after GENERATE_TAGFILE, doxygen will create
+# a tag file that is based on the input files it reads.
+
+GENERATE_TAGFILE =
+
+# If the ALLEXTERNALS tag is set to YES all external classes will be listed
+# in the class index. If set to NO only the inherited external classes
+# will be listed.
+
+ALLEXTERNALS = NO
+
+# If the EXTERNAL_GROUPS tag is set to YES all external groups will be listed
+# in the modules index. If set to NO, only the current project's groups will
+# be listed.
+
+EXTERNAL_GROUPS = YES
+
+# The PERL_PATH should be the absolute path and name of the perl script
+# interpreter (i.e. the result of `which perl').
+
+PERL_PATH = /usr/bin/perl
+
+#---------------------------------------------------------------------------
+# Configuration options related to the dot tool
+#---------------------------------------------------------------------------
+
+# If the CLASS_DIAGRAMS tag is set to YES (the default) Doxygen will
+# generate a inheritance diagram (in Html, RTF and LaTeX) for classes with base or
+# super classes. Setting the tag to NO turns the diagrams off. Note that this
+# option is superceded by the HAVE_DOT option below. This is only a fallback. It is
+# recommended to install and use dot, since it yield more powerful graphs.
+
+CLASS_DIAGRAMS = YES
+
+# If you set the HAVE_DOT tag to YES then doxygen will assume the dot tool is
+# available from the path. This tool is part of Graphviz, a graph visualization
+# toolkit from AT&T and Lucent Bell Labs. The other options in this section
+# have no effect if this option is set to NO (the default)
+
+HAVE_DOT = YES
+
+# If the CLASS_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect inheritance relations. Setting this tag to YES will force the
+# the CLASS_DIAGRAMS tag to NO.
+
+CLASS_GRAPH = YES
+
+# If the COLLABORATION_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect implementation dependencies (inheritance, containment, and
+# class references variables) of the class with other documented classes.
+
+COLLABORATION_GRAPH = YES
+
+# If set to YES, the inheritance and collaboration graphs will show the
+# relations between templates and their instances.
+
+TEMPLATE_RELATIONS = YES
+
+# If set to YES, the inheritance and collaboration graphs will hide
+# inheritance and usage relations if the target is undocumented
+# or is not a class.
+
+HIDE_UNDOC_RELATIONS = NO
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDE_GRAPH, and HAVE_DOT
+# tags are set to YES then doxygen will generate a graph for each documented
+# file showing the direct and indirect include dependencies of the file with
+# other documented files.
+
+INCLUDE_GRAPH = YES
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDED_BY_GRAPH, and
+# HAVE_DOT tags are set to YES then doxygen will generate a graph for each
+# documented header file showing the documented files that directly or
+# indirectly include this file.
+
+INCLUDED_BY_GRAPH = YES
+
+# If the GRAPHICAL_HIERARCHY and HAVE_DOT tags are set to YES then doxygen
+# will graphical hierarchy of all classes instead of a textual one.
+
+GRAPHICAL_HIERARCHY = YES
+
+# The DOT_IMAGE_FORMAT tag can be used to set the image format of the images
+# generated by dot. Possible values are gif, jpg, and png
+# If left blank gif will be used.
+
+DOT_IMAGE_FORMAT = gif
+
+# The tag DOT_PATH can be used to specify the path where the dot tool can be
+# found. If left blank, it is assumed the dot tool can be found on the path.
+
+DOT_PATH =
+
+# The DOTFILE_DIRS tag can be used to specify one or more directories that
+# contain dot files that are included in the documentation (see the
+# \dotfile command).
+
+DOTFILE_DIRS =
+
+# The MAX_DOT_GRAPH_WIDTH tag can be used to set the maximum allowed width
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_WIDTH = 1024
+
+# The MAX_DOT_GRAPH_HEIGHT tag can be used to set the maximum allows height
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_HEIGHT = 1024
+
+# If the GENERATE_LEGEND tag is set to YES (the default) Doxygen will
+# generate a legend page explaining the meaning of the various boxes and
+# arrows in the dot generated graphs.
+
+GENERATE_LEGEND = YES
+
+# If the DOT_CLEANUP tag is set to YES (the default) Doxygen will
+# remove the intermedate dot files that are used to generate
+# the various graphs.
+
+DOT_CLEANUP = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to the search engine
+#---------------------------------------------------------------------------
+
+# The SEARCHENGINE tag specifies whether or not a search engine should be
+# used. If set to NO the values of all tags below this one will be ignored.
+
+SEARCHENGINE = YES
+
+# The CGI_NAME tag should be the name of the CGI script that
+# starts the search engine (doxysearch) with the correct parameters.
+# A script with this name will be generated by doxygen.
+
+CGI_NAME = search.cgi
+
+# The CGI_URL tag should be the absolute URL to the directory where the
+# cgi binaries are located. See the documentation of your http daemon for
+# details.
+
+CGI_URL =
+
+# The DOC_URL tag should be the absolute URL to the directory where the
+# documentation is located. If left blank the absolute path to the
+# documentation, with file:// prepended to it, will be used.
+
+DOC_URL =
+
+# The DOC_ABSPATH tag should be the absolute path to the directory where the
+# documentation is located. If left blank the directory on the local machine
+# will be used.
+
+DOC_ABSPATH =
+
+# The BIN_ABSPATH tag must point to the directory where the doxysearch binary
+# is installed.
+
+BIN_ABSPATH = /usr/local/bin/
+
+# The EXT_DOC_PATHS tag can be used to specify one or more paths to
+# documentation generated for other projects. This allows doxysearch to search
+# the documentation for these projects as well.
+
+EXT_DOC_PATHS =
diff --git a/ndb/docs/doxygen/Doxyfile.test b/ndb/docs/doxygen/Doxyfile.test
new file mode 100644
index 00000000000..34ee21873ff
--- /dev/null
+++ b/ndb/docs/doxygen/Doxyfile.test
@@ -0,0 +1,921 @@
+# Doxyfile 1.2.14
+
+# This file describes the settings to be used by the documentation system
+# doxygen (www.doxygen.org) for a project
+#
+# All text after a hash (#) is considered a comment and will be ignored
+# The format is:
+# TAG = value [value, ...]
+# For lists items can also be appended using:
+# TAG += value [value, ...]
+# Values that contain spaces should be placed between quotes (" ")
+
+#---------------------------------------------------------------------------
+# General configuration options
+#---------------------------------------------------------------------------
+
+# The PROJECT_NAME tag is a single word (or a sequence of words surrounded
+# by quotes) that should identify the project.
+
+PROJECT_NAME = "NDB Cluster Test Programs"
+
+# The PROJECT_NUMBER tag can be used to enter a project or revision number.
+# This could be handy for archiving the generated documentation or
+# if some version control system is used.
+
+PROJECT_NUMBER =
+
+# The OUTPUT_DIRECTORY tag is used to specify the (relative or absolute)
+# base path where the generated documentation will be put.
+# If a relative path is entered, it will be relative to the location
+# where doxygen was started. If left blank the current directory will be used.
+
+OUTPUT_DIRECTORY =
+
+# The OUTPUT_LANGUAGE tag is used to specify the language in which all
+# documentation generated by doxygen is written. Doxygen will use this
+# information to generate all constant output in the proper language.
+# The default language is English, other supported languages are:
+# Brazilian, Chinese, Croatian, Czech, Danish, Dutch, Finnish, French,
+# German, Greek, Hungarian, Italian, Japanese, Korean, Norwegian, Polish,
+# Portuguese, Romanian, Russian, Slovak, Slovene, Spanish and Swedish.
+
+OUTPUT_LANGUAGE = English
+
+# If the EXTRACT_ALL tag is set to YES doxygen will assume all entities in
+# documentation are documented, even if no documentation was available.
+# Private class members and static file members will be hidden unless
+# the EXTRACT_PRIVATE and EXTRACT_STATIC tags are set to YES
+
+EXTRACT_ALL = YES
+
+# If the EXTRACT_PRIVATE tag is set to YES all private members of a class
+# will be included in the documentation.
+
+EXTRACT_PRIVATE = YES
+
+# If the EXTRACT_STATIC tag is set to YES all static members of a file
+# will be included in the documentation.
+
+EXTRACT_STATIC = YES
+
+# If the EXTRACT_LOCAL_CLASSES tag is set to YES classes (and structs)
+# defined locally in source files will be included in the documentation.
+# If set to NO only classes defined in header files are included.
+
+EXTRACT_LOCAL_CLASSES = YES
+
+# If the HIDE_UNDOC_MEMBERS tag is set to YES, Doxygen will hide all
+# undocumented members of documented classes, files or namespaces.
+# If set to NO (the default) these members will be included in the
+# various overviews, but no documentation section is generated.
+# This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_MEMBERS = NO
+
+# If the HIDE_UNDOC_CLASSES tag is set to YES, Doxygen will hide all
+# undocumented classes that are normally visible in the class hierarchy.
+# If set to NO (the default) these class will be included in the various
+# overviews. This option has no effect if EXTRACT_ALL is enabled.
+
+HIDE_UNDOC_CLASSES = NO
+
+# If the BRIEF_MEMBER_DESC tag is set to YES (the default) Doxygen will
+# include brief member descriptions after the members that are listed in
+# the file and class documentation (similar to JavaDoc).
+# Set to NO to disable this.
+
+BRIEF_MEMBER_DESC = YES
+
+# If the REPEAT_BRIEF tag is set to YES (the default) Doxygen will prepend
+# the brief description of a member or function before the detailed description.
+# Note: if both HIDE_UNDOC_MEMBERS and BRIEF_MEMBER_DESC are set to NO, the
+# brief descriptions will be completely suppressed.
+
+REPEAT_BRIEF = YES
+
+# If the ALWAYS_DETAILED_SEC and REPEAT_BRIEF tags are both set to YES then
+# Doxygen will generate a detailed section even if there is only a brief
+# description.
+
+ALWAYS_DETAILED_SEC = NO
+
+# If the INLINE_INHERITED_MEMB tag is set to YES, doxygen will show all inherited
+# members of a class in the documentation of that class as if those members were
+# ordinary class members. Constructors, destructors and assignment operators of
+# the base classes will not be shown.
+
+INLINE_INHERITED_MEMB = NO
+
+# If the FULL_PATH_NAMES tag is set to YES then Doxygen will prepend the full
+# path before files name in the file list and in the header files. If set
+# to NO the shortest path that makes the file name unique will be used.
+
+FULL_PATH_NAMES = YES
+
+# If the FULL_PATH_NAMES tag is set to YES then the STRIP_FROM_PATH tag
+# can be used to strip a user defined part of the path. Stripping is
+# only done if one of the specified strings matches the left-hand part of
+# the path. It is allowed to use relative paths in the argument list.
+
+STRIP_FROM_PATH = .
+
+# The INTERNAL_DOCS tag determines if documentation
+# that is typed after a \internal command is included. If the tag is set
+# to NO (the default) then the documentation will be excluded.
+# Set it to YES to include the internal documentation.
+
+INTERNAL_DOCS = YES
+
+# Setting the STRIP_CODE_COMMENTS tag to YES (the default) will instruct
+# doxygen to hide any special comment blocks from generated source code
+# fragments. Normal C and C++ comments will always remain visible.
+
+STRIP_CODE_COMMENTS = YES
+
+# If the CASE_SENSE_NAMES tag is set to NO then Doxygen will only generate
+# file names in lower case letters. If set to YES upper case letters are also
+# allowed. This is useful if you have classes or files whose names only differ
+# in case and if your file system supports case sensitive file names. Windows
+# users are adviced to set this option to NO.
+
+CASE_SENSE_NAMES = YES
+
+# If the SHORT_NAMES tag is set to YES, doxygen will generate much shorter
+# (but less readable) file names. This can be useful is your file systems
+# doesn't support long names like on DOS, Mac, or CD-ROM.
+
+SHORT_NAMES = NO
+
+# If the HIDE_SCOPE_NAMES tag is set to NO (the default) then Doxygen
+# will show members with their full class and namespace scopes in the
+# documentation. If set to YES the scope will be hidden.
+
+HIDE_SCOPE_NAMES = NO
+
+# If the VERBATIM_HEADERS tag is set to YES (the default) then Doxygen
+# will generate a verbatim copy of the header file for each class for
+# which an include is specified. Set to NO to disable this.
+
+VERBATIM_HEADERS = YES
+
+# If the SHOW_INCLUDE_FILES tag is set to YES (the default) then Doxygen
+# will put list of the files that are included by a file in the documentation
+# of that file.
+
+SHOW_INCLUDE_FILES = YES
+
+# If the JAVADOC_AUTOBRIEF tag is set to YES then Doxygen
+# will interpret the first line (until the first dot) of a JavaDoc-style
+# comment as the brief description. If set to NO, the JavaDoc
+# comments will behave just like the Qt-style comments (thus requiring an
+# explict @brief command for a brief description.
+
+JAVADOC_AUTOBRIEF = NO
+
+# If the INHERIT_DOCS tag is set to YES (the default) then an undocumented
+# member inherits the documentation from any documented member that it
+# reimplements.
+
+INHERIT_DOCS = YES
+
+# If the INLINE_INFO tag is set to YES (the default) then a tag [inline]
+# is inserted in the documentation for inline members.
+
+INLINE_INFO = YES
+
+# If the SORT_MEMBER_DOCS tag is set to YES (the default) then doxygen
+# will sort the (detailed) documentation of file and class members
+# alphabetically by member name. If set to NO the members will appear in
+# declaration order.
+
+SORT_MEMBER_DOCS = YES
+
+# If member grouping is used in the documentation and the DISTRIBUTE_GROUP_DOC
+# tag is set to YES, then doxygen will reuse the documentation of the first
+# member in the group (if any) for the other members of the group. By default
+# all members of a group must be documented explicitly.
+
+DISTRIBUTE_GROUP_DOC = NO
+
+# The TAB_SIZE tag can be used to set the number of spaces in a tab.
+# Doxygen uses this value to replace tabs by spaces in code fragments.
+
+TAB_SIZE = 8
+
+# The GENERATE_TODOLIST tag can be used to enable (YES) or
+# disable (NO) the todo list. This list is created by putting \todo
+# commands in the documentation.
+
+GENERATE_TODOLIST = YES
+
+# The GENERATE_TESTLIST tag can be used to enable (YES) or
+# disable (NO) the test list. This list is created by putting \test
+# commands in the documentation.
+
+GENERATE_TESTLIST = YES
+
+# The GENERATE_BUGLIST tag can be used to enable (YES) or
+# disable (NO) the bug list. This list is created by putting \bug
+# commands in the documentation.
+
+GENERATE_BUGLIST = YES
+
+# This tag can be used to specify a number of aliases that acts
+# as commands in the documentation. An alias has the form "name=value".
+# For example adding "sideeffect=\par Side Effects:\n" will allow you to
+# put the command \sideeffect (or @sideeffect) in the documentation, which
+# will result in a user defined paragraph with heading "Side Effects:".
+# You can put \n's in the value part of an alias to insert newlines.
+
+ALIASES =
+
+# The ENABLED_SECTIONS tag can be used to enable conditional
+# documentation sections, marked by \if sectionname ... \endif.
+
+ENABLED_SECTIONS =
+
+# The MAX_INITIALIZER_LINES tag determines the maximum number of lines
+# the initial value of a variable or define consist of for it to appear in
+# the documentation. If the initializer consists of more lines than specified
+# here it will be hidden. Use a value of 0 to hide initializers completely.
+# The appearance of the initializer of individual variables and defines in the
+# documentation can be controlled using \showinitializer or \hideinitializer
+# command in the documentation regardless of this setting.
+
+MAX_INITIALIZER_LINES = 30
+
+# Set the OPTIMIZE_OUTPUT_FOR_C tag to YES if your project consists of C sources
+# only. Doxygen will then generate output that is more tailored for C.
+# For instance some of the names that are used will be different. The list
+# of all members will be omitted, etc.
+
+OPTIMIZE_OUTPUT_FOR_C = NO
+
+# Set the SHOW_USED_FILES tag to NO to disable the list of files generated
+# at the bottom of the documentation of classes and structs. If set to YES the
+# list will mention the files that were used to generate the documentation.
+
+SHOW_USED_FILES = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to warning and progress messages
+#---------------------------------------------------------------------------
+
+# The QUIET tag can be used to turn on/off the messages that are generated
+# by doxygen. Possible values are YES and NO. If left blank NO is used.
+
+QUIET = NO
+
+# The WARNINGS tag can be used to turn on/off the warning messages that are
+# generated by doxygen. Possible values are YES and NO. If left blank
+# NO is used.
+
+WARNINGS = YES
+
+# If WARN_IF_UNDOCUMENTED is set to YES, then doxygen will generate warnings
+# for undocumented members. If EXTRACT_ALL is set to YES then this flag will
+# automatically be disabled.
+
+WARN_IF_UNDOCUMENTED = YES
+
+# The WARN_FORMAT tag determines the format of the warning messages that
+# doxygen can produce. The string should contain the $file, $line, and $text
+# tags, which will be replaced by the file and line number from which the
+# warning originated and the warning text.
+
+WARN_FORMAT = "$file:$line: $text"
+
+# The WARN_LOGFILE tag can be used to specify a file to which warning
+# and error messages should be written. If left blank the output is written
+# to stderr.
+
+WARN_LOGFILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the input files
+#---------------------------------------------------------------------------
+
+# The INPUT tag can be used to specify the files and/or directories that contain
+# documented source files. You may enter file names like "myfile.cpp" or
+# directories like "/usr/src/myproject". Separate the files or directories
+# with spaces.
+
+INPUT = test
+
+# If the value of the INPUT tag contains directories, you can use the
+# FILE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank the following patterns are tested:
+# *.c *.cc *.cxx *.cpp *.c++ *.java *.ii *.ixx *.ipp *.i++ *.inl *.h *.hh *.hxx *.hpp
+# *.h++ *.idl *.odl
+
+FILE_PATTERNS =
+
+# The RECURSIVE tag can be used to turn specify whether or not subdirectories
+# should be searched for input files as well. Possible values are YES and NO.
+# If left blank NO is used.
+
+RECURSIVE = YES
+
+# The EXCLUDE tag can be used to specify files and/or directories that should
+# excluded from the INPUT source files. This way you can easily exclude a
+# subdirectory from a directory tree whose root is specified with the INPUT tag.
+
+EXCLUDE =
+
+# The EXCLUDE_SYMLINKS tag can be used select whether or not files or directories
+# that are symbolic links (a Unix filesystem feature) are excluded from the input.
+
+EXCLUDE_SYMLINKS = NO
+
+# If the value of the INPUT tag contains directories, you can use the
+# EXCLUDE_PATTERNS tag to specify one or more wildcard patterns to exclude
+# certain files from those directories.
+
+EXCLUDE_PATTERNS =
+
+# The EXAMPLE_PATH tag can be used to specify one or more files or
+# directories that contain example code fragments that are included (see
+# the \include command).
+
+EXAMPLE_PATH =
+
+# If the value of the EXAMPLE_PATH tag contains directories, you can use the
+# EXAMPLE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp
+# and *.h) to filter out the source-files in the directories. If left
+# blank all files are included.
+
+EXAMPLE_PATTERNS =
+
+# If the EXAMPLE_RECURSIVE tag is set to YES then subdirectories will be
+# searched for input files to be used with the \include or \dontinclude
+# commands irrespective of the value of the RECURSIVE tag.
+# Possible values are YES and NO. If left blank NO is used.
+
+EXAMPLE_RECURSIVE = NO
+
+# The IMAGE_PATH tag can be used to specify one or more files or
+# directories that contain image that are included in the documentation (see
+# the \image command).
+
+IMAGE_PATH =
+
+# The INPUT_FILTER tag can be used to specify a program that doxygen should
+# invoke to filter for each input file. Doxygen will invoke the filter program
+# by executing (via popen()) the command <filter> <input-file>, where <filter>
+# is the value of the INPUT_FILTER tag, and <input-file> is the name of an
+# input file. Doxygen will then use the output that the filter program writes
+# to standard output.
+
+INPUT_FILTER =
+
+# If the FILTER_SOURCE_FILES tag is set to YES, the input filter (if set using
+# INPUT_FILTER) will be used to filter the input files when producing source
+# files to browse.
+
+FILTER_SOURCE_FILES = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to source browsing
+#---------------------------------------------------------------------------
+
+# If the SOURCE_BROWSER tag is set to YES then a list of source files will
+# be generated. Documented entities will be cross-referenced with these sources.
+
+SOURCE_BROWSER = YES
+
+# Setting the INLINE_SOURCES tag to YES will include the body
+# of functions and classes directly in the documentation.
+
+INLINE_SOURCES = YES
+
+# If the REFERENCED_BY_RELATION tag is set to YES (the default)
+# then for each documented function all documented
+# functions referencing it will be listed.
+
+REFERENCED_BY_RELATION = YES
+
+# If the REFERENCES_RELATION tag is set to YES (the default)
+# then for each documented function all documented entities
+# called/used by that function will be listed.
+
+REFERENCES_RELATION = YES
+
+#---------------------------------------------------------------------------
+# configuration options related to the alphabetical class index
+#---------------------------------------------------------------------------
+
+# If the ALPHABETICAL_INDEX tag is set to YES, an alphabetical index
+# of all compounds will be generated. Enable this if the project
+# contains a lot of classes, structs, unions or interfaces.
+
+ALPHABETICAL_INDEX = YES
+
+# If the alphabetical index is enabled (see ALPHABETICAL_INDEX) then
+# the COLS_IN_ALPHA_INDEX tag can be used to specify the number of columns
+# in which this list will be split (can be a number in the range [1..20])
+
+COLS_IN_ALPHA_INDEX = 5
+
+# In case all classes in a project start with a common prefix, all
+# classes will be put under the same header in the alphabetical index.
+# The IGNORE_PREFIX tag can be used to specify one or more prefixes that
+# should be ignored while generating the index headers.
+
+IGNORE_PREFIX =
+
+#---------------------------------------------------------------------------
+# configuration options related to the HTML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_HTML tag is set to YES (the default) Doxygen will
+# generate HTML output.
+
+GENERATE_HTML = YES
+
+# The HTML_OUTPUT tag is used to specify where the HTML docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `html' will be used as the default path.
+
+HTML_OUTPUT = html
+
+# The HTML_FILE_EXTENSION tag can be used to specify the file extension for
+# each generated HTML page (for example: .htm,.php,.asp). If it is left blank
+# doxygen will generate files with .html extension.
+
+HTML_FILE_EXTENSION = .html
+
+# The HTML_HEADER tag can be used to specify a personal HTML header for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard header.
+
+HTML_HEADER =
+
+# The HTML_FOOTER tag can be used to specify a personal HTML footer for
+# each generated HTML page. If it is left blank doxygen will generate a
+# standard footer.
+
+HTML_FOOTER =
+
+# The HTML_STYLESHEET tag can be used to specify a user defined cascading
+# style sheet that is used by each HTML page. It can be used to
+# fine-tune the look of the HTML output. If the tag is left blank doxygen
+# will generate a default style sheet
+
+HTML_STYLESHEET =
+
+# If the HTML_ALIGN_MEMBERS tag is set to YES, the members of classes,
+# files or namespaces will be aligned in HTML using tables. If set to
+# NO a bullet list will be used.
+
+HTML_ALIGN_MEMBERS = YES
+
+# If the GENERATE_HTMLHELP tag is set to YES, additional index files
+# will be generated that can be used as input for tools like the
+# Microsoft HTML help workshop to generate a compressed HTML help file (.chm)
+# of the generated HTML documentation.
+
+GENERATE_HTMLHELP = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the GENERATE_CHI flag
+# controls if a separate .chi index file is generated (YES) or that
+# it should be included in the master .chm file (NO).
+
+GENERATE_CHI = NO
+
+# If the GENERATE_HTMLHELP tag is set to YES, the BINARY_TOC flag
+# controls whether a binary table of contents is generated (YES) or a
+# normal table of contents (NO) in the .chm file.
+
+BINARY_TOC = NO
+
+# The TOC_EXPAND flag can be set to YES to add extra items for group members
+# to the contents of the Html help documentation and to the tree view.
+
+TOC_EXPAND = NO
+
+# The DISABLE_INDEX tag can be used to turn on/off the condensed index at
+# top of each HTML page. The value NO (the default) enables the index and
+# the value YES disables it.
+
+DISABLE_INDEX = NO
+
+# This tag can be used to set the number of enum values (range [1..20])
+# that doxygen will group on one line in the generated HTML documentation.
+
+ENUM_VALUES_PER_LINE = 4
+
+# If the GENERATE_TREEVIEW tag is set to YES, a side panel will be
+# generated containing a tree-like index structure (just like the one that
+# is generated for HTML Help). For this to work a browser that supports
+# JavaScript and frames is required (for instance Mozilla, Netscape 4.0+,
+# or Internet explorer 4.0+). Note that for large projects the tree generation
+# can take a very long time. In such cases it is better to disable this feature.
+# Windows users are probably better off using the HTML help feature.
+
+GENERATE_TREEVIEW = NO
+
+# If the treeview is enabled (see GENERATE_TREEVIEW) then this tag can be
+# used to set the initial width (in pixels) of the frame in which the tree
+# is shown.
+
+TREEVIEW_WIDTH = 250
+
+#---------------------------------------------------------------------------
+# configuration options related to the LaTeX output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_LATEX tag is set to YES (the default) Doxygen will
+# generate Latex output.
+
+GENERATE_LATEX = NO
+
+# The LATEX_OUTPUT tag is used to specify where the LaTeX docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `latex' will be used as the default path.
+
+LATEX_OUTPUT = latex
+
+# If the COMPACT_LATEX tag is set to YES Doxygen generates more compact
+# LaTeX documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_LATEX = NO
+
+# The PAPER_TYPE tag can be used to set the paper type that is used
+# by the printer. Possible values are: a4, a4wide, letter, legal and
+# executive. If left blank a4wide will be used.
+
+PAPER_TYPE = a4wide
+
+# The EXTRA_PACKAGES tag can be to specify one or more names of LaTeX
+# packages that should be included in the LaTeX output.
+
+EXTRA_PACKAGES =
+
+# The LATEX_HEADER tag can be used to specify a personal LaTeX header for
+# the generated latex document. The header should contain everything until
+# the first chapter. If it is left blank doxygen will generate a
+# standard header. Notice: only use this tag if you know what you are doing!
+
+LATEX_HEADER =
+
+# If the PDF_HYPERLINKS tag is set to YES, the LaTeX that is generated
+# is prepared for conversion to pdf (using ps2pdf). The pdf file will
+# contain links (just like the HTML output) instead of page references
+# This makes the output suitable for online browsing using a pdf viewer.
+
+PDF_HYPERLINKS = NO
+
+# If the USE_PDFLATEX tag is set to YES, pdflatex will be used instead of
+# plain latex in the generated Makefile. Set this option to YES to get a
+# higher quality PDF documentation.
+
+USE_PDFLATEX = NO
+
+# If the LATEX_BATCHMODE tag is set to YES, doxygen will add the \\batchmode.
+# command to the generated LaTeX files. This will instruct LaTeX to keep
+# running if errors occur, instead of asking the user for help.
+# This option is also used when generating formulas in HTML.
+
+LATEX_BATCHMODE = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the RTF output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_RTF tag is set to YES Doxygen will generate RTF output
+# The RTF output is optimised for Word 97 and may not look very pretty with
+# other RTF readers or editors.
+
+GENERATE_RTF = NO
+
+# The RTF_OUTPUT tag is used to specify where the RTF docs will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `rtf' will be used as the default path.
+
+RTF_OUTPUT = rtf
+
+# If the COMPACT_RTF tag is set to YES Doxygen generates more compact
+# RTF documents. This may be useful for small projects and may help to
+# save some trees in general.
+
+COMPACT_RTF = NO
+
+# If the RTF_HYPERLINKS tag is set to YES, the RTF that is generated
+# will contain hyperlink fields. The RTF file will
+# contain links (just like the HTML output) instead of page references.
+# This makes the output suitable for online browsing using WORD or other
+# programs which support those fields.
+# Note: wordpad (write) and others do not support links.
+
+RTF_HYPERLINKS = NO
+
+# Load stylesheet definitions from file. Syntax is similar to doxygen's
+# config file, i.e. a series of assigments. You only have to provide
+# replacements, missing definitions are set to their default value.
+
+RTF_STYLESHEET_FILE =
+
+# Set optional variables used in the generation of an rtf document.
+# Syntax is similar to doxygen's config file.
+
+RTF_EXTENSIONS_FILE =
+
+#---------------------------------------------------------------------------
+# configuration options related to the man page output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_MAN tag is set to YES (the default) Doxygen will
+# generate man pages
+
+GENERATE_MAN = NO
+
+# The MAN_OUTPUT tag is used to specify where the man pages will be put.
+# If a relative path is entered the value of OUTPUT_DIRECTORY will be
+# put in front of it. If left blank `man' will be used as the default path.
+
+MAN_OUTPUT = man
+
+# The MAN_EXTENSION tag determines the extension that is added to
+# the generated man pages (default is the subroutine's section .3)
+
+MAN_EXTENSION = .3
+
+# If the MAN_LINKS tag is set to YES and Doxygen generates man output,
+# then it will generate one additional man file for each entity
+# documented in the real man page(s). These additional files
+# only source the real man page, but without them the man command
+# would be unable to find the correct page. The default is NO.
+
+MAN_LINKS = NO
+
+#---------------------------------------------------------------------------
+# configuration options related to the XML output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_XML tag is set to YES Doxygen will
+# generate an XML file that captures the structure of
+# the code including all documentation. Note that this
+# feature is still experimental and incomplete at the
+# moment.
+
+GENERATE_XML = NO
+
+#---------------------------------------------------------------------------
+# configuration options for the AutoGen Definitions output
+#---------------------------------------------------------------------------
+
+# If the GENERATE_AUTOGEN_DEF tag is set to YES Doxygen will
+# generate an AutoGen Definitions (see autogen.sf.net) file
+# that captures the structure of the code including all
+# documentation. Note that this feature is still experimental
+# and incomplete at the moment.
+
+GENERATE_AUTOGEN_DEF = NO
+
+#---------------------------------------------------------------------------
+# Configuration options related to the preprocessor
+#---------------------------------------------------------------------------
+
+# If the ENABLE_PREPROCESSING tag is set to YES (the default) Doxygen will
+# evaluate all C-preprocessor directives found in the sources and include
+# files.
+
+ENABLE_PREPROCESSING = YES
+
+# If the MACRO_EXPANSION tag is set to YES Doxygen will expand all macro
+# names in the source code. If set to NO (the default) only conditional
+# compilation will be performed. Macro expansion can be done in a controlled
+# way by setting EXPAND_ONLY_PREDEF to YES.
+
+MACRO_EXPANSION = NO
+
+# If the EXPAND_ONLY_PREDEF and MACRO_EXPANSION tags are both set to YES
+# then the macro expansion is limited to the macros specified with the
+# PREDEFINED and EXPAND_AS_PREDEFINED tags.
+
+EXPAND_ONLY_PREDEF = NO
+
+# If the SEARCH_INCLUDES tag is set to YES (the default) the includes files
+# in the INCLUDE_PATH (see below) will be search if a #include is found.
+
+SEARCH_INCLUDES = YES
+
+# The INCLUDE_PATH tag can be used to specify one or more directories that
+# contain include files that are not input files but should be processed by
+# the preprocessor.
+
+INCLUDE_PATH =
+
+# You can use the INCLUDE_FILE_PATTERNS tag to specify one or more wildcard
+# patterns (like *.h and *.hpp) to filter out the header-files in the
+# directories. If left blank, the patterns specified with FILE_PATTERNS will
+# be used.
+
+INCLUDE_FILE_PATTERNS =
+
+# The PREDEFINED tag can be used to specify one or more macro names that
+# are defined before the preprocessor is started (similar to the -D option of
+# gcc). The argument of the tag is a list of macros of the form: name
+# or name=definition (no spaces). If the definition and the = are
+# omitted =1 is assumed.
+
+PREDEFINED =
+
+# If the MACRO_EXPANSION and EXPAND_PREDEF_ONLY tags are set to YES then
+# this tag can be used to specify a list of macro names that should be expanded.
+# The macro definition that is found in the sources will be used.
+# Use the PREDEFINED tag if you want to use a different macro definition.
+
+EXPAND_AS_DEFINED =
+
+# If the SKIP_FUNCTION_MACROS tag is set to YES (the default) then
+# doxygen's preprocessor will remove all function-like macros that are alone
+# on a line and do not end with a semicolon. Such function macros are typically
+# used for boiler-plate code, and will confuse the parser if not removed.
+
+SKIP_FUNCTION_MACROS = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to external references
+#---------------------------------------------------------------------------
+
+# The TAGFILES tag can be used to specify one or more tagfiles.
+
+TAGFILES =
+
+# When a file name is specified after GENERATE_TAGFILE, doxygen will create
+# a tag file that is based on the input files it reads.
+
+GENERATE_TAGFILE =
+
+# If the ALLEXTERNALS tag is set to YES all external classes will be listed
+# in the class index. If set to NO only the inherited external classes
+# will be listed.
+
+ALLEXTERNALS = NO
+
+# If the EXTERNAL_GROUPS tag is set to YES all external groups will be listed
+# in the modules index. If set to NO, only the current project's groups will
+# be listed.
+
+EXTERNAL_GROUPS = YES
+
+# The PERL_PATH should be the absolute path and name of the perl script
+# interpreter (i.e. the result of `which perl').
+
+PERL_PATH = /usr/bin/perl
+
+#---------------------------------------------------------------------------
+# Configuration options related to the dot tool
+#---------------------------------------------------------------------------
+
+# If the CLASS_DIAGRAMS tag is set to YES (the default) Doxygen will
+# generate a inheritance diagram (in Html, RTF and LaTeX) for classes with base or
+# super classes. Setting the tag to NO turns the diagrams off. Note that this
+# option is superceded by the HAVE_DOT option below. This is only a fallback. It is
+# recommended to install and use dot, since it yield more powerful graphs.
+
+CLASS_DIAGRAMS = YES
+
+# If you set the HAVE_DOT tag to YES then doxygen will assume the dot tool is
+# available from the path. This tool is part of Graphviz, a graph visualization
+# toolkit from AT&T and Lucent Bell Labs. The other options in this section
+# have no effect if this option is set to NO (the default)
+
+HAVE_DOT = YES
+
+# If the CLASS_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect inheritance relations. Setting this tag to YES will force the
+# the CLASS_DIAGRAMS tag to NO.
+
+CLASS_GRAPH = YES
+
+# If the COLLABORATION_GRAPH and HAVE_DOT tags are set to YES then doxygen
+# will generate a graph for each documented class showing the direct and
+# indirect implementation dependencies (inheritance, containment, and
+# class references variables) of the class with other documented classes.
+
+COLLABORATION_GRAPH = YES
+
+# If set to YES, the inheritance and collaboration graphs will show the
+# relations between templates and their instances.
+
+TEMPLATE_RELATIONS = YES
+
+# If set to YES, the inheritance and collaboration graphs will hide
+# inheritance and usage relations if the target is undocumented
+# or is not a class.
+
+HIDE_UNDOC_RELATIONS = NO
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDE_GRAPH, and HAVE_DOT
+# tags are set to YES then doxygen will generate a graph for each documented
+# file showing the direct and indirect include dependencies of the file with
+# other documented files.
+
+INCLUDE_GRAPH = YES
+
+# If the ENABLE_PREPROCESSING, SEARCH_INCLUDES, INCLUDED_BY_GRAPH, and
+# HAVE_DOT tags are set to YES then doxygen will generate a graph for each
+# documented header file showing the documented files that directly or
+# indirectly include this file.
+
+INCLUDED_BY_GRAPH = YES
+
+# If the GRAPHICAL_HIERARCHY and HAVE_DOT tags are set to YES then doxygen
+# will graphical hierarchy of all classes instead of a textual one.
+
+GRAPHICAL_HIERARCHY = YES
+
+# The DOT_IMAGE_FORMAT tag can be used to set the image format of the images
+# generated by dot. Possible values are gif, jpg, and png
+# If left blank gif will be used.
+
+DOT_IMAGE_FORMAT = gif
+
+# The tag DOT_PATH can be used to specify the path where the dot tool can be
+# found. If left blank, it is assumed the dot tool can be found on the path.
+
+DOT_PATH =
+
+# The DOTFILE_DIRS tag can be used to specify one or more directories that
+# contain dot files that are included in the documentation (see the
+# \dotfile command).
+
+DOTFILE_DIRS =
+
+# The MAX_DOT_GRAPH_WIDTH tag can be used to set the maximum allowed width
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_WIDTH = 1024
+
+# The MAX_DOT_GRAPH_HEIGHT tag can be used to set the maximum allows height
+# (in pixels) of the graphs generated by dot. If a graph becomes larger than
+# this value, doxygen will try to truncate the graph, so that it fits within
+# the specified constraint. Beware that most browsers cannot cope with very
+# large images.
+
+MAX_DOT_GRAPH_HEIGHT = 1024
+
+# If the GENERATE_LEGEND tag is set to YES (the default) Doxygen will
+# generate a legend page explaining the meaning of the various boxes and
+# arrows in the dot generated graphs.
+
+GENERATE_LEGEND = YES
+
+# If the DOT_CLEANUP tag is set to YES (the default) Doxygen will
+# remove the intermedate dot files that are used to generate
+# the various graphs.
+
+DOT_CLEANUP = YES
+
+#---------------------------------------------------------------------------
+# Configuration::addtions related to the search engine
+#---------------------------------------------------------------------------
+
+# The SEARCHENGINE tag specifies whether or not a search engine should be
+# used. If set to NO the values of all tags below this one will be ignored.
+
+SEARCHENGINE = NO
+
+# The CGI_NAME tag should be the name of the CGI script that
+# starts the search engine (doxysearch) with the correct parameters.
+# A script with this name will be generated by doxygen.
+
+CGI_NAME = search.cgi
+
+# The CGI_URL tag should be the absolute URL to the directory where the
+# cgi binaries are located. See the documentation of your http daemon for
+# details.
+
+CGI_URL =
+
+# The DOC_URL tag should be the absolute URL to the directory where the
+# documentation is located. If left blank the absolute path to the
+# documentation, with file:// prepended to it, will be used.
+
+DOC_URL =
+
+# The DOC_ABSPATH tag should be the absolute path to the directory where the
+# documentation is located. If left blank the directory on the local machine
+# will be used.
+
+DOC_ABSPATH =
+
+# The BIN_ABSPATH tag must point to the directory where the doxysearch binary
+# is installed.
+
+BIN_ABSPATH = /usr/local/bin
+
+# The EXT_DOC_PATHS tag can be used to specify one or more paths to
+# documentation generated for other projects. This allows doxysearch to search
+# the documentation for these projects as well.
+
+EXT_DOC_PATHS =
diff --git a/ndb/docs/doxygen/header.mgmapi.tex b/ndb/docs/doxygen/header.mgmapi.tex
new file mode 100644
index 00000000000..1b55ceb15c7
--- /dev/null
+++ b/ndb/docs/doxygen/header.mgmapi.tex
@@ -0,0 +1,44 @@
+\documentclass[a4paper]{book}
+\usepackage{a4wide}
+\usepackage{makeidx}
+\usepackage{fancyhdr}
+\usepackage{graphicx}
+\usepackage{multicol}
+\usepackage{float}
+\usepackage{textcomp}
+\usepackage{alltt}
+\usepackage{times}
+\ifx\pdfoutput\undefined
+\usepackage[ps2pdf,
+ pagebackref=true,
+ colorlinks=true,
+ linkcolor=blue
+ ]{hyperref}
+\usepackage{pspicture}
+\else
+\usepackage[pdftex,
+ pagebackref=true,
+ colorlinks=true,
+ linkcolor=blue
+ ]{hyperref}
+\fi
+\usepackage{doxygen}
+\makeindex
+\setcounter{tocdepth}{1}
+\renewcommand{\footrulewidth}{0.4pt}
+\begin{document}
+\begin{titlepage}
+\vspace*{7cm}
+\begin{center}
+{\Huge NDB Cluster MGM API Guide \mbox{}\vspace{-3cm}\mbox{}\hrule\bigskip\bigskip\bigskip\bigskip\mbox{}\Huge{}}\\\vspace*{1cm}
+\begin{center}\LARGE{MySQL AB}\end{center}\hfill\bigskip\bigskip\bigskip\hrule\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip NDB Cluster Release RELEASE
+\bigskip\bigskip\bigskip\bigskip\bigskip\hfill\vspace*{0.5cm}
+{\small DATE}\\
+\end{center}
+\end{titlepage}
+\clearemptydoublepage
+\pagenumbering{roman}
+\tableofcontents
+\clearemptydoublepage
+\pagenumbering{arabic}
+
diff --git a/ndb/docs/doxygen/header.ndbapi.tex b/ndb/docs/doxygen/header.ndbapi.tex
new file mode 100644
index 00000000000..c37ce286ed8
--- /dev/null
+++ b/ndb/docs/doxygen/header.ndbapi.tex
@@ -0,0 +1,44 @@
+\documentclass[a4paper]{book}
+\usepackage{a4wide}
+\usepackage{makeidx}
+\usepackage{fancyhdr}
+\usepackage{graphicx}
+\usepackage{multicol}
+\usepackage{float}
+\usepackage{textcomp}
+\usepackage{alltt}
+\usepackage{times}
+\ifx\pdfoutput\undefined
+\usepackage[ps2pdf,
+ pagebackref=true,
+ colorlinks=true,
+ linkcolor=blue
+ ]{hyperref}
+\usepackage{pspicture}
+\else
+\usepackage[pdftex,
+ pagebackref=true,
+ colorlinks=true,
+ linkcolor=blue
+ ]{hyperref}
+\fi
+\usepackage{doxygen}
+\makeindex
+\setcounter{tocdepth}{1}
+\renewcommand{\footrulewidth}{0.4pt}
+\begin{document}
+\begin{titlepage}
+\vspace*{7cm}
+\begin{center}
+{\Huge NDB API Programmer's Guide \mbox{}\vspace{-3cm}\mbox{}\hrule\bigskip\bigskip\bigskip\bigskip\mbox{}\Huge{}}\\\vspace*{1cm}
+\begin{center}\LARGE{MySQL AB}\end{center}\hfill\bigskip\bigskip\bigskip\hrule\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip\bigskip NDB Cluster Release RELEASE
+\bigskip\bigskip\bigskip\bigskip\bigskip\hfill\vspace*{0.5cm}
+{\small DATE}\\
+\end{center}
+\end{titlepage}
+\clearemptydoublepage
+\pagenumbering{roman}
+\tableofcontents
+\clearemptydoublepage
+\pagenumbering{arabic}
+
diff --git a/ndb/docs/doxygen/postdoxy.pl b/ndb/docs/doxygen/postdoxy.pl
new file mode 100755
index 00000000000..95062d1899f
--- /dev/null
+++ b/ndb/docs/doxygen/postdoxy.pl
@@ -0,0 +1,97 @@
+#!/usr/local/bin/perl
+#
+# Written by Lars Thalmann, lars@mysql.com, 2003.
+#
+
+use strict;
+umask 000;
+
+# -----------------------------------------------------------------------------
+# Settings
+# -----------------------------------------------------------------------------
+
+$ENV{LD_LIBRARY_PATH} = "/usr/local/lib:/opt/as/local/lib";
+$ENV{LD_LIBRARY_PATH} = $ENV{LD_LIBRARY_PATH} . ":/opt/as/forte6/SUNWspro/lib";
+$ENV{PATH} = $ENV{PATH} . ":/usr/local/bin:/opt/as/local/bin";
+$ENV{PATH} = $ENV{PATH} . ":/opt/as/local/teTeX/bin/sparc-sun-solaris2.8";
+
+my $destdir = @ARGV[0];
+my $title = ""; # $ARGV[1];
+
+my $release;
+if (defined $ENV{'NDB_RELEASE'}) {
+ $release = $ENV{'NDB_RELEASE'};
+ print "----------------------------------------------------------------\n";
+ print "Relase = " . $release . "\n";
+ print "----------------------------------------------------------------\n";
+} else {
+ print "----------------------------------------------------------------\n";
+ print "NDB Documentation is being modified to statndard format\n";
+ print "(If you want this automatic, use env variable NDB_RELEASE.)\n";
+ print "Enter release (Examples: \"1.43.0 (alpha)\" or \"2.1.0 (gamma)\"): ";
+ $release = <stdin>;
+ print "----------------------------------------------------------------\n";
+}
+
+# -----------------------------------------------------------------------------
+# Change a little in refman.tex
+# -----------------------------------------------------------------------------
+
+open (INFILE, "< ${destdir}/refman.tex")
+ or die "Error opening ${destdir}/refman.tex.\n";
+open (OUTFILE, "> ${destdir}/refman.tex.new")
+ or die "Error opening ${destdir}/refman.tex.new.\n";
+
+while (<INFILE>)
+{
+ if (/(.*)(RELEASE)(.*)$/) {
+ print OUTFILE $1 . $release . $3;
+ } elsif (/(.*)(DATE)(.*)$/) {
+ print OUTFILE $1 . localtime() . $3;
+ } elsif (/\\chapter\{File Index\}/) {
+ # Erase
+ } elsif (/\\input\{files\}/) {
+ # Erase
+ } elsif (/\\chapter\{Hierarchical Index\}/) {
+ # Erase
+ } elsif (/\\input\{hierarchy\}/) {
+ # Erase
+ } elsif (/\\chapter\{Page Index\}/) {
+ # Erase
+ } elsif (/\\input\{pages\}/) {
+ # Erase
+ } else {
+ print OUTFILE;
+ }
+}
+
+close INFILE;
+close OUTFILE;
+
+system("mv ${destdir}/refman.tex.new ${destdir}/refman.tex");
+
+# -----------------------------------------------------------------------------
+# Change a little in doxygen.sty
+# -----------------------------------------------------------------------------
+
+open (INFILE, "< ${destdir}/doxygen.sty")
+ or die "Error opening INFILE.\n";
+open (OUTFILE, "> ${destdir}/doxygen.sty.new")
+ or die "Error opening OUTFILE.\n";
+
+while (<INFILE>)
+{
+ if (/\\rfoot/) {
+ print OUTFILE "\\rfoot[\\fancyplain{}{\\bfseries\\small \\copyright~Copyright 2003-2004 MySQL AB\\hfill support-cluster\@mysql.com}]{}\n";
+ } elsif (/\\lfoot/) {
+ print OUTFILE "\\lfoot[]{\\fancyplain{}{\\bfseries\\small support-cluster\@mysql.com\\hfill \\copyright~Copyright 2003-2004 MySQL AB}}\n";
+ } else {
+ print OUTFILE;
+ }
+}
+
+close INFILE;
+close OUTFILE;
+
+system("mv ${destdir}/doxygen.sty.new ${destdir}/doxygen.sty");
+
diff --git a/ndb/docs/doxygen/predoxy.pl b/ndb/docs/doxygen/predoxy.pl
new file mode 100755
index 00000000000..461ad02478a
--- /dev/null
+++ b/ndb/docs/doxygen/predoxy.pl
@@ -0,0 +1,34 @@
+#!/usr/local/bin/perl
+#
+# Written by Lars Thalmann, lars@mysql.com, 2003.
+#
+
+use strict;
+umask 000;
+
+# -----------------------------------------------------------------------------
+# Fix HTML Footer
+# -----------------------------------------------------------------------------
+
+open (OUTFILE, "> footer.html");
+
+print OUTFILE<<EOT;
+<hr>
+<address>
+<small>
+<center>
+EOT
+print OUTFILE "Documentation generated " . localtime() .
+ " from NDB Cluster source files.";
+print OUTFILE<<EOT;
+<br>
+&copy; 2003-2004
+<a href="http://www.mysql.com">MySQL AB</a>
+<br>
+</center>
+</small></address>
+</body>
+</html>
+EOT
+
+print "Preformat finished\n\n";
diff --git a/ndb/env.sh b/ndb/env.sh
new file mode 100644
index 00000000000..c84a61b2c6f
--- /dev/null
+++ b/ndb/env.sh
@@ -0,0 +1,8 @@
+#
+
+NDB_TOP=`pwd`
+export NDB_TOP
+
+NDB_PROJ_HOME=$NDB_TOP/home
+export NDB_PROJ_HOME
+
diff --git a/ndb/examples/Makefile b/ndb/examples/Makefile
new file mode 100644
index 00000000000..e1fb71a1817
--- /dev/null
+++ b/ndb/examples/Makefile
@@ -0,0 +1,26 @@
+-include .defs.mk
+
+#ifneq ($(C++),)
+#OPTS = CC=$(CC) CXX=$(C++)
+#endif
+
+# XXX ndbapi_example4 commented out until fixed
+BIN_DIRS := ndbapi_example1 ndbapi_example2 ndbapi_example3 $(ndbapi_example4) \
+ ndbapi_example5 select_all
+
+bins: $(patsubst %, _bins_%, $(BIN_DIRS))
+
+$(patsubst %, _bins_%, $(BIN_DIRS)) :
+ $(MAKE) -C $(patsubst _bins_%, %, $@) $(OPTS)
+
+libs:
+
+clean:
+ for f in ${BIN_DIRS}; do \
+ $(MAKE) -C $$f $@;\
+ done
+
+cleanall: clean
+tidy: clean
+distclean: clean
+
diff --git a/ndb/examples/configurations/demos.tar b/ndb/examples/configurations/demos.tar
new file mode 100644
index 00000000000..d8cae90ec5b
--- /dev/null
+++ b/ndb/examples/configurations/demos.tar
Binary files differ
diff --git a/ndb/examples/ndbapi_async_example/Makefile b/ndb/examples/ndbapi_async_example/Makefile
new file mode 100644
index 00000000000..7910a4a1d12
--- /dev/null
+++ b/ndb/examples/ndbapi_async_example/Makefile
@@ -0,0 +1,34 @@
+-include ../../Defs.mk
+#NDB_OS = OS_YOU_ARE_RUNNING_ON
+#NDB_OS = LINUX
+#You need to set the NDB_OS variable here (LINUX, SOLARIS, MACOSX)
+TARGET = ndbapi_async
+SRCS = ndbapi_async.cpp
+OBJS = ndbapi_async.o
+CC = g++
+CFLAGS = -c -Wall -fno-rtti -D$(NDB_OS)
+DEBUG =
+LFLAGS = -Wall
+INCLUDE_DIR = ../../include
+LIB_DIR = ../../lib
+ifeq ($(NDB_OS), SOLARIS)
+# Here is the definition of system libraries necessary for Solaris 7
+SYS_LIB = -lpthread -lsocket -lnsl -lrt
+endif
+ifeq ($(NDB_OS), LINUX)
+# Here is the definition of system libraries necessary for Linux 2.4
+SYS_LIB = -lpthread
+endif
+ifeq ($(NDB_OS), MACOSX)
+# Here is the definition of system libraries necessary for Mac OS X
+SYS_LIB =
+endif
+
+$(TARGET): $(OBJS)
+ $(CC) $(LFLAGS) -L$(LIB_DIR) -lNDB_API $(OBJS) $(SYS_LIB) -o $(TARGET)
+
+$(TARGET).o: $(SRCS)
+ $(CC) $(CFLAGS) -I$(INCLUDE_DIR) -I$(INCLUDE_DIR)/ndbapi $(SRCS)
+
+clean:
+ rm -f *.o $(TARGET)
diff --git a/ndb/examples/ndbapi_async_example/ndbapi_async.cpp b/ndb/examples/ndbapi_async_example/ndbapi_async.cpp
new file mode 100644
index 00000000000..078ac0c5cbf
--- /dev/null
+++ b/ndb/examples/ndbapi_async_example/ndbapi_async.cpp
@@ -0,0 +1,505 @@
+
+
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/**
+ * ndbapi_async.cpp:
+ * Illustrates how to use callbacks and error handling using the asynchronous
+ * part of the NDBAPI.
+ *
+ * Classes and methods in NDBAPI used in this example:
+ *
+ * Ndb
+ * init()
+ * waitUntilRead()
+ * getDictionary()
+ * startTransaction()
+ * closeTransaction()
+ * sendPollNdb()
+ * getNdbError()
+ *
+ * NdbConnection
+ * getNdbOperation()
+ * executeAsynchPrepare()
+ * getNdbError()
+ *
+ * NdbDictionary::Dictionary
+ * getTable()
+ * dropTable()
+ * createTable()
+ * getNdbError()
+ *
+ * NdbDictionary::Column
+ * setName()
+ * setPrimaryKey()
+ * setType()
+ * setLength()
+ * setNullable()
+ *
+ * NdbDictionary::Table
+ * setName()
+ * addColumn()
+ *
+ * NdbOperation
+ * insertTuple()
+ * equal()
+ * setValue()
+ *
+ */
+
+
+#include <NdbApi.hpp>
+#include <NdbScanFilter.hpp>
+#include <iostream> // Used for cout
+
+#ifdef SOLARIS
+#include <sys/types.h>
+#include <unistd.h>
+#endif
+
+#if defined LINUX || defined MACOSX
+#include <time.h>
+#include <unistd.h>
+#endif
+
+/**
+ * Helper sleep function
+ */
+int
+milliSleep(int milliseconds){
+ int result = 0;
+ struct timespec sleeptime;
+ sleeptime.tv_sec = milliseconds / 1000;
+ sleeptime.tv_nsec = (milliseconds - (sleeptime.tv_sec * 1000)) * 1000000;
+ result = nanosleep(&sleeptime, NULL);
+ return result;
+}
+
+/**
+ * error printout macro
+ */
+#define APIERROR(error) \
+ { std::cout << "Error in " << __FILE__ << ", line:" << __LINE__ << ", code:" \
+ << error.code << ", msg: " << error.message << "." << std::endl; \
+ exit(-1); }
+
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+/**
+ * callback struct.
+ * transaction : index of the transaction in transaction[] array below
+ * data : the data that the transaction was modifying.
+ * retries : counter for how many times the trans. has been retried
+ */
+typedef struct {
+ Ndb * ndb;
+ int transaction;
+ int data;
+ int retries;
+} async_callback_t;
+
+/**
+ * Structure used in "free list" to a NdbConnection
+ */
+typedef struct {
+ NdbConnection* conn;
+ int used;
+} transaction_t;
+
+/**
+ * Free list holding transactions
+ */
+transaction_t transaction[1024]; //1024 - max number of outstanding
+ //transaction in one Ndb object
+
+#endif
+/**
+ * prototypes
+ */
+
+/**
+ * Prepare and send transaction
+ */
+int populate(Ndb * myNdb, int data, async_callback_t * cbData);
+
+/**
+ * Error handler.
+ */
+bool asynchErrorHandler(NdbConnection * trans, Ndb* ndb);
+
+/**
+ * Exit function
+ */
+void asynchExitHandler(Ndb * m_ndb) ;
+
+/**
+ * Helper function used in callback(...)
+ */
+void closeTransaction(Ndb * ndb , async_callback_t * cb);
+
+/**
+ * Function to create table
+ */
+int create_table(Ndb * myNdb);
+
+/**
+ * stat. variables
+ */
+int tempErrors = 0;
+int permErrors = 0;
+
+/**
+ * Helper function for callback(...)
+ */
+void
+closeTransaction(Ndb * ndb , async_callback_t * cb)
+{
+ ndb->closeTransaction(transaction[cb->transaction].conn);
+ transaction[cb->transaction].conn = 0;
+ transaction[cb->transaction].used = 0;
+ cb->retries++;
+}
+
+/**
+ * Callback executed when transaction has return from NDB
+ */
+static void
+callback(int result, NdbConnection* trans, void* aObject)
+{
+ async_callback_t * cbData = (async_callback_t *)aObject;
+ if (result<0)
+ {
+ /**
+ * Error: Temporary or permanent?
+ */
+ if (asynchErrorHandler(trans, (Ndb*)cbData->ndb))
+ {
+ closeTransaction((Ndb*)cbData->ndb, cbData);
+ while(populate((Ndb*)cbData->ndb, cbData->data, cbData) < 0)
+ milliSleep(10);
+ }
+ else
+ {
+ std::cout << "Restore: Failed to restore data "
+ << "due to a unrecoverable error. Exiting..." << std::endl;
+ delete cbData;
+ asynchExitHandler((Ndb*)cbData->ndb);
+ }
+ }
+ else
+ {
+ /**
+ * OK! close transaction
+ */
+ closeTransaction((Ndb*)cbData->ndb, cbData);
+ delete cbData;
+ }
+}
+
+
+/**
+ * Create table "GARAGE"
+ */
+int create_table(Ndb * myNdb)
+{
+ NdbDictionary::Table myTable;
+ NdbDictionary::Column myColumn;
+
+ NdbDictionary::Dictionary* myDict = myNdb->getDictionary();
+
+ /*********************************************************
+ * Create a table named GARAGE if it does not exist *
+ *********************************************************/
+ if (myDict->getTable("GARAGE") != NULL)
+ {
+ std::cout << "NDB already has example table: GARAGE. "
+ << "Dropping it..." << std::endl;
+ if(myDict->dropTable("GARAGE") == -1)
+ {
+ std::cout << "Failed to drop: GARAGE." << std::endl;
+ exit(1);
+ }
+ }
+
+ myTable.setName("GARAGE");
+
+/**
+ * Column REG_NO
+ */
+ myColumn.setName("REG_NO");
+ myColumn.setPrimaryKey(true);
+ myColumn.setType(NdbDictionary::Column::Unsigned);
+ myColumn.setLength(1);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+/**
+ * Column BRAND
+ */
+ myColumn.setName("BRAND");
+ myColumn.setPrimaryKey(false);
+ myColumn.setType(NdbDictionary::Column::Char);
+ myColumn.setLength(20);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+/**
+ * Column COLOR
+ */
+ myColumn.setName("COLOR");
+ myColumn.setPrimaryKey(false);
+ myColumn.setType(NdbDictionary::Column::Char);
+ myColumn.setLength(20);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+ if (myDict->createTable(myTable) == -1) {
+ APIERROR(myDict->getNdbError());
+ }
+ return 1;
+}
+
+void asynchExitHandler(Ndb * m_ndb)
+{
+ if (m_ndb != NULL)
+ delete m_ndb;
+ exit(-1);
+}
+
+/* returns true if is recoverable (temporary),
+ * false if it is an error that is permanent.
+ */
+bool asynchErrorHandler(NdbConnection * trans, Ndb* ndb)
+{
+ NdbError error = trans->getNdbError();
+ switch(error.status)
+ {
+ case NdbError::Success:
+ return false;
+ break;
+
+ case NdbError::TemporaryError:
+ /**
+ * The error code indicates a temporary error.
+ * The application should typically retry.
+ * (Includes classifications: NdbError::InsufficientSpace,
+ * NdbError::TemporaryResourceError, NdbError::NodeRecoveryError,
+ * NdbError::OverloadError, NdbError::NodeShutdown
+ * and NdbError::TimeoutExpired.)
+ *
+ * We should sleep for a while and retry, except for insufficient space
+ */
+ if(error.classification == NdbError::InsufficientSpace)
+ return false;
+ milliSleep(10);
+ tempErrors++;
+ return true;
+ break;
+ case NdbError::UnknownResult:
+ std::cout << error.message << std::endl;
+ return false;
+ break;
+ default:
+ case NdbError::PermanentError:
+ switch (error.code)
+ {
+ case 499:
+ case 250:
+ milliSleep(10);
+ return true; // SCAN errors that can be retried. Requires restart of scan.
+ default:
+ break;
+ }
+ //ERROR
+ std::cout << error.message << std::endl;
+ return false;
+ break;
+ }
+ return false;
+}
+
+static int nPreparedTransactions = 0;
+static int MAX_RETRIES = 10;
+static int parallelism = 100;
+
+
+/************************************************************************
+ * populate()
+ * 1. Prepare 'parallelism' number of insert transactions.
+ * 2. Send transactions to NDB and wait for callbacks to execute
+ */
+int populate(Ndb * myNdb, int data, async_callback_t * cbData)
+{
+
+ NdbOperation* myNdbOperation; // For operations
+
+ async_callback_t * cb;
+ int retries;
+ int current = 0;
+ for(int i=0; i<1024; i++)
+ {
+ if(transaction[i].used == 0)
+ {
+ current = i;
+ if (cbData == 0)
+ {
+ /**
+ * We already have a callback
+ * This is an absolutely new transaction
+ */
+ cb = new async_callback_t;
+ cb->retries = 0;
+ }
+ else
+ {
+ /**
+ * We already have a callback
+ */
+ cb =cbData;
+ retries = cbData->retries;
+ }
+ /**
+ * Set data used by the callback
+ */
+ cb->ndb = myNdb; //handle to Ndb object so that we can close transaction
+ // in the callback (alt. make myNdb global).
+
+ cb->data = data; //this is the data we want to insert
+ cb->transaction = current; //This is the number (id) of this transaction
+ transaction[current].used = 1 ; //Mark the transaction as used
+ break;
+ }
+ }
+ if(!current)
+ return -1;
+
+ while(retries < MAX_RETRIES)
+ {
+ transaction[current].conn = myNdb->startTransaction();
+ if (transaction[current].conn == NULL) {
+ if (asynchErrorHandler(transaction[current].conn, myNdb))
+ {
+ /**
+ * no transaction to close since conn == null
+ */
+ milliSleep(10);
+ retries++;
+ continue;
+ }
+ asynchExitHandler(myNdb);
+ }
+ // Error check. If error, then maybe table GARAGE is not in database
+ myNdbOperation = transaction[current].conn->getNdbOperation("GARAGE");
+ if (myNdbOperation == NULL)
+ {
+ if (asynchErrorHandler(transaction[current].conn, myNdb))
+ {
+ myNdb->closeTransaction(transaction[current].conn);
+ transaction[current].conn = 0;
+ milliSleep(10);
+ retries++;
+ continue;
+ }
+ asynchExitHandler(myNdb);
+ } // if
+ if(myNdbOperation->insertTuple() < 0 ||
+ myNdbOperation->equal("REG_NO", data) < 0 ||
+ myNdbOperation->setValue("BRAND", "Mercedes") <0 ||
+ myNdbOperation->setValue("COLOR", "Blue") < 0)
+ {
+ if (asynchErrorHandler(transaction[current].conn, myNdb))
+ {
+ myNdb->closeTransaction(transaction[current].conn);
+ transaction[current].conn = 0;
+ retries++;
+ milliSleep(10);
+ continue;
+ }
+ asynchExitHandler(myNdb);
+ }
+
+ /*Prepare transaction (the transaction is NOT yet sent to NDB)*/
+ transaction[current].conn->executeAsynchPrepare(Commit,
+ &callback,
+ cb);
+ /**
+ * When we have prepared parallelism number of transactions ->
+ * send the transaction to ndb.
+ * Next time we will deal with the transactions are in the
+ * callback. There we will see which ones that were successful
+ * and which ones to retry.
+ */
+ if (nPreparedTransactions == parallelism-1)
+ {
+ // send-poll all transactions
+ // close transaction is done in callback
+ myNdb->sendPollNdb(3000, parallelism );
+ nPreparedTransactions=0;
+ }
+ else
+ nPreparedTransactions++;
+ return 1;
+ }
+ std::cout << "Unable to recover from errors. Exiting..." << std::endl;
+ asynchExitHandler(myNdb);
+ return -1;
+}
+
+int main()
+{
+ Ndb* myNdb = new Ndb( "TEST_DB" ); // Object representing the database
+
+ /*******************************************
+ * Initialize NDB and wait until its ready *
+ *******************************************/
+ if (myNdb->init(1024) == -1) { // Set max 1024 parallel transactions
+ APIERROR(myNdb->getNdbError());
+ }
+
+ if (myNdb->waitUntilReady(30) != 0) {
+ std::cout << "NDB was not ready within 30 secs." << std::endl;
+ exit(-1);
+ }
+ create_table(myNdb);
+
+
+ /**
+ * Initialise transaction array
+ */
+ for(int i = 0 ; i < 1024 ; i++)
+ {
+ transaction[i].used = 0;
+ transaction[i].conn = 0;
+
+ }
+ int i=0;
+ /**
+ * Do 20000 insert transactions.
+ */
+ while(i < 20000)
+ {
+ while(populate(myNdb,i,0)<0) // <0, no space on free list. Sleep and try again.
+ milliSleep(10);
+
+ i++;
+ }
+ std::cout << "Number of temporary errors: " << tempErrors << std::endl;
+ delete myNdb;
+}
+
+
diff --git a/ndb/examples/ndbapi_async_example/readme.txt b/ndb/examples/ndbapi_async_example/readme.txt
new file mode 100644
index 00000000000..47cb4bf9ffa
--- /dev/null
+++ b/ndb/examples/ndbapi_async_example/readme.txt
@@ -0,0 +1,3 @@
+1. Set NDB_OS in Makefile
+2. Add path to libNDB_API.so in LD_LIBRARY_PATH
+3. Set NDB_CONNECTSTRING
diff --git a/ndb/examples/ndbapi_example1/Makefile b/ndb/examples/ndbapi_example1/Makefile
new file mode 100644
index 00000000000..eb0142ce673
--- /dev/null
+++ b/ndb/examples/ndbapi_example1/Makefile
@@ -0,0 +1,33 @@
+-include .defs.mk
+#NDB_OS = OS_YOU_ARE_RUNNING_ON
+#You need to set the NDB_OS variable here
+TARGET = ndbapi_example1
+SRCS = ndbapi_example1.cpp
+OBJS = ndbapi_example1.o
+CXX = g++
+CFLAGS = -c -Wall -fno-rtti -fno-exceptions
+DEBUG =
+LFLAGS = -Wall
+INCLUDE_DIR = ../../include
+LIB_DIR = ../../lib
+ifeq ($(NDB_OS), SOLARIS)
+# Here is the definition of system libraries necessary for Solaris 7
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), LINUX)
+# Here is the definition of system libraries necessary for Linux 2.4
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), MACOSX)
+# Here is the definition of system libraries necessary for Mac OS X
+SYS_LIB =
+endif
+
+$(TARGET): $(OBJS)
+ $(CXX) $(LFLAGS) -L$(LIB_DIR) $(OBJS) -lNDB_API $(SYS_LIB) -o $(TARGET)
+
+$(TARGET).o: $(SRCS)
+ $(CXX) $(CFLAGS) -I$(INCLUDE_DIR) -I$(INCLUDE_DIR)/ndbapi $(SRCS)
+
+clean:
+ rm -f *.o $(TARGET)
diff --git a/ndb/examples/ndbapi_example1/ndbapi_example1.cpp b/ndb/examples/ndbapi_example1/ndbapi_example1.cpp
new file mode 100644
index 00000000000..879d86de824
--- /dev/null
+++ b/ndb/examples/ndbapi_example1/ndbapi_example1.cpp
@@ -0,0 +1,193 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//
+// ndbapi_example1.cpp: Using synchronous transactions in NDB API
+//
+// Correct output from this program is:
+//
+// ATTR1 ATTR2
+// 0 10
+// 1 1
+// 2 12
+// Detected that deleted tuple doesn't exist!
+// 4 14
+// 5 5
+// 6 16
+// 7 7
+// 8 18
+// 9 9
+
+#include <NdbApi.hpp>
+
+// Used for cout
+#include <stdio.h>
+#include <iostream>
+
+#define APIERROR(error) \
+ { std::cout << "Error in " << __FILE__ << ", line:" << __LINE__ << ", code:" \
+ << error.code << ", msg: " << error.message << "." << std::endl; \
+ exit(-1); }
+
+int main()
+{
+ Ndb* myNdb = new Ndb( "TEST_DB_1" ); // Object representing the database
+ NdbDictionary::Table myTable;
+ NdbDictionary::Column myColumn;
+
+ NdbConnection *myConnection; // For other transactions
+ NdbOperation *myOperation; // For other operations
+ NdbRecAttr *myRecAttr; // Result of reading attribute value
+
+ /********************************************
+ * Initialize NDB and wait until it's ready *
+ ********************************************/
+ if (myNdb->init() == -1) {
+ APIERROR(myNdb->getNdbError());
+ exit(-1);
+ }
+
+ if (myNdb->waitUntilReady(30) != 0) {
+ std::cout << "NDB was not ready within 30 secs." << std::endl;
+ exit(-1);
+ }
+
+ NdbDictionary::Dictionary* myDict = myNdb->getDictionary();
+
+ /*********************************************************
+ * Create a table named MYTABLENAME if it does not exist *
+ *********************************************************/
+ if (myDict->getTable("MYTABLENAME") != NULL) {
+ std::cout << "NDB already has example table: MYTABLENAME." << std::endl;
+ exit(-1);
+ }
+
+ myTable.setName("MYTABLENAME");
+
+ myColumn.setName("ATTR1");
+ myColumn.setPrimaryKey(true);
+ myColumn.setType(NdbDictionary::Column::Unsigned);
+ myColumn.setLength(1);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+ myColumn.setName("ATTR2");
+ myColumn.setPrimaryKey(false);
+ myColumn.setType(NdbDictionary::Column::Unsigned);
+ myColumn.setLength(1);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+ if (myDict->createTable(myTable) == -1)
+ APIERROR(myDict->getNdbError());
+
+ /**************************************************************************
+ * Using 5 transactions, insert 10 tuples in table: (0,0),(1,1),...,(9,9) *
+ **************************************************************************/
+ for (int i = 0; i < 5; i++) {
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myOperation = myConnection->getNdbOperation("MYTABLENAME");
+ if (myOperation == NULL) APIERROR(myConnection->getNdbError());
+
+ myOperation->insertTuple();
+ myOperation->equal("ATTR1", i);
+ myOperation->setValue("ATTR2", i);
+
+ myOperation = myConnection->getNdbOperation("MYTABLENAME");
+ if (myOperation == NULL) APIERROR(myConnection->getNdbError());
+
+ myOperation->insertTuple();
+ myOperation->equal("ATTR1", i+5);
+ myOperation->setValue("ATTR2", i+5);
+
+ if (myConnection->execute( Commit ) == -1)
+ APIERROR(myConnection->getNdbError());
+
+ myNdb->closeTransaction(myConnection);
+ }
+
+ /*****************************************************************
+ * Update the second attribute in half of the tuples (adding 10) *
+ *****************************************************************/
+ for (int i = 0; i < 10; i+=2) {
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myOperation = myConnection->getNdbOperation("MYTABLENAME");
+ if (myOperation == NULL) APIERROR(myConnection->getNdbError());
+
+ myOperation->updateTuple();
+ myOperation->equal( "ATTR1", i );
+ myOperation->setValue( "ATTR2", i+10);
+
+ if( myConnection->execute( Commit ) == -1 )
+ APIERROR(myConnection->getNdbError());
+
+ myNdb->closeTransaction(myConnection);
+ }
+
+ /*************************************************
+ * Delete one tuple (the one with primary key 3) *
+ *************************************************/
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myOperation = myConnection->getNdbOperation("MYTABLENAME");
+ if (myOperation == NULL)
+ APIERROR(myConnection->getNdbError());
+
+ myOperation->deleteTuple();
+ myOperation->equal( "ATTR1", 3 );
+
+ if (myConnection->execute(Commit) == -1)
+ APIERROR(myConnection->getNdbError());
+
+ myNdb->closeTransaction(myConnection);
+
+ /*****************************
+ * Read and print all tuples *
+ *****************************/
+ std::cout << "ATTR1 ATTR2" << std::endl;
+
+ for (int i = 0; i < 10; i++) {
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myOperation = myConnection->getNdbOperation("MYTABLENAME");
+ if (myOperation == NULL) APIERROR(myConnection->getNdbError());
+
+ myOperation->readTuple();
+ myOperation->equal("ATTR1", i);
+
+ myRecAttr = myOperation->getValue("ATTR2", NULL);
+ if (myRecAttr == NULL) APIERROR(myConnection->getNdbError());
+
+ if(myConnection->execute( Commit ) == -1)
+ if (i == 3) {
+ std::cout << "Detected that deleted tuple doesn't exist!" << std::endl;
+ } else {
+ APIERROR(myConnection->getNdbError());
+ }
+
+ if (i != 3) {
+ printf(" %2d %2d\n", i, myRecAttr->u_32_value());
+ }
+ myNdb->closeTransaction(myConnection);
+ }
+ delete myNdb;
+}
diff --git a/ndb/examples/ndbapi_example2/Makefile b/ndb/examples/ndbapi_example2/Makefile
new file mode 100644
index 00000000000..17b2b1528fc
--- /dev/null
+++ b/ndb/examples/ndbapi_example2/Makefile
@@ -0,0 +1,33 @@
+-include .defs.mk
+#NDB_OS = OS_YOU_ARE_RUNNING_ON
+#You need to set the NDB_OS variable here
+TARGET = ndbapi_example2
+SRCS = ndbapi_example2.cpp
+OBJS = ndbapi_example2.o
+CXX = g++
+CFLAGS = -c -Wall -fno-rtti -fno-exceptions
+DEBUG =
+LFLAGS = -Wall
+INCLUDE_DIR = ../../include
+LIB_DIR = ../../lib
+ifeq ($(NDB_OS), SOLARIS)
+# Here is the definition of system libraries necessary for Solaris 7
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), LINUX)
+# Here is the definition of system libraries necessary for Linux 2.4
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), MACOSX)
+# Here is the definition of system libraries necessary for Mac OS X
+SYS_LIB =
+endif
+
+$(TARGET): $(OBJS)
+ $(CXX) $(LFLAGS) -L$(LIB_DIR) $(OBJS) -lNDB_API $(SYS_LIB) -o $(TARGET)
+
+$(TARGET).o: $(SRCS)
+ $(CXX) $(CFLAGS) -I$(INCLUDE_DIR) -I$(INCLUDE_DIR)/ndbapi $(SRCS)
+
+clean:
+ rm -f *.o $(TARGET)
diff --git a/ndb/examples/ndbapi_example2/ndbapi_example2.cpp b/ndb/examples/ndbapi_example2/ndbapi_example2.cpp
new file mode 100644
index 00000000000..1c61721c037
--- /dev/null
+++ b/ndb/examples/ndbapi_example2/ndbapi_example2.cpp
@@ -0,0 +1,110 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//
+// ndbapi_example2.cpp: Using asynchronous transactions in NDB API
+//
+// Execute ndbapi_example1 to create the table "MYTABLENAME"
+// before executing this program.
+//
+// Correct output from this program is:
+//
+// Successful insert.
+// Successful insert.
+
+#include <NdbApi.hpp>
+
+// Used for cout
+#include <iostream>
+
+#define APIERROR(error) \
+ { std::cout << "Error in " << __FILE__ << ", line:" << __LINE__ << ", code:" \
+ << error.code << ", msg: " << error.message << "." << std::endl; \
+ exit(-1); }
+
+static void callback(int result, NdbConnection* NdbObject, void* aObject);
+
+int main()
+{
+ Ndb* myNdb = new Ndb( "TEST_DB_2" ); // Object representing the database
+
+ NdbConnection* myNdbConnection[2]; // For transactions
+ NdbOperation* myNdbOperation; // For operations
+
+ /*******************************************
+ * Initialize NDB and wait until its ready *
+ *******************************************/
+ if (myNdb->init(2) == -1) { // Want two parallel insert transactions
+ APIERROR(myNdb->getNdbError());
+ exit(-1);
+ }
+
+ if (myNdb->waitUntilReady(30) != 0) {
+ std::cout << "NDB was not ready within 30 secs." << std::endl;
+ exit(-1);
+ }
+
+ /******************************************************
+ * Insert (we do two insert transactions in parallel) *
+ ******************************************************/
+ for (int i = 0; i < 2; i++) {
+ myNdbConnection[i] = myNdb->startTransaction();
+ if (myNdbConnection[i] == NULL) APIERROR(myNdb->getNdbError());
+
+ myNdbOperation = myNdbConnection[i]->getNdbOperation("MYTABLENAME");
+ // Error check. If error, then maybe table MYTABLENAME is not in database
+ if (myNdbOperation == NULL) APIERROR(myNdbConnection[i]->getNdbError());
+
+ myNdbOperation->insertTuple();
+ myNdbOperation->equal("ATTR1", 20 + i);
+ myNdbOperation->setValue("ATTR2", 20 + i);
+
+ // Prepare transaction (the transaction is NOT yet sent to NDB)
+ myNdbConnection[i]->executeAsynchPrepare(Commit, &callback, NULL);
+ }
+
+ // Send all transactions to NDB
+ myNdb->sendPreparedTransactions(0);
+
+ // Poll all transactions
+ myNdb->pollNdb(3000, 2);
+
+ // Close all transactions
+ for (int i = 0; i < 2; i++)
+ myNdb->closeTransaction(myNdbConnection[i]);
+
+ delete myNdb;
+}
+
+/*
+ * callback : This is called when the transaction is polled
+ *
+ * (This function must have three arguments:
+ * - The result of the transaction,
+ * - The NdbConnection object, and
+ * - A pointer to an arbitrary object.)
+ */
+
+static void
+callback(int result, NdbConnection* myTrans, void* aObject)
+{
+ if (result == -1) {
+ std::cout << "Poll error: " << std::endl;
+ APIERROR(myTrans->getNdbError());
+ } else {
+ std::cout << "Successful insert." << std::endl;
+ }
+}
diff --git a/ndb/examples/ndbapi_example3/Makefile b/ndb/examples/ndbapi_example3/Makefile
new file mode 100644
index 00000000000..bd6f0182aa4
--- /dev/null
+++ b/ndb/examples/ndbapi_example3/Makefile
@@ -0,0 +1,33 @@
+-include .defs.mk
+#NDB_OS = OS_YOU_ARE_RUNNING_ON
+#You need to set the NDB_OS variable here
+TARGET = ndbapi_example3
+SRCS = ndbapi_example3.cpp
+OBJS = ndbapi_example3.o
+CXX = g++
+CFLAGS = -c -Wall -fno-rtti -fno-exceptions
+DEBUG =
+LFLAGS = -Wall
+INCLUDE_DIR = ../../include
+LIB_DIR = ../../lib
+ifeq ($(NDB_OS), SOLARIS)
+# Here is the definition of system libraries necessary for Solaris 7
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), LINUX)
+# Here is the definition of system libraries necessary for Linux 2.4
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), MACOSX)
+# Here is the definition of system libraries necessary for Mac OS X
+SYS_LIB =
+endif
+
+$(TARGET): $(OBJS)
+ $(CXX) $(LFLAGS) -L$(LIB_DIR) $(OBJS) -lNDB_API $(SYS_LIB) -o $(TARGET)
+
+$(TARGET).o: $(SRCS)
+ $(CXX) $(CFLAGS) -I$(INCLUDE_DIR) -I$(INCLUDE_DIR)/ndbapi $(SRCS)
+
+clean:
+ rm -f *.o $(TARGET)
diff --git a/ndb/examples/ndbapi_example3/ndbapi_example3.cpp b/ndb/examples/ndbapi_example3/ndbapi_example3.cpp
new file mode 100644
index 00000000000..36d2cf1608c
--- /dev/null
+++ b/ndb/examples/ndbapi_example3/ndbapi_example3.cpp
@@ -0,0 +1,202 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//
+// ndbapi_example3.cpp: Error handling and transaction retries
+//
+// Execute ndbapi_example1 to create the table "MYTABLENAME"
+// before executing this program.
+//
+// There are many ways to program using the NDB API. In this example
+// we execute two inserts in the same transaction using
+// NdbConnection::Ndbexecute(NoCommit).
+//
+// Transaction failing is handled by re-executing the transaction
+// in case of non-permanent transaction errors.
+// Application errors (i.e. errors at points marked with APIERROR)
+// should be handled by the application programmer.
+
+#include <NdbApi.hpp>
+
+// Used for cout
+#include <iostream>
+
+// Used for sleep (use your own version of sleep)
+#include <unistd.h>
+#define TIME_TO_SLEEP_BETWEEN_TRANSACTION_RETRIES 1
+
+//
+// APIERROR prints an NdbError object
+//
+#define APIERROR(error) \
+ { std::cout << "API ERROR: " << error.code << " " << error.message \
+ << std::endl \
+ << " " << "Status: " << error.status \
+ << ", Classification: " << error.classification << std::endl\
+ << " " << "File: " << __FILE__ \
+ << " (Line: " << __LINE__ << ")" << std::endl \
+ ; \
+ }
+
+//
+// CONERROR prints all error info regarding an NdbConnection
+//
+#define CONERROR(ndbConnection) \
+ { NdbError error = ndbConnection->getNdbError(); \
+ std::cout << "CON ERROR: " << error.code << " " << error.message \
+ << std::endl \
+ << " " << "Status: " << error.status \
+ << ", Classification: " << error.classification << std::endl \
+ << " " << "File: " << __FILE__ \
+ << " (Line: " << __LINE__ << ")" << std::endl \
+ ; \
+ printTransactionError(ndbConnection); \
+ }
+
+void printTransactionError(NdbConnection *ndbConnection) {
+ const NdbOperation *ndbOp = NULL;
+ int i=0;
+
+ /****************************************************************
+ * Print NdbError object of every operations in the transaction *
+ ****************************************************************/
+ while ((ndbOp = ndbConnection->getNextCompletedOperation(ndbOp)) != NULL) {
+ NdbError error = ndbOp->getNdbError();
+ std::cout << " OPERATION " << i+1 << ": "
+ << error.code << " " << error.message << std::endl
+ << " Status: " << error.status
+ << ", Classification: " << error.classification << std::endl;
+ i++;
+ }
+}
+
+
+//
+// Example insert
+// @param myNdb Ndb object representing NDB Cluster
+// @param myConnection NdbConnection used for transaction
+// @param error NdbError object returned in case of errors
+// @return -1 in case of failures, 0 otherwise
+//
+int insert(int transactionId, NdbConnection* myConnection) {
+ NdbOperation *myOperation; // For other operations
+
+ myOperation = myConnection->getNdbOperation("MYTABLENAME");
+ if (myOperation == NULL) return -1;
+
+ if (myOperation->insertTuple() ||
+ myOperation->equal("ATTR1", transactionId) ||
+ myOperation->setValue("ATTR2", transactionId)) {
+ APIERROR(myOperation->getNdbError());
+ exit(-1);
+ }
+
+ return myConnection->execute(NoCommit);
+}
+
+
+//
+// Execute function which re-executes (tries 10 times) the transaction
+// if there are temporary errors (e.g. the NDB Cluster is overloaded).
+// @return -1 failure, 1 success
+//
+int executeInsertTransaction(int transactionId, Ndb* myNdb) {
+ int result = 0; // No result yet
+ int noOfRetriesLeft = 10;
+ NdbConnection *myConnection; // For other transactions
+ NdbError ndberror;
+
+ while (noOfRetriesLeft > 0 && !result) {
+
+ /*********************************
+ * Start and execute transaction *
+ *********************************/
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) {
+ APIERROR(myNdb->getNdbError());
+ ndberror = myNdb->getNdbError();
+ result = -1; // Failure
+ } else if (insert(transactionId, myConnection) ||
+ insert(10000+transactionId, myConnection) ||
+ myConnection->execute(Commit)) {
+ CONERROR(myConnection);
+ ndberror = myConnection->getNdbError();
+ result = -1; // Failure
+ } else {
+ result = 1; // Success
+ }
+
+ /**********************************
+ * If failure, then analyze error *
+ **********************************/
+ if (result == -1) {
+ switch (ndberror.status) {
+ case NdbError::Success:
+ break;
+ case NdbError::TemporaryError:
+ std::cout << "Retrying transaction..." << std::endl;
+ sleep(TIME_TO_SLEEP_BETWEEN_TRANSACTION_RETRIES);
+ --noOfRetriesLeft;
+ result = 0; // No completed transaction yet
+ break;
+
+ case NdbError::UnknownResult:
+ case NdbError::PermanentError:
+ std::cout << "No retry of transaction..." << std::endl;
+ result = -1; // Permanent failure
+ break;
+ }
+ }
+
+ /*********************
+ * Close transaction *
+ *********************/
+ if (myConnection != NULL) {
+ myNdb->closeTransaction(myConnection);
+ }
+ }
+
+ if (result != 1) exit(-1);
+ return result;
+}
+
+
+int main()
+{
+ Ndb* myNdb = new Ndb( "TEST_DB_1" ); // Object representing the database
+
+ /*******************************************
+ * Initialize NDB and wait until its ready *
+ *******************************************/
+ if (myNdb->init() == -1) {
+ APIERROR(myNdb->getNdbError());
+ exit(-1);
+ }
+
+ if (myNdb->waitUntilReady(30) != 0) {
+ std::cout << "NDB was not ready within 30 secs." << std::endl;
+ exit(-1);
+ }
+
+ /************************************
+ * Execute some insert transactions *
+ ************************************/
+ for (int i = 10000; i < 20000; i++) {
+ executeInsertTransaction(i, myNdb);
+ }
+
+ delete myNdb;
+}
diff --git a/ndb/examples/ndbapi_example4/Makefile b/ndb/examples/ndbapi_example4/Makefile
new file mode 100644
index 00000000000..b0ce852d347
--- /dev/null
+++ b/ndb/examples/ndbapi_example4/Makefile
@@ -0,0 +1,33 @@
+-include .defs.mk
+#NDB_OS = OS_YOU_ARE_RUNNING_ON
+#You need to set the NDB_OS variable here
+TARGET = ndbapi_example4
+SRCS = ndbapi_example4.cpp
+OBJS = ndbapi_example4.o
+CXX = g++
+CFLAGS = -c -Wall -fno-rtti -fno-exceptions
+DEBUG =
+LFLAGS = -Wall
+INCLUDE_DIR = ../../include
+LIB_DIR = ../../lib
+ifeq ($(NDB_OS), SOLARIS)
+# Here is the definition of system libraries necessary for Solaris 7
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), LINUX)
+# Here is the definition of system libraries necessary for Linux 2.4
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), MACOSX)
+# Here is the definition of system libraries necessary for Mac OS X
+SYS_LIB =
+endif
+
+$(TARGET): $(OBJS)
+ $(CXX) $(LFLAGS) -L$(LIB_DIR) $(OBJS) -lNDB_API $(SYS_LIB) -o $(TARGET)
+
+$(TARGET).o: $(SRCS)
+ $(CXX) $(CFLAGS) -I$(INCLUDE_DIR) -I$(INCLUDE_DIR)/ndbapi $(SRCS)
+
+clean:
+ rm -f *.o $(TARGET)
diff --git a/ndb/examples/ndbapi_example4/ndbapi_example4.cpp b/ndb/examples/ndbapi_example4/ndbapi_example4.cpp
new file mode 100644
index 00000000000..520172b9b0c
--- /dev/null
+++ b/ndb/examples/ndbapi_example4/ndbapi_example4.cpp
@@ -0,0 +1,252 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//
+// ndbapi_example4.cpp: Using secondary indexes in NDB API
+//
+// Correct output from this program is:
+//
+// ATTR1 ATTR2
+// 0 10
+// 1 1
+// 2 12
+// Detected that deleted tuple doesn't exist!
+// 4 14
+// 5 5
+// 6 16
+// 7 7
+// 8 18
+// 9 9
+
+#include <NdbApi.hpp>
+
+// Used for cout
+#include <stdio.h>
+#include <iostream>
+
+#define APIERROR(error) \
+ { std::cout << "Error in " << __FILE__ << ", line:" << __LINE__ << ", code:" \
+ << error.code << ", msg: " << error.message << "." << std::endl; \
+ exit(-1); }
+
+int main()
+{
+ Ndb* myNdb = new Ndb( "TEST_DB_1" ); // Object representing the database
+ NdbDictionary::Table myTable;
+ NdbDictionary::Column myColumn;
+ NdbDictionary::Index myIndex;
+
+ NdbConnection *myConnection; // For transactions
+ NdbOperation *myOperation; // For primary key operations
+ NdbIndexOperation *myIndexOperation; // For index operations
+ NdbRecAttr *myRecAttr; // Result of reading attribute value
+
+ /********************************************
+ * Initialize NDB and wait until it's ready *
+ ********************************************/
+ if (myNdb->init() == -1) {
+ APIERROR(myNdb->getNdbError());
+ exit(-1);
+ }
+
+ if (myNdb->waitUntilReady(30) != 0) {
+ std::cout << "NDB was not ready within 30 secs." << std::endl;
+ exit(-1);
+ }
+
+ /*********************************************************
+ * Create a table named MYTABLENAME if it does not exist *
+ *********************************************************/
+ NdbDictionary::Dictionary* myDict = myNdb->getDictionary();
+ if (myDict->getTable("MYTABLENAME") != NULL) {
+ std::cout << "NDB already has example table: MYTABLENAME." << std::endl;
+ exit(-1);
+ }
+
+ myTable.setName("MYTABLENAME");
+
+ myColumn.setName("ATTR1");
+ myColumn.setPrimaryKey(true);
+ myColumn.setType(NdbDictionary::Column::Unsigned);
+ myColumn.setLength(1);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+ myColumn.setName("ATTR2");
+ myColumn.setPrimaryKey(false);
+ myColumn.setType(NdbDictionary::Column::Unsigned);
+ myColumn.setLength(1);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+ if (myDict->createTable(myTable) == -1)
+ APIERROR(myDict->getNdbError());
+
+
+ /**********************************************************
+ * Create an index named MYINDEXNAME if it does not exist *
+ **********************************************************/
+ if (myDict->getIndex("MYINDEXNAME", "MYTABLENAME") != NULL) {
+ std::cout << "NDB already has example index: MYINDEXNAME." << std::endl;
+ exit(-1);
+ }
+
+ myIndex.setName("MYINDEXNAME");
+ myIndex.setTable("MYTABLENAME");
+ myIndex.setType(NdbDictionary::Index::UniqueHashIndex);
+ const char* attr_arr[] = {"ATTR2"};
+ myIndex.addIndexColumns(1, attr_arr);
+
+ if (myDict->createIndex(myIndex) == -1)
+ APIERROR(myDict->getNdbError());
+
+
+ /**************************************************************************
+ * Using 5 transactions, insert 10 tuples in table: (0,0),(1,1),...,(9,9) *
+ **************************************************************************/
+ for (int i = 0; i < 5; i++) {
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myOperation = myConnection->getNdbOperation("MYTABLENAME");
+ if (myOperation == NULL) APIERROR(myConnection->getNdbError());
+
+ myOperation->insertTuple();
+ myOperation->equal("ATTR1", i);
+ myOperation->setValue("ATTR2", i);
+
+ myOperation = myConnection->getNdbOperation("MYTABLENAME");
+ if (myOperation == NULL) APIERROR(myConnection->getNdbError());
+
+ myOperation->insertTuple();
+ myOperation->equal("ATTR1", i+5);
+ myOperation->setValue("ATTR2", i+5);
+
+ if (myConnection->execute( Commit ) == -1)
+ APIERROR(myConnection->getNdbError());
+
+ myNdb->closeTransaction(myConnection);
+ }
+
+ /*****************************************
+ * Read and print all tuples using index *
+ *****************************************/
+ std::cout << "ATTR1 ATTR2" << std::endl;
+
+ for (int i = 0; i < 10; i++) {
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myIndexOperation = myConnection->getNdbIndexOperation("MYINDEXNAME",
+ "MYTABLENAME");
+ if (myIndexOperation == NULL) APIERROR(myConnection->getNdbError());
+
+ myIndexOperation->readTuple();
+ myIndexOperation->equal("ATTR2", i);
+
+ myRecAttr = myIndexOperation->getValue("ATTR1", NULL);
+ if (myRecAttr == NULL) APIERROR(myConnection->getNdbError());
+
+ if(myConnection->execute( Commit ) != -1)
+ printf(" %2d %2d\n", myRecAttr->u_32_value(), i);
+ }
+ myNdb->closeTransaction(myConnection);
+
+ /*****************************************************************
+ * Update the second attribute in half of the tuples (adding 10) *
+ *****************************************************************/
+ for (int i = 0; i < 10; i+=2) {
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myIndexOperation = myConnection->getNdbIndexOperation("MYINDEXNAME",
+ "MYTABLENAME");
+ if (myIndexOperation == NULL) APIERROR(myConnection->getNdbError());
+
+ myIndexOperation->updateTuple();
+ myIndexOperation->equal( "ATTR2", i );
+ myIndexOperation->setValue( "ATTR2", i+10);
+
+ if( myConnection->execute( Commit ) == -1 )
+ APIERROR(myConnection->getNdbError());
+
+ myNdb->closeTransaction(myConnection);
+ }
+
+ /*************************************************
+ * Delete one tuple (the one with primary key 3) *
+ *************************************************/
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myIndexOperation = myConnection->getNdbIndexOperation("MYINDEXNAME",
+ "MYTABLENAME");
+ if (myIndexOperation == NULL)
+ APIERROR(myConnection->getNdbError());
+
+ myIndexOperation->deleteTuple();
+ myIndexOperation->equal( "ATTR2", 3 );
+
+ if (myConnection->execute(Commit) == -1)
+ APIERROR(myConnection->getNdbError());
+
+ myNdb->closeTransaction(myConnection);
+
+ /*****************************
+ * Read and print all tuples *
+ *****************************/
+ std::cout << "ATTR1 ATTR2" << std::endl;
+
+ for (int i = 0; i < 10; i++) {
+ myConnection = myNdb->startTransaction();
+ if (myConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myOperation = myConnection->getNdbOperation("MYTABLENAME");
+ if (myOperation == NULL) APIERROR(myConnection->getNdbError());
+
+ myOperation->readTuple();
+ myOperation->equal("ATTR1", i);
+
+ myRecAttr = myOperation->getValue("ATTR2", NULL);
+ if (myRecAttr == NULL) APIERROR(myConnection->getNdbError());
+
+ if(myConnection->execute( Commit ) == -1)
+ if (i == 3) {
+ std::cout << "Detected that deleted tuple doesn't exist!" << std::endl;
+ } else {
+ APIERROR(myConnection->getNdbError());
+ }
+
+ if (i != 3) {
+ printf(" %2d %2d\n", i, myRecAttr->u_32_value());
+ }
+ myNdb->closeTransaction(myConnection);
+ }
+
+ /**************
+ * Drop index *
+ **************/
+ if (myDict->dropIndex("MYINDEXNAME", "MYTABLENAME") == -1)
+ APIERROR(myDict->getNdbError());
+
+ /**************
+ * Drop table *
+ **************/
+ if (myDict->dropTable("MYTABLENAME") == -1)
+ APIERROR(myDict->getNdbError());
+
+ delete myNdb;
+}
diff --git a/ndb/examples/ndbapi_example5/Makefile b/ndb/examples/ndbapi_example5/Makefile
new file mode 100644
index 00000000000..e2e3f06374a
--- /dev/null
+++ b/ndb/examples/ndbapi_example5/Makefile
@@ -0,0 +1,33 @@
+-include .defs.mk
+#NDB_OS = OS_YOU_ARE_RUNNING_ON
+#You need to set the NDB_OS variable here
+TARGET = ndbapi_example5
+SRCS = ndbapi_example5.cpp
+OBJS = ndbapi_example5.o
+CXX = g++
+CFLAGS = -c -Wall -fno-rtti -fno-exceptions
+DEBUG =
+LFLAGS = -Wall
+INCLUDE_DIR = ../../include
+LIB_DIR = ../../lib
+ifeq ($(NDB_OS), SOLARIS)
+# Here is the definition of system libraries necessary for Solaris 7
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), LINUX)
+# Here is the definition of system libraries necessary for Linux 2.4
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), MACOSX)
+# Here is the definition of system libraries necessary for Mac OS X
+SYS_LIB =
+endif
+
+$(TARGET): $(OBJS)
+ $(CXX) $(LFLAGS) -L$(LIB_DIR) $(OBJS) -lNDB_API $(SYS_LIB) -o $(TARGET)
+
+$(TARGET).o: $(SRCS)
+ $(CXX) $(CFLAGS) -I$(INCLUDE_DIR) -I$(INCLUDE_DIR)/ndbapi $(SRCS)
+
+clean:
+ rm -f *.o $(TARGET)
diff --git a/ndb/examples/ndbapi_example5/ndbapi_example5.cpp b/ndb/examples/ndbapi_example5/ndbapi_example5.cpp
new file mode 100644
index 00000000000..a9d3099883c
--- /dev/null
+++ b/ndb/examples/ndbapi_example5/ndbapi_example5.cpp
@@ -0,0 +1,230 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * ndbapi_example5.cpp: Using API level events in NDB API
+ */
+
+#include <NdbApi.hpp>
+#include <NdbEventOperation.hpp>
+
+// Used for cout
+#include <stdio.h>
+#include <iostream>
+#include <unistd.h>
+
+
+/**
+ *
+ * Assume that there is a table TAB0 which is being updated by
+ * another process (e.g. flexBench -l 0 -stdtables).
+ * We want to monitor what happens with columns COL0, COL2, COL11
+ *
+ * or together with the mysqlcluster client;
+ *
+ * shell> mysqlcluster -u root
+ * mysql> create database TEST_DB;
+ * mysql> use TEST_DB;
+ * mysql> create table TAB0 (COL0 int primary key, COL1 int, COL11 int);
+ *
+ * In another window start ndbapi_example5, wait until properly started
+ *
+ * mysql> insert into TAB0 values (1,2,3);
+ * mysql> insert into TAB0 values (2,2,3);
+ * mysql> insert into TAB0 values (3,2,9);
+ * mysql>
+ *
+ * you should see the data popping up in the example window
+ *
+ */
+
+#define APIERROR(error) \
+ { std::cout << "Error in " << __FILE__ << ", line:" << __LINE__ << ", code:" \
+ << error.code << ", msg: " << error.message << "." << std::endl; \
+ exit(-1); }
+
+Ndb* myCreateNdb();
+int myCreateEvent(Ndb* myNdb,
+ const char *eventName,
+ const char *eventTableName,
+ const char **eventComlumnName,
+ const int noEventComlumnName);
+
+int main()
+{
+ Ndb* myNdb = myCreateNdb();
+ NdbDictionary::Dictionary *myDict;
+
+ const char *eventName = "CHNG_IN_TAB0";
+ const char *eventTableName = "TAB0";
+ const int noEventColumnName = 3;
+ const char *eventColumnName[noEventColumnName] =
+ {"COL0",
+ "COL1",
+ "COL11"};
+
+ myDict = myNdb->getDictionary();
+
+ // Create events
+ myCreateEvent(myNdb,
+ eventName,
+ eventTableName,
+ eventColumnName,
+ noEventColumnName);
+ int j = 0;
+ while (j < 5) {
+
+ // Start "transaction" for handling events
+ NdbEventOperation* op;
+ printf("create EventOperation\n");
+ if ((op = myNdb->createEventOperation(eventName,100)) == NULL) {
+ printf("Event operation creation failed\n");
+ exit(-1);
+ }
+
+ printf("get values\n");
+ NdbRecAttr* recAttr[noEventColumnName];
+ NdbRecAttr* recAttrPre[noEventColumnName];
+ // primary keys should always be a part of the result
+ for (int i = 0; i < noEventColumnName; i++) {
+ recAttr[i] = op->getValue(eventColumnName[i]);
+ recAttrPre[i] = op->getPreValue(eventColumnName[i]);
+ }
+
+ // set up the callbacks
+ printf("execute\n");
+ if (op->execute()) { // This starts changes to "start flowing"
+ printf("operationd execution failed\n");
+ exit(-1);
+ }
+
+ int i = 0;
+
+ while(i < 40) {
+ //printf("now waiting for event...\n");
+ int r = myNdb->pollEvents(1000); // wait for event or 1000 ms
+ if (r>0) {
+ //printf("got data! %d\n", r);
+ int overrun;
+ while (op->next(&overrun) > 0) {
+ i++;
+ if (!op->isConsistent())
+ printf("A node failiure has occured and events might be missing\n");
+ switch (op->getEventType()) {
+ case NdbDictionary::Event::TE_INSERT:
+ printf("%u INSERT: ", i);
+ break;
+ case NdbDictionary::Event::TE_DELETE:
+ printf("%u DELETE: ", i);
+ break;
+ case NdbDictionary::Event::TE_UPDATE:
+ printf("%u UPDATE: ", i);
+ break;
+ }
+ printf("overrun %u pk %u: ", overrun, recAttr[0]->u_32_value());
+ for (int i = 1; i < noEventColumnName; i++) {
+ if (recAttr[i]->isNULL() >= 0) { // we have a value
+ printf(" post[%u]=", i);
+ if (recAttr[i]->isNULL() == 0) // we have a non-null value
+ printf("%u", recAttr[i]->u_32_value());
+ else // we have a null value
+ printf("NULL");
+ }
+ if (recAttrPre[i]->isNULL() >= 0) { // we have a value
+ printf(" post[%u]=", i);
+ if (recAttrPre[i]->isNULL() == 0) // we have a non-null value
+ printf("%u", recAttrPre[i]->u_32_value());
+ else // we have a null value
+ printf("NULL");
+ }
+ }
+ printf("\n");
+ }
+ } else
+ ;//printf("timed out\n");
+ }
+ // don't want to listen to eventsanymore
+ myNdb->dropEventOperation(op);
+
+ j++;
+ }
+
+ myDict->dropEvent(eventName); // remove event from database
+
+ delete myNdb;
+}
+
+Ndb* myCreateNdb()
+{
+ Ndb* myNdb = new Ndb("TEST_DB");
+
+ /********************************************
+ * Initialize NDB and wait until it's ready *
+ ********************************************/
+ if (myNdb->init() == -1) {
+ APIERROR(myNdb->getNdbError());
+ exit(-1);
+ }
+
+ if (myNdb->waitUntilReady(30) != 0) {
+ std::cout << "NDB was not ready within 30 secs." << std::endl;
+ exit(-1);
+ }
+
+ return myNdb;
+}
+
+int myCreateEvent(Ndb* myNdb,
+ const char *eventName,
+ const char *eventTableName,
+ const char **eventColumnName,
+ const int noEventColumnName)
+{
+ NdbDictionary::Dictionary *myDict = myNdb->getDictionary();
+
+ if (!myDict) {
+ printf("Event Creation failedDictionary not found");
+ exit(-1);
+ }
+
+ NdbDictionary::Event myEvent(eventName);
+ myEvent.setTable(eventTableName);
+ myEvent.addTableEvent(NdbDictionary::Event::TE_ALL);
+ // myEvent.addTableEvent(NdbDictionary::Event::TE_INSERT);
+ // myEvent.addTableEvent(NdbDictionary::Event::TE_UPDATE);
+ // myEvent.addTableEvent(NdbDictionary::Event::TE_DELETE);
+
+ for (int i = 0; i < noEventColumnName; i++)
+ myEvent.addEventColumn(eventColumnName[i]);
+
+ int res = myDict->createEvent(myEvent); // Add event to database
+
+ if (res == 0)
+ myEvent.print();
+ else {
+ printf("Event creation failed\n");
+ printf("trying drop Event, maybe event exists\n");
+ res = myDict->dropEvent(eventName);
+ if (res)
+ exit(-1);
+ // try again
+ res = myDict->createEvent(myEvent); // Add event to database
+ if (res)
+ exit(-1);
+ }
+
+ return res;
+}
diff --git a/ndb/examples/ndbapi_scan_example/Makefile b/ndb/examples/ndbapi_scan_example/Makefile
new file mode 100644
index 00000000000..6e53317f8bf
--- /dev/null
+++ b/ndb/examples/ndbapi_scan_example/Makefile
@@ -0,0 +1,35 @@
+-include ../../Defs.mk
+#NDB_OS = OS_YOU_ARE_RUNNING_ON
+#You need to set the NDB_OS variable here (LINUX, SOLARIS, MACOSX)
+#NDB_OS = LINUX
+
+TARGET = ndbapi_scan
+SRCS = ndbapi_scan.cpp
+OBJS = ndbapi_scan.o
+CC = g++
+CFLAGS = -c -Wall -fno-rtti
+DEBUG =
+LFLAGS = -Wall
+INCLUDE_DIR = ../../include
+LIB_DIR = ../../lib
+ifeq ($(NDB_OS), SOLARIS)
+# Here is the definition of system libraries necessary for Solaris 7
+SYS_LIB = -lpthread -lsocket -lnsl -lrt
+endif
+ifeq ($(NDB_OS), LINUX)
+# Here is the definition of system libraries necessary for Linux 2.4
+SYS_LIB = -lpthread
+endif
+ifeq ($(NDB_OS), MACOSX)
+# Here is the definition of system libraries necessary for Mac OS X
+SYS_LIB =
+endif
+
+$(TARGET): $(OBJS)
+ $(CC) $(LFLAGS) -L$(LIB_DIR) -lNDB_API $(OBJS) $(SYS_LIB) -o $(TARGET)
+
+$(TARGET).o: $(SRCS)
+ $(CC) $(CFLAGS) -I$(INCLUDE_DIR) -I$(INCLUDE_DIR)/ndbapi $(SRCS)
+
+clean:
+ rm -f *.o $(TARGET)
diff --git a/ndb/examples/ndbapi_scan_example/ndbapi_scan.cpp b/ndb/examples/ndbapi_scan_example/ndbapi_scan.cpp
new file mode 100644
index 00000000000..186afdb9471
--- /dev/null
+++ b/ndb/examples/ndbapi_scan_example/ndbapi_scan.cpp
@@ -0,0 +1,824 @@
+
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/*
+ * ndbapi_scan.cpp:
+ * Illustrates how to use the scan api in the NDBAPI.
+ * The example shows how to do scan, scan for update and scan for delete
+ * using NdbScanFilter and NdbScanOperation
+ *
+ * Classes and methods used in this example:
+ *
+ * Ndb
+ * init()
+ * waitUntilRead()
+ * getDictionary()
+ * startTransaction()
+ * closeTransaction()
+ * sendPreparedTransactions()
+ * pollNdb()
+ *
+ * NdbConnection
+ * getNdbOperation()
+ * executeAsynchPrepare()
+ * getNdbError()
+ * executeScan()
+ * nextScanResult()
+ *
+ * NdbDictionary::Dictionary
+ * getTable()
+ * dropTable()
+ * createTable()
+ *
+ * NdbDictionary::Column
+ * setName()
+ * setPrimaryKey()
+ * setType()
+ * setLength()
+ * setNullable()
+ *
+ * NdbDictionary::Table
+ * setName()
+ * addColumn()
+ *
+ * NdbOperation
+ * insertTuple()
+ * equal()
+ * setValue()
+ * openScanRead()
+ * openScanExclusive()
+ *
+ * NdbRecAttr
+ * aRef()
+ * u_32_value()
+ *
+ * NdbResultSet
+ * nextResult()
+ * deleteTuple()
+ * updateTuple()
+ *
+ * NdbScanOperation
+ * getValue()
+ * readTuplesExclusive()
+ *
+ * NdbScanFilter
+ * begin()
+ * eq()
+ * end()
+ *
+ *
+ */
+
+
+#include <NdbApi.hpp>
+#include <NdbScanFilter.hpp>
+// Used for cout
+#include <iostream>
+
+#ifdef SOLARIS
+#include <sys/types.h>
+#include <unistd.h>
+#endif
+
+#if defined LINUX || defined MACOSX
+#include <time.h>
+#include <unistd.h>
+#endif
+
+/**
+ * Helper sleep function
+ */
+int
+milliSleep(int milliseconds){
+ int result = 0;
+ struct timespec sleeptime;
+ sleeptime.tv_sec = milliseconds / 1000;
+ sleeptime.tv_nsec = (milliseconds - (sleeptime.tv_sec * 1000)) * 1000000;
+ result = nanosleep(&sleeptime, NULL);
+ return result;
+}
+
+
+/**
+ * Helper sleep function
+ */
+#define APIERROR(error) \
+ { std::cout << "Error in " << __FILE__ << ", line:" << __LINE__ << ", code:" \
+ << error.code << ", msg: " << error.message << "." << std::endl; \
+ exit(-1); }
+
+/*
+ * callback : This is called when the transaction is polled
+ *
+ * (This function must have three arguments:
+ * - The result of the transaction,
+ * - The NdbConnection object, and
+ * - A pointer to an arbitrary object.)
+ */
+static void
+callback(int result, NdbConnection* myTrans, void* aObject)
+{
+ if (result == -1) {
+ std::cout << "In callback: " << std::endl;
+ /**
+ * Put error checking code here (see ndb_async_example)
+ */
+ APIERROR(myTrans->getNdbError());
+ } else {
+ /**
+ * Ok!
+ */
+ return;
+ }
+}
+
+/**
+ * Function to create table
+ */
+int create_table(Ndb * myNdb)
+{
+ NdbDictionary::Table myTable;
+ NdbDictionary::Column myColumn;
+
+ NdbDictionary::Dictionary* myDict = myNdb->getDictionary();
+
+ /*********************************************************
+ * Create a table named GARAGE if it does not exist *
+ *********************************************************/
+ if (myDict->getTable("GARAGE") != NULL) {
+ std::cout << "NDB already has example table: GARAGE. "
+ << "Dropping it..." << std::endl;
+ if(myDict->dropTable("GARAGE") == -1)
+ {
+ std::cout << "Failed to drop: GARAGE." << std::endl;
+ exit(1);
+ }
+ }
+
+ myTable.setName("GARAGE");
+
+ myColumn.setName("REG_NO");
+ myColumn.setPrimaryKey(true);
+ myColumn.setType(NdbDictionary::Column::Unsigned);
+ myColumn.setLength(1);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+ myColumn.setName("BRAND");
+ myColumn.setPrimaryKey(false);
+ myColumn.setType(NdbDictionary::Column::Char);
+ myColumn.setLength(20);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+
+ myColumn.setName("COLOR");
+ myColumn.setPrimaryKey(false);
+ myColumn.setType(NdbDictionary::Column::Char);
+ myColumn.setLength(20);
+ myColumn.setNullable(false);
+ myTable.addColumn(myColumn);
+
+ if (myDict->createTable(myTable) == -1) {
+ APIERROR(myDict->getNdbError());
+ return -1;
+ }
+ return 1;
+}
+
+
+int populate(Ndb * myNdb)
+{
+ NdbConnection* myNdbConnection[15]; // For transactions
+ NdbOperation* myNdbOperation; // For operations
+ /******************************************************
+ * Insert (we do 15 insert transactions in parallel) *
+ ******************************************************/
+ /**
+ * Five blue mercedes
+ */
+ for (int i = 0; i < 5; i++)
+ {
+ myNdbConnection[i] = myNdb->startTransaction();
+ if (myNdbConnection[i] == NULL)
+ APIERROR(myNdb->getNdbError());
+ myNdbOperation = myNdbConnection[i]->getNdbOperation("GARAGE");
+ // Error check. If error, then maybe table GARAGE is not in database
+ if (myNdbOperation == NULL)
+ APIERROR(myNdbConnection[i]->getNdbError());
+ myNdbOperation->insertTuple();
+ myNdbOperation->equal("REG_NO", i);
+ myNdbOperation->setValue("BRAND", "Mercedes");
+ myNdbOperation->setValue("COLOR", "Blue");
+ // Prepare transaction (the transaction is NOT yet sent to NDB)
+ myNdbConnection[i]->executeAsynchPrepare(Commit, &callback, NULL);
+ }
+
+
+ /**
+ * Five black bmw
+ */
+ for (int i = 5; i < 10; i++)
+ {
+ myNdbConnection[i] = myNdb->startTransaction();
+ if (myNdbConnection[i] == NULL)
+ APIERROR(myNdb->getNdbError());
+ myNdbOperation = myNdbConnection[i]->getNdbOperation("GARAGE");
+ // Error check. If error, then maybe table MYTABLENAME is not in database
+ if (myNdbOperation == NULL)
+ APIERROR(myNdbConnection[i]->getNdbError());
+ myNdbOperation->insertTuple();
+ myNdbOperation->equal("REG_NO", i);
+ myNdbOperation->setValue("BRAND", "BMW");
+ myNdbOperation->setValue("COLOR", "Black");
+ // Prepare transaction (the transaction is NOT yet sent to NDB)
+ myNdbConnection[i]->executeAsynchPrepare(Commit, &callback, NULL);
+ }
+
+ /**
+ * Five pink toyotas
+ */
+ for (int i = 10; i < 15; i++) {
+ myNdbConnection[i] = myNdb->startTransaction();
+ if (myNdbConnection[i] == NULL) APIERROR(myNdb->getNdbError());
+ myNdbOperation = myNdbConnection[i]->getNdbOperation("GARAGE");
+ // Error check. If error, then maybe table MYTABLENAME is not in database
+ if (myNdbOperation == NULL) APIERROR(myNdbConnection[i]->getNdbError());
+ myNdbOperation->insertTuple();
+ myNdbOperation->equal("REG_NO", i);
+ myNdbOperation->setValue("BRAND", "Toyota");
+ myNdbOperation->setValue("COLOR", "Pink");
+ // Prepare transaction (the transaction is NOT yet sent to NDB)
+ myNdbConnection[i]->executeAsynchPrepare(Commit, &callback, NULL);
+ }
+
+ // Send all transactions to NDB
+ myNdb->sendPreparedTransactions(0);
+ // Poll all transactions
+ myNdb->pollNdb(3000, 0);
+
+ // it is also possible to use sendPollNdb instead of
+ // myNdb->sendPreparedTransactions(0); and myNdb->pollNdb(3000, 15); above.
+ // myNdb->sendPollNdb(3000,0);
+ // Note! Neither sendPollNdb or pollNdb returs until all 15 callbacks have
+ // executed.
+
+ // Close all transactions. It is also possible to close transactions
+ // in the callback.
+ for (int i = 0; i < 15; i++)
+ myNdb->closeTransaction(myNdbConnection[i]);
+ return 1;
+}
+
+int scan_delete(Ndb* myNdb,
+ int parallelism,
+ int column,
+ int column_len,
+ const char * color)
+
+{
+
+ // Scan all records exclusive and delete
+ // them one by one
+ int retryAttempt = 0;
+ const int retryMax = 10;
+ int deletedRows = 0;
+ int check;
+ NdbError err;
+ NdbConnection *myTrans;
+ NdbScanOperation *myScanOp;
+
+ /**
+ * Loop as long as :
+ * retryMax not reached
+ * failed operations due to TEMPORARY erros
+ *
+ * Exit loop;
+ * retyrMax reached
+ * Permanent error (return -1)
+ */
+ while (true)
+ {
+ if (retryAttempt >= retryMax)
+ {
+ std::cout << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << std::endl;
+ return -1;
+ }
+
+ myTrans = myNdb->startTransaction();
+ if (myTrans == NULL)
+ {
+ const NdbError err = myNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError)
+ {
+ milliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ std::cout << err.message << std::endl;
+ return -1;
+ }
+
+ /**
+ * Get a scan operation.
+ */
+ myScanOp = myTrans->getNdbScanOperation("GARAGE");
+ if (myScanOp == NULL)
+ {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * Define a result set for the scan.
+ */
+ NdbResultSet * rs = myScanOp->readTuplesExclusive(parallelism);
+ if( rs == 0 ) {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * Use NdbScanFilter to define a search critera
+ */
+ NdbScanFilter filter(myScanOp) ;
+ if(filter.begin(NdbScanFilter::AND) < 0 ||
+ filter.eq(column, color, column_len, false) <0||
+ filter.end() <0)
+ {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * Start scan (NoCommit since we are only reading at this stage);
+ */
+ if(myTrans->execute(NoCommit) != 0){
+ err = myTrans->getNdbError();
+ if(err.status == NdbError::TemporaryError){
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ milliSleep(50);
+ continue;
+ }
+ std::cout << err.code << std::endl;
+ std::cout << myTrans->getNdbError().code << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+
+ /**
+ * start of loop: nextResult(true) means that "parallelism" number of
+ * rows are fetched from NDB and cached in NDBAPI
+ */
+ while((check = rs->nextResult(true)) == 0){
+ do {
+ if (rs->deleteTuple() != 0){
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+ deletedRows++;
+
+ /**
+ * nextResult(false) means that the records
+ * cached in the NDBAPI are modified before
+ * fetching more rows from NDB.
+ */
+ } while((check = rs->nextResult(false)) == 0);
+
+ /**
+ * Commit when all cached tuple have been marked for deletion
+ */
+ if(check != -1){
+ check = myTrans->execute(Commit);
+ myTrans->releaseCompletedOperations();
+ }
+ /**
+ * Check for errors
+ */
+ err = myTrans->getNdbError();
+ if(check == -1){
+ if(err.status == NdbError::TemporaryError){
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ milliSleep(50);
+ continue;
+ }
+ }
+ /**
+ * End of loop
+ */
+ }
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return 0;
+
+
+ }
+ if(myTrans!=0) {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ }
+ return -1;
+}
+
+
+int scan_update(Ndb* myNdb,
+ int parallelism,
+ int column_len,
+ int update_column,
+ const char * column_name,
+ const char * before_color,
+ const char * after_color)
+
+{
+
+ // Scan all records exclusive and update
+ // them one by one
+ int retryAttempt = 0;
+ const int retryMax = 10;
+ int updatedRows = 0;
+ int check;
+ NdbError err;
+ NdbConnection *myTrans;
+ NdbScanOperation *myScanOp;
+
+ /**
+ * Loop as long as :
+ * retryMax not reached
+ * failed operations due to TEMPORARY erros
+ *
+ * Exit loop;
+ * retyrMax reached
+ * Permanent error (return -1)
+ */
+ while (true)
+ {
+
+ if (retryAttempt >= retryMax)
+ {
+ std::cout << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << std::endl;
+ return -1;
+ }
+
+ myTrans = myNdb->startTransaction();
+ if (myTrans == NULL)
+ {
+ const NdbError err = myNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError)
+ {
+ milliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ std::cout << err.message << std::endl;
+ return -1;
+ }
+
+ /**
+ * Get a scan operation.
+ */
+ myScanOp = myTrans->getNdbScanOperation("GARAGE");
+ if (myScanOp == NULL)
+ {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * Define a result set for the scan.
+ */
+ NdbResultSet * rs = myScanOp->readTuplesExclusive(parallelism);
+ if( rs == 0 ) {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * Use NdbScanFilter to define a search critera
+ */
+ NdbScanFilter filter(myScanOp) ;
+ if(filter.begin(NdbScanFilter::AND) < 0 ||
+ filter.eq(update_column, before_color, column_len, false) <0||
+ filter.end() <0)
+ {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * Start scan (NoCommit since we are only reading at this stage);
+ */
+ if(myTrans->execute(NoCommit) != 0){
+ err = myTrans->getNdbError();
+ if(err.status == NdbError::TemporaryError){
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ milliSleep(50);
+ continue;
+ }
+ std::cout << myTrans->getNdbError().code << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * Define an update operation
+ */
+ NdbOperation * myUpdateOp;
+ /**
+ * start of loop: nextResult(true) means that "parallelism" number of
+ * rows are fetched from NDB and cached in NDBAPI
+ */
+ while((check = rs->nextResult(true)) == 0){
+ do {
+ /**
+ * Get update operation
+ */
+ myUpdateOp = rs->updateTuple();
+ if (myUpdateOp == 0){
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+ updatedRows++;
+ /**
+ * do the update
+ */
+ myUpdateOp->setValue(update_column,after_color);
+ /**
+ * nextResult(false) means that the records
+ * cached in the NDBAPI are modified before
+ * fetching more rows from NDB.
+ */
+ } while((check = rs->nextResult(false)) == 0);
+
+ /**
+ * Commit when all cached tuple have been updated
+ */
+ if(check != -1){
+ check = myTrans->execute(Commit);
+ myTrans->releaseCompletedOperations();
+ }
+ /**
+ * Check for errors
+ */
+ err = myTrans->getNdbError();
+ if(check == -1){
+ if(err.status == NdbError::TemporaryError){
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ milliSleep(50);
+ continue;
+ }
+ }
+ /**
+ * End of loop
+ */
+ }
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return 0;
+
+
+ }
+ if(myTrans!=0) {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ }
+ return -1;
+}
+
+
+
+int scan_print(Ndb * myNdb, int parallelism,
+ int column_len_brand,
+ int column_len_color)
+{
+// Scan all records exclusive and update
+ // them one by one
+ int retryAttempt = 0;
+ const int retryMax = 10;
+ int fetchedRows = 0;
+ int check;
+ NdbError err;
+ NdbConnection *myTrans;
+ NdbScanOperation *myScanOp;
+ /* Result of reading attribute value, three columns:
+ REG_NO, BRAND, and COLOR
+ */
+ NdbRecAttr * myRecAttr[3];
+
+ /**
+ * Loop as long as :
+ * retryMax not reached
+ * failed operations due to TEMPORARY erros
+ *
+ * Exit loop;
+ * retyrMax reached
+ * Permanent error (return -1)
+ */
+ while (true)
+ {
+
+ if (retryAttempt >= retryMax)
+ {
+ std::cout << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << std::endl;
+ return -1;
+ }
+
+ myTrans = myNdb->startTransaction();
+ if (myTrans == NULL)
+ {
+ const NdbError err = myNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError)
+ {
+ milliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ std::cout << err.message << std::endl;
+ return -1;
+ }
+ /*
+ * Define a scan operation.
+ * NDBAPI.
+ */
+ myScanOp = myTrans->getNdbScanOperation("GARAGE");
+ if (myScanOp == NULL)
+ {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * Define a result set for the scan.
+ */
+ NdbResultSet * rs = myScanOp->readTuplesExclusive(parallelism);
+ if( rs == 0 ) {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * Define storage for fetched attributes.
+ * E.g., the resulting attributes of executing
+ * myOp->getValue("REG_NO") is placed in myRecAttr[0].
+ * No data exists in myRecAttr until transaction has commited!
+ */
+ myRecAttr[0] = myScanOp->getValue("REG_NO");
+ myRecAttr[1] = myScanOp->getValue("BRAND");
+ myRecAttr[2] = myScanOp->getValue("COLOR");
+ if(myRecAttr[0] ==NULL || myRecAttr[1] == NULL || myRecAttr[2]==NULL)
+ {
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+ /**
+ * Start scan (NoCommit since we are only reading at this stage);
+ */
+ if(myTrans->execute(NoCommit) != 0){
+ err = myTrans->getNdbError();
+ if(err.status == NdbError::TemporaryError){
+ std::cout << myTrans->getNdbError().message << std::endl;
+ myNdb->closeTransaction(myTrans);
+ milliSleep(50);
+ continue;
+ }
+ std::cout << err.code << std::endl;
+ std::cout << myTrans->getNdbError().code << std::endl;
+ myNdb->closeTransaction(myTrans);
+ return -1;
+ }
+
+ /**
+ * start of loop: nextResult(true) means that "parallelism" number of
+ * rows are fetched from NDB and cached in NDBAPI
+ */
+ while((check = rs->nextResult(true)) == 0){
+ do {
+
+ fetchedRows++;
+ /**
+ * print REG_NO unsigned int
+ */
+ std::cout << myRecAttr[0]->u_32_value() << "\t";
+ char * buf_brand = new char[column_len_brand+1];
+ char * buf_color = new char[column_len_color+1];
+ /**
+ * print BRAND character string
+ */
+ memcpy(buf_brand, myRecAttr[1]->aRef(), column_len_brand);
+ buf_brand[column_len_brand] = 0;
+ std::cout << buf_brand << "\t";
+ delete [] buf_brand;
+ /**
+ * print COLOR character string
+ */
+ memcpy(buf_color, myRecAttr[2]->aRef(), column_len_color);
+ buf_brand[column_len_color] = 0;
+ std::cout << buf_color << std::endl;
+ delete [] buf_color;
+ /**
+ * nextResult(false) means that the records
+ * cached in the NDBAPI are modified before
+ * fetching more rows from NDB.
+ */
+ } while((check = rs->nextResult(false)) == 0);
+
+ }
+ myNdb->closeTransaction(myTrans);
+ return 1;
+ }
+ return -1;
+
+}
+
+
+int main()
+{
+ Ndb* myNdb = new Ndb( "TEST_DB" ); // Object representing the database
+
+
+
+ /*******************************************
+ * Initialize NDB and wait until its ready *
+ *******************************************/
+ if (myNdb->init(1024) == -1) { // Set max 1024 parallel transactions
+ APIERROR(myNdb->getNdbError());
+ exit(-1);
+ }
+
+ if (myNdb->waitUntilReady(30) != 0) {
+ std::cout << "NDB was not ready within 30 secs." << std::endl;
+ exit(-1);
+ }
+ create_table(myNdb);
+
+ NdbDictionary::Dictionary* myDict = myNdb->getDictionary();
+ int column_color = myDict->getTable("GARAGE")->getColumn("COLOR")->getColumnNo();
+ int column_len_color =
+ myDict->getTable("GARAGE")->getColumn("COLOR")->getLength();
+ int column_len_brand =
+ myDict->getTable("GARAGE")->getColumn("BRAND")->getLength();
+ int parallelism = 16;
+
+
+ if(populate(myNdb) > 0)
+ std::cout << "populate: Success!" << std::endl;
+
+ if(scan_print(myNdb, parallelism, column_len_brand, column_len_color) > 0)
+ std::cout << "scan_print: Success!" << std::endl << std::endl;
+
+ std::cout << "Going to delete all pink cars!" << std::endl;
+ if(scan_delete(myNdb, parallelism, column_color,
+ column_len_color, "Pink") > 0)
+ std::cout << "scan_delete: Success!" << std::endl << std::endl;
+
+ if(scan_print(myNdb, parallelism, column_len_brand, column_len_color) > 0)
+ std::cout << "scan_print: Success!" << std::endl << std::endl;
+
+ std::cout << "Going to update all blue cars to black cars!" << std::endl;
+ if(scan_update(myNdb, parallelism, column_len_color, column_color,
+ "COLOR", "Blue", "Black") > 0)
+ {
+ std::cout << "scan_update: Success!" << std::endl << std::endl;
+ }
+ if(scan_print(myNdb, parallelism, column_len_brand, column_len_color) > 0)
+ std::cout << "scan_print: Success!" << std::endl << std::endl;
+
+ delete myNdb;
+}
+
diff --git a/ndb/examples/ndbapi_scan_example/readme.txt b/ndb/examples/ndbapi_scan_example/readme.txt
new file mode 100644
index 00000000000..47cb4bf9ffa
--- /dev/null
+++ b/ndb/examples/ndbapi_scan_example/readme.txt
@@ -0,0 +1,3 @@
+1. Set NDB_OS in Makefile
+2. Add path to libNDB_API.so in LD_LIBRARY_PATH
+3. Set NDB_CONNECTSTRING
diff --git a/ndb/examples/select_all/Makefile b/ndb/examples/select_all/Makefile
new file mode 100644
index 00000000000..2bec205fa99
--- /dev/null
+++ b/ndb/examples/select_all/Makefile
@@ -0,0 +1,33 @@
+-include .defs.mk
+#NDB_OS = OS_YOU_ARE_RUNNING_ON
+#You need to set the NDB_OS variable here
+TARGET = select_all
+SRCS = select_all.cpp
+OBJS = select_all.o
+CXX = g++
+CFLAGS = -c -Wall -fno-rtti -fno-exceptions
+DEBUG =
+LFLAGS = -Wall
+INCLUDE_DIR = ../../include
+LIB_DIR = ../../lib
+ifeq ($(NDB_OS), SOLARIS)
+# Here is the definition of system libraries necessary for Solaris 7
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), LINUX)
+# Here is the definition of system libraries necessary for Linux 2.4
+SYS_LIB =
+endif
+ifeq ($(NDB_OS), MACOSX)
+# Here is the definition of system libraries necessary for Mac OS X
+SYS_LIB =
+endif
+
+$(TARGET): $(OBJS)
+ $(CXX) $(LFLAGS) -L$(LIB_DIR) $(OBJS) -lNDB_API $(SYS_LIB) -o $(TARGET)
+
+$(TARGET).o: $(SRCS)
+ $(CXX) $(CFLAGS) -I$(INCLUDE_DIR) -I$(INCLUDE_DIR)/ndbapi $(SRCS)
+
+clean:
+ rm -f *.o $(TARGET)
diff --git a/ndb/examples/select_all/select_all.cpp b/ndb/examples/select_all/select_all.cpp
new file mode 100644
index 00000000000..3cdbdc47e62
--- /dev/null
+++ b/ndb/examples/select_all/select_all.cpp
@@ -0,0 +1,258 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//
+// select_all.cpp: Prints all rows of a table
+//
+// Usage: select_all <table_name>+
+
+#include <NdbApi.hpp>
+
+// Used for cout
+#include <iostream>
+using namespace std;
+#include <stdio.h>
+#include <string.h>
+
+#define APIERROR(error) \
+ { cout << "Error in " << __FILE__ << ", line:" << __LINE__ << ", code:" \
+ << error.code << ", msg: " << error.message << "." << endl; \
+ exit(-1); }
+
+void usage(const char* prg) {
+ cout << "Usage: " << prg << " <table name>" << endl;
+ cout << "Prints all rows of table named <table name>" << endl;
+ exit(0);
+}
+
+/*****************************************************************************
+ *************************** Result Set Container ****************************
+ *****************************************************************************/
+
+/*
+ * Container of NdbRecAttr objects.
+ * (NdbRecAttr objects are database rows read by a scan operation.)
+ */
+class ResultSetContainer {
+public:
+ /**
+ * Initialize ResultSetContainer object for table named <tableName>
+ * - Allocates memory
+ * - Fetches attribute names from NDB Cluster
+ */
+ void init(NdbDictionary::Dictionary* dict, const char* tableName);
+
+ /**
+ * Get no of attributes for stored NdbRecAttr objects
+ */
+ int getNoOfAttributes() const;
+
+ /**
+ * Get NdbRecAttr object no i
+ */
+ NdbRecAttr* & getAttrStore(int i);
+
+ /**
+ * Get attribute name of attribute no i
+ */
+ const char* getAttrName(int i) const;
+
+ /**
+ * Print header of rows
+ */
+ void header() const;
+
+private:
+ int m_cols; // No of attributes for stored NdbRecAttr objects
+ char **m_names; // Names of attributes
+ NdbRecAttr **m_data; // The actual stored NdbRecAttr objects
+};
+
+void ResultSetContainer::init(NdbDictionary::Dictionary * dict,
+ const char* tableName)
+{
+ // Get Table object from NDB (this contains metadata about all tables)
+ const NdbDictionary::Table * tab = dict->getTable(tableName);
+
+ // Get table id of the table we are interested in
+ if (tab == 0) APIERROR(dict->getNdbError()); // E.g. table didn't exist
+
+ // Get no of attributes and allocate memory
+ m_cols = tab->getNoOfColumns();
+ m_names = new char* [m_cols];
+ m_data = new NdbRecAttr* [m_cols];
+
+ // Store all attribute names for the table
+ for (int i = 0; i < m_cols; i++) {
+ m_names[i] = new char[255];
+ snprintf(m_names[i], 255, "%s", tab->getColumn(i)->getName());
+ }
+}
+
+int ResultSetContainer::getNoOfAttributes() const {return m_cols;}
+NdbRecAttr*& ResultSetContainer::getAttrStore(int i) {return m_data[i];}
+const char* ResultSetContainer::getAttrName(int i) const {return m_names[i];}
+
+/*****************************************************************************
+ ********************************** MAIN ***********************************
+ *****************************************************************************/
+
+int main(int argc, const char** argv)
+{
+ Ndb* myNdb = new Ndb("ndbapi_example4"); // Object representing the database
+ NdbConnection* myNdbConnection; // For transactions
+ NdbOperation* myNdbOperation; // For operations
+ int check;
+
+ if (argc != 2) {
+ usage(argv[0]);
+ exit(0);
+ }
+ const char* tableName = argv[1];
+
+ /*******************************************
+ * Initialize NDB and wait until its ready *
+ *******************************************/
+ if (myNdb->init() == -1) {
+ APIERROR(myNdb->getNdbError());
+ exit(-1);
+ }
+
+ if (myNdb->waitUntilReady(30) != 0) {
+ cout << "NDB was not ready within 30 secs." << endl;
+ exit(-1);
+ }
+
+ /***************************
+ * Define and execute scan *
+ ***************************/
+ cout << "Select * from " << tableName << endl;
+
+ ResultSetContainer * container = new ResultSetContainer;
+ container->init(myNdb->getDictionary(), tableName);
+
+ myNdbConnection = myNdb->startTransaction();
+ if (myNdbConnection == NULL) APIERROR(myNdb->getNdbError());
+
+ myNdbOperation = myNdbConnection->getNdbOperation(tableName);
+ if (myNdbOperation == NULL) APIERROR(myNdbConnection->getNdbError());
+
+ // Define the operation to be an 'openScanRead' operation.
+ check = myNdbOperation->openScanRead(1);
+ if (check == -1) APIERROR(myNdbConnection->getNdbError());
+
+ // Set interpreted program to just be the single instruction
+ // 'interpret_exit_ok'. (This approves all rows of the table.)
+ if (myNdbOperation->interpret_exit_ok() == -1)
+ APIERROR(myNdbConnection->getNdbError());
+
+ // Get all attribute values of the row
+ for(int i = 0; i < container->getNoOfAttributes(); i++){
+ if((container->getAttrStore(i) =
+ myNdbOperation->getValue(container->getAttrName(i))) == 0)
+ APIERROR(myNdbConnection->getNdbError());
+ }
+
+ // Execute scan operation
+ check = myNdbConnection->executeScan();
+ if (check == -1) APIERROR(myNdbConnection->getNdbError());
+
+ /****************
+ * Print header *
+ ****************/
+ for (int i = 0; i < container->getNoOfAttributes(); i++)
+ cout << container->getAttrName(i) << "\t";
+
+ cout << endl;
+ for (int i = 0; i < container->getNoOfAttributes(); i++) {
+ for (int j = strlen(container->getAttrName(i)); j > 0; j--)
+ cout << "-";
+ cout << "\t";
+ }
+ cout << "\n";
+
+ /**************
+ * Scan table *
+ **************/
+ int eof;
+ int rows = 0;
+
+ // Print all rows of table
+ while ((eof = myNdbConnection->nextScanResult()) == 0) {
+ rows++;
+
+ for (int i = 0; i < container->getNoOfAttributes(); i++) {
+ if (container->getAttrStore(i)->isNULL()) {
+ cout << "NULL";
+ } else {
+
+ // Element size of value (No of bits per element in attribute value)
+ const int size = container->getAttrStore(i)->attrSize();
+
+ // No of elements in an array attribute (Is 1 if non-array attribute)
+ const int aSize = container->getAttrStore(i)->arraySize();
+
+ switch(container->getAttrStore(i)->attrType()){
+ case UnSigned:
+ switch(size) {
+ case 8: cout << container->getAttrStore(i)->u_64_value(); break;
+ case 4: cout << container->getAttrStore(i)->u_32_value(); break;
+ case 2: cout << container->getAttrStore(i)->u_short_value(); break;
+ case 1: cout << (unsigned) container->getAttrStore(i)->u_char_value();
+ break;
+ default: cout << "Unknown size" << endl;
+ }
+ break;
+
+ case Signed:
+ switch(size) {
+ case 8: cout << container->getAttrStore(i)->int64_value(); break;
+ case 4: cout << container->getAttrStore(i)->int32_value(); break;
+ case 2: cout << container->getAttrStore(i)->short_value(); break;
+ case 1: cout << (int) container->getAttrStore(i)->char_value(); break;
+ default: cout << "Unknown size" << endl;
+ }
+ break;
+
+ case String:
+ {
+ char* buf = new char[aSize+1];
+ memcpy(buf, container->getAttrStore(i)->aRef(), aSize);
+ buf[aSize] = 0;
+ cout << buf;
+ delete [] buf;
+ }
+ break;
+
+ case Float:
+ cout << container->getAttrStore(i)->float_value();
+ break;
+
+ default:
+ cout << "Unknown";
+ break;
+ }
+ }
+ cout << "\t";
+ }
+ cout << endl;
+ }
+ if (eof == -1) APIERROR(myNdbConnection->getNdbError());
+
+ myNdb->closeTransaction(myNdbConnection);
+
+ cout << "Selected " << rows << " rows." << endl;
+}
diff --git a/ndb/home/bin/Linuxmkisofs b/ndb/home/bin/Linuxmkisofs
new file mode 100755
index 00000000000..a531f4cca7b
--- /dev/null
+++ b/ndb/home/bin/Linuxmkisofs
Binary files differ
diff --git a/ndb/home/bin/Solarismkisofs b/ndb/home/bin/Solarismkisofs
new file mode 100755
index 00000000000..b239eaed6ad
--- /dev/null
+++ b/ndb/home/bin/Solarismkisofs
Binary files differ
diff --git a/ndb/home/bin/cvs2cl.pl b/ndb/home/bin/cvs2cl.pl
new file mode 100755
index 00000000000..9e6da5acf5b
--- /dev/null
+++ b/ndb/home/bin/cvs2cl.pl
@@ -0,0 +1,1865 @@
+#!/bin/sh
+exec perl -w -x $0 ${1+"$@"} # -*- mode: perl; perl-indent-level: 2; -*-
+#!perl -w
+
+##############################################################
+### ###
+### cvs2cl.pl: produce ChangeLog(s) from `cvs log` output. ###
+### ###
+##############################################################
+
+## $Revision: 2.38 $
+## $Date: 2001/02/12 19:54:35 $
+## $Author: kfogel $
+##
+## (C) 1999 Karl Fogel <kfogel@red-bean.com>, under the GNU GPL.
+##
+## (Extensively hacked on by Melissa O'Neill <oneill@cs.sfu.ca>.)
+##
+## cvs2cl.pl 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, or (at your option)
+## any later version.
+##
+## cvs2cl.pl 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 may have received a copy of the GNU General Public License
+## along with cvs2cl.pl; see the file COPYING. If not, write to the
+## Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+## Boston, MA 02111-1307, USA.
+
+
+
+use strict;
+use Text::Wrap;
+use Time::Local;
+use File::Basename;
+
+
+# The Plan:
+#
+# Read in the logs for multiple files, spit out a nice ChangeLog that
+# mirrors the information entered during `cvs commit'.
+#
+# The problem presents some challenges. In an ideal world, we could
+# detect files with the same author, log message, and checkin time --
+# each <filelist, author, time, logmessage> would be a changelog entry.
+# We'd sort them; and spit them out. Unfortunately, CVS is *not atomic*
+# so checkins can span a range of times. Also, the directory structure
+# could be hierarchical.
+#
+# Another question is whether we really want to have the ChangeLog
+# exactly reflect commits. An author could issue two related commits,
+# with different log entries, reflecting a single logical change to the
+# source. GNU style ChangeLogs group these under a single author/date.
+# We try to do the same.
+#
+# So, we parse the output of `cvs log', storing log messages in a
+# multilevel hash that stores the mapping:
+# directory => author => time => message => filelist
+# As we go, we notice "nearby" commit times and store them together
+# (i.e., under the same timestamp), so they appear in the same log
+# entry.
+#
+# When we've read all the logs, we twist this mapping into
+# a time => author => message => filelist mapping for each directory.
+#
+# If we're not using the `--distributed' flag, the directory is always
+# considered to be `./', even as descend into subdirectories.
+
+
+############### Globals ################
+
+
+# What we run to generate it:
+my $Log_Source_Command = "cvs log";
+
+# In case we have to print it out:
+my $VERSION = '$Revision: 2.38 $';
+$VERSION =~ s/\S+\s+(\S+)\s+\S+/$1/;
+
+## Vars set by options:
+
+# Print debugging messages?
+my $Debug = 0;
+
+# Just show version and exit?
+my $Print_Version = 0;
+
+# Just print usage message and exit?
+my $Print_Usage = 0;
+
+# Single top-level ChangeLog, or one per subdirectory?
+my $Distributed = 0;
+
+# What file should we generate (defaults to "ChangeLog")?
+my $Log_File_Name = "ChangeLog";
+
+# Grab most recent entry date from existing ChangeLog file, just add
+# to that ChangeLog.
+my $Cumulative = 0;
+
+# Expand usernames to email addresses based on a map file?
+my $User_Map_File = "";
+
+# Output to a file or to stdout?
+my $Output_To_Stdout = 0;
+
+# Eliminate empty log messages?
+my $Prune_Empty_Msgs = 0;
+
+# Don't call Text::Wrap on the body of the message
+my $No_Wrap = 0;
+
+# Separates header from log message. Code assumes it is either " " or
+# "\n\n", so if there's ever an option to set it to something else,
+# make sure to go through all conditionals that use this var.
+my $After_Header = " ";
+
+# Format more for programs than for humans.
+my $XML_Output = 0;
+
+# Do some special tweaks for log data that was written in FSF
+# ChangeLog style.
+my $FSF_Style = 0;
+
+# Show times in UTC instead of local time
+my $UTC_Times = 0;
+
+# Show day of week in output?
+my $Show_Day_Of_Week = 0;
+
+# Show revision numbers in output?
+my $Show_Revisions = 0;
+
+# Show tags (symbolic names) in output?
+my $Show_Tags = 0;
+
+# Show branches by symbolic name in output?
+my $Show_Branches = 0;
+
+# Show only revisions on these branches or their ancestors.
+my @Follow_Branches;
+
+# Don't bother with files matching this regexp.
+my @Ignore_Files;
+
+# How exactly we match entries. We definitely want "o",
+# and user might add "i" by using --case-insensitive option.
+my $Case_Insensitive = 0;
+
+# Maybe only show log messages matching a certain regular expression.
+my $Regexp_Gate = "";
+
+# Pass this global option string along to cvs, to the left of `log':
+my $Global_Opts = "";
+
+# Pass this option string along to the cvs log subcommand:
+my $Command_Opts = "";
+
+# Read log output from stdin instead of invoking cvs log?
+my $Input_From_Stdin = 0;
+
+# Don't show filenames in output.
+my $Hide_Filenames = 0;
+
+# Max checkin duration. CVS checkin is not atomic, so we may have checkin
+# times that span a range of time. We assume that checkins will last no
+# longer than $Max_Checkin_Duration seconds, and that similarly, no
+# checkins will happen from the same users with the same message less
+# than $Max_Checkin_Duration seconds apart.
+my $Max_Checkin_Duration = 180;
+
+# What to put at the front of [each] ChangeLog.
+my $ChangeLog_Header = "";
+
+## end vars set by options.
+
+# In 'cvs log' output, one long unbroken line of equal signs separates
+# files:
+my $file_separator = "======================================="
+ . "======================================";
+
+# In 'cvs log' output, a shorter line of dashes separates log messages
+# within a file:
+my $logmsg_separator = "----------------------------";
+
+
+############### End globals ############
+
+
+
+
+&parse_options ();
+&derive_change_log ();
+
+
+
+### Everything below is subroutine definitions. ###
+
+# If accumulating, grab the boundary date from pre-existing ChangeLog.
+sub maybe_grab_accumulation_date ()
+{
+ if (! $Cumulative) {
+ return "";
+ }
+
+ # else
+
+ open (LOG, "$Log_File_Name")
+ or die ("trouble opening $Log_File_Name for reading ($!)");
+
+ my $boundary_date;
+ while (<LOG>)
+ {
+ if (/^(\d\d\d\d-\d\d-\d\d\s+\d\d:\d\d)/)
+ {
+ $boundary_date = "$1";
+ last;
+ }
+ }
+
+ close (LOG);
+ return $boundary_date;
+}
+
+
+# Fills up a ChangeLog structure in the current directory.
+sub derive_change_log ()
+{
+ # See "The Plan" above for a full explanation.
+
+ my %grand_poobah;
+
+ my $file_full_path;
+ my $time;
+ my $revision;
+ my $author;
+ my $msg_txt;
+ my $detected_file_separator;
+
+ # Might be adding to an existing ChangeLog
+ my $accumulation_date = &maybe_grab_accumulation_date ();
+ if ($accumulation_date) {
+ $Log_Source_Command .= " -d\'>${accumulation_date}\'";
+ }
+
+ # We might be expanding usernames
+ my %usermap;
+
+ # In general, it's probably not very maintainable to use state
+ # variables like this to tell the loop what it's doing at any given
+ # moment, but this is only the first one, and if we never have more
+ # than a few of these, it's okay.
+ my $collecting_symbolic_names = 0;
+ my %symbolic_names; # Where tag names get stored.
+ my %branch_names; # We'll grab branch names while we're at it.
+ my %branch_numbers; # Save some revisions for @Follow_Branches
+ my @branch_roots; # For showing which files are branch ancestors.
+
+ # Bleargh. Compensate for a deficiency of custom wrapping.
+ if (($After_Header ne " ") and $FSF_Style)
+ {
+ $After_Header .= "\t";
+ }
+
+ if (! $Input_From_Stdin) {
+ open (LOG_SOURCE, "$Log_Source_Command |")
+ or die "unable to run \"${Log_Source_Command}\"";
+ }
+ else {
+ open (LOG_SOURCE, "-") or die "unable to open stdin for reading";
+ }
+
+ %usermap = &maybe_read_user_map_file ();
+
+ while (<LOG_SOURCE>)
+ {
+ # If on a new file and don't see filename, skip until we find it, and
+ # when we find it, grab it.
+ if ((! (defined $file_full_path)) and /^Working file: (.*)/)
+ {
+ $file_full_path = $1;
+ if (@Ignore_Files)
+ {
+ my $base;
+ ($base, undef, undef) = fileparse ($file_full_path);
+ # Ouch, I wish trailing operators in regexps could be
+ # evaluated on the fly!
+ if ($Case_Insensitive) {
+ if (grep ($file_full_path =~ m|$_|i, @Ignore_Files)) {
+ undef $file_full_path;
+ }
+ }
+ elsif (grep ($file_full_path =~ m|$_|, @Ignore_Files)) {
+ undef $file_full_path;
+ }
+ }
+ next;
+ }
+
+ # Just spin wheels if no file defined yet.
+ next if (! $file_full_path);
+
+ # Collect tag names in case we're asked to print them in the output.
+ if (/^symbolic names:$/) {
+ $collecting_symbolic_names = 1;
+ next; # There's no more info on this line, so skip to next
+ }
+ if ($collecting_symbolic_names)
+ {
+ # All tag names are listed with whitespace in front in cvs log
+ # output; so if see non-whitespace, then we're done collecting.
+ if (/^\S/) {
+ $collecting_symbolic_names = 0;
+ }
+ else # we're looking at a tag name, so parse & store it
+ {
+ # According to the Cederqvist manual, in node "Tags", tag
+ # names must start with an uppercase or lowercase letter and
+ # can contain uppercase and lowercase letters, digits, `-',
+ # and `_'. However, it's not our place to enforce that, so
+ # we'll allow anything CVS hands us to be a tag:
+ /^\s+([^:]+): ([\d.]+)$/;
+ my $tag_name = $1;
+ my $tag_rev = $2;
+
+ # A branch number either has an odd number of digit sections
+ # (and hence an even number of dots), or has ".0." as the
+ # second-to-last digit section. Test for these conditions.
+ my $real_branch_rev = "";
+ if (($tag_rev =~ /^(\d+\.\d+\.)+\d+$/) # Even number of dots...
+ and (! ($tag_rev =~ /^(1\.)+1$/))) # ...but not "1.[1.]1"
+ {
+ $real_branch_rev = $tag_rev;
+ }
+ elsif ($tag_rev =~ /(\d+\.(\d+\.)+)0.(\d+)/) # Has ".0."
+ {
+ $real_branch_rev = $1 . $3;
+ }
+ # If we got a branch, record its number.
+ if ($real_branch_rev)
+ {
+ $branch_names{$real_branch_rev} = $tag_name;
+ if (@Follow_Branches) {
+ if (grep ($_ eq $tag_name, @Follow_Branches)) {
+ $branch_numbers{$tag_name} = $real_branch_rev;
+ }
+ }
+ }
+ else {
+ # Else it's just a regular (non-branch) tag.
+ push (@{$symbolic_names{$tag_rev}}, $tag_name);
+ }
+ }
+ }
+ # End of code for collecting tag names.
+
+ # If have file name, but not revision, and see revision, then grab
+ # it. (We collect unconditionally, even though we may or may not
+ # ever use it.)
+ if ((! (defined $revision)) and (/^revision (\d+\.[\d.]+)/))
+ {
+ $revision = $1;
+
+ if (@Follow_Branches)
+ {
+ foreach my $branch (@Follow_Branches)
+ {
+ # Special case for following trunk revisions
+ if (($branch =~ /^trunk$/i) and ($revision =~ /^[0-9]+\.[0-9]+$/))
+ {
+ goto dengo;
+ }
+
+ my $branch_number = $branch_numbers{$branch};
+ if ($branch_number)
+ {
+ # Are we on one of the follow branches or an ancestor of
+ # same?
+ #
+ # If this revision is a prefix of the branch number, or
+ # possibly is less in the minormost number, OR if this
+ # branch number is a prefix of the revision, then yes.
+ # Otherwise, no.
+ #
+ # So below, we determine if any of those conditions are
+ # met.
+
+ # Trivial case: is this revision on the branch?
+ # (Compare this way to avoid regexps that screw up Emacs
+ # indentation, argh.)
+ if ((substr ($revision, 0, ((length ($branch_number)) + 1)))
+ eq ($branch_number . "."))
+ {
+ goto dengo;
+ }
+ # Non-trivial case: check if rev is ancestral to branch
+ elsif ((length ($branch_number)) > (length ($revision)))
+ {
+ $revision =~ /^((?:\d+\.)+)(\d+)$/;
+ my $r_left = $1; # still has the trailing "."
+ my $r_end = $2;
+
+ $branch_number =~ /^((?:\d+\.)+)(\d+)\.\d+$/;
+ my $b_left = $1; # still has trailing "."
+ my $b_mid = $2; # has no trailing "."
+
+ if (($r_left eq $b_left)
+ && ($r_end <= $b_mid))
+ {
+ goto dengo;
+ }
+ }
+ }
+ }
+ }
+ else # (! @Follow_Branches)
+ {
+ next;
+ }
+
+ # Else we are following branches, but this revision isn't on the
+ # path. So skip it.
+ undef $revision;
+ dengo:
+ next;
+ }
+
+ # If we don't have a revision right now, we couldn't possibly
+ # be looking at anything useful.
+ if (! (defined ($revision))) {
+ $detected_file_separator = /^$file_separator$/o;
+ if ($detected_file_separator) {
+ # No revisions for this file; can happen, e.g. "cvs log -d DATE"
+ goto CLEAR;
+ }
+ else {
+ next;
+ }
+ }
+
+ # If have file name but not date and author, and see date or
+ # author, then grab them:
+ unless (defined $time)
+ {
+ if (/^date: .*/)
+ {
+ ($time, $author) = &parse_date_and_author ($_);
+ if (defined ($usermap{$author}) and $usermap{$author}) {
+ $author = $usermap{$author};
+ }
+ }
+ else {
+ $detected_file_separator = /^$file_separator$/o;
+ if ($detected_file_separator) {
+ # No revisions for this file; can happen, e.g. "cvs log -d DATE"
+ goto CLEAR;
+ }
+ }
+ # If the date/time/author hasn't been found yet, we couldn't
+ # possibly care about anything we see. So skip:
+ next;
+ }
+
+ # A "branches: ..." line here indicates that one or more branches
+ # are rooted at this revision. If we're showing branches, then we
+ # want to show that fact as well, so we collect all the branches
+ # that this is the latest ancestor of and store them in
+ # @branch_roots. Just for reference, the format of the line we're
+ # seeing at this point is:
+ #
+ # branches: 1.5.2; 1.5.4; ...;
+ #
+ # Okay, here goes:
+
+ if (/^branches:\s+(.*);$/)
+ {
+ if ($Show_Branches)
+ {
+ my $lst = $1;
+ $lst =~ s/(1\.)+1;|(1\.)+1$//; # ignore the trivial branch 1.1.1
+ if ($lst) {
+ @branch_roots = split (/;\s+/, $lst);
+ }
+ else {
+ undef @branch_roots;
+ }
+ next;
+ }
+ else
+ {
+ # Ugh. This really bothers me. Suppose we see a log entry
+ # like this:
+ #
+ # ----------------------------
+ # revision 1.1
+ # date: 1999/10/17 03:07:38; author: jrandom; state: Exp;
+ # branches: 1.1.2;
+ # Intended first line of log message begins here.
+ # ----------------------------
+ #
+ # The question is, how we can tell the difference between that
+ # log message and a *two*-line log message whose first line is
+ #
+ # "branches: 1.1.2;"
+ #
+ # See the problem? The output of "cvs log" is inherently
+ # ambiguous.
+ #
+ # For now, we punt: we liberally assume that people don't
+ # write log messages like that, and just toss a "branches:"
+ # line if we see it but are not showing branches. I hope no
+ # one ever loses real log data because of this.
+ next;
+ }
+ }
+
+ # If have file name, time, and author, then we're just grabbing
+ # log message texts:
+ $detected_file_separator = /^$file_separator$/o;
+ if ($detected_file_separator && ! (defined $revision)) {
+ # No revisions for this file; can happen, e.g. "cvs log -d DATE"
+ goto CLEAR;
+ }
+ unless ($detected_file_separator || /^$logmsg_separator$/o)
+ {
+ $msg_txt .= $_; # Normally, just accumulate the message...
+ next;
+ }
+ # ... until a msg separator is encountered:
+ # Ensure the message contains something:
+ if ((! $msg_txt)
+ || ($msg_txt =~ /^\s*\.\s*$|^\s*$/)
+ || ($msg_txt =~ /\*\*\* empty log message \*\*\*/))
+ {
+ if ($Prune_Empty_Msgs) {
+ goto CLEAR;
+ }
+ # else
+ $msg_txt = "[no log message]\n";
+ }
+
+ ### Store it all in the Grand Poobah:
+ {
+ my $dir_key; # key into %grand_poobah
+ my %qunk; # complicated little jobbie, see below
+
+ # Each revision of a file has a little data structure (a `qunk')
+ # associated with it. That data structure holds not only the
+ # file's name, but any additional information about the file
+ # that might be needed in the output, such as the revision
+ # number, tags, branches, etc. The reason to have these things
+ # arranged in a data structure, instead of just appending them
+ # textually to the file's name, is that we may want to do a
+ # little rearranging later as we write the output. For example,
+ # all the files on a given tag/branch will go together, followed
+ # by the tag in parentheses (so trunk or otherwise non-tagged
+ # files would go at the end of the file list for a given log
+ # message). This rearrangement is a lot easier to do if we
+ # don't have to reparse the text.
+ #
+ # A qunk looks like this:
+ #
+ # {
+ # filename => "hello.c",
+ # revision => "1.4.3.2",
+ # time => a timegm() return value (moment of commit)
+ # tags => [ "tag1", "tag2", ... ],
+ # branch => "branchname" # There should be only one, right?
+ # branchroots => [ "branchtag1", "branchtag2", ... ]
+ # }
+
+ if ($Distributed) {
+ # Just the basename, don't include the path.
+ ($qunk{'filename'}, $dir_key, undef) = fileparse ($file_full_path);
+ }
+ else {
+ $dir_key = "./";
+ $qunk{'filename'} = $file_full_path;
+ }
+
+ # This may someday be used in a more sophisticated calculation
+ # of what other files are involved in this commit. For now, we
+ # don't use it, because the common-commit-detection algorithm is
+ # hypothesized to be "good enough" as it stands.
+ $qunk{'time'} = $time;
+
+ # We might be including revision numbers and/or tags and/or
+ # branch names in the output. Most of the code from here to
+ # loop-end deals with organizing these in qunk.
+
+ $qunk{'revision'} = $revision;
+
+ # Grab the branch, even though we may or may not need it:
+ $qunk{'revision'} =~ /((?:\d+\.)+)\d+/;
+ my $branch_prefix = $1;
+ $branch_prefix =~ s/\.$//; # strip off final dot
+ if ($branch_names{$branch_prefix}) {
+ $qunk{'branch'} = $branch_names{$branch_prefix};
+ }
+
+ # If there's anything in the @branch_roots array, then this
+ # revision is the root of at least one branch. We'll display
+ # them as branch names instead of revision numbers, the
+ # substitution for which is done directly in the array:
+ if (@branch_roots) {
+ my @roots = map { $branch_names{$_} } @branch_roots;
+ $qunk{'branchroots'} = \@roots;
+ }
+
+ # Save tags too.
+ if (defined ($symbolic_names{$revision})) {
+ $qunk{'tags'} = $symbolic_names{$revision};
+ delete $symbolic_names{$revision};
+ }
+
+ # Add this file to the list
+ # (We use many spoonfuls of autovivication magic. Hashes and arrays
+ # will spring into existence if they aren't there already.)
+
+ &debug ("(pushing log msg for ${dir_key}$qunk{'filename'})\n");
+
+ # Store with the files in this commit. Later we'll loop through
+ # again, making sure that revisions with the same log message
+ # and nearby commit times are grouped together as one commit.
+ push (@{$grand_poobah{$dir_key}{$author}{$time}{$msg_txt}}, \%qunk);
+ }
+
+ CLEAR:
+ # Make way for the next message
+ undef $msg_txt;
+ undef $time;
+ undef $revision;
+ undef $author;
+ undef @branch_roots;
+
+ # Maybe even make way for the next file:
+ if ($detected_file_separator) {
+ undef $file_full_path;
+ undef %branch_names;
+ undef %branch_numbers;
+ undef %symbolic_names;
+ }
+ }
+
+ close (LOG_SOURCE);
+
+ ### Process each ChangeLog
+
+ while (my ($dir,$authorhash) = each %grand_poobah)
+ {
+ &debug ("DOING DIR: $dir\n");
+
+ # Here we twist our hash around, from being
+ # author => time => message => filelist
+ # in %$authorhash to
+ # time => author => message => filelist
+ # in %changelog.
+ #
+ # This is also where we merge entries. The algorithm proceeds
+ # through the timeline of the changelog with a sliding window of
+ # $Max_Checkin_Duration seconds; within that window, entries that
+ # have the same log message are merged.
+ #
+ # (To save space, we zap %$authorhash after we've copied
+ # everything out of it.)
+
+ my %changelog;
+ while (my ($author,$timehash) = each %$authorhash)
+ {
+ my $lasttime;
+ my %stamptime;
+ foreach my $time (sort {$main::a <=> $main::b} (keys %$timehash))
+ {
+ my $msghash = $timehash->{$time};
+ while (my ($msg,$qunklist) = each %$msghash)
+ {
+ my $stamptime = $stamptime{$msg};
+ if ((defined $stamptime)
+ and (($time - $stamptime) < $Max_Checkin_Duration)
+ and (defined $changelog{$stamptime}{$author}{$msg}))
+ {
+ push(@{$changelog{$stamptime}{$author}{$msg}}, @$qunklist);
+ }
+ else {
+ $changelog{$time}{$author}{$msg} = $qunklist;
+ $stamptime{$msg} = $time;
+ }
+ }
+ }
+ }
+ undef (%$authorhash);
+
+ ### Now we can write out the ChangeLog!
+
+ my ($logfile_here, $logfile_bak, $tmpfile);
+
+ if (! $Output_To_Stdout) {
+ $logfile_here = $dir . $Log_File_Name;
+ $logfile_here =~ s/^\.\/\//\//; # fix any leading ".//" problem
+ $tmpfile = "${logfile_here}.cvs2cl$$.tmp";
+ $logfile_bak = "${logfile_here}.bak";
+
+ open (LOG_OUT, ">$tmpfile") or die "Unable to open \"$tmpfile\"";
+ }
+ else {
+ open (LOG_OUT, ">-") or die "Unable to open stdout for writing";
+ }
+
+ print LOG_OUT $ChangeLog_Header;
+
+ if ($XML_Output) {
+ print LOG_OUT "<?xml version=\"1.0\"?>\n\n"
+ . "<changelog xmlns=\"http://www.red-bean.com/xmlns/cvs2cl/\">\n\n";
+ }
+
+ foreach my $time (sort {$main::b <=> $main::a} (keys %changelog))
+ {
+ my $authorhash = $changelog{$time};
+ while (my ($author,$mesghash) = each %$authorhash)
+ {
+ # If XML, escape in outer loop to avoid compound quoting:
+ if ($XML_Output) {
+ $author = &xml_escape ($author);
+ }
+
+ while (my ($msg,$qunklist) = each %$mesghash)
+ {
+ my $files = &pretty_file_list ($qunklist);
+ my $header_line; # date and author
+ my $body; # see below
+ my $wholething; # $header_line + $body
+
+ # Set up the date/author line.
+ # kff todo: do some more XML munging here, on the header
+ # part of the entry:
+ my ($ignore,$min,$hour,$mday,$mon,$year,$wday)
+ = $UTC_Times ? gmtime($time) : localtime($time);
+
+ # XML output includes everything else, we might as well make
+ # it always include Day Of Week too, for consistency.
+ if ($Show_Day_Of_Week or $XML_Output) {
+ $wday = ("Sunday", "Monday", "Tuesday", "Wednesday",
+ "Thursday", "Friday", "Saturday")[$wday];
+ $wday = ($XML_Output) ? "<weekday>${wday}</weekday>\n" : " $wday";
+ }
+ else {
+ $wday = "";
+ }
+
+ if ($XML_Output) {
+ $header_line =
+ sprintf ("<date>%4u-%02u-%02u</date>\n"
+ . "${wday}"
+ . "<time>%02u:%02u</time>\n"
+ . "<author>%s</author>\n",
+ $year+1900, $mon+1, $mday, $hour, $min, $author);
+ }
+ else {
+ $header_line =
+ sprintf ("%4u-%02u-%02u${wday} %02u:%02u %s\n\n",
+ $year+1900, $mon+1, $mday, $hour, $min, $author);
+ }
+
+ # Reshape the body according to user preferences.
+ if ($XML_Output)
+ {
+ $msg = &preprocess_msg_text ($msg);
+ $body = $files . $msg;
+ }
+ elsif ($No_Wrap)
+ {
+ $msg = &preprocess_msg_text ($msg);
+ $files = wrap ("\t", " ", "$files");
+ $msg =~ s/\n(.*)/\n\t$1/g;
+ unless ($After_Header eq " ") {
+ $msg =~ s/^(.*)/\t$1/g;
+ }
+ $body = $files . $After_Header . $msg;
+ }
+ else # do wrapping, either FSF-style or regular
+ {
+ if ($FSF_Style)
+ {
+ $files = wrap ("\t", " ", "$files");
+
+ my $files_last_line_len = 0;
+ if ($After_Header eq " ")
+ {
+ $files_last_line_len = &last_line_len ($files);
+ $files_last_line_len += 1; # for $After_Header
+ }
+
+ $msg = &wrap_log_entry
+ ($msg, "\t", 69 - $files_last_line_len, 69);
+ $body = $files . $After_Header . $msg;
+ }
+ else # not FSF-style
+ {
+ $msg = &preprocess_msg_text ($msg);
+ $body = $files . $After_Header . $msg;
+ $body = wrap ("\t", " ", "$body");
+ }
+ }
+
+ $wholething = $header_line . $body;
+
+ if ($XML_Output) {
+ $wholething = "<entry>\n${wholething}</entry>\n";
+ }
+
+ # One last check: make sure it passes the regexp test, if the
+ # user asked for that. We have to do it here, so that the
+ # test can match against information in the header as well
+ # as in the text of the log message.
+
+ # How annoying to duplicate so much code just because I
+ # can't figure out a way to evaluate scalars on the trailing
+ # operator portion of a regular expression. Grrr.
+ if ($Case_Insensitive) {
+ unless ($Regexp_Gate && ($wholething !~ /$Regexp_Gate/oi)) {
+ print LOG_OUT "${wholething}\n";
+ }
+ }
+ else {
+ unless ($Regexp_Gate && ($wholething !~ /$Regexp_Gate/o)) {
+ print LOG_OUT "${wholething}\n";
+ }
+ }
+ }
+ }
+ }
+
+ if ($XML_Output) {
+ print LOG_OUT "</changelog>\n";
+ }
+
+ close (LOG_OUT);
+
+ if (! $Output_To_Stdout)
+ {
+ # If accumulating, append old data to new before renaming. But
+ # don't append the most recent entry, since it's already in the
+ # new log due to CVS's idiosyncratic interpretation of "log -d".
+ if ($Cumulative && -f $logfile_here)
+ {
+ open (NEW_LOG, ">>$tmpfile")
+ or die "trouble appending to $tmpfile ($!)";
+
+ open (OLD_LOG, "<$logfile_here")
+ or die "trouble reading from $logfile_here ($!)";
+
+ my $started_first_entry = 0;
+ my $passed_first_entry = 0;
+ while (<OLD_LOG>)
+ {
+ if (! $passed_first_entry)
+ {
+ if ((! $started_first_entry)
+ && /^(\d\d\d\d-\d\d-\d\d\s+\d\d:\d\d)/) {
+ $started_first_entry = 1;
+ }
+ elsif (/^(\d\d\d\d-\d\d-\d\d\s+\d\d:\d\d)/) {
+ $passed_first_entry = 1;
+ print NEW_LOG $_;
+ }
+ }
+ else {
+ print NEW_LOG $_;
+ }
+ }
+
+ close (NEW_LOG);
+ close (OLD_LOG);
+ }
+
+ if (-f $logfile_here) {
+ rename ($logfile_here, $logfile_bak);
+ }
+ rename ($tmpfile, $logfile_here);
+ }
+ }
+}
+
+
+sub parse_date_and_author ()
+{
+ # Parses the date/time and author out of a line like:
+ #
+ # date: 1999/02/19 23:29:05; author: apharris; state: Exp;
+
+ my $line = shift;
+
+ my ($year, $mon, $mday, $hours, $min, $secs, $author) = $line =~
+ m#(\d+)/(\d+)/(\d+)\s+(\d+):(\d+):(\d+);\s+author:\s+([^;]+);#
+ or die "Couldn't parse date ``$line''";
+ die "Bad date or Y2K issues" unless ($year > 1969 and $year < 2258);
+ # Kinda arbitrary, but useful as a sanity check
+ my $time = timegm($secs,$min,$hours,$mday,$mon-1,$year-1900);
+
+ return ($time, $author);
+}
+
+
+# Here we take a bunch of qunks and convert them into printed
+# summary that will include all the information the user asked for.
+sub pretty_file_list ()
+{
+ if ($Hide_Filenames and (! $XML_Output)) {
+ return "";
+ }
+
+ my $qunksref = shift;
+ my @qunkrefs = @$qunksref;
+ my @filenames;
+ my $beauty = ""; # The accumulating header string for this entry.
+ my %non_unanimous_tags; # Tags found in a proper subset of qunks
+ my %unanimous_tags; # Tags found in all qunks
+ my %all_branches; # Branches found in any qunk
+ my $common_dir = undef; # Dir prefix common to all files ("" if none)
+ my $fbegun = 0; # Did we begin printing filenames yet?
+
+ # First, loop over the qunks gathering all the tag/branch names.
+ # We'll put them all in non_unanimous_tags, and take out the
+ # unanimous ones later.
+ foreach my $qunkref (@qunkrefs)
+ {
+ # Keep track of whether all the files in this commit were in the
+ # same directory, and memorize it if so. We can make the output a
+ # little more compact by mentioning the directory only once.
+ if ((scalar (@qunkrefs)) > 1)
+ {
+ if (! (defined ($common_dir)))
+ {
+ my ($base, $dir);
+ ($base, $dir, undef) = fileparse ($$qunkref{'filename'});
+
+ if ((! (defined ($dir))) # this first case is sheer paranoia
+ or ($dir eq "")
+ or ($dir eq "./")
+ or ($dir eq ".\\"))
+ {
+ $common_dir = "";
+ }
+ else
+ {
+ $common_dir = $dir;
+ }
+ }
+ elsif ($common_dir ne "")
+ {
+ # Already have a common dir prefix, so how much of it can we preserve?
+ $common_dir = &common_path_prefix ($$qunkref{'filename'}, $common_dir);
+ }
+ }
+ else # only one file in this entry anyway, so common dir not an issue
+ {
+ $common_dir = "";
+ }
+
+ if (defined ($$qunkref{'branch'})) {
+ $all_branches{$$qunkref{'branch'}} = 1;
+ }
+ if (defined ($$qunkref{'tags'})) {
+ foreach my $tag (@{$$qunkref{'tags'}}) {
+ $non_unanimous_tags{$tag} = 1;
+ }
+ }
+ }
+
+ # Any tag held by all qunks will be printed specially... but only if
+ # there are multiple qunks in the first place!
+ if ((scalar (@qunkrefs)) > 1) {
+ foreach my $tag (keys (%non_unanimous_tags)) {
+ my $everyone_has_this_tag = 1;
+ foreach my $qunkref (@qunkrefs) {
+ if ((! (defined ($$qunkref{'tags'})))
+ or (! (grep ($_ eq $tag, @{$$qunkref{'tags'}})))) {
+ $everyone_has_this_tag = 0;
+ }
+ }
+ if ($everyone_has_this_tag) {
+ $unanimous_tags{$tag} = 1;
+ delete $non_unanimous_tags{$tag};
+ }
+ }
+ }
+
+ if ($XML_Output)
+ {
+ # If outputting XML, then our task is pretty simple, because we
+ # don't have to detect common dir, common tags, branch prefixing,
+ # etc. We just output exactly what we have, and don't worry about
+ # redundancy or readability.
+
+ foreach my $qunkref (@qunkrefs)
+ {
+ my $filename = $$qunkref{'filename'};
+ my $revision = $$qunkref{'revision'};
+ my $tags = $$qunkref{'tags'};
+ my $branch = $$qunkref{'branch'};
+ my $branchroots = $$qunkref{'branchroots'};
+
+ $filename = &xml_escape ($filename); # probably paranoia
+ $revision = &xml_escape ($revision); # definitely paranoia
+
+ $beauty .= "<file>\n";
+ $beauty .= "<name>${filename}</name>\n";
+ $beauty .= "<revision>${revision}</revision>\n";
+ if ($branch) {
+ $branch = &xml_escape ($branch); # more paranoia
+ $beauty .= "<branch>${branch}</branch>\n";
+ }
+ foreach my $tag (@$tags) {
+ $tag = &xml_escape ($tag); # by now you're used to the paranoia
+ $beauty .= "<tag>${tag}</tag>\n";
+ }
+ foreach my $root (@$branchroots) {
+ $root = &xml_escape ($root); # which is good, because it will continue
+ $beauty .= "<branchroot>${root}</branchroot>\n";
+ }
+ $beauty .= "</file>\n";
+ }
+
+ # Theoretically, we could go home now. But as long as we're here,
+ # let's print out the common_dir and utags, as a convenience to
+ # the receiver (after all, earlier code calculated that stuff
+ # anyway, so we might as well take advantage of it).
+
+ if ((scalar (keys (%unanimous_tags))) > 1) {
+ foreach my $utag ((keys (%unanimous_tags))) {
+ $utag = &xml_escape ($utag); # the usual paranoia
+ $beauty .= "<utag>${utag}</utag>\n";
+ }
+ }
+ if ($common_dir) {
+ $common_dir = &xml_escape ($common_dir);
+ $beauty .= "<commondir>${common_dir}</commondir>\n";
+ }
+
+ # That's enough for XML, time to go home:
+ return $beauty;
+ }
+
+ # Else not XML output, so complexly compactify for chordate
+ # consumption. At this point we have enough global information
+ # about all the qunks to organize them non-redundantly for output.
+
+ if ($common_dir) {
+ # Note that $common_dir still has its trailing slash
+ $beauty .= "$common_dir: ";
+ }
+
+ if ($Show_Branches)
+ {
+ # For trailing revision numbers.
+ my @brevisions;
+
+ foreach my $branch (keys (%all_branches))
+ {
+ foreach my $qunkref (@qunkrefs)
+ {
+ if ((defined ($$qunkref{'branch'}))
+ and ($$qunkref{'branch'} eq $branch))
+ {
+ if ($fbegun) {
+ # kff todo: comma-delimited in XML too? Sure.
+ $beauty .= ", ";
+ }
+ else {
+ $fbegun = 1;
+ }
+ my $fname = substr ($$qunkref{'filename'}, length ($common_dir));
+ $beauty .= $fname;
+ $$qunkref{'printed'} = 1; # Just setting a mark bit, basically
+
+ if ($Show_Tags && (defined @{$$qunkref{'tags'}})) {
+ my @tags = grep ($non_unanimous_tags{$_}, @{$$qunkref{'tags'}});
+ if (@tags) {
+ $beauty .= " (tags: ";
+ $beauty .= join (', ', @tags);
+ $beauty .= ")";
+ }
+ }
+
+ if ($Show_Revisions) {
+ # Collect the revision numbers' last components, but don't
+ # print them -- they'll get printed with the branch name
+ # later.
+ $$qunkref{'revision'} =~ /.+\.([\d]+)$/;
+ push (@brevisions, $1);
+
+ # todo: we're still collecting branch roots, but we're not
+ # showing them anywhere. If we do show them, it would be
+ # nifty to just call them revision "0" on a the branch.
+ # Yeah, that's the ticket.
+ }
+ }
+ }
+ $beauty .= " ($branch";
+ if (@brevisions) {
+ if ((scalar (@brevisions)) > 1) {
+ $beauty .= ".[";
+ $beauty .= (join (',', @brevisions));
+ $beauty .= "]";
+ }
+ else {
+ $beauty .= ".$brevisions[0]";
+ }
+ }
+ $beauty .= ")";
+ }
+ }
+
+ # Okay; any qunks that were done according to branch are taken care
+ # of, and marked as printed. Now print everyone else.
+
+ foreach my $qunkref (@qunkrefs)
+ {
+ next if (defined ($$qunkref{'printed'})); # skip if already printed
+
+ if ($fbegun) {
+ $beauty .= ", ";
+ }
+ else {
+ $fbegun = 1;
+ }
+ $beauty .= substr ($$qunkref{'filename'}, length ($common_dir));
+ # todo: Shlomo's change was this:
+ # $beauty .= substr ($$qunkref{'filename'},
+ # (($common_dir eq "./") ? "" : length ($common_dir)));
+ $$qunkref{'printed'} = 1; # Set a mark bit.
+
+ if ($Show_Revisions || $Show_Tags)
+ {
+ my $started_addendum = 0;
+
+ if ($Show_Revisions) {
+ $started_addendum = 1;
+ $beauty .= " (";
+ $beauty .= "$$qunkref{'revision'}";
+ }
+ if ($Show_Tags && (defined $$qunkref{'tags'})) {
+ my @tags = grep ($non_unanimous_tags{$_}, @{$$qunkref{'tags'}});
+ if ((scalar (@tags)) > 0) {
+ if ($started_addendum) {
+ $beauty .= ", ";
+ }
+ else {
+ $beauty .= " (tags: ";
+ }
+ $beauty .= join (', ', @tags);
+ $started_addendum = 1;
+ }
+ }
+ if ($started_addendum) {
+ $beauty .= ")";
+ }
+ }
+ }
+
+ # Unanimous tags always come last.
+ if ($Show_Tags && %unanimous_tags)
+ {
+ $beauty .= " (utags: ";
+ $beauty .= join (', ', keys (%unanimous_tags));
+ $beauty .= ")";
+ }
+
+ # todo: still have to take care of branch_roots?
+
+ $beauty = "* $beauty:";
+
+ return $beauty;
+}
+
+
+sub common_path_prefix ()
+{
+ my $path1 = shift;
+ my $path2 = shift;
+
+ my ($dir1, $dir2);
+ (undef, $dir1, undef) = fileparse ($path1);
+ (undef, $dir2, undef) = fileparse ($path2);
+
+ # Transmogrify Windows filenames to look like Unix.
+ # (It is far more likely that someone is running cvs2cl.pl under
+ # Windows than that they would genuinely have backslashes in their
+ # filenames.)
+ $dir1 =~ tr#\\#/#;
+ $dir2 =~ tr#\\#/#;
+
+ my $accum1 = "";
+ my $accum2 = "";
+ my $last_common_prefix = "";
+
+ while ($accum1 eq $accum2)
+ {
+ $last_common_prefix = $accum1;
+ last if ($accum1 eq $dir1);
+ my ($tmp1) = split (/\//, (substr ($dir1, length ($accum1))));
+ my ($tmp2) = split (/\//, (substr ($dir2, length ($accum2))));
+ $accum1 .= "$tmp1/" if ((defined ($tmp1)) and $tmp1);
+ $accum2 .= "$tmp2/" if ((defined ($tmp2)) and $tmp2);
+ }
+
+ return $last_common_prefix;
+}
+
+
+sub preprocess_msg_text ()
+{
+ my $text = shift;
+
+ # Strip out carriage returns (as they probably result from DOSsy editors).
+ $text =~ s/\r\n/\n/g;
+
+ # If it *looks* like two newlines, make it *be* two newlines:
+ $text =~ s/\n\s*\n/\n\n/g;
+
+ if ($XML_Output)
+ {
+ $text = &xml_escape ($text);
+ $text = "<msg>${text}</msg>\n";
+ }
+ elsif (! $No_Wrap)
+ {
+ # Strip off lone newlines, but only for lines that don't begin with
+ # whitespace or a mail-quoting character, since we want to preserve
+ # that kind of formatting. Also don't strip newlines that follow a
+ # period; we handle those specially next. And don't strip
+ # newlines that precede an open paren.
+ 1 while ($text =~ s/(^|\n)([^>\s].*[^.\n])\n([^>\n])/$1$2 $3/g);
+
+ # If a newline follows a period, make sure that when we bring up the
+ # bottom sentence, it begins with two spaces.
+ 1 while ($text =~ s/(^|\n)([^>\s].*)\n([^>\n])/$1$2 $3/g);
+ }
+
+ return $text;
+}
+
+
+sub last_line_len ()
+{
+ my $files_list = shift;
+ my @lines = split (/\n/, $files_list);
+ my $last_line = pop (@lines);
+ return length ($last_line);
+}
+
+
+# A custom wrap function, sensitive to some common constructs used in
+# log entries.
+sub wrap_log_entry ()
+{
+ my $text = shift; # The text to wrap.
+ my $left_pad_str = shift; # String to pad with on the left.
+
+ # These do NOT take left_pad_str into account:
+ my $length_remaining = shift; # Amount left on current line.
+ my $max_line_length = shift; # Amount left for a blank line.
+
+ my $wrapped_text = ""; # The accumulating wrapped entry.
+ my $user_indent = ""; # Inherited user_indent from prev line.
+
+ my $first_time = 1; # First iteration of the loop?
+ my $suppress_line_start_match = 0; # Set to disable line start checks.
+
+ my @lines = split (/\n/, $text);
+ while (@lines) # Don't use `foreach' here, it won't work.
+ {
+ my $this_line = shift (@lines);
+ chomp $this_line;
+
+ if ($this_line =~ /^(\s+)/) {
+ $user_indent = $1;
+ }
+ else {
+ $user_indent = "";
+ }
+
+ # If it matches any of the line-start regexps, print a newline now...
+ if ($suppress_line_start_match)
+ {
+ $suppress_line_start_match = 0;
+ }
+ elsif (($this_line =~ /^(\s*)\*\s+[a-zA-Z0-9]/)
+ || ($this_line =~ /^(\s*)\* [a-zA-Z0-9_\.\/\+-]+/)
+ || ($this_line =~ /^(\s*)\([a-zA-Z0-9_\.\/\+-]+(\)|,\s*)/)
+ || ($this_line =~ /^(\s+)(\S+)/)
+ || ($this_line =~ /^(\s*)- +/)
+ || ($this_line =~ /^()\s*$/)
+ || ($this_line =~ /^(\s*)\*\) +/)
+ || ($this_line =~ /^(\s*)[a-zA-Z0-9](\)|\.|\:) +/))
+ {
+ # Make a line break immediately, unless header separator is set
+ # and this line is the first line in the entry, in which case
+ # we're getting the blank line for free already and shouldn't
+ # add an extra one.
+ unless (($After_Header ne " ") and ($first_time))
+ {
+ if ($this_line =~ /^()\s*$/) {
+ $suppress_line_start_match = 1;
+ $wrapped_text .= "\n${left_pad_str}";
+ }
+
+ $wrapped_text .= "\n${left_pad_str}";
+ }
+
+ $length_remaining = $max_line_length - (length ($user_indent));
+ }
+
+ # Now that any user_indent has been preserved, strip off leading
+ # whitespace, so up-folding has no ugly side-effects.
+ $this_line =~ s/^\s*//;
+
+ # Accumulate the line, and adjust parameters for next line.
+ my $this_len = length ($this_line);
+ if ($this_len == 0)
+ {
+ # Blank lines should cancel any user_indent level.
+ $user_indent = "";
+ $length_remaining = $max_line_length;
+ }
+ elsif ($this_len >= $length_remaining) # Line too long, try breaking it.
+ {
+ # Walk backwards from the end. At first acceptable spot, break
+ # a new line.
+ my $idx = $length_remaining - 1;
+ if ($idx < 0) { $idx = 0 };
+ while ($idx > 0)
+ {
+ if (substr ($this_line, $idx, 1) =~ /\s/)
+ {
+ my $line_now = substr ($this_line, 0, $idx);
+ my $next_line = substr ($this_line, $idx);
+ $this_line = $line_now;
+
+ # Clean whitespace off the end.
+ chomp $this_line;
+
+ # The current line is ready to be printed.
+ $this_line .= "\n${left_pad_str}";
+
+ # Make sure the next line is allowed full room.
+ $length_remaining = $max_line_length - (length ($user_indent));
+
+ # Strip next_line, but then preserve any user_indent.
+ $next_line =~ s/^\s*//;
+
+ # Sneak a peek at the user_indent of the upcoming line, so
+ # $next_line (which will now precede it) can inherit that
+ # indent level. Otherwise, use whatever user_indent level
+ # we currently have, which might be none.
+ my $next_next_line = shift (@lines);
+ if ((defined ($next_next_line)) && ($next_next_line =~ /^(\s+)/)) {
+ $next_line = $1 . $next_line if (defined ($1));
+ # $length_remaining = $max_line_length - (length ($1));
+ $next_next_line =~ s/^\s*//;
+ }
+ else {
+ $next_line = $user_indent . $next_line;
+ }
+ if (defined ($next_next_line)) {
+ unshift (@lines, $next_next_line);
+ }
+ unshift (@lines, $next_line);
+
+ # Our new next line might, coincidentally, begin with one of
+ # the line-start regexps, so we temporarily turn off
+ # sensitivity to that until we're past the line.
+ $suppress_line_start_match = 1;
+
+ last;
+ }
+ else
+ {
+ $idx--;
+ }
+ }
+
+ if ($idx == 0)
+ {
+ # We bottomed out because the line is longer than the
+ # available space. But that could be because the space is
+ # small, or because the line is longer than even the maximum
+ # possible space. Handle both cases below.
+
+ if ($length_remaining == ($max_line_length - (length ($user_indent))))
+ {
+ # The line is simply too long -- there is no hope of ever
+ # breaking it nicely, so just insert it verbatim, with
+ # appropriate padding.
+ $this_line = "\n${left_pad_str}${this_line}";
+ }
+ else
+ {
+ # Can't break it here, but may be able to on the next round...
+ unshift (@lines, $this_line);
+ $length_remaining = $max_line_length - (length ($user_indent));
+ $this_line = "\n${left_pad_str}";
+ }
+ }
+ }
+ else # $this_len < $length_remaining, so tack on what we can.
+ {
+ # Leave a note for the next iteration.
+ $length_remaining = $length_remaining - $this_len;
+
+ if ($this_line =~ /\.$/)
+ {
+ $this_line .= " ";
+ $length_remaining -= 2;
+ }
+ else # not a sentence end
+ {
+ $this_line .= " ";
+ $length_remaining -= 1;
+ }
+ }
+
+ # Unconditionally indicate that loop has run at least once.
+ $first_time = 0;
+
+ $wrapped_text .= "${user_indent}${this_line}";
+ }
+
+ # One last bit of padding.
+ $wrapped_text .= "\n";
+
+ return $wrapped_text;
+}
+
+
+sub xml_escape ()
+{
+ my $txt = shift;
+ $txt =~ s/&/&amp;/g;
+ $txt =~ s/</&lt;/g;
+ $txt =~ s/>/&gt;/g;
+ return $txt;
+}
+
+
+sub maybe_read_user_map_file ()
+{
+ my %expansions;
+
+ if ($User_Map_File)
+ {
+ open (MAPFILE, "<$User_Map_File")
+ or die ("Unable to open $User_Map_File ($!)");
+
+ while (<MAPFILE>)
+ {
+ next if /^\s*#/; # Skip comment lines.
+ next if not /:/; # Skip lines without colons.
+
+ # It is now safe to split on ':'.
+ my ($username, $expansion) = split ':';
+ chomp $expansion;
+ $expansion =~ s/^'(.*)'$/$1/;
+ $expansion =~ s/^"(.*)"$/$1/;
+
+ # If it looks like the expansion has a real name already, then
+ # we toss the username we got from CVS log. Otherwise, keep
+ # it to use in combination with the email address.
+
+ if ($expansion =~ /^\s*<{0,1}\S+@.*/) {
+ # Also, add angle brackets if none present
+ if (! ($expansion =~ /<\S+@\S+>/)) {
+ $expansions{$username} = "$username <$expansion>";
+ }
+ else {
+ $expansions{$username} = "$username $expansion";
+ }
+ }
+ else {
+ $expansions{$username} = $expansion;
+ }
+ }
+
+ close (MAPFILE);
+ }
+
+ return %expansions;
+}
+
+
+sub parse_options ()
+{
+ # Check this internally before setting the global variable.
+ my $output_file;
+
+ # If this gets set, we encountered unknown options and will exit at
+ # the end of this subroutine.
+ my $exit_with_admonishment = 0;
+
+ while (my $arg = shift (@ARGV))
+ {
+ if ($arg =~ /^-h$|^-help$|^--help$|^--usage$|^-?$/) {
+ $Print_Usage = 1;
+ }
+ elsif ($arg =~ /^--debug$/) { # unadvertised option, heh
+ $Debug = 1;
+ }
+ elsif ($arg =~ /^--version$/) {
+ $Print_Version = 1;
+ }
+ elsif ($arg =~ /^-g$|^--global-opts$/) {
+ my $narg = shift (@ARGV) || die "$arg needs argument.\n";
+ # Don't assume CVS is called "cvs" on the user's system:
+ $Log_Source_Command =~ s/(^\S*)/$1 $narg/;
+ }
+ elsif ($arg =~ /^-l$|^--log-opts$/) {
+ my $narg = shift (@ARGV) || die "$arg needs argument.\n";
+ $Log_Source_Command .= " $narg";
+ }
+ elsif ($arg =~ /^-f$|^--file$/) {
+ my $narg = shift (@ARGV) || die "$arg needs argument.\n";
+ $output_file = $narg;
+ }
+ elsif ($arg =~ /^--accum$/) {
+ $Cumulative = 1;
+ }
+ elsif ($arg =~ /^--fsf$/) {
+ $FSF_Style = 1;
+ }
+ elsif ($arg =~ /^-U$|^--usermap$/) {
+ my $narg = shift (@ARGV) || die "$arg needs argument.\n";
+ $User_Map_File = $narg;
+ }
+ elsif ($arg =~ /^-W$|^--window$/) {
+ my $narg = shift (@ARGV) || die "$arg needs argument.\n";
+ $Max_Checkin_Duration = $narg;
+ }
+ elsif ($arg =~ /^-I$|^--ignore$/) {
+ my $narg = shift (@ARGV) || die "$arg needs argument.\n";
+ push (@Ignore_Files, $narg);
+ }
+ elsif ($arg =~ /^-C$|^--case-insensitive$/) {
+ $Case_Insensitive = 1;
+ }
+ elsif ($arg =~ /^-R$|^--regexp$/) {
+ my $narg = shift (@ARGV) || die "$arg needs argument.\n";
+ $Regexp_Gate = $narg;
+ }
+ elsif ($arg =~ /^--stdout$/) {
+ $Output_To_Stdout = 1;
+ }
+ elsif ($arg =~ /^--version$/) {
+ $Print_Version = 1;
+ }
+ elsif ($arg =~ /^-d$|^--distributed$/) {
+ $Distributed = 1;
+ }
+ elsif ($arg =~ /^-P$|^--prune$/) {
+ $Prune_Empty_Msgs = 1;
+ }
+ elsif ($arg =~ /^-S$|^--separate-header$/) {
+ $After_Header = "\n\n";
+ }
+ elsif ($arg =~ /^--no-wrap$/) {
+ $No_Wrap = 1;
+ }
+ elsif ($arg =~ /^--gmt$|^--utc$/) {
+ $UTC_Times = 1;
+ }
+ elsif ($arg =~ /^-w$|^--day-of-week$/) {
+ $Show_Day_Of_Week = 1;
+ }
+ elsif ($arg =~ /^-r$|^--revisions$/) {
+ $Show_Revisions = 1;
+ }
+ elsif ($arg =~ /^-t$|^--tags$/) {
+ $Show_Tags = 1;
+ }
+ elsif ($arg =~ /^-b$|^--branches$/) {
+ $Show_Branches = 1;
+ }
+ elsif ($arg =~ /^-F$|^--follow$/) {
+ my $narg = shift (@ARGV) || die "$arg needs argument.\n";
+ push (@Follow_Branches, $narg);
+ }
+ elsif ($arg =~ /^--stdin$/) {
+ $Input_From_Stdin = 1;
+ }
+ elsif ($arg =~ /^--header$/) {
+ my $narg = shift (@ARGV) || die "$arg needs argument.\n";
+ $ChangeLog_Header = &slurp_file ($narg);
+ if (! defined ($ChangeLog_Header)) {
+ $ChangeLog_Header = "";
+ }
+ }
+ elsif ($arg =~ /^--xml$/) {
+ $XML_Output = 1;
+ }
+ elsif ($arg =~ /^--hide-filenames$/) {
+ $Hide_Filenames = 1;
+ $After_Header = "";
+ }
+ else {
+ # Just add a filename as argument to the log command
+ $Log_Source_Command .= " $arg";
+ }
+ }
+
+ ## Check for contradictions...
+
+ if ($Output_To_Stdout && $Distributed) {
+ print STDERR "cannot pass both --stdout and --distributed\n";
+ $exit_with_admonishment = 1;
+ }
+
+ if ($Output_To_Stdout && $output_file) {
+ print STDERR "cannot pass both --stdout and --file\n";
+ $exit_with_admonishment = 1;
+ }
+
+ if ($XML_Output && $Cumulative) {
+ print STDERR "cannot pass both --xml and --accum\n";
+ $exit_with_admonishment = 1;
+ }
+
+ # Or if any other error message has already been printed out, we
+ # just leave now:
+ if ($exit_with_admonishment) {
+ &usage ();
+ exit (1);
+ }
+ elsif ($Print_Usage) {
+ &usage ();
+ exit (0);
+ }
+ elsif ($Print_Version) {
+ &version ();
+ exit (0);
+ }
+
+ ## Else no problems, so proceed.
+
+ if ($output_file) {
+ $Log_File_Name = $output_file;
+ }
+}
+
+
+sub slurp_file ()
+{
+ my $filename = shift || die ("no filename passed to slurp_file()");
+ my $retstr;
+
+ open (SLURPEE, "<${filename}") or die ("unable to open $filename ($!)");
+ my $saved_sep = $/;
+ undef $/;
+ $retstr = <SLURPEE>;
+ $/ = $saved_sep;
+ close (SLURPEE);
+ return $retstr;
+}
+
+
+sub debug ()
+{
+ if ($Debug) {
+ my $msg = shift;
+ print STDERR $msg;
+ }
+}
+
+
+sub version ()
+{
+ print "cvs2cl.pl version ${VERSION}; distributed under the GNU GPL.\n";
+}
+
+
+sub usage ()
+{
+ &version ();
+ print <<'END_OF_INFO';
+Generate GNU-style ChangeLogs in CVS working copies.
+
+Notes about the output format(s):
+
+ The default output of cvs2cl.pl is designed to be compact, formally
+ unambiguous, but still easy for humans to read. It is largely
+ self-explanatory, I hope; the one abbreviation that might not be
+ obvious is "utags". That stands for "universal tags" -- a
+ universal tag is one held by all the files in a given change entry.
+
+ If you need output that's easy for a program to parse, use the
+ --xml option. Note that with XML output, just about all available
+ information is included with each change entry, whether you asked
+ for it or not, on the theory that your parser can ignore anything
+ it's not looking for.
+
+Notes about the options and arguments (the actual options are listed
+last in this usage message):
+
+ * The -I and -F options may appear multiple times.
+
+ * To follow trunk revisions, use "-F trunk" ("-F TRUNK" also works).
+ This is okay because no would ever, ever be crazy enough to name a
+ branch "trunk", right? Right.
+
+ * For the -U option, the UFILE should be formatted like
+ CVSROOT/users. That is, each line of UFILE looks like this
+ jrandom:jrandom@red-bean.com
+ or maybe even like this
+ jrandom:'Jesse Q. Random <jrandom@red-bean.com>'
+ Don't forget to quote the portion after the colon if necessary.
+
+ * Many people want to filter by date. To do so, invoke cvs2cl.pl
+ like this:
+ cvs2cl.pl -l "-d'DATESPEC'"
+ where DATESPEC is any date specification valid for "cvs log -d".
+ (Note that CVS 1.10.7 and below requires there be no space between
+ -d and its argument).
+
+Options/Arguments:
+
+ -h, -help, --help, or -? Show this usage and exit
+ --version Show version and exit
+ -r, --revisions Show revision numbers in output
+ -b, --branches Show branch names in revisions when possible
+ -t, --tags Show tags (symbolic names) in output
+ --stdin Read from stdin, don't run cvs log
+ --stdout Output to stdout not to ChangeLog
+ -d, --distributed Put ChangeLogs in subdirs
+ -f FILE, --file FILE Write to FILE instead of "ChangeLog"
+ --fsf Use this if log data is in FSF ChangeLog style
+ -W SECS, --window SECS Window of time within which log entries unify
+ -U UFILE, --usermap UFILE Expand usernames to email addresses from UFILE
+ -R REGEXP, --regexp REGEXP Include only entries that match REGEXP
+ -I REGEXP, --ignore REGEXP Ignore files whose names match REGEXP
+ -C, --case-insensitive Any regexp matching is done case-insensitively
+ -F BRANCH, --follow BRANCH Show only revisions on or ancestral to BRANCH
+ -S, --separate-header Blank line between each header and log message
+ --no-wrap Don't auto-wrap log message (recommend -S also)
+ --gmt, --utc Show times in GMT/UTC instead of local time
+ --accum Add to an existing ChangeLog (incompat w/ --xml)
+ -w, --day-of-week Show day of week
+ --header FILE Get ChangeLog header from FILE ("-" means stdin)
+ --xml Output XML instead of ChangeLog format
+ --hide-filenames Don't show filenames (ignored for XML output)
+ -P, --prune Don't show empty log messages
+ -g OPTS, --global-opts OPTS Invoke like this "cvs OPTS log ..."
+ -l OPTS, --log-opts OPTS Invoke like this "cvs ... log OPTS"
+ FILE1 [FILE2 ...] Show only log information for the named FILE(s)
+
+See http://www.red-bean.com/cvs2cl for maintenance and bug info.
+END_OF_INFO
+}
+
+__END__
+
+=head1 NAME
+
+cvs2cl.pl - produces GNU-style ChangeLogs in CVS working copies, by
+ running "cvs log" and parsing the output. Shared log entries are
+ unified in an intuitive way.
+
+=head1 DESCRIPTION
+
+This script generates GNU-style ChangeLog files from CVS log
+information. Basic usage: just run it inside a working copy and a
+ChangeLog will appear. It requires repository access (i.e., 'cvs log'
+must work). Run "cvs2cl.pl --help" to see more advanced options.
+
+See http://www.red-bean.com/cvs2cl for updates, and for instructions
+on getting anonymous CVS access to this script.
+
+Maintainer: Karl Fogel <kfogel@red-bean.com>
+Please report bugs to <bug-cvs2cl@red-bean.com>.
+
+=head1 README
+
+This script generates GNU-style ChangeLog files from CVS log
+information. Basic usage: just run it inside a working copy and a
+ChangeLog will appear. It requires repository access (i.e., 'cvs log'
+must work). Run "cvs2cl.pl --help" to see more advanced options.
+
+See http://www.red-bean.com/cvs2cl for updates, and for instructions
+on getting anonymous CVS access to this script.
+
+Maintainer: Karl Fogel <kfogel@red-bean.com>
+Please report bugs to <bug-cvs2cl@red-bean.com>.
+
+=head1 PREREQUISITES
+
+This script requires C<Text::Wrap>, C<Time::Local>, and
+C<File::Basename>.
+It also seems to require C<Perl 5.004_04> or higher.
+
+=pod OSNAMES
+
+any
+
+=pod SCRIPT CATEGORIES
+
+Version_Control/CVS
+
+=cut
+
+
+-*- -*- -*- -*- -*- -*- -*- -*- -*- -*- -*- -*- -*- -*- -*- -*- -*- -*-
+
+Note about a bug-slash-opportunity:
+-----------------------------------
+
+There's a bug in Text::Wrap, which affects cvs2cl. This script
+reveals it:
+
+ #!/usr/bin/perl -w
+
+ use Text::Wrap;
+
+ my $test_text =
+ "This script demonstrates a bug in Text::Wrap. The very long line
+ following this paragraph will be relocated relative to the surrounding
+ text:
+
+ ====================================================================
+
+ See? When the bug happens, we'll get the line of equal signs below
+ this paragraph, even though it should be above.";
+
+
+ # Print out the test text with no wrapping:
+ print "$test_text";
+ print "\n";
+ print "\n";
+
+ # Now print it out wrapped, and see the bug:
+ print wrap ("\t", " ", "$test_text");
+ print "\n";
+ print "\n";
+
+If the line of equal signs were one shorter, then the bug doesn't
+happen. Interesting.
+
+Anyway, rather than fix this in Text::Wrap, we might as well write a
+new wrap() which has the following much-needed features:
+
+* initial indentation, like current Text::Wrap()
+* subsequent line indentation, like current Text::Wrap()
+* user chooses among: force-break long words, leave them alone, or die()?
+* preserve existing indentation: chopped chunks from an indented line
+ are indented by same (like this line, not counting the asterisk!)
+* optional list of things to preserve on line starts, default ">"
+
+Note that the last two are essentially the same concept, so unify in
+implementation and give a good interface to controlling them.
+
+And how about:
+
+Optionally, when encounter a line pre-indented by same as previous
+line, then strip the newline and refill, but indent by the same.
+Yeah...
diff --git a/ndb/home/bin/cvschk b/ndb/home/bin/cvschk
new file mode 100755
index 00000000000..4510cc30888
--- /dev/null
+++ b/ndb/home/bin/cvschk
@@ -0,0 +1,569 @@
+#!/usr/bin/perl -w
+#
+# cvschk -- fast offline check for new files and modifications of files
+
+# cvschk : A perl program which checks the status of the CVS controlled
+# files and gives an ASCII table sorted after the status of files.
+#
+# If you have used CVS, then you know that it is hard to
+# get a good overview the CVS-status of the files in you
+# directories. Any new files? Any files changes?
+# cvschk will help the programmer get the overview in the
+# situation, where we do not have access to the CVS repository.
+#
+# Note that the program does only local checks of the files
+# If you have fast access to the CVS repositiory, then consider
+# the cvsstat-program - which additionally can tell if other
+# people have made newer versions of the files.
+#
+# The program requires Perl 5.004 (maybe previous versions also work).
+#
+# It is tuned to parse the output of cvs(1) version 1.9.
+# Earlier and later versions may require modifications to the script.
+#
+# ** Note that the first line might be wrong depending **
+# ** on the location of your perl program. **
+#
+# Sample output:
+# The directory ./mytempdir is not under CVS control
+#
+# Changed files
+# ---------------
+# ./cvs2html
+# ./cvschk
+# ./cvsstat
+#
+# New files
+# ---------------
+# ./.#cvschk
+# ./XX
+# ./cvs2html.ok
+#
+# Deleted files
+# ---------------
+# (none)
+
+# Changelog:
+#
+# Ver Date Author Changelog
+# --- ---------- -------------------- -------------------------------------
+# 1.12 2002-01-04 Michael Kohne Fixed a $foo=<> warning for
+# 5.004_01 with defined($foo=<>)
+# Added a --tabular|-t switch
+#
+# 1.11 2001-12-27 Michael Kohne Added cvsignore functionality
+# Handling of 'dummy timestamp'
+# Handling of 'Result of Merge'
+#
+# 1.10 2001-11-06 Michael Kohne Added -r and -l options
+#
+# 1.9 2001-08-03 Lars G. T. Jørgensen Hack to allow special entry-line
+#
+# 1.8 2001-06-07 Peter Toft Back to the same as 1.6
+# CVS is my friend
+#
+# 1.7 2001-06-04 Peter Toft Peter was very tired and
+# applied a wrong patch -
+# version 1.7 is crap
+#
+# 1.6 2000-12-17 Peter Toft Better description added
+#
+# 1.5 2000-11-04 Peter Toft URL of cvsstat changed
+#
+# 1.4 2000-09-20 Peter Toft Must show deleted files also
+# as the default
+#
+# 1.3 2000-08-08 Ole Tange and Initial version
+# Peter Toft
+# ---- ---------- -------------------- -------------------------------------
+#
+# -----------------------------------------------------------------------------
+#
+# This program is protected by the GPL, and all modifications of
+# general interest should be emailed to the maintainer (pto@sslug.dk).
+#
+# This program also uses code parts from cvsstat
+# (same homepage as cvschk)
+#
+# Copyright 2000,2001 by Peter Toft <pto@sslug.dk> and Ole Tange <ole@tange.dk>
+# as well as
+# Lars G. T. Jørgensen <larsj@diku.dk>
+#
+# The URL of the home page of cvschk is shown below.
+
+
+use Time::Local;
+use strict;
+use Getopt::Long;
+
+my $startdir = ".";
+
+my $debug = 0;
+my (%files,%filesok,%seen,%skip);
+
+
+# Michael Kohne 12/16/01
+#
+# Simulation of .cvsignore as CVS does it...
+#
+# using .cvsignore handling makes cvschk take from 2 to 3 times
+# longer to run over the same set of files.
+# in my tests, disabling cvsignore altogether, cvschk takes .2
+# seconds on my working directory. Adding cvsignore,takes
+# .4 seconds.
+# Note that I do not use individual .cvsignore files - if there
+# are a lot of them in your directory tree, it will add run time
+#
+# variables used for .cvsignore handling
+my $initcvsignoreregex;# regex holding all startup cvsignore pattersn (no ())
+my $cvsignoreregex;# one regex holding all current cvsignore patterns
+my $disable_cvsignore=0;# set to 1 to disable cvsignore emulation
+ # (available in case it's REALLY screwed up)
+my $disable_ind_cvsignore=0;# set to 1 to disable finding .cvsignore files
+ # in each directory.
+my $debug_cvsignore = 0; # For debugging .cvsignore problems
+
+my %mon;
+@mon{qw(Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec)}=
+ 0..11; # Perl months are 0 .. 11
+
+my ($version) = ('$Revision: 1.12 $ ' =~ /^\$\w+: (.*) \$ $/);
+my $URL = "http://cvs.sslug.dk/cvs2html";
+my $version_line = "cvschk version $version (see $URL)\n";
+
+my $opt_all;
+my $restrict;
+my $local;
+my $tabular;
+
+my $opt_restrict;
+
+sub show_version {print $version_line}
+
+sub die_version {die $version_line}
+
+sub die_usage {
+ my $bundled = ($] > 5.00399
+ ? "can be bundled"
+ : "can't be bundled, because your Perl is too old");
+ die <<END_OF_USAGE; # Help in the style of GNU `ls --help' or `make --help'
+Usage: $0 [OPTION]...
+ Show the CVS status of FILEs (the current directory by default),
+ traversing directories recursively and telling if new files exist
+ in the repository.
+Options:
+ -a, --all Show all statistics, including the names of files that
+ are up to date, used tags, ignored patterns and more
+ -r, --restrict Don't show the names of the unknown files
+ (useful if you have many temporary files)
+ -t, --tabular Show one file per line, each preceeded with a status word,
+ Sorted by filename.
+ -l, --local Don't descend into sub-directories
+ -d, --debug Debug info
+ -h, --help Show this help end exit immediately
+ -V, --version Show the version line and exit immediately
+The one-letter options $bundled.
+END_OF_USAGE
+}
+
+sub die_help {show_version; die_usage}
+
+# Let `-ar' mean `-a -r' and require `--all' (or -a) instead of `-all'.
+if ($] > 5.00399) { # This requires 5.004, so silently skip it for older Perls.
+ eval {Getopt::Long::config("bundling")}; # avoid 5.003 compilation error
+ warn $@ if $@; # For Perl 5.004+ we do want to see any compilation error
+}
+
+
+GetOptions( "all|a" => \$opt_all,
+ "tabular|t" => \$tabular,
+ "restrict|r" => \$restrict,
+ "local|l" => \$local,
+ "help|h" => \&die_help,
+ "debug|d" => \$debug,
+ "version|V" => \&die_version,
+ ) or die_usage;
+
+sub cvs_changed_in_dir($); #define prototype (for recursion)
+
+# functions for .cvsignore handling
+
+# converts a given filename pattern
+# (of the sort that sh(1) takes) to
+# a perl regex of similar meaning.
+#
+# It works by doing the following:
+#
+# change:
+# . to \.
+# $ to \$
+# * to .*
+# ? to .
+#
+sub fpat_to_regex($)
+{
+ my $fexp;
+ $fexp = shift;
+ $fexp =~ s/\./\\\./g;#change . to \.
+ $fexp =~ s/\$/\\\$/g;#change dollar sign to \dollar sign
+ $fexp =~ s/\*/.*/g;# change * to .*
+ $fexp =~ s/\?/./g; # change ? to .
+ return $fexp;
+}
+
+# copy the input list to one single regex,
+# items seperated by | symbols.
+# return the regex string
+sub do_regex_convert
+{
+ my $rx = "";
+ my $first = 1;#true for first element only
+
+
+ # convert each element of cvsignore into a regex
+ # this makes the patterns usable in perl
+ my $cp;
+ foreach $cp (@_) {
+ if (not $first) { $rx = $rx . "|"; }
+ if ($first) { $first = 0; }
+ $rx = $rx . fpat_to_regex($cp);
+ }
+
+ return $rx;
+}
+
+# first parameter is a reference to the array
+# to be loaded
+# the rest of the parameters are just items
+# that need to be loaded into the array.
+# Note that if a ! is found, the list is
+# emptied, then further items are added.
+# returns true if a ! was found
+sub load_list_from_list
+{
+ my $arref = shift;# get reference to array from front
+ my $item;
+ my $ret=0;#false means no ! found
+
+ chomp @_;#kill newlines
+ foreach $item (@_) {
+ $item =~ s/^\s*(.*?)\s*$/$1/;#kill leading/trailing whitespace
+ if ($item) { # empty string is false
+ push @$arref,$item;
+ }
+ if ($item eq "!") {
+ @$arref = ();# '!' causes list to clear
+ $ret = 1;# ! found
+ }
+ }
+
+ return $ret;
+}
+
+# loads the given list with lines from the
+# specified file. Note that if a '!' is found
+# all prior patterns are removed from the list
+# before the following patterns are loaded
+# first param is the filename,
+# second param is a reference to an array
+# that the data is to go into
+# returns true if a ! was found
+sub load_list_from_file
+{
+ my @inlist;
+ my $fname = shift;#filename to read from
+ #if (not -e $fname) { return; }
+ my $arref = shift;#array to store into
+ open CVSIGNORE,"$fname" or return;#file might not exist, that's OK
+ @inlist = <CVSIGNORE>;
+ close CVSIGNORE;
+ return load_list_from_list($arref,@inlist);
+}
+
+# loads $cvsignoreregex from
+# $initcvsignoreregex and the .cvsignore file
+# in the local directory
+sub load_cvsignore
+{
+ if ($disable_ind_cvsignore) {return;}#don't look for local .cvsignore files
+ if ($disable_cvsignore) {return;}#don't do anything
+
+ my $dir = shift;
+ my @cvsignore;
+
+ # bang will be true if a ! was found. In such cases, I need
+ # to not use the pre-exisitng regex list.
+ my $bang = load_list_from_file("$dir/.cvsignore",\@cvsignore);
+
+ # if we get a local cvsignore list, then...
+ my $rx = do_regex_convert(@cvsignore);
+ if ($rx) {
+ $cvsignoreregex = "(";
+ if (not $bang) {$cvsignoreregex = $cvsignoreregex . $initcvsignoreregex . "|";}
+ $cvsignoreregex = $cvsignoreregex . $rx . ")";
+ } else {
+ if ($bang) {$cvsignoreregex = "";}
+ else {$cvsignoreregex = "(" . $initcvsignoreregex . ")";}
+ }
+
+ if ($debug_cvsignore) {print $dir,":",$cvsignoreregex, "\n";}
+}
+
+
+# loads all of the cvsignore patterns that
+# can be loaded at script startup
+sub load_initial_cvsignore()
+{
+ #load the default patterns
+ # (taken from http://www.gnu.org/manual/cvs-1.9/html_node/cvs_141.html#IDX399)
+ #
+ # this gives you the patterns that cvs normally starts with
+ my @initcvsignore;
+ push @initcvsignore,("RCS");
+ push @initcvsignore,("SCCS");
+ push @initcvsignore,("CVS");
+ push @initcvsignore,("CVS.adm");
+ push @initcvsignore,("RCSLOG");
+ push @initcvsignore,("cvslog.*");
+ push @initcvsignore,("tags");
+ push @initcvsignore,("TAGS");
+ push @initcvsignore,(".make.state");
+ push @initcvsignore,(".nse_depinfo");
+ push @initcvsignore,("*~");
+ push @initcvsignore,("\#*");
+ push @initcvsignore,(".\#*");
+ push @initcvsignore,("\,*");
+ push @initcvsignore,("_\$\*");
+ push @initcvsignore,("*\$");
+ push @initcvsignore,("*.old");
+ push @initcvsignore,("*.bak");
+ push @initcvsignore,("*.BAK");
+ push @initcvsignore,("*.orig");
+ push @initcvsignore,("*.rej");
+ push @initcvsignore,(".del-*");
+ push @initcvsignore,("*.a");
+ push @initcvsignore,("*.olb");
+ push @initcvsignore,("*.o");
+ push @initcvsignore,("*.obj");
+ push @initcvsignore,("*.so");
+ push @initcvsignore,("*.exe");
+ push @initcvsignore,("*.Z");
+ push @initcvsignore,("*.elc");
+ push @initcvsignore,("*.ln");
+ push @initcvsignore,("core");
+
+
+ # now, load (in proper order!)
+ # each of the possible cvsignore files
+
+ # there are 4 possible .cvsignore files:
+
+ # $CVSROOT/CVSROOT/cvsignore
+ # ~/.cvsignore
+ # $CVSIGNORE environment variable
+ # .cvsignore in current directory
+
+ # The first (CVSROOT/cvsignore) would require calling cvs, so
+ # we won't do that one.
+ # The last (.cvsignore in current directory) is done
+ # for each directory. It's handled in the load_cvsignore routine.
+
+ # ~/.cvsignore
+ my @inlist;
+ my $item;
+ my $HOME=$ENV{"HOME"};
+ if (not $HOME) {$HOME = ".";}
+ load_list_from_file("$HOME/.cvsignore",\@initcvsignore);
+
+ # $CVSIGNORE environment variable
+ my $igstr = $ENV{"CVSIGNORE"}; # get env var
+ if ($igstr) {
+ my @iglist = split(/\s+/, $igstr); #if it exists, convert to list
+ load_list_from_list(\@initcvsignore,@iglist);
+ }
+
+ # now that @initcvsignore is setup,
+ # turn it into a regex string
+ $initcvsignoreregex = do_regex_convert(@initcvsignore);
+
+ # now preset the cvsignore regex string to match
+ # @initcvsignore. That way, if we aren't using local
+ # cvsignore files, we do nothing.
+ $cvsignoreregex = "(" . $initcvsignoreregex . ")";
+}
+# routine to see if the given name is in the cvsignore regex
+# returns true if it is, false if it's not
+sub ignore_file($)
+{
+ #allow user to disable the cvsignore stuff
+ if ($disable_cvsignore) {return 0;}
+ if (not $cvsignoreregex) {return 0;}# if regex is empty, nothing matches the regex
+ my $filename = shift;
+
+ if ($debug_cvsignore) {print "ignore_file:",$filename,"\n";}
+
+ if ($filename =~ $cvsignoreregex) {
+ if ($debug_cvsignore) {print $filename," matches\n";}
+ return 1;
+ }
+
+ if ($debug_cvsignore) {print $filename," doesn't match\n";}
+ return 0;
+}
+
+sub cvs_changed_in_dir($) {
+ my $dir = shift;
+
+ my ($line,$filename,$version,$mtime,$date,
+ $dir_filename,$cvstime,@subdirs,
+ @new_in_dir,$i);
+
+ # Examine status of files in CVS/Entries
+ if(not open(ENTRIES,"$dir/CVS/Entries")) {
+ if ($tabular) {
+ push @{$files{Unknown}}, $dir;
+ }
+ else {
+ warn "The directory $dir is not under CVS control\n";
+ }
+ } else {
+ load_cvsignore($dir);#load up proper cvsignore for given directory
+
+ while(defined ($line=<ENTRIES>)) {
+ # Parse CVS/Entries-line
+ $line=~m!^/(.*)/(.*)/(.*)/.*/! or do {
+ $debug and warn("Skipping entry-line $line");
+ next;
+ };
+ ($filename,$version,$date) = ($1,$2,$3);
+ $dir_filename=$dir."/".$filename;
+
+ # Mark this file as seen
+ $seen{$dir_filename}=1;
+
+ # if not exists: Deleted
+ if(not -e $dir_filename) {
+ push @{$files{Deleted}}, $dir_filename; next;
+ }
+ # if dir: save name for recursion
+ -d $dir_filename and do {
+ push @subdirs, $dir_filename; next;
+ };
+
+ # modification time of $dir_filename
+ $mtime= (stat $dir_filename)[9];
+
+
+ if($date eq "dummy timestamp") {
+ # dummy timestamp means it's new to the repository.
+ push @{$files{Changed}}, $dir_filename;
+ if ($debug) {
+ print "$dir_filename is changed\n";
+ }
+ }
+ elsif($date eq "Result of merge") {
+ # result of merge means it's changed, then updated.
+ push @{$files{Changed}}, $dir_filename;
+ if ($debug) {
+ print "$dir_filename is changed\n";
+ }
+ }
+ elsif(not
+ $date=~/... (...)\s+(\d+)\s+(\d+):(\d+):(\d+) (\d{4})/)
+ {
+ #bogus entry in Entires
+ warn "Warning: $dir_filename -> '$date' ".
+ "not in ctime(3) format\n";
+ } else {
+ $cvstime=timegm($5,$4,$3,$2,$mon{$1},$6);
+ if($cvstime != $mtime) {
+ push @{$files{Changed}}, $dir_filename;
+ if ($debug) {
+ print "$dir_filename is changed\n";
+ }
+ } else {
+ push @{$files{Unchanged}}, $dir_filename;
+ if ($debug) {
+ print "$dir_filename is Unchanged\n";
+ }
+ }
+ }
+ }
+ close ENTRIES;
+
+ # Locate any new files/dirs
+ if(not opendir(D,$dir)) {
+ warn("Cannot open $dir");
+ @new_in_dir= ();
+ } else {
+ @skip{qw(. .. CVS)}=1..3; # Filenames that that we want to ignore
+ #(note: these are exact filenames)
+ @new_in_dir=
+ (grep { not $seen{$_} } # files we have not already processed
+ map { $dir."/".$_ } # map from file to dir/file
+ grep { not ignore_file($_) } # ignore files in the cvsignore list
+ grep { not $skip{$_} } # skip files to be ignored
+ readdir(D));
+ closedir(D);
+ }
+
+ # Remember new files (actually non-directories)
+ push @{$files{New}}, grep { not -d $_ } @new_in_dir;
+ if ($debug) { print "@{$files{New}} are new in $dir\n"; }
+
+ # Remember new subdirs
+ push @subdirs, grep { -d $_ } @new_in_dir;
+
+ # Recurse all subdirs
+ if (not $local) {
+ for $i (@subdirs) { cvs_changed_in_dir($i); }
+ }
+ }
+}
+
+sub print_status()
+{
+ my $k;
+ my %show_these_states = ("Changed" => 1);
+ if(not $restrict) {
+ $show_these_states{"New"} = 1;
+ $show_these_states{"Deleted"} = 1;
+ }
+
+ if($opt_all) { $show_these_states{"Unchanged"} = 1; }
+
+ if ($tabular) {
+ my %allfiles; # key: filesname, value: state
+ my ($file, $state, $statefiles);
+
+ $show_these_states{"Unknown"} = 1;
+ while (($state, $statefiles) = each %files) {
+ for my $f (@{$statefiles}) {
+ $allfiles{$f} = $state;
+ }
+ }
+ for $file (sort keys %allfiles) {
+ $state = $allfiles{$file};
+ printf("%-10s %s\n", $state, $file) if $show_these_states{$state};
+ }
+ }
+ else {
+ print "\n";
+ for $k (keys %show_these_states) {
+ if(not $files{$k} or not @{$files{$k}}) {
+ # no files
+ $files{$k}=["(none)"];
+ }
+ print("$k files\n",
+ "---------------\n",
+ map { "$_\n" } sort @{$files{$k}});
+ print "\n";
+ }
+ }
+}
+
+load_initial_cvsignore();
+if ($debug_cvsignore) {print "initial regex:",$cvsignoreregex,"\n";}
+cvs_changed_in_dir($startdir);
+print_status();
+
diff --git a/ndb/home/bin/fix-cvs-root b/ndb/home/bin/fix-cvs-root
new file mode 100755
index 00000000000..2c4f158f825
--- /dev/null
+++ b/ndb/home/bin/fix-cvs-root
@@ -0,0 +1,17 @@
+#! /bin/sh
+
+# change all CVS/Root to current CVSROOT
+
+[ "$CVSROOT" ] || { echo "no CVSROOT in environment" >&2; exit 1; }
+
+echo "changing all CVS/Root files under `pwd`"
+sleep 1
+
+find . -path '*/CVS/Root' -print |
+while read file; do
+ echo "$file"
+ chmod +w $file || exit 1
+ echo $CVSROOT >$file || exit 1
+done
+
+echo "done"
diff --git a/ndb/home/bin/import-from-bk.sh b/ndb/home/bin/import-from-bk.sh
new file mode 100755
index 00000000000..4e3957be6d5
--- /dev/null
+++ b/ndb/home/bin/import-from-bk.sh
@@ -0,0 +1,158 @@
+#! /bin/sh
+
+# XXX does not delete files
+# XXX does not handle nested new dirs
+# this script screams for perl, no time now
+# look for bk2cvs on the net
+
+PATH=/usr/local/bin:$PATH; export PATH
+LD_LIBRARY_PATH=/usr/local/lib:$LD_LIBRARY_PATH; export LD_LIBRARY_PATH
+
+batch=n
+if [ "$1" = "-batch" ]; then
+ batch=y
+ shift
+fi
+
+say() {
+ echo "$*"
+}
+
+die() {
+ case $# in
+ 0) set -- "command failed" ;;
+ esac
+ say "$* -- aborted" >&2
+ exit 1
+}
+
+usage() {
+ die "usage: $0 [-batch] top -- copy from mysql/ndb to another NDB_TOP"
+}
+
+doit() {
+ cmd="$*"
+ if [ $batch = n ]; then
+ echo -n "$cmd [y]"
+ read junk
+ sh -c "$cmd"
+ return 0
+ else
+ echo "$cmd"
+ sh -c "$cmd"
+ return $?
+ fi
+}
+
+say "======================"
+say "`date`"
+
+case $# in
+1) [ -d $1/src/CVS ] || die "$1 is not an NDB_TOP"
+ top=$1 ;;
+*) usage ;;
+esac
+
+if ! fgrep ndb_kernel_version.h $top/include/kernel/CVS/Entries >/dev/null 2>&1; then
+ die "$top is not an NDB_TOP"
+fi
+
+if find $top -path '*/CVS/Tag' -print | grep . >/dev/null; then
+ die "$top: contains CVS/Tag files, not accepted"
+fi
+
+if [ ! -f include/SCCS/s.ndb_version.h ]; then
+ die "current dir ($PWD) is not an NDB_TOP"
+fi
+
+doit "bk pull" || exit 1
+doit "bk -r clean"
+doit "bk -r get -q"
+
+files=`bk -r. sfiles -g |
+ fgrep -v ' ' |
+ fgrep -v /.cvsignore`
+
+n=0
+files2=
+for f in $files; do
+ if [ ! -f $f ]; then
+ die "$f: no such file"
+ fi
+ if [ -w $f ]; then
+ say "$f: is writable, accept anyway"
+ fi
+ files2="$files2 $f"
+ n=$((n+1))
+done
+files=$files2
+say "$n files..."
+
+adddirs= addfiles= updfiles=
+for f in $files; do
+ d=`dirname $f`
+ b=`basename $f`
+ if [ ! -f $top/$d/CVS/Entries ]; then
+ found=n
+ for x in $adddirs; do
+ if [ $x = $d ]; then found=y; break; fi
+ done
+ if [ $found = n ]; then
+ say "$d: to create dir"
+ adddirs="$adddirs $d"
+ fi
+ addfiles="$addfiles $f"
+ say "$f: to create"
+ elif ! fgrep "/$b/" $top/$d/CVS/Entries >/dev/null; then
+ addfiles="$addfiles $f"
+ say "$f: to create"
+ else
+ cmp $f $top/$f >/dev/null
+ case $? in
+ 0) continue ;;
+ 1) ;;
+ *) die "$f: unknown error" ;;
+ esac
+ updfiles="$updfiles $f"
+ say "$f: to update"
+ fi
+done
+
+for d in $adddirs; do
+ doit "cd $top && mkdir -p $d" || die
+done
+
+for f in $addfiles $updfiles; do
+ doit "cp -fp $f $top/$f" || die
+done
+
+for d in $adddirs; do
+ # fix 1 level up
+ d2=`dirname $d`
+ if [ ! -d $top/$d2/CVS ]; then
+ doit "cd $top && cvs add $d2" || die
+ fi
+ doit "cd $top && cvs add $d" || die
+done
+
+for f in $addfiles; do
+ kb=
+ if echo $f | perl -nle "print(-B $_)" | grep 1 >/dev/null; then
+ kb="-kb"
+ fi
+ doit "cd $top && cvs add $kb $f" || die
+done
+
+tag=import_bk_`date +%Y_%m_%d`
+
+doit "cd $top && cvs commit -m $tag" || die
+doit "cd $top && cvs tag -F $tag" || die
+
+env="NDB_TOP=$top; export NDB_TOP"
+env="$env; USER_FLAGS='-DAPI_TRACE -fmessage-length=0'; export USER_FLAGS"
+doit "$env; cd $top && ./configure"
+doit "$env; cd $top && sh config/GuessConfig.sh"
+doit "$env; cd $top && make clean nuke-deps vim-tags"
+doit "$env; cd $top && make" || die
+
+say "imported ok"
diff --git a/ndb/home/bin/ndb_deploy b/ndb/home/bin/ndb_deploy
new file mode 100755
index 00000000000..773fc9b8fd7
--- /dev/null
+++ b/ndb/home/bin/ndb_deploy
@@ -0,0 +1,27 @@
+#!/bin/sh
+
+if [ $# -eq 0 ]
+then
+ for i in $DEPLOY_DST
+ do
+ rsync -r -v --exclude '*.a' $NDB_TOP/bin $NDB_TOP/lib $i/
+ done
+else
+ while [ $# -gt 0 ]
+ do
+ arg=$1
+ shift;
+ if [ `echo $arg | grep -c lib` -eq 0 ]
+ then
+ dst=bin/
+ else
+ dst=lib/
+ fi
+
+ for i in $DEPLOY_DST
+ do
+ rsync -v $arg $i/$dst
+ done
+ done
+fi
+
diff --git a/ndb/home/bin/ndbdoxy.pl b/ndb/home/bin/ndbdoxy.pl
new file mode 100755
index 00000000000..89b7de8440e
--- /dev/null
+++ b/ndb/home/bin/ndbdoxy.pl
@@ -0,0 +1,184 @@
+#!/usr/local/bin/perl
+#
+# ndbdoxy.pl Executes doxygen on a checked out version of NDB Cluster
+#
+# Written by Lars Thalmann, 2003.
+
+use strict;
+umask 000;
+
+# -----------------------------------------------------------------------------
+# Settings
+# -----------------------------------------------------------------------------
+
+my $root = "/home/elathal/public_html/cvsdoxy";
+
+$ENV{LD_LIBRARY_PATH} = "/usr/local/lib:/opt/as/local/lib";
+$ENV{LD_LIBRARY_PATH} = $ENV{LD_LIBRARY_PATH} . ":/opt/as/forte6/SUNWspro/lib";
+$ENV{PATH} = $ENV{PATH} . ":/usr/local/bin:/opt/as/local/bin";
+$ENV{PATH} = $ENV{PATH} . ":/opt/as/local/teTeX/bin/sparc-sun-solaris2.8";
+
+my $DOXYGEN = "doxygen";
+my $PDFLATEX = "pdflatex";
+my $MAKEINDEX = "makeindex";
+
+# -----------------------------------------------------------------------------
+# Argument handling
+# -----------------------------------------------------------------------------
+
+if (@ARGV != 3) {
+ print<<END;
+Usage:
+ ndbdoxy.pl <module> <title> <version>
+
+ where
+ <module> is cvsdoxy module to doxgenify
+ <title> is title of report
+ <version> is version of NDB Cluster
+END
+ exit;
+}
+my $module = $ARGV[0];
+my $title = $ARGV[1];
+my $version = $ARGV[2];
+my $destdir = ".";
+
+# -----------------------------------------------------------------------------
+# Execute Doxygen -g
+# -----------------------------------------------------------------------------
+
+if (-r "${root}/doxyfiles/${module}.doxyfile") {
+ system("cd ${destdir}; \
+ cp ${root}/doxyfiles/${module}.doxyfile Doxyfile");
+} elsif (-r "${root}/doxyfiles/default.doxyfile") {
+ system("cd ${destdir}; \
+ cp ${root}/doxyfiles/default.doxyfile Doxyfile");
+} else {
+ system("cd ${destdir}; $DOXYGEN -g");
+}
+
+# -----------------------------------------------------------------------------
+# HTML Footer
+# -----------------------------------------------------------------------------
+
+if (-r "${root}/doxyfiles/htmlfooter") {
+ system("cd ${destdir}; \
+ cp ${root}/doxyfiles/htmlfooter footer.html");
+
+ open (INFILE, "< ${destdir}/footer.html")
+ or die "Error opening ${destdir}/footer.html.\n";
+ open (OUTFILE, "> ${destdir}/footer.html.new")
+ or die "Error opening ${destdir}/footer.html.new.\n";
+ while (<INFILE>) {
+ if (/(.*)DATE(.*)$/) {
+ print OUTFILE $1 . localtime() . $2;
+ } else {
+ print OUTFILE;
+ }
+ }
+ close INFILE;
+ close OUTFILE;
+
+ system("mv ${destdir}/footer.html.new ${destdir}/footer.html");
+} else {
+ print("Warning: No ${root}/doxyfiles/${module}.htmlfooter");
+}
+
+# -----------------------------------------------------------------------------
+# Execute Doxygen
+# -----------------------------------------------------------------------------
+
+system("cd ${destdir}; $DOXYGEN");
+
+# -----------------------------------------------------------------------------
+# Change a little in refman.tex
+# -----------------------------------------------------------------------------
+
+open (INFILE, "< ${destdir}/latex/refman.tex")
+ or die "Error opening ${destdir}/latex/refman.tex.\n";
+open (OUTFILE, "> ${destdir}/latex/refman.tex.new")
+ or die "Error opening ${destdir}/latex/refman.tex.new.\n";
+
+while (<INFILE>)
+{
+ if (/(.*)Reference Manual(.*)$/) {
+ print OUTFILE $1 .
+ "\\mbox{}\\vspace{-3cm}\\mbox{}" .
+ "\\hrule\\bigskip\\bigskip\\bigskip\\bigskip" .
+ "\\Huge{" . $title . "}" . $2;
+ } elsif (/(.*)Generated by Doxygen 1.2.1[0-9](.*)$/) {
+ print OUTFILE $1 .
+ "\\begin{center}" .
+ "\\LARGE{MySQL AB}" .
+ "\\end{center}".
+ "\\hfill\\bigskip\\bigskip\\bigskip\\hrule" .
+ "\\bigskip\\bigskip\\bigskip\\bigskip\\bigskip" .
+ "\\bigskip\\bigskip\\bigskip\\bigskip\\bigskip" .
+ "\\bigskip\\bigskip NDB Cluster Release " . $version .
+ "\\bigskip\\bigskip\\bigskip\\bigskip\\bigskip\\hfill" .
+ $2;
+ } elsif (/\\chapter\{File Index\}/) {
+ print OUTFILE "\%\\chapter{File Index}\n";
+ } elsif (/\\input{files}/) {
+ print OUTFILE "\%\\input{files}\n";
+ } elsif (/\\chapter\{Page Index\}/) {
+ print OUTFILE "\%\\chapter{Page Index}\n";
+ } elsif (/\\input{pages}/) {
+ print OUTFILE "\%\\input{pages}\n";
+ } else {
+ print OUTFILE;
+ }
+}
+
+close INFILE;
+close OUTFILE;
+
+system("mv ${destdir}/latex/refman.tex.new ${destdir}/latex/refman.tex");
+
+# -----------------------------------------------------------------------------
+# Change a little in doxygen.sty
+# -----------------------------------------------------------------------------
+
+open (INFILE, "< ${destdir}/latex/doxygen.sty")
+ or die "Error opening INFILE.\n";
+open (OUTFILE, "> ${destdir}/latex/doxygen.sty.new")
+ or die "Error opening OUTFILE.\n";
+
+while (<INFILE>)
+{
+ if (/\\rfoot/) {
+ print OUTFILE "\\rfoot[\\fancyplain{}{\\bfseries\\small \\copyright~Copyright 2003 MySQL AB\\hfill support-cluster\@mysql.com}]{}\n";
+ } elsif (/\\lfoot/) {
+ print OUTFILE "\\lfoot[]{\\fancyplain{}{\\bfseries\\small support-cluster\@mysql.com\\hfill \\copyright~Copyright 2003 MySQL AB}}\n";
+ } else {
+ print OUTFILE;
+ }
+}
+
+close INFILE;
+close OUTFILE;
+
+system("mv ${destdir}/latex/doxygen.sty.new ${destdir}/latex/doxygen.sty");
+
+# -----------------------------------------------------------------------------
+# Other
+# -----------------------------------------------------------------------------
+
+#system("cd ${root}/tmp/${module}; \
+# mkdir html.tar; \
+# cd html.tar; \
+# cp -r ../html ${module}; \
+# tar cf ${module}.html.tar ${module}; \
+# /usr/local/bin/gzip ${module}.html.tar; \
+# /bin/rm -rf ${root}/tmp/${module}/html.tar/${module}");
+
+#system("cd ${destdir}/latex/; \
+# $PDFLATEX refman.tex \
+# $MAKEINDEX refman.idx \
+# $PDFLATEX refman.tex \
+# mv -f refman.pdf ${module}.pdf");
+
+print<<END;
+Execute:
+ latex refman; makeindex refman; latex refman
+END
diff --git a/ndb/home/bin/ngcalc b/ndb/home/bin/ngcalc
new file mode 100755
index 00000000000..a289d384db9
--- /dev/null
+++ b/ndb/home/bin/ngcalc
@@ -0,0 +1,78 @@
+#! /usr/local/bin/perl
+
+use strict;
+use Getopt::Long;
+
+sub usage {
+ print <<END;
+ngcalc -- calculate node groups and table fragments
+usage: ngcalc [ options ] f1 f2 ...
+-g num number of node groups (default 2)
+-r num number of replicas (default 2)
+-n list comma-separated list of db nodes (default 1,2,...)
+fX number of fragments per node group in table X (e.g. 1,2,8)
+ (all replicas count as same fragment)
+END
+ exit(1);
+};
+
+use vars qw($cnoOfNodeGroups $cnoReplicas $nodeArray);
+
+$cnoOfNodeGroups = 2;
+$cnoReplicas = 2;
+GetOptions(
+ "g=i" => \$cnoOfNodeGroups,
+ "r=i" => \$cnoReplicas,
+ "n=s" => \$nodeArray,
+) or &usage;
+
+my @tableList = @ARGV;
+
+$cnoOfNodeGroups > 0 or &usage;
+$cnoReplicas > 0 or &usage;
+if (! defined($nodeArray)) {
+ $nodeArray = join(',', 1..($cnoOfNodeGroups*$cnoReplicas));
+}
+$nodeArray =~ /^\d+(,\d+)*$/ or &usage;
+my @nodeArray = split(/,/, $nodeArray);
+@nodeArray == $cnoOfNodeGroups*$cnoReplicas or &usage;
+
+my @nodeGroupRecord;
+for (my $i = 0; $i < $cnoOfNodeGroups; $i++) {
+ my $rec = {};
+ my $nodes = [];
+ for (my $j = 0; $j < $cnoReplicas; $j++) {
+ push(@$nodes, $nodeArray[$i * $cnoReplicas + $j]);
+ }
+ $rec->{nodesInGroup} = $nodes;
+ $rec->{nodeCount} = $cnoReplicas;
+ $rec->{nextReplicaNode} = 0;
+ $nodeGroupRecord[$i] = $rec;
+ print "NG $i: ", join(" ", @{$rec->{nodesInGroup}}), "\n";
+}
+
+# see Dbdih::execCREATE_FRAGMENTATION_REQ
+
+my $c_nextNodeGroup = 0;
+for (my $t = 0; $t < @tableList; $t++) {
+ use integer;
+ my $f = $tableList[$t];
+ my $ng = $c_nextNodeGroup++;
+ $c_nextNodeGroup = 0 if $c_nextNodeGroup == $cnoOfNodeGroups;
+ my $noOfFragments = $f * $cnoOfNodeGroups;
+ my @fragments;
+ for (my $fragNo = 0; $fragNo < $noOfFragments; $fragNo++) {
+ my $rec = $nodeGroupRecord[$ng];
+ my $max = $rec->{nodeCount};
+ my $ind = $rec->{nextReplicaNode};
+ $rec->{nextReplicaNode} = ($ind + 1 >= $max ? 0 : $ind + 1);
+ for (my $replicaNo = 0; $replicaNo < $cnoReplicas; $replicaNo++) {
+ my $nodeId = $rec->{nodesInGroup}[$ind++];
+ push(@fragments, $nodeId);
+ $ind = ($ind == $max ? 0 : $ind);
+ }
+ $ng++;
+ $ng = ($ng == $cnoOfNodeGroups ? 0 : $ng);
+ }
+ printf "%02d %s\n", $t, join(" ", @fragments);
+}
diff --git a/ndb/home/bin/signallog2html.lib/signallog2list.awk b/ndb/home/bin/signallog2html.lib/signallog2list.awk
new file mode 100644
index 00000000000..9839f314556
--- /dev/null
+++ b/ndb/home/bin/signallog2html.lib/signallog2list.awk
@@ -0,0 +1,102 @@
+BEGIN{
+ PRINT=0;
+ SIGNAL_ARRAY[0]="";
+ BLOCK_ID=0;
+ SIGNAL_ID=-22;
+}
+{
+ SIGNAL_ARRAY[SIGNAL_ID]=SIGNAL_ID;
+}
+
+/^---- Send ----- Signal ----------------/ {
+ DIRECTION="S";
+ SENDER="";
+ SENDPROCESS="";
+ RECEIVER="";
+ RECPROCESS="";
+ SIGNAL="";
+ RECSIGID="?";
+ SIGID="?";
+ DELAY="N/A";
+}
+
+/^---- Send delay Signal/ {
+ DIRECTION="SD";
+ SENDER="";
+ SENDPROCESS="";
+ RECEIVER="";
+ RECPROCESS="";
+ SIGNAL="";
+ RECSIGID="?";
+ SIGID="?";
+ DELAY=$5;
+
+ LEN=length(DELAY);
+ DELAY=substr(DELAY,2,LEN);
+}
+
+/^---- Received - Signal ----------------/ {
+ DIRECTION="R";
+ SENDER="";
+ SENDPROCESS="";
+ RECEIVER="";
+ RECPROCESS="";
+ SIGNAL="";
+ RECSIGID="?";
+ SIGID="?";
+ DELAY="N/A";
+}
+
+/r.bn:/{
+
+ RECEIVER=$3;
+ RECPROCESS=$5;
+
+ if(DIRECTION == "R"){
+ SIGNAL=$10;
+ RECSIGID=$7;
+ }
+ else
+ SIGNAL=$8;
+}
+
+/s.bn:/{
+
+ SENDER=$3;
+ SIGID=$7;
+
+ if(SIGID == SIGNAL_ARRAY[SIGID]){
+ PRINT=1;
+ if(DIRECTION == "R"){
+ SIGNAL_ARRAY[RECSIGID]=RECSIGID;
+ };
+ }
+
+ SENDPROCESS=$5;
+
+ LEN=length(RECEIVER);
+ RECEIVER=substr(RECEIVER,2,LEN-3);
+
+ if(BLOCK_ID == "ALL" || RECEIVER==BLOCK_ID){PRINT=1; }
+
+ LEN=length(SENDER);
+ SENDER=substr(SENDER,2,LEN-3);
+ if(BLOCK_ID == "ALL" || SENDER == BLOCK_ID){ PRINT=1;}
+
+ LEN=length(SIGNAL);
+ SIGNAL=substr(SIGNAL,2,LEN-2);
+
+ LEN=length(SENDPROCESS);
+ SENDPROCESS=substr(SENDPROCESS,1,LEN-1);
+
+ LEN=length(RECPROCESS);
+ RECPROCESS=substr(RECPROCESS,1,LEN-1);
+
+ if( PRINT == 1){
+ print DIRECTION" "SENDPROCESS" "SENDER" "RECPROCESS" "RECEIVER" "SIGNAL" "SIGID" "RECSIGID" "DELAY;
+ }
+
+ PRINT=0;
+}
+
+
diff --git a/ndb/home/bin/signallog2html.lib/uniq_blocks.awk b/ndb/home/bin/signallog2html.lib/uniq_blocks.awk
new file mode 100644
index 00000000000..43f48d1cde1
--- /dev/null
+++ b/ndb/home/bin/signallog2html.lib/uniq_blocks.awk
@@ -0,0 +1,29 @@
+BEGIN{
+ NAMES[""]="";
+ ORDER[0]="";
+ NUM=0;
+}
+
+{
+ if(NAMES[$2$3]!=$2$3){
+ NAMES[$2$3]=$2$3;
+ ORDER[NUM]=$2$3;
+ NUM++;
+ }
+
+ if(NAMES[$4$5]!=$4$5){
+ NAMES[$4$5]=$4$5;
+ ORDER[NUM]=$4$5;
+ NUM++;
+ }
+
+
+}
+END{
+ for(i=0; i<NUM; i++){
+ LIST=ORDER[i]" "LIST;
+
+ }
+ print LIST;
+}
+
diff --git a/ndb/home/bin/signallog2html.sh b/ndb/home/bin/signallog2html.sh
new file mode 100755
index 00000000000..5665275807c
--- /dev/null
+++ b/ndb/home/bin/signallog2html.sh
@@ -0,0 +1,349 @@
+#!/bin/sh
+# NAME
+# signallog2html.sh
+#
+# SYNOPSIS
+# signallog2html.sh [ -b <block_name | ALL> ] [ -s <signal_id> ] -f signal_log_file
+#
+# DESCRIPTION
+# Creates a signal sequence diagram in HTML format that can be
+# viewed from a web browser. The HTML file is created from a signal
+# log file and it contains a big table with jpeg files in every
+# table cell. Every row in the table is a signal. The block_name
+# could be one of the following: CMVMI MISSRA NDBFS NDBCNTR DBACC
+# DBDICT DBLQH DBDIH DBTC DBTUP QMGR ALL. The signal_id is a
+# number. If no block_name or signal_id is given the default
+# block_name "ALL" is used.
+#
+#
+#
+# OPTIONS
+#
+# EXAMPLES
+#
+#
+# ENVIRONMENT
+# NDB_PROJ_HOME Home dir for ndb
+#
+# FILES
+# $NDB_PROJ_HOME/lib/funcs.sh General shell script functions.
+# uniq_blocks.awk Creates a list of unique blocks
+# in the signal_log_file.
+# signallog2list.awk Creates a list file from the signal_log_file.
+# empty.JPG Jpeg file, must exist in the HTML file
+# directory for viewing.
+# left_line.JPG
+# line.JPG
+# right_line.JPG
+# self_line.JPG
+#
+#
+# SEE ALSO
+#
+# DIAGNOSTICTS
+#
+# VERSION
+# 1.0
+#
+# DATE
+# 011029
+#
+# AUTHOR
+# Jan Markborg
+#
+
+progname=`basename $0`
+synopsis="signallog2html.sh [ -b <block_name | ALL> ] [ -s <signal_id> ] -f signal_log_file"
+block_name=""
+signal_id=""
+verbose=yes
+signal_log_file=""
+
+: ${NDB_PROJ_HOME:?} # If undefined, exit with error message
+
+: ${NDB_LOCAL_BUILD_OPTIONS:=--} # If undef, set to --. Keeps getopts happy.
+ # You may have to experiment a bit
+ # to get quoting right (if you need it).
+
+
+. $NDB_PROJ_HOME/lib/funcs.sh # Load some good stuff
+
+# defaults for options related variables
+#
+report_date=`date '+%Y-%m-%d'`
+
+# Option parsing for the the command line.
+#
+
+while getopts f:b:s: i
+do
+ case $i in
+ f) signal_log_file=$OPTARG;;
+ b) block_name=$OPTARG;;
+ s) signal_id=$OPTARG;;
+ \?) syndie ;; # print synopsis and exit
+ esac
+done
+
+# -- Verify
+trace "Verifying signal_log_file $signal_log_file"
+
+if [ x$signal_log_file = "x" ]
+then
+ syndie "Invalid signal_log_file name: $signal_log_file not found"
+fi
+
+
+if [ ! -r $signal_log_file ]
+then
+ syndie "Invalid signal_log_file name: $signal_log_file not found"
+fi
+
+
+
+if [ blocknameSET = 1 ]
+then
+
+ trace "Verifying block_name"
+ case $block_name in
+ CMVMI| MISSRA| NDBFS| NDBCNTR| DBACC| DBDICT| DBLQH| DBDIH| DBTC| DBTUP| QMGR);;
+ ALL) trace "Signals to/from every block will be traced!";;
+ *) syndie "Unknown block name: $block_name";;
+ esac
+fi
+
+if [ block_name="" -a signal_id="" ]
+then
+ block_name=ALL
+ trace "block_name = $block_name"
+fi
+
+trace "Arguments OK"
+
+###
+#
+# General html functions
+header(){
+ cat <<EOF
+<html><head><title>$*</title></head>
+<body>
+EOF
+}
+
+footer(){
+ cat <<EOF
+</body></html>
+EOF
+}
+
+heading(){
+ h=$1; shift
+ cat <<EOF
+<h$h>$*</h$h>
+EOF
+}
+
+table(){
+ echo "<table $*>"
+}
+
+table_header(){
+ echo "<th>$*</th>"
+}
+
+end_table(){
+ echo "</table>"
+}
+
+row(){
+ echo "<tr>"
+}
+
+end_row(){
+ echo "</tr>"
+}
+
+c_column(){
+ cat <<EOF
+<td valign=center align=center>$*</td>
+EOF
+}
+
+bold(){
+ cat <<EOF
+<b>$*</b>
+EOF
+}
+
+column(){
+ cat <<EOF
+<td align=left>$*</td>
+EOF
+}
+
+para(){
+ cat <<EOF
+<p></p>
+EOF
+}
+
+hr(){
+ cat <<EOF
+<hr>
+EOF
+}
+
+img_column(){
+ cat <<EOF
+<td><center><$* height=100% width=100%></center></td>
+EOF
+}
+
+# Check the direction of arrow.
+# arrowDirection(){ $columnarray $sendnode$sendblock $recnode$recblock
+arrowDirection(){
+if [ $2 = $3 ]
+then
+ arrow=SELF
+ return;
+else
+ for x in $1
+ do
+ if [ $x = $2 ]
+ then
+ arrow=RIGHT
+ break
+ elif [ $x = $3 ]
+ then
+ arrow=LEFT
+ break
+ fi
+ done
+fi
+}
+
+drawImages(){
+for x in $columnarray
+do
+ case $arrow in
+ SELF)
+ if [ $x = $sendnode$sendblock ]
+ then
+ img_column img SRC=\"self_line.JPG\"
+ else
+ img_column img SRC=\"empty.JPG\"
+ fi;;
+
+ RIGHT)
+ if [ $x = $recnode$recblock ]
+ then
+ img_column img SRC=\"right_line.JPG\"
+ weHavePassedRec=1
+ elif [ $x = $sendnode$sendblock ]
+ then
+ img_column img SRC=\"empty.JPG\"
+ weHavePassedSen=1
+ elif [ $weHavePassedRec = 1 -o $weHavePassedSen = 0 ]
+ then
+ img_column img SRC=\"empty.JPG\"
+ elif [ $weHavePassedRec = 0 -a $weHavePassedSen = 1 ]
+ then
+ img_column img SRC=\"line.JPG\"
+ fi;;
+
+ LEFT)
+ if [ $x = $recnode$recblock ]
+ then
+ img_column img SRC=\"empty.JPG\"
+ weHaveJustPassedRec=1
+ weHavePassedRec=1
+ continue
+ fi
+ if [ $x = $sendnode$sendblock -a $weHaveJustPassedRec = 1 ]
+ then
+ img_column img SRC=\"left_line.JPG\"
+ weHaveJustPassedRec=0
+ weHavePassedSen=1
+ continue
+ fi
+ if [ $x = $sendnode$sendblock ]
+ then
+ img_column img SRC=\"line.JPG\"
+ weHavePassedSen=1
+ continue
+ fi
+ if [ $weHaveJustPassedRec = 1 ]
+ then
+ img_column img SRC=\"left_line.JPG\"
+ weHaveJustPassedRec=0
+ continue
+ fi
+ if [ $weHavePassedSen = 1 -o $weHavePassedRec = 0 ]
+ then
+ img_column img SRC=\"empty.JPG\"
+ continue
+ fi
+
+ if [ $weHavePassedRec = 1 -a $weHavePassedSen = 0 ]
+ then
+ img_column img SRC=\"line.JPG\"
+ continue
+
+ fi
+ column ERROR;;
+
+ *)
+ echo ERROR;;
+ esac
+done
+column $signal
+}
+
+### Main
+trace "Making HTML file"
+(
+ header "Signal sequence diagram $report_date"
+ heading 1 "Signal sequence diagram $report_date"
+
+ trace "Making list file"
+ #make a signal list file from the signal log file.
+ `awk -f /home/ndb/bin/signallog2html.lib/signallog2list.awk SIGNAL_ID=$signal_id BLOCK_ID=$block_name $signal_log_file > $signal_log_file.list`
+
+ COLUMNS=`awk -f /home/ndb/bin/signallog2html.lib/uniq_blocks.awk $signal_log_file.list | wc -w`
+
+ table "border=0 cellspacing=0 cellpadding=0 cols=`expr $COLUMNS + 1`"
+
+ columnarray=`awk -f /home/ndb/bin/signallog2html.lib/uniq_blocks.awk $signal_log_file.list`
+
+ row
+ column #make an empty first column!
+ for col in $columnarray
+ do
+ table_header $col
+ done
+
+ grep "" $signal_log_file.list | \
+ while read direction sendnode sendblock recnode recblock signal sigid recsigid delay
+ do
+ if [ $direction = "R" ]
+ then
+ row
+ weHavePassedRec=0
+ weHavePassedSen=0
+ weHaveJustPassedRec=0
+ arrow=""
+
+ # calculate the direction of the arrow.
+ arrowDirection "$columnarray" "$sendnode$sendblock" "$recnode$recblock"
+
+ # Draw the arrow images.
+ drawImages
+ end_row
+ fi
+ done
+ end_table
+
+ footer
+) > $signal_log_file.html
+
+exit 0
diff --git a/ndb/home/bin/stripcr b/ndb/home/bin/stripcr
new file mode 100755
index 00000000000..540418f88cf
--- /dev/null
+++ b/ndb/home/bin/stripcr
@@ -0,0 +1,90 @@
+#!/bin/sh
+
+
+# NAME
+# stripcr - a program for removing carriage return chars from dos-files.
+#
+# SYNOPSIS
+# stripcr [file...]
+#
+# DESCRIPTION
+# stripcr deletes all CR characters from the given files.
+# The files are edited in place.
+# If no files are given, stdin and stdout are used instead.
+#
+# OPTIONS
+# -s extension Make a copy of the original of each file, and
+# give it the given extension (.bak, .orig, -bak, ...).
+#
+# EXAMPLES
+# stripcr file.txt innerloop.cc
+# stripcr -i.bak *.cc
+#
+# ENVIRONMENT
+# NDB_PROJ_HOME Home dir for ndb
+#
+# FILES
+# $NDB_PROJ_HOME/lib/funcs.sh Some userful functions for safe execution
+# of commands, printing, and tracing.
+#
+# VERSION
+# 1.0
+#
+# AUTHOR
+# Jonas Mölsä
+#
+
+
+progname=`basename $0`
+synopsis="stripcr [-s extension] [file...]"
+
+
+: ${NDB_PROJ_HOME:?} # If undefined, exit with error message
+
+: ${STRIPCR_OPTIONS:=--} # If undefined, set to --, to keep getopts happy.
+ # You may have to experiment, to get quoting right.
+
+. $NDB_PROJ_HOME/lib/funcs.sh
+
+
+# defaults for options related variables
+#
+extension=
+options="$STRIPCR_OPTIONS"
+
+# used if error when parsing the options environment variable
+#
+env_opterr="options environment variable: <<$options>>"
+
+
+
+# We want to be able to set options in an environment variable,
+# as well as on the command line. In order not to have to repeat
+# the same getopts information twice, we loop two times over the
+# getopts while loop. The first time, we process options from
+# the options environment variable, the second time we process
+# options from the command line.
+#
+# The things to change are the actual options and what they do.
+#
+#
+for optstring in "$options" "" # 1. options variable 2. cmd line
+do
+ while getopts s: i $optstring # optstring empty => no arg => cmd line
+ do
+ case $i in
+
+ s) extension="$OPTARG";;
+ \?) syndie $env_opterr;; # print synopsis and exit
+
+ esac
+ done
+
+ [ -n "$optstring" ] && OPTIND=1 # Reset for round 2, cmd line options
+
+ env_opterr= # Round 2 should not use the value
+done
+shift `expr $OPTIND - 1`
+
+
+safe perl -i$extension -lpe 'tr/\r//d' $*
diff --git a/ndb/include/debugger/DebuggerNames.hpp b/ndb/include/debugger/DebuggerNames.hpp
new file mode 100644
index 00000000000..cf9b1b57226
--- /dev/null
+++ b/ndb/include/debugger/DebuggerNames.hpp
@@ -0,0 +1,71 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DEBUGGER_NAMES
+#define DEBUGGER_NAMES
+
+#include <kernel_types.h>
+#include <signaldata/SignalDataPrint.hpp>
+
+/**
+ * getSignalName
+ *
+ * NOTES: Very quick
+ *
+ * RETURNS: Signal name or 0 if none found
+ */
+const char *
+getSignalName(GlobalSignalNumber gsn, const char * defualtValue = "Unknown");
+
+/**
+ * getGsn
+ *
+ * NOTES: Very slow
+ *
+ * RETURNS: Gsn or 0 if none found
+ */
+GlobalSignalNumber
+getGsn(const char * signalName);
+
+/**
+ * getBlockName
+ *
+ * NOTES: Very quick
+ *
+ * RETURNS: Block name or
+ * defValue if not a valid block number
+ */
+const char *
+getBlockName(BlockNumber blockNo, const char * defValue = 0);
+
+/**
+ * getBlockNo
+ *
+ * NOTES: Very slow
+ *
+ * RETURNS: BlockNo or 0 if none found
+ */
+BlockNumber
+getBlockNo(const char * blockName);
+
+/**
+ * Find a print function for a signal
+ *
+ * RETURNS: 0 if none found
+ */
+SignalDataPrintFunction findPrintFunction(GlobalSignalNumber);
+
+#endif
diff --git a/ndb/include/debugger/EventLogger.hpp b/ndb/include/debugger/EventLogger.hpp
new file mode 100644
index 00000000000..c49bd176ee8
--- /dev/null
+++ b/ndb/include/debugger/EventLogger.hpp
@@ -0,0 +1,234 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef EVENTLOGGER_H
+#define EVENTLOGGER_H
+
+#include <Logger.hpp>
+#include <FileLogHandler.hpp>
+#include <GrepError.hpp>
+#include <kernel_types.h>
+#include <kernel/LogLevel.hpp>
+#include <signaldata/EventReport.hpp>
+
+/**
+ * The EventLogger is primarily used for logging NDB events
+ * in the Management Server. It inherits all logging functionality of Logger.
+ *
+ * HOW TO USE
+ *
+ * 1) Create an EventLogger
+ *
+ * EventLogger myEventLogger = new EventLogger();
+ *
+ * 2) Log NDB events and other log messages.
+ *
+ * myEventLogger->info("Changing log levels.");
+ *
+ * EventReport* report = (EventReport*)&theSignalData[0];
+ * myEventLogger->log(eventReport->getEventType(), theSignalData, aNodeId);
+ *
+ *
+ * The following NDB event categories and log levels are enabled as default:
+ *
+ * EVENT-CATEGORY LOG-LEVEL
+ *
+ * Startup 4
+ * Shutdown 1
+ * Statistic 2
+ * Checkpoint 5
+ * NodeRestart 8
+ * Connection 2
+ * Error 15
+ * Info 10
+ *
+ * @see Logger
+ * @version #@ $Id: EventLogger.hpp,v 1.3 2003/09/01 10:15:52 innpeno Exp $
+ */
+class EventLogger : public Logger
+{
+public:
+ /**
+ * Default constructor. Enables default log levels and
+ * sets the log category to 'EventLogger'.
+ */
+ EventLogger();
+
+ /**
+ * Destructor.
+ */
+ ~EventLogger();
+
+ /**
+ * Open/create the eventlog, the default name is 'cluster.log'.
+ *
+ * @return true if successful.
+ */
+ bool open();
+
+ /**
+ * Opens/creates the eventlog with the specified filename.
+ *
+ * @param aFileName the eventlog filename.
+ * @param maxNoFiles the maximum no of archived eventlog files.
+ * @param maxFileSize the maximum eventlog file size.
+ * @param maxLogEntries the maximum number of log entries before
+ * checking time to archive.
+ * @return true if successful.
+ */
+ bool open(const char* logFileName,
+ int maxNoFiles = FileLogHandler::MAX_NO_FILES,
+ long int maxFileSize = FileLogHandler::MAX_FILE_SIZE,
+ unsigned int maxLogEntries = FileLogHandler::MAX_LOG_ENTRIES);
+
+ /**
+ * Closes the eventlog.
+ */
+ void close();
+
+ /**
+ * Logs the NDB event.
+ *
+ * @param nodeId the node id of event origin.
+ * @param eventType the type of event.
+ * @param theData the event data.
+ * @deprecated use log(int eventType, const Uint32* theData, NodeId nodeId)
+ */
+ void log(NodeId nodeId, int eventType, const Uint32* theData);
+
+ /**
+ * Logs the NDB event.
+ *
+ * @param eventType the type of event.
+ * @param theData the event data.
+ * @param nodeId the node id of event origin.
+ */
+ void log(int eventType, const Uint32* theData, NodeId nodeId = 0);
+
+ /**
+ * Returns the current log levels.
+ * Enable, disable log levels to filter the events that are sent to the
+ * eventlog.
+ *
+ * @return the log level.
+ */
+ LogLevel& getLoglevel();
+
+ /**
+ * Returns the log level that is used to filter an event. The event will not
+ * be logged unless its event category's log level is <= levelFilter.
+ *
+ * @return the log level filter that is used for all event categories.
+ */
+ int getFilterLevel() const;
+ /**
+ * Sets log level filter. The event will be logged if
+ * the event category's log level is <= 'filterLevel'.
+ *
+ * @param level the log level to filter.
+ */
+ void setFilterLevel(int filterLevel);
+
+ /**
+ * Returns the event text for the specified event report type.
+ *
+ * @param type the event type.
+ * @param theData the event data.
+ * @param nodeId a node id.
+ * @return the event report text.
+ */
+ static const char* getText(int type,
+ const Uint32* theData, NodeId nodeId = 0);
+
+ /**
+ * Find a category matching the string
+ *
+ * @param str string to match.
+ * @param cat the event category.
+ * @param exactMatch only do exact matching.
+ *
+ * @return TRUE if match is found, then cat is modified
+ * FALSE if match is not found
+ */
+ static bool matchEventCategory(const char * str,
+ LogLevel::EventCategory * cat,
+ bool exactMatch = false);
+
+ /**
+ * Returns category name or NULL if not found.
+ *
+ * @param cat the event category.
+ * @return category name.
+ */
+ static const char * getEventCategoryName(LogLevel::EventCategory cat);
+
+ /**
+ * Specifies allowed event categories/log levels.
+ */
+ struct EventCategoryName {
+ LogLevel::EventCategory category;
+ const char * name;
+ };
+
+ static const EventCategoryName eventCategoryNames[];
+ static const Uint32 noOfEventCategoryNames;
+
+ /**
+ * This matrix defines which event should be printed when
+ *
+ * threshold - is in range [0-15]
+ * severity - DEBUG to ALERT (Type of log message)
+ */
+ struct EventRepLogLevelMatrix {
+ EventReport::EventType eventType;
+ LogLevel::EventCategory eventCategory;
+ Uint32 threshold;
+ Logger::LoggerLevel severity;
+ };
+
+ static const EventRepLogLevelMatrix matrix[];
+
+ /**
+ * Default log levels for management nodes.
+ *
+ * threshold - is in range [0-15]
+ */
+ struct EventLogMatrix {
+ LogLevel::EventCategory eventCategory;
+ Uint32 threshold;
+ };
+
+ static const EventLogMatrix defEventLogMatrix[];
+
+
+ static const Uint32 matrixSize;
+ static const Uint32 defEventLogMatrixSize;
+
+private:
+ /** Prohibit */
+ EventLogger(const EventLogger&);
+ EventLogger operator = (const EventLogger&);
+ bool operator == (const EventLogger&);
+
+ LogLevel m_logLevel;
+ Uint32 m_filterLevel;
+
+ STATIC_CONST(MAX_TEXT_LENGTH = 256);
+ static char m_text[MAX_TEXT_LENGTH];
+};
+
+
+#endif
diff --git a/ndb/include/debugger/GrepError.hpp b/ndb/include/debugger/GrepError.hpp
new file mode 100644
index 00000000000..ab6a7b272a5
--- /dev/null
+++ b/ndb/include/debugger/GrepError.hpp
@@ -0,0 +1,94 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GREP_ERROR_H
+#define GREP_ERROR_H
+
+#include <ndb_types.h>
+
+/**
+ *
+ */
+class GrepError {
+public:
+ enum Code {
+ NO_ERROR = 0,
+ SUBSCRIPTION_ID_NOMEM = 1,
+ SUBSCRIPTION_ID_NOT_FOUND = 2,
+ SUBSCRIPTION_ID_NOT_UNIQUE = 3,
+ SUBSCRIPTION_ID_SUMA_FAILED_CREATE = 4,
+ SUBSCRIPTION_ID_ALREADY_EXIST = 5,
+ COULD_NOT_ALLOCATE_MEM_FOR_SIGNAL = 6,
+ NULL_VALUE = 7,
+ SEQUENCE_ERROR = 8,
+ NOSPACE_IN_POOL= 9,
+ SUBSCRIPTION_NOT_FOUND = 10,
+
+ NF_FakeErrorREF = 11,
+
+ // Error that the user can get when issuing commands
+ SUBSCRIPTION_NOT_STARTED = 100,
+ START_OF_COMPONENT_IN_WRONG_STATE,
+ START_ALREADY_IN_PROGRESS,
+ ILLEGAL_STOP_EPOCH_ID,
+ WRONG_NO_OF_SECTIONS,
+ ILLEGAL_ACTION_WHEN_STOPPING,
+ ILLEGAL_USE_OF_COMMAND,
+ CHANNEL_NOT_STOPPABLE,
+
+ // subscriber releated 20 - 30
+ SUBSCRIBER_NOT_FOUND = 20,
+
+ //SUMA specific 400 - 600
+ SELECTED_TABLE_NOT_FOUND = 400,
+ SELECTED_TABLE_ALREADY_ADDED = 401,
+
+ //REP ERRORS starts at 1000
+ REP_NO_CONNECTED_NODES = 1001,
+ REP_DELETE_NEGATIVE_EPOCH = 1002,
+ REP_DELETE_NONEXISTING_EPOCH = 1003,
+ REP_APPLY_LOGRECORD_FAILED = 1012,
+ REP_APPLY_METARECORD_FAILED = 1013,
+ REP_APPLY_NONCOMPLETE_GCIBUFFER = 1004,
+ REP_APPLY_NULL_GCIBUFFER = 1005,
+ REP_APPLIER_START_TRANSACTION = 1006,
+ REP_APPLIER_NO_TABLE = 1007,
+ REP_APPLIER_NO_OPERATION = 1007,
+ REP_APPLIER_EXECUTE_TRANSACTION = 1008,
+ REP_APPLIER_CREATE_TABLE = 1009,
+ REP_APPLIER_PREPARE_TABLE = 1010,
+ REP_DISCONNECT = 1011,
+ REQUESTOR_ILLEGAL_STATE_FOR_SLOWSTOP = 1200,
+ REQUESTOR_ILLEGAL_STATE_FOR_FASTSTOP = 1201,
+ REP_NOT_PROPER_TABLE = 1202,
+ REP_TABLE_ALREADY_SELECTED = 1203,
+ REP_TABLE_NOT_FOUND = 1204,
+
+ NOT_YET_IMPLEMENTED,
+ NO_OF_ERROR_CODES
+ };
+
+ struct ErrorDescription {
+ Code errCode;
+ const char * name;
+ };
+ static const ErrorDescription errorDescriptions[];
+ static const Uint32 noOfErrorDescs;
+ static const char * getErrorDesc(GrepError::Code err);
+
+};
+
+#endif
diff --git a/ndb/include/debugger/SignalLoggerManager.hpp b/ndb/include/debugger/SignalLoggerManager.hpp
new file mode 100644
index 00000000000..cf777505399
--- /dev/null
+++ b/ndb/include/debugger/SignalLoggerManager.hpp
@@ -0,0 +1,169 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//****************************************************************************
+//
+// .NAME
+// SignalLoggerManager - Handle signal loggers
+//
+//****************************************************************************
+#ifndef SignalLoggerManager_H
+#define SignalLoggerManager_H
+
+
+#include <NdbStdio.h>
+#include <kernel_types.h>
+#include <BlockNumbers.h>
+#include <TransporterDefinitions.hpp>
+
+class SignalLoggerManager
+{
+public:
+ SignalLoggerManager();
+ virtual ~SignalLoggerManager();
+
+ /**
+ * Sets output
+ * @Returns old output stream
+ */
+ FILE * setOutputStream(FILE * output);
+
+ /**
+ * Gets current output
+ */
+ FILE * getOutputStream() const;
+
+ void flushSignalLog();
+
+ /**
+ * For direct signals
+ * @See also SimulatedBlock EXECUTE_DIRECT
+ */
+ void executeDirect(const SignalHeader&,
+ Uint8 prio, const Uint32 * theData, Uint32 node);
+
+ /**
+ * For input signals
+ */
+ void executeSignal(const SignalHeader&, Uint8 prio,
+ const Uint32 * theData, Uint32 node,
+ const SegmentedSectionPtr ptr[3], Uint32 secs);
+
+ void executeSignal(const SignalHeader&, Uint8 prio,
+ const Uint32 * theData, Uint32 node,
+ const LinearSectionPtr ptr[3], Uint32 secs);
+
+ /**
+ * For output signals
+ */
+ void sendSignal(const SignalHeader&, Uint8 prio,
+ const Uint32 * theData, Uint32 node,
+ const SegmentedSectionPtr ptr[3], Uint32 secs);
+
+ void sendSignal(const SignalHeader&, Uint8 prio,
+ const Uint32 * theData, Uint32 node,
+ const LinearSectionPtr ptr[3], Uint32 secs);
+
+ /**
+ * For output signals
+ */
+ void sendSignalWithDelay(Uint32 delayInMilliSeconds,
+ const SignalHeader&,
+ Uint8 prio, const Uint32 * data, Uint32 node,
+ const SegmentedSectionPtr ptr[3], Uint32 secs);
+
+ /**
+ * Generic messages in the signal log
+ */
+ void log(BlockNumber bno, const char * msg);
+
+ /**
+ * LogModes
+ */
+ enum LogMode {
+ LogOff = 0,
+ LogIn = 1,
+ LogOut = 2,
+ LogInOut = 3
+ };
+
+ /**
+ * Returns no of loggers affected
+ */
+ int log(LogMode logMode, const char * params);
+ int logOn(bool allBlocks, BlockNumber bno, LogMode logMode);
+ int logOff(bool allBlocks, BlockNumber bno, LogMode logMode);
+ int logToggle(bool allBlocks, BlockNumber bno, LogMode logMode);
+
+ void setTrace(unsigned long trace);
+ unsigned long getTrace() const;
+
+ /**
+ * Print header
+ */
+ static void printSignalHeader(FILE * output,
+ const SignalHeader & sh,
+ Uint8 prio,
+ Uint32 node,
+ bool printReceiversSignalId);
+
+ /**
+ * Function for printing the Signal Data
+ */
+ static void printSignalData(FILE * out,
+ const SignalHeader & sh, const Uint32 *);
+
+ /**
+ * Print linear section.
+ */
+ static void printLinearSection(FILE * output,
+ const SignalHeader & sh,
+ const LinearSectionPtr ptr[3],
+ unsigned i);
+
+ /**
+ * Print segmented section.
+ */
+ static void printSegmentedSection(FILE * output,
+ const SignalHeader & sh,
+ const SegmentedSectionPtr ptr[3],
+ unsigned i);
+
+ /**
+ * Print data word in hex. Adds line break before the word
+ * when pos > 0 && pos % 7 == 0. Increments pos.
+ */
+ static void printDataWord(FILE * output, Uint32 & pos, const Uint32 data);
+
+private:
+ FILE * outputStream;
+ int log(int cmd, BlockNumber bno, LogMode logMode);
+
+ Uint32 traceId;
+ Uint8 logModes[NO_OF_BLOCKS];
+
+ inline bool
+ logMatch(BlockNumber bno, LogMode mask)
+ {
+ // avoid addressing outside logModes
+ return
+ bno < MIN_BLOCK_NO || bno > MAX_BLOCK_NO ||
+ (logModes[bno-MIN_BLOCK_NO] & mask);
+ }
+};
+
+#endif // SignalLoggerManager_H
+
diff --git a/ndb/include/editline/editline.h b/ndb/include/editline/editline.h
new file mode 100644
index 00000000000..2757e385968
--- /dev/null
+++ b/ndb/include/editline/editline.h
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* $Id: editline.h,v 1.1 2002/12/11 13:53:46 hin Exp $ */
+
+/*
+ * Public include file for editline, to be included instead of readline.h
+ */
+
+#ifndef __EDITLINE_H_INCLUDED__
+#define __EDITLINE_H_INCLUDED__
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+extern char *readline(const char *);
+extern void add_history(char *);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* !__EDITLINE_H_INCLUDED__ */
+
diff --git a/ndb/include/kernel/AttributeDescriptor.hpp b/ndb/include/kernel/AttributeDescriptor.hpp
new file mode 100644
index 00000000000..071d45e2607
--- /dev/null
+++ b/ndb/include/kernel/AttributeDescriptor.hpp
@@ -0,0 +1,250 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ATTRIBUTE_DESCRIPTOR_HPP
+#define ATTRIBUTE_DESCRIPTOR_HPP
+
+class AttributeDescriptor {
+ friend class Dbdict;
+ friend class Dbtup;
+ friend class Dbtux;
+
+private:
+ static void setType(Uint32 &, Uint32 type);
+ static void setSize(Uint32 &, Uint32 size);
+ static void setArray(Uint32 &, Uint32 arraySize);
+ static void setOriginal(Uint32 &, Uint32 original);
+ static void setNullable(Uint32 &, Uint32 nullable);
+ static void setDGroup(Uint32 &, Uint32 dgroup);
+ static void setDKey(Uint32 &, Uint32 dkey);
+ static void setPrimaryKey(Uint32 &, Uint32 dkey);
+ static void setStoredInTup(Uint32 &, Uint32 storedInTup);
+ static void setDynamic(Uint32 &, Uint32 dynamicInd);
+
+ static Uint32 getType(const Uint32 &);
+ static Uint32 getSize(const Uint32 &);
+ static Uint32 getSizeInWords(const Uint32 &);
+ static Uint32 getArrayType(const Uint32 &);
+ static Uint32 getArraySize(const Uint32 &);
+ static Uint32 getOriginal(const Uint32 &);
+ static Uint32 getNullable(const Uint32 &);
+ static Uint32 getDGroup(const Uint32 &);
+ static Uint32 getDKey(const Uint32 &);
+ static Uint32 getPrimaryKey(const Uint32 &);
+ static Uint32 getStoredInTup(const Uint32 &);
+ static Uint32 getDynamic(const Uint32 &);
+};
+
+/**
+ *
+ * a = Array type - 2 Bits -> Max 3 (Bit 0-1)
+ * t = Attribute type - 2 Bits -> Max 3 (Bit 2-3)
+ * s = Attribute size - 3 Bits -> Max 7 (Bit 4-6)
+ * o = Original attribute - 1 Bit 7
+ * n = Nullable - 1 Bit 8
+ * ? = Stored in tup - 1 Bit 9
+ * d = Disk based - 1 Bit 10
+ * g = Distribution Group Ind- 1 Bit 11
+ * k = Distribution Key Ind - 1 Bit 12
+ * r = Distribution group sz - 1 Bit 13
+ * p = Primary key attribute - 1 Bit 14
+ * y = Dynamic attribute - 1 Bit 15
+ * z = Array size - 16 Bits -> Max 65535 (Bit 16-31)
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * aattsss n dgkrpyzzzzzzzzzzzzzzzz
+ *
+ */
+
+#define AD_ARRAY_TYPE_SHIFT (0)
+#define AD_ARRAY_TYPE_MASK (3)
+
+#define AD_TYPE_SHIFT (2)
+#define AD_TYPE_MASK (3)
+
+#define AD_SIZE_SHIFT (4)
+#define AD_SIZE_MASK (7)
+
+#define AD_SIZE_IN_WORDS_OFFSET (31)
+#define AD_SIZE_IN_WORDS_SHIFT (5)
+
+#define AD_ORIGINAL_SHIFT (8)
+#define AD_NULLABLE_SHIFT (8)
+#define AD_TUP_STORED_SHIFT (9)
+
+#define AD_DISTR_GROUP_SHIFT (11)
+#define AD_DISTR_KEY_SHIFT (12)
+#define AD_DISTR_GROUP_SZ (13)
+#define AD_PRIMARY_KEY (14)
+#define AD_DYNAMIC (15)
+
+#define AD_ARRAY_SIZE_SHIFT (16)
+#define AD_ARRAY_SIZE_MASK (65535)
+
+inline
+void
+AttributeDescriptor::setType(Uint32 & desc, Uint32 type){
+ ASSERT_MAX(type, AD_TYPE_MASK, "AttributeDescriptor::setType");
+ desc |= (type << AD_TYPE_SHIFT);
+}
+
+inline
+void
+AttributeDescriptor::setSize(Uint32 & desc, Uint32 size){
+ ASSERT_MAX(size, AD_SIZE_MASK, "AttributeDescriptor::setSize");
+ desc |= (size << AD_SIZE_SHIFT);
+}
+
+inline
+void
+AttributeDescriptor::setArray(Uint32 & desc, Uint32 size){
+ ASSERT_MAX(size, AD_ARRAY_SIZE_MASK, "AttributeDescriptor::setArray");
+ desc |= (size << AD_ARRAY_SIZE_SHIFT);
+ if(size <= 1){
+ desc |= (size << AD_ARRAY_TYPE_SHIFT);
+ } else {
+ desc |= (2 << AD_ARRAY_TYPE_SHIFT);
+ }
+}
+
+inline
+void
+AttributeDescriptor::setNullable(Uint32 & desc, Uint32 nullable){
+ ASSERT_BOOL(nullable, "AttributeDescriptor::setNullable");
+ desc |= (nullable << AD_NULLABLE_SHIFT);
+}
+
+inline
+void
+AttributeDescriptor::setOriginal(Uint32 & desc, Uint32 original){
+ ASSERT_BOOL(original, "AttributeDescriptor::setOriginal");
+ desc |= (original << AD_ORIGINAL_SHIFT);
+}
+
+inline
+void
+AttributeDescriptor::setDGroup(Uint32 & desc, Uint32 dgroup){
+ ASSERT_BOOL(dgroup, "AttributeDescriptor::setDGroup");
+ desc |= (dgroup << AD_DISTR_GROUP_SHIFT);
+}
+
+inline
+void
+AttributeDescriptor::setDKey(Uint32 & desc, Uint32 dkey){
+ ASSERT_BOOL(dkey, "AttributeDescriptor::setDKey");
+ desc |= (dkey << AD_DISTR_KEY_SHIFT);
+}
+
+inline
+void
+AttributeDescriptor::setPrimaryKey(Uint32 & desc, Uint32 dkey){
+ ASSERT_BOOL(dkey, "AttributeDescriptor::setPrimaryKey");
+ desc |= (dkey << AD_PRIMARY_KEY);
+}
+
+inline
+void
+AttributeDescriptor::setStoredInTup(Uint32 & desc, Uint32 storedInTup){
+ ASSERT_BOOL(storedInTup, "AttributeDescriptor::setStoredInTup");
+ desc |= (storedInTup << AD_TUP_STORED_SHIFT);
+}
+
+inline
+void
+AttributeDescriptor::setDynamic(Uint32 & desc, Uint32 dynamic){
+ ASSERT_BOOL(dynamic, "AttributeDescriptor::setDynamic");
+ desc |= (dynamic << AD_DYNAMIC);
+}
+
+/**
+ * Getters
+ */
+inline
+Uint32
+AttributeDescriptor::getType(const Uint32 & desc){
+ return (desc >> AD_TYPE_SHIFT) & AD_TYPE_MASK;
+}
+
+inline
+Uint32
+AttributeDescriptor::getSize(const Uint32 & desc){
+ return (desc >> AD_SIZE_SHIFT) & AD_SIZE_MASK;
+}
+
+inline
+Uint32
+AttributeDescriptor::getSizeInWords(const Uint32 & desc){
+ return ((getArraySize(desc) << getSize(desc))
+ + AD_SIZE_IN_WORDS_OFFSET)
+ >> AD_SIZE_IN_WORDS_SHIFT;
+}
+
+inline
+Uint32
+AttributeDescriptor::getArrayType(const Uint32 & desc){
+ return (desc >> AD_ARRAY_TYPE_SHIFT) & AD_ARRAY_TYPE_MASK;
+}
+
+inline
+Uint32
+AttributeDescriptor::getArraySize(const Uint32 & desc){
+ return (desc >> AD_ARRAY_SIZE_SHIFT) & AD_ARRAY_SIZE_MASK;
+}
+
+inline
+Uint32
+AttributeDescriptor::getNullable(const Uint32 & desc){
+ return (desc >> AD_NULLABLE_SHIFT) & 1;
+}
+
+inline
+Uint32
+AttributeDescriptor::getOriginal(const Uint32 & desc){
+ return (desc >> AD_ORIGINAL_SHIFT) & 1;
+}
+
+inline
+Uint32
+AttributeDescriptor::getDGroup(const Uint32 & desc){
+ return (desc >> AD_DISTR_GROUP_SHIFT) & 1;
+}
+
+inline
+Uint32
+AttributeDescriptor::getDKey(const Uint32 & desc){
+ return (desc >> AD_DISTR_KEY_SHIFT) & 1;
+}
+
+inline
+Uint32
+AttributeDescriptor::getPrimaryKey(const Uint32 & desc){
+ return (desc >> AD_PRIMARY_KEY) & 1;
+}
+
+inline
+Uint32
+AttributeDescriptor::getDynamic(const Uint32 & desc){
+ return (desc >> AD_DYNAMIC) & 1;
+}
+
+inline
+Uint32
+AttributeDescriptor::getStoredInTup(const Uint32 & desc){
+ return (desc >> AD_TUP_STORED_SHIFT) & 1;
+}
+
+#endif
diff --git a/ndb/include/kernel/AttributeHeader.hpp b/ndb/include/kernel/AttributeHeader.hpp
new file mode 100644
index 00000000000..91190fdd223
--- /dev/null
+++ b/ndb/include/kernel/AttributeHeader.hpp
@@ -0,0 +1,204 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ATTRIBUTE_HEADER
+#define ATTRIBUTE_HEADER
+
+#include <new>
+/**
+ * @class AttributeHeader
+ * @brief Header passed in front of every attribute value in AttrInfo signal
+ */
+class AttributeHeader {
+ friend class Dbtup;
+ friend class Backup;
+ friend class NdbOperation;
+ friend class DbUtil;
+ friend class Suma;
+
+public:
+ /** Initialize AttributeHeader at location aHeaderPtr */
+ static AttributeHeader& init(void* aHeaderPtr, Uint32 anAttributeId,
+ Uint32 aDataSize);
+
+ /** Returns size of AttributeHeader (usually one or two words) */
+ Uint32 getHeaderSize() const; // In 32-bit words
+
+ /** Store AttributeHeader in location given as argument */
+ void insertHeader(Uint32*);
+
+ /** Get next attribute header (if there is one) */
+ AttributeHeader* getNext() const;
+
+ /** Get location of attribute value */
+ Uint32* getDataPtr() const;
+
+ /** Getters and Setters */
+ Uint32 getAttributeId() const;
+ void setAttributeId(Uint32);
+ Uint32 getDataSize() const; // In 32-bit words
+ void setDataSize(Uint32);
+ bool isNULL() const;
+ void setNULL();
+
+ /** Print **/
+ //void print(NdbOut&);
+ void print(FILE*);
+
+ static Uint32 getDataSize(Uint32);
+
+public:
+ AttributeHeader(Uint32 = 0);
+ AttributeHeader(Uint32 anAttributeId, Uint32 aDataSize);
+ ~AttributeHeader();
+
+ Uint32 m_value;
+};
+
+/**
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * ssssssssssssss eiiiiiiiiiiiiiiii
+ *
+ * i = Attribute Id
+ * s = Size of current "chunk" - 14 Bits -> 16384 (words) = 65k
+ * Including optional extra word(s).
+ * e - Element data/Blob, read element of array
+ * If == 0 next data word contains attribute value.
+ * If == 1 next data word contains:
+ * For Array of Fixed size Elements
+ * Start Index (16 bit), Stop Index(16 bit)
+ * For Blob
+ * Start offset (32 bit) (length is defined in previous word)
+ *
+ * An attribute value equal to "null" is represented by setting s == 0.
+ *
+ * Bit 14 is not yet used.
+ */
+
+inline
+AttributeHeader& AttributeHeader::init(void* aHeaderPtr, Uint32 anAttributeId,
+ Uint32 aDataSize)
+{
+ return * new (aHeaderPtr) AttributeHeader(anAttributeId, aDataSize);
+}
+
+inline
+AttributeHeader::AttributeHeader(Uint32 aHeader)
+{
+ m_value = aHeader;
+}
+
+inline
+AttributeHeader::AttributeHeader(Uint32 anAttributeId, Uint32 aDataSize)
+{
+ m_value = 0;
+ this->setAttributeId(anAttributeId);
+ this->setDataSize(aDataSize);
+}
+
+inline
+AttributeHeader::~AttributeHeader()
+{}
+
+inline
+Uint32 AttributeHeader::getHeaderSize() const
+{
+ // Should check 'e' bit here
+ return 1;
+}
+
+inline
+Uint32 AttributeHeader::getAttributeId() const
+{
+ return (m_value & 0xFFFF0000) >> 16;
+}
+
+inline
+void AttributeHeader::setAttributeId(Uint32 anAttributeId)
+{
+ m_value &= 0x0000FFFF; // Clear attribute id
+ m_value |= (anAttributeId << 16);
+}
+
+inline
+Uint32 AttributeHeader::getDataSize() const
+{
+ return (m_value & 0x3FFF);
+}
+
+inline
+void AttributeHeader::setDataSize(Uint32 aDataSize)
+{
+ m_value &= (~0x3FFF);
+ m_value |= aDataSize;
+}
+
+inline
+bool AttributeHeader::isNULL() const
+{
+ return (getDataSize() == 0);
+}
+
+inline
+void AttributeHeader::setNULL()
+{
+ setDataSize(0);
+}
+
+inline
+Uint32* AttributeHeader::getDataPtr() const
+{
+ return (Uint32*)&m_value + getHeaderSize();
+}
+
+inline
+void AttributeHeader::insertHeader(Uint32* target)
+{
+ *target = m_value;
+}
+
+inline
+AttributeHeader*
+AttributeHeader::getNext() const {
+ return (AttributeHeader*)(getDataPtr() + getDataSize());
+}
+
+inline
+void
+//AttributeHeader::print(NdbOut& output) {
+AttributeHeader::print(FILE* output) {
+ fprintf(output, "AttributeId: H\'%.8x (D\'%d), DataSize: H\'%.8x (D\'%d), "
+ "isNULL: %d\n",
+ getAttributeId(), getAttributeId(),
+ getDataSize(), getDataSize(),
+ isNULL());
+}
+
+inline
+Uint32
+AttributeHeader::getDataSize(Uint32 m_value){
+ return (m_value & 0x3FFF);
+}
+
+#endif
+
+
+
+
+
+
+
diff --git a/ndb/include/kernel/AttributeList.hpp b/ndb/include/kernel/AttributeList.hpp
new file mode 100644
index 00000000000..7c6f71df3d2
--- /dev/null
+++ b/ndb/include/kernel/AttributeList.hpp
@@ -0,0 +1,32 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ATTRIBUTE_LIST_HPP
+#define ATTRIBUTE_LIST_HPP
+
+/**
+ * Masks and lists used by index and trigger. Must be plain old Uint32 data.
+ * XXX depends on other headers XXX move to some common file
+ */
+
+typedef Bitmask<MAXNROFATTRIBUTESINWORDS> AttributeMask;
+
+struct AttributeList {
+ Uint32 sz;
+ Uint32 id[MAX_ATTRIBUTES_IN_INDEX];
+};
+
+#endif
diff --git a/ndb/include/kernel/BlockNumbers.h b/ndb/include/kernel/BlockNumbers.h
new file mode 100644
index 00000000000..84c3fc656a9
--- /dev/null
+++ b/ndb/include/kernel/BlockNumbers.h
@@ -0,0 +1,81 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BLOCK_NUMBERS_H
+#define BLOCK_NUMBERS_H
+
+#include <kernel_types.h>
+#include <RefConvert.hpp>
+
+// 240
+#define MIN_API_BLOCK_NO 0x8000
+
+// 2047
+#define API_PACKED 0x07ff
+
+// 4002
+#define API_CLUSTERMGR 0x0FA2
+
+#define BACKUP 0xF4
+#define DBTC 0xF5
+#define DBDIH 0xF6
+#define DBLQH 0xF7
+#define DBACC 0xF8
+#define DBTUP 0xF9
+#define DBDICT 0xFA
+#define NDBCNTR 0xFB
+#define QMGR 0xFC
+#define NDBFS 0xFD
+#define CMVMI 0xFE
+#define TRIX 0xFF
+#define DBUTIL 0x100
+#define SUMA 0x101
+#define GREP 0x102
+#define DBTUX 0x103
+
+const BlockReference BACKUP_REF = numberToRef(BACKUP, 0);
+const BlockReference DBTC_REF = numberToRef(DBTC, 0);
+const BlockReference DBDIH_REF = numberToRef(DBDIH, 0);
+const BlockReference DBLQH_REF = numberToRef(DBLQH, 0);
+const BlockReference DBACC_REF = numberToRef(DBACC, 0);
+const BlockReference DBTUP_REF = numberToRef(DBTUP, 0);
+const BlockReference DBDICT_REF = numberToRef(DBDICT, 0);
+const BlockReference NDBCNTR_REF = numberToRef(NDBCNTR, 0);
+const BlockReference QMGR_REF = numberToRef(QMGR, 0);
+const BlockReference NDBFS_REF = numberToRef(NDBFS, 0);
+const BlockReference CMVMI_REF = numberToRef(CMVMI, 0);
+const BlockReference TRIX_REF = numberToRef(TRIX, 0);
+const BlockReference DBUTIL_REF = numberToRef(DBUTIL, 0);
+const BlockReference SUMA_REF = numberToRef(SUMA, 0);
+const BlockReference GREP_REF = numberToRef(GREP, 0);
+const BlockReference DBTUX_REF = numberToRef(DBTUX, 0);
+
+const BlockNumber MIN_BLOCK_NO = BACKUP;
+const BlockNumber MAX_BLOCK_NO = DBTUX;
+const BlockNumber NO_OF_BLOCKS = (MAX_BLOCK_NO - MIN_BLOCK_NO + 1);
+
+/**
+ * Used for printing and stuff
+ */
+struct BlockName {
+ const char* name;
+ BlockNumber number;
+};
+
+extern const BlockName BlockNames[];
+extern const BlockNumber NO_OF_BLOCK_NAMES;
+
+#endif
diff --git a/ndb/include/kernel/GlobalSignalNumbers.h b/ndb/include/kernel/GlobalSignalNumbers.h
new file mode 100644
index 00000000000..87385de1f14
--- /dev/null
+++ b/ndb/include/kernel/GlobalSignalNumbers.h
@@ -0,0 +1,957 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GLOBAL_SIGNAL_NUMBERS_H
+#define GLOBAL_SIGNAL_NUMBERS_H
+
+#include <kernel_types.h>
+/**
+ * NOTE
+ *
+ * When adding a new signal, remember to update MAX_GSN and SignalNames.cpp
+ */
+
+
+
+const GlobalSignalNumber MAX_GSN = 712;
+
+
+
+
+struct GsnName {
+ GlobalSignalNumber gsn;
+ const char * name;
+};
+
+extern const GsnName SignalNames[];
+extern const GlobalSignalNumber NO_OF_SIGNAL_NAMES;
+
+/**
+ * These are used by API and kernel
+ */
+#define GSN_API_REGCONF 1
+#define GSN_API_REGREF 2
+#define GSN_API_REGREQ 3
+
+#define GSN_ATTRINFO 4
+#define GSN_TRANSID_AI 5
+#define GSN_KEYINFO 6
+#define GSN_READCONF 7
+
+#define GSN_TCKEY_FAILCONF 8
+#define GSN_TCKEY_FAILREF 9
+#define GSN_TCKEYCONF 10
+#define GSN_TCKEYREF 11
+#define GSN_TCKEYREQ 12
+
+#define GSN_TCROLLBACKCONF 13
+#define GSN_TCROLLBACKREF 14
+#define GSN_TCROLLBACKREQ 15
+#define GSN_TCROLLBACKREP 16
+
+#define GSN_TC_COMMITCONF 17
+#define GSN_TC_COMMITREF 18
+#define GSN_TC_COMMITREQ 19
+#define GSN_TC_HBREP 20
+
+#define GSN_TRANSID_AI_R 21
+#define GSN_KEYINFO20_R 22
+
+#define GSN_GET_TABINFOREF 23
+#define GSN_GET_TABINFOREQ 24
+#define GSN_GET_TABINFO_CONF 190
+
+#define GSN_GET_TABLEID_REQ 683
+#define GSN_GET_TABLEID_REF 684
+#define GSN_GET_TABLEID_CONF 685
+
+#define GSN_DIHNDBTAMPER 25
+#define GSN_NODE_FAILREP 26
+#define GSN_NF_COMPLETEREP 27
+
+#define GSN_SCAN_NEXTREQ 28
+#define GSN_SCAN_TABCONF 29
+#define GSN_SCAN_TABINFO 30
+#define GSN_SCAN_TABREF 31
+#define GSN_SCAN_TABREQ 32
+#define GSN_KEYINFO20 33
+
+#define GSN_TCRELEASECONF 34
+#define GSN_TCRELEASEREF 35
+#define GSN_TCRELEASEREQ 36
+
+#define GSN_TCSEIZECONF 37
+#define GSN_TCSEIZEREF 38
+#define GSN_TCSEIZEREQ 39
+
+// 40 unused
+// 41 unused
+// 42 unused
+// 43 unused
+// 44 unused
+// 45 unused
+// 46 unused
+// 47 unused
+// 48 unused
+// 49 unused
+// 50 unused
+// 51 unused
+// 52 unused
+// 53 unused
+// 54 unused
+// 55 unused
+// 56 unused
+// 57 unused
+// 58 unused
+// 59 unused
+// 60 unused
+// 61 unused
+// 62 unused
+// 63 unused
+// 64 unused
+// 65 unused
+// 66 unused
+
+/**
+ * These are used only by kernel
+ */
+
+#define GSN_ACC_ABORTCONF 67
+// 68 unused
+// 69 unused
+// 70 unused
+#define GSN_ACC_ABORTREQ 71
+#define GSN_ACC_CHECK_SCAN 72
+#define GSN_ACC_COMMITCONF 73
+#define GSN_ACC_COMMITREQ 74
+#define GSN_ACC_CONTOPCONF 75
+#define GSN_ACC_CONTOPREQ 76
+#define GSN_ACC_LCPCONF 77
+#define GSN_ACC_LCPREF 78
+#define GSN_ACC_LCPREQ 79
+#define GSN_ACC_LCPSTARTED 80
+#define GSN_ACC_OVER_REC 81
+
+#define GSN_ACC_SAVE_PAGES 83
+#define GSN_ACC_SCAN_INFO 84
+#define GSN_ACC_SCAN_INFO24 85
+#define GSN_ACC_SCANCONF 86
+#define GSN_ACC_SCANREF 87
+#define GSN_ACC_SCANREQ 88
+#define GSN_ACC_SRCONF 89
+#define GSN_ACC_SRREF 90
+#define GSN_ACC_SRREQ 91
+#define GSN_ACC_TO_CONF 92
+#define GSN_ACC_TO_REF 93
+#define GSN_ACC_TO_REQ 94
+#define GSN_ACCFRAGCONF 95
+#define GSN_ACCFRAGREF 96
+#define GSN_ACCFRAGREQ 97
+#define GSN_ACCKEYCONF 98
+#define GSN_ACCKEYREF 99
+#define GSN_ACCKEYREQ 100
+#define GSN_ACCMINUPDATE 101
+#define GSN_ACCSEIZECONF 103
+#define GSN_ACCSEIZEREF 104
+#define GSN_ACCSEIZEREQ 105
+#define GSN_ACCUPDATECONF 106
+#define GSN_ACCUPDATEKEY 107
+#define GSN_ACCUPDATEREF 108
+
+#define GSN_ADD_FRAGCONF 109
+#define GSN_ADD_FRAGREF 110
+#define GSN_ADD_FRAGREQ 111
+
+#define GSN_API_FAILCONF 113
+#define GSN_API_FAILREQ 114
+#define GSN_APPL_CHANGEREP 115
+// 116 not unused
+#define GSN_APPL_HB 117
+#define GSN_APPL_HBREQ 118
+#define GSN_APPL_REGCONF 119
+#define GSN_APPL_REGREF 120
+#define GSN_APPL_REGREQ 121
+#define GSN_APPL_RUN 122
+#define GSN_APPL_STARTCONF 123
+#define GSN_APPL_STARTREG 124
+#define GSN_CHECK_LCP_STOP 125
+#define GSN_CLOSE_COMCONF 126
+#define GSN_CLOSE_COMREQ 127
+#define GSN_CM_ACKADD 128
+#define GSN_CM_ACKALARM 129
+#define GSN_CM_ADD 130
+#define GSN_CM_APPCHG 131
+// 132 not unused
+// 133 not unused
+#define GSN_CM_HEARTBEAT 134
+#define GSN_CM_INFOCONF 135
+#define GSN_CM_INFOREQ 136
+#define GSN_CM_INIT 137
+#define GSN_CM_NODEINFOCONF 138
+#define GSN_CM_NODEINFOREF 139
+#define GSN_CM_NODEINFOREQ 140
+#define GSN_CM_REGCONF 141
+#define GSN_CM_REGREF 142
+#define GSN_CM_REGREQ 143
+#define GSN_CM_RUN 144
+#define GSN_CMVMI_CFGCONF 145
+#define GSN_CMVMI_CFGREQ 146
+#define GSN_CNTR_CHANGEREP 147
+#define GSN_CNTR_MASTERCONF 148
+#define GSN_CNTR_MASTERREF 149
+#define GSN_CNTR_MASTERREQ 150
+#define GSN_CNTR_WAITREP 151
+#define GSN_COMMIT 152
+#define GSN_COMMIT_FAILCONF 153
+#define GSN_COMMIT_FAILREQ 154
+#define GSN_COMMITCONF 155
+#define GSN_COMMITREQ 156
+#define GSN_COMMITTED 157
+#define GSN_COMPLETE 159
+#define GSN_COMPLETECONF 160
+#define GSN_COMPLETED 161
+#define GSN_COMPLETEREQ 162
+#define GSN_CONNECT_REP 163
+#define GSN_CONTINUEB 164
+// 165 not unused
+#define GSN_COPY_ACTIVECONF 166
+#define GSN_COPY_ACTIVEREF 167
+#define GSN_COPY_ACTIVEREQ 168
+#define GSN_COPY_FRAGCONF 169
+#define GSN_COPY_FRAGREF 170
+#define GSN_COPY_FRAGREQ 171
+#define GSN_COPY_GCICONF 172
+#define GSN_COPY_GCIREQ 173
+#define GSN_COPY_STATECONF 174
+#define GSN_COPY_STATEREQ 175
+#define GSN_COPY_TABCONF 176
+#define GSN_COPY_TABREQ 177
+#define GSN_CREATE_FRAGCONF 178
+#define GSN_CREATE_FRAGREF 179
+#define GSN_CREATE_FRAGREQ 180
+#define GSN_DEBUG_SIG 181
+#define GSN_DI_FCOUNTCONF 182
+#define GSN_DI_FCOUNTREF 183
+#define GSN_DI_FCOUNTREQ 184
+#define GSN_DIADDTABCONF 185
+#define GSN_DIADDTABREF 186
+#define GSN_DIADDTABREQ 187
+// 188 not unused
+// 189 not unused
+// 190 not unused
+#define GSN_DICTSTARTCONF 191
+#define GSN_DICTSTARTREQ 192
+
+#define GSN_LIST_TABLES_REQ 193
+#define GSN_LIST_TABLES_CONF 194
+
+#define GSN_ABORT 195
+#define GSN_ABORTCONF 196
+#define GSN_ABORTED 197
+#define GSN_ABORTREQ 198
+
+/******************************************
+ * DROP TABLE
+ *
+ */
+
+/**
+ * This is drop table's public interface
+ */
+#define GSN_DROP_TABLE_REQ 82
+#define GSN_DROP_TABLE_REF 102
+#define GSN_DROP_TABLE_CONF 112
+
+/**
+ * This is used for implementing drop table
+ */
+#define GSN_PREP_DROP_TAB_REQ 199
+#define GSN_PREP_DROP_TAB_REF 200
+#define GSN_PREP_DROP_TAB_CONF 201
+
+#define GSN_DROP_TAB_REQ 202
+#define GSN_DROP_TAB_REF 203
+#define GSN_DROP_TAB_CONF 204
+
+#define GSN_WAIT_DROP_TAB_REQ 208
+#define GSN_WAIT_DROP_TAB_REF 209
+#define GSN_WAIT_DROP_TAB_CONF 216
+
+/*****************************************/
+
+#define GSN_UPDATE_TOCONF 205
+#define GSN_UPDATE_TOREF 206
+#define GSN_UPDATE_TOREQ 207
+
+#define GSN_DIGETNODESCONF 210
+#define GSN_DIGETNODESREF 211
+#define GSN_DIGETNODESREQ 212
+#define GSN_DIGETPRIMCONF 213
+#define GSN_DIGETPRIMREF 214
+#define GSN_DIGETPRIMREQ 215
+
+#define GSN_DIH_RESTARTCONF 217
+#define GSN_DIH_RESTARTREF 218
+#define GSN_DIH_RESTARTREQ 219
+
+// 220 not unused
+// 221 not unused
+// 222 not unused
+
+#define GSN_EMPTY_LCP_REQ 223
+#define GSN_EMPTY_LCP_CONF 224
+
+#define GSN_SCHEMA_INFO 225
+#define GSN_SCHEMA_INFOCONF 226
+
+#define GSN_MASTER_GCPCONF 227
+#define GSN_MASTER_GCPREF 228
+#define GSN_MASTER_GCPREQ 229
+
+// 230 not unused
+// 231 not unused
+
+#define GSN_DIRELEASECONF 232
+#define GSN_DIRELEASEREF 233
+#define GSN_DIRELEASEREQ 234
+#define GSN_DISCONNECT_REP 235
+#define GSN_DISEIZECONF 236
+#define GSN_DISEIZEREF 237
+#define GSN_DISEIZEREQ 238
+#define GSN_DIVERIFYCONF 239
+#define GSN_DIVERIFYREF 240
+#define GSN_DIVERIFYREQ 241
+#define GSN_ENABLE_COMORD 242
+#define GSN_END_LCPCONF 243
+#define GSN_END_LCPREQ 244
+#define GSN_END_TOCONF 245
+#define GSN_END_TOREQ 246
+#define GSN_EVENT_REP 247
+#define GSN_EXEC_FRAGCONF 248
+#define GSN_EXEC_FRAGREF 249
+#define GSN_EXEC_FRAGREQ 250
+#define GSN_EXEC_SRCONF 251
+#define GSN_EXEC_SRREQ 252
+#define GSN_EXPANDCHECK2 253
+#define GSN_FAIL_REP 254
+#define GSN_FSCLOSECONF 255
+#define GSN_FSCLOSEREF 256
+#define GSN_FSCLOSEREQ 257
+#define GSN_FSAPPENDCONF 258
+#define GSN_FSOPENCONF 259
+#define GSN_FSOPENREF 260
+#define GSN_FSOPENREQ 261
+#define GSN_FSREADCONF 262
+#define GSN_FSREADREF 263
+#define GSN_FSREADREQ 264
+#define GSN_FSSYNCCONF 265
+#define GSN_FSSYNCREF 266
+#define GSN_FSSYNCREQ 267
+#define GSN_FSAPPENDREQ 268
+#define GSN_FSAPPENDREF 269
+#define GSN_FSWRITECONF 270
+#define GSN_FSWRITEREF 271
+#define GSN_FSWRITEREQ 272
+#define GSN_GCP_ABORT 273
+#define GSN_GCP_ABORTED 274
+#define GSN_GCP_COMMIT 275
+#define GSN_GCP_NODEFINISH 276
+#define GSN_GCP_NOMORETRANS 277
+#define GSN_GCP_PREPARE 278
+#define GSN_GCP_PREPARECONF 279
+#define GSN_GCP_PREPAREREF 280
+#define GSN_GCP_SAVECONF 281
+#define GSN_GCP_SAVEREF 282
+#define GSN_GCP_SAVEREQ 283
+#define GSN_GCP_TCFINISHED 284
+#define GSN_SR_FRAGIDCONF 285
+#define GSN_SR_FRAGIDREF 286
+#define GSN_SR_FRAGIDREQ 287
+#define GSN_GETGCICONF 288
+#define GSN_GETGCIREQ 289
+#define GSN_HOT_SPAREREP 290
+#define GSN_INCL_NODECONF 291
+#define GSN_INCL_NODEREF 292
+#define GSN_INCL_NODEREQ 293
+#define GSN_LCP_FRAGIDCONF 294
+#define GSN_LCP_FRAGIDREF 295
+#define GSN_LCP_FRAGIDREQ 296
+#define GSN_LCP_HOLDOPCONF 297
+#define GSN_LCP_HOLDOPREF 298
+#define GSN_LCP_HOLDOPREQ 299
+#define GSN_SHRINKCHECK2 301
+#define GSN_GET_SCHEMA_INFOREQ 302
+// 303 not unused
+// 304 not unused
+#define GSN_LQH_RESTART_OP 305
+#define GSN_LQH_TRANSCONF 306
+#define GSN_LQH_TRANSREQ 307
+#define GSN_LQHADDATTCONF 308
+#define GSN_LQHADDATTREF 309
+#define GSN_LQHADDATTREQ 310
+#define GSN_LQHFRAGCONF 311
+#define GSN_LQHFRAGREF 312
+#define GSN_LQHFRAGREQ 313
+#define GSN_LQHKEYCONF 314
+#define GSN_LQHKEYREF 315
+#define GSN_LQHKEYREQ 316
+
+#define GSN_MASTER_LCPCONF 318
+#define GSN_MASTER_LCPREF 319
+#define GSN_MASTER_LCPREQ 320
+
+#define GSN_MEMCHECKCONF 321
+#define GSN_MEMCHECKREQ 322
+#define GSN_NDB_FAILCONF 323
+#define GSN_NDB_STARTCONF 324
+#define GSN_NDB_STARTREF 325
+#define GSN_NDB_STARTREQ 326
+#define GSN_NDB_STTOR 327
+#define GSN_NDB_STTORRY 328
+#define GSN_NDB_TAMPER 329
+#define GSN_NEXT_SCANCONF 330
+#define GSN_NEXT_SCANREF 331
+#define GSN_NEXT_SCANREQ 332
+#define GSN_NEXTOPERATION 333
+#define GSN_SIZEALT_ACK 334
+#define GSN_SIZEALT_REP 335
+#define GSN_NODE_STATESCONF 336
+#define GSN_NODE_STATESREF 337
+#define GSN_NODE_STATESREQ 338
+#define GSN_OPEN_COMCONF 339
+#define GSN_OPEN_COMREF 340
+#define GSN_OPEN_COMREQ 341
+#define GSN_PACKED_SIGNAL 342
+#define GSN_PREP_FAILCONF 343
+#define GSN_PREP_FAILREF 344
+#define GSN_PREP_FAILREQ 345
+#define GSN_PRES_TOCONF 346
+#define GSN_PRES_TOREQ 347
+#define GSN_READ_NODESCONF 348
+#define GSN_READ_NODESREF 349
+#define GSN_READ_NODESREQ 350
+#define GSN_SCAN_FRAGCONF 351
+#define GSN_SCAN_FRAGREF 352
+#define GSN_SCAN_FRAGREQ 353
+#define GSN_SCAN_HBREP 354
+#define GSN_SCAN_PROCCONF 355
+#define GSN_SCAN_PROCREQ 356
+#define GSN_SEND_PACKED 357
+#define GSN_SET_LOGLEVELORD 358
+
+#define GSN_LQH_ALLOCREQ 359
+#define GSN_TUP_ALLOCREQ 360
+#define GSN_TUP_DEALLOCREQ 361
+
+// 362 not unused
+
+#define GSN_TUP_WRITELOG_REQ 363
+#define GSN_LQH_WRITELOG_REQ 364
+
+#define GSN_LCP_FRAG_REP 300
+#define GSN_LCP_FRAG_ORD 365
+#define GSN_LCP_COMPLETE_REP 158
+
+#define GSN_START_LCP_REQ 317
+#define GSN_START_LCP_CONF 366
+
+#define GSN_UNBLO_DICTCONF 367
+#define GSN_UNBLO_DICTREQ 368
+#define GSN_START_COPYCONF 369
+#define GSN_START_COPYREF 370
+#define GSN_START_COPYREQ 371
+#define GSN_START_EXEC_SR 372
+#define GSN_START_FRAGCONF 373
+#define GSN_START_FRAGREF 374
+#define GSN_START_FRAGREQ 375
+#define GSN_START_LCP_REF 376
+#define GSN_START_LCP_ROUND 377
+#define GSN_START_MECONF 378
+#define GSN_START_MEREF 379
+#define GSN_START_MEREQ 380
+#define GSN_START_PERMCONF 381
+#define GSN_START_PERMREF 382
+#define GSN_START_PERMREQ 383
+#define GSN_START_RECCONF 384
+#define GSN_START_RECREF 385
+#define GSN_START_RECREQ 386
+#define GSN_START_TOCONF 387
+#define GSN_START_TOREQ 388
+#define GSN_STORED_PROCCONF 389
+#define GSN_STORED_PROCREF 390
+#define GSN_STORED_PROCREQ 391
+#define GSN_STTOR 392
+#define GSN_STTORRY 393
+#define GSN_BACKUP_TRIG_REQ 394
+#define GSN_SYSTEM_ERROR 395
+#define GSN_TAB_COMMITCONF 396
+#define GSN_TAB_COMMITREF 397
+#define GSN_TAB_COMMITREQ 398
+#define GSN_TAKE_OVERTCCONF 399
+#define GSN_TAKE_OVERTCREQ 400
+#define GSN_TC_CLOPSIZECONF 401
+#define GSN_TC_CLOPSIZEREQ 402
+#define GSN_TC_SCHVERCONF 403
+#define GSN_TC_SCHVERREQ 404
+#define GSN_TCGETOPSIZECONF 405
+#define GSN_TCGETOPSIZEREQ 406
+#define GSN_TEST_ORD 407
+#define GSN_TESTSIG 408
+#define GSN_TIME_SIGNAL 409
+#define GSN_VOTE_MASTERORD 410
+// 411 unused
+// 412 unused
+#define GSN_TUP_ABORTREQ 414
+#define GSN_TUP_ADD_ATTCONF 415
+#define GSN_TUP_ADD_ATTRREF 416
+#define GSN_TUP_ADD_ATTRREQ 417
+#define GSN_TUP_ATTRINFO 418
+#define GSN_TUP_COMMITREQ 419
+// 420 unused
+#define GSN_TUP_LCPCONF 421
+#define GSN_TUP_LCPREF 422
+#define GSN_TUP_LCPREQ 423
+#define GSN_TUP_LCPSTARTED 424
+#define GSN_TUP_PREPLCPCONF 425
+#define GSN_TUP_PREPLCPREF 426
+#define GSN_TUP_PREPLCPREQ 427
+#define GSN_TUP_SRCONF 428
+#define GSN_TUP_SRREF 429
+#define GSN_TUP_SRREQ 430
+#define GSN_TUPFRAGCONF 431
+#define GSN_TUPFRAGREF 432
+#define GSN_TUPFRAGREQ 433
+#define GSN_TUPKEYCONF 434
+#define GSN_TUPKEYREF 435
+#define GSN_TUPKEYREQ 436
+#define GSN_TUPRELEASECONF 437
+#define GSN_TUPRELEASEREF 438
+#define GSN_TUPRELEASEREQ 439
+#define GSN_TUPSEIZECONF 440
+#define GSN_TUPSEIZEREF 441
+#define GSN_TUPSEIZEREQ 442
+
+#define GSN_ABORT_ALL_REQ 445
+#define GSN_ABORT_ALL_REF 446
+#define GSN_ABORT_ALL_CONF 447
+
+#define GSN_STATISTICS_REQ 448
+#define GSN_STOP_ORD 449
+#define GSN_TAMPER_ORD 450
+#define GSN_SET_VAR_REQ 451
+#define GSN_SET_VAR_CONF 452
+#define GSN_SET_VAR_REF 453
+#define GSN_STATISTICS_CONF 454
+
+#define GSN_START_ORD 455
+// 456 unused
+// 457 unused
+
+#define GSN_EVENT_SUBSCRIBE_REQ 458
+#define GSN_EVENT_SUBSCRIBE_CONF 459
+#define GSN_EVENT_SUBSCRIBE_REF 460
+#define GSN_ACC_COM_BLOCK 461
+#define GSN_ACC_COM_UNBLOCK 462
+#define GSN_TUP_COM_BLOCK 463
+#define GSN_TUP_COM_UNBLOCK 464
+
+#define GSN_DUMP_STATE_ORD 465
+
+#define GSN_START_INFOREQ 466
+#define GSN_START_INFOREF 467
+#define GSN_START_INFOCONF 468
+
+#define GSN_TC_COMMIT_ACK 469
+#define GSN_REMOVE_MARKER_ORD 470
+
+#define GSN_CHECKNODEGROUPSREQ 471
+#define GSN_CHECKNODEGROUPSCONF 472
+
+#define GSN_ARBIT_CFG 473
+#define GSN_ARBIT_PREPREQ 474
+#define GSN_ARBIT_PREPCONF 475
+#define GSN_ARBIT_PREPREF 476
+#define GSN_ARBIT_STARTREQ 477
+#define GSN_ARBIT_STARTCONF 478
+#define GSN_ARBIT_STARTREF 479
+#define GSN_ARBIT_CHOOSEREQ 480
+#define GSN_ARBIT_CHOOSECONF 481
+#define GSN_ARBIT_CHOOSEREF 482
+#define GSN_ARBIT_STOPORD 483
+#define GSN_ARBIT_STOPREP 484
+
+#define GSN_BLOCK_COMMIT_ORD 485
+#define GSN_UNBLOCK_COMMIT_ORD 486
+
+#define GSN_NODE_STATE_REP 487
+#define GSN_CHANGE_NODE_STATE_REQ 488
+#define GSN_CHANGE_NODE_STATE_CONF 489
+
+#define GSN_DIH_SWITCH_REPLICA_REQ 490
+#define GSN_DIH_SWITCH_REPLICA_CONF 491
+#define GSN_DIH_SWITCH_REPLICA_REF 492
+
+#define GSN_STOP_PERM_REQ 493
+#define GSN_STOP_PERM_REF 494
+#define GSN_STOP_PERM_CONF 495
+
+#define GSN_STOP_ME_REQ 496
+#define GSN_STOP_ME_REF 497
+#define GSN_STOP_ME_CONF 498
+
+#define GSN_WAIT_GCP_REQ 499
+#define GSN_WAIT_GCP_REF 500
+#define GSN_WAIT_GCP_CONF 501
+
+// 502 not used
+
+/**
+ * Trigger and index signals
+ */
+
+/**
+ * These are used by API and kernel
+ */
+#define GSN_TRIG_ATTRINFO 503
+#define GSN_CREATE_TRIG_REQ 504
+#define GSN_CREATE_TRIG_CONF 505
+#define GSN_CREATE_TRIG_REF 506
+#define GSN_ALTER_TRIG_REQ 507
+#define GSN_ALTER_TRIG_CONF 508
+#define GSN_ALTER_TRIG_REF 509
+#define GSN_CREATE_INDX_REQ 510
+#define GSN_CREATE_INDX_CONF 511
+#define GSN_CREATE_INDX_REF 512
+#define GSN_DROP_TRIG_REQ 513
+#define GSN_DROP_TRIG_CONF 514
+#define GSN_DROP_TRIG_REF 515
+#define GSN_DROP_INDX_REQ 516
+#define GSN_DROP_INDX_CONF 517
+#define GSN_DROP_INDX_REF 518
+#define GSN_TCINDXREQ 519
+#define GSN_TCINDXCONF 520
+#define GSN_TCINDXREF 521
+#define GSN_INDXKEYINFO 522
+#define GSN_INDXATTRINFO 523
+#define GSN_TCINDXNEXTREQ 524
+#define GSN_TCINDXNEXTCONF 525
+#define GSN_TCINDXNEXREF 526
+#define GSN_FIRE_TRIG_ORD 527
+
+/**
+ * These are used only by kernel
+ */
+#define GSN_BUILDINDXREQ 528
+#define GSN_BUILDINDXCONF 529
+#define GSN_BUILDINDXREF 530
+
+/**
+ * Backup interface
+ */
+#define GSN_BACKUP_REQ 531
+#define GSN_BACKUP_DATA 532
+#define GSN_BACKUP_REF 533
+#define GSN_BACKUP_CONF 534
+
+#define GSN_ABORT_BACKUP_ORD 535
+
+#define GSN_BACKUP_ABORT_REP 536
+#define GSN_BACKUP_COMPLETE_REP 537
+#define GSN_BACKUP_NF_COMPLETE_REP 538
+
+/**
+ * Internal backup signals
+ */
+#define GSN_DEFINE_BACKUP_REQ 539
+#define GSN_DEFINE_BACKUP_REF 540
+#define GSN_DEFINE_BACKUP_CONF 541
+
+#define GSN_START_BACKUP_REQ 542
+#define GSN_START_BACKUP_REF 543
+#define GSN_START_BACKUP_CONF 544
+
+#define GSN_BACKUP_FRAGMENT_REQ 545
+#define GSN_BACKUP_FRAGMENT_REF 546
+#define GSN_BACKUP_FRAGMENT_CONF 547
+
+#define GSN_STOP_BACKUP_REQ 548
+#define GSN_STOP_BACKUP_REF 549
+#define GSN_STOP_BACKUP_CONF 550
+
+/**
+ * Used for master take-over / API status request
+ */
+#define GSN_BACKUP_STATUS_REQ 551
+#define GSN_BACKUP_STATUS_REF 116
+#define GSN_BACKUP_STATUS_CONF 165
+
+/**
+ * Db sequence signals
+ */
+#define GSN_UTIL_SEQUENCE_REQ 552
+#define GSN_UTIL_SEQUENCE_REF 553
+#define GSN_UTIL_SEQUENCE_CONF 554
+
+#define GSN_FSREMOVEREQ 555
+#define GSN_FSREMOVEREF 556
+#define GSN_FSREMOVECONF 557
+
+#define GSN_UTIL_PREPARE_REQ 558
+#define GSN_UTIL_PREPARE_CONF 559
+#define GSN_UTIL_PREPARE_REF 560
+
+#define GSN_UTIL_EXECUTE_REQ 561
+#define GSN_UTIL_EXECUTE_CONF 562
+#define GSN_UTIL_EXECUTE_REF 563
+
+#define GSN_UTIL_RELEASE_REQ 564
+#define GSN_UTIL_RELEASE_CONF 565
+#define GSN_UTIL_RELEASE_REF 566
+
+/**
+ * When dropping a long signal due to lack of memory resources
+ */
+#define GSN_SIGNAL_DROPPED_REP 567
+#define GSN_CONTINUE_FRAGMENTED 568
+
+/**
+ * Suma participant interface
+ */
+#define GSN_SUB_REMOVE_REQ 569
+#define GSN_SUB_REMOVE_REF 570
+#define GSN_SUB_REMOVE_CONF 571
+#define GSN_SUB_STOP_REQ 572
+#define GSN_SUB_STOP_REF 573
+#define GSN_SUB_STOP_CONF 574
+// 575 unused
+#define GSN_SUB_CREATE_REQ 576
+#define GSN_SUB_CREATE_REF 577
+#define GSN_SUB_CREATE_CONF 578
+#define GSN_SUB_START_REQ 579
+#define GSN_SUB_START_REF 580
+#define GSN_SUB_START_CONF 581
+#define GSN_SUB_SYNC_REQ 582
+#define GSN_SUB_SYNC_REF 583
+#define GSN_SUB_SYNC_CONF 584
+#define GSN_SUB_META_DATA 585
+#define GSN_SUB_TABLE_DATA 586
+
+#define GSN_CREATE_TABLE_REQ 587
+#define GSN_CREATE_TABLE_REF 588
+#define GSN_CREATE_TABLE_CONF 589
+
+#define GSN_ALTER_TABLE_REQ 624
+#define GSN_ALTER_TABLE_REF 625
+#define GSN_ALTER_TABLE_CONF 626
+
+#define GSN_SUB_SYNC_CONTINUE_REQ 590
+#define GSN_SUB_SYNC_CONTINUE_REF 591
+#define GSN_SUB_SYNC_CONTINUE_CONF 592
+#define GSN_SUB_GCP_COMPLETE_REP 593
+
+#define GSN_CREATE_FRAGMENTATION_REQ 594
+#define GSN_CREATE_FRAGMENTATION_REF 595
+#define GSN_CREATE_FRAGMENTATION_CONF 596
+
+#define GSN_CREATE_TAB_REQ 597
+#define GSN_CREATE_TAB_REF 598
+#define GSN_CREATE_TAB_CONF 599
+
+#define GSN_ALTER_TAB_REQ 600
+#define GSN_ALTER_TAB_REF 601
+#define GSN_ALTER_TAB_CONF 602
+
+#define GSN_ALTER_INDX_REQ 603
+#define GSN_ALTER_INDX_REF 604
+#define GSN_ALTER_INDX_CONF 605
+
+/**
+ * Grep signals
+ */
+#define GSN_GREP_SUB_CREATE_REQ 606
+#define GSN_GREP_SUB_CREATE_REF 607
+#define GSN_GREP_SUB_CREATE_CONF 608
+#define GSN_GREP_CREATE_REQ 609
+#define GSN_GREP_CREATE_REF 610
+#define GSN_GREP_CREATE_CONF 611
+
+#define GSN_GREP_SUB_START_REQ 612
+#define GSN_GREP_SUB_START_REF 613
+#define GSN_GREP_SUB_START_CONF 614
+#define GSN_GREP_START_REQ 615
+#define GSN_GREP_START_REF 616
+#define GSN_GREP_START_CONF 617
+
+#define GSN_GREP_SUB_SYNC_REQ 618
+#define GSN_GREP_SUB_SYNC_REF 619
+#define GSN_GREP_SUB_SYNC_CONF 620
+#define GSN_GREP_SYNC_REQ 621
+#define GSN_GREP_SYNC_REF 622
+#define GSN_GREP_SYNC_CONF 623
+
+/**
+ * REP signals
+ */
+#define GSN_REP_WAITGCP_REQ 627
+#define GSN_REP_WAITGCP_REF 628
+#define GSN_REP_WAITGCP_CONF 629
+#define GSN_GREP_WAITGCP_REQ 630
+#define GSN_GREP_WAITGCP_REF 631
+#define GSN_GREP_WAITGCP_CONF 632
+#define GSN_REP_GET_GCI_REQ 633
+#define GSN_REP_GET_GCI_REF 634
+#define GSN_REP_GET_GCI_CONF 635
+#define GSN_REP_GET_GCIBUFFER_REQ 636
+#define GSN_REP_GET_GCIBUFFER_REF 637
+#define GSN_REP_GET_GCIBUFFER_CONF 638
+#define GSN_REP_INSERT_GCIBUFFER_REQ 639
+#define GSN_REP_INSERT_GCIBUFFER_REF 640
+#define GSN_REP_INSERT_GCIBUFFER_CONF 641
+#define GSN_REP_CLEAR_PS_GCIBUFFER_REQ 642
+#define GSN_REP_CLEAR_PS_GCIBUFFER_REF 643
+#define GSN_REP_CLEAR_PS_GCIBUFFER_CONF 644
+#define GSN_REP_CLEAR_SS_GCIBUFFER_REQ 645
+#define GSN_REP_CLEAR_SS_GCIBUFFER_REF 646
+#define GSN_REP_CLEAR_SS_GCIBUFFER_CONF 647
+#define GSN_REP_DATA_PAGE 648
+#define GSN_REP_GCIBUFFER_ACC_REP 649
+
+#define GSN_GREP_SUB_REMOVE_REQ 650
+#define GSN_GREP_SUB_REMOVE_REF 651
+#define GSN_GREP_SUB_REMOVE_CONF 652
+#define GSN_GREP_REMOVE_REQ 653
+#define GSN_GREP_REMOVE_REF 654
+#define GSN_GREP_REMOVE_CONF 655
+
+// Start Global Replication
+#define GSN_GREP_REQ 656
+
+/**
+ * Management server
+ */
+#define GSN_MGM_LOCK_CONFIG_REQ 657
+#define GSN_MGM_LOCK_CONFIG_REP 658
+#define GSN_MGM_UNLOCK_CONFIG_REQ 659
+#define GSN_MGM_UNLOCK_CONFIG_REP 660
+
+#define GSN_UTIL_CREATE_LOCK_REQ 132
+#define GSN_UTIL_CREATE_LOCK_REF 133
+#define GSN_UTIL_CREATE_LOCK_CONF 188
+
+#define GSN_UTIL_DESTROY_LOCK_REQ 189
+#define GSN_UTIL_DESTROY_LOCK_REF 220
+#define GSN_UTIL_DESTROY_LOCK_CONF 221
+
+#define GSN_UTIL_LOCK_REQ 222
+#define GSN_UTIL_LOCK_REF 230
+#define GSN_UTIL_LOCK_CONF 231
+
+#define GSN_UTIL_UNLOCK_REQ 303
+#define GSN_UTIL_UNLOCK_REF 304
+#define GSN_UTIL_UNLOCK_CONF 362
+
+/* SUMA */
+#define GSN_CREATE_SUBID_REQ 661
+#define GSN_CREATE_SUBID_REF 662
+#define GSN_CREATE_SUBID_CONF 663
+
+/* GREP */
+#define GSN_GREP_CREATE_SUBID_REQ 664
+#define GSN_GREP_CREATE_SUBID_REF 665
+#define GSN_GREP_CREATE_SUBID_CONF 666
+#define GSN_REP_DROP_TABLE_REQ 667
+#define GSN_REP_DROP_TABLE_REF 668
+#define GSN_REP_DROP_TABLE_CONF 669
+
+/*
+ * TUX
+ */
+#define GSN_TUXFRAGREQ 670
+#define GSN_TUXFRAGCONF 671
+#define GSN_TUXFRAGREF 672
+#define GSN_TUX_ADD_ATTRREQ 673
+#define GSN_TUX_ADD_ATTRCONF 674
+#define GSN_TUX_ADD_ATTRREF 675
+
+/*
+ * REP
+ */
+#define GSN_REP_DISCONNECT_REP 676
+
+#define GSN_TUX_MAINT_REQ 677
+#define GSN_TUX_MAINT_CONF 678
+#define GSN_TUX_MAINT_REF 679
+
+/*
+ * TUP access
+ */
+#define GSN_TUP_READ_ATTRS 680
+#define GSN_TUP_QUERY_TH 712
+#define GSN_TUP_STORE_TH 681
+
+/**
+ * from mgmtsrvr to NDBCNTR
+ */
+#define GSN_RESUME_REQ 682
+#define GSN_STOP_REQ 443
+#define GSN_STOP_REF 444
+#define GSN_API_VERSION_REQ 697
+#define GSN_API_VERSION_CONF 698
+
+// not used 686
+// not used 687
+// not used 689
+// not used 690
+
+/**
+ * SUMA restart protocol
+ */
+#define GSN_SUMA_START_ME 691
+#define GSN_SUMA_HANDOVER_REQ 692
+#define GSN_SUMA_HANDOVER_CONF 693
+
+// not used 694
+// not used 695
+// not used 696
+
+/**
+ * GREP restart protocol
+ */
+#define GSN_GREP_START_ME 706
+#define GSN_GREP_ADD_SUB_REQ 707
+#define GSN_GREP_ADD_SUB_REF 708
+#define GSN_GREP_ADD_SUB_CONF 709
+
+
+/*
+ * EVENT Signals
+ */
+#define GSN_SUB_GCP_COMPLETE_ACC 699
+
+#define GSN_CREATE_EVNT_REQ 700
+#define GSN_CREATE_EVNT_CONF 701
+#define GSN_CREATE_EVNT_REF 702
+
+#define GSN_DROP_EVNT_REQ 703
+#define GSN_DROP_EVNT_CONF 704
+#define GSN_DROP_EVNT_REF 705
+
+#define GSN_TUX_BOUND_INFO 710
+
+#define GSN_ACC_LOCKREQ 711
+
+
+#endif
diff --git a/ndb/include/kernel/GrepEvent.hpp b/ndb/include/kernel/GrepEvent.hpp
new file mode 100644
index 00000000000..2073a7072c9
--- /dev/null
+++ b/ndb/include/kernel/GrepEvent.hpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GREP_EVENT_H
+#define GREP_EVENT_H
+
+class GrepEvent {
+public:
+ enum Subscription {
+ GrepSS_CreateSubIdConf = 1,
+ GrepSS_SubCreateConf = 2,
+ GrepSS_SubStartMetaConf = 3,
+ GrepSS_SubStartDataConf = 4,
+ GrepSS_SubSyncDataConf = 5,
+ GrepSS_SubSyncMetaConf = 6,
+ GrepSS_SubRemoveConf = 7,
+
+ GrepPS_CreateSubIdConf = 8,
+ GrepPS_SubCreateConf = 9,
+ GrepPS_SubStartMetaConf = 10,
+ GrepPS_SubStartDataConf = 11,
+ GrepPS_SubSyncMetaConf = 12,
+ GrepPS_SubSyncDataConf = 13,
+ GrepPS_SubRemoveConf = 14,
+
+ GrepPS_CreateSubIdRef = 15,
+ GrepPS_SubCreateRef = 16,
+ GrepPS_SubStartMetaRef = 17,
+ GrepPS_SubStartDataRef = 18,
+ GrepPS_SubSyncMetaRef = 19,
+ GrepPS_SubSyncDataRef = 20,
+ GrepPS_SubRemoveRef = 21,
+
+ GrepSS_CreateSubIdRef = 22,
+ GrepSS_SubCreateRef = 23,
+ GrepSS_SubStartMetaRef = 24,
+ GrepSS_SubStartDataRef = 25,
+ GrepSS_SubSyncMetaRef = 26,
+ GrepSS_SubSyncDataRef = 27,
+ GrepSS_SubRemoveRef = 28,
+
+ Rep_Disconnect = 29
+
+ };
+};
+#endif
diff --git a/ndb/include/kernel/Interpreter.hpp b/ndb/include/kernel/Interpreter.hpp
new file mode 100644
index 00000000000..2c282be361c
--- /dev/null
+++ b/ndb/include/kernel/Interpreter.hpp
@@ -0,0 +1,284 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_INTERPRETER_HPP
+#define NDB_INTERPRETER_HPP
+
+#include <ndb_types.h>
+
+class Interpreter {
+public:
+
+ inline static Uint32 mod4(Uint32 len){
+ return len + ((4 - (len & 3)) & 3);
+ }
+
+
+ /**
+ * General Mnemonic format
+ *
+ * i = Instruction - 5 Bits ( 0 - 5 ) max 63
+ * x = Register 1 - 3 Bits ( 6 - 8 ) max 7
+ * y = Register 2 - 3 Bits ( 9 -11 ) max 7
+ * b = Branch offset (only branches)
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * iiiiiixxxyyy bbbbbbbbbbbbbbbb
+ *
+ *
+ */
+
+ /**
+ * Instructions
+ */
+ static const Uint32 READ_ATTR_INTO_REG = 1;
+ static const Uint32 WRITE_ATTR_FROM_REG = 2;
+ static const Uint32 LOAD_CONST_NULL = 3;
+ static const Uint32 LOAD_CONST16 = 4;
+ static const Uint32 LOAD_CONST32 = 5;
+ static const Uint32 LOAD_CONST64 = 6;
+ static const Uint32 ADD_REG_REG = 7;
+ static const Uint32 SUB_REG_REG = 8;
+ static const Uint32 BRANCH = 9;
+ static const Uint32 BRANCH_REG_EQ_NULL = 10;
+ static const Uint32 BRANCH_REG_NE_NULL = 11;
+ static const Uint32 BRANCH_EQ_REG_REG = 12;
+ static const Uint32 BRANCH_NE_REG_REG = 13;
+ static const Uint32 BRANCH_LT_REG_REG = 14;
+ static const Uint32 BRANCH_LE_REG_REG = 15;
+ static const Uint32 BRANCH_GT_REG_REG = 16;
+ static const Uint32 BRANCH_GE_REG_REG = 17;
+ static const Uint32 EXIT_OK = 18;
+ static const Uint32 EXIT_REFUSE = 19;
+ static const Uint32 CALL = 20;
+ static const Uint32 RETURN = 21;
+ static const Uint32 EXIT_OK_LAST = 22;
+ static const Uint32 BRANCH_ATTR_OP_ARG = 23;
+ static const Uint32 BRANCH_ATTR_EQ_NULL = 24;
+ static const Uint32 BRANCH_ATTR_NE_NULL = 25;
+
+ /**
+ * Macros for creating code
+ */
+ static Uint32 Read(Uint32 AttrId, Uint32 Register);
+ static Uint32 Write(Uint32 AttrId, Uint32 Register);
+
+ static Uint32 LoadNull(Uint32 Register);
+ static Uint32 LoadConst16(Uint32 Register, Uint32 Value);
+ static Uint32 LoadConst32(Uint32 Register); // Value in next word
+ static Uint32 LoadConst64(Uint32 Register); // Value in next 2 words
+ static Uint32 Add(Uint32 DstReg, Uint32 SrcReg1, Uint32 SrcReg2);
+ static Uint32 Sub(Uint32 DstReg, Uint32 SrcReg1, Uint32 SrcReg2);
+ static Uint32 Branch(Uint32 Inst, Uint32 R1, Uint32 R2);
+ static Uint32 ExitOK();
+
+ /**
+ * Branch string
+ *
+ * i = Instruction - 5 Bits ( 0 - 5 ) max 63
+ * a = Attribute id
+ * l = Length of string
+ * b = Branch offset
+ * t = branch type
+ * d = Array length diff
+ * v = Varchar flag
+ * p = No-blank-padding flag for char compare
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * iiiiii ddvtttpbbbbbbbbbbbbbbbb
+ * aaaaaaaaaaaaaaaallllllllllllllll
+ * -string.... -
+ */
+ enum UnaryCondition {
+ IS_NULL = 0,
+ IS_NOT_NULL = 1
+ };
+
+ enum BinaryCondition {
+ EQ = 0,
+ NE = 1,
+ LT = 2,
+ LE = 3,
+ GT = 4,
+ GE = 5,
+ LIKE = 6,
+ NOT_LIKE = 7
+ };
+ static Uint32 BranchCol(BinaryCondition cond,
+ Uint32 arrayLengthDiff, Uint32 varchar, bool nopad);
+ static Uint32 BranchCol_2(Uint32 AttrId);
+ static Uint32 BranchCol_2(Uint32 AttrId, Uint32 Len);
+
+ static Uint32 getBinaryCondition(Uint32 op1);
+ static Uint32 getArrayLengthDiff(Uint32 op1);
+ static Uint32 isVarchar(Uint32 op1);
+ static Uint32 isNopad(Uint32 op1);
+ static Uint32 getBranchCol_AttrId(Uint32 op2);
+ static Uint32 getBranchCol_Len(Uint32 op2);
+
+ /**
+ * Macros for decoding code
+ */
+ static Uint32 getOpCode(Uint32 op);
+ static Uint32 getReg1(Uint32 op);
+ static Uint32 getReg2(Uint32 op);
+ static Uint32 getReg3(Uint32 op);
+};
+
+inline
+Uint32
+Interpreter::Read(Uint32 AttrId, Uint32 Register){
+ return (AttrId << 16) + (Register << 6) + READ_ATTR_INTO_REG;
+}
+
+inline
+Uint32
+Interpreter::Write(Uint32 AttrId, Uint32 Register){
+ return (AttrId << 16) + (Register << 6) + WRITE_ATTR_FROM_REG;
+}
+
+inline
+Uint32
+Interpreter::LoadConst16(Uint32 Register, Uint32 Value){
+ return (Value << 16) + (Register << 6) + LOAD_CONST16;
+}
+
+inline
+Uint32
+Interpreter::LoadConst32(Uint32 Register){
+ return (Register << 6) + LOAD_CONST32;
+}
+
+inline
+Uint32
+Interpreter::LoadConst64(Uint32 Register){
+ return (Register << 6) + LOAD_CONST64;
+}
+
+inline
+Uint32
+Interpreter::Add(Uint32 Dcoleg, Uint32 SrcReg1, Uint32 SrcReg2){
+ return (SrcReg1 << 6) + (SrcReg2 << 9) + (Dcoleg << 16) + ADD_REG_REG;
+}
+
+inline
+Uint32
+Interpreter::Sub(Uint32 Dcoleg, Uint32 SrcReg1, Uint32 SrcReg2){
+ return (SrcReg1 << 6) + (SrcReg2 << 9) + (Dcoleg << 16) + SUB_REG_REG;
+}
+
+inline
+Uint32
+Interpreter::Branch(Uint32 Inst, Uint32 R1, Uint32 R2){
+ return (R1 << 9) + (R2 << 6) + Inst;
+}
+
+inline
+Uint32
+Interpreter::BranchCol(BinaryCondition cond,
+ Uint32 arrayLengthDiff,
+ Uint32 varchar, bool nopad){
+ //ndbout_c("BranchCol: cond=%d diff=%u varchar=%u nopad=%d",
+ //cond, arrayLengthDiff, varchar, nopad);
+ return
+ BRANCH_ATTR_OP_ARG +
+ (arrayLengthDiff << 9) +
+ (varchar << 11) +
+ (cond << 12) +
+ (nopad << 15);
+}
+
+inline
+Uint32
+Interpreter::BranchCol_2(Uint32 AttrId, Uint32 Len){
+ return (AttrId << 16) + Len;
+}
+
+inline
+Uint32
+Interpreter::BranchCol_2(Uint32 AttrId){
+ return (AttrId << 16);
+}
+
+inline
+Uint32
+Interpreter::getBinaryCondition(Uint32 op){
+ return (op >> 12) & 0x7;
+}
+
+inline
+Uint32
+Interpreter::getArrayLengthDiff(Uint32 op){
+ return (op >> 9) & 0x3;
+}
+
+inline
+Uint32
+Interpreter::isVarchar(Uint32 op){
+ return (op >> 11) & 1;
+}
+
+inline
+Uint32
+Interpreter::isNopad(Uint32 op){
+ return (op >> 15) & 1;
+}
+
+inline
+Uint32
+Interpreter::getBranchCol_AttrId(Uint32 op){
+ return (op >> 16) & 0xFFFF;
+}
+
+inline
+Uint32
+Interpreter::getBranchCol_Len(Uint32 op){
+ return op & 0xFFFF;
+}
+
+inline
+Uint32
+Interpreter::ExitOK(){
+ return EXIT_OK;
+}
+
+inline
+Uint32
+Interpreter::getOpCode(Uint32 op){
+ return op & 0x3f;
+}
+
+inline
+Uint32
+Interpreter::getReg1(Uint32 op){
+ return (op >> 6) & 0x7;
+}
+
+inline
+Uint32
+Interpreter::getReg2(Uint32 op){
+ return (op >> 9) & 0x7;
+}
+
+inline
+Uint32
+Interpreter::getReg3(Uint32 op){
+ return (op >> 16) & 0x7;
+}
+
+#endif
diff --git a/ndb/include/kernel/LogLevel.hpp b/ndb/include/kernel/LogLevel.hpp
new file mode 100644
index 00000000000..0902f3e488b
--- /dev/null
+++ b/ndb/include/kernel/LogLevel.hpp
@@ -0,0 +1,173 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef _LOG_LEVEL_HPP
+#define _LOG_LEVEL_HPP
+
+#include <ndb_types.h>
+#include <assert.h>
+
+/**
+ *
+ */
+class LogLevel {
+ friend class Config;
+public:
+ /**
+ * Constructor
+ */
+ LogLevel();
+
+ /**
+ * Howto add a new event category:
+ * 1. Add the new event category to EventCategory below
+ * 2. Update #define _LOGLEVEL_CATEGORIES (found below) with the number of
+ * items in EventCategory
+ * 3. Update LogLevelCategoryName in LogLevel.cpp
+ * 4. Add the event in EventLogger
+ */
+
+
+ /**
+ * Copy operator
+ */
+ LogLevel & operator= (const LogLevel &);
+
+ enum EventCategory {
+ /**
+ * Events during all kind of startups
+ */
+ llStartUp = 0,
+
+ /**
+ * Events during shutdown
+ */
+ llShutdown = 1,
+
+ /**
+ * Transaction statistics
+ * Job level
+ * TCP/IP speed
+ */
+ llStatistic = 2,
+
+ /**
+ * Checkpoints
+ */
+ llCheckpoint = 3,
+
+ /**
+ * Events during node restart
+ */
+ llNodeRestart = 4,
+
+ /**
+ * Events related to connection / communication
+ */
+ llConnection = 5,
+
+ /**
+ * Assorted event w.r.t unexpected happenings
+ */
+ llError = 6,
+
+ /**
+ * Assorted event w.r.t information
+ */
+ llInfo = 7,
+
+ /**
+ * Events related to global replication
+ */
+ llGrep = 8
+ };
+
+ struct LogLevelCategoryName {
+ const char* name;
+ };
+
+ /**
+ * Log/event level category names. Remember to update the names whenever
+ * a new category is added.
+ */
+ static const LogLevelCategoryName LOGLEVEL_CATEGORY_NAME[];
+
+ /**
+ * No of categories
+ */
+#define _LOGLEVEL_CATEGORIES 9
+ static const Uint32 LOGLEVEL_CATEGORIES = _LOGLEVEL_CATEGORIES;
+
+ void clear();
+
+ /**
+ * Note level is valid as 0-15
+ */
+ void setLogLevel(EventCategory ec, Uint32 level = 7);
+
+ /**
+ * Get the loglevel (0-15) for a category
+ */
+ Uint32 getLogLevel(EventCategory ec) const;
+
+private:
+ /**
+ * The actual data
+ */
+ Uint32 logLevelData[LOGLEVEL_CATEGORIES];
+
+ LogLevel(const LogLevel &);
+};
+
+inline
+LogLevel::LogLevel(){
+ clear();
+}
+
+inline
+LogLevel &
+LogLevel::operator= (const LogLevel & org){
+ for(Uint32 i = 0; i<LOGLEVEL_CATEGORIES; i++){
+ logLevelData[i] = org.logLevelData[i];
+ }
+ return * this;
+}
+
+inline
+void
+LogLevel::clear(){
+ for(Uint32 i = 0; i<LOGLEVEL_CATEGORIES; i++){
+ logLevelData[i] = 0;
+ }
+}
+
+inline
+void
+LogLevel::setLogLevel(EventCategory ec, Uint32 level){
+ assert(ec >= 0 && (Uint32) ec < LOGLEVEL_CATEGORIES);
+ logLevelData[ec] = level;
+}
+
+inline
+Uint32
+LogLevel::getLogLevel(EventCategory ec) const{
+ assert(ec >= 0 && (Uint32) ec < LOGLEVEL_CATEGORIES);
+
+ return logLevelData[ec];
+}
+
+
+#endif
diff --git a/ndb/include/kernel/NodeBitmask.hpp b/ndb/include/kernel/NodeBitmask.hpp
new file mode 100644
index 00000000000..423c01cd841
--- /dev/null
+++ b/ndb/include/kernel/NodeBitmask.hpp
@@ -0,0 +1,89 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODE_BITMASK_HPP
+#define NODE_BITMASK_HPP
+
+#include <ndb_limits.h>
+#include <kernel_types.h>
+#include <Bitmask.hpp>
+
+/**
+ * No of 32 bits words needed to store a node bitmask
+ * containing all the nodes in the system
+ * Both NDB nodes and API, MGM... nodes
+ *
+ * Note that this is used in a lot of signals
+ */
+#define _NODE_BITMASK_SIZE 2
+
+/**
+ * No of 32 bits words needed to store a node bitmask
+ * containing all the ndb nodes in the system
+ *
+ * Note that this is used in a lot of signals
+ */
+#define _NDB_NODE_BITMASK_SIZE 2
+
+/**
+ * No of 32 bits word needed to store B bits for N nodes
+ */
+#define NODE_ARRAY_SIZE(N, B) (((N)*(B)+31) >> 5)
+
+typedef Bitmask<(unsigned int)_NODE_BITMASK_SIZE> NodeBitmask;
+
+typedef Bitmask<(unsigned int)_NDB_NODE_BITMASK_SIZE> NdbNodeBitmask;
+
+#define __NBM_SZ ((MAX_NODES >> 5) + ((MAX_NODES & 31) != 0))
+#define __NNBM_SZ ((MAX_NDB_NODES >> 5) + ((MAX_NDB_NODES & 31) != 0))
+
+#if ( __NBM_SZ > _NODE_BITMASK_SIZE)
+#error "MAX_NODES can not fit into NODE_BITMASK_SIZE"
+#endif
+
+#if ( __NNBM_SZ > _NDB_NODE_BITMASK_SIZE)
+#error "MAX_NDB_NODES can not fit into NDB_NODE_BITMASK_SIZE"
+#endif
+
+/**
+ * General B Bits operations
+ *
+ * Get(x, A[], B)
+ * w = x >> S1
+ * s = (x & S2) << S3
+ * return (A[w] >> s) & S4
+ *
+ * Set(x, A[], v, B)
+ * w = x >> S1
+ * s = (x & S2) << S3
+ * m = ~(S4 << s)
+ * t = A[w] & m;
+ * A[w] = t | ((v & S4) << s)
+ *
+ * B(Bits) S1 S2 S3 S4
+ * 1 5 31 0 1
+ * 2 4 15 1 3
+ * 4 3 7 2 15
+ * 8 2 3 3 255
+ * 16 1 1 4 65535
+ *
+ * S1 = 5 - 2log(B)
+ * S2 = 2^S1 - 1
+ * S3 = 2log(B)
+ * S4 = 2^B - 1
+ */
+
+#endif
diff --git a/ndb/include/kernel/NodeInfo.hpp b/ndb/include/kernel/NodeInfo.hpp
new file mode 100644
index 00000000000..86aca7d6883
--- /dev/null
+++ b/ndb/include/kernel/NodeInfo.hpp
@@ -0,0 +1,94 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODE_INFO_HPP
+#define NODE_INFO_HPP
+
+#include <NdbOut.hpp>
+
+class NodeInfo {
+public:
+ NodeInfo();
+
+ /**
+ * NodeType
+ */
+ enum NodeType {
+ DB = 0, ///< Database node
+ API = 1, ///< NDB API node
+ MGM = 2, ///< Management node (incl. NDB API)
+ REP = 3, ///< Replication node (incl. NDB API)
+ INVALID = 255 ///< Invalid type
+ };
+ NodeType getType() const;
+
+ Uint32 m_version; ///< Node version
+ Uint32 m_signalVersion; ///< Signal version
+ Uint32 m_type; ///< Node type
+ Uint32 m_connectCount; ///< No of times connected
+ bool m_connected; ///< Node is connected
+
+ friend NdbOut & operator<<(NdbOut&, const NodeInfo&);
+};
+
+
+inline
+NodeInfo::NodeInfo(){
+ m_version = 0;
+ m_signalVersion = 0;
+ m_type = INVALID;
+ m_connectCount = 0;
+}
+
+inline
+NodeInfo::NodeType
+NodeInfo::getType() const {
+ return (NodeType)m_type;
+}
+
+inline
+NdbOut &
+operator<<(NdbOut& ndbout, const NodeInfo & info){
+ ndbout << "[NodeInfo: ";
+ switch(info.m_type){
+ case NodeInfo::DB:
+ ndbout << "DB";
+ break;
+ case NodeInfo::API:
+ ndbout << "API";
+ break;
+ case NodeInfo::MGM:
+ ndbout << "MGM";
+ break;
+ case NodeInfo::REP:
+ ndbout << "REP";
+ break;
+ case NodeInfo::INVALID:
+ ndbout << "INVALID";
+ break;
+ default:
+ ndbout << "<Unknown: " << info.m_type << ">";
+ break;
+ }
+
+ ndbout << " version: " << info.m_version
+ << " sig. version; " << info.m_signalVersion
+ << " connect count: " << info.m_connectCount
+ << "]";
+ return ndbout;
+}
+
+#endif
diff --git a/ndb/include/kernel/NodeState.hpp b/ndb/include/kernel/NodeState.hpp
new file mode 100644
index 00000000000..1bc7806876d
--- /dev/null
+++ b/ndb/include/kernel/NodeState.hpp
@@ -0,0 +1,308 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODE_STATE_HPP
+#define NODE_STATE_HPP
+
+#include <NdbOut.hpp>
+
+class NodeState {
+public:
+ enum StartLevel {
+ /**
+ * SL_NOTHING
+ * Nothing is started
+ */
+ SL_NOTHING = 0,
+
+ /**
+ * SL_CMVMI
+ * CMVMI is started
+ * Listening to management server
+ * Qmgr knows nothing...
+ */
+ SL_CMVMI = 1,
+
+ /**
+ * SL_STARTING
+ * All blocks are starting
+ * Initial or restart
+ * During this phase is <b>startPhase</b> valid
+ */
+ SL_STARTING = 2,
+
+ /**
+ * The database is started open for connections
+ */
+ SL_STARTED = 3,
+
+ SL_SINGLEUSER = 4,
+
+ /**
+ * SL_STOPPING_1 - Inform API
+ * API is informed not to start transactions on node
+ * The database is about to close
+ *
+ * New TcSeize(s) are refused (TcSeizeRef)
+ */
+ SL_STOPPING_1 = 5,
+
+ /**
+ * SL_STOPPING_2 - Close TC
+ * New transactions(TC) are refused
+ */
+ SL_STOPPING_2 = 6,
+
+
+
+
+ /**
+ * SL_STOPPING_3 - Wait for reads in LQH
+ * No transactions are running in TC
+ * New scans(s) and read(s) are refused in LQH
+ * NS: The node is not Primary for any fragment
+ * NS: No node is allow to start
+ */
+ SL_STOPPING_3 = 7,
+
+ /**
+ * SL_STOPPING_4 - Close LQH
+ * Node is out of DIGETNODES
+ * Insert/Update/Delete can still be running in LQH
+ * GCP is refused
+ * Node is not startable w.o Node Recovery
+ */
+ SL_STOPPING_4 = 8
+ };
+
+ enum StartType {
+ ST_INITIAL_START = 0,
+ ST_SYSTEM_RESTART = 1,
+ ST_NODE_RESTART = 2,
+ ST_INITIAL_NODE_RESTART = 3,
+ ST_ILLEGAL_TYPE = 4
+ };
+
+ /**
+ * Length in 32-bit words
+ */
+ static const Uint32 DataLength = 8;
+
+ /**
+ * Constructor(s)
+ */
+ NodeState();
+ NodeState(StartLevel);
+ NodeState(StartLevel, bool systemShutdown);
+ NodeState(StartLevel, Uint32 startPhase, StartType);
+
+ /**
+ * Current start level
+ */
+ Uint32 startLevel;
+
+ /**
+ * Node group
+ */
+ Uint32 nodeGroup; // valid when startLevel == SL_STARTING
+
+ /**
+ * Dynamic id
+ */
+ union {
+ Uint32 dynamicId; // valid when startLevel == SL_STARTING to API
+ Uint32 masterNodeId; // When from cntr
+ };
+
+ /**
+ *
+ */
+ union {
+ struct {
+ Uint32 startPhase; // valid when startLevel == SL_STARTING
+ Uint32 restartType; // valid when startLevel == SL_STARTING
+ } starting;
+ struct {
+ Uint32 systemShutdown; // valid when startLevel == SL_STOPPING_{X}
+ Uint32 timeout;
+ Uint32 alarmTime;
+ } stopping;
+
+
+ };
+ Uint32 singleUserMode;
+ Uint32 singleUserApi; //the single user node
+
+ void setDynamicId(Uint32 dynamic);
+ void setNodeGroup(Uint32 group);
+ void setSingleUser(Uint32 s);
+ void setSingleUserApi(Uint32 n);
+
+
+ /**
+ * Is a node restart in progress (ordinary or initial)
+ */
+ bool getNodeRestartInProgress() const;
+
+ /**
+ * Is a system restart ongoing
+ */
+ bool getSystemRestartInProgress() const;
+
+ /**
+ * Is in single user mode?
+ */
+ bool getSingleUserMode() const;
+
+ /**
+ * Is in single user mode
+ */
+ Uint32 getSingleUserApi() const;
+
+ friend NdbOut & operator<<(NdbOut&, const NodeState&);
+};
+
+inline
+NodeState::NodeState(){
+ startLevel = SL_CMVMI;
+ nodeGroup = 0xFFFFFFFF;
+ dynamicId = 0xFFFFFFFF;
+ singleUserMode = 0;
+ singleUserApi = 0xFFFFFFFF;
+}
+
+inline
+NodeState::NodeState(StartLevel sl){
+ NodeState::NodeState();
+ startLevel = sl;
+ singleUserMode = 0;
+ singleUserApi = 0xFFFFFFFF;
+}
+
+inline
+NodeState::NodeState(StartLevel sl, Uint32 sp, StartType typeOfStart){
+ NodeState::NodeState();
+ startLevel = sl;
+ starting.startPhase = sp;
+ starting.restartType = typeOfStart;
+ singleUserMode = 0;
+ singleUserApi = 0xFFFFFFFF;
+}
+
+inline
+NodeState::NodeState(StartLevel sl, bool sys){
+ NodeState::NodeState();
+ startLevel = sl;
+ stopping.systemShutdown = sys;
+ singleUserMode = 0;
+ singleUserApi = 0xFFFFFFFF;
+}
+
+inline
+void NodeState::setDynamicId(Uint32 dynamic){
+ dynamicId = dynamic;
+}
+
+inline
+void NodeState::setNodeGroup(Uint32 group){
+ nodeGroup = group;
+}
+
+inline
+void NodeState::setSingleUser(Uint32 s) {
+ singleUserMode = s;
+}
+
+inline
+void NodeState::setSingleUserApi(Uint32 n) {
+ singleUserApi = n;
+}
+inline
+bool NodeState::getNodeRestartInProgress() const {
+ return startLevel == SL_STARTING &&
+ (starting.restartType == ST_NODE_RESTART ||
+ starting.restartType == ST_INITIAL_NODE_RESTART);
+}
+
+inline
+bool NodeState::getSingleUserMode() const {
+ return singleUserMode;
+}
+
+inline
+Uint32 NodeState::getSingleUserApi() const {
+ return singleUserApi;
+}
+
+inline
+bool NodeState::getSystemRestartInProgress() const {
+ return startLevel == SL_STARTING && starting.restartType == ST_SYSTEM_RESTART;
+}
+
+inline
+NdbOut &
+operator<<(NdbOut& ndbout, const NodeState & state){
+ ndbout << "[NodeState: startLevel: ";
+ switch(state.startLevel){
+ case NodeState::SL_NOTHING:
+ ndbout << "<NOTHING> ]";
+ break;
+ case NodeState::SL_CMVMI:
+ ndbout << "<CMVMI> ]";
+ break;
+ case NodeState::SL_STARTING:
+ ndbout << "<STARTING type: ";
+ switch(state.starting.restartType){
+ case NodeState::ST_INITIAL_START:
+ ndbout << " INITIAL START";
+ break;
+ case NodeState::ST_SYSTEM_RESTART:
+ ndbout << " SYSTEM RESTART ";
+ break;
+ case NodeState::ST_NODE_RESTART:
+ ndbout << " NODE RESTART ";
+ break;
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ ndbout << " INITIAL NODE RESTART ";
+ break;
+ case NodeState::ST_ILLEGAL_TYPE:
+ default:
+ ndbout << " UNKNOWN " << state.starting.restartType;
+ }
+ ndbout << " phase: " << state.starting.startPhase << "> ]";
+ break;
+ case NodeState::SL_STARTED:
+ ndbout << "<STARTED> ]";
+ break;
+ case NodeState::SL_STOPPING_1:
+ ndbout << "<STOPPING 1 sys: " << state.stopping.systemShutdown << "> ]";
+ break;
+ case NodeState::SL_STOPPING_2:
+ ndbout << "<STOPPING 2 sys: " << state.stopping.systemShutdown << "> ]";
+ break;
+ case NodeState::SL_STOPPING_3:
+ ndbout << "<STOPPING 3 sys: " << state.stopping.systemShutdown << "> ]";
+ break;
+ case NodeState::SL_STOPPING_4:
+ ndbout << "<STOPPING 4 sys: " << state.stopping.systemShutdown << "> ]";
+ break;
+ default:
+ ndbout << "<UNKNOWN " << state.startLevel << "> ]";
+ }
+ return ndbout;
+}
+
+#endif
diff --git a/ndb/include/kernel/RefConvert.hpp b/ndb/include/kernel/RefConvert.hpp
new file mode 100644
index 00000000000..7604b1cf224
--- /dev/null
+++ b/ndb/include/kernel/RefConvert.hpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REFCONVERT_H
+#define REFCONVERT_H
+
+#include "kernel_types.h"
+
+/**
+ * Convert BlockReference to BlockNumber
+ */
+inline
+BlockNumber refToBlock(BlockReference ref){
+ return (BlockNumber)(ref >> 16);
+}
+
+/**
+ * Convert BlockReference to NodeId
+ */
+inline
+NodeId refToNode(BlockReference ref){
+ return (NodeId)(ref & 0xFFFF);
+}
+
+/**
+ * Convert NodeId and BlockNumber to BlockReference
+ */
+inline
+BlockReference numberToRef(BlockNumber bnr, NodeId proc){
+ return (((Uint32)bnr) << 16) + proc;
+}
+
+#endif
+
diff --git a/ndb/include/kernel/kernel_types.h b/ndb/include/kernel/kernel_types.h
new file mode 100644
index 00000000000..b176d20798c
--- /dev/null
+++ b/ndb/include/kernel/kernel_types.h
@@ -0,0 +1,43 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_KERNEL_TYPES_H
+#define NDB_KERNEL_TYPES_H
+
+#include <ndb_types.h>
+
+typedef Uint16 NodeId;
+typedef Uint16 BlockNumber;
+typedef Uint32 BlockReference;
+typedef Uint16 GlobalSignalNumber;
+
+enum Operation_t {
+ ZREAD = 0
+ ,ZUPDATE = 1
+ ,ZINSERT = 2
+ ,ZDELETE = 3
+ ,ZWRITE = 4
+ ,ZREAD_EX = 5
+#if 0
+ ,ZREAD_CONSISTENT = 6
+#endif
+};
+
+#endif
+
+
+
+
diff --git a/ndb/include/kernel/ndb_limits.h b/ndb/include/kernel/ndb_limits.h
new file mode 100644
index 00000000000..65f729af1f2
--- /dev/null
+++ b/ndb/include/kernel/ndb_limits.h
@@ -0,0 +1,94 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_LIMITS_H
+#define NDB_LIMITS_H
+
+#define RNIL 0xffffff00
+
+/**
+ * Note that actual value = MAX_NODES - 1,
+ * since NodeId = 0 can not be used
+ */
+#define MAX_NDB_NODES 49
+#define MAX_NODES 64
+
+/**
+ * MAX_API_NODES = MAX_NODES - No of NDB Nodes in use
+ */
+
+/**
+ * The maximum number of replicas in the system
+ */
+#define MAX_REPLICAS 4
+
+/**
+ * The maximum number of local checkpoints stored at a time
+ */
+#define MAX_LCP_STORED 3
+
+/**
+ * The maximum number of log execution rounds at system restart
+ */
+#define MAX_LOG_EXEC 4
+
+/**
+ * The maximum number of tuples per page
+ **/
+#define MAX_TUPLES_PER_PAGE 8191
+#define MAX_TUPLES_BITS 13 /* 13 bits = 8191 tuples per page */
+//#define MAX_NO_OF_TUPLEKEY 16 Not currently used
+#define MAX_TABLES 1600
+#define MAX_TAB_NAME_SIZE 128
+#define MAX_ATTR_NAME_SIZE 32
+#define MAX_ATTR_DEFAULT_VALUE_SIZE 128
+#define MAX_ATTRIBUTES_IN_TABLE 128
+#define MAX_ATTRIBUTES_IN_INDEX 32
+#define MAX_TUPLE_SIZE_IN_WORDS 2013
+#define MAX_FIXED_KEY_LENGTH_IN_WORDS 8
+#define MAX_KEY_SIZE_IN_WORDS 1023
+#define MAX_FRM_DATA_SIZE 6000
+
+#define MIN_ATTRBUF ((MAX_ATTRIBUTES_IN_TABLE/24) + 1)
+/*
+ * Number of Records to fetch per SCAN_NEXTREQ in a scan in LQH. The
+ * API can order a multiple of this number of records at a time since
+ * fragments can be scanned in parallel.
+ */
+#define MAX_PARALLEL_OP_PER_SCAN 16
+/*
+ * Maximum number of Parallel Scan queries on one hash index fragment
+ */
+#define MAX_PARALLEL_SCANS_PER_FRAG 12
+/*
+ * Maximum parallel ordered index scans per primary table fragment.
+ * Implementation limit is (256 minus 12).
+ */
+#define MAX_PARALLEL_INDEX_SCANS_PER_FRAG 32
+
+/**
+ * Computed defines
+ */
+#define MAXNROFATTRIBUTESINWORDS (MAX_ATTRIBUTES_IN_TABLE / 32)
+
+/*
+ * Ordered index constants. Make configurable per index later.
+ */
+#define MAX_TTREE_NODE_SIZE 64 // total words in node
+#define MAX_TTREE_PREF_SIZE 4 // words in min/max prefix each
+#define MAX_TTREE_NODE_SLACK 3 // diff between max and min occupancy
+
+#endif
diff --git a/ndb/include/kernel/signaldata/AbortAll.hpp b/ndb/include/kernel/signaldata/AbortAll.hpp
new file mode 100644
index 00000000000..a3d7f483953
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AbortAll.hpp
@@ -0,0 +1,88 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ABORT_ALL_REQ_HPP
+#define ABORT_ALL_REQ_HPP
+
+#include "SignalData.hpp"
+
+class AbortAllReq {
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * Sender
+ */
+ friend class Ndbcntr;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+public:
+
+ Uint32 senderRef;
+ Uint32 senderData;
+};
+
+class AbortAllConf {
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Sender
+ */
+ friend class Dbtc;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+public:
+ Uint32 senderData;
+};
+
+class AbortAllRef {
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Sender
+ */
+ friend class Dbtc;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ enum ErrorCode {
+ InvalidState = 1,
+ AbortAlreadyInProgress = 2,
+ FunctionNotImplemented = 3
+ };
+public:
+ Uint32 senderData;
+ Uint32 errorCode;
+};
+
+#endif
+
diff --git a/ndb/include/kernel/signaldata/AccFrag.hpp b/ndb/include/kernel/signaldata/AccFrag.hpp
new file mode 100644
index 00000000000..e28ab0d1ee6
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AccFrag.hpp
@@ -0,0 +1,89 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ACC_FRAG_HPP
+#define ACC_FRAG_HPP
+
+#include "SignalData.hpp"
+
+class AccFragReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbacc;
+public:
+ STATIC_CONST( SignalLength = 12 );
+
+private:
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 tableId;
+ Uint32 reqInfo;
+ Uint32 fragId;
+ Uint32 localKeyLen;
+ Uint32 maxLoadFactor;
+ Uint32 minLoadFactor;
+ Uint32 kValue;
+ Uint32 lhFragBits;
+ Uint32 lhDirBits;
+ Uint32 keyLength;
+};
+
+class AccFragConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbacc;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+public:
+ STATIC_CONST( SignalLength = 7 );
+
+private:
+ Uint32 userPtr;
+ Uint32 rootFragPtr;
+ Uint32 fragId[2];
+ Uint32 fragPtr[2];
+ Uint32 rootHashCheck;
+};
+
+class AccFragRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dbacc;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 userPtr;
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/AccLock.hpp b/ndb/include/kernel/signaldata/AccLock.hpp
new file mode 100644
index 00000000000..1a41b4c9334
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AccLock.hpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ACC_LOCK_HPP
+#define ACC_LOCK_HPP
+
+#include "SignalData.hpp"
+
+/*
+ * Lock or unlock tuple. If lock request is queued, the reply is later
+ * via ACCKEYCONF.
+ */
+class AccLockReq {
+ friend class Dbtux;
+ friend class Dbacc;
+ friend bool printACC_LOCKREQ(FILE *, const Uint32*, Uint32, Uint16);
+public:
+ enum RequestType { // first byte
+ LockShared = 1,
+ LockExclusive = 2,
+ Unlock = 3,
+ Abort = 4,
+ AbortWithConf = 5
+ };
+ enum RequestFlag { // second byte
+ };
+ enum ReturnCode {
+ Success = 0,
+ IsBlocked = 1, // was put in lock queue
+ WouldBlock = 2, // if we add non-blocking option
+ Refused = 3,
+ NoFreeOp = 4
+ };
+ STATIC_CONST( LockSignalLength = 12 );
+ STATIC_CONST( UndoSignalLength = 3 );
+private:
+ Uint32 returnCode;
+ Uint32 requestInfo;
+ Uint32 accOpPtr;
+ // rest only if lock request
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 fragPtrI;
+ Uint32 hashValue;
+ Uint32 tupAddr;
+ Uint32 transId1;
+ Uint32 transId2;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/AccScan.hpp b/ndb/include/kernel/signaldata/AccScan.hpp
new file mode 100644
index 00000000000..eab1c3262fc
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AccScan.hpp
@@ -0,0 +1,164 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ACC_SCAN_HPP
+#define ACC_SCAN_HPP
+
+#include "SignalData.hpp"
+
+/*
+ * Used by ACC and TUX scan.
+ */
+
+class AccScanReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbacc;
+ friend class Dbtux;
+public:
+ STATIC_CONST( SignalLength = 8 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 tableId;
+ Uint32 fragmentNo;
+ Uint32 requestInfo;
+ Uint32 transId1;
+ Uint32 transId2;
+ Uint32 savePointId;
+
+ /**
+ * Previously there where also a scan type
+ */
+ static Uint32 getLockMode(const Uint32 & requestInfo);
+ static Uint32 getKeyinfoFlag(const Uint32 & requestInfo);
+ static Uint32 getReadCommittedFlag(const Uint32 & requestInfo);
+
+ static void setLockMode(Uint32 & requestInfo, Uint32 lockMode);
+ static void setKeyinfoFlag(Uint32 & requestInfo, Uint32 keyinfo);
+ static void setReadCommittedFlag(Uint32 & requestInfo, Uint32 readCommitted);
+};
+
+/**
+ * Request Info
+ *
+ * l = Lock Mode - 1 Bit 2
+ * k = Keyinfo - 1 Bit 4
+ * h = Read Committed - 1 Bit 5
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * l kh
+ */
+#define AS_LOCK_MODE_SHIFT (2)
+#define AS_LOCK_MODE_MASK (1)
+
+#define AS_KEYINFO_SHIFT (4)
+#define AS_READ_COMMITTED_SHIFT (5)
+
+inline
+Uint32
+AccScanReq::getLockMode(const Uint32 & requestInfo){
+ return (requestInfo >> AS_LOCK_MODE_SHIFT) & AS_LOCK_MODE_MASK;
+}
+
+inline
+Uint32
+AccScanReq::getKeyinfoFlag(const Uint32 & requestInfo){
+ return (requestInfo >> AS_KEYINFO_SHIFT) & 1;
+}
+
+inline
+Uint32
+AccScanReq::getReadCommittedFlag(const Uint32 & requestInfo){
+ return (requestInfo >> AS_READ_COMMITTED_SHIFT) & 1;
+}
+
+inline
+void
+AccScanReq::setLockMode(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, AS_LOCK_MODE_MASK, "AccScanReq::setLockMode");
+ requestInfo |= (val << AS_LOCK_MODE_SHIFT);
+}
+
+inline
+void
+AccScanReq::setKeyinfoFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "AccScanReq::setKeyinfoFlag");
+ requestInfo |= (val << AS_KEYINFO_SHIFT);
+}
+
+inline
+void
+AccScanReq::setReadCommittedFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "AccScanReq::setReadCommittedFlag");
+ requestInfo |= (val << AS_READ_COMMITTED_SHIFT);
+}
+
+class AccScanConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbacc;
+ friend class Dbtux;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dblqh;
+
+ enum {
+ ZEMPTY_FRAGMENT = 0,
+ ZNOT_EMPTY_FRAGMENT = 1
+ };
+
+public:
+ STATIC_CONST( SignalLength = 8 );
+
+private:
+ Uint32 scanPtr;
+ Uint32 accPtr;
+ Uint32 unused1;
+ Uint32 unused2;
+ Uint32 unused3;
+ Uint32 unused4;
+ Uint32 unused5;
+ Uint32 flag;
+};
+
+class AccCheckScan {
+ friend class Dbacc;
+ friend class Dbtux;
+ friend class Dblqh;
+ enum {
+ ZCHECK_LCP_STOP = 0,
+ ZNOT_CHECK_LCP_STOP = 1
+ };
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+ Uint32 accPtr; // scanptr.i in ACC or TUX
+ Uint32 checkLcpStop; // from enum
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/AccSizeAltReq.hpp b/ndb/include/kernel/signaldata/AccSizeAltReq.hpp
new file mode 100644
index 00000000000..ac348444826
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AccSizeAltReq.hpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ACC_SIZE_ALT_REQ_H
+#define ACC_SIZE_ALT_REQ_H
+
+#include "SignalData.hpp"
+
+class AccSizeAltReq {
+ /**
+ * Sender(s)
+ */
+ friend class ClusterConfiguration;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbacc;
+private:
+ /**
+ * Indexes in theData
+ */
+ STATIC_CONST( IND_BLOCK_REF = 0 );
+ STATIC_CONST( IND_DIR_RANGE = 1 );
+ STATIC_CONST( IND_DIR_ARRAY = 2 );
+ STATIC_CONST( IND_FRAGMENT = 3 );
+ STATIC_CONST( IND_OP_RECS = 4 );
+ STATIC_CONST( IND_OVERFLOW_RECS = 5 );
+ STATIC_CONST( IND_PAGE8 = 6 );
+ STATIC_CONST( IND_ROOT_FRAG = 7 );
+ STATIC_CONST( IND_TABLE = 8 );
+ STATIC_CONST( IND_SCAN = 9 );
+
+ /**
+ * Use the index definitions to use the signal data
+ */
+ UintR theData[10];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/AlterIndx.hpp b/ndb/include/kernel/signaldata/AlterIndx.hpp
new file mode 100644
index 00000000000..1f464ded010
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AlterIndx.hpp
@@ -0,0 +1,268 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ALTER_INDX_HPP
+#define ALTER_INDX_HPP
+
+#include "SignalData.hpp"
+#include <Bitmask.hpp>
+#include <trigger_definitions.h>
+
+/**
+ * AlterIndxReq.
+ */
+class AlterIndxReq {
+ friend bool printALTER_INDX_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum RequestType {
+ RT_UNDEFINED = 0,
+ RT_USER = 1,
+ RT_CREATE_INDEX = 2,
+ RT_DROP_INDEX = 3,
+ RT_SYSTEMRESTART = 4,
+ RT_NODERESTART = 5,
+ RT_DICT_PREPARE = 1 << 4,
+ RT_DICT_TC = 5 << 4,
+ RT_DICT_COMMIT = 0xC << 4,
+ RT_DICT_ABORT = 0xF << 4,
+ RT_TC = 5 << 8,
+ RT_TUX = 8 << 8
+ };
+ STATIC_CONST( SignalLength = 7 );
+
+private:
+ Uint32 m_userRef;
+ Uint32 m_connectionPtr;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexId; // only set by DICT
+ Uint32 m_indexVersion;
+ Uint32 m_online; // new state 0-offline 1-online
+ // extra
+ Uint32 m_opKey;
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ AlterIndxReq::RequestType getRequestType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_requestInfo, 0, 16);
+ return (AlterIndxReq::RequestType)val;
+ }
+ void setRequestType(AlterIndxReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getRequestFlag() const {
+ return BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ };
+ void addRequestFlag(Uint32 val) {
+ val |= BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ BitmaskImpl::setField(1, &m_requestInfo, 16, 16, val);
+ };
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getIndexVersion() const {
+ return m_indexVersion;
+ }
+ void setIndexVersion(Uint32 val) {
+ m_indexVersion = val;
+ }
+ Uint32 getOnline() const {
+ return m_online;
+ }
+ void setOnline(Uint32 val) {
+ m_online = val;
+ }
+ Uint32 getOpKey() const {
+ return m_opKey;
+ }
+ void setOpKey(Uint32 val) {
+ m_opKey = val;
+ }
+};
+
+/**
+ * AlterIndxConf.
+ */
+class AlterIndxConf {
+ friend bool printALTER_INDX_CONF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( InternalLength = 3 );
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+ Uint32 m_userRef;
+ Uint32 m_connectionPtr;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexId;
+ Uint32 m_indexVersion;
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ AlterIndxReq::RequestType getRequestType() const {
+ return (AlterIndxReq::RequestType)m_requestInfo;
+ }
+ void setRequestType(AlterIndxReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getIndexVersion() const {
+ return m_indexVersion;
+ }
+ void setIndexVersion(Uint32 val) {
+ m_indexVersion = val;
+ }
+};
+
+/**
+ * AlterIndxRef.
+ */
+class AlterIndxRef {
+ friend bool printALTER_INDX_REF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum ErrorCode {
+ NoError = 0,
+ Busy = 701,
+ IndexNotFound = 4243,
+ IndexExists = 4244,
+ BadRequestType = 4247,
+ NotAnIndex = 4254,
+ BadState = 4347,
+ Inconsistency = 4348
+ };
+ STATIC_CONST( SignalLength = AlterIndxConf::SignalLength + 3 );
+
+private:
+ AlterIndxConf m_conf;
+ //Uint32 m_userRef;
+ //Uint32 m_connectionPtr;
+ //Uint32 m_requestInfo;
+ //Uint32 m_tableId;
+ //Uint32 m_indexId;
+ //Uint32 m_indexVersion;
+ Uint32 m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+
+public:
+ AlterIndxConf* getConf() {
+ return &m_conf;
+ }
+ const AlterIndxConf* getConf() const {
+ return &m_conf;
+ }
+ Uint32 getUserRef() const {
+ return m_conf.getUserRef();
+ }
+ void setUserRef(Uint32 val) {
+ m_conf.setUserRef(val);
+ }
+ Uint32 getConnectionPtr() const {
+ return m_conf.getConnectionPtr();
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_conf.setConnectionPtr(val);
+ }
+ AlterIndxReq::RequestType getRequestType() const {
+ return m_conf.getRequestType();
+ }
+ void setRequestType(AlterIndxReq::RequestType val) {
+ m_conf.setRequestType(val);
+ }
+ Uint32 getTableId() const {
+ return m_conf.getTableId();
+ }
+ void setTableId(Uint32 val) {
+ m_conf.setTableId(val);
+ }
+ Uint32 getIndexId() const {
+ return m_conf.getIndexId();
+ }
+ void setIndexId(Uint32 val) {
+ m_conf.setIndexId(val);
+ }
+ Uint32 getIndexVersion() const {
+ return m_conf.getIndexVersion();
+ }
+ void setIndexVersion(Uint32 val) {
+ m_conf.setIndexVersion(val);
+ }
+ AlterIndxRef::ErrorCode getErrorCode() const {
+ return (AlterIndxRef::ErrorCode)m_errorCode;
+ }
+ void setErrorCode(AlterIndxRef::ErrorCode val) {
+ m_errorCode = (Uint32)val;
+ }
+ Uint32 getErrorLine() const {
+ return m_errorLine;
+ }
+ void setErrorLine(Uint32 val) {
+ m_errorLine = val;
+ }
+ Uint32 getErrorNode() const {
+ return m_errorNode;
+ }
+ void setErrorNode(Uint32 val) {
+ m_errorNode = val;
+ }
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/AlterTab.hpp b/ndb/include/kernel/signaldata/AlterTab.hpp
new file mode 100644
index 00000000000..02d4eb95d2e
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AlterTab.hpp
@@ -0,0 +1,125 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ALTER_TAB_HPP
+#define ALTER_TAB_HPP
+
+#include "SignalData.hpp"
+#include "GlobalSignalNumbers.h"
+
+/**
+ * AlterTab
+ *
+ * Implemenatation of AlterTable
+ */
+class AlterTabReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+ friend class Dbdih;
+ friend class Dbtc;
+ friend class Dblqh;
+
+ /**
+ * For printing
+ */
+ friend bool printALTER_TAB_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 9 );
+
+ enum RequestType {
+ AlterTablePrepare = 0, // Prepare alter table
+ AlterTableCommit = 1, // Commit alter table
+ AlterTableRevert = 2 // Prepare failed, revert instead
+ };
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 clientRef;
+ Uint32 clientData;
+
+ Uint32 changeMask;
+ Uint32 tableId;
+ Uint32 tableVersion;
+ Uint32 gci;
+ Uint32 requestType;
+
+ SECTION( DICT_TAB_INFO = 0 );
+};
+
+struct AlterTabRef {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+ friend class Dbdih;
+ friend class Dbtc;
+ friend class Dblqh;
+ friend class Dbtup;
+ friend class SafeCounter;
+
+ /**
+ * For printing
+ */
+ friend bool printALTER_TAB_REF(FILE *, const Uint32 *, Uint32, Uint16);
+
+ STATIC_CONST( SignalLength = 7 );
+ STATIC_CONST( GSN = GSN_ALTER_TAB_REF );
+
+ enum ErrorCode {
+ NF_FakeErrorREF = 255
+ };
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 errorCode;
+ Uint32 errorLine;
+ Uint32 errorKey;
+ Uint32 errorStatus;
+ Uint32 requestType;
+};
+
+class AlterTabConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+ friend class Dbdih;
+ friend class Dbtc;
+ friend class Dblqh;
+ friend class Dbtup;
+
+ /**
+ * For printing
+ */
+ friend bool printALTER_TAB_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 7 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 changeMask;
+ Uint32 tableId;
+ Uint32 tableVersion;
+ Uint32 gci;
+ Uint32 requestType;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/AlterTable.hpp b/ndb/include/kernel/signaldata/AlterTable.hpp
new file mode 100644
index 00000000000..30f8727551d
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AlterTable.hpp
@@ -0,0 +1,179 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ALTER_TABLE_HPP
+#define ALTER_TABLE_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * AlterTable
+ *
+ * This signal is sent by API to DICT/TRIX
+ * as a request to alter a secondary index
+ * and then from TRIX to TRIX(n) and TRIX to TC.
+ */
+class AlterTableReq {
+ /**
+ * Sender(s)
+ */
+ // API
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class NdbTableImpl;
+ friend class NdbDictInterface;
+ friend class Dbdict;
+
+ /**
+ * For printing
+ */
+ friend bool printALTER_TABLE_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 changeMask;
+ Uint32 tableId;
+ Uint32 tableVersion;
+
+ SECTION( DICT_TAB_INFO = 0 );
+
+/**
+ * ChangeMask
+ */
+
+/*
+ n = Changed name
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ n-------------------------------
+*/
+#define NAME_SHIFT (0)
+
+ /**
+ * Getters and setters
+ */
+ static Uint8 getNameFlag(const UintR & changeMask);
+ static void setNameFlag(UintR & changeMask, Uint32 nameFlg);
+};
+
+inline
+Uint8
+AlterTableReq::getNameFlag(const UintR & changeMask){
+ return (Uint8)((changeMask >> NAME_SHIFT) & 1);
+}
+
+inline
+void
+AlterTableReq::setNameFlag(UintR & changeMask, Uint32 nameFlg){
+ changeMask |= (nameFlg << NAME_SHIFT);
+}
+
+
+class AlterTableRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Ndbcntr;
+ friend class NdbDictInterface;
+
+ /**
+ * For printing
+ */
+ friend bool printALTER_TABLE_REF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 7 );
+
+ enum ErrorCode {
+ NoError = 0,
+ InvalidTableVersion = 241,
+ DropInProgress = 283,
+ Busy = 701,
+ NotMaster = 702,
+ InvalidFormat = 703,
+ AttributeNameTooLong = 704,
+ TableNameTooLong = 705,
+ Inconsistency = 706,
+ NoMoreTableRecords = 707,
+ NoMoreAttributeRecords = 708,
+ NoSuchTable = 709,
+ AttributeNameTwice = 720,
+ TableAlreadyExist = 721,
+ ArraySizeTooBig = 737,
+ RecordTooBig = 738,
+ InvalidPrimaryKeySize = 739,
+ NullablePrimaryKey = 740,
+ UnsupportedChange = 741
+ };
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 masterNodeId;
+ Uint32 errorCode;
+ Uint32 errorLine;
+ Uint32 errorKey;
+ Uint32 status;
+
+public:
+ Uint32 getErrorCode() const {
+ return errorCode;
+ }
+ Uint32 getErrorLine() const {
+ return errorLine;
+ }
+};
+
+class AlterTableConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Ndbcntr;
+ friend class NdbDictInterface;
+
+ /**
+ * For printing
+ */
+ friend bool printALTER_TABLE_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 tableId;
+ Uint32 tableVersion;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/AlterTrig.hpp b/ndb/include/kernel/signaldata/AlterTrig.hpp
new file mode 100644
index 00000000000..a97c1fd0196
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AlterTrig.hpp
@@ -0,0 +1,288 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ALTER_TRIG_HPP
+#define ALTER_TRIG_HPP
+
+#include "SignalData.hpp"
+#include <Bitmask.hpp>
+#include <trigger_definitions.h>
+
+/**
+ * AlterTrigReq.
+ */
+class AlterTrigReq {
+ friend bool printALTER_TRIG_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum RequestType {
+ RT_UNDEFINED = 0,
+ RT_USER = 1,
+ RT_CREATE_TRIGGER = 2,
+ RT_DROP_TRIGGER = 3,
+ RT_DICT_PREPARE = 1 << 4,
+ RT_DICT_TC = 5 << 4,
+ RT_DICT_LQH = 6 << 4,
+ RT_DICT_COMMIT = 0xC << 4,
+ RT_DICT_ABORT = 0xF << 4
+ };
+ STATIC_CONST( SignalLength = 8 );
+
+private:
+ Uint32 m_userRef;
+ Uint32 m_connectionPtr;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_triggerId;
+ Uint32 m_triggerInfo;
+ Uint32 m_online; // new state 0-offline 1-online
+ Uint32 m_receiverRef; // receiver for subscription trigger
+ // extra
+ Uint32 m_opKey;
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ AlterTrigReq::RequestType getRequestType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_requestInfo, 0, 16);
+ return (AlterTrigReq::RequestType)val;
+ }
+ void setRequestType(AlterTrigReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getRequestFlag() const {
+ return BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ };
+ void addRequestFlag(Uint32 val) {
+ val |= BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ BitmaskImpl::setField(1, &m_requestInfo, 16, 16, val);
+ };
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getTriggerId() const {
+ return m_triggerId;
+ }
+ void setTriggerId(Uint32 val) {
+ m_triggerId = val;
+ }
+ Uint32 getTriggerInfo() const {
+ return m_triggerInfo;
+ }
+ void setTriggerInfo(Uint32 val) {
+ m_triggerInfo = val;
+ }
+ TriggerType::Value getTriggerType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 0, 8);
+ return (TriggerType::Value)val;
+ }
+ void setTriggerType(TriggerType::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 0, 8, (Uint32)val);
+ }
+ TriggerActionTime::Value getTriggerActionTime() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 8, 8);
+ return (TriggerActionTime::Value)val;
+ }
+ void setTriggerActionTime(TriggerActionTime::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 8, 8, (Uint32)val);
+ }
+ TriggerEvent::Value getTriggerEvent() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 16, 8);
+ return (TriggerEvent::Value)val;
+ }
+ void setTriggerEvent(TriggerEvent::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 16, 8, (Uint32)val);
+ }
+ bool getMonitorReplicas() const {
+ return BitmaskImpl::getField(1, &m_triggerInfo, 24, 1);
+ }
+ void setMonitorReplicas(bool val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 24, 1, val);
+ }
+ bool getMonitorAllAttributes() const {
+ return BitmaskImpl::getField(1, &m_triggerInfo, 25, 1);
+ }
+ void setMonitorAllAttributes(bool val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 25, 1, val);
+ }
+ Uint32 getOnline() const {
+ return m_online;
+ }
+ void setOnline(Uint32 val) {
+ m_online = val;
+ }
+ Uint32 getReceiverRef() const {
+ return m_receiverRef;
+ }
+ void setReceiverRef(Uint32 val) {
+ m_receiverRef = val;
+ }
+ Uint32 getOpKey() const {
+ return m_opKey;
+ }
+ void setOpKey(Uint32 val) {
+ m_opKey = val;
+ }
+};
+
+/**
+ * AlterTrigConf.
+ */
+class AlterTrigConf {
+ friend bool printALTER_TRIG_CONF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( InternalLength = 3 );
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ Uint32 m_userRef;
+ Uint32 m_connectionPtr;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_triggerId;
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ AlterTrigReq::RequestType getRequestType() const {
+ return (AlterTrigReq::RequestType)m_requestInfo;
+ }
+ void setRequestType(AlterTrigReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getTriggerId() const {
+ return m_triggerId;
+ }
+ void setTriggerId(Uint32 val) {
+ m_triggerId = val;
+ }
+};
+
+/**
+ * AlterTrigRef.
+ */
+class AlterTrigRef {
+ friend bool printALTER_TRIG_REF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum ErrorCode {
+ NoError = 0,
+ Busy = 701,
+ TriggerNotFound = 4238,
+ TriggerExists = 4239,
+ BadRequestType = 4247
+ };
+ STATIC_CONST( SignalLength = AlterTrigConf::SignalLength + 3 );
+
+private:
+ AlterTrigConf m_conf;
+ //Uint32 m_userRef;
+ //Uint32 m_connectionPtr;
+ //Uint32 m_requestInfo;
+ //Uint32 m_tableId;
+ //Uint32 m_triggerId;
+ Uint32 m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+
+public:
+ AlterTrigConf* getConf() {
+ return &m_conf;
+ }
+ const AlterTrigConf* getConf() const {
+ return &m_conf;
+ }
+ Uint32 getUserRef() const {
+ return m_conf.getUserRef();
+ }
+ void setUserRef(Uint32 val) {
+ m_conf.setUserRef(val);
+ }
+ Uint32 getConnectionPtr() const {
+ return m_conf.getConnectionPtr();
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_conf.setConnectionPtr(val);
+ }
+ AlterTrigReq::RequestType getRequestType() const {
+ return m_conf.getRequestType();
+ }
+ void setRequestType(AlterTrigReq::RequestType val) {
+ m_conf.setRequestType(val);
+ }
+ Uint32 getTableId() const {
+ return m_conf.getTableId();
+ }
+ void setTableId(Uint32 val) {
+ m_conf.setTableId(val);
+ }
+ Uint32 getTriggerId() const {
+ return m_conf.getTriggerId();
+ }
+ void setTriggerId(Uint32 val) {
+ m_conf.setTriggerId(val);
+ }
+ ErrorCode getErrorCode() const {
+ return (ErrorCode)m_errorCode;
+ }
+ void setErrorCode(ErrorCode val) {
+ m_errorCode = (Uint32)val;
+ }
+ Uint32 getErrorLine() const {
+ return m_errorLine;
+ }
+ void setErrorLine(Uint32 val) {
+ m_errorLine = val;
+ }
+ Uint32 getErrorNode() const {
+ return m_errorNode;
+ }
+ void setErrorNode(Uint32 val) {
+ m_errorNode = val;
+ }
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ApiRegSignalData.hpp b/ndb/include/kernel/signaldata/ApiRegSignalData.hpp
new file mode 100644
index 00000000000..84dca8fb260
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ApiRegSignalData.hpp
@@ -0,0 +1,92 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef API_REGCONF_HPP
+#define API_REGCONF_HPP
+
+#include <NodeState.hpp>
+
+class ApiRegReq {
+ /**
+ * Sender(s)
+ */
+ friend class ClusterMgr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 ref;
+ Uint32 version; // Version of API node
+};
+
+/**
+ *
+ */
+class ApiRegRef {
+ /**
+ * Sender(s)
+ */
+ friend class Qmgr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class ClusterMgr;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ enum ErrorCode {
+ WrongType = 1,
+ UnsupportedVersion = 2
+ };
+private:
+ Uint32 ref; // Qmgr ref
+ Uint32 version; // Version of NDB node
+ Uint32 errorCode;
+};
+
+/**
+ *
+ */
+class ApiRegConf {
+ /**
+ * Sender(s)
+ */
+ friend class Qmgr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class ClusterMgr;
+
+public:
+ STATIC_CONST( SignalLength = 3 + NodeState::DataLength );
+private:
+
+ Uint32 qmgrRef;
+ Uint32 version; // Version of NDB node
+ Uint32 apiHeartbeatFrequency;
+ NodeState nodeState;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ApiVersion.hpp b/ndb/include/kernel/signaldata/ApiVersion.hpp
new file mode 100644
index 00000000000..28281e7d186
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ApiVersion.hpp
@@ -0,0 +1,60 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef API_VERSION_HPP
+#define API_VERSION_HPP
+
+class ApiVersionReq {
+/**
+ * Sender(s)
+ */
+ friend class MgmtSrv;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Qmgr;
+public:
+ STATIC_CONST( SignalLength = 3 );
+ Uint32 senderRef;
+ Uint32 nodeId; //api node id
+ Uint32 version; // Version of API node
+
+
+};
+
+
+
+class ApiVersionConf {
+/**
+ * Sender(s)
+ */
+ friend class Qmgr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class MgmtSrv;
+public:
+ STATIC_CONST( SignalLength = 3 );
+ Uint32 senderRef;
+ Uint32 nodeId; //api node id
+ Uint32 version; // Version of API node
+
+
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ArbitSignalData.hpp b/ndb/include/kernel/signaldata/ArbitSignalData.hpp
new file mode 100644
index 00000000000..271b9920cd0
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ArbitSignalData.hpp
@@ -0,0 +1,154 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ARBIT_SIGNAL_DATA_H
+#define ARBIT_SIGNAL_DATA_H
+
+#include <string.h>
+#include <NodeBitmask.hpp>
+#include <NdbTick.h>
+#include <NdbHost.h>
+#include "SignalData.hpp"
+#include "SignalDataPrint.hpp"
+
+/**
+ * The ticket.
+ */
+class ArbitTicket {
+private:
+ Uint32 data[2];
+
+public:
+ STATIC_CONST( DataLength = 2 );
+ STATIC_CONST( TextLength = DataLength * 8 ); // hex digits
+
+ inline void clear() {
+ data[0] = 0;
+ data[1] = 0;
+ }
+
+ inline void update() {
+ Uint16 cnt = data[0] & 0xFFFF; // previous count
+ Uint16 pid = NdbHost_GetProcessId();
+ data[0] = (pid << 16) | (cnt + 1);
+ data[1] = NdbTick_CurrentMillisecond();
+ }
+
+ inline bool match(ArbitTicket& aTicket) const {
+ return
+ data[0] == aTicket.data[0] &&
+ data[1] == aTicket.data[1];
+ }
+
+ inline void getText(char *buf, size_t buf_len) const {
+ snprintf(buf, buf_len, "%08x%08x", data[0], data[1]);
+ }
+
+/* inline char* getText() const {
+ static char buf[TextLength + 1];
+ getText(buf, sizeof(buf));
+ return buf;
+ } */
+};
+
+/**
+ * Result codes. Part of signal data. Each signal uses only
+ * a subset but a common namespace is convenient.
+ */
+class ArbitCode {
+public:
+ STATIC_CONST( ErrTextLength = 80 );
+
+ enum {
+ NoInfo = 0,
+
+ // CFG signals
+ CfgRank1 = 1, // these have to be 1 and 2
+ CfgRank2 = 2,
+
+ // QMGR continueB thread state
+ ThreadStart = 11, // continueB thread started
+
+ // PREP signals
+ PrepPart1 = 21, // zero old ticket
+ PrepPart2 = 22, // get new ticket
+ PrepAtrun = 23, // late joiner gets ticket at RUN time
+
+ // arbitrator state
+ ApiStart = 31, // arbitrator thread started
+ ApiFail = 32, // arbitrator died
+ ApiExit = 33, // arbitrator reported it will exit
+
+ // arbitration result
+ LoseNodes = 41, // lose on ndb node count
+ WinGroups = 42, // we win, no need for arbitration
+ LoseGroups = 43, // we lose, missing node group
+ Partitioning = 44, // possible network partitioning
+ WinChoose = 45, // positive reply
+ LoseChoose = 46, // negative reply
+ LoseNorun = 47, // arbitrator required but not running
+ LoseNocfg = 48, // arbitrator required but none configured
+
+ // general error codes
+ ErrTicket = 91, // invalid arbitrator-ticket
+ ErrToomany = 92, // too many requests
+ ErrState = 93, // invalid state
+ ErrTimeout = 94, // timeout waiting for signals
+ ErrUnknown = 95 // unknown error
+ };
+
+ static inline void getErrText(Uint32 code, char* buf, size_t buf_len) {
+ switch (code) {
+ case ErrTicket:
+ snprintf(buf, buf_len, "invalid arbitrator-ticket");
+ break;
+ case ErrToomany:
+ snprintf(buf, buf_len, "too many requests");
+ break;
+ case ErrState:
+ snprintf(buf, buf_len, "invalid state");
+ break;
+ case ErrTimeout:
+ snprintf(buf, buf_len, "timeout");
+ break;
+ default:
+ snprintf(buf, buf_len, "unknown error [code=%u]", code);
+ break;
+ }
+ }
+};
+
+/**
+ * Common class for arbitration signal data.
+ */
+class ArbitSignalData {
+public:
+ Uint32 sender; // sender's node id (must be word 0)
+ Uint32 code; // result code or other info
+ Uint32 node; // arbitrator node id
+ ArbitTicket ticket; // ticket
+ NodeBitmask mask; // set of nodes
+
+ STATIC_CONST( SignalLength = 3 + ArbitTicket::DataLength + NodeBitmask::Size );
+
+ inline bool match(ArbitSignalData& aData) const {
+ return
+ node == aData.node &&
+ ticket.match(aData.ticket);
+ }
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/AttrInfo.hpp b/ndb/include/kernel/signaldata/AttrInfo.hpp
new file mode 100644
index 00000000000..18bd9b22c40
--- /dev/null
+++ b/ndb/include/kernel/signaldata/AttrInfo.hpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ATTRINFO_HPP
+#define ATTRINFO_HPP
+
+#include "SignalData.hpp"
+
+class AttrInfo {
+ /**
+ * Sender(s)
+ */
+ friend class DbUtil;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtup;
+
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class Dbtc;
+ friend class Dblqh;
+
+ friend bool printATTRINFO(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( HeaderLength = 3 );
+ STATIC_CONST( DataLength = 22 );
+ STATIC_CONST( MaxSignalLength = HeaderLength + DataLength );
+
+private:
+ Uint32 connectPtr;
+ Uint32 transId[2];
+ Uint32 attrData[DataLength];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/BackupContinueB.hpp b/ndb/include/kernel/signaldata/BackupContinueB.hpp
new file mode 100644
index 00000000000..d3d3f79f310
--- /dev/null
+++ b/ndb/include/kernel/signaldata/BackupContinueB.hpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BACKUP_CONTINUEB_H
+#define BACKUP_CONTINUEB_H
+
+#include "SignalData.hpp"
+
+class BackupContinueB {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Backup;
+ friend bool printCONTINUEB_BACKUP(FILE * output, const Uint32 * theData, Uint32 len);
+private:
+ enum {
+ START_FILE_THREAD = 0,
+ BUFFER_UNDERFLOW = 1,
+ BUFFER_FULL_SCAN = 2,
+ BUFFER_FULL_FRAG_COMPLETE = 3,
+ BUFFER_FULL_META = 4
+ };
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/BackupImpl.hpp b/ndb/include/kernel/signaldata/BackupImpl.hpp
new file mode 100644
index 00000000000..1872069daa7
--- /dev/null
+++ b/ndb/include/kernel/signaldata/BackupImpl.hpp
@@ -0,0 +1,366 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BACKUP_IMPL_HPP
+#define BACKUP_IMPL_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+
+class DefineBackupReq {
+ /**
+ * Sender(s)
+ */
+ friend class BackupMaster;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+
+ friend bool printDEFINE_BACKUP_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 8 + NdbNodeBitmask::Size);
+
+private:
+ /**
+ * i - value of backup object
+ */
+ Uint32 backupPtr;
+
+ Uint32 backupId;
+ Uint32 clientRef;
+ Uint32 clientData;
+ Uint32 senderRef;
+
+ /**
+ * Which node(s) is participating in the backup
+ */
+ NdbNodeBitmask nodes;
+
+ /**
+ * Generated random number
+ */
+ Uint32 backupKey[2];
+
+ /**
+ * Length of backup data
+ */
+ Uint32 backupDataLen;
+};
+
+class DefineBackupRef {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class BackupMaster;
+
+ friend bool printDEFINE_BACKUP_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ enum ErrorCode {
+ Undefined = 200,
+ FailedToAllocateBuffers = 202,
+ FailedToSetupFsBuffers = 203,
+ FailedToAllocateTables = 204,
+ FailedInsertFileHeader = 205,
+ FailedInsertTableList = 206,
+ FailedAllocateTableMem = 207,
+ FailedToAllocateFileRecord = 208,
+ FailedToAllocateAttributeRecord = 209
+ };
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 errorCode;
+};
+
+class DefineBackupConf {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class BackupMaster;
+
+ friend bool printDEFINE_BACKUP_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+};
+
+class StartBackupReq {
+ /**
+ * Sender(s)
+ */
+ friend class BackupMaster;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+
+ friend bool printSTART_BACKUP_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+
+ STATIC_CONST( MaxTableTriggers = 4 );
+ STATIC_CONST( HeaderLength = 5 );
+ STATIC_CONST( TableTriggerLength = 4);
+
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 signalNo;
+ Uint32 noOfSignals;
+ Uint32 noOfTableTriggers;
+
+ struct TableTriggers {
+ Uint32 tableId;
+ Uint32 triggerIds[3];
+ } tableTriggers[MaxTableTriggers];
+};
+
+class StartBackupRef {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class BackupMaster;
+
+ friend bool printSTART_BACKUP_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum ErrorCode {
+ FailedToAllocateTriggerRecord = 1
+ };
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 signalNo;
+ Uint32 errorCode;
+};
+
+class StartBackupConf {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class BackupMaster;
+
+ friend bool printSTART_BACKUP_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 signalNo;
+};
+
+class BackupFragmentReq {
+ /**
+ * Sender(s)
+ */
+ friend class BackupMaster;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+
+ friend bool printBACKUP_FRAGMENT_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 tableId;
+ Uint32 fragmentNo;
+ Uint32 count;
+};
+
+class BackupFragmentRef {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class BackupMaster;
+
+ friend bool printBACKUP_FRAGMENT_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 tableId;
+ Uint32 fragmentNo;
+ Uint32 errorCode;
+};
+
+class BackupFragmentConf {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class BackupMaster;
+
+ friend bool printBACKUP_FRAGMENT_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 tableId;
+ Uint32 fragmentNo;
+ Uint32 noOfRecords;
+ Uint32 noOfBytes;
+};
+
+class StopBackupReq {
+ /**
+ * Sender(s)
+ */
+ friend class BackupMaster;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+
+ friend bool printSTOP_BACKUP_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 startGCP;
+ Uint32 stopGCP;
+};
+
+class StopBackupRef {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class BackupMaster;
+
+ friend bool printSTOP_BACKUP_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 errorCode;
+};
+
+class StopBackupConf {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class BackupMaster;
+
+ friend bool printSTOP_BACKUP_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 backupId;
+ Uint32 backupPtr;
+ Uint32 noOfLogBytes;
+ Uint32 noOfLogRecords;
+};
+
+class BackupStatusReq {
+ /**
+ * Sender(s)
+ */
+ friend class BackupMaster;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+
+ friend bool printBACKUP_STATUS_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+};
+
+class BackupStatusConf {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class BackupMaster;
+
+ friend bool printBACKUP_STATUS_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/BackupSignalData.hpp b/ndb/include/kernel/signaldata/BackupSignalData.hpp
new file mode 100644
index 00000000000..42eb8464d53
--- /dev/null
+++ b/ndb/include/kernel/signaldata/BackupSignalData.hpp
@@ -0,0 +1,252 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BACKUP_HPP
+#define BACKUP_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+
+/**
+ * Request to start a backup
+ */
+class BackupReq {
+ /**
+ * Sender(s)
+ */
+ friend class MgmtSrvr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+
+ friend bool printBACKUP_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderData;
+ Uint32 backupDataLen;
+};
+
+class BackupData {
+ /**
+ * Sender(s)
+ */
+ friend class BackupMaster;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+
+ friend bool printBACKUP_DATA(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 25 );
+
+ enum KeyValues {
+ /**
+ * Buffer(s) and stuff
+ */
+ BufferSize = 1, // In MB
+ BlockSize = 2, // Write in chunks of this (in bytes)
+ MinWrite = 3, // Minimum write as multiple of blocksize
+ MaxWrite = 4, // Maximum write as multiple of blocksize
+
+ // Max throughput
+ // Parallell files
+
+ NoOfTables = 1000,
+ TableName = 1001 // char*
+ };
+private:
+ enum RequestType {
+ ClientToMaster = 1,
+ MasterToSlave = 2
+ };
+ Uint32 requestType;
+
+ union {
+ Uint32 backupPtr;
+ Uint32 senderData;
+ };
+ Uint32 backupId;
+
+ /**
+ * totalLen = totalLen_offset >> 16
+ * offset = totalLen_offset & 0xFFFF
+ */
+ Uint32 totalLen_offset;
+
+ /**
+ * Length in this = signal->length() - 3
+ * Sender block ref = signal->senderBlockRef()
+ */
+ Uint32 backupData[21];
+};
+
+/**
+ * The request to start a backup was refused
+ */
+class BackupRef {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class MgmtSrvr;
+
+ friend bool printBACKUP_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ enum ErrorCodes {
+ Undefined = 100,
+ IAmNotMaster = 101,
+ OutOfBackupRecord = 102,
+ OutOfResources = 103,
+ SequenceFailure = 104,
+ BackupDefinitionNotImplemented = 105
+ };
+ Uint32 senderData;
+ Uint32 errorCode;
+ union {
+ Uint32 masterRef;
+ };
+};
+
+/**
+ * The backup has started
+ */
+class BackupConf {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class MgmtSrvr;
+
+ friend bool printBACKUP_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 + NdbNodeBitmask::Size );
+
+private:
+ Uint32 senderData;
+ Uint32 backupId;
+ NdbNodeBitmask nodes;
+};
+
+/**
+ * A backup has been aborted
+ */
+class BackupAbortRep {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class MgmtSrvr;
+
+ friend bool printBACKUP_ABORT_REP(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 senderData;
+ Uint32 backupId;
+ Uint32 reason;
+};
+
+/**
+ * A backup has been completed
+ */
+class BackupCompleteRep {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class MgmtSrvr;
+
+ friend bool printBACKUP_COMPLETE_REP(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 8 + NdbNodeBitmask::Size );
+private:
+ Uint32 senderData;
+ Uint32 backupId;
+ Uint32 startGCP;
+ Uint32 stopGCP;
+ Uint32 noOfBytes;
+ Uint32 noOfRecords;
+ Uint32 noOfLogBytes;
+ Uint32 noOfLogRecords;
+ NdbNodeBitmask nodes;
+};
+
+/**
+ * A master has finished taking-over backup responsiblility
+ */
+class BackupNFCompleteRep {
+ friend bool printBACKUP_NF_COMPLETE_REP(FILE*, const Uint32*, Uint32, Uint16);
+};
+
+/**
+ * Abort of backup
+ */
+class AbortBackupOrd {
+ /**
+ * Sender / Reciver
+ */
+ friend class Backup;
+ friend class MgmtSrvr;
+
+ friend bool printABORT_BACKUP_ORD(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ enum RequestType {
+ ClientAbort = 1,
+ BackupComplete = 2,
+ BackupFailure = 3, // General backup failure coordinator -> slave
+ LogBufferFull = 4, // slave -> coordinator
+ FileOrScanError = 5, // slave -> coordinator
+ BackupFailureDueToNodeFail = 6, // slave -> slave
+ OkToClean = 7 // master -> slave
+ };
+private:
+ Uint32 requestType;
+ Uint32 backupId;
+ union {
+ Uint32 backupPtr;
+ Uint32 senderData;
+ };
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/BlockCommitOrd.hpp b/ndb/include/kernel/signaldata/BlockCommitOrd.hpp
new file mode 100644
index 00000000000..3b33dceb758
--- /dev/null
+++ b/ndb/include/kernel/signaldata/BlockCommitOrd.hpp
@@ -0,0 +1,62 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BLOCK_COMMIT_ORD_HPP
+#define BLOCK_COMMIT_ORD_HPP
+
+/**
+ * These two signals are sent via EXECUTE_DIRECT
+ * to DBDIH from QMGR
+ *
+ * Block make sure that no commit is performed
+ * Unblock turns on commit again
+ */
+
+class BlockCommitOrd {
+ /**
+ * Sender(s)
+ */
+ friend class Qmgr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ Uint32 failNo; // As used by Qmgr
+};
+
+class UnblockCommitOrd {
+ /**
+ * Sender(s)
+ */
+ friend class Qmgr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ Uint32 failNo; // As used by Qmgr
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/BuildIndx.hpp b/ndb/include/kernel/signaldata/BuildIndx.hpp
new file mode 100644
index 00000000000..9cf1123cc61
--- /dev/null
+++ b/ndb/include/kernel/signaldata/BuildIndx.hpp
@@ -0,0 +1,308 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BUILD_INDX_HPP
+#define BUILD_INDX_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+#include <NdbString.h>
+#include <signaldata/DictTabInfo.hpp>
+
+/**
+ * BuildIndxReq
+ *
+ * This signal is sent by DICT to TRIX(n)
+ * as a request to build a secondary index
+ */
+class BuildIndxReq {
+ friend bool printBUILD_INDX_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ enum RequestType {
+ RT_UNDEFINED = 0,
+ RT_USER = 1,
+ RT_ALTER_INDEX = 2,
+ RT_SYSTEMRESTART = 3,
+ RT_DICT_PREPARE = 1 << 4,
+ RT_DICT_TC = 5 << 4,
+ RT_DICT_TRIX = 7 << 4,
+ RT_DICT_TUX = 8 << 4,
+ RT_DICT_COMMIT = 0xC << 4,
+ RT_DICT_ABORT = 0xF << 4,
+ RT_TRIX = 7 << 8
+ };
+ STATIC_CONST( SignalLength = 9 );
+ STATIC_CONST( INDEX_COLUMNS = 0 );
+ STATIC_CONST( KEY_COLUMNS = 1 );
+ STATIC_CONST( NoOfSections = 2 );
+
+private:
+ Uint32 m_userRef; // user block reference
+ Uint32 m_connectionPtr; // user "schema connection"
+ Uint32 m_requestInfo;
+ Uint32 m_buildId; // Suma subscription id
+ Uint32 m_buildKey; // Suma subscription key
+ Uint32 m_tableId; // table being indexed
+ Uint32 m_indexType; // from DictTabInfo::TableType
+ Uint32 m_indexId; // table storing index
+ Uint32 m_parallelism; // number of parallel insert transactions
+ // extra
+ Uint32 m_opKey;
+ // Sent data ends here
+ Uint32 m_slack[25 - SignalLength - 1];
+ Uint32 m_sectionBuffer[MAX_ATTRIBUTES_IN_TABLE * 2];
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ BuildIndxReq::RequestType getRequestType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_requestInfo, 0, 16);
+ return (BuildIndxReq::RequestType)val;
+ }
+ void setRequestType(BuildIndxReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getRequestFlag() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ return (BuildIndxReq::RequestType)val;
+ };
+ void addRequestFlag(Uint32 val) {
+ val |= BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ BitmaskImpl::setField(1, &m_requestInfo, 16, 16, val);
+ };
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getBuildId() const {
+ return m_buildId;
+ }
+ void setBuildId(Uint32 val) {
+ m_buildId = val;
+ }
+ Uint32 getBuildKey() const {
+ return m_buildKey;
+ }
+ void setBuildKey(Uint32 val) {
+ m_buildKey = val;
+ }
+ Uint32 getIndexType() const {
+ return m_indexType;
+ }
+ void setIndexType(Uint32 val) {
+ m_indexType = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getParallelism() const {
+ return m_parallelism;
+ }
+ void setParallelism(Uint32 val) {
+ m_parallelism = val;
+ }
+ Uint32 getOpKey() const {
+ return m_opKey;
+ }
+ void setOpKey(Uint32 val) {
+ m_opKey = val;
+ }
+ // Column order
+ void setColumnOrder(Uint32* indexBuf, Uint32 indexLen,
+ Uint32* keyBuf, Uint32 keyLen,
+ struct LinearSectionPtr orderPtr[]);
+};
+
+inline
+void BuildIndxReq::setColumnOrder(Uint32* indexBuf, Uint32 indexLen,
+ Uint32* keyBuf, Uint32 keyLen,
+ struct LinearSectionPtr orderPtr[])
+
+{
+ printf("BuildIndxReq::setColumnOrder: indexLen %u, keyLen %u\n", indexLen, keyLen);
+ // Copy buffers
+ MEMCOPY_NO_WORDS(m_sectionBuffer, indexBuf, indexLen);
+ MEMCOPY_NO_WORDS(m_sectionBuffer + indexLen, keyBuf, keyLen);
+ orderPtr[INDEX_COLUMNS].p = m_sectionBuffer;
+ orderPtr[INDEX_COLUMNS].sz = indexLen;
+ orderPtr[KEY_COLUMNS].p = m_sectionBuffer + indexLen;
+ orderPtr[KEY_COLUMNS].sz = keyLen;
+}
+
+/**
+ * BuildIndxConf
+ *
+ * This signal is sent back to DICT from TRIX
+ * as confirmation of succesfull index build
+ * (BuildIndxReq).
+ */
+class BuildIndxConf {
+ friend bool printBUILD_INDX_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( InternalLength = 3 );
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+ friend class BuildIndxRef;
+ Uint32 m_userRef;
+ Uint32 m_connectionPtr;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexType;
+ Uint32 m_indexId;
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ BuildIndxReq::RequestType getRequestType() const {
+ return (BuildIndxReq::RequestType)m_requestInfo;
+ }
+ void setRequestType(BuildIndxReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getIndexType() const {
+ return m_indexType;
+ }
+ void setIndexType(Uint32 val) {
+ m_indexType = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+};
+
+/**
+ * BuildIndxRef
+ *
+ * This signal is sent back to API from DICT/TRIX
+ * as refusal of a failed index creation
+ * (BuildIndxReq). It is also sent as refusal
+ * from TC to TRIX and TRIX to DICT.
+ */
+class BuildIndxRef {
+ friend bool printBUILD_INDX_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ enum ErrorCode {
+ NoError = 0,
+ Busy = 701,
+ BadRequestType = 4247,
+ InvalidPrimaryTable = 4249,
+ InvalidIndexType = 4250,
+ IndexNotUnique = 4251,
+ AllocationFailure = 4252,
+ InternalError = 4346
+ };
+ STATIC_CONST( SignalLength = BuildIndxConf::SignalLength + 1 );
+
+private:
+ //Uint32 m_userRef;
+ //Uint32 m_connectionPtr;
+ //Uint32 m_requestInfo;
+ //Uint32 m_tableId;
+ //Uint32 m_indexType;
+ //Uint32 m_indexId;
+ BuildIndxConf m_conf;
+ Uint32 m_errorCode;
+
+public:
+ BuildIndxConf* getConf() {
+ return &m_conf;
+ }
+ const BuildIndxConf* getConf() const {
+ return &m_conf;
+ }
+ Uint32 getUserRef() const {
+ return m_conf.getUserRef();
+ }
+ void setUserRef(Uint32 val) {
+ m_conf.setUserRef(val);
+ }
+ Uint32 getConnectionPtr() const {
+ return m_conf.getConnectionPtr();
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_conf.setConnectionPtr(val);
+ }
+ BuildIndxReq::RequestType getRequestType() const {
+ return m_conf.getRequestType();
+ }
+ void setRequestType(BuildIndxReq::RequestType val) {
+ m_conf.setRequestType(val);
+ }
+ Uint32 getTableId() const {
+ return m_conf.getTableId();
+ }
+ void setTableId(Uint32 val) {
+ m_conf.setTableId(val);
+ }
+ Uint32 getIndexType() const {
+ return m_conf.getIndexType();
+ }
+ void setIndexType(Uint32 val) {
+ m_conf.setIndexType(val);
+ }
+ Uint32 getIndexId() const {
+ return m_conf.getIndexId();
+ }
+ void setIndexId(Uint32 val) {
+ m_conf.setIndexId(val);
+ }
+ BuildIndxRef::ErrorCode getErrorCode() const {
+ return (BuildIndxRef::ErrorCode)m_errorCode;
+ }
+ void setErrorCode(BuildIndxRef::ErrorCode val) {
+ m_errorCode = (Uint32)val;
+ }
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CheckNodeGroups.hpp b/ndb/include/kernel/signaldata/CheckNodeGroups.hpp
new file mode 100644
index 00000000000..9b2f847e128
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CheckNodeGroups.hpp
@@ -0,0 +1,63 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CHECKNODEGROUPS_H
+#define CHECKNODEGROUPS_H
+
+#include <string.h>
+#include <NodeBitmask.hpp>
+#include "SignalData.hpp"
+#include "SignalDataPrint.hpp"
+
+/**
+ * Ask DIH to check if a node set can survive i.e. if it
+ * has at least one node in every node group. Returns one
+ * of Win, Lose, Partitioning.
+ *
+ * Same class is used for REQ and CONF. The REQ can also
+ * be executed as a direct signal.
+ */
+class CheckNodeGroups {
+public:
+ Uint32 blockRef; // sender's node id
+ union {
+ Uint32 requestType; // direct flag, output code
+ Uint32 output;
+ };
+ union {
+ Uint32 nodeId; // nodeId input for GetNodeGroupMembers
+ NodeBitmask mask; /* set of NDB nodes, input for ArbitCheck,
+ * output for GetNodeGroupMembers
+ */
+ };
+
+ enum RequestType {
+ Direct = 0x1,
+ ArbitCheck = 0x2,
+ GetNodeGroup = 0x4,
+ GetNodeGroupMembers = 0x8
+ };
+
+ enum Output {
+ Lose = 1, // we cannot survive
+ Win = 2, // we and only we can survive
+ Partitioning = 3 // possible network partitioning
+ };
+
+ STATIC_CONST( SignalLength = 2 + NodeBitmask::Size );
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CloseComReqConf.hpp b/ndb/include/kernel/signaldata/CloseComReqConf.hpp
new file mode 100644
index 00000000000..3d3dc54ba64
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CloseComReqConf.hpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CLOSE_COMREQCONF_HPP
+#define CLOSE_COMREQCONF_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+
+/**
+ * The Req signal is sent by Qmgr to Cmvmi
+ * and the Conf signal is sent back
+ *
+ * NOTE that the signals are identical
+ */
+class CloseComReqConf {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Qmgr;
+ friend class Cmvmi;
+
+ /**
+ * For printing
+ */
+ friend bool printCLOSECOMREQCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 3 + NodeBitmask::Size );
+private:
+
+ Uint32 xxxBlockRef;
+ Uint32 failNo;
+
+ Uint32 noOfNodes;
+ Uint32 theNodes[NodeBitmask::Size];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CmInit.hpp b/ndb/include/kernel/signaldata/CmInit.hpp
new file mode 100644
index 00000000000..b59547b767b
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CmInit.hpp
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CM_INIT_HPP
+#define CM_INIT_HPP
+
+#include <NodeBitmask.hpp>
+
+/**
+ *
+ */
+class CmInit {
+ /**
+ * Sender(s)
+ */
+ friend class Cmvmi;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 4 + NodeBitmask::Size );
+private:
+
+ Uint32 heartbeatDbDb;
+ Uint32 heartbeatDbApi;
+ Uint32 inactiveTransactionCheck;
+ Uint32 arbitTimeout;
+
+ Uint32 allNdbNodes[NodeBitmask::Size];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CmRegSignalData.hpp b/ndb/include/kernel/signaldata/CmRegSignalData.hpp
new file mode 100644
index 00000000000..f33c991249f
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CmRegSignalData.hpp
@@ -0,0 +1,192 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CM_REG_HPP
+#define CM_REG_HPP
+
+#include <NodeBitmask.hpp>
+
+/**
+ * This is the first distributed signal
+ * (the node tries to register in the cluster)
+ */
+class CmRegReq {
+ /**
+ * Sender(s) & Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+private:
+
+ Uint32 blockRef;
+ Uint32 nodeId;
+ Uint32 version; // See ndb_version.h
+};
+
+/**
+ * The node receving this signal has been accepted into the cluster
+ */
+class CmRegConf {
+ /**
+ * Sender(s) & Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 4 + NdbNodeBitmask::Size );
+private:
+
+ Uint32 presidentBlockRef;
+ Uint32 presidentNodeId;
+ Uint32 presidentVersion;
+
+ /**
+ * The dynamic id that the node reciving this signal has
+ */
+ Uint32 dynamicId;
+
+ Uint32 allNdbNodes[NdbNodeBitmask::Size];
+};
+
+/**
+ *
+ */
+class CmRegRef {
+ /**
+ * Sender(s) & Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum ErrorCode {
+ ZBUSY = 0, /* Only the president can send this */
+ ZBUSY_PRESIDENT = 1,/* Only the president can send this */
+ ZBUSY_TO_PRES = 2, /* Only the president can send this */
+ ZNOT_IN_CFG = 3, /* Only the president can send this */
+ ZELECTION = 4, /* Receiver is definitely not president,
+ * but we are not sure if sender ends up
+ * as president. */
+ ZNOT_PRESIDENT = 5, /* We are not president */
+ ZNOT_DEAD = 6, /* We are not dead when we are starting */
+ ZINCOMPATIBLE_VERSION = 7
+ };
+private:
+
+ Uint32 blockRef;
+ Uint32 nodeId;
+ Uint32 errorCode;
+ Uint32 presidentCandidate;
+};
+
+class CmAdd {
+ /**
+ * Sender(s) & Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ enum RequestType {
+ Prepare = 0,
+ AddCommit = 1,
+ CommitNew = 2
+ };
+
+ Uint32 requestType;
+ Uint32 startingNodeId;
+ Uint32 startingVersion;
+};
+
+class CmAckAdd {
+ /**
+ * Sender(s) & Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 senderNodeId;
+ Uint32 requestType; // see CmAdd::RequestType
+ Uint32 startingNodeId;
+};
+
+class CmNodeInfoReq {
+ /**
+ * Sender(s) & Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ /**
+ * This is information for sending node (starting node)
+ */
+ Uint32 nodeId;
+ Uint32 dynamicId;
+ Uint32 version;
+};
+
+class CmNodeInfoRef {
+ /**
+ * Sender(s) & Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ enum ErrorCode {
+ NotRunning = 1
+ };
+
+private:
+ Uint32 nodeId;
+ Uint32 errorCode;
+};
+
+class CmNodeInfoConf {
+ /**
+ * Sender(s) & Reciver(s)
+ */
+ friend class Qmgr;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 nodeId;
+ Uint32 dynamicId;
+ Uint32 version;
+};
+
+#endif
+
+
+
+
+
+
+
diff --git a/ndb/include/kernel/signaldata/CmvmiCfgConf.hpp b/ndb/include/kernel/signaldata/CmvmiCfgConf.hpp
new file mode 100644
index 00000000000..12b785723d9
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CmvmiCfgConf.hpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CMVMI_CFGCONF_H
+#define CMVMI_CFGCONF_H
+
+#include "SignalData.hpp"
+
+/**
+ * This signal is used for transfering the
+ * ISP_X Data
+ *
+ * I.e. Configuration data which is sent in a specific start phase
+ *
+ */
+class CmvmiCfgConf {
+ /**
+ * Sender(s)
+ */
+ friend class Cmvmi;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+
+public:
+ STATIC_CONST( NO_OF_WORDS = 16 );
+ STATIC_CONST( LENGTH = 17 );
+private:
+
+ Uint32 startPhase;
+ Uint32 theData[NO_OF_WORDS];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CntrMasterConf.hpp b/ndb/include/kernel/signaldata/CntrMasterConf.hpp
new file mode 100644
index 00000000000..e6bf363ea68
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CntrMasterConf.hpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CNTR_MASTERCONF_HPP
+#define CNTR_MASTERCONF_HPP
+
+#include <NodeBitmask.hpp>
+
+/**
+ * This signals is sent by NdbCntr-Master to NdbCntr
+ */
+class CntrMasterConf {
+ /**
+ * Sender(s)
+ */
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Reciver(s)
+ */
+
+public:
+ STATIC_CONST( SignalLength = 1 + NodeBitmask::Size );
+private:
+
+ Uint32 noStartNodes;
+ Uint32 theNodes[NodeBitmask::Size];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CntrMasterReq.hpp b/ndb/include/kernel/signaldata/CntrMasterReq.hpp
new file mode 100644
index 00000000000..caf9efb1243
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CntrMasterReq.hpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CNTR_MASTERREQ_HPP
+#define CNTR_MASTERREQ_HPP
+
+#include <NodeBitmask.hpp>
+
+/**
+ * This signals is sent by NdbCntr-Master to NdbCntr
+ */
+class CntrMasterReq {
+ /**
+ * Sender(s)
+ */
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Reciver(s)
+ */
+
+public:
+ STATIC_CONST( SignalLength = 4 + NodeBitmask::Size );
+private:
+
+ Uint32 userBlockRef;
+ Uint32 userNodeId;
+ Uint32 typeOfStart;
+ Uint32 noRestartNodes;
+ Uint32 theNodes[NodeBitmask::Size];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ConfigParamId.hpp b/ndb/include/kernel/signaldata/ConfigParamId.hpp
new file mode 100644
index 00000000000..9d9e04957ab
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ConfigParamId.hpp
@@ -0,0 +1,71 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ConfigParamId_H
+#define ConfigParamId_H
+
+ enum ConfigParamId {
+
+ Id,
+ ExecuteOnComputer,
+ MaxNoOfSavedMessages,
+ ShmKey,
+
+ LockPagesInMainMemory,
+ TimeBetweenWatchDogCheck,
+ StopOnError,
+
+ MaxNoOfConcurrentOperations,
+ MaxNoOfConcurrentTransactions,
+ MemorySpaceIndexes,
+ MemorySpaceTuples,
+ MemoryDiskPages,
+ NoOfFreeDiskClusters,
+ NoOfDiskClusters,
+
+ TimeToWaitAlive,
+ HeartbeatIntervalDbDb,
+ HeartbeatIntervalDbApi,
+ ArbitTimeout,
+
+ TimeBetweenLocalCheckpoints,
+ TimeBetweenGlobalCheckpoints,
+ NoOfFragmentLogFiles,
+ NoOfConcurrentCheckpointsDuringRestart,
+ TransactionDeadlockDetectionTimeout,
+ TransactionInactiveTime,
+ NoOfConcurrentProcessesHandleTakeover,
+
+ NoOfConcurrentCheckpointsAfterRestart,
+
+ NoOfDiskPagesToDiskDuringRestartTUP,
+ NoOfDiskPagesToDiskAfterRestartTUP,
+ NoOfDiskPagesToDiskDuringRestartACC,
+ NoOfDiskPagesToDiskAfterRestartACC,
+
+ NoOfDiskClustersPerDiskFile,
+ NoOfDiskFiles,
+
+ MaxNoOfSavedEvents
+ };
+
+#endif // ConfigParamId_H
+
+
+
+
+
+
diff --git a/ndb/include/kernel/signaldata/ContinueFragmented.hpp b/ndb/include/kernel/signaldata/ContinueFragmented.hpp
new file mode 100644
index 00000000000..3d12b9e51eb
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ContinueFragmented.hpp
@@ -0,0 +1,36 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CONTINUE_FRAGMENTED_HPP
+#define CONTINUE_FRAGMENTED_HPP
+
+#include "SignalData.hpp"
+
+class ContinueFragmented {
+
+ /**
+ * Sender/Reciver(s)
+ */
+ friend class SimulatedBlock;
+
+ friend bool printCONTINUE_FRAGMENTED(FILE *,const Uint32 *, Uint32, Uint16);
+public:
+
+private:
+ Uint32 line;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CopyActive.hpp b/ndb/include/kernel/signaldata/CopyActive.hpp
new file mode 100644
index 00000000000..19b05bda072
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CopyActive.hpp
@@ -0,0 +1,84 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef COPY_ACTIVE_HPP
+#define COPY_ACTIVE_HPP
+
+#include "SignalData.hpp"
+
+class CopyActiveReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 distributionKey;
+};
+
+class CopyActiveConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ Uint32 userPtr;
+ Uint32 startingNodeId;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 startGci;
+};
+class CopyActiveRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ Uint32 userPtr;
+ Uint32 startingNodeId;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CopyFrag.hpp b/ndb/include/kernel/signaldata/CopyFrag.hpp
new file mode 100644
index 00000000000..67b935dda64
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CopyFrag.hpp
@@ -0,0 +1,87 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef COPY_FRAG_HPP
+#define COPY_FRAG_HPP
+
+#include "SignalData.hpp"
+
+class CopyFragReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+public:
+ STATIC_CONST( SignalLength = 7 );
+
+private:
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 nodeId;
+ Uint32 schemaVersion;
+ Uint32 distributionKey;
+};
+
+class CopyFragConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ Uint32 userPtr;
+ Uint32 sendingNodeId;
+ Uint32 startingNodeId;
+ Uint32 tableId;
+ Uint32 fragId;
+};
+class CopyFragRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+ Uint32 userPtr;
+ Uint32 sendingNodeId;
+ Uint32 startingNodeId;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CopyGCIReq.hpp b/ndb/include/kernel/signaldata/CopyGCIReq.hpp
new file mode 100644
index 00000000000..4b401654de3
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CopyGCIReq.hpp
@@ -0,0 +1,63 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef COPY_GCI_REQ_HPP
+#define COPY_GCI_REQ_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * This signal is used for transfering the sysfile
+ * between Dih on different nodes.
+ *
+ * The master will distributes the file to the other nodes
+ *
+ * Since the Sysfile can be larger than on StartMeConf signal,
+ * there might be more than on of these signals sent before
+ * the entire sysfile is transfered
+
+ */
+class CopyGCIReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printCOPY_GCI_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ enum CopyReason {
+ IDLE = 0,
+ LOCAL_CHECKPOINT = 1,
+ RESTART = 2,
+ GLOBAL_CHECKPOINT = 3,
+ INITIAL_START_COMPLETED = 4
+ };
+
+private:
+
+ Uint32 anyData;
+ Uint32 copyReason;
+ Uint32 startWord;
+
+ /**
+ * No of free words to carry data
+ */
+ STATIC_CONST( DATA_SIZE = 22 );
+
+ Uint32 data[DATA_SIZE];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CreateEvnt.hpp b/ndb/include/kernel/signaldata/CreateEvnt.hpp
new file mode 100644
index 00000000000..7398fb6d8cc
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CreateEvnt.hpp
@@ -0,0 +1,488 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CREATE_EVNT_HPP
+#define CREATE_EVNT_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+#include <signaldata/DictTabInfo.hpp>
+
+/**
+ * DropEvntReq.
+ */
+class DropEvntReq {
+ friend bool printDROP_EVNT_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+ SECTION( EVENT_NAME_SECTION = 0 );
+
+ union { // user block reference
+ Uint32 senderRef;
+ Uint32 m_userRef;
+ };
+ union {
+ Uint32 senderData;
+ Uint32 m_userData; // user
+ };
+
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getUserData() const {
+ return m_userData;
+ }
+ void setUserData(Uint32 val) {
+ m_userData = val;
+ }
+};
+
+/**
+ * DropEvntConf.
+ */
+class DropEvntConf {
+ friend bool printDROP_EVNT_CONF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ union { // user block reference
+ Uint32 senderRef;
+ Uint32 m_userRef;
+ };
+ union {
+ Uint32 senderData;
+ Uint32 m_userData; // user
+ };
+
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getUserData() const {
+ return m_userData;
+ }
+ void setUserData(Uint32 val) {
+ m_userData = val;
+ }
+};
+
+/**
+ * DropEvntRef.
+ */
+class DropEvntRef {
+ friend bool printDROP_EVNT_REF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum ErrorCode {
+ NoError = 0,
+ Undefined = 1,
+ UndefinedTCError = 2,
+ NF_FakeErrorREF = 11,
+ Busy = 701,
+ NotMaster = 702,
+ SeizeError = 703,
+ EventNotFound = 4238,
+ EventNameTooLong = 4241,
+ TooManyEvents = 4242,
+ BadRequestType = 4247,
+ InvalidName = 4248,
+ InvalidPrimaryTable = 4249,
+ InvalidEventType = 4250,
+ NotUnique = 4251,
+ AllocationError = 4252,
+ CreateEventTableFailed = 4253,
+ InvalidAttributeOrder = 4255,
+ Temporary = 0x1 << 16
+ };
+
+ STATIC_CONST( SignalLength = 5 );
+
+ union { // user block reference
+ Uint32 senderRef;
+ Uint32 m_userRef;
+ };
+ union {
+ Uint32 senderData;
+ Uint32 m_userData; // user
+ };
+ union {
+ Uint32 errorCode;
+ Uint32 m_errorCode;
+ };
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+
+ bool isTemporary() const
+ { return (errorCode & Temporary) > 0; }
+
+ void setTemporary()
+ { errorCode |= Temporary; }
+
+ ErrorCode setTemporary(ErrorCode ec)
+ { return (ErrorCode) (errorCode = ((Uint32) ec | (Uint32)Temporary)); }
+
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getUserData() const {
+ return m_userData;
+ }
+ void setUserData(Uint32 val) {
+ m_userData = val;
+ }
+ DropEvntRef::ErrorCode getErrorCode() const {
+ return (DropEvntRef::ErrorCode)m_errorCode;
+ }
+ void setErrorCode(DropEvntRef::ErrorCode val) {
+ m_errorCode = (Uint32)val;
+ }
+ Uint32 getErrorLine() const {
+ return m_errorLine;
+ }
+ void setErrorLine(Uint32 val) {
+ m_errorLine = val;
+ }
+ Uint32 getErrorNode() const {
+ return m_errorNode;
+ }
+ void setErrorNode(Uint32 val) {
+ m_errorNode = val;
+ }
+};
+
+/**
+ * CreateEvntReq.
+ */
+struct CreateEvntReq {
+ friend bool printCREATE_EVNT_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+ enum RequestType {
+ RT_UNDEFINED = 0,
+ RT_USER_CREATE = 1,
+ RT_USER_GET = 2,
+
+ RT_DICT_AFTER_GET = 0x1 << 4
+ // RT_DICT_MASTER = 0x2 << 4,
+
+ // RT_DICT_COMMIT = 0xC << 4,
+ // RT_DICT_ABORT = 0xF << 4,
+ // RT_TC = 5 << 8
+ };
+ STATIC_CONST( SignalLengthGet = 3 );
+ STATIC_CONST( SignalLengthCreate = 5+MAXNROFATTRIBUTESINWORDS );
+ STATIC_CONST( SignalLength = 7+MAXNROFATTRIBUTESINWORDS );
+ // SECTION( ATTRIBUTE_LIST_SECTION = 0 );
+ SECTION( EVENT_NAME_SECTION = 0 );
+
+ union {
+ Uint32 m_userRef; // user block reference
+ Uint32 senderRef; // user block reference
+ };
+ union {
+ Uint32 m_userData; // user
+ Uint32 senderData; // user
+ };
+ Uint32 m_requestInfo;
+ Uint32 m_tableId; // table to event
+ AttributeMask::Data m_attrListBitmask;
+ Uint32 m_eventType; // from DictTabInfo::TableType
+ Uint32 m_eventId; // event table id set by DICT/SUMA
+ Uint32 m_eventKey; // event table key set by DICT/SUMA
+
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getUserData() const {
+ return m_userData;
+ }
+ void setUserData(Uint32 val) {
+ m_userData = val;
+ }
+ CreateEvntReq::RequestType getRequestType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_requestInfo, 0, 16);
+ return (CreateEvntReq::RequestType)val;
+ }
+ void setRequestType(CreateEvntReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getRequestFlag() const {
+ return BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ };
+ void addRequestFlag(Uint32 val) {
+ val |= BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ BitmaskImpl::setField(1, &m_requestInfo, 16, 16, val);
+ };
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ AttributeMask getAttrListBitmask() const {
+ AttributeMask tmp;
+ tmp.assign(m_attrListBitmask);
+ return tmp;
+ }
+ void setAttrListBitmask(const AttributeMask & val) {
+ m_attrListBitmask = val;
+ }
+ Uint32 getEventType() const {
+ return m_eventType;
+ }
+ void setEventType(Uint32 val) {
+ m_eventType = (Uint32)val;
+ }
+ Uint32 getEventId() const {
+ return m_eventId;
+ }
+ void setEventId(Uint32 val) {
+ m_eventId = val;
+ }
+ Uint32 getEventKey() const {
+ return m_eventKey;
+ }
+ void setEventKey(Uint32 val) {
+ m_eventKey = val;
+ }
+};
+
+/**
+ * CreateEvntConf.
+ */
+class CreateEvntConf {
+ friend bool printCREATE_EVNT_CONF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ // STATIC_CONST( InternalLength = 3 );
+ STATIC_CONST( SignalLength = 7+MAXNROFATTRIBUTESINWORDS );
+
+ union {
+ Uint32 m_userRef; // user block reference
+ Uint32 senderRef; // user block reference
+ };
+ union {
+ Uint32 m_userData; // user
+ Uint32 senderData; // user
+ };
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ AttributeMask m_attrListBitmask;
+ Uint32 m_eventType;
+ Uint32 m_eventId;
+ Uint32 m_eventKey;
+
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getUserData() const {
+ return m_userData;
+ }
+ void setUserData(Uint32 val) {
+ m_userData = val;
+ }
+ CreateEvntReq::RequestType getRequestType() const {
+ return (CreateEvntReq::RequestType)m_requestInfo;
+ }
+ void setRequestType(CreateEvntReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ AttributeMask getAttrListBitmask() const {
+ return m_attrListBitmask;
+ }
+ void setAttrListBitmask(const AttributeMask & val) {
+ m_attrListBitmask = val;
+ }
+ Uint32 getEventType() const {
+ return m_eventType;
+ }
+ void setEventType(Uint32 val) {
+ m_eventType = (Uint32)val;
+ }
+ Uint32 getEventId() const {
+ return m_eventId;
+ }
+ void setEventId(Uint32 val) {
+ m_eventId = val;
+ }
+ Uint32 getEventKey() const {
+ return m_eventKey;
+ }
+ void setEventKey(Uint32 val) {
+ m_eventKey = val;
+ }
+};
+
+/**
+ * CreateEvntRef.
+ */
+struct CreateEvntRef {
+ friend class SafeCounter;
+ friend bool printCREATE_EVNT_REF(FILE*, const Uint32*, Uint32, Uint16);
+
+ STATIC_CONST( SignalLength = 10 );
+ enum ErrorCode {
+ NoError = 0,
+ Undefined = 1,
+ UndefinedTCError = 2,
+ NF_FakeErrorREF = 11,
+ Busy = 701,
+ NotMaster = 702,
+ SeizeError = 703,
+ EventNotFound = 4238,
+ EventExists = 4239,
+ EventNameTooLong = 4241,
+ TooManyEvents = 4242,
+ // EventExists = 4244,
+ AttributeNotStored = 4245,
+ AttributeNullable = 4246,
+ BadRequestType = 4247,
+ InvalidName = 4248,
+ InvalidPrimaryTable = 4249,
+ InvalidEventType = 4250,
+ NotUnique = 4251,
+ AllocationError = 4252,
+ CreateEventTableFailed = 4253,
+ InvalidAttributeOrder = 4255,
+ Temporary = 0x1 << 16
+ };
+ bool isTemporary() const;
+ void setTemporary();
+ ErrorCode setTemporary(ErrorCode ec);
+ static ErrorCode makeTemporary(ErrorCode ec);
+
+ union {
+ Uint32 m_userRef; // user block reference
+ Uint32 senderRef; // user block reference
+ };
+ union {
+ Uint32 m_userData; // user
+ Uint32 senderData; // user
+ };
+
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_eventType;
+ Uint32 m_eventId;
+ Uint32 m_eventKey;
+ Uint32 errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+
+#if 0
+ CreateEvntConf* getConf() {
+ return &m_conf;
+ }
+ const CreateEvntConf* getConf() const {
+ return &m_conf;
+ }
+#endif
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getUserData() const {
+ return m_userData;
+ }
+ void setUserData(Uint32 val) {
+ m_userData = val;
+ }
+ CreateEvntReq::RequestType getRequestType() const {
+ return (CreateEvntReq::RequestType)m_requestInfo;
+ }
+ void setRequestType(CreateEvntReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+
+ Uint32 getEventType() const {
+ return m_eventType;
+ }
+ void setEventType(Uint32 val) {
+ m_eventType = (Uint32)val;
+ }
+ Uint32 getEventId() const {
+ return m_eventId;
+ }
+ void setEventId(Uint32 val) {
+ m_eventId = val;
+ }
+ Uint32 getEventKey() const {
+ return m_eventKey;
+ }
+ void setEventKey(Uint32 val) {
+ m_eventKey = val;
+ }
+
+ CreateEvntRef::ErrorCode getErrorCode() const {
+ return (CreateEvntRef::ErrorCode)errorCode;
+ }
+ void setErrorCode(CreateEvntRef::ErrorCode val) {
+ errorCode = (Uint32)val;
+ }
+ Uint32 getErrorLine() const {
+ return m_errorLine;
+ }
+ void setErrorLine(Uint32 val) {
+ m_errorLine = val;
+ }
+ Uint32 getErrorNode() const {
+ return m_errorNode;
+ }
+ void setErrorNode(Uint32 val) {
+ m_errorNode = val;
+ }
+};
+inline bool CreateEvntRef::isTemporary() const
+{ return (errorCode & CreateEvntRef::Temporary) > 0; };
+inline void CreateEvntRef::setTemporary()
+{ errorCode |= CreateEvntRef::Temporary; };
+inline CreateEvntRef::ErrorCode CreateEvntRef::setTemporary(ErrorCode ec)
+{ return (CreateEvntRef::ErrorCode)
+ (errorCode = ((Uint32) ec | (Uint32)CreateEvntRef::Temporary)); };
+inline CreateEvntRef::ErrorCode CreateEvntRef::makeTemporary(ErrorCode ec)
+{ return (CreateEvntRef::ErrorCode)
+ ( (Uint32) ec | (Uint32)CreateEvntRef::Temporary ); };
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CreateFrag.hpp b/ndb/include/kernel/signaldata/CreateFrag.hpp
new file mode 100644
index 00000000000..a7b3f836353
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CreateFrag.hpp
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CREATE_FRAG_HPP
+#define CREATE_FRAG_HPP
+
+class CreateFragReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 8 );
+
+ enum ReplicaType {
+ STORED = 7,
+ COMMIT_STORED = 9
+ };
+private:
+
+ Uint32 userPtr;
+ BlockReference userRef;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 startingNodeId;
+ Uint32 copyNodeId;
+ Uint32 startGci;
+ Uint32 replicaType;
+};
+
+class CreateFragConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 5 );
+private:
+
+ Uint32 userPtr;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 sendingNodeId;
+ Uint32 startingNodeId;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/CreateFragmentation.hpp b/ndb/include/kernel/signaldata/CreateFragmentation.hpp
new file mode 100644
index 00000000000..a2f45a9580d
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CreateFragmentation.hpp
@@ -0,0 +1,101 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CREATE_FRAGMENTATION_REQ_HPP
+#define CREATE_FRAGMENTATION_REQ_HPP
+
+#include "SignalData.hpp"
+
+class CreateFragmentationReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printCREATE_FRAGMENTATION_REQ(FILE *,
+ const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 fragmentationType;
+ Uint32 noOfFragments;
+ Uint32 fragmentNode;
+ Uint32 primaryTableId; // use same fragmentation as this table if not RNIL
+};
+
+class CreateFragmentationRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printCREATE_FRAGMENTATION_REF(FILE *,
+ const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ enum ErrorCode {
+ OK = 0
+ ,InvalidFragmentationType = 1
+ ,InvalidNodeId = 2
+ ,InvalidNodeType = 3
+ ,InvalidPrimaryTable = 4
+ };
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 errorCode;
+};
+
+class CreateFragmentationConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printCREATE_FRAGMENTATION_CONF(FILE *,
+ const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+ SECTION( FRAGMENTS = 0 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 noOfReplicas;
+ Uint32 noOfFragments;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CreateIndx.hpp b/ndb/include/kernel/signaldata/CreateIndx.hpp
new file mode 100644
index 00000000000..3e277b38dea
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CreateIndx.hpp
@@ -0,0 +1,295 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CREATE_INDX_HPP
+#define CREATE_INDX_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+#include <signaldata/DictTabInfo.hpp>
+
+/**
+ * CreateIndxReq.
+ */
+class CreateIndxReq {
+ friend bool printCREATE_INDX_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum RequestType {
+ RT_UNDEFINED = 0,
+ RT_USER = 1,
+ RT_DICT_PREPARE = 1 << 4,
+ RT_DICT_COMMIT = 0xC << 4,
+ RT_DICT_ABORT = 0xF << 4,
+ RT_TC = 5 << 8
+ };
+ STATIC_CONST( SignalLength = 8 );
+ SECTION( ATTRIBUTE_LIST_SECTION = 0 );
+ SECTION( INDEX_NAME_SECTION = 1 );
+
+private:
+ Uint32 m_connectionPtr; // user "schema connection"
+ Uint32 m_userRef; // user block reference
+ Uint32 m_requestInfo;
+ Uint32 m_tableId; // table to index
+ Uint32 m_indexType; // from DictTabInfo::TableType
+ Uint32 m_indexId; // index table id set by DICT
+ Uint32 m_indexVersion; // index table version set by DICT
+ Uint32 m_online; // alter online
+ // extra
+ Uint32 m_opKey;
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ CreateIndxReq::RequestType getRequestType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_requestInfo, 0, 16);
+ return (CreateIndxReq::RequestType)val;
+ }
+ void setRequestType(CreateIndxReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getRequestFlag() const {
+ return BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ };
+ void addRequestFlag(Uint32 val) {
+ val |= BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ BitmaskImpl::setField(1, &m_requestInfo, 16, 16, val);
+ };
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ DictTabInfo::TableType getIndexType() const {
+ return (DictTabInfo::TableType)m_indexType;
+ }
+ void setIndexType(DictTabInfo::TableType val) {
+ m_indexType = (Uint32)val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getOnline() const {
+ return m_online;
+ }
+ void setOnline(Uint32 val) {
+ m_online = val;
+ }
+ Uint32 getIndexVersion() const {
+ return m_indexVersion;
+ }
+ void setIndexVersion(Uint32 val) {
+ m_indexVersion = val;
+ }
+ Uint32 getOpKey() const {
+ return m_opKey;
+ }
+ void setOpKey(Uint32 val) {
+ m_opKey = val;
+ }
+};
+
+/**
+ * CreateIndxConf.
+ */
+class CreateIndxConf {
+ friend bool printCREATE_INDX_CONF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( InternalLength = 3 );
+ STATIC_CONST( SignalLength = 7 );
+
+private:
+ Uint32 m_connectionPtr;
+ Uint32 m_userRef;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexType;
+ Uint32 m_indexId;
+ Uint32 m_indexVersion;
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ CreateIndxReq::RequestType getRequestType() const {
+ return (CreateIndxReq::RequestType)m_requestInfo;
+ }
+ void setRequestType(CreateIndxReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ DictTabInfo::TableType getIndexType() const {
+ return (DictTabInfo::TableType)m_indexType;
+ }
+ void setIndexType(DictTabInfo::TableType val) {
+ m_indexType = (Uint32)val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getIndexVersion() const {
+ return m_indexVersion;
+ }
+ void setIndexVersion(Uint32 val) {
+ m_indexVersion = val;
+ }
+};
+
+/**
+ * CreateIndxRef.
+ */
+class CreateIndxRef {
+ friend bool printCREATE_INDX_REF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = CreateIndxReq::SignalLength + 3 );
+ enum ErrorCode {
+ NoError = 0,
+ Busy = 701,
+ NotMaster = 702,
+ TriggerNotFound = 4238,
+ TriggerExists = 4239,
+ IndexNameTooLong = 4241,
+ TooManyIndexes = 4242,
+ IndexExists = 4244,
+ AttributeNotStored = 4245,
+ AttributeNullable = 4246,
+ BadRequestType = 4247,
+ InvalidName = 4248,
+ InvalidPrimaryTable = 4249,
+ InvalidIndexType = 4250,
+ NotUnique = 4251,
+ AllocationError = 4252,
+ CreateIndexTableFailed = 4253,
+ InvalidAttributeOrder = 4255
+ };
+
+private:
+ CreateIndxConf m_conf;
+ //Uint32 m_userRef;
+ //Uint32 m_connectionPtr;
+ //Uint32 m_requestInfo;
+ //Uint32 m_tableId;
+ //Uint32 m_indexType;
+ //Uint32 m_indexId;
+ //Uint32 m_indexVersion;
+ Uint32 m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+
+public:
+ CreateIndxConf* getConf() {
+ return &m_conf;
+ }
+ const CreateIndxConf* getConf() const {
+ return &m_conf;
+ }
+ Uint32 getUserRef() const {
+ return m_conf.getUserRef();
+ }
+ void setUserRef(Uint32 val) {
+ m_conf.setUserRef(val);
+ }
+ Uint32 getConnectionPtr() const {
+ return m_conf.getConnectionPtr();
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_conf.setConnectionPtr(val);
+ }
+ CreateIndxReq::RequestType getRequestType() const {
+ return m_conf.getRequestType();
+ }
+ void setRequestType(CreateIndxReq::RequestType val) {
+ m_conf.setRequestType(val);
+ }
+ Uint32 getTableId() const {
+ return m_conf.getTableId();
+ }
+ void setTableId(Uint32 val) {
+ m_conf.setTableId(val);
+ }
+ DictTabInfo::TableType getIndexType() const {
+ return m_conf.getIndexType();
+ }
+ void setIndexType(DictTabInfo::TableType val) {
+ m_conf.setIndexType(val);
+ }
+ Uint32 getIndexId() const {
+ return m_conf.getIndexId();
+ }
+ void setIndexId(Uint32 val) {
+ m_conf.setIndexId(val);
+ }
+ Uint32 getIndexVersion() const {
+ return m_conf.getIndexVersion();
+ }
+ void setIndexVersion(Uint32 val) {
+ m_conf.setIndexVersion(val);
+ }
+ CreateIndxRef::ErrorCode getErrorCode() const {
+ return (CreateIndxRef::ErrorCode)m_errorCode;
+ }
+ void setErrorCode(CreateIndxRef::ErrorCode val) {
+ m_errorCode = (Uint32)val;
+ }
+ Uint32 getErrorLine() const {
+ return m_errorLine;
+ }
+ void setErrorLine(Uint32 val) {
+ m_errorLine = val;
+ }
+ Uint32 getErrorNode() const {
+ return m_errorNode;
+ }
+ void setErrorNode(Uint32 val) {
+ m_errorNode = val;
+ }
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CreateTab.hpp b/ndb/include/kernel/signaldata/CreateTab.hpp
new file mode 100644
index 00000000000..b2ef52a6bf7
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CreateTab.hpp
@@ -0,0 +1,108 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CREATE_TAB_HPP
+#define CREATE_TAB_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * CreateTab
+ *
+ * Implemenatation of CreateTable
+ */
+class CreateTabReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * For printing
+ */
+ friend bool printCREATE_TAB_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 8 );
+
+ enum RequestType {
+ CreateTablePrepare = 0, // Prepare create table
+ CreateTableCommit = 1, // Commit create table
+ CreateTableDrop = 2 // Prepare failed, drop instead
+ };
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 clientRef;
+ Uint32 clientData;
+
+ Uint32 tableId;
+ Uint32 tableVersion;
+ Uint32 gci;
+ Uint32 requestType;
+
+ SECTION( DICT_TAB_INFO = 0 );
+ SECTION( FRAGMENTATION = 1 );
+};
+
+struct CreateTabRef {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+ friend class SafeCounter;
+
+ /**
+ * For printing
+ */
+ friend bool printCREATE_TAB_REF(FILE *, const Uint32 *, Uint32, Uint16);
+
+ STATIC_CONST( SignalLength = 6 );
+ STATIC_CONST( GSN = GSN_CREATE_TAB_REF );
+
+ enum ErrorCode {
+ NF_FakeErrorREF = 255
+ };
+
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 errorCode;
+ Uint32 errorLine;
+ Uint32 errorKey;
+ Uint32 errorStatus;
+};
+
+class CreateTabConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * For printing
+ */
+ friend bool printCREATE_TAB_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CreateTable.hpp b/ndb/include/kernel/signaldata/CreateTable.hpp
new file mode 100644
index 00000000000..424367f28d5
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CreateTable.hpp
@@ -0,0 +1,140 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CREATE_TABLE_HPP
+#define CREATE_TABLE_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * CreateTable
+ *
+ * This signal is sent by API to DICT/TRIX
+ * as a request to create a secondary index
+ * and then from TRIX to TRIX(n) and TRIX to TC.
+ */
+class CreateTableReq {
+ /**
+ * Sender(s)
+ */
+ // API
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class NdbDictInterface;
+ friend class Dbdict;
+ friend class Ndbcntr;
+
+ /**
+ * For printing
+ */
+ friend bool printCREATE_TABLE_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+
+ SECTION( DICT_TAB_INFO = 0 );
+};
+
+class CreateTableRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Ndbcntr;
+ friend class NdbDictInterface;
+
+ /**
+ * For printing
+ */
+ friend bool printCREATE_TABLE_REF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 7 );
+
+ enum ErrorCode {
+ NoError = 0,
+ Busy = 701,
+ NotMaster = 702,
+ InvalidFormat = 703,
+ AttributeNameTooLong = 704,
+ TableNameTooLong = 705,
+ Inconsistency = 706,
+ NoMoreTableRecords = 707,
+ NoMoreAttributeRecords = 708,
+ AttributeNameTwice = 720,
+ TableAlreadyExist = 721,
+ ArraySizeTooBig = 737,
+ RecordTooBig = 738,
+ InvalidPrimaryKeySize = 739,
+ NullablePrimaryKey = 740
+ };
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 masterNodeId;
+ Uint32 errorCode;
+ Uint32 errorLine;
+ Uint32 errorKey;
+ Uint32 status;
+
+public:
+ Uint32 getErrorCode() const {
+ return errorCode;
+ }
+ Uint32 getErrorLine() const {
+ return errorLine;
+ }
+};
+
+class CreateTableConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Ndbcntr;
+ friend class NdbDictInterface;
+
+ /**
+ * For printing
+ */
+ friend bool printCREATE_TABLE_REF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 tableId;
+ Uint32 tableVersion;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/CreateTrig.hpp b/ndb/include/kernel/signaldata/CreateTrig.hpp
new file mode 100644
index 00000000000..a8de9e50dd4
--- /dev/null
+++ b/ndb/include/kernel/signaldata/CreateTrig.hpp
@@ -0,0 +1,414 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CREATE_TRIG_HPP
+#define CREATE_TRIG_HPP
+
+#include "SignalData.hpp"
+#include <Bitmask.hpp>
+#include <trigger_definitions.h>
+#include <AttributeList.hpp>
+
+/**
+ * CreateTrigReq.
+ */
+class CreateTrigReq {
+ friend bool printCREATE_TRIG_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum RequestType {
+ RT_UNDEFINED = 0,
+ RT_USER = 1,
+ RT_ALTER_INDEX = 2,
+ RT_BUILD_INDEX = 3,
+ RT_DICT_PREPARE = 1 << 4,
+ RT_DICT_CREATE = 2 << 4,
+ RT_DICT_COMMIT = 0xC << 4,
+ RT_DICT_ABORT = 0xF << 4,
+ RT_TC = 5 << 8,
+ RT_LQH = 6 << 8
+ };
+ STATIC_CONST( SignalLength = 9 + MAXNROFATTRIBUTESINWORDS);
+ SECTION( TRIGGER_NAME_SECTION = 0 );
+ SECTION( ATTRIBUTE_MASK_SECTION = 1 ); // not yet in use
+ enum KeyValues {
+ TriggerNameKey = 0xa1
+ };
+
+private:
+ Uint32 m_userRef;
+ Uint32 m_connectionPtr;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexId; // only for index trigger
+ Uint32 m_triggerId; // only set by DICT
+ Uint32 m_triggerInfo; // flags | event | timing | type
+ Uint32 m_online; // alter online (not normally for subscription)
+ Uint32 m_receiverRef; // receiver for subscription trigger
+ AttributeMask m_attributeMask;
+ // extra
+ Uint32 m_opKey;
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ CreateTrigReq::RequestType getRequestType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_requestInfo, 0, 16);
+ return (CreateTrigReq::RequestType)val;
+ }
+ void setRequestType(CreateTrigReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getRequestFlag() const {
+ return BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ };
+ void addRequestFlag(Uint32 val) {
+ val |= BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ BitmaskImpl::setField(1, &m_requestInfo, 16, 16, val);
+ };
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getTriggerId() const {
+ return m_triggerId;
+ }
+ void setTriggerId(Uint32 val) {
+ m_triggerId = val;
+ }
+ Uint32 getTriggerInfo() const {
+ return m_triggerInfo;
+ }
+ void setTriggerInfo(Uint32 val) {
+ m_triggerInfo = val;
+ }
+ TriggerType::Value getTriggerType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 0, 8);
+ return (TriggerType::Value)val;
+ }
+ void setTriggerType(TriggerType::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 0, 8, (Uint32)val);
+ }
+ TriggerActionTime::Value getTriggerActionTime() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 8, 8);
+ return (TriggerActionTime::Value)val;
+ }
+ void setTriggerActionTime(TriggerActionTime::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 8, 8, (Uint32)val);
+ }
+ TriggerEvent::Value getTriggerEvent() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 16, 8);
+ return (TriggerEvent::Value)val;
+ }
+ void setTriggerEvent(TriggerEvent::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 16, 8, (Uint32)val);
+ }
+ bool getMonitorReplicas() const {
+ return BitmaskImpl::getField(1, &m_triggerInfo, 24, 1);
+ }
+ void setMonitorReplicas(bool val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 24, 1, val);
+ }
+ bool getMonitorAllAttributes() const {
+ return BitmaskImpl::getField(1, &m_triggerInfo, 25, 1);
+ }
+ void setMonitorAllAttributes(bool val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 25, 1, val);
+ }
+ Uint32 getOnline() const {
+ return m_online;
+ }
+ void setOnline(Uint32 val) {
+ m_online = val;
+ }
+ Uint32 getReceiverRef() const {
+ return m_receiverRef;
+ }
+ void setReceiverRef(Uint32 val) {
+ m_receiverRef = val;
+ }
+ AttributeMask& getAttributeMask() {
+ return m_attributeMask;
+ }
+ const AttributeMask& getAttributeMask() const {
+ return m_attributeMask;
+ }
+ void clearAttributeMask() {
+ m_attributeMask.clear();
+ }
+ void setAttributeMask(const AttributeMask& val) {
+ m_attributeMask = val;
+ }
+ void setAttributeMask(Uint16 val) {
+ m_attributeMask.set(val);
+ }
+ Uint32 getOpKey() const {
+ return m_opKey;
+ }
+ void setOpKey(Uint32 val) {
+ m_opKey = val;
+ }
+};
+
+/**
+ * CreateTrigConf.
+ */
+class CreateTrigConf {
+ friend bool printCREATE_TRIG_CONF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( InternalLength = 3 );
+ STATIC_CONST( SignalLength = 7 );
+
+private:
+ Uint32 m_userRef;
+ Uint32 m_connectionPtr;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexId;
+ Uint32 m_triggerId;
+ Uint32 m_triggerInfo; // BACKUP wants this
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ CreateTrigReq::RequestType getRequestType() const {
+ return (CreateTrigReq::RequestType)m_requestInfo;
+ }
+ void setRequestType(CreateTrigReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getTriggerId() const {
+ return m_triggerId;
+ }
+ void setTriggerId(Uint32 val) {
+ m_triggerId = val;
+ }
+ Uint32 getTriggerInfo() const {
+ return m_triggerInfo;
+ }
+ void setTriggerInfo(Uint32 val) {
+ m_triggerInfo = val;
+ }
+ TriggerType::Value getTriggerType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 0, 8);
+ return (TriggerType::Value)val;
+ }
+ void setTriggerType(TriggerType::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 0, 8, (Uint32)val);
+ }
+ TriggerActionTime::Value getTriggerActionTime() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 8, 8);
+ return (TriggerActionTime::Value)val;
+ }
+ void setTriggerActionTime(TriggerActionTime::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 8, 8, (Uint32)val);
+ }
+ TriggerEvent::Value getTriggerEvent() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 16, 8);
+ return (TriggerEvent::Value)val;
+ }
+ void setTriggerEvent(TriggerEvent::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 16, 8, (Uint32)val);
+ }
+ bool getMonitorReplicas() const {
+ return BitmaskImpl::getField(1, &m_triggerInfo, 24, 1);
+ }
+ void setMonitorReplicas(bool val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 24, 1, val);
+ }
+ bool getMonitorAllAttributes() const {
+ return BitmaskImpl::getField(1, &m_triggerInfo, 25, 1);
+ }
+ void setMonitorAllAttributes(bool val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 25, 1, val);
+ }
+};
+
+/**
+ * CreateTrigRef.
+ */
+class CreateTrigRef {
+ friend bool printCREATE_TRIG_REF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum ErrorCode {
+ NoError = 0,
+ Busy = 701,
+ TriggerNameTooLong = 4236,
+ TooManyTriggers = 4237,
+ TriggerNotFound = 4238,
+ TriggerExists = 4239,
+ UnsupportedTriggerType = 4240,
+ BadRequestType = 4247,
+ InvalidName = 4248,
+ InvalidTable = 4249
+ };
+ STATIC_CONST( SignalLength = CreateTrigConf::SignalLength + 3 );
+
+private:
+ CreateTrigConf m_conf;
+ //Uint32 m_userRef;
+ //Uint32 m_connectionPtr;
+ //Uint32 m_requestInfo;
+ //Uint32 m_tableId;
+ //Uint32 m_indexId;
+ //Uint32 m_triggerId;
+ //Uint32 m_triggerInfo;
+ Uint32 m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+
+public:
+ CreateTrigConf* getConf() {
+ return &m_conf;
+ }
+ const CreateTrigConf* getConf() const {
+ return &m_conf;
+ }
+ Uint32 getUserRef() const {
+ return m_conf.getUserRef();
+ }
+ void setUserRef(Uint32 val) {
+ m_conf.setUserRef(val);
+ }
+ Uint32 getConnectionPtr() const {
+ return m_conf.getConnectionPtr();
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_conf.setConnectionPtr(val);
+ }
+ CreateTrigReq::RequestType getRequestType() const {
+ return m_conf.getRequestType();
+ }
+ void setRequestType(CreateTrigReq::RequestType val) {
+ m_conf.setRequestType(val);
+ }
+ Uint32 getTableId() const {
+ return m_conf.getTableId();
+ }
+ void setTableId(Uint32 val) {
+ m_conf.setTableId(val);
+ }
+ Uint32 getIndexId() const {
+ return m_conf.getIndexId();
+ }
+ void setIndexId(Uint32 val) {
+ m_conf.setIndexId(val);
+ }
+ Uint32 getTriggerId() const {
+ return m_conf.getTriggerId();
+ }
+ void setTriggerId(Uint32 val) {
+ m_conf.setTriggerId(val);
+ }
+ Uint32 getTriggerInfo() const {
+ return m_conf.getTriggerInfo();
+ }
+ void setTriggerInfo(Uint32 val) {
+ m_conf.setTriggerInfo(val);
+ }
+ TriggerType::Value getTriggerType() const {
+ return m_conf.getTriggerType();
+ }
+ void setTriggerType(TriggerType::Value val) {
+ m_conf.setTriggerType(val);
+ }
+ TriggerActionTime::Value getTriggerActionTime() const {
+ return m_conf.getTriggerActionTime();
+ }
+ void setTriggerActionTime(TriggerActionTime::Value val) {
+ m_conf.setTriggerActionTime(val);
+ }
+ TriggerEvent::Value getTriggerEvent() const {
+ return m_conf.getTriggerEvent();
+ }
+ void setTriggerEvent(TriggerEvent::Value val) {
+ m_conf.setTriggerEvent(val);
+ }
+ bool getMonitorReplicas() const {
+ return m_conf.getMonitorReplicas();
+ }
+ void setMonitorReplicas(bool val) {
+ m_conf.setMonitorReplicas(val);
+ }
+ bool getMonitorAllAttributes() const {
+ return m_conf.getMonitorAllAttributes();
+ }
+ void setMonitorAllAttributes(bool val) {
+ m_conf.setMonitorAllAttributes(val);
+ }
+ CreateTrigRef::ErrorCode getErrorCode() const {
+ return (CreateTrigRef::ErrorCode)m_errorCode;
+ }
+ void setErrorCode(CreateTrigRef::ErrorCode val) {
+ m_errorCode = (Uint32)val;
+ }
+ Uint32 getErrorLine() const {
+ return m_errorLine;
+ }
+ void setErrorLine(Uint32 val) {
+ m_errorLine = val;
+ }
+ Uint32 getErrorNode() const {
+ return m_errorNode;
+ }
+ void setErrorNode(Uint32 val) {
+ m_errorNode = val;
+ }
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DiAddTab.hpp b/ndb/include/kernel/signaldata/DiAddTab.hpp
new file mode 100644
index 00000000000..6b17515eb6f
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DiAddTab.hpp
@@ -0,0 +1,90 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DIADDTABREQ_HPP
+#define DIADDTABREQ_HPP
+
+#include "SignalData.hpp"
+
+class DiAddTabReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( SignalLength = 9 );
+ SECTION( FRAGMENTATION = 0 );
+
+private:
+ Uint32 connectPtr;
+ Uint32 tableId;
+ Uint32 fragType;
+ Uint32 kValue;
+ Uint32 noOfReplicas; //Currently not used
+ Uint32 storedTable;
+ Uint32 tableType;
+ Uint32 schemaVersion;
+ Uint32 primaryTableId;
+};
+
+class DiAddTabRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ union {
+ Uint32 connectPtr;
+ Uint32 senderData;
+ };
+ Uint32 errorCode;
+};
+
+class DiAddTabConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ union {
+ Uint32 connectPtr;
+ Uint32 senderData;
+ };
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DiGetNodes.hpp b/ndb/include/kernel/signaldata/DiGetNodes.hpp
new file mode 100644
index 00000000000..05ab6bfebb3
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DiGetNodes.hpp
@@ -0,0 +1,62 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DIGETNODES_HPP
+#define DIGETNODES_HPP
+
+#include <NodeBitmask.hpp>
+#include <ndb_limits.h>
+
+/**
+ *
+ */
+class DiGetNodesConf {
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtc;
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( SignalLength = 3 + MAX_REPLICAS );
+private:
+ Uint32 zero;
+ Uint32 fragId;
+ Uint32 reqinfo;
+ Uint32 nodes[MAX_REPLICAS];
+};
+/**
+ *
+ */
+class DiGetNodesReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( SignalLength = 3 );
+private:
+ Uint32 notUsed;
+ Uint32 tableId;
+ Uint32 hashValue;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/DictSchemaInfo.hpp b/ndb/include/kernel/signaldata/DictSchemaInfo.hpp
new file mode 100644
index 00000000000..d7f82abc299
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DictSchemaInfo.hpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DICT_SCHEMA_INFO_HPP
+#define DICT_SCHEMA_INFO_HPP
+
+#include "SignalData.hpp"
+
+class DictSchemaInfo {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+
+public:
+ static const unsigned HeaderLength = 3;
+ static const unsigned DataLength = 22;
+
+private:
+ Uint32 senderRef;
+ Uint32 offset;
+ Uint32 totalLen;
+
+ /**
+ * Length in this = signal->length() - 3
+ * Sender block ref = signal->senderBlockRef()
+ */
+
+ Uint32 schemaInfoData[22];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DictSizeAltReq.hpp b/ndb/include/kernel/signaldata/DictSizeAltReq.hpp
new file mode 100644
index 00000000000..b40f0c8c1af
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DictSizeAltReq.hpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DICT_SIZE_ALT_REQ_H
+#define DICT_SIZE_ALT_REQ_H
+
+
+
+#include "SignalData.hpp"
+
+class DictSizeAltReq {
+ /**
+ * Sender(s)
+ */
+ friend class ClusterConfiguration;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdict;
+private:
+ /**
+ * Indexes in theData
+ */
+ STATIC_CONST( IND_BLOCK_REF = 0 );
+ STATIC_CONST( IND_ATTRIBUTE = 1 );
+ STATIC_CONST( IND_CONNECT = 2 );
+ STATIC_CONST( IND_FRAG_CONNECT = 3 );
+ STATIC_CONST( IND_TABLE = 4 );
+ STATIC_CONST( IND_TC_CONNECT = 5 );
+
+ /**
+ * Use the index definitions to use the signal data
+ */
+ UintR theData[6];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DictStart.hpp b/ndb/include/kernel/signaldata/DictStart.hpp
new file mode 100644
index 00000000000..59310601f48
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DictStart.hpp
@@ -0,0 +1,54 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DICT_START_HPP
+#define DICT_START_HPP
+
+class DictStartReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+
+ Uint32 restartGci;
+ Uint32 senderRef;
+};
+
+class DictStartConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+
+public:
+private:
+
+ Uint32 startingNodeId;
+ Uint32 startWord;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/DictTabInfo.hpp b/ndb/include/kernel/signaldata/DictTabInfo.hpp
new file mode 100644
index 00000000000..791388d5df8
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DictTabInfo.hpp
@@ -0,0 +1,483 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DICT_TAB_INFO_HPP
+#define DICT_TAB_INFO_HPP
+
+#include "SignalData.hpp"
+#include <AttributeDescriptor.hpp>
+#include <SimpleProperties.hpp>
+#include <ndb_limits.h>
+#include <trigger_definitions.h>
+#include <NdbSqlUtil.hpp>
+
+#define DTIMAP(x, y, z) \
+ { DictTabInfo::y, offsetof(x, z), SimpleProperties::Uint32Value, 0, (~0), 0 }
+
+#define DTIMAP2(x, y, z, u, v) \
+ { DictTabInfo::y, offsetof(x, z), SimpleProperties::Uint32Value, u, v, 0 }
+
+#define DTIMAPS(x, y, z, u, v) \
+ { DictTabInfo::y, offsetof(x, z), SimpleProperties::StringValue, u, v, 0 }
+
+#define DTIMAPB(x, y, z, u, v, l) \
+ { DictTabInfo::y, offsetof(x, z), SimpleProperties::BinaryValue, u, v, \
+ offsetof(x, l) }
+
+#define DTIBREAK(x) \
+ { DictTabInfo::x, 0, SimpleProperties::InvalidValue, 0, 0 }
+
+class DictTabInfo {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ // Blocks
+ friend class Backup;
+ friend class Dbdict;
+ friend class Ndbcntr;
+ friend class Trix;
+ friend class DbUtil;
+ // API
+ friend class Table;
+ friend class NdbSchemaOp;
+
+ /**
+ * For printing
+ */
+ friend bool printDICTTABINFO(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo);
+
+public:
+ enum RequestType {
+ CreateTableFromAPI = 1,
+ AddTableFromDict = 2, // Between DICT's
+ CopyTable = 3, // Between DICT's
+ ReadTableFromDiskSR = 4, // Local in DICT
+ GetTabInfoConf = 5,
+ AlterTableFromAPI = 6
+ };
+
+ enum KeyValues {
+ TableName = 1, // String, Mandatory
+ TableId = 2, //Mandatory between DICT's otherwise not allowed
+ TableVersion = 3, //Mandatory between DICT's otherwise not allowed
+ TableLoggedFlag = 4, //Default Logged
+ NoOfKeyAttr = 5, //Default 1
+ NoOfAttributes = 6, //Mandatory
+ NoOfNullable = 7, //Deafult 0
+ NoOfVariable = 8, //Default 0
+ TableKValue = 9, //Default 6
+ MinLoadFactor = 10, //Default 70
+ MaxLoadFactor = 11, //Default 80
+ KeyLength = 12, //Default 1 (No of words in primary key)
+ FragmentTypeVal = 13, //Default AllNodesSmallTable
+ TableStorageVal = 14, //Default StorageType::MainMemory
+ ScanOptimised = 15, //Default updateOptimised
+ FragmentKeyTypeVal = 16, //Default PrimaryKey
+ SecondTableId = 17, //Mandatory between DICT's otherwise not allowed
+ TableTypeVal = 18, //Default TableType::UserTable
+ PrimaryTable = 19, //Mandatory for index otherwise RNIL
+ PrimaryTableId = 20, //ditto
+ IndexState = 21,
+ InsertTriggerId = 22,
+ UpdateTriggerId = 23,
+ DeleteTriggerId = 24,
+ CustomTriggerId = 25,
+ FrmLen = 26,
+ FrmData = 27,
+ TableEnd = 999,
+
+ AttributeName = 1000, // String, Mandatory
+ AttributeId = 1001, //Mandatory between DICT's otherwise not allowed
+ AttributeType = 1002, //Default UnSignedType
+ AttributeSize = 1003, //Default DictTabInfo::a32Bit
+ AttributeArraySize = 1005, //Default 1
+ AttributeKeyFlag = 1006, //Default noKey
+ AttributeStorage = 1007, //Default MainMemory
+ AttributeNullableFlag = 1008, //Default NotNullable
+ AttributeDGroup = 1009, //Default NotDGroup
+ AttributeDKey = 1010, //Default NotDKey
+ AttributeStoredInd = 1011, //Default NotStored
+ AttributeGroup = 1012, //Default 0
+ AttributeExtType = 1013, //Default 0 (undefined)
+ AttributeExtPrecision = 1014, //Default 0
+ AttributeExtScale = 1015, //Default 0
+ AttributeExtLength = 1016, //Default 0
+ AttributeAutoIncrement = 1017, //Default false
+ AttributeDefaultValue = 1018, //Default value (printable string)
+ AttributeEnd = 1999 //
+ };
+ // ----------------------------------------------------------------------
+ // Part of the protocol is that we only transfer parameters which do not
+ // have a default value. Thus the default values are part of the protocol.
+ // ----------------------------------------------------------------------
+
+ // FragmentKeyType constants
+ enum FragmentKeyType {
+ PrimaryKey = 0,
+ DistributionKey = 1,
+ DistributionGroup = 2
+ };
+
+ // FragmentType constants
+ enum FragmentType {
+ AllNodesSmallTable = 0,
+ AllNodesMediumTable = 1,
+ AllNodesLargeTable = 2,
+ SingleFragment = 3
+ };
+
+ // TableStorage AND AttributeStorage constants
+ enum StorageType {
+ MainMemory = 0,
+ DiskMemory = 1
+ };
+
+ // TableType constants + objects
+ enum TableType {
+ UndefTableType = 0,
+ SystemTable = 1,
+ UserTable = 2,
+ UniqueHashIndex = 3,
+ HashIndex = 4,
+ UniqueOrderedIndex = 5,
+ OrderedIndex = 6,
+ // constant 10 hardcoded in Dbdict.cpp
+ HashIndexTrigger = 10 + TriggerType::SECONDARY_INDEX,
+ SubscriptionTrigger = 10 + TriggerType::SUBSCRIPTION,
+ ReadOnlyConstraint = 10 + TriggerType::READ_ONLY_CONSTRAINT,
+ IndexTrigger = 10 + TriggerType::ORDERED_INDEX
+ };
+ static inline bool
+ isTable(int tableType) {
+ return
+ tableType == SystemTable ||
+ tableType == UserTable;
+ }
+ static inline bool
+ isIndex(int tableType) {
+ return
+ tableType == UniqueHashIndex ||
+ tableType == HashIndex ||
+ tableType == UniqueOrderedIndex ||
+ tableType == OrderedIndex;
+ }
+ static inline bool
+ isUniqueIndex(int tableType) {
+ return
+ tableType == UniqueHashIndex ||
+ tableType == UniqueOrderedIndex;
+ }
+ static inline bool
+ isNonUniqueIndex(int tableType) {
+ return
+ tableType == HashIndex ||
+ tableType == OrderedIndex;
+ }
+ static inline bool
+ isHashIndex(int tableType) {
+ return
+ tableType == UniqueHashIndex ||
+ tableType == HashIndex;
+ }
+ static inline bool
+ isOrderedIndex(int tableType) {
+ return
+ tableType == UniqueOrderedIndex ||
+ tableType == OrderedIndex;
+ }
+
+ // Object state for translating from/to API
+ enum ObjectState {
+ StateUndefined = 0,
+ StateOffline = 1,
+ StateBuilding = 2,
+ StateDropping = 3,
+ StateOnline = 4,
+ StateBroken = 9
+ };
+
+ // Object store for translating from/to API
+ enum ObjectStore {
+ StoreUndefined = 0,
+ StoreTemporary = 1,
+ StorePermanent = 2
+ };
+
+ // ScanOptimised constants
+ static const unsigned updateOptimised = 0;
+ static const unsigned scanOptimised = 1;
+
+ // AttributeType constants
+ static const unsigned SignedType = 0;
+ static const unsigned UnSignedType = 1;
+ static const unsigned FloatingPointType = 2;
+ static const unsigned StringType = 3;
+
+ // AttributeSize constants
+ static const unsigned an8Bit = 3;
+ static const unsigned a16Bit = 4;
+ static const unsigned a32Bit = 5;
+ static const unsigned a64Bit = 6;
+ static const unsigned a128Bit = 7;
+
+ // AttributeDGroup constants
+ static const unsigned NotDGroup = 0;
+ static const unsigned DGroup = 1;
+
+ // AttributeDKey constants
+ static const unsigned NotDKey = 0;
+ static const unsigned DKey = 1;
+
+ // AttributeStoredInd constants
+ static const unsigned NotStored = 0;
+ static const unsigned Stored = 1;
+
+ // Table data interpretation
+ struct Table {
+ char TableName[MAX_TAB_NAME_SIZE];
+ Uint32 TableId;
+ Uint32 SecondTableId;
+ char PrimaryTable[MAX_TAB_NAME_SIZE]; // Only used when "index"
+ Uint32 PrimaryTableId;
+ Uint32 TableLoggedFlag;
+ Uint32 NoOfKeyAttr;
+ Uint32 NoOfAttributes;
+ Uint32 NoOfNullable;
+ Uint32 NoOfVariable;
+ Uint32 TableKValue;
+ Uint32 MinLoadFactor;
+ Uint32 MaxLoadFactor;
+ Uint32 KeyLength;
+ Uint32 FragmentType;
+ Uint32 TableStorage;
+ Uint32 ScanOptimised;
+ Uint32 FragmentKeyType;
+ Uint32 TableType;
+ Uint32 TableVersion;
+ Uint32 IndexState;
+ Uint32 InsertTriggerId;
+ Uint32 UpdateTriggerId;
+ Uint32 DeleteTriggerId;
+ Uint32 CustomTriggerId;
+ Uint32 FrmLen;
+ char FrmData[MAX_FRM_DATA_SIZE];
+
+ void init();
+ };
+
+ static const
+ SimpleProperties::SP2StructMapping TableMapping[];
+
+ static const Uint32 TableMappingSize;
+
+ // AttributeExtType values
+ enum ExtType {
+ ExtUndefined = NdbSqlUtil::Type::Undefined,
+ ExtTinyint = NdbSqlUtil::Type::Tinyint,
+ ExtTinyunsigned = NdbSqlUtil::Type::Tinyunsigned,
+ ExtSmallint = NdbSqlUtil::Type::Smallint,
+ ExtSmallunsigned = NdbSqlUtil::Type::Smallunsigned,
+ ExtMediumint = NdbSqlUtil::Type::Mediumint,
+ ExtMediumunsigned = NdbSqlUtil::Type::Mediumunsigned,
+ ExtInt = NdbSqlUtil::Type::Int,
+ ExtUnsigned = NdbSqlUtil::Type::Unsigned,
+ ExtBigint = NdbSqlUtil::Type::Bigint,
+ ExtBigunsigned = NdbSqlUtil::Type::Bigunsigned,
+ ExtFloat = NdbSqlUtil::Type::Float,
+ ExtDouble = NdbSqlUtil::Type::Double,
+ ExtDecimal = NdbSqlUtil::Type::Decimal,
+ ExtChar = NdbSqlUtil::Type::Char,
+ ExtVarchar = NdbSqlUtil::Type::Varchar,
+ ExtBinary = NdbSqlUtil::Type::Binary,
+ ExtVarbinary = NdbSqlUtil::Type::Varbinary,
+ ExtDatetime = NdbSqlUtil::Type::Datetime,
+ ExtTimespec = NdbSqlUtil::Type::Timespec
+ };
+
+ // Attribute data interpretation
+ struct Attribute {
+ char AttributeName[MAX_TAB_NAME_SIZE];
+ Uint32 AttributeId;
+ Uint32 AttributeType;
+ Uint32 AttributeSize;
+ Uint32 AttributeArraySize;
+ Uint32 AttributeKeyFlag;
+ Uint32 AttributeStorage;
+ Uint32 AttributeNullableFlag;
+ Uint32 AttributeDGroup;
+ Uint32 AttributeDKey;
+ Uint32 AttributeStoredInd;
+ Uint32 AttributeGroup;
+ Uint32 AttributeExtType;
+ Uint32 AttributeExtPrecision;
+ Uint32 AttributeExtScale;
+ Uint32 AttributeExtLength;
+ Uint32 AttributeAutoIncrement;
+ char AttributeDefaultValue[MAX_ATTR_DEFAULT_VALUE_SIZE];
+
+ void init();
+
+ inline
+ Uint32 sizeInWords()
+ {
+ return ((1 << AttributeSize) * AttributeArraySize + 31) >> 5;
+ }
+
+ // translate to old kernel types and sizes
+ inline bool
+ translateExtType() {
+ switch (AttributeExtType) {
+ case DictTabInfo::ExtUndefined:
+ break;
+ case DictTabInfo::ExtTinyint:
+ AttributeType = DictTabInfo::SignedType;
+ AttributeSize = DictTabInfo::an8Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtTinyunsigned:
+ AttributeType = DictTabInfo::UnSignedType;
+ AttributeSize = DictTabInfo::an8Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtSmallint:
+ AttributeType = DictTabInfo::SignedType;
+ AttributeSize = DictTabInfo::a16Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtSmallunsigned:
+ AttributeType = DictTabInfo::UnSignedType;
+ AttributeSize = DictTabInfo::a16Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtMediumint:
+ AttributeType = DictTabInfo::SignedType;
+ AttributeSize = DictTabInfo::an8Bit;
+ AttributeArraySize = 3 * AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtMediumunsigned:
+ AttributeType = DictTabInfo::UnSignedType;
+ AttributeSize = DictTabInfo::an8Bit;
+ AttributeArraySize = 3 * AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtInt:
+ AttributeType = DictTabInfo::SignedType;
+ AttributeSize = DictTabInfo::a32Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtUnsigned:
+ AttributeType = DictTabInfo::UnSignedType;
+ AttributeSize = DictTabInfo::a32Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtBigint:
+ AttributeType = DictTabInfo::SignedType;
+ AttributeSize = DictTabInfo::a64Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtBigunsigned:
+ AttributeType = DictTabInfo::UnSignedType;
+ AttributeSize = DictTabInfo::a64Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtFloat:
+ AttributeType = DictTabInfo::FloatingPointType;
+ AttributeSize = DictTabInfo::a32Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtDouble:
+ AttributeType = DictTabInfo::FloatingPointType;
+ AttributeSize = DictTabInfo::a64Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtDecimal:
+ // not yet implemented anywhere
+ break;
+ case DictTabInfo::ExtChar:
+ case DictTabInfo::ExtBinary:
+ AttributeType = DictTabInfo::StringType;
+ AttributeSize = DictTabInfo::an8Bit;
+ AttributeArraySize = AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtVarchar:
+ case DictTabInfo::ExtVarbinary:
+ AttributeType = DictTabInfo::StringType;
+ AttributeSize = DictTabInfo::an8Bit;
+ AttributeArraySize = AttributeExtLength + 2;
+ return true;
+ case DictTabInfo::ExtDatetime:
+ AttributeType = DictTabInfo::StringType;
+ AttributeSize = DictTabInfo::an8Bit;
+ AttributeArraySize = 8 * AttributeExtLength;
+ return true;
+ case DictTabInfo::ExtTimespec:
+ AttributeType = DictTabInfo::StringType;
+ AttributeSize = DictTabInfo::an8Bit;
+ AttributeArraySize = 12 * AttributeExtLength;
+ return true;
+ };
+ return false;
+ }
+
+ inline void print(FILE *out) {
+ fprintf(out, "AttributeId = %d\n", AttributeId);
+ fprintf(out, "AttributeType = %d\n", AttributeType);
+ fprintf(out, "AttributeSize = %d\n", AttributeSize);
+ fprintf(out, "AttributeArraySize = %d\n", AttributeArraySize);
+ fprintf(out, "AttributeKeyFlag = %d\n", AttributeKeyFlag);
+ fprintf(out, "AttributeStorage = %d\n", AttributeStorage);
+ fprintf(out, "AttributeNullableFlag = %d\n", AttributeNullableFlag);
+ fprintf(out, "AttributeDGroup = %d\n", AttributeDGroup);
+ fprintf(out, "AttributeDKey = %d\n", AttributeDKey);
+ fprintf(out, "AttributeStoredInd = %d\n", AttributeStoredInd);
+ fprintf(out, "AttributeGroup = %d\n", AttributeGroup);
+ fprintf(out, "AttributeAutoIncrement = %d\n", AttributeAutoIncrement);
+ fprintf(out, "AttributeExtType = %d\n", AttributeExtType);
+ fprintf(out, "AttributeExtPrecision = %d\n", AttributeExtPrecision);
+ fprintf(out, "AttributeExtScale = %d\n", AttributeExtScale);
+ fprintf(out, "AttributeExtLength = %d\n", AttributeExtLength);
+ fprintf(out, "AttributeDefaultValue = \"%s\"\n",
+ AttributeDefaultValue ? AttributeDefaultValue : "");
+ }
+ };
+
+ static const
+ SimpleProperties::SP2StructMapping AttributeMapping[];
+
+ static const Uint32 AttributeMappingSize;
+
+ // Signal constants
+ STATIC_CONST( DataLength = 20 );
+ STATIC_CONST( HeaderLength = 5 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 requestType;
+ Uint32 totalLen;
+ Uint32 offset;
+
+ /**
+ * Length of this data = signal->length() - HeaderLength
+ * Sender block ref = signal->senderBlockRef()
+ */
+
+ Uint32 tabInfoData[DataLength];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DihAddFrag.hpp b/ndb/include/kernel/signaldata/DihAddFrag.hpp
new file mode 100644
index 00000000000..6e5a24ee413
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DihAddFrag.hpp
@@ -0,0 +1,62 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DIHADDFRAG_HPP
+#define DIHADDFRAG_HPP
+
+#include <NodeBitmask.hpp>
+#include <ndb_limits.h>
+
+/**
+ *
+ */
+class DihAddFragConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+ Uint32 senderNodeId;
+ Uint32 tableId;
+};
+/**
+ *
+ */
+class DihAddFragReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 10 + MAX_REPLICAS );
+private:
+ Uint32 masterRef;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 kValue;
+ Uint32 method;
+ Uint32 mask;
+ Uint32 hashPointer;
+ Uint32 noOfFragments;
+ Uint32 noOfBackups;
+ Uint32 storedTable;
+ Uint32 nodes[MAX_REPLICAS];
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/DihContinueB.hpp b/ndb/include/kernel/signaldata/DihContinueB.hpp
new file mode 100644
index 00000000000..e683b55351c
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DihContinueB.hpp
@@ -0,0 +1,75 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DIH_CONTINUEB_H
+#define DIH_CONTINUEB_H
+
+#include "SignalData.hpp"
+
+class DihContinueB {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Dbdih;
+ friend bool printCONTINUEB_DBDIH(FILE * output, const Uint32 * theData, Uint32 len);
+private:
+ enum Type {
+ ZPACK_TABLE_INTO_PAGES = 1,
+ ZPACK_FRAG_INTO_PAGES = 2,
+ ZREAD_PAGES_INTO_TABLE = 3,
+ ZREAD_PAGES_INTO_FRAG = 4,
+ //ZREAD_TAB_DESCRIPTION = 5,
+ ZCOPY_TABLE = 6,
+ ZCOPY_TABLE_NODE = 7,
+ ZSTART_FRAGMENT = 8,
+ ZCOMPLETE_RESTART = 9,
+ ZREAD_TABLE_FROM_PAGES = 10,
+ ZSR_PHASE2_READ_TABLE = 11,
+ ZCHECK_TC_COUNTER = 12,
+ ZCALCULATE_KEEP_GCI = 13,
+ ZSTORE_NEW_LCP_ID = 14,
+ ZTABLE_UPDATE = 15,
+ ZCHECK_LCP_COMPLETED = 16,
+ ZINIT_LCP = 17,
+ ZADD_TABLE_MASTER_PAGES = 19,
+ ZDIH_ADD_TABLE_MASTER = 20,
+ ZADD_TABLE_SLAVE_PAGES = 21,
+ ZDIH_ADD_TABLE_SLAVE = 22,
+ ZSTART_GCP = 23,
+ ZCOPY_GCI = 24,
+ ZEMPTY_VERIFY_QUEUE = 25,
+ ZCHECK_GCP_STOP = 26,
+ ZREMOVE_NODE_FROM_TABLE = 27,
+ ZCOPY_NODE = 28,
+ ZSTART_TAKE_OVER = 29,
+ ZCHECK_START_TAKE_OVER = 30,
+ ZTO_START_COPY_FRAG = 31,
+ ZINITIALISE_RECORDS = 33,
+ ZINVALIDATE_NODE_LCP = 34,
+ ZSTART_PERMREQ_AGAIN = 35,
+ SwitchReplica = 36,
+ ZSEND_START_TO = 37,
+ ZSEND_ADD_FRAG = 38,
+ ZSEND_CREATE_FRAG = 39,
+ ZSEND_UPDATE_TO = 40,
+ ZSEND_END_TO = 41,
+
+ WAIT_DROP_TAB_WRITING_TO_FILE = 42,
+ CHECK_WAIT_DROP_TAB_FAILED_LQH = 43
+ };
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DihSizeAltReq.hpp b/ndb/include/kernel/signaldata/DihSizeAltReq.hpp
new file mode 100644
index 00000000000..73279447859
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DihSizeAltReq.hpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DIH_SIZE_ALT_REQ_H
+#define DIH_SIZE_ALT_REQ_H
+
+#include "SignalData.hpp"
+
+class DihSizeAltReq {
+ /**
+ * Sender(s)
+ */
+ friend class ClusterConfiguration;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdih;
+private:
+ /**
+ * Indexes in theData
+ */
+ STATIC_CONST( IND_BLOCK_REF = 0 );
+ STATIC_CONST( IND_API_CONNECT = 1 );
+ STATIC_CONST( IND_CONNECT = 2 );
+ STATIC_CONST( IND_FRAG_CONNECT = 3 );
+ STATIC_CONST( IND_MORE_NODES = 4 );
+ STATIC_CONST( IND_REPLICAS = 5 );
+ STATIC_CONST( IND_TABLE = 6 );
+
+ /**
+ * Use the index definitions to use the signal data
+ */
+ UintR theData[7];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DihStartTab.hpp b/ndb/include/kernel/signaldata/DihStartTab.hpp
new file mode 100644
index 00000000000..75443e6070e
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DihStartTab.hpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DIH_STARTTAB__HPP
+#define DIH_STARTTAB__HPP
+
+#include "SignalData.hpp"
+
+class DihStartTabReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+public:
+ STATIC_CONST( HeaderLength = 3 );
+
+private:
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 noOfTables;
+
+ struct {
+ Uint32 tableId;
+ Uint32 schemaVersion;
+ } tables[10];
+};
+
+class DihStartTabConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DihSwitchReplica.hpp b/ndb/include/kernel/signaldata/DihSwitchReplica.hpp
new file mode 100644
index 00000000000..d4212f510f3
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DihSwitchReplica.hpp
@@ -0,0 +1,72 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DIH_SWITCH_REPLICA_HPP
+#define DIH_SWITCH_REPLICA_HPP
+
+/**
+ * This signal is sent from master DIH to all DIH's
+ * switches primary / backup nodes for replica(s)
+ *
+ */
+class DihSwitchReplicaReq {
+ /**
+ * Sender/Reciver
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 4 + MAX_REPLICAS );
+
+private:
+ /**
+ * Request Info
+ *
+ */
+ Uint32 senderRef;
+ Uint32 tableId;
+ Uint32 fragNo;
+ Uint32 noOfReplicas;
+ Uint32 newNodeOrder[MAX_REPLICAS];
+};
+
+class DihSwitchReplicaRef {
+ /**
+ * Sender/Reciver
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderNode;
+ Uint32 errorCode; // See StopPermRef::ErrorCode
+};
+
+class DihSwitchReplicaConf {
+ /**
+ * Sender/Reciver
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ Uint32 senderNode;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/DisconnectRep.hpp b/ndb/include/kernel/signaldata/DisconnectRep.hpp
new file mode 100644
index 00000000000..d7fcdc4fb35
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DisconnectRep.hpp
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DISCONNECT_REP_HPP
+#define DISCONNECT_REP_HPP
+
+#include "SignalData.hpp"
+
+/**
+ *
+ */
+class DisconnectRep {
+ /**
+ * Receiver(s)
+ */
+ friend class Qmgr;
+ friend class Cmvmi; // Cmvmi
+
+ /**
+ * Senders
+ */
+ friend class Dbtc;
+ friend void reportDisconnect(void * , NodeId, Uint32); // TransporterCallback
+
+ /**
+ * For printing
+ */
+ friend bool printDISCONNECT_REP(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ enum ErrCode {
+ // ErrorCodes come from different sources
+ // for example TransporterCallback.hpp
+ // or inet errno
+ // This one is selected not to conflict with any of them
+ TcReportNodeFailed = 0xFF000001
+ };
+
+private:
+
+ Uint32 nodeId;
+ Uint32 err;
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DropIndx.hpp b/ndb/include/kernel/signaldata/DropIndx.hpp
new file mode 100644
index 00000000000..0c0cf31aec8
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DropIndx.hpp
@@ -0,0 +1,253 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DROP_INDX_HPP
+#define DROP_INDX_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+
+/**
+ * DropIndxReq.
+ */
+class DropIndxReq {
+ friend bool printDROP_INDX_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum RequestType {
+ RT_UNDEFINED = 0,
+ RT_USER = 1,
+ RT_DICT_PREPARE = 1 << 4,
+ RT_DICT_COMMIT = 0xC << 4,
+ RT_DICT_ABORT = 0xF << 4,
+ RT_TC = 5 << 8
+ };
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+ Uint32 m_connectionPtr;
+ Uint32 m_userRef;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexId;
+ Uint32 m_indexVersion;
+ // extra
+ Uint32 m_opKey;
+
+public:
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ DropIndxReq::RequestType getRequestType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_requestInfo, 0, 16);
+ return (DropIndxReq::RequestType)val;
+ }
+ void setRequestType(DropIndxReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getRequestFlag() const {
+ return BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ };
+ void addRequestFlag(Uint32 val) {
+ val |= BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ BitmaskImpl::setField(1, &m_requestInfo, 16, 16, val);
+ };
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getIndexVersion() const {
+ return m_indexVersion;
+ }
+ void setIndexVersion(Uint32 val) {
+ m_indexVersion = val;
+ }
+ Uint32 getOpKey() const {
+ return m_opKey;
+ }
+ void setOpKey(Uint32 val) {
+ m_opKey = val;
+ }
+};
+
+/**
+ * DropIndxConf.
+ */
+class DropIndxConf {
+ friend bool printDROP_INDX_CONF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( InternalLength = 3 );
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+ Uint32 m_connectionPtr;
+ Uint32 m_userRef;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexId;
+ Uint32 m_indexVersion;
+
+public:
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ DropIndxReq::RequestType getRequestType() const {
+ return (DropIndxReq::RequestType)m_requestInfo;
+ }
+ void setRequestType(DropIndxReq::RequestType val) {
+ m_requestInfo = val;
+ }
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getIndexVersion() const {
+ return m_indexVersion;
+ }
+ void setIndexVersion(Uint32 val) {
+ m_indexVersion = val;
+ }
+};
+
+/**
+ * DropIndxRef.
+ */
+class DropIndxRef {
+ friend bool printDROP_INDX_REF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum ErrorCode {
+ NoError = 0,
+ InvalidIndexVersion = 241,
+ Busy = 701,
+ IndexNotFound = 4243,
+ BadRequestType = 4247,
+ InvalidName = 4248,
+ NotAnIndex = 4254
+ };
+ STATIC_CONST( SignalLength = DropIndxConf::SignalLength + 3 );
+
+private:
+ DropIndxConf m_conf;
+ //Uint32 m_userRef;
+ //Uint32 m_connectionPtr;
+ //Uint32 m_requestInfo;
+ //Uint32 m_tableId;
+ //Uint32 m_indexId;
+ //Uint32 m_indexVersion;
+ Uint32 m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+
+public:
+ DropIndxConf* getConf() {
+ return &m_conf;
+ }
+ const DropIndxConf* getConf() const {
+ return &m_conf;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_conf.getConnectionPtr();
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_conf.setConnectionPtr(val);
+ }
+ Uint32 getUserRef() const {
+ return m_conf.getUserRef();
+ }
+ void setUserRef(Uint32 val) {
+ m_conf.setUserRef(val);
+ }
+ DropIndxReq::RequestType getRequestType() const {
+ return m_conf.getRequestType();
+ }
+ void setRequestType(DropIndxReq::RequestType val) {
+ m_conf.setRequestType(val);
+ }
+ Uint32 getTableId() const {
+ return m_conf.getTableId();
+ }
+ void setTableId(Uint32 val) {
+ m_conf.setTableId(val);
+ }
+ Uint32 getIndexId() const {
+ return m_conf.getIndexId();
+ }
+ void setIndexId(Uint32 val) {
+ m_conf.setIndexId(val);
+ }
+ Uint32 getIndexVersion() const {
+ return m_conf.getIndexVersion();
+ }
+ void setIndexVersion(Uint32 val) {
+ m_conf.setIndexVersion(val);
+ }
+ DropIndxRef::ErrorCode getErrorCode() const {
+ return (DropIndxRef::ErrorCode)m_errorCode;
+ }
+ void setErrorCode(DropIndxRef::ErrorCode val) {
+ m_errorCode = (Uint32)val;
+ }
+ Uint32 getErrorLine() const {
+ return m_errorLine;
+ }
+ void setErrorLine(Uint32 val) {
+ m_errorLine = val;
+ }
+ Uint32 getErrorNode() const {
+ return m_errorNode;
+ }
+ void setErrorNode(Uint32 val) {
+ m_errorNode = val;
+ }
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DropTab.hpp b/ndb/include/kernel/signaldata/DropTab.hpp
new file mode 100644
index 00000000000..906f952d852
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DropTab.hpp
@@ -0,0 +1,114 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DROP_TAB_HPP
+#define DROP_TAB_HPP
+
+#include "SignalData.hpp"
+
+class DropTabReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtc;
+ friend class Dblqh;
+ friend class Dbacc;
+ friend class Dbtup;
+ friend class Dbtux;
+ friend class Dbdih;
+
+ friend bool printDROP_TAB_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum RequestType {
+ OnlineDropTab = 0,
+ CreateTabDrop = 1,
+ RestartDropTab = 2
+ };
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 tableId;
+ Uint32 requestType;
+};
+
+class DropTabConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+ friend class Dblqh;
+ friend class Dbacc;
+ friend class Dbtup;
+ friend class Dbtux;
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printDROP_TAB_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 tableId;
+};
+
+class DropTabRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+ friend class Dblqh;
+ friend class Dbacc;
+ friend class Dbtup;
+ friend class Dbtux;
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printDROP_TAB_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum ErrorCode {
+ NoSuchTable = 1,
+ DropWoPrep = 2, // Calling Drop with first calling PrepDrop
+ PrepDropInProgress = 3,
+ DropInProgress = 4
+ };
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 tableId;
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DropTabFile.hpp b/ndb/include/kernel/signaldata/DropTabFile.hpp
new file mode 100644
index 00000000000..9ae4dae41c1
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DropTabFile.hpp
@@ -0,0 +1,64 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DROP_TABFILE_HPP
+#define DROP_TABFILE_HPP
+
+#include "SignalData.hpp"
+
+class DropTabFileReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+ friend class Dbacc;
+ friend class Dbtup;
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 primaryTableId;
+ Uint32 secondaryTableId;
+};
+class DropTabFileConf {
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+ friend class Dbacc;
+ friend class Dbtup;
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 userPtr;
+ Uint32 senderRef;
+ Uint32 nodeId;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DropTable.hpp b/ndb/include/kernel/signaldata/DropTable.hpp
new file mode 100644
index 00000000000..7a5b96e4cd1
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DropTable.hpp
@@ -0,0 +1,80 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DROP_TABLE_HPP
+#define DROP_TABLE_HPP
+
+#include "SignalData.hpp"
+
+class DropTableReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+public:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 tableId;
+ Uint32 tableVersion;
+};
+
+class DropTableRef {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+
+public:
+ STATIC_CONST( SignalLength = 6 );
+
+public:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 tableId;
+ Uint32 tableVersion;
+ Uint32 errorCode;
+ Uint32 masterNodeId;
+
+ enum ErrorCode {
+ Busy = 701,
+ NotMaster = 702,
+ NoSuchTable = 709,
+ InvalidTableVersion = 241,
+ DropInProgress = 283,
+ NoDropTableRecordAvailable = 1229
+ };
+};
+
+class DropTableConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdict;
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+public:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 tableId;
+ Uint32 tableVersion;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DropTrig.hpp b/ndb/include/kernel/signaldata/DropTrig.hpp
new file mode 100644
index 00000000000..7c5049f3de8
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DropTrig.hpp
@@ -0,0 +1,300 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DROP_TRIG_HPP
+#define DROP_TRIG_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+#include <trigger_definitions.h>
+
+/**
+ * DropTrigReq.
+ */
+class DropTrigReq {
+ friend bool printDROP_TRIG_REQ(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum RequestType {
+ RT_UNDEFINED = 0,
+ RT_USER = 1,
+ RT_ALTER_INDEX = 2,
+ RT_BUILD_INDEX = 3,
+ RT_DICT_PREPARE = 1 << 4,
+ RT_DICT_COMMIT = 0xC << 4,
+ RT_DICT_ABORT = 0xF << 4,
+ RT_TC = 5 << 8,
+ RT_LQH = 6 << 8
+ };
+ STATIC_CONST( SignalLength = 7 );
+ SECTION( TRIGGER_NAME_SECTION = 0 ); // optional
+ enum KeyValues {
+ TriggerNameKey = 0xa1
+ };
+
+private:
+ Uint32 m_userRef;
+ Uint32 m_connectionPtr;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexId; // set by DICT if index trigger
+ Uint32 m_triggerId; // set by DICT based on name
+ Uint32 m_triggerInfo; // only for TUP
+ // extra
+ Uint32 m_opKey;
+
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ DropTrigReq::RequestType getRequestType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_requestInfo, 0, 16);
+ return (DropTrigReq::RequestType)val;
+ }
+ void setRequestType(DropTrigReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getRequestFlag() const {
+ return BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ };
+ void addRequestFlag(Uint32 val) {
+ val |= BitmaskImpl::getField(1, &m_requestInfo, 16, 16);
+ BitmaskImpl::setField(1, &m_requestInfo, 16, 16, val);
+ };
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getTriggerId() const {
+ return m_triggerId;
+ }
+ void setTriggerId(Uint32 val) {
+ m_triggerId = val;
+ }
+ Uint32 getTriggerInfo() const {
+ return m_triggerInfo;
+ }
+ void setTriggerInfo(Uint32 val) {
+ m_triggerInfo = val;
+ }
+ TriggerType::Value getTriggerType() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 0, 8);
+ return (TriggerType::Value)val;
+ }
+ void setTriggerType(TriggerType::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 0, 8, (Uint32)val);
+ }
+ TriggerActionTime::Value getTriggerActionTime() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 8, 8);
+ return (TriggerActionTime::Value)val;
+ }
+ void setTriggerActionTime(TriggerActionTime::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 8, 8, (Uint32)val);
+ }
+ TriggerEvent::Value getTriggerEvent() const {
+ const Uint32 val = BitmaskImpl::getField(1, &m_triggerInfo, 16, 8);
+ return (TriggerEvent::Value)val;
+ }
+ void setTriggerEvent(TriggerEvent::Value val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 16, 8, (Uint32)val);
+ }
+ bool getMonitorReplicas() const {
+ return BitmaskImpl::getField(1, &m_triggerInfo, 24, 1);
+ }
+ void setMonitorReplicas(bool val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 24, 1, val);
+ }
+ bool getMonitorAllAttributes() const {
+ return BitmaskImpl::getField(1, &m_triggerInfo, 25, 1);
+ }
+ void setMonitorAllAttributes(bool val) {
+ BitmaskImpl::setField(1, &m_triggerInfo, 25, 1, val);
+ }
+ Uint32 getOpKey() const {
+ return m_opKey;
+ }
+ void setOpKey(Uint32 val) {
+ m_opKey = val;
+ }
+};
+
+/**
+ * DropTrigConf.
+ */
+class DropTrigConf {
+ friend bool printDROP_TRIG_CONF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ STATIC_CONST( InternalLength = 3 );
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+ Uint32 m_userRef;
+ Uint32 m_connectionPtr;
+ Uint32 m_requestInfo;
+ Uint32 m_tableId;
+ Uint32 m_indexId;
+ Uint32 m_triggerId;
+
+ // Public methods
+public:
+ Uint32 getUserRef() const {
+ return m_userRef;
+ }
+ void setUserRef(Uint32 val) {
+ m_userRef = val;
+ }
+ Uint32 getConnectionPtr() const {
+ return m_connectionPtr;
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_connectionPtr = val;
+ }
+ DropTrigReq::RequestType getRequestType() const {
+ return (DropTrigReq::RequestType)m_requestInfo;
+ }
+ void setRequestType(DropTrigReq::RequestType val) {
+ m_requestInfo = (Uint32)val;
+ }
+ Uint32 getTableId() const {
+ return m_tableId;
+ }
+ void setTableId(Uint32 val) {
+ m_tableId = val;
+ }
+ Uint32 getIndexId() const {
+ return m_indexId;
+ }
+ void setIndexId(Uint32 val) {
+ m_indexId = val;
+ }
+ Uint32 getTriggerId() const {
+ return m_triggerId;
+ }
+ void setTriggerId(Uint32 val) {
+ m_triggerId = val;
+ }
+};
+
+/**
+ * DropTrigRef.
+ */
+class DropTrigRef {
+ friend bool printDROP_TRIG_REF(FILE*, const Uint32*, Uint32, Uint16);
+
+public:
+ enum ErrorCode {
+ NoError = 0,
+ Busy = 701,
+ TriggerNotFound = 4238,
+ BadRequestType = 4247,
+ InvalidName = 4248
+ };
+ STATIC_CONST( SignalLength = DropTrigConf::SignalLength + 3 );
+
+private:
+ DropTrigConf m_conf;
+ //Uint32 m_userRef;
+ //Uint32 m_connectionPtr;
+ //Uint32 m_requestInfo;
+ //Uint32 m_tableId;
+ //Uint32 m_indexId;
+ //Uint32 m_triggerId;
+ Uint32 m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+
+public:
+ DropTrigConf* getConf() {
+ return &m_conf;
+ }
+ const DropTrigConf* getConf() const {
+ return &m_conf;
+ }
+ Uint32 getUserRef() const {
+ return m_conf.getUserRef();
+ }
+ void setUserRef(Uint32 val) {
+ m_conf.setUserRef(val);
+ }
+ Uint32 getConnectionPtr() const {
+ return m_conf.getConnectionPtr();
+ }
+ void setConnectionPtr(Uint32 val) {
+ m_conf.setConnectionPtr(val);
+ }
+ DropTrigReq::RequestType getRequestType() const {
+ return m_conf.getRequestType();
+ }
+ void setRequestType(DropTrigReq::RequestType val) {
+ m_conf.setRequestType(val);
+ }
+ Uint32 getTableId() const {
+ return m_conf.getTableId();
+ }
+ void setTableId(Uint32 val) {
+ m_conf.setTableId(val);
+ }
+ Uint32 getIndexId() const {
+ return m_conf.getIndexId();
+ }
+ void setIndexId(Uint32 val) {
+ m_conf.setIndexId(val);
+ }
+ Uint32 getTriggerId() const {
+ return m_conf.getTriggerId();
+ }
+ void setTriggerId(Uint32 val) {
+ m_conf.setTriggerId(val);
+ }
+ DropTrigRef::ErrorCode getErrorCode() const {
+ return (DropTrigRef::ErrorCode)m_errorCode;
+ }
+ void setErrorCode(DropTrigRef::ErrorCode val) {
+ m_errorCode = (Uint32)val;
+ }
+ Uint32 getErrorLine() const {
+ return m_errorLine;
+ }
+ void setErrorLine(Uint32 val) {
+ m_errorLine = val;
+ }
+ Uint32 getErrorNode() const {
+ return m_errorNode;
+ }
+ void setErrorNode(Uint32 val) {
+ m_errorNode = val;
+ }
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/DumpStateOrd.hpp b/ndb/include/kernel/signaldata/DumpStateOrd.hpp
new file mode 100644
index 00000000000..6403a52926f
--- /dev/null
+++ b/ndb/include/kernel/signaldata/DumpStateOrd.hpp
@@ -0,0 +1,134 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DUMP_STATE_ORD_HPP
+#define DUMP_STATE_ORD_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * DumpStateOrd is sent by the mgmtsrvr to CMVMI.
+ * CMVMI the redirect the signal to all blocks.
+ *
+ * The implementation of the DumpStateOrd should dump state information
+ * (typically using the infoEvent-function)
+ */
+class DumpStateOrd {
+ /**
+ * Sender/Reciver
+ */
+ friend class Cmvmi;
+
+ /**
+ * Sender(s)
+ */
+ friend class MgmtSrvr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbacc;
+ friend class Dblqh;
+ friend class Dbtup;
+ friend class Dbtc;
+ friend class Ndbcntr;
+ friend class Qmgr;
+ friend class Dbdih;
+ friend class Dbdict;
+ friend class Ndbfs;
+
+public:
+ enum DumpStateType {
+ // 1 QMGR Dump information about phase 1 variables
+ // 13 CMVMI Dump signal counter
+ // 13 NDBCNTR Dump start phase information
+ // 13 NDBCNTR_REF Dump start phase information
+ CommitAckMarkersSize = 14, // TC+LQH Dump free size in commitAckMarkerP
+ CommitAckMarkersDump = 15, // TC+LQH Dump info in commitAckMarkerPool
+ DihDumpNodeRestartInfo = 16, // 16 DIH Dump node restart info
+ DihDumpNodeStatusInfo = 17,// 17 DIH Dump node status info
+ DihPrintFragmentation = 18,// 18 DIH Print fragmentation
+ // 19 NDBFS Fipple with O_SYNC, O_CREATE etc.
+ // 20-24 BACKUP
+ NdbcntrTestStopOnError = 25,
+ // 100-105 TUP and ACC
+ // 200-240 UTIL
+ // 300-305 TRIX
+ NdbfsDumpFileStat = 400,
+ NdbfsDumpAllFiles = 401,
+ NdbfsDumpOpenFiles = 402,
+ NdbfsDumpIdleFiles = 403,
+ // 1222-1225 DICT
+ LqhDumpAllDefinedTabs = 1332,
+ LqhDumpNoLogPages = 1333,
+ LqhDumpOneScanRec = 2300,
+ LqhDumpAllScanRec = 2301,
+ LqhDumpAllActiveScanRec = 2302,
+ LqhDumpLcpState = 2303,
+ AccDumpOneScanRec = 2400,
+ AccDumpAllScanRec = 2401,
+ AccDumpAllActiveScanRec = 2402,
+ AccDumpOneOperationRec = 2403,
+ AccDumpNumOpRecs = 2404,
+ AccDumpFreeOpRecs = 2405,
+ AccDumpNotFreeOpRecs = 2406,
+ DumpPageMemory = 1000, // Acc & TUP
+ TcDumpAllScanFragRec = 2500,
+ TcDumpOneScanFragRec = 2501,
+ TcDumpAllScanRec = 2502,
+ TcDumpAllActiveScanRec = 2503,
+ TcDumpOneScanRec = 2504,
+ TcDumpOneApiConnectRec = 2505,
+ TcDumpAllApiConnectRec = 2506,
+ TcSetTransactionTimeout = 2507,
+ CmvmiDumpConnections = 2600,
+ CmvmiDumpLongSignalMemory = 2601,
+ CmvmiSetRestartOnErrorInsert = 2602,
+ CmvmiTestLongSigWithDelay = 2603,
+ // 7000 DIH
+ // 7001 DIH
+ // 7002 DIH
+ // 7003 DIH
+ // 7004 DIH
+ // 7005 DIH
+ // 7006 DIH
+ // 7006 DIH
+ // 7007 DIH
+ // 7008 DIH
+ // 7009 DIH
+ // 7010 DIH
+ // 7011 DIH
+ // 7012 DIH
+ DihDumpLCPState= 7013,
+ DihDumpLCPMasterTakeOver = 7014,
+ // 7015 DIH
+ DihAllAllowNodeStart = 7016,
+ DihMinTimeBetweenLCP = 7017,
+ DihMaxTimeBetweenLCP = 7018,
+ EnableUndoDelayDataWrite = 7080, // DIH+ACC+TUP
+ DihStartLcpImmediately = 7099,
+ // 8000 Suma
+ // 12000 Tux
+ TuxLogToFile = 12001,
+ TuxSetLogFlags = 12002,
+ TuxMetaDataJunk = 12009
+ };
+public:
+
+ Uint32 args[25]; // Generic argument
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/EmptyLcp.hpp b/ndb/include/kernel/signaldata/EmptyLcp.hpp
new file mode 100644
index 00000000000..32ea6c13231
--- /dev/null
+++ b/ndb/include/kernel/signaldata/EmptyLcp.hpp
@@ -0,0 +1,77 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef EMPTY_LCPREQ_HPP
+#define EMPTY_LCPREQ_HPP
+
+/**
+ * This signals is sent by Dbdih-Master to Dblqh
+ * as part of master take over after node crash
+ */
+class EmptyLcpReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Sender(s) / Receiver(s)
+ */
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+private:
+
+ Uint32 senderRef;
+};
+
+/**
+ * This signals is sent by Dblqh to Dbdih
+ * as part of master take over after node crash
+ */
+class EmptyLcpConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Sender(s) / Receiver(s)
+ */
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 6 );
+private:
+
+ Uint32 senderNodeId;
+ Uint32 tableId;
+ Uint32 fragmentId;
+ Uint32 lcpNo;
+ Uint32 lcpId;
+ Uint32 idle;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/EndTo.hpp b/ndb/include/kernel/signaldata/EndTo.hpp
new file mode 100644
index 00000000000..944cca3ca98
--- /dev/null
+++ b/ndb/include/kernel/signaldata/EndTo.hpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef END_TO_HPP
+#define END_TO_HPP
+
+class EndToReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+private:
+ Uint32 userPtr;
+ BlockReference userRef;
+ Uint32 startingNodeId;
+ Uint32 nodeTakenOver;
+};
+
+class EndToConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+private:
+
+ Uint32 userPtr;
+ Uint32 sendingNodeId;
+ Uint32 startingNodeId;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/EventReport.hpp b/ndb/include/kernel/signaldata/EventReport.hpp
new file mode 100644
index 00000000000..b6106bb0ca4
--- /dev/null
+++ b/ndb/include/kernel/signaldata/EventReport.hpp
@@ -0,0 +1,159 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SD_EVENT_REPORT_H
+#define SD_EVENT_REPORT_H
+
+#include "SignalData.hpp"
+
+/**
+ * Send by different block to report that a event has taken place
+ *
+ * SENDER: *Block*
+ * RECIVER: SimBlockCMCtrBlck
+ */
+class EventReport {
+ friend class SimulatedBlock;
+ friend class Cmvmi;
+ friend class SimblockMissra;
+ friend class Dbacc;
+ friend class Dblqh;
+ friend class Dbtup;
+ friend class Dbtc;
+ friend class Ndbcntr;
+ friend class Qmgr;
+ friend class Dbdih;
+ friend class Dbdict;
+ friend class MgmtSrvr;
+ friend class Grep;
+public:
+ /*
+ EventType defines what event reports to send.
+
+ The ORDER is NOT important anymore. //ejonore 2003-07-24 15:03
+
+ HOW TO ADD A NEW EVENT
+ --------------------
+ 1) Add SentHeartbeat EventType in the category where it belongs.
+ ...
+ // INFO
+ SentHeartbeat,
+ InfoEvent
+ ...
+
+ 2) remeber to update # of events below. Just to keep count...
+ Number of event types = 53
+
+ 3) Add a new SentHeartBeat entry to EventLogger::matrix[].
+ ...
+ // INFO
+ { EventReport::SentHeartbeat, LogLevel::llInfo, 11, INFO },
+ { EventReport::InfoEvent, LogLevel::llInfo, 2, INFO }
+ ...
+
+ 4) Add SentHeartbeat in EventLogger::getText()
+
+ */
+ enum EventType {
+ // CONNECTION
+ Connected = 0,
+ Disconnected = 1,
+ CommunicationClosed = 2,
+ CommunicationOpened = 3,
+ ConnectedApiVersion = 51,
+ // CHECKPOINT
+ GlobalCheckpointStarted = 4,
+ GlobalCheckpointCompleted = 5,
+ LocalCheckpointStarted = 6,
+ LocalCheckpointCompleted = 7,
+ LCPStoppedInCalcKeepGci = 8,
+ LCPFragmentCompleted = 9,
+ // STARTUP
+ NDBStartStarted = 10,
+ NDBStartCompleted = 11,
+ STTORRYRecieved = 12,
+ StartPhaseCompleted = 13,
+ CM_REGCONF = 14,
+ CM_REGREF = 15,
+ FIND_NEIGHBOURS = 16,
+ NDBStopStarted = 17,
+ NDBStopAborted = 18,
+ StartREDOLog = 19,
+ StartLog = 20,
+ UNDORecordsExecuted = 21,
+
+ // NODERESTART
+ NR_CopyDict = 22,
+ NR_CopyDistr = 23,
+ NR_CopyFragsStarted = 24,
+ NR_CopyFragDone = 25,
+ NR_CopyFragsCompleted = 26,
+
+ // NODEFAIL
+ NodeFailCompleted = 27,
+ NODE_FAILREP = 28,
+ ArbitState = 29,
+ ArbitResult = 30,
+ GCP_TakeoverStarted = 31,
+ GCP_TakeoverCompleted = 32,
+ LCP_TakeoverStarted = 33,
+ LCP_TakeoverCompleted = 34,
+
+ // STATISTIC
+ TransReportCounters = 35,
+ OperationReportCounters = 36,
+ TableCreated = 37,
+ UndoLogBlocked = 38,
+ JobStatistic = 39,
+ SendBytesStatistic = 40,
+ ReceiveBytesStatistic = 41,
+ MemoryUsage = 50,
+
+ // ERROR
+ TransporterError = 42,
+ TransporterWarning = 43,
+ MissedHeartbeat = 44,
+ DeadDueToHeartbeat = 45,
+ WarningEvent = 46,
+ // INFO
+ SentHeartbeat = 47,
+ CreateLogBytes = 48,
+ InfoEvent = 49,
+
+ //GREP
+ GrepSubscriptionInfo = 52,
+ GrepSubscriptionAlert = 53
+ };
+
+ void setEventType(EventType type);
+ EventType getEventType() const;
+private:
+ UintR eventType; // DATA 0
+};
+
+inline
+void
+EventReport::setEventType(EventType type){
+ eventType = (UintR) type;
+}
+
+inline
+EventReport::EventType
+EventReport::getEventType() const {
+ return (EventType)eventType;
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/EventSubscribeReq.hpp b/ndb/include/kernel/signaldata/EventSubscribeReq.hpp
new file mode 100644
index 00000000000..2ac62be19a3
--- /dev/null
+++ b/ndb/include/kernel/signaldata/EventSubscribeReq.hpp
@@ -0,0 +1,60 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SD_EVENT_SUB_REQ_H
+#define SD_EVENT_SUB_REQ_H
+
+#include "SignalData.hpp"
+
+/**
+ * Requests change (new, update, delete) of event subscription,
+ * i.e. forwarding of events.
+ *
+ * SENDER: Mgm server
+ * RECIVER: SimBlockCMCtrBlck
+ */
+
+class EventSubscribeReq {
+ /**
+ * Receiver(s)
+ */
+ friend class Cmvmi;
+
+ /**
+ * Sender(s)
+ */
+ friend class MgmtSrvr;
+
+public:
+ STATIC_CONST( SignalLength = 14 );
+private:
+ /**
+ * Note: If you use the same blockRef as you have used earlier,
+ * you update your ongoing subscription
+ */
+ Uint32 blockRef;
+
+ /**
+ * If you specify 0 entries, it's the same as cancelling an
+ * subscription
+ */
+ Uint32 noOfEntries;
+
+ Uint32 theCategories[6];
+ Uint32 theLevels[6];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ExecFragReq.hpp b/ndb/include/kernel/signaldata/ExecFragReq.hpp
new file mode 100644
index 00000000000..e40213d6e29
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ExecFragReq.hpp
@@ -0,0 +1,43 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef EXEC_FRAGREQ_HPP
+#define EXEC_FRAGREQ_HPP
+
+#include "SignalData.hpp"
+
+class ExecFragReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+public:
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 startGci;
+ Uint32 lastGci;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/FailRep.hpp b/ndb/include/kernel/signaldata/FailRep.hpp
new file mode 100644
index 00000000000..44577f07fdc
--- /dev/null
+++ b/ndb/include/kernel/signaldata/FailRep.hpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FAIL_REP_HPP
+#define FAIL_REP_HPP
+
+#include "SignalData.hpp"
+
+/**
+ *
+ */
+class FailRep {
+ /**
+ * Sender(s) & Reciver(s)
+ */
+ friend class Qmgr;
+
+ /**
+ * For printing
+ */
+ friend bool printFAIL_REP(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ enum FailCause {
+ ZOWN_FAILURE=0,
+ ZOTHER_NODE_WHEN_WE_START=1,
+ ZIN_PREP_FAIL_REQ=2,
+ ZSTART_IN_REGREQ=3,
+ ZHEARTBEAT_FAILURE=4,
+ ZLINK_FAILURE=5,
+ ZOTHERNODE_FAILED_DURING_START=6
+ };
+
+private:
+
+ Uint32 failNodeId;
+ Uint32 failCause;
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/FireTrigOrd.hpp b/ndb/include/kernel/signaldata/FireTrigOrd.hpp
new file mode 100644
index 00000000000..20a0a863094
--- /dev/null
+++ b/ndb/include/kernel/signaldata/FireTrigOrd.hpp
@@ -0,0 +1,200 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FIRE_TRIG_ORD_HPP
+#define FIRE_TRIG_ORD_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+#include <trigger_definitions.h>
+#include <string.h>
+
+/**
+ * FireTrigOrd
+ *
+ * This signal is sent by TUP to signal
+ * that a trigger has fired
+ */
+class FireTrigOrd {
+ /**
+ * Sender(s)
+ */
+ // API
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbtup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+ friend class Backup;
+ friend class SumaParticipant;
+
+ /**
+ * For printing
+ */
+ friend bool printFIRE_TRIG_ORD(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 7 );
+ STATIC_CONST( SignalWithGCILength = 8 );
+ STATIC_CONST( SignalWithHashValueLength = 9 );
+
+private:
+ Uint32 m_connectionPtr;
+ Uint32 m_userRef;
+ Uint32 m_triggerId;
+ TriggerEvent::Value m_triggerEvent;
+ Uint32 m_noPrimKeyWords;
+ Uint32 m_noBeforeValueWords;
+ Uint32 m_noAfterValueWords;
+ Uint32 m_gci;
+ Uint32 m_hashValue;
+ // Public methods
+public:
+ Uint32 getConnectionPtr() const;
+ void setConnectionPtr(Uint32);
+ Uint32 getUserRef() const;
+ void setUserRef(Uint32);
+ Uint32 getTriggerId() const;
+ void setTriggerId(Uint32 anIndxId);
+ TriggerEvent::Value getTriggerEvent() const;
+ void setTriggerEvent(TriggerEvent::Value);
+ Uint32 getNoOfPrimaryKeyWords() const;
+ void setNoOfPrimaryKeyWords(Uint32);
+ Uint32 getNoOfBeforeValueWords() const;
+ void setNoOfBeforeValueWords(Uint32);
+ Uint32 getNoOfAfterValueWords() const;
+ void setNoOfAfterValueWords(Uint32);
+ Uint32 getGCI() const;
+ void setGCI(Uint32);
+ Uint32 getHashValue() const;
+ void setHashValue(Uint32);
+};
+
+inline
+Uint32 FireTrigOrd::getConnectionPtr() const
+{
+ return m_connectionPtr;
+}
+
+inline
+void FireTrigOrd::setConnectionPtr(Uint32 aConnectionPtr)
+{
+ m_connectionPtr = aConnectionPtr;
+}
+
+inline
+Uint32 FireTrigOrd::getUserRef() const
+{
+ return m_userRef;
+}
+
+inline
+void FireTrigOrd::setUserRef(Uint32 aUserRef)
+{
+ m_userRef = aUserRef;
+}
+
+inline
+Uint32 FireTrigOrd::getTriggerId() const
+{
+ return m_triggerId;
+}
+
+inline
+void FireTrigOrd::setTriggerId(Uint32 aTriggerId)
+{
+ m_triggerId = aTriggerId;
+}
+
+inline
+TriggerEvent::Value FireTrigOrd::getTriggerEvent() const
+{
+ return m_triggerEvent;
+}
+
+inline
+void FireTrigOrd::setTriggerEvent(TriggerEvent::Value aTriggerEvent)
+{
+ m_triggerEvent = aTriggerEvent;
+}
+
+inline
+Uint32 FireTrigOrd::getNoOfPrimaryKeyWords() const
+{
+ return m_noPrimKeyWords;
+}
+
+inline
+void FireTrigOrd::setNoOfPrimaryKeyWords(Uint32 noPrim)
+{
+ m_noPrimKeyWords = noPrim;
+}
+
+inline
+Uint32 FireTrigOrd::getNoOfBeforeValueWords() const
+{
+ return m_noBeforeValueWords;
+}
+
+inline
+void FireTrigOrd::setNoOfBeforeValueWords(Uint32 noBefore)
+{
+ m_noBeforeValueWords = noBefore;
+}
+
+inline
+Uint32 FireTrigOrd::getNoOfAfterValueWords() const
+{
+ return m_noAfterValueWords;
+}
+
+inline
+void FireTrigOrd::setNoOfAfterValueWords(Uint32 noAfter)
+{
+ m_noAfterValueWords = noAfter;
+}
+
+inline
+Uint32 FireTrigOrd::getGCI() const
+{
+ return m_gci;
+}
+
+inline
+void FireTrigOrd::setGCI(Uint32 aGCI)
+{
+ m_gci = aGCI;
+}
+
+inline
+Uint32 FireTrigOrd::getHashValue() const
+{
+ return m_hashValue;
+}
+
+inline
+void FireTrigOrd::setHashValue(Uint32 flag)
+{
+ m_hashValue = flag;
+}
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/FsAppendReq.hpp b/ndb/include/kernel/signaldata/FsAppendReq.hpp
new file mode 100644
index 00000000000..e2fd61f8a11
--- /dev/null
+++ b/ndb/include/kernel/signaldata/FsAppendReq.hpp
@@ -0,0 +1,57 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FS_APPENDREQ_H
+#define FS_APPENDREQ_H
+
+#include "SignalData.hpp"
+
+/**
+ *
+ * SENDER:
+ * RECIVER: Ndbfs
+ */
+class FsAppendReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbfs;
+ friend class VoidFs;
+
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+
+ friend bool printFSAPPENDREQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo);
+public:
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ UintR filePointer; // DATA 0
+ UintR userReference; // DATA 1
+ UintR userPointer; // DATA 2
+ UintR varIndex; // DATA 3
+ UintR offset; // DATA 4
+ UintR size; // DATA 5
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/FsCloseReq.hpp b/ndb/include/kernel/signaldata/FsCloseReq.hpp
new file mode 100644
index 00000000000..8ff47145f87
--- /dev/null
+++ b/ndb/include/kernel/signaldata/FsCloseReq.hpp
@@ -0,0 +1,85 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FS_CLOSE_REQ_H
+#define FS_CLOSE_REQ_H
+
+#include "SignalData.hpp"
+
+/**
+ *
+ * SENDER:
+ * RECIVER: Ndbfs
+ */
+class FsCloseReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbfs; // Reciver
+ friend class VoidFs;
+
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+ friend class Dbdict;
+
+ /**
+ * For printing
+ */
+ friend bool printFSCLOSEREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+
+ UintR filePointer; // DATA 0
+ UintR userReference; // DATA 1
+ UintR userPointer; // DATA 2
+ UintR fileFlag; // DATA 3
+
+ static bool getRemoveFileFlag(const UintR & fileflag);
+ static void setRemoveFileFlag(UintR & fileflag, bool removefile);
+
+};
+
+
+inline
+bool
+FsCloseReq::getRemoveFileFlag(const UintR & fileflag){
+ return (fileflag == 1);
+}
+
+inline
+void
+FsCloseReq::setRemoveFileFlag(UintR & fileflag, bool removefile){
+ ASSERT_BOOL(removefile, "FsCloseReq::setRemoveFileFlag");
+ if (removefile == true)
+ fileflag = 1;
+ else
+ fileflag = 0;
+}
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/FsConf.hpp b/ndb/include/kernel/signaldata/FsConf.hpp
new file mode 100644
index 00000000000..f66d9feea49
--- /dev/null
+++ b/ndb/include/kernel/signaldata/FsConf.hpp
@@ -0,0 +1,77 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FS_CONF_H
+#define FS_CONF_H
+
+#include "SignalData.hpp"
+
+/**
+ * FsConf - Common signal class for all CONF signals sent from Ndbfs
+ * GSN_FSCLOSECONF, GSN_FSOPENCONF, GSN_FSWRITECONF, GSN_FSREADCONF,
+ * GSN_FSSYNCCONF, GSN_FSREMOVECONF
+ */
+
+/**
+ *
+ * SENDER: Ndbfs
+ * RECIVER:
+ */
+class FsConf {
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+ friend class Dbacc;
+ friend class Dbtup;
+ friend class Dbdict;
+
+ /**
+ * Sender(s)
+ */
+ friend class Ndbfs;
+ friend class VoidFs;
+
+ /**
+ * For printing
+ */
+ friend bool printFSCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Length of signal
+ */
+ /**
+ * FSOPENCONF: static const UintR SignalLength = 2;
+ * FSCLOSECONF, FSREADCONF, FSWRITECONF, FSSYNCCONF: static const UintR SignalLength = 2;
+ */
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ UintR userPointer; // DATA 0
+
+ /**
+ * Only used if FSOPENCONF
+ */
+ UintR filePointer; // DATA 1
+};
+
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/FsOpenReq.hpp b/ndb/include/kernel/signaldata/FsOpenReq.hpp
new file mode 100644
index 00000000000..b84d78ba9dd
--- /dev/null
+++ b/ndb/include/kernel/signaldata/FsOpenReq.hpp
@@ -0,0 +1,266 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FS_OPEN_REQ_H
+#define FS_OPEN_REQ_H
+
+#include "SignalData.hpp"
+
+/**
+ *
+ * SENDER:
+ * RECIVER: Ndbfs
+ */
+class FsOpenReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbfs; // Reciver
+ friend class AsyncFile; // Uses FsOpenReq to decode file open flags
+ friend class Filename;
+ friend class VoidFs;
+
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+ friend class Dbdict;
+ friend class Ndbcntr; // For initial start...
+
+ /**
+ * For printing
+ */
+ friend bool printFSOPENREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 7 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+
+ UintR userReference; // DATA 0
+ UintR userPointer; // DATA 1
+ UintR fileNumber[4]; // DATA 2 - 5
+ UintR fileFlags; // DATA 6
+
+ STATIC_CONST( OM_READONLY = 0 );
+ STATIC_CONST( OM_WRITEONLY = 1 );
+ STATIC_CONST( OM_READWRITE = 2 );
+
+ STATIC_CONST( OM_APPEND = 0x8 ); // Not Implemented on W2k
+ STATIC_CONST( OM_SYNC = 0x10 );
+ STATIC_CONST( OM_CREATE = 0x100 );
+ STATIC_CONST( OM_TRUNCATE = 0x200 );
+
+ enum Suffixes {
+ S_DATA = 0,
+ S_FRAGLOG = 1,
+ S_LOGLOG = 2,
+ S_FRAGLIST = 3,
+ S_TABLELIST = 4,
+ S_SCHEMALOG = 5,
+ S_SYSFILE = 6,
+ S_LOG = 7,
+ S_CTL = 8
+ };
+
+ static Uint32 getVersion(const Uint32 fileNumber[]);
+ static Uint32 getSuffix(const Uint32 fileNumber[]);
+
+ static void setVersion(Uint32 fileNumber[], Uint8 val);
+ static void setSuffix(Uint32 fileNumber[], Uint8 val);
+
+ /**
+ * V1
+ */
+ static Uint32 v1_getDisk(const Uint32 fileNumber[]);
+ static Uint32 v1_getTable(const Uint32 fileNumber[]);
+ static Uint32 v1_getFragment(const Uint32 fileNumber[]);
+ static Uint32 v1_getS(const Uint32 fileNumber[]);
+ static Uint32 v1_getP(const Uint32 fileNumber[]);
+
+ static void v1_setDisk(Uint32 fileNumber[], Uint8 val);
+ static void v1_setTable(Uint32 fileNumber[], Uint32 val);
+ static void v1_setFragment(Uint32 fileNumber[], Uint32 val);
+ static void v1_setS(Uint32 fileNumber[], Uint32 val);
+ static void v1_setP(Uint32 fileNumber[], Uint8 val);
+
+ /**
+ * V2 - Backup
+ */
+ static Uint32 v2_getSequence(const Uint32 fileNumber[]);
+ static Uint32 v2_getNodeId(const Uint32 fileNumber[]);
+ static Uint32 v2_getCount(const Uint32 fileNumber[]);
+
+ static void v2_setSequence(Uint32 fileNumber[], Uint32 no);
+ static void v2_setNodeId(Uint32 fileNumber[], Uint32 no);
+ static void v2_setCount(Uint32 fileNumber[], Uint32 no);
+};
+
+/**
+ * File flags (set according to solaris standard)
+ *
+ o = Open mode - 2 Bits -> max 3
+ c = create new file - 1 Bit
+ t = truncate existing - 1 Bit
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ oo ct
+*/
+
+
+/**
+ * -- v1 --
+ * File number[0] = Table
+ * File number[1] = Fragment
+ * File number[2] = S-value
+ * File number[3] =
+ * p = v1_P 0 - 7
+ * d = v1_disk 8 - 15
+ * s = v1_suffix 16 - 23
+ * v = version 24 - 31
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * ppppppppddddddddssssssssvvvvvvvv
+ *
+ * -- v2 --
+ * File number[0] = Backup Sequence Number
+ * File number[1] = Node Id
+ * File number[3] =
+ * v = version 24 - 31
+ * s = v1_suffix 16 - 23
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * ppppppppddddddddssssssssvvvvvvvv
+ *
+ */
+inline
+Uint32 FsOpenReq::getVersion(const Uint32 fileNumber[]){
+ return (fileNumber[3] >> 24) & 0xff;
+}
+
+inline
+void FsOpenReq::setVersion(Uint32 fileNumber[], Uint8 val){
+ const Uint32 t = fileNumber[3];
+ fileNumber[3] = t & 0x00FFFFFF | (((Uint32)val) << 24);
+}
+
+inline
+Uint32 FsOpenReq::getSuffix(const Uint32 fileNumber[]){
+ return (fileNumber[3] >> 16)& 0xff;
+}
+
+inline
+void FsOpenReq::setSuffix(Uint32 fileNumber[], Uint8 val){
+ const Uint32 t = fileNumber[3];
+ fileNumber[3] = t & 0xFF00FFFF | (((Uint32)val) << 16);
+}
+
+inline
+Uint32 FsOpenReq::v1_getDisk(const Uint32 fileNumber[]){
+ return (fileNumber[3]>>8) & 0xff;
+}
+
+inline
+void FsOpenReq::v1_setDisk(Uint32 fileNumber[], Uint8 val){
+ const Uint32 t = fileNumber[3];
+ fileNumber[3] = t & 0xFFFF00FF | (((Uint32)val) << 8);
+}
+
+inline
+Uint32 FsOpenReq::v1_getTable(const Uint32 fileNumber[]){
+ return fileNumber[0];
+}
+
+inline
+void FsOpenReq::v1_setTable(Uint32 fileNumber[], Uint32 val){
+ fileNumber[0] = val;
+}
+
+inline
+Uint32 FsOpenReq::v1_getFragment(const Uint32 fileNumber[]){
+ return fileNumber[1];
+}
+
+inline
+void FsOpenReq::v1_setFragment(Uint32 fileNumber[], Uint32 val){
+ fileNumber[1] = val;
+}
+
+inline
+Uint32 FsOpenReq::v1_getS(const Uint32 fileNumber[]){
+ return fileNumber[2];
+}
+
+inline
+void FsOpenReq::v1_setS(Uint32 fileNumber[], Uint32 val){
+ fileNumber[2] = val;
+}
+
+inline
+Uint32 FsOpenReq::v1_getP(const Uint32 fileNumber[]){
+ return fileNumber[3] & 0xff;
+}
+
+inline
+void FsOpenReq::v1_setP(Uint32 fileNumber[], Uint8 val){
+ const Uint32 t = fileNumber[3];
+ fileNumber[3] = t & 0xFFFFFF00 | val;
+}
+
+/****************/
+inline
+Uint32 FsOpenReq::v2_getSequence(const Uint32 fileNumber[]){
+ return fileNumber[0];
+}
+
+inline
+void FsOpenReq::v2_setSequence(Uint32 fileNumber[], Uint32 val){
+ fileNumber[0] = val;
+}
+
+inline
+Uint32 FsOpenReq::v2_getNodeId(const Uint32 fileNumber[]){
+ return fileNumber[1];
+}
+
+inline
+void FsOpenReq::v2_setNodeId(Uint32 fileNumber[], Uint32 val){
+ fileNumber[1] = val;
+}
+
+inline
+Uint32 FsOpenReq::v2_getCount(const Uint32 fileNumber[]){
+ return fileNumber[2];
+}
+
+inline
+void FsOpenReq::v2_setCount(Uint32 fileNumber[], Uint32 val){
+ fileNumber[2] = val;
+}
+
+
+#endif
+
diff --git a/ndb/include/kernel/signaldata/FsReadWriteReq.hpp b/ndb/include/kernel/signaldata/FsReadWriteReq.hpp
new file mode 100644
index 00000000000..6e4fa4d260e
--- /dev/null
+++ b/ndb/include/kernel/signaldata/FsReadWriteReq.hpp
@@ -0,0 +1,152 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FS_READWRITEREQ_H
+#define FS_READWRITEREQ_H
+
+#include "SignalData.hpp"
+
+/**
+ * FsReadWriteReq - Common signal class for FSWRITEREQ and FSREADREQ
+ *
+ */
+
+/**
+ *
+ * SENDER:
+ * RECIVER: Ndbfs
+ */
+class FsReadWriteReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbfs;
+ friend class VoidFs;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+
+ /**
+ * For printing
+ */
+ friend bool printFSREADWRITEREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Enum type for errorCode
+ */
+ enum NdbfsFormatType {
+ fsFormatListOfPairs=0,
+ fsFormatArrayOfPages=1,
+ fsFormatListOfMemPages=2,
+ fsFormatMax
+ };
+
+ /**
+ * Length of signal
+ */
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ UintR filePointer; // DATA 0
+ UintR userReference; // DATA 1
+ UintR userPointer; // DATA 2
+ UintR operationFlag; // DATA 3
+ UintR varIndex; // DATA 4
+ UintR numberOfPages; // DATA 5
+
+//-------------------------------------------------------------
+// Variable sized part. Those will contain
+// info about memory/file pages to read/write
+//-------------------------------------------------------------
+ union {
+ UintR pageData[16]; // DATA 6 - 21
+ struct {
+ Uint32 varIndex; // In unit cluster size
+ Uint32 fileOffset; // In unit page size
+ } listOfPair[8];
+ struct {
+ Uint32 varIndex;
+ Uint32 fileOffset;
+ } arrayOfPages;
+ struct {
+ Uint32 varIndex[1]; // Size = numberOfPages
+ Uint32 fileOffset;
+ } listOfMemPages;
+ } data;
+
+ static Uint8 getSyncFlag(const UintR & opFlag);
+ static void setSyncFlag(UintR & opFlag, Uint8 flag);
+
+ static NdbfsFormatType getFormatFlag(const UintR & opFlag);
+ static void setFormatFlag(UintR & opFlag, Uint8 flag);
+
+};
+
+/**
+ * Operation flag
+ *
+ f = Format of pageData - 4 Bits -> max 15
+ s = sync after write flag - 1 Bit
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ ffffs
+*/
+
+#define SYNC_SHIFT (4)
+#define SYNC_MASK (0x01)
+
+#define FORMAT_MASK (0x0F)
+
+
+inline
+Uint8
+FsReadWriteReq::getSyncFlag(const UintR & opFlag){
+ return (Uint8)((opFlag >> SYNC_SHIFT) & SYNC_MASK);
+}
+
+inline
+FsReadWriteReq::NdbfsFormatType
+FsReadWriteReq::getFormatFlag(const UintR & opFlag){
+ return (NdbfsFormatType)(opFlag & FORMAT_MASK);
+}
+
+inline
+void
+FsReadWriteReq::setSyncFlag(UintR & opFlag, Uint8 flag){
+ ASSERT_BOOL(flag, "FsReadWriteReq::setSyncFlag");
+ opFlag |= (flag << SYNC_SHIFT);
+}
+
+inline
+void
+FsReadWriteReq::setFormatFlag(UintR & opFlag, Uint8 flag){
+ ASSERT_MAX(flag, fsFormatMax, "FsReadWriteReq::setSyncFlag");
+ opFlag |= flag;
+}
+
+
+
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/FsRef.hpp b/ndb/include/kernel/signaldata/FsRef.hpp
new file mode 100644
index 00000000000..650f6520fb5
--- /dev/null
+++ b/ndb/include/kernel/signaldata/FsRef.hpp
@@ -0,0 +1,116 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FS_REF_H
+#define FS_REF_H
+
+#include "SignalData.hpp"
+
+/**
+ * FsRef - Common signal class for all REF signals sent from Ndbfs
+ * GSN_FSCLOSEREF, GSN_FSOPENREF, GSN_FSWRITEREF, GSN_FSREADREF,
+ * GSN_FSSYNCREF
+ */
+
+
+/**
+ *
+ * SENDER: Ndbfs
+ * RECIVER:
+ */
+class FsRef {
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdict;
+ friend class Backup;
+
+ /**
+ * Sender(s)
+ */
+ friend class Ndbfs;
+ friend class VoidFs;
+
+ /**
+ * For printing
+ */
+ friend bool printFSREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Enum type for errorCode
+ */
+ enum NdbfsErrorCodeType {
+ fsErrNone=0,
+ fsErrHardwareFailed=1,
+ fsErrUserError=2,
+ fsErrEnvironmentError=3,
+ fsErrTemporaryNotAccessible=4,
+ fsErrNoSpaceLeftOnDevice=5,
+ fsErrPermissionDenied=6,
+ fsErrInvalidParameters=7,
+ fsErrUnknown=8,
+ fsErrNoMoreResources=9,
+ fsErrFileDoesNotExist=10,
+ fsErrReadUnderflow = 11,
+ fsErrMax
+ };
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ UintR userPointer; // DATA 0
+ UintR errorCode; // DATA 1
+ UintR osErrorCode; // DATA 2
+ UintR senderData;
+
+ static NdbfsErrorCodeType getErrorCode(const UintR & errorcode);
+ static void setErrorCode(UintR & errorcode, NdbfsErrorCodeType errorcodetype);
+ static void setErrorCode(UintR & errorcode, UintR errorcodetype);
+
+};
+
+
+inline
+FsRef::NdbfsErrorCodeType
+FsRef::getErrorCode(const UintR & errorcode){
+ return (NdbfsErrorCodeType)errorcode;
+}
+
+inline
+void
+FsRef::setErrorCode(UintR & errorcode, NdbfsErrorCodeType errorcodetype){
+ ASSERT_MAX(errorcodetype, fsErrMax, "FsRef::setErrorCode");
+ errorcode = (UintR)errorcodetype;
+}
+
+inline
+void
+FsRef::setErrorCode(UintR & errorcode, UintR errorcodetype){
+ ASSERT_MAX(errorcodetype, fsErrMax, "FsRef::setErrorCode");
+ errorcode = errorcodetype;
+}
+
+
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/FsRemoveReq.hpp b/ndb/include/kernel/signaldata/FsRemoveReq.hpp
new file mode 100644
index 00000000000..efb566d883a
--- /dev/null
+++ b/ndb/include/kernel/signaldata/FsRemoveReq.hpp
@@ -0,0 +1,78 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FS_REMOVE_REQ_H
+#define FS_REMOVE_REQ_H
+
+#include "SignalData.hpp"
+#include "FsOpenReq.hpp"
+
+/**
+ *
+ * SENDER:
+ * RECIVER: Ndbfs
+ */
+class FsRemoveReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbfs; // Reciver
+ friend class AsyncFile; // Uses FsOpenReq to decode file open flags
+ friend class Filename;
+ friend class VoidFs;
+
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+ friend class Dbdict;
+ friend class Dbacc;
+ friend class Dbtup;
+ friend class Ndbcntr; // For initial start...
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 8 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+
+ UintR userReference; // DATA 0
+ UintR userPointer; // DATA 1
+ UintR fileNumber[4]; // DATA 2 - 5 // See FsOpen for interpretation
+
+ /**
+ * 0 = File -> rm file
+ * 1 = Directory -> rm -r path
+ */
+ UintR directory;
+
+ /**
+ * If directory = 1
+ *
+ * 0 = remove only files/direcories in directory specified in fileNumber
+ * 1 = remove directory specified in fileNumber
+ */
+ UintR ownDirectory;
+};
+
+#endif
+
diff --git a/ndb/include/kernel/signaldata/GCPSave.hpp b/ndb/include/kernel/signaldata/GCPSave.hpp
new file mode 100644
index 00000000000..2b4a25e6bb2
--- /dev/null
+++ b/ndb/include/kernel/signaldata/GCPSave.hpp
@@ -0,0 +1,98 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GCP_SAVE_HPP
+#define GCP_SAVE_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * GCPSaveReq / (Ref/Conf) is sent as part of GCP
+ */
+class GCPSaveReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dblqh;
+
+ friend bool printGCPSaveReq(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 dihBlockRef;
+ Uint32 dihPtr;
+ Uint32 gci;
+};
+
+class GCPSaveRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printGCPSaveRef(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum ErrorCode {
+ NodeShutdownInProgress = 1,
+ FakedSignalDueToNodeFailure = 2,
+ NodeRestartInProgress = 3
+ };
+
+private:
+ Uint32 dihPtr;
+ Uint32 nodeId;
+ Uint32 gci;
+ Uint32 errorCode;
+};
+
+class GCPSaveConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printGCPSaveConf(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 dihPtr;
+ Uint32 nodeId;
+ Uint32 gci;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/GetTabInfo.hpp b/ndb/include/kernel/signaldata/GetTabInfo.hpp
new file mode 100644
index 00000000000..cb6e38872d3
--- /dev/null
+++ b/ndb/include/kernel/signaldata/GetTabInfo.hpp
@@ -0,0 +1,126 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GET_INFO_TAB_HPP
+#define GET_INFO_TAB_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * GetTabInfo - Get table info from DICT
+ *
+ * Successfull return = series of DICTTABINFO-signals
+ */
+class GetTabInfoReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ // Blocks
+ friend class Dbdict;
+ friend class Backup;
+ friend class Trix;
+ friend class DbUtil;
+ // API
+ friend class Table;
+
+ friend bool printGET_TABINFO_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ // STATIC_CONST( MaxTableNameLengthInWords = 20 );
+public:
+ Uint32 senderData;
+ Uint32 senderRef;
+
+ /**
+ * 0 = request by id, 1 = request by name
+ */
+ Uint32 requestType;
+
+ union {
+ Uint32 tableId;
+ Uint32 tableNameLen;
+ };
+ Uint32 unused; // This is located here so that Req & Ref have the same format
+ // Uint32 tableName[MaxTableNameLengthInWords];
+
+ enum RequestType {
+ RequestById = 0,
+ RequestByName = 1,
+ LongSignalConf = 2
+ };
+ SECTION( TABLE_NAME = 0 );
+};
+
+class GetTabInfoRef {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ // Blocks
+ friend class Dbdict;
+ friend class Backup;
+ friend class Trix;
+ friend class DbUtil;
+ // API
+ friend class Table;
+
+ friend bool printGET_TABINFO_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+public:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 requestType; // 0 = request by id, 1 = request by name
+ union {
+ Uint32 tableId;
+ Uint32 tableNameLen;
+ };
+ Uint32 errorCode;
+
+ enum ErrorCode {
+ InvalidTableId = 709,
+ TableNotDefined = 723,
+ TableNameTooLong = 702,
+ Busy = 701
+ };
+};
+
+class GetTabInfoConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ // Blocks
+ friend class Dbdict;
+ friend class Backup;
+ friend class Trix;
+ friend class DbUtil;
+ friend class Suma;
+ // API
+ friend class Table;
+
+ friend bool printGET_TABINFO_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ SECTION( DICT_TAB_INFO = 0 );
+public:
+ Uint32 senderData;
+ Uint32 tableId;
+ Uint32 gci; // For table
+ Uint32 totalLen; // In words
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/GetTableId.hpp b/ndb/include/kernel/signaldata/GetTableId.hpp
new file mode 100644
index 00000000000..fb91c2e10d7
--- /dev/null
+++ b/ndb/include/kernel/signaldata/GetTableId.hpp
@@ -0,0 +1,93 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GET_TABLEID_HPP
+#define GET_TABLEID_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * Convert tabname to table id
+ */
+class GetTableIdReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ // Blocks
+ friend class Dbdict;
+ friend class SumaParticipant;
+
+ friend bool printGET_TABLEID_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+public:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 len;
+ SECTION( TABLE_NAME = 0 );
+};
+
+
+/**
+ * Convert tabname to table id
+ */
+class GetTableIdRef {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ // Blocks
+ friend class Dbdict;
+ friend class SumaParticipant;
+ friend bool printGET_TABLEID_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+public:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 err;
+
+ enum ErrorCode {
+ InvalidTableId = 709,
+ TableNotDefined = 723,
+ TableNameTooLong = 702,
+ EmptyTable = 1111
+ };
+};
+
+
+/**
+ * Convert tabname to table id
+ */
+class GetTableIdConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ // Blocks
+ friend class Dbdict;
+ friend class SumaParticipant;
+ friend bool printGET_TABLEID_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+public:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 tableId;
+ Uint32 schemaVersion;
+
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/GrepImpl.hpp b/ndb/include/kernel/signaldata/GrepImpl.hpp
new file mode 100644
index 00000000000..95b93df0a58
--- /dev/null
+++ b/ndb/include/kernel/signaldata/GrepImpl.hpp
@@ -0,0 +1,891 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GREP_IMPL_HPP
+#define GREP_IMPL_HPP
+
+#include "SignalData.hpp"
+#include <GrepError.hpp>
+#include <NodeBitmask.hpp>
+
+
+
+/*****************************************************************************
+ * GREP REQ Request a Global Replication (between SS and PS)
+ *****************************************************************************/
+/**
+ * @class GrepReq
+ * @brief
+ */
+class GrepReq
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+public:
+ enum Request {
+ START = 0, ///< Start Global Replication (all phases)
+ SLOWSTOP = 1, ///< Stop after finishing applying current GCI epoch
+ FASTSTOP = 2, ///< Stop after finishing applying all PS GCI epochs
+ STATUS = 3, ///< Status
+ REMOVE_BUFFERS = 4, ///< Remove buffers from PS and SS
+
+ START_SUBSCR = 5,
+ START_METALOG = 6, ///< Start Global Replication Logging of Metadata
+ START_METASCAN = 7, ///< Start Global Replication Scanning of Metadata
+ START_DATALOG = 8, ///< Start Global Replication Logging of table data
+ START_DATASCAN = 9, ///< Start Global Replication Scanning of table data
+ START_REQUESTOR = 10, ///< Start Global Replication Requestor
+ START_TRANSFER = 11, ///< Start SS-PS transfer
+ START_APPLY = 12, ///< Start applying GCI epochs in SS
+ START_DELETE = 13, ///< Start deleting buffers at PS/SS REP automatic.
+
+ STOP_SUBSCR = 14, ///< Remove subscription
+ STOP_METALOG = 15, ///< Stop Global Replication Logging of Metadata
+ STOP_METASCAN = 16, ///< Stop Global Replication Scanning of Metadata
+ STOP_DATALOG = 17, ///< Stop Global Replication Logging of table data
+ STOP_DATASCAN = 18, ///< Stop Global Replication Scanning of table data
+ STOP_REQUESTOR = 19, ///< Stop Global Replication Requestor
+ STOP_TRANSFER = 20, ///< Stop SS-PS transfer
+ STOP_APPLY = 21, ///< Stop applying GCI epochs in SS
+ STOP_DELETE = 22, ///< Stop deleting buffers at PS/SS REP automatically
+ CREATE_SUBSCR = 23, ///< Create subscription ID in SUMA
+ DROP_TABLE = 24, ///< Create subscription ID in SUMA
+ STOP = 25,
+
+ NO_REQUEST = 0xffffffff
+ };
+
+ STATIC_CONST( SignalLength = 2 );
+
+ Uint32 senderRef;
+ Uint32 request;
+};
+
+
+/*****************************************************************************
+ * CREATE Between SS and PS (DB and REP nodes)
+ *****************************************************************************/
+/**
+ * @class GrepSubCreateReq
+ * @brief
+ */
+class GrepSubCreateReq
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_CREATE_REQ(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+ Uint32 senderRef;
+ Uint32 senderData;
+ SECTION( TABLE_LIST = 0 );
+};
+
+/**
+ * @class GrepSubCreateReq
+ * @brief
+ */
+class GrepSubCreateRef
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_CREATE_REF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+ Uint32 err;
+ Uint32 senderRef;
+ Uint32 senderData;
+};
+
+
+/**
+ * @class GrepSubCreateConf
+ * @brief
+ */
+class GrepSubCreateConf
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_CREATE_CONF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 noOfNodeGroups;
+};
+
+
+
+/*****************************************************************************
+ * CREATE Internal between PS DB nodes
+ *****************************************************************************/
+
+/**
+ * @class GrepCreateReq
+ * @brief
+ */
+class GrepCreateReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class GrepParticipant;
+
+ friend bool printGREP_CREATE_REQ(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 8 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriberData;
+ Uint32 subscriberRef;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+ SECTION( TABLE_LIST = 0 );
+};
+
+
+/**
+ * @class GrepCreateRef
+ * @brief
+ */
+class GrepCreateRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_CREATE_REF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ enum ErrorCode {
+ NF_FakeErrorREF = GrepError::NF_FakeErrorREF
+ };
+ STATIC_CONST( SignalLength = 6 );
+ Uint32 senderRef;
+ Uint32 senderData;
+ union {
+ Uint32 err;
+ Uint32 errorCode;
+ };
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+};
+
+
+/**
+ * @class GrepCreateConf
+ * @brief
+ */
+class GrepCreateConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_CREATE_CONF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+ Uint32 senderNodeId;
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+};
+
+
+/*****************************************************************************
+ * START Between SS and PS (DB and REP nodes)
+ *****************************************************************************/
+
+/**
+ * @class GrepSubStartReq
+ * @brief
+ */
+class GrepSubStartReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_START_REQ(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 part;
+};
+
+/**
+ * @class GrepSubStartRef
+ * @brief
+ */
+class GrepSubStartRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_START_REF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 err;
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 part;
+};
+
+
+
+/**
+ * @class GrepSubStartConf
+ * @brief
+ */
+class GrepSubStartConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_START_CONF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 part;
+ Uint32 firstGCI;
+};
+
+
+/*****************************************************************************
+ * START Internal between PS DB nodes
+ *****************************************************************************/
+
+/**
+ * @class GrepStartReq
+ * @brief
+ */
+class GrepStartReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class GrepParticipant;
+
+ friend bool printGREP_START_REQ(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ Uint32 senderData;
+ Uint32 part;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+};
+
+
+/**
+ * @class GrepStartRef
+ * @brief
+ */
+class GrepStartRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_START_REF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ enum ErrorCode {
+ NF_FakeErrorREF = GrepError::NF_FakeErrorREF
+ };
+ STATIC_CONST( SignalLength = 6 );
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 part;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ union {
+ Uint32 err;
+ Uint32 errorCode;
+ };
+};
+
+
+/**
+ * @class GrepStartConf
+ * @brief
+ */
+class GrepStartConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_START_CONF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+ public:
+ STATIC_CONST( SignalLength = 7 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 part;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 firstGCI;
+ Uint32 senderNodeId;
+ };
+
+
+/*****************************************************************************
+ * SCAN (SYNC) Between SS and PS (REP and DB nodes)
+ *****************************************************************************/
+
+/**
+ * @class GrepSubSyncReq
+ * @brief
+ */
+class GrepSubSyncReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_SYNC_REQ(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 part;
+};
+
+
+/**
+ * @class GrepSubSyncRef
+ * @brief
+ */
+class GrepSubSyncRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_SYNC_REF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderRef;
+ Uint32 err;
+ Uint32 senderData;
+ Uint32 part;
+};
+
+
+/**
+ * @class GrepSubSyncConf
+ * @brief
+ */
+class GrepSubSyncConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_SYNC_CONF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+ public:
+ STATIC_CONST( SignalLength = 7 );
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 part;
+ Uint32 firstGCI;
+ Uint32 lastGCI;
+};
+
+
+
+/*****************************************************************************
+ * SCAN (SYNC) Internal between PS DB nodes
+ *****************************************************************************/
+
+/**
+ * @class GrepSyncReq
+ * @brief
+ */
+class GrepSyncReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class GrepParticipant;
+
+ friend bool printGREP_SYNC_REQ(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ Uint32 senderData;
+ Uint32 part;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+};
+
+
+/**
+ * @class GrepSyncRef
+ * @brief
+ */
+class GrepSyncRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_SYNC_REF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ enum ErrorCode {
+ NF_FakeErrorREF = GrepError::NF_FakeErrorREF
+ };
+ STATIC_CONST( SignalLength = 6 );
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 part;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ union {
+ Uint32 err;
+ Uint32 errorCode;
+ };
+};
+
+
+/**
+ * @class GrepSyncConf
+ * @brief
+ */
+class GrepSyncConf
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class GrepParticipant;
+
+ friend bool printGREP_SYNC_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 8 );
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 part;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderNodeId;
+ Uint32 firstGCI;
+ Uint32 lastGCI;
+};
+
+/*****************************************************************************
+ * ABORT - remove subscription
+ *****************************************************************************/
+
+/**
+ * @class GrepSubRemoveReq
+ * @brief Between PS and SS
+ */
+class GrepSubRemoveReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_REMOVE_REQ(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+};
+
+
+/**
+ * @class GrepSubRemoveRef
+ * @brief Between PS and SS
+ */
+class GrepSubRemoveRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_REMOVE_REF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 err;
+};
+
+
+/**
+ * @class
+ * @brief
+ */
+class GrepSubRemoveConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printGREP_SUB_REMOVE_CONF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+};
+
+
+/**
+ * @class
+ * @brief
+ */
+class GrepRemoveReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class GrepParticipant;
+
+ friend bool printGREP_REMOVE_REQ(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+};
+
+
+/**
+ * @class
+ * @brief
+ */
+class GrepRemoveRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_REMOVE_REF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ enum ErrorCode {
+ NF_FakeErrorREF = GrepError::NF_FakeErrorREF
+ };
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ union {
+ Uint32 err;
+ Uint32 errorCode;
+ };
+};
+
+
+/**
+ * @class
+ * @brief
+ */
+class GrepRemoveConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_REMOVE_CONF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderNodeId;
+};
+
+
+/*****************************************************************************
+ * WAIT FOR CGP
+ *****************************************************************************/
+
+/**
+ * @class GrepWaitGcpReq
+ * @brief
+ */
+class GrepWaitGcpReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_WAITGCP_REQ(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+ Uint32 senderData;
+ Uint32 gcp;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderNodeId;
+};
+
+/**
+ * @class GrepWaitGcpConf
+ * @brief
+ */
+class GrepWaitGcpConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_WAITGCP_CONF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderNodeId;
+};
+
+
+
+class GrepCreateSubscriptionIdConf {
+ friend class Grep;
+
+ friend bool printGREP_CREATE_SUBSCRIPTION_ID_CONF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+};
+
+
+
+class GrepStartMe {
+ friend class Grep;
+ friend bool printGREP_START_ME(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 1 );
+ Uint32 senderRef;
+};
+
+
+
+
+/**
+ * @class GrepAddSubReq
+ * @brief
+ */
+class GrepAddSubReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class GrepParticipant;
+
+ friend bool printGREP_ADD_SUB_REQ(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 7 );
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriberData;
+ Uint32 subscriberRef;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+};
+
+
+/**
+ * @class GrepAddSubRef
+ * @brief
+ */
+class GrepAddSubRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_CREATE_REF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 err;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+};
+
+
+/**
+ * @class GrepAddSubConf
+ * @brief
+ */
+class GrepAddSubConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+
+ friend class GrepParticipant;
+
+ friend bool printGREP_CREATE_CONF(FILE *,
+ const Uint32 *,
+ Uint32,
+ Uint16);
+public:
+ STATIC_CONST( SignalLength = 1 );
+ Uint32 noOfSub;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/HotSpareRep.hpp b/ndb/include/kernel/signaldata/HotSpareRep.hpp
new file mode 100644
index 00000000000..fb9d338be1b
--- /dev/null
+++ b/ndb/include/kernel/signaldata/HotSpareRep.hpp
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef HOT_SPAREREP_HPP
+#define HOT_SPAREREP_HPP
+
+#include <NodeBitmask.hpp>
+
+/**
+ * This signals is sent by Dbdih to Dbdict
+ */
+class HotSpareRep {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdict;
+
+public:
+ STATIC_CONST( SignalLength = 1 + NodeBitmask::Size );
+private:
+
+ Uint32 noHotSpareNodes;
+ Uint32 theHotSpareNodes[NodeBitmask::Size];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/IndxAttrInfo.hpp b/ndb/include/kernel/signaldata/IndxAttrInfo.hpp
new file mode 100755
index 00000000000..ec5790d84f3
--- /dev/null
+++ b/ndb/include/kernel/signaldata/IndxAttrInfo.hpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef INDX_ATTRINFO_HPP
+#define INDX_ATTRINFO_HPP
+
+#include "SignalData.hpp"
+
+class IndxAttrInfo {
+ /**
+ * Sender(s)
+ */
+ friend class NdbIndexOperation;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtc;
+
+ friend bool printINDXATTRINFO(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( HeaderLength = 3 );
+ STATIC_CONST( DataLength = 22 );
+ STATIC_CONST( MaxSignalLength = HeaderLength + DataLength);
+
+ // Public methods
+public:
+ Uint32* getData() const;
+
+private:
+ Uint32 connectPtr;
+ Uint32 transId[2];
+ Uint32 attrData[DataLength];
+};
+
+inline
+Uint32* IndxAttrInfo::getData() const
+{
+ return (Uint32*)&attrData[0];
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/IndxKeyInfo.hpp b/ndb/include/kernel/signaldata/IndxKeyInfo.hpp
new file mode 100755
index 00000000000..7cd7795ec71
--- /dev/null
+++ b/ndb/include/kernel/signaldata/IndxKeyInfo.hpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef INDX_KEY_INFO_HPP
+#define INDX_KEY_INFO_HPP
+
+#include "SignalData.hpp"
+
+class IndxKeyInfo {
+ /**
+ * Sender(s)
+ */
+ friend class NdbIndexOperation;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+
+ friend bool printINDXKEYINFO(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( HeaderLength = 3 );
+ STATIC_CONST( DataLength = 20 );
+ STATIC_CONST( MaxSignalLength = HeaderLength + DataLength );
+
+ // Public methods
+public:
+ Uint32* getData() const;
+
+private:
+ Uint32 connectPtr;
+ Uint32 transId[2];
+ Uint32 keyData[DataLength];
+};
+
+inline
+Uint32* IndxKeyInfo::getData() const
+{
+ return (Uint32*)&keyData[0];
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/InvalidateNodeLCPConf.hpp b/ndb/include/kernel/signaldata/InvalidateNodeLCPConf.hpp
new file mode 100644
index 00000000000..2497af354ce
--- /dev/null
+++ b/ndb/include/kernel/signaldata/InvalidateNodeLCPConf.hpp
@@ -0,0 +1,41 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef INVALIDATE_NODE_LCP_CONF_HPP
+#define INVALIDATE_NODE_LCP_CONF_HPP
+
+/**
+ * This signal is sent from the non-master DIH to master DIHs
+ *
+ */
+class InvalidateNodeLCPConf {
+
+ /**
+ * Sender/Receiver
+ */
+ friend class Dbdih;
+
+ /**
+ * NodeId of sending node
+ * which is "done"
+ */
+ Uint32 sendingNodeId;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/InvalidateNodeLCPReq.hpp b/ndb/include/kernel/signaldata/InvalidateNodeLCPReq.hpp
new file mode 100644
index 00000000000..e55a58710b4
--- /dev/null
+++ b/ndb/include/kernel/signaldata/InvalidateNodeLCPReq.hpp
@@ -0,0 +1,42 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef INVALIDATE_NODE_LCP_REQ_HPP
+#define INVALIDATE_NODE_LCP_REQ_HPP
+
+/**
+ * This signal is sent from the master DIH to all DIHs
+ * when a node is starting without filesystem.
+ *
+ * All DIHs must then "forgett" that the starting node has
+ * performed LCP
+ *
+ * @see StartPermReq
+ */
+class InvalidateNodeLCPReq {
+
+ /**
+ * Sender/Receiver
+ */
+ friend class Dbdih;
+
+ Uint32 startingNodeId;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/KeyInfo.hpp b/ndb/include/kernel/signaldata/KeyInfo.hpp
new file mode 100644
index 00000000000..b839a2c2035
--- /dev/null
+++ b/ndb/include/kernel/signaldata/KeyInfo.hpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef KEY_INFO_HPP
+#define KEY_INFO_HPP
+
+#include "SignalData.hpp"
+
+class KeyInfo {
+ /**
+ * Sender(s)
+ */
+ friend class DbUtil;
+ friend class NdbOperation;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+
+public:
+ STATIC_CONST( HeaderLength = 3 );
+ STATIC_CONST( DataLength = 20 );
+ STATIC_CONST( MaxSignalLength = HeaderLength + DataLength );
+
+private:
+ Uint32 connectPtr;
+ Uint32 transId[2];
+ Uint32 keyData[DataLength];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/LCP.hpp b/ndb/include/kernel/signaldata/LCP.hpp
new file mode 100644
index 00000000000..7d3fb71ae7e
--- /dev/null
+++ b/ndb/include/kernel/signaldata/LCP.hpp
@@ -0,0 +1,154 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LCP_SIGNAL_DATA_HPP
+#define LCP_SIGNAL_DATA_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+
+class StartLcpReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Sender(s) / Receiver(s)
+ */
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+
+ friend bool printSTART_LCP_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+
+ STATIC_CONST( SignalLength = 2 + 2 * NdbNodeBitmask::Size );
+private:
+ Uint32 senderRef;
+ Uint32 lcpId;
+
+ NdbNodeBitmask participatingDIH;
+ NdbNodeBitmask participatingLQH;
+};
+
+class StartLcpConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Sender(s) / Receiver(s)
+ */
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printSTART_LCP_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+
+ STATIC_CONST( SignalLength = 2 );
+private:
+ Uint32 senderRef;
+ Uint32 lcpId;
+};
+
+/**
+ * This signals is sent by Dbdih to Dblqh
+ * to order checkpointing of a certain
+ * fragment.
+ */
+class LcpFragOrd {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Sender(s) / Receiver(s)
+ */
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+
+ friend bool printLCP_FRAG_ORD(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+private:
+
+ Uint32 tableId;
+ Uint32 fragmentId;
+ Uint32 lcpNo;
+ Uint32 lcpId;
+ Uint32 lastFragmentFlag;
+ Uint32 keepGci;
+};
+
+
+class LcpFragRep {
+ /**
+ * Sender(s) and receiver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ friend bool printLCP_FRAG_REP(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 7 );
+
+private:
+ Uint32 nodeId;
+ Uint32 lcpId;
+ Uint32 lcpNo;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 maxGciCompleted;
+ Uint32 maxGciStarted;
+};
+
+class LcpCompleteRep {
+ /**
+ * Sender(s) and receiver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ friend bool printLCP_COMPLETE_REP(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 nodeId;
+ Uint32 blockNo;
+ Uint32 lcpId;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ListTables.hpp b/ndb/include/kernel/signaldata/ListTables.hpp
new file mode 100644
index 00000000000..7fbfab1294c
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ListTables.hpp
@@ -0,0 +1,166 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LIST_TABLES_HPP
+#define LIST_TABLES_HPP
+
+#include <Bitmask.hpp>
+#include "SignalData.hpp"
+
+/**
+ * It is convenient to pack request/response data per table in one
+ * 32-bit word...
+ */
+class ListTablesData {
+public:
+ static Uint32 getTableId(Uint32 data) {
+ return BitmaskImpl::getField(1, &data, 0, 12);
+ }
+ static void setTableId(Uint32& data, Uint32 val) {
+ BitmaskImpl::setField(1, &data, 0, 12, val);
+ }
+ static Uint32 getTableType(Uint32 data) {
+ return BitmaskImpl::getField(1, &data, 12, 8);
+ }
+ static void setTableType(Uint32& data, Uint32 val) {
+ BitmaskImpl::setField(1, &data, 12, 8, val);
+ }
+ static Uint32 getTableStore(Uint32 data) {
+ return BitmaskImpl::getField(1, &data, 20, 4);
+ }
+ static void setTableStore(Uint32& data, Uint32 val) {
+ BitmaskImpl::setField(1, &data, 20, 4, val);
+ }
+ static Uint32 getTableState(Uint32 data) {
+ return BitmaskImpl::getField(1, &data, 24, 4);
+ }
+ static void setTableState(Uint32& data, Uint32 val) {
+ BitmaskImpl::setField(1, &data, 24, 4, val);
+ }
+ static Uint32 getListNames(Uint32 data) {
+ return BitmaskImpl::getField(1, &data, 28, 1);
+ }
+ static void setListNames(Uint32& data, Uint32 val) {
+ BitmaskImpl::setField(1, &data, 28, 1, val);
+ }
+ static Uint32 getListIndexes(Uint32 data) {
+ return BitmaskImpl::getField(1, &data, 29, 1);
+ }
+ static void setListIndexes(Uint32& data, Uint32 val) {
+ BitmaskImpl::setField(1, &data, 29, 1, val);
+ }
+};
+
+class ListTablesReq {
+ /**
+ * Sender(s)
+ */
+ friend class Backup;
+ friend class Table;
+ friend class Suma;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdict;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+public:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 requestData;
+
+ Uint32 getTableId() {
+ return ListTablesData::getTableId(requestData);
+ }
+ void setTableId(Uint32 val) {
+ ListTablesData::setTableId(requestData, val);
+ }
+ Uint32 getTableType() const {
+ return ListTablesData::getTableType(requestData);
+ }
+ void setTableType(Uint32 val) {
+ ListTablesData::setTableType(requestData, val);
+ }
+ Uint32 getListNames() const {
+ return ListTablesData::getListNames(requestData);
+ }
+ void setListNames(Uint32 val) {
+ ListTablesData::setListNames(requestData, val);
+ }
+ Uint32 getListIndexes() const {
+ return ListTablesData::getListIndexes(requestData);
+ }
+ void setListIndexes(Uint32 val) {
+ ListTablesData::setListIndexes(requestData, val);
+ }
+};
+
+class ListTablesConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+ friend class Table;
+ friend class Suma;
+
+public:
+ /**
+ * Note: last signal is indicated by having length < 25
+ */
+ STATIC_CONST( SignalLength = 25 );
+ STATIC_CONST( HeaderLength = 2 );
+ STATIC_CONST( DataLength = 23 );
+
+public:
+ Uint32 senderData;
+ Uint32 counter;
+ Uint32 tableData[DataLength];
+
+ static Uint32 getTableId(Uint32 data) {
+ return ListTablesData::getTableId(data);
+ }
+ void setTableId(unsigned pos, Uint32 val) {
+ ListTablesData::setTableId(tableData[pos], val);
+ }
+ static Uint32 getTableType(Uint32 data) {
+ return ListTablesData::getTableType(data);
+ }
+ void setTableType(unsigned pos, Uint32 val) {
+ ListTablesData::setTableType(tableData[pos], val);
+ }
+ static Uint32 getTableStore(Uint32 data) {
+ return ListTablesData::getTableStore(data);
+ }
+ void setTableStore(unsigned pos, Uint32 val) {
+ ListTablesData::setTableStore(tableData[pos], val);
+ }
+ static Uint32 getTableState(Uint32 data) {
+ return ListTablesData::getTableState(data);
+ }
+ void setTableState(unsigned pos, Uint32 val) {
+ ListTablesData::setTableState(tableData[pos], val);
+ }
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/LqhFrag.hpp b/ndb/include/kernel/signaldata/LqhFrag.hpp
new file mode 100644
index 00000000000..116e9c01ca0
--- /dev/null
+++ b/ndb/include/kernel/signaldata/LqhFrag.hpp
@@ -0,0 +1,252 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LQH_FRAG_HPP
+#define LQH_FRAG_HPP
+
+#include "SignalData.hpp"
+
+class AddFragReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printADD_FRAG_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 9 );
+
+ enum RequestInfo {
+ CreateInRunning = 0x8000000,
+ TemporaryTable = 0x00000010
+ };
+private:
+ Uint32 dihPtr;
+ Uint32 senderData; // The same data as sent in DIADDTABREQ
+ Uint32 fragmentId;
+ Uint32 requestInfo;
+ Uint32 tableId;
+ Uint32 nextLCP;
+ Uint32 nodeId;
+ Uint32 totalFragments;
+ Uint32 startGci;
+};
+
+class AddFragRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printADD_FRAG_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ Uint32 dihPtr;
+};
+
+class AddFragConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printADD_FRAG_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 dihPtr;
+ Uint32 fragId;
+};
+
+class LqhFragReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+
+ friend bool printLQH_FRAG_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 25 );
+
+ enum RequestInfo {
+ CreateInRunning = 0x8000000,
+ TemporaryTable = 0x00000010
+ };
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 fragmentId;
+ Uint32 requestInfo;
+ Uint32 tableId;
+ Uint32 localKeyLength;
+ Uint32 maxLoadFactor;
+ Uint32 minLoadFactor;
+ Uint32 kValue;
+ Uint32 lh3DistrBits;
+ Uint32 lh3PageBits;
+ Uint32 noOfAttributes;
+ Uint32 noOfNullAttributes;
+ Uint32 noOfPagesToPreAllocate;
+ Uint32 schemaVersion;
+ Uint32 keyLength;
+ Uint32 nextLCP;
+ Uint32 noOfKeyAttr;
+ Uint32 noOfNewAttr;
+ Uint32 checksumIndicator;
+ Uint32 noOfAttributeGroups;
+ Uint32 GCPIndicator;
+ Uint32 startGci;
+ Uint32 tableType; // DictTabInfo::TableType
+ Uint32 primaryTableId; // table of index or RNIL
+};
+
+class LqhFragConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printLQH_FRAG_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderData;
+ Uint32 lqhFragPtr;
+};
+
+class LqhFragRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printLQH_FRAG_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderData;
+ Uint32 errorCode;
+};
+
+class LqhAddAttrReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+
+ friend bool printLQH_ADD_ATTR_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( HeaderLength = 4 );
+ STATIC_CONST( EntryLength = 3 );
+ STATIC_CONST( MAX_ATTRIBUTES = 6 );
+ struct Entry {
+ Uint32 attrId; // for index, includes primary attr id << 16
+ Uint32 attrDescriptor; // 2 words type info
+ Uint32 extTypeInfo;
+ };
+private:
+ Uint32 lqhFragPtr;
+ Uint32 noOfAttributes;
+ Uint32 senderData;
+ Uint32 senderAttrPtr;
+ Entry attributes[MAX_ATTRIBUTES];
+};
+
+class LqhAddAttrRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printLQH_ADD_ATTR_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderData;
+ Uint32 errorCode;
+};
+
+class LqhAddAttrConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printLQH_ADD_ATTR_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderAttrPtr;
+ Uint32 fragId;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/LqhKey.hpp b/ndb/include/kernel/signaldata/LqhKey.hpp
new file mode 100644
index 00000000000..e937180e3f7
--- /dev/null
+++ b/ndb/include/kernel/signaldata/LqhKey.hpp
@@ -0,0 +1,534 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LQH_KEY_H
+#define LQH_KEY_H
+
+#include "SignalData.hpp"
+
+class LqhKeyReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Dblqh; // Reciver
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * For printing
+ */
+ friend bool printLQHKEYREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( FixedSignalLength = 11 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+//-------------------------------------------------------------
+// Unconditional part. First 10 words
+//-------------------------------------------------------------
+ UintR clientConnectPtr; // DATA 0
+ UintR attrLen; // DATA 1
+ UintR hashValue; // DATA 2
+ UintR requestInfo; // DATA 3
+ UintR tcBlockref; // DATA 4
+ UintR tableSchemaVersion; // DATA 5
+ UintR fragmentData; // DATA 6
+ UintR transId1; // DATA 7
+ UintR transId2; // DATA 8
+ UintR savePointId; // DATA 9
+ union {
+ /**
+ * When sent from TC -> LQH this variable contains scanInfo
+ * When send from LQH -> LQH this variable contains noFiredTriggers
+ */
+ UintR noFiredTriggers; // DATA 10
+ Uint32 scanInfo; // DATA 10
+ };
+
+//-------------------------------------------------------------
+// Variable sized key part. Those will be placed to
+// pack the signal in an appropriate manner.
+//-------------------------------------------------------------
+ UintR variableData[10]; // DATA 11 - 21
+
+ static UintR getAttrLen(const UintR & scanInfoAttrLen);
+ static UintR getScanTakeOverFlag(const UintR & scanInfoAttrLen);
+ static UintR getStoredProcFlag(const UintR & scanData);
+ static UintR getDistributionKey(const UintR & scanData);
+
+ static UintR getTableId(const UintR & tableSchemaVersion);
+ static UintR getSchemaVersion(const UintR & tableSchemaVersion);
+
+ static UintR getFragmentId(const UintR & fragmentData);
+ static UintR getNextReplicaNodeId(const UintR & fragmentData);
+
+ static Uint8 getLockType(const UintR & requestInfo);
+ static Uint8 getDirtyFlag(const UintR & requestInfo);
+ static Uint8 getInterpretedFlag(const UintR & requestInfo);
+ static Uint8 getSimpleFlag(const UintR & requestInfo);
+ static Uint8 getOperation(const UintR & requestInfo);
+ static Uint8 getSeqNoReplica(const UintR & requestInfo);
+ static Uint8 getLastReplicaNo(const UintR & requestInfo);
+ static Uint8 getAIInLqhKeyReq(const UintR & requestInfo);
+ static UintR getKeyLen(const UintR & requestInfo);
+ static UintR getSameClientAndTcFlag(const UintR & requestInfo);
+ static UintR getReturnedReadLenAIFlag(const UintR & requestInfo);
+ static UintR getApplicationAddressFlag(const UintR & requestInfo);
+ static UintR getMarkerFlag(const UintR & requestInfo);
+
+ /**
+ * Setters
+ */
+
+ static void setAttrLen(UintR & scanInfoAttrLen, UintR val);
+ static void setScanTakeOverFlag(UintR & scanInfoAttrLen, UintR val);
+ static void setStoredProcFlag(UintR & scanData, UintR val);
+ static void setDistributionKey(UintR & scanData, UintR val);
+
+ static void setTableId(UintR & tableSchemaVersion, UintR val);
+ static void setSchemaVersion(UintR & tableSchemaVersion, UintR val);
+
+ static void setFragmentId(UintR & fragmentData, UintR val);
+ static void setNextReplicaNodeId(UintR & fragmentData, UintR val);
+
+ static void setLockType(UintR & requestInfo, UintR val);
+ static void setDirtyFlag(UintR & requestInfo, UintR val);
+ static void setInterpretedFlag(UintR & requestInfo, UintR val);
+ static void setSimpleFlag(UintR & requestInfo, UintR val);
+ static void setOperation(UintR & requestInfo, UintR val);
+ static void setSeqNoReplica(UintR & requestInfo, UintR val);
+ static void setLastReplicaNo(UintR & requestInfo, UintR val);
+ static void setAIInLqhKeyReq(UintR & requestInfo, UintR val);
+ static void setKeyLen(UintR & requestInfo, UintR val);
+ static void setSameClientAndTcFlag(UintR & requestInfo, UintR val);
+ static void setReturnedReadLenAIFlag(UintR & requestInfo, UintR val);
+ static void setApplicationAddressFlag(UintR & requestInfo, UintR val);
+ static void setMarkerFlag(UintR & requestInfo, UintR val);
+};
+
+/**
+ * Request Info
+ *
+ * k = Key len - 10 Bits (0-9) max 1023
+ * l = Last Replica No - 2 Bits -> Max 3 (10-11)
+ * t = Lock type - 3 Bits -> Max 7 (12-14)
+ * p = Application Addr. Ind - 1 Bit (15)
+ * d = Dirty indicator - 1 Bit (16)
+ * i = Interpreted indicator - 1 Bit (17)
+ * s = Simple indicator - 1 Bit (18)
+ * o = Operation - 3 Bits (19-21)
+ * r = Sequence replica - 2 Bits (22-23)
+ * a = Attr Info in LQHKEYREQ - 3 Bits (24-26)
+ * c = Same client and tc - 1 Bit (27)
+ * u = Read Len Return Ind - 1 Bit (28)
+ * m = Commit ack marker - 1 Bit (29)
+ * - = Unused - 2 Bits (30-31)
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * kkkkkkkkkklltttpdisooorraaacum--
+ */
+
+#define RI_KEYLEN_SHIFT (0)
+#define RI_KEYLEN_MASK (1023)
+#define RI_LAST_REPL_SHIFT (10)
+#define RI_LAST_REPL_MASK (3)
+#define RI_LOCK_TYPE_SHIFT (12)
+#define RI_LOCK_TYPE_MASK (7)
+#define RI_APPL_ADDR_SHIFT (15)
+#define RI_DIRTY_SHIFT (16)
+#define RI_INTERPRETED_SHIFT (17)
+#define RI_SIMPLE_SHIFT (18)
+#define RI_OPERATION_SHIFT (19)
+#define RI_OPERATION_MASK (7)
+#define RI_SEQ_REPLICA_SHIFT (22)
+#define RI_SEQ_REPLICA_MASK (3)
+#define RI_AI_IN_THIS_SHIFT (24)
+#define RI_AI_IN_THIS_MASK (7)
+#define RI_SAME_CLIENT_SHIFT (27)
+#define RI_RETURN_AI_SHIFT (28)
+#define RI_MARKER_SHIFT (29)
+
+/**
+ * Scan Info
+ *
+ * a = Attr Len - 16 Bits -> max 65535 (0-15)
+ * p = Stored Procedure Ind - 1 Bit (16)
+ * d = Distribution key - 8 Bit -> max 255 (17-24)
+ * t = Scan take over indicator - 1 Bit (25)
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * aaaaaaaaaaaaaaaapddddddddt
+ */
+
+#define SI_ATTR_LEN_MASK (65535)
+#define SI_ATTR_LEN_SHIFT (0)
+#define SI_STORED_PROC_SHIFT (16)
+#define SI_DISTR_KEY_MASK (255)
+#define SI_DISTR_KEY_SHIFT (17)
+#define SI_SCAN_TO_SHIFT (25)
+#define SI_SCAN_INFO_MASK (63)
+#define SI_SCAN_INFO_SHIFT (26)
+
+inline
+UintR
+LqhKeyReq::getAttrLen(const UintR & scanData)
+{
+ return (scanData >> SI_ATTR_LEN_SHIFT) & SI_ATTR_LEN_MASK;
+}
+
+inline
+Uint32
+LqhKeyReq::getScanTakeOverFlag(const UintR & scanData)
+{
+ return (scanData >> SI_SCAN_TO_SHIFT) & 1;
+}
+
+inline
+UintR
+LqhKeyReq::getStoredProcFlag(const UintR & scanData){
+ return (scanData >> SI_STORED_PROC_SHIFT) & 1;
+}
+
+inline
+UintR
+LqhKeyReq::getDistributionKey(const UintR & scanData){
+ return (scanData >> SI_DISTR_KEY_SHIFT) & SI_DISTR_KEY_MASK;
+}
+
+inline
+UintR LqhKeyReq::getTableId(const UintR & tableSchemaVersion)
+{
+ return tableSchemaVersion & 0xFFFF;
+}
+
+inline
+UintR LqhKeyReq::getSchemaVersion(const UintR & tableSchemaVersion)
+{
+ return tableSchemaVersion >> 16;
+}
+
+inline
+UintR LqhKeyReq::getFragmentId(const UintR & fragmentData)
+{
+ return fragmentData & 0xFFFF;
+}
+
+inline
+UintR LqhKeyReq::getNextReplicaNodeId(const UintR & fragmentData)
+{
+ return fragmentData >> 16;
+}
+
+inline
+Uint8 LqhKeyReq::getLastReplicaNo(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_LAST_REPL_SHIFT) & RI_LAST_REPL_MASK;
+}
+
+inline
+Uint8 LqhKeyReq::getLockType(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_LOCK_TYPE_SHIFT) & RI_LOCK_TYPE_MASK;
+}
+
+inline
+Uint8 LqhKeyReq::getDirtyFlag(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_DIRTY_SHIFT) & 1;
+}
+
+inline
+Uint8 LqhKeyReq::getInterpretedFlag(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_INTERPRETED_SHIFT) & 1;
+}
+
+inline
+Uint8 LqhKeyReq::getSimpleFlag(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_SIMPLE_SHIFT) & 1;
+}
+
+inline
+Uint8 LqhKeyReq::getOperation(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_OPERATION_SHIFT) & RI_OPERATION_MASK;
+}
+
+inline
+Uint8 LqhKeyReq::getSeqNoReplica(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_SEQ_REPLICA_SHIFT) & RI_SEQ_REPLICA_MASK;
+}
+
+
+inline
+Uint8 LqhKeyReq::getAIInLqhKeyReq(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_AI_IN_THIS_SHIFT) & RI_AI_IN_THIS_MASK;
+}
+
+inline
+UintR LqhKeyReq::getKeyLen(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_KEYLEN_SHIFT) & RI_KEYLEN_MASK;
+}
+
+inline
+UintR
+LqhKeyReq::getSameClientAndTcFlag(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_SAME_CLIENT_SHIFT) & 1;
+}
+
+inline
+UintR LqhKeyReq::getReturnedReadLenAIFlag(const UintR & requestInfo)
+{
+ return (requestInfo >> RI_RETURN_AI_SHIFT) & 1;
+}
+
+inline
+UintR
+LqhKeyReq::getApplicationAddressFlag(const UintR & requestInfo){
+ return (requestInfo >> RI_APPL_ADDR_SHIFT) & 1;
+}
+
+inline
+void
+LqhKeyReq::setAttrLen(UintR & scanInfoAttrLen, UintR val){
+ ASSERT_MAX(val, SI_ATTR_LEN_MASK, "LqhKeyReq::setAttrLen");
+ scanInfoAttrLen |= (val << SI_ATTR_LEN_SHIFT);
+}
+
+
+inline
+void
+LqhKeyReq::setScanTakeOverFlag(UintR & scanInfoAttrLen, UintR val){
+ ASSERT_BOOL(val, "LqhKeyReq::setScanTakeOverFlag");
+ scanInfoAttrLen |= (val << SI_SCAN_TO_SHIFT);
+}
+inline
+void
+
+LqhKeyReq::setStoredProcFlag(UintR & scanData, UintR val){
+ ASSERT_BOOL(val, "LqhKeyReq::setStoredProcFlag");
+ scanData |= (val << SI_STORED_PROC_SHIFT);
+}
+
+inline
+void
+
+LqhKeyReq::setDistributionKey(UintR & scanData, UintR val){
+ ASSERT_MAX(val, SI_DISTR_KEY_MASK, "LqhKeyReq::setDistributionKey");
+ scanData |= (val << SI_DISTR_KEY_SHIFT);
+}
+
+#if 0
+inline
+void
+
+LqhKeyReq::setTableId(UintR & tableSchemaVersion, UintR val){
+
+}
+inline
+void
+LqhKeyReq::setSchemaVersion(UintR & tableSchemaVersion, UintR val);
+
+inline
+void
+LqhKeyReq::setFragmentId(UintR & fragmentData, UintR val);
+
+inline
+void
+LqhKeyReq::setNextReplicaNodeId(UintR & fragmentData, UintR val);
+#endif
+
+inline
+void
+LqhKeyReq::setLockType(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, RI_LOCK_TYPE_MASK, "LqhKeyReq::setLockType");
+ requestInfo |= (val << RI_LOCK_TYPE_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setDirtyFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhKeyReq::setDirtyFlag");
+ requestInfo |= (val << RI_DIRTY_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setInterpretedFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhKeyReq::setInterpretedFlag");
+ requestInfo |= (val << RI_INTERPRETED_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setSimpleFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhKeyReq::setSimpleFlag");
+ requestInfo |= (val << RI_SIMPLE_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setOperation(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, RI_OPERATION_MASK, "LqhKeyReq::setOperation");
+ requestInfo |= (val << RI_OPERATION_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setSeqNoReplica(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, RI_SEQ_REPLICA_MASK, "LqhKeyReq::setSeqNoReplica");
+ requestInfo |= (val << RI_SEQ_REPLICA_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setLastReplicaNo(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, RI_LAST_REPL_MASK, "LqhKeyReq::setLastReplicaNo");
+ requestInfo |= (val << RI_LAST_REPL_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setAIInLqhKeyReq(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, RI_AI_IN_THIS_MASK, "LqhKeyReq::setAIInLqhKeyReq");
+ requestInfo |= (val << RI_AI_IN_THIS_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setKeyLen(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, RI_KEYLEN_MASK, "LqhKeyReq::setKeyLen");
+ requestInfo |= (val << RI_KEYLEN_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setSameClientAndTcFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhKeyReq::setSameClientAndTcFlag");
+ requestInfo |= (val << RI_SAME_CLIENT_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setReturnedReadLenAIFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhKeyReq::setReturnedReadLenAIFlag");
+ requestInfo |= (val << RI_RETURN_AI_SHIFT);
+}
+
+inline
+void
+LqhKeyReq::setApplicationAddressFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhKeyReq::setApplicationAddressFlag");
+ requestInfo |= (val << RI_APPL_ADDR_SHIFT);
+}
+
+/**** */
+
+inline
+void
+LqhKeyReq::setMarkerFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhKeyReq::setMarkerFlag");
+ requestInfo |= (val << RI_MARKER_SHIFT);
+}
+
+inline
+UintR
+LqhKeyReq::getMarkerFlag(const UintR & requestInfo){
+ return (requestInfo >> RI_MARKER_SHIFT) & 1;
+}
+
+class LqhKeyConf {
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ // Sent in a packed signal
+ friend class PackedSignal;
+ /**
+ * For printing
+ */
+ friend bool printPACKED_SIGNAL(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+ friend bool printLQHKEYCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 7 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ Uint32 connectPtr;
+ Uint32 opPtr;
+ Uint32 userRef;
+ Uint32 readLen;
+ Uint32 transId1;
+ Uint32 transId2;
+ Uint32 noFiredTriggers;
+};
+
+class LqhKeyRef {
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * For printing
+ */
+ friend bool printLQHKEYREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ Uint32 userRef;
+ Uint32 connectPtr;
+ Uint32 errorCode;
+ Uint32 transId1;
+ Uint32 transId2;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/LqhSizeAltReq.hpp b/ndb/include/kernel/signaldata/LqhSizeAltReq.hpp
new file mode 100644
index 00000000000..e47ce39897a
--- /dev/null
+++ b/ndb/include/kernel/signaldata/LqhSizeAltReq.hpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LQH_SIZE_ALT_REQ_H
+#define LQH_SIZE_ALT_REQ_H
+
+
+
+#include "SignalData.hpp"
+
+class LqhSizeAltReq {
+ /**
+ * Sender(s)
+ */
+ friend class ClusterConfiguration;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dblqh;
+private:
+ /**
+ * Indexes in theData
+ */
+ STATIC_CONST( IND_BLOCK_REF = 0);
+ STATIC_CONST( IND_FRAG = 1);
+ STATIC_CONST( IND_CONNECT = 2);
+ STATIC_CONST( IND_TABLE = 3);
+ STATIC_CONST( IND_TC_CONNECT = 4);
+ STATIC_CONST( IND_REPLICAS = 5);
+ STATIC_CONST( IND_LOG_FILES = 6);
+ STATIC_CONST( IND_SCAN = 7);
+
+ /**
+ * Use the index definitions to use the signal data
+ */
+ UintR theData[8];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/LqhTransConf.hpp b/ndb/include/kernel/signaldata/LqhTransConf.hpp
new file mode 100644
index 00000000000..f62dfd07f51
--- /dev/null
+++ b/ndb/include/kernel/signaldata/LqhTransConf.hpp
@@ -0,0 +1,218 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LQH_TRANS_CONF_H
+#define LQH_TRANS_CONF_H
+
+#include "SignalData.hpp"
+
+/**
+ * This signal is sent as response to a LQH_TRANSREQ
+ * which is sent as by a take-over TC
+ */
+class LqhTransConf {
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ friend bool printLQH_TRANSCONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 15 );
+private:
+
+ /**
+ * This type describes the state of the operation returned in this signal
+ */
+ enum OperationStatus {
+ InvalidStatus = 0, /**< This status should never be sent in a signal
+ it is only used for initializing variables so that
+ you can easily later check if they have changed */
+ LastTransConf = 4, /**< This status indicates that LQH has finished the scan
+ of operations belonging to the died TC.
+ Data 0 - 2 is valid */
+
+ Prepared = 2,
+ Committed = 3,
+ Aborted = 1,
+ Marker = 5 /**< This means that the only thing left is a marker,
+ Data 0 - 6 is valid */
+ };
+
+ /**
+ * DATA VARIABLES
+ */
+ Uint32 tcRef; // 0
+ Uint32 lqhNodeId; // 1
+ Uint32 operationStatus; // 2 See enum OperationStatus
+ Uint32 transId1; // 3
+ Uint32 transId2; // 4
+ Uint32 apiRef; // 5
+ Uint32 apiOpRec; // 6
+ Uint32 lqhConnectPtr;
+ Uint32 oldTcOpRec;
+ Uint32 requestInfo;
+ Uint32 gci;
+ Uint32 nextNodeId1;
+ Uint32 nextNodeId2;
+ Uint32 nextNodeId3;
+ Uint32 tableId;
+
+ /**
+ * Getters
+ */
+ static Uint32 getReplicaNo(Uint32 & requestInfo);
+ static Uint32 getReplicaType(Uint32 & requestInfo);
+ static Uint32 getLastReplicaNo(Uint32 & requestInfo);
+ static Uint32 getSimpleFlag(Uint32 & requestInfo);
+ static Uint32 getDirtyFlag(Uint32 & requestInfo);
+ static Uint32 getOperation(Uint32 & requestInfo);
+ static Uint32 getMarkerFlag(Uint32 & requestInfo);
+
+ static void setReplicaNo(UintR & requestInfo, UintR val);
+ static void setReplicaType(UintR & requestInfo, UintR val);
+ static void setLastReplicaNo(UintR & requestInfo, UintR val);
+ static void setSimpleFlag(UintR & requestInfo, UintR val);
+ static void setDirtyFlag(UintR & requestInfo, UintR val);
+ static void setOperation(UintR & requestInfo, UintR val);
+ static void setMarkerFlag(Uint32 & requestInfo, Uint32 val);
+};
+
+/**
+ * Request Info
+ *
+ * t = replica type - 2 Bits (0-1)
+ * r = Replica No - 2 Bits (2-3)
+ * l = Last Replica No - 2 Bits (4-5)
+ * s = Simple - 1 Bits (6)
+ * d = Dirty - 1 Bit (7)
+ * o = Operation - 3 Bit (8-9)
+ * m = Marker present - 1 Bit (10)
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * ttrrllsdooom
+ */
+#define LTC_REPLICA_TYPE_SHIFT (0)
+#define LTC_REPLICA_TYPE_MASK (3)
+#define LTC_REPLICA_NO_SHIFT (2)
+#define LTC_REPLICA_NO_MASK (3)
+#define LTC_LAST_REPLICA_SHIFT (4)
+#define LTC_LAST_REPLICA_MASK (3)
+#define LTC_SIMPLE_SHIFT (6)
+#define LTC_DIRTY_SHIFT (7)
+#define LTC_OPERATION_SHIFT (8)
+#define LTC_OPERATION_MASK (7)
+#define LTC_MARKER_SHIFT (10)
+
+inline
+Uint32
+LqhTransConf::getReplicaType(Uint32 & requestInfo){
+ return (requestInfo >> LTC_REPLICA_TYPE_SHIFT) & LTC_REPLICA_TYPE_MASK;
+}
+
+inline
+Uint32
+LqhTransConf::getReplicaNo(Uint32 & requestInfo){
+ return (requestInfo >> LTC_REPLICA_NO_SHIFT) & LTC_REPLICA_NO_MASK;
+}
+
+inline
+Uint32
+LqhTransConf::getLastReplicaNo(Uint32 & requestInfo){
+ return (requestInfo >> LTC_LAST_REPLICA_SHIFT) & LTC_LAST_REPLICA_MASK;
+}
+
+inline
+Uint32
+LqhTransConf::getSimpleFlag(Uint32 & requestInfo){
+ return (requestInfo >> LTC_SIMPLE_SHIFT) & 1;
+}
+
+inline
+Uint32
+LqhTransConf::getDirtyFlag(Uint32 & requestInfo){
+ return (requestInfo >> LTC_DIRTY_SHIFT) & 1;
+}
+
+inline
+Uint32
+LqhTransConf::getOperation(Uint32 & requestInfo){
+ return (requestInfo >> LTC_OPERATION_SHIFT) & LTC_OPERATION_MASK;
+}
+
+inline
+Uint32
+LqhTransConf::getMarkerFlag(Uint32 & requestInfo){
+ return (requestInfo >> LTC_MARKER_SHIFT) & 1;
+}
+
+
+inline
+void
+LqhTransConf::setReplicaNo(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, LTC_REPLICA_NO_MASK, "LqhTransConf::setReplicaNo");
+ requestInfo |= (val << LTC_REPLICA_NO_SHIFT);
+}
+
+inline
+void
+LqhTransConf::setReplicaType(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, LTC_REPLICA_TYPE_MASK, "LqhTransConf::setReplicaType");
+ requestInfo |= (val << LTC_REPLICA_TYPE_SHIFT);
+}
+
+inline
+void
+LqhTransConf::setLastReplicaNo(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, LTC_LAST_REPLICA_MASK, "LqhTransConf::setLastReplicaNo");
+ requestInfo |= (val << LTC_LAST_REPLICA_SHIFT);
+}
+
+inline
+void
+LqhTransConf::setSimpleFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhTransConf::setSimpleFlag");
+ requestInfo |= (val << LTC_SIMPLE_SHIFT);
+}
+
+inline
+void
+LqhTransConf::setDirtyFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhTransConf::setDirtyFlag");
+ requestInfo |= (val << LTC_DIRTY_SHIFT);
+}
+
+inline
+void
+LqhTransConf::setOperation(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, LTC_OPERATION_MASK, "LqhTransConf::setOperation");
+ requestInfo |= (val << LTC_OPERATION_SHIFT);
+}
+
+inline
+void
+LqhTransConf::setMarkerFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "LqhTransConf::setMarkerFlag");
+ requestInfo |= (val << LTC_MARKER_SHIFT);
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ManagementServer.hpp b/ndb/include/kernel/signaldata/ManagementServer.hpp
new file mode 100644
index 00000000000..ce14e30c81d
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ManagementServer.hpp
@@ -0,0 +1,87 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MANAGEMENTSERVER_HPP
+#define MANAGEMENTSERVER_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * Request to lock configuration
+ */
+class MgmLockConfigReq {
+ friend class MgmtSrvr;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ Uint32 newConfigGeneration;
+};
+
+/**
+ * Confirm configuration lock
+ */
+class MgmLockConfigRep {
+ friend class MgmtSrvr;
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+ /* Error codes */
+ enum ErrorCode {
+ OK,
+ UNKNOWN_ERROR,
+ GENERATION_MISMATCH,
+ ALREADY_LOCKED
+ };
+
+private:
+ Uint32 errorCode;
+};
+
+/**
+ * Unlock configuration
+ */
+class MgmUnlockConfigReq {
+ friend class MgmtSrvr;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ Uint32 commitConfig;
+};
+
+/**
+ * Confirm config unlock
+ */
+class MgmUnlockConfigRep {
+ friend class MgmtSrvr;
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+ /* Error codes */
+ enum ErrorCode {
+ OK,
+ UNKNOWN_ERROR,
+ NOT_LOCKED
+ };
+
+private:
+ Uint32 errorCode;
+};
+
+#endif /* !MANAGEMENTSERVER_HPP */
diff --git a/ndb/include/kernel/signaldata/MasterGCP.hpp b/ndb/include/kernel/signaldata/MasterGCP.hpp
new file mode 100644
index 00000000000..ebe6857a107
--- /dev/null
+++ b/ndb/include/kernel/signaldata/MasterGCP.hpp
@@ -0,0 +1,84 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MASTER_GCP_HPP
+#define MASTER_GCP_HPP
+
+#include <NodeBitmask.hpp>
+
+/**
+ *
+ */
+class MasterGCPConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 8 + NdbNodeBitmask::Size );
+
+ enum State {
+ GCP_READY = 0,
+ GCP_PREPARE_RECEIVED = 1,
+ GCP_COMMIT_RECEIVED = 2,
+ GCP_TC_FINISHED = 3
+ };
+private:
+ /**
+ * Data replied
+ */
+ Uint32 gcpState;
+ Uint32 senderNodeId;
+ Uint32 failedNodeId;
+ Uint32 newGCP;
+ Uint32 latestLCP;
+ Uint32 oldestRestorableGCI;
+ Uint32 keepGCI;
+ Uint32 lcpActive[NdbNodeBitmask::Size];
+};
+/**
+ *
+ */
+class MasterGCPReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+ Uint32 masterRef;
+ Uint32 failedNodeId;
+};
+
+/**
+ *
+ */
+class MasterGCPRef {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+ Uint32 senderNodeId;
+ Uint32 failedNodeId;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/MasterLCP.hpp b/ndb/include/kernel/signaldata/MasterLCP.hpp
new file mode 100644
index 00000000000..bf84ac73309
--- /dev/null
+++ b/ndb/include/kernel/signaldata/MasterLCP.hpp
@@ -0,0 +1,86 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MASTER_LCP_HPP
+#define MASTER_LCP_HPP
+
+#include <NdbOut.hpp>
+#include "SignalData.hpp"
+
+/**
+ *
+ */
+class MasterLCPConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printMASTER_LCP_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ enum State {
+ LCP_STATUS_IDLE = 0,
+ LCP_STATUS_ACTIVE = 2,
+ LCP_TAB_COMPLETED = 8,
+ LCP_TAB_SAVED = 9
+ };
+
+ friend NdbOut& operator<<(NdbOut&, const State&);
+
+private:
+ /**
+ * Data replied
+ */
+ Uint32 senderNodeId;
+ Uint32 lcpState;
+ Uint32 failedNodeId;
+};
+/**
+ *
+ */
+class MasterLCPReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printMASTER_LCP_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+ Uint32 masterRef;
+ Uint32 failedNodeId;
+};
+
+class MasterLCPRef {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printMASTER_LCP_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+ /**
+ * Data replied
+ */
+ Uint32 senderNodeId;
+ Uint32 failedNodeId;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/NFCompleteRep.hpp b/ndb/include/kernel/signaldata/NFCompleteRep.hpp
new file mode 100644
index 00000000000..c8bde705a86
--- /dev/null
+++ b/ndb/include/kernel/signaldata/NFCompleteRep.hpp
@@ -0,0 +1,80 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NF_COMPLETE_REP_HPP
+#define NF_COMPLETE_REP_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * NFCompleteRep - Node Fail Complete Report
+ *
+ * This signal is sent by a block(or a node)
+ * when it has finished cleaning up after a node failure.
+ *
+ * It's also sent from Qmgr to the clusterMgr in API
+ * to tell the API that it can now abort all transactions still waiting for response
+ * from the failed NDB node
+ *
+ */
+class NFCompleteRep {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+ friend class Dblqh;
+ friend class Dbtc;
+ friend class Qmgr;
+
+ /**
+ * Sender/Reciver
+ */
+ friend class Dbdih;
+ friend class ClusterMgr;
+
+ friend bool printNF_COMPLETE_REP(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+
+ /**
+ * Which block has completed...
+ *
+ * NOTE: 0 means the node has completed
+ */
+ Uint32 blockNo;
+
+ /**
+ * Which node has completed...
+ */
+ Uint32 nodeId;
+
+ /**
+ * Which node has failed
+ */
+ Uint32 failedNodeId;
+
+ /**
+ * Is this the original message or a delayed variant.
+ */
+ Uint32 unused; // originalMessage
+
+ Uint32 from;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/NdbSttor.hpp b/ndb/include/kernel/signaldata/NdbSttor.hpp
new file mode 100644
index 00000000000..edd93ef96a8
--- /dev/null
+++ b/ndb/include/kernel/signaldata/NdbSttor.hpp
@@ -0,0 +1,85 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_STTOR_HPP
+#define NDB_STTOR_HPP
+
+#include "SignalData.hpp"
+
+class NdbSttor {
+ /**
+ * Sender(s)
+ */
+ friend class NdbCntr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+ friend class Dbdict;
+ friend class Dbdih;
+ friend class Dblqh;
+ friend class Dbtc;
+ friend class ClusterMgr;
+ friend class Trix;
+ friend class Backup;
+ friend class Suma;
+ friend class Grep;
+
+ friend bool printNDB_STTOR(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ STATIC_CONST( DataLength = 16 );
+private:
+
+ Uint32 senderRef;
+ Uint32 nodeId;
+ Uint32 internalStartPhase;
+ Uint32 typeOfStart;
+ Uint32 masterNodeId;
+ Uint32 unused;
+ Uint32 config[DataLength];
+};
+
+class NdbSttorry {
+ /**
+ * Receiver(s)
+ */
+ friend class NdbCntr;
+
+ /**
+ * Sender(s)
+ */
+ friend class Ndbcntr;
+ friend class Dbdict;
+ friend class Dbdih;
+ friend class Dblqh;
+ friend class Dbtc;
+ friend class ClusterMgr;
+ friend class Trix;
+ friend class Backup;
+ friend class Suma;
+ friend class Grep;
+
+ friend bool printNDB_STTORRY(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 1 );
+private:
+
+ Uint32 senderRef;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/NdbfsContinueB.hpp b/ndb/include/kernel/signaldata/NdbfsContinueB.hpp
new file mode 100644
index 00000000000..2d569be721f
--- /dev/null
+++ b/ndb/include/kernel/signaldata/NdbfsContinueB.hpp
@@ -0,0 +1,35 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBFS_CONTINUEB_H
+#define NDBFS_CONTINUEB_H
+
+#include "SignalData.hpp"
+
+class NdbfsContinueB {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Ndbfs;
+ friend bool printCONTINUEB_NDBFS(FILE * output, const Uint32 * theData, Uint32 len);
+private:
+ enum {
+ ZSCAN_MEMORYCHANNEL_10MS_DELAY = 0,
+ ZSCAN_MEMORYCHANNEL_NO_DELAY = 1
+ };
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/NextScan.hpp b/ndb/include/kernel/signaldata/NextScan.hpp
new file mode 100644
index 00000000000..3a1882f94e8
--- /dev/null
+++ b/ndb/include/kernel/signaldata/NextScan.hpp
@@ -0,0 +1,67 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NEXT_SCAN_HPP
+#define NEXT_SCAN_HPP
+
+#include "SignalData.hpp"
+
+class NextScanReq {
+ friend class Dblqh;
+ friend class Dbacc;
+ friend class Dbtux;
+public:
+ // two sets of defs picked from lqh/acc
+ enum ScanFlag {
+ ZSCAN_NEXT = 1,
+ ZSCAN_NEXT_COMMIT = 2,
+ ZSCAN_COMMIT = 3, // new
+ ZSCAN_CLOSE = 6,
+ ZSCAN_NEXT_ABORT = 12
+ };
+ enum CopyFlag {
+ todo_ZCOPY_NEXT = 1,
+ todo_ZCOPY_NEXT_COMMIT = 2,
+ todo_ZCOPY_COMMIT = 3,
+ todo_ZCOPY_REPEAT = 4,
+ todo_ZCOPY_ABORT = 5,
+ todo_ZCOPY_CLOSE = 6
+ };
+ STATIC_CONST( SignalLength = 3 );
+private:
+ Uint32 accPtr; // scan record in ACC/TUX
+ Uint32 accOperationPtr;
+ Uint32 scanFlag;
+};
+
+class NextScanConf {
+ friend class Dbacc;
+ friend class Dbtux;
+ friend class Dblqh;
+public:
+ // length is less if no keyinfo or no next result
+ STATIC_CONST( SignalLength = 11 );
+private:
+ Uint32 scanPtr; // scan record in LQH
+ Uint32 accOperationPtr;
+ Uint32 fragId;
+ Uint32 localKey[2];
+ Uint32 localKeyLength;
+ Uint32 keyLength;
+ Uint32 key[4];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/NodeFailRep.hpp b/ndb/include/kernel/signaldata/NodeFailRep.hpp
new file mode 100644
index 00000000000..060acd6a3e2
--- /dev/null
+++ b/ndb/include/kernel/signaldata/NodeFailRep.hpp
@@ -0,0 +1,68 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODE_FAILREP_HPP
+#define NODE_FAILREP_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+
+/**
+ * This signals is sent by Qmgr to NdbCntr
+ * and then from NdbCntr sent to: dih, dict, lqh, tc & API
+ */
+class NodeFailRep {
+ /**
+ * Sender(s)
+ */
+ friend class Qmgr;
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Ndbcntr;
+ friend class Dbdict;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdih;
+ friend class Dblqh;
+ friend class Dbtc;
+ friend class ClusterMgr;
+ friend class Trix;
+ friend class Backup;
+ friend class Suma;
+ friend class Grep;
+ friend class SafeCounterManager;
+
+public:
+ STATIC_CONST( SignalLength = 3 + NodeBitmask::Size );
+private:
+
+ Uint32 failNo;
+
+ /**
+ * Note: This field is only set when signals is sent FROM Ndbcntr
+ * (not when signal is sent from Qmgr)
+ */
+ Uint32 masterNodeId;
+
+ Uint32 noOfNodes;
+ Uint32 theNodes[NodeBitmask::Size];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/NodeStateSignalData.hpp b/ndb/include/kernel/signaldata/NodeStateSignalData.hpp
new file mode 100644
index 00000000000..391d8f89566
--- /dev/null
+++ b/ndb/include/kernel/signaldata/NodeStateSignalData.hpp
@@ -0,0 +1,94 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODE_STATE_SIGNAL_DATA_HPP
+#define NODE_STATE_SIGNAL_DATA_HPP
+
+#include <NodeState.hpp>
+
+/**
+ * NodeStateRep
+ *
+ * Sent so that all blocks will update their NodeState
+ */
+class NodeStateRep {
+ /**
+ * Sender(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Reciver
+ */
+ friend class SimulatedBlock;
+
+public:
+ STATIC_CONST( SignalLength = NodeState::DataLength );
+private:
+
+ NodeState nodeState;
+};
+
+/**
+ * ChangeNodeStateReq
+ *
+ * Sent by NdbCntr when synchronous NodeState updates are needed
+ */
+class ChangeNodeStateReq {
+ /**
+ * Sender(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Reciver
+ */
+ friend class SimulatedBlock;
+
+public:
+ STATIC_CONST( SignalLength = 2 + NodeState::DataLength );
+public:
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ NodeState nodeState;
+};
+
+/**
+ * ChangeNodeStateConf
+ *
+ * Sent by SimulatedBlock as a confirmation to ChangeNodeStateReq
+ */
+class ChangeNodeStateConf {
+ /**
+ * Sender(s)
+ */
+ friend class SimulatedBlock;
+
+ /**
+ * Reciver
+ */
+ friend class NdbCntr;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+private:
+
+ Uint32 senderData;
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/PackedSignal.hpp b/ndb/include/kernel/signaldata/PackedSignal.hpp
new file mode 100644
index 00000000000..057bb39b25a
--- /dev/null
+++ b/ndb/include/kernel/signaldata/PackedSignal.hpp
@@ -0,0 +1,43 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PACKED_SIGNAL_HPP
+#define PACKED_SIGNAL_HPP
+
+#include "SignalData.hpp"
+
+// -------- CODES FOR COMPRESSED SIGNAL (PACKED_SIGNAL) -------
+#define ZCOMMIT 0
+#define ZCOMPLETE 1
+#define ZCOMMITTED 2
+#define ZCOMPLETED 3
+#define ZLQHKEYCONF 4
+#define ZREMOVE_MARKER 5
+
+class PackedSignal {
+
+ static Uint32 getSignalType(Uint32 data);
+
+ /**
+ * For printing
+ */
+ friend bool printPACKED_SIGNAL(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+};
+
+inline
+Uint32 PackedSignal::getSignalType(Uint32 data) { return data >> 28; };
+
+#endif
diff --git a/ndb/include/kernel/signaldata/PrepDropTab.hpp b/ndb/include/kernel/signaldata/PrepDropTab.hpp
new file mode 100644
index 00000000000..e9cc28fed0c
--- /dev/null
+++ b/ndb/include/kernel/signaldata/PrepDropTab.hpp
@@ -0,0 +1,170 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PREP_DROP_TAB_HPP
+#define PREP_DROP_TAB_HPP
+
+#include "SignalData.hpp"
+
+class PrepDropTabReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtc;
+ friend class Dblqh;
+ friend class Dbdih;
+
+ friend bool printPREP_DROP_TAB_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 tableId;
+ Uint32 requestType; // @see DropTabReq::RequestType
+};
+
+class PrepDropTabConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+ friend class Dblqh;
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printPREP_DROP_TAB_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 tableId;
+};
+
+class PrepDropTabRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+ friend class Dblqh;
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdict;
+
+ friend bool printPREP_DROP_TAB_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum ErrorCode {
+ OK = 0,
+ NoSuchTable = 1,
+ PrepDropInProgress = 2,
+ DropInProgress = 3,
+ InvalidTableState = 4
+ };
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 tableId;
+ Uint32 errorCode;
+};
+
+class WaitDropTabReq {
+ /**
+ * Sender
+ */
+ friend class Dbtc;
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+
+ friend bool printWAIT_DROP_TAB_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ Uint32 tableId;
+ Uint32 senderRef;
+};
+
+class WaitDropTabRef {
+ /**
+ * Sender
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtc;
+ friend class Dbdih;
+
+ friend bool printWAIT_DROP_TAB_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum ErrorCode {
+ NoSuchTable = 1,
+ IllegalTableState = 2,
+ DropInProgress = 3
+ };
+
+ Uint32 tableId;
+ Uint32 senderRef;
+ Uint32 errorCode;
+ Uint32 tableStatus;
+};
+
+
+class WaitDropTabConf {
+ /**
+ * Sender
+ */
+ friend class Dblqh;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtc;
+ friend class Dbdih;
+
+ friend bool printWAIT_DROP_TAB_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ Uint32 tableId;
+ Uint32 senderRef;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/PrepFailReqRef.hpp b/ndb/include/kernel/signaldata/PrepFailReqRef.hpp
new file mode 100644
index 00000000000..90b568237b8
--- /dev/null
+++ b/ndb/include/kernel/signaldata/PrepFailReqRef.hpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PREP_FAILREQREF_HPP
+#define PREP_FAILREQREF_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+
+/**
+ * The Req signal is sent by Qmgr to Qmgr
+ * and the Ref signal might be sent back
+ *
+ * NOTE that the signals are identical
+ */
+class PrepFailReqRef {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Qmgr;
+
+ friend bool printPREPFAILREQREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 3 + NodeBitmask::Size );
+private:
+
+ Uint32 xxxBlockRef;
+ Uint32 failNo;
+
+ Uint32 noOfNodes;
+ Uint32 theNodes[NodeBitmask::Size];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ReadNodesConf.hpp b/ndb/include/kernel/signaldata/ReadNodesConf.hpp
new file mode 100644
index 00000000000..f3176cbf0e8
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ReadNodesConf.hpp
@@ -0,0 +1,109 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef READ_NODESCONF_HPP
+#define READ_NODESCONF_HPP
+
+#include <NodeBitmask.hpp>
+
+/**
+ * This signals is sent by Qmgr to NdbCntr
+ * and then from NdbCntr sent to: dih, dict, lqh, tc
+ *
+ * NOTE Only noOfNodes & allNodes are valid when sent from Qmgr
+ */
+class ReadNodesConf {
+ /**
+ * Sender(s)
+ */
+ friend class Qmgr;
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbdih;
+ friend class Dbdict;
+ friend class Dblqh;
+ friend class Dbtc;
+ friend class Trix;
+ friend class Backup;
+ friend class Suma;
+ friend class Grep;
+
+public:
+ STATIC_CONST( SignalLength = 2 + 6*NodeBitmask::Size );
+private:
+
+ Uint32 noOfNodes;
+
+ /**
+ *
+ * NOTE Not valid when send from Qmgr
+ */
+ Uint32 masterNodeId;
+
+ /**
+ * This array defines all the ndb nodes in the system
+ */
+ Uint32 allNodes[NodeBitmask::Size];
+
+ /**
+ * This array describes wheather the nodes are currently active
+ *
+ * NOTE Not valid when send from Qmgr
+ */
+ Uint32 inactiveNodes[NodeBitmask::Size];
+
+ /**
+ * This array describes the version id of the nodes
+ * The version id is a 4 bit number
+ *
+ * NOTE Not valid when send from Qmgr
+ */
+ Uint32 theVersionIds[4*NodeBitmask::Size];
+
+ static void setVersionId(NodeId, Uint8 versionId, Uint32 theVersionIds[]);
+ static Uint8 getVersionId(NodeId, const Uint32 theVersionIds[]);
+};
+
+inline
+void
+ReadNodesConf::setVersionId(NodeId nodeId, Uint8 versionId,
+ Uint32 theVersionIds[]){
+ const int word = nodeId >> 3;
+ const int shift = (nodeId & 7) << 2;
+
+ const Uint32 mask = ~(((Uint32)15) << shift);
+ const Uint32 tmp = theVersionIds[word];
+
+ theVersionIds[word] = (tmp & mask) | ((((Uint32)versionId) & 15) << shift);
+}
+
+inline
+Uint8
+ReadNodesConf::getVersionId(NodeId nodeId, const Uint32 theVersionIds[]){
+ const int word = nodeId >> 3;
+ const int shift = (nodeId & 7) << 2;
+
+ return (theVersionIds[word] >> shift) & 15;
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/RelTabMem.hpp b/ndb/include/kernel/signaldata/RelTabMem.hpp
new file mode 100644
index 00000000000..9cf1787bba4
--- /dev/null
+++ b/ndb/include/kernel/signaldata/RelTabMem.hpp
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REL_TABMEM_HPP
+#define REL_TABMEM_HPP
+
+#include "SignalData.hpp"
+
+class RelTabMemReq {
+ /**
+ * Sender(s) and Receiver(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbacc;
+ friend class Dbdih;
+ friend class Dblqh;
+ friend class Dbtc;
+ friend class Dbtup;
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 primaryTableId;
+ Uint32 secondaryTableId;
+};
+
+class RelTabMemConf {
+ /**
+ * Sender(s) and Receiver(s)
+ */
+ friend class Dbdict;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbacc;
+ friend class Dbdih;
+ friend class Dblqh;
+ friend class Dbtc;
+ friend class Dbtup;
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 userPtr;
+ Uint32 senderRef;
+ Uint32 nodeId;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/RepImpl.hpp b/ndb/include/kernel/signaldata/RepImpl.hpp
new file mode 100644
index 00000000000..affffe46f9c
--- /dev/null
+++ b/ndb/include/kernel/signaldata/RepImpl.hpp
@@ -0,0 +1,500 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REP_IMPL_HPP
+#define REP_IMPL_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+#include <ndb_limits.h>
+#include <debugger/GrepError.hpp>
+
+/**
+ * RecordType
+ * sz = no of elems in enum
+ * @todo support for meta_log must be added
+ */
+enum RecordType
+{
+ DATA_SCAN = 0,
+ DATA_LOG = 1,
+ META_SCAN = 2,
+ // META_LOG = 3, //removed META_LOG. not supported
+ RecordTypeSize = 3 // =4 if meta log is supported
+};
+
+/**
+ * Wait GCP
+ */
+class RepWaitGcpReq
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+ friend class GrepParticipant;
+ friend bool printREP_WAITGCP_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 gcp;
+ Uint32 senderNodeId;
+};
+
+class RepWaitGcpConf
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+ friend class GrepParticipant;
+
+ friend bool printREP_WAITGCP_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderNodeId;
+};
+
+class RepWaitGcpRef
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+ friend class GrepParticipant;
+
+ friend bool printREP_WAITGCP_REF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 6 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 senderNodeId;
+ GrepError::Code err;
+};
+
+class RepGetGciReq
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+ friend class Grep;
+
+ friend bool printREP_GET_GCI_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 nodeGrp;
+};
+
+class RepGetGciConf
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_GET_GCI_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 7 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 nodeGrp;
+ Uint32 firstPSGCI;
+ Uint32 lastPSGCI;
+ Uint32 firstSSGCI;
+ Uint32 lastSSGCI;
+};
+
+class RepGetGciRef
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_GET_GCI_REF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 8);
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 nodeGrp;
+ Uint32 firstPSGCI;
+ Uint32 lastPSGCI;
+ Uint32 firstSSGCI;
+ Uint32 lastSSGCI;
+ GrepError::Code err;
+};
+
+class RepGetGciBufferReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_GET_GCIBUFFER_REQ(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 firstGCI;
+ Uint32 lastGCI;
+ Uint32 nodeGrp;
+};
+
+
+class RepGetGciBufferConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_GET_GCIBUFFER_CONF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 8 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 firstPSGCI;
+ Uint32 lastPSGCI;
+ Uint32 firstSSGCI;
+ Uint32 lastSSGCI;
+ Uint32 currentGCIBuffer;
+ Uint32 nodeGrp;
+};
+
+class RepGetGciBufferRef
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_GET_GCIBUFFER_REF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 9 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 firstPSGCI;
+ Uint32 lastPSGCI;
+ Uint32 firstSSGCI;
+ Uint32 lastSSGCI;
+ Uint32 currentGCIBuffer;
+ Uint32 nodeGrp;
+ GrepError::Code err;
+};
+
+class RepInsertGciBufferReq
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_INSERT_GCIBUFFER_REQ(FILE *, const Uint32 *,
+ Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 gci;
+ Uint32 nodeGrp;
+ Uint32 force;
+};
+
+class RepInsertGciBufferRef
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_INSERT_GCIBUFFER_REF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 7 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 gci;
+ Uint32 nodeGrp;
+ Uint32 tableId;
+ Uint32 force;
+ GrepError::Code err;
+};
+
+class RepInsertGciBufferConf
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_INSERT_GCIBUFFER_CONF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 gci;
+ Uint32 nodeGrp;
+ Uint32 force;
+};
+
+
+class RepClearPSGciBufferReq
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_CLEAR_PS_GCIBUFFER_REQ(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 firstGCI;
+ Uint32 lastGCI;
+ Uint32 nodeGrp;
+};
+
+class RepClearPSGciBufferRef
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_CLEAR_PS_GCIBUFFER_REF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 7 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 firstGCI;
+ Uint32 lastGCI;
+ Uint32 currentGCI;
+ Uint32 nodeGrp;
+ GrepError::Code err;
+};
+
+class RepClearPSGciBufferConf
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_CLEAR_PS_GCIBUFFER_CONF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 firstGCI;
+ Uint32 lastGCI;
+ Uint32 nodeGrp;
+};
+
+class RepClearSSGciBufferReq
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_CLEAR_SS_GCIBUFFER_REQ(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 firstGCI;
+ Uint32 lastGCI;
+ Uint32 nodeGrp;
+};
+
+class RepClearSSGciBufferRef
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_CLEAR_SS_GCIBUFFER_REF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 7 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 firstGCI;
+ Uint32 lastGCI;
+ Uint32 currentGCI;
+ Uint32 nodeGrp;
+ GrepError::Code err;
+};
+
+class RepClearSSGciBufferConf
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_CLEAR_SS_GCIBUFFER_CONF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 firstGCI;
+ Uint32 lastGCI;
+ Uint32 nodeGrp;
+};
+
+
+class RepDataPage
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_DATA_PAGE(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 nodeGrp;
+ Uint32 gci;
+};
+
+
+class RepGciBufferAccRep
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_GCIBUFFER_ACC_REP(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 nodeGrp;
+ Uint32 gci;
+ Uint32 totalSentBytes;
+};
+
+class RepDropTableReq
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_DROP_TABLE_REQ(FILE *, const Uint32 *,
+ Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+ Uint32 tableId;
+ // char tableName[MAX_TAB_NAME_SIZE];
+};
+
+class RepDropTableRef
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_DROP_TABLE_REF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+ Uint32 tableId;
+ // char tableName[MAX_TAB_NAME_SIZE];
+};
+
+class RepDropTableConf
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+
+ friend bool printREP_DROP_TABLE_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+ Uint32 tableId;
+ //char tableName[MAX_TAB_NAME_SIZE];
+};
+
+class RepDisconnectRep
+{
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Rep;
+ friend class Grep;
+
+ friend bool printREP_DISCONNECT_REP(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ enum NodeType {
+ DB = 0,
+ REP = 1
+ };
+ STATIC_CONST( SignalLength = 7 );
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 nodeId;
+ Uint32 nodeType;
+ Uint32 subId;
+ Uint32 subKey;
+ Uint32 err;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ResumeReq.hpp b/ndb/include/kernel/signaldata/ResumeReq.hpp
new file mode 100644
index 00000000000..a4880474ca8
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ResumeReq.hpp
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef RESUME_REQ_HPP
+#define RESUME_REQ_HPP
+
+#include "SignalData.hpp"
+
+class ResumeReq {
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Sender
+ */
+ friend class MgmtSrvr;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+public:
+
+ Uint32 senderRef;
+ Uint32 senderData;
+};
+
+class ResumeRef {
+
+ /**
+ * Reciver(s)
+ */
+ friend class MgmtSrvr;
+
+ /**
+ * Sender
+ */
+ friend class Ndbcntr;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ enum ErrorCode {
+ OK = 0,
+ NodeShutdownInProgress = 1,
+ SystemShutdownInProgress = 2,
+ NodeShutdownWouldCauseSystemCrash = 3
+ };
+
+public:
+ Uint32 senderData;
+ Uint32 errorCode;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/ScanFrag.hpp b/ndb/include/kernel/signaldata/ScanFrag.hpp
new file mode 100644
index 00000000000..65ab6f7e411
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ScanFrag.hpp
@@ -0,0 +1,325 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SCAN_FRAG_HPP
+#define SCAN_FRAG_HPP
+
+#include "SignalData.hpp"
+#include "ndb_limits.h"
+
+class ScanFragReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+ friend class Backup;
+ friend class Suma;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dblqh;
+public:
+ STATIC_CONST( SignalLength = 25 );
+
+public:
+ Uint32 senderData;
+ Uint32 resultRef; // Where to send the result
+ Uint32 savePointId;
+ Uint32 requestInfo;
+ Uint32 tableId;
+ Uint32 fragmentNo;
+ Uint32 schemaVersion;
+ Uint32 transId1;
+ Uint32 transId2;
+ Uint32 clientOpPtr[MAX_PARALLEL_OP_PER_SCAN];
+
+ static Uint32 getConcurrency(const Uint32 & requestInfo);
+ static Uint32 getLockMode(const Uint32 & requestInfo);
+ static Uint32 getHoldLockFlag(const Uint32 & requestInfo);
+ static Uint32 getKeyinfoFlag(const Uint32 & requestInfo);
+ static Uint32 getReadCommittedFlag(const Uint32 & requestInfo);
+ static Uint32 getRangeScanFlag(const Uint32 & requestInfo);
+ static Uint32 getAttrLen(const Uint32 & requestInfo);
+
+ static void setConcurrency(Uint32 & requestInfo, Uint32 concurrency);
+ static void setLockMode(Uint32 & requestInfo, Uint32 lockMode);
+ static void setHoldLockFlag(Uint32 & requestInfo, Uint32 holdLock);
+ static void setKeyinfoFlag(Uint32 & requestInfo, Uint32 keyinfo);
+ static void setReadCommittedFlag(Uint32 & requestInfo, Uint32 readCommitted);
+ static void setRangeScanFlag(Uint32 & requestInfo, Uint32 rangeScan);
+ static void setAttrLen(Uint32 & requestInfo, Uint32 attrLen);
+};
+
+class KeyInfo20 {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Backup;
+ friend class NdbOperation;
+ friend class NdbScanReceiver;
+public:
+ //STATIC_CONST( SignalLength = 21 );
+ STATIC_CONST( HeaderLength = 5);
+ STATIC_CONST( DataLength = 20 );
+
+
+ static Uint32 setScanInfo(Uint32 noOfOps, Uint32 scanNo);
+ static Uint32 getScanNo(Uint32 scanInfo);
+ static Uint32 getScanOp(Uint32 scanInfo);
+
+public:
+ Uint32 clientOpPtr;
+ Uint32 keyLen;
+ Uint32 scanInfo_Node;
+ Uint32 transId1;
+ Uint32 transId2;
+ Uint32 keyData[DataLength];
+};
+
+class ScanFragConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+ friend class Backup;
+ friend class Suma;
+public:
+ STATIC_CONST( SignalLength = 21 );
+
+public:
+ Uint32 senderData;
+ Uint32 completedOps;
+ Uint32 fragmentCompleted;
+ Uint32 opReturnDataLen[16];
+ Uint32 transId1;
+ Uint32 transId2;
+};
+
+class ScanFragRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+ friend class Backup;
+ friend class Suma;
+public:
+ STATIC_CONST( SignalLength = 4 );
+public:
+ enum ErrorCode {
+ ZNO_FREE_TC_CONREC_ERROR = 484,
+ ZTOO_FEW_CONCURRENT_OPERATIONS = 485,
+ ZTOO_MANY_CONCURRENT_OPERATIONS = 486,
+ ZSCAN_NO_FRAGMENT_ERROR = 487,
+ ZTOO_MANY_ACTIVE_SCAN_ERROR = 488,
+ ZNO_FREE_SCANREC_ERROR = 489,
+ ZSTANDBY_SCAN_ERROR = 1209,
+ ZSCAN_BOOK_ACC_OP_ERROR = 1219,
+ ZUNKNOWN_TRANS_ERROR = 1227
+ };
+
+ Uint32 senderData;
+ Uint32 transId1;
+ Uint32 transId2;
+ Uint32 errorCode;
+};
+
+/**
+ * This is part of Scan Fragment protocol
+ *
+ * Not to be confused with ScanNextReq in Scan Table protocol
+ */
+class ScanFragNextReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+ friend class Backup;
+ friend class Suma;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dblqh;
+
+ friend bool printSCANFRAGNEXTREQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+public:
+ Uint32 senderData;
+ Uint32 closeFlag;
+ Uint32 transId1;
+ Uint32 transId2;
+};
+
+/**
+ * Request Info
+ *
+ * a = Length of attrinfo - 16 Bits (16-31)
+ * c = Concurrency - 5 Bits (0-4) -> Max 31
+ * l = Lock Mode - 1 Bit 5
+ * h = Hold lock - 1 Bit 7
+ * k = Keyinfo - 1 Bit 8
+ * r = read committed - 1 Bit 9
+ * x = range scan - 1 Bit 6
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * ccccclxhkr aaaaaaaaaaaaaaaa
+ */
+#define SF_CONCURRENCY_SHIFT (0)
+#define SF_CONCURRENCY_MASK (31)
+
+#define SF_LOCK_MODE_SHIFT (5)
+#define SF_LOCK_MODE_MASK (1)
+
+#define SF_HOLD_LOCK_SHIFT (7)
+#define SF_KEYINFO_SHIFT (8)
+#define SF_READ_COMMITTED_SHIFT (9)
+#define SF_RANGE_SCAN_SHIFT (6)
+
+#define SF_ATTR_LEN_SHIFT (16)
+#define SF_ATTR_LEN_MASK (65535)
+
+inline
+Uint32
+ScanFragReq::getConcurrency(const Uint32 & requestInfo){
+ return (requestInfo >> SF_CONCURRENCY_SHIFT) & SF_CONCURRENCY_MASK;
+}
+
+inline
+Uint32
+ScanFragReq::getLockMode(const Uint32 & requestInfo){
+ return (requestInfo >> SF_LOCK_MODE_SHIFT) & SF_LOCK_MODE_MASK;
+}
+
+inline
+Uint32
+ScanFragReq::getHoldLockFlag(const Uint32 & requestInfo){
+ return (requestInfo >> SF_HOLD_LOCK_SHIFT) & 1;
+}
+
+inline
+Uint32
+ScanFragReq::getKeyinfoFlag(const Uint32 & requestInfo){
+ return (requestInfo >> SF_KEYINFO_SHIFT) & 1;
+}
+
+inline
+Uint32
+ScanFragReq::getReadCommittedFlag(const Uint32 & requestInfo){
+ return (requestInfo >> SF_READ_COMMITTED_SHIFT) & 1;
+}
+
+inline
+Uint32
+ScanFragReq::getRangeScanFlag(const Uint32 & requestInfo){
+ return (requestInfo >> SF_RANGE_SCAN_SHIFT) & 1;
+}
+
+inline
+Uint32
+ScanFragReq::getAttrLen(const Uint32 & requestInfo){
+ return (requestInfo >> SF_ATTR_LEN_SHIFT) & SF_ATTR_LEN_MASK;
+}
+
+inline
+void
+ScanFragReq::setConcurrency(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, SF_CONCURRENCY_MASK, "ScanFragReq::setConcurrency");
+ requestInfo |= (val << SF_CONCURRENCY_SHIFT);
+}
+
+inline
+void
+ScanFragReq::setLockMode(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, SF_LOCK_MODE_MASK, "ScanFragReq::setLockMode");
+ requestInfo |= (val << SF_LOCK_MODE_SHIFT);
+}
+
+inline
+void
+ScanFragReq::setHoldLockFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "ScanFragReq::setHoldLockFlag");
+ requestInfo |= (val << SF_HOLD_LOCK_SHIFT);
+}
+
+inline
+void
+ScanFragReq::setKeyinfoFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "ScanFragReq::setKeyinfoFlag");
+ requestInfo |= (val << SF_KEYINFO_SHIFT);
+}
+
+inline
+void
+ScanFragReq::setReadCommittedFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "ScanFragReq::setReadCommittedFlag");
+ requestInfo |= (val << SF_READ_COMMITTED_SHIFT);
+}
+
+inline
+void
+ScanFragReq::setRangeScanFlag(UintR & requestInfo, UintR val){
+ ASSERT_BOOL(val, "ScanFragReq::setRangeScanFlag");
+ requestInfo |= (val << SF_RANGE_SCAN_SHIFT);
+}
+
+inline
+void
+ScanFragReq::setAttrLen(UintR & requestInfo, UintR val){
+ ASSERT_MAX(val, SF_ATTR_LEN_MASK, "ScanFragReq::setAttrLen");
+ requestInfo |= (val << SF_ATTR_LEN_SHIFT);
+}
+
+inline
+Uint32
+KeyInfo20::setScanInfo(Uint32 opNo, Uint32 scanNo){
+ ASSERT_MAX(opNo, 15, "KeyInfo20::setScanInfo");
+ ASSERT_MAX(scanNo, 255, "KeyInfo20::setScanInfo");
+ return (opNo << 8) + scanNo;
+}
+
+inline
+Uint32
+KeyInfo20::getScanNo(Uint32 scanInfo){
+ return scanInfo & 0xFF;
+}
+
+inline
+Uint32
+KeyInfo20::getScanOp(Uint32 scanInfo){
+ return (scanInfo >> 8) & 0xF;
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/ScanTab.hpp b/ndb/include/kernel/signaldata/ScanTab.hpp
new file mode 100644
index 00000000000..efd8a4918ab
--- /dev/null
+++ b/ndb/include/kernel/signaldata/ScanTab.hpp
@@ -0,0 +1,453 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SCAN_TAB_H
+#define SCAN_TAB_H
+
+#include "SignalData.hpp"
+
+/**
+ *
+ * SENDER: API
+ * RECIVER: Dbtc
+ */
+class ScanTabReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc; // Reciver
+
+ /**
+ * Sender(s)
+ */
+ friend class NdbOperation;
+ friend class NdbConnection;
+
+ /**
+ * For printing
+ */
+ friend bool printSCANTABREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 25 );
+
+private:
+
+ // Type definitions
+
+ /**
+ * DATA VARIABLES
+ */
+ UintR apiConnectPtr; // DATA 0
+ UintR attrLen; // DATA 1
+ UintR requestInfo; // DATA 2
+ UintR tableId; // DATA 3
+ UintR tableSchemaVersion; // DATA 4
+ UintR storedProcId; // DATA 5
+ UintR transId1; // DATA 6
+ UintR transId2; // DATA 7
+ UintR buddyConPtr; // DATA 8
+ UintR apiOperationPtr[16]; // DATA 9-25
+
+ /**
+ * Get:ers for requestInfo
+ */
+ static Uint8 getParallelism(const UintR & requestInfo);
+ static Uint8 getLockMode(const UintR & requestInfo);
+ static Uint8 getHoldLockFlag(const UintR & requestInfo);
+ static Uint8 getReadCommittedFlag(const UintR & requestInfo);
+ static Uint8 getRangeScanFlag(const UintR & requestInfo);
+
+ /**
+ * Set:ers for requestInfo
+ */
+ static void clearRequestInfo(UintR & requestInfo);
+ static void setParallelism(UintR & requestInfo, Uint32 flag);
+ static void setLockMode(UintR & requestInfo, Uint32 flag);
+ static void setHoldLockFlag(UintR & requestInfo, Uint32 flag);
+ static void setReadCommittedFlag(UintR & requestInfo, Uint32 flag);
+ static void setRangeScanFlag(UintR & requestInfo, Uint32 flag);
+
+};
+
+/**
+ * Request Info
+ *
+ p = Parallelism - 8 Bits -> Max 256 (Bit 0-7)
+ l = Lock mode - 1 Bit 8
+ h = Hold lock mode - 1 Bit 10
+ c = Read Committed - 1 Bit 11
+ x = Range Scan (TUX) - 1 Bit 15
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ ppppppppl hc x
+*/
+
+#define PARALLELL_SHIFT (0)
+#define PARALLELL_MASK (255)
+
+#define LOCK_MODE_SHIFT (8)
+#define LOCK_MODE_MASK (1)
+
+#define HOLD_LOCK_SHIFT (10)
+#define HOLD_LOCK_MASK (1)
+
+#define READ_COMMITTED_SHIFT (11)
+#define READ_COMMITTED_MASK (1)
+
+#define RANGE_SCAN_SHIFT (15)
+#define RANGE_SCAN_MASK (1)
+
+inline
+Uint8
+ScanTabReq::getParallelism(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> PARALLELL_SHIFT) & PARALLELL_MASK);
+}
+
+inline
+Uint8
+ScanTabReq::getLockMode(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> LOCK_MODE_SHIFT) & LOCK_MODE_MASK);
+}
+
+inline
+Uint8
+ScanTabReq::getHoldLockFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> HOLD_LOCK_SHIFT) & HOLD_LOCK_MASK);
+}
+
+inline
+Uint8
+ScanTabReq::getReadCommittedFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> READ_COMMITTED_SHIFT) & READ_COMMITTED_MASK);
+}
+
+inline
+Uint8
+ScanTabReq::getRangeScanFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> RANGE_SCAN_SHIFT) & RANGE_SCAN_MASK);
+}
+
+inline
+void
+ScanTabReq::clearRequestInfo(UintR & requestInfo){
+ requestInfo = 0;
+}
+
+inline
+void
+ScanTabReq::setParallelism(UintR & requestInfo, Uint32 type){
+ ASSERT_MAX(type, PARALLELL_MASK, "ScanTabReq::setParallellism");
+ requestInfo |= (type << PARALLELL_SHIFT);
+}
+
+inline
+void
+ScanTabReq::setLockMode(UintR & requestInfo, Uint32 mode){
+ ASSERT_MAX(mode, LOCK_MODE_MASK, "ScanTabReq::setLockMode");
+ requestInfo |= (mode << LOCK_MODE_SHIFT);
+}
+
+inline
+void
+ScanTabReq::setHoldLockFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "ScanTabReq::setHoldLockFlag");
+ requestInfo |= (flag << HOLD_LOCK_SHIFT);
+}
+
+inline
+void
+ScanTabReq::setReadCommittedFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "ScanTabReq::setReadCommittedFlag");
+ requestInfo |= (flag << READ_COMMITTED_SHIFT);
+}
+
+inline
+void
+ScanTabReq::setRangeScanFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "ScanTabReq::setRangeScanFlag");
+ requestInfo |= (flag << RANGE_SCAN_SHIFT);
+}
+
+
+/**
+ *
+ * SENDER: Dbtc
+ * RECIVER: API
+ */
+class ScanTabConf {
+ /**
+ * Reciver(s)
+ */
+ friend class NdbConnection; // Reciver
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * For printing
+ */
+ friend bool printSCANTABCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+
+ // Type definitions
+
+ /**
+ * DATA VARIABLES
+ */
+ UintR apiConnectPtr; // DATA 0
+ UintR requestInfo; // DATA 1
+ UintR transId1; // DATA 2
+ UintR transId2; // DATA 3
+#if 0
+ UintR operLenAndIdx[16]; // DATA 4-19
+
+ /**
+ * Get:ers for operLenAndIdx
+ */
+ static Uint32 getLen(const UintR & operLenAndIdx);
+ static Uint8 getIdx(const UintR & operLenAndIdx);
+#endif
+
+ /**
+ * Get:ers for requestInfo
+ */
+ static Uint8 getOperations(const UintR & reqInfo);
+ static Uint8 getScanStatus(const UintR & reqInfo);
+
+ /**
+ * Set:ers for requestInfo
+ */
+ static void setOperations(UintR & reqInfo, Uint32 ops);
+ static void setScanStatus(UintR & reqInfo, Uint32 stat);
+
+
+};
+
+/**
+ * request info
+ *
+ o = received operations - 7 Bits -> Max 255 (Bit 0-7)
+ s = status of scan - 2 Bits -> Max ??? (Bit 8-?)
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ ooooooooss
+*/
+
+#define OPERATIONS_SHIFT (0)
+#define OPERATIONS_MASK (0xFF)
+
+#define STATUS_SHIFT (8)
+#define STATUS_MASK (0xFF)
+
+inline
+Uint8
+ScanTabConf::getOperations(const UintR & reqInfo){
+ return (Uint8)((reqInfo >> OPERATIONS_SHIFT) & OPERATIONS_MASK);
+}
+
+inline
+void
+ScanTabConf::setOperations(UintR & requestInfo, Uint32 ops){
+ ASSERT_MAX(ops, OPERATIONS_MASK, "ScanTabConf::setOperations");
+ requestInfo |= (ops << OPERATIONS_SHIFT);
+}
+
+inline
+Uint8
+ScanTabConf::getScanStatus(const UintR & reqInfo){
+ return (Uint8)((reqInfo >> STATUS_SHIFT) & STATUS_MASK);
+}
+
+inline
+void
+ScanTabConf::setScanStatus(UintR & requestInfo, Uint32 stat){
+ ASSERT_MAX(stat, STATUS_MASK, "ScanTabConf::setScanStatus");
+ requestInfo |= (stat << STATUS_SHIFT);
+}
+
+
+/**
+ *
+ * SENDER: Dbtc, API
+ * RECIVER: API, Dbtc
+ */
+class ScanTabInfo {
+ /**
+ * Reciver(s) and Sender(s)
+ */
+ friend class NdbConnection;
+ friend class Dbtc;
+
+ /**
+ * For printing
+ */
+ friend bool printSCANTABINFO(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 17 );
+
+private:
+
+ // Type definitions
+
+ /**
+ * DATA VARIABLES
+ */
+ UintR apiConnectPtr; // DATA 0
+ UintR operLenAndIdx[16]; // DATA 1-16
+
+ /**
+ * Get:ers for operLenAndIdx
+ */
+ static Uint32 getLen(const UintR & operLenAndIdx);
+ static Uint8 getIdx(const UintR & operLenAndIdx);
+
+};
+
+
+/**
+ * Operation length and index
+ *
+ l = Length of operation - 24 Bits -> Max 16777215 (Bit 0-24)
+ i = Index of operation - 7 Bits -> Max 255 (Bit 25-32)
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ llllllllllllllllllllllllliiiiiii
+*/
+
+#define LENGTH_SHIFT (0)
+#define LENGTH_MASK (0xFFFFFF)
+
+#define INDEX_SHIFT (24)
+#define INDEX_MASK (0xFF)
+
+inline
+Uint32
+ScanTabInfo::getLen(const UintR & operLenAndIdx){
+ return (Uint32)((operLenAndIdx >> LENGTH_SHIFT) & LENGTH_MASK);
+}
+
+inline
+Uint8
+ScanTabInfo::getIdx(const UintR & operLenAndIdx){
+ return (Uint8)((operLenAndIdx >> INDEX_SHIFT) & INDEX_MASK);
+}
+
+/**
+ *
+ * SENDER: Dbtc
+ * RECIVER: API
+ */
+class ScanTabRef {
+ /**
+ * Reciver(s)
+ */
+ friend class NdbConnection; // Reciver
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * For printing
+ */
+ friend bool printSCANTABREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+
+ // Type definitions
+
+ /**
+ * DATA VARIABLES
+ */
+ UintR apiConnectPtr; // DATA 0
+ UintR transId1; // DATA 1
+ UintR transId2; // DATA 2
+ UintR errorCode; // DATA 3
+ // UintR sendScanNextReqWithClose; // DATA 4
+
+};
+
+/**
+ *
+ * SENDER: API
+ * RECIVER: Dbtc
+ */
+class ScanNextReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc; // Reciver
+
+ /**
+ * Sender(s)
+ */
+ friend class NdbOperation;
+
+ /**
+ * For printing
+ */
+ friend bool printSCANNEXTREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+
+ // Type definitions
+
+ /**
+ * DATA VARIABLES
+ */
+ UintR apiConnectPtr; // DATA 0
+ UintR stopScan; // DATA 1
+ UintR transId1; // DATA 2
+ UintR transId2; // DATA 3
+
+ // stopScan = 1, stop this scan
+
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/SetLogLevelOrd.hpp b/ndb/include/kernel/signaldata/SetLogLevelOrd.hpp
new file mode 100644
index 00000000000..680e9b25a49
--- /dev/null
+++ b/ndb/include/kernel/signaldata/SetLogLevelOrd.hpp
@@ -0,0 +1,70 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SET_LOGLEVEL_ORD_HPP
+#define SET_LOGLEVEL_ORD_HPP
+
+#include <LogLevel.hpp>
+
+/**
+ *
+ */
+class SetLogLevelOrd {
+ /**
+ * Sender(s)
+ */
+ friend class MgmtSrvr; /* XXX can probably be removed */
+ friend class MgmApiSession;
+ friend class CommandInterpreter;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Cmvmi;
+
+ friend class NodeLogLevel;
+
+private:
+ STATIC_CONST( SignalLength = 25 );
+
+ Uint32 noOfEntries;
+ Uint32 theCategories[12];
+ Uint32 theLevels[12];
+
+ void clear();
+
+ /**
+ * Note level is valid as 0-15
+ */
+ void setLogLevel(LogLevel::EventCategory ec, int level = 7);
+};
+
+inline
+void
+SetLogLevelOrd::clear(){
+ noOfEntries = 0;
+}
+
+inline
+void
+SetLogLevelOrd::setLogLevel(LogLevel::EventCategory ec, int level){
+ assert(noOfEntries < 12);
+ theCategories[noOfEntries] = ec;
+ theLevels[noOfEntries] = level;
+ noOfEntries++;
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/SetVarReq.hpp b/ndb/include/kernel/signaldata/SetVarReq.hpp
new file mode 100644
index 00000000000..8cb3e78be8b
--- /dev/null
+++ b/ndb/include/kernel/signaldata/SetVarReq.hpp
@@ -0,0 +1,84 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SETVARREQ_H
+#define SETVARREQ_H
+
+#include "SignalData.hpp"
+#include "ConfigParamId.hpp"
+
+class SetVarReq {
+
+public:
+
+
+ static UintR size();
+
+ void mgmtSrvrBlockRef(UintR mgmtSrvrBlockRef);
+ UintR mgmtSrvrBlockRef(void) const;
+
+ void variable(ConfigParamId variable);
+ ConfigParamId variable(void) const;
+
+ void value(UintR value);
+ UintR value(void) const;
+
+
+private:
+
+ UintR _mgmtSrvrBlockRef;
+ UintR _variable;
+ UintR _value;
+};
+
+
+
+inline UintR SetVarReq::size(void) {
+ return 3;
+}
+
+
+inline void SetVarReq::mgmtSrvrBlockRef(UintR mgmtSrvrBlockRef) {
+ _mgmtSrvrBlockRef = mgmtSrvrBlockRef;
+}
+
+inline UintR SetVarReq::mgmtSrvrBlockRef(void) const {
+ return _mgmtSrvrBlockRef;
+}
+
+
+inline void SetVarReq::variable(ConfigParamId variable) {
+ _variable = variable;
+}
+
+
+inline ConfigParamId SetVarReq::variable(void) const {
+ return static_cast<ConfigParamId>(_variable);
+}
+
+
+inline void SetVarReq::value(UintR value) {
+ _value = value;
+}
+
+inline UintR SetVarReq::value(void) const {
+ return _value;
+}
+
+
+
+#endif // SETVARREQ_H
+
diff --git a/ndb/include/kernel/signaldata/SignalData.hpp b/ndb/include/kernel/signaldata/SignalData.hpp
new file mode 100644
index 00000000000..071bd9b9104
--- /dev/null
+++ b/ndb/include/kernel/signaldata/SignalData.hpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SIGNAL_DATA_H
+#define SIGNAL_DATA_H
+
+#include <ndb_limits.h>
+#include <kernel_types.h>
+#include <NdbStdio.h>
+#include <NdbConstant.hpp>
+#include <stdlib.h>
+
+#ifndef NDB_ASSERT
+#ifdef VM_TRACE
+#define NDB_ASSERT(test, message) { if(!(test)) { printf(message); exit(-1); }}
+#else
+#define NDB_ASSERT(test, message)
+#endif
+#endif
+
+// Useful ASSERT macros...
+#define ASSERT_BOOL(flag, message) NDB_ASSERT( (flag<=1), (message) )
+#define ASSERT_RANGE(value, min, max, message) \
+ NDB_ASSERT((value) >= (min) && (value) <= (max), (message))
+#define ASSERT_MAX(value, max, message) \
+ NDB_ASSERT((value) <= (max), (message))
+
+#define SECTION(x) STATIC_CONST(x)
+
+// defines for setter and getters on commonly used member data in signals
+
+#define GET_SET_SENDERDATA \
+ Uint32 getSenderData() { return senderData; }; \
+ void setSenderData(Uint32 _s) { senderData = _s; };
+
+#define GET_SET_SENDERREF \
+ Uint32 getSenderRef() { return senderRef; }; \
+ void setSenderRef(Uint32 _s) { senderRef = _s; };
+
+#define GET_SET_PREPAREID \
+ Uint32 getPrepareId() { return prepareId; }; \
+ void setPrepareId(Uint32 _s) { prepareId = _s; };
+
+#define GET_SET_ERRORCODE \
+ Uint32 getErrorCode() { return errorCode; }; \
+ void setErrorCode(Uint32 _s) { errorCode = _s; };
+
+#define GET_SET_TCERRORCODE \
+ Uint32 getTCErrorCode() { return TCErrorCode; }; \
+ void setTCErrorCode(Uint32 _s) { TCErrorCode = _s; };
+
+#endif
diff --git a/ndb/include/kernel/signaldata/SignalDataPrint.hpp b/ndb/include/kernel/signaldata/SignalDataPrint.hpp
new file mode 100644
index 00000000000..588e2893214
--- /dev/null
+++ b/ndb/include/kernel/signaldata/SignalDataPrint.hpp
@@ -0,0 +1,36 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SIGNAL_DATA_PRINT_H
+#define SIGNAL_DATA_PRINT_H
+
+#include <kernel_types.h>
+#include <NdbStdio.h>
+
+/**
+ * Typedef for a Signal Data Print Function
+ */
+typedef bool (* SignalDataPrintFunction)(FILE * output, const Uint32 * theData, Uint32 len, BlockNumber receiverBlockNo);
+
+struct NameFunctionPair {
+ GlobalSignalNumber gsn;
+ SignalDataPrintFunction function;
+};
+
+extern const NameFunctionPair SignalDataPrintFunctions[];
+extern const unsigned short NO_OF_PRINT_FUNCTIONS;
+
+#endif
diff --git a/ndb/include/kernel/signaldata/SignalDroppedRep.hpp b/ndb/include/kernel/signaldata/SignalDroppedRep.hpp
new file mode 100644
index 00000000000..20863524358
--- /dev/null
+++ b/ndb/include/kernel/signaldata/SignalDroppedRep.hpp
@@ -0,0 +1,44 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SIGNAL_DROPPED_HPP
+#define SIGNAL_DROPPED_HPP
+
+#include "SignalData.hpp"
+
+class SignalDroppedRep {
+
+ /**
+ * Reciver(s)
+ */
+ friend class SimulatedBlock;
+
+ /**
+ * Sender (TransporterCallback.cpp)
+ */
+ friend void execute(void * , struct SignalHeader* const, Uint8,
+ Uint32* const, struct LinearSectionPtr ptr[3]);
+
+ friend bool printSIGNAL_DROPPED_REP(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+private:
+ Uint32 originalGsn;
+ Uint32 originalLength;
+ Uint32 originalSectionCount;
+ Uint32 originalData[1];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/SrFragidConf.hpp b/ndb/include/kernel/signaldata/SrFragidConf.hpp
new file mode 100644
index 00000000000..9a6088ad57f
--- /dev/null
+++ b/ndb/include/kernel/signaldata/SrFragidConf.hpp
@@ -0,0 +1,43 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SR_FRAGIDCONF_HPP
+#define SR_FRAGIDCONF_HPP
+
+#include "SignalData.hpp"
+
+class SrFragidConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbacc;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+public:
+ STATIC_CONST( SignalLength = 10 );
+
+private:
+ Uint32 lcpPtr;
+ Uint32 accPtr;
+ Uint32 noLocFrag;
+ Uint32 fragId[4];
+ Uint32 fragPtr[2];
+ Uint32 hashCheckBit;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/StartFragReq.hpp b/ndb/include/kernel/signaldata/StartFragReq.hpp
new file mode 100644
index 00000000000..ec05c1ee366
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StartFragReq.hpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef START_FRAGREQ_HPP
+#define START_FRAGREQ_HPP
+
+#include "SignalData.hpp"
+
+class StartFragReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+public:
+ STATIC_CONST( SignalLength = 19 );
+
+private:
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 lcpNo;
+ Uint32 lcpId;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 noOfLogNodes;
+ Uint32 lqhLogNode[4];
+ Uint32 startGci[4];
+ Uint32 lastGci[4];
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/StartInfo.hpp b/ndb/include/kernel/signaldata/StartInfo.hpp
new file mode 100644
index 00000000000..da032adba8a
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StartInfo.hpp
@@ -0,0 +1,84 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef START_INFO_HPP
+#define START_INFO_HPP
+
+/**
+ * This signal is sent from the master DIH to all DIHs
+ * when a node is starting.
+ * If the typeStart is initial node restart then the node
+ * has started without filesystem.
+ * All DIHs must then "forget" that the starting node has
+ * performed LCP's ever.
+ *
+ * @see StartPermReq
+ */
+
+class StartInfoReq {
+ /**
+ * Sender/Receiver
+ */
+ friend class Dbdih;
+
+ Uint32 startingNodeId;
+ Uint32 typeStart;
+ Uint32 systemFailureNo;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+};
+
+class StartInfoConf {
+
+ /**
+ * Sender/Receiver
+ */
+ friend class Dbdih;
+
+ /**
+ * NodeId of sending node
+ * which is "done"
+ */
+ Uint32 sendingNodeId;
+ Uint32 startingNodeId;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+};
+
+class StartInfoRef {
+
+ /**
+ * Sender/Receiver
+ */
+ friend class Dbdih;
+
+ /**
+ * NodeId of sending node
+ * The node was refused to start. This could be
+ * because there are still processes handling
+ * previous information from the starting node.
+ */
+ Uint32 sendingNodeId;
+ Uint32 startingNodeId;
+ Uint32 errorCode;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/StartMe.hpp b/ndb/include/kernel/signaldata/StartMe.hpp
new file mode 100644
index 00000000000..6593a9e9741
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StartMe.hpp
@@ -0,0 +1,63 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef START_ME_HPP
+#define START_ME_HPP
+
+/**
+ * This signal is sent...
+ *
+ * It also contains the Sysfile.
+ * Since the Sysfile can be larger than on StartMeConf signal,
+ * there might be more than on of these signals sent before
+ * the entire sysfile is transfered
+ *
+ */
+class StartMeReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+
+ Uint32 startingRef;
+ Uint32 startingVersion;
+};
+
+class StartMeConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 25 );
+private:
+
+ Uint32 startingNodeId;
+ Uint32 startWord;
+
+ /**
+ * No of free words to carry data
+ */
+ STATIC_CONST( DATA_SIZE = 23 );
+
+ Uint32 data[DATA_SIZE];
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/StartOrd.hpp b/ndb/include/kernel/signaldata/StartOrd.hpp
new file mode 100644
index 00000000000..43a48f70ba9
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StartOrd.hpp
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef START_ORD_HPP
+#define START_ORD_HPP
+
+#include "SignalData.hpp"
+#include "StopReq.hpp"
+
+class StartOrd {
+public:
+ /**
+ * Senders
+ */
+ friend class ThreadConfig;
+ friend class MgmtSrvr;
+ friend class Ndbcntr;
+
+ /**
+ * Receivers
+ */
+ friend class SimBlockCMCtrBlck;
+
+ /**
+ * RequestInfo - See StopReq for getters/setters
+ */
+ Uint32 restartInfo;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+};
+
+
+#endif
+
diff --git a/ndb/include/kernel/signaldata/StartPerm.hpp b/ndb/include/kernel/signaldata/StartPerm.hpp
new file mode 100644
index 00000000000..38be72835a3
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StartPerm.hpp
@@ -0,0 +1,68 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef START_PERM_REQ_HPP
+#define START_PERM_REQ_HPP
+
+/**
+ * This signal is sent by starting DIH to master DIH
+ *
+ * Used when starting in an already started cluster
+ *
+ */
+class StartPermReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+private:
+
+ Uint32 blockRef;
+ Uint32 nodeId;
+ Uint32 startType;
+};
+
+class StartPermConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+
+ Uint32 startingNodeId;
+ Uint32 systemFailureNo;
+};
+
+class StartPermRef {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+
+ Uint32 startingNodeId;
+ Uint32 errorCode;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/StartRec.hpp b/ndb/include/kernel/signaldata/StartRec.hpp
new file mode 100644
index 00000000000..f8a4e01a094
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StartRec.hpp
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef START_REC_HPP
+#define START_REC_HPP
+
+#include "SignalData.hpp"
+
+class StartRecReq {
+ /**
+ * Sender(s)
+ */
+ friend class Dbdih;
+ /**
+ * Receiver(s)
+ */
+ friend class Dblqh;
+
+ friend bool printSTART_REC_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+private:
+
+ Uint32 receivingNodeId;
+ Uint32 senderRef;
+ Uint32 keepGci;
+ Uint32 lastCompletedGci;
+ Uint32 newestGci;
+};
+
+class StartRecConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+ /**
+ * Receiver(s)
+ */
+ friend class Dbdih;
+
+ friend bool printSTART_REC_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 1 );
+private:
+
+ Uint32 startingNodeId;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/StartTo.hpp b/ndb/include/kernel/signaldata/StartTo.hpp
new file mode 100644
index 00000000000..5aecef6275d
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StartTo.hpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef START_TO_HPP
+#define START_TO_HPP
+
+class StartToReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 5 );
+private:
+ Uint32 userPtr;
+ BlockReference userRef;
+ Uint32 startingNodeId;
+ Uint32 nodeTakenOver;
+ bool nodeRestart;
+};
+
+class StartToConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+private:
+
+ Uint32 userPtr;
+ Uint32 sendingNodeId;
+ Uint32 startingNodeId;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/StopMe.hpp b/ndb/include/kernel/signaldata/StopMe.hpp
new file mode 100644
index 00000000000..51d944a3b96
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StopMe.hpp
@@ -0,0 +1,70 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef STOP_ME_HPP
+#define STOP_ME_HPP
+
+/**
+ * This signal is sent by ndbcntr to local DIH
+ *
+ * If local DIH then sends it to all DIH's
+ *
+ * @see StopPermReq
+ * @see StartMeReq
+ * @see StartPermReq
+ */
+class StopMeReq {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Sender
+ */
+ friend class Ndbcntr;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+
+ Uint32 senderRef;
+ Uint32 senderData;
+};
+
+class StopMeConf {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderRef;
+ Uint32 senderData;
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/StopPerm.hpp b/ndb/include/kernel/signaldata/StopPerm.hpp
new file mode 100644
index 00000000000..95fb82c8cde
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StopPerm.hpp
@@ -0,0 +1,96 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef STOP_PERM_HPP
+#define STOP_PERM_HPP
+
+/**
+ * This signal is sent by ndbcntr to local DIH
+ *
+ * If local DIH is not master, it forwards it to master DIH
+ * and start acting as a proxy
+ *
+ * @see StopMeReq
+ * @see StartMeReq
+ * @see StartPermReq
+ */
+class StopPermReq {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Sender
+ */
+ friend class Ndbcntr;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+public:
+
+ Uint32 senderRef;
+ Uint32 senderData;
+};
+
+class StopPermConf {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ Uint32 senderData;
+};
+
+class StopPermRef {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ enum ErrorCode {
+ StopOK = 0,
+ NodeStartInProgress = 1,
+ NodeShutdownInProgress = 2,
+ NF_CausedAbortOfStopProcedure = 3
+ };
+
+private:
+ Uint32 errorCode;
+ Uint32 senderData;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/StopReq.hpp b/ndb/include/kernel/signaldata/StopReq.hpp
new file mode 100644
index 00000000000..ea453ae115d
--- /dev/null
+++ b/ndb/include/kernel/signaldata/StopReq.hpp
@@ -0,0 +1,202 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef STOP_REQ_HPP
+#define STOP_REQ_HPP
+
+#include "SignalData.hpp"
+
+class StopReq
+{
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Sender
+ */
+ friend class MgmtSrvr;
+
+public:
+ STATIC_CONST( SignalLength = 9 );
+
+public:
+ Uint32 senderRef;
+ Uint32 senderData;
+
+ Uint32 requestInfo;
+ Uint32 singleuser; // Indicates whether or not to enter
+ // single user mode.
+ // Only in conjunction with system stop
+ Uint32 singleUserApi; // allowed api in singleuser
+
+ Int32 apiTimeout; // Timeout before api transactions are refused
+ Int32 transactionTimeout; // Timeout before transactions are aborted
+ Int32 readOperationTimeout; // Timeout before read operations are aborted
+ Int32 operationTimeout; // Timeout before all operations are aborted
+
+ static void setSystemStop(Uint32 & requestInfo, bool value);
+ static void setPerformRestart(Uint32 & requestInfo, bool value);
+ static void setNoStart(Uint32 & requestInfo, bool value);
+ static void setInitialStart(Uint32 & requestInfo, bool value);
+ static void setEscalateOnNodeFail(Uint32 & requestInfo, bool value);
+ /**
+ * Don't perform "graceful" shutdown/restart...
+ */
+ static void setStopAbort(Uint32 & requestInfo, bool value);
+
+ static bool getSystemStop(const Uint32 & requestInfo);
+ static bool getPerformRestart(const Uint32 & requestInfo);
+ static bool getNoStart(const Uint32 & requestInfo);
+ static bool getInitialStart(const Uint32 & requestInfo);
+ static bool getEscalateOnNodeFail(const Uint32 & requestInfo);
+ static bool getStopAbort(const Uint32 & requestInfo);
+};
+
+class StopRef
+{
+ /**
+ * Reciver(s)
+ */
+ friend class MgmtSrvr;
+
+ /**
+ * Sender
+ */
+ friend class Ndbcntr;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ enum ErrorCode {
+ OK = 0,
+ NodeShutdownInProgress = 1,
+ SystemShutdownInProgress = 2,
+ NodeShutdownWouldCauseSystemCrash = 3
+ };
+
+public:
+ Uint32 senderData;
+ Uint32 errorCode;
+};
+
+inline
+bool
+StopReq::getSystemStop(const Uint32 & requestInfo)
+{
+ return requestInfo & 1;
+}
+
+inline
+bool
+StopReq::getPerformRestart(const Uint32 & requestInfo)
+{
+ return requestInfo & 2;
+}
+
+inline
+bool
+StopReq::getNoStart(const Uint32 & requestInfo)
+{
+ return requestInfo & 4;
+}
+
+inline
+bool
+StopReq::getInitialStart(const Uint32 & requestInfo)
+{
+ return requestInfo & 8;
+}
+
+inline
+bool
+StopReq::getEscalateOnNodeFail(const Uint32 & requestInfo)
+{
+ return requestInfo & 16;
+}
+
+inline
+bool
+StopReq::getStopAbort(const Uint32 & requestInfo)
+{
+ return requestInfo & 32;
+}
+
+
+inline
+void
+StopReq::setSystemStop(Uint32 & requestInfo, bool value)
+{
+ if(value)
+ requestInfo |= 1;
+ else
+ requestInfo &= ~1;
+}
+
+inline
+void
+StopReq::setPerformRestart(Uint32 & requestInfo, bool value)
+{
+ if(value)
+ requestInfo |= 2;
+ else
+ requestInfo &= ~2;
+}
+
+inline
+void
+StopReq::setNoStart(Uint32 & requestInfo, bool value)
+{
+ if(value)
+ requestInfo |= 4;
+ else
+ requestInfo &= ~4;
+}
+
+inline
+void
+StopReq::setInitialStart(Uint32 & requestInfo, bool value)
+{
+ if(value)
+ requestInfo |= 8;
+ else
+ requestInfo &= ~8;
+}
+
+inline
+void
+StopReq::setEscalateOnNodeFail(Uint32 & requestInfo, bool value)
+{
+ if(value)
+ requestInfo |= 16;
+ else
+ requestInfo &= ~16;
+}
+
+inline
+void
+StopReq::setStopAbort(Uint32 & requestInfo, bool value)
+{
+ if(value)
+ requestInfo |= 32;
+ else
+ requestInfo &= ~32;
+}
+
+
+#endif
+
diff --git a/ndb/include/kernel/signaldata/SumaImpl.hpp b/ndb/include/kernel/signaldata/SumaImpl.hpp
new file mode 100644
index 00000000000..089132cd9aa
--- /dev/null
+++ b/ndb/include/kernel/signaldata/SumaImpl.hpp
@@ -0,0 +1,619 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SUMA_IMPL_HPP
+#define SUMA_IMPL_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+
+
+class SubCreateReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+ friend class SumaParticipant;
+
+ friend bool printSUB_CREATE_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+ enum SubscriptionType {
+ SingleTableScan = 1, //
+ DatabaseSnapshot = 2, // All tables/all data (including new ones)
+ TableEvent = 3, //
+ SelectiveTableSnapshot = 4, // User defines tables
+ RemoveFlags = 0xff,
+ GetFlags = 0xff << 16,
+ AddTableFlag = 0x1 << 16,
+ RestartFlag = 0x2 << 16
+ };
+
+ Uint32 subscriberRef;
+ Uint32 subscriberData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+ union {
+ Uint32 tableId; // Used when doing SingelTableScan
+ };
+ SECTION( ATTRIBUTE_LIST = 0); // Used when doing SingelTableScan
+ SECTION( TABLE_LIST = 1 );
+
+};
+
+class SubCreateRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+ friend class SumaParticipant;
+
+ friend bool printSUB_CREATE_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+
+ Uint32 subscriberRef;
+ Uint32 subscriberData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriptionType;
+ Uint32 err;
+
+ SECTION( ATTRIBUTE_LIST = 0); // Used when doing SingelTableScan
+ union {
+ Uint32 tableId; // Used when doing SingelTableScan
+ };
+};
+
+class SubCreateConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+ friend class SumaParticipant;
+
+ friend bool printSUB_CREATE_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ Uint32 subscriberData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+};
+
+class SubscriptionData {
+public:
+ enum Part {
+ MetaData = 1,
+ TableData = 2
+ };
+};
+
+class SubStartReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Suma;
+
+ friend bool printSUB_START_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 6 );
+ STATIC_CONST( SignalLength2 = SignalLength+1 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 part; // SubscriptionData::Part
+ Uint32 subscriberData;
+ Uint32 subscriberRef;
+};
+
+class SubStartRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Suma;
+
+ friend bool printSUB_START_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ enum ErrorCode {
+ Undefined = 0,
+ NF_FakeErrorREF = 11,
+ Busy = 701,
+ Temporary = 0x1 << 16
+ };
+ bool isTemporary() const;
+ void setTemporary();
+ ErrorCode setTemporary(ErrorCode ec);
+
+ STATIC_CONST( SignalLength = 7 );
+ STATIC_CONST( SignalLength2 = SignalLength+1 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 part; // SubscriptionData::Part
+ Uint32 subscriberData;
+ union { // do not change the order here!
+ Uint32 err;
+ Uint32 errorCode;
+ };
+ // with SignalLength2
+ Uint32 subscriberRef;
+};
+inline bool SubStartRef::isTemporary() const
+{ return (errorCode & SubStartRef::Temporary) > 0; };
+inline void SubStartRef::setTemporary()
+{ errorCode |= SubStartRef::Temporary; };
+inline SubStartRef::ErrorCode SubStartRef::setTemporary(ErrorCode ec)
+{ return (SubStartRef::ErrorCode)
+ (errorCode = ((Uint32) ec | (Uint32)SubStartRef::Temporary)); };
+
+class SubStartConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printSUB_START_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 7 );
+ STATIC_CONST( SignalLength2 = SignalLength+1 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 firstGCI;
+ Uint32 part; // SubscriptionData::Part
+ Uint32 subscriberData;
+ // with SignalLength2
+ Uint32 subscriberRef;
+};
+
+class SubStopReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Suma;
+
+ friend bool printSUB_STOP_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 7 );
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 part; // SubscriptionData::Part
+ Uint32 subscriberData;
+ Uint32 subscriberRef;
+};
+
+class SubStopRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Suma;
+
+ friend bool printSUB_STOP_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ enum ErrorCode {
+ Undefined = 0,
+ NF_FakeErrorREF = 11,
+ Busy = 701,
+ Temporary = 0x1 << 16
+ };
+ bool isTemporary() const;
+ void setTemporary();
+ ErrorCode setTemporary(ErrorCode ec);
+
+ STATIC_CONST( SignalLength = 8 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 part; // SubscriptionData::Part
+ Uint32 subscriberData;
+ Uint32 subscriberRef;
+ union {
+ Uint32 err;
+ Uint32 errorCode;
+ };
+};
+inline bool SubStopRef::isTemporary() const
+{ return (errorCode & SubStopRef::Temporary) > 0; };
+inline void SubStopRef::setTemporary()
+{ errorCode |= SubStopRef::Temporary; };
+inline SubStopRef::ErrorCode SubStopRef::setTemporary(ErrorCode ec)
+{ return (SubStopRef::ErrorCode)
+ (errorCode = ((Uint32) ec | (Uint32)SubStopRef::Temporary)); };
+
+class SubStopConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+
+ friend bool printSUB_STOP_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 7 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 part; // SubscriptionData::Part
+ Uint32 subscriberData;
+ Uint32 subscriberRef;
+};
+
+class SubSyncReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Suma;
+ friend class Grep;
+
+ friend bool printSUB_SYNC_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+public:
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 subscriberData;
+ Uint32 part; // SubscriptionData::Part
+};
+
+class SubSyncRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Suma;
+ friend class Grep;
+
+ friend bool printSUB_SYNC_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ enum ErrorCode {
+ Undefined = 0,
+ Temporary = 0x1 << 16
+ };
+ STATIC_CONST( SignalLength = 5 );
+
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 part; // SubscriptionData::Part
+ Uint32 subscriberData;
+ union {
+ Uint32 errorCode;
+ Uint32 err;
+ };
+};
+
+class SubSyncConf {
+
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Suma;
+ friend class Grep;
+
+ friend bool printSUB_SYNC_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 part; // SubscriptionData::Part
+ Uint32 subscriberData;
+};
+
+class SubMetaData {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class SumaParticipant;
+ friend class Grep;
+
+ friend bool printSUB_META_DATA(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+ SECTION( DICT_TAB_INFO = 0 );
+
+ Uint32 gci;
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+ union {
+ Uint32 tableId;
+ };
+};
+
+class SubTableData {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class SumaParticipant;
+ friend class Grep;
+
+ friend bool printSUB_TABLE_DATA(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 7 );
+
+ enum LogType {
+ SCAN = 1,
+ LOG = 2,
+ REMOVE_FLAGS = 0xff,
+ GCINOTCONSISTENT = 0x1 << 16
+ };
+
+ void setGCINotConsistent() { logType |= (Uint32)GCINOTCONSISTENT; };
+ bool isGCIConsistent()
+ { return (logType & (Uint32)GCINOTCONSISTENT) == 0 ? true : false; };
+
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+ Uint32 gci;
+ Uint32 tableId;
+ Uint32 operation;
+ Uint32 noOfAttributes;
+ Uint32 dataSize;
+ Uint32 logType;
+};
+
+class SubSyncContinueReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class SumaParticipant;
+ friend class Grep;
+ friend class Trix;
+
+ friend bool printSUB_SYNC_CONTINUE_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+ Uint32 noOfRowsSent;
+};
+
+class SubSyncContinueRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class SumaParticipant;
+ friend class Grep;
+ friend class Trix;
+
+ friend bool printSUB_SYNC_CONTINUE_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+};
+
+class SubSyncContinueConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class SumaParticipant;
+ friend class Grep;
+ friend class Trix;
+
+ friend bool printSUB_SYNC_CONTINUE_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+};
+
+class SubGcpCompleteRep {
+
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Dbdih;
+ friend class SumaParticipant;
+ friend class Grep;
+ friend class Trix;
+
+ friend bool printSUB_GCP_COMPLETE_REP(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ Uint32 gci;
+ Uint32 senderRef;
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+};
+
+class SubGcpCompleteAcc {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+public:
+ STATIC_CONST( SignalLength = SubGcpCompleteRep::SignalLength );
+
+ SubGcpCompleteRep rep;
+};
+
+class SubRemoveReq {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+ friend class SumaParticipant;
+
+ friend bool printSUB_REMOVE_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+};
+
+class SubRemoveRef {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+ friend class SumaParticipant;
+
+ friend bool printSUB_REMOVE_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+ enum ErrorCode {
+ Undefined = 0,
+ NF_FakeErrorREF = 11,
+ Busy = 701,
+ Temporary = 0x1 << 16
+ };
+ bool isTemporary() const;
+ void setTemporary();
+ ErrorCode setTemporary(ErrorCode ec);
+
+ Uint32 senderRef;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ union {
+ Uint32 err;
+ Uint32 errorCode;
+ };
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+};
+inline bool SubRemoveRef::isTemporary() const
+{ return (err & SubRemoveRef::Temporary) > 0; };
+inline void SubRemoveRef::setTemporary()
+{ err |= SubRemoveRef::Temporary; };
+inline SubRemoveRef::ErrorCode SubRemoveRef::setTemporary(ErrorCode ec)
+{ return (SubRemoveRef::ErrorCode)
+ (errorCode = ((Uint32) ec | (Uint32)SubRemoveRef::Temporary)); };
+
+class SubRemoveConf {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Grep;
+ friend class SumaParticipant;
+
+ friend bool printSUB_REMOVE_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+ Uint32 senderRef;
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 err;
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+
+};
+
+
+class CreateSubscriptionIdReq {
+ friend class Grep;
+ friend class SumaParticipant;
+
+ friend bool printCREATE_SUBSCRIPTION_ID_REQ(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+};
+
+
+class CreateSubscriptionIdConf {
+ friend class Grep;
+ friend class SumaParticipant;
+
+ friend bool printCREATE_SUBSCRIPTION_ID_CONF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+};
+
+
+class CreateSubscriptionIdRef {
+ friend class Grep;
+ friend class SumaParticipant;
+
+ friend bool printCREATE_SUBSCRIPTION_ID_REF(FILE *, const Uint32 *,
+ Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ Uint32 subscriptionId;
+ Uint32 subscriptionKey;
+ Uint32 err;
+ union { // Haven't decide what to call it
+ Uint32 senderData;
+ Uint32 subscriberData;
+ };
+};
+
+class SumaStartMe {
+public:
+ STATIC_CONST( SignalLength = 1 );
+ Uint32 unused;
+};
+
+class SumaHandoverReq {
+public:
+ STATIC_CONST( SignalLength = 1 );
+ Uint32 gci;
+};
+
+class SumaHandoverConf {
+public:
+ STATIC_CONST( SignalLength = 1 );
+ Uint32 gci;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/SystemError.hpp b/ndb/include/kernel/signaldata/SystemError.hpp
new file mode 100644
index 00000000000..7b4d47c5c2e
--- /dev/null
+++ b/ndb/include/kernel/signaldata/SystemError.hpp
@@ -0,0 +1,60 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SYSTEM_ERROR_HPP
+#define SYSTEM_ERROR_HPP
+
+#include "SignalData.hpp"
+
+class SystemError {
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+
+ /**
+ * Sender
+ */
+ friend class Dbtc;
+ friend class Dbdih;
+
+ /**
+ * For printing
+ */
+ friend bool printSYSTEM_ERROR(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum ErrorCode {
+ ScanfragStateError = 1,
+ ScanfragTimeout = 2,
+ GCPStopDetected = 3,
+ StartInProgressError = 4,
+ CopyFragRefError = 5,
+ TestStopOnError = 6
+ };
+
+private:
+ Uint32 errorRef;
+ Uint32 errorCode;
+ Uint32 data1;
+ Uint32 data2;
+};
+
+#endif
+
diff --git a/ndb/include/kernel/signaldata/TamperOrd.hpp b/ndb/include/kernel/signaldata/TamperOrd.hpp
new file mode 100644
index 00000000000..eb6cd47b093
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TamperOrd.hpp
@@ -0,0 +1,40 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TAMPERORD_H
+#define TAMPERORD_H
+
+#include "SignalData.hpp"
+
+class TamperOrd {
+ /**
+ * Sender
+ */
+ friend class MgmtSrvr;
+
+ /**
+ * Receiver
+ */
+ friend class Cmvmi;
+
+private:
+ STATIC_CONST( SignalLength = 1 );
+
+ UintR errorNo;
+};
+
+#endif // TAMPERORD_H
+
diff --git a/ndb/include/kernel/signaldata/TcCommit.hpp b/ndb/include/kernel/signaldata/TcCommit.hpp
new file mode 100644
index 00000000000..43eb7be1c39
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcCommit.hpp
@@ -0,0 +1,74 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TCCOMMITCONF_HPP
+#define TCCOMMITCONF_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * This is signal is sent from TC to API
+ * It means that the transaction was committed
+ */
+class TcCommitConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndb;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+private:
+
+ /**
+ * apiConnectPtr
+ *
+ * Bit 0 (lowest) is used as indicator
+ * if == 1 then tc expects a commit ack
+ */
+ Uint32 apiConnectPtr;
+
+ Uint32 transId1;
+ Uint32 transId2;
+};
+
+class TcCommitRef {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * Reciver(s)
+ */
+ friend class NdbConnection;
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+private:
+
+ Uint32 apiConnectPtr;
+ Uint32 transId1;
+ Uint32 transId2;
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TcContinueB.hpp b/ndb/include/kernel/signaldata/TcContinueB.hpp
new file mode 100644
index 00000000000..7a093b457e8
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcContinueB.hpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TC_CONTINUEB_H
+#define TC_CONTINUEB_H
+
+#include "SignalData.hpp"
+
+class TcContinueB {
+ /**
+ * Sender(s)/Reciver(s)
+ */
+ friend class Dbtc;
+private:
+ enum {
+ ZRETURN_FROM_QUEUED_DELIVERY = 1,
+ ZCOMPLETE_TRANS_AT_TAKE_OVER = 2,
+ ZCONTINUE_TIME_OUT_CONTROL = 3,
+ ZNODE_TAKE_OVER_COMPLETED = 4,
+ ZINITIALISE_RECORDS = 5,
+ ZSEND_COMMIT_LOOP = 6,
+ ZSEND_COMPLETE_LOOP = 7,
+ ZHANDLE_FAILED_API_NODE = 8,
+ ZTRANS_EVENT_REP = 9,
+ ZABORT_BREAK = 10,
+ ZABORT_TIMEOUT_BREAK = 11,
+ ZCONTINUE_TIME_OUT_FRAG_CONTROL = 12,
+ ZHANDLE_FAILED_API_NODE_REMOVE_MARKERS = 13,
+ ZWAIT_ABORT_ALL = 14,
+ ZCHECK_SCAN_ACTIVE_FAILED_LQH = 15,
+ CHECK_WAIT_DROP_TAB_FAILED_LQH = 16,
+ TRIGGER_PENDING = 17
+ };
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TcHbRep.hpp b/ndb/include/kernel/signaldata/TcHbRep.hpp
new file mode 100644
index 00000000000..58ab015917a
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcHbRep.hpp
@@ -0,0 +1,64 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TC_HB_REP_H
+#define TC_HB_REP_H
+
+#include "SignalData.hpp"
+
+/**
+ * @class TcHbRep
+ * @brief Order tc refresh(exetend) the timeout counters for this
+ * transaction
+ *
+ * - SENDER: API
+ * - RECEIVER: TC
+ */
+class TcHbRep {
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtc; // Receiver
+
+ /**
+ * Sender(s)
+ */
+ friend class NdbConnection;
+
+ /**
+ * For printing
+ */
+ friend bool printTC_HBREP(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+
+ Uint32 apiConnectPtr; // DATA 0
+ UintR transId1; // DATA 1
+ UintR transId2; // DATA 2
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TcIndx.hpp b/ndb/include/kernel/signaldata/TcIndx.hpp
new file mode 100644
index 00000000000..764d4e9fcd7
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcIndx.hpp
@@ -0,0 +1,528 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TC_INDX_H
+#define TC_INDX_H
+
+#include "SignalData.hpp"
+
+class TcIndxReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc; // Reciver
+
+ /**
+ * Sender(s)
+ */
+ friend class NdbIndexOperation;
+
+ /**
+ * For printing
+ */
+ friend bool printTCINDXREQ(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( StaticLength = 8 );
+ STATIC_CONST( SignalLength = 25 );
+ STATIC_CONST( MaxKeyInfo = 8 );
+ STATIC_CONST( MaxAttrInfo = 5 );
+
+private:
+
+ enum CommitType {
+ CommitIfFailFree = 0,
+ TryCommit = 1,
+ CommitAsMuchAsPossible = 2
+ };
+
+ /**
+ * DATA VARIABLES
+ */
+//-------------------------------------------------------------
+// Unconditional part. First 8 words
+//-------------------------------------------------------------
+ UintR apiConnectPtr; // DATA 0
+ UintR senderData; // DATA 1
+ UintR attrLen; // DATA 2 (including API Version)
+ UintR indexId; // DATA 3
+ UintR requestInfo; // DATA 4
+ UintR indexSchemaVersion; // DATA 5
+ UintR transId1; // DATA 6
+ UintR transId2; // DATA 7
+//-------------------------------------------------------------
+// Conditional part. Those four words will be sent only if their
+// indicator is set.
+//-------------------------------------------------------------
+ UintR scanInfo; // DATA 8
+ UintR distrGroupHashValue; // DATA 9
+ UintR distributionKeySize; // DATA 10
+ UintR storedProcId; // DATA 11
+
+//-------------------------------------------------------------
+// Variable sized key and attrinfo part. Those will be placed to
+// pack the signal in an appropriate manner.
+//-------------------------------------------------------------
+ UintR keyInfo[MaxKeyInfo]; // DATA 12 - 19
+ UintR attrInfo[MaxAttrInfo]; // DATA 20 - 24
+
+ static Uint8 getAPIVersion(const UintR & attrLen);
+
+ /**
+ * Get:ers for requestInfo
+ */
+ static Uint8 getCommitFlag(const UintR & requestInfo);
+ static Uint8 getCommitType(const UintR & requestInfo);
+ static Uint8 getStartFlag(const UintR & requestInfo);
+ static Uint8 getSimpleFlag(const UintR & requestInfo);
+ static Uint8 getDirtyFlag(const UintR & requestInfo);
+ static Uint8 getInterpretedFlag(const UintR & requestInfo);
+ static Uint8 getDistributionGroupFlag(const UintR & requestInfo);
+ static Uint8 getDistributionGroupTypeFlag(const UintR & requestInfo);
+ static Uint8 getDistributionKeyFlag(const UintR & requestInfo);
+ static Uint8 getScanIndFlag(const UintR & requestInfo);
+
+ static Uint8 getOperationType(const UintR & requestInfo);
+
+ static Uint16 getIndexLength(const UintR & requestInfo);
+ static Uint8 getAIInTcIndxReq(const UintR & requestInfo);
+
+ /**
+ * Get:ers for scanInfo
+ */
+
+ static void setAPIVersion(UintR & attrLen, Uint16 apiVersion);
+
+ /**
+ * Set:ers for requestInfo
+ */
+ static void clearRequestInfo(UintR & requestInfo);
+ static void setCommitType(UintR & requestInfo, Uint32 type);
+ static void setCommitFlag(UintR & requestInfo, Uint32 flag);
+ static void setStartFlag(UintR & requestInfo, Uint32 flag);
+ static void setSimpleFlag(UintR & requestInfo, Uint32 flag);
+ static void setDirtyFlag(UintR & requestInfo, Uint32 flag);
+ static void setInterpretedFlag(UintR & requestInfo, Uint32 flag);
+ static void setDistributionGroupFlag(UintR & requestInfo, Uint32 flag);
+ static void setDistributionGroupTypeFlag(UintR & requestInfo, Uint32 flag);
+ static void setDistributionKeyFlag(UintR & requestInfo, Uint32 flag);
+ static void setScanIndFlag(UintR & requestInfo, Uint32 flag);
+
+ static void setOperationType(UintR & requestInfo, Uint32 type);
+
+ static void setIndexLength(UintR & requestInfo, Uint32 len);
+ static void setAIInTcIndxReq(UintR & requestInfo, Uint32 len);
+
+ /**
+ * Set:ers for scanInfo
+ */
+
+};
+
+#define API_VER_NO_SHIFT (16)
+#define API_VER_NO_MASK (65535)
+
+/**
+ * Request Info
+ *
+ a = Attr Info in TCINDXREQ - 3 Bits -> Max 7 (Bit 16-18)
+ b = Distribution Key Ind - 1 Bit 2
+ c = Commit Indicator - 1 Bit 4
+ d = Dirty Indicator - 1 Bit 0
+ e = Scan Indicator - 1 Bit 14
+ g = Distribution Group Ind - 1 Bit 1
+ i = Interpreted Indicator - 1 Bit 15
+ k = Index lengt - 12 Bits -> Max 4095 (Bit 20 - 31)
+ o = Operation Type - 3 Bits -> Max 7 (Bit 5-7)
+ p = Simple Indicator - 1 Bit 8
+ s = Start Indicator - 1 Bit 11
+ t = Distribution GroupType - 1 Bit 3
+ y = Commit Type - 2 Bit 12-13
+ x = Last Op in execute - 1 Bit 19
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ dgbtcooop syyeiaaa-kkkkkkkkkkkk
+*/
+
+#define COMMIT_SHIFT (4)
+#define START_SHIFT (11)
+#define SIMPLE_SHIFT (8)
+#define DIRTY_SHIFT (0)
+#define INTERPRETED_SHIFT (15)
+#define DISTR_GROUP_SHIFT (1)
+#define DISTR_GROUP_TYPE_SHIFT (3)
+#define DISTR_KEY_SHIFT (2)
+#define SCAN_SHIFT (14)
+
+#define OPERATION_SHIFT (5)
+#define OPERATION_MASK (7)
+
+#define AINFO_SHIFT (16)
+#define AINFO_MASK (7)
+
+#define INDEX_LEN_SHIFT (20)
+#define INDEX_LEN_MASK (4095)
+
+#define COMMIT_TYPE_SHIFT (12)
+#define COMMIT_TYPE_MASK (3)
+
+#define LAST_OP_IN_EXEC_SHIFT (19)
+
+/**
+ * Scan Info
+ *
+
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+
+*/
+
+inline
+Uint8
+TcIndxReq::getCommitFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> COMMIT_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcIndxReq::getCommitType(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> COMMIT_TYPE_SHIFT) & COMMIT_TYPE_MASK);
+}
+
+inline
+Uint8
+TcIndxReq::getStartFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> START_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcIndxReq::getSimpleFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> SIMPLE_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcIndxReq::getDirtyFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> DIRTY_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcIndxReq::getInterpretedFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> INTERPRETED_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcIndxReq::getDistributionGroupFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> DISTR_GROUP_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcIndxReq::getDistributionGroupTypeFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> DISTR_GROUP_TYPE_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcIndxReq::getDistributionKeyFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> DISTR_KEY_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcIndxReq::getScanIndFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> SCAN_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcIndxReq::getOperationType(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> OPERATION_SHIFT) & OPERATION_MASK);
+}
+
+inline
+Uint16
+TcIndxReq::getIndexLength(const UintR & requestInfo){
+ return (Uint16)((requestInfo >> INDEX_LEN_SHIFT) & INDEX_LEN_MASK);
+}
+
+inline
+Uint8
+TcIndxReq::getAIInTcIndxReq(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> AINFO_SHIFT) & AINFO_MASK);
+}
+
+inline
+void
+TcIndxReq::clearRequestInfo(UintR & requestInfo){
+ requestInfo = 0;
+}
+
+inline
+void
+TcIndxReq::setCommitType(UintR & requestInfo, Uint32 type){
+ ASSERT_MAX(type, COMMIT_TYPE_MASK, "TcIndxReq::setCommitType");
+ requestInfo |= (type << COMMIT_TYPE_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setCommitFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxReq::setCommitFlag");
+ requestInfo &= ~(1 << COMMIT_SHIFT);
+ requestInfo |= (flag << COMMIT_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setStartFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxReq::setStartFlag");
+ requestInfo &= ~(1 << START_SHIFT);
+ requestInfo |= (flag << START_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setSimpleFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxReq::setSimpleFlag");
+ requestInfo &= ~(1 << SIMPLE_SHIFT);
+ requestInfo |= (flag << SIMPLE_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setDirtyFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxReq::setDirtyFlag");
+ requestInfo &= ~(1 << DIRTY_SHIFT);
+ requestInfo |= (flag << DIRTY_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setInterpretedFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxReq::setInterpretedFlag");
+ requestInfo &= ~(1 << INTERPRETED_SHIFT);
+ requestInfo |= (flag << INTERPRETED_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setDistributionGroupTypeFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxReq::setDistributionGroupTypeFlag");
+ requestInfo &= ~(1 << DISTR_GROUP_TYPE_SHIFT);
+ requestInfo |= (flag << DISTR_GROUP_TYPE_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setDistributionGroupFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxReq::setDistributionGroupFlag");
+ requestInfo &= ~(1 << DISTR_GROUP_SHIFT);
+ requestInfo |= (flag << DISTR_GROUP_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setDistributionKeyFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxReq::setDistributionKeyFlag");
+ requestInfo &= ~(1 << DISTR_KEY_SHIFT);
+ requestInfo |= (flag << DISTR_KEY_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setScanIndFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxReq::setScanIndFlag");
+ requestInfo &= ~(1 << SCAN_SHIFT);
+ requestInfo |= (flag << SCAN_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setOperationType(UintR & requestInfo, Uint32 type){
+ ASSERT_MAX(type, OPERATION_MASK, "TcIndxReq::setOperationType");
+ requestInfo |= (type << OPERATION_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setIndexLength(UintR & requestInfo, Uint32 len){
+ ASSERT_MAX(len, INDEX_LEN_MASK, "TcIndxReq::setKeyLength");
+ requestInfo |= (len << INDEX_LEN_SHIFT);
+}
+
+inline
+void
+TcIndxReq::setAIInTcIndxReq(UintR & requestInfo, Uint32 len){
+ ASSERT_MAX(len, AINFO_MASK, "TcIndxReq::setAIInTcIndxReq");
+ requestInfo |= (len << AINFO_SHIFT);
+}
+
+inline
+Uint8
+TcIndxReq::getAPIVersion(const UintR & anAttrLen){
+ return (Uint16)((anAttrLen >> API_VER_NO_SHIFT) & API_VER_NO_MASK);
+}
+
+inline
+void
+TcIndxReq::setAPIVersion(UintR & anAttrLen, Uint16 apiVersion){
+// ASSERT_MAX(apiVersion, API_VER_NO_MASK, "TcIndxReq::setAPIVersion");
+ anAttrLen |= (apiVersion << API_VER_NO_SHIFT);
+}
+
+class TcIndxConf {
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndb;
+ friend class NdbConnection;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * For printing
+ */
+ friend bool printTCINDXCONF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ /**
+ * DATA VARIABLES
+ */
+ //-------------------------------------------------------------
+ // Unconditional part. First 5 words
+ //-------------------------------------------------------------
+
+ Uint32 apiConnectPtr;
+ Uint32 gci;
+ Uint32 confInfo;
+ Uint32 transId1;
+ Uint32 transId2;
+
+ struct OperationConf {
+ Uint32 apiOperationPtr;
+ Uint32 attrInfoLen;
+ };
+ //-------------------------------------------------------------
+ // Operations confirmations,
+ // No of actually sent = getNoOfOperations(confInfo)
+ //-------------------------------------------------------------
+ OperationConf operations[10];
+
+ /**
+ * Get:ers for confInfo
+ */
+ static Uint32 getNoOfOperations(const Uint32 & confInfo);
+ static Uint32 getCommitFlag(const Uint32 & confInfo);
+ static bool getMarkerFlag(const Uint32 & confInfo);
+
+ /**
+ * Set:ers for confInfo
+ */
+ static void setCommitFlag(Uint32 & confInfo, Uint8 flag);
+ static void setNoOfOperations(Uint32 & confInfo, Uint32 noOfOps);
+ static void setMarkerFlag(Uint32 & confInfo, Uint32 flag);
+};
+
+inline
+Uint32
+TcIndxConf::getNoOfOperations(const Uint32 & confInfo){
+ return confInfo & 65535;
+}
+
+inline
+Uint32
+TcIndxConf::getCommitFlag(const Uint32 & confInfo){
+ return ((confInfo >> 16) & 1);
+}
+
+inline
+bool
+TcIndxConf::getMarkerFlag(const Uint32 & confInfo){
+ const Uint32 bits = 3 << 16; // Marker only valid when doing commit
+ return (confInfo & bits) == bits;
+}
+
+inline
+void
+TcIndxConf::setNoOfOperations(Uint32 & confInfo, Uint32 noOfOps){
+ ASSERT_MAX(noOfOps, 65535, "TcIndxConf::setNoOfOperations");
+ confInfo |= noOfOps;
+}
+
+inline
+void
+TcIndxConf::setCommitFlag(Uint32 & confInfo, Uint8 flag){
+ ASSERT_BOOL(flag, "TcIndxConf::setCommitFlag");
+ confInfo |= (flag << 16);
+}
+
+inline
+void
+TcIndxConf::setMarkerFlag(Uint32 & confInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcIndxConf::setMarkerFlag");
+ confInfo |= (flag << 17);
+}
+
+class TcIndxRef {
+
+ /**
+ * Reciver(s)
+ */
+ friend class NdbIndexOperation;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * For printing
+ */
+ friend bool printTCINDXREF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ /**
+ * Length of signal
+ */
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 connectPtr;
+ Uint32 transId[2];
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TcKeyConf.hpp b/ndb/include/kernel/signaldata/TcKeyConf.hpp
new file mode 100644
index 00000000000..bfd684b4af4
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcKeyConf.hpp
@@ -0,0 +1,131 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TC_KEY_CONF_H
+#define TC_KEY_CONF_H
+
+#include "SignalData.hpp"
+
+/**
+ *
+ */
+class TcKeyConf {
+ /**
+ * Reciver(s)
+ */
+ friend class Ndb;
+ friend class NdbConnection;
+ friend class Ndbcntr;
+ friend class DbUtil;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * For printing
+ */
+ friend bool printTCKEYCONF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( StaticLength = 5 );
+ STATIC_CONST( OperationLength = 2 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ //-------------------------------------------------------------
+ // Unconditional part. First 5 words
+ //-------------------------------------------------------------
+
+ Uint32 apiConnectPtr;
+ Uint32 gci;
+ Uint32 confInfo;
+ Uint32 transId1;
+ Uint32 transId2;
+
+ struct OperationConf {
+ Uint32 apiOperationPtr;
+ Uint32 attrInfoLen;
+ };
+ //-------------------------------------------------------------
+ // Operations confirmations,
+ // No of actually sent = getNoOfOperations(confInfo)
+ //-------------------------------------------------------------
+ OperationConf operations[10];
+
+ /**
+ * Get:ers for confInfo
+ */
+ static Uint32 getNoOfOperations(const Uint32 & confInfo);
+ static Uint32 getCommitFlag(const Uint32 & confInfo);
+ static bool getMarkerFlag(const Uint32 & confInfo);
+
+ /**
+ * Set:ers for confInfo
+ */
+ static void setCommitFlag(Uint32 & confInfo, Uint8 flag);
+ static void setNoOfOperations(Uint32 & confInfo, Uint32 noOfOps);
+ static void setMarkerFlag(Uint32 & confInfo, Uint32 flag);
+};
+
+inline
+Uint32
+TcKeyConf::getNoOfOperations(const Uint32 & confInfo){
+ return confInfo & 65535;
+}
+
+inline
+Uint32
+TcKeyConf::getCommitFlag(const Uint32 & confInfo){
+ return ((confInfo >> 16) & 1);
+}
+
+inline
+bool
+TcKeyConf::getMarkerFlag(const Uint32 & confInfo){
+ const Uint32 bits = 3 << 16; // Marker only valid when doing commit
+ return (confInfo & bits) == bits;
+}
+
+inline
+void
+TcKeyConf::setNoOfOperations(Uint32 & confInfo, Uint32 noOfOps){
+ ASSERT_MAX(noOfOps, 65535, "TcKeyConf::setNoOfOperations");
+ confInfo |= noOfOps;
+}
+
+inline
+void
+TcKeyConf::setCommitFlag(Uint32 & confInfo, Uint8 flag){
+ ASSERT_BOOL(flag, "TcKeyConf::setCommitFlag");
+ confInfo |= (flag << 16);
+}
+
+inline
+void
+TcKeyConf::setMarkerFlag(Uint32 & confInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyConf::setMarkerFlag");
+ confInfo |= (flag << 17);
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TcKeyFailConf.hpp b/ndb/include/kernel/signaldata/TcKeyFailConf.hpp
new file mode 100644
index 00000000000..d8207b63262
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcKeyFailConf.hpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TCKEYFAILCONF_HPP
+#define TCKEYFAILCONF_HPP
+
+#include <NodeBitmask.hpp>
+
+/**
+ * This is signal is sent from "Take-Over" TC after a node crash
+ * It means that the transaction was committed
+ */
+class TcKeyFailConf {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndb;
+ friend class NdbConnection;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+private:
+
+ /**
+ * apiConnectPtr
+ *
+ * Bit 0 (lowest) is used as indicator
+ * if == 1 then tc expects a commit ack
+ */
+ Uint32 apiConnectPtr;
+ Uint32 transId1;
+ Uint32 transId2;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TcKeyRef.hpp b/ndb/include/kernel/signaldata/TcKeyRef.hpp
new file mode 100644
index 00000000000..c773920713a
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcKeyRef.hpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TCKEYREF_HPP
+#define TCKEYREF_HPP
+
+#include "SignalData.hpp"
+
+class TcKeyRef {
+
+ /**
+ * Receiver(s)
+ */
+ friend class NdbOperation;
+ friend class Ndbcntr;
+ friend class DbUtil;
+
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class Dbtc;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ friend bool printTCKEYREF(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 connectPtr;
+ Uint32 transId[2];
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TcKeyReq.hpp b/ndb/include/kernel/signaldata/TcKeyReq.hpp
new file mode 100644
index 00000000000..df0a00da3e0
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcKeyReq.hpp
@@ -0,0 +1,547 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TC_KEY_REQ_H
+#define TC_KEY_REQ_H
+
+#include "SignalData.hpp"
+
+/**
+ * @class TcKeyReq
+ * @brief Contains KeyInfo and AttrInfo and is commonly followed by more signals
+ *
+ * - SENDER: API, NDBCNTR
+ * - RECEIVER: TC
+ */
+class TcKeyReq {
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtc; // Receiver
+
+ /**
+ * Sender(s)
+ */
+ friend class Ndbcntr;
+ friend class NdbOperation;
+ friend class NdbIndexOperation;
+ friend class DbUtil;
+
+ /**
+ * For printing
+ */
+ friend bool printTCKEYREQ(FILE *, const Uint32 *, Uint32, Uint16);
+ friend bool printTCINDXREQ(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ /**
+ * Length of signal
+ */
+ STATIC_CONST( StaticLength = 8 );
+ STATIC_CONST( SignalLength = 25 );
+ STATIC_CONST( MaxKeyInfo = 8 );
+ STATIC_CONST( MaxAttrInfo = 5 );
+ STATIC_CONST( MaxTotalAttrInfo = 0xFFFF );
+
+private:
+
+ enum AbortOption {
+ CommitIfFailFree = 0, AbortOnError = 0,
+ CommitAsMuchAsPossible = 2, IgnoreError = 2
+ };
+
+ typedef AbortOption CommitType;
+
+ /**
+ * DATA VARIABLES
+ */
+
+ // ----------------------------------------------------------------------
+ // Unconditional part = must be present in signal. First 8 words
+ // ----------------------------------------------------------------------
+ Uint32 apiConnectPtr; // DATA 0
+ union {
+ Uint32 senderData;
+ UintR apiOperationPtr; // DATA 1
+ };
+ /**
+ * ATTRIBUTE INFO (attrinfo) LENGTH
+ * This is the total length of all attribute info that is sent from
+ * the application as part of this operation.
+ * It includes all attribute info sent in possible attrinfo
+ * signals as well as the attribute info sent in TCKEYREQ.
+ */
+ UintR attrLen; // DATA 2 (also stores API Version)
+ UintR tableId; // DATA 3
+ UintR requestInfo; // DATA 4 Various transaction flags
+ UintR tableSchemaVersion; // DATA 5
+ UintR transId1; // DATA 6
+ UintR transId2; // DATA 7
+
+ // ----------------------------------------------------------------------
+ // Conditional part = can be present in signal.
+ // These four words will be sent only if their indicator is set.
+ // ----------------------------------------------------------------------
+ UintR scanInfo; // DATA 8 Various flags for scans
+ UintR distrGroupHashValue; // DATA 9
+ UintR distributionKeySize; // DATA 10
+ UintR storedProcId; // DATA 11
+
+ // ----------------------------------------------------------------------
+ // Variable sized KEY and ATTRINFO part.
+ // These will be placed to pack the signal in an appropriate manner.
+ // ----------------------------------------------------------------------
+ UintR keyInfo[MaxKeyInfo]; // DATA 12 - 19
+ UintR attrInfo[MaxAttrInfo]; // DATA 20 - 24
+
+ /**
+ * Get:ers for attrLen
+ */
+
+ static Uint16 getAPIVersion(const UintR & attrLen);
+ static Uint16 getAttrinfoLen(const UintR & attrLen);
+ static void setAPIVersion(UintR & attrLen, Uint16 apiVersion);
+ static void setAttrinfoLen(UintR & attrLen, Uint16 aiLen);
+
+
+ /**
+ * Get:ers for requestInfo
+ */
+ static Uint8 getCommitFlag(const UintR & requestInfo);
+ static Uint8 getAbortOption(const UintR & requestInfo);
+ static Uint8 getStartFlag(const UintR & requestInfo);
+ static Uint8 getSimpleFlag(const UintR & requestInfo);
+ static Uint8 getDirtyFlag(const UintR & requestInfo);
+ static Uint8 getInterpretedFlag(const UintR & requestInfo);
+ static Uint8 getDistributionGroupFlag(const UintR & requestInfo);
+ static Uint8 getDistributionGroupTypeFlag(const UintR & requestInfo);
+ static Uint8 getDistributionKeyFlag(const UintR & requestInfo);
+ static Uint8 getScanIndFlag(const UintR & requestInfo);
+ static Uint8 getOperationType(const UintR & requestInfo);
+ static Uint8 getExecuteFlag(const UintR & requestInfo);
+
+ static Uint16 getKeyLength(const UintR & requestInfo);
+ static Uint8 getAIInTcKeyReq(const UintR & requestInfo);
+ static Uint8 getExecutingTrigger(const UintR & requestInfo);
+
+ /**
+ * Get:ers for scanInfo
+ */
+ static Uint8 getTakeOverScanFlag(const UintR & scanInfo);
+ static Uint16 getTakeOverScanNode(const UintR & scanInfo);
+ static Uint16 getTakeOverScanInfo(const UintR & scanInfo);
+
+
+ /**
+ * Set:ers for requestInfo
+ */
+ static void clearRequestInfo(UintR & requestInfo);
+ static void setAbortOption(UintR & requestInfo, Uint32 type);
+ static void setCommitFlag(UintR & requestInfo, Uint32 flag);
+ static void setStartFlag(UintR & requestInfo, Uint32 flag);
+ static void setSimpleFlag(UintR & requestInfo, Uint32 flag);
+ static void setDirtyFlag(UintR & requestInfo, Uint32 flag);
+ static void setInterpretedFlag(UintR & requestInfo, Uint32 flag);
+ static void setDistributionGroupFlag(UintR & requestInfo, Uint32 flag);
+ static void setDistributionGroupTypeFlag(UintR & requestInfo, Uint32 flag);
+ static void setDistributionKeyFlag(UintR & requestInfo, Uint32 flag);
+ static void setScanIndFlag(UintR & requestInfo, Uint32 flag);
+ static void setExecuteFlag(UintR & requestInfo, Uint32 flag);
+ static void setOperationType(UintR & requestInfo, Uint32 type);
+
+ static void setKeyLength(UintR & requestInfo, Uint32 len);
+ static void setAIInTcKeyReq(UintR & requestInfo, Uint32 len);
+ static void setExecutingTrigger(UintR & requestInfo, Uint32 flag);
+
+ /**
+ * Set:ers for scanInfo
+ */
+ static void setTakeOverScanFlag(UintR & scanInfo, Uint8 flag);
+ static void setTakeOverScanNode(UintR & scanInfo, Uint16 node);
+ static void setTakeOverScanInfo(UintR & scanInfo, Uint16 aScanInfo);
+};
+
+/**
+ * Request Info
+ *
+ a = Attr Info in TCKEYREQ - 3 Bits -> Max 7 (Bit 16-18)
+ b = Distribution Key Ind - 1 Bit 2
+ c = Commit Indicator - 1 Bit 4
+ d = Dirty Indicator - 1 Bit 0
+ e = Scan Indicator - 1 Bit 14
+ f = Execute fired trigger - 1 Bit 19
+ g = Distribution Group Ind- 1 Bit 1
+ i = Interpreted Indicator - 1 Bit 15
+ k = Key length - 12 Bits -> Max 4095 (Bit 20 - 31)
+ o = Operation Type - 3 Bits -> Max 7 (Bit 5-7)
+ l = Execute - 1 Bit 10
+ p = Simple Indicator - 1 Bit 8
+ s = Start Indicator - 1 Bit 11
+ t = Distribution GroupType- 1 Bit 3
+ y = Commit Type - 2 Bit 12-13
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ dgbtcooop lsyyeiaaafkkkkkkkkkkkk
+*/
+
+#define COMMIT_SHIFT (4)
+#define START_SHIFT (11)
+#define SIMPLE_SHIFT (8)
+#define DIRTY_SHIFT (0)
+#define EXECUTE_SHIFT (10)
+#define INTERPRETED_SHIFT (15)
+#define DISTR_GROUP_SHIFT (1)
+#define DISTR_GROUP_TYPE_SHIFT (3)
+#define DISTR_KEY_SHIFT (2)
+#define SCAN_SHIFT (14)
+
+#define OPERATION_SHIFT (5)
+#define OPERATION_MASK (7)
+
+#define AINFO_SHIFT (16)
+#define AINFO_MASK (7)
+
+#define KEY_LEN_SHIFT (20)
+#define KEY_LEN_MASK (4095)
+
+#define COMMIT_TYPE_SHIFT (12)
+#define COMMIT_TYPE_MASK (3)
+
+#define EXECUTING_TRIGGER_SHIFT (19)
+
+/**
+ * Scan Info
+ *
+ t = Scan take over indicator - 1 Bit
+ n = Take over node - 16 Bits -> max 65535
+ p = Scan Info - 12 Bits -> max 4095
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ tpppppppppppp nnnnnnnnnnnnnnnn
+*/
+
+#define TAKE_OVER_SHIFT (0)
+
+#define TAKE_OVER_NODE_SHIFT (16)
+#define TAKE_OVER_NODE_MASK (65535)
+
+#define SCAN_INFO_SHIFT (1)
+#define SCAN_INFO_MASK (4095)
+
+/**
+ * Attr Len
+ *
+ n = Attrinfo length(words) - 16 Bits -> max 65535
+ a = API version no - 16 Bits -> max 65535
+
+ 1111111111222222222233
+ 01234567890123456789012345678901
+ aaaaaaaaaaaaaaaannnnnnnnnnnnnnnn
+*/
+
+#define API_VER_NO_SHIFT (16)
+#define API_VER_NO_MASK (65535)
+
+#define ATTRLEN_SHIFT (0)
+#define ATTRLEN_MASK (65535)
+
+inline
+Uint8
+TcKeyReq::getCommitFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> COMMIT_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getAbortOption(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> COMMIT_TYPE_SHIFT) & COMMIT_TYPE_MASK);
+}
+
+inline
+Uint8
+TcKeyReq::getStartFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> START_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getSimpleFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> SIMPLE_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getExecuteFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> EXECUTE_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getDirtyFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> DIRTY_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getInterpretedFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> INTERPRETED_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getDistributionGroupFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> DISTR_GROUP_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getDistributionGroupTypeFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> DISTR_GROUP_TYPE_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getDistributionKeyFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> DISTR_KEY_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getScanIndFlag(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> SCAN_SHIFT) & 1);
+}
+
+inline
+Uint8
+TcKeyReq::getOperationType(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> OPERATION_SHIFT) & OPERATION_MASK);
+}
+
+inline
+Uint16
+TcKeyReq::getKeyLength(const UintR & requestInfo){
+ return (Uint16)((requestInfo >> KEY_LEN_SHIFT) & KEY_LEN_MASK);
+}
+
+inline
+Uint8
+TcKeyReq::getAIInTcKeyReq(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> AINFO_SHIFT) & AINFO_MASK);
+}
+
+inline
+Uint8
+TcKeyReq::getExecutingTrigger(const UintR & requestInfo){
+ return (Uint8)((requestInfo >> EXECUTING_TRIGGER_SHIFT) & 1);
+}
+
+inline
+void
+TcKeyReq::clearRequestInfo(UintR & requestInfo){
+ requestInfo = 0;
+}
+
+inline
+void
+TcKeyReq::setAbortOption(UintR & requestInfo, Uint32 type){
+ ASSERT_MAX(type, COMMIT_TYPE_MASK, "TcKeyReq::setAbortOption");
+ requestInfo &= ~(COMMIT_TYPE_MASK << COMMIT_TYPE_SHIFT);
+ requestInfo |= (type << COMMIT_TYPE_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setCommitFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setCommitFlag");
+ requestInfo &= ~(1 << COMMIT_SHIFT);
+ requestInfo |= (flag << COMMIT_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setStartFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setStartFlag");
+ requestInfo &= ~(1 << START_SHIFT);
+ requestInfo |= (flag << START_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setSimpleFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setSimpleFlag");
+ requestInfo &= ~(1 << SIMPLE_SHIFT);
+ requestInfo |= (flag << SIMPLE_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setDirtyFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setDirstFlag");
+ requestInfo &= ~(1 << DIRTY_SHIFT);
+ requestInfo |= (flag << DIRTY_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setExecuteFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setExecuteFlag");
+ requestInfo &= ~(1 << EXECUTE_SHIFT);
+ requestInfo |= (flag << EXECUTE_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setInterpretedFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setInterpretedFlag");
+ requestInfo &= ~(1 << INTERPRETED_SHIFT);
+ requestInfo |= (flag << INTERPRETED_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setDistributionGroupTypeFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setDistributionGroupTypeFlag");
+ requestInfo &= ~(1 << DISTR_GROUP_TYPE_SHIFT);
+ requestInfo |= (flag << DISTR_GROUP_TYPE_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setDistributionGroupFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setDistributionGroupFlag");
+ requestInfo &= ~(1 << DISTR_GROUP_SHIFT);
+ requestInfo |= (flag << DISTR_GROUP_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setDistributionKeyFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setDistributionKeyFlag");
+ requestInfo &= ~(1 << DISTR_KEY_SHIFT);
+ requestInfo |= (flag << DISTR_KEY_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setScanIndFlag(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setScanIndFlag");
+ requestInfo &= ~(1 << SCAN_SHIFT);
+ requestInfo |= (flag << SCAN_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setOperationType(UintR & requestInfo, Uint32 type){
+ ASSERT_MAX(type, OPERATION_MASK, "TcKeyReq::setOperationType");
+ requestInfo &= ~(OPERATION_MASK << OPERATION_SHIFT);
+ requestInfo |= (type << OPERATION_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setKeyLength(UintR & requestInfo, Uint32 len){
+ ASSERT_MAX(len, KEY_LEN_MASK, "TcKeyReq::setKeyLength");
+ requestInfo &= ~(KEY_LEN_MASK << KEY_LEN_SHIFT);
+ requestInfo |= (len << KEY_LEN_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setAIInTcKeyReq(UintR & requestInfo, Uint32 len){
+ ASSERT_MAX(len, AINFO_MASK, "TcKeyReq::setAIInTcKeyReq");
+ requestInfo &= ~(AINFO_MASK << AINFO_SHIFT);
+ requestInfo |= (len << AINFO_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setExecutingTrigger(UintR & requestInfo, Uint32 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setExecutingTrigger");
+ requestInfo &= ~(1 << EXECUTING_TRIGGER_SHIFT);
+ requestInfo |= (flag << EXECUTING_TRIGGER_SHIFT);
+}
+
+inline
+Uint8
+TcKeyReq::getTakeOverScanFlag(const UintR & scanInfo){
+ return (Uint8)((scanInfo >> TAKE_OVER_SHIFT) & 1);
+}
+
+inline
+Uint16
+TcKeyReq::getTakeOverScanNode(const UintR & scanInfo){
+ return (Uint16)((scanInfo >> TAKE_OVER_NODE_SHIFT) & TAKE_OVER_NODE_MASK);
+}
+
+inline
+Uint16
+TcKeyReq::getTakeOverScanInfo(const UintR & scanInfo){
+ return (Uint16)((scanInfo >> SCAN_INFO_SHIFT) & SCAN_INFO_MASK);
+}
+
+
+inline
+void
+TcKeyReq::setTakeOverScanFlag(UintR & scanInfo, Uint8 flag){
+ ASSERT_BOOL(flag, "TcKeyReq::setTakeOverScanFlag");
+ scanInfo |= (flag << TAKE_OVER_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setTakeOverScanNode(UintR & scanInfo, Uint16 node){
+// ASSERT_MAX(node, TAKE_OVER_NODE_MASK, "TcKeyReq::setTakeOverScanNode");
+ scanInfo |= (node << TAKE_OVER_NODE_SHIFT);
+}
+
+inline
+void
+TcKeyReq::setTakeOverScanInfo(UintR & scanInfo, Uint16 aScanInfo){
+// ASSERT_MAX(aScanInfo, SCAN_INFO_MASK, "TcKeyReq::setTakeOverScanInfo");
+ scanInfo |= (aScanInfo << SCAN_INFO_SHIFT);
+}
+
+
+inline
+Uint16
+TcKeyReq::getAPIVersion(const UintR & anAttrLen){
+ return (Uint16)((anAttrLen >> API_VER_NO_SHIFT) & API_VER_NO_MASK);
+}
+
+inline
+void
+TcKeyReq::setAPIVersion(UintR & anAttrLen, Uint16 apiVersion){
+// ASSERT_MAX(apiVersion, API_VER_NO_MASK, "TcKeyReq::setAPIVersion");
+ anAttrLen |= (apiVersion << API_VER_NO_SHIFT);
+}
+
+inline
+Uint16
+TcKeyReq::getAttrinfoLen(const UintR & anAttrLen){
+ return (Uint16)((anAttrLen) & ATTRLEN_MASK);
+}
+
+inline
+void
+TcKeyReq::setAttrinfoLen(UintR & anAttrLen, Uint16 aiLen){
+// ASSERT_MAX(aiLen, ATTRLEN_MASK, "TcKeyReq::setAttrinfoLen");
+ anAttrLen |= aiLen;
+}
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TcRollbackRep.hpp b/ndb/include/kernel/signaldata/TcRollbackRep.hpp
new file mode 100644
index 00000000000..b00731a04a6
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcRollbackRep.hpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TCROLLBACKREP_HPP
+#define TCROLLBACKREP_HPP
+
+#include "SignalData.hpp"
+
+class TcRollbackRep {
+ /**
+ * Sender(s)
+ */
+ friend class NdbConnection;
+ friend class DbUtil;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtup;
+
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class Dbtc;
+
+ friend bool printTCROLBACKREP(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 connectPtr;
+ Uint32 transId[2];
+ Uint32 returnCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TcSizeAltReq.hpp b/ndb/include/kernel/signaldata/TcSizeAltReq.hpp
new file mode 100644
index 00000000000..34eacfe5a93
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TcSizeAltReq.hpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TC_SIZE_ALT_REQ_H
+#define TC_SIZE_ALT_REQ_H
+
+
+
+#include "SignalData.hpp"
+
+class TcSizeAltReq {
+ /**
+ * Sender(s)
+ */
+ friend class ClusterConfiguration;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+private:
+ /**
+ * Indexes in theData
+ */
+ STATIC_CONST( IND_BLOCK_REF = 0 );
+ STATIC_CONST( IND_API_CONNECT = 1 );
+ STATIC_CONST( IND_TC_CONNECT = 2 );
+ STATIC_CONST( IND_UNUSED = 3 );
+ STATIC_CONST( IND_TABLE = 4 );
+ STATIC_CONST( IND_TC_SCAN = 5 );
+ STATIC_CONST( IND_LOCAL_SCAN = 6 );
+
+ /**
+ * Use the index definitions to use the signal data
+ */
+ UintR theData[7];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TestOrd.hpp b/ndb/include/kernel/signaldata/TestOrd.hpp
new file mode 100644
index 00000000000..1600df08884
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TestOrd.hpp
@@ -0,0 +1,229 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TEST_ORD_H
+#define TEST_ORD_H
+
+#include "SignalData.hpp"
+
+/**
+ * Send by API to preform TEST ON / TEST OFF
+ *
+ * SENDER: API
+ * RECIVER: SimBlockCMCtrBlck
+ */
+class TestOrd {
+ friend class Ndb;
+ friend class Cmvmi;
+ friend class MgmtSrvr;
+public:
+
+ enum Command {
+ KeepUnchanged = 0,
+ On = 1,
+ Off = 2,
+ Toggle = 3,
+ COMMAND_MASK = 3
+ };
+
+ enum SignalLoggerSpecification {
+ InputSignals = 1,
+ OutputSignals = 2,
+ InputOutputSignals = 3,
+ LOG_MASK = 3
+ };
+
+ enum TraceSpecification {
+ TraceALL = 0,
+ TraceAPI = 1,
+ TraceGlobalCheckpoint = 2,
+ TraceLocalCheckpoint = 4,
+ TraceDisconnect = 8,
+ TRACE_MASK = 15
+ };
+
+private:
+ STATIC_CONST( SignalLength = 25 );
+
+ /**
+ * Clear Signal
+ */
+ void clear();
+
+ /**
+ * Set/Get test command
+ */
+ void setTestCommand(Command);
+ void getTestCommand(Command&) const;
+
+ /**
+ * Set trace command
+ */
+ void setTraceCommand(Command, TraceSpecification);
+
+ /**
+ * Get trace command
+ */
+ void getTraceCommand(Command&, TraceSpecification&) const;
+
+ /**
+ * Return no of signal logger commands
+ *
+ * -1 Means apply command(0) to all blocks
+ *
+ */
+ UintR getNoOfSignalLoggerCommands() const;
+
+ /**
+ * Add a signal logger command to a specific block
+ */
+ void addSignalLoggerCommand(BlockNumber, Command, SignalLoggerSpecification);
+
+ /**
+ * Add a signal logger command to all blocks
+ *
+ * Note removes all previously added commands
+ *
+ */
+ void addSignalLoggerCommand(Command, SignalLoggerSpecification);
+
+ /**
+ * Get Signal logger command
+ */
+ void getSignalLoggerCommand(int no, BlockNumber&, Command&, SignalLoggerSpecification&) const;
+
+ UintR testCommand; // DATA 0
+ UintR traceCommand; // DATA 1
+ UintR noOfSignalLoggerCommands; // DATA 2
+ UintR signalLoggerCommands[22]; // DATA 3 - 25
+};
+
+#define COMMAND_SHIFT (0)
+#define TRACE_SHIFT (2)
+#define LOG_SHIFT (2)
+
+#define BLOCK_NO_SHIFT (16)
+#define BLOCK_NO_MASK 65535
+
+/**
+ * Clear Signal
+ */
+inline
+void
+TestOrd::clear(){
+ setTestCommand(KeepUnchanged);
+ setTraceCommand(KeepUnchanged, TraceAPI); //
+ noOfSignalLoggerCommands = 0;
+}
+
+/**
+ * Set/Get test command
+ */
+inline
+void
+TestOrd::setTestCommand(Command cmd){
+ ASSERT_RANGE(cmd, 0, COMMAND_MASK, "TestOrd::setTestCommand");
+ testCommand = cmd;
+}
+
+inline
+void
+TestOrd::getTestCommand(Command & cmd) const{
+ cmd = (Command)(testCommand >> COMMAND_SHIFT);
+}
+
+/**
+ * Set trace command
+ */
+inline
+void
+TestOrd::setTraceCommand(Command cmd, TraceSpecification spec){
+ ASSERT_RANGE(cmd, 0, COMMAND_MASK, "TestOrd::setTraceCommand");
+ ASSERT_RANGE(spec, 0, TRACE_MASK, "TestOrd::setTraceCommand");
+ traceCommand = (cmd << COMMAND_SHIFT) | (spec << TRACE_SHIFT);
+}
+
+/**
+ * Get trace command
+ */
+inline
+void
+TestOrd::getTraceCommand(Command & cmd, TraceSpecification & spec) const{
+ cmd = (Command)((traceCommand >> COMMAND_SHIFT) & COMMAND_MASK);
+ spec = (TraceSpecification)((traceCommand >> TRACE_SHIFT) & TRACE_MASK);
+}
+
+/**
+ * Return no of signal logger commands
+ *
+ * -1 Means apply command(0) to all blocks
+ *
+ */
+inline
+UintR
+TestOrd::getNoOfSignalLoggerCommands() const{
+ return noOfSignalLoggerCommands;
+}
+
+/**
+ * Add a signal logger command to a specific block
+ */
+inline
+void
+TestOrd::addSignalLoggerCommand(BlockNumber bnr,
+ Command cmd, SignalLoggerSpecification spec){
+ ASSERT_RANGE(cmd, 0, COMMAND_MASK, "TestOrd::addSignalLoggerCommand");
+ ASSERT_RANGE(spec, 0, LOG_MASK, "TestOrd::addSignalLoggerCommand");
+ //ASSERT_MAX(bnr, BLOCK_NO_MASK, "TestOrd::addSignalLoggerCommand");
+
+ signalLoggerCommands[noOfSignalLoggerCommands] =
+ (bnr << BLOCK_NO_SHIFT) | (cmd << COMMAND_SHIFT) | (spec << LOG_SHIFT);
+ noOfSignalLoggerCommands ++;
+}
+
+/**
+ * Add a signal logger command to all blocks
+ *
+ * Note removes all previously added commands
+ *
+ */
+inline
+void
+TestOrd::addSignalLoggerCommand(Command cmd, SignalLoggerSpecification spec){
+ ASSERT_RANGE(cmd, 0, COMMAND_MASK, "TestOrd::addSignalLoggerCommand");
+ ASSERT_RANGE(spec, 0, LOG_MASK, "TestOrd::addSignalLoggerCommand");
+
+ noOfSignalLoggerCommands = ~0;
+ signalLoggerCommands[0] = (cmd << COMMAND_SHIFT) | (spec << LOG_SHIFT);
+}
+
+/**
+ * Get Signal logger command
+ */
+inline
+void
+TestOrd::getSignalLoggerCommand(int no, BlockNumber & bnr,
+ Command & cmd,
+ SignalLoggerSpecification & spec) const{
+ bnr = (BlockNumber)((signalLoggerCommands[no] >> BLOCK_NO_SHIFT)
+ & BLOCK_NO_MASK);
+ cmd = (Command)((signalLoggerCommands[no] >> COMMAND_SHIFT)
+ & COMMAND_MASK);
+ spec = (SignalLoggerSpecification)((signalLoggerCommands[no] >> LOG_SHIFT)
+ & LOG_MASK);
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TransIdAI.hpp b/ndb/include/kernel/signaldata/TransIdAI.hpp
new file mode 100755
index 00000000000..4df7bf2a126
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TransIdAI.hpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TRANSID_AI_HPP
+#define TRANSID_AI_HPP
+
+#include "SignalData.hpp"
+
+class TransIdAI {
+ /**
+ * Sender(s)
+ */
+ friend class Dbtup;
+
+ /**
+ * Receiver(s)
+ */
+ friend class NdbConnection;
+ friend class Dbtc;
+ friend class Dbutil;
+ friend class Dblqh;
+ friend class Suma;
+
+ friend bool printTRANSID_AI(FILE *, const Uint32 *, Uint32, Uint16);
+
+public:
+ STATIC_CONST( HeaderLength = 3 );
+ STATIC_CONST( DataLength = 22 );
+
+ // Public methods
+public:
+ Uint32* getData() const;
+
+public:
+ Uint32 connectPtr;
+ Uint32 transId[2];
+ Uint32 attrData[DataLength];
+};
+
+inline
+Uint32* TransIdAI::getData() const
+{
+ return (Uint32*)&attrData[0];
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TrigAttrInfo.hpp b/ndb/include/kernel/signaldata/TrigAttrInfo.hpp
new file mode 100644
index 00000000000..e2c029b9033
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TrigAttrInfo.hpp
@@ -0,0 +1,138 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TRIG_ATTRINFO_HPP
+#define TRIG_ATTRINFO_HPP
+
+#include "SignalData.hpp"
+#include <NodeBitmask.hpp>
+#include <trigger_definitions.h>
+#include <string.h>
+
+/**
+ * TrigAttrInfo
+ *
+ * This signal is sent by TUP to signal
+ * that a trigger has fired
+ */
+class TrigAttrInfo {
+ /**
+ * Sender(s)
+ */
+ // API
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbtup;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtc;
+ friend class Backup;
+ friend class SumaParticipant;
+
+ /**
+ * For printing
+ */
+ friend bool printTRIG_ATTRINFO(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+enum AttrInfoType {
+ PRIMARY_KEY = 0,
+ BEFORE_VALUES = 1,
+ AFTER_VALUES = 2
+};
+
+ STATIC_CONST( DataLength = 22 );
+ STATIC_CONST( StaticLength = 3 );
+
+private:
+ Uint32 m_connectionPtr;
+ Uint32 m_trigId;
+ Uint32 m_type;
+ Uint32 m_data[DataLength];
+
+ // Public methods
+public:
+ Uint32 getConnectionPtr() const;
+ void setConnectionPtr(Uint32);
+ AttrInfoType getAttrInfoType() const;
+ void setAttrInfoType(AttrInfoType anAttrType);
+ Uint32 getTriggerId() const;
+ void setTriggerId(Uint32 aTriggerId);
+ Uint32 getTransactionId1() const;
+ void setTransactionId1(Uint32 aTransId);
+ Uint32 getTransactionId2() const;
+ void setTransactionId2(Uint32 aTransId);
+ Uint32* getData() const;
+ int setData(Uint32* aDataBuf, Uint32 aDataLen);
+};
+
+inline
+Uint32 TrigAttrInfo::getConnectionPtr() const
+{
+ return m_connectionPtr;
+}
+
+inline
+void TrigAttrInfo::setConnectionPtr(Uint32 aConnectionPtr)
+{
+ m_connectionPtr = aConnectionPtr;
+}
+
+inline
+TrigAttrInfo::AttrInfoType TrigAttrInfo::getAttrInfoType() const
+{
+ return (TrigAttrInfo::AttrInfoType) m_type;
+}
+
+inline
+void TrigAttrInfo::setAttrInfoType(TrigAttrInfo::AttrInfoType anAttrType)
+{
+ m_type = (Uint32) anAttrType;
+}
+
+inline
+Uint32 TrigAttrInfo::getTriggerId() const
+{
+ return m_trigId;
+}
+
+inline
+void TrigAttrInfo::setTriggerId(Uint32 aTriggerId)
+{
+ m_trigId = aTriggerId;
+}
+
+inline
+Uint32* TrigAttrInfo::getData() const
+{
+ return (Uint32*)&m_data[0];
+}
+
+inline
+int TrigAttrInfo::setData(Uint32* aDataBuf, Uint32 aDataLen)
+{
+ if (aDataLen > DataLength)
+ return -1;
+ memcpy(m_data, aDataBuf, aDataLen*sizeof(Uint32));
+
+ return 0;
+}
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TupAccess.hpp b/ndb/include/kernel/signaldata/TupAccess.hpp
new file mode 100644
index 00000000000..5cfb8c0d153
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TupAccess.hpp
@@ -0,0 +1,172 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TUP_ACCESS_HPP
+#define TUP_ACCESS_HPP
+
+#include "SignalData.hpp"
+
+/*
+ * Direct signals used by ACC and TUX to access the TUP block in the
+ * same thread.
+ *
+ * NOTE: Caller must set errorCode to RNIL. Signal printer uses this to
+ * distinguish between input and output (no better way exists).
+ */
+
+/*
+ * Read attributes from any table.
+ */
+class TupReadAttrs {
+ friend class Dbtup;
+ friend class Dbacc;
+ friend class Dbtux;
+ friend bool printTUP_READ_ATTRS(FILE*, const Uint32*, Uint32, Uint16);
+public:
+ enum Flag {
+ /*
+ * Read primary key attributes. No input attribute ids are
+ * specified. Instead TUP fills in both input and output sections.
+ * Tuple version is not used.
+ */
+ ReadKeys = (1 << 0)
+ };
+ STATIC_CONST( SignalLength = 10 );
+private:
+ /*
+ * Error code set by TUP. Zero means no error.
+ */
+ Uint32 errorCode;
+ /*
+ * Request info contains flags (see Flags above).
+ */
+ Uint32 requestInfo;
+ /*
+ * Table i-value.
+ */
+ Uint32 tableId;
+ /*
+ * Fragment is given by logical id within the table or by direct
+ * i-value (faster). Unknown values are given as RNIL. On return TUP
+ * fills in both values.
+ */
+ Uint32 fragId;
+ Uint32 fragPtrI;
+ /*
+ * Logical address ("local key") of "original" tuple (the latest
+ * version) consisting of logical fragment page id and tuple index
+ * within the page (shifted left by 1).
+ */
+ Uint32 tupAddr;
+ /*
+ * Version of the tuple to read. Not used if ReadKeys.
+ */
+ Uint32 tupVersion;
+ /*
+ * Real page id and offset of the "original" tuple. Unknown page is
+ * given as RNIL. On return TUP fills in these.
+ */
+ Uint32 pageId;
+ Uint32 pageOffset;
+ /*
+ * Shared buffer id. Currently must be 0 which means to use rest of
+ * signal data.
+ */
+ Uint32 bufferId;
+ /*
+ * Shared buffer 0 starts after signal class. Input is number of
+ * attributes and list of attribute ids in AttributeHeader format.
+ * Output is placed after the input and consists of a list of entries
+ * where each entry has an AttributeHeader followed by words of data.
+ */
+};
+
+/*
+ * Query status of tuple version. Used by TUX to decide if a tuple
+ * version found in index tree is visible to the transaction.
+ */
+class TupQueryTh {
+ friend class Dbtup;
+ friend class Dbtux;
+ friend bool printTUP_QUERY_TH(FILE*, const Uint32*, Uint32, Uint16);
+public:
+ enum Flag {
+ };
+ STATIC_CONST( SignalLength = 7 );
+private:
+ /*
+ TUX wants to check if tuple is visible to the scan query.
+ Input data is tuple address (tableId, fragId, tupAddr, tupVersion),
+ and transaction data so that TUP knows how to deduct if tuple is
+ visible (transId1, transId2, savePointId).
+ returnCode is set in return signal to indicate whether tuple is visible.
+ */
+ union {
+ Uint32 returnCode; // 1 if tuple visible
+ Uint32 tableId;
+ };
+ Uint32 fragId;
+ Uint32 tupAddr;
+ Uint32 tupVersion;
+ Uint32 transId1;
+ Uint32 transId2;
+ Uint32 savePointId;
+};
+
+/*
+ * Operate on entire tuple. Used by TUX where the table has a single
+ * Uint32 array attribute representing an index tree node.
+ */
+class TupStoreTh {
+ friend class Dbtup;
+ friend class Dbtux;
+ friend bool printTUP_STORE_TH(FILE*, const Uint32*, Uint32, Uint16);
+public:
+ enum OpCode {
+ OpUndefined = 0,
+ OpRead = 1,
+ OpInsert = 2,
+ OpUpdate = 3,
+ OpDelete = 4
+ };
+ STATIC_CONST( SignalLength = 12 );
+private:
+ /*
+ * These are as in TupReadAttrs (except opCode). Version must be
+ * zero. Ordered index tuple (tree node) has only current version.
+ */
+ Uint32 errorCode;
+ Uint32 opCode;
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 fragPtrI;
+ Uint32 tupAddr;
+ Uint32 tupVersion;
+ Uint32 pageId;
+ Uint32 pageOffset;
+ Uint32 bufferId;
+ /*
+ * Data offset and size in words. Applies to both the buffer and the
+ * tuple. Used e.g. to read only node header.
+ */
+ Uint32 dataOffset;
+ Uint32 dataSize;
+ /*
+ * Shared buffer 0 starts after signal class.
+ */
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TupCommit.hpp b/ndb/include/kernel/signaldata/TupCommit.hpp
new file mode 100644
index 00000000000..7c5a7931e6c
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TupCommit.hpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TUP_COMMIT_H
+#define TUP_COMMIT_H
+
+#include "SignalData.hpp"
+
+class TupCommitReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtup;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * For printing
+ */
+ friend bool printTUPCOMMITREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ Uint32 opPtr;
+ Uint32 gci;
+ Uint32 hashValue;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TupFrag.hpp b/ndb/include/kernel/signaldata/TupFrag.hpp
new file mode 100644
index 00000000000..ffde2217893
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TupFrag.hpp
@@ -0,0 +1,188 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TUP_FRAG_HPP
+#define TUP_FRAG_HPP
+
+#include "SignalData.hpp"
+
+/*
+ * Add fragment and add attribute signals between LQH and TUP,TUX.
+ * NOTE: return signals from TUP,TUX to LQH must have same format.
+ */
+
+// TUP: add fragment
+
+class TupFragReq {
+ friend class Dblqh;
+ friend class Dbtup;
+public:
+ STATIC_CONST( SignalLength = 14 );
+private:
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 reqInfo;
+ Uint32 tableId;
+ Uint32 noOfAttr;
+ Uint32 fragId;
+ Uint32 todo[8];
+};
+
+class TupFragConf {
+ friend class Dblqh;
+ friend class Dbtup;
+public:
+ STATIC_CONST( SignalLength = 4 );
+private:
+ Uint32 userPtr;
+ Uint32 tupConnectPtr;
+ Uint32 fragPtr;
+ Uint32 fragId;
+};
+
+class TupFragRef {
+ friend class Dblqh;
+ friend class Dbtup;
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+ Uint32 userPtr;
+ Uint32 errorCode;
+};
+
+// TUX: add fragment
+
+class TuxFragReq {
+ friend class Dblqh;
+ friend class Dbtux;
+public:
+ STATIC_CONST( SignalLength = 9 );
+private:
+ Uint32 userPtr;
+ Uint32 userRef;
+ Uint32 reqInfo;
+ Uint32 tableId;
+ Uint32 noOfAttr;
+ Uint32 fragId;
+ Uint32 fragOff;
+ Uint32 tableType;
+ Uint32 primaryTableId;
+};
+
+class TuxFragConf {
+ friend class Dblqh;
+ friend class Dbtux;
+public:
+ STATIC_CONST( SignalLength = 4 );
+private:
+ Uint32 userPtr;
+ Uint32 tuxConnectPtr;
+ Uint32 fragPtr;
+ Uint32 fragId;
+};
+
+class TuxFragRef {
+ friend class Dblqh;
+ friend class Dbtux;
+public:
+ STATIC_CONST( SignalLength = 2 );
+ enum ErrorCode {
+ NoError = 0,
+ InvalidRequest = 800,
+ NoFreeFragmentOper = 830,
+ NoFreeIndexFragment = 852,
+ NoFreeFragment = 604,
+ NoFreeAttributes = 827
+ };
+private:
+ Uint32 userPtr;
+ Uint32 errorCode;
+};
+
+// TUP: add attribute
+
+class TupAddAttrReq {
+ friend class Dblqh;
+ friend class Dbtux;
+public:
+ STATIC_CONST( SignalLength = 4 );
+private:
+ Uint32 tupConnectPtr;
+ Uint32 notused1;
+ Uint32 attrId;
+ Uint32 attrDescriptor;
+};
+
+class TupAddAttrConf {
+ friend class Dblqh;
+ friend class Dbtup;
+public:
+ STATIC_CONST( SignalLength = 1 );
+private:
+ Uint32 userPtr;
+};
+
+class TupAddAttrRef {
+ friend class Dblqh;
+ friend class Dbtup;
+public:
+ STATIC_CONST( SignalLength = 2 );
+private:
+ Uint32 userPtr;
+ Uint32 errorCode;
+};
+
+// TUX: add attribute
+
+class TuxAddAttrReq {
+ friend class Dblqh;
+ friend class Dbtux;
+public:
+ STATIC_CONST( SignalLength = 6 );
+private:
+ Uint32 tuxConnectPtr;
+ Uint32 notused1;
+ Uint32 attrId;
+ Uint32 attrDescriptor;
+ Uint32 extTypeInfo;
+ Uint32 primaryAttrId;
+};
+
+class TuxAddAttrConf {
+ friend class Dblqh;
+ friend class Dbtux;
+public:
+ STATIC_CONST( SignalLength = 1 );
+private:
+ Uint32 userPtr;
+};
+
+class TuxAddAttrRef {
+ friend class Dblqh;
+ friend class Dbtux;
+public:
+ STATIC_CONST( SignalLength = 2 );
+ enum ErrorCode {
+ NoError = 0,
+ InvalidAttributeType = 831,
+ InvalidNodeSize = 832
+ };
+private:
+ Uint32 userPtr;
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TupKey.hpp b/ndb/include/kernel/signaldata/TupKey.hpp
new file mode 100644
index 00000000000..304bebbec88
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TupKey.hpp
@@ -0,0 +1,126 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TUP_KEY_H
+#define TUP_KEY_H
+
+#include "SignalData.hpp"
+
+class TupKeyReq {
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtup;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * For printing
+ */
+ friend bool printTUPKEYREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 18 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ Uint32 connectPtr;
+ Uint32 request;
+ Uint32 tableRef;
+ Uint32 fragId;
+ Uint32 keyRef1;
+ Uint32 keyRef2;
+ Uint32 attrBufLen;
+ Uint32 opRef;
+ Uint32 applRef;
+ Uint32 schemaVersion;
+ Uint32 storedProcedure;
+ Uint32 transId1;
+ Uint32 transId2;
+ Uint32 fragPtr;
+ Uint32 primaryReplica;
+ Uint32 coordinatorTC;
+ Uint32 tcOpIndex;
+ Uint32 savePointId;
+};
+
+class TupKeyConf {
+ /**
+ * Reciver(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbtup;
+
+ /**
+ * For printing
+ */
+ friend bool printTUPKEYCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 6 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ Uint32 userPtr;
+ Uint32 pageId;
+ Uint32 pageIndex;
+ Uint32 readLength;
+ Uint32 writeLength;
+ Uint32 noFiredTriggers;
+};
+
+class TupKeyRef {
+ /**
+ * Reciver(s)
+ */
+ friend class Dblqh;
+
+ /**
+ * Sender(s)
+ */
+ friend class Dbtup;
+
+ /**
+ * For printing
+ */
+ friend bool printTUPKEYREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+
+ /**
+ * DATA VARIABLES
+ */
+ Uint32 userRef;
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TupSizeAltReq.hpp b/ndb/include/kernel/signaldata/TupSizeAltReq.hpp
new file mode 100644
index 00000000000..215493bc188
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TupSizeAltReq.hpp
@@ -0,0 +1,58 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TUP_SIZE_ALT_REQ_H
+#define TUP_SIZE_ALT_REQ_H
+
+
+
+#include "SignalData.hpp"
+
+class TupSizeAltReq {
+ /**
+ * Sender(s)
+ */
+ friend class ClusterConfiguration;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Dbtup;
+private:
+ /**
+ * Indexes in theData
+ */
+ STATIC_CONST( IND_BLOCK_REF = 0 );
+ STATIC_CONST( IND_DISK_PAGE_ARRAY = 1 );
+ STATIC_CONST( IND_DISK_PAGE_REPRESENT = 2 );
+ STATIC_CONST( IND_FRAG = 3 );
+ STATIC_CONST( IND_PAGE_CLUSTER = 4 );
+ STATIC_CONST( IND_LOGIC_PAGE = 5 );
+ STATIC_CONST( IND_OP_RECS = 6 );
+ STATIC_CONST( IND_PAGE = 7 );
+ STATIC_CONST( IND_PAGE_RANGE = 8 );
+ STATIC_CONST( IND_TABLE = 9 );
+ STATIC_CONST( IND_TABLE_DESC = 10 );
+ STATIC_CONST( IND_DELETED_BLOCKS = 11 );
+ STATIC_CONST( IND_STORED_PROC = 12 );
+
+ /**
+ * Use the index definitions to use the signal data
+ */
+ UintR theData[13];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TuxBound.hpp b/ndb/include/kernel/signaldata/TuxBound.hpp
new file mode 100644
index 00000000000..1f256150573
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TuxBound.hpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TUX_BOUND_HPP
+#define TUX_BOUND_HPP
+
+#include "SignalData.hpp"
+
+class TuxBoundInfo {
+ friend class Dblqh;
+ friend class Dbtux;
+public:
+ // must match API (0-4 and no changes expected)
+ enum BoundType {
+ BoundLE = 0, // bit 1 for less/greater
+ BoundLT = 1, // bit 0 for strict
+ BoundGE = 2,
+ BoundGT = 3,
+ BoundEQ = 4
+ };
+ enum ErrorCode {
+ InvalidAttrInfo = 4110,
+ InvalidBounds = 4259,
+ OutOfBuffers = 873
+ };
+ STATIC_CONST( SignalLength = 3 );
+private:
+ /*
+ * Error code set by TUX. Zero means no error.
+ */
+ Uint32 errorCode;
+ /*
+ * Pointer (i-value) to scan operation in TUX.
+ */
+ Uint32 tuxScanPtrI;
+ /*
+ * Number of words of bound info included after fixed signal data.
+ * Starts with 5 unused words (word 0 is length used by LQH).
+ */
+ Uint32 boundAiLength;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TuxContinueB.hpp b/ndb/include/kernel/signaldata/TuxContinueB.hpp
new file mode 100644
index 00000000000..385d85715e2
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TuxContinueB.hpp
@@ -0,0 +1,30 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TUX_CONTINUEB_H
+#define TUX_CONTINUEB_H
+
+#include "SignalData.hpp"
+
+class TuxContinueB {
+ friend class Dbtux;
+private:
+ enum {
+ DropIndex = 1
+ };
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TuxMaint.hpp b/ndb/include/kernel/signaldata/TuxMaint.hpp
new file mode 100644
index 00000000000..44deb33be80
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TuxMaint.hpp
@@ -0,0 +1,66 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TUX_MAINT_HPP
+#define TUX_MAINT_HPP
+
+#include "SignalData.hpp"
+
+/*
+ * Ordered index maintenance operation.
+ */
+
+class TuxMaintReq {
+ friend class Dbtup;
+ friend class Dbtux;
+ friend bool printTUX_MAINT_REQ(FILE*, const Uint32*, Uint32, Uint16);
+public:
+ enum OpCode { // first byte of opInfo
+ OpAdd = 1,
+ OpRemove = 2
+ };
+ enum OpFlag { // second byte of opInfo
+ };
+ enum ErrorCode {
+ NoError = 0, // must be zero
+ SearchError = 895, // add + found or remove + not found
+ NoMemError = 827
+ };
+ STATIC_CONST( SignalLength = 7 );
+private:
+ /*
+ * Error code set by TUX. Zero means no error.
+ */
+ Uint32 errorCode;
+ /*
+ * Table, index, fragment.
+ */
+ Uint32 tableId;
+ Uint32 indexId;
+ Uint32 fragId;
+ /*
+ * Tuple version identified by logical address of "original" tuple and
+ * version number.
+ */
+ Uint32 tupAddr;
+ Uint32 tupVersion;
+ /*
+ * Operation code and flags.
+ */
+ Uint32 opInfo;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/TuxSizeAltReq.hpp b/ndb/include/kernel/signaldata/TuxSizeAltReq.hpp
new file mode 100644
index 00000000000..5d5a0e102ba
--- /dev/null
+++ b/ndb/include/kernel/signaldata/TuxSizeAltReq.hpp
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TUX_SIZE_ALT_REQ_H
+#define TUX_SIZE_ALT_REQ_H
+
+#include "SignalData.hpp"
+
+class TuxSizeAltReq {
+ /**
+ * Sender(s)
+ */
+ friend class ClusterConfiguration;
+
+ /**
+ * Receiver(s)
+ */
+ friend class Dbtux;
+private:
+ /**
+ * Indexes in theData
+ */
+ STATIC_CONST( IND_BLOCK_REF = 0 );
+ STATIC_CONST( IND_INDEX = 1 );
+ STATIC_CONST( IND_FRAGMENT = 2 );
+ STATIC_CONST( IND_ATTRIBUTE = 3 );
+ STATIC_CONST( IND_SCAN = 4 );
+
+ /**
+ * Use the index definitions to use the signal data
+ */
+ UintR theData[4];
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/UpdateTo.hpp b/ndb/include/kernel/signaldata/UpdateTo.hpp
new file mode 100644
index 00000000000..0fa5f31b6b4
--- /dev/null
+++ b/ndb/include/kernel/signaldata/UpdateTo.hpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef UPDATE_TO_HPP
+#define UPDATE_TO_HPP
+
+class UpdateToReq {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 6 );
+private:
+ enum UpdateState {
+ TO_COPY_FRAG_COMPLETED = 0,
+ TO_COPY_COMPLETED = 1
+ };
+ Uint32 userPtr;
+ BlockReference userRef;
+ UpdateState updateState;
+ Uint32 startingNodeId;
+
+ /**
+ * Only when TO_COPY_FRAG_COMPLETED
+ */
+ Uint32 tableId;
+ Uint32 fragmentNo;
+};
+
+class UpdateToConf {
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+private:
+
+ Uint32 userPtr;
+ Uint32 sendingNodeId;
+ Uint32 startingNodeId;
+};
+#endif
diff --git a/ndb/include/kernel/signaldata/UtilDelete.hpp b/ndb/include/kernel/signaldata/UtilDelete.hpp
new file mode 100644
index 00000000000..67c13b8c2d5
--- /dev/null
+++ b/ndb/include/kernel/signaldata/UtilDelete.hpp
@@ -0,0 +1,121 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef UTIL_DELETE_HPP
+#define UTIL_DELETE_HPP
+
+#include "SignalData.hpp"
+#include <SimpleProperties.hpp>
+
+/**
+ * UTIL_DELETE_REQ, UTIL_DELETE_CONF, UTIL_DELETE_REF
+ */
+
+/**
+ * @class UtilDeleteReq
+ * @brief Delete transaction in Util block
+ *
+ * Data format:
+ * - UTIL_DELETE_REQ <prepareId> <ListOfAttributeHeaderValuePairs>
+ */
+
+class UtilDeleteReq {
+ /** Sender(s) / Receiver(s) */
+ friend class DbUtil;
+
+ /** For printing */
+ friend bool printUTIL_DELETE_REQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo);
+public:
+ STATIC_CONST( DataLength = 22 );
+ STATIC_CONST( HeaderLength = 3 );
+
+private:
+ Uint32 senderData;
+ Uint32 prepareId; // Which prepared transaction to execute
+ Uint32 totalDataLen; // Total length of attrData (including AttributeHeaders
+ // and possibly spanning over multiple signals)
+
+ /**
+ * Length in this = signal->length() - 3
+ * Sender block ref = signal->senderBlockRef()
+ */
+
+ Uint32 attrData[DataLength];
+};
+
+
+
+/**
+ * @class UtilDeleteConf
+ *
+ * Data format:
+ * - UTIL_PREPARE_CONF <UtilPrepareId>
+ */
+
+class UtilDeleteConf {
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class DbUtil;
+
+ /**
+ * For printing
+ */
+ friend bool printUTIL_DELETE_CONF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo);
+
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ Uint32 senderData; ///< The client data provided by the client sending
+ ///< UTIL_DELETE_REQ
+};
+
+
+/**
+ * @class UtilDeleteRef
+ *
+ * Data format:
+ * - UTIL_PREPARE_REF
+ */
+
+class UtilDeleteRef {
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class DbUtil;
+
+ /**
+ * For printing
+ */
+ friend bool printUTIL_DELETE_REF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo);
+
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderData;
+ Uint32 errorCode; ///< See UtilExecuteRef::errorCode
+ Uint32 TCErrorCode;
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/UtilExecute.hpp b/ndb/include/kernel/signaldata/UtilExecute.hpp
new file mode 100644
index 00000000000..551fb172cac
--- /dev/null
+++ b/ndb/include/kernel/signaldata/UtilExecute.hpp
@@ -0,0 +1,136 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef UTIL_EXECUTE_HPP
+#define UTIL_EXECUTE_HPP
+
+#include "SignalData.hpp"
+#include <SimpleProperties.hpp>
+
+/**
+ * UTIL_EXECUTE_REQ, UTIL_EXECUTE_CONF, UTIL_EXECUTE_REF
+ */
+
+/**
+ * @class UtilExecuteReq
+ * @brief Execute transaction in Util block
+ *
+ * Data format:
+ * - UTIL_EXECUTE_REQ <prepareId> <ListOfAttributeHeaderValuePairs>
+ */
+
+class UtilExecuteReq {
+ /** Sender(s) / Receiver(s) */
+ friend class DbUtil;
+ friend class Trix;
+
+ /** For printing */
+ friend bool printUTIL_EXECUTE_REQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo);
+public:
+ STATIC_CONST( SignalLength = 3 );
+ STATIC_CONST( HEADER_SECTION = 0 );
+ STATIC_CONST( DATA_SECTION = 1 );
+ STATIC_CONST( NoOfSections = 2 );
+
+ GET_SET_SENDERREF
+ GET_SET_SENDERDATA
+ void setPrepareId(Uint32 pId) { prepareId = pId; }; // !! unsets release flag
+ Uint32 getPrepareId() { return prepareId & 0xFF; };
+ void setReleaseFlag() { prepareId |= 0x100; };
+ bool getReleaseFlag() { return (prepareId & 0x100) != 0; };
+private:
+ Uint32 senderData; // MUST be no 1!
+ Uint32 senderRef;
+ Uint32 prepareId; // Which prepared transaction to execute
+};
+
+/**
+ * @class UtilExecuteConf
+ *
+ * Data format:
+ * - UTIL_PREPARE_CONF <UtilPrepareId>
+ */
+
+class UtilExecuteConf {
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class DbUtil;
+ friend class Trix;
+
+ /**
+ * For printing
+ */
+ friend bool printUTIL_EXECUTE_CONF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo);
+public:
+ STATIC_CONST( SignalLength = 1 );
+
+ GET_SET_SENDERDATA
+private:
+ Uint32 senderData; // MUST be no 1!
+};
+
+
+/**
+ * @class UtilExecuteRef
+ *
+ * Data format:
+ * - UTIL_PREPARE_REF
+ */
+
+class UtilExecuteRef {
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class DbUtil;
+ friend class Trix;
+
+ /**
+ * For printing
+ */
+ friend bool printUTIL_EXECUTE_REF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ enum ErrorCode {
+ IllegalKeyNumber = 1,
+ IllegalAttrNumber = 2,
+ TCError = 3,
+ IllegalPrepareId = 4,
+ AllocationError = 5,
+ MissingDataSection = 6,
+ MissingData = 7
+ };
+
+ GET_SET_SENDERDATA
+ GET_SET_ERRORCODE
+ GET_SET_TCERRORCODE
+private:
+ Uint32 senderData; // MUST be no 1!
+ Uint32 errorCode;
+ Uint32 TCErrorCode;
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/UtilLock.hpp b/ndb/include/kernel/signaldata/UtilLock.hpp
new file mode 100644
index 00000000000..1cac467daa0
--- /dev/null
+++ b/ndb/include/kernel/signaldata/UtilLock.hpp
@@ -0,0 +1,334 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef UTIL_LOCK_HPP
+#define UTIL_LOCK_HPP
+
+#include "SignalData.hpp"
+
+class UtilLockReq {
+
+ /**
+ * Receiver
+ */
+ friend class DbUtil;
+
+ /**
+ * Sender
+ */
+ friend class Dbdih;
+ friend class MutexManager;
+
+ friend bool printUTIL_LOCK_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum RequestInfo {
+ TryLock = 1
+ };
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+ Uint32 requestInfo;
+};
+
+class UtilLockConf {
+
+ /**
+ * Receiver
+ */
+ friend class Dbdih;
+ friend class MutexManager;
+
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ friend bool printUTIL_LOCK_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+ Uint32 lockKey;
+};
+
+class UtilLockRef {
+
+ /**
+ * Reciver
+ */
+ friend class Dbdih;
+ friend class MutexManager;
+
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ friend bool printUTIL_LOCK_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum ErrorCode {
+ OK = 0,
+ NoSuchLock = 1,
+ OutOfLockRecords = 2,
+ DistributedLockNotSupported = 3,
+ LockAlreadyHeld = 4
+
+ };
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+ Uint32 errorCode;
+};
+
+class UtilUnlockReq {
+
+ /**
+ * Receiver
+ */
+ friend class DbUtil;
+
+ /**
+ * Sender
+ */
+ friend class Dbdih;
+ friend class MutexManager;
+
+ friend bool printUTIL_UNLOCK_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+ Uint32 lockKey;
+};
+
+class UtilUnlockConf {
+
+ /**
+ * Receiver
+ */
+ friend class Dbdih;
+ friend class MutexManager;
+
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ friend bool printUTIL_UNLOCK_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+};
+
+class UtilUnlockRef {
+
+ /**
+ * Reciver
+ */
+ friend class Dbdih;
+ friend class MutexManager;
+
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ friend bool printUTIL_UNLOCK_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+ enum ErrorCode {
+ OK = 0,
+ NoSuchLock = 1,
+ NotLockOwner = 2
+ };
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+ Uint32 errorCode;
+};
+
+/**
+ * Creating a lock
+ */
+class UtilCreateLockReq {
+ /**
+ * Receiver
+ */
+ friend class DbUtil;
+
+ /**
+ * Sender
+ */
+ friend class MutexManager;
+
+ friend bool printUTIL_CREATE_LOCK_REQ(FILE *, const Uint32*, Uint32, Uint16);
+public:
+ enum LockType {
+ Mutex = 0 // Lock with only exclusive locks
+ };
+
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+ Uint32 lockType;
+};
+
+class UtilCreateLockRef {
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ /**
+ * Receiver
+ */
+ friend class MutexManager;
+
+ friend bool printUTIL_CREATE_LOCK_REF(FILE *, const Uint32*, Uint32, Uint16);
+public:
+ enum ErrorCode {
+ OK = 0,
+ OutOfLockQueueRecords = 1,
+ LockIdAlreadyUsed = 2,
+ UnsupportedLockType = 3
+ };
+
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+ Uint32 errorCode;
+};
+
+class UtilCreateLockConf {
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ /**
+ * Receiver
+ */
+ friend class MutexManager;
+
+ friend bool printUTIL_CREATE_LOCK_CONF(FILE*, const Uint32*, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+};
+
+/**
+ * Creating a lock
+ */
+class UtilDestroyLockReq {
+ /**
+ * Receiver
+ */
+ friend class DbUtil;
+
+ /**
+ * Sender
+ */
+ friend class MutexManager;
+
+ friend bool printUTIL_DESTROY_LOCK_REQ(FILE *, const Uint32*, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+ Uint32 lockKey;
+};
+
+class UtilDestroyLockRef {
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ /**
+ * Receiver
+ */
+ friend class MutexManager;
+
+ friend bool printUTIL_DESTROY_LOCK_REF(FILE *, const Uint32*, Uint32, Uint16);
+public:
+ enum ErrorCode {
+ OK = 0,
+ NoSuchLock = 1,
+ NotLockOwner = 2
+ };
+
+ STATIC_CONST( SignalLength = 4 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+ Uint32 errorCode;
+};
+
+class UtilDestroyLockConf {
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ /**
+ * Receiver
+ */
+ friend class MutexManager;
+
+ friend bool printUTIL_DESTROY_LOCK_CONF(FILE*, const Uint32*, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 senderData;
+ Uint32 senderRef;
+ Uint32 lockId;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/UtilPrepare.hpp b/ndb/include/kernel/signaldata/UtilPrepare.hpp
new file mode 100644
index 00000000000..8508487ce15
--- /dev/null
+++ b/ndb/include/kernel/signaldata/UtilPrepare.hpp
@@ -0,0 +1,161 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef UTIL_PREPARE_REQ_HPP
+#define UTIL_PREPARE_REQ_HPP
+
+#include "SignalData.hpp"
+#include <SimpleProperties.hpp>
+
+#ifdef NDB_WIN32
+#ifdef NO_ERROR
+#undef NO_ERROR
+#endif
+#endif
+
+/**
+ * UTIL_PREPARE_REQ, UTIL_PREPARE_CONF, UTIL_PREPARE_REF
+ */
+
+/**
+ * @class UtilPrepareReq
+ * @brief Prepare transaction in Util block
+ *
+ * Data format:
+ * - UTIL_PREPARE_REQ <NoOfOps> (<OperationType> <TableName> <AttrName>+)+
+ */
+class UtilPrepareReq {
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class DbUtil;
+ friend class Trix;
+
+ /**
+ * For printing
+ */
+ friend bool printUTIL_PREPARE_REQ(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo);
+
+public:
+ enum OperationTypeValue {
+ Read = 0,
+ Update = 1,
+ Insert = 2,
+ Delete = 3,
+ Write = 4
+
+ };
+
+ enum KeyValue {
+ NoOfOperations = 1, ///< No of operations in transaction
+ OperationType = 2, ///
+ TableName = 3, ///< String
+ AttributeName = 4, ///< String
+ TableId = 5,
+ AttributeId = 6
+ };
+
+ // Signal constants
+ STATIC_CONST( SignalLength = 2 );
+ STATIC_CONST( PROPERTIES_SECTION = 0 );
+ STATIC_CONST( NoOfSections = 1 );
+
+ GET_SET_SENDERREF
+ GET_SET_SENDERDATA
+private:
+ Uint32 senderData; // MUST be no 1!
+ Uint32 senderRef;
+};
+
+/**
+ * @class UtilPrepareConf
+ *
+ * Data format:
+ * - UTIL_PREPARE_CONF <UtilPrepareId>
+ */
+
+class UtilPrepareConf {
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class DbUtil;
+ friend class Trix;
+
+ /**
+ * For printing
+ */
+ friend bool printUTIL_PREPARE_CONF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo);
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ GET_SET_SENDERDATA
+ GET_SET_PREPAREID
+private:
+ Uint32 senderData; // MUST be no 1!
+ Uint32 prepareId;
+};
+
+
+/**
+ * @class UtilPrepareRef
+ *
+ * Data format:
+ * - UTIL_PREPARE_REF
+ */
+
+class UtilPrepareRef {
+ /**
+ * Sender(s) / Receiver(s)
+ */
+ friend class DbUtil;
+ friend class Trix;
+
+ /**
+ * For printing
+ */
+ friend bool printUTIL_PREPARE_REF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo);
+
+public:
+ enum ErrorCode {
+ NO_ERROR = 0,
+ PREPARE_SEIZE_ERROR = 1,
+ PREPARE_PAGES_SEIZE_ERROR = 2,
+ PREPARED_OPERATION_SEIZE_ERROR = 3,
+ DICT_TAB_INFO_ERROR = 4,
+ MISSING_PROPERTIES_SECTION = 5
+ };
+
+ STATIC_CONST( SignalLength = 2 );
+
+ GET_SET_SENDERDATA
+ GET_SET_ERRORCODE
+private:
+ Uint32 senderData; // MUST be no 1!
+ Uint32 errorCode;
+};
+
+
+#endif
diff --git a/ndb/include/kernel/signaldata/UtilRelease.hpp b/ndb/include/kernel/signaldata/UtilRelease.hpp
new file mode 100644
index 00000000000..d2864f02f47
--- /dev/null
+++ b/ndb/include/kernel/signaldata/UtilRelease.hpp
@@ -0,0 +1,83 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef UTIL_RELEASE_HPP
+#define UTIL_PREPARE_HPP
+
+#include "SignalData.hpp"
+
+/**
+ * @class UtilReleaseReq
+ * @brief Release Prepared transaction in Util block
+ *
+ * Data format:
+ * - UTIL_PREPARE_RELEASE_REQ <UtilPrepareId>
+ */
+class UtilReleaseReq {
+ friend class DbUtil;
+ friend class Trix;
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+private:
+ Uint32 senderData; // MUST be no 1!
+ Uint32 prepareId;
+};
+
+
+/**
+ * @class UtilReleaseConf
+ *
+ * Data format:
+ * - UTIL_PREPARE_CONF <UtilPrepareId>
+ */
+
+class UtilReleaseConf {
+ friend class DbUtil;
+ friend class Trix;
+
+ STATIC_CONST( SignalLength = 1 );
+
+private:
+ Uint32 senderData; // MUST be no 1!
+};
+
+
+/**
+ * @class UtilReleaseRef
+ *
+ * Data format:
+ * - UTIL_PREPARE_RELEASE_REF
+ */
+
+class UtilReleaseRef {
+ friend class DbUtil;
+ friend class Trix;
+
+ enum ErrorCode {
+ NO_ERROR = 0,
+ NO_SUCH_PREPARE_SEIZED = 1
+ };
+
+ STATIC_CONST( SignalLength = 3 );
+
+private:
+ Uint32 senderData; // MUST be no 1!
+ Uint32 prepareId;
+ Uint32 errorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/UtilSequence.hpp b/ndb/include/kernel/signaldata/UtilSequence.hpp
new file mode 100644
index 00000000000..50e5d673e99
--- /dev/null
+++ b/ndb/include/kernel/signaldata/UtilSequence.hpp
@@ -0,0 +1,101 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef UTIL_SEQUENCE_HPP
+#define UTIL_SEQUENCE_HPP
+
+#include "SignalData.hpp"
+
+class UtilSequenceReq {
+
+ /**
+ * Receiver
+ */
+ friend class DbUtil;
+
+ /**
+ * Sender
+ */
+ friend class Backup;
+ friend class Suma;
+
+ friend bool printUTIL_SEQUENCE_REQ(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 3 );
+
+ enum RequestType {
+ NextVal = 1, // Return uniq value
+ CurrVal = 2, // Read
+ Create = 3 // Create a sequence
+ };
+private:
+ Uint32 senderData;
+ Uint32 sequenceId; // Number of sequence variable
+ Uint32 requestType;
+};
+
+class UtilSequenceConf {
+
+ /**
+ * Receiver
+ */
+ friend class Backup;
+ friend class Suma;
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ friend bool printUTIL_SEQUENCE_CONF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+private:
+ Uint32 senderData;
+ Uint32 sequenceId;
+ Uint32 requestType;
+ Uint32 sequenceValue[2];
+};
+
+class UtilSequenceRef {
+
+ /**
+ * Reciver
+ */
+ friend class Backup;
+ friend class Suma;
+ /**
+ * Sender
+ */
+ friend class DbUtil;
+
+ friend bool printUTIL_SEQUENCE_REF(FILE *, const Uint32 *, Uint32, Uint16);
+public:
+ STATIC_CONST( SignalLength = 5 );
+
+ enum ErrorCode {
+ NoSuchSequence = 1,
+ TCError = 2
+ };
+private:
+ Uint32 senderData;
+ Uint32 sequenceId;
+ Uint32 requestType;
+ Uint32 errorCode;
+ Uint32 TCErrorCode;
+};
+
+#endif
diff --git a/ndb/include/kernel/signaldata/WaitGCP.hpp b/ndb/include/kernel/signaldata/WaitGCP.hpp
new file mode 100644
index 00000000000..ebed28714d2
--- /dev/null
+++ b/ndb/include/kernel/signaldata/WaitGCP.hpp
@@ -0,0 +1,109 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef WAIT_GCP_HPP
+#define WAIT_GCP_HPP
+
+/**
+ * This signal is sent by anyone to local DIH
+ *
+ * If local DIH is not master, it forwards it to master DIH
+ * and start acting as a proxy
+ *
+ */
+class WaitGCPReq {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Sender
+ */
+ friend class Ndbcntr;
+ friend class Dbdict;
+ friend class Backup;
+ //friend class Grep::PSCoord;
+
+public:
+ STATIC_CONST( SignalLength = 3 );
+public:
+ enum RequestType {
+ Complete = 1, ///< Wait for a GCP to complete
+ CompleteForceStart = 2, ///< Wait for a GCP to complete start one if needed
+ CompleteIfRunning = 3, ///< Wait for ongoing GCP
+ CurrentGCI = 8 ///< Immediately return current GCI
+ };
+
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 requestType;
+};
+
+class WaitGCPConf {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+ friend class Dbdict;
+ friend class Backup;
+ //friend class Grep::PSCoord;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+public:
+ Uint32 senderData;
+ Uint32 gcp;
+};
+
+class WaitGCPRef {
+
+ /**
+ * Sender(s) / Reciver(s)
+ */
+ friend class Dbdih;
+
+ /**
+ * Reciver(s)
+ */
+ friend class Ndbcntr;
+ friend class Dbdict;
+ friend class Backup;
+ friend class Grep;
+
+public:
+ STATIC_CONST( SignalLength = 2 );
+
+ enum ErrorCode {
+ StopOK = 0,
+ NF_CausedAbortOfProcedure = 1,
+ NoWaitGCPRecords = 2
+ };
+
+private:
+ Uint32 errorCode;
+ Uint32 senderData;
+};
+
+#endif
diff --git a/ndb/include/kernel/trigger_definitions.h b/ndb/include/kernel/trigger_definitions.h
new file mode 100644
index 00000000000..a5e7fb1953c
--- /dev/null
+++ b/ndb/include/kernel/trigger_definitions.h
@@ -0,0 +1,66 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_TRIGGER_DEFINITIONS_H
+#define NDB_TRIGGER_DEFINITIONS_H
+
+#include "ndb_limits.h"
+
+#ifndef MIN
+#define MIN(x,y) (((x)<(y))?(x):(y))
+#endif
+
+#ifndef MAX
+#define MAX(x,y) (((x)>(y))?(x):(y))
+#endif
+
+#define ILLEGAL_TRIGGER_ID ((Uint32)(~0))
+
+struct TriggerType {
+ enum Value {
+ CONSTRAINT = 0,
+ SECONDARY_INDEX = 1,
+ FOREIGN_KEY = 2,
+ SCHEMA_UPGRADE = 3,
+ API_TRIGGER = 4,
+ SQL_TRIGGER = 5,
+ SUBSCRIPTION = 6,
+ READ_ONLY_CONSTRAINT = 7,
+ ORDERED_INDEX = 8,
+ SUBSCRIPTION_BEFORE = 9
+ };
+};
+
+struct TriggerActionTime {
+ enum Value {
+ TA_BEFORE = 0, // Immediate, before operation
+ TA_AFTER = 1, // Immediate, after operation
+ TA_DEFERRED = 2, // Before commit
+ TA_DETACHED = 3, // After commit in a separate transaction, NYI
+ TA_CUSTOM = 4 // Hardcoded per TriggerType
+ };
+};
+
+struct TriggerEvent {
+ enum Value {
+ TE_INSERT = 0,
+ TE_DELETE = 1,
+ TE_UPDATE = 2,
+ TE_CUSTOM = 3 // Hardcoded per TriggerType
+ };
+};
+
+#endif
diff --git a/ndb/include/logger/ConsoleLogHandler.hpp b/ndb/include/logger/ConsoleLogHandler.hpp
new file mode 100644
index 00000000000..ae77b13d3b7
--- /dev/null
+++ b/ndb/include/logger/ConsoleLogHandler.hpp
@@ -0,0 +1,57 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CONSOLELOGHANDLER_H
+#define CONSOLELOGHANDLER_H
+
+#include "LogHandler.hpp"
+
+/**
+ * Logs messages to the console/stdout.
+ *
+ * @see LogHandler
+ * @version #@ $Id: ConsoleLogHandler.hpp,v 1.2 2003/09/01 10:15:53 innpeno Exp $
+ */
+class ConsoleLogHandler : public LogHandler
+{
+public:
+ /**
+ * Default constructor.
+ */
+ ConsoleLogHandler();
+ /**
+ * Destructor.
+ */
+ virtual ~ConsoleLogHandler();
+
+ virtual bool open();
+ virtual bool close();
+
+ virtual bool setParam(const BaseString &param, const BaseString &value);
+
+protected:
+ virtual void writeHeader(const char* pCategory, Logger::LoggerLevel level);
+ virtual void writeMessage(const char* pMsg);
+ virtual void writeFooter();
+
+private:
+ /** Prohibit*/
+ ConsoleLogHandler(const ConsoleLogHandler&);
+ ConsoleLogHandler operator = (const ConsoleLogHandler&);
+ bool operator == (const ConsoleLogHandler&);
+
+};
+#endif
diff --git a/ndb/include/logger/FileLogHandler.hpp b/ndb/include/logger/FileLogHandler.hpp
new file mode 100644
index 00000000000..ae69a2f5418
--- /dev/null
+++ b/ndb/include/logger/FileLogHandler.hpp
@@ -0,0 +1,110 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FILELOGHANDLER_H
+#define FILELOGHANDLER_H
+
+#include "LogHandler.hpp"
+
+class File;
+
+/**
+ * Logs messages to a file. The log file will be archived depending on
+ * the file's size or after N number of log entries.
+ * There will be only a specified number of archived logs
+ * which will be "recycled".
+ *
+ * The archived log file will be named as <filename>.1..N.
+ *
+ *
+ * @see LogHandler
+ * @version #@ $Id: FileLogHandler.hpp,v 1.2 2003/09/01 10:15:53 innpeno Exp $
+ */
+class FileLogHandler : public LogHandler
+{
+public:
+ /** Max number of log files to archive. */
+ static const int MAX_NO_FILES = 6;
+ /** Max file size of the log before archiving. */
+ static const long MAX_FILE_SIZE = 1024000;
+ /** Max number of log entries before archiving. */
+ static const unsigned int MAX_LOG_ENTRIES = 10000;
+
+ /**
+ * Default constructor.
+ */
+ FileLogHandler();
+
+ /**
+ * Creates a new file handler with the specified filename,
+ * max number of archived log files and max log size for each log.
+ *
+ * @param aFileName the log filename.
+ * @param maxNoFiles the maximum number of archived log files.
+ * @param maxFileSize the maximum log file size before archiving.
+ * @param maxLogEntries the maximum number of log entries before checking time to archive.
+ */
+ FileLogHandler(const char* aFileName,
+ int maxNoFiles = MAX_NO_FILES,
+ long maxFileSize = MAX_FILE_SIZE,
+ unsigned int maxLogEntries = MAX_LOG_ENTRIES);
+
+ /**
+ * Destructor.
+ */
+ virtual ~FileLogHandler();
+
+ virtual bool open();
+ virtual bool close();
+
+ virtual bool setParam(const BaseString &param, const BaseString &value);
+ virtual bool checkParams();
+
+protected:
+ virtual void writeHeader(const char* pCategory, Logger::LoggerLevel level);
+ virtual void writeMessage(const char* pMsg);
+ virtual void writeFooter();
+
+private:
+ /** Prohibit */
+ FileLogHandler(const FileLogHandler&);
+ FileLogHandler operator = (const FileLogHandler&);
+ bool operator == (const FileLogHandler&);
+
+ /**
+ * Returns true if it is time to create a new log file.
+ */
+ bool isTimeForNewFile();
+
+ /**
+ * Archives the current log file and creates a new one.
+ * The archived log filename will be in the format of <filename>.N
+ *
+ * @return true if successful.
+ */
+ bool createNewFile();
+
+ bool setFilename(const BaseString &filename);
+ bool setMaxSize(const BaseString &size);
+ bool setMaxFiles(const BaseString &files);
+
+ int m_maxNoFiles;
+ long m_maxFileSize;
+ unsigned int m_maxLogEntries;
+ File* m_pLogFile;
+};
+
+#endif
diff --git a/ndb/include/logger/LogHandler.hpp b/ndb/include/logger/LogHandler.hpp
new file mode 100644
index 00000000000..8c5c9298f69
--- /dev/null
+++ b/ndb/include/logger/LogHandler.hpp
@@ -0,0 +1,198 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LOGHANDLER_H
+#define LOGHANDLER_H
+
+#include "Logger.hpp"
+
+#include <NdbStdio.h> // Defines NULL
+
+/**
+ * This class is the base class for all log handlers. A log handler is
+ * responsible for formatting and writing log messages to a specific output.
+ *
+ * A log entry consists of three parts: a header, <body/log message and a footer.
+ * <pre>
+ * 09:17:37 2002-03-13 [MgmSrv] INFO -- Local checkpoint 13344 started.
+ * </pre>
+ *
+ * Header format: TIME&DATE CATEGORY LEVEL --
+ * TIME&DATE = ctime() format.
+ * CATEGORY = Any string.
+ * LEVEL = ALERT to DEBUG (Log levels)
+ *
+ * Footer format: \n (currently only newline)
+ *
+ * @version #@ $Id: LogHandler.hpp,v 1.7 2003/09/01 10:15:53 innpeno Exp $
+ */
+class LogHandler
+{
+public:
+ /**
+ * Default constructor.
+ */
+ LogHandler();
+
+ /**
+ * Destructor.
+ */
+ virtual ~LogHandler();
+
+ /**
+ * Opens/initializes the log handler.
+ *
+ * @return true if successful.
+ */
+ virtual bool open() = 0;
+
+ /**
+ * Closes/free any allocated resources used by the log handler.
+ *
+ * @return true if successful.
+ */
+ virtual bool close() = 0;
+
+ /**
+ * Append a log message to the output stream/file whatever.
+ * append() will call writeHeader(), writeMessage() and writeFooter() for
+ * a child class and in that order.
+ *
+ * @param pCategory the category/name to tag the log entry with.
+ * @param level the log level.
+ * @param pMsg the log message.
+ */
+ void append(const char* pCategory, Logger::LoggerLevel level,
+ const char* pMsg);
+
+ /**
+ * Returns a default formatted header. It currently has the
+ * follwing default format: '%H:%M:%S %Y-%m-%d [CATEGORY] LOGLEVEL --'
+ *
+ * @param pStr the header string to format.
+ * @param pCategory a category/name to tag the log entry with.
+ * @param level the log level.
+ * @return the header.
+ */
+ const char* getDefaultHeader(char* pStr, const char* pCategory,
+ Logger::LoggerLevel level) const;
+
+ /**
+ * Returns a default formatted footer. Currently only returns a newline.
+ *
+ * @return the footer.
+ */
+ const char* getDefaultFooter() const;
+
+ /**
+ * Returns the date and time format used by ctime().
+ *
+ * @return the date and time format.
+ */
+ const char* getDateTimeFormat() const;
+
+ /**
+ * Sets the date and time format. It needs to have the same arguments
+ * a ctime().
+ *
+ * @param pFormat the date and time format.
+ */
+ void setDateTimeFormat(const char* pFormat);
+
+ /**
+ * Returns a string date and time string.
+ *
+ * @param pStr a string.
+ * @return a string with date and time.
+ */
+ char* getTimeAsString(char* pStr) const;
+
+ /**
+ * Returns the error code.
+ */
+ int getErrorCode() const;
+
+ /**
+ * Sets the error code.
+ *
+ * @param code the error code.
+ */
+ void setErrorCode(int code);
+
+ /**
+ * Parse logstring parameters
+ *
+ * @param params list of parameters, formatted as "param=value",
+ * entries separated by ","
+ * @return true on success, false on failure
+ */
+ bool parseParams(const BaseString &params);
+
+ /**
+ * Sets a parameters. What parameters are accepted depends on the subclass.
+ *
+ * @param param name of parameter
+ * @param value value of parameter
+ */
+ virtual bool setParam(const BaseString &param, const BaseString &value) = 0;
+
+ /**
+ * Checks that all necessary parameters have been set.
+ *
+ * @return true if all parameters are correctly set, false otherwise
+ */
+ virtual bool checkParams();
+
+protected:
+ /** Max length of the date and time header in the log. */
+ static const int MAX_DATE_TIME_HEADER_LENGTH = 64;
+ /** Max length of the header the log. */
+ static const int MAX_HEADER_LENGTH = 128;
+ /** Max lenght of footer in the log. */
+ static const int MAX_FOOTER_LENGTH = 128;
+
+ /**
+ * Write the header to the log.
+ *
+ * @param pCategory the category to tag the log with.
+ * @param level the log level.
+ */
+ virtual void writeHeader(const char* category, Logger::LoggerLevel level) = 0;
+
+ /**
+ * Write the message to the log.
+ *
+ * @param pMsg the message to log.
+ */
+ virtual void writeMessage(const char* pMsg) = 0;
+
+ /**
+ * Write the footer to the log.
+ *
+ */
+ virtual void writeFooter() = 0;
+
+private:
+ /** Prohibit */
+ LogHandler(const LogHandler&);
+ LogHandler* operator = (const LogHandler&);
+ bool operator == (const LogHandler&);
+
+ const char* m_pDateTimeFormat;
+ int m_errorCode;
+};
+
+#endif
diff --git a/ndb/include/logger/Logger.hpp b/ndb/include/logger/Logger.hpp
new file mode 100644
index 00000000000..2d12a5b8a6e
--- /dev/null
+++ b/ndb/include/logger/Logger.hpp
@@ -0,0 +1,294 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef Logger_H
+#define Logger_H
+
+#include <BaseString.hpp>
+#include <stdarg.h>
+
+class LogHandler;
+class LogHandlerList;
+
+/**
+ * Logger should be used whenver you need to log a message like
+ * general information or debug messages. By creating/adding different
+ * log handlers, a single log message can be sent to
+ * different outputs (stdout, file or syslog).
+ *
+ * Each log entry is created with a log level (or severity) which is
+ * used to identity the type of the entry, e.g., if it is a debug
+ * or an error message.
+ *
+ * Example of a log entry:
+ *
+ * 09:17:39 2002-03-13 [myLogger] INFO -- Local checkpoint started.
+ *
+ * HOW TO USE
+ *
+ * 1) Create a new instance of the Logger.
+ *
+ * Logger myLogger = new Logger();
+ *
+ * 2) Add the log handlers that you want, i.e., where the log entries
+ * should be written/shown.
+ *
+ * myLogger->createConsoleHandler(); // Output to console/stdout
+ * myLogger->addHandler(new FileLogHandler("mylog.txt")); // use mylog.txt
+ *
+ * 3) Tag each log entry with a category/name.
+ *
+ * myLogger->setCategory("myLogger");
+ *
+ * 4) Start log messages.
+ *
+ * myLogger->alert("T-9 to lift off");
+ * myLogger->info("Here comes the sun, la la");
+ * myLogger->debug("Why does this not work!!!, We should not be here...")
+ *
+ * 5) Log only debug messages.
+ *
+ * myLogger->enable(Logger::LL_DEBUG);
+ *
+ * 6) Log only ALERTS and ERRORS.
+ *
+ * myLogger->enable(Logger::LL_ERROR, Logger::LL_ALERT);
+ *
+ * 7) Do not log any messages.
+ *
+ * myLogger->disable(Logger::LL_ALL);
+ *
+ *
+ * LOG LEVELS (Matches the severity levels of syslog)
+ * <pre>
+ *
+ * ALERT A condition that should be corrected
+ * immediately, such as a corrupted system
+ * database.
+ *
+ * CRITICAL Critical conditions, such as hard device
+ * errors.
+ *
+ * ERROR Errors.
+ *
+ * WARNING Warning messages.
+ *
+ * INFO Informational messages.
+ *
+ * DEBUG Messages that contain information nor-
+ * mally of use only when debugging a pro-
+ * gram.
+ * </pre>
+ *
+ * @version #@ $Id: Logger.hpp,v 1.7 2003/09/01 10:15:53 innpeno Exp $
+ */
+class Logger
+{
+public:
+ /** The log levels. NOTE: Could not use the name LogLevel since
+ * it caused conflicts with another class.
+ */
+ enum LoggerLevel {LL_OFF, LL_DEBUG, LL_INFO, LL_WARNING, LL_ERROR,
+ LL_CRITICAL, LL_ALERT, LL_ALL};
+
+ /**
+ * String representation of the the log levels.
+ */
+ static const char* LoggerLevelNames[];
+
+ /**
+ * Default constructor.
+ */
+ Logger();
+
+ /**
+ * Destructor.
+ */
+ virtual ~Logger();
+
+ /**
+ * Set a category/name that each log entry will have.
+ *
+ * @param pCategory the category.
+ */
+ void setCategory(const char* pCategory);
+
+ /**
+ * Create a default handler that logs to the console/stdout.
+ *
+ * @return true if successful.
+ */
+ bool createConsoleHandler();
+
+ /**
+ * Remove the default console handler.
+ */
+ void removeConsoleHandler();
+
+ /**
+ * Create a default handler that logs to a file called logger.log.
+ *
+ * @return true if successful.
+ */
+ bool createFileHandler();
+
+ /**
+ * Remove the default file handler.
+ */
+ void removeFileHandler();
+
+ /**
+ * Create a default handler that logs to the syslog.
+ *
+ * On OSE a ConsoleHandler will be created since there is no syslog support.
+ *
+ * @return true if successful.
+ */
+ bool createSyslogHandler();
+
+ /**
+ * Remove the default syslog handler.
+ */
+ void removeSyslogHandler();
+
+ /**
+ * Add a new log handler.
+ *
+ * @param pHandler a log handler.
+ * @return true if successful.
+ */
+ bool addHandler(LogHandler* pHandler);
+
+ /**
+ * Add a new handler
+ *
+ * @param logstring string describing the handler to add
+ */
+ bool addHandler(const BaseString &logstring);
+
+ /**
+ * Remove a log handler.
+ *
+ * @param pHandler log handler to remove.
+ * @return true if successful.
+ */
+ bool removeHandler(LogHandler* pHandler);
+
+ /**
+ * Remove all log handlers.
+ */
+ void removeAllHandlers();
+
+ /**
+ * Returns true if the specified log level is enabled.
+ *
+ * @return true if enabled.
+ */
+ bool isEnable(LoggerLevel logLevel) const;
+
+ /**
+ * Enable the specified log level.
+ *
+ * @param logLevel the loglevel to enable.
+ */
+ void enable(LoggerLevel logLevel);
+
+ /**
+ * Enable log levels.
+ *
+ * @param fromLogLevel enable from log level.
+ * @param toLogLevel enable to log level.
+ */
+ void enable (LoggerLevel fromLogLevel, LoggerLevel toLogLevel);
+
+ /**
+ * Disable log level.
+ *
+ * @param logLevel disable log level.
+ */
+ void disable(LoggerLevel logLevel);
+
+ /**
+ * Log an alert message.
+ *
+ * @param pMsg the message.
+ */
+ virtual void alert(const char* pMsg, ...) const;
+ virtual void alert(BaseString &pMsg) const { alert(pMsg.c_str()); };
+
+ /**
+ * Log a critical message.
+ *
+ * @param pMsg the message.
+ */
+ virtual void critical(const char* pMsg, ...) const;
+ virtual void critical(BaseString &pMsg) const { critical(pMsg.c_str()); };
+
+ /**
+ * Log an error message.
+ *
+ * @param pMsg the message.
+ */
+ virtual void error(const char* pMsg, ...) const;
+ virtual void error(BaseString &pMsg) const { error(pMsg.c_str()); };
+
+ /**
+ * Log a warning message.
+ *
+ * @param pMsg the message.
+ */
+ virtual void warning(const char* pMsg, ...) const;
+ virtual void warning(BaseString &pMsg) const { warning(pMsg.c_str()); };
+
+ /**
+ * Log an info message.
+ *
+ * @param pMsg the message.
+ */
+ virtual void info(const char* pMsg, ...) const;
+ virtual void info(BaseString &pMsg) const { info(pMsg.c_str()); };
+
+ /**
+ * Log a debug message.
+ *
+ * @param pMsg the message.
+ */
+ virtual void debug(const char* pMsg, ...) const;
+ virtual void debug(BaseString &pMsg) const { debug(pMsg.c_str()); };
+
+protected:
+
+ void log(LoggerLevel logLevel, const char* msg, va_list ap) const;
+
+private:
+ /** Prohibit */
+ Logger(const Logger&);
+ Logger operator = (const Logger&);
+ bool operator == (const Logger&);
+
+ static const int MAX_LOG_LEVELS = 8;
+
+ bool m_logLevels[MAX_LOG_LEVELS];
+
+ LogHandlerList* m_pHandlerList;
+ const char* m_pCategory;
+ /* Default handlers */
+ LogHandler* m_pConsoleHandler;
+ LogHandler* m_pFileHandler;
+ LogHandler* m_pSyslogHandler;
+};
+
+#endif
diff --git a/ndb/include/logger/SysLogHandler.hpp b/ndb/include/logger/SysLogHandler.hpp
new file mode 100644
index 00000000000..4f13308d61b
--- /dev/null
+++ b/ndb/include/logger/SysLogHandler.hpp
@@ -0,0 +1,97 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SYSLOGHANDLER_H
+#define SYSLOGHANDLER_H
+
+#include "LogHandler.hpp"
+#include <syslog.h>
+
+/**
+ * Logs messages to syslog. The default identity is 'NDB'.
+ * See 'man 3 syslog'.
+ *
+ * It logs the following severity levels.
+ * <pre>
+ *
+ * LOG_ALERT A condition that should be corrected
+ * immediately, such as a corrupted system
+ * database.
+ *
+ * LOG_CRIT Critical conditions, such as hard device
+ * errors.
+ *
+ * LOG_ERR Errors.
+ *
+ * LOG_WARNING Warning messages.
+ *
+ * LOG_INFO Informational messages.
+ *
+ * LOG_DEBUG Messages that contain information nor-
+ * mally of use only when debugging a pro-
+ * gram.
+ * </pre>
+ *
+ * @see LogHandler
+ * @version #@ $Id: SysLogHandler.hpp,v 1.2 2003/09/01 10:15:53 innpeno Exp $
+ */
+class SysLogHandler : public LogHandler
+{
+public:
+ /**
+ * Default constructor.
+ */
+ SysLogHandler();
+
+ /**
+ * Create a new syslog handler with the specified identity.
+ *
+ * @param pIdentity a syslog identity.
+ * @param facility syslog facility, defaults to LOG_USER
+ */
+ SysLogHandler(const char* pIdentity, int facility = LOG_USER);
+
+ /**
+ * Destructor.
+ */
+ virtual ~SysLogHandler();
+
+ virtual bool open();
+ virtual bool close();
+
+ virtual bool setParam(const BaseString &param, const BaseString &value);
+ bool setFacility(const BaseString &facility);
+
+protected:
+ virtual void writeHeader(const char* pCategory, Logger::LoggerLevel level);
+ virtual void writeMessage(const char* pMsg);
+ virtual void writeFooter();
+
+private:
+ /** Prohibit*/
+ SysLogHandler(const SysLogHandler&);
+ SysLogHandler operator = (const SysLogHandler&);
+ bool operator == (const SysLogHandler&);
+
+ int m_severity;
+ const char* m_pCategory;
+
+ /** Syslog identity for all log entries. */
+ const char* m_pIdentity;
+ int m_facility;
+};
+
+#endif
diff --git a/ndb/include/mgmapi/mgmapi.h b/ndb/include/mgmapi/mgmapi.h
new file mode 100644
index 00000000000..c74a046b7e7
--- /dev/null
+++ b/ndb/include/mgmapi/mgmapi.h
@@ -0,0 +1,663 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MGMAPI_H
+#define MGMAPI_H
+
+/**
+ * @mainpage NDB Cluster Management API
+ *
+ * The NDB Cluster Management API (MGM API) is a C API
+ * that is used to:
+ * - Start/stop database nodes (DB nodes)
+ * - Start/stop NDB Cluster backups
+ * - Control the NDB Cluster log
+ * - Other administrative tasks
+ *
+ * @section General Concepts
+ *
+ * Each MGM API function needs a management server handle
+ * (of type Mgm_C_Api::NdbMgmHandle).
+ * This handle is initally is created by calling the
+ * function ndb_mgm_create_handle().
+ *
+ * A function can return:
+ * -# An integer value.
+ * If it returns -1 then this indicates an error, and then
+ * -# A pointer value. If it returns NULL then check the latest error.
+ * If it didn't return NULL, then a "something" is returned.
+ * This "something" has to be free:ed by the user of the MGM API.
+ *
+ * If there are an error, then the get latest error functions
+ * can be used to check what the error was.
+ */
+
+/** @addtogroup MGM_C_API
+ * @{
+ */
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+ /**
+ * Format of statistical information from the NDB Cluster.
+ * STATISTIC_LINE is sent on the statistical port from the Management server,
+ * each line is timestamped with STATISTIC_DATE.
+ */
+#define STATISTIC_LINE "date=%s epochsecs=%d nodeid=%u trans=%u commit=%u " \
+ "read=%u insert=%u attrinfo=%u cops=%u abort=%u"
+ /**
+ * Format of statistical information from the NDB Cluster.
+ * STATISTIC_LINE is sent on the statistical port from the Management server,
+ * each line is timestamped with STATISTIC_DATE.
+ */
+#define STATISTIC_DATE "%d-%.2d-%.2d/%.2d:%.2d:%.2d"
+ /**
+ * Format of statistical information from the NDB Cluster.
+ */
+#define OP_STATISTIC_LINE "date=%s epochsecs=%d nodeid=%d operations=%u"
+
+ /**
+ * The NdbMgmHandle.
+ */
+ typedef struct ndb_mgm_handle * NdbMgmHandle;
+
+ /**
+ * NDB Cluster node types
+ */
+ enum ndb_mgm_node_type {
+ NDB_MGM_NODE_TYPE_UNKNOWN = -1, ///< Node type not known
+ NDB_MGM_NODE_TYPE_API = 0, ///< An application node (API)
+ NDB_MGM_NODE_TYPE_NDB = 1, ///< A database node (DB)
+ NDB_MGM_NODE_TYPE_MGM = 2, ///< A management server node (MGM)
+
+ NDB_MGM_NODE_TYPE_MIN = 0, ///< Min valid value
+ NDB_MGM_NODE_TYPE_MAX = 2 ///< Max valid value
+ };
+
+ /**
+ * Database node status
+ */
+ enum ndb_mgm_node_status {
+ NDB_MGM_NODE_STATUS_UNKNOWN = 0, ///< Node status not known
+ NDB_MGM_NODE_STATUS_NO_CONTACT = 1, ///< No contact with node
+ NDB_MGM_NODE_STATUS_NOT_STARTED = 2, ///< Has not run starting protocol
+ NDB_MGM_NODE_STATUS_STARTING = 3, ///< Is running starting protocol
+ NDB_MGM_NODE_STATUS_STARTED = 4, ///< Running
+ NDB_MGM_NODE_STATUS_SHUTTING_DOWN = 5, ///< Is shutting down
+ NDB_MGM_NODE_STATUS_RESTARTING = 6, ///< Is restarting
+ NDB_MGM_NODE_STATUS_SINGLEUSER = 7, ///< Maintenance mode
+ NDB_MGM_NODE_STATUS_RESUME = 8, ///< Resume mode
+
+ NDB_MGM_NODE_STATUS_MIN = 0, ///< Min valid value
+ NDB_MGM_NODE_STATUS_MAX = 6 ///< Max valid value
+ };
+
+ /**
+ * Error codes
+ */
+ enum ndb_mgm_error {
+ NDB_MGM_NO_ERROR = 0,
+
+ /* Request for service errors */
+ NDB_MGM_ILLEGAL_CONNECT_STRING = 1001,
+ NDB_MGM_ILLEGAL_PORT_NUMBER = 1002,
+ NDB_MGM_ILLEGAL_SOCKET = 1003,
+ NDB_MGM_ILLEGAL_IP_ADDRESS = 1004,
+ NDB_MGM_ILLEGAL_SERVER_HANDLE = 1005,
+ NDB_MGM_ILLEGAL_SERVER_REPLY = 1006,
+ NDB_MGM_ILLEGAL_NUMBER_OF_NODES = 1007,
+ NDB_MGM_ILLEGAL_NODE_STATUS = 1008,
+ NDB_MGM_OUT_OF_MEMORY = 1009,
+ NDB_MGM_SERVER_NOT_CONNECTED = 1010,
+ NDB_MGM_COULD_NOT_CONNECT_TO_SOCKET = 1011,
+
+ /* Service errors - Start/Stop Node or System */
+ NDB_MGM_START_FAILED = 2001,
+ NDB_MGM_STOP_FAILED = 2002,
+ NDB_MGM_RESTART_FAILED = 2003,
+
+ /* Service errors - Backup */
+ NDB_MGM_COULD_NOT_START_BACKUP = 3001,
+ NDB_MGM_COULD_NOT_ABORT_BACKUP = 3002,
+
+ /* Service errors - Single User Mode */
+ NDB_MGM_COULD_NOT_ENTER_SINGLE_USER_MODE = 4001,
+ NDB_MGM_COULD_NOT_EXIT_SINGLE_USER_MODE = 4002
+ };
+
+ struct Ndb_Mgm_Error_Msg {
+ enum ndb_mgm_error code;
+ const char * msg;
+ };
+
+ const struct Ndb_Mgm_Error_Msg ndb_mgm_error_msgs[] = {
+ { NDB_MGM_NO_ERROR, "No error" },
+
+ { NDB_MGM_ILLEGAL_CONNECT_STRING, "Illegal connect string" },
+ { NDB_MGM_ILLEGAL_PORT_NUMBER, "Illegal port number" },
+ { NDB_MGM_ILLEGAL_SOCKET, "Illegal socket" },
+ { NDB_MGM_ILLEGAL_IP_ADDRESS, "Illegal IP address" },
+ { NDB_MGM_ILLEGAL_SERVER_HANDLE, "Illegal server handle" },
+ { NDB_MGM_ILLEGAL_SERVER_REPLY, "Illegal reply from server" },
+ { NDB_MGM_ILLEGAL_NUMBER_OF_NODES, "Illegal number of nodes" },
+ { NDB_MGM_ILLEGAL_NODE_STATUS, "Illegal node status" },
+ { NDB_MGM_OUT_OF_MEMORY, "Out of memory" },
+ { NDB_MGM_SERVER_NOT_CONNECTED, "Management server not connected" },
+ { NDB_MGM_COULD_NOT_CONNECT_TO_SOCKET, "Could not connect to socket" },
+
+ /* Service errors - Start/Stop Node or System */
+ { NDB_MGM_START_FAILED, "Start failed" },
+ { NDB_MGM_STOP_FAILED, "Stop failed" },
+ { NDB_MGM_RESTART_FAILED, "Restart failed" },
+
+ /* Service errors - Backup */
+ { NDB_MGM_COULD_NOT_START_BACKUP, "Could not start backup" },
+ { NDB_MGM_COULD_NOT_ABORT_BACKUP, "Could not abort backup" },
+
+ /* Service errors - Single User Mode */
+ { NDB_MGM_COULD_NOT_ENTER_SINGLE_USER_MODE,
+ "Could not enter single user mode" },
+ { NDB_MGM_COULD_NOT_EXIT_SINGLE_USER_MODE,
+ "Could not exit single user mode" }
+ };
+
+ const int ndb_mgm_noOfErrorMsgs =
+ sizeof(ndb_mgm_error_msgs)/sizeof(struct Ndb_Mgm_Error_Msg);
+
+ /**
+ * Structure returned by ndb_mgm_get_status
+ */
+ struct ndb_mgm_node_state {
+ int node_id; ///< NDB Cluster node id
+ enum ndb_mgm_node_type node_type; ///< Type of NDB Cluster node
+ enum ndb_mgm_node_status node_status; ///< State of node
+ int start_phase; ///< Start phase.
+ ///< @note Start phase is only
+ ///< valid if
+ ///< node_type is
+ ///< NDB_MGM_NODE_TYPE_NDB and
+ ///< node_status is
+ ///< NDB_MGM_NODE_STATUS_STARTING
+ int dynamic_id; ///< Id for heartbeats and
+ ///< master take-over
+ ///< (only valid for DB nodes)
+ int node_group; ///< Node group of node
+ ///< (only valid for DB nodes)
+ int version; ///< Internal version number
+ };
+
+ /**
+ * Cluster status
+ */
+ struct ndb_mgm_cluster_state {
+ int no_of_nodes; ///< No of entries in the
+ ///< node_states array
+ struct ndb_mgm_node_state ///< An array with node_states
+ node_states[1];
+ };
+
+ /**
+ * Default reply from the server
+ */
+ struct ndb_mgm_reply {
+ int return_code; ///< 0 if successful,
+ ///< otherwise error code.
+ char message[256]; ///< Error or reply message.
+ };
+
+ /**
+ * Default information types
+ */
+ enum ndb_mgm_info {
+ NDB_MGM_INFO_CLUSTER, ///< ?
+ NDB_MGM_INFO_CLUSTERLOG ///< Cluster log
+ };
+
+ /**
+ * Signal log modes
+ * (Used only in the development of NDB Cluster.)
+ */
+ enum ndb_mgm_signal_log_mode {
+ NDB_MGM_SIGNAL_LOG_MODE_IN, ///< Log receiving signals
+ NDB_MGM_SIGNAL_LOG_MODE_OUT, ///< Log sending signals
+ NDB_MGM_SIGNAL_LOG_MODE_INOUT, ///< Log both sending/receiving
+ NDB_MGM_SIGNAL_LOG_MODE_OFF ///< Log off
+ };
+
+ /**
+ * Log severities (used to filter the cluster log)
+ */
+ enum ndb_mgm_clusterlog_level {
+ NDB_MGM_CLUSTERLOG_OFF = 0, ///< Cluster log off
+ NDB_MGM_CLUSTERLOG_DEBUG = 1, ///< Used in NDB Cluster
+ ///< developement
+ NDB_MGM_CLUSTERLOG_INFO = 2, ///< Informational messages
+ NDB_MGM_CLUSTERLOG_WARNING = 3, ///< Conditions that are not
+ ///< error condition, but
+ ///< might require handling
+ NDB_MGM_CLUSTERLOG_ERROR = 4, ///< Conditions that should be
+ ///< corrected
+ NDB_MGM_CLUSTERLOG_CRITICAL = 5, ///< Critical conditions, like
+ ///< device errors or out of
+ ///< resources
+ NDB_MGM_CLUSTERLOG_ALERT = 6, ///< A condition that should be
+ ///< corrected immediately,
+ ///< such as a corrupted system
+ NDB_MGM_CLUSTERLOG_ALL = 7 ///< All severities on
+ };
+
+ /**
+ * Log categories
+ */
+ enum ndb_mgm_event_category {
+ NDB_MGM_EVENT_CATEGORY_STARTUP, ///< Events during all kinds
+ ///< of startups
+ NDB_MGM_EVENT_CATEGORY_SHUTDOWN, ///< Events during shutdown
+ NDB_MGM_EVENT_CATEGORY_STATISTIC, ///< Transaction statistics
+ ///< (Job level, TCP/IP speed)
+ NDB_MGM_EVENT_CATEGORY_CHECKPOINT, ///< Checkpoints
+ NDB_MGM_EVENT_CATEGORY_NODE_RESTART, ///< Events during node restart
+ NDB_MGM_EVENT_CATEGORY_CONNECTION, ///< Events related to connection
+ ///< and communication
+ NDB_MGM_EVENT_CATEGORY_ERROR ///< Assorted event w.r.t.
+ ///< unexpected happenings
+ };
+
+ /***************************************************************************/
+ /**
+ * @name Functions: Error Handling
+ * @{
+ */
+
+ /**
+ * Get latest error associated with a management server handle
+ *
+ * @param handle Management handle
+ * @return Latest error code
+ */
+ int ndb_mgm_get_latest_error(const NdbMgmHandle handle);
+
+ /**
+ * Get latest main error message associated with a handle
+ *
+ * @param handle Management handle.
+ * @return Latest error message
+ */
+ const char * ndb_mgm_get_latest_error_msg(const NdbMgmHandle handle);
+
+ /**
+ * Get latest error description associated with a handle
+ *
+ * The error description gives some additional information to
+ * the error message.
+ *
+ * @param handle Management handle.
+ * @return Latest error description
+ */
+ const char * ndb_mgm_get_latest_error_desc(const NdbMgmHandle handle);
+
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+ /**
+ * Get latest internal source code error line associated with a handle
+ *
+ * @param handle Management handle.
+ * @return Latest internal source code line of latest error
+ * @deprecated
+ */
+ int ndb_mgm_get_latest_error_line(const NdbMgmHandle handle);
+#endif
+
+ /** @} *********************************************************************/
+ /**
+ * @name Functions: Create/Destroy Management Server Handles
+ * @{
+ */
+
+ /**
+ * Create a handle to a management server
+ *
+ * @return A management handle<br>
+ * or NULL if no management handle could be created.
+ */
+ NdbMgmHandle ndb_mgm_create_handle();
+
+ /**
+ * Destroy a management server handle
+ *
+ * @param handle Management handle
+ */
+ void ndb_mgm_destroy_handle(NdbMgmHandle * handle);
+
+ /** @} *********************************************************************/
+ /**
+ * @name Functions: Connect/Disconnect Management Server
+ * @{
+ */
+
+ /**
+ * Connect to a management server
+ *
+ * @param handle Management handle.
+ * @param mgmsrv Hostname and port of the management server,
+ * "hostname:port".
+ * @return -1 on error.
+ */
+ int ndb_mgm_connect(NdbMgmHandle handle, const char * mgmsrv);
+
+ /**
+ * Disconnect from a management server
+ *
+ * @param handle Management handle.
+ * @return -1 on error.
+ */
+ int ndb_mgm_disconnect(NdbMgmHandle handle);
+
+ /** @} *********************************************************************/
+ /**
+ * @name Functions: Convert between different data formats
+ * @{
+ */
+
+ /**
+ * Convert a string to a ndb_mgm_node_type
+ *
+ * @param type Node type as string.
+ * @return NDB_MGM_NODE_TYPE_UNKNOWN if invalid string.
+ */
+ enum ndb_mgm_node_type ndb_mgm_match_node_type(const char * type);
+
+ /**
+ * Convert an ndb_mgm_node_type to a string
+ *
+ * @param type Node type.
+ * @return NULL if invalid id.
+ */
+ const char * ndb_mgm_get_node_type_string(enum ndb_mgm_node_type type);
+
+ /**
+ * Convert a string to a ndb_mgm_node_status
+ *
+ * @param status NDB node status string.
+ * @return NDB_MGM_NODE_STATUS_UNKNOWN if invalid string.
+ */
+ enum ndb_mgm_node_status ndb_mgm_match_node_status(const char * status);
+
+ /**
+ * Convert an id to a string
+ *
+ * @param status NDB node status.
+ * @return NULL if invalid id.
+ */
+ const char * ndb_mgm_get_node_status_string(enum ndb_mgm_node_status status);
+
+ /** @} *********************************************************************/
+ /**
+ * @name Functions: State of cluster
+ * @{
+ */
+
+ /**
+ * Get status of the nodes in an NDB Cluster
+ *
+ * Note the caller must free the pointer returned.
+ *
+ * @param handle Management handle.
+ * @return Cluster state (or NULL on error).
+ */
+ struct ndb_mgm_cluster_state * ndb_mgm_get_status(NdbMgmHandle handle);
+
+ /** @} *********************************************************************/
+ /**
+ * @name Functions: Start/stop nodes
+ * @{
+ */
+
+ /**
+ * Stop database nodes
+ *
+ * @param handle Management handle.
+ * @param no_of_nodes no of database nodes<br>
+ * 0 - means all database nodes in cluster<br>
+ * n - Means stop n node(s) specified in the
+ * array node_list
+ * @param node_list List of node ids of database nodes to be stopped
+ * @return No of nodes stopped (or -1 on error)
+ *
+ * @note The function is equivalent
+ * to ndb_mgm_stop2(handle, no_of_nodes, node_list, 0)
+ */
+ int ndb_mgm_stop(NdbMgmHandle handle, int no_of_nodes,
+ const int * node_list);
+
+ /**
+ * Stop database nodes
+ *
+ * @param handle Management handle.
+ * @param no_of_nodes No of database nodes<br>
+ * 0 - means all database nodes in cluster<br>
+ * n - Means stop n node(s) specified in
+ * the array node_list
+ * @param node_list List of node ids of database nodes to be stopped
+ * @param abort Don't perform gracefull stop,
+ * but rather stop immediatly
+ * @return No of nodes stopped (or -1 on error).
+ */
+ int ndb_mgm_stop2(NdbMgmHandle handle, int no_of_nodes,
+ const int * node_list, int abort);
+
+ /**
+ * Restart database nodes
+ *
+ * @param handle Management handle.
+ * @param no_of_nodes No of database nodes<br>
+ * 0 - means all database nodes in cluster<br>
+ * n - Means stop n node(s) specified in the
+ * array node_list
+ * @param node_list List of node ids of database nodes to be stopped
+ * @return No of nodes stopped (or -1 on error).
+ *
+ * @note The function is equivalent to
+ * ndb_mgm_restart2(handle, no_of_nodes, node_list, 0, 0, 0);
+ */
+ int ndb_mgm_restart(NdbMgmHandle handle, int no_of_nodes,
+ const int * node_list);
+
+ /**
+ * Restart database nodes
+ *
+ * @param handle Management handle.
+ * @param no_of_nodes No of database nodes<br>
+ * 0 - means all database nodes in cluster<br>
+ * n - Means stop n node(s) specified in the
+ * array node_list
+ * @param node_list List of node ids of database nodes to be stopped
+ * @param initial Remove filesystem from node(s) restarting
+ * @param nostart Don't actually start node(s) but leave them
+ * waiting for start command
+ * @param abort Don't perform gracefull restart,
+ * but rather restart immediatly
+ * @return No of nodes stopped (or -1 on error).
+ */
+ int ndb_mgm_restart2(NdbMgmHandle handle, int no_of_nodes,
+ const int * node_list, int initial,
+ int nostart, int abort);
+
+ /**
+ * Start database nodes
+ *
+ * @param handle Management handle.
+ * @param no_of_nodes No of database nodes<br>
+ * 0 - means all database nodes in cluster<br>
+ * n - Means start n node(s) specified in
+ * the array node_list
+ * @param node_list List of node ids of database nodes to be started
+ * @return No of nodes started (or -1 on error).
+ *
+ * @note The nodes to start must have been started with nostart(-n)
+ * argument.
+ * This means that the database node binary is started and
+ * waiting for a START management command which will
+ * actually start the database node functionality
+ */
+ int ndb_mgm_start(NdbMgmHandle handle,
+ int no_of_nodes,
+ const int * node_list);
+
+ /** @} *********************************************************************/
+ /**
+ * @name Functions: Logging and Statistics
+ * @{
+ */
+
+ /**
+ * Filter cluster log
+ *
+ * @param handle NDB management handle.
+ * @param level A cluster log level to filter.
+ * @param reply Reply message.
+ * @return -1 on error.
+ */
+ int ndb_mgm_filter_clusterlog(NdbMgmHandle handle,
+ enum ndb_mgm_clusterlog_level level,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Get log filter
+ *
+ * @param handle NDB management handle
+ * @return A vector of seven elements,
+ * where each element contains
+ * 1 if a severity is enabled and 0 if not.
+ * A severity is stored at position
+ * ndb_mgm_clusterlog_level,
+ * for example the "error" level is stored in position
+ * [NDB_MGM_CLUSTERLOG_ERROR-1].
+ * The first element in the vector signals
+ * whether the clusterlog
+ * is disabled or enabled.
+ */
+ unsigned int *ndb_mgm_get_logfilter(NdbMgmHandle handle);
+
+ /**
+ * Set log category and levels for the cluster log
+ *
+ * @param handle NDB management handle.
+ * @param nodeId Node id.
+ * @param category Event category.
+ * @param level Log level (0-15).
+ * @param reply Reply message.
+ * @return -1 on error.
+ */
+ int ndb_mgm_set_loglevel_clusterlog(NdbMgmHandle handle,
+ int nodeId,
+ /*enum ndb_mgm_event_category category*/
+ char * category,
+ int level,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Set log category and levels for the Node
+ *
+ * @param handle NDB management handle.
+ * @param nodeId Node id.
+ * @param category Event category.
+ * @param level Log level (0-15).
+ * @param reply Reply message.
+ * @return -1 on error.
+ */
+ int ndb_mgm_set_loglevel_node(NdbMgmHandle handle,
+ int nodeId,
+ /*enum ndb_mgm_event_category category*/
+ char * category,
+ int level,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Returns the port number where statistics information is sent
+ *
+ * @param handle NDB management handle.
+ * @param reply Reply message.
+ * @return -1 on error.
+ */
+ int ndb_mgm_get_stat_port(NdbMgmHandle handle,
+ struct ndb_mgm_reply* reply);
+
+ /** @} *********************************************************************/
+ /**
+ * @name Functions: Backup
+ * @{
+ */
+
+ /**
+ * Start backup
+ *
+ * @param handle NDB management handle.
+ * @param backup_id Backup id is returned from function.
+ * @param reply Reply message.
+ * @return -1 on error.
+ */
+ int ndb_mgm_start_backup(NdbMgmHandle handle, unsigned int* backup_id,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Abort backup
+ *
+ * @param handle NDB management handle.
+ * @param backup_id Backup Id.
+ * @param reply Reply message.
+ * @return -1 on error.
+ */
+ int ndb_mgm_abort_backup(NdbMgmHandle handle, unsigned int backup_id,
+ struct ndb_mgm_reply* reply);
+
+
+ /** @} *********************************************************************/
+ /**
+ * @name Functions: Single User Mode
+ * @{
+ */
+
+ /**
+ * Enter Single user mode
+ *
+ * @param handle NDB management handle.
+ * @param nodeId Node Id of the single user node
+ * @param reply Reply message.
+ * @return -1 on error.
+ */
+ int ndb_mgm_enter_single_user(NdbMgmHandle handle, unsigned int nodeId,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Exit Single user mode
+ *
+ * @param handle NDB management handle.
+ * @param nodeId Node Id of the single user node
+ * @param reply Reply message.
+ * @return -1 on error.
+ */
+ int ndb_mgm_exit_single_user(NdbMgmHandle handle,
+ struct ndb_mgm_reply* reply);
+
+#ifdef __cplusplus
+}
+#endif
+
+/** @} */
+
+#endif
diff --git a/ndb/include/mgmapi/mgmapi_debug.h b/ndb/include/mgmapi/mgmapi_debug.h
new file mode 100644
index 00000000000..2723263e7a7
--- /dev/null
+++ b/ndb/include/mgmapi/mgmapi_debug.h
@@ -0,0 +1,114 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MGMAPI_DEBUG_H
+#define MGMAPI_DEBUG_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+ /**
+ * Start signal logging.
+ *
+ * @param handle the NDB management handle.
+ * @param nodeId the node Id.
+ * @param reply the reply message.
+ * @return 0 if successful.
+ */
+ int ndb_mgm_start_signallog(NdbMgmHandle handle,
+ int nodeId,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Stop signal logging.
+ *
+ * @param handle the NDB management handle.
+ * @param nodeId the node Id.
+ * @param reply the reply message.
+ * @return 0 if successful.
+ */
+ int ndb_mgm_stop_signallog(NdbMgmHandle handle,
+ int nodeId,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Set the signals to log.
+ *
+ * @param handle the NDB management handle.
+ * @param nodeId the node id.
+ * @param mode the signal log mode.
+ * @param blockNames the block names (space separated).
+ * @param reply the reply message.
+ * @return 0 if successful or an error code.
+ */
+ int ndb_mgm_log_signals(NdbMgmHandle handle,
+ int nodeId,
+ enum ndb_mgm_signal_log_mode mode,
+ const char* blockNames,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Set trace.
+ *
+ * @param handle the NDB management handle.
+ * @param nodeId the node id.
+ * @param traceNumber the trace number.
+ * @param reply the reply message.
+ * @return 0 if successful or an error code.
+ */
+ int ndb_mgm_set_trace(NdbMgmHandle handle,
+ int nodeId,
+ int traceNumber,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Provoke an error.
+ *
+ * @param handle the NDB management handle.
+ * @param nodeId the node id.
+ * @param errrorCode the errorCode.
+ * @param reply the reply message.
+ * @return 0 if successful or an error code.
+ */
+ int ndb_mgm_insert_error(NdbMgmHandle handle,
+ int nodeId,
+ int errorCode,
+ struct ndb_mgm_reply* reply);
+
+ /**
+ * Dump state
+ *
+ * @param handle the NDB management handle.
+ * @param nodeId the node id.
+ * @param args integer array
+ * @param number of args in int array
+ * @param reply the reply message.
+ * @return 0 if successful or an error code.
+ */
+ int ndb_mgm_dump_state(NdbMgmHandle handle,
+ int nodeId,
+ int * args,
+ int num_args,
+ struct ndb_mgm_reply* reply);
+
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif
diff --git a/ndb/include/mgmcommon/ConfigRetriever.hpp b/ndb/include/mgmcommon/ConfigRetriever.hpp
new file mode 100644
index 00000000000..ff60e730d45
--- /dev/null
+++ b/ndb/include/mgmcommon/ConfigRetriever.hpp
@@ -0,0 +1,116 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ConfigRetriever_H
+#define ConfigRetriever_H
+
+#include <ndb_types.h>
+#include <Properties.hpp>
+
+/**
+ * @class ConfigRetriever
+ * @brief Used by nodes (DB, MGM, API) to get their config from MGM server.
+ */
+class ConfigRetriever {
+public:
+ ConfigRetriever();
+ ConfigRetriever(const int id, const char* remoteHost, const int port);
+ ~ConfigRetriever();
+
+ /**
+ * Read local config
+ * @return Own node id, -1 means fail
+ */
+ int init(bool onlyNodeId = false);
+
+ /**
+ * Get configuration for current (nodeId given in local config file) node.
+ *
+ * Configuration is fetched from one MGM server configured in local config
+ * file. The method loops over all the configured MGM servers and tries
+ * to establish a connection. This is repeated until a connection is
+ * established, so the function hangs until a connection is established.
+ *
+ * @return Properties object if succeeded,
+ * NULL if erroneous local config file or configuration error.
+ */
+ class Properties * getConfig(const char * nodeType, int versionId);
+
+ const char * getErrorString();
+
+ /**
+ * Sets connectstring which can be used instead of local config file
+ */
+ void setConnectString(const char * connectString);
+
+ /**
+ * Sets name of local config file (usually not needed)
+ */
+ void setLocalConfigFileName(const char * connectString);
+
+ /**
+ * Sets connectstring which can be used instead of local config file
+ * environment variables and Ndb.cfg has precidence over this
+ */
+ void setDefaultConnectString(const char * defaultConnectString);
+
+ /**
+ * @return Node id of this node (as stated in local config or connectString)
+ */
+ inline Uint32 getOwnNodeId() { return _ownNodeId; }
+
+ /**
+ * Get configuration object
+ */
+ class Properties * getConfig(int versionId);
+
+ /**
+ * Get config using socket
+ */
+ class Properties * getConfig(const char * mgmhost, unsigned int port,
+ Uint32 nodeId, int versionId);
+ /**
+ * Get config from file
+ */
+ class Properties * getConfig(const char * filename, Uint32 nodeId,
+ int versionId);
+private:
+ char * errorString;
+ enum ErrorType {
+ CR_ERROR = 0,
+ CR_RETRY = 1
+ };
+ ErrorType latestErrorType;
+
+ void setError(ErrorType, const char * errorMsg);
+
+ /**
+ * Verifies that received configuration is correct
+ */
+ bool verifyProperties(const char* nodeType, Properties *p,
+ Uint32 nodeId, int versionId);
+
+ char * _localConfigFileName;
+ struct LocalConfig * _localConfig;
+ int _ownNodeId;
+
+ char * m_connectString;
+ char * m_defaultConnectString;
+};
+
+#endif
+
+
diff --git a/ndb/include/mgmcommon/IPCConfig.hpp b/ndb/include/mgmcommon/IPCConfig.hpp
new file mode 100644
index 00000000000..626242865cb
--- /dev/null
+++ b/ndb/include/mgmcommon/IPCConfig.hpp
@@ -0,0 +1,79 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef IPCConfig_H
+#define IPCConfig_H
+
+#include <ndb_types.h>
+#include <ndb_limits.h>
+#include <kernel_types.h>
+#include <Properties.hpp>
+
+/**
+ * @class IPCConfig
+ * @brief Config transporters in TransporterRegistry using Properties config
+ */
+class IPCConfig
+{
+public:
+ IPCConfig(Properties * props);
+ ~IPCConfig();
+
+ /** @return 0 for OK */
+ int init();
+
+ NodeId ownId() const;
+
+ /** @return No of transporters configured */
+ int configureTransporters(class TransporterRegistry * theTransporterRegistry);
+
+ /**
+ * Supply a nodeId,
+ * and get next higher node id
+ * @return false if none found, true otherwise
+ *
+ * getREPHBFrequency and getNodeType uses the last Id supplied to
+ * getNextRemoteNodeId.
+ */
+ bool getNextRemoteNodeId(NodeId & nodeId) const;
+ Uint32 getREPHBFrequency(NodeId id) const;
+ const char* getNodeType(NodeId id) const;
+
+ NodeId getNoOfRemoteNodes() const {
+ return theNoOfRemoteNodes;
+ }
+
+ void print() const { props->print(); }
+
+private:
+ NodeId the_ownId;
+ Properties * props;
+
+ bool addRemoteNodeId(NodeId nodeId);
+ NodeId theNoOfRemoteNodes;
+ NodeId theRemoteNodeIds[MAX_NODES];
+};
+
+inline
+NodeId
+IPCConfig::ownId() const
+{
+ return the_ownId;
+}
+
+
+
+#endif // IPCConfig_H
diff --git a/ndb/include/mgmcommon/MgmtErrorReporter.hpp b/ndb/include/mgmcommon/MgmtErrorReporter.hpp
new file mode 100644
index 00000000000..acc44b14d8e
--- /dev/null
+++ b/ndb/include/mgmcommon/MgmtErrorReporter.hpp
@@ -0,0 +1,74 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//******************************************************************************
+// Description: This file contains the error reporting macros to be used
+// within management server.
+//
+// Author: Peter Lind
+//******************************************************************************
+
+
+#include <NdbOut.hpp>
+#include <stdlib.h> // exit
+
+#define REPORT_WARNING(message) \
+ ndbout << "WARNING: " << message << endl
+
+//****************************************************************************
+// Description: Report a warning, the message is printed on ndbout.
+// Parameters:
+// message: A text describing the warning.
+// Returns: -
+//****************************************************************************
+
+
+#define REPORT_ERROR(message) \
+ ndbout << "ERROR: " << message << endl
+
+//****************************************************************************
+// Description: Report an error, the message is printed on ndbout.
+// Parameters:
+// message: A text describing the error.
+// Returns: -
+//****************************************************************************
+
+
+#ifdef MGMT_TRACE
+
+#define TRACE(message) \
+ ndbout << "MGMT_TRACE: " << message << endl
+#else
+#define TRACE(message)
+
+#endif
+
+//****************************************************************************
+// Description: Print a message on ndbout.
+// Parameters:
+// message: The message
+// Returns: -
+//****************************************************************************
+
+#ifndef NDB_ASSERT
+#define NDB_ASSERT(trueToContinue, message) \
+ if ( !(trueToContinue) ) { \
+ndbout << "ASSERT FAILED. FILE: " << __FILE__ << ", LINE: " << __LINE__ << ", MSG: " << message << endl;exit(-1);}
+#endif
+
+#define MGM_REQUIRE(x) \
+ if (!(x)) { ndbout << __FILE__ << " " << __LINE__ \
+ << ": Warning! Requirement failed" << endl; }
diff --git a/ndb/include/mgmcommon/NdbConfig.h b/ndb/include/mgmcommon/NdbConfig.h
new file mode 100644
index 00000000000..d9b484edcc5
--- /dev/null
+++ b/ndb/include/mgmcommon/NdbConfig.h
@@ -0,0 +1,34 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_CONFIG_H
+#define NDB_CONFIG_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+const char* NdbConfig_HomePath(char* buf, int buflen);
+
+const char* NdbConfig_NdbCfgName(char* buf, int buflen, int with_ndb_home);
+const char* NdbConfig_ErrorFileName(char* buf, int buflen);
+const char* NdbConfig_ClusterLogFileName(char* buf, int buflen);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/ndb/include/ndb_types.h b/ndb/include/ndb_types.h
new file mode 100644
index 00000000000..40d73b0f230
--- /dev/null
+++ b/ndb/include/ndb_types.h
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * @file ndb_types.h
+ */
+
+#ifndef SYS_TYPES_H
+#define SYS_TYPES_H
+
+#include <sys/types.h>
+#include <stddef.h>
+
+typedef char Int8;
+typedef unsigned char Uint8;
+typedef short Int16;
+typedef unsigned short Uint16;
+typedef int Int32;
+typedef unsigned int Uint32;
+
+typedef unsigned int UintR;
+
+#ifdef __SIZE_TYPE__
+typedef __SIZE_TYPE__ UintPtr;
+#else
+#include <stdint.h>
+typedef uintptr_t UintPtr;
+#endif
+
+#if defined(WIN32) || defined(NDB_WIN32)
+typedef unsigned __int64 Uint64;
+typedef __int64 Int64;
+#else
+typedef unsigned long long Uint64;
+typedef long long Int64;
+#endif
+
+#endif
diff --git a/ndb/include/ndb_version.h b/ndb/include/ndb_version.h
new file mode 100644
index 00000000000..4fb6ec18fce
--- /dev/null
+++ b/ndb/include/ndb_version.h
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_VERSION_H
+#define NDB_VERSION_H
+
+#include <stdio.h>
+#include <string.h>
+
+#include <version.h>
+
+#define MAKE_VERSION(A,B,C) (((A) << 16) | ((B) << 8) | ((C) << 0))
+
+/**
+ * version of this build
+ */
+
+#define NDB_VERSION_MAJOR 3
+#define NDB_VERSION_MINOR 4
+#define NDB_VERSION_BUILD 5
+#define NDB_VERSION_STATUS "alpha"
+
+#define NDB_VERSION_D MAKE_VERSION(NDB_VERSION_MAJOR, NDB_VERSION_MINOR, NDB_VERSION_BUILD)
+
+#define NDB_VERSION_STRING (getVersionString(NDB_VERSION, NDB_VERSION_STATUS))
+
+#define NDB_VERSION_TAG_STRING "$Name: $"
+
+#define NDB_VERSION ndbGetOwnVersion()
+
+/**
+ * Version id
+ *
+ * Used by transporter and when communicating with
+ * managment server
+ */
+//#define NDB_VERSION_ID 0
+
+#endif
+
diff --git a/ndb/include/ndbapi/AttrType.hpp b/ndb/include/ndbapi/AttrType.hpp
new file mode 100644
index 00000000000..e6e00c77130
--- /dev/null
+++ b/ndb/include/ndbapi/AttrType.hpp
@@ -0,0 +1,329 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * @file AttrType.hpp
+ */
+
+#ifndef AttrType_H
+#define AttrType_H
+
+/**
+ * Max number of Ndb objects in different threads.
+ * (Ndb objects should not be shared by different threads.)
+ */
+const unsigned MAX_NO_THREADS = 4711;
+
+/**
+ * Max number of attributes in a table.
+ */
+const unsigned MAXNROFATTRIBUTES = 128;
+
+/**
+ * Max number of tuple keys for a table in NDB Cluster.
+ *
+ * A <em>tuple key</em> of a table is an attribute
+ * which is either part of the
+ * <em>primary key</em> or the <em>tuple id</em> of a table.
+ */
+const unsigned MAXNROFTUPLEKEY = 16;
+
+/**
+ * Max number of words in a tuple key attribute.
+ *
+ * Tuple keys can not have values larger than
+ * 4092 bytes (i.e. 1023 words).
+ */
+const unsigned MAXTUPLEKEYLENOFATTERIBUTEINWORD = 1023;
+
+/**
+ * Max number of ErrorCode in NDB Cluster range 0 - 1999.
+ */
+const unsigned MAXNDBCLUSTERERROR = 1999;
+
+/**
+ * Max number of theErrorCode NDB API range 4000 - 4999.
+ */
+const unsigned MAXNROFERRORCODE = 5000;
+
+/**
+ * <i>Missing explanation</i>
+ */
+enum ReturnType {
+ ReturnSuccess, ///< <i>Missing explanation</i>
+ ReturnFailure ///< <i>Missing explanation</i>
+};
+
+/**
+ *
+ */
+enum SendStatusType {
+ NotInit, ///< <i>Missing explanation</i>
+ InitState, ///< <i>Missing explanation</i>
+ sendOperations, ///< <i>Missing explanation</i>
+ sendCompleted, ///< <i>Missing explanation</i>
+ sendCOMMITstate, ///< <i>Missing explanation</i>
+ sendABORT, ///< <i>Missing explanation</i>
+ sendABORTfail, ///< <i>Missing explanation</i>
+ sendTC_ROLLBACK, ///< <i>Missing explanation</i>
+ sendTC_COMMIT, ///< <i>Missing explanation</i>
+ sendTC_OP ///< <i>Missing explanation</i>
+};
+
+/**
+ * <i>Missing explanation</i>
+ */
+enum ListState {
+ NotInList, ///< <i>Missing explanation</i>
+ InPreparedList, ///< <i>Missing explanation</i>
+ InSendList, ///< <i>Missing explanation</i>
+ InCompletedList ///< <i>Missing explanation</i>
+};
+
+/**
+ * Commit status of the transaction
+ */
+enum CommitStatusType {
+ NotStarted, ///< Transaction not yet started
+ Started, ///< <i>Missing explanation</i>
+ Committed, ///< Transaction has been committed
+ Aborted, ///< Transaction has been aborted
+ NeedAbort ///< <i>Missing explanation</i>
+};
+
+/**
+ * Commit type of transaction
+ */
+enum AbortOption {
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ CommitIfFailFree = 0,
+ CommitAsMuchAsPossible = 2, ///< Commit transaction with as many
+ TryCommit = 0, ///< <i>Missing explanation</i>
+#endif
+ AbortOnError = 0, ///< Abort transaction on failed operation
+ IgnoreError = 2 ///< Transaction continues on failed operation
+};
+
+typedef AbortOption CommitType;
+
+/**
+ * <i>Missing explanation</i>
+ */
+enum InitType {
+ NotConstructed, ///< <i>Missing explanation</i>
+ NotInitialised, ///< <i>Missing explanation</i>
+ StartingInit, ///< <i>Missing explanation</i>
+ Initialised, ///< <i>Missing explanation</i>
+ InitConfigError ///< <i>Missing explanation</i>
+};
+
+/**
+ * Type of attribute
+ */
+enum AttrType {
+ Signed, ///< Attributes of this type can be read with:
+ ///< NdbRecAttr::int64_value,
+ ///< NdbRecAttr::int32_value,
+ ///< NdbRecAttr::short_value,
+ ///< NdbRecAttr::char_value
+ UnSigned, ///< Attributes of this type can be read with:
+ ///< NdbRecAttr::u_64_value,
+ ///< NdbRecAttr::u_32_value,
+ ///< NdbRecAttr::u_short_value,
+ ///< NdbRecAttr::u_char_value
+ Float, ///< Attributes of this type can be read with:
+ ///< NdbRecAttr::float_value and
+ ///< NdbRecAttr::double_value
+ String, ///< Attributes of this type can be read with:
+ ///< NdbRecAttr::aRef,
+ ///< NdbRecAttr::getAttributeObject
+ NoAttrTypeDef ///< Used for debugging only
+};
+
+/**
+ * Execution type of transaction
+ */
+enum ExecType {
+ NoExecTypeDef = -1, ///< Erroneous type (Used for debugging only)
+ Prepare, ///< <i>Missing explanation</i>
+ NoCommit, ///< Execute the transaction as far as it has
+ ///< been defined, but do not yet commit it
+ Commit, ///< Execute and try to commit the transaction
+ Rollback ///< Rollback transaction
+};
+
+/**
+ * Indicates whether the attribute is part of a primary key or not
+ */
+enum KeyType {
+ Undefined = -1, ///< Used for debugging only
+ NoKey, ///< Attribute is not part of primary key
+ ///< or tuple identity
+ TupleKey, ///< Attribute is part of primary key
+ TupleId ///< Attribute is part of tuple identity
+ ///< (This type of attribute is created
+ ///< internally, and should not be
+ ///< manually created.)
+};
+
+/**
+ * Indicate whether the attribute should be stored on disk or not
+ */
+enum StorageMode {
+ MMBased = 0, ///< Main memory
+ DiskBased = 1, ///< Disk (Not yet supported.)
+ NoStorageTypeDef ///< Used for debugging only
+};
+
+/**
+ * Where attribute is stored.
+ *
+ * This is used to indicate whether a primary key
+ * should only be stored in the index storage and not in the data storage
+ * or if it should be stored in both places.
+ * The first alternative makes the attribute take less space,
+ * but makes it impossible to scan using attribute.
+ *
+ * @note Use NormalStorageAttribute for most cases.
+ * (IndexStorageAttribute should only be used on primary key
+ * attributes and only if you do not want to scan using the attribute.)
+ */
+enum StorageAttributeType {
+ NoStorageAttributeTypeDefined = -1, ///< <i>Missing explanation</i>
+ IndexStorageAttribute, ///< Attribute is only stored in
+ ///< index storage (ACC)
+ NormalStorageAttribute ///< Attribute values are stored
+ ///< both in the index (ACC) and
+ ///< in the data storage (TUP)
+};
+
+/**
+ * <i>Missing explanation</i>
+ */
+enum OperationStatus{
+ Init, ///< <i>Missing explanation</i>
+ OperationDefined, ///< <i>Missing explanation</i>
+ TupleKeyDefined, ///< <i>Missing explanation</i>
+ GetValue, ///< <i>Missing explanation</i>
+ SetValue, ///< <i>Missing explanation</i>
+ ExecInterpretedValue, ///< <i>Missing explanation</i>
+ SetValueInterpreted, ///< <i>Missing explanation</i>
+ FinalGetValue, ///< <i>Missing explanation</i>
+ SubroutineExec, ///< <i>Missing explanation</i>
+ SubroutineEnd, ///< <i>Missing explanation</i>
+ SetBound, ///< Setting bounds in range scan
+ WaitResponse, ///< <i>Missing explanation</i>
+ WaitCommitResponse, ///< <i>Missing explanation</i>
+ Finished, ///< <i>Missing explanation</i>
+ ReceiveFinished ///< <i>Missing explanation</i>
+};
+
+/**
+ * Type of operation
+ */
+enum OperationType {
+ ReadRequest = 0, ///< Read operation
+ UpdateRequest = 1, ///< Update Operation
+ InsertRequest = 2, ///< Insert Operation
+ DeleteRequest = 3, ///< Delete Operation
+ WriteRequest = 4, ///< Write Operation
+ ReadExclusive = 5, ///< Read exclusive
+ OpenScanRequest, ///< Scan Operation
+ OpenRangeScanRequest, ///< Range scan operation
+ NotDefined2, ///< <i>Missing explanation</i>
+ NotDefined ///< <i>Missing explanation</i>
+};
+
+/**
+ * <i>Missing explanation</i>
+ */
+enum ConStatusType {
+ NotConnected, ///< <i>Missing explanation</i>
+ Connecting, ///< <i>Missing explanation</i>
+ Connected, ///< <i>Missing explanation</i>
+ DisConnecting, ///< <i>Missing explanation</i>
+ ConnectFailure ///< <i>Missing explanation</i>
+};
+
+/**
+ * <i>Missing explanation</i>
+ */
+enum CompletionStatus {
+ NotCompleted, ///< <i>Missing explanation</i>
+ CompletedSuccess, ///< <i>Missing explanation</i>
+ CompletedFailure, ///< <i>Missing explanation</i>
+ DefinitionFailure ///< <i>Missing explanation</i>
+};
+
+/**
+ * Type of fragmentation used for a table
+ */
+enum FragmentType {
+ Default = 0, ///< (All is default!)
+ Single = 1, ///< Only one fragment
+ All = 2, ///< Default value. One fragment per node group
+ DistributionGroup = 3, ///< Distribution Group used for fragmentation.
+ ///< One fragment per node group
+ DistributionKey = 4, ///< Distribution Key used for fragmentation.
+ ///< One fragment per node group.
+ AllLarge = 5, ///< Sixten fragments per node group.
+ DGroupLarge = 6, ///< Distribution Group used for fragmentation.
+ ///< Sixten fragments per node group
+ DKeyLarge = 7 ///< Distribution Key used for fragmentation.
+ ///< Sixten fragments per node group
+};
+
+/**
+ * Type of table or index.
+ */
+enum TableType {
+ UndefTableType = 0,
+ SystemTable = 1, ///< Internal. Table cannot be updated by user
+ UserTable = 2, ///< Normal application table
+ UniqueHashIndex = 3, ///< Unique un-ordered hash index
+ HashIndex = 4, ///< Non-unique un-ordered hash index
+ UniqueOrderedIndex = 5, ///< Unique ordered index
+ OrderedIndex = 6 ///< Non-unique ordered index
+};
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+/**
+ * Different types of tampering with the NDB Cluster.
+ * <b>Only for debugging purposes only.</b>
+ */
+enum TamperType {
+ LockGlbChp = 1, ///< Lock GCP
+ UnlockGlbChp, ///< Unlock GCP
+ CrashNode, ///< Crash an NDB node
+ ReadRestartGCI, ///< Request the restart GCI id from NDB Cluster
+ InsertError ///< Execute an error in NDB Cluster
+ ///< (may crash system)
+};
+#endif
+
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+/**
+ * @deprecated
+ */
+enum NullAttributeType {
+ NoNullTypeDefined = -1,
+ NotNullAttribute,
+ NullAttribute,
+ AttributeDefined
+};
+#endif
+
+#endif
diff --git a/ndb/include/ndbapi/Ndb.hpp b/ndb/include/ndbapi/Ndb.hpp
new file mode 100644
index 00000000000..59bdd212919
--- /dev/null
+++ b/ndb/include/ndbapi/Ndb.hpp
@@ -0,0 +1,1702 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ @mainpage NDB API Programmers' Guide
+
+ This guide assumes a basic familiarity with NDB Cluster concepts.
+ Some of the fundamental ones are described in section @ref secConcepts.
+
+ The <em>NDB API</em> is an NDB Cluster application interface
+ that implements both synchronous and asynchronous transactions.
+ The NDB API consists of the following fundamental classes:
+ - Ndb is the main class representing the database,
+ - NdbConnection represents a transaction,
+ - NdbOperation represents a transaction operation using primary key,
+ - NdbIndexOperation represents a transaction operation using a secondary
+ index,
+ - NdbRecAttr represents the value of an attribute, and
+ - NdbDictionary represents meta information about tables and attributes.
+ - NdbError represents an error condition
+ There are also some auxiliary classes.
+
+ The main structure of an application program is as follows:
+ -# Construct and initialize Ndb object(s).
+ -# Define and execute (synchronous or asynchronous) transactions.
+ -# Delete Ndb objects
+
+ The main structure of a transaction is as follows:
+ -# Start transaction
+ -# Add and define operations (associated with the transaction)
+ -# Execute transaction
+
+ The execute can be of two different types,
+ <em>Commit</em> or <em>NoCommit</em>.
+ (The execute can also be divided into three
+ steps: prepare, send, and poll to get asynchronous
+ transactions. More about this later.)
+
+ If the execute is of type NoCommit,
+ then the application program executes part of a transaction,
+ but without committing the transaction.
+ After a NoCommit type of execute, the program can continue
+ to add and define more operations to the transaction
+ for later execution.
+
+ If the execute is of type Commit, then the transaction is
+ committed and no further adding and defining of operations
+ is allowed.
+
+
+ @section secSync Synchronous Transactions
+
+ Synchronous transactions are defined and executed in the following way.
+
+ -# Start (create) transaction (the transaction will be
+ referred to by an NdbConnection object,
+ typically created by Ndb::startTransaction).
+ At this step the transaction is being defined.
+ It is not yet sent to the NDB kernel.
+ -# Add and define operations to the transaction
+ (using NdbConnection::getNdbOperation and
+ methods from class NdbOperation).
+ The transaction is still not sent to the NDB kernel.
+ -# Execute the transaction (using NdbConnection::execute).
+ -# Close the transaction (using Ndb::closeTransaction).
+
+ See example program in section @ref ndbapi_example1.cpp.
+
+ To execute several parallel synchronous transactions, one can either
+ use multiple Ndb objects in several threads or start multiple
+ applications programs.
+ Another way to execute several parallel transactions is to use
+ asynchronous transactions.
+
+
+ @section secNdbOperations Operations
+
+ Each transaction (NdbConnection object) consist of a list of
+ operations (NdbOperation or NdbIndexOperation objects.
+ NdbIndexOperation is used for accessing tables through secondary indexes).
+ Operations are of two different kinds:
+ -# standard operations, and
+ -# interpreted program operations.
+
+ <h3>Standard Operations</h3>
+ After the operation is created using NdbConnection::getNdbOperation
+ (or NdbConnection::getNdbIndexOperation),
+ it is defined in the following three steps:
+ -# Defining standard operation type
+ (e.g. using NdbOperation::readTuple)
+ -# Specifying search conditions
+ (e.g. using NdbOperation::equal)
+ -# Specify attribute actions
+ (e.g. using NdbOperation::getValue)
+
+ Example code (using an NdbOperation):
+ @code
+ MyOperation = MyConnection->getNdbOperation("MYTABLENAME"); // 1. Create
+ if (MyOperation == NULL) APIERROR(MyConnection->getNdbError());
+
+ MyOperation->readTuple(); // 2. Define type of operation
+ MyOperation->equal("ATTR1", i); // 3. Specify Search Conditions
+
+ MyRecAttr = MyOperation->getValue("ATTR2", NULL); // 4. Attribute Actions
+ if (MyRecAttr == NULL) APIERROR(MyConnection->getNdbError());
+ @endcode
+ For more examples, see @ref ndbapi_example1.cpp and @ref ndbapi_example2.cpp.
+
+ Example code using an NdbIndexOperation:
+ @code
+ MyOperation = // 1. Create
+ MyConnection->getNdbIndexOperation("MYINDEX", "MYTABLENAME");
+ if (MyOperation == NULL) APIERROR(MyConnection->getNdbError());
+
+ MyOperation->readTuple(); // 2. Define type of operation
+ MyOperation->equal("ATTR1", i); // 3. Specify Search Conditions
+
+ MyRecAttr = MyOperation->getValue("ATTR2", NULL); // 4. Attribute Actions
+ if (MyRecAttr == NULL) APIERROR(MyConnection->getNdbError());
+ @endcode
+ For more examples, see @ref ndbapi_example4.cpp.
+
+
+ <h4>Step 1: Define Standard Operation Type</h4>
+ The following types of standard operations exist:
+ -# NdbOperation::insertTuple :
+ inserts a non-existing tuple
+ -# NdbOperation::writeTuple :
+ updates an existing tuple if is exists,
+ otherwise inserts a new tuple
+ -# NdbOperation::updateTuple :
+ updates an existing tuple
+ -# NdbOperation::deleteTuple :
+ deletes an existing tuple
+ -# NdbOperation::readTuple :
+ reads an existing tuple
+ -# NdbOperation::readTupleExclusive :
+ reads an existing tuple using an exclusive lock
+ -# NdbOperation::simpleRead :
+ reads an existing tuple (using shared read lock),
+ but releases lock immediately after read
+ -# NdbOperation::committedRead :
+ reads committed tuple
+ -# NdbOperation::dirtyUpdate :
+ updates an existing tuple, but releases lock immediately
+ after read (uses dirty lock)
+ -# NdbOperation::dirtyWrite :
+ updates or writes a tuple, but releases lock immediately
+ after read (uses dirty lock)
+
+ All of these operations operate on the unique tuple key.
+ (When NdbIndexOperation is used then all of these operations
+ operate on a defined secondary index.)
+
+
+ Some comments:
+ - NdbOperation::simpleRead and
+ NdbOperation::committedRead can execute on the same transaction
+ as the above operations but will release its locks immediately
+ after reading the tuple.
+ NdbOperation::simpleRead will always read the latest version
+ of the tuple.
+ Thus it will wait until it can acquire a shared read lock on
+ the tuple.
+ NdbOperation::committedRead will read the latest committed
+ version of the tuple.
+ <br>
+ Both NdbOperation::simpleRead and NdbOperation::committedRead
+ are examples of consistent reads which are not repeatable.
+ All reads read the latest version if updates were made by the same
+ transaction.
+ Errors on simple read are only reported by the NdbOperation object.
+ These error codes are not transferred to the NdbConnection object.
+ - NdbOperation::dirtyUpdate and NdbOperation::dirtyWrite
+ will execute in the same transaction
+ but will release the lock immediately after updating the
+ tuple.
+ It will wait on the lock until it can acquire an exclusive
+ write lock.
+ In a replicated version of NDB Cluster NdbOperation::dirtyUpdate
+ can lead to inconsistency between the replicas.
+ Examples of when it could be used is
+ to update statistical counters on tuples which are "hot-spots".
+
+ @note If you want to define multiple operations within the same transaction,
+ then you need to call NdbConnection::getNdbOperation
+ (or NdbConnection::getNdbIndexOperation) for each
+ operation.
+
+
+ <h4>Step 2: Specify Search Conditions</h4>
+ The search condition is used to select tuples.
+ (In the current NdbIndexOperation implementation
+ this means setting the value of
+ the secondary index attributes of the wanted tuple.)
+
+ If a tuple identity is used, then NdbOperation::setTupleId
+ is used to define the search key when inserting new tuples.
+ Otherwise, NdbOperation::equal is used.
+
+ For NdbOperation::insertTuple it is also allowed to define the
+ search key by using NdbOperation::setValue.
+ The NDB API will automatically detect that it is
+ supposed to use NdbOperation::equal instead.
+ For NdbOperation::insertTuple it is not necessary to use
+ NdbOperation::setValue on key attributes before other attributes.
+
+
+ <h4>Step 3: Specify Attribute Actions</h4>
+ Now it is time to define which attributes should be read or updated.
+ Deletes can neither read nor set values, read can only read values and
+ updates can only set values.
+ Normally the attribute is defined by its name but it is
+ also possible to use the attribute identity to define the
+ attribute.
+ The mapping from name to identity is performed by the Table object.
+
+ NdbIndexOperation::getValue returns an NdbRecAttr object
+ containing the read value.
+ To get the value, there is actually two methods.
+ The application can either
+ - use its own memory (passed through a pointer aValue) to
+ NdbIndexOperation::getValue, or
+ - receive the attribute value in an NdbRecAttr object allocated
+ by the NDB API.
+
+ The NdbRecAttr object is released when Ndb::closeTransaction
+ is called.
+ Thus, the application can not reference this object after
+ Ndb::closeTransaction have been called.
+ The result of reading data from an NdbRecAttr object before
+ calling NdbConnection::execute is undefined.
+
+
+
+ <h3>Interpreted Program Operations</h3>
+ The following types of interpreted program operations exist:
+ -# NdbOperation::interpretedUpdateTuple :
+ updates a tuple using an interpreted program
+ -# NdbOperation::interpretedDeleteTuple :
+ delete a tuple using an interpreted program
+ -# NdbOperation::openScanRead :
+ scans a table with read lock on each tuple
+ -# NdbOperation::openScanExclusive :
+ scans a table with exclusive update lock on each tuple
+
+ The operations interpretedUpdateTuple and interpretedDeleteTuple both
+ work using the unique tuple key.
+
+ These <em>interpreted programs</em>
+ make it possible to perform computations
+ inside the NDB Cluster Kernel instead of in the application
+ program.
+ This is sometimes very effective, since no intermediate results
+ are sent to the application, only the final result.
+
+
+ <h3>Interpreted Update and Delete</h3>
+
+ Operations for interpreted updates and deletes must follow a
+ certain order when defining operations on a tuple.
+ As for read and write operations,
+ one must first define the operation type and then the search key.
+ -# The first step is to define the initial readings.
+ In this phase it is only allowed to use the
+ NdbOperation::getValue method.
+ This part might be empty.
+ -# The second step is to define the interpreted part.
+ The methods supported are the methods listed below except
+ NdbOperation::def_subroutine and NdbOperation::ret_sub
+ which can only be used in a subroutine.
+ NdbOperation::incValue and NdbOperation::subValue
+ increment and decrement attributes
+ (currently only unsigned integers supported).
+ This part can also be empty since interpreted updates
+ can be used for reading and updating the same tuple.
+ <p>
+ Even though getValue and setValue are not really interpreted
+ program instructions, it is still allowed to use them as
+ the last instruction of the program.
+ (If a getValue or setValue is found when an interpret_exit_ok
+ could have been issued then the interpreted_exit_ok
+ will be inserted.
+ A interpret_exit_ok should be viewed as a jump to the first
+ instruction after the interpreted instructions.)
+ -# The third step is to define all updates without any
+ interpreted program instructions.
+ Here a set of NdbOperation::setValue methods are called.
+ There might be zero such calls.
+ -# The fourth step is the final readings.
+ The initial readings reads the initial value of attributes
+ and the final readings reads them after their updates.
+ There might be zero NdbOperation::getValue calls.
+ -# The fifth step is possible subroutine definitions using
+ NdbOperation::def_subroutine and NdbOperation::ret_sub.
+
+
+ @subsection secScan Scanning
+ The most common use of interpreted programs is for scanning
+ tables. Scanning is a search of all tuples in a table.
+ Tuples which satisfy conditions (a search filter)
+ stated in the interpreted program
+ are sent to the application.
+
+ Reasons for using scan transactions include
+ need to use a search key different from the primary key
+ and any secondary index.
+ Or that the query needs to access so many tuples so that
+ it is more efficient to scan the entire table.
+
+ Scanning can also be used to update information.
+ The scanning transaction itself is however
+ not allowed to update any tuples.
+ To do updates via scanning transactions, the tuples
+ need to be handed over to another transaction which is
+ executing the actual update.
+
+ Even though a scan operation is part of a transaction,
+ the scan transaction is not a normal transaction.
+ The locks are <em>not</em> kept throughout the entire
+ scan transaction, since this would imply non-optimal performance.
+ <em>
+ A transaction containing a scan operation can only
+ contain that operation.
+ No other operations are allowed in the same transaction.
+ </em>
+
+ The NdbOperation::openScanRead operation
+ only sets a temporary read lock while
+ reading the tuple.
+ The tuple lock is released already when the
+ result of the read reaches the application.
+ The NdbOperation::openScanExclusive operation sets an
+ exclusive lock on the tuple
+ and sends the result to the application.
+ Thus when the application reads the data it is still
+ locked with the exclusive lock.
+
+ If the application desires to update the tuple it may transfer
+ the tuple to another transaction which updates the tuple.
+ The updating transaction can consist of a combination of tuples
+ received from the scan and normal operations.
+
+ For transferred operations it is not necessary to provide the
+ primary key. It is part of the transfer.
+ You only need to give the operation type and the
+ actions to perform on the tuple.
+
+ The scan transaction starts like a usual transaction,
+ but is of the following form:
+ -# Start transaction
+ -# Get NdbOperation for the table to be scanned
+ -# Set the operation type using NdbOperation::openScanRead or
+ NdbOperation::openScanExclusive
+ -# Search conditions are defined by an interpreted program
+ (setValue and write_attr are not allowed, since scan transactions
+ are only allowed to read information).
+ The instruction interpret_exit_nok does in this case
+ not abort the transaction, it only skips the tuple and
+ proceeds with the next.
+ The skipped tuple will not be reported to the application.
+ -# Call NdbConnection::executeScan to define (and start) the scan.
+ -# Call NdbConnection::nextScanResult to proceed with next tuple.
+ When calling NdbConnection::nextScanResult, the lock on any
+ previous tuples are released.
+ <br>
+ If the tuple should be updated then it must be transferred over
+ to another updating transaction.
+ This is performed by calling
+ NdbOperation::takeOverForUpdate or takeOverForDelete on
+ the scanning transactions NdbOperation object with the updating
+ transactions NdbConnection object as parameter.
+ <p>
+ If NdbOperation::takeOverFor* returns NULL then the
+ operation was not successful, otherwise it returns a reference
+ to the NdbOperation which the updating transaction has received
+ -# Use Ndb::closeTransaction as usual to close the transaction.
+ This can be performed even if there are more tuples to scan.
+
+ See also example program in section @ref select_all.cpp.
+
+ However, a new scan api is under development, using NdbScanOperation
+ and NdbScanFilter. NdbScanFilter makes it easier to define a search
+ criteria and is recommended instead of using Interpreted Programs.
+
+ The scan transaction starts like a usual transaction,
+ but is of the following form:
+ -# Start transaction
+ -# Get NdbScanOperation for the table to be scanned
+ -# NdbScanOperation::readTuplesExclusive returns a handle to a
+ NdbResultSet.
+ -# Search conditions are defined by NdbScanFilter
+ -# Call NdbConnection::execute(NoCommit) to start the scan.
+ -# Call NdbResultSet::nextResult to proceed with next tuple.
+ When calling NdbResultSet::nextResult(false), the lock on any
+ previous tuples are released and the next tuple cached in the API
+ is fetched.
+ <br>
+ If the tuple should be updated then define a new update operation
+ (NdbOperation) using NdbResultSet::updateTuple().
+ The new update operation can the be used to modify the tuple.
+ When nextResult(false) returns != 0, then no more tuples
+ are cached in the API. Updated tuples is now commit using
+ NdbConnection::execute(Commit).
+ After the commit, more tuples are fetched from NDB using
+ nextResult(true).
+ -# Use Ndb::closeTransaction as usual to close the transaction.
+ This can be performed even if there are more tuples to scan.
+
+ See the scan example program in @ref ndbapi_scan.cppn for example
+ usage of the new scan api.
+
+
+ <h3>Interpreted Programs</h3>
+ Interpretation programs are executed in a
+ register-based virtual machine.
+ The virtual machine has eight 64 bit registers numbered 0-7.
+ Each register contains type information which is used both
+ for type conversion and for type checking.
+
+ @note Arrays are currently <b>not</b> supported in the virtual machine.
+ Currently only unsigned integers are supported and of size
+ maximum 64 bits.
+
+ All errors in the interpretation program will cause a
+ transaction abort, but will not affect any other transactions.
+
+ The following are legal interpreted program instructions:
+ -# incValue : Add to an attribute
+ -# subValue : Subtract from an attribute
+ -# def_label : Define a label in the interpreted program
+ -# add_reg : Add two registers
+ -# sub_reg : Subtract one register from another
+ -# load_const_u32 : Load an unsigned 32 bit value into a register
+ -# load_const_u64 : Load an unsigned 64 bit value into a register
+ -# load_const_null : Load a NULL value into a register
+ -# read_attr : Read attribute value into a register
+ -# write_attr : Write a register value into an attribute
+ -# branch_ge : Compares registers and possibly jumps to specified label
+ -# branch_gt : Compares registers and possibly jumps to specified label
+ -# branch_le : Compares registers and possibly jumps to specified label
+ -# branch_lt : Compares registers and possibly jumps to specified label
+ -# branch_eq : Compares registers and possibly jumps to specified label
+ -# branch_ne : Compares registers and possibly jumps to specified label
+ -# branch_ne_null : Jumps if register does not contain NULL value
+ -# branch_eq_null : Jumps if register contains NULL value
+ -# branch_label : Unconditional jump to label
+ -# interpret_exit_ok : Exit interpreted program
+ (approving tuple if used in scan)
+ -# interpret_exit_nok : Exit interpreted program
+ (disqualifying tuple if used in scan)
+
+ There are also three instructions for subroutines, which
+ are described in the next section.
+
+ @subsection subsubSub Interpreted Programs: Subroutines
+
+ The following are legal interpreted program instructions for
+ subroutines:
+ -# NdbOperation::def_subroutine :
+ Defines start of subroutine in interpreted program code
+ -# NdbOperation::call_sub :
+ Calls a subroutine
+ -# NdbOperation::ret_sub :
+ Return from subroutine
+
+ The virtual machine executes subroutines using a stack for
+ its operation.
+ The stack allows for up to 24 subroutine calls in succession.
+ Deeper subroutine nesting will cause an abort of the transaction.
+
+ All subroutines starts with the instruction
+ NdbOperation::def_subroutine and ends with the instruction
+ NdbOperation::ret_sub.
+ If it is necessary to return earlier in the subroutine
+ it has to be done using a branch_label instruction
+ to a label defined right before the
+ NdbOperation::ret_sub instruction.
+
+ @note The subroutines are automatically numbered starting with 0.
+ The parameter used by NdbOperation::def_subroutine
+ should match the automatic numbering to make it easier to
+ debug the interpreted program.
+
+
+ @section secAsync Asynchronous Transactions
+ The asynchronous interface is used to increase the speed of
+ transaction executing by better utilizing the connection
+ between the application and the NDB Kernel.
+ The interface is used to send many transactions
+ at the same time to the NDB kernel.
+ This is often much more efficient than using synchronous transactions.
+ The main reason for using this method is to ensure that
+ Sending many transactions at the same time ensures that bigger
+ chunks of data are sent when actually sending and thus decreasing
+ the operating system overhead.
+
+ The synchronous call to NdbConnection::execute
+ normally performs three main steps:<br>
+ -# <b>Prepare</b>
+ Check transaction status
+ - if problems, abort the transaction
+ - if ok, proceed
+ -# <b>Send</b>
+ Send the defined operations since last execute
+ or since start of transaction.
+ -# <b>Poll</b>
+ Wait for response from NDB kernel.
+
+ The asynchronous method NdbConnection::executeAsynchPrepare
+ only perform step 1.
+ (The abort part in step 1 is only prepared for. The actual
+ aborting of the transaction is performed in a later step.)
+
+ Asynchronous transactions are defined and executed
+ in the following way.
+ -# Start (create) transactions (same way as for the
+ synchronous transactions)
+ -# Add and define operations (also as in the synchronous case)
+ -# <b>Prepare</b> transactions
+ (using NdbConnection::executeAsynchPrepare or
+ NdbConnection::executeAsynch)
+ -# <b>Send</b> transactions to NDB Kernel
+ (using Ndb::sendPreparedTransactions,
+ NdbConnection::executeAsynch, or Ndb::sendPollNdb)
+ -# <b>Poll</b> NDB kernel to find completed transactions
+ (using Ndb::pollNdb or Ndb::sendPollNdb)
+ -# Close transactions (same way as for the synchronous transactions)
+
+ See example program in section @ref ndbapi_example2.cpp.
+
+ This prepare-send-poll protocol actually exists in four variants:
+ - (Prepare-Send-Poll). This is the one-step variant provided
+ by synchronous transactions.
+ - (Prepare-Send)-Poll. This is the two-step variant using
+ NdbConnection::executeAsynch and Ndb::pollNdb.
+ - Prepare-(Send-Poll). This is the two-step variant using
+ NdbConnection::executeAsynchPrepare and Ndb::sendPollNdb.
+ - Prepare-Send-Poll. This is the three-step variant using
+ NdbConnection::executeAsynchPrepare, Ndb::sendPreparedTransactions, and
+ Ndb::pollNdb.
+
+ Transactions first has to be prepared by using method
+ NdbConnection::executeAsynchPrepare or NdbConnection::executeAsynch.
+ The difference between these is that
+ NdbConnection::executeAsynch also sends the transaction to
+ the NDB kernel.
+ One of the arguments to these methods is a callback method.
+ The callback method is executed during polling (item 5 above).
+
+ Note that NdbConnection::executeAsynchPrepare does not
+ send the transaction to the NDB kernel. When using
+ NdbConnection::executeAsynchPrepare, you either have to call
+ Ndb::sendPreparedTransactions or Ndb::sendPollNdb to send the
+ database operations.
+ (Ndb::sendPollNdb also polls Ndb for completed transactions.)
+
+ The methods Ndb::pollNdb and Ndb::sendPollNdb checks if any
+ sent transactions are completed. The method Ndb::sendPollNdb
+ also send all prepared transactions before polling NDB.
+ Transactions still in the definition phase (i.e. items 1-3 above,
+ transactions which has not yet been sent to the NDB kernel) are not
+ affected by poll-calls.
+ The poll method invoked (either Ndb::pollNdb or Ndb::sendPollNdb)
+ will return when:
+ -# at least 'minNoOfEventsToWakeup' of the transactions
+ are finished processing, and
+ -# all of these transactions have executed their
+ callback methods.
+
+ The poll method returns the number of transactions that
+ have finished processing and executed their callback methods.
+
+ @note When an asynchronous transaction has been started and sent to
+ the NDB kernel, it is not allowed to execute any methods on
+ objects belonging to this transaction until the transaction
+ callback method have been executed.
+ (The transaction is stated and sent by either
+ NdbConnection::executeAsynch or through the combination of
+ NdbConnection::executeAsynchPrepare and either
+ Ndb::sendPreparedTransactions or Ndb::sendPollNdb).
+
+ More about how transactions are send the NDB Kernel is
+ available in section @ref secAdapt.
+
+
+ @section secError Error Handling
+
+ Errors can occur when
+ -# operations are being defined, or when the
+ -# transaction is being executed.
+
+ One recommended way to handle a transaction failure
+ (i.e. an error is reported) is to:
+ -# Rollback transaction (NdbConnection::execute with a special parameter)
+ -# Close transaction
+ -# Restart transaction (if the error was temporary)
+
+ @note Transaction are not automatically closed when an error occur.
+
+ Several errors can occur when a transaction holds multiple
+ operations which are simultaneously executed.
+ In this case the application has to go through the operation
+ objects and query for their NdbError objects to find out what really
+ happened.
+
+ NdbConnection::getNdbErrorOperation returns a reference to the
+ operation causing the latest error.
+ NdbConnection::getNdbErrorLine delivers the method number of the
+ erroneous method in the operation.
+
+ @code
+ theConnection = theNdb->startTransaction();
+ theOperation = theConnection->getNdbOperation("TEST_TABLE");
+ if (theOperation == NULL) goto error;
+ theOperation->readTuple();
+ theOperation->setValue("ATTR_1", at1);
+ theOperation->setValue("ATTR_2", at1); //Here an error occurs
+ theOperation->setValue("ATTR_3", at1);
+ theOperation->setValue("ATTR_4", at1);
+
+ if (theConnection->execute(Commit) == -1) {
+ errorLine = theConnection->getNdbErrorLine();
+ errorOperation = theConnection->getNdbErrorOperation();
+ @endcode
+
+ Here errorLine will be 3 as the error occurred in the third method
+ on the operation object.
+ Getting errorLine == 0 means that the error occurred when executing the
+ operations.
+ Here errorOperation will be a pointer to the theOperation object.
+ NdbConnection::getNdbError will return the NdbError object
+ including holding information about the error.
+
+ Since errors could have occurred even when a commit was reported,
+ there is also a special method, NdbConnection::commitStatus,
+ to check the commit status of the transaction.
+
+*******************************************************************************/
+
+/**
+ * @page ndbapi_example1.cpp ndbapi_example1.cpp
+ * @include ndbapi_example1.cpp
+ */
+
+/**
+ * @page ndbapi_example2.cpp ndbapi_example2.cpp
+ * @include ndbapi_example2.cpp
+ */
+
+/**
+ * @page ndbapi_example3.cpp ndbapi_example3.cpp
+ * @include ndbapi_example3.cpp
+ */
+
+/**
+ * @page ndbapi_example4.cpp ndbapi_example4.cpp
+ * @include ndbapi_example4.cpp
+ */
+
+/**
+ * @page select_all.cpp select_all.cpp
+ * @include select_all.cpp
+ */
+
+/**
+ * @page ndbapi_async.cpp ndbapi_async.cpp
+ * @include ndbapi_async.cpp
+ */
+
+/**
+ * @page ndbapi_scan.cpp ndbapi_scan.cpp
+ * @include ndbapi_scan.cpp
+ */
+
+
+/**
+ @page secAdapt Adaptive Send Algorithm
+
+ At the time of "sending" the transaction
+ (using NdbConnection::execute, NdbConnection::executeAsynch,
+ Ndb::sendPreparedTransactions, or Ndb::sendPollNdb), the transactions
+ are in reality <em>not</em> immediately transfered to the NDB Kernel.
+ Instead, the "sent" transactions are only kept in a
+ special send list (buffer) in the Ndb object to which they belong.
+ The adaptive send algorithm decides when transactions should
+ be transfered to the NDB kernel.
+
+ For each of these "sent" transactions, there are three
+ possible states:
+ -# Waiting to be transfered to NDB Kernel.
+ -# Has been transfered to the NDB Kernel and is currently
+ being processed.
+ -# Has been transfered to the NDB Kernel and has
+ finished processing.
+ Now it is waiting for a call to a poll method.
+ (When the poll method is invoked,
+ then the transaction callback method will be executed.)
+
+ The poll method invoked (either Ndb::pollNdb or Ndb::sendPollNdb)
+ will return when:
+ -# at least 'minNoOfEventsToWakeup' of the transactions
+ in the send list have transitioned to state 3 as described above, and
+ -# all of these transactions have executed their callback methods.
+
+
+ Since the NDB API is designed as a multi-threaded interface,
+ it is desirable to transfer database operations from more than
+ one thread at a time.
+ The NDB API keeps track of which Ndb objects are active in transfering
+ information to the NDB kernel and the expected amount of threads to
+ interact with the NDB kernel.
+ Note that an Ndb object should be used in at most one thread.
+ Two different threads should <em>not</em> use the same Ndb object.
+
+ There are four reasons leading to transfering of database
+ operations:
+ -# The NDB Transporter (TCP/IP, OSE, SCI or shared memory)
+ decides that a buffer is full and sends it off.
+ The buffer size is implementation dependent and
+ might change between NDB Cluster releases.
+ On TCP/IP the buffer size is usually around 64 kByte and
+ on OSE/Delta it is usually less than 2000 bytes.
+ In each Ndb object there is one buffer per DB node,
+ so this criteria of a full buffer is only
+ local to the connection to one DB node.
+ -# Statistical information on the transfered information
+ may force sending of buffers to all DB nodes.
+ -# Every 10 ms a special send-thread checks whether
+ any send activity has occurred. If not, then the thread will
+ force sending to all nodes.
+ This means that 20 ms is the maximum time database operations
+ are waiting before being sent off. The 10 millisecond limit
+ is likely to become a configuration parameter in
+ later releases of NDB Cluster.
+ However, to support faster than 10 ms checks,
+ there has to be support from the operating system.
+ -# When calling NdbConnection::execute synchronously or calling any
+ of the poll-methods, there is a force parameter that overrides the
+ adaptive algorithm and forces the send to all nodes.
+
+ @note The times mentioned above are examples. These might
+ change in later releases of NDB Cluster.
+*/
+
+/**
+ @page secConcepts NDB Cluster Concepts
+
+ The <em>NDB Kernel</em> is the collection of database (DB) nodes
+ belonging to an NDB Cluster.
+ The application programmer can for most purposes view the
+ set of all DB nodes as one entity.
+ Each DB node has three main components:
+ - TC : The transaction coordinator
+ - ACC : The index storage
+ - TUP : The data storage
+
+ When the application program executes a transaction,
+ it connects to one TC on one DB node.
+ Usually, the programmer does not need to specify which TC to use,
+ but some cases when performance is important,
+ transactions can be hinted to use a certain TC.
+ (If the node with the TC is down, then another TC will
+ automatically take over the work.)
+
+ Every DB node has an ACC and a TUP which stores
+ the index and the data part of the database.
+ Even though one TC is responsible for the transaction,
+ several ACCs and TUPs on other DB nodes might be involved in the
+ execution of the transaction.
+
+
+ @section secNdbKernelConnection Selecting Transaction Coordinator
+ The default method is round robin,
+ where each new set of transactions
+ is placed on the next DB node.
+ The application chooses a TC for a number of transactions
+ and then lets the next TC (on the next DB node) carry out
+ the next set of transactions.
+
+ The application programmer can however hint the NDB API which
+ transaction coordinator to use
+ by providing a <em>distribution key</em> (usually the primary key).
+ By using the primary key as distribution key,
+ the transaction will be placed on the node where the primary replica
+ of that record resides.
+ Note that this is only a hint, the system can be
+ reconfigured and then the NDB API will choose a transaction
+ coordinator without using the hint.
+ For more information, see NdbDictionary::Column::setDistributionKey.
+
+
+ @section secRecordStruct Record Structure
+ NDB Cluster is a relational database with tables of records.
+ Table rows represent tuples of relational data stored as records.
+ When created, the attribute schema of the table is specified,
+ and thus each record of the table has the same schema.
+
+
+ @subsection secKeys Tuple Keys
+ Each record has from zero up to four attributes which belong
+ to the primary key of the table.
+ If no attribute belongs to the primary key, then
+ the NDB Cluster creates an attribute named <em>NDB$TID</em>
+ which stores a tuple identity.
+ The <em>tuple key</em> of a table is thus either
+ the primary key attributes or the special NDB$TID attribute.
+
+
+ @subsection secArrays Array Attributes
+ A table attribute in NDB Cluster can be of <em>array type</em>.
+ This means that the attribute consists of an array of
+ <em>elements</em>. The <em>attribute size</em> is the size
+ of one element of the array (expressed in bits) and the
+ <em>array size</em> is the number of elements of the array.
+
+
+ @section secTrans Transactions
+
+ Transactions are committed to main memory,
+ and are committed to disk after a global checkpoint, GCP.
+ Since all data is (in most NDB Cluster configurations)
+ synchronously replicated and stored on multiple NDB nodes,
+ the system can still handle processor failures without loss
+ of data.
+ However, in the case of a system failure (e.g. the whole system goes down),
+ then all (committed or not) transactions after the latest GCP are lost.
+
+
+ @subsection secConcur Concurrency Control
+ NDB Cluster uses pessimistic concurrency control based on locking.
+ If a requested lock (implicit and depending on database operation)
+ cannot be attained within a specified time,
+ then a timeout error occurs.
+
+ Concurrent transactions (parallel application programs, thread-based
+ applications, or applications with asynchronous transactions)
+ sometimes deadlock when they try to access the same information.
+ Applications need to be programmed so that timeout errors
+ occurring due to deadlocks are handled. This generally
+ means that the transaction encountering timeout
+ should be rolled back and restarted.
+*/
+
+#ifndef Ndb_H
+#define Ndb_H
+
+#include <ndb_types.h>
+#include <ndbapi_limits.h>
+#include "AttrType.hpp"
+#include <NdbError.hpp>
+#include "NdbDictionary.hpp"
+
+class NdbObjectIdMap;
+class NdbOperation;
+class NdbEventOperationImpl;
+class NdbScanOperation;
+class NdbIndexOperation;
+class NdbConnection;
+class NdbSchemaOp;
+class NdbSchemaCon;
+class NdbApiSignal;
+class NdbRecAttr;
+class NdbLabel;
+class NdbBranch;
+class NdbSubroutine;
+class NdbCall;
+class NdbScanReceiver;
+class Table;
+class BaseString;
+class NdbEventOperation;
+
+typedef void (* NdbEventCallback)(NdbEventOperation*, Ndb*, void*);
+
+
+#if defined NDB_OSE
+/**
+ * Default time to wait for response after request has been sent to
+ * NDB Cluster (Set to 10 seconds usually, but to 100 s in
+ * the OSE operating system)
+ */
+#define WAITFOR_RESPONSE_TIMEOUT 100000 // Milliseconds
+#else
+#define WAITFOR_RESPONSE_TIMEOUT 120000 // Milliseconds
+#endif
+
+#define NDB_MAX_INTERNAL_TABLE_LENGTH NDB_MAX_DATABASE_NAME_SIZE + \
+ NDB_MAX_SCHEMA_NAME_SIZE + \
+ NDB_MAX_TAB_NAME_SIZE*2
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+class NdbWaiter {
+public:
+ NdbWaiter();
+ ~NdbWaiter();
+
+ void wait(int waitTime);
+ void nodeFail(Uint32 node);
+ void signal(Uint32 state);
+
+ Uint32 m_node;
+ Uint32 m_state;
+ void * m_mutex;
+ struct NdbCondition * m_condition;
+};
+#endif
+
+/**
+ * @class Ndb
+ * @brief Represents the NDB kernel and is the main class of the NDB API.
+ *
+ * Always start your application program by creating an Ndb object.
+ * By using several Ndb objects it is possible to design
+ * a multi-threaded application, but note that Ndb objects
+ * cannot be shared by several threads.
+ * Different threads should use different Ndb objects.
+ * A thread might however use multiple Ndb objects.
+ * Currently there is a limit of maximum 128 Ndb objects
+ * per application process.
+ *
+ * @note It is not allowed to call methods in the NDB API
+ * on the same Ndb object in different threads
+ * simultaneously (without special handling of the
+ * Ndb object).
+ *
+ * @note The Ndb object is multi-thread safe in the following manner.
+ * Each Ndb object can ONLY be handled in one thread.
+ * If an Ndb object is handed over to another thread then the
+ * application must ensure that a memory barrier is used to
+ * ensure that the new thread see all updates performed by
+ * the previous thread.
+ * Semaphores, mutexes and so forth are easy ways of issuing memory
+ * barriers without having to bother about the memory barrier concept.
+ *
+ * If one Ndb object is used to handle parallel transactions through the
+ * asynchronous programming interface, please read the notes regarding
+ * asynchronous transactions (Section @ref secAsync).
+ * The asynchronous interface provides much higher performance
+ * in some situations, but is more complicated for the application designer.
+ *
+ * @note Each Ndb object should either use the methods for
+ * asynchronous transaction or the methods for
+ * synchronous transactions but not both.
+ */
+class Ndb
+{
+ friend class NdbReceiver;
+ friend class NdbOperation;
+ friend class NdbEventOperationImpl;
+ friend class NdbConnection;
+ friend class NdbSchemaOp;
+ friend class NdbSchemaCon;
+ friend class Table;
+ friend class NdbApiSignal;
+ friend class NdbScanReceiver;
+ friend class NdbIndexOperation;
+ friend class NdbDictionaryImpl;
+ friend class NdbDictInterface;
+
+public:
+ /**
+ * @name General
+ * @{
+ */
+ /**
+ * The starting point of your application code is to create an
+ * Ndb object.
+ * This object represents the NDB kernel and is the main
+ * object used in interaction with the NDB kernel.
+ *
+ * @param aCatalogName is the name of the catalog you want to use.
+ * @note The catalog name provides a name space for the tables and
+ * indexes created in any connection from the Ndb object.
+ * @param aSchemaName is the name of the schema you
+ * want to use. It is optional and defaults to the "def" schema.
+ * @note The schema name provides an additional name space
+ * for the tables and indexes created in a given catalog.
+ * @note The methods get/setDatabaseName and get/setDatabaseSchemaName
+ * are equivalent to get/setCatalogName and get/setSchemaName.
+ * The get/setDatabaseName and get/setDatabaseSchemaName are
+ * deprecated.
+ */
+ Ndb(const char* aCatalogName = "", const char* aSchemaName = "def");
+
+ ~Ndb();
+
+ /**
+ * The current catalog name can be fetched by getCatalogName.
+ *
+ * @return the current catalog name
+ */
+ const char * getCatalogName() const;
+
+ /**
+ * The current catalog name can be set by setCatalogName.
+ *
+ * @param aCatalogName is the new name of the current catalog
+ */
+ void setCatalogName(const char * aCatalogName);
+
+ /**
+ * The current schema name can be fetched by getSchemaName.
+ *
+ * @return the current schema name
+ */
+ const char * getSchemaName() const;
+
+ /**
+ * The current schema name can be set by setSchemaName.
+ *
+ * @param aSchemaName is the new name of the current schema
+ */
+ void setSchemaName(const char * aSchemaName);
+
+
+ /**
+ * The current database name can be fetched by getDatabaseName.
+ *
+ * @return the current database name
+ */
+ const char * getDatabaseName() const;
+
+ /**
+ * The current database name can be set by setDatabaseName.
+ *
+ * @param aDatabaseName is the new name of the current database
+ */
+ void setDatabaseName(const char * aDatabaseName);
+
+ /**
+ * The current database schema name can be fetched by getDatabaseSchemaName.
+ *
+ * @return the current database schema name
+ */
+ const char * getDatabaseSchemaName() const;
+
+ /**
+ * The current database schema name can be set by setDatabaseSchemaName.
+ *
+ * @param aDatabaseSchemaName is the new name of the current database schema
+ */
+ void setDatabaseSchemaName(const char * aDatabaseSchemaName);
+
+ /**
+ * Before anything else it is necessary to initialize (start)
+ * the Ndb object.
+ *
+ * @param maxNoOfTransactions
+ * Maximum number of parallel
+ * NdbConnection objects that should be handled by the Ndb object.
+ * A value larger than 1024 will be downgraded to 1024.
+ * This means that one Ndb object can handle at most 1024 parallel
+ * transactions.
+ * There is a maximum of 128 simultaneous
+ * Ndb object within one application process.
+ * @return 0 if successful, -1 otherwise.
+ *
+ * @note The internal implementation multiplies this value
+ * with 3.
+ */
+ int init(int maxNoOfTransactions = 4);
+
+ /**
+ * Wait for Ndb object to successfully set-up connections to
+ * the NDB kernel.
+ * Starting to use the Ndb object without using this method
+ * gives unspecified behavior.
+ *
+ * @param timeout The maximum time we will wait for
+ * the initiation process to finish.
+ * Timeout is expressed in seconds.
+ * @return 0: Ndb is ready and timeout has not occurred.<br>
+ * -1: Timeout has expired
+ */
+ int waitUntilReady(int timeout = 60);
+
+ /** @} *********************************************************************/
+
+ /**
+ * @name Meta Information
+ * @{
+ */
+
+ /**
+ * Query the database for schema information
+ * (without performing any transaction).
+ *
+ * @return Object containing meta information about all tables
+ * in NDB Cluster.
+ */
+ class NdbDictionary::Dictionary* getDictionary() const;
+
+ NdbEventOperation* createEventOperation(const char* eventName,
+ const int bufferLength);
+ int dropEventOperation(NdbEventOperation*);
+ void monitorEvent(NdbEventOperation *, NdbEventCallback, void*);
+ int pollEvents(int aMillisecondNumber);
+
+ /**
+ * Get the application node identity.
+ *
+ * Each node (DB nodes, Applications, and Management Servers)
+ * has its own node identity in the NDB Cluster.
+ * See documentation for the management server configuration file.
+ *
+ * @return Node id of this application.
+ */
+ int getNodeId();
+
+ /** @} *********************************************************************/
+
+ /**
+ * @name Starting and Closing Transactions
+ * @{
+ */
+
+ /**
+ * This method returns an NdbConnection which caters for the transaction.
+ * When the transaction is completed it must be closed.
+ * The Ndb::closeTransaction also return the NdbConnection object
+ * and all other memory related to the transaction.
+ * Failure to close the transaction will lead to memory leakage.
+ * The transaction must be closed independent of its outcome, i.e.
+ * even if there is an error.
+ *
+ * NDB API can be hinted to select a particular transaction coordinator.
+ * The default method is round robin where each set of new transactions
+ * is placed on the next NDB kernel node.
+ * By providing a distribution key (usually the primary key
+ * of the mostly used table of the transaction) for a record
+ * the transaction will be placed on the node where the primary replica
+ * of that record resides.
+ * Note that this is only a hint, the system can
+ * be under reconfiguration and then the NDB API
+ * will use select the transaction coordinator without using
+ * this hint.
+ *
+ * Placing the transaction coordinator close
+ * to the actual data used in the transaction can in many cases
+ * improve performance significantly. This is particularly true for
+ * systems using TCP/IP. A system using Solaris and a 500 MHz processor
+ * has a cost model for TCP/IP communication which is:
+ *
+ * 30 microseconds + (100 nanoseconds * no of Bytes)
+ *
+ * This means that if we can ensure that we use "popular" links we increase
+ * buffering and thus drastically reduce the communication cost.
+ * Systems using SCI has a different cost model which is:
+ *
+ * 5 microseconds + (10 nanoseconds * no of Bytes)
+ *
+ * Thus SCI systems are much less dependent on selection of
+ * transaction coordinators.
+ * Typically TCP/IP systems spend 30-60% of the time during communication,
+ * whereas SCI systems typically spend 5-10% of the time during
+ * communication.
+ * Thus SCI means that less care from the NDB API programmer is
+ * needed and great scalability can be achieved even for applications using
+ * data from many parts of the database.
+ *
+ * A simple example is an application that uses many simple updates where
+ * a transaction needs to update one record.
+ * This record has a 32 bit primary key,
+ * which is also the distribution key.
+ * Then the keyData will be the address of the integer
+ * of the primary key and keyLen will be 4.
+ *
+ * @note Transaction priorities are not yet supported.
+ *
+ * @param prio The priority of the transaction.<br>
+ * Priority 0 is the highest priority and is used
+ * for short transactions with requirements on low delay.<br>
+ * Priority 1 is a medium priority for short transactions.
+ * <br>
+ * Priority 2 is a medium priority for long transactions.<br>
+ * Priority 3 is a low priority for long transactions.<br>
+ * <em>This parameter is not currently used,
+ * and can be set to any value</em>
+ * @param keyData Pointer to distribution key
+ * @param keyLen Length of distribution key expressed in bytes
+ *
+ * @return NdbConnection object, or NULL if method failed.
+ */
+ NdbConnection* startTransaction(Uint32 prio = 0,
+ const char * keyData = 0,
+ Uint32 keyLen = 0);
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ /**
+ * This method is a modification of Ndb::startTransaction,
+ * in which we use only the first two chars of keyData to
+ * select transaction coordinator.
+ * This is referred to as a distribution group.
+ * There are two ways to use the method:
+ * - In the first, the two characters are used directly as
+ * the distribution key, and
+ * - in the second the distribution is calculated as:
+ * (10 * (char[0] - 0x30) + (char[1] - 0x30)).
+ * Thus, in the second way, the two ASCII digits '78'
+ * will provide the distribution key = 78.
+ *
+ * @note Transaction priorities are not yet supported.
+ *
+ * @param aPrio Priority of the transaction.<br>
+ * Priority 0 is the highest priority and is used for short transactions
+ * with requirements on low delay.<br>
+ * Priority 1 is a medium priority for short transactions.<br>
+ * Priority 2 is a medium priority for long transactions.<br>
+ * Priority 3 is a low priority for long transactions.
+ * @param keyData is a string of which the two first characters
+ * is used to compute which fragement the data is stored in.
+ * @param type is the type of distribution group.<br>
+ * 0 means direct usage of the two characters, and<br>
+ * 1 means the ASCII digit variant.
+ * @return NdbConnection, or NULL if it failed.
+ */
+ NdbConnection* startTransactionDGroup(Uint32 aPrio,
+ const char * keyData, int type);
+#endif
+
+ /**
+ * When a transactions is completed, the transaction has to be closed.
+ *
+ * @note It is not allowed to call Ndb::closeTransaction after sending the
+ * transaction asynchronously with either
+ * Ndb::sendPreparedTransactions or
+ * Ndb::sendPollNdb before the callback method has been called.
+ * (The application should keep track of the number of
+ * outstanding transactions and wait until all of them
+ * has completed before calling Ndb::closeTransaction).
+ * If the transaction is not committed it will be aborted.
+ */
+ void closeTransaction(NdbConnection* aConnection);
+
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+ /**
+ * To create a table it is necessary to obtain a schema transaction
+ * object.
+ * All schema transactions need to closed when they are
+ * completed.
+ *
+ * @return NdbSchemaCon
+ */
+ NdbSchemaCon* startSchemaTransaction();
+
+ /**
+ * Close schema transaction when finished.
+ */
+ void closeSchemaTransaction(NdbSchemaCon* aSchemaCon);
+#endif
+
+ /** @} *********************************************************************/
+
+ /**
+ * @name Asynchronous Transactions
+ * @{
+ */
+
+ /**
+ * Wait for prepared transactions.
+ * Will return as soon as at least 'minNoOfEventsToWakeUp'
+ * of them have completed, or the maximum time given as timeout has passed.
+ *
+ * @param aMillisecondNumber Maximum time to wait for transactions
+ * to complete.
+ * Polling without wait is achieved by setting the
+ * timer to zero.
+ * Time is expressed in milliseconds.
+ * @param minNoOfEventsToWakeup Minimum number of transactions
+ * which has to wake up before the poll-call will return.
+ * If minNoOfEventsToWakeup is
+ * set to a value larger than 1 then this is the minimum
+ * number of transactions that need to complete before the
+ * poll will return.
+ * Setting it to zero means that one should wait for all
+ * outstanding transactions to return before waking up.
+ * @return Number of transactions polled.
+ */
+ int pollNdb(int aMillisecondNumber = WAITFOR_RESPONSE_TIMEOUT,
+ int minNoOfEventsToWakeup = 1);
+
+ /**
+ * This send method will send all prepared database operations.
+ * The default method is to do it non-force and instead
+ * use the adaptive algorithm. (See Section @ref secAdapt.)
+ * The second option is to force the sending and
+ * finally there is the third alternative which is
+ * also non-force but also making sure that the
+ * adaptive algorithm do not notice the send.
+ * In this case the sending will be performed on a
+ * cyclical 10 millisecond event.
+ *
+ * @param forceSend When operations should be sent to NDB Kernel.
+ * (See @ref secAdapt.)
+ * - 0: non-force, adaptive algorithm notices it (default);
+ * - 1: force send, adaptive algorithm notices it;
+ * - 2: non-force, adaptive algorithm do not notice the send.
+ */
+ void sendPreparedTransactions(int forceSend = 0);
+
+ /**
+ * This is a send-poll variant that first calls
+ * Ndb::sendPreparedTransactions and then Ndb::pollNdb.
+ * It is however somewhat faster than calling the methods
+ * separately, since some mutex-operations are avoided.
+ * See documentation of Ndb::pollNdb and Ndb::sendPreparedTransactions
+ * for more details.
+ *
+ * @param aMillisecondNumber Timeout specifier
+ * Polling without wait is achieved by setting the
+ * millisecond timer to zero.
+ * @param minNoOfEventsToWakeup Minimum number of transactions
+ * which has to wake up before the poll-call will return.
+ * If minNoOfEventsToWakeup is
+ * set to a value larger than 1 then this is the minimum
+ * number of transactions that need to complete before the
+ * poll-call will return.
+ * Setting it to zero means that one should wait for all
+ * outstanding transactions to return before waking up.
+ * @param forceSend When operations should be sent to NDB Kernel.
+ * (See @ref secAdapt.)
+ * - 0: non-force, adaptive algorithm notices it (default);
+ * - 1: force send, adaptive algorithm notices it;
+ * - 2: non-force, adaptive algorithm does not notice the send.
+ * @return Number of transactions polled.
+ */
+ int sendPollNdb(int aMillisecondNumber = WAITFOR_RESPONSE_TIMEOUT,
+ int minNoOfEventsToWakeup = 1,
+ int forceSend = 0);
+
+ /** @} *********************************************************************/
+
+ /**
+ * @name Error Handling
+ * @{
+ */
+
+ /**
+ * Get the NdbError object
+ *
+ * The NdbError object is valid until you call a new NDB API method.
+ */
+ const NdbError & getNdbError() const;
+
+ /**
+ * Get a NdbError object for a specific error code
+ *
+ * The NdbError object is valid until you call a new NDB API method.
+ */
+ const NdbError & getNdbError(int errorCode);
+
+
+ /**
+ * setConnectString
+ * @param connectString - the connectString has the following format:
+ * @code
+ * "nodeid=<ID>;host=host://<HOSTNAME>:<PORT>;
+ * host=host://<HOSTNAME2>:<PORT>;..."
+ * @endcode
+ * or
+ * @code
+ * "nodeid=<ID>;host=<HOSTNAME>:<PORT>;host=<HOSTNAME2>:<PORT>;..."
+ * @endcode
+ */
+ static void setConnectString(const char * connectString);
+
+ /**
+ * useFullyQualifiedNames
+ * Enables unique name space for different databases and schemas
+ * by defining table names as DATABASENAME/SCHEMANAME/TABLENAME and
+ * index names as DATABASENAME/SCHEMANAME/TABLENAME/INDEXNAME
+ * @param turnNamingOn bool true - turn naming on, false - turn naming off
+ */
+ static void useFullyQualifiedNames(bool turnNamingOn = true);
+
+ static bool usingFullyQualifiedNames();
+
+ /** @} *********************************************************************/
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ /**
+ * For testing purposes it is possible to tamper with the NDB Cluster
+ * (i.e. send a special signal to DBDIH, the NDB distribution handler).
+ * <b>This feature should only used for debugging purposes.</b>
+ * In a release versions of NDB Cluster,
+ * this call always return -1 and does nothing.
+ *
+ * @param aAction Action to be taken
+ * - 1: Lock global checkpointing
+ * (Can only be sent to master DIH,
+ * Parameter aNode ignored).
+ * - 2: UnLock global checkpointing
+ * (Can only be sent to master DIH,
+ * Parameter aNode ignored).
+ * - 3: Crash node.
+ *
+ * @param aNode Which node the action will be taken
+ * -1: Master DIH.
+ * 0-16: Nodnumber.
+ * @return -1 indicates error, other values have meaning dependent
+ * on type of tampering.
+ */
+ int NdbTamper(TamperType aAction, int aNode);
+#endif
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ /**
+ * Return a unique tuple id for a table. The id sequence is
+ * ascending but may contain gaps.
+ *
+ * @param aTableName table name
+ *
+ * @param cacheSize number of values to cache in this Ndb object
+ *
+ * @return tuple id or 0 on error
+ */
+ Uint64 getAutoIncrementValue(const char* aTableName, Uint32 cacheSize = 1);
+ bool setAutoIncrementValue(const char* aTableName, Uint64 val);
+ Uint64 getTupleIdFromNdb(const char* aTableName, Uint32 cacheSize = 1000 );
+ Uint64 getTupleIdFromNdb(Uint32 aTableId, Uint32 cacheSize = 1000 );
+ bool setTupleIdInNdb(const char* aTableName, Uint64 val);
+ bool setTupleIdInNdb(Uint32 aTableId, Uint64 val);
+ Uint64 opTupleIdOnNdb(Uint32 aTableId, Uint64 opValue, Uint32 op);
+#endif
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ /**
+ */
+ NdbConnection* hupp( NdbConnection* );
+ Uint32 getReference() const { return theMyRef;}
+#endif
+
+/*****************************************************************************
+ * These are service routines used by the other classes in the NDBAPI.
+ ****************************************************************************/
+private:
+
+ NdbConnection* startTransactionLocal(Uint32 aPrio, Uint32 aFragmentId);
+
+// Connect the connection object to the Database.
+ int NDB_connect(Uint32 tNode);
+ NdbConnection* doConnect(Uint32 nodeId);
+ void doDisconnect();
+
+ NdbScanReceiver* getNdbScanRec();// Get a NdbScanReceiver from idle list
+ NdbLabel* getNdbLabel(); // Get a NdbLabel from idle list
+ NdbBranch* getNdbBranch(); // Get a NdbBranch from idle list
+ NdbSubroutine* getNdbSubroutine();// Get a NdbSubroutine from idle
+ NdbCall* getNdbCall(); // Get a NdbCall from idle list
+ NdbApiSignal* getSignal(); // Get an operation from idle list
+ NdbRecAttr* getRecAttr(); // Get a receeive attribute object from
+ // idle list of the Ndb object.
+ NdbOperation* getOperation(); // Get an operation from idle list
+ NdbScanOperation* getScanOperation(); // Get a scan operation from idle
+ NdbIndexOperation* getIndexOperation();// Get an index operation from idle
+
+ class NdbGlobalEventBufferHandle* getGlobalEventBufferHandle();
+
+ void releaseSignal(NdbApiSignal* anApiSignal);
+ void releaseSignalsInList(NdbApiSignal** pList);
+ void releaseNdbScanRec(NdbScanReceiver* aNdbScanRec);
+ void releaseNdbLabel(NdbLabel* anNdbLabel);
+ void releaseNdbBranch(NdbBranch* anNdbBranch);
+ void releaseNdbSubroutine(NdbSubroutine* anNdbSubroutine);
+ void releaseNdbCall(NdbCall* anNdbCall);
+ void releaseRecAttr (NdbRecAttr* aRecAttr);
+ void releaseOperation(NdbOperation* anOperation);
+ void releaseScanOperation(NdbScanOperation* aScanOperation);
+
+ void check_send_timeout();
+ void remove_sent_list(Uint32);
+ Uint32 insert_completed_list(NdbConnection*);
+ Uint32 insert_sent_list(NdbConnection*);
+
+ // Handle a received signal. Used by both
+ // synchronous and asynchronous interface
+ void handleReceivedSignal(NdbApiSignal* anApiSignal, struct LinearSectionPtr ptr[3]);
+
+ // Receive response signals
+ int receiveResponse(int waitTime = WAITFOR_RESPONSE_TIMEOUT);
+
+ int sendRecSignal(Uint16 aNodeId,
+ Uint32 aWaitState,
+ NdbApiSignal* aSignal,
+ Uint32 nodeSequence);
+
+ // Sets Restart GCI in Ndb object
+ void RestartGCI(int aRestartGCI);
+
+ // Get block number of this NDBAPI object
+ int getBlockNumber();
+
+ /****************************************************************************
+ * These are local service routines used by this class.
+ ***************************************************************************/
+
+ int createConIdleList(int aNrOfCon);
+ int createOpIdleList( int nrOfOp );
+
+ void freeOperation(); // Free the first idle operation.
+ void freeScanOperation(); // Free the first idle scan operation.
+ void freeIndexOperation(); // Free the first idle index operation.
+ void freeNdbCon(); // Free the first idle connection.
+ void freeSignal(); // Free the first idle signal
+ void freeRecAttr(); // Free the first idle receive attr obj
+ void freeNdbLabel(); // Free the first idle NdbLabel obj
+ void freeNdbBranch();// Free the first idle NdbBranch obj
+ void freeNdbSubroutine();// Free the first idle NdbSubroutine obj
+ void freeNdbCall(); // Free the first idle NdbCall obj
+ void freeNdbScanRec(); // Free the first idle NdbScanRec obj
+
+ NdbConnection* getNdbCon(); // Get a connection from idle list
+
+ /**
+ * Get a connected NdbConnection to nodeId
+ * Returns NULL if none found
+ */
+ NdbConnection* getConnectedNdbConnection(Uint32 nodeId);
+
+ // Release and disconnect from DBTC a connection
+ // and seize it to theConIdleList
+ void releaseConnectToNdb (NdbConnection* aConnectConnection);
+
+ // Release a connection to idle list
+ void releaseNdbCon (NdbConnection* aConnection);
+
+ int checkInitState(); // Check that we are initialized
+ void report_node_failure(Uint32 node_id); // Report Failed node
+ void report_node_failure_completed(Uint32 node_id); // Report Failed node(NF comp.)
+
+ void checkFailedNode(); // Check for failed nodes
+
+ int NDB_connect(); // Perform connect towards NDB Kernel
+
+ // Release arrays of NdbConnection pointers
+ void releaseTransactionArrays();
+
+ Uint32 pollCompleted(NdbConnection** aCopyArray);
+ void sendPrepTrans(int forceSend);
+ void reportCallback(NdbConnection** aCopyArray, Uint32 aNoOfComplTrans);
+ void waitCompletedTransactions(int milliSecs, int noOfEventsToWaitFor);
+ void completedTransaction(NdbConnection* aTransaction);
+ void completedScanTransaction(NdbConnection* aTransaction);
+
+ void abortTransactionsAfterNodeFailure(Uint16 aNodeId);
+
+ static
+ const char * externalizeTableName(const char * internalTableName);
+ const char * internalizeTableName(const char * externalTableName);
+
+ static
+ const char * externalizeIndexName(const char * internalIndexName);
+ const char * internalizeIndexName(const NdbTableImpl * table,
+ const char * externalIndexName);
+
+ static
+ const BaseString getDatabaseFromInternalName(const char * internalName);
+ static
+ const BaseString getSchemaFromInternalName(const char * internalName);
+
+ void* int2void (Uint32 val);
+ NdbReceiver* void2rec (void* val);
+ NdbConnection* void2con (void* val);
+ NdbScanReceiver* void2rec_srec(void* val);
+ NdbOperation* void2rec_op (void* val);
+ NdbIndexOperation* void2rec_iop (void* val);
+
+/******************************************************************************
+ * These are the private variables in this class.
+ *****************************************************************************/
+ NdbObjectIdMap* theNdbObjectIdMap;
+
+ NdbConnection** thePreparedTransactionsArray;
+ NdbConnection** theSentTransactionsArray;
+ NdbConnection** theCompletedTransactionsArray;
+
+ Uint32 theNoOfPreparedTransactions;
+ Uint32 theNoOfSentTransactions;
+ Uint32 theNoOfCompletedTransactions;
+ Uint32 theNoOfAllocatedTransactions;
+ Uint32 theMaxNoOfTransactions;
+ Uint32 theMinNoOfEventsToWakeUp;
+
+ Uint32 theNextConnectNode;
+
+ NdbWaiter theWaiter;
+
+ // Ndb database name.
+ char theDataBase[NDB_MAX_DATABASE_NAME_SIZE];
+ // Ndb database schema name.
+ char theDataBaseSchema[NDB_MAX_SCHEMA_NAME_SIZE];
+ char prefixName[NDB_MAX_INTERNAL_TABLE_LENGTH];
+ char * prefixEnd;
+
+ //Table* theTable; // The table object
+ class NdbImpl * theImpl;
+ class NdbDictionaryImpl* theDictionary;
+ class NdbGlobalEventBufferHandle* theGlobalEventBufferHandle;
+
+ NdbConnection* theConIdleList; // First connection in idle list.
+
+ NdbOperation* theOpIdleList; // First operation in the idle list.
+
+ NdbScanOperation* theScanOpIdleList; // First scan operation in the idle list.
+ NdbIndexOperation* theIndexOpIdleList; // First index operation in the idle list.
+ NdbSchemaCon* theSchemaConIdleList; // First schemaCon in idle list.
+
+ NdbSchemaCon* theSchemaConToNdbList; // Connected schemaCon object.
+ NdbConnection* theTransactionList;
+ NdbConnection** theConnectionArray;
+ NdbRecAttr* theRecAttrIdleList;
+ NdbApiSignal* theSignalIdleList; // First signal in idlelist.
+ NdbLabel* theLabelList; // First label descriptor in list
+ NdbBranch* theBranchList; // First branch descriptor in list
+ NdbSubroutine* theSubroutineList; // First subroutine descriptor in
+ NdbCall* theCallList; // First call descriptor in list
+ NdbScanReceiver* theScanList;
+
+ Uint32 theMyRef; // My block reference
+ Uint32 theNode; // The node number of our node
+
+ Uint32 theNoOfDBnodes; // The number of DB nodes
+ Uint32 * theDBnodes; // The node number of the DB nodes
+ Uint8 *the_release_ind;// 1 indicates to release all connections to node
+
+ Uint64 the_last_check_time;
+ Uint64 theFirstTransId;
+
+ // The tupleId is retreived from DB the
+ // tupleId is unique for each tableid.
+ Uint64 theFirstTupleId[2048];
+ Uint64 theLastTupleId[2048];
+
+ Uint32 theRestartGCI; // the Restart GCI used by DIHNDBTAMPER
+
+ NdbError theError;
+
+ Int32 theNdbBlockNumber;
+ InitType theInitState;
+
+ // Ensure good distribution of connects
+ Uint32 theCurrentConnectIndex;
+ Uint32 theCurrentConnectCounter;
+
+ /**
+ * Computes fragement id for primary key
+ *
+ * Note that keydata has to be "shaped" as it is being sent in KEYINFO
+ */
+ Uint32 computeFragmentId(const char * keyData, Uint32 keyLen);
+ Uint32 getFragmentId(Uint32 hashValue);
+
+ /**
+ * Make a guess to which node is the primary for the fragment
+ */
+ Uint32 guessPrimaryNode(Uint32 fragmentId);
+
+ /**
+ * Structure containing values for guessing primary node
+ */
+ struct StartTransactionNodeSelectionData {
+ StartTransactionNodeSelectionData():
+ fragment2PrimaryNodeMap(NULL) {};
+ Uint32 kValue;
+ Uint32 hashValueMask;
+ Uint32 hashpointerValue;
+ Uint32 noOfFragments;
+ Uint32 * fragment2PrimaryNodeMap;
+
+ void init(Uint32 noOfNodes, Uint32 nodeIds[]);
+ void release();
+ } startTransactionNodeSelectionData;
+
+ NdbApiSignal* theCommitAckSignal;
+
+ int cfreeSignals;
+ int cnewSignals;
+ int cgetSignals;
+ int creleaseSignals;
+
+ static void executeMessage(void*, NdbApiSignal *,
+ struct LinearSectionPtr ptr[3]);
+ static void statusMessage(void*, Uint16, bool, bool);
+#ifdef VM_TRACE
+ void printState(const char* fmt, ...);
+#endif
+};
+
+#endif
diff --git a/ndb/include/ndbapi/NdbApi.hpp b/ndb/include/ndbapi/NdbApi.hpp
new file mode 100644
index 00000000000..e5efc9756ce
--- /dev/null
+++ b/ndb/include/ndbapi/NdbApi.hpp
@@ -0,0 +1,33 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbApi_H
+#define NdbApi_H
+
+#include "Ndb.hpp"
+#include "AttrType.hpp"
+#include "NdbConnection.hpp"
+#include "NdbOperation.hpp"
+#include "NdbScanOperation.hpp"
+#include "NdbIndexOperation.hpp"
+#include "NdbSchemaCon.hpp"
+#include "NdbSchemaOp.hpp"
+#include "NdbRecAttr.hpp"
+#include "NdbResultSet.hpp"
+#include "NdbDictionary.hpp"
+#include "NdbEventOperation.hpp"
+#include "NdbPool.hpp"
+#endif
diff --git a/ndb/include/ndbapi/NdbConnection.hpp b/ndb/include/ndbapi/NdbConnection.hpp
new file mode 100644
index 00000000000..a1532bb2f0e
--- /dev/null
+++ b/ndb/include/ndbapi/NdbConnection.hpp
@@ -0,0 +1,885 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbConnection_H
+#define NdbConnection_H
+
+#include <ndb_types.h>
+#include "AttrType.hpp"
+#include <NdbError.hpp>
+#include <stdlib.h>
+
+class NdbConnection;
+class NdbOperation;
+class NdbCursorOperation;
+class NdbScanOperation;
+class NdbIndexOperation;
+class NdbApiSignal;
+class Ndb;
+class NdbScanReceiver;
+
+
+/**
+ * NdbAsynchCallback functions are used when executing asynchronous
+ * transactions (using NdbConnection::executeAsynchPrepare, or
+ * NdbConnection::executeAsynch).
+ * The functions are called when the execute has finished.
+ * See @ref secAsync for more information.
+ */
+typedef void (* NdbAsynchCallback)(int, NdbConnection*, void*);
+
+/**
+ * @class NdbConnection
+ * @brief Represents a transaction.
+ *
+ * A transaction (represented by an NdbConnection object)
+ * belongs to an Ndb object and is typically created using
+ * Ndb::startTransaction.
+ * A transaction consists of a list of operations
+ * (represented by NdbOperation objects).
+ * Each operation access exactly one table.
+ *
+ * After getting the NdbConnection object,
+ * the first step is to get (allocate) an operation given the table name.
+ * Then the operation is defined.
+ * Several operations can be defined in parallel on the same
+ * NdbConnection object.
+ * When all operations are defined, the NdbConnection::execute
+ * method sends them to the NDB kernel for execution.
+ *
+ * The NdbConnection::execute method returns when the NDB kernel has
+ * completed execution of all operations defined before the call to
+ * NdbConnection::execute.
+ * All allocated operations should be properly defined
+ * before calling NdbConnection::execute.
+ *
+ * A call to NdbConnection::execute uses one out of three types of execution:
+ * -# ExecType::NoCommit Executes operations without committing them.
+ * -# ExecType::Commit Executes remaining operation and commits the
+ * complete transaction
+ * -# ExecType::Rollback Rollbacks the entire transaction.
+ *
+ * NdbConnection::execute is equipped with an extra error handling parameter
+ * There are two alternatives:
+ * -# AbortOption::AbortOnError (default).
+ * The transaction is aborted if there are any error during the
+ * execution
+ * -# AbortOption::IgnoreError
+ * Continue execution of transaction even if operation fails
+ *
+ * NdbConnection::execute can sometimes indicate an error
+ * (return with -1) while the error code on the NdbConnection is 0.
+ * This is an indication that one of the operations found a record
+ * problem. The transaction is still ok and can continue as usual.
+ * The NdbConnection::execute returns -1 together with error code
+ * on NdbConnection object equal to 0 always means that an
+ * operation was not successful but that the total transaction was OK.
+ * By checking error codes on the individual operations it is possible
+ * to find out which operation was not successful.
+ *
+ * NdbConnection::executeScan is used to setup a scan in the NDB kernel
+ * after it has been defined.
+ * NdbConnection::nextScanResult is used to iterate through the
+ * scanned tuples.
+ * After each call to NdbConnection::nextScanResult, the pointers
+ * of NdbRecAttr objects defined in the NdbOperation::getValue
+ * operations are updated with the values of the new the scanned tuple.
+ */
+
+/* FUTURE IMPLEMENTATION:
+ * Later a prepare mode will be added when Ndb supports Prepare-To-Commit
+ * The NdbConnection can deliver the Transaction Id of the transaction.
+ * After committing a transaction it is also possible to retrieve the
+ * global transaction checkpoint which the transaction was put in.
+ *
+ * FUTURE IMPLEMENTATION:
+ * There are three methods for acquiring the NdbOperation.
+ * -# The first method is the normal where a table name is
+ * provided. In this case the primary key must be supplied through
+ * the use of the NdbOperation::equal methods on the NdbOperation object.
+ * -# The second method provides the tuple identity of the tuple to be
+ * read. The tuple identity contains a table identifier and will
+ * thus be possible to use to ensure the attribute names provided
+ * are correct. If an object-oriented layer is put on top of NDB
+ * Cluster it is essential that all tables derived from a base
+ * class has the same attributes with the same type and the same
+ * name. Thus the application can use the tuple identity and need
+ * not known the table of the tuple. As long as the table is
+ * derived from the known base class everything is ok.
+ * It is not possible to provide any primary key since it is
+ * already supplied with the call to NdbConnection::getNdbOperation.
+ * -# The third method is used when a scanned tuple is to be transferred to
+ * another transaction. In this case it is not possible to define the
+ * primary key since it came along from the scanned tuple.
+ *
+ */
+class NdbConnection
+{
+ friend class Ndb;
+ friend class NdbOperation;
+ friend class NdbScanOperation;
+ friend class NdbIndexOperation;
+ friend class NdbScanReceiver;
+
+public:
+
+ /**
+ * Get an NdbOperation for a table.
+ * Note that the operation has to be defined before it is executed.
+ *
+ * @note All operations within the same transaction need to
+ * be initialized with this method.
+ *
+ * @param aTableName The table name.
+ * @return Pointer to an NdbOperation object if successful, otherwise NULL.
+ */
+ NdbOperation* getNdbOperation(const char* aTableName);
+
+ /**
+ * Get an NdbOperation for index scan of a table.
+ * Note that the operation has to be defined before it is executed.
+ *
+ * @note All operations within the same transaction need to
+ * be initialized with this method.
+ *
+ * @param anIndexName The index name.
+ * @param aTableName The table name.
+ * @return Pointer to an NdbOperation object if successful, otherwise NULL.
+ */
+ NdbOperation* getNdbOperation(const char* anIndexName,
+ const char* aTableName);
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ /**
+ * Get an operation from NdbScanOperation idlelist and
+ * get the NdbConnection object which
+ * was fetched by startTransaction pointing to this operation.
+ * This operation will set the theTableId
+ * in the NdbOperation object.synchronous.
+ *
+ * @param aTableName a table name.
+ * @return pointer to an NdbOperation object if successful, otherwise NULL
+ */
+ NdbScanOperation* getNdbScanOperation(const char* aTableName);
+#endif
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ /**
+ * Get an operation from NdbScanOperation idlelist and
+ * get the NdbConnection object which
+ * was fetched by startTransaction pointing to this operation.
+ * This operation will set the theTableId
+ * in the NdbOperation object.synchronous.
+ *
+ * @param anIndexName The index name.
+ * @param aTableName a table name.
+ * @return pointer to an NdbOperation object if successful, otherwise NULL
+ */
+ NdbScanOperation* getNdbScanOperation(const char* anIndexName,
+ const char* aTableName);
+#endif
+
+
+ /**
+ * Get an operation from NdbIndexOperation idlelist and
+ * get the NdbConnection object that
+ * was fetched by startTransaction pointing to this operation.
+ * This operation will set the theTableId
+ * in the NdbOperation object. Synchronous.
+ *
+ * @param indexName An index name (as created by createIndex).
+ * @param tableName A table name.
+ * @return Pointer to an NdbIndexOperation object if
+ * successful, otherwise NULL
+ */
+ NdbIndexOperation* getNdbIndexOperation(const char* indexName,
+ const char* tableName);
+
+ /**
+ * @name Execute Transaction
+ * @{
+ */
+
+ /**
+ * Executes transaction.
+ *
+ * @param execType Execution type:<br>
+ * ExecType::NoCommit executes operations without
+ * committing them.<br>
+ * ExecType::Commit executes remaining operations and
+ * commits the complete transaction.<br>
+ * ExecType::Rollback rollbacks the entire transaction.
+ * @param abortOption Handling of error while excuting
+ * AbortOnError - Abort transaction if an operation fail
+ * IgnoreError - Accept failing operations
+ * @param force When operations should be sent to NDB Kernel.
+ * (See @ref secAdapt.)
+ * - 0: non-force, adaptive algorithm notices it
+ * (default);
+ * - 1: force send, adaptive algorithm notices it;
+ * - 2: non-force, adaptive algorithm do not notice
+ * the send.
+ * @return 0 if successful otherwise -1.
+ */
+ int execute(ExecType execType,
+ AbortOption abortOption = AbortOnError,
+ int force = 0 );
+
+ /**
+ * Prepare an asynchronous transaction.
+ *
+ * See @ref secAsync for more information on
+ * how to use this method.
+ *
+ * @param execType Execution type:<br>
+ * ExecType::NoCommit executes operations without committing them.<br>
+ * ExecType::Commit executes remaining operations and commits the
+ * complete transaction.<br>
+ * ExecType::Rollback rollbacks the entire transaction.
+ * @param callback A callback method. This method gets
+ * called when the transaction has been
+ * executed. See @ref ndbapi_example2.cpp
+ * for an example on how to specify and use
+ * a callback method.
+ * @param anyObject A void pointer. This pointer is forwarded to the
+ * callback method and can be used to give
+ * the callback method some data to work on.
+ * It is up to the application programmer
+ * to decide on the use of this pointer.
+ * @param abortOption see @ref execute
+ */
+ void executeAsynchPrepare(ExecType execType,
+ NdbAsynchCallback callback,
+ void* anyObject,
+ AbortOption abortOption = AbortOnError);
+
+ /**
+ * Prepare and send an asynchronous transaction.
+ *
+ * This method perform the same action as
+ * NdbConnection::executeAsynchPrepare
+ * but also sends the operations to the NDB kernel.
+ *
+ * See NdbConnection::executeAsynchPrepare for information
+ * about the parameters of this method.
+ *
+ * See @ref secAsync for more information on
+ * how to use this method.
+ */
+ void executeAsynch(ExecType aTypeOfExec,
+ NdbAsynchCallback aCallback,
+ void* anyObject,
+ AbortOption abortOption = AbortOnError);
+
+ /**
+ * Refresh
+ * Update timeout counter of this transaction
+ * in the database. If you want to keep the transaction
+ * active in the database longer than the
+ * transaction abort timeout.
+ * @note It's not advised to take a lock on a record and keep it
+ * for a extended time since this can impact other transactions.
+ *
+ */
+ int refresh();
+
+ /**
+ * Close transaction
+ * @note It is not allowed to call NdbConnection::close after sending the
+ * transaction asynchronously before the callback method has
+ * been called.
+ * (The application should keep track of the number of
+ * outstanding transactions and wait until all of them
+ * has completed before calling NdbConnection::close).
+ * If the transaction is not committed it will be aborted.
+ */
+ void close();
+
+ /** @} *********************************************************************/
+
+ /**
+ * @name Scan Transactions
+ * @{
+ */
+
+ /**
+ * Execute a scan transaction. This will define
+ * and start the scan transaction in the NDB kernel.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int executeScan();
+
+ /**
+ * Get the next tuple in a scan transaction.
+ *
+ * After each call to NdbConnection::nextScanResult
+ * the buffers and NdbRecAttr objects defined in
+ * NdbOperation::getValue are updated with values
+ * from the scanned tuple.
+ *
+ * @param fetchAllowed If set to false, then fetching is disabled
+ *
+ * The NDB API will contact the NDB Kernel for more tuples
+ * when necessary to do so unless you set the fetchAllowed
+ * to false.
+ * This will force NDB to process any records it
+ * already has in it's caches. When there are no more cached
+ * records it will return 2. You must then call nextScanResult
+ * with fetchAllowed = true in order to contact NDB for more
+ * records.
+ *
+ * fetchAllowed = false is useful when you want to update or
+ * delete all the records fetched in one transaction(This will save a
+ * lot of round trip time and make updates or deletes of scanned
+ * records a lot faster).
+ * While nextScanResult(false)
+ * returns 0 take over the record to another transaction. When
+ * nextScanResult(false) returns 2 you must execute and commit the other
+ * transaction. This will cause the locks to be transferred to the
+ * other transaction, updates or deletes will be made and then the
+ * locks will be released.
+ * After that, call nextScanResult(true) which will fetch new records and
+ * cache them in the NdbApi.
+ *
+ * @note If you don't take over the records to another transaction the
+ * locks on those records will be released the next time NDB Kernel
+ * is contacted for more records.
+ *
+ * @note Please contact for examples of efficient scan
+ * updates and deletes.
+ *
+ * @return
+ * - -1: if unsuccessful,<br>
+ * - 0: if another tuple was received, and<br>
+ * - 1: if there are no more tuples to scan.
+ * - 2: if there are no more cached records in NdbApi
+ */
+ int nextScanResult(bool fetchAllowed = true);
+
+ /**
+ * Stops the scan. Used if no more tuples are wanted.
+ * The transaction should still be closed with
+ * Ndb::closeTransaction.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int stopScan();
+
+ /**
+ * @name Meta Information
+ * @{
+ */
+
+ /**
+ * Get global checkpoint identity (GCI) of transaction.
+ *
+ * Each committed transaction belong to a GCI.
+ * The log for the committed transaction is saved on
+ * disk when a global checkpoint occurs.
+ *
+ * Whether or not the global checkpoint with this GCI has been
+ * saved on disk or not cannot be determined by this method.
+ *
+ * By comparing the GCI of a transaction with the value
+ * last GCI restored in a restarted NDB Cluster one can determine
+ * whether the transaction was restored or not.
+ *
+ * @note Global Checkpoint Identity is undefined for scan transactions
+ * (This is because no updates are performed in scan transactions.)
+ *
+ * @return GCI of transaction or -1 if GCI is not available.
+ * (Note that there has to be an NdbConnection::execute call
+ * with Ndb::Commit for the GCI to be available.)
+ */
+ int getGCI();
+
+ /**
+ * Get transaction identity.
+ *
+ * @return Transaction id.
+ */
+ Uint64 getTransactionId();
+
+ /**
+ * Returns the commit status of the transaction.
+ *
+ * @return The commit status of the transaction, i.e. one of
+ * { NotStarted, Started, TimeOut, Committed, Aborted, NeedAbort }
+ */
+ CommitStatusType commitStatus();
+
+ /** @} *********************************************************************/
+
+ /**
+ * @name Error Handling
+ * @{
+ */
+
+ /**
+ * Get error object with information about the latest error.
+ *
+ * @return An error object with information about the latest error.
+ */
+ const NdbError & getNdbError() const;
+
+ /**
+ * Get the latest NdbOperation which had an error.
+ * This method is used on the NdbConnection object to find the
+ * NdbOperation causing an error.
+ * To find more information about the
+ * actual error, use method NdbOperation::getNdbError
+ * on the returned NdbOperation object.
+ *
+ * @return The NdbOperation causing the latest error.
+ */
+ NdbOperation* getNdbErrorOperation();
+
+ /**
+ * Get the method number where the latest error occured.
+ *
+ * @return Line number where latest error occured.
+ */
+ int getNdbErrorLine();
+
+ /**
+ * Get completed (i.e. executed) operations of a transaction
+ *
+ * This method should only be used <em>after</em> a transaction
+ * has been executed.
+ * - NdbConnection::getNextCompletedOperation(NULL) returns the
+ * first NdbOperation object.
+ * - NdbConnection::getNextCompletedOperation(op) returns the
+ * NdbOperation object defined after the NdbOperation "op".
+ *
+ * This method is typically used to fetch all NdbOperation:s of
+ * a transaction to check for errors (use NdbOperation::getNdbError
+ * to fetch the NdbError object of an NdbOperation).
+ *
+ * @note This method should only be used after the transaction has been
+ * executed and before the transaction has been closed.
+ *
+ * @param op Operation, NULL means get first operation
+ * @return Operation "after" op
+ */
+ const NdbOperation * getNextCompletedOperation(const NdbOperation * op)const;
+
+ /**
+ * Release completed operations
+ */
+ void releaseCompletedOperations();
+
+
+ /** @} *********************************************************************/
+
+private:
+
+ typedef Uint64 TimeMillis_t;
+ /**************************************************************************
+ * These methods are service methods to other classes in the NDBAPI. *
+ **************************************************************************/
+
+ /**************************************************************************
+ * These are the create and delete methods of this class. *
+ **************************************************************************/
+
+ NdbConnection(Ndb* aNdb);
+
+ ~NdbConnection();
+
+ void init(); // Initialize connection object for new transaction
+
+ /**
+ * Set Connected node id
+ * and sequence no
+ */
+ void setConnectedNodeId( Uint32 nodeId, Uint32 sequence);
+
+ Uint32 getConnectedNodeId(); // Get Connected node id
+ void setMyBlockReference( int ); // Set my block refrerence
+ void setTC_ConnectPtr( Uint32 ); // Sets TC Connect pointer
+ int getTC_ConnectPtr(); // Gets TC Connect pointer
+ void setBuddyConPtr(Uint32); // Sets Buddy Con Ptr
+ Uint32 getBuddyConPtr(); // Gets Buddy Con Ptr
+ NdbConnection* next(); // Returns the next pointer
+ void next(NdbConnection*); // Sets the next pointer
+ ConStatusType Status(); // Read the status information
+ void Status(ConStatusType); // Set the status information
+ Uint32 get_send_size(); // Get size to send
+ void set_send_size(Uint32); // Set size to send;
+
+ int receiveDIHNDBTAMPER(NdbApiSignal* anApiSignal);
+ int receiveTCSEIZECONF(NdbApiSignal* anApiSignal);
+ int receiveTCSEIZEREF(NdbApiSignal* anApiSignal);
+ int receiveTCRELEASECONF(NdbApiSignal* anApiSignal);
+ int receiveTCRELEASEREF(NdbApiSignal* anApiSignal);
+ int receiveTC_COMMITCONF(const class TcCommitConf *);
+ int receiveTCKEYCONF(const class TcKeyConf *, Uint32 aDataLength);
+ int receiveTCKEY_FAILCONF(const class TcKeyFailConf *);
+ int receiveTCKEY_FAILREF(NdbApiSignal* anApiSignal);
+ int receiveTC_COMMITREF(NdbApiSignal* anApiSignal);
+ int receiveTCROLLBACKCONF(NdbApiSignal* anApiSignal); // Rec TCPREPARECONF ?
+ int receiveTCROLLBACKREF(NdbApiSignal* anApiSignal); // Rec TCPREPAREREF ?
+ int receiveTCROLLBACKREP(NdbApiSignal* anApiSignal);
+ int receiveTCINDXCONF(const class TcIndxConf *, Uint32 aDataLength);
+ int receiveTCINDXREF(NdbApiSignal*);
+ int receiveSCAN_TABREF(NdbApiSignal*);
+ int receiveSCAN_TABCONF(NdbApiSignal*);
+ int receiveSCAN_TABINFO(NdbApiSignal*);
+
+ int checkNextScanResultComplete();
+ int sendScanStart();
+ int sendScanNext(bool stopScanFlag);
+ int fetchNextScanResult();
+
+ int doSend(); // Send all operations
+ int sendROLLBACK(); // Send of an ROLLBACK
+ int sendTC_HBREP(); // Send a TCHBREP signal;
+ int sendCOMMIT(); // Send a TC_COMMITREQ signal;
+ void setGCI(int GCI); // Set the global checkpoint identity
+
+ int OpCompleteFailure(); // Operation Completed with success
+ int OpCompleteSuccess(); // Operation Completed with success
+
+ void CompletedOperations(); // Move active ops to list of completed
+
+ void OpSent(); // Operation Sent with success
+
+ // Free connection related resources and close transaction
+ void release();
+
+ // Release all operations in connection
+ void releaseOperations();
+
+ // Release all cursor operations in connection
+ void releaseOps(NdbOperation*);
+ void releaseCursorOperations(NdbCursorOperation*);
+
+ // Set the transaction identity of the transaction
+ void setTransactionId(Uint64 aTransactionId);
+
+ // Indicate something went wrong in the definition phase
+ void setErrorCode(int anErrorCode);
+
+ // Indicate something went wrong in the definition phase
+ void setOperationErrorCode(int anErrorCode);
+
+ // Indicate something went wrong in the definition phase
+ void setOperationErrorCodeAbort(int anErrorCode);
+
+ int checkMagicNumber(); // Verify correct object
+ NdbOperation* getNdbOperation(class NdbTableImpl* aTable);
+ NdbScanOperation* getNdbScanOperation(class NdbTableImpl* aTable);
+ NdbIndexOperation* getNdbIndexOperation(class NdbIndexImpl* anIndex,
+ class NdbTableImpl* aTable);
+
+ void handleExecuteCompletion();
+
+ /****************************************************************************
+ * These are the private variables of this class.
+ ****************************************************************************/
+
+ Uint32 ptr2int();
+ Uint32 theId;
+
+ // Keeps track of what the send method should do.
+ SendStatusType theSendStatus;
+ NdbAsynchCallback theCallbackFunction; // Pointer to the callback function
+ void* theCallbackObject; // The callback object pointer
+ Uint32 theTransArrayIndex; // Current index in a transaction
+ // array for this object
+ TimeMillis_t theStartTransTime; // Start time of the transaction
+
+ NdbError theError; // Errorcode on transaction
+ int theErrorLine; // Method number of last error in NdbOperation
+ NdbOperation* theErrorOperation; // The NdbOperation where the error occurred
+
+ Ndb* theNdb; // Pointer to Ndb object
+ NdbConnection* theNext; // Next pointer. Used in idle list.
+
+ NdbOperation* theFirstOpInList; // First operation in defining list.
+ NdbOperation* theLastOpInList; // Last operation in defining list.
+
+ NdbOperation* theFirstExecOpInList; // First executing operation in list
+ NdbOperation* theLastExecOpInList; // Last executing operation in list.
+
+
+ NdbOperation* theCompletedFirstOp; // First operation in completed
+ // operation list.
+
+ Uint32 theNoOfOpSent; // How many operations have been sent
+ Uint32 theNoOfOpCompleted; // How many operations have completed
+ Uint32 theNoOfOpFetched; // How many operations was actually fetched
+ Uint32 theNoOfSCANTABCONFRecv; // How many SCAN_TABCONF have been received
+ Uint32 theMyRef; // Our block reference
+ Uint32 theTCConPtr; // Transaction Co-ordinator connection pointer.
+ Uint64 theTransactionId; // theTransactionId of the transaction
+ Uint32 theGlobalCheckpointId; // The gloabl checkpoint identity of the transaction
+ ConStatusType theStatus; // The status of the connection
+
+ CompletionStatus theCompletionStatus; // The Completion status of the transaction
+ CommitStatusType theCommitStatus; // The commit status of the transaction
+ Uint32 theMagicNumber; // Magic Number to verify correct object
+
+ Uint32 thePriority; // Transaction Priority
+ ReturnType theReturnStatus; // Did we have any read/update/delete failing
+ // to find the tuple.
+ bool theTransactionIsStarted;
+ bool theInUseState;
+ bool theSimpleState;
+ Uint8 m_abortOption; // Type of commit
+
+ ListState theListState;
+
+ Uint32 theDBnode; // The database node we are connected to
+ Uint32 theNodeSequence; // The sequence no of the db node
+ bool theReleaseOnClose;
+
+ // Cursor operations
+ bool m_waitForReply;
+ NdbCursorOperation* m_theFirstCursorOperation;
+ NdbCursorOperation* m_theLastCursorOperation;
+
+ NdbCursorOperation* m_firstExecutedCursorOp;
+ // Scan operations
+ bool theScanFinished;
+
+ NdbScanReceiver* theCurrentScanRec; // The current operation to
+ // distribute to the app.
+ NdbScanReceiver* thePreviousScanRec; // The previous operation read by
+ // nextScanResult.
+ NdbOperation* theScanningOp; // The operation actually performing the scan
+ Uint32 theBuddyConPtr;
+
+ static void sendTC_COMMIT_ACK(NdbApiSignal *,
+ Uint32 transId1, Uint32 transId2,
+ Uint32 aBlockRef);
+
+ void completedFail(const char * s);
+#ifdef VM_TRACE
+ void printState();
+#endif
+};
+
+inline
+Uint32
+NdbConnection::get_send_size()
+{
+ return 0;
+}
+
+inline
+void
+NdbConnection::set_send_size(Uint32 send_size)
+{
+ return;
+}
+
+inline
+int
+NdbConnection::checkMagicNumber()
+{
+ if (theMagicNumber == 0x37412619)
+ return 0;
+ else {
+#ifdef NDB_NO_DROPPED_SIGNAL
+ abort();
+#endif
+ return -1;
+ }
+}
+
+/************************************************************************************************
+void setTransactionId(Uint64 aTransactionId);
+
+Remark: Set the transaction identity.
+************************************************************************************************/
+inline
+void
+NdbConnection::setTransactionId(Uint64 aTransactionId)
+{
+ theTransactionId = aTransactionId;
+}
+
+inline
+void
+NdbConnection::setConnectedNodeId(Uint32 aNode, Uint32 aSequenceNo)
+{
+ theDBnode = aNode;
+ theNodeSequence = aSequenceNo;
+}
+/******************************************************************************
+int getConnectedNodeId();
+
+Return Value: Return theDBnode.
+Remark: Get Connected node id.
+******************************************************************************/
+inline
+Uint32
+NdbConnection::getConnectedNodeId()
+{
+ return theDBnode;
+}
+/******************************************************************************
+void setMyBlockReference(int aBlockRef);
+
+Parameters: aBlockRef: The block refrerence.
+Remark: Set my block refrerence.
+******************************************************************************/
+inline
+void
+NdbConnection::setMyBlockReference(int aBlockRef)
+{
+ theMyRef = aBlockRef;
+}
+/******************************************************************************
+void setTC_ConnectPtr(Uint32 aTCConPtr);
+
+Parameters: aTCConPtr: The connection pointer.
+Remark: Sets TC Connect pointer.
+******************************************************************************/
+inline
+void
+NdbConnection::setTC_ConnectPtr(Uint32 aTCConPtr)
+{
+ theTCConPtr = aTCConPtr;
+}
+
+/******************************************************************************
+int getTC_ConnectPtr();
+
+Return Value: Return theTCConPtr.
+Remark: Gets TC Connect pointer.
+******************************************************************************/
+inline
+int
+NdbConnection::getTC_ConnectPtr()
+{
+ return theTCConPtr;
+}
+
+inline
+void
+NdbConnection::setBuddyConPtr(Uint32 aBuddyConPtr)
+{
+ theBuddyConPtr = aBuddyConPtr;
+}
+
+inline
+Uint32 NdbConnection::getBuddyConPtr()
+{
+ return theBuddyConPtr;
+}
+
+/******************************************************************************
+NdbConnection* next();
+
+inline
+void
+NdbConnection::setBuddyConPtr(Uint32 aBuddyConPtr)
+{
+ theBuddyConPtr = aBuddyConPtr;
+}
+
+inline
+Uint32 NdbConnection::getBuddyConPtr()
+{
+ return theBuddyConPtr;
+}
+
+Return Value: Return next pointer to NdbConnection object.
+Remark: Get the next pointer.
+******************************************************************************/
+inline
+NdbConnection*
+NdbConnection::next()
+{
+ return theNext;
+}
+
+/******************************************************************************
+void next(NdbConnection aConnection);
+
+Parameters: aConnection: The connection object.
+Remark: Sets the next pointer.
+******************************************************************************/
+inline
+void
+NdbConnection::next(NdbConnection* aConnection)
+{
+ theNext = aConnection;
+}
+
+/******************************************************************************
+ConStatusType Status();
+
+Return Value Return the ConStatusType.
+Parameters: aStatus: The status.
+Remark: Sets Connect status.
+******************************************************************************/
+inline
+ConStatusType
+NdbConnection::Status()
+{
+ return theStatus;
+}
+
+/******************************************************************************
+void Status(ConStatusType aStatus);
+
+Parameters: aStatus: The status.
+Remark: Sets Connect status.
+******************************************************************************/
+inline
+void
+NdbConnection::Status( ConStatusType aStatus )
+{
+ theStatus = aStatus;
+}
+
+/******************************************************************************
+ void setGCI();
+
+Remark: Set global checkpoint identity of the transaction
+******************************************************************************/
+inline
+void
+NdbConnection::setGCI(int aGlobalCheckpointId)
+{
+ theGlobalCheckpointId = aGlobalCheckpointId;
+}
+
+/******************************************************************************
+void OpSent();
+
+Remark: An operation was sent with success that expects a response.
+******************************************************************************/
+inline
+void
+NdbConnection::OpSent()
+{
+ theNoOfOpSent++;
+}
+
+inline
+Uint32
+NdbConnection::ptr2int(){
+ return theId;
+}
+
+#endif
+
+
diff --git a/ndb/include/ndbapi/NdbCursorOperation.hpp b/ndb/include/ndbapi/NdbCursorOperation.hpp
new file mode 100644
index 00000000000..cd76b045ea2
--- /dev/null
+++ b/ndb/include/ndbapi/NdbCursorOperation.hpp
@@ -0,0 +1,93 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbCursorOperation_H
+#define NdbCursorOperation_H
+
+#include <NdbOperation.hpp>
+
+class NdbResultSet;
+
+/**
+ * @class NdbCursorOperation
+ * @brief Operation using cursors
+ */
+class NdbCursorOperation : public NdbOperation
+{
+ friend class NdbResultSet;
+ friend class NdbConnection;
+
+public:
+ /**
+ * Type of cursor
+ */
+ enum CursorType {
+ NoCursor = 0,
+ ScanCursor = 1,
+ IndexCursor = 2
+ };
+
+ /**
+ * Lock when performing scan
+ */
+ enum LockMode {
+ LM_Read = 0,
+ LM_Exclusive = 1,
+ LM_CommittedRead = 2,
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ LM_Dirty = 2
+#endif
+ };
+
+ virtual CursorType cursorType() = 0;
+
+ /**
+ * readTuples returns a NdbResultSet where tuples are stored.
+ * Tuples are not stored in NdbResultSet until execute(NoCommit)
+ * has been executed and nextResult has been called.
+ *
+ * @param parallel Scan parallelism
+ * @param LockMode Scan lock handling
+ * @returns NdbResultSet.
+ */
+ virtual NdbResultSet* readTuples(unsigned parallel = 0,
+ LockMode = LM_Read ) = 0;
+
+ inline NdbResultSet* readTuplesExclusive(int parallell = 0){
+ return readTuples(parallell, LM_Exclusive);
+ }
+
+protected:
+ NdbCursorOperation(Ndb* aNdb);
+
+ ~NdbCursorOperation();
+
+ void cursInit();
+
+ virtual int executeCursor(int ProcessorId) = 0;
+
+ NdbResultSet* getResultSet();
+ NdbResultSet* m_resultSet;
+
+private:
+
+ virtual int nextResult(bool fetchAllowed) = 0;
+
+ virtual void closeScan() = 0;
+};
+
+
+#endif
diff --git a/ndb/include/ndbapi/NdbDictionary.hpp b/ndb/include/ndbapi/NdbDictionary.hpp
new file mode 100644
index 00000000000..a8a3bc86786
--- /dev/null
+++ b/ndb/include/ndbapi/NdbDictionary.hpp
@@ -0,0 +1,1033 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbDictionary.hpp
+ * Include:
+ * Link:
+ * Author: Jonas Oreland
+ * Date: 2003-05-14
+ * Version: 0.1
+ * Description: Data dictionary support
+ * Documentation:
+ * Adjust: 2003-05-14 Jonas Oreland First version.
+ ****************************************************************************/
+
+#ifndef NdbDictionary_H
+#define NdbDictionary_H
+
+#include <ndb_types.h>
+
+class Ndb;
+
+/**
+ * @class NdbDictionary
+ * @brief Data dictionary class
+ *
+ * This class supports all schema data definition and enquiry such as:
+ * -# Creating tables (Dictionary::createTable) and table columns
+ * -# Dropping tables (Dictionary::dropTable)
+ * -# Creating secondary indexes (Dictionary::createIndex)
+ * -# Dropping secondary indexes (Dictionary::dropIndex)
+ * -# Enquiries about tables
+ * (Dictionary::getTable, Table::getNoOfColumns,
+ * Table::getPrimaryKey, and Table::getNoOfPrimaryKeys)
+ * -# Enquiries about indexes
+ * (Dictionary::getIndex, Index::getNoOfColumns,
+ * and Index::getColumn)
+ *
+ * NdbDictionary has several help (inner) classes:
+ * -# NdbDictionary::Table for creating tables
+ * -# NdbDictionary::Column for creating table columns
+ * -# NdbDictionary::Index for creating secondary indexes
+ *
+ * See @ref ndbapi_example4.cpp for details of usage.
+ */
+class NdbDictionary {
+public:
+ /**
+ * @class Object
+ * @brief Meta information about a database object (a table, index, etc)
+ */
+ class Object {
+ public:
+ /**
+ * Status of object
+ */
+ enum Status {
+ New, ///< The object only exist in memory and
+ ///< has not been created in the NDB Kernel
+ Changed, ///< The object has been modified in memory
+ ///< and has to be commited in NDB Kernel for
+ ///< changes to take effect
+ Retrieved ///< The object exist and has been read
+ ///< into main memory from NDB Kernel
+ };
+
+ /**
+ * Get status of object
+ */
+ virtual Status getObjectStatus() const = 0;
+
+ /**
+ * Get version of object
+ */
+ virtual int getObjectVersion() const = 0;
+
+ /**
+ * Object type
+ */
+ enum Type {
+ TypeUndefined = 0, ///< Undefined
+ SystemTable = 1, ///< System table
+ UserTable = 2, ///< User table (may be temporary)
+ UniqueHashIndex = 3, ///< Unique un-ordered hash index
+ HashIndex = 4, ///< Non-unique un-ordered hash index
+ UniqueOrderedIndex = 5, ///< Unique ordered index
+ OrderedIndex = 6, ///< Non-unique ordered index
+ HashIndexTrigger = 7, ///< Index maintenance, internal
+ IndexTrigger = 8, ///< Index maintenance, internal
+ SubscriptionTrigger = 9,///< Backup or replication, internal
+ ReadOnlyConstraint = 10 ///< Trigger, internal
+ };
+
+ /**
+ * Object state
+ */
+ enum State {
+ StateUndefined = 0, ///< Undefined
+ StateOffline = 1, ///< Offline, not usable
+ StateBuilding = 2, ///< Building, not yet usable
+ StateDropping = 3, ///< Offlining or dropping, not usable
+ StateOnline = 4, ///< Online, usable
+ StateBroken = 9 ///< Broken, should be dropped and re-created
+ };
+
+ /**
+ * Object store
+ */
+ enum Store {
+ StoreUndefined = 0, ///< Undefined
+ StoreTemporary = 1, ///< Object or data deleted on system restart
+ StorePermanent = 2 ///< Permanent. logged to disk
+ };
+
+ /**
+ * Type of fragmentation.
+ *
+ * This parameter specifies how data in the table or index will
+ * be distributed among the db nodes in the cluster.<br>
+ * The bigger the table the more number of fragments should be used.
+ * Note that all replicas count as same "fragment".<br>
+ * For a table, default is FragAllMedium. For a unique hash index,
+ * default is taken from underlying table and cannot currently
+ * be changed.
+ */
+ enum FragmentType {
+ FragUndefined = 0, ///< Fragmentation type undefined or default
+ FragSingle = 1, ///< Only one fragment
+ FragAllSmall = 2, ///< One fragment per node group
+ FragAllMedium = 3, ///< Default value. Two fragments per node group.
+ FragAllLarge = 4 ///< Eight fragments per node group.
+ };
+ };
+
+ /**
+ * @class Column
+ * @brief Represents an column in an NDB Cluster table
+ *
+ * Each column has a type. The type of a column is determind by a number
+ * of type specifiers.
+ * The type specifiers are:
+ * - Builtin type
+ * - Array length or max length
+ * - Precision and scale
+ */
+ class Column {
+ public:
+ /**
+ * The builtin column types
+ */
+ enum Type {
+ Undefined=0,///< Undefined
+ Tinyint, ///< 8 bit. 1 byte signed integer, can be used in array
+ Tinyunsigned, ///< 8 bit. 1 byte unsigned integer, can be used in array
+ Smallint, ///< 16 bit. 2 byte signed integer, can be used in array
+ Smallunsigned, ///< 16 bit. 2 byte unsigned integer, can be used in array
+ Mediumint, ///< 24 bit. 3 byte signed integer, can be used in array
+ Mediumunsigned,///< 24 bit. 3 byte unsigned integer, can be used in array
+ Int, ///< 32 bit. 4 byte signed integer, can be used in array
+ Unsigned, ///< 32 bit. 4 byte unsigned integer, can be used in array
+ Bigint, ///< 64 bit. 8 byte signed integer, can be used in array
+ Bigunsigned, ///< 64 Bit. 8 byte signed integer, can be used in array
+ Float, ///< 32-bit float. 4 bytes float, can be used in array
+ Double, ///< 64-bit float. 8 byte float, can be used in array
+ Decimal, ///< Precision, Scale are applicable
+ Char, ///< Len. A fixed array of 1-byte chars
+ Varchar, ///< Max len
+ Binary, ///< Len
+ Varbinary, ///< Max len
+ Datetime, ///< Precision down to 1 sec (sizeof(Datetime) == 8 bytes )
+ Timespec, ///< Precision down to 1 nsec(sizeof(Datetime) == 12 bytes )
+ Blob ///< Binary large object (see NdbBlob)
+ };
+
+ /**
+ * @name General
+ * @{
+ */
+ /**
+ * Constructor
+ * @param name Name of column
+ */
+ Column(const char * name = "");
+ /**
+ * Copy constructor
+ * @param column Column to be copied
+ */
+ Column(const Column& column);
+ ~Column();
+
+ /**
+ * Set name of column
+ * @param name Name of the column
+ */
+ void setName(const char * name);
+
+ /**
+ * Get name of column
+ * @return Name of the column
+ */
+ const char* getName() const;
+
+ /**
+ * Set whether column is nullable or not
+ */
+ void setNullable(bool);
+
+ /**
+ * Get if the column is nullable or not
+ */
+ bool getNullable() const;
+
+ /**
+ * Set that column is part of primary key
+ */
+ void setPrimaryKey(bool);
+
+ /**
+ * Check if column is part of primary key
+ */
+ bool getPrimaryKey() const;
+
+ /**
+ * Get number of column (horizontal position within table)
+ */
+ int getColumnNo() const;
+
+ /**
+ * Check if column is equal to some other column
+ * @param column Column to compare with
+ * @return true if column is equal to some other column otherwise false.
+ */
+ bool equal(const Column& column) const;
+
+ /** @} *******************************************************************/
+ /**
+ * @name Type Specifiers
+ * @{
+ */
+
+ /**
+ * Set type of column
+ * @param type Type of column
+ */
+ void setType(Type type);
+
+ /**
+ * Get type of column
+ */
+ Type getType() const;
+
+ /**
+ * Set precision of column.
+ * @note Only applicable for builtin type Decimal
+ */
+ void setPrecision(int);
+
+ /**
+ * Get precision of column.
+ * @note Only applicable for builtin type Decimal
+ */
+ int getPrecision() const;
+
+ /**
+ * Set scale of column.
+ * @note Only applicable for builtin type Decimal
+ */
+ void setScale(int);
+
+ /**
+ * Get scale of column.
+ * @note Only applicable for builtin type Decimal
+ */
+ int getScale() const;
+
+ /**
+ * Set length for column
+ * Array length for column or max length for variable length arrays.
+ */
+ void setLength(int length);
+
+ /**
+ * Get length for column
+ * Array length for column or max length for variable length arrays.
+ */
+ int getLength() const;
+
+ /**
+ * Set distribution key
+ *
+ * A <em>distribution key</em> is a set of attributes which are used
+ * to distribute the tuples onto the NDB nodes.
+ * The distribution key uses the NDB Cluster hashing function.
+ *
+ * An example where this is useful is TPC-C where it might be
+ * good to use the warehouse id and district id as the distribution key.
+ * This would place all data for a specific district and warehouse
+ * in the same database node.
+ *
+ * Locally in the fragments the full primary key
+ * will still be used with the hashing algorithm.
+ *
+ * @param enable If set to true, then the column will be part of
+ * the distribution key.
+ */
+ void setDistributionKey(bool enable);
+
+ /**
+ * Check if column is part of distribution key
+ * @see setDistributionKey
+ */
+ bool getDistributionKey() const;
+ /** @} *******************************************************************/
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ void setTupleKey(bool);
+ bool getTupleKey() const;
+
+ void setDistributionGroup(bool, int bits = 16);
+ bool getDistributionGroup() const;
+ int getDistributionGroupBits() const;
+
+ void setIndexOnlyStorage(bool);
+ bool getIndexOnlyStorage() const;
+
+ /**
+ * @name ODBC Specific methods
+ * @{
+ */
+ void setAutoIncrement(bool);
+ bool getAutoIncrement() const;
+ void setAutoIncrementInitialValue(Uint64 val);
+ void setDefaultValue(const char*);
+ const char* getDefaultValue() const;
+ /** @} *******************************************************************/
+#endif
+
+ private:
+ friend class NdbColumnImpl;
+ class NdbColumnImpl & m_impl;
+ Column(NdbColumnImpl&);
+ Column& operator=(const Column&);
+ };
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ /**
+ * ???
+ */
+ typedef Column Attribute;
+#endif
+
+ /**
+ * @brief Represents a table in NDB Cluster
+ *
+ * <em>TableSize</em><br>
+ * When calculating the data storage one should add the size of all
+ * attributes (each attributeconsumes at least 4 bytes) and also an overhead
+ * of 12 byte. Variable size attributes (not supported yet) will have a
+ * size of 12 bytes plus the actual data storage parts where there is an
+ * additional overhead based on the size of the variable part.<br>
+ * An example table with 5 attributes:
+ * one 64 bit attribute, one 32 bit attribute,
+ * two 16 bit attributes and one array of 64 8 bits.
+ * This table will consume
+ * 12 (overhead) + 8 + 4 + 2*4 (4 is minimum) + 64 = 96 bytes per record.
+ * Additionally an overhead of about 2 % as page headers and waste should
+ * be allocated. Thus, 1 million records should consume 96 MBytes
+ * plus the overhead 2 MByte and rounded up to 100 000 kBytes.<br>
+ *
+ */
+ class Table : public Object {
+ public:
+ /**
+ * @name General
+ * @{
+ */
+ /**
+ * Constructor
+ * @param name Name of table
+ */
+ Table(const char * name = "");
+
+ /**
+ * Copy constructor
+ * @param table Table to be copied
+ */
+ Table(const Table& table);
+ virtual ~Table();
+
+ /**
+ * Assignment operator, deep copy
+ * @param table Table to be copied
+ */
+ Table& operator=(const Table&);
+
+ /**
+ * Name of table
+ * @param name Name of table
+ */
+ void setName(const char * name);
+
+ /**
+ * Get table name
+ */
+ const char * getName() const;
+
+ /**
+ * Get table id
+ */
+ int getTableId() const;
+
+ /**
+ * Add a column definition to a table
+ * @note creates a copy
+ */
+ void addColumn(const Column &);
+
+ /**
+ * Get column definition via name.
+ * @return null if none existing name
+ */
+ const Column* getColumn(const char * name) const;
+
+ /**
+ * Get column definition via index in table.
+ * @return null if none existing name
+ */
+ const Column* getColumn(const int attributeId) const;
+
+ /** @} *******************************************************************/
+ /**
+ * @name Storage
+ * @{
+ */
+
+ /**
+ * If set to false, then the table is a temporary
+ * table and is not logged to disk.
+ *
+ * In case of a system restart the table will still
+ * be defined and exist but will be empty.
+ * Thus no checkpointing and no logging is performed on the table.
+ *
+ * The default value is true and indicates a normal table
+ * with full checkpointing and logging activated.
+ */
+ void setLogging(bool);
+
+ /**
+ * @see NdbDictionary::Table::setLogging.
+ */
+ bool getLogging() const;
+
+ /**
+ * Set fragmentation type
+ */
+ void setFragmentType(FragmentType);
+
+ /**
+ * Get fragmentation type
+ */
+ FragmentType getFragmentType() const;
+
+ /**
+ * Set KValue (Hash parameter.)
+ * Only allowed value is 6.
+ * Later implementations might add flexibility in this parameter.
+ */
+ void setKValue(int kValue);
+
+ /**
+ * Get KValue (Hash parameter.)
+ * Only allowed value is 6.
+ * Later implementations might add flexibility in this parameter.
+ */
+ int getKValue() const;
+
+ /**
+ * Set MinLoadFactor (Hash parameter.)
+ * This value specifies the load factor when starting to shrink
+ * the hash table.
+ * It must be smaller than MaxLoadFactor.
+ * Both these factors are given in percentage.
+ */
+ void setMinLoadFactor(int);
+
+ /**
+ * Get MinLoadFactor (Hash parameter.)
+ * This value specifies the load factor when starting to shrink
+ * the hash table.
+ * It must be smaller than MaxLoadFactor.
+ * Both these factors are given in percentage.
+ */
+ int getMinLoadFactor() const;
+
+ /**
+ * Set MaxLoadFactor (Hash parameter.)
+ * This value specifies the load factor when starting to split
+ * the containers in the local hash tables.
+ * 100 is the maximum which will optimize memory usage.
+ * A lower figure will store less information in each container and thus
+ * find the key faster but consume more memory.
+ */
+ void setMaxLoadFactor(int);
+
+ /**
+ * Get MaxLoadFactor (Hash parameter.)
+ * This value specifies the load factor when starting to split
+ * the containers in the local hash tables.
+ * 100 is the maximum which will optimize memory usage.
+ * A lower figure will store less information in each container and thus
+ * find the key faster but consume more memory.
+ */
+ int getMaxLoadFactor() const;
+
+ /** @} *******************************************************************/
+ /**
+ * @name Other
+ * @{
+ */
+
+ /**
+ * Get number of columns in the table
+ */
+ int getNoOfColumns() const;
+
+ /**
+ * Get number of primary keys in the table
+ */
+ int getNoOfPrimaryKeys() const;
+
+ /**
+ * Get name of primary key
+ */
+ const char* getPrimaryKey(int no) const;
+
+ /**
+ * Check if table is equal to some other table
+ */
+ bool equal(const Table&) const;
+
+ /**
+ * Get frm file stored with this table
+ */
+ const void* getFrmData() const;
+ Uint32 getFrmLength() const;
+
+ /**
+ * Set frm file to store with this table
+ */
+ void setFrm(const void* data, Uint32 len);
+
+ /**
+ * Set table object type
+ */
+ void setObjectType(Object::Type type);
+
+ /**
+ * Get table object type
+ */
+ Object::Type getObjectType() const;
+
+ /**
+ * Get object status
+ */
+ virtual Object::Status getObjectStatus() const;
+
+ /**
+ * Get object version
+ */
+ virtual int getObjectVersion() const;
+
+ /** @} *******************************************************************/
+
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+ void setStoredTable(bool x) { setLogging(x); }
+ bool getStoredTable() const { return getLogging(); }
+
+ int getRowSizeInBytes() const ;
+ int createTableInDb(Ndb*, bool existingEqualIsOk = true) const ;
+#endif
+
+ private:
+ friend class NdbTableImpl;
+ class NdbTableImpl & m_impl;
+ Table(NdbTableImpl&);
+ };
+
+ /**
+ * @class Index
+ * @brief Represents an index in an NDB Cluster
+ */
+ class Index : public Object {
+ public:
+ /**
+ * Constructor
+ * @param name Name of index
+ */
+ Index(const char * name = "");
+ virtual ~Index();
+
+ /**
+ * Set the name of an index
+ */
+ void setName(const char * name);
+
+ /**
+ * Get the name of an index
+ */
+ const char * getName() const;
+
+ /**
+ * Define the name of the table to be indexed
+ */
+ void setTable(const char * name);
+
+ /**
+ * Get the name of the table being indexed
+ */
+ const char * getTable() const;
+
+ /**
+ * Get the number of columns in the index
+ */
+ unsigned getNoOfColumns() const;
+
+ /**
+ * Get the number of columns in the index
+ * Depricated, use getNoOfColumns instead.
+ */
+ int getNoOfIndexColumns() const;
+
+ /**
+ * Get a specific column in the index
+ */
+ const NdbDictionary::Column * getColumn(unsigned no) const ;
+
+ /**
+ * Get a specific column name in the index
+ * Depricated, use getColumn instead.
+ */
+ const char * getIndexColumn(int no) const ;
+
+ /**
+ * Add a column to the index definition
+ * Note that the order of columns will be in
+ * the order they are added (only matters for ordered indexes).
+ */
+ void addColumn(const Column & c);
+
+ /**
+ * Add a column name to the index definition
+ * Note that the order of indexes will be in
+ * the order they are added (only matters for ordered indexes).
+ */
+ void addColumnName(const char * name);
+
+ /**
+ * Add a column name to the index definition
+ * Note that the order of indexes will be in
+ * the order they are added (only matters for ordered indexes).
+ * Depricated, use addColumnName instead.
+ */
+ void addIndexColumn(const char * name);
+
+ /**
+ * Add several column names to the index definition
+ * Note that the order of indexes will be in
+ * the order they are added (only matters for ordered indexes).
+ */
+ void addColumnNames(unsigned noOfNames, const char ** names);
+
+ /**
+ * Add several column names to the index definition
+ * Note that the order of indexes will be in
+ * the order they are added (only matters for ordered indexes).
+ * Depricated, use addColumnNames instead.
+ */
+ void addIndexColumns(int noOfNames, const char ** names);
+
+ /**
+ * Represents type of index
+ */
+ enum Type {
+ Undefined = 0, ///< Undefined object type (initial value)
+ UniqueHashIndex = 3, ///< Unique un-ordered hash index
+ ///< (only one currently supported)
+ HashIndex = 4, ///< Non-unique un-ordered hash index
+ UniqueOrderedIndex = 5, ///< Unique ordered index
+ OrderedIndex = 6 ///< Non-unique ordered index
+ };
+
+ /**
+ * Set index type of the index
+ */
+ void setType(Type type);
+
+ /**
+ * Get index type of the index
+ */
+ Type getType() const;
+
+ /**
+ * Enable/Disable index storage on disk
+ *
+ * @param enable If enable is set to true, then logging becomes enabled
+ *
+ * @see NdbDictionary::Table::setLogging
+ *
+ * @note Non-logged indexes are rebuilt at system restart.
+ * @note Ordered index does not currently support logging.
+ */
+ void setLogging(bool enable);
+
+ /**
+ * Check if index is set to be stored on disk
+ *
+ * @see NdbDictionary::Index::setLogging
+ */
+ bool getLogging() const;
+
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+ void setStoredIndex(bool x) { setLogging(x); }
+ bool getStoredIndex() const { return getLogging(); }
+#endif
+
+ /**
+ * Get object status
+ */
+ virtual Object::Status getObjectStatus() const;
+
+ /**
+ * Get object version
+ */
+ virtual int getObjectVersion() const;
+
+ private:
+ friend class NdbIndexImpl;
+
+ class NdbIndexImpl & m_impl;
+ Index(NdbIndexImpl&);
+ };
+
+ /**
+ * @brief Represents an Event in NDB Cluster
+ *
+ */
+ class Event : public Object {
+ public:
+ enum TableEvent { TE_INSERT=1, TE_DELETE=2, TE_UPDATE=4, TE_ALL=7 };
+ enum EventDurability {
+ ED_UNDEFINED = 0,
+#if 0 // not supported
+ ED_SESSION = 1,
+ // Only this API can use it
+ // and it's deleted after api has disconnected or ndb has restarted
+
+ ED_TEMPORARY = 2,
+ // All API's can use it,
+ // But's its removed when ndb is restarted
+#endif
+ ED_PERMANENT = 3
+ // All API's can use it,
+ // It's still defined after a restart
+ };
+
+ Event(const char *name);
+ virtual ~Event();
+ void setName(const char *);
+ void setTable(const char *);
+ void addTableEvent(const TableEvent);
+ void setDurability(const EventDurability);
+ void addColumn(const Column &c);
+ void addEventColumn(unsigned attrId);
+ void addEventColumn(const char * columnName);
+ void addEventColumns(int n, const char ** columnNames);
+
+ /**
+ * Get object status
+ */
+ virtual Object::Status getObjectStatus() const;
+
+ /**
+ * Get object version
+ */
+ virtual int getObjectVersion() const;
+
+ void print();
+
+ private:
+ friend class NdbEventImpl;
+ friend class NdbEventOperationImpl;
+ class NdbEventImpl & m_impl;
+ Event(NdbEventImpl&);
+ };
+
+ /**
+ * @class Dictionary
+ * @brief Dictionary for defining and retreiving meta data
+ */
+ class Dictionary {
+ public:
+ /**
+ * @class List
+ * @brief Structure for retrieving lists of object names
+ */
+ struct List {
+ /**
+ * @struct Element
+ * @brief Object to be stored in an NdbDictionary::Dictionary::List
+ */
+ struct Element {
+ unsigned id; ///< Id of object
+ Object::Type type; ///< Type of object
+ Object::State state; ///< State of object
+ Object::Store store; ///< How object is stored
+ char * database; ///< In what database the object resides
+ char * schema; ///< What schema the object is defined in
+ char * name; ///< Name of object
+ Element() :
+ id(0),
+ type(Object::TypeUndefined),
+ state(Object::StateUndefined),
+ store(Object::StoreUndefined),
+ database(0),
+ schema(0),
+ name(0) {
+ }
+ };
+ unsigned count; ///< Number of elements in list
+ Element * elements; ///< Pointer to array of elements
+ List() : count(0), elements(0) {}
+ ~List() {
+ if (elements != 0) {
+ for (unsigned i = 0; i < count; i++) {
+ delete[] elements[i].database;
+ delete[] elements[i].schema;
+ delete[] elements[i].name;
+ elements[i].name = 0;
+ }
+ delete[] elements;
+ count = 0;
+ elements = 0;
+ }
+ }
+ };
+
+ /**
+ * @name General
+ * @{
+ */
+
+ /**
+ * Fetch list of all objects, optionally restricted to given type.
+ */
+ int listObjects(List & list, Object::Type type = Object::TypeUndefined);
+
+ /**
+ * Get the latest error
+ *
+ * @return Error object.
+ */
+ const struct NdbError & getNdbError() const;
+
+ /** @} *******************************************************************/
+ /**
+ * @name Tables
+ * @{
+ */
+
+ /**
+ * Create defined table given defined Table instance
+ * @param Table Table to create
+ * @return 0 if successful otherwise -1.
+ */
+ int createTable(const Table &);
+
+ /**
+ * Drop table given retrieved Table instance
+ * @param Table Table to drop
+ * @return 0 if successful otherwise -1.
+ */
+ int dropTable(Table &);
+
+ /**
+ * Drop table given table name
+ * @param name Name of table to drop
+ * @return 0 if successful otherwise -1.
+ */
+ int dropTable(const char * name);
+
+ /**
+ * Alter defined table given defined Table instance
+ * @param Table Table to alter
+ * @return -2 (incompatible version) <br>
+ * -1 general error <br>
+ * 0 success
+ */
+ int alterTable(const Table &);
+
+ /**
+ * Get table with given name, NULL if undefined
+ * @param name Name of table to get
+ * @return table if successful otherwise NULL.
+ */
+ const Table * getTable(const char * name);
+
+ /**
+ * Get table with given name for alteration.
+ * @param name Name of table to alter
+ * @return table if successful. NULL if undefined
+ */
+ Table getTableForAlteration(const char * name);
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ /**
+ * Invalidate cached table object
+ * @param name Name of table to invalidate
+ */
+ void invalidateTable(const char * name);
+#endif
+
+ /**
+ * Remove table/index from local cache
+ */
+ void removeCachedTable(const char * table);
+ void removeCachedIndex(const char * index, const char * table);
+
+
+ /** @} *******************************************************************/
+ /**
+ * @name Indexes
+ * @{
+ */
+
+ /**
+ * Create index given defined Index instance
+ * @param Index to create
+ * @return 0 if successful otherwise -1.
+ */
+ int createIndex(const Index &);
+
+ /**
+ * Drop index with given name
+ * @param indexName Name of index to drop.
+ * @param tableName Name of table that index belongs to.
+ * @return 0 if successful otherwise -1.
+ */
+ int dropIndex(const char * indexName,
+ const char * tableName);
+
+ /**
+ * Get index with given name, NULL if undefined
+ * @param indexName Name of index to get.
+ * @param tableName Name of table that index belongs to.
+ * @return index if successful, otherwise 0.
+ */
+ const Index * getIndex(const char * indexName,
+ const char * tableName);
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+ /**
+ * Invalidate cached index object
+ */
+ void invalidateIndex(const char * indexName,
+ const char * tableName);
+#endif
+
+ /**
+ * Fetch list of indexes of given table.
+ * @param list Reference to list where to store the listed indexes
+ * @param tableName Name of table that index belongs to.
+ * @return 0 if successful, otherwise -1
+ */
+ int listIndexes(List & list, const char * tableName);
+
+ /** @} *******************************************************************/
+ /**
+ * @name Events
+ * @{
+ */
+
+ /**
+ * Create event given defined Event instance
+ * @param Event to create
+ * @return 0 if successful otherwise -1.
+ */
+ int createEvent(const Event &);
+
+ /**
+ * Drop event with given name
+ * @param eventName Name of event to drop.
+ * @return 0 if successful otherwise -1.
+ */
+ int dropEvent(const char * eventName);
+
+ /**
+ * Get event with given name.
+ * @param eventName Name of event to get.
+ * @return an Event if successful, otherwise NULL.
+ */
+ const Event * getEvent(const char * eventName);
+
+ /** @} *******************************************************************/
+
+ protected:
+ Dictionary(Ndb & ndb);
+ ~Dictionary();
+
+ private:
+ friend class NdbDictionaryImpl;
+ friend class UtilTransactions;
+ class NdbDictionaryImpl & m_impl;
+ Dictionary(NdbDictionaryImpl&);
+ const Table * getIndexTable(const char * indexName,
+ const char * tableName);
+ };
+};
+
+#endif
diff --git a/ndb/include/ndbapi/NdbError.hpp b/ndb/include/ndbapi/NdbError.hpp
new file mode 100644
index 00000000000..b08dd1041b2
--- /dev/null
+++ b/ndb/include/ndbapi/NdbError.hpp
@@ -0,0 +1,212 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_ERROR_HPP
+#define NDB_ERROR_HPP
+
+/**
+ * @struct NdbError
+ * @brief Contains error information
+ *
+ * A NdbError consists of five parts:
+ * -# Error status : Application impact
+ * -# Error classification : Logical error group
+ * -# Error code : Internal error code
+ * -# Error message : Context independent description of error
+ * -# Error details : Context dependent information
+ * (not always available)
+ *
+ * <em>Error status</em> is usually used for programming against errors.
+ * If more detailed error control is needed, it is possible to
+ * use the <em>error classification</em>.
+ *
+ * It is not recommended to write application programs dependent on
+ * specific <em>error codes</em>.
+ *
+ * The <em>error messages</em> and <em>error details</em> may
+ * change without notice.
+ *
+ * For example of use, see @ref ndbapi_example3.cpp.
+ */
+struct NdbError {
+ /**
+ * Status categorizes error codes into status values reflecting
+ * what the application should do when encountering errors
+ */
+ enum Status {
+ /**
+ * The error code indicate success<br>
+ * (Includes classification: NdbError::NoError)
+ */
+ Success = 0,
+
+ /**
+ * The error code indicates a temporary error.
+ * The application should typically retry.<br>
+ * (Includes classifications: NdbError::InsufficientSpace,
+ * NdbError::TemporaryResourceError, NdbError::NodeRecoveryError,
+ * NdbError::OverloadError, NdbError::NodeShutdown
+ * and NdbError::TimeoutExpired.)
+ */
+ TemporaryError = 1,
+
+ /**
+ * The error code indicates a permanent error.<br>
+ * (Includes classificatons: NdbError::PermanentError,
+ * NdbError::ApplicationError, NdbError::NoDataFound,
+ * NdbError::ConstraintViolation, NdbError::SchemaError,
+ * NdbError::UserDefinedError, NdbError::InternalError, and,
+ * NdbError::FunctionNotImplemented.)
+ */
+ PermanentError = 2,
+
+ /**
+ * The result/status is unknown.<br>
+ * (Includes classifications: NdbError::UnknownResultError, and
+ * NdbError::UnknownErrorCode.)
+ */
+ UnknownResult = 3
+ };
+
+ /**
+ * Type of error
+ */
+ enum Classification {
+ /**
+ * Success. No error occurred.
+ */
+ NoError = 0,
+
+ /**
+ * Error in application program.
+ */
+ ApplicationError = 1,
+
+ /**
+ * Read operation failed due to missing record.
+ */
+ NoDataFound = 2,
+
+ /**
+ * E.g. inserting a tuple with a primary key already existing
+ * in the table.
+ */
+ ConstraintViolation = 3,
+
+ /**
+ * Error in creating table or usage of table.
+ */
+ SchemaError = 4,
+
+ /**
+ * Error occurred in interpreted program.
+ */
+ UserDefinedError = 5,
+
+ /**
+ * E.g. insufficient memory for data or indexes.
+ */
+ InsufficientSpace = 6,
+
+ /**
+ * E.g. too many active transactions.
+ */
+ TemporaryResourceError = 7,
+
+ /**
+ * Temporary failures which are probably inflicted by a node
+ * recovery in progress. Examples: information sent between
+ * application and NDB lost, distribution change.
+ */
+ NodeRecoveryError = 8,
+
+ /**
+ * E.g. out of log file space.
+ */
+ OverloadError = 9,
+
+ /**
+ * Timeouts, often inflicted by deadlocks in NDB.
+ */
+ TimeoutExpired = 10,
+
+ /**
+ * Is is unknown whether the transaction was committed or not.
+ */
+ UnknownResultError = 11,
+
+ /**
+ * A serious error in NDB has occurred.
+ */
+ InternalError = 12,
+
+ /**
+ * A function used is not yet implemented.
+ */
+ FunctionNotImplemented = 13,
+
+ /**
+ * Error handler could not determine correct error code.
+ */
+ UnknownErrorCode = 14,
+
+ /**
+ * Node shutdown
+ */
+ NodeShutdown = 15
+ };
+
+ /**
+ * Error status.
+ */
+ Status status;
+
+ /**
+ * Error type
+ */
+ Classification classification;
+
+ /**
+ * Error code
+ */
+ int code;
+
+ /**
+ * Error message
+ */
+ const char * message;
+
+ /**
+ * The detailed description. This is extra information regarding the
+ * error which is not included in the error message.
+ *
+ * @note Is NULL when no details specified
+ */
+ char * details;
+
+ NdbError(){
+ status = UnknownResult;
+ classification = NoError;
+ code = 0;
+ message = 0;
+ details = 0;
+ }
+};
+
+class NdbOut& operator <<(class NdbOut&, const NdbError &);
+class NdbOut& operator <<(class NdbOut&, const NdbError::Status&);
+class NdbOut& operator <<(class NdbOut&, const NdbError::Classification&);
+#endif
diff --git a/ndb/include/ndbapi/NdbEventOperation.hpp b/ndb/include/ndbapi/NdbEventOperation.hpp
new file mode 100644
index 00000000000..911b00b02c4
--- /dev/null
+++ b/ndb/include/ndbapi/NdbEventOperation.hpp
@@ -0,0 +1,205 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbEventOperation.hpp
+ * Include:
+ * Link:
+ * Author: Tomas Ulin MySQL AB
+ * Date: 2003-11-21
+ * Version: 0.1
+ * Description: Event support
+ * Documentation:
+ * Adjust: 2003-11-21 Tomas Ulin First version.
+ * Adjust: 2003-12-11 Tomas Ulin Alpha Release.
+ ****************************************************************************/
+
+#ifndef NdbEventOperation_H
+#define NdbEventOperation_H
+
+class NdbGlobalEventBuffer;
+class NdbEventOperationImpl;
+
+/**
+ * @class NdbEventOperation
+ * @brief Class of operations for getting change events from database.
+ *
+ * An NdbEventOperation object is instantiated by
+ * NdbEventOperation *Ndb::createEventOperation(const char *eventName,
+ * int bufferLength)
+ *
+ * Prior to that an event must have been created in the Database through
+ * int NdbDictionary::createEvent(NdbDictionary::Event)
+ *
+ * bufferLength indicates size of circular buffer to store event info as
+ * they occur.
+ *
+ * The instance is removed by Ndb::dropEventOperation(NdbEventOperation*)
+ *
+ * For more info see:
+ * ndbapi_example5.cpp
+ * Ndb.hpp
+ * NdbDictionary.hpp
+ *
+ * Known limitations:
+ *
+ * Maximum number of active NdbEventOperations are now set at compile time.
+ * Today 100. This will become a configuration parameter later.
+ *
+ * Maximum number of NdbEventOperations tied to same event are maximum 16
+ * per process.
+ *
+ * Known issues:
+ *
+ * When several NdbEventOperation s are tied to the same event in the same
+ * process they will share the circular buffer. The BufferLength will then
+ * be the same for all and decided by the first NdbEventOperation
+ * instantiation. Just make sure to instantiate the "largest" one first.
+ *
+ * Today all events INSERT/DELETE/UPDATE and all changed attributes are
+ * sent to the API, even if only specific attributes have been specified.
+ * These are however hidden from the user and only relevant data is shown
+ * after next(). However false exits from pollEvents() may occur and thus
+ * the subsequent next() will return zero, since there was no available
+ * data. Just do pollEvents() again. Will be fixed in later versions.
+ *
+ * Event code does not check table schema version. Make sure to drop events
+ * after table is dropped. Will be fixed in later
+ * versions.
+ *
+ * On a replicated system one will receive each event 2 times, one for each
+ * replica. If a node fails events will not be received twice anymore
+ * for data in corresponding fragment. Will be optimized in later versions.
+ *
+ * If a nodefailiure has occured not all events will be recieved
+ * anymore. Drop NdbEventOperation and Create again after nodes are up
+ * again. Will be fixed in later versions.
+ *
+ * Test status:
+ * Tests have been run on 1-node and 2-node systems
+ *
+ * Known bugs:
+ *
+ * None, except if we can call some of the "isses" above bugs
+ *
+ * Useful API programs:
+ *
+ * select_all -d sys 'NDB$EVENTS_0'
+ * Will show contents in the system table containing created events.
+ *
+ */
+class NdbEventOperation {
+public:
+ enum State {CREATED,EXECUTING,ERROR};
+
+ State getState();
+
+ /**
+ * Activates the NdbEventOperation to start receiving events. The
+ * changed attribute values may be retrieved after next() has returned
+ * a value greater than zero. The getValue() methods below must be called
+ * prior to execute().
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int execute();
+
+ // about the event operation
+ // getting data
+ // NdbResultSet* getResultSet();
+
+ /**
+ * Defines a retrieval operation of an attribute value.
+ * The NDB API allocate memory for the NdbRecAttr object that
+ * will hold the returned attribute value.
+ *
+ * @note Note that it is the applications responsibility
+ * to allocate enough memory for aValue (if non-NULL).
+ * The buffer aValue supplied by the application must be
+ * aligned appropriately. The buffer is used directly
+ * (avoiding a copy penalty) only if it is aligned on a
+ * 4-byte boundary and the attribute size in bytes
+ * (i.e. NdbRecAttr::attrSize times NdbRecAttr::arraySize is
+ * a multiple of 4).
+ *
+ * @note There are two versions, NdbOperation::getValue and
+ * NdbOperation::getPreValue for retrieving the current and
+ * previous value repectively.
+ *
+ * @note This method does not fetch the attribute value from
+ * the database! The NdbRecAttr object returned by this method
+ * is <em>not</em> readable/printable before the
+ * NdbEventConnection::execute has been made and
+ * NdbEventConnection::next has returned a value greater than
+ * zero. If a specific attribute has not changed the corresponding
+ * NdbRecAttr will be in state UNDEFINED. This is checked by
+ * NdbRecAttr::isNull which then returns -1.
+ *
+ * @param anAttrName Attribute name
+ * @param aValue If this is non-NULL, then the attribute value
+ * will be returned in this parameter.<br>
+ * If NULL, then the attribute value will only
+ * be stored in the returned NdbRecAttr object.
+ * @return An NdbRecAttr object to hold the value of
+ * the attribute, or a NULL pointer
+ * (indicating error).
+ */
+ NdbRecAttr *getValue(const char *anAttrName, char *aValue = NULL);
+ NdbRecAttr *getPreValue(const char *anAttrName, char *aValue = NULL);
+
+ /**
+ * Retrieves event resultset if available, inserted into the NdbRecAttrs
+ * specified in getValue() and getPreValue(). To avoid polling for
+ * a resultset, one can use Ndb::pollEvents(int millisecond_timeout)
+ * which will wait on a mutex until an event occurs or the specified
+ * timeout occurs.
+ *
+ * @return >=0 if successful otherwise -1. Return value inicates number
+ * of available events. By sending pOverRun one may query for buffer
+ * overflow and *pOverRun will indicate the number of events that have
+ * overwritten.
+ */
+ int next(int *pOverRun=NULL);
+
+ /**
+ * In the current implementation a nodefailiure may cause loss of events,
+ * in which case isConsistent() will return false
+ */
+ bool isConsistent();
+
+ /**
+ * Query for occured event type.
+ * NdbDictionary::Event::{TE_INSERT,TE_UPDATE,TE_DELETE}
+ * Only valid after next() has returned value >= 0
+ */
+ NdbDictionary::Event::TableEvent getEventType();
+
+ Uint32 getGCI();
+ Uint32 getLatestGCI();
+ void print();
+
+private:
+ friend class NdbEventOperationImpl;
+ friend class Ndb;
+ NdbEventOperation(Ndb *theNdb, const char* eventName,int bufferLength);
+ ~NdbEventOperation();
+ static int wait(void *p, int aMillisecondNumber);
+ class NdbEventOperationImpl &m_impl;
+ NdbEventOperation(NdbEventOperationImpl& impl);
+};
+
+typedef void (* NdbEventCallback)(NdbEventOperation*, Ndb*, void*);
+#endif
diff --git a/ndb/include/ndbapi/NdbIndexOperation.hpp b/ndb/include/ndbapi/NdbIndexOperation.hpp
new file mode 100644
index 00000000000..3b8e5f7a888
--- /dev/null
+++ b/ndb/include/ndbapi/NdbIndexOperation.hpp
@@ -0,0 +1,192 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbIndexOperation.hpp
+ * Include:
+ * Link:
+ * Author: Martin Sköld
+ * Date: 2002-04-01
+ * Version: 0.1
+ * Description: Secondary index support
+ * Documentation:
+ * Adjust: 2002-04-01 Martin Sköld First version.
+ ****************************************************************************/
+
+#ifndef NdbIndexOperation_H
+#define NdbIndexOperation_H
+
+#include <NdbCursorOperation.hpp>
+
+class Index;
+class NdbResultSet;
+
+/**
+ * @class NdbIndexOperation
+ * @brief Class of index operations for use in transactions
+ */
+class NdbIndexOperation : public NdbOperation
+{
+ friend class Ndb;
+ friend class NdbConnection;
+
+public:
+ /**
+ * @name Define Standard Operation
+ * @{
+ */
+
+ /**
+ * Define the NdbIndexOperation to be a standard operation of type readTuple.
+ * When calling NdbConnection::execute, this operation
+ * reads a tuple.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int readTuple();
+
+ /**
+ * Define the NdbIndexOperation to be a standard operation of type
+ * readTupleExclusive.
+ * When calling NdbConnection::execute, this operation
+ * read a tuple using an exclusive lock.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int readTupleExclusive();
+
+ /**
+ * Define the NdbIndexOperation to be a standard operation of type simpleRead.
+ * When calling NdbConnection::execute, this operation
+ * reads an existing tuple (using shared read lock),
+ * but releases lock immediately after read.
+ *
+ * @note Using this operation twice in the same transaction
+ * may produce different results (e.g. if there is another
+ * transaction which updates the value between the
+ * simple reads).
+ *
+ * Note that simpleRead can read the value from any database node while
+ * standard read always read the value on the database node which is
+ * primary for the record.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int simpleRead();
+
+ /**
+ * Define the NdbOperation to be a standard operation of type committedRead.
+ * When calling NdbConnection::execute, this operation
+ * read latest committed value of the record.
+ *
+ * This means that if another transaction is updating the
+ * record, then the current transaction will not wait.
+ * It will instead use the latest committed value of the
+ * record.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int dirtyRead();
+
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+ int committedRead();
+#endif
+
+ /**
+ * Define the NdbIndexOperation to be a standard operation of type
+ * updateTuple.
+ *
+ * When calling NdbConnection::execute, this operation
+ * updates a tuple in the table.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int updateTuple();
+
+ /**
+ * Define the NdbIndexOperation to be a standard operation of type
+ * deleteTuple.
+ *
+ * When calling NdbConnection::execute, this operation
+ * deletes a tuple.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int deleteTuple();
+
+ /**
+ * Define the NdbIndexOperation to be a standard operation of type
+ * dirtyUpdate.
+ *
+ * When calling NdbConnection::execute, this operation
+ * updates without two-phase commit.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int dirtyUpdate();
+
+ /** @} *********************************************************************/
+ /**
+ * @name Define Interpreted Program Operation
+ * @{
+ */
+
+ /**
+ * Update a tuple using an interpreted program.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int interpretedUpdateTuple();
+
+ /**
+ * Delete a tuple using an interpreted program.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int interpretedDeleteTuple();
+
+ /** @} *********************************************************************/
+
+private:
+ NdbIndexOperation(Ndb* aNdb);
+ ~NdbIndexOperation();
+
+ void closeScan();
+
+ int receiveTCINDXREF(NdbApiSignal* aSignal);
+
+ // Overloaded method from NdbOperation
+ void setLastFlag(NdbApiSignal* signal, Uint32 lastFlag);
+
+ // Overloaded methods from NdbCursorOperation
+ int executeCursor(int ProcessorId);
+
+ // Overloaded methods from NdbCursorOperation
+ int indxInit(class NdbIndexImpl* anIndex,
+ class NdbTableImpl* aTable,
+ NdbConnection* myConnection);
+
+ int equal_impl(const class NdbColumnImpl*, const char* aValue, Uint32 len);
+ int prepareSend(Uint32 TC_ConnectPtr, Uint64 TransactionId);
+
+ // Private attributes
+ NdbIndexImpl* m_theIndex;
+ Uint32 m_theIndexDefined[MAXNROFTUPLEKEY][3];
+ Uint32 m_theIndexLen; // Length of the index in words
+ Uint32 m_theNoOfIndexDefined; // The number of index attributes
+};
+
+#endif
diff --git a/ndb/include/ndbapi/NdbOperation.hpp b/ndb/include/ndbapi/NdbOperation.hpp
new file mode 100644
index 00000000000..4f5f4597937
--- /dev/null
+++ b/ndb/include/ndbapi/NdbOperation.hpp
@@ -0,0 +1,1338 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbOperation_H
+#define NdbOperation_H
+
+#include <stdlib.h>
+#include <assert.h>
+
+#include <ndb_types.h>
+#include "AttrType.hpp"
+#include "NdbError.hpp"
+#include "NdbReceiver.hpp"
+#include <stdlib.h>
+
+class Ndb;
+class NdbApiSignal;
+class NdbRecAttr;
+class NdbOperation;
+class NdbConnection;
+class NdbColumnImpl;
+
+/**
+ * @class NdbOperation
+ * @brief Class of operations for use in transactions.
+ */
+class NdbOperation
+{
+ friend class Ndb;
+ friend class NdbConnection;
+ friend class NdbScanOperation;
+ friend class NdbScanReceiver;
+ friend class NdbScanFilter;
+ friend class NdbScanFilterImpl;
+
+public:
+ /**
+ * @name Define Standard Operation Type
+ * @{
+ */
+
+ /**
+ * Define the NdbOperation to be a standard operation of type insertTuple.
+ * When calling NdbConnection::execute, this operation
+ * adds a new tuple to the table.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int insertTuple();
+
+ /**
+ * Define the NdbOperation to be a standard operation of type updateTuple.
+ * When calling NdbConnection::execute, this operation
+ * updates a tuple in the table.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int updateTuple();
+
+ /**
+ * Define the NdbOperation to be a standard operation of type writeTuple.
+ * When calling NdbConnection::execute, this operation
+ * writes a tuple to the table.
+ * If the tuple exists, it updates it, otherwise an insert takes place.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int writeTuple();
+
+ /**
+ * Define the NdbOperation to be a standard operation of type deleteTuple.
+ * When calling NdbConnection::execute, this operation
+ * delete a tuple.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int deleteTuple();
+
+ /**
+ * Define the NdbOperation to be a standard operation of type readTuple.
+ * When calling NdbConnection::execute, this operation
+ * reads a tuple.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int readTuple();
+
+ /**
+ * Define the NdbOperation to be a standard operation of type
+ * readTupleExclusive.
+ * When calling NdbConnection::execute, this operation
+ * read a tuple using an exclusive lock.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int readTupleExclusive();
+
+ /**
+ * Define the NdbOperation to be a standard operation of type
+ * simpleRead.
+ * When calling NdbConnection::execute, this operation
+ * reads an existing tuple (using shared read lock),
+ * but releases lock immediately after read.
+ *
+ * @note Using this operation twice in the same transaction
+ * may produce different results (e.g. if there is another
+ * transaction which updates the value between the
+ * simple reads).
+ *
+ * Note that simpleRead can read the value from any database node while
+ * standard read always read the value on the database node which is
+ * primary for the record.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int simpleRead();
+
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+ /**
+ * Define the NdbOperation to be a standard operation of type committedRead.
+ * When calling NdbConnection::execute, this operation
+ * read latest committed value of the record.
+ *
+ * This means that if another transaction is updating the
+ * record, then the current transaction will not wait.
+ * It will instead use the latest committed value of the
+ * record.
+ * dirtyRead is a deprecated name for committedRead
+ *
+ * @return 0 if successful otherwise -1.
+ * @depricated
+ */
+ virtual int dirtyRead();
+#endif
+
+ /**
+ * Define the NdbOperation to be a standard operation of type committedRead.
+ * When calling NdbConnection::execute, this operation
+ * read latest committed value of the record.
+ *
+ * This means that if another transaction is updating the
+ * record, then the current transaction will not wait.
+ * It will instead use the latest committed value of the
+ * record.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int committedRead();
+
+ /**
+ * Define the NdbOperation to be a standard operation of type dirtyUpdate.
+ * When calling NdbConnection::execute, this operation
+ * updates without two-phase commit.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int dirtyUpdate();
+
+ /**
+ * Define the NdbOperation to be a standard operation of type dirtyWrite.
+ * When calling NdbConnection::execute, this operation
+ * writes without two-phase commit.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int dirtyWrite();
+
+ /** @} *********************************************************************/
+ /**
+ * @name Define Interpreted Program Operation Type
+ * @{
+ */
+
+ /**
+ * Update a tuple using an interpreted program.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int interpretedUpdateTuple();
+
+ /**
+ * Delete a tuple using an interpreted program.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ virtual int interpretedDeleteTuple();
+
+ /**
+ * Scan a table to read tuples.
+ *
+ * The operation only sets a temporary read lock while
+ * reading the tuple.
+ * The tuple lock is released when the result of the read reaches the
+ * application.
+ *
+ * @param Parallelism Number of parallel tuple reads are performed
+ * in the scan.
+ * Currently a maximum of 256 parallel tuple
+ * reads are allowed.
+ * The parallelism can in reality be lower
+ * than specified
+ * depending on the number of nodes
+ * in the cluster
+ * @return 0 if successful otherwise -1.
+ */
+ int openScanRead(Uint32 Parallelism = 16 );
+
+ /**
+ * Scan a table to write or update tuples.
+ *
+ * The operation sets an exclusive lock on the tuple and sends the result
+ * to the application.
+ * Thus when the application reads the data, the tuple is
+ * still locked with an exclusive lock.
+ *
+ * @param parallelism Number of parallel tuple reads are performed
+ * in the scan.
+ * Currently a maximum of 256 parallel tuple
+ * reads are allowed.
+ * The parallelism can in reality be lower
+ * than specified depending on the number
+ * of nodes in the cluster
+ * @return 0 if successful otherwise -1.
+ *
+ */
+ int openScanExclusive(Uint32 parallelism = 16);
+
+ /**
+ * Scan a table to read tuples.
+ *
+ * The operation only sets a read lock while
+ * reading the tuple.
+ * Thus when the application reads the data, the tuple is
+ * still locked with a read lock.
+ *
+ * @param parallelism Number of parallel tuple reads are performed
+ * in the scan.
+ * Currently a maximum of 256 parallel tuple
+ * reads are allowed.
+ * The parallelism can in reality be lower
+ * than specified
+ * depending on the number of nodes
+ * in the cluster
+ * @return 0 if successful otherwise -1.
+ */
+ int openScanReadHoldLock(Uint32 parallelism = 16);
+
+ /**
+ * Scan a table to read tuples.
+ *
+ * The operation does not wait for locks held by other transactions
+ * but returns the latest committed tuple instead.
+ *
+ * @param parallelism Number of parallel tuple reads are performed
+ * in the scan.
+ * Currently a maximum of 256 parallel tuple
+ * reads are allowed.
+ * The parallelism can in reality be lower
+ * than specified
+ * depending on the number of nodes
+ * in the cluster
+ * @return 0 if successful otherwise -1.
+ */
+ int openScanReadCommitted(Uint32 parallelism = 16);
+
+ /** @} *********************************************************************/
+
+ /**
+ * @name Define Range Scan
+ *
+ * A range scan is a scan on an ordered index. The operation is on
+ * the index table but tuples are returned from the primary table.
+ * The index contains all tuples where at least one index key has not
+ * null value.
+ *
+ * A range scan is currently opened via a normal open scan method.
+ * Bounds can be defined for each index key. After setting bounds,
+ * usual scan methods can be used (get value, interpreter, take over).
+ * These operate on the primary table.
+ *
+ * @{
+ */
+
+ /**
+ * Type of ordered index key bound. The values (0-4) will not change
+ * and can be used explicitly (e.g. they could be computed).
+ */
+ enum BoundType {
+ BoundLE = 0, ///< lower bound,
+ BoundLT = 1, ///< lower bound, strict
+ BoundGE = 2, ///< upper bound
+ BoundGT = 3, ///< upper bound, strict
+ BoundEQ = 4 ///< equality
+ };
+
+ /**
+ * Define bound on index key in range scan.
+ *
+ * Each index key can have not null lower and/or upper bound, or can
+ * be set equal to not null value. The bounds can be defined in any
+ * order but a duplicate definition is an error.
+ *
+ * The scan is most effective when bounds are given for an initial
+ * sequence of non-nullable index keys, and all but the last one is an
+ * equality. In this case the scan returns a contiguous range from
+ * each ordered index fragment.
+ *
+ * @note This release implements only the case described above,
+ * except for the non-nullable limitation. Other sets of
+ * bounds return error or empty result set.
+ *
+ * @note In this release a null key value satisfies any lower
+ * bound and no upper bound. This may change.
+ *
+ * @param attrName Attribute name, alternatively:
+ * @param anAttrId Index column id (starting from 0).
+ * @param type Type of bound
+ * @param value Pointer to bound value
+ * @param len Value length in bytes.
+ * Fixed per datatype and can be omitted
+ * @return 0 if successful otherwise -1
+ */
+ int setBound(const char* anAttrName, int type, const void* aValue, Uint32 len = 0);
+
+ /**
+ * Define bound on index key in range scan using index column id.
+ * See the other setBound() method for details.
+ */
+ int setBound(Uint32 anAttrId, int type, const void* aValue, Uint32 len = 0);
+
+ /** @} *********************************************************************/
+
+ /**
+ * Validate parallelism parameter by checking the number
+ * against number of executing Ndb nodes.
+ *
+ * @param Parallelism
+ * @return 0 if correct parallelism value, otherwise -1.
+ *
+ */
+ int checkParallelism(Uint32 Parallelism);
+
+ /**
+ * Transfer scan operation to an updating transaction. Use this function
+ * when a scan has found a record that you want to update.
+ * 1. Start a new transaction.
+ * 2. Call the function takeOverForUpdate using your new transaction
+ * as parameter, all the properties of the found record will be copied
+ * to the new transaction.
+ * 3. When you execute the new transaction, the lock held by the scan will
+ * be transferred to the new transaction(it's taken over).
+ *
+ * @note You must have started the scan with openScanExclusive
+ * to be able to update the found tuple.
+ *
+ * @param updateTrans the update transaction connection.
+ * @return an NdbOperation or NULL.
+ */
+ NdbOperation* takeOverForUpdate(NdbConnection* updateTrans);
+
+ /**
+ * Transfer scan operation to a deleting transaction. Use this function
+ * when a scan has found a record that you want to delete.
+ * 1. Start a new transaction.
+ * 2. Call the function takeOverForDelete using your new transaction
+ * as parameter, all the properties of the found record will be copied
+ * to the new transaction.
+ * 3. When you execute the new transaction, the lock held by the scan will
+ * be transferred to the new transaction(its taken over).
+ *
+ * @note You must have started the scan with openScanExclusive
+ * to be able to delete the found tuple.
+ *
+ * @param deleteTrans the delete transaction connection.
+ * @return an NdbOperation or NULL.
+ */
+ NdbOperation* takeOverForDelete(NdbConnection* deleteTrans);
+
+ /**
+ * @name Specify Search Conditions
+ * @{
+ */
+ /**
+ * Define a search condition with equality.
+ * The condition is true if the attribute has the given value.
+ * To set search conditions on multiple attributes,
+ * use several equals (then all of them must be satisfied for the
+ * tuple to be selected).
+ *
+ * @note There are 10 versions of NdbOperation::equal with
+ * slightly different parameters.
+ *
+ * @note When using NdbOperation::equal with a string (char *) as
+ * second argument, the string needs to be padded with
+ * zeros in the following sense:
+ * @code
+ * // Equal needs strings to be padded with zeros
+ * strncpy(buf, str, sizeof(buf));
+ * NdbOperation->equal("Attr1", buf);
+ * @endcode
+ *
+ * @param anAttrName Attribute name
+ * @param aValue Attribute value.
+ * @param len Attribute length expressed in bytes.
+ * @return -1 if unsuccessful.
+ */
+ int equal(const char* anAttrName, const char* aValue, Uint32 len = 0);
+ int equal(const char* anAttrName, Uint32 aValue);
+ int equal(const char* anAttrName, Int32 aValue);
+ int equal(const char* anAttrName, Int64 aValue);
+ int equal(const char* anAttrName, Uint64 aValue);
+ int equal(Uint32 anAttrId, const char* aValue, Uint32 len = 0);
+ int equal(Uint32 anAttrId, Int32 aValue);
+ int equal(Uint32 anAttrId, Uint32 aValue);
+ int equal(Uint32 anAttrId, Int64 aValue);
+ int equal(Uint32 anAttrId, Uint64 aValue);
+
+ /**
+ * Generate a tuple id and set it as search argument.
+ *
+ * The Tuple id has NDB$TID as attribute name and 0 as attribute id.
+ *
+ * The generated tuple id is returned by the method.
+ * If zero is returned there is an error.
+ *
+ * This is mostly used for tables without any primary key
+ * attributes.
+ *
+ * @return Generated tuple id if successful, otherwise 0.
+ */
+ Uint64 setTupleId();
+
+ /** @} *********************************************************************/
+ /**
+ * @name Specify Attribute Actions for Operations
+ * @{
+ */
+
+ /**
+ * Defines a retrieval operation of an attribute value.
+ * The NDB API allocate memory for the NdbRecAttr object that
+ * will hold the returned attribute value.
+ *
+ * @note Note that it is the applications responsibility
+ * to allocate enough memory for aValue (if non-NULL).
+ * The buffer aValue supplied by the application must be
+ * aligned appropriately. The buffer is used directly
+ * (avoiding a copy penalty) only if it is aligned on a
+ * 4-byte boundary and the attribute size in bytes
+ * (i.e. NdbRecAttr::attrSize times NdbRecAttr::arraySize is
+ * a multiple of 4).
+ *
+ * @note There are two versions of NdbOperation::getValue with
+ * slightly different parameters.
+ *
+ * @note This method does not fetch the attribute value from
+ * the database! The NdbRecAttr object returned by this method
+ * is <em>not</em> readable/printable before the
+ * transaction has been executed with NdbConnection::execute.
+ *
+ * @param anAttrName Attribute name
+ * @param aValue If this is non-NULL, then the attribute value
+ * will be returned in this parameter.<br>
+ * If NULL, then the attribute value will only
+ * be stored in the returned NdbRecAttr object.
+ * @return An NdbRecAttr object to hold the value of
+ * the attribute, or a NULL pointer
+ * (indicating error).
+ */
+ NdbRecAttr* getValue(const char* anAttrName, char* aValue = NULL);
+ NdbRecAttr* getValue(Uint32 anAttrId, char* aValue = NULL);
+
+ /**
+ * Define an attribute to set or update in query.
+ *
+ * To set a NULL value, use the following construct:
+ * @code
+ * setValue("ATTR_NAME", (char*)NULL);
+ * @endcode
+ *
+ * There are a number of NdbOperation::setValue methods that
+ * take a certain type as input
+ * (pass by value rather than passing a pointer).
+ * As the interface is currently implemented it is the responsibility
+ * of the application programmer to use the correct types.
+ *
+ * The NDB API will however check that the application sends
+ * a correct length to the interface as given in the length parameter.
+ * The passing of char* as the value can contain any type or
+ * any type of array.
+ * If length is not provided or set to zero,
+ * then the API will assume that the pointer
+ * is correct and not bother with checking it.
+ *
+ * @note There are 14 versions of NdbOperation::setValue with
+ * slightly different parameters.
+ *
+ * @param anAttrName Name (or Id) of attribute.
+ * @param aValue Attribute value to set.
+ * @param len Attribute length expressed in bytes.
+ * @return -1 if unsuccessful.
+ */
+ virtual int setValue(const char* anAttrName, const char* aValue,
+ Uint32 len = 0);
+ virtual int setValue(const char* anAttrName, Int32 aValue);
+ virtual int setValue(const char* anAttrName, Uint32 aValue);
+ virtual int setValue(const char* anAttrName, Uint64 aValue);
+ virtual int setValue(const char* anAttrName, Int64 aValue);
+ virtual int setValue(const char* anAttrName, float aValue);
+ virtual int setValue(const char* anAttrName, double aValue);
+
+ virtual int setValue(Uint32 anAttrId, const char* aValue, Uint32 len = 0);
+ virtual int setValue(Uint32 anAttrId, Int32 aValue);
+ virtual int setValue(Uint32 anAttrId, Uint32 aValue);
+ virtual int setValue(Uint32 anAttrId, Uint64 aValue);
+ virtual int setValue(Uint32 anAttrId, Int64 aValue);
+ virtual int setValue(Uint32 anAttrId, float aValue);
+ virtual int setValue(Uint32 anAttrId, double aValue);
+
+ /** @} *********************************************************************/
+ /**
+ * @name Specify Interpreted Program Instructions
+ * @{
+ */
+
+ /**
+ * Interpreted program instruction: Add a value to an attribute.
+ *
+ * @note Destroys the contents of registers 6 and 7.
+ * (The instruction uses these registers for its operation.)
+ *
+ * @note There are four versions of NdbOperation::incValue with
+ * slightly different parameters.
+ *
+ * @param anAttrName Attribute name.
+ * @param aValue Value to add.
+ * @return -1 if unsuccessful.
+ */
+ int incValue(const char* anAttrName, Uint32 aValue);
+ int incValue(const char* anAttrName, Uint64 aValue);
+ int incValue(Uint32 anAttrId, Uint32 aValue);
+ int incValue(Uint32 anAttrId, Uint64 aValue);
+
+ /**
+ * Interpreted program instruction:
+ * Subtract a value from an attribute in an interpreted operation.
+ *
+ * @note Destroys the contents of registers 6 and 7.
+ * (The instruction uses these registers for its operation.)
+ *
+ * @note There are four versions of NdbOperation::subValue with
+ * slightly different parameters.
+ *
+ * @param anAttrName Attribute name.
+ * @param aValue Value to subtract.
+ * @return -1 if unsuccessful.
+ */
+ int subValue(const char* anAttrName, Uint32 aValue);
+ int subValue(const char* anAttrName, Uint64 aValue);
+ int subValue(Uint32 anAttrId, Uint32 aValue);
+ int subValue(Uint32 anAttrId, Uint64 aValue);
+
+ /**
+ * Interpreted program instruction:
+ * Define a jump label in an interpreted operation.
+ *
+ * @note The labels are automatically numbered starting with 0.
+ * The parameter used by NdbOperation::def_label should
+ * match the automatic numbering to make it easier to
+ * debug the interpreted program.
+ *
+ * @param labelNumber Label number.
+ * @return -1 if unsuccessful.
+ */
+ int def_label(int labelNumber);
+
+ /**
+ * Interpreted program instruction:
+ * Add two registers into a third.
+ *
+ * @param RegSource1 First register.
+ * @param RegSource2 Second register.
+ * @param RegDest Destination register where the result will be stored.
+ * @return -1 if unsuccessful.
+ */
+ int add_reg(Uint32 RegSource1, Uint32 RegSource2, Uint32 RegDest);
+
+ /**
+ * Interpreted program instruction:
+ * Substract RegSource1 from RegSource2 and put the result in RegDest.
+ *
+ * @param RegSource1 First register.
+ * @param RegSource2 Second register.
+ * @param RegDest Destination register where the result will be stored.
+ * @return -1 if unsuccessful.
+ */
+ int sub_reg(Uint32 RegSource1, Uint32 RegSource2, Uint32 RegDest);
+
+ /**
+ * Interpreted program instruction:
+ * Load a constant into a register.
+ *
+ * @param RegDest Destination register.
+ * @param Constant Value to load.
+ * @return -1 if unsuccessful.
+ */
+ int load_const_u32(Uint32 RegDest, Uint32 Constant);
+ int load_const_u64(Uint32 RegDest, Uint64 Constant);
+
+ /**
+ * Interpreted program instruction:
+ * Load NULL value into a register.
+ *
+ * @param RegDest Destination register.
+ * @return -1 if unsuccessful.
+ */
+ int load_const_null(Uint32 RegDest);
+
+ /**
+ * Interpreted program instruction:
+ * Read an attribute into a register.
+ *
+ * @param anAttrName Attribute name.
+ * @param RegDest Destination register.
+ * @return -1 if unsuccessful.
+ */
+ int read_attr(const char* anAttrName, Uint32 RegDest);
+
+ /**
+ * Interpreted program instruction:
+ * Write an attribute from a register.
+ *
+ * @param anAttrName Attribute name.
+ * @param RegSource Source register.
+ * @return -1 if unsuccessful.
+ */
+ int write_attr(const char* anAttrName, Uint32 RegSource);
+
+ /**
+ * Interpreted program instruction:
+ * Read an attribute into a register.
+ *
+ * @param anAttrId the attribute id.
+ * @param RegDest the destination register.
+ * @return -1 if unsuccessful.
+ */
+ int read_attr(Uint32 anAttrId, Uint32 RegDest);
+
+ /**
+ * Interpreted program instruction:
+ * Write an attribute from a register.
+ *
+ * @param anAttrId the attribute id.
+ * @param RegSource the source register.
+ * @return -1 if unsuccessful.
+ */
+ int write_attr(Uint32 anAttrId, Uint32 RegSource);
+
+ /**
+ * Interpreted program instruction:
+ * Define a search condition. Last two letters in the function name
+ * describes the search condition.
+ * The condition compares RegR with RegL and therefore appears
+ * to be reversed.
+ *
+ * - ge RegR >= RegL
+ * - gt RegR > RegL
+ * - le RegR <= RegL
+ * - lt RegR < RegL
+ * - eq RegR = RegL
+ * - ne RegR <> RegL
+ *
+ * @param RegLvalue left value.
+ * @param RegRvalue right value.
+ * @param Label the label to jump to.
+ * @return -1 if unsuccessful.
+ */
+ int branch_ge(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label);
+ int branch_gt(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label);
+ int branch_le(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label);
+ int branch_lt(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label);
+ int branch_eq(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label);
+ int branch_ne(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label);
+
+ /**
+ * Interpreted program instruction:
+ * Jump to Label if RegLvalue is not NULL.
+ *
+ * @param RegLvalue the value to check.
+ * @param Label the label to jump to.
+ * @return -1 if unsuccessful.
+ */
+ int branch_ne_null(Uint32 RegLvalue, Uint32 Label);
+
+ /**
+ * Interpreted program instruction:
+ * Jump to Label if RegLvalue is equal to NULL.
+ *
+ * @param RegLvalue Value to check.
+ * @param Label Label to jump to.
+ * @return -1 if unsuccessful.
+ */
+ int branch_eq_null(Uint32 RegLvalue, Uint32 Label);
+
+ /**
+ * Interpreted program instruction:
+ * Jump to Label.
+ *
+ * @param Label Label to jump to.
+ * @return -1 if unsuccessful.
+ */
+ int branch_label(Uint32 Label);
+
+ /**
+ * Interpreted program instruction: branch after memcmp
+ * @param ColId Column to check
+ * @param Label Label to jump to
+ * @return -1 if unsuccessful
+ */
+ int branch_col_eq_null(Uint32 ColId, Uint32 Label);
+ int branch_col_ne_null(Uint32 ColId, Uint32 Label);
+
+ /**
+ * Interpreted program instruction: branch after memcmp
+ * @param ColId column to check
+ * @param val search value
+ * @param len length of search value
+ * @param nopad force non-padded comparison for a Char column
+ * @param Label label to jump to
+ * @return -1 if unsuccessful
+ */
+ int branch_col_eq(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label);
+ int branch_col_ne(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label);
+ int branch_col_lt(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label);
+ int branch_col_le(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label);
+ int branch_col_gt(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label);
+ int branch_col_ge(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label);
+ int branch_col_like(Uint32 ColId, const char *, Uint32 len,
+ bool nopad, Uint32 Label);
+ int branch_col_notlike(Uint32 ColId, const char *, Uint32 len,
+ bool nopad, Uint32 Label);
+
+ /**
+ * Interpreted program instruction: Exit with Ok
+ *
+ * For scanning transactions,
+ * end interpreted operation and return the row to the application.
+ *
+ * For non-scanning transactions,
+ * exit interpreted program.
+ *
+ * @return -1 if unsuccessful.
+ */
+ int interpret_exit_ok();
+
+ /**
+ * Interpreted program instruction: Exit with Not Ok
+ *
+ * For scanning transactions,
+ * continue with the next row without returning the current row.
+ *
+ * For non-scanning transactions,
+ * abort the whole transaction.
+ *
+ * @note A method also exists without the error parameter.
+ *
+ * @param ErrorCode An error code given by the application programmer.
+ * @return -1 if unsuccessful.
+ */
+ int interpret_exit_nok(Uint32 ErrorCode);
+ int interpret_exit_nok();
+
+ /**
+ * Interpreted program instruction:
+ * Define a subroutine in an interpreted operation.
+ *
+ * @param SubroutineNumber the subroutine number.
+ * @return -1 if unsuccessful.
+ */
+ int def_subroutine(int SubroutineNumber);
+
+ /**
+ * Interpreted program instruction:
+ * Call a subroutine.
+ *
+ * @param Subroutine the subroutine to call.
+ * @return -1 if unsuccessful.
+ */
+ int call_sub(Uint32 Subroutine);
+
+ /**
+ * Interpreted program instruction:
+ * End a subroutine.
+ *
+ * @return -1 if unsuccessful.
+ */
+ int ret_sub();
+
+ /** @} *********************************************************************/
+
+ /**
+ * @name Error Handling
+ * @{
+ */
+
+ /**
+ * Get the latest error code.
+ *
+ * @return error code.
+ */
+ const NdbError & getNdbError() const;
+
+ /**
+ * Get the method number where the error occured.
+ *
+ * @return method number where the error occured.
+ */
+ int getNdbErrorLine();
+
+ /** @} *********************************************************************/
+
+protected:
+/******************************************************************************
+ * These are the methods used to create and delete the NdbOperation objects.
+ *****************************************************************************/
+ NdbOperation(Ndb* aNdb);
+ virtual ~NdbOperation();
+
+ bool needReply();
+/******************************************************************************
+ * These methods are service routines used by the other NDB API classes.
+ *****************************************************************************/
+//--------------------------------------------------------------
+// Initialise after allocating operation to a transaction
+//--------------------------------------------------------------
+ int init(class NdbTableImpl*, NdbConnection* aCon);
+
+ void initScan(); // Initialise after allocating operation
+ // to a scan transaction
+ virtual void releaseScan(); // Release scan parts of transaction
+ void releaseSignals();
+ void releaseScanSignals();
+ void prepareNextScanResult();
+
+ // Common part for Read and Exclusive
+ int openScan(Uint32 aParallelism, bool, bool, bool);
+
+ void next(NdbOperation*); // Set next pointer
+
+ NdbOperation* next(); // Get next pointer
+
+ OperationStatus Status(); // Read the status information
+
+ void Status(OperationStatus); // Set the status information
+
+ OperationType RequestType();
+
+ void NdbCon(NdbConnection*); // Set reference to connection
+ // object.
+
+ virtual void release(); // Release all operations
+ // connected to
+ // the operations object.
+ void setStartIndicator();
+
+ void setCommitIndicator(CommitType aCommitType);
+
+/******************************************************************************
+ * The methods below is the execution part of the NdbOperation
+ * class. This is where the NDB signals are sent and received. The
+ * operation can send TC[KEY/INDX]REQ, [INDX]ATTRINFO.
+ * It can receive TC[KEY/INDX]CONF, TC[KEY/INDX]REF, [INDX]ATTRINFO.
+ * When an operation is received in its fulness or a refuse message
+ * was sent, then the connection object is told about this situation.
+ *****************************************************************************/
+
+ int doSend(int ProcessorId, Uint32 lastFlag);
+ int doSendScan(int ProcessorId);
+
+ int prepareSendScan(Uint32 TC_ConnectPtr,
+ Uint64 TransactionId);
+
+ virtual int prepareSend(Uint32 TC_ConnectPtr,
+ Uint64 TransactionId);
+ virtual void setLastFlag(NdbApiSignal* signal, Uint32 lastFlag);
+
+ int prepareSendInterpreted(); // Help routine to prepare*
+
+ void TCOPCONF(Uint32 anNdbColumnImplLen); // Handle TC[KEY/INDX]CONF signal
+
+ int receiveTCKEYREF(NdbApiSignal*);
+
+
+ int receiveTRANSID_AI(const Uint32* aDataPtr, Uint32 aDataLength);
+ int receiveREAD_CONF(const Uint32* aDataPtr, Uint32 aDataLength);
+
+
+ int checkMagicNumber(); // Verify correct object
+
+ int checkState_TransId(NdbApiSignal* aSignal);
+
+/******************************************************************************
+ * These are support methods only used locally in this class.
+******************************************************************************/
+
+ virtual int equal_impl(const NdbColumnImpl* anAttrObject,
+ const char* aValue,
+ Uint32 len);
+ NdbRecAttr* getValue(const NdbColumnImpl* anAttrObject, char* aValue = NULL);
+ int setValue(const NdbColumnImpl* anAttrObject, const char* aValue, Uint32 len);
+ int incValue(const NdbColumnImpl* anAttrObject, Uint32 aValue);
+ int incValue(const NdbColumnImpl* anAttrObject, Uint64 aValue);
+ int subValue(const NdbColumnImpl* anAttrObject, Uint32 aValue);
+ int subValue(const NdbColumnImpl* anAttrObject, Uint64 aValue);
+ int read_attr(const NdbColumnImpl* anAttrObject, Uint32 RegDest);
+ int write_attr(const NdbColumnImpl* anAttrObject, Uint32 RegSource);
+ int branch_reg_reg(Uint32 type, Uint32, Uint32, Uint32);
+ int branch_col(Uint32 type, Uint32, const char *, Uint32, bool, Uint32 Label);
+ int branch_col_null(Uint32 type, Uint32 col, Uint32 Label);
+ int setBound(const NdbColumnImpl* anAttrObject, int type, const void* aValue, Uint32 len);
+
+ // Handle ATTRINFO signals
+ int receiveREAD_AI(Uint32* aDataPtr, Uint32 aLength);
+
+ int insertATTRINFO(Uint32 aData);
+ int insertATTRINFOloop(const Uint32* aDataPtr, Uint32 aLength);
+ int getFirstATTRINFOScan();
+ int saveBoundATTRINFO();
+
+ int insertKEYINFO(const char* aValue,
+ Uint32 aStartPosition,
+ Uint32 aKeyLenInByte,
+ Uint32 anAttrBitsInLastWord);
+
+ virtual void setErrorCode(int aErrorCode);
+ virtual void setErrorCodeAbort(int aErrorCode);
+
+ void handleFailedAI_ElemLen(); // When not all attribute data
+ // were received
+
+ int incCheck(const NdbColumnImpl* anAttrObject);
+ int initial_interpreterCheck();
+ int intermediate_interpreterCheck();
+ int read_attrCheck(const NdbColumnImpl* anAttrObject);
+ int write_attrCheck(const NdbColumnImpl* anAttrObject);
+ int labelCheck();
+ int insertCall(Uint32 aCall);
+ int insertBranch(Uint32 aBranch);
+
+ Uint32 ptr2int() { return theReceiver.getId(); };
+
+ NdbOperation*
+ takeOverScanOp(OperationType opType, NdbConnection* updateTrans);
+
+/******************************************************************************
+ * These are the private variables that are defined in the operation objects.
+ *****************************************************************************/
+
+ NdbReceiver theReceiver;
+
+ NdbError theError; // Errorcode
+ int theErrorLine; // Error line
+
+ Ndb* theNdb; // Point back to the Ndb object.
+ NdbConnection* theNdbCon; // Point back to the connection object.
+ NdbOperation* theNext; // Next pointer to operation.
+ NdbOperation* theNextScanOp;
+ NdbApiSignal* theTCREQ; // The TC[KEY/INDX]REQ signal object
+ NdbApiSignal* theFirstATTRINFO; // The first ATTRINFO signal object
+ NdbApiSignal* theCurrentATTRINFO; // The current ATTRINFO signal object
+ Uint32 theTotalCurrAI_Len; // The total number of attribute info
+ // words currently defined
+ Uint32 theAI_LenInCurrAI; // The number of words defined in the
+ // current ATTRINFO signal
+ NdbApiSignal* theFirstKEYINFO; // The first KEYINFO signal object
+ NdbApiSignal* theLastKEYINFO; // The first KEYINFO signal object
+
+ NdbRecAttr* theFirstRecAttr; // The first receive attribute object
+ NdbRecAttr* theCurrentRecAttr; // The current receive attribute object
+
+ class NdbLabel* theFirstLabel;
+ class NdbLabel* theLastLabel;
+ class NdbBranch* theFirstBranch;
+ class NdbBranch* theLastBranch;
+ class NdbCall* theFirstCall;
+ class NdbCall* theLastCall;
+ class NdbSubroutine* theFirstSubroutine;
+ class NdbSubroutine* theLastSubroutine;
+ Uint32 theNoOfLabels;
+ Uint32 theNoOfSubroutines;
+
+ Uint32* theKEYINFOptr; // Pointer to where to write KEYINFO
+ Uint32* theATTRINFOptr; // Pointer to where to write ATTRINFO
+
+ Uint32 theTotalRecAI_Len; // The total length received according
+ // to the TCKEYCONF signal
+ Uint32 theCurrRecAI_Len; // The currently received length
+ Uint32 theAI_ElementLen; // How many words long is this element
+ Uint32* theCurrElemPtr; // The current pointer to the element
+ //Uint32 theTableId; // Table id.
+ //Uint32 theAccessTableId; // The id of table for initial access,
+ // changed by NdbIndexOperation
+ //Uint32 theSchemaVersion; // The schema version on the table.
+ class NdbTableImpl* m_currentTable; // The current table
+ class NdbTableImpl* m_accessTable;
+
+ // Set to TRUE when a tuple key attribute has been defined.
+ // A tuple key is allowed to consist of 64 attributes.
+ Uint32 theTupleKeyDefined[MAXNROFTUPLEKEY][3];
+
+ Uint32 theTotalNrOfKeyWordInSignal; // The total number of
+ // keyword in signal.
+
+ Uint32 theTupKeyLen; // Length of the tuple key in words
+ Uint32 theNoOfTupKeyDefined; // The number of tuple key attributes
+ // currently defined
+ OperationType theOperationType; // Read Request, Update Req......
+
+ Uint8 theLockMode; // Can be set to WRITE if read operation
+ OperationStatus theStatus; // The status of the operation.
+ Uint32 theMagicNumber; // Magic number to verify that object
+ // is correct
+ Uint32 theScanInfo; // Scan info bits (take over flag etc)
+ Uint32 theDistrKeySize; // Distribution Key size if used
+ Uint32 theDistributionGroup; // Distribution Group if used
+
+ Uint32 theSubroutineSize; // Size of subroutines for interpretation
+ Uint32 theInitialReadSize; // Size of initial reads for interpretation
+ Uint32 theInterpretedSize; // Size of interpretation
+ Uint32 theFinalUpdateSize; // Size of final updates for interpretation
+ Uint32 theFinalReadSize; // Size of final reads for interpretation
+
+ Uint8 theStartIndicator; // Indicator of whether start operation
+ Uint8 theCommitIndicator; // Indicator of whether commit operation
+ Uint8 theSimpleIndicator; // Indicator of whether simple operation
+ Uint8 theDirtyIndicator; // Indicator of whether dirty operation
+ Uint8 theInterpretIndicator; // Indicator of whether interpreted operation
+ Uint8 theDistrGroupIndicator; // Indicates whether distribution grp is used
+ Uint8 theDistrGroupType; // Type of distribution group used
+ Uint8 theDistrKeyIndicator; // Indicates whether distr. key is used
+
+ Uint16 m_tcReqGSN;
+ Uint16 m_keyInfoGSN;
+ Uint16 m_attrInfoGSN;
+
+ // Scan related variables
+ Uint32 theParallelism;
+ NdbScanReceiver** theScanReceiversArray;
+ NdbApiSignal* theSCAN_TABREQ;
+ NdbApiSignal* theFirstSCAN_TABINFO_Send;
+ NdbApiSignal* theLastSCAN_TABINFO_Send;
+ NdbApiSignal* theFirstSCAN_TABINFO_Recv;
+ NdbApiSignal* theLastSCAN_TABINFO_Recv;
+ NdbApiSignal* theSCAN_TABCONF_Recv;
+ // saveBoundATTRINFO() moves ATTRINFO here when setBound() is ready
+ NdbApiSignal* theBoundATTRINFO;
+ Uint32 theTotalBoundAI_Len;
+
+};
+
+inline
+int
+NdbOperation::checkMagicNumber()
+{
+ if (theMagicNumber != 0xABCDEF01){
+#ifdef NDB_NO_DROPPED_SIGNAL
+ abort();
+#endif
+ return -1;
+ }
+ return 0;
+}
+
+inline
+void
+NdbOperation::setStartIndicator()
+{
+ theStartIndicator = 1;
+}
+
+#if 0
+inline
+void
+NdbOperation::setCommitIndicator(CommitType aTypeOfCommit)
+{
+ theCommitIndicator = 1;
+ theCommitType = (Uint8)aTypeOfCommit;
+}
+#endif
+
+inline
+int
+NdbOperation::getNdbErrorLine()
+{
+ return theErrorLine;
+}
+
+/******************************************************************************
+void next(NdbOperation* aNdbOperation);
+
+Parameters: aNdbOperation: Pointers to the NdbOperation object.
+Remark: Set the next variable of the operation object.
+******************************************************************************/
+inline
+void
+NdbOperation::next(NdbOperation* aNdbOperation)
+{
+ theNext = aNdbOperation;
+}
+
+/******************************************************************************
+NdbOperation* next();
+
+Return Value: Return next pointer to NdbOperation object.
+Remark: Get the next variable of the operation object.
+******************************************************************************/
+inline
+NdbOperation*
+NdbOperation::next()
+{
+ return theNext;
+}
+
+/******************************************************************************
+OperationStatus Status();
+
+Return Value Return the OperationStatus.
+Parameters: aStatus: The status.
+Remark: Sets Operation status.
+******************************************************************************/
+inline
+OperationStatus
+NdbOperation::Status()
+{
+ return theStatus;
+}
+
+/******************************************************************************
+void Status(OperationStatus aStatus);
+
+Parameters: aStatus: The status.
+Remark: Sets Operation
+ status.
+******************************************************************************/
+inline
+void
+NdbOperation::Status( OperationStatus aStatus )
+{
+ theStatus = aStatus;
+}
+
+/******************************************************************************
+void NdbCon(NdbConnection* aNdbCon);
+
+Parameters: aNdbCon: Pointers to NdbConnection object.
+Remark: Set the reference to the connection in the operation object.
+******************************************************************************/
+inline
+void
+NdbOperation::NdbCon(NdbConnection* aNdbCon)
+{
+ theNdbCon = aNdbCon;
+}
+
+/******************************************************************************
+OperationType RequestType();
+
+Remark: Return the request typ of the operation..
+******************************************************************************/
+inline
+OperationType
+NdbOperation::RequestType()
+{
+ return theOperationType;
+}
+
+inline
+int
+NdbOperation::equal(const char* anAttrName, Int32 aPar)
+{
+ return equal(anAttrName, (const char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::equal(const char* anAttrName, Uint32 aPar)
+{
+ return equal(anAttrName, (const char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::equal(const char* anAttrName, Int64 aPar)
+{
+ return equal(anAttrName, (const char*)&aPar, (Uint32)8);
+}
+
+inline
+int
+NdbOperation::equal(const char* anAttrName, Uint64 aPar)
+{
+ return equal(anAttrName, (const char*)&aPar, (Uint32)8);
+}
+
+inline
+int
+NdbOperation::equal(Uint32 anAttrId, Int32 aPar)
+{
+ return equal(anAttrId, (const char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::equal(Uint32 anAttrId, Uint32 aPar)
+{
+ return equal(anAttrId, (const char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::equal(Uint32 anAttrId, Int64 aPar)
+{
+ return equal(anAttrId, (const char*)&aPar, (Uint32)8);
+}
+
+inline
+int
+NdbOperation::equal(Uint32 anAttrId, Uint64 aPar)
+{
+ return equal(anAttrId, (const char*)&aPar, (Uint32)8);
+}
+
+inline
+int
+NdbOperation::setValue(const char* anAttrName, Int32 aPar)
+{
+ return setValue(anAttrName, (const char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::setValue(const char* anAttrName, Uint32 aPar)
+{
+ return setValue(anAttrName, (const char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::setValue(const char* anAttrName, Int64 aPar)
+{
+ return setValue(anAttrName, (const char*)&aPar, (Uint32)8);
+}
+
+inline
+int
+NdbOperation::setValue(const char* anAttrName, Uint64 aPar)
+{
+ return setValue(anAttrName, (const char*)&aPar, (Uint32)8);
+}
+
+inline
+int
+NdbOperation::setValue(const char* anAttrName, float aPar)
+{
+ return setValue(anAttrName, (const char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::setValue(const char* anAttrName, double aPar)
+{
+ return setValue(anAttrName, (const char*)&aPar, (Uint32)8);
+}
+
+inline
+int
+NdbOperation::setValue(Uint32 anAttrId, Int32 aPar)
+{
+ return setValue(anAttrId, (const char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::setValue(Uint32 anAttrId, Uint32 aPar)
+{
+ return setValue(anAttrId, (const char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::setValue(Uint32 anAttrId, Int64 aPar)
+{
+ return setValue(anAttrId, (const char*)&aPar, (Uint32)8);
+}
+
+inline
+int
+NdbOperation::setValue(Uint32 anAttrId, Uint64 aPar)
+{
+ return setValue(anAttrId, (const char*)&aPar, (Uint32)8);
+}
+
+inline
+int
+NdbOperation::setValue(Uint32 anAttrId, float aPar)
+{
+ return setValue(anAttrId, (char*)&aPar, (Uint32)4);
+}
+
+inline
+int
+NdbOperation::setValue(Uint32 anAttrId, double aPar)
+{
+ return setValue(anAttrId, (const char*)&aPar, (Uint32)8);
+}
+
+#endif
+
+
diff --git a/ndb/include/ndbapi/NdbPool.hpp b/ndb/include/ndbapi/NdbPool.hpp
new file mode 100644
index 00000000000..64cba5a008c
--- /dev/null
+++ b/ndb/include/ndbapi/NdbPool.hpp
@@ -0,0 +1,35 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+class Ndb;
+class NdbPool;
+
+bool
+create_instance(Uint32 max_ndb_objects,
+ Uint32 no_conn_obj,
+ Uint32 init_no_ndb_objects);
+
+void
+drop_instance();
+
+Ndb*
+get_ndb_object(Uint32 &hint_id,
+ const char* a_catalog_name,
+ const char* a_schema_name);
+
+void
+return_ndb_object(Ndb* returned_object, Uint32 id);
+
diff --git a/ndb/include/ndbapi/NdbRecAttr.hpp b/ndb/include/ndbapi/NdbRecAttr.hpp
new file mode 100644
index 00000000000..a5595096bf6
--- /dev/null
+++ b/ndb/include/ndbapi/NdbRecAttr.hpp
@@ -0,0 +1,512 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbRecAttr_H
+#define NdbRecAttr_H
+
+#include <stdlib.h>
+
+#include <ndb_types.h>
+#include <NdbDictionary.hpp>
+#include "AttrType.hpp"
+
+class NdbOperation;
+class AttrInfo;
+
+/**
+ * @class NdbRecAttr
+ * @brief Contains value of an attribute.
+ *
+ * NdbRecAttr objects are used to store the attribute value
+ * after retrieving the value from the NDB Cluster using the method
+ * NdbOperation::getValue. The objects are allocated by the NDB API.
+ * An example application program follows:
+ *
+ * @code
+ * MyRecAttr = MyOperation->getValue("ATTR2", NULL);
+ * if (MyRecAttr == NULL) goto error;
+ *
+ * if (MyConnection->execute(Commit) == -1) goto error;
+ *
+ * ndbout << MyRecAttr->u_32_value();
+ * @endcode
+ * For more examples, see
+ * @ref ndbapi_example1.cpp and
+ * @ref ndbapi_example2.cpp.
+ *
+ * @note The NdbRecAttr object is instantiated with its value when
+ * NdbConnection::execute is called. Before this, the value is
+ * undefined. (NdbRecAttr::isNULL can be used to check
+ * if the value is defined or not.)
+ * This means that an NdbRecAttr object only has valid information
+ * between the time of calling NdbConnection::execute and
+ * the time of Ndb::closeTransaction.
+ * The value of the null indicator is -1 until the
+ * NdbConnection::execute method have been called.
+ *
+ * For simple types, there are methods which directly getting the value
+ * from the NdbRecAttr object.
+ *
+ * To get a reference to the value, there are two methods:
+ * NdbRecAttr::aRef (memory is released by NDB API) and
+ * NdbRecAttr::getAttributeObject (memory must be released
+ * by application program).
+ * The two methods may return different pointers.
+ *
+ * There are also methods to check attribute type, attribute size and
+ * array size.
+ * The method NdbRecAttr::arraySize returns the number of elements in the
+ * array (where each element is of size given by NdbRecAttr::attrSize).
+ * The NdbRecAttr::arraySize method is needed when reading variable-sized
+ * attributes.
+ *
+ * @note Variable-sized attributes are not yet supported.
+ */
+class NdbRecAttr
+{
+ friend class NdbOperation;
+ friend class NdbEventOperationImpl;
+ friend class NdbScanReceiver;
+ friend class Ndb;
+
+public:
+ /**
+ * @name Getting meta information
+ * @{
+ */
+ const NdbDictionary::Column * getColumn() const;
+
+ /**
+ * Get attribute type.
+ *
+ * @return Type of attribute: { Signed, UnSigned, Float, String }
+ */
+ AttrType attrType() const ;
+ NdbDictionary::Column::Type getType() const;
+
+ /**
+ * Get attribute (element) size in bytes.
+ *
+ * @note For arrays, the method only gives the size of an element.
+ * The total attribute size is calculated by
+ * multiplying this value with the value
+ * returned by NdbRecAttr::arraySize.
+ *
+ * @return Attribute size in 32 bit unsigned int.
+ */
+ Uint32 attrSize() const ;
+
+ /**
+ * Get array size of attribute.
+ * For variable-sized arrays this method returns the
+ * size of the attribute read.
+ *
+ * @return array size in 32 unsigned int.
+ */
+ Uint32 arraySize() const ;
+ Uint32 getLength() const ;
+
+ /** @} *********************************************************************/
+ /**
+ * @name Getting stored value
+ * @{
+ */
+
+ /**
+ * Check if attribute value is NULL.
+ *
+ * @return -1 = Not defined (Failure or
+ * NdbConnection::execute not yet called).<br>
+ * 0 = Attribute value is defined, but not equal to NULL.<br>
+ * 1 = Attribute value is defined and equal to NULL.
+ */
+ int isNULL() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return 64 bit long value.
+ */
+ Int64 int64_value() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return 32 bit int value.
+ */
+ Int32 int32_value() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return Short value.
+ */
+ short short_value() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return Char value.
+ */
+ char char_value() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return 64 bit unsigned value.
+ */
+ Uint64 u_64_value() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return 32 bit unsigned value.
+ */
+ Uint32 u_32_value() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return Unsigned short value.
+ */
+ Uint16 u_short_value() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return Unsigned char value.
+ */
+ Uint8 u_char_value() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return Float value.
+ */
+ float float_value() const;
+
+ /**
+ * Get value stored in NdbRecAttr object.
+ *
+ * @return Double value.
+ */
+ double double_value() const;
+
+ /** @} *********************************************************************/
+ /**
+ * @name Getting reference to stored value
+ * @{
+ */
+
+ /**
+ * Get reference to attribute value.
+ *
+ * Returns a char*-pointer to the value.
+ * The pointer is aligned appropriately for the data type.
+ * The memory is released when Ndb::closeTransaction is executed
+ * for the transaction which read the value.
+ *
+ * @note The memory is released by NDB API.
+ *
+ * @note The pointer to the attribute value stored in an NdbRecAttr
+ * object (i.e. the pointer returned by aRef) is constant.
+ * This means that this method can be called anytime after
+ * NdbOperation::getValue has been called.
+ *
+ * @return Pointer to attribute value.
+ */
+ char* aRef() const;
+
+ /** @} *********************************************************************/
+
+ /**
+ * Make a copy of RecAttr object including all data.
+ *
+ * @note Copy needs to be deleted by application program.
+ */
+ NdbRecAttr * clone() const;
+
+ /**
+ * Destructor
+ *
+ * @note You should only delete RecAttr-copies,
+ * i.e. objects that has been cloned.
+ */
+ ~NdbRecAttr();
+private:
+ NdbRecAttr();
+
+ Uint32 attrId() const; /* Get attribute id */
+ void setNULL(); /* Set NULL indicator */
+ void setNotNULL(); /* Set Not NULL indicator */
+ void setUNDEFINED(); /* Set UNDEFINED indicator */
+
+ void release(); /* Release memory if allocated */
+ void init(); /* Initialise object when allocated */
+
+ void next(NdbRecAttr* aRecAttr);
+ NdbRecAttr* next() const;
+
+ int setup(const class NdbColumnImpl* anAttrInfo, char* aValue);
+ /* Set up attributes and buffers */
+ bool copyoutRequired() const; /* Need to copy data to application */
+ void copyout(); /* Copy from storage to application */
+
+ Uint64 theStorage[4]; /* The data storage here if <= 32 bytes */
+ Uint64* theStorageX; /* The data storage here if > 32 bytes */
+ char* theValue; /* The data storage in the application */
+ void* theRef; /* Pointer to one of above */
+
+ NdbRecAttr* theNext; /* Next pointer */
+ Uint32 theAttrId; /* The attribute id */
+
+ int theNULLind;
+ Uint32 theAttrSize;
+ Uint32 theArraySize;
+ const NdbDictionary::Column* m_column;
+};
+
+inline
+NdbDictionary::Column::Type
+NdbRecAttr::getType() const {
+ return m_column->getType();
+}
+
+inline
+const NdbDictionary::Column *
+NdbRecAttr::getColumn() const {
+ return m_column;
+}
+
+inline
+Uint32
+NdbRecAttr::attrSize() const {
+
+ switch(getType()){
+ case NdbDictionary::Column::Int:
+ case NdbDictionary::Column::Unsigned:
+ case NdbDictionary::Column::Float:
+ return 4;
+ case NdbDictionary::Column::Decimal:
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Binary:
+ case NdbDictionary::Column::Varbinary:
+ return 1;
+ case NdbDictionary::Column::Bigint:
+ case NdbDictionary::Column::Bigunsigned:
+ case NdbDictionary::Column::Double:
+ case NdbDictionary::Column::Datetime:
+ return 8;
+ case NdbDictionary::Column::Timespec:
+ return 12;
+ case NdbDictionary::Column::Undefined:
+ default:
+ return 0;
+ }
+}
+
+inline
+AttrType
+NdbRecAttr::attrType() const {
+ switch(getType()){
+ case NdbDictionary::Column::Bigint:
+ case NdbDictionary::Column::Int:
+ return Signed;
+ case NdbDictionary::Column::Bigunsigned:
+ case NdbDictionary::Column::Unsigned:
+ return UnSigned;
+ case NdbDictionary::Column::Float:
+ case NdbDictionary::Column::Decimal:
+ case NdbDictionary::Column::Double:
+ return Float;
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Binary:
+ case NdbDictionary::Column::Varbinary:
+ return String;
+ case NdbDictionary::Column::Datetime:
+ case NdbDictionary::Column::Timespec:
+ case NdbDictionary::Column::Undefined:
+ default:
+ return NoAttrTypeDef;
+ }
+}
+
+inline
+Uint32
+NdbRecAttr::arraySize() const
+{
+ return theArraySize;
+}
+
+inline
+Int64
+NdbRecAttr::int64_value() const
+{
+ return *(Int64*)theRef;
+}
+
+inline
+Int32
+NdbRecAttr::int32_value() const
+{
+ return *(Int32*)theRef;
+}
+
+inline
+short
+NdbRecAttr::short_value() const
+{
+ return *(short*)theRef;
+}
+
+inline
+char
+NdbRecAttr::char_value() const
+{
+ return *(char*)theRef;
+}
+
+inline
+Uint64
+NdbRecAttr::u_64_value() const
+{
+ return *(Uint64*)theRef;
+}
+
+inline
+Uint32
+NdbRecAttr::u_32_value() const
+{
+ return *(Uint32*)theRef;
+}
+
+inline
+Uint16
+NdbRecAttr::u_short_value() const
+{
+ return *(Uint16*)theRef;
+}
+
+inline
+Uint8
+NdbRecAttr::u_char_value() const
+{
+ return *(Uint8*)theRef;
+}
+
+inline
+float
+NdbRecAttr::float_value() const
+{
+ return *(float*)theRef;
+}
+
+inline
+double
+NdbRecAttr::double_value() const
+{
+ return *(double*)theRef;
+}
+
+inline
+void
+NdbRecAttr::release()
+{
+ if (theStorageX != NULL) {
+ delete [] theStorageX;
+ theStorageX = NULL;
+ }
+}
+
+inline
+void
+NdbRecAttr::init()
+{
+ theStorageX = NULL;
+ theValue = NULL;
+ theRef = NULL;
+ theNext = NULL;
+ theAttrId = 0xFFFF;
+ theNULLind = -1;
+}
+
+inline
+void
+NdbRecAttr::next(NdbRecAttr* aRecAttr)
+{
+ theNext = aRecAttr;
+}
+
+inline
+NdbRecAttr*
+NdbRecAttr::next() const
+{
+ return theNext;
+}
+
+inline
+char*
+NdbRecAttr::aRef() const
+{
+ return (char*)theRef;
+}
+
+inline
+bool
+NdbRecAttr::copyoutRequired() const
+{
+ return theRef != theValue && theValue != NULL;
+}
+
+inline
+Uint32
+NdbRecAttr::attrId() const
+{
+ return theAttrId;
+}
+
+inline
+void
+NdbRecAttr::setNULL()
+{
+ theNULLind = 1;
+}
+
+inline
+void
+NdbRecAttr::setNotNULL()
+{
+ theNULLind = 0;
+}
+
+inline
+void
+NdbRecAttr::setUNDEFINED()
+{
+ theNULLind = -1;
+}
+
+inline
+int
+NdbRecAttr::isNULL() const
+{
+ return theNULLind;
+}
+
+#endif
+
diff --git a/ndb/include/ndbapi/NdbReceiver.hpp b/ndb/include/ndbapi/NdbReceiver.hpp
new file mode 100644
index 00000000000..952803f8e70
--- /dev/null
+++ b/ndb/include/ndbapi/NdbReceiver.hpp
@@ -0,0 +1,72 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbReceiver_H
+#define NdbReceiver_H
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL // Not part of public interface
+
+#include <stdlib.h>
+
+class Ndb;
+class NdbReceiver
+{
+public:
+ enum ReceiverType { NDB_UNINITIALIZED,
+ NDB_OPERATION = 1,
+ NDB_SCANRECEIVER = 2,
+ NDB_INDEX_OPERATION = 3
+ };
+
+ NdbReceiver(Ndb *aNdb);
+ void init(ReceiverType type, void* owner);
+ ~NdbReceiver();
+
+ Uint32 getId(){
+ return m_id;
+ }
+
+ ReceiverType getType(){
+ return m_type;
+ }
+
+ void* getOwner(){
+ return m_owner;
+ }
+
+ bool checkMagicNumber() const;
+
+private:
+ Uint32 theMagicNumber;
+ Ndb* m_ndb;
+ Uint32 m_id;
+ ReceiverType m_type;
+ void* m_owner;
+};
+
+inline
+bool
+NdbReceiver::checkMagicNumber() const {
+ bool retVal = (theMagicNumber == 0x11223344);
+#ifdef NDB_NO_DROPPED_SIGNAL
+ if(!retVal){
+ abort();
+ }
+#endif
+ return retVal;
+}
+
+#endif
+#endif
diff --git a/ndb/include/ndbapi/NdbResultSet.hpp b/ndb/include/ndbapi/NdbResultSet.hpp
new file mode 100644
index 00000000000..d48df01214e
--- /dev/null
+++ b/ndb/include/ndbapi/NdbResultSet.hpp
@@ -0,0 +1,114 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbResultSet.hpp
+ * Include:
+ * Link:
+ * Author: Martin Sköld
+ * Date: 2002-04-01
+ * Version: 0.1
+ * Description: Cursor class
+ * Documentation:
+ * Adjust: 2002-04-01 Martin Sköld First version.
+ ****************************************************************************/
+
+#ifndef NdbResultSet_H
+#define NdbResultSet_H
+
+
+#include <NdbCursorOperation.hpp>
+#include <NdbIndexOperation.hpp>
+#include <NdbScanOperation.hpp>
+
+/**
+ * @class NdbResultSet
+ * @brief NdbResultSet contains a NdbCursorOperation.
+ */
+class NdbResultSet
+{
+ friend class NdbCursorOperation;
+
+public:
+
+ /**
+ * Get the next tuple in a scan transaction.
+ *
+ * After each call to NdbResult::nextResult
+ * the buffers and NdbRecAttr objects defined in
+ * NdbOperation::getValue are updated with values
+ * from the scanned tuple.
+ *
+ * @param fetchAllowed If set to false, then fetching is disabled
+ *
+ * The NDB API will contact the NDB Kernel for more tuples
+ * when necessary to do so unless you set the fetchAllowed
+ * to false.
+ * This will force NDB to process any records it
+ * already has in it's caches. When there are no more cached
+ * records it will return 2. You must then call nextResult
+ * with fetchAllowed = true in order to contact NDB for more
+ * records.
+ *
+ * fetchAllowed = false is useful when you want to update or
+ * delete all the records fetched in one transaction(This will save a
+ * lot of round trip time and make updates or deletes of scanned
+ * records a lot faster).
+ * While nextResult(false)
+ * returns 0 take over the record to another transaction. When
+ * nextResult(false) returns 2 you must execute and commit the other
+ * transaction. This will cause the locks to be transferred to the
+ * other transaction, updates or deletes will be made and then the
+ * locks will be released.
+ * After that, call nextResult(true) which will fetch new records and
+ * cache them in the NdbApi.
+ *
+ * @note If you don't take over the records to another transaction the
+ * locks on those records will be released the next time NDB Kernel
+ * is contacted for more records.
+ *
+ * @note Please contact for examples of efficient scan
+ * updates and deletes.
+ *
+ * @note See ndb/examples/ndbapi_scan_example for usage.
+ *
+ * @return
+ * - -1: if unsuccessful,<br>
+ * - 0: if another tuple was received, and<br>
+ * - 1: if there are no more tuples to scan.
+ * - 2: if there are no more cached records in NdbApi
+ */
+ int nextResult(bool fetchAllowed = true);
+
+ void close();
+
+ NdbOperation* updateTuple();
+ NdbOperation* updateTuple(NdbConnection* takeOverTransaction);
+
+ int deleteTuple();
+ int deleteTuple(NdbConnection* takeOverTransaction);
+
+private:
+ NdbResultSet(NdbCursorOperation*);
+
+ ~NdbResultSet();
+
+ void init();
+
+ NdbCursorOperation* m_operation;
+};
+
+#endif
diff --git a/ndb/include/ndbapi/NdbScanFilter.hpp b/ndb/include/ndbapi/NdbScanFilter.hpp
new file mode 100644
index 00000000000..9f8a01b1059
--- /dev/null
+++ b/ndb/include/ndbapi/NdbScanFilter.hpp
@@ -0,0 +1,177 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_SCAN_FILTER_HPP
+#define NDB_SCAN_FILTER_HPP
+
+#include <ndb_types.h>
+
+/**
+ * @class NdbScanFilter
+ * @brief A simple way to specify filters for scan operations
+ *
+ * @note This filter interface is under development and may change in
+ * the future!
+ *
+ */
+class NdbScanFilter {
+public:
+ /**
+ * Constructor
+ * @param op The NdbOperation that the filter belongs to (is applied to).
+ */
+ NdbScanFilter(class NdbOperation * op);
+ ~NdbScanFilter();
+
+ /**
+ * Group operators
+ */
+ enum Group {
+ AND = 1, ///< (x1 AND x2 AND x3)
+ OR = 2, ///< (x1 OR x2 OR X3)
+ NAND = 3, ///< NOT (x1 AND x2 AND x3)
+ NOR = 4 ///< NOT (x1 OR x2 OR x3)
+ };
+
+ /**
+ * @name Grouping
+ * @{
+ */
+
+ /**
+ * Begin of compound.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int begin(Group group = AND);
+
+ /**
+ * End of compound.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int end();
+
+ /** @} *********************************************************************/
+
+ /**
+ * <i>Explanation missing</i>
+ */
+ int istrue();
+
+ /**
+ * <i>Explanation missing</i>
+ */
+ int isfalse();
+
+ /**
+ * @name Integer Comparators
+ * @{
+ */
+ /** Compare column value with integer for equal
+ * ®return 0 if successful, -1 otherwize
+ */
+ int eq(int ColId, Uint32 value);
+ /** Compare column value with integer for not equal.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int ne(int ColId, Uint32 value);
+ /** Compare column value with integer for less than.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int lt(int ColId, Uint32 value);
+ /** Compare column value with integer for less than or equal.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int le(int ColId, Uint32 value);
+ /** Compare column value with integer for greater than.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int gt(int ColId, Uint32 value);
+ /** Compare column value with integer for greater than or equal.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int ge(int ColId, Uint32 value);
+
+ /** Compare column value with integer for equal. 64-bit.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int eq(int ColId, Uint64 value);
+ /** Compare column value with integer for not equal. 64-bit.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int ne(int ColId, Uint64 value);
+ /** Compare column value with integer for less than. 64-bit.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int lt(int ColId, Uint64 value);
+ /** Compare column value with integer for less than or equal. 64-bit.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int le(int ColId, Uint64 value);
+ /** Compare column value with integer for greater than. 64-bit.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int gt(int ColId, Uint64 value);
+ /** Compare column value with integer for greater than or equal. 64-bit.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int ge(int ColId, Uint64 value);
+ /** @} *********************************************************************/
+
+ /** Check if column value is NULL */
+ int isnull(int ColId);
+ /** Check if column value is non-NULL */
+ int isnotnull(int ColId);
+
+ /**
+ * @name String Comparators
+ * @{
+ */
+ /**
+ * Compare string against a Char or Varchar column.
+ *
+ * By default Char comparison blank-pads both sides to common length.
+ * Varchar comparison does not blank-pad.
+ *
+ * The extra <i>nopad</i> argument can be used to
+ * force non-padded comparison for a Char column.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int eq(int ColId, const char * val, Uint32 len, bool nopad=false);
+ int ne(int ColId, const char * val, Uint32 len, bool nopad=false);
+ int lt(int ColId, const char * val, Uint32 len, bool nopad=false);
+ int le(int ColId, const char * val, Uint32 len, bool nopad=false);
+ int gt(int ColId, const char * val, Uint32 len, bool nopad=false);
+ int ge(int ColId, const char * val, Uint32 len, bool nopad=false);
+
+ /**
+ * Like comparison operator.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int like(int ColId, const char * val, Uint32 len, bool nopad=false);
+ /**
+ * Notlike comparison operator.
+ * ®return 0 if successful, -1 otherwize
+ */
+ int notlike(int ColId, const char * val, Uint32 len, bool nopad=false);
+ /** @} *********************************************************************/
+
+private:
+ friend class NdbScanFilterImpl;
+ class NdbScanFilterImpl & m_impl;
+ NdbScanFilter& operator=(const NdbScanFilter&); ///< Defined not implemented
+};
+
+#endif
diff --git a/ndb/include/ndbapi/NdbScanOperation.hpp b/ndb/include/ndbapi/NdbScanOperation.hpp
new file mode 100644
index 00000000000..2a27d8b34a1
--- /dev/null
+++ b/ndb/include/ndbapi/NdbScanOperation.hpp
@@ -0,0 +1,248 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbScanOperation.hpp
+ * Include:
+ * Link:
+ * Author: Martin Sköld
+ * Date: 2002-04-01
+ * Version: 0.1
+ * Description: Table scan support
+ * Documentation:
+ * Adjust: 2002-04-01 Martin Sköld First version.
+ ****************************************************************************/
+
+#ifndef NdbScanOperation_H
+#define NdbScanOperation_H
+
+
+#include <NdbOperation.hpp>
+#include <NdbCursorOperation.hpp>
+
+/**
+ * @class NdbScanOperation
+ * @brief Class of scan operations for use in transactions.
+ */
+class NdbScanOperation : public NdbCursorOperation
+{
+ friend class Ndb;
+ friend class NdbConnection;
+ friend class NdbResultSet;
+ friend class NdbOperation;
+
+public:
+ /**
+ * readTuples returns a NdbResultSet where tuples are stored.
+ * Tuples are not stored in NdbResultSet until execute(NoCommit)
+ * has been executed and nextResult has been called.
+ *
+ * @param parallel Scan parallelism
+ * @param LockMode Scan lock handling
+ * @returns NdbResultSet.
+ */
+ virtual NdbResultSet* readTuples(unsigned parallel = 0,
+ LockMode = LM_Read );
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+
+ int updateTuples();
+ int updateTuples(Uint32 parallelism);
+
+ int deleteTuples();
+ int deleteTuples(Uint32 parallelism);
+
+ // Overload setValue for updateTuples
+ int setValue(const char* anAttrName, const char* aValue, Uint32 len = 0);
+ int setValue(const char* anAttrName, Int32 aValue);
+ int setValue(const char* anAttrName, Uint32 aValue);
+ int setValue(const char* anAttrName, Int64 aValue);
+ int setValue(const char* anAttrName, Uint64 aValue);
+ int setValue(const char* anAttrName, float aValue);
+ int setValue(const char* anAttrName, double aValue);
+
+ int setValue(Uint32 anAttrId, const char* aValue, Uint32 len = 0);
+ int setValue(Uint32 anAttrId, Int32 aValue);
+ int setValue(Uint32 anAttrId, Uint32 aValue);
+ int setValue(Uint32 anAttrId, Int64 aValue);
+ int setValue(Uint32 anAttrId, Uint64 aValue);
+ int setValue(Uint32 anAttrId, float aValue);
+ int setValue(Uint32 anAttrId, double aValue);
+#endif
+private:
+ NdbScanOperation(Ndb* aNdb);
+
+ ~NdbScanOperation();
+
+ NdbCursorOperation::CursorType cursorType();
+
+ virtual int nextResult(bool fetchAllowed = true);
+ virtual void release();
+
+ void closeScan();
+
+ // Overloaded methods from NdbCursorOperation
+ int executeCursor(int ProcessorId);
+
+ // Overloaded private methods from NdbOperation
+ int init(NdbTableImpl* tab, NdbConnection* myConnection);
+ int prepareSend(Uint32 TC_ConnectPtr, Uint64 TransactionId);
+ int doSend(int ProcessorId);
+
+ virtual void setErrorCode(int aErrorCode);
+ virtual void setErrorCodeAbort(int aErrorCode);
+
+ virtual int equal_impl(const NdbColumnImpl* anAttrObject,
+ const char* aValue,
+ Uint32 len);
+private:
+ NdbConnection *m_transConnection;
+ bool m_autoExecute;
+ bool m_updateOp;
+ bool m_writeOp;
+ bool m_deleteOp;
+ class SetValueRecList* m_setValueList;
+};
+
+#ifndef DOXYGEN_SHOULD_SKIP_INTERNAL
+class AttrInfo;
+class SetValueRecList;
+
+class SetValueRec {
+ friend class SetValueRecList;
+public:
+ SetValueRec();
+ ~SetValueRec();
+
+ enum SetValueType {
+ SET_STRING_ATTR1 = 0,
+ SET_INT32_ATTR1 = 1,
+ SET_UINT32_ATTR1 = 2,
+ SET_INT64_ATTR1 = 3,
+ SET_UINT64_ATTR1 = 4,
+ SET_FLOAT_ATTR1 = 5,
+ SET_DOUBLE_ATTR1 = 6,
+ SET_STRING_ATTR2 = 7,
+ SET_INT32_ATTR2 = 8,
+ SET_UINT32_ATTR2 = 9,
+ SET_INT64_ATTR2 = 10,
+ SET_UINT64_ATTR2 = 11,
+ SET_FLOAT_ATTR2 = 12,
+ SET_DOUBLE_ATTR2 = 13
+ };
+
+ SetValueType stype;
+ union {
+ char* anAttrName;
+ Uint32 anAttrId;
+ };
+ typedef struct String {
+ char* aStringValue;
+ Uint32 len;
+ };
+ union {
+ String stringStruct;
+ Int32 anInt32Value;
+ Uint32 anUint32Value;
+ Int64 anInt64Value;
+ Uint64 anUint64Value;
+ float aFloatValue;
+ double aDoubleValue;
+ };
+private:
+ SetValueRec* next;
+};
+
+inline
+SetValueRec::SetValueRec() :
+ next(0)
+{
+}
+
+inline
+SetValueRec::~SetValueRec()
+{
+ if ((stype == SET_STRING_ATTR1) ||
+ (stype == SET_INT32_ATTR1) ||
+ (stype == SET_UINT32_ATTR1) ||
+ (stype == SET_INT64_ATTR1) ||
+ (stype == SET_UINT64_ATTR1) ||
+ (stype == SET_FLOAT_ATTR1) ||
+ (stype == SET_DOUBLE_ATTR1))
+ free(anAttrName);
+
+ if ((stype == SET_STRING_ATTR1) ||
+ (stype == SET_STRING_ATTR2))
+ free(stringStruct.aStringValue);
+ if (next) delete next;
+ next = 0;
+}
+
+class SetValueRecList {
+public:
+ SetValueRecList();
+ ~SetValueRecList();
+
+ void add(const char* anAttrName, const char* aValue, Uint32 len = 0);
+ void add(const char* anAttrName, Int32 aValue);
+ void add(const char* anAttrName, Uint32 aValue);
+ void add(const char* anAttrName, Int64 aValue);
+ void add(const char* anAttrName, Uint64 aValue);
+ void add(const char* anAttrName, float aValue);
+ void add(const char* anAttrName, double aValue);
+ void add(Uint32 anAttrId, const char* aValue, Uint32 len = 0);
+ void add(Uint32 anAttrId, Int32 aValue);
+ void add(Uint32 anAttrId, Uint32 aValue);
+ void add(Uint32 anAttrId, Int64 aValue);
+ void add(Uint32 anAttrId, Uint64 aValue);
+ void add(Uint32 anAttrId, float aValue);
+ void add(Uint32 anAttrId, double aValue);
+
+ typedef void(* IterateFn)(SetValueRec&, NdbOperation&);
+ static void callSetValueFn(SetValueRec&, NdbOperation&);
+ void iterate(IterateFn nextfn, NdbOperation&);
+private:
+ SetValueRec* first;
+ SetValueRec* last;
+};
+
+inline
+SetValueRecList::SetValueRecList() :
+ first(0),
+ last(0)
+{
+}
+
+inline
+SetValueRecList::~SetValueRecList() {
+ if (first) delete first;
+ first = last = 0;
+}
+
+
+inline
+void SetValueRecList::iterate(SetValueRecList::IterateFn nextfn, NdbOperation& oper)
+{
+ SetValueRec* recPtr = first;
+ while(recPtr) {
+ (*nextfn)(*recPtr, oper);
+ recPtr = recPtr->next; // Move to next in list - MASV
+ }
+}
+
+#endif
+
+#endif
diff --git a/ndb/include/ndbapi/NdbSchemaCon.hpp b/ndb/include/ndbapi/NdbSchemaCon.hpp
new file mode 100644
index 00000000000..9d6b49df8f9
--- /dev/null
+++ b/ndb/include/ndbapi/NdbSchemaCon.hpp
@@ -0,0 +1,132 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbSchemaCon_H
+#define NdbSchemaCon_H
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+
+#include <ndb_types.h>
+#include "AttrType.hpp"
+#include "NdbError.hpp"
+
+class NdbSchemaOp;
+class NdbApiSignal;
+class Ndb;
+
+/**
+ * @class NdbSchemaCon
+ * @brief Represents a schema transaction.
+ *
+ * When creating a new table,
+ * the first step is to get a NdbSchemaCon object to represent
+ * the schema transaction.
+ * This is done by calling Ndb::startSchemaTransaction.
+ *
+ * The next step is to get a NdbSchemaOp object by calling
+ * NdbSchemaCon::getNdbSchemaOp.
+ * The NdbSchemaOp object then has methods to define the table and
+ * its attributes.
+ *
+ * Finally, the NdbSchemaCon::execute method inserts the table
+ * into the database.
+ *
+ * @note Currently only one table can be added per transaction.
+ */
+class NdbSchemaCon
+{
+friend class Ndb;
+friend class NdbSchemaOp;
+
+public:
+ /**
+ * Execute a schema transaction.
+ *
+ * @return 0 if successful otherwise -1.
+ */
+ int execute();
+
+ /**
+ * Get a schemaoperation.
+ *
+ * @note Currently, only one operation per transaction is allowed.
+ *
+ * @return Pointer to a NdbSchemaOp or NULL if unsuccessful.
+ */
+ NdbSchemaOp* getNdbSchemaOp();
+
+ /**
+ * Get the latest error
+ *
+ * @return Error object.
+ */
+ const NdbError & getNdbError() const;
+
+private:
+/******************************************************************************
+ * These are the create and delete methods of this class.
+ *****************************************************************************/
+
+ NdbSchemaCon(Ndb* aNdb);
+ ~NdbSchemaCon();
+
+/******************************************************************************
+ * These are the private methods of this class.
+ *****************************************************************************/
+ void init(); // Initialise connection object for new
+ // transaction.
+
+ void release(); // Release all schemaop in schemaCon
+
+ /***************************************************************************
+ * These methods are service methods to other classes in the NDBAPI.
+ ***************************************************************************/
+
+ int checkMagicNumber(); // Verify correct object
+ int receiveDICTTABCONF(NdbApiSignal* aSignal);
+ int receiveDICTTABREF(NdbApiSignal* aSignal);
+
+
+ int receiveCREATE_INDX_CONF(NdbApiSignal*);
+ int receiveCREATE_INDX_REF(NdbApiSignal*);
+ int receiveDROP_INDX_CONF(NdbApiSignal*);
+ int receiveDROP_INDX_REF(NdbApiSignal*);
+
+
+
+/*****************************************************************************
+ * These are the private variables of this class.
+ *****************************************************************************/
+
+
+ NdbError theError; // Errorcode
+ Ndb* theNdb; // Pointer to Ndb object
+
+ NdbSchemaOp* theFirstSchemaOpInList; // First operation in operation list.
+ int theMagicNumber; // Magic number
+};
+
+inline
+int
+NdbSchemaCon::checkMagicNumber()
+{
+ if (theMagicNumber != 0x75318642)
+ return -1;
+ return 0;
+}//NdbSchemaCon::checkMagicNumber()
+#endif
+#endif
+
+
diff --git a/ndb/include/ndbapi/NdbSchemaOp.hpp b/ndb/include/ndbapi/NdbSchemaOp.hpp
new file mode 100644
index 00000000000..90837bbc66b
--- /dev/null
+++ b/ndb/include/ndbapi/NdbSchemaOp.hpp
@@ -0,0 +1,458 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbSchemaOp_H
+#define NdbSchemaOp_H
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+
+#include <ndb_types.h>
+#include "AttrType.hpp"
+#include "NdbSchemaCon.hpp"
+#include <stdlib.h>
+#include "NdbDictionary.hpp"
+
+class NdbApiSignal;
+class Ndb;
+
+/**
+ * @class NdbSchemaOp
+ * @brief Represents various operations for use in schema transactions
+ *
+ * This class is used for schema operations, e.g. creating tables and
+ * attributes.
+ *
+ * The NdbSchemaOp object is created using NdbSchemaCon::getNdbSchemaOp.
+ *
+ * @note This class is depricated and is now replaced with the class
+ * NdbDictionary.
+ */
+class NdbSchemaOp
+{
+ friend class Ndb;
+ friend class NdbSchemaCon;
+
+public:
+ /**
+ * Create a new table in the database.
+ *
+ * @note The NdbSchemaCon should be closed with
+ * Ndb::closeSchemaTransaction, even if this method fails.
+ *
+ * @param aTableName Table name. Should not be NULL.
+ * @param aTableSize (Performance parameter.)
+ * Initial size of the data part of the table
+ * expressed in kByte.
+ * The database handles
+ * bad parameter setting but at a certain
+ * loss in performance.
+ * The size given here is
+ * the initial size allocated for the table
+ * storage (the data part).
+ * When calculating the data storage one should
+ * add the size of all attributes (each attribute
+ * consumes at least 4 bytes) and also an overhead
+ * of 12 byte.
+ * Variable size attributes (not supported yet)
+ * will have a size of 12 bytes plus the actual
+ * data storage parts where there is an
+ * additional overhead based on the size of the
+ * variable part.
+ * <br>
+ * An example table with 5 attributes:
+ * one 64 bit attribute, one 32 bit attribute,
+ * two 16 bit attributes and one array of 64 8 bits.
+ * This table will consume
+ * 12 (overhead) + 8 + 4 + 2*4 (4 is minimum) + 64 =
+ * 96 bytes per record.
+ * Additionally an overhead of about 2 % as page
+ * headers and waste should be allocated.
+ * Thus, 1 million records should consume 96 MBytes
+ * plus the overhead 2 MByte and rounded up to
+ * 100 000 kBytes.
+ * <br><em>
+ * This parameter is currently not used.
+ * </em>
+ * @param aTupleKey Indicates if the table has a primary key or not.
+ * <br>
+ * <b>TupleKey</b> means that a <em>primary key</em>
+ * consisting of one to four attributes
+ * (at most one of variable size)
+ * uniquely identifies each record in the created
+ * table.
+ * <br>
+ * <b>TupleId</b> means that a <em>tuple identity</em>
+ * is used. The tuple identity is
+ * a unique key indentifying each record of the
+ * created table.
+ * The tuple identity is a (non-stored)
+ * 64 bit attribute named <b>NDB$TID</b>.
+ * <br>
+ * When inserting a record (tuple), the method
+ * NdbOperation::setTupleId
+ * will generate a unique tuple identity
+ * and return it to the user.
+ * <br>
+ * When reading, updating or deleting a record
+ * in a table with <b>TupleId</b>,
+ * NdbOperation::equal("NDB$TID", value_Uint64)
+ * can be used to identify the record.
+ * <br>
+ * Legal values: TupleKey or TupleId.
+ * @param aNrOfPages (Performance parameter.)
+ * Specifies the initial size of the index storage.
+ * When calculating the index storage,
+ * each key has approximately 14 byte of
+ * overhead plus the size of the key.
+ * Each key attribute takes up at least 4 bytes
+ * of storage.
+ * Thus a mixed key consisting of a
+ * 64 bit attribute, a 32 bit attribute
+ * and a 16 bit attribute will
+ * consume approx. 30 bytes per key.
+ * Thus, the if initial size is to be 1 million rows,
+ * then aNrOfPages should be set to
+ * 30 M / 8k = 2670 pages.
+ * <br><em>
+ * This parameter is currently not used.
+ * </em>
+ * @param aFragmentType Type of fragmentation.<br>
+ * <b>All</b> (default) means that the
+ * table fragments are automatically
+ * distributed on all nodes in the system.<br>
+ * <b>DistributionGroup</b> and
+ * <b>DistributionKey</b> are
+ * also supported. For further details about
+ * these types see the documentation of
+ * Ndb::startTransaction.
+ * @param aKValue (Hash parameter.)
+ * Only allowed value is 6.
+ * Later implementations might add flexibility
+ * in this parameter.
+ * @param aMinLoadFactor (Hash parameter.)
+ * This value specifies the load factor when
+ * starting to shrink the hash table.
+ * It must be smaller than aMaxLoadFactor.
+ * Both these factors are given in percentage.
+ * @param aMaxLoadFactor (Hash parameter.)
+ * This value specifies the load factor when
+ * starting to split the containers in the local
+ * hash tables. 100 is the maximum which will
+ * optimize memory usage (this is the figure
+ * used for the above calculations).
+ * A lower figure will store less information in
+ * each container and thus
+ * find the key faster but consume more memory.
+ * @param aMemoryType Currently only 1 is allowed which specifies
+ * storage of table in main memory.
+ * Later 2 will be added where the table is stored
+ * completely on disk
+ * and 3 where the index is in main memory but
+ * data is on disk.
+ * If 1 is chosen an individual attribute can
+ * still be specified as a disk attribute.
+ * @param aStoredTable If set to false it indicates that the table is
+ * a temporary table and should not be logged
+ * to disk.
+ * In case of a system restart the table will still
+ * be defined and exist but will be empty.
+ * Thus no checkpointing and
+ * no logging is performed on the table.
+ * The default value is true and indicates a
+ * normal table with full checkpointing and
+ * logging activated.
+ * @return Returns 0 when successful and returns -1 otherwise.
+ */
+ int createTable( const char* aTableName,
+ Uint32 aTableSize = 8,
+ KeyType aTupleKey = TupleKey,
+ int aNrOfPages = 2,
+ FragmentType aFragmentType = All,
+ int aKValue = 6,
+ int aMinLoadFactor = 78,
+ int aMaxLoadFactor = 80,
+ int aMemoryType = 1,
+ bool aStoredTable = true);
+
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+ /**
+ * This is the old function declaration, don't use.
+ *
+ * @deprecated do not use!
+ */
+ inline int createTable( const char* aTableName,
+ Uint32 aTableSize,
+ KeyType aTupleKey,
+ int aNrOfPages,
+ FragmentType aFragmentType,
+ int aKValue,
+ int aMinLoadFactor,
+ int aMaxLoadFactor,
+ int aMemoryType,
+ int aStoredTable){
+ return createTable(aTableName,
+ aTableSize,
+ aTupleKey,
+ aNrOfPages,
+ aFragmentType,
+ aKValue,
+ aMinLoadFactor,
+ aMaxLoadFactor,
+ aMemoryType,
+ (aStoredTable == 1 ? true : false));
+ }
+#endif
+
+ /**
+ * Add a new attribute to a database table.
+ *
+ * Attributes can only be added to a table in the same transaction
+ * as the transaction creating the table.
+ *
+ * @note The NdbSchemaCon transaction should be closed with
+ * Ndb::closeSchemaTransaction, even if this method fails.
+ *
+ * Example creating an unsigned int attribute belonging to the primary key
+ * of the table it is created in:
+ * @code
+ * MySchemaOp->createAttribute("Attr1", // Attribute name
+ * TupleKey, // Belongs to primary key
+ * 32, // 32 bits
+ * 1, // Not an array attribute
+ * UnSigned, // Unsigned type
+ * );
+ * @endcode
+ *
+ * Example creating a string attribute belonging to the primary key
+ * of the table it is created in:
+ * @code
+ * MySchemaOp->createAttribute("Attr1", // Attribute name
+ * TupleKey, // Belongs to primary key
+ * 8, // Each character is 8 bits
+ * 12, // Max 12 chars in string
+ * String, // Attribute if of type string
+ * );
+ * @endcode
+ *
+ * A <em>distribution key</em> is a set of attributes which are used
+ * to distribute the tuples onto the NDB nodes.
+ * A <em>distribution group</em> is a part (currently 16 bits)
+ * of an attribute used to distribute the tuples onto the NDB nodes.
+ * The distribution key uses the NDB Cluster hashing function,
+ * while the distribution group uses a simpler function.
+ *
+ * @param aAttrName Attribute name. Should not be NULL.
+ * @param aTupleKey This parameter specifies whether the
+ * attribute is part of the primary key or not.
+ * Floats are not allowed in the primary key.
+ * <br>
+ * Legal values: NoKey, TupleKey
+ * @param aAttrSize Specifies the size of the elements of the
+ * attribute. (An attribute can consist
+ * of an array of elements.)
+ * <br>
+ * Legal values: 8, 16, 32, 64 and 128 bits.
+ * @param aArraySize Size of array.
+ * <br>
+ * Legal values:
+ * 0 = variable-sized array,
+ * 1 = no array, and
+ * 2- = fixed size array.
+ * <br>
+ * <em>
+ * Variable-sized array attributes are
+ * not yet supported.
+ * </em>
+ * <br>
+ * There is no upper limit of the array size
+ * for a single attribute.
+ * @param aAttrType The attribute type.
+ * This is only of interest if calculations are
+ * made within NDB.
+ * <br>
+ * Legal values: UnSigned, Signed, Float, String
+ * @param aStorageMode Main memory based or disk based attribute.<br>
+ * Legal values: MMBased, DiskBased
+ * <br>
+ * <em>
+ * Disk-based attributes are not yet supported.
+ * </em>
+ * @param nullable Set to true if NULL is a correct value for
+ * the attribute.
+ * <br>
+ * Legal values: true, false
+ * @param aStType Stored in both index and data storage or
+ * only store in index data storage.
+ * <br>
+ * This parameter is only of interest for tuple
+ * key attributes.
+ * All tuple key attributes values are always stored
+ * in the index storage part.
+ * If this parameter is set to
+ * IndexStorageAttribute, then the attribute values
+ * will <em>only</em> be stored in the index
+ * storage part and <em>not</em> in the data
+ * storage part.
+ * <br>
+ * If there will be no scans using the primary
+ * key attribute and if the size of the attribute
+ * is large, then this might be of interest.
+ * A typical example is a table where
+ * http-addresses are used as primary key.
+ * <br>
+ * Legal values: NormalStorageAttribute,
+ * IndexStorageAttribute
+ * @param aDistributionKey Sometimes it is preferable to use a subset
+ * of the primary key as the distribution key.
+ * An example is TPC-C where it might be
+ * good to use the warehouse id and district id
+ * as the distribution key.
+ * <br>
+ * Locally in the fragments the full primary key
+ * will still be used with the hashing algorithm.
+ * Set to 1 if this attribute is part of the
+ * distribution key.
+ * All distribution key attributes must be
+ * defined before
+ * any other attributes are defined.
+ * @param aDistributionGroup In other applications it is desirable to use
+ * only a part of an attribute to create the
+ * distribution key.
+ * This is applicable for some telecom
+ * applications.
+ * <br>
+ * In these situations one must provide how many
+ * bits of the attribute that is to
+ * be used as the distribution hash value.
+ * <br>
+ * This provides some control to the
+ * application of the distribution.
+ * It still needs to be part of a primary key
+ * the attribute and must be defined as the
+ * first attribute.
+ * @param aDistributionGroupNoOfBits
+ * Number of bits to use of the
+ * distribution group attribute in the
+ * distribution hash value.
+ * <br>
+ * Currently, only 16 bits is supported. It will
+ * always be the last 16 bits in the attribute
+ * which is used for the distribution group.
+ * @param aAutoIncrement Set to autoincrement attribute.
+ * @param aDefaultValue Set a default value of attribute.
+ *
+ * @return Returns 0 when successful and returns -1 otherwise.
+ ****************************************************************************/
+ int createAttribute(const char* aAttrName,
+ KeyType aTupleKey = NoKey,
+ int aAttrSize = 32,
+ int aArraySize = 1,
+ AttrType aAttrType = UnSigned,
+ StorageMode aStorageMode = MMBased,
+ bool nullable = false,
+ StorageAttributeType aStType= NormalStorageAttribute,
+ int aDistributionKey = 0,
+ int aDistributionGroup = 0,
+ int aDistributionGroupNoOfBits = 16,
+ bool aAutoIncrement = false,
+ const char* aDefaultValue = 0);
+
+#ifndef DOXYGEN_SHOULD_SKIP_DEPRECATED
+ /**
+ * @deprecated do not use!
+ */
+ int createAttribute(const char* aAttrName,
+ KeyType aTupleKey,
+ int aAttrSize,
+ int aArraySize,
+ AttrType aAttrType,
+ StorageMode aStorageMode,
+ NullAttributeType aNullAttr,
+ StorageAttributeType aStType = NormalStorageAttribute,
+ int aDistributionKey = 0,
+ int aDistributionGroup = 0,
+ int aDistributionGroupNoOfBits = 16){
+ return createAttribute(aAttrName,
+ aTupleKey,
+ aAttrSize,
+ aArraySize,
+ aAttrType,
+ aStorageMode,
+ aNullAttr == NullAttribute,
+ aStType,
+ aDistributionKey,
+ aDistributionGroup,
+ aDistributionGroupNoOfBits);
+ }
+#endif
+
+ /**
+ * Get the last error which occurred during the transaction.
+ *
+ * If an error occured (NdbSchemaCon::execute returned -1 or
+ * NdbSchemaCon::getNdbSchemaOp returned NULL), then this method
+ * retrieves the error object containing information about
+ * the error.
+ *
+ * @return Error object containing information about last error.
+ */
+ const NdbError & getNdbError() const;
+
+protected:
+
+/*****************************************************************************
+ * These are the methods used to create and delete the NdbOperation objects.
+ ****************************************************************************/
+ NdbSchemaOp(Ndb* aNdb);
+
+ ~NdbSchemaOp();
+
+/******************************************************************************
+ * These methods are service routines used by the other NDBAPI classes.
+ *****************************************************************************/
+
+ void release(); // Release all memory connected
+ // to the operations object.
+
+/****************************************************************************
+ * The methods below is the execution part of the NdbSchemaOp class.
+ *****************************************************************************/
+
+ int sendRec();
+ int sendSignals(Uint32 aNodeId, bool HaveMutex);
+
+ int init(NdbSchemaCon* aSchemaCon);
+
+ /**************************************************************************
+ * These are the private variables that are defined in the operation
+ * objects.
+ **************************************************************************/
+ Ndb* theNdb; // Point back to the Ndb object.
+ NdbSchemaCon* theSchemaCon; // Point back to the connection object.
+
+ class NdbDictionary::Table * m_currentTable;
+};
+
+inline
+const NdbError &
+NdbSchemaOp::getNdbError() const
+{
+ return theSchemaCon->getNdbError();
+}
+
+#endif
+#endif
+
+
diff --git a/ndb/include/ndbapi/ndbapi_limits.h b/ndb/include/ndbapi/ndbapi_limits.h
new file mode 100644
index 00000000000..bcfba7d3f9d
--- /dev/null
+++ b/ndb/include/ndbapi/ndbapi_limits.h
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBAPI_LIMITS_H
+#define NDBAPI_LIMITS_H
+
+#define NDB_MAX_NO_OF_ATTRIBUTES_IN_KEY 32
+#define NDB_MAX_TABLES 1600
+#define NDB_MAX_DATABASE_NAME_SIZE 128
+#define NDB_MAX_SCHEMA_NAME_SIZE 128
+#define NDB_MAX_TAB_NAME_SIZE 128
+#define NDB_MAX_ATTR_NAME_SIZE 32
+#define NDB_MAX_ATTR_DEFAULT_VALUE_SIZE 128
+#define NDB_MAX_ATTRIBUTES_IN_TABLE 91
+#define NDB_MAX_ATTRIBUTES_IN_INDEX NDB_MAX_NO_OF_ATTRIBUTES_IN_KEY
+#define NDB_MAX_TUPLE_SIZE_IN_WORDS 1023
+#define NDB_MAX_FIXED_KEY_LENGTH_IN_WORDS 8
+#define NDB_MAX_KEYSIZE_IN_WORDS 1023
+#define NDB_MAX_KEY_SIZE NDB_MAX_KEYSIZE_IN_WORDS*sizeof(Uint32)
+#define NDB_MAX_TUPLE_SIZE 8191
+#define NDB_MAX_CONNECTIONS 127
+#define NDB_MAX_TRANSACTIONS 1024
+#define NDB_MAX_PARALLEL_SCANS 12
+#define NDB_MAX_ACTIVE_EVENTS 100
+
+#ifndef MIN
+#define MIN(x,y) (((x)<(y))?(x):(y))
+#endif
+
+#ifndef MAX
+#define MAX(x,y) (((x)>(y))?(x):(y))
+#endif
+
+#endif
diff --git a/ndb/include/newtonapi/dba.h b/ndb/include/newtonapi/dba.h
new file mode 100644
index 00000000000..eb5b3f9b5c2
--- /dev/null
+++ b/ndb/include/newtonapi/dba.h
@@ -0,0 +1,732 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * @mainpage DBA User Guide
+ *
+ * @section secIntro Introduction
+ * DBA is an API to access the NDB Cluster.
+ *
+ * DBA supports transactions using an asynchronous execution model.
+ * Everything but transactions is synchronous.
+ *
+ * DBA uses the concept of bindings to simplify database access.
+ * A <em>binding</em> is a relation between a database table and
+ * one or several C structs.
+ * A binding is created initially and then used multiple time during
+ * application execution.
+ *
+ * Each of the data accessing functions in DBA is implemented as a
+ * transaction, i.e. the call will either fully complete or
+ * nothing happens (the transaction fails).
+ *
+ * DBA also supports "read as much as possible" with bulk read.
+ * With bulk read the application can specify a set of primary keys and
+ * try to read all of the corresponding rows. The bulk read will not fail
+ * if a row does not exist but will instead inform the application using a
+ * RowFoundIndicator variable.
+ *
+ * A <em>request</em> is a transaction or a bulk read.
+ *
+ * @section secError Error Handling
+ * When a synchronous method in DBA fails these methods are applicable:
+ * -# DBA_GetLatestError()
+ * -# DBA_GetLatestNdbError()
+ * -# DBA_GetLatestErrorMsg()
+ *
+ * The DBA_GetLatestErrorMsg() will then return a description of
+ * what has failed.
+ *
+ * For asynchronous methods the application should:
+ * -# check that the RequestId returned by function is not
+ * @ref DBA_INVALID_REQID
+ * -# check Status supplied in callback (see @ref DBA_AsyncCallbackFn_t)
+ *
+ * If @ref DBA_INVALID_REQID is returned,
+ * the details of error can be found using
+ * "latest"-functions.
+ *
+ * If error is indicated in callback (using Status), when the
+ * "latest"-functions are <b>NOT</b> applicable.
+ *
+ * @section secExamples Example Programs
+ *
+ * - @ref common.hpp
+ * - @ref basic.cpp
+ * - @ref br_test.cpp
+ * - @ref ptr_binding_test.cpp
+ *
+ */
+
+/**
+ * @page basic.cpp basic.cpp
+ * @include basic.cpp
+ */
+
+/**
+ * @page common.hpp common.hpp
+ * @include common.hpp
+ */
+
+/**
+ * @page br_test.cpp br_test.cpp
+ * @include br_test.cpp
+ */
+
+/**
+ * @page ptr_binding_test.cpp ptr_binding_test.cpp
+ * @include ptr_binding_test.cpp
+ */
+
+/** @addtogroup DBA
+ * @{
+ */
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+#ifndef DBA_H
+#define DBA_H
+
+/* --- Include files ---- */
+
+#include <defs/pcn_types.h>
+
+#include <stdio.h>
+
+
+/* --- Types and definitions --- */
+
+/**
+ * Possible error status for DBA functions.
+ */
+typedef enum {
+ DBA_NO_ERROR = 0, /**< Success */
+
+ DBA_NOT_IMPLEMENTED = -1, /**< Function not implemented */
+ DBA_NDB_ERROR = -2, /**< Uncategorised error from NDB */
+ DBA_ERROR = -3, /**< Uncategorised error from DBA implementation */
+
+ DBA_APPLICATION_ERROR = 1, /**< Function called with invalid argument(s)
+ or other application errors */
+ DBA_NO_DATA = 2, /**< No row with specified PK existed */
+ DBA_CONSTRAINT_VIOLATION = 3, /**< There already exists a row with that PK*/
+
+ DBA_SCHEMA_ERROR = 4, /**< Table already exists */
+ DBA_INSUFFICIENT_SPACE = 5, /**< The DB is full */
+ DBA_TEMPORARY_ERROR = 6, /**< Some temporary problem occured */
+ DBA_TIMEOUT = 7, /**< The request timed out, probably due to
+ dead-lock */
+ DBA_OVERLOAD = 8, /**< The DB is overloaded */
+ DBA_UNKNOWN_RESULT = 9 /**< It is unknown wheater transaction was
+ commited or aborted */
+} DBA_Error_t;
+
+/**
+ * Error code. This is the error code that is returned by NDB.
+ * Not to be confused by the status returned by the DBA implementation.
+ */
+typedef int DBA_ErrorCode_t;
+
+/**
+ * DBA column types
+ */
+typedef enum {
+ DBA_CHAR, /**< String */
+ DBA_INT /**< Integer */
+} DBA_DataTypes_t;
+
+
+/**
+ * Column description.
+ * Used for creating tables.
+ */
+typedef struct DBA_ColumnDesc {
+
+ const char* Name; /**< Name of table column */
+ DBA_DataTypes_t DataType; /**< Datatype of table column*/
+ Size_t Size; /**< Column size in bytes */
+ Boolean_t IsKey; /**< True if column is part of primary key */
+
+} DBA_ColumnDesc_t;
+
+/**
+ * Used to simplify binding definitions. See @ref DBA_ColumnBinding
+ * for example.
+ *
+ * @param ColName Name of column in db table
+ * @param Type Column/field type.
+ * @param Struct Structure
+ * @param Field Field in structure
+ * @return Arg list for defining binding of type @ref DBA_Binding_t
+ */
+#define DBA_BINDING( ColName, Type, Struct, Field ) \
+ { ColName, Type, PCN_SIZE_OF( Struct, Field ), \
+ PCN_OFFSET_OF( Struct, Field ), 0, 0 }
+
+/**
+ * Used to simplify ptr binding definitions. See @ref DBA_ColumnBinding
+ * for example.
+ *
+ * @param Struct Structure
+ * @param Field Field in structure
+ * @return Arg list for defining binding of type @ref DBA_Binding_t
+ */
+#define DBA_BINDING_PTR(Struct, Field, ColBindings, NbCBindings) \
+ { 0, DBA_CHAR, NbCBindings, PCN_OFFSET_OF( Struct, Field ), \
+ 1, ColBindings }
+
+/**
+ * The @ref DBA_ColumnBinding_t is used to describe a binding between one
+ * column and one field of a C struct.
+ *
+ *<pre>
+ * typedef struct Address {
+ * char StreetName[30];
+ * int StreetNumber;
+ * } Address_t;
+ *
+ * typdef struct Person {
+ * char Name[30];
+ * Address_t * AddressPtr;
+ * } Person_t; </pre>
+ *
+ *
+ * For example, if the field Name of a Person_t data structure is
+ * bound to the column "NAME", the corresponding binding would be
+ * defined as:
+ *
+ *<pre>
+ * DBA_ColumnBinding_t NameBinding =
+ * DBA_BINDING( "name", DBA_CHAR, Person_t, Name ); </pre>
+ *
+ *
+ * There is also the @ref DBA_BINDING_PTR which is used when
+ * several linked structures should be put into one table.
+ *
+ * For example, if data in a Person_t data structure should be saved
+ * in the same table as the Address_t data structure
+ * (as the address belongs to the person), the corresponding binding would be
+ * defined as:
+ *
+ *<pre>
+ * DBA_ColumnBinding_t AddrBinding[AddrLen]; This binding describes how the
+ * fields in the Address_t
+ * structure is linked to the
+ * table PERSON_ADDRESS
+ *
+ * DBA_ColumnBinding_t AddressBinding =
+ * DBA_BINDING_PTR(Person_t, AddressPtr, AddrBinding, AddrLen); </pre>
+ *
+ *
+ */
+struct DBA_ColumnBinding {
+ const char* Name; /**< Name of table column */
+ DBA_DataTypes_t DataType; /**< Type of member in structure */
+ Size_t Size; /**< Size in bytes of member
+ or no of @ref DBA_ColumnBinding's
+ when doing ptr binding */
+ Size_t Offset; /**< Offset of the member */
+
+ Boolean_t Ptr; /**< True if binding is of ptr type */
+ const struct DBA_ColumnBinding * SubBinding; /**< Address of Binding Ptr
+ valid if Ptr is true */
+};
+
+/**
+ * Typedef: @ref DBA_ColumnBinding
+ */
+typedef struct DBA_ColumnBinding DBA_ColumnBinding_t;
+
+/**
+ * A @ref DBA_Binding_t object is used to establish a binding between
+ * one or more columns of a table to the fields of C structs.
+ *
+ * It is used with insert, and update and read transactions to define
+ * on which columns of the table the operations is performed, and to
+ * which members of a C data structure they map.
+ *
+ * All key columns must be bound to a field of the struct.
+ *
+ * The function @ref DBA_CreateBinding is used to create this binding.
+ */
+typedef struct DBA_Binding DBA_Binding_t;
+
+/* --- Exported functions --- */
+
+/**
+ * Set DBA configuration parameter
+ *<pre>
+ * Id Description Default Min Max
+ * == =========================== ======= ==== ====
+ * 0 NBP Interval 10 4 -
+ * 1 Operations/Bulkread 1000 1 5000
+ * 2 Start transaction timeout 0 0 -
+ * 3 Force send algorithm 1 0 2
+ *</pre>
+ * @return Status
+ */
+DBA_Error_t DBA_SetParameter(int ParameterId, int Value);
+
+/**
+ * Set DBA configuration parameter.
+ * See @ref DBA_SetParameter for description of parameters.
+ *
+ * @return Status
+ */
+DBA_Error_t DBA_GetParameter(int ParameterId, int * Value);
+
+/**
+ * Initialize DBA library and connect to NDB Cluster.
+ *
+ * @return Status
+ */
+DBA_Error_t DBA_Open( );
+
+/**
+ * Close connection to NDB cluster and free allocated memory.
+ *
+ * @return Error status
+ */
+DBA_Error_t DBA_Close(void);
+
+/**
+ * Get latest DBA error.
+ *
+ * @note Only applicable to synchronous methods
+ */
+DBA_Error_t DBA_GetLatestError();
+
+/**
+ * Get latest NDB error.
+ *
+ * @note Only applicable to synchronous methods
+ */
+DBA_ErrorCode_t DBA_GetLatestNdbError();
+
+/**
+ * Get latest error string associated with DBA_GetLatestError().
+ *
+ * @note String must not be free by caller of this method.
+ * @note Only applicable to synchronous methods.
+ */
+const char * DBA_GetLatestErrorMsg();
+
+/**
+ * Get error msg associated with code
+ *
+ * @note String must not be free by caller of this method
+ */
+const char * DBA_GetErrorMsg(DBA_Error_t);
+
+/**
+ * Get error msg associated with code
+ *
+ * @note String must not be free by caller of this method
+ */
+const char * DBA_GetNdbErrorMsg(DBA_ErrorCode_t);
+
+/**
+ * Create a table.
+ *
+ * @param TableName Name of table to create.
+ * @param NbColumns numbers of columns.
+ * @param Columns Column descriptions.
+ * @return Status.
+ */
+DBA_Error_t
+DBA_CreateTable( const char* TableName, int NbColumns,
+ const DBA_ColumnDesc_t Columns[] );
+
+/**
+ * Destroy a table.
+ *
+ * @param TableName Table name.
+ * @return Status.
+ * @note Not implemented
+ */
+DBA_Error_t
+DBA_DropTable( const char* TableName );
+
+
+/**
+ * Test for existence of a table.
+ *
+ * @param TableName Table name.
+ * @return Boolean value indicating if table exists or not.
+ */
+Boolean_t
+DBA_TableExists( const char* TableName );
+
+/**
+ * Define a binding between the columns of a table and a C structure.
+ *
+ * @param TableName table
+ * @param NbCol number of columns bindings
+ * @param ColBinding bindings
+ * @param StructSz Sizeof structure.
+ * @return Created binding, or NULL if binding could not be created.
+ */
+DBA_Binding_t*
+DBA_CreateBinding( const char* TableName,
+ int NbCol, const DBA_ColumnBinding_t ColsBinding[],
+ Size_t StructSz );
+
+/**
+ * Destroys a @ref DBA_Binding_t allocated with @ref
+ * DBA_CreateBinding.
+ *
+ * @param pBinding Pointer to binding.
+ * @return Status.
+ */
+DBA_Error_t
+DBA_DestroyBinding( DBA_Binding_t* Binding );
+
+/**
+ * Used to identify a pending db request
+ */
+typedef long DBA_ReqId_t;
+
+/**
+ * An asynchronous call returning this means that the function was called
+ * with invalid arguments. The application should check error status
+ * with DBA_GetLatestError() etc.
+ */
+#define DBA_INVALID_REQID 0
+
+/**
+ * Callback function for transactions.
+ * Will be called in NBP process (Newton Batch Process).
+ *
+ * @note The implementation of the callback function is not allowed to
+ * make an asynchronous database call.
+ *
+ * @param ReqId Request identifier
+ * @param Status Status of the request
+ * @param ErrorCode Error code given by NDB
+ * @see DBA_Error_t
+ */
+typedef void (*DBA_AsyncCallbackFn_t)( DBA_ReqId_t ReqId,
+ DBA_Error_t Status,
+ DBA_ErrorCode_t ErrorCode );
+/**
+ * Insert row(s) in the table (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of pointers to structures.
+ * @param NbRows No of rows to insert (i.e. length of pData array)
+ * @return Request identifier
+ *
+ * @note All the table columns must be part of the binding.
+ */
+DBA_ReqId_t
+DBA_InsertRows( const DBA_Binding_t* pBinding, const void * const pData[],
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Insert row(s) in the table (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of structures.
+ * @param NbRows No of rows to insert (i.e. length of pData array)
+ * @return Request identifier
+ *
+ * @note All the table columns must be part of the binding.
+ */
+DBA_ReqId_t
+DBA_ArrayInsertRows( const DBA_Binding_t* pBinding, const void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Update row(s) in the table (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of pointers to structures. Fields that are part of the
+ * key are used to generate the where clause, the
+ * other fields are used to update the row.
+ * @param NbRows No of rows to update (i.e. length of pData array).
+ * @return Request identifier
+ */
+DBA_ReqId_t
+DBA_UpdateRows( const DBA_Binding_t* pBinding, const void * const pData[],
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Update row(s) in the table (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of structures. Fields that are part of the
+ * key are used to generate the where clause, the
+ * other fields are used to update the row.
+ * @param NbRows No of rows to update (i.e. length of pData array).
+ * @return Request identifier
+ */
+DBA_ReqId_t
+DBA_ArrayUpdateRows( const DBA_Binding_t* pBinding, const void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Delete row(s) from the table (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of pointers to structures.
+ * Only fields part of the primary key needs to be set.
+ * @param NbRows No of rows to delete (i.e. length of pData array)
+ * @return Request identifier
+ */
+DBA_ReqId_t
+DBA_DeleteRows( const DBA_Binding_t* pBinding, const void * const pData[],
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+
+/**
+ * Delete row(s) from the table (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of structures. Only fields part of the primary
+ * key needs to be set.
+ * @param NbRows No of rows to delete (i.e. length of pData array)
+ * @return Request identifier
+ */
+DBA_ReqId_t
+DBA_ArrayDeleteRows( const DBA_Binding_t* pBinding, const void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Updates/Inserts row(s) in the table (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of pointers to structures.
+ * @param NbRows No of rows to update/insert (i.e. length of pData array)
+ * @return Request identifier
+ * @note All the table columns must be part of the binding.
+ */
+DBA_ReqId_t
+DBA_WriteRows( const DBA_Binding_t* pBinding, const void * const pData[],
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Update/Insert row(s) in the table (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of structures.
+ * @param NbRows No of rows to update/insert (i.e. length of pData array)
+ * @return Request identifier
+ * @note All the table columns must be part of the binding.
+ */
+DBA_ReqId_t
+DBA_ArrayWriteRows( const DBA_Binding_t* pBinding, const void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Read row(s) from a table of the database (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of pointers to structures.
+ * Only fields part of the primary key needs to be set.
+ * The other fields in the binding will be populated.
+ * @param NbRows No of rows to read (i.e. length of pData array)
+ * @return Request identifier
+ */
+DBA_ReqId_t
+DBA_ReadRows( const DBA_Binding_t* pBinding, void * const pData[],
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Read row(s) from a table of the database (one transaction)
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of structures.
+ * Only fields part of the primary key needs to be set.
+ * The other fields in the binding will be populated.
+ * @param NbRows No of rows to read (i.e. length of pData array)
+ * @return Request identifier
+ */
+DBA_ReqId_t
+DBA_ArrayReadRows( const DBA_Binding_t* pBinding, void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+/**
+ * Insert <b>one</b> row for each specified binding (as one transaction).
+ *
+ * @param pBindings Array of pointers to bindings.
+ * @param pData Array of pointers to structures.
+ * @param NbBindings No of bindings (tables) to insert into,
+ * i.e. length of arrays pBindings and pData
+ * @return Request identifier
+ * @note It is valid to specify the same binding twice
+ * (with corresponding data pointer) if you want to insert two
+ * rows in one table
+ */
+DBA_ReqId_t
+DBA_MultiInsertRow(const DBA_Binding_t * const pBindings[],
+ const void * const pData[],
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Update <b>one</b> row for each specified binding (as one transaction).
+ *
+ * @param pBindings Array of pointers to bindings.
+ * @param pData Array of pointers to structures.
+ * @param NbBindings No of bindings (tables) to insert into
+ * i.e. length of arrays pBindings and pData
+ * @return Request identifier
+ * @note It is valid to specify the same binding twice
+ * (with corresponding data pointer) if you want to update two
+ * rows in one table
+ */
+DBA_ReqId_t
+DBA_MultiUpdateRow(const DBA_Binding_t * const pBindings[],
+ const void * const pData[],
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Update/insert <b>one</b> row for each specified binding (as one transaction).
+ *
+ * @param pBindings Array of pointers to bindings.
+ * @param pData Array of pointers to structures.
+ * @param NbBindings No of bindings (tables) to insert into
+ * i.e. length of arrays pBindings and pData
+ * @return Request identifier
+ * @note It is valid to specify the same binding twice
+ * (with corresponding data pointer) if you want to update/insert two
+ * rows in one table
+ */
+DBA_ReqId_t
+DBA_MultiWriteRow(const DBA_Binding_t * const pBindings[],
+ const void * const pData[],
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Delete <b>one</b> row for each specified binding (as one transaction).
+ *
+ * @param pBindings Array of pointers to bindings.
+ * @param pData Array of pointers to structures.
+ * @param NbBindings No of bindings (tables) to insert into
+ * i.e. length of arrays pBindings and pData
+ * @return Request identifier
+ * @note It is valid to specify the same binding twice
+ * (with corresponding data pointer) if you want to delete two
+ * rows in one table
+ */
+DBA_ReqId_t
+DBA_MultiDeleteRow(const DBA_Binding_t * const pBindings[],
+ const void * const pData[],
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Read <b>one</b> row for each specified binding (as one transaction).
+ *
+ * @param pBindings Array of pointers to bindings.
+ * @param pData Array of pointers to structures.
+ * @param NbBindings No of bindings (tables) to insert into
+ * i.e. length of arrays pBindings and pData
+ * @return Request identifier
+ * @note It is valid to specify the same binding twice
+ * (with corresponding data pointer) if you want to read two
+ * rows in one table
+ */
+DBA_ReqId_t
+DBA_MultiReadRow(const DBA_Binding_t * const pBindings[],
+ void * const pData[],
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+/**
+ * A structure used for bulk reads.
+ * The structure contains a pointer to the data and an indicator.
+ * After the bulk read has completed, the indicator is set to 1 if the row
+ * was found and to 0 if the row was not found.
+ *
+ */
+typedef struct DBA_BulkReadResultSet {
+ void * DataPtr; /**< Pointer to data. Only fields part of
+ primary key members needs
+ to be set before bulk read. */
+ Boolean_t RowFoundIndicator; /**< This indicator has a valid value
+ only after bulk read has completed.
+ If the value is 1 then the row was found */
+} DBA_BulkReadResultSet_t;
+
+/**
+ * Read rows from a table of the database (potentially multiple transactions)
+ * The users should for each NbRows specify the fields part of the primary key
+ *
+ * @param pBinding Binding between table columns and struct fields.
+ * @param pData Array of DBA_BulkReadResultSet_t, with DataPtr pointing to
+ * structure. Only the fields which are part of the
+ * primary key need be set.
+ * The RowFoundIndicator will be set when the request returns.
+ * @param NbRows No of rows to read (i.e. length of pData array)
+ * @return Request identifier
+ *
+ */
+DBA_ReqId_t
+DBA_BulkReadRows(const DBA_Binding_t * pBinding,
+ DBA_BulkReadResultSet_t pData[],
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/**
+ * Read rows from several tables of the database in potentially multiple
+ * transactions.
+ *
+ *<pre>
+ * The pData array <b>must</b> be organized as follows:
+ * NbRows with DataPtr pointing to structure of type pBindings[0]
+ * NbRows with DataPtr pointing to structure of type pBindings[1]
+ * ... </pre>
+ * Meaning that the pData array must be (NbBindings * NbRows) in length.
+ *
+ * The user should for each (NbRows * NbBindings) specify the primary key
+ * fields.
+ *
+ * @param pBindings Array of pointers to bindings
+ * @param pData Array of DBA_BulkReadResultSet_t.
+ * With DataPtr pointing to structure. Only the fields which
+ * are part of the key need be set.
+ * The RowFoundIndicator will be set when the operations returns.
+ * @param NbBindings No of bindings (i.e. length of pBindings array)
+ * @param NbRows No of rows per binding to read
+ * @return Request identifier
+ */
+DBA_ReqId_t
+DBA_BulkMultiReadRows(const DBA_Binding_t * const pBindings[],
+ DBA_BulkReadResultSet_t pData[],
+ int NbBindings,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+/** @} */
+
+#endif
diff --git a/ndb/include/newtonapi/defs/pcn_types.h b/ndb/include/newtonapi/defs/pcn_types.h
new file mode 100644
index 00000000000..a823846d7be
--- /dev/null
+++ b/ndb/include/newtonapi/defs/pcn_types.h
@@ -0,0 +1,41 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PCN_TYPES_H
+#define PCN_TYPES_H
+
+#include <stddef.h>
+#include <NdbUnistd.h>
+
+#ifdef NDB_MACOSX
+typedef unsigned int Size_t;
+#elif defined(NDB_SPARC_64)
+typedef unsigned int Size_t;
+#else
+typedef size_t Size_t;
+#endif
+
+typedef int Boolean_t;
+
+typedef unsigned UInt32_t;
+
+#define PCN_TRUE true
+#define PCN_FALSE false
+
+#define PCN_SIZE_OF(s, m ) sizeof(((s *)0)->m)
+#define PCN_OFFSET_OF(s, m) offsetof(s, m)
+
+#endif
diff --git a/ndb/include/portlib/NdbCondition.h b/ndb/include/portlib/NdbCondition.h
new file mode 100644
index 00000000000..fb1f2fcd69e
--- /dev/null
+++ b/ndb/include/portlib/NdbCondition.h
@@ -0,0 +1,94 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_CONDITION_H
+#define NDB_CONDITION_H
+
+#include "NdbMutex.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+struct NdbCondition;
+
+
+/*
+// Create a condition
+//
+// * returnvalue: pointer to the condition structure
+*/
+struct NdbCondition* NdbCondition_Create(void);
+
+/*
+// Wait for a condition, allows a thread to wait for
+// a condition and atomically releases the associated mutex.
+//
+// * p_cond: pointer to the condition structure
+// * p_mutex: pointer to the mutex structure
+// * returnvalue: 0 = succeeded, 1 = failed
+*/
+int NdbCondition_Wait(struct NdbCondition* p_cond,
+ NdbMutex* p_mutex);
+
+/*
+ * Wait for a condition with timeout, allows a thread to
+ * wait for a condition and atomically releases the associated mutex.
+ *
+ * @param p_cond - pointer to the condition structure
+ * @param p_mutex - pointer to the mutex structure
+ * @param msec - Wait for msec milli seconds the most
+ * @return 0 = succeeded, 1 = failed
+ * @
+ */
+int
+NdbCondition_WaitTimeout(struct NdbCondition* p_cond,
+ NdbMutex* p_mutex,
+ int msec);
+
+
+/*
+// Signal a condition
+//
+// * p_cond: pointer to the condition structure
+// * returnvalue: 0 = succeeded, 1 = failed
+*/
+int NdbCondition_Signal(struct NdbCondition* p_cond);
+
+
+/*
+// Broadcast a condition
+//
+// * p_cond: pointer to the condition structure
+// * returnvalue: 0 = succeeded, 1 = failed
+*/
+int NdbCondition_Broadcast(struct NdbCondition* p_cond);
+
+/*
+// Destroy a condition
+//
+// * p_cond: pointer to the condition structure
+// * returnvalue: 0 = succeeded, 1 = failed
+*/
+int NdbCondition_Destroy(struct NdbCondition* p_cond);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
+
+
diff --git a/ndb/include/portlib/NdbConstant.hpp b/ndb/include/portlib/NdbConstant.hpp
new file mode 100644
index 00000000000..bd45209d2b5
--- /dev/null
+++ b/ndb/include/portlib/NdbConstant.hpp
@@ -0,0 +1,28 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_CONSTANT_HPP
+#define NDB_CONSTANT_HPP
+
+#include <ndb_types.h>
+
+#ifdef NDB_VC98
+#define STATIC_CONST(x) enum { x }
+#else
+#define STATIC_CONST(x) static const Uint32 x
+#endif
+
+#endif
diff --git a/ndb/include/portlib/NdbDaemon.h b/ndb/include/portlib/NdbDaemon.h
new file mode 100644
index 00000000000..74ea3f06419
--- /dev/null
+++ b/ndb/include/portlib/NdbDaemon.h
@@ -0,0 +1,72 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_DAEMON_H
+#define NDB_DAEMON_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/*
+ * Become a daemon.
+ * lockfile the "pid file" or other resource to lock exclusively
+ * logfile daemon output is directed here (input is set to /dev/null)
+ * if NULL, output redirection is not done
+ * flags none currently
+ * returns 0 on success, on error -1
+ */
+extern int
+NdbDaemon_Make(const char* lockfile, const char* logfile, unsigned flags);
+
+/*
+ * Test if the daemon is running (file is locked).
+ * lockfile the "pid file"
+ * flags none currently
+ * return 0 no, 1 yes, -1
+ */
+extern int
+NdbDaemon_Test(const char* lockfile, unsigned flags);
+
+/*
+ * Kill the daemon.
+ * lockfile the "pid file"
+ * flags none currently
+ * return 0 killed, 1 not running, -1 other error
+ */
+extern int
+NdbDaemon_Kill(const char* lockfile, unsigned flags);
+
+/*
+ * Pid from last call, either forked off or found in lock file.
+ */
+extern long NdbDaemon_DaemonPid;
+
+/*
+ * Error code from last failed call.
+ */
+extern int NdbDaemon_ErrorCode;
+
+/*
+ * Error text from last failed call.
+ */
+extern char NdbDaemon_ErrorText[];
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/ndb/include/portlib/NdbEnv.h b/ndb/include/portlib/NdbEnv.h
new file mode 100644
index 00000000000..1611bf3152e
--- /dev/null
+++ b/ndb/include/portlib/NdbEnv.h
@@ -0,0 +1,34 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_ENV_H
+#define NDB_ENV_H
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+ const char* NdbEnv_GetEnv(const char* name, char * buf, int buflen);
+
+#ifdef __cplusplus
+ }
+#endif
+
+#endif
+
+
+
diff --git a/ndb/include/portlib/NdbHost.h b/ndb/include/portlib/NdbHost.h
new file mode 100644
index 00000000000..90e7b781137
--- /dev/null
+++ b/ndb/include/portlib/NdbHost.h
@@ -0,0 +1,43 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_HOST_H
+#define NDB_HOST_H
+
+#ifndef NDB_WIN32
+#include <sys/param.h>
+#include <netdb.h>
+#endif
+
+#ifndef MAXHOSTNAMELEN
+#define MAXHOSTNAMELEN 255
+#endif
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+ int NdbHost_GetHostName(char*);
+ int NdbHost_GetProcessId();
+
+#ifdef __cplusplus
+ }
+#endif
+
+#endif
+
+
+
diff --git a/ndb/include/portlib/NdbMain.h b/ndb/include/portlib/NdbMain.h
new file mode 100644
index 00000000000..7cc7a877750
--- /dev/null
+++ b/ndb/include/portlib/NdbMain.h
@@ -0,0 +1,66 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBMAIN_H
+#define NDBMAIN_H
+
+#if defined NDB_SOFTOSE || defined NDB_OSE
+#include <ose.h>
+#include <shell.h>
+
+/* Define an OSE_PROCESS that can be started from osemain.con */
+#define NDB_MAIN(name) \
+int main_ ## name(int argc, const char** argv); \
+OS_PROCESS(name){ \
+ main_ ## name(0, 0); \
+ stop(current_process()); \
+ exit(0); \
+} \
+int main_ ## name(int argc, const char** argv)
+
+/* Define an function that can be started from the command line */
+#define NDB_COMMAND(name, str_name, syntax, description, stacksize) \
+int main_ ## name(int argc, const char** argv); \
+ \
+static int run_ ## name(int argc, char *argv[]){ \
+ return main_ ## name (argc, argv); \
+} \
+ \
+OS_PROCESS(init_ ## name){ \
+ shell_add_cmd_attrs(str_name, syntax, description, \
+ run_ ## name, OS_PRI_PROC, 25, stacksize); \
+ stop(current_process()); \
+ return; \
+} \
+ \
+int main_ ## name(int argc, const char** argv)
+
+
+
+
+#else
+
+#define NDB_MAIN(name) \
+int main(int argc, const char** argv)
+
+#define NDB_COMMAND(name, str_name, syntax, description, stacksize) \
+int main(int argc, const char** argv)
+
+
+#endif
+
+
+#endif
diff --git a/ndb/include/portlib/NdbMem.h b/ndb/include/portlib/NdbMem.h
new file mode 100644
index 00000000000..38ad3f60448
--- /dev/null
+++ b/ndb/include/portlib/NdbMem.h
@@ -0,0 +1,82 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_MEM_H
+#define NDB_MEM_H
+
+#include <stddef.h>
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+/**
+ * NdbMem_Create
+ * Create and initalise internal data structures for Ndb
+ */
+void NdbMem_Create(void);
+
+
+/**
+ * NdbMem_Destroy
+ * Destroy all memory allocated by NdbMem
+ */
+void NdbMem_Destroy(void);
+
+/**
+ * NdbMem_Allocate
+ * Allocate size of memory
+ * @parameter size - size in bytes of memory to allocate
+ * @returns - pointer to memory if succesful otherwise NULL
+ */
+void* NdbMem_Allocate(size_t size);
+
+/**
+ * NdbMem_AllocateAlign
+ * Allocate size of memory
+ * @parameter size - size in bytes of memory to allocate
+ * @paramter alignment - byte boundary to align the data at
+ * @returns - pointer to memory if succesful otherwise NULL
+ */
+void* NdbMem_AllocateAlign(size_t size, size_t alignment);
+
+
+/**
+ * NdbMem_Free
+ * Free the memory that ptr points to
+ * @parameter ptr - pointer to the memory to free
+ */
+void NdbMem_Free(void* ptr);
+
+/**
+ * NdbMem_MemLockAll
+ * Locks virtual memory in main memory
+ */
+int NdbMem_MemLockAll(void);
+
+/**
+ * NdbMem_MemUnlockAll
+ * Unlocks virtual memory
+ */
+int NdbMem_MemUnlockAll(void);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/ndb/include/portlib/NdbMutex.h b/ndb/include/portlib/NdbMutex.h
new file mode 100644
index 00000000000..d2cb6328b03
--- /dev/null
+++ b/ndb/include/portlib/NdbMutex.h
@@ -0,0 +1,114 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_MUTEX_H
+#define NDB_MUTEX_H
+
+#ifdef NDB_WIN32
+#include <winsock2.h>
+#include <ws2tcpip.h>
+#include <windows.h>
+#endif
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+#include <ose.h>
+typedef SEMAPHORE NdbMutex;
+#define NDB_MUTEX_INITIALIZER { 1, 0, 0 }
+#elif defined NDB_WIN32
+typedef CRITICAL_SECTION NdbMutex;
+#else
+#include <pthread.h>
+typedef pthread_mutex_t NdbMutex;
+#define NDB_MUTEX_INITIALIZER PTHREAD_MUTEX_INITIALIZER
+#endif
+
+/**
+ * Create a mutex
+ *
+ * p_mutex: pointer to the mutex structure
+ * returnvalue: pointer to the mutex structure
+ */
+NdbMutex* NdbMutex_Create(void);
+
+/**
+ * Destroy a mutex
+ *
+ * * p_mutex: pointer to the mutex structure
+ * * returnvalue: 0 = succeeded, -1 = failed
+ */
+int NdbMutex_Destroy(NdbMutex* p_mutex);
+
+/**
+ * Lock a mutex
+ *
+ * * p_mutex: pointer to the mutex structure
+ * * returnvalue: 0 = succeeded, -1 = failed
+ */
+int NdbMutex_Lock(NdbMutex* p_mutex);
+
+/**
+ * Unlock a mutex
+ *
+ * * p_mutex: pointer to the mutex structure
+ * * returnvalue: 0 = succeeded, -1 = failed
+ */
+int NdbMutex_Unlock(NdbMutex* p_mutex);
+
+/**
+ * Try to lock a mutex
+ *
+ * * p_mutex: pointer to the mutex structure
+ * * returnvalue: 0 = succeeded, -1 = failed
+ */
+int NdbMutex_Trylock(NdbMutex* p_mutex);
+
+#ifdef __cplusplus
+}
+#endif
+
+#ifdef __cplusplus
+class NdbLockable {
+ friend class Guard;
+public:
+ NdbLockable() { m_mutex = NdbMutex_Create(); }
+ ~NdbLockable() { NdbMutex_Destroy(m_mutex); }
+
+ void lock() { NdbMutex_Lock(m_mutex); }
+ void unlock(){ NdbMutex_Unlock(m_mutex);}
+ bool tryLock(){ return NdbMutex_Trylock(m_mutex) == 0;}
+
+ NdbMutex* getMutex() {return m_mutex;};
+
+protected:
+ NdbMutex * m_mutex;
+};
+
+class Guard {
+public:
+ Guard(NdbMutex *mtx) : m_mtx(mtx) { NdbMutex_Lock(m_mtx); };
+ Guard(NdbLockable & l) : m_mtx(l.m_mutex) { NdbMutex_Lock(m_mtx); };
+ ~Guard() { NdbMutex_Unlock(m_mtx); };
+private:
+ NdbMutex *m_mtx;
+};
+
+#endif
+
+#endif
diff --git a/ndb/include/portlib/NdbSleep.h b/ndb/include/portlib/NdbSleep.h
new file mode 100644
index 00000000000..3b26710154f
--- /dev/null
+++ b/ndb/include/portlib/NdbSleep.h
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBSLEEP_H
+#define NDBSLEEP_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * Sleep for some time
+ *
+ * returnvalue: true = time is up, false = failed
+ */
+int NdbSleep_MicroSleep(int microseconds);
+int NdbSleep_MilliSleep(int milliseconds);
+int NdbSleep_SecSleep(int seconds);
+
+#ifdef __cplusplus
+}
+#endif
+
+
+#endif
diff --git a/ndb/include/portlib/NdbStdio.h b/ndb/include/portlib/NdbStdio.h
new file mode 100644
index 00000000000..163b7eeef6f
--- /dev/null
+++ b/ndb/include/portlib/NdbStdio.h
@@ -0,0 +1,36 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*
+ * NdbStdio.h - stdio.h for ndb
+ *
+ *
+ */
+
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+/* On OSE Delta the snprintf is declare in outfmt.h */
+#include <outfmt.h>
+#endif
+
+#include <stdio.h>
+
+#ifdef NDB_WIN32
+#define snprintf _snprintf
+#define vsnprintf _vsnprintf
+#define strtok_r(s1, s2, l) strtok(s1, s2)
+#endif
+
diff --git a/ndb/include/portlib/NdbTCP.h b/ndb/include/portlib/NdbTCP.h
new file mode 100644
index 00000000000..6e2f18b61b2
--- /dev/null
+++ b/ndb/include/portlib/NdbTCP.h
@@ -0,0 +1,137 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_TCP_H
+#define NDB_TCP_H
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+/**
+ * Include files needed
+ */
+#include "inet.h"
+
+#include <netdb.h>
+#include <errno.h>
+
+#define NDB_NONBLOCK FNDELAY
+#define NDB_SOCKET_TYPE int
+#define NDB_INVALID_SOCKET -1
+#define NDB_CLOSE_SOCKET(x) close(x)
+
+/**
+ * socklen_t not defined in the header files of OSE
+ */
+typedef int socklen_t;
+
+#define InetErrno (* inet_errno())
+
+#endif
+
+#if defined NDB_SOLARIS || defined NDB_HPUX || defined NDB_IBMAIX || defined NDB_TRU64X
+/**
+ * Include files needed
+ */
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <netinet/tcp.h>
+#include <arpa/inet.h>
+#include <unistd.h>
+
+#include <netdb.h>
+#include <errno.h>
+#include <fcntl.h>
+
+#define NDB_NONBLOCK O_NONBLOCK
+#define NDB_SOCKET_TYPE int
+#define NDB_INVALID_SOCKET -1
+#define NDB_CLOSE_SOCKET(x) close(x)
+
+#define InetErrno errno
+
+#endif
+
+#if defined NDB_LINUX || defined NDB_MACOSX
+/**
+ * Include files needed
+ */
+#include <sys/time.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <netinet/tcp.h>
+#include <arpa/inet.h>
+#include <unistd.h>
+
+#include <netdb.h>
+#include <errno.h>
+#include <fcntl.h>
+
+#define NDB_NONBLOCK O_NONBLOCK
+#define NDB_SOCKET_TYPE int
+#define NDB_INVALID_SOCKET -1
+#define NDB_CLOSE_SOCKET(x) close(x)
+
+#define InetErrno errno
+
+#endif
+
+
+#ifdef NDB_WIN32
+/**
+ * Include files needed
+ */
+#include <winsock2.h>
+#include <ws2tcpip.h>
+#include <errno.h>
+
+#define InetErrno WSAGetLastError()
+#define EWOULDBLOCK WSAEWOULDBLOCK
+#define NDB_SOCKET_TYPE SOCKET
+#define NDB_INVALID_SOCKET INVALID_SOCKET
+#define NDB_CLOSE_SOCKET(x) closesocket(x)
+
+#endif
+
+#ifndef NDB_MACOSX
+#define NDB_SOCKLEN_T socklen_t
+#else
+#define NDB_SOCKLEN_T int
+#endif
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * Convert host name or ip address to in_addr
+ *
+ * Returns 0 on success
+ * -1 on failure
+ *
+ * Implemented as:
+ * gethostbyname
+ * if not success
+ * inet_addr
+ */
+int Ndb_getInAddr(struct in_addr * dst, const char *address);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/ndb/include/portlib/NdbThread.h b/ndb/include/portlib/NdbThread.h
new file mode 100644
index 00000000000..516022903e3
--- /dev/null
+++ b/ndb/include/portlib/NdbThread.h
@@ -0,0 +1,103 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_THREAD_H
+#define NDB_THREAD_H
+
+#include <sys/types.h>
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+typedef enum NDB_THREAD_PRIO_ENUM {
+ NDB_THREAD_PRIO_HIGHEST,
+ NDB_THREAD_PRIO_HIGH,
+ NDB_THREAD_PRIO_MEAN,
+ NDB_THREAD_PRIO_LOW,
+ NDB_THREAD_PRIO_LOWEST
+} NDB_THREAD_PRIO;
+
+typedef void* (NDB_THREAD_FUNC)(void*);
+typedef void* NDB_THREAD_ARG;
+typedef size_t NDB_THREAD_STACKSIZE;
+
+struct NdbThread;
+
+/**
+ * Create a thread
+ *
+ * * p_thread_func: pointer of the function to run in the thread
+ * * p_thread_arg: pointer to argument to be passed to the thread
+ * * thread_stack_size: stack size for this thread
+ * * p_thread_name: pointer to name of this thread
+ * * returnvalue: pointer to the created thread
+ */
+struct NdbThread* NdbThread_Create(NDB_THREAD_FUNC *p_thread_func,
+ NDB_THREAD_ARG *p_thread_arg,
+ const NDB_THREAD_STACKSIZE thread_stack_size,
+ const char* p_thread_name,
+ NDB_THREAD_PRIO thread_prio);
+
+/**
+ * Destroy a thread
+ * Deallocates memory for thread
+ * And NULL the pointer
+ *
+ */
+void NdbThread_Destroy(struct NdbThread** p_thread);
+
+
+/**
+ * Waitfor a thread, suspend the execution of the calling thread
+ * until the wait_thread_id completes
+ *
+ * * p_wait_thread, pointer to the thread to wait for
+ * * status: exit code from thread waited for
+ * * returnvalue: true = succeded, false = failed
+ */
+int NdbThread_WaitFor(struct NdbThread* p_wait_thread, void** status);
+
+/**
+ * Exit thread, terminates the calling thread
+ *
+ * * status: exit code
+ */
+void NdbThread_Exit(int status);
+
+/**
+ * Set thread concurrency level
+ *
+ * *
+ */
+int NdbThread_SetConcurrencyLevel(int level);
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
+
+
+
+
+
+
+
+
+
diff --git a/ndb/include/portlib/NdbTick.h b/ndb/include/portlib/NdbTick.h
new file mode 100644
index 00000000000..762f65331cc
--- /dev/null
+++ b/ndb/include/portlib/NdbTick.h
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_TICK_H
+#define NDB_TICK_H
+
+#include <ndb_types.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+typedef unsigned long NDB_TICKS;
+#else
+typedef Uint64 NDB_TICKS;
+#endif
+
+/**
+ * Returns the current millisecond since 1970
+ */
+NDB_TICKS NdbTick_CurrentMillisecond(void);
+
+/**
+ * Get current micro second
+ * Second method is simply abstraction on top of the first
+ *
+ * Returns 0 - Success
+ */
+int NdbTick_CurrentMicrosecond(NDB_TICKS * secs, Uint32 * micros);
+
+//#define TIME_MEASUREMENT
+#ifdef TIME_MEASUREMENT
+
+struct MicroSecondTimer {
+ NDB_TICKS seconds;
+ NDB_TICKS micro_seconds;
+};
+
+/**
+ * Get time between start and stop time in microseconds
+ * Abstraction to get time in struct
+ *
+ * 0 means stop happened at or before start time
+ */
+NDB_TICKS NdbTick_getMicrosPassed(struct MicroSecondTimer start,
+ struct MicroSecondTimer stop);
+int NdbTick_getMicroTimer(struct MicroSecondTimer* time_now);
+#endif
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
+
diff --git a/ndb/include/portlib/NdbUnistd.h b/ndb/include/portlib/NdbUnistd.h
new file mode 100644
index 00000000000..42f67e2aeb3
--- /dev/null
+++ b/ndb/include/portlib/NdbUnistd.h
@@ -0,0 +1,39 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_UNISTD_H
+#define NDB_UNISTD_H
+
+#ifdef NDB_WIN32
+#include <winsock2.h>
+#include <ws2tcpip.h>
+#include <windows.h>
+#include <limits.h>
+
+#define DIR_SEPARATOR "\\"
+#define PATH_MAX 256
+
+#pragma warning(disable: 4503 4786)
+
+#else
+#include <unistd.h>
+#include <limits.h>
+
+#define DIR_SEPARATOR "/"
+
+#endif
+
+#endif
diff --git a/ndb/include/portlib/PortDefs.h b/ndb/include/portlib/PortDefs.h
new file mode 100644
index 00000000000..6cd5be0149f
--- /dev/null
+++ b/ndb/include/portlib/PortDefs.h
@@ -0,0 +1,96 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PORT_DEFS_H
+#define PORT_DEFS_H
+/*
+ This file contains varoius declarations/definitions needed in the port of AXEVM to NT, as well as backporting
+ to Solaris...
+
+ $Id: PortDefs.h,v 1.5 2003/10/07 07:59:59 mikael Exp $
+*/
+#ifdef NDB_WIN32
+#include <time.h>
+
+
+struct tms
+{
+ time_t tms_utime; // user time
+ time_t tms_stime; // system time
+ time_t tms_cutime; // user time of children
+ time_t tms_cstime; // system time of children
+};
+
+struct timespec
+{
+ long tv_sec; // Seconds
+ long tv_nsec; // Nanoseconds
+};
+
+#define strcasecmp(a,b) _strcmpi(a,b)
+
+ // Exports a WIN32 getopt function
+extern int optind;
+extern char *optarg;
+int getopt(int, char **, char *opts);
+#endif // NDB_WIN32
+
+#ifdef NDB_ALPHA
+#ifdef NDB_GCC
+extern int gnuShouldNotUseRPCC();
+#define RPCC() gnuShouldNotUseRPCC();
+#else
+#ifdef NDB_WIN32
+#ifdef __cplusplus
+extern "C" {
+#endif //__cplusplus
+ u_int64 __asm(char *, ...);
+ double __dasm(char *, ...);
+ float __fasm(char *, ...);
+ void _AcquireSpinLock(long *);
+ void _ReleaseSpinLock(long *);
+ int __ADD_ATOMIC_LONG2(void *, int);
+#ifdef __cplusplus
+};
+#endif //__cplusplus
+#pragma intrinsic (__asm, __dasm, __fasm)
+#pragma intrinsic(_ReleaseSpinLock, _AcquireSpinLock)
+#pragma intrinsic(__ADD_ATOMIC_LONG2)
+#endif // NDB_WIN32
+
+#define RPCC() ((int)__asm(" rpcc v0;"))
+#define MB() __asm(" mb;");
+#define WMB() __asm(" wmb;");
+#ifdef USE_INITIALSP
+#define IS_IP() (__asm(" mov sp,v0;") < IPinitialSP)
+#else // USE_INITIALSP
+#define IS_IP() (((__asm(" rpcc v0;") >> 32) & 0x7) == IP_CPU)
+#endif
+#endif //NDB_GCC
+#else // NDB_ALPHA
+#if defined NDB_SPARC
+#define MB() asm ("membar 0x0;"); // LoadLoad
+#define WMB() asm ("membar 0x3;"); // StoreStore
+#else // NDB_SPARC
+#define MB()
+#define WMB()
+#endif // NDB_SPARC
+#define IS_IP() (1==1)
+extern int shouldNotUseRPCC();
+#define RPCC() shouldNotUseRPCC();
+#endif // NDB_ALPHA
+
+#endif
diff --git a/ndb/include/portlib/prefetch.h b/ndb/include/portlib/prefetch.h
new file mode 100644
index 00000000000..d663dd4c40d
--- /dev/null
+++ b/ndb/include/portlib/prefetch.h
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PREFETCH_H
+#define PREFETCH_H
+
+#ifdef NDB_FORTE6
+#include <sun_prefetch.h>
+#endif
+
+#ifdef USE_PREFETCH
+#define PREFETCH(addr) prefetch(addr)
+#else
+#define PREFETCH(addr)
+#endif
+
+#ifdef USE_PREFETCH
+#define WRITEHINT(addr) writehint(addr)
+#else
+#define WRITEHINT(addr)
+#endif
+
+#include "PortDefs.h"
+
+#ifdef NDB_FORTE6
+#pragma optimize("", off)
+#endif
+inline void prefetch(void* p)
+{
+#ifdef NDB_ALPHA
+ __asm(" ldl r31,0(a0);", p);
+#endif // NDB_ALPHA
+#ifdef NDB_FORTE6
+ sparc_prefetch_read_once(p);
+#else
+ (void)p;
+#endif
+}
+
+inline void writehint(void* p)
+{
+#ifdef NDB_ALPHA
+ __asm(" wh64 (a0);", p);
+#endif // NDB_ALPHA
+#ifdef NDB_FORTE6
+ sparc_prefetch_write_once(p);
+#else
+ (void)p;
+#endif
+}
+#ifdef NDB_FORTE6
+#pragma optimize("", on)
+#endif
+
+#endif
+
diff --git a/ndb/include/transporter/TransporterCallback.hpp b/ndb/include/transporter/TransporterCallback.hpp
new file mode 100644
index 00000000000..9f910f31728
--- /dev/null
+++ b/ndb/include/transporter/TransporterCallback.hpp
@@ -0,0 +1,345 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//****************************************************************************
+//
+// AUTHOR
+// Åsa Fransson
+//
+// NAME
+// TransporterCallback
+//
+//
+//***************************************************************************/
+#ifndef TRANSPORTER_CALLBACK_H
+#define TRANSPORTER_CALLBACK_H
+
+#include <kernel_types.h>
+#include "TransporterDefinitions.hpp"
+
+
+/**
+ * Call back functions
+ */
+
+/**
+ * The execute function
+ */
+void
+execute(void * callbackObj,
+ SignalHeader * const header,
+ Uint8 prio,
+ Uint32 * const signalData,
+ LinearSectionPtr ptr[3]);
+
+/**
+ * A function to avoid job buffer overflow in NDB kernel, empty in API
+ * Non-zero return means we executed signals. This is necessary information
+ * to the transporter to ensure that it properly uses the transporter after
+ * coming back again.
+ */
+int
+checkJobBuffer();
+
+/**
+ * Report send length
+ */
+void
+reportSendLen(void * callbackObj,
+ NodeId nodeId, Uint32 count, Uint64 bytes);
+
+/**
+ * Report average receive length
+ */
+void
+reportReceiveLen(void * callbackObj,
+ NodeId nodeId, Uint32 count, Uint64 bytes);
+
+/**
+ * Report connection established
+ */
+void
+reportConnect(void * callbackObj, NodeId nodeId);
+
+/**
+ * Report connection broken
+ */
+
+void
+reportDisconnect(void * callbackObj,
+ NodeId nodeId, Uint32 errNo);
+
+enum TransporterError {
+ TE_NO_ERROR = 0,
+ /**
+ * TE_ERROR_CLOSING_SOCKET
+ *
+ * Error found during closing of socket
+ *
+ * Recommended behavior: Ignore
+ */
+ TE_ERROR_CLOSING_SOCKET = 0x1,
+
+ /**
+ * TE_ERROR_IN_SELECT_BEFORE_ACCEPT
+ *
+ * Error found during accept (just before)
+ * The transporter will retry.
+ *
+ * Recommended behavior: Ignore
+ * (or possible do setPerformState(PerformDisconnect)
+ */
+ TE_ERROR_IN_SELECT_BEFORE_ACCEPT = 0x2,
+
+ /**
+ * TE_INVALID_MESSAGE_LENGTH
+ *
+ * Error found in message (message length)
+ *
+ * Recommended behavior: setPerformState(PerformDisconnect)
+ */
+ TE_INVALID_MESSAGE_LENGTH = 0x8003,
+
+ /**
+ * TE_INVALID_CHECKSUM
+ *
+ * Error found in message (checksum)
+ *
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ TE_INVALID_CHECKSUM = 0x8004,
+
+ /**
+ * TE_COULD_NOT_CREATE_SOCKET
+ *
+ * Error found while creating socket
+ *
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ TE_COULD_NOT_CREATE_SOCKET = 0x8005,
+
+ /**
+ * TE_COULD_NOT_BIND_SOCKET
+ *
+ * Error found while binding server socket
+ *
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ TE_COULD_NOT_BIND_SOCKET = 0x8006,
+
+ /**
+ * TE_LISTEN_FAILED
+ *
+ * Error found while listening to server socket
+ *
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ TE_LISTEN_FAILED = 0x8007,
+
+ /**
+ * TE_ACCEPT_RETURN_ERROR
+ *
+ * Error found during accept
+ * The transporter will retry.
+ *
+ * Recommended behavior: Ignore
+ * (or possible do setPerformState(PerformDisconnect)
+ */
+ TE_ACCEPT_RETURN_ERROR = 0x8008
+
+ /**
+ * TE_SHM_DISCONNECT
+ *
+ * The remote node has disconnected
+ *
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SHM_DISCONNECT = 0x800b
+
+ /**
+ * TE_SHM_IPC_STAT
+ *
+ * Unable to check shm segment
+ * probably because remote node
+ * has disconnected and removed it
+ *
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SHM_IPC_STAT = 0x800c
+
+ /**
+ * TE_SHM_UNABLE_TO_CREATE_SEGMENT
+ *
+ * Unable to create shm segment
+ * probably os something error
+ *
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SHM_UNABLE_TO_CREATE_SEGMENT = 0x800d
+
+ /**
+ * TE_SHM_UNABLE_TO_ATTACH_SEGMENT
+ *
+ * Unable to attach shm segment
+ * probably invalid group / user
+ *
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SHM_UNABLE_TO_ATTACH_SEGMENT = 0x800e
+
+ /**
+ * TE_SHM_UNABLE_TO_REMOVE_SEGMENT
+ *
+ * Unable to remove shm segment
+ *
+ * Recommended behavior: Ignore (not much to do)
+ * Print warning to logfile
+ */
+ ,TE_SHM_UNABLE_TO_REMOVE_SEGMENT = 0x800f
+
+ ,TE_TOO_SMALL_SIGID = 0x0010
+ ,TE_TOO_LARGE_SIGID = 0x0011
+ ,TE_WAIT_STACK_FULL = 0x8012
+ ,TE_RECEIVE_BUFFER_FULL = 0x8013
+
+ /**
+ * TE_SIGNAL_LOST_SEND_BUFFER_FULL
+ *
+ * Send buffer is full, and trying to force send fails
+ * a signal is dropped!! very bad very bad
+ *
+ */
+ ,TE_SIGNAL_LOST_SEND_BUFFER_FULL = 0x8014
+
+ /**
+ * TE_SIGNAL_LOST
+ *
+ * Send failed for unknown reason
+ * a signal is dropped!! very bad very bad
+ *
+ */
+ ,TE_SIGNAL_LOST = 0x8015
+
+ /**
+ * TE_SEND_BUFFER_FULL
+ *
+ * The send buffer was full, but sleeping for a while solved it
+ */
+ ,TE_SEND_BUFFER_FULL = 0x0016
+
+ /**
+ * TE_SCI_UNABLE_TO_CLOSE_CHANNEL
+ *
+ * Unable to close the sci channel and the resources allocated by
+ * the sisci api.
+ */
+ ,TE_SCI_UNABLE_TO_CLOSE_CHANNEL = 0x8016
+
+ /**
+ * TE_SCI_LINK_ERROR
+ *
+ * There is no link from this node to the switch.
+ * No point in continuing. Must check the connections.
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SCI_LINK_ERROR = 0x8017
+
+ /**
+ * TE_SCI_UNABLE_TO_START_SEQUENCE
+ *
+ * Could not start a sequence, because system resources
+ * are exumed or no sequence has been created.
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SCI_UNABLE_TO_START_SEQUENCE = 0x8018
+
+ /**
+ * TE_SCI_UNABLE_TO_REMOVE_SEQUENCE
+ *
+ * Could not remove a sequence
+ */
+ ,TE_SCI_UNABLE_TO_REMOVE_SEQUENCE = 0x8019
+
+ /**
+ * TE_SCI_UNABLE_TO_CREATE_SEQUENCE
+ *
+ * Could not create a sequence, because system resources are
+ * exempted. Must reboot.
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SCI_UNABLE_TO_CREATE_SEQUENCE = 0x801a
+
+ /**
+ * TE_SCI_UNRECOVERABLE_DATA_TFX_ERROR
+ *
+ * Tried to send data on redundant link but failed.
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SCI_UNRECOVERABLE_DATA_TFX_ERROR = 0x801b
+
+ /**
+ * TE_SCI_CANNOT_INIT_LOCALSEGMENT
+ *
+ * Cannot initialize local segment. A whole lot of things has
+ * gone wrong (no system resources). Must reboot.
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SCI_CANNOT_INIT_LOCALSEGMENT = 0x801c
+
+ /**
+ * TE_SCI_CANNOT_MAP_REMOTESEGMENT
+ *
+ * Cannot map remote segment. No system resources are left.
+ * Must reboot system.
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SCI_CANNOT_MAP_REMOTESEGMENT = 0x801d
+
+ /**
+ * TE_SCI_UNABLE_TO_UNMAP_SEGMENT
+ *
+ * Cannot free the resources used by this segment (step 1).
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SCI_UNABLE_TO_UNMAP_SEGMENT = 0x801e
+
+ /**
+ * TE_SCI_UNABLE_TO_REMOVE_SEGMENT
+ *
+ * Cannot free the resources used by this segment (step 2).
+ * Cannot guarantee that enough resources exist for NDB
+ * to map more segment
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SCI_UNABLE_TO_REMOVE_SEGMENT = 0x801f
+
+ /**
+ * TE_SCI_UNABLE_TO_DISCONNECT_SEGMENT
+ *
+ * Cannot disconnect from a remote segment.
+ * Recommended behavior: setPerformState(PerformDisonnect)
+ */
+ ,TE_SCI_UNABLE_TO_DISCONNECT_SEGMENT = 0x8020
+
+};
+
+/**
+ * Report error
+ */
+void
+reportError(void * callbackObj, NodeId nodeId, TransporterError errorCode);
+
+#endif
diff --git a/ndb/include/transporter/TransporterDefinitions.hpp b/ndb/include/transporter/TransporterDefinitions.hpp
new file mode 100644
index 00000000000..5bbf7c79491
--- /dev/null
+++ b/ndb/include/transporter/TransporterDefinitions.hpp
@@ -0,0 +1,152 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TransporterDefinitions_H
+#define TransporterDefinitions_H
+
+#include <kernel_types.h>
+#include <string.h>
+#include <NdbOut.hpp>
+
+/**
+ * The maximum number of transporters allowed
+ * A maximum is needed to be able to allocate the array of transporters
+ */
+const int MAX_NTRANSPORTERS = 128;
+
+/**
+ * The sendbuffer limit after which the contents of the buffer is sent
+ */
+const int TCP_SEND_LIMIT = 64000;
+
+enum SendStatus {
+ SEND_OK = 0,
+ SEND_BLOCKED = 1,
+ SEND_DISCONNECTED = 2,
+ SEND_BUFFER_FULL = 3,
+ SEND_MESSAGE_TOO_BIG = 4,
+ SEND_UNKNOWN_NODE = 5
+};
+
+/**
+ * Protocol6 Header +
+ * (optional signal id) + (optional checksum) + (signal data)
+ */
+//const Uint32 MAX_MESSAGE_SIZE = (12+4+4+(4*25));
+const Uint32 MAX_MESSAGE_SIZE = (12+4+4+(4*25)+(3*4)+4*4096);
+
+/**
+ * TCP Transporter Configuration
+ */
+struct TCP_TransporterConfiguration {
+ Uint32 port;
+ const char *remoteHostName;
+ const char *localHostName;
+ NodeId remoteNodeId;
+ NodeId localNodeId;
+ Uint32 sendBufferSize; // Size of SendBuffer of priority B
+ Uint32 maxReceiveSize; // Maximum no of bytes to receive
+ Uint32 byteOrder;
+ bool compression;
+ bool checksum;
+ bool signalId;
+};
+
+/**
+ * SHM Transporter Configuration
+ */
+struct SHM_TransporterConfiguration {
+ NodeId remoteNodeId;
+ NodeId localNodeId;
+ bool compression;
+ bool checksum;
+ bool signalId;
+ int byteOrder;
+
+ Uint32 shmKey;
+ Uint32 shmSize;
+};
+
+/**
+ * OSE Transporter Configuration
+ */
+struct OSE_TransporterConfiguration {
+ const char *remoteHostName;
+ const char *localHostName;
+ NodeId remoteNodeId;
+ NodeId localNodeId;
+ bool compression;
+ bool checksum;
+ bool signalId;
+ int byteOrder;
+
+ Uint32 prioASignalSize;
+ Uint32 prioBSignalSize;
+ Uint32 receiveBufferSize; // In number of signals
+};
+
+/**
+ * SCI Transporter Configuration
+ */
+struct SCI_TransporterConfiguration {
+ Uint32 sendLimit; // Packet size
+ Uint32 bufferSize; // Buffer size
+
+ Uint32 nLocalAdapters; // 1 or 2, the number of adapters on local host
+
+ Uint32 nRemoteAdapters;
+ Uint32 remoteSciNodeId0; // SCInodeId for adapter 1
+ Uint32 remoteSciNodeId1; // SCInodeId for adapter 2
+
+ NodeId localNodeId; // Local node Id
+ NodeId remoteNodeId; // Remote node Id
+
+ Uint32 byteOrder;
+ bool compression;
+ bool checksum;
+ bool signalId;
+
+};
+
+struct SignalHeader {
+ Uint32 theVerId_signalNumber; // 4 bit ver id - 16 bit gsn
+ Uint32 theReceiversBlockNumber; // Only 16 bit blocknum
+ Uint32 theSendersBlockRef;
+ Uint32 theLength;
+ Uint32 theSendersSignalId;
+ Uint32 theSignalId;
+ Uint16 theTrace;
+ Uint8 m_noOfSections;
+ Uint8 m_fragmentInfo;
+}; /** 7x4 = 32 Bytes */
+
+struct LinearSectionPtr {
+ Uint32 sz;
+ Uint32 * p;
+};
+
+struct SegmentedSectionPtr {
+ Uint32 sz;
+ Uint32 i;
+ struct SectionSegment * p;
+
+ void setNull() { p = 0;}
+ bool isNull() const { return p == 0;}
+};
+
+class NdbOut & operator <<(class NdbOut & out, SignalHeader & sh);
+
+#endif // Define of TransporterDefinitions_H
diff --git a/ndb/include/transporter/TransporterRegistry.hpp b/ndb/include/transporter/TransporterRegistry.hpp
new file mode 100644
index 00000000000..6c979777f18
--- /dev/null
+++ b/ndb/include/transporter/TransporterRegistry.hpp
@@ -0,0 +1,281 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//****************************************************************************
+//
+// NAME
+// TransporterRegistry
+//
+// DESCRIPTION
+// TransporterRegistry (singelton) is the interface to the
+// transporter layer. It handles transporter states and
+// holds the transporter arrays.
+//
+//***************************************************************************/
+#ifndef TransporterRegistry_H
+#define TransporterRegistry_H
+
+#include "TransporterDefinitions.hpp"
+
+#include <NdbTCP.h>
+
+// A transporter is always in a PerformState.
+// PerformIO is used initially and as long as any of the events
+// PerformConnect, ...
+enum PerformState {
+ PerformNothing = 4, // Does nothing
+ PerformIO = 0, // Is connected
+ PerformConnect = 1, // Is trying to connect
+ PerformDisconnect = 2, // Trying to disconnect
+ RemoveTransporter = 3 // Will be removed
+};
+
+// A transporter is always in an IOState.
+// NoHalt is used initially and as long as it is no restrictions on
+// sending or receiving.
+enum IOState {
+ NoHalt = 0,
+ HaltInput = 1,
+ HaltOutput = 2,
+ HaltIO = 3
+};
+
+enum TransporterType {
+ tt_TCP_TRANSPORTER = 1,
+ tt_SCI_TRANSPORTER = 2,
+ tt_SHM_TRANSPORTER = 3,
+ tt_OSE_TRANSPORTER = 4
+};
+
+class Transporter;
+class TCP_Transporter;
+class SCI_Transporter;
+class SHM_Transporter;
+class OSE_Transporter;
+
+/**
+ * @class TransporterRegistry
+ * @brief ...
+ */
+class TransporterRegistry {
+ friend class OSE_Receiver;
+public:
+ /**
+ * Constructor
+ */
+ TransporterRegistry(void * callback = 0 ,
+ unsigned maxTransporters = MAX_NTRANSPORTERS,
+ unsigned sizeOfLongSignalMemory = 100);
+
+ bool init(NodeId localNodeId);
+
+ /**
+ * Remove all transporters
+ */
+ void removeAll();
+
+ /**
+ * Disconnect all transporters
+ */
+ void disconnectAll();
+
+ /**
+ * Stops the server, disconnects all the transporter
+ * and deletes them and remove it from the transporter arrays
+ */
+ ~TransporterRegistry();
+
+ /**
+ * Start/Stop receiving
+ */
+ void startReceiving();
+ void stopReceiving();
+
+ /**
+ * Start/Stop sending
+ */
+ void startSending();
+ void stopSending();
+
+ /**
+ * Get and set methods for PerformState
+ */
+ PerformState performState(NodeId nodeId);
+ void setPerformState(NodeId nodeId, PerformState state);
+
+ /**
+ * Set perform state for all transporters
+ */
+ void setPerformState(PerformState state);
+
+ /**
+ * Get and set methods for IOState
+ */
+ IOState ioState(NodeId nodeId);
+ void setIOState(NodeId nodeId, IOState state);
+
+ /**
+ * createTransporter
+ *
+ * If the config object indicates that the transporter
+ * to be created will act as a server and no server is
+ * started, startServer is called. A transporter of the selected kind
+ * is created and it is put in the transporter arrays.
+ */
+ bool createTransporter(struct TCP_TransporterConfiguration * config);
+ bool createTransporter(struct SCI_TransporterConfiguration * config);
+ bool createTransporter(struct SHM_TransporterConfiguration * config);
+ bool createTransporter(struct OSE_TransporterConfiguration * config);
+
+ /**
+ * prepareSend
+ *
+ * When IOState is HaltOutput or HaltIO do not send or insert any
+ * signals in the SendBuffer, unless it is intended for the remote
+ * CMVMI block (blockno 252)
+ * Perform prepareSend on the transporter.
+ *
+ * NOTE signalHeader->xxxBlockRef should contain block numbers and
+ * not references
+ */
+ SendStatus prepareSend(const SignalHeader * const signalHeader, Uint8 prio,
+ const Uint32 * const signalData,
+ NodeId nodeId,
+ const LinearSectionPtr ptr[3]);
+
+ SendStatus prepareSend(const SignalHeader * const signalHeader, Uint8 prio,
+ const Uint32 * const signalData,
+ NodeId nodeId,
+ class SectionSegmentPool & pool,
+ const SegmentedSectionPtr ptr[3]);
+
+ /**
+ * external_IO
+ *
+ * Equal to: poll(...); perform_IO()
+ *
+ */
+ void external_IO(Uint32 timeOutMillis);
+
+ Uint32 pollReceive(Uint32 timeOutMillis);
+ void performReceive();
+ void performSend();
+
+ void checkConnections();
+
+ /**
+ * Force sending if more than or equal to sendLimit
+ * number have asked for send. Returns 0 if not sending
+ * and 1 if sending.
+ */
+ int forceSendCheck(int sendLimit);
+
+#ifdef DEBUG_TRANSPORTER
+ void printState();
+#endif
+
+protected:
+
+private:
+ void * callbackObj;
+
+ int sendCounter;
+ NodeId localNodeId;
+ bool nodeIdSpecified;
+ unsigned maxTransporters;
+ int nTransporters;
+ int nTCPTransporters;
+ int nSCITransporters;
+ int nSHMTransporters;
+ int nOSETransporters;
+
+ int m_ccCount;
+ int m_ccIndex;
+ int m_ccStep;
+ int m_nTransportersPerformConnect;
+ bool m_ccReady;
+ /**
+ * Arrays holding all transporters in the order they are created
+ */
+ TCP_Transporter** theTCPTransporters;
+ SCI_Transporter** theSCITransporters;
+ SHM_Transporter** theSHMTransporters;
+ OSE_Transporter** theOSETransporters;
+
+ /**
+ * Array, indexed by nodeId, holding all transporters
+ */
+ TransporterType* theTransporterTypes;
+ Transporter** theTransporters;
+
+ /**
+ * OSE Receiver
+ */
+ class OSE_Receiver * theOSEReceiver;
+
+ /**
+ * In OSE you for some bizar reason needs to create a socket
+ * the first thing you do when using inet functions.
+ *
+ * Furthermore a process doing select has to "own" a socket
+ *
+ */
+ int theOSEJunkSocketSend;
+ int theOSEJunkSocketRecv;
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ PROCESS theReceiverPid;
+#endif
+
+ /**
+ * State arrays, index by host id
+ */
+ PerformState* performStates;
+ IOState* ioStates;
+
+ /**
+ * Unpack signal data
+ */
+ Uint32 unpack(Uint32 * readPtr,
+ Uint32 bufferSize,
+ NodeId remoteNodeId,
+ IOState state);
+
+ Uint32 * unpack(Uint32 * readPtr,
+ Uint32 * eodPtr,
+ NodeId remoteNodeId,
+ IOState state);
+
+ /**
+ * Disconnect the transporter and remove it from
+ * theTransporters array. Do not allow any holes
+ * in theTransporters. Delete the transporter
+ * and remove it from theIndexedTransporters array
+ */
+ void removeTransporter(NodeId nodeId);
+
+ /**
+ * Used in polling if exists TCP_Transporter
+ */
+ int tcpReadSelectReply;
+ fd_set tcpReadset;
+
+ Uint32 poll_OSE(Uint32 timeOutMillis);
+ Uint32 poll_TCP(Uint32 timeOutMillis);
+ Uint32 poll_SCI(Uint32 timeOutMillis);
+ Uint32 poll_SHM(Uint32 timeOutMillis);
+};
+
+#endif // Define of TransporterRegistry_H
diff --git a/ndb/include/util/Base64.hpp b/ndb/include/util/Base64.hpp
new file mode 100644
index 00000000000..a8678da946c
--- /dev/null
+++ b/ndb/include/util/Base64.hpp
@@ -0,0 +1,26 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __BASE64_HPP_INCLUDED__
+#define __BASE64_HPP_INCLUDED__
+
+#include <UtilBuffer.hpp>
+#include <BaseString.hpp>
+
+int base64_encode(UtilBuffer &src, BaseString &dst);
+int base64_decode(BaseString &src, UtilBuffer &dst);
+
+#endif /* !__BASE64_HPP_INCLUDED__ */
diff --git a/ndb/include/util/BaseString.hpp b/ndb/include/util/BaseString.hpp
new file mode 100644
index 00000000000..a88bd97ffc5
--- /dev/null
+++ b/ndb/include/util/BaseString.hpp
@@ -0,0 +1,260 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __UTIL_BASESTRING_HPP_INCLUDED__
+#define __UTIL_BASESTRING_HPP_INCLUDED__
+
+#include <ctype.h>
+#include <stdlib.h>
+
+#include <Vector.hpp>
+
+/**
+ * @class BaseString
+ * @brief Null terminated strings
+ */
+class BaseString {
+public:
+ /** @brief Constructs an empty string */
+ BaseString();
+
+ /** @brief Constructs a copy of a char * */
+ BaseString(const char* s);
+
+ /** @brief Constructs a copy of another BaseString */
+ BaseString(const BaseString& str);
+
+ /** @brief Destructor */
+ ~BaseString();
+
+ /** @brief Returns a C-style string */
+ const char* c_str() const;
+
+ /** @brief Returns the length of the string */
+ unsigned length() const;
+
+ /** @brief Checks if the string is empty */
+ bool empty() const;
+
+ /** @brief Convert to uppercase */
+ void ndb_toupper();
+
+ /** @brief Convert to lowercase */
+ void ndb_tolower();
+
+ /** @brief Assigns from a char * */
+ BaseString& assign(const char* s);
+
+ /** @brief Assigns from another BaseString */
+ BaseString& assign(const BaseString& str);
+
+ /** @brief Assigns from char *s, with maximum length n */
+ BaseString& assign(const char* s, size_t n);
+
+ /** @brief Assigns from another BaseString, with maximum length n */
+ BaseString& assign(const BaseString& str, size_t n);
+
+ /**
+ * Assings from a Vector of BaseStrings, each Vector entry
+ * separated by separator.
+ *
+ * @param vector Vector of BaseStrings to append
+ * @param separator Separation between appended strings
+ */
+ BaseString& assign(const Vector<BaseString> &vector,
+ const BaseString &separator = BaseString(" "));
+
+ /** @brief Appends a char * to the end */
+ BaseString& append(const char* s);
+
+ /** @brief Appends a char to the end */
+ BaseString& append(char c);
+
+ /** @brief Appends another BaseString to the end */
+ BaseString& append(const BaseString& str);
+
+ /**
+ * Appends a Vector of BaseStrings to the end, each Vector entry
+ * separated by separator.
+ *
+ * @param vector Vector of BaseStrings to append
+ * @param separator Separation between appended strings
+ */
+ BaseString& append(const Vector<BaseString> &vector,
+ const BaseString &separator = BaseString(" "));
+
+ /** @brief Assigns from a format string a la printf() */
+ BaseString& assfmt(const char* ftm, ...);
+
+ /** @brief Appends a format string a la printf() to the end */
+ BaseString& appfmt(const char* ftm, ...);
+
+ /**
+ * Split a string into a vector of strings. Separate the string where
+ * any character included in separator exists.
+ * Maximally maxSize entries are added to the vector, if more separators
+ * exist in the string, the remainder of the string will be appended
+ * to the last entry in the vector.
+ * The vector will not be cleared, so any existing strings in the
+ * vector will remain.
+ *
+ * @param separator characters separating the entries
+ * @param vector where the separated strings will be stored
+ * @param maximum number of strings extracted
+ *
+ * @returns the number of string added to the vector
+ */
+ int split(Vector<BaseString> &vector,
+ const BaseString &separator = BaseString(" "),
+ int maxSize = -1) const;
+
+ /**
+ * Returns the index of the first occurance of the character c.
+ *
+ * @params c character to look for
+ * @returns index of character, of -1 if no character found
+ */
+ ssize_t indexOf(char c);
+
+ /**
+ * Returns the index of the last occurance of the character c.
+ *
+ * @params c character to look for
+ * @returns index of character, of -1 if no character found
+ */
+ ssize_t lastIndexOf(char c);
+
+ /**
+ * Returns a subset of a string
+ *
+ * @param start index of first character
+ * @param stop index of last character
+ * @return a new string
+ */
+ BaseString substr(ssize_t start, ssize_t stop = -1);
+
+ /**
+ * @brief Assignment operator
+ */
+ BaseString& operator=(const BaseString& str);
+
+ /** @brief Compare two strings */
+ bool operator<(const BaseString& str) const;
+ /** @brief Are two strings equal? */
+ bool operator==(const BaseString& str) const;
+ /** @brief Are two strings equal? */
+ bool operator==(const char *str) const;
+ /** @brief Are two strings not equal? */
+ bool operator!=(const BaseString& str) const;
+ /** @brief Are two strings not equal? */
+ bool operator!=(const char *str) const;
+
+ /**
+ * Trim string from <i>delim</i>
+ */
+ BaseString& trim(const char * delim = " \t");
+
+ /**
+ * Return c-array with strings suitable for execve
+ * When whitespace is detected, the characters '"' and '\' are honored,
+ * to make it possible to give arguments containing whitespace.
+ * The semantics of '"' and '\' match that of most Unix shells.
+ */
+ static char** argify(const char *argv0, const char *src);
+
+ /**
+ * Trim string from <i>delim</i>
+ */
+ static char* trim(char * src, const char * delim = " \t");
+private:
+ char* m_chr;
+ unsigned m_len;
+};
+
+inline const char*
+BaseString::c_str() const
+{
+ return m_chr;
+}
+
+inline unsigned
+BaseString::length() const
+{
+ return m_len;
+}
+
+inline bool
+BaseString::empty() const
+{
+ return m_len == 0;
+}
+
+inline void
+BaseString::ndb_toupper() {
+ for(unsigned i = 0; i < length(); i++)
+ m_chr[i] = ::toupper(m_chr[i]);
+}
+
+inline void
+BaseString::ndb_tolower() {
+ for(unsigned i = 0; i < length(); i++)
+ m_chr[i] = ::tolower(m_chr[i]);
+}
+
+inline bool
+BaseString::operator<(const BaseString& str) const
+{
+ return strcmp(m_chr, str.m_chr) < 0;
+}
+
+inline bool
+BaseString::operator==(const BaseString& str) const
+{
+ return strcmp(m_chr, str.m_chr) == 0;
+}
+
+inline bool
+BaseString::operator==(const char *str) const
+{
+ return strcmp(m_chr, str) == 0;
+}
+
+inline bool
+BaseString::operator!=(const BaseString& str) const
+{
+ return strcmp(m_chr, str.m_chr) != 0;
+}
+
+inline bool
+BaseString::operator!=(const char *str) const
+{
+ return strcmp(m_chr, str) != 0;
+}
+
+inline BaseString&
+BaseString::assign(const BaseString& str)
+{
+ return assign(str.m_chr);
+}
+
+inline BaseString&
+BaseString::assign(const Vector<BaseString> &vector,
+ const BaseString &separator) {
+ assign("");
+ return append(vector, separator);
+}
+
+#endif /* !__UTIL_BASESTRING_HPP_INCLUDED__ */
diff --git a/ndb/include/util/Bitmask.hpp b/ndb/include/util/Bitmask.hpp
new file mode 100644
index 00000000000..1f95d62bcb6
--- /dev/null
+++ b/ndb/include/util/Bitmask.hpp
@@ -0,0 +1,755 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_BITMASK_H
+#define NDB_BITMASK_H
+
+#include <ndb_types.h>
+#include <NdbConstant.hpp>
+
+#ifndef NDB_ASSERT
+#include <stdio.h>
+#include <stdlib.h>
+#define NDB_ASSERT(x, s) \
+ do { if (!(x)) { printf("%s\n", s); abort(); } } while (0)
+#endif
+
+/**
+ * Bitmask implementation. Size is given explicitly
+ * (as first argument). All methods are static.
+ */
+class BitmaskImpl {
+public:
+ STATIC_CONST( NotFound = (unsigned)-1 );
+
+ /**
+ * get - Check if bit n is set.
+ */
+ static bool get(unsigned size, const Uint32 data[], unsigned n);
+
+ /**
+ * set - Set bit n to given value (true/false).
+ */
+ static void set(unsigned size, Uint32 data[], unsigned n, bool value);
+
+ /**
+ * set - Set bit n.
+ */
+ static void set(unsigned size, Uint32 data[], unsigned n);
+
+ /**
+ * set - Set all bits.
+ */
+ static void set(unsigned size, Uint32 data[]);
+
+ /**
+ * assign - Set all bits in <em>dst</em> to corresponding in <em>src/<em>
+ */
+ static void assign(unsigned size, Uint32 dst[], const Uint32 src[]);
+
+ /**
+ * clear - Clear bit n.
+ */
+ static void clear(unsigned size, Uint32 data[], unsigned n);
+
+ /**
+ * clear - Clear all bits.
+ */
+ static void clear(unsigned size, Uint32 data[]);
+
+ /**
+ * isclear - Check if all bits are clear. This is faster
+ * than checking count() == 0.
+ */
+ static bool isclear(unsigned size, const Uint32 data[]);
+
+ /**
+ * count - Count number of set bits.
+ */
+ static unsigned count(unsigned size, const Uint32 data[]);
+
+ /**
+ * find - Find first set bit, starting at given position.
+ * Returns NotFound when not found.
+ */
+ static unsigned find(unsigned size, const Uint32 data[], unsigned n);
+
+ /**
+ * equal - Bitwise equal.
+ */
+ static bool equal(unsigned size, const Uint32 data[], const Uint32 data2[]);
+
+ /**
+ * bitOR - Bitwise (x | y) into first operand.
+ */
+ static void bitOR(unsigned size, Uint32 data[], const Uint32 data2[]);
+
+ /**
+ * bitAND - Bitwise (x & y) into first operand.
+ */
+ static void bitAND(unsigned size, Uint32 data[], const Uint32 data2[]);
+
+ /**
+ * bitANDC - Bitwise (x & ~y) into first operand.
+ */
+ static void bitANDC(unsigned size, Uint32 data[], const Uint32 data2[]);
+
+ /**
+ * bitXOR - Bitwise (x ^ y) into first operand.
+ */
+ static void bitXOR(unsigned size, Uint32 data[], const Uint32 data2[]);
+
+ /**
+ * contains - Check if all bits set in data2 are set in data
+ */
+ static bool contains(unsigned size, Uint32 data[], const Uint32 data2[]);
+
+ /**
+ * overlaps - Check if any bit set in data is set in data2
+ */
+ static bool overlaps(unsigned size, Uint32 data[], const Uint32 data2[]);
+
+ /**
+ * getField - Get bitfield at given position and length (max 32 bits)
+ */
+ static Uint32 getField(unsigned size, const Uint32 data[],
+ unsigned pos, unsigned len);
+
+ /**
+ * setField - Set bitfield at given position and length (max 32 bits)
+ */
+ static void setField(unsigned size, Uint32 data[],
+ unsigned pos, unsigned len, Uint32 val);
+
+ /**
+ * getText - Return as hex-digits (only for debug routines).
+ */
+ static void getText(unsigned size, const Uint32 data[], char* buf);
+};
+
+inline bool
+BitmaskImpl::get(unsigned size, const Uint32 data[], unsigned n)
+{
+ NDB_ASSERT(n < (size << 5), "bit get out of range");
+ return (data[n >> 5] & (1 << (n & 31))) != 0;
+}
+
+inline void
+BitmaskImpl::set(unsigned size, Uint32 data[], unsigned n, bool value)
+{
+ value ? set(size, data, n) : clear(size, data, n);
+}
+
+inline void
+BitmaskImpl::set(unsigned size, Uint32 data[], unsigned n)
+{
+ NDB_ASSERT(n < (size << 5), "bit set out of range");
+ data[n >> 5] |= (1 << (n & 31));
+}
+
+inline void
+BitmaskImpl::set(unsigned size, Uint32 data[])
+{
+ for (unsigned i = 0; i < size; i++) {
+ data[i] = ~0;
+ }
+}
+
+inline void
+BitmaskImpl::assign(unsigned size, Uint32 dst[], const Uint32 src[])
+{
+ for (unsigned i = 0; i < size; i++) {
+ dst[i] = src[i];
+ }
+}
+
+inline void
+BitmaskImpl::clear(unsigned size, Uint32 data[], unsigned n)
+{
+ NDB_ASSERT(n < (size << 5), "bit clear out of range");
+ data[n >> 5] &= ~(1 << (n & 31));
+}
+
+inline void
+BitmaskImpl::clear(unsigned size, Uint32 data[])
+{
+ for (unsigned i = 0; i < size; i++) {
+ data[i] = 0;
+ }
+}
+
+inline bool
+BitmaskImpl::isclear(unsigned size, const Uint32 data[])
+{
+ for (unsigned i = 0; i < size; i++) {
+ if (data[i] != 0)
+ return false;
+ }
+ return true;
+}
+
+inline unsigned
+BitmaskImpl::count(unsigned size, const Uint32 data[])
+{
+ unsigned cnt = 0;
+ for (unsigned i = 0; i < size; i++) {
+ Uint32 x = data[i];
+ while (x) {
+ x &= (x - 1);
+ cnt++;
+ }
+ }
+ return cnt;
+}
+
+inline unsigned
+BitmaskImpl::find(unsigned size, const Uint32 data[], unsigned n)
+{
+ while (n < (size << 5)) { // XXX make this smarter
+ if (get(size, data, n)) {
+ return n;
+ }
+ n++;
+ }
+ return NotFound;
+}
+
+inline bool
+BitmaskImpl::equal(unsigned size, const Uint32 data[], const Uint32 data2[])
+{
+ for (unsigned i = 0; i < size; i++) {
+ if (data[i] != data2[i])
+ return false;
+ }
+ return true;
+}
+
+inline void
+BitmaskImpl::bitOR(unsigned size, Uint32 data[], const Uint32 data2[])
+{
+ for (unsigned i = 0; i < size; i++) {
+ data[i] |= data2[i];
+ }
+}
+
+inline void
+BitmaskImpl::bitAND(unsigned size, Uint32 data[], const Uint32 data2[])
+{
+ for (unsigned i = 0; i < size; i++) {
+ data[i] &= data2[i];
+ }
+}
+
+inline void
+BitmaskImpl::bitANDC(unsigned size, Uint32 data[], const Uint32 data2[])
+{
+ for (unsigned i = 0; i < size; i++) {
+ data[i] &= ~data2[i];
+ }
+}
+
+inline void
+BitmaskImpl::bitXOR(unsigned size, Uint32 data[], const Uint32 data2[])
+{
+ for (unsigned i = 0; i < size; i++) {
+ data[i] ^= data2[i];
+ }
+}
+
+inline bool
+BitmaskImpl::contains(unsigned size, Uint32 data[], const Uint32 data2[])
+{
+ for (unsigned int i = 0; i < size; i++)
+ if ((data[i] & data2[i]) != data2[i])
+ return false;
+ return true;
+}
+
+inline bool
+BitmaskImpl::overlaps(unsigned size, Uint32 data[], const Uint32 data2[])
+{
+ for (unsigned int i = 0; i < size; i++)
+ if ((data[i] & data2[i]) != 0)
+ return true;
+ return false;
+}
+
+inline Uint32
+BitmaskImpl::getField(unsigned size, const Uint32 data[],
+ unsigned pos, unsigned len)
+{
+ Uint32 val = 0;
+ for (unsigned i = 0; i < len; i++)
+ val |= get(size, data, pos + i) << i;
+ return val;
+}
+
+inline void
+BitmaskImpl::setField(unsigned size, Uint32 data[],
+ unsigned pos, unsigned len, Uint32 val)
+{
+ for (unsigned i = 0; i < len; i++)
+ set(size, data, pos + i, val & (1 << i));
+}
+
+inline void
+BitmaskImpl::getText(unsigned size, const Uint32 data[], char* buf)
+{
+ const char* const hex = "0123456789abcdef";
+ for (int i = (size-1); i >= 0; i--) {
+ Uint32 x = data[i];
+ for (unsigned j = 0; j < 8; j++) {
+ buf[7-j] = hex[x & 0xf];
+ x >>= 4;
+ }
+ buf += 8;
+ }
+ *buf = 0;
+}
+
+/**
+ * Bitmasks. The size is number of 32-bit words (Uint32).
+ * Unused bits in the last word must be zero.
+ *
+ * XXX replace size by length in bits
+ */
+template <unsigned size>
+class Bitmask {
+public:
+ /**
+ * POD data representation
+ */
+ struct Data {
+ Uint32 data[size];
+
+ Data & operator=(const Bitmask<size> & src) {
+ src.assign(size, data);
+ return *this;
+ }
+ };
+private:
+
+ Data rep;
+public:
+ STATIC_CONST( Size = size );
+ STATIC_CONST( NotFound = BitmaskImpl::NotFound );
+ STATIC_CONST( TextLength = size * 8 );
+
+ /**
+ * assign - Set all bits in <em>dst</em> to corresponding in <em>src/<em>
+ */
+ void assign(const Bitmask<size>::Data & src);
+
+ /**
+ * assign - Set all bits in <em>dst</em> to corresponding in <em>src/<em>
+ */
+ static void assign(Uint32 dst[], const Uint32 src[]);
+ void assign(const Bitmask<size> & src);
+
+ /**
+ * assign <em>dst</em> of size <em>sz</em> to <em>this</em>
+ */
+ void assign(unsigned sz, Uint32 dst[]) const;
+
+ /**
+ * assign <em>this</em> according to <em>src/em>
+ */
+ void assign(unsigned sz, const Uint32 src[]);
+
+ /**
+ * get - Check if bit n is set.
+ */
+ static bool get(const Uint32 data[], unsigned n);
+ bool get(unsigned n) const;
+
+ /**
+ * set - Set bit n to given value (true/false).
+ */
+ static void set(Uint32 data[], unsigned n, bool value);
+ void set(unsigned n, bool value);
+
+ /**
+ * set - Set bit n.
+ */
+ static void set(Uint32 data[], unsigned n);
+ void set(unsigned n);
+
+ /**
+ * set - set all bits.
+ */
+ static void set(Uint32 data[]);
+ void set();
+
+ /**
+ * clear - Clear bit n.
+ */
+ static void clear(Uint32 data[], unsigned n);
+ void clear(unsigned n);
+
+ /**
+ * clear - Clear all bits.
+ */
+ static void clear(Uint32 data[]);
+ void clear();
+
+ /**
+ * isclear - Check if all bits are clear. This is faster
+ * than checking count() == 0.
+ */
+ static bool isclear(const Uint32 data[]);
+ bool isclear() const;
+
+ /**
+ * count - Count number of set bits.
+ */
+ static unsigned count(const Uint32 data[]);
+ unsigned count() const;
+
+ /**
+ * find - Find first set bit, starting at given position.
+ * Returns NotFound when not found.
+ */
+ static unsigned find(const Uint32 data[], unsigned n);
+ unsigned find(unsigned n) const;
+
+ /**
+ * equal - Bitwise equal.
+ */
+ static bool equal(const Uint32 data[], const Uint32 data2[]);
+ bool equal(const Bitmask<size>& mask2) const;
+
+ /**
+ * bitOR - Bitwise (x | y) into first operand.
+ */
+ static void bitOR(Uint32 data[], const Uint32 data2[]);
+ Bitmask<size>& bitOR(const Bitmask<size>& mask2);
+
+ /**
+ * bitAND - Bitwise (x & y) into first operand.
+ */
+ static void bitAND(Uint32 data[], const Uint32 data2[]);
+ Bitmask<size>& bitAND(const Bitmask<size>& mask2);
+
+ /**
+ * bitANDC - Bitwise (x & ~y) into first operand.
+ */
+ static void bitANDC(Uint32 data[], const Uint32 data2[]);
+ Bitmask<size>& bitANDC(const Bitmask<size>& mask2);
+
+ /**
+ * bitXOR - Bitwise (x ^ y) into first operand.
+ */
+ static void bitXOR(Uint32 data[], const Uint32 data2[]);
+ Bitmask<size>& bitXOR(const Bitmask<size>& mask2);
+
+ /**
+ * contains - Check if all bits set in data2 (that) are also set in data (this)
+ */
+ static bool contains(Uint32 data[], const Uint32 data2[]);
+ bool contains(Bitmask<size> that);
+
+ /**
+ * overlaps - Check if any bit set in this Bitmask (data) is also set in that (data2)
+ */
+ static bool overlaps(Uint32 data[], const Uint32 data2[]);
+ bool overlaps(Bitmask<size> that);
+
+ /**
+ * getText - Return as hex-digits (only for debug routines).
+ */
+ static void getText(const Uint32 data[], char* buf);
+ char* getText(char* buf) const;
+};
+
+template <unsigned size>
+inline void
+Bitmask<size>::assign(Uint32 dst[], const Uint32 src[])
+{
+ BitmaskImpl::assign(size, dst, src);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::assign(const Bitmask<size>::Data & src)
+{
+ assign(rep.data, src.data);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::assign(const Bitmask<size> & src)
+{
+ assign(rep.data, src);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::assign(unsigned sz, Uint32 dst[]) const
+{
+ BitmaskImpl::assign(sz, dst, rep.data);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::assign(unsigned sz, const Uint32 src[])
+{
+ BitmaskImpl::assign(sz, rep.data, src);
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::get(const Uint32 data[], unsigned n)
+{
+ return BitmaskImpl::get(size, data, n);
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::get(unsigned n) const
+{
+ return get(rep.data, n);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::set(Uint32 data[], unsigned n, bool value)
+{
+ BitmaskImpl::set(size, data, n, value);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::set(unsigned n, bool value)
+{
+ set(rep.data, n, value);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::set(Uint32 data[], unsigned n)
+{
+ BitmaskImpl::set(size, data, n);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::set(unsigned n)
+{
+ set(rep.data, n);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::set(Uint32 data[])
+{
+ BitmaskImpl::set(size, data);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::set()
+{
+ set(rep.data);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::clear(Uint32 data[], unsigned n)
+{
+ BitmaskImpl::clear(size, data, n);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::clear(unsigned n)
+{
+ clear(rep.data, n);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::clear(Uint32 data[])
+{
+ BitmaskImpl::clear(size, data);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::clear()
+{
+ clear(rep.data);
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::isclear(const Uint32 data[])
+{
+ return BitmaskImpl::isclear(size, data);
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::isclear() const
+{
+ return isclear(rep.data);
+}
+
+template <unsigned size>
+unsigned
+Bitmask<size>::count(const Uint32 data[])
+{
+ return BitmaskImpl::count(size, data);
+}
+
+template <unsigned size>
+inline unsigned
+Bitmask<size>::count() const
+{
+ return count(rep.data);
+}
+
+template <unsigned size>
+unsigned
+Bitmask<size>::find(const Uint32 data[], unsigned n)
+{
+ return BitmaskImpl::find(size, data, n);
+}
+
+template <unsigned size>
+inline unsigned
+Bitmask<size>::find(unsigned n) const
+{
+ return find(rep.data, n);
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::equal(const Uint32 data[], const Uint32 data2[])
+{
+ return BitmaskImpl::equal(size, data, data2);
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::equal(const Bitmask<size>& mask2) const
+{
+ return equal(rep.data, mask2.rep.data);
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::bitOR(Uint32 data[], const Uint32 data2[])
+{
+ BitmaskImpl::bitOR(size,data, data2);
+}
+
+template <unsigned size>
+inline Bitmask<size>&
+Bitmask<size>::bitOR(const Bitmask<size>& mask2)
+{
+ bitOR(rep.data, mask2.rep.data);
+ return *this;
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::bitAND(Uint32 data[], const Uint32 data2[])
+{
+ BitmaskImpl::bitAND(size,data, data2);
+}
+
+template <unsigned size>
+inline Bitmask<size>&
+Bitmask<size>::bitAND(const Bitmask<size>& mask2)
+{
+ bitAND(rep.data, mask2.rep.data);
+ return *this;
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::bitANDC(Uint32 data[], const Uint32 data2[])
+{
+ BitmaskImpl::bitANDC(size,data, data2);
+}
+
+template <unsigned size>
+inline Bitmask<size>&
+Bitmask<size>::bitANDC(const Bitmask<size>& mask2)
+{
+ bitANDC(rep.data, mask2.rep.data);
+ return *this;
+}
+
+template <unsigned size>
+inline void
+Bitmask<size>::bitXOR(Uint32 data[], const Uint32 data2[])
+{
+ BitmaskImpl::bitXOR(size,data, data2);
+}
+
+template <unsigned size>
+inline Bitmask<size>&
+Bitmask<size>::bitXOR(const Bitmask<size>& mask2)
+{
+ bitXOR(rep.data, mask2.rep.data);
+ return *this;
+}
+
+template <unsigned size>
+void
+Bitmask<size>::getText(const Uint32 data[], char* buf)
+{
+ BitmaskImpl::getText(size, data, buf);
+}
+
+template <unsigned size>
+inline char *
+Bitmask<size>::getText(char* buf) const
+{
+ getText(rep.data, buf);
+ return buf;
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::contains(Uint32 data[], const Uint32 data2[])
+{
+ return BitmaskImpl::contains(size, data, data2);
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::contains(Bitmask<size> that)
+{
+ return contains(this->rep.data, that.rep.data);
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::overlaps(Uint32 data[], const Uint32 data2[])
+{
+ return BitmaskImpl::overlaps(size, data, data2);
+}
+
+template <unsigned size>
+inline bool
+Bitmask<size>::overlaps(Bitmask<size> that)
+{
+ return overlaps(this->rep.data, that.rep.data);
+}
+
+#endif
diff --git a/ndb/include/util/File.hpp b/ndb/include/util/File.hpp
new file mode 100644
index 00000000000..fe3d2642b18
--- /dev/null
+++ b/ndb/include/util/File.hpp
@@ -0,0 +1,206 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FILE_H
+#define FILE_H
+
+#include <NdbStdio.h>
+#include <NdbConstant.hpp>
+
+/**
+ * This class provides a file abstraction . It has operations
+ * to create, read, write and delete a file.
+ *
+ * @version #@ $Id: File.hpp,v 1.5 2002/04/26 13:15:38 ejonore Exp $
+ */
+class File
+{
+public:
+ /**
+ * Returns true if the file exist.
+ *
+ * @param aFileName a filename to check.
+ * @return true if the file exists.
+ */
+ static bool exists(const char* aFileName);
+
+ /**
+ * Returns the size of a file.
+ *
+ * @param f a pointer to a FILE descriptor.
+ * @return the size of the file.
+ */
+ static long size(FILE* f);
+
+ /**
+ * Renames a file.
+ *
+ * @param currFileName the current name of the file.
+ * @param newFileName the new name of the file.
+ * @return true if successful.
+ */
+ static bool rename(const char* currFileName, const char* newFileName);
+
+ /**
+ * Removes/deletes a file.
+ *
+ * @param aFileName the file to remove.
+ * @return true if successful.
+ */
+ static bool remove(const char* aFileName);
+
+ /**
+ * Default constructor.
+ */
+ File();
+
+ /**
+ * Creates a new File with the specified filename and file mode.
+ * The real file itself will not be created unless open() is called!
+ *
+ * To see the available file modes use 'man 3 fopen'.
+ *
+ * @param aFileName a filename.
+ * @param mode the mode which the file should be opened/created with, default "r".
+ */
+ File(const char* aFileName, const char* mode = "r");
+
+ /**
+ * Destructor.
+ */
+ ~File();
+
+ /**
+ * Opens/creates the file. If open() fails then 'errno' and perror()
+ * should be used to determine the exact failure cause.
+ *
+ * @return true if successful. Check errno if unsuccessful.
+ */
+ bool open();
+
+ /**
+ * Opens/creates the file with the specified name and mode.
+ * If open() fails then 'errno' and perror() should be used to
+ * determine the exact failure cause.
+ *
+ * @param aFileName the file to open.
+ * @param mode the file mode to use.
+ * @return true if successful. Check errno if unsuccessful.
+ */
+ bool open(const char* aFileName, const char* mode);
+
+ /**
+ * Removes the file.
+ *
+ * @return true if successful.
+ */
+ bool remove();
+
+ /**
+ * Closes the file, i.e., the FILE descriptor is closed.
+ */
+ bool close();
+
+ /**
+ * Read from the file. See fread() for more info.
+ *
+ * @param buf the buffer to read into.
+ * @param itemSize the size of each item.
+ * @param nitems read max n number of items.
+ * @return 0 if successful.
+ */
+ int read(void* buf, size_t itemSize, size_t nitems) const;
+
+ /**
+ * Read char from the file. See fread() for more info.
+ *
+ * @param buf the buffer to read into.
+ * @param start the start index of the buf.
+ * @param length the length of the buffer.
+ * @return 0 if successful.
+ */
+ int readChar(char* buf, long start, long length) const;
+
+ /**
+ * Read chars from the file. See fread() for more info.
+ *
+ * @param buf the buffer to read into.
+ * @return 0 if successful.
+ */
+ int readChar(char* buf);
+
+ /**
+ * Write to file. See fwrite() for more info.
+ *
+ * @param buf the buffer to read from.
+ * @param itemSize the size of each item.
+ * @param nitems write max n number of items.
+ * @return 0 if successful.
+ */
+ int write(const void* buf, size_t itemSize, size_t nitems);
+
+ /**
+ * Write chars to file. See fwrite() for more info.
+ *
+ * @param buf the buffer to read from.
+ * @param start the start index of the buf.
+ * @param length the length of the buffer.
+ * @return 0 if successful.
+ */
+ int writeChar(const char* buf, long start, long length);
+
+ /**
+ * Write chars to file. See fwrite() for more info.
+ *
+ * @param buf the buffer to read from.
+ * @return 0 if successful.
+ */
+ int writeChar(const char* buf);
+
+ /**
+ * Returns the file size.
+ *
+ * @return the file size.
+ */
+ long size() const;
+
+ /**
+ * Returns the filename.
+ *
+ * @return the filename.
+ */
+ const char* getName() const;
+
+ /**
+ * Flush the buffer.
+ *
+ * @return 0 if successful.
+ */
+ int flush() const;
+
+private:
+ STATIC_CONST( MAX_FILE_NAME_SIZE = 128 );
+
+ FILE* m_file;
+ char m_fileName[MAX_FILE_NAME_SIZE];
+ const char* m_fileMode;
+ /* Prohibit */
+ File(const File& aCopy);
+ File operator = (const File&);
+ bool operator == (const File&);
+};
+#endif
+
diff --git a/ndb/include/util/InputStream.hpp b/ndb/include/util/InputStream.hpp
new file mode 100644
index 00000000000..6b4cf262db4
--- /dev/null
+++ b/ndb/include/util/InputStream.hpp
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef INPUT_STREAM_HPP
+#define INPUT_STREAM_HPP
+
+#include <stdio.h>
+#include <NdbTCP.h>
+
+/**
+ * Input stream
+ */
+class InputStream {
+public:
+ virtual char* gets(char * buf, int bufLen) = 0;
+};
+
+class FileInputStream : public InputStream {
+ FILE * f;
+public:
+ FileInputStream(FILE * file = stdin);
+ char* gets(char * buf, int bufLen);
+};
+
+extern FileInputStream Stdin;
+
+class SocketInputStream : public InputStream {
+ NDB_SOCKET_TYPE m_socket;
+ unsigned m_timeout;
+public:
+ SocketInputStream(NDB_SOCKET_TYPE socket, unsigned readTimeout = 1000);
+ char* gets(char * buf, int bufLen);
+};
+
+#endif
diff --git a/ndb/include/util/NdbAutoPtr.hpp b/ndb/include/util/NdbAutoPtr.hpp
new file mode 100644
index 00000000000..2078714d98d
--- /dev/null
+++ b/ndb/include/util/NdbAutoPtr.hpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __NDB_AUTO_PTR_HPP
+#define __NDB_AUTO_PTR_HPP
+
+#include <stdlib.h>
+
+template<typename T>
+class NdbAutoPtr {
+ T * m_obj;
+public:
+ NdbAutoPtr(T * obj = 0){ m_obj = obj;}
+ void reset(T * obj = 0) { if (m_obj) free(m_obj); m_obj = obj; }
+ ~NdbAutoPtr() { if (m_obj) free(m_obj);}
+};
+
+template<typename T>
+class NdbAutoObjPtr {
+ T * m_obj;
+public:
+ NdbAutoObjPtr(T * obj = 0){ m_obj = obj;}
+ void reset(T * obj = 0) { if (m_obj) delete m_obj; m_obj = obj; }
+ ~NdbAutoObjPtr() { if (m_obj) delete m_obj;}
+};
+
+template<typename T>
+class NdbAutoObjArrayPtr {
+ T * m_obj;
+public:
+ NdbAutoObjArrayPtr(T * obj = 0){ m_obj = obj;}
+ void reset(T * obj = 0) { if (m_obj) delete[] m_obj; m_obj = obj; }
+ ~NdbAutoObjArrayPtr() { if (m_obj) delete[] m_obj;}
+};
+
+#endif
diff --git a/ndb/include/util/NdbOut.hpp b/ndb/include/util/NdbOut.hpp
new file mode 100644
index 00000000000..d85d5cc6305
--- /dev/null
+++ b/ndb/include/util/NdbOut.hpp
@@ -0,0 +1,132 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBOUT_H
+#define NDBOUT_H
+
+#ifdef __cplusplus
+
+#include <ndb_types.h>
+#include <util/BaseString.hpp>
+
+/**
+ * Class used for outputting logging messages to screen.
+ * Since the output capabilities are different on different platforms
+ * this middle layer class should be used for all output messages
+ */
+
+/*
+ Example usage:
+
+ #include "NdbOut.hpp"
+
+ / * Use ndbout as you would use cout:
+
+ ndbout << "Hello World! "<< 1 << " Hello again"
+ << 67 << anIntegerVar << "Hup << endl;
+
+
+ / * Use ndbout_c as you would use printf:
+
+ ndbout_c("Hello World %d\n", 1);
+*/
+
+class NdbOut;
+class OutputStream;
+class NullOutputStream;
+
+/* Declare a static variable of NdbOut as ndbout */
+extern NdbOut ndbout;
+
+class NdbOut
+{
+public:
+ NdbOut& operator<<(NdbOut& (* _f)(NdbOut&));
+ NdbOut& operator<<(Int8);
+ NdbOut& operator<<(Uint8);
+ NdbOut& operator<<(Int16);
+ NdbOut& operator<<(Uint16);
+ NdbOut& operator<<(Int32);
+ NdbOut& operator<<(Uint32);
+ NdbOut& operator<<(Int64);
+ NdbOut& operator<<(Uint64);
+ NdbOut& operator<<(long unsigned int);
+ NdbOut& operator<<(const char*);
+ NdbOut& operator<<(const unsigned char*);
+ NdbOut& operator<<(BaseString &);
+ NdbOut& operator<<(const void*);
+ NdbOut& operator<<(float);
+ NdbOut& operator<<(double);
+ NdbOut& endline(void);
+ NdbOut& flushline(void);
+ NdbOut& setHexFormat(int _format);
+
+ NdbOut(OutputStream &);
+ virtual ~NdbOut();
+
+ void print(const char * fmt, ...);
+ void println(const char * fmt, ...);
+
+ OutputStream * m_out;
+private:
+ int isHex;
+};
+
+inline NdbOut& NdbOut::operator<<(NdbOut& (* _f)(NdbOut&)) {
+ (* _f)(*this);
+ return * this;
+}
+
+inline NdbOut& endl(NdbOut& _NdbOut) {
+ return _NdbOut.endline();
+}
+
+inline NdbOut& flush(NdbOut& _NdbOut) {
+ return _NdbOut.flushline();
+}
+
+inline NdbOut& hex(NdbOut& _NdbOut) {
+ return _NdbOut.setHexFormat(1);
+}
+
+inline NdbOut& dec(NdbOut& _NdbOut) {
+ return _NdbOut.setHexFormat(0);
+}
+extern "C"
+void ndbout_c(const char * fmt, ...);
+
+class FilteredNdbOut : public NdbOut {
+public:
+ FilteredNdbOut(OutputStream &, int threshold = 0, int level = 0);
+ virtual ~FilteredNdbOut();
+
+ void setLevel(int i);
+ void setThreshold(int i);
+
+ int getLevel() const;
+ int getThreshold() const;
+
+private:
+ int m_threshold, m_level;
+ OutputStream * m_org;
+ NullOutputStream * m_null;
+};
+
+#else
+void ndbout_c(const char * fmt, ...);
+#endif
+
+#endif
diff --git a/ndb/include/util/NdbSqlUtil.hpp b/ndb/include/util/NdbSqlUtil.hpp
new file mode 100644
index 00000000000..bb573eea17b
--- /dev/null
+++ b/ndb/include/util/NdbSqlUtil.hpp
@@ -0,0 +1,357 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_SQL_UTIL_HPP
+#define NDB_SQL_UTIL_HPP
+
+#include <string.h>
+#include <ndb_types.h>
+
+class NdbSqlUtil {
+public:
+ /**
+ * Compare strings, optionally with padded semantics. Returns
+ * negative (less), zero (equal), or positive (greater).
+ */
+ static int char_compare(const char* s1, unsigned n1,
+ const char* s2, unsigned n2, bool padded);
+
+ /**
+ * Like operator, optionally with padded semantics. Returns true or
+ * false.
+ */
+ static bool char_like(const char* s1, unsigned n1,
+ const char* s2, unsigned n2, bool padded);
+
+ /**
+ * Compare kernel attribute values. Returns -1, 0, +1 for less,
+ * equal, greater, respectively. Parameters are pointers to values,
+ * full attribute size in words, and size of available data in words.
+ * There are 2 special return values to check first. All values fit
+ * into a signed char.
+ */
+ typedef int Cmp(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size);
+
+ enum CmpResult {
+ CmpLess = -1,
+ CmpEqual = 0,
+ CmpGreater = 1,
+ CmpUnknown = 126, // insufficient partial data
+ CmpError = 127 // bad data format or unimplemented comparison
+ };
+
+ /**
+ * Kernel data types. Must match m_typeList in NdbSqlUtil.cpp.
+ */
+ struct Type {
+ enum Enum {
+ Undefined = 0, // Undefined
+ Tinyint, // 8 bit
+ Tinyunsigned, // 8 bit
+ Smallint, // 16 bit
+ Smallunsigned, // 16 bit
+ Mediumint, // 24 bit
+ Mediumunsigned, // 24 bit
+ Int, // 32 bit
+ Unsigned, // 32 bit
+ Bigint, // 64 bit
+ Bigunsigned, // 64 Bit
+ Float, // 32-bit float
+ Double, // 64-bit float
+ Decimal, // Precision, Scale
+ Char, // Len
+ Varchar, // Max len
+ Binary, // Len
+ Varbinary, // Max len
+ Datetime, // Precision down to 1 sec (size 8 bytes)
+ Timespec // Precision down to 1 nsec (size 12 bytes)
+ };
+ Enum m_typeId;
+ Cmp* m_cmp; // set to NULL if cmp not implemented
+ };
+
+ /**
+ * Get type by id. Can return the Undefined type.
+ */
+ static const Type& type(Uint32 typeId);
+
+ /**
+ * Inline comparison method. Most or all real methods Type::m_cmp are
+ * implemented via this (trusting dead code elimination).
+ */
+ static int cmp(Uint32 typeId, const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size);
+
+private:
+ /**
+ * List of all types. Must match Type::Enum.
+ */
+ static const Type m_typeList[];
+ /**
+ * Comparison methods.
+ */
+ static Cmp cmpTinyint;
+ static Cmp cmpTinyunsigned;
+ static Cmp cmpSmallint;
+ static Cmp cmpSmallunsigned;
+ static Cmp cmpMediumint;
+ static Cmp cmpMediumunsigned;
+ static Cmp cmpInt;
+ static Cmp cmpUnsigned;
+ static Cmp cmpBigint;
+ static Cmp cmpBigunsigned;
+ static Cmp cmpFloat;
+ static Cmp cmpDouble;
+ static Cmp cmpDecimal;
+ static Cmp cmpChar;
+ static Cmp cmpVarchar;
+ static Cmp cmpBinary;
+ static Cmp cmpVarbinary;
+ static Cmp cmpDatetime;
+ static Cmp cmpTimespec;
+};
+
+inline int
+NdbSqlUtil::cmp(Uint32 typeId, const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ if (size > full)
+ return CmpError;
+ switch ((Type::Enum)typeId) {
+ case Type::Undefined:
+ break;
+ case Type::Tinyint:
+ {
+ if (size >= 1) {
+ union { Uint32 p[1]; Int8 v; } u1, u2;
+ u1.p[0] = p1[0];
+ u2.p[0] = p2[0];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Tinyunsigned:
+ {
+ if (size >= 1) {
+ union { Uint32 p[1]; Uint8 v; } u1, u2;
+ u1.p[0] = p1[0];
+ u2.p[0] = p2[0];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Smallint:
+ {
+ if (size >= 1) {
+ union { Uint32 p[1]; Int16 v; } u1, u2;
+ u1.p[0] = p1[0];
+ u2.p[0] = p2[0];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Smallunsigned:
+ {
+ if (size >= 1) {
+ union { Uint32 p[1]; Uint16 v; } u1, u2;
+ u1.p[0] = p1[0];
+ u2.p[0] = p2[0];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Mediumint: // XXX fix these
+ break;
+ case Type::Mediumunsigned:
+ break;
+ case Type::Int:
+ {
+ if (size >= 1) {
+ union { Uint32 p[1]; Int32 v; } u1, u2;
+ u1.p[0] = p1[0];
+ u2.p[0] = p2[0];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Unsigned:
+ {
+ if (size >= 1) {
+ union { Uint32 p[1]; Uint32 v; } u1, u2;
+ u1.v = p1[0];
+ u2.v = p2[0];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Bigint:
+ {
+ if (size >= 2) {
+ union { Uint32 p[2]; Int64 v; } u1, u2;
+ u1.p[0] = p1[0];
+ u1.p[1] = p1[1];
+ u2.p[0] = p2[0];
+ u2.p[1] = p2[1];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Bigunsigned:
+ {
+ if (size >= 2) {
+ union { Uint32 p[2]; Uint64 v; } u1, u2;
+ u1.p[0] = p1[0];
+ u1.p[1] = p1[1];
+ u2.p[0] = p2[0];
+ u2.p[1] = p2[1];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Float:
+ {
+ if (size >= 1) {
+ union { Uint32 p[1]; float v; } u1, u2;
+ u1.p[0] = p1[0];
+ u2.p[0] = p2[0];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Double:
+ {
+ if (size >= 2) {
+ union { Uint32 p[2]; double v; } u1, u2;
+ u1.p[0] = p1[0];
+ u1.p[1] = p1[1];
+ u2.p[0] = p2[0];
+ u2.p[1] = p2[1];
+ if (u1.v < u2.v)
+ return -1;
+ if (u1.v > u2.v)
+ return +1;
+ return 0;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Decimal:
+ break;
+ case Type::Char:
+ {
+ /*
+ * Char is blank-padded to length and null-padded to word size.
+ * There is no terminator so we must compare the full values.
+ */
+ union { const Uint32* p; const char* v; } u1, u2;
+ u1.p = p1;
+ u2.p = p2;
+ int k = memcmp(u1.v, u2.v, size << 2);
+ return k < 0 ? -1 : k > 0 ? +1 : full == size ? 0 : CmpUnknown;
+ }
+ break;
+ case Type::Varchar:
+ {
+ /*
+ * Varchar is not allowed to contain a null byte and the stored
+ * value is null-padded. Therefore comparison does not need to
+ * use the length.
+ */
+ if (size >= 1) {
+ union { const Uint32* p; const char* v; } u1, u2;
+ u1.p = p1;
+ u2.p = p2;
+ // length in first 2 bytes
+ int k = strncmp(u1.v + 2, u2.v + 2, (size << 2) - 2);
+ return k < 0 ? -1 : k > 0 ? +1 : full == size ? 0 : CmpUnknown;
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Binary: // XXX fix these
+ break;
+ case Type::Varbinary:
+ break;
+ case Type::Datetime:
+ {
+ /*
+ * Datetime is CC YY MM DD hh mm ss \0
+ */
+ if (size >= 1) {
+ union { const Uint32* p; const char* v; } u1, u2;
+ u1.p = p1;
+ u2.p = p2;
+ // skip format check
+ int k = strncmp(u1.v, u2.v, 4);
+ if (k != 0)
+ return k;
+ if (size >= 2) {
+ return strncmp(u1.v + 4, u2.v + 4, 4);
+ }
+ }
+ return CmpUnknown;
+ }
+ break;
+ case Type::Timespec: // XXX fix this
+ break;
+ }
+ return CmpError;
+}
+
+#endif
diff --git a/ndb/include/util/NdbString.h b/ndb/include/util/NdbString.h
new file mode 100644
index 00000000000..97646f813ac
--- /dev/null
+++ b/ndb/include/util/NdbString.h
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __NDBSTRING_H_INCLUDED__
+#define __NDBSTRING_H_INCLUDED__
+
+#include <sys/types.h>
+#include <string.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifndef HAVE_STRDUP
+extern char * strdup(const char *s);
+#endif
+
+#ifndef HAVE_STRLCPY
+extern size_t strlcpy (char *dst, const char *src, size_t dst_sz);
+#endif
+
+#ifndef HAVE_STRLCAT
+extern size_t strlcat (char *dst, const char *src, size_t dst_sz);
+#endif
+
+#ifndef HAVE_STRCASECMP
+extern int strcasecmp(const char *s1, const char *s2);
+extern int strncasecmp(const char *s1, const char *s2, size_t n);
+#endif
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* !__NDBSTRING_H_INCLUDED__ */
diff --git a/ndb/include/util/OutputStream.hpp b/ndb/include/util/OutputStream.hpp
new file mode 100644
index 00000000000..9d33ead7eb9
--- /dev/null
+++ b/ndb/include/util/OutputStream.hpp
@@ -0,0 +1,67 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef OUTPUT_STREAM_HPP
+#define OUTPUT_STREAM_HPP
+
+#include <stdio.h>
+#include <NdbTCP.h>
+
+/**
+ * Output stream
+ */
+class OutputStream {
+public:
+ virtual int print(const char * fmt, ...) = 0;
+ virtual int println(const char * fmt, ...) = 0;
+ virtual void flush() {};
+};
+
+class FileOutputStream : public OutputStream {
+ FILE * f;
+public:
+ FileOutputStream(FILE * file = stdout);
+
+ int print(const char * fmt, ...);
+ int println(const char * fmt, ...);
+ void flush() { fflush(f); }
+};
+
+class SocketOutputStream : public OutputStream {
+ NDB_SOCKET_TYPE m_socket;
+ unsigned m_timeout;
+public:
+ SocketOutputStream(NDB_SOCKET_TYPE socket, unsigned writeTimeout = 1000);
+
+ int print(const char * fmt, ...);
+ int println(const char * fmt, ...);
+};
+
+class SoftOseOutputStream : public OutputStream {
+public:
+ SoftOseOutputStream();
+
+ int print(const char * fmt, ...);
+ int println(const char * fmt, ...);
+};
+
+class NullOutputStream : public OutputStream {
+public:
+ int print(const char * /* unused */, ...) { return 1;}
+ int println(const char * /* unused */, ...) { return 1;}
+};
+
+#endif
diff --git a/ndb/include/util/Parser.hpp b/ndb/include/util/Parser.hpp
new file mode 100644
index 00000000000..9c2f02b6024
--- /dev/null
+++ b/ndb/include/util/Parser.hpp
@@ -0,0 +1,290 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CPCD_PARSER_HPP
+#define CPCD_PARSER_HPP
+
+#include "Vector.hpp"
+#include "Properties.hpp"
+#include "InputStream.hpp"
+#include "NdbOut.hpp"
+
+class ParserImpl;
+template<class T> class ParserRow;
+
+//#define PARSER_DEBUG
+#ifdef PARSER_DEBUG
+#define DEBUG(x) \
+ ndbout_c("%s:%d:%s", __FILE__, __LINE__, x);
+#else
+#define DEBUG(x)
+#endif
+
+/**
+ * A generic parser
+ */
+template<class T>
+class Parser {
+public:
+ /**
+ * Status for parser
+ */
+ enum ParserStatus {
+ Ok = 0,
+ Eof = 1,
+ NoLine = 2,
+ EmptyLine = 3,
+ UnknownCommand = 4,
+ UnknownArgument = 5,
+ TypeMismatch = 6,
+ InvalidArgumentFormat = 7,
+ UnknownArgumentType = 8,
+ CommandWithoutFunction = 9,
+ ArgumentGivenTwice = 10,
+ ExternalStop = 11,
+ MissingMandatoryArgument = 12
+ };
+
+ /**
+ * Context for parse
+ */
+ struct Context {
+ ParserStatus m_status;
+ const ParserRow<T> * m_currentCmd;
+ const ParserRow<T> * m_currentArg;
+ char * m_currentToken;
+ char m_tokenBuffer[512];
+
+ Vector<const ParserRow<T> *> m_aliasUsed;
+ };
+
+ /**
+ * Initialize parser
+ */
+ Parser(const ParserRow<T> rows[], class InputStream & in = Stdin,
+ bool breakOnCommand = false,
+ bool breakOnEmptyLine = true,
+ bool breakOnInvalidArg = false);
+ ~Parser();
+
+ /**
+ * Run parser
+ */
+ bool run(Context &, T &, volatile bool * stop = 0) const;
+
+ /**
+ * Parse only one entry and return Properties object representing
+ * the message
+ */
+ const Properties *parse(Context &, T &);
+
+ bool getBreakOnCommand() const;
+ void setBreakOnCommand(bool v);
+
+ bool getBreakOnEmptyLine() const;
+ void setBreakOnEmptyLine(bool v);
+
+ bool getBreakOnInvalidArg() const;
+ void setBreakOnInvalidArg(bool v);
+
+private:
+ ParserImpl * impl;
+};
+
+template<class T>
+struct ParserRow {
+public:
+ enum Type { Cmd, Arg, CmdAlias, ArgAlias };
+ enum ArgType { String, Int, Properties };
+ enum ArgRequired { Mandatory, Optional };
+ enum ArgMinMax { CheckMinMax, IgnoreMinMax };
+
+ const char * name;
+ const char * realName;
+ Type type;
+ ArgType argType;
+ ArgRequired argRequired;
+ ArgMinMax argMinMax;
+ int minVal;
+ int maxVal;
+ void (T::* function)(typename Parser<T>::Context & ctx,
+ const class Properties& args);
+ const char * description;
+ void *user_value;
+};
+
+/**
+ * The void* equivalent implementation
+ */
+class ParserImpl {
+ class Dummy {};
+ typedef ParserRow<Dummy> DummyRow;
+ typedef Parser<Dummy>::Context Context;
+ template<class T> friend class Parser;
+private:
+
+ ParserImpl(const DummyRow rows[], class InputStream & in,
+ bool b_cmd, bool b_empty, bool b_iarg);
+ ~ParserImpl();
+
+ bool run(Context *ctx, const class Properties **, volatile bool *) const ;
+
+ static const DummyRow* matchCommand(Context*, const char*, const DummyRow*);
+ static const DummyRow* matchArg(Context*, const char *, const DummyRow *);
+ static bool parseArg(Context*, char*, const DummyRow*, Properties*);
+ static bool checkMandatory(Context*, const Properties*);
+private:
+ const DummyRow * const m_rows;
+ class ParseInputStream & input;
+ bool m_breakOnEmpty;
+ bool m_breakOnCmd;
+ bool m_breakOnInvalidArg;
+};
+
+template<class T>
+inline
+Parser<T>::Parser(const ParserRow<T> rows[], class InputStream & in,
+ bool b_cmd, bool b_empty, bool b_iarg){
+ impl = new ParserImpl((ParserImpl::DummyRow *)rows, in,
+ b_cmd, b_empty, b_iarg);
+}
+
+template<class T>
+inline
+Parser<T>::~Parser(){
+}
+
+template<class T>
+inline
+bool
+Parser<T>::run(Context & ctx, T & t, volatile bool * stop) const {
+ const Properties * p;
+ DEBUG("Executing Parser<T>::run");
+ if(impl->run((ParserImpl::Context*)&ctx, &p, stop)){
+ const ParserRow<T> * cmd = ctx.m_currentCmd; // Cast to correct type
+ if(cmd == 0){
+ /**
+ * Should happen if run returns true
+ */
+ abort();
+ }
+
+ for(unsigned i = 0; i<ctx.m_aliasUsed.size(); i++){
+ const ParserRow<T> * alias = ctx.m_aliasUsed[i];
+ if(alias->function != 0){
+ /**
+ * Report alias usage with callback (if specified by user)
+ */
+ DEBUG("Alias usage with callback");
+ (t.* alias->function)(ctx, * p);
+ }
+ }
+
+ if(cmd->function == 0){
+ ctx.m_status = CommandWithoutFunction;
+ DEBUG("CommandWithoutFunction");
+ delete p;
+ return false;
+ }
+ (t.* cmd->function)(ctx, * p); // Call the function
+ delete p;
+ return true;
+ }
+ DEBUG("");
+ return false;
+}
+
+template<class T>
+inline
+const Properties *
+Parser<T>::parse(Context &ctx, T &t) {
+ const Properties * p;
+ volatile bool stop = false;
+ DEBUG("Executing Parser<T>::parse");
+
+ if(impl->run((ParserImpl::Context*)&ctx, &p, &stop)){
+ const ParserRow<T> * cmd = ctx.m_currentCmd; // Cast to correct type
+ if(cmd == 0){
+ /**
+ * Should happen if run returns true
+ */
+ abort();
+ }
+
+ for(unsigned i = 0; i<ctx.m_aliasUsed.size(); i++){
+ const ParserRow<T> * alias = ctx.m_aliasUsed[i];
+ if(alias->function != 0){
+ /**
+ * Report alias usage with callback (if specified by user)
+ */
+ DEBUG("Alias usage with callback");
+ (t.* alias->function)(ctx, * p);
+ }
+ }
+
+ if(cmd->function == 0){
+ DEBUG("CommandWithoutFunction");
+ ctx.m_status = CommandWithoutFunction;
+ return p;
+ }
+ return p;
+ }
+ DEBUG("");
+ return NULL;
+}
+
+template<class T>
+inline
+bool
+Parser<T>::getBreakOnCommand() const{
+ return impl->m_breakOnCmd;
+}
+
+template<class T>
+inline
+void
+Parser<T>::setBreakOnCommand(bool v){
+ impl->m_breakOnCmd = v;
+}
+
+template<class T>
+inline
+bool
+Parser<T>::getBreakOnEmptyLine() const{
+ return impl->m_breakOnEmpty;
+}
+template<class T>
+inline
+void
+Parser<T>::setBreakOnEmptyLine(bool v){
+ impl->m_breakOnEmpty = v;
+}
+
+template<class T>
+inline
+bool
+Parser<T>::getBreakOnInvalidArg() const{
+ return impl->m_breakOnInvalidArg;
+}
+
+template<class T>
+inline
+void
+Parser<T>::setBreakOnInvalidArg(bool v){
+ impl->m_breakOnInvalidArg;
+}
+
+#endif
diff --git a/ndb/include/util/Properties.hpp b/ndb/include/util/Properties.hpp
new file mode 100644
index 00000000000..dbdc5f2b480
--- /dev/null
+++ b/ndb/include/util/Properties.hpp
@@ -0,0 +1,246 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PROPERTIES_HPP
+#define PROPERTIES_HPP
+
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <ndb_types.h>
+#include <BaseString.hpp>
+#include <UtilBuffer.hpp>
+
+enum PropertiesType {
+ PropertiesType_Uint32,
+ PropertiesType_char,
+ PropertiesType_Properties
+};
+
+/**
+ * @struct Property
+ * @brief Stores one (name, value)-pair
+ *
+ * Value can be of type Properties, i.e. a Property may contain
+ * a Properties object.
+ */
+struct Property {
+ Property(const char* name, Uint32 val);
+ Property(const char* name, const char * value);
+ Property(const char* name, const class Properties * value);
+ ~Property();
+private:
+ friend class Properties;
+ struct PropertyImpl * impl;
+};
+
+/**
+ * @class Properties
+ * @brief Stores information in (name, value)-pairs
+ */
+class Properties {
+public:
+ static const char delimiter = ':';
+ static const char version[];
+
+ Properties();
+ Properties(const Properties &);
+ Properties(const Property *, int len);
+ virtual ~Properties();
+
+ /**
+ * Set/Get wheather names in the Properties should be compared
+ * w/o case.
+ * NOTE: The property is automatically applied to all propoerties put
+ * into this after a called to setCaseInsensitiveNames has been made
+ * But properties already in when calling setCaseInsensitiveNames will
+ * not be affected
+ */
+ void setCaseInsensitiveNames(bool value);
+ bool getCaseInsensitiveNames() const;
+
+ /**
+ * Insert an array of value(s)
+ */
+ void put(const Property *, int len);
+
+ bool put(const char * name, Uint32 value, bool replace = false);
+ bool put(const char * name, const char * value, bool replace = false);
+ bool put(const char * name, const Properties * value, bool replace = false);
+
+ /**
+ * Same as put above,
+ * except that _%d (where %d is a number) is added to the name
+ * Compare get(name, no)
+ */
+ bool put(const char *, Uint32 no, Uint32, bool replace = false);
+ bool put(const char *, Uint32 no, const char *, bool replace = false);
+ bool put(const char *, Uint32 no, const Properties *, bool replace = false);
+
+
+ bool getTypeOf(const char * name, PropertiesType * type) const;
+
+ /** @return true if Properties object contains name */
+ bool contains(const char * name) const;
+
+ bool get(const char * name, Uint32 * value) const;
+ bool get(const char * name, const char ** value) const;
+ bool get(const char * name, BaseString & value) const;
+ bool get(const char * name, const Properties ** value) const;
+
+ bool getCopy(const char * name, char ** value) const;
+ bool getCopy(const char * name, Properties ** value) const;
+
+ /**
+ * Same as get above
+ * except that _%d (where %d = no) is added to the name
+ */
+ bool getTypeOf(const char * name, Uint32 no, PropertiesType * type) const;
+ bool contains(const char * name, Uint32 no) const;
+
+ bool get(const char * name, Uint32 no, Uint32 * value) const;
+ bool get(const char * name, Uint32 no, const char ** value) const;
+ bool get(const char * name, Uint32 no, const Properties ** value) const;
+
+ bool getCopy(const char * name, Uint32 no, char ** value) const;
+ bool getCopy(const char * name, Uint32 no, Properties ** value) const;
+
+ void clear();
+
+ void remove(const char * name);
+
+ void print(FILE * file = stdout, const char * prefix = 0) const;
+ /**
+ * Iterator over names
+ */
+ class Iterator {
+ public:
+ Iterator(const Properties* prop);
+
+ const char* first();
+ const char* next();
+ private:
+ const Properties* m_prop;
+ Uint32 m_iterator;
+ };
+ friend class Properties::Iterator;
+
+ Uint32 getPackedSize() const;
+ bool pack(Uint32 * buf) const;
+ bool pack(UtilBuffer &buf) const;
+ bool unpack(const Uint32 * buf, Uint32 bufLen);
+ bool unpack(UtilBuffer &buf);
+
+ Uint32 getPropertiesErrno() const { return propErrno; }
+ Uint32 getOSErrno() const { return osErrno; }
+private:
+ Uint32 propErrno;
+ Uint32 osErrno;
+
+ friend class PropertiesImpl;
+ class PropertiesImpl * impl;
+ class Properties * parent;
+
+ void setErrno(Uint32 pErr, Uint32 osErr = 0) const ;
+};
+
+/**
+ * Error code for properties
+ */
+
+/**
+ * No error
+ */
+extern const Uint32 E_PROPERTIES_OK;
+
+/**
+ * Invalid name in put, names can not contain Properties::delimiter
+ */
+extern const Uint32 E_PROPERTIES_INVALID_NAME;
+
+/**
+ * Element did not exist when using get
+ */
+extern const Uint32 E_PROPERTIES_NO_SUCH_ELEMENT;
+
+/**
+ * Element had wrong type when using get
+ */
+extern const Uint32 E_PROPERTIES_INVALID_TYPE;
+
+/**
+ * Element already existed when using put, and replace was not specified
+ */
+extern const Uint32 E_PROPERTIES_ELEMENT_ALREADY_EXISTS;
+
+/**
+ * Invalid version on properties file you are trying to read
+ */
+extern const Uint32 E_PROPERTIES_INVALID_VERSION_WHILE_UNPACKING;
+
+/**
+ * When unpacking an buffer
+ * found that buffer is to short
+ *
+ * Probably an invlaid buffer
+ */
+extern const Uint32 E_PROPERTIES_INVALID_BUFFER_TO_SHORT;
+
+/**
+ * Error when packing, can not allocate working buffer
+ *
+ * Note: OS error is set
+ */
+extern const Uint32 E_PROPERTIES_ERROR_MALLOC_WHILE_PACKING;
+
+/**
+ * Error when unpacking, can not allocate working buffer
+ *
+ * Note: OS error is set
+ */
+extern const Uint32 E_PROPERTIES_ERROR_MALLOC_WHILE_UNPACKING;
+
+/**
+ * Error when unpacking, invalid checksum
+ *
+ */
+extern const Uint32 E_PROPERTIES_INVALID_CHECKSUM;
+
+/**
+ * Error when unpacking
+ * No of items > 0 while size of buffer (left) <= 0
+ */
+extern const Uint32 E_PROPERTIES_BUFFER_TO_SMALL_WHILE_UNPACKING;
+
+inline bool
+Properties::unpack(UtilBuffer &buf) {
+ return unpack((const Uint32 *)buf.get_data(), buf.length());
+}
+
+inline bool
+Properties::pack(UtilBuffer &buf) const {
+ Uint32 size = getPackedSize();
+ char *tmp_buf = new char[size];
+ bool ret = pack((Uint32 *)tmp_buf);
+ if(ret == false)
+ return false;
+ buf.append(tmp_buf, size);
+ return true;
+}
+
+
+
+#endif
diff --git a/ndb/include/util/SimpleProperties.hpp b/ndb/include/util/SimpleProperties.hpp
new file mode 100644
index 00000000000..37e28ea91d8
--- /dev/null
+++ b/ndb/include/util/SimpleProperties.hpp
@@ -0,0 +1,290 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SIMPLE_PROPERTIES_HPP
+#define SIMPLE_PROPERTIES_HPP
+
+#include <ndb_types.h>
+#include <stddef.h> // offsetof
+#include <NdbOut.hpp>
+
+/**
+ * @class SimpleProperties
+ * @brief Key-value-pair container. Actully a list of named elements.
+ *
+ * SimpleProperties:
+ * - The keys are Uint16
+ * - The values are either Uint32 or null terminated c-strings
+ *
+ * @note Keys may be repeated.
+ *
+ * Examples of things that can be stored in a SimpleProperties object:
+ * - Lists like: ((1, "foo"), (2, "bar"), (3, 32), (2, "baz"))
+ */
+class SimpleProperties {
+public:
+ /**
+ * Value types
+ */
+ enum ValueType {
+ Uint32Value = 0,
+ StringValue = 1,
+ BinaryValue = 2,
+ InvalidValue = 3
+ };
+
+ /**
+ * Struct for defining mapping to be used with unpack
+ */
+ struct SP2StructMapping {
+ Uint16 Key;
+ Uint32 Offset;
+ SimpleProperties::ValueType Type;
+ Uint32 minValue;
+ Uint32 maxValue;
+ Uint32 Length_Offset; // Offset used for looking up length of
+ // data if Type = BinaryValue
+ };
+
+ /**
+ * UnpackStatus - Value returned from unpack
+ */
+ enum UnpackStatus {
+ Eof = 0, // Success, end of SimpleProperties object reached
+ Break = 1, // Success
+ TypeMismatch = 2,
+ ValueTooLow = 3,
+ ValueTooHigh = 4,
+ UnknownKey = 5,
+ OutOfMemory = 6 // Only used when packing
+ };
+
+ /**
+ * Unpack
+ */
+ class Reader;
+ static UnpackStatus unpack(class Reader & it,
+ void * dst,
+ const SP2StructMapping[], Uint32 mapSz,
+ bool ignoreMinMax,
+ bool ignoreUnknownKeys);
+
+ class Writer;
+ static UnpackStatus pack(class Writer &,
+ const void * src,
+ const SP2StructMapping[], Uint32 mapSz,
+ bool ignoreMinMax);
+
+ /**
+ * Reader class
+ */
+ class Reader {
+ public:
+ /**
+ * Move to first element
+ * Return true if element exist
+ */
+ bool first();
+
+ /**
+ * Move to next element
+ * Return true if element exist
+ */
+ bool next();
+
+ /**
+ * Is this valid
+ */
+ bool valid() const;
+
+ /**
+ * Get key
+ * Note only valid is valid() == true
+ */
+ Uint16 getKey() const;
+
+ /**
+ * Get value length in bytes - (including terminating 0 for strings)
+ * Note only valid is valid() == true
+ */
+ Uint16 getValueLen() const;
+
+ /**
+ * Get value type
+ * Note only valid is valid() == true
+ */
+ ValueType getValueType() const;
+
+ /**
+ * Get value
+ * Note only valid is valid() == true
+ */
+ Uint32 getUint32() const;
+ char * getString(char * dst) const;
+
+ /**
+ * Print the complete simple properties (for debugging)
+ */
+ void printAll(NdbOut& ndbout);
+
+ private:
+ bool readValue();
+
+ Uint16 m_key;
+ Uint16 m_itemLen;
+ union {
+ Uint32 m_ui32_value;
+ Uint32 m_strLen; // Including 0-byte in words
+ };
+ ValueType m_type;
+ protected:
+ Reader();
+ virtual void reset() = 0;
+
+ virtual bool step(Uint32 len) = 0;
+ virtual bool getWord(Uint32 * dst) = 0;
+ virtual bool peekWord(Uint32 * dst) const = 0;
+ virtual bool peekWords(Uint32 * dst, Uint32 len) const = 0;
+ };
+
+ /**
+ * Writer class
+ */
+ class Writer {
+ public:
+ bool first();
+ bool add(Uint16 key, Uint32 value);
+ bool add(Uint16 key, const char * value);
+ bool add(Uint16 key, const void* value, int len);
+ protected:
+ virtual bool reset() = 0;
+ virtual bool putWord(Uint32 val) = 0;
+ virtual bool putWords(const Uint32 * src, Uint32 len) = 0;
+ };
+};
+
+/**
+ * Reader for linear memory
+ */
+class SimplePropertiesLinearReader : public SimpleProperties::Reader {
+public:
+ SimplePropertiesLinearReader(const Uint32 * src, Uint32 len);
+
+ virtual void reset();
+ virtual bool step(Uint32 len);
+ virtual bool getWord(Uint32 * dst);
+ virtual bool peekWord(Uint32 * dst) const ;
+ virtual bool peekWords(Uint32 * dst, Uint32 len) const;
+private:
+ Uint32 m_len;
+ Uint32 m_pos;
+ const Uint32 * m_src;
+};
+
+/**
+ * Writer for linear memory
+ */
+class LinearWriter : public SimpleProperties::Writer {
+public:
+ LinearWriter(Uint32 * src, Uint32 len);
+
+ virtual bool reset();
+ virtual bool putWord(Uint32 val);
+ virtual bool putWords(const Uint32 * src, Uint32 len);
+ Uint32 getWordsUsed() const;
+private:
+ Uint32 m_len;
+ Uint32 m_pos;
+ Uint32 * m_src;
+};
+
+/**
+ * Writer for linear memory
+ */
+class UtilBufferWriter : public SimpleProperties::Writer {
+public:
+ UtilBufferWriter(class UtilBuffer & buf);
+
+ virtual bool reset();
+ virtual bool putWord(Uint32 val);
+ virtual bool putWords(const Uint32 * src, Uint32 len);
+ Uint32 getWordsUsed() const;
+private:
+ class UtilBuffer & m_buf;
+};
+
+/**
+ * Reader for long signal section memory
+ *
+ *
+ * Implemented in kernel/vm/SimplePropertiesSection.cpp
+ */
+class SimplePropertiesSectionReader : public SimpleProperties::Reader {
+public:
+ SimplePropertiesSectionReader(class SegmentedSectionPtr &,
+ class SectionSegmentPool &);
+
+ virtual void reset();
+ virtual bool step(Uint32 len);
+ virtual bool getWord(Uint32 * dst);
+ virtual bool peekWord(Uint32 * dst) const ;
+ virtual bool peekWords(Uint32 * dst, Uint32 len) const;
+ Uint32 getSize() const;
+ bool getWords(Uint32 * dst, Uint32 len);
+
+private:
+ Uint32 m_pos;
+ Uint32 m_len;
+ class SectionSegmentPool & m_pool;
+ class SectionSegment * m_head;
+ class SectionSegment * m_currentSegment;
+};
+
+inline
+Uint32 SimplePropertiesSectionReader::getSize() const
+{
+ return m_len;
+}
+
+/**
+ * Writer for long signal section memory
+ *
+ *
+ * Implemented in kernel/vm/SimplePropertiesSection.cpp
+ */
+class SimplePropertiesSectionWriter : public SimpleProperties::Writer {
+public:
+ SimplePropertiesSectionWriter(class SectionSegmentPool &);
+
+ virtual bool reset();
+ virtual bool putWord(Uint32 val);
+ virtual bool putWords(const Uint32 * src, Uint32 len);
+
+ /**
+ * This "unlinks" the writer from the memory
+ */
+ void getPtr(class SegmentedSectionPtr & dst);
+
+private:
+ Int32 m_pos;
+ Uint32 m_sz;
+ class SectionSegmentPool & m_pool;
+ class SectionSegment * m_head;
+ Uint32 m_prevPtrI; // Prev to m_currentSegment
+ class SectionSegment * m_currentSegment;
+};
+
+#endif
diff --git a/ndb/include/util/SocketServer.hpp b/ndb/include/util/SocketServer.hpp
new file mode 100644
index 00000000000..f1ce5182183
--- /dev/null
+++ b/ndb/include/util/SocketServer.hpp
@@ -0,0 +1,132 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SOCKET_SERVER_HPP
+#define SOCKET_SERVER_HPP
+
+#include <NdbTCP.h>
+#include <NdbMutex.h>
+#include <NdbThread.h>
+#include <Vector.hpp>
+
+extern "C" void* sessionThread_C(void*);
+extern "C" void* socketServerThread_C(void*);
+
+/**
+ * Socket Server
+ */
+class SocketServer {
+public:
+ /**
+ * A Session
+ */
+ class Session {
+ public:
+ virtual ~Session() {}
+ virtual void runSession(){}
+ virtual void stopSession(){}
+ protected:
+ friend class SocketServer;
+ friend void* sessionThread_C(void*);
+ Session(NDB_SOCKET_TYPE sock): m_socket(sock){ m_stop = m_stopped = false;}
+
+ bool m_stop; // Has the session been ordered to stop?
+ bool m_stopped; // Has the session stopped?
+
+ NDB_SOCKET_TYPE m_socket;
+ };
+
+ /**
+ * A service i.e. a session factory
+ */
+ class Service {
+ public:
+ virtual ~Service(){}
+
+ /**
+ * Returned Session will be ran in own thread
+ *
+ * To manage threads self, just return NULL
+ */
+ virtual Session * newSession(NDB_SOCKET_TYPE theSock) = 0;
+
+ virtual void stopSessions() {}
+ };
+
+ /**
+ * Constructor / Destructor
+ */
+ SocketServer(int maxSessions = 32);
+ ~SocketServer();
+
+ /**
+ * Setup socket and bind it
+ * then close the socket
+ * Returns true if succeding in binding
+ */
+ bool tryBind(unsigned short port, const char * intface = 0) const;
+
+ /**
+ * Setup socket
+ * bind & listen
+ * Returns false if no success
+ */
+ bool setup(Service *, unsigned short port, const char * pinterface = 0);
+
+ /**
+ * start/stop the server
+ */
+ void startServer();
+ void stopServer();
+
+ /**
+ * stop sessions
+ *
+ * Note: Implies stopServer
+ */
+ void stopSessions(bool wait = false);
+
+private:
+ struct SessionInstance {
+ Service * m_service;
+ Session * m_session;
+ NdbThread * m_thread;
+ };
+ struct ServiceInstance {
+ Service * m_service;
+ NDB_SOCKET_TYPE m_socket;
+ };
+ MutexVector<SessionInstance> m_sessions;
+ MutexVector<ServiceInstance> m_services;
+ unsigned m_maxSessions;
+
+ void doAccept();
+ void checkSessions();
+ void startSession(SessionInstance &);
+
+ /**
+ * Note, this thread is only used when running interactive
+ *
+ */
+ bool m_stopThread;
+ struct NdbThread * m_thread;
+ NdbLockable m_threadLock;
+ void doRun();
+ friend void* socketServerThread_C(void*);
+ friend void* sessionThread_C(void*);
+};
+
+#endif
diff --git a/ndb/include/util/UtilBuffer.hpp b/ndb/include/util/UtilBuffer.hpp
new file mode 100644
index 00000000000..97821ee3f9b
--- /dev/null
+++ b/ndb/include/util/UtilBuffer.hpp
@@ -0,0 +1,90 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __BUFFER_HPP_INCLUDED__
+#define __BUFFER_HPP_INCLUDED__
+
+#include <unistd.h>
+#include <errno.h>
+#include <stdlib.h>
+
+/* This class represents a buffer of binary data, where you can append
+ * data at the end, and later read the entire bunch.
+ * It will take care of the hairy details of realloc()ing the space
+ * for you
+ */
+class UtilBuffer {
+public:
+ UtilBuffer() { data = NULL; len = 0; alloc_size = 0; };
+ ~UtilBuffer() { if(data) free(data); data = NULL; len = 0; alloc_size = 0; };
+
+
+ int reallocate(size_t newsize) {
+ if(newsize < len) {
+ errno = EINVAL;
+ return -1;
+ }
+ void *newdata;
+ if((newdata = realloc(data, newsize)) == NULL) {
+ errno = ENOMEM;
+ return -1;
+ }
+ alloc_size = newsize;
+ data = newdata;
+ return 0;
+ };
+
+ int grow(size_t l) {
+ if(l > alloc_size)
+ return reallocate(l);
+ return 0;
+ };
+
+ int append(const void *d, size_t l) {
+ int ret;
+ ret = grow(len+l);
+ if(ret != 0)
+ return ret;
+
+ memcpy((char *)data+len, d, l);
+ len+=l;
+
+ return 0;
+ };
+
+ int assign(const void * d, size_t l) {
+ if (data) free(data);
+ data = NULL;
+ len = 0;
+ alloc_size = 0;
+ return append(d, l);
+ }
+
+ void clear() {
+ len = 0;
+ }
+
+ int length() const { return len; };
+
+ void *get_data() const { return data; };
+private:
+ void *data; /* Pointer to data storage */
+ size_t len; /* Size of the stored data */
+ size_t alloc_size; /* Size of the allocated space,
+ * i.e. len can grow to this size */
+};
+
+#endif /* !__BUFFER_HPP_INCLUDED__ */
diff --git a/ndb/include/util/Vector.hpp b/ndb/include/util/Vector.hpp
new file mode 100644
index 00000000000..a717dfecd7e
--- /dev/null
+++ b/ndb/include/util/Vector.hpp
@@ -0,0 +1,289 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_VECTOR_HPP
+#define NDB_VECTOR_HPP
+
+#include <stdlib.h>
+#include <NdbMutex.h>
+#include <string.h>
+
+template<class T>
+struct Vector {
+public:
+ Vector(int sz = 10);
+ ~Vector();
+
+ T& operator[](unsigned i);
+ const T& operator[](unsigned i) const;
+ unsigned size() const { return m_size; };
+
+ void push_back(const T &);
+ T& back();
+
+ void erase(unsigned index);
+
+ void clear();
+
+ void fill(unsigned new_size, T & obj);
+
+ Vector<T>& operator=(const Vector<T>&);
+
+ T* getBase() { return m_items;}
+ const T* getBase() const { return m_items;}
+private:
+ T * m_items;
+ unsigned m_size;
+ unsigned m_incSize;
+ unsigned m_arraySize;
+};
+
+template<class T>
+Vector<T>::Vector(int i){
+ m_items = new T[i];
+ m_size = 0;
+ m_arraySize = i;
+ m_incSize = 50;
+}
+
+template<class T>
+Vector<T>::~Vector(){
+ delete[] m_items;
+}
+
+template<class T>
+T &
+Vector<T>::operator[](unsigned i){
+ if(i >= m_size)
+ abort();
+ return m_items[i];
+}
+
+template<class T>
+const T &
+Vector<T>::operator[](unsigned i) const {
+ if(i >= m_size)
+ abort();
+ return m_items[i];
+}
+
+template<class T>
+T &
+Vector<T>::back(){
+ return (* this)[m_size - 1];
+}
+
+template<class T>
+void
+Vector<T>::push_back(const T & t){
+ if(m_size == m_arraySize){
+ T * tmp = new T [m_arraySize + m_incSize];
+ for (unsigned k = 0; k < m_size; k++)
+ tmp[k] = m_items[k];
+ delete[] m_items;
+ m_items = tmp;
+ m_arraySize = m_arraySize + m_incSize;
+ }
+ m_items[m_size] = t;
+ m_size++;
+}
+
+template<class T>
+void
+Vector<T>::erase(unsigned i){
+ if(i >= m_size)
+ abort();
+
+ for (unsigned k = i; k + 1 < m_size; k++)
+ m_items[k] = m_items[k + 1];
+ m_size--;
+}
+
+template<class T>
+void
+Vector<T>::clear(){
+ m_size = 0;
+}
+
+template<class T>
+void
+Vector<T>::fill(unsigned new_size, T & obj){
+ while(m_size <= new_size)
+ push_back(obj);
+}
+
+template<class T>
+Vector<T>&
+Vector<T>::operator=(const Vector<T>& obj){
+ if(this != &obj){
+ clear();
+ for(size_t i = 0; i<obj.size(); i++){
+ push_back(obj[i]);
+ }
+ }
+ return * this;
+}
+
+template<class T>
+struct MutexVector : public NdbLockable {
+ MutexVector(int sz = 10);
+ ~MutexVector();
+
+ T& operator[](unsigned i);
+ const T& operator[](unsigned i) const;
+ unsigned size() const { return m_size; };
+
+ void push_back(const T &);
+ void push_back(const T &, bool lockMutex);
+ T& back();
+
+ void erase(unsigned index);
+ void erase(unsigned index, bool lockMutex);
+
+ void clear();
+ void clear(bool lockMutex);
+
+ void fill(unsigned new_size, T & obj);
+private:
+ T * m_items;
+ unsigned m_size;
+ unsigned m_incSize;
+ unsigned m_arraySize;
+};
+
+template<class T>
+MutexVector<T>::MutexVector(int i){
+ m_items = new T[i];
+ m_size = 0;
+ m_arraySize = i;
+ m_incSize = 50;
+}
+
+template<class T>
+MutexVector<T>::~MutexVector(){
+ delete[] m_items;
+}
+
+template<class T>
+T &
+MutexVector<T>::operator[](unsigned i){
+ if(i >= m_size)
+ abort();
+ return m_items[i];
+}
+
+template<class T>
+const T &
+MutexVector<T>::operator[](unsigned i) const {
+ if(i >= m_size)
+ abort();
+ return m_items[i];
+}
+
+template<class T>
+T &
+MutexVector<T>::back(){
+ return (* this)[m_size - 1];
+}
+
+template<class T>
+void
+MutexVector<T>::push_back(const T & t){
+ lock();
+ if(m_size == m_arraySize){
+ T * tmp = new T [m_arraySize + m_incSize];
+ for (unsigned k = 0; k < m_size; k++)
+ tmp[k] = m_items[k];
+ delete[] m_items;
+ m_items = tmp;
+ m_arraySize = m_arraySize + m_incSize;
+ }
+ m_items[m_size] = t;
+ m_size++;
+ unlock();
+}
+
+template<class T>
+void
+MutexVector<T>::push_back(const T & t, bool lockMutex){
+ if(lockMutex)
+ lock();
+ if(m_size == m_arraySize){
+ T * tmp = new T [m_arraySize + m_incSize];
+ for (unsigned k = 0; k < m_size; k++)
+ tmp[k] = m_items[k];
+ delete[] m_items;
+ m_items = tmp;
+ m_arraySize = m_arraySize + m_incSize;
+ }
+ m_items[m_size] = t;
+ m_size++;
+ if(lockMutex)
+ unlock();
+}
+
+template<class T>
+void
+MutexVector<T>::erase(unsigned i){
+ if(i >= m_size)
+ abort();
+
+ lock();
+ for (unsigned k = i; k + 1 < m_size; k++)
+ m_items[k] = m_items[k + 1];
+ m_size--;
+ unlock();
+}
+
+template<class T>
+void
+MutexVector<T>::erase(unsigned i, bool _lock){
+ if(i >= m_size)
+ abort();
+
+ if(_lock)
+ lock();
+ for (unsigned k = i; k + 1 < m_size; k++)
+ m_items[k] = m_items[k + 1];
+ m_size--;
+ if(_lock)
+ unlock();
+}
+
+template<class T>
+void
+MutexVector<T>::clear(){
+ lock();
+ m_size = 0;
+ unlock();
+}
+
+template<class T>
+void
+MutexVector<T>::clear(bool l){
+ if(l) lock();
+ m_size = 0;
+ if(l) unlock();
+}
+
+template<class T>
+void
+MutexVector<T>::fill(unsigned new_size, T & obj){
+ while(m_size <= new_size)
+ push_back(obj);
+}
+
+#endif
diff --git a/ndb/include/util/getarg.h b/ndb/include/util/getarg.h
new file mode 100644
index 00000000000..713cf6e4b32
--- /dev/null
+++ b/ndb/include/util/getarg.h
@@ -0,0 +1,115 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*
+ * Copyright (c) 1997, 1999 Kungliga Tekniska Högskolan
+ * (Royal Institute of Technology, Stockholm, Sweden).
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of the Institute nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+/* $KTH: getarg.h,v 1.9 2000/09/01 21:25:55 lha Exp $ */
+
+#ifndef __GETARG_H__
+#define __GETARG_H__
+
+#include <stddef.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+typedef enum {
+ arg_integer,
+ arg_string,
+ arg_flag,
+ arg_negative_flag,
+ arg_strings,
+ arg_double,
+ arg_collect,
+ arg_counter
+} arg_type;
+
+struct getargs{
+ const char *long_name;
+ char short_name;
+ arg_type type;
+ void *value;
+ const char *help;
+ const char *arg_help;
+};
+
+enum {
+ ARG_ERR_NO_MATCH = 1,
+ ARG_ERR_BAD_ARG,
+ ARG_ERR_NO_ARG
+};
+
+typedef struct getarg_strings {
+ int num_strings;
+ char **strings;
+} getarg_strings;
+
+typedef int (*getarg_collect_func)(int short_opt,
+ int argc,
+ const char **argv,
+ int *optind,
+ int *optarg,
+ void *data);
+
+typedef struct getarg_collect_info {
+ getarg_collect_func func;
+ void *data;
+} getarg_collect_info;
+
+int getarg(struct getargs *args, size_t num_args,
+ int argc, const char **argv, int *optind);
+
+void arg_printusage (struct getargs *args,
+ size_t num_args,
+ const char *progname,
+ const char *extra_string);
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __GETARG_H__ */
diff --git a/ndb/include/util/md5_hash.hpp b/ndb/include/util/md5_hash.hpp
new file mode 100644
index 00000000000..4c3cf239881
--- /dev/null
+++ b/ndb/include/util/md5_hash.hpp
@@ -0,0 +1,25 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MD5_HASH_H
+#define MD5_HASH_H
+
+#include <ndb_types.h>
+
+// External declaration of hash function
+Uint32 md5_hash(const Uint64* keybuf, Uint32 no_of_32_words);
+
+#endif
diff --git a/ndb/include/util/random.h b/ndb/include/util/random.h
new file mode 100644
index 00000000000..1b83e5fec93
--- /dev/null
+++ b/ndb/include/util/random.h
@@ -0,0 +1,84 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef RANDOM_H
+#define RANDOM_H
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+typedef struct {
+ unsigned int length;
+ unsigned int *values;
+ unsigned int currentIndex;
+}RandomSequence;
+
+typedef struct {
+ unsigned int length;
+ unsigned int value;
+}SequenceValues;
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+extern double getTps(unsigned int count, double timeValue);
+
+/*----------------------------*/
+/* Random Sequences Functions */
+/*----------------------------*/
+extern int initSequence(RandomSequence *seq, SequenceValues *inputValues);
+extern unsigned int getNextRandom(RandomSequence *seq);
+extern void printSequence(RandomSequence *seq, unsigned int numPerRow);
+
+/*---------------------------------------------------*/
+/* Code from the glibc, to make sure the same random */
+/* number generator is used by all */
+/*---------------------------------------------------*/
+extern void myRandom48Init(long int seedval);
+extern long int myRandom48(unsigned int maxValue);
+
+#ifdef __cplusplus
+}
+#endif
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+
+
+#endif /* RANDOM_H */
+
diff --git a/ndb/include/util/socket_io.h b/ndb/include/util/socket_io.h
new file mode 100644
index 00000000000..bbd1bf115ae
--- /dev/null
+++ b/ndb/include/util/socket_io.h
@@ -0,0 +1,40 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef _SOCKET_IO_H
+#define _SOCKET_IO_H
+
+#include <NdbTCP.h>
+#include <stdarg.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+ int read_socket(NDB_SOCKET_TYPE, int timeout_ms, char *, int len);
+ int readln_socket(NDB_SOCKET_TYPE, int timeout_ms, char *, int len);
+ int write_socket(NDB_SOCKET_TYPE, int timeout_ms, const char[], int len);
+
+ int print_socket(NDB_SOCKET_TYPE, int timeout_ms, const char *, ...);
+ int println_socket(NDB_SOCKET_TYPE, int timeout_ms, const char *, ...);
+ int vprint_socket(NDB_SOCKET_TYPE, int timeout_ms, const char *, va_list);
+ int vprintln_socket(NDB_SOCKET_TYPE, int timeout_ms, const char *, va_list);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/ndb/include/util/uucode.h b/ndb/include/util/uucode.h
new file mode 100644
index 00000000000..138a79fa3ae
--- /dev/null
+++ b/ndb/include/util/uucode.h
@@ -0,0 +1,36 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef UUCODE_H
+#define UUCODE_H
+
+#include <stdio.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+ void uuencode(const char * data, int dataLen, FILE * out);
+ int uudecode(FILE * input, char * outBuf, int bufLen);
+
+ int uuencode_mem(char * dst, const char * src, int src_len);
+ int uudecode_mem(char * dst, int dst_len, const char * src);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/ndb/include/util/version.h b/ndb/include/util/version.h
new file mode 100644
index 00000000000..a82ae4b8b52
--- /dev/null
+++ b/ndb/include/util/version.h
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#ifndef VERSION_H
+#define VERSION_H
+#include <ndb_types.h>
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+ Uint32 getMajor(Uint32 version);
+
+ Uint32 getMinor(Uint32 version);
+
+ Uint32 getBuild(Uint32 version);
+
+ Uint32 makeVersion(Uint32 major, Uint32 minor, Uint32 build);
+
+ char* getVersionString(Uint32 version, char * status);
+
+ void ndbPrintVersion();
+ Uint32 ndbGetOwnVersion();
+
+ int ndbCompatible_mgmt_ndb(Uint32 ownVersion, Uint32 otherVersion);
+ int ndbCompatible_ndb_mgmt(Uint32 ownVersion, Uint32 otherVersion);
+ int ndbCompatible_mgmt_api(Uint32 ownVersion, Uint32 otherVersion);
+ int ndbCompatible_api_mgmt(Uint32 ownVersion, Uint32 otherVersion);
+ int ndbCompatible_api_ndb(Uint32 ownVersion, Uint32 otherVersion);
+ int ndbCompatible_ndb_api(Uint32 ownVersion, Uint32 otherVersion);
+ int ndbCompatible_ndb_ndb(Uint32 ownVersion, Uint32 otherVersion);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/ndb/lib/.empty b/ndb/lib/.empty
new file mode 100644
index 00000000000..e69de29bb2d
--- /dev/null
+++ b/ndb/lib/.empty
diff --git a/ndb/mysqlclusterenv.sh b/ndb/mysqlclusterenv.sh
new file mode 100644
index 00000000000..e151186d01e
--- /dev/null
+++ b/ndb/mysqlclusterenv.sh
@@ -0,0 +1,51 @@
+# Sets necessary environment variables for mysqlcluster install scripts
+mytop=
+if [ -f bin/mysql ]; then
+ mytop=`/bin/pwd`
+elif [ -f bin/ndb ]; then
+ mytop=`dirname \`/bin/pwd\``
+fi
+if [ "$mytop" ]; then
+ MYSQLCLUSTER_TOP=$mytop
+ PATH=$MYSQLCLUSTER_TOP/bin:$MYSQLCLUSTER_TOP/ndb/bin:$PATH
+ LD_LIBRARY_PATH=$MYSQLCLUSTER_TOP/lib:$LD_LIBRARY_PATH
+ LD_LIBRARY_PATH=$MYSQLCLUSTER_TOP/ndb/lib:$LD_LIBRARY_PATH
+ export MYSQLCLUSTER_TOP PATH LD_LIBRARY_PATH
+else
+if [ -d SCCS ]; then
+if [ -f ndb/mysqlclusterenv.sh ]; then
+ mytop=`/bin/pwd`
+elif [ -f mysqlclusterenv.sh ]; then
+ mytop=`dirname \`/bin/pwd\``
+fi
+fi
+if [ "$mytop" ]; then
+# we're in the development tree
+ if [ "$REAL_EMAIL" ]; then :; else
+#Guessing REAL_EMAIL
+ REAL_EMAIL=`whoami`@mysql.com
+ export REAL_EMAIL
+ echo Setting REAL_EMAIL=$REAL_EMAIL
+ fi
+
+ MYSQLCLUSTER_TOP=$mytop
+
+ NDB_TOP=$MYSQLCLUSTER_TOP/ndb
+ export NDB_TOP
+
+ NDB_PROJ_HOME=$NDB_TOP/home
+ export NDB_PROJ_HOME
+
+ PATH=$MYSQLCLUSTER_TOP/ndb/bin:$MYSQLCLUSTER_TOP/ndb/home/bin:$PATH
+ PATH=$MYSQLCLUSTER_TOP/client:$PATH
+ PATH=$MYSQLCLUSTER_TOP/sql:$PATH
+ LD_LIBRARY_PATH=$MYSQLCLUSTER_TOP/libmysql:$LD_LIBRARY_PATH
+ LD_LIBRARY_PATH=$MYSQLCLUSTER_TOP/libmysqld:$LD_LIBRARY_PATH
+ LD_LIBRARY_PATH=$MYSQLCLUSTER_TOP/ndb/lib:$LD_LIBRARY_PATH
+ LD_LIBRARY_PATH=$MYSQLCLUSTER_TOP/libmysql_r/.libs:$LD_LIBRARY_PATH
+ export MYSQLCLUSTER_TOP PATH LD_LIBRARY_PATH
+else
+ echo "Please source this file (mysqlclusterenv.sh) from installation top directory"
+fi
+fi
+mytop=
diff --git a/ndb/src/Makefile b/ndb/src/Makefile
new file mode 100644
index 00000000000..c4a58871f3b
--- /dev/null
+++ b/ndb/src/Makefile
@@ -0,0 +1,34 @@
+include .defs.mk
+
+DIRS := \
+ client \
+ common \
+ kernel \
+ ndbapi \
+ env \
+ mgmsrv \
+ mgmapi \
+ rep \
+ mgmclient \
+ cw \
+ newtonapi \
+ ndbbaseclient
+ifneq ($(NDB_ODBC),N)
+ DIRS += ndbclient
+endif
+ifeq ($(findstring OSE, $(NDB_OS)),)
+ DIRS += scripts
+endif
+include $(NDB_TOP)/Epilogue.mk
+
+_bins_mgmsrv: _libs_ndbapi
+_bins_mgmsrv: _libs_mgmapi
+_bins_mgmclient: _libs_mgmapi
+_bins_mgmclient: _libs_common
+_bins_client: _bins_ndbapi
+_bins_common: _bins_mgmapi
+_bins_kernel: _bins_ndbapi
+_bins_newtonapi: _bins_ndbapi
+_bins_mgmapi : _libs_common
+_bins_rep: _libs_common
+_bins_rep: _libs_ndbapi
diff --git a/ndb/src/client/Makefile b/ndb/src/client/Makefile
new file mode 100644
index 00000000000..1751a98bdfe
--- /dev/null
+++ b/ndb/src/client/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+DIRS =
+
+ifneq ($(NDB_ODBC),N)
+DIRS += odbc
+endif
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/client/odbc/Extra.mk b/ndb/src/client/odbc/Extra.mk
new file mode 100644
index 00000000000..762fb0bedd0
--- /dev/null
+++ b/ndb/src/client/odbc/Extra.mk
@@ -0,0 +1,59 @@
+# before Epilogue.mk
+
+CCFLAGS_LOC += -I..
+
+CCFLAGS_LOC += \
+ -I$(call fixpath,$(NDB_TOP)/include/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/include/util) \
+ -I$(call fixpath,$(NDB_TOP)/include/portlib)
+
+ifeq ($(NDB_OS),SOLARIS)
+
+CCFLAGS_LOC += -I/usr/local/include
+
+ifeq ($(NDB_COMPILER),GCC)
+LIBS_LOC += -Wl,-z,text
+CCFLAGS_WARNINGS += -Wno-unused -Wformat
+CCFLAGS_TOP += -D__STL_PTHREADS
+endif
+
+ifeq ($(NDB_COMPILER),FORTE6)
+LIBS_LOC += -z text
+LIBS_SPEC += /usr/lib/libCrun.so.1
+endif
+
+LIB_TARGET_LIBS += -lthread -lrt
+
+endif
+ifneq ($(filter $(NDB_OS), LINUX MACOSX IBMAIX TRU64X),)
+
+LIBS_LOC += -Wl,-z,text
+CCFLAGS_WARNINGS += -Wno-unused -Wformat
+GCC_VER := $(shell $(CC) --version)
+
+ifeq ($(GCC_VER),2.96)
+CCFLAGS_TOP += -D__STL_PTHREADS
+CCFLAGS_TOP += -fmessage-length=0
+endif
+
+CCFLAGS_TOP += -fno-rtti
+
+LIB_TARGET_LIBS += -lpthread
+
+endif
+
+ifeq ($(NDB_OS),WIN32)
+ifeq (RELEASE, $(NDB_VERSION))
+CCFLAGS_WIN += /MT /GR /GS /Zi -D_WINDOWS -D_USRDLL -DNDBODBC_EXPORTS -DNO_COMMAND_HANDLER -D_MBCS -D_WINDLL -U_LIB
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+CCFLAGS_WIN += /MT /GR /GS /Zi -D_WINDOWS -D_USRDLL -DNDBODBC_EXPORTS -DNO_COMMAND_HANDLER -D_MBCS -D_WINDLL -U_LIB
+else
+CCFLAGS_WIN += /MT /GR /GS /Zi -D_WINDOWS -D_USRDLL -DNDBODBC_EXPORTS -DNO_COMMAND_HANDLER -D_MBCS -D_WINDLL -U_LIB
+endif
+endif
+endif
+
+CCFLAGS_TOP += -DYYDEBUG=0 -fexceptions
+
+CCFLAGS_TOP += -DHAVE_LONG_LONG
diff --git a/ndb/src/client/odbc/Makefile b/ndb/src/client/odbc/Makefile
new file mode 100644
index 00000000000..2da683e7d86
--- /dev/null
+++ b/ndb/src/client/odbc/Makefile
@@ -0,0 +1,75 @@
+include .defs.mk
+
+TYPE = *
+
+A_LIB = N
+PIC_LIB = Y
+SO_LIB = Y
+
+LIB_TARGET = NDB_ODBC
+
+LIB_TARGET_ARCHIVES = $(LIB_DIRS:%=odbc%) NDB_API
+
+# Overide Defs.mk
+LDFLAGS_LAST = -lstdc++ -lm
+
+XXX = \
+ ndbapi \
+ mgmsrvcommon \
+ transporter \
+ general \
+ signaldataprint \
+ portlib \
+ logger \
+ trace
+
+ifeq ($(NDB_OS),WIN32)
+
+LIB_DIRS = \
+ handles \
+ dictionary \
+ codegen \
+ executor \
+ common
+
+SOURCES += NdbOdbc.cpp
+CFLAGS_NdbOdbc.cpp += -I. -I$(call fixpath,driver)
+
+PIC_ARCHIVE := Y
+NONPIC_ARCHIVE := N
+ARCHIVE_TARGET := ndb_odbcwin32
+LIB_TARGET_ARCHIVES += ndb_odbcwin32
+
+ifeq (RELEASE, $(NDB_VERSION))
+WIN_LIBS += /VERSION:2.0x /NODEFAULTLIB:"odbc32" /INCREMENTAL:NO /DLL /SUBSYSTEM:WINDOWS /MACHINE:IX86 /OPT:REF /OPT:ICF /DEF:NdbOdbc.def odbc32.lib odbccp32.lib user32.lib
+else
+ifeq (RELEASE_TRACE, $(NDB_VERSION))
+WIN_LIBS += /VERSION:2.0x /NODEFAULTLIB:"odbc32" /INCREMENTAL:NO /DLL /SUBSYSTEM:WINDOWS /MACHINE:IX86 /OPT:REF /OPT:ICF /DEF:NdbOdbc.def odbc32.lib odbccp32.lib user32.lib
+else
+WIN_LIBS += /VERSION:2.0x /NODEFAULTLIB:"odbc32" /INCREMENTAL /DLL /DEBUG /SUBSYSTEM:WINDOWS /MACHINE:IX86 /DEF:NdbOdbc.def odbc32.lib odbccp32.lib user32.lib
+endif
+endif
+
+else
+
+LIB_DIRS = \
+ driver \
+ handles \
+ dictionary \
+ codegen \
+ executor \
+ common
+
+endif
+
+include Extra.mk
+include $(NDB_TOP)/Epilogue.mk
+
+# yo
+
+test:
+ $(MAKE) -j4
+ $(MAKE) -C $(NDB_TOP)/tools/ndbsql
+ $(MAKE) -C $(NDB_TOP)/test/odbc/driver tidy
+ $(MAKE) -C $(NDB_TOP)/test/odbc/driver
+ $(MAKE) -C $(NDB_TOP)/test/ndbapi/testOIBasic
diff --git a/ndb/src/client/odbc/NdbOdbc.cpp b/ndb/src/client/odbc/NdbOdbc.cpp
new file mode 100755
index 00000000000..67c6b5e0004
--- /dev/null
+++ b/ndb/src/client/odbc/NdbOdbc.cpp
@@ -0,0 +1,78 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbUnistd.h>
+#include <odbcinst.h>
+
+#include "driver.cpp"
+
+
+BOOL APIENTRY DllMain( HANDLE hModule,
+ DWORD ul_reason_for_call,
+ LPVOID lpReserved
+ )
+{
+ switch (ul_reason_for_call)
+ {
+ case DLL_PROCESS_ATTACH:
+ case DLL_THREAD_ATTACH:
+ case DLL_THREAD_DETACH:
+ case DLL_PROCESS_DETACH:
+ break;
+ }
+ return TRUE;
+}
+
+
+BOOL INSTAPI ConfigDSN(
+ HWND hwndParent,
+ WORD fRequest,
+ LPCSTR lpszDriver,
+ LPCSTR lpszAttributes)
+{
+ const char* szDSN = "NDB";
+
+ switch(fRequest)
+ {
+ case ODBC_ADD_DSN:
+ SQLWriteDSNToIni(szDSN, lpszDriver);
+ break;
+
+ case ODBC_CONFIG_DSN:
+ break;
+
+ case ODBC_REMOVE_DSN:
+ SQLRemoveDSNFromIni(szDSN);
+ break;
+ }
+
+ return TRUE;
+}
+
+
+int FAR PASCAL
+DriverConnectProc(HWND hdlg, WORD wMsg, WPARAM wParam, LPARAM lParam)
+{
+ return FALSE;
+}
+
+void __declspec( dllexport) FAR PASCAL LoadByOrdinal(void);
+/* Entry point to cause DM to load using ordinals */
+void __declspec( dllexport) FAR PASCAL LoadByOrdinal(void)
+{
+}
+
diff --git a/ndb/src/client/odbc/NdbOdbc.def b/ndb/src/client/odbc/NdbOdbc.def
new file mode 100755
index 00000000000..85619b91915
--- /dev/null
+++ b/ndb/src/client/odbc/NdbOdbc.def
@@ -0,0 +1,85 @@
+LIBRARY NdbOdbc.DLL
+VERSION 03.51.00
+EXPORTS
+SQLAllocConnect
+SQLAllocEnv
+SQLAllocHandle
+SQLAllocHandleStd
+SQLAllocStmt
+SQLBindCol
+SQLBindParam
+SQLBindParameter
+SQLBrowseConnect
+SQLBulkOperations
+SQLCancel
+SQLCloseCursor
+SQLColAttribute
+SQLColAttributes
+SQLColumnPrivileges
+SQLColumns
+SQLConnect
+SQLCopyDesc
+SQLDataSources
+SQLDescribeCol
+SQLDescribeParam
+SQLDisconnect
+SQLDriverConnect
+SQLDrivers
+SQLEndTran
+SQLError
+SQLExecDirect
+SQLExecute
+SQLExtendedFetch
+SQLFetch
+SQLFetchScroll
+SQLForeignKeys
+SQLFreeConnect
+SQLFreeEnv
+SQLFreeHandle
+SQLFreeStmt
+SQLGetConnectAttr
+SQLGetConnectOption
+SQLGetCursorName
+SQLGetData
+SQLGetDescField
+SQLGetDescRec
+SQLGetDiagField
+SQLGetDiagRec
+SQLGetEnvAttr
+SQLGetFunctions
+SQLGetInfo
+SQLGetStmtAttr
+SQLGetStmtOption
+SQLGetTypeInfo
+SQLMoreResults
+SQLNativeSql
+SQLNumParams
+SQLNumResultCols
+SQLParamData
+SQLParamOptions
+SQLPrepare
+SQLPrimaryKeys
+SQLProcedureColumns
+SQLProcedures
+SQLPutData
+SQLRowCount
+SQLSetConnectAttr
+SQLSetConnectOption
+SQLSetCursorName
+SQLSetDescField
+SQLSetDescRec
+SQLSetEnvAttr
+SQLSetParam
+SQLSetPos
+SQLSetScrollOptions
+SQLSetStmtAttr
+SQLSetStmtOption
+SQLSpecialColumns
+SQLStatistics
+SQLTablePrivileges
+SQLTables
+SQLTransact
+DllMain
+DriverConnectProc
+ConfigDSN
+LoadByOrdinal
diff --git a/ndb/src/client/odbc/codegen/CodeGen.cpp b/ndb/src/client/odbc/codegen/CodeGen.cpp
new file mode 100644
index 00000000000..6be78b62bd9
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/CodeGen.cpp
@@ -0,0 +1,229 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <common/CodeTree.hpp>
+#include <executor/Executor.hpp>
+#include "CodeGen.hpp"
+#include "Code_root.hpp"
+
+#include <FlexLexer.h>
+#include "SimpleParser.hpp"
+
+void
+CodeGen::prepare(Ctx& ctx)
+{
+ parse(ctx);
+ if (! ctx.ok())
+ return;
+ analyze(ctx);
+ if (! ctx.ok())
+ return;
+ describe(ctx);
+}
+
+void
+CodeGen::execute(Ctx& ctx)
+{
+ DescArea& ipd = m_stmtArea.descArea(Desc_usage_IPD);
+ if (m_stmtArea.m_unbound) {
+ analyze(ctx);
+ if (! ctx.ok())
+ return;
+ describe(ctx);
+ if (! ctx.ok())
+ return;
+ if (m_stmtArea.m_unbound) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "%u input parameters have unbound SQL type", m_stmtArea.m_unbound);
+ return;
+ }
+ ipd.setBound(true);
+ }
+ if (! ipd.isBound()) {
+ ctx_log2(("IPD changed between executes - reanalyze"));
+ // jdbc can change parameter length at each execute
+ analyze(ctx);
+ if (! ctx.ok())
+ return;
+ describe(ctx);
+ if (! ctx.ok())
+ return;
+ freeExec(ctx);
+ codegen(ctx);
+ if (! ctx.ok())
+ return;
+ alloc(ctx);
+ if (! ctx.ok())
+ return;
+ ipd.setBound(true);
+ }
+ if (m_stmtArea.m_execTree == 0) {
+ codegen(ctx);
+ if (! ctx.ok())
+ return;
+ alloc(ctx);
+ if (! ctx.ok())
+ return;
+ }
+ Executor executor(m_stmtArea);
+ executor.execute(ctx);
+}
+
+void
+CodeGen::fetch(Ctx& ctx)
+{
+ // XXX parameter types are not checked any more
+ ctx_assert(! m_stmtArea.m_unbound);
+ Executor executor(m_stmtArea);
+ executor.fetch(ctx);
+}
+
+void
+CodeGen::parse(Ctx& ctx)
+{
+ Plan_root* planRoot = new Plan_root(m_stmtArea);
+ SimpleParser simpleParser(ctx, m_stmtArea, planRoot);
+ simpleParser.yyparse();
+ if (! ctx.ok())
+ return;
+ planRoot->m_paramList.resize(1 + simpleParser.paramNumber());
+ ctx_log2(("CodeGen: parse done - plan tree follows"));
+ if (ctx.logLevel() >= 2)
+ planRoot->print(ctx);
+ m_stmtArea.m_planTree = planRoot;
+}
+
+void
+CodeGen::analyze(Ctx& ctx)
+{
+ Plan_root* planRoot = static_cast<Plan_root*>(m_stmtArea.m_planTree);
+ ctx_assert(planRoot != 0);
+ Plan_base::Ctl ctl(0);
+ planRoot->analyze(ctx, ctl); // returns itself
+ if (! ctx.ok())
+ return;
+ ctx_log2(("CodeGen: analyze done - plan tree follows"));
+ if (ctx.logLevel() >= 2)
+ planRoot->print(ctx);
+}
+
+void
+CodeGen::describe(Ctx& ctx)
+{
+ Plan_root* planRoot = static_cast<Plan_root*>(m_stmtArea.m_planTree);
+ ctx_assert(planRoot != 0);
+ planRoot->describe(ctx);
+ ctx_log2(("CodeGen: describe done"));
+}
+
+void
+CodeGen::codegen(Ctx& ctx)
+{
+ Plan_root* planRoot = static_cast<Plan_root*>(m_stmtArea.m_planTree);
+ ctx_assert(planRoot != 0);
+ Plan_base::Ctl ctl(0);
+ Exec_root* execRoot = static_cast<Exec_root*>(planRoot->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return;
+ ctx_assert(execRoot != 0);
+ ctx_log2(("CodeGen: codegen done - code tree follows"));
+ if (ctx.logLevel() >= 2)
+ execRoot->print(ctx);
+ m_stmtArea.m_execTree = execRoot;
+}
+
+void
+CodeGen::alloc(Ctx& ctx)
+{
+ Exec_root* execRoot = static_cast<Exec_root*>(m_stmtArea.m_execTree);
+ ctx_assert(execRoot != 0);
+ Exec_base::Ctl ctl(0);
+ execRoot->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ ctx_log2(("CodeGen: alloc done"));
+}
+
+void
+CodeGen::close(Ctx& ctx)
+{
+ Exec_root* execRoot = static_cast<Exec_root*>(m_stmtArea.m_execTree);
+ if (execRoot != 0) {
+ execRoot->close(ctx);
+ ctx_log2(("CodeGen: close done"));
+ }
+}
+
+void
+CodeGen::free(Ctx& ctx)
+{
+ freePlan(ctx);
+ freeExec(ctx);
+}
+
+void
+CodeGen::freePlan(Ctx & ctx)
+{
+ if (m_stmtArea.m_planTree != 0) {
+ Plan_root* planRoot = static_cast<Plan_root*>(m_stmtArea.m_planTree);
+ ctx_assert(planRoot != 0);
+ unsigned count = 1 + planRoot->m_nodeList.size();
+ planRoot->freeNodeList();
+ delete planRoot;
+ m_stmtArea.m_planTree = 0;
+ ctx_log3(("CodeGen: freed %u plan tree nodes", count));
+ }
+}
+
+void
+CodeGen::freeExec(Ctx & ctx)
+{
+ if (m_stmtArea.m_execTree != 0) {
+ Exec_root* execRoot = static_cast<Exec_root*>(m_stmtArea.m_execTree);
+ ctx_assert(execRoot != 0);
+ unsigned count = 1 + execRoot->m_nodeList.size();
+ execRoot->freeNodeList();
+ delete execRoot;
+ m_stmtArea.m_execTree = 0;
+ ctx_log3(("CodeGen: freed %u exec tree nodes", count));
+ }
+}
+
+// odbc support
+
+void
+CodeGen::sqlGetData(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind)
+{
+ Exec_root* execRoot = static_cast<Exec_root*>(m_stmtArea.m_execTree);
+ ctx_assert(execRoot != 0);
+ execRoot->sqlGetData(ctx, columnNumber, targetType, targetValue, bufferLength, strlen_or_Ind);
+}
+
+void
+CodeGen::sqlParamData(Ctx& ctx, SQLPOINTER* value)
+{
+ Exec_root* execRoot = static_cast<Exec_root*>(m_stmtArea.m_execTree);
+ ctx_assert(execRoot != 0);
+ execRoot->sqlParamData(ctx, value);
+}
+
+void
+CodeGen::sqlPutData(Ctx& ctx, SQLPOINTER data, SQLINTEGER strlen_or_Ind)
+{
+ Exec_root* execRoot = static_cast<Exec_root*>(m_stmtArea.m_execTree);
+ ctx_assert(execRoot != 0);
+ execRoot->sqlPutData(ctx, data, strlen_or_Ind);
+}
diff --git a/ndb/src/client/odbc/codegen/CodeGen.hpp b/ndb/src/client/odbc/codegen/CodeGen.hpp
new file mode 100644
index 00000000000..ae61dab0c2a
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/CodeGen.hpp
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_CodeGen_hpp
+#define ODBC_CODEGEN_CodeGen_hpp
+
+#include <common/common.hpp>
+
+class StmtArea;
+class SqlField;
+class ExtField;
+
+/**
+ * @class CodeGen
+ * @brief Compiles SQL text into ExecTree::Code
+ */
+class CodeGen {
+public:
+ CodeGen(StmtArea& stmtArea);
+ ~CodeGen();
+ // parse and analyze SQL statement
+ void prepare(Ctx& ctx);
+ // these are passed to Executor
+ void execute(Ctx& ctx);
+ void fetch(Ctx& ctx);
+ // close statement (mainly scan)
+ void close(Ctx& ctx);
+ // free data structures
+ void free(Ctx& ctx);
+ // odbc support
+ void sqlGetData(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind);
+ void sqlParamData(Ctx& ctx, SQLPOINTER* value);
+ void sqlPutData(Ctx& ctx, SQLPOINTER data, SQLINTEGER strlen_or_Ind);
+private:
+ void parse(Ctx& ctx);
+ void analyze(Ctx& ctx);
+ void describe(Ctx& ctx);
+ void codegen(Ctx& ctx);
+ void alloc(Ctx& ctx);
+ void freePlan(Ctx& ctx);
+ void freeExec(Ctx& ctx);
+ StmtArea& m_stmtArea;
+};
+
+inline
+CodeGen::CodeGen(StmtArea& stmtArea) :
+ m_stmtArea(stmtArea)
+{
+}
+
+inline
+CodeGen::~CodeGen()
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_base.cpp b/ndb/src/client/odbc/codegen/Code_base.cpp
new file mode 100644
index 00000000000..dc02e071156
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_base.cpp
@@ -0,0 +1,167 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_base.hpp"
+#include "Code_root.hpp"
+
+// Plan_base
+
+Plan_base::~Plan_base()
+{
+}
+
+StmtArea&
+Plan_base::stmtArea() const
+{
+ ctx_assert(m_root != 0);
+ return m_root->m_stmtArea;
+}
+
+DescArea&
+Plan_base::descArea(DescUsage u) const
+{
+ return stmtArea().descArea(u);
+}
+
+ConnArea&
+Plan_base::connArea() const
+{
+ return stmtArea().connArea();
+}
+
+DictCatalog&
+Plan_base::dictCatalog() const
+{
+ return connArea().dictCatalog();
+}
+
+DictSchema&
+Plan_base::dictSchema() const
+{
+ return connArea().dictSchema();
+}
+
+Ndb*
+Plan_base::ndbObject() const
+{
+ Ndb* ndb = connArea().ndbObject();
+ ctx_assert(ndb != 0);
+ return ndb;
+}
+
+NdbSchemaCon*
+Plan_base::ndbSchemaCon() const
+{
+ NdbSchemaCon* ndbSchemaCon = connArea().ndbSchemaCon();
+ ctx_assert(ndbSchemaCon != 0);
+ return ndbSchemaCon;
+}
+
+NdbConnection*
+Plan_base::ndbConnection() const
+{
+ NdbConnection* ndbConnection = connArea().ndbConnection();
+ ctx_assert(ndbConnection != 0);
+ return ndbConnection;
+}
+
+void
+Plan_base::printList(Ctx& ctx, Plan_base* a[], unsigned n)
+{
+ for (unsigned i = 0; i < n; i++) {
+ if (a[i] == 0)
+ ctx.print(" -");
+ else
+ a[i]->print(ctx);
+ }
+}
+
+// Exec_base
+
+Exec_base::Code::~Code()
+{
+}
+
+Exec_base::Data::~Data()
+{
+}
+
+Exec_base::~Exec_base()
+{
+ delete m_code; // remove when code becomes shared
+ m_code = 0;
+ delete m_data;
+ m_data = 0;
+}
+
+StmtArea&
+Exec_base::stmtArea() const
+{
+ ctx_assert(m_root != 0);
+ return m_root->m_stmtArea;
+}
+
+DescArea&
+Exec_base::descArea(DescUsage u) const
+{
+ return stmtArea().descArea(u);
+}
+
+ConnArea&
+Exec_base::connArea() const
+{
+ return stmtArea().connArea();
+}
+
+DictSchema&
+Exec_base::dictSchema() const
+{
+ return connArea().dictSchema();
+}
+
+Ndb*
+Exec_base::ndbObject() const
+{
+ Ndb* ndb = connArea().ndbObject();
+ ctx_assert(ndb != 0);
+ return ndb;
+}
+
+NdbSchemaCon*
+Exec_base::ndbSchemaCon() const
+{
+ NdbSchemaCon* ndbSchemaCon = connArea().ndbSchemaCon();
+ ctx_assert(ndbSchemaCon != 0);
+ return ndbSchemaCon;
+}
+
+NdbConnection*
+Exec_base::ndbConnection() const
+{
+ NdbConnection* ndbConnection = connArea().ndbConnection();
+ ctx_assert(ndbConnection != 0);
+ return ndbConnection;
+}
+
+void
+Exec_base::printList(Ctx& ctx, Exec_base* a[], unsigned n)
+{
+ for (unsigned i = 0; i < n; i++) {
+ ctx_assert(a[i] != 0);
+ a[i]->print(ctx);
+ }
+}
diff --git a/ndb/src/client/odbc/codegen/Code_base.hpp b/ndb/src/client/odbc/codegen/Code_base.hpp
new file mode 100644
index 00000000000..c67c0ca7adb
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_base.hpp
@@ -0,0 +1,237 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_base_hpp
+#define ODBC_CODEGEN_Code_base_hpp
+
+#include <set>
+#include <list>
+#include <vector>
+#include <common/common.hpp>
+#include <common/CodeTree.hpp>
+#include <common/DescArea.hpp>
+
+class Ctx;
+class ConnArea;
+class StmtArea;
+class DescArea;
+class DictCatalog;
+class DictSchema;
+class ResultArea;
+class ResultSet;
+class SpecRow;
+class Ndb;
+class NdbSchemaCon;
+class NdbConnection;
+class NdbOperation;
+class NdbScanFilter;
+
+class Plan_root;
+class Plan_table;
+class Plan_column;
+class Plan_expr;
+class Plan_expr_param;
+class Plan_pred;
+class Plan_dml_row;
+class Plan_dml_column;
+class Plan_ddl_column;
+class Plan_ddl_constr;
+class Plan_idx_column;
+class Exec_root;
+class Exec_base;
+class Exec_query;
+class Exec_expr;
+class Exec_expr_row;
+class Exec_expr_param;
+
+/**
+ * @class Plan_base
+ * @brief Base class for plan trees
+ */
+class Plan_base : public PlanTree {
+public:
+ Plan_base(Plan_root* root);
+ virtual ~Plan_base() = 0;
+ // get references to StmtArea via Plan_root
+ StmtArea& stmtArea() const;
+ DescArea& descArea(DescUsage u) const;
+ ConnArea& connArea() const;
+ // catalogs
+ DictCatalog& dictCatalog() const;
+ DictSchema& dictSchema() const;
+ // ndb
+ Ndb* ndbObject() const;
+ NdbSchemaCon* ndbSchemaCon() const;
+ NdbConnection* ndbConnection() const;
+ // containers for Plan classes
+ typedef std::vector<Plan_table*> TableVector;
+ typedef std::vector<Plan_column*> ColumnVector;
+ typedef std::vector<Plan_dml_column*> DmlColumnVector;
+ typedef std::vector<Plan_ddl_column*> DdlColumnVector;
+ typedef std::vector<Plan_ddl_constr*> DdlConstrVector;
+ typedef std::vector<Plan_idx_column*> IdxColumnVector;
+ typedef std::vector<Plan_expr*> ExprVector;
+ typedef std::list<Plan_expr*> ExprList;
+ typedef std::vector<ExprList> ExprListVector;
+ typedef std::list<Plan_pred*> PredList;
+ typedef std::set<Plan_table*> TableSet;
+ typedef std::vector<Plan_expr_param*> ParamVector;
+ // control area on the stack XXX needs to be designed
+ struct Ctl {
+ Ctl(Ctl* up);
+ Ctl* m_up; // up the stack
+ // analyze
+ TableVector m_tableList; // resolve column names
+ bool m_topand; // in top-level where clause
+ bool m_extra; // anything but single pk=expr
+ bool m_aggrok; // aggregate allowed
+ bool m_aggrin; // within aggregate args
+ bool m_const; // only constants in set clause
+ PredList m_topcomp; // top level comparisons
+ Plan_dml_row *m_dmlRow; // row type to convert to
+ Plan_table* m_topTable; // top level table for interpreted progs
+ bool m_having; // in having-predicate
+ // codegen
+ Exec_root* m_execRoot; // root of Exec tree
+ const Exec_query* m_execQuery; // pass to column
+ };
+ // semantic analysis and optimization
+ virtual Plan_base* analyze(Ctx& ctx, Ctl& ctl) = 0;
+ // generate "executable" code
+ virtual Exec_base* codegen(Ctx& ctx, Ctl& ctl) = 0;
+ // misc
+ virtual void print(Ctx& ctx) = 0;
+protected:
+ Plan_root* m_root;
+ void printList(Ctx& ctx, Plan_base* a[], unsigned n);
+};
+
+inline
+Plan_base::Plan_base(Plan_root* root) :
+ m_root(root)
+{
+ ctx_assert(m_root != 0);
+}
+
+inline
+Plan_base::Ctl::Ctl(Ctl* up) :
+ m_up(up),
+ m_tableList(1), // 1-based
+ m_topand(false),
+ m_extra(false),
+ m_aggrok(false),
+ m_aggrin(false),
+ m_dmlRow(0),
+ m_topTable(0),
+ m_having(false),
+ m_execRoot(0),
+ m_execQuery(0)
+{
+}
+
+/**
+ * @class Exec_base
+ * @brief Base class for exec trees
+ */
+class Exec_base : public ExecTree {
+public:
+ class Code : public ExecTree::Code {
+ public:
+ virtual ~Code() = 0;
+ };
+ class Data : public ExecTree::Data {
+ public:
+ virtual ~Data() = 0;
+ };
+ Exec_base(Exec_root* root);
+ virtual ~Exec_base() = 0;
+ // get references to StmtArea via Exec_root
+ virtual StmtArea& stmtArea() const;
+ DescArea& descArea(DescUsage u) const;
+ ConnArea& connArea() const;
+ // catalogs
+ DictSchema& dictSchema() const;
+ // ndb
+ Ndb* ndbObject() const;
+ NdbSchemaCon* ndbSchemaCon() const;
+ NdbConnection* ndbConnection() const;
+ // containers for Exec classes
+ typedef std::vector<Exec_expr*> ExprVector;
+ typedef std::vector<Exec_expr_param*> ParamVector;
+ // control area on the stack
+ struct Ctl {
+ Ctl(Ctl* up);
+ Ctl* m_up; // up the stack
+ const Exec_query* m_query; // pass Data
+ ExprVector m_exprList; // pass Data
+ NdbOperation* m_scanOp; // scan operation
+ bool m_postEval; // for rownum
+ unsigned m_groupIndex; // for group by
+ bool m_groupInit; // first in group
+ Exec_expr_row* m_sortRow; // from sort to group by
+ NdbScanFilter* m_scanFilter; // scan filter
+ };
+ // allocate and deallocate Data instances
+ virtual void alloc(Ctx& ctx, Ctl& ctl) = 0;
+ virtual void close(Ctx& ctx) = 0;
+ // set Code and Data
+ void setCode(const Code& code);
+ void setData(Data& data);
+ // misc
+ virtual void print(Ctx& ctx) = 0;
+protected:
+ const Code* m_code;
+ Data* m_data;
+ Exec_root* m_root;
+ void printList(Ctx& ctx, Exec_base* a[], unsigned n);
+};
+
+inline
+Exec_base::Exec_base(Exec_root* root) :
+ m_code(0),
+ m_data(0),
+ m_root(root)
+{
+ ctx_assert(m_root != 0);
+}
+
+inline void
+Exec_base::setCode(const Code& code)
+{
+ ctx_assert(m_code == 0);
+ m_code = &code;
+}
+
+inline void
+Exec_base::setData(Data& data)
+{
+ ctx_assert(m_data == 0);
+ m_data = &data;
+}
+
+inline
+Exec_base::Ctl::Ctl(Ctl* up) :
+ m_up(up),
+ m_scanOp(0),
+ m_postEval(false),
+ m_groupIndex(0),
+ m_groupInit(false),
+ m_sortRow(0),
+ m_scanFilter(0)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_column.cpp b/ndb/src/client/odbc/codegen/Code_column.cpp
new file mode 100644
index 00000000000..c4c0480a5e7
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_column.cpp
@@ -0,0 +1,72 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <dictionary/DictSchema.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_column.hpp"
+#include "Code_table_list.hpp"
+#include "Code_table.hpp"
+
+// Plan_column
+
+Plan_column::~Plan_column()
+{
+}
+
+void
+Plan_column::analyzeColumn(Ctx& ctx, Plan_base::Ctl& ctl)
+{
+ if (m_resTable != 0) // done on previous pass
+ return;
+ if (! (ctl.m_tableList.size() > 1)) {
+ ctx.pushStatus(Sqlstate::_42000, Error::Gen, "column %s not allowed here", getPrintName());
+ return;
+ }
+ unsigned resCount = 0;
+ for (unsigned i = 1; i < ctl.m_tableList.size(); i++) {
+ Plan_table* table = ctl.m_tableList[i];
+ ctx_assert(table != 0);
+ int ret = table->resolveColumn(ctx, this);
+ if (ret < 0)
+ return;
+ if (ret)
+ resCount++;
+ }
+ if (resCount == 0) {
+ // XXX try to strip "schema name" from table name
+ for (unsigned i = 1; i < ctl.m_tableList.size(); i++) {
+ Plan_table* table = ctl.m_tableList[i];
+ ctx_assert(table != 0);
+ int ret = table->resolveColumn(ctx, this, true);
+ if (ret < 0)
+ return;
+ if (ret)
+ resCount++;
+ }
+ }
+ if (resCount == 0) {
+ ctx.pushStatus(Sqlstate::_42S22, Error::Gen, "column %s not found", getPrintName());
+ return;
+ }
+ if (resCount > 1) {
+ ctx.pushStatus(Error::Gen, "column %s is ambiguous", getPrintName());
+ return;
+ }
+ // copy SQL type
+ m_sqlType = dictColumn().sqlType();
+}
diff --git a/ndb/src/client/odbc/codegen/Code_column.hpp b/ndb/src/client/odbc/codegen/Code_column.hpp
new file mode 100644
index 00000000000..af0dcea690d
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_column.hpp
@@ -0,0 +1,122 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_column_hpp
+#define ODBC_CODEGEN_Code_column_hpp
+
+#include <common/common.hpp>
+#include <common/DataType.hpp>
+#include "Code_base.hpp"
+
+class DictColumn;
+class Plan_table;
+
+/**
+ * @class Plan_column
+ * @brief Abstract base class for columns
+ */
+class Plan_column {
+public:
+ enum Type {
+ Type_expr = 1,
+ Type_dml = 2,
+ Type_ddl = 3, // new columns in create table
+ Type_idx = 4 // old columns in create index
+ };
+ Plan_column(Type type, const BaseString& name);
+ virtual ~Plan_column() = 0;
+ void analyzeColumn(Ctx& ctx, Plan_base::Ctl& ctl);
+ // attributes
+ const BaseString& getName() const;
+ const BaseString& getCname() const;
+ const char* getPrintName() const;
+ void setCname(const BaseString& cname);
+ const DictColumn& dictColumn() const;
+ const SqlType& sqlType() const;
+protected:
+ friend class Plan_table;
+ friend class Plan_comp_op;
+ Type m_type;
+ BaseString m_name;
+ BaseString m_cname;
+ BaseString m_printName;
+ DictColumn* m_dictColumn;
+ /**
+ * Resolve to table and operational position (for example
+ * column number in scan query).
+ */
+ Plan_table* m_resTable;
+ unsigned m_resPos;
+ SqlType m_sqlType;
+};
+
+inline
+Plan_column::Plan_column(Type type, const BaseString& name) :
+ m_type(type),
+ m_name(name),
+ m_printName(name),
+ m_dictColumn(0),
+ m_resTable(0),
+ m_resPos(0)
+{
+}
+
+inline const BaseString&
+Plan_column::getName() const
+{
+ return m_name;
+}
+
+inline const BaseString&
+Plan_column::getCname() const
+{
+ return m_cname;
+}
+
+inline const char*
+Plan_column::getPrintName() const
+{
+ return m_printName.c_str();
+}
+
+inline void
+Plan_column::setCname(const BaseString& cname)
+{
+ m_cname.assign(cname);
+ if (m_cname.empty())
+ m_printName.assign(m_name);
+ else {
+ m_printName.assign(m_cname);
+ m_printName.append(".");
+ m_printName.append(m_name);
+ }
+}
+
+inline const DictColumn&
+Plan_column::dictColumn() const
+{
+ ctx_assert(m_dictColumn != 0);
+ return *m_dictColumn;
+}
+
+inline const SqlType&
+Plan_column::sqlType() const
+{
+ ctx_assert(m_sqlType.type() != SqlType::Undef);
+ return m_sqlType;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_comp_op.cpp b/ndb/src/client/odbc/codegen/Code_comp_op.cpp
new file mode 100644
index 00000000000..7782ed1ea2a
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_comp_op.cpp
@@ -0,0 +1,485 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <dictionary/DictColumn.hpp>
+#include "Code_pred.hpp"
+#include "Code_comp_op.hpp"
+#include "Code_expr_conv.hpp"
+#include "Code_expr_column.hpp"
+#include "Code_table.hpp"
+#include "Code_root.hpp"
+
+// Comp_op
+
+const char*
+Comp_op::name() const
+{
+ switch (m_opcode) {
+ case Eq:
+ return "=";
+ case Noteq:
+ return "!=";
+ case Lt:
+ return "<";
+ case Lteq:
+ return "<=";
+ case Gt:
+ return ">";
+ case Gteq:
+ return ">=";
+ case Like:
+ return "like";
+ case Notlike:
+ return "not like";
+ case Isnull:
+ return "is null";
+ case Isnotnull:
+ return "is not null";
+ }
+ ctx_assert(false);
+ return "";
+}
+
+unsigned
+Comp_op::arity() const
+{
+ switch (m_opcode) {
+ case Eq:
+ case Noteq:
+ case Lt:
+ case Lteq:
+ case Gt:
+ case Gteq:
+ case Like:
+ case Notlike:
+ return 2;
+ case Isnull:
+ case Isnotnull:
+ return 1;
+ }
+ ctx_assert(false);
+ return 0;
+}
+
+// Plan_comp_op
+
+Plan_comp_op::~Plan_comp_op()
+{
+}
+
+Plan_base*
+Plan_comp_op::analyze(Ctx& ctx, Ctl& ctl)
+{
+ m_exec = 0;
+ const unsigned arity = m_op.arity();
+ // analyze operands
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ m_expr[i]->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ }
+ // for each operand, find type to convert to
+ SqlType con[1 + 2];
+ if (arity == 1) {
+ const SqlType& t1 = m_expr[1]->sqlType();
+ switch (t1.type()) {
+ case SqlType::Char:
+ case SqlType::Varchar:
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ case SqlType::Real:
+ case SqlType::Double:
+ case SqlType::Datetime:
+ case SqlType::Null:
+ case SqlType::Unbound:
+ con[1] = t1;
+ break;
+ default:
+ break;
+ }
+ if (con[1].type() == SqlType::Undef) {
+ char b1[40];
+ t1.print(b1, sizeof(b1));
+ ctx.pushStatus(Error::Gen, "type mismatch in comparison: %s %s", b1, m_op.name());
+ return 0;
+ }
+ } else if (arity == 2) {
+ const SqlType& t1 = m_expr[1]->sqlType();
+ const SqlType& t2 = m_expr[2]->sqlType();
+ switch (t1.type()) {
+ case SqlType::Char:
+ switch (t2.type()) {
+ case SqlType::Char:
+ case SqlType::Varchar:
+ case SqlType::Null:
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Unbound:
+ con[1] = con[2] = t2;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Varchar:
+ switch (t2.type()) {
+ case SqlType::Char:
+ case SqlType::Varchar:
+ case SqlType::Null:
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Unbound:
+ con[1] = con[2] = t2;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ switch (t2.type()) {
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ // conversion would mask primary key optimization
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Real:
+ case SqlType::Double:
+ con[1].setType(ctx, SqlType::Double);
+ con[2] = con[1];
+ break;
+ case SqlType::Null:
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Unbound:
+ con[1] = con[2] = t2;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Real:
+ case SqlType::Double:
+ switch (t2.type()) {
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ case SqlType::Real:
+ case SqlType::Double:
+ con[1].setType(ctx, SqlType::Double);
+ con[2] = con[1];
+ break;
+ case SqlType::Null:
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Unbound:
+ con[1] = con[2] = t2;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Datetime:
+ switch (t2.type()) {
+ case SqlType::Datetime:
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Unbound:
+ con[1] = con[2] = t2;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Null:
+ switch (t2.type()) {
+ case SqlType::Char:
+ case SqlType::Varchar:
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ case SqlType::Real:
+ case SqlType::Double:
+ case SqlType::Datetime:
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Unbound:
+ con[1] = con[2] = t2;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Unbound:
+ con[1] = con[2] = t1;
+ break;
+ default:
+ break;
+ }
+ if (con[1].type() == SqlType::Undef || con[2].type() == SqlType::Undef) {
+ char b1[40], b2[40];
+ t1.print(b1, sizeof(b1));
+ t2.print(b2, sizeof(b2));
+ ctx.pushStatus(Error::Gen, "type mismatch in comparison: %s %s %s", b1, m_op.name(), b2);
+ return 0;
+ }
+ } else {
+ ctx_assert(false);
+ return 0;
+ }
+ if (! ctx.ok())
+ return 0;
+ // insert required conversions
+ for (unsigned i = 1; i <= arity; i++) {
+ if (con[i].type() == SqlType::Unbound) {
+ continue;
+ }
+ Plan_expr_conv* exprConv = new Plan_expr_conv(m_root, con[i]);
+ m_root->saveNode(exprConv);
+ exprConv->setExpr(m_expr[i]);
+ m_expr[i] = static_cast<Plan_expr*>(exprConv->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_expr[i] != 0);
+ }
+ // look for column=expr
+ if (ctl.m_topand && m_op.m_opcode == Comp_op::Eq) {
+ ctx_assert(arity == 2);
+ for (unsigned i = 1, j = 2; i <= 2; i++, j--) {
+ if (m_expr[i]->type() != Plan_expr::TypeColumn)
+ continue;
+ Plan_expr_column* column = static_cast<Plan_expr_column*>(m_expr[i]);
+ if (! column->resolveEq(ctx, m_expr[j]))
+ ctl.m_extra = true;
+ }
+ } else {
+ ctl.m_extra = true;
+ }
+ // save top level comparison on list
+ if (ctl.m_topand) {
+ ctl.m_topcomp.push_back(this);
+ }
+ // table dependencies are union from operands
+ m_tableSet.clear();
+ for (unsigned i = 1; i <= arity; i++) {
+ const TableSet& ts = m_expr[i]->tableSet();
+ m_tableSet.insert(ts.begin(), ts.end());
+ }
+ // set of tables for which interpreter cannot be used
+ m_noInterp.clear();
+ // convenient
+#undef ustype
+#define ustype(b, n) (((b) ? 1 : 0) * 100 + (n))
+ if (arity == 1) {
+ for (unsigned i = 1; i <= 1; i++) {
+ const SqlType t1 = m_expr[i]->sqlType();
+ switch (m_op.m_opcode) {
+ case Comp_op::Isnull:
+ case Comp_op::Isnotnull:
+ if (m_expr[i]->type() == Plan_expr::TypeColumn) {
+ switch (ustype(t1.unSigned(), t1.type())) {
+ // all types accepted now
+ default:
+ {
+ Plan_expr_column* column = static_cast<Plan_expr_column*>(m_expr[i]);
+ ctx_assert(column->m_resTable != 0);
+ m_interpColumn[i] = column;
+ continue; // ok
+ }
+ break;
+ }
+ }
+ break;
+ default:
+ break;
+ }
+ const TableSet& ts = m_expr[i]->tableSet();
+ m_noInterp.insert(ts.begin(), ts.end());
+ }
+ } else if (arity == 2) {
+ for (unsigned i = 1, j = 2; i <= 2; i++, j--) {
+ const SqlType t1 = m_expr[i]->sqlType();
+ switch (m_op.m_opcode) {
+ case Comp_op::Like:
+ case Comp_op::Notlike:
+ if (i == 2) // col like val but not val like col
+ break;
+ /*FALLTHRU*/
+ case Comp_op::Eq:
+ case Comp_op::Noteq:
+ case Comp_op::Lt:
+ case Comp_op::Lteq:
+ case Comp_op::Gt:
+ case Comp_op::Gteq:
+ if (m_expr[i]->type() == Plan_expr::TypeColumn) {
+ switch (ustype(t1.unSigned(), t1.type())) {
+ case ustype(false, SqlType::Char):
+ case ustype(false, SqlType::Varchar):
+ case ustype(true, SqlType::Smallint):
+ case ustype(true, SqlType::Integer):
+ case ustype(true, SqlType::Bigint):
+ {
+ Plan_expr_column* column = static_cast<Plan_expr_column*>(m_expr[i]);
+ ctx_assert(column->m_resTable != 0);
+ const TableSet& ts = m_expr[j]->tableSet();
+ if (ts.find(column->m_resTable) == ts.end()) {
+ // candidate for column=const
+ m_interpColumn[i] = column;
+ continue; // ok
+ }
+ }
+ break;
+ default:
+ break;
+ }
+ }
+ break;
+ default:
+ break;
+ }
+ const TableSet& ts = m_expr[i]->tableSet();
+ m_noInterp.insert(ts.begin(), ts.end());
+ }
+ } else {
+ ctx_assert(false);
+ return 0;
+ }
+#undef ustype
+ return this;
+}
+
+Exec_base*
+Plan_comp_op::codegen(Ctx& ctx, Ctl& ctl)
+{
+ if (m_exec != 0)
+ return m_exec;
+ const unsigned arity = m_op.arity();
+ Exec_comp_op* exec = new Exec_comp_op(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // create code for operands
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ Exec_expr* execExpr = static_cast<Exec_expr*>(m_expr[i]->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ exec->setExpr(i, execExpr);
+ }
+ // create the code
+ Exec_comp_op::Code& code = *new Exec_comp_op::Code(m_op);
+ // interpreted column=const
+ if (! ctl.m_having) {
+ ctx_assert(ctl.m_topTable != 0);
+ for (unsigned i = 1; i <= arity; i++) {
+ Plan_expr_column* column = m_interpColumn[i];
+ if (column == 0)
+ continue;
+ ctx_assert(column->m_resTable != 0);
+ if (column->m_resTable != ctl.m_topTable)
+ continue;
+ ctx_assert(code.m_interpColumn == 0);
+ code.m_interpColumn = i;
+ code.m_interpAttrId = column->dictColumn().getAttrId();
+ ctx_log2(("can use interpreter on %s", column->getPrintName()));
+ }
+ }
+ exec->setCode(code);
+ m_exec = exec;
+ return exec;
+}
+
+void
+Plan_comp_op::print(Ctx& ctx)
+{
+ ctx.print(" [%s", m_op.name());
+ Plan_base* a[] = { m_expr[1], m_expr[2] };
+ printList(ctx, a, m_op.arity());
+ ctx.print("]");
+}
+
+bool
+Plan_comp_op::isGroupBy(const Plan_expr_row* row) const
+{
+ const unsigned arity = m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ if (! m_expr[i]->isGroupBy(row))
+ return false;
+ }
+ return true;
+}
+
+// Code_comp_op
+
+Exec_comp_op::Code::~Code()
+{
+}
+
+Exec_comp_op::Data::~Data()
+{
+}
+
+Exec_comp_op::~Exec_comp_op()
+{
+}
+
+void
+Exec_comp_op::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // allocate subexpressions
+ unsigned arity = code.m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ m_expr[i]->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_comp_op::close(Ctx& ctx)
+{
+ const Code& code = getCode();
+ unsigned arity = code.m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ m_expr[i]->close(ctx);
+ }
+}
+
+void
+Exec_comp_op::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [%s", code.m_op.name());
+ Exec_base* a[] = { m_expr[1], m_expr[2] };
+ printList(ctx, a, code.m_op.arity());
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_comp_op.hpp b/ndb/src/client/odbc/codegen/Code_comp_op.hpp
new file mode 100644
index 00000000000..0585ab1dabf
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_comp_op.hpp
@@ -0,0 +1,172 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_comp_op_hpp
+#define ODBC_CODEGEN_Code_comp_op_hpp
+
+#include <common/common.hpp>
+#include <common/DataField.hpp>
+#include "Code_pred.hpp"
+#include "Code_expr.hpp"
+#include "Code_expr_column.hpp"
+
+/**
+ * @class Comp_op
+ * @brief Comparison operations
+ */
+struct Comp_op {
+ enum Opcode {
+ Eq = 1, // binary
+ Noteq,
+ Lt,
+ Lteq,
+ Gt,
+ Gteq,
+ Like,
+ Notlike,
+ Isnull, // unary
+ Isnotnull
+ };
+ Comp_op(Opcode opcode);
+ const char* name() const;
+ unsigned arity() const;
+ Opcode m_opcode;
+};
+
+inline
+Comp_op::Comp_op(Opcode opcode) :
+ m_opcode(opcode)
+{
+}
+
+/**
+ * @class Plan_comp_op
+ * @brief Comparison operator node in PlanTree
+ */
+class Plan_comp_op : public Plan_pred {
+public:
+ Plan_comp_op(Plan_root* root, Comp_op op);
+ virtual ~Plan_comp_op();
+ void setExpr(unsigned i, Plan_expr* expr);
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ virtual bool isGroupBy(const Plan_expr_row* row) const;
+protected:
+ Comp_op m_op;
+ Plan_expr* m_expr[1 + 2];
+ Plan_expr_column* m_interpColumn[1 + 2]; // candidates
+};
+
+inline
+Plan_comp_op::Plan_comp_op(Plan_root* root, Comp_op op) :
+ Plan_pred(root, TypeComp),
+ m_op(op)
+{
+ m_expr[0] = m_expr[1] = m_expr[2] = 0;
+ m_interpColumn[0] = m_interpColumn[1] = m_interpColumn[2] = 0;
+}
+
+inline void
+Plan_comp_op::setExpr(unsigned i, Plan_expr* expr)
+{
+ ctx_assert(1 <= i && i <= 2);
+ m_expr[i] = expr;
+}
+
+/**
+ * @class Exec_comp_op
+ * @brief Comparison operator node in ExecTree
+ */
+class Exec_comp_op : public Exec_pred {
+public:
+ class Code : public Exec_pred::Code {
+ public:
+ Code(Comp_op op);
+ virtual ~Code();
+ protected:
+ friend class Plan_comp_op;
+ friend class Exec_comp_op;
+ Comp_op m_op;
+ unsigned m_interpColumn; // 1 or 2 if interpreted column, 0 if both constant
+ NdbAttrId m_interpAttrId;
+ };
+ class Data : public Exec_pred::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_comp_op;
+ };
+ Exec_comp_op(Exec_root* root);
+ virtual ~Exec_comp_op();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execInterp(Ctx& ctx, Ctl& ctl);
+ void evaluate(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setExpr(unsigned i, Exec_expr* expr);
+protected:
+ Exec_expr* m_expr[1 + 2];
+};
+
+inline
+Exec_comp_op::Code::Code(Comp_op op) :
+ m_op(op),
+ m_interpColumn(0),
+ m_interpAttrId((NdbAttrId)-1)
+{
+}
+
+inline
+Exec_comp_op::Data::Data()
+{
+}
+
+inline
+Exec_comp_op::Exec_comp_op(Exec_root* root) :
+ Exec_pred(root)
+{
+ m_expr[0] = m_expr[1] = m_expr[2] = 0;
+}
+
+// children
+
+inline const Exec_comp_op::Code&
+Exec_comp_op::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_comp_op::Data&
+Exec_comp_op::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_comp_op::setExpr(unsigned i, Exec_expr* expr)
+{
+ ctx_assert(1 <= i && i <= 2 && m_expr[i] == 0);
+ m_expr[i] = expr;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_create_index.cpp b/ndb/src/client/odbc/codegen/Code_create_index.cpp
new file mode 100644
index 00000000000..84f319338a4
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_create_index.cpp
@@ -0,0 +1,124 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_create_index.hpp"
+#include "Code_root.hpp"
+
+// Plan_create_index
+
+Plan_create_index::~Plan_create_index()
+{
+}
+
+Plan_base*
+Plan_create_index::analyze(Ctx& ctx, Ctl& ctl)
+{
+ stmtArea().stmtInfo().setName(Stmt_name_create_index);
+ // analyze the table
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ // analyze the columns
+ ctl.m_tableList.resize(1 + 1); // indexed from 1
+ ctl.m_tableList[1] = m_table;
+ for (unsigned i = 1, n = countColumn(); i <= n; i++) {
+ Plan_idx_column* column = getColumn(i);
+ column->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ }
+ return this;
+}
+
+void
+Plan_create_index::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "CREATE INDEX", SQL_DIAG_CREATE_INDEX);
+}
+
+Exec_base*
+Plan_create_index::codegen(Ctx& ctx, Ctl& ctl)
+{
+ Exec_create_index* exec = new Exec_create_index(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ const unsigned count = countColumn();
+ const char** attrList = new const char* [1 + count];
+ attrList[0] = 0; // unused
+ for (unsigned i = 1; i <= count; i++) {
+ Plan_idx_column* column = getColumn(i);
+ const char* cname = column->getName().c_str();
+ attrList[i] = strcpy(new char[strlen(cname) + 1], cname);
+ }
+ Exec_create_index::Code& code = *new Exec_create_index::Code(m_name, m_table->getName(), m_type, count, attrList);
+ exec->setCode(code);
+ code.m_fragmentType = m_fragmentType;
+ code.m_logging = m_logging;
+ return exec;
+}
+
+void
+Plan_create_index::print(Ctx& ctx)
+{
+ ctx.print(" [create_index name=%s table=%s type=%d", m_name.c_str(), m_table->getName().c_str(), (int)m_type);
+ ctx.print(" [");
+ for (unsigned i = 1; i <= countColumn(); i++) {
+ Plan_idx_column* column = getColumn(i);
+ if (i > 1)
+ ctx.print(" ");
+ column->print(ctx);
+ }
+ ctx.print("]");
+}
+
+// Exec_create_index
+
+Exec_create_index::Code::~Code()
+{
+ for (unsigned i = 1; i <= m_attrCount; i++) {
+ delete[] m_attrList[i];
+ m_attrList[i] = 0;
+ }
+ delete[] m_attrList;
+}
+
+Exec_create_index::Data::~Data()
+{
+}
+
+Exec_create_index::~Exec_create_index()
+{
+}
+
+void
+Exec_create_index::alloc(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_create_index::close(Ctx& ctx)
+{
+}
+
+void
+Exec_create_index::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [create_index %s]", code.m_tableName.c_str());
+}
diff --git a/ndb/src/client/odbc/codegen/Code_create_index.hpp b/ndb/src/client/odbc/codegen/Code_create_index.hpp
new file mode 100644
index 00000000000..ebd757e1118
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_create_index.hpp
@@ -0,0 +1,203 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_create_index_hpp
+#define ODBC_CODEGEN_Code_create_index_hpp
+
+#include <vector>
+#include <NdbApi.hpp>
+#include <common/common.hpp>
+#include "Code_ddl.hpp"
+#include "Code_table.hpp"
+#include "Code_idx_column.hpp"
+
+class DictTable;
+class DictColumn;
+
+/**
+ * @class Plan_create_index
+ * @brief Create table in PlanTree
+ */
+class Plan_create_index : public Plan_ddl {
+public:
+ Plan_create_index(Plan_root* root, const BaseString& name);
+ virtual ~Plan_create_index();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx & ctx);
+ void print(Ctx& ctx);
+ // attributes
+ const BaseString& getName() const;
+ // children
+ void setType(NdbDictionary::Object::Type type);
+ void setTable(Plan_table* table);
+ unsigned countColumn() const;
+ void addColumn(Plan_idx_column* column);
+ Plan_idx_column* getColumn(unsigned i) const;
+ void setFragmentType(NdbDictionary::Object::FragmentType fragmentType);
+ void setLogging(bool logging);
+protected:
+ BaseString m_name;
+ NdbDictionary::Object::Type m_type;
+ Plan_table* m_table;
+ IdxColumnVector m_columnList;
+ NdbDictionary::Object::FragmentType m_fragmentType;
+ bool m_logging;
+};
+
+inline
+Plan_create_index::Plan_create_index(Plan_root* root, const BaseString& name) :
+ Plan_ddl(root),
+ m_name(name),
+ m_type(NdbDictionary::Object::TypeUndefined),
+ m_columnList(1),
+ m_fragmentType(NdbDictionary::Object::FragUndefined),
+ m_logging(true)
+{
+}
+
+inline const BaseString&
+Plan_create_index::getName() const
+{
+ return m_name;
+}
+
+// children
+
+inline void
+Plan_create_index::setType(NdbDictionary::Object::Type type)
+{
+ m_type = type;
+}
+
+inline void
+Plan_create_index::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+inline unsigned
+Plan_create_index::countColumn() const
+{
+ return m_columnList.size() - 1;
+}
+
+inline void
+Plan_create_index::addColumn(Plan_idx_column* column)
+{
+ ctx_assert(column != 0);
+ m_columnList.push_back(column);
+}
+
+inline Plan_idx_column*
+Plan_create_index::getColumn(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= countColumn() && m_columnList[i] != 0);
+ return m_columnList[i];
+}
+
+inline void
+Plan_create_index::setFragmentType(NdbDictionary::Object::FragmentType fragmentType)
+{
+ m_fragmentType = fragmentType;
+}
+
+inline void
+Plan_create_index::setLogging(bool logging)
+{
+ m_logging = logging;
+}
+
+/**
+ * @class Exec_create_index
+ * @brief Create table in ExecTree
+ */
+class Exec_create_index : public Exec_ddl {
+public:
+ class Code : public Exec_ddl::Code {
+ public:
+ Code(const BaseString& indexName, const BaseString& tableName, NdbDictionary::Object::Type type, unsigned attrCount, const char** attrList);
+ virtual ~Code();
+ protected:
+ friend class Plan_create_index;
+ friend class Exec_create_index;
+ const BaseString m_indexName;
+ const BaseString m_tableName;
+ NdbDictionary::Object::Type m_type;
+ const unsigned m_attrCount;
+ const char** m_attrList;
+ NdbDictionary::Object::FragmentType m_fragmentType;
+ bool m_logging;
+ };
+ class Data : public Exec_ddl::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_create_index;
+ };
+ Exec_create_index(Exec_root* root);
+ virtual ~Exec_create_index();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execute(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_create_index::Code::Code(const BaseString& indexName, const BaseString& tableName, NdbDictionary::Object::Type type, unsigned attrCount, const char** attrList) :
+ m_indexName(indexName),
+ m_tableName(tableName),
+ m_type(type),
+ m_attrCount(attrCount),
+ m_attrList(attrList),
+ m_fragmentType(NdbDictionary::Object::FragUndefined),
+ m_logging(true)
+{
+}
+
+inline
+Exec_create_index::Data::Data()
+{
+}
+
+inline
+Exec_create_index::Exec_create_index(Exec_root* root) :
+ Exec_ddl(root)
+{
+}
+
+// children
+
+inline const Exec_create_index::Code&
+Exec_create_index::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_create_index::Data&
+Exec_create_index::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_create_row.cpp b/ndb/src/client/odbc/codegen/Code_create_row.cpp
new file mode 100644
index 00000000000..5b90b658ed7
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_create_row.cpp
@@ -0,0 +1,162 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_create_row.hpp"
+#include "Code_root.hpp"
+
+Plan_create_row::~Plan_create_row()
+{
+}
+
+Plan_base*
+Plan_create_row::analyze(Ctx& ctx, Ctl& ctl)
+{
+ // check for duplicate column name
+ for (unsigned i = 1, n = countColumn(); i < n; i++) {
+ const BaseString& a = getColumn(i)->getName();
+ for (unsigned i2 = i + 1; i2 <= n; i2++) {
+ const BaseString& a2 = getColumn(i2)->getName();
+ if (strcmp(a.c_str(), a2.c_str()) == 0) {
+ ctx.pushStatus(Error::Gen, "duplicate column %s", a.c_str());
+ return 0;
+ }
+ }
+ }
+ // move single-column primary key constraint to constraint list
+ for (unsigned i = 1, n = countColumn(); i <= n; i++) {
+ Plan_ddl_column* column = getColumn(i);
+ if (column->m_primaryKey) {
+ Plan_ddl_row* ddlRow = new Plan_ddl_row(m_root);
+ m_root->saveNode(ddlRow);
+ ddlRow->addColumn(column);
+ Plan_ddl_constr* constr = new Plan_ddl_constr(m_root);
+ m_root->saveNode(constr);
+ constr->setRow(ddlRow);
+ addConstr(constr);
+ column->m_primaryKey = false; // will be set again
+ }
+ }
+ // check primary key constraints
+ if (countConstr() < 1) {
+ ctx.pushStatus(Error::Gen, "table must have a primary key");
+ return 0;
+ }
+ if (countConstr() > 1) {
+ ctx.pushStatus(Error::Gen, "table can have only one primary key");
+ return 0;
+ }
+ Plan_ddl_row* ddlRow = getConstr(1)->getRow();
+ for (unsigned i = 1, n = ddlRow->countColumn(); i <= n; i++) {
+ Plan_ddl_column* column = ddlRow->getColumn(i);
+ const BaseString& a = column->getName();
+ bool found = false;
+ for (unsigned i2 = 1, n2 = countColumn(); i2 <= n2; i2++) {
+ Plan_ddl_column* column2 = getColumn(i2);
+ const BaseString& a2 = column2->getName();
+ if (strcmp(a.c_str(), a2.c_str()) != 0)
+ continue;
+ if (column2->getPrimaryKey()) {
+ ctx.pushStatus(Error::Gen, "duplicate primary key constraint on %s", a.c_str());
+ return 0;
+ }
+ column2->setPrimaryKey();
+ found = true;
+ break;
+ }
+ if (! found) {
+ ctx.pushStatus(Error::Gen, "undefined primary key column %s", a.c_str());
+ return 0;
+ }
+ }
+ // analyze column types
+ for (unsigned i = 1, n = countColumn(); i <= n; i++) {
+ getColumn(i)->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ }
+ // check TupleId
+ unsigned tupleId = 0;
+ for (unsigned i = 1, n = countColumn(); i <= n; i++) {
+ Plan_ddl_column* column = getColumn(i);
+ if (! column->getTupleId())
+ continue;
+ if (i != 1) {
+ ctx.pushStatus(Error::Gen, "tuple id column %u is not first column", i);
+ return 0;
+ }
+ if (tupleId != 0) { // cannot happen now since attr name is fixed
+ ctx.pushStatus(Error::Gen, "duplicate tuple id column %u", i);
+ return 0;
+ }
+ tupleId = i;
+ }
+ if (tupleId != 0) {
+ for (unsigned i = 1, n = countColumn(); i <= n; i++) {
+ Plan_ddl_column* column = getColumn(i);
+ if (i == tupleId)
+ continue;
+ if (! column->getPrimaryKey())
+ continue;
+ ctx.pushStatus(Error::Gen, "cannot have both tuple id and other primary key column %u", i);
+ return 0;
+ }
+ }
+ // check auto-increment
+ unsigned autoIncrement = 0;
+ for (unsigned i = 1, n = countColumn(); i <= n; i++) {
+ Plan_ddl_column* column = getColumn(i);
+ if (! column->getAutoIncrement())
+ continue;
+ if (autoIncrement != 0) {
+ ctx.pushStatus(Error::Gen, "duplicate auto-increment column %u", i);
+ return 0;
+ }
+ autoIncrement = i;
+ }
+ if (autoIncrement != 0) {
+ for (unsigned i = 1, n = countColumn(); i <= n; i++) {
+ Plan_ddl_column* column = getColumn(i);
+ if (i == autoIncrement)
+ continue;
+ if (! column->getPrimaryKey())
+ continue;
+ ctx.pushStatus(Error::Gen, "cannot have both auto-increment column and other primary key column %u", i);
+ return 0;
+ }
+ }
+ return this;
+}
+
+Exec_base*
+Plan_create_row::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_create_row::print(Ctx& ctx)
+{
+ ctx.print(" [create_row");
+ for (unsigned i = 1; i <= countColumn(); i++) {
+ Plan_base* a = m_columnList[i];
+ printList(ctx, &a, 1);
+ }
+ for (unsigned i = 1; i <= countConstr(); i++) {
+ Plan_base* a = m_constrList[i];
+ printList(ctx, &a, 1);
+ }
+}
diff --git a/ndb/src/client/odbc/codegen/Code_create_row.hpp b/ndb/src/client/odbc/codegen/Code_create_row.hpp
new file mode 100644
index 00000000000..f03455ff28e
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_create_row.hpp
@@ -0,0 +1,99 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_create_row_hpp
+#define ODBC_CODEGEN_Code_create_row_hpp
+
+#include <vector>
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_ddl_column.hpp"
+#include "Code_ddl_constr.hpp"
+
+/**
+ * @class Plan_create_row
+ * @brief Row of columns and constraints in create statement
+ */
+class Plan_create_row : public Plan_base {
+public:
+ Plan_create_row(Plan_root* root);
+ virtual ~Plan_create_row();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ unsigned countColumn() const;
+ void addColumn(Plan_ddl_column* column);
+ Plan_ddl_column* getColumn(unsigned i) const;
+ unsigned countConstr() const;
+ void addConstr(Plan_ddl_constr* constr);
+ Plan_ddl_constr* getConstr(unsigned i) const;
+protected:
+ DdlColumnVector m_columnList;
+ DdlConstrVector m_constrList;
+};
+
+inline
+Plan_create_row::Plan_create_row(Plan_root* root) :
+ Plan_base(root),
+ m_columnList(1),
+ m_constrList(1)
+{
+}
+
+// children
+
+inline unsigned
+Plan_create_row::countColumn() const
+{
+ return m_columnList.size() - 1;
+}
+
+inline void
+Plan_create_row::addColumn(Plan_ddl_column* column)
+{
+ ctx_assert(column != 0);
+ m_columnList.push_back(column);
+}
+
+inline Plan_ddl_column*
+Plan_create_row::getColumn(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= m_columnList.size() && m_columnList[i] != 0);
+ return m_columnList[i];
+}
+
+inline unsigned
+Plan_create_row::countConstr() const
+{
+ return m_constrList.size() - 1;
+}
+
+inline void
+Plan_create_row::addConstr(Plan_ddl_constr* constr)
+{
+ ctx_assert(constr != 0);
+ m_constrList.push_back(constr);
+}
+
+inline Plan_ddl_constr*
+Plan_create_row::getConstr(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= m_constrList.size() && m_constrList[i] != 0);
+ return m_constrList[i];
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_create_table.cpp b/ndb/src/client/odbc/codegen/Code_create_table.cpp
new file mode 100644
index 00000000000..14e4abbd7fe
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_create_table.cpp
@@ -0,0 +1,137 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_create_table.hpp"
+#include "Code_root.hpp"
+
+// Plan_create_table
+
+Plan_create_table::~Plan_create_table()
+{
+}
+
+Plan_base*
+Plan_create_table::analyze(Ctx& ctx, Ctl& ctl)
+{
+ stmtArea().stmtInfo().setName(Stmt_name_create_table);
+ // analyze the create row
+ ctx_assert(m_createRow != 0);
+ m_createRow->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+void
+Plan_create_table::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "CREATE TABLE", SQL_DIAG_CREATE_TABLE);
+}
+
+Exec_base*
+Plan_create_table::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_createRow != 0);
+ Exec_create_table* exec = new Exec_create_table(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ const unsigned count = m_createRow->countColumn();
+ Exec_create_table::Code::Attr* attrList = new Exec_create_table::Code::Attr[1 + count];
+ unsigned tupleId = 0;
+ unsigned autoIncrement = 0;
+ for (unsigned i = 1; i <= count; i++) {
+ Plan_ddl_column* column = m_createRow->getColumn(i);
+ Exec_create_table::Code::Attr& attr = attrList[i];
+ attr.m_attrName.assign(column->getName());
+ attr.m_sqlType = column->sqlType();
+ attr.m_tupleKey = column->getPrimaryKey();
+ attr.m_tupleId = column->getTupleId();
+ attr.m_autoIncrement = column->getAutoIncrement();
+ if (attr.m_tupleId)
+ tupleId = i;
+ if (attr.m_autoIncrement)
+ autoIncrement = i;
+ attr.m_defaultValue = 0;
+ Plan_expr* expr;
+ if ((expr = column->getDefaultValue()) != 0) {
+ Exec_expr* execExpr = static_cast<Exec_expr*>(expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ attr.m_defaultValue = execExpr;
+ }
+ }
+ Exec_create_table::Code& code = *new Exec_create_table::Code(m_name, count, attrList, tupleId, autoIncrement);
+ exec->setCode(code);
+ code.m_fragmentType = m_fragmentType;
+ code.m_logging = m_logging;
+ return exec;
+}
+
+void
+Plan_create_table::print(Ctx& ctx)
+{
+ ctx.print(" [create_table '%s'", m_name.c_str());
+ Plan_base* a[] = { m_createRow };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
+
+// Exec_create_table
+
+Exec_create_table::Code::~Code()
+{
+ delete[] m_attrList;
+}
+
+Exec_create_table::Data::~Data()
+{
+}
+
+Exec_create_table::~Exec_create_table()
+{
+}
+
+void
+Exec_create_table::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ const Code::Attr& attr = code.m_attrList[i];
+ if (attr.m_defaultValue != 0)
+ attr.m_defaultValue->alloc(ctx, ctl);
+ }
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_create_table::close(Ctx& ctx)
+{
+ const Code& code = getCode();
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ const Code::Attr& attr = code.m_attrList[i];
+ if (attr.m_defaultValue != 0)
+ attr.m_defaultValue->close(ctx);
+ }
+}
+
+void
+Exec_create_table::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [create_table %s]", code.m_tableName.c_str());
+}
diff --git a/ndb/src/client/odbc/codegen/Code_create_table.hpp b/ndb/src/client/odbc/codegen/Code_create_table.hpp
new file mode 100644
index 00000000000..cbb2189d8ce
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_create_table.hpp
@@ -0,0 +1,178 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_create_table_hpp
+#define ODBC_CODEGEN_Code_create_table_hpp
+
+#include <vector>
+#include <common/common.hpp>
+#include "Code_ddl.hpp"
+#include "Code_ddl_row.hpp"
+#include "Code_create_row.hpp"
+
+class DictTable;
+class DictColumn;
+
+/**
+ * @class Plan_create_table
+ * @brief Create table in PlanTree
+ */
+class Plan_create_table : public Plan_ddl {
+public:
+ Plan_create_table(Plan_root* root, const BaseString& name);
+ virtual ~Plan_create_table();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx & ctx);
+ void print(Ctx& ctx);
+ // attributes
+ const BaseString& getName() const;
+ // children
+ void setCreateRow(Plan_create_row* createRow);
+ void setFragmentType(NdbDictionary::Object::FragmentType fragmentType);
+ void setLogging(bool logging);
+protected:
+ BaseString m_name;
+ Plan_create_row* m_createRow;
+ NdbDictionary::Object::FragmentType m_fragmentType;
+ bool m_logging;
+};
+
+inline
+Plan_create_table::Plan_create_table(Plan_root* root, const BaseString& name) :
+ Plan_ddl(root),
+ m_name(name),
+ m_createRow(0),
+ m_fragmentType(NdbDictionary::Object::FragUndefined),
+ m_logging(true)
+{
+}
+
+inline const BaseString&
+Plan_create_table::getName() const
+{
+ return m_name;
+}
+
+// children
+
+inline void
+Plan_create_table::setCreateRow(Plan_create_row* createRow)
+{
+ ctx_assert(createRow != 0);
+ m_createRow = createRow;
+}
+
+inline void
+Plan_create_table::setFragmentType(NdbDictionary::Object::FragmentType fragmentType)
+{
+ m_fragmentType = fragmentType;
+}
+
+inline void
+Plan_create_table::setLogging(bool logging)
+{
+ m_logging = logging;
+}
+
+/**
+ * @class Exec_create_table
+ * @brief Create table in ExecTree
+ */
+class Exec_create_table : public Exec_ddl {
+public:
+ class Code : public Exec_ddl::Code {
+ public:
+ struct Attr {
+ Attr() : m_defaultValue(0) {}
+ BaseString m_attrName;
+ SqlType m_sqlType;
+ bool m_tupleKey;
+ bool m_tupleId;
+ bool m_autoIncrement;
+ Exec_expr* m_defaultValue;
+ };
+ Code(const BaseString& tableName, unsigned attrCount, const Attr* attrList, unsigned tupleId, unsigned autoIncrement);
+ virtual ~Code();
+ protected:
+ friend class Plan_create_table;
+ friend class Exec_create_table;
+ const BaseString m_tableName;
+ const unsigned m_attrCount;
+ const Attr* const m_attrList;
+ unsigned m_tupleId;
+ unsigned m_autoIncrement;
+ NdbDictionary::Object::FragmentType m_fragmentType;
+ bool m_logging;
+ };
+ class Data : public Exec_ddl::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_create_table;
+ };
+ Exec_create_table(Exec_root* root);
+ virtual ~Exec_create_table();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execute(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_create_table::Code::Code(const BaseString& tableName, unsigned attrCount, const Attr* attrList, unsigned tupleId, unsigned autoIncrement) :
+ m_tableName(tableName),
+ m_attrCount(attrCount),
+ m_attrList(attrList),
+ m_tupleId(tupleId),
+ m_autoIncrement(autoIncrement),
+ m_fragmentType(NdbDictionary::Object::FragUndefined),
+ m_logging(true)
+{
+}
+
+inline
+Exec_create_table::Data::Data()
+{
+}
+
+inline
+Exec_create_table::Exec_create_table(Exec_root* root) :
+ Exec_ddl(root)
+{
+}
+
+// children
+
+inline const Exec_create_table::Code&
+Exec_create_table::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_create_table::Data&
+Exec_create_table::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_data_type.cpp b/ndb/src/client/odbc/codegen/Code_data_type.cpp
new file mode 100644
index 00000000000..1ff0fcebcbe
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_data_type.cpp
@@ -0,0 +1,44 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include "Code_data_type.hpp"
+
+// Plan_data_type
+
+Plan_data_type::~Plan_data_type()
+{
+}
+
+Plan_base*
+Plan_data_type::analyze(Ctx& ctx, Ctl& ctl)
+{
+ return this;
+}
+
+Exec_base*
+Plan_data_type::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_data_type::print(Ctx& ctx)
+{
+ ctx.print(" [data_type]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_data_type.hpp b/ndb/src/client/odbc/codegen/Code_data_type.hpp
new file mode 100644
index 00000000000..735dc05014f
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_data_type.hpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_data_type_hpp
+#define ODBC_CODEGEN_Code_data_type_hpp
+
+#include <common/common.hpp>
+#include <common/DataType.hpp>
+#include "Code_base.hpp"
+
+/**
+ * @class Plan_data_type
+ * @brief Data type in DDL statement
+ *
+ * This is pure plan node.
+ */
+class Plan_data_type : public Plan_base {
+public:
+ Plan_data_type(Plan_root* root, const SqlType& sqlType);
+ virtual ~Plan_data_type();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+private:
+ friend class Plan_ddl_column;
+ SqlType m_sqlType;
+};
+
+inline
+Plan_data_type::Plan_data_type(Plan_root* root, const SqlType& sqlType) :
+ Plan_base(root),
+ m_sqlType(sqlType)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_ddl.cpp b/ndb/src/client/odbc/codegen/Code_ddl.cpp
new file mode 100644
index 00000000000..2ba4291a0e8
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_ddl.cpp
@@ -0,0 +1,37 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_ddl.hpp"
+
+// Plan_ddl
+
+Plan_ddl::~Plan_ddl()
+{
+}
+
+// Exec_ddl
+
+Exec_ddl::Code::~Code()
+{
+}
+
+Exec_ddl::Data::~Data()
+{
+}
+
+Exec_ddl::~Exec_ddl()
+{
+}
diff --git a/ndb/src/client/odbc/codegen/Code_ddl.hpp b/ndb/src/client/odbc/codegen/Code_ddl.hpp
new file mode 100644
index 00000000000..1ceca62d55d
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_ddl.hpp
@@ -0,0 +1,63 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_ddl_hpp
+#define ODBC_CODEGEN_Code_ddl_hpp
+
+#include <common/common.hpp>
+#include "Code_stmt.hpp"
+
+/**
+ * @class Plan_ddl
+ * @brief Base class for DDL statements in PlanTree
+ */
+class Plan_ddl : public Plan_stmt {
+public:
+ Plan_ddl(Plan_root* root);
+ virtual ~Plan_ddl() = 0;
+};
+
+inline
+Plan_ddl::Plan_ddl(Plan_root* root) :
+ Plan_stmt(root)
+{
+}
+
+/**
+ * @class Exec_ddl
+ * @brief Base class for DDL statements in ExecTree
+ */
+class Exec_ddl : public Exec_stmt {
+public:
+ class Code : public Exec_stmt::Code {
+ public:
+ virtual ~Code() = 0;
+ };
+ class Data : public Exec_stmt::Data {
+ public:
+ virtual ~Data() = 0;
+ };
+ Exec_ddl(Exec_root* root);
+ virtual ~Exec_ddl() = 0;
+};
+
+inline
+Exec_ddl::Exec_ddl(Exec_root* root) :
+ Exec_stmt(root)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_ddl_column.cpp b/ndb/src/client/odbc/codegen/Code_ddl_column.cpp
new file mode 100644
index 00000000000..ee037e54c1f
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_ddl_column.cpp
@@ -0,0 +1,104 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include "Code_ddl_column.hpp"
+#include "Code_expr_conv.hpp"
+#include "Code_root.hpp"
+
+// Plan_ddl_column
+
+Plan_ddl_column::~Plan_ddl_column()
+{
+}
+
+Plan_base*
+Plan_ddl_column::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_type != 0);
+ if (! m_type->m_sqlType.nullable()) {
+ m_nullable = false;
+ }
+ m_sqlType = m_type->m_sqlType;
+ m_sqlType.nullable(m_nullable);
+ const BaseString& name = getName();
+ if (m_unSigned) {
+ switch (m_sqlType.type()) {
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "invalid unsigned qualifier on column %s", name.c_str());
+ return 0;
+ }
+ m_sqlType.unSigned(true);
+ }
+ if (strcmp(name.c_str(), "NDB$TID") == 0) {
+ if (! m_primaryKey) {
+ ctx.pushStatus(Error::Gen, "column %s must be a primary key", name.c_str());
+ return 0;
+ }
+ if (sqlType().type() != SqlType::Bigint || ! sqlType().unSigned()) {
+ ctx.pushStatus(Error::Gen, "tuple id %s must have type BIGINT UNSIGNED", name.c_str());
+ return 0;
+ }
+ setTupleId();
+ }
+ if (m_autoIncrement) {
+ if (! m_primaryKey) {
+ ctx.pushStatus(Error::Gen, "auto-increment column %s must be a primary key", name.c_str());
+ return 0;
+ }
+ if (sqlType().type() != SqlType::Smallint && sqlType().type() != SqlType::Integer && sqlType().type() != SqlType::Bigint) {
+ ctx.pushStatus(Error::Gen, "auto-increment column %s must have an integral type", name.c_str());
+ return 0;
+ }
+ }
+ if (m_defaultValue != 0) {
+ if (m_primaryKey) {
+ ctx.pushStatus(Sqlstate::_42000, Error::Gen, "default value not allowed on primary key column %s", name.c_str());
+ return 0;
+ }
+ m_defaultValue->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ // insert conversion node
+ Plan_expr_conv* exprConv = new Plan_expr_conv(m_root, sqlType());
+ m_root->saveNode(exprConv);
+ exprConv->setExpr(m_defaultValue);
+ Plan_expr* expr = static_cast<Plan_expr*>(exprConv->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(expr != 0);
+ m_defaultValue = expr;
+ }
+ return this;
+}
+
+Exec_base*
+Plan_ddl_column::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_ddl_column::print(Ctx& ctx)
+{
+ ctx.print(" [ddl_column %s key=%d id=%d]", getPrintName(), m_primaryKey, m_tupleId);
+}
diff --git a/ndb/src/client/odbc/codegen/Code_ddl_column.hpp b/ndb/src/client/odbc/codegen/Code_ddl_column.hpp
new file mode 100644
index 00000000000..7d089d37440
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_ddl_column.hpp
@@ -0,0 +1,150 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_ddl_column_hpp
+#define ODBC_CODEGEN_Code_ddl_column_hpp
+
+#include <common/common.hpp>
+#include "Code_column.hpp"
+#include "Code_data_type.hpp"
+#include "Code_expr.hpp"
+
+class DictColumn;
+class Plan_table;
+
+/**
+ * @class Plan_ddl_column
+ * @brief Column in DDL statement
+ */
+class Plan_ddl_column : public Plan_base, public Plan_column {
+public:
+ Plan_ddl_column(Plan_root* root, const BaseString& name);
+ virtual ~Plan_ddl_column();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // attributes
+ void setNotNull();
+ void setUnSigned();
+ void setPrimaryKey();
+ bool getPrimaryKey() const;
+ void setTupleId();
+ bool getTupleId() const;
+ void setAutoIncrement();
+ bool getAutoIncrement() const;
+ // children
+ void setType(Plan_data_type* type);
+ void setDefaultValue(Plan_expr* defaultValue);
+ Plan_expr* getDefaultValue() const;
+protected:
+ friend class Plan_create_row;
+ Plan_data_type* m_type;
+ Plan_expr* m_defaultValue;
+ bool m_nullable;
+ bool m_unSigned;
+ bool m_primaryKey;
+ bool m_tupleId;
+ bool m_autoIncrement;
+};
+
+inline
+Plan_ddl_column::Plan_ddl_column(Plan_root* root, const BaseString& name) :
+ Plan_base(root),
+ Plan_column(Type_ddl, name),
+ m_type(0),
+ m_defaultValue(0),
+ m_nullable(true),
+ m_unSigned(false),
+ m_primaryKey(false),
+ m_tupleId(false),
+ m_autoIncrement(false)
+{
+}
+
+inline void
+Plan_ddl_column::setNotNull()
+{
+ m_nullable = false;
+}
+
+inline void
+Plan_ddl_column::setUnSigned()
+{
+ m_unSigned = true;
+}
+
+inline void
+Plan_ddl_column::setPrimaryKey()
+{
+ m_nullable = false;
+ m_primaryKey = true;
+}
+
+inline bool
+Plan_ddl_column::getPrimaryKey() const
+{
+ return m_primaryKey;
+}
+
+inline void
+Plan_ddl_column::setTupleId()
+{
+ m_nullable = false;
+ m_tupleId = true;
+}
+
+inline bool
+Plan_ddl_column::getTupleId() const
+{
+ return m_tupleId;
+}
+
+inline void
+Plan_ddl_column::setAutoIncrement()
+{
+ m_nullable = false;
+ m_autoIncrement = true;
+}
+
+inline bool
+Plan_ddl_column::getAutoIncrement() const
+{
+ return m_autoIncrement;
+}
+
+// children
+
+inline void
+Plan_ddl_column::setType(Plan_data_type* type)
+{
+ ctx_assert(type != 0);
+ m_type = type;
+}
+
+inline void
+Plan_ddl_column::setDefaultValue(Plan_expr* defaultValue)
+{
+ ctx_assert(defaultValue != 0);
+ m_defaultValue = defaultValue;
+}
+
+inline Plan_expr*
+Plan_ddl_column::getDefaultValue() const
+{
+ return m_defaultValue;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_ddl_constr.cpp b/ndb/src/client/odbc/codegen/Code_ddl_constr.cpp
new file mode 100644
index 00000000000..78c23e38d97
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_ddl_constr.cpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include "Code_ddl_constr.hpp"
+
+// Plan_ddl_constr
+
+Plan_ddl_constr::~Plan_ddl_constr()
+{
+}
+
+Plan_base*
+Plan_ddl_constr::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_ddlRow != 0);
+ m_ddlRow->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_ddl_constr::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_ddl_constr::print(Ctx& ctx)
+{
+ ctx.print(" [ddl_constr");
+ Plan_base* a[] = { m_ddlRow };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_ddl_constr.hpp b/ndb/src/client/odbc/codegen/Code_ddl_constr.hpp
new file mode 100644
index 00000000000..ea7808b37cb
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_ddl_constr.hpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_ddl_constr_hpp
+#define ODBC_CODEGEN_Code_ddl_constr_hpp
+
+#include <common/common.hpp>
+#include "Code_ddl_row.hpp"
+
+/**
+ * @class Plan_ddl_constr
+ * @brief Constraint in DDL statement
+ *
+ * Only unnamed primary key constraint exists.
+ */
+class Plan_ddl_constr : public Plan_base {
+public:
+ Plan_ddl_constr(Plan_root* root);
+ virtual ~Plan_ddl_constr();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setRow(Plan_ddl_row* ddlRow);
+ Plan_ddl_row* getRow() const;
+protected:
+ Plan_ddl_row* m_ddlRow;
+};
+
+inline
+Plan_ddl_constr::Plan_ddl_constr(Plan_root* root) :
+ Plan_base(root)
+{
+}
+
+// children
+
+inline void
+Plan_ddl_constr::setRow(Plan_ddl_row* ddlRow)
+{
+ ctx_assert(ddlRow != 0);
+ m_ddlRow = ddlRow;
+}
+
+inline Plan_ddl_row*
+Plan_ddl_constr::getRow() const
+{
+ ctx_assert(m_ddlRow != 0);
+ return m_ddlRow;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_ddl_row.cpp b/ndb/src/client/odbc/codegen/Code_ddl_row.cpp
new file mode 100644
index 00000000000..87589ebbaa0
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_ddl_row.cpp
@@ -0,0 +1,54 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_ddl_row.hpp"
+#include "Code_ddl_column.hpp"
+#include "Code_ddl_constr.hpp"
+
+Plan_ddl_row::~Plan_ddl_row()
+{
+}
+
+Plan_base*
+Plan_ddl_row::analyze(Ctx& ctx, Ctl& ctl)
+{
+ // analyze the columns
+ for (unsigned i = 1, n = countColumn(); i <= n; i++) {
+ Plan_ddl_column* column = getColumn(i);
+ column->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ }
+ // node was not replaced
+ return this;
+}
+
+Exec_base*
+Plan_ddl_row::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_ddl_row::print(Ctx& ctx)
+{
+ ctx.print(" [ddl_row");
+ for (unsigned i = 1, n = countColumn(); i <= n; i++) {
+ Plan_base* a = m_columnList[i];
+ printList(ctx, &a, 1);
+ }
+}
diff --git a/ndb/src/client/odbc/codegen/Code_ddl_row.hpp b/ndb/src/client/odbc/codegen/Code_ddl_row.hpp
new file mode 100644
index 00000000000..ac3eded1b2e
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_ddl_row.hpp
@@ -0,0 +1,72 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_ddl_row_hpp
+#define ODBC_CODEGEN_Code_ddl_row_hpp
+
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_ddl_column.hpp"
+
+/**
+ * @class Plan_ddl_row
+ * @brief Row of columns in create statement
+ */
+class Plan_ddl_row : public Plan_base {
+public:
+ Plan_ddl_row(Plan_root* root);
+ virtual ~Plan_ddl_row();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ unsigned countColumn() const;
+ void addColumn(Plan_ddl_column* column);
+ Plan_ddl_column* getColumn(unsigned i) const;
+protected:
+ DdlColumnVector m_columnList;
+};
+
+inline
+Plan_ddl_row::Plan_ddl_row(Plan_root* root) :
+ Plan_base(root),
+ m_columnList(1)
+{
+}
+
+// children
+
+inline unsigned
+Plan_ddl_row::countColumn() const
+{
+ return m_columnList.size() - 1;
+}
+
+inline void
+Plan_ddl_row::addColumn(Plan_ddl_column* column)
+{
+ ctx_assert(column != 0);
+ m_columnList.push_back(column);
+}
+
+inline Plan_ddl_column*
+Plan_ddl_row::getColumn(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= countColumn() && m_columnList[i] != 0);
+ return m_columnList[i];
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_delete.cpp b/ndb/src/client/odbc/codegen/Code_delete.cpp
new file mode 100644
index 00000000000..35b3daa1aca
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_delete.cpp
@@ -0,0 +1,205 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_delete.hpp"
+#include "Code_delete_lookup.hpp"
+#include "Code_delete_index.hpp"
+#include "Code_delete_scan.hpp"
+#include "Code_query_filter.hpp"
+#include "Code_query_lookup.hpp"
+#include "Code_query_index.hpp"
+#include "Code_query_scan.hpp"
+#include "Code_query_range.hpp"
+#include "Code_query_repeat.hpp"
+#include "Code_table.hpp"
+#include "Code_root.hpp"
+
+Plan_delete::~Plan_delete()
+{
+}
+
+Plan_base*
+Plan_delete::analyze(Ctx& ctx, Ctl& ctl)
+{
+ stmtArea().stmtInfo().setName(Stmt_name_delete);
+ // analyze the table
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ // set name resolution scope
+ ctl.m_tableList.resize(1 + 1); // indexed from 1
+ ctl.m_tableList[1] = m_table;
+ Plan_dml* stmt = 0;
+ if (m_pred != 0) {
+ // analyze the predicate
+ ctl.m_topand = true;
+ ctl.m_extra = false;
+ m_pred = static_cast<Plan_pred*>(m_pred->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_pred != 0);
+ // check for key match
+ Plan_table::Index* indexBest = 0;
+ for (unsigned i = 0; i <= m_table->indexCount(); i++) {
+ Plan_table::Index& index = m_table->m_indexList[i];
+ TableSet tsDone;
+ m_table->resolveSet(ctx, index, tsDone);
+ if (! ctx.ok())
+ return 0;
+ if (! index.m_keyFound)
+ continue;
+ // prefer smaller rank, less unused keys
+ int k;
+ (k = (indexBest == 0)) ||
+ (k = (indexBest->m_rank - index.m_rank)) ||
+ (k = (indexBest->m_keyCountUnused - index.m_keyCountUnused));
+ if (k > 0)
+ indexBest = &index;
+ }
+ if (indexBest != 0) {
+ const bool exactKey = indexBest->m_rank <= 1 ? m_table->exactKey(ctx, indexBest) : false;
+ const bool direct = ! ctl.m_extra && exactKey;
+ ctx_log3(("delete direct=%d: extra=%d exact=%d", direct, ctl.m_extra, exactKey));
+ if (indexBest->m_rank == 0) {
+ // primary key
+ Plan_delete_lookup* deleteLookup = new Plan_delete_lookup(m_root);
+ m_root->saveNode(deleteLookup);
+ deleteLookup->setTable(m_table);
+ if (direct) {
+ // key match with no extra conditions
+ Plan_query_repeat* queryRepeat = new Plan_query_repeat(m_root, 1);
+ m_root->saveNode(queryRepeat);
+ deleteLookup->setQuery(queryRepeat);
+ } else {
+ // key match with extra conditions
+ Plan_query_lookup* queryLookup = new Plan_query_lookup(m_root);
+ m_root->saveNode(queryLookup);
+ Plan_query_filter* queryFilter = new Plan_query_filter(m_root);
+ m_root->saveNode(queryFilter);
+ queryLookup->setTable(m_table);
+ queryFilter->setQuery(queryLookup);
+ queryFilter->setPred(m_pred);
+ queryFilter->m_topTable = m_table;
+ deleteLookup->setQuery(queryFilter);
+ }
+ stmt = deleteLookup;
+ } else if (indexBest->m_rank == 1) {
+ // hash index
+ Plan_delete_index* deleteIndex = new Plan_delete_index(m_root);
+ m_root->saveNode(deleteIndex);
+ deleteIndex->setTable(m_table, indexBest);
+ if (direct) {
+ // key match with no extra conditions
+ Plan_query_repeat* queryRepeat = new Plan_query_repeat(m_root, 1);
+ m_root->saveNode(queryRepeat);
+ deleteIndex->setQuery(queryRepeat);
+ } else {
+ // key match with extra conditions
+ Plan_query_index* queryIndex = new Plan_query_index(m_root);
+ m_root->saveNode(queryIndex);
+ Plan_query_filter* queryFilter = new Plan_query_filter(m_root);
+ m_root->saveNode(queryFilter);
+ queryIndex->setTable(m_table, indexBest);
+ queryFilter->setQuery(queryIndex);
+ queryFilter->setPred(m_pred);
+ queryFilter->m_topTable = m_table;
+ deleteIndex->setQuery(queryFilter);
+ }
+ stmt = deleteIndex;
+ } else if (indexBest->m_rank == 2) {
+ // ordered index
+ Plan_delete_scan* deleteScan = new Plan_delete_scan(m_root);
+ m_root->saveNode(deleteScan);
+ Plan_query_filter* queryFilter = new Plan_query_filter(m_root);
+ m_root->saveNode(queryFilter);
+ Plan_query_range* queryRange = new Plan_query_range(m_root);
+ m_root->saveNode(queryRange);
+ queryRange->setTable(m_table, indexBest);
+ queryRange->setExclusive();
+ queryFilter->setQuery(queryRange);
+ queryFilter->setPred(m_pred);
+ queryFilter->m_topTable = m_table;
+ const TableSet& ts2 = m_pred->noInterp();
+ ctx_assert(ts2.size() <= 1);
+ if (ts2.size() == 0) {
+ queryRange->setInterp(m_pred);
+ }
+ deleteScan->setQuery(queryFilter);
+ stmt = deleteScan;
+ } else {
+ ctx_assert(false);
+ }
+ } else {
+ // scan delete with filter
+ Plan_delete_scan* deleteScan = new Plan_delete_scan(m_root);
+ m_root->saveNode(deleteScan);
+ Plan_query_filter* queryFilter = new Plan_query_filter(m_root);
+ m_root->saveNode(queryFilter);
+ Plan_query_scan* queryScan = new Plan_query_scan(m_root);
+ m_root->saveNode(queryScan);
+ queryScan->setTable(m_table);
+ queryScan->setExclusive();
+ queryFilter->setQuery(queryScan);
+ queryFilter->setPred(m_pred);
+ queryFilter->m_topTable = m_table;
+ // interpeter
+ const TableSet& ts2 = m_pred->noInterp();
+ ctx_assert(ts2.size() <= 1);
+ if (ts2.size() == 0) {
+ queryScan->setInterp(m_pred);
+ }
+ deleteScan->setQuery(queryFilter);
+ stmt = deleteScan;
+ }
+ } else {
+ // scan delete without filter
+ Plan_delete_scan* deleteScan = new Plan_delete_scan(m_root);
+ m_root->saveNode(deleteScan);
+ Plan_query_scan* queryScan = new Plan_query_scan(m_root);
+ m_root->saveNode(queryScan);
+ queryScan->setTable(m_table);
+ queryScan->setExclusive();
+ deleteScan->setQuery(queryScan);
+ stmt = deleteScan;
+ }
+ // set base for column position offsets
+ m_table->m_resOff = 1;
+ return stmt;
+}
+
+void
+Plan_delete::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "DELETE WHERE", SQL_DIAG_DELETE_WHERE);
+}
+
+Exec_base*
+Plan_delete::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_delete::print(Ctx& ctx)
+{
+ ctx.print(" [delete");
+ Plan_base* a[] = { m_table, m_pred };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_delete.hpp b/ndb/src/client/odbc/codegen/Code_delete.hpp
new file mode 100644
index 00000000000..c7fa245497b
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_delete.hpp
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_delete_hpp
+#define ODBC_CODEGEN_Code_delete_hpp
+
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_dml.hpp"
+#include "Code_table.hpp"
+#include "Code_query.hpp"
+#include "Code_pred.hpp"
+
+/**
+ * @class Plan_delete
+ * @brief Delete in PlanTree
+ */
+class Plan_delete : public Plan_dml {
+public:
+ Plan_delete(Plan_root* root);
+ virtual ~Plan_delete();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setTable(Plan_table* table);
+ void setPred(Plan_pred* pred);
+protected:
+ Plan_table* m_table;
+ Plan_pred* m_pred;
+};
+
+inline
+Plan_delete::Plan_delete(Plan_root* root) :
+ Plan_dml(root),
+ m_table(0),
+ m_pred(0)
+{
+}
+
+inline void
+Plan_delete::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+inline void
+Plan_delete::setPred(Plan_pred* pred)
+{
+ ctx_assert(pred != 0);
+ m_pred = pred;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_delete_index.cpp b/ndb/src/client/odbc/codegen/Code_delete_index.cpp
new file mode 100644
index 00000000000..8f2c3be2848
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_delete_index.cpp
@@ -0,0 +1,164 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include "Code_expr.hpp"
+#include "Code_delete_index.hpp"
+#include "Code_table.hpp"
+#include "Code_root.hpp"
+
+Plan_delete_index::~Plan_delete_index()
+{
+}
+
+Plan_base*
+Plan_delete_index::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+void
+Plan_delete_index::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "DELETE WHERE", SQL_DIAG_DELETE_WHERE);
+}
+
+Exec_base*
+Plan_delete_index::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // create code for the query
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // set up
+ ctx_assert(m_table != 0 && m_index != 0);
+ const BaseString& tableName = m_table->getName();
+ ctx_assert(m_index->m_dictIndex != 0);
+ const DictIndex& dictIndex = *m_index->m_dictIndex;
+ const BaseString& indexName = dictIndex.getName();
+ const unsigned keyCount = m_index->m_keyCount;
+ // create the code
+ Exec_delete_index::Code& code = *new Exec_delete_index::Code(keyCount);
+ code.m_tableName = strcpy(new char[tableName.length() + 1], tableName.c_str());
+ code.m_indexName = strcpy(new char[indexName.length() + 1], indexName.c_str());
+ // key attributes
+ code.m_keyId = new NdbAttrId[1 + keyCount];
+ code.m_keyId[0] = (NdbAttrId)-1;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ const DictColumn* keyColumn = dictIndex.getColumn(k);
+ const SqlType& sqlType = keyColumn->sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_keySpecs.setEntry(k, sqlSpec);
+ code.m_keyId[k] = k - 1; // index column order
+ }
+ // matching expressions
+ ctx_assert(m_index->m_keyFound);
+ const ExprVector& keyEq = m_index->m_keyEq;
+ ctx_assert(keyEq.size() == 1 + keyCount);
+ code.m_keyMatch = new Exec_expr* [1 + keyCount];
+ code.m_keyMatch[0] = 0;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ Plan_expr* expr = keyEq[k];
+ Exec_expr* execExpr = static_cast<Exec_expr*>(expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ code.m_keyMatch[k] = execExpr;
+ }
+ // create the exec
+ Exec_delete_index* exec = new Exec_delete_index(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ return exec;
+}
+
+void
+Plan_delete_index::print(Ctx& ctx)
+{
+ ctx.print(" [delete_index");
+ Plan_base* a[] = { m_query, m_table };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
+
+// Exec_delete_index
+
+Exec_delete_index::Code::~Code()
+{
+ delete[] m_tableName;
+ delete[] m_keyId;
+ delete[] m_keyMatch;
+}
+
+Exec_delete_index::Data::~Data()
+{
+}
+
+Exec_delete_index::~Exec_delete_index()
+{
+}
+
+void
+Exec_delete_index::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // allocate the subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // allocate matching expressions
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* expr = code.m_keyMatch[k];
+ ctx_assert(expr != 0);
+ expr->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ // create data
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_delete_index::close(Ctx& ctx)
+{
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+}
+
+void
+Exec_delete_index::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [delete_index");
+ Exec_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ printList(ctx, (Exec_base**)&code.m_keyMatch[1], code.m_keyCount);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_delete_index.hpp b/ndb/src/client/odbc/codegen/Code_delete_index.hpp
new file mode 100644
index 00000000000..1aaaa18abcb
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_delete_index.hpp
@@ -0,0 +1,156 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_delete_index_hpp
+#define ODBC_CODEGEN_Code_delete_index_hpp
+
+#include <common/common.hpp>
+#include "Code_dml.hpp"
+#include "Code_query.hpp"
+#include "Code_table.hpp"
+
+/**
+ * @class Plan_delete_index
+ * @brief Delete by primary key
+ */
+class Plan_delete_index : public Plan_dml {
+public:
+ Plan_delete_index(Plan_root* root);
+ virtual ~Plan_delete_index();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setQuery(Plan_query* query);
+ void setTable(Plan_table* table, Plan_table::Index* index);
+protected:
+ Plan_query* m_query;
+ Plan_table* m_table;
+ Plan_table::Index* m_index;
+};
+
+inline
+Plan_delete_index::Plan_delete_index(Plan_root* root) :
+ Plan_dml(root),
+ m_query(0),
+ m_table(0)
+{
+}
+
+inline void
+Plan_delete_index::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Plan_delete_index::setTable(Plan_table* table, Plan_table::Index* index)
+{
+ ctx_assert(table != 0 && index != 0 && index == &table->m_indexList[index->m_pos] && index->m_pos != 0);
+ m_table = table;
+ m_index = index;
+}
+
+/**
+ * @class Exec_delete_index
+ * @brief Delete by primary key
+ */
+class Exec_delete_index : public Exec_dml {
+public:
+ class Code : public Exec_dml::Code {
+ public:
+ Code(unsigned keyCount);
+ virtual ~Code();
+ protected:
+ friend class Plan_delete_index;
+ friend class Exec_delete_index;
+ const char* m_tableName;
+ const char* m_indexName;
+ unsigned m_keyCount;
+ SqlSpecs m_keySpecs; // key types
+ NdbAttrId* m_keyId;
+ Exec_expr** m_keyMatch; // XXX pointers for now
+ };
+ class Data : public Exec_dml::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_delete_index;
+ };
+ Exec_delete_index(Exec_root* root);
+ virtual ~Exec_delete_index();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+protected:
+ Exec_query* m_query;
+};
+
+inline
+Exec_delete_index::Code::Code(unsigned keyCount) :
+ m_tableName(0),
+ m_indexName(0),
+ m_keyCount(keyCount),
+ m_keySpecs(keyCount),
+ m_keyId(0),
+ m_keyMatch(0)
+{
+}
+
+inline
+Exec_delete_index::Data::Data()
+{
+}
+
+inline
+Exec_delete_index::Exec_delete_index(Exec_root* root) :
+ Exec_dml(root),
+ m_query(0)
+{
+}
+
+// children
+
+inline const Exec_delete_index::Code&
+Exec_delete_index::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_delete_index::Data&
+Exec_delete_index::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_delete_index::setQuery(Exec_query* query)
+{
+ ctx_assert(query != 0 && m_query == 0);
+ m_query = query;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_delete_lookup.cpp b/ndb/src/client/odbc/codegen/Code_delete_lookup.cpp
new file mode 100644
index 00000000000..4a6dec64654
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_delete_lookup.cpp
@@ -0,0 +1,162 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include "Code_expr.hpp"
+#include "Code_delete_lookup.hpp"
+#include "Code_table.hpp"
+#include "Code_root.hpp"
+
+Plan_delete_lookup::~Plan_delete_lookup()
+{
+}
+
+Plan_base*
+Plan_delete_lookup::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+void
+Plan_delete_lookup::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "DELETE WHERE", SQL_DIAG_DELETE_WHERE);
+}
+
+Exec_base*
+Plan_delete_lookup::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // create code for the query
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // set up
+ ctx_assert(m_table != 0);
+ const BaseString& tableName = m_table->getName();
+ const DictTable& dictTable = m_table->dictTable();
+ const unsigned keyCount = dictTable.keyCount();
+ // create the code
+ Exec_delete_lookup::Code& code = *new Exec_delete_lookup::Code(keyCount);
+ code.m_tableName = strcpy(new char[tableName.length() + 1], tableName.c_str());
+ // key attributes
+ code.m_keyId = new NdbAttrId[1 + keyCount];
+ code.m_keyId[0] = (NdbAttrId)-1;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ const DictColumn* keyColumn = dictTable.getKey(k);
+ const SqlType& sqlType = keyColumn->sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_keySpecs.setEntry(k, sqlSpec);
+ code.m_keyId[k] = keyColumn->getAttrId();
+ }
+ // matching expressions
+ const Plan_table::Index& index = m_table->m_indexList[0];
+ ctx_assert(index.m_keyFound);
+ const ExprVector& keyEq = index.m_keyEq;
+ ctx_assert(keyEq.size() == 1 + keyCount);
+ code.m_keyMatch = new Exec_expr* [1 + keyCount];
+ code.m_keyMatch[0] = 0;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ Plan_expr* expr = keyEq[k];
+ Exec_expr* execExpr = static_cast<Exec_expr*>(expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ code.m_keyMatch[k] = execExpr;
+ }
+ // create the exec
+ Exec_delete_lookup* exec = new Exec_delete_lookup(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ return exec;
+}
+
+void
+Plan_delete_lookup::print(Ctx& ctx)
+{
+ ctx.print(" [delete_lookup");
+ Plan_base* a[] = { m_query, m_table };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
+
+// Exec_delete_lookup
+
+Exec_delete_lookup::Code::~Code()
+{
+ delete[] m_tableName;
+ delete[] m_keyId;
+ delete[] m_keyMatch;
+}
+
+Exec_delete_lookup::Data::~Data()
+{
+}
+
+Exec_delete_lookup::~Exec_delete_lookup()
+{
+}
+
+void
+Exec_delete_lookup::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // allocate the subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // allocate matching expressions
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* expr = code.m_keyMatch[k];
+ ctx_assert(expr != 0);
+ expr->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ // create data
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_delete_lookup::close(Ctx& ctx)
+{
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+}
+
+void
+Exec_delete_lookup::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [delete_lookup");
+ Exec_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ printList(ctx, (Exec_base**)&code.m_keyMatch[1], code.m_keyCount);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_delete_lookup.hpp b/ndb/src/client/odbc/codegen/Code_delete_lookup.hpp
new file mode 100644
index 00000000000..4138baefa4c
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_delete_lookup.hpp
@@ -0,0 +1,152 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_delete_lookup_hpp
+#define ODBC_CODEGEN_Code_delete_lookup_hpp
+
+#include <common/common.hpp>
+#include "Code_dml.hpp"
+#include "Code_query.hpp"
+#include "Code_table.hpp"
+
+/**
+ * @class Plan_delete_lookup
+ * @brief Delete by primary key
+ */
+class Plan_delete_lookup : public Plan_dml {
+public:
+ Plan_delete_lookup(Plan_root* root);
+ virtual ~Plan_delete_lookup();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setQuery(Plan_query* query);
+ void setTable(Plan_table* table);
+protected:
+ Plan_query* m_query;
+ Plan_table* m_table;
+};
+
+inline
+Plan_delete_lookup::Plan_delete_lookup(Plan_root* root) :
+ Plan_dml(root),
+ m_query(0),
+ m_table(0)
+{
+}
+
+inline void
+Plan_delete_lookup::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Plan_delete_lookup::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+/**
+ * @class Exec_delete_lookup
+ * @brief Delete by primary key
+ */
+class Exec_delete_lookup : public Exec_dml {
+public:
+ class Code : public Exec_dml::Code {
+ public:
+ Code(unsigned keyCount);
+ virtual ~Code();
+ protected:
+ friend class Plan_delete_lookup;
+ friend class Exec_delete_lookup;
+ char* m_tableName;
+ unsigned m_keyCount;
+ SqlSpecs m_keySpecs; // key types
+ NdbAttrId* m_keyId;
+ Exec_expr** m_keyMatch; // XXX pointers for now
+ };
+ class Data : public Exec_dml::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_delete_lookup;
+ };
+ Exec_delete_lookup(Exec_root* root);
+ virtual ~Exec_delete_lookup();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+protected:
+ Exec_query* m_query;
+};
+
+inline
+Exec_delete_lookup::Code::Code(unsigned keyCount) :
+ m_tableName(0),
+ m_keyCount(keyCount),
+ m_keySpecs(keyCount),
+ m_keyId(0),
+ m_keyMatch(0)
+{
+}
+
+inline
+Exec_delete_lookup::Data::Data()
+{
+}
+
+inline
+Exec_delete_lookup::Exec_delete_lookup(Exec_root* root) :
+ Exec_dml(root),
+ m_query(0)
+{
+}
+
+// children
+
+inline const Exec_delete_lookup::Code&
+Exec_delete_lookup::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_delete_lookup::Data&
+Exec_delete_lookup::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_delete_lookup::setQuery(Exec_query* query)
+{
+ ctx_assert(query != 0 && m_query == 0);
+ m_query = query;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_delete_scan.cpp b/ndb/src/client/odbc/codegen/Code_delete_scan.cpp
new file mode 100644
index 00000000000..fed7244a026
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_delete_scan.cpp
@@ -0,0 +1,110 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_delete_scan.hpp"
+#include "Code_root.hpp"
+
+Plan_delete_scan::~Plan_delete_scan()
+{
+}
+
+Plan_base*
+Plan_delete_scan::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+void
+Plan_delete_scan::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "DELETE WHERE", SQL_DIAG_DELETE_WHERE);
+}
+
+Exec_base*
+Plan_delete_scan::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // create code for the subquery
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // create the code
+ Exec_delete_scan* exec = new Exec_delete_scan(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ Exec_delete_scan::Code& code = *new Exec_delete_scan::Code;
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ return exec;
+}
+
+void
+Plan_delete_scan::print(Ctx& ctx)
+{
+ ctx.print(" [delete_scan");
+ Plan_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
+
+// Exec_delete_scan
+
+Exec_delete_scan::Code::~Code()
+{
+}
+
+Exec_delete_scan::Data::~Data()
+{
+}
+
+Exec_delete_scan::~Exec_delete_scan()
+{
+}
+
+void
+Exec_delete_scan::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate the subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // create data
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_delete_scan::close(Ctx& ctx)
+{
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+}
+
+void
+Exec_delete_scan::print(Ctx& ctx)
+{
+ ctx.print(" [delete_scan");
+ Exec_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
+
diff --git a/ndb/src/client/odbc/codegen/Code_delete_scan.hpp b/ndb/src/client/odbc/codegen/Code_delete_scan.hpp
new file mode 100644
index 00000000000..eb013a8257e
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_delete_scan.hpp
@@ -0,0 +1,130 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_delete_scan_hpp
+#define ODBC_CODEGEN_Code_delete_scan_hpp
+
+#include <common/common.hpp>
+#include "Code_dml.hpp"
+#include "Code_query.hpp"
+
+/**
+ * @class Plan_delete_scan
+ * @brief Scan delete
+ */
+class Plan_delete_scan : public Plan_dml {
+public:
+ Plan_delete_scan(Plan_root* root);
+ virtual ~Plan_delete_scan();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setQuery(Plan_query* query);
+protected:
+ Plan_query* m_query;
+};
+
+inline
+Plan_delete_scan::Plan_delete_scan(Plan_root* root) :
+ Plan_dml(root),
+ m_query(0)
+{
+}
+
+inline void
+Plan_delete_scan::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+/**
+ * @class Exec_delete_scan
+ * @brief Scan delete
+ */
+class Exec_delete_scan : public Exec_dml {
+public:
+ class Code : public Exec_dml::Code {
+ public:
+ Code();
+ virtual ~Code();
+ protected:
+ friend class Exec_delete_scan;
+ };
+ class Data : public Exec_dml::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_delete_scan;
+ };
+ Exec_delete_scan(Exec_root* root);
+ virtual ~Exec_delete_scan();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+protected:
+ Exec_query* m_query;
+};
+
+inline
+Exec_delete_scan::Code::Code()
+{
+}
+
+inline
+Exec_delete_scan::Data::Data()
+{
+}
+
+inline
+Exec_delete_scan::Exec_delete_scan(Exec_root* root) :
+ Exec_dml(root),
+ m_query(0)
+{
+}
+
+// children
+
+inline const Exec_delete_scan::Code&
+Exec_delete_scan::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_delete_scan::Data&
+Exec_delete_scan::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_delete_scan::setQuery(Exec_query* query)
+{
+ ctx_assert(query != 0 && m_query == 0);
+ m_query = query;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_dml.cpp b/ndb/src/client/odbc/codegen/Code_dml.cpp
new file mode 100644
index 00000000000..44fd4478646
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_dml.cpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_dml.hpp"
+
+// Plan_dml
+
+Plan_dml::~Plan_dml()
+{
+}
+
+// Exec_dml
+
+Exec_dml::Code::~Code()
+{
+}
+
+Exec_dml::Data::~Data()
+{
+}
+
+Exec_dml::~Exec_dml()
+{
+}
+
+void
+Exec_dml::execute(Ctx& ctx, Ctl& ctl)
+{
+ execImpl(ctx, ctl);
+ if (m_topLevel) {
+ if (ctx.ok()) {
+ if (stmtArea().getRowCount() == 0) {
+ ctx.setCode(SQL_NO_DATA);
+ }
+ }
+ }
+}
diff --git a/ndb/src/client/odbc/codegen/Code_dml.hpp b/ndb/src/client/odbc/codegen/Code_dml.hpp
new file mode 100644
index 00000000000..0618f583984
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_dml.hpp
@@ -0,0 +1,67 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_dml_hpp
+#define ODBC_CODEGEN_Code_dml_hpp
+
+#include <common/common.hpp>
+#include <common/ResultArea.hpp>
+#include "Code_stmt.hpp"
+
+/**
+ * @class Plan_dml
+ * @brief Base class for DML statements in PlanTree
+ */
+class Plan_dml : public Plan_stmt {
+public:
+ Plan_dml(Plan_root* root);
+ virtual ~Plan_dml() = 0;
+};
+
+inline
+Plan_dml::Plan_dml(Plan_root* root) :
+ Plan_stmt(root)
+{
+}
+
+/**
+ * @class Exec_dml
+ * @brief Base class for DML statements in ExecTree
+ */
+class Exec_dml : public Exec_stmt {
+public:
+ class Code : public Exec_stmt::Code {
+ public:
+ virtual ~Code() = 0;
+ };
+ class Data : public Exec_stmt::Data, public ResultArea {
+ public:
+ virtual ~Data() = 0;
+ };
+ Exec_dml(Exec_root* root);
+ virtual ~Exec_dml() = 0;
+ void execute(Ctx& ctx, Ctl& ctl);
+protected:
+ virtual void execImpl(Ctx& ctx, Ctl& ctl) = 0;
+};
+
+inline
+Exec_dml::Exec_dml(Exec_root* root) :
+ Exec_stmt(root)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_dml_column.cpp b/ndb/src/client/odbc/codegen/Code_dml_column.cpp
new file mode 100644
index 00000000000..808e2ac8c4b
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_dml_column.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include "Code_dml_column.hpp"
+
+// Plan_dml_column
+
+Plan_dml_column::~Plan_dml_column()
+{
+}
+
+Plan_base*
+Plan_dml_column::analyze(Ctx& ctx, Ctl& ctl)
+{
+ analyzeColumn(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_dml_column::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_dml_column::print(Ctx& ctx)
+{
+ ctx.print(" [dml_column %s]", getPrintName());
+}
diff --git a/ndb/src/client/odbc/codegen/Code_dml_column.hpp b/ndb/src/client/odbc/codegen/Code_dml_column.hpp
new file mode 100644
index 00000000000..0fb33944a3a
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_dml_column.hpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_dml_column_hpp
+#define ODBC_CODEGEN_Code_dml_column_hpp
+
+#include <common/common.hpp>
+#include "Code_column.hpp"
+
+class DictColumn;
+class Plan_table;
+
+/**
+ * @class Plan_dml_column
+ * @brief Column in query expression
+ */
+class Plan_dml_column : public Plan_base, public Plan_column {
+public:
+ Plan_dml_column(Plan_root* root, const BaseString& name);
+ virtual ~Plan_dml_column();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+};
+
+inline
+Plan_dml_column::Plan_dml_column(Plan_root* root, const BaseString& name) :
+ Plan_base(root),
+ Plan_column(Type_dml, name)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_dml_row.cpp b/ndb/src/client/odbc/codegen/Code_dml_row.cpp
new file mode 100644
index 00000000000..ceb63a9f7b9
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_dml_row.cpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_dml_row.hpp"
+#include "Code_dml_column.hpp"
+
+Plan_dml_row::~Plan_dml_row()
+{
+}
+
+Plan_base*
+Plan_dml_row::analyze(Ctx& ctx, Ctl& ctl)
+{
+ unsigned size = getSize();
+ // analyze the columns
+ for (unsigned i = 1; i <= size; i++) {
+ Plan_dml_column* column = getColumn(i);
+ column->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ }
+ // node was not replaced
+ return this;
+}
+
+Exec_base*
+Plan_dml_row::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_dml_row::print(Ctx& ctx)
+{
+ unsigned size = getSize();
+ ctx.print(" [dml_row");
+ for (unsigned i = 1; i <= size; i++) {
+ Plan_base* a = m_columnList[i];
+ a == 0 ? ctx.print(" -") : a->print(ctx);
+ }
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_dml_row.hpp b/ndb/src/client/odbc/codegen/Code_dml_row.hpp
new file mode 100644
index 00000000000..6c7e46ba9af
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_dml_row.hpp
@@ -0,0 +1,76 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_dml_row_hpp
+#define ODBC_CODEGEN_Code_dml_row_hpp
+
+#include <vector>
+#include <common/common.hpp>
+#include <common/DataRow.hpp>
+#include "Code_base.hpp"
+#include "Code_dml_column.hpp"
+
+class Plan_dml_column;
+
+/**
+ * @class Plan_dml_row
+ * @brief Row of lvalue columns in insert or update
+ */
+class Plan_dml_row : public Plan_base {
+public:
+ Plan_dml_row(Plan_root* root);
+ virtual ~Plan_dml_row();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ unsigned getSize() const;
+ void addColumn(Plan_dml_column* column);
+ Plan_dml_column* getColumn(unsigned i) const;
+protected:
+ DmlColumnVector m_columnList;
+};
+
+inline
+Plan_dml_row::Plan_dml_row(Plan_root* root) :
+ Plan_base(root),
+ m_columnList(1)
+{
+}
+
+// children
+
+inline unsigned
+Plan_dml_row::getSize() const
+{
+ return m_columnList.size() - 1;
+}
+
+inline void
+Plan_dml_row::addColumn(Plan_dml_column* column)
+{
+ ctx_assert(column != 0);
+ m_columnList.push_back(column);
+}
+
+inline Plan_dml_column*
+Plan_dml_row::getColumn(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= m_columnList.size() && m_columnList[i] != 0);
+ return m_columnList[i];
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_drop_index.cpp b/ndb/src/client/odbc/codegen/Code_drop_index.cpp
new file mode 100644
index 00000000000..b6bae88e270
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_drop_index.cpp
@@ -0,0 +1,87 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_drop_index.hpp"
+#include "Code_root.hpp"
+
+// Plan_drop_index
+
+Plan_drop_index::~Plan_drop_index()
+{
+}
+
+Plan_base*
+Plan_drop_index::analyze(Ctx& ctx, Ctl& ctl)
+{
+ stmtArea().stmtInfo().setName(Stmt_name_drop_index);
+ return this;
+}
+
+void
+Plan_drop_index::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "DROP INDEX", SQL_DIAG_DROP_INDEX);
+}
+
+Exec_base*
+Plan_drop_index::codegen(Ctx& ctx, Ctl& ctl)
+{
+ Exec_drop_index* exec = new Exec_drop_index(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ Exec_drop_index::Code& code = *new Exec_drop_index::Code(m_name, m_tableName);
+ exec->setCode(code);
+ return exec;
+}
+
+void
+Plan_drop_index::print(Ctx& ctx)
+{
+ ctx.print(" [drop_index %s]", m_name.c_str());
+}
+
+// Exec_drop_index
+
+Exec_drop_index::Code::~Code()
+{
+}
+
+Exec_drop_index::Data::~Data()
+{
+}
+
+Exec_drop_index::~Exec_drop_index()
+{
+}
+
+void
+Exec_drop_index::alloc(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_drop_index::close(Ctx& ctx)
+{
+}
+
+void
+Exec_drop_index::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [drop_index %s]", code.m_indexName.c_str());
+}
diff --git a/ndb/src/client/odbc/codegen/Code_drop_index.hpp b/ndb/src/client/odbc/codegen/Code_drop_index.hpp
new file mode 100644
index 00000000000..99891c9a52f
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_drop_index.hpp
@@ -0,0 +1,136 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_drop_index_hpp
+#define ODBC_CODEGEN_Code_drop_index_hpp
+
+#include <vector>
+#include <NdbApi.hpp>
+#include <common/common.hpp>
+#include "Code_ddl.hpp"
+
+class DictTable;
+class DictColumn;
+
+/**
+ * @class Plan_drop_index
+ * @brief Drop index in PlanTree
+ */
+class Plan_drop_index : public Plan_ddl {
+public:
+ Plan_drop_index(Plan_root* root, const BaseString& name);
+ Plan_drop_index(Plan_root* root, const BaseString& name, const BaseString& tableName);
+ virtual ~Plan_drop_index();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx & ctx);
+ void print(Ctx& ctx);
+ // attributes
+ const BaseString& getName() const;
+protected:
+ BaseString m_name;
+ BaseString m_tableName;
+};
+
+inline
+Plan_drop_index::Plan_drop_index(Plan_root* root, const BaseString& name) :
+ Plan_ddl(root),
+ m_name(name)
+{
+}
+
+inline
+Plan_drop_index::Plan_drop_index(Plan_root* root, const BaseString& name, const BaseString& tableName) :
+ Plan_ddl(root),
+ m_name(name),
+ m_tableName(tableName)
+{
+}
+
+inline const BaseString&
+Plan_drop_index::getName() const
+{
+ return m_name;
+}
+
+/**
+ * @class Exec_drop_index
+ * @brief Drop index in ExecTree
+ */
+class Exec_drop_index : public Exec_ddl {
+public:
+ class Code : public Exec_ddl::Code {
+ public:
+ Code(const BaseString& indexName, const BaseString& tableName);
+ virtual ~Code();
+ protected:
+ friend class Exec_drop_index;
+ const BaseString m_indexName;
+ const BaseString m_tableName;
+ };
+ class Data : public Exec_ddl::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_drop_index;
+ };
+ Exec_drop_index(Exec_root* root);
+ virtual ~Exec_drop_index();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execute(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_drop_index::Code::Code(const BaseString& indexName, const BaseString& tableName) :
+ m_indexName(indexName),
+ m_tableName(tableName)
+{
+}
+
+inline
+Exec_drop_index::Data::Data()
+{
+}
+
+inline
+Exec_drop_index::Exec_drop_index(Exec_root* root) :
+ Exec_ddl(root)
+{
+}
+
+// children
+
+inline const Exec_drop_index::Code&
+Exec_drop_index::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_drop_index::Data&
+Exec_drop_index::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_drop_table.cpp b/ndb/src/client/odbc/codegen/Code_drop_table.cpp
new file mode 100644
index 00000000000..f20bf9fdae0
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_drop_table.cpp
@@ -0,0 +1,87 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_drop_table.hpp"
+#include "Code_root.hpp"
+
+// Plan_drop_table
+
+Plan_drop_table::~Plan_drop_table()
+{
+}
+
+Plan_base*
+Plan_drop_table::analyze(Ctx& ctx, Ctl& ctl)
+{
+ stmtArea().stmtInfo().setName(Stmt_name_drop_table);
+ return this;
+}
+
+void
+Plan_drop_table::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "DROP TABLE", SQL_DIAG_DROP_TABLE);
+}
+
+Exec_base*
+Plan_drop_table::codegen(Ctx& ctx, Ctl& ctl)
+{
+ Exec_drop_table* exec = new Exec_drop_table(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ Exec_drop_table::Code& code = *new Exec_drop_table::Code(m_name);
+ exec->setCode(code);
+ return exec;
+}
+
+void
+Plan_drop_table::print(Ctx& ctx)
+{
+ ctx.print(" [drop_table %s]", m_name.c_str());
+}
+
+// Exec_drop_table
+
+Exec_drop_table::Code::~Code()
+{
+}
+
+Exec_drop_table::Data::~Data()
+{
+}
+
+Exec_drop_table::~Exec_drop_table()
+{
+}
+
+void
+Exec_drop_table::alloc(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_drop_table::close(Ctx& ctx)
+{
+}
+
+void
+Exec_drop_table::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [drop_table %s]", code.m_tableName.c_str());
+}
diff --git a/ndb/src/client/odbc/codegen/Code_drop_table.hpp b/ndb/src/client/odbc/codegen/Code_drop_table.hpp
new file mode 100644
index 00000000000..849a472ed94
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_drop_table.hpp
@@ -0,0 +1,124 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_drop_table_hpp
+#define ODBC_CODEGEN_Code_drop_table_hpp
+
+#include <vector>
+#include <NdbApi.hpp>
+#include <common/common.hpp>
+#include "Code_ddl.hpp"
+
+class DictTable;
+class DictColumn;
+
+/**
+ * @class Plan_drop_table
+ * @brief Drop table in PlanTree
+ */
+class Plan_drop_table : public Plan_ddl {
+public:
+ Plan_drop_table(Plan_root* root, const BaseString& name);
+ virtual ~Plan_drop_table();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx & ctx);
+ void print(Ctx& ctx);
+ // attributes
+ const BaseString& getName() const;
+protected:
+ BaseString m_name;
+};
+
+inline
+Plan_drop_table::Plan_drop_table(Plan_root* root, const BaseString& name) :
+ Plan_ddl(root),
+ m_name(name)
+{
+}
+
+inline const BaseString&
+Plan_drop_table::getName() const
+{
+ return m_name;
+}
+
+/**
+ * @class Exec_drop_table
+ * @brief Drop table in ExecTree
+ */
+class Exec_drop_table : public Exec_ddl {
+public:
+ class Code : public Exec_ddl::Code {
+ public:
+ Code(const BaseString& tableName);
+ virtual ~Code();
+ protected:
+ friend class Exec_drop_table;
+ const BaseString m_tableName;
+ };
+ class Data : public Exec_ddl::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_drop_table;
+ };
+ Exec_drop_table(Exec_root* root);
+ virtual ~Exec_drop_table();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execute(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_drop_table::Code::Code(const BaseString& tableName) :
+ m_tableName(tableName)
+{
+}
+
+inline
+Exec_drop_table::Data::Data()
+{
+}
+
+inline
+Exec_drop_table::Exec_drop_table(Exec_root* root) :
+ Exec_ddl(root)
+{
+}
+
+// children
+
+inline const Exec_drop_table::Code&
+Exec_drop_table::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_drop_table::Data&
+Exec_drop_table::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_expr.cpp b/ndb/src/client/odbc/codegen/Code_expr.cpp
new file mode 100644
index 00000000000..4afa75986a0
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr.cpp
@@ -0,0 +1,79 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_expr.hpp"
+#include "Code_expr_row.hpp"
+
+// Plan_expr
+
+Plan_expr::~Plan_expr()
+{
+}
+
+bool
+Plan_expr::isEqual(const Plan_expr* expr) const
+{
+ return false;
+}
+
+bool
+Plan_expr::isAnyEqual(const Plan_expr_row* row) const
+{
+ ctx_assert(row != 0);
+ const unsigned size = row->getSize();
+ for (unsigned i = 1; i <= size; i++) {
+ if (isEqual(row->getExpr(i)))
+ return true;
+ }
+ return false;
+}
+
+bool
+Plan_expr::isGroupBy(const Plan_expr_row* row) const
+{
+ return false;
+}
+
+// Exec_expr
+
+Exec_expr::Code::~Code()
+{
+}
+
+Exec_expr::Data::~Data()
+{
+}
+
+Exec_expr::~Exec_expr()
+{
+}
+
+SqlField&
+Exec_expr::Data::groupField(const SqlType& sqlType, unsigned i, bool initFlag)
+{
+ if (m_groupField.size() == 0) {
+ m_groupField.resize(1);
+ }
+ if (initFlag) {
+ //unsigned i2 = m_groupField.size();
+ //ctx_assert(i == i2);
+ const SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ const SqlField sqlField(sqlSpec);
+ m_groupField.push_back(sqlField);
+ }
+ ctx_assert(i != 0 && i < m_groupField.size());
+ return m_groupField[i];
+}
diff --git a/ndb/src/client/odbc/codegen/Code_expr.hpp b/ndb/src/client/odbc/codegen/Code_expr.hpp
new file mode 100644
index 00000000000..b6f07471b4d
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr.hpp
@@ -0,0 +1,219 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_expr_hpp
+#define ODBC_CODEGEN_Code_expr_hpp
+
+#include <common/common.hpp>
+#include <common/DataField.hpp>
+#include "Code_base.hpp"
+
+class Ctx;
+class Plan_expr_row;
+class Exec_expr;
+
+/**
+ * @class Plan_expr
+ * @brief Base class for expressions in PlanTree
+ */
+class Plan_expr : public Plan_base {
+public:
+ // type is convenient since RTTI cannot be used
+ enum Type {
+ TypeUndefined = 0,
+ TypeColumn,
+ TypeConst,
+ TypeConv,
+ TypeFunc,
+ TypeOp,
+ TypeParam,
+ TypeValue
+ };
+ Plan_expr(Plan_root* root, Type type);
+ virtual ~Plan_expr() = 0;
+ Type type() const;
+ const SqlType& sqlType() const; // data type set by analyze
+ const TableSet& tableSet() const;
+ const BaseString& getAlias() const;
+ bool isAggr() const;
+ bool isBound() const;
+ bool isAnyEqual(const Plan_expr_row* row) const;
+ virtual bool isEqual(const Plan_expr* expr) const;
+ virtual bool isGroupBy(const Plan_expr_row* row) const;
+protected:
+ friend class Plan_expr_row;
+ friend class Plan_expr_op;
+ friend class Plan_expr_func;
+ friend class Plan_comp_op;
+ const Type m_type;
+ SqlType m_sqlType; // subclass must set
+ BaseString m_alias; // for row expression alias
+ TableSet m_tableSet; // depends on these tables
+ bool m_isAggr; // contains an aggregate expression
+ bool m_isBound; // only constants and aggregates
+ Exec_expr* m_exec; // XXX wrong move
+};
+
+inline
+Plan_expr::Plan_expr(Plan_root* root, Type type) :
+ Plan_base(root),
+ m_type(type),
+ m_isAggr(false),
+ m_isBound(false),
+ m_exec(0)
+{
+}
+
+inline Plan_expr::Type
+Plan_expr::type() const
+{
+ return m_type;
+}
+
+inline const SqlType&
+Plan_expr::sqlType() const
+{
+ ctx_assert(m_sqlType.type() != SqlType::Undef);
+ return m_sqlType;
+}
+
+inline const Plan_base::TableSet&
+Plan_expr::tableSet() const
+{
+ return m_tableSet;
+}
+
+inline const BaseString&
+Plan_expr::getAlias() const
+{
+ return m_alias;
+}
+
+inline bool
+Plan_expr::isAggr() const
+{
+ return m_isAggr;
+}
+
+inline bool
+Plan_expr::isBound() const
+{
+ return m_isBound;
+}
+
+/**
+ * @class Exec_expr
+ * @brief Base class for expressions in ExecTree
+ */
+class Exec_expr : public Exec_base {
+public:
+ /**
+ * Exec_expr::Code includes reference to SqlSpec which
+ * specifies data type and access method.
+ */
+ class Code : public Exec_base::Code {
+ public:
+ Code(const SqlSpec& sqlSpec);
+ virtual ~Code() = 0;
+ const SqlSpec& sqlSpec() const;
+ protected:
+ friend class Exec_expr;
+ const SqlSpec& m_sqlSpec; // subclass must contain
+ };
+ /**
+ * Exec_expr::Data includes reference to SqlField which
+ * contains specification and data address.
+ */
+ class Data : public Exec_base::Data {
+ public:
+ Data(const SqlField& sqlField);
+ virtual ~Data() = 0;
+ const SqlField& sqlField() const;
+ const SqlField& groupField(unsigned i) const;
+ protected:
+ friend class Exec_expr;
+ const SqlField& m_sqlField; // subclass must contain
+ // group-by data
+ typedef std::vector<SqlField> GroupField;
+ GroupField m_groupField;
+ SqlField& groupField(const SqlType& sqlType, unsigned i, bool initFlag);
+ };
+ Exec_expr(Exec_root* root);
+ virtual ~Exec_expr() = 0;
+ /**
+ * Evaluate the expression. Must be implemented by all
+ * subclasses. Check ctx.ok() for errors.
+ */
+ virtual void evaluate(Ctx& ctx, Ctl& ctl) = 0;
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_expr::Code::Code(const SqlSpec& sqlSpec) :
+ m_sqlSpec(sqlSpec)
+{
+}
+
+inline const SqlSpec&
+Exec_expr::Code::sqlSpec() const {
+ return m_sqlSpec;
+}
+
+inline
+Exec_expr::Data::Data(const SqlField& sqlField) :
+ m_sqlField(sqlField)
+{
+}
+
+inline const SqlField&
+Exec_expr::Data::sqlField() const
+{
+ return m_sqlField;
+}
+
+inline const SqlField&
+Exec_expr::Data::groupField(unsigned i) const
+{
+ ctx_assert(i != 0 && i < m_groupField.size());
+ return m_groupField[i];
+}
+
+inline
+Exec_expr::Exec_expr(Exec_root* root) :
+ Exec_base(root)
+{
+}
+
+// children
+
+inline const Exec_expr::Code&
+Exec_expr::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_expr::Data&
+Exec_expr::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_expr_column.cpp b/ndb/src/client/odbc/codegen/Code_expr_column.cpp
new file mode 100644
index 00000000000..17a9a502d4c
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_column.cpp
@@ -0,0 +1,160 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <dictionary/DictSchema.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_query.hpp"
+#include "Code_table.hpp"
+#include "Code_expr_column.hpp"
+#include "Code_root.hpp"
+
+// Plan_expr_column
+
+Plan_expr_column::~Plan_expr_column()
+{
+}
+
+Plan_base*
+Plan_expr_column::analyze(Ctx& ctx, Ctl& ctl)
+{
+ m_exec = 0;
+ analyzeColumn(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ Plan_expr::m_sqlType = Plan_column::m_sqlType;
+ // depends on one table
+ m_tableSet.insert(m_resTable);
+ // not constant as set-value
+ ctl.m_const = false;
+ // set alias name
+ m_alias = m_name;
+ return this;
+}
+
+Exec_base*
+Plan_expr_column::codegen(Ctx& ctx, Ctl& ctl)
+{
+ if (m_exec != 0)
+ return m_exec;
+ // connect column to query column
+ const Exec_query* execQuery = ctl.m_execQuery;
+ ctx_assert(execQuery != 0);
+ const Exec_query::Code& codeQuery = execQuery->getCode();
+ const SqlSpec sqlSpec(Plan_expr::m_sqlType, SqlSpec::Reference);
+ // offset in final output row
+ ctx_assert(m_resTable != 0 && m_resTable->m_resOff != 0 && m_resPos != 0);
+ unsigned resOff = m_resTable->m_resOff + (m_resPos - 1);
+ // create the code
+ Exec_expr_column* exec = new Exec_expr_column(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ Exec_expr_column::Code& code = *new Exec_expr_column::Code(sqlSpec, resOff);
+ exec->setCode(code);
+ m_exec = exec;
+ return exec;
+}
+
+bool
+Plan_expr_column::resolveEq(Ctx& ctx, Plan_expr* expr)
+{
+ ctx_assert(m_resTable != 0 && expr != 0);
+ return m_resTable->resolveEq(ctx, this, expr);
+}
+
+void
+Plan_expr_column::print(Ctx& ctx)
+{
+ ctx.print(" [expr_column %s]", getPrintName());
+}
+
+bool
+Plan_expr_column::isEqual(const Plan_expr* expr) const
+{
+ ctx_assert(expr != 0);
+ if (expr->type() != Plan_expr::TypeColumn)
+ return false;
+ const Plan_expr_column* expr2 = static_cast<const Plan_expr_column*>(expr);
+ ctx_assert(m_resTable != 0 && expr2->m_resTable != 0);
+ if (m_resTable != expr2->m_resTable)
+ return false;
+ ctx_assert(m_dictColumn != 0 && expr2->m_dictColumn != 0);
+ if (m_dictColumn != expr2->m_dictColumn)
+ return false;
+ return true;
+}
+
+bool
+Plan_expr_column::isGroupBy(const Plan_expr_row* row) const
+{
+ if (isAnyEqual(row))
+ return true;
+ return false;
+}
+
+// Exec_expr_column
+
+Exec_expr_column::Code::~Code()
+{
+}
+
+Exec_expr_column::Data::~Data()
+{
+}
+
+Exec_expr_column::~Exec_expr_column()
+{
+}
+
+void
+Exec_expr_column::alloc(Ctx& ctx, Ctl& ctl)
+{
+ if (m_data != 0)
+ return;
+ const Code& code = getCode();
+ // connect column to query column
+ ctx_assert(ctl.m_query != 0);
+ const SqlRow& sqlRow = ctl.m_query->getData().sqlRow();
+ SqlField& sqlField = sqlRow.getEntry(code.m_resOff);
+ // create the data
+ Data& data = *new Data(sqlField);
+ setData(data);
+}
+
+void
+Exec_expr_column::evaluate(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ if (ctl.m_groupIndex != 0) {
+ SqlField& out = data.groupField(code.sqlSpec().sqlType(), ctl.m_groupIndex, ctl.m_groupInit);
+ data.sqlField().copy(ctx, out);
+ }
+}
+
+void
+Exec_expr_column::close(Ctx& ctx)
+{
+ Data& data = getData();
+ data.m_groupField.clear();
+}
+
+void
+Exec_expr_column::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [column %d]", code.m_resOff);
+}
diff --git a/ndb/src/client/odbc/codegen/Code_expr_column.hpp b/ndb/src/client/odbc/codegen/Code_expr_column.hpp
new file mode 100644
index 00000000000..2ce7c441e45
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_column.hpp
@@ -0,0 +1,120 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_expr_column_hpp
+#define ODBC_CODEGEN_Code_expr_column_hpp
+
+#include <common/common.hpp>
+#include "Code_column.hpp"
+#include "Code_expr.hpp"
+
+class DictColumn;
+class Plan_table;
+
+/**
+ * @class Plan_expr_column
+ * @brief Column in query expression
+ */
+class Plan_expr_column : public Plan_expr, public Plan_column {
+public:
+ Plan_expr_column(Plan_root* root, const BaseString& name);
+ virtual ~Plan_expr_column();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ bool resolveEq(Ctx& ctx, Plan_expr* expr);
+ void print(Ctx& ctx);
+ bool isEqual(const Plan_expr* expr) const;
+ bool isGroupBy(const Plan_expr_row* row) const;
+};
+
+inline
+Plan_expr_column::Plan_expr_column(Plan_root* root, const BaseString& name) :
+ Plan_expr(root, TypeColumn),
+ Plan_column(Type_expr, name)
+{
+}
+
+/**
+ * @class Exec_expr_column
+ * @brief Column in query expression
+ */
+class Exec_expr_column : public Exec_expr {
+public:
+ class Code : public Exec_expr::Code {
+ public:
+ Code(const SqlSpec& sqlSpec, unsigned resOff);
+ virtual ~Code();
+ protected:
+ friend class Exec_expr_column;
+ SqlSpec m_sqlSpec;
+ unsigned m_resOff;
+ };
+ class Data : public Exec_expr::Data {
+ public:
+ Data(SqlField& sqlField);
+ virtual ~Data();
+ protected:
+ friend class Exec_expr_column;
+ // set reference to SqlField in query
+ };
+ Exec_expr_column(Exec_root* root);
+ virtual ~Exec_expr_column();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void evaluate(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_expr_column::Code::Code(const SqlSpec& sqlSpec, unsigned resOff) :
+ Exec_expr::Code(m_sqlSpec),
+ m_sqlSpec(sqlSpec),
+ m_resOff(resOff)
+{
+}
+
+inline
+Exec_expr_column::Data::Data(SqlField& sqlField) :
+ Exec_expr::Data(sqlField)
+{
+}
+
+inline
+Exec_expr_column::Exec_expr_column(Exec_root* root) :
+ Exec_expr(root)
+{
+}
+
+// children
+
+inline const Exec_expr_column::Code&
+Exec_expr_column::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_expr_column::Data&
+Exec_expr_column::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_expr_const.cpp b/ndb/src/client/odbc/codegen/Code_expr_const.cpp
new file mode 100644
index 00000000000..564d307a4f8
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_const.cpp
@@ -0,0 +1,138 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_expr_const.hpp"
+#include "Code_root.hpp"
+
+// Plan_expr_const
+
+Plan_expr_const::~Plan_expr_const()
+{
+}
+
+Plan_base*
+Plan_expr_const::analyze(Ctx& ctx, Ctl& ctl)
+{
+ m_exec = 0;
+ // convert data type
+ m_lexType.convert(ctx, m_sqlType, m_string.length());
+ if (! ctx.ok())
+ return 0;
+ // depends on no tables
+ // set alias name
+ m_alias = m_string;
+ // node was not changed
+ return this;
+}
+
+Exec_base*
+Plan_expr_const::codegen(Ctx& ctx, Ctl& ctl)
+{
+ if (m_exec != 0)
+ return m_exec;
+ // convert data
+ SqlSpec sqlSpec(m_sqlType, SqlSpec::Physical);
+ SqlField sqlField(sqlSpec);
+ LexSpec lexSpec(m_lexType);
+ lexSpec.convert(ctx, m_string, sqlField);
+ if (! ctx.ok())
+ return 0;
+ // create code
+ Exec_expr_const* exec = new Exec_expr_const(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ Exec_expr_const::Code& code = *new Exec_expr_const::Code(sqlField);
+ exec->setCode(code);
+ m_exec = exec;
+ return exec;
+}
+
+void
+Plan_expr_const::print(Ctx& ctx)
+{
+ ctx.print(" [const %s]", m_string.c_str());
+}
+
+bool
+Plan_expr_const::isEqual(const Plan_expr* expr) const
+{
+ ctx_assert(expr != 0);
+ if (expr->type() != Plan_expr::TypeConst)
+ return false;
+ const Plan_expr_const* expr2 = static_cast<const Plan_expr_const*>(expr);
+ if (strcmp(m_string.c_str(), expr2->m_string.c_str()) != 0)
+ return false;
+ return true;
+}
+
+bool
+Plan_expr_const::isGroupBy(const Plan_expr_row* row) const
+{
+ return true;
+}
+
+// Exec_expr_const
+
+Exec_expr_const::Code::~Code()
+{
+}
+
+Exec_expr_const::Data::~Data()
+{
+}
+
+Exec_expr_const::~Exec_expr_const()
+{
+}
+
+void
+Exec_expr_const::alloc(Ctx& ctx, Ctl& ctl)
+{
+ if (m_data != 0)
+ return;
+ // copy the value for const correctness reasons
+ SqlField sqlField(getCode().m_sqlField);
+ Data& data = *new Data(sqlField);
+ setData(data);
+}
+
+void
+Exec_expr_const::evaluate(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ if (ctl.m_groupIndex != 0) {
+ SqlField& out = data.groupField(code.sqlSpec().sqlType(), ctl.m_groupIndex, ctl.m_groupInit);
+ data.sqlField().copy(ctx, out);
+ }
+}
+
+void
+Exec_expr_const::close(Ctx& ctx)
+{
+ Data& data = getData();
+ data.m_groupField.clear();
+}
+
+void
+Exec_expr_const::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [");
+ char buf[500];
+ code.m_sqlField.print(buf, sizeof(buf));
+ ctx.print("%s", buf);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_expr_const.hpp b/ndb/src/client/odbc/codegen/Code_expr_const.hpp
new file mode 100644
index 00000000000..2e26c637a23
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_const.hpp
@@ -0,0 +1,120 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_expr_const_hpp
+#define ODBC_CODEGEN_Code_expr_const_hpp
+
+#include <common/common.hpp>
+#include <common/DataField.hpp>
+#include "Code_expr.hpp"
+
+/**
+ * @class Plan_expr_const
+ * @brief Constant expression value in PlanTree
+ */
+class Plan_expr_const : public Plan_expr {
+public:
+ Plan_expr_const(Plan_root* root, LexType lexType, const char* value);
+ virtual ~Plan_expr_const();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ bool isEqual(const Plan_expr* expr) const;
+ bool isGroupBy(const Plan_expr_row* row) const;
+protected:
+ // lexical type and token set by the parser
+ LexType m_lexType;
+ BaseString m_string;
+};
+
+inline
+Plan_expr_const::Plan_expr_const(Plan_root* root, LexType lexType, const char* value) :
+ Plan_expr(root, TypeConst),
+ m_lexType(lexType),
+ m_string(value)
+{
+}
+
+/**
+ * @class Exec_expr_const
+ * @brief Constant expression value in ExecTree
+ */
+class Exec_expr_const : public Exec_expr {
+public:
+ class Code : public Exec_expr::Code {
+ public:
+ Code(const SqlField& sqlField);
+ virtual ~Code();
+ protected:
+ friend class Exec_expr_const;
+ const SqlField m_sqlField;
+ };
+ class Data : public Exec_expr::Data {
+ public:
+ Data(SqlField& sqlField);
+ virtual ~Data();
+ protected:
+ friend class Exec_expr_const;
+ SqlField m_sqlField;
+ };
+ Exec_expr_const(Exec_root* root);
+ virtual ~Exec_expr_const();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void evaluate(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_expr_const::Code::Code(const SqlField& sqlField) :
+ Exec_expr::Code(m_sqlField.sqlSpec()),
+ m_sqlField(sqlField)
+{
+}
+
+inline
+Exec_expr_const::Data::Data(SqlField& sqlField) :
+ Exec_expr::Data(m_sqlField),
+ m_sqlField(sqlField)
+{
+}
+
+inline
+Exec_expr_const::Exec_expr_const(Exec_root* root) :
+ Exec_expr(root)
+{
+}
+
+// children
+
+inline const Exec_expr_const::Code&
+Exec_expr_const::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_expr_const::Data&
+Exec_expr_const::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_expr_conv.cpp b/ndb/src/client/odbc/codegen/Code_expr_conv.cpp
new file mode 100644
index 00000000000..bc89482fedc
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_conv.cpp
@@ -0,0 +1,273 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_expr.hpp"
+#include "Code_expr_conv.hpp"
+#include "Code_root.hpp"
+
+// Plan_expr_conv
+
+Plan_expr_conv::~Plan_expr_conv()
+{
+}
+
+Plan_base*
+Plan_expr_conv::analyze(Ctx& ctx, Ctl& ctl)
+{
+ m_exec = 0;
+ const SqlType& t1 = sqlType();
+ ctx_assert(m_expr != 0);
+ m_expr->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ const SqlType& t2 = m_expr->sqlType();
+ if (t2.type() == SqlType::Unbound) {
+ return m_expr;
+ }
+ if (t1.equal(t2)) {
+ return m_expr;
+ }
+ // XXX move to runtime or make table-driven
+ bool ok = false;
+ if (t2.type() == SqlType::Null) {
+ ok = true;
+ } else if (t1.type() == SqlType::Char) {
+ if (t2.type() == SqlType::Char) {
+ ok = true;
+ } else if (t2.type() == SqlType::Varchar) {
+ ok = true;
+ } else if (t2.type() == SqlType::Binary) {
+ ok = true;
+ } else if (t2.type() == SqlType::Varbinary) {
+ ok = true;
+ }
+ } else if (t1.type() == SqlType::Varchar) {
+ if (t2.type() == SqlType::Char) {
+ ok = true;
+ } else if (t2.type() == SqlType::Varchar) {
+ ok = true;
+ } else if (t2.type() == SqlType::Binary) {
+ ok = true;
+ } else if (t2.type() == SqlType::Varbinary) {
+ ok = true;
+ }
+ } else if (t1.type() == SqlType::Binary) {
+ if (t2.type() == SqlType::Char) {
+ ok = true;
+ } else if (t2.type() == SqlType::Varchar) {
+ ok = true;
+ } else if (t2.type() == SqlType::Binary) {
+ ok = true;
+ } else if (t2.type() == SqlType::Varbinary) {
+ ok = true;
+ }
+ } else if (t1.type() == SqlType::Varbinary) {
+ if (t2.type() == SqlType::Char) {
+ ok = true;
+ } else if (t2.type() == SqlType::Varchar) {
+ ok = true;
+ } else if (t2.type() == SqlType::Binary) {
+ ok = true;
+ } else if (t2.type() == SqlType::Varbinary) {
+ ok = true;
+ }
+ } else if (t1.type() == SqlType::Smallint) {
+ if (t2.type() == SqlType::Smallint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Integer) {
+ ok = true;
+ } else if (t2.type() == SqlType::Bigint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Real) {
+ ok = true;
+ } else if (t2.type() == SqlType::Double) {
+ ok = true;
+ }
+ } else if (t1.type() == SqlType::Integer) {
+ if (t2.type() == SqlType::Smallint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Integer) {
+ ok = true;
+ } else if (t2.type() == SqlType::Bigint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Real) {
+ ok = true;
+ } else if (t2.type() == SqlType::Double) {
+ ok = true;
+ }
+ } else if (t1.type() == SqlType::Bigint) {
+ if (t2.type() == SqlType::Smallint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Integer) {
+ ok = true;
+ } else if (t2.type() == SqlType::Bigint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Real) {
+ ok = true;
+ } else if (t2.type() == SqlType::Double) {
+ ok = true;
+ }
+ } else if (t1.type() == SqlType::Real) {
+ if (t2.type() == SqlType::Smallint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Integer) {
+ ok = true;
+ } else if (t2.type() == SqlType::Bigint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Real) {
+ ok = true;
+ } else if (t2.type() == SqlType::Double) {
+ ok = true;
+ }
+ } else if (t1.type() == SqlType::Double) {
+ if (t2.type() == SqlType::Smallint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Integer) {
+ ok = true;
+ } else if (t2.type() == SqlType::Bigint) {
+ ok = true;
+ } else if (t2.type() == SqlType::Real) {
+ ok = true;
+ } else if (t2.type() == SqlType::Double) {
+ ok = true;
+ }
+ } else if (t1.type() == SqlType::Datetime) {
+ if (t2.type() == SqlType::Datetime) {
+ ok = true;
+ }
+ }
+ if (! ok) {
+ char b1[40], b2[40];
+ t1.print(b1, sizeof(b1));
+ t2.print(b2, sizeof(b2));
+ ctx.pushStatus(Error::Gen, "cannot convert %s to %s", b2, b1);
+ return 0;
+ }
+ // depend on same tables
+ const TableSet& ts = m_expr->tableSet();
+ m_tableSet.insert(ts.begin(), ts.end());
+ // set alias
+ m_alias = m_expr->getAlias();
+ return this;
+}
+
+Exec_base*
+Plan_expr_conv::codegen(Ctx& ctx, Ctl& ctl)
+{
+ if (m_exec != 0)
+ return m_exec;
+ Exec_expr_conv* exec = new Exec_expr_conv(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // create code for subexpression
+ ctx_assert(m_expr != 0);
+ Exec_expr* execExpr = static_cast<Exec_expr*>(m_expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ exec->setExpr(execExpr);
+ // create the code
+ SqlSpec sqlSpec(sqlType(), SqlSpec::Physical);
+ Exec_expr_conv::Code& code = *new Exec_expr_conv::Code(sqlSpec);
+ exec->setCode(code);
+ m_exec = exec;
+ return exec;
+}
+
+void
+Plan_expr_conv::print(Ctx& ctx)
+{
+ ctx.print(" [expr_conv ");
+ char buf[100];
+ m_sqlType.print(buf, sizeof(buf));
+ ctx.print("%s", buf);
+ Plan_base* a[] = { m_expr };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
+
+bool
+Plan_expr_conv::isEqual(const Plan_expr* expr) const
+{
+ ctx_assert(expr != 0);
+ if (expr->type() != Plan_expr::TypeConv)
+ return false;
+ const Plan_expr_conv* expr2 = static_cast<const Plan_expr_conv*>(expr);
+ if (! m_sqlType.equal(expr2->m_sqlType))
+ return false;
+ ctx_assert(m_expr != 0 && expr2->m_expr != 0);
+ if (! m_expr->isEqual(expr2->m_expr))
+ return false;
+ return true;
+}
+
+bool
+Plan_expr_conv::isGroupBy(const Plan_expr_row* row) const
+{
+ if (isAnyEqual(row))
+ return true;
+ ctx_assert(m_expr != 0);
+ if (m_expr->isGroupBy(row))
+ return true;
+ return false;
+}
+
+// Code_expr_conv
+
+Exec_expr_conv::Code::~Code()
+{
+}
+
+Exec_expr_conv::Data::~Data()
+{
+}
+
+Exec_expr_conv::~Exec_expr_conv()
+{
+}
+
+void
+Exec_expr_conv::alloc(Ctx& ctx, Ctl& ctl)
+{
+ if (m_data != 0)
+ return;
+ const Code& code = getCode();
+ // allocate subexpression
+ ctx_assert(m_expr != 0);
+ m_expr->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ SqlField sqlField(code.m_sqlSpec);
+ Data& data = *new Data(sqlField);
+ setData(data);
+}
+
+void
+Exec_expr_conv::close(Ctx& ctx)
+{
+ ctx_assert(m_expr != 0);
+ m_expr->close(ctx);
+ Data& data = getData();
+ data.m_groupField.clear();
+}
+
+void
+Exec_expr_conv::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [expr_conv");
+ Exec_base* a[] = { m_expr };
+ printList(ctx, a, sizeof(a)/sizeof(a[0]));
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_expr_conv.hpp b/ndb/src/client/odbc/codegen/Code_expr_conv.hpp
new file mode 100644
index 00000000000..3294960c7b3
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_conv.hpp
@@ -0,0 +1,141 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_expr_conv_hpp
+#define ODBC_CODEGEN_Code_expr_conv_hpp
+
+#include <common/common.hpp>
+#include <common/DataField.hpp>
+#include "Code_expr.hpp"
+
+/**
+ * @class Plan_expr_conv
+ * @brief Data type conversion in PlanTree
+ *
+ * Inserted to convert value to another compatible type.
+ */
+class Plan_expr_conv : public Plan_expr {
+public:
+ Plan_expr_conv(Plan_root* root, const SqlType& sqlType);
+ virtual ~Plan_expr_conv();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ bool isEqual(const Plan_expr* expr) const;
+ bool isGroupBy(const Plan_expr_row* row) const;
+ // children
+ void setExpr(Plan_expr* expr);
+protected:
+ Plan_expr* m_expr;
+};
+
+inline
+Plan_expr_conv::Plan_expr_conv(Plan_root* root, const SqlType& sqlType) :
+ Plan_expr(root, TypeConv),
+ m_expr(0)
+{
+ ctx_assert(sqlType.type() != SqlType::Undef);
+ m_sqlType = sqlType;
+}
+
+inline void
+Plan_expr_conv::setExpr(Plan_expr* expr)
+{
+ ctx_assert(expr != 0);
+ m_expr = expr;
+}
+
+/**
+ * @class Exec_expr_conv
+ * @brief Data type conversion in ExecTree
+ */
+class Exec_expr_conv : public Exec_expr {
+public:
+ class Code : public Exec_expr::Code {
+ public:
+ Code(const SqlSpec& spec);
+ virtual ~Code();
+ protected:
+ friend class Exec_expr_conv;
+ const SqlSpec m_sqlSpec;
+ };
+ class Data : public Exec_expr::Data {
+ public:
+ Data(const SqlField& sqlField);
+ virtual ~Data();
+ protected:
+ friend class Exec_expr_conv;
+ SqlField m_sqlField;
+ };
+ Exec_expr_conv(Exec_root* root);
+ virtual ~Exec_expr_conv();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void evaluate(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setExpr(Exec_expr* expr);
+protected:
+ Exec_expr* m_expr;
+};
+
+inline
+Exec_expr_conv::Code::Code(const SqlSpec& sqlSpec) :
+ Exec_expr::Code(m_sqlSpec),
+ m_sqlSpec(sqlSpec)
+{
+}
+
+inline
+Exec_expr_conv::Data::Data(const SqlField& sqlField) :
+ Exec_expr::Data(m_sqlField),
+ m_sqlField(sqlField)
+{
+}
+
+inline
+Exec_expr_conv::Exec_expr_conv(Exec_root* root) :
+ Exec_expr(root),
+ m_expr(0)
+{
+}
+
+// children
+
+inline const Exec_expr_conv::Code&
+Exec_expr_conv::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_expr_conv::Data&
+Exec_expr_conv::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_expr_conv::setExpr(Exec_expr* expr)
+{
+ ctx_assert(m_expr == 0 && expr != 0);
+ m_expr = expr;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_expr_func.cpp b/ndb/src/client/odbc/codegen/Code_expr_func.cpp
new file mode 100644
index 00000000000..96b461a72d9
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_func.cpp
@@ -0,0 +1,401 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_expr.hpp"
+#include "Code_expr_func.hpp"
+#include "Code_expr_conv.hpp"
+#include "Code_root.hpp"
+#include "PortDefs.h"
+
+
+// Expr_func
+
+static const struct { const char* alias; const char* name; }
+expr_func_alias[] = {
+ { "SUBSTRING", "SUBSTR" },
+ { 0, 0 }
+};
+
+static const Expr_func
+expr_func[] = {
+ Expr_func(Expr_func::Substr, "SUBSTR", false ),
+ Expr_func(Expr_func::Left, "LEFT", false ),
+ Expr_func(Expr_func::Right, "RIGHT", false ),
+ Expr_func(Expr_func::Count, "COUNT", true ),
+ Expr_func(Expr_func::Max, "MAX", true ),
+ Expr_func(Expr_func::Min, "MIN", true ),
+ Expr_func(Expr_func::Sum, "SUM", true ),
+ Expr_func(Expr_func::Avg, "AVG", true ),
+ Expr_func(Expr_func::Rownum, "ROWNUM", false ),
+ Expr_func(Expr_func::Sysdate, "SYSDATE", false ),
+ Expr_func(Expr_func::Undef, 0, false )
+};
+
+const Expr_func&
+Expr_func::find(const char* name)
+{
+ for (unsigned i = 0; expr_func_alias[i].alias != 0; i++) {
+ if (strcasecmp(expr_func_alias[i].alias, name) == 0) {
+ name = expr_func_alias[i].name;
+ break;
+ }
+ }
+ const Expr_func* p;
+ for (p = expr_func; p->m_name != 0; p++) {
+ if (strcasecmp(p->m_name, name) == 0)
+ break;
+ }
+ return *p;
+}
+
+// Plan_expr_func
+
+Plan_expr_func::~Plan_expr_func()
+{
+ delete[] m_conv;
+ m_conv = 0;
+}
+
+Plan_base*
+Plan_expr_func::analyze(Ctx& ctx, Ctl& ctl)
+{
+ m_exec = 0;
+ ctx_assert(m_narg == 0 || m_args != 0);
+ // aggregate check
+ if (m_func.m_aggr) {
+ if (! ctl.m_aggrok) {
+ ctx.pushStatus(Error::Gen, "%s: invalid use of aggregate function", m_func.m_name);
+ return 0;
+ }
+ if (ctl.m_aggrin) {
+ // XXX actually possible with group-by but too hard
+ ctx.pushStatus(Error::Gen, "%s: nested aggregate function", m_func.m_name);
+ return 0;
+ }
+ ctl.m_aggrin = true;
+ m_isAggr = true;
+ m_isBound = true;
+ }
+ // analyze argument expressions
+ if (m_func.m_code != Expr_func::Rownum)
+ m_isBound = true;
+ for (unsigned i = 1; i <= m_narg; i++) {
+ Plan_expr* expr = m_args->getExpr(i);
+ expr = static_cast<Plan_expr*>(expr->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(expr != 0);
+ if (expr->m_isAggr)
+ m_isAggr = true;
+ if (! m_func.m_aggr && ! expr->m_isBound)
+ m_isBound = false;
+ }
+ if (m_func.m_aggr)
+ ctl.m_aggrin = false;
+ // find result type and conversion types
+ SqlType res;
+ const Expr_func::Code fc = m_func.m_code;
+ const char* const invalidArgCount = "%s: argument count %u is invalid";
+ const char* const invalidArgType = "%s: argument %u has invalid type";
+ if (fc == Expr_func::Substr || fc == Expr_func::Left || fc == Expr_func::Right) {
+ if ((m_narg != (unsigned)2) && (m_narg != (unsigned)(fc == Expr_func::Substr ? 3 : 2))) {
+ ctx.pushStatus(Error::Gen, invalidArgCount, m_func.m_name, m_narg);
+ return 0;
+ }
+ const SqlType& t1 = m_args->getExpr(1)->sqlType();
+ switch (t1.type()) {
+ case SqlType::Char:
+ {
+ // XXX convert to varchar for now to get length right
+ SqlType tx(SqlType::Varchar, t1.length());
+ res = m_conv[1] = tx;
+ }
+ break;
+ case SqlType::Varchar:
+ case SqlType::Unbound:
+ res = m_conv[1] = t1;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, invalidArgType, m_func.m_name, 1);
+ return 0;
+ }
+ for (unsigned i = 2; i <= m_narg; i++) {
+ const SqlType& t2 = m_args->getExpr(i)->sqlType();
+ switch (t2.type()) {
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ case SqlType::Unbound:
+ m_conv[i] = t2;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, invalidArgType, m_func.m_name, i);
+ return 0;
+ }
+ }
+ } else if (fc == Expr_func::Count) {
+ ctx_assert(m_args != 0);
+ if (m_args->getAsterisk()) {
+ ctx_assert(m_narg == 0);
+ } else {
+ if (m_narg != 1) {
+ ctx.pushStatus(Error::Gen, invalidArgCount, m_func.m_name, m_narg);
+ return 0;
+ }
+ m_conv[1] = m_args->getExpr(1)->sqlType();
+ }
+ res.setType(ctx, SqlType::Bigint);
+ } else if (fc == Expr_func::Min || fc == Expr_func::Max) {
+ if (m_narg != 1) {
+ ctx.pushStatus(Error::Gen, invalidArgCount, m_func.m_name, m_narg);
+ return 0;
+ }
+ const SqlType& t1 = m_args->getExpr(1)->sqlType();
+ res = m_conv[1] = t1;
+ } else if (fc == Expr_func::Sum) {
+ if (m_narg != 1) {
+ ctx.pushStatus(Error::Gen, invalidArgCount, m_func.m_name, m_narg);
+ return 0;
+ }
+ const SqlType& t1 = m_args->getExpr(1)->sqlType();
+ switch (t1.type()) {
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ res.setType(ctx, SqlType::Bigint);
+ m_conv[1] = res;
+ break;
+ case SqlType::Real:
+ case SqlType::Double:
+ res.setType(ctx, SqlType::Double);
+ m_conv[1] = res;
+ break;
+ case SqlType::Unbound:
+ res = m_conv[1] = t1;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, invalidArgType, m_func.m_name, 1);
+ return 0;
+ }
+ } else if (fc == Expr_func::Avg) {
+ if (m_narg != 1) {
+ ctx.pushStatus(Error::Gen, invalidArgCount, m_func.m_name, m_narg);
+ return 0;
+ }
+ const SqlType& t1 = m_args->getExpr(1)->sqlType();
+ switch (t1.type()) {
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ case SqlType::Real:
+ case SqlType::Double:
+ res.setType(ctx, SqlType::Double);
+ m_conv[1] = res;
+ break;
+ case SqlType::Unbound:
+ res = m_conv[1] = t1;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, invalidArgType, m_func.m_name, 1);
+ return 0;
+ }
+ } else if (fc == Expr_func::Rownum) {
+ ctx_assert(m_narg == 0 && m_args == 0);
+ res.setType(ctx, SqlType::Bigint);
+ } else if (fc == Expr_func::Sysdate) {
+ ctx_assert(m_narg == 0 && m_args == 0);
+ res.setType(ctx, SqlType::Datetime);
+ } else {
+ ctx_assert(false);
+ }
+ // insert required conversions
+ for (unsigned i = 1; i <= m_narg; i++) {
+ if (m_conv[i].type() == SqlType::Unbound) {
+ // parameter type not yet bound
+ continue;
+ }
+ Plan_expr_conv* exprConv = new Plan_expr_conv(m_root, m_conv[i]);
+ m_root->saveNode(exprConv);
+ exprConv->setExpr(m_args->getExpr(i));
+ Plan_expr* expr = static_cast<Plan_expr*>(exprConv->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ m_args->setExpr(i, expr);
+ }
+ // set result type
+ m_sqlType = res;
+ // set table dependencies
+ for (unsigned i = 1; i <= m_narg; i++) {
+ const TableSet& ts = m_args->getExpr(i)->tableSet();
+ m_tableSet.insert(ts.begin(), ts.end());
+ }
+ // set alias name
+ m_alias.assign(m_func.m_name);
+ if (m_narg == 0) {
+ if (fc == Expr_func::Count)
+ m_alias.append("(*)");
+ } else {
+ m_alias.append("(");
+ for (unsigned i = 1; i <= m_narg; i++) {
+ if (i > 1)
+ m_alias.append(",");
+ m_alias.append(m_args->getExpr(i)->getAlias());
+ }
+ m_alias.append(")");
+ }
+ return this;
+}
+
+Exec_base*
+Plan_expr_func::codegen(Ctx& ctx, Ctl& ctl)
+{
+ if (m_exec != 0)
+ return m_exec;
+ Exec_expr_func* exec = new Exec_expr_func(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ SqlSpec sqlSpec(sqlType(), SqlSpec::Physical);
+ Exec_expr_func::Code& code = *new Exec_expr_func::Code(m_func, sqlSpec);
+ exec->setCode(code);
+ code.m_narg = m_narg;
+ code.m_args = new Exec_expr* [1 + m_narg];
+ for (unsigned i = 0; i <= m_narg; i++)
+ code.m_args[i] = 0;
+ // create code for arguments
+ for (unsigned i = 1; i <= m_narg; i++) {
+ Plan_expr* expr = m_args->getExpr(i);
+ ctx_assert(expr != 0);
+ Exec_expr* execExpr = static_cast<Exec_expr*>(expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ code.m_args[i] = execExpr;
+ }
+ m_exec = exec;
+ return exec;
+}
+
+void
+Plan_expr_func::print(Ctx& ctx)
+{
+ ctx.print(" [%s", m_func.m_name);
+ Plan_base* a[] = { m_args };
+ printList(ctx, a, sizeof(a)/sizeof(a[1]));
+ ctx.print("]");
+}
+
+bool
+Plan_expr_func::isEqual(const Plan_expr* expr) const
+{
+ ctx_assert(expr != 0);
+ if (expr->type() != Plan_expr::TypeFunc)
+ return false;
+ const Plan_expr_func* expr2 = static_cast<const Plan_expr_func*>(expr);
+ if (m_func.m_code != expr2->m_func.m_code)
+ return false;
+ if (m_narg != expr2->m_narg)
+ return false;
+ ctx_assert(m_args != 0 && expr2->m_args != 0);
+ for (unsigned i = 1; i <= m_narg; i++) {
+ if (! m_args->getExpr(i)->isEqual(expr2->m_args->getExpr(i)))
+ return false;
+ }
+ return true;
+}
+
+bool
+Plan_expr_func::isGroupBy(const Plan_expr_row* row) const
+{
+ if (m_func.m_aggr)
+ return true;
+ switch (m_func.m_code) {
+ case Expr_func::Substr:
+ case Expr_func::Left:
+ case Expr_func::Right:
+ ctx_assert(m_narg >= 1);
+ if (m_args->getExpr(1)->isGroupBy(row))
+ return true;
+ break;
+ case Expr_func::Sysdate:
+ return true;
+ default:
+ break;
+ }
+ if (isAnyEqual(row))
+ return true;
+ return false;
+}
+
+// Exec_expr_func
+
+Exec_expr_func::Code::~Code()
+{
+ delete[] m_args;
+ m_args = 0;
+}
+
+Exec_expr_func::Data::~Data()
+{
+}
+
+Exec_expr_func::~Exec_expr_func()
+{
+}
+
+void
+Exec_expr_func::alloc(Ctx& ctx, Ctl& ctl)
+{
+ if (m_data != 0)
+ return;
+ const Code& code = getCode();
+ // allocate arguments
+ for (unsigned i = 1; i <= code.m_narg; i++) {
+ ctx_assert(code.m_args != 0 && code.m_args[i] != 0);
+ code.m_args[i]->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ SqlField sqlField(code.m_sqlSpec);
+ Data& data = *new Data(sqlField);
+ setData(data);
+ ctx_assert(ctl.m_groupIndex == 0);
+ init(ctx, ctl);
+}
+
+void
+Exec_expr_func::close(Ctx& ctx)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ for (unsigned i = 1; i <= code.m_narg; i++) {
+ ctx_assert(code.m_args != 0 && code.m_args[i] != 0);
+ code.m_args[i]->close(ctx);
+ }
+ data.m_groupField.clear();
+ Ctl ctl(0);
+ init(ctx, ctl);
+}
+
+void
+Exec_expr_func::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [%s", code.m_func.m_name);
+ for (unsigned i = 1; i <= code.m_narg; i++) {
+ Exec_base* a[] = { code.m_args[i] };
+ printList(ctx, a, sizeof(a)/sizeof(a[0]));
+ }
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_expr_func.hpp b/ndb/src/client/odbc/codegen/Code_expr_func.hpp
new file mode 100644
index 00000000000..856d7529875
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_func.hpp
@@ -0,0 +1,193 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_expr_func_hpp
+#define ODBC_CODEGEN_Code_expr_func_hpp
+
+#include <common/common.hpp>
+#include <common/DataField.hpp>
+#include "Code_expr.hpp"
+#include "Code_expr_row.hpp"
+
+/**
+ * @class Expr_func
+ * @brief Specifies a function
+ */
+struct Expr_func {
+ enum Code {
+ Undef = 0,
+ Substr,
+ Left,
+ Right,
+ Count,
+ Max,
+ Min,
+ Sum,
+ Avg,
+ Rownum,
+ Sysdate
+ };
+ Expr_func(Code code, const char* name, bool aggr);
+ Code m_code;
+ const char* m_name;
+ bool m_aggr;
+ static const Expr_func& find(const char* name);
+};
+
+inline
+Expr_func::Expr_func(Code code, const char* name, bool aggr) :
+ m_code(code),
+ m_name(name),
+ m_aggr(aggr)
+{
+}
+
+/**
+ * @class Plan_expr_func
+ * @brief Function node in an expression in PlanTree
+ */
+class Plan_expr_func : public Plan_expr {
+public:
+ Plan_expr_func(Plan_root* root, const Expr_func& func);
+ virtual ~Plan_expr_func();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ bool isEqual(const Plan_expr* expr) const;
+ bool isGroupBy(const Plan_expr_row* row) const;
+ // children
+ void setArgs(Plan_expr_row* args);
+protected:
+ const Expr_func& m_func;
+ Plan_expr_row* m_args;
+ unsigned m_narg;
+ SqlType* m_conv; // temp work area
+};
+
+inline
+Plan_expr_func::Plan_expr_func(Plan_root* root, const Expr_func& func) :
+ Plan_expr(root, TypeFunc),
+ m_func(func),
+ m_args(0),
+ m_narg(0),
+ m_conv(0)
+{
+}
+
+inline void
+Plan_expr_func::setArgs(Plan_expr_row* args)
+{
+ if (args == 0) {
+ m_args = 0;
+ m_narg = 0;
+ } else {
+ m_args = args;
+ m_narg = m_args->getSize();
+ delete[] m_conv;
+ m_conv = new SqlType[1 + m_narg];
+ }
+}
+
+/**
+ * @class Exec_expr_func
+ * @brief Function node in an expression in ExecTree
+ */
+class Exec_expr_func : public Exec_expr {
+public:
+ class Code : public Exec_expr::Code {
+ public:
+ Code(const Expr_func& func, const SqlSpec& spec);
+ virtual ~Code();
+ protected:
+ friend class Plan_expr_func;
+ friend class Exec_expr_func;
+ const Expr_func& m_func;
+ const SqlSpec m_sqlSpec;
+ unsigned m_narg;
+ Exec_expr** m_args; // XXX pointers for now
+ };
+ class Data : public Exec_expr::Data {
+ public:
+ Data(const SqlField& sqlField);
+ virtual ~Data();
+ protected:
+ friend class Exec_expr_func;
+ SqlField m_sqlField;
+ struct Acc { // accumulators etc
+ SqlBigint m_count; // current row count
+ union {
+ SqlBigint m_bigint;
+ SqlDouble m_double;
+ SqlDatetime m_sysdate;
+ };
+ };
+ // group-by extra accumulators (default in entry 0)
+ typedef std::vector<Acc> GroupAcc;
+ GroupAcc m_groupAcc;
+ };
+ Exec_expr_func(Exec_root* root);
+ virtual ~Exec_expr_func();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void evaluate(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+protected:
+ void init(Ctx &ctx, Ctl& ctl); // initialize values
+};
+
+inline
+Exec_expr_func::Code::Code(const Expr_func& func, const SqlSpec& sqlSpec) :
+ Exec_expr::Code(m_sqlSpec),
+ m_func(func),
+ m_sqlSpec(sqlSpec),
+ m_args(0)
+{
+}
+
+inline
+Exec_expr_func::Data::Data(const SqlField& sqlField) :
+ Exec_expr::Data(m_sqlField),
+ m_sqlField(sqlField),
+ m_groupAcc(1)
+{
+}
+
+inline
+Exec_expr_func::Exec_expr_func(Exec_root* root) :
+ Exec_expr(root)
+{
+}
+
+// children
+
+inline const Exec_expr_func::Code&
+Exec_expr_func::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_expr_func::Data&
+Exec_expr_func::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_expr_op.cpp b/ndb/src/client/odbc/codegen/Code_expr_op.cpp
new file mode 100644
index 00000000000..7e8314c1741
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_op.cpp
@@ -0,0 +1,424 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_expr.hpp"
+#include "Code_expr_op.hpp"
+#include "Code_expr_conv.hpp"
+#include "Code_root.hpp"
+
+// Expr_op
+
+const char*
+Expr_op::name() const
+{
+ switch (m_opcode) {
+ case Add:
+ return "+";
+ case Subtract:
+ return "-";
+ case Multiply:
+ return "*";
+ case Divide:
+ return "/";
+ case Plus:
+ return "+";
+ case Minus:
+ return "-";
+ }
+ ctx_assert(false);
+ return "";
+}
+
+unsigned
+Expr_op::arity() const
+{
+ switch (m_opcode) {
+ case Add:
+ case Subtract:
+ case Multiply:
+ case Divide:
+ return 2;
+ case Plus:
+ case Minus:
+ return 1;
+ }
+ ctx_assert(false);
+ return 0;
+}
+
+// Plan_expr_op
+
+Plan_expr_op::~Plan_expr_op()
+{
+}
+
+Plan_base*
+Plan_expr_op::analyze(Ctx& ctx, Ctl& ctl)
+{
+ m_exec = 0;
+ unsigned arity = m_op.arity();
+ // analyze operands
+ m_isAggr = false;
+ m_isBound = true;
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ m_expr[i]->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ if (m_expr[i]->m_isAggr)
+ m_isAggr = true;
+ if (! m_expr[i]->m_isBound)
+ m_isBound = false;
+ }
+ // find result type and conversion types (currently same)
+ SqlType res;
+ SqlType con[1 + 2];
+ if (arity == 1) {
+ const SqlType& t1 = m_expr[1]->sqlType();
+ switch (t1.type()) {
+ case SqlType::Char:
+ case SqlType::Varchar:
+ break;
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ res.setType(ctx, SqlType::Bigint);
+ con[1] = res;
+ break;
+ case SqlType::Real:
+ case SqlType::Double:
+ res.setType(ctx, SqlType::Double);
+ con[1] = res;
+ break;
+ case SqlType::Null:
+ res.setType(ctx, SqlType::Null);
+ con[1] = res;
+ break;
+ case SqlType::Unbound:
+ res.setType(ctx, SqlType::Unbound);
+ con[1] = res;
+ default:
+ break;
+ }
+ if (con[1].type() == SqlType::Undef) {
+ char b1[40];
+ t1.print(b1, sizeof(b1));
+ ctx.pushStatus(Error::Gen, "type mismatch in operation: %s %s", m_op.name(), b1);
+ return 0;
+ }
+ } else if (arity == 2) {
+ const SqlType& t1 = m_expr[1]->sqlType();
+ const SqlType& t2 = m_expr[2]->sqlType();
+ switch (t1.type()) {
+ case SqlType::Char: // handle char types as in oracle
+ switch (t2.type()) {
+ case SqlType::Char:
+ res.setType(ctx, SqlType::Char, t1.length() + t2.length());
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Varchar:
+ res.setType(ctx, SqlType::Varchar, t1.length() + t2.length());
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Null:
+ res.setType(ctx, SqlType::Varchar, t1.length());
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Unbound:
+ res.setType(ctx, SqlType::Unbound);
+ con[1] = con[2] = res;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Varchar:
+ switch (t2.type()) {
+ case SqlType::Char:
+ res.setType(ctx, SqlType::Varchar, t1.length() + t2.length());
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Varchar:
+ res.setType(ctx, SqlType::Varchar, t1.length() + t2.length());
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Null:
+ res.setType(ctx, SqlType::Varchar, t1.length());
+ con[1] = t1;
+ con[2] = t2;
+ break;
+ case SqlType::Unbound:
+ res.setType(ctx, SqlType::Unbound);
+ con[1] = con[2] = res;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ switch (t2.type()) {
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ res.setType(ctx, SqlType::Bigint);
+ con[1] = con[2] = res;
+ if (t1.unSigned() || t2.unSigned()) {
+ con[1].unSigned(true);
+ con[2].unSigned(true);
+ }
+ break;
+ case SqlType::Real:
+ case SqlType::Double:
+ res.setType(ctx, SqlType::Double);
+ con[1] = con[2] = res;
+ break;
+ case SqlType::Null:
+ res.setType(ctx, SqlType::Null);
+ con[1] = con[2] = res;
+ break;
+ case SqlType::Unbound:
+ res.setType(ctx, SqlType::Unbound);
+ con[1] = con[2] = res;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Real:
+ case SqlType::Double:
+ switch (t2.type()) {
+ case SqlType::Smallint:
+ case SqlType::Integer:
+ case SqlType::Bigint:
+ case SqlType::Real:
+ case SqlType::Double:
+ res.setType(ctx, SqlType::Double);
+ con[1] = con[2] = res;
+ break;
+ case SqlType::Null:
+ res.setType(ctx, SqlType::Null);
+ con[1] = con[2] = res;
+ break;
+ case SqlType::Unbound:
+ res.setType(ctx, SqlType::Unbound);
+ con[1] = con[2] = res;
+ break;
+ default:
+ break;
+ }
+ break;
+ case SqlType::Null:
+ switch (t2.type()) {
+ case SqlType::Char:
+ case SqlType::Varchar:
+ res.setType(ctx, SqlType::Varchar, t2.length());
+ con[1] = con[2] = res;
+ break;
+ case SqlType::Unbound:
+ res.setType(ctx, SqlType::Unbound);
+ con[1] = con[2] = res;
+ break;
+ default:
+ res.setType(ctx, SqlType::Null);
+ con[1] = con[2] = res;
+ break;
+ }
+ break;
+ case SqlType::Unbound:
+ res.setType(ctx, SqlType::Unbound);
+ con[1] = con[2] = res;
+ break;
+ default:
+ break;
+ }
+ if (con[1].type() == SqlType::Undef || con[2].type() == SqlType::Undef) {
+ char b1[40], b2[40];
+ t1.print(b1, sizeof(b1));
+ t2.print(b2, sizeof(b2));
+ ctx.pushStatus(Error::Gen, "type mismatch in operation: %s %s %s", b1, m_op.name(), b2);
+ return 0;
+ }
+ } else {
+ ctx_assert(false);
+ return 0;
+ }
+ if (! ctx.ok())
+ return 0;
+ // insert required conversions
+ for (unsigned i = 1; i <= arity; i++) {
+ if (con[i].type() == SqlType::Undef) {
+ ctx.pushStatus(Error::Gen, "mismatched types in operation");
+ return 0;
+ }
+ if (con[i].type() == SqlType::Unbound) {
+ // parameter type not yet bound
+ continue;
+ }
+ Plan_expr_conv* exprConv = new Plan_expr_conv(m_root, con[i]);
+ m_root->saveNode(exprConv);
+ exprConv->setExpr(m_expr[i]);
+ m_expr[i] = static_cast<Plan_expr*>(exprConv->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_expr[i] != 0);
+ }
+ // set result type
+ m_sqlType = res;
+ // table dependencies are union from operands
+ for (unsigned i = 1; i <= arity; i++) {
+ const TableSet& ts = m_expr[i]->tableSet();
+ m_tableSet.insert(ts.begin(), ts.end());
+ }
+ // set alias name XXX misses operator precedence
+ if (arity == 1) {
+ m_alias.assign(m_op.name());
+ m_alias.append(m_expr[1]->m_alias);
+ } else if (arity == 2) {
+ m_alias.assign(m_expr[1]->m_alias);
+ m_alias.append(m_op.name());
+ m_alias.append(m_expr[2]->m_alias);
+ }
+ return this;
+}
+
+Exec_base*
+Plan_expr_op::codegen(Ctx& ctx, Ctl& ctl)
+{
+ if (m_exec != 0)
+ return m_exec;
+ unsigned arity = m_op.arity();
+ Exec_expr_op* exec = new Exec_expr_op(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // create code for operands
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ Exec_expr* execExpr = static_cast<Exec_expr*>(m_expr[i]->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ exec->setExpr(i, execExpr);
+ }
+ // create the code
+ SqlSpec sqlSpec(sqlType(), SqlSpec::Physical);
+ Exec_expr_op::Code& code = *new Exec_expr_op::Code(m_op, sqlSpec);
+ exec->setCode(code);
+ m_exec = exec;
+ return exec;
+}
+
+void
+Plan_expr_op::print(Ctx& ctx)
+{
+ ctx.print(" [%s", m_op.name());
+ Plan_base* a[] = { m_expr[1], m_expr[2] };
+ printList(ctx, a, m_op.arity());
+ ctx.print("]");
+}
+
+bool
+Plan_expr_op::isEqual(const Plan_expr* expr) const
+{
+ ctx_assert(expr != 0);
+ if (expr->type() != Plan_expr::TypeOp)
+ return false;
+ const Plan_expr_op* expr2 = static_cast<const Plan_expr_op*>(expr);
+ if (m_op.m_opcode != expr2->m_op.m_opcode)
+ return false;
+ const unsigned arity = m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ if (! m_expr[i]->isEqual(expr2->m_expr[i]))
+ return false;
+ }
+ return true;
+}
+
+bool
+Plan_expr_op::isGroupBy(const Plan_expr_row* row) const
+{
+ if (isAnyEqual(row))
+ return true;
+ const unsigned arity = m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ if (! m_expr[i]->isGroupBy(row))
+ return false;
+ }
+ return true;
+}
+
+// Code_expr_op
+
+Exec_expr_op::Code::~Code()
+{
+}
+
+Exec_expr_op::Data::~Data()
+{
+}
+
+Exec_expr_op::~Exec_expr_op()
+{
+}
+
+void
+Exec_expr_op::alloc(Ctx& ctx, Ctl& ctl)
+{
+ if (m_data != 0)
+ return;
+ const Code& code = getCode();
+ // allocate subexpressions
+ unsigned arity = code.m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ m_expr[i]->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ SqlField sqlField(code.m_sqlSpec);
+ Data& data = *new Data(sqlField);
+ setData(data);
+}
+
+void
+Exec_expr_op::close(Ctx& ctx)
+{
+ const Code& code = getCode();
+ unsigned arity = code.m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_expr[i] != 0);
+ m_expr[i]->close(ctx);
+ }
+ Data& data = getData();
+ data.m_groupField.clear();
+}
+
+void
+Exec_expr_op::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [%s", code.m_op.name());
+ Exec_base* a[] = { m_expr[1], m_expr[2] };
+ printList(ctx, a, code.m_op.arity());
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_expr_op.hpp b/ndb/src/client/odbc/codegen/Code_expr_op.hpp
new file mode 100644
index 00000000000..f9686cad151
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_op.hpp
@@ -0,0 +1,166 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_expr_op_hpp
+#define ODBC_CODEGEN_Code_expr_op_hpp
+
+#include <common/common.hpp>
+#include <common/DataField.hpp>
+#include "Code_expr.hpp"
+
+/**
+ * @class Expr_op
+ * @brief Arithmetic and string operators
+ */
+struct Expr_op {
+ enum Opcode {
+ Add = 1, // binary
+ Subtract,
+ Multiply,
+ Divide,
+ Plus, // unary
+ Minus
+ };
+ Expr_op(Opcode opcode);
+ const char* name() const;
+ unsigned arity() const;
+ Opcode m_opcode;
+};
+
+inline
+Expr_op::Expr_op(Opcode opcode) :
+ m_opcode(opcode)
+{
+}
+
+/**
+ * @class Plan_expr_op
+ * @brief Operator node in an expression in PlanTree
+ */
+class Plan_expr_op : public Plan_expr {
+public:
+ Plan_expr_op(Plan_root* root, Expr_op op);
+ virtual ~Plan_expr_op();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ bool isEqual(const Plan_expr* expr) const;
+ bool isGroupBy(const Plan_expr_row* row) const;
+ // children
+ void setExpr(unsigned i, Plan_expr* expr);
+protected:
+ Expr_op m_op;
+ Plan_expr* m_expr[1 + 2];
+};
+
+inline
+Plan_expr_op::Plan_expr_op(Plan_root* root, Expr_op op) :
+ Plan_expr(root, TypeOp),
+ m_op(op)
+{
+ m_expr[0] = m_expr[1] = m_expr[2] = 0;
+}
+
+inline void
+Plan_expr_op::setExpr(unsigned i, Plan_expr* expr)
+{
+ ctx_assert(1 <= i && i <= 2 && expr != 0);
+ m_expr[i] = expr;
+}
+
+/**
+ * @class Exec_expr_op
+ * @brief Operator node in an expression in ExecTree
+ */
+class Exec_expr_op : public Exec_expr {
+public:
+ class Code : public Exec_expr::Code {
+ public:
+ Code(Expr_op op, const SqlSpec& spec);
+ virtual ~Code();
+ protected:
+ friend class Exec_expr_op;
+ Expr_op m_op;
+ const SqlSpec m_sqlSpec;
+ };
+ class Data : public Exec_expr::Data {
+ public:
+ Data(const SqlField& sqlField);
+ virtual ~Data();
+ protected:
+ friend class Exec_expr_op;
+ SqlField m_sqlField;
+ };
+ Exec_expr_op(Exec_root* root);
+ virtual ~Exec_expr_op();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void evaluate(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setExpr(unsigned i, Exec_expr* expr);
+protected:
+ Exec_expr* m_expr[1 + 2];
+};
+
+inline
+Exec_expr_op::Code::Code(Expr_op op, const SqlSpec& sqlSpec) :
+ Exec_expr::Code(m_sqlSpec),
+ m_op(op),
+ m_sqlSpec(sqlSpec)
+{
+}
+
+inline
+Exec_expr_op::Data::Data(const SqlField& sqlField) :
+ Exec_expr::Data(m_sqlField),
+ m_sqlField(sqlField)
+{
+}
+
+inline
+Exec_expr_op::Exec_expr_op(Exec_root* root) :
+ Exec_expr(root)
+{
+ m_expr[0] = m_expr[1] = m_expr[2] = 0;
+}
+
+// children
+
+inline const Exec_expr_op::Code&
+Exec_expr_op::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_expr_op::Data&
+Exec_expr_op::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_expr_op::setExpr(unsigned i, Exec_expr* expr)
+{
+ ctx_assert(1 <= i && i <= 2 && m_expr[i] == 0);
+ m_expr[i] = expr;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_expr_param.cpp b/ndb/src/client/odbc/codegen/Code_expr_param.cpp
new file mode 100644
index 00000000000..93892cae5e6
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_param.cpp
@@ -0,0 +1,279 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_expr_param.hpp"
+#include "Code_root.hpp"
+
+// Plan_expr_param
+
+Plan_expr_param::~Plan_expr_param()
+{
+}
+
+Plan_base*
+Plan_expr_param::analyze(Ctx& ctx, Ctl& ctl)
+{
+ m_exec = 0;
+ ctx_assert(m_paramNumber != 0);
+ ctx_assert(m_paramNumber < m_root->m_paramList.size());
+ m_root->m_paramList[m_paramNumber] = this;
+ m_sqlType.setType(ctx, SqlType::Unbound);
+ // check if type is bound now
+ DescArea& ipd = descArea(Desc_usage_IPD);
+ if (m_paramNumber <= ipd.getCount()) {
+ DescRec& rec = ipd.getRecord(m_paramNumber);
+ OdbcData descData;
+ rec.getField(ctx, SQL_DESC_TYPE, descData);
+ if (descData.type() != OdbcData::Undef) {
+ SQLSMALLINT desc_TYPE = descData.smallint();
+ // XXX wrong but fixes sun.jdbc.odbc
+ if (desc_TYPE == SQL_CHAR)
+ desc_TYPE = SQL_VARCHAR;
+ if (desc_TYPE == SQL_CHAR) {
+ rec.getField(ctx, SQL_DESC_LENGTH, descData);
+ if (descData.type() != OdbcData::Undef) {
+ unsigned desc_LENGTH = descData.uinteger();
+ m_sqlType.setType(ctx, SqlType::Char, desc_LENGTH);
+ }
+ } else if (desc_TYPE == SQL_VARCHAR) {
+ rec.getField(ctx, SQL_DESC_LENGTH, descData);
+ if (descData.type() != OdbcData::Undef) {
+ unsigned desc_LENGTH = descData.uinteger();
+ m_sqlType.setType(ctx, SqlType::Varchar, desc_LENGTH);
+ }
+ } else if (desc_TYPE == SQL_BINARY) {
+ rec.getField(ctx, SQL_DESC_LENGTH, descData);
+ if (descData.type() != OdbcData::Undef) {
+ unsigned desc_LENGTH = descData.uinteger();
+ m_sqlType.setType(ctx, SqlType::Binary, desc_LENGTH);
+ }
+ } else if (desc_TYPE == SQL_VARBINARY) {
+ rec.getField(ctx, SQL_DESC_LENGTH, descData);
+ if (descData.type() != OdbcData::Undef) {
+ unsigned desc_LENGTH = descData.uinteger();
+ m_sqlType.setType(ctx, SqlType::Varbinary, desc_LENGTH);
+ } else {
+ // XXX BLOB hack
+ unsigned desc_LENGTH = FAKE_BLOB_SIZE;
+ m_sqlType.setType(ctx, SqlType::Varbinary, desc_LENGTH);
+ }
+ } else if (desc_TYPE == SQL_SMALLINT) {
+ m_sqlType.setType(ctx, SqlType::Smallint);
+ } else if (desc_TYPE == SQL_INTEGER) {
+ m_sqlType.setType(ctx, SqlType::Integer);
+ } else if (desc_TYPE == SQL_BIGINT) {
+ m_sqlType.setType(ctx, SqlType::Bigint);
+ } else if (desc_TYPE == SQL_REAL) {
+ m_sqlType.setType(ctx, SqlType::Real);
+ } else if (desc_TYPE == SQL_DOUBLE) {
+ m_sqlType.setType(ctx, SqlType::Double);
+ } else if (desc_TYPE == SQL_TYPE_TIMESTAMP) {
+ m_sqlType.setType(ctx, SqlType::Datetime);
+ // XXX BLOB hack
+ } else if (desc_TYPE == SQL_LONGVARBINARY) {
+ m_sqlType.setType(ctx, SqlType::Varbinary, (unsigned)FAKE_BLOB_SIZE);
+ } else {
+ ctx.pushStatus(Error::Gen, "parameter %u unsupported SQL type %d", m_paramNumber, (int)desc_TYPE);
+ return 0;
+ }
+ char buf[100];
+ m_sqlType.print(buf, sizeof(buf));
+ ctx_log2(("parameter %u SQL type bound to %s", m_paramNumber, buf));
+ }
+ }
+ return this;
+}
+
+void
+Plan_expr_param::describe(Ctx& ctx)
+{
+ DescArea& ipd = descArea(Desc_usage_IPD);
+ if (ipd.getCount() < m_paramNumber)
+ ipd.setCount(ctx, m_paramNumber);
+ // XXX describe if possible
+ DescRec& rec = ipd.getRecord(m_paramNumber);
+}
+
+Exec_base*
+Plan_expr_param::codegen(Ctx& ctx, Ctl& ctl)
+{
+ if (m_exec != 0)
+ return m_exec;
+ SqlSpec sqlSpec(m_sqlType, SqlSpec::Physical);
+ // create code
+ Exec_expr_param* exec = new Exec_expr_param(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ ctx_assert(m_paramNumber != 0);
+ Exec_expr_param::Code& code = *new Exec_expr_param::Code(sqlSpec, m_paramNumber);
+ exec->setCode(code);
+ m_exec = exec;
+ return exec;
+}
+
+void
+Plan_expr_param::print(Ctx& ctx)
+{
+ ctx.print(" [param %u]", m_paramNumber);
+}
+
+bool
+Plan_expr_param::isEqual(const Plan_expr* expr) const
+{
+ ctx_assert(expr != 0);
+ if (expr->type() != Plan_expr::TypeParam)
+ return false;
+ const Plan_expr_param* expr2 = static_cast<const Plan_expr_param*>(expr);
+ // params are not equal ever
+ return false;
+}
+
+bool
+Plan_expr_param::isGroupBy(const Plan_expr_row* row) const
+{
+ // params are constants
+ return true;
+}
+
+// Exec_expr_param
+
+Exec_expr_param::Code::~Code()
+{
+}
+
+Exec_expr_param::Data::~Data()
+{
+ delete m_extField;
+ m_extField = 0;
+}
+
+Exec_expr_param::~Exec_expr_param()
+{
+}
+
+void
+Exec_expr_param::alloc(Ctx& ctx, Ctl& ctl)
+{
+ if (m_data != 0)
+ return;
+ const Code& code = getCode();
+ SqlField sqlField(code.sqlSpec());
+ Data& data = *new Data(sqlField);
+ setData(data);
+}
+
+void
+Exec_expr_param::bind(Ctx& ctx)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ DescArea& apd = descArea(Desc_usage_APD);
+ if (apd.getCount() < code.m_paramNumber) {
+ ctx_log1(("parameter %u is not bound", code.m_paramNumber));
+ return;
+ }
+ const unsigned paramNumber = code.m_paramNumber;
+ DescRec& rec = apd.getRecord(paramNumber);
+ OdbcData descData;
+ // create type
+ rec.getField(ctx, SQL_DESC_TYPE, descData);
+ if (descData.type() == OdbcData::Undef) {
+ ctx.pushStatus(Error::Gen, "parameter %u external type not defined", paramNumber);
+ return;
+ }
+ ExtType extType;
+ SQLSMALLINT desc_TYPE = descData.smallint();
+ switch (desc_TYPE) {
+ case SQL_C_CHAR:
+ case SQL_C_SHORT: // for sun.jdbc.odbc
+ case SQL_C_SSHORT:
+ case SQL_C_USHORT:
+ case SQL_C_LONG: // for sun.jdbc.odbc
+ case SQL_C_SLONG:
+ case SQL_C_ULONG:
+ case SQL_C_SBIGINT:
+ case SQL_C_UBIGINT:
+ case SQL_C_FLOAT:
+ case SQL_C_DOUBLE:
+ case SQL_C_TYPE_TIMESTAMP:
+ case SQL_C_BINARY: // XXX BLOB hack
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "parameter %u unsupported external type %d", paramNumber, (int)desc_TYPE);
+ return;
+ }
+ extType.setType(ctx, static_cast<ExtType::Type>(desc_TYPE));
+ ExtSpec extSpec(extType);
+ // create data field
+ rec.getField(ctx, SQL_DESC_DATA_PTR, descData);
+ if (descData.type() == OdbcData::Undef) {
+ ctx.pushStatus(Error::Gen, "parameter %u data address not defined", paramNumber);
+ return;
+ }
+ SQLPOINTER desc_DATA_PTR = descData.pointer();
+ rec.getField(ctx, SQL_DESC_OCTET_LENGTH, descData);
+ if (descData.type() == OdbcData::Undef) {
+ ctx.pushStatus(Error::Gen, "parameter %u data length not defined", paramNumber);
+ return;
+ }
+ SQLINTEGER desc_OCTET_LENGTH = descData.integer();
+ rec.getField(ctx, SQL_DESC_INDICATOR_PTR, descData);
+ if (descData.type() == OdbcData::Undef) {
+ ctx.pushStatus(Error::Gen, "parameter %u indicator address not defined", paramNumber);
+ return;
+ }
+ SQLINTEGER* desc_INDICATOR_PTR = descData.integerPtr();
+ ctx_log4(("parameter %u bind to 0x%x %d 0x%x", paramNumber, (unsigned)desc_DATA_PTR, (int)desc_OCTET_LENGTH, (unsigned)desc_INDICATOR_PTR));
+ ExtField& extField = *new ExtField(extSpec, desc_DATA_PTR, desc_OCTET_LENGTH, desc_INDICATOR_PTR, paramNumber);
+ data.m_atExec = false;
+ if (desc_INDICATOR_PTR != 0 && *desc_INDICATOR_PTR < 0) {
+ if (*desc_INDICATOR_PTR == SQL_NULL_DATA) {
+ ;
+ } else if (*desc_INDICATOR_PTR == SQL_DATA_AT_EXEC) {
+ data.m_atExec = true;
+ } else if (*desc_INDICATOR_PTR <= SQL_LEN_DATA_AT_EXEC(0)) {
+ data.m_atExec = true;
+ }
+ }
+ delete data.m_extField;
+ data.m_extField = &extField;
+}
+
+void
+Exec_expr_param::evaluate(Ctx& ctx, Ctl& ctl)
+{
+ if (ctl.m_postEval)
+ return;
+ const Code& code = getCode();
+ Data& data = getData();
+ if (data.m_atExec)
+ return;
+ ctx_assert(data.m_extField != 0);
+ data.m_sqlField.copyin(ctx, *data.m_extField);
+}
+
+void
+Exec_expr_param::close(Ctx& ctx)
+{
+ Data& data = getData();
+ data.m_extPos = -1;
+}
+
+void
+Exec_expr_param::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [param %u]", code.m_paramNumber);
+}
diff --git a/ndb/src/client/odbc/codegen/Code_expr_param.hpp b/ndb/src/client/odbc/codegen/Code_expr_param.hpp
new file mode 100644
index 00000000000..783e5c087b4
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_param.hpp
@@ -0,0 +1,136 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_expr_param_hpp
+#define ODBC_CODEGEN_Code_expr_param_hpp
+
+#include <common/common.hpp>
+#include <common/DataField.hpp>
+#include "Code_expr.hpp"
+
+/**
+ * @class Plan_expr_param
+ * @brief Constant expression value in PlanTree
+ */
+class Plan_expr_param : public Plan_expr {
+public:
+ Plan_expr_param(Plan_root* root, unsigned paramNumber);
+ virtual ~Plan_expr_param();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ bool isEqual(const Plan_expr* expr) const;
+ bool isGroupBy(const Plan_expr_row* row) const;
+protected:
+ const unsigned m_paramNumber;
+};
+
+inline
+Plan_expr_param::Plan_expr_param(Plan_root* root, unsigned paramNumber) :
+ Plan_expr(root, TypeParam),
+ m_paramNumber(paramNumber)
+{
+}
+
+/**
+ * @class Exec_expr_param
+ * @brief Constant expression value in ExecTree
+ */
+class Exec_expr_param : public Exec_expr {
+public:
+ class Code : public Exec_expr::Code {
+ public:
+ Code(const SqlSpec& sqlSpec, unsigned paramNumber);
+ virtual ~Code();
+ protected:
+ friend class Plan_expr_param;
+ friend class Exec_expr_param;
+ const SqlSpec m_sqlSpec;
+ const unsigned m_paramNumber;
+ };
+ class Data : public Exec_expr::Data {
+ public:
+ Data(SqlField& sqlField);
+ virtual ~Data();
+ ExtField* extField() const;
+ protected:
+ friend class Exec_expr_param;
+ friend class Exec_root;
+ SqlField m_sqlField;
+ ExtField* m_extField; // input binding
+ bool m_atExec; // data at exec
+ int m_extPos; // position for SQLPutData (-1 before first call)
+ };
+ Exec_expr_param(Exec_root* root);
+ virtual ~Exec_expr_param();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void bind(Ctx& ctx);
+ void evaluate(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_expr_param::Code::Code(const SqlSpec& sqlSpec, unsigned paramNumber) :
+ Exec_expr::Code(m_sqlSpec),
+ m_sqlSpec(sqlSpec),
+ m_paramNumber(paramNumber)
+{
+}
+
+inline
+Exec_expr_param::Data::Data(SqlField& sqlField) :
+ Exec_expr::Data(m_sqlField),
+ m_sqlField(sqlField),
+ m_extField(0),
+ m_atExec(false),
+ m_extPos(-1)
+{
+}
+
+inline ExtField*
+Exec_expr_param::Data::extField() const
+{
+ return m_extField;
+}
+
+inline
+Exec_expr_param::Exec_expr_param(Exec_root* root) :
+ Exec_expr(root)
+{
+}
+
+// children
+
+inline const Exec_expr_param::Code&
+Exec_expr_param::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_expr_param::Data&
+Exec_expr_param::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_expr_row.cpp b/ndb/src/client/odbc/codegen/Code_expr_row.cpp
new file mode 100644
index 00000000000..da1751d41d1
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_row.cpp
@@ -0,0 +1,204 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/DataType.hpp>
+#include "Code_expr_row.hpp"
+#include "Code_expr.hpp"
+#include "Code_expr_conv.hpp"
+#include "Code_dml_row.hpp"
+#include "Code_root.hpp"
+
+// Plan_expr_row
+
+Plan_expr_row::~Plan_expr_row()
+{
+}
+
+Plan_base*
+Plan_expr_row::analyze(Ctx& ctx, Ctl& ctl)
+{
+ unsigned size = getSize();
+ // analyze subexpressions
+ m_anyAggr = false;
+ m_allBound = true;
+ for (unsigned i = 1; i <= size; i++) {
+ Plan_expr* expr1 = getExpr(i);
+ Plan_expr* expr2 = static_cast<Plan_expr*>(expr1->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ setExpr(i, expr2);
+ if (expr2->isAggr())
+ m_anyAggr = true;
+ if (! expr2->isBound())
+ m_allBound = false;
+ }
+ // insert conversions if requested XXX ugly hack
+ if (ctl.m_dmlRow != 0) {
+ if (ctl.m_dmlRow->getSize() > getSize()) {
+ ctx.pushStatus(Sqlstate::_21S01, Error::Gen, "not enough values (%u > %u)", ctl.m_dmlRow->getSize(), getSize());
+ return 0;
+ }
+ if (ctl.m_dmlRow->getSize() < getSize()) {
+ ctx.pushStatus(Sqlstate::_21S01, Error::Gen, "too many values (%u < %u)", ctl.m_dmlRow->getSize(), getSize());
+ return 0;
+ }
+ for (unsigned i = 1; i <= size; i++) {
+ const SqlType& sqlType = ctl.m_dmlRow->getColumn(i)->sqlType();
+ Plan_expr_conv* exprConv = new Plan_expr_conv(m_root, sqlType);
+ m_root->saveNode(exprConv);
+ exprConv->setExpr(getExpr(i));
+ Plan_expr* expr = static_cast<Plan_expr*>(exprConv->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(expr != 0);
+ setExpr(i, expr);
+ }
+ }
+ // set aliases
+ m_aliasList.resize(1 + size);
+ for (unsigned i = 1; i <= size; i++) {
+ if (m_aliasList[i].empty()) {
+ setAlias(i, getExpr(i)->getAlias());
+ }
+ }
+ // node was not replaced
+ return this;
+}
+
+Exec_base*
+Plan_expr_row::codegen(Ctx& ctx, Ctl& ctl)
+{
+ unsigned size = getSize();
+ Exec_expr_row* exec = new Exec_expr_row(ctl.m_execRoot, size);
+ ctl.m_execRoot->saveNode(exec);
+ SqlSpecs sqlSpecs(size);
+ // create code for subexpressions
+ for (unsigned i = 1; i <= size; i++) {
+ Plan_expr* planExpr = getExpr(i);
+ Exec_expr* execExpr = static_cast<Exec_expr*>(planExpr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ exec->setExpr(i, execExpr);
+ const SqlSpec sqlSpec(execExpr->getCode().sqlSpec(), SqlSpec::Reference);
+ sqlSpecs.setEntry(i, sqlSpec);
+ }
+ // create alias list
+ Exec_expr_row::Code::Alias* aliasList = new Exec_expr_row::Code::Alias[1 + size];
+ strcpy(aliasList[0], "?");
+ for (unsigned i = 1; i <= size; i++) {
+ const char* s = m_aliasList[i].c_str();
+ if (strlen(s) == 0)
+ s = getExpr(i)->getAlias().c_str();
+ unsigned n = strlen(s);
+ if (n >= sizeof(Exec_expr_row::Code::Alias))
+ n = sizeof(Exec_expr_row::Code::Alias) - 1;
+ strncpy(aliasList[i], s, n);
+ aliasList[i][n] = 0;
+ }
+ // create the code
+ Exec_expr_row::Code& code = *new Exec_expr_row::Code(sqlSpecs, aliasList);
+ exec->setCode(code);
+ return exec;
+}
+
+void
+Plan_expr_row::print(Ctx& ctx)
+{
+ const unsigned size = getSize();
+ ctx.print(" [expr_row");
+ for (unsigned i = 1; i <= size; i++) {
+ Plan_base* a = m_exprList[i];
+ a == 0 ? ctx.print(" -") : a->print(ctx);
+ }
+ ctx.print("]");
+}
+
+bool
+Plan_expr_row::isAllGroupBy(const Plan_expr_row* row) const
+{
+ const unsigned size = getSize();
+ for (unsigned i = 1; i <= size; i++) {
+ if (! getExpr(i)->isGroupBy(row))
+ return false;
+ }
+ return true;
+}
+
+// Exec_expr_row
+
+Exec_expr_row::Code::~Code()
+{
+ delete[] m_aliasList;
+}
+
+Exec_expr_row::Data::~Data()
+{
+}
+
+Exec_expr_row::~Exec_expr_row()
+{
+ delete[] m_expr;
+}
+
+void
+Exec_expr_row::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate subexpressions
+ for (unsigned i = 1; i <= m_size; i++) {
+ getExpr(i)->alloc(ctx, ctl);
+ }
+ // construct SqlRow of references
+ const Code& code = getCode();
+ SqlRow sqlRow(getCode().m_sqlSpecs);
+ for (unsigned i = 1; i <= m_size; i++) {
+ const Exec_expr::Data& dataExpr = getExpr(i)->getData();
+ const SqlSpec& sqlSpec = code.m_sqlSpecs.getEntry(i);
+ const SqlField sqlField(sqlSpec, &dataExpr.sqlField());
+ sqlRow.setEntry(i, sqlField);
+ }
+ // create the data
+ Data& data = *new Data(sqlRow);
+ setData(data);
+}
+
+void
+Exec_expr_row::evaluate(Ctx& ctx, Ctl& ctl)
+{
+ for (unsigned i = 1; i <= m_size; i++) {
+ getExpr(i)->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+}
+
+void
+Exec_expr_row::close(Ctx& ctx)
+{
+ for (unsigned i = 1; i <= m_size; i++) {
+ getExpr(i)->close(ctx);
+ }
+}
+
+void
+Exec_expr_row::print(Ctx& ctx)
+{
+ ctx.print(" [expr_row");
+ for (unsigned i = 1; i <= m_size; i++) {
+ getExpr(i)->print(ctx);
+ }
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_expr_row.hpp b/ndb/src/client/odbc/codegen/Code_expr_row.hpp
new file mode 100644
index 00000000000..94527931dba
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_expr_row.hpp
@@ -0,0 +1,272 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_expr_row_hpp
+#define ODBC_CODEGEN_Code_expr_row_hpp
+
+#include <vector>
+#include <common/common.hpp>
+#include <common/DataRow.hpp>
+#include "Code_base.hpp"
+#include "Code_expr.hpp"
+
+class Plan_expr;
+
+/**
+ * @class Plan_expr_row
+ * @brief Row of expressions in PlanTree
+ *
+ * Used for select, value, and order by rows.
+ */
+class Plan_expr_row : public Plan_base {
+public:
+ Plan_expr_row(Plan_root* root);
+ virtual ~Plan_expr_row();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setAsterisk();
+ bool getAsterisk() const;
+ unsigned getSize() const;
+ Plan_expr* getExpr(unsigned i) const;
+ void setExpr(unsigned i, Plan_expr* expr);
+ void addExpr(Plan_expr* expr);
+ void addExpr(Plan_expr* expr, const BaseString& alias);
+ void setAlias(unsigned i, const BaseString& alias);
+ void addExpr(Plan_expr* expr, bool asc);
+ bool anyAggr() const;
+ bool allBound() const;
+ bool isAllGroupBy(const Plan_expr_row* row) const;
+protected:
+ friend class Plan_query;
+ friend class Plan_query_sort;
+ bool m_asterisk; // early plan node type
+ ExprVector m_exprList;
+ typedef std::vector<BaseString> AliasList;
+ AliasList m_aliasList;
+ typedef std::vector<bool> AscList;
+ AscList m_ascList;
+ bool m_anyAggr; // at least one aggreate
+ bool m_allBound; // all bound
+};
+
+inline
+Plan_expr_row::Plan_expr_row(Plan_root* root) :
+ Plan_base(root),
+ m_asterisk(false),
+ m_exprList(1),
+ m_aliasList(1),
+ m_ascList(1),
+ m_anyAggr(false),
+ m_allBound(false)
+{
+}
+
+// children
+
+inline void
+Plan_expr_row::setAsterisk()
+{
+ m_asterisk = true;
+}
+
+inline bool
+Plan_expr_row::getAsterisk() const
+{
+ return m_asterisk;
+}
+
+inline unsigned
+Plan_expr_row::getSize() const
+{
+ ctx_assert(m_exprList.size() >= 1);
+ return m_exprList.size() - 1;
+}
+
+inline void
+Plan_expr_row::addExpr(Plan_expr* expr)
+{
+ ctx_assert(expr != 0);
+ addExpr(expr, expr->m_alias);
+}
+
+inline void
+Plan_expr_row::addExpr(Plan_expr* expr, const BaseString& alias)
+{
+ ctx_assert(expr != 0);
+ m_exprList.push_back(expr);
+ m_aliasList.push_back(alias);
+}
+
+inline void
+Plan_expr_row::addExpr(Plan_expr* expr, bool asc)
+{
+ ctx_assert(expr != 0);
+ m_exprList.push_back(expr);
+ m_ascList.push_back(asc);
+}
+
+inline void
+Plan_expr_row::setExpr(unsigned i, Plan_expr* expr)
+{
+ ctx_assert(1 <= i && i < m_exprList.size() && expr != 0);
+ m_exprList[i] = expr;
+}
+
+inline Plan_expr*
+Plan_expr_row::getExpr(unsigned i) const
+{
+ ctx_assert(1 <= i && i < m_exprList.size() && m_exprList[i] != 0);
+ return m_exprList[i];
+}
+
+inline void
+Plan_expr_row::setAlias(unsigned i, const BaseString& alias)
+{
+ ctx_assert(1 <= i && i < m_aliasList.size());
+ m_aliasList[i] = alias;
+}
+
+inline bool
+Plan_expr_row::anyAggr() const
+{
+ return m_anyAggr;
+}
+
+inline bool
+Plan_expr_row::allBound() const
+{
+ return m_allBound;
+}
+
+/**
+ * @class Expr_expr_row
+ * @brief Row of expressions in ExecTree
+ */
+class Exec_expr_row : public Exec_base {
+public:
+ class Code : public Exec_base::Code {
+ public:
+ typedef char Alias[40];
+ Code(const SqlSpecs& sqlSpecs, const Alias* aliasList);
+ virtual ~Code();
+ const SqlSpecs& sqlSpecs() const;
+ const char* getAlias(unsigned i) const;
+ protected:
+ friend class Exec_expr_row;
+ const SqlSpecs m_sqlSpecs;
+ const Alias* m_aliasList;
+ };
+ class Data : public Exec_base::Data {
+ public:
+ Data(const SqlRow& sqlRow);
+ virtual ~Data();
+ const SqlRow& sqlRow() const;
+ protected:
+ friend class Exec_expr_row;
+ SqlRow m_sqlRow;
+ };
+ Exec_expr_row(Exec_root* root, unsigned size);
+ virtual ~Exec_expr_row();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void evaluate(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ Exec_expr* getExpr(unsigned i) const;
+ void setExpr(unsigned i, Exec_expr* expr);
+protected:
+ Exec_expr** m_expr; // numbered from 1
+ unsigned m_size;
+};
+
+inline
+Exec_expr_row::Code::Code(const SqlSpecs& sqlSpecs, const Alias* aliasList) :
+ m_sqlSpecs(sqlSpecs),
+ m_aliasList(aliasList)
+{
+}
+
+inline const SqlSpecs&
+Exec_expr_row::Code::sqlSpecs() const
+{
+ return m_sqlSpecs;
+}
+
+inline const char*
+Exec_expr_row::Code::getAlias(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= m_sqlSpecs.count() && m_aliasList != 0);
+ return m_aliasList[i];
+}
+
+inline
+Exec_expr_row::Data::Data(const SqlRow& sqlRow) :
+ m_sqlRow(sqlRow)
+{
+}
+
+inline const SqlRow&
+Exec_expr_row::Data::sqlRow() const
+{
+ return m_sqlRow;
+}
+
+inline
+Exec_expr_row::Exec_expr_row(Exec_root* root, unsigned size) :
+ Exec_base(root),
+ m_expr(new Exec_expr* [1 + size]),
+ m_size(size)
+{
+ m_expr[0] = (Exec_expr*)-1;
+ for (unsigned i = 1; i <= m_size; i++)
+ m_expr[i] = 0;
+}
+
+// children
+
+inline const Exec_expr_row::Code&
+Exec_expr_row::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_expr_row::Data&
+Exec_expr_row::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline Exec_expr*
+Exec_expr_row::getExpr(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= m_size && m_expr != 0 && m_expr[i] != 0);
+ return m_expr[i];
+}
+
+inline void
+Exec_expr_row::setExpr(unsigned i, Exec_expr* expr)
+{
+ ctx_assert(1 <= i && i <= m_size && m_expr != 0 && m_expr[i] == 0);
+ m_expr[i] = expr;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_idx_column.cpp b/ndb/src/client/odbc/codegen/Code_idx_column.cpp
new file mode 100644
index 00000000000..584ffef3e01
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_idx_column.cpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include "Code_idx_column.hpp"
+#include "Code_expr_conv.hpp"
+#include "Code_root.hpp"
+
+// Plan_idx_column
+
+Plan_idx_column::~Plan_idx_column()
+{
+}
+
+Plan_base*
+Plan_idx_column::analyze(Ctx& ctx, Ctl& ctl)
+{
+ analyzeColumn(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_idx_column::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_idx_column::print(Ctx& ctx)
+{
+ ctx.print(" [idx_column %s]", getPrintName());
+}
diff --git a/ndb/src/client/odbc/codegen/Code_idx_column.hpp b/ndb/src/client/odbc/codegen/Code_idx_column.hpp
new file mode 100644
index 00000000000..209ed705b48
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_idx_column.hpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_idx_column_hpp
+#define ODBC_CODEGEN_Code_idx_column_hpp
+
+#include <common/common.hpp>
+#include "Code_column.hpp"
+#include "Code_data_type.hpp"
+#include "Code_expr.hpp"
+
+class DictColumn;
+class Plan_table;
+
+/**
+ * @class Plan_idx_column
+ * @brief Column in create index statement
+ */
+class Plan_idx_column : public Plan_base, public Plan_column {
+public:
+ Plan_idx_column(Plan_root* root, const BaseString& name);
+ virtual ~Plan_idx_column();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+protected:
+ friend class Plan_create_row;
+};
+
+inline
+Plan_idx_column::Plan_idx_column(Plan_root* root, const BaseString& name) :
+ Plan_base(root),
+ Plan_column(Type_idx, name)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_insert.cpp b/ndb/src/client/odbc/codegen/Code_insert.cpp
new file mode 100644
index 00000000000..c442186c181
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_insert.cpp
@@ -0,0 +1,253 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_insert.hpp"
+#include "Code_query_repeat.hpp"
+#include "Code_query_project.hpp"
+#include "Code_table.hpp"
+#include "Code_root.hpp"
+
+// Plan_insert
+
+Plan_insert::~Plan_insert()
+{
+}
+
+Plan_base*
+Plan_insert::analyze(Ctx& ctx, Ctl& ctl)
+{
+ stmtArea().stmtInfo().setName(Stmt_name_insert);
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ // handle MySql syntax
+ if (m_mysqlRow != 0) {
+ setDmlRow(m_mysqlRow->m_dmlRow);
+ setExprRow(m_mysqlRow->m_exprRow);
+ m_mysqlRow = 0;
+ }
+ if (m_dmlRow == 0) {
+ // construct column list
+ setDmlRow(new Plan_dml_row(m_root));
+ m_root->saveNode(m_dmlRow);
+ const DictTable& dictTable = m_table->dictTable();
+ unsigned n = dictTable.getSize();
+ for (unsigned i = 1; i <= n; i++) {
+ DictColumn* dictColumn = dictTable.getColumn(i);
+ Plan_dml_column* column = new Plan_dml_column(m_root, dictColumn->getName());
+ m_root->saveNode(column);
+ m_dmlRow->addColumn(column);
+ }
+ }
+ // set name resolution scope
+ ctl.m_tableList.resize(1 + 1); // indexed from 1
+ ctl.m_tableList[1] = m_table;
+ // analyze the dml columns
+ m_dmlRow->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctl.m_dmlRow = m_dmlRow; // row type to convert to
+ if (m_query != 0) {
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ } else if (m_select == 0) {
+ // analyze the expression row
+ m_exprRow->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ // transform the row into query
+ Plan_query_repeat* queryRepeat = new Plan_query_repeat(m_root, 1);
+ m_root->saveNode(queryRepeat);
+ Plan_query_project* queryProject = new Plan_query_project(m_root);
+ m_root->saveNode(queryProject);
+ queryProject->setQuery(queryRepeat);
+ queryProject->setRow(m_exprRow);
+ setQuery(queryProject);
+ } else {
+ // analyze the select into query
+ Plan_query* query = static_cast<Plan_query*>(m_select->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ setQuery(query);
+ }
+ return this;
+}
+
+void
+Plan_insert::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "INSERT", SQL_DIAG_INSERT);
+}
+
+Exec_base*
+Plan_insert::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // create code for the query
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // set up
+ ctx_assert(m_table != 0);
+ const BaseString& tableName = m_table->getName();
+ const DictTable& dictTable = m_table->dictTable();
+ const ColumnVector& columns = m_table->dmlColumns();
+ ctx_assert(columns.size() > 0);
+ const unsigned attrCount = columns.size() - 1;
+ // create the code
+ Exec_insert::Code& code = *new Exec_insert::Code();
+ code.m_insertOp = m_insertOp;
+ code.m_tableName = strcpy(new char[tableName.length() + 1], tableName.c_str());
+ code.m_attrCount = attrCount;
+ code.m_attrId = new NdbAttrId[1 + attrCount];
+ code.m_isKey = new bool[1 + attrCount];
+ code.m_attrId[0] = (NdbAttrId)-1;
+ code.m_tupleId = dictTable.tupleId(); // maybe 0
+ code.m_autoIncrement = dictTable.autoIncrement(); // maybe 0
+ unsigned k;
+ if ((k = code.m_tupleId) != 0 || (k = code.m_autoIncrement) != 0) {
+ const DictColumn& dictColumn = *dictTable.getColumn(k);
+ code.m_idType = dictColumn.sqlType();
+ }
+ for (unsigned i = 1; i <= attrCount; i++) {
+ Plan_column* column = columns[i];
+ ctx_assert(column != 0);
+ const DictColumn& dictColumn = column->dictColumn();
+ code.m_attrId[i] = dictColumn.getAttrId();
+ code.m_isKey[i] = dictColumn.isKey();
+ }
+ // default values XXX a mess
+ code.m_defaultCount = 0;
+ for (unsigned j = 1; j <= dictTable.getSize(); j++) {
+ const DictColumn& dictColumn = *dictTable.getColumn(j);
+ if (dictColumn.getDefaultValue() != 0 && ! code.findAttrId(dictColumn.getAttrId()))
+ code.m_defaultCount++;
+ }
+ if (code.m_defaultCount != 0) {
+ code.m_defaultId = new NdbAttrId[1 + code.m_defaultCount];
+ for (unsigned i = 0, j = 1; j <= dictTable.getSize(); j++) {
+ const DictColumn& dictColumn = *dictTable.getColumn(j);
+ if (dictColumn.getDefaultValue() != 0 && ! code.findAttrId(dictColumn.getAttrId()))
+ code.m_defaultId[++i] = dictColumn.getAttrId();
+ }
+ SqlSpecs sqlSpecs(code.m_defaultCount);
+ for (unsigned i = 0, j = 1; j <= dictTable.getSize(); j++) {
+ const DictColumn& dictColumn = *dictTable.getColumn(j);
+ if (dictColumn.getDefaultValue() != 0 && ! code.findAttrId(dictColumn.getAttrId())) {
+ SqlSpec sqlSpec(dictColumn.sqlType(), SqlSpec::Physical);
+ sqlSpecs.setEntry(++i, sqlSpec);
+ }
+ }
+ code.m_defaultValue = new SqlRow(sqlSpecs);
+ for (unsigned i = 0, j = 1; j <= dictTable.getSize(); j++) {
+ const DictColumn& dictColumn = *dictTable.getColumn(j);
+ if (dictColumn.getDefaultValue() != 0 && ! code.findAttrId(dictColumn.getAttrId())) {
+ const char* defaultValue = dictColumn.getDefaultValue();
+ ExtType extType(ExtType::Char);
+ ExtSpec extSpec(extType);
+ SQLINTEGER ind = SQL_NTS;
+ ExtField extField(extSpec, (SQLPOINTER)defaultValue, 0, &ind);
+ SqlField& f = code.m_defaultValue->getEntry(++i);
+ f.copyin(ctx, extField);
+ if (! ctx.ok())
+ return 0;
+ }
+ }
+ }
+ // create the exec
+ Exec_insert* exec = new Exec_insert(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ return exec;
+}
+
+void
+Plan_insert::print(Ctx& ctx)
+{
+ ctx.print(" [%s", m_insertOp == Insert_op_insert ? "insert" : "write");
+ Plan_base* a[] = { m_table, m_dmlRow, m_exprRow, m_query };
+ printList(ctx, a, 4);
+ ctx.print("]");
+}
+
+// Exec_insert
+
+Exec_insert::Code::~Code()
+{
+ delete[] m_tableName;
+ delete[] m_attrId;
+ delete[] m_isKey;
+ delete[] m_defaultId;
+ delete m_defaultValue;
+}
+
+bool
+Exec_insert::Code::findAttrId(NdbAttrId attrId) const
+{
+ for (unsigned i = 1; i <= m_attrCount; i++) {
+ if (m_attrId[i] == attrId)
+ return true;
+ }
+ return false;
+}
+
+Exec_insert::Data::~Data()
+{
+}
+
+Exec_insert::~Exec_insert()
+{
+}
+
+void
+Exec_insert::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate the query
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ // create data
+ Data& data = *new Data();
+ setData(data);
+}
+
+void
+Exec_insert::close(Ctx& ctx)
+{
+}
+
+void
+Exec_insert::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx_assert(m_query != 0);
+ ctx.print(" [insert");
+ ctx.print(" attrId=");
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_attrId[i]);
+ }
+ ctx.print(" table=%s", code.m_tableName);
+ m_query->print(ctx);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_insert.hpp b/ndb/src/client/odbc/codegen/Code_insert.hpp
new file mode 100644
index 00000000000..748b092e33a
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_insert.hpp
@@ -0,0 +1,229 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_insert_hpp
+#define ODBC_CODEGEN_Code_insert_hpp
+
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_dml.hpp"
+#include "Code_dml_row.hpp"
+#include "Code_expr_row.hpp"
+#include "Code_select.hpp"
+#include "Code_query.hpp"
+#include "Code_table.hpp"
+#include "Code_set_row.hpp"
+
+enum Insert_op {
+ Insert_op_undef = 0,
+ Insert_op_insert = 1,
+ Insert_op_write = 2
+};
+
+/**
+ * @class Plan_insert
+ * @brief Insert in PlanTree
+ *
+ * Insert. Becomes directly executable.
+ */
+class Plan_insert : public Plan_dml {
+public:
+ Plan_insert(Plan_root* root, Insert_op insertOp);
+ virtual ~Plan_insert();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setTable(Plan_table* table);
+ Plan_dml_row* getDmlRow() const;
+ void setDmlRow(Plan_dml_row* dmlRow);
+ void setExprRow(Plan_expr_row* exprRow);
+ void setSelect(Plan_select* select);
+ void setQuery(Plan_query* query);
+ void setMysqlRow(Plan_set_row* mysqlRow);
+protected:
+ Insert_op m_insertOp;
+ Plan_table* m_table;
+ Plan_dml_row* m_dmlRow;
+ Plan_expr_row* m_exprRow;
+ Plan_select* m_select;
+ Plan_query* m_query;
+ Plan_set_row* m_mysqlRow;
+};
+
+inline
+Plan_insert::Plan_insert(Plan_root* root, Insert_op insertOp) :
+ Plan_dml(root),
+ m_insertOp(insertOp),
+ m_table(0),
+ m_dmlRow(0),
+ m_exprRow(0),
+ m_select(0),
+ m_query(0),
+ m_mysqlRow(0)
+{
+}
+
+// children
+
+inline void
+Plan_insert::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+inline void
+Plan_insert::setDmlRow(Plan_dml_row* dmlRow)
+{
+ ctx_assert(dmlRow != 0);
+ m_dmlRow = dmlRow;
+}
+
+inline Plan_dml_row*
+Plan_insert::getDmlRow() const
+{
+ ctx_assert(m_dmlRow != 0);
+ return m_dmlRow;
+}
+
+inline void
+Plan_insert::setExprRow(Plan_expr_row* exprRow)
+{
+ ctx_assert(exprRow != 0);
+ m_exprRow = exprRow;
+}
+
+inline void
+Plan_insert::setSelect(Plan_select* select)
+{
+ ctx_assert(select != 0);
+ m_select = select;
+}
+
+inline void
+Plan_insert::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Plan_insert::setMysqlRow(Plan_set_row* mysqlRow)
+{
+ ctx_assert(mysqlRow != 0);
+ m_mysqlRow = mysqlRow;
+}
+
+/**
+ * @class Exec_insert
+ * @brief Executable insert
+ */
+class Exec_insert : public Exec_dml {
+public:
+ class Code : public Exec_dml::Code {
+ public:
+ Code();
+ virtual ~Code();
+ protected:
+ friend class Plan_insert;
+ friend class Exec_insert;
+ Insert_op m_insertOp;
+ char* m_tableName;
+ unsigned m_attrCount;
+ NdbAttrId* m_attrId;
+ bool* m_isKey;
+ unsigned m_tupleId; // position of tuple id
+ unsigned m_autoIncrement; // position of ai key
+ SqlType m_idType; // type of tuple id or ai key
+ unsigned m_defaultCount;
+ NdbAttrId* m_defaultId;
+ SqlRow* m_defaultValue;
+ bool findAttrId(NdbAttrId attrId) const;
+ };
+ class Data : public Exec_dml::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_insert;
+ };
+ Exec_insert(Exec_root* root);
+ virtual ~Exec_insert();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+private:
+ Exec_query* m_query;
+};
+
+inline
+Exec_insert::Code::Code() :
+ m_insertOp(Insert_op_undef),
+ m_tableName(0),
+ m_attrCount(0),
+ m_attrId(0),
+ m_isKey(0),
+ m_tupleId(0),
+ m_autoIncrement(0),
+ m_defaultCount(0),
+ m_defaultId(0),
+ m_defaultValue(0)
+{
+}
+
+inline
+Exec_insert::Data::Data()
+{
+}
+
+inline
+Exec_insert::Exec_insert(Exec_root* root) :
+ Exec_dml(root),
+ m_query(0)
+{
+}
+
+// children
+
+inline const Exec_insert::Code&
+Exec_insert::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_insert::Data&
+Exec_insert::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_insert::setQuery(Exec_query* query)
+{
+ ctx_assert(m_query == 0 && query != 0);
+ m_query = query;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_pred.cpp b/ndb/src/client/odbc/codegen/Code_pred.cpp
new file mode 100644
index 00000000000..fe7cac7606e
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_pred.cpp
@@ -0,0 +1,70 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_pred.hpp"
+#include "Code_pred_op.hpp"
+#include "Code_root.hpp"
+
+// Plan_pred
+
+Plan_pred::~Plan_pred()
+{
+}
+
+bool
+Plan_pred::isGroupBy(const Plan_expr_row* row) const
+{
+ return false;
+}
+
+Plan_pred*
+Plan_pred::opAnd(Plan_pred* pred2)
+{
+ Plan_pred_op* predAnd = new Plan_pred_op(m_root, Pred_op::And);
+ m_root->saveNode(predAnd);
+ predAnd->setPred(1, this);
+ predAnd->setPred(2, pred2);
+ return predAnd;
+}
+
+// Exec_pred
+
+Exec_pred::Code::~Code()
+{
+}
+
+Exec_pred::Data::~Data()
+{
+}
+
+Exec_pred::~Exec_pred()
+{
+}
+
+Pred_value&
+Exec_pred::Data::groupValue(unsigned i, bool initFlag)
+{
+ if (m_groupValue.size() == 0) {
+ m_groupValue.resize(1);
+ }
+ if (initFlag) {
+ //unsigned i2 = m_groupValue.size();
+ //ctx_assert(i == i2);
+ m_groupValue.push_back(Pred_value_unknown);
+ }
+ ctx_assert(i != 0 && i < m_groupValue.size());
+ return m_groupValue[i];
+}
diff --git a/ndb/src/client/odbc/codegen/Code_pred.hpp b/ndb/src/client/odbc/codegen/Code_pred.hpp
new file mode 100644
index 00000000000..a77c1161fa1
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_pred.hpp
@@ -0,0 +1,172 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_pred_hpp
+#define ODBC_CODEGEN_Code_pred_hpp
+
+#include <common/common.hpp>
+#include <common/DataField.hpp>
+#include "Code_base.hpp"
+
+enum Pred_value {
+ Pred_value_unknown = -1,
+ Pred_value_false = 0,
+ Pred_value_true = 1
+};
+
+class Ctx;
+class Plan_expr_row;
+class Exec_pred;
+
+/**
+ * @class Plan_pred
+ * @brief Base class for predicates in PlanTree
+ *
+ * Predicate represents a boolean value.
+ */
+class Plan_pred : public Plan_base {
+public:
+ // type is convenient since RTTI cannot be used
+ enum Type {
+ TypeUndefined = 0,
+ TypeComp = 1,
+ TypeOp = 2
+ };
+ Plan_pred(Plan_root* root, Type type);
+ virtual ~Plan_pred() = 0;
+ Type type() const;
+ const TableSet& tableSet() const;
+ const TableSet& noInterp() const;
+ virtual bool isGroupBy(const Plan_expr_row* row) const;
+ // helpers
+ Plan_pred* opAnd(Plan_pred* pred2);
+protected:
+ const Type m_type;
+ TableSet m_tableSet; // depends on these tables
+ TableSet m_noInterp; // cannot use interpreted TUP program on these tables
+ Exec_pred* m_exec; // probably stupid
+};
+
+inline
+Plan_pred::Plan_pred(Plan_root* root, Type type) :
+ Plan_base(root),
+ m_type(type),
+ m_exec(0)
+{
+}
+
+inline Plan_pred::Type
+Plan_pred::type() const
+{
+ return m_type;
+}
+
+inline const Plan_pred::TableSet&
+Plan_pred::tableSet() const
+{
+ return m_tableSet;
+}
+
+inline const Plan_pred::TableSet&
+Plan_pred::noInterp() const
+{
+ return m_noInterp;
+}
+
+/**
+ * @class Exec_pred
+ * @brief Base class for predicates in ExecTree
+ */
+class Exec_pred : public Exec_base {
+public:
+ class Code : public Exec_base::Code {
+ public:
+ Code();
+ virtual ~Code() = 0;
+ protected:
+ friend class Exec_pred;
+ };
+ class Data : public Exec_base::Data {
+ public:
+ Data();
+ virtual ~Data() = 0;
+ Pred_value getValue() const;
+ Pred_value groupValue(unsigned i) const;
+ protected:
+ friend class Exec_pred;
+ Pred_value m_value; // the value
+ // group-by data
+ typedef std::vector<Pred_value> GroupValue;
+ GroupValue m_groupValue;
+ Pred_value& groupValue(unsigned i, bool initFlag);
+ };
+ Exec_pred(Exec_root* root);
+ virtual ~Exec_pred() = 0;
+ virtual void execInterp(Ctx& ctx, Ctl& ctl) = 0;
+ virtual void evaluate(Ctx& ctx, Ctl& ctl) = 0;
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_pred::Code::Code()
+{
+}
+
+inline
+Exec_pred::Data::Data() :
+ m_value(Pred_value_unknown)
+{
+}
+
+inline Pred_value
+Exec_pred::Data::getValue() const
+{
+ return m_value;
+}
+
+inline Pred_value
+Exec_pred::Data::groupValue(unsigned i) const
+{
+ ctx_assert(i != 0 && i < m_groupValue.size());
+ return m_groupValue[i];
+}
+
+inline
+Exec_pred::Exec_pred(Exec_root* root) :
+ Exec_base(root)
+{
+}
+
+// children
+
+inline const Exec_pred::Code&
+Exec_pred::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_pred::Data&
+Exec_pred::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_pred_op.cpp b/ndb/src/client/odbc/codegen/Code_pred_op.cpp
new file mode 100644
index 00000000000..29736e45818
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_pred_op.cpp
@@ -0,0 +1,188 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_pred.hpp"
+#include "Code_pred_op.hpp"
+#include "Code_root.hpp"
+
+// Pred_op
+
+const char*
+Pred_op::name() const
+{
+ switch (m_opcode) {
+ case And:
+ return "and";
+ case Or:
+ return "or";
+ case Not:
+ return "not";
+ }
+ ctx_assert(false);
+ return "";
+}
+
+unsigned
+Pred_op::arity() const
+{
+ switch (m_opcode) {
+ case And:
+ case Or:
+ return 2;
+ case Not:
+ return 1;
+ }
+ ctx_assert(false);
+ return 0;
+}
+
+// Plan_pred_op
+
+Plan_pred_op::~Plan_pred_op()
+{
+}
+
+Plan_base*
+Plan_pred_op::analyze(Ctx& ctx, Ctl& ctl)
+{
+ m_exec = 0;
+ unsigned arity = m_op.arity();
+ // check if we remain in top-level AND-clause
+ const bool topand = ctl.m_topand;
+ if (m_op.m_opcode != Pred_op::And)
+ ctl.m_topand = false;
+ // analyze sub-predicates
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_pred[i] != 0);
+ m_pred[i]->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ }
+ // save top level predicate on list
+ if (topand && ! ctl.m_topand) {
+ ctl.m_topcomp.push_back(this);
+ }
+ ctl.m_topand = topand;
+ // table dependencies are union from operands
+ m_tableSet.clear();
+ for (unsigned i = 1; i <= arity; i++) {
+ const TableSet& ts = m_pred[i]->tableSet();
+ m_tableSet.insert(ts.begin(), ts.end());
+ }
+ // set of tables for which interpreter cannot be used
+ m_noInterp.clear();
+ for (unsigned i = 1; i <= arity; i++) {
+ const TableSet& ts = m_pred[i]->noInterp();
+ m_noInterp.insert(ts.begin(), ts.end());
+ }
+ return this;
+}
+
+Exec_base*
+Plan_pred_op::codegen(Ctx& ctx, Ctl& ctl)
+{
+ if (m_exec != 0)
+ return m_exec;
+ unsigned arity = m_op.arity();
+ Exec_pred_op* exec = new Exec_pred_op(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // create code for operands
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_pred[i] != 0);
+ Exec_pred* execPred = static_cast<Exec_pred*>(m_pred[i]->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execPred != 0);
+ exec->setPred(i, execPred);
+ }
+ // create the code
+ Exec_pred_op::Code& code = *new Exec_pred_op::Code(m_op);
+ exec->setCode(code);
+ m_exec = exec;
+ return exec;
+}
+
+void
+Plan_pred_op::print(Ctx& ctx)
+{
+ ctx.print(" [%s", m_op.name());
+ Plan_base* a[] = { m_pred[1], m_pred[2] };
+ printList(ctx, a, m_op.arity());
+ ctx.print("]");
+}
+
+bool
+Plan_pred_op::isGroupBy(const Plan_expr_row* row) const
+{
+ const unsigned arity = m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_pred[i] != 0);
+ if (! m_pred[i]->isGroupBy(row))
+ return false;
+ }
+ return true;
+}
+
+// Code_pred_op
+
+Exec_pred_op::Code::~Code()
+{
+}
+
+Exec_pred_op::Data::~Data()
+{
+}
+
+Exec_pred_op::~Exec_pred_op()
+{
+}
+
+void
+Exec_pred_op::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // allocate sub-predicates
+ unsigned arity = code.m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_pred[i] != 0);
+ m_pred[i]->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_pred_op::close(Ctx& ctx)
+{
+ const Code& code = getCode();
+ unsigned arity = code.m_op.arity();
+ for (unsigned i = 1; i <= arity; i++) {
+ ctx_assert(m_pred[i] != 0);
+ m_pred[i]->close(ctx);
+ }
+}
+
+void
+Exec_pred_op::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [%s", code.m_op.name());
+ Exec_base* a[] = { m_pred[1], m_pred[2] };
+ printList(ctx, a, code.m_op.arity());
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_pred_op.hpp b/ndb/src/client/odbc/codegen/Code_pred_op.hpp
new file mode 100644
index 00000000000..9130bc3cb81
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_pred_op.hpp
@@ -0,0 +1,158 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_pred_op_hpp
+#define ODBC_CODEGEN_Code_pred_op_hpp
+
+#include <common/common.hpp>
+#include <common/DataField.hpp>
+#include "Code_pred.hpp"
+
+/**
+ * @class Pred_op
+ * @brief Boolean operators
+ */
+struct Pred_op {
+ enum Opcode {
+ And = 1, // binary
+ Or,
+ Not // unary
+ };
+ Pred_op(Opcode opcode);
+ const char* name() const;
+ unsigned arity() const;
+ Opcode m_opcode;
+};
+
+inline
+Pred_op::Pred_op(Opcode opcode) :
+ m_opcode(opcode)
+{
+}
+
+/**
+ * @class Plan_pred_op
+ * @brief Operator node in a predicate in PlanTree
+ */
+class Plan_pred_op : public Plan_pred {
+public:
+ Plan_pred_op(Plan_root* root, Pred_op op);
+ virtual ~Plan_pred_op();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ bool isGroupBy(const Plan_expr_row* row) const;
+ // children
+ void setPred(unsigned i, Plan_pred* pred);
+protected:
+ friend class Plan_pred;
+ Pred_op m_op;
+ Plan_pred* m_pred[1 + 2];
+};
+
+inline
+Plan_pred_op::Plan_pred_op(Plan_root* root, Pred_op op) :
+ Plan_pred(root, TypeOp),
+ m_op(op)
+{
+ m_pred[0] = m_pred[1] = m_pred[2] = 0;
+}
+
+inline void
+Plan_pred_op::setPred(unsigned i, Plan_pred* pred)
+{
+ ctx_assert(1 <= i && i <= m_op.arity() && pred != 0);
+ m_pred[i] = pred;
+}
+
+/**
+ * @class Exec_pred_op
+ * @brief Operator node in a predicate in ExecTree
+ */
+class Exec_pred_op : public Exec_pred {
+public:
+ class Code : public Exec_pred::Code {
+ public:
+ Code(Pred_op op);
+ virtual ~Code();
+ protected:
+ friend class Exec_pred_op;
+ Pred_op m_op;
+ };
+ class Data : public Exec_pred::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_pred_op;
+ };
+ Exec_pred_op(Exec_root* root);
+ virtual ~Exec_pred_op();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execInterp(Ctx& ctx, Ctl& ctl);
+ void evaluate(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setPred(unsigned i, Exec_pred* pred);
+protected:
+ Exec_pred* m_pred[1 + 2];
+};
+
+inline
+Exec_pred_op::Code::Code(Pred_op op) :
+ m_op(op)
+{
+}
+
+inline
+Exec_pred_op::Data::Data()
+{
+}
+
+inline
+Exec_pred_op::Exec_pred_op(Exec_root* root) :
+ Exec_pred(root)
+{
+ m_pred[0] = m_pred[1] = m_pred[2] = 0;
+}
+
+// children
+
+inline const Exec_pred_op::Code&
+Exec_pred_op::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_pred_op::Data&
+Exec_pred_op::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_pred_op::setPred(unsigned i, Exec_pred* pred)
+{
+ ctx_assert(1 <= i && i <= 2 && m_pred[i] == 0);
+ m_pred[i] = pred;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query.cpp b/ndb/src/client/odbc/codegen/Code_query.cpp
new file mode 100644
index 00000000000..9e983942601
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query.cpp
@@ -0,0 +1,299 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_query.hpp"
+#include "Code_query_project.hpp"
+#include "Code_query_count.hpp"
+
+// Plan_query
+
+Plan_query::~Plan_query()
+{
+}
+
+Plan_expr_row*
+Plan_query::getRow()
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_query::describe(Ctx& ctx)
+{
+ const Plan_expr_row* exprRow = getRow();
+ const unsigned count = exprRow->getSize();
+ // create IRD
+ DescArea& ird = descArea(Desc_usage_IRD);
+ ird.setCount(ctx, count);
+ for (unsigned i = 1; i <= count; i++) {
+ DescRec& rec = ird.getRecord(i);
+ const Plan_expr* expr = exprRow->m_exprList[i];
+ const SqlType& sqlType = expr->sqlType();
+ // data type
+ SQLSMALLINT desc_TYPE = sqlType.type();
+ rec.setField(SQL_DESC_TYPE, desc_TYPE);
+ SQLSMALLINT desc_CONCISE_TYPE = desc_TYPE;
+ rec.setField(SQL_DESC_CONCISE_TYPE, desc_CONCISE_TYPE);
+ SQLSMALLINT desc_DESC_DATETIME_INTERVAL_CODE = 0;
+ rec.setField(SQL_DESC_DATETIME_INTERVAL_CODE, desc_DESC_DATETIME_INTERVAL_CODE);
+ // nullable
+ SQLSMALLINT desc_NULLABLE = sqlType.nullable() ? SQL_NULLABLE : SQL_NO_NULLS;
+ rec.setField(SQL_DESC_NULLABLE, desc_NULLABLE);
+ // unsigned
+ SQLSMALLINT desc_UNSIGNED;
+ switch (sqlType.type()) {
+ case SQL_SMALLINT:
+ case SQL_INTEGER:
+ case SQL_BIGINT:
+ desc_UNSIGNED = sqlType.unSigned() ? SQL_TRUE : SQL_FALSE;
+ break;
+ default:
+ desc_UNSIGNED = SQL_TRUE; // thus spake microsoft
+ break;
+ }
+ rec.setField(SQL_DESC_UNSIGNED, desc_UNSIGNED);
+ // sizes
+ SQLUINTEGER desc_LENGTH = sqlType.length();
+ rec.setField(SQL_DESC_LENGTH, desc_LENGTH);
+ SQLINTEGER desc_OCTET_LENGTH = sqlType.size();
+ rec.setField(SQL_DESC_OCTET_LENGTH, desc_OCTET_LENGTH);
+ SQLINTEGER desc_DISPLAY_SIZE = sqlType.displaySize();
+ rec.setField(SQL_DESC_DISPLAY_SIZE, desc_DISPLAY_SIZE);
+ // name
+ ctx_assert(i < exprRow->m_aliasList.size());
+ const char* desc_NAME = exprRow->m_aliasList[i].c_str();
+ rec.setField(SQL_DESC_NAME, desc_NAME);
+ }
+ ctx_log3(("describe %u columns done", count));
+ stmtArea().setFunction(ctx, "SELECT CURSOR", SQL_DIAG_SELECT_CURSOR);
+}
+
+// Exec_query
+
+Exec_query::Code::~Code()
+{
+}
+
+Exec_query::Data::~Data()
+{
+ delete m_extRow;
+ m_extRow = 0;
+ delete[] m_extPos;
+ m_extPos = 0;
+}
+
+Exec_query::~Exec_query()
+{
+}
+
+const Exec_query*
+Exec_query::getRawQuery() const
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Exec_query::bind(Ctx& ctx)
+{
+ const Code& code = getCode();
+ const SqlSpecs& sqlSpecs = code.sqlSpecs();
+ const unsigned count = sqlSpecs.count();
+ // read ARD
+ DescArea& ard = descArea(Desc_usage_ARD);
+ const unsigned ardCount = ard.getCount();
+ // create specification row
+ ExtSpecs extSpecs(count);
+ for (unsigned i = 1; i <= count; i++) {
+ ExtType extType;
+ if (i <= ardCount) {
+ OdbcData descData;
+ DescRec& rec = ard.getRecord(i);
+ // check for unbound column
+ rec.getField(ctx, SQL_DESC_DATA_PTR, descData);
+ SQLPOINTER desc_DATA_PTR = descData.type() != OdbcData::Undef ? descData.pointer() : 0;
+ if (desc_DATA_PTR == 0) {
+ extType.setType(ctx, ExtType::Unbound);
+ } else {
+ rec.getField(ctx, SQL_DESC_TYPE, descData);
+ if (descData.type() == OdbcData::Undef) {
+ ctx.pushStatus(Error::Gen, "query column %u: external type not defined", i);
+ return;
+ }
+ SQLSMALLINT desc_TYPE = descData.smallint();
+ if (desc_TYPE == SQL_C_DEFAULT) {
+ if (i <= code.m_sqlSpecs.count())
+ desc_TYPE = code.m_sqlSpecs.getEntry(i).sqlType().sqlcdefault(ctx);
+ }
+ switch (desc_TYPE) {
+ case SQL_C_CHAR:
+ case SQL_C_BINARY:
+ case SQL_C_SHORT: // for sun.jdbc.odbc
+ case SQL_C_SSHORT:
+ case SQL_C_USHORT:
+ case SQL_C_LONG: // for sun.jdbc.odbc
+ case SQL_C_SLONG:
+ case SQL_C_ULONG:
+ case SQL_C_SBIGINT:
+ case SQL_C_UBIGINT:
+ case SQL_C_FLOAT:
+ case SQL_C_DOUBLE:
+ case SQL_C_TYPE_TIMESTAMP:
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "query column %u: unsupported external type %d", i, (int)desc_TYPE);
+ return;
+ }
+ extType.setType(ctx, static_cast<ExtType::Type>(desc_TYPE));
+ }
+ } else {
+ extType.setType(ctx, ExtType::Unbound);
+ }
+ const ExtSpec extSpec(extType);
+ extSpecs.setEntry(i, extSpec);
+ }
+ // create data row
+ ExtRow& extRow = *new ExtRow(extSpecs);
+ unsigned boundCount = 0;
+ for (unsigned i = 1; i <= count; i++) {
+ const ExtSpec& extSpec = extSpecs.getEntry(i);
+ if (extSpec.extType().type() != ExtType::Unbound) {
+ OdbcData descData;
+ DescRec& rec = ard.getRecord(i);
+ rec.getField(ctx, SQL_DESC_DATA_PTR, descData);
+ SQLPOINTER desc_DATA_PTR = descData.type() != OdbcData::Undef ? descData.pointer() : 0;
+ rec.getField(ctx, SQL_DESC_OCTET_LENGTH, descData);
+ SQLINTEGER desc_OCTET_LENGTH = descData.type() != OdbcData::Undef ? descData.integer() : 0;
+ rec.getField(ctx, SQL_DESC_INDICATOR_PTR, descData);
+ SQLINTEGER* desc_INDICATOR_PTR = descData.type() != OdbcData::Undef ? descData.integerPtr() : 0;
+ ctx_log4(("column %u: bind to 0x%x %d 0x%x", i, (unsigned)desc_DATA_PTR, (int)desc_OCTET_LENGTH, (unsigned)desc_INDICATOR_PTR));
+ ExtField extField(extSpec, desc_DATA_PTR, desc_OCTET_LENGTH, desc_INDICATOR_PTR, i);
+ extRow.setEntry(i, extField);
+ boundCount++;
+ } else {
+ ExtField extField(extSpec, i);
+ extRow.setEntry(i, extField);
+ }
+ }
+ Data& data = getData();
+ delete data.m_extRow;
+ data.m_extRow = &extRow;
+ ctx_log3(("bound %u out of %u columns", boundCount, count));
+}
+
+// execute and fetch
+
+void
+Exec_query::execute(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = getData();
+ execImpl(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ data.initState();
+ if (m_topLevel) {
+ stmtArea().setRowCount(ctx, data.getCount());
+ }
+}
+
+bool
+Exec_query::fetch(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ if (data.fetch(ctx, ctl)) {
+ if (m_topLevel) {
+ stmtArea().setRowCount(ctx, data.getCount());
+ }
+ if (data.m_extRow != 0) {
+ data.sqlRow().copyout(ctx, *data.m_extRow);
+ if (! ctx.ok())
+ return false;
+ }
+ if (data.m_extPos != 0) {
+ const unsigned count = code.sqlSpecs().count();
+ for (unsigned i = 0; i <= count; i++) {
+ data.m_extPos[i] = 0;
+ }
+ }
+ return true;
+ }
+ if (m_topLevel) {
+ stmtArea().setRowCount(ctx, data.getCount());
+ if (ctx.ok()) {
+ ctx.setCode(SQL_NO_DATA);
+ }
+ }
+ return false;
+}
+
+// odbc support
+
+void
+Exec_query::sqlGetData(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ const SqlSpecs& sqlSpecs = code.m_sqlSpecs;
+ const unsigned count = sqlSpecs.count();
+ if (columnNumber == 0 || columnNumber > count) {
+ ctx.pushStatus(Sqlstate::_07009, Error::Gen, "column index %u is not within 1 to %u", (unsigned)columnNumber, count);
+ return;
+ }
+ // create positions array on first use
+ if (data.m_extPos == 0) {
+ data.m_extPos = new int[1 + count];
+ for (unsigned i = 0; i <= count; i++) {
+ data.m_extPos[i] = 0;
+ }
+ }
+ if (targetType == SQL_ARD_TYPE) {
+ // get type from ARD
+ DescArea& ard = descArea(Desc_usage_ARD);
+ const unsigned ardCount = ard.getCount();
+ if (columnNumber <= ardCount) {
+ OdbcData descData;
+ DescRec& rec = ard.getRecord(columnNumber);
+ rec.getField(ctx, SQL_DESC_CONCISE_TYPE, descData);
+ if (descData.type() != OdbcData::Undef) {
+ targetType = descData.smallint();
+ }
+ }
+ if (targetType == SQL_ARD_TYPE) {
+ ctx.pushStatus(Sqlstate::_07009, Error::Gen, "output column %u type not bound - cannot use SQL_ARD_TYPE", (unsigned)columnNumber);
+ return;
+ }
+ }
+ ExtType extType;
+ if (targetValue != 0) {
+ extType.setType(ctx, static_cast<ExtType::Type>(targetType));
+ // check if supported
+ if (! ctx.ok())
+ return;
+ } else {
+ extType.setType(ctx, ExtType::Unbound);
+ }
+ ExtSpec extSpec(extType);
+ ExtField extField(extSpec, targetValue, bufferLength, strlen_or_Ind, columnNumber);
+ // copy out and update position
+ extField.setPos(data.m_extPos[columnNumber]);
+ const SqlRow& sqlRow = data.sqlRow();
+ const SqlField& sqlField = sqlRow.getEntry(columnNumber);
+ sqlField.copyout(ctx, extField);
+ data.m_extPos[columnNumber] = extField.getPos();
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query.hpp b/ndb/src/client/odbc/codegen/Code_query.hpp
new file mode 100644
index 00000000000..97f98f859ff
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query.hpp
@@ -0,0 +1,155 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_hpp
+#define ODBC_CODEGEN_Code_query_hpp
+
+#include <common/common.hpp>
+#include <common/DataRow.hpp>
+#include <common/ResultArea.hpp>
+#include "Code_stmt.hpp"
+
+class Plan_expr_row;
+class Plan_table;
+class Exec_expr_row;
+
+/**
+ * @class Plan_query
+ * @brief Base class for queries in PlanTree
+ */
+class Plan_query : public Plan_stmt {
+public:
+ Plan_query(Plan_root* root);
+ virtual ~Plan_query() = 0;
+ void describe(Ctx& ctx);
+ virtual Plan_expr_row* getRow();
+};
+
+inline
+Plan_query::Plan_query(Plan_root* root) :
+ Plan_stmt(root)
+{
+}
+
+/**
+ * @class Exec_query
+ * @brief Base class for executable queries.
+ *
+ * Executable queriable statement.
+ */
+class Exec_query : public Exec_stmt {
+public:
+ class Code : public Exec_stmt::Code {
+ public:
+ Code(const SqlSpecs& sqlSpecs);
+ virtual ~Code() = 0;
+ const SqlSpecs& sqlSpecs() const;
+ protected:
+ friend class Exec_query;
+ const SqlSpecs& m_sqlSpecs; // subclass must contain
+ };
+ class Data : public Exec_stmt::Data, public ResultSet {
+ public:
+ Data(Exec_query* node, const SqlRow& sqlRow);
+ virtual ~Data() = 0;
+ const SqlRow& sqlRow() const;
+ ExtRow* extRow() const;
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ protected:
+ friend class Exec_query;
+ Exec_query* const m_node;
+ ExtRow* m_extRow; // output bindings
+ int* m_extPos; // positions for SQLGetData
+ };
+ Exec_query(Exec_root* root);
+ virtual ~Exec_query() = 0;
+ void bind(Ctx& ctx);
+ void execute(Ctx& ctx, Ctl& ctl);
+ bool fetch(Ctx& ctx, Ctl& ctl);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ virtual const Exec_query* getRawQuery() const;
+ // odbc support
+ void sqlGetData(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind);
+protected:
+ friend class Data;
+ virtual void execImpl(Ctx& ctx, Ctl& ctl) = 0;
+ virtual bool fetchImpl(Ctx& ctx, Ctl& ctl) = 0;
+};
+
+inline
+Exec_query::Code::Code(const SqlSpecs& sqlSpecs) :
+ m_sqlSpecs(sqlSpecs)
+{
+}
+
+inline const SqlSpecs&
+Exec_query::Code::sqlSpecs() const
+{
+ return m_sqlSpecs;
+}
+
+inline
+Exec_query::Data::Data(Exec_query* node, const SqlRow& sqlRow) :
+ ResultSet(sqlRow),
+ m_node(node),
+ m_extRow(0),
+ m_extPos(0)
+{
+}
+
+inline const SqlRow&
+Exec_query::Data::sqlRow() const
+{
+ return static_cast<const SqlRow&>(m_dataRow);
+}
+
+inline ExtRow*
+Exec_query::Data::extRow() const
+{
+ return m_extRow;
+}
+
+inline bool
+Exec_query::Data::fetchImpl(Ctx& ctx, Ctl& ctl)
+{
+ return m_node->fetchImpl(ctx, ctl);
+}
+
+inline
+Exec_query::Exec_query(Exec_root* root) :
+ Exec_stmt(root)
+{
+}
+
+// children
+
+inline const Exec_query::Code&
+Exec_query::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query::Data&
+Exec_query::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_count.cpp b/ndb/src/client/odbc/codegen/Code_query_count.cpp
new file mode 100644
index 00000000000..f52c41df802
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_count.cpp
@@ -0,0 +1,177 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_query_count.hpp"
+#include "Code_column.hpp"
+#include "Code_expr_row.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_count
+
+Plan_query_count::~Plan_query_count()
+{
+}
+
+Plan_expr_row*
+Plan_query_count::getRow()
+{
+ ctx_assert(m_exprRow != 0);
+ return m_exprRow;
+}
+
+Plan_base*
+Plan_query_count::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_exprRow != 0);
+ ctl.m_aggrok = true;
+ ctl.m_aggrin = false;
+ m_exprRow->analyze(ctx, ctl);
+ ctl.m_aggrok = false;
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_query_count::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // create code for the subquery
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // create code for the row based on query code
+ ctx_assert(m_exprRow != 0);
+ ctl.m_execQuery = execQuery;
+ Exec_expr_row* execRow = static_cast<Exec_expr_row*>(m_exprRow->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execRow != 0);
+ Exec_query_count* exec = new Exec_query_count(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // re-use SqlSpecs from the row
+ const SqlSpecs& sqlSpecs = execRow->getCode().sqlSpecs();
+ Exec_query_count::Code& code = *new Exec_query_count::Code(sqlSpecs);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ exec->setRow(execRow);
+ return exec;
+}
+
+void
+Plan_query_count::print(Ctx& ctx)
+{
+ ctx.print(" [query_count");
+ Plan_base* a[] = { m_query, m_exprRow };
+ printList(ctx, a, sizeof(a)/sizeof(a[0]));
+ ctx.print("]");
+}
+
+// Exec_query_count
+
+Exec_query_count::Code::~Code()
+{
+}
+
+Exec_query_count::Data::~Data()
+{
+}
+
+Exec_query_count::~Exec_query_count()
+{
+}
+
+const Exec_query*
+Exec_query_count::getRawQuery() const
+{
+ ctx_assert(m_query != 0);
+ return m_query;
+}
+
+void
+Exec_query_count::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate the subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ // allocate the row based on subquery data
+ ctx_assert(m_exprRow != 0);
+ ctl.m_query = m_query;
+ m_exprRow->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // re-use SqlRow from the expression row
+ const SqlRow& sqlRow = m_exprRow->getData().sqlRow();
+ Data& data = *new Data(this, sqlRow);
+ setData(data);
+}
+
+void
+Exec_query_count::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = getData();
+ // zero counters
+ ctx_assert(m_exprRow != 0);
+ m_exprRow->close(ctx);
+ data.m_done = false;
+ // execute the subquery
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+}
+
+bool
+Exec_query_count::fetchImpl(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = getData();
+ // returns one row only
+ if (data.m_done)
+ return false;
+ ctx_assert(m_query != 0 && m_exprRow != 0);
+ while (m_query->fetch(ctx, ctl)) {
+ // accumulate values
+ m_exprRow->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return false;
+ }
+ data.m_done = true;
+ return true;
+}
+
+void
+Exec_query_count::close(Ctx& ctx)
+{
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+ ctx_assert(m_exprRow != 0);
+ m_exprRow->close(ctx);
+}
+
+void
+Exec_query_count::print(Ctx& ctx)
+{
+ ctx.print(" [query_count");
+ Exec_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_count.hpp b/ndb/src/client/odbc/codegen/Code_query_count.hpp
new file mode 100644
index 00000000000..a094eba4519
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_count.hpp
@@ -0,0 +1,162 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_count_hpp
+#define ODBC_CODEGEN_Code_query_count_hpp
+
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_expr_row.hpp"
+#include "Code_root.hpp"
+
+class Ctx;
+
+/**
+ * @class Plan_query_count
+ * @brief Select count and other aggregates (no group by)
+ */
+class Plan_query_count : public Plan_query {
+public:
+ Plan_query_count(Plan_root* root);
+ virtual ~Plan_query_count();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setQuery(Plan_query* query);
+ void setRow(Plan_expr_row* exprRow);
+protected:
+ Plan_expr_row* getRow();
+ Plan_query* m_query;
+ Plan_expr_row* m_exprRow;
+};
+
+inline
+Plan_query_count::Plan_query_count(Plan_root* root) :
+ Plan_query(root),
+ m_query(0),
+ m_exprRow(0)
+{
+}
+
+// children
+
+inline void
+Plan_query_count::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Plan_query_count::setRow(Plan_expr_row* exprRow)
+{
+ ctx_assert(exprRow != 0);
+ m_exprRow = exprRow;
+}
+
+/**
+ * @class Exec_query_count
+ * @brief Select count and other aggregates (no group by)
+ */
+class Exec_query_count : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(const SqlSpecs& sqlSpecs);
+ virtual ~Code();
+ protected:
+ friend class Exec_query_count;
+ // sets reference to Sqlspecs from the row
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_count* node, const SqlRow& sqlRow);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_count;
+ // sets reference to SqlRow from the row
+ bool m_done; // returns one row
+ };
+ Exec_query_count(Exec_root* root);
+ virtual ~Exec_query_count();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+ void setRow(Exec_expr_row* exprRow);
+ const Exec_query* getRawQuery() const;
+protected:
+ Exec_query* m_query;
+ Exec_expr_row* m_exprRow;
+};
+
+inline
+Exec_query_count::Code::Code(const SqlSpecs& sqlSpecs) :
+ Exec_query::Code(sqlSpecs)
+{
+}
+
+inline
+Exec_query_count::Data::Data(Exec_query_count* node, const SqlRow& sqlRow) :
+ Exec_query::Data(node, sqlRow)
+{
+}
+
+inline
+Exec_query_count::Exec_query_count(Exec_root* root) :
+ Exec_query(root),
+ m_query(0),
+ m_exprRow(0)
+{
+}
+
+// children
+
+inline const Exec_query_count::Code&
+Exec_query_count::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_count::Data&
+Exec_query_count::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_query_count::setQuery(Exec_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Exec_query_count::setRow(Exec_expr_row* exprRow)
+{
+ ctx_assert(m_exprRow == 0 && exprRow != 0);
+ m_exprRow = exprRow;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_distinct.cpp b/ndb/src/client/odbc/codegen/Code_query_distinct.cpp
new file mode 100644
index 00000000000..4cbfbfe812d
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_distinct.cpp
@@ -0,0 +1,204 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <algorithm>
+#include "Code_query_distinct.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_distinct
+
+Plan_query_distinct::~Plan_query_distinct()
+{
+}
+
+Plan_expr_row*
+Plan_query_distinct::getRow()
+{
+ ctx_assert(m_query != 0);
+ return m_query->getRow();
+}
+
+Plan_base*
+Plan_query_distinct::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_query_distinct::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // create code for the subquery
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // the exec node
+ Exec_query_distinct* exec = new Exec_query_distinct(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // re-use SqlSpecs from subquery
+ const Exec_query::Code& codeQuery = execQuery->getCode();
+ const SqlSpecs& sqlSpecs = codeQuery.sqlSpecs();
+ Exec_query_distinct::Code& code = *new Exec_query_distinct::Code(sqlSpecs);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ return exec;
+}
+
+void
+Plan_query_distinct::print(Ctx& ctx)
+{
+ ctx.print(" [query_distinct");
+ Plan_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
+
+// Exec_query_distinct
+
+Exec_query_distinct::Code::~Code()
+{
+}
+
+Exec_query_distinct::Data::~Data()
+{
+ for (DistinctList::iterator i = m_groupList.begin(); i != m_groupList.end(); i++) {
+ delete (*i).first;
+ }
+}
+
+Exec_query_distinct::~Exec_query_distinct()
+{
+}
+
+const Exec_query*
+Exec_query_distinct::getRawQuery() const
+{
+ ctx_assert(m_query != 0);
+ return m_query->getRawQuery();
+}
+
+void
+Exec_query_distinct::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ Data& data = *new Data(this, getCode().sqlSpecs());
+ setData(data);
+}
+
+void
+Exec_query_distinct::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+}
+
+bool
+DistinctLess::operator()(const SqlRow* s1, const SqlRow* s2) const
+{
+ ctx_assert(s1 != 0 && s2 != 0);
+ const SqlRow& r1 = *s1;
+ const SqlRow& r2 = *s2;
+ for (unsigned i = 1; i <= r1.count(); i++) {
+ const SqlField& f1 = r1.getEntry(i);
+ const SqlField& f2 = r2.getEntry(i);
+ // nulls last is default in oracle
+ const bool f1null = f1.sqlNull();
+ const bool f2null = f2.sqlNull();
+ if (f1null && f2null)
+ continue;
+ if (! f1null && f2null)
+ return true;
+ if (f1null && ! f2null)
+ return false;
+ if (f1.less(f2))
+ return true;
+ if (f2.less(f1))
+ return false;
+ }
+ return false;
+}
+
+bool
+Exec_query_distinct::fetchImpl(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = getData();
+ ctx_assert(m_query != 0);
+ if (! data.m_grouped) {
+ // read and group all rows
+ while (m_query->fetch(ctx, ctl)) {
+ const SqlRow* dataRow = &m_query->getData().sqlRow();
+ DistinctList::iterator i = data.m_groupList.find(dataRow);
+ if (i != data.m_groupList.end())
+ continue;
+ unsigned index = data.m_count++;
+ dataRow = dataRow->copy();
+ const DistinctList::value_type groupPair(dataRow, index);
+ data.m_groupList.insert(groupPair);
+ data.m_groupVector.push_back(dataRow);
+ }
+ if (! ctx.ok())
+ return false;
+ data.m_index = 0;
+ data.m_grouped = true;
+ }
+ ctx_assert(data.m_count == data.m_groupVector.size());
+ if (data.m_index < data.m_count) {
+ const SqlRow* currRow = data.m_groupVector[data.m_index];
+ // make our SqlRow reference to it
+ for (unsigned i = 1; i <= data.m_sqlRow.count(); i++) {
+ const SqlField& currField = currRow->getEntry(i);
+ SqlSpec sqlSpec(currField.sqlSpec(), SqlSpec::Reference);
+ SqlField sqlField(sqlSpec, &currField);
+ data.m_sqlRow.setEntry(i, sqlField);
+ }
+ data.m_index++;
+ return true;
+ }
+ return false;
+}
+
+void
+Exec_query_distinct::close(Ctx& ctx)
+{
+ Data& data = getData();
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+ data.m_grouped = false;
+ data.m_count = 0;
+ for (DistinctList::iterator i = data.m_groupList.begin(); i != data.m_groupList.end(); i++) {
+ delete (*i).first;
+ }
+ data.m_groupList.clear();
+ data.m_groupVector.clear();
+}
+
+void
+Exec_query_distinct::print(Ctx& ctx)
+{
+ ctx.print(" [query_distinct");
+ Exec_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_distinct.hpp b/ndb/src/client/odbc/codegen/Code_query_distinct.hpp
new file mode 100644
index 00000000000..62c46bda901
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_distinct.hpp
@@ -0,0 +1,165 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_distinct_hpp
+#define ODBC_CODEGEN_Code_query_distinct_hpp
+
+#include <functional>
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_expr_row.hpp"
+#include "Code_pred.hpp"
+
+/**
+ * @class Plan_query_distinct
+ * @brief Group-by node in PlanTree
+ */
+class Plan_query_distinct : public Plan_query {
+public:
+ Plan_query_distinct(Plan_root* root);
+ virtual ~Plan_query_distinct();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setQuery(Plan_query* query);
+ Plan_expr_row* getRow();
+protected:
+ Plan_query* m_query;
+};
+
+inline
+Plan_query_distinct::Plan_query_distinct(Plan_root* root) :
+ Plan_query(root),
+ m_query(0)
+{
+}
+
+// children
+
+inline void
+Plan_query_distinct::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+/**
+ * Distinct preserves order of input rows so we use 2 data structures:
+ * map<row> = index and vector<index> = row (index >= 0).
+ */
+
+class Exec_query_distinct;
+
+struct DistinctLess : std::binary_function<const SqlRow*, const SqlRow*, bool> {
+ bool operator()(const SqlRow* s1, const SqlRow* s2) const;
+};
+
+typedef std::map<const SqlRow*, unsigned, DistinctLess> DistinctList;
+
+typedef std::vector<const SqlRow*> DistinctVector;
+
+/**
+ * @class Exec_query_distinct
+ * @brief Group-by node in ExecTree
+ */
+class Exec_query_distinct : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(const SqlSpecs& sqlSpecs);
+ virtual ~Code();
+ protected:
+ friend class Exec_query_distinct;
+ // sets reference to Sqlspecs from subquery
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_distinct* node, const SqlSpecs& sqlSpecs);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_distinct;
+ SqlRow m_sqlRow; // current row
+ bool m_grouped; // fetch and group-by done
+ unsigned m_count;
+ DistinctList m_groupList;
+ DistinctVector m_groupVector;
+ unsigned m_index;
+ };
+ Exec_query_distinct(Exec_root* root);
+ virtual ~Exec_query_distinct();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+ const Exec_query* getRawQuery() const;
+protected:
+ friend class Exec_query;
+ Exec_query* m_query;
+};
+
+inline
+Exec_query_distinct::Code::Code(const SqlSpecs& sqlSpecs) :
+ Exec_query::Code(sqlSpecs)
+{
+}
+
+inline
+Exec_query_distinct::Data::Data(Exec_query_distinct* node, const SqlSpecs& sqlSpecs) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlSpecs),
+ m_grouped(false),
+ m_count(0),
+ m_index(0)
+{
+}
+
+inline
+Exec_query_distinct::Exec_query_distinct(Exec_root* root) :
+ Exec_query(root),
+ m_query(0)
+{
+}
+
+// children
+
+inline const Exec_query_distinct::Code&
+Exec_query_distinct::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_distinct::Data&
+Exec_query_distinct::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_query_distinct::setQuery(Exec_query* query)
+{
+ ctx_assert(m_query == 0 && query != 0);
+ m_query = query;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_filter.cpp b/ndb/src/client/odbc/codegen/Code_query_filter.cpp
new file mode 100644
index 00000000000..934a24d182d
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_filter.cpp
@@ -0,0 +1,161 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_query_filter.hpp"
+#include "Code_query_join.hpp"
+#include "Code_query_scan.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_filter
+
+Plan_query_filter::~Plan_query_filter()
+{
+}
+
+Plan_base*
+Plan_query_filter::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_pred != 0);
+ m_pred->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_query_filter::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // generate code for the subquery
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // create code for the predicate based on query code
+ Exec_pred* execPred = 0;
+ ctl.m_execQuery = execQuery;
+ ctx_assert(m_topTable != 0);
+ ctl.m_topTable = m_topTable;
+ ctx_assert(m_pred != 0);
+ execPred = static_cast<Exec_pred*>(m_pred->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execPred != 0);
+ ctl.m_topTable = 0;
+ // re-use SqlSpecs from subquery
+ const Exec_query::Code& codeQuery = execQuery->getCode();
+ const SqlSpecs& sqlSpecs = codeQuery.sqlSpecs();
+ Exec_query_filter* exec = new Exec_query_filter(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ Exec_query_filter::Code& code = *new Exec_query_filter::Code(sqlSpecs);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ exec->setPred(execPred);
+ return exec;
+}
+
+void
+Plan_query_filter::print(Ctx& ctx)
+{
+ ctx.print(" [query_filter");
+ Plan_base* a[] = { m_query, m_pred };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
+
+// Exec_query_filter
+
+Exec_query_filter::Code::~Code()
+{
+}
+
+Exec_query_filter::Data::~Data()
+{
+}
+
+Exec_query_filter::~Exec_query_filter()
+{
+}
+
+void
+Exec_query_filter::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate the subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // allocate the predicate
+ ctl.m_query = m_query;
+ ctx_assert(m_pred != 0);
+ m_pred->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // re-use SqlRow from subquery
+ Exec_query::Data& dataQuery = m_query->getData();
+ Data& data = *new Data(this, dataQuery.sqlRow());
+ setData(data);
+}
+
+void
+Exec_query_filter::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ // execute subquery
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+}
+
+bool
+Exec_query_filter::fetchImpl(Ctx& ctx, Ctl& ctl)
+{
+ // invoke fetch on subquery until predicate is true
+ ctx_assert(m_query != 0);
+ while (m_query->fetch(ctx, ctl)) {
+ ctx_assert(m_pred != 0);
+ m_pred->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return false;
+ if (m_pred->getData().getValue() == Pred_value_true) {
+ ctl.m_postEval = true;
+ m_pred->evaluate(ctx, ctl);
+ ctl.m_postEval = false;
+ return true;
+ }
+ }
+ return false;
+}
+
+void
+Exec_query_filter::close(Ctx& ctx)
+{
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+ ctx_assert(m_pred != 0);
+ m_pred->close(ctx);
+}
+
+void
+Exec_query_filter::print(Ctx& ctx)
+{
+ ctx.print(" [query_filter");
+ Exec_base* a[] = { m_query, m_pred };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_filter.hpp b/ndb/src/client/odbc/codegen/Code_query_filter.hpp
new file mode 100644
index 00000000000..60cbf0f86a7
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_filter.hpp
@@ -0,0 +1,162 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_filter_hpp
+#define ODBC_CODEGEN_Code_query_filter_hpp
+
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_table_list.hpp"
+#include "Code_pred.hpp"
+
+/**
+ * @class Plan_query_filter
+ * @brief Filter node in PlanTree
+ */
+class Plan_query_filter : public Plan_query {
+public:
+ Plan_query_filter(Plan_root* root);
+ virtual ~Plan_query_filter();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setQuery(Plan_query* query);
+ void setPred(Plan_pred* pred);
+protected:
+ friend class Plan_select;
+ friend class Plan_update;
+ friend class Plan_delete;
+ Plan_query* m_query;
+ Plan_pred* m_pred;
+ Plan_table* m_topTable; // top level table for interpreted progs
+};
+
+inline
+Plan_query_filter::Plan_query_filter(Plan_root* root) :
+ Plan_query(root),
+ m_query(0),
+ m_pred(0),
+ m_topTable(0)
+{
+}
+
+// children
+
+inline void
+Plan_query_filter::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Plan_query_filter::setPred(Plan_pred* pred)
+{
+ ctx_assert(pred != 0);
+ m_pred = pred;
+}
+
+/**
+ * @class Exec_query_filter
+ * @brief Filter node in ExecTree
+ */
+class Exec_query_filter : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(const SqlSpecs& sqlSpecs);
+ virtual ~Code();
+ protected:
+ friend class Exec_query_filter;
+ // sets reference to SqlSpecs from subquery
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_filter* node, const SqlRow& sqlRow);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_filter;
+ // sets reference to SqlRow from subquery
+ };
+ Exec_query_filter(Exec_root* root);
+ virtual ~Exec_query_filter();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+ void setPred(Exec_pred* pred);
+protected:
+ Exec_query* m_query;
+ Exec_pred* m_pred;
+};
+
+inline
+Exec_query_filter::Code::Code(const SqlSpecs& sqlSpecs) :
+ Exec_query::Code(sqlSpecs)
+{
+}
+
+inline
+Exec_query_filter::Data::Data(Exec_query_filter* node, const SqlRow& sqlRow) :
+ Exec_query::Data(node, sqlRow)
+{
+}
+
+inline
+Exec_query_filter::Exec_query_filter(Exec_root* root) :
+ Exec_query(root),
+ m_query(0),
+ m_pred(0)
+{
+}
+
+// children
+
+inline const Exec_query_filter::Code&
+Exec_query_filter::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_filter::Data&
+Exec_query_filter::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_query_filter::setQuery(Exec_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Exec_query_filter::setPred(Exec_pred* pred)
+{
+ ctx_assert(pred != 0);
+ m_pred = pred;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_group.cpp b/ndb/src/client/odbc/codegen/Code_query_group.cpp
new file mode 100644
index 00000000000..c3019efaa85
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_group.cpp
@@ -0,0 +1,301 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <algorithm>
+#include "Code_query_group.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_group
+
+Plan_query_group::~Plan_query_group()
+{
+}
+
+Plan_expr_row*
+Plan_query_group::getRow()
+{
+ ctx_assert(m_dataRow != 0);
+ return m_dataRow;
+}
+
+Plan_base*
+Plan_query_group::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_dataRow != 0);
+ m_dataRow->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_groupRow != 0);
+ m_groupRow->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ if (m_havingPred != 0) {
+ ctl.m_having = true;
+ m_havingPred->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctl.m_having = false;
+ }
+ return this;
+}
+
+Exec_base*
+Plan_query_group::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // create code for the subquery
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // create code for the rows based on query code
+ ctl.m_execQuery = execQuery;
+ ctx_assert(m_dataRow != 0);
+ Exec_expr_row* execDataRow = static_cast<Exec_expr_row*>(m_dataRow->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execDataRow != 0);
+ ctx_assert(m_groupRow != 0);
+ Exec_expr_row* execGroupRow = static_cast<Exec_expr_row*>(m_groupRow->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execGroupRow != 0);
+ Exec_pred* execHavingPred = 0;
+ if (m_havingPred != 0) {
+ ctl.m_having = true;
+ execHavingPred = static_cast<Exec_pred*>(m_havingPred->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execHavingPred != 0);
+ ctl.m_having = false;
+ }
+ // the exec node
+ Exec_query_group* exec = new Exec_query_group(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // re-use SqlSpecs from data row
+ const SqlSpecs& sqlSpecs = execDataRow->getCode().sqlSpecs();
+ Exec_query_group::Code& code = *new Exec_query_group::Code(sqlSpecs);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ exec->setDataRow(execDataRow);
+ exec->setGroupRow(execGroupRow);
+ if (execHavingPred != 0)
+ exec->setHavingPred(execHavingPred);
+ return exec;
+}
+
+void
+Plan_query_group::print(Ctx& ctx)
+{
+ ctx.print(" [query_group");
+ Plan_base* a[] = { m_query, m_dataRow, m_groupRow };
+ printList(ctx, a, 3);
+ ctx.print("]");
+}
+
+// Exec_query_group
+
+Exec_query_group::Code::~Code()
+{
+}
+
+Exec_query_group::Data::~Data()
+{
+ for (GroupList::iterator i = m_groupList.begin(); i != m_groupList.end(); i++) {
+ delete (*i).first;
+ }
+}
+
+Exec_query_group::~Exec_query_group()
+{
+}
+
+const Exec_query*
+Exec_query_group::getRawQuery() const
+{
+ ctx_assert(m_query != 0);
+ return m_query;
+}
+
+void
+Exec_query_group::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // allocate rows based on subquery data
+ ctl.m_query = m_query;
+ ctx_assert(m_dataRow != 0);
+ m_dataRow->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ ctx_assert(m_groupRow != 0);
+ m_groupRow->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ if (m_havingPred != 0) {
+ m_havingPred->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ Data& data = *new Data(this, getCode().sqlSpecs());
+ setData(data);
+}
+
+void
+Exec_query_group::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+}
+
+bool
+GroupLess::operator()(const SqlRow* s1, const SqlRow* s2) const
+{
+ ctx_assert(s1 != 0 && s2 != 0);
+ const SqlRow& r1 = *s1;
+ const SqlRow& r2 = *s2;
+ for (unsigned i = 1; i <= r1.count(); i++) {
+ const SqlField& f1 = r1.getEntry(i);
+ const SqlField& f2 = r2.getEntry(i);
+ // nulls last is default in oracle
+ const bool f1null = f1.sqlNull();
+ const bool f2null = f2.sqlNull();
+ if (f1null && f2null)
+ continue;
+ if (! f1null && f2null)
+ return true;
+ if (f1null && ! f2null)
+ return false;
+ if (f1.less(f2))
+ return true;
+ if (f2.less(f1))
+ return false;
+ }
+ return false;
+}
+
+bool
+Exec_query_group::fetchImpl(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = getData();
+ ctx_assert(m_query != 0 && m_groupRow != 0);
+ if (! data.m_grouped) {
+ // read and group all rows
+ while (m_query->fetch(ctx, ctl)) {
+ // evaluate and insert group-by values
+ m_groupRow->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return false;
+ const SqlRow* groupRow = 0;
+ unsigned index = 0;
+ bool init;
+ GroupList::iterator i = data.m_groupList.find(&m_groupRow->getData().sqlRow());
+ if (i == data.m_groupList.end()) {
+ groupRow = m_groupRow->getData().sqlRow().copy();
+ index = ++data.m_count;
+ const GroupList::value_type groupPair(groupRow, index);
+ data.m_groupList.insert(groupPair);
+ init = true;
+ } else {
+ groupRow = (*i).first;
+ index = (*i).second;
+ ctx_assert(groupRow != 0 && index != 0);
+ init = false;
+ }
+ // evaluate rows saving expression values at index position
+ ctl.m_groupIndex = index;
+ ctl.m_groupInit = init;
+ m_dataRow->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return false;
+ if (m_havingPred != 0) {
+ m_havingPred->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return false;
+ }
+ if (ctl.m_sortRow != 0) {
+ ctl.m_sortRow->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return false;
+ }
+ ctl.m_groupIndex = 0;
+ }
+ if (! ctx.ok())
+ return false;
+ data.m_iterator = data.m_groupList.begin();
+ data.m_grouped = true;
+ }
+ while (data.m_iterator != data.m_groupList.end()) {
+ const SqlRow* groupRow = (*data.m_iterator).first;
+ const unsigned index = (*data.m_iterator).second;
+ ctx_assert(groupRow != 0 && index != 0);
+ if (m_havingPred != 0) {
+ Pred_value v = m_havingPred->getData().groupValue(index);
+ if (v != Pred_value_true) {
+ data.m_iterator++;
+ continue;
+ }
+ }
+ // make our SqlRow reference to the saved values
+ for (unsigned i = 1; i <= data.m_sqlRow.count(); i++) {
+ const SqlField& currField = m_dataRow->getExpr(i)->getData().groupField(index);
+ SqlSpec sqlSpec(currField.sqlSpec(), SqlSpec::Reference);
+ SqlField sqlField(sqlSpec, &currField);
+ data.m_sqlRow.setEntry(i, sqlField);
+ }
+ // send group index up for possible order by
+ ctl.m_groupIndex = index;
+ data.m_iterator++;
+ return true;
+ }
+ return false;
+}
+
+void
+Exec_query_group::close(Ctx& ctx)
+{
+ Data& data = getData();
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+ ctx_assert(m_dataRow != 0);
+ m_dataRow->close(ctx);
+ ctx_assert(m_groupRow != 0);
+ m_groupRow->close(ctx);
+ if (m_havingPred != 0)
+ m_havingPred->close(ctx);
+ data.m_grouped = false;
+ data.m_count = 0;
+ for (GroupList::iterator i = data.m_groupList.begin(); i != data.m_groupList.end(); i++) {
+ delete (*i).first;
+ }
+ data.m_groupList.clear();
+}
+
+void
+Exec_query_group::print(Ctx& ctx)
+{
+ ctx.print(" [query_group");
+ Exec_base* a[] = { m_query, m_dataRow, m_groupRow };
+ printList(ctx, a, 3);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_group.hpp b/ndb/src/client/odbc/codegen/Code_query_group.hpp
new file mode 100644
index 00000000000..e79022c5284
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_group.hpp
@@ -0,0 +1,221 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_group_hpp
+#define ODBC_CODEGEN_Code_query_group_hpp
+
+#include <functional>
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_expr_row.hpp"
+#include "Code_pred.hpp"
+
+/**
+ * @class Plan_query_group
+ * @brief Group-by node in PlanTree
+ */
+class Plan_query_group : public Plan_query {
+public:
+ Plan_query_group(Plan_root* root);
+ virtual ~Plan_query_group();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setQuery(Plan_query* query);
+ void setDataRow(Plan_expr_row* dataRow);
+ void setGroupRow(Plan_expr_row* groupRow);
+ void setHavingPred(Plan_pred* havingPred);
+ Plan_expr_row* getRow();
+protected:
+ Plan_query* m_query;
+ Plan_expr_row* m_dataRow;
+ Plan_expr_row* m_groupRow;
+ Plan_pred* m_havingPred;
+};
+
+inline
+Plan_query_group::Plan_query_group(Plan_root* root) :
+ Plan_query(root),
+ m_query(0),
+ m_dataRow(0),
+ m_groupRow(0),
+ m_havingPred(0)
+{
+}
+
+// children
+
+inline void
+Plan_query_group::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Plan_query_group::setDataRow(Plan_expr_row* dataRow)
+{
+ ctx_assert(dataRow != 0);
+ m_dataRow = dataRow;
+}
+
+inline void
+Plan_query_group::setGroupRow(Plan_expr_row* groupRow)
+{
+ ctx_assert(groupRow != 0);
+ m_groupRow = groupRow;
+}
+
+inline void
+Plan_query_group::setHavingPred(Plan_pred* havingPred)
+{
+ ctx_assert(havingPred != 0);
+ m_havingPred = havingPred;
+}
+
+/**
+ * Group-by uses a std::map. Key is values grouped by. Data is unique index
+ * (starting at 1) into arrays in expression data.
+ */
+
+class Exec_query_group;
+
+struct GroupLess : std::binary_function<const SqlRow*, const SqlRow*, bool> {
+ bool operator()(const SqlRow* s1, const SqlRow* s2) const;
+};
+
+typedef std::map<const SqlRow*, unsigned, GroupLess> GroupList;
+
+/**
+ * @class Exec_query_group
+ * @brief Group-by node in ExecTree
+ */
+class Exec_query_group : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(const SqlSpecs& sqlSpecs);
+ virtual ~Code();
+ protected:
+ friend class Exec_query_group;
+ // sets reference to Sqlspecs from subquery
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_group* node, const SqlSpecs& sqlSpecs);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_group;
+ SqlRow m_sqlRow; // current row
+ bool m_grouped; // fetch and group-by done
+ unsigned m_count;
+ GroupList m_groupList;
+ GroupList::iterator m_iterator;
+ };
+ Exec_query_group(Exec_root* root);
+ virtual ~Exec_query_group();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+ void setDataRow(Exec_expr_row* dataRow);
+ void setGroupRow(Exec_expr_row* groupRow);
+ void setHavingPred(Exec_pred* havingPred);
+ const Exec_query* getRawQuery() const;
+protected:
+ friend class Exec_query;
+ Exec_query* m_query;
+ Exec_expr_row* m_dataRow;
+ Exec_expr_row* m_groupRow;
+ Exec_pred* m_havingPred;
+};
+
+inline
+Exec_query_group::Code::Code(const SqlSpecs& sqlSpecs) :
+ Exec_query::Code(sqlSpecs)
+{
+}
+
+inline
+Exec_query_group::Data::Data(Exec_query_group* node, const SqlSpecs& sqlSpecs) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlSpecs),
+ m_grouped(false),
+ m_count(0)
+{
+}
+
+inline
+Exec_query_group::Exec_query_group(Exec_root* root) :
+ Exec_query(root),
+ m_query(0),
+ m_dataRow(0),
+ m_groupRow(0),
+ m_havingPred(0)
+{
+}
+
+// children
+
+inline const Exec_query_group::Code&
+Exec_query_group::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_group::Data&
+Exec_query_group::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_query_group::setQuery(Exec_query* query)
+{
+ ctx_assert(m_query == 0 && query != 0);
+ m_query = query;
+}
+
+inline void
+Exec_query_group::setDataRow(Exec_expr_row* dataRow)
+{
+ ctx_assert(m_dataRow == 0 && dataRow != 0);
+ m_dataRow = dataRow;
+}
+
+inline void
+Exec_query_group::setGroupRow(Exec_expr_row* groupRow)
+{
+ ctx_assert(m_groupRow == 0 && groupRow != 0);
+ m_groupRow = groupRow;
+}
+
+inline void
+Exec_query_group::setHavingPred(Exec_pred* havingPred)
+{
+ ctx_assert(m_havingPred == 0 && havingPred != 0);
+ m_havingPred = havingPred;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_index.cpp b/ndb/src/client/odbc/codegen/Code_query_index.cpp
new file mode 100644
index 00000000000..ee19d6123cc
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_index.cpp
@@ -0,0 +1,186 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_query_index.hpp"
+#include "Code_column.hpp"
+#include "Code_expr.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_index
+
+Plan_query_index::~Plan_query_index()
+{
+}
+
+Plan_base*
+Plan_query_index::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_query_index::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // set up
+ ctx_assert(m_table != 0 && m_index != 0);
+ const BaseString& tableName = m_table->getName();
+ ctx_assert(m_index->m_dictIndex != 0);
+ const DictIndex& dictIndex = *m_index->m_dictIndex;
+ const BaseString& indexName = dictIndex.getName();
+ const unsigned keyCount = m_index->m_keyCount;
+ const ColumnVector& columns = m_table->exprColumns();
+ ctx_assert(columns.size() > 0);
+ const unsigned attrCount = columns.size() - 1;
+ // create the code
+ Exec_query_index::Code& code = *new Exec_query_index::Code(keyCount, attrCount);
+ code.m_tableName = strcpy(new char[tableName.length() + 1], tableName.c_str());
+ code.m_indexName = strcpy(new char[indexName.length() + 1], indexName.c_str());
+ // key attributes
+ code.m_keyId = new NdbAttrId[1 + keyCount];
+ code.m_keyId[0] = (NdbAttrId)-1;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ const DictColumn* keyColumn = dictIndex.getColumn(k);
+ const SqlType& sqlType = keyColumn->sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_keySpecs.setEntry(k, sqlSpec);
+ code.m_keyId[k] = k - 1; // index column order
+ }
+ // matching expressions
+ ctx_assert(m_index->m_keyFound);
+ const ExprVector& keyEq = m_index->m_keyEq;
+ ctx_assert(keyEq.size() == 1 + keyCount);
+ code.m_keyMatch = new Exec_expr* [1 + keyCount];
+ code.m_keyMatch[0] = 0;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ Plan_expr* expr = keyEq[k];
+ Exec_expr* execExpr = static_cast<Exec_expr*>(expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ code.m_keyMatch[k] = execExpr;
+ }
+ // queried attributes
+ code.m_attrId = new NdbAttrId[1 + attrCount];
+ code.m_attrId[0] = (NdbAttrId)-1;
+ for (unsigned i = 1; i <= attrCount; i++) {
+ Plan_column* column = columns[i];
+ ctx_assert(column != 0);
+ const DictColumn& dictColumn = column->dictColumn();
+ const SqlType& sqlType = dictColumn.sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_sqlSpecs.setEntry(i, sqlSpec);
+ code.m_attrId[i] = dictColumn.getAttrId();
+ }
+ // create the exec
+ Exec_query_index* exec = new Exec_query_index(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ return exec;
+}
+
+void
+Plan_query_index::print(Ctx& ctx)
+{
+ ctx.print(" [query_index");
+ Plan_base* a[] = { m_table };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
+
+// Exec_query_index
+
+Exec_query_index::Code::~Code()
+{
+ delete[] m_tableName;
+ delete[] m_keyId;
+ delete[] m_keyMatch;
+ delete[] m_attrId;
+}
+
+Exec_query_index::Data::~Data()
+{
+ delete[] m_recAttr;
+}
+
+Exec_query_index::~Exec_query_index()
+{
+}
+
+void
+Exec_query_index::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // create data
+ Data& data = *new Data(this, code.sqlSpecs());
+ setData(data);
+ // allocate matching expressions
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* expr = code.m_keyMatch[k];
+ ctx_assert(expr != 0);
+ expr->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ // needed for isNULL
+ data.m_recAttr = new NdbRecAttr* [1 + code.m_attrCount];
+ for (unsigned i = 0; i <= code.m_attrCount; i++) {
+ data.m_recAttr[i] = 0;
+ }
+}
+
+void
+Exec_query_index::close(Ctx& ctx)
+{
+ Data& data = getData();
+ if (data.m_con != 0) {
+ Ndb* const ndb = ndbObject();
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ data.m_done = true;
+ ctx_log2(("lookup closed at statement close"));
+ }
+}
+
+void
+Exec_query_index::print(Ctx& ctx)
+{
+ ctx.print(" [query_index");
+ if (m_code != 0) {
+ const Code& code = getCode();
+ ctx.print(" keyId=");
+ for (unsigned i = 1; i <= code.m_keyCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_keyId[i]);
+ }
+ ctx.print(" attrId=");
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_attrId[i]);
+ }
+ ctx.print(" table=%s", code.m_tableName);
+ }
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_index.hpp b/ndb/src/client/odbc/codegen/Code_query_index.hpp
new file mode 100644
index 00000000000..87affd50580
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_index.hpp
@@ -0,0 +1,160 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_index_hpp
+#define ODBC_CODEGEN_Code_query_index_hpp
+
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_table.hpp"
+
+class Ctx;
+class StmtArea;
+class NdbConnection;
+class NdbOperation;
+class NdbRecAttr;
+
+/**
+ * @class Plan_query_index
+ * @brief Full select (no where clause)
+ */
+class Plan_query_index : public Plan_query {
+public:
+ Plan_query_index(Plan_root* root);
+ virtual ~Plan_query_index();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setTable(Plan_table* table, Plan_table::Index* index);
+protected:
+ Plan_table* m_table;
+ Plan_table::Index* m_index;
+};
+
+inline
+Plan_query_index::Plan_query_index(Plan_root* root) :
+ Plan_query(root),
+ m_table(0),
+ m_index(0)
+{
+}
+
+// children
+
+inline void
+Plan_query_index::setTable(Plan_table* table, Plan_table::Index* index)
+{
+ ctx_assert(table != 0 && index != 0 && index == &table->m_indexList[index->m_pos] && index->m_pos != 0);
+ m_table = table;
+ m_index = index;
+}
+
+/**
+ * @class Exec_query_index
+ * @brief Full select (no where clause)
+ */
+class Exec_query_index : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(unsigned keyCount, unsigned attrCount);
+ virtual ~Code();
+ protected:
+ friend class Plan_query_index;
+ friend class Exec_query_index;
+ const char* m_tableName;
+ const char* m_indexName;
+ unsigned m_keyCount;
+ SqlSpecs m_keySpecs; // key types
+ NdbAttrId* m_keyId;
+ Exec_expr** m_keyMatch; // XXX pointers for now
+ unsigned m_attrCount;
+ SqlSpecs m_sqlSpecs;
+ NdbAttrId* m_attrId;
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_index* node, const SqlSpecs& sqlSpecs);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_index;
+ SqlRow m_sqlRow;
+ NdbConnection* m_con;
+ NdbOperation* m_op;
+ NdbRecAttr** m_recAttr;
+ bool m_done; // returns one row
+ };
+ Exec_query_index(Exec_root* root);
+ virtual ~Exec_query_index();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_query_index::Code::Code(unsigned keyCount, unsigned attrCount) :
+ Exec_query::Code(m_sqlSpecs),
+ m_tableName(0),
+ m_indexName(0),
+ m_keyCount(keyCount),
+ m_keySpecs(keyCount),
+ m_keyId(0),
+ m_attrCount(attrCount),
+ m_sqlSpecs(attrCount),
+ m_attrId(0)
+{
+}
+
+inline
+Exec_query_index::Data::Data(Exec_query_index* node, const SqlSpecs& sqlSpecs) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlSpecs),
+ m_con(0),
+ m_op(0),
+ m_recAttr(0),
+ m_done(false)
+{
+}
+
+inline
+Exec_query_index::Exec_query_index(Exec_root* root) :
+ Exec_query(root)
+{
+}
+
+// children
+
+inline const Exec_query_index::Code&
+Exec_query_index::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_index::Data&
+Exec_query_index::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_join.cpp b/ndb/src/client/odbc/codegen/Code_query_join.cpp
new file mode 100644
index 00000000000..89aafe13610
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_join.cpp
@@ -0,0 +1,192 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_query.hpp"
+#include "Code_query_join.hpp"
+#include "Code_query_scan.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_join
+
+Plan_query_join::~Plan_query_join()
+{
+}
+
+Plan_base*
+Plan_query_join::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_inner != 0);
+ m_inner->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_outer != 0);
+ m_outer->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_query_join::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // generate code for subqueries
+ ctx_assert(m_inner != 0);
+ Exec_query* execInner = static_cast<Exec_query*>(m_inner->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execInner != 0);
+ ctx_assert(m_outer != 0);
+ ctl.m_execQuery = execInner;
+ Exec_query* execOuter = static_cast<Exec_query*>(m_outer->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execOuter != 0);
+ // combine sql specs from subqueries
+ const SqlSpecs& specsInner = execInner->getCode().sqlSpecs();
+ const SqlSpecs& specsOuter = execOuter->getCode().sqlSpecs();
+ SqlSpecs sqlSpecs(specsInner.count() + specsOuter.count());
+ for (unsigned i = 1; i <= specsInner.count(); i++) {
+ const SqlSpec sqlSpec(specsInner.getEntry(i), SqlSpec::Reference);
+ sqlSpecs.setEntry(i, sqlSpec);
+ }
+ for (unsigned i = 1; i <= specsOuter.count(); i++) {
+ const SqlSpec sqlSpec(specsOuter.getEntry(i), SqlSpec::Reference);
+ sqlSpecs.setEntry(specsInner.count() + i, sqlSpec);
+ }
+ // create the code
+ Exec_query_join* exec = new Exec_query_join(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setInner(execInner);
+ exec->setOuter(execOuter);
+ Exec_query_join::Code& code = *new Exec_query_join::Code(sqlSpecs);
+ exec->setCode(code);
+ return exec;
+}
+
+void
+Plan_query_join::print(Ctx& ctx)
+{
+ ctx.print(" [query_join");
+ Plan_base* a[] = { m_inner, m_outer };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
+
+// Exec_query_join
+
+Exec_query_join::Code::~Code()
+{
+}
+
+Exec_query_join::Data::~Data()
+{
+}
+
+Exec_query_join::~Exec_query_join()
+{
+}
+
+void
+Exec_query_join::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate the subqueries
+ ctx_assert(m_inner != 0);
+ m_inner->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ ctx_assert(m_outer != 0);
+ ctl.m_query = m_inner;
+ m_outer->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // combine data rows from subqueries
+ const Code& code = getCode();
+ const SqlRow& rowInner = m_inner->getData().sqlRow();
+ const SqlRow& rowOuter = m_outer->getData().sqlRow();
+ SqlRow sqlRow(code.m_sqlSpecs);
+ for (unsigned i = 1; i <= rowInner.count(); i++) {
+ const SqlSpec& sqlSpec = code.m_sqlSpecs.getEntry(i);
+ const SqlField sqlField(sqlSpec, &rowInner.getEntry(i));
+ sqlRow.setEntry(i, sqlField);
+ }
+ for (unsigned i = 1; i <= rowOuter.count(); i++) {
+ const SqlSpec& sqlSpec = code.m_sqlSpecs.getEntry(rowInner.count() + i);
+ const SqlField sqlField(sqlSpec, &rowOuter.getEntry(i));
+ sqlRow.setEntry(rowInner.count() + i, sqlField);
+ }
+ // create the data
+ Data& data = *new Data(this, sqlRow);
+ setData(data);
+}
+
+void
+Exec_query_join::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ // execute only inner query
+ ctx_assert(m_inner != 0);
+ m_inner->execute(ctx, ctl);
+}
+
+bool
+Exec_query_join::fetchImpl(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_inner != 0);
+ ctx_assert(m_outer != 0);
+ if (getData().getState() == ResultSet::State_init) {
+ // fetch first row from inner
+ if (! m_inner->fetch(ctx, ctl))
+ return false;
+ }
+ while (1) {
+ if (m_outer->getData().getState() == ResultSet::State_end) {
+ // execute or re-execute outer
+ Ctl ctl(0);
+ m_outer->close(ctx);
+ if (! ctx.ok())
+ return false;
+ m_outer->execute(ctx, ctl);
+ if (! ctx.ok())
+ return false;
+ }
+ if (! m_outer->fetch(ctx, ctl)) {
+ if (! ctx.ok())
+ return false;
+ // fetch next row from inner
+ if (! m_inner->fetch(ctx, ctl))
+ return false;
+ }
+ else
+ return true;
+ }
+}
+
+void
+Exec_query_join::close(Ctx& ctx)
+{
+ ctx_assert(m_inner != 0);
+ m_inner->close(ctx);
+ ctx_assert(m_outer != 0);
+ m_outer->close(ctx);
+}
+
+void
+Exec_query_join::print(Ctx& ctx)
+{
+ ctx.print(" [query_join");
+ Exec_base* a[] = { m_inner, m_outer };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_join.hpp b/ndb/src/client/odbc/codegen/Code_query_join.hpp
new file mode 100644
index 00000000000..f6ac9205329
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_join.hpp
@@ -0,0 +1,159 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_join_hpp
+#define ODBC_CODEGEN_Code_query_join_hpp
+
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_table_list.hpp"
+#include "Code_pred.hpp"
+
+/**
+ * @class Plan_query_join
+ * @brief Filter node in PlanTree
+ */
+class Plan_query_join : public Plan_query {
+public:
+ Plan_query_join(Plan_root* root);
+ virtual ~Plan_query_join();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setInner(Plan_query* query);
+ void setOuter(Plan_query* query);
+protected:
+ Plan_query* m_inner;
+ Plan_query* m_outer;
+};
+
+inline
+Plan_query_join::Plan_query_join(Plan_root* root) :
+ Plan_query(root),
+ m_inner(0),
+ m_outer(0)
+{
+}
+
+// children
+
+inline void
+Plan_query_join::setInner(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_inner = query;
+}
+
+inline void
+Plan_query_join::setOuter(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_outer = query;
+}
+
+/**
+ * @class Exec_query_join
+ * @brief Filter node in ExecTree
+ */
+class Exec_query_join : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(const SqlSpecs& sqlSpecs);
+ virtual ~Code();
+ protected:
+ friend class Exec_query_join;
+ SqlSpecs m_sqlSpecs;
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_join* node, const SqlRow& sqlRow);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_join;
+ SqlRow m_sqlRow;
+ };
+ Exec_query_join(Exec_root* root);
+ virtual ~Exec_query_join();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setInner(Exec_query* query);
+ void setOuter(Exec_query* query);
+protected:
+ Exec_query* m_inner;
+ Exec_query* m_outer;
+};
+
+inline
+Exec_query_join::Code::Code(const SqlSpecs& sqlSpecs) :
+ Exec_query::Code(m_sqlSpecs),
+ m_sqlSpecs(sqlSpecs)
+{
+}
+
+inline
+Exec_query_join::Data::Data(Exec_query_join* node, const SqlRow& sqlRow) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlRow)
+{
+}
+
+inline
+Exec_query_join::Exec_query_join(Exec_root* root) :
+ Exec_query(root),
+ m_inner(0),
+ m_outer(0)
+{
+}
+
+// children
+
+inline const Exec_query_join::Code&
+Exec_query_join::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_join::Data&
+Exec_query_join::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_query_join::setInner(Exec_query* query)
+{
+ ctx_assert(query != 0);
+ m_inner = query;
+}
+
+inline void
+Exec_query_join::setOuter(Exec_query* query)
+{
+ ctx_assert(query != 0);
+ m_outer = query;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_lookup.cpp b/ndb/src/client/odbc/codegen/Code_query_lookup.cpp
new file mode 100644
index 00000000000..bad4199190b
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_lookup.cpp
@@ -0,0 +1,184 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_query_lookup.hpp"
+#include "Code_column.hpp"
+#include "Code_expr.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_lookup
+
+Plan_query_lookup::~Plan_query_lookup()
+{
+}
+
+Plan_base*
+Plan_query_lookup::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_query_lookup::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // set up
+ ctx_assert(m_table != 0);
+ const BaseString& tableName = m_table->getName();
+ const DictTable& dictTable = m_table->dictTable();
+ const unsigned keyCount = dictTable.keyCount();
+ const ColumnVector& columns = m_table->exprColumns();
+ ctx_assert(columns.size() > 0);
+ const unsigned attrCount = columns.size() - 1;
+ // create the code
+ Exec_query_lookup::Code& code = *new Exec_query_lookup::Code(keyCount, attrCount);
+ code.m_tableName = strcpy(new char[tableName.length() + 1], tableName.c_str());
+ // key attributes
+ code.m_keyId = new NdbAttrId[1 + keyCount];
+ code.m_keyId[0] = (NdbAttrId)-1;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ const DictColumn* keyColumn = dictTable.getKey(k);
+ const SqlType& sqlType = keyColumn->sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_keySpecs.setEntry(k, sqlSpec);
+ code.m_keyId[k] = keyColumn->getAttrId();
+ }
+ // matching expressions
+ const Plan_table::Index& index = m_table->m_indexList[0];
+ ctx_assert(index.m_keyFound);
+ const ExprVector& keyEq = index.m_keyEq;
+ ctx_assert(keyEq.size() == 1 + keyCount);
+ code.m_keyMatch = new Exec_expr* [1 + keyCount];
+ code.m_keyMatch[0] = 0;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ Plan_expr* expr = keyEq[k];
+ Exec_expr* execExpr = static_cast<Exec_expr*>(expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ code.m_keyMatch[k] = execExpr;
+ }
+ // queried attributes
+ code.m_attrId = new NdbAttrId[1 + attrCount];
+ code.m_attrId[0] = (NdbAttrId)-1;
+ for (unsigned i = 1; i <= attrCount; i++) {
+ Plan_column* column = columns[i];
+ ctx_assert(column != 0);
+ const DictColumn& dictColumn = column->dictColumn();
+ const SqlType& sqlType = dictColumn.sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_sqlSpecs.setEntry(i, sqlSpec);
+ code.m_attrId[i] = dictColumn.getAttrId();
+ }
+ // create the exec
+ Exec_query_lookup* exec = new Exec_query_lookup(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ return exec;
+}
+
+void
+Plan_query_lookup::print(Ctx& ctx)
+{
+ ctx.print(" [query_lookup");
+ Plan_base* a[] = { m_table };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
+
+// Exec_query_lookup
+
+Exec_query_lookup::Code::~Code()
+{
+ delete[] m_tableName;
+ delete[] m_keyId;
+ delete[] m_keyMatch;
+ delete[] m_attrId;
+}
+
+Exec_query_lookup::Data::~Data()
+{
+ delete[] m_recAttr;
+}
+
+Exec_query_lookup::~Exec_query_lookup()
+{
+}
+
+void
+Exec_query_lookup::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // create data
+ Data& data = *new Data(this, code.sqlSpecs());
+ setData(data);
+ // allocate matching expressions
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* expr = code.m_keyMatch[k];
+ ctx_assert(expr != 0);
+ expr->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ // needed for isNULL
+ data.m_recAttr = new NdbRecAttr* [1 + code.m_attrCount];
+ for (unsigned i = 0; i <= code.m_attrCount; i++) {
+ data.m_recAttr[i] = 0;
+ }
+}
+
+void
+Exec_query_lookup::close(Ctx& ctx)
+{
+ Data& data = getData();
+ if (data.m_con != 0) {
+ Ndb* const ndb = ndbObject();
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ data.m_done = true;
+ ctx_log2(("lookup closed at statement close"));
+ }
+}
+
+void
+Exec_query_lookup::print(Ctx& ctx)
+{
+ ctx.print(" [query_lookup");
+ if (m_code != 0) {
+ const Code& code = getCode();
+ ctx.print(" keyId=");
+ for (unsigned i = 1; i <= code.m_keyCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_keyId[i]);
+ }
+ ctx.print(" attrId=");
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_attrId[i]);
+ }
+ ctx.print(" table=%s", code.m_tableName);
+ }
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_lookup.hpp b/ndb/src/client/odbc/codegen/Code_query_lookup.hpp
new file mode 100644
index 00000000000..e66623d4030
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_lookup.hpp
@@ -0,0 +1,155 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_lookup_hpp
+#define ODBC_CODEGEN_Code_query_lookup_hpp
+
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_table.hpp"
+
+class Ctx;
+class StmtArea;
+class NdbConnection;
+class NdbOperation;
+class NdbRecAttr;
+
+/**
+ * @class Plan_query_lookup
+ * @brief Full select (no where clause)
+ */
+class Plan_query_lookup : public Plan_query {
+public:
+ Plan_query_lookup(Plan_root* root);
+ virtual ~Plan_query_lookup();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setTable(Plan_table* table);
+protected:
+ Plan_table* m_table;
+};
+
+inline
+Plan_query_lookup::Plan_query_lookup(Plan_root* root) :
+ Plan_query(root),
+ m_table(0)
+{
+}
+
+// children
+
+inline void
+Plan_query_lookup::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+/**
+ * @class Exec_query_lookup
+ * @brief Full select (no where clause)
+ */
+class Exec_query_lookup : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(unsigned keyCount, unsigned attrCount);
+ virtual ~Code();
+ protected:
+ friend class Plan_query_lookup;
+ friend class Exec_query_lookup;
+ char* m_tableName;
+ unsigned m_keyCount;
+ SqlSpecs m_keySpecs; // key types
+ NdbAttrId* m_keyId;
+ Exec_expr** m_keyMatch; // XXX pointers for now
+ unsigned m_attrCount;
+ SqlSpecs m_sqlSpecs;
+ NdbAttrId* m_attrId;
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_lookup* node, const SqlSpecs& sqlSpecs);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_lookup;
+ SqlRow m_sqlRow;
+ NdbConnection* m_con;
+ NdbOperation* m_op;
+ NdbRecAttr** m_recAttr;
+ bool m_done; // returns one row
+ };
+ Exec_query_lookup(Exec_root* root);
+ virtual ~Exec_query_lookup();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_query_lookup::Code::Code(unsigned keyCount, unsigned attrCount) :
+ Exec_query::Code(m_sqlSpecs),
+ m_tableName(0),
+ m_keyCount(keyCount),
+ m_keySpecs(keyCount),
+ m_keyId(0),
+ m_attrCount(attrCount),
+ m_sqlSpecs(attrCount),
+ m_attrId(0)
+{
+}
+
+inline
+Exec_query_lookup::Data::Data(Exec_query_lookup* node, const SqlSpecs& sqlSpecs) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlSpecs),
+ m_con(0),
+ m_op(0),
+ m_recAttr(0),
+ m_done(false)
+{
+}
+
+inline
+Exec_query_lookup::Exec_query_lookup(Exec_root* root) :
+ Exec_query(root)
+{
+}
+
+// children
+
+inline const Exec_query_lookup::Code&
+Exec_query_lookup::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_lookup::Data&
+Exec_query_lookup::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_project.cpp b/ndb/src/client/odbc/codegen/Code_query_project.cpp
new file mode 100644
index 00000000000..54043ce3d5d
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_project.cpp
@@ -0,0 +1,184 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_query_project.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_project
+
+Plan_query_project::~Plan_query_project()
+{
+}
+
+Plan_base*
+Plan_query_project::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_exprRow != 0);
+ ctl.m_aggrok = true;
+ ctl.m_aggrin = false;
+ m_exprRow->analyze(ctx, ctl);
+ ctl.m_aggrok = false;
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Plan_expr_row*
+Plan_query_project::getRow()
+{
+ ctx_assert(m_exprRow != 0);
+ return m_exprRow;
+}
+
+Exec_base*
+Plan_query_project::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // create code for the subquery
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // create code for the row based on query code
+ ctx_assert(m_exprRow != 0);
+ ctl.m_execQuery = execQuery;
+ Exec_expr_row* execRow = static_cast<Exec_expr_row*>(m_exprRow->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execRow != 0);
+ Exec_query_project* exec = new Exec_query_project(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // re-use SqlSpecs from the row
+ const SqlSpecs& sqlSpecs = execRow->getCode().sqlSpecs();
+ Exec_query_project::Code& code = *new Exec_query_project::Code(sqlSpecs);
+ code.m_limitOff = m_limitOff;
+ code.m_limitCnt = m_limitCnt;
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ exec->setRow(execRow);
+ return exec;
+}
+
+void
+Plan_query_project::print(Ctx& ctx)
+{
+ ctx.print(" [query_project");
+ Plan_base* a[] = { m_query, m_exprRow };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
+
+// Exec_query_project
+
+Exec_query_project::Code::~Code()
+{
+}
+
+Exec_query_project::Data::~Data()
+{
+}
+
+Exec_query_project::~Exec_query_project()
+{
+}
+
+const Exec_query*
+Exec_query_project::getRawQuery() const
+{
+ ctx_assert(m_query != 0);
+ return m_query;
+}
+
+void
+Exec_query_project::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate the subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // allocate the row based on subquery data
+ ctx_assert(m_exprRow != 0);
+ ctl.m_query = m_query;
+ m_exprRow->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // re-use SqlRow from the expression row
+ const SqlRow& sqlRow = m_exprRow->getData().sqlRow();
+ Data& data = *new Data(this, sqlRow);
+ setData(data);
+}
+
+void
+Exec_query_project::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+}
+
+bool
+Exec_query_project::fetchImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ ctx_assert(m_query != 0);
+ while (1) {
+ if (! m_query->fetch(ctx, ctl))
+ return false;
+ ctx_assert(m_exprRow != 0);
+ m_exprRow->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return false;
+ ctl.m_postEval = true;
+ m_exprRow->evaluate(ctx, ctl);
+ ctl.m_postEval = false;
+ const int n = ++data.m_cnt;
+ const int o = code.m_limitOff <= 0 ? 0 : code.m_limitOff;
+ const int c = code.m_limitCnt;
+ if (n <= o)
+ continue;
+ if (c < 0)
+ break;
+ if (n - o <= c)
+ break;
+ return false;
+ }
+ return true;
+}
+
+void
+Exec_query_project::close(Ctx& ctx)
+{
+ Data& data = getData();
+ data.m_cnt = 0;
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+ ctx_assert(m_exprRow != 0);
+ m_exprRow->close(ctx);
+}
+
+void
+Exec_query_project::print(Ctx& ctx)
+{
+ ctx.print(" [query_project");
+ Exec_base* a[] = { m_query, m_exprRow };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_project.hpp b/ndb/src/client/odbc/codegen/Code_query_project.hpp
new file mode 100644
index 00000000000..545685ab9df
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_project.hpp
@@ -0,0 +1,178 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_project_hpp
+#define ODBC_CODEGEN_Code_query_project_hpp
+
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_expr_row.hpp"
+
+/**
+ * @class Plan_query_project
+ * @brief Project node in PlanTree
+ */
+class Plan_query_project : public Plan_query {
+public:
+ Plan_query_project(Plan_root* root);
+ virtual ~Plan_query_project();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setQuery(Plan_query* query);
+ void setRow(Plan_expr_row* exprRow);
+ void setLimit(int off, int cnt);
+protected:
+ Plan_expr_row* getRow();
+ Plan_query* m_query;
+ Plan_expr_row* m_exprRow;
+ int m_limitOff;
+ int m_limitCnt;
+};
+
+inline
+Plan_query_project::Plan_query_project(Plan_root* root) :
+ Plan_query(root),
+ m_query(0),
+ m_exprRow(0),
+ m_limitOff(0),
+ m_limitCnt(-1)
+{
+}
+
+// children
+
+inline void
+Plan_query_project::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Plan_query_project::setRow(Plan_expr_row* exprRow)
+{
+ ctx_assert(exprRow != 0);
+ m_exprRow = exprRow;
+}
+
+inline void
+Plan_query_project::setLimit(int off, int cnt)
+{
+ m_limitOff = off;
+ m_limitCnt = cnt;
+}
+
+/**
+ * @class Exec_query_project
+ * @brief Project node in ExecTree
+ */
+class Exec_query_project : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(const SqlSpecs& sqlSpecs);
+ virtual ~Code();
+ protected:
+ friend class Plan_query_project;
+ friend class Exec_query_project;
+ // sets reference to Sqlspecs from the row
+ int m_limitOff;
+ int m_limitCnt;
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_project* node, const SqlRow& sqlRow);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_project;
+ // sets reference to SqlRow from the row
+ unsigned m_cnt;
+ };
+ Exec_query_project(Exec_root* root);
+ virtual ~Exec_query_project();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+ void setRow(Exec_expr_row* exprRow);
+ const Exec_query* getRawQuery() const;
+protected:
+ friend class Exec_query;
+ Exec_query* m_query;
+ Exec_expr_row* m_exprRow;
+};
+
+inline
+Exec_query_project::Code::Code(const SqlSpecs& sqlSpecs) :
+ Exec_query::Code(sqlSpecs),
+ m_limitOff(0),
+ m_limitCnt(-1)
+{
+}
+
+inline
+Exec_query_project::Data::Data(Exec_query_project* node, const SqlRow& sqlRow) :
+ Exec_query::Data(node, sqlRow),
+ m_cnt(0)
+{
+}
+
+inline
+Exec_query_project::Exec_query_project(Exec_root* root) :
+ Exec_query(root),
+ m_query(0),
+ m_exprRow(0)
+{
+}
+
+// children
+
+inline const Exec_query_project::Code&
+Exec_query_project::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_project::Data&
+Exec_query_project::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_query_project::setQuery(Exec_query* query)
+{
+ ctx_assert(m_query == 0 && query != 0);
+ m_query = query;
+}
+
+inline void
+Exec_query_project::setRow(Exec_expr_row* exprRow)
+{
+ ctx_assert(m_exprRow == 0 && exprRow != 0);
+ m_exprRow = exprRow;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_range.cpp b/ndb/src/client/odbc/codegen/Code_query_range.cpp
new file mode 100644
index 00000000000..5d29c5af315
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_range.cpp
@@ -0,0 +1,211 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_query_range.hpp"
+#include "Code_column.hpp"
+#include "Code_expr.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_range
+
+Plan_query_range::~Plan_query_range()
+{
+}
+
+Plan_base*
+Plan_query_range::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_query_range::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // set up
+ ctx_assert(m_table != 0 && m_index != 0);
+ const BaseString& tableName = m_table->getName();
+ ctx_assert(m_index->m_dictIndex != 0);
+ const DictIndex& dictIndex = *m_index->m_dictIndex;
+ const BaseString& indexName = dictIndex.getName();
+ const unsigned keyCount = m_index->m_keyCountUsed;
+ const ColumnVector& columns = m_table->exprColumns();
+ ctx_assert(columns.size() > 0);
+ const unsigned attrCount = columns.size() - 1;
+ // create the code
+ Exec_query_range::Code& code = *new Exec_query_range::Code(keyCount, attrCount);
+ code.m_tableName = strcpy(new char[tableName.length() + 1], tableName.c_str());
+ code.m_indexName = strcpy(new char[indexName.length() + 1], indexName.c_str());
+ code.m_exclusive = m_exclusive;
+ // key attributes
+ code.m_keyId = new NdbAttrId[1 + keyCount];
+ code.m_keyId[0] = (NdbAttrId)-1;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ const DictColumn* keyColumn = dictIndex.getColumn(k);
+ const SqlType& sqlType = keyColumn->sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_keySpecs.setEntry(k, sqlSpec);
+ code.m_keyId[k] = k - 1; // index column order
+ }
+ // matching expressions
+ ctx_assert(m_index->m_keyFound);
+ const ExprVector& keyEq = m_index->m_keyEq;
+ // check size matches
+ ctx_assert(keyEq.size() == 1 + keyCount);
+ code.m_keyMatch = new Exec_expr* [1 + keyCount];
+ code.m_keyMatch[0] = 0;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ Plan_expr* expr = keyEq[k];
+ Exec_expr* execExpr = static_cast<Exec_expr*>(expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ code.m_keyMatch[k] = execExpr;
+ }
+ // queried attributes
+ code.m_attrId = new NdbAttrId[1 + attrCount];
+ code.m_attrId[0] = (NdbAttrId)-1;
+ for (unsigned i = 1; i <= attrCount; i++) {
+ Plan_column* column = columns[i];
+ ctx_assert(column != 0);
+ const DictColumn& dictColumn = column->dictColumn();
+ const SqlType& sqlType = dictColumn.sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_sqlSpecs.setEntry(i, sqlSpec);
+ code.m_attrId[i] = dictColumn.getAttrId();
+ }
+ // create the exec
+ Exec_query_range* exec = new Exec_query_range(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ // interpreter
+ ctl.m_execQuery = exec;
+ Exec_pred* execInterp = 0;
+ ctl.m_topTable = m_table;
+ if (m_interp != 0) {
+ execInterp = static_cast<Exec_pred*>(m_interp->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execInterp != 0);
+ }
+ ctl.m_topTable = 0;
+ if (m_interp != 0)
+ exec->setInterp(execInterp);
+ return exec;
+}
+
+void
+Plan_query_range::print(Ctx& ctx)
+{
+ ctx.print(" [query_range");
+ Plan_base* a[] = { m_table };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
+
+// Exec_query_range
+
+Exec_query_range::Code::~Code()
+{
+ delete[] m_tableName;
+ delete[] m_keyId;
+ delete[] m_keyMatch;
+ delete[] m_attrId;
+}
+
+Exec_query_range::Data::~Data()
+{
+ delete[] m_recAttr;
+}
+
+Exec_query_range::~Exec_query_range()
+{
+}
+
+void
+Exec_query_range::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // create data
+ Data& data = *new Data(this, code.sqlSpecs());
+ setData(data);
+ // allocate matching expressions
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* expr = code.m_keyMatch[k];
+ ctx_assert(expr != 0);
+ expr->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+ // needed for isNULL
+ data.m_recAttr = new NdbRecAttr* [1 + code.m_attrCount];
+ for (unsigned i = 0; i <= code.m_attrCount; i++) {
+ data.m_recAttr[i] = 0;
+ }
+ // parallel
+ data.m_parallel = code.m_exclusive ? 1 : 240; // best supported
+ // interpreter
+ if (m_interp != 0) {
+ //m_interp->alloc(ctx, ctl); XXX
+ if (! ctx.ok())
+ return;
+ }
+}
+
+void
+Exec_query_range::close(Ctx& ctx)
+{
+ Data& data = getData();
+ if (data.m_con != 0) {
+ Ndb* const ndb = ndbObject();
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ data.m_done = true;
+ ctx_log2(("lookup closed at statement close"));
+ }
+ // if (m_interp != 0)
+ // m_interp->close(ctx);
+}
+
+void
+Exec_query_range::print(Ctx& ctx)
+{
+ ctx.print(" [query_range");
+ if (m_code != 0) {
+ const Code& code = getCode();
+ ctx.print(" keyId=");
+ for (unsigned i = 1; i <= code.m_keyCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_keyId[i]);
+ }
+ ctx.print(" attrId=");
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_attrId[i]);
+ }
+ ctx.print(" table=%s", code.m_tableName);
+ }
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_range.hpp b/ndb/src/client/odbc/codegen/Code_query_range.hpp
new file mode 100644
index 00000000000..4438189522c
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_range.hpp
@@ -0,0 +1,186 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_range_hpp
+#define ODBC_CODEGEN_Code_query_range_hpp
+
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_table.hpp"
+#include "Code_pred.hpp"
+
+class Ctx;
+class StmtArea;
+class NdbConnection;
+class NdbOperation;
+class NdbRecAttr;
+
+/*
+ * Range scan via ordered index. We implement only the case of equality
+ * on an initial sequence of index keys.
+ */
+
+class Plan_query_range : public Plan_query {
+public:
+ Plan_query_range(Plan_root* root);
+ virtual ~Plan_query_range();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ void setTable(Plan_table* table, Plan_table::Index* index);
+ void setInterp(Plan_pred* interp);
+ void setExclusive();
+protected:
+ Plan_table* m_table;
+ Plan_table::Index* m_index;
+ Plan_pred* m_interp;
+ bool m_exclusive;
+};
+
+inline
+Plan_query_range::Plan_query_range(Plan_root* root) :
+ Plan_query(root),
+ m_table(0),
+ m_index(0),
+ m_interp(0),
+ m_exclusive(false)
+{
+}
+
+inline void
+Plan_query_range::setTable(Plan_table* table, Plan_table::Index* index)
+{
+ ctx_assert(table != 0 && index != 0 && index == &table->m_indexList[index->m_pos] && index->m_pos != 0);
+ m_table = table;
+ m_index = index;
+}
+
+inline void
+Plan_query_range::setInterp(Plan_pred* interp)
+{
+ ctx_assert(interp != 0);
+ m_interp = interp;
+}
+
+inline void
+Plan_query_range::setExclusive()
+{
+ m_exclusive = true;
+}
+
+class Exec_query_range : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(unsigned keyCount, unsigned attrCount);
+ virtual ~Code();
+ protected:
+ friend class Plan_query_range;
+ friend class Exec_query_range;
+ const char* m_tableName;
+ const char* m_indexName;
+ unsigned m_keyCount;
+ SqlSpecs m_keySpecs; // key types
+ NdbAttrId* m_keyId;
+ Exec_expr** m_keyMatch; // XXX pointers for now
+ unsigned m_attrCount;
+ SqlSpecs m_sqlSpecs;
+ NdbAttrId* m_attrId;
+ bool m_exclusive;
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_range* node, const SqlSpecs& sqlSpecs);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_range;
+ SqlRow m_sqlRow;
+ NdbConnection* m_con;
+ NdbOperation* m_op;
+ NdbRecAttr** m_recAttr;
+ unsigned m_parallel;
+ bool m_done; // if no match possible due to range
+ };
+ Exec_query_range(Exec_root* root);
+ virtual ~Exec_query_range();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ const Code& getCode() const;
+ Data& getData() const;
+ void setInterp(Exec_pred* interp);
+protected:
+ Exec_pred* m_interp;
+};
+
+inline
+Exec_query_range::Code::Code(unsigned keyCount, unsigned attrCount) :
+ Exec_query::Code(m_sqlSpecs),
+ m_tableName(0),
+ m_indexName(0),
+ m_keyCount(keyCount),
+ m_keySpecs(keyCount),
+ m_keyId(0),
+ m_attrCount(attrCount),
+ m_sqlSpecs(attrCount),
+ m_attrId(0),
+ m_exclusive(false)
+{
+}
+
+inline
+Exec_query_range::Data::Data(Exec_query_range* node, const SqlSpecs& sqlSpecs) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlSpecs),
+ m_con(0),
+ m_op(0),
+ m_recAttr(0),
+ m_parallel(1),
+ m_done(false)
+{
+}
+
+inline
+Exec_query_range::Exec_query_range(Exec_root* root) :
+ Exec_query(root),
+ m_interp(0)
+{
+}
+
+inline const Exec_query_range::Code&
+Exec_query_range::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_range::Data&
+Exec_query_range::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_query_range::setInterp(Exec_pred* interp)
+{
+ ctx_assert(interp != 0);
+ m_interp = interp;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_repeat.cpp b/ndb/src/client/odbc/codegen/Code_query_repeat.cpp
new file mode 100644
index 00000000000..8b295a97916
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_repeat.cpp
@@ -0,0 +1,109 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_query_repeat.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_repeat
+
+Plan_query_repeat::~Plan_query_repeat()
+{
+}
+
+Plan_base*
+Plan_query_repeat::analyze(Ctx& ctx, Ctl& ctl)
+{
+ return this;
+}
+
+Exec_base*
+Plan_query_repeat::codegen(Ctx& ctx, Ctl& ctl)
+{
+ Exec_query_repeat* exec = new Exec_query_repeat(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // SqlSpecs is empty
+ const SqlSpecs sqlSpecs(0);
+ Exec_query_repeat::Code& code = *new Exec_query_repeat::Code(sqlSpecs, m_forever, m_maxcount);
+ exec->setCode(code);
+ return exec;
+}
+
+void
+Plan_query_repeat::print(Ctx& ctx)
+{
+ ctx.print(" [query_repeat");
+ if (! m_forever)
+ ctx.print(" %ld", (long)m_maxcount);
+ ctx.print("]");
+}
+
+// Exec_query_repeat
+
+Exec_query_repeat::Code::~Code()
+{
+}
+
+Exec_query_repeat::Data::~Data()
+{
+}
+
+Exec_query_repeat::~Exec_query_repeat()
+{
+}
+
+void
+Exec_query_repeat::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // SqlRow is empty
+ Data& data = *new Data(this, code.sqlSpecs());
+ setData(data);
+}
+
+void
+Exec_query_repeat::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = getData();
+ data.m_count = 0;
+}
+
+bool
+Exec_query_repeat::fetchImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ // fetch until count is up
+ if (code.m_forever || data.m_count < code.m_maxcount) {
+ data.m_count++;
+ return true;
+ }
+ return false;
+}
+
+void
+Exec_query_repeat::close(Ctx& ctx)
+{
+}
+
+void
+Exec_query_repeat::print(Ctx& ctx)
+{
+ const Code& code = getCode();
+ ctx.print(" [query_repeat");
+ if (! code.m_forever)
+ ctx.print(" %ld", (long)code.m_maxcount);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_repeat.hpp b/ndb/src/client/odbc/codegen/Code_query_repeat.hpp
new file mode 100644
index 00000000000..90d6ef55104
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_repeat.hpp
@@ -0,0 +1,133 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_repeat_hpp
+#define ODBC_CODEGEN_Code_query_repeat_hpp
+
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_expr_row.hpp"
+
+/**
+ * @class Plan_query_repeat
+ * @brief Constant query node in PlanTree
+ */
+class Plan_query_repeat : public Plan_query {
+public:
+ Plan_query_repeat(Plan_root* root);
+ Plan_query_repeat(Plan_root* root, CountType maxcount);
+ virtual ~Plan_query_repeat();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+private:
+ bool m_forever;
+ CountType m_maxcount;
+};
+
+inline
+Plan_query_repeat::Plan_query_repeat(Plan_root* root) :
+ Plan_query(root),
+ m_forever(true),
+ m_maxcount(0)
+{
+}
+
+inline
+Plan_query_repeat::Plan_query_repeat(Plan_root* root, CountType maxcount) :
+ Plan_query(root),
+ m_forever(false),
+ m_maxcount(maxcount)
+{
+}
+
+/**
+ * @class Exec_query_repeat
+ * @brief Constant query node in ExecTree
+ */
+class Exec_query_repeat : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(const SqlSpecs& sqlSpecs, bool forever, CountType maxcount);
+ virtual ~Code();
+ protected:
+ friend class Exec_query_repeat;
+ SqlSpecs m_sqlSpecs;
+ bool m_forever;
+ CountType m_maxcount;
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_repeat* node, const SqlSpecs& sqlSpecs);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_repeat;
+ SqlRow m_sqlRow;
+ CountType m_count;
+ };
+ Exec_query_repeat(Exec_root* root);
+ virtual ~Exec_query_repeat();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_query_repeat::Code::Code(const SqlSpecs& sqlSpecs, bool forever, CountType maxcount) :
+ Exec_query::Code(m_sqlSpecs),
+ m_sqlSpecs(sqlSpecs),
+ m_forever(forever),
+ m_maxcount(maxcount)
+{
+}
+
+inline
+Exec_query_repeat::Data::Data(Exec_query_repeat* node, const SqlSpecs& sqlSpecs) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlSpecs),
+ m_count(0)
+{
+}
+
+inline
+Exec_query_repeat::Exec_query_repeat(Exec_root* root) :
+ Exec_query(root)
+{
+}
+
+// children
+
+inline const Exec_query_repeat::Code&
+Exec_query_repeat::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_repeat::Data&
+Exec_query_repeat::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_scan.cpp b/ndb/src/client/odbc/codegen/Code_query_scan.cpp
new file mode 100644
index 00000000000..1c0f58980e5
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_scan.cpp
@@ -0,0 +1,177 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_query_scan.hpp"
+#include "Code_column.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_scan
+
+Plan_query_scan::~Plan_query_scan()
+{
+}
+
+Plan_base*
+Plan_query_scan::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ if (m_interp != 0) {
+ m_interp = static_cast<Plan_pred*>(m_interp->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_interp != 0);
+ }
+ return this;
+}
+
+Exec_base*
+Plan_query_scan::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // set up
+ ctx_assert(m_table != 0);
+ const BaseString& tableName = m_table->getName();
+ const DictTable& dictTable = m_table->dictTable();
+ const ColumnVector& columns = m_table->exprColumns();
+ ctx_assert(columns.size() > 0);
+ const unsigned attrCount = columns.size() - 1;
+ // create the code
+ Exec_query_scan::Code& code = *new Exec_query_scan::Code(attrCount);
+ code.m_tableName = strcpy(new char[tableName.length() + 1], tableName.c_str());
+ code.m_exclusive = m_exclusive;
+ // queried attributes
+ code.m_attrId = new NdbAttrId[1 + attrCount];
+ code.m_attrId[0] = (NdbAttrId)-1;
+ for (unsigned i = 1; i <= attrCount; i++) {
+ Plan_column* column = columns[i];
+ ctx_assert(column != 0);
+ const DictColumn& dictColumn = column->dictColumn();
+ const SqlType& sqlType = dictColumn.sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_sqlSpecs.setEntry(i, sqlSpec);
+ code.m_attrId[i] = dictColumn.getAttrId();
+ }
+ // create the exec
+ Exec_query_scan* exec = new Exec_query_scan(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ // interpreter
+ Exec_pred* execInterp = 0;
+ ctl.m_execQuery = exec;
+ ctl.m_topTable = m_table;
+ if (m_interp != 0) {
+ execInterp = static_cast<Exec_pred*>(m_interp->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execInterp != 0);
+ }
+ ctl.m_topTable = 0;
+ if (m_interp != 0)
+ exec->setInterp(execInterp);
+ return exec;
+}
+
+void
+Plan_query_scan::print(Ctx& ctx)
+{
+ ctx.print(" [query_scan");
+ Plan_base* a[] = { m_table, m_interp };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
+
+// Exec_query_scan
+
+Exec_query_scan::Code::~Code()
+{
+ delete[] m_tableName;
+ delete[] m_attrId;
+}
+
+Exec_query_scan::Data::~Data()
+{
+ delete[] m_recAttr;
+}
+
+Exec_query_scan::~Exec_query_scan()
+{
+}
+
+void
+Exec_query_scan::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // create data
+ Data& data = *new Data(this, code.sqlSpecs());
+ // needed for isNULL
+ data.m_recAttr = new NdbRecAttr* [1 + code.m_attrCount];
+ for (unsigned i = 0; i <= code.m_attrCount; i++) {
+ data.m_recAttr[i] = 0;
+ }
+ data.m_parallel = code.m_exclusive ? 1 : 240; // best supported
+ setData(data);
+ // interpreter
+ ctl.m_query = this;
+ if (m_interp != 0) {
+ //m_interp->alloc(ctx, ctl); XXX
+ if (! ctx.ok())
+ return;
+ }
+}
+
+void
+Exec_query_scan::close(Ctx& ctx)
+{
+ Data& data = getData();
+ if (data.m_con != 0) {
+ Ndb* const ndb = ndbObject();
+ int ret = data.m_con->stopScan();
+ if (ret == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "stopScan");
+ }
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ ctx_log2(("scan closed at statement close"));
+ }
+ if (m_interp != 0)
+ m_interp->close(ctx);
+}
+
+void
+Exec_query_scan::print(Ctx& ctx)
+{
+ ctx.print(" [query_scan");
+ if (m_code != 0) {
+ const Code& code = getCode();
+ ctx.print(" attrId=");
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_attrId[i]);
+ }
+ ctx.print(" table=%s", code.m_tableName);
+ }
+ if (m_interp != 0)
+ m_interp->print(ctx);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_scan.hpp b/ndb/src/client/odbc/codegen/Code_query_scan.hpp
new file mode 100644
index 00000000000..d6d1630ddf8
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_scan.hpp
@@ -0,0 +1,174 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_scan_hpp
+#define ODBC_CODEGEN_Code_query_scan_hpp
+
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_table.hpp"
+#include "Code_pred.hpp"
+
+class Ctx;
+class StmtArea;
+class NdbConnection;
+class NdbOperation;
+class NdbRecAttr;
+
+/*
+ * Table scan.
+ */
+
+class Plan_query_scan : public Plan_query {
+public:
+ Plan_query_scan(Plan_root* root);
+ virtual ~Plan_query_scan();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ void setTable(Plan_table* table);
+ void setInterp(Plan_pred* interp);
+ void setExclusive();
+protected:
+ Plan_table* m_table;
+ Plan_pred* m_interp;
+ bool m_exclusive; // exclusive
+};
+
+inline
+Plan_query_scan::Plan_query_scan(Plan_root* root) :
+ Plan_query(root),
+ m_table(0),
+ m_interp(0),
+ m_exclusive(false)
+{
+}
+
+inline void
+Plan_query_scan::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+inline void
+Plan_query_scan::setInterp(Plan_pred* interp)
+{
+ ctx_assert(interp != 0);
+ m_interp = interp;
+}
+
+inline void
+Plan_query_scan::setExclusive()
+{
+ m_exclusive = true;
+}
+
+class Exec_query_scan : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(unsigned attrCount);
+ virtual ~Code();
+ protected:
+ friend class Plan_query_scan;
+ friend class Exec_query_scan;
+ char* m_tableName;
+ unsigned m_attrCount;
+ SqlSpecs m_sqlSpecs;
+ NdbAttrId* m_attrId;
+ bool m_exclusive;
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_scan* node, const SqlSpecs& sqlSpecs);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_scan;
+ SqlRow m_sqlRow;
+ NdbConnection* m_con;
+ NdbOperation* m_op;
+ NdbRecAttr** m_recAttr;
+ unsigned m_parallel; // parallelism could be runtime option
+ };
+ Exec_query_scan(Exec_root* root);
+ virtual ~Exec_query_scan();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setInterp(Exec_pred* interp);
+protected:
+ Exec_pred* m_interp;
+};
+
+inline
+Exec_query_scan::Code::Code(unsigned attrCount) :
+ Exec_query::Code(m_sqlSpecs),
+ m_tableName(0),
+ m_attrCount(attrCount),
+ m_sqlSpecs(attrCount),
+ m_attrId(0),
+ m_exclusive(false)
+{
+}
+
+inline
+Exec_query_scan::Data::Data(Exec_query_scan* node, const SqlSpecs& sqlSpecs) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlSpecs),
+ m_con(0),
+ m_op(0),
+ m_recAttr(0),
+ m_parallel(1)
+{
+}
+
+inline
+Exec_query_scan::Exec_query_scan(Exec_root* root) :
+ Exec_query(root),
+ m_interp(0)
+{
+}
+
+// children
+
+inline const Exec_query_scan::Code&
+Exec_query_scan::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_scan::Data&
+Exec_query_scan::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_query_scan::setInterp(Exec_pred* interp)
+{
+ ctx_assert(interp != 0);
+ m_interp = interp;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_sort.cpp b/ndb/src/client/odbc/codegen/Code_query_sort.cpp
new file mode 100644
index 00000000000..4ea6db8c4e2
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_sort.cpp
@@ -0,0 +1,239 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <algorithm>
+#include "Code_query_sort.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_sort
+
+Plan_query_sort::~Plan_query_sort()
+{
+}
+
+Plan_expr_row*
+Plan_query_sort::getRow()
+{
+ ctx_assert(m_query != 0);
+ return m_query->getRow();
+}
+
+Plan_base*
+Plan_query_sort::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_sortRow != 0);
+ m_sortRow->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_query_sort::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // create code for the subquery
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // create code for the row based on query code
+ ctx_assert(m_sortRow != 0);
+ ctl.m_execQuery = execQuery->getRawQuery();
+ Exec_expr_row* execRow = static_cast<Exec_expr_row*>(m_sortRow->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execRow != 0);
+ Exec_query_sort* exec = new Exec_query_sort(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ // re-use SqlSpecs from subquery
+ const Exec_query::Code& codeQuery = execQuery->getCode();
+ const SqlSpecs& sqlSpecs = codeQuery.sqlSpecs();
+ // make asc
+ unsigned size = m_sortRow->getSize();
+ bool* asc = new bool[1 + size];
+ for (unsigned i = 1; i <= size; i++) {
+ asc[i] = m_sortRow->m_ascList[i];
+ }
+ Exec_query_sort::Code& code = *new Exec_query_sort::Code(sqlSpecs, asc);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ exec->setRow(execRow);
+ return exec;
+}
+
+void
+Plan_query_sort::print(Ctx& ctx)
+{
+ ctx.print(" [query_sort");
+ Plan_base* a[] = { m_query, m_sortRow };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
+
+// Exec_query_sort
+
+Exec_query_sort::Code::~Code()
+{
+}
+
+Exec_query_sort::Data::~Data()
+{
+ for (unsigned i = 0; i < m_sortList.size(); i++) {
+ SortItem& sortItem = m_sortList[i];
+ delete sortItem.m_dataRow;
+ delete sortItem.m_sortRow;
+ }
+}
+
+Exec_query_sort::~Exec_query_sort()
+{
+}
+
+void
+Exec_query_sort::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // allocate sort row based on subquery data
+ ctx_assert(m_sortRow != 0);
+ ctl.m_query = m_query->getRawQuery();
+ m_sortRow->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ Data& data = *new Data(this, getCode().sqlSpecs());
+ setData(data);
+}
+
+void
+Exec_query_sort::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_query != 0 && m_sortRow != 0);
+ ctl.m_sortRow = m_sortRow;
+ m_query->execute(ctx, ctl);
+}
+
+bool
+SortLess::operator()(SortItem s1, SortItem s2) const
+{
+ const Exec_query_sort::Code& code = m_node->getCode();
+ const SqlRow& r1 = *s1.m_sortRow;
+ const SqlRow& r2 = *s2.m_sortRow;
+ for (unsigned i = 1; i <= r1.count(); i++) {
+ const SqlField& f1 = r1.getEntry(i);
+ const SqlField& f2 = r2.getEntry(i);
+ // nulls last is default in oracle
+ bool f1null = f1.sqlNull();
+ bool f2null = f2.sqlNull();
+ if (f1null && f2null)
+ continue;
+ if (! f1null && f2null)
+ return code.getAsc(i) ? true : false;
+ if (f1null && ! f2null)
+ return code.getAsc(i) ? false : true;
+ if (f1.less(f2))
+ return code.getAsc(i) ? true : false;
+ if (f2.less(f1))
+ return code.getAsc(i) ? false : true;
+ }
+ return false;
+}
+
+bool
+Exec_query_sort::fetchImpl(Ctx& ctx, Ctl& ctl)
+{
+ Data& data = getData();
+ ctx_assert(m_query != 0 && m_sortRow != 0);
+ ctl.m_sortRow = m_sortRow;
+ if (! data.m_sorted) {
+ // read and cache all rows
+ data.m_count = 0;
+ while (m_query->fetch(ctx, ctl)) {
+ const SqlRow* dataRow = m_query->getData().sqlRow().copy();
+ const SqlRow* sortRow = 0;
+ if (ctl.m_groupIndex == 0) {
+ // evaluate sort row
+ m_sortRow->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return false;
+ sortRow = m_sortRow->getData().sqlRow().copy();
+ } else {
+ // evaluate done by group-by
+ SqlRow tmpSortRow(m_sortRow->getCode().sqlSpecs());
+ for (unsigned i = 1; i <= tmpSortRow.count(); i++) {
+ tmpSortRow.setEntry(i, m_sortRow->getExpr(i)->getData().groupField(ctl.m_groupIndex));
+ }
+ sortRow = tmpSortRow.copy();
+ }
+ SortItem sortItem(dataRow, sortRow);
+ data.m_sortList.push_back(sortItem);
+ data.m_count++;
+ }
+ data.m_index = 0;
+ if (! ctx.ok())
+ return false;
+ // sort the rows XXX use iterated stable_sort
+ SortLess sortLess(this);
+ std::sort(data.m_sortList.begin(), data.m_sortList.end(), sortLess);
+ data.m_sorted = true;
+ }
+ if (data.m_index < data.m_count) {
+ // make our SqlRow reference to current row
+ const SqlRow& currRow = *data.m_sortList[data.m_index].m_dataRow;
+ for (unsigned i = 1; i <= data.m_sqlRow.count(); i++) {
+ const SqlField& currField = currRow.getEntry(i);
+ SqlSpec sqlSpec(currField.sqlSpec(), SqlSpec::Reference);
+ SqlField sqlField(sqlSpec, &currField);
+ data.m_sqlRow.setEntry(i, sqlField);
+ }
+ data.m_index++;
+ return true;
+ }
+ return false;
+}
+
+void
+Exec_query_sort::close(Ctx& ctx)
+{
+ Data& data = getData();
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+ data.m_sorted = false;
+ for (unsigned i = 0; i < data.m_sortList.size(); i++) {
+ SortItem& sortItem = data.m_sortList[i];
+ delete sortItem.m_dataRow;
+ delete sortItem.m_sortRow;
+ }
+ data.m_sortList.clear();
+ data.m_count = 0;
+ data.m_index = 0;
+}
+
+void
+Exec_query_sort::print(Ctx& ctx)
+{
+ ctx.print(" [query_sort");
+ Exec_base* a[] = { m_query, m_sortRow };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_sort.hpp b/ndb/src/client/odbc/codegen/Code_query_sort.hpp
new file mode 100644
index 00000000000..d1aa03d9aef
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_sort.hpp
@@ -0,0 +1,208 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_sort_hpp
+#define ODBC_CODEGEN_Code_query_sort_hpp
+
+#include <functional>
+#include <common/common.hpp>
+#include "Code_query.hpp"
+#include "Code_expr_row.hpp"
+
+/**
+ * @class Plan_query_sort
+ * @brief Project node in PlanTree
+ */
+class Plan_query_sort : public Plan_query {
+public:
+ Plan_query_sort(Plan_root* root);
+ virtual ~Plan_query_sort();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setQuery(Plan_query* query);
+ void setRow(Plan_expr_row* sortRow);
+protected:
+ Plan_expr_row* getRow();
+ Plan_query* m_query;
+ Plan_expr_row* m_sortRow;
+};
+
+inline
+Plan_query_sort::Plan_query_sort(Plan_root* root) :
+ Plan_query(root),
+ m_query(0),
+ m_sortRow(0)
+{
+}
+
+// children
+
+inline void
+Plan_query_sort::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+inline void
+Plan_query_sort::setRow(Plan_expr_row* sortRow)
+{
+ ctx_assert(sortRow != 0);
+ m_sortRow = sortRow;
+}
+
+/**
+ * Item to sort includes data row and sort row.
+ */
+struct SortItem {
+ SortItem(const SqlRow* dataRow, const SqlRow* sortRow);
+ const SqlRow* m_dataRow; // copy of fetched row from subquery
+ const SqlRow* m_sortRow; // copy of values to sort on
+};
+
+typedef std::vector<SortItem> SortList;
+
+class Exec_query_sort;
+
+struct SortLess : std::binary_function<SortItem, SortItem, bool> {
+ SortLess(const Exec_query_sort* node);
+ const Exec_query_sort* m_node;
+ bool operator()(SortItem s1, SortItem s2) const;
+};
+
+inline
+SortItem::SortItem(const SqlRow* dataRow, const SqlRow* sortRow) :
+ m_dataRow(dataRow),
+ m_sortRow(sortRow)
+{
+}
+
+inline
+SortLess::SortLess(const Exec_query_sort* node) :
+ m_node(node)
+{
+}
+
+/**
+ * @class Exec_query_sort
+ * @brief Project node in ExecTree
+ */
+class Exec_query_sort : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(const SqlSpecs& sqlSpecs, bool* asc);
+ virtual ~Code();
+ bool getAsc(unsigned i) const;
+ protected:
+ friend class Exec_query_sort;
+ const bool* const m_asc;
+ // sets reference to Sqlspecs from subquery
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_sort* node, const SqlSpecs& sqlSpecs);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_sort;
+ SqlRow m_sqlRow; // current row
+ bool m_sorted; // fetch and sort done
+ SortList m_sortList;
+ unsigned m_count; // number of rows
+ unsigned m_index; // current fetch index
+ };
+ Exec_query_sort(Exec_root* root);
+ virtual ~Exec_query_sort();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+ void setRow(Exec_expr_row* sortRow);
+protected:
+ friend class Exec_query;
+ Exec_query* m_query;
+ Exec_expr_row* m_sortRow;
+};
+
+inline
+Exec_query_sort::Code::Code(const SqlSpecs& sqlSpecs, bool* asc) :
+ Exec_query::Code(sqlSpecs),
+ m_asc(asc)
+{
+}
+
+inline bool
+Exec_query_sort::Code::getAsc(unsigned i) const
+{
+ return m_asc[i];
+}
+
+inline
+Exec_query_sort::Data::Data(Exec_query_sort* node, const SqlSpecs& sqlSpecs) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlSpecs),
+ m_sorted(false),
+ m_count(0),
+ m_index(0)
+{
+}
+
+inline
+Exec_query_sort::Exec_query_sort(Exec_root* root) :
+ Exec_query(root),
+ m_query(0),
+ m_sortRow(0)
+{
+}
+
+// children
+
+inline const Exec_query_sort::Code&
+Exec_query_sort::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_sort::Data&
+Exec_query_sort::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_query_sort::setQuery(Exec_query* query)
+{
+ ctx_assert(m_query == 0 && query != 0);
+ m_query = query;
+}
+
+inline void
+Exec_query_sort::setRow(Exec_expr_row* sortRow)
+{
+ ctx_assert(m_sortRow == 0 && sortRow != 0);
+ m_sortRow = sortRow;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_query_sys.cpp b/ndb/src/client/odbc/codegen/Code_query_sys.cpp
new file mode 100644
index 00000000000..affe3dc1264
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_sys.cpp
@@ -0,0 +1,130 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_query_sys.hpp"
+#include "Code_column.hpp"
+#include "Code_root.hpp"
+
+// Plan_query_sys
+
+Plan_query_sys::~Plan_query_sys()
+{
+}
+
+Plan_base*
+Plan_query_sys::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+Exec_base*
+Plan_query_sys::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // set up
+ ctx_assert(m_table != 0);
+ const DictTable& dictTable = m_table->dictTable();
+ const ColumnVector& columns = m_table->exprColumns();
+ ctx_assert(columns.size() > 0);
+ const unsigned attrCount = columns.size() - 1;
+ // create the code
+ Exec_query_sys::Code& code = *new Exec_query_sys::Code(attrCount);
+ code.m_sysId = dictTable.sysId();
+ // queried attributes
+ code.m_attrId = new NdbAttrId[1 + attrCount];
+ code.m_attrId[0] = (NdbAttrId)-1;
+ for (unsigned i = 1; i <= attrCount; i++) {
+ Plan_column* column = columns[i];
+ ctx_assert(column != 0);
+ const DictColumn& dictColumn = column->dictColumn();
+ const SqlType& sqlType = dictColumn.sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_sqlSpecs.setEntry(i, sqlSpec);
+ code.m_attrId[i] = dictColumn.getAttrId();
+ }
+ // create the exec
+ Exec_query_sys* exec = new Exec_query_sys(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ return exec;
+}
+
+void
+Plan_query_sys::print(Ctx& ctx)
+{
+ ctx.print(" [query_sys");
+ Plan_base* a[] = { m_table };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
+
+// Exec_query_sys
+
+Exec_query_sys::Code::~Code()
+{
+ delete[] m_attrId;
+}
+
+Exec_query_sys::Data::~Data()
+{
+}
+
+Exec_query_sys::~Exec_query_sys()
+{
+}
+
+void
+Exec_query_sys::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // create data
+ Data& data = *new Data(this, code.sqlSpecs());
+ setData(data);
+}
+
+void
+Exec_query_sys::close(Ctx& ctx)
+{
+ Data& data = getData();
+ data.m_rowPos = 0;
+ data.m_tablePos = 0;
+ data.m_attrPos = 0;
+ data.m_keyPos = 0;
+}
+
+void
+Exec_query_sys::print(Ctx& ctx)
+{
+ ctx.print(" [query_sys");
+ if (m_code != 0) {
+ const Code& code = getCode();
+ ctx.print(" attrId=");
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_attrId[i]);
+ }
+ ctx.print(" sysId=%u", (unsigned)code.m_sysId);
+ }
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_query_sys.hpp b/ndb/src/client/odbc/codegen/Code_query_sys.hpp
new file mode 100644
index 00000000000..8eb069d0413
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_query_sys.hpp
@@ -0,0 +1,148 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_query_sys_hpp
+#define ODBC_CODEGEN_Code_query_sys_hpp
+
+#include <common/common.hpp>
+#include <dictionary/DictSys.hpp>
+#include "Code_query.hpp"
+#include "Code_table.hpp"
+
+class Ctx;
+class StmtArea;
+class NdbConnection;
+class NdbOperation;
+class NdbRecAttr;
+
+/**
+ * @class Plan_query_sys
+ * @brief Full select (no where clause)
+ */
+class Plan_query_sys : public Plan_query {
+public:
+ Plan_query_sys(Plan_root* root);
+ virtual ~Plan_query_sys();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setTable(Plan_table* table);
+protected:
+ Plan_table* m_table;
+};
+
+inline
+Plan_query_sys::Plan_query_sys(Plan_root* root) :
+ Plan_query(root),
+ m_table(0)
+{
+}
+
+// children
+
+inline void
+Plan_query_sys::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+/**
+ * @class Exec_query_sys
+ * @brief Full select (no where clause)
+ */
+class Exec_query_sys : public Exec_query {
+public:
+ class Code : public Exec_query::Code {
+ public:
+ Code(unsigned attrCount);
+ virtual ~Code();
+ protected:
+ friend class Plan_query_sys;
+ friend class Exec_query_sys;
+ DictSys::Id m_sysId;
+ unsigned m_attrCount;
+ SqlSpecs m_sqlSpecs;
+ NdbAttrId* m_attrId;
+ };
+ class Data : public Exec_query::Data {
+ public:
+ Data(Exec_query_sys* node, const SqlSpecs& sqlSpecs);
+ virtual ~Data();
+ protected:
+ friend class Exec_query_sys;
+ SqlRow m_sqlRow;
+ // for typeinfo
+ unsigned m_rowPos;
+ // for tables and columns
+ NdbDictionary::Dictionary::List m_objectList;
+ unsigned m_tablePos;
+ unsigned m_attrPos;
+ unsigned m_keyPos;
+ };
+ Exec_query_sys(Exec_root* root);
+ virtual ~Exec_query_sys();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ bool fetchImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+};
+
+inline
+Exec_query_sys::Code::Code(unsigned attrCount) :
+ Exec_query::Code(m_sqlSpecs),
+ m_sysId(DictSys::Undef),
+ m_attrCount(attrCount),
+ m_sqlSpecs(attrCount),
+ m_attrId(0)
+{
+}
+
+inline
+Exec_query_sys::Data::Data(Exec_query_sys* node, const SqlSpecs& sqlSpecs) :
+ Exec_query::Data(node, m_sqlRow),
+ m_sqlRow(sqlSpecs)
+{
+}
+
+inline
+Exec_query_sys::Exec_query_sys(Exec_root* root) :
+ Exec_query(root)
+{
+}
+
+// children
+
+inline const Exec_query_sys::Code&
+Exec_query_sys::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_query_sys::Data&
+Exec_query_sys::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_root.cpp b/ndb/src/client/odbc/codegen/Code_root.cpp
new file mode 100644
index 00000000000..4f45bdffdaf
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_root.cpp
@@ -0,0 +1,307 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include "Code_root.hpp"
+#include "Code_stmt.hpp"
+#include "Code_query.hpp"
+#include "Code_expr_param.hpp"
+#include "Code_root.hpp"
+
+// Plan_root
+
+Plan_root::~Plan_root()
+{
+}
+
+Plan_base*
+Plan_root::analyze(Ctx& ctx, Ctl& ctl)
+{
+ // analyze statement
+ ctx_assert(m_stmt != 0);
+ m_stmt = static_cast<Plan_stmt*>(m_stmt->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_stmt != 0);
+ // analyze parameters
+ ctx_assert(m_paramList.size() > 0);
+ const unsigned paramCount = m_paramList.size() - 1;
+ DescArea& ipd = descArea(Desc_usage_IPD);
+ ipd.setCount(ctx, paramCount);
+ for (unsigned i = 1; i <= paramCount; i++) {
+ Plan_expr_param* param = m_paramList[i];
+ ctx_assert(param != 0);
+ // analyze the parameter
+ param->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ }
+ // must return self
+ return this;
+}
+
+void
+Plan_root::describe(Ctx& ctx)
+{
+ // describe statement
+ ctx_assert(m_stmt != 0);
+ m_stmt->describe(ctx);
+ // describe parameters
+ ctx_assert(m_paramList.size() > 0);
+ const unsigned paramCount = m_paramList.size() - 1;
+ DescArea& ipd = descArea(Desc_usage_IPD);
+ ipd.setCount(ctx, paramCount);
+ unsigned unbound = 0;
+ for (unsigned i = 1; i <= paramCount; i++) {
+ Plan_expr_param* param = m_paramList[i];
+ ctx_assert(param != 0);
+ // describe the parameter
+ param->describe(ctx);
+ // check if SQL type is bound
+ ctx_assert(param->sqlType().type() != SqlType::Undef);
+ if (param->sqlType().type() == SqlType::Unbound)
+ unbound++;
+ }
+ if (unbound > 0)
+ ctx_log2(("%u out of %u params have unbound SQL type", unbound, paramCount));
+ m_stmtArea.m_unbound = unbound;
+}
+
+Exec_base*
+Plan_root::codegen(Ctx& ctx, Ctl& ctl)
+{
+ Exec_root* execRoot = new Exec_root(m_stmtArea);
+ Exec_root::Code& code = *new Exec_root::Code;
+ execRoot->setCode(code);
+ // set root in helper struct
+ ctl.m_execRoot = execRoot;
+ // generate code for the statement
+ ctx_assert(m_stmt != 0);
+ Exec_stmt* execStmt = static_cast<Exec_stmt*>(m_stmt->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ execRoot->setStmt(execStmt);
+ // create parameters list
+ execRoot->m_paramList.resize(m_paramList.size());
+ for (unsigned i = 1; i < m_paramList.size(); i++) {
+ Plan_expr_param* param = m_paramList[i];
+ ctx_assert(param != 0);
+ Exec_expr_param* execParam = static_cast<Exec_expr_param*>(param->codegen(ctx, ctl));
+ ctx_assert(execParam != 0);
+ execRoot->m_paramList[i] = execParam;
+ }
+ return execRoot;
+}
+
+void
+Plan_root::print(Ctx& ctx)
+{
+ ctx.print("[root");
+ Plan_base* a[] = { m_stmt };
+ printList(ctx, a, 1);
+ ctx.print("]\n");
+}
+
+void
+Plan_root::saveNode(Plan_base* node)
+{
+ ctx_assert(node != 0);
+ m_nodeList.push_back(node);
+}
+
+void
+Plan_root::freeNodeList()
+{
+ for (NodeList::iterator i = m_nodeList.begin(); i != m_nodeList.end(); i++) {
+ Plan_base* node = *i;
+ *i = 0;
+ delete node;
+ }
+ m_nodeList.clear();
+}
+
+// Exec_root
+
+Exec_root::Code::~Code()
+{
+}
+
+Exec_root::Data::~Data()
+{
+}
+
+Exec_root::~Exec_root()
+{
+}
+
+StmtArea&
+Exec_root::stmtArea() const
+{
+ return m_stmtArea;
+}
+
+void
+Exec_root::alloc(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_stmt != 0);
+ m_stmt->alloc(ctx, ctl);
+}
+
+void
+Exec_root::bind(Ctx& ctx)
+{
+ // bind output cols
+ ctx_assert(m_stmt != 0);
+ m_stmt->bind(ctx);
+ // bind input params
+ for (unsigned i = 1; i < m_paramList.size(); i++) {
+ Exec_expr_param* param = m_paramList[i];
+ ctx_assert(param != 0);
+ param->bind(ctx);
+ if (! ctx.ok())
+ return;
+ }
+}
+
+void
+Exec_root::execute(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_stmt != 0);
+ // check if data is needed
+ for (unsigned i = 1; i < m_paramList.size(); i++) {
+ Exec_expr_param* param = m_paramList[i];
+ ctx_assert(param != 0);
+ Exec_expr_param::Data& paramData = param->getData();
+ if (paramData.m_atExec && paramData.m_extPos == -1) {
+ ctx.setCode(SQL_NEED_DATA);
+ return;
+ }
+ }
+ m_stmt->execute(ctx, ctl);
+}
+
+void
+Exec_root::fetch(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(m_stmt != 0);
+ Exec_query* query = static_cast<Exec_query*>(m_stmt);
+ ctx_assert(query != 0);
+ query->fetch(ctx, ctl);
+}
+
+void
+Exec_root::close(Ctx& ctx)
+{
+ ctx_assert(m_stmt != 0);
+ m_stmt->close(ctx);
+ for (unsigned i = 1; i < m_paramList.size(); i++) {
+ Exec_expr_param* param = m_paramList[i];
+ ctx_assert(param != 0);
+ param->close(ctx);
+ }
+}
+
+void
+Exec_root::print(Ctx& ctx)
+{
+ ctx.print("[root");
+ Exec_base* a[] = { m_stmt };
+ printList(ctx, a, sizeof(a)/sizeof(a[0]));
+ ctx.print("]\n");
+}
+
+void
+Exec_root::saveNode(Exec_base* node)
+{
+ ctx_assert(node != 0);
+ m_nodeList.push_back(node);
+}
+
+void
+Exec_root::freeNodeList()
+{
+ for (NodeList::iterator i = m_nodeList.begin(); i != m_nodeList.end(); i++) {
+ Exec_base* node = *i;
+ *i = 0;
+ delete node;
+ }
+ m_nodeList.clear();
+}
+
+// odbc support
+
+void
+Exec_root::sqlGetData(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind)
+{
+ ctx_assert(m_stmt != 0);
+ Exec_query* query = static_cast<Exec_query*>(m_stmt);
+ ctx_assert(query != 0);
+ query->sqlGetData(ctx, columnNumber, targetType, targetValue, bufferLength, strlen_or_Ind);
+}
+
+void
+Exec_root::sqlParamData(Ctx& ctx, SQLPOINTER* value)
+{
+ ctx_assert(m_paramList.size() > 0);
+ unsigned count = m_paramList.size() - 1;
+ for (unsigned i = 1; i <= count; i++) {
+ Exec_expr_param* param = m_paramList[i];
+ ctx_assert(param != 0);
+ Exec_expr_param::Data& paramData = param->getData();
+ if (! paramData.m_atExec || paramData.m_extPos >= 0)
+ continue;
+ ctx_assert(paramData.m_extField != 0);
+ ExtField& extField = *paramData.m_extField;
+ if (value != 0)
+ *value = extField.m_dataPtr;
+ m_paramData = i;
+ ctx.setCode(SQL_NEED_DATA);
+ return;
+ }
+}
+
+void
+Exec_root::sqlPutData(Ctx& ctx, SQLPOINTER data, SQLINTEGER strlen_or_Ind)
+{
+ ctx_assert(m_paramList.size() > 0);
+ unsigned count = m_paramList.size() - 1;
+ unsigned i = m_paramData;
+ if (i == 0) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "missing call to SQLParamData");
+ return;
+ }
+ if (i > count) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "parameter %u out of range 1 to %u", i, count);
+ return;
+ }
+ Exec_expr_param* param = m_paramList[i];
+ ctx_assert(param != 0);
+ Exec_expr_param::Data& paramData = param->getData();
+ if (! paramData.m_atExec) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "parameter %u not marked for data-at-exec", i);
+ return;
+ }
+ ctx_assert(paramData.m_extField != 0);
+ ExtField extField(paramData.m_extField->extSpec(), data, 0, &strlen_or_Ind, i);
+ if (paramData.m_extPos == -1)
+ paramData.m_extPos = 0;
+ extField.setPos(paramData.m_extPos);
+ // copy in and update position
+ SqlField& sqlField = paramData.m_sqlField;
+ sqlField.copyin(ctx, extField);
+ paramData.m_extPos = extField.getPos();
+ ctx_log4(("parameter %u data received", i));
+}
diff --git a/ndb/src/client/odbc/codegen/Code_root.hpp b/ndb/src/client/odbc/codegen/Code_root.hpp
new file mode 100644
index 00000000000..4f0f96725e3
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_root.hpp
@@ -0,0 +1,162 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_root_hpp
+#define ODBC_CODEGEN_Code_root_hpp
+
+#include <list>
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_stmt.hpp"
+
+class SqlField;
+class ExtField;
+
+/**
+ * @class Plan_root
+ * @brief Root node above top level statement node
+ */
+class Plan_root : public Plan_base {
+public:
+ Plan_root(StmtArea& stmtArea);
+ virtual ~Plan_root();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setStmt(Plan_stmt* stmt);
+ // save and free nodes
+ void saveNode(Plan_base* node);
+ void freeNodeList();
+private:
+ friend class CodeGen;
+ friend class Plan_base;
+ friend class Plan_expr_param;
+ StmtArea& m_stmtArea;
+ Plan_stmt* m_stmt;
+ ParamVector m_paramList;
+ typedef std::list<Plan_base*> NodeList;
+ NodeList m_nodeList;
+};
+
+inline
+Plan_root::Plan_root(StmtArea& stmtArea) :
+ Plan_base(this),
+ m_stmtArea(stmtArea),
+ m_stmt(0)
+{
+}
+
+inline void
+Plan_root::setStmt(Plan_stmt* stmt)
+{
+ ctx_assert(stmt != 0);
+ m_stmt = stmt;
+}
+
+/**
+ * @class Exec_root
+ * @brief Root node above top level statement node
+ */
+class Exec_root : public Exec_base {
+public:
+ class Code : public Exec_base::Code {
+ public:
+ Code();
+ virtual ~Code();
+ };
+ class Data : public Exec_base::Data {
+ public:
+ Data();
+ virtual ~Data();
+ };
+ Exec_root(StmtArea& stmtArea);
+ virtual ~Exec_root();
+ StmtArea& stmtArea() const;
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void bind(Ctx& ctx);
+ void execute(Ctx& ctx, Ctl& ctl);
+ void fetch(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setStmt(Exec_stmt* stmt);
+ // save and free nodes
+ void saveNode(Exec_base* node);
+ void freeNodeList();
+ // odbc support
+ void sqlGetData(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind);
+ void sqlParamData(Ctx& ctx, SQLPOINTER* value);
+ void sqlPutData(Ctx& ctx, SQLPOINTER data, SQLINTEGER strlen_or_Ind);
+private:
+ friend class Plan_root;
+ friend class Exec_base;
+ friend class CodeGen;
+ StmtArea& m_stmtArea;
+ Exec_stmt* m_stmt;
+ ParamVector m_paramList;
+ unsigned m_paramData; // position of SQLParamData
+ typedef std::list<Exec_base*> NodeList;
+ NodeList m_nodeList;
+};
+
+inline
+Exec_root::Code::Code()
+{
+}
+
+inline
+Exec_root::Data::Data()
+{
+}
+
+inline
+Exec_root::Exec_root(StmtArea& stmtArea) :
+ Exec_base(this),
+ m_stmtArea(stmtArea),
+ m_stmt(0),
+ m_paramData(0)
+{
+}
+
+// children
+
+inline const Exec_root::Code&
+Exec_root::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_root::Data&
+Exec_root::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_root::setStmt(Exec_stmt* stmt)
+{
+ ctx_assert(stmt != 0);
+ m_stmt = stmt;
+ m_stmt->m_topLevel = true;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_select.cpp b/ndb/src/client/odbc/codegen/Code_select.cpp
new file mode 100644
index 00000000000..611b491968d
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_select.cpp
@@ -0,0 +1,406 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <algorithm>
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include "Code_select.hpp"
+#include "Code_query_lookup.hpp"
+#include "Code_query_index.hpp"
+#include "Code_query_scan.hpp"
+#include "Code_query_range.hpp"
+#include "Code_query_sys.hpp"
+#include "Code_query_project.hpp"
+#include "Code_query_filter.hpp"
+#include "Code_query_join.hpp"
+#include "Code_query_count.hpp"
+#include "Code_query_sort.hpp"
+#include "Code_query_group.hpp"
+#include "Code_query_distinct.hpp"
+#include "Code_expr_column.hpp"
+#include "Code_expr_const.hpp"
+#include "Code_pred_op.hpp"
+#include "Code_root.hpp"
+
+Plan_select::~Plan_select()
+{
+}
+
+Plan_base*
+Plan_select::analyze(Ctx& ctx, Ctl& ctl)
+{
+ stmtArea().stmtInfo().setName(Stmt_name_select);
+ // analyze tables
+ ctx_assert(m_tableList != 0);
+ for (unsigned i = 1; i <= m_tableList->countTable(); i++) {
+ Plan_table* table = m_tableList->getTable(i);
+ table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ }
+ ctx_assert(m_exprRow != 0);
+ if (m_exprRow->getAsterisk()) {
+ // expand unqualified asterisk to table-qualified columns
+ setRow(new Plan_expr_row(m_root));
+ m_root->saveNode(m_exprRow);
+ for (unsigned i = 1; i <= m_tableList->countTable(); i++) {
+ const Plan_table* table = m_tableList->getTable(i);
+ const DictTable& dictTable = table->dictTable();
+ for (unsigned i = 1; i <= dictTable.getSize(); i++) {
+ DictColumn* dictColumn = dictTable.getColumn(i);
+ Plan_expr_column* column = new Plan_expr_column(m_root, dictColumn->getName());
+ m_root->saveNode(column);
+ column->setCname(table->getCname());
+ m_exprRow->addExpr(column);
+ }
+ }
+ }
+ // set name resolution scope
+ ctl.m_tableList = m_tableList->m_tableList;
+ ctx_assert(ctl.m_tableList.size() >= 1 + 1);
+ ctl.m_aggrin = false;
+ // analyze select row
+ ctl.m_aggrok = true;
+ ctx_assert(m_exprRow != 0);
+ m_exprRow = static_cast<Plan_expr_row*>(m_exprRow->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_exprRow != 0);
+ // analyze group by row
+ ctl.m_aggrok = false;
+ if (m_groupRow != 0) {
+ m_groupRow = static_cast<Plan_expr_row*>(m_groupRow->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_groupRow != 0);
+ }
+ // analyze having predicate
+ ctl.m_aggrok = true;
+ if (m_havingPred != 0) {
+ m_havingPred = static_cast<Plan_pred*>(m_havingPred->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_havingPred != 0);
+ }
+ // ana|yze order by row
+ ctl.m_aggrok = true;
+ if (m_sortRow != 0) {
+ m_sortRow = static_cast<Plan_expr_row*>(m_sortRow->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_sortRow != 0);
+ }
+ // analyze the predicate
+ ctl.m_aggrok = false;
+ ctl.m_topand = true;
+ ctl.m_extra = false;
+ if (m_pred != 0) {
+ m_pred = static_cast<Plan_pred*>(m_pred->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_pred != 0);
+ }
+ // check if group by required
+ if (m_exprRow->anyAggr() && ! m_exprRow->allBound() && m_groupRow == 0) {
+ ctx.pushStatus(Error::Gen, "missing GROUP BY clause");
+ return 0;
+ }
+ // in special cases add "group by 1"
+ if (m_groupRow == 0) {
+ bool addgb = false;
+ if (m_havingPred != 0) {
+ // allowed by oracle but nearly useless
+ addgb = true;
+ } else if (m_exprRow->anyAggr() && m_sortRow != 0) {
+ // allowed by oracle but useless
+ ctx_assert(m_exprRow->allBound());
+ addgb = true;
+ }
+ if (addgb) {
+ ctx_log2(("adding 'group by 1'"));
+ m_groupRow = new Plan_expr_row(m_root);
+ m_root->saveNode(m_groupRow);
+ LexType type(LexType::Integer);
+ Plan_expr* expr = new Plan_expr_const(m_root, type, "1");
+ m_root->saveNode(expr);
+ m_groupRow->addExpr(expr);
+ m_groupRow = static_cast<Plan_expr_row*>(m_groupRow->analyze(ctx, ctl));
+ ctx_assert(ctx.ok());
+ ctx_assert(m_groupRow != 0);
+ }
+ }
+ // check group by allowed
+ if (m_groupRow != 0) {
+ if (! m_exprRow->isAllGroupBy(m_groupRow)) {
+ ctx.pushStatus(Error::Gen, "invalid GROUP BY expression in SELECT list");
+ return 0;
+ }
+ if (m_havingPred != 0) {
+ if (! m_havingPred->isGroupBy(m_groupRow)) {
+ ctx.pushStatus(Error::Gen, "invalid GROUP BY expression in HAVING clause");
+ return 0;
+ }
+ }
+ if (m_sortRow != 0) {
+ if (! m_sortRow->isAllGroupBy(m_groupRow)) {
+ ctx.pushStatus(Error::Gen, "invalid GROUP BY expression in ORDER BY clause");
+ return 0;
+ }
+ }
+ }
+ // log top level predicate
+ {
+ unsigned n = 0;
+ for (PredList::iterator i = ctl.m_topcomp.begin(); i != ctl.m_topcomp.end(); i++)
+ ctx_log2(("top level pred %u: count tables = %u, not interp = %u",
+ ++n,
+ (unsigned)(*i)->tableSet().size(),
+ (unsigned)(*i)->noInterp().size()));
+ }
+ // compose the raw query from lookups and scans
+ Plan_query* queryRaw = 0;
+ TableVector tableVector(1);
+ TableSet tsDone;
+ while (tableVector.size() < ctl.m_tableList.size()) {
+ Plan_table* tableBest = 0;
+ Plan_table::Index* indexBest = 0;
+ for (unsigned n = 1; n < ctl.m_tableList.size(); n++) {
+ Plan_table* table = ctl.m_tableList[n];
+ if (tsDone.find(table) != tsDone.end())
+ continue;
+ // get system table out of the way
+ if (table->dictTable().sysId()) {
+ tableBest = table;
+ break;
+ }
+ // find best match for primary key or index
+ for (unsigned i = 0; i <= table->indexCount(); i++) {
+ Plan_table::Index& index = table->m_indexList[i];
+ table->resolveSet(ctx, index, tsDone);
+ if (! ctx.ok())
+ return 0;
+ if (! index.m_keyFound)
+ continue;
+ // prefer smaller dependency set, smaller rank, less unused keys
+ int k;
+ (k = (indexBest == 0)) ||
+ (k = (indexBest->m_keySet.size() - index.m_keySet.size())) ||
+ (k = (indexBest->m_rank - index.m_rank)) ||
+ (k = (indexBest->m_keyCountUnused - index.m_keyCountUnused));
+ if (k > 0) {
+ tableBest = table;
+ indexBest = &index;
+ }
+ }
+ }
+ Plan_query* queryNext = 0;
+ Plan_table* tableNext = 0;
+ Plan_query_scan* queryScan = 0; // for pushing interpreted program
+ Plan_query_range* queryRange = 0; // ditto
+ if (tableBest == 0) {
+ // scan first unprocessed table
+ for (unsigned n = 1; n < ctl.m_tableList.size(); n++) {
+ Plan_table* table = ctl.m_tableList[n];
+ if (tsDone.find(table) != tsDone.end())
+ continue;
+ tableNext = table;
+ break;
+ }
+ ctx_assert(tableNext != 0);
+ queryScan = new Plan_query_scan(m_root);
+ m_root->saveNode(queryScan);
+ queryScan->setTable(tableNext);
+ queryNext = queryScan;
+ ctx_log2(("optim: scan %s", tableNext->getPrintName()));
+ } else if (tableBest->dictTable().sysId()) {
+ // "scan" system table
+ tableNext = tableBest;
+ Plan_query_sys* querySys = new Plan_query_sys(m_root);
+ m_root->saveNode(querySys);
+ querySys->setTable(tableNext);
+ queryNext = querySys;
+ ctx_log2(("optim: scan %s", tableNext->getPrintName()));
+ } else if (indexBest->m_keySet.size() > 0) {
+ // scan first table this one depends on
+ const TableSet& keySet = indexBest->m_keySet;
+ for (unsigned n = 1; n < ctl.m_tableList.size(); n++) {
+ Plan_table* table = ctl.m_tableList[n];
+ if (keySet.find(table) == keySet.end())
+ continue;
+ ctx_assert(tsDone.find(table) == tsDone.end());
+ tableNext = table;
+ break;
+ }
+ ctx_assert(tableNext != 0);
+ queryScan = new Plan_query_scan(m_root);
+ m_root->saveNode(queryScan);
+ queryScan->setTable(tableNext);
+ queryNext = queryScan;
+ ctx_log2(("optim: scan %s for %s", tableNext->getPrintName(), tableBest->getPrintName()));
+ } else if (indexBest->m_rank == 0) {
+ // primary key depends only on processed tables
+ tableNext = tableBest;
+ Plan_query_lookup* queryLookup = new Plan_query_lookup(m_root);
+ m_root->saveNode(queryLookup);
+ queryLookup->setTable(tableNext);
+ queryNext = queryLookup;
+ ctx_log2(("optim: lookup %s", tableNext->getPrintName()));
+ } else if (indexBest->m_rank == 1) {
+ // hash index key depends only on processed tables
+ tableNext = tableBest;
+ Plan_query_index* queryIndex = new Plan_query_index(m_root);
+ m_root->saveNode(queryIndex);
+ queryIndex->setTable(tableNext, indexBest);
+ queryNext = queryIndex;
+ ctx_log2(("optim: lookup %s via index %s", tableNext->getPrintName(), indexBest->m_dictIndex->getName().c_str()));
+ } else if (indexBest->m_rank == 2) {
+ // ordered index key depends only on processed tables
+ tableNext = tableBest;
+ queryRange = new Plan_query_range(m_root);
+ m_root->saveNode(queryRange);
+ queryRange->setTable(tableNext, indexBest);
+ queryNext = queryRange;
+ ctx_log2(("optim: range scan %s via index %s", tableNext->getPrintName(), indexBest->m_dictIndex->getName().c_str()));
+ } else {
+ ctx_assert(false);
+ }
+ if (queryRaw == 0) {
+ queryRaw = queryNext;
+ } else {
+ Plan_query_join* queryJoin = new Plan_query_join(m_root);
+ m_root->saveNode(queryJoin);
+ queryJoin->setInner(queryRaw);
+ queryJoin->setOuter(queryNext);
+ queryRaw = queryJoin;
+ }
+ tableVector.push_back(tableNext);
+ tsDone.insert(tableNext);
+ // push down part of top level predicate to table scan or range scan
+ Plan_pred* predPush = 0;
+ Plan_pred* predInterp = 0;
+ PredList::iterator i = ctl.m_topcomp.begin();
+ while (i != ctl.m_topcomp.end()) {
+ const TableSet& ts = (*i)->tableSet();
+ if (! std::includes(tsDone.begin(), tsDone.end(), ts.begin(), ts.end())) {
+ i++;
+ continue;
+ }
+ predPush = predPush == 0 ? *i : predPush->opAnd(*i);
+ if (queryScan != 0) {
+ const TableSet& ts2 = (*i)->noInterp();
+ if (ts2.find(tableNext) == ts2.end())
+ predInterp = predInterp == 0 ? *i : predInterp->opAnd(*i);
+ }
+ if (queryRange != 0) {
+ const TableSet& ts2 = (*i)->noInterp();
+ if (ts2.find(tableNext) == ts2.end())
+ predInterp = predInterp == 0 ? *i : predInterp->opAnd(*i);
+ }
+ // remove it from top level predicate
+ PredList::iterator j = i;
+ i++;
+ ctl.m_topcomp.erase(j);
+ }
+ if (predPush != 0) {
+ Plan_query_filter* queryPush = new Plan_query_filter(m_root);
+ m_root->saveNode(queryPush);
+ queryPush->setQuery(queryRaw);
+ queryPush->setPred(predPush);
+ queryPush->m_topTable = tableNext;
+ queryRaw = queryPush;
+ }
+ if (predInterp != 0) {
+ if (queryScan != 0)
+ queryScan->setInterp(predInterp);
+ else if (queryRange != 0)
+ queryRange->setInterp(predInterp);
+ else
+ ctx_assert(false);
+ }
+ }
+ ctx_assert(ctl.m_topcomp.empty());
+ // set base for column position offsets
+ for (unsigned n = 1; n < tableVector.size(); n++) {
+ Plan_table* table = tableVector[n];
+ if (n == 1) {
+ table->m_resOff = 1;
+ } else {
+ Plan_table* tablePrev = tableVector[n - 1];
+ table->m_resOff = tablePrev->m_resOff + tablePrev->m_exprColumns.size() - 1;
+ }
+ }
+ // next level up is one of project, count, group by
+ Plan_query* queryTop;
+ if (m_groupRow == 0) {
+ if (! m_exprRow->anyAggr()) {
+ Plan_query_project* queryProject = new Plan_query_project(m_root);
+ m_root->saveNode(queryProject);
+ queryProject->setQuery(queryRaw);
+ queryProject->setRow(m_exprRow);
+ queryProject->setLimit(m_limitOff, m_limitCnt);
+ queryTop = queryProject;
+ } else {
+ ctx_assert(m_exprRow->allBound());
+ Plan_query_count* queryCount = new Plan_query_count(m_root);
+ m_root->saveNode(queryCount);
+ queryCount->setQuery(queryRaw);
+ queryCount->setRow(m_exprRow);
+ queryTop = queryCount;
+ }
+ } else {
+ Plan_query_group* queryGroup = new Plan_query_group(m_root);
+ m_root->saveNode(queryGroup);
+ queryGroup->setQuery(queryRaw);
+ queryGroup->setDataRow(m_exprRow);
+ queryGroup->setGroupRow(m_groupRow);
+ if (m_havingPred != 0)
+ queryGroup->setHavingPred(m_havingPred);
+ queryTop = queryGroup;
+ }
+ // optional sort becomes new top level
+ if (m_sortRow != 0) {
+ Plan_query_sort* querySort = new Plan_query_sort(m_root);
+ m_root->saveNode(querySort);
+ querySort->setQuery(queryTop);
+ querySort->setRow(m_sortRow);
+ queryTop = querySort;
+ }
+ // optional distinct becomes new top level
+ if (m_distinct) {
+ Plan_query_distinct* queryDistinct = new Plan_query_distinct(m_root);
+ m_root->saveNode(queryDistinct);
+ queryDistinct->setQuery(queryTop);
+ queryTop = queryDistinct;
+ }
+ // return top node
+ return queryTop;
+}
+
+Exec_base*
+Plan_select::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_select::print(Ctx& ctx)
+{
+ ctx.print(" [select");
+ Plan_base* a[] = { m_tableList, m_exprRow, m_pred, m_groupRow, m_havingPred };
+ printList(ctx, a, 5);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_select.hpp b/ndb/src/client/odbc/codegen/Code_select.hpp
new file mode 100644
index 00000000000..eaa9b801f29
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_select.hpp
@@ -0,0 +1,132 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_select_hpp
+#define ODBC_CODEGEN_Code_select_hpp
+
+#include <common/common.hpp>
+#include "Code_stmt.hpp"
+#include "Code_expr_row.hpp"
+#include "Code_table_list.hpp"
+#include "Code_pred.hpp"
+
+/**
+ * @class Plan_select
+ * @brief General select in PlanTree
+ *
+ * General select. An initial PlanTree node.
+ */
+class Plan_select : public Plan_stmt {
+public:
+ Plan_select(Plan_root* root);
+ virtual ~Plan_select();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setList(Plan_table_list* tableList);
+ void setRow(Plan_expr_row* exprRow);
+ void setPred(Plan_pred* pred);
+ void setSort(Plan_expr_row* sortRow);
+ void setDistinct(bool distinct);
+ void setGroup(Plan_expr_row* groupRow);
+ void setHaving(Plan_pred* havingPred);
+ void setLimit(int off, int cnt);
+protected:
+ Plan_table_list* m_tableList;
+ Plan_expr_row* m_exprRow;
+ Plan_pred* m_pred;
+ Plan_expr_row* m_sortRow;
+ bool m_distinct;
+ Plan_expr_row* m_groupRow;
+ Plan_pred* m_havingPred;
+ int m_limitOff;
+ int m_limitCnt;
+};
+
+inline
+Plan_select::Plan_select(Plan_root* root) :
+ Plan_stmt(root),
+ m_tableList(0),
+ m_exprRow(0),
+ m_pred(0),
+ m_sortRow(0),
+ m_distinct(false),
+ m_groupRow(0),
+ m_havingPred(0),
+ m_limitOff(0),
+ m_limitCnt(-1)
+{
+}
+
+// children
+
+inline void
+Plan_select::setList(Plan_table_list* tableList)
+{
+ ctx_assert(tableList != 0);
+ m_tableList = tableList;
+}
+
+inline void
+Plan_select::setRow(Plan_expr_row* exprRow)
+{
+ ctx_assert(exprRow != 0);
+ m_exprRow = exprRow;
+}
+
+inline void
+Plan_select::setPred(Plan_pred* pred)
+{
+ ctx_assert(pred != 0);
+ m_pred = pred;
+}
+
+inline void
+Plan_select::setSort(Plan_expr_row* sortRow)
+{
+ ctx_assert(sortRow != 0);
+ m_sortRow = sortRow;
+}
+
+inline void
+Plan_select::setDistinct(bool distinct)
+{
+ m_distinct = distinct;
+}
+
+inline void
+Plan_select::setGroup(Plan_expr_row* groupRow)
+{
+ ctx_assert(groupRow != 0);
+ m_groupRow = groupRow;
+}
+
+inline void
+Plan_select::setHaving(Plan_pred* havingPred)
+{
+ ctx_assert(havingPred != 0);
+ m_havingPred = havingPred;
+}
+
+inline void
+Plan_select::setLimit(int off, int cnt)
+{
+ m_limitOff = off;
+ m_limitCnt = cnt;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_set_row.cpp b/ndb/src/client/odbc/codegen/Code_set_row.cpp
new file mode 100644
index 00000000000..dd13ba0c3f7
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_set_row.cpp
@@ -0,0 +1,44 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_set_row.hpp"
+#include "Code_dml_column.hpp"
+
+Plan_set_row::~Plan_set_row()
+{
+}
+
+Plan_base*
+Plan_set_row::analyze(Ctx& ctx, Ctl& ctl)
+{
+ return this;
+}
+
+Exec_base*
+Plan_set_row::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_set_row::print(Ctx& ctx)
+{
+ ctx.print(" [set_row");
+ Plan_base* a[] = { m_dmlRow, m_exprRow };
+ printList(ctx, a, 2);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_set_row.hpp b/ndb/src/client/odbc/codegen/Code_set_row.hpp
new file mode 100644
index 00000000000..10d62826ac7
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_set_row.hpp
@@ -0,0 +1,76 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_set_row_hpp
+#define ODBC_CODEGEN_Code_set_row_hpp
+
+#include <vector>
+#include <common/common.hpp>
+#include <common/DataRow.hpp>
+#include "Code_base.hpp"
+#include "Code_dml_row.hpp"
+#include "Code_expr_row.hpp"
+#include "Code_root.hpp"
+
+/**
+ * @class Plan_set_row
+ * @brief Row of column assigments in update
+ *
+ * Used only in parse. The column and expression rows are moved
+ * to the update node immediately after parse.
+ */
+class Plan_set_row : public Plan_base {
+public:
+ Plan_set_row(Plan_root* root);
+ virtual ~Plan_set_row();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void addColumn(Plan_dml_column* column);
+ void addExpr(Plan_expr* expr);
+protected:
+ friend class Plan_update;
+ friend class Plan_insert; // for MySql
+ Plan_dml_row* m_dmlRow;
+ Plan_expr_row* m_exprRow;
+};
+
+inline
+Plan_set_row::Plan_set_row(Plan_root* root) :
+ Plan_base(root)
+{
+ m_dmlRow = new Plan_dml_row(root);
+ root->saveNode(m_dmlRow);
+ m_exprRow = new Plan_expr_row(root);
+ root->saveNode(m_exprRow);
+}
+
+// children
+
+inline void
+Plan_set_row::addColumn(Plan_dml_column* column)
+{
+ m_dmlRow->addColumn(column);
+}
+
+inline void
+Plan_set_row::addExpr(Plan_expr* expr)
+{
+ m_exprRow->addExpr(expr);
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_stmt.cpp b/ndb/src/client/odbc/codegen/Code_stmt.cpp
new file mode 100644
index 00000000000..d790f667b84
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_stmt.cpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_stmt.hpp"
+
+// Plan_stmt
+
+Plan_stmt::~Plan_stmt()
+{
+}
+
+// XXX remove
+void
+Plan_stmt::describe(Ctx& ctx)
+{
+ ctx_log1(("unimplemented describe"));
+}
+
+// Exec_stmt
+
+Exec_stmt::Code::~Code()
+{
+}
+
+Exec_stmt::Data::~Data()
+{
+}
+
+Exec_stmt::~Exec_stmt()
+{
+}
+
+void
+Exec_stmt::bind(Ctx& ctx)
+{
+}
diff --git a/ndb/src/client/odbc/codegen/Code_stmt.hpp b/ndb/src/client/odbc/codegen/Code_stmt.hpp
new file mode 100644
index 00000000000..20b7fb965fb
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_stmt.hpp
@@ -0,0 +1,76 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_stmt_hpp
+#define ODBC_CODEGEN_Code_stmt_hpp
+
+#include <common/common.hpp>
+#include <common/DataType.hpp>
+#include "Code_base.hpp"
+
+class Ctx;
+
+/**
+ * @class Plan_stmt
+ * @brief Base class for statements in PlanTree
+ *
+ * A statement is a complete or partial SQL statement which can
+ * be optimized into executable statements Exec_stmt.
+ */
+class Plan_stmt : public Plan_base {
+public:
+ Plan_stmt(Plan_root* root);
+ virtual ~Plan_stmt() = 0;
+ virtual void describe(Ctx& ctx);
+};
+
+inline
+Plan_stmt::Plan_stmt(Plan_root* root) :
+ Plan_base(root)
+{
+}
+
+/**
+ * @class Exec_stmt
+ * @brief Base class for statements in ExecTree
+ */
+class Exec_stmt : public Exec_base {
+public:
+ class Code : public Exec_base::Code {
+ public:
+ virtual ~Code() = 0;
+ };
+ class Data : public Exec_base::Data {
+ public:
+ virtual ~Data() = 0;
+ };
+ Exec_stmt(Exec_root* root);
+ virtual ~Exec_stmt() = 0;
+ virtual void bind(Ctx& ctx);
+ virtual void execute(Ctx& ctx, Ctl& ctl) = 0;
+protected:
+ friend class Exec_root;
+ bool m_topLevel;
+};
+
+inline
+Exec_stmt::Exec_stmt(Exec_root* root) :
+ Exec_base(root),
+ m_topLevel(false)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_table.cpp b/ndb/src/client/odbc/codegen/Code_table.cpp
new file mode 100644
index 00000000000..ee3c2a2ed07
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_table.cpp
@@ -0,0 +1,254 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <dictionary/DictSchema.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_table.hpp"
+#include "Code_column.hpp"
+#include "Code_expr_column.hpp"
+
+Plan_table::~Plan_table()
+{
+}
+
+Plan_base*
+Plan_table::analyze(Ctx& ctx, Ctl& ctl)
+{
+ if (m_dictTable != 0) // already done
+ return this;
+ DictTable* table = dictSchema().findTable(m_name);
+ if (table == 0) {
+ table = dictSchema().loadTable(ctx, m_name);
+ if (table == 0) {
+ ctx.pushStatus(Sqlstate::_42S02, Error::Gen, "table %s not found", m_name.c_str());
+ return 0;
+ }
+ }
+ m_dictTable = table;
+ // indexes
+ m_indexList.resize(1 + m_dictTable->indexCount());
+ for (unsigned i = 0; i <= indexCount(); i++) {
+ Index& index = m_indexList[i];
+ index.m_pos = i;
+ if (index.m_pos == 0) {
+ index.m_keyCount = m_dictTable->keyCount();
+ index.m_rank = 0;
+ } else {
+ index.m_dictIndex = m_dictTable->getIndex(i);
+ index.m_keyCount = index.m_dictIndex->getSize();
+ if (index.m_dictIndex->getType() == NdbDictionary::Object::UniqueHashIndex) {
+ index.m_rank = 1;
+ } else if (index.m_dictIndex->getType() == NdbDictionary::Object::OrderedIndex) {
+ index.m_rank = 2;
+ } else {
+ ctx_assert(false);
+ }
+ }
+ index.m_keyEqList.resize(1 + index.m_keyCount);
+ }
+ return this;
+}
+
+int
+Plan_table::resolveColumn(Ctx& ctx, Plan_column* column, bool stripSchemaName)
+{
+ ctx_assert(column != 0);
+ bool dml, unq;
+ switch (column->m_type) {
+ case Plan_column::Type_expr:
+ dml = false;
+ unq = false;
+ break;
+ case Plan_column::Type_dml:
+ dml = true;
+ unq = true;
+ break;
+ case Plan_column::Type_idx:
+ dml = false;
+ unq = true;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ ColumnVector& columns = ! dml ? m_exprColumns : m_dmlColumns;
+ const BaseString& name = column->m_name;
+ const BaseString& cname = column->m_cname;
+ ctx_log3(("resolve %s column %s in table %s", ! dml ? "expr" : "dml", column->getPrintName(), getPrintName()));
+ // find column in table
+ DictColumn* dictColumn = dictTable().findColumn(name);
+ if (dictColumn == 0)
+ return 0;
+ // qualified column must match table correlation name
+ if (! cname.empty()) {
+ const char* str;
+ if (! m_cname.empty()) {
+ str = m_cname.c_str();
+ } else {
+ str = m_name.c_str();
+ if (stripSchemaName && strrchr(str, '.') != 0)
+ str = strrchr(str, '.') + 1;
+ }
+ if (strcmp(cname.c_str(), str) != 0)
+ return 0;
+ }
+ // find in positional list or add to it
+ unsigned resPos;
+ for (resPos = 1; resPos < columns.size(); resPos++) {
+ if (strcmp(columns[resPos]->getName().c_str(), name.c_str()) != 0)
+ continue;
+ // these columns must be unique
+ if (unq) {
+ ctx.pushStatus(Error::Gen, "duplicate column %s", column->getName().c_str());
+ return -1;
+ }
+ break;
+ }
+ if (resPos >= columns.size()) {
+ columns.push_back(column);
+ }
+ ctx_log3(("resolve to attrId %u pos %u", (unsigned)dictColumn->getAttrId(), resPos));
+ column->m_dictColumn = dictColumn;
+ column->m_resTable = this;
+ column->m_resPos = resPos;
+ // found
+ return 1;
+}
+
+bool
+Plan_table::resolveEq(Ctx& ctx, Plan_expr_column* column, Plan_expr* expr)
+{
+ ctx_assert(m_dictTable != 0);
+ const TableSet& ts = expr->tableSet();
+ TableSet::const_iterator i = ts.find(this);
+ if (i != ts.end())
+ return false;
+ unsigned found = 0;
+ for (unsigned i = 0; i <= indexCount(); i++) {
+ Index& index = m_indexList[i];
+ for (unsigned n = 1, cnt = 0; n <= index.m_keyCount; n++) {
+ const DictColumn* dictColumn = 0;
+ if (index.m_pos == 0) {
+ ctx_assert(m_dictTable != 0);
+ dictColumn = m_dictTable->getKey(n);
+ } else {
+ ctx_assert(index.m_dictIndex != 0);
+ dictColumn = index.m_dictIndex->getColumn(n);
+ }
+ if (dictColumn != &column->dictColumn())
+ continue;
+ ctx_assert(++cnt == 1);
+ index.m_keyEqList[n].push_back(expr);
+ if (index.m_pos == 0)
+ ctx_log2(("%s: found match to primary key column %s pos %u", getPrintName(), column->getPrintName(), n));
+ else
+ ctx_log2(("%s: found match to index %s column %s pos %u", getPrintName(), index.m_dictIndex->getName().c_str(), column->getPrintName(), n));
+ found++;
+ }
+ }
+ return (found != 0);
+}
+
+void
+Plan_table::resolveSet(Ctx& ctx, Index& index, const TableSet& tsDone)
+{
+ index.m_keyFound = false;
+ ExprVector keyEq;
+ keyEq.resize(1 + index.m_keyCount);
+ resolveSet(ctx, index, tsDone, keyEq, 1);
+}
+
+void
+Plan_table::resolveSet(Ctx& ctx, Index& index, const TableSet& tsDone, ExprVector& keyEq, unsigned n)
+{
+ if (n <= index.m_keyCount) {
+ // building up combinations
+ ExprList& keyEqList = index.m_keyEqList[n];
+ for (ExprList::iterator i = keyEqList.begin(); i != keyEqList.end(); i++) {
+ keyEq[n] = *i;
+ resolveSet(ctx, index, tsDone, keyEq, n + 1);
+ }
+ if (! keyEqList.empty() || index.m_rank <= 1 || n == 1)
+ return;
+ // ordered index with maximal initial key match
+ }
+ TableSet keySet;
+ for (unsigned i = 1; i <= n - 1; i++) {
+ const TableSet& tableSet = keyEq[i]->tableSet();
+ for (TableSet::const_iterator j = tableSet.begin(); j != tableSet.end(); j++) {
+ if (tsDone.find(*j) == tsDone.end())
+ keySet.insert(*j);
+ }
+ }
+ if (! index.m_keyFound || index.m_keySet.size() > keySet.size()) {
+ index.m_keyFound = true;
+ index.m_keyEq = keyEq;
+ index.m_keySet = keySet;
+ index.m_keyCountUsed = n - 1;
+ index.m_keyCountUnused = index.m_keyCount - index.m_keyCountUsed;
+ // set matching size
+ index.m_keyEq.resize(1 + index.m_keyCountUsed);
+ }
+}
+
+bool
+Plan_table::exactKey(Ctx& ctx, const Index* indexKey) const
+{
+ ctx_assert(indexKey != 0 && indexKey == &m_indexList[indexKey->m_pos]);
+ for (unsigned i = 0; i <= indexCount(); i++) {
+ const Index& index = m_indexList[i];
+ const ExprListVector& keyEqList = index.m_keyEqList;
+ for (unsigned n = 1; n <= index.m_keyCount; n++) {
+ if (index.m_pos == indexKey->m_pos) {
+ ctx_assert(keyEqList[n].size() >= 1);
+ if (keyEqList[n].size() > 1) {
+ ctx_log2(("index %u not exact: column %u has %u > 1 matches",
+ indexKey->m_pos,
+ n,
+ (unsigned)keyEqList[n].size()));
+ return false;
+ }
+ } else {
+ if (keyEqList[n].size() > 0) {
+ ctx_log2(("index %u not exact: index %u column %u has %u > 0 matches",
+ indexKey->m_pos,
+ index.m_pos,
+ n,
+ (unsigned)keyEqList[n].size()));
+ return false;
+ }
+ }
+ }
+ }
+ ctx_log2(("index %u is exact", indexKey->m_pos));
+ return true;
+}
+
+Exec_base*
+Plan_table::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_table::print(Ctx& ctx)
+{
+ ctx.print(" [table %s]", getPrintName());
+}
diff --git a/ndb/src/client/odbc/codegen/Code_table.hpp b/ndb/src/client/odbc/codegen/Code_table.hpp
new file mode 100644
index 00000000000..8a95b8fa26c
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_table.hpp
@@ -0,0 +1,202 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_table_hpp
+#define ODBC_CODEGEN_Code_table_hpp
+
+#include <vector>
+#include <common/common.hpp>
+#include "Code_base.hpp"
+
+class DictTable;
+class DictColumn;
+class DictIndex;
+class Plan_query_filter;
+class Plan_query_lookup;
+class Plan_query_range;
+class Plan_column;
+class Plan_expr_column;
+class Plan_select;
+class Plan_delete;
+class Plan_delete_lookup;
+class Plan_update;
+class Plan_update_lookup;
+
+/**
+ * @class Plan_table
+ * @brief Table node in PlanTree
+ *
+ * This is a pure Plan node. Final executable nodes have table
+ * information built-in.
+ */
+class Plan_table : public Plan_base {
+public:
+ Plan_table(Plan_root* root, const BaseString& name);
+ virtual ~Plan_table();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // attributes
+ const BaseString& getName() const;
+ const BaseString& getCname() const;
+ const char* getPrintName() const;
+ void setCname(const BaseString& cname);
+ const DictTable& dictTable() const;
+ unsigned indexCount() const;
+ // resolve
+ const ColumnVector& exprColumns() const;
+ const ColumnVector& dmlColumns() const;
+protected:
+ friend class Plan_column;
+ friend class Plan_query_filter;
+ friend class Plan_query_lookup;
+ friend class Plan_query_index;
+ friend class Plan_query_range;
+ friend class Plan_expr_column;
+ friend class Plan_select;
+ friend class Plan_delete;
+ friend class Plan_delete_lookup;
+ friend class Plan_delete_index;
+ friend class Plan_update;
+ friend class Plan_update_lookup;
+ friend class Plan_update_index;
+ BaseString m_name;
+ BaseString m_cname;
+ BaseString m_printName;
+ DictTable* m_dictTable;
+ /*
+ * Resolve column. Returns 1 on found, 0 on not found, and -1 on error.
+ * Modifies both table and column data.
+ */
+ int resolveColumn(Ctx& ctx, Plan_column* column, bool stripSchemaName = false);
+ ColumnVector m_exprColumns;
+ ColumnVector m_dmlColumns;
+ /*
+ * Offset for resolved columns in join. This is sum over m_exprColumns
+ * lengths for all preceding tables.
+ */
+ unsigned m_resOff;
+ /*
+ * Each column in primary key and unique hash index has list of
+ * expressions it is set equal to in the where-clause (at top level).
+ */
+ bool resolveEq(Ctx& ctx, Plan_expr_column* column, Plan_expr* expr);
+ /*
+ * Index struct for primary key and indexes.
+ */
+ struct Index {
+ Index() :
+ m_pos(0),
+ m_keyFound(false),
+ m_dictIndex(0),
+ m_rank(~0),
+ m_keyCount(0),
+ m_keyCountUsed(0) {
+ }
+ unsigned m_pos;
+ ExprListVector m_keyEqList;
+ bool m_keyFound;
+ ExprVector m_keyEq;
+ TableSet m_keySet;
+ const DictIndex* m_dictIndex; // for index only
+ unsigned m_rank; // 0-pk 1-hash index 2-ordered index
+ unsigned m_keyCount; // number of columns
+ unsigned m_keyCountUsed; // may be less for ordered index
+ unsigned m_keyCountUnused; // m_keyCount - m_keyCountUsed
+ };
+ typedef std::vector<Index> IndexList; // primary key is entry 0
+ IndexList m_indexList;
+ /*
+ * Find set of additional tables (maybe empty) required to resolve the key
+ * columns.
+ */
+ void resolveSet(Ctx& ctx, Index& index, const TableSet& tsDone);
+ void resolveSet(Ctx& ctx, Index& index, const TableSet& tsDone, ExprVector& keyEq, unsigned n);
+ /*
+ * Check for exactly one key or index match.
+ */
+ bool exactKey(Ctx& ctx, const Index* indexKey) const;
+};
+
+inline
+Plan_table::Plan_table(Plan_root* root, const BaseString& name) :
+ Plan_base(root),
+ m_name(name),
+ m_printName(name),
+ m_dictTable(0),
+ m_exprColumns(1), // 1-based
+ m_dmlColumns(1), // 1-based
+ m_resOff(0),
+ m_indexList(1)
+{
+}
+
+inline const BaseString&
+Plan_table::getName() const
+{
+ return m_name;
+}
+
+inline const BaseString&
+Plan_table::getCname() const
+{
+ return m_cname;
+}
+
+inline const char*
+Plan_table::getPrintName() const
+{
+ return m_printName.c_str();
+}
+
+inline void
+Plan_table::setCname(const BaseString& cname)
+{
+ m_cname.assign(cname);
+ m_printName.assign(m_name);
+ if (! m_cname.empty()) {
+ m_printName.append(" ");
+ m_printName.append(m_cname);
+ }
+}
+
+inline const DictTable&
+Plan_table::dictTable() const
+{
+ ctx_assert(m_dictTable != 0);
+ return *m_dictTable;
+}
+
+inline unsigned
+Plan_table::indexCount() const
+{
+ ctx_assert(m_indexList.size() > 0);
+ return m_indexList.size() - 1;
+}
+
+inline const Plan_table::ColumnVector&
+Plan_table::exprColumns() const
+{
+ return m_exprColumns;
+}
+
+inline const Plan_table::ColumnVector&
+Plan_table::dmlColumns() const
+{
+ return m_dmlColumns;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_table_list.cpp b/ndb/src/client/odbc/codegen/Code_table_list.cpp
new file mode 100644
index 00000000000..ea9f4fdc26e
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_table_list.cpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Code_table_list.hpp"
+
+Plan_table_list::~Plan_table_list()
+{
+}
+
+Plan_base*
+Plan_table_list::analyze(Ctx& ctx, Ctl& ctl)
+{
+ // analyze the tables
+ for (unsigned i = 1, n = countTable(); i <= n; i++) {
+ Plan_table* table = getTable(i);
+ table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ }
+ // node was not replaced
+ return this;
+}
+
+Exec_base*
+Plan_table_list::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_table_list::print(Ctx& ctx)
+{
+ ctx.print(" [table_list");
+ for (unsigned i = 1, n = countTable(); i <= n; i++) {
+ Plan_base* a[] = { m_tableList[i] };
+ printList(ctx, a, 1);
+ }
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_table_list.hpp b/ndb/src/client/odbc/codegen/Code_table_list.hpp
new file mode 100644
index 00000000000..47989166cac
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_table_list.hpp
@@ -0,0 +1,73 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_table_list_hpp
+#define ODBC_CODEGEN_Code_table_list_hpp
+
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_table.hpp"
+
+/**
+ * @class Plan_table_list
+ * @brief List of tables in select statement
+ */
+class Plan_table_list : public Plan_base {
+public:
+ Plan_table_list(Plan_root* root);
+ virtual ~Plan_table_list();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ unsigned countTable() const;
+ void addTable(Plan_table* table);
+ Plan_table* getTable(unsigned i) const;
+protected:
+ friend class Plan_select;
+ TableVector m_tableList;
+};
+
+inline
+Plan_table_list::Plan_table_list(Plan_root* root) :
+ Plan_base(root),
+ m_tableList(1)
+{
+}
+
+// children
+
+inline unsigned
+Plan_table_list::countTable() const
+{
+ return m_tableList.size() - 1;
+}
+
+inline void
+Plan_table_list::addTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_tableList.push_back(table);
+}
+
+inline Plan_table*
+Plan_table_list::getTable(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= countTable() && m_tableList[i] != 0);
+ return m_tableList[i];
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_update.cpp b/ndb/src/client/odbc/codegen/Code_update.cpp
new file mode 100644
index 00000000000..0b33cd628b4
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_update.cpp
@@ -0,0 +1,246 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_update.hpp"
+#include "Code_update_lookup.hpp"
+#include "Code_update_index.hpp"
+#include "Code_update_scan.hpp"
+#include "Code_table.hpp"
+#include "Code_query_project.hpp"
+#include "Code_query_filter.hpp"
+#include "Code_query_scan.hpp"
+#include "Code_query_lookup.hpp"
+#include "Code_query_index.hpp"
+#include "Code_query_range.hpp"
+#include "Code_query_repeat.hpp"
+#include "Code_root.hpp"
+
+// Plan_update
+
+Plan_update::~Plan_update()
+{
+}
+
+Plan_base*
+Plan_update::analyze(Ctx& ctx, Ctl& ctl)
+{
+ stmtArea().stmtInfo().setName(Stmt_name_update);
+ // analyze the table
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ // get column and expression rows
+ ctx_assert(m_setRow != 0);
+ setDmlRow(m_setRow->m_dmlRow);
+ setExprRow(m_setRow->m_exprRow);
+ m_setRow = 0;
+ // implied by parse
+ ctx_assert(m_dmlRow->getSize() == m_exprRow->getSize());
+ // set name resolution scope
+ ctl.m_tableList.resize(1 + 1); // indexed from 1
+ ctl.m_tableList[1] = m_table;
+ // analyze the rows
+ m_dmlRow->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctl.m_dmlRow = m_dmlRow; // row type to convert to
+ ctl.m_const = true; // set to constants
+ m_exprRow->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ bool setConst = ctl.m_const;
+ ctl.m_dmlRow = 0;
+ Plan_dml* stmt = 0;
+ // top level query is a project
+ Plan_query_project* queryProject = new Plan_query_project(m_root);
+ m_root->saveNode(queryProject);
+ queryProject->setRow(m_exprRow);
+ if (m_pred != 0) {
+ // analyze the predicate
+ ctl.m_topand = true;
+ ctl.m_extra = false;
+ m_pred = static_cast<Plan_pred*>(m_pred->analyze(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_pred != 0);
+ // check for key match
+ Plan_table::Index* indexBest = 0;
+ for (unsigned i = 0; i <= m_table->indexCount(); i++) {
+ Plan_table::Index& index = m_table->m_indexList[i];
+ TableSet tsDone;
+ m_table->resolveSet(ctx, index, tsDone);
+ if (! ctx.ok())
+ return 0;
+ if (! index.m_keyFound)
+ continue;
+ // prefer smaller rank, less unused keys
+ int k;
+ (k = (indexBest == 0)) ||
+ (k = (indexBest->m_rank - index.m_rank)) ||
+ (k = (indexBest->m_keyCountUnused - index.m_keyCountUnused));
+ if (k > 0)
+ indexBest = &index;
+ }
+ if (indexBest != 0) {
+ const bool exactKey = indexBest->m_rank <= 1 ? m_table->exactKey(ctx, indexBest) : false;
+ const bool direct = setConst && ! ctl.m_extra && exactKey;
+ ctx_log3(("update direct=%d: const=%d extra=%d exact=%d", direct, setConst, ctl.m_extra, exactKey));
+ if (indexBest->m_rank == 0) {
+ // primary key
+ Plan_update_lookup* updateLookup = new Plan_update_lookup(m_root);
+ m_root->saveNode(updateLookup);
+ updateLookup->setTable(m_table);
+ updateLookup->setDmlRow(m_dmlRow);
+ if (direct) {
+ // constant values and exact key match
+ Plan_query_repeat* queryRepeat = new Plan_query_repeat(m_root, 1);
+ m_root->saveNode(queryRepeat);
+ queryProject->setQuery(queryRepeat);
+ } else {
+ // more conditions or non-constant values
+ Plan_query_lookup* queryLookup = new Plan_query_lookup(m_root);
+ m_root->saveNode(queryLookup);
+ Plan_query_filter* queryFilter = new Plan_query_filter(m_root);
+ m_root->saveNode(queryFilter);
+ queryLookup->setTable(m_table);
+ queryFilter->setQuery(queryLookup);
+ queryFilter->setPred(m_pred);
+ queryFilter->m_topTable = m_table;
+ queryProject->setQuery(queryFilter);
+ }
+ updateLookup->setQuery(queryProject);
+ stmt = updateLookup;
+ } else if (indexBest->m_rank == 1) {
+ // hash index
+ Plan_update_index* updateIndex = new Plan_update_index(m_root);
+ m_root->saveNode(updateIndex);
+ updateIndex->setTable(m_table, indexBest);
+ updateIndex->setDmlRow(m_dmlRow);
+ if (direct) {
+ // constant values and exact key match
+ Plan_query_repeat* queryRepeat = new Plan_query_repeat(m_root, 1);
+ m_root->saveNode(queryRepeat);
+ queryProject->setQuery(queryRepeat);
+ } else {
+ // more conditions or non-constant values
+ Plan_query_index* queryIndex = new Plan_query_index(m_root);
+ m_root->saveNode(queryIndex);
+ Plan_query_filter* queryFilter = new Plan_query_filter(m_root);
+ m_root->saveNode(queryFilter);
+ queryIndex->setTable(m_table, indexBest);
+ queryFilter->setQuery(queryIndex);
+ queryFilter->setPred(m_pred);
+ queryFilter->m_topTable = m_table;
+ queryProject->setQuery(queryFilter);
+ }
+ updateIndex->setQuery(queryProject);
+ stmt = updateIndex;
+ } else if (indexBest->m_rank == 2) {
+ // ordered index
+ Plan_update_scan* updateScan = new Plan_update_scan(m_root);
+ m_root->saveNode(updateScan);
+ updateScan->setTable(m_table);
+ updateScan->setDmlRow(m_dmlRow);
+ Plan_query_range* queryRange = new Plan_query_range(m_root);
+ m_root->saveNode(queryRange);
+ queryRange->setTable(m_table, indexBest);
+ queryRange->setExclusive();
+ Plan_query_filter* queryFilter = new Plan_query_filter(m_root);
+ m_root->saveNode(queryFilter);
+ queryFilter->setQuery(queryRange);
+ queryFilter->setPred(m_pred);
+ queryFilter->m_topTable = m_table;
+ // interpeter
+ const TableSet& ts2 = m_pred->noInterp();
+ ctx_assert(ts2.size() <= 1);
+ if (ts2.size() == 0) {
+ queryRange->setInterp(m_pred);
+ }
+ queryProject->setQuery(queryFilter);
+ updateScan->setQuery(queryProject);
+ stmt = updateScan;
+ } else {
+ ctx_assert(false);
+ }
+ } else {
+ // scan update with filter
+ Plan_update_scan* updateScan = new Plan_update_scan(m_root);
+ m_root->saveNode(updateScan);
+ updateScan->setTable(m_table);
+ updateScan->setDmlRow(m_dmlRow);
+ Plan_query_scan* queryScan = new Plan_query_scan(m_root);
+ m_root->saveNode(queryScan);
+ queryScan->setTable(m_table);
+ queryScan->setExclusive();
+ Plan_query_filter* queryFilter = new Plan_query_filter(m_root);
+ m_root->saveNode(queryFilter);
+ queryFilter->setQuery(queryScan);
+ queryFilter->setPred(m_pred);
+ queryFilter->m_topTable = m_table;
+ // interpeter
+ const TableSet& ts2 = m_pred->noInterp();
+ ctx_assert(ts2.size() <= 1);
+ if (ts2.size() == 0) {
+ queryScan->setInterp(m_pred);
+ }
+ queryProject->setQuery(queryFilter);
+ updateScan->setQuery(queryProject);
+ stmt = updateScan;
+ }
+ } else {
+ // scan update without filter
+ Plan_update_scan* updateScan = new Plan_update_scan(m_root);
+ m_root->saveNode(updateScan);
+ updateScan->setTable(m_table);
+ updateScan->setDmlRow(m_dmlRow);
+ Plan_query_scan* queryScan = new Plan_query_scan(m_root);
+ m_root->saveNode(queryScan);
+ queryScan->setTable(m_table);
+ queryScan->setExclusive();
+ queryProject->setQuery(queryScan);
+ updateScan->setQuery(queryProject);
+ stmt = updateScan;
+ }
+ // set base for column position offsets
+ m_table->m_resOff = 1;
+ return stmt;
+}
+
+void
+Plan_update::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "UPDATE WHERE", SQL_DIAG_UPDATE_WHERE);
+}
+
+Exec_base*
+Plan_update::codegen(Ctx& ctx, Ctl& ctl)
+{
+ ctx_assert(false);
+ return 0;
+}
+
+void
+Plan_update::print(Ctx& ctx)
+{
+ ctx.print(" [update");
+ Plan_base* a[] = { m_table, m_setRow, m_dmlRow, m_exprRow };
+ printList(ctx, a, 4);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_update.hpp b/ndb/src/client/odbc/codegen/Code_update.hpp
new file mode 100644
index 00000000000..380b651518b
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_update.hpp
@@ -0,0 +1,102 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_update_hpp
+#define ODBC_CODEGEN_Code_update_hpp
+
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_dml.hpp"
+#include "Code_set_row.hpp"
+#include "Code_table.hpp"
+#include "Code_pred.hpp"
+#include "Code_query.hpp"
+
+/**
+ * @class Plan_update
+ * @brief Update in PlanTree
+ */
+class Plan_update : public Plan_dml {
+public:
+ Plan_update(Plan_root* root);
+ virtual ~Plan_update();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ void setTable(Plan_table* table);
+ void setRow(Plan_set_row* setRow);
+ void setDmlRow(Plan_dml_row* dmlRow);
+ void setExprRow(Plan_expr_row* exprRow);
+ void setPred(Plan_pred* pred);
+protected:
+ Plan_table* m_table;
+ Plan_set_row* m_setRow;
+ Plan_dml_row* m_dmlRow;
+ Plan_expr_row* m_exprRow;
+ Plan_pred* m_pred;
+};
+
+inline
+Plan_update::Plan_update(Plan_root* root) :
+ Plan_dml(root),
+ m_table(0),
+ m_setRow(0),
+ m_dmlRow(0),
+ m_exprRow(0),
+ m_pred(0)
+{
+}
+
+// children
+
+inline void
+Plan_update::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+inline void
+Plan_update::setRow(Plan_set_row* setRow)
+{
+ ctx_assert(setRow != 0);
+ m_setRow = setRow;
+}
+
+inline void
+Plan_update::setDmlRow(Plan_dml_row* dmlRow)
+{
+ ctx_assert(dmlRow != 0);
+ m_dmlRow = dmlRow;
+}
+
+inline void
+Plan_update::setExprRow(Plan_expr_row* exprRow)
+{
+ ctx_assert(exprRow != 0);
+ m_exprRow = exprRow;
+}
+
+inline void
+Plan_update::setPred(Plan_pred* pred)
+{
+ ctx_assert(pred != 0);
+ m_pred = pred;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_update_index.cpp b/ndb/src/client/odbc/codegen/Code_update_index.cpp
new file mode 100644
index 00000000000..6f74db0d913
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_update_index.cpp
@@ -0,0 +1,196 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_dml_column.hpp"
+#include "Code_expr.hpp"
+#include "Code_update_index.hpp"
+#include "Code_table.hpp"
+#include "Code_root.hpp"
+
+// Plan_update_index
+
+Plan_update_index::~Plan_update_index()
+{
+}
+
+Plan_base*
+Plan_update_index::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctl.m_dmlRow = m_dmlRow; // row type to convert to
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+void
+Plan_update_index::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "UPDATE WHERE", SQL_DIAG_UPDATE_WHERE);
+}
+
+Exec_base*
+Plan_update_index::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // generate code for the query
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // set up
+ ctx_assert(m_table != 0 && m_index != 0);
+ const BaseString& tableName = m_table->getName();
+ ctx_assert(m_index->m_dictIndex != 0);
+ const DictIndex& dictIndex = *m_index->m_dictIndex;
+ const BaseString& indexName = dictIndex.getName();
+ const unsigned keyCount = m_index->m_keyCount;
+ const ColumnVector& columns = m_table->dmlColumns();
+ ctx_assert(columns.size() > 0);
+ const unsigned attrCount = columns.size() - 1;
+ // create the code
+ Exec_update_index::Code& code = *new Exec_update_index::Code(keyCount);
+ code.m_tableName = strcpy(new char[tableName.length() + 1], tableName.c_str());
+ code.m_indexName = strcpy(new char[indexName.length() + 1], indexName.c_str());
+ // key attributes
+ code.m_keyId = new NdbAttrId[1 + keyCount];
+ code.m_keyId[0] = (NdbAttrId)-1;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ const DictColumn* keyColumn = dictIndex.getColumn(k);
+ const SqlType& sqlType = keyColumn->sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_keySpecs.setEntry(k, sqlSpec);
+ code.m_keyId[k] = k - 1; // index column order
+ }
+ // matching expressions
+ ctx_assert(m_index->m_keyFound);
+ const ExprVector& keyEq = m_index->m_keyEq;
+ ctx_assert(keyEq.size() == 1 + keyCount);
+ code.m_keyMatch = new Exec_expr* [1 + keyCount];
+ code.m_keyMatch[0] = 0;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ Plan_expr* expr = keyEq[k];
+ Exec_expr* execExpr = static_cast<Exec_expr*>(expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ code.m_keyMatch[k] = execExpr;
+ }
+ // updated attributes
+ code.m_attrCount = attrCount;
+ code.m_attrId = new NdbAttrId[1 + attrCount];
+ code.m_attrId[0] = (NdbAttrId)-1;
+ for (unsigned i = 1; i <= attrCount; i++) {
+ Plan_column* column = columns[i];
+ ctx_assert(column != 0);
+ const DictColumn& dictColumn = column->dictColumn();
+ code.m_attrId[i] = dictColumn.getAttrId();
+ }
+ // create the exec
+ Exec_update_index* exec = new Exec_update_index(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ return exec;
+}
+
+void
+Plan_update_index::print(Ctx& ctx)
+{
+ ctx.print(" [update_index");
+ Plan_base* a[] = { m_table, m_query };
+ printList(ctx, a, sizeof(a)/sizeof(a[0]));
+ ctx.print("]");
+}
+
+// Exec_delete
+
+Exec_update_index::Code::~Code()
+{
+ delete[] m_tableName;
+ delete[] m_keyId;
+ delete[] m_keyMatch;
+ delete[] m_attrId;
+}
+
+Exec_update_index::Data::~Data()
+{
+}
+
+Exec_update_index::~Exec_update_index()
+{
+}
+
+void
+Exec_update_index::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // allocate the subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // create data
+ Data& data = *new Data;
+ setData(data);
+ // allocate matching expressions
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* expr = code.m_keyMatch[k];
+ ctx_assert(expr != 0);
+ expr->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+}
+
+void
+Exec_update_index::close(Ctx& ctx)
+{
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+}
+
+void
+Exec_update_index::print(Ctx& ctx)
+{
+ ctx.print(" [update_index");
+ if (m_code != 0) {
+ const Code& code = getCode();
+ ctx.print(" keyId=");
+ for (unsigned i = 1; i <= code.m_keyCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_keyId[i]);
+ }
+ ctx.print(" attrId=");
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_attrId[i]);
+ }
+ }
+ Exec_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_update_index.hpp b/ndb/src/client/odbc/codegen/Code_update_index.hpp
new file mode 100644
index 00000000000..bbad822650a
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_update_index.hpp
@@ -0,0 +1,171 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_update_index_hpp
+#define ODBC_CODEGEN_Code_update_index_hpp
+
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_dml.hpp"
+#include "Code_table.hpp"
+#include "Code_query.hpp"
+
+/**
+ * @class Plan_update_index
+ * @brief Update in PlanTree
+ */
+class Plan_update_index : public Plan_dml {
+public:
+ Plan_update_index(Plan_root* root);
+ virtual ~Plan_update_index();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setTable(Plan_table* table, Plan_table::Index* index);
+ void setDmlRow(Plan_dml_row* dmlRow);
+ void setQuery(Plan_query* query);
+protected:
+ Plan_table* m_table;
+ Plan_table::Index* m_index;
+ Plan_dml_row* m_dmlRow;
+ Plan_query* m_query;
+};
+
+inline
+Plan_update_index::Plan_update_index(Plan_root* root) :
+ Plan_dml(root),
+ m_table(0),
+ m_dmlRow(0),
+ m_query(0)
+{
+}
+
+inline void
+Plan_update_index::setTable(Plan_table* table, Plan_table::Index* index)
+{
+ ctx_assert(table != 0 && index != 0 && index == &table->m_indexList[index->m_pos] && index->m_pos != 0);
+ m_table = table;
+ m_index = index;
+}
+
+inline void
+Plan_update_index::setDmlRow(Plan_dml_row* dmlRow)
+{
+ ctx_assert(dmlRow != 0);
+ m_dmlRow = dmlRow;
+}
+
+inline void
+Plan_update_index::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+/**
+ * @class Exec_update_index
+ * @brief Insert in ExecTree
+ */
+class Exec_update_index : public Exec_dml {
+public:
+ class Code : public Exec_dml::Code {
+ public:
+ Code(unsigned keyCount);
+ virtual ~Code();
+ protected:
+ friend class Plan_update_index;
+ friend class Exec_update_index;
+ const char* m_tableName;
+ const char* m_indexName;
+ unsigned m_keyCount;
+ SqlSpecs m_keySpecs; // key types
+ NdbAttrId* m_keyId;
+ Exec_expr** m_keyMatch; // XXX pointers for now
+ unsigned m_attrCount;
+ NdbAttrId* m_attrId;
+ };
+ class Data : public Exec_dml::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_update_index;
+ };
+ Exec_update_index(Exec_root* root);
+ virtual ~Exec_update_index();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+protected:
+ Exec_query* m_query;
+};
+
+inline
+Exec_update_index::Code::Code(unsigned keyCount) :
+ m_tableName(0),
+ m_indexName(0),
+ m_keyCount(keyCount),
+ m_keySpecs(keyCount),
+ m_keyId(0),
+ m_keyMatch(0),
+ m_attrCount(0),
+ m_attrId(0)
+{
+}
+
+inline
+Exec_update_index::Data::Data()
+{
+}
+
+inline
+Exec_update_index::Exec_update_index(Exec_root* root) :
+ Exec_dml(root),
+ m_query(0)
+{
+}
+
+// children
+
+inline const Exec_update_index::Code&
+Exec_update_index::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_update_index::Data&
+Exec_update_index::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_update_index::setQuery(Exec_query* query)
+{
+ ctx_assert(query != 0 && m_query == 0);
+ m_query = query;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_update_lookup.cpp b/ndb/src/client/odbc/codegen/Code_update_lookup.cpp
new file mode 100644
index 00000000000..7525fb72692
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_update_lookup.cpp
@@ -0,0 +1,194 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_dml_column.hpp"
+#include "Code_expr.hpp"
+#include "Code_update_lookup.hpp"
+#include "Code_table.hpp"
+#include "Code_root.hpp"
+
+// Plan_update_lookup
+
+Plan_update_lookup::~Plan_update_lookup()
+{
+}
+
+Plan_base*
+Plan_update_lookup::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctl.m_dmlRow = m_dmlRow; // row type to convert to
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+void
+Plan_update_lookup::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "UPDATE WHERE", SQL_DIAG_UPDATE_WHERE);
+}
+
+Exec_base*
+Plan_update_lookup::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // generate code for the query
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // set up
+ ctx_assert(m_table != 0);
+ const BaseString& tableName = m_table->getName();
+ const DictTable& dictTable = m_table->dictTable();
+ const unsigned keyCount = dictTable.keyCount();
+ const ColumnVector& columns = m_table->dmlColumns();
+ ctx_assert(columns.size() > 0);
+ const unsigned attrCount = columns.size() - 1;
+ // create the code
+ Exec_update_lookup::Code& code = *new Exec_update_lookup::Code(keyCount);
+ code.m_tableName = strcpy(new char[tableName.length() + 1], tableName.c_str());
+ // key attributes
+ code.m_keyId = new NdbAttrId[1 + keyCount];
+ code.m_keyId[0] = (NdbAttrId)-1;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ const DictColumn* keyColumn = dictTable.getKey(k);
+ const SqlType& sqlType = keyColumn->sqlType();
+ SqlSpec sqlSpec(sqlType, SqlSpec::Physical);
+ code.m_keySpecs.setEntry(k, sqlSpec);
+ code.m_keyId[k] = keyColumn->getAttrId();
+ }
+ // matching expressions
+ const Plan_table::Index& index = m_table->m_indexList[0];
+ ctx_assert(index.m_keyFound);
+ const ExprVector& keyEq = index.m_keyEq;
+ ctx_assert(keyEq.size() == 1 + keyCount);
+ code.m_keyMatch = new Exec_expr* [1 + keyCount];
+ code.m_keyMatch[0] = 0;
+ for (unsigned k = 1; k <= keyCount; k++) {
+ Plan_expr* expr = keyEq[k];
+ Exec_expr* execExpr = static_cast<Exec_expr*>(expr->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execExpr != 0);
+ code.m_keyMatch[k] = execExpr;
+ }
+ // updated attributes
+ code.m_attrCount = attrCount;
+ code.m_attrId = new NdbAttrId[1 + attrCount];
+ code.m_attrId[0] = (NdbAttrId)-1;
+ for (unsigned i = 1; i <= attrCount; i++) {
+ Plan_column* column = columns[i];
+ ctx_assert(column != 0);
+ const DictColumn& dictColumn = column->dictColumn();
+ code.m_attrId[i] = dictColumn.getAttrId();
+ }
+ // create the exec
+ Exec_update_lookup* exec = new Exec_update_lookup(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ return exec;
+}
+
+void
+Plan_update_lookup::print(Ctx& ctx)
+{
+ ctx.print(" [update_lookup");
+ Plan_base* a[] = { m_table, m_query };
+ printList(ctx, a, sizeof(a)/sizeof(a[0]));
+ ctx.print("]");
+}
+
+// Exec_delete
+
+Exec_update_lookup::Code::~Code()
+{
+ delete[] m_tableName;
+ delete[] m_keyId;
+ delete[] m_keyMatch;
+ delete[] m_attrId;
+}
+
+Exec_update_lookup::Data::~Data()
+{
+}
+
+Exec_update_lookup::~Exec_update_lookup()
+{
+}
+
+void
+Exec_update_lookup::alloc(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ // allocate the subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // create data
+ Data& data = *new Data;
+ setData(data);
+ // allocate matching expressions
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* expr = code.m_keyMatch[k];
+ ctx_assert(expr != 0);
+ expr->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ }
+}
+
+void
+Exec_update_lookup::close(Ctx& ctx)
+{
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+}
+
+void
+Exec_update_lookup::print(Ctx& ctx)
+{
+ ctx.print(" [update_lookup");
+ if (m_code != 0) {
+ const Code& code = getCode();
+ ctx.print(" keyId=");
+ for (unsigned i = 1; i <= code.m_keyCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_keyId[i]);
+ }
+ ctx.print(" attrId=");
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_attrId[i]);
+ }
+ }
+ Exec_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_update_lookup.hpp b/ndb/src/client/odbc/codegen/Code_update_lookup.hpp
new file mode 100644
index 00000000000..fc4341880dd
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_update_lookup.hpp
@@ -0,0 +1,167 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_update_lookup_hpp
+#define ODBC_CODEGEN_Code_update_lookup_hpp
+
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_dml.hpp"
+#include "Code_table.hpp"
+#include "Code_query.hpp"
+
+/**
+ * @class Plan_update_lookup
+ * @brief Update in PlanTree
+ */
+class Plan_update_lookup : public Plan_dml {
+public:
+ Plan_update_lookup(Plan_root* root);
+ virtual ~Plan_update_lookup();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setTable(Plan_table* table);
+ void setDmlRow(Plan_dml_row* dmlRow);
+ void setQuery(Plan_query* query);
+protected:
+ Plan_table* m_table;
+ Plan_dml_row* m_dmlRow;
+ Plan_query* m_query;
+};
+
+inline
+Plan_update_lookup::Plan_update_lookup(Plan_root* root) :
+ Plan_dml(root),
+ m_table(0),
+ m_dmlRow(0),
+ m_query(0)
+{
+}
+
+inline void
+Plan_update_lookup::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+inline void
+Plan_update_lookup::setDmlRow(Plan_dml_row* dmlRow)
+{
+ ctx_assert(dmlRow != 0);
+ m_dmlRow = dmlRow;
+}
+
+inline void
+Plan_update_lookup::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+/**
+ * @class Exec_update_lookup
+ * @brief Insert in ExecTree
+ */
+class Exec_update_lookup : public Exec_dml {
+public:
+ class Code : public Exec_dml::Code {
+ public:
+ Code(unsigned keyCount);
+ virtual ~Code();
+ protected:
+ friend class Plan_update_lookup;
+ friend class Exec_update_lookup;
+ char* m_tableName;
+ unsigned m_keyCount;
+ SqlSpecs m_keySpecs; // key types
+ NdbAttrId* m_keyId;
+ Exec_expr** m_keyMatch; // XXX pointers for now
+ unsigned m_attrCount;
+ NdbAttrId* m_attrId;
+ };
+ class Data : public Exec_dml::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_update_lookup;
+ };
+ Exec_update_lookup(Exec_root* root);
+ virtual ~Exec_update_lookup();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+protected:
+ Exec_query* m_query;
+};
+
+inline
+Exec_update_lookup::Code::Code(unsigned keyCount) :
+ m_tableName(0),
+ m_keyCount(keyCount),
+ m_keySpecs(keyCount),
+ m_keyId(0),
+ m_keyMatch(0),
+ m_attrCount(0),
+ m_attrId(0)
+{
+}
+
+inline
+Exec_update_lookup::Data::Data()
+{
+}
+
+inline
+Exec_update_lookup::Exec_update_lookup(Exec_root* root) :
+ Exec_dml(root),
+ m_query(0)
+{
+}
+
+// children
+
+inline const Exec_update_lookup::Code&
+Exec_update_lookup::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_update_lookup::Data&
+Exec_update_lookup::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_update_lookup::setQuery(Exec_query* query)
+{
+ ctx_assert(query != 0 && m_query == 0);
+ m_query = query;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Code_update_scan.cpp b/ndb/src/client/odbc/codegen/Code_update_scan.cpp
new file mode 100644
index 00000000000..9fac1728469
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_update_scan.cpp
@@ -0,0 +1,146 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/StmtArea.hpp>
+#include <dictionary/DictTable.hpp>
+#include <dictionary/DictColumn.hpp>
+#include "Code_dml_column.hpp"
+#include "Code_update_scan.hpp"
+#include "Code_table.hpp"
+#include "Code_root.hpp"
+
+// Plan_update_scan
+
+Plan_update_scan::~Plan_update_scan()
+{
+}
+
+Plan_base*
+Plan_update_scan::analyze(Ctx& ctx, Ctl& ctl)
+{
+ ctl.m_dmlRow = m_dmlRow; // row type to convert to
+ ctx_assert(m_query != 0);
+ m_query->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(m_table != 0);
+ m_table->analyze(ctx, ctl);
+ if (! ctx.ok())
+ return 0;
+ return this;
+}
+
+void
+Plan_update_scan::describe(Ctx& ctx)
+{
+ stmtArea().setFunction(ctx, "UPDATE WHERE", SQL_DIAG_UPDATE_WHERE);
+}
+
+Exec_base*
+Plan_update_scan::codegen(Ctx& ctx, Ctl& ctl)
+{
+ // generate code for the query
+ ctx_assert(m_query != 0);
+ Exec_query* execQuery = static_cast<Exec_query*>(m_query->codegen(ctx, ctl));
+ if (! ctx.ok())
+ return 0;
+ ctx_assert(execQuery != 0);
+ // set up
+ ctx_assert(m_table != 0);
+ const ColumnVector& columns = m_table->dmlColumns();
+ ctx_assert(columns.size() > 0);
+ const unsigned attrCount = columns.size() - 1;
+ // create the code
+ Exec_update_scan::Code& code = *new Exec_update_scan::Code();
+ // updated attributes
+ code.m_attrCount = attrCount;
+ code.m_attrId = new NdbAttrId[1 + attrCount];
+ code.m_attrId[0] = (NdbAttrId)-1;
+ for (unsigned i = 1; i <= attrCount; i++) {
+ Plan_column* column = columns[i];
+ ctx_assert(column != 0);
+ const DictColumn& dictColumn = column->dictColumn();
+ code.m_attrId[i] = dictColumn.getAttrId();
+ }
+ // create the exec
+ Exec_update_scan* exec = new Exec_update_scan(ctl.m_execRoot);
+ ctl.m_execRoot->saveNode(exec);
+ exec->setCode(code);
+ exec->setQuery(execQuery);
+ return exec;
+}
+
+void
+Plan_update_scan::print(Ctx& ctx)
+{
+ ctx.print(" [update_scan");
+ Plan_base* a[] = { m_table, m_query };
+ printList(ctx, a, sizeof(a)/sizeof(a[0]));
+ ctx.print("]");
+}
+
+// Exec_delete
+
+Exec_update_scan::Code::~Code()
+{
+ delete[] m_attrId;
+}
+
+Exec_update_scan::Data::~Data()
+{
+}
+
+Exec_update_scan::~Exec_update_scan()
+{
+}
+
+void
+Exec_update_scan::alloc(Ctx& ctx, Ctl& ctl)
+{
+ // allocate the subquery
+ ctx_assert(m_query != 0);
+ m_query->alloc(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // create data
+ Data& data = *new Data;
+ setData(data);
+}
+
+void
+Exec_update_scan::close(Ctx& ctx)
+{
+ ctx_assert(m_query != 0);
+ m_query->close(ctx);
+}
+
+void
+Exec_update_scan::print(Ctx& ctx)
+{
+ ctx.print(" [update_scan");
+ if (m_code != 0) {
+ const Code& code = getCode();
+ ctx.print(" attrId=");
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ if (i > 1)
+ ctx.print(",");
+ ctx.print("%u", (unsigned)code.m_attrId[i]);
+ }
+ }
+ Exec_base* a[] = { m_query };
+ printList(ctx, a, 1);
+ ctx.print("]");
+}
diff --git a/ndb/src/client/odbc/codegen/Code_update_scan.hpp b/ndb/src/client/odbc/codegen/Code_update_scan.hpp
new file mode 100644
index 00000000000..d742883e561
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Code_update_scan.hpp
@@ -0,0 +1,160 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_Code_update_scan_hpp
+#define ODBC_CODEGEN_Code_update_scan_hpp
+
+#include <common/common.hpp>
+#include "Code_base.hpp"
+#include "Code_dml.hpp"
+#include "Code_table.hpp"
+#include "Code_pred.hpp"
+#include "Code_query.hpp"
+
+/**
+ * @class Plan_update_scan
+ * @brief Update in PlanTree
+ */
+class Plan_update_scan : public Plan_dml {
+public:
+ Plan_update_scan(Plan_root* root);
+ virtual ~Plan_update_scan();
+ Plan_base* analyze(Ctx& ctx, Ctl& ctl);
+ void describe(Ctx& ctx);
+ Exec_base* codegen(Ctx& ctx, Ctl& ctl);
+ void print(Ctx& ctx);
+ // children
+ void setTable(Plan_table* table);
+ void setDmlRow(Plan_dml_row* dmlRow);
+ void setQuery(Plan_query* query);
+protected:
+ Plan_table* m_table;
+ Plan_dml_row* m_dmlRow;
+ Plan_query* m_query;
+};
+
+inline
+Plan_update_scan::Plan_update_scan(Plan_root* root) :
+ Plan_dml(root),
+ m_table(0),
+ m_dmlRow(0),
+ m_query(0)
+{
+}
+
+// children
+
+inline void
+Plan_update_scan::setTable(Plan_table* table)
+{
+ ctx_assert(table != 0);
+ m_table = table;
+}
+
+inline void
+Plan_update_scan::setDmlRow(Plan_dml_row* dmlRow)
+{
+ ctx_assert(dmlRow != 0);
+ m_dmlRow = dmlRow;
+}
+
+inline void
+Plan_update_scan::setQuery(Plan_query* query)
+{
+ ctx_assert(query != 0);
+ m_query = query;
+}
+
+/**
+ * @class Exec_update_scan
+ * @brief Insert in ExecTree
+ */
+class Exec_update_scan : public Exec_dml {
+public:
+ class Code : public Exec_dml::Code {
+ public:
+ Code();
+ virtual ~Code();
+ protected:
+ friend class Plan_update_scan;
+ friend class Exec_update_scan;
+ unsigned m_attrCount;
+ NdbAttrId* m_attrId;
+ };
+ class Data : public Exec_dml::Data {
+ public:
+ Data();
+ virtual ~Data();
+ protected:
+ friend class Exec_update_scan;
+ };
+ Exec_update_scan(Exec_root* root);
+ virtual ~Exec_update_scan();
+ void alloc(Ctx& ctx, Ctl& ctl);
+ void execImpl(Ctx& ctx, Ctl& ctl);
+ void close(Ctx& ctx);
+ void print(Ctx& ctx);
+ // children
+ const Code& getCode() const;
+ Data& getData() const;
+ void setQuery(Exec_query* query);
+protected:
+ Exec_query* m_query;
+};
+
+inline
+Exec_update_scan::Code::Code() :
+ m_attrCount(0),
+ m_attrId(0)
+{
+}
+
+inline
+Exec_update_scan::Data::Data()
+{
+}
+
+inline
+Exec_update_scan::Exec_update_scan(Exec_root* root) :
+ Exec_dml(root),
+ m_query(0)
+{
+}
+
+// children
+
+inline const Exec_update_scan::Code&
+Exec_update_scan::getCode() const
+{
+ const Code* code = static_cast<const Code*>(m_code);
+ return *code;
+}
+
+inline Exec_update_scan::Data&
+Exec_update_scan::getData() const
+{
+ Data* data = static_cast<Data*>(m_data);
+ return *data;
+}
+
+inline void
+Exec_update_scan::setQuery(Exec_query* query)
+{
+ ctx_assert(query != 0 && m_query == 0);
+ m_query = query;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/Makefile b/ndb/src/client/odbc/codegen/Makefile
new file mode 100644
index 00000000000..49e5439556d
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/Makefile
@@ -0,0 +1,104 @@
+include .defs.mk
+
+TYPE = *
+
+NONPIC_ARCHIVE = N
+
+PIC_ARCHIVE = Y
+
+ARCHIVE_TARGET = odbccodegen
+
+SOURCES = \
+ SimpleScan.lpp \
+ SimpleGram.ypp \
+ SimpleParser.cpp \
+ CodeGen.cpp \
+ Code_base.cpp \
+ Code_root.cpp \
+ Code_stmt.cpp \
+ Code_query.cpp \
+ Code_dml.cpp \
+ Code_ddl.cpp \
+ Code_select.cpp \
+ Code_pred.cpp \
+ Code_pred_op.cpp \
+ Code_comp_op.cpp \
+ Code_query_project.cpp \
+ Code_query_filter.cpp \
+ Code_query_join.cpp \
+ Code_query_lookup.cpp \
+ Code_query_index.cpp \
+ Code_query_scan.cpp \
+ Code_query_range.cpp \
+ Code_query_sys.cpp \
+ Code_query_repeat.cpp \
+ Code_query_count.cpp \
+ Code_query_sort.cpp \
+ Code_query_group.cpp \
+ Code_query_distinct.cpp \
+ Code_expr_row.cpp \
+ Code_expr.cpp \
+ Code_expr_op.cpp \
+ Code_expr_func.cpp \
+ Code_expr_conv.cpp \
+ Code_expr_column.cpp \
+ Code_expr_const.cpp \
+ Code_expr_param.cpp \
+ Code_update.cpp \
+ Code_update_lookup.cpp \
+ Code_update_index.cpp \
+ Code_update_scan.cpp \
+ Code_set_row.cpp \
+ Code_insert.cpp \
+ Code_dml_row.cpp \
+ Code_dml_column.cpp \
+ Code_delete.cpp \
+ Code_delete_lookup.cpp \
+ Code_delete_index.cpp \
+ Code_delete_scan.cpp \
+ Code_column.cpp \
+ Code_table_list.cpp \
+ Code_table.cpp \
+ Code_create_table.cpp \
+ Code_create_index.cpp \
+ Code_create_row.cpp \
+ Code_ddl_row.cpp \
+ Code_ddl_column.cpp \
+ Code_ddl_constr.cpp \
+ Code_idx_column.cpp \
+ Code_data_type.cpp \
+ Code_drop_table.cpp \
+ Code_drop_index.cpp
+
+ifeq ($(NDB_OS),WIN32)
+CCFLAGS += -I$(call fixpath,.)
+
+_libs:: FlexLexer.h unistd.h
+
+endif
+
+include ../Extra.mk
+include $(NDB_TOP)/Epilogue.mk
+
+ifeq ($(NDB_OS),LINUX)
+FLEXHACK = perl -i -pe 's/\bisatty\b/ouencunbwdb2y1bdc/g'
+BISONHACK = perl -i -pe 's/^\s*__attribute__\s*\(\(.*\)\)//'
+endif
+
+ifeq ($(NDB_OS),MACOSX)
+FLEXHACK = perl -i -pe 's/\bisatty\b/ouencunbwdb2y1bdc/g'
+BISONHACK = perl -i -pe 's/^\s*__attribute__\s*\(\(.*\)\)//'
+endif
+
+ifeq ($(NDB_OS),SOLARIS)
+BISONHACK = perl -i -pe 's/^\s*__attribute__\s*\(\(.*\)\)//'
+endif
+
+ifeq ($(NDB_OS),WIN32)
+unistd.h:
+ touch unistd.h
+
+FlexLexer.h:
+ cp /usr/include/FlexLexer.h .
+
+endif
diff --git a/ndb/src/client/odbc/codegen/SimpleGram.ypp b/ndb/src/client/odbc/codegen/SimpleGram.ypp
new file mode 100644
index 00000000000..d49344849d2
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/SimpleGram.ypp
@@ -0,0 +1,1629 @@
+%{
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/DataRow.hpp>
+#include "CodeGen.hpp"
+#include <FlexLexer.h>
+#include "SimpleParser.hpp"
+
+/* redefine globals after headers */
+#define yyparse SimpleParser_yyparse
+#if YYDEBUG
+#define yydebug SimpleParser_yydebug
+#endif
+
+#define YYLEX_PARAM simpleParserPtr
+#define YYPARSE_PARAM simpleParserPtr
+#define simpleParser (*static_cast<SimpleParser*>(simpleParserPtr))
+
+static int yylex(YYSTYPE* lvalp, void* simpleParserPtr);
+
+#define yyerror(s) simpleParser.parseError(s)
+
+#if YYDEBUG
+// does not work in bison 1.75
+#undef stderr
+#define stderr 0
+#define YYFPRINTF simpleParser.ctx().print
+#endif
+
+// scanner states
+
+#define pushState(sc) simpleParser.pushState(sc)
+#define popState() simpleParser.popState()
+
+#define StateEval SimpleParser_stateEval
+#define StateType SimpleParser_stateType
+#define StatePhys SimpleParser_statePhys
+extern int SimpleParser_stateEval;
+extern int SimpleParser_stateType;
+extern int SimpleParser_statePhys;
+
+struct LimitPair { int off; int cnt; };
+
+struct PhysAttr { int storage; int logging; };
+
+%}
+
+%defines
+%pure-parser
+%verbose
+
+%union {
+ Plan_root* m_root;
+ Plan_stmt* m_stmt;
+ Plan_select* m_select;
+ Insert_op m_insert_op;
+ Plan_insert* m_insert;
+ Plan_update* m_update;
+ Plan_delete* m_delete;
+ Plan_create_table* m_create_table;
+ Plan_create_index* m_create_index;
+ NdbDictionary::Object::Type m_index_type;
+ Plan_create_row* m_create_row;
+ Plan_ddl_row* m_ddl_row;
+ Plan_ddl_column* m_ddl_column;
+ Plan_ddl_constr* m_ddl_constr;
+ Plan_idx_column* m_idx_column;
+ Plan_data_type* m_data_type;
+ Plan_drop_table* m_drop_table;
+ Plan_drop_index* m_drop_index;
+ Plan_set_row* m_set_row;
+ Plan_expr_row* m_expr_row;
+ bool m_asc_desc;
+ Plan_pred* m_pred;
+ Pred_op::Opcode m_pred_opcode;
+ Comp_op::Opcode m_comp_opcode;
+ Plan_expr* m_expr;
+ Expr_op::Opcode m_expr_opcode;
+ Plan_dml_row* m_dml_row;
+ Plan_dml_column* m_dml_column;
+ Plan_table* m_table;
+ Plan_table_list* m_table_list;
+ const char* m_string;
+ struct LimitPair* m_limit;
+ int m_signed_integer;
+ bool m_distinct;
+ struct PhysAttr* m_phys_attr;
+ NdbDictionary::Object::FragmentType m_storage_attr;
+ bool m_logging_attr;
+}
+
+/* keywords */
+%token
+ T_AND
+ T_ASC
+ T_AUTO_INCREMENT
+ T_BIGINT
+ T_BINARY
+ T_BLOB
+ T_BY
+ T_CHAR
+ T_CONSTRAINT
+ T_CREATE
+ T_DATETIME
+ T_DEFAULT
+ T_DELETE
+ T_DESC
+ T_DISTINCT
+ T_DOUBLE
+ T_DROP
+ T_FLOAT
+ T_FOREIGN
+ T_FROM
+ T_GROUP
+ T_HASH
+ T_HAVING
+ T_IN
+ T_INDEX
+ T_INSERT
+ T_INT
+ T_INTEGER
+ T_INTO
+ T_IS
+ T_KEY
+ T_LARGE
+ T_LIKE
+ T_LIMIT
+ T_LOGGING
+ T_LONGBLOB
+ T_MEDIUM
+ T_NOLOGGING
+ T_NOT
+ T_NULL
+ T_OFFSET
+ T_ON
+ T_OR
+ T_ORDER
+ T_PRECISION
+ T_PRIMARY
+ T_REAL
+ T_REFERENCES
+ T_ROWNUM
+ T_SELECT
+ T_SET
+ T_SINGLE
+ T_SMALL
+ T_SMALLINT
+ T_STORAGE
+ T_SYSDATE
+ T_TABLE
+ T_UNIQUE
+ T_UNSIGNED
+ T_UPDATE
+ T_VALUES
+ T_VARBINARY
+ T_VARCHAR
+ T_WHERE
+ T_WRITE
+
+/* identifiers and constants */
+%token
+ <m_string> T_IDENTIFIER
+ <m_string> T_LINTEGER
+ <m_string> T_LDECIMAL
+ <m_string> T_LREAL
+ <m_string> T_STRING
+
+/* expressions and predicates */
+%token
+ T_PLUS
+ T_MINUS
+ T_TIMES
+ T_DIVIDE
+ T_EQ
+ T_NOTEQ
+ T_LT
+ T_LTEQ
+ T_GT
+ T_GTEQ
+ T_QUES
+
+/* common special symbols */
+%token
+ T_PERIOD
+ T_COMMA
+ T_PARENLEFT
+ T_PARENRIGHT
+ T_ASTERISK
+ T_ASSIGN
+
+%type <m_root> root
+%type <m_stmt> stmt
+%type <m_select> stmt_select
+%type <m_insert> stmt_insert
+%type <m_insert_op> insert_op
+%type <m_update> stmt_update
+%type <m_delete> stmt_delete
+%type <m_create_table> create_table
+%type <m_create_index> create_index
+%type <m_index_type> index_type
+%type <m_create_row> create_row
+%type <m_ddl_column> create_column
+%type <m_ddl_constr> create_constr
+%type <m_idx_column> idx_column
+%type <m_data_type> data_type
+%type <m_ddl_row> ddl_row
+%type <m_ddl_column> ddl_column
+%type <m_drop_table> drop_table
+%type <m_drop_index> drop_index
+%type <m_asc_desc> asc_desc
+%type <m_set_row> set_row
+%type <m_expr_row> expr_row
+%type <m_expr_row> sort_row
+%type <m_pred> where_clause
+%type <m_expr_row> order_clause
+%type <m_pred> pred
+%type <m_pred> pred1
+%type <m_pred_opcode> pred1_op
+%type <m_pred> pred2
+%type <m_pred_opcode> pred2_op
+%type <m_pred> pred3
+%type <m_pred_opcode> pred3_op
+%type <m_pred> pred4
+%type <m_comp_opcode> comp_op
+%type <m_expr> expr
+%type <m_expr> expr1
+%type <m_expr_opcode> expr1_op
+%type <m_expr> expr2
+%type <m_expr_opcode> expr2_op
+%type <m_expr> expr3
+%type <m_expr_opcode> expr3_op
+%type <m_expr> expr4
+%type <m_expr> expr_column
+%type <m_dml_row> dml_row
+%type <m_dml_column> dml_column
+%type <m_expr_row> value_row
+%type <m_expr> value_expr
+%type <m_table> table
+%type <m_table_list> table_list
+%type <m_string> dot_identifier
+%type <m_limit> limit_clause
+%type <m_signed_integer> signed_integer
+%type <m_distinct> distinct_clause
+%type <m_expr_row> group_clause
+%type <m_pred> having_clause
+%type <m_phys_attr> phys_attr
+%type <m_phys_attr> phys_attr2
+%type <m_storage_attr> storage_attr
+%type <m_logging_attr> logging_attr
+
+%%
+
+root:
+ stmt
+ {
+ Plan_root* root = simpleParser.root();
+ root->setStmt($1);
+ }
+ ;
+stmt:
+ stmt_select
+ {
+ $$ = $1;
+ }
+ |
+ stmt_insert
+ {
+ $$ = $1;
+ }
+ |
+ stmt_update
+ {
+ $$ = $1;
+ }
+ |
+ stmt_delete
+ {
+ $$ = $1;
+ }
+ |
+ create_table
+ {
+ $$ = $1;
+ }
+ |
+ create_index
+ {
+ $$ = $1;
+ }
+ |
+ drop_table
+ {
+ $$ = $1;
+ }
+ |
+ drop_index
+ {
+ $$ = $1;
+ }
+ ;
+stmt_select:
+ T_SELECT distinct_clause expr_row T_FROM table_list where_clause group_clause having_clause order_clause limit_clause
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_select* stmt = new Plan_select(root);
+ root->saveNode(stmt);
+ stmt->setDistinct($2);
+ stmt->setRow($3);
+ stmt->setList($5);
+ if ($6 != 0)
+ stmt->setPred($6);
+ if ($7 != 0)
+ stmt->setGroup($7);
+ if ($8 != 0)
+ stmt->setHaving($8);
+ if ($9 != 0)
+ stmt->setSort($9);
+ if ($10 != 0) {
+ stmt->setLimit($10->off, $10->cnt);
+ delete $10;
+ }
+ $$ = stmt;
+ }
+ ;
+stmt_insert:
+ insert_op T_INTO table T_VALUES T_PARENLEFT value_row T_PARENRIGHT
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_insert* stmt = new Plan_insert(root, $1);
+ root->saveNode(stmt);
+ stmt->setTable($3);
+ stmt->setExprRow($6);
+ $$ = stmt;
+ }
+ |
+ insert_op T_INTO table T_PARENLEFT dml_row T_PARENRIGHT T_VALUES T_PARENLEFT value_row T_PARENRIGHT
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_insert* stmt = new Plan_insert(root, $1);
+ root->saveNode(stmt);
+ stmt->setTable($3);
+ stmt->setDmlRow($5);
+ stmt->setExprRow($9);
+ $$ = stmt;
+ }
+ |
+ insert_op T_INTO table stmt_select
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_insert* stmt = new Plan_insert(root, $1);
+ root->saveNode(stmt);
+ stmt->setTable($3);
+ stmt->setSelect($4);
+ $$ = stmt;
+ }
+ |
+ insert_op T_INTO table T_PARENLEFT dml_row T_PARENRIGHT stmt_select
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_insert* stmt = new Plan_insert(root, $1);
+ root->saveNode(stmt);
+ stmt->setTable($3);
+ stmt->setDmlRow($5);
+ stmt->setSelect($7);
+ $$ = stmt;
+ }
+ |
+ insert_op T_INTO table T_SET set_row
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_insert* stmt = new Plan_insert(root, $1);
+ root->saveNode(stmt);
+ stmt->setTable($3);
+ stmt->setMysqlRow($5);
+ $$ = stmt;
+ }
+ ;
+insert_op:
+ T_INSERT
+ {
+ $$ = Insert_op_insert;
+ }
+ |
+ T_WRITE
+ {
+ $$ = Insert_op_write;
+ }
+ ;
+stmt_update:
+ T_UPDATE table T_SET set_row where_clause
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_update* stmt = new Plan_update(root);
+ root->saveNode(stmt);
+ stmt->setTable($2);
+ stmt->setRow($4);
+ if ($5 != 0)
+ stmt->setPred($5);
+ $$ = stmt;
+ }
+ ;
+stmt_delete:
+ T_DELETE T_FROM table where_clause
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_delete* stmt = new Plan_delete(root);
+ root->saveNode(stmt);
+ stmt->setTable($3);
+ if ($4 != 0)
+ stmt->setPred($4);
+ $$ = stmt;
+ }
+ ;
+create_table:
+ T_CREATE T_TABLE dot_identifier T_PARENLEFT create_row T_PARENRIGHT phys_attr
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_create_table* stmt = new Plan_create_table(root, $3);
+ root->saveNode(stmt);
+ delete[] $3;
+ stmt->setCreateRow($5);
+ if ($7->storage != -1)
+ stmt->setFragmentType((NdbDictionary::Object::FragmentType)$7->storage);
+ if ($7->logging != -1)
+ stmt->setLogging($7->logging);
+ delete $7;
+ $$ = stmt;
+ }
+ ;
+create_row:
+ create_column
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_create_row* createRow = new Plan_create_row(root);
+ root->saveNode(createRow);
+ createRow->addColumn($1);
+ $$ = createRow;
+ }
+ |
+ create_constr
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_create_row* createRow = new Plan_create_row(root);
+ root->saveNode(createRow);
+ createRow->addConstr($1);
+ $$ = createRow;
+ }
+ |
+ create_row T_COMMA create_column
+ {
+ Plan_create_row* createRow = $1;
+ createRow->addColumn($3);
+ $$ = createRow;
+ }
+ |
+ create_row T_COMMA create_constr
+ {
+ Plan_create_row* createRow = $1;
+ createRow->addConstr($3);
+ $$ = createRow;
+ }
+ |
+ create_row T_COMMA create_ignore
+ {
+ $$ = $1;
+ }
+ ;
+create_column:
+ T_IDENTIFIER { pushState(StateType); } data_type { popState(); }
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_ddl_column* ddlColumn = new Plan_ddl_column(root, $1);
+ root->saveNode(ddlColumn);
+ delete[] $1;
+ ddlColumn->setType($3);
+ simpleParser.curr(ddlColumn);
+ }
+ create_column_rest
+ {
+ $$ = simpleParser.curr((Plan_ddl_column*)0);
+ }
+ ;
+data_type:
+ T_CHAR T_PARENLEFT T_LINTEGER T_PARENRIGHT dummy_binary
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(simpleParser.ctx(), SqlType::Char, atoi($3), true);
+ delete[] $3;
+ if (! simpleParser.ctx().ok()) {
+ YYABORT;
+ }
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_BINARY T_PARENLEFT T_LINTEGER T_PARENRIGHT
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(simpleParser.ctx(), SqlType::Binary, atoi($3), true);
+ delete[] $3;
+ if (! simpleParser.ctx().ok()) {
+ YYABORT;
+ }
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_VARCHAR T_PARENLEFT T_LINTEGER T_PARENRIGHT dummy_binary
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(simpleParser.ctx(), SqlType::Varchar, atoi($3), true);
+ delete[] $3;
+ if (! simpleParser.ctx().ok()) {
+ YYABORT;
+ }
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_VARBINARY T_PARENLEFT T_LINTEGER T_PARENRIGHT
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(simpleParser.ctx(), SqlType::Varbinary, atoi($3), true);
+ delete[] $3;
+ if (! simpleParser.ctx().ok()) {
+ YYABORT;
+ }
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_SMALLINT
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(SqlType::Smallint, true);
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_INTEGER
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(SqlType::Integer, true);
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_INT
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(SqlType::Integer, true);
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_BIGINT
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(SqlType::Bigint, true);
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_REAL
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(SqlType::Real, true);
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_FLOAT
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(SqlType::Double, true);
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_DOUBLE T_PRECISION
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(SqlType::Double, true);
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ T_DATETIME
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(SqlType::Datetime, true);
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ |
+ blob_keyword
+ {
+ Plan_root* root = simpleParser.root();
+ SqlType sqlType(SqlType::Blob, true);
+ Plan_data_type* dataType = new Plan_data_type(root, sqlType);
+ root->saveNode(dataType);
+ $$ = dataType;
+ }
+ ;
+dummy_binary:
+ /* empty */
+ |
+ T_BINARY
+ ;
+blob_keyword:
+ T_BLOB
+ |
+ T_LONGBLOB
+ ;
+create_column_rest:
+ /* empty */
+ |
+ data_constr_list
+ ;
+data_constr_list:
+ data_constr
+ |
+ data_constr_list data_constr
+ ;
+data_constr:
+ T_NULL
+ |
+ T_NOT T_NULL
+ {
+ Plan_ddl_column* ddlColumn = simpleParser.curr((Plan_ddl_column*)0);
+ ddlColumn->setNotNull();
+ }
+ |
+ T_UNSIGNED
+ {
+ Plan_ddl_column* ddlColumn = simpleParser.curr((Plan_ddl_column*)0);
+ ddlColumn->setUnSigned();
+ }
+ |
+ T_PRIMARY T_KEY
+ {
+ Plan_ddl_column* ddlColumn = simpleParser.curr((Plan_ddl_column*)0);
+ ddlColumn->setPrimaryKey();
+ }
+ |
+ T_AUTO_INCREMENT
+ {
+ Plan_ddl_column* ddlColumn = simpleParser.curr((Plan_ddl_column*)0);
+ ddlColumn->setAutoIncrement();
+ }
+ |
+ T_DEFAULT expr
+ {
+ Plan_ddl_column* ddlColumn = simpleParser.curr((Plan_ddl_column*)0);
+ ddlColumn->setDefaultValue($2);
+ }
+ ;
+create_constr:
+ T_PRIMARY T_KEY T_PARENLEFT ddl_row T_PARENRIGHT
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_ddl_constr* ddlConstr = new Plan_ddl_constr(root);
+ root->saveNode(ddlConstr);
+ ddlConstr->setRow($4);
+ $$ = ddlConstr;
+ }
+ |
+ T_CONSTRAINT dot_identifier T_PRIMARY T_KEY T_PARENLEFT ddl_row T_PARENRIGHT
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_ddl_constr* ddlConstr = new Plan_ddl_constr(root);
+ root->saveNode(ddlConstr);
+ ddlConstr->setRow($6);
+ $$ = ddlConstr;
+ }
+ ;
+create_ignore:
+ T_INDEX dot_identifier T_PARENLEFT ddl_row T_PARENRIGHT
+ |
+ T_FOREIGN T_KEY T_PARENLEFT ddl_row T_PARENRIGHT T_REFERENCES dot_identifier T_PARENLEFT ddl_row T_PARENRIGHT
+ ;
+ddl_row:
+ ddl_column
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_ddl_row* ddlRow = new Plan_ddl_row(root);
+ root->saveNode(ddlRow);
+ ddlRow->addColumn($1);
+ $$ = ddlRow;
+ }
+ |
+ ddl_row T_COMMA ddl_column
+ {
+ Plan_ddl_row* ddlRow = $1;
+ ddlRow->addColumn($3);
+ $$ = ddlRow;
+ }
+ ;
+ddl_column:
+ T_IDENTIFIER
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_ddl_column* column = new Plan_ddl_column(root, $1);
+ root->saveNode(column);
+ delete[] $1;
+ $$ = column;
+ }
+ ;
+create_index:
+ T_CREATE index_type T_INDEX dot_identifier T_ON table
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_create_index* stmt = new Plan_create_index(root, $4);
+ root->saveNode(stmt);
+ delete[] $4;
+ stmt->setType($2);
+ stmt->setTable($6);
+ simpleParser.curr(stmt);
+ }
+ T_PARENLEFT idx_row T_PARENRIGHT phys_attr
+ {
+ $$ = simpleParser.curr((Plan_create_index*)0);
+ if ($11->storage != -1)
+ $$->setFragmentType((NdbDictionary::Object::FragmentType)$11->storage);
+ if ($11->logging != -1)
+ $$->setLogging($11->logging);
+ delete $11;
+ }
+ ;
+index_type:
+ T_HASH
+ {
+ $$ = NdbDictionary::Object::HashIndex;
+ }
+ |
+ T_UNIQUE T_HASH
+ {
+ $$ = NdbDictionary::Object::UniqueHashIndex;
+ }
+ |
+ /* empty */
+ {
+ $$ = NdbDictionary::Object::OrderedIndex;
+ }
+ |
+ T_UNIQUE
+ {
+ $$ = NdbDictionary::Object::UniqueOrderedIndex;
+ }
+ ;
+idx_row:
+ idx_column
+ {
+ }
+ |
+ idx_row T_COMMA idx_column
+ {
+ }
+ ;
+idx_column:
+ T_IDENTIFIER
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_idx_column* column = new Plan_idx_column(root, $1);
+ root->saveNode(column);
+ delete[] $1;
+ Plan_create_index* stmt = simpleParser.curr((Plan_create_index*)0);
+ stmt->addColumn(column);
+ }
+ ;
+phys_attr:
+ { pushState(StatePhys); } phys_attr2 { popState(); }
+ {
+ $$ = $2;
+ }
+ ;
+phys_attr2:
+ /* empty */
+ {
+ $$ = new PhysAttr();
+ $$->storage = $$->logging = -1;
+ }
+ |
+ phys_attr2 storage_attr
+ {
+ if ($1->storage != -1 && $1->storage != $2) {
+ simpleParser.ctx().pushStatus(Sqlstate::_42000, Error::Gen, "conflicting STORAGE clauses");
+ YYABORT;
+ }
+ $$->storage = $2;
+ }
+ |
+ phys_attr2 logging_attr
+ {
+ if ($1->logging != -1 && $1->logging != $2) {
+ simpleParser.ctx().pushStatus(Sqlstate::_42000, Error::Gen, "conflicting LOGGING clauses");
+ YYABORT;
+ }
+ $$->logging = $2;
+ }
+ ;
+logging_attr:
+ T_LOGGING
+ {
+ $$ = true;
+ }
+ |
+ T_NOLOGGING
+ {
+ $$ = false;
+ }
+ ;
+storage_attr:
+ T_STORAGE T_PARENLEFT T_SINGLE T_PARENRIGHT
+ {
+ $$ = NdbDictionary::Object::FragSingle;
+ }
+ |
+ T_STORAGE T_PARENLEFT T_SMALL T_PARENRIGHT
+ {
+ $$ = NdbDictionary::Object::FragAllSmall;
+ }
+ |
+ T_STORAGE T_PARENLEFT T_MEDIUM T_PARENRIGHT
+ {
+ $$ = NdbDictionary::Object::FragAllMedium;
+ }
+ |
+ T_STORAGE T_PARENLEFT T_LARGE T_PARENRIGHT
+ {
+ $$ = NdbDictionary::Object::FragAllLarge;
+ }
+ ;
+drop_table:
+ T_DROP T_TABLE dot_identifier
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_drop_table* stmt = new Plan_drop_table(root, $3);
+ root->saveNode(stmt);
+ delete[] $3;
+ $$ = stmt;
+ }
+ ;
+drop_index:
+ T_DROP T_INDEX dot_identifier
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_drop_index* stmt = new Plan_drop_index(root, $3);
+ root->saveNode(stmt);
+ delete[] $3;
+ $$ = stmt;
+ }
+ |
+ T_DROP T_INDEX dot_identifier T_ON dot_identifier
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_drop_index* stmt = new Plan_drop_index(root, $3, $5);
+ root->saveNode(stmt);
+ delete[] $3;
+ delete[] $5;
+ $$ = stmt;
+ }
+ ;
+distinct_clause:
+ /* empty */
+ {
+ $$ = false;
+ }
+ |
+ T_DISTINCT
+ {
+ $$ = true;
+ }
+ ;
+where_clause:
+ /* empty */
+ {
+ $$ = 0;
+ }
+ |
+ T_WHERE pred
+ {
+ $$ = $2;
+ }
+ ;
+group_clause:
+ /* empty */
+ {
+ $$ = 0;
+ }
+ |
+ T_GROUP T_BY value_row
+ {
+ $$ = $3;
+ }
+ ;
+having_clause:
+ /* empty */
+ {
+ $$ = 0;
+ }
+ |
+ T_HAVING pred
+ {
+ $$ = $2;
+ }
+ ;
+order_clause:
+ /* empty */
+ {
+ $$ = 0;
+ }
+ |
+ T_ORDER T_BY sort_row
+ {
+ $$ = $3;
+ }
+ ;
+limit_clause:
+ /* empty */
+ {
+ $$ = 0;
+ }
+ |
+ T_LIMIT signed_integer
+ {
+ LimitPair* p = new LimitPair;
+ p->off = 0;
+ p->cnt = $2;
+ $$ = p;
+ }
+ |
+ T_LIMIT signed_integer T_COMMA signed_integer
+ {
+ LimitPair* p = new LimitPair;
+ p->off = $2,
+ p->cnt = $4;
+ $$ = p;
+ }
+ |
+ T_LIMIT signed_integer T_OFFSET signed_integer
+ {
+ LimitPair* p = new LimitPair;
+ p->off = $4;
+ p->cnt = $2;
+ $$ = p;
+ }
+ ;
+signed_integer:
+ T_LINTEGER
+ {
+ $$ = atoi($1);
+ delete[] $1;
+ }
+ |
+ T_MINUS T_LINTEGER
+ {
+ $$ = (-1) * atoi($2);
+ delete[] $2;
+ }
+ ;
+set_row:
+ dml_column T_ASSIGN expr
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_set_row* row = new Plan_set_row(root);
+ root->saveNode(row);
+ row->addColumn($1);
+ row->addExpr($3);
+ $$ = row;
+ }
+ |
+ set_row T_COMMA dml_column T_ASSIGN expr
+ {
+ Plan_set_row* row = $1;
+ row->addColumn($3);
+ row->addExpr($5);
+ $$ = row;
+ }
+ ;
+dml_row:
+ dml_column
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_dml_row* row = new Plan_dml_row(root);
+ root->saveNode(row);
+ row->addColumn($1);
+ $$ = row;
+ }
+ |
+ dml_row T_COMMA dml_column
+ {
+ Plan_dml_row* row = $1;
+ row->addColumn($3);
+ $$ = row;
+ }
+ ;
+dml_column:
+ T_IDENTIFIER
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_dml_column* column = new Plan_dml_column(root, $1);
+ root->saveNode(column);
+ delete[] $1;
+ $$ = column;
+ }
+ ;
+value_row:
+ value_expr
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_expr_row* row = new Plan_expr_row(root);
+ root->saveNode(row);
+ row->addExpr($1);
+ $$ = row;
+ }
+ |
+ value_row T_COMMA value_expr
+ {
+ Plan_expr_row* row = $1;
+ row->addExpr($3);
+ $$ = row;
+ }
+ ;
+value_expr:
+ expr
+ {
+ $$ = $1;
+ }
+ ;
+sort_row:
+ expr asc_desc
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_expr_row* row = new Plan_expr_row(root);
+ root->saveNode(row);
+ row->addExpr($1, $2);
+ $$ = row;
+ }
+ |
+ sort_row T_COMMA expr asc_desc
+ {
+ Plan_expr_row* row = $1;
+ row->addExpr($3, $4);
+ $$ = row;
+ }
+ ;
+asc_desc:
+ /* empty */
+ {
+ $$ = true;
+ }
+ |
+ T_ASC
+ {
+ $$ = true;
+ }
+ |
+ T_DESC
+ {
+ $$ = false;
+ }
+ ;
+expr_row:
+ T_ASTERISK
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_expr_row* row = new Plan_expr_row(root);
+ root->saveNode(row);
+ row->setAsterisk();
+ $$ = row;
+ }
+ |
+ T_TIMES /* XXX fix scanner state */
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_expr_row* row = new Plan_expr_row(root);
+ root->saveNode(row);
+ row->setAsterisk();
+ $$ = row;
+ }
+ |
+ expr
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_expr_row* row = new Plan_expr_row(root);
+ root->saveNode(row);
+ row->addExpr($1);
+ $$ = row;
+ }
+ |
+ expr T_IDENTIFIER
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_expr_row* row = new Plan_expr_row(root);
+ root->saveNode(row);
+ row->addExpr($1, BaseString($2));
+ $$ = row;
+ }
+ |
+ expr_row T_COMMA expr
+ {
+ Plan_expr_row* row = $1;
+ row->addExpr($3);
+ $$ = row;
+ }
+ |
+ expr_row T_COMMA expr T_IDENTIFIER
+ {
+ Plan_expr_row* row = $1;
+ row->addExpr($3, BaseString($4));
+ $$ = row;
+ }
+ ;
+pred:
+ { pushState(StateEval); } pred1 { popState(); }
+ {
+ $$ = $2;
+ }
+ ;
+pred1:
+ pred2
+ {
+ $$ = $1;
+ }
+ |
+ pred1 pred1_op pred2
+ {
+ Plan_root* root = simpleParser.root();
+ Pred_op op($2);
+ Plan_pred_op* pred = new Plan_pred_op(root, op);
+ root->saveNode(pred);
+ pred->setPred(1, $1);
+ pred->setPred(2, $3);
+ $$ = pred;
+ }
+ ;
+pred1_op:
+ T_OR
+ {
+ $$ = Pred_op::Or;
+ }
+ ;
+pred2:
+ pred3
+ {
+ $$ = $1;
+ }
+ |
+ pred2 pred2_op pred3
+ {
+ Plan_root* root = simpleParser.root();
+ Pred_op op($2);
+ Plan_pred_op* pred = new Plan_pred_op(root, op);
+ root->saveNode(pred);
+ pred->setPred(1, $1);
+ pred->setPred(2, $3);
+ $$ = pred;
+ }
+ ;
+pred2_op:
+ T_AND
+ {
+ $$ = Pred_op::And;
+ }
+ ;
+pred3:
+ pred4
+ {
+ $$ = $1;
+ }
+ |
+ pred3_op pred3
+ {
+ Plan_root* root = simpleParser.root();
+ Pred_op op($1);
+ Plan_pred_op* pred = new Plan_pred_op(root, op);
+ root->saveNode(pred);
+ pred->setPred(1, $2);
+ $$ = pred;
+ }
+ ;
+pred3_op:
+ T_NOT
+ {
+ $$ = Pred_op::Not;
+ }
+ ;
+pred4:
+ T_PARENLEFT pred1 T_PARENRIGHT
+ {
+ $$ = $2;
+ }
+ |
+ expr1 comp_op expr1
+ {
+ Plan_root* root = simpleParser.root();
+ Comp_op op($2);
+ Plan_comp_op* comp = new Plan_comp_op(root, op);
+ root->saveNode(comp);
+ comp->setExpr(1, $1);
+ comp->setExpr(2, $3);
+ $$ = comp;
+ }
+ |
+ expr1 T_IS T_NULL
+ {
+ Plan_root* root = simpleParser.root();
+ Comp_op op(Comp_op::Isnull);
+ Plan_comp_op* comp = new Plan_comp_op(root, op);
+ root->saveNode(comp);
+ comp->setExpr(1, $1);
+ $$ = comp;
+ }
+ |
+ expr1 T_IS T_NOT T_NULL
+ {
+ Plan_root* root = simpleParser.root();
+ Comp_op op(Comp_op::Isnotnull);
+ Plan_comp_op* comp = new Plan_comp_op(root, op);
+ root->saveNode(comp);
+ comp->setExpr(1, $1);
+ $$ = comp;
+ }
+ |
+ expr1 T_IN T_PARENLEFT value_row T_PARENRIGHT
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_pred* predOut = 0; // hack directly into Or of Eq
+ Plan_expr* exprLeft = $1;
+ Plan_expr_row* row = $4;
+ for (unsigned i = row->getSize(); i >= 1; i--) {
+ Plan_expr* exprRight = row->getExpr(i);
+ Plan_comp_op* comp = new Plan_comp_op(root, Comp_op::Eq);
+ root->saveNode(comp);
+ comp->setExpr(1, exprLeft);
+ comp->setExpr(2, exprRight);
+ if (predOut == 0) {
+ predOut = comp;
+ } else {
+ Plan_pred_op* pred = new Plan_pred_op(root, Pred_op::Or);
+ root->saveNode(pred);
+ pred->setPred(1, predOut);
+ pred->setPred(2, comp);
+ predOut = pred;
+ }
+ }
+ $$ = predOut;
+ }
+ |
+ expr1 T_NOT T_IN T_PARENLEFT value_row T_PARENRIGHT
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_pred* predOut = 0; // hack directly into And of Noteq
+ Plan_expr* exprLeft = $1;
+ Plan_expr_row* row = $5;
+ for (unsigned i = row->getSize(); i >= 1; i--) {
+ Plan_expr* exprRight = row->getExpr(i);
+ Plan_comp_op* comp = new Plan_comp_op(root, Comp_op::Noteq);
+ root->saveNode(comp);
+ comp->setExpr(1, exprLeft);
+ comp->setExpr(2, exprRight);
+ if (predOut == 0) {
+ predOut = comp;
+ } else {
+ Plan_pred_op* pred = new Plan_pred_op(root, Pred_op::And);
+ root->saveNode(pred);
+ pred->setPred(1, predOut);
+ pred->setPred(2, comp);
+ predOut = pred;
+ }
+ }
+ $$ = predOut;
+ }
+ ;
+comp_op:
+ T_EQ
+ {
+ $$ = Comp_op::Eq;
+ }
+ |
+ T_NOTEQ
+ {
+ $$ = Comp_op::Noteq;
+ }
+ |
+ T_LT
+ {
+ $$ = Comp_op::Lt;
+ }
+ |
+ T_LTEQ
+ {
+ $$ = Comp_op::Lteq;
+ }
+ |
+ T_GT
+ {
+ $$ = Comp_op::Gt;
+ }
+ |
+ T_GTEQ
+ {
+ $$ = Comp_op::Gteq;
+ }
+ |
+ T_LIKE
+ {
+ $$ = Comp_op::Like;
+ }
+ |
+ T_NOT T_LIKE
+ {
+ $$ = Comp_op::Notlike;
+ }
+ ;
+expr:
+ { pushState(StateEval); } expr1 { popState(); }
+ {
+ $$ = $2;
+ }
+ ;
+expr1:
+ expr2
+ {
+ $$ = $1;
+ }
+ |
+ expr1 expr1_op expr2
+ {
+ Plan_root* root = simpleParser.root();
+ Expr_op op($2);
+ Plan_expr_op* expr = new Plan_expr_op(root, op);
+ root->saveNode(expr);
+ expr->setExpr(1, $1);
+ expr->setExpr(2, $3);
+ $$ = expr;
+ }
+ ;
+expr1_op:
+ T_PLUS
+ {
+ $$ = Expr_op::Add;
+ }
+ |
+ T_MINUS
+ {
+ $$ = Expr_op::Subtract;
+ }
+ ;
+expr2:
+ expr3
+ {
+ $$ = $1;
+ }
+ |
+ expr2 expr2_op expr3
+ {
+ Plan_root* root = simpleParser.root();
+ Expr_op op($2);
+ Plan_expr_op* expr = new Plan_expr_op(root, op);
+ root->saveNode(expr);
+ expr->setExpr(1, $1);
+ expr->setExpr(2, $3);
+ $$ = expr;
+ }
+ ;
+expr2_op:
+ T_TIMES
+ {
+ $$ = Expr_op::Multiply;
+ }
+ |
+ T_DIVIDE
+ {
+ $$ = Expr_op::Divide;
+ }
+ ;
+expr3:
+ expr4
+ {
+ $$ = $1;
+ }
+ |
+ expr3_op expr3
+ {
+ Plan_root* root = simpleParser.root();
+ Expr_op op($1);
+ Plan_expr_op* expr = new Plan_expr_op(root, op);
+ root->saveNode(expr);
+ expr->setExpr(1, $2);
+ $$ = expr;
+ }
+ ;
+expr3_op:
+ T_PLUS
+ {
+ $$ = Expr_op::Plus;
+ }
+ |
+ T_MINUS
+ {
+ $$ = Expr_op::Minus;
+ }
+ ;
+expr4:
+ T_PARENLEFT expr1 T_PARENRIGHT
+ {
+ $$ = $2;
+ }
+ |
+ T_IDENTIFIER T_PARENLEFT expr_row T_PARENRIGHT
+ {
+ Plan_root* root = simpleParser.root();
+ const Expr_func& spec = Expr_func::find($1);
+ if (spec.m_name == 0) {
+ simpleParser.ctx().pushStatus(Sqlstate::_42000, Error::Gen, "unknown function %s", $1);
+ delete[] $1;
+ YYABORT;
+ }
+ Plan_expr_func* func = new Plan_expr_func(root, spec);
+ root->saveNode(func);
+ delete[] $1;
+ func->setArgs($3);
+ $$ = func;
+ }
+ |
+ T_ROWNUM
+ {
+ Plan_root* root = simpleParser.root();
+ const Expr_func& spec = Expr_func::find("ROWNUM");
+ ctx_assert(spec.m_name != 0);
+ Plan_expr_func* func = new Plan_expr_func(root, spec);
+ root->saveNode(func);
+ func->setArgs(0);
+ $$ = func;
+ }
+ |
+ T_SYSDATE
+ {
+ Plan_root* root = simpleParser.root();
+ const Expr_func& spec = Expr_func::find("SYSDATE");
+ ctx_assert(spec.m_name != 0);
+ Plan_expr_func* func = new Plan_expr_func(root, spec);
+ root->saveNode(func);
+ func->setArgs(0);
+ $$ = func;
+ }
+ |
+ expr_column
+ {
+ $$ = $1;
+ }
+ |
+ T_STRING
+ {
+ Plan_root* root = simpleParser.root();
+ LexType lexType(LexType::Char);
+ Plan_expr_const* expr = new Plan_expr_const(root, lexType, $1);
+ root->saveNode(expr);
+ delete[] $1;
+ $$ = expr;
+ }
+ |
+ T_LINTEGER
+ {
+ Plan_root* root = simpleParser.root();
+ LexType lexType(LexType::Integer);
+ Plan_expr_const* expr = new Plan_expr_const(root, lexType, $1);
+ root->saveNode(expr);
+ delete[] $1;
+ $$ = expr;
+ }
+ |
+ T_LDECIMAL
+ {
+ Plan_root* root = simpleParser.root();
+ LexType lexType(LexType::Float);
+ Plan_expr_const* expr = new Plan_expr_const(root, lexType, $1);
+ root->saveNode(expr);
+ delete[] $1;
+ $$ = expr;
+ }
+ |
+ T_LREAL
+ {
+ Plan_root* root = simpleParser.root();
+ LexType lexType(LexType::Float);
+ Plan_expr_const* expr = new Plan_expr_const(root, lexType, $1);
+ root->saveNode(expr);
+ delete[] $1;
+ $$ = expr;
+ }
+ |
+ T_NULL
+ {
+ Plan_root* root = simpleParser.root();
+ LexType lexType(LexType::Null);
+ Plan_expr_const* expr = new Plan_expr_const(root, lexType, "");
+ root->saveNode(expr);
+ $$ = expr;
+ }
+ |
+ T_QUES
+ {
+ Plan_root* root = simpleParser.root();
+ unsigned paramNumber = simpleParser.paramNumber();
+ ctx_assert(paramNumber != 0);
+ Plan_expr_param* expr = new Plan_expr_param(root, paramNumber);
+ root->saveNode(expr);
+ $$ = expr;
+ }
+ ;
+expr_column:
+ T_IDENTIFIER
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_expr_column* column = new Plan_expr_column(root, $1);
+ root->saveNode(column);
+ delete[] $1;
+ $$ = column;
+ }
+ |
+ T_IDENTIFIER T_PERIOD T_IDENTIFIER
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_expr_column* column = new Plan_expr_column(root, $3);
+ root->saveNode(column);
+ delete[] $3;
+ column->setCname($1);
+ $$ = column;
+ }
+ |
+ T_IDENTIFIER T_PERIOD T_IDENTIFIER T_PERIOD T_IDENTIFIER
+ {
+ Plan_root* root = simpleParser.root();
+ BaseString str;
+ str.append($1);
+ str.append(".");
+ str.append($3);
+ delete[] $1;
+ delete[] $3;
+ Plan_expr_column* column = new Plan_expr_column(root, $5);
+ root->saveNode(column);
+ delete[] $5;
+ column->setCname(str);
+ $$ = column;
+ }
+ ;
+table_list:
+ table
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_table_list* tableList = new Plan_table_list(root);
+ root->saveNode(tableList);
+ tableList->addTable($1);
+ $$ = tableList;
+ }
+ |
+ table_list T_COMMA table
+ {
+ Plan_table_list* tableList = $1;
+ tableList->addTable($3);
+ $$ = tableList;
+ }
+ ;
+table:
+ dot_identifier
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_table* table = new Plan_table(root, $1);
+ root->saveNode(table);
+ delete[] $1;
+ $$ = table;
+ }
+ |
+ dot_identifier T_IDENTIFIER
+ {
+ Plan_root* root = simpleParser.root();
+ Plan_table* table = new Plan_table(root, $1);
+ root->saveNode(table);
+ delete[] $1;
+ table->setCname($2);
+ delete[] $2;
+ $$ = table;
+ }
+ ;
+dot_identifier:
+ T_IDENTIFIER
+ {
+ $$ = $1;
+ }
+ |
+ T_IDENTIFIER T_PERIOD T_IDENTIFIER
+ {
+ char* s = new char[strlen($1) + 1 + strlen($3) + 1];
+ strcpy(s, $1);
+ strcat(s, ".");
+ strcat(s, $3);
+ delete[] $1;
+ delete[] $3;
+ $$ = s;
+ }
+ ;
+
+%%
+
+static int
+yylex(YYSTYPE* lvalp, void* simpleParserPtr)
+{
+ int ret = simpleParser.yylex();
+ *lvalp = simpleParser.yylval();
+ return ret;
+}
+
+/* vim: set filetype=yacc: */
diff --git a/ndb/src/client/odbc/codegen/SimpleParser.cpp b/ndb/src/client/odbc/codegen/SimpleParser.cpp
new file mode 100644
index 00000000000..62162172bc8
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/SimpleParser.cpp
@@ -0,0 +1,95 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbMutex.h>
+#include <common/StmtArea.hpp>
+#include <FlexLexer.h>
+#include "SimpleParser.hpp"
+
+SimpleParser::~SimpleParser()
+{
+}
+
+#ifdef NDB_WIN32
+static NdbMutex & parse_mutex = * NdbMutex_Create();
+#else
+static NdbMutex parse_mutex = NDB_MUTEX_INITIALIZER;
+#endif
+
+void
+SimpleParser::yyparse()
+{
+ Ctx& ctx = this->ctx();
+ NdbMutex_Lock(&parse_mutex);
+ ctx_log2(("parse: %s", stmtArea().sqlText().c_str()));
+#if YYDEBUG
+ SimpleParser_yydebug = (m_ctx.logLevel() >= 5);
+#endif
+ SimpleParser_yyparse((void*)this);
+ NdbMutex_Unlock(&parse_mutex);
+}
+
+void
+SimpleParser::pushState(int sc)
+{
+ yy_push_state(sc);
+ m_stacksize++;
+}
+
+void
+SimpleParser::popState()
+{
+ ctx_assert(m_stacksize > 0);
+ yy_pop_state();
+ m_stacksize--;
+}
+
+void
+SimpleParser::parseError(const char* msg)
+{
+ ctx().pushStatus(Sqlstate::_42000, Error::Gen, "%s at '%*s' position %u", msg, yyleng, yytext, m_parsePos - yyleng);
+}
+
+int
+SimpleParser::LexerInput(char* buf, int max_size)
+{
+ const BaseString& text = stmtArea().sqlText();
+ int n = 0;
+ const char* const t = text.c_str();
+ const unsigned m = text.length();
+ while (n < max_size && m_textPos < m) {
+ buf[n++] = t[m_textPos++];
+ m_parsePos++; // XXX simple hack
+ break;
+ }
+ return n;
+}
+
+// XXX just a catch-all (scanner should match all input)
+void
+SimpleParser::LexerOutput(const char* buf, int size)
+{
+ if (! ctx().ok())
+ return;
+ const char* msg = "unrecognized input";
+ ctx().pushStatus(Sqlstate::_42000, Error::Gen, "%s at '%*s' position %u", msg, size, buf, m_parsePos);
+}
+
+void
+SimpleParser::LexerError(const char* msg)
+{
+ ctx().pushStatus(Sqlstate::_42000, Error::Gen, "%s at '%*s' position %u", msg, yyleng, yytext, m_parsePos);
+}
diff --git a/ndb/src/client/odbc/codegen/SimpleParser.hpp b/ndb/src/client/odbc/codegen/SimpleParser.hpp
new file mode 100644
index 00000000000..abadae8f905
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/SimpleParser.hpp
@@ -0,0 +1,161 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_SimpleParser_hpp
+#define ODBC_CODEGEN_SimpleParser_hpp
+
+#include <common/common.hpp>
+#include "Code_root.hpp"
+#include "Code_stmt.hpp"
+#include "Code_select.hpp"
+#include "Code_pred.hpp"
+#include "Code_pred_op.hpp"
+#include "Code_comp_op.hpp"
+#include "Code_expr_row.hpp"
+#include "Code_expr.hpp"
+#include "Code_expr_op.hpp"
+#include "Code_expr_func.hpp"
+#include "Code_expr_column.hpp"
+#include "Code_expr_const.hpp"
+#include "Code_expr_param.hpp"
+#include "Code_update.hpp"
+#include "Code_set_row.hpp"
+#include "Code_insert.hpp"
+#include "Code_dml_row.hpp"
+#include "Code_dml_column.hpp"
+#include "Code_delete.hpp"
+#include "Code_table_list.hpp"
+#include "Code_table.hpp"
+#include "Code_create_table.hpp"
+#include "Code_create_index.hpp"
+#include "Code_ddl_row.hpp"
+#include "Code_ddl_column.hpp"
+#include "Code_ddl_constr.hpp"
+#include "Code_data_type.hpp"
+#include "Code_drop_table.hpp"
+#include "Code_drop_index.hpp"
+
+#include "SimpleGram.tab.hpp"
+
+class StmtArea;
+class Plan_root;
+
+class SimpleParser : public yyFlexLexer {
+public:
+ SimpleParser(Ctx& ctx, StmtArea& stmtArea, Plan_root* root);
+ ~SimpleParser();
+ Ctx& ctx();
+ StmtArea& stmtArea();
+ Plan_root* root();
+ void yyparse(); // calls real yyparse
+ int yylex(); // generated by flex
+ YYSTYPE yylval();
+ void pushState(int sc); // push start condition
+ void popState(); // pop start condition
+ unsigned paramNumber() const;
+ void parseError(const char* msg);
+ // parser helpers - to avoid creating new Plan tree types
+ Plan_ddl_column* curr(Plan_ddl_column* p);
+ Plan_create_index* curr(Plan_create_index* p);
+protected:
+ virtual int LexerInput(char* buf, int max_size);
+ virtual void LexerOutput(const char* buf, int size);
+ virtual void LexerError(const char* msg);
+private:
+ Ctx& m_ctx;
+ StmtArea& m_stmtArea;
+ Plan_root* const m_root;
+ unsigned m_textPos; // position in sql text
+ unsigned m_parsePos; // parse position, to report error
+ YYSTYPE m_yylval; // token value
+ BaseString m_string; // storage for edited string token
+ unsigned m_stacksize; // state stack size
+ unsigned m_paramNumber; // parameter number
+ // parser helpers
+ Plan_ddl_column* m_ddl_column;
+ Plan_create_index* m_create_index;
+};
+
+extern int SimpleParser_yyparse(void* simpleParserPtr);
+#if YYDEBUG
+extern int SimpleParser_yydebug;
+#endif
+
+inline
+SimpleParser::SimpleParser(Ctx& ctx, StmtArea& stmtArea, Plan_root* root) :
+ m_ctx(ctx),
+ m_stmtArea(stmtArea),
+ m_root(root),
+ m_textPos(0),
+ m_parsePos(0),
+ m_stacksize(0),
+ m_paramNumber(0),
+ // parser helpers
+ m_ddl_column(0)
+{
+}
+
+inline Ctx&
+SimpleParser::ctx()
+{
+ return m_ctx;
+}
+
+inline StmtArea&
+SimpleParser::stmtArea()
+{
+ return m_stmtArea;
+}
+
+inline Plan_root*
+SimpleParser::root()
+{
+ return m_root;
+}
+
+inline YYSTYPE
+SimpleParser::yylval()
+{
+ return m_yylval;
+}
+
+inline unsigned
+SimpleParser::paramNumber() const
+{
+ return m_paramNumber;
+}
+
+// parser helpers
+
+inline Plan_ddl_column*
+SimpleParser::curr(Plan_ddl_column* p)
+{
+ if (p != 0)
+ m_ddl_column = p;
+ ctx_assert(m_ddl_column != 0);
+ return m_ddl_column;
+}
+
+inline Plan_create_index*
+SimpleParser::curr(Plan_create_index* p)
+{
+ if (p != 0)
+ m_create_index = p;
+ ctx_assert(m_create_index != 0);
+ return m_create_index;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/codegen/SimpleScan.lpp b/ndb/src/client/odbc/codegen/SimpleScan.lpp
new file mode 100644
index 00000000000..bd930c08cfa
--- /dev/null
+++ b/ndb/src/client/odbc/codegen/SimpleScan.lpp
@@ -0,0 +1,241 @@
+%{
+#include <ctype.h>
+#include "SimpleParser.hpp"
+
+struct SqlKeyword {
+ const char* m_name;
+ int m_value;
+ int m_state;
+ static const SqlKeyword* find(Ctx& ctx, const char* name, int state);
+};
+
+%}
+
+%option c++
+%option yyclass="SimpleParser"
+%option stack
+%option noyywrap
+
+space [\040\t\n\r\f]
+digit [0-9]
+letter [A-Za-z_]
+special ("$")
+identifier ({letter}({letter}|{digit}|{special})*)
+integer {digit}++
+decimal (({digit}*\.{digit}+)|({digit}+\.{digit}*))
+real ((({digit}*\.{digit}+)|({digit}+\.{digit}*)|({digit}+))([Ee][-+]?{digit}+))
+
+%s StateEval
+%s StateType
+%s StatePhys
+%x StateString
+%x StateQuoted
+
+%%
+
+{space} {
+ }
+{identifier} {
+ const SqlKeyword* key = SqlKeyword::find(m_ctx, (char*)yytext, YYSTATE);
+ if (key != 0)
+ return key->m_value;
+ for (unsigned char* a = (unsigned char*)yytext; *a != 0; a++) {
+ if (islower(*a))
+ *a = toupper(*a);
+ }
+ m_yylval.m_string = strcpy(new char[yyleng + 1], yytext);
+ return T_IDENTIFIER;
+ }
+{integer} {
+ m_yylval.m_string = strcpy(new char[yyleng + 1], yytext);
+ return T_LINTEGER;
+ }
+{decimal} {
+ m_yylval.m_string = strcpy(new char[yyleng + 1], yytext);
+ return T_LDECIMAL;
+ }
+{real} {
+ m_yylval.m_string = strcpy(new char[yyleng + 1], yytext);
+ return T_LREAL;
+ }
+"--".* {
+ }
+"/*" {
+ int c = 0, d;
+ while (1) {
+ d = c;
+ if ((c = yyinput()) == EOF) {
+ parseError("unterminated comment");
+ yyterminate();
+ }
+ if (d == '*' && c == '/')
+ break;
+ }
+ }
+<StateEval>{
+"+" return T_PLUS;
+"-" return T_MINUS;
+"*" return T_TIMES;
+"/" return T_DIVIDE;
+"=" return T_EQ;
+"!=" return T_NOTEQ;
+"^=" return T_NOTEQ;
+"<>" return T_NOTEQ;
+"<" return T_LT;
+"<=" return T_LTEQ;
+">" return T_GT;
+">=" return T_GTEQ;
+"?" m_paramNumber++; return T_QUES;
+}
+
+"." return T_PERIOD;
+"," return T_COMMA;
+"(" return T_PARENLEFT;
+")" return T_PARENRIGHT;
+"*" return T_ASTERISK;
+"=" return T_ASSIGN;
+
+"'" {
+ pushState(StateString);
+ m_string.assign("");
+ }
+<StateString>{
+[^']* {
+ m_string.append(yytext);
+ }
+"''" {
+ m_string.append("'");
+ }
+"'" {
+ m_yylval.m_string = strcpy(new char[m_string.length() + 1], m_string.c_str());
+ popState();
+ return T_STRING;
+ }
+}
+
+\" {
+ pushState(StateQuoted);
+ m_string.assign("");
+ }
+<StateQuoted>{
+[^"]* {
+ m_string.append(yytext);
+ }
+\\\" {
+ m_string.append("\"");
+ }
+\" {
+ m_yylval.m_string = strcpy(new char[m_string.length() + 1], m_string.c_str());
+ popState();
+ return T_IDENTIFIER;
+ }
+}
+
+%%
+
+// scan states
+int SimpleParser_stateEval = StateEval;
+int SimpleParser_stateType = StateType;
+int SimpleParser_statePhys = StatePhys;
+
+// keep sorted
+
+static const SqlKeyword sqlKeyword[] = {
+ { "AND", T_AND, -1 },
+ { "ASC", T_ASC, -1 },
+ { "AUTO_INCREMENT", T_AUTO_INCREMENT, -1 },
+ { "BIGINT", T_BIGINT, StateType },
+ { "BINARY", T_BINARY, StateType },
+ { "BLOB", T_BLOB, StateType },
+ { "BY", T_BY, -1 },
+ { "CHAR", T_CHAR, StateType },
+ { "CONSTRAINT", T_CONSTRAINT, -1 },
+ { "CREATE", T_CREATE, -1 },
+ { "DATETIME", T_DATETIME, StateType },
+ { "DEFAULT", T_DEFAULT, -1 },
+ { "DELETE", T_DELETE, -1 },
+ { "DESC", T_DESC, -1 },
+ { "DISTINCT", T_DISTINCT, -1 },
+ { "DOUBLE", T_DOUBLE, StateType },
+ { "DROP", T_DROP, -1 },
+ { "FLOAT", T_FLOAT, StateType },
+ { "FOREIGN", T_FOREIGN, -1 },
+ { "FROM", T_FROM, -1 },
+ { "GROUP", T_GROUP, -1 },
+ { "HASH", T_HASH, -1 },
+ { "HAVING", T_HAVING, -1 },
+ { "IN", T_IN, -1 },
+ { "INDEX", T_INDEX, -1 },
+ { "INSERT", T_INSERT, -1 },
+ { "INT", T_INT, StateType },
+ { "INTEGER", T_INTEGER, StateType },
+ { "INTO", T_INTO, -1 },
+ { "IS", T_IS, -1 },
+ { "KEY", T_KEY, -1 },
+ { "LARGE", T_LARGE, StatePhys },
+ { "LIKE", T_LIKE, -1 },
+ { "LIMIT", T_LIMIT, -1 },
+ { "LOGGING", T_LOGGING, StatePhys },
+ { "LONGBLOB", T_LONGBLOB, StateType },
+ { "MEDIUM", T_MEDIUM, StatePhys },
+ { "NOLOGGING", T_NOLOGGING, StatePhys },
+ { "NOT", T_NOT, -1 },
+ { "NULL", T_NULL, -1 },
+ { "OFFSET", T_OFFSET, -1 },
+ { "ON", T_ON, -1 },
+ { "OR", T_OR, -1 },
+ { "ORDER", T_ORDER, -1 },
+ { "PRECISION", T_PRECISION, StateType },
+ { "PRIMARY", T_PRIMARY, -1 },
+ { "REAL", T_REAL, StateType },
+ { "REFERENCES", T_REFERENCES, -1 },
+ { "ROWNUM", T_ROWNUM, -1 },
+ { "SELECT", T_SELECT, -1 },
+ { "SET", T_SET, -1 },
+ { "SINGLE", T_SINGLE, StatePhys },
+ { "SMALL", T_SMALL, StatePhys },
+ { "SMALLINT", T_SMALLINT, StateType },
+ { "STORAGE", T_STORAGE, StatePhys },
+ { "SYSDATE", T_SYSDATE, -1 },
+ { "TABLE", T_TABLE, -1 },
+ { "UNIQUE", T_UNIQUE, -1 },
+ { "UNSIGNED", T_UNSIGNED, -1 },
+ { "UPDATE", T_UPDATE, -1 },
+ { "VALUES", T_VALUES, -1 },
+ { "VARBINARY", T_VARBINARY, StateType },
+ { "VARCHAR", T_VARCHAR, StateType },
+ { "WHERE", T_WHERE, -1 },
+ { "WRITE", T_WRITE, -1 }
+};
+
+static const unsigned sqlKeywordCount = sizeof(sqlKeyword) / sizeof(sqlKeyword[0]);
+
+const SqlKeyword*
+SqlKeyword::find(Ctx& ctx, const char* name, int state)
+{
+ ctx_log4(("find keyword '%s' lex state = %d", name, state));
+ const unsigned maxlen = 99;
+ char buf[maxlen + 1];
+ char* a = buf;
+ const char* b = name;
+ while (*b != 0) {
+ if (a >= buf + maxlen) // will not be found
+ break;
+ char c = *b++;
+ if ('a' <= c && c <= 'z') // locale independent
+ c -= 'a' - 'A';
+ *a++ = c;
+ }
+ *a = 0;
+ for (unsigned i = 0; i < sqlKeywordCount; i++) {
+ const SqlKeyword* key = &sqlKeyword[i];
+ if (strcmp(key->m_name, buf) == 0) {
+ if (key->m_state != -1 && key->m_state != state)
+ return 0;
+ return key;
+ }
+ }
+ return 0;
+}
+
+/* vim: set filetype=lex: */
diff --git a/ndb/src/client/odbc/common/AttrArea.cpp b/ndb/src/client/odbc/common/AttrArea.cpp
new file mode 100644
index 00000000000..ff9e085a7f6
--- /dev/null
+++ b/ndb/src/client/odbc/common/AttrArea.cpp
@@ -0,0 +1,91 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "AttrArea.hpp"
+
+// AttrSpec
+
+// AttrField
+
+// AttrArea
+
+AttrArea::AttrArea(const AttrSpec* specList) :
+ m_specList(specList)
+{
+}
+
+AttrArea::~AttrArea()
+{
+}
+
+const AttrSpec&
+AttrArea::findSpec(int id) const
+{
+ const AttrSpec* p;
+ for (p = m_specList; p->m_mode != Attr_mode_undef; p++) {
+ if (p->m_id == id)
+ break;
+ }
+ return *p;
+}
+
+void
+AttrArea::setAttr(Ctx& ctx, int id, const OdbcData& data)
+{
+ const AttrSpec& spec = findSpec(id);
+ if (spec.m_mode == Attr_mode_undef) {
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "undefined attribute id %d", id);
+ return;
+ }
+ ctx_assert(spec.m_type == data.type());
+ ctx_assert(spec.m_set != 0);
+ spec.m_set(ctx, m_handle, data);
+ if (! ctx.ok())
+ return;
+ Fields::iterator iter;
+ if (ctx.logLevel() >= 3) {
+ char buf[100];
+ data.print(buf, sizeof(buf));
+ ctx_log3(("attr handle 0x%x set id %d = %s", (unsigned)m_handle, id, buf));
+ }
+ iter = m_fields.find(id);
+ if (iter != m_fields.end()) {
+ AttrField& field = (*iter).second;
+ field.setData(data);
+ return;
+ }
+ AttrField field(spec, data);
+ m_fields.insert(Fields::value_type(id, field));
+}
+
+void
+AttrArea::getAttr(Ctx& ctx, int id, OdbcData& data)
+{
+ const AttrSpec& spec = findSpec(id);
+ if (spec.m_mode == Attr_mode_undef) {
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "undefined attribute id %d", id);
+ return;
+ }
+ Fields::iterator iter;
+ iter = m_fields.find(id);
+ if (iter != m_fields.end()) {
+ AttrField& field = (*iter).second;
+ data.setValue(field.getData());
+ return;
+ }
+ ctx_assert(spec.m_default != 0);
+ spec.m_default(ctx, m_handle, data);
+}
diff --git a/ndb/src/client/odbc/common/AttrArea.hpp b/ndb/src/client/odbc/common/AttrArea.hpp
new file mode 100644
index 00000000000..050cce719bf
--- /dev/null
+++ b/ndb/src/client/odbc/common/AttrArea.hpp
@@ -0,0 +1,135 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_HANDLES_AttrArea_hpp
+#define ODBC_HANDLES_AttrArea_hpp
+
+#include <map>
+#include <common/common.hpp>
+#include "OdbcData.hpp"
+
+class HandleBase;
+
+enum AttrMode {
+ Attr_mode_undef,
+ Attr_mode_readonly,
+ Attr_mode_writeonly,
+ Attr_mode_readwrite
+};
+
+/**
+ * @struct AttrSpec
+ * @brief Attribute specifier
+ *
+ * Each handle class has a list of attribute specifiers.
+ */
+struct AttrSpec {
+ int m_id; // SQL_ATTR_ identifier
+ OdbcData::Type m_type; // data type
+ AttrMode m_mode; // access mode, undef indicates end of list
+ /**
+ * Callback for checks and side effects. Called before the
+ * attribute is stored. May set error status.
+ */
+ typedef void CallbackSet(Ctx& ctx, HandleBase* self, const OdbcData& data);
+ CallbackSet* m_set;
+ /**
+ * Callback to set default value. May set error status.
+ */
+ typedef void CallbackDefault(Ctx& ctx, HandleBase* self, OdbcData& data);
+ CallbackDefault* m_default;
+};
+
+/**
+ * @class AttrField
+ * @brief Attribute value (stored as OdbcData)
+ */
+class AttrField {
+public:
+ AttrField(const AttrSpec& attrSpec, const OdbcData& data);
+ AttrField(const AttrField& field);
+ ~AttrField();
+ void setData(const OdbcData& data);
+ const OdbcData& getData();
+private:
+ const AttrSpec& m_spec;
+ OdbcData m_data;
+};
+
+inline
+AttrField::AttrField(const AttrSpec& spec, const OdbcData& data) :
+ m_spec(spec),
+ m_data(data)
+{
+}
+
+inline
+AttrField::AttrField(const AttrField& field) :
+ m_spec(field.m_spec),
+ m_data(field.m_data)
+{
+}
+
+inline
+AttrField::~AttrField()
+{
+}
+
+inline void
+AttrField::setData(const OdbcData& data)
+{
+ ctx_assert(m_spec.m_type == data.type());
+ m_data.setValue(data);
+}
+
+inline const OdbcData&
+AttrField::getData()
+{
+ ctx_assert(m_data.type() != OdbcData::Undef);
+ return m_data;
+}
+
+/**
+ * @class AttrArea
+ * @brief Handle attributes
+ *
+ * Each handle instance has a list of attribute values stored
+ * under an AttrArea. Callbacks to handle code provide for
+ * default values, extra checks, and side-effects.
+ */
+class AttrArea {
+public:
+ AttrArea(const AttrSpec* specList);
+ ~AttrArea();
+ void setHandle(HandleBase* handle);
+ const AttrSpec& findSpec(int id) const;
+ void setAttr(Ctx& ctx, int id, const OdbcData& data);
+ void getAttr(Ctx& ctx, int id, OdbcData& data);
+private:
+ HandleBase* m_handle;
+ const AttrSpec* const m_specList;
+ typedef std::map<int, AttrField> Fields;
+ Fields m_fields;
+};
+
+inline void
+AttrArea::setHandle(HandleBase* handle)
+{
+ ctx_assert(handle != 0);
+ m_handle = handle;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/CodeTree.cpp b/ndb/src/client/odbc/common/CodeTree.cpp
new file mode 100644
index 00000000000..ebe4840c5f6
--- /dev/null
+++ b/ndb/src/client/odbc/common/CodeTree.cpp
@@ -0,0 +1,37 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "CodeTree.hpp"
+
+// PlanTree
+
+PlanTree::~PlanTree()
+{
+}
+
+// ExecTree
+
+ExecTree::Code::~Code()
+{
+}
+
+ExecTree::Data::~Data()
+{
+}
+
+ExecTree::~ExecTree()
+{
+}
diff --git a/ndb/src/client/odbc/common/CodeTree.hpp b/ndb/src/client/odbc/common/CodeTree.hpp
new file mode 100644
index 00000000000..1b0ae3199af
--- /dev/null
+++ b/ndb/src/client/odbc/common/CodeTree.hpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_CODEGEN_CodeTree_hpp
+#define ODBC_CODEGEN_CodeTree_hpp
+
+#include <common/common.hpp>
+
+/*
+ * Intermediary statement evalution plan. Starts as parse tree. Final
+ * format maps directly to ExecTree.
+ */
+class PlanTree {
+public:
+ virtual ~PlanTree() = 0;
+};
+
+/*
+ * Executable code and runtime data. Currently looks like PlanTree.
+ * Later may change code format to linear "byte code" and move execution
+ * to a SQL block in NDB kernel.
+ */
+class ExecTree {
+public:
+ class Code {
+ public:
+ virtual ~Code() = 0;
+ };
+ class Data {
+ public:
+ virtual ~Data() = 0;
+ };
+ virtual ~ExecTree() = 0;
+};
+
+#endif
diff --git a/ndb/src/client/odbc/common/ConnArea.cpp b/ndb/src/client/odbc/common/ConnArea.cpp
new file mode 100644
index 00000000000..d4d3be52a3c
--- /dev/null
+++ b/ndb/src/client/odbc/common/ConnArea.cpp
@@ -0,0 +1,109 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <dictionary/DictCatalog.hpp>
+#include <dictionary/DictSchema.hpp>
+#include "ConnArea.hpp"
+
+ConnArea::ConnArea() :
+ m_state(Free),
+ m_ndbObject(0),
+ m_ndbSchemaCon(0),
+ m_ndbConnection(0),
+ m_useSchemaCon(0),
+ m_useConnection(0),
+ m_autocommit(true),
+ m_uncommitted(false)
+{
+ // initialize connection catalog
+ m_catalog = new DictCatalog(*this);
+ m_schema = new DictSchema(*this, "NDB");
+ m_catalog->addSchema(m_schema);
+}
+
+ConnArea::~ConnArea()
+{
+ delete m_catalog;
+}
+
+bool
+ConnArea::useSchemaCon(Ctx& ctx, bool use)
+{
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "not connected");
+ return false;
+ }
+ Ndb* ndb = m_ndbObject;
+ ctx_assert(ndb != 0);
+ NdbSchemaCon* scon = m_ndbSchemaCon;
+ if (use) {
+ if (scon == 0) {
+ ctx_assert(m_useSchemaCon == 0);
+ scon = ndb->startSchemaTransaction();
+ if (scon == 0) {
+ ctx.pushStatus(ndb, scon, 0, "startSchemaTransaction");
+ return false;
+ }
+ m_ndbSchemaCon = scon;
+ }
+ m_useSchemaCon++;
+ } else {
+ ctx_assert(scon != 0 && m_useSchemaCon != 0);
+ if (--m_useSchemaCon == 0) {
+ ndb->closeSchemaTransaction(scon);
+ m_ndbSchemaCon = 0;
+ }
+ }
+ return true;
+}
+
+bool
+ConnArea::useConnection(Ctx& ctx, bool use)
+{
+ ctx_log3(("useConnection: count before=%u on-off=%d", m_useConnection, (int)use));
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "not connected");
+ return false;
+ }
+ Ndb* ndb = m_ndbObject;
+ ctx_assert(ndb != 0);
+ NdbConnection* tcon = m_ndbConnection;
+ if (use) {
+ if (tcon == 0) {
+ ctx_assert(m_useConnection == 0);
+ tcon = ndb->startTransaction();
+ if (tcon == 0) {
+ ctx.pushStatus(ndb, tcon, 0, "startTransaction");
+ return false;
+ }
+ m_ndbConnection = tcon;
+ m_state = Transacting;
+ ctx_log2(("transaction opened"));
+ }
+ m_useConnection++;
+ } else {
+ ctx_assert(tcon != 0 && m_useConnection != 0);
+ if (--m_useConnection == 0) {
+ ndb->closeTransaction(tcon);
+ m_ndbConnection = 0;
+ m_uncommitted = false;
+ m_state = Connected;
+ ctx_log2(("transaction closed"));
+ }
+ }
+ return true;
+}
diff --git a/ndb/src/client/odbc/common/ConnArea.hpp b/ndb/src/client/odbc/common/ConnArea.hpp
new file mode 100644
index 00000000000..36367a39bae
--- /dev/null
+++ b/ndb/src/client/odbc/common/ConnArea.hpp
@@ -0,0 +1,135 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_ConnArea_hpp
+#define ODBC_COMMON_ConnArea_hpp
+
+#include <common/common.hpp>
+
+class Ctx;
+class Ndb;
+class NdbSchemaCon;
+class NdbConnection;
+class DictCatalog;
+class DictSchema;
+
+/**
+ * @class ConnArea
+ * @brief Public part of connection handle
+ */
+class ConnArea {
+public:
+ // state between ODBC function calls
+ enum State {
+ Free = 1, // not in use, no Ndb object
+ Connected = 2, // has Ndb object but no Ndb connection
+ Transacting = 3 // has Ndb connection
+ };
+ State getState() const;
+ DictCatalog& dictCatalog() const;
+ DictSchema& dictSchema() const;
+ Ndb* ndbObject() const;
+ NdbSchemaCon* ndbSchemaCon() const;
+ NdbConnection* ndbConnection() const;
+ // called from StmtArea
+ bool useSchemaCon(Ctx& ctx, bool use);
+ bool useConnection(Ctx& ctx, bool use);
+ // these just get and set the flag - no semantics
+ bool autocommit() const;
+ void autocommit(bool flag);
+ bool uncommitted() const;
+ void uncommitted(bool flag);
+protected:
+ ConnArea();
+ ~ConnArea();
+ State m_state;
+ DictCatalog* m_catalog;
+ DictSchema* m_schema;
+ Ndb* m_ndbObject;
+ NdbSchemaCon* m_ndbSchemaCon;
+ NdbConnection* m_ndbConnection;
+ unsigned m_useSchemaCon;
+ unsigned m_useConnection;
+ bool m_autocommit;
+ bool m_uncommitted; // has uncommitted changes
+};
+
+inline ConnArea::State
+ConnArea::getState() const
+{
+ return m_state;
+}
+
+inline DictCatalog&
+ConnArea::dictCatalog() const
+{
+ ctx_assert(m_catalog != 0);
+ return *m_catalog;
+}
+
+inline DictSchema&
+ConnArea::dictSchema() const
+{
+ ctx_assert(m_schema != 0);
+ return *m_schema;
+}
+
+inline Ndb*
+ConnArea::ndbObject() const
+{
+ ctx_assert(m_ndbObject != 0);
+ return m_ndbObject;
+}
+
+inline NdbSchemaCon*
+ConnArea::ndbSchemaCon() const
+{
+ ctx_assert(m_ndbSchemaCon != 0);
+ return m_ndbSchemaCon;
+}
+
+inline NdbConnection*
+ConnArea::ndbConnection() const
+{
+ ctx_assert(m_ndbConnection != 0);
+ return m_ndbConnection;
+}
+
+inline bool
+ConnArea::autocommit() const
+{
+ return m_autocommit;
+}
+
+inline void
+ConnArea::autocommit(bool flag)
+{
+ m_autocommit = flag;
+}
+
+inline bool
+ConnArea::uncommitted() const
+{
+ return m_uncommitted;
+}
+
+inline void
+ConnArea::uncommitted(bool flag)
+{
+ m_uncommitted = flag;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/Ctx.cpp b/ndb/src/client/odbc/common/Ctx.cpp
new file mode 100644
index 00000000000..85edbd1a63f
--- /dev/null
+++ b/ndb/src/client/odbc/common/Ctx.cpp
@@ -0,0 +1,360 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbUnistd.h>
+#include <stdlib.h>
+#include <string.h>
+#include <NdbStdio.h>
+#include <stdarg.h>
+#include <NdbApi.hpp>
+#include <common/common.hpp>
+#include "DiagArea.hpp"
+
+// ctor
+
+Ctx::Ctx() :
+ m_diagArea(0) // create on demand
+{
+ const char* p;
+ if ((p = getenv("NDB_ODBC_TRACE")) != 0)
+ m_logLevel = atoi(p);
+ if ((p = getenv("NDB_ODBC_TRACE_FILE")) != 0 && *p != 0)
+ strcpy(m_szTraceFile, p);
+}
+
+Ctx::~Ctx()
+{
+ delete m_diagArea;
+ m_diagArea = 0;
+}
+
+// handle exceptions
+
+CtxAssert::CtxAssert(const char* file, int line) :
+ m_file(file),
+ m_line(line)
+{
+ const char* p;
+ if ((p = getenv("NDB_ODBC_DEBUG")) != 0 && atoi(p) != 0) {
+ char buf[200];
+ snprintf(buf, sizeof(buf), "%s, line %d: assert failed\n", m_file, m_line);
+ if ((p = getenv("NDB_ODBC_TRACE_FILE")) != 0 && *p != 0) {
+ FILE* pFile = fopen(p, "a");
+ fprintf(pFile, buf);
+ fflush(pFile);
+ fclose(pFile);
+ } else {
+ fprintf(stderr, buf);
+ fflush(stderr);
+ }
+ abort();
+ exit(1);
+ }
+}
+
+void
+Ctx::handleEx(CtxAssert& ctxAssert)
+{
+ pushStatus(Sqlstate::_IM001, Error::Gen, "exception at %s line %d", ctxAssert.m_file, ctxAssert.m_line);
+}
+
+// logging methods
+
+int Ctx::m_logLevel = 0;
+char Ctx::m_szTraceFile[MAX_PATH];
+
+void
+Ctx::log(const char* fmt, ...)
+{
+ va_list ap;
+ va_start(ap, fmt);
+ if (m_szTraceFile[0]) {
+ FILE* pFile = fopen(m_szTraceFile, "a");
+ fprintf(pFile, "[NdbOdbc] ");
+ vfprintf(pFile, fmt, ap);
+ fprintf(pFile, "\n");
+ fflush(pFile);
+ fclose(pFile);
+ } else {
+ printf("[NdbOdbc] ");
+ vprintf(fmt, ap);
+ printf("\n");
+ fflush(stdout);
+ }
+ va_end(ap);
+}
+
+void
+Ctx::logSqlEnter(const char* sqlFunction)
+{
+ Ctx& ctx = *this;
+ snprintf(m_sqlFunction, sizeof(m_sqlFunction), "%s", sqlFunction);
+ ctx_log3(("%s", m_sqlFunction));
+}
+
+void
+Ctx::logSqlExit()
+{
+ Ctx& ctx = *this;
+ if (m_diagArea == 0) {
+ ctx_log3(("%s ret=%d", m_sqlFunction, getCode()));
+ return;
+ }
+ int logLevel = diagArea().numStatus() != 0 ? 2 : 3;
+ ctx_logN(logLevel, ("%s ret=%d diag=%d", m_sqlFunction, diagArea().getCode(), diagArea().numStatus()));
+ for (unsigned i = 1; i <= diagArea().numStatus(); i++) {
+ OdbcData state;
+ OdbcData message;
+ diagArea().getRecord(ctx, i, SQL_DIAG_SQLSTATE, state);
+ diagArea().getRecord(ctx, i, SQL_DIAG_MESSAGE_TEXT, message);
+ ctx_logN(logLevel, ("diag %u: %s - %s", i, state.sqlstate().state(), message.sqlchar()));
+ }
+}
+
+void
+Ctx::print(const char* fmt, ...)
+{
+ va_list ap;
+ va_start(ap, fmt);
+ if (m_szTraceFile[0]) {
+ FILE* pFile = fopen(m_szTraceFile, "a");
+ vfprintf(pFile, fmt, ap);
+ unsigned n = strlen(fmt);
+ if (n > 0 && fmt[n-1] == '\n')
+ fflush(pFile);
+ fclose(pFile);
+ } else {
+ vprintf(fmt, ap);
+ unsigned n = strlen(fmt);
+ if (n > 0 && fmt[n-1] == '\n')
+ fflush(stdout);
+ }
+ va_end(ap);
+}
+
+void
+Ctx::print(int level, const char* fmt, ...)
+{
+ if (level > m_logLevel)
+ return;
+ va_list ap;
+ va_start(ap, fmt);
+ if (m_szTraceFile[0]) {
+ FILE* pFile = fopen(m_szTraceFile, "a");
+ vfprintf(pFile, fmt, ap);
+ unsigned n = strlen(fmt);
+ if (n > 0 && fmt[n-1] == '\n')
+ fflush(pFile);
+ fclose(pFile);
+ } else {
+ vprintf(fmt, ap);
+ unsigned n = strlen(fmt);
+ if (n > 0 && fmt[n-1] == '\n')
+ fflush(stdout);
+ }
+ va_end(ap);
+}
+
+// diagnostics
+
+static const unsigned MessageSize = 512;
+
+DiagArea&
+Ctx::diagArea() const
+{
+ ctx_assert(m_diagArea != 0);
+ return *m_diagArea;
+}
+
+DiagArea&
+Ctx::diagArea()
+{
+ if (m_diagArea == 0)
+ m_diagArea = new DiagArea;
+ return *m_diagArea;
+}
+
+SQLRETURN
+Ctx::getCode() const
+{
+ if (m_diagArea == 0)
+ return SQL_SUCCESS;
+ return diagArea().getCode();
+}
+
+void
+Ctx::setCode(SQLRETURN ret)
+{
+ diagArea().setCode(ret);
+}
+
+void
+Ctx::pushStatus(const Sqlstate& state, SQLINTEGER code, const char* fmt, ...)
+{
+ char message[MessageSize];
+ va_list ap;
+ va_start(ap, fmt);
+ vsnprintf(message, sizeof(message), fmt, ap);
+ va_end(ap);
+ Error error(state);
+ error.m_status = NdbError::PermanentError;
+ error.m_classification = NdbError::ApplicationError;
+ error.m_code = code;
+ error.m_message = message;
+ error.m_sqlFunction = m_sqlFunction;
+ diagArea().pushStatus(error);
+}
+
+void
+Ctx::pushStatus(SQLINTEGER code, const char* fmt, ...)
+{
+ char message[MessageSize];
+ va_list ap;
+ va_start(ap, fmt);
+ vsnprintf(message, sizeof(message), fmt, ap);
+ va_end(ap);
+ Error error(Sqlstate::_IM000);
+ error.m_status = NdbError::PermanentError;
+ error.m_classification = NdbError::ApplicationError;
+ error.m_code = code;
+ error.m_message = message;
+ error.m_sqlFunction = m_sqlFunction;
+ diagArea().pushStatus(error);
+}
+
+void
+Ctx::pushStatus(const NdbError& ndbError, const char* fmt, ...)
+{
+ char message[MessageSize];
+ va_list ap;
+ va_start(ap, fmt);
+ snprintf(message, sizeof(message), "%s", ndbError.message);
+ snprintf(message + strlen(message), sizeof(message) - strlen(message), "%s", " - at ");
+ vsnprintf(message + strlen(message), sizeof(message) - strlen(message), fmt, ap);
+ va_end(ap);
+ Error error(Sqlstate::_IM000);
+ error.m_status = ndbError.status;
+ error.m_classification = ndbError.classification;
+ error.m_code = ndbError.code;
+ error.m_message = message;
+ error.m_sqlFunction = m_sqlFunction;
+ diagArea().pushStatus(error);
+}
+
+void
+Ctx::pushStatus(const Ndb* ndb, const char* fmt, ...)
+{
+ char message[MessageSize];
+ va_list ap;
+ va_start(ap, fmt);
+ vsnprintf(message, sizeof(message), fmt, ap);
+ va_end(ap);
+ bool found = false;
+ if (ndb != 0) {
+ const NdbError& ndbError = ndb->getNdbError();
+ if (ndbError.code != 0) {
+ pushStatus(ndbError, "%s", message);
+ found = true;
+ }
+ }
+ if (! found) {
+ pushStatus(Error::Gen, "unknown NDB error");
+ }
+}
+
+void
+Ctx::pushStatus(const Ndb* ndb, const NdbConnection* tcon, const NdbOperation* op, const char* fmt, ...)
+{
+ char message[MessageSize];
+ va_list ap;
+ va_start(ap, fmt);
+ vsnprintf(message, sizeof(message), fmt, ap);
+ va_end(ap);
+ bool found = false;
+ if (op != 0) {
+ const NdbError& ndbError = op->getNdbError();
+ if (ndbError.code != 0) {
+ pushStatus(ndbError, "%s", message);
+ found = true;
+ }
+ }
+ if (tcon != 0) {
+ const NdbError& ndbError = tcon->getNdbError();
+ if (ndbError.code != 0) {
+ pushStatus(ndbError, "%s", message);
+ found = true;
+ }
+ }
+ if (ndb != 0) {
+ const NdbError& ndbError = ndb->getNdbError();
+ if (ndbError.code != 0) {
+ pushStatus(ndbError, "%s", message);
+ found = true;
+ }
+ }
+ if (! found) {
+ pushStatus(Error::Gen, "unknown NDB error");
+ }
+}
+
+void
+Ctx::pushStatus(const Ndb* ndb, const NdbSchemaCon* scon, const NdbSchemaOp* op, const char* fmt, ...)
+{
+ char message[MessageSize];
+ va_list ap;
+ va_start(ap, fmt);
+ vsnprintf(message, sizeof(message), fmt, ap);
+ va_end(ap);
+ bool found = false;
+ if (op != 0) {
+ const NdbError& ndbError = op->getNdbError();
+ if (ndbError.code != 0) {
+ pushStatus(ndbError, "%s", message);
+ found = true;
+ }
+ }
+ if (scon != 0) {
+ const NdbError& ndbError = scon->getNdbError();
+ if (ndbError.code != 0) {
+ pushStatus(ndbError, "%s", message);
+ found = true;
+ }
+ }
+ if (ndb != 0) {
+ const NdbError& ndbError = ndb->getNdbError();
+ if (ndbError.code != 0) {
+ pushStatus(ndbError, "%s", message);
+ found = true;
+ }
+ }
+ if (! found) {
+ pushStatus(Error::Gen, "unknown NDB error");
+ }
+}
+
+// check for error
+
+bool
+Ctx::ok()
+{
+ if (m_diagArea == 0)
+ return true;
+ if (diagArea().getCode() == SQL_SUCCESS)
+ return true;
+ if (diagArea().getCode() == SQL_SUCCESS_WITH_INFO)
+ return true;
+ return false;
+}
diff --git a/ndb/src/client/odbc/common/Ctx.hpp b/ndb/src/client/odbc/common/Ctx.hpp
new file mode 100644
index 00000000000..d25d45ff0c7
--- /dev/null
+++ b/ndb/src/client/odbc/common/Ctx.hpp
@@ -0,0 +1,182 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_Ctx_hpp
+#define ODBC_COMMON_Ctx_hpp
+
+#include <NdbDictionary.hpp>
+
+class Ndb;
+class NdbConnection;
+class NdbOperation;
+class NdbSchemaCon;
+class NdbSchemaOp;
+class NdbError;
+
+class Sqlstate;
+class DiagArea;
+class CtxOwner;
+
+#ifndef MAX_PATH
+#define MAX_PATH 1024
+#endif
+
+/**
+ * @class Error
+ * @brief Sql state, error codes, and message
+ */
+struct Error {
+ enum {
+ Gen = NDB_ODBC_ERROR_MIN + 1 // unclassified
+ };
+ explicit Error(const Sqlstate& sqlstate);
+ const Sqlstate& m_sqlstate;
+ int m_status;
+ int m_classification;
+ int m_code;
+ const char* m_message;
+ const char* m_sqlFunction;
+ bool driverError() const;
+};
+
+inline
+Error::Error(const Sqlstate& sqlstate) :
+ m_sqlstate(sqlstate),
+ m_status(0),
+ m_classification(0),
+ m_code(0),
+ m_sqlFunction(0)
+{
+}
+
+inline bool
+Error::driverError() const
+{
+ return NDB_ODBC_ERROR_MIN <= m_code && m_code < NDB_ODBC_ERROR_MAX;
+}
+
+#define ctx_assert(x) \
+ do { if (x) break; throw CtxAssert(__FILE__, __LINE__); } while (0)
+
+/**
+ * @class Assert
+ * @brief Assert thrown
+ */
+class CtxAssert {
+public:
+ CtxAssert(const char* file, int line);
+ const char* const m_file;
+ const int m_line;
+};
+
+/**
+ * @class Ctx
+ * @brief Context for one ODBC SQL function
+ *
+ * Local to the function (not member of the handle) because methods on
+ * a handle can call methods on other handles. Created in driver/
+ * before method calls and saved under the handle on return. Contains
+ * diag area for the function. Also used as logger.
+ */
+class Ctx {
+public:
+ Ctx();
+ ~Ctx();
+ // handle exceptions
+ void handleEx(CtxAssert& ctxAssert);
+ // logging methods
+ int logLevel() const;
+ void log(const char* fmt, ...) PRINTFLIKE(2,3);
+ void logSqlEnter(const char* sqlFunction);
+ void logSqlExit();
+ const char* sqlFunction() const;
+ void print(const char* fmt, ...) PRINTFLIKE(2,3);
+ void print(int level, const char* fmt, ...) PRINTFLIKE(3,4);
+ // diagnostics area.
+ DiagArea& diagArea() const;
+ DiagArea& diagArea();
+ SQLRETURN getCode() const;
+ void setCode(SQLRETURN ret);
+ // push diagnostic record
+ void pushStatus(const Sqlstate& state, SQLINTEGER code, const char* fmt, ...) PRINTFLIKE(4,5);
+ void pushStatus(SQLINTEGER code, const char* fmt, ...) PRINTFLIKE(3,4);
+ void pushStatus(const NdbError& ndbError, const char* fmt, ...) PRINTFLIKE(3,4);
+ void pushStatus(const Ndb* ndb, const char* fmt, ...) PRINTFLIKE(3,4);
+ void pushStatus(const Ndb* ndb, const NdbConnection* tcon, const NdbOperation* op, const char* fmt, ...) PRINTFLIKE(5,6);
+ void pushStatus(const Ndb* ndb, const NdbSchemaCon* scon, const NdbSchemaOp* op, const char* fmt, ...) PRINTFLIKE(5,6);
+ // check if we should continue executing.
+ bool ok();
+private:
+ static int m_logLevel;
+ static char m_szTraceFile[MAX_PATH];
+ char m_sqlFunction[32]; // max needed is 20
+ DiagArea* m_diagArea;
+};
+
+inline int
+Ctx::logLevel() const
+{
+ return m_logLevel;
+}
+
+inline const char*
+Ctx::sqlFunction() const
+{
+ return m_sqlFunction;
+}
+
+// logging macros can be used only when ctx is in scope
+
+#define ctx_logN(n, x) \
+ do { if (ctx.logLevel() < (n)) break; ctx.log x; } while (0)
+
+#if NDB_ODBC_MAX_LOG_LEVEL >= 0
+#define ctx_log0(x) ctx_logN(0, x)
+#else
+#define ctx_log0(x)
+#endif
+
+#if NDB_ODBC_MAX_LOG_LEVEL >= 1
+#define ctx_log1(x) ctx_logN(1, x)
+#else
+#define ctx_log1(x)
+#endif
+
+#if NDB_ODBC_MAX_LOG_LEVEL >= 2
+#define ctx_log2(x) ctx_logN(2, x)
+#else
+#define ctx_log2(x)
+#endif
+
+#if NDB_ODBC_MAX_LOG_LEVEL >= 3
+#define ctx_log3(x) ctx_logN(3, x)
+#else
+#define ctx_log3(x)
+#endif
+
+#if NDB_ODBC_MAX_LOG_LEVEL >= 4
+#define ctx_log4(x) ctx_logN(4, x)
+#else
+#define ctx_log4(x)
+#endif
+
+#if NDB_ODBC_MAX_LOG_LEVEL >= 5
+#define ctx_log5(x) ctx_logN(5, x)
+#else
+#define ctx_log5(x)
+#endif
+
+#endif
diff --git a/ndb/src/client/odbc/common/DataField.cpp b/ndb/src/client/odbc/common/DataField.cpp
new file mode 100644
index 00000000000..5853f90c08f
--- /dev/null
+++ b/ndb/src/client/odbc/common/DataField.cpp
@@ -0,0 +1,3030 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbUnistd.h>
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <errno.h>
+#include <limits.h>
+#include <math.h>
+#include "DataField.hpp"
+#include <time.h>
+
+#ifndef INT_MAX
+#define INT_MAX (2147483647)
+#endif
+
+#ifndef INT_MIN
+#define INT_MIN (-INT_MAX - 1)
+#endif
+
+#ifndef UINT_MAX
+#define UINT_MAX 4294967295U
+#endif
+
+#ifndef FLT_MAX
+#define FLT_MAX (3.402823466E+38F)
+#endif
+#ifndef FLT_MIN
+#define FLT_MIN (1.175494351E-38F)
+#endif
+
+#ifdef NDB_WIN32
+#define FMT_I64 "%I64d"
+#define FMT_U64 "%I64u"
+#else
+#define FMT_I64 "%lld"
+#define FMT_U64 "%llu"
+#endif
+
+#ifdef NDB_WIN32
+#define strtoll(str, endptr, base) strtoint64(str, endptr, base)
+#define strtoull(str, endptr, base) strtouint64(str, endptr, base)
+
+static Int64
+strtoint64(const char *str, char **endptr, int base)
+{
+ Int64 x = 0;
+ while (*str == ' ')
+ str++;
+ const char* p = str;
+ while ('0' <= *p && *p <= '9')
+ x = 10 * x + *p++ - '0';
+ if (p == str) {
+ *endptr = 0;
+ return 0;
+ }
+ *endptr = (char*)p;
+ return x;
+}
+
+static Uint64
+strtouint64(const char *str, char **endptr, int base)
+{
+ Uint64 x = 0;
+ while (*str == ' ')
+ str++;
+ const char* p = str;
+ while ('0' <= *p && *p <= '9')
+ x = 10 * x + *p++ - '0';
+ if (p == str) {
+ *endptr = 0;
+ return 0;
+ }
+ *endptr = (char*)p;
+ return x;
+}
+#endif
+
+// LexSpec
+
+void
+LexSpec::convert(Ctx& ctx, const BaseString& value, SqlField& out)
+{
+ const SqlSpec& sqlSpec = out.sqlSpec();
+ const SqlType& sqlType = sqlSpec.sqlType();
+ out.alloc();
+ if (sqlType.type() == SqlType::Char) {
+ const SqlChar* s = (const SqlChar*)value.c_str();
+ out.sqlChar(s, SQL_NTS);
+ return;
+ }
+ if (sqlType.type() == SqlType::Bigint) {
+ char* endptr = 0;
+ SqlBigint n = static_cast<SqlBigint>(strtoll(value.c_str(), &endptr, 10));
+ if (endptr == 0 || *endptr != 0) {
+ ctx.pushStatus(Error::Gen, "cannot convert '%s' to integer", value.c_str());
+ return;
+ }
+ out.sqlBigint(n);
+ return;
+ }
+ if (sqlType.type() == SqlType::Double) {
+ char* endptr = 0;
+ SqlDouble x = static_cast<SqlDouble>(strtod(value.c_str(), &endptr));
+ if (endptr == 0 || *endptr != 0) {
+ ctx.pushStatus(Error::Gen, "cannot convert '%s' to number", value.c_str());
+ return;
+ }
+ out.sqlDouble(x);
+ return;
+ }
+ if (sqlType.type() == SqlType::Null) {
+ out.u_null.m_nullFlag = true;
+ return;
+ }
+ ctx_assert(false);
+}
+
+// SqlField
+
+void
+SqlField::alloc()
+{
+ ctx_assert(sqlSpec().store() == SqlSpec::Physical);
+ const SqlType& sqlType = sqlSpec().sqlType();
+ if (sqlType.type() == SqlType::Char || sqlType.type() == SqlType::Varchar) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varchar)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ u_data.m_sqlChar = new SqlChar[n];
+ }
+ }
+ if (sqlType.type() == SqlType::Binary || sqlType.type() == SqlType::Varbinary) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varbinary)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ u_data.m_sqlChar = new SqlChar[n];
+ }
+ }
+}
+
+void
+SqlField::alloc(const SqlField& sqlField)
+{
+ alloc();
+ const SqlType& sqlType = sqlSpec().sqlType();
+ if (sqlType.type() == SqlType::Char || sqlType.type() == SqlType::Varchar) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varchar)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ memcpy(u_data.m_sqlChar, sqlField.u_data.m_sqlChar, n);
+ }
+ }
+ if (sqlType.type() == SqlType::Binary || sqlType.type() == SqlType::Varbinary) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varbinary)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ memcpy(u_data.m_sqlChar, sqlField.u_data.m_sqlChar, n);
+ }
+ }
+}
+
+const void*
+SqlField::addr() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->addr();
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ if (sqlType.type() == SqlType::Char || sqlType.type() == SqlType::Varchar) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varchar)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ return static_cast<const void*>(u_data.m_sqlChar);
+ }
+ return static_cast<const void*>(u_data.m_sqlCharSmall);
+ }
+ if (sqlType.type() == SqlType::Binary || sqlType.type() == SqlType::Varbinary) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varbinary)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ return static_cast<const void*>(u_data.m_sqlChar);
+ }
+ return static_cast<const void*>(u_data.m_sqlCharSmall);
+ }
+ if (sqlType.type() == SqlType::Smallint) {
+ return static_cast<const void*>(&u_data.m_sqlSmallint);
+ }
+ if (sqlType.type() == SqlType::Integer) {
+ return static_cast<const void*>(&u_data.m_sqlInteger);
+ }
+ if (sqlType.type() == SqlType::Bigint) {
+ return static_cast<const void*>(&u_data.m_sqlBigint);
+ }
+ if (sqlType.type() == SqlType::Real) {
+ return static_cast<const void*>(&u_data.m_sqlReal);
+ }
+ if (sqlType.type() == SqlType::Double) {
+ return static_cast<const void*>(&u_data.m_sqlDouble);
+ }
+ if (sqlType.type() == SqlType::Datetime) {
+ return static_cast<const void*>(&u_data.m_sqlDatetime);
+ }
+ ctx_assert(false); // SqlType::Null has no address
+ return 0;
+}
+
+void*
+SqlField::addr()
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ if (sqlType.type() == SqlType::Char || sqlType.type() == SqlType::Varchar) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varchar)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ return static_cast<void*>(u_data.m_sqlChar);
+ }
+ return static_cast<void*>(u_data.m_sqlCharSmall);
+ }
+ if (sqlType.type() == SqlType::Binary || sqlType.type() == SqlType::Varbinary) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varbinary)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ return static_cast<void*>(u_data.m_sqlChar);
+ }
+ return static_cast<void*>(u_data.m_sqlCharSmall);
+ }
+ if (sqlType.type() == SqlType::Smallint) {
+ return static_cast<void*>(&u_data.m_sqlSmallint);
+ }
+ if (sqlType.type() == SqlType::Integer) {
+ return static_cast<void*>(&u_data.m_sqlInteger);
+ }
+ if (sqlType.type() == SqlType::Bigint) {
+ return static_cast<void*>(&u_data.m_sqlBigint);
+ }
+ if (sqlType.type() == SqlType::Real) {
+ return static_cast<void*>(&u_data.m_sqlReal);
+ }
+ if (sqlType.type() == SqlType::Double) {
+ return static_cast<void*>(&u_data.m_sqlDouble);
+ }
+ if (sqlType.type() == SqlType::Datetime) {
+ return static_cast<void*>(&u_data.m_sqlDatetime);
+ }
+ ctx_assert(false); // SqlType::Null has no address
+ return 0;
+}
+
+unsigned
+SqlField::allocSize() const
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ unsigned n = sqlType.size();
+ if (sqlType.type() == SqlType::Varchar || sqlType.type() == SqlType::Varbinary) {
+ n += 2;
+ }
+ return n;
+}
+
+void
+SqlField::free()
+{
+ ctx_assert(sqlSpec().store() == SqlSpec::Physical);
+ const SqlType& sqlType = sqlSpec().sqlType();
+ if (sqlType.type() == SqlType::Char || sqlType.type() == SqlType::Varchar) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varchar)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ delete[] u_data.m_sqlChar;
+ u_data.m_sqlChar = 0; // safety since dtor used explicitly
+ }
+ }
+ if (sqlType.type() == SqlType::Binary || sqlType.type() == SqlType::Varbinary) {
+ unsigned n = sqlType.length();
+ if (sqlType.type() == SqlType::Varbinary)
+ n += 2;
+ if (n > SqlField_CharSmall) {
+ delete[] u_data.m_sqlChar;
+ u_data.m_sqlChar = 0; // safety since dtor used explicitly
+ }
+ }
+}
+
+// get
+
+const SqlChar*
+SqlField::sqlChar() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlChar();
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Char);
+ if (sqlType.length() > SqlField_CharSmall)
+ return u_data.m_sqlChar;
+ return u_data.m_sqlCharSmall;
+}
+
+const SqlChar*
+SqlField::sqlVarchar(unsigned* length) const
+{
+#if NDB_VERSION_MAJOR >= 3
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlVarchar(length);
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Varchar);
+ const SqlChar* sqlChar;
+ unsigned n = sqlType.length();
+ if (2 + n > SqlField_CharSmall)
+ sqlChar = u_data.m_sqlChar;
+ else
+ sqlChar = u_data.m_sqlCharSmall;
+ if (length != 0)
+ *length = (sqlChar[0] << 8) | sqlChar[1]; // big-endian
+ return sqlChar + 2;
+#else
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlVarchar(length);
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Varchar);
+ const SqlChar* sqlChar;
+ unsigned n = sqlType.length();
+ if (n + 2 > SqlField_CharSmall)
+ sqlChar = u_data.m_sqlChar;
+ else
+ sqlChar = u_data.m_sqlCharSmall;
+ if (length != 0)
+ *length = (sqlChar[n + 0] << 8) | sqlChar[n + 1]; // big-endian
+ return sqlChar;
+#endif
+}
+
+const SqlChar*
+SqlField::sqlBinary() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlChar();
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Binary);
+ if (sqlType.length() > SqlField_CharSmall)
+ return u_data.m_sqlChar;
+ return u_data.m_sqlCharSmall;
+}
+
+const SqlChar*
+SqlField::sqlVarbinary(unsigned* length) const
+{
+#if NDB_VERSION_MAJOR >= 3
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlVarchar(length);
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Varbinary);
+ const SqlChar* sqlChar;
+ unsigned n = sqlType.length();
+ if (2 + n > SqlField_CharSmall)
+ sqlChar = u_data.m_sqlChar;
+ else
+ sqlChar = u_data.m_sqlCharSmall;
+ if (length != 0)
+ *length = (sqlChar[0] << 8) | sqlChar[1]; // big-endian
+ return sqlChar + 2;
+#else
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlVarchar(length);
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Varbinary);
+ const SqlChar* sqlChar;
+ unsigned n = sqlType.length();
+ if (n + 2 > SqlField_CharSmall)
+ sqlChar = u_data.m_sqlChar;
+ else
+ sqlChar = u_data.m_sqlCharSmall;
+ if (length != 0)
+ *length = (sqlChar[n + 0] << 8) | sqlChar[n + 1]; // big-endian
+ return sqlChar;
+#endif
+}
+
+SqlSmallint
+SqlField::sqlSmallint() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlSmallint();
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Smallint);
+ return u_data.m_sqlSmallint;
+}
+
+SqlInteger
+SqlField::sqlInteger() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlInteger();
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Integer);
+ return u_data.m_sqlInteger;
+}
+
+SqlBigint
+SqlField::sqlBigint() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlBigint();
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Bigint);
+ return u_data.m_sqlBigint;
+}
+
+SqlReal
+SqlField::sqlReal() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlReal();
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Real);
+ return u_data.m_sqlReal;
+}
+
+SqlDouble
+SqlField::sqlDouble() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlDouble();
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Double);
+ return u_data.m_sqlDouble;
+}
+
+SqlDatetime
+SqlField::sqlDatetime() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlDatetime();
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Datetime);
+ return u_data.m_sqlDatetime;
+}
+
+// set
+
+void
+SqlField::sqlChar(const SqlChar* value, int length)
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Char);
+ unsigned n = sqlType.length();
+ SqlChar* p = n > SqlField_CharSmall ? u_data.m_sqlChar : u_data.m_sqlCharSmall;
+ const SqlChar* q = value;
+ unsigned m = length == SQL_NTS ? strlen((const char*)q) : length;
+ ctx_assert(m <= n);
+ for (unsigned i = 0; i < m; i++)
+ *p++ = *q++;
+ for (unsigned i = m; i < n; i++)
+ *p++ = 0x20; // space
+ u_null.m_nullFlag = false;
+}
+
+void
+SqlField::sqlVarchar(const SqlChar* value, int length)
+{
+#if NDB_VERSION_MAJOR >= 3
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Varchar);
+ unsigned n = sqlType.length();
+ SqlChar* p = 2 + n > SqlField_CharSmall ? u_data.m_sqlChar : u_data.m_sqlCharSmall;
+ const SqlChar* q = value;
+ unsigned m = length == SQL_NTS ? strlen((const char*)q) : length;
+ ctx_assert(m <= n);
+ *p++ = (m >> 8) & 0xff; // big-endian
+ *p++ = (m & 0xff);
+ for (unsigned i = 0; i < m; i++)
+ *p++ = *q++;
+ for (unsigned i = m; i < n; i++)
+ *p++ = 0x0; // null
+ u_null.m_nullFlag = false;
+#else
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Varchar);
+ unsigned n = sqlType.length();
+ SqlChar* p = n + 2 > SqlField_CharSmall ? u_data.m_sqlChar : u_data.m_sqlCharSmall;
+ const SqlChar* q = value;
+ unsigned m = length == SQL_NTS ? strlen((const char*)q) : length;
+ ctx_assert(m <= n);
+ for (unsigned i = 0; i < m; i++)
+ *p++ = *q++;
+ for (unsigned i = m; i < n; i++)
+ *p++ = 0x0; // null
+ *p++ = (m >> 8) & 0xff; // big-endian
+ *p++ = (m & 0xff);
+ u_null.m_nullFlag = false;
+#endif
+}
+
+void
+SqlField::sqlBinary(const SqlChar* value, int length)
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Binary);
+ unsigned n = sqlType.length();
+ SqlChar* p = n > SqlField_CharSmall ? u_data.m_sqlChar : u_data.m_sqlCharSmall;
+ const SqlChar* q = value;
+ unsigned m = length;
+ ctx_assert(m <= n);
+ for (unsigned i = 0; i < m; i++)
+ *p++ = *q++;
+ for (unsigned i = m; i < n; i++)
+ *p++ = 0x0; // null
+ u_null.m_nullFlag = false;
+}
+
+void
+SqlField::sqlVarbinary(const SqlChar* value, int length)
+{
+#if NDB_VERSION_MAJOR >= 3
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Varbinary);
+ unsigned n = sqlType.length();
+ SqlChar* p = 2 + n > SqlField_CharSmall ? u_data.m_sqlChar : u_data.m_sqlCharSmall;
+ const SqlChar* q = value;
+ unsigned m = length;
+ ctx_assert(m <= n);
+ *p++ = (m >> 8) & 0xff; // big-endian
+ *p++ = (m & 0xff);
+ for (unsigned i = 0; i < m; i++)
+ *p++ = *q++;
+ for (unsigned i = m; i < n; i++)
+ *p++ = 0x0; // null
+ u_null.m_nullFlag = false;
+#else
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Varbinary);
+ unsigned n = sqlType.length();
+ SqlChar* p = n + 2 > SqlField_CharSmall ? u_data.m_sqlChar : u_data.m_sqlCharSmall;
+ const SqlChar* q = value;
+ unsigned m = length;
+ ctx_assert(m <= n);
+ for (unsigned i = 0; i < m; i++)
+ *p++ = *q++;
+ for (unsigned i = m; i < n; i++)
+ *p++ = 0x0; // null
+ *p++ = (m >> 8) & 0xff; // big-endian
+ *p++ = (m & 0xff);
+ u_null.m_nullFlag = false;
+#endif
+}
+
+void
+SqlField::sqlSmallint(SqlSmallint value)
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Smallint);
+ u_data.m_sqlSmallint = value;
+ u_null.m_nullFlag = false;
+}
+
+void
+SqlField::sqlInteger(SqlInteger value)
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Integer);
+ u_data.m_sqlInteger = value;
+ u_null.m_nullFlag = false;
+}
+
+void
+SqlField::sqlBigint(SqlBigint value)
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Bigint);
+ u_data.m_sqlBigint = value;
+ u_null.m_nullFlag = false;
+}
+
+void
+SqlField::sqlReal(SqlReal value)
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Real);
+ u_data.m_sqlReal = value;
+ u_null.m_nullFlag = false;
+}
+
+void
+SqlField::sqlDouble(SqlDouble value)
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Double);
+ u_data.m_sqlDouble = value;
+ u_null.m_nullFlag = false;
+}
+
+void
+SqlField::sqlDatetime(SqlDatetime value)
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ ctx_assert(sqlType.type() == SqlType::Datetime);
+ u_data.m_sqlDatetime = value;
+ u_null.m_nullFlag = false;
+}
+
+// get and and set null
+
+bool
+SqlField::sqlNull() const
+{
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ return u_data.m_sqlField->sqlNull();
+ }
+ return u_null.m_nullFlag;
+}
+
+void
+SqlField::sqlNull(bool value)
+{
+ u_null.m_nullFlag = value;
+}
+
+unsigned
+SqlField::trim() const
+{
+ const SqlType& sqlType = sqlSpec().sqlType();
+ unsigned n = 0;
+ const SqlChar* s = 0;
+ if (sqlType.type() == SqlType::Char) {
+ n = sqlType.length();
+ s = sqlChar();
+ } else if (sqlType.type() == SqlType::Varchar) {
+ s = sqlVarchar(&n);
+ } else {
+ ctx_assert(false);
+ return 0;
+ }
+ while (n > 0 && s[n - 1] == 0x20)
+ n--;
+ return n;
+}
+
+void
+SqlField::copy(Ctx& ctx, SqlField& out) const
+{
+ const SqlField& f1 = *this;
+ SqlField& f2 = out;
+ const SqlType& t1 = f1.sqlSpec().sqlType();
+ const SqlType& t2 = f2.sqlSpec().sqlType();
+ ctx_assert(t1.type() == t2.type());
+ if (f1.sqlNull()) {
+ f2.sqlNull(true);
+ return;
+ }
+ if (t1.type() == SqlType::Char) {
+ f2.sqlChar(f1.sqlChar(), t1.length());
+ return;
+ }
+ if (t1.type() == SqlType::Varchar) {
+ unsigned length;
+ const SqlChar* s1 = f1.sqlVarchar(&length);
+ f2.sqlVarchar(s1, length);
+ return;
+ }
+ if (t1.type() == SqlType::Binary) {
+ f2.sqlBinary(f1.sqlBinary(), t1.length());
+ return;
+ }
+ if (t1.type() == SqlType::Varbinary) {
+ unsigned length;
+ const SqlChar* s1 = f1.sqlVarbinary(&length);
+ f2.sqlVarbinary(s1, length);
+ return;
+ }
+ if (t1.type() == SqlType::Smallint) {
+ f2.sqlSmallint(f1.sqlSmallint());
+ return;
+ }
+ if (t1.type() == SqlType::Integer) {
+ f2.sqlInteger(f1.sqlInteger());
+ return;
+ }
+ if (t1.type() == SqlType::Bigint) {
+ f2.sqlBigint(f1.sqlBigint());
+ return;
+ }
+ if (t1.type() == SqlType::Real) {
+ f2.sqlReal(f1.sqlReal());
+ return;
+ }
+ if (t1.type() == SqlType::Double) {
+ f2.sqlDouble(f1.sqlDouble());
+ return;
+ }
+ if (t1.type() == SqlType::Datetime) {
+ f2.sqlDatetime(f1.sqlDatetime());
+ return;
+ }
+ ctx_assert(false);
+}
+
+bool
+SqlField::cast(Ctx& ctx, SqlField& out) const
+{
+ const SqlField& f1 = *this;
+ SqlField& f2 = out;
+ if (f1.sqlNull()) {
+ f2.sqlNull(true);
+ return true;
+ }
+ const SqlType& t1 = f1.sqlSpec().sqlType();
+ const SqlType& t2 = f2.sqlSpec().sqlType();
+ if (t1.type() == SqlType::Char) {
+ if (t2.type() == SqlType::Char) {
+ unsigned n1 = f1.trim();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlChar(f1.sqlChar(), n1);
+ return true;
+ }
+ if (t2.type() == SqlType::Varchar) {
+ unsigned n1 = t1.length();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlVarchar(f1.sqlChar(), n1);
+ return true;
+ }
+ if (t2.type() == SqlType::Binary) {
+ unsigned n1 = t1.length();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlBinary(f1.sqlChar(), n1);
+ return true;
+ }
+ if (t2.type() == SqlType::Varbinary) {
+ unsigned n1 = t1.length();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlVarbinary(f1.sqlChar(), n1);
+ return true;
+ }
+ ctx_assert(false);
+ return false;
+ }
+ if (t1.type() == SqlType::Varchar) {
+ if (t2.type() == SqlType::Char) {
+ unsigned n1 = f1.trim();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlChar(f1.sqlVarchar(0), n1);
+ return true;
+ }
+ if (t2.type() == SqlType::Varchar) {
+ unsigned n1 = f1.trim();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlVarchar(f1.sqlVarchar(0), n1);
+ return true;
+ }
+ if (t2.type() == SqlType::Binary) {
+ unsigned n1 = t1.length();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlBinary(f1.sqlVarchar(0), n1);
+ return true;
+ }
+ if (t2.type() == SqlType::Varbinary) {
+ unsigned n1 = t1.length();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlVarbinary(f1.sqlVarchar(0), n1);
+ return true;
+ }
+ ctx_assert(false);
+ return false;
+ }
+ if (t1.type() == SqlType::Binary) {
+ if (t2.type() == SqlType::Binary) {
+ unsigned n1 = t1.length();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlBinary(f1.sqlBinary(), n1);
+ return true;
+ }
+ if (t2.type() == SqlType::Varbinary) {
+ unsigned n1 = t1.length();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlVarbinary(f1.sqlBinary(), n1);
+ return true;
+ }
+ ctx_assert(false);
+ return false;
+ }
+ if (t1.type() == SqlType::Varbinary) {
+ if (t2.type() == SqlType::Binary) {
+ unsigned n1 = t1.length();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlBinary(f1.sqlVarbinary(0), n1);
+ return true;
+ }
+ if (t2.type() == SqlType::Varbinary) {
+ unsigned n1 = t1.length();
+ if (n1 > t2.length())
+ return false;
+ f2.sqlVarbinary(f1.sqlVarbinary(0), n1);
+ return true;
+ }
+ ctx_assert(false);
+ return false;
+ }
+ if (t1.type() == SqlType::Smallint) {
+ if (! t2.unSigned()) {
+ SqlSmallint x1 = f1.sqlSmallint();
+ if (t2.type() == SqlType::Smallint) {
+ f2.sqlSmallint(x1);
+ return true;
+ }
+ if (t2.type() == SqlType::Integer) {
+ SqlInteger x2 = static_cast<SqlInteger>(x1);
+ f2.sqlInteger(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Bigint) {
+ SqlBigint x2 = static_cast<SqlBigint>(x1);
+ f2.sqlBigint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Real) {
+ SqlReal x2 = static_cast<SqlReal>(x1);
+ f2.sqlReal(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Double) {
+ SqlDouble x2 = static_cast<SqlDouble>(x1);
+ f2.sqlDouble(x2);
+ return true;
+ }
+ } else {
+ SqlUsmallint x1 = f1.sqlSmallint();
+ if (t2.type() == SqlType::Smallint) {
+ f2.sqlSmallint(x1);
+ return true;
+ }
+ if (t2.type() == SqlType::Integer) {
+ SqlUinteger x2 = static_cast<SqlUinteger>(x1);
+ f2.sqlInteger(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Bigint) {
+ SqlUbigint x2 = static_cast<SqlUbigint>(x1);
+ f2.sqlBigint(x2);
+ return true;
+ }
+ }
+ ctx_assert(false);
+ return false;
+ }
+ if (t1.type() == SqlType::Integer) {
+ if (! t2.unSigned()) {
+ SqlInteger x1 = f1.sqlInteger();
+ if (t2.type() == SqlType::Smallint) {
+ SqlSmallint x2 = static_cast<SqlSmallint>(x1);
+ if (x1 != static_cast<SqlInteger>(x2))
+ return false;
+ f2.sqlSmallint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Integer) {
+ f2.sqlInteger(x1);
+ return true;
+ }
+ if (t2.type() == SqlType::Bigint) {
+ SqlBigint x2 = static_cast<SqlBigint>(x1);
+ f2.sqlBigint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Real) {
+ SqlReal x2 = static_cast<SqlReal>(x1);
+ f2.sqlReal(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Double) {
+ SqlDouble x2 = static_cast<SqlDouble>(x1);
+ f2.sqlDouble(x2);
+ return true;
+ }
+ } else {
+ SqlUinteger x1 = f1.sqlInteger();
+ if (t2.type() == SqlType::Smallint) {
+ SqlUsmallint x2 = static_cast<SqlUsmallint>(x1);
+ if (x1 != static_cast<SqlUinteger>(x2))
+ return false;
+ f2.sqlSmallint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Integer) {
+ f2.sqlInteger(x1);
+ return true;
+ }
+ if (t2.type() == SqlType::Bigint) {
+ SqlUbigint x2 = static_cast<SqlUbigint>(x1);
+ f2.sqlBigint(x2);
+ return true;
+ }
+ }
+ ctx_assert(false);
+ return false;
+ }
+ if (t1.type() == SqlType::Bigint) {
+ if (! t2.unSigned()) {
+ SqlBigint x1 = f1.sqlBigint();
+ if (t2.type() == SqlType::Smallint) {
+ SqlSmallint x2 = static_cast<SqlSmallint>(x1);
+ if (x1 != static_cast<SqlBigint>(x2))
+ return false;
+ f2.sqlSmallint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Integer) {
+ SqlInteger x2 = static_cast<SqlInteger>(x1);
+ if (x1 != static_cast<SqlBigint>(x2))
+ return false;
+ f2.sqlInteger(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Bigint) {
+ f2.sqlBigint(x1);
+ return true;
+ }
+ if (t2.type() == SqlType::Real) {
+ SqlReal x2 = static_cast<SqlReal>(x1);
+ f2.sqlReal(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Double) {
+ SqlDouble x2 = static_cast<SqlDouble>(x1);
+ f2.sqlDouble(x2);
+ return true;
+ }
+ } else {
+ SqlUbigint x1 = f1.sqlBigint();
+ if (t2.type() == SqlType::Smallint) {
+ SqlUsmallint x2 = static_cast<SqlUsmallint>(x1);
+ if (x1 != static_cast<SqlUbigint>(x2))
+ return false;
+ f2.sqlSmallint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Integer) {
+ SqlUinteger x2 = static_cast<SqlUinteger>(x1);
+ if (x1 != static_cast<SqlUbigint>(x2))
+ return false;
+ f2.sqlInteger(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Bigint) {
+ f2.sqlBigint(x1);
+ return true;
+ }
+ }
+ ctx_assert(false);
+ return false;
+ }
+ if (t1.type() == SqlType::Real) {
+ SqlReal x1 = f1.sqlReal();
+ int off = 0;
+ if (x1 > 0.0 && x1 - floor(x1) >= 0.5)
+ off = 1;
+ if (x1 < 0.0 && x1 - floor(x1) <= 0.5)
+ off = -1;
+ bool b = (x1 - floor(x1) < 0.5);
+ if (t2.type() == SqlType::Smallint) {
+ SqlSmallint x2 = static_cast<SqlSmallint>(x1) + off;
+ if (fabs(x1 - static_cast<SqlReal>(x2)) >= 1.0)
+ return false;
+ f2.sqlSmallint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Integer) {
+ SqlInteger x2 = static_cast<SqlInteger>(x1) + off;
+ if (fabs(x1 - static_cast<SqlReal>(x2)) >= 1.0)
+ return false;
+ f2.sqlInteger(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Bigint) {
+ SqlBigint x2 = static_cast<SqlBigint>(x1) + off;
+ if (fabs(x1 - static_cast<SqlReal>(x2)) >= 1.0)
+ return false;
+ f2.sqlBigint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Real) {
+ f2.sqlReal(x1);
+ return true;
+ }
+ if (t2.type() == SqlType::Double) {
+ SqlDouble x2 = static_cast<SqlDouble>(x1);
+ f2.sqlDouble(x2);
+ return true;
+ }
+ ctx_assert(false);
+ return false;
+ }
+ if (t1.type() == SqlType::Double) {
+ SqlDouble x1 = f1.sqlDouble();
+ int off = 0;
+ if (x1 > 0.0 && x1 - floor(x1) >= 0.5)
+ off = 1;
+ if (x1 < 0.0 && x1 - floor(x1) <= 0.5)
+ off = -1;
+ bool b = (x1 - floor(x1) < 0.5);
+ if (t2.type() == SqlType::Smallint) {
+ SqlSmallint x2 = static_cast<SqlSmallint>(x1) + off;
+ if (fabs(x1 - static_cast<SqlDouble>(x2)) >= 1.0)
+ return false;
+ f2.sqlSmallint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Integer) {
+ SqlInteger x2 = static_cast<SqlInteger>(x1) + off;
+ if (fabs(x1 - static_cast<SqlDouble>(x2)) >= 1.0)
+ return false;
+ f2.sqlInteger(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Bigint) {
+ SqlBigint x2 = static_cast<SqlBigint>(x1) + off;
+ if (fabs(x1 - static_cast<SqlDouble>(x2)) >= 1.0)
+ return false;
+ f2.sqlBigint(x2);
+ return true;
+ }
+ if (t2.type() == SqlType::Real) {
+ SqlReal x2 = static_cast<SqlReal>(x1);
+ if (fabs(x1 - static_cast<SqlDouble>(x2)) >= 1.0) // XXX
+ return false;
+ f2.sqlReal(x1);
+ return true;
+ }
+ if (t2.type() == SqlType::Double) {
+ f2.sqlDouble(x1);
+ return true;
+ }
+ ctx_assert(false);
+ return false;
+ }
+ ctx_assert(false);
+ return false;
+}
+
+bool
+SqlField::less(const SqlField& sqlField) const
+{
+ const SqlField& f1 = *this;
+ const SqlField& f2 = sqlField;
+ const SqlType& t1 = f1.sqlSpec().sqlType();
+ const SqlType& t2 = f2.sqlSpec().sqlType();
+ ctx_assert(t1.type() == t2.type());
+ if (t1.type() == SqlType::Char) {
+ const SqlChar* s1 = f1.sqlChar();
+ const SqlChar* s2 = f2.sqlChar();
+ unsigned n1 = t1.length();
+ unsigned n2 = t2.length();
+ SqlChar c1 = 0;
+ SqlChar c2 = 0;
+ unsigned i = 0;
+ while (i < n1 || i < n2) {
+ c1 = i < n1 ? s1[i] : 0x20;
+ c2 = i < n2 ? s2[i] : 0x20;
+ if (c1 != c2)
+ break;
+ i++;
+ }
+ return (c1 < c2);
+ }
+ if (t1.type() == SqlType::Varchar) {
+ unsigned n1, n2;
+ const SqlChar* s1 = f1.sqlVarchar(&n1);
+ const SqlChar* s2 = f2.sqlVarchar(&n2);
+ SqlChar c1 = 0;
+ SqlChar c2 = 0;
+ unsigned i = 0;
+ while (i < n1 || i < n2) {
+ c1 = i < n1 ? s1[i] : 0x0;
+ c2 = i < n2 ? s2[i] : 0x0;
+ if (c1 != c2)
+ break;
+ i++;
+ }
+ return (c1 < c2);
+ }
+ if (t1.type() == SqlType::Smallint) {
+ ctx_assert(t1.unSigned() == t2.unSigned());
+ if (! t1.unSigned()) {
+ SqlSmallint x1 = f1.sqlSmallint();
+ SqlSmallint x2 = f2.sqlSmallint();
+ return (x1 < x2);
+ } else {
+ SqlUsmallint x1 = f1.sqlSmallint();
+ SqlUsmallint x2 = f2.sqlSmallint();
+ return (x1 < x2);
+ }
+ }
+ if (t1.type() == SqlType::Integer) {
+ ctx_assert(t1.unSigned() == t2.unSigned());
+ if (! t1.unSigned()) {
+ SqlInteger x1 = f1.sqlInteger();
+ SqlInteger x2 = f2.sqlInteger();
+ return (x1 < x2);
+ } else {
+ SqlUinteger x1 = f1.sqlInteger();
+ SqlUinteger x2 = f2.sqlInteger();
+ return (x1 < x2);
+ }
+ }
+ if (t1.type() == SqlType::Bigint) {
+ ctx_assert(t1.unSigned() == t2.unSigned());
+ if (! t1.unSigned()) {
+ SqlBigint x1 = f1.sqlBigint();
+ SqlBigint x2 = f2.sqlBigint();
+ return (x1 < x2);
+ } else {
+ SqlUbigint x1 = f1.sqlBigint();
+ SqlUbigint x2 = f2.sqlBigint();
+ return (x1 < x2);
+ }
+ }
+ if (t1.type() == SqlType::Real) {
+ SqlReal x1 = f1.sqlReal();
+ SqlReal x2 = f2.sqlReal();
+ return (x1 < x2);
+ }
+ if (t1.type() == SqlType::Double) {
+ SqlDouble x1 = f1.sqlDouble();
+ SqlDouble x2 = f2.sqlDouble();
+ return (x1 < x2);
+ }
+ if (t1.type() == SqlType::Datetime) {
+ SqlDatetime x1 = f1.sqlDatetime();
+ SqlDatetime x2 = f2.sqlDatetime();
+ return x1.less(x2);
+ }
+ ctx_assert(false);
+}
+
+// copy from external
+
+static bool
+copyin_char_char(Ctx& ctx, char* value, unsigned n, const char* ptr, const SQLINTEGER* ind, int* off, SqlChar* addr, int fieldId)
+{
+ if (off != 0 && *off >= 0) {
+ if ((unsigned)*off > n) {
+ ctx.pushStatus(Sqlstate::_22001, Error::Gen, "input parameter %d truncated (%u > %u)", fieldId, (unsigned)*off, n);
+ return false;
+ }
+ value += *off;
+ n -= *off;
+ }
+ unsigned m;
+ if (ind == 0 || *ind == SQL_NTS)
+ m = strlen(ptr);
+ else
+ m = *ind;
+ if (m > n) {
+ ctx.pushStatus(Sqlstate::_22001, Error::Gen, "input parameter %d truncated (%u > %u)", fieldId, m, n);
+ return false;
+ }
+ for (unsigned i = 0; i < m; i++)
+ value[i] = ptr[i];
+ if (off != 0 && *off >= 0)
+ *off += m;
+ for (unsigned i = m; i < n; i++)
+ value[i] = addr == 0 ? 0x20 : 0x0;
+ if (addr != 0) {
+ if (off != 0 && *off >= 0)
+ m = *off;
+ addr[0] = (m >> 8) & 0xff;
+ addr[1] = (m & 0xff);
+ }
+ return true;
+}
+
+static bool
+copyin_binary_binary(Ctx& ctx, char* value, unsigned n, const char* ptr, const SQLINTEGER* ind, int* off, SqlChar* addr, int fieldId)
+{
+ if (off != 0 && *off >= 0) {
+ if ((unsigned)*off > n) {
+ ctx.pushStatus(Sqlstate::_22001, Error::Gen, "input parameter %d truncated (%u > %u)", fieldId, (unsigned)*off, n);
+ return false;
+ }
+ value += *off;
+ n -= *off;
+ }
+ if (ind == 0) {
+ ctx.pushStatus(Sqlstate::_22001, Error::Gen, "input parameter %d missing length", fieldId);
+ return false;
+ }
+ if (*ind < 0) {
+ ctx.pushStatus(Sqlstate::_22001, Error::Gen, "input parameter %d invalid length %d", fieldId, (int)*ind);
+ return false;
+ }
+ unsigned m;
+ m = *ind;
+ if (m > n) {
+ ctx.pushStatus(Sqlstate::_22001, Error::Gen, "input parameter %d truncated (%u > %u)", fieldId, m, n);
+ return false;
+ }
+ for (unsigned i = 0; i < m; i++)
+ value[i] = ptr[i];
+ if (off != 0 && *off >= 0)
+ *off += m;
+ for (unsigned i = m; i < n; i++)
+ value[i] = addr == 0 ? 0x0 : 0x0; // just null
+ if (addr != 0) {
+ if (off != 0 && *off >= 0)
+ m = *off;
+ addr[0] = (m >> 8) & 0xff;
+ addr[1] = (m & 0xff);
+ }
+ return true;
+}
+
+static bool
+copyin_signed_char(Ctx& ctx, SqlBigint* value, const char* ptr, int fieldId)
+{
+ errno = 0;
+ char* endptr = 0;
+ SqlBigint x = strtoll(ptr, &endptr, 10);
+ if (endptr == 0 || *endptr != 0) {
+ errno = 0;
+ endptr = 0;
+ double y = strtod(ptr, &endptr);
+ if (endptr == 0 || *endptr != 0) {
+ ctx.pushStatus(Sqlstate::_22005, Error::Gen, "input parameter %d value %s not numeric", fieldId, ptr);
+ return false;
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "input parameter %d value %s overflow", fieldId, ptr);
+ return false;
+ }
+ // XXX should handle 123.000
+ ctx.pushStatus(Sqlstate::_01004, Error::Gen, "input parameter %d value %s truncated", fieldId, ptr);
+ x = static_cast<SqlBigint>(y);
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "input parameter %d value %s overflow", fieldId, ptr);
+ return false;
+ }
+ *value = x;
+ return true;
+}
+
+static bool
+copyin_double_char(Ctx& ctx, SqlDouble* value, const char* ptr, int fieldId)
+{
+ errno = 0;
+ char* endptr = 0;
+ double x = strtod(ptr, &endptr);
+ if (endptr == 0 || *endptr != 0) {
+ ctx.pushStatus(Sqlstate::_22005, Error::Gen, "input parameter %d value %s not numeric", fieldId, ptr);
+ return false;
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "input parameter %d value %s overflow", fieldId, ptr);
+ return false;
+ }
+ *value = x;
+ return true;
+}
+
+void
+SqlField::copyin(Ctx& ctx, ExtField& extField)
+{
+ ctx_assert(extField.extSpec().extType().type() != ExtType::Unbound);
+ ctx_assert(sqlSpec().store() == SqlSpec::Physical);
+ SQLINTEGER* indPtr = extField.m_indPtr;
+ const int fieldId = extField.fieldId();
+ if (indPtr != 0 && *indPtr == SQL_NULL_DATA) {
+ sqlNull(true);
+ return;
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ const ExtType& extType = extField.extSpec().extType();
+ if (extField.m_pos > 0) {
+ if (sqlType.type() == SqlType::Char && extType.type() == ExtType::Char)
+ ;
+ else if (sqlType.type() == SqlType::Varchar && extType.type() == ExtType::Char)
+ ;
+ else {
+ char buf[40];
+ sqlType.print(buf, sizeof(buf));
+ ctx.pushStatus(Sqlstate::_HY019, Error::Gen, "cannot send %s data in pieces", buf);
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Char || sqlType.type() == SqlType::Varchar) {
+ unsigned length = 0;
+ char* value = 0;
+ SqlChar* laddr = 0; // Varchar length address
+ if (sqlType.type() == SqlType::Char) {
+ length = sqlType.length();
+ if (length > SqlField_CharSmall)
+ value = reinterpret_cast<char *>(u_data.m_sqlChar);
+ else
+ value = reinterpret_cast<char *>(u_data.m_sqlCharSmall);
+ laddr = 0;
+ } else {
+#if NDB_VERSION_MAJOR >= 3
+ length = sqlType.length();
+ if (2 + length > SqlField_CharSmall)
+ value = reinterpret_cast<char *>(u_data.m_sqlChar + 2);
+ else
+ value = reinterpret_cast<char *>(u_data.m_sqlCharSmall + 2);
+ laddr = (SqlChar*)value - 2;
+#else
+ length = sqlType.length();
+ if (length + 2 > SqlField_CharSmall)
+ value = reinterpret_cast<char *>(u_data.m_sqlChar);
+ else
+ value = reinterpret_cast<char *>(u_data.m_sqlCharSmall);
+ laddr = (SqlChar*)value + length;
+#endif
+ }
+ if (extType.type() == ExtType::Char) {
+ const char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ int* off = 0;
+ if (extField.m_pos >= 0)
+ off = &extField.m_pos;
+ if (! copyin_char_char(ctx, value, length, dataPtr, indPtr, off, laddr, fieldId))
+ return;
+ sqlNull(false);
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ const short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ char buf[100];
+ sprintf(buf, "%hd", *dataPtr);
+ if (! copyin_char_char(ctx, value, length, buf, indPtr, 0, laddr, fieldId))
+ return;
+ sqlNull(false);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ const unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ char buf[100];
+ sprintf(buf, "%hu", *dataPtr);
+ if (! copyin_char_char(ctx, value, length, buf, indPtr, 0, laddr, fieldId))
+ return;
+ sqlNull(false);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ const long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ char buf[100];
+ sprintf(buf, "%ld", *dataPtr);
+ if (! copyin_char_char(ctx, value, length, buf, indPtr, 0, laddr, fieldId))
+ return;
+ sqlNull(false);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ const unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ char buf[100];
+ sprintf(buf, "%lu", *dataPtr);
+ if (! copyin_char_char(ctx, value, length, buf, indPtr, 0, laddr, fieldId))
+ return;
+ sqlNull(false);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ const SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ char buf[100];
+ sprintf(buf, FMT_I64, *dataPtr);
+ if (! copyin_char_char(ctx, value, length, buf, indPtr, 0, laddr, fieldId))
+ return;
+ sqlNull(false);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ const SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ char buf[100];
+ sprintf(buf, FMT_U64, *dataPtr);
+ if (! copyin_char_char(ctx, value, length, buf, indPtr, 0, laddr, fieldId))
+ return;
+ sqlNull(false);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ const float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ char buf[100];
+ sprintf(buf, "%.7f", (double)*dataPtr);
+ if (! copyin_char_char(ctx, value, length, buf, indPtr, 0, laddr, fieldId))
+ return;
+ sqlNull(false);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ const double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ char buf[100];
+ sprintf(buf, "%.14f", *dataPtr);
+ if (! copyin_char_char(ctx, value, length, buf, indPtr, 0, laddr, fieldId))
+ return;
+ sqlNull(false);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Binary || sqlType.type() == SqlType::Varbinary) {
+ unsigned length = 0;
+ char* value = 0;
+ SqlChar* laddr = 0; // Varbinary length address
+ if (sqlType.type() == SqlType::Binary) {
+ length = sqlType.length();
+ if (length > SqlField_CharSmall)
+ value = reinterpret_cast<char *>(u_data.m_sqlChar);
+ else
+ value = reinterpret_cast<char *>(u_data.m_sqlCharSmall);
+ laddr = 0;
+ } else {
+#if NDB_VERSION_MAJOR >= 3
+ length = sqlType.length();
+ if (2 + length > SqlField_CharSmall)
+ value = reinterpret_cast<char *>(u_data.m_sqlChar + 2);
+ else
+ value = reinterpret_cast<char *>(u_data.m_sqlCharSmall + 2);
+ laddr = (SqlChar*)value - 2;
+#else
+ length = sqlType.length();
+ if (length + 2 > SqlField_CharSmall)
+ value = reinterpret_cast<char *>(u_data.m_sqlChar);
+ else
+ value = reinterpret_cast<char *>(u_data.m_sqlCharSmall);
+ laddr = (SqlChar*)value + length;
+#endif
+ }
+ if (extType.type() == ExtType::Binary) {
+ const char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ int* off = 0;
+ if (extField.m_pos >= 0)
+ off = &extField.m_pos;
+ if (! copyin_binary_binary(ctx, value, length, dataPtr, indPtr, off, laddr, fieldId))
+ return;
+ sqlNull(false);
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Smallint) {
+ SqlSmallint value;
+ if (extType.type() == ExtType::Char) {
+ const char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ SqlBigint x;
+ if (! copyin_signed_char(ctx, &x, dataPtr, fieldId))
+ return;
+ value = x;
+ sqlSmallint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlSmallint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlSmallint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlSmallint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlSmallint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlSmallint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlSmallint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ value = (SqlSmallint)*dataPtr;
+ sqlSmallint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ value = (SqlSmallint)*dataPtr;
+ sqlSmallint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Integer) {
+ SqlInteger value;
+ if (extType.type() == ExtType::Char) {
+ const char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ SqlBigint x;
+ if (! copyin_signed_char(ctx, &x, dataPtr, fieldId))
+ return;
+ value = x;
+ sqlInteger(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlInteger(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlInteger(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlInteger(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlInteger(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlInteger(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlInteger(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ value = (SqlInteger)*dataPtr;
+ sqlInteger(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ value = (SqlInteger)*dataPtr;
+ sqlInteger(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Bigint) {
+ SqlBigint value;
+ if (extType.type() == ExtType::Char) {
+ const char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ SqlBigint x;
+ if (! copyin_signed_char(ctx, &x, dataPtr, fieldId))
+ return;
+ value = x;
+ sqlBigint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlBigint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlBigint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlBigint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlBigint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlBigint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlBigint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ value = (SqlBigint)*dataPtr;
+ sqlBigint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ value = (SqlBigint)*dataPtr;
+ sqlBigint(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Real) {
+ SqlReal value;
+ if (extType.type() == ExtType::Char) {
+ const char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ SqlDouble x;
+ if (! copyin_double_char(ctx, &x, dataPtr, fieldId))
+ return;
+ value = x;
+ sqlReal(x);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlReal(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlReal(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlReal(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlReal(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlReal(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlReal(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlReal(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlReal(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Double) {
+ SqlDouble value;
+ if (extType.type() == ExtType::Char) {
+ const char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ SqlDouble x;
+ if (! copyin_double_char(ctx, &x, dataPtr, fieldId))
+ return;
+ value = x;
+ sqlDouble(x);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlDouble(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlDouble(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlDouble(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlDouble(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlDouble(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlDouble(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlDouble(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ value = *dataPtr;
+ sqlDouble(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Datetime) {
+ SqlDatetime value;
+ if (extType.type() == ExtType::Char) {
+ // XXX replace sscanf by manual scan or regex
+ const char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ int cc = 0;
+ unsigned yy = 0, mm = 0, dd = 0, HH = 0, MM = 0, SS = 0, ff = 0;
+ bool setdate = false;
+ char dummy[10];
+ if (sscanf(dataPtr, "%2d%2u-%2u-%2u %2u:%2u:%2u.%4u%1s", &cc, &yy, &mm, &dd, &HH, &MM, &SS, &ff, dummy) == 8) {
+ ;
+ } else if (sscanf(dataPtr, "%2d%2u-%2u-%2u %2u:%2u:%2u%1s", &cc, &yy, &mm, &dd, &HH, &MM, &SS, dummy) == 7) {
+ ;
+ } else if (sscanf(dataPtr, "%2d%2u-%2u-%2u%1s", &cc, &yy, &mm, &dd, dummy) == 4) {
+ ;
+ } else if (sscanf(dataPtr, "%2u:%2u:%2u.%4u%1s", &HH, &MM, &SS, &ff, dummy) == 4) {
+ setdate = true;
+ } else if (sscanf(dataPtr, "%2u:%2u:%2u%1s", &HH, &MM, &SS, dummy) == 3) {
+ setdate = true;
+ } else {
+ ctx.pushStatus(Sqlstate::_22008, Error::Gen, "invalid timestamp format '%s'", dataPtr);
+ return;
+ }
+ if (setdate) {
+ time_t clock = time(0);
+ struct tm* t = localtime(&clock);
+ cc = (1900 + t->tm_year) / 100;
+ yy = (1900 + t->tm_year) % 100;
+ mm = 1 + t->tm_mon;
+ dd = t->tm_mday;
+ }
+ value.cc(cc);
+ value.yy(yy);
+ value.mm(mm);
+ value.dd(dd);
+ value.HH(HH);
+ value.MM(MM);
+ value.SS(SS);
+ value.ff(ff);
+ // XXX write date routines later
+ if (! value.valid()) {
+ ctx.pushStatus(Sqlstate::_22008, Error::Gen, "invalid timestamp values '%s'", dataPtr);
+ return;
+ }
+ sqlDatetime(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Timestamp) {
+ SQL_TIMESTAMP_STRUCT* dataPtr = static_cast<SQL_TIMESTAMP_STRUCT*>(extField.m_dataPtr);
+ // XXX assume same datatype
+ value.cc(dataPtr->year / 100);
+ value.yy(dataPtr->year / 100);
+ value.mm(dataPtr->month);
+ value.dd(dataPtr->day);
+ value.HH(dataPtr->hour);
+ value.MM(dataPtr->minute);
+ value.SS(dataPtr->second);
+ value.ff(dataPtr->fraction);
+ if (! value.valid()) {
+ ctx.pushStatus(Sqlstate::_22008, Error::Gen, "invalid timestamp struct");
+ return;
+ }
+ sqlDatetime(value);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ ctx_assert(false); // SqlType::Null not applicable
+}
+
+// copy to external
+
+static bool
+copyout_char_char(Ctx& ctx, const char* value, unsigned n, char* ptr, unsigned len, SQLINTEGER* ind, int* off)
+{
+ unsigned n2 = n;
+ if (off != 0 && *off >= 0) {
+ ctx_assert((unsigned)*off <= n2);
+ value += *off;
+ n2 -= *off;
+ if (len < n2 + 1) {
+ ctx.pushStatus(Sqlstate::_01004, Error::Gen, "more data at offset %d, current fetch %u, available %u", *off, len, n2);
+ n2 = len - 1;
+ }
+ } else {
+ if (len < n + 1) { // room for null byte
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "char value '%.*s' overflow (%u < %u)", (int)n, value, (unsigned)len, (unsigned)(len + 1));
+ return false;
+ }
+ }
+ memcpy(ptr, value, n2);
+ ptr[n2] = 0;
+ if (off != 0 && *off >= 0) {
+ if (ind != 0)
+ *ind = n - *off;
+ *off += n2;
+ } else {
+ if (ind != 0)
+ *ind = n;
+ }
+ return true;
+}
+
+static bool
+copyout_binary_binary(Ctx& ctx, const char* value, unsigned n, char* ptr, unsigned len, SQLINTEGER* ind, int* off)
+{
+ unsigned n2 = n;
+ if (off != 0 && *off >= 0) {
+ ctx_assert((unsigned)*off <= n2);
+ value += *off;
+ n2 -= *off;
+ if (len < n2 + 1) {
+ ctx.pushStatus(Sqlstate::_01004, Error::Gen, "more data at offset %d, current fetch %u, available %u", *off, len, n2);
+ n2 = len - 1;
+ }
+ } else {
+ if (len < n) { // no room for null byte
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "binary value '%.*s' overflow (%u < %u)", (int)n, value, (unsigned)len, (unsigned)n);
+ return false;
+ }
+ }
+ memcpy(ptr, value, n2);
+ ptr[n2] = 0;
+ if (off != 0 && *off >= 0) {
+ if (ind != 0)
+ *ind = n - *off;
+ *off += n2;
+ } else {
+ if (ind != 0)
+ *ind = n;
+ }
+ return true;
+}
+
+static bool
+copyout_char_signed(Ctx& ctx, const char* value, unsigned n, long* ptr)
+{
+ while (n > 0 && value[0] == 0x20) {
+ value++;
+ n--;
+ }
+ char buf[200];
+ if (n >= 200)
+ n = 200 - 1;
+ memcpy(buf, value, n);
+ buf[n] = 0;
+ errno = 0;
+ char* endptr = 0;
+ long x = strtol(buf, &endptr, 10);
+ if (endptr == 0 || *endptr != 0) {
+ errno = 0;
+ endptr = 0;
+ double y = strtod(buf, &endptr);
+ if (endptr == 0 || *endptr != 0) {
+ ctx.pushStatus(Sqlstate::_22005, Error::Gen, "value %s not numeric", buf);
+ return false;
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ // XXX should handle 123.000
+ ctx.pushStatus(Sqlstate::_01004, Error::Gen, "value %s truncated", buf);
+ x = static_cast<long>(y);
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ *ptr = x;
+ return true;
+}
+
+static bool
+copyout_char_bigsigned(Ctx& ctx, const char* value, unsigned n, SQLBIGINT* ptr)
+{
+ while (n > 0 && value[0] == 0x20) {
+ value++;
+ n--;
+ }
+ char buf[200];
+ if (n >= 200)
+ n = 200 - 1;
+ memcpy(buf, value, n);
+ buf[n] = 0;
+ errno = 0;
+ char* endptr = 0;
+ SQLBIGINT x = strtoll(buf, &endptr, 10);
+ if (endptr == 0 || *endptr != 0) {
+ errno = 0;
+ endptr = 0;
+ double y = strtod(buf, &endptr);
+ if (endptr == 0 || *endptr != 0) {
+ ctx.pushStatus(Sqlstate::_22005, Error::Gen, "value %s not numeric", buf);
+ return false;
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ // XXX should handle 123.000
+ ctx.pushStatus(Sqlstate::_01004, Error::Gen, "value %s truncated", buf);
+ x = static_cast<long>(y);
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ *ptr = x;
+ return true;
+}
+
+static bool
+copyout_char_unsigned(Ctx& ctx, const char* value, unsigned n, unsigned long* ptr)
+{
+ while (n > 0 && value[0] == 0x20) {
+ value++;
+ n--;
+ }
+ char buf[200];
+ if (n >= 200)
+ n = 200 - 1;
+ memcpy(buf, value, n);
+ buf[n] = 0;
+ errno = 0;
+ char* endptr = 0;
+ unsigned long x = strtoul(buf, &endptr, 10);
+ if (endptr == 0 || *endptr != 0) {
+ errno = 0;
+ endptr = 0;
+ double y = strtod(buf, &endptr);
+ if (endptr == 0 || *endptr != 0) {
+ ctx.pushStatus(Sqlstate::_22005, Error::Gen, "value %s not numeric", buf);
+ return false;
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ // XXX should handle 123.000
+ ctx.pushStatus(Sqlstate::_01004, Error::Gen, "value %s truncated", buf);
+ x = static_cast<unsigned long>(y);
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ *ptr = x;
+ return true;
+}
+
+static bool
+copyout_char_bigunsigned(Ctx& ctx, const char* value, unsigned n, SQLUBIGINT* ptr)
+{
+ while (n > 0 && value[0] == 0x20) {
+ value++;
+ n--;
+ }
+ char buf[200];
+ if (n >= 200)
+ n = 200 - 1;
+ memcpy(buf, value, n);
+ buf[n] = 0;
+ errno = 0;
+ char* endptr = 0;
+ SQLUBIGINT x = strtoull(buf, &endptr, 10);
+ if (endptr == 0 || *endptr != 0) {
+ errno = 0;
+ endptr = 0;
+ double y = strtod(buf, &endptr);
+ if (endptr == 0 || *endptr != 0) {
+ ctx.pushStatus(Sqlstate::_22005, Error::Gen, "value %s not numeric", buf);
+ return false;
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ // XXX should handle 123.000
+ ctx.pushStatus(Sqlstate::_01004, Error::Gen, "value %s truncated", buf);
+ x = static_cast<unsigned long>(y);
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ *ptr = x;
+ return true;
+}
+
+static bool
+copyout_char_double(Ctx& ctx, const char* value, unsigned n, double* ptr)
+{
+ while (n > 0 && value[0] == 0x20) {
+ value++;
+ n--;
+ }
+ char buf[200];
+ if (n >= 200)
+ n = 200 - 1;
+ memcpy(buf, value, n);
+ buf[n] = 0;
+ errno = 0;
+ char* endptr = 0;
+ double x = strtod(value, &endptr);
+ if (endptr == 0 || *endptr != 0) {
+ ctx.pushStatus(Sqlstate::_22005, Error::Gen, "value %s not numeric", value);
+ return false;
+ } else if (errno != 0) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", value);
+ return false;
+ }
+ *ptr = x;
+ return true;
+}
+
+static bool
+copyout_signed_char(Ctx& ctx, Int64 value, char* ptr, int len, SQLINTEGER* ind)
+{
+ char buf[100];
+ sprintf(buf, FMT_I64, value);
+ unsigned n = strlen(buf);
+ if (len <= 0) {
+ ctx.pushStatus(Sqlstate::_HY090, Error::Gen, "invalid output buffer length %d", len);
+ return false;
+ }
+ if ((unsigned)len < n + 1) { // room for null byte
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ strcpy(ptr, buf);
+ if (ind != 0)
+ *ind = n;
+ return true;
+}
+
+static bool
+copyout_unsigned_char(Ctx& ctx, Uint64 uvalue, char* ptr, int len, SQLINTEGER* ind)
+{
+ char buf[100];
+ sprintf(buf, FMT_U64, uvalue);
+ unsigned n = strlen(buf);
+ if (len <= 0) {
+ ctx.pushStatus(Sqlstate::_HY090, Error::Gen, "invalid output buffer length %d", len);
+ return false;
+ }
+ if ((unsigned)len < n + 1) { // room for null byte
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ strcpy(ptr, buf);
+ if (ind != 0)
+ *ind = n;
+ return true;
+}
+
+static bool
+copyout_double_char(Ctx& ctx, double value, unsigned prec, char* ptr, int len, SQLINTEGER* ind)
+{
+ char buf[100];
+ sprintf(buf, "%.*f", (int)prec, value);
+ char* p = buf + strlen(buf);
+ while (p > buf + prec)
+ *--p = 0;
+ while (p > buf && *(p - 1) == '0')
+ *--p = 0;
+ if (p > buf && *(p - 1) == '.') {
+ *p++ = '0';
+ *p = 0;
+ }
+ unsigned n = strlen(buf);
+ if (len <= 0) {
+ ctx.pushStatus(Sqlstate::_HY090, Error::Gen, "invalid output buffer length %d", len);
+ return false;
+ }
+ if ((unsigned)len < n + 1) { // room for null byte
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", buf);
+ return false;
+ }
+ strcpy(ptr, buf);
+ if (ind != 0)
+ *ind = n;
+ return true;
+}
+
+void
+SqlField::copyout(Ctx& ctx, ExtField& extField) const
+{
+ if (extField.extSpec().extType().type() == ExtType::Unbound) {
+ return; // output buffer may be unbound
+ }
+ if (sqlSpec().store() == SqlSpec::Reference) {
+ ctx_assert(u_data.m_sqlField != 0);
+ u_data.m_sqlField->copyout(ctx, extField);
+ return;
+ }
+ SQLINTEGER* indPtr = extField.m_indPtr;
+ if (u_null.m_nullFlag) {
+ if (extField.m_pos > 0) { // second time from SQLGetData
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ if (indPtr == 0) {
+ ctx.pushStatus(Sqlstate::_22002, Error::Gen, "indicator variable required");
+ return;
+ }
+ *indPtr = SQL_NULL_DATA;
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ const SqlType& sqlType = sqlSpec().sqlType();
+ const ExtType& extType = extField.extSpec().extType();
+ if (sqlType.type() == SqlType::Char || sqlType.type() == SqlType::Varchar) {
+ unsigned n = 0;
+ const char* value = 0;
+ if (sqlType.type() == SqlType::Char) {
+ n = sqlType.length();
+ value = reinterpret_cast<const char*>(sqlChar());
+ } else {
+ value = reinterpret_cast<const char*>(sqlVarchar(&n));
+ }
+ if (extType.type() == ExtType::Char) {
+ char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ if (extField.m_dataLen <= 0) {
+ ctx.pushStatus(Sqlstate::_HY090, Error::Gen, "invalid output buffer length %d", (int)extField.m_dataLen);
+ return;
+ }
+ int* off = 0;
+ if (extField.m_pos >= 0) {
+ off = &extField.m_pos;
+ if ((unsigned)*off >= n) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ }
+ if (! copyout_char_char(ctx, value, n, dataPtr, extField.m_dataLen, indPtr, off))
+ return;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ long x;
+ if (! copyout_char_signed(ctx, value, n, &x))
+ return;
+ if (x < SHRT_MIN || x > SHRT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", value);
+ return;
+ }
+ *dataPtr = static_cast<short>(x);
+ if (indPtr != 0)
+ *indPtr = sizeof(short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ unsigned long x;
+ if (! copyout_char_unsigned(ctx, value, n, &x))
+ return;
+ if (x > USHRT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", value);
+ return;
+ }
+ *dataPtr = static_cast<unsigned short>(x);
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ if (! copyout_char_signed(ctx, value, n, dataPtr))
+ return;
+ if (indPtr != 0)
+ *indPtr = sizeof(long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ if (! copyout_char_unsigned(ctx, value, n, dataPtr))
+ return;
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ if (! copyout_char_bigsigned(ctx, value, n, dataPtr))
+ return;
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ if (! copyout_char_bigunsigned(ctx, value, n, dataPtr))
+ return;
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLUBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ double x;
+ if (! copyout_char_double(ctx, value, n, &x))
+ return;
+ if (fabs(x) < FLT_MIN || fabs(x) > FLT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %s overflow", value);
+ return;
+ }
+ *dataPtr = static_cast<float>(x);
+ if (indPtr != 0)
+ *indPtr = sizeof(float);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ double x;
+ if (! copyout_char_double(ctx, value, n, &x))
+ return;
+ *dataPtr = static_cast<double>(x);
+ if (indPtr != 0)
+ *indPtr = sizeof(double);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Binary || sqlType.type() == SqlType::Varbinary) {
+ unsigned n = 0;
+ const char* value = 0;
+ if (sqlType.type() == SqlType::Binary) {
+ n = sqlType.length();
+ value = reinterpret_cast<const char*>(sqlBinary());
+ } else {
+ value = reinterpret_cast<const char*>(sqlVarbinary(&n));
+ }
+ if (extType.type() == ExtType::Binary) {
+ char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ if (extField.m_dataLen <= 0) {
+ ctx.pushStatus(Sqlstate::_HY090, Error::Gen, "invalid output buffer length %d", (int)extField.m_dataLen);
+ return;
+ }
+ int* off = 0;
+ if (extField.m_pos >= 0) {
+ off = &extField.m_pos;
+ if ((unsigned)*off >= n) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ }
+ if (! copyout_binary_binary(ctx, value, n, dataPtr, extField.m_dataLen, indPtr, off))
+ return;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Smallint) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ const SqlSmallint value = sqlSmallint();
+ const SqlUsmallint uvalue = value;
+ if (extType.type() == ExtType::Char) {
+ char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ if (! sqlType.unSigned()) {
+ if (! copyout_signed_char(ctx, value, dataPtr, extField.m_dataLen, indPtr))
+ return;
+ } else {
+ if (! copyout_unsigned_char(ctx, uvalue, dataPtr, extField.m_dataLen, indPtr))
+ return;
+ }
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ *dataPtr = static_cast<short>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ *dataPtr = static_cast<unsigned short>(uvalue);
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ *dataPtr = static_cast<long>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ *dataPtr = static_cast<unsigned long>(uvalue);
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLBIGINT>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLUBIGINT>(uvalue);
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLUBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ *dataPtr = static_cast<float>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(float);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ *dataPtr = static_cast<double>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(double);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Integer) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ const SqlInteger value = sqlInteger();
+ const SqlUinteger uvalue = value;
+ if (extType.type() == ExtType::Char) {
+ char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ if (! sqlType.unSigned()) {
+ if (! copyout_signed_char(ctx, value, dataPtr, extField.m_dataLen, indPtr))
+ return;
+ } else {
+ if (! copyout_unsigned_char(ctx, uvalue, dataPtr, extField.m_dataLen, indPtr))
+ return;
+ }
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ if (value < SHRT_MIN || value > SHRT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %d overflow", (int)value);
+ return;
+ }
+ *dataPtr = static_cast<short>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ if (uvalue > USHRT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %u overflow", uvalue);
+ return;
+ }
+ *dataPtr = static_cast<unsigned short>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ *dataPtr = static_cast<long>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ *dataPtr = static_cast<unsigned long>(uvalue);
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLBIGINT>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLUBIGINT>(uvalue);
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLUBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ *dataPtr = static_cast<float>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(float);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ *dataPtr = static_cast<double>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(double);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Bigint) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ const SqlBigint value = sqlBigint();
+ const SqlUbigint uvalue = value;
+ if (extType.type() == ExtType::Char) {
+ char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ if (! sqlType.unSigned()) {
+ if (! copyout_signed_char(ctx, value, dataPtr, extField.m_dataLen, indPtr))
+ return;
+ } else {
+ if (! copyout_unsigned_char(ctx, uvalue, dataPtr, extField.m_dataLen, indPtr))
+ return;
+ }
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ if (value < SHRT_MIN || value > SHRT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value " FMT_I64 " overflow", (Int64)value);
+ return;
+ }
+ *dataPtr = static_cast<short>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ if (uvalue > USHRT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value " FMT_U64 " overflow", (Uint64)uvalue);
+ return;
+ }
+ *dataPtr = static_cast<short>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ if (value < INT_MIN || value > INT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value " FMT_I64 " overflow", (Int64)value);
+ return;
+ }
+ *dataPtr = static_cast<long>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ if (uvalue > UINT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value " FMT_U64 " overflow", (Uint64)uvalue);
+ return;
+ }
+ *dataPtr = static_cast<unsigned long>(uvalue);
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLBIGINT>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLUBIGINT>(uvalue);
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLUBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ *dataPtr = static_cast<float>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(float);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ *dataPtr = static_cast<double>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(double);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Real) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ const SqlReal value = sqlReal();
+ if (extType.type() == ExtType::Char) {
+ char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ if (! copyout_double_char(ctx, value, 7, dataPtr, extField.m_dataLen, indPtr))
+ return;
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ *dataPtr = static_cast<short>(value); // XXX todo
+ if (indPtr != 0)
+ *indPtr = sizeof(short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ if (value < 0 || value > USHRT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %g overflow", (double)value);
+ return;
+ }
+ *dataPtr = static_cast<short>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ *dataPtr = static_cast<long>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ *dataPtr = static_cast<unsigned long>(value); // XXX todo
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLBIGINT>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLUBIGINT>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLUBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ *dataPtr = static_cast<float>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(float);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ *dataPtr = static_cast<double>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(double);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Double) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ SqlDouble value = sqlDouble();
+ if (extType.type() == ExtType::Char) {
+ char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ if (! copyout_double_char(ctx, value, 14, dataPtr, extField.m_dataLen, indPtr))
+ return;
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Short || extType.type() == ExtType::Sshort) {
+ short* dataPtr = static_cast<short*>(extField.m_dataPtr);
+ *dataPtr = static_cast<short>(value); // XXX todo
+ if (indPtr != 0)
+ *indPtr = sizeof(short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ushort) {
+ unsigned short* dataPtr = static_cast<unsigned short*>(extField.m_dataPtr);
+ if (value < 0 || value > USHRT_MAX) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "value %g overflow", (double)value);
+ return;
+ }
+ *dataPtr = static_cast<short>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned short);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Long || extType.type() == ExtType::Slong) {
+ long* dataPtr = static_cast<long*>(extField.m_dataPtr);
+ *dataPtr = static_cast<long>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ulong) {
+ unsigned long* dataPtr = static_cast<unsigned long*>(extField.m_dataPtr);
+ *dataPtr = static_cast<unsigned long>(value); // XXX todo
+ if (indPtr != 0)
+ *indPtr = sizeof(unsigned long);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Sbigint) {
+ SQLBIGINT* dataPtr = static_cast<SQLBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLBIGINT>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Ubigint) {
+ SQLUBIGINT* dataPtr = static_cast<SQLUBIGINT*>(extField.m_dataPtr);
+ *dataPtr = static_cast<SQLUBIGINT>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(SQLUBIGINT);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Float) {
+ float* dataPtr = static_cast<float*>(extField.m_dataPtr);
+ *dataPtr = static_cast<float>(value); // big enough
+ if (indPtr != 0)
+ *indPtr = sizeof(float);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Double) {
+ double* dataPtr = static_cast<double*>(extField.m_dataPtr);
+ *dataPtr = static_cast<double>(value);
+ if (indPtr != 0)
+ *indPtr = sizeof(double);
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ }
+ if (sqlType.type() == SqlType::Datetime) {
+ if (extField.m_pos > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ SqlDatetime value = sqlDatetime();
+ if (extType.type() == ExtType::Char) {
+ char* dataPtr = static_cast<char*>(extField.m_dataPtr);
+ char buf[100];
+ sprintf(buf, "%02d%02u-%02u-%02u\040%02u:%02u:%02u.%09u", value.cc(), value.yy(), value.mm(), value.dd(), value.HH(), value.MM(), value.SS(), value.ff());
+ int n = strlen(buf);
+ if (extField.m_dataLen < 20) {
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "buffer too small for timestamp %s", buf);
+ return;
+ }
+ if (extField.m_dataLen < n) {
+ ctx.pushStatus(Sqlstate::_01004, Error::Gen, "truncating fractional part of timestamp %s", buf);
+ n = extField.m_dataLen;
+ }
+ if (! copyout_char_char(ctx, buf, n, dataPtr, extField.m_dataLen, indPtr, 0))
+ return;
+ if (extField.m_pos >= 0)
+ extField.m_pos = 1;
+ return;
+ }
+ if (extType.type() == ExtType::Timestamp) {
+ SQL_TIMESTAMP_STRUCT* dataPtr = static_cast<SQL_TIMESTAMP_STRUCT*>(extField.m_dataPtr);
+ // XXX assume same datatype
+ dataPtr->year = value.cc() * 100 + value.yy();
+ dataPtr->month = value.mm();
+ dataPtr->day = value.dd();
+ dataPtr->hour = value.HH();
+ dataPtr->minute = value.MM();
+ dataPtr->second = value.SS();
+ dataPtr->fraction = value.ff();
+ return;
+ }
+ }
+ ctx_assert(false); // SqlType::Null not applicable
+}
+
+void
+SqlField::print(char* buf, unsigned size) const
+{
+ Ctx ctx;
+ unsigned n = sqlSpec().sqlType().displaySize();
+ SQLINTEGER ind = 0;
+ ExtType extType(ExtType::Char);
+ ExtSpec extSpec(extType);
+ ExtField extField(extSpec, (SQLPOINTER)buf, size, &ind);
+ buf[0] = 0;
+ copyout(ctx, extField);
+ if (ind == SQL_NULL_DATA)
+ snprintf(buf, size, "NULL");
+}
diff --git a/ndb/src/client/odbc/common/DataField.hpp b/ndb/src/client/odbc/common/DataField.hpp
new file mode 100644
index 00000000000..65138df25f1
--- /dev/null
+++ b/ndb/src/client/odbc/common/DataField.hpp
@@ -0,0 +1,446 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_DataField_hpp
+#define ODBC_COMMON_DataField_hpp
+
+#include <NdbApi.hpp>
+#include <common/common.hpp>
+#include "DataType.hpp"
+
+/**
+ * @class SqlSpec
+ * @brief Specification of data in SQL format
+ */
+class SqlSpec {
+public:
+ enum Store {
+ Undef = 0,
+ Reference = 1, // reference to read-only SqlField of same type
+ Physical = 2 // stored within or in allocated storage
+ };
+ SqlSpec();
+ SqlSpec(const SqlType& sqlType, Store store);
+ SqlSpec(const SqlSpec& sqlSpec);
+ SqlSpec(const SqlSpec& sqlSpec, Store store);
+ const SqlType& sqlType() const;
+ const Store store() const;
+ unsigned size() const;
+private:
+ //SqlSpec& operator=(const SqlSpec& sqlSpec); // disallowed
+ SqlType m_sqlType;
+ Store m_store;
+};
+
+/**
+ * @class ExtSpec
+ * @brief Specification of data in external format
+ */
+class ExtSpec {
+public:
+ ExtSpec();
+ ExtSpec(const ExtType& extType);
+ ExtSpec(const ExtSpec& extSpec);
+ const ExtType& extType() const;
+ unsigned size() const;
+ void setValue(const ExtType& extType);
+private:
+ ExtType m_extType;
+};
+
+/**
+ * @class LexSpec
+ * @brief Specification of lexical data
+ *
+ * Used only for converting lexical data to SQL data.
+ */
+class LexSpec {
+public:
+ LexSpec();
+ LexSpec(const LexType& lexType);
+ /**
+ * Lexical data is represented as string. Following
+ * converts it to SQL data.
+ */
+ void convert(Ctx& ctx, const BaseString& value, class SqlField& out);
+private:
+ LexType m_lexType;
+};
+
+// SqlSpec
+
+inline
+SqlSpec::SqlSpec() :
+ m_store(Undef)
+{
+}
+
+inline
+SqlSpec::SqlSpec(const SqlType& sqlType, Store store) :
+ m_sqlType(sqlType),
+ m_store(store)
+{
+}
+
+inline
+SqlSpec::SqlSpec(const SqlSpec& sqlSpec) :
+ m_sqlType(sqlSpec.m_sqlType),
+ m_store(sqlSpec.m_store)
+{
+}
+
+inline
+SqlSpec::SqlSpec(const SqlSpec& sqlSpec, Store store) :
+ m_sqlType(sqlSpec.m_sqlType),
+ m_store(store)
+{
+}
+
+inline const SqlType&
+SqlSpec::sqlType() const
+{
+ return m_sqlType;
+}
+
+inline const SqlSpec::Store
+SqlSpec::store() const
+{
+ return m_store;
+}
+
+inline unsigned
+SqlSpec::size() const
+{
+ return sqlType().size();
+}
+
+// ExtSpec
+
+inline
+ExtSpec::ExtSpec()
+{
+}
+
+inline
+ExtSpec::ExtSpec(const ExtType& extType) :
+ m_extType(extType)
+{
+}
+
+inline
+ExtSpec::ExtSpec(const ExtSpec& extSpec) :
+ m_extType(extSpec.m_extType)
+{
+}
+
+inline const ExtType&
+ExtSpec::extType() const
+{
+ return m_extType;
+}
+
+inline unsigned
+ExtSpec::size() const
+{
+ return m_extType.size();
+}
+
+inline void
+ExtSpec::setValue(const ExtType& extType)
+{
+ m_extType = extType;
+}
+
+// LexSpec
+
+inline
+LexSpec::LexSpec(const LexType& lexType) :
+ m_lexType(lexType)
+{
+}
+
+/**
+ * @class SqlField
+ * @brief Sql data field accessor
+ */
+class SqlField {
+public:
+ SqlField();
+ SqlField(const SqlSpec& sqlSpec);
+ SqlField(const SqlSpec& sqlSpec, const SqlField* sqlField);
+ SqlField(const SqlField& sqlField);
+ ~SqlField();
+ const SqlSpec& sqlSpec() const;
+ const void* addr() const; // address of data
+ void* addr();
+ unsigned allocSize() const;
+ const SqlChar* sqlChar() const; // get
+ const SqlChar* sqlVarchar(unsigned* length) const;
+ const SqlChar* sqlBinary() const;
+ const SqlChar* sqlVarbinary(unsigned* length) const;
+ SqlSmallint sqlSmallint() const;
+ SqlInteger sqlInteger() const;
+ SqlBigint sqlBigint() const;
+ SqlReal sqlReal() const;
+ SqlDouble sqlDouble() const;
+ SqlDatetime sqlDatetime() const;
+ void sqlChar(const char* value, int length); // set
+ void sqlChar(const SqlChar* value, int length);
+ void sqlVarchar(const char* value, int length);
+ void sqlVarchar(const SqlChar* value, int length);
+ void sqlBinary(const char* value, int length);
+ void sqlBinary(const SqlChar* value, int length);
+ void sqlVarbinary(const char* value, int length);
+ void sqlVarbinary(const SqlChar* value, int length);
+ void sqlSmallint(SqlSmallint value);
+ void sqlInteger(SqlInteger value);
+ void sqlBigint(SqlBigint value);
+ void sqlReal(SqlReal value);
+ void sqlDouble(SqlDouble value);
+ void sqlDatetime(SqlDatetime value);
+ bool sqlNull() const; // get and set null
+ void sqlNull(bool value);
+ unsigned trim() const; // right trimmed length
+ void copy(Ctx& ctx, SqlField& out) const;
+ bool cast(Ctx& ctx, SqlField& out) const;
+ bool less(const SqlField& sqlField) const;
+ // application input and output
+ void copyin(Ctx& ctx, class ExtField& extField);
+ void copyout(Ctx& ctx, class ExtField& extField) const;
+ // print for debugging
+ void print(char* buf, unsigned size) const;
+ // public for forte6
+ //enum { CharSmall = 20 };
+#define SqlField_CharSmall 20 // redhat-6.2 (egcs-2.91.66)
+private:
+ friend class LexSpec;
+ friend class SqlRow;
+ void alloc(); // allocate Physical
+ void alloc(const SqlField& sqlField);
+ void free(); // free Physical
+ SqlSpec m_sqlSpec;
+ union Data {
+ Data();
+ Data(const SqlField* sqlField);
+ const SqlField* m_sqlField;
+ // Physical
+ SqlChar* m_sqlChar; // all char types
+ SqlChar m_sqlCharSmall[SqlField_CharSmall];
+ SqlSmallint m_sqlSmallint;
+ SqlInteger m_sqlInteger;
+ SqlBigint m_sqlBigint;
+ SqlReal m_sqlReal;
+ SqlDouble m_sqlDouble;
+ SqlDatetime m_sqlDatetime;
+ } u_data;
+ union Null {
+ Null();
+ bool m_nullFlag;
+ } u_null;
+};
+
+/**
+ * @class ExtField
+ * @brief External data field accessor
+ */
+class ExtField {
+public:
+ ExtField();
+ ExtField(const ExtSpec& extSpec, int fieldId = 0);
+ ExtField(const ExtSpec& extSpec, SQLPOINTER dataPtr, SQLINTEGER dataLen, SQLINTEGER* indPtr, int fieldId = 0);
+ ~ExtField();
+ const ExtSpec& extSpec() const;
+ void setValue(SQLPOINTER dataPtr, SQLINTEGER dataLen);
+ void setValue(const ExtSpec& extSpec, SQLPOINTER dataPtr, SQLINTEGER dataLen, SQLINTEGER* indPtr);
+ int fieldId() const;
+ void setPos(int pos);
+ int getPos() const;
+private:
+ friend class SqlField;
+ friend class Exec_root;
+ ExtSpec m_extSpec;
+ SQLPOINTER m_dataPtr; // data buffer
+ SQLINTEGER m_dataLen; // data buffer length
+ SQLINTEGER* m_indPtr; // null indicator or length
+ int m_fieldId; // field id > 0 for error messages
+ int m_pos; // output position for SQLGetData (if != -1)
+};
+
+inline int
+ExtField::fieldId() const
+{
+ return m_fieldId;
+}
+
+inline void
+ExtField::setPos(int pos)
+{
+ m_pos = pos;
+}
+
+inline int
+ExtField::getPos() const
+{
+ return m_pos;
+}
+
+// SqlField
+
+inline
+SqlField::SqlField()
+{
+}
+
+inline
+SqlField::SqlField(const SqlSpec& sqlSpec) :
+ m_sqlSpec(sqlSpec)
+{
+ if (m_sqlSpec.store() == SqlSpec::Physical)
+ alloc();
+}
+
+inline
+SqlField::SqlField(const SqlSpec& sqlSpec, const SqlField* sqlField) :
+ m_sqlSpec(sqlSpec),
+ u_data(sqlField)
+{
+ ctx_assert(m_sqlSpec.store() == SqlSpec::Reference);
+}
+
+inline
+SqlField::SqlField(const SqlField& sqlField) :
+ m_sqlSpec(sqlField.m_sqlSpec),
+ u_data(sqlField.u_data),
+ u_null(sqlField.u_null)
+{
+ if (m_sqlSpec.store() == SqlSpec::Physical)
+ alloc(sqlField);
+}
+
+inline
+SqlField::Data::Data()
+{
+}
+
+inline
+SqlField::Data::Data(const SqlField* sqlField)
+{
+ m_sqlField = sqlField;
+}
+
+inline
+SqlField::Null::Null()
+{
+}
+
+inline
+SqlField::~SqlField()
+{
+ if (m_sqlSpec.store() == SqlSpec::Physical)
+ free();
+}
+
+inline const SqlSpec&
+SqlField::sqlSpec() const
+{
+ return m_sqlSpec;
+}
+
+inline void
+SqlField::sqlChar(const char* value, int length)
+{
+ sqlChar(reinterpret_cast<const SqlChar*>(value), length);
+}
+
+inline void
+SqlField::sqlVarchar(const char* value, int length)
+{
+ sqlVarchar(reinterpret_cast<const SqlChar*>(value), length);
+}
+
+inline void
+SqlField::sqlBinary(const char* value, int length)
+{
+ sqlBinary(reinterpret_cast<const SqlChar*>(value), length);
+}
+
+inline void
+SqlField::sqlVarbinary(const char* value, int length)
+{
+ sqlVarbinary(reinterpret_cast<const SqlChar*>(value), length);
+}
+
+// ExtField
+
+inline
+ExtField::ExtField() :
+ m_dataPtr(0),
+ m_dataLen(0),
+ m_indPtr(0),
+ m_pos(-1)
+{
+}
+
+inline
+ExtField::ExtField(const ExtSpec& extSpec, int fieldId) :
+ m_extSpec(extSpec),
+ m_dataPtr(0),
+ m_dataLen(0),
+ m_indPtr(0),
+ m_fieldId(fieldId),
+ m_pos(-1)
+{
+}
+
+inline
+ExtField::ExtField(const ExtSpec& extSpec, SQLPOINTER dataPtr, SQLINTEGER dataLen, SQLINTEGER* indPtr, int fieldId) :
+ m_extSpec(extSpec),
+ m_dataPtr(dataPtr),
+ m_dataLen(dataLen),
+ m_indPtr(indPtr),
+ m_fieldId(fieldId),
+ m_pos(-1)
+{
+}
+
+inline
+ExtField::~ExtField()
+{
+}
+
+inline const ExtSpec&
+ExtField::extSpec() const
+{
+ return m_extSpec;
+}
+
+inline void
+ExtField::setValue(SQLPOINTER dataPtr, SQLINTEGER dataLen)
+{
+ m_dataPtr = dataPtr;
+ m_dataLen = dataLen;
+}
+
+inline void
+ExtField::setValue(const ExtSpec& extSpec, SQLPOINTER dataPtr, SQLINTEGER dataLen, SQLINTEGER* indPtr)
+{
+ m_extSpec.setValue(extSpec.extType());
+ m_dataPtr = dataPtr;
+ m_dataLen = dataLen;
+ m_indPtr = indPtr;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/DataRow.cpp b/ndb/src/client/odbc/common/DataRow.cpp
new file mode 100644
index 00000000000..509f2673e0d
--- /dev/null
+++ b/ndb/src/client/odbc/common/DataRow.cpp
@@ -0,0 +1,140 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "DataRow.hpp"
+
+// SqlSpecs
+
+SqlSpecs::SqlSpecs(unsigned count) :
+ m_count(count)
+{
+ m_sqlSpec = new SqlSpec[1 + count];
+}
+
+SqlSpecs::SqlSpecs(const SqlSpecs& sqlSpecs) :
+ m_count(sqlSpecs.m_count)
+{
+ m_sqlSpec = new SqlSpec[1 + m_count];
+ for (unsigned i = 1; i <= m_count; i++) {
+ void* place = static_cast<void*>(&m_sqlSpec[i]);
+ new (place) SqlSpec(sqlSpecs.m_sqlSpec[i]);
+ }
+}
+
+SqlSpecs::~SqlSpecs()
+{
+ delete[] m_sqlSpec;
+}
+
+// ExtSpecs
+
+ExtSpecs::ExtSpecs(unsigned count) :
+ m_count(count)
+{
+ m_extSpec = new ExtSpec[1 + count];
+}
+
+ExtSpecs::ExtSpecs(const ExtSpecs& extSpecs) :
+ m_count(extSpecs.m_count)
+{
+ m_extSpec = new ExtSpec[1 + m_count];
+ for (unsigned i = 1; i <= m_count; i++) {
+ void* place = static_cast<void*>(&m_extSpec[i]);
+ new (place) ExtSpec(extSpecs.m_extSpec[i]);
+ }
+}
+
+ExtSpecs::~ExtSpecs()
+{
+ delete[] m_extSpec;
+}
+
+// SqlRow
+
+SqlRow::SqlRow(const SqlSpecs& sqlSpecs) :
+ m_sqlSpecs(sqlSpecs)
+{
+ m_sqlField = new SqlField[1 + count()];
+ for (unsigned i = 1; i <= count(); i++) {
+ SqlField sqlField(m_sqlSpecs.getEntry(i));
+ setEntry(i, sqlField);
+ }
+}
+
+SqlRow::SqlRow(const SqlRow& sqlRow) :
+ m_sqlSpecs(sqlRow.m_sqlSpecs)
+{
+ m_sqlField = new SqlField[1 + count()];
+ for (unsigned i = 1; i <= count(); i++) {
+ void* place = static_cast<void*>(&m_sqlField[i]);
+ new (place) SqlField(sqlRow.getEntry(i));
+ }
+}
+
+SqlRow::~SqlRow()
+{
+ for (unsigned i = 1; i <= count(); i++) {
+ m_sqlField[i].~SqlField();
+ }
+ delete[] m_sqlField;
+}
+
+SqlRow*
+SqlRow::copy() const
+{
+ SqlRow* copyRow = new SqlRow(m_sqlSpecs);
+ for (unsigned i = 1; i <= count(); i++) {
+ const SqlField* sqlField = &m_sqlField[i];
+ while (sqlField->sqlSpec().store() == SqlSpec::Reference) {
+ sqlField = sqlField->u_data.m_sqlField;
+ }
+ copyRow->setEntry(i, *sqlField);
+ }
+ return copyRow;
+}
+
+void
+SqlRow::copyout(Ctx& ctx, class ExtRow& extRow) const
+{
+ for (unsigned i = 1; i <= count(); i++) {
+ const SqlField& sqlField = getEntry(i);
+ ExtField& extField = extRow.getEntry(i);
+ sqlField.copyout(ctx, extField);
+ }
+}
+
+// ExtRow
+
+ExtRow::ExtRow(const ExtSpecs& extSpecs) :
+ m_extSpecs(extSpecs)
+{
+ m_extField = new ExtField[1 + count()];
+}
+
+ExtRow::ExtRow(const ExtRow& extRow) :
+ m_extSpecs(extRow.m_extSpecs)
+{
+ m_extField = new ExtField[1 + count()];
+ for (unsigned i = 1; i <= count(); i++) {
+ void* place = static_cast<void*>(&m_extField[i]);
+ new (place) ExtField(extRow.getEntry(i));
+ }
+}
+
+ExtRow::~ExtRow()
+{
+ delete[] m_extField;
+}
diff --git a/ndb/src/client/odbc/common/DataRow.hpp b/ndb/src/client/odbc/common/DataRow.hpp
new file mode 100644
index 00000000000..4a5a1e905b9
--- /dev/null
+++ b/ndb/src/client/odbc/common/DataRow.hpp
@@ -0,0 +1,185 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_DataRow_hpp
+#define ODBC_COMMON_DataRow_hpp
+
+#include <new>
+#include <common/common.hpp>
+#include "DataField.hpp"
+
+class Ctx;
+
+/**
+ * @class SqlSpecs
+ * @brief Specification of row of SQL data
+ */
+class SqlSpecs {
+public:
+ SqlSpecs(unsigned count);
+ SqlSpecs(const SqlSpecs& sqlSpecs);
+ ~SqlSpecs();
+ unsigned count() const;
+ void setEntry(unsigned i, const SqlSpec& sqlSpec);
+ const SqlSpec& getEntry(unsigned i) const;
+private:
+ SqlSpecs& operator=(const SqlSpecs& sqlSpecs); // disallowed
+ const unsigned m_count;
+ SqlSpec* m_sqlSpec;
+};
+
+inline unsigned
+SqlSpecs::count() const
+{
+ return m_count;
+}
+
+inline void
+SqlSpecs::setEntry(unsigned i, const SqlSpec& sqlSpec)
+{
+ ctx_assert(m_sqlSpec != 0 && 1 <= i && i <= m_count);
+ void* place = static_cast<void*>(&m_sqlSpec[i]);
+ new (place) SqlSpec(sqlSpec);
+}
+
+inline const SqlSpec&
+SqlSpecs::getEntry(unsigned i) const
+{
+ ctx_assert(m_sqlSpec != 0 && 1 <= i && i <= m_count);
+ return m_sqlSpec[i];
+}
+
+/**
+ * @class ExtSpecs
+ * @brief Specification of row of external data
+ */
+class ExtSpecs {
+public:
+ ExtSpecs(unsigned count);
+ ExtSpecs(const ExtSpecs& extSpecs);
+ ~ExtSpecs();
+ unsigned count() const;
+ void setEntry(unsigned i, const ExtSpec& extSpec);
+ const ExtSpec& getEntry(unsigned i) const;
+private:
+ ExtSpecs& operator=(const ExtSpecs& extSpecs); // disallowed
+ const unsigned m_count;
+ ExtSpec* m_extSpec;
+};
+
+inline unsigned
+ExtSpecs::count() const
+{
+ return m_count;
+}
+
+inline void
+ExtSpecs::setEntry(unsigned i, const ExtSpec& extSpec)
+{
+ ctx_assert(m_extSpec != 0 && 1 <= i && i <= m_count);
+ void* place = static_cast<void*>(&m_extSpec[i]);
+ new (place) ExtSpec(extSpec);
+}
+
+inline const ExtSpec&
+ExtSpecs::getEntry(unsigned i) const
+{
+ ctx_assert(m_extSpec != 0 && 1 <= i && i <= m_count);
+ return m_extSpec[i];
+}
+
+/**
+ * @class SqlRow
+ * @brief Sql data row
+ */
+class SqlRow {
+public:
+ SqlRow(const SqlSpecs& sqlSpecs);
+ SqlRow(const SqlRow& sqlRow);
+ ~SqlRow();
+ unsigned count() const;
+ void setEntry(unsigned i, const SqlField& sqlField);
+ SqlField& getEntry(unsigned i) const;
+ SqlRow* copy() const;
+ void copyout(Ctx& ctx, class ExtRow& extRow) const;
+private:
+ SqlRow& operator=(const SqlRow& sqlRow); // disallowed
+ SqlSpecs m_sqlSpecs;
+ SqlField* m_sqlField;
+};
+
+inline unsigned
+SqlRow::count() const
+{
+ return m_sqlSpecs.count();
+}
+
+inline void
+SqlRow::setEntry(unsigned i, const SqlField& sqlField)
+{
+ ctx_assert(1 <= i && i <= count() && m_sqlField != 0);
+ m_sqlField[i].~SqlField();
+ void* place = static_cast<void*>(&m_sqlField[i]);
+ new (place) SqlField(sqlField);
+}
+
+inline SqlField&
+SqlRow::getEntry(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= count() && m_sqlField != 0);
+ return m_sqlField[i];
+}
+
+/**
+ * @class ExtRow
+ * @brief External data row
+ */
+class ExtRow {
+public:
+ ExtRow(const ExtSpecs& extSpecs);
+ ExtRow(const ExtRow& extRow);
+ ~ExtRow();
+ unsigned count() const;
+ void setEntry(unsigned i, const ExtField& extField);
+ ExtField& getEntry(unsigned i) const;
+private:
+ ExtRow& operator=(const ExtRow& extRow); // disallowed
+ ExtSpecs m_extSpecs;
+ ExtField* m_extField;
+};
+
+inline unsigned
+ExtRow::count() const
+{
+ return m_extSpecs.count();
+}
+
+inline void
+ExtRow::setEntry(unsigned i, const ExtField& extField)
+{
+ ctx_assert(1 <= i && i <= count() && m_extField != 0);
+ void* place = static_cast<void*>(&m_extField[i]);
+ new (place) ExtField(extField);
+}
+
+inline ExtField&
+ExtRow::getEntry(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= count() && m_extField != 0);
+ return m_extField[i];
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/DataType.cpp b/ndb/src/client/odbc/common/DataType.cpp
new file mode 100644
index 00000000000..62bd622b9b5
--- /dev/null
+++ b/ndb/src/client/odbc/common/DataType.cpp
@@ -0,0 +1,546 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "DataType.hpp"
+#include <NdbStdio.h>
+
+// SqlType
+
+SqlType::SqlType() :
+ m_type(Undef)
+{
+}
+
+SqlType::SqlType(Type type, bool nullable)
+{
+ Ctx ctx;
+ setType(ctx, type, nullable);
+ ctx_assert(ctx.ok());
+}
+
+SqlType::SqlType(Type type, unsigned length, bool nullable)
+{
+ Ctx ctx;
+ setType(ctx, type, length, nullable);
+ ctx_assert(ctx.ok());
+}
+
+SqlType::SqlType(Type type, unsigned precision, unsigned scale, bool nullable)
+{
+ Ctx ctx;
+ setType(ctx, type, precision, scale, nullable);
+ ctx_assert(ctx.ok());
+}
+
+SqlType::SqlType(Ctx& ctx, Type type, bool nullable)
+{
+ setType(ctx, type, nullable);
+}
+
+SqlType::SqlType(Ctx& ctx, Type type, unsigned length, bool nullable)
+{
+ setType(ctx, type, length, nullable);
+}
+
+SqlType::SqlType(Ctx& ctx, Type type, unsigned precision, unsigned scale, bool nullable)
+{
+ setType(ctx, type, precision, scale, nullable);
+}
+
+SqlType::SqlType(Ctx& ctx, const NdbDictionary::Column* ndbColumn)
+{
+ setType(ctx, ndbColumn);
+}
+
+void
+SqlType::setType(Ctx& ctx, Type type, bool nullable)
+{
+ switch (type) {
+ case Smallint:
+ case Integer:
+ case Bigint:
+ case Real:
+ case Double:
+ case Datetime:
+ break;
+ case Blob:
+ setType(ctx, Varbinary, FAKE_BLOB_SIZE, nullable); // XXX BLOB hack
+ return;
+ case Null:
+ case Unbound:
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ m_type = type;
+ m_precision = 0;
+ m_scale = 0;
+ m_length = 0;
+ m_nullable = nullable;
+ m_unSigned = false;
+}
+
+void
+SqlType::setType(Ctx& ctx, Type type, unsigned length, bool nullable)
+{
+ switch (type) {
+ case Char:
+ case Varchar:
+ case Binary:
+ case Varbinary:
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ m_type = type;
+ m_precision = 0;
+ m_scale = 0;
+ m_length = length;
+ m_nullable = nullable;
+ m_unSigned = false;
+}
+
+void
+SqlType::setType(Ctx& ctx, Type type, unsigned precision, unsigned scale, bool nullable)
+{
+ ctx_assert(false); // not yet
+}
+
+void
+SqlType::setType(Ctx& ctx, const NdbDictionary::Column* ndbColumn)
+{
+ NdbDictionary::Column::Type type = ndbColumn->getType();
+ unsigned length = ndbColumn->getLength();
+ unsigned precision = ndbColumn->getPrecision();
+ unsigned scale = ndbColumn->getScale();
+ bool nullable = ndbColumn->getNullable();
+ switch (type) {
+ case NdbDictionary::Column::Undefined:
+ break;
+ case NdbDictionary::Column::Int:
+ if (length == 1)
+ setType(ctx, Integer, nullable);
+ else
+ setType(ctx, Binary, length * sizeof(SqlInteger), nullable);
+ return;
+ case NdbDictionary::Column::Unsigned:
+ if (length == 1) {
+ setType(ctx, Integer, nullable);
+ unSigned(true);
+ } else
+ setType(ctx, Binary, length * sizeof(SqlUinteger), nullable);
+ return;
+ case NdbDictionary::Column::Bigint:
+ if (length == 1)
+ setType(ctx, Bigint, nullable);
+ else
+ setType(ctx, Binary, length * sizeof(SqlBigint), nullable);
+ return;
+ case NdbDictionary::Column::Bigunsigned:
+ if (length == 1) {
+ setType(ctx, Bigint, nullable);
+ unSigned(true);
+ } else
+ setType(ctx, Binary, length * sizeof(SqlBigint), nullable);
+ return;
+ case NdbDictionary::Column::Float:
+ if (length == 1)
+ setType(ctx, Real, nullable);
+ else
+ setType(ctx, Binary, length * sizeof(SqlReal), nullable);
+ return;
+ case NdbDictionary::Column::Double:
+ if (length == 1)
+ setType(ctx, Double, nullable);
+ else
+ setType(ctx, Binary, length * sizeof(SqlDouble), nullable);
+ return;
+ case NdbDictionary::Column::Decimal:
+ setType(ctx, Decimal, precision, scale, nullable);
+ return;
+ case NdbDictionary::Column::Char:
+ setType(ctx, Char, length, nullable);
+ return;
+ case NdbDictionary::Column::Varchar:
+ setType(ctx, Varchar, length, nullable);
+ return;
+ case NdbDictionary::Column::Binary:
+ setType(ctx, Binary, length, nullable);
+ return;
+ case NdbDictionary::Column::Varbinary:
+ setType(ctx, Varbinary, length, nullable);
+ return;
+ case NdbDictionary::Column::Datetime:
+ // XXX not yet
+ break;
+ case NdbDictionary::Column::Timespec:
+ setType(ctx, Datetime, nullable);
+ return;
+ case NdbDictionary::Column::Blob:
+ setType(ctx, Blob, nullable);
+ return;
+ default:
+ break;
+ }
+ ctx.pushStatus(Error::Gen, "unsupported NDB type %d", (signed)type);
+}
+
+bool
+SqlType::equal(const SqlType& sqlType) const
+{
+ return
+ m_type == sqlType.m_type &&
+ m_precision == sqlType.m_precision &&
+ m_scale == sqlType.m_scale &&
+ m_length == sqlType.m_length;
+}
+
+unsigned
+SqlType::size() const
+{
+ switch (m_type) {
+ case Char:
+ case Varchar:
+ case Binary:
+ case Varbinary:
+ return m_length;
+ case Smallint:
+ return sizeof(SqlSmallint);
+ case Integer:
+ return sizeof(SqlInteger);
+ case Bigint:
+ return sizeof(SqlBigint);
+ case Real:
+ return sizeof(SqlReal);
+ case Double:
+ return sizeof(SqlDouble);
+ case Datetime:
+ return sizeof(SqlDatetime);
+ case Null:
+ return 0;
+ default:
+ break;
+ }
+ ctx_assert(false);
+ return 0;
+}
+
+unsigned
+SqlType::displaySize() const
+{
+ switch (m_type) {
+ case Char:
+ case Varchar:
+ return m_length;
+ case Binary:
+ case Varbinary:
+ return m_length;
+ case Smallint:
+ return m_unSigned ? 5 : 6;
+ case Integer:
+ return m_unSigned ? 10 : 11;
+ case Bigint:
+ return m_unSigned ? 20 : 21;
+ case Real:
+ return 10;
+ case Double:
+ return 20;
+ case Datetime:
+ return 30;
+ case Null:
+ return 0;
+ default:
+ break;
+ }
+ ctx_assert(false);
+ return 0;
+}
+
+void
+SqlType::getType(Ctx& ctx, NdbDictionary::Column* ndbColumn) const
+{
+ switch (m_type) {
+ case Char:
+ ndbColumn->setType(NdbDictionary::Column::Char);
+ ndbColumn->setLength(m_length);
+ break;
+ case Varchar:
+ ndbColumn->setType(NdbDictionary::Column::Varchar);
+ ndbColumn->setLength(m_length);
+ break;
+ case Binary:
+ ndbColumn->setType(NdbDictionary::Column::Binary);
+ ndbColumn->setLength(m_length);
+ break;
+ case Varbinary:
+ ndbColumn->setType(NdbDictionary::Column::Varbinary);
+ ndbColumn->setLength(m_length);
+ break;
+ case Smallint:
+ break; // XXX
+ case Integer:
+ if (! m_unSigned)
+ ndbColumn->setType(NdbDictionary::Column::Int);
+ else
+ ndbColumn->setType(NdbDictionary::Column::Unsigned);
+ ndbColumn->setLength(1);
+ break;
+ case Bigint:
+ if (! m_unSigned)
+ ndbColumn->setType(NdbDictionary::Column::Bigint);
+ else
+ ndbColumn->setType(NdbDictionary::Column::Bigunsigned);
+ ndbColumn->setLength(1);
+ break;
+ case Real:
+ ndbColumn->setType(NdbDictionary::Column::Float);
+ ndbColumn->setLength(1);
+ break;
+ case Double:
+ ndbColumn->setType(NdbDictionary::Column::Double);
+ ndbColumn->setLength(1);
+ break;
+ case Datetime:
+ ndbColumn->setType(NdbDictionary::Column::Timespec);
+ ndbColumn->setLength(1);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ ndbColumn->setNullable(m_nullable);
+}
+
+const char*
+SqlType::typeName() const
+{
+ switch (m_type) {
+ case Char:
+ return "CHAR";
+ case Varchar:
+ return "VARCHAR";
+ case Binary:
+ return "BINARY";
+ case Varbinary:
+ return "VARBINARY";
+ case Smallint:
+ return "SMALLINT";
+ case Integer:
+ return "INTEGER";
+ case Bigint:
+ return "BIGINT";
+ case Real:
+ return "REAL";
+ case Double:
+ return "FLOAT";
+ case Datetime:
+ return "DATETIME";
+ default:
+ break;
+ }
+ return "UNKNOWN";
+}
+
+void
+SqlType::print(char* buf, unsigned size) const
+{
+ switch (m_type) {
+ case Char:
+ snprintf(buf, size, "char(%d)", m_length);
+ break;
+ case Varchar:
+ snprintf(buf, size, "varchar(%d)", m_length);
+ break;
+ case Binary:
+ snprintf(buf, size, "binary(%d)", m_length);
+ break;
+ case Varbinary:
+ snprintf(buf, size, "varbinary(%d)", m_length);
+ break;
+ case Smallint:
+ snprintf(buf, size, "smallint%s", m_unSigned ? " unsigned" : "");
+ break;
+ case Integer:
+ snprintf(buf, size, "integer%s", m_unSigned ? " unsigned" : "");
+ break;
+ case Bigint:
+ snprintf(buf, size, "bigint%s", m_unSigned ? " unsigned" : "");
+ break;
+ case Real:
+ snprintf(buf, size, "real");
+ break;
+ case Double:
+ snprintf(buf, size, "double");
+ break;
+ case Datetime:
+ snprintf(buf, size, "datetime");
+ break;
+ case Null:
+ snprintf(buf, size, "null");
+ break;
+ case Unbound:
+ snprintf(buf, size, "unbound");
+ break;
+ default:
+ snprintf(buf, size, "sqltype(%d)", (int)m_type);
+ break;
+ }
+}
+
+// ExtType
+
+ExtType::ExtType() :
+ m_type(Undef)
+{
+}
+
+ExtType::ExtType(Type type)
+{
+ Ctx ctx;
+ setType(ctx, type);
+ ctx_assert(ctx.ok());
+}
+
+ExtType::ExtType(Ctx& ctx, Type type)
+{
+ setType(ctx, type);
+}
+
+void
+ExtType::setType(Ctx& ctx, Type type)
+{
+ switch (type) {
+ case Char:
+ case Short:
+ case Sshort:
+ case Ushort:
+ case Long:
+ case Slong:
+ case Ulong:
+ case Sbigint:
+ case Ubigint:
+ case Float:
+ case Double:
+ case Timestamp:
+ case Binary: // XXX BLOB hack
+ case Unbound:
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "unsupported external type %d", (int)type);
+ return;
+ }
+ m_type = type;
+}
+
+unsigned
+ExtType::size() const
+{
+ ctx_assert(false);
+ return 0;
+}
+
+// LexType
+
+LexType::LexType() :
+ m_type(Undef)
+{
+}
+
+LexType::LexType(Type type)
+{
+ Ctx ctx;
+ setType(ctx, type);
+ ctx_assert(ctx.ok());
+}
+
+LexType::LexType(Ctx& ctx, Type type)
+{
+ setType(ctx, type);
+}
+
+void
+LexType::setType(Ctx& ctx, Type type)
+{
+ switch (type) {
+ case Char:
+ case Integer:
+ case Float:
+ case Null:
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ m_type = type;
+}
+
+// convert types
+
+SQLSMALLINT
+SqlType::sqlcdefault(Ctx& ctx) const
+{
+ switch (m_type) {
+ case Char:
+ return SQL_C_CHAR;
+ case Varchar:
+ return SQL_C_CHAR;
+ case Binary:
+ return SQL_C_BINARY;
+ case Varbinary:
+ return SQL_C_BINARY;
+ case Smallint:
+ return m_unSigned ? SQL_C_USHORT : SQL_C_SSHORT;
+ case Integer:
+ return m_unSigned ? SQL_C_ULONG : SQL_C_SLONG;
+ case Bigint:
+ return SQL_C_CHAR;
+ // or maybe this
+ return m_unSigned ? SQL_C_UBIGINT : SQL_C_SBIGINT;
+ case Real:
+ return SQL_C_FLOAT;
+ case Double:
+ return SQL_C_DOUBLE;
+ case Datetime:
+ return SQL_C_TYPE_TIMESTAMP;
+ default:
+ break;
+ }
+ return SQL_C_DEFAULT; // no default
+}
+
+void
+LexType::convert(Ctx& ctx, SqlType& out, unsigned length) const
+{
+ switch (m_type) {
+ case Char:
+ out.setType(ctx, SqlType::Char, length, true);
+ return;
+ case Integer:
+ out.setType(ctx, SqlType::Bigint, false);
+ return;
+ case Float:
+ out.setType(ctx, SqlType::Double, false);
+ return;
+ case Null:
+ out.setType(ctx, SqlType::Null, true);
+ return;
+ default:
+ break;
+ }
+ ctx.pushStatus(Error::Gen, "unsupported lexical to SQL type conversion");
+}
diff --git a/ndb/src/client/odbc/common/DataType.hpp b/ndb/src/client/odbc/common/DataType.hpp
new file mode 100644
index 00000000000..ac2ca337e22
--- /dev/null
+++ b/ndb/src/client/odbc/common/DataType.hpp
@@ -0,0 +1,292 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_DataType_hpp
+#define ODBC_COMMON_DataType_hpp
+
+#include <map>
+#include <ndb_types.h>
+#include <AttrType.hpp>
+#include <NdbDictionary.hpp>
+#include <common/common.hpp>
+
+/**
+ * Sql data exists in several formats:
+ *
+ * - as NDB data at the bottom
+ * - as SQL data during intermediary processing
+ * - as external data in user input and output buffers
+ * - as lexical constants in SQL statement text
+ *
+ * Each data format has specific types (e.g. number) and each
+ * type has specific attributes (e.g. precision).
+ */
+enum DataFormat {
+ Undef_format = 0,
+ Ndb_format = 1, // not used in NDB version >= v2.10
+ Sql_format = 2,
+ Ext_format = 3,
+ Lex_format = 4
+};
+
+#define UndefDataType 990
+#define NullDataType 991
+#define UnboundDataType 992
+
+class SqlType;
+class ExtType;
+class LexType;
+
+/**
+ * @class SqlType
+ * @brief Sql data type
+ */
+class SqlType {
+public:
+ enum Type {
+ Undef = UndefDataType,
+ Char = SQL_CHAR,
+ Varchar = SQL_VARCHAR,
+ Longvarchar = SQL_LONGVARCHAR,
+ Binary = SQL_BINARY,
+ Varbinary = SQL_VARBINARY,
+ Longvarbinary = SQL_LONGVARBINARY,
+ Decimal = SQL_DECIMAL,
+ Tinyint = SQL_TINYINT,
+ Smallint = SQL_SMALLINT,
+ Integer = SQL_INTEGER,
+ Bigint = SQL_BIGINT,
+ Real = SQL_REAL,
+ Double = SQL_DOUBLE,
+ Date = SQL_DATE,
+ Datetime = SQL_TYPE_TIMESTAMP,
+ Blob = SQL_BLOB,
+ Null = NullDataType, // not an ODBC SQL type
+ Unbound = UnboundDataType // special for placeholders
+ };
+ SqlType();
+ SqlType(Type type, bool nullable = true);
+ SqlType(Type type, unsigned length, bool nullable = true);
+ SqlType(Type type, unsigned precision, unsigned scale, bool nullable = true);
+ SqlType(Ctx& ctx, Type type, bool nullable = true);
+ SqlType(Ctx& ctx, Type type, unsigned length, bool nullable = true);
+ SqlType(Ctx& ctx, Type type, unsigned precision, unsigned scale, bool nullable = true);
+ SqlType(Ctx& ctx, const NdbDictionary::Column* ndbColumn);
+ Type type() const;
+ void setType(Ctx& ctx, Type type, bool nullable = true);
+ void setType(Ctx& ctx, Type type, unsigned length, bool nullable = true);
+ void setType(Ctx& ctx, Type type, unsigned precision, unsigned scale, bool nullable = true);
+ void setType(Ctx& ctx, const NdbDictionary::Column* ndbColumn);
+ bool equal(const SqlType& sqlType) const;
+ unsigned size() const;
+ unsigned displaySize() const;
+ const char* typeName() const;
+ unsigned length() const;
+ bool nullable() const;
+ void nullable(bool value);
+ bool unSigned() const;
+ void unSigned(bool value);
+ // forwards compatible
+ void getType(Ctx& ctx, NdbDictionary::Column* ndbColumn) const;
+ // type conversion
+ SQLSMALLINT sqlcdefault(Ctx& ctx) const;
+ // print for debugging
+ void print(char* buf, unsigned size) const;
+private:
+ friend class LexType;
+ Type m_type;
+ unsigned m_precision;
+ unsigned m_scale;
+ unsigned m_length;
+ bool m_nullable;
+ bool m_unSigned; // qualifier instead of separate types
+};
+
+inline SqlType::Type
+SqlType::type() const
+{
+ return m_type;
+}
+
+inline unsigned
+SqlType::length() const
+{
+ return m_length;
+}
+
+inline bool
+SqlType::nullable() const
+{
+ return m_nullable;
+}
+
+inline void
+SqlType::nullable(bool value)
+{
+ m_nullable = value;
+}
+
+inline bool
+SqlType::unSigned() const
+{
+ return m_unSigned;
+}
+
+inline void
+SqlType::unSigned(bool value)
+{
+ ctx_assert(m_type == Smallint || m_type == Integer || m_type == Bigint);
+ m_unSigned = value;
+}
+
+/**
+ * Actual SQL datatypes.
+ */
+typedef unsigned char SqlChar; // Char and Varchar via pointer
+typedef Int16 SqlSmallint;
+typedef Int32 SqlInteger;
+typedef Int64 SqlBigint;
+typedef Uint16 SqlUsmallint;
+typedef Uint32 SqlUinteger;
+typedef Uint64 SqlUbigint;
+typedef float SqlReal;
+typedef double SqlDouble;
+
+// datetime cc yy mm dd HH MM SS 00 ff ff ff ff stored as String(12)
+struct SqlDatetime {
+ int cc() const { return *(signed char*)&m_data[0]; }
+ void cc(int x) { *(signed char*)&m_data[0] = x; }
+ unsigned yy() const { return *(unsigned char*)&m_data[1]; }
+ void yy(unsigned x) { *(unsigned char*)&m_data[1] = x; }
+ unsigned mm() const { return *(unsigned char*)&m_data[2]; }
+ void mm(unsigned x) { *(unsigned char*)&m_data[2] = x; }
+ unsigned dd() const { return *(unsigned char*)&m_data[3]; }
+ void dd(unsigned x) { *(unsigned char*)&m_data[3] = x; }
+ unsigned HH() const { return *(unsigned char*)&m_data[4]; }
+ void HH(unsigned x) { *(unsigned char*)&m_data[4] = x; }
+ unsigned MM() const { return *(unsigned char*)&m_data[5]; }
+ void MM(unsigned x) { *(unsigned char*)&m_data[5] = x; }
+ unsigned SS() const { return *(unsigned char*)&m_data[6]; }
+ void SS(unsigned x) { *(unsigned char*)&m_data[6] = x; }
+ unsigned ff() const {
+ const unsigned char* p = (unsigned char*)&m_data[8];
+ unsigned x = 0;
+ x += *p++ << 24;
+ x += *p++ << 16;
+ x += *p++ << 8;
+ x += *p++;
+ return x;
+ }
+ void ff(unsigned x) {
+ unsigned char* p = (unsigned char*)&m_data[8];
+ *p++ = (x >> 24) & 0xff;
+ *p++ = (x >> 16) & 0xff;
+ *p++ = (x >> 8) & 0xff;
+ *p++ = x & 0xff;
+ }
+ bool valid() { return true; } // XXX later
+ bool less(const SqlDatetime t) const {
+ if (cc() != t.cc())
+ return cc() < t.cc();
+ if (yy() != t.yy())
+ return yy() < t.yy();
+ if (mm() != t.mm())
+ return mm() < t.mm();
+ if (dd() != t.dd())
+ return dd() < t.dd();
+ if (HH() != t.HH())
+ return HH() < t.HH();
+ if (MM() != t.MM())
+ return MM() < t.MM();
+ if (SS() != t.SS())
+ return SS() < t.SS();
+ if (ff() != t.ff())
+ return ff() < t.ff();
+ return false;
+ }
+private:
+ char m_data[12]; // use array to avoid gaps
+};
+
+/**
+ * @class ExtType
+ * @brief External data type
+ */
+class ExtType {
+public:
+ enum Type {
+ Undef = UndefDataType,
+ Char = SQL_C_CHAR,
+ Short = SQL_C_SHORT,
+ Sshort = SQL_C_SSHORT,
+ Ushort = SQL_C_USHORT,
+ Long = SQL_C_LONG, // for sun.jdbc.odbc
+ Slong = SQL_C_SLONG,
+ Ulong = SQL_C_ULONG,
+ Sbigint = SQL_C_SBIGINT,
+ Ubigint = SQL_C_UBIGINT,
+ Float = SQL_C_FLOAT,
+ Double = SQL_C_DOUBLE,
+ Timestamp = SQL_C_TYPE_TIMESTAMP,
+ Binary = SQL_C_BINARY, // XXX BLOB hack
+ Unbound = UnboundDataType
+ };
+ ExtType();
+ ExtType(Type type);
+ ExtType(Ctx& ctx, Type type);
+ Type type() const;
+ void setType(Ctx& ctx, Type type);
+ unsigned size() const;
+private:
+ Type m_type;
+};
+
+inline ExtType::Type
+ExtType::type() const
+{
+ return m_type;
+}
+
+/**
+ * @class LexType
+ * @class Lexical data type
+ */
+class LexType {
+public:
+ enum Type {
+ Undef = UndefDataType,
+ Char = 1,
+ Integer = 2,
+ Float = 3,
+ Null = 4
+ };
+ LexType();
+ LexType(Type type);
+ LexType(Ctx& ctx, Type type);
+ Type type() const;
+ void setType(Ctx& ctx, Type type);
+ void convert(Ctx& ctx, SqlType& out, unsigned length = 0) const;
+private:
+ Type m_type;
+};
+
+inline LexType::Type
+LexType::type() const
+{
+ return m_type;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/DescArea.cpp b/ndb/src/client/odbc/common/DescArea.cpp
new file mode 100644
index 00000000000..bad9f23d3ef
--- /dev/null
+++ b/ndb/src/client/odbc/common/DescArea.cpp
@@ -0,0 +1,167 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <vector>
+#include "DescArea.hpp"
+
+// DescField
+
+// DescRec
+
+void
+DescRec::setField(int id, const OdbcData& data)
+{
+ Ctx ctx;
+ setField(ctx, id, data);
+ ctx_assert(ctx.ok());
+}
+
+void
+DescRec::getField(int id, OdbcData& data)
+{
+ Ctx ctx;
+ getField(ctx, id, data);
+ ctx_assert(ctx.ok());
+}
+
+void
+DescRec::setField(Ctx& ctx, int id, const OdbcData& data)
+{
+ Fields::iterator iter;
+ iter = m_fields.find(id);
+ if (ctx.logLevel() >= 3) {
+ char buf[100];
+ data.print(buf, sizeof(buf));
+ ctx_log3(("set %s rec %d id %d = %s", DescArea::nameUsage(m_area->getUsage()), m_num, id, buf));
+ }
+ if (iter != m_fields.end()) {
+ DescField& field = (*iter).second;
+ field.setData(data);
+ m_area->setBound(false); // XXX could compare data values
+ return;
+ }
+ const DescSpec& spec = m_area->findSpec(id);
+ if (spec.m_pos != Desc_pos_end) {
+ DescField field(spec, data);
+ m_fields.insert(Fields::value_type(id, field));
+ m_area->setBound(false);
+ return;
+ }
+ ctx_assert(false);
+}
+
+void
+DescRec::getField(Ctx& ctx, int id, OdbcData& data)
+{
+ Fields::iterator iter;
+ iter = m_fields.find(id);
+ if (iter != m_fields.end()) {
+ DescField& field = (*iter).second;
+ data.setValue(field.getData());
+ return;
+ }
+ const DescSpec& spec = m_area->findSpec(id);
+ if (spec.m_pos != Desc_pos_end) {
+ data.setValue();
+ return; // XXX default value
+ }
+ ctx_assert(false);
+}
+
+// DescArea
+
+DescArea::DescArea(HandleBase* handle, const DescSpec* specList) :
+ m_handle(handle),
+ m_specList(specList),
+ m_alloc(Desc_alloc_undef),
+ m_usage(Desc_usage_undef),
+ m_bound(true) // no bind necessary since empty
+{
+ m_header.m_area = this;
+ m_header.m_num = -1;
+ DescRec rec;
+ rec.m_area = this;
+ rec.m_num = m_recs.size();
+ m_recs.push_back(rec); // add bookmark record
+ SQLSMALLINT count = 0;
+ getHeader().setField(SQL_DESC_COUNT, count);
+ m_bound = true;
+}
+
+DescArea::~DescArea()
+{
+}
+
+const DescSpec&
+DescArea::findSpec(int id)
+{
+ const DescSpec* p;
+ for (p = m_specList; p->m_pos != Desc_pos_end; p++) {
+ if (p->m_id == id)
+ break;
+ }
+ return *p;
+}
+
+unsigned
+DescArea::getCount() const
+{
+ ctx_assert(m_recs.size() > 0);
+ return m_recs.size() - 1;
+}
+
+void
+DescArea::setCount(Ctx& ctx, unsigned count)
+{
+ if (m_recs.size() - 1 == count)
+ return;
+ ctx_log3(("set %s count %d to %d",
+ DescArea::nameUsage(m_usage),
+ (unsigned)(m_recs.size() - 1),
+ count));
+ m_recs.resize(1 + count);
+ for (unsigned i = 0; i <= count; i++) {
+ m_recs[i].m_area = this;
+ m_recs[i].m_num = i;
+ }
+ getHeader().setField(SQL_DESC_COUNT, static_cast<SQLSMALLINT>(count));
+}
+
+DescRec&
+DescArea::pushRecord()
+{
+ ctx_assert(m_recs.size() > 0);
+ DescRec rec;
+ rec.m_area = this;
+ rec.m_num = m_recs.size();
+ m_recs.push_back(rec);
+ SQLSMALLINT count = m_recs.size() - 1;
+ getHeader().setField(SQL_DESC_COUNT, count);
+ return m_recs.back();
+}
+
+DescRec&
+DescArea::getHeader()
+{
+ return m_header;
+}
+
+DescRec&
+DescArea::getRecord(unsigned num)
+{
+ ctx_assert(num < m_recs.size());
+ return m_recs[num];
+}
diff --git a/ndb/src/client/odbc/common/DescArea.hpp b/ndb/src/client/odbc/common/DescArea.hpp
new file mode 100644
index 00000000000..e9f552d758d
--- /dev/null
+++ b/ndb/src/client/odbc/common/DescArea.hpp
@@ -0,0 +1,266 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_DescArea_hpp
+#define ODBC_COMMON_DescArea_hpp
+
+#include <map>
+#include <vector>
+#include <common/common.hpp>
+#include "OdbcData.hpp"
+
+/**
+ * Descriptor records. Contains:
+ * -# header, not called a "record" in this context
+ * -# bookmark record at index position 0
+ * -# descriptor records at index positions starting from 1
+ *
+ * These classes are in common/ since the code is general.
+ * However each area is associated with a HandleDesc.
+ *
+ * DescField - field identified by an SQL_DESC_* constant
+ * DescRec - header or record, a list of fields
+ * DescArea - header and all records
+ */
+
+class HandleBase;
+class DescField;
+class DescRec;
+class DescArea;
+
+enum DescPos {
+ Desc_pos_undef = 0,
+ Desc_pos_header,
+ Desc_pos_record,
+ Desc_pos_end
+};
+
+enum DescMode {
+ Desc_mode_undef = 0,
+ Desc_mode_readonly,
+ Desc_mode_writeonly,
+ Desc_mode_readwrite,
+ Desc_mode_unused
+};
+
+struct DescSpec {
+ DescPos m_pos; // header or record
+ int m_id; // SQL_DESC_ identifier
+ OdbcData::Type m_type; // data type
+ DescMode m_mode[1+4]; // access mode IPD APD IRD ARD
+ // called before setting value
+ typedef void CallbackSet(Ctx& ctx, HandleBase* self, const OdbcData& data);
+ CallbackSet* m_set;
+ // called to get default value
+ typedef void CallbackDefault(Ctx& ctx, HandleBase* self, OdbcData& data);
+ CallbackDefault* m_default;
+};
+
+enum DescAlloc {
+ Desc_alloc_undef = 0,
+ Desc_alloc_auto,
+ Desc_alloc_user
+};
+
+enum DescUsage {
+ Desc_usage_undef = 0,
+ Desc_usage_IPD = 1, // these must be 1-4
+ Desc_usage_IRD = 2,
+ Desc_usage_APD = 3,
+ Desc_usage_ARD = 4
+};
+
+/**
+ * @class DescField
+ * @brief Field identified by an SQL_DESC_* constant
+ */
+class DescField {
+public:
+ DescField(const DescSpec& spec, const OdbcData& data);
+ DescField(const DescField& field);
+ ~DescField();
+private:
+ friend class DescRec;
+ void setData(const OdbcData& data);
+ const OdbcData& getData();
+ const DescSpec& m_spec;
+ OdbcData m_data;
+};
+
+inline
+DescField::DescField(const DescSpec& spec, const OdbcData& data) :
+ m_spec(spec),
+ m_data(data)
+{
+}
+
+inline
+DescField::DescField(const DescField& field) :
+ m_spec(field.m_spec),
+ m_data(field.m_data)
+{
+}
+
+inline
+DescField::~DescField()
+{
+}
+
+inline void
+DescField::setData(const OdbcData& data)
+{
+ ctx_assert(m_spec.m_type == data.type());
+ m_data.setValue(data);
+}
+
+inline const OdbcData&
+DescField::getData()
+{
+ ctx_assert(m_data.type() != OdbcData::Undef);
+ return m_data;
+}
+
+/**
+ * @class DescRec
+ * @brief Descriptor record, a list of fields
+ */
+class DescRec {
+ friend class DescArea;
+public:
+ DescRec();
+ ~DescRec();
+ void setField(int id, const OdbcData& data);
+ void getField(int id, OdbcData& data);
+ void setField(Ctx& ctx, int id, const OdbcData& data);
+ void getField(Ctx& ctx, int id, OdbcData& data);
+private:
+ DescArea* m_area;
+ int m_num; // for logging only -1 = header 0 = bookmark
+ typedef std::map<int, DescField> Fields;
+ Fields m_fields;
+};
+
+inline
+DescRec::DescRec() :
+ m_area(0)
+{
+}
+
+inline
+DescRec::~DescRec()
+{
+}
+
+/**
+ * @class DescArea
+ * @brief All records, including header (record 0)
+ *
+ * Descriptor area includes a header (record 0)
+ * and zero or more records at position >= 1.
+ * Each of these describes one parameter or one column.
+ *
+ * - DescArea : Collection of records
+ * - DescRec : Collection of fields
+ * - DescField : Contains data of type OdbcData
+ */
+class DescArea {
+public:
+ DescArea(HandleBase* handle, const DescSpec* specList);
+ ~DescArea();
+ void setAlloc(DescAlloc alloc);
+ DescAlloc getAlloc() const;
+ void setUsage(DescUsage usage);
+ DescUsage getUsage() const;
+ static const char* nameUsage(DescUsage u);
+ // find specifier
+ const DescSpec& findSpec(int id);
+ // get or set number of records (record 0 not counted)
+ unsigned getCount() const;
+ void setCount(Ctx& ctx, unsigned count);
+ // paush new record (record 0 exists always)
+ DescRec& pushRecord();
+ // get ref to header or to any record
+ DescRec& getHeader();
+ DescRec& getRecord(unsigned num);
+ // modified since last bind
+ void setBound(bool bound);
+ bool isBound() const;
+private:
+ HandleBase* m_handle;
+ const DescSpec* const m_specList;
+ DescRec m_header;
+ typedef std::vector<DescRec> Recs;
+ Recs m_recs;
+ DescAlloc m_alloc;
+ DescUsage m_usage;
+ bool m_bound;
+};
+
+inline void
+DescArea::setAlloc(DescAlloc alloc)
+{
+ m_alloc = alloc;
+}
+
+inline DescAlloc
+DescArea::getAlloc() const
+{
+ return m_alloc;
+}
+
+inline void
+DescArea::setUsage(DescUsage usage)
+{
+ m_usage = usage;
+}
+
+inline DescUsage
+DescArea::getUsage() const
+{
+ return m_usage;
+}
+
+inline const char*
+DescArea::nameUsage(DescUsage u)
+{
+ switch (u) {
+ case Desc_usage_undef:
+ break;
+ case Desc_usage_IPD:
+ return "IPD";
+ case Desc_usage_IRD:
+ return "IRD";
+ case Desc_usage_APD:
+ return "APD";
+ case Desc_usage_ARD:
+ return "ARD";
+ }
+ return "?";
+}
+
+inline void
+DescArea::setBound(bool bound)
+{
+ m_bound = bound;
+}
+
+inline bool
+DescArea::isBound() const
+{
+ return m_bound;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/DiagArea.cpp b/ndb/src/client/odbc/common/DiagArea.cpp
new file mode 100644
index 00000000000..06e8da89495
--- /dev/null
+++ b/ndb/src/client/odbc/common/DiagArea.cpp
@@ -0,0 +1,284 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include "OdbcData.hpp"
+#include "DiagArea.hpp"
+
+// DiagSpec
+
+static const DiagSpec
+diag_spec_list[] = {
+ { Diag_pos_header,
+ SQL_DIAG_CURSOR_ROW_COUNT,
+ OdbcData::Integer,
+ Odbc_handle_stmt
+ },
+ { Diag_pos_header,
+ SQL_DIAG_DYNAMIC_FUNCTION,
+ OdbcData::Sqlchar,
+ Odbc_handle_stmt
+ },
+ { Diag_pos_header,
+ SQL_DIAG_DYNAMIC_FUNCTION_CODE,
+ OdbcData::Integer,
+ Odbc_handle_stmt
+ },
+ { Diag_pos_header,
+ SQL_DIAG_NUMBER,
+ OdbcData::Integer,
+ Odbc_handle_all
+ },
+ { Diag_pos_header,
+ SQL_DIAG_RETURNCODE,
+ OdbcData::Smallint,
+ Odbc_handle_all
+ },
+ { Diag_pos_header,
+ SQL_DIAG_ROW_COUNT,
+ OdbcData::Integer,
+ Odbc_handle_stmt
+ },
+ { Diag_pos_status,
+ SQL_DIAG_CLASS_ORIGIN,
+ OdbcData::Sqlchar,
+ Odbc_handle_all
+ },
+ { Diag_pos_status,
+ SQL_DIAG_COLUMN_NUMBER,
+ OdbcData::Integer,
+ Odbc_handle_all
+ },
+ { Diag_pos_status,
+ SQL_DIAG_CONNECTION_NAME,
+ OdbcData::Sqlchar,
+ Odbc_handle_all
+ },
+ { Diag_pos_status,
+ SQL_DIAG_MESSAGE_TEXT,
+ OdbcData::Sqlchar,
+ Odbc_handle_all
+ },
+ { Diag_pos_status,
+ SQL_DIAG_NATIVE,
+ OdbcData::Integer,
+ Odbc_handle_all
+ },
+ { Diag_pos_status,
+ SQL_DIAG_ROW_NUMBER,
+ OdbcData::Integer,
+ Odbc_handle_all
+ },
+ { Diag_pos_status,
+ SQL_DIAG_SERVER_NAME,
+ OdbcData::Sqlchar,
+ Odbc_handle_all
+ },
+ { Diag_pos_status,
+ SQL_DIAG_SQLSTATE,
+ OdbcData::Sqlchar,
+ Odbc_handle_all
+ },
+ { Diag_pos_status,
+ SQL_DIAG_SUBCLASS_ORIGIN,
+ OdbcData::Sqlchar,
+ Odbc_handle_all
+ },
+ { Diag_pos_end,
+ 0,
+ OdbcData::Undef,
+ 0
+ }
+};
+
+const DiagSpec&
+DiagSpec::find(int id)
+{
+ const DiagSpec* p;
+ for (p = diag_spec_list; p->m_pos != Diag_pos_end; p++) {
+ if (p->m_id == id)
+ break;
+ }
+ return *p;
+}
+
+// DiagField
+
+// DiagRec
+
+void
+DiagRec::setField(int id, const OdbcData& data)
+{
+ Fields::iterator iter;
+ iter = m_fields.find(id);
+ if (iter != m_fields.end()) {
+ DiagField& field = (*iter).second;
+ field.setData(data);
+ return;
+ }
+ const DiagSpec& spec = DiagSpec::find(id);
+ if (spec.m_pos != Diag_pos_end) {
+ DiagField field(spec, data);
+ m_fields.insert(Fields::value_type(id, field));
+ return;
+ }
+ ctx_assert(false);
+}
+
+void
+DiagRec::getField(Ctx& ctx, int id, OdbcData& data)
+{
+ Fields::iterator iter;
+ iter = m_fields.find(id);
+ if (iter != m_fields.end()) {
+ DiagField& field = (*iter).second;
+ data.setValue(field.getData());
+ return;
+ }
+ const DiagSpec& spec = DiagSpec::find(id);
+ if (spec.m_pos != Diag_pos_end) {
+ // success and undefined value says the MS doc
+ data.setValue();
+ return;
+ }
+ ctx_assert(false);
+}
+
+// DiagArea
+
+DiagArea::DiagArea() :
+ m_recs(1), // add header
+ m_code(SQL_SUCCESS),
+ m_recNumber(0)
+{
+ setHeader(SQL_DIAG_NUMBER, (SQLINTEGER)0);
+}
+
+DiagArea::~DiagArea() {
+}
+
+unsigned
+DiagArea::numStatus()
+{
+ ctx_assert(m_recs.size() > 0);
+ return m_recs.size() - 1;
+}
+
+void
+DiagArea::pushStatus()
+{
+ ctx_assert(m_recs.size() > 0);
+ DiagRec rec;
+ m_recs.push_back(rec);
+ SQLINTEGER diagNumber = m_recs.size() - 1;
+ setHeader(SQL_DIAG_NUMBER, diagNumber);
+}
+
+void
+DiagArea::setHeader(int id, const OdbcData& data)
+{
+ ctx_assert(m_recs.size() > 0);
+ getHeader().setField(id, data);
+}
+
+// set status
+
+void
+DiagArea::setStatus(int id, const OdbcData& data)
+{
+ getStatus().setField(id, data);
+}
+
+void
+DiagArea::setStatus(const Sqlstate& state)
+{
+ getStatus().setField(SQL_DIAG_SQLSTATE, state);
+ setCode(state.getCode(m_code));
+}
+
+void
+DiagArea::setStatus(const Error& error)
+{
+ BaseString message("");
+ // bracketed prefixes
+ message.append(NDB_ODBC_COMPONENT_VENDOR);
+ message.append(NDB_ODBC_COMPONENT_DRIVER);
+ if (! error.driverError())
+ message.append(NDB_ODBC_COMPONENT_DATABASE);
+ // native error code
+ char nativeString[20];
+ sprintf(nativeString, "%02d%02d%04d",
+ (unsigned)error.m_status % 100,
+ (unsigned)error.m_classification % 100,
+ (unsigned)error.m_code % 10000);
+ SQLINTEGER native = atoi(nativeString);
+ message.appfmt("NDB-%s", nativeString);
+ // message text
+ message.append(" ");
+ message.append(error.m_message);
+ if (error.m_sqlFunction != 0)
+ message.appfmt(" (in %s)", error.m_sqlFunction);
+ // set diag fields
+ setStatus(error.m_sqlstate);
+ setStatus(SQL_DIAG_NATIVE, native);
+ setStatus(SQL_DIAG_MESSAGE_TEXT, message.c_str());
+}
+
+// push status
+
+void
+DiagArea::pushStatus(const Error& error)
+{
+ pushStatus();
+ setStatus(error);
+}
+
+// record access
+
+DiagRec&
+DiagArea::getHeader()
+{
+ ctx_assert(m_recs.size() > 0);
+ return m_recs[0];
+}
+
+DiagRec&
+DiagArea::getStatus()
+{
+ ctx_assert(m_recs.size() > 1);
+ return m_recs.back();
+}
+
+DiagRec&
+DiagArea::getRecord(unsigned num)
+{
+ ctx_assert(num < m_recs.size());
+ return m_recs[num];
+}
+
+void
+DiagArea::getRecord(Ctx& ctx, unsigned num, int id, OdbcData& data)
+{
+ DiagRec& rec = getRecord(num);
+ rec.getField(ctx, id, data);
+}
+
+void
+DiagArea::setCode(SQLRETURN code)
+{
+ m_code = code;
+ getHeader().setField(SQL_DIAG_RETURNCODE, (SQLSMALLINT)code);
+}
diff --git a/ndb/src/client/odbc/common/DiagArea.hpp b/ndb/src/client/odbc/common/DiagArea.hpp
new file mode 100644
index 00000000000..79c03de6623
--- /dev/null
+++ b/ndb/src/client/odbc/common/DiagArea.hpp
@@ -0,0 +1,196 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_DiagArea_hpp
+#define ODBC_COMMON_DiagArea_hpp
+
+#include <map>
+#include <vector>
+#include <common/common.hpp>
+#include "OdbcData.hpp"
+
+enum DiagPos {
+ Diag_pos_undef = 0,
+ Diag_pos_header,
+ Diag_pos_status,
+ Diag_pos_end
+};
+
+/**
+ * @class DiagSpec
+ * @brief Field specification
+ */
+struct DiagSpec {
+ DiagPos m_pos; // header or status
+ int m_id; // SQL_DIAG_ identifier
+ OdbcData::Type m_type; // data type
+ unsigned m_handles; // defined for these handle types
+ // find the spec
+ static const DiagSpec& find(int id);
+};
+
+/**
+ * @class DiagField
+ * @brief Field identified by an SQL_DIAG_* constant
+ */
+class DiagField {
+public:
+ DiagField(const DiagSpec& spec, const OdbcData& data);
+ DiagField(const DiagField& field);
+ ~DiagField();
+ void setData(const OdbcData& data);
+ const OdbcData& getData();
+private:
+ const DiagSpec& m_spec;
+ OdbcData m_data;
+};
+
+inline
+DiagField::DiagField(const DiagSpec& spec, const OdbcData& data) :
+ m_spec(spec),
+ m_data(data)
+{
+}
+
+inline
+DiagField::DiagField(const DiagField& field) :
+ m_spec(field.m_spec),
+ m_data(field.m_data)
+{
+}
+
+inline
+DiagField::~DiagField()
+{
+}
+
+inline void
+DiagField::setData(const OdbcData& data)
+{
+ ctx_assert(m_spec.m_type == data.type());
+ m_data.setValue(data);
+}
+
+inline const OdbcData&
+DiagField::getData()
+{
+ ctx_assert(m_data.type() != OdbcData::Undef);
+ return m_data;
+}
+
+/**
+ * @class DiagRec
+ * @brief One diagnostic record, a list of fields
+ */
+class DiagRec {
+public:
+ DiagRec();
+ ~DiagRec();
+ void setField(int id, const OdbcData& data);
+ void getField(Ctx& ctx, int id, OdbcData& data);
+private:
+ typedef std::map<int, DiagField> Fields;
+ Fields m_fields;
+};
+
+inline
+DiagRec::DiagRec()
+{
+}
+
+inline
+DiagRec::~DiagRec()
+{
+}
+
+/**
+ * @class DiagArea
+ * @brief All records, including header (record 0)
+ *
+ * Diagnostic area includes a header (record 0) and zero or more
+ * status records at positions >= 1.
+ */
+class DiagArea {
+public:
+ DiagArea();
+ ~DiagArea();
+ /**
+ * Get number of status records.
+ */
+ unsigned numStatus();
+ /**
+ * Push new status record.
+ */
+ void pushStatus();
+ /**
+ * Set field in header.
+ */
+ void setHeader(int id, const OdbcData& data);
+ /**
+ * Set field in latest status record. The arguments can
+ * also be plain int, char*, Sqlstate. The NDB and other
+ * native errors set Sqlstate _IM000 automatically.
+ */
+ void setStatus(int id, const OdbcData& data);
+ void setStatus(const Sqlstate& state);
+ void setStatus(const Error& error);
+ /**
+ * Convenience methods to push new status record and set
+ * some common fields in it. Sqlstate is set always.
+ */
+ void pushStatus(const Error& error);
+ /**
+ * Get refs to various records.
+ */
+ DiagRec& getHeader();
+ DiagRec& getStatus();
+ DiagRec& getRecord(unsigned num);
+ /**
+ * Get diag values.
+ */
+ void getRecord(Ctx& ctx, unsigned num, int id, OdbcData& data);
+ /**
+ * Get or set return code.
+ */
+ SQLRETURN getCode() const;
+ void setCode(SQLRETURN ret);
+ /**
+ * Get "next" record number (0 when no more).
+ * Used only by the deprecated SQLError function.
+ */
+ unsigned nextRecNumber();
+private:
+ typedef std::vector<DiagRec> Recs;
+ Recs m_recs;
+ SQLRETURN m_code;
+ unsigned m_recNumber; // for SQLError
+};
+
+inline SQLRETURN
+DiagArea::getCode() const
+{
+ return m_code;
+}
+
+inline unsigned
+DiagArea::nextRecNumber()
+{
+ if (m_recNumber >= numStatus())
+ return 0;
+ return ++m_recNumber;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/Makefile b/ndb/src/client/odbc/common/Makefile
new file mode 100644
index 00000000000..7ee29738d86
--- /dev/null
+++ b/ndb/src/client/odbc/common/Makefile
@@ -0,0 +1,29 @@
+include .defs.mk
+
+TYPE = *
+
+NONPIC_ARCHIVE = N
+
+PIC_ARCHIVE = Y
+
+ARCHIVE_TARGET = odbccommon
+
+SOURCES = \
+ common.cpp \
+ Ctx.cpp \
+ Sqlstate.cpp \
+ OdbcData.cpp \
+ DiagArea.cpp \
+ AttrArea.cpp \
+ DescArea.cpp \
+ ConnArea.cpp \
+ StmtInfo.cpp \
+ StmtArea.cpp \
+ CodeTree.cpp \
+ ResultArea.cpp \
+ DataType.cpp \
+ DataField.cpp \
+ DataRow.cpp
+
+include ../Extra.mk
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/client/odbc/common/OdbcData.cpp b/ndb/src/client/odbc/common/OdbcData.cpp
new file mode 100644
index 00000000000..d2402c7e0ab
--- /dev/null
+++ b/ndb/src/client/odbc/common/OdbcData.cpp
@@ -0,0 +1,563 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <new>
+#include <string.h>
+#include <NdbStdio.h>
+#include "OdbcData.hpp"
+
+OdbcData::OdbcData() :
+ m_type(Undef)
+{
+}
+
+OdbcData::OdbcData(Type type) :
+ m_type(type)
+{
+ switch (m_type) {
+ case Smallint:
+ m_smallint = 0;
+ break;
+ case Usmallint:
+ m_usmallint = 0;
+ break;
+ case Integer:
+ m_integer = 0;
+ break;
+ case Uinteger:
+ m_uinteger = 0;
+ break;
+ case Pointer:
+ m_pointer = 0;
+ break;
+ case SmallintPtr:
+ m_smallintPtr = 0;
+ break;
+ case UsmallintPtr:
+ m_usmallintPtr = 0;
+ break;
+ case IntegerPtr:
+ m_integerPtr = 0;
+ break;
+ case UintegerPtr:
+ m_uintegerPtr = 0;
+ break;
+ case PointerPtr:
+ m_pointerPtr = 0;
+ break;
+ case Sqlchar:
+ m_sqlchar = 0;
+ break;
+ case Sqlstate:
+ m_sqlstate = 0;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ };
+}
+
+OdbcData::OdbcData(const OdbcData& odbcData) :
+ m_type(odbcData.m_type)
+{
+ switch (m_type) {
+ case Smallint:
+ m_smallint = odbcData.m_smallint;
+ break;
+ case Usmallint:
+ m_usmallint = odbcData.m_usmallint;
+ break;
+ case Integer:
+ m_integer = odbcData.m_integer;
+ break;
+ case Uinteger:
+ m_uinteger = odbcData.m_uinteger;
+ break;
+ case Pointer:
+ m_pointer = odbcData.m_pointer;
+ break;
+ case SmallintPtr:
+ m_smallintPtr = odbcData.m_smallintPtr;
+ break;
+ case UsmallintPtr:
+ m_usmallintPtr = odbcData.m_usmallintPtr;
+ break;
+ case IntegerPtr:
+ m_integerPtr = odbcData.m_integerPtr;
+ break;
+ case UintegerPtr:
+ m_uintegerPtr = odbcData.m_uintegerPtr;
+ break;
+ case PointerPtr:
+ m_pointerPtr = odbcData.m_pointerPtr;
+ break;
+ case Sqlchar: {
+ unsigned n = strlen(odbcData.m_sqlchar);
+ m_sqlchar = new char[n + 1];
+ memcpy(m_sqlchar, odbcData.m_sqlchar, n + 1);
+ break;
+ }
+ case Sqlstate:
+ m_sqlstate = odbcData.m_sqlstate;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ };
+}
+
+OdbcData::~OdbcData()
+{
+ switch (m_type) {
+ case Sqlchar:
+ delete[] m_sqlchar;
+ break;
+ default:
+ break;
+ }
+}
+
+void
+OdbcData::setValue()
+{
+ m_type = Undef;
+}
+
+void
+OdbcData::setValue(Type type)
+{
+ if (m_type == Sqlchar) {
+ delete[] m_sqlchar;
+ m_sqlchar = 0;
+ }
+ switch (m_type) {
+ case Smallint:
+ m_smallint = 0;
+ break;
+ case Usmallint:
+ m_usmallint = 0;
+ break;
+ case Integer:
+ m_integer = 0;
+ break;
+ case Uinteger:
+ m_uinteger = 0;
+ break;
+ case Pointer:
+ m_pointer = 0;
+ break;
+ case SmallintPtr:
+ m_smallintPtr = 0;
+ break;
+ case UsmallintPtr:
+ m_usmallintPtr = 0;
+ break;
+ case IntegerPtr:
+ m_integerPtr = 0;
+ break;
+ case UintegerPtr:
+ m_uintegerPtr = 0;
+ break;
+ case PointerPtr:
+ m_pointerPtr = 0;
+ break;
+ case Sqlchar:
+ m_sqlchar = 0;
+ break;
+ case Sqlstate:
+ m_sqlstate = 0;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ };
+}
+
+void
+OdbcData::setValue(const OdbcData odbcData)
+{
+ if (m_type == Sqlchar) {
+ delete[] m_sqlchar;
+ m_sqlchar = 0;
+ }
+ m_type = odbcData.m_type;
+ switch (m_type) {
+ case Smallint:
+ m_smallint = odbcData.m_smallint;
+ break;
+ case Usmallint:
+ m_usmallint = odbcData.m_usmallint;
+ break;
+ case Integer:
+ m_integer = odbcData.m_integer;
+ break;
+ case Uinteger:
+ m_uinteger = odbcData.m_uinteger;
+ break;
+ case Pointer:
+ m_pointer = odbcData.m_pointer;
+ break;
+ case SmallintPtr:
+ m_smallintPtr = odbcData.m_smallintPtr;
+ break;
+ case UsmallintPtr:
+ m_usmallintPtr = odbcData.m_usmallintPtr;
+ break;
+ case IntegerPtr:
+ m_integerPtr = odbcData.m_integerPtr;
+ break;
+ case UintegerPtr:
+ m_uintegerPtr = odbcData.m_uintegerPtr;
+ break;
+ case PointerPtr:
+ m_pointerPtr = odbcData.m_pointerPtr;
+ break;
+ case Sqlchar: {
+ unsigned n = strlen(odbcData.m_sqlchar);
+ m_sqlchar = new char[n + 1];
+ memcpy(m_sqlchar, odbcData.m_sqlchar, n + 1);
+ break;
+ }
+ case Sqlstate:
+ m_sqlstate = odbcData.m_sqlstate;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ };
+}
+
+// copy in from user buffer
+
+void
+OdbcData::copyin(Ctx& ctx, Type type, SQLPOINTER buf, SQLINTEGER length)
+{
+ if (m_type == Sqlchar) {
+ delete[] m_sqlchar;
+ m_sqlchar = 0;
+ }
+ m_type = type;
+ switch (m_type) {
+ case Smallint: {
+ SQLSMALLINT val = 0;
+ switch (length) {
+ case 0:
+ case SQL_IS_SMALLINT:
+ val = (SQLSMALLINT)(SQLINTEGER)buf;
+ break;
+ case SQL_IS_USMALLINT:
+ val = (SQLUSMALLINT)(SQLUINTEGER)buf;
+ break;
+ case SQL_IS_INTEGER:
+ val = (SQLINTEGER)buf;
+ break;
+ case SQL_IS_UINTEGER:
+ val = (SQLUINTEGER)buf;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "smallint input - invalid length %d", (int)length);
+ return;
+ }
+ m_smallint = val;
+ break;
+ }
+ case Usmallint: {
+ SQLUSMALLINT val = 0;
+ switch (length) {
+ case SQL_IS_SMALLINT:
+ val = (SQLSMALLINT)(SQLINTEGER)buf;
+ break;
+ case 0:
+ case SQL_IS_USMALLINT:
+ val = (SQLUSMALLINT)(SQLUINTEGER)buf;
+ break;
+ case SQL_IS_INTEGER:
+ val = (SQLINTEGER)buf;
+ break;
+ case SQL_IS_UINTEGER:
+ val = (SQLUINTEGER)buf;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "unsigned smallint input - invalid length %d", (int)length);
+ return;
+ }
+ m_usmallint = val;
+ break;
+ }
+ case Integer: {
+ SQLINTEGER val = 0;
+ switch (length) {
+ case SQL_IS_SMALLINT:
+ val = (SQLSMALLINT)(SQLINTEGER)buf;
+ break;
+ case SQL_IS_USMALLINT:
+ val = (SQLUSMALLINT)(SQLUINTEGER)buf;
+ break;
+ case 0:
+ case SQL_IS_INTEGER:
+ val = (SQLINTEGER)buf;
+ break;
+ case SQL_IS_UINTEGER:
+ val = (SQLUINTEGER)buf;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "integer input - invalid length %d", (int)length);
+ return;
+ }
+ m_integer = val;
+ break;
+ }
+ case Uinteger: {
+ SQLUINTEGER val = 0;
+ switch (length) {
+ case SQL_IS_SMALLINT:
+ val = (SQLSMALLINT)(SQLINTEGER)buf;
+ break;
+ case SQL_IS_USMALLINT:
+ val = (SQLUSMALLINT)(SQLUINTEGER)buf;
+ break;
+ case SQL_IS_INTEGER:
+ val = (SQLINTEGER)buf;
+ break;
+ case 0:
+ case SQL_IS_UINTEGER:
+ val = (SQLUINTEGER)buf;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "unsigned integer input - invalid length %d", (int)length);
+ return;
+ }
+ m_uinteger = val;
+ break;
+ }
+ case Pointer: {
+ SQLPOINTER val = 0;
+ switch (length) {
+ case 0:
+ case SQL_IS_POINTER:
+ val = (SQLPOINTER)buf;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "pointer input - invalid length %d", (int)length);
+ return;
+ }
+ m_pointer = val;
+ break;
+ }
+ case SmallintPtr: {
+ SQLSMALLINT* val = 0;
+ switch (length) {
+ case 0:
+ case SQL_IS_POINTER:
+ val = (SQLSMALLINT*)buf;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "smallint pointer input - invalid length %d", (int)length);
+ return;
+ }
+ m_smallintPtr = val;
+ break;
+ }
+ case UsmallintPtr: {
+ SQLUSMALLINT* val = 0;
+ switch (length) {
+ case 0:
+ case SQL_IS_POINTER:
+ val = (SQLUSMALLINT*)buf;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "unsigned smallint pointer input - invalid length %d", (int)length);
+ return;
+ }
+ m_usmallintPtr = val;
+ break;
+ }
+ case IntegerPtr: {
+ SQLINTEGER* val = 0;
+ switch (length) {
+ case 0:
+ case SQL_IS_POINTER:
+ val = (SQLINTEGER*)buf;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "integer pointer input - invalid length %d", (int)length);
+ return;
+ }
+ m_integerPtr = val;
+ break;
+ }
+ case UintegerPtr: {
+ SQLUINTEGER* val = 0;
+ switch (length) {
+ case 0:
+ case SQL_IS_POINTER:
+ val = (SQLUINTEGER*)buf;
+ break;
+ default:
+ ctx.pushStatus(Error::Gen, "unsigned integer pointer input - invalid length %d", (int)length);
+ return;
+ }
+ m_uintegerPtr = val;
+ break;
+ }
+ case Sqlchar: {
+ const char* val = (char*)buf;
+ if (val == 0) {
+ ctx.pushStatus(Sqlstate::_HY009, Error::Gen, "null string input");
+ return;
+ }
+ if (length < 0 && length != SQL_NTS) {
+ ctx.pushStatus(Error::Gen, "string input - invalid length %d", (int)length);
+ return;
+ }
+ if (length == SQL_NTS) {
+ m_sqlchar = strcpy(new char[strlen(val) + 1], val);
+ } else {
+ m_sqlchar = (char*)memcpy(new char[length + 1], val, length);
+ m_sqlchar[length] = 0;
+ }
+ break;
+ }
+ default:
+ ctx_assert(false);
+ break;
+ }
+}
+
+// copy out to user buffer
+
+void
+OdbcData::copyout(Ctx& ctx, SQLPOINTER buf, SQLINTEGER length, SQLINTEGER* total, SQLSMALLINT* total2)
+{
+ if (buf == 0) {
+ ctx.setCode(SQL_ERROR);
+ return;
+ }
+ switch (m_type) {
+ case Smallint: {
+ SQLSMALLINT* ptr = static_cast<SQLSMALLINT*>(buf);
+ *ptr = m_smallint;
+ break;
+ }
+ case Usmallint: {
+ SQLUSMALLINT* ptr = static_cast<SQLUSMALLINT*>(buf);
+ *ptr = m_usmallint;
+ break;
+ }
+ case Integer: {
+ SQLINTEGER* ptr = static_cast<SQLINTEGER*>(buf);
+ *ptr = m_integer;
+ break;
+ }
+ case Uinteger: {
+ SQLUINTEGER* ptr = static_cast<SQLUINTEGER*>(buf);
+ *ptr = m_uinteger;
+ break;
+ }
+ case Pointer: {
+ SQLPOINTER* ptr = static_cast<SQLPOINTER*>(buf);
+ *ptr = m_pointer;
+ break;
+ }
+ case Sqlchar: {
+ char* ptr = static_cast<char*>(buf);
+ if (length < 0 && length != SQL_NTS) {
+ ctx.setCode(SQL_ERROR);
+ return;
+ }
+ if (length == SQL_NTS) {
+ strcpy(ptr, m_sqlchar);
+ } else {
+ strncpy(ptr, m_sqlchar, length);
+ }
+ if (total != 0)
+ *total = strlen(m_sqlchar);
+ if (total2 != 0)
+ *total2 = strlen(m_sqlchar);
+ break;
+ }
+ case Sqlstate: {
+ char* ptr = static_cast<char*>(buf);
+ const char* state = m_sqlstate->state();
+ if (length < 0 && length != SQL_NTS) {
+ ctx.setCode(SQL_ERROR);
+ return;
+ }
+ if (length == SQL_NTS) {
+ strcpy(ptr, state);
+ } else {
+ strncpy(ptr, state, length);
+ }
+ if (total != 0)
+ *total = strlen(state);
+ if (total2 != 0)
+ *total2 = strlen(state);
+ break;
+ }
+ default:
+ ctx_assert(false);
+ break;
+ }
+}
+
+void
+OdbcData::print(char* buf, unsigned size) const
+{
+ switch (m_type) {
+ case Undef:
+ snprintf(buf, size, "undef");
+ break;
+ case Smallint:
+ snprintf(buf, size, "%d", (int)m_smallint);
+ break;
+ case Usmallint:
+ snprintf(buf, size, "%u", (unsigned)m_usmallint);
+ break;
+ case Integer:
+ snprintf(buf, size, "%ld", (long)m_integer);
+ break;
+ case Uinteger:
+ snprintf(buf, size, "%lu", (unsigned long)m_uinteger);
+ break;
+ case Pointer:
+ snprintf(buf, size, "0x%lx", (unsigned long)m_pointer);
+ break;
+ case SmallintPtr:
+ snprintf(buf, size, "0x%lx", (unsigned long)m_smallintPtr);
+ break;
+ case UsmallintPtr:
+ snprintf(buf, size, "0x%lx", (unsigned long)m_usmallintPtr);
+ break;
+ case IntegerPtr:
+ snprintf(buf, size, "0x%lx", (unsigned long)m_integerPtr);
+ break;
+ case UintegerPtr:
+ snprintf(buf, size, "0x%lx", (unsigned long)m_uintegerPtr);
+ break;
+ case PointerPtr:
+ snprintf(buf, size, "0x%lx", (unsigned long)m_pointerPtr);
+ break;
+ case Sqlchar:
+ snprintf(buf, size, "%s", m_sqlchar);
+ break;
+ case Sqlstate:
+ snprintf(buf, size, "%s", m_sqlstate->state());
+ break;
+ default:
+ snprintf(buf, size, "data(%d)", (int)m_type);
+ break;
+ };
+}
diff --git a/ndb/src/client/odbc/common/OdbcData.hpp b/ndb/src/client/odbc/common/OdbcData.hpp
new file mode 100644
index 00000000000..c1884507cfe
--- /dev/null
+++ b/ndb/src/client/odbc/common/OdbcData.hpp
@@ -0,0 +1,283 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_OdbcData_hpp
+#define ODBC_COMMON_OdbcData_hpp
+
+#include <ndb_types.h>
+#include <common/common.hpp>
+
+/**
+ * @class OdbcData
+ * @brief Odbc data types and storage
+ *
+ * Stores diagnostics, attributes, and descriptors. Also used
+ * for converting to and from driver function arguments.
+ */
+class OdbcData {
+public:
+ enum Type {
+ Undef = 0,
+ Smallint,
+ Usmallint,
+ Integer,
+ Uinteger,
+ Pointer,
+ SmallintPtr,
+ UsmallintPtr,
+ IntegerPtr,
+ UintegerPtr,
+ PointerPtr,
+ Sqlchar,
+ Sqlstate
+ };
+ OdbcData();
+ OdbcData(Type type);
+ OdbcData(const OdbcData& odbcData);
+ OdbcData(SQLSMALLINT smallint);
+ OdbcData(SQLUSMALLINT usmallint);
+ OdbcData(SQLINTEGER integer);
+ OdbcData(SQLUINTEGER uinteger);
+ OdbcData(SQLPOINTER pointer);
+ OdbcData(SQLSMALLINT* smallintPtr);
+ OdbcData(SQLUSMALLINT* usmallintPtr);
+ OdbcData(SQLINTEGER* integerPtr);
+ OdbcData(SQLUINTEGER* uintegerPtr);
+ OdbcData(SQLPOINTER* pointerPtr);
+ OdbcData(const char* sqlchar);
+ OdbcData(const ::Sqlstate& sqlstate);
+ ~OdbcData();
+ Type type() const;
+ void setValue();
+ void setValue(Type type);
+ void setValue(const OdbcData odbcData);
+ // get value
+ SQLSMALLINT smallint() const;
+ SQLUSMALLINT usmallint() const;
+ SQLINTEGER integer() const;
+ SQLUINTEGER uinteger() const;
+ SQLPOINTER pointer() const;
+ SQLSMALLINT* smallintPtr() const;
+ SQLUSMALLINT* usmallintPtr() const;
+ SQLINTEGER* integerPtr() const;
+ SQLUINTEGER* uintegerPtr() const;
+ SQLPOINTER* pointerPtr() const;
+ const char* sqlchar() const;
+ const ::Sqlstate& sqlstate() const;
+ // copy in from user buffer
+ void copyin(Ctx& ctx, Type type, SQLPOINTER buf, SQLINTEGER length);
+ // copy out to user buffer
+ void copyout(Ctx& ctx, SQLPOINTER buf, SQLINTEGER length, SQLINTEGER* total, SQLSMALLINT* total2 = 0);
+ // logging
+ void print(char* buf, unsigned size) const;
+private:
+ OdbcData& operator=(const OdbcData& odbcData); // disallowed
+ Type m_type;
+ union {
+ SQLSMALLINT m_smallint;
+ SQLUSMALLINT m_usmallint;
+ SQLINTEGER m_integer;
+ SQLUINTEGER m_uinteger;
+ SQLPOINTER m_pointer;
+ SQLSMALLINT* m_smallintPtr;
+ SQLUSMALLINT* m_usmallintPtr;
+ SQLINTEGER* m_integerPtr;
+ SQLUINTEGER* m_uintegerPtr;
+ SQLPOINTER* m_pointerPtr;
+ char* m_sqlchar;
+ const ::Sqlstate* m_sqlstate;
+ };
+};
+
+inline OdbcData::Type
+OdbcData::type() const
+{
+ return m_type;
+}
+
+inline
+OdbcData::OdbcData(SQLSMALLINT smallint) :
+ m_type(Smallint),
+ m_smallint(smallint)
+{
+}
+
+inline
+OdbcData::OdbcData(SQLUSMALLINT usmallint) :
+ m_type(Usmallint),
+ m_usmallint(usmallint)
+{
+}
+
+inline
+OdbcData::OdbcData(SQLINTEGER integer) :
+ m_type(Integer),
+ m_integer(integer)
+{
+}
+
+inline
+OdbcData::OdbcData(SQLUINTEGER uinteger) :
+ m_type(Uinteger),
+ m_uinteger(uinteger)
+{
+}
+
+inline
+OdbcData::OdbcData(SQLPOINTER pointer) :
+ m_type(Pointer),
+ m_pointer(pointer)
+{
+}
+
+inline
+OdbcData::OdbcData(SQLSMALLINT* smallintPtr) :
+ m_type(SmallintPtr),
+ m_smallintPtr(smallintPtr)
+{
+}
+
+inline
+OdbcData::OdbcData(SQLUSMALLINT* usmallintPtr) :
+ m_type(UsmallintPtr),
+ m_usmallintPtr(usmallintPtr)
+{
+}
+
+inline
+OdbcData::OdbcData(SQLINTEGER* integerPtr) :
+ m_type(IntegerPtr),
+ m_integerPtr(integerPtr)
+{
+}
+
+inline
+OdbcData::OdbcData(SQLUINTEGER* uintegerPtr) :
+ m_type(UintegerPtr),
+ m_uintegerPtr(uintegerPtr)
+{
+}
+
+inline
+OdbcData::OdbcData(SQLPOINTER* pointerPtr) :
+ m_type(PointerPtr),
+ m_pointerPtr(pointerPtr)
+{
+}
+
+inline
+OdbcData::OdbcData(const char* sqlchar) :
+ m_type(Sqlchar)
+{
+ unsigned n = strlen(sqlchar);
+ m_sqlchar = new char[n + 1];
+ strcpy(m_sqlchar, sqlchar);
+}
+
+inline
+OdbcData::OdbcData(const ::Sqlstate& sqlstate) :
+ m_type(Sqlstate),
+ m_sqlstate(&sqlstate)
+{
+}
+
+// get value
+
+inline SQLSMALLINT
+OdbcData::smallint() const
+{
+ ctx_assert(m_type == Smallint);
+ return m_smallint;
+}
+
+inline SQLUSMALLINT
+OdbcData::usmallint() const
+{
+ ctx_assert(m_type == Usmallint);
+ return m_usmallint;
+}
+
+inline SQLINTEGER
+OdbcData::integer() const
+{
+ ctx_assert(m_type == Integer);
+ return m_integer;
+}
+
+inline SQLUINTEGER
+OdbcData::uinteger() const
+{
+ ctx_assert(m_type == Uinteger);
+ return m_uinteger;
+}
+
+inline SQLPOINTER
+OdbcData::pointer() const
+{
+ ctx_assert(m_type == Pointer);
+ return m_pointer;
+}
+
+inline SQLSMALLINT*
+OdbcData::smallintPtr() const
+{
+ ctx_assert(m_type == SmallintPtr);
+ return m_smallintPtr;
+}
+
+inline SQLUSMALLINT*
+OdbcData::usmallintPtr() const
+{
+ ctx_assert(m_type == UsmallintPtr);
+ return m_usmallintPtr;
+}
+
+inline SQLINTEGER*
+OdbcData::integerPtr() const
+{
+ ctx_assert(m_type == IntegerPtr);
+ return m_integerPtr;
+}
+
+inline SQLUINTEGER*
+OdbcData::uintegerPtr() const
+{
+ ctx_assert(m_type == UintegerPtr);
+ return m_uintegerPtr;
+}
+
+inline SQLPOINTER*
+OdbcData::pointerPtr() const
+{
+ ctx_assert(m_type == PointerPtr);
+ return m_pointerPtr;
+}
+
+inline const char*
+OdbcData::sqlchar() const
+{
+ ctx_assert(m_type == Sqlchar);
+ return m_sqlchar;
+}
+
+inline const ::Sqlstate&
+OdbcData::sqlstate() const
+{
+ ctx_assert(m_type == Sqlstate);
+ return *m_sqlstate;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/ResultArea.cpp b/ndb/src/client/odbc/common/ResultArea.cpp
new file mode 100644
index 00000000000..79d7fb0ccc4
--- /dev/null
+++ b/ndb/src/client/odbc/common/ResultArea.cpp
@@ -0,0 +1,29 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "ResultArea.hpp"
+
+// ResultArea
+
+ResultArea::~ResultArea()
+{
+}
+
+// ResultSet
+
+ResultSet::~ResultSet()
+{
+}
diff --git a/ndb/src/client/odbc/common/ResultArea.hpp b/ndb/src/client/odbc/common/ResultArea.hpp
new file mode 100644
index 00000000000..d4890c44d99
--- /dev/null
+++ b/ndb/src/client/odbc/common/ResultArea.hpp
@@ -0,0 +1,162 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_ResultArea_hpp
+#define ODBC_COMMON_ResultArea_hpp
+
+#include <common/common.hpp>
+#include <codegen/Code_base.hpp>
+
+class SqlRow;
+
+/**
+ * @class ResultArea
+ * @brief Execution result
+ *
+ * ResultArea contains general results from executing SQL
+ * statements. Data rows from queries are fetched via derived
+ * class ResultSet.
+ */
+class ResultArea {
+public:
+ ResultArea();
+ virtual ~ResultArea();
+ /**
+ * Get number of rows:
+ *
+ * - for queries: number of rows fetched so far
+ * - for DML statements: number of rows affected
+ * - for DDL and other statements: 0
+ */
+ CountType getCount() const;
+protected:
+ void setCount(CountType count);
+ void addCount(unsigned count = 1);
+private:
+ CountType m_count;
+};
+
+inline
+ResultArea::ResultArea() :
+ m_count(0)
+{
+}
+
+inline CountType
+ResultArea::getCount() const
+{
+ return m_count;
+}
+
+inline void
+ResultArea::setCount(CountType count)
+{
+ m_count = count;
+}
+
+inline void
+ResultArea::addCount(unsigned count)
+{
+ m_count += count;
+}
+
+/**
+ * @class ResultSet
+ * @brief Data rows from queries
+ *
+ * ResultSet is an interface implemented by SQL queries and
+ * virtual queries (such as SQLTables). It has an associated
+ * data row accessor SqlRow.
+ */
+class ResultSet : public ResultArea {
+public:
+ ResultSet(const SqlRow& dataRow);
+ virtual ~ResultSet();
+ enum State {
+ State_init = 1, // before first fetch
+ State_ok = 2, // last fetch succeeded
+ State_end = 3 // end of fetch or error
+ };
+ void initState();
+ State getState() const;
+ /**
+ * Get data accessor. Can be retrieved once and used after
+ * each successful ResultSet::fetch().
+ */
+ const SqlRow& dataRow() const;
+ /**
+ * Try to fetch one more row from this result set.
+ * - returns true if a row was fetched
+ * - returns false on end of fetch
+ * - returns false on error and sets error status
+ * It is a fatal error to call fetch after end of fetch.
+ */
+ bool fetch(Ctx& ctx, Exec_base::Ctl& ctl);
+protected:
+ /**
+ * Implementation of ResultSet::fetch() must be provided by
+ * each concrete subclass.
+ */
+ virtual bool fetchImpl(Ctx& ctx, Exec_base::Ctl& ctl) = 0;
+ const SqlRow& m_dataRow;
+ State m_state;
+};
+
+inline
+ResultSet::ResultSet(const SqlRow& dataRow) :
+ m_dataRow(dataRow),
+ m_state(State_end) // explicit initState() is required
+{
+}
+
+inline const SqlRow&
+ResultSet::dataRow() const
+{
+ return m_dataRow;
+}
+
+inline void
+ResultSet::initState()
+{
+ m_state = State_init;
+ setCount(0);
+}
+
+inline ResultSet::State
+ResultSet::getState() const
+{
+ return m_state;
+}
+
+inline bool
+ResultSet::fetch(Ctx& ctx, Exec_base::Ctl& ctl)
+{
+ if (! (m_state == State_init || m_state == State_ok)) {
+ // should not happen but return error instead of assert
+ ctx.pushStatus(Error::Gen, "invalid fetch state %d", (int)m_state);
+ m_state = State_end;
+ return false;
+ }
+ if (fetchImpl(ctx, ctl)) {
+ m_state = State_ok;
+ addCount();
+ return true;
+ }
+ m_state = State_end;
+ return false;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/Sqlstate.cpp b/ndb/src/client/odbc/common/Sqlstate.cpp
new file mode 100644
index 00000000000..2d625a7c159
--- /dev/null
+++ b/ndb/src/client/odbc/common/Sqlstate.cpp
@@ -0,0 +1,93 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/common.hpp>
+
+// Initialize Sqlstate records statically.
+// They are not used by other static initializers.
+
+#define make_Sqlstate(state, code) \
+ const Sqlstate Sqlstate::_##state(#state, code)
+
+make_Sqlstate(00000, SQL_SUCCESS);
+make_Sqlstate(01004, SQL_SUCCESS_WITH_INFO);
+make_Sqlstate(01S02, SQL_SUCCESS_WITH_INFO);
+make_Sqlstate(07009, SQL_ERROR);
+make_Sqlstate(08003, SQL_ERROR);
+make_Sqlstate(21S01, SQL_ERROR);
+make_Sqlstate(22001, SQL_ERROR);
+make_Sqlstate(22002, SQL_ERROR);
+make_Sqlstate(22003, SQL_ERROR);
+make_Sqlstate(22005, SQL_ERROR);
+make_Sqlstate(22008, SQL_ERROR);
+make_Sqlstate(22012, SQL_ERROR);
+make_Sqlstate(24000, SQL_ERROR);
+make_Sqlstate(25000, SQL_ERROR);
+make_Sqlstate(42000, SQL_ERROR);
+make_Sqlstate(42S02, SQL_ERROR);
+make_Sqlstate(42S22, SQL_ERROR);
+make_Sqlstate(HY004, SQL_ERROR);
+make_Sqlstate(HY009, SQL_ERROR);
+make_Sqlstate(HY010, SQL_ERROR);
+make_Sqlstate(HY011, SQL_ERROR);
+make_Sqlstate(HY012, SQL_ERROR);
+make_Sqlstate(HY014, SQL_ERROR);
+make_Sqlstate(HY019, SQL_ERROR);
+make_Sqlstate(HY024, SQL_ERROR);
+make_Sqlstate(HY090, SQL_ERROR);
+make_Sqlstate(HY091, SQL_ERROR);
+make_Sqlstate(HY092, SQL_ERROR);
+make_Sqlstate(HY095, SQL_ERROR);
+make_Sqlstate(HY096, SQL_ERROR);
+make_Sqlstate(HYC00, SQL_ERROR);
+make_Sqlstate(HYT00, SQL_ERROR);
+make_Sqlstate(IM000, SQL_ERROR); // consider all to be errors for now
+make_Sqlstate(IM001, SQL_ERROR);
+make_Sqlstate(IM999, SQL_ERROR);
+
+SQLRETURN
+Sqlstate::getCode(SQLRETURN code) const
+{
+ int codes[2];
+ int ranks[2];
+ codes[0] = code;
+ codes[1] = m_code;
+ for (int i = 0; i < 2; i++) {
+ switch (codes[i]) {
+ case SQL_SUCCESS:
+ ranks[i] = 0;
+ break;
+ case SQL_SUCCESS_WITH_INFO:
+ ranks[i] = 1;
+ break;
+ case SQL_NO_DATA:
+ ranks[i] = 2;
+ break;
+ case SQL_NEED_DATA:
+ ranks[i] = 3;
+ break;
+ case SQL_ERROR:
+ ranks[i] = 9;
+ break;
+ default:
+ ctx_assert(false);
+ ranks[i] = 9;
+ }
+ }
+ if (ranks[0] < ranks[1])
+ code = m_code;
+ return code;
+}
diff --git a/ndb/src/client/odbc/common/Sqlstate.hpp b/ndb/src/client/odbc/common/Sqlstate.hpp
new file mode 100644
index 00000000000..3b4665dc6ca
--- /dev/null
+++ b/ndb/src/client/odbc/common/Sqlstate.hpp
@@ -0,0 +1,85 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_SqlState_hpp
+#define ODBC_COMMON_SqlState_hpp
+
+#include <string.h>
+
+/**
+ * SQL states.
+ */
+class Sqlstate {
+public:
+ static const Sqlstate _00000; // no state
+ static const Sqlstate _01004; // data converted with truncation
+ static const Sqlstate _01S02; // option value changed
+ static const Sqlstate _07009; // invalid descriptor index
+ static const Sqlstate _08003; // connection not open
+ static const Sqlstate _21S01; // insert value list does not match column list
+ static const Sqlstate _22001; // string data, right truncation
+ static const Sqlstate _22002; // indicator variable required but not supplied
+ static const Sqlstate _22003; // data overflow
+ static const Sqlstate _22005; // data is not numeric-literal
+ static const Sqlstate _22008; // data value is not a valid timestamp
+ static const Sqlstate _22012; // division by zero
+ static const Sqlstate _24000; // invalid cursor state
+ static const Sqlstate _25000; // invalid transaction state
+ static const Sqlstate _42000; // syntax error or access violation
+ static const Sqlstate _42S02; // base table or view not found
+ static const Sqlstate _42S22; // column not found
+ static const Sqlstate _HY004; // invalid SQL data type
+ static const Sqlstate _HY009; // invalid use of null pointer
+ static const Sqlstate _HY010; // function sequence error
+ static const Sqlstate _HY011; // attribute cannot be set now
+ static const Sqlstate _HY012; // invalid transaction operation code
+ static const Sqlstate _HY014; // limit on number of handles exceeded
+ static const Sqlstate _HY019; // non-character and non-binary data sent in pieces
+ static const Sqlstate _HY024; // invalid attribute value
+ static const Sqlstate _HY090; // invalid string or buffer length
+ static const Sqlstate _HY091; // invalid descriptor field identifier
+ static const Sqlstate _HY092; // invalid attribute/option identifier
+ static const Sqlstate _HY095; // function type out of range
+ static const Sqlstate _HY096; // information type out of range
+ static const Sqlstate _HYC00; // optional feature not implemented
+ static const Sqlstate _HYT00; // timeout expired
+ static const Sqlstate _IM000; // implementation defined
+ static const Sqlstate _IM001; // driver does not support this function
+ static const Sqlstate _IM999; // fill in the right state please
+ // get the 5-char text string
+ const char* state() const;
+ // get code or "upgrade" existing code
+ SQLRETURN getCode(SQLRETURN code = SQL_SUCCESS) const;
+private:
+ Sqlstate(const char* state, const SQLRETURN code);
+ const char* const m_state;
+ const SQLRETURN m_code;
+};
+
+inline const char*
+Sqlstate::state() const
+{
+ return m_state;
+}
+
+inline
+Sqlstate::Sqlstate(const char* state, const SQLRETURN code) :
+ m_state(state),
+ m_code(code)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/StmtArea.cpp b/ndb/src/client/odbc/common/StmtArea.cpp
new file mode 100644
index 00000000000..5ce2d47d31a
--- /dev/null
+++ b/ndb/src/client/odbc/common/StmtArea.cpp
@@ -0,0 +1,112 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "DiagArea.hpp"
+#include "StmtArea.hpp"
+#include <codegen/CodeGen.hpp>
+
+StmtArea::StmtArea(ConnArea& connArea) :
+ m_connArea(connArea),
+ m_state(Free),
+ m_useSchemaCon(false),
+ m_useConnection(false),
+ m_planTree(0),
+ m_execTree(0),
+ m_unbound(0),
+ m_rowCount(0),
+ m_tuplesFetched(0)
+{
+ for (unsigned i = 0; i <= 4; i++)
+ m_descArea[i] = 0;
+}
+
+StmtArea::~StmtArea()
+{
+}
+
+void
+StmtArea::free(Ctx &ctx)
+{
+ CodeGen codegen(*this);
+ codegen.close(ctx);
+ codegen.free(ctx);
+ m_sqlText.assign("");
+ m_nativeText.assign("");
+ useSchemaCon(ctx, false);
+ useConnection(ctx, false);
+ m_stmtInfo.free(ctx);
+ m_planTree = 0;
+ m_execTree = 0;
+ m_rowCount = 0;
+ m_tuplesFetched = 0;
+ m_unbound = 0;
+ m_state = Free;
+}
+
+void
+StmtArea::setRowCount(Ctx& ctx, CountType rowCount)
+{
+ m_rowCount = rowCount;
+ // location
+ DescArea& ird = descArea(Desc_usage_IRD);
+ OdbcData data;
+ ird.getHeader().getField(ctx, SQL_DESC_ROWS_PROCESSED_PTR, data);
+ if (data.type() != OdbcData::Undef) {
+ SQLUINTEGER* countPtr = data.uintegerPtr();
+ if (countPtr != 0) {
+ *countPtr = static_cast<SQLUINTEGER>(m_rowCount);
+ }
+ }
+ // diagnostic
+ SQLINTEGER count = static_cast<SQLINTEGER>(m_rowCount);
+ ctx.diagArea().setHeader(SQL_DIAG_ROW_COUNT, count);
+}
+
+void
+StmtArea::setFunction(Ctx& ctx, const char* function, SQLINTEGER functionCode)
+{
+ m_stmtInfo.m_function = function;
+ m_stmtInfo.m_functionCode = functionCode;
+}
+
+void
+StmtArea::setFunction(Ctx& ctx)
+{
+ OdbcData function(m_stmtInfo.m_function);
+ ctx.diagArea().setHeader(SQL_DIAG_DYNAMIC_FUNCTION, function);
+ OdbcData functionCode(m_stmtInfo.m_functionCode);
+ ctx.diagArea().setHeader(SQL_DIAG_DYNAMIC_FUNCTION_CODE, functionCode);
+}
+
+bool
+StmtArea::useSchemaCon(Ctx& ctx, bool use)
+{
+ if (m_useSchemaCon != use)
+ if (! m_connArea.useSchemaCon(ctx, use))
+ return false;
+ m_useSchemaCon = use;
+ return true;
+}
+
+bool
+StmtArea::useConnection(Ctx& ctx, bool use)
+{
+ if (m_useConnection != use)
+ if (! m_connArea.useConnection(ctx, use))
+ return false;
+ m_useConnection = use;
+ return true;
+}
diff --git a/ndb/src/client/odbc/common/StmtArea.hpp b/ndb/src/client/odbc/common/StmtArea.hpp
new file mode 100644
index 00000000000..a88c6d36e6d
--- /dev/null
+++ b/ndb/src/client/odbc/common/StmtArea.hpp
@@ -0,0 +1,157 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_StmtArea_hpp
+#define ODBC_COMMON_StmtArea_hpp
+
+#include <common/common.hpp>
+#include "ConnArea.hpp"
+#include "StmtInfo.hpp"
+#include "DescArea.hpp"
+
+class PlanTree;
+class ExecTree;
+
+/**
+ * @class StmtArea
+ * @brief Public part of statement handle
+ */
+class StmtArea {
+public:
+ // state between ODBC function calls
+ enum State {
+ Free = 1, // not in use
+ Prepared = 2, // statement prepared, maybe unbound
+ NeedData = 3, // executed, SQLParamData expected
+ Open = 4 // cursor open
+ };
+ // connection area shared by all statements
+ ConnArea& connArea();
+ State getState() const;
+ // SQL text
+ const BaseString& sqlText();
+ BaseString& nativeText();
+ // allocate or unallocate connections if necessary
+ bool useSchemaCon(Ctx& ctx, bool use);
+ bool useConnection(Ctx& ctx, bool use);
+ // statement info
+ StmtInfo& stmtInfo();
+ DescArea& descArea(DescUsage u);
+ unsigned unbound() const;
+ // set row count here and in diagnostics
+ void setRowCount(Ctx& ctx, CountType rowCount);
+ CountType getRowCount() const;
+ // raw tuple count (tuples fetched from NDB)
+ void resetTuplesFetched();
+ void incTuplesFetched();
+ CountType getTuplesFetched() const;
+ // set dynamic function in StmtInfo only (at prepare)
+ void setFunction(Ctx& ctx, const char* function, SQLINTEGER functionCode);
+ // set dynamic function in diagnostics (at execute)
+ void setFunction(Ctx& ctx);
+protected:
+ friend class CodeGen;
+ friend class Executor;
+ friend class Plan_root;
+ StmtArea(ConnArea& connArea);
+ ~StmtArea();
+ void free(Ctx& ctx);
+ ConnArea& m_connArea;
+ State m_state;
+ BaseString m_sqlText;
+ BaseString m_nativeText;
+ bool m_useSchemaCon;
+ bool m_useConnection;
+ StmtInfo m_stmtInfo;
+ // plan tree output from parser and rewritten by analyze
+ PlanTree* m_planTree;
+ // exec tree output from analyze
+ ExecTree* m_execTree;
+ // pointers within HandleDesc allocated via HandleStmt
+ DescArea* m_descArea[1+4];
+ // parameters with unbound SQL type
+ unsigned m_unbound;
+ CountType m_rowCount;
+ CountType m_tuplesFetched;
+};
+
+inline ConnArea&
+StmtArea::connArea()
+{
+ return m_connArea;
+}
+
+inline StmtArea::State
+StmtArea::getState() const
+{
+ return m_state;
+}
+
+inline const BaseString&
+StmtArea::sqlText() {
+ return m_sqlText;
+}
+
+inline BaseString&
+StmtArea::nativeText() {
+ return m_nativeText;
+}
+
+inline StmtInfo&
+StmtArea::stmtInfo()
+{
+ return m_stmtInfo;
+}
+
+inline DescArea&
+StmtArea::descArea(DescUsage u)
+{
+ ctx_assert(1 <= u && u <= 4);
+ ctx_assert(m_descArea[u] != 0);
+ return *m_descArea[u];
+}
+
+inline unsigned
+StmtArea::unbound() const
+{
+ return m_unbound;
+}
+
+inline CountType
+StmtArea::getRowCount() const
+{
+ return m_rowCount;
+}
+
+inline void
+StmtArea::resetTuplesFetched()
+{
+ m_tuplesFetched = 0;
+}
+
+inline void
+StmtArea::incTuplesFetched()
+{
+ m_tuplesFetched++;
+}
+
+inline CountType
+StmtArea::getTuplesFetched() const
+{
+ return m_tuplesFetched;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/StmtInfo.cpp b/ndb/src/client/odbc/common/StmtInfo.cpp
new file mode 100644
index 00000000000..3467fb5023e
--- /dev/null
+++ b/ndb/src/client/odbc/common/StmtInfo.cpp
@@ -0,0 +1,78 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "StmtInfo.hpp"
+
+const char*
+StmtInfo::getDesc() const
+{
+ switch (m_name) {
+ case Stmt_name_select:
+ return "select";
+ case Stmt_name_insert:
+ return "insert";
+ case Stmt_name_update:
+ return "update";
+ case Stmt_name_delete:
+ return "delete";
+ case Stmt_name_create_table:
+ return "create table";
+ case Stmt_name_create_index:
+ return "create index";
+ case Stmt_name_drop_table:
+ return "drop table";
+ case Stmt_name_drop_index:
+ return "drop index";
+ default:
+ ctx_assert(false);
+ break;
+ }
+ return "";
+}
+
+StmtType
+StmtInfo::getType() const
+{
+ StmtType type = Stmt_type_undef;
+ switch (m_name) {
+ case Stmt_name_select: // query
+ type = Stmt_type_query;
+ break;
+ case Stmt_name_insert: // DML
+ case Stmt_name_update:
+ case Stmt_name_delete:
+ type = Stmt_type_DML;
+ break;
+ case Stmt_name_create_table: // DDL
+ case Stmt_name_create_index:
+ case Stmt_name_drop_table:
+ case Stmt_name_drop_index:
+ type = Stmt_type_DDL;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ return type;
+}
+
+void
+StmtInfo::free(Ctx& ctx)
+{
+ m_name = Stmt_name_undef;
+ m_function = "";
+ m_functionCode = SQL_DIAG_UNKNOWN_STATEMENT;
+}
diff --git a/ndb/src/client/odbc/common/StmtInfo.hpp b/ndb/src/client/odbc/common/StmtInfo.hpp
new file mode 100644
index 00000000000..9cd489be6da
--- /dev/null
+++ b/ndb/src/client/odbc/common/StmtInfo.hpp
@@ -0,0 +1,86 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_StmtInfo_hpp
+#define ODBC_COMMON_StmtInfo_hpp
+
+#include <common/common.hpp>
+
+// general type (determined by SQL command)
+enum StmtType {
+ Stmt_type_undef = 0,
+ Stmt_type_query, // select
+ Stmt_type_DML, // insert, update, delete
+ Stmt_type_DDL, // create, drop, alter table
+ Stmt_type_info // virtual query
+};
+
+// specific SQL command (first 1-2 words)
+enum StmtName {
+ Stmt_name_undef = 0,
+ Stmt_name_select,
+ Stmt_name_insert,
+ Stmt_name_update,
+ Stmt_name_delete,
+ Stmt_name_create_table,
+ Stmt_name_create_index,
+ Stmt_name_drop_table,
+ Stmt_name_drop_index
+};
+
+/**
+ * @class StmtInfo
+ * @brief Statement Info
+ *
+ * Statement info. This is a separate class which could
+ * be used in cases not tied to statement execution.
+ */
+class StmtInfo {
+public:
+ StmtInfo();
+ void setName(StmtName name);
+ StmtName getName() const;
+ const char* getDesc() const;
+ StmtType getType() const;
+ void free(Ctx& ctx);
+private:
+ friend class StmtArea;
+ StmtName m_name;
+ const char* m_function; // not allocated
+ SQLINTEGER m_functionCode;
+};
+
+inline
+StmtInfo::StmtInfo() :
+ m_name(Stmt_name_undef),
+ m_function(""),
+ m_functionCode(SQL_DIAG_UNKNOWN_STATEMENT)
+{
+}
+
+inline void
+StmtInfo::setName(StmtName name)
+{
+ m_name = name;
+}
+
+inline StmtName
+StmtInfo::getName() const
+{
+ return m_name;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/common/common.cpp b/ndb/src/client/odbc/common/common.cpp
new file mode 100644
index 00000000000..73d14c82efe
--- /dev/null
+++ b/ndb/src/client/odbc/common/common.cpp
@@ -0,0 +1,17 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "common.hpp"
diff --git a/ndb/src/client/odbc/common/common.hpp b/ndb/src/client/odbc/common/common.hpp
new file mode 100644
index 00000000000..e90950df9c5
--- /dev/null
+++ b/ndb/src/client/odbc/common/common.hpp
@@ -0,0 +1,120 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_COMMON_common_hpp
+#define ODBC_COMMON_common_hpp
+
+// misc defs
+
+#ifdef NDB_GCC
+#define PRINTFLIKE(i,j) __attribute__ ((format (printf, i, j)))
+#else
+#define PRINTFLIKE(i,j)
+#endif
+
+// odbc defs
+
+#define ODBCVER 0x0351
+
+#ifdef NDB_WIN32
+#include <windows.h>
+#endif
+
+extern "C" {
+#include <sqlext.h>
+}
+// some types which may be missing
+#ifndef SQL_BLOB
+#define SQL_BLOB 30
+#endif
+#ifndef SQL_BLOB_LOCATOR
+#define SQL_BLOB_LOCATOR 31
+#endif
+#ifndef SQL_CLOB
+#define SQL_CLOB 40
+#endif
+#ifndef SQL_CLOB_LOCATOR
+#define SQL_CLOB_LOCATOR 41
+#endif
+
+// until real blobs use Varchar of this size
+#define FAKE_BLOB_SIZE 2000
+
+#define SQL_HANDLE_ROOT 0 // assume real handles != 0
+
+enum OdbcHandle {
+ Odbc_handle_root = 0, // not an odbc handle
+ Odbc_handle_env = 1,
+ Odbc_handle_dbc = 2,
+ Odbc_handle_stmt = 4,
+ Odbc_handle_desc = 8,
+ Odbc_handle_all = (1|2|4|8)
+};
+
+// ndb defs
+
+#undef BOOL
+#include <ndb_types.h>
+// this info not yet on api side
+#include <kernel/ndb_limits.h>
+#include <ndb_version.h>
+
+#ifndef MAX_TAB_NAME_SIZE
+#define MAX_TAB_NAME_SIZE 128
+#endif
+
+#ifndef MAX_ATTR_NAME_SIZE
+#define MAX_ATTR_NAME_SIZE 32
+#endif
+
+#ifndef MAX_ATTR_DEFAULT_VALUE_SIZE
+#define MAX_ATTR_DEFAULT_VALUE_SIZE 128
+#endif
+
+typedef Uint32 NdbAttrId;
+typedef Uint64 CountType;
+
+// ndb odbc defs
+
+#define NDB_ODBC_COMPONENT_VENDOR "[MySQL]"
+#define NDB_ODBC_COMPONENT_DRIVER "[ODBC driver]"
+#define NDB_ODBC_COMPONENT_DATABASE "[NDB Cluster]"
+
+#define NDB_ODBC_VERSION_MAJOR 0
+#define NDB_ODBC_VERSION_MINOR 22
+#define NDB_ODBC_VERSION_STRING "0.22"
+
+// reserved error codes for non-NDB errors
+#define NDB_ODBC_ERROR_MIN 5000
+#define NDB_ODBC_ERROR_MAX 5100
+
+// maximum log level compiled in
+#ifdef VM_TRACE
+#define NDB_ODBC_MAX_LOG_LEVEL 5
+#else
+#define NDB_ODBC_MAX_LOG_LEVEL 3
+#endif
+
+// driver specific statement attribute for number of NDB tuples fetched
+#define SQL_ATTR_NDB_TUPLES_FETCHED 66601
+
+#include <BaseString.hpp>
+#include <common/Sqlstate.hpp>
+#include <common/Ctx.hpp>
+
+#undef assert
+
+#endif
diff --git a/ndb/src/client/odbc/dictionary/DictCatalog.cpp b/ndb/src/client/odbc/dictionary/DictCatalog.cpp
new file mode 100644
index 00000000000..433347c9a70
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictCatalog.cpp
@@ -0,0 +1,42 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/ConnArea.hpp>
+#include "DictCatalog.hpp"
+#include "DictSchema.hpp"
+
+DictCatalog::~DictCatalog()
+{
+ for (Schemas::iterator i = m_schemas.begin(); i != m_schemas.end(); i++) {
+ delete *i;
+ *i = 0;
+ }
+}
+
+DictSchema*
+DictCatalog::findSchema(Ctx& ctx, const BaseString& name)
+{
+ for (Schemas::iterator i = m_schemas.begin(); i != m_schemas.end(); i++) {
+ DictSchema* schema = *i;
+ ctx_assert(schema != 0);
+ if (strcmp(schema->getName().c_str(), name.c_str()) == 0)
+ return schema;
+ }
+ ctx_assert(strcmp(name.c_str(), "NDB") == 0);
+ DictSchema* schema = new DictSchema(m_connArea, "NDB");
+ m_schemas.push_back(schema);
+ return schema;
+}
diff --git a/ndb/src/client/odbc/dictionary/DictCatalog.hpp b/ndb/src/client/odbc/dictionary/DictCatalog.hpp
new file mode 100644
index 00000000000..5452990a51b
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictCatalog.hpp
@@ -0,0 +1,64 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_DICTIONARY_DictCatalog_hpp
+#define ODBC_DICTIONARY_DictCatalog_hpp
+
+#include <list>
+#include <common/common.hpp>
+#include "DictSchema.hpp"
+
+class Ctx;
+class ConnArea;
+class DictSchema;
+
+/**
+ * @class DictCatalog
+ * @brief Collection of schemas
+ */
+class DictCatalog {
+public:
+ DictCatalog(const ConnArea& connArea);
+ ~DictCatalog();
+ const ConnArea& connArea() const;
+ DictSchema* findSchema(Ctx& ctx, const BaseString& name);
+ void addSchema(DictSchema* schema);
+protected:
+ const ConnArea& m_connArea;
+ typedef std::list<DictSchema*> Schemas;
+ Schemas m_schemas;
+};
+
+inline
+DictCatalog::DictCatalog(const ConnArea& connArea) :
+ m_connArea(connArea)
+{
+}
+
+inline const ConnArea&
+DictCatalog::connArea() const
+{
+ return m_connArea;
+}
+
+inline void
+DictCatalog::addSchema(DictSchema* schema)
+{
+ m_schemas.push_back(schema);
+ schema->setParent(this);
+}
+
+#endif
diff --git a/ndb/src/client/odbc/dictionary/DictColumn.cpp b/ndb/src/client/odbc/dictionary/DictColumn.cpp
new file mode 100644
index 00000000000..fa0128f1ddb
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictColumn.cpp
@@ -0,0 +1,23 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "DictColumn.hpp"
+
+DictColumn::~DictColumn()
+{
+ delete[] m_defaultValue;
+ m_defaultValue = 0;
+}
diff --git a/ndb/src/client/odbc/dictionary/DictColumn.hpp b/ndb/src/client/odbc/dictionary/DictColumn.hpp
new file mode 100644
index 00000000000..945fb86367b
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictColumn.hpp
@@ -0,0 +1,143 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_DICTIONARY_DictColumn_hpp
+#define ODBC_DICTIONARY_DictColumn_hpp
+
+#include <common/common.hpp>
+#include <common/DataType.hpp>
+
+class Ctx;
+class SqlType;
+class ConnArea;
+class DictTable;
+
+/**
+ * @class DictColumn
+ * @brief Table column
+ */
+class DictColumn {
+public:
+ DictColumn(const ConnArea& connArea, const BaseString& name, const SqlType& sqlType);
+ ~DictColumn();
+ const BaseString& getName() const;
+ const SqlType& sqlType() const;
+ void setPosition(unsigned position);
+ unsigned getPosition() const;
+ void setParent(DictTable* parent);
+ DictTable* getParent() const;
+ NdbAttrId getAttrId() const;
+ bool isKey() const;
+ bool isTupleId() const;
+ bool isAutoIncrement() const;
+ const char* getDefaultValue() const;
+protected:
+ friend class DictSys;
+ friend class DictTable;
+ const ConnArea& m_connArea;
+ const BaseString m_name;
+ SqlType m_sqlType;
+ unsigned m_position;
+ DictTable* m_parent;
+ bool m_key; // part of key
+ bool m_tupleId; // the tuple id
+ bool m_autoIncrement;
+ const char* m_defaultValue;
+};
+
+inline
+DictColumn::DictColumn(const ConnArea& connArea, const BaseString& name, const SqlType& sqlType) :
+ m_connArea(connArea),
+ m_name(name),
+ m_sqlType(sqlType),
+ m_position(0),
+ m_parent(0),
+ m_key(false),
+ m_tupleId(false),
+ m_autoIncrement(false),
+ m_defaultValue(0)
+{
+}
+
+inline const SqlType&
+DictColumn::sqlType() const
+{
+ return m_sqlType;
+}
+
+inline void
+DictColumn::setPosition(unsigned position)
+{
+ ctx_assert(position != 0);
+ m_position = position;
+}
+
+inline unsigned
+DictColumn::getPosition() const
+{
+ return m_position;
+}
+
+inline void
+DictColumn::setParent(DictTable* parent)
+{
+ m_parent = parent;
+}
+
+inline DictTable*
+DictColumn::getParent() const
+{
+ return m_parent;
+}
+
+inline const BaseString&
+DictColumn::getName() const
+{
+ return m_name;
+}
+
+inline NdbAttrId
+DictColumn::getAttrId() const
+{
+ ctx_assert(m_position != 0);
+ return static_cast<NdbAttrId>(m_position - 1);
+}
+
+inline bool
+DictColumn::isKey() const
+{
+ return m_key;
+}
+
+inline bool
+DictColumn::isTupleId() const
+{
+ return m_tupleId;
+}
+
+inline bool
+DictColumn::isAutoIncrement() const
+{
+ return m_autoIncrement;
+}
+
+inline const char*
+DictColumn::getDefaultValue() const
+{
+ return m_defaultValue;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/dictionary/DictIndex.cpp b/ndb/src/client/odbc/dictionary/DictIndex.cpp
new file mode 100644
index 00000000000..95d93318902
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictIndex.cpp
@@ -0,0 +1,29 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "DictTable.hpp"
+#include "DictIndex.hpp"
+
+DictIndex::~DictIndex()
+{
+}
+
+void
+DictIndex::setColumn(unsigned i, DictColumn* column)
+{
+ ctx_assert(1 <= i && i <= m_size);
+ m_columns[i] = column;
+}
diff --git a/ndb/src/client/odbc/dictionary/DictIndex.hpp b/ndb/src/client/odbc/dictionary/DictIndex.hpp
new file mode 100644
index 00000000000..7ba46daaae3
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictIndex.hpp
@@ -0,0 +1,108 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_DICTIONARY_DictIndex_hpp
+#define ODBC_DICTIONARY_DictIndex_hpp
+
+#include <vector>
+#include <common/common.hpp>
+#include "DictColumn.hpp"
+
+class Ctx;
+class ConnArea;
+class DictTable;
+class DictColumn;
+class DictIndex;
+
+/**
+ * @class DictIndex
+ * @brief Database table
+ */
+class DictIndex {
+ friend class DictSchema;
+public:
+ DictIndex(const ConnArea& connArea, const BaseString& name, NdbDictionary::Object::Type type, unsigned size);
+ ~DictIndex();
+ NdbDictionary::Object::Type getType() const;
+ unsigned getSize() const;
+ void setTable(DictTable* table);
+ DictTable* getTable() const;
+ void setColumn(unsigned i, DictColumn* column);
+ DictColumn* getColumn(unsigned i) const;
+ const BaseString& getName() const;
+ DictColumn* findColumn(const BaseString& name) const;
+protected:
+ const ConnArea& m_connArea;
+ const BaseString m_name;
+ const NdbDictionary::Object::Type m_type;
+ const unsigned m_size;
+ DictSchema* m_parent;
+ DictTable* m_table;
+ typedef std::vector<DictColumn*> Columns; // pointers to table columns
+ Columns m_columns;
+};
+
+inline
+DictIndex::DictIndex(const ConnArea& connArea, const BaseString& name, NdbDictionary::Object::Type type, unsigned size) :
+ m_connArea(connArea),
+ m_name(name),
+ m_type(type),
+ m_size(size),
+ m_parent(0),
+ m_columns(1 + size)
+{
+}
+
+inline NdbDictionary::Object::Type
+DictIndex::getType() const
+{
+ return m_type;
+}
+
+inline unsigned
+DictIndex::getSize() const
+{
+ ctx_assert(m_columns.size() == 1 + m_size);
+ return m_size;
+}
+
+inline void
+DictIndex::setTable(DictTable* table)
+{
+ m_table = table;
+}
+
+inline DictTable*
+DictIndex::getTable() const
+{
+ return m_table;
+}
+
+inline DictColumn*
+DictIndex::getColumn(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= m_size);
+ ctx_assert(m_columns[i] != 0);
+ return m_columns[i];
+}
+
+inline const BaseString&
+DictIndex::getName() const
+{
+ return m_name;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/dictionary/DictSchema.cpp b/ndb/src/client/odbc/dictionary/DictSchema.cpp
new file mode 100644
index 00000000000..91939cb2f26
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictSchema.cpp
@@ -0,0 +1,155 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/ConnArea.hpp>
+#include "DictCatalog.hpp"
+#include "DictSchema.hpp"
+#include "DictTable.hpp"
+#include "DictTable.hpp"
+#include "DictColumn.hpp"
+#include "DictIndex.hpp"
+#include "DictSys.hpp"
+
+DictSchema::~DictSchema()
+{
+ for (Tables::iterator i = m_tables.begin(); i != m_tables.end(); i++) {
+ delete *i;
+ *i = 0;
+ }
+}
+
+DictTable*
+DictSchema::findTable(const BaseString& name)
+{
+ for (Tables::iterator i = m_tables.begin(); i != m_tables.end(); i++) {
+ DictTable* table = *i;
+ ctx_assert(table != 0);
+ if (strcmp(table->getName().c_str(), name.c_str()) == 0)
+ return table;
+ }
+ return 0;
+}
+
+void
+DictSchema::deleteTable(Ctx& ctx, const BaseString& name)
+{
+ Tables::iterator i = m_tables.begin();
+ while (i != m_tables.end()) {
+ DictTable* table = *i;
+ ctx_assert(table != 0);
+ if (strcmp(table->getName().c_str(), name.c_str()) == 0) {
+ ctx_log2(("purge table %s from dictionary", name.c_str()));
+ delete table;
+ Tables::iterator j = i;
+ i++;
+ m_tables.erase(j);
+ break;
+ }
+ i++;
+ }
+}
+
+void
+DictSchema::deleteTableByIndex(Ctx& ctx, const BaseString& indexName)
+{
+ DictTable* foundTable = 0;
+ for (Tables::iterator i = m_tables.begin(); i != m_tables.end(); i++) {
+ DictTable* table = *i;
+ ctx_assert(table != 0);
+ for (unsigned k = 1; k <= table->indexCount(); k++) {
+ const DictIndex* index = table->getIndex(k);
+ if (strcmp(index->getName().c_str(), indexName.c_str()) == 0) {
+ foundTable = table;
+ break;
+ }
+ }
+ if (foundTable != 0)
+ break;
+ }
+ if (foundTable != 0)
+ deleteTable(ctx, foundTable->getName());
+}
+
+DictTable*
+DictSchema::loadTable(Ctx& ctx, const BaseString& name)
+{
+ ctx_log4(("%s: load from NDB", name.c_str()));
+ Ndb* ndb = m_connArea.ndbObject();
+ NdbDictionary::Dictionary* ndbDictionary = ndb->getDictionary();
+ if (ndbDictionary == 0) {
+ ctx.pushStatus(ndb, "getDictionary");
+ return 0;
+ }
+ const NdbDictionary::Table* ndbTable = ndbDictionary->getTable(name.c_str());
+ if (ndbTable == 0) {
+ const NdbError& ndbError = ndbDictionary->getNdbError();
+ if (ndbError.code == 709) {
+ // try built-in system table
+ DictTable* table = DictSys::loadTable(ctx, this, name);
+ if (table != 0) {
+ return table;
+ }
+ ctx_log3(("%s: not found in NDB", name.c_str()));
+ return 0;
+ }
+ ctx.pushStatus(ndbDictionary->getNdbError(), "getTable");
+ return 0;
+ }
+ int nattr = ndbTable->getNoOfColumns();
+ DictTable* table = new DictTable(m_connArea, name, nattr);
+ for (unsigned position = 1; position <= (unsigned)nattr; position++) {
+ DictColumn* column = table->loadColumn(ctx, position);
+ if (column == 0)
+ return 0;
+ ctx_log4(("add column %u %s", column->getPosition(), column->getName().c_str()));
+ }
+ // load indexes
+ NdbDictionary::Dictionary::List list;
+ if (ndbDictionary->listIndexes(list, name.c_str()) == -1) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "listIndexes");
+ return 0;
+ }
+ for (unsigned i = 0; i < list.count; i++) {
+ const NdbDictionary::Dictionary::List::Element& elt = list.elements[i];
+ if (elt.state != NdbDictionary::Object::StateOnline) {
+ ctx_log1(("%s: skip broken index %s", name.c_str(), elt.name));
+ continue;
+ }
+ if (elt.type != NdbDictionary::Object::UniqueHashIndex && elt.type != NdbDictionary::Object::OrderedIndex) {
+ ctx_log1(("%s: skip unknown index type %s", name.c_str(), elt.name));
+ continue;
+ }
+ const NdbDictionary::Index* ndbIndex = ndbDictionary->getIndex(elt.name, name.c_str());
+ if (ndbIndex == 0) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "table %s getIndex %s", name.c_str(), elt.name);
+ return 0;
+ }
+ DictIndex* index = new DictIndex(m_connArea, elt.name, elt.type, ndbIndex->getNoOfIndexColumns());
+ for (unsigned j = 0; j < index->getSize(); j++) {
+ const char* cname = ndbIndex->getIndexColumn(j);
+ ctx_assert(cname != 0);
+ DictColumn* icolumn = table->findColumn(cname);
+ ctx_assert(icolumn != 0);
+ index->setColumn(1 + j, icolumn);
+ }
+ table->addIndex(index);
+ ctx_log3(("%s: index %s: load from NDB done", name.c_str(), elt.name));
+ }
+ addTable(table);
+ ctx_log3(("%s: load from NDB done", name.c_str()));
+ return table;
+}
diff --git a/ndb/src/client/odbc/dictionary/DictSchema.hpp b/ndb/src/client/odbc/dictionary/DictSchema.hpp
new file mode 100644
index 00000000000..099352edbb9
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictSchema.hpp
@@ -0,0 +1,89 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_DICTIONARY_DictSchema_hpp
+#define ODBC_DICTIONARY_DictSchema_hpp
+
+#include <list>
+#include <common/common.hpp>
+#include "DictTable.hpp"
+
+class Ctx;
+class ConnArea;
+class DictCatalog;
+class DictTable;
+
+/**
+ * @class DictSchema
+ * @brief Collection of tables
+ */
+class DictSchema {
+public:
+ DictSchema(const ConnArea& connArea, const BaseString& name);
+ ~DictSchema();
+ const BaseString& getName() const;
+ void setParent(DictCatalog* parent);
+ DictCatalog* getParent() const;
+ void addTable(DictTable* table);
+ DictTable* findTable(const BaseString& name);
+ DictTable* loadTable(Ctx& ctx, const BaseString& name);
+ void deleteTable(Ctx& ctx, const BaseString& name);
+ void deleteTableByIndex(Ctx& ctx, const BaseString& indexName);
+protected:
+ friend class DictCatalog;
+ friend class DictSys;
+ const ConnArea& m_connArea;
+ BaseString m_name;
+ DictCatalog* m_parent;
+ typedef std::list<DictTable*> Tables;
+ Tables m_tables;
+};
+
+inline
+DictSchema::DictSchema(const ConnArea& connArea, const BaseString& name) :
+ m_connArea(connArea),
+ m_name(name),
+ m_parent(0)
+{
+ ctx_assert(strcmp(name.c_str(), "NDB") == 0);
+}
+
+inline const BaseString&
+DictSchema::getName() const
+{
+ return m_name;
+}
+
+inline void
+DictSchema::setParent(DictCatalog* parent)
+{
+ m_parent = parent;
+}
+
+inline DictCatalog*
+DictSchema::getParent() const
+{
+ return m_parent;
+}
+
+inline void
+DictSchema::addTable(DictTable* table)
+{
+ m_tables.push_back(table);
+ table->setParent(this);
+}
+
+#endif
diff --git a/ndb/src/client/odbc/dictionary/DictSys.cpp b/ndb/src/client/odbc/dictionary/DictSys.cpp
new file mode 100644
index 00000000000..1ceef66ee57
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictSys.cpp
@@ -0,0 +1,433 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/ConnArea.hpp>
+#include "DictSchema.hpp"
+#include "DictTable.hpp"
+#include "DictColumn.hpp"
+#include "DictSys.hpp"
+
+#define arraySize(x) sizeof(x)/sizeof(x[0])
+
+#define MAX_SCHEMA_NAME_LENGTH 32
+#define MAX_REMARKS_LENGTH 256
+
+// typeinfo
+
+static DictSys::Column
+column_ODBC_TYPEINFO[] = {
+ DictSys::Column(
+ 1,
+ "TYPE_NAME",
+ false,
+ SqlType(SqlType::Varchar, 20, false)
+ ),
+ DictSys::Column(
+ 2,
+ "DATA_TYPE",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 3,
+ "COLUMN_SIZE",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 4,
+ "LITERAL_PREFIX",
+ false,
+ SqlType(SqlType::Varchar, 1, true)
+ ),
+ DictSys::Column(
+ 5,
+ "LITERAL_SUFFIX",
+ false,
+ SqlType(SqlType::Varchar, 1, true)
+ ),
+ DictSys::Column(
+ 6,
+ "CREATE_PARAMS",
+ false,
+ SqlType(SqlType::Varchar, 20, true)
+ ),
+ DictSys::Column(
+ 7,
+ "NULLABLE",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 8,
+ "CASE_SENSITIVE",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 9,
+ "SEARCHABLE",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 10,
+ "UNSIGNED_ATTRIBUTE",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 11,
+ "FIXED_PREC_SCALE",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 12,
+ "AUTO_UNIQUE_VALUE",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 13,
+ "LOCAL_TYPE_NAME",
+ false,
+ SqlType(SqlType::Varchar, 20, true)
+ ),
+ DictSys::Column(
+ 14,
+ "MINIMUM_SCALE",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 15,
+ "MAXIMUM_SCALE",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 16,
+ "SQL_DATA_TYPE",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 17,
+ "SQL_DATETIME_SUB",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 18,
+ "NUM_PREC_RADIX",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 19,
+ "INTERVAL_PRECISION",
+ false,
+ SqlType(SqlType::Integer, true)
+ )
+};
+
+static DictSys::Table
+table_ODBC_TYPEINFO(
+ DictSys::OdbcTypeinfo,
+ "ODBC$TYPEINFO",
+ column_ODBC_TYPEINFO,
+ arraySize(column_ODBC_TYPEINFO)
+);
+
+// tables
+
+static DictSys::Column
+column_ODBC_TABLES[] = {
+ // perl docs/systables.pl tables -c
+ DictSys::Column(
+ 1,
+ "TABLE_CAT",
+ false,
+ SqlType(SqlType::Varchar, MAX_SCHEMA_NAME_LENGTH, true)
+ ),
+ DictSys::Column(
+ 2,
+ "TABLE_SCHEM",
+ false,
+ SqlType(SqlType::Varchar, MAX_SCHEMA_NAME_LENGTH, true)
+ ),
+ DictSys::Column(
+ 3,
+ "TABLE_NAME",
+ false,
+ SqlType(SqlType::Varchar, MAX_TAB_NAME_SIZE, false)
+ ),
+ DictSys::Column(
+ 4,
+ "TABLE_TYPE",
+ false,
+ SqlType(SqlType::Varchar, 20, false)
+ ),
+ DictSys::Column(
+ 5,
+ "REMARKS",
+ false,
+ SqlType(SqlType::Varchar, MAX_REMARKS_LENGTH, true)
+ )
+};
+
+static DictSys::Table
+table_ODBC_TABLES(
+ DictSys::OdbcTables,
+ "ODBC$TABLES",
+ column_ODBC_TABLES,
+ arraySize(column_ODBC_TABLES)
+);
+
+// columns
+
+static DictSys::Column
+column_ODBC_COLUMNS[] = {
+ // perl docs/systables.pl columns -c
+ DictSys::Column(
+ 1,
+ "TABLE_CAT",
+ false,
+ SqlType(SqlType::Varchar, MAX_SCHEMA_NAME_LENGTH, true)
+ ),
+ DictSys::Column(
+ 2,
+ "TABLE_SCHEM",
+ false,
+ SqlType(SqlType::Varchar, MAX_SCHEMA_NAME_LENGTH, true)
+ ),
+ DictSys::Column(
+ 3,
+ "TABLE_NAME",
+ false,
+ SqlType(SqlType::Varchar, MAX_TAB_NAME_SIZE, false)
+ ),
+ DictSys::Column(
+ 4,
+ "COLUMN_NAME",
+ false,
+ SqlType(SqlType::Varchar, MAX_ATTR_NAME_SIZE, false)
+ ),
+ DictSys::Column(
+ 5,
+ "DATA_TYPE",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 6,
+ "TYPE_NAME",
+ false,
+ SqlType(SqlType::Varchar, 20, false)
+ ),
+ DictSys::Column(
+ 7,
+ "COLUMN_SIZE",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 8,
+ "BUFFER_LENGTH",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 9,
+ "DECIMAL_DIGITS",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 10,
+ "NUM_PREC_RADIX",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 11,
+ "NULLABLE",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 12,
+ "REMARKS",
+ false,
+ SqlType(SqlType::Varchar, MAX_REMARKS_LENGTH, true)
+ ),
+ DictSys::Column(
+ 13,
+ "COLUMN_DEF",
+ false,
+ SqlType(SqlType::Varchar, MAX_ATTR_DEFAULT_VALUE_SIZE, true)
+ ),
+ DictSys::Column(
+ 14,
+ "SQL_DATA_TYPE",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 15,
+ "SQL_DATETIME_SUB",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 16,
+ "CHAR_OCTET_LENGTH",
+ false,
+ SqlType(SqlType::Integer, true)
+ ),
+ DictSys::Column(
+ 17,
+ "ORDINAL_POSITION",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 18,
+ "IS_NULLABLE",
+ false,
+ SqlType(SqlType::Varchar, 3, true)
+ )
+};
+
+static DictSys::Table
+table_ODBC_COLUMNS(
+ DictSys::OdbcColumns,
+ "ODBC$COLUMNS",
+ column_ODBC_COLUMNS,
+ arraySize(column_ODBC_COLUMNS)
+);
+
+// primarykeys
+
+static DictSys::Column
+column_ODBC_PRIMARYKEYS[] = {
+ DictSys::Column(
+ 1,
+ "TABLE_CAT",
+ false,
+ SqlType(SqlType::Varchar, MAX_SCHEMA_NAME_LENGTH, true)
+ ),
+ DictSys::Column(
+ 2,
+ "TABLE_SCHEM",
+ false,
+ SqlType(SqlType::Varchar, MAX_SCHEMA_NAME_LENGTH, true)
+ ),
+ DictSys::Column(
+ 3,
+ "TABLE_NAME",
+ false,
+ SqlType(SqlType::Varchar, MAX_TAB_NAME_SIZE, false)
+ ),
+ DictSys::Column(
+ 4,
+ "COLUMN_NAME",
+ false,
+ SqlType(SqlType::Varchar, MAX_ATTR_NAME_SIZE, false)
+ ),
+ DictSys::Column(
+ 5,
+ "KEY_SEQ",
+ false,
+ SqlType(SqlType::Integer, false)
+ ),
+ DictSys::Column(
+ 6,
+ "PK_NAME",
+ false,
+ SqlType(SqlType::Varchar, MAX_ATTR_NAME_SIZE, true)
+ )
+};
+
+static DictSys::Table
+table_ODBC_PRIMARYKEYS(
+ DictSys::OdbcPrimarykeys,
+ "ODBC$PRIMARYKEYS",
+ column_ODBC_PRIMARYKEYS,
+ arraySize(column_ODBC_PRIMARYKEYS)
+);
+
+static DictSys::Column
+column_DUAL[] = {
+ DictSys::Column(
+ 1,
+ "DUMMY",
+ false,
+ SqlType(SqlType::Varchar, 1, false)
+ )
+};
+
+static DictSys::Table
+table_DUAL(
+ DictSys::Dual,
+ "DUAL",
+ column_DUAL,
+ arraySize(column_DUAL)
+);
+
+// all tables
+
+static const DictSys::Table*
+tableList[] = {
+ &table_ODBC_TYPEINFO,
+ &table_ODBC_TABLES,
+ &table_ODBC_COLUMNS,
+ &table_ODBC_PRIMARYKEYS,
+ &table_DUAL
+};
+
+static const unsigned tableCount = arraySize(tableList);
+
+DictTable*
+DictSys::loadTable(Ctx& ctx, DictSchema* schema, const BaseString& name)
+{
+ const Table* tp = 0;
+ for (unsigned i = 0; i < tableCount; i++) {
+ if (strcmp(tableList[i]->m_name, name.c_str()) == 0) {
+ tp = tableList[i];
+ break;
+ }
+ }
+ if (tp == 0)
+ return 0;
+ DictTable* table = new DictTable(schema->m_connArea, tp->m_name, tp->m_columnCount);
+ table->sysId(tp->m_id);
+ schema->addTable(table);
+ for (unsigned position = 1; position <= tp->m_columnCount; position++) {
+ const Column* cp = &tp->m_columnList[position - 1];
+ ctx_assert(cp->m_position == position);
+ const SqlType& sqlType = cp->m_sqlType;
+ DictColumn* column = new DictColumn(table->m_connArea, cp->m_name, sqlType);
+ table->setColumn(position, column);
+ column->m_key = cp->m_key;
+ if (column->m_key)
+ table->m_keys.push_back(column);
+ }
+ ctx_log3(("%s: system table defined", name.c_str()));
+ return table;
+}
diff --git a/ndb/src/client/odbc/dictionary/DictSys.hpp b/ndb/src/client/odbc/dictionary/DictSys.hpp
new file mode 100644
index 00000000000..e6fa661fd59
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictSys.hpp
@@ -0,0 +1,77 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_DICTIONARY_DictSys_hpp
+#define ODBC_DICTIONARY_DictSys_hpp
+
+#include <common/common.hpp>
+#include <common/DataType.hpp>
+
+class Ctx;
+class DictSchema;
+class DictTable;
+class SqlType;
+
+/**
+ * @class DictSys
+ * @brief Built-in tables (replaced later by real systables)
+ */
+class DictSys {
+public:
+ enum Id {
+ Undef = 0,
+ OdbcTypeinfo = 1,
+ OdbcTables = 2,
+ OdbcColumns = 3,
+ OdbcPrimarykeys = 4,
+ Dual = 5
+ };
+ struct Column {
+ Column(unsigned position, const char* name, bool key, const SqlType& sqlType);
+ const unsigned m_position;
+ const char* const m_name;
+ const bool m_key;
+ const SqlType m_sqlType;
+ };
+ struct Table {
+ Table(Id id, const char* name, const Column* columnList, unsigned columnCount);
+ const Id m_id;
+ const char* m_name;
+ const Column* const m_columnList;
+ const unsigned m_columnCount;
+ };
+ static DictTable* loadTable(Ctx& ctx, DictSchema* schema, const BaseString& name);
+};
+
+inline
+DictSys::Column::Column(unsigned position, const char* name, bool key, const SqlType& sqlType) :
+ m_position(position),
+ m_name(name),
+ m_key(key),
+ m_sqlType(sqlType)
+{
+}
+
+inline
+DictSys::Table::Table(DictSys::Id id, const char* name, const Column* columnList, unsigned columnCount) :
+ m_id(id),
+ m_name(name),
+ m_columnList(columnList),
+ m_columnCount(columnCount)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/dictionary/DictTable.cpp b/ndb/src/client/odbc/dictionary/DictTable.cpp
new file mode 100644
index 00000000000..4db7d3b3aec
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictTable.cpp
@@ -0,0 +1,91 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/common.hpp>
+#include <common/Ctx.hpp>
+#include <common/ConnArea.hpp>
+#include "DictSchema.hpp"
+#include "DictTable.hpp"
+#include "DictColumn.hpp"
+#include "DictColumn.hpp"
+
+DictTable::~DictTable()
+{
+ for (Columns::iterator i = m_columns.begin(); i != m_columns.end(); i++) {
+ delete *i;
+ *i = 0;
+ }
+ for (Indexes::iterator i = m_indexes.begin(); i != m_indexes.end(); i++) {
+ delete *i;
+ *i = 0;
+ }
+}
+
+DictColumn*
+DictTable::findColumn(const BaseString& name) const
+{
+ for (unsigned i = 1; i <= getSize(); i++) {
+ DictColumn* column = m_columns[i];
+ ctx_assert(column != 0);
+ if (strcmp(column->getName().c_str(), name.c_str()) == 0)
+ return column;
+ }
+ return 0;
+}
+
+DictColumn*
+DictTable::loadColumn(Ctx& ctx, unsigned position)
+{
+ Ndb* ndb = m_connArea.ndbObject();
+ NdbDictionary::Dictionary* ndbDictionary = ndb->getDictionary();
+ if (ndbDictionary == 0) {
+ ctx.pushStatus(ndb, "getDictionary");
+ return 0;
+ }
+ const NdbDictionary::Table* ndbTable = ndbDictionary->getTable(m_name.c_str());
+ ctx_assert(ndbTable != 0);
+ ctx_assert(position != 0);
+ NdbAttrId attrId = position - 1;
+ const NdbDictionary::Column* ndbColumn = ndbTable->getColumn(attrId);
+ ctx_assert(ndbColumn != 0);
+ SqlType sqlType(ctx, ndbColumn);
+ if (! ctx.ok())
+ return 0;
+ DictColumn* column = new DictColumn(m_connArea, ndbColumn->getName(), sqlType);
+ setColumn(position, column);
+ column->m_key = column->m_tupleId = false;
+ if (ndbColumn->getPrimaryKey())
+ column->m_key = true;
+ if (ndbColumn->getTupleKey())
+ column->m_key = column->m_tupleId = true;
+ if (column->m_key)
+ m_keys.push_back(column);
+ // props
+ const char* value;
+ column->m_autoIncrement = false;
+ if (ndbColumn->getAutoIncrement())
+ column->m_autoIncrement = true;
+ column->m_defaultValue = 0;
+ if ((value = ndbColumn->getDefaultValue()) != 0 && strlen(value) != 0)
+ column->m_defaultValue = strcpy(new char[strlen(value) + 1], value);
+ ctx_log4(("column %u %s keyFlag=%d idFlag=%d", position, ndbColumn->getName(), column->m_key, column->m_tupleId));
+ if (column->m_tupleId)
+ m_tupleId = position;
+ if (column->m_autoIncrement)
+ m_autoIncrement = position;
+ return column;
+}
diff --git a/ndb/src/client/odbc/dictionary/DictTable.hpp b/ndb/src/client/odbc/dictionary/DictTable.hpp
new file mode 100644
index 00000000000..5cecfff9562
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/DictTable.hpp
@@ -0,0 +1,192 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_DICTIONARY_DictTable_hpp
+#define ODBC_DICTIONARY_DictTable_hpp
+
+#include <vector>
+#include <list>
+#include <common/common.hpp>
+#include "DictColumn.hpp"
+#include "DictIndex.hpp"
+#include "DictSys.hpp"
+
+class Ctx;
+class ConnArea;
+class DictSchema;
+class DictColumn;
+class DictIndex;
+
+/**
+ * @class DictTable
+ * @brief Database table
+ */
+class DictTable {
+ friend class DictSchema;
+public:
+ DictTable(const ConnArea& connArea, const BaseString& name, unsigned size);
+ ~DictTable();
+ unsigned getSize() const;
+ void setParent(DictSchema* parent);
+ DictSchema* getParent() const;
+ void setColumn(unsigned i, DictColumn* column);
+ DictColumn* getColumn(unsigned i) const;
+ const BaseString& getName() const;
+ DictColumn* findColumn(const BaseString& name) const;
+ DictColumn* loadColumn(Ctx& ctx, unsigned position);
+ unsigned keyCount() const;
+ DictColumn* getKey(unsigned i) const;
+ unsigned tupleId() const;
+ unsigned autoIncrement() const;
+ void sysId(DictSys::Id id);
+ DictSys::Id sysId() const;
+ // indexes
+ void addIndex(DictIndex* index);
+ unsigned indexCount() const;
+ const DictIndex* getIndex(unsigned i) const; // indexed from 1
+protected:
+ friend class DictSys;
+ const ConnArea& m_connArea;
+ const BaseString m_name;
+ unsigned m_size;
+ DictSchema* m_parent;
+ typedef std::vector<DictColumn*> Columns;
+ Columns m_columns;
+ Columns m_keys;
+ unsigned m_tupleId; // tuple id column
+ unsigned m_autoIncrement; // autoincrement key
+ DictSys::Id m_sysId; // built-in system table id (if non-zero)
+ typedef std::vector<DictIndex*> Indexes;
+ Indexes m_indexes;
+};
+
+inline
+DictTable::DictTable(const ConnArea& connArea, const BaseString& name, unsigned size) :
+ m_connArea(connArea),
+ m_name(name),
+ m_size(size),
+ m_parent(0),
+ m_columns(1 + size),
+ m_keys(1), // indexed from 1
+ m_tupleId(0),
+ m_autoIncrement(0),
+ m_sysId(DictSys::Undef),
+ m_indexes(1)
+{
+}
+
+inline unsigned
+DictTable::getSize() const
+{
+ ctx_assert(m_columns.size() == 1 + m_size);
+ return m_size;
+}
+
+inline void
+DictTable::setParent(DictSchema* parent)
+{
+ m_parent = parent;
+}
+
+inline DictSchema*
+DictTable::getParent() const
+{
+ return m_parent;
+}
+
+inline void
+DictTable::setColumn(unsigned i, DictColumn* column)
+{
+ ctx_assert(1 <= i && i <= m_size);
+ m_columns[i] = column;
+ column->setPosition(i);
+ column->setParent(this);
+}
+
+inline DictColumn*
+DictTable::getColumn(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= m_size);
+ ctx_assert(m_columns[i] != 0);
+ return m_columns[i];
+}
+
+inline const BaseString&
+DictTable::getName() const
+{
+ return m_name;
+}
+
+inline unsigned
+DictTable::keyCount() const
+{
+ ctx_assert(m_keys.size() >= 1);
+ return m_keys.size() - 1;
+}
+
+inline DictColumn*
+DictTable::getKey(unsigned i) const
+{
+ ctx_assert(1 <= i && i <= m_keys.size() && m_keys[i] != 0);
+ return m_keys[i];
+}
+
+inline unsigned
+DictTable::tupleId() const
+{
+ return m_tupleId;
+}
+
+inline unsigned
+DictTable::autoIncrement() const
+{
+ return m_autoIncrement;
+}
+
+inline void
+DictTable::sysId(DictSys::Id id)
+{
+ m_sysId = id;
+}
+
+inline DictSys::Id
+DictTable::sysId() const
+{
+ return m_sysId;
+}
+
+inline void
+DictTable::addIndex(DictIndex* index)
+{
+ m_indexes.push_back(index);
+ index->setTable(this);
+}
+
+inline unsigned
+DictTable::indexCount() const
+{
+ ctx_assert(m_indexes.size() >= 1);
+ return m_indexes.size() - 1;
+}
+
+inline const DictIndex*
+DictTable::getIndex(unsigned i) const
+{
+ ctx_assert(1 <= i && i < m_indexes.size() && m_indexes[i] != 0);
+ return m_indexes[i];
+}
+
+#endif
diff --git a/ndb/src/client/odbc/dictionary/Makefile b/ndb/src/client/odbc/dictionary/Makefile
new file mode 100644
index 00000000000..cdfd3b6ea0c
--- /dev/null
+++ b/ndb/src/client/odbc/dictionary/Makefile
@@ -0,0 +1,20 @@
+include .defs.mk
+
+TYPE = *
+
+NONPIC_ARCHIVE = N
+
+PIC_ARCHIVE = Y
+
+ARCHIVE_TARGET = odbcdictionary
+
+SOURCES = \
+ DictCatalog.cpp \
+ DictSchema.cpp \
+ DictTable.cpp \
+ DictColumn.cpp \
+ DictIndex.cpp \
+ DictSys.cpp
+
+include ../Extra.mk
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/client/odbc/docs/class.fig b/ndb/src/client/odbc/docs/class.fig
new file mode 100644
index 00000000000..38c24c1fba4
--- /dev/null
+++ b/ndb/src/client/odbc/docs/class.fig
@@ -0,0 +1,332 @@
+#FIG 3.2
+Landscape
+Flush left
+Inches
+A4
+100.00
+Single
+-2
+1200 2
+6 600 6600 1500 9600
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 600 6600 1500 6600 1500 7200 600 7200 600 6600
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 600 7800 1500 7800 1500 8400 600 8400 600 7800
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 600 9000 1500 9000 1500 9600 600 9600 600 9000
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 2
+ 1 1 1.00 60.00 120.00
+ 900 9000 900 8400
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 2
+ 1 1 1.00 60.00 120.00
+ 900 7800 900 7200
+4 0 0 50 0 12 12 0.0000 4 180 420 750 6825 Diag\001
+4 0 0 50 0 12 12 0.0000 4 180 420 750 8025 Diag\001
+4 0 0 50 0 12 12 0.0000 4 135 630 750 8325 Record\001
+4 0 0 50 0 12 12 0.0000 4 180 420 750 9225 Diag\001
+4 0 0 50 0 12 12 0.0000 4 135 525 750 9525 Field\001
+4 0 0 50 0 12 12 0.0000 4 120 420 750 7125 Area\001
+-6
+6 2700 6600 3600 9600
+6 2700 6600 3600 9600
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 2700 6600 3600 6600 3600 7200 2700 7200 2700 6600
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 2700 9000 3600 9000 3600 9600 2700 9600 2700 9000
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 2
+ 1 1 1.00 60.00 120.00
+ 3000 9000 3000 7200
+4 0 0 50 0 12 12 0.0000 4 120 420 2850 6825 Attr\001
+4 0 0 50 0 12 12 0.0000 4 120 420 2850 9225 Attr\001
+4 0 0 50 0 12 12 0.0000 4 135 525 2850 9525 Field\001
+4 0 0 50 0 12 12 0.0000 4 120 420 2850 7125 Area\001
+-6
+-6
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 3300 3900 4200 3900 4200 4500 3300 4500 3300 3900
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 3300 2700 4200 2700 4200 3300 3300 3300 3300 2700
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 3300 1500 4200 1500 4200 2100 3300 2100 3300 1500
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 3300 300 4200 300 4200 900 3300 900 3300 300
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 1800 2700 2700 2700 2700 3300 1800 3300 1800 2700
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 0 1 2
+ 1 1 1.00 60.00 120.00
+ 3300 1800 3000 1800
+2 1 0 1 0 7 50 0 -1 0.000 0 0 7 0 1 2
+ 1 1 1.00 60.00 120.00
+ 3300 3000 3000 3000
+2 1 0 1 0 7 50 0 -1 0.000 0 0 7 0 1 2
+ 1 1 1.00 60.00 120.00
+ 3300 4200 3000 4200
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 1 2
+ 1 1 1.00 60.00 120.00
+ 1 1 1.00 60.00 120.00
+ 4200 4200 4800 4200
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 2
+ 1 1 1.00 60.00 120.00
+ 3600 3900 3600 3300
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 2
+ 1 1 1.00 60.00 120.00
+ 3600 2700 3600 2100
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 2
+ 1 1 1.00 60.00 120.00
+ 3600 1500 3600 900
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 4800 5100 5700 5100 5700 5700 4800 5700 4800 5100
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 4800 2700 5700 2700 5700 3300 4800 3300 4800 2700
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 4800 3900 5700 3900 5700 4500 4800 4500 4800 3900
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 1 2
+ 1 1 1.00 60.00 120.00
+ 1 1 1.00 60.00 120.00
+ 5100 6600 5100 5700
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 6
+ 1 1 1.00 60.00 120.00
+ 5100 5100 5100 4800 4500 4800 4500 3600 3900 3600 3900 3300
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 3
+ 1 1 1.00 60.00 120.00
+ 3600 4500 3600 5400 4800 5400
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 4800 6600 5700 6600 5700 7200 4800 7200 4800 6600
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 4800 7800 5700 7800 5700 8400 4800 8400 4800 7800
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 4800 9000 5700 9000 5700 9600 4800 9600 4800 9000
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 2
+ 1 1 1.00 60.00 120.00
+ 5100 9000 5100 8400
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 2
+ 1 1 1.00 60.00 120.00
+ 5100 7800 5100 7200
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 1 2
+ 1 1 1.00 60.00 120.00
+ 1 1 1.00 60.00 120.00
+ 4200 3000 4800 3000
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 6900 2700 7800 2700 7800 3300 6900 3300 6900 2700
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 6900 1500 7800 1500 7800 2100 6900 2100 6900 1500
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 3
+ 0 0 1.00 60.00 120.00
+ 7200 3300 7200 4050 5700 4050
+2 1 0 1 0 7 50 0 -1 0.000 0 0 -1 1 0 3
+ 0 0 1.00 60.00 120.00
+ 11700 3300 11700 4350 5700 4350
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 5100 3900 5100 3300
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 8400 300 9300 300 9300 900 8400 900 8400 300
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 5100 2700 5100 900
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 2
+ 3000 6600 3000 1800
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 6900 5100 7800 5100 7800 6000 6900 6000 6900 5100
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 8400 5100 9300 5100 9300 6000 8400 6000 8400 5100
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 9900 5100 10800 5100 10800 6000 9900 6000 9900 5100
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 11400 5100 12300 5100 12300 6000 11400 6000 11400 5100
+2 1 1 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 7800 5550 8400 5550
+2 1 1 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 9300 5550 9900 5550
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 7500 5100 7500 3300
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 10500 5100 10500 3300
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 12000 5100 12000 3300
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 9900 2700 10800 2700 10800 3300 9900 3300 9900 2700
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 11400 2700 12300 2700 12300 3300 11400 3300 11400 2700
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 11400 1500 12300 1500 12300 2100 11400 2100 11400 1500
+2 2 0 1 0 7 50 0 -1 0.000 0 0 -1 0 0 5
+ 9900 1500 10800 1500 10800 2100 9900 2100 9900 1500
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 6900 6600 7800 6600 7800 7200 6900 7200 6900 6600
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 6900 7800 7800 7800 7800 8400 6900 8400 6900 7800
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 8400 6600 9300 6600 9300 7200 8400 7200 8400 6600
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 8400 7800 9300 7800 9300 8400 8400 8400 8400 7800
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 4
+ 0 0 1.00 60.00 120.00
+ 5700 3000 6300 3000 6300 6900 6900 6900
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 4
+ 0 0 1.00 60.00 120.00
+ 5700 4200 6000 4200 6000 8100 6900 8100
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 8400 6900 7800 6900
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 8400 8100 7800 8100
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 9900 6900 9300 6900
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 9900 8100 9300 8100
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 11400 6900 10800 6900
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 11400 8100 10800 8100
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 11400 6600 12300 6600 12300 7200 11400 7200 11400 6600
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 11400 7800 12300 7800 12300 8400 11400 8400 11400 7800
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 9900 6600 10800 6600 10800 7200 9900 7200 9900 6600
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 9900 7800 10800 7800 10800 8400 9900 8400 9900 7800
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 8400 2700 9300 2700 9300 3300 8400 3300 8400 2700
+2 2 0 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 8400 1500 9300 1500 9300 2100 8400 2100 8400 1500
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 9000 5100 9000 3300
+2 1 1 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 7800 3000 8400 3000
+2 1 1 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 9300 3000 9900 3000
+2 1 1 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 7800 1800 8400 1800
+2 1 1 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 9300 1800 9900 1800
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 11400 3000 10800 3000
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 11400 1800 10800 1800
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 11400 5550 10800 5550
+2 4 0 2 0 7 50 0 -1 6.000 0 0 7 0 0 5
+ 5700 900 5700 300 4800 300 4800 900 5700 900
+2 2 1 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 12900 6600 13800 6600 13800 7200 12900 7200 12900 6600
+2 2 1 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 12900 5100 13800 5100 13800 5700 12900 5700 12900 5100
+2 1 1 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 13200 7800 13200 7200
+2 1 1 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 13200 6600 13200 5700
+2 2 1 1 0 7 50 0 -1 4.000 0 0 -1 0 0 5
+ 12900 7800 13800 7800 13800 8400 12900 8400 12900 7800
+2 1 1 1 0 7 50 0 -1 4.000 0 0 -1 1 0 3
+ 0 0 1.00 60.00 120.00
+ 13200 5100 13200 1800 12300 1800
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 1 4
+ 0 0 1.00 60.00 120.00
+ 0 0 1.00 60.00 120.00
+ 5700 7050 6600 7050 6600 7875 6900 7875
+4 0 0 50 0 12 12 0.0000 4 135 630 3375 525 Handle\001
+4 0 0 50 0 12 12 0.0000 4 135 630 3375 1725 Handle\001
+4 0 0 50 0 12 12 0.0000 4 135 630 3375 2925 Handle\001
+4 0 0 50 0 12 12 0.0000 4 135 630 3375 4125 Handle\001
+4 0 0 50 0 12 12 0.0000 4 120 420 3450 825 Root\001
+4 0 0 50 0 12 12 0.0000 4 120 315 3450 2025 Env\001
+4 0 0 50 0 12 12 0.0000 4 135 315 3450 3225 Dbc\001
+4 0 0 50 0 12 12 0.0000 4 120 420 3450 4425 Stmt\001
+4 0 0 50 0 12 12 0.0000 4 135 630 1875 2925 Handle\001
+4 0 0 50 0 12 12 0.0000 4 120 420 1950 3225 Base\001
+4 0 0 50 0 12 12 0.0000 4 135 630 4875 5325 Handle\001
+4 0 0 50 0 12 12 0.0000 4 120 420 4950 5625 Desc\001
+4 0 0 50 0 12 12 0.0000 4 120 420 4875 3225 Area\001
+4 0 0 50 0 12 12 0.0000 4 120 420 4875 2925 Conn\001
+4 0 0 50 0 12 12 0.0000 4 120 420 4875 4425 Area\001
+4 0 0 50 0 12 12 0.0000 4 120 420 4875 4125 Stmt\001
+4 0 0 50 0 12 12 0.0000 4 120 420 4950 6825 Desc\001
+4 0 0 50 0 12 12 0.0000 4 120 420 4950 8025 Desc\001
+4 0 0 50 0 12 12 0.0000 4 135 630 4950 8325 Record\001
+4 0 0 50 0 12 12 0.0000 4 120 420 4950 9225 Desc\001
+4 0 0 50 0 12 12 0.0000 4 120 420 4950 7125 Area\001
+4 0 0 50 0 12 12 0.0000 4 135 525 4950 9525 Field\001
+4 0 0 50 0 12 12 0.0000 4 135 735 3675 5925 ird ard\001
+4 0 0 50 0 12 12 0.0000 4 180 735 3675 5625 ipd apd\001
+4 0 0 50 0 12 12 0.0000 4 135 420 6975 2925 Plan\001
+4 0 0 50 0 12 12 0.0000 4 120 420 6975 3225 root\001
+4 0 0 50 0 12 12 0.0000 4 135 420 6975 1725 Plan\001
+4 0 0 50 0 12 12 0.0000 4 120 420 6975 2025 Tree\001
+4 0 0 50 0 12 12 0.0000 4 120 420 8475 525 Base\001
+4 0 0 50 0 12 12 0.0000 4 120 420 8475 825 Tree\001
+4 0 0 50 0 12 12 0.0000 4 120 315 5025 675 NDB\001
+4 0 0 50 0 14 14 0.0000 4 195 1755 300 525 Class Diagram\001
+4 0 0 50 0 12 12 0.0000 4 135 420 6975 5325 Plan\001
+4 0 0 50 0 12 12 0.0000 4 135 630 6975 5625 delete\001
+4 0 0 50 0 12 12 0.0000 4 135 840 6975 5925 searched\001
+4 0 0 50 0 12 12 0.0000 4 135 420 8475 5325 Plan\001
+4 0 0 50 0 12 12 0.0000 4 135 630 8475 5625 delete\001
+4 0 0 50 0 12 12 0.0000 4 135 420 8475 5925 full\001
+4 0 0 50 0 12 12 0.0000 4 135 420 9975 5325 Code\001
+4 0 0 50 0 12 12 0.0000 4 135 630 9975 5625 delete\001
+4 0 0 50 0 12 12 0.0000 4 135 420 9975 5925 full\001
+4 0 0 50 0 12 12 0.0000 4 120 315 11475 5325 Run\001
+4 0 0 50 0 12 12 0.0000 4 135 630 11475 5625 delete\001
+4 0 0 50 0 12 12 0.0000 4 135 420 11475 5925 full\001
+4 0 0 50 0 12 12 0.0000 4 120 420 9975 3225 root\001
+4 0 0 50 0 12 12 0.0000 4 120 315 11475 2925 Run\001
+4 0 0 50 0 12 12 0.0000 4 120 420 11475 3225 root\001
+4 0 0 50 0 12 12 0.0000 4 120 315 11475 1725 Run\001
+4 0 0 50 0 12 12 0.0000 4 120 420 11475 2025 Tree\001
+4 0 0 50 0 12 12 0.0000 4 135 420 9975 1725 Code\001
+4 0 0 50 0 12 12 0.0000 4 120 420 9975 2025 Tree\001
+4 0 0 50 0 12 12 0.0000 4 135 420 9975 2925 Code\001
+4 0 0 50 0 12 12 0.0000 4 120 420 6975 6825 Conn\001
+4 0 0 50 0 12 12 0.0000 4 180 735 6975 7125 Catalog\001
+4 0 0 50 0 12 12 0.0000 4 120 420 6975 8025 Stmt\001
+4 0 0 50 0 12 12 0.0000 4 180 735 6975 8325 Catalog\001
+4 0 0 50 0 12 12 0.0000 4 120 420 8475 6825 Conn\001
+4 0 0 50 0 12 12 0.0000 4 120 420 8475 8025 Stmt\001
+4 0 0 50 0 12 12 0.0000 4 135 630 8475 7125 Schema\001
+4 0 0 50 0 12 12 0.0000 4 135 630 8475 8325 Schema\001
+4 0 0 50 0 12 12 0.0000 4 120 420 11475 6825 Conn\001
+4 0 0 50 0 12 12 0.0000 4 120 420 11475 8025 Stmt\001
+4 0 0 50 0 12 12 0.0000 4 135 630 11475 7125 Column\001
+4 0 0 50 0 12 12 0.0000 4 135 525 11475 8325 Field\001
+4 0 0 50 0 12 12 0.0000 4 120 420 9975 6825 Conn\001
+4 0 0 50 0 12 12 0.0000 4 120 420 9975 8025 Stmt\001
+4 0 0 50 0 12 12 0.0000 4 135 525 9975 7125 Table\001
+4 0 0 50 0 12 12 0.0000 4 120 315 9975 8325 Row\001
+4 0 0 50 0 12 12 0.0000 4 135 420 8475 1725 Plan\001
+4 0 0 50 0 12 12 0.0000 4 120 420 8475 2025 Tree\001
+4 0 0 50 0 12 12 0.0000 4 135 420 8475 2925 Plan\001
+4 0 0 50 0 12 12 0.0000 4 120 420 8475 3225 root\001
+4 0 0 50 0 14 11 0.0000 4 180 840 675 825 (prelim)\001
+4 0 0 50 0 12 12 0.0000 4 180 525 6375 1350 input\001
+4 0 0 50 0 12 12 0.0000 4 180 840 7725 1350 optimize\001
+4 0 0 50 0 12 12 0.0000 4 165 630 9375 1350 output\001
+4 0 0 50 0 12 12 0.0000 4 120 735 10650 1350 execute\001
+4 0 0 50 0 12 12 0.0000 4 135 525 12075 1350 fetch\001
+4 0 0 50 0 12 12 0.0000 4 135 630 13050 5325 Result\001
+4 0 0 50 0 12 12 0.0000 4 120 315 13050 5625 Set\001
+4 0 0 50 0 12 12 0.0000 4 135 630 13050 6825 Result\001
+4 0 0 50 0 12 12 0.0000 4 120 315 13125 7125 Row\001
+4 0 0 50 0 12 12 0.0000 4 135 630 13050 8025 Result\001
+4 0 0 50 0 12 12 0.0000 4 135 525 13050 8325 Field\001
diff --git a/ndb/src/client/odbc/docs/descfield.pl b/ndb/src/client/odbc/docs/descfield.pl
new file mode 100644
index 00000000000..80fef22f303
--- /dev/null
+++ b/ndb/src/client/odbc/docs/descfield.pl
@@ -0,0 +1,1482 @@
+# usage perl Desc.data
+# prints template for DescSpec.cpp
+use strict;
+my $order = 0;
+
+# XXX do it later
+
+#
+# odbcsqlsetdescfield.htm
+#
+my $descSpec = {
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLSetDescField</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqlsetdescfield"></A>SQLSetDescField</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 3.0<BR>
+# Standards Compliance: ISO 92</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLSetDescField</B> sets the value of a single field of a descriptor record.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLSetDescField</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHDESC&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>DescriptorHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>RecNumber</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>FieldIdentifier</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLPOINTER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>ValuePtr</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLINTEGER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>BufferLength</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>DescriptorHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Descriptor handle.</dd>
+#
+# <DT><I>RecNumber</I></DT>
+#
+# <DD>[Input]<BR>
+# Indicates the descriptor record containing the field that the application seeks to set. Descriptor records are numbered from 0, with record number 0 being the bookmark record. The <I>RecNumber</I> argument is ignored for header fields.</dd>
+#
+# <DT><I>FieldIdentifier</I></DT>
+#
+# <DD>[Input]<BR>
+# Indicates the field of the descriptor whose value is to be set. For more information, see "<I>FieldIdentifier</I> Argument" in the "Comments" section.</dd>
+#
+# <DT><I>ValuePtr</I></DT>
+#
+# <DD>[Input]<BR>
+# Pointer to a buffer containing the descriptor information, or a 4-byte value. The data type depends on the value of <I>FieldIdentifier</I>. If <I>ValuePtr</I> is a 4-byte value, either all four of the bytes are used or just two of the four are used, depending on the value of the <I>FieldIdentifier</I> argument.</dd>
+#
+# <DT><I>BufferLength</I></DT>
+#
+# <DD>[Input]<BR>
+# If <I>FieldIdentifier</I> is an ODBC-defined field and <I>ValuePtr</I> points to a character string or a binary buffer, this argument should be the length of *<I>ValuePtr</I>. If <I>FieldIdentifier</I> is an ODBC-defined field and <I>ValuePtr</I> is an integer, <I>BufferLength</I> is ignored.
+#
+# <P>If <I>FieldIdentifier</I> is a driver-defined field, the application indicates the nature of the field to the Driver Manager by setting the <I>BufferLength</I> argument. <I>BufferLength</I> can have the following values:
+#
+#
+# <UL type=disc>
+# <LI>If <I>ValuePtr</I> is a pointer to a character string, then <I>BufferLength</I> is the length of the string or SQL_NTS.</li>
+#
+# <LI>If <I>ValuePtr</I> is a pointer to a binary buffer, then the application places the result of the SQL_LEN_BINARY_ATTR(<I>length</I>) macro in <I>BufferLength</I>. This places a negative value in <I>BufferLength</I>.</li>
+#
+# <LI>If <I>ValuePtr</I> is a pointer to a value other than a character string or a binary string, then <I>BufferLength</I> should have the value SQL_IS_POINTER. </li>
+#
+# <LI>If <I>ValuePtr</I> contains a fixed-length value, then <I>BufferLength</I> is either SQL_IS_INTEGER, SQL_IS_UINTEGER, SQL_IS_SMALLINT, or SQL_IS_USMALLINT, as appropriate.</li>
+# </UL>
+# </dd>
+# </DL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLSetDescField</B> returns SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value can be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_DESC and a <I>Handle</I> of <I>DescriptorHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLSetDescField</B> and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=16%>SQLSTATE</TH>
+# <TH width=30%>Error</TH>
+# <TH width=54%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>01000</TD>
+# <TD width=30%>General warning</TD>
+# <TD width=54%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>01S02</TD>
+# <TD width=30%>Option value changed</TD>
+# <TD width=54%>The driver did not support the value specified in <I>*ValuePtr</I> (if <I>ValuePtr</I> was a pointer) or the value in <I>ValuePtr</I> (if <I>ValuePtr </I>was a 4-byte value), or <I>*ValuePtr</I> was invalid because of implementation working conditions, so the driver substituted a similar value. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>07009</TD>
+# <TD width=30%>Invalid descriptor index</TD>
+# <TD width=54%>The <I>FieldIdentifier</I> argument was a record field, the <I>RecNumber</I> argument was 0, and the <I>DescriptorHandle</I> argument referred to an IPD handle.
+# <P>The <I>RecNumber</I> argument was less than 0, and the <I>DescriptorHandle</I> argument referred to an ARD or an APD.</P>
+#
+# <P>The <I>RecNumber</I> argument was greater than the maximum number of columns or parameters that the data source can support, and the <I>DescriptorHandle</I> argument referred to an APD or ARD.</P>
+#
+# <P>(DM) The <I>FieldIdentifier</I> argument was SQL_DESC_COUNT, and <I>*ValuePtr</I> argument was less than 0.</P>
+#
+# <P>The <I>RecNumber</I> argument was equal to 0, and the <I>DescriptorHandle</I> argument referred to an implicitly allocated APD. (This error does not occur with an explicitly allocated application descriptor, because it is not known whether an explicitly allocated application descriptor is an APD or ARD until execute time.)</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>08S01</TD>
+# <TD width=30%>Communication link failure</TD>
+# <TD width=54%>The communication link between the driver and the data source to which the driver was connected failed before the function completed processing.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>22001</TD>
+# <TD width=30%>String data, right <BR>
+# truncated</TD>
+# <TD width=54%>The <I>FieldIdentifier</I> argument was SQL_DESC_NAME, and the <I>BufferLength</I> argument was a value larger than SQL_MAX_IDENTIFIER_LEN.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY000</TD>
+# <TD width=30%>General error</TD>
+# <TD width=54%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY001</TD>
+# <TD width=30%>Memory allocation <BR>
+# error</TD>
+# <TD width=54%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY010</TD>
+# <TD width=30%>Function sequence error</TD>
+# <TD width=54%>(DM) The <I>DescriptorHandle</I> was associated with a <I>StatementHandle</I> for which an asynchronously executing function (not this one) was called and was still executing when this function was called.
+# <P>(DM) <B>SQLExecute</B>, <B>SQLExecDirect</B>, <B>SQLBulkOperations</B>, or <B>SQLSetPos</B> was called for the <I>StatementHandle</I> with which the <I>DescriptorHandle</I> was associated and returned SQL_NEED_DATA. This function was called before data was sent for all data-at-execution parameters or columns.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY013</TD>
+# <TD width=30%>Memory management error</TD>
+# <TD width=54%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY016</TD>
+# <TD width=30%>Cannot modify an implementation row descriptor</TD>
+# <TD width=54%>The <I>DescriptorHandle</I> argument was associated with an IRD, and the <I>FieldIdentifier</I> argument was not SQL_DESC_ARRAY_STATUS_PTR or SQL_DESC_ROWS_PROCESSED_PTR.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY021</TD>
+# <TD width=30%>Inconsistent descriptor information</TD>
+# <TD width=54%>The SQL_DESC_TYPE and SQL_DESC_DATETIME_INTERVAL_CODE fields do not form a valid ODBC SQL type or a valid driver-specific SQL type (for IPDs) or a valid ODBC C type (for APDs or ARDs).
+# <P>Descriptor information checked during a consistency check was not consistent. (See "Consistency Check" in <B>SQLSetDescRec</B>.)</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY090</TD>
+# <TD width=30%>Invalid string or buffer length</TD>
+# <TD width=54%>(DM) <I>*ValuePtr</I> is a character string, and <I>BufferLength</I> was less than zero but was not equal to SQL_NTS.
+# <P>(DM) The driver was an ODBC 2<I>.x</I> driver, the descriptor was an ARD, the <I>ColumnNumber</I> argument was set to 0, and the value specified for the argument <I>BufferLength</I> was not equal to 4. </P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY091</TD>
+# <TD width=30%>Invalid descriptor field identifier</TD>
+# <TD width=54%>The value specified for the <I>FieldIdentifier</I> argument was not an ODBC-defined field and was not an implementation-defined value.
+# <P>The <I>FieldIdentifier</I> argument was invalid for the <I>DescriptorHandle</I> argument.</P>
+#
+# <P>The <I>FieldIdentifier</I> argument was a read-only, ODBC-defined field.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY092</TD>
+# <TD width=30%>Invalid attribute/option identifier</TD>
+# <TD width=54%>The value in <I>*ValuePtr</I> was not valid for the <I>FieldIdentifier</I> argument.
+# <P>The <I>FieldIdentifier</I> argument was SQL_DESC_UNNAMED, and <I>ValuePtr</I> was SQL_NAMED.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HY105</TD>
+# <TD width=30%>Invalid parameter type</TD>
+# <TD width=54%>(DM) The value specified for the SQL_DESC_PARAMETER_TYPE field was invalid. (For more information, see the "<I>InputOutputType</I> Argument" section in <B>SQLBindParameter</B>.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>HYT01</TD>
+# <TD width=30%>Connection timeout expired</TD>
+# <TD width=54%>The connection timeout period expired before the data source responded to the request. The connection timeout period is set through <B>SQLSetConnectAttr</B>, SQL_ATTR_CONNECTION_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=16%>IM001</TD>
+# <TD width=30%>Driver does not support this function</TD>
+# <TD width=54%>(DM) The driver associated with the <I>DescriptorHandle</I> does not support the function.</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Comments</B></P>
+#
+# <P>An application can call <B>SQLSetDescField</B> to set any descriptor field one at a time. One call to <B>SQLSetDescField</B> sets a single field in a single descriptor. This function can be called to set any field in any descriptor type, provided the field can be set. (See the table later in this section.)</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;If a call to <B>SQLSetDescField</B> fails, the contents of the descriptor record identified by the <I>RecNumber</I> argument are undefined.</P>
+#
+# <P>Other functions can be called to set multiple descriptor fields with a single call of the function. The <B>SQLSetDescRec</B> function sets a variety of fields that affect the data type and buffer bound to a column or parameter (the SQL_DESC_TYPE, SQL_DESC_DATETIME_INTERVAL_CODE, SQL_DESC_OCTET_LENGTH, SQL_DESC_PRECISION, SQL_DESC_SCALE, SQL_DESC_DATA_PTR, SQL_DESC_OCTET_LENGTH_PTR, and SQL_DESC_INDICATOR_PTR fields). <B>SQLBindCol </B>or <B>SQLBindParameter</B> can be used to make a complete specification for the binding of a column or parameter. These functions set a specific group of descriptor fields with one function call.</P>
+#
+# <P><B>SQLSetDescField</B> can be called to change the binding buffers by adding an offset to the binding pointers (SQL_DESC_DATA_PTR, SQL_DESC_INDICATOR_PTR, or SQL_DESC_OCTET_LENGTH_PTR). This changes the binding buffers without calling <B>SQLBindCol </B>or <B>SQLBindParameter</B>, which allows an application to change SQL_DESC_DATA_PTR without changing other fields, such as SQL_DESC_DATA_TYPE.</P>
+#
+# <P>If an application calls <B>SQLSetDescField</B> to set any field other than SQL_DESC_COUNT or the deferred fields SQL_DESC_DATA_PTR, SQL_DESC_OCTET_LENGTH_PTR, or SQL_DESC_INDICATOR_PTR, the record becomes unbound.</P>
+#
+# <P>Descriptor header fields are set by calling <B>SQLSetDescField </B>with the appropriate <I>FieldIdentifier</I>. Many header fields are also statement attributes, so they can also be set by a call to <B>SQLSetStmtAttr</B>. This allows applications to set a descriptor field without first obtaining a descriptor handle. When <B>SQLSetDescField</B> is called to set a header field, the <I>RecNumber</I> argument is ignored.</P>
+#
+# <P>A <I>RecNumber</I> of 0 is used to set bookmark fields.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;The statement attribute SQL_ATTR_USE_BOOKMARKS should always be set before calling <B>SQLSetDescField</B> to set bookmark fields. While this is not mandatory, it is strongly recommended.</P>
+#
+# <H1>Sequence of Setting Descriptor Fields</H1>
+#
+# <P>When setting descriptor fields by calling <B>SQLSetDescField</B>, the application must follow a specific sequence:
+#
+# <OL type=1>
+# <LI>The application must first set the SQL_DESC_TYPE, SQL_DESC_CONCISE_TYPE, or SQL_DESC_DATETIME_INTERVAL_CODE field. </li>
+#
+# <LI>After one of these fields has been set, the application can set an attribute of a data type, and the driver sets data type attribute fields to the appropriate default values for the data type. Automatic defaulting of type attribute fields ensures that the descriptor is always ready to use once the application has specified a data type. If the application explicitly sets a data type attribute, it is overriding the default attribute.</li>
+#
+# <LI>After one of the fields listed in step 1 has been set, and data type attributes have been set, the application can set SQL_DESC_DATA_PTR. This prompts a consistency check of descriptor fields. If the application changes the data type or attributes after setting the SQL_DESC_DATA_PTR field, the driver sets SQL_DESC_DATA_PTR to a null pointer, unbinding the record. This forces the application to complete the proper steps in sequence, before the descriptor record is usable.</li>
+# </OL>
+#
+# <H1>Initialization of Descriptor Fields</H1>
+#
+# <P>When a descriptor is allocated, the fields in the descriptor can be initialized to a default value, be initialized without a default value, or be undefined for the type of descriptor. The following tables indicate the initialization of each field for each type of descriptor, with "D" indicating that the field is initialized with a default, and "ND" indicating that the field is initialized without a default. If a number is shown, the default value of the field is that number. The tables also indicate whether a field is read/write (R/W) or read-only (R). </P>
+#
+# <P>The fields of an IRD have a default value only after the statement has been prepared or executed and the IRD has been populated, not when the statement handle or descriptor has been allocated. Until the IRD has been populated, any attempt to gain access to a field of an IRD will return an error.</P>
+#
+# <P>Some descriptor fields are defined for one or more, but not all, of the descriptor types (ARDs and IRDs, and APDs and IPDs). When a field is undefined for a type of descriptor, it is not needed by any of the functions that use that descriptor.</P>
+#
+# <P>The fields that can be accessed by <B>SQLGetDescField</B> cannot necessarily be set by <B>SQLSetDescField</B>. Fields that can be set by <B>SQLSetDescField</B> are listed in the following tables.</P>
+#
+# <P>The initialization of header fields is outlined in the table that follows.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=27%>Header field name</TH>
+# <TH width=21%>Type</TH>
+# <TH width=19%>R/W</TH>
+# <TH width=33%>Default</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_ALLOC_TYPE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: R<BR>
+# APD: R<BR>
+# IRD: R<BR>
+# IPD: R </TD>
+# <TD width=33%>ARD: SQL_DESC_ALLOC_AUTO for implicit or SQL_DESC_ALLOC_USER for explicit
+# <P>APD: SQL_DESC_ALLOC_AUTO for implicit or SQL_DESC_ALLOC_USER for explicit</P>
+#
+# <P>IRD: SQL_DESC_ALLOC_AUTO</P>
+#
+# <P>IPD: SQL_DESC_ALLOC_AUTO</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_ARRAY_SIZE</TD>
+# <TD width=21%>SQLUINTEGER</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD:<SUP>[1]</SUP><BR>
+# APD:<SUP>[1]</SUP><BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_ARRAY_STATUS_PTR</TD>
+# <TD width=21%>SQLUSMALLINT*</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R/W<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: Null ptr<BR>
+# APD: Null ptr<BR>
+# IRD: Null ptr<BR>
+# IPD: Null ptr</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_BIND_OFFSET_PTR</TD>
+# <TD width=21%>SQLINTEGER*</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Null ptr<BR>
+# APD: Null ptr<BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_BIND_TYPE</TD>
+# <TD width=21%>SQLINTEGER</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: SQL_BIND_BY_COLUMN
+# <P>APD: SQL_BIND_BY_COLUMN</P>
+#
+# <P>IRD: Unused</P>
+#
+# <P>IPD: Unused</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_COUNT</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: 0<BR>
+# APD: 0<BR>
+# IRD: D<BR>
+# IPD: 0</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_ROWS_PROCESSED_PTR</TD>
+# <TD width=21%>SQLUINTEGER*</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R/W<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: Null ptr<BR>
+# IPD: Null ptr</TD>
+# </TR>
+# </table></div>
+#
+# <P class="fineprint">[1]&nbsp;&nbsp;&nbsp;These fields are defined only when the IPD is automatically populated by the driver. If not, they are undefined. If an application attempts to set these fields, SQLSTATE HY091 (Invalid descriptor field identifier) will be returned.</p>
+# <P>The initialization of record fields is as shown in the following table.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=27%>Record field name</TH>
+# <TH width=21%>Type</TH>
+# <TH width=19%>R/W</TH>
+# <TH width=33%>Default</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_AUTO_UNIQUE_VALUE</TD>
+# <TD width=21%>SQLINTEGER</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_BASE_COLUMN_NAME</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_BASE_TABLE_NAME</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_CASE_SENSITIVE</TD>
+# <TD width=21%>SQLINTEGER</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: R</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: D<SUP>[1]</SUP></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_CATALOG_NAME</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_CONCISE_TYPE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: SQL_C_<BR>
+# DEFAULT<BR>
+# APD: SQL_C_<BR>
+# DEFAULT<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_DATA_PTR</TD>
+# <TD width=21%>SQLPOINTER</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Null ptr<BR>
+# APD: Null ptr<BR>
+# IRD: Unused<BR>
+# IPD: Unused<SUP>[2]</SUP></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_DATETIME_INTERVAL_CODE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_DATETIME_INTERVAL_PRECISION</TD>
+# <TD width=21%>SQLINTEGER</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_DISPLAY_SIZE</TD>
+# <TD width=21%>SQLINTEGER</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_FIXED_PREC_SCALE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: R</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: D<SUP>[1]</SUP></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_INDICATOR_PTR</TD>
+# <TD width=21%>SQLINTEGER *</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Null ptr<BR>
+# APD: Null ptr<BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_LABEL</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_LENGTH</TD>
+# <TD width=21%>SQLUINTEGER</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_LITERAL_PREFIX</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_LITERAL_SUFFIX</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_LOCAL_TYPE_NAME</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: R</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: D<SUP>[1]</SUP></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_NAME</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_NULLABLE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: R</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_NUM_PREC_RADIX</TD>
+# <TD width=21%>SQLINTEGER</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_OCTET_LENGTH</TD>
+# <TD width=21%>SQLINTEGER</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_OCTET_LENGTH_PTR</TD>
+# <TD width=21%>SQLINTEGER *</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Null ptr<BR>
+# APD: Null ptr<BR>
+# IRD: Unused<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_PARAMETER_TYPE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: Unused<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: Unused<BR>
+# IPD: D=SQL_PARAM_INPUT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_PRECISION</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_ROWVER</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: Unused
+# <P>APD: Unused</P>
+#
+# <P>IRD: R</P>
+#
+# <P>IPD: R</P>
+# </TD>
+# <TD width=33%>ARD: Unused
+# <P>APD: Unused</P>
+#
+# <P>IRD: ND</P>
+#
+# <P>IPD: ND</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_SCALE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_SCHEMA_NAME</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_SEARCHABLE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_TABLE_NAME</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_TYPE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: R/W<BR>
+# APD: R/W<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: SQL_C_DEFAULT<BR>
+# APD: SQL_C_DEFAULT<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_TYPE_NAME</TD>
+# <TD width=21%>SQLCHAR *</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: R</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: D<SUP>[1]</SUP></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_UNNAMED</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: R/W</TD>
+# <TD width=33%>ARD: ND<BR>
+# APD: ND<BR>
+# IRD: D<BR>
+# IPD: ND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_UNSIGNED</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: R</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: D<SUP>[1]</SUP></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>SQL_DESC_UPDATABLE</TD>
+# <TD width=21%>SQLSMALLINT</TD>
+# <TD width=19%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: R<BR>
+# IPD: Unused</TD>
+# <TD width=33%>ARD: Unused<BR>
+# APD: Unused<BR>
+# IRD: D<BR>
+# IPD: Unused</TD>
+# </TR>
+# </table></div>
+#
+# <P class="fineprint">[1]&nbsp;&nbsp;&nbsp;These fields are defined only when the IPD is automatically populated by the driver. If not, they are undefined. If an application attempts to set these fields, SQLSTATE HY091 (Invalid descriptor field identifier) will be returned.</p>
+# <P class="fineprint">[2]&nbsp;&nbsp;&nbsp;The SQL_DESC_DATA_PTR field in the IPD can be set to force a consistency check. In a subsequent call to <B>SQLGetDescField</B> or <B>SQLGetDescRec</B>, the driver is not required to return the value that SQL_DESC_DATA_PTR was set to.</p>
+# <P class="label"><B><I>FieldIdentifier</I> Argument</B></P>
+#
+# <P>The <I>FieldIdentifier</I> argument indicates the descriptor field to be set. A descriptor contains the <I>descriptor header,</I> consisting of the header fields described in the next section, "Header Fields," and zero or more <I>descriptor records,</I> consisting of the record fields described in the section following the "Header Fields" section.</P>
+#
+# <H1>Header Fields</H1>
+#
+# <P>Each descriptor has a header consisting of the following fields:
+#
+# <DL>
+# <DT><B>SQL_DESC_ALLOC_TYPE [All]</B></DT>
+#
+# <DD>This read-only SQLSMALLINT header field specifies whether the descriptor was allocated automatically by the driver or explicitly by the application. The application can obtain, but not modify, this field. The field is set to SQL_DESC_ALLOC_AUTO by the driver if the descriptor was automatically allocated by the driver. It is set to SQL_DESC_ALLOC_USER by the driver if the descriptor was explicitly allocated by the application.</dd>
+#
+# <DT><B>SQL_DESC_ARRAY_SIZE [Application descriptors]</B></DT>
+#
+# <DD>In ARDs, this SQLUINTEGER header field specifies the number of rows in the rowset. This is the number of rows to be returned by a call to <B>SQLFetch</B> or <B>SQLFetchScroll</B> or to be operated on by a call to <B>SQLBulkOperations</B> or <B>SQLSetPos</B>.
+#
+# <P>In APDs, this SQLUINTEGER header field specifies the number of values for each parameter.
+#
+#
+# <P>The default value of this field is 1. If SQL_DESC_ARRAY_SIZE is greater than 1, SQL_DESC_DATA_PTR, SQL_DESC_INDICATOR_PTR, and SQL_DESC_OCTET_LENGTH_PTR of the APD or ARD point to arrays. The cardinality of each array is equal to the value of this field.
+#
+#
+# <P>This field in the ARD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_ROW_ARRAY_SIZE attribute. This field in the APD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_PARAMSET_SIZE attribute.
+# </dd>
+#
+# <DT><B>SQL_DESC_ARRAY_STATUS_PTR [All]</B></DT>
+#
+# <DD>For each descriptor type, this SQLUSMALLINT * header field points to an array of SQLUSMALLINT values. These arrays are named as follows: row status array (IRD), parameter status array (IPD), row operation array (ARD), and parameter operation array (APD).
+#
+# <P>In the IRD, this header field points to a row status array containing status values after a call to <B>SQLBulkOperations</B>, <B>SQLFetch</B>, <B>SQLFetchScroll</B>, or <B>SQLSetPos</B>. The array has as many elements as there are rows in the rowset. The application must allocate an array of SQLUSMALLINTs and set this field to point to the array. The field is set to a null pointer by default. The driver will populate the array&#0151;unless the SQL_DESC_ARRAY_STATUS_PTR field is set to a null pointer, in which case no status values are generated and the array is not populated.
+#
+#
+# <P class="indent"><b class="le">Caution</b>&nbsp;&nbsp;&nbsp;Driver behavior is undefined if the application sets the elements of the row status array pointed to by the SQL_DESC_ARRAY_STATUS_PTR field of the IRD.
+#
+#
+# <P>The array is initially populated by a call to <B>SQLBulkOperations</B>, <B>SQLFetch</B>, <B>SQLFetchScroll</B>, or <B>SQLSetPos</B>. If the call did not return SQL_SUCCESS or SQL_SUCCESS_WITH_INFO, the contents of the array pointed to by this field are undefined. The elements in the array can contain the following values:
+#
+#
+# <UL type=disc>
+# <LI>SQL_ROW_SUCCESS: The row was successfully fetched and has not changed since it was last fetched.</li>
+#
+# <LI>SQL_ROW_SUCCESS_WITH_INFO: The row was successfully fetched and has not changed since it was last fetched. However, a warning was returned about the row.</li>
+#
+# <LI>SQL_ROW_ERROR: An error occurred while fetching the row.</li>
+#
+# <LI>SQL_ROW_UPDATED: The row was successfully fetched and has been updated since it was last fetched. If the row is fetched again, its status is SQL_ROW_SUCCESS.</li>
+#
+# <LI>SQL_ROW_DELETED: The row has been deleted since it was last fetched.</li>
+#
+# <LI>SQL_ROW_ADDED: The row was inserted by <B>SQLBulkOperations</B>. If the row is fetched again, its status is SQL_ROW_SUCCESS.</li>
+#
+# <LI>SQL_ROW_NOROW: The rowset overlapped the end of the result set, and no row was returned that corresponded to this element of the row status array.</li>
+# </UL>
+#
+#
+# <P>This field in the IRD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_ROW_STATUS_PTR attribute.
+#
+#
+# <P>The SQL_DESC_ARRAY_STATUS_PTR field of the IRD is valid only after SQL_SUCCESS or SQL_SUCCESS_WITH_INFO has been returned. If the return code is not one of these, the location pointed to by SQL_DESC_ROWS_PROCESSED_PTR is undefined.
+#
+#
+# <P>In the IPD, this header field points to a parameter status array containing status information for each set of parameter values after a call to <B>SQLExecute</B> or <B>SQLExecDirect</B>. If the call to <B>SQLExecute</B> or <B>SQLExecDirect</B> did not return SQL_SUCCESS or SQL_SUCCESS_WITH_INFO, the contents of the array pointed to by this field are undefined. The application must allocate an array of SQLUSMALLINTs and set this field to point to the array. The driver will populate the array&#0151;unless the SQL_DESC_ARRAY_STATUS_PTR field is set to a null pointer, in which case no status values are generated and the array is not populated. The elements in the array can contain the following values:
+#
+#
+# <UL type=disc>
+# <LI>SQL_PARAM_SUCCESS: The SQL statement was successfully executed for this set of parameters.</li>
+#
+# <LI>SQL_PARAM_SUCCESS_WITH_INFO: The SQL statement was successfully executed for this set of parameters; however, warning information is available in the diagnostics data structure.</li>
+#
+# <LI>SQL_PARAM_ERROR: An error occurred in processing this set of parameters. Additional error information is available in the diagnostics data structure.</li>
+#
+# <LI>SQL_PARAM_UNUSED: This parameter set was unused, possibly due to the fact that some previous parameter set caused an error that aborted further processing, or because SQL_PARAM_IGNORE was set for that set of parameters in the array specified by the SQL_DESC_ARRAY_STATUS_PTR field of the APD.</li>
+#
+# <LI>SQL_PARAM_DIAG_UNAVAILABLE: Diagnostic information is not available. An example of this is when the driver treats arrays of parameters as a monolithic unit and so does not generate this level of error information.</li>
+# </UL>
+#
+#
+# <P>This field in the IPD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_PARAM_STATUS_PTR attribute.
+#
+#
+# <P>In the ARD, this header field points to a row operation array of values that can be set by the application to indicate whether this row is to be ignored for <B>SQLSetPos</B> operations. The elements in the array can contain the following values:
+#
+#
+# <UL type=disc>
+# <LI>SQL_ROW_PROCEED: The row is included in the bulk operation using <B>SQLSetPos</B>. (This setting does not guarantee that the operation will occur on the row. If the row has the status SQL_ROW_ERROR in the IRD row status array, the driver might not be able to perform the operation in the row.)</li>
+#
+# <LI>SQL_ROW_IGNORE: The row is excluded from the bulk operation using <B>SQLSetPos</B>.</li>
+# </UL>
+#
+#
+# <P>If no elements of the array are set, all rows are included in the bulk operation. If the value in the SQL_DESC_ARRAY_STATUS_PTR field of the ARD is a null pointer, all rows are included in the bulk operation; the interpretation is the same as if the pointer pointed to a valid array and all elements of the array were SQL_ROW_PROCEED. If an element in the array is set to SQL_ROW_IGNORE, the value in the row status array for the ignored row is not changed.
+#
+#
+# <P>This field in the ARD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_ROW_OPERATION_PTR attribute.
+#
+#
+# <P>In the APD, this header field points to a parameter operation array of values that can be set by the application to indicate whether this set of parameters is to be ignored when <B>SQLExecute</B> or<B> SQLExecDirect</B> is called. The elements in the array can contain the following values:
+#
+#
+# <UL type=disc>
+# <LI>SQL_PARAM_PROCEED: The set of parameters is included in the <B>SQLExecute</B> or <B>SQLExecDirect</B> call.</li>
+#
+# <LI>SQL_PARAM_IGNORE: The set of parameters is excluded from the <B>SQLExecute</B> or <B>SQLExecDirect</B> call.</li>
+# </UL>
+#
+#
+# <P>If no elements of the array are set, all sets of parameters in the array are used in the <B>SQLExecute</B> or <B>SQLExecDirect</B> calls. If the value in the SQL_DESC_ARRAY_STATUS_PTR field of the APD is a null pointer, all sets of parameters are used; the interpretation is the same as if the pointer pointed to a valid array and all elements of the array were SQL_PARAM_PROCEED.
+#
+#
+# <P>This field in the APD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_PARAM_OPERATION_PTR attribute.
+# </dd>
+#
+# <DT><B>SQL_DESC_BIND_OFFSET_PTR [Application descriptors]</B></DT>
+#
+# <DD>This SQLINTEGER * header field points to the binding offset. It is set to a null pointer by default. If this field is not a null pointer, the driver dereferences the pointer and adds the dereferenced value to each of the deferred fields that has a non-null value in the descriptor record (SQL_DESC_DATA_PTR, SQL_DESC_INDICATOR_PTR, and SQL_DESC_OCTET_LENGTH_PTR) at fetch time and uses the new pointer values when binding.
+#
+# <P>The binding offset is always added directly to the values in the SQL_DESC_DATA_PTR, SQL_DESC_INDICATOR_PTR, and SQL_DESC_OCTET_LENGTH_PTR fields. If the offset is changed to a different value, the new value is still added directly to the value in each descriptor field. The new offset is not added to the field value plus any earlier offset.
+#
+#
+# <P>This field is a <I>deferred field</I>: It is not used at the time it is set but is used at a later time by the driver when it needs to determine addresses for data buffers.
+#
+#
+# <P>This field in the ARD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_ROW_BIND_OFFSET_PTR attribute. This field in the ARD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_PARAM_BIND_OFFSET_PTR attribute.
+#
+#
+# <P>For more information, see the description of row-wise binding in <A HREF="odbcsqlfetchscroll.htm">SQLFetchScroll</A> and <A HREF="odbcsqlbindparameter.htm">SQLBindParameter</A>.
+# </dd>
+#
+# <DT><B>SQL_DESC_BIND_TYPE [Application descriptors]</B></DT>
+#
+# <DD>This SQLUINTEGER header field sets the binding orientation to be used for binding either columns or parameters.
+#
+# <P>In ARDs, this field specifies the binding orientation when <B>SQLFetchScroll</B> or <B>SQLFetch</B> is called on the associated statement handle.
+#
+#
+# <P>To select column-wise binding for columns, this field is set to SQL_BIND_BY_COLUMN (the default).
+#
+#
+# <P>This field in the ARD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_ROW_BIND_TYPE <I>Attribute</I>.
+#
+#
+# <P>In APDs, this field specifies the binding orientation to be used for dynamic parameters.
+#
+#
+# <P>To select column-wise binding for parameters, this field is set to SQL_BIND_BY_COLUMN (the default).
+#
+#
+# <P>This field in the APD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_PARAM_BIND_TYPE <I>Attribute</I>.
+# </dd>
+#
+# <DT><B>SQL_DESC_COUNT [All]</B></DT>
+#
+# <DD>This SQLSMALLINT header field specifies the 1-based index of the highest-numbered record that contains data. When the driver sets the data structure for the descriptor, it must also set the SQL_DESC_COUNT field to show how many records are significant. When an application allocates an instance of this data structure, it does not have to specify how many records to reserve room for. As the application specifies the contents of the records, the driver takes any required action to ensure that the descriptor handle refers to a data structure of the adequate size.
+#
+# <P>SQL_DESC_COUNT is not a count of all data columns that are bound (if the field is in an ARD) or of all parameters that are bound (if the field is in an APD), but the number of the highest-numbered record. If the highest-numbered column or parameter is unbound, then SQL_DESC_COUNT is changed to the number of the next highest-numbered column or parameter. If a column or a parameter with a number that is less than the number of the highest-numbered column is unbound (by calling <B>SQLBindCol</B> with the <I>TargetValuePtr</I> argument set to a null pointer, or <B>SQLBindParameter</B> with the <I>ParameterValuePtr</I> argument set to a null pointer), SQL_DESC_COUNT is not changed. If additional columns or parameters are bound with numbers greater than the highest-numbered record that contains data, the driver automatically increases the value in the SQL_DESC_COUNT field. If all columns are unbound by calling <B>SQLFreeStmt</B> with the SQL_UNBIND option, the SQL_DESC_COUNT fields in the ARD and IRD are set to 0. If <B>SQLFreeStmt</B> is called with the SQL_RESET_PARAMS option, the SQL_DESC_COUNT fields in the APD and IPD are set to 0.
+#
+#
+# <P>The value in SQL_DESC_COUNT can be set explicitly by an application by calling <B>SQLSetDescField</B>. If the value in SQL_DESC_COUNT is explicitly decreased, all records with numbers greater than the new value in SQL_DESC_COUNT are effectively removed. If the value in SQL_DESC_COUNT is explicitly set to 0 and the field is in an ARD, all data buffers except a bound bookmark column are released.
+#
+#
+# <P>The record count in this field of an ARD does not include a bound bookmark column. The only way to unbind a bookmark column is to set the SQL_DESC_DATA_PTR field to a null pointer.
+# </dd>
+#
+# <DT><B>SQL_DESC_ROWS_PROCESSED_PTR [Implementation descriptors]</B></DT>
+#
+# <DD>In an IRD, this SQLUINTEGER * header field points to a buffer containing the number of rows fetched after a call to <B>SQLFetch</B> or <B>SQLFetchScroll</B>, or the number of rows affected in a bulk operation performed by a call to <B>SQLBulkOperations</B> or <B>SQLSetPos</B>, including error rows.
+#
+# <P>In an IPD, this SQLUINTEGER * header field points to a buffer containing the number of sets of parameters that have been processed, including error sets. No number will be returned if this is a null pointer.
+#
+#
+# <P>SQL_DESC_ROWS_PROCESSED_PTR is valid only after SQL_SUCCESS or SQL_SUCCESS_WITH_INFO has been returned after a call to <B>SQLFetch</B> or <B>SQLFetchScroll </B>(for an IRD field) or <B>SQLExecute</B>, <B>SQLExecDirect</B>, or <B>SQLParamData</B> (for an IPD field). If the call that fills in the buffer pointed to by this field does not return SQL_SUCCESS or SQL_SUCCESS_WITH_INFO, the contents of the buffer are undefined, unless it returns SQL_NO_DATA, in which case the value in the buffer is set to 0.
+#
+#
+# <P>This field in the ARD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_ROWS_FETCHED_PTR attribute. This field in the APD can also be set by calling <B>SQLSetStmtAttr</B> with the SQL_ATTR_PARAMS_PROCESSED_PTR attribute.
+#
+#
+# <P>The buffer pointed to by this field is allocated by the application. It is a deferred output buffer that is set by the driver. It is set to a null pointer by default.
+# </dd>
+# </DL>
+#
+# <H1>Record Fields</H1>
+#
+# <P>Each descriptor contains one or more records consisting of fields that define either column data or dynamic parameters, depending on the type of descriptor. Each record is a complete definition of a single column or parameter.
+#
+# <DL>
+# <DT><B>SQL_DESC_AUTO_UNIQUE_VALUE [IRDs]</B></DT>
+#
+# <DD>This read-only SQLINTEGER record field contains SQL_TRUE if the column is an auto-incrementing column, or SQL_FALSE if the column is not an auto-incrementing column. This field is read-only, but the underlying auto-incrementing column is not necessarily read-only.</dd>
+#
+# <DT><B>SQL_DESC_BASE_COLUMN_NAME [IRDs]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains the base column name for the result set column. If a base column name does not exist (as in the case of columns that are expressions), this variable contains an empty string.</dd>
+#
+# <DT><B>SQL_DESC_BASE_TABLE_NAME [IRDs]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains the base table name for the result set column. If a base table name cannot be defined or is not applicable, this variable contains an empty string.</dd>
+#
+# <DT><B>SQL_DESC_CASE_SENSITIVE [Implementation descriptors]</B></DT>
+#
+# <DD>This read-only SQLINTEGER record field contains SQL_TRUE if the column or parameter is treated as case-sensitive for collations and comparisons, or SQL_FALSE if the column is not treated as case-sensitive for collations and comparisons or if it is a noncharacter column.</dd>
+#
+# <DT><B>SQL_DESC_CATALOG_NAME [IRDs]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains the catalog for the base table that contains the column. The return value is driver-dependent if the column is an expression or if the column is part of a view. If the data source does not support catalogs or the catalog cannot be determined, this variable contains an empty string.</dd>
+#
+# <DT><B>SQL_DESC_CONCISE_TYPE [All]</B></DT>
+#
+# <DD>This SQLSMALLINT header field specifies the concise data type for all data types, including the datetime and interval data types.
+#
+# <P>The values in the SQL_DESC_CONCISE_TYPE, SQL_DESC_TYPE, and SQL_DESC_DATETIME_INTERVAL_CODE fields are interdependent. Each time one of the fields is set, the other must also be set. SQL_DESC_CONCISE_TYPE can be set by a call to <B>SQLBindCol</B> or <B>SQLBindParameter</B>, or <B>SQLSetDescField</B>. SQL_DESC_TYPE can be set by a call to <B>SQLSetDescField</B> or <B>SQLSetDescRec</B>.
+#
+#
+# <P>If SQL_DESC_CONCISE_TYPE is set to a concise data type other than an interval or datetime data type, the SQL_DESC_TYPE field is set to the same value and the SQL_DESC_DATETIME_INTERVAL_CODE field is set to 0.
+#
+#
+# <P>If SQL_DESC_CONCISE_TYPE is set to the concise datetime or interval data type, the SQL_DESC_TYPE field is set to the corresponding verbose type (SQL_DATETIME or SQL_INTERVAL) and the SQL_DESC_DATETIME_INTERVAL_CODE field is set to the appropriate subcode.
+# </dd>
+#
+# <DT><B>SQL_DESC_DATA_PTR [Application descriptors and IPDs]</B></DT>
+#
+# <DD>This SQLPOINTER record field points to a variable that will contain the parameter value (for APDs) or the column value (for ARDs). This field is a <I>deferred field</I>. It is not used at the time it is set but is used at a later time by the driver to retrieve data.
+#
+# <P>The column specified by the SQL_DESC_DATA_PTR field of the ARD is unbound if the <I>TargetValuePtr</I> argument in a call to <B>SQLBindCol</B> is a null pointer or if the SQL_DESC_DATA_PTR field in the ARD is set by a call to <B>SQLSetDescField</B> or <B>SQLSetDescRec</B> to a null pointer. Other fields are not affected if the SQL_DESC_DATA_PTR field is set to a null pointer.
+#
+#
+# <P>If the call to <B>SQLFetch</B> or <B>SQLFetchScroll </B>that fills in the buffer pointed to by this field did not return SQL_SUCCESS or SQL_SUCCESS_WITH_INFO, the contents of the buffer are undefined.
+#
+#
+# <P>Whenever the SQL_DESC_DATA_PTR field of an APD, ARD, or IPD is set, the driver checks that the value in the SQL_DESC_TYPE field contains one of the valid ODBC C data types or a driver-specific data type, and that all other fields affecting the data types are consistent. Prompting a consistency check is the only use of the SQL_DESC_DATA_PTR field of an IPD. Specifically, if an application sets the SQL_DESC_DATA_PTR field of an IPD and later calls <B>SQLGetDescField</B> on this field, it is not necessarily returned the value that it had set. For more information, see "Consistency Checks" in <A HREF="odbcsqlsetdescrec.htm">SQLSetDescRec</A>.
+# </dd>
+#
+# <DT><B>SQL_DESC_DATETIME_INTERVAL_CODE [All]</B></DT>
+#
+# <DD>This SQLSMALLINT record field contains the subcode for the specific datetime or interval data type when the SQL_DESC_TYPE field is SQL_DATETIME or SQL_INTERVAL. This is true for both SQL and C data types. The code consists of the data type name with "CODE" substituted for either "TYPE" or "C_TYPE" (for datetime types), or "CODE" substituted for "INTERVAL" or "C_INTERVAL" (for interval types).
+#
+# <P>If SQL_DESC_TYPE and SQL_DESC_CONCISE_TYPE in an application descriptor are set to SQL_C_DEFAULT and the descriptor is not associated with a statement handle, the contents of SQL_DESC_DATETIME_INTERVAL_CODE are undefined.
+#
+#
+# <P>This field can be set for the datetime data types listed in the following table.
+#
+# <!--TS:-->
+# <div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=50%>Datetime types</TH>
+# <TH width=50%>DATETIME_INTERVAL_CODE</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_TYPE_DATE/SQL_C_TYPE_DATE</TD>
+# <TD width=50%>SQL_CODE_DATE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_TYPE_TIME/SQL_C_TYPE_TIME</TD>
+# <TD width=50%>SQL_CODE_TIME</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_TYPE_TIMESTAMP/<BR>
+# SQL_C_TYPE_TIMESTAMP</TD>
+# <TD width=50%>SQL_CODE_TIMESTAMP</TD>
+# </TR>
+# </table></div>
+#
+# <!--TS:-->
+#
+# <P>This field can be set for the interval data types listed in the following table.
+#
+# <!--TS:-->
+# <div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=50%>Interval type</TH>
+# <TH width=50%>DATETIME_INTERVAL_CODE</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_DAY/<BR>
+# SQL_C_INTERVAL_DAY</TD>
+# <TD width=50%>SQL_CODE_DAY</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_DAY_TO_HOUR/<BR>
+# SQL_C_INTERVAL_DAY_TO_HOUR</TD>
+# <TD width=50%>SQL_CODE_DAY_TO_HOUR</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_DAY_TO_MINUTE/<BR>
+# SQL_C_INTERVAL_DAY_TO_MINUTE</TD>
+# <TD width=50%>SQL_CODE_DAY_TO_MINUTE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_DAY_TO_SECOND/<BR>
+# SQL_C_INTERVAL_DAY_TO_SECOND</TD>
+# <TD width=50%>SQL_CODE_DAY_TO_SECOND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_HOUR/<BR>
+# SQL_C_INTERVAL_HOUR</TD>
+# <TD width=50%>SQL_CODE_HOUR</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_HOUR_TO_MINUTE/<BR>
+# SQL_C_INTERVAL_HOUR_TO_MINUTE</TD>
+# <TD width=50%>SQL_CODE_HOUR_TO_MINUTE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_HOUR_TO_SECOND/<BR>
+# SQL_C_INTERVAL_HOUR_TO_SECOND</TD>
+# <TD width=50%>SQL_CODE_HOUR_TO_SECOND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_MINUTE/<BR>
+# SQL_C_INTERVAL_MINUTE</TD>
+# <TD width=50%>SQL_CODE_MINUTE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_MINUTE_TO_SECOND/<BR>
+# SQL_C_INTERVAL_MINUTE_TO_SECOND</TD>
+# <TD width=50%>SQL_CODE_MINUTE_TO_SECOND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_MONTH/<BR>
+# SQL_C_INTERVAL_MONTH</TD>
+# <TD width=50%>SQL_CODE_MONTH</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_SECOND/<BR>
+# SQL_C_INTERVAL_SECOND</TD>
+# <TD width=50%>SQL_CODE_SECOND</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_YEAR/<BR>
+# SQL_C_INTERVAL_YEAR</TD>
+# <TD width=50%>SQL_CODE_YEAR</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTERVAL_YEAR_TO_MONTH/<BR>
+# SQL_C_INTERVAL_YEAR_TO_MONTH</TD>
+# <TD width=50%>SQL_CODE_YEAR_TO_MONTH</TD>
+# </TR>
+# </table></div>
+#
+# <!--TS:-->
+#
+# <P>For more information about the data intervals and this field, see "<A HREF="odbcdata_type_identifiers_and_descriptors.htm">Data Type Identifiers and Descriptors</A>" in Appendix D: Data Types.
+# </dd>
+#
+# <DT><B>SQL_DESC_DATETIME_INTERVAL_PRECISION [All]</B></DT>
+#
+# <DD>This SQLINTEGER record field contains the interval leading precision if the SQL_DESC_TYPE field is SQL_INTERVAL. When the SQL_DESC_DATETIME_INTERVAL_CODE field is set to an interval data type, this field is set to the default interval leading precision.</dd>
+#
+# <DT><B>SQL_DESC_DISPLAY_SIZE [IRDs]</B></DT>
+#
+# <DD>This read-only SQLINTEGER record field contains the maximum number of characters required to display the data from the column. </dd>
+#
+# <DT><B>SQL_DESC_FIXED_PREC_SCALE [Implementation descriptors]</B></DT>
+#
+# <DD>This read-only SQLSMALLINT record field is set to SQL_TRUE if the column is an exact numeric column and has a fixed precision and nonzero scale, or to SQL_FALSE if the column is not an exact numeric column with a fixed precision and scale.</dd>
+#
+# <DT><B>SQL_DESC_INDICATOR_PTR [Application descriptors]</B></DT>
+#
+# <DD>In ARDs, this SQLINTEGER * record field points to the indicator variable. This variable contains SQL_NULL_DATA if the column value is a NULL. For APDs, the indicator variable is set to SQL_NULL_DATA to specify NULL dynamic arguments. Otherwise, the variable is zero (unless the values in SQL_DESC_INDICATOR_PTR and SQL_DESC_OCTET_LENGTH_PTR are the same pointer).
+#
+# <P>If the SQL_DESC_INDICATOR_PTR field in an ARD is a null pointer, the driver is prevented from returning information about whether the column is NULL or not. If the column is NULL and SQL_DESC_INDICATOR_PTR is a null pointer, SQLSTATE 22002 (Indicator variable required but not supplied) is returned when the driver attempts to populate the buffer after a call to <B>SQLFetch</B> or <B>SQLFetchScroll</B>. If the call to <B>SQLFetch</B> or <B>SQLFetchScroll </B>did not return SQL_SUCCESS or SQL_SUCCESS_WITH_INFO, the contents of the buffer are undefined.
+#
+#
+# <P>The SQL_DESC_INDICATOR_PTR field determines whether the field pointed to by SQL_DESC_OCTET_LENGTH_PTR is set. If the data value for a column is NULL, the driver sets the indicator variable to SQL_NULL_DATA. The field pointed to by SQL_DESC_OCTET_LENGTH_PTR is then not set. If a NULL value is not encountered during the fetch, the buffer pointed to by SQL_DESC_INDICATOR_PTR is set to zero and the buffer pointed to by SQL_DESC_OCTET_LENGTH_PTR is set to the length of the data.
+#
+#
+# <P>If the SQL_DESC_INDICATOR_PTR field in an APD is a null pointer, the application cannot use this descriptor record to specify NULL arguments.
+#
+#
+# <P>This field is a <I>deferred field</I>: It is not used at the time it is set but is used at a later time by the driver to indicate nullability (for ARDs) or to determine nullability (for APDs).
+# </dd>
+#
+# <DT><B>SQL_DESC_LABEL [IRDs]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains the column label or title. If the column does not have a label, this variable contains the column name. If the column is unnamed and unlabeled, this variable contains an empty string.</dd>
+#
+# <DT><B>SQL_DESC_LENGTH [All]</B></DT>
+#
+# <DD>This SQLUINTEGER record field is either the maximum or actual length of a character string in characters or a binary data type in bytes. It is the maximum length for a fixed-length data type, or the actual length for a variable-length data type. Its value always excludes the null-termination character that ends the character string. For values whose type is SQL_TYPE_DATE, SQL_TYPE_TIME, SQL_TYPE_TIMESTAMP, or one of the SQL interval data types, this field has the length in characters of the character string representation of the datetime or interval value.
+#
+# <P>The value in this field may be different from the value for "length" as defined in ODBC 2<I>.x</I>. For more information, see <A HREF="odbcdata_types.htm">Appendix D: Data Types</A>.
+# </dd>
+#
+# <DT><B>SQL_DESC_LITERAL_PREFIX [IRDs]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains the character or characters that the driver recognizes as a prefix for a literal of this data type. This variable contains an empty string for a data type for which a literal prefix is not applicable.</dd>
+#
+# <DT><B>SQL_DESC_LITERAL_SUFFIX [IRDs]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains the character or characters that the driver recognizes as a suffix for a literal of this data type. This variable contains an empty string for a data type for which a literal suffix is not applicable.</dd>
+#
+# <DT><B>SQL_DESC_LOCAL_TYPE_NAME [Implementation descriptors]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains any localized (native language) name for the data type that may be different from the regular name of the data type. If there is no localized name, an empty string is returned. This field is for display purposes only.</dd>
+#
+# <DT><B>SQL_DESC_NAME [Implementation descriptors]</B></DT>
+#
+# <DD>This SQLCHAR * record field in a row descriptor contains the column alias, if it applies. If the column alias does not apply, the column name is returned. In either case, the driver sets the SQL_DESC_UNNAMED field to SQL_NAMED when it sets the SQL_DESC_NAME field. If there is no column name or a column alias, the driver returns an empty string in the SQL_DESC_NAME field and sets the SQL_DESC_UNNAMED field to SQL_UNNAMED.
+#
+# <P>An application can set the SQL_DESC_NAME field of an IPD to a parameter name or alias to specify stored procedure parameters by name. (For more information, see "<A HREF="odbcbinding_parameters_by_name__named_parameters_.htm">Binding Parameters by Name (Named Parameters)</A>" in Chapter 9: Executing Statements.) The SQL_DESC_NAME field of an IRD is a read-only field; SQLSTATE HY091 (Invalid descriptor field identifier) will be returned if an application attempts to set it.
+#
+#
+# <P>In IPDs, this field is undefined if the driver does not support named parameters. If the driver supports named parameters and is capable of describing parameters, the parameter name is returned in this field.
+# </dd>
+#
+# <DT><B>SQL_DESC_NULLABLE [Implementation descriptors]</B></DT>
+#
+# <DD>In IRDs, this read-only SQLSMALLINT record field is SQL_NULLABLE if the column can have NULL values, SQL_NO_NULLS if the column does not have NULL values, or SQL_NULLABLE_UNKNOWN if it is not known whether the column accepts NULL values. This field pertains to the result set column, not the base column.
+#
+# <P>In IPDs, this field is always set to SQL_NULLABLE because dynamic parameters are always nullable and cannot be set by an application.
+# </dd>
+#
+# <DT><B>SQL_DESC_NUM_PREC_RADIX [All]</B></DT>
+#
+# <DD>This SQLINTEGER field contains a value of 2 if the data type in the SQL_DESC_TYPE field is an approximate numeric data type, because the SQL_DESC_PRECISION field contains the number of bits. This field contains a value of 10 if the data type in the SQL_DESC_TYPE field is an exact numeric data type, because the SQL_DESC_PRECISION field contains the number of decimal digits. This field is set to 0 for all non-numeric data types.</dd>
+#
+# <DT><B>SQL_DESC_OCTET_LENGTH [All]</B></DT>
+#
+# <DD>This SQLINTEGER record field contains the length, in bytes, of a character string or binary data type. For fixed-length character or binary types, this is the actual length in bytes. For variable-length character or binary types, this is the maximum length in bytes. This value always excludes space for the null-termination character for implementation descriptors and always includes space for the null-termination character for application descriptors. For application data, this field contains the size of the buffer. For APDs, this field is defined only for output or input/output parameters.</dd>
+#
+# <DT><B>SQL_DESC_OCTET_LENGTH_PTR [Application descriptors]</B></DT>
+#
+# <DD>This SQLINTEGER * record field points to a variable that will contain the total length in bytes of a dynamic argument (for parameter descriptors) or of a bound column value (for row descriptors).
+#
+# <P>For an APD, this value is ignored for all arguments except character string and binary; if this field points to SQL_NTS, the dynamic argument must be null-terminated. To indicate that a bound parameter will be a data-at-execution parameter, an application sets this field in the appropriate record of the APD to a variable that, at execute time, will contain the value SQL_DATA_AT_EXEC or the result of the SQL_LEN_DATA_AT_EXEC macro. If there is more than one such field, SQL_DESC_DATA_PTR can be set to a value uniquely identifying the parameter to help the application determine which parameter is being requested.
+#
+#
+# <P>If the OCTET_LENGTH_PTR field of an ARD is a null pointer, the driver does not return length information for the column. If the SQL_DESC_OCTET_LENGTH_PTR field of an APD is a null pointer, the driver assumes that character strings and binary values are null-terminated. (Binary values should not be null-terminated but should be given a length to avoid truncation.)
+#
+#
+# <P>If the call to <B>SQLFetch</B> or <B>SQLFetchScroll</B> that fills in the buffer pointed to by this field did not return SQL_SUCCESS or SQL_SUCCESS_WITH_INFO, the contents of the buffer are undefined. This field is a <I>deferred field</I>. It is not used at the time it is set but is used at a later time by the driver to determine or indicate the octet length of the data.
+# </dd>
+#
+# <DT><B>SQL_DESC_PARAMETER_TYPE [IPDs]</B></DT>
+#
+# <DD>This SQLSMALLINT record field is set to SQL_PARAM_INPUT for an input parameter, SQL_PARAM_INPUT_OUTPUT for an input/output parameter, or SQL_PARAM_OUTPUT for an output parameter. It is set to SQL_PARAM_INPUT by default.
+#
+# <P>For an IPD, the field is set to SQL_PARAM_INPUT by default if the IPD is not automatically populated by the driver (the SQL_ATTR_ENABLE_AUTO_IPD statement attribute is SQL_FALSE). An application should set this field in the IPD for parameters that are not input parameters.
+# </dd>
+#
+# <DT><B>SQL_DESC_PRECISION [All]</B></DT>
+#
+# <DD>This SQLSMALLINT record field contains the number of digits for an exact numeric type, the number of bits in the mantissa (binary precision) for an approximate numeric type, or the numbers of digits in the fractional seconds component for the SQL_TYPE_TIME, SQL_TYPE_TIMESTAMP, or SQL_INTERVAL_SECOND data type. This field is undefined for all other data types.
+#
+# <P>The value in this field may be different from the value for "precision" as defined in ODBC 2<I>.x</I>. For more information, see <A HREF="odbcdata_types.htm">Appendix D: Data Types</A>.
+# </dd>
+#
+# <DT><B>SQL_DESC_ROWVER [Implementation descriptors]</B></DT>
+#
+# <DD>This SQLSMALLINT<B> </B>record field indicates whether a column is automatically modified by the DBMS when a row is updated (for example, a column of the type "timestamp" in SQL Server). The value of this record field is set to SQL_TRUE if the column is a row versioning column, and to SQL_FALSE otherwise. This column attribute is similar to calling <B>SQLSpecialColumns</B> with IdentifierType of SQL_ROWVER to determine whether a column is automatically updated.</dd>
+#
+# <DT><B>SQL_DESC_SCALE [All]</B></DT>
+#
+# <DD>This SQLSMALLINT record field contains the defined scale for decimal and numeric data types. The field is undefined for all other data types.
+#
+# <P>The value in this field may be different from the value for "scale" as defined in ODBC 2<I>.x</I>. For more information, see <A HREF="odbcdata_types.htm">Appendix D: Data Types</A>.
+# </dd>
+#
+# <DT><B>SQL_DESC_SCHEMA_NAME [IRDs]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains the schema name of the base table that contains the column. The return value is driver-dependent if the column is an expression or if the column is part of a view. If the data source does not support schemas or the schema name cannot be determined, this variable contains an empty string.</dd>
+#
+# <DT><B>SQL_DESC_SEARCHABLE [IRDs]</B></DT>
+#
+# <DD>This read-only SQLSMALLINT record field is set to one of the following values:
+#
+# <UL type=disc>
+# <LI>SQL_PRED_NONE if the column cannot be used in a <B>WHERE</B> clause. (This is the same as the SQL_UNSEARCHABLE value in ODBC 2<I>.x</I>.)</li>
+#
+# <LI>SQL_PRED_CHAR if the column can be used in a <B>WHERE</B> clause but only with the <B>LIKE</B> predicate. (This is the same as the SQL_LIKE_ONLY value in ODBC 2<I>.x</I>.)</li>
+#
+# <LI>SQL_PRED_BASIC if the column can be used in a <B>WHERE</B> clause with all the comparison operators except <B>LIKE</B>. (This is the same as the SQL_EXCEPT_LIKE value in ODBC 2<I>.x</I>.)</li>
+#
+# <LI>SQL_PRED_SEARCHABLE if the column can be used in a <B>WHERE</B> clause with any comparison operator.</li>
+# </UL>
+# </dd>
+#
+# <DT><B>SQL_DESC_TABLE_NAME [IRDs]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains the name of the base table that contains this column. The return value is driver-dependent if the column is an expression or if the column is part of a view.</dd>
+#
+# <DT><B>SQL_DESC_TYPE [All]</B></DT>
+#
+# <DD>This SQLSMALLINT record field specifies the concise SQL or C data type for all data types except datetime and interval data types. For the datetime and interval data types, this field specifies the verbose data type, which is SQL_DATETIME or SQL_INTERVAL.
+#
+# <P>Whenever this field contains SQL_DATETIME or SQL_INTERVAL, the SQL_DESC_DATETIME_INTERVAL_CODE field must contain the appropriate subcode for the concise type. For datetime data types, SQL_DESC_TYPE contains SQL_DATETIME, and the SQL_DESC_DATETIME_INTERVAL_CODE field contains a subcode for the specific datetime data type. For interval data types, SQL_DESC_TYPE contains SQL_INTERVAL and the SQL_DESC_DATETIME_INTERVAL_CODE field contains a subcode for the specific interval data type.
+#
+#
+# <P>The values in the SQL_DESC_TYPE and SQL_DESC_CONCISE_TYPE fields are interdependent. Each time one of the fields is set, the other must also be set. SQL_DESC_TYPE can be set by a call to <B>SQLSetDescField</B> or <B>SQLSetDescRec</B>. SQL_DESC_CONCISE_TYPE can be set by a call to <B>SQLBindCol</B> or <B>SQLBindParameter</B>, or <B>SQLSetDescField</B>.
+#
+#
+# <P>If SQL_DESC_TYPE is set to a concise data type other than an interval or datetime data type, the SQL_DESC_CONCISE_TYPE field is set to the same value and the SQL_DESC_DATETIME_INTERVAL_CODE field is set to 0.
+#
+#
+# <P>If SQL_DESC_TYPE is set to the verbose datetime or interval data type (SQL_DATETIME or SQL_INTERVAL) and the SQL_DESC_DATETIME_INTERVAL_CODE field is set to the appropriate subcode, the SQL_DESC_CONCISE TYPE field is set to the corresponding concise type. Trying to set SQL_DESC_TYPE to one of the concise datetime or interval types will return SQLSTATE HY021 (Inconsistent descriptor information).
+#
+#
+# <P>When the SQL_DESC_TYPE field is set by a call to <B>SQLBindCol</B>, <B>SQLBindParameter</B>, or <B>SQLSetDescField</B>, the following fields are set to the following default values, as shown in the table below. The values of the remaining fields of the same record are undefined.
+#
+# <!--TS:-->
+# <div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=48%>Value of SQL_DESC_TYPE</TH>
+# <TH width=52%>Other fields implicitly set</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>SQL_CHAR, SQL_VARCHAR, SQL_C_CHAR, SQL_C_VARCHAR</TD>
+# <TD width=52%>SQL_DESC_LENGTH is set to 1. SQL_DESC_PRECISION is set to 0.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>SQL_DATETIME</TD>
+# <TD width=52%>When SQL_DESC_DATETIME_INTERVAL_CODE is set to SQL_CODE_DATE or SQL_CODE_TIME, SQL_DESC_PRECISION is set to 0. When it is set to SQL_DESC_TIMESTAMP, SQL_DESC_PRECISION is set to 6.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>SQL_DECIMAL, SQL_NUMERIC,<BR>
+# SQL_C_NUMERIC</TD>
+# <TD width=52%>SQL_DESC_SCALE is set to 0. SQL_DESC_PRECISION is set to the implementation-defined precision for the respective data type.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>SQL_FLOAT, SQL_C_FLOAT</TD>
+# <TD width=52%>SQL_DESC_PRECISION is set to the implementation-defined default precision for SQL_FLOAT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>SQL_INTERVAL</TD>
+# <TD width=52%>When SQL_DESC_DATETIME_INTERVAL_CODE is set to an interval data type, SQL_DESC_DATETIME_INTERVAL_PRECISION is set to 2 (the default interval leading precision). When the interval has a seconds component, SQL_DESC_PRECISION is set to 6 (the default interval seconds precision).</TD>
+# </TR>
+# </table></div>
+#
+# <!--TS:-->
+#
+# <P>When an application calls <B>SQLSetDescField</B> to set fields of a descriptor rather than calling <B>SQLSetDescRec</B>, the application must first declare the data type. When it does, the other fields indicated in the previous table are implicitly set. If any of the values implicitly set are unacceptable, the application can then call <B>SQLSetDescField</B> or <B>SQLSetDescRec</B> to set the unacceptable value explicitly.
+# </dd>
+#
+# <DT><B>SQL_DESC_TYPE_NAME [Implementation descriptors]</B></DT>
+#
+# <DD>This read-only SQLCHAR * record field contains the data source&#0150;dependent type name (for example, "CHAR", "VARCHAR", and so on). If the data type name is unknown, this variable contains an empty string.</dd>
+#
+# <DT><B>SQL_DESC_UNNAMED [Implementation descriptors]</B></DT>
+#
+# <DD>This SQLSMALLINT record field in a row descriptor is set by the driver to either SQL_NAMED or SQL_UNNAMED when it sets the SQL_DESC_NAME field. If the SQL_DESC_NAME field contains a column alias or if the column alias does not apply, the driver sets the SQL_DESC_UNNAMED field to SQL_NAMED. If an application sets the SQL_DESC_NAME field of an IPD to a parameter name or alias, the driver sets the SQL_DESC_UNNAMED field of the IPD to SQL_NAMED. If there is no column name or a column alias, the driver sets the SQL_DESC_UNNAMED field to SQL_UNNAMED.
+#
+# <P>An application can set the SQL_DESC_UNNAMED field of an IPD to SQL_UNNAMED. A driver returns SQLSTATE HY091 (Invalid descriptor field identifier) if an application attempts to set the SQL_DESC_UNNAMED field of an IPD to SQL_NAMED. The SQL_DESC_UNNAMED field of an IRD is read-only; SQLSTATE HY091 (Invalid descriptor field identifier) will be returned if an application attempts to set it.
+# </dd>
+#
+# <DT><B>SQL_DESC_UNSIGNED [Implementation descriptors]</B></DT>
+#
+# <DD>This read-only SQLSMALLINT record field is set to SQL_TRUE if the column type is unsigned or non-numeric, or SQL_FALSE if the column type is signed.</dd>
+#
+# <DT><B>SQL_DESC_UPDATABLE [IRDs]</B></DT>
+#
+# <DD>This read-only SQLSMALLINT record field is set to one of the following values:
+#
+# <UL type=disc>
+# <LI>SQL_ATTR_READ_ONLY if the result set column is read-only.</li>
+#
+# <LI>SQL_ATTR_WRITE if the result set column is read-write.</li>
+#
+# <LI>SQL_ATTR_READWRITE_UNKNOWN if it is not known whether the result set column is updatable or not.</li>
+# </UL>
+#
+#
+# <P>SQL_DESC_UPDATABLE describes the updatability of the column in the result set, not the column in the base table. The updatability of the column in the base table on which this result set column is based may be different than the value in this field. Whether a column is updatable can be based on the data type, user privileges, and the definition of the result set itself. If it is unclear whether a column is updatable, SQL_ATTR_READWRITE_UNKNOWN should be returned.
+# </dd>
+# </DL>
+#
+# <H1>Consistency Checks</H1>
+#
+# <P>A consistency check is performed by the driver automatically whenever an application passes in a value for the SQL_DESC_DATA_PTR field of the ARD, APD, or IPD. If any of the fields is inconsistent with other fields, <B>SQLSetDescField</B> will return SQLSTATE HY021 (Inconsistent descriptor information). For more information, see "Consistency Check" in <A HREF="odbcsqlsetdescrec.htm">SQLSetDescRec</A>.</P>
+#
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=48%>For information about</TH>
+# <TH width=52%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>Binding a column</TD>
+# <TD width=52%><A HREF="odbcsqlbindcol.htm">SQLBindCol</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>Binding a parameter</TD>
+# <TD width=52%><A HREF="odbcsqlbindparameter.htm">SQLBindParameter</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>Getting a descriptor field</TD>
+# <TD width=52%><A HREF="odbcsqlgetdescfield.htm">SQLGetDescField</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>Getting multiple descriptor fields</TD>
+# <TD width=52%><A HREF="odbcsqlgetdescrec.htm">SQLGetDescRec</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>Setting multiple descriptor fields</TD>
+# <TD width=52%><A HREF="odbcsqlsetdescrec.htm">SQLSetDescRec</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+};
diff --git a/ndb/src/client/odbc/docs/diag.txt b/ndb/src/client/odbc/docs/diag.txt
new file mode 100644
index 00000000000..a9a0e0f42d0
--- /dev/null
+++ b/ndb/src/client/odbc/docs/diag.txt
@@ -0,0 +1,48 @@
+# Header Fields
+
+SQL_DIAG_CURSOR_ROW_COUNT
+SQLINTEGER
+
+SQL_DIAG_DYNAMIC_FUNCTION
+SQLCHAR *
+
+SQL_DIAG_DYNAMIC_FUNCTION_CODE
+SQLINTEGER
+
+SQL_DIAG_NUMBER
+SQLINTEGER
+
+SQL_DIAG_RETURNCODE
+SQLRETURN
+
+SQL_DIAG_ROW_COUNT
+SQLINTEGER
+
+# Record Fields
+
+SQL_DIAG_CLASS_ORIGIN
+SQLCHAR *
+
+SQL_DIAG_COLUMN_NUMBER
+SQLINTEGER
+
+SQL_DIAG_CONNECTION_NAME
+SQLCHAR *
+
+SQL_DIAG_MESSAGE_TEXT
+SQLCHAR *
+
+SQL_DIAG_NATIVE
+SQLINTEGER
+
+SQL_DIAG_ROW_NUMBER
+SQLINTEGER
+
+SQL_DIAG_SERVER_NAME
+SQLCHAR *
+
+SQL_DIAG_SQLSTATE
+SQLCHAR *
+
+SQL_DIAG_SUBCLASS_ORIGIN
+SQLCHAR *
diff --git a/ndb/src/client/odbc/docs/getinfo.pl b/ndb/src/client/odbc/docs/getinfo.pl
new file mode 100644
index 00000000000..34e26b47bab
--- /dev/null
+++ b/ndb/src/client/odbc/docs/getinfo.pl
@@ -0,0 +1,3676 @@
+#
+use strict;
+
+#
+# odbcsqlgetinfo.htm
+#
+my $info = {
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLGetInfo</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqlgetinfo"></A>SQLGetInfo</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 1.0<BR>
+# Standards Compliance: ISO 92</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLGetInfo</B> returns general information about the driver and data source associated with a connection.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLGetInfo</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHDBC&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>ConnectionHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLUSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>InfoType</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLPOINTER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>InfoValuePtr</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>BufferLength</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StringLengthPtr</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>ConnectionHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Connection handle.</dd>
+#
+# <DT><I>InfoType</I></DT>
+#
+# <DD>[Input]<BR>
+# Type of information.</dd>
+#
+# <DT><I>InfoValuePtr</I></DT>
+#
+# <DD>[Output]<BR>
+# Pointer to a buffer in which to return the information. Depending on the <I>InfoType</I> requested, the information returned will be one of the following: a null-terminated character string, an SQLUSMALLINT value, an SQLUINTEGER bitmask, an SQLUINTEGER flag, or a SQLUINTEGER binary value.
+#
+# <P>If the <I>InfoType</I> argument is SQL_DRIVER_HDESC or SQL_DRIVER_HSTMT, the <I>InfoValuePtr</I> argument is both input and output. (See the SQL_DRIVER_HDESC or SQL_DRIVER_HSTMT descriptors later in this function description for more information.)
+# </dd>
+#
+# <DT><I>BufferLength</I></DT>
+#
+# <DD>[Input]<BR>
+# Length of the *<I>InfoValuePtr</I> buffer. If the value in <I>*InfoValuePtr</I> is not a character string or if <I>InfoValuePtr</I> is a null pointer,<I> </I>the <I>BufferLength</I> argument is ignored. The driver assumes that the size of <I>*InfoValuePtr</I> is SQLUSMALLINT or SQLUINTEGER, based on the <I>InfoType</I>. If <I>*InfoValuePtr</I> is a Unicode string (when calling <B>SQLGetInfoW</B>), the <I>BufferLength</I> argument must be an even number; if not, SQLSTATE HY090 (Invalid string or buffer length) is returned. </dd>
+#
+# <DT><I>StringLengthPtr</I></DT>
+#
+# <DD>[Output]<BR>
+# Pointer to a buffer in which to return the total number of bytes (excluding the null-termination character for character data) available to return in *<I>InfoValuePtr</I>.
+#
+# <P>For character data, if the number of bytes available to return is greater than or equal to <I>BufferLength</I>, the information in *<I>InfoValuePtr</I> is truncated to <I>BufferLength</I> bytes minus the length of a null-termination character and is null-terminated by the driver.
+#
+#
+# <P>For all other types of data, the value of <I>BufferLength</I> is ignored and the driver assumes the size of *<I>InfoValuePtr</I> is SQLUSMALLINT or SQLUINTEGER, depending on the <I>InfoType</I>.
+# </dd>
+# </DL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLGetInfo</B> returns either SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value can be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_DBC and a <I>Handle</I> of <I>ConnectionHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLGetInfo</B> and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=23%>SQLSTATE</TH>
+# <TH width=26%>Error</TH>
+# <TH width=51%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>01000</TD>
+# <TD width=26%>General warning</TD>
+# <TD width=51%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>01004</TD>
+# <TD width=26%>String data, right truncated</TD>
+# <TD width=51%>The buffer *<I>InfoValuePtr</I> was not large enough to return all of the requested information, so the information was truncated. The length of the requested information in its untruncated form is returned in *<I>StringLengthPtr</I>. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>08003</TD>
+# <TD width=26%>Connection does not exist</TD>
+# <TD width=51%>(DM) The type of information requested in <I>InfoType</I> requires an open connection. Of the information types reserved by ODBC, only SQL_ODBC_VER can be returned without an open connection.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>08S01</TD>
+# <TD width=26%>Communication link failure</TD>
+# <TD width=51%>The communication link between the driver and the data source to which the driver was connected failed before the function completed processing.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>HY000</TD>
+# <TD width=26%>General error</TD>
+# <TD width=51%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>HY001</TD>
+# <TD width=26%>Memory allocation <BR>
+# error</TD>
+# <TD width=51%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>HY013</TD>
+# <TD width=26%>Memory management error</TD>
+# <TD width=51%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>HY024</TD>
+# <TD width=26%>Invalid attribute value</TD>
+# <TD width=51%>(DM) The <I>InfoType</I> argument was SQL_DRIVER_HSTMT, and the value pointed to by <I>InfoValuePtr</I> was not a valid statement handle.
+# <P>(DM) The <I>InfoType</I> argument was SQL_DRIVER_HDESC, and the value pointed to by <I>InfoValuePtr</I> was not a valid descriptor handle.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>HY090</TD>
+# <TD width=26%>Invalid string or buffer length</TD>
+# <TD width=51%>(DM) The value specified for argument <I>BufferLength</I> was less than 0.
+# <P>(DM) The value specified for <I>BufferLength</I> was an odd number, and <I>*InfoValuePtr </I>was of a Unicode data type.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>HY096</TD>
+# <TD width=26%>Information type out of range</TD>
+# <TD width=51%>The value specified for the argument <I>InfoType</I> was not valid for the version of ODBC supported by the driver.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>HYC00</TD>
+# <TD width=26%>Optional field not implemented</TD>
+# <TD width=51%>The value specified for the argument <I>InfoType</I> was a driver-specific value that is not supported by the driver.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>HYT01</TD>
+# <TD width=26%>Connection timeout expired</TD>
+# <TD width=51%>The connection timeout period expired before the data source responded to the request. The connection timeout period is set through <B>SQLSetConnectAttr</B>, SQL_ATTR_CONNECTION_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=23%>IM001</TD>
+# <TD width=26%>Driver does not support this function</TD>
+# <TD width=51%>(DM) The driver corresponding to the <I>ConnectionHandle</I> does not support the function.</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Comments</B></P>
+#
+# <P>The currently defined information types are shown in "Information Types," later in this section; it is expected that more will be defined to take advantage of different data sources. A range of information types is reserved by ODBC; driver developers must reserve values for their own driver-specific use from X/Open. <B>SQLGetInfo</B> performs no Unicode conversion or <I>thunking</I> (see <A HREF="odbcodbc_error_codes.htm">Appendix A</A> of the <I>ODBC Programmer's Reference</I>) for driver-defined <I>InfoTypes</I>. For more information, see "<A HREF="odbcdriver_specific_data_types__descriptor_types__information_types.htm">Driver-Specific Data Types, Descriptor Types, Information Types, Diagnostic Types, and Attributes</A>" in Chapter 17: Programming Considerations. The format of the information returned in *<I>InfoValuePtr</I> depends on the <I>InfoType</I> requested. <B>SQLGetInfo</B> will return information in one of five different formats:
+#
+# <UL type=disc>
+# <LI>A null-terminated character string</li>
+#
+# <LI>An SQLUSMALLINT value</li>
+#
+# <LI>An SQLUINTEGER bitmask</li>
+#
+# <LI>An SQLUINTEGER value</li>
+#
+# <LI>A SQLUINTEGER binary value</li>
+# </UL>
+#
+# <P>The format of each of the following information types is noted in the type's description. The application must cast the value returned in *<I>InfoValuePtr</I> accordingly. For an example of how an application could retrieve data from a SQLUINTEGER bitmask, see "Code Example."</P>
+#
+# <P>A driver must return a value for each of the information types defined in the tables below. If an information type does not apply to the driver or data source, the driver returns one of the values listed in the following table.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=49%>Format of *<I>InfoValuePtr</I></TH>
+# <TH width=51%>Returned value</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>Character string ("Y" or "N")</TD>
+# <TD width=51%>"N"</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>Character string (not "Y" or "N")</TD>
+# <TD width=51%>Empty string</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQLUSMALLINT</TD>
+# <TD width=51%>0</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQLUINTEGER bitmask or SQLUINTEGER binary value</TD>
+# <TD width=51%>0L</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P>For example, if a data source does not support procedures, <B>SQLGetInfo</B> returns the values listed in the following table for the values of <I>InfoType</I> that are related to procedures.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=49%><I>InfoType</I></TH>
+# <TH width=51%>Returned value</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_PROCEDURES</TD>
+# <TD width=51%>"N"</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ACCESSIBLE_PROCEDURES</TD>
+# <TD width=51%>"N"</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_PROCEDURE_NAME_LEN</TD>
+# <TD width=51%>0</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_PROCEDURE_TERM</TD>
+# <TD width=51%>Empty string</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P><B>SQLGetInfo</B> returns SQLSTATE HY096 (Invalid argument value) for values of <I>InfoType</I> that are in the range of information types reserved for use by ODBC but are not defined by the version of ODBC supported by the driver. To determine what version of ODBC a driver conforms to, an application calls <B>SQLGetInfo</B> with the SQL_DRIVER_ODBC_VER information type. <B>SQLGetInfo</B> returns SQLSTATE HYC00 (Optional feature not implemented) for values of <I>InfoType</I> that are in the range of information types reserved for driver-specific use but are not supported by the driver.</P>
+#
+# <P>All calls to <B>SQLGetInfo</B> require an open connection, except when the <I>InfoType</I> is SQL_ODBC_VER, which returns the version of the Driver Manager.</P>
+#
+# <H1>Information Types</H1>
+#
+# <P>This section lists the information types supported by <B>SQLGetInfo</B>. Information types are grouped categorically and listed alphabetically. Information types that were added or renamed for ODBC 3<I>.x</I> are also listed.</P>
+#
+# <H2>Driver Information</H2>
+#
+# <P>The following values of the <I>InfoType</I> argument return information about the ODBC driver, such as the number of active statements, the data source name, and the interface standards compliance level:</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_ACTIVE_ENVIRONMENTS</TD>
+# <TD width=57%>SQL_GETDATA_EXTENSIONS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_ASYNC_MODE</TD>
+# <TD width=57%>SQL_INFO_SCHEMA_VIEWS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_BATCH_ROW_COUNT</TD>
+# <TD width=57%>SQL_KEYSET_CURSOR_ATTRIBUTES1</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_BATCH_SUPPORT</TD>
+# <TD width=57%>SQL_KEYSET_CURSOR_ATTRIBUTES2</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DATA_SOURCE_NAME</TD>
+# <TD width=57%>SQL_MAX_ASYNC_CONCURRENT_STATEMENTS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DRIVER_HDBC</TD>
+# <TD width=57%>SQL_MAX_CONCURRENT_ACTIVITIES</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DRIVER_HDESC</TD>
+# <TD width=57%>SQL_MAX_DRIVER_CONNECTIONS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DRIVER_HENV</TD>
+# <TD width=57%>SQL_ODBC_INTERFACE_CONFORMANCE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DRIVER_HLIB</TD>
+# <TD width=57%>SQL_ODBC_STANDARD_CLI_CONFORMANCE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DRIVER_HSTMT</TD>
+# <TD width=57%>SQL_ODBC_VER</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DRIVER_NAME</TD>
+# <TD width=57%>SQL_PARAM_ARRAY_ROW_COUNTS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DRIVER_ODBC_VER</TD>
+# <TD width=57%>SQL_PARAM_ARRAY_SELECTS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DRIVER_VER</TD>
+# <TD width=57%>SQL_ROW_UPDATES</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DYNAMIC_CURSOR_ATTRIBUTES1</TD>
+# <TD width=57%>SQL_SEARCH_PATTERN_ESCAPE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_DYNAMIC_CURSOR_ATTRIBUTES2</TD>
+# <TD width=57%>SQL_SERVER_NAME</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES1</TD>
+# <TD width=57%>SQL_STATIC_CURSOR_ATTRIBUTES1</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES2</TD>
+# <TD width=57%>SQL_STATIC_CURSOR_ATTRIBUTES2</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>SQL_FILE_USAGE</TD>
+# <TD width=57%>&nbsp;</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <H2>DBMS Product Information</H2>
+#
+# <P>The following values of the <I>InfoType</I> argument return information about the DBMS product, such as the DBMS name and version:</P>
+#
+# <P>SQL_DATABASE_NAME<BR>
+# SQL_DBMS_NAME<BR>
+# SQL_DBMS_VER</P>
+#
+# <H2>Data Source Information</H2>
+#
+# <P>The following values of the <I>InfoType</I> argument return information about the data source, such as cursor characteristics and transaction capabilities:</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_ACCESSIBLE_PROCEDURES</TD>
+# <TD width=46%>SQL_MULT_RESULT_SETS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_ACCESSIBLE_TABLES</TD>
+# <TD width=46%>SQL_MULTIPLE_ACTIVE_TXN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_BOOKMARK_PERSISTENCE</TD>
+# <TD width=46%>SQL_NEED_LONG_DATA_LEN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_CATALOG_TERM</TD>
+# <TD width=46%>SQL_NULL_COLLATION</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_COLLATION_SEQ</TD>
+# <TD width=46%>SQL_PROCEDURE_TERM</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_CONCAT_NULL_BEHAVIOR</TD>
+# <TD width=46%>SQL_SCHEMA_TERM</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_CURSOR_COMMIT_BEHAVIOR</TD>
+# <TD width=46%>SQL_SCROLL_OPTIONS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_CURSOR_ROLLBACK_BEHAVIOR</TD>
+# <TD width=46%>SQL_TABLE_TERM</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_CURSOR_SENSITIVITY</TD>
+# <TD width=46%>SQL_TXN_CAPABLE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_DATA_SOURCE_READ_ONLY</TD>
+# <TD width=46%>SQL_TXN_ISOLATION_OPTION</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_DEFAULT_TXN_ISOLATION</TD>
+# <TD width=46%>SQL_USER_NAME</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=54%>SQL_DESCRIBE_PARAMETER</TD>
+# <TD width=46%>&nbsp;</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <H2>Supported SQL</H2>
+#
+# <P>The following values of the <I>InfoType</I> argument return information about the SQL statements supported by the data source. The SQL syntax of each feature described by these information types is the SQL-92 syntax. These information types do not exhaustively describe the entire SQL-92 grammar. Instead, they describe those parts of the grammar for which data sources commonly offer different levels of support. Specifically, most of the DDL statements in SQL-92 are covered.</P>
+#
+# <P>Applications should determine the general level of supported grammar from the SQL_SQL_CONFORMANCE information type and use the other information types to determine variations from the stated standards compliance level.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_AGGREGATE_FUNCTIONS</TD>
+# <TD width=51%>SQL_DROP_TABLE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ALTER_DOMAIN</TD>
+# <TD width=51%>SQL_DROP_TRANSLATION</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ALTER_SCHEMA</TD>
+# <TD width=51%>SQL_DROP_VIEW</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ALTER_TABLE</TD>
+# <TD width=51%>SQL_EXPRESSIONS_IN_ORDERBY</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ANSI_SQL_DATETIME_LITERALS</TD>
+# <TD width=51%>SQL_GROUP_BY</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CATALOG_LOCATION </TD>
+# <TD width=51%>SQL_IDENTIFIER_CASE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CATALOG_NAME</TD>
+# <TD width=51%>SQL_IDENTIFIER_QUOTE_CHAR</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CATALOG_NAME_SEPARATOR</TD>
+# <TD width=51%>SQL_INDEX_KEYWORDS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CATALOG_USAGE</TD>
+# <TD width=51%>SQL_INSERT_STATEMENT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_COLUMN_ALIAS</TD>
+# <TD width=51%>SQL_INTEGRITY</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CORRELATION_NAME</TD>
+# <TD width=51%>SQL_KEYWORDS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_ASSERTION</TD>
+# <TD width=51%>SQL_LIKE_ESCAPE_CLAUSE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_CHARACTER_SET</TD>
+# <TD width=51%>SQL_NON_NULLABLE_COLUMNS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_COLLATION</TD>
+# <TD width=51%>SQL_SQL_CONFORMANCE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_DOMAIN</TD>
+# <TD width=51%>SQL_OJ_CAPABILITIES</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_SCHEMA</TD>
+# <TD width=51%>SQL_ORDER_BY_COLUMNS_IN_SELECT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_TABLE</TD>
+# <TD width=51%>SQL_OUTER_JOINS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_TRANSLATION</TD>
+# <TD width=51%>SQL_PROCEDURES</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DDL_INDEX</TD>
+# <TD width=51%>SQL_QUOTED_IDENTIFIER_CASE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DROP_ASSERTION</TD>
+# <TD width=51%>SQL_SCHEMA_USAGE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DROP_CHARACTER_SET</TD>
+# <TD width=51%>SQL_SPECIAL_CHARACTERS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DROP_COLLATION</TD>
+# <TD width=51%>SQL_SUBQUERIES</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DROP_DOMAIN</TD>
+# <TD width=51%>SQL_UNION</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DROP_SCHEMA</TD>
+# <TD width=51%>&nbsp;</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>SQL Limits</B></P>
+#
+# <P>The following values of the <I>InfoType</I> argument return information about the limits applied to identifiers and clauses in SQL statements, such as the maximum lengths of identifiers and the maximum number of columns in a select list. Limitations can be imposed by either the driver or the data source.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_BINARY_LITERAL_LEN</TD>
+# <TD width=51%>SQL_MAX_IDENTIFIER_LEN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_CATALOG_NAME_LEN</TD>
+# <TD width=51%>SQL_MAX_INDEX_SIZE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_CHAR_LITERAL_LEN</TD>
+# <TD width=51%>SQL_MAX_PROCEDURE_NAME_LEN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_COLUMN_NAME_LEN</TD>
+# <TD width=51%>SQL_MAX_ROW_SIZE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_COLUMNS_IN_GROUP_BY</TD>
+# <TD width=51%>SQL_MAX_ROW_SIZE_INCLUDES_LONG</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_COLUMNS_IN_INDEX</TD>
+# <TD width=51%>SQL_MAX_SCHEMA_NAME_LEN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_COLUMNS_IN_ORDER_BY</TD>
+# <TD width=51%>SQL_MAX_STATEMENT_LEN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_COLUMNS_IN_SELECT</TD>
+# <TD width=51%>SQL_MAX_TABLE_NAME_LEN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_COLUMNS_IN_TABLE</TD>
+# <TD width=51%>SQL_MAX_TABLES_IN_SELECT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_MAX_CURSOR_NAME_LEN</TD>
+# <TD width=51%>SQL_MAX_USER_NAME_LEN</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Scalar Function Information</B></P>
+#
+# <P>The following values of the <I>InfoType</I> argument return information about the scalar functions supported by the data source and the driver. For more information about scalar functions, see <A HREF="odbcscalar_functions.htm">Appendix E: Scalar Functions</A>.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CONVERT_FUNCTIONS</TD>
+# <TD width=51%>SQL_TIMEDATE_ADD_INTERVALS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_NUMERIC_FUNCTIONS</TD>
+# <TD width=51%>SQL_TIMEDATE_DIFF_INTERVALS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_STRING_FUNCTIONS</TD>
+# <TD width=51%>SQL_TIMEDATE_FUNCTIONS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_SYSTEM_FUNCTIONS</TD>
+# <TD width=51%>&nbsp;</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Conversion Information</B></P>
+#
+# <P>The following values of the <I>InfoType</I> argument return a list of the SQL data types to which the data source can convert the specified SQL data type with the <B>CONVERT</B> scalar function:</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_BIGINT</TD>
+# <TD width=47%>SQL_CONVERT_LONGVARBINARY</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_BINARY</TD>
+# <TD width=47%>SQL_CONVERT_LONGVARCHAR</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_BIT</TD>
+# <TD width=47%>SQL_CONVERT_NUMERIC</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_CHAR</TD>
+# <TD width=47%>SQL_CONVERT_REAL</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_DATE</TD>
+# <TD width=47%>SQL_CONVERT_SMALLINT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_DECIMAL</TD>
+# <TD width=47%>SQL_CONVERT_TIME</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_DOUBLE</TD>
+# <TD width=47%>SQL_CONVERT_TIMESTAMP</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_FLOAT</TD>
+# <TD width=47%>SQL_CONVERT_TINYINT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_INTEGER</TD>
+# <TD width=47%>SQL_CONVERT_VARBINARY</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_INTERVAL_YEAR_MONTH</TD>
+# <TD width=47%>SQL_CONVERT_VARCHAR</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=53%>SQL_CONVERT_INTERVAL_DAY_TIME</TD>
+# <TD width=47%>&nbsp;</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Information Types Added for ODBC 3<I>.x</I></B></P>
+#
+# <P>The following values of the <I>InfoType</I> argument have been added for ODBC 3<I>.x</I>:</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ACTIVE_ENVIRONMENTS</TD>
+# <TD width=51%>SQL_DROP_ASSERTION</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_AGGREGATE_FUNCTIONS</TD>
+# <TD width=51%>SQL_DROP_CHARACTER_SET</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ALTER_DOMAIN</TD>
+# <TD width=51%>SQL_DROP_COLLATION</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ALTER_SCHEMA</TD>
+# <TD width=51%>SQL_DROP_DOMAIN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ANSI_SQL_DATETIME_LITERALS</TD>
+# <TD width=51%>SQL_DROP_SCHEMA</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_ASYNC_MODE</TD>
+# <TD width=51%>SQL_DROP_TABLE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_BATCH_ROW_COUNT</TD>
+# <TD width=51%>SQL_DROP_TRANSLATION</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_BATCH_SUPPORT</TD>
+# <TD width=51%>SQL_DROP_VIEW</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CATALOG_NAME</TD>
+# <TD width=51%>SQL_DYNAMIC_CURSOR_ATTRIBUTES1</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_COLLATION_SEQ</TD>
+# <TD width=51%>SQL_DYNAMIC_CURSOR_ATTRIBUTES2</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CONVERT_INTERVAL_YEAR_MONTH</TD>
+# <TD width=51%>SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES1</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CONVERT_INTERVAL_DAY_TIME</TD>
+# <TD width=51%>SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES2</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_ASSERTION</TD>
+# <TD width=51%>SQL_INFO_SCHEMA_VIEWS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_CHARACTER_SET</TD>
+# <TD width=51%>SQL_INSERT_STATEMENT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_COLLATION</TD>
+# <TD width=51%>SQL_KEYSET_CURSOR_ATTRIBUTES1</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_DOMAIN</TD>
+# <TD width=51%>SQL_KEYSET_CURSOR_ATTRIBUTES2</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_SCHEMA</TD>
+# <TD width=51%>SQL_MAX_ASYNC_CONCURRENT_STATEMENTS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_TABLE</TD>
+# <TD width=51%>SQL_MAX_IDENTIFIER_LEN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CREATE_TRANSLATION</TD>
+# <TD width=51%>SQL_PARAM_ARRAY_ROW_COUNTS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_CURSOR_SENSITIVITY</TD>
+# <TD width=51%>SQL_PARAM_ARRAY_SELECTS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DDL_INDEX</TD>
+# <TD width=51%>SQL_STATIC_CURSOR_ATTRIBUTES1</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DESCRIBE_PARAMETER</TD>
+# <TD width=51%>SQL_STATIC_CURSOR_ATTRIBUTES2</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DM_VER</TD>
+# <TD width=51%>SQL_XOPEN_CLI_YEAR</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=49%>SQL_DRIVER_HDESC</TD>
+# <TD width=51%>&nbsp;</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Information Types Renamed for ODBC 3<I>.x</I></B></P>
+#
+# <P>The following values of the <I>InfoType</I> argument have been renamed for ODBC 3<I>.x</I>.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=55%>ODBC 2.0 <I>InfoType</I></TH>
+# <TH width=45%>ODBC 3<I>.x</I> <I>InfoType</I></TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_ACTIVE_CONNECTIONS</TD>
+# <TD width=45%>SQL_MAX_DRIVER_CONNECTIONS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_ACTIVE_STATEMENTS</TD>
+# <TD width=45%>SQL_MAX_CONCURRENT_ACTIVITIES</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_MAX_OWNER_NAME_LEN</TD>
+# <TD width=45%>SQL_MAX_SCHEMA_NAME_LEN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_MAX_QUALIFIER_NAME_LEN</TD>
+# <TD width=45%>SQL_MAX_CATALOG_NAME_LEN</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_ODBC_SQL_OPT_IEF</TD>
+# <TD width=45%>SQL_INTEGRITY</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_OWNER_TERM</TD>
+# <TD width=45%>SQL_SCHEMA_TERM</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_OWNER_USAGE</TD>
+# <TD width=45%>SQL_SCHEMA_USAGE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_QUALIFIER_LOCATION</TD>
+# <TD width=45%>SQL_CATALOG_LOCATION</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_QUALIFIER_NAME_SEPARATOR</TD>
+# <TD width=45%>SQL_CATALOG_NAME_SEPARATOR</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_QUALIFIER_TERM</TD>
+# <TD width=45%>SQL_CATALOG_TERM</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=55%>SQL_QUALIFIER_USAGE</TD>
+# <TD width=45%>SQL_CATALOG_USAGE</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Information Types Deprecated in ODBC 3<I>.x</I></B></P>
+#
+# <P>The following values of the <I>InfoType</I> argument have been deprecated in ODBC 3<I>.x</I>. ODBC 3<I>.x</I> drivers must continue to support these information types for backward compatibility with ODBC 2<I>.x</I> applications. (For more information on these types, see "<A HREF="odbcsqlgetinfo_support.htm">SQLGetInfo Support</A>" in Appendix G: Driver Guidelines for Backward Compatibility.)</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TD width=51%>SQL_FETCH_DIRECTION</TD>
+# <TD width=49%>SQL_POS_OPERATIONS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=51%>SQL_LOCK_TYPES</TD>
+# <TD width=49%>SQL_POSITIONED_STATEMENTS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=51%>SQL_ODBC_API_CONFORMANCE</TD>
+# <TD width=49%>SQL_SCROLL_CONCURRENCY</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=51%>SQL_ODBC_SQL_CONFORMANCE</TD>
+# <TD width=49%>SQL_STATIC_SENSITIVITY</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <H2>Information Type Descriptions</H2>
+#
+# <P>The following table alphabetically lists each information type, the version of ODBC in which it was introduced, and its description.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=50%><I>InfoType</I></TH>
+# <TH width=50%>Returns</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ACCESSIBLE_PROCEDURES<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string: "Y" if the user can execute all procedures returned by <B>SQLProcedures</B>; "N" if there may be procedures returned that the user cannot execute.</TD>
+# </TR>
+ SQL_ACCESSIBLE_PROCEDURES => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ACCESSIBLE_TABLES<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string: "Y" if the user is guaranteed <B>SELECT</B> privileges to all tables returned by <B>SQLTables</B>; "N" if there may be tables returned that the user cannot access.</TD>
+# </TR>
+ SQL_ACCESSIBLE_TABLES => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ACTIVE_ENVIRONMENTS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum number of active environments that the driver can support. If there is no specified limit or the limit is unknown, this value is set to zero.</TD>
+# </TR>
+ SQL_ACTIVE_ENVIRONMENTS => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_AGGREGATE_FUNCTIONS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating support for aggregation functions:
+# <P>SQL_AF_ALL<BR>
+# SQL_AF_AVG<BR>
+# SQL_AF_COUNT<BR>
+# SQL_AF_DISTINCT<BR>
+# SQL_AF_MAX<BR>
+# SQL_AF_MIN<BR>
+# SQL_AF_SUM </P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_AGGREGATE_FUNCTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ALTER_DOMAIN<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>ALTER DOMAIN</B> statement, as defined in SQL-92, supported by the data source. An SQL-92 Full level&#0150;compliant driver will always return all of the bitmasks. A return value of "0" means that the <B>ALTER DOMAIN</B> statement is not supported.
+# <P>The SQL-92 or FIPS conformance level at which this feature needs to be supported is shown in parentheses next to each bitmask.</P>
+#
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_AD_ADD_DOMAIN_CONSTRAINT = Adding a domain constraint is supported (Full level)</P>
+#
+# <P>SQL_AD_ADD_DOMAIN_DEFAULT = &lt;alter domain&gt; &lt;set domain default clause&gt; is supported (Full level)</P>
+#
+# <P>SQL_AD_CONSTRAINT_NAME_DEFINITION = &lt;constraint name definition clause&gt; is supported for naming domain constraint (Intermediate level)</P>
+#
+# <P>SQL_AD_DROP_DOMAIN_CONSTRAINT = &lt;drop domain constraint clause&gt; is supported (Full level)</P>
+#
+# <P>SQL_AD_DROP_DOMAIN_DEFAULT = &lt;alter domain&gt; &lt;drop domain default clause&gt; is supported (Full level)</P>
+#
+# <P>The following bits specify the supported &lt;constraint attributes&gt; if &lt;add domain constraint&gt; is supported (the SQL_AD_ADD_DOMAIN_CONSTRAINT bit is set):</P>
+#
+# <P>SQL_AD_ADD_CONSTRAINT_DEFERRABLE (Full level)<BR>
+# SQL_AD_ADD_CONSTRAINT_NON_DEFERRABLE (Full level)<BR>
+# SQL_AD_ADD_CONSTRAINT_INITIALLY_DEFERRED (Full level)<BR>
+# SQL_AD_ADD_CONSTRAINT_INITIALLY_IMMEDIATE (Full level)</P>
+# </TD>
+# </TR>
+ SQL_ALTER_DOMAIN => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ALTER_TABLE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>ALTER TABLE</B> statement supported by the data source.
+# <P>The SQL-92 or FIPS conformance level at which this feature needs to be supported is shown in parentheses next to each bitmask.</P>
+#
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_AT_ADD_COLUMN_COLLATION = &lt;add column&gt; clause is supported, with facility to specify column collation (Full level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_ADD_COLUMN_DEFAULT = &lt;add column&gt; clause is supported, with facility to specify column defaults (FIPS Transitional level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_ADD_COLUMN_SINGLE = &lt;add column&gt; is supported (FIPS Transitional level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_ADD_CONSTRAINT = &lt;add column&gt; clause is supported, with facility to specify column constraints (FIPS Transitional level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_ADD_TABLE_CONSTRAINT = &lt;add table constraint&gt; clause is supported (FIPS Transitional level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_CONSTRAINT_NAME_DEFINITION = &lt;constraint name definition&gt; is supported for naming column and table constraints (Intermediate level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_DROP_COLUMN_CASCADE = &lt;drop column&gt; CASCADE is supported (FIPS Transitional level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_DROP_COLUMN_DEFAULT = &lt;alter column&gt; &lt;drop column default clause&gt; is supported (Intermediate level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_DROP_COLUMN_RESTRICT = &lt;drop column&gt; RESTRICT is supported (FIPS Transitional level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_DROP_TABLE_CONSTRAINT_CASCADE (ODBC 3.0)</P>
+#
+# <P>SQL_AT_DROP_TABLE_CONSTRAINT_RESTRICT = &lt;drop column&gt; RESTRICT is supported (FIPS Transitional level) (ODBC 3.0)</P>
+#
+# <P>SQL_AT_SET_COLUMN_DEFAULT = &lt;alter column&gt; &lt;set column default clause&gt; is supported (Intermediate level) (ODBC 3.0)</P>
+#
+# <P>The following bits specify the support &lt;constraint attributes&gt; if specifying column or table constraints is supported (the SQL_AT_ADD_CONSTRAINT bit is set):</P>
+#
+# <P>SQL_AT_CONSTRAINT_INITIALLY_DEFERRED (Full level) (ODBC 3.0)<BR>
+# SQL_AT_CONSTRAINT_INITIALLY_IMMEDIATE (Full level) (ODBC 3.0)<BR>
+# SQL_AT_CONSTRAINT_DEFERRABLE (Full level) (ODBC 3.0)<BR>
+# SQL_AT_CONSTRAINT_NON_DEFERRABLE (Full level) (ODBC 3.0)</P>
+# </TD>
+# </TR>
+ SQL_ALTER_TABLE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ASYNC_MODE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER value indicating the level of asynchronous support in the driver:
+# <P>SQL_AM_CONNECTION = Connection level asynchronous execution is supported. Either all statement handles associated with a given connection handle are in asynchronous mode or all are in synchronous mode. A statement handle on a connection cannot be in asynchronous mode while another statement handle on the same connection is in synchronous mode, and vice versa.</P>
+#
+# <P>SQL_AM_STATEMENT = Statement level asynchronous execution is supported. Some statement handles associated with a connection handle can be in asynchronous mode, while other statement handles on the same connection are in synchronous mode.</P>
+#
+# <P>SQL_AM_NONE = Asynchronous mode is not supported.</P>
+# </TD>
+# </TR>
+ SQL_ASYNC_MODE => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_BATCH_ROW_COUNT <BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the behavior of the driver with respect to the availability of row counts. The following bitmasks are used in conjunction with the information type:
+# <P>SQL_BRC_ROLLED_UP = Row counts for consecutive INSERT, DELETE, or UPDATE statements are rolled up into one. If this bit is not set, then row counts are available for each individual statement.</P>
+#
+# <P>SQL_BRC_PROCEDURES = Row counts, if any, are available when a batch is executed in a stored procedure. If row counts are available, they can be rolled up or individually available, depending on the SQL_BRC_ROLLED_UP bit.</P>
+#
+# <P>SQL_BRC_EXPLICIT = Row counts, if any, are available when a batch is executed directly by calling <B>SQLExecute</B> or <B>SQLExecDirect</B>. If row counts are available, they can be rolled up or individually available, depending on the SQL_BRC_ROLLED_UP bit.</P>
+# </TD>
+# </TR>
+ SQL_BATCH_ROW_COUNT => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_BATCH_SUPPORT <BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the driver's support for batches. The following bitmasks are used to determine which level is supported:
+# <P>SQL_BS_SELECT_EXPLICIT = The driver supports explicit batches that can have result-set generating statements.</P>
+#
+# <P>SQL_BS_ROW_COUNT_EXPLICIT = The driver supports explicit batches that can have row-count generating statements.</P>
+#
+# <P>SQL_BS_SELECT_PROC = The driver supports explicit procedures that can have result-set generating statements.</P>
+#
+# <P>SQL_BS_ROW_COUNT_PROC = The driver supports explicit procedures that can have row-count generating statements.</P>
+# </TD>
+# </TR>
+ SQL_BATCH_SUPPORT => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_BOOKMARK_PERSISTENCE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the operations through which bookmarks persist.
+# <P>The following bitmasks are used in conjunction with the flag to determine through which options bookmarks persist:</P>
+#
+# <P>SQL_BP_CLOSE = Bookmarks are valid after an application calls <B>SQLFreeStmt</B> with the SQL_CLOSE option, or <B>SQLCloseCursor</B> to close the cursor associated with a statement.</P>
+#
+# <P>SQL_BP_DELETE = The bookmark for a row is valid after that row has been deleted.</P>
+#
+# <P>SQL_BP_DROP = Bookmarks are valid after an application calls <B>SQLFreeHandle</B> with a <I>HandleType</I> of SQL_HANDLE_STMT to drop a statement.</P>
+#
+# <P>SQL_BP_TRANSACTION = Bookmarks are valid after an application commits or rolls back a transaction.</P>
+#
+# <P>SQL_BP_UPDATE = The bookmark for a row is valid after any column in that row has been updated, including key columns.</P>
+#
+# <P>SQL_BP_OTHER_HSTMT = A bookmark associated with one statement can be used with another statement. Unless SQL_BP_CLOSE or SQL_BP_DROP is specified, the cursor on the first statement must be open.</P>
+# </TD>
+# </TR>
+ SQL_BOOKMARK_PERSISTENCE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CATALOG_LOCATION<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value indicating the position of the catalog in a qualified table name:
+# <P>SQL_CL_START<BR>
+# SQL_CL_END</P>
+#
+# <P>For example, an Xbase driver returns SQL_CL_START because the directory (catalog) name is at the start of the table name, as in \EMPDATA\EMP.DBF. An ORACLE Server driver returns SQL_CL_END because the catalog is at the end of the table name, as in ADMIN.EMP@EMPDATA.</P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return SQL_CL_START. A value of 0 is returned if catalogs are not supported by the data source. To find out whether catalogs are supported, an application calls <B>SQLGetInfo</B> with the SQL_CATALOG_NAME information type.</P>
+#
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_QUALIFIER_LOCATION.</P>
+# </TD>
+# </TR>
+ SQL_CATALOG_LOCATION => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CATALOG_NAME<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>A character string: "Y" if the server supports catalog names, or "N" if it does not.
+# <P>An SQL-92 Full level&#0150;conformant driver will always return "Y".</P>
+# </TD>
+# </TR>
+ SQL_CATALOG_NAME => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CATALOG_NAME_SEPARATOR<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string: the character or characters that the data source defines as the separator between a catalog name and the qualified name element that follows or precedes it.
+# <P>An empty string is returned if catalogs are not supported by the data source. To find out whether catalogs are supported, an application calls <B>SQLGetInfo</B> with the SQL_CATALOG_NAME information type. An SQL-92 Full level&#0150;conformant driver will always return ".".</P>
+#
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_QUALIFIER_NAME_SEPARATOR.</P>
+# </TD>
+# </TR>
+ SQL_CATALOG_NAME_SEPARATOR => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CATALOG_TERM<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the data source vendor's name for a catalog; for example, "database" or "directory". This string can be in upper, lower, or mixed case.
+# <P>An empty string is returned if catalogs are not supported by the data source. To find out whether catalogs are supported, an application calls <B>SQLGetInfo</B> with the SQL_CATALOG_NAME information type. An SQL-92 Full level&#0150;conformant driver will always return "catalog".</P>
+#
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_QUALIFIER_TERM.</P>
+# </TD>
+# </TR>
+ SQL_CATALOG_TERM => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CATALOG_USAGE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the statements in which catalogs can be used.
+# <P>The following bitmasks are used to determine where catalogs can be used:</P>
+#
+# <P>SQL_CU_DML_STATEMENTS = Catalogs are supported in all Data Manipulation Language statements: <B>SELECT</B>, <B>INSERT</B>, <B>UPDATE</B>, <B>DELETE</B>, and if supported, <B>SELECT FOR UPDATE</B> and positioned update and delete statements.</P>
+#
+# <P>SQL_CU_PROCEDURE_INVOCATION = Catalogs are supported in the ODBC procedure invocation statement.</P>
+#
+# <P>SQL_CU_TABLE_DEFINITION = Catalogs are supported in all table definition statements: <B>CREATE TABLE</B>, <B>CREATE VIEW</B>, <B>ALTER TABLE</B>, <B>DROP TABLE</B>, and <B>DROP VIEW</B>.</P>
+#
+# <P>SQL_CU_INDEX_DEFINITION = Catalogs are supported in all index definition statements: <B>CREATE INDEX</B> and <B>DROP INDEX</B>.</P>
+#
+# <P>SQL_CU_PRIVILEGE_DEFINITION = Catalogs are supported in all privilege definition statements: <B>GRANT</B> and <B>REVOKE</B>. </P>
+#
+# <P>A value of 0 is returned if catalogs are not supported by the data source. To find out whether catalogs are supported, an application calls <B>SQLGetInfo</B> with the SQL_CATALOG_NAME information type. An SQL-92 Full level&#0150;conformant driver will always return a bitmask with all of these bits set.</P>
+#
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_QUALIFIER_USAGE.</P>
+# </TD>
+# </TR>
+ SQL_CATALOG_USAGE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_COLLATION_SEQ<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>The name of the collation sequence. This is a character string that indicates the name of the default collation for the default character set for this server (for example, 'ISO 8859-1' or EBCDIC). If this is unknown, an empty string will be returned. An SQL-92 Full level&#0150;conformant driver will always return a non-empty string.</TD>
+# </TR>
+ SQL_COLLATION_SEQ => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_COLUMN_ALIAS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>A character string: "Y" if the data source supports column aliases; otherwise, "N".
+# <P>A column alias is an alternate name that can be specified for a column in the select list by using an AS clause. An SQL-92 Entry level&#0150;conformant driver will always return "Y".</P>
+# </TD>
+# </TR>
+ SQL_COLUMN_ALIAS => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CONCAT_NULL_BEHAVIOR<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value indicating how the data source handles the concatenation of NULL valued character data type columns with non-NULL valued character data type columns:
+# <P>SQL_CB_NULL = Result is NULL valued.</P>
+#
+# <P>SQL_CB_NON_NULL = Result is concatenation of non-NULL valued column or columns. </P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return SQL_CB_NULL.</P>
+# </TD>
+# </TR>
+ SQL_CONCAT_NULL_BEHAVIOR => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CONVERT_BIGINT<BR>
+ SQL_CONVERT_BIGINT => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_BINARY<BR>
+ SQL_CONVERT_BINARY => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_BIT <BR>
+ SQL_CONVERT_BIT => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_CHAR <BR>
+ SQL_CONVERT_CHAR => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_GUID<BR>
+ SQL_CONVERT_GUID => {
+ type => q(Bitmask),
+ omit => 1,
+ },
+# SQL_CONVERT_DATE<BR>
+ SQL_CONVERT_DATE => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_DECIMAL<BR>
+ SQL_CONVERT_DECIMAL => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_DOUBLE<BR>
+ SQL_CONVERT_DOUBLE => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_FLOAT<BR>
+ SQL_CONVERT_FLOAT => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_INTEGER<BR>
+ SQL_CONVERT_INTEGER => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_INTERVAL_YEAR_MONTH<BR>
+ SQL_CONVERT_INTERVAL_YEAR_MONTH => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_INTERVAL_DAY_TIME<BR>
+ SQL_CONVERT_INTERVAL_DAY_TIME => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_LONGVARBINARY<BR>
+ SQL_CONVERT_LONGVARBINARY => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_LONGVARCHAR<BR>
+ SQL_CONVERT_LONGVARCHAR => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_NUMERIC<BR>
+ SQL_CONVERT_NUMERIC => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_REAL<BR>
+ SQL_CONVERT_REAL => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_SMALLINT<BR>
+ SQL_CONVERT_SMALLINT => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_TIME<BR>
+ SQL_CONVERT_TIME => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_TIMESTAMP<BR>
+ SQL_CONVERT_TIMESTAMP => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_TINYINT<BR>
+ SQL_CONVERT_TINYINT => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_VARBINARY<BR>
+ SQL_CONVERT_VARBINARY => {
+ type => q(Bitmask),
+ },
+# SQL_CONVERT_VARCHAR <BR>
+ SQL_CONVERT_VARCHAR => {
+ type => q(Bitmask),
+ },
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask. The bitmask indicates the conversions supported by the data source with the <B>CONVERT</B> scalar function for data of the type named in the <I>InfoType</I>. If the bitmask equals zero, the data source does not support any conversions from data of the named type, including conversion to the same data type.
+# <P>For example, to find out if a data source supports the conversion of SQL_INTEGER data to the SQL_BIGINT data type, an application calls <B>SQLGetInfo</B> with the <I>InfoType</I> of SQL_CONVERT_INTEGER. The application performs an <B>AND</B> operation with the returned bitmask and SQL_CVT_BIGINT. If the resulting value is nonzero, the conversion is supported. </P>
+#
+# <P>The following bitmasks are used to determine which conversions are supported:</P>
+#
+# <P>SQL_CVT_BIGINT (ODBC 1.0)<BR>
+# SQL_CVT_BINARY (ODBC 1.0)<BR>
+# SQL_CVT_BIT (ODBC 1.0) <BR>
+# SQL_CVT_GUID (ODBC 3.5)<BR>
+# SQL_CVT_CHAR (ODBC 1.0) <BR>
+# SQL_CVT_DATE (ODBC 1.0)<BR>
+# SQL_CVT_DECIMAL (ODBC 1.0)<BR>
+# SQL_CVT_DOUBLE (ODBC 1.0)<BR>
+# SQL_CVT_FLOAT (ODBC 1.0)<BR>
+# SQL_CVT_INTEGER (ODBC 1.0)<BR>
+# SQL_CVT_INTERVAL_YEAR_MONTH (ODBC 3.0)<BR>
+# SQL_CVT_INTERVAL_DAY_TIME (ODBC 3.0)<BR>
+# SQL_CVT_LONGVARBINARY (ODBC 1.0)<BR>
+# SQL_CVT_LONGVARCHAR (ODBC 1.0)<BR>
+# SQL_CVT_NUMERIC (ODBC 1.0)<BR>
+# SQL_CVT_REAL ODBC 1.0)<BR>
+# SQL_CVT_SMALLINT (ODBC 1.0)<BR>
+# SQL_CVT_TIME (ODBC 1.0)<BR>
+# SQL_CVT_TIMESTAMP (ODBC 1.0)<BR>
+# SQL_CVT_TINYINT (ODBC 1.0)<BR>
+# SQL_CVT_VARBINARY (ODBC 1.0)<BR>
+# SQL_CVT_VARCHAR (ODBC 1.0)</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CONVERT_FUNCTIONS<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the scalar conversion functions supported by the driver and associated data source.
+# <P>The following bitmask is used to determine which conversion functions are supported:</P>
+#
+# <P>SQL_FN_CVT_CAST<BR>
+# SQL_FN_CVT_CONVERT</P>
+# </TD>
+# </TR>
+ SQL_CONVERT_FUNCTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CORRELATION_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value indicating whether table correlation names are supported:
+# <P>SQL_CN_NONE = Correlation names are not supported.</P>
+#
+# <P>SQL_CN_DIFFERENT = Correlation names are supported but must differ from the names of the tables they represent.</P>
+#
+# <P>SQL_CN_ANY = Correlation names are supported and can be any valid user-defined name. </P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return SQL_CN_ANY.</P>
+# </TD>
+# </TR>
+ SQL_CORRELATION_NAME => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CREATE_ASSERTION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>CREATE ASSERTION</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_CA_CREATE_ASSERTION</P>
+#
+# <P>The following bits specify the supported constraint attribute if the ability to specify constraint attributes explicitly is supported (see the SQL_ALTER_TABLE and SQL_CREATE_TABLE information types):</P>
+#
+# <P>SQL_CA_CONSTRAINT_INITIALLY_DEFERRED<BR>
+# SQL_CA_CONSTRAINT_INITIALLY_IMMEDIATE<BR>
+# SQL_CA_CONSTRAINT_DEFERRABLE<BR>
+# SQL_CA_CONSTRAINT_NON_DEFERRABLE</P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return all of these options as supported. A return value of "0" means that the <B>CREATE ASSERTION</B> statement is not supported.</P>
+# </TD>
+# </TR>
+ SQL_CREATE_ASSERTION => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CREATE_CHARACTER_SET<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>CREATE CHARACTER SET</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_CCS_CREATE_CHARACTER_SET<BR>
+# SQL_CCS_COLLATE_CLAUSE<BR>
+# SQL_CCS_LIMITED_COLLATION</P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return all of these options as supported. A return value of "0" means that the <B>CREATE CHARACTER SET</B> statement is not supported.</P>
+# </TD>
+# </TR>
+ SQL_CREATE_CHARACTER_SET => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CREATE_COLLATION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>CREATE COLLATION</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmask is used to determine which clauses are supported:</P>
+#
+# <P>SQL_CCOL_CREATE_COLLATION </P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return this option as supported. A return value of "0" means that the <B>CREATE COLLATION</B> statement is not supported.</P>
+# </TD>
+# </TR>
+ SQL_CREATE_COLLATION => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CREATE_DOMAIN<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>CREATE DOMAIN</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_CDO_CREATE_DOMAIN = The CREATE DOMAIN statement is supported (Intermediate level).</P>
+#
+# <P>SQL_CDO_CONSTRAINT_NAME_DEFINITION = &lt;constraint name definition&gt; is supported for naming domain constraints (Intermediate level).</P>
+#
+# <P>The following bits specify the ability to create column constraints:<BR>
+# SQL_CDO_DEFAULT = Specifying domain constraints is supported (Intermediate level)<BR>
+# SQL_CDO_CONSTRAINT = Specifying domain defaults is supported (Intermediate level)<BR>
+# SQL_CDO_COLLATION = Specifying domain collation is supported (Full level)</P>
+#
+# <P>The following bits specify the supported constraint attributes if specifying domain constraints is supported (SQL_CDO_DEFAULT is set):</P>
+#
+# <P>SQL_CDO_CONSTRAINT_INITIALLY_DEFERRED (Full level)<BR>
+# SQL_CDO_CONSTRAINT_INITIALLY_IMMEDIATE (Full level)<BR>
+# SQL_CDO_CONSTRAINT_DEFERRABLE (Full level)<BR>
+# SQL_CDO_CONSTRAINT_NON_DEFERRABLE (Full level)</P>
+#
+# <P>A return value of "0" means that the <B>CREATE DOMAIN</B> statement is not supported.</P>
+# </TD>
+# </TR>
+ SQL_CREATE_DOMAIN => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CREATE_SCHEMA<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>CREATE SCHEMA</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_CS_CREATE_SCHEMA<BR>
+# SQL_CS_AUTHORIZATION<BR>
+# SQL_CS_DEFAULT_CHARACTER_SET </P>
+#
+# <P>An SQL-92 Intermediate level&#0150;conformant driver will always return the SQL_CS_CREATE_SCHEMA and SQL_CS_AUTHORIZATION options as supported. These must also be supported at the SQL-92 Entry level, but not necessarily as SQL statements. An SQL-92 Full level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_CREATE_SCHEMA => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CREATE_TABLE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>CREATE TABLE</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The SQL-92 or FIPS conformance level at which this feature needs to be supported is shown in parentheses next to each bitmask.</P>
+#
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_CT_CREATE_TABLE = The CREATE TABLE statement is supported. (Entry level)</P>
+#
+# <P>SQL_CT_TABLE_CONSTRAINT = Specifying table constraints is supported (FIPS Transitional level)</P>
+#
+# <P>SQL_CT_CONSTRAINT_NAME_DEFINITION = The &lt;constraint name definition&gt; clause is supported for naming column and table constraints (Intermediate level)</P>
+#
+# <P>The following bits specify the ability to create temporary tables:</P>
+#
+# <P>SQL_CT_COMMIT_PRESERVE = Deleted rows are preserved on commit. (Full level)<BR>
+# SQL_CT_COMMIT_DELETE = Deleted rows are deleted on commit. (Full level)<BR>
+# SQL_CT_GLOBAL_TEMPORARY = Global temporary tables can be created. (Full level)<BR>
+# SQL_CT_LOCAL_TEMPORARY = Local temporary tables can be created. (Full level)</P>
+#
+# <P>The following bits specify the ability to create column constraints:</P>
+#
+# <P>SQL_CT_COLUMN_CONSTRAINT = Specifying column constraints is supported (FIPS Transitional level)<BR>
+# SQL_CT_COLUMN_DEFAULT = Specifying column defaults is supported (FIPS Transitional level)<BR>
+# SQL_CT_COLUMN_COLLATION = Specifying column collation is supported (Full level)</P>
+#
+# <P>The following bits specify the supported constraint attributes if specifying column or table constraints is supported:</P>
+#
+# <P>SQL_CT_CONSTRAINT_INITIALLY_DEFERRED (Full level)<BR>
+# SQL_CT_CONSTRAINT_INITIALLY_IMMEDIATE (Full level)<BR>
+# SQL_CT_CONSTRAINT_DEFERRABLE (Full level)<BR>
+# SQL_CT_CONSTRAINT_NON_DEFERRABLE (Full level)</P>
+# </TD>
+# </TR>
+ SQL_CREATE_TABLE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CREATE_TRANSLATION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>CREATE TRANSLATION</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmask is used to determine which clauses are supported:</P>
+#
+# <P>SQL_CTR_CREATE_TRANSLATION</P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return these options as supported. A return value of "0" means that the <B>CREATE TRANSLATION</B> statement is not supported.</P>
+# </TD>
+# </TR>
+ SQL_CREATE_TRANSLATION => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CREATE_VIEW<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>CREATE VIEW</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_CV_CREATE_VIEW<BR>
+# SQL_CV_CHECK_OPTION<BR>
+# SQL_CV_CASCADED<BR>
+# SQL_CV_LOCAL </P>
+#
+# <P>A return value of "0" means that the <B>CREATE VIEW</B> statement is not supported.</P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return the SQL_CV_CREATE_VIEW and SQL_CV_CHECK_OPTION options as supported. </P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_CREATE_VIEW => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CURSOR_COMMIT_BEHAVIOR<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value indicating how a <B>COMMIT</B> operation affects cursors and prepared statements in the data source:
+# <P>SQL_CB_DELETE = Close cursors and delete prepared statements. To use the cursor again, the application must reprepare and reexecute the statement.</P>
+#
+# <P>SQL_CB_CLOSE = Close cursors. For prepared statements, the application can call <B>SQLExecute</B> on the statement without calling <B>SQLPrepare</B> again.</P>
+#
+# <P>SQL_CB_PRESERVE = Preserve cursors in the same position as before the <B>COMMIT</B> operation. The application can continue to fetch data, or it can close the cursor and reexecute the statement without repreparing it.</P>
+# </TD>
+# </TR>
+ SQL_CURSOR_COMMIT_BEHAVIOR => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CURSOR_ROLLBACK_BEHAVIOR<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value indicating how a <B>ROLLBACK</B> operation affects cursors and prepared statements in the data source:
+# <P>SQL_CB_DELETE = Close cursors and delete prepared statements. To use the cursor again, the application must reprepare and reexecute the statement.</P>
+#
+# <P>SQL_CB_CLOSE = Close cursors. For prepared statements, the application can call <B>SQLExecute</B> on the statement without calling <B>SQLPrepare</B> again.</P>
+#
+# <P>SQL_CB_PRESERVE = Preserve cursors in the same position as before the <B>ROLLBACK</B> operation. The application can continue to fetch data, or it can close the cursor and reexecute the statement without repreparing it.</P>
+# </TD>
+# </TR>
+ SQL_CURSOR_ROLLBACK_BEHAVIOR => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_CURSOR_ROLLBACK_SQL_CURSOR_SENSITIVITY<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER value indicating the support for cursor sensitivity:
+# <P>SQL_INSENSITIVE = All cursors on the statement handle show the result set without reflecting any changes made to it by any other cursor within the same transaction.</P>
+#
+# <P>SQL_UNSPECIFIED = It is unspecified whether cursors on the statement handle make visible the changes made to a result set by another cursor within the same transaction. Cursors on the statement handle may make visible none, some, or all such changes.</P>
+#
+# <P>SQL_SENSITIVE = Cursors are sensitive to changes made by other cursors within the same transaction.</P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return the SQL_UNSPECIFIED option as supported. </P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return the SQL_INSENSITIVE option as supported.</P>
+# </TD>
+# </TR>
+ SQL_CURSOR_SENSITIVITY => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DATA_SOURCE_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the data source name used during connection. If the application called <B>SQLConnect</B>, this is the value of the <I>szDSN</I> argument. If the application called <B>SQLDriverConnect</B> or <B>SQLBrowseConnect</B>, this is the value of the DSN keyword in the connection string passed to the driver. If the connection string did not contain the <B>DSN</B> keyword (such as when it contains the <B>DRIVER</B> keyword), this is an empty string.</TD>
+# </TR>
+ SQL_DATA_SOURCE_NAME => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DATA_SOURCE_READ_ONLY<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string. "Y" if the data source is set to READ ONLY mode, "N" if it is otherwise.
+# <P>This characteristic pertains only to the data source itself; it is not a characteristic of the driver that enables access to the data source. A driver that is read/write can be used with a data source that is read-only. If a driver is read-only, all of its data sources must be read-only and must return SQL_DATA_SOURCE_READ_ONLY.</P>
+# </TD>
+# </TR>
+ SQL_DATA_SOURCE_READ_ONLY => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DATABASE_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the name of the current database in use, if the data source defines a named object called "database".
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;In ODBC 3<I>.x</I>, the value returned for this <I>InfoType</I> can also be returned by calling <B>SQLGetConnectAttr</B> with an <I>Attribute</I> argument of SQL_ATTR_CURRENT_CATALOG.</P>
+# </TD>
+# </TR>
+ SQL_DATABASE_NAME => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DATETIME_LITERALS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the SQL-92 datetime literals supported by the data source. Note that these are the datetime literals listed in the SQL-92 specification and are separate from the datetime literal escape clauses defined by ODBC. For more information about the ODBC datetime literal escape clauses, see "Date, Time, Timestamp, and Datetime Interval Literals" in Chapter 8: SQL Statements.
+# <P>A FIPS Transitional level&#0150;conformant driver will always return the "1" value in the bitmask for the bits listed below. A value of "0" means that SQL-92 datetime literals are not supported.</P>
+#
+# <P>The following bitmasks are used to determine which literals are supported:</P>
+#
+# <P>SQL_DL_SQL92_DATE<BR>
+# SQL_DL_SQL92_TIME<BR>
+# SQL_DL_SQL92_TIMESTAMP<BR>
+# SQL_DL_SQL92_INTERVAL_YEAR<BR>
+# SQL_DL_SQL92_INTERVAL_MONTH<BR>
+# SQL_DL_SQL92_INTERVAL_DAY<BR>
+# SQL_DL_SQL92_INTERVAL_HOUR<BR>
+# SQL_DL_SQL92_INTERVAL_MINUTE<BR>
+# SQL_DL_SQL92_INTERVAL_SECOND<BR>
+# SQL_DL_SQL92_INTERVAL_YEAR_TO_MONTH<BR>
+# SQL_DL_SQL92_INTERVAL_DAY_TO_HOUR</P>
+#
+# <P>SQL_DL_SQL92_INTERVAL_DAY_TO_MINUTE<BR>
+# SQL_DL_SQL92_INTERVAL_DAY_TO_SECOND<BR>
+# SQL_DL_SQL92_INTERVAL_HOUR_TO_MINUTE<BR>
+# SQL_DL_SQL92_INTERVAL_HOUR_TO_SECOND<BR>
+# SQL_DL_SQL92_INTERVAL_MINUTE_TO_SECOND</P>
+# </TD>
+# </TR>
+ SQL_DATETIME_LITERALS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DBMS_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the name of the DBMS product accessed by the driver.</TD>
+# </TR>
+ SQL_DBMS_NAME => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DBMS_VER<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string indicating the version of the DBMS product accessed by the driver. The version is of the form ##.##.####, where the first two digits are the major version, the next two digits are the minor version, and the last four digits are the release version. The driver must render the DBMS product version in this form but can also append the DBMS product-specific version as well. For example, "04.01.0000 Rdb 4.1".</TD>
+# </TR>
+ SQL_DBMS_VER => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DDL_INDEX<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER value that indicates support for creation and dropping of indexes:
+# <P>SQL_DI_CREATE_INDEX<BR>
+# SQL_DI_DROP_INDEX </P>
+# </TD>
+# </TR>
+ SQL_DDL_INDEX => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DEFAULT_TXN_ISOLATION<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUINTEGER value that indicates the default transaction isolation level supported by the driver or data source, or zero if the data source does not support transactions. The following terms are used to define transaction isolation levels:
+# <P><B>Dirty Read </B>Transaction 1 changes a row. Transaction 2 reads the changed row before transaction 1 commits the change. If transaction 1 rolls back the change, transaction 2 will have read a row that is considered to have never existed.</P>
+#
+# <P><B>Nonrepeatable Read </B>Transaction 1 reads a row. Transaction 2 updates or deletes that row and commits this change. If transaction 1 attempts to reread the row, it will receive different row values or discover that the row has been deleted.</P>
+#
+# <P><B>Phantom </B>Transaction 1 reads a set of rows that satisfy some search criteria. Transaction 2 generates one or more rows (through either inserts or updates) that match the search criteria. If transaction 1 reexecutes the statement that reads the rows, it receives a different set of rows.</P>
+#
+# <P>If the data source supports transactions, the driver returns one of the following bitmasks:</P>
+#
+# <P>SQL_TXN_READ_UNCOMMITTED = Dirty reads, nonrepeatable reads, and phantoms are possible.</P>
+#
+# <P>SQL_TXN_READ_COMMITTED = Dirty reads are not possible. Nonrepeatable reads and phantoms are possible.</P>
+#
+# <P>SQL_TXN_REPEATABLE_READ = Dirty reads and nonrepeatable reads are not possible. Phantoms are possible.</P>
+#
+# <P>SQL_TXN_SERIALIZABLE = Transactions are serializable. Serializable transactions do not allow dirty reads, nonrepeatable reads, or phantoms.</P>
+# </TD>
+# </TR>
+ SQL_DEFAULT_TXN_ISOLATION => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DESCRIBE_PARAMETER<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>A character string: "Y" if parameters can be described; "N", if not.
+# <P>An SQL-92 Full level&#0150;conformant driver will usually return "Y" because it will support the <B>DESCRIBE INPUT</B> statement. Because this does not directly specify the underlying SQL support, however, describing parameters might not be supported, even in a SQL-92 Full level&#0150;conformant driver.</P>
+# </TD>
+# </TR>
+ SQL_DESCRIBE_PARAMETER => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DM_VER<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>A character string with the version of the Driver Manager. The version is of the form ##.##.####.####, where:
+# <P>The first set of two digits is the major ODBC version, as given by the constant SQL_SPEC_MAJOR.</P>
+#
+# <P>The second set of two digits is the minor ODBC version, as given by the constant SQL_SPEC_MINOR.</P>
+#
+# <P>The third set of four digits is the Driver Manager major build number.</P>
+#
+# <P>The last set of four digits is the Driver Manager minor build number.</P>
+# </TD>
+# </TR>
+ SQL_DM_VER => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DRIVER_HDBC<BR>
+# SQL_DRIVER_HENV<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUINTEGER value, the driver's environment handle or connection handle, determined by the argument <I>InfoType</I>.
+# <P>These information types are implemented by the Driver Manager alone.</P>
+# </TD>
+# </TR>
+ SQL_DRIVER_HDBC => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DRIVER_HDESC<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER value, the driver's descriptor handle determined by the Driver Manager's descriptor handle, which must be passed on input in *<I>InfoValuePtr</I> from the application. In this case, <I>InfoValuePtr</I> is both an input and output argument. The input descriptor handle passed in *<I>InfoValuePtr</I> must have been either explicitly or implicitly allocated on the <I>ConnectionHandle</I>.
+# <P>The application should make a copy of the Driver Manager's descriptor handle before calling <B>SQLGetInfo</B> with this information type, to ensure that the handle is not overwritten on output.</P>
+#
+# <P>This information type is implemented by the Driver Manager alone.</P>
+# </TD>
+# </TR>
+ SQL_DRIVER_HDESC => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DRIVER_HLIB<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER value, the <I>hinst</I> from the load library returned to the Driver Manager when it loaded the driver DLL (on a Microsoft&reg; Windows&reg; platform) or equivalent on a non-Windows platform. The handle is valid only for the connection handle specified in the call to <B>SQLGetInfo</B>.
+# <P>This information type is implemented by the Driver Manager alone.</P>
+# </TD>
+# </TR>
+ SQL_DRIVER_HLIB => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DRIVER_HSTMT<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUINTEGER value, the driver's statement handle determined by the Driver Manager statement handle, which must be passed on input in *<I>InfoValuePtr</I> from the application. In this case, <I>InfoValuePtr</I> is both an input and an output argument. The input statement handle passed in *<I>InfoValuePtr</I> must have been allocated on the argument <I>ConnectionHandle</I>.
+# <P>The application should make a copy of the Driver Manager's statement handle before calling <B>SQLGetInfo</B> with this information type, to ensure that the handle is not overwritten on output.</P>
+#
+# <P>This information type is implemented by the Driver Manager alone.</P>
+# </TD>
+# </TR>
+ SQL_DRIVER_HSTMT => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DRIVER_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the file name of the driver used to access the data source.</TD>
+# </TR>
+ SQL_DRIVER_NAME => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DRIVER_ODBC_VER<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>A character string with the version of ODBC that the driver supports. The version is of the form ##.##, where the first two digits are the major version and the next two digits are the minor version. SQL_SPEC_MAJOR and SQL_SPEC_MINOR define the major and minor version numbers. For the version of ODBC described in this manual, these are 3 and 0, and the driver should return "03.00".</TD>
+# </TR>
+ SQL_DRIVER_ODBC_VER => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DRIVER_VER<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the version of the driver and optionally, a description of the driver. At a minimum, the version is of the form ##.##.####, where the first two digits are the major version, the next two digits are the minor version, and the last four digits are the release version.</TD>
+# </TR>
+ SQL_DRIVER_VER => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DROP_ASSERTION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>DROP ASSERTION</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmask is used to determine which clauses are supported:</P>
+#
+# <P>SQL_DA_DROP_ASSERTION </P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return this option as supported.</P>
+# </TD>
+# </TR>
+ SQL_DROP_ASSERTION => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DROP_CHARACTER_SET<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>DROP CHARACTER SET</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmask is used to determine which clauses are supported:</P>
+#
+# <P>SQL_DCS_DROP_CHARACTER_SET </P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return this option as supported.</P>
+# </TD>
+# </TR>
+ SQL_DROP_CHARACTER_SET => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DROP_COLLATION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>DROP COLLATION</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmask is used to determine which clauses are supported:</P>
+#
+# <P>SQL_DC_DROP_COLLATION </P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return this option as supported.</P>
+# </TD>
+# </TR>
+ SQL_DROP_COLLATION => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DROP_DOMAIN<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>DROP DOMAIN</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_DD_DROP_DOMAIN<BR>
+# SQL_DD_CASCADE<BR>
+# SQL_DD_RESTRICT </P>
+#
+# <P>An SQL-92 Intermediate level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_DROP_DOMAIN => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DROP_SCHEMA<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>DROP SCHEMA</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_DS_DROP_SCHEMA<BR>
+# SQL_DS_CASCADE<BR>
+# SQL_DS_RESTRICT </P>
+#
+# <P>An SQL-92 Intermediate level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_DROP_SCHEMA => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DROP_TABLE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>DROP TABLE</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_DT_DROP_TABLE<BR>
+# SQL_DT_CASCADE<BR>
+# SQL_DT_RESTRICT </P>
+#
+# <P>An FIPS Transitional level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_DROP_TABLE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DROP_TRANSLATION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>DROP TRANSLATION</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmask is used to determine which clauses are supported:</P>
+#
+# <P>SQL_DTR_DROP_TRANSLATION </P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return this option as supported.</P>
+# </TD>
+# </TR>
+ SQL_DROP_TRANSLATION => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DROP_VIEW<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses in the <B>DROP VIEW</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The following bitmasks are used to determine which clauses are supported:</P>
+#
+# <P>SQL_DV_DROP_VIEW<BR>
+# SQL_DV_CASCADE<BR>
+# SQL_DV_RESTRICT </P>
+#
+# <P>An FIPS Transitional level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_DROP_VIEW => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DYNAMIC_CURSOR_ATTRIBUTES1<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that describes the attributes of a dynamic cursor that are supported by the driver. This bitmask contains the first subset of attributes; for the second subset, see SQL_DYNAMIC_CURSOR_ATTRIBUTES2.
+# <P>The following bitmasks are used to determine which attributes are supported:</P>
+#
+# <P>SQL_CA1_NEXT = A <I>FetchOrientation</I> argument of SQL_FETCH_NEXT is supported in a call to <B>SQLFetchScroll</B> when the cursor is a dynamic cursor.</P>
+#
+# <P>SQL_CA1_ABSOLUTE = <I>FetchOrientation</I> arguments of SQL_FETCH_FIRST, SQL_FETCH_LAST, and SQL_FETCH_ABSOLUTE are supported in a call to <B>SQLFetchScroll</B> when the cursor is a dynamic cursor. (The rowset that will be fetched is independent of the current cursor position.) </P>
+#
+# <P>SQL_CA1_RELATIVE = <I>FetchOrientation</I> arguments of SQL_FETCH_PRIOR and SQL_FETCH_RELATIVE are supported in a call to <B>SQLFetchScroll</B> when the cursor is a dynamic cursor. (The rowset that will be fetched is dependent on the current cursor position. Note that this is separated from SQL_FETCH_NEXT because in a forward-only cursor, only SQL_FETCH_NEXT is supported.) </P>
+#
+# <P>SQL_CA1_BOOKMARK = A <I>FetchOrientation</I> argument of SQL_FETCH_BOOKMARK is supported in a call to <B>SQLFetchScroll</B> when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA1_LOCK_EXCLUSIVE = A <I>LockType</I> argument of SQL_LOCK_EXCLUSIVE is supported in a call to <B>SQLSetPos</B> when the cursor is a dynamic cursor.</P>
+#
+# <P>SQL_CA1_LOCK_NO_CHANGE = A <I>LockType</I> argument of SQL_LOCK_NO_CHANGE is supported in a call to <B>SQLSetPos</B> when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA1_LOCK_UNLOCK = A <I>LockType</I> argument of SQL_LOCK_UNLOCK is supported in a call to <B>SQLSetPos</B> when the cursor is a dynamic cursor.</P>
+#
+# <P>SQL_CA1_POS_POSITION = An <I>Operation</I> argument of SQL_POSITION is supported in a call to <B>SQLSetPos</B> when the cursor is a dynamic cursor.</P>
+#
+# <P>SQL_CA1_POS_UPDATE = An <I>Operation</I> argument of SQL_UPDATE is supported in a call to <B>SQLSetPos</B> when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA1_POS_DELETE = An <I>Operation</I> argument of SQL_DELETE is supported in a call to <B>SQLSetPos</B> when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA1_POS_REFRESH = An <I>Operation</I> argument of SQL_REFRESH is supported in a call to <B>SQLSetPos</B> when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA1_POSITIONED_UPDATE = An UPDATE WHERE CURRENT OF SQL statement is supported when the cursor is a dynamic cursor. (An SQL-92 Entry level&#0150;conformant driver will always return this option as supported.)</P>
+#
+# <P>SQL_CA1_POSITIONED_DELETE = A DELETE WHERE CURRENT OF SQL statement is supported when the cursor is a dynamic cursor. (An SQL-92 Entry level&#0150;conformant driver will always return this option as supported.)</P>
+#
+# <P>SQL_CA1_SELECT_FOR_UPDATE = A SELECT FOR UPDATE SQL statement is supported when the cursor is a dynamic cursor. (An SQL-92 Entry level&#0150;conformant driver will always return this option as supported.)</P>
+#
+# <P>SQL_CA1_BULK_ADD = An <I>Operation</I> argument of SQL_ADD is supported in a call to <B>SQLBulkOperations</B> when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA1_BULK_UPDATE_BY_BOOKMARK = An <I>Operation</I> argument of SQL_UPDATE_BY_BOOKMARK is supported in a call to <B>SQLBulkOperations</B> when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA1_BULK_DELETE_BY_BOOKMARK = An <I>Operation</I> argument of SQL_DELETE_BY_BOOKMARK is supported in a call to <B>SQLBulkOperations</B> when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA1_BULK_FETCH_BY_BOOKMARK = An <I>Operation</I> argument of SQL_FETCH_BY_BOOKMARK is supported in a call to <B>SQLBulkOperations</B> when the cursor is a dynamic cursor. </P>
+#
+# <P>An SQL-92 Intermediate level&#0150;conformant driver will usually return the SQL_CA1_NEXT, SQL_CA1_ABSOLUTE, and SQL_CA1_RELATIVE options as supported, because it supports scrollable cursors through the embedded SQL FETCH statement. Because this does not directly determine the underlying SQL support, however, scrollable cursors may not be supported, even for an SQL-92 Intermediate level&#0150;conformant driver.</P>
+# </TD>
+# </TR>
+ SQL_DYNAMIC_CURSOR_ATTRIBUTES1 => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_DYNAMIC_CURSOR_ATTRIBUTES2<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that describes the attributes of a dynamic cursor that are supported by the driver. This bitmask contains the second subset of attributes; for the first subset, see SQL_DYNAMIC_CURSOR_ATTRIBUTES1.
+# <P>The following bitmasks are used to determine which attributes are supported:</P>
+#
+# <P>SQL_CA2_READ_ONLY_CONCURRENCY = A read-only dynamic cursor, in which no updates are allowed, is supported. (The SQL_ATTR_CONCURRENCY statement attribute can be SQL_CONCUR_READ_ONLY for a dynamic cursor). </P>
+#
+# <P>SQL_CA2_LOCK_CONCURRENCY = A dynamic cursor that uses the lowest level of locking sufficient to ensure that the row can be updated is supported. (The SQL_ATTR_CONCURRENCY statement attribute can be SQL_CONCUR_LOCK for a dynamic cursor.) These locks must be consistent with the transaction isolation level set by the SQL_ATTR_TXN_ISOLATION connection attribute.</P>
+#
+# <P>SQL_CA2_OPT_ROWVER_CONCURRENCY = A dynamic cursor that uses the optimistic concurrency control comparing row versions is supported. (The SQL_ATTR_CONCURRENCY statement attribute can be SQL_CONCUR_ROWVER for a dynamic cursor.) </P>
+#
+# <P>SQL_CA2_OPT_VALUES_CONCURRENCY = A dynamic cursor that uses the optimistic concurrency control comparing values is supported. (The SQL_ATTR_CONCURRENCY statement attribute can be SQL_CONCUR_VALUES for a dynamic cursor.) </P>
+#
+# <P>SQL_CA2_SENSITIVITY_ADDITIONS = Added rows are visible to a dynamic cursor; the cursor can scroll to those rows. (Where these rows are added to the cursor is driver-dependent.) </P>
+#
+# <P>SQL_CA2_SENSITIVITY_DELETIONS = Deleted rows are no longer available to a dynamic cursor, and do not leave a "hole" in the result set; after the dynamic cursor scrolls from a deleted row, it cannot return to that row. </P>
+#
+# <P>SQL_CA2_SENSITIVITY_UPDATES = Updates to rows are visible to a dynamic cursor; if the dynamic cursor scrolls from and returns to an updated row, the data returned by the cursor is the updated data, not the original data. </P>
+#
+# <P>SQL_CA2_MAX_ROWS_SELECT = The SQL_ATTR_MAX_ROWS statement attribute affects <B>SELECT</B> statements when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA2_MAX_ROWS_INSERT = The SQL_ATTR_MAX_ROWS statement attribute affects <B>INSERT</B> statements when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA2_MAX_ROWS_DELETE = The SQL_ATTR_MAX_ROWS statement attribute affects <B>DELETE</B> statements when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA2_MAX_ROWS_UPDATE = The SQL_ATTR_MAX_ROWS statement attribute affects <B>UPDATE</B> statements when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA2_MAX_ROWS_CATALOG = The SQL_ATTR_MAX_ROWS statement attribute affects <B>CATALOG</B> result sets when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA2_MAX_ROWS_AFFECTS_ALL = The SQL_ATTR_MAX_ROWS statement attribute affects <B>SELECT</B>, <B>INSERT</B>, <B>DELETE</B>, and <B>UPDATE</B> statements, and <B>CATALOG</B> result sets, when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA2_CRC_EXACT = The exact row count is available in the SQL_DIAG_CURSOR_ROW_COUNT diagnostic field when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA2_CRC_APPROXIMATE = An approximate row count is available in the SQL_DIAG_CURSOR_ROW_COUNT diagnostic field when the cursor is a dynamic cursor. </P>
+#
+# <P>SQL_CA2_SIMULATE_NON_UNIQUE = The driver does not guarantee that simulated positioned update or delete statements will affect only one row when the cursor is a dynamic cursor; it is the application's responsibility to guarantee this. (If a statement affects more than one row, <B>SQLExecute</B> or <B>SQLExecDirect</B> returns SQLSTATE 01001 [Cursor operation conflict].) To set this behavior, the application calls <B>SQLSetStmtAttr</B> with the SQL_ATTR_SIMULATE_CURSOR attribute set to SQL_SC_NON_UNIQUE. </P>
+#
+# <P>SQL_CA2_SIMULATE_TRY_UNIQUE = The driver attempts to guarantee that simulated positioned update or delete statements will affect only one row when the cursor is a dynamic cursor. The driver always executes such statements, even if they might affect more than one row, such as when there is no unique key. (If a statement affects more than one row, <B>SQLExecute</B> or <B>SQLExecDirect</B> returns SQLSTATE 01001 [Cursor operation conflict].) To set this behavior, the application calls <B>SQLSetStmtAttr</B> with the SQL_ATTR_SIMULATE_CURSOR attribute set to SQL_SC_TRY_UNIQUE. </P>
+#
+# <P>SQL_CA2_SIMULATE_UNIQUE = The driver guarantees that simulated positioned update or delete statements will affect only one row when the cursor is a dynamic cursor. If the driver cannot guarantee this for a given statement, <B>SQLExecDirect</B> or <B>SQLPrepare</B> return SQLSTATE 01001 (Cursor operation conflict). To set this behavior, the application calls <B>SQLSetStmtAttr</B> with the SQL_ATTR_SIMULATE_CURSOR attribute set to SQL_SC_UNIQUE.</P>
+# </TD>
+# </TR>
+ SQL_DYNAMIC_CURSOR_ATTRIBUTES2 => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_EXPRESSIONS_IN_ORDERBY<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string: "Y" if the data source supports expressions in the <B>ORDER BY</B> list; "N" if it does not.</TD>
+# </TR>
+ SQL_EXPRESSIONS_IN_ORDERBY => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_FILE_USAGE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value indicating how a single-tier driver directly treats files in a data source:
+# <P>SQL_FILE_NOT_SUPPORTED = The driver is not a single-tier driver. For example, an ORACLE driver is a two-tier driver.</P>
+#
+# <P>SQL_FILE_TABLE = A single-tier driver treats files in a data source as tables. For example, an Xbase driver treats each Xbase file as a table.</P>
+#
+# <P>SQL_FILE_CATALOG = A single-tier driver treats files in a data source as a catalog. For example, a Microsoft&reg; Access driver treats each Microsoft Access file as a complete database.</P>
+#
+# <P>An application might use this to determine how users will select data. For example, Xbase users often think of data as stored in files, while ORACLE and MicrosoftAccess users generally think of data as stored in tables.</P>
+#
+# <P>When a user selects an Xbase data source, the application could display the Windows <B>File Open</B> common dialog box; when the user selects a Microsoft Access or ORACLE data source, the application could display a custom <B>Select Table</B> dialog box.</P>
+# </TD>
+# </TR>
+ SQL_FILE_USAGE => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES1<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that describes the attributes of a forward-only cursor that are supported by the driver. This bitmask contains the first subset of attributes; for the second subset, see SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES2.
+# <P>The following bitmasks are used to determine which attributes are supported:</P>
+#
+# <P>SQL_CA1_NEXT<BR>
+# SQL_CA1_LOCK_EXCLUSIVE<BR>
+# SQL_CA1_LOCK_NO_CHANGE<BR>
+# SQL_CA1_LOCK_UNLOCK<BR>
+# SQL_CA1_POS_POSITION<BR>
+# SQL_CA1_POS_UPDATE<BR>
+# SQL_CA1_POS_DELETE<BR>
+# SQL_CA1_POS_REFRESH<BR>
+# SQL_CA1_POSITIONED_UPDATE<BR>
+# SQL_CA1_POSITIONED_DELETE<BR>
+# SQL_CA1_SELECT_FOR_UPDATE<BR>
+# SQL_CA1_BULK_ADD<BR>
+# SQL_CA1_BULK_UPDATE_BY_BOOKMARK<BR>
+# SQL_CA1_BULK_DELETE_BY_BOOKMARK<BR>
+# SQL_CA1_BULK_FETCH_BY_BOOKMARK</P>
+#
+# <P>For descriptions of these bitmasks, see SQL_DYNAMIC_CURSOR_ATTRIBUTES1 (and substitute "forward-only cursor" for "dynamic cursor" in the descriptions). </P>
+# </TD>
+# </TR>
+ SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES1 => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES2<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that describes the attributes of a forward-only cursor that are supported by the driver. This bitmask contains the second subset of attributes; for the first subset, see SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES1.
+# <P>The following bitmasks are used to determine which attributes are supported:</P>
+#
+# <P>SQL_CA2_READ_ONLY_CONCURRENCY<BR>
+# SQL_CA2_LOCK_CONCURRENCY<BR>
+# SQL_CA2_OPT_ROWVER_CONCURRENCY<BR>
+# SQL_CA2_OPT_VALUES_CONCURRENCY<BR>
+# SQL_CA2_SENSITIVITY_ADDITIONS<BR>
+# SQL_CA2_SENSITIVITY_DELETIONS<BR>
+# SQL_CA2_SENSITIVITY_UPDATES<BR>
+# SQL_CA2_MAX_ROWS_SELECT<BR>
+# SQL_CA2_MAX_ROWS_INSERT<BR>
+# SQL_CA2_MAX_ROWS_DELETE<BR>
+# SQL_CA2_MAX_ROWS_UPDATE<BR>
+# SQL_CA2_MAX_ROWS_CATALOG<BR>
+# SQL_CA2_MAX_ROWS_AFFECTS_ALL<BR>
+# SQL_CA2_CRC_EXACT<BR>
+# SQL_CA2_CRC_APPROXIMATE<BR>
+# SQL_CA2_SIMULATE_NON_UNIQUE<BR>
+# SQL_CA2_SIMULATE_TRY_UNIQUE<BR>
+# SQL_CA2_SIMULATE_UNIQUE </P>
+#
+# <P>For descriptions of these bitmasks, see SQL_DYNAMIC_CURSOR_ATTRIBUTES2 (and substitute "forward-only cursor" for "dynamic cursor" in the descriptions).</P>
+# </TD>
+# </TR>
+ SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES2 => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_GETDATA_EXTENSIONS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating extensions to <B>SQLGetData</B>.
+# <P>The following bitmasks are used in conjunction with the flag to determine what common extensions the driver supports for <B>SQLGetData</B>:</P>
+#
+# <P>SQL_GD_ANY_COLUMN = <B>SQLGetData</B> can be called for any unbound column, including those before the last bound column. Note that the columns must be called in order of ascending column number unless SQL_GD_ANY_ORDER is also returned.</P>
+#
+# <P>SQL_GD_ANY_ORDER = <B>SQLGetData</B> can be called for unbound columns in any order. Note that <B>SQLGetData</B> can be called only for columns after the last bound column unless SQL_GD_ANY_COLUMN is also returned.</P>
+#
+# <P>SQL_GD_BLOCK = <B>SQLGetData</B> can be called for an unbound column in any row in a block (where the rowset size is greater than 1) of data after positioning to that row with <B>SQLSetPos</B>.</P>
+#
+# <P>SQL_GD_BOUND = <B>SQLGetData</B> can be called for bound columns as well as unbound columns. A driver cannot return this value unless it also returns SQL_GD_ANY_COLUMN.</P>
+#
+# <P><B>SQLGetData</B> is required to return data only from unbound columns that occur after the last bound column, are called in order of increasing column number, and are not in a row in a block of rows.</P>
+#
+# <P>If a driver supports bookmarks (either fixed-length or variable-length), it must support calling <B>SQLGetData</B> on column 0. This support is required regardless of what the driver returns for a call to <B>SQLGetInfo</B> with the SQL_GETDATA_EXTENSIONS <I>InfoType</I>.</P>
+# </TD>
+# </TR>
+ SQL_GETDATA_EXTENSIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_GROUP_BY<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the relationship between the columns in the <B>GROUP BY</B> clause and the nonaggregated columns in the select list:
+# <P>SQL_GB_COLLATE = A <B>COLLATE</B> clause can be specified at the end of each grouping column. (ODBC 3.0)</P>
+#
+# <P>SQL_GB_NOT_SUPPORTED = <B>GROUP BY</B> clauses are not supported. (ODBC 2.0)</P>
+#
+# <P>SQL_GB_GROUP_BY_EQUALS_SELECT = The <B>GROUP BY</B> clause must contain all nonaggregated columns in the select list. It cannot contain any other columns. For example, <B>SELECT DEPT, MAX(SALARY) FROM EMPLOYEE GROUP BY DEPT</B>. (ODBC 2.0)</P>
+#
+# <P>SQL_GB_GROUP_BY_CONTAINS_SELECT = The <B>GROUP BY</B> clause must contain all nonaggregated columns in the select list. It can contain columns that are not in the select list. For example, <B>SELECT DEPT, MAX(SALARY) FROM EMPLOYEE GROUP BY DEPT, AGE</B>. (ODBC 2.0)</P>
+#
+# <P>SQL_GB_NO_RELATION = The columns in the <B>GROUP BY</B> clause and the select list are not related. The meaning of nongrouped, nonaggregated columns in the select list is data source&#0150;dependent. For example, <B>SELECT DEPT, SALARY FROM EMPLOYEE GROUP BY DEPT, AGE</B>. (ODBC 2.0)</P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return the SQL_GB_GROUP_BY_EQUALS_SELECT option as supported. An SQL-92 Full level&#0150;conformant driver will always return the SQL_GB_COLLATE option as supported. If none of the options is supported, the <B>GROUP BY</B> clause is not supported by the data source.</P>
+# </TD>
+# </TR>
+ SQL_GROUP_BY => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_IDENTIFIER_CASE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value as follows:
+# <P>SQL_IC_UPPER = Identifiers in SQL are not case-sensitive and are stored in uppercase in system catalog.</P>
+#
+# <P>SQL_IC_LOWER = Identifiers in SQL are not case-sensitive and are stored in lowercase in system catalog.</P>
+#
+# <P>SQL_IC_SENSITIVE = Identifiers in SQL are case-sensitive and are stored in mixed case in system catalog.</P>
+#
+# <P>SQL_IC_MIXED = Identifiers in SQL are not case-sensitive and are stored in mixed case in system catalog. </P>
+#
+# <P>Because identifiers in SQL-92 are never case-sensitive, a driver that conforms strictly to SQL-92 (any level) will never return the SQL_IC_SENSITIVE option as supported.</P>
+# </TD>
+# </TR>
+ SQL_IDENTIFIER_CASE => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_IDENTIFIER_QUOTE_CHAR<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>The character string used as the starting and ending delimiter of a quoted (delimited) identifier in SQL statements. (Identifiers passed as arguments to ODBC functions do not need to be quoted.) If the data source does not support quoted identifiers, a blank is returned.
+# <P>This character string can also be used for quoting catalog function arguments when the connection attribute SQL_ATTR_METADATA_ID is set to SQL_TRUE.</P>
+#
+# <P>Because the identifier quote character in SQL-92 is the double quotation mark ("), a driver that conforms strictly to SQL-92 will always return the double quotation mark character.</P>
+# </TD>
+# </TR>
+ SQL_IDENTIFIER_QUOTE_CHAR => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INDEX_KEYWORDS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that enumerates keywords in the CREATE INDEX statement that are supported by the driver:
+# <P>SQL_IK_NONE = None of the keywords is supported.</P>
+#
+# <P>SQL_IK_ASC = ASC keyword is supported.</P>
+#
+# <P>SQL_IK_DESC = DESC keyword is supported.</P>
+#
+# <P>SQL_IK_ALL = All keywords are supported.</P>
+#
+# <P>To see if the CREATE INDEX statement is supported, an application calls <B>SQLGetInfo</B> with the SQL_DLL_INDEX information type.</P>
+# </TD>
+# </TR>
+ SQL_INDEX_KEYWORDS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INFO_SCHEMA_VIEWS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the views in the INFORMATION_SCHEMA that are supported by the driver. The views in, and the contents of, INFORMATION_SCHEMA are as defined in SQL-92.
+# <P>The SQL-92 or FIPS conformance level at which this feature needs to be supported is shown in parentheses next to each bitmask.</P>
+#
+# <P>The following bitmasks are used to determine which views are supported:</P>
+#
+# <P>SQL_ISV_ASSERTIONS = Identifies the catalog's assertions that are owned by a given user. (Full level)</P>
+#
+# <P>SQL_ISV_CHARACTER_SETS = Identifies the catalog's character sets that are accessible to a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_CHECK_CONSTRAINTS = Identifies the CHECK constraints that are owned by a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_COLLATIONS = Identifies the character collations for the catalog that are accessible to a given user. (Full level)</P>
+#
+# <P>SQL_ISV_COLUMN_DOMAIN_USAGE = Identifies columns for the catalog that are dependent on domains defined in the catalog and are owned by a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_COLUMN_PRIVILEGES = Identifies the privileges on columns of persistent tables that are available to or granted by a given user. (FIPS Transitional level)</P>
+#
+# <P>SQL_ISV_COLUMNS = Identifies the columns of persistent tables that are accessible to a given user. (FIPS Transitional level)</P>
+#
+# <P>SQL_ISV_CONSTRAINT_COLUMN_USAGE = Similar to CONSTRAINT_TABLE_USAGE view, columns are identified for the various constraints that are owned by a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_CONSTRAINT_TABLE_USAGE = Identifies the tables that are used by constraints (referential, unique, and assertions), and are owned by a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_DOMAIN_CONSTRAINTS = Identifies the domain constraints (of the domains in the catalog) that are accessible to a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_DOMAINS = Identifies the domains defined in a catalog that are accessible to the user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_KEY_COLUMN_USAGE = Identifies columns defined in the catalog that are constrained as keys by a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_REFERENTIAL_CONSTRAINTS = Identifies the referential constraints that are owned by a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_SCHEMATA = Identifies the schemas that are owned by a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_SQL_LANGUAGES = Identifies the SQL conformance levels, options, and dialects supported by the SQL implementation. (Intermediate level)</P>
+#
+# <P>SQL_ISV_TABLE_CONSTRAINTS = Identifies the table constraints that are owned by a given user. (Intermediate level)</P>
+#
+# <P>SQL_ISV_TABLE_PRIVILEGES = Identifies the privileges on persistent tables that are available to or granted by a given user. (FIPS Transitional level)</P>
+#
+# <P>SQL_ISV_TABLES = Identifies the persistent tables defined in a catalog that are accessible to a given user. (FIPS Transitional level)</P>
+#
+# <P>SQL_ISV_TRANSLATIONS = Identifies character translations for the catalog that are accessible to a given user. (Full level) </P>
+#
+# <P>SQL_ISV_USAGE_PRIVILEGES = Identifies the USAGE privileges on catalog objects that are available to or owned by a given user. (FIPS Transitional level)</P>
+#
+# <P>SQL_ISV_VIEW_COLUMN_USAGE = Identifies the columns on which the catalog's views that are owned by a given user are dependent. (Intermediate level)</P>
+#
+# <P>SQL_ISV_VIEW_TABLE_USAGE = Identifies the tables on which the catalog's views that are owned by a given user are dependent. (Intermediate level) </P>
+#
+# <P>SQL_ISV_VIEWS = Identifies the viewed tables defined in this catalog that are accessible to a given user. (FIPS Transitional level)</P>
+# </TD>
+# </TR>
+ SQL_INFO_SCHEMA_VIEWS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INSERT_STATEMENT<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that indicates support for <B>INSERT</B> statements:
+# <P>SQL_IS_INSERT_LITERALS</P>
+#
+# <P>SQL_IS_INSERT_SEARCHED</P>
+#
+# <P>SQL_IS_SELECT_INTO </P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_INSERT_STATEMENT => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_INTEGRITY<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string: "Y" if the data source supports the Integrity Enhancement Facility; "N" if it does not.
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_ODBC_SQL_OPT_IEF.</P>
+# </TD>
+# </TR>
+ SQL_INTEGRITY => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_KEYSET_CURSOR_ATTRIBUTES1<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that describes the attributes of a keyset cursor that are supported by the driver. This bitmask contains the first subset of attributes; for the second subset, see SQL_KEYSET_CURSOR_ATTRIBUTES2.
+# <P>The following bitmasks are used to determine which attributes are supported: </P>
+#
+# <P>SQL_CA1_NEXT<BR>
+# SQL_CA1_ABSOLUTE<BR>
+# SQL_CA1_RELATIVE<BR>
+# SQL_CA1_BOOKMARK<BR>
+# SQL_CA1_LOCK_EXCLUSIVE<BR>
+# SQL_CA1_LOCK_NO_CHANGE<BR>
+# SQL_CA1_LOCK_UNLOCK<BR>
+# SQL_CA1_POS_POSITION<BR>
+# SQL_CA1_POS_UPDATE<BR>
+# SQL_CA1_POS_DELETE<BR>
+# SQL_CA1_POS_REFRESH<BR>
+# SQL_CA1_POSITIONED_UPDATE<BR>
+# SQL_CA1_POSITIONED_DELETE<BR>
+# SQL_CA1_SELECT_FOR_UPDATE<BR>
+# SQL_CA1_BULK_ADD<BR>
+# SQL_CA1_BULK_UPDATE_BY_BOOKMARK<BR>
+# SQL_CA1_BULK_DELETE_BY_BOOKMARK<BR>
+# SQL_CA1_BULK_FETCH_BY_BOOKMARK</P>
+#
+# <P>For descriptions of these bitmasks, see SQL_DYNAMIC_CURSOR_ATTRIBUTES1 (and substitute "keyset-driven cursor" for "dynamic cursor" in the descriptions).</P>
+#
+# <P>An SQL-92 Intermediate level&#0150;conformant driver will usually return the SQL_CA1_NEXT, SQL_CA1_ABSOLUTE, and SQL_CA1_RELATIVE options as supported, because the driver supports scrollable cursors through the embedded SQL FETCH statement. Because this does not directly determine the underlying SQL support, however, scrollable cursors may not be supported, even for an SQL-92 Intermediate level&#0150;conformant driver.</P>
+# </TD>
+# </TR>
+ SQL_KEYSET_CURSOR_ATTRIBUTES1 => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_KEYSET_CURSOR_ATTRIBUTES2<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that describes the attributes of a keyset cursor that are supported by the driver. This bitmask contains the second subset of attributes; for the first subset, see SQL_KEYSET_CURSOR_ATTRIBUTES1.
+# <P>The following bitmasks are used to determine which attributes are supported:</P>
+#
+# <P>SQL_CA2_READ_ONLY_CONCURRENCY<BR>
+# SQL_CA2_LOCK_CONCURRENCY<BR>
+# SQL_CA2_OPT_ROWVER_CONCURRENCY<BR>
+# SQL_CA2_OPT_VALUES_CONCURRENCY<BR>
+# SQL_CA2_SENSITIVITY_ADDITIONS<BR>
+# SQL_CA2_SENSITIVITY_DELETIONS<BR>
+# SQL_CA2_SENSITIVITY_UPDATES<BR>
+# SQL_CA2_MAX_ROWS_SELECT<BR>
+# SQL_CA2_MAX_ROWS_INSERT<BR>
+# SQL_CA2_MAX_ROWS_DELETE<BR>
+# SQL_CA2_MAX_ROWS_UPDATE<BR>
+# SQL_CA2_MAX_ROWS_CATALOG<BR>
+# SQL_CA2_MAX_ROWS_AFFECTS_ALL<BR>
+# SQL_CA2_CRC_EXACT<BR>
+# SQL_CA2_CRC_APPROXIMATE<BR>
+# SQL_CA2_SIMULATE_NON_UNIQUE<BR>
+# SQL_CA2_SIMULATE_TRY_UNIQUE<BR>
+# SQL_CA2_SIMULATE_UNIQUE</P>
+#
+# <P>For descriptions of these bitmasks, see SQL_DYNAMIC_CURSOR_ATTRIBUTES1 (and substitute "keyset-driven cursor" for "dynamic cursor" in the descriptions).</P>
+# </TD>
+# </TR>
+ SQL_KEYSET_CURSOR_ATTRIBUTES2 => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_KEYWORDS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>A character string containing a comma-separated list of all data source&#0150;specific keywords. This list does not contain keywords specific to ODBC or keywords used by both the data source and ODBC. This list represents all the reserved keywords; interoperable applications should not use these words in object names.
+# <P>For a list of ODBC keywords, see "<A HREF="odbclist_of_reserved_keywords.htm">List of Reserved Keywords</A>" in Appendix C, "SQL Grammar." The <B>#define</B> value SQL_ODBC_KEYWORDS contains a comma-separated list of ODBC keywords.</P>
+# </TD>
+# </TR>
+ SQL_KEYWORDS => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_LIKE_ESCAPE_CLAUSE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>A character string: "Y" if the data source supports an escape character for the percent character (%) and underscore character (_) in a <B>LIKE</B> predicate and the driver supports the ODBC syntax for defining a <B>LIKE</B> predicate escape character; "N" otherwise.</TD>
+# </TR>
+ SQL_LIKE_ESCAPE_CLAUSE => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_ASYNC_CONCURRENT_STATEMENTS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER value specifying the maximum number of active concurrent statements in asynchronous mode that the driver can support on a given connection. If there is no specific limit or the limit is unknown, this value is zero.</TD>
+# </TR>
+ SQL_MAX_ASYNC_CONCURRENT_STATEMENTS => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_BINARY_LITERAL_LEN<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER value specifying the maximum length (number of hexadecimal characters, excluding the literal prefix and suffix returned by <B>SQLGetTypeInfo</B>) of a binary literal in an SQL statement. For example, the binary literal 0xFFAA has a length of 4. If there is no maximum length or the length is unknown, this value is set to zero.</TD>
+# </TR>
+ SQL_MAX_BINARY_LITERAL_LEN => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_CATALOG_NAME_LEN<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum length of a catalog name in the data source. If there is no maximum length or the length is unknown, this value is set to zero.
+# <P>An FIPS Full level&#0150;conformant driver will return at least 128.</P>
+#
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_MAX_QUALIFIER_NAME_LEN.</P>
+# </TD>
+# </TR>
+ SQL_MAX_CATALOG_NAME_LEN => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_CHAR_LITERAL_LEN<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER value specifying the maximum length (number of characters, excluding the literal prefix and suffix returned by <B>SQLGetTypeInfo</B>) of a character literal in an SQL statement. If there is no maximum length or the length is unknown, this value is set to zero.</TD>
+# </TR>
+ SQL_MAX_CHAR_LITERAL_LEN => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_COLUMN_NAME_LEN<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum length of a column name in the data source. If there is no maximum length or the length is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 18. An FIPS Intermediate level&#0150;conformant driver will return at least 128.</P>
+# </TD>
+# </TR>
+ SQL_MAX_COLUMN_NAME_LEN => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_COLUMNS_IN_GROUP_BY<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum number of columns allowed in a <B>GROUP BY</B> clause. If there is no specified limit or the limit is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 6. An FIPS Intermediate level&#0150;conformant driver will return at least 15.</P>
+# </TD>
+# </TR>
+ SQL_MAX_COLUMNS_IN_GROUP_BY => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_COLUMNS_IN_INDEX<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum number of columns allowed in an index. If there is no specified limit or the limit is unknown, this value is set to zero.</TD>
+# </TR>
+ SQL_MAX_COLUMNS_IN_INDEX => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_COLUMNS_IN_ORDER_BY<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum number of columns allowed in an <B>ORDER BY</B> clause. If there is no specified limit or the limit is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 6. An FIPS Intermediate level&#0150;conformant driver will return at least 15.</P>
+# </TD>
+# </TR>
+ SQL_MAX_COLUMNS_IN_ORDER_BY => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_COLUMNS_IN_SELECT<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum number of columns allowed in a select list. If there is no specified limit or the limit is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 100. An FIPS Intermediate level&#0150;conformant driver will return at least 250.</P>
+# </TD>
+# </TR>
+ SQL_MAX_COLUMNS_IN_SELECT => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_COLUMNS_IN_TABLE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum number of columns allowed in a table. If there is no specified limit or the limit is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 100. An FIPS Intermediate level&#0150;conformant driver will return at least 250.</P>
+# </TD>
+# </TR>
+ SQL_MAX_COLUMNS_IN_TABLE => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_CONCURRENT_ACTIVITIES<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum number of active statements that the driver can support for a connection. A statement is defined as active if it has results pending, with the term "results" meaning rows from a <B>SELECT</B> operation or rows affected by an <B>INSERT</B>, <B>UPDATE</B>, or <B>DELETE</B> operation (such as a row count), or if it is in a NEED_DATA state. This value can reflect a limitation imposed by either the driver or the data source. If there is no specified limit or the limit is unknown, this value is set to zero.
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_ACTIVE_STATEMENTS.</P>
+# </TD>
+# </TR>
+ SQL_MAX_CONCURRENT_ACTIVITIES => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_CURSOR_NAME_LEN<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum length of a cursor name in the data source. If there is no maximum length or the length is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 18. An FIPS Intermediate level&#0150;conformant driver will return at least 128.</P>
+# </TD>
+# </TR>
+ SQL_MAX_CURSOR_NAME_LEN => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_DRIVER_CONNECTIONS<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum number of active connections that the driver can support for an environment. This value can reflect a limitation imposed by either the driver or the data source. If there is no specified limit or the limit is unknown, this value is set to zero.
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_ACTIVE_CONNECTIONS.</P>
+# </TD>
+# </TR>
+ SQL_MAX_DRIVER_CONNECTIONS => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_IDENTIFIER_LEN<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUSMALLINT that indicates the maximum size in characters that the data source supports for user-defined names.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 18. An FIPS Intermediate level&#0150;conformant driver will return at least 128.</P>
+# </TD>
+# </TR>
+ SQL_MAX_IDENTIFIER_LEN => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_INDEX_SIZE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER value specifying the maximum number of bytes allowed in the combined fields of an index. If there is no specified limit or the limit is unknown, this value is set to zero.</TD>
+# </TR>
+ SQL_MAX_INDEX_SIZE => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_PROCEDURE_NAME_LEN<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum length of a procedure name in the data source. If there is no maximum length or the length is unknown, this value is set to zero.</TD>
+# </TR>
+ SQL_MAX_PROCEDURE_NAME_LEN => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_ROW_SIZE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER value specifying the maximum length of a single row in a table. If there is no specified limit or the limit is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 2,000. An FIPS Intermediate level&#0150;conformant driver will return at least 8,000.</P>
+# </TD>
+# </TR>
+ SQL_MAX_ROW_SIZE => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_ROW_SIZE_INCLUDES_LONG<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>A character string: "Y" if the maximum row size returned for the SQL_MAX_ROW_SIZE information type includes the length of all SQL_LONGVARCHAR and SQL_LONGVARBINARY columns in the row; "N" otherwise.</TD>
+# </TR>
+ SQL_MAX_ROW_SIZE_INCLUDES_LONG => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_SCHEMA_NAME_LEN<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum length of a schema name in the data source. If there is no maximum length or the length is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 18. An FIPS Intermediate level&#0150;conformant driver will return at least 128.</P>
+#
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_MAX_OWNER_NAME_LEN.</P>
+# </TD>
+# </TR>
+ SQL_MAX_SCHEMA_NAME_LEN => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_STATEMENT_LEN<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER value specifying the maximum length (number of characters, including white space) of an SQL statement. If there is no maximum length or the length is unknown, this value is set to zero.</TD>
+# </TR>
+ SQL_MAX_STATEMENT_LEN => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_TABLE_NAME_LEN<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum length of a table name in the data source. If there is no maximum length or the length is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 18. An FIPS Intermediate level&#0150;conformant driver will return at least 128.</P>
+# </TD>
+# </TR>
+ SQL_MAX_TABLE_NAME_LEN => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_TABLES_IN_SELECT<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum number of tables allowed in the <B>FROM</B> clause of a <B>SELECT </B>statement. If there is no specified limit or the limit is unknown, this value is set to zero.
+# <P>An FIPS Entry level&#0150;conformant driver will return at least 15. An FIPS Intermediate level&#0150;conformant driver will return at least 50.</P>
+# </TD>
+# </TR>
+ SQL_MAX_TABLES_IN_SELECT => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MAX_USER_NAME_LEN<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying the maximum length of a user name in the data source. If there is no maximum length or the length is unknown, this value is set to zero.</TD>
+# </TR>
+ SQL_MAX_USER_NAME_LEN => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MULT_RESULT_SETS<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string: "Y" if the data source supports multiple result sets, "N" if it does not.
+# <P>For more information on multiple result sets, see "<A HREF="odbcmultiple_results.htm">Multiple Results</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+# </TD>
+# </TR>
+ SQL_MULT_RESULT_SETS => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_MULTIPLE_ACTIVE_TXN<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string: "Y" if the driver supports more than one active transaction at the same time, "N" if only one transaction can be active at any time.
+# <P>The information returned for this information type does not apply in the case of distributed transactions.</P>
+# </TD>
+# </TR>
+ SQL_MULTIPLE_ACTIVE_TXN => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_NEED_LONG_DATA_LEN<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>A character string: "Y" if the data source needs the length of a long data value (the data type is SQL_LONGVARCHAR, SQL_LONGVARBINARY, or a long data source&#0150;specific data type) before that value is sent to the data source, "N" if it does not. For more information, see <B>SQLBindParameter</B> and <B>SQLSetPos</B>.</TD>
+# </TR>
+ SQL_NEED_LONG_DATA_LEN => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_NON_NULLABLE_COLUMNS<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying whether the data source supports NOT NULL in columns:
+# <P>SQL_NNC_NULL = All columns must be nullable.</P>
+#
+# <P>SQL_NNC_NON_NULL = Columns cannot be nullable. (The data source supports the <B>NOT NULL</B> column constraint in <B>CREATE TABLE</B> statements.) </P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will return SQL_NNC_NON_NULL.</P>
+# </TD>
+# </TR>
+ SQL_NON_NULLABLE_COLUMNS => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_NULL_COLLATION<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value specifying where NULLs are sorted in a result set:
+# <P>SQL_NC_END = NULLs are sorted at the end of the result set, regardless of the ASC or DESC keywords.</P>
+#
+# <P>SQL_NC_HIGH = NULLs are sorted at the high end of the result set, depending on the ASC or DESC keywords.</P>
+#
+# <P>SQL_NC_LOW = NULLs are sorted at the low end of the result set, depending on the ASC or DESC keywords.</P>
+#
+# <P>SQL_NC_START = NULLs are sorted at the start of the result set, regardless of the ASC or DESC keywords.</P>
+# </TD>
+# </TR>
+ SQL_NULL_COLLATION => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_NUMERIC_FUNCTIONS<BR>
+# (ODBC 1.0)
+# <P>The information type was introduced in ODBC 1.0; each bitmask is labeled with the version in which it was introduced.</P>
+# </TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the scalar numeric functions supported by the driver and associated data source.
+# <P>The following bitmasks are used to determine which numeric functions are supported:</P>
+#
+# <P>SQL_FN_NUM_ABS (ODBC 1.0)<BR>
+# SQL_FN_NUM_ACOS (ODBC 1.0)<BR>
+# SQL_FN_NUM_ASIN (ODBC 1.0)<BR>
+# SQL_FN_NUM_ATAN (ODBC 1.0)<BR>
+# SQL_FN_NUM_ATAN2 (ODBC 1.0)<BR>
+# SQL_FN_NUM_CEILING (ODBC 1.0)<BR>
+# SQL_FN_NUM_COS (ODBC 1.0)<BR>
+# SQL_FN_NUM_COT (ODBC 1.0)<BR>
+# SQL_FN_NUM_DEGREES (ODBC 2.0)<BR>
+# SQL_FN_NUM_EXP (ODBC 1.0)<BR>
+# SQL_FN_NUM_FLOOR (ODBC 1.0)<BR>
+# SQL_FN_NUM_LOG (ODBC 1.0)<BR>
+# SQL_FN_NUM_LOG10 (ODBC 2.0)<BR>
+# SQL_FN_NUM_MOD (ODBC 1.0)<BR>
+# SQL_FN_NUM_PI (ODBC 1.0)<BR>
+# SQL_FN_NUM_POWER (ODBC 2.0)<BR>
+# SQL_FN_NUM_RADIANS (ODBC 2.0)<BR>
+# SQL_FN_NUM_RAND (ODBC 1.0)<BR>
+# SQL_FN_NUM_ROUND (ODBC 2.0)<BR>
+# SQL_FN_NUM_SIGN (ODBC 1.0)<BR>
+# SQL_FN_NUM_SIN (ODBC 1.0)<BR>
+# SQL_FN_NUM_SQRT (ODBC 1.0)<BR>
+# SQL_FN_NUM_TAN (ODBC 1.0)<BR>
+# SQL_FN_NUM_TRUNCATE (ODBC 2.0)</P>
+# </TD>
+# </TR>
+ SQL_NUMERIC_FUNCTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ODBC_INTERFACE_CONFORMANCE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER value indicating the level of the ODBC 3<I>.x</I> interface that the driver conforms to.
+# <P>SQL_OIC_CORE: The minimum level that all ODBC drivers are expected to conform to. This level includes basic interface elements such as connection functions, functions for preparing and executing an SQL statement, basic result set metadata functions, basic catalog functions, and so on.</P>
+#
+# <P>SQL_OIC_LEVEL1: A level including the core standards compliance level functionality, plus scrollable cursors, bookmarks, positioned updates and deletes, and so on.</P>
+#
+# <P>SQL_OIC_LEVEL2: A level including level 1 standards compliance level functionality, plus advanced features such as sensitive cursors; update, delete, and refresh by bookmarks; stored procedure support; catalog functions for primary and foreign keys; multicatalog support; and so on.</P>
+#
+# <P>For more information, see "<A HREF="odbcinterface_conformance_levels.htm">Interface Conformance Levels</A>" in Chapter 4: ODBC Fundamentals.</P>
+# </TD>
+# </TR>
+ SQL_ODBC_INTERFACE_CONFORMANCE => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ODBC_VER<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the version of ODBC to which the Driver Manager conforms. The version is of the form ##.##.0000, where the first two digits are the major version and the next two digits are the minor version. This is implemented solely in the Driver Manager.</TD>
+# </TR>
+ SQL_ODBC_VER => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_OJ_CAPABILITIES<BR>
+# (ODBC 2.01)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the types of outer joins supported by the driver and data source. The following bitmasks are used to determine which types are supported:
+# <P>SQL_OJ_LEFT = Left outer joins are supported.</P>
+#
+# <P>SQL_OJ_RIGHT = Right outer joins are supported.</P>
+#
+# <P>SQL_OJ_FULL = Full outer joins are supported.</P>
+#
+# <P>SQL_OJ_NESTED = Nested outer joins are supported.</P>
+#
+# <P>SQL_OJ_NOT_ORDERED = The column names in the ON clause of the outer join do not have to be in the same order as their respective table names in the <B>OUTER JOIN </B>clause.</P>
+#
+# <P>SQL_OJ_INNER = The inner table (the right table in a left outer join or the left table in a right outer join) can also be used in an inner join. This does not apply to full outer joins, which do not have an inner table.</P>
+#
+# <P>SQL_OJ_ALL_COMPARISON_OPS = The comparison operator in the ON clause can be any of the ODBC comparison operators. If this bit is not set, only the equals (=) comparison operator can be used in outer joins.</P>
+#
+# <P>If none of these options is returned as supported, no outer join clause is supported.</P>
+#
+# <P>For information on the support of relational join operators in a SELECT statement, as defined by SQL-92, see SQL_SQL92_RELATIONAL_JOIN_OPERATORS.</P>
+# </TD>
+# </TR>
+ SQL_OJ_CAPABILITIES => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ORDER_BY_COLUMNS_IN_SELECT<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>A character string: "Y" if the columns in the <B>ORDER BY</B> clause must be in the select list; otherwise, "N".</TD>
+# </TR>
+ SQL_ORDER_BY_COLUMNS_IN_SELECT => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_PARAM_ARRAY_ROW_COUNTS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER enumerating the driver's properties regarding the availability of row counts in a parameterized execution. Has the following values:
+# <P>SQL_PARC_BATCH = Individual row counts are available for each set of parameters. This is conceptually equivalent to the driver generating a batch of SQL statements, one for each parameter set in the array. Extended error information can be retrieved by using the SQL_PARAM_STATUS_PTR descriptor field.</P>
+#
+# <P>SQL_PARC_NO_BATCH = There is only one row count available, which is the cumulative row count resulting from the execution of the statement for the entire array of parameters. This is conceptually equivalent to treating the statement along with the entire parameter array as one atomic unit. Errors are handled the same as if one statement were executed.</P>
+# </TD>
+# </TR>
+ SQL_PARAM_ARRAY_ROW_COUNTS => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_PARAM_ARRAY_SELECTS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER enumerating the driver's properties regarding the availability of result sets in a parameterized execution. Has the following values:
+# <P>SQL_PAS_BATCH = There is one result set available per set of parameters. This is conceptually equivalent to the driver generating a batch of SQL statements, one for each parameter set in the array.</P>
+#
+# <P>SQL_PAS_NO_BATCH = There is only one result set available, which represents the cumulative result set resulting from the execution of the statement for the entire array of parameters. This is conceptually equivalent to treating the statement along with the entire parameter array as one atomic unit.</P>
+#
+# <P>SQL_PAS_NO_SELECT = A driver does not allow a result-set generating statement to be executed with an array of parameters.</P>
+# </TD>
+# </TR>
+ SQL_PARAM_ARRAY_SELECTS => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_PROCEDURE_TERM<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the data source vendor's name for a procedure; for example, "database procedure", "stored procedure", "procedure", "package", or "stored query".</TD>
+# </TR>
+ SQL_PROCEDURE_TERM => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_PROCEDURES<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string: "Y" if the data source supports procedures and the driver supports the ODBC procedure invocation syntax; "N" otherwise.</TD>
+# </TR>
+ SQL_PROCEDURES => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_POS_OPERATIONS (ODBC 2.0)</TD>
+# <TD width=50%>An SQLINTEGER bitmask enumerating the support operations in <B>SQLSetPos</B>.
+# <P>The following bitmasks are used in conjunction with the flag to determine which options are supported.</P>
+#
+# <P>SQL_POS_POSITION (ODBC 2.0) SQL_POS_REFRESH (ODBC 2.0) SQL_POS_UPDATE (ODBC 2.0) SQL_POS_DELETE (ODBC 2.0) SQL_POS_ADD (ODBC 2.0) </P>
+# </TD>
+# </TR>
+ SQL_POS_OPERATIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_QUOTED_IDENTIFIER_CASE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUSMALLINT value as follows:
+# <P>SQL_IC_UPPER = Quoted identifiers in SQL are not case-sensitive and are stored in uppercase in the system catalog.</P>
+#
+# <P>SQL_IC_LOWER = Quoted identifiers in SQL are not case-sensitive and are stored in lowercase in the system catalog.</P>
+#
+# <P>SQL_IC_SENSITIVE = Quoted identifiers in SQL are case-sensitive and are stored in mixed case in the system catalog. (In an SQL-92&#0150;compliant database, quoted identifiers are always case-sensitive.)</P>
+#
+# <P>SQL_IC_MIXED = Quoted identifiers in SQL are not case-sensitive and are stored in mixed case in the system catalog.</P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return SQL_IC_SENSITIVE.</P>
+# </TD>
+# </TR>
+ SQL_QUOTED_IDENTIFIER_CASE => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ROW_UPDATES<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string: "Y" if a keyset-driven or mixed cursor maintains row versions or values for all fetched rows and therefore can detect any updates made to a row by any user since the row was last fetched. (This applies only to updates, not to deletions or insertions.) The driver can return the SQL_ROW_UPDATED flag to the row status array when <B>SQLFetchScroll</B> is called. Otherwise, "N".</TD>
+# </TR>
+ SQL_ROW_UPDATES => {
+ type => q(YesNo),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SCHEMA_TERM<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the data source vendor's name for an schema; for example, "owner", "Authorization ID", or "Schema".
+# <P>The character string can be returned in upper, lower, or mixed case.</P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return "schema".</P>
+#
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_OWNER_TERM.</P>
+# </TD>
+# </TR>
+ SQL_SCHEMA_TERM => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SCHEMA_USAGE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the statements in which schemas can be used:
+# <P>SQL_SU_DML_STATEMENTS = Schemas are supported in all Data Manipulation Language statements: <B>SELECT</B>, <B>INSERT</B>, <B>UPDATE</B>, <B>DELETE</B>, and if supported, <B>SELECT FOR UPDATE</B> and positioned update and delete statements.</P>
+#
+# <P>SQL_SU_PROCEDURE_INVOCATION = Schemas are supported in the ODBC procedure invocation statement.</P>
+#
+# <P>SQL_SU_TABLE_DEFINITION = Schemas are supported in all table definition statements: <B>CREATE TABLE</B>, <B>CREATE VIEW</B>, <B>ALTER TABLE</B>, <B>DROP TABLE</B>, and <B>DROP VIEW</B>.</P>
+#
+# <P>SQL_SU_INDEX_DEFINITION = Schemas are supported in all index definition statements: <B>CREATE INDEX</B> and <B>DROP INDEX</B>.</P>
+#
+# <P>SQL_SU_PRIVILEGE_DEFINITION = Schemas are supported in all privilege definition statements: <B>GRANT</B> and <B>REVOKE</B>. </P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return the SQL_SU_DML_STATEMENTS, SQL_SU_TABLE_DEFINITION, and SQL_SU_PRIVILEGE_DEFINITION options, as supported.</P>
+#
+# <P>This <I>InfoType</I> has been renamed for ODBC 3.0 from the ODBC 2.0 <I>InfoType</I> SQL_OWNER_USAGE.</P>
+# </TD>
+# </TR>
+ SQL_SCHEMA_USAGE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SCROLL_OPTIONS<BR>
+# (ODBC 1.0)
+# <P>The information type was introduced in ODBC 1.0; each bitmask is labeled with the version in which it was introduced.</P>
+# </TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the scroll options supported for scrollable cursors.
+# <P>The following bitmasks are used to determine which options are supported:</P>
+#
+# <P>SQL_SO_FORWARD_ONLY = The cursor only scrolls forward. (ODBC 1.0)</P>
+#
+# <P>SQL_SO_STATIC = The data in the result set is static. (ODBC 2.0)</P>
+#
+# <P>SQL_SO_KEYSET_DRIVEN = The driver saves and uses the keys for every row in the result set. (ODBC 1.0)</P>
+#
+# <P>SQL_SO_DYNAMIC = The driver keeps the keys for every row in the rowset (the keyset size is the same as the rowset size). (ODBC 1.0)</P>
+#
+# <P>SQL_SO_MIXED = The driver keeps the keys for every row in the keyset, and the keyset size is greater than the rowset size. The cursor is keyset-driven inside the keyset and dynamic outside the keyset. (ODBC 1.0)</P>
+#
+# <P>For information about scrollable cursors, see "<A HREF="odbcscrollable_cursors.htm">Scrollable Cursors</A>" in Chapter 11: Retrieving Results (Advanced)</P>
+# </TD>
+# </TR>
+ SQL_SCROLL_OPTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SEARCH_PATTERN_ESCAPE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string specifying what the driver supports as an escape character that permits the use of the pattern match metacharacters underscore (_) and percent sign (%) as valid characters in search patterns. This escape character applies only for those catalog function arguments that support search strings. If this string is empty, the driver does not support a search-pattern escape character.
+# <P>Because this information type does not indicate general support of the escape character in the <B>LIKE</B> predicate, SQL-92 does not include requirements for this character string.</P>
+#
+# <P>This <I>InfoType</I> is limited to catalog functions. For a description of the use of the escape character in search pattern strings, see "<A HREF="odbcpattern_value_arguments.htm">Pattern Value Arguments</A>" in Chapter 7: Catalog Functions.</P>
+# </TD>
+# </TR>
+ SQL_SEARCH_PATTERN_ESCAPE => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SERVER_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the actual data source&#0150;specific server name; useful when a data source name is used during <B>SQLConnect</B>, <B>SQLDriverConnect</B>, and<B> SQLBrowseConnect</B>.</TD>
+# </TR>
+ SQL_SERVER_NAME => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SPECIAL_CHARACTERS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>A character string containing all special characters (that is, all characters except a through z, A through Z, 0 through 9, and underscore) that can be used in an identifier name, such as a table name, column column name, or index name, on the data source. For example, "#$^". If an identifier contains one or more of these characters, the identifier must be a delimited identifier.</TD>
+# </TR>
+ SQL_SPECIAL_CHARACTERS => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL_CONFORMANCE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER value indicating the level of SQL-92 supported by the driver:
+# <P>SQL_SC_SQL92_ENTRY = Entry level SQL-92 compliant.</P>
+#
+# <P>SQL_SC_FIPS127_2_TRANSITIONAL = FIPS 127-2 transitional level compliant.</P>
+#
+# <P>SQL_SC_SQL92_FULL = Full level SQL-92 compliant.</P>
+#
+# <P>SQL_SC_ SQL92_INTERMEDIATE = Intermediate level SQL-92 compliant.</P>
+# </TD>
+# </TR>
+ SQL_SQL_CONFORMANCE => {
+ type => q(Long),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_DATETIME_FUNCTIONS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the datetime scalar functions that are supported by the driver and the associated data source, as defined in SQL-92.
+# <P>The following bitmasks are used to determine which datetime functions are supported:</P>
+#
+# <P>SQL_SDF_CURRENT_DATE<BR>
+# SQL_SDF_CURRENT_TIME<BR>
+# SQL_SDF_CURRENT_TIMESTAMP</P>
+# </TD>
+# </TR>
+ SQL_SQL92_DATETIME_FUNCTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_FOREIGN_KEY_DELETE_RULE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the rules supported for a foreign key in a <B>DELETE</B> statement, as defined in SQL-92.
+# <P>The following bitmasks are used to determine which clauses are supported by the data source:</P>
+#
+# <P>SQL_SFKD_CASCADE<BR>
+# SQL_SFKD_NO_ACTION<BR>
+# SQL_SFKD_SET_DEFAULT<BR>
+# SQL_SFKD_SET_NULL</P>
+#
+# <P>An FIPS Transitional level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_SQL92_FOREIGN_KEY_DELETE_RULE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_FOREIGN_KEY_UPDATE_RULE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the rules supported for a foreign key in an <B>UPDATE</B> statement, as defined in SQL-92.
+# <P>The following bitmasks are used to determine which clauses are supported by the data source:</P>
+#
+# <P>SQL_SFKU_CASCADE<BR>
+# SQL_SFKU_NO_ACTION<BR>
+# SQL_SFKU_SET_DEFAULT<BR>
+# SQL_SFKU_SET_NULL </P>
+#
+# <P>An SQL-92 Full level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_SQL92_FOREIGN_KEY_UPDATE_RULE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_GRANT<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses supported in the <B>GRANT</B> statement, as defined in SQL-92.
+# <P>The SQL-92 or FIPS conformance level at which this feature needs to be supported is shown in parentheses next to each bitmask.</P>
+#
+# <P>The following bitmasks are used to determine which clauses are supported by the data source:</P>
+#
+# <P>SQL_SG_DELETE_TABLE (Entry level)<BR>
+# SQL_SG_INSERT_COLUMN (Intermediate level)<BR>
+# SQL_SG_INSERT_TABLE (Entry level) <BR>
+# SQL_SG_REFERENCES_TABLE (Entry level)<BR>
+# SQL_SG_REFERENCES_COLUMN (Entry level)<BR>
+# SQL_SG_SELECT_TABLE (Entry level)<BR>
+# SQL_SG_UPDATE_COLUMN (Entry level)<BR>
+# SQL_SG_UPDATE_TABLE (Entry level) <BR>
+# SQL_SG_USAGE_ON_DOMAIN (FIPS Transitional level)<BR>
+# SQL_SG_USAGE_ON_CHARACTER_SET (FIPS Transitional level)<BR>
+# SQL_SG_USAGE_ON_COLLATION (FIPS Transitional level)<BR>
+# SQL_SG_USAGE_ON_TRANSLATION (FIPS Transitional level)<BR>
+# SQL_SG_WITH_GRANT_OPTION (Entry level)</P>
+# </TD>
+# </TR>
+ SQL_SQL92_GRANT => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_NUMERIC_VALUE_FUNCTIONS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the numeric value scalar functions that are supported by the driver and the associated data source, as defined in SQL-92.
+# <P>The following bitmasks are used to determine which numeric functions are supported:</P>
+#
+# <P>SQL_SNVF_BIT_LENGTH<BR>
+# SQL_SNVF_CHAR_LENGTH<BR>
+# SQL_SNVF_CHARACTER_LENGTH<BR>
+# SQL_SNVF_EXTRACT<BR>
+# SQL_SNVF_OCTET_LENGTH<BR>
+# SQL_SNVF_POSITION</P>
+# </TD>
+# </TR>
+ SQL_SQL92_NUMERIC_VALUE_FUNCTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_PREDICATES<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the predicates supported in a <B>SELECT</B> statement, as defined in SQL-92.
+# <P>The SQL-92 or FIPS conformance level at which this feature needs to be supported is shown in parentheses next to each bitmask.</P>
+#
+# <P>The following bitmasks are used to determine which options are supported by the data source:</P>
+#
+# <P>SQL_SP_BETWEEN (Entry level)<BR>
+# SQL_SP_COMPARISON (Entry level)<BR>
+# SQL_SP_EXISTS (Entry level)<BR>
+# SQL_SP_IN (Entry level)<BR>
+# SQL_SP_ISNOTNULL (Entry level)<BR>
+# SQL_SP_ISNULL (Entry level)<BR>
+# SQL_SP_LIKE (Entry level)<BR>
+# SQL_SP_MATCH_FULL (Full level)<BR>
+# SQL_SP_MATCH_PARTIAL(Full level)<BR>
+# SQL_SP_MATCH_UNIQUE_FULL (Full level)<BR>
+# SQL_SP_MATCH_UNIQUE_PARTIAL (Full level)<BR>
+# SQL_SP_OVERLAPS (FIPS Transitional level)<BR>
+# SQL_SP_QUANTIFIED_COMPARISON (Entry level)<BR>
+# SQL_SP_UNIQUE (Entry level)</P>
+# </TD>
+# </TR>
+ SQL_SQL92_PREDICATES => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_RELATIONAL_JOIN_OPERATORS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the relational join operators supported in a <B>SELECT</B> statement, as defined in SQL-92.
+# <P>The SQL-92 or FIPS conformance level at which this feature needs to be supported is shown in parentheses next to each bitmask.</P>
+#
+# <P>The following bitmasks are used to determine which options are supported by the data source:</P>
+#
+# <P>SQL_SRJO_CORRESPONDING_CLAUSE (Intermediate level)<BR>
+# SQL_SRJO_CROSS_JOIN (Full level)<BR>
+# SQL_SRJO_EXCEPT_JOIN (Intermediate level)<BR>
+# SQL_SRJO_FULL_OUTER_JOIN (Intermediate level) <BR>
+# SQL_SRJO_INNER_JOIN (FIPS Transitional level)<BR>
+# SQL_SRJO_INTERSECT_JOIN (Intermediate level)<BR>
+# SQL_SRJO_LEFT_OUTER_JOIN (FIPS Transitional level)<BR>
+# SQL_SRJO_NATURAL_JOIN (FIPS Transitional level)<BR>
+# SQL_SRJO_RIGHT_OUTER_JOIN (FIPS Transitional level)<BR>
+# SQL_SRJO_UNION_JOIN (Full level)</P>
+#
+# <P>SQL_SRJO_INNER_JOIN indicates support for the <B>INNER JOIN </B>syntax, not for the inner join capability. Support for the <B>INNER JOIN</B> syntax is FIPS TRANSITIONAL, while support for the inner join capability is <B>ENTRY</B>.</P>
+# </TD>
+# </TR>
+ SQL_SQL92_RELATIONAL_JOIN_OPERATORS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_REVOKE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the clauses supported in the <B>REVOKE</B> statement, as defined in SQL-92, supported by the data source.
+# <P>The SQL-92 or FIPS conformance level at which this feature needs to be supported is shown in parentheses next to each bitmask.</P>
+#
+# <P>The following bitmasks are used to determine which clauses are supported by the data source:</P>
+#
+# <P>SQL_SR_CASCADE (FIPS Transitional level) <BR>
+# SQL_SR_DELETE_TABLE (Entry level)<BR>
+# SQL_SR_GRANT_OPTION_FOR (Intermediate level) <BR>
+# SQL_SR_INSERT_COLUMN (Intermediate level)<BR>
+# SQL_SR_INSERT_TABLE (Entry level)<BR>
+# SQL_SR_REFERENCES_COLUMN (Entry level)<BR>
+# SQL_SR_REFERENCES_TABLE (Entry level)<BR>
+# SQL_SR_RESTRICT (FIPS Transitional level)<BR>
+# SQL_SR_SELECT_TABLE (Entry level)<BR>
+# SQL_SR_UPDATE_COLUMN (Entry level)<BR>
+# SQL_SR_UPDATE_TABLE (Entry level)<BR>
+# SQL_SR_USAGE_ON_DOMAIN (FIPS Transitional level)<BR>
+# SQL_SR_USAGE_ON_CHARACTER_SET (FIPS Transitional level)<BR>
+# SQL_SR_USAGE_ON_COLLATION (FIPS Transitional level)<BR>
+# SQL_SR_USAGE_ON_TRANSLATION (FIPS Transitional level)</P>
+# </TD>
+# </TR>
+ SQL_SQL92_REVOKE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_ROW_VALUE_CONSTRUCTOR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the row value constructor expressions supported in a <B>SELECT</B> statement, as defined in SQL-92. The following bitmasks are used to determine which options are supported by the data source:
+# <P>SQL_SRVC_VALUE_EXPRESSION <BR>
+# SQL_SRVC_NULL <BR>
+# SQL_SRVC_DEFAULT <BR>
+# SQL_SRVC_ROW_SUBQUERY</P>
+# </TD>
+# </TR>
+ SQL_SQL92_ROW_VALUE_CONSTRUCTOR => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_STRING_FUNCTIONS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the string scalar functions that are supported by the driver and the associated data source, as defined in SQL-92.
+# <P>The following bitmasks are used to determine which string functions are supported:</P>
+#
+# <P>SQL_SSF_CONVERT<BR>
+# SQL_SSF_LOWER<BR>
+# SQL_SSF_UPPER<BR>
+# SQL_SSF_SUBSTRING<BR>
+# SQL_SSF_TRANSLATE<BR>
+# SQL_SSF_TRIM_BOTH<BR>
+# SQL_SSF_TRIM_LEADING<BR>
+# SQL_SSF_TRIM_TRAILING</P>
+# </TD>
+# </TR>
+ SQL_SQL92_STRING_FUNCTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SQL92_VALUE_EXPRESSIONS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the value expressions supported, as defined in SQL-92.
+# <P>The SQL-92 or FIPS conformance level at which this feature needs to be supported is shown in parentheses next to each bitmask.</P>
+#
+# <P>The following bitmasks are used to determine which options are supported by the data source:</P>
+#
+# <P>SQL_SVE_CASE (Intermediate level)<BR>
+# SQL_SVE_CAST (FIPS Transitional level)<BR>
+# SQL_SVE_COALESCE (Intermediate level)<BR>
+# SQL_SVE_NULLIF (Intermediate level)</P>
+# </TD>
+# </TR>
+ SQL_SQL92_VALUE_EXPRESSIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_STANDARD_CLI_CONFORMANCE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the CLI standard or standards to which the driver conforms. The following bitmasks are used to determine which levels the driver conforms to:
+# <P>SQL_SCC_XOPEN_CLI_VERSION1: The driver conforms to the X/Open CLI version 1.</P>
+#
+# <P>SQL_SCC_ISO92_CLI: The driver conforms to the ISO 92 CLI.</P>
+# </TD>
+# </TR>
+ SQL_STANDARD_CLI_CONFORMANCE => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_STATIC_CURSOR_ATTRIBUTES1<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that describes the attributes of a static cursor that are supported by the driver. This bitmask contains the first subset of attributes; for the second subset, see SQL_STATIC_CURSOR_ATTRIBUTES2.
+# <P>The following bitmasks are used to determine which attributes are supported:</P>
+#
+# <P>SQL_CA1_NEXT<BR>
+# SQL_CA1_ABSOLUTE<BR>
+# SQL_CA1_RELATIVE<BR>
+# SQL_CA1_BOOKMARK<BR>
+# SQL_CA1_LOCK_NO_CHANGE<BR>
+# SQL_CA1_LOCK_EXCLUSIVE<BR>
+# SQL_CA1_LOCK_UNLOCK<BR>
+# SQL_CA1_POS_POSITION<BR>
+# SQL_CA1_POS_UPDATE<BR>
+# SQL_CA1_POS_DELETE<BR>
+# SQL_CA1_POS_REFRESH<BR>
+# SQL_CA1_POSITIONED_UPDATE<BR>
+# SQL_CA1_POSITIONED_DELETE<BR>
+# SQL_CA1_SELECT_FOR_UPDATE<BR>
+# SQL_CA1_BULK_ADD<BR>
+# SQL_CA1_BULK_UPDATE_BY_BOOKMARK<BR>
+# SQL_CA1_BULK_DELETE_BY_BOOKMARK<BR>
+# SQL_CA1_BULK_FETCH_BY_BOOKMARK</P>
+#
+# <P>For descriptions of these bitmasks, see SQL_DYNAMIC_CURSOR_ATTRIBUTES1 (and substitute "static cursor" for "dynamic cursor" in the descriptions).</P>
+#
+# <P>An SQL-92 Intermediate level&#0150;conformant driver will usually return the SQL_CA1_NEXT, SQL_CA1_ABSOLUTE, and SQL_CA1_RELATIVE options as supported, because the driver supports scrollable cursors through the embedded SQL FETCH statement. Because this does not directly determine the underlying SQL support, however, scrollable cursors may not be supported, even for an SQL-92 Intermediate level&#0150;conformant driver.</P>
+# </TD>
+# </TR>
+ SQL_STATIC_CURSOR_ATTRIBUTES1 => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_STATIC_CURSOR_ATTRIBUTES2<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask that describes the attributes of a static cursor that are supported by the driver. This bitmask contains the second subset of attributes; for the first subset, see SQL_STATIC_CURSOR_ATTRIBUTES1.
+# <P>The following bitmasks are used to determine which attributes are supported:</P>
+#
+# <P>SQL_CA2_READ_ONLY_CONCURRENCY<BR>
+# SQL_CA2_LOCK_CONCURRENCY<BR>
+# SQL_CA2_OPT_ROWVER_CONCURRENCY<BR>
+# SQL_CA2_OPT_VALUES_CONCURRENCY<BR>
+# SQL_CA2_SENSITIVITY_ADDITIONS<BR>
+# SQL_CA2_SENSITIVITY_DELETIONS<BR>
+# SQL_CA2_SENSITIVITY_UPDATES<BR>
+# SQL_CA2_MAX_ROWS_SELECT<BR>
+# SQL_CA2_MAX_ROWS_INSERT<BR>
+# SQL_CA2_MAX_ROWS_DELETE<BR>
+# SQL_CA2_MAX_ROWS_UPDATE<BR>
+# SQL_CA2_MAX_ROWS_CATALOG<BR>
+# SQL_CA2_MAX_ROWS_AFFECTS_ALL<BR>
+# SQL_CA2_CRC_EXACT<BR>
+# SQL_CA2_CRC_APPROXIMATE<BR>
+# SQL_CA2_SIMULATE_NON_UNIQUE<BR>
+# SQL_CA2_SIMULATE_TRY_UNIQUE<BR>
+# SQL_CA2_SIMULATE_UNIQUE</P>
+#
+# <P>For descriptions of these bitmasks, see SQL_DYNAMIC_CURSOR_ATTRIBUTES2 (and substitute "static cursor" for "dynamic cursor" in the descriptions).</P>
+# </TD>
+# </TR>
+ SQL_STATIC_CURSOR_ATTRIBUTES2 => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_STRING_FUNCTIONS<BR>
+# (ODBC 1.0)
+# <P>The information type was introduced in ODBC 1.0; each bitmask is labeled with the version in which it was introduced.</P>
+# </TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the scalar string functions supported by the driver and associated data source.
+# <P>The following bitmasks are used to determine which string functions are supported:</P>
+#
+# <P>SQL_FN_STR_ASCII (ODBC 1.0)<BR>
+# SQL_FN_STR_BIT_LENGTH (ODBC 3.0)<BR>
+# SQL_FN_STR_CHAR (ODBC 1.0)<BR>
+# SQL_FN_STR_CHAR_<BR>
+# LENGTH (ODBC 3.0)<BR>
+# SQL_FN_STR_CHARACTER_<BR>
+# LENGTH (ODBC 3.0)<BR>
+# SQL_FN_STR_CONCAT (ODBC 1.0)<BR>
+# SQL_FN_STR_DIFFERENCE (ODBC 2.0)<BR>
+# SQL_FN_STR_INSERT (ODBC 1.0)<BR>
+# SQL_FN_STR_LCASE (ODBC 1.0)<BR>
+# SQL_FN_STR_LEFT (ODBC 1.0)<BR>
+# SQL_FN_STR_LENGTH (ODBC 1.0)<BR>
+# SQL_FN_STR_LOCATE (ODBC 1.0)<BR>
+# SQL_FN_STR_LTRIM (ODBC 1.0) <BR>
+# SQL_FN_STR_OCTET_<BR>
+# LENGTH (ODBC 3.0) <BR>
+# SQL_FN_STR_POSITION (ODBC 3.0)<BR>
+# SQL_FN_STR_REPEAT (ODBC 1.0)<BR>
+# SQL_FN_STR_REPLACE (ODBC 1.0)<BR>
+# SQL_FN_STR_RIGHT (ODBC 1.0)<BR>
+# SQL_FN_STR_RTRIM (ODBC 1.0)<BR>
+# SQL_FN_STR_SOUNDEX (ODBC 2.0)<BR>
+# SQL_FN_STR_SPACE (ODBC 2.0)<BR>
+# SQL_FN_STR_SUBSTRING (ODBC 1.0)<BR>
+# SQL_FN_STR_UCASE (ODBC 1.0)</P>
+#
+# <P>If an application can call the <B>LOCATE</B> scalar function with the <I>string_exp1</I>, <I>string_exp2</I>, and <I>start</I> arguments, the driver returns the SQL_FN_STR_LOCATE bitmask. If an application can call the LOCATE scalar function with only the <I>string_exp1</I> and <I>string_exp2</I> arguments, the driver returns the SQL_FN_STR_LOCATE_2 bitmask. Drivers that fully support the <B>LOCATE</B> scalar function return both bitmasks.</P>
+#
+# <P>(For more information, see <A HREF="odbcstring_functions.htm">String Functions</A> in Appendix E, "Scalar Functions.")</P>
+# </TD>
+# </TR>
+ SQL_STRING_FUNCTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SUBQUERIES<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the predicates that support subqueries:
+# <P>SQL_SQ_CORRELATED_SUBQUERIES<BR>
+# SQL_SQ_COMPARISON<BR>
+# SQL_SQ_EXISTS<BR>
+# SQL_SQ_IN<BR>
+# SQL_SQ_QUANTIFIED</P>
+#
+# <P>The SQL_SQ_CORRELATED_SUBQUERIES bitmask indicates that all predicates that support subqueries support correlated subqueries.</P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return a bitmask in which all of these bits are set.</P>
+# </TD>
+# </TR>
+ SQL_SUBQUERIES => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_SYSTEM_FUNCTIONS<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the scalar system functions supported by the driver and associated data source.
+# <P>The following bitmasks are used to determine which system functions are supported:</P>
+#
+# <P>SQL_FN_SYS_DBNAME<BR>
+# SQL_FN_SYS_IFNULL<BR>
+# SQL_FN_SYS_USERNAME</P>
+# </TD>
+# </TR>
+ SQL_SYSTEM_FUNCTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_TABLE_TERM<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the data source vendor's name for a table; for example, "table" or "file".
+# <P>This character string can be in upper, lower, or mixed case. </P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return "table".</P>
+# </TD>
+# </TR>
+ SQL_TABLE_TERM => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_TIMEDATE_ADD_INTERVALS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the timestamp intervals supported by the driver and associated data source for the TIMESTAMPADD scalar function.
+# <P>The following bitmasks are used to determine which intervals are supported:</P>
+#
+# <P>SQL_FN_TSI_FRAC_SECOND<BR>
+# SQL_FN_TSI_SECOND<BR>
+# SQL_FN_TSI_MINUTE<BR>
+# SQL_FN_TSI_HOUR<BR>
+# SQL_FN_TSI_DAY<BR>
+# SQL_FN_TSI_WEEK<BR>
+# SQL_FN_TSI_MONTH<BR>
+# SQL_FN_TSI_QUARTER<BR>
+# SQL_FN_TSI_YEAR</P>
+#
+# <P>An FIPS Transitional level&#0150;conformant driver will always return a bitmask in which all of these bits are set.</P>
+# </TD>
+# </TR>
+ SQL_TIMEDATE_ADD_INTERVALS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_TIMEDATE_DIFF_INTERVALS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the timestamp intervals supported by the driver and associated data source for the TIMESTAMPDIFF scalar function.
+# <P>The following bitmasks are used to determine which intervals are supported:</P>
+#
+# <P>SQL_FN_TSI_FRAC_SECOND<BR>
+# SQL_FN_TSI_SECOND<BR>
+# SQL_FN_TSI_MINUTE<BR>
+# SQL_FN_TSI_HOUR<BR>
+# SQL_FN_TSI_DAY<BR>
+# SQL_FN_TSI_WEEK<BR>
+# SQL_FN_TSI_MONTH<BR>
+# SQL_FN_TSI_QUARTER<BR>
+# SQL_FN_TSI_YEAR </P>
+#
+# <P>An FIPS Transitional level&#0150;conformant driver will always return a bitmask in which all of these bits are set.</P>
+# </TD>
+# </TR>
+ SQL_TIMEDATE_DIFF_INTERVALS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_TIMEDATE_FUNCTIONS<BR>
+# (ODBC 1.0)
+# <P>The information type was introduced in ODBC 1.0; each bitmask is labeled with the version in which it was introduced.</P>
+# </TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the scalar date and time functions supported by the driver and associated data source.
+# <P>The following bitmasks are used to determine which date and time functions are supported:</P>
+#
+# <P>SQL_FN_TD_CURRENT_DATE ODBC 3.0)<BR>
+# SQL_FN_TD_CURRENT_TIME (ODBC 3.0)<BR>
+# SQL_FN_TD_CURRENT_TIMESTAMP (ODBC 3.0)<BR>
+# SQL_FN_TD_CURDATE (ODBC 1.0)<BR>
+# SQL_FN_TD_CURTIME (ODBC 1.0) <BR>
+# SQL_FN_TD_DAYNAME (ODBC 2.0)<BR>
+# SQL_FN_TD_DAYOFMONTH (ODBC 1.0)<BR>
+# SQL_FN_TD_DAYOFWEEK (ODBC 1.0)<BR>
+# SQL_FN_TD_DAYOFYEAR (ODBC 1.0) <BR>
+# SQL_FN_TD_EXTRACT (ODBC 3.0)<BR>
+# SQL_FN_TD_HOUR (ODBC 1.0)<BR>
+# SQL_FN_TD_MINUTE (ODBC 1.0)<BR>
+# SQL_FN_TD_MONTH (ODBC 1.0)<BR>
+# SQL_FN_TD_MONTHNAME (ODBC 2.0)<BR>
+# SQL_FN_TD_NOW (ODBC 1.0)<BR>
+# SQL_FN_TD_QUARTER (ODBC 1.0)<BR>
+# SQL_FN_TD_SECOND (ODBC 1.0)<BR>
+# SQL_FN_TD_TIMESTAMPADD (ODBC 2.0)<BR>
+# SQL_FN_TD_TIMESTAMPDIFF (ODBC 2.0)<BR>
+# SQL_FN_TD_WEEK (ODBC 1.0)<BR>
+# SQL_FN_TD_YEAR (ODBC 1.0)</P>
+# </TD>
+# </TR>
+ SQL_TIMEDATE_FUNCTIONS => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_TXN_CAPABLE<BR>
+# (ODBC 1.0)
+# <P>The information type was introduced in ODBC 1.0; each return value is labeled with the version in which it was introduced.</P>
+# </TD>
+# <TD width=50%>An SQLUSMALLINT value describing the transaction support in the driver or data source:
+# <P>SQL_TC_NONE = Transactions not supported. (ODBC 1.0)</P>
+#
+# <P>SQL_TC_DML = Transactions can contain only Data Manipulation Language (DML) statements (<B>SELECT</B>, <B>INSERT</B>, <B>UPDATE</B>, <B>DELETE</B>). Data Definition Language (DDL) statements encountered in a transaction cause an error. (ODBC 1.0)</P>
+#
+# <P>SQL_TC_DDL_COMMIT = Transactions can contain only DML statements. DDL statements (<B>CREATE TABLE</B>, <B>DROP INDEX</B>, and so on) encountered in a transaction cause the transaction to be committed. (ODBC 2.0)</P>
+#
+# <P>SQL_TC_DDL_IGNORE = Transactions can contain only DML statements. DDL statements encountered in a transaction are ignored. (ODBC 2.0)</P>
+#
+# <P>SQL_TC_ALL = Transactions can contain DDL statements and DML statements in any order. (ODBC 1.0) </P>
+#
+# <P>(Because support of transactions is mandatory in SQL-92, an SQL-92 conformant driver [any level] will never return SQL_TC_NONE.)</P>
+# </TD>
+# </TR>
+ SQL_TXN_CAPABLE => {
+ type => q(Short),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_TXN_ISOLATION_OPTION<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the transaction isolation levels available from the driver or data source.
+# <P>The following bitmasks are used in conjunction with the flag to determine which options are supported:</P>
+#
+# <P>SQL_TXN_READ_UNCOMMITTED<BR>
+# SQL_TXN_READ_COMMITTED<BR>
+# SQL_TXN_REPEATABLE_READ<BR>
+# SQL_TXN_SERIALIZABLE</P>
+#
+# <P>For descriptions of these isolation levels, see the description of SQL_DEFAULT_TXN_ISOLATION.</P>
+#
+# <P>To set the transaction isolation level, an application calls <B>SQLSetConnectAttr</B> to set the SQL_ATTR_TXN_ISOLATION attribute. For more information, see <B>SQLSetConnectAttr</B>. </P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return SQL_TXN_SERIALIZABLE as supported. A FIPS Transitional level&#0150;conformant driver will always return all of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_TXN_ISOLATION_OPTION => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_UNION<BR>
+# (ODBC 2.0)</TD>
+# <TD width=50%>An SQLUINTEGER bitmask enumerating the support for the <B>UNION</B> clause:
+# <P>SQL_U_UNION = The data source supports the <B>UNION</B> clause.</P>
+#
+# <P>SQL_U_UNION_ALL = The data source supports the <B>ALL</B> keyword in the <B>UNION</B> clause. (<B>SQLGetInfo</B> returns both SQL_U_UNION and SQL_U_UNION_ALL in this case.)</P>
+#
+# <P>An SQL-92 Entry level&#0150;conformant driver will always return both of these options as supported.</P>
+# </TD>
+# </TR>
+ SQL_UNION => {
+ type => q(Bitmask),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_USER_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=50%>A character string with the name used in a particular database, which can be different from the login name.</TD>
+# </TR>
+ SQL_USER_NAME => {
+ type => q(Char),
+ },
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_XOPEN_CLI_YEAR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=50%>A character string that indicates the year of publication of the X/Open specification with which the version of the ODBC Driver Manager fully complies.</TD>
+# </TR>
+ SQL_XOPEN_CLI_YEAR => {
+ type => q(Char),
+ },
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Code Example</B></P>
+#
+# <P><B>SQLGetInfo</B> returns lists of supported options as an SQLUINTEGER bitmask in *<I>InfoValuePtr</I>. The bitmask for each option is used in conjunction with the flag to determine whether the option is supported.</P>
+#
+# <P>For example, an application could use the following code to determine whether the SUBSTRING scalar function is supported by the driver associated with the connection:</P>
+#
+# <PRE class="code">SQLUINTEGER fFuncs;
+#
+# SQLGetInfo(hdbc,
+# SQL_STRING_FUNCTIONS,
+# (SQLPOINTER)&amp;fFuncs,
+# sizeof(fFuncs),
+# NULL);
+#
+# if (fFuncs &amp; SQL_FN_STR_SUBSTRING)&nbsp;&nbsp;&nbsp;/* SUBSTRING supported */
+# ;
+# else&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;/* SUBSTRING not supported */
+# ;</PRE>
+#
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=50%>For information about</TH>
+# <TH width=50%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning the setting of a connection attribute</TD>
+# <TD width=50%><A HREF="odbcsqlgetconnectattr.htm">SQLGetConnectAttr</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Determining whether a driver supports a function</TD>
+# <TD width=50%><A HREF="odbcsqlgetfunctions.htm">SQLGetFunctions</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning the setting of a statement attribute</TD>
+# <TD width=50%><A HREF="odbcsqlgetstmtattr.htm">SQLGetStmtAttr</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning information about a data source's data types</TD>
+# <TD width=50%><A HREF="odbcsqlgettypeinfo.htm">SQLGetTypeInfo</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+};
+
+print <<END;
+#include "HandleDbc.hpp"
+
+HandleDbc::InfoTab
+HandleDbc::m_infoTab[] = {
+END
+
+my @name = sort keys %$info;
+for my $name (@name) {
+ my $p = $info->{$name};
+ my $type = $p->{type};
+ $type =~ /^(Char|YesNo|Short|Long|Bitmask)$/
+ or die "$name: bad type $type";
+ my $defstr = $type eq "YesNo" ? q("N") : $type eq "Char" ? q("") : q(0);
+ my $s = <<END;
+ { $name,
+ InfoTab\::$type,
+ 0L,
+ $defstr
+ },
+END
+ if ($p->{omit}) {
+ $s ="#if 0\n" . $s . "#endif\n";
+ }
+ print $s;
+};
+
+print <<END;
+ { 0,
+ InfoTab::End,
+ 0L,
+ 0
+ }
+};
+END
+
+# vim: set sw=4:
diff --git a/ndb/src/client/odbc/docs/gettypeinfo.pl b/ndb/src/client/odbc/docs/gettypeinfo.pl
new file mode 100644
index 00000000000..0a999fd7249
--- /dev/null
+++ b/ndb/src/client/odbc/docs/gettypeinfo.pl
@@ -0,0 +1,645 @@
+# usage: perl TypeInfo.data
+# prints template for typeinfo
+use strict;
+my $position = 0;
+
+#
+# odbcsqlgettypeinfo.htm
+#
+my @typeinfo = (
+ { name => "UNDEF",
+ type => q(Undef),
+ nullable => q(true),
+ position => 0,
+ },
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLGetTypeInfo</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqlgettypeinfo"></A>SQLGetTypeInfo</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 1.0<BR>
+# Standards Compliance: ISO 92</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLGetTypeInfo</B> returns information about data types supported by the data source. The driver returns the information in the form of an SQL result set. The data types are intended for use in Data Definition Language (DDL) statements.</P>
+#
+# <P class="indent"><b class="le">Important&nbsp;&nbsp;&nbsp;</b>Applications must use the type names returned in the TYPE_NAME column of the <B>SQLGetTypeInfo</B> result set in <B>ALTER TABLE</B> and <B>CREATE TABLE</B> statements. <B>SQLGetTypeInfo</B> may return more than one row with the same value in the DATA_TYPE column.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLGetTypeInfo</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHSTMT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StatementHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>DataType</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>StatementHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Statement handle for the result set.</dd>
+#
+# <DT><I>DataType</I></DT>
+#
+# <DD>[Input]<BR>
+# The SQL data type. This must be one of the values in the "<A HREF="odbcsql_data_types.htm">SQL Data Types</A>" section of Appendix D: Data Types, or a driver-specific SQL data type. SQL_ALL_TYPES specifies that information about all data types should be returned.</dd>
+# </DL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_STILL_EXECUTING, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLGetTypeInfo</B> returns SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value can be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_STMT and a <I>Handle</I> of <I>StatementHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLGetTypeInfo </B>and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=22%>SQLSTATE</TH>
+# <TH width=26%>Error</TH>
+# <TH width=52%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01000</TD>
+# <TD width=26%>General warning</TD>
+# <TD width=52%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01S02</TD>
+# <TD width=26%>Option value changed</TD>
+# <TD width=52%>A specified statement attribute was invalid because of implementation working conditions, so a similar value was temporarily substituted. (Call <B>SQLGetStmtAttr</B> to determine the temporarily substituted value.) The substitute value is valid for the <I>StatementHandle</I> until the cursor is closed. The statement attributes that can be changed are: SQL_ATTR_CONCURRENCY, SQL_ATTR_CURSOR_TYPE, SQL_ATTR_KEYSET_SIZE, SQL_ATTR_MAX_LENGTH, SQL_ATTR_MAX_ROWS, SQL_ATTR_QUERY_TIMEOUT, and SQL_ATTR_SIMULATE_CURSOR. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>08S01</TD>
+# <TD width=26%>Communication link failure</TD>
+# <TD width=52%>The communication link between the driver and the data source to which the driver was connected failed before the function completed processing.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>24000</TD>
+# <TD width=26%>Invalid cursor state</TD>
+# <TD width=52%>A cursor was open on the <I>StatementHandle,</I> and <B>SQLFetch</B> or <B>SQLFetchScroll</B> had been called. This error is returned by the Driver Manager if <B>SQLFetch</B> or <B>SQLFetchScroll</B> has not returned SQL_NO_DATA, and is returned by the driver if <B>SQLFetch</B> or <B>SQLFetchScroll</B> has returned SQL_NO_DATA.
+# <P>A result set was open on the <I>StatementHandle</I>, but <B>SQLFetch</B> or <B>SQLFetchScroll</B> had not been called.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40001</TD>
+# <TD width=26%>Serialization failure</TD>
+# <TD width=52%>The transaction was rolled back due to a resource deadlock with another transaction.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40003</TD>
+# <TD width=26%>Statement completion unknown</TD>
+# <TD width=52%>The associated connection failed during the execution of this function and the state of the transaction cannot be determined.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY000</TD>
+# <TD width=26%>General error</TD>
+# <TD width=52%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause. </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY001</TD>
+# <TD width=26%>Memory allocation error</TD>
+# <TD width=52%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY004</TD>
+# <TD width=26%>Invalid SQL data type</TD>
+# <TD width=52%>The value specified for the argument <I>DataType</I> was neither a valid ODBC SQL data type identifier nor a driver-specific data type identifier supported by the driver.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY008</TD>
+# <TD width=26%>Operation canceled</TD>
+# <TD width=52%>Asynchronous processing was enabled for the <I>StatementHandle</I>, then the function was called and, before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I>. Then the function was called again on the <I>StatementHandle</I>.
+# <P>The function was called and, before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I> from a different thread in a multithread application.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY010</TD>
+# <TD width=26%>Function sequence error</TD>
+# <TD width=52%>(DM) An asynchronously executing function (not this one) was called for the <I>StatementHandle</I> and was still executing when this function was called.
+# <P>(DM) <B>SQLExecute</B>, <B>SQLExecDirect</B>, <B>SQLBulkOperations</B>, or <B>SQLSetPos</B> was called for the <I>StatementHandle</I> and returned SQL_NEED_DATA. This function was called before data was sent for all data-at-execution parameters or columns.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY013</TD>
+# <TD width=26%>Memory management error</TD>
+# <TD width=52%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYC00</TD>
+# <TD width=26%>Optional feature not implemented</TD>
+# <TD width=52%>The combination of the current settings of the SQL_ATTR_CONCURRENCY and SQL_ATTR_CURSOR_TYPE statement attributes was not supported by the driver or data source.
+# <P>The SQL_ATTR_USE_BOOKMARKS statement attribute was set to SQL_UB_VARIABLE, and the SQL_ATTR_CURSOR_TYPE statement attribute was set to a cursor type for which the driver does not support bookmarks.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT00</TD>
+# <TD width=26%>Timeout expired</TD>
+# <TD width=52%>The query timeout period expired before the data source returned the result set. The timeout period is set through <B>SQLSetStmtAttr</B>, SQL_ATTR_QUERY_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT01</TD>
+# <TD width=26%>Connection timeout expired</TD>
+# <TD width=52%>The connection timeout period expired before the data source responded to the request. The connection timeout period is set through <B>SQLSetConnectAttr</B>, SQL_ATTR_CONNECTION_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>IM001</TD>
+# <TD width=26%>Driver does not support this function</TD>
+# <TD width=52%>(DM) The driver corresponding to the <I>StatementHandle</I> does not support the function.</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Comments</B></P>
+#
+# <P><B>SQLGetTypeInfo</B> returns the results as a standard result set, ordered by DATA_TYPE and then by how closely the data type maps to the corresponding ODBC SQL data type. Data types defined by the data source take precedence over user-defined data types. Consequently, the sort order is not necessarily consistent but can be generalized as DATA_TYPE first, followed by TYPE_NAME, both ascending. For example, suppose that a data source defined INTEGER and COUNTER data types, where COUNTER is auto-incrementing, and that a user-defined data type WHOLENUM has also been defined. These would be returned in the order INTEGER, WHOLENUM, and COUNTER, because WHOLENUM maps closely to the ODBC SQL data type SQL_INTEGER, while the auto-incrementing data type, even though supported by the data source, does not map closely to an ODBC SQL data type. For information about how this information might be used, see "<A HREF="odbcddl_statements.htm">DDL Statements</A>" in Chapter 8: SQL Statements.</P>
+#
+# <P>If the <I>DataType</I> argument specifies a data type which is valid for the version of ODBC supported by the driver, but is not supported by the driver, then it will return an empty result set. </P>
+#
+# <P class="indent"><b class="le">Note&nbsp;&nbsp;&nbsp;</b>For more information about the general use, arguments, and returned data of ODBC catalog functions, see <A HREF="odbccatalog_functions.htm">Chapter 7: Catalog Functions</A>.</P>
+#
+# <P>The following columns have been renamed for ODBC 3.<I>x</I>. The column name changes do not affect backward compatibility because applications bind by column number.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=48%>ODBC 2.0 column</TH>
+# <TH width=52%>ODBC 3.<I>x</I> column</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>PRECISION</TD>
+# <TD width=52%>COLUMN_SIZE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>MONEY</TD>
+# <TD width=52%>FIXED_PREC_SCALE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>AUTO_INCREMENT</TD>
+# <TD width=52%>AUTO_UNIQUE_VALUE</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P>The following columns have been added to the results set returned by <B>SQLGetTypeInfo</B> for ODBC 3.<I>x</I>:
+#
+# <UL type=disc>
+# <LI>SQL_DATA_TYPE</li>
+#
+# <LI>INTERVAL_PRECISION</li>
+#
+# <LI>SQL_DATETIME_SUB</li>
+#
+# <LI>NUM_PREC_RADIX</li>
+# </UL>
+#
+# <P>The following table lists the columns in the result set. Additional columns beyond column 19 (INTERVAL_PRECISION) can be defined by the driver. An application should gain access to driver-specific columns by counting down from the end of the result set rather than specifying an explicit ordinal position. For more information, see "<A HREF="odbcdata_returned_by_catalog_functions.htm">Data Returned by Catalog Functions</A>" in Chapter 7: Catalog Functions.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;<B>SQLGetTypeInfo</B> might not return all data types. For example, a driver might not return user-defined data types. Applications can use any valid data type, regardless of whether it is returned by <B>SQLGetTypeInfo</B>.</P>
+#
+# <P class="indent">The data types returned by <B>SQLGetTypeInfo</B> are those supported by the data source. They are intended for use in Data Definition Language (DDL) statements. Drivers can return result-set data using data types other than the types returned by <B>SQLGetTypeInfo</B>. In creating the result set for a catalog function, the driver might use a data type that is not supported by the data source. </P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=33%><BR>
+# Column name</TH>
+# <TH width=14%>Column <BR>
+# number</TH>
+# <TH width=15%><BR>
+# Data type</TH>
+# <TH width=38%><BR>
+# Comments</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=33%>TYPE_NAME<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>1</TD>
+# <TD width=15%>Varchar<BR>
+# not NULL</TD>
+# <TD width=38%>Data source&#0150;dependent data-type name; for example, "CHAR()", "VARCHAR()", "MONEY", "LONG VARBINARY", or "CHAR ( ) FOR BIT DATA". Applications must use this name in <B>CREATE TABLE</B> and <B>ALTER TABLE</B> statements.</TD>
+# </TR>
+ { name => "TYPE_NAME",
+ type => q(Varchar),
+ length => 20,
+ nullable => q(false),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>DATA_TYPE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>2</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>SQL data type. This can be an ODBC SQL data type or a driver-specific SQL data type. For datetime or interval data types, this column returns the concise data type (such as SQL_TYPE_TIME or SQL_INTERVAL_YEAR_TO_MONTH). For a list of valid ODBC SQL data types, see "<A HREF="odbcsql_data_types.htm">SQL Data Types</A>" in Appendix D: Data Types. For information about driver-specific SQL data types, see the driver’s documentation.</TD>
+# </TR>
+ { name => "DATA_TYPE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(false),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>COLUMN_SIZE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>3</TD>
+# <TD width=15%>Integer</TD>
+# <TD width=38%>The maximum column size that the server supports for this data type. For numeric data, this is the maximum precision. For string data, this is the length in characters. For datetime data types, this is the length in characters of the string representation (assuming the maximum allowed precision of the fractional seconds component). NULL is returned for data types where column size is not applicable. For interval data types, this is the number of characters in the character representation of the interval literal (as defined by the interval leading precision; see "<A HREF="odbcinterval_data_type_length.htm">Interval Data Type Length</A>" in Appendix D: Data Types).
+# <P>For more information on column size, see "<A HREF="odbccolumn_size__decimal_digits__transfer_octet_length__and_display_size.htm">Column Size, Decimal Digits, Transfer Octet Length, and Display Size</A>" in Appendix D: Data Types.</P>
+# </TD>
+# </TR>
+ { name => "COLUMN_SIZE",
+ type => q(Integer),
+ length => undef,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>LITERAL_PREFIX<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>4</TD>
+# <TD width=15%>Varchar</TD>
+# <TD width=38%>Character or characters used to prefix a literal; for example, a single quotation mark (') for character data types or 0x for binary data types; NULL is returned for data types where a literal prefix is not applicable.</TD>
+# </TR>
+ { name => "LITERAL_PREFIX",
+ type => q(Varchar),
+ length => 1,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>LITERAL_SUFFIX<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>5</TD>
+# <TD width=15%>Varchar</TD>
+# <TD width=38%>Character or characters used to terminate a literal; for example, a single quotation mark (') for character data types; NULL is returned for data types where a literal suffix is not applicable.</TD>
+# </TR>
+ { name => "LITERAL_SUFFIX",
+ type => q(Varchar),
+ length => 1,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>CREATE_PARAMS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>6</TD>
+# <TD width=15%>Varchar</TD>
+# <TD width=38%>A list of keywords, separated by commas, corresponding to each parameter that the application may specify in parentheses when using the name that is returned in the TYPE_NAME field. The keywords in the list can be any of the following: length, precision, or scale. They appear in the order that the syntax requires them to be used. For example, CREATE_PARAMS for DECIMAL would be "precision,scale"; CREATE_PARAMS for VARCHAR would equal "length." NULL is returned if there are no parameters for the data type definition; for example, INTEGER.
+# <P>The driver supplies the CREATE_PARAMS text in the language of the country where it is used.</P>
+# </TD>
+# </TR>
+ { name => "CREATE_PARAMS",
+ type => q(Varchar),
+ length => 20,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>NULLABLE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>7</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>Whether the data type accepts a NULL value:
+# <P>SQL_NO_NULLS if the data type does not accept NULL values.</P>
+#
+# <P>SQL_NULLABLE if the data type accepts NULL values.</P>
+#
+# <P>SQL_NULLABLE_UNKNOWN if it is not known whether the column accepts NULL values.</P>
+# </TD>
+# </TR>
+ { name => "NULLABLE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(false),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>CASE_SENSITIVE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>8</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>Whether a character data type is case-sensitive in collations and comparisons:
+# <P>SQL_TRUE if the data type is a character data type and is case-sensitive.</P>
+#
+# <P>SQL_FALSE if the data type is not a character data type or is not case-sensitive.</P>
+# </TD>
+# </TR>
+ { name => "CASE_SENSITIVE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(false),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>SEARCHABLE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>9</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>How the data type is used in a <B>WHERE</B> clause:
+# <P>SQL_PRED_NONE if the column cannot be used in a <B>WHERE</B> clause. (This is the same as the SQL_UNSEARCHABLE value in ODBC 2.<I>x</I>.)</P>
+#
+# <P>SQL_PRED_CHAR if the column can be used in a <B>WHERE</B> clause, but only with the <B>LIKE</B> predicate. (This is the same as the SQL_LIKE_ONLY value in ODBC 2.<I>x</I>.)</P>
+#
+# <P>SQL_PRED_BASIC if the column can be used in a <B>WHERE</B> clause with all the comparison operators except <B>LIKE</B> (comparison, quantified comparison, <B>BETWEEN</B>, <B>DISTINCT</B>, <B>IN</B>, <B>MATCH</B>, and <B>UNIQUE</B>). (This is the same as the SQL_ALL_EXCEPT_LIKE value in ODBC 2.<I>x</I>.)</P>
+#
+# <P>SQL_SEARCHABLE if the column can be used in a <B>WHERE</B> clause with any comparison operator.</P>
+# </TD>
+# </TR>
+ { name => "SEARCHABLE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(false),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>UNSIGNED_ATTRIBUTE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>10</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>Whether the data type is unsigned:
+# <P>SQL_TRUE if the data type is unsigned.</P>
+#
+# <P>SQL_FALSE if the data type is signed.</P>
+#
+# <P>NULL is returned if the attribute is not applicable to the data type or the data type is not numeric.</P>
+# </TD>
+# </TR>
+ { name => "UNSIGNED_ATTRIBUTE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>FIXED_PREC_SCALE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>11</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>Whether the data type has predefined fixed precision and scale (which are data source&#0150;specific), such as a money data type:
+# <P>SQL_TRUE if it has predefined fixed precision and scale.</P>
+#
+# <P>SQL_FALSE if it does not have predefined fixed precision and scale.</P>
+# </TD>
+# </TR>
+ { name => "FIXED_PREC_SCALE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(false),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>AUTO_UNIQUE_VALUE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>12</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>Whether the data type is autoincrementing:
+# <P>SQL_TRUE if the data type is autoincrementing.</P>
+#
+# <P>SQL_FALSE if the data type is not autoincrementing.</P>
+#
+# <P>NULL is returned if the attribute is not applicable to the data type or the data type is not numeric.</P>
+#
+# <P>An application can insert values into a column having this attribute, but typically cannot update the values in the column. </P>
+#
+# <P>When an insert is made into an auto-increment column, a unique value is inserted into the column at insert time. The increment is not defined, but is data source&#0150;specific. An application should not assume that an auto-increment column starts at any particular point or increments by any particular value.</P>
+# </TD>
+# </TR>
+ { name => "AUTO_UNIQUE_VALUE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>LOCAL_TYPE_NAME<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>13</TD>
+# <TD width=15%>Varchar</TD>
+# <TD width=38%>Localized version of the data source&#0150;dependent name of the data type. NULL is returned if a localized name is not supported by the data source. This name is intended for display only, such as in dialog boxes.</TD>
+# </TR>
+ { name => "LOCAL_TYPE_NAME",
+ type => q(Varchar),
+ length => 20,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>MINIMUM_SCALE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>14</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>The minimum scale of the data type on the data source. If a data type has a fixed scale, the MINIMUM_SCALE and MAXIMUM_SCALE columns both contain this value. For example, an SQL_TYPE_TIMESTAMP column might have a fixed scale for fractional seconds. NULL is returned where scale is not applicable. For more information, see "<A HREF="odbccolumn_size__decimal_digits__transfer_octet_length__and_display_size.htm">Column Size, Decimal Digits, Transfer Octet Length, and Display Size</A>" in Appendix D: Data Types.</TD>
+# </TR>
+ { name => "MINIMUM_SCALE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>MAXIMUM_SCALE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>15</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>The maximum scale of the data type on the data source. NULL is returned where scale is not applicable. If the maximum scale is not defined separately on the data source, but is instead defined to be the same as the maximum precision, this column contains the same value as the COLUMN_SIZE column. For more information, see "<A HREF="odbccolumn_size__decimal_digits__transfer_octet_length__and_display_size.htm">Column Size, Decimal Digits, Transfer Octet Length, and Display Size</A>" in Appendix D: Data Types.</TD>
+# </TR>
+ { name => "MAXIMUM_SCALE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>SQL_DATA_TYPE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=14%>16</TD>
+# <TD width=15%>Smallint NOT NULL</TD>
+# <TD width=38%>The value of the SQL data type as it appears in the SQL_DESC_TYPE field of the descriptor. This column is the same as the DATA_TYPE column, except for interval and datetime data types.
+# <P>For interval and datetime data types, the SQL_DATA_TYPE field in the result set will return SQL_INTERVAL or SQL_DATETIME, and the SQL_DATETIME_SUB field will return the subcode for the specific interval or datetime data type. (See <A HREF="odbcdata_types.htm">Appendix D: Data Types</A>.) </P>
+# </TD>
+# </TR>
+ { name => "SQL_DATA_TYPE",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(false),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>SQL_DATETIME_SUB<BR>
+# (ODBC 3.0)</TD>
+# <TD width=14%>17</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>When the value of SQL_DATA_TYPE is SQL_DATETIME or SQL_INTERVAL, this column contains the datetime/interval subcode. For data types other than datetime and interval, this field is NULL.
+# <P>For interval or datetime data types, the SQL_DATA_TYPE field in the result set will return SQL_INTERVAL or SQL_DATETIME, and the SQL_DATETIME_SUB field will return the subcode for the specific interval or datetime data type. (See <A HREF="odbcdata_types.htm">Appendix D: Data Types</A>.)</P>
+# </TD>
+# </TR>
+ { name => "SQL_DATETIME_SUB",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>NUM_PREC_RADIX<BR>
+# (ODBC 3.0)</TD>
+# <TD width=14%>18</TD>
+# <TD width=15%>Integer</TD>
+# <TD width=38%>If the data type is an approximate numeric type, this column contains the value 2 to indicate that COLUMN_SIZE specifies a number of bits. For exact numeric types, this column contains the value 10 to indicate that COLUMN_SIZE specifies a number of decimal digits. Otherwise, this column is NULL.</TD>
+# </TR>
+ { name => "NUM_PREC_RADIX",
+ type => q(Integer),
+ length => undef,
+ nullable => q(true),
+ position => ++$position,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>INTERVAL_PRECISION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=14%>19</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>If the data type is an interval data type, then this column contains the value of the interval leading precision. (See "<A HREF="odbcinterval_data_type_precision.htm">Interval Data Type Precision</A>" in Appendix D: Data Types.) Otherwise, this column is NULL.</TD>
+# </TR>
+ { name => "INTERVAL_PRECISION",
+ type => q(Smallint),
+ length => undef,
+ nullable => q(true),
+ position => ++$position,
+ },
+# </table></div>
+# <!--TS:-->
+# <P>Attribute information can apply to data types or to specific columns in a result set. <B>SQLGetTypeInfo</B> returns information about attributes associated with data types; <B>SQLColAttribute</B> returns information about attributes associated with columns in a result set.</P>
+#
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=50%>For information about</TH>
+# <TH width=50%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Binding a buffer to a column in a result set</TD>
+# <TD width=50%><A HREF="odbcsqlbindcol.htm">SQLBindCol</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Canceling statement processing</TD>
+# <TD width=50%><A HREF="odbcsqlcancel.htm">SQLCancel</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning information about a column in a result set</TD>
+# <TD width=50%><A HREF="odbcsqlcolattribute.htm">SQLColAttribute</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Fetching a block of data or scrolling through a result set</TD>
+# <TD width=50%><A HREF="odbcsqlfetchscroll.htm">SQLFetchScroll</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Fetching a single row or a block of data in a forward-only direction</TD>
+# <TD width=50%><A HREF="odbcsqlfetch.htm">SQLFetch</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning information about a driver or data source</TD>
+# <TD width=50%><A HREF="odbcsqlgetinfo.htm">SQLGetInfo</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+);
+
+my $i4 = " " x 4;
+print "// template-begin\n";
+print "#define Varchar Char\n";
+print "#define Smallint Integer\n";
+print "const SqlTypeInfo::Column\nSqlTypeInfo::m_columnList[] = {\n";
+for my $p (@typeinfo) {
+ print "$i4\{\t$p->{position},\n";
+ print "\t\"$p->{name}\",\n";
+ my $type = $p->{type};
+ if ($p->{position} == 0) {
+ print "\tSqlType()\n";
+ } elsif (! $p->{length}) {
+ print "\tSqlType(SqlType::$type, $p->{nullable})\n";
+ } else {
+ print "\tSqlType(SqlType::$type, $p->{length}, $p->{nullable})\n";
+ }
+ my $c = $p == $typeinfo[-1] ? "" : ",";
+ print "$i4\}$c\n";
+}
+print "};\n";
+print "#undef Varchar\n";
+print "#undef Smallint\n";
+print "const unsigned\nSqlTypeInfo::m_columnCount = $position;\n";
+print "// template-end\n";
+
+# vim: set sw=4:
diff --git a/ndb/src/client/odbc/docs/handleattr.pl b/ndb/src/client/odbc/docs/handleattr.pl
new file mode 100644
index 00000000000..892d34b105b
--- /dev/null
+++ b/ndb/src/client/odbc/docs/handleattr.pl
@@ -0,0 +1,2232 @@
+# usage: perl Attr.data X (X = Env,Dbc,Stmt)
+# prints template for AttrX.cpp
+use strict;
+my $type = shift;
+my $order = 0;
+
+#
+# odbcsqlsetenvattr.htm
+#
+my $attrEnv = {
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLSetEnvAttr</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqlsetenvattr"></A>SQLSetEnvAttr</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 3.0<BR>
+# Standards Compliance: ISO 92</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLSetEnvAttr</B> sets attributes that govern aspects of environments.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLSetEnvAttr</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHENV&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>EnvironmentHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLINTEGER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>Attribute</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLPOINTER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>ValuePtr</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLINTEGER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StringLength</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>EnvironmentHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Environment handle.</dd>
+#
+# <DT><I>Attribute</I></DT>
+#
+# <DD>[Input]<BR>
+# Attribute to set, listed in "Comments."</dd>
+#
+# <DT><I>ValuePtr</I></DT>
+#
+# <DD>[Input]<BR>
+# Pointer to the value to be associated with <I>Attribute</I>. Depending on the value of <I>Attribute</I>, <I>ValuePtr</I> will be a 32-bit integer value or point to a null-terminated character string.</dd>
+#
+# <DT><I>StringLength</I></DT>
+#
+# <DD>[Input] If <I>ValuePtr</I> points to a character string or a binary buffer, this argument should be the length of *<I>ValuePtr</I>. If <I>ValuePtr</I> is an integer, <I>StringLength</I> is ignored.</dd>
+# </DL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLSetEnvAttr</B> returns SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value can be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_ENV and a <I>Handle</I> of <I>EnvironmentHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLSetEnvAttr</B> and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise. If a driver does not support an environment attribute, the error can be returned only during connect time.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=22%>SQLSTATE</TH>
+# <TH width=26%>Error</TH>
+# <TH width=52%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01000</TD>
+# <TD width=26%>General warning</TD>
+# <TD width=52%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01S02</TD>
+# <TD width=26%>Option value changed</TD>
+# <TD width=52%>The driver did not support the value specified in <I>ValuePtr</I> and substituted a similar value. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY000</TD>
+# <TD width=26%>General error</TD>
+# <TD width=52%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY001</TD>
+# <TD width=26%>Memory allocation <BR>
+# error</TD>
+# <TD width=52%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY009</TD>
+# <TD width=26%>Invalid use of null pointer</TD>
+# <TD width=52%>The Attribute argument identified an environment attribute that required a string value, and the <I>ValuePtr</I> argument was a null pointer.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY010</TD>
+# <TD width=26%>Function sequence error</TD>
+# <TD width=52%>(DM) A connection handle has been allocated on <I>EnvironmentHandle</I>. </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY013</TD>
+# <TD width=26%>Memory management error</TD>
+# <TD width=52%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY024</TD>
+# <TD width=26%>Invalid attribute value</TD>
+# <TD width=52%>Given the specified <I>Attribute</I> value, an invalid value was specified in <I>ValuePtr</I>.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY090</TD>
+# <TD width=26%>Invalid string or buffer length</TD>
+# <TD width=52%>The <I>StringLength</I> argument was less than 0 but was not SQL_NTS.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY092</TD>
+# <TD width=26%>Invalid attribute/option identifier</TD>
+# <TD width=52%>(DM) The value specified for the argument <I>Attribute</I> was not valid for the version of ODBC supported by the driver.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYC00</TD>
+# <TD width=26%>Optional feature not implemented</TD>
+# <TD width=52%>The value specified for the argument <I>Attribute</I> was a valid ODBC environment attribute for the version of ODBC supported by the driver, but was not supported by the driver.
+# <P>(DM) The <I>Attribute</I> argument was SQL_ATTR_OUTPUT_NTS, and <I>ValuePtr</I> was SQL_FALSE.</P>
+# </TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Comments</B></P>
+#
+# <P>An application can call <B>SQLSetEnvAttr</B> only if no connection handle is allocated on the environment. All environment attributes successfully set by the application for the environment persist until <B>SQLFreeHandle</B> is called on the environment. More than one environment handle can be allocated simultaneously in ODBC 3<I>.x</I>.</P>
+#
+# <P>The format of information set through <I>ValuePtr</I> depends on the specified <I>Attribute</I>. <B>SQLSetEnvAttr</B> will accept attribute information in one of two different formats: a null-terminated character string or a 32-bit integer value. The format of each is noted in the attribute's description.</P>
+#
+# <P>There are no driver-specific environment attributes.</P>
+#
+# <P>Connection attributes cannot be set by a call to <B>SQLSetEnvAttr</B>. Attempting to do so will return SQLSTATE HY092 (Invalid attribute/option identifier).</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=35%><I>Attribute</I></TH>
+# <TH width=65%><I>ValuePtr</I> contents</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=35%>SQL_ATTR_CONNECTION_POOLING<BR>
+# (ODBC 3.0)</TD>
+# <TD width=65%>A 32-bit SQLUINTEGER value that enables or disables connection pooling at the environment level. The following values are used:
+# <P>SQL_CP_OFF = Connection pooling is turned off. This is the default.</P>
+#
+# <P>SQL_CP_ONE_PER_DRIVER = A single connection pool is supported for each driver. Every connection in a pool is associated with one driver.</P>
+#
+# <P>SQL_CP_ONE_PER_HENV = A single connection pool is supported for each environment. Every connection in a pool is associated with one environment.</P>
+#
+# <P>Connection pooling is enabled by calling <B>SQLSetEnvAttr</B> to set the SQL_ATTR_CONNECTION_POOLING attribute to SQL_CP_ONE_PER_DRIVER or SQL_CP_ONE_PER_HENV. This call must be made before the application allocates the shared environment for which connection pooling is to be enabled. The environment handle in the call to <B>SQLSetEnvAttr</B> is set to null, which makes SQL_ATTR_CONNECTION_POOLING a process-level attribute. After connection pooling is enabled, the application then allocates an implicit shared environment by calling <B>SQLAllocHandle</B> with the <I>InputHandle</I> argument set to SQL_HANDLE_ENV.</P>
+#
+# <P>After connection pooling has been enabled and a shared environment has been selected for an application, SQL_ATTR_CONNECTION_POOLING cannot be reset for that environment, because <B>SQLSetEnvAttr</B> is called with a null environment handle when setting this attribute. If this attribute is set while connection pooling is already enabled on a shared environment, the attribute affects only shared environments that are allocated subsequently.</P>
+#
+# <P>For more information, see "<A HREF="odbcodbc_connection_pooling.htm">ODBC Connection Pooling</A>" in Chapter 6: Connecting to a Data Source or Driver.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_CONNECTION_POOLING => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_CP_OFF SQL_CP_ONE_PER_DRIVER SQL_CP_ONE_PER_HENV) ],
+ default => q(SQL_CP_OFF),
+ mode => 'rw',
+ order => ++$order,
+ },
+# <TR VALIGN="top">
+# <TD width=35%>SQL_ATTR_CP_MATCH<BR>
+# (ODBC 3.0)</TD>
+# <TD width=65%>A 32-bit SQLUINTEGER value that determines how a connection is chosen from a connection pool. When <B>SQLConnect</B> or <B>SQLDriverConnect</B> is called, the Driver Manager determines which connection is reused from the pool. The Driver Manager attempts to match the connection options in the call and the connection attributes set by the application to the keywords and connection attributes of the connections in the pool. The value of this attribute determines the level of precision of the matching criteria.
+# <P>The following values are used to set the value of this attribute:</P>
+#
+# <P>SQL_CP_STRICT_MATCH = Only connections that exactly match the connection options in the call and the connection attributes set by the application are reused. This is the default.</P>
+#
+# <P>SQL_CP_RELAXED_MATCH = Connections with matching connection string keywords can be used. Keywords must match, but not all connection attributes must match.</P>
+#
+# <P>For more information on how the Driver Manager performs the match in connecting to a pooled connection, see <A HREF="odbcsqlconnect.htm">SQLConnect</A>. For more information on connection pooling, see "<A HREF="odbcodbc_connection_pooling.htm">ODBC Connection Pooling</A>" in Chapter 6: Connecting to a Data Source or Driver.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_CP_MATCH => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_CP_STRICT_MATCH SQL_CP_RELAXED_MATCH) ],
+ default => q(SQL_CP_STRICT_MATCH),
+ mode => 'rw',
+ order => ++$order,
+ },
+# <TR VALIGN="top">
+# <TD width=35%>SQL_ATTR_ODBC_VERSION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=65%>A 32-bit integer that determines whether certain functionality exhibits ODBC 2<I>.x</I> behavior or ODBC 3<I>.x</I> behavior. The following values are used to set the value of this attribute:
+# <P>SQL_OV_ODBC3 = The Driver Manager and driver exhibit the following ODBC 3<I>.x</I> behavior:
+#
+# <UL type=disc>
+# <LI>The driver returns and expects ODBC 3<I>.x</I> codes for date, time, and timestamp.</li>
+#
+# <LI>The driver returns ODBC 3<I>.x</I> SQLSTATE codes when <B>SQLError</B>, <B>SQLGetDiagField</B>, or <B>SQLGetDiagRec</B> is called.</li>
+#
+# <LI>The <I>CatalogName</I> argument in a call to <B>SQLTables</B> accepts a search pattern.</li>
+# </UL>
+#
+# <P>SQL_OV_ODBC2 = The Driver Manager and driver exhibit the following ODBC 2<I>.x </I>behavior. This is especially useful for an ODBC 2<I>.x</I> application working with an ODBC 3<I>.x</I> driver.
+#
+# <UL type=disc>
+# <LI>The driver returns and expects ODBC 2<I>.x</I> codes for date, time, and timestamp.</li>
+#
+# <LI>The driver returns ODBC 2<I>.x</I> SQLSTATE codes when <B>SQLError</B>, <B>SQLGetDiagField</B>, or <B>SQLGetDiagRec</B> is called.</li>
+#
+# <LI>The <I>CatalogName</I> argument in a call to <B>SQLTables</B> does not accept a search pattern.</li>
+# </UL>
+#
+# <P>An application must set this environment attribute before calling any function that has an SQLHENV argument, or the call will return SQLSTATE HY010 (Function sequence error). It is driver-specific whether or not additional behaviors exist for these environmental flags.
+#
+# <UL type=disc>
+# <LI>For more information, see "<A HREF="odbcdeclaring_the_application_s_odbc_version.htm">Declaring the Application's ODBC Version</A>" in Chapter 6: Connecting to a Data Source or Driver and "<A HREF="odbcbehavioral_changes.htm">Behavioral Changes</A>" in Chapter 17: Programming Considerations.</li>
+# </UL>
+# </TD>
+# </TR>
+ SQL_ATTR_ODBC_VERSION => {
+ type => q(SQLINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_OV_ODBC3 SQL_OV_ODBC2) ],
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+# <TR VALIGN="top">
+# <TD width=35%>SQL_ATTR_OUTPUT_NTS<BR>
+# (ODBC 3.0)</TD>
+# <TD width=65%>A 32-bit integer that determines how the driver returns string data. If SQL_TRUE, the driver returns string data null-terminated. If SQL_FALSE, the driver does not return string data null-terminated.
+# <P>This attribute defaults to SQL_TRUE. A call to <B>SQLSetEnvAttr</B> to set it to SQL_TRUE returns SQL_SUCCESS. A call to <B>SQLSetEnvAttr</B> to set it to SQL_FALSE returns SQL_ERROR and SQLSTATE HYC00 (Optional feature not implemented).</P>
+# </TD>
+# </TR>
+ SQL_ATTR_OUTPUT_NTS => {
+ type => q(SQLINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_FALSE SQL_TRUE) ],
+ default => q(SQL_TRUE),
+ mode => 'rw',
+ order => ++$order,
+ }
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=48%>For information about</TH>
+# <TH width=52%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>Allocating a handle</TD>
+# <TD width=52%><A HREF="odbcsqlallochandle.htm">SQLAllocHandle</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>Returning the setting of an environment attribute</TD>
+# <TD width=52%><A HREF="odbcsqlgetenvattr.htm">SQLGetEnvAttr</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+};
+
+#
+# odbcsqlsetconnectattr.htm
+#
+my $attrDbc = {
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLSetConnectAttr</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqlsetconnectattr"></A>SQLSetConnectAttr</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 3.0<BR>
+# Standards Compliance: ISO 92</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLSetConnectAttr</B> sets attributes that govern aspects of connections.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;For more information about what the Driver Manager maps this function to when an ODBC 3<I>.x</I> application is working with an ODBC 2<I>.x</I> driver, see "<A HREF="odbcmapping_replacement_functions_for_backward_compatibility_of_applications.htm">Mapping Replacement Functions for Backward Compatibility of Applications</A>" in Chapter 17: Programming Considerations.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLSetConnectAttr</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHDBC&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>ConnectionHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLINTEGER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>Attribute</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLPOINTER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>ValuePtr</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLINTEGER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StringLength</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>ConnectionHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Connection handle.</dd>
+#
+# <DT><I>Attribute</I></DT>
+#
+# <DD>[Input]<BR>
+# Attribute to set, listed in "Comments."</dd>
+#
+# <DT><I>ValuePtr</I></DT>
+#
+# <DD>[Input]<BR>
+# Pointer to the value to be associated with <I>Attribute</I>. Depending on the value of <I>Attribute</I>, <I>ValuePtr</I> will be a 32-bit unsigned integer value or will point to a null-terminated character string. Note that if the <I>Attribute</I> argument is a driver-specific value, the value in <I>ValuePtr</I> may be a signed integer.</dd>
+#
+# <DT><I>StringLength</I></DT>
+#
+# <DD>[Input]<BR>
+# If <I>Attribute</I> is an ODBC-defined attribute and <I>ValuePtr</I> points to a character string or a binary buffer, this argument should be the length of *<I>ValuePtr</I>. If <I>Attribute</I> is an ODBC-defined attribute and <I>ValuePtr</I> is an integer, <I>StringLength</I> is ignored.
+#
+# <P>If <I>Attribute</I> is a driver-defined attribute, the application indicates the nature of the attribute to the Driver Manager by setting the <I>StringLength</I> argument. <I>StringLength</I> can have the following values:
+#
+#
+# <UL type=disc>
+# <LI>If <I>ValuePtr</I> is a pointer to a character string, then <I>StringLength</I> is the length of the string or SQL_NTS.</li>
+#
+# <LI>If <I>ValuePtr</I> is a pointer to a binary buffer, then the application places the result of the SQL_LEN_BINARY_ATTR(<I>length</I>) macro in <I>StringLength</I>. This places a negative value in <I>StringLength</I>.</li>
+#
+# <LI>If <I>ValuePtr</I> is a pointer to a value other than a character string or a binary string, then <I>StringLength</I> should have the value SQL_IS_POINTER.</li>
+#
+# <LI>If <I>ValuePtr</I> contains a fixed-length value, then <I>StringLength</I> is either SQL_IS_INTEGER or SQL_IS_UINTEGER, as appropriate.</li>
+# </UL>
+# </dd>
+# </DL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLSetConnectAttr</B> returns SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value can be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_DBC and a <I>Handle</I> of <I>ConnectionHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLSetConnectAttr</B> and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise.</P>
+#
+# <P>The driver can return SQL_SUCCESS_WITH_INFO to provide information about the result of setting an option.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=22%>SQLSTATE</TH>
+# <TH width=26%>Error</TH>
+# <TH width=52%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01000</TD>
+# <TD width=26%>General warning</TD>
+# <TD width=52%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01S02</TD>
+# <TD width=26%>Option value changed</TD>
+# <TD width=52%>The driver did not support the value specified in <I>ValuePtr</I> and substituted a similar value. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>08002</TD>
+# <TD width=26%>Connection name in use</TD>
+# <TD width=52%>The <I>Attribute</I> argument was SQL_ATTR_ODBC_CURSORS, and the driver was already connected to the data source.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>08003</TD>
+# <TD width=26%>Connection does not exist</TD>
+# <TD width=52%>(DM) An <I>Attribute</I> value was specified that required an open connection, but the <I>ConnectionHandle</I> was not in a connected state.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>08S01</TD>
+# <TD width=26%>Communication link failure</TD>
+# <TD width=52%>The communication link between the driver and the data source to which the driver was connected failed before the function completed processing.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>24000</TD>
+# <TD width=26%>Invalid cursor state</TD>
+# <TD width=52%>The <I>Attribute</I> argument was SQL_ATTR_CURRENT_CATALOG, and a result set was pending.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>3D000</TD>
+# <TD width=26%>Invalid catalog name</TD>
+# <TD width=52%>The <I>Attribute</I> argument was SQL_CURRENT_CATALOG, and the specified catalog name was invalid.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY000</TD>
+# <TD width=26%>General error</TD>
+# <TD width=52%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY001</TD>
+# <TD width=26%>Memory allocation error</TD>
+# <TD width=52%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY009</TD>
+# <TD width=26%>Invalid use of null pointer</TD>
+# <TD width=52%>The <I>Attribute</I> argument identified a connection attribute that required a string value, and the <I>ValuePtr </I>argument was a null pointer.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY010</TD>
+# <TD width=26%>Function sequence error</TD>
+# <TD width=52%>(DM) An asynchronously executing function was called for a <I>StatementHandle</I> associated with the <I>ConnectionHandle</I> and was still executing when <B>SQLSetConnectAttr</B> was called.
+# <P>(DM) <B>SQLExecute</B>, <B>SQLExecDirect</B>, <B>SQLBulkOperations</B>, or <B>SQLSetPos</B> was called for a <I>StatementHandle</I> associated with the <I>ConnectionHandle</I> and returned SQL_NEED_DATA. This function was called before data was sent for all data-at-execution parameters or columns.</P>
+#
+# <P>(DM) <B>SQLBrowseConnect</B> was called for the <I>ConnectionHandle</I> and returned SQL_NEED_DATA. This function was called before <B>SQLBrowseConnect</B> returned SQL_SUCCESS_WITH_INFO or SQL_SUCCESS.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY011</TD>
+# <TD width=26%>Attribute cannot be set now</TD>
+# <TD width=52%>The <I>Attribute</I> argument was SQL_ATTR_TXN_ISOLATION, and a transaction was open.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY013</TD>
+# <TD width=26%>Memory management error</TD>
+# <TD width=52%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY024</TD>
+# <TD width=26%>Invalid attribute value</TD>
+# <TD width=52%>Given the specified <I>Attribute</I> value, an invalid value was specified in <I>ValuePtr</I>. (The Driver Manager returns this SQLSTATE only for connection and statement attributes that accept a discrete set of values, such as SQL_ATTR_ACCESS_MODE or SQL_ATTR_ASYNC_ENABLE. For all other connection and statement attributes, the driver must verify the value specified in <I>ValuePtr</I>.)
+# <P>The <I>Attribute</I> argument was SQL_ATTR_TRACEFILE or SQL_ATTR_TRANSLATE_LIB, and <I>ValuePtr</I> was an empty string.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY090</TD>
+# <TD width=26%>Invalid string or buffer length</TD>
+# <TD width=52%><I>(DM) *ValuePtr </I>is a character string, and the <I>StringLength</I> argument was less than 0 but was not SQL_NTS.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY092</TD>
+# <TD width=26%>Invalid attribute/option identifier</TD>
+# <TD width=52%>(DM) The value specified for the argument <I>Attribute</I> was not valid for the version of ODBC supported by the driver.
+# <P>(DM) The value specified for the argument <I>Attribute</I> was a read-only attribute.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYC00</TD>
+# <TD width=26%>Optional feature not implemented</TD>
+# <TD width=52%>The value specified for the argument <I>Attribute</I> was a valid ODBC connection or statement attribute for the version of ODBC supported by the driver but was not supported by the driver.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT01</TD>
+# <TD width=26%>Connection timeout expired</TD>
+# <TD width=52%>The connection timeout period expired before the data source responded to the request. The connection timeout period is set through <B>SQLSetConnectAttr</B>, SQL_ATTR_CONNECTION_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>IM001</TD>
+# <TD width=26%>Driver does not support this function</TD>
+# <TD width=52%>(DM) The driver associated with the <I>ConnectionHandle</I> does not support the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>IM009</TD>
+# <TD width=26%>Unable to load translation DLL</TD>
+# <TD width=52%>The driver was unable to load the translation DLL that was specified for the connection. This error can be returned only when <I>Attribute</I> is SQL_ATTR_TRANSLATE_LIB.</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P>When <I>Attribute</I> is a statement attribute, <B>SQLSetConnectAttr</B> can return any SQLSTATEs returned by <B>SQLSetStmtAttr</B>.</P>
+#
+# <P class="label"><B>Comments</B></P>
+#
+# <P>For general information about connection attributes, see "<A HREF="odbcconnection_attributes.htm">Connection Attributes</A>" in Chapter 6: Connecting to a Data Source or Driver.</P>
+#
+# <P>The currently defined attributes and the version of ODBC in which they were introduced are shown in the table later in this section; it is expected that more attributes will be defined to take advantage of different data sources. A range of attributes is reserved by ODBC; driver developers must reserve values for their own driver-specific use from X/Open.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;The ability to set statement attributes at the connection level by calling <B>SQLSetConnectAttr</B> has been deprecated in ODBC 3<I>.x</I>. ODBC 3<I>.x</I> applications should never set statement attributes at the connection level. ODBC 3<I>.x</I> statement attributes cannot be set at the connection level, with the exception of the SQL_ATTR_METADATA_ID and SQL_ATTR_ASYNC_ENABLE attributes, which are both connection attributes and statement attributes and can be set at either the connection level or the statement level.</P>
+#
+# <P class="indent">ODBC 3<I>.x</I> drivers need only support this functionality if they should work with ODBC 2<I>.x</I> applications that set ODBC 2<I>.x</I> statement options at the connection level. For more information, see "<A HREF="odbcsqlsetconnectoption_mapping.htm">SQLSetConnectOption Mapping</A>" in Appendix G: Driver Guidelines for Backward Compatibility.</P>
+#
+# <P>An application can call <B>SQLSetConnectAttr</B> at any time between the time the connection is allocated and freed. All connection and statement attributes successfully set by the application for the connection persist until <B>SQLFreeHandle</B> is called on the connection. For example, if an application calls <B>SQLSetConnectAttr</B> before connecting to a data source, the attribute persists even if <B>SQLSetConnectAttr</B> fails in the driver when the application connects to the data source; if an application sets a driver-specific attribute, the attribute persists even if the application connects to a different driver on the connection.</P>
+#
+# <P>Some connection attributes can be set only before a connection has been made; others can be set only after a connection has been made. The following table indicates those connection attributes that must be set either before or after a connection has been made. <I>Either</I> indicates that the attribute can be set either before or after connection.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=50%>Attribute</TH>
+# <TH width=50%>Set before or after connection?</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_ACCESS_MODE</TD>
+# <TD width=50%>Either<SUP>[1]</SUP></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_ASYNC_ENABLE</TD>
+# <TD width=50%>Either<SUP>[2]</SUP></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_AUTOCOMMIT</TD>
+# <TD width=50%>Either</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_CONNECTION_TIMEOUT</TD>
+# <TD width=50%>Either</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_CURRENT_CATALOG</TD>
+# <TD width=50%>Either<SUP>[1]</SUP></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_LOGIN_TIMEOUT</TD>
+# <TD width=50%>Before</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_METADATA_ID</TD>
+# <TD width=50%>Either</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_ODBC_CURSORS</TD>
+# <TD width=50%>Before</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_PACKET_SIZE</TD>
+# <TD width=50%>Before</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_QUIET_MODE</TD>
+# <TD width=50%>Either</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_TRACE</TD>
+# <TD width=50%>Either</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_TRACEFILE</TD>
+# <TD width=50%>Either</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_TRANSLATE_LIB</TD>
+# <TD width=50%>After</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_TRANSLATE_OPTION</TD>
+# <TD width=50%>After</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>SQL_ATTR_TXN_ISOLATION</TD>
+# <TD width=50%>Either<SUP>[3]</SUP></TD>
+# </TR>
+# </table></div>
+#
+# <P class="fineprint">[1]&nbsp;&nbsp;&nbsp;SQL_ATTR_ACCESS_MODE and SQL_ATTR_CURRENT_CATALOG can be set before or after connecting, depending on the driver. However, interoperable applications set them before connecting because some drivers do not support changing these after connecting.</p>
+# <P class="fineprint">[2]&nbsp;&nbsp;&nbsp;SQL_ATTR_ASYNC_ENABLE must be set before there is an active statement.</p>
+# <P class="fineprint">[3]&nbsp;&nbsp;&nbsp;SQL_ATTR_TXN_ISOLATION can be set only if there are no open transactions on the connection. Some connection attributes support substitution of a similar value if the data source does not support the value specified in *<I>ValuePtr</I>. In such cases, the driver returns SQL_SUCCESS_WITH_INFO and SQLSTATE 01S02 (Option value changed). For example, if <I>Attribute</I> is SQL_ATTR_PACKET_SIZE and *<I>ValuePtr</I> exceeds the maximum packet size, the driver substitutes the maximum size. To determine the substituted value, an application calls <B>SQLGetConnectAttr</B>.</p>
+# <P>The format of information set in the *<I>ValuePtr</I> buffer depends on the specified <I>Attribute</I>. <B>SQLSetConnectAttr</B> will accept attribute information in one of two different formats: a null-terminated character string or a 32-bit integer value. The format of each is noted in the attribute's description. Character strings pointed to by the <I>ValuePtr</I> argument of <B>SQLSetConnectAttr</B> have a length of <I>StringLength</I> bytes.</P>
+#
+# <P>The <I>StringLength</I> argument is ignored if the length is defined by the attribute, as is the case for all attributes introduced in ODBC 2<I>.x</I> or earlier.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=38%><I>Attribute</I></TH>
+# <TH width=62%><I>ValuePtr</I> contents</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ACCESS_MODE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value. SQL_MODE_READ_ONLY is used by the driver or data source as an indicator that the connection is not required to support SQL statements that cause updates to occur. This mode can be used to optimize locking strategies, transaction management, or other areas as appropriate to the driver or data source. The driver is not required to prevent such statements from being submitted to the data source. The behavior of the driver and data source when asked to process SQL statements that are not read-only during a read-only connection is implementation-defined. SQL_MODE_READ_WRITE is the default.</TD>
+# </TR>
+ SQL_ATTR_ACCESS_MODE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_MODE_READ_ONLY SQL_MODE_READ_WRITE) ],
+ default => q(SQL_MODE_READ_WRITE),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ASYNC_ENABLE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies whether a function called with a statement on the specified connection is executed asynchronously:
+# <P>SQL_ASYNC_ENABLE_OFF = Off (the default)<BR>
+# SQL_ASYNC_ENABLE_ON = On</P>
+#
+# <P>Setting SQL_ASYNC_ENABLE_ON enables asynchronous execution for all future statement handles allocated on this connection. It is driver-defined whether this enables asynchronous execution for existing statement handles associated with this connection. An error is returned if asynchronous execution is enabled while there is an active statement on the connection.</P>
+#
+# <P>This attribute can be set whether <B>SQLGetInfo</B> with the SQL_ASYNC_MODE information type returns SQL_AM_CONNECTION or SQL_AM_STATEMENT.</P>
+#
+# <P>After a function has been called asynchronously, only the original function, <B>SQLAllocHandle</B>, <B>SQLCancel</B>, <B>SQLGetDiagField</B>, or <B>SQLGetDiagRec </B>can be called on the statement or the connection associated with <I>StatementHandle</I>, until the original function returns a code other than SQL_STILL_EXECUTING. Any other function called on <I>StatementHandle</I> or the connection associated with <I>StatementHandle</I> returns SQL_ERROR with an SQLSTATE of HY010 (Function sequence error). Functions can be called on other statements. For more information, see "<A HREF="odbcasynchronous_execution.htm">Asynchronous Execution</A>" in Chapter 9: Executing Statements.</P>
+#
+# <P>In general, applications should execute functions asynchronously only on single-thread operating systems. On multithread operating systems, applications should execute functions on separate threads rather than executing them asynchronously on the same thread. Drivers that operate only on multithread operating systems do not need to support asynchronous execution.</P>
+#
+# <P>The following functions can be executed asynchronously:</P>
+#
+# <P><B>SQLBulkOperations<BR>
+# SQLColAttribute</B><BR>
+# <B>SQLColumnPrivileges</B><BR>
+# <B>SQLColumns</B><BR>
+# <B>SQLCopyDesc</B><BR>
+# <B>SQLDescribeCol</B><BR>
+# <B>SQLDescribeParam</B><BR>
+# <B>SQLExecDirect</B><BR>
+# <B>SQLExecute</B><BR>
+# <B>SQLFetch</B><BR>
+# <B>SQLFetchScroll</B><BR>
+# <B>SQLForeignKeys</B><BR>
+# <B>SQLGetData</B><BR>
+# <B>SQLGetDescField</B><SUP>[1]<BR>
+# </SUP><B>SQLGetDescRec</B><SUP>[1]</SUP><B><BR>
+# SQLGetDiagField</B><BR>
+# <B>SQLGetDiagRec<BR>
+# SQLGetTypeInfo</B><BR>
+# <B>SQLMoreResults</B><BR>
+# <B>SQLNumParams</B><BR>
+# <B>SQLNumResultCols</B><BR>
+# <B>SQLParamData</B><BR>
+# <B>SQLPrepare</B><BR>
+# <B>SQLPrimaryKeys</B><BR>
+# <B>SQLProcedureColumns</B><BR>
+# <B>SQLProcedures</B><BR>
+# <B>SQLPutData</B><BR>
+# <B>SQLSetPos</B><BR>
+# <B>SQLSpecialColumns</B><BR>
+# <B>SQLStatistics</B><BR>
+# <B>SQLTablePrivileges</B><BR>
+# <B>SQLTables</B></P>
+# </TD>
+# </TR>
+ SQL_ASYNC_ENABLE_ON => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_ASYNC_ENABLE_OFF SQL_ASYNC_ENABLE_ON) ],
+ default => q(SQL_ASYNC_ENABLE_OFF),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_AUTO_IPD<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>A read-only SQLUINTEGER value that specifies whether automatic population of the IPD after a call to <B>SQLPrepare </B>is supported:
+# <P>SQL_TRUE = Automatic population of the IPD after a call to <B>SQLPrepare </B>is supported by the driver.</P>
+#
+# <P>SQL_FALSE = Automatic population of the IPD after a call to <B>SQLPrepare </B>is not supported by the driver. Servers that do not support prepared statements will not be able to populate the IPD automatically. </P>
+#
+# <P>If SQL_TRUE is returned for the SQL_ATTR_AUTO_IPD connection attribute, the statement attribute SQL_ATTR_ENABLE_AUTO_IPD can be set to turn automatic population of the IPD on or off. If SQL_ATTR_AUTO_IPD is SQL_FALSE, SQL_ATTR_ENABLE_AUTO_IPD cannot be set to SQL_TRUE. The default value of SQL_ATTR_ENABLE_AUTO_IPD is equal to the value of SQL_ATTR_AUTO_IPD.</P>
+#
+# <P>This connection attribute can be returned by <B>SQLGetConnectAttr</B> but cannot be set by <B>SQLSetConnectAttr</B>.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_AUTO_IPD => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_FALSE SQL_TRUE) ],
+ default => undef,
+ mode => 'ro',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_AUTOCOMMIT<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies whether to use autocommit or manual-commit mode:
+# <P>SQL_AUTOCOMMIT_OFF = The driver uses manual-commit mode, and the application must explicitly commit or roll back transactions with <B>SQLEndTran</B>.</P>
+#
+# <P>SQL_AUTOCOMMIT_ON = The driver uses autocommit mode. Each statement is committed immediately after it is executed. This is the default. Any open transactions on the connection are committed when SQL_ATTR_AUTOCOMMIT is set to SQL_AUTOCOMMIT_ON to change from manual-commit mode to autocommit mode.</P>
+#
+# <P>For more information, see "<A HREF="odbccommit_mode.htm">Commit Mode</A>" in Chapter 14: Transactions.</P>
+#
+# <P class="indent"><b class="le">Important&nbsp;&nbsp;&nbsp;</b>Some data sources delete the access plans and close the cursors for all statements on a connection each time a statement is committed; autocommit mode can cause this to happen after each nonquery statement is executed or when the cursor is closed for a query. For more information, see the SQL_CURSOR_COMMIT_BEHAVIOR and SQL_CURSOR_ROLLBACK_BEHAVIOR information types in <A HREF="odbcsqlgetinfo.htm">SQLGetInfo</A> and "<A HREF="odbceffect_of_transactions_on_cursors_and_prepared_statements.htm">Effect of Transactions on Cursors and Prepared Statements</A>" in Chapter 14: Transactions.</P>
+#
+# <P>When a batch is executed in autocommit mode, two things are possible. The entire batch can be treated as an autocommitable unit, or each statement in a batch is treated as an autocommitable unit. Certain data sources can support both these behaviors and may provide a way of choosing one or the other. It is driver-defined whether a batch is treated as an autocommitable unit or whether each individual statement within the batch is autocommitable.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_AUTOCOMMIT => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_AUTOCOMMIT_OFF SQL_AUTOCOMMIT_ON) ],
+ default => q(SQL_AUTOCOMMIT_ON),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_CONNECTION_DEAD
+# <P>(ODBC 3.5)</P>
+# </TD>
+# <TD width=62%>An SQLUINTERGER value that indicates the state of the connection. If SQL_CD_TRUE, the connection has been lost. If SQL_CD_FALSE, the connection is still active.</TD>
+# </TR>
+ SQL_ATTR_CONNECTION_DEAD => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_CD_FALSE SQL_CD_TRUE) ],
+ default => undef,
+ mode => 'ro',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_CONNECTION_TIMEOUT<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value corresponding to the number of seconds to wait for any request on the connection to complete before returning to the application. The driver should return SQLSTATE HYT00 (Timeout expired) anytime that it is possible to time out in a situation not associated with query execution or login.
+# <P>If <I>ValuePtr</I> is equal to 0 (the default), there is no timeout.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_CONNECTION_TIMEOUT => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => 0,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_CURRENT_CATALOG<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>A character string containing the name of the catalog to be used by the data source. For example, in SQL Server, the catalog is a database, so the driver sends a <B>USE</B> <I>database</I> statement to the data source, where <I>database</I> is the database specified in *<I>ValuePtr</I>. For a single-tier driver, the catalog might be a directory, so the driver changes its current directory to the directory specified in *<I>ValuePtr</I>.</TD>
+# </TR>
+ SQL_ATTR_CURRENT_CATALOG => {
+ type => q(SQLCHAR),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_LOGIN_TIMEOUT<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value corresponding to the number of seconds to wait for a login request to complete before returning to the application. The default is driver-dependent. If <I>ValuePtr</I> is 0, the timeout is disabled and a connection attempt will wait indefinitely.
+# <P>If the specified timeout exceeds the maximum login timeout in the data source, the driver substitutes that value and returns SQLSTATE 01S02 (Option value changed).</P>
+# </TD>
+# </TR>
+ SQL_ATTR_LOGIN_TIMEOUT => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => 0,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_METADATA_ID<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that determines how the string arguments of catalog functions are treated.
+# <P>If SQL_TRUE, the string argument of catalog functions are treated as identifiers. The case is not significant. For nondelimited strings, the driver removes any trailing spaces and the string is folded to uppercase. For delimited strings, the driver removes any leading or trailing spaces and takes literally whatever is between the delimiters. If one of these arguments is set to a null pointer, the function returns SQL_ERROR and SQLSTATE HY009 (Invalid use of null pointer). </P>
+#
+# <P>If SQL_FALSE, the string arguments of catalog functions are not treated as identifiers. The case is significant. They can either contain a string search pattern or not, depending on the argument.</P>
+#
+# <P>The default value is SQL_FALSE.</P>
+#
+# <P>The <I>TableType</I> argument of <B>SQLTables</B>, which takes a list of values, is not affected by this attribute.</P>
+#
+# <P>SQL_ATTR_METADATA_ID can also be set on the statement level. (It is the only connection attribute that is also a statement attribute.)</P>
+#
+# <P>For more information, see "<A HREF="odbcarguments_in_catalog_functions.htm">Arguments in Catalog Functions</A>" in Chapter 7: Catalog Functions.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_METADATA_ID => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_FALSE SQL_TRUE) ],
+ default => q(SQL_FALSE),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ODBC_CURSORS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>An SQLUINTEGER value specifying how the Driver Manager uses the ODBC cursor library:
+# <P>SQL_CUR_USE_IF_NEEDED = The Driver Manager uses the ODBC cursor library only if it is needed. If the driver supports the SQL_FETCH_PRIOR option in <B>SQLFetchScroll</B>, the Driver Manager uses the scrolling capabilities of the driver. Otherwise, it uses the ODBC cursor library.</P>
+#
+# <P>SQL_CUR_USE_ODBC = The Driver Manager uses the ODBC cursor library.</P>
+#
+# <P>SQL_CUR_USE_DRIVER = The Driver Manager uses the scrolling capabilities of the driver. This is the default setting.</P>
+#
+# <P>For more information about the ODBC cursor library, see <A HREF="odbcodbc_cursor_library.htm">Appendix F: ODBC Cursor Library</A>.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_ODBC_CURSORS => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_CUR_USE_IF_NEEDED SQL_CUR_USE_ODBC SQL_CUR_USE_DRIVER) ],
+ default => q(SQL_CUR_USE_DRIVER),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_PACKET_SIZE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>An SQLUINTEGER value specifying the network packet size in bytes.
+# <P class="indent"><B>Note</B>&nbsp;&nbsp;&nbsp;Many data sources either do not support this option or only can return but not set the network packet size.</P>
+#
+# <P>If the specified size exceeds the maximum packet size or is smaller than the minimum packet size, the driver substitutes that value and returns SQLSTATE 01S02 (Option value changed).</P>
+#
+# <P>If the application sets packet size after a connection has already been made, the driver will return SQLSTATE HY011 (Attribute cannot be set now).</P>
+# </TD>
+# </TR>
+ SQL_ATTR_PACKET_SIZE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_QUIET_MODE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>A 32-bit window handle (<I>hwnd</I>).
+# <P>If the window handle is a null pointer, the driver does not display any dialog boxes.</P>
+#
+# <P>If the window handle is not a null pointer, it should be the parent window handle of the application. This is the default. The driver uses this handle to display dialog boxes.</P>
+#
+# <P class="indent"><b class="le">Note&nbsp;&nbsp;&nbsp;</b>The SQL_ATTR_QUIET_MODE connection attribute does not apply to dialog boxes displayed by <B>SQLDriverConnect</B>.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_QUIET_MODE => {
+ type => q(SQLPOINTER),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_TRACE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value telling the Driver Manager whether to perform tracing:
+# <P>SQL_OPT_TRACE_OFF = Tracing off (the default)</P>
+#
+# <P>SQL_OPT_TRACE_ON = Tracing on</P>
+#
+# <P>When tracing is on, the Driver Manager writes each ODBC function call to the trace file.</P>
+#
+# <P class="indent"><b class="le">Note&nbsp;&nbsp;&nbsp;</b>When tracing is on, the Driver Manager can return SQLSTATE IM013 (Trace file error) from any function.</P>
+#
+# <P>An application specifies a trace file with the SQL_ATTR_TRACEFILE option. If the file already exists, the Driver Manager appends to the file. Otherwise, it creates the file. If tracing is on and no trace file has been specified, the Driver Manager writes to the file SQL.LOG in the root directory. </P>
+#
+# <P>An application can set the variable <B>ODBCSharedTraceFlag</B> to enable tracing dynamically. Tracing is then enabled for all ODBC applications currently running. If an application turns tracing off, it is turned off only for that application.</P>
+#
+# <P>If the <B>Trace</B> keyword in the system information is set to 1 when an application calls <B>SQLAllocHandle</B> with a <I>HandleType</I> of SQL_HANDLE_ENV, tracing is enabled for all handles. It is enabled only for the application that called <B>SQLAllocHandle</B>.</P>
+#
+# <P>Calling <B>SQLSetConnectAttr</B> with an <I>Attribute</I> of SQL_ATTR_TRACE does not require that the <I>ConnectionHandle</I> argument be valid and will not return SQL_ERROR if <I>ConnectionHandle</I> is NULL. This attribute applies to all connections.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_TRACE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_OPT_TRACE_OFF SQL_OPT_TRACE_ON) ],
+ default => q(SQL_OPT_TRACE_OFF),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_TRACEFILE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>A null-terminated character string containing the name of the trace file.
+# <P>The default value of the SQL_ATTR_TRACEFILE attribute is specified with the <B>TraceFile</B> keyword in the system information. For more information, see "<A HREF="odbcodbc_subkey.htm">ODBC Subkey</A>" in Chapter 19: Configuring Data Sources.</P>
+#
+# <P>Calling <B>SQLSetConnectAttr</B> with an <I>Attribute</I> of SQL_ATTR_ TRACEFILE does not require the <I>ConnectionHandle</I> argument to be valid and will not return SQL_ERROR if <I>ConnectionHandle</I> is invalid. This attribute applies to all connections.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_TRACEFILE => {
+ type => q(SQLCHAR),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_TRANSLATE_LIB<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>A null-terminated character string containing the name of a library containing the functions <B>SQLDriverToDataSource</B> and <B>SQLDataSourceToDriver</B> that the driver accesses to perform tasks such as character set translation. This option may be specified only if the driver has connected to the data source. The setting of this attribute will persist across connections. For more information about translating data, see "<A HREF="odbctranslation_dlls.htm">Translation DLLs</A>" in Chapter 17: Programming Considerations, and <A HREF="odbctranslation_dll_function_reference.htm">Chapter 24: Translation DLL Function Reference</A>.</TD>
+# </TR>
+ SQL_ATTR_TRANSLATE_LIB => {
+ type => q(SQLCHAR),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_TRANSLATE_OPTION<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>A 32-bit flag value that is passed to the translation DLL. This attribute can be specified only if the driver has connected to the data source. For information about translating data, see "<A HREF="odbctranslation_dlls.htm">Translation DLLs</A>" in Chapter 17: Programming Considerations.</TD>
+# </TR>
+ SQL_ATTR_TRANSLATE_OPTION => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_TXN_ISOLATION<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>A 32-bit bitmask that sets the transaction isolation level for the current connection. An application must call <B>SQLEndTran</B> to commit or roll back all open transactions on a connection, before calling <B>SQLSetConnectAttr</B> with this option.
+# <P>The valid values for <I>ValuePtr</I> can be determined by calling <B>SQLGetInfo</B> with <I>InfoType</I> equal to SQL_TXN_ISOLATION_OPTIONS.</P>
+#
+# <P>For a description of transaction isolation levels, see the description of the SQL_DEFAULT_TXN_ISOLATION information type in <B>SQLGetInfo</B> and "<A HREF="odbctransaction_isolation_levels.htm">Transaction Isolation Levels</A>" in Chapter 14: Transactions.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_TXN_ISOLATION => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+# </table></div>
+#
+# <P class="fineprint">[1]&nbsp;&nbsp;&nbsp;These functions can be called asynchronously only if the descriptor is an implementation descriptor, not an application descriptor.</p>
+# <P class="label"><B>Code Example</B></P>
+#
+# <P>See <A HREF="odbcsqlconnect.htm">SQLConnect</A>.</P>
+#
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=48%>For information about</TH>
+# <TH width=52%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>Allocating a handle</TD>
+# <TD width=52%><A HREF="odbcsqlallochandle.htm">SQLAllocHandle</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>Returning the setting of a connection <BR>
+# attribute</TD>
+# <TD width=52%><A HREF="odbcsqlgetconnectattr.htm">SQLGetConnectAttr</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+};
+
+#
+# odbcsqlsetstmtattr.htm
+#
+my $attrStmt = {
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLSetStmtAttr</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqlsetstmtattr"></A>SQLSetStmtAttr</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 3.0<BR>
+# Standards Compliance: ISO 92</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLSetStmtAttr</B> sets attributes related to a statement.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;For more information about what the Driver Manager maps this function to when an ODBC 3<I>.x</I> application is working with an ODBC 2<I>.x</I> driver, see "<A HREF="odbcmapping_replacement_functions_for_backward_compatibility_of_applications.htm">Mapping Replacement Functions for Backward Compatibility of Applications</A>" in Chapter 17: Programming Considerations.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLSetStmtAttr</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHSTMT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StatementHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLINTEGER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>Attribute</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLPOINTER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>ValuePtr</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLINTEGER&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StringLength</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>StatementHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Statement handle.</dd>
+#
+# <DT><I>Attribute</I></DT>
+#
+# <DD>[Input]<BR>
+# Option to set, listed in "Comments."</dd>
+#
+# <DT><I>ValuePtr</I></DT>
+#
+# <DD>[Input]<BR>
+# Pointer to the value to be associated with <I>Attribute</I>. Depending on the value of <I>Attribute</I>, <I>ValuePtr</I> will be a 32-bit unsigned integer value or a pointer to a null-terminated character string, a binary buffer, or a driver-defined value. If the <I>Attribute</I> argument is a driver-specific value, <I>ValuePtr</I> may be a signed integer.</dd>
+#
+# <DT><I>StringLength</I></DT>
+#
+# <DD>[Input]<BR>
+# If <I>Attribute</I> is an ODBC-defined attribute and <I>ValuePtr</I> points to a character string or a binary buffer, this argument should be the length of *<I>ValuePtr</I>. If <I>Attribute</I> is an ODBC-defined attribute and <I>ValuePtr</I> is an integer, <I>StringLength</I> is ignored.
+#
+# <P>If <I>Attribute</I> is a driver-defined attribute, the application indicates the nature of the attribute to the Driver Manager by setting the <I>StringLength</I> argument. <I>StringLength</I> can have the following values:
+# </dd>
+# </DL>
+#
+# <UL type=disc>
+# <LI>If <I>ValuePtr</I> is a pointer to a character string, then <I>StringLength</I> is the length of the string or SQL_NTS.</li>
+#
+# <LI>If <I>ValuePtr</I> is a pointer to a binary buffer, then the application places the result of the SQL_LEN_BINARY_ATTR(<I>length</I>) macro in <I>StringLength</I>. This places a negative value in <I>StringLength</I>.</li>
+#
+# <LI>If <I>ValuePtr</I> is a pointer to a value other than a character string or a binary string, then <I>StringLength</I> should have the value SQL_IS_POINTER. </li>
+#
+# <LI>If <I>ValuePtr</I> contains a fixed-length value, then <I>StringLength</I> is either SQL_IS_INTEGER or SQL_IS_UINTEGER, as appropriate.</li>
+# </UL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLSetStmtAttr</B> returns SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value may be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_STMT and a <I>Handle</I> of <I>StatementHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLSetStmtAttr</B> and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=22%>SQLSTATE</TH>
+# <TH width=26%>Error</TH>
+# <TH width=52%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01000</TD>
+# <TD width=26%>General warning</TD>
+# <TD width=52%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01S02</TD>
+# <TD width=26%>Option value changed</TD>
+# <TD width=52%>The driver did not support the value specified in <I>ValuePtr</I>, or the value specified in <I>ValuePtr</I> was invalid because of implementation working conditions, so the driver substituted a similar value. (<B>SQLGetStmtAttr</B> can be called to determine the temporarily substituted value.) The substitute value is valid for the <I>StatementHandle</I> until the cursor is closed, at which point the statement attribute reverts to its previous value. The statement attributes that can be changed are:
+# <P>SQL_ ATTR_CONCURRENCY<BR>
+# SQL_ ATTR_CURSOR_TYPE<BR>
+# SQL_ ATTR_KEYSET_SIZE<BR>
+# SQL_ ATTR_MAX_LENGTH<BR>
+# SQL_ ATTR_MAX_ROWS<BR>
+# SQL_ ATTR_QUERY_TIMEOUT <BR>
+# SQL_ATTR_ROW_ARRAY_SIZE<BR>
+# SQL_ ATTR_SIMULATE_CURSOR </P>
+#
+# <P>(Function returns SQL_SUCCESS_WITH_INFO.)</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>08S01</TD>
+# <TD width=26%>Communication link failure</TD>
+# <TD width=52%>The communication link between the driver and the data source to which the driver was connected failed before the function completed processing.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>24000</TD>
+# <TD width=26%>Invalid cursor state</TD>
+# <TD width=52%>The <I>Attribute</I> was SQL_ATTR_CONCURRENCY, SQL_ATTR_CURSOR_TYPE, SQL_ATTR_SIMULATE_CURSOR, or SQL_ATTR_USE_BOOKMARKS, and the cursor was open.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY000</TD>
+# <TD width=26%>General error</TD>
+# <TD width=52%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY001</TD>
+# <TD width=26%>Memory allocation <BR>
+# error</TD>
+# <TD width=52%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY009</TD>
+# <TD width=26%>Invalid use of null pointer</TD>
+# <TD width=52%>The <I>Attribute</I> argument identified a statement attribute that required a string attribute, and the <I>ValuePtr </I>argument was a null pointer.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY010</TD>
+# <TD width=26%>Function sequence error</TD>
+# <TD width=52%>(DM) An asynchronously executing function was called for the <I>StatementHandle</I> and was still executing when this function was called.
+# <P>(DM) <B>SQLExecute</B>, <B>SQLExecDirect</B>, <B>SQLBulkOperations</B>, or <B>SQLSetPos</B> was called for the <I>StatementHandle</I> and returned SQL_NEED_DATA. This function was called before data was sent for all data-at-execution parameters or columns.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY011</TD>
+# <TD width=26%>Attribute cannot be set now</TD>
+# <TD width=52%>The <I>Attribute</I> was SQL_ATTR_CONCURRENCY, SQL_ ATTR_CURSOR_TYPE, SQL_ ATTR_SIMULATE_CURSOR, or SQL_ ATTR_USE_BOOKMARKS, and the statement was prepared.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY013</TD>
+# <TD width=26%>Memory management error</TD>
+# <TD width=52%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY017</TD>
+# <TD width=26%>Invalid use of an automatically allocated descriptor handle</TD>
+# <TD width=52%>(DM) The <I>Attribute</I> argument was SQL_ATTR_IMP_ROW_DESC or SQL_ATTR_IMP_PARAM_DESC.
+# <P>(DM) The <I>Attribute</I> argument was SQL_ATTR_APP_ROW_DESC or SQL_ATTR_APP_PARAM_DESC, and the value in <I>ValuePtr</I> was an implicitly allocated descriptor handle other than the handle originally allocated for the ARD or APD.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY024</TD>
+# <TD width=26%>Invalid attribute value</TD>
+# <TD width=52%>Given the specified <I>Attribute</I> value, an invalid value was specified in <I>ValuePtr</I>. (The Driver Manager returns this SQLSTATE only for connection and statement attributes that accept a discrete set of values, such as SQL_ATTR_ACCESS_MODE or SQL_ ATTR_ASYNC_ENABLE. For all other connection and statement attributes, the driver must verify the value specified in <I>ValuePtr</I>.)
+# <P>The <I>Attribute</I> argument was SQL_ATTR_APP_ROW_DESC or SQL_ATTR_APP_PARAM_DESC, and <I>ValuePtr</I> was an explicitly allocated descriptor handle that is not on the same connection as the <I>StatementHandle</I> argument.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY090</TD>
+# <TD width=26%>Invalid string or buffer length</TD>
+# <TD width=52%>(DM) <I>*ValuePtr</I> is a character string, and the <I>StringLength</I> argument was less than 0 but was not SQL_NTS.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY092</TD>
+# <TD width=26%>Invalid attribute/option identifier</TD>
+# <TD width=52%>(DM) The value specified for the argument <I>Attribute</I> was not valid for the version of ODBC supported by the driver.
+# <P>(DM) The value specified for the argument <I>Attribute</I> was a read-only attribute.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYC00</TD>
+# <TD width=26%>Optional feature not implemented</TD>
+# <TD width=52%>The value specified for the argument <I>Attribute</I> was a valid ODBC statement attribute for the version of ODBC supported by the driver but was not supported by the driver.
+# <P>The <I>Attribute</I> argument was SQL_ATTR_ASYNC_ENABLE, and a call to <B>SQLGetInfo</B> with an <I>InfoType</I> of SQL_ASYNC_MODE returns SQL_AM_CONNECTION.</P>
+#
+# <P>The <I>Attribute</I> argument was SQL_ATTR_ENABLE_AUTO_IPD, and the value of the connection attribute SQL_ATTR_AUTO_IPD was SQL_FALSE.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT01</TD>
+# <TD width=26%>Connection timeout expired</TD>
+# <TD width=52%>The connection timeout period expired before the data source responded to the request. The connection timeout period is set through <B>SQLSetConnectAttr</B>, SQL_ATTR_CONNECTION_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>IM001</TD>
+# <TD width=26%>Driver does not support this function</TD>
+# <TD width=52%>(DM) The driver associated with the <I>StatementHandle</I> does not support the function.</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Comments</B></P>
+#
+# <P>Statement attributes for a statement remain in effect until they are changed by another call to <B>SQLSetStmtAttr</B> or until the statement is dropped by calling <B>SQLFreeHandle</B>. Calling <B>SQLFreeStmt</B> with the SQL_CLOSE, SQL_UNBIND, or SQL_RESET_PARAMS option does not reset statement attributes.</P>
+#
+# <P>Some statement attributes support substitution of a similar value if the data source does not support the value specified in <I>ValuePtr</I>. In such cases, the driver returns SQL_SUCCESS_WITH_INFO and SQLSTATE 01S02 (Option value changed). For example, if <I>Attribute</I> is SQL_ATTR_CONCURRENCY and <I>ValuePtr</I> is SQL_CONCUR_ROWVER, and if the data source does not support this, the driver substitutes SQL_CONCUR_VALUES and returns SQL_SUCCESS_WITH_INFO. To determine the substituted value, an application calls <B>SQLGetStmtAttr</B>.</P>
+#
+# <P>The format of information set with <I>ValuePtr</I> depends on the specified <I>Attribute</I>. <B>SQLSetStmtAttr</B> accepts attribute information in one of two different formats: a character string or a 32-bit integer value. The format of each is noted in the attribute's description. This format applies to the information returned for each attribute in <B>SQLGetStmtAttr</B>. Character strings pointed to by the <I>ValuePtr</I> argument of <B>SQLSetStmtAttr</B> have a length of <I>StringLength</I>.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;The ability to set statement attributes at the connection level by calling <B>SQLSetConnectAttr</B> has been deprecated in ODBC 3<I>.x</I>. ODBC 3<I>.x</I> applications should never set statement attributes at the connection level. ODBC 3<I>.x</I> statement attributes cannot be set at the connection level, with the exception of the SQL_ATTR_METADATA_ID and SQL_ATTR_ASYNC_ENABLE attributes, which are both connection attributes and statement attributes, and can be set at either the connection level or the statement level.</P>
+#
+# <P class="indent">ODBC 3<I>.x</I> drivers need only support this functionality if they should work with ODBC 2<I>.x</I> applications that set ODBC 2<I>.x</I> statement options at the connection level. For more information, see "Setting Statement Options on the Connection Level" under "<A HREF="odbcsqlsetconnectoption_mapping.htm">SQLSetConnectOption Mapping</A>" in Appendix G: Driver Guidelines for Backward Compatibility.</P>
+#
+# <H1>Statement Attributes That Set Descriptor Fields</H1>
+#
+# <P>Many statement attributes correspond to a header field of a descriptor. Setting these attributes actually results in the setting of the descriptor fields. Setting fields by a call to <B>SQLSetStmtAttr</B> rather than to <B>SQLSetDescField</B> has the advantage that a descriptor handle does not have to be obtained for the function call.</P>
+#
+# <P class="indent"><b class="le">Caution</b>&nbsp;&nbsp;&nbsp;Calling <B>SQLSetStmtAttr</B> for one statement can affect other statements. This occurs when the APD or ARD associated with the statement is explicitly allocated and is also associated with other statements. Because <B>SQLSetStmtAttr</B> modifies the APD or ARD, the modifications apply to all statements with which this descriptor is associated. If this is not the required behavior, the application should dissociate this descriptor from the other statements (by calling <B>SQLSetStmtAttr</B> to set the SQL_ATTR_APP_ROW_DESC or SQL_ATTR_APP_PARAM_DESC field to a different descriptor handle) before calling <B>SQLSetStmtAttr</B> again.</P>
+#
+# <P>When a descriptor field is set as a result of the corresponding statement attribute being set, the field is set only for the applicable descriptors that are currently associated with the statement identified by the <I>StatementHandle</I> argument, and the attribute setting does not affect any descriptors that may be associated with that statement in the future. When a descriptor field that is also a statement attribute is set by a call to <B>SQLSetDescField</B>, the corresponding statement attribute is set. If an explicitly allocated descriptor is dissociated from a statement, a statement attribute that corresponds to a header field will revert to the value of the field in the implicitly allocated descriptor.</P>
+#
+# <P>When a statement is allocated (see <A HREF="odbcsqlallochandle.htm">SQLAllocHandle</A>), four descriptor handles are automatically allocated and associated with the statement. Explicitly allocated descriptor handles can be associated with the statement by calling <B>SQLAllocHandle</B> with an <I>fHandleType</I> of SQL_HANDLE_DESC to allocate a descriptor handle and then calling <B>SQLSetStmtAttr</B> to associate the descriptor handle with the statement. </P>
+#
+# <P>The statement attributes in the following table correspond to descriptor header fields.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=42%>Statement attribute</TH>
+# <TH width=45%>Header field</TH>
+# <TH width=13%>Desc.</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_PARAM_BIND_OFFSET_PTR</TD>
+# <TD width=45%>SQL_DESC_BIND_OFFSET_PTR</TD>
+# <TD width=13%>APD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_PARAM_BIND_TYPE</TD>
+# <TD width=45%>SQL_DESC_BIND_TYPE</TD>
+# <TD width=13%>APD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_PARAM_OPERATION_PTR</TD>
+# <TD width=45%>SQL_DESC_ARRAY_STATUS_PTR</TD>
+# <TD width=13%>APD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_PARAM_STATUS_PTR</TD>
+# <TD width=45%>SQL_DESC_ARRAY_STATUS_PTR</TD>
+# <TD width=13%>IPD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_PARAMS_PROCESSED_PTR</TD>
+# <TD width=45%>SQL_DESC_ROWS_PROCESSED_PTR</TD>
+# <TD width=13%>IPD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_PARAMSET_SIZE</TD>
+# <TD width=45%>SQL_DESC_ARRAY_SIZE</TD>
+# <TD width=13%>APD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_ROW_ARRAY_SIZE</TD>
+# <TD width=45%>SQL_DESC_ARRAY_SIZE</TD>
+# <TD width=13%>ARD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_ROW_BIND_OFFSET_PTR</TD>
+# <TD width=45%>SQL_DESC_BIND_OFFSET_PTR</TD>
+# <TD width=13%>ARD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_ROW_BIND_TYPE</TD>
+# <TD width=45%>SQL_DESC_BIND_TYPE</TD>
+# <TD width=13%>ARD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_ROW_OPERATION_PTR</TD>
+# <TD width=45%>SQL_DESC_ARRAY_STATUS_PTR</TD>
+# <TD width=13%>ARD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_ROW_STATUS_PTR</TD>
+# <TD width=45%>SQL_DESC_ARRAY_STATUS_PTR</TD>
+# <TD width=13%>IRD</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=42%>SQL_ATTR_ROWS_FETCHED_PTR</TD>
+# <TD width=45%>SQL_DESC_ROWS_PROCESSED_PTR</TD>
+# <TD width=13%>IRD</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <H1>Statement Attributes</H1>
+#
+# <P>The currently defined attributes and the version of ODBC in which they were introduced are shown in the following table; it is expected that more attributes will be defined by drivers to take advantage of different data sources. A range of attributes is reserved by ODBC; driver developers must reserve values for their own driver-specific use from X/Open. For more information, see "<A HREF="odbcdriver_specific_data_types__descriptor_types__information_types.htm">Driver-Specific Data Types, Descriptor Types, Information Types, Diagnostic Types, and Attributes</A>" in Chapter 17: Programming Considerations.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=38%>Attribute</TH>
+# <TH width=62%><I>ValuePtr</I> contents</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_APP_PARAM_DESC<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>The handle to the APD for subsequent calls to <B>SQLExecute</B> and <B>SQLExecDirect</B> on the statement handle. The initial value of this attribute is the descriptor implicitly allocated when the statement was initially allocated. If the value of this attribute is set to SQL_NULL_DESC or the handle originally allocated for the descriptor, an explicitly allocated APD handle that was previously associated with the statement handle is dissociated from it and the statement handle reverts to the implicitly allocated APD handle.
+# <P>This attribute cannot be set to a descriptor handle that was implicitly allocated for another statement or to another descriptor handle that was implicitly set on the same statement; implicitly allocated descriptor handles cannot be associated with more than one statement or descriptor handle.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_APP_PARAM_DESC => {
+ type => q(SQLPOINTER),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_APP_ROW_DESC<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>The handle to the ARD for subsequent fetches on the statement handle. The initial value of this attribute is the descriptor implicitly allocated when the statement was initially allocated. If the value of this attribute is set to SQL_NULL_DESC or the handle originally allocated for the descriptor, an explicitly allocated ARD handle that was previously associated with the statement handle is dissociated from it and the statement handle reverts to the implicitly allocated ARD handle.
+# <P>This attribute cannot be set to a descriptor handle that was implicitly allocated for another statement or to another descriptor handle that was implicitly set on the same statement; implicitly allocated descriptor handles cannot be associated with more than one statement or descriptor handle.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_APP_ROW_DESC => {
+ type => q(SQLPOINTER),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ASYNC_ENABLE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies whether a function called with the specified statement is executed asynchronously:
+# <P>SQL_ASYNC_ENABLE_OFF = Off (the default)<BR>
+# SQL_ASYNC_ENABLE_ON = On</P>
+#
+# <P>Once a function has been called asynchronously, only the original function, <B>SQLCancel</B>, <B>SQLGetDiagField</B>, or <B>SQLGetDiagRec</B> can be called on the statement, and only the original function, <B>SQLAllocHandle </B>(with a <I>HandleType</I> of SQL_HANDLE_STMT), <B>SQLGetDiagField</B>, <B>SQLGetDiagRec</B>, or <B>SQLGetFunctions</B> can be called on the connection associated with the statement, until the original function returns a code other than SQL_STILL_EXECUTING. Any other function called on the statement or the connection associated with the statement returns SQL_ERROR with an SQLSTATE of HY010 (Function sequence error). Functions can be called on other statements. For more information, see "<A HREF="odbcasynchronous_execution.htm">Asynchronous Execution</A>" in Chapter 9: Executing Statements.</P>
+#
+# <P>For drivers with statement level asynchronous execution support, the statement attribute SQL_ATTR_ASYNC_ENABLE may be set. Its initial value is the same as the value of the connection level attribute with the same name at the time the statement handle was allocated. </P>
+#
+# <P>For drivers with connection-level, asynchronous-execution support, the statement attribute SQL_ATTR_ASYNC_ENABLE is read-only. Its value is the same as the value of the connection level attribute with the same name at the time the statement handle was allocated. Calling <B>SQLSetStmtAttr</B> to set SQL_ATTR_ASYNC_ENABLE when the SQL_ASYNC_MODE <I>InfoType</I> returns SQL_AM_CONNECTION returns SQLSTATE HYC00 (Optional feature not implemented). (See <B>SQLSetConnectAttr</B> for more information.)</P>
+#
+# <P>As a standard practice, applications should execute functions asynchronously only on single-thread operating systems. On multithread operating systems, applications should execute functions on separate threads rather than executing them asynchronously on the same thread. No functionality is lost if drivers that operate only on multithread operating systems do not need to support asynchronous execution. </P>
+#
+# <P>The following functions can be executed asynchronously:</P>
+#
+# <P><B>SQLBulkOperations<BR>
+# SQLColAttribute</B><BR>
+# <B>SQLColumnPrivileges</B><BR>
+# <B>SQLColumns</B><BR>
+# <B>SQLCopyDesc</B><BR>
+# <B>SQLDescribeCol</B><BR>
+# <B>SQLDescribeParam</B><BR>
+# <B>SQLExecDirect</B><BR>
+# <B>SQLExecute</B><BR>
+# <B>SQLFetch</B><BR>
+# <B>SQLFetchScroll</B><BR>
+# <B>SQLForeignKeys</B><BR>
+# <B>SQLGetData</B><BR>
+# <B>SQLGetDescField</B><SUP>[1]</SUP><BR>
+# <B>SQLGetDescRec</B><SUP>[1]</SUP><B><BR>
+# SQLGetDiagField</B><BR>
+# <B>SQLGetDiagRec<BR>
+# SQLGetTypeInfo</B><BR>
+# <B>SQLMoreResults</B><BR>
+# <B>SQLNumParams</B><BR>
+# <B>SQLNumResultCols</B><BR>
+# <B>SQLParamData</B><BR>
+# <B>SQLPrepare</B><BR>
+# <B>SQLPrimaryKeys</B><BR>
+# <B>SQLProcedureColumns</B><BR>
+# <B>SQLProcedures</B><BR>
+# <B>SQLPutData</B><BR>
+# <B>SQLSetPos</B><BR>
+# <B>SQLSpecialColumns</B><BR>
+# <B>SQLStatistics</B><BR>
+# <B>SQLTablePrivileges</B><BR>
+# <B>SQLTables</B></P>
+# </TD>
+# </TR>
+ SQL_ATTR_ASYNC_ENABLE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_CONCURRENCY<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies the cursor concurrency:
+# <P>SQL_CONCUR_READ_ONLY = Cursor is read-only. No updates are allowed.</P>
+#
+# <P>SQL_CONCUR_LOCK = Cursor uses the lowest level of locking sufficient to ensure that the row can be updated.</P>
+#
+# <P>SQL_CONCUR_ROWVER = Cursor uses optimistic concurrency control, comparing row versions such as SQLBase ROWID or Sybase TIMESTAMP.</P>
+#
+# <P>SQL_CONCUR_VALUES = Cursor uses optimistic concurrency control, comparing values.</P>
+#
+# <P>The default value for SQL_ATTR_CONCURRENCY is SQL_CONCUR_READ_ONLY.</P>
+#
+# <P>This attribute cannot be specified for an open cursor. For more information, see "<A HREF="odbcconcurrency_types.htm">Concurrency Types</A>" in Chapter 14: Transactions.</P>
+#
+# <P>If the SQL_ATTR_CURSOR_TYPE <I>Attribute</I> is changed to a type that does not support the current value of SQL_ATTR_CONCURRENCY, the value of SQL_ATTR_CONCURRENCY will be changed at execution time, and a warning issued when <B>SQLExecDirect</B> or <B>SQLPrepare</B> is called.</P>
+#
+# <P>If the driver supports the <B>SELECT FOR UPDATE</B> statement and such a statement is executed while the value of SQL_ATTR_CONCURRENCY is set to SQL_CONCUR_READ_ONLY, an error will be returned. If the value of SQL_ATTR_CONCURRENCY is changed to a value that the driver supports for some value of SQL_ATTR_CURSOR_TYPE but not for the current value of SQL_ATTR_CURSOR_TYPE, the value of SQL_ATTR_CURSOR_TYPE will be changed at execution time and SQLSTATE 01S02 (Option value changed) is issued when <B>SQLExecDirect</B> or <B>SQLPrepare</B> is called.</P>
+#
+# <P>If the specified concurrency is not supported by the data source, the driver substitutes a different concurrency and returns SQLSTATE 01S02 (Option value changed). For SQL_CONCUR_VALUES, the driver substitutes SQL_CONCUR_ROWVER, and vice versa. For SQL_CONCUR_LOCK, the driver substitutes, in order, SQL_CONCUR_ROWVER or SQL_CONCUR_VALUES. The validity of the substituted value is not checked until execution time.</P>
+#
+# <P>For more information about the relationship between SQL_ATTR_CONCURRENCY and the other cursor attributes, see "<A HREF="odbccursor_characteristics_and_cursor_type.htm">Cursor Characteristics and Cursor Type</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+# </TD>
+# </TR>
+ SQL_ATTR_CONCURRENCY => {
+ type => q(SQLUINTEGER),
+ ptr => undef,
+ value => [ qw(SQL_CONCUR_READ_ONLY SQL_CONCUR_LOCK SQL_CONCUR_ROWVER SQL_CONCUR_ROWVER) ],
+ default => q(SQL_CONCUR_READ_ONLY),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_CURSOR_SCROLLABLE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies the level of support that the application requires. Setting this attribute affects subsequent calls to <B>SQLExecDirect</B> and <B>SQLExecute</B>.
+# <P>SQL_NONSCROLLABLE = Scrollable cursors are not required on the statement handle. If the application calls <B>SQLFetchScroll</B> on this handle, the only valid value of <I>FetchOrientation</I> is SQL_FETCH_NEXT. This is the default.</P>
+#
+# <P>SQL_SCROLLABLE = Scrollable cursors are required on the statement handle. When calling <B>SQLFetchScroll</B>, the application may specify any valid value of <I>FetchOrientation</I>, achieving cursor positioning in modes other than the sequential mode. </P>
+#
+# <P>For more information about scrollable cursors, see "<A HREF="odbcscrollable_cursors.htm">Scrollable Cursors</A>" in Chapter 11: Retrieving Results (Advanced). For more information about the relationship between SQL_ATTR_CURSOR_SCROLLABLE and the other cursor attributes, see "<A HREF="odbccursor_characteristics_and_cursor_type.htm">Cursor Characteristics and Cursor Type</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+# </TD>
+# </TR>
+ SQL_ATTR_CURSOR_SCROLLABLE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_NONSCROLLABLE SQL_SCROLLABLE) ],
+ default => q(SQL_NONSCROLLABLE),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_CURSOR_SENSITIVITY<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies whether cursors on the statement handle make visible the changes made to a result set by another cursor. Setting this attribute affects subsequent calls to <B>SQLExecDirect</B> and <B>SQLExecute</B>. An application can read back the value of this attribute to obtain its initial state or its state as most recently set by the application.
+# <P>SQL_UNSPECIFIED = It is unspecified what the cursor type is and whether cursors on the statement handle make visible the changes made to a result set by another cursor. Cursors on the statement handle may make visible none, some, or all such changes. This is the default.</P>
+#
+# <P>SQL_INSENSITIVE = All cursors on the statement handle show the result set without reflecting any changes made to it by any other cursor. Insensitive cursors are read-only. This corresponds to a static cursor, which has a concurrency that is read-only.</P>
+#
+# <P>SQL_SENSITIVE = All cursors on the statement handle make visible all changes made to a result set by another cursor. </P>
+#
+# <P>For more information about the relationship between SQL_ATTR_CURSOR_SENSITIVITY and the other cursor attributes, see "<A HREF="odbccursor_characteristics_and_cursor_type.htm">Cursor Characteristics and Cursor Type</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+# </TD>
+# </TR>
+ SQL_ATTR_CURSOR_SENSITIVITY => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_UNSPECIFIED SQL_INSENSITIVE SQL_SENSITIVE) ],
+ default => q(SQL_UNSPECIFIED),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_CURSOR_TYPE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies the cursor type:
+# <P>SQL_CURSOR_FORWARD_ONLY = The cursor only scrolls forward.</P>
+#
+# <P>SQL_CURSOR_STATIC = The data in the result set is static.</P>
+#
+# <P>SQL_CURSOR_KEYSET_DRIVEN = The driver saves and uses the keys for the number of rows specified in the SQL_ATTR_KEYSET_SIZE statement attribute.</P>
+#
+# <P>SQL_CURSOR_DYNAMIC = The driver saves and uses only the keys for the rows in the rowset.</P>
+#
+# <P>The default value is SQL_CURSOR_FORWARD_ONLY. This attribute cannot be specified after the SQL statement has been prepared.</P>
+#
+# <P>If the specified cursor type is not supported by the data source, the driver substitutes a different cursor type and returns SQLSTATE 01S02 (Option value changed). For a mixed or dynamic cursor, the driver substitutes, in order, a keyset-driven or static cursor. For a keyset-driven cursor, the driver substitutes a static cursor. </P>
+#
+# <P>For more information about scrollable cursor types, see "<A HREF="odbcscrollable_cursor_types.htm">Scrollable Cursor Types</A>" in Chapter 11: Retrieving Results (Advanced). For more information about the relationship between SQL_ATTR_CURSOR_TYPE and the other cursor attributes, see "<A HREF="odbccursor_characteristics_and_cursor_type.htm">Cursor Characteristics and Cursor Type</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+# </TD>
+# </TR>
+ SQL_ATTR_CURSOR_TYPE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_CURSOR_FORWARD_ONLY SQL_CURSOR_STATIC SQL_CURSOR_KEYSET_DRIVEN SQL_CURSOR_DYNAMIC) ],
+ default => q(SQL_CURSOR_FORWARD_ONLY),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ENABLE_AUTO_IPD<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies whether automatic population of the IPD is performed:
+# <P>SQL_TRUE = Turns on automatic population of the IPD after a call to <B>SQLPrepare</B>. SQL_FALSE = Turns off automatic population of the IPD after a call to <B>SQLPrepare</B>. (An application can still obtain IPD field information by calling <B>SQLDescribeParam</B>, if supported.) The default value of the statement attribute SQL_ATTR_ENABLE_AUTO_IPD is SQL_FALSE. For more information, see "<A HREF="odbcautomatic_population_of_the_ipd.htm">Automatic Population of the IPD</A>" in Chapter 13: Descriptors.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_ENABLE_AUTO_IPD => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_FALSE SQL_TRUE) ],
+ default => q(SQL_FALSE),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_FETCH_BOOKMARK_PTR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>A pointer that points to a binary bookmark value. When <B>SQLFetchScroll</B> is called with <I>fFetchOrientation</I> equal to SQL_FETCH_BOOKMARK, the driver picks up the bookmark value from this field. This field defaults to a null pointer. For more information, see "<A HREF="odbcscrolling_by_bookmark.htm">Scrolling by Bookmark</A>" in Chapter 11: Retrieving Results (Advanced).
+# <P>The value pointed to by this field is not used for delete by bookmark, update by bookmark, or fetch by bookmark operations in <B>SQLBulkOperations</B>, which use bookmarks cached in rowset buffers.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_FETCH_BOOKMARK_PTR => {
+ type => q(SQLPOINTER),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_IMP_PARAM_DESC<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>The handle to the IPD. The value of this attribute is the descriptor allocated when the statement was initially allocated. The application cannot set this attribute.
+# <P>This attribute can be retrieved by a call to <B>SQLGetStmtAttr</B> but not set by a call to <B>SQLSetStmtAttr</B>.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_IMP_PARAM_DESC => {
+ type => q(SQLPOINTER),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'ro',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_IMP_ROW_DESC<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>The handle to the IRD. The value of this attribute is the descriptor allocated when the statement was initially allocated. The application cannot set this attribute.
+# <P>This attribute can be retrieved by a call to <B>SQLGetStmtAttr</B> but not set by a call to <B>SQLSetStmtAttr</B>.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_IMP_ROW_DESC => {
+ type => q(SQLPOINTER),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'ro',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_KEYSET_SIZE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>An SQLUINTEGER that specifies the number of rows in the keyset for a keyset-driven cursor. If the keyset size is 0 (the default), the cursor is fully keyset-driven. If the keyset size is greater than 0, the cursor is mixed (keyset-driven within the keyset and dynamic outside of the keyset). The default keyset size is 0. For more information about keyset-driven cursors, see "<A HREF="odbckeyset_driven_cursors.htm">Keyset-Driven Cursors</A>" in Chapter 11: Retrieving Results (Advanced).
+# <P>If the specified size exceeds the maximum keyset size, the driver substitutes that size and returns SQLSTATE 01S02 (Option value changed).</P>
+#
+# <P><B>SQLFetch</B> or <B>SQLFetchScroll</B> returns an error if the keyset size is greater than 0 and less than the rowset size.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_KEYSET_SIZE => {
+ type => q(SQLUINTEGER),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_MAX_LENGTH<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies the maximum amount of data that the driver returns from a character or binary column. If <I>ValuePtr</I> is less than the length of the available data, <B>SQLFetch</B> or <B>SQLGetData</B> truncates the data and returns SQL_SUCCESS. If <I>ValuePtr</I> is 0 (the default), the driver attempts to return all available data.
+# <P>If the specified length is less than the minimum amount of data that the data source can return or greater than the maximum amount of data that the data source can return, the driver substitutes that value and returns SQLSTATE 01S02 (Option value changed).</P>
+#
+# <P>The value of this attribute can be set on an open cursor; however, the setting might not take effect immediately, in which case the driver will return SQLSTATE 01S02 (Option value changed) and reset the attribute to its original value.</P>
+#
+# <P>This attribute is intended to reduce network traffic and should be supported only when the data source (as opposed to the driver) in a multiple-tier driver can implement it. This mechanism should not be used by applications to truncate data; to truncate data received, an application should specify the maximum buffer length in the <I>BufferLength </I>argument in <B>SQLBindCol</B> or <B>SQLGetData</B>.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_MAX_LENGTH => {
+ type => q(SQLUINTEGER),
+ ptr => undef,
+ value => undef,
+ default => 0,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_MAX_ROWS<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value corresponding to the maximum number of rows to return to the application for a <B>SELECT</B> statement. If *<I>ValuePtr</I> equals 0 (the default), the driver returns all rows.
+# <P>This attribute is intended to reduce network traffic. Conceptually, it is applied when the result set is created and limits the result set to the first <I>ValuePtr</I> rows. If the number of rows in the result set is greater than <I>ValuePtr</I>, the result set is truncated. </P>
+#
+# <P>SQL_ATTR_MAX_ROWS applies to all result sets on the <I>Statement</I>, including those returned by catalog functions. SQL_ATTR_MAX_ROWS establishes a maximum for the value of the cursor row count.</P>
+#
+# <P>A driver should not emulate SQL_ATTR_MAX_ROWS behavior for <B>SQLFetch</B> or <B>SQLFetchScroll</B> (if result set size limitations cannot be implemented at the data source) if it cannot guarantee that SQL_ATTR_MAX_ROWS will be implemented properly.</P>
+#
+# <P>It is driver-defined whether SQL_ATTR_MAX_ROWS applies to statements other than SELECT statements (such as catalog functions). </P>
+#
+# <P>The value of this attribute can be set on an open cursor; however, the setting might not take effect immediately, in which case the driver will return SQLSTATE 01S02 (Option value changed) and reset the attribute to its original value.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_MAX_ROWS => {
+ type => q(SQLUINTEGER),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_METADATA_ID<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that determines how the string arguments of catalog functions are treated.
+# <P>If SQL_TRUE, the string argument of catalog functions are treated as identifiers. The case is not significant. For nondelimited strings, the driver removes any trailing spaces and the string is folded to uppercase. For delimited strings, the driver removes any leading or trailing spaces and takes whatever is between the delimiters literally. If one of these arguments is set to a null pointer, the function returns SQL_ERROR and SQLSTATE HY009 (Invalid use of null pointer). </P>
+#
+# <P>If SQL_FALSE, the string arguments of catalog functions are not treated as identifiers. The case is significant. They can either contain a string search pattern or not, depending on the argument.</P>
+#
+# <P>The default value is SQL_FALSE.</P>
+#
+# <P>The <I>TableType</I> argument of <B>SQLTables</B>, which takes a list of values, is not affected by this attribute.</P>
+#
+# <P>SQL_ATTR_METADATA_ID can also be set on the connection level. (It and SQL_ATTR_ASYNC_ENABLE are the only statement attributes that are also connection attributes.)</P>
+#
+# <P>For more information, see "<A HREF="odbcarguments_in_catalog_functions.htm">Arguments in Catalog Functions</A>" in Chapter 7: Catalog Functions.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_METADATA_ID => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_FALSE SQL_TRUE) ],
+ default => q(SQL_FALSE),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_NOSCAN<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that indicates whether the driver should scan SQL strings for escape sequences:
+# <P>SQL_NOSCAN_OFF = The driver scans SQL strings for escape sequences (the default).</P>
+#
+# <P>SQL_NOSCAN_ON = The driver does not scan SQL strings for escape sequences. Instead, the driver sends the statement directly to the data source.</P>
+#
+# <P>For more information, see "<A HREF="odbcescape_sequences_in_odbc.htm">Escape Sequences in ODBC</A>" in Chapter 8: SQL Statements.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_NOSCAN => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_NOSCAN_OFF SQL_NOSCAN_ON) ],
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_PARAM_BIND_OFFSET_PTR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER * value that points to an offset added to pointers to change binding of dynamic parameters. If this field is non-null, the driver dereferences the pointer, adds the dereferenced value to each of the deferred fields in the descriptor record (SQL_DESC_DATA_PTR, SQL_DESC_INDICATOR_PTR, and SQL_DESC_OCTET_LENGTH_PTR), and uses the new pointer values when binding. It is set to null by default.
+# <P>The bind offset is always added directly to the SQL_DESC_DATA_PTR, SQL_DESC_INDICATOR_PTR, and SQL_DESC_OCTET_LENGTH_PTR fields. If the offset is changed to a different value, the new value is still added directly to the value in the descriptor field. The new offset is not added to the field value plus any earlier offsets.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_PARAM_BIND_OFFSET_PTR => {
+ type => q(SQLUINTEGER),
+ ptr => 1,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_PARAM_BIND_OFFSET_PTR<BR>
+# (ODBC 3.0) (<I>continued</I>)</TD>
+# <TD width=62%>For more information, see "<A HREF="odbcparameter_binding_offsets.htm">Parameter Binding Offsets</A>" in Chapter 9: Executing Statements.
+# <P>Setting this statement attribute sets the SQL_DESC_BIND_OFFSET_PTR field in the APD header.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_PARAM_BIND_TYPE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that indicates the binding orientation to be used for dynamic parameters.
+# <P>This field is set to SQL_PARAM_BIND_BY_COLUMN (the default) to select column-wise binding. </P>
+#
+# <P>To select row-wise binding, this field is set to the length of the structure or an instance of a buffer that will be bound to a set of dynamic parameters. This length must include space for all of the bound parameters and any padding of the structure or buffer to ensure that when the address of a bound parameter is incremented with the specified length, the result will point to the beginning of the same parameter in the next set of parameters. When using the <I>sizeof</I> operator in ANSI C, this behavior is guaranteed.</P>
+#
+# <P>For more information, see "<A HREF="odbcbinding_arrays_of_parameters.htm">Binding Arrays of Parameters</A>" in Chapter 9: Executing Statements.</P>
+#
+# <P>Setting this statement attribute sets the SQL_DESC_ BIND_TYPE field in the APD header.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_PARAM_BIND_TYPE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_PARAM_OPERATION_PTR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUSMALLINT * value that points to an array of SQLUSMALLINT values used to ignore a parameter during execution of an SQL statement. Each value is set to either SQL_PARAM_PROCEED (for the parameter to be executed) or SQL_PARAM_IGNORE (for the parameter to be ignored).
+# <P>A set of parameters can be ignored during processing by setting the status value in the array pointed to by SQL_DESC_ARRAY_STATUS_PTR in the APD to SQL_PARAM_IGNORE. A set of parameters is processed if its status value is set to SQL_PARAM_PROCEED or if no elements in the array are set.</P>
+#
+# <P>This statement attribute can be set to a null pointer, in which case the driver does not return parameter status values. This attribute can be set at any time, but the new value is not used until the next time <B>SQLExecDirect</B> or <B>SQLExecute</B> is called.</P>
+#
+# <P>For more information, see "<A HREF="odbcusing_arrays_of_parameters.htm">Using Arrays of Parameters</A>" in Chapter 9: Executing Statements.</P>
+#
+# <P>Setting this statement attribute sets the SQL_DESC_ARRAY_STATUS_PTR field in the APD header.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_PARAM_OPERATION_PTR => {
+ type => q(SQLUSMALLINT),
+ ptr => 1,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_PARAM_STATUS_PTR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUSMALLINT * value that points to an array of SQLUSMALLINT values containing status information for each row of parameter values after a call to <B>SQLExecute</B> or <B>SQLExecDirect</B>. This field is required only if PARAMSET_SIZE is greater than 1.
+# <P>The status values can contain the following values:</P>
+#
+# <P>SQL_PARAM_SUCCESS: The SQL statement was successfully executed for this set of parameters.</P>
+#
+# <P>SQL_PARAM_SUCCESS_WITH_INFO: The SQL statement was successfully executed for this set of parameters; however, warning information is available in the diagnostics data structure.</P>
+#
+# <P>SQL_PARAM_ERROR: There was an error in processing this set of parameters. Additional error information is available in the diagnostics data structure.</P>
+#
+# <P>SQL_PARAM_UNUSED: This parameter set was unused, possibly due to the fact that some previous parameter set caused an error that aborted further processing, or because SQL_PARAM_IGNORE was set for that set of parameters in the array specified by the SQL_ATTR_PARAM_OPERATION_PTR.</P>
+#
+# <P>SQL_PARAM_DIAG_UNAVAILABLE: The driver treats arrays of parameters as a monolithic unit and so does not generate this level of error information. </P>
+#
+# <P>This statement attribute can be set to a null pointer, in which case the driver does not return parameter status values. This attribute can be set at any time, but the new value is not used until the next time <B>SQLExecute</B> or <B>SQLExecDirect</B> is called. Note that setting this attribute can affect the output parameter behavior implemented by the driver.</P>
+#
+# <P>For more information, see "<A HREF="odbcusing_arrays_of_parameters.htm">Using Arrays of Parameters</A>" in Chapter 9: Executing Statements.</P>
+#
+# <P>Setting this statement attribute sets the SQL_DESC_ARRAY_STATUS_PTR field in the IPD header.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_PARAM_STATUS_PTR => {
+ type => q(SQLUSMALLINT),
+ ptr => 1,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_PARAMS_PROCESSED_PTR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER * record field that points to a buffer in which to return the number of sets of parameters that have been processed, including error sets. No number will be returned if this is a null pointer.
+# <P>Setting this statement attribute sets the SQL_DESC_ROWS_PROCESSED_PTR field in the IPD header.</P>
+#
+# <P>If the call to <B>SQLExecDirect</B> or <B>SQLExecute</B> that fills in the buffer pointed to by this attribute does not return SQL_SUCCESS or SQL_SUCCESS_WITH_INFO, the contents of the buffer are undefined.</P>
+#
+# <P>For more information, see "<A HREF="odbcusing_arrays_of_parameters.htm">Using Arrays of Parameters</A>" in Chapter 9: Executing Statements.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_PARAMS_PROCESSED_PTR => {
+ type => q(SQLUINTEGER),
+ ptr => 1,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_PARAMSET_SIZE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies the number of values for each parameter. If SQL_ATTR_PARAMSET_SIZE is greater than 1, SQL_DESC_DATA_PTR, SQL_DESC_INDICATOR_PTR, and SQL_DESC_OCTET_LENGTH_PTR of the APD point to arrays. The cardinality of each array is equal to the value of this field.
+# <P>For more information, see "<A HREF="odbcusing_arrays_of_parameters.htm">Using Arrays of Parameters</A>" in Chapter 9: Executing Statements.</P>
+#
+# <P>Setting this statement attribute sets the SQL_DESC_ARRAY_SIZE field in the APD header.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_PARAMSET_SIZE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_QUERY_TIMEOUT<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value corresponding to the number of seconds to wait for an SQL statement to execute before returning to the application. If <I>ValuePtr</I> is equal to 0 (default), there is no timeout.
+# <P>If the specified timeout exceeds the maximum timeout in the data source or is smaller than the minimum timeout, <B>SQLSetStmtAttr</B> substitutes that value and returns SQLSTATE 01S02 (Option value changed).</P>
+#
+# <P>Note that the application need not call <B>SQLCloseCursor</B> to reuse the statement if a <B>SELECT</B> statement timed out.</P>
+#
+# <P>The query timeout set in this statement attribute is valid in both synchronous and asynchronous modes.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_QUERY_TIMEOUT => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_RETRIEVE_DATA<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>An SQLUINTEGER value:
+# <P>SQL_RD_ON = <B>SQLFetchScroll</B> and, in ODBC 3<I>.x</I>, <B>SQLFetch</B> retrieve data after it positions the cursor to the specified location. This is the default.</P>
+#
+# <P>SQL_RD_OFF = <B>SQLFetchScroll</B> and, in ODBC 3<I>.x</I>, <B>SQLFetch</B> do not retrieve data after it positions the cursor.</P>
+#
+# <P>By setting SQL_RETRIEVE_DATA to SQL_RD_OFF, an application can verify that a row exists or retrieve a bookmark for the row without incurring the overhead of retrieving rows. For more information, see "<A HREF="odbcscrolling_and_fetching_rows.htm">Scrolling and Fetching Rows</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+#
+# <P>The value of this attribute can be set on an open cursor; however, the setting might not take effect immediately, in which case the driver will return SQLSTATE 01S02 (Option value changed) and reset the attribute to its original value.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_RETRIEVE_DATA => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_RD_ON SQL_RD_OFF) ],
+ default => q(SQL_RD_ON),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ROW_ARRAY_SIZE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies the number of rows returned by each call to <B>SQLFetch</B> or <B>SQLFetchScroll</B>. It is also the number of rows in a bookmark array used in a bulk bookmark operation in <B>SQLBulkOperations</B>. The default value is 1.
+# <P>If the specified rowset size exceeds the maximum rowset size supported by the data source, the driver substitutes that value and returns SQLSTATE 01S02 (Option value changed).</P>
+#
+# <P>For more information, see "<A HREF="odbcrowset_size.htm">Rowset Size</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+#
+# <P>Setting this statement attribute sets the SQL_DESC_ARRAY_SIZE field in the ARD header.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_ROW_ARRAY_SIZE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ROW_BIND_OFFSET_PTR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER * value that points to an offset added to pointers to change binding of column data. If this field is non-null, the driver dereferences the pointer, adds the dereferenced value to each of the deferred fields in the descriptor record (SQL_DESC_DATA_PTR, SQL_DESC_INDICATOR_PTR, and SQL_DESC_OCTET_LENGTH_PTR), and uses the new pointer values when binding. It is set to null by default.
+# <P>Setting this statement attribute sets the SQL_DESC_BIND_OFFSET_PTR field in the ARD header.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_ROW_BIND_OFFSET_PTR => {
+ type => q(SQLUINTEGER),
+ ptr => undef,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ROW_BIND_TYPE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that sets the binding orientation to be used when <B>SQLFetch</B> or <B>SQLFetchScroll</B> is called on the associated statement. Column-wise binding is selected by setting the value to SQL_BIND_BY_COLUMN. Row-wise binding is selected by setting the value to the length of a structure or an instance of a buffer into which result columns will be bound.
+# <P>If a length is specified, it must include space for all of the bound columns and any padding of the structure or buffer to ensure that when the address of a bound column is incremented with the specified length, the result will point to the beginning of the same column in the next row. When using the <B>sizeof</B> operator with structures or unions in ANSI C, this behavior is guaranteed.</P>
+#
+# <P>Column-wise binding is the default binding orientation for <B>SQLFetch</B> and <B>SQLFetchScroll</B>.</P>
+#
+# <P>For more information, see "<A HREF="odbcbinding_columns_for_use_with_block_cursors.htm">Binding Columns for Use with Block Cursors</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+#
+# <P>Setting this statement attribute sets the SQL_DESC_BIND_TYPE field in the ARD header.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_ROW_BIND_TYPE => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_BIND_BY_COLUMN etc) ],
+ default => q(SQL_BIND_BY_COLUMN),
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ROW_NUMBER<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that is the number of the current row in the entire result set. If the number of the current row cannot be determined or there is no current row, the driver returns 0.
+# <P>This attribute can be retrieved by a call to <B>SQLGetStmtAttr</B> but not set by a call to <B>SQLSetStmtAttr</B>.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_ROW_NUMBER => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ROW_OPERATION_PTR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUSMALLINT * value that points to an array of SQLUSMALLINT values used to ignore a row during a bulk operation using <B>SQLSetPos</B>. Each value is set to either SQL_ROW_PROCEED (for the row to be included in the bulk operation) or SQL_ROW_IGNORE (for the row to be excluded from the bulk operation). (Rows cannot be ignored by using this array during calls to <B>SQLBulkOperations</B>.)
+# <P>This statement attribute can be set to a null pointer, in which case the driver does not return row status values. This attribute can be set at any time, but the new value is not used until the next time <B>SQLSetPos</B> is called.</P>
+#
+# <P>For more information, see "<A HREF="odbcupdating_rows_in_the_rowset_with_sqlsetpos.htm">Updating Rows in the Rowset with SQLSetPos</A>" and "<A HREF="odbcdeleting_rows_in_the_rowset_with_sqlsetpos.htm">Deleting Rows in the Rowset with SQLSetPos</A>" in Chapter 12: Updating Data.</P>
+#
+# <P>Setting this statement attribute sets the SQL_DESC_ARRAY_STATUS_PTR field in the ARD.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_ROW_OPERATION_PTR => {
+ type => q(SQLUSMALLINT),
+ ptr => 1,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ROW_STATUS_PTR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUSMALLINT * value that points to an array of SQLUSMALLINT values containing row status values after a call to <B>SQLFetch</B> or <B>SQLFetchScroll</B>. The array has as many elements as there are rows in the rowset.
+# <P>This statement attribute can be set to a null pointer, in which case the driver does not return row status values. This attribute can be set at any time, but the new value is not used until the next time <B>SQLBulkOperations</B>, <B>SQLFetch</B>, <B>SQLFetchScroll</B>, or <B>SQLSetPos</B> is called.</P>
+#
+# <P>For more information, see "<A HREF="odbcnumber_of_rows_fetched_and_status.htm">Number of Rows Fetched and Status</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+#
+# <P>Setting this statement attribute sets the SQL_DESC_ARRAY_STATUS_PTR field in the IRD header.</P>
+#
+# <P>This attribute is mapped by an ODBC 2<I>.x</I> driver to the <I>rgbRowStatus</I> array in a call to <B>SQLExtendedFetch</B>.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_ROW_STATUS_PTR => {
+ type => q(SQLUSMALLINT),
+ ptr => 1,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_ROWS_FETCHED_PTR<BR>
+# (ODBC 3.0)</TD>
+# <TD width=62%>An SQLUINTEGER * value that points to a buffer in which to return the number of rows fetched after a call to <B>SQLFetch</B> or <B>SQLFetchScroll</B>; the number of rows affected by a bulk operation performed by a call to <B>SQLSetPos</B> with an <I>Operation</I> argument of SQL_REFRESH; or the number of rows affected by a bulk operation performed by <B>SQLBulkOperations</B>. This number includes error rows.
+# <P>For more information, see "<A HREF="odbcnumber_of_rows_fetched_and_status.htm">Number of Rows Fetched and Status</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+#
+# <P>Setting this statement attribute sets the SQL_DESC_ROWS_PROCESSED_PTR field in the IRD header. </P>
+#
+# <P>If the call to <B>SQLFetch</B> or <B>SQLFetchScroll</B> that fills in the buffer pointed to by this attribute does not return SQL_SUCCESS or SQL_SUCCESS_WITH_INFO, the contents of the buffer are undefined.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_ROWS_FETCHED_PTR => {
+ type => q(SQLUINTEGER),
+ ptr => 1,
+ value => undef,
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_SIMULATE_CURSOR<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies whether drivers that simulate positioned update and delete statements guarantee that such statements affect only one single row.
+# <P>To simulate positioned update and delete statements, most drivers construct a searched <B>UPDATE</B> or <B>DELETE</B> statement containing a <B>WHERE</B> clause that specifies the value of each column in the current row. Unless these columns make up a unique key, such a statement can affect more than one row.</P>
+#
+# <P>To guarantee that such statements affect only one row, the driver determines the columns in a unique key and adds these columns to the result set. If an application guarantees that the columns in the result set make up a unique key, the driver is not required to do so. This may reduce execution time.</P>
+#
+# <P>SQL_SC_NON_UNIQUE = The driver does not guarantee that simulated positioned update or delete statements will affect only one row; it is the application's responsibility to do so. If a statement affects more than one row, <B>SQLExecute</B>, <B>SQLExecDirect</B>, or <B>SQLSetPos</B> returns SQLSTATE 01001 (Cursor operation conflict).</P>
+#
+# <P>SQL_SC_TRY_UNIQUE = The driver attempts to guarantee that simulated positioned update or delete statements affect only one row. The driver always executes such statements, even if they might affect more than one row, such as when there is no unique key. If a statement affects more than one row, <B>SQLExecute</B>, <B>SQLExecDirect</B>, or <B>SQLSetPos</B> returns SQLSTATE 01001 (Cursor operation conflict).</P>
+#
+# <P>SQL_SC_UNIQUE = The driver guarantees that simulated positioned update or delete statements affect only one row. If the driver cannot guarantee this for a given statement, <B>SQLExecDirect</B> or <B>SQLPrepare</B> returns an error.</P>
+#
+# <P>If the data source provides native SQL support for positioned update and delete statements and the driver does not simulate cursors, SQL_SUCCESS is returned when SQL_SC_UNIQUE is requested for SQL_SIMULATE_CURSOR. SQL_SUCCESS_WITH_INFO is returned if SQL_SC_TRY_UNIQUE or SQL_SC_NON_UNIQUE is requested. If the data source provides the SQL_SC_TRY_UNIQUE level of support and the driver does not, SQL_SUCCESS is returned for SQL_SC_TRY_UNIQUE and SQL_SUCCESS_WITH_INFO is returned for SQL_SC_NON_UNIQUE.</P>
+#
+# <P>If the specified cursor simulation type is not supported by the data source, the driver substitutes a different simulation type and returns SQLSTATE 01S02 (Option value changed). For SQL_SC_UNIQUE, the driver substitutes, in order, SQL_SC_TRY_UNIQUE or SQL_SC_NON_UNIQUE. For SQL_SC_TRY_UNIQUE, the driver substitutes SQL_SC_NON_UNIQUE.</P>
+#
+# <P>For more information, see "<A HREF="odbcsimulating_positioned_update_and_delete_statements.htm">Simulating Positioned Update and Delete Statements</A>" in Chapter 12: Updating Data.</P>
+# </TD>
+# </TR>
+ SQL_ATTR_SIMULATE_CURSOR => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_SC_NON_UNIQUE SQL_SC_TRY_UNIQUE SQL_SC_UNIQUE) ],
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=38%>SQL_ATTR_USE_BOOKMARKS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=62%>An SQLUINTEGER value that specifies whether an application will use bookmarks with a cursor:
+# <P>SQL_UB_OFF = Off (the default)</P>
+#
+# <P>SQL_UB_VARIABLE = An application will use bookmarks with a cursor, and the driver will provide variable-length bookmarks if they are supported. SQL_UB_FIXED is deprecated in ODBC 3<I>.x</I>. ODBC 3<I>.x</I> applications should always use variable-length bookmarks, even when working with ODBC 2<I>.x</I> drivers (which supported only 4-byte, fixed-length bookmarks). This is because a fixed-length bookmark is just a special case of a variable-length bookmark. When working with an ODBC 2<I>.x</I> driver, the Driver Manager maps SQL_UB_VARIABLE to SQL_UB_FIXED.</P>
+#
+# <P>To use bookmarks with a cursor, the application must specify this attribute with the SQL_UB_VARIABLE value before opening the cursor.</P>
+#
+# <P>For more information, see "<A HREF="odbcretrieving_bookmarks.htm">Retrieving Bookmarks</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+# </TD>
+# </TR>
+ SQL_ATTR_USE_BOOKMARKS => {
+ type => q(SQLUINTEGER),
+ ptr => 0,
+ value => [ qw(SQL_UB_OFF SQL_UB_VARIABLE SQL_UB_FIXED) ],
+ default => undef,
+ mode => 'rw',
+ order => ++$order,
+ },
+# </table></div>
+#
+# <P class="fineprint">[1]&nbsp;&nbsp;&nbsp;These functions can be called asynchronously only if the descriptor is an implementation descriptor, not an application descriptor.</p>
+# <P>See "<A HREF="odbccolumn_wise_binding.htm">Column-Wise Binding</A>" and "<A HREF="odbcrow_wise_binding.htm">Row-Wise Binding</A>" in Chapter 11: Retrieving Results (Advanced).</P>
+#
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=47%>For information about</TH>
+# <TH width=53%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=47%>Canceling statement processing</TD>
+# <TD width=53%><A HREF="odbcsqlcancel.htm">SQLCancel</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=47%>Returning the setting of a connection attribute</TD>
+# <TD width=53%><A HREF="odbcsqlgetconnectattr.htm">SQLGetConnectAttr</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=47%>Returning the setting of a statement attribute</TD>
+# <TD width=53%><A HREF="odbcsqlgetstmtattr.htm">SQLGetStmtAttr</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=47%>Setting a connection attribute</TD>
+# <TD width=53%><A HREF="odbcsqlsetconnectattr.htm">SQLSetConnectAttr</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=47%>Setting a single field of the descriptor</TD>
+# <TD width=53%><A HREF="odbcsqlsetdescfield.htm">SQLSetDescField</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+};
+
+my $i3 = " " x 3;
+my $i4 = " " x 4;
+my $i8 = " " x 8;
+
+my $type2type = {
+ SQLSMALLINT => 'Smallint',
+ SQLUSMALLINT => 'Usmallint',
+ SQLINTEGER => 'Integer',
+ SQLUINTEGER => 'Uinteger',
+ SQLPOINTER => 'Pointer',
+ SQLCHAR => 'Sqlchar',
+};
+
+my $attr =
+ $type eq 'Env' ? $attrEnv :
+ $type eq 'Dbc' ? $attrDbc :
+ $type eq 'Stmt' ? $attrStmt : die "bad type $type";
+
+my @name = sort {
+ $attr->{$a}{order} <=> $attr->{$b}{order}
+} keys %$attr;
+
+print "#include \"Handle$type.hpp\"\n";
+my $class = "OdbcData";
+
+for my $name (@name) {
+ my $p = $attr->{$name};
+ my $odbctype = $type2type->{$p->{type}} or die $name;
+ $odbctype .= "Ptr" if $p->{ptr};
+ print "\nstatic void\n";
+ print "callback_${name}_set(Ctx& ctx, HandleBase* self, const $class& data)\n";
+ print "{\n";
+ print "${i4}Handle$type* p$type = dynamic_cast<Handle$type*>(self);\n";
+ print "${i4}assert(p$type != 0 && data.type() == ${class}::$odbctype);\n";
+ print "}\n";
+ print "\nstatic void\n";
+ print "callback_${name}_default(Ctx& ctx, HandleBase* self, $class& data)\n";
+ print "{\n";
+ print "${i4}Handle$type* p$type = dynamic_cast<Handle$type*>(self);\n";
+ print "${i4}assert(p$type != 0);\n";
+ print "${i4}data.set();\n";
+ print "}\n";
+}
+
+print "\nAttrSpec Handle${type}::m_attrSpec\[\] = {\n";
+for my $name (@name) {
+ my $p = $attr->{$name};
+ my $odbctype = $type2type->{$p->{type}} or die $name;
+ $odbctype .= "Ptr" if $p->{ptr};
+ print "${i4}\{${i3}$name,\n";
+ print "${i8}${class}::$odbctype,\n";
+ my $attrmode =
+ $p->{mode} eq 'rw' ? 'Attr_mode_readwrite' :
+ $p->{mode} eq 'ro' ? 'Attr_mode_readonly' : die "bad mode $p->{mode}";
+ print "${i8}$attrmode,\n";
+ print "${i8}callback_${name}_set,\n";
+ print "${i8}callback_${name}_default,\n";
+ print "${i4}\},\n";
+}
+print "${i4}\{${i3}0,\n";
+print "${i8}${class}::Undef,\n";
+print "${i8}Attr_mode_undef,\n";
+print "${i8}0,\n";
+print "${i8}0,\n";
+print "${i4}\},\n";
+
+print "};\n";
diff --git a/ndb/src/client/odbc/docs/main.hpp b/ndb/src/client/odbc/docs/main.hpp
new file mode 100644
index 00000000000..ebb5b1f235a
--- /dev/null
+++ b/ndb/src/client/odbc/docs/main.hpp
@@ -0,0 +1,104 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ @mainpage NDB ODBC
+
+ The ODBC Driver Frontend has:
+ -# HandleBase : Various ODBC handles
+ -# AttrArea : Attributes of handles
+ -# ConnArea : Communication area on connection level between driver parts
+ -# StmtArea : Communication area on statement level between driver parts
+
+ and controls the following steps:
+ -# SQL_compiler : Compiles SQL into SQL_code_tree:s
+ -# Parser : Bison grammar
+ -# Analyzer : Syntactic and semantic checks (binds names)
+ -# PlanGen : Generate initial (execution) plan (PlanTree)
+ -# CodeGen : Generates CodeTree:s out of PlanTree:s
+ -# Optimizer : Optimizes PlanTree:s
+ -# Output : Outputs executable CodeTree:s
+ -# Executor : Executes CodeTree:s
+ -# CodeTree::allocRun : Allocates runtime data structures (RunTree:s)
+ -# Dataflow machine : Executes and evaluates statement and expressions
+
+ The Dataflow machine works in four different ways:
+ -# Non-query statements
+ -# CodeStmt::execute : Executes (non-query) statement
+ -# Query statements
+ -# CodeQuery::execute : Execute Query statement
+ -# CodeQuery::fetch : Fetch row from CodeQuery node
+ -# Boolean expressions
+ -# CodePred::evaluate : Evaluates boolean expression
+ -# Arithmetical expressions
+ -# CodeExpr::evaluate : Evaluates arithmetic expression
+
+ The following components are used throughout the NDB ODBC:
+ -# Context (Ctx) : Info regarding execution/evaluation
+ -# Diagnostic area (DiagArea) : Errors and warnings (for ODBC user)
+ -# DescArea : Description of ODBC user input/output bind varibles/columns
+ -# Dictionary (DictBase) : Lookup info stored in NDB Dictionary
+ and info regarding temporary
+ materialized results
+ -# ResultArea : Execution (temporary) results
+
+
+ @section secCompiler SQL_compiler : SQL to SQL_code_tree
+
+ The SQL_compiler takes an <em>SQL statement</em> and translates
+ it into an SQL_code_tree. The compiler uses an SQL_code_tree
+ internally during the compilation and the result of the compilation
+ is a simlified SQL_code_tree.
+
+ The compiler works in the following steps:
+ -# Parse SQL statments and create SQL_code_tree representing the
+ statement.
+ -# Apply Syntax Rules to the SQL_code_tree. Syntax rules are
+ rules which are <em>not</em> expressed in the SQL grammar,
+ but are expressed in natural language in the SQL specification.
+ -# Apply Access Rules to the SQL_code_tree
+ (this is not implemented, since NDB Cluster has no access control)
+ -# Apply General Rules to the SQL_code_tree
+ -# Apply Conformance Rules to the SQL_code_tree
+
+ The resulting simplified SQL_code_tree is represented by a
+ tree of C++ objects.
+
+
+ @section secCodegen Codegen : SQL_code_tree to CodeTree
+
+ CodeGen takes simplified SQL_code_tree:s and transforms them into
+ CodeTree:s.
+
+
+ @section secOptimizer Optimizer : CodeTree to CodeTree
+
+ The implementation of the ODBC optimizer will uses the
+ PlanTree:s to represent statements and transforms them
+ into executable format (still PlanTree format).
+
+ @note In the future, more optimizations can be implemented.
+
+
+ @section secExecutor Executor : Execute CodeTree
+
+ The Executor uses the following data structures:
+ -# CodeTree : A read-only quary evaluation plan
+ -# RunTree : Runtime data structures containing ResultSet:s
+
+ The execution mechanism is actually implemented as a
+ part of the CodeTree.
+*/
diff --git a/ndb/src/client/odbc/docs/ndbodbc.html b/ndb/src/client/odbc/docs/ndbodbc.html
new file mode 100644
index 00000000000..6be624dfa1b
--- /dev/null
+++ b/ndb/src/client/odbc/docs/ndbodbc.html
@@ -0,0 +1,659 @@
+<HTML>
+<HEAD>
+<META HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=windows-1252">
+<TITLE>ODBC and SQL</TITLE>
+</HEAD>
+<BODY LINK="#0000ff" VLINK="#800080" BGCOLOR="#ffffff">
+
+<h2>ODBC and SQL - NDB Cluster v2.11</h2>
+
+<p>
+NDB Cluster v2.11 includes a version of ODBC and SQL.
+<p>
+This document has 4 sections.
+<ol>
+<li>PLATFORMS
+<li>ODBC
+<li>SQL
+<li>DIAGNOSTICS
+</ol>
+<p>
+Features which are currently incomplete or planned for next release
+are marked with <b>v2.x</b>.
+
+<h3>1. PLATFORMS</h3>
+
+<h4>1.1 Linux / Unix</h4>
+<p>
+We use RedHat package names to describe supporting software.
+Packages starting with <b>perl-</b> are perl modules.
+If your installation does not include them you can get them
+from a CPAN archive ( <tt><b>ftp://ftp.funet.fi/pub/languages/perl/CPAN</b></tt> ).
+<p>
+Version numbers are given only as examples.
+Other versions will work.
+<p>
+An ODBC driver manager is required, one of:
+<ul>
+<li>unixODBC-2.2.3 (this is more common)
+<li>libiodbc-3.0.5
+</ul>
+<p>
+Additional packages are convenient.
+Following include perl scripting interface
+and an "interactive SQL" tool <b>dbish</b>:
+<ul>
+<li>perl-DBI-1.30
+<li>perl-DBD-ODBC-0.43
+<li>readline-4.2
+<li>perl-Term-ReadLine-Gnu-1.11
+</ul>
+<p>
+The NDB ODBC driver is located under NDB Cluster installation
+directory and is named <tt><b>libNDB_ODBC.so</b></tt>.
+It includes NDB API.
+To use it create a text file
+<tt><b>/etc/odbc.ini</b></tt> or <tt><b>$HOME/.odbc.ini</b></tt>
+containing at least:
+<p>
+<tt>
+<b>
+[ndb]
+<br>
+Driver = &lt;path-to-your-NDB-installation&gt;/lib/libNDB_ODBC.so
+</b>
+</tt>
+<p>
+Then try the shell command <tt><b>dbish dbi:ODBC:ndb</b></tt>
+in an NDB API node directory.
+
+<h4>1.2 Windows</h4>
+
+[ TODO - documentation ]
+
+<h3>2. ODBC</h3>
+
+<h4>2.1 External data types</h4>
+
+Usual external data types are supported and converted to and from SQL
+data types.
+<p>
+<table width="80%" border=1>
+<tr align="left"><th width="40%">type</th> <th>description</th></tr>
+<tr align="left"><td>SQL_C_CHAR</td><td>character buffers</tr>
+<tr align="left"><td>SQL_C_SLONG, etc</td><td>integer types</tr>
+<tr align="left"><td>SQL_C_DOUBLE, etc</td><td>floating types</tr>
+<tr align="left"><td>SQL_C_TYPE_TIMESTAMP</td><td>timestamp</tr>
+</table>
+
+<h4>2.2 ODBC functions</h4>
+<p>
+The driver implements basic ODBC functions.
+Main exceptions are:
+<ul>
+<li>no named cursors
+<li>no scrollable cursors
+<li>no bulk operations
+<li>no asynchronous execution
+</ul>
+<p>
+Following lists main ODBC 3.0 functions and
+their status in the driver.
+<p>
+<table width="80%" border=1>
+<tr align="left"><th width="40%">function</th><th>supported</th></tr>
+<tr align="left"><td>SQLAllocHandle</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLConnect</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetInfo</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetFunctions</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetTypeInfo</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLSetConnectAttr</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetConnectAttr</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLSetEnvAttr</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetEnvAttr</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLSetStmtAttr</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetStmtAttr</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetDescField</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetDescRec</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLSetDescField</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLSetDescRec</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLPrepare</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLBindParameter</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetCursorName</td><td>
+yes, but cursor names cannot be used in SQL
+</td></tr>
+<tr align="left"><td>SQLSetCursorName</td><td>
+yes, but cursor names cannot be used in SQL
+</td></tr>
+<tr align="left"><td>SQLSetScrollOptions</td><td>
+not implemented
+</td></tr>
+<tr align="left"><td>SQLExecute</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLExecDirect</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLNativeSql</td><td>
+not implemented
+</td></tr>
+<tr align="left"><td>SQLDescribeParam</td><td>
+not supported
+</td></tr>
+<tr align="left"><td>SQLNumParams</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLParamData</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLPutData</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLRowCount</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLNumResultCols</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLDescribeCol</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLColAttribute</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLBindCol</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLFetch</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLFetchScroll</td><td>
+not implemented
+</td></tr>
+<tr align="left"><td>SQLGetData</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLSetPos</td><td>
+not implemented
+</td></tr>
+<tr align="left"><td>SQLBulkOperations</td><td>
+not implemented
+</td></tr>
+<tr align="left"><td>SQLMoreResults</td><td>
+yes, but multiple result sets are not supported
+</td></tr>
+<tr align="left"><td>SQLGetDiagField</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLGetDiagRec</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLColumnPrivileges</td><td>
+not applicable
+</td></tr>
+<tr align="left"><td>SQLColumns</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLForeignKeys</td><td>
+not applicable
+</td></tr>
+<tr align="left"><td>SQLPrimaryKeys</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLProcedureColumns</td><td>
+not applicable
+</td></tr>
+<tr align="left"><td>SQLProcedures</td><td>
+not applicable
+</td></tr>
+<tr align="left"><td>SQLSpecialColumns</td><td>
+yes <b>v2.x</b>
+</td></tr>
+<tr align="left"><td>SQLStatistics</td><td>
+not applicable
+</td></tr>
+<tr align="left"><td>SQLTablePrivileges</td><td>
+not applicable
+</td></tr>
+<tr align="left"><td>SQLTables</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLFreeStmt</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLCloseCursor</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLCancel</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLEndTran</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLDisconnect</td><td>
+yes
+</td></tr>
+<tr align="left"><td>SQLFreeHandle</td><td>
+yes
+</td></tr>
+</table>
+
+<h3>3. SQL</h3>
+
+<h4>3.1 Data types</h4>
+
+<table width="80%" border=1>
+<tr align="left"><th width="40%">type</th> <th>description</th></tr>
+<tr align="left"><td>CHAR(n)</td><td>fixed-width blank-padded string</tr>
+<tr align="left"><td>VARCHAR(n)</td><td>variable length string</tr>
+<tr align="left"><td>INT<br>INTEGER</td><td>integer 32 bits</tr>
+<tr align="left"><td>BIGINT</td><td>integer 64 bits</tr>
+<tr align="left"><td>DECIMAL(m,n)</td><td>exact number with precision and scale <b>v2.x</b></tr>
+<tr align="left"><td>REAL</td><td>float 32 bits</tr>
+<tr align="left"><td>FLOAT<br>DOUBLE PRECISION</td><td>float, at least 64 bits</tr>
+<tr align="left"><td>DATE</td><td>date with precision 1 second <b>v2.x</b></tr>
+<tr align="left"><td>DATETIME</td><td>date with precision 1 nanosecond (SQL_TYPE_TIMESTAMP)</tr>
+</table>
+<p>
+
+Integer types may be qualified as UNSIGNED.
+<br><br>
+Strings and numbers are not converted to each other automatically.
+Following is an error (unlike in oracle).
+<br>
+<pre><tt>select 123 + '456' from tab</tt></pre>
+
+<h4>3.2 Expressions</h4>
+
+<table width="80%" border=1>
+<tr align="left"><th width="40%">syntax</th> <th>description</th></tr>
+<tr align="left"><td align="center"><b>NULL</b></td><td>null value</td></tr>
+<tr align="left"><td align="center">12.34<b>e</b>5</td><td>integer or decimal or float constant</td></tr>
+<tr align="left"><td align="center"><b>'</b>abc<b>'</b></td><td>string constant</td></tr>
+<tr align="left"><td align="center"><b>+ - * / ( )</b></td><td>arithmetic operations</td></tr>
+<tr align="left"><td align="center"><b>||</b></td><td>string concatenation <b>v2.x</b></td></tr>
+</table>
+
+<br>
+Integer and decimal arithmetic is done in BIGINT.
+<br>
+Floating arithmetic is done in DOUBLE PRECISION.
+<br>
+Numeric literals use largest applicable type.
+<br>
+String operations are done in CHAR or in VARCHAR (if any operand is VARCHAR).
+<br>
+String literals have type CHAR.
+
+<h4>3.3 Functions : non-aggregate</h4>
+
+<table width="80%" border=1>
+<tr align="left"><th width="40%">syntax</th> <th>description</th></tr>
+<tr align="left"><td align="center">SUBSTR() LEFT() RIGHT()</td><td>substring</td></tr>
+<tr align="left"><td align="center">TO_NUMBER() TO_CHAR()</td><td>basic conversions <b>v2.x</b></td></tr>
+<tr align="left"><td align="center">ROWNUM</td><td>row number in query</td></tr>
+<tr align="left"><td align="center">SYSDATE</td><td>current date as DATETIME</td></tr>
+</table>
+
+<h4>3.4 Functions : aggregate</h4>
+
+<table width="80%" border=1>
+<tr align="left"><th width="40%">syntax</th> <th>description</th></tr>
+<tr align="left"><td align="center">COUNT(*) COUNT(expr)</td><td>count rows or non-NULL values</td></tr>
+<tr align="left"><td align="center">MIN(expr) MAX(expr)</td><td>min and max of strings and numbers</td></tr>
+<tr align="left"><td align="center">SUM(expr) AVG(expr)</td><td>sum and average of numbers</td></tr>
+</table>
+<br>
+GROUP BY and HAVING are supported.
+
+<h4>3.5 Predicates</h4>
+
+<table width="80%" border=1>
+<tr align="left"><th width="40%">syntax</th> <th>description</th></tr>
+<tr align="left"><td align="center">IS NULL / IS NOT NULL</td><td>test if value is null</td></tr>
+<tr align="left"><td align="center"><b>&lt; &lt;= = != &gt; &gt=</b></td><td>comparisons</td></tr>
+<tr align="left"><td align="center">LIKE / NOT LIKE</td><td>string matching</td></tr>
+<tr align="left"><td align="center">AND OR NOT <b>( )</b></td><td>boolean operators</td></tr>
+</table>
+
+<h4>3.6 Tables</h4>
+
+An NDB table requires a primary key.
+There are 2 ways to specify it.
+
+<p>
+<h4>Case 1</h4>
+<pre><tt>create table t (
+ a integer not null,
+ b char(20) not null,
+ c float,
+ primary key(a, b)
+)
+</tt></pre>
+<p>
+<h4>Case 2</h4>
+<p>
+A column can be specified as AUTO_INCREMENT.
+The column has following requirements.
+<ul>
+<li>it must be the primary key (not just part of one)
+<li>its type must be one of the integer types
+</ul>
+<pre><tt>create table t (
+ a int unsigned auto_increment primary key,
+ b char(20) not null,
+ c float
+)
+</tt></pre>
+<p>
+The values of an AUTO_INCREMENT column are unique (until wrap-around)
+and form an ascending sequence.
+Gaps in the sequence are possible.
+<h4>Default values</h4>
+Columns can be specified with DEFAULT value
+which is used on insert if the column is not on the insert list.
+<p>
+<pre><tt>create table t (
+ a int primary key,
+ b int default 100
+)
+insert into t(a) values(1) -- inserts (1,100)
+</tt></pre>
+<p>
+The value must evaluate to constant.
+Using SYSDATE (if allowed at all) evaluates to table creation time.
+<p>
+
+<h4>Logging / nologging</h4>
+
+By default tables are created in logging mode, meaning the data
+is preserved across database restart.
+The mode can be specified explicitly:
+<p>
+<tt>create table t1 (a int primary key, b int) logging</tt>
+<br>
+<tt>create table t1 (a int primary key, b int) nologging</tt>
+
+<h4>Schemas</h4>
+
+Schemas do not exist in current NDB Cluster.
+As a convenience, a single period is allowed in table names:
+<p>
+<pre><tt>create table mydb.mytable (a int primary key)</tt></pre>
+<p>
+
+<h4>Drop table</h4>
+
+Deletes a table, all of its indexes, and all data:
+<p>
+<tt>drop table t</tt>
+
+<h4>3.7 Indexes</h4>
+Only unique non-ordered indexes exist currently.
+The columns must be not nullable and are stored in same
+order as underlying table columns.
+<p>
+<tt>create unique hash index x1 on t1(b, c) logging</tt>
+<p>
+Internally, a unique hash index is a table where index key is primary key.
+If the index is <tt>nologging</tt>, it is rebuilt on database restart
+before the database is opened.
+<p>
+Indexes can of course be dropped:
+<p>
+<tt>drop index x1</tt>
+
+<h4>3.8 Select</h4>
+
+Features:
+
+<ul>
+<li>Expressions and predicates
+<br><tt>select a + b * c from t where a &lt; b + c and (b &gt; c or c &gt; 10)</tt>
+<li>JOIN to any depth
+<br><tt>select a.x, b.y, c.z from t1 a, t2 b, t2 c where a.x + b.y &lt; c.z</tt>
+<li>ORDER BY
+<br><tt>select * from t1, t2 where a1 &gt; 5 order by b1 + b2, c1 desc</tt>
+<li>DISTINCT
+<br><tt>select distinct a, b + c from t</tt>
+<li>Aggregates without grouping.
+<br><tt>select count(*), max(a), 1 + sum(b) + avg(c * d) from t</tt>
+<li>Aggregates with grouping.
+<br><tt>select a, sum(b) from t group by a having sum(c) &gt; 0 order by a, sum(d)</tt>
+</ul>
+
+Major omissions:
+<ul>
+<li>no OUTER JOIN
+<li>no subqueries and no EXISTS clause
+</ul>
+
+Queries are optimized to minimize scans,
+by using primary keys and existing unique hash indexes.
+Simple predicates in scans (column compared to constant)
+are passed to an interpreter in NDB kernel.
+Joins are done via <em>nested loops</em> only.
+<p>
+<ul>
+<li>SCAN
+<br><tt>select * from t where a &lt; b</tt>
+<li>INTERPRETED SCAN (much faster)
+<br><tt>select * from t1, t2 where t1.a &lt; 10 and t2.b &gt; t1.c</tt>
+<li>PRIMARY KEY lookup
+<br><tt>select * from t where pk = 5 and b &gt; 10</tt>
+<li>NESTED LOOPS
+<br><tt>select * from t1, t2, t3 where t1.pk = t2.x and t2.pk = t3.y</tt>
+</ul>
+
+<h4>3.9 Insert and write</h4>
+
+Both VALUES and subquery variants can be used.
+<p>
+<pre><tt>insert into t(a, c) values (123, 'abc')
+insert into t1(a, c) select a + 10 * b, c from t2
+</tt></pre>
+<p>
+For convenience, the non-standard <i>MySql</i> syntax is also supported.
+<p>
+<pre><tt>insert into t set a = 123, c = 'abc'</tt></pre>
+<p>
+The non-standard operation WRITE is used exactly like INSERT.
+The record is updated if it exists.
+Otherwise a new record is inserted.
+<p>
+<pre><tt>write into t(a, c) values (123, 'abc')
+</tt></pre>
+
+<h4>3.10 Update</h4>
+
+Update allows no subqueries.
+Update is optimized to minimize scans and reads,
+by using primary keys and existing unique hash indexes.
+<p>
+<ul>
+<li>SCAN
+<br><tt>update t set a = b + 5, c = d where c &gt; 10</tt>
+<li>PRIMARY KEY or INDEX lookup
+<br><tt>update t set a = b + 5, c = d where pk = 5 and c &gt; 10</tt>
+<li>PRIMARY KEY or INDEX direct
+<br><tt>update t set a = 5, c = 7 where pk = 5</tt>
+</ul>
+
+<h4>3.11 Delete</h4>
+
+Delete allows no subqueries.
+Delete is optimized to minimize scans and reads,
+by using primary keys and existing unique hash indexes.
+<p>
+<ul>
+<li>SCAN
+<br><tt>delete from t where c &gt; 10</tt>
+<li>PRIMARY KEY or INDEX lookup
+<br><tt>delete from t where pk = 5 and c &gt; 10</tt>
+<li>PRIMARY KEY or INDEX direct
+<br><tt>delete from t where pk = 5</tt>
+</ul>
+
+<h4>3.12 Virtual tables</h4>
+
+The driver implements some virtual tables.
+They can only be queried, not modified.
+<p>
+<ul>
+<li>DUAL
+<br>A 1-row table - example: select SYSDATE from DUAL.
+<li>ODBC$TYPEINFO
+<br>Corresponds to SQLGetTypeInfo.
+<li>ODBC$TABLES
+<br>Corresponds to SQLTables but shows all NDB kernel objects.
+<li>ODBC$COLUMNS
+<br>Corresponds to SQLColumns.
+<li>ODBC$PRIMARYKEYS
+<br>Corresponds to SQLPrimaryKeys.
+</ul>
+
+<h3>4. DIAGNOSTICS</h3>
+
+<h4>4.1 Diagnostic records</h4>
+
+The driver manager and driver return 3 main diagnostics
+(see <b><tt>SQLGetDiagRec</tt></b>).
+<ul>
+<li>SQLSTATE (a string of 5 characters)
+<li>Native error code
+<li>Message text
+</ul>
+<p>
+Message text format is
+<br><br>
+<b><tt>[Alzato][ODBC driver][NDB Cluster] NDB-ssccnnn <i>error text</i> (in SQLXxx)</tt></b>
+<br><br>
+Here <b>ssccnnnn</b> is native error code (decimal number), with following parts:
+<p>
+<li><b><tt>ss</tt></b> - status
+<li><b><tt>cc</tt></b> - classification
+<li><b><tt>nnnn</tt></b> - error code
+</ul>
+<p>
+See NDB API guide for further information.
+<p>
+For non-database errors the last prefix <b><tt>[NDB Cluster]</tt></b> is omitted
+and native error code is always 02015001.
+
+<h4>4.2 Tracing</h4>
+
+Following environment variables may be useful.
+<ul>
+<li><b><tt>NDB_ODBC_TRACE</tt></b>
+<br>
+Values &ge; 2 cause SQL execution plan
+to be printed on standard output.
+<br>
+Values &ge; 3 show the ODBC API functions
+called by the driver manager.
+<br><br>
+<li><b><tt>NDB_ODBC_DEBUG</tt></b>
+<br>
+Non-zero value makes the driver abort
+the application on unhandled conditions.
+<br>
+By default the ODBC API function is aborted gracefully.
+</ul>
+
+<h4>4.3 Thread safety</h4>
+<p>
+The driver has same thread-safety model as NDB API.
+In NDB API each thread must use its own <b><tt>Ndb</tt></b> object.
+In NDB ODBC a <b><tt>SQLConnect</tt></b> corresponds to an <b><tt>Ndb</tt></b> object.
+It is required that each thread allocates its
+own ODBC handles (of all types).
+
+<h4>4.4 Data formats</h4>
+<p>
+SQL types are represented as (old) NDB types as follows.
+<p>
+<table width="80%" border=1>
+<tr align="left"><th width="20%">SQL type</th> <th>NDB type</th></tr>
+<tr align="left"><td align="left">CHAR(n)</td><td>String(n), blank-padded to n</td></tr>
+<tr align="left"><td align="left">VARCHAR(n)</td><td>String(n+2), zero-padded to n, length in last 2 bytes (big-endian)</td></tr>
+<tr align="left"><td align="left">integers</td><td>Signed(x) or UnSigned(x), x=16,32,64, native format</td></tr>
+<tr align="left"><td align="left">floats</td><td>Float(x), x=32,64, native format</td></tr>
+<tr align="left"><td align="left">DATETIME</td><td>String(12) = cc yy mm dd HH MM SS \0 ff ff ff ff (big-endian)</td></tr>
+</table>
+<p>
+Note: SQL types exist now in NDB API in <b><tt>NdbDictionary</tt></b> class.
+However they are not yet understood by NDB API operations.
+
+<h4>4.5 NDB Cluster limitations</h4>
+<p>
+<ul>
+<li>Isolation level is READ COMMITTED.
+A scan (non-primary-key select of several rows) does not see consistent data.
+<br><br>
+<li>Inserting into a table from itself is likely to cause a deadlock
+or a random result.
+<br><tt><b>no:</b> insert into t(a, b) select a*100, b+100 from t</tt>
+<br><br>
+<li>Number of uncommitted rows is limited by NDB configuration
+parameter <b><tt>MaxNoOfConcurrentOperations</tt></b> (typical default 4096).
+To delete all rows from a large table one may need to do repeatedly:
+<br><tt>delete from t where rownum < 4000</tt>
+</ul>
+
+<h4>4.6 Known problems v2.11</h4>
+<p>
+Following lists specific known problems.
+<ul>
+<li>ORDER BY works only with expressions,
+not with column aliases or positions.
+<br><tt><b>no:</b> select a+b x from t order by x</tt>
+<br><tt><b>no:</b> select * from t order by 1, 2, 3</tt>
+<br><br>
+<li>Join optimizer does not always minimize number of scans.
+Changing the order of tables in the statement may help.
+</ul>
+
+<h4>4.7 Useful links</h4>
+<p>
+<a href="http://msdn.microsoft.com/library/default.asp?url=/library/en-us/odbc/htm/dasdkodbcoverview_whatsnew.asp/">Microsoft ODBC page</a>
+<br>
+<a href="http://www.unixodbc.org/">unixODBC home page</a>
+<br>
+<a href="http://www.iodbc.org/">iODBC home page</a>
+
+</BODY>
+</HTML>
diff --git a/ndb/src/client/odbc/docs/select.fig b/ndb/src/client/odbc/docs/select.fig
new file mode 100644
index 00000000000..4f51a2085b4
--- /dev/null
+++ b/ndb/src/client/odbc/docs/select.fig
@@ -0,0 +1,94 @@
+#FIG 3.2
+Landscape
+Center
+Inches
+A4
+92.00
+Single
+-2
+1200 2
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 2700 2700 1500 3900
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 3150 2700 3150 3900
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 3600 4500 4800 5700
+2 1 1 1 0 7 50 0 -1 4.000 0 0 7 1 0 2
+ 0 0 1.00 60.00 120.00
+ 3600 2700 4800 3900
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 2775 4500 1200 5700
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 3150 4500 3150 5700
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 2100 4500 2100 3900 600 3900 600 4500 2100 4500
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 2100 6300 2100 5700 600 5700 600 6300 2100 6300
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 3900 4500 3900 3900 2400 3900 2400 4500 3900 4500
+2 4 1 1 0 7 50 0 -1 4.000 0 0 7 0 0 5
+ 5700 4500 5700 3900 4200 3900 4200 4500 5700 4500
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 3900 6300 3900 5700 2400 5700 2400 6300 3900 6300
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 5700 6300 5700 5700 4200 5700 4200 6300 5700 6300
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 8400 2700 8400 2100 6900 2100 6900 2700 8400 2700
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 8400 6300 8400 5700 6900 5700 6900 6300 8400 6300
+2 1 0 1 0 7 50 0 -1 4.000 0 0 7 1 0 2
+ 0 0 1.00 60.00 120.00
+ 7650 2700 7650 5700
+2 1 0 1 0 7 50 0 -1 4.000 0 0 7 1 0 2
+ 0 0 1.00 60.00 120.00
+ 7650 6300 7650 7500
+2 1 1 1 0 7 50 0 -1 4.000 0 0 7 1 0 2
+ 0 0 1.00 60.00 120.00
+ 8100 6300 10575 6900
+2 1 0 1 0 7 50 0 -1 4.000 0 0 7 1 0 2
+ 0 0 1.00 60.00 120.00
+ 8100 2700 10575 3300
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 11700 3900 11700 3300 10200 3300 10200 3900 11700 3900
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 9900 5400 9900 4800 8400 4800 8400 5400 9900 5400
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 11700 5400 11700 4800 10200 4800 10200 5400 11700 5400
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 13500 5400 13500 4800 12000 4800 12000 5400 13500 5400
+2 1 0 1 0 7 50 0 -1 4.000 0 0 7 1 0 2
+ 0 0 1.00 60.00 120.00
+ 10500 3900 9300 4800
+2 1 0 1 0 7 50 0 -1 4.000 0 0 7 1 0 2
+ 0 0 1.00 60.00 120.00
+ 11400 3900 12600 4800
+2 1 0 1 0 7 50 0 -1 4.000 0 0 -1 1 0 2
+ 0 0 1.00 60.00 120.00
+ 10950 3900 10950 4800
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 8400 8100 8400 7500 6900 7500 6900 8100 8400 8100
+2 4 0 1 0 7 50 0 -1 0.000 0 0 7 0 0 5
+ 3900 2700 3900 2100 2400 2100 2400 2700 3900 2700
+2 4 1 1 0 7 50 0 -1 4.000 0 0 7 0 0 5
+ 11700 7500 11700 6900 10200 6900 10200 7500 11700 7500
+4 0 0 50 0 14 12 0.0000 4 135 840 900 4275 table T1\001
+4 0 0 50 0 14 20 0.0000 4 240 4125 900 1125 select A, C, 123 from T1/\001
+4 0 0 50 0 14 12 0.0000 4 135 840 825 6075 column A\001
+4 0 0 50 0 14 12 0.0000 4 135 840 2775 6075 column C\001
+4 0 0 50 0 14 12 0.0000 4 135 945 4425 6075 const 123\001
+4 0 0 50 0 14 12 0.0000 4 135 630 4575 4275 clause\001
+4 0 0 50 0 14 12 0.0000 4 90 315 2925 4275 row\001
+4 0 0 50 0 14 12 0.0000 4 180 735 7200 2475 project\001
+4 0 0 50 0 14 12 0.0000 4 135 630 7200 6000 filter\001
+4 0 0 50 0 14 12 0.0000 4 135 840 8625 5100 column 1\001
+4 0 0 50 0 14 12 0.0000 4 135 840 10500 5100 column 2\001
+4 0 0 50 0 14 12 0.0000 4 135 945 12300 5100 const 123\001
+4 0 0 50 0 14 12 0.0000 4 90 315 10650 3600 row\001
+4 0 0 50 0 14 12 0.0000 4 150 840 7200 7800 scan 1,2\001
+4 0 0 50 0 14 12 0.0000 4 135 630 2850 2475 select\001
+4 0 0 50 0 14 12 0.0000 4 135 630 10500 7200 clause\001
diff --git a/ndb/src/client/odbc/docs/systables.pl b/ndb/src/client/odbc/docs/systables.pl
new file mode 100644
index 00000000000..728d966a7a4
--- /dev/null
+++ b/ndb/src/client/odbc/docs/systables.pl
@@ -0,0 +1,2192 @@
+# usage: perl systables.pl {typeinfo|tables|columns|primarykeys} {-l|-c}
+use strict;
+my $what = shift;
+my $opt = shift;
+my $listWhat = {};
+
+#
+# odbcsqlgettypeinfo.htm
+#
+$listWhat->{typeinfo} = [
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLGetTypeInfo</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqlgettypeinfo"></A>SQLGetTypeInfo</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 1.0<BR>
+# Standards Compliance: ISO 92</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLGetTypeInfo</B> returns information about data types supported by the data source. The driver returns the information in the form of an SQL result set. The data types are intended for use in Data Definition Language (DDL) statements.</P>
+#
+# <P class="indent"><b class="le">Important&nbsp;&nbsp;&nbsp;</b>Applications must use the type names returned in the TYPE_NAME column of the <B>SQLGetTypeInfo</B> result set in <B>ALTER TABLE</B> and <B>CREATE TABLE</B> statements. <B>SQLGetTypeInfo</B> may return more than one row with the same value in the DATA_TYPE column.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLGetTypeInfo</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHSTMT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StatementHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>DataType</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>StatementHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Statement handle for the result set.</dd>
+#
+# <DT><I>DataType</I></DT>
+#
+# <DD>[Input]<BR>
+# The SQL data type. This must be one of the values in the "<A HREF="odbcsql_data_types.htm">SQL Data Types</A>" section of Appendix D: Data Types, or a driver-specific SQL data type. SQL_ALL_TYPES specifies that information about all data types should be returned.</dd>
+# </DL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_STILL_EXECUTING, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLGetTypeInfo</B> returns SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value can be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_STMT and a <I>Handle</I> of <I>StatementHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLGetTypeInfo </B>and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=22%>SQLSTATE</TH>
+# <TH width=26%>Error</TH>
+# <TH width=52%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01000</TD>
+# <TD width=26%>General warning</TD>
+# <TD width=52%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01S02</TD>
+# <TD width=26%>Option value changed</TD>
+# <TD width=52%>A specified statement attribute was invalid because of implementation working conditions, so a similar value was temporarily substituted. (Call <B>SQLGetStmtAttr</B> to determine the temporarily substituted value.) The substitute value is valid for the <I>StatementHandle</I> until the cursor is closed. The statement attributes that can be changed are: SQL_ATTR_CONCURRENCY, SQL_ATTR_CURSOR_TYPE, SQL_ATTR_KEYSET_SIZE, SQL_ATTR_MAX_LENGTH, SQL_ATTR_MAX_ROWS, SQL_ATTR_QUERY_TIMEOUT, and SQL_ATTR_SIMULATE_CURSOR. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>08S01</TD>
+# <TD width=26%>Communication link failure</TD>
+# <TD width=52%>The communication link between the driver and the data source to which the driver was connected failed before the function completed processing.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>24000</TD>
+# <TD width=26%>Invalid cursor state</TD>
+# <TD width=52%>A cursor was open on the <I>StatementHandle,</I> and <B>SQLFetch</B> or <B>SQLFetchScroll</B> had been called. This error is returned by the Driver Manager if <B>SQLFetch</B> or <B>SQLFetchScroll</B> has not returned SQL_NO_DATA, and is returned by the driver if <B>SQLFetch</B> or <B>SQLFetchScroll</B> has returned SQL_NO_DATA.
+# <P>A result set was open on the <I>StatementHandle</I>, but <B>SQLFetch</B> or <B>SQLFetchScroll</B> had not been called.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40001</TD>
+# <TD width=26%>Serialization failure</TD>
+# <TD width=52%>The transaction was rolled back due to a resource deadlock with another transaction.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40003</TD>
+# <TD width=26%>Statement completion unknown</TD>
+# <TD width=52%>The associated connection failed during the execution of this function and the state of the transaction cannot be determined.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY000</TD>
+# <TD width=26%>General error</TD>
+# <TD width=52%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause. </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY001</TD>
+# <TD width=26%>Memory allocation error</TD>
+# <TD width=52%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY004</TD>
+# <TD width=26%>Invalid SQL data type</TD>
+# <TD width=52%>The value specified for the argument <I>DataType</I> was neither a valid ODBC SQL data type identifier nor a driver-specific data type identifier supported by the driver.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY008</TD>
+# <TD width=26%>Operation canceled</TD>
+# <TD width=52%>Asynchronous processing was enabled for the <I>StatementHandle</I>, then the function was called and, before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I>. Then the function was called again on the <I>StatementHandle</I>.
+# <P>The function was called and, before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I> from a different thread in a multithread application.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY010</TD>
+# <TD width=26%>Function sequence error</TD>
+# <TD width=52%>(DM) An asynchronously executing function (not this one) was called for the <I>StatementHandle</I> and was still executing when this function was called.
+# <P>(DM) <B>SQLExecute</B>, <B>SQLExecDirect</B>, <B>SQLBulkOperations</B>, or <B>SQLSetPos</B> was called for the <I>StatementHandle</I> and returned SQL_NEED_DATA. This function was called before data was sent for all data-at-execution parameters or columns.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY013</TD>
+# <TD width=26%>Memory management error</TD>
+# <TD width=52%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYC00</TD>
+# <TD width=26%>Optional feature not implemented</TD>
+# <TD width=52%>The combination of the current settings of the SQL_ATTR_CONCURRENCY and SQL_ATTR_CURSOR_TYPE statement attributes was not supported by the driver or data source.
+# <P>The SQL_ATTR_USE_BOOKMARKS statement attribute was set to SQL_UB_VARIABLE, and the SQL_ATTR_CURSOR_TYPE statement attribute was set to a cursor type for which the driver does not support bookmarks.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT00</TD>
+# <TD width=26%>Timeout expired</TD>
+# <TD width=52%>The query timeout period expired before the data source returned the result set. The timeout period is set through <B>SQLSetStmtAttr</B>, SQL_ATTR_QUERY_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT01</TD>
+# <TD width=26%>Connection timeout expired</TD>
+# <TD width=52%>The connection timeout period expired before the data source responded to the request. The connection timeout period is set through <B>SQLSetConnectAttr</B>, SQL_ATTR_CONNECTION_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>IM001</TD>
+# <TD width=26%>Driver does not support this function</TD>
+# <TD width=52%>(DM) The driver corresponding to the <I>StatementHandle</I> does not support the function.</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Comments</B></P>
+#
+# <P><B>SQLGetTypeInfo</B> returns the results as a standard result set, ordered by DATA_TYPE and then by how closely the data type maps to the corresponding ODBC SQL data type. Data types defined by the data source take precedence over user-defined data types. Consequently, the sort order is not necessarily consistent but can be generalized as DATA_TYPE first, followed by TYPE_NAME, both ascending. For example, suppose that a data source defined INTEGER and COUNTER data types, where COUNTER is auto-incrementing, and that a user-defined data type WHOLENUM has also been defined. These would be returned in the order INTEGER, WHOLENUM, and COUNTER, because WHOLENUM maps closely to the ODBC SQL data type SQL_INTEGER, while the auto-incrementing data type, even though supported by the data source, does not map closely to an ODBC SQL data type. For information about how this information might be used, see "<A HREF="odbcddl_statements.htm">DDL Statements</A>" in Chapter 8: SQL Statements.</P>
+#
+# <P>If the <I>DataType</I> argument specifies a data type which is valid for the version of ODBC supported by the driver, but is not supported by the driver, then it will return an empty result set. </P>
+#
+# <P class="indent"><b class="le">Note&nbsp;&nbsp;&nbsp;</b>For more information about the general use, arguments, and returned data of ODBC catalog functions, see <A HREF="odbccatalog_functions.htm">Chapter 7: Catalog Functions</A>.</P>
+#
+# <P>The following columns have been renamed for ODBC 3.<I>x</I>. The column name changes do not affect backward compatibility because applications bind by column number.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=48%>ODBC 2.0 column</TH>
+# <TH width=52%>ODBC 3.<I>x</I> column</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>PRECISION</TD>
+# <TD width=52%>COLUMN_SIZE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>MONEY</TD>
+# <TD width=52%>FIXED_PREC_SCALE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>AUTO_INCREMENT</TD>
+# <TD width=52%>AUTO_UNIQUE_VALUE</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P>The following columns have been added to the results set returned by <B>SQLGetTypeInfo</B> for ODBC 3.<I>x</I>:
+#
+# <UL type=disc>
+# <LI>SQL_DATA_TYPE</li>
+#
+# <LI>INTERVAL_PRECISION</li>
+#
+# <LI>SQL_DATETIME_SUB</li>
+#
+# <LI>NUM_PREC_RADIX</li>
+# </UL>
+#
+# <P>The following table lists the columns in the result set. Additional columns beyond column 19 (INTERVAL_PRECISION) can be defined by the driver. An application should gain access to driver-specific columns by counting down from the end of the result set rather than specifying an explicit ordinal position. For more information, see "<A HREF="odbcdata_returned_by_catalog_functions.htm">Data Returned by Catalog Functions</A>" in Chapter 7: Catalog Functions.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;<B>SQLGetTypeInfo</B> might not return all data types. For example, a driver might not return user-defined data types. Applications can use any valid data type, regardless of whether it is returned by <B>SQLGetTypeInfo</B>.</P>
+#
+# <P class="indent">The data types returned by <B>SQLGetTypeInfo</B> are those supported by the data source. They are intended for use in Data Definition Language (DDL) statements. Drivers can return result-set data using data types other than the types returned by <B>SQLGetTypeInfo</B>. In creating the result set for a catalog function, the driver might use a data type that is not supported by the data source. </P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=33%><BR>
+# Column name</TH>
+# <TH width=14%>Column <BR>
+# number</TH>
+# <TH width=15%><BR>
+# Data type</TH>
+# <TH width=38%><BR>
+# Comments</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=33%>TYPE_NAME<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>1</TD>
+# <TD width=15%>Varchar<BR>
+# not NULL</TD>
+# <TD width=38%>Data source&#0150;dependent data-type name; for example, "CHAR()", "VARCHAR()", "MONEY", "LONG VARBINARY", or "CHAR ( ) FOR BIT DATA". Applications must use this name in <B>CREATE TABLE</B> and <B>ALTER TABLE</B> statements.</TD>
+# </TR>
+ { name => "type_name",
+ type => "varchar",
+ length => 20,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>DATA_TYPE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>2</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>SQL data type. This can be an ODBC SQL data type or a driver-specific SQL data type. For datetime or interval data types, this column returns the concise data type (such as SQL_TYPE_TIME or SQL_INTERVAL_YEAR_TO_MONTH). For a list of valid ODBC SQL data types, see "<A HREF="odbcsql_data_types.htm">SQL Data Types</A>" in Appendix D: Data Types. For information about driver-specific SQL data types, see the driver’s documentation.</TD>
+# </TR>
+ { name => "data_type",
+ type => "smallint",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>COLUMN_SIZE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>3</TD>
+# <TD width=15%>Integer</TD>
+# <TD width=38%>The maximum column size that the server supports for this data type. For numeric data, this is the maximum precision. For string data, this is the length in characters. For datetime data types, this is the length in characters of the string representation (assuming the maximum allowed precision of the fractional seconds component). NULL is returned for data types where column size is not applicable. For interval data types, this is the number of characters in the character representation of the interval literal (as defined by the interval leading precision; see "<A HREF="odbcinterval_data_type_length.htm">Interval Data Type Length</A>" in Appendix D: Data Types).
+# <P>For more information on column size, see "<A HREF="odbccolumn_size__decimal_digits__transfer_octet_length__and_display_size.htm">Column Size, Decimal Digits, Transfer Octet Length, and Display Size</A>" in Appendix D: Data Types.</P>
+# </TD>
+# </TR>
+ { name => "column_size",
+ type => "integer",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>LITERAL_PREFIX<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>4</TD>
+# <TD width=15%>Varchar</TD>
+# <TD width=38%>Character or characters used to prefix a literal; for example, a single quotation mark (') for character data types or 0x for binary data types; NULL is returned for data types where a literal prefix is not applicable.</TD>
+# </TR>
+ { name => "literal_prefix",
+ type => "varchar",
+ length => 1,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>LITERAL_SUFFIX<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>5</TD>
+# <TD width=15%>Varchar</TD>
+# <TD width=38%>Character or characters used to terminate a literal; for example, a single quotation mark (') for character data types; NULL is returned for data types where a literal suffix is not applicable.</TD>
+# </TR>
+ { name => "literal_suffix",
+ type => "varchar",
+ length => 1,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>CREATE_PARAMS<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>6</TD>
+# <TD width=15%>Varchar</TD>
+# <TD width=38%>A list of keywords, separated by commas, corresponding to each parameter that the application may specify in parentheses when using the name that is returned in the TYPE_NAME field. The keywords in the list can be any of the following: length, precision, or scale. They appear in the order that the syntax requires them to be used. For example, CREATE_PARAMS for DECIMAL would be "precision,scale"; CREATE_PARAMS for VARCHAR would equal "length." NULL is returned if there are no parameters for the data type definition; for example, INTEGER.
+# <P>The driver supplies the CREATE_PARAMS text in the language of the country where it is used.</P>
+# </TD>
+# </TR>
+ { name => "create_params",
+ type => "varchar",
+ length => 20,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>NULLABLE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>7</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>Whether the data type accepts a NULL value:
+# <P>SQL_NO_NULLS if the data type does not accept NULL values.</P>
+#
+# <P>SQL_NULLABLE if the data type accepts NULL values.</P>
+#
+# <P>SQL_NULLABLE_UNKNOWN if it is not known whether the column accepts NULL values.</P>
+# </TD>
+# </TR>
+ { name => "nullable",
+ type => "smallint",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>CASE_SENSITIVE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>8</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>Whether a character data type is case-sensitive in collations and comparisons:
+# <P>SQL_TRUE if the data type is a character data type and is case-sensitive.</P>
+#
+# <P>SQL_FALSE if the data type is not a character data type or is not case-sensitive.</P>
+# </TD>
+# </TR>
+ { name => "case_sensitive",
+ type => "smallint",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>SEARCHABLE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>9</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>How the data type is used in a <B>WHERE</B> clause:
+# <P>SQL_PRED_NONE if the column cannot be used in a <B>WHERE</B> clause. (This is the same as the SQL_UNSEARCHABLE value in ODBC 2.<I>x</I>.)</P>
+#
+# <P>SQL_PRED_CHAR if the column can be used in a <B>WHERE</B> clause, but only with the <B>LIKE</B> predicate. (This is the same as the SQL_LIKE_ONLY value in ODBC 2.<I>x</I>.)</P>
+#
+# <P>SQL_PRED_BASIC if the column can be used in a <B>WHERE</B> clause with all the comparison operators except <B>LIKE</B> (comparison, quantified comparison, <B>BETWEEN</B>, <B>DISTINCT</B>, <B>IN</B>, <B>MATCH</B>, and <B>UNIQUE</B>). (This is the same as the SQL_ALL_EXCEPT_LIKE value in ODBC 2.<I>x</I>.)</P>
+#
+# <P>SQL_SEARCHABLE if the column can be used in a <B>WHERE</B> clause with any comparison operator.</P>
+# </TD>
+# </TR>
+ { name => "searchable",
+ type => "smallint",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>UNSIGNED_ATTRIBUTE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>10</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>Whether the data type is unsigned:
+# <P>SQL_TRUE if the data type is unsigned.</P>
+#
+# <P>SQL_FALSE if the data type is signed.</P>
+#
+# <P>NULL is returned if the attribute is not applicable to the data type or the data type is not numeric.</P>
+# </TD>
+# </TR>
+ { name => "unsigned_attribute",
+ type => "smallint",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>FIXED_PREC_SCALE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>11</TD>
+# <TD width=15%>Smallint<BR>
+# not NULL</TD>
+# <TD width=38%>Whether the data type has predefined fixed precision and scale (which are data source&#0150;specific), such as a money data type:
+# <P>SQL_TRUE if it has predefined fixed precision and scale.</P>
+#
+# <P>SQL_FALSE if it does not have predefined fixed precision and scale.</P>
+# </TD>
+# </TR>
+ { name => "fixed_prec_scale",
+ type => "smallint",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>AUTO_UNIQUE_VALUE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>12</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>Whether the data type is autoincrementing:
+# <P>SQL_TRUE if the data type is autoincrementing.</P>
+#
+# <P>SQL_FALSE if the data type is not autoincrementing.</P>
+#
+# <P>NULL is returned if the attribute is not applicable to the data type or the data type is not numeric.</P>
+#
+# <P>An application can insert values into a column having this attribute, but typically cannot update the values in the column. </P>
+#
+# <P>When an insert is made into an auto-increment column, a unique value is inserted into the column at insert time. The increment is not defined, but is data source&#0150;specific. An application should not assume that an auto-increment column starts at any particular point or increments by any particular value.</P>
+# </TD>
+# </TR>
+ { name => "auto_unique_value",
+ type => "smallint",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>LOCAL_TYPE_NAME<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>13</TD>
+# <TD width=15%>Varchar</TD>
+# <TD width=38%>Localized version of the data source&#0150;dependent name of the data type. NULL is returned if a localized name is not supported by the data source. This name is intended for display only, such as in dialog boxes.</TD>
+# </TR>
+ { name => "local_type_name",
+ type => "varchar",
+ length => 20,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>MINIMUM_SCALE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>14</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>The minimum scale of the data type on the data source. If a data type has a fixed scale, the MINIMUM_SCALE and MAXIMUM_SCALE columns both contain this value. For example, an SQL_TYPE_TIMESTAMP column might have a fixed scale for fractional seconds. NULL is returned where scale is not applicable. For more information, see "<A HREF="odbccolumn_size__decimal_digits__transfer_octet_length__and_display_size.htm">Column Size, Decimal Digits, Transfer Octet Length, and Display Size</A>" in Appendix D: Data Types.</TD>
+# </TR>
+ { name => "minimum_scale",
+ type => "smallint",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>MAXIMUM_SCALE<BR>
+# (ODBC 2.0)</TD>
+# <TD width=14%>15</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>The maximum scale of the data type on the data source. NULL is returned where scale is not applicable. If the maximum scale is not defined separately on the data source, but is instead defined to be the same as the maximum precision, this column contains the same value as the COLUMN_SIZE column. For more information, see "<A HREF="odbccolumn_size__decimal_digits__transfer_octet_length__and_display_size.htm">Column Size, Decimal Digits, Transfer Octet Length, and Display Size</A>" in Appendix D: Data Types.</TD>
+# </TR>
+ { name => "maximum_scale",
+ type => "smallint",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>SQL_DATA_TYPE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=14%>16</TD>
+# <TD width=15%>Smallint NOT NULL</TD>
+# <TD width=38%>The value of the SQL data type as it appears in the SQL_DESC_TYPE field of the descriptor. This column is the same as the DATA_TYPE column, except for interval and datetime data types.
+# <P>For interval and datetime data types, the SQL_DATA_TYPE field in the result set will return SQL_INTERVAL or SQL_DATETIME, and the SQL_DATETIME_SUB field will return the subcode for the specific interval or datetime data type. (See <A HREF="odbcdata_types.htm">Appendix D: Data Types</A>.) </P>
+# </TD>
+# </TR>
+ { name => "sql_data_type",
+ type => "smallint",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>SQL_DATETIME_SUB<BR>
+# (ODBC 3.0)</TD>
+# <TD width=14%>17</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>When the value of SQL_DATA_TYPE is SQL_DATETIME or SQL_INTERVAL, this column contains the datetime/interval subcode. For data types other than datetime and interval, this field is NULL.
+# <P>For interval or datetime data types, the SQL_DATA_TYPE field in the result set will return SQL_INTERVAL or SQL_DATETIME, and the SQL_DATETIME_SUB field will return the subcode for the specific interval or datetime data type. (See <A HREF="odbcdata_types.htm">Appendix D: Data Types</A>.)</P>
+# </TD>
+# </TR>
+ { name => "sql_datetime_sub",
+ type => "smallint",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>NUM_PREC_RADIX<BR>
+# (ODBC 3.0)</TD>
+# <TD width=14%>18</TD>
+# <TD width=15%>Integer</TD>
+# <TD width=38%>If the data type is an approximate numeric type, this column contains the value 2 to indicate that COLUMN_SIZE specifies a number of bits. For exact numeric types, this column contains the value 10 to indicate that COLUMN_SIZE specifies a number of decimal digits. Otherwise, this column is NULL.</TD>
+# </TR>
+ { name => "num_prec_radix",
+ type => "integer",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=33%>INTERVAL_PRECISION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=14%>19</TD>
+# <TD width=15%>Smallint</TD>
+# <TD width=38%>If the data type is an interval data type, then this column contains the value of the interval leading precision. (See "<A HREF="odbcinterval_data_type_precision.htm">Interval Data Type Precision</A>" in Appendix D: Data Types.) Otherwise, this column is NULL.</TD>
+# </TR>
+ { name => "interval_precision",
+ type => "smallint",
+ length => undef,
+ nullable => 1,
+ },
+# </table></div>
+# <!--TS:-->
+# <P>Attribute information can apply to data types or to specific columns in a result set. <B>SQLGetTypeInfo</B> returns information about attributes associated with data types; <B>SQLColAttribute</B> returns information about attributes associated with columns in a result set.</P>
+#
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=50%>For information about</TH>
+# <TH width=50%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Binding a buffer to a column in a result set</TD>
+# <TD width=50%><A HREF="odbcsqlbindcol.htm">SQLBindCol</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Canceling statement processing</TD>
+# <TD width=50%><A HREF="odbcsqlcancel.htm">SQLCancel</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning information about a column in a result set</TD>
+# <TD width=50%><A HREF="odbcsqlcolattribute.htm">SQLColAttribute</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Fetching a block of data or scrolling through a result set</TD>
+# <TD width=50%><A HREF="odbcsqlfetchscroll.htm">SQLFetchScroll</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Fetching a single row or a block of data in a forward-only direction</TD>
+# <TD width=50%><A HREF="odbcsqlfetch.htm">SQLFetch</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning information about a driver or data source</TD>
+# <TD width=50%><A HREF="odbcsqlgetinfo.htm">SQLGetInfo</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+];
+
+#
+# odbcsqltables.htm
+#
+$listWhat->{tables} = [
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLTables</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqltables"></A>SQLTables</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 1.0<BR>
+# Standards Compliance: X/Open</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLTables</B> returns the list of table, catalog, or schema names, and table types, stored in a specific data source. The driver returns the information as a result set.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLTables</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHSTMT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StatementHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>CatalogName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength1</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>SchemaName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength2</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>TableName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength3</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>TableType</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength4</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>StatementHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Statement handle for retrieved results.</dd>
+#
+# <DT><I>CatalogName</I></DT>
+#
+# <DD>[Input]<BR>
+# Catalog name. The <I>CatalogName</I> argument accepts search patterns if the SQL_ODBC_VERSION environment attribute is SQL_OV_ODBC3; it does not accept search patterns if SQL_OV_ODBC2 is set. If a driver supports catalogs for some tables but not for others, such as when a driver retrieves data from different DBMSs, an empty string ("") denotes those tables that do not have catalogs.
+#
+# <P>If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>CatalogName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>CatalogName</I> is a pattern value argument; it is treated literally, and its case is significant. For more information, see "<A HREF="odbcarguments_in_catalog_functions.htm">Arguments in Catalog Functions</A>" in Chapter 7: Catalog Functions.
+# </dd>
+#
+# <DT><I>NameLength1</I></DT>
+#
+# <DD>[Input]<BR>
+# Length of *<I>CatalogName</I>.</dd>
+#
+# <DT><I>SchemaName</I></DT>
+#
+# <DD>[Input]<BR>
+# String search pattern for schema names. If a driver supports schemas for some tables but not for others, such as when the driver retrieves data from different DBMSs, an empty string ("") denotes those tables that do not have schemas.
+#
+# <P>If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>SchemaName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>SchemaName</I> is a pattern value argument; it is treated literally, and its case is significant.
+# </dd>
+#
+# <DT><I>NameLength2</I></DT>
+#
+# <DD>[Input]<BR>
+# Length of *<I>SchemaName</I>.</dd>
+#
+# <DT><I>TableName</I></DT>
+#
+# <DD>[Input]<BR>
+# String search pattern for table names.
+#
+# <P>If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>TableName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>TableName</I> is a pattern value argument; it is treated literally, and its case is significant.
+# </dd>
+#
+# <DT><I>NameLength3</I></DT>
+#
+# <DD>[Input]<BR>
+# Length of *<I>TableName</I>.</dd>
+#
+# <DT><I>TableType</I></DT>
+#
+# <DD>[Input]<BR>
+# List of table types to match.
+#
+# <P>Note that the SQL_ATTR_METADATA_ID statement attribute has no effect upon the <I>TableType</I> argument. <I>TableType</I> is a value list argument, no matter what the setting of SQL_ATTR_METADATA_ID.
+# </dd>
+#
+# <DT><I>NameLength4</I></DT>
+#
+# <DD>[Input]<BR>
+# Length of *<I>TableType</I>.</dd>
+# </DL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_STILL_EXECUTING, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLTables</B> returns SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value may be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_STMT and a <I>Handle</I> of <I>StatementHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLTables</B> and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=22%>SQLSTATE</TH>
+# <TH width=26%>Error</TH>
+# <TH width=52%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01000</TD>
+# <TD width=26%>General warning</TD>
+# <TD width=52%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>08S01</TD>
+# <TD width=26%>Communication link failure</TD>
+# <TD width=52%>The communication link between the driver and the data source to which the driver was connected failed before the function completed processing.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>24000</TD>
+# <TD width=26%>Invalid cursor state</TD>
+# <TD width=52%>A cursor was open on the <I>StatementHandle</I>, and <B>SQLFetch</B> or <B>SQLFetchScroll</B> had been called. This error is returned by the Driver Manager if <B>SQLFetch</B> or <B>SQLFetchScroll</B> has not returned SQL_NO_DATA and is returned by the driver if <B>SQLFetch</B> or <B>SQLFetchScroll</B> has returned SQL_NO_DATA.
+# <P>A cursor was open on the <I>StatementHandle</I>, but <B>SQLFetch</B> or <B>SQLFetchScroll</B> had not been called.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40001</TD>
+# <TD width=26%>Serialization failure</TD>
+# <TD width=52%>The transaction was rolled back due to a resource deadlock with another transaction.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40003</TD>
+# <TD width=26%>Statement completion unknown</TD>
+# <TD width=52%>The associated connection failed during the execution of this function, and the state of the transaction cannot be determined.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY000</TD>
+# <TD width=26%>General error</TD>
+# <TD width=52%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY001</TD>
+# <TD width=26%>Memory allocation <BR>
+# error</TD>
+# <TD width=52%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY008</TD>
+# <TD width=26%>Operation canceled</TD>
+# <TD width=52%>Asynchronous processing was enabled for the <I>StatementHandle</I>. The function was called, and before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I>. Then the function was called again on the <I>StatementHandle</I>.
+# <P>The function was called, and before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I> from a different thread in a multithread application.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY009</TD>
+# <TD width=26%>Invalid use of null pointer</TD>
+# <TD width=52%>The SQL_ATTR_METADATA_ID statement attribute was set to SQL_TRUE, the <I>CatalogName</I> argument was a null pointer, and the SQL_CATALOG_NAME <I>InfoType</I> returns that catalog names are supported.
+# <P>(DM) The SQL_ATTR_METADATA_ID statement attribute was set to SQL_TRUE, and the <I>SchemaName</I> or <I>TableName</I> argument was a null pointer.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY010</TD>
+# <TD width=26%>Function sequence error</TD>
+# <TD width=52%>(DM) An asynchronously executing function (not this one) was called for the <I>StatementHandle</I> and was still executing when this function was called.
+# <P>(DM) <B>SQLExecute</B>, <B>SQLExecDirect</B>, <B>SQLBulkOperations</B>, or <B>SQLSetPos</B> was called for the <I>StatementHandle</I> and returned SQL_NEED_DATA. This function was called before data was sent for all data-at-execution parameters or columns.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY013</TD>
+# <TD width=26%>Memory management error</TD>
+# <TD width=52%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY090</TD>
+# <TD width=26%>Invalid string or buffer length</TD>
+# <TD width=52%>(DM) The value of one of the length arguments was less than 0 but not equal to SQL_NTS.
+# <P>The value of one of the name length arguments exceeded the maximum length value for the corresponding name.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYC00</TD>
+# <TD width=26%>Optional feature not implemented</TD>
+# <TD width=52%>A catalog was specified, and the driver or data source does not support catalogs.
+# <P>A schema was specified, and the driver or data source does not support schemas.</P>
+#
+# <P>A string search pattern was specified for the catalog name, table schema, or table name, and the data source does not support search patterns for one or more of those arguments.</P>
+#
+# <P>The combination of the current settings of the SQL_ATTR_CONCURRENCY and SQL_ATTR_CURSOR_TYPE statement attributes was not supported by the driver or data source. </P>
+#
+# <P>The SQL_ATTR_USE_BOOKMARKS statement attribute was set to SQL_UB_VARIABLE, and the SQL_ATTR_CURSOR_TYPE statement attribute was set to a cursor type for which the driver does not support bookmarks.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT00</TD>
+# <TD width=26%>Timeout expired</TD>
+# <TD width=52%>The query timeout period expired before the data source returned the requested result set. The timeout period is set through <B>SQLSetStmtAttr</B>, SQL_ATTR_QUERY_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT01</TD>
+# <TD width=26%>Connection timeout expired</TD>
+# <TD width=52%>The connection timeout period expired before the data source responded to the request. The connection timeout period is set through <B>SQLSetConnectAttr</B>, SQL_ATTR_CONNECTION_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>IM001</TD>
+# <TD width=26%>Driver does not support this function</TD>
+# <TD width=52%>(DM) The driver associated with the <I>StatementHandle</I> does not support the function.</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Comments</B></P>
+#
+# <P><B>SQLTables</B> lists all tables in the requested range. A user may or may not have SELECT privileges to any of these tables. To check accessibility, an application can:
+#
+# <UL type=disc>
+# <LI>Call <B>SQLGetInfo</B> and check the SQL_ACCESSIBLE_TABLES information type.</li>
+#
+# <LI>Call <B>SQLTablePrivileges</B> to check the privileges for each table.</li>
+# </UL>
+#
+# <P>Otherwise, the application must be able to handle a situation where the user selects a table for which <B>SELECT</B> privileges are not granted.</P>
+#
+# <P>The <I>SchemaName</I> and <I>TableName</I> arguments accept search patterns. The <I>CatalogName</I> argument accepts search patterns if the SQL_ODBC_VERSION environment attribute is SQL_OV_ODBC3; it does not accept search patterns if SQL_OV_ODBC2 is set. If SQL_OV_ODBC3 is set, an ODBC 3<I>.x</I> driver will require that wildcard characters in the <I>CatalogName</I> argument be escaped to be treated literally. For more information about valid search patterns, see "<A HREF="odbcpattern_value_arguments.htm">Pattern Value Arguments</A>" in Chapter 7: Catalog Functions.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;For more information about the general use, arguments, and returned data of ODBC catalog functions, see <A HREF="odbccatalog_functions.htm">Chapter 7: Catalog Functions</A>.</P>
+#
+# <P>To support enumeration of catalogs, schemas, and table types, the following special semantics are defined for the <I>CatalogName</I>, <I>SchemaName</I>, <I>TableName</I>, and <I>TableType</I> arguments of <B>SQLTables</B>:
+#
+# <UL type=disc>
+# <LI>If <I>CatalogName</I> is SQL_ALL_CATALOGS and <I>SchemaName</I> and <I>TableName</I> are empty strings, the result set contains a list of valid catalogs for the data source. (All columns except the TABLE_CAT column contain NULLs.)</li>
+#
+# <LI>If <I>SchemaName</I> is SQL_ALL_SCHEMAS and <I>CatalogName</I> and <I>TableName</I> are empty strings, the result set contains a list of valid schemas for the data source. (All columns except the TABLE_SCHEM column contain NULLs.)</li>
+#
+# <LI>If <I>TableType</I> is SQL_ALL_TABLE_TYPES and <I>CatalogName</I>, <I>SchemaName</I>, and <I>TableName</I> are empty strings, the result set contains a list of valid table types for the data source. (All columns except the TABLE_TYPE column contain NULLs.)</li>
+# </UL>
+#
+# <P>If <I>TableType</I> is not an empty string, it must contain a list of comma-separated values for the types of interest; each value may be enclosed in single quotation marks (') or unquoted&#0151;for example, 'TABLE', 'VIEW' or TABLE, VIEW. An application should always specify the table type in uppercase; the driver should convert the table type to whatever case is needed by the data source. If the data source does not support a specified table type, <B>SQLTables</B> does not return any results for that type.</P>
+#
+# <P><B>SQLTables</B> returns the results as a standard result set, ordered by TABLE_TYPE, TABLE_CAT, TABLE_SCHEM, and TABLE_NAME. For information about how this information might be used, see "<A HREF="odbcuses_of_catalog_data.htm">Uses of Catalog Data</A>" in Chapter 7: Catalog Functions.</P>
+#
+# <P>To determine the actual lengths of the TABLE_CAT, TABLE_SCHEM, and TABLE_NAME columns, an application can call <B>SQLGetInfo</B> with the SQL_MAX_CATALOG_NAME_LEN, SQL_MAX_SCHEMA_NAME_LEN, and SQL_MAX_TABLE_NAME_LEN information types.</P>
+#
+# <P>The following columns have been renamed for ODBC 3<I>.x</I>. The column name changes do not affect backward compatibility because applications bind by column number.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=48%>ODBC 2.0 column</TH>
+# <TH width=52%>ODBC 3<I>.x</I> column</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>TABLE_QUALIFIER</TD>
+# <TD width=52%>TABLE_CAT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>TABLE_OWNER</TD>
+# <TD width=52%>TABLE_SCHEM</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P>The following table lists the columns in the result set. Additional columns beyond column 5 (REMARKS) can be defined by the driver. An application should gain access to driver-specific columns by counting down from the end of the result set rather than specifying an explicit ordinal position. For more information, see "<A HREF="odbcdata_returned_by_catalog_functions.htm">Data Returned by Catalog Functions</A>" in Chapter 7: Catalog Functions.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=27%><BR>
+# Column name</TH>
+# <TH width=15%>Column number</TH>
+# <TH width=16%><BR>
+# Data type</TH>
+# <TH width=42%><BR>
+# Comments</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>TABLE_CAT<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>1</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=42%>Catalog name; NULL if not applicable to the data source. If a driver supports catalogs for some tables but not for others, such as when the driver retrieves data from different DBMSs, it returns an empty string ("") for those tables that do not have catalogs.</TD>
+# </TR>
+ { name => "table_cat",
+ type => "varchar",
+ length => 16,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=27%>TABLE_SCHEM<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>2</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=42%>Schema name; NULL if not applicable to the data source. If a driver supports schemas for some tables but not for others, such as when the driver retrieves data from different DBMSs, it returns an empty string ("") for those tables that do not have schemas.</TD>
+# </TR>
+ { name => "table_schem",
+ type => "varchar",
+ length => 16,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=27%>TABLE_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>3</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=42%>Table name.</TD>
+# </TR>
+ { name => "table_name",
+ type => "varchar",
+ length => 16,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=27%>TABLE_TYPE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>4</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=42%>Table type name; one of the following: "TABLE", "VIEW", "SYSTEM TABLE", "GLOBAL TEMPORARY", "LOCAL TEMPORARY", "ALIAS", "SYNONYM", or a data source&#0150;specific type name.
+# <P>The meanings of "ALIAS" and "SYNONYM" are driver-specific.</P>
+# </TD>
+# </TR>
+ { name => "table_type",
+ type => "varchar",
+ length => 20,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=27%>REMARKS<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>5</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=42%>A description of the table.</TD>
+# </TR>
+ { name => "remarks",
+ type => "varchar",
+ length => 200,
+ nullable => 1,
+ },
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Code Example</B></P>
+#
+# <P>For a code example of a similar function, see <A HREF="odbcsqlcolumns.htm">SQLColumns</A>.</P>
+#
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=43%>For information about</TH>
+# <TH width=57%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>Binding a buffer to a column in a result set</TD>
+# <TD width=57%><A HREF="odbcsqlbindcol.htm">SQLBindCol</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>Canceling statement processing</TD>
+# <TD width=57%><A HREF="odbcsqlcancel.htm">SQLCancel</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>Returning privileges for a column or columns</TD>
+# <TD width=57%><A HREF="odbcsqlcolumnprivileges.htm">SQLColumnPrivileges</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>Returning the columns in a table or tables</TD>
+# <TD width=57%><A HREF="odbcsqlcolumns.htm">SQLColumns</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>Fetching a single row or a block of data in a forward-only direction</TD>
+# <TD width=57%><A HREF="odbcsqlfetch.htm">SQLFetch</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>Fetching a block of data or scrolling through a result set</TD>
+# <TD width=57%><A HREF="odbcsqlfetchscroll.htm">SQLFetchScroll</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>Returning table statistics and indexes</TD>
+# <TD width=57%><A HREF="odbcsqlstatistics.htm">SQLStatistics</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=43%>Returning privileges for a table or tables</TD>
+# <TD width=57%><A HREF="odbcsqltableprivileges.htm">SQLTablePrivileges</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+];
+
+#
+# odbcsqlcolumns.htm
+#
+$listWhat->{columns} = [
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLColumns</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqlcolumns"></A>SQLColumns</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 1.0<BR>
+# Standards Compliance: X/Open</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLColumns</B> returns the list of column names in specified tables. The driver returns this information as a result set on the specified <I>StatementHandle</I>.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLColumns</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHSTMT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StatementHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>CatalogName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength1</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>SchemaName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength2</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>TableName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength3</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>ColumnName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength4</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>StatementHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Statement handle.</dd>
+#
+# <DT><I>CatalogName</I></DT>
+#
+# <DD>[Input]<BR>
+# Catalog name. If a driver supports catalogs for some tables but not for others, such as when the driver retrieves data from different DBMSs, an empty string ("") denotes those tables that do not have catalogs. <I>CatalogName</I> cannot contain a string search pattern.</dd>
+# </DL>
+#
+# <BLOCKQUOTE>
+# If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>CatalogName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>CatalogName</I> is an ordinary argument; it is treated literally, and its case is significant. For more information, see "<A HREF="odbcarguments_in_catalog_functions.htm">Arguments in Catalog Functions</A>" in Chapter 7: Catalog Functions.</BLOCKQUOTE>
+#
+# <DL>
+# <DT><I>NameLength1</I></DT>
+#
+# <DD>[Input]<BR>
+# Length of *<I>CatalogName</I>.</dd>
+#
+# <DT><I>SchemaName</I></DT>
+#
+# <DD>[Input]<BR>
+# String search pattern for schema names. If a driver supports schemas for some tables but not for others, such as when the driver retrieves data from different DBMSs, an empty string ("") denotes those tables that do not have schemas.</dd>
+# </DL>
+#
+# <BLOCKQUOTE>
+# If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>SchemaName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>SchemaName</I> is a pattern value argument; it is treated literally, and its case is significant.</BLOCKQUOTE>
+#
+# <DL>
+# <DT><I>NameLength2</I></DT>
+#
+# <DD>[Input]<BR>
+# Length of *<I>SchemaName</I>.</dd>
+#
+# <DT><I>TableName</I></DT>
+#
+# <DD>[Input]<BR>
+# String search pattern for table names.</dd>
+# </DL>
+#
+# <BLOCKQUOTE>
+# If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>TableName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>TableName</I> is a pattern value argument; it is treated literally, and its case is significant.</BLOCKQUOTE>
+#
+# <DL>
+# <DT><I>NameLength3</I></DT>
+#
+# <DD>[Input]<BR>
+# Length of *<I>TableName</I>.</dd>
+#
+# <DT><I>ColumnName</I></DT>
+#
+# <DD>[Input]<BR>
+# String search pattern for column names. </dd>
+# </DL>
+#
+# <BLOCKQUOTE>
+# If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>ColumnName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>ColumnName</I> is a pattern value argument; it is treated literally, and its case is significant.</BLOCKQUOTE>
+#
+# <DL>
+# <DT><I>NameLength4</I></DT>
+#
+# <DD>[Input]<BR>
+# Length of *<I>ColumnName</I>.</dd>
+# </DL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_STILL_EXECUTING, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLColumns</B> returns SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value may be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_STMT and a <I>Handle</I> of <I>StatementHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLColumns</B> and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=22%>SQLSTATE</TH>
+# <TH width=26%>Error</TH>
+# <TH width=52%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01000</TD>
+# <TD width=26%>General warning</TD>
+# <TD width=52%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>08S01</TD>
+# <TD width=26%>Communication link failure</TD>
+# <TD width=52%>The communication link between the driver and the data source to which the driver was connected failed before the function completed processing.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>24000</TD>
+# <TD width=26%>Invalid cursor state</TD>
+# <TD width=52%>A cursor was open on the <I>StatementHandle</I>, and <B>SQLFetch</B> or <B>SQLFetchScroll</B> had been called. This error is returned by the Driver Manager if <B>SQLFetch</B> or <B>SQLFetchScroll</B> has not returned SQL_NO_DATA, and is returned by the driver if <B>SQLFetch</B> or <B>SQLFetchScroll</B> has returned SQL_NO_DATA.
+# <P>A cursor was open on the <I>StatementHandle</I> but <B>SQLFetch</B> or <B>SQLFetchScroll</B> had not been called.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40001</TD>
+# <TD width=26%>Serialization failure</TD>
+# <TD width=52%>The transaction was rolled back due to a resource deadlock with another transaction.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40003</TD>
+# <TD width=26%>Statement completion unknown</TD>
+# <TD width=52%>The associated connection failed during the execution of this function, and the state of the transaction cannot be determined.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY000</TD>
+# <TD width=26%>General error</TD>
+# <TD width=52%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY001</TD>
+# <TD width=26%>Memory allocation error</TD>
+# <TD width=52%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY008</TD>
+# <TD width=26%>Operation canceled</TD>
+# <TD width=52%>Asynchronous processing was enabled for the <I>StatementHandle</I>. The function was called, and before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I>. Then the function was called again on the <I>StatementHandle</I>.
+# <P>The function was called, and before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I> from a different thread in a multithread application.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY009</TD>
+# <TD width=26%>Invalid use of null pointer</TD>
+# <TD width=52%>The SQL_ATTR_METADATA_ID statement attribute was set to SQL_TRUE, the <I>CatalogName</I> argument was a null pointer, and the SQL_CATALOG_NAME <I>InfoType</I> returns that catalog names are supported.
+# <P>(DM) The SQL_ATTR_METADATA_ID statement attribute was set to SQL_TRUE, and the <I>SchemaName</I>, <I>TableName</I>, or <I>ColumnName</I> argument was a null pointer.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY010</TD>
+# <TD width=26%>Function sequence error</TD>
+# <TD width=52%>(DM) An asynchronously executing function (not this one) was called for the <I>StatementHandle</I> and was still executing when this function was called.
+# <P>(DM) <B>SQLExecute</B>, <B>SQLExecDirect</B>, <B>SQLBulkOperations</B>, or <B>SQLSetPos</B> was called for the <I>StatementHandle</I> and returned SQL_NEED_DATA. This function was called before data was sent for all data-at-execution parameters or columns.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY013</TD>
+# <TD width=26%>Memory management error</TD>
+# <TD width=52%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY090</TD>
+# <TD width=26%>Invalid string or buffer length</TD>
+# <TD width=52%>(DM) The value of one of the name length arguments was less than 0 but not equal to SQL_NTS.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>&nbsp;</TD>
+# <TD width=26%>&nbsp;</TD>
+# <TD width=52%>The value of one of the name length arguments exceeded the maximum length value for the corresponding catalog or name. The maximum length of each catalog or name may be obtained by calling <B>SQLGetInfo</B> with the <I>InfoType</I> values. (See "Comments.")</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYC00</TD>
+# <TD width=26%>Optional feature not implemented</TD>
+# <TD width=52%>A catalog name was specified, and the driver or data source does not support catalogs.
+# <P>A schema name was specified, and the driver or data source does not support schemas.</P>
+#
+# <P>A string search pattern was specified for the schema name, table name, or column name, and the data source does not support search patterns for one or more of those arguments.</P>
+#
+# <P>The combination of the current settings of the SQL_ATTR_CONCURRENCY and SQL_ATTR_CURSOR_TYPE statement attributes was not supported by the driver or data source. </P>
+#
+# <P>The SQL_ATTR_USE_BOOKMARKS statement attribute was set to SQL_UB_VARIABLE, and the SQL_ATTR_CURSOR_TYPE statement attribute was set to a cursor type for which the driver does not support bookmarks.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT00</TD>
+# <TD width=26%>Timeout expired</TD>
+# <TD width=52%>The query timeout period expired before the data source returned the result set. The timeout period is set through <B>SQLSetStmtAttr</B>, SQL_ATTR_QUERY_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT01</TD>
+# <TD width=26%>Connection timeout expired</TD>
+# <TD width=52%>The connection timeout period expired before the data source responded to the request. The connection timeout period is set through <B>SQLSetConnectAttr</B>, SQL_ATTR_CONNECTION_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>IM001</TD>
+# <TD width=26%>Driver does not support this function</TD>
+# <TD width=52%>(DM) The driver associated with the <I>StatementHandle</I> does not support the function.</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Comments</B></P>
+#
+# <P>This function typically is used before statement execution to retrieve information about columns for a table or tables from the data source's catalog. <B>SQLColumns</B> can be used to retrieve data for all types of items returned by <B>SQLTables</B>. In addition to base tables, this may include (but is not limited to) views, synonyms, system tables, and so on. By contrast, the functions <B>SQLColAttribute</B> and <B>SQLDescribeCol</B> describe the columns in a result set and the function <B>SQLNumResultCols</B> returns the number of columns in a result set. For more information, see "<A HREF="odbcuses_of_catalog_data.htm">Uses of Catalog Data</A>" in Chapter 7: Catalog Functions.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;For more information about the general use, arguments, and returned data of ODBC catalog functions, see <A HREF="odbccatalog_functions.htm">Chapter 7: Catalog Functions</A>.</P>
+#
+# <P><B>SQLColumns</B> returns the results as a standard result set, ordered by TABLE_CAT, TABLE_SCHEM, TABLE_NAME, and ORDINAL_POSITION. </P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;When an application works with an ODBC 2.<I>x</I> driver, no ORDINAL_POSITION column is returned in the result set. As a result, when working with ODBC 2.<I>x</I> drivers, the order of the columns in the column list returned by <B>SQLColumns</B> is not necessarily the same as the order of the columns returned when the application performs a SELECT statement on all columns in that table.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;<B>SQLColumns</B> might not return all columns. For example, a driver might not return information about pseudo-columns, such as Oracle ROWID. Applications can use any valid column, whether or not it is returned by <B>SQLColumns</B>. </P>
+#
+# <P class="indent">Some columns that can be returned by <B>SQLStatistics</B> are not returned by <B>SQLColumns</B>. For example, <B>SQLColumns</B> does not return the columns in an index created over an expression or filter, such as SALARY + BENEFITS or DEPT = 0012.</P>
+#
+# <P>The lengths of VARCHAR columns are not shown in the table; the actual lengths depend on the data source. To determine the actual lengths of the TABLE_CAT, TABLE_SCHEM, TABLE_NAME, and COLUMN_NAME columns, an application can call <B>SQLGetInfo</B> with the SQL_MAX_CATALOG_NAME_LEN, SQL_MAX_SCHEMA_NAME_LEN, SQL_MAX_TABLE_NAME_LEN, and SQL_MAX_COLUMN_NAME_LEN options.</P>
+#
+# <P>The following columns have been renamed for ODBC 3.<I>x</I>. The column name changes do not affect backward compatibility because applications bind by column number.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=48%>ODBC 2.0 column</TH>
+# <TH width=52%>ODBC 3.<I>x</I> column</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>TABLE_QUALIFIER</TD>
+# <TD width=52%>TABLE_CAT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>TABLE_OWNER</TD>
+# <TD width=52%>TABLE_SCHEM</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>PRECISION</TD>
+# <TD width=52%>COLUMN_SIZE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>LENGTH</TD>
+# <TD width=52%>BUFFER_LENGTH</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>SCALE</TD>
+# <TD width=52%>DECIMAL_DIGITS</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>RADIX</TD>
+# <TD width=52%>NUM_PREC_RADIX</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P>The following columns have been added to the result set returned by <B>SQLColumns</B> for ODBC 3.<I>x</I>:</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TD width=50%>&nbsp;&nbsp;&nbsp;CHAR_OCTET_LENGTH </TD>
+# <TD width=50%>ORDINAL_POSITION</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>&nbsp;&nbsp;&nbsp;COLUMN_DEF</TD>
+# <TD width=50%>SQL_DATA_TYPE</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>&nbsp;&nbsp;&nbsp;IS_NULLABLE </TD>
+# <TD width=50%>SQL_DATETIME_SUB</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P>The following table lists the columns in the result set. Additional columns beyond column 18 (IS_NULLABLE) can be defined by the driver. An application should gain access to driver-specific columns by counting down from the end of the result set rather than specifying an explicit ordinal position. For more information, see "<A HREF="odbcdata_returned_by_catalog_functions.htm">Data Returned by Catalog Functions</A>" in Chapter 7: Catalog Functions.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=24%><BR>
+# Column name</TH>
+# <TH width=17%>Column<BR>
+# number</TH>
+# <TH width=16%><BR>
+# Data type</TH>
+# <TH width=43%><BR>
+# Comments</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=24%>TABLE_CAT<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>1</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=43%>Catalog name; NULL if not applicable to the data source. If a driver supports catalogs for some tables but not for others, such as when the driver retrieves data from different DBMSs, it returns an empty string ("") for those tables that do not have catalogs.</TD>
+# </TR>
+ { name => "table_cat",
+ type => "varchar",
+ length => 16,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>TABLE_SCHEM<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>2</TD>
+# <TD width=16%>Varchar </TD>
+# <TD width=43%>Schema name; NULL if not applicable to the data source. If a driver supports schemas for some tables but not for others, such as when the driver retrieves data from different DBMSs, it returns an empty string ("") for those tables that do not have schemas.</TD>
+# </TR>
+ { name => "table_schem",
+ type => "varchar",
+ length => 16,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>TABLE_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>3</TD>
+# <TD width=16%>Varchar not NULL</TD>
+# <TD width=43%>Table name.</TD>
+# </TR>
+ { name => "table_name",
+ type => "varchar",
+ length => 16,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>COLUMN_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>4</TD>
+# <TD width=16%>Varchar not NULL</TD>
+# <TD width=43%>Column name. The driver returns an empty string for a column that does not have a name.</TD>
+# </TR>
+ { name => "column_name",
+ type => "varchar",
+ length => 16,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>DATA_TYPE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>5</TD>
+# <TD width=16%>Smallint not NULL</TD>
+# <TD width=43%>SQL data type. This can be an ODBC SQL data type or a driver-specific SQL data type. For datetime and interval data types, this column returns the concise data type (such as SQL_TYPE_DATE or SQL_INTERVAL_YEAR_TO_MONTH, rather than the nonconcise data type such as SQL_DATETIME or SQL_INTERVAL). For a list of valid ODBC SQL data types, see "<A HREF="odbcsql_data_types.htm">SQL Data Types</A>" in Appendix D: Data Types. For information about driver-specific SQL data types, see the driver's documentation.
+# <P>The data types returned for ODBC 3.<I>x</I> and ODBC 2.<I>x</I> applications may be different. For more information, see "<A HREF="odbcbackward_compatibility_and_standards_compliance.htm">Backward Compatibility and Standards Compliance</A>" in Chapter 17: Programming Considerations.</P>
+# </TD>
+# </TR>
+ { name => "data_type",
+ type => "smallint",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>TYPE_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>6</TD>
+# <TD width=16%>Varchar not NULL</TD>
+# <TD width=43%>Data source&#0150;dependent data type name; for example, "CHAR", "VARCHAR", "MONEY", "LONG VARBINAR", or "CHAR ( ) FOR BIT DATA".</TD>
+# </TR>
+ { name => "type_name",
+ type => "varchar",
+ length => 20,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>COLUMN_SIZE<BR>
+# (ODBC 1.0) </TD>
+# <TD width=17%>7</TD>
+# <TD width=16%>Integer</TD>
+# <TD width=43%>If DATA_TYPE is SQL_CHAR or SQL_VARCHAR, this column contains the maximum length in characters of the column. For datetime data types, this is the total number of characters required to display the value when converted to characters. For numeric data types, this is either the total number of digits or the total number of bits allowed in the column, according to the NUM_PREC_RADIX column. For interval data types, this is the number of characters in the character representation of the interval literal (as defined by the interval leading precision, see "<A HREF="odbcinterval_data_type_length.htm">Interval Data Type Length</A>" in Appendix D: Data Types). For more information, see "<A HREF="odbccolumn_size__decimal_digits__transfer_octet_length__and_display_size.htm">Column Size, Decimal Digits, Transfer Octet Length, and Display Size</A>" in Appendix D: Data Types.</TD>
+# </TR>
+ { name => "column_size",
+ type => "integer",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>BUFFER_LENGTH<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>8</TD>
+# <TD width=16%>Integer</TD>
+# <TD width=43%>The length in bytes of data transferred on an SQLGetData, SQLFetch, or SQLFetchScroll operation if SQL_C_DEFAULT is specified. For numeric data, this size may be different than the size of the data stored on the data source. This value might be different than COLUMN_SIZE column for character data. For more information about length, see "<A HREF="odbccolumn_size__decimal_digits__transfer_octet_length__and_display_size.htm">Column Size, Decimal Digits, Transfer Octet Length, and Display Size</A>" in Appendix D: Data Types.</TD>
+# </TR>
+ { name => "buffer_length",
+ type => "integer",
+ length => 16,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>DECIMAL_DIGITS<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>9</TD>
+# <TD width=16%>Smallint</TD>
+# <TD width=43%>The total number of significant digits to the right of the decimal point. For SQL_TYPE_TIME and SQL_TYPE_TIMESTAMP, this column contains the number of digits in the fractional seconds component. For the other data types, this is the decimal digits of the column on the data source. For interval data types that contain a time component, this column contains the number of digits to the right of the decimal point (fractional seconds). For interval data types that do not contain a time component, this column is 0. For more information about decimal digits, see "<A HREF="odbccolumn_size__decimal_digits__transfer_octet_length__and_display_size.htm">Column Size, Decimal Digits, Transfer Octet Length, and Display Size</A>" in Appendix D: Data Types. NULL is returned for data types where DECIMAL_DIGITS is not applicable.</TD>
+# </TR>
+ { name => "decimal_digits",
+ type => "smallint",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>NUM_PREC_RADIX<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>10</TD>
+# <TD width=16%>Smallint</TD>
+# <TD width=43%>For numeric data types, either 10 or 2. If it is 10, the values in COLUMN_SIZE and DECIMAL_DIGITS give the number of decimal digits allowed for the column. For example, a DECIMAL(12,5) column would return a NUM_PREC_RADIX of 10, a COLUMN_SIZE of 12, and a DECIMAL_DIGITS of 5; a FLOAT column could return a NUM_PREC_RADIX of 10, a COLUMN_SIZE of 15, and a DECIMAL_DIGITS of NULL.
+# <P>If it is 2, the values in COLUMN_SIZE and DECIMAL_DIGITS give the number of bits allowed in the column. For example, a FLOAT column could return a RADIX of 2, a COLUMN_SIZE of 53, and a DECIMAL_DIGITS of NULL.</P>
+#
+# <P>NULL is returned for data types where NUM_PREC_RADIX is not applicable.</P>
+# </TD>
+# </TR>
+ { name => "num_prec_radix",
+ type => "smallint",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>NULLABLE<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>11</TD>
+# <TD width=16%>Smallint not NULL</TD>
+# <TD width=43%>SQL_NO_NULLS if the column could not include NULL values.
+# <P>SQL_NULLABLE if the column accepts NULL values.</P>
+#
+# <P>SQL_NULLABLE_UNKNOWN if it is not known whether the column accepts NULL values.</P>
+#
+# <P>The value returned for this column is different from the value returned for the IS_NULLABLE column. The NULLABLE column indicates with certainty that a column can accept NULLs, but cannot indicate with certainty that a column does not accept NULLs. The IS_NULLABLE column indicates with certainty that a column cannot accept NULLs, but cannot indicate with certainty that a column accepts NULLs.</P>
+# </TD>
+# </TR>
+ { name => "nullable",
+ type => "smallint",
+ length => 16,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>REMARKS<BR>
+# (ODBC 1.0)</TD>
+# <TD width=17%>12</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=43%>A description of the column.</TD>
+# </TR>
+ { name => "remarks",
+ type => "varchar",
+ length => 200,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>COLUMN_DEF<BR>
+# (ODBC 3.0)</TD>
+# <TD width=17%>13</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=43%>The default value of the column. The value in this column should be interpreted as a string if it is enclosed in quotation marks.
+# <P>If NULL was specified as the default value, then this column is the word NULL, not enclosed in quotation marks. If the default value cannot be represented without truncation, then this column contains TRUNCATED, with no enclosing single quotation marks. If no default value was specified, then this column is NULL.</P>
+#
+# <P>The value of COLUMN_DEF can be used in generating a new column definition, except when it contains the value TRUNCATED.</P>
+# </TD>
+# </TR>
+ { name => "column_def",
+ type => "varchar",
+ length => 100,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>SQL_DATA_TYPE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=17%>14</TD>
+# <TD width=16%>Smallint not NULL</TD>
+# <TD width=43%>SQL data type, as it appears in the SQL_DESC_TYPE record field in the IRD. This can be an ODBC SQL data type or a driver-specific SQL data type. This column is the same as the DATA_TYPE column, with the exception of datetime and interval data types. This column returns the nonconcise data type (such as SQL_DATETIME or SQL_INTERVAL), rather than the concise data type (such as SQL_TYPE_DATE or SQL_INTERVAL_YEAR_TO_MONTH) for datetime and interval data types. If this column returns SQL_DATETIME or SQL_INTERVAL, the specific data type can be determined from the SQL_DATETIME_SUB column. For a list of valid ODBC SQL data types, see "<A HREF="odbcsql_data_types.htm">SQL Data Types</A>" in Appendix D: Data Types. For information about driver-specific SQL data types, see the driver's documentation.
+# <P>The data types returned for ODBC 3.<I>x</I> and ODBC 2.<I>x</I> applications may be different. For more information, see "<A HREF="odbcbackward_compatibility_and_standards_compliance.htm">Backward Compatibility and Standards Compliance</A>" in Chapter 17: Programming Considerations.</P>
+# </TD>
+# </TR>
+ { name => "sql_data_type",
+ type => "smallint",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>SQL_DATETIME_SUB<BR>
+# (ODBC 3.0)</TD>
+# <TD width=17%>15</TD>
+# <TD width=16%>Smallint</TD>
+# <TD width=43%>The subtype code for datetime and interval data types. For other data types, this column returns a NULL. For more information about datetime and interval subcodes, see "SQL_DESC_DATETIME_INTERVAL_CODE" in <A HREF="odbcsqlsetdescfield.htm">SQLSetDescField</A>.</TD>
+# </TR>
+ { name => "sql_datetime_sub",
+ type => "smallint",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>CHAR_OCTET_LENGTH<BR>
+# (ODBC 3.0)</TD>
+# <TD width=17%>16</TD>
+# <TD width=16%>Integer</TD>
+# <TD width=43%>The maximum length in bytes of a character or binary data type column. For all other data types, this column returns a NULL.</TD>
+# </TR>
+ { name => "char_octet_length",
+ type => "integer",
+ length => undef,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>ORDINAL_POSITION<BR>
+# (ODBC 3.0)</TD>
+# <TD width=17%>17</TD>
+# <TD width=16%>Integer not NULL</TD>
+# <TD width=43%>The ordinal position of the column in the table. The first column in the table is number 1.</TD>
+# </TR>
+ { name => "ordinal_position",
+ type => "integer",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=24%>IS_NULLABLE<BR>
+# (ODBC 3.0)</TD>
+# <TD width=17%>18</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=43%>"NO" if the column does not include NULLs.
+# <P>"YES" if the column could include NULLs.</P>
+#
+# <P>This column returns a zero-length string if nullability is unknown. </P>
+#
+# <P>ISO rules are followed to determine nullability. An ISO SQL&#0150;compliant DBMS cannot return an empty string. </P>
+#
+# <P>The value returned for this column is different from the value returned for the NULLABLE column. (See the description of the NULLABLE column.)</P>
+# </TD>
+# </TR>
+ { name => "is_nullable",
+ type => "varchar",
+ length => 3,
+ nullable => 1,
+ },
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Code Example</B></P>
+#
+# <P>In the following example, an application declares buffers for the result set returned by <B>SQLColumns</B>. It calls <B>SQLColumns</B> to return a result set that describes each column in the EMPLOYEE table. It then calls <B>SQLBindCol</B> to bind the columns in the result set to the buffers. Finally, the application fetches each row of data with <B>SQLFetch</B> and processes it.</P>
+#
+# <PRE class="code">#define STR_LEN 128+1
+# #define REM_LEN 254+1
+#
+# /* Declare buffers for result set data */
+#
+# SQLCHAR&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;szCatalog[STR_LEN], szSchema[STR_LEN];
+# SQLCHAR&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;szTableName[STR_LEN], szColumnName[STR_LEN];
+# SQLCHAR&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;szTypeName[STR_LEN], szRemarks[REM_LEN];
+# SQLCHAR&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;szColumnDefault[STR_LEN], szIsNullable[STR_LEN];
+# SQLINTEGER&nbsp;&nbsp;&nbsp;&nbsp;ColumnSize, BufferLength, CharOctetLength, OrdinalPosition;
+# SQLSMALLINT&nbsp;&nbsp;&nbsp;DataType, DecimalDigits, NumPrecRadix, Nullable;
+# SQLSMALLINT&nbsp;&nbsp;&nbsp;SQLDataType, DatetimeSubtypeCode;
+# SQLRETURN&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;retcode;
+# SQLHSTMT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;hstmt;
+#
+# /* Declare buffers for bytes available to return */
+#
+# SQLINTEGER cbCatalog, cbSchema, cbTableName, cbColumnName;
+# SQLINTEGER cbDataType, cbTypeName, cbColumnSize, cbBufferLength;
+# SQLINTEGER cbDecimalDigits, cbNumPrecRadix, cbNullable, cbRemarks;
+# SQLINTEGER cbColumnDefault, cbSQLDataType, cbDatetimeSubtypeCode, cbCharOctetLength;
+# SQLINTEGER cbOrdinalPosition, cbIsNullable;
+#
+# retcode = SQLColumns(hstmt,
+# NULL, 0,&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;/* All catalogs */
+# NULL, 0,&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;/* All schemas */
+# "CUSTOMERS", SQL_NTS,&nbsp;&nbsp;&nbsp;/* CUSTOMERS table */
+# NULL, 0);&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;/* All columns */
+#
+# if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+#
+# /* Bind columns in result set to buffers */
+#
+# SQLBindCol(hstmt, 1, SQL_C_CHAR, szCatalog, STR_LEN,&amp;cbCatalog);
+# SQLBindCol(hstmt, 2, SQL_C_CHAR, szSchema, STR_LEN, &amp;cbSchema);
+# SQLBindCol(hstmt, 3, SQL_C_CHAR, szTableName, STR_LEN,&amp;cbTableName);
+# SQLBindCol(hstmt, 4, SQL_C_CHAR, szColumnName, STR_LEN, &amp;cbColumnName);
+# SQLBindCol(hstmt, 5, SQL_C_SSHORT, &amp;DataType, 0, &amp;cbDataType);
+# SQLBindCol(hstmt, 6, SQL_C_CHAR, szTypeName, STR_LEN, &amp;cbTypeName);
+# SQLBindCol(hstmt, 7, SQL_C_SLONG, &amp;ColumnSize, 0, &amp;cbColumnSize);
+# SQLBindCol(hstmt, 8, SQL_C_SLONG, &amp;BufferLength, 0, &amp;cbBufferLength);
+# SQLBindCol(hstmt, 9, SQL_C_SSHORT, &amp;DecimalDigits, 0, &amp;cbDecimalDigits);
+# SQLBindCol(hstmt, 10, SQL_C_SSHORT, &amp;NumPrecRadix, 0, &amp;cbNumPrecRadix);
+# SQLBindCol(hstmt, 11, SQL_C_SSHORT, &amp;Nullable, 0, &amp;cbNullable);
+# SQLBindCol(hstmt, 12, SQL_C_CHAR, szRemarks, REM_LEN, &amp;cbRemarks);
+# SQLBindCol(hstmt, 13, SQL_C_CHAR, szColumnDefault, STR_LEN, &amp;cbColumnDefault);
+# SQLBindCol(hstmt, 14, SQL_C_SSHORT, &amp;SQLDataType, 0, &amp;cbSQLDataType);
+# SQLBindCol(hstmt, 15, SQL_C_SSHORT, &amp;DatetimeSubtypeCode, 0,
+# &amp;cbDatetimeSubtypeCode);
+# SQLBindCol(hstmt, 16, SQL_C_SLONG, &amp;CharOctetLength, 0, &amp;cbCharOctetLength);
+# SQLBindCol(hstmt, 17, SQL_C_SLONG, &amp;OrdinalPosition, 0, &amp;cbOrdinalPosition);
+# SQLBindCol(hstmt, 18, SQL_C_CHAR, szIsNullable, STR_LEN, &amp;cbIsNullable);
+# while(TRUE) {
+# retcode = SQLFetch(hstmt);
+# if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+# show_error( );
+# }
+# if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO){
+# ; /* Process fetched data */
+# } else {
+# break;
+# }
+# }
+# }</PRE>
+#
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=50%>For information about</TH>
+# <TH width=50%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Binding a buffer to a column in a result set</TD>
+# <TD width=50%><A HREF="odbcsqlbindcol.htm">SQLBindCol</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Canceling statement processing</TD>
+# <TD width=50%><A HREF="odbcsqlcancel.htm">SQLCancel</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning privileges for a column or columns</TD>
+# <TD width=50%><A HREF="odbcsqlcolumnprivileges.htm">SQLColumnPrivileges</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Fetching a block of data or scrolling through a result set</TD>
+# <TD width=50%><A HREF="odbcsqlfetchscroll.htm">SQLFetchScroll</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Fetching multiple rows of data</TD>
+# <TD width=50%><A HREF="odbcsqlfetch.htm">SQLFetch</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning columns that uniquely identify a row, or columns automatically updated by a transaction</TD>
+# <TD width=50%><A HREF="odbcsqlspecialcolumns.htm">SQLSpecialColumns</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning table statistics and indexes</TD>
+# <TD width=50%><A HREF="odbcsqlstatistics.htm">SQLStatistics</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning a list of tables in a data source</TD>
+# <TD width=50%><A HREF="odbcsqltables.htm">SQLTables</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning privileges for a table or tables</TD>
+# <TD width=50%><A HREF="odbcsqltableprivileges.htm">SQLTablePrivileges</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+];
+
+#
+# odbcsqlprimarykeys.htm
+#
+$listWhat->{primarykeys} = [
+# <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 3.2 Final//EN">
+# <HTML DIR="LTR"><HEAD>
+# <META HTTP-EQUIV="Content-Type" Content="text/html; charset=Windows-1252">
+# <TITLE>SQLPrimaryKeys</TITLE>
+# <SCRIPT SRC="/stylesheets/vs70link.js"></SCRIPT>
+# <SCRIPT SRC="/stylesheets/vs70.js"></SCRIPT>
+# <SCRIPT LANGUAGE="JScript" SRC="/stylesheets/odbc.js"></SCRIPT>
+# </HEAD>
+# <body topmargin=0 id="bodyID">
+#
+# <div id="nsbanner">
+# <div id="bannertitle">
+# <TABLE CLASS="bannerparthead" CELLSPACING=0>
+# <TR ID="hdr">
+# <TD CLASS="bannertitle" nowrap>
+# ODBC Programmer's Reference
+# </TD><TD valign=middle><a href="#Feedback"><IMG name="feedb" onclick=EMailStream(SDKFeedB) style="CURSOR: hand;" hspace=15 alt="" src="/stylesheets/mailto.gif" align=right></a></TD>
+# </TR>
+# </TABLE>
+# </div>
+# </div>
+# <DIV id="nstext" valign="bottom">
+#
+# <H1><A NAME="odbcsqlprimarykeys"></A>SQLPrimaryKeys</H1>
+#
+# <P class="label"><B>Conformance</B></P>
+#
+# <P>Version Introduced: ODBC 1.0<BR>
+# Standards Compliance: ODBC</P>
+#
+# <P class="label"><B>Summary</B></P>
+#
+# <P><B>SQLPrimaryKeys</B> returns the column names that make up the primary key for a table. The driver returns the information as a result set. This function does not support returning primary keys from multiple tables in a single call.</P>
+#
+# <P class="label"><B>Syntax</B></P>
+#
+# <PRE class="syntax">SQLRETURN <B>SQLPrimaryKeys</B>(
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLHSTMT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>StatementHandle</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>CatalogName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength1</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>SchemaName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength2</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLCHAR *&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>TableName</I>,
+# &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SQLSMALLINT&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<I>NameLength3</I>);</PRE>
+#
+# <P class="label"><B>Arguments</B>
+#
+# <DL>
+# <DT><I>StatementHandle</I></DT>
+#
+# <DD>[Input]<BR>
+# Statement handle.</dd>
+#
+# <DT><I>CatalogName</I></DT>
+#
+# <DD>[Input]<BR>
+# Catalog name. If a driver supports catalogs for some tables but not for others, such as when the driver retrieves data from different DBMSs, an empty string ("") denotes those tables that do not have catalogs. <I>CatalogName </I>cannot contain a string search pattern.
+#
+# <P>If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>CatalogName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>CatalogName</I> is an ordinary argument; it is treated literally, and its case is significant. For more information, see "<A HREF="odbcarguments_in_catalog_functions.htm">Arguments in Catalog Functions</A>" in Chapter 7: Catalog Functions.
+# </dd>
+#
+# <DT><I>NameLength1</I></DT>
+#
+# <DD>[Input]<BR>
+# Length in bytes of *<I>CatalogName</I>.</dd>
+#
+# <DT><I>SchemaName</I></DT>
+#
+# <DD>[Input]<BR>
+# Schema name. If a driver supports schemas for some tables but not for others, such as when the driver retrieves data from different DBMSs, an empty string ("") denotes those tables that do not have schemas. <I>SchemaName </I>cannot contain a string search pattern.
+#
+# <P>If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>SchemaName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>SchemaName</I> is an ordinary argument; it is treated literally, and its case is not significant.
+# </dd>
+#
+# <DT><I>NameLength2</I></DT>
+#
+# <DD>[Input]<BR>
+# Length in bytes of *<I>SchemaName</I>.</dd>
+#
+# <DT><I>TableName</I></DT>
+#
+# <DD>[Input]<BR>
+# Table name. This argument cannot be a null pointer. <I>TableName </I>cannot contain a string search pattern.
+#
+# <P>If the SQL_ATTR_METADATA_ID statement attribute is set to SQL_TRUE, <I>TableName</I> is treated as an identifier and its case is not significant. If it is SQL_FALSE, <I>TableName</I> is an ordinary argument; it is treated literally, and its case is not significant.
+# </dd>
+#
+# <DT><I>NameLength3</I></DT>
+#
+# <DD>[Input]<BR>
+# Length in bytes of *<I>TableName</I>.</dd>
+# </DL>
+#
+# <P class="label"><B>Returns</B></P>
+#
+# <P>SQL_SUCCESS, SQL_SUCCESS_WITH_INFO, SQL_STILL_EXECUTING, SQL_ERROR, or SQL_INVALID_HANDLE.</P>
+#
+# <P class="label"><B>Diagnostics</B></P>
+#
+# <P>When <B>SQLPrimaryKeys</B> returns SQL_ERROR or SQL_SUCCESS_WITH_INFO, an associated SQLSTATE value can be obtained by calling <B>SQLGetDiagRec</B> with a <I>HandleType</I> of SQL_HANDLE_STMT and a <I>Handle</I> of <I>StatementHandle</I>. The following table lists the SQLSTATE values commonly returned by <B>SQLPrimaryKeys</B> and explains each one in the context of this function; the notation "(DM)" precedes the descriptions of SQLSTATEs returned by the Driver Manager. The return code associated with each SQLSTATE value is SQL_ERROR, unless noted otherwise.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=22%>SQLSTATE</TH>
+# <TH width=26%>Error</TH>
+# <TH width=52%>Description</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>01000</TD>
+# <TD width=26%>General warning</TD>
+# <TD width=52%>Driver-specific informational message. (Function returns SQL_SUCCESS_WITH_INFO.)</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>08S01</TD>
+# <TD width=26%>Communication link failure</TD>
+# <TD width=52%>The communication link between the driver and the data source to which the driver was connected failed before the function completed processing.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>24000</TD>
+# <TD width=26%>Invalid cursor state</TD>
+# <TD width=52%>(DM) A cursor was open on the <I>StatementHandle</I>, and <B>SQLFetch</B> or <B>SQLFetchScroll</B> had been called.
+# <P>A cursor was open on the <I>StatementHandle</I>, but <B>SQLFetch</B> or <B>SQLFetchScroll</B> had not been called.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40001</TD>
+# <TD width=26%>Serialization failure</TD>
+# <TD width=52%>The transaction was rolled back due to a resource deadlock with another transaction.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>40003</TD>
+# <TD width=26%>Statement completion unknown</TD>
+# <TD width=52%>The associated connection failed during the execution of this function, and the state of the transaction cannot be determined.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY000</TD>
+# <TD width=26%>General error</TD>
+# <TD width=52%>An error occurred for which there was no specific SQLSTATE and for which no implementation-specific SQLSTATE was defined. The error message returned by <B>SQLGetDiagRec</B> in the <I>*MessageText</I> buffer describes the error and its cause.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY001</TD>
+# <TD width=26%>Memory allocation error</TD>
+# <TD width=52%>The driver was unable to allocate memory required to support execution or completion of the function.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY008</TD>
+# <TD width=26%>Operation canceled</TD>
+# <TD width=52%>Asynchronous processing was enabled for the <I>StatementHandle</I>. The function was called, and before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I>. Then the function was called again on the <I>StatementHandle</I>.
+# <P>The function was called, and before it completed execution, <B>SQLCancel</B> was called on the <I>StatementHandle</I> from a different thread in a multithread application.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY009</TD>
+# <TD width=26%>Invalid use of null pointer</TD>
+# <TD width=52%>(DM) The <I>TableName</I> argument was a null pointer.
+# <P>The SQL_ATTR_METADATA_ID statement attribute was set to SQL_TRUE, the <I>CatalogName</I> argument was a null pointer, and <B>SQLGetInfo</B> with the SQL_CATALOG_NAME information type returns that catalog names are supported.</P>
+#
+# <P>(DM) The SQL_ATTR_METADATA_ID statement attribute was set to SQL_TRUE, and the <I>SchemaName</I> argument was a null pointer.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY010</TD>
+# <TD width=26%>Function sequence error</TD>
+# <TD width=52%>(DM) An asynchronously executing function (not this one) was called for the <I>StatementHandle</I> and was still executing when this function was called.
+# <P>(DM) <B>SQLExecute</B>, <B>SQLExecDirect</B>, <B>SQLBulkOperations</B>, or <B>SQLSetPos</B> was called for the <I>StatementHandle</I> and returned SQL_NEED_DATA. This function was called before data was sent for all data-at-execution parameters or columns.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY013</TD>
+# <TD width=26%>Memory management error</TD>
+# <TD width=52%>The function call could not be processed because the underlying memory objects could not be accessed, possibly because of low memory conditions.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HY090</TD>
+# <TD width=26%>Invalid string or buffer length</TD>
+# <TD width=52%>(DM) The value of one of the name length arguments was less than 0 but not equal to SQL_NTS, and the associated name argument is not a null pointer.
+# <P>The value of one of the name length arguments exceeded the maximum length value for the corresponding name.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYC00</TD>
+# <TD width=26%>Optional feature not implemented</TD>
+# <TD width=52%>A catalog was specified, and the driver or data source does not support catalogs.
+# <P>A schema was specified and the driver or data source does not support schemas.</P>
+#
+# <P>The combination of the current settings of the SQL_ATTR_CONCURRENCY and SQL_ATTR_CURSOR_TYPE statement attributes was not supported by the driver or data source.</P>
+#
+# <P>The SQL_ATTR_USE_BOOKMARKS statement attribute was set to SQL_UB_VARIABLE, and the SQL_ATTR_CURSOR_TYPE statement attribute was set to a cursor type for which the driver does not support bookmarks.</P>
+# </TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT00</TD>
+# <TD width=26%>Timeout expired</TD>
+# <TD width=52%>The timeout period expired before the data source returned the requested result set. The timeout period is set through <B>SQLSetStmtAttr</B>, SQL_ATTR_QUERY_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>HYT01</TD>
+# <TD width=26%>Connection timeout expired</TD>
+# <TD width=52%>The connection timeout period expired before the data source responded to the request. The connection timeout period is set through <B>SQLSetConnectAttr</B>, SQL_ATTR_CONNECTION_TIMEOUT.</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=22%>IM001</TD>
+# <TD width=26%>Driver does not support this function</TD>
+# <TD width=52%>(DM) The driver associated with the <I>StatementHandle</I> does not support the function.</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Comments</B></P>
+#
+# <P><B>SQLPrimaryKeys</B> returns the results as a standard result set, ordered by TABLE_CAT, TABLE_SCHEM, TABLE_NAME, and KEY_SEQ. For information about how this information might be used, see "<A HREF="odbcuses_of_catalog_data.htm">Uses of Catalog Data</A>" in Chapter 7: Catalog Functions.</P>
+#
+# <P>The following columns have been renamed for ODBC 3.<I>x</I>. The column name changes do not affect backward compatibility because applications bind by column number.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=48%>ODBC 2.0 column</TH>
+# <TH width=52%>ODBC 3.<I>x</I> column</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>TABLE_QUALIFIER</TD>
+# <TD width=52%>TABLE_CAT</TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=48%>TABLE_OWNER</TD>
+# <TD width=52%>TABLE_SCHEM</TD>
+# </TR>
+# </table></div>
+# <!--TS:-->
+# <P>To determine the actual lengths of the TABLE_CAT, TABLE_SCHEM, TABLE_NAME, and COLUMN_NAME columns, call <B>SQLGetInfo</B> with the SQL_MAX_CATALOG_NAME_LEN, SQL_MAX_SCHEMA_NAME_LEN, SQL_MAX_TABLE_NAME_LEN, and SQL_MAX_COLUMN_NAME_LEN options.</P>
+#
+# <P class="indent"><b class="le">Note</b>&nbsp;&nbsp;&nbsp;For more information about the general use, arguments, and returned data of ODBC catalog functions, see <A HREF="odbccatalog_functions.htm">Chapter 7: Catalog Functions</A>.</P>
+#
+# <P>The following table lists the columns in the result set. Additional columns beyond column 6 (PK_NAME) can be defined by the driver. An application should gain access to driver-specific columns by counting down from the end of the result set rather than specifying an explicit ordinal position. For more information, see "<A HREF="odbcdata_returned_by_catalog_functions.htm">Data Returned by Catalog Functions</A>" in Chapter 7: Catalog Functions.</P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=27%><BR>
+# Column name</TH>
+# <TH width=15%>Column number</TH>
+# <TH width=16%><BR>
+# Data type</TH>
+# <TH width=42%><BR>
+# Comments</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=27%>TABLE_CAT<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>1</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=42%>Primary key table catalog name; NULL if not applicable to the data source. If a driver supports catalogs for some tables but not for others, such as when the driver retrieves data from different DBMSs, it returns an empty string ("") for those tables that do not have catalogs.</TD>
+# </TR>
+ { name => "table_cat",
+ type => "varchar",
+ length => 16,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=27%>TABLE_SCHEM<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>2</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=42%>Primary key table schema name; NULL if not applicable to the data source. If a driver supports schemas for some tables but not for others, such as when the driver retrieves data from different DBMSs, it returns an empty string ("") for those tables that do not have schemas.</TD>
+# </TR>
+ { name => "table_schem",
+ type => "varchar",
+ length => 16,
+ nullable => 1,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=27%>TABLE_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>3</TD>
+# <TD width=16%>Varchar<BR>
+# not NULL</TD>
+# <TD width=42%>Primary key table name.</TD>
+# </TR>
+ { name => "table_name",
+ type => "varchar",
+ length => 16,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=27%>COLUMN_NAME<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>4</TD>
+# <TD width=16%>Varchar<BR>
+# not NULL</TD>
+# <TD width=42%>Primary key column name. The driver returns an empty string for a column that does not have a name.</TD>
+# </TR>
+ { name => "column_name",
+ type => "varchar",
+ length => 16,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=27%>KEY_SEQ<BR>
+# (ODBC 1.0)</TD>
+# <TD width=15%>5</TD>
+# <TD width=16%>Smallint<BR>
+# not NULL</TD>
+# <TD width=42%>Column sequence number in key (starting with 1).</TD>
+# </TR>
+ { name => "key_seq",
+ type => "smallint",
+ length => undef,
+ nullable => 0,
+ },
+#
+# <TR VALIGN="top">
+# <TD width=27%>PK_NAME<BR>
+# (ODBC 2.0)</TD>
+# <TD width=15%>6</TD>
+# <TD width=16%>Varchar</TD>
+# <TD width=42%>Primary key name. NULL if not applicable to the data source.</TD>
+# </TR>
+ { name => "pk_name",
+ type => "varchar",
+ length => 16,
+ nullable => 1,
+ },
+# </table></div>
+# <!--TS:-->
+# <P class="label"><B>Code Example</B></P>
+#
+# <P>See <A HREF="odbcsqlforeignkeys.htm">SQLForeignKeys</A>.</P>
+#
+# <P class="label"><B>Related Functions</B></P>
+# <!--TS:--><div class="tablediv"><table>
+#
+# <TR VALIGN="top">
+# <TH width=50%>For information about</TH>
+# <TH width=50%>See</TH>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Binding a buffer to a column in a result set</TD>
+# <TD width=50%><A HREF="odbcsqlbindcol.htm">SQLBindCol</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Canceling statement processing</TD>
+# <TD width=50%><A HREF="odbcsqlcancel.htm">SQLCancel</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Fetching a block of data or scrolling through a result set</TD>
+# <TD width=50%><A HREF="odbcsqlfetchscroll.htm">SQLFetchScroll</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Fetching a single row or a block of data in a forward-only direction</TD>
+# <TD width=50%><A HREF="odbcsqlfetch.htm">SQLFetch</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning the columns of foreign keys</TD>
+# <TD width=50%><A HREF="odbcsqlforeignkeys.htm">SQLForeignKeys</A></TD>
+# </TR>
+#
+# <TR VALIGN="top">
+# <TD width=50%>Returning table statistics and indexes</TD>
+# <TD width=50%><A HREF="odbcsqlstatistics.htm">SQLStatistics</A></TD>
+# </TR>
+# </table></div>
+# <!--TS:--><H4><A NAME="feedback"></A></H4>
+# <SPAN id="SDKFeedB"></SPAN>
+# </div>
+#
+# </BODY>
+# </HTML>
+];
+
+my $list = $listWhat->{$what} or die "$what?";
+my $i4 = " " x 4;
+if ($opt eq '-l') {
+ print join(", ", map($_->{name}, @$list)), "\n";
+ exit;
+}
+if ($opt eq '-c') {
+ my $pos = 0;
+ for my $p (@$list) {
+ print "${i4}ConnSys::Column::Column(\n";
+ $pos++;
+ print "\t$pos,\n";
+ print "\t\"" . uc($p->{name}) . "\",\n";
+ print "\tfalse,\n";
+ print "\tNdbType(NdbType::";
+ if ($p->{type} eq 'varchar') {
+ print "String, 8, $p->{length}";
+ } else {
+ print "Signed, 32, 1";
+ }
+ print ", " . ($p->{nullable} ? "true" : "false");
+ print ")\n";
+ print "${i4}),\n";
+ }
+ exit;
+}
+print "$opt?\n";
+
+# vim: set sw=4:
diff --git a/ndb/src/client/odbc/docs/type.txt b/ndb/src/client/odbc/docs/type.txt
new file mode 100644
index 00000000000..d7b391afc55
--- /dev/null
+++ b/ndb/src/client/odbc/docs/type.txt
@@ -0,0 +1,333 @@
+ODBC Programmer's Reference
+****** SQL Data Types ******
+Each DBMS defines its own SQL types. Each ODBC driver exposes only those SQL
+data types that the associated DBMS defines. How a driver maps DBMS SQL types
+to the ODBC-defined SQL type identifiers and how a driver maps DBMS SQL types
+to its own driver-specific SQL type identifiers are returned through a call to
+SQLGetTypeInfo. A driver also returns the SQL data types when describing the
+data types of columns and parameters through calls to SQLColAttribute,
+SQLColumns, SQLDescribeCol, SQLDescribeParam, SQLProcedureColumns, and
+SQLSpecialColumns.
+Note The SQL data types are contained in the SQL_DESC_ CONCISE_TYPE,
+SQL_DESC_TYPE, and SQL_DESC_DATETIME_INTERVAL_CODE fields of the implementation
+descriptors. Characteristics of the SQL data types are contained in the
+SQL_DESC_PRECISION, SQL_DESC_SCALE, SQL_DESC_LENGTH, and SQL_DESC_OCTET_LENGTH
+fields of the implementation descriptors. For more information, see "Data_Type
+Identifiers_and_Descriptors" later in this appendix.
+A given driver and data source do not necessarily support all of the SQL data
+types defined in this appendix. A driver's support for SQL data types depends
+on the level of SQL-92 that the driver conforms to. To determine the level of
+SQL-92 grammar supported by the driver, an application calls SQLGetInfo with
+the SQL_SQL_CONFORMANCE information type. Furthermore, a given driver and data
+source may support additional, driver-specific SQL data types. To determine
+which data types a driver supports, an application calls SQLGetTypeInfo. For
+information about driver-specific SQL data types, see the driver's
+documentation. For information about the data types in a specific data source,
+see the documentation for that data source.
+Important The tables throughout this appendix are only guidelines and show
+commonly used names, ranges, and limits of SQL data types. A given data source
+might support only some of the listed data types, and the characteristics of
+the supported data types can differ from those listed.
+The following table lists valid SQL type identifiers for all SQL data types.
+The table also lists the name and description of the corresponding data type
+from SQL-92 (if one exists).
+SQL type identifier[1] Typical SQL data Typical type description
+ type[2]
+SQL_CHAR CHAR(n) Character string of fixed
+ string length n.
+SQL_VARCHAR VARCHAR(n) Variable-length character
+ string with a maximum
+ string length n.
+SQL_LONGVARCHAR LONG VARCHAR Variable length character
+ data. Maximum length is
+ data source–dependent.[9]
+SQL_WCHAR WCHAR(n) Unicode character string
+ of fixed string length n
+SQL_WVARCHAR VARWCHAR(n) Unicode variable-length
+ character string with a
+ maximum string length n
+SQL_WLONGVARCHAR LONGWVARCHAR Unicode variable-length
+ character data. Maximum
+ length is data
+ source–dependent
+SQL_DECIMAL DECIMAL(p,s) Signed, exact, numeric
+ value with a precision of
+ at least p and scale s.
+ (The maximum precision is
+ driver-defined.)
+ (1 <= p <= 15; s <= p).
+ [4]
+SQL_NUMERIC NUMERIC(p,s) Signed, exact, numeric
+ value with a precision p
+ and scale s
+ (1 <= p <= 15; s <= p).
+ [4]
+SQL_SMALLINT SMALLINT Exact numeric value with
+ precision 5 and scale 0
+ (signed:
+ –32,768 <= n <= 32,767,
+ unsigned:
+ 0 <= n <= 65,535)[3].
+SQL_INTEGER INTEGER Exact numeric value with
+ precision 10 and scale 0
+ (signed:
+ –2[31] <= n <= 2[31] – 1,
+ unsigned:
+ 0 <= n <= 2[32] – 1)[3].
+SQL_REAL REAL Signed, approximate,
+ numeric value with a
+ binary precision 24 (zero
+ or absolute value 10[–38]
+ to 10[38]).
+SQL_FLOAT FLOAT(p) Signed, approximate,
+ numeric value with a
+ binary precision of at
+ least p. (The maximum
+ precision is driver-
+ defined.)[5]
+SQL_DOUBLE DOUBLE PRECISION Signed, approximate,
+ numeric value with a
+ binary precision 53 (zero
+ or absolute value 10
+ [–308] to 10[308]).
+SQL_BIT BIT Single bit binary data.
+ [8]
+SQL_TINYINT TINYINT Exact numeric value with
+ precision 3 and scale 0
+ (signed:
+ –128 <= n <= 127,
+ unsigned:
+ 0 <= n <= 255)[3].
+SQL_BIGINT BIGINT Exact numeric value with
+ precision 19 (if signed)
+ or 20 (if unsigned) and
+ scale 0
+ (signed:
+ –2[63] <= n <= 2[63] – 1,
+
+ unsigned:
+ 0 <= n <= 2[64] – 1)[3],
+ [9].
+SQL_BINARY BINARY(n) Binary data of fixed
+ length n.[9]
+SQL_VARBINARY VARBINARY(n) Variable length binary
+ data of maximum length n.
+ The maximum is set by the
+ user.[9]
+SQL_LONGVARBINARY LONG VARBINARY Variable length binary
+ data. Maximum length is
+ data source–dependent.[9]
+SQL_TYPE_DATE[6] DATE Year, month, and day
+ fields, conforming to the
+ rules of the Gregorian
+ calendar. (See
+ "Constraints_of_the
+ Gregorian_Calendar,"
+ later in this appendix.)
+SQL_TYPE_TIME[6] TIME(p) Hour, minute, and second
+ fields, with valid values
+ for hours of 00 to 23,
+ valid values for minutes
+ of 00 to 59, and valid
+ values for seconds of 00
+ to 61. Precision p
+ indicates the seconds
+ precision.
+SQL_TYPE_TIMESTAMP[6] TIMESTAMP(p) Year, month, day, hour,
+ minute, and second
+ fields, with valid values
+ as defined for the DATE
+ and TIME data types.
+SQL_INTERVAL_MONTH[7] INTERVAL MONTH(p) Number of months between
+ two dates; p is the
+ interval leading
+ precision.
+SQL_INTERVAL_YEAR[7] INTERVAL YEAR(p) Number of years between
+ two dates; p is the
+ interval leading
+ precision.
+SQL_INTERVAL_YEAR_TO_MONTH[7] INTERVAL YEAR(p) TO Number of years and
+ MONTH months between two dates;
+ p is the interval leading
+ precision.
+SQL_INTERVAL_DAY[7] INTERVAL DAY(p) Number of days between
+ two dates; p is the
+ interval leading
+ precision.
+SQL_INTERVAL_HOUR[7] INTERVAL HOUR(p) Number of hours between
+ two date/times; p is the
+ interval leading
+ precision.
+SQL_INTERVAL_MINUTE[7] INTERVAL MINUTE(p) Number of minutes between
+ two date/times; p is the
+ interval leading
+ precision.
+SQL_INTERVAL_SECOND[7] INTERVAL SECOND(p,q) Number of seconds between
+ two date/times; p is the
+ interval leading
+ precision and q is the
+ interval seconds
+ precision.
+SQL_INTERVAL_DAY_TO_HOUR[7] INTERVAL DAY(p) TO HOUR Number of days/hours
+ between two date/times; p
+ is the interval leading
+ precision.
+SQL_INTERVAL_DAY_TO_MINUTE[7] INTERVAL DAY(p) TO Number of days/hours/
+ MINUTE minutes between two date/
+ times; p is the interval
+ leading precision.
+SQL_INTERVAL_DAY_TO_SECOND[7] INTERVAL DAY(p) TO Number of days/hours/
+ SECOND(q) minutes/seconds between
+ two date/times; p is the
+ interval leading
+ precision and q is the
+ interval seconds
+ precision.
+SQL_INTERVAL_HOUR_TO_MINUTE INTERVAL HOUR(p) TO Number of hours/minutes
+[7] MINUTE between two date/times; p
+ is the interval leading
+ precision.
+SQL_INTERVAL_HOUR_TO_SECOND INTERVAL HOUR(p) TO Number of hours/minutes/
+[7] SECOND(q) seconds between two date/
+ times; p is the interval
+ leading precision and q
+ is the interval seconds
+ precision.
+SQL_INTERVAL_MINUTE_TO_SECOND INTERVAL MINUTE(p) TO Number of minutes/seconds
+[7] SECOND(q) between two date/times; p
+ is the interval leading
+ precision and q is the
+ interval seconds
+ precision.
+SQL_GUID GUID Fixed length Globally
+ Unique Identifier.
+[1] This is the value returned in the DATA_TYPE column by a call to
+SQLGetTypeInfo.
+[2] This is the value returned in the NAME and CREATE PARAMS column by a call
+to SQLGetTypeInfo. The NAME column returns the designation—for example,
+CHAR—while the CREATE PARAMS column returns a comma-separated list of creation
+parameters such as precision, scale, and length.
+[3] An application uses SQLGetTypeInfo or SQLColAttribute to determine if a
+particular data type or a particular column in a result set is unsigned.
+[4] SQL_DECIMAL and SQL_NUMERIC data types differ only in their precision.
+The precision of a DECIMAL(p,s) is an implementation-defined decimal precision
+that is no less than p, while the precision of a NUMERIC(p,s) is exactly equal
+to p.
+[5] Depending on the implementation, the precision of SQL_FLOAT can be either
+24 or 53: if it is 24, the SQL_FLOAT data type is the same as SQL_REAL; if it
+is 53, the SQL_FLOAT data type is the same as SQL_DOUBLE.
+[6] In ODBC 3.x, the SQL date, time, and timestamp data types are
+SQL_TYPE_DATE, SQL_TYPE_TIME, and SQL_TYPE_TIMESTAMP, respectively; in ODBC
+2.x, the data types are SQL_DATE, SQL_TIME, and SQL_TIMESTAMP.
+[7] For more information on the interval SQL data types, see the "Interval
+Data_Types" section, later in this appendix.
+[8] The SQL_BIT data type has different characteristics than the BIT type in
+SQL-92.
+[9] This data type has no corresponding data type in SQL-92.
+ODBC Programmer's Reference
+************ CC DDaattaa TTyyppeess ************
+ODBC C data types indicate the data type of C buffers used to store data in the
+application.
+All drivers must support all C data types. This is required because all drivers
+must support all C types to which SQL types that they support can be converted,
+and all drivers support at least one character SQL type. Because the character
+SQL type can be converted to and from all C types, all drivers must support all
+C types.
+The C data type is specified in the SSQQLLBBiinnddCCoolland SSQQLLGGeettDDaattaa functions with the
+TargetType argument and in the SSQQLLBBiinnddPPaarraammeetteerr function with the ValueType
+argument. It can also be specified by calling SSQQLLSSeettDDeessccFFiieelldd to set the
+SQL_DESC_CONCISE_TYPE field of an ARD or APD, or by calling SSQQLLSSeettDDeessccRReecc with
+the Type argument (and the SubType argument if needed) and the DescriptorHandle
+argument set to the handle of an ARD or APD.
+The following table lists valid type identifiers for the C data types. The
+table also lists the ODBC C data type that corresponds to each identifier and
+the definition of this data type.
+CC ttyyppee iiddeennttiiffiieerr OODDBBCC CC ttyyppeeddeeff CC ttyyppee
+SQL_C_CHAR SQLCHAR * unsigned char *
+SQL_C_SSHORT[j] SQLSMALLINT short int
+SQL_C_USHORT[j] SQLUSMALLINT unsigned short int
+SQL_C_SLONG[j] SQLINTEGER long int
+SQL_C_ULONG[j] SQLUINTEGER unsigned long int
+SQL_C_FLOAT SQLREAL float
+SQL_C_DOUBLE SQLDOUBLE, SQLFLOAT double
+SQL_C_BIT SQLCHAR unsigned char
+SQL_C_STINYINT[j] SQLSCHAR signed char
+SQL_C_UTINYINT[j] SQLCHAR unsigned char
+SQL_C_SBIGINT SQLBIGINT _int64[h]
+SQL_C_UBIGINT SQLUBIGINT unsigned _int64[h]
+SQL_C_BINARY SQLCHAR * unsigned char *
+SQL_C_BOOKMARK[i] BOOKMARK unsigned long int[d]
+SQL_C_VARBOOKMARK SQLCHAR * unsigned char *
+SQL_C_TYPE_DATE[c] SQL_DATE_STRUCT struct tagDATE_STRUCT {
+ SQLSMALLINT year;
+ SQLUSMALLINT month;
+ SQLUSMALLINT day;
+ } DATE_STRUCT;[a]
+SQL_C_TYPE_TIME[c] SQL_TIME_STRUCT struct tagTIME_STRUCT {
+ SQLUSMALLINT hour;
+ SQLUSMALLINT minute;
+ SQLUSMALLINT second;
+ } TIME_STRUCT;[a]
+SQL_C_TYPE_TIMESTAMP[c] SQL_TIMESTAMP_STRUCT struct tagTIMESTAMP_STRUCT {
+ SQLSMALLINT year;
+ SQLUSMALLINT month;
+ SQLUSMALLINT day;
+ SQLUSMALLINT hour;
+ SQLUSMALLINT minute;
+ SQLUSMALLINT second;
+ SQLUINTEGER fraction;[b]
+ } TIMESTAMP_STRUCT;[a]
+SQL_C_NUMERIC SQL_NUMERIC_STRUCT struct tagSQL_NUMERIC_STRUCT {
+ SQLCHAR precision;
+ SQLSCHAR scale;
+ SQLCHAR sign[g];
+ SQLCHAR
+ val
+ [SQL_MAX_NUMERIC_L EN];
+ [e], [f]
+ } SQL_NUMERIC_STRUCT;
+SQL_C_GUID SQLGUID struct tagSQLGUID {
+ DWORD Data1;
+ WORD Data2;
+ WORD Data3;
+ BYTE Data4[8];
+ } SQLGUID;[k]
+All C interval data SQL_INTERVAL_STRUCT See the "_C_ _I_n_t_e_r_v_a_l_ _S_t_r_u_c_t_u_r_e"
+types section, later in this appendix.
+[a] The values of the year, month, day, hour, minute, and second fields in
+the datetime C data types must conform to the constraints of the Gregorian
+calendar. (See "_C_o_n_s_t_r_a_i_n_t_s_ _o_f_ _t_h_e_ _G_r_e_g_o_r_i_a_n_ _C_a_l_e_n_d_a_r" later in this appendix.)
+[b] The value of the fraction field is the number of billionths of a second
+and ranges from 0 through 999,999,999 (1 less than 1 billion). For example, the
+value of the fraction field for a half-second is 500,000,000, for a thousandth
+of a second (one millisecond) is 1,000,000, for a millionth of a second (one
+microsecond) is 1,000, and for a billionth of a second (one nanosecond) is 1.
+[c] In ODBC 2.x, the C date, time, and timestamp data types are SQL_C_DATE,
+SQL_C_TIME, and SQL_C_TIMESTAMP.
+[d] ODBC 3.x applications should use SQL_C_VARBOOKMARK, not SQL_C_BOOKMARK.
+When an ODBC 3.x application works with an ODBC 2.x driver, the ODBC 3.x Driver
+Manager will map SQL_C_VARBOOKMARK to SQL_C_BOOKMARK.
+[e] A number is stored in the val field of the SQL_NUMERIC_STRUCT structure
+as a scaled integer, in little endian mode (the leftmost byte being the least-
+significant byte). For example, the number 10.001 base 10, with a scale of 4,
+is scaled to an integer of 100010. Because this is 186AA in hexadecimal format,
+the value in SQL_NUMERIC_STRUCT would be "AA 86 01 00 00 … 00", with the number
+of bytes defined by the SQL_MAX_NUMERIC_LEN ##ddeeffiinnee.
+[f] The precision and scale fields of the SQL_C_NUMERIC data type are never
+used for input from an application, only for output from the driver to the
+application. When the driver writes a numeric value into the
+SQL_NUMERIC_STRUCT, it will use its own driver-specific default as the value
+for the precision field, and it will use the value in the SQL_DESC_SCALE field
+of the application descriptor (which defaults to 0) for the scale field. An
+application can provide its own values for precision and scale by setting the
+SQL_DESC_PRECISION and SQL_DESC_SCALE fields of the application descriptor.
+[g] The sign field is 1 if positive, 0 if negative.
+[h] _int64 might not be supplied by some compilers.
+[i] _SQL_C_BOOKMARK has been deprecated in ODBC 3.x.
+[j] _SQL_C_SHORT, SQL_C_LONG, and SQL_C_TINYINT have been replaced in ODBC by
+signed and unsigned types: SQL_C_SSHORT and SQL_C_USHORT, SQL_C_SLONG and
+SQL_C_ULONG, and SQL_C_STINYINT and SQL_C_UTINYINT. An ODBC 3.x driver that
+should work with ODBC 2.x applications should support SQL_C_SHORT, SQL_C_LONG,
+and SQL_C_TINYINT, because when they are called, the Driver Manager passes them
+through to the driver.
+[k] SQL_C_GUID can be converted only to SQL_CHAR or SQL_WCHAR.
diff --git a/ndb/src/client/odbc/driver/Func.data b/ndb/src/client/odbc/driver/Func.data
new file mode 100644
index 00000000000..c32671e1135
--- /dev/null
+++ b/ndb/src/client/odbc/driver/Func.data
@@ -0,0 +1,2822 @@
+$func = {
+ SQLAllocConnect => {
+ type => 'SQLRETURN',
+ name => 'SQLAllocConnect',
+ param => [
+ {
+ type => 'SQLHENV',
+ ptr => 0,
+ name => 'EnvironmentHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLHDBC',
+ ptr => 1,
+ name => 'ConnectionHandle',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLAllocEnv => {
+ type => 'SQLRETURN',
+ name => 'SQLAllocEnv',
+ param => [
+ {
+ type => 'SQLHENV',
+ ptr => 1,
+ name => 'EnvironmentHandle',
+ index => 0,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLAllocHandle => {
+ type => 'SQLRETURN',
+ name => 'SQLAllocHandle',
+ param => [
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'HandleType',
+ index => 0,
+ },
+ {
+ type => 'SQLHANDLE',
+ ptr => 0,
+ name => 'InputHandle',
+ index => 1,
+ },
+ {
+ type => 'SQLHANDLE',
+ ptr => 1,
+ name => 'OutputHandle',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLAllocHandleStd => {
+ type => 'SQLRETURN',
+ name => 'SQLAllocHandleStd',
+ param => [
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'fHandleType',
+ index => 0,
+ },
+ {
+ type => 'SQLHANDLE',
+ ptr => 0,
+ name => 'hInput',
+ index => 1,
+ },
+ {
+ type => 'SQLHANDLE',
+ ptr => 1,
+ name => 'phOutput',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLAllocStmt => {
+ type => 'SQLRETURN',
+ name => 'SQLAllocStmt',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLHSTMT',
+ ptr => 1,
+ name => 'StatementHandle',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLBindCol => {
+ type => 'SQLRETURN',
+ name => 'SQLBindCol',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'ColumnNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'TargetType',
+ index => 2,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'TargetValue',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 4,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'StrLen_or_Ind',
+ index => 5,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLBindParam => {
+ type => 'SQLRETURN',
+ name => 'SQLBindParam',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'ParameterNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'ValueType',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'ParameterType',
+ index => 3,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 0,
+ name => 'LengthPrecision',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'ParameterScale',
+ index => 5,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'ParameterValue',
+ index => 6,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'StrLen_or_Ind',
+ index => 7,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLBindParameter => {
+ type => 'SQLRETURN',
+ name => 'SQLBindParameter',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'ipar',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'fParamType',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'fCType',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'fSqlType',
+ index => 4,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 0,
+ name => 'cbColDef',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'ibScale',
+ index => 6,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'rgbValue',
+ index => 7,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'cbValueMax',
+ index => 8,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'pcbValue',
+ index => 9,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLBrowseConnect => {
+ type => 'SQLRETURN',
+ name => 'SQLBrowseConnect',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'hdbc',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szConnStrIn',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbConnStrIn',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szConnStrOut',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbConnStrOutMax',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'pcbConnStrOut',
+ index => 5,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLBulkOperations => {
+ type => 'SQLRETURN',
+ name => 'SQLBulkOperations',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'Operation',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLCancel => {
+ type => 'SQLRETURN',
+ name => 'SQLCancel',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLCloseCursor => {
+ type => 'SQLRETURN',
+ name => 'SQLCloseCursor',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLColAttribute => {
+ type => 'SQLRETURN',
+ name => 'SQLColAttribute',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'ColumnNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'FieldIdentifier',
+ index => 2,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'CharacterAttribute',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'StringLength',
+ index => 5,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'NumericAttribute',
+ index => 6,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLColAttributes => {
+ type => 'SQLRETURN',
+ name => 'SQLColAttributes',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'icol',
+ index => 1,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'fDescType',
+ index => 2,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'rgbDesc',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbDescMax',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'pcbDesc',
+ index => 5,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'pfDesc',
+ index => 6,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLColumnPrivileges => {
+ type => 'SQLRETURN',
+ name => 'SQLColumnPrivileges',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szCatalogName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbCatalogName',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szSchemaName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbSchemaName',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szTableName',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbTableName',
+ index => 6,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szColumnName',
+ index => 7,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbColumnName',
+ index => 8,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLColumns => {
+ type => 'SQLRETURN',
+ name => 'SQLColumns',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'CatalogName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength1',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'SchemaName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength2',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'TableName',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength3',
+ index => 6,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'ColumnName',
+ index => 7,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength4',
+ index => 8,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLConnect => {
+ type => 'SQLRETURN',
+ name => 'SQLConnect',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'ServerName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength1',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'UserName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength2',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'Authentication',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength3',
+ index => 6,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLCopyDesc => {
+ type => 'SQLRETURN',
+ name => 'SQLCopyDesc',
+ param => [
+ {
+ type => 'SQLHDESC',
+ ptr => 0,
+ name => 'SourceDescHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLHDESC',
+ ptr => 0,
+ name => 'TargetDescHandle',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLDataSources => {
+ type => 'SQLRETURN',
+ name => 'SQLDataSources',
+ param => [
+ {
+ type => 'SQLHENV',
+ ptr => 0,
+ name => 'EnvironmentHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Direction',
+ index => 1,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'ServerName',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength1',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'NameLength1',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'Description',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength2',
+ index => 6,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'NameLength2',
+ index => 7,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLDescribeCol => {
+ type => 'SQLRETURN',
+ name => 'SQLDescribeCol',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'ColumnNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'ColumnName',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'NameLength',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'DataType',
+ index => 5,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 1,
+ name => 'ColumnSize',
+ index => 6,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'DecimalDigits',
+ index => 7,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'Nullable',
+ index => 8,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLDescribeParam => {
+ type => 'SQLRETURN',
+ name => 'SQLDescribeParam',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'ipar',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'pfSqlType',
+ index => 2,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 1,
+ name => 'pcbParamDef',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'pibScale',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'pfNullable',
+ index => 5,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLDisconnect => {
+ type => 'SQLRETURN',
+ name => 'SQLDisconnect',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLDriverConnect => {
+ type => 'SQLRETURN',
+ name => 'SQLDriverConnect',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'hdbc',
+ index => 0,
+ },
+ {
+ type => 'SQLHWND',
+ ptr => 0,
+ name => 'hwnd',
+ index => 1,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szConnStrIn',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbConnStrIn',
+ index => 3,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szConnStrOut',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbConnStrOutMax',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'pcbConnStrOut',
+ index => 6,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'fDriverCompletion',
+ index => 7,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLDrivers => {
+ type => 'SQLRETURN',
+ name => 'SQLDrivers',
+ param => [
+ {
+ type => 'SQLHENV',
+ ptr => 0,
+ name => 'henv',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'fDirection',
+ index => 1,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szDriverDesc',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbDriverDescMax',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'pcbDriverDesc',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szDriverAttributes',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbDrvrAttrMax',
+ index => 6,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'pcbDrvrAttr',
+ index => 7,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLEndTran => {
+ type => 'SQLRETURN',
+ name => 'SQLEndTran',
+ param => [
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'HandleType',
+ index => 0,
+ },
+ {
+ type => 'SQLHANDLE',
+ ptr => 0,
+ name => 'Handle',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'CompletionType',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLError => {
+ type => 'SQLRETURN',
+ name => 'SQLError',
+ param => [
+ {
+ type => 'SQLHENV',
+ ptr => 0,
+ name => 'EnvironmentHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 1,
+ },
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'Sqlstate',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'NativeError',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'MessageText',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 6,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'TextLength',
+ index => 7,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLExecDirect => {
+ type => 'SQLRETURN',
+ name => 'SQLExecDirect',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'StatementText',
+ index => 1,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'TextLength',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLExecute => {
+ type => 'SQLRETURN',
+ name => 'SQLExecute',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLExtendedFetch => {
+ type => 'SQLRETURN',
+ name => 'SQLExtendedFetch',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'fFetchType',
+ index => 1,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'irow',
+ index => 2,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 1,
+ name => 'pcrow',
+ index => 3,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 1,
+ name => 'rgfRowStatus',
+ index => 4,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLFetch => {
+ type => 'SQLRETURN',
+ name => 'SQLFetch',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLFetchScroll => {
+ type => 'SQLRETURN',
+ name => 'SQLFetchScroll',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'FetchOrientation',
+ index => 1,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'FetchOffset',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLForeignKeys => {
+ type => 'SQLRETURN',
+ name => 'SQLForeignKeys',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szPkCatalogName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbPkCatalogName',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szPkSchemaName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbPkSchemaName',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szPkTableName',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbPkTableName',
+ index => 6,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szFkCatalogName',
+ index => 7,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbFkCatalogName',
+ index => 8,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szFkSchemaName',
+ index => 9,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbFkSchemaName',
+ index => 10,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szFkTableName',
+ index => 11,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbFkTableName',
+ index => 12,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLFreeConnect => {
+ type => 'SQLRETURN',
+ name => 'SQLFreeConnect',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLFreeEnv => {
+ type => 'SQLRETURN',
+ name => 'SQLFreeEnv',
+ param => [
+ {
+ type => 'SQLHENV',
+ ptr => 0,
+ name => 'EnvironmentHandle',
+ index => 0,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLFreeHandle => {
+ type => 'SQLRETURN',
+ name => 'SQLFreeHandle',
+ param => [
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'HandleType',
+ index => 0,
+ },
+ {
+ type => 'SQLHANDLE',
+ ptr => 0,
+ name => 'Handle',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLFreeStmt => {
+ type => 'SQLRETURN',
+ name => 'SQLFreeStmt',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Option',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLGetConnectAttr => {
+ type => 'SQLRETURN',
+ name => 'SQLGetConnectAttr',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'Attribute',
+ index => 1,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'StringLength',
+ index => 4,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLGetConnectOption => {
+ type => 'SQLRETURN',
+ name => 'SQLGetConnectOption',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Option',
+ index => 1,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLGetCursorName => {
+ type => 'SQLRETURN',
+ name => 'SQLGetCursorName',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'CursorName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'NameLength',
+ index => 3,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLGetData => {
+ type => 'SQLRETURN',
+ name => 'SQLGetData',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'ColumnNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'TargetType',
+ index => 2,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'TargetValue',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 4,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'StrLen_or_Ind',
+ index => 5,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLGetDescField => {
+ type => 'SQLRETURN',
+ name => 'SQLGetDescField',
+ param => [
+ {
+ type => 'SQLHDESC',
+ ptr => 0,
+ name => 'DescriptorHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'RecNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'FieldIdentifier',
+ index => 2,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 4,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'StringLength',
+ index => 5,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLGetDescRec => {
+ type => 'SQLRETURN',
+ name => 'SQLGetDescRec',
+ param => [
+ {
+ type => 'SQLHDESC',
+ ptr => 0,
+ name => 'DescriptorHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'RecNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'Name',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'StringLength',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'Type',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'SubType',
+ index => 6,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'Length',
+ index => 7,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'Precision',
+ index => 8,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'Scale',
+ index => 9,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'Nullable',
+ index => 10,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLGetDiagField => {
+ type => 'SQLRETURN',
+ name => 'SQLGetDiagField',
+ param => [
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'HandleType',
+ index => 0,
+ },
+ {
+ type => 'SQLHANDLE',
+ ptr => 0,
+ name => 'Handle',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'RecNumber',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'DiagIdentifier',
+ index => 3,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'DiagInfo',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'StringLength',
+ index => 6,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLGetDiagRec => {
+ type => 'SQLRETURN',
+ name => 'SQLGetDiagRec',
+ param => [
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'HandleType',
+ index => 0,
+ },
+ {
+ type => 'SQLHANDLE',
+ ptr => 0,
+ name => 'Handle',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'RecNumber',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'Sqlstate',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'NativeError',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'MessageText',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 6,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'TextLength',
+ index => 7,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLGetEnvAttr => {
+ type => 'SQLRETURN',
+ name => 'SQLGetEnvAttr',
+ param => [
+ {
+ type => 'SQLHENV',
+ ptr => 0,
+ name => 'EnvironmentHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'Attribute',
+ index => 1,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'StringLength',
+ index => 4,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLGetFunctions => {
+ type => 'SQLRETURN',
+ name => 'SQLGetFunctions',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'FunctionId',
+ index => 1,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 1,
+ name => 'Supported',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLGetInfo => {
+ type => 'SQLRETURN',
+ name => 'SQLGetInfo',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'InfoType',
+ index => 1,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'InfoValue',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'StringLength',
+ index => 4,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLGetStmtAttr => {
+ type => 'SQLRETURN',
+ name => 'SQLGetStmtAttr',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'Attribute',
+ index => 1,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'StringLength',
+ index => 4,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLGetStmtOption => {
+ type => 'SQLRETURN',
+ name => 'SQLGetStmtOption',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Option',
+ index => 1,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLGetTypeInfo => {
+ type => 'SQLRETURN',
+ name => 'SQLGetTypeInfo',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'DataType',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLMoreResults => {
+ type => 'SQLRETURN',
+ name => 'SQLMoreResults',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLNativeSql => {
+ type => 'SQLRETURN',
+ name => 'SQLNativeSql',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'hdbc',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szSqlStrIn',
+ index => 1,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'cbSqlStrIn',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szSqlStr',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'cbSqlStrMax',
+ index => 4,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'pcbSqlStr',
+ index => 5,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLNumParams => {
+ type => 'SQLRETURN',
+ name => 'SQLNumParams',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'pcpar',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLNumResultCols => {
+ type => 'SQLRETURN',
+ name => 'SQLNumResultCols',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 1,
+ name => 'ColumnCount',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLParamData => {
+ type => 'SQLRETURN',
+ name => 'SQLParamData',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 1,
+ name => 'Value',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLParamOptions => {
+ type => 'SQLRETURN',
+ name => 'SQLParamOptions',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 0,
+ name => 'crow',
+ index => 1,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 1,
+ name => 'pirow',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLPrepare => {
+ type => 'SQLRETURN',
+ name => 'SQLPrepare',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'StatementText',
+ index => 1,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'TextLength',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLPrimaryKeys => {
+ type => 'SQLRETURN',
+ name => 'SQLPrimaryKeys',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szCatalogName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbCatalogName',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szSchemaName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbSchemaName',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szTableName',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbTableName',
+ index => 6,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLProcedureColumns => {
+ type => 'SQLRETURN',
+ name => 'SQLProcedureColumns',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szCatalogName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbCatalogName',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szSchemaName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbSchemaName',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szProcName',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbProcName',
+ index => 6,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szColumnName',
+ index => 7,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbColumnName',
+ index => 8,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLProcedures => {
+ type => 'SQLRETURN',
+ name => 'SQLProcedures',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szCatalogName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbCatalogName',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szSchemaName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbSchemaName',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szProcName',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbProcName',
+ index => 6,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLPutData => {
+ type => 'SQLRETURN',
+ name => 'SQLPutData',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Data',
+ index => 1,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'StrLen_or_Ind',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLRowCount => {
+ type => 'SQLRETURN',
+ name => 'SQLRowCount',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'RowCount',
+ index => 1,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLSetConnectAttr => {
+ type => 'SQLRETURN',
+ name => 'SQLSetConnectAttr',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'Attribute',
+ index => 1,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'StringLength',
+ index => 3,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLSetConnectOption => {
+ type => 'SQLRETURN',
+ name => 'SQLSetConnectOption',
+ param => [
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Option',
+ index => 1,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLSetCursorName => {
+ type => 'SQLRETURN',
+ name => 'SQLSetCursorName',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'CursorName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLSetDescField => {
+ type => 'SQLRETURN',
+ name => 'SQLSetDescField',
+ param => [
+ {
+ type => 'SQLHDESC',
+ ptr => 0,
+ name => 'DescriptorHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'RecNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'FieldIdentifier',
+ index => 2,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'BufferLength',
+ index => 4,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLSetDescRec => {
+ type => 'SQLRETURN',
+ name => 'SQLSetDescRec',
+ param => [
+ {
+ type => 'SQLHDESC',
+ ptr => 0,
+ name => 'DescriptorHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'RecNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'Type',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'SubType',
+ index => 3,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'Length',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'Precision',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'Scale',
+ index => 6,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Data',
+ index => 7,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'StringLength',
+ index => 8,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'Indicator',
+ index => 9,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLSetEnvAttr => {
+ type => 'SQLRETURN',
+ name => 'SQLSetEnvAttr',
+ param => [
+ {
+ type => 'SQLHENV',
+ ptr => 0,
+ name => 'EnvironmentHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'Attribute',
+ index => 1,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'StringLength',
+ index => 3,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLSetParam => {
+ type => 'SQLRETURN',
+ name => 'SQLSetParam',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'ParameterNumber',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'ValueType',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'ParameterType',
+ index => 3,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 0,
+ name => 'LengthPrecision',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'ParameterScale',
+ index => 5,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'ParameterValue',
+ index => 6,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 1,
+ name => 'StrLen_or_Ind',
+ index => 7,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLSetPos => {
+ type => 'SQLRETURN',
+ name => 'SQLSetPos',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'irow',
+ index => 1,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'fOption',
+ index => 2,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'fLock',
+ index => 3,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLSetScrollOptions => {
+ type => 'SQLRETURN',
+ name => 'SQLSetScrollOptions',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'fConcurrency',
+ index => 1,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'crowKeyset',
+ index => 2,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'crowRowset',
+ index => 3,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLSetStmtAttr => {
+ type => 'SQLRETURN',
+ name => 'SQLSetStmtAttr',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'Attribute',
+ index => 1,
+ },
+ {
+ type => 'SQLPOINTER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ {
+ type => 'SQLINTEGER',
+ ptr => 0,
+ name => 'StringLength',
+ index => 3,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0300',
+ },
+ SQLSetStmtOption => {
+ type => 'SQLRETURN',
+ name => 'SQLSetStmtOption',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Option',
+ index => 1,
+ },
+ {
+ type => 'SQLUINTEGER',
+ ptr => 0,
+ name => 'Value',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLSpecialColumns => {
+ type => 'SQLRETURN',
+ name => 'SQLSpecialColumns',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'IdentifierType',
+ index => 1,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'CatalogName',
+ index => 2,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength1',
+ index => 3,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'SchemaName',
+ index => 4,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength2',
+ index => 5,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'TableName',
+ index => 6,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength3',
+ index => 7,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Scope',
+ index => 8,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Nullable',
+ index => 9,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLStatistics => {
+ type => 'SQLRETURN',
+ name => 'SQLStatistics',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'CatalogName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength1',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'SchemaName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength2',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'TableName',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength3',
+ index => 6,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Unique',
+ index => 7,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'Reserved',
+ index => 8,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLTablePrivileges => {
+ type => 'SQLRETURN',
+ name => 'SQLTablePrivileges',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'hstmt',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szCatalogName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbCatalogName',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szSchemaName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbSchemaName',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'szTableName',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'cbTableName',
+ index => 6,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLTables => {
+ type => 'SQLRETURN',
+ name => 'SQLTables',
+ param => [
+ {
+ type => 'SQLHSTMT',
+ ptr => 0,
+ name => 'StatementHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'CatalogName',
+ index => 1,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength1',
+ index => 2,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'SchemaName',
+ index => 3,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength2',
+ index => 4,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'TableName',
+ index => 5,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength3',
+ index => 6,
+ },
+ {
+ type => 'SQLCHAR',
+ ptr => 1,
+ name => 'TableType',
+ index => 7,
+ },
+ {
+ type => 'SQLSMALLINT',
+ ptr => 0,
+ name => 'NameLength4',
+ index => 8,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+ SQLTransact => {
+ type => 'SQLRETURN',
+ name => 'SQLTransact',
+ param => [
+ {
+ type => 'SQLHENV',
+ ptr => 0,
+ name => 'EnvironmentHandle',
+ index => 0,
+ },
+ {
+ type => 'SQLHDBC',
+ ptr => 0,
+ name => 'ConnectionHandle',
+ index => 1,
+ },
+ {
+ type => 'SQLUSMALLINT',
+ ptr => 0,
+ name => 'CompletionType',
+ index => 2,
+ },
+ ],
+ odbcver => 'ODBCVER >= 0x0000',
+ },
+};
diff --git a/ndb/src/client/odbc/driver/Func.pl b/ndb/src/client/odbc/driver/Func.pl
new file mode 100644
index 00000000000..1064a6a6c6e
--- /dev/null
+++ b/ndb/src/client/odbc/driver/Func.pl
@@ -0,0 +1,352 @@
+#
+
+use strict;
+select(STDOUT);
+$| = 1;
+use vars qw($func);
+
+my $action = shift;
+my @args = @ARGV;
+if (! $action) {
+ print <<END;
+usage: perl $0 <data|name|code|move> ...
+data -unixodbc -- write new Func.data to stdout
+name [-[no]auto -type] [suffix] -- list function names
+code -- write auto/*.cpp
+diff -- diff against auto/*.cpp
+move -- move auto/*.cpp to .
+functab -- write struct entiries for SQLGetFunctions
+END
+ exit(0);
+}
+
+# indents
+my $i1 = " " x (1*4);
+my $i2 = " " x (2*4);
+my $i3 = " " x (3*4);
+my $i4 = " " x (4*4);
+
+if ($action eq 'data') {
+ my @entry = ();
+ while (@args) {
+ my $file = shift(@args);
+ if ($file eq '-unixodbc') {
+ unshift(@args, </usr/local/include/{sql,sqlext}.h>);
+ next;
+ }
+ if ($file eq '-iodbc') {
+ unshift(@args, </usr/local/opt/iODBC/include/{sql,sqlext}.h>);
+ next;
+ }
+ warn "read $file\n";
+ open(F, "<$file") || die "$file: $!";
+ my $text = undef;
+ my $odbcver = undef;
+ while ($_ = <F>) {
+ chomp;
+ if (/^\s*$/) {
+ next;
+ }
+ if (/^\s*#\s*if\s+(.*\bODBCVER\b.*)$/) {
+ $odbcver = $1;
+ $odbcver =~ s/^\s+|\s+$//g;
+ $odbcver =~ s/^\(+|\)+$//g;
+ next;
+ }
+ if (/^\s*#\s*endif\b/) {
+ $odbcver = undef;
+ next;
+ }
+ if (/^\s*SQLRETURN\b/) {
+ $text = "";
+ }
+ if (defined($text)) {
+ $text .= $_;
+ if (/;\s*$/) {
+ push(@entry, {
+ text => $text,
+ odbcver => $odbcver || 'ODBCVER >= 0x0000',
+ });
+ $text = undef;
+ }
+ }
+ }
+ close(F);
+ }
+ warn "@{[ scalar @entry ]} entries\n";
+ $func = {};
+ for my $e (@entry) {
+ my $text = $e->{text};
+ $text =~ s!/\*.+?\*/!!g;
+ $text =~ s/^\s+|\s+$//g;
+ $text =~ s/\s\s*/\040/g;
+ $text =~ /^(SQLRETURN)\s+(SQL_API\s+)?(\w+)\s*\((.*)\)\s*;/
+ or warn "discard: $_\n", next;
+ my $type = $1;
+ my $name = $3;
+ my $body = $4;
+ my $f = {};
+ $f->{type} = $type;
+ $f->{name} = $name;
+ my @body = split(/,/, $body);
+ my $param = [];
+ for my $s (@body) {
+ $s =~ s/^\s+|\s+$//g;
+ my($ptype, $pptr, $pname);
+ if ($s =~ /^(\w+)\s+(\w+)$/) {
+ $ptype = $1;
+ $pptr = 0;
+ $pname = $2;
+ } elsif ($s =~ /^(\w+)\s*\*\s*(\w+)$/) {
+ $ptype = $1;
+ $pptr = 1;
+ $pname = $2;
+ } else {
+ warn "discard: $name: param $s\n";
+ $param = undef;
+ last;
+ }
+ my $pindex = scalar @$param;
+ push(@$param, {
+ type => $ptype,
+ ptr => $pptr,
+ name => $pname,
+ index => $pindex,
+ });
+ }
+ $param or next;
+ $f->{param} = $param;
+ $f->{odbcver} = $e->{odbcver};
+ $func->{$name}
+ and warn "duplicate: $name\n", next;
+ $func->{$name} = $f;
+ }
+ print "\$func = {\n";
+ for my $name (sort keys %$func) {
+ my $f = $func->{$name};
+ print "${i1}$name => {\n";
+ print "${i2}type => '$f->{type}',\n";
+ print "${i2}name => '$f->{name}',\n";
+ print "${i2}param => [\n";
+ for my $p (@{$f->{param}}) {
+ print "${i3}\{\n";
+ print "${i4}type => '$p->{type}',\n";
+ print "${i4}ptr => $p->{ptr},\n";
+ print "${i4}name => '$p->{name}',\n";
+ print "${i4}index => $p->{index},\n";
+ print "${i3}\},\n";
+ }
+ print "${i2}],\n";
+ print "${i2}odbcver => '$f->{odbcver}',\n";
+ print "${i1}},\n";
+ }
+ printf "};\n";
+ $action = undef;
+}
+
+if ($action eq 'name') {
+ my %functab = (); # bit in FuncTab
+ my $functab = "../handles/FuncTab.cpp";
+ if (! open(F, "<$functab")) {
+ warn "$functab: $!";
+ } else {
+ while ($_ = <F>) {
+ if (/SQL_API_([A-Z]+)[\s,]*([01])/) {
+ defined $functab{$1} and die "$_";
+ $functab{$1} = $2;
+ }
+ }
+ close(F);
+ }
+ require './Func.data';
+ my $auto = 1;
+ my $noauto = 1;
+ my $type = 0;
+ while ($args[0] =~ /^-(\w+)$/) {
+ $noauto = 0 if $1 eq 'auto';
+ $auto = 0 if $1 eq 'noauto';
+ $type = 1 if $1 eq 'type';
+ shift(@args);
+ }
+ my $suffix = shift(@args);
+ for my $name (sort keys %$func) {
+ my $f = $func->{$name};
+ local $/ = undef;
+ my($x1);
+ if (open(F, "<$name.cpp")) {
+ $x1 = <F>;
+ close(F);
+ if ($x1 =~ /\bauto_$name\b/) {
+ $auto || next;
+ print "A " if $type;
+ } else {
+ $noauto || next;
+ print "- " if $type;
+ }
+ if ($type) {
+ my $y = $functab{uc $name};
+ $y = "?" if $y !~ /^[01]$/;
+ print "$y ";
+ my $z = $f->{odbcver};
+ $z =~ s/^.*(...)$/$1/;
+ print "$z ";
+ }
+ }
+ print "$name$suffix\n";
+ }
+ $action = undef;
+}
+
+if ($action eq 'code') {
+ require './Func.data';
+ system("rm -rf auto; mkdir auto");
+ for my $name (sort keys %$func) {
+ my $f = $func->{$name};
+ my $file = "auto/$name.cpp";
+ open(F, ">$file") || die "$file: $!\n";
+ print F "#include \"driver.hpp\"\n";
+ print F "\n";
+ printf F "#if $f->{odbcver}\n";
+ print F "$f->{type} SQL_API\n";
+ print F "$f->{name}(";
+ for my $p (@{$f->{param}}) {
+ print F "," if $p->{index} > 0;
+ print F "\n${i1}$p->{type}";
+ for (my $i = 0; $i < $p->{ptr}; $i++) {
+ print F "*";
+ }
+ print F " $p->{name}";
+ }
+ print F ")\n";
+ print F "{\n";
+ print F "${i1}const char* const sqlFunction = \"$f->{name}\";\n";
+ print F "#ifndef auto_$name\n";
+ print F "${i1}Ctx ctx;\n";
+ print F "${i1}ctx.log(1, \"*** not implemented: %s\", sqlFunction);\n";
+ print F "${i1}return SQL_ERROR;\n";
+ print F "#else\n";
+ my @ihandle = ();
+ my @ohandle = ();
+ for my $p (@{$f->{param}}) {
+ if ($p->{type} =~ /^SQLH(ENV|DBC|STMT|DESC)$/) {
+ $p->{btype} = lc $1;
+ my $h = ! $p->{ptr} ? \@ihandle : \@ohandle;
+ push(@$h, $p);
+ }
+ }
+ if (! @ihandle) { # use root handle instance
+ push(@ihandle, {
+ type => 'SQLHROOT',
+ name => '(SQLHANDLE*)0',
+ });
+ }
+ for my $p (@ihandle, @ohandle) {
+ $p->{htype} = "Handle" . (ucfirst lc $p->{btype});
+ $p->{hname} = "p" . (ucfirst lc $p->{btype});
+ }
+ if (@ihandle) {
+ print F "${i1}HandleRoot* const pRoot = HandleRoot::instance();\n";
+ }
+ for my $p (@ihandle) {
+ print F "${i1}$p->{htype}* $p->{hname} = ";
+ print F "pRoot->find" . ucfirst($p->{btype}). "($p->{name});\n";
+ print F "${i1}if ($p->{hname} == 0)\n";
+ print F "${i2}return SQL_INVALID_HANDLE;\n";
+ }
+ {
+ my $p = $ihandle[0];
+ print F "${i1}Ctx& ctx = $p->{hname}->initCtx();\n";
+ print F "${i1}ctx.logSqlEnter(sqlFunction);\n";
+ }
+ for my $p (@ohandle) {
+ print F "${i1}$p->{htype}* $p->{hname} = 0;\n";
+ }
+ {
+ my $p = $ihandle[0];
+ my $fname = $f->{name};
+ $fname =~ s/^SQL/sql/; # keep sql prefix
+ print F "${i1}if (ctx.ok())\n";
+ print F "${i2}$p->{hname}->$fname(\n";
+ print F "${i3}ctx";
+ }
+ for my $p (@{$f->{param}}) {
+ if ($p == $ihandle[0]) {
+ next;
+ }
+ print F ",";
+ print F "\n${i3}";
+ if (grep($_ == $p, @ihandle)) {
+ print F "$p->{hname}";
+ } elsif (grep($_ == $p, @ohandle)) {
+ print F "$p->{name} != 0 ? &$p->{hname} : 0";
+ } else {
+ print F "&" if $p->{ptr} > 0;
+ print F "$p->{name}";
+ }
+ }
+ print F "\n${i2});\n";
+ for my $p (@ohandle) {
+ print F "${i1}if ($p->{name} != 0)\n";
+ print F "${i2}*$p->{name} = ";
+ print F "pRoot->from" . ucfirst($p->{btype}) . "($p->{hname});\n";
+ }
+ {
+ my $p = $ihandle[0];
+ print F "${i1}$p->{hname}->saveCtx(ctx);\n";
+ }
+ print F "${i1}ctx.logSqlExit();\n";
+ print F "${i1}return ctx.getCode();\n";
+ print F "#endif\n";
+ print F "}\n";
+ print F "#endif // $f->{odbcver}\n";
+ close(F);
+ }
+ $action = undef;
+}
+
+if ($action eq 'diff' || $action eq 'move') {
+ require './Func.data';
+ for my $name (sort keys %$func) {
+ local $/ = undef;
+ my($x1, $x2);
+ if (open(F, "<$name.cpp")) {
+ $x1 = <F>;
+ close(F);
+ if ($x1 !~ /\bauto_$name\b/) {
+ warn "$name.cpp: not auto-generated\n" if $action eq 'move';
+ next;
+ }
+ }
+ if (! open(F, "<auto/$name.cpp")) {
+ die "auto/$name.cpp: $!\n";
+ }
+ $x2 = <F>;
+ close(F);
+ if ($x1 eq $x2) {
+ warn "$name: no changes\n" if $action eq 'move';
+ next;
+ }
+ if ($action eq 'diff') {
+ print "=" x 40, "\n";
+ print "diff $name.cpp auto/", "\n";
+ system("diff $name.cpp auto/$name.cpp");
+ } else {
+ rename("auto/$name.cpp", "$name.cpp")
+ or die "rename $name: $!\n";
+ warn "$name: updated\n" if 0;
+ }
+ }
+ $action = undef;
+}
+
+if ($action eq 'functab') {
+ require './Func.data';
+ for my $name (sort keys %$func) {
+ printf "%4s{%3s%-30s, 0 },\n", "", "", uc "SQL_API_$name";
+ }
+ $action = undef;
+}
+
+$action && die "$action: undefined\n";
+
+# vim: set sw=4:
diff --git a/ndb/src/client/odbc/driver/Makefile b/ndb/src/client/odbc/driver/Makefile
new file mode 100644
index 00000000000..62f82371da4
--- /dev/null
+++ b/ndb/src/client/odbc/driver/Makefile
@@ -0,0 +1,16 @@
+include .defs.mk
+
+TYPE = *
+
+NONPIC_ARCHIVE = N
+
+PIC_ARCHIVE = Y
+
+ARCHIVE_TARGET = odbcdriver
+
+SOURCES = driver.cpp
+
+SOURCES_EXTRA = $(shell perl Func.pl name .cpp)
+
+include ../Extra.mk
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/client/odbc/driver/SQLAllocConnect.cpp b/ndb/src/client/odbc/driver/SQLAllocConnect.cpp
new file mode 100644
index 00000000000..a7ffd8c89d1
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLAllocConnect.cpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLAllocConnect(
+ SQLHENV EnvironmentHandle,
+ SQLHDBC* ConnectionHandle)
+{
+ driver_enter(SQL_API_SQLALLOCCONNECT);
+ const char* const sqlFunction = "SQLAllocConnect";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleEnv* pEnv = pRoot->findEnv(EnvironmentHandle);
+ if (pEnv == 0) {
+ driver_exit(SQL_API_SQLALLOCCONNECT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ HandleDbc* pDbc = 0;
+ HandleDbc** ppDbc = 0;
+ if (ConnectionHandle != 0)
+ ppDbc = &pDbc;
+ try {
+ pEnv->sqlAllocConnect(ctx, ppDbc);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ if (ConnectionHandle != 0)
+ *ConnectionHandle = static_cast<SQLHDBC>(pDbc);
+ pEnv->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLALLOCCONNECT);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLAllocEnv.cpp b/ndb/src/client/odbc/driver/SQLAllocEnv.cpp
new file mode 100644
index 00000000000..a62dae61008
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLAllocEnv.cpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLAllocEnv(
+ SQLHENV* EnvironmentHandle)
+{
+ driver_enter(SQL_API_SQLALLOCENV);
+ const char* const sqlFunction = "SQLAllocEnv";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ HandleEnv* pEnv = 0;
+ HandleEnv** ppEnv = 0;
+ if (EnvironmentHandle != 0)
+ ppEnv = &pEnv;
+ try {
+ pRoot->sqlAllocEnv(ctx, ppEnv);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ if (EnvironmentHandle != 0)
+ *EnvironmentHandle = static_cast<SQLHENV>(pEnv);
+ pRoot->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLALLOCENV);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLAllocHandle.cpp b/ndb/src/client/odbc/driver/SQLAllocHandle.cpp
new file mode 100644
index 00000000000..9daf6ead946
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLAllocHandle.cpp
@@ -0,0 +1,62 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLAllocHandle(
+ SQLSMALLINT HandleType,
+ SQLHANDLE InputHandle,
+ SQLHANDLE* OutputHandle)
+{
+ driver_enter(SQL_API_SQLALLOCHANDLE);
+ const char* const sqlFunction = "SQLAllocHandle";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ SQLSMALLINT parentType = pRoot->findParentType(HandleType);
+ if (parentType == -1) {
+ driver_exit(SQL_API_SQLALLOCHANDLE);
+ return SQL_INVALID_HANDLE;
+ }
+ HandleBase* pParent = pRoot->findBase(parentType, InputHandle);
+ if (pParent == 0) {
+ driver_exit(SQL_API_SQLALLOCHANDLE);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ HandleBase* pChild = 0;
+ HandleBase** ppChild = 0;
+ if (OutputHandle != 0)
+ ppChild = &pChild;
+ try {
+ pParent->sqlAllocHandle(ctx, HandleType, ppChild);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ if (OutputHandle != 0)
+ *OutputHandle = static_cast<SQLHANDLE>(pChild);
+ if (pRoot == pParent)
+ pRoot->lockHandle();
+ pParent->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ if (pRoot == pParent)
+ pRoot->unlockHandle();
+ driver_exit(SQL_API_SQLALLOCHANDLE);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLAllocHandleStd.cpp b/ndb/src/client/odbc/driver/SQLAllocHandleStd.cpp
new file mode 100644
index 00000000000..61290e37b7b
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLAllocHandleStd.cpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLAllocHandleStd(
+ SQLSMALLINT fHandleType,
+ SQLHANDLE hInput,
+ SQLHANDLE* phOutput)
+{
+#ifndef auto_SQLAllocHandleStd
+ const char* const sqlFunction = "SQLAllocHandleStd";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLALLOCHANDLESTD);
+ const char* const sqlFunction = "SQLAllocHandleStd";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ Handle* p = pRoot->find((SQLHANDLE*)0);
+ if (p == 0) {
+ driver_exit(SQL_API_SQLALLOCHANDLESTD);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ p->sqlAllocHandleStd(
+ ctx,
+ fHandleType,
+ hInput,
+ &phOutput
+ );
+ p->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLALLOCHANDLESTD);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLAllocStmt.cpp b/ndb/src/client/odbc/driver/SQLAllocStmt.cpp
new file mode 100644
index 00000000000..bf3f149f5de
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLAllocStmt.cpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLAllocStmt(
+ SQLHDBC ConnectionHandle,
+ SQLHSTMT* StatementHandle)
+{
+ driver_enter(SQL_API_SQLALLOCSTMT);
+ const char* const sqlFunction = "SQLAllocStmt";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLALLOCSTMT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ HandleStmt* pStmt = 0;
+ HandleStmt** ppStmt = 0;
+ if (StatementHandle != 0)
+ ppStmt = &pStmt;
+ try {
+ pDbc->sqlAllocStmt(ctx, ppStmt);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ if (StatementHandle != 0)
+ *StatementHandle = static_cast<SQLHSTMT>(pStmt);
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLALLOCSTMT);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLBindCol.cpp b/ndb/src/client/odbc/driver/SQLBindCol.cpp
new file mode 100644
index 00000000000..5562334e8cc
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLBindCol.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLBindCol(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT ColumnNumber,
+ SQLSMALLINT TargetType,
+ SQLPOINTER TargetValue,
+ SQLINTEGER BufferLength,
+ SQLINTEGER* StrLen_or_Ind)
+{
+ driver_enter(SQL_API_SQLBINDCOL);
+ const char* const sqlFunction = "SQLBindCol";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLBINDCOL);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlBindCol(ctx, ColumnNumber, TargetType, TargetValue, BufferLength, StrLen_or_Ind);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLBINDCOL);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLBindParam.cpp b/ndb/src/client/odbc/driver/SQLBindParam.cpp
new file mode 100644
index 00000000000..2fcc17b872f
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLBindParam.cpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLBindParam(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT ParameterNumber,
+ SQLSMALLINT ValueType,
+ SQLSMALLINT ParameterType,
+ SQLUINTEGER LengthPrecision,
+ SQLSMALLINT ParameterScale,
+ SQLPOINTER ParameterValue,
+ SQLINTEGER* StrLen_or_Ind)
+{
+ driver_enter(SQL_API_SQLBINDPARAM);
+ const char* const sqlFunction = "SQLBindParam";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLBINDPARAM);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlBindParam(ctx, ParameterNumber, ValueType, ParameterType, LengthPrecision, ParameterScale, ParameterValue, StrLen_or_Ind);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLBINDPARAM);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLBindParameter.cpp b/ndb/src/client/odbc/driver/SQLBindParameter.cpp
new file mode 100644
index 00000000000..e4ca5bbc731
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLBindParameter.cpp
@@ -0,0 +1,54 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLBindParameter(
+ SQLHSTMT hstmt,
+ SQLUSMALLINT ipar,
+ SQLSMALLINT fParamType,
+ SQLSMALLINT fCType,
+ SQLSMALLINT fSqlType,
+ SQLUINTEGER cbColDef,
+ SQLSMALLINT ibScale,
+ SQLPOINTER rgbValue,
+ SQLINTEGER cbValueMax,
+ SQLINTEGER* pcbValue)
+{
+ driver_enter(SQL_API_SQLBINDPARAMETER);
+ const char* const sqlFunction = "SQLBindParameter";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLBINDPARAMETER);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlBindParameter(ctx, ipar, fParamType, fCType, fSqlType, cbColDef, ibScale, rgbValue, cbValueMax, pcbValue);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLBINDPARAMETER);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLBrowseConnect.cpp b/ndb/src/client/odbc/driver/SQLBrowseConnect.cpp
new file mode 100644
index 00000000000..7e629e199e5
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLBrowseConnect.cpp
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLBrowseConnect(
+ SQLHDBC hdbc,
+ SQLCHAR* szConnStrIn,
+ SQLSMALLINT cbConnStrIn,
+ SQLCHAR* szConnStrOut,
+ SQLSMALLINT cbConnStrOutMax,
+ SQLSMALLINT* pcbConnStrOut)
+{
+#ifndef auto_SQLBrowseConnect
+ const char* const sqlFunction = "SQLBrowseConnect";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLBROWSECONNECT);
+ const char* const sqlFunction = "SQLBrowseConnect";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(hdbc);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLBROWSECONNECT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pDbc->sqlBrowseConnect(
+ ctx,
+ &szConnStrIn,
+ cbConnStrIn,
+ &szConnStrOut,
+ cbConnStrOutMax,
+ &pcbConnStrOut
+ );
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLBROWSECONNECT);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLBulkOperations.cpp b/ndb/src/client/odbc/driver/SQLBulkOperations.cpp
new file mode 100644
index 00000000000..7d256d66e3f
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLBulkOperations.cpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLBulkOperations(
+ SQLHSTMT StatementHandle,
+ SQLSMALLINT Operation)
+{
+#ifndef auto_SQLBulkOperations
+ const char* const sqlFunction = "SQLBulkOperations";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLBULKOPERATIONS);
+ const char* const sqlFunction = "SQLBulkOperations";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLBULKOPERATIONS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlBulkOperations(
+ ctx,
+ Operation
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLBULKOPERATIONS);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLCancel.cpp b/ndb/src/client/odbc/driver/SQLCancel.cpp
new file mode 100644
index 00000000000..ac4e43c6e89
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLCancel.cpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLCancel(
+ SQLHSTMT StatementHandle)
+{
+ driver_enter(SQL_API_SQLCANCEL);
+ const char* const sqlFunction = "SQLCancel";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLCANCEL);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlCancel(ctx);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLCANCEL);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLCloseCursor.cpp b/ndb/src/client/odbc/driver/SQLCloseCursor.cpp
new file mode 100644
index 00000000000..26d88c91e3b
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLCloseCursor.cpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLCloseCursor(
+ SQLHSTMT StatementHandle)
+{
+ driver_enter(SQL_API_SQLCLOSECURSOR);
+ const char* const sqlFunction = "SQLCloseCursor";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLCLOSECURSOR);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlCloseCursor(ctx);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLCLOSECURSOR);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLColAttribute.cpp b/ndb/src/client/odbc/driver/SQLColAttribute.cpp
new file mode 100644
index 00000000000..0e7e5446932
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLColAttribute.cpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLColAttribute(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT ColumnNumber,
+ SQLUSMALLINT FieldIdentifier,
+ SQLPOINTER CharacterAttribute,
+ SQLSMALLINT BufferLength,
+ SQLSMALLINT* StringLength,
+ SQLPOINTER NumericAttribute)
+{
+ driver_enter(SQL_API_SQLCOLATTRIBUTE);
+ const char* const sqlFunction = "SQLColAttribute";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLCOLATTRIBUTE);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlColAttribute(ctx, ColumnNumber, FieldIdentifier, CharacterAttribute, BufferLength, StringLength, NumericAttribute);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLCOLATTRIBUTE);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLColAttributes.cpp b/ndb/src/client/odbc/driver/SQLColAttributes.cpp
new file mode 100644
index 00000000000..05a4c1d4d37
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLColAttributes.cpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLColAttributes(
+ SQLHSTMT hstmt,
+ SQLUSMALLINT icol,
+ SQLUSMALLINT fDescType,
+ SQLPOINTER rgbDesc,
+ SQLSMALLINT cbDescMax,
+ SQLSMALLINT* pcbDesc,
+ SQLINTEGER* pfDesc)
+{
+ driver_enter(SQL_API_SQLCOLATTRIBUTES);
+ const char* const sqlFunction = "SQLColAttributes";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLCOLATTRIBUTES);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlColAttributes(ctx, icol, fDescType, rgbDesc, cbDescMax, pcbDesc, pfDesc);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLCOLATTRIBUTES);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLColumnPrivileges.cpp b/ndb/src/client/odbc/driver/SQLColumnPrivileges.cpp
new file mode 100644
index 00000000000..cfbc9c2bc57
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLColumnPrivileges.cpp
@@ -0,0 +1,67 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLColumnPrivileges(
+ SQLHSTMT hstmt,
+ SQLCHAR* szCatalogName,
+ SQLSMALLINT cbCatalogName,
+ SQLCHAR* szSchemaName,
+ SQLSMALLINT cbSchemaName,
+ SQLCHAR* szTableName,
+ SQLSMALLINT cbTableName,
+ SQLCHAR* szColumnName,
+ SQLSMALLINT cbColumnName)
+{
+#ifndef auto_SQLColumnPrivileges
+ const char* const sqlFunction = "SQLColumnPrivileges";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLCOLUMNPRIVILEGES);
+ const char* const sqlFunction = "SQLColumnPrivileges";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLCOLUMNPRIVILEGES);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlColumnPrivileges(
+ ctx,
+ &szCatalogName,
+ cbCatalogName,
+ &szSchemaName,
+ cbSchemaName,
+ &szTableName,
+ cbTableName,
+ &szColumnName,
+ cbColumnName
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLCOLUMNPRIVILEGES);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLColumns.cpp b/ndb/src/client/odbc/driver/SQLColumns.cpp
new file mode 100644
index 00000000000..4e0b646ee7d
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLColumns.cpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLColumns(
+ SQLHSTMT StatementHandle,
+ SQLCHAR* CatalogName,
+ SQLSMALLINT NameLength1,
+ SQLCHAR* SchemaName,
+ SQLSMALLINT NameLength2,
+ SQLCHAR* TableName,
+ SQLSMALLINT NameLength3,
+ SQLCHAR* ColumnName,
+ SQLSMALLINT NameLength4)
+{
+ driver_enter(SQL_API_SQLCOLUMNS);
+ const char* const sqlFunction = "SQLColumns";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLCOLUMNS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ pStmt->sqlColumns(ctx, CatalogName, NameLength1, SchemaName, NameLength2, TableName, NameLength3, ColumnName, NameLength4);
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLCOLUMNS);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLConnect.cpp b/ndb/src/client/odbc/driver/SQLConnect.cpp
new file mode 100644
index 00000000000..d8f30ed47e7
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLConnect.cpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLConnect(
+ SQLHDBC ConnectionHandle,
+ SQLCHAR* ServerName,
+ SQLSMALLINT NameLength1,
+ SQLCHAR* UserName,
+ SQLSMALLINT NameLength2,
+ SQLCHAR* Authentication,
+ SQLSMALLINT NameLength3)
+{
+ driver_enter(SQL_API_SQLCONNECT);
+ const char* const sqlFunction = "SQLConnect";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLCONNECT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlConnect(ctx, ServerName, NameLength1, UserName, NameLength2, Authentication, NameLength3);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLCONNECT);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLCopyDesc.cpp b/ndb/src/client/odbc/driver/SQLCopyDesc.cpp
new file mode 100644
index 00000000000..b4d4b2e4122
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLCopyDesc.cpp
@@ -0,0 +1,58 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLCopyDesc(
+ SQLHDESC SourceDescHandle,
+ SQLHDESC TargetDescHandle)
+{
+#ifndef auto_SQLCopyDesc
+ const char* const sqlFunction = "SQLCopyDesc";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLCOPYDESC);
+ const char* const sqlFunction = "SQLCopyDesc";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDesc* pDesc = pRoot->findDesc(SourceDescHandle);
+ if (pDesc == 0) {
+ driver_exit(SQL_API_SQLCOPYDESC);
+ return SQL_INVALID_HANDLE;
+ }
+ HandleDesc* pDesc = pRoot->findDesc(TargetDescHandle);
+ if (pDesc == 0) {
+ driver_exit(SQL_API_SQLCOPYDESC);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pDesc->sqlCopyDesc(
+ ctx,
+ pDesc
+ );
+ pDesc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLCOPYDESC);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLDataSources.cpp b/ndb/src/client/odbc/driver/SQLDataSources.cpp
new file mode 100644
index 00000000000..6115e7175f9
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLDataSources.cpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLDataSources(
+ SQLHENV EnvironmentHandle,
+ SQLUSMALLINT Direction,
+ SQLCHAR* ServerName,
+ SQLSMALLINT BufferLength1,
+ SQLSMALLINT* NameLength1,
+ SQLCHAR* Description,
+ SQLSMALLINT BufferLength2,
+ SQLSMALLINT* NameLength2)
+{
+#ifndef auto_SQLDataSources
+ const char* const sqlFunction = "SQLDataSources";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLDATASOURCES);
+ const char* const sqlFunction = "SQLDataSources";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleEnv* pEnv = pRoot->findEnv(EnvironmentHandle);
+ if (pEnv == 0) {
+ driver_exit(SQL_API_SQLDATASOURCES);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pEnv->sqlDataSources(
+ ctx,
+ Direction,
+ &ServerName,
+ BufferLength1,
+ &NameLength1,
+ &Description,
+ BufferLength2,
+ &NameLength2
+ );
+ pEnv->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLDATASOURCES);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLDescribeCol.cpp b/ndb/src/client/odbc/driver/SQLDescribeCol.cpp
new file mode 100644
index 00000000000..f15ce8962f1
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLDescribeCol.cpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLDescribeCol(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT ColumnNumber,
+ SQLCHAR* ColumnName,
+ SQLSMALLINT BufferLength,
+ SQLSMALLINT* NameLength,
+ SQLSMALLINT* DataType,
+ SQLUINTEGER* ColumnSize,
+ SQLSMALLINT* DecimalDigits,
+ SQLSMALLINT* Nullable)
+{
+ driver_enter(SQL_API_SQLDESCRIBECOL);
+ const char* const sqlFunction = "SQLDescribeCol";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLDESCRIBECOL);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlDescribeCol(ctx, ColumnNumber, ColumnName, BufferLength, NameLength, DataType, ColumnSize, DecimalDigits, Nullable);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLDESCRIBECOL);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLDescribeParam.cpp b/ndb/src/client/odbc/driver/SQLDescribeParam.cpp
new file mode 100644
index 00000000000..beff41396fe
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLDescribeParam.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLDescribeParam(
+ SQLHSTMT hstmt,
+ SQLUSMALLINT ipar,
+ SQLSMALLINT* pfSqlType,
+ SQLUINTEGER* pcbParamDef,
+ SQLSMALLINT* pibScale,
+ SQLSMALLINT* pfNullable)
+{
+ driver_enter(SQL_API_SQLDESCRIBEPARAM);
+ const char* const sqlFunction = "SQLDescribeParam";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLDESCRIBEPARAM);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlDescribeParam(ctx, ipar, pfSqlType, pcbParamDef, pibScale, pfNullable);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLDESCRIBEPARAM);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLDisconnect.cpp b/ndb/src/client/odbc/driver/SQLDisconnect.cpp
new file mode 100644
index 00000000000..75db5604da8
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLDisconnect.cpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLDisconnect(
+ SQLHDBC ConnectionHandle)
+{
+ driver_enter(SQL_API_SQLDISCONNECT);
+ const char* const sqlFunction = "SQLDisconnect";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLDISCONNECT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlDisconnect(ctx);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLDISCONNECT);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLDriverConnect.cpp b/ndb/src/client/odbc/driver/SQLDriverConnect.cpp
new file mode 100644
index 00000000000..340babd8523
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLDriverConnect.cpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLDriverConnect(
+ SQLHDBC hdbc,
+ SQLHWND hwnd,
+ SQLCHAR* szConnStrIn,
+ SQLSMALLINT cbConnStrIn,
+ SQLCHAR* szConnStrOut,
+ SQLSMALLINT cbConnStrOutMax,
+ SQLSMALLINT* pcbConnStrOut,
+ SQLUSMALLINT fDriverCompletion)
+{
+ driver_enter(SQL_API_SQLDRIVERCONNECT);
+ const char* const sqlFunction = "SQLDriverConnect";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(hdbc);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLDRIVERCONNECT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlDriverConnect(ctx, hwnd, szConnStrIn, cbConnStrIn, szConnStrOut, cbConnStrOutMax, pcbConnStrOut, fDriverCompletion);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLDRIVERCONNECT);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLDrivers.cpp b/ndb/src/client/odbc/driver/SQLDrivers.cpp
new file mode 100644
index 00000000000..9c52f900992
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLDrivers.cpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLDrivers(
+ SQLHENV henv,
+ SQLUSMALLINT fDirection,
+ SQLCHAR* szDriverDesc,
+ SQLSMALLINT cbDriverDescMax,
+ SQLSMALLINT* pcbDriverDesc,
+ SQLCHAR* szDriverAttributes,
+ SQLSMALLINT cbDrvrAttrMax,
+ SQLSMALLINT* pcbDrvrAttr)
+{
+#ifndef auto_SQLDrivers
+ const char* const sqlFunction = "SQLDrivers";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLDRIVERS);
+ const char* const sqlFunction = "SQLDrivers";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleEnv* pEnv = pRoot->findEnv(henv);
+ if (pEnv == 0) {
+ driver_exit(SQL_API_SQLDRIVERS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pEnv->sqlDrivers(
+ ctx,
+ fDirection,
+ &szDriverDesc,
+ cbDriverDescMax,
+ &pcbDriverDesc,
+ &szDriverAttributes,
+ cbDrvrAttrMax,
+ &pcbDrvrAttr
+ );
+ pEnv->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLDRIVERS);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLEndTran.cpp b/ndb/src/client/odbc/driver/SQLEndTran.cpp
new file mode 100644
index 00000000000..20b0b2203f5
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLEndTran.cpp
@@ -0,0 +1,70 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLEndTran(
+ SQLSMALLINT HandleType,
+ SQLHANDLE Handle,
+ SQLSMALLINT CompletionType)
+{
+ driver_enter(SQL_API_SQLENDTRAN);
+ const char* const sqlFunction = "SQLEndTran";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ if (HandleType == SQL_HANDLE_DBC) {
+ HandleDbc* pDbc = pRoot->findDbc(Handle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLENDTRAN);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlEndTran(ctx, CompletionType);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLENDTRAN);
+ return ret;
+ }
+ if (HandleType == SQL_HANDLE_ENV) {
+ HandleEnv* pEnv = pRoot->findEnv(Handle);
+ if (pEnv == 0) {
+ driver_exit(SQL_API_SQLENDTRAN);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pEnv->sqlEndTran(ctx, CompletionType);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pEnv->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLENDTRAN);
+ return ret;
+ }
+ driver_exit(SQL_API_SQLENDTRAN);
+ return SQL_INVALID_HANDLE;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLError.cpp b/ndb/src/client/odbc/driver/SQLError.cpp
new file mode 100644
index 00000000000..af78c931d37
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLError.cpp
@@ -0,0 +1,67 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLError(
+ SQLHENV EnvironmentHandle,
+ SQLHDBC ConnectionHandle,
+ SQLHSTMT StatementHandle,
+ SQLCHAR* Sqlstate,
+ SQLINTEGER* NativeError,
+ SQLCHAR* MessageText,
+ SQLSMALLINT BufferLength,
+ SQLSMALLINT* TextLength)
+{
+ driver_enter(SQL_API_SQLERROR);
+ const char* const sqlFunction = "SQLError";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleEnv* pEnv = pRoot->findEnv(EnvironmentHandle);
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0 && pDbc == 0 && pEnv == 0) {
+ driver_exit(SQL_API_SQLERROR);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx ctx; // discard diagnostics
+ ctx.logSqlEnter(sqlFunction);
+ if (pStmt != 0) {
+ try {
+ pStmt->sqlError(ctx, Sqlstate, NativeError, MessageText, BufferLength, TextLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ } else if (pDbc != 0) {
+ try {
+ pDbc->sqlError(ctx, Sqlstate, NativeError, MessageText, BufferLength, TextLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ } else {
+ try {
+ pEnv->sqlError(ctx, Sqlstate, NativeError, MessageText, BufferLength, TextLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ }
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLERROR);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLExecDirect.cpp b/ndb/src/client/odbc/driver/SQLExecDirect.cpp
new file mode 100644
index 00000000000..0ad99d29cd9
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLExecDirect.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLExecDirect(
+ SQLHSTMT StatementHandle,
+ SQLCHAR* StatementText,
+ SQLINTEGER TextLength)
+{
+ driver_enter(SQL_API_SQLEXECDIRECT);
+ const char* const sqlFunction = "SQLExecDirect";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLEXECDIRECT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlExecDirect(ctx, StatementText, TextLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLEXECDIRECT);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLExecute.cpp b/ndb/src/client/odbc/driver/SQLExecute.cpp
new file mode 100644
index 00000000000..9c30d418f09
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLExecute.cpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLExecute(
+ SQLHSTMT StatementHandle)
+{
+ driver_enter(SQL_API_SQLEXECUTE);
+ const char* const sqlFunction = "SQLExecute";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLEXECUTE);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlExecute(ctx);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLEXECUTE);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLExtendedFetch.cpp b/ndb/src/client/odbc/driver/SQLExtendedFetch.cpp
new file mode 100644
index 00000000000..e0dd078b5d0
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLExtendedFetch.cpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLExtendedFetch(
+ SQLHSTMT hstmt,
+ SQLUSMALLINT fFetchType,
+ SQLINTEGER irow,
+ SQLUINTEGER* pcrow,
+ SQLUSMALLINT* rgfRowStatus)
+{
+#ifndef auto_SQLExtendedFetch
+ const char* const sqlFunction = "SQLExtendedFetch";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLEXTENDEDFETCH);
+ const char* const sqlFunction = "SQLExtendedFetch";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLEXTENDEDFETCH);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlExtendedFetch(
+ ctx,
+ fFetchType,
+ irow,
+ &pcrow,
+ &rgfRowStatus
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLEXTENDEDFETCH);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLFetch.cpp b/ndb/src/client/odbc/driver/SQLFetch.cpp
new file mode 100644
index 00000000000..addba7b998c
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLFetch.cpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLFetch(
+ SQLHSTMT StatementHandle)
+{
+ driver_enter(SQL_API_SQLFETCH);
+ const char* const sqlFunction = "SQLFetch";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLFETCH);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlFetch(ctx);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLFETCH);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLFetchScroll.cpp b/ndb/src/client/odbc/driver/SQLFetchScroll.cpp
new file mode 100644
index 00000000000..cfbfc813fca
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLFetchScroll.cpp
@@ -0,0 +1,55 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLFetchScroll(
+ SQLHSTMT StatementHandle,
+ SQLSMALLINT FetchOrientation,
+ SQLINTEGER FetchOffset)
+{
+#ifndef auto_SQLFetchScroll
+ const char* const sqlFunction = "SQLFetchScroll";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLFETCHSCROLL);
+ const char* const sqlFunction = "SQLFetchScroll";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLFETCHSCROLL);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlFetchScroll(
+ ctx,
+ FetchOrientation,
+ FetchOffset
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLFETCHSCROLL);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLForeignKeys.cpp b/ndb/src/client/odbc/driver/SQLForeignKeys.cpp
new file mode 100644
index 00000000000..886ac6bdaa5
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLForeignKeys.cpp
@@ -0,0 +1,75 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLForeignKeys(
+ SQLHSTMT hstmt,
+ SQLCHAR* szPkCatalogName,
+ SQLSMALLINT cbPkCatalogName,
+ SQLCHAR* szPkSchemaName,
+ SQLSMALLINT cbPkSchemaName,
+ SQLCHAR* szPkTableName,
+ SQLSMALLINT cbPkTableName,
+ SQLCHAR* szFkCatalogName,
+ SQLSMALLINT cbFkCatalogName,
+ SQLCHAR* szFkSchemaName,
+ SQLSMALLINT cbFkSchemaName,
+ SQLCHAR* szFkTableName,
+ SQLSMALLINT cbFkTableName)
+{
+#ifndef auto_SQLForeignKeys
+ const char* const sqlFunction = "SQLForeignKeys";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLFOREIGNKEYS);
+ const char* const sqlFunction = "SQLForeignKeys";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLFOREIGNKEYS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlForeignKeys(
+ ctx,
+ &szPkCatalogName,
+ cbPkCatalogName,
+ &szPkSchemaName,
+ cbPkSchemaName,
+ &szPkTableName,
+ cbPkTableName,
+ &szFkCatalogName,
+ cbFkCatalogName,
+ &szFkSchemaName,
+ cbFkSchemaName,
+ &szFkTableName,
+ cbFkTableName
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLFOREIGNKEYS);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLFreeConnect.cpp b/ndb/src/client/odbc/driver/SQLFreeConnect.cpp
new file mode 100644
index 00000000000..9ac84710cce
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLFreeConnect.cpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLFreeConnect(
+ SQLHDBC ConnectionHandle)
+{
+ driver_enter(SQL_API_SQLFREECONNECT);
+ const char* const sqlFunction = "SQLFreeConnect";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLFREECONNECT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ HandleEnv* pEnv = pDbc->getEnv();
+ try {
+ pEnv->sqlFreeConnect(ctx, pDbc);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ if (! ctx.ok()) {
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLFREECONNECT);
+ return ret;
+ }
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ delete &ctx;
+ driver_exit(SQL_API_SQLFREECONNECT);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLFreeEnv.cpp b/ndb/src/client/odbc/driver/SQLFreeEnv.cpp
new file mode 100644
index 00000000000..7e35056feb5
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLFreeEnv.cpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLFreeEnv(
+ SQLHENV EnvironmentHandle)
+{
+ driver_enter(SQL_API_SQLFREEENV);
+ const char* const sqlFunction = "SQLFreeEnv";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleEnv* pEnv = pRoot->findEnv(EnvironmentHandle);
+ if (pEnv == 0) {
+ driver_exit(SQL_API_SQLFREEENV);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pRoot->sqlFreeEnv(ctx, pEnv);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ if (! ctx.ok()) {
+ pEnv->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLFREEENV);
+ return ret;
+ }
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ delete &ctx;
+ driver_exit(SQL_API_SQLFREEENV);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLFreeHandle.cpp b/ndb/src/client/odbc/driver/SQLFreeHandle.cpp
new file mode 100644
index 00000000000..284463cbb07
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLFreeHandle.cpp
@@ -0,0 +1,60 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLFreeHandle(
+ SQLSMALLINT HandleType,
+ SQLHANDLE Handle)
+{
+ driver_enter(SQL_API_SQLFREEHANDLE);
+ const char* const sqlFunction = "SQLFreeHandle";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ SQLSMALLINT parentType = pRoot->findParentType(HandleType);
+ if (parentType == -1) {
+ driver_exit(SQL_API_SQLFREEHANDLE);
+ return SQL_INVALID_HANDLE;
+ }
+ HandleBase* pChild = pRoot->findBase(HandleType, Handle);
+ if (pChild == 0) {
+ driver_exit(SQL_API_SQLFREEHANDLE);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ HandleBase* pParent = pChild->getParent();
+ ctx_assert(pParent != 0);
+ try {
+ pParent->sqlFreeHandle(ctx, HandleType, pChild);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ if (! ctx.ok()) {
+ pChild->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLFREEHANDLE);
+ return ret;
+ }
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ delete &ctx;
+ driver_exit(SQL_API_SQLFREEHANDLE);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLFreeStmt.cpp b/ndb/src/client/odbc/driver/SQLFreeStmt.cpp
new file mode 100644
index 00000000000..7af6623a37a
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLFreeStmt.cpp
@@ -0,0 +1,54 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLFreeStmt(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT Option)
+{
+ driver_enter(SQL_API_SQLFREESTMT);
+ const char* const sqlFunction = "SQLFreeStmt";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLFREESTMT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ HandleDbc* pDbc = pStmt->getDbc();
+ try {
+ pDbc->sqlFreeStmt(ctx, pStmt, Option);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ if (! ctx.ok() || Option != SQL_DROP) {
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLFREESTMT);
+ return ret;
+ }
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ delete &ctx;
+ driver_exit(SQL_API_SQLFREESTMT);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLGetConnectAttr.cpp b/ndb/src/client/odbc/driver/SQLGetConnectAttr.cpp
new file mode 100644
index 00000000000..66c1f3827e1
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetConnectAttr.cpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLGetConnectAttr(
+ SQLHDBC ConnectionHandle,
+ SQLINTEGER Attribute,
+ SQLPOINTER Value,
+ SQLINTEGER BufferLength,
+ SQLINTEGER* StringLength)
+{
+ driver_enter(SQL_API_SQLGETCONNECTATTR);
+ const char* const sqlFunction = "SQLGetConnectAttr";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLGETCONNECTATTR);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlGetConnectAttr(ctx, Attribute, Value, BufferLength, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETCONNECTATTR);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLGetConnectOption.cpp b/ndb/src/client/odbc/driver/SQLGetConnectOption.cpp
new file mode 100644
index 00000000000..514bedb12b9
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetConnectOption.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLGetConnectOption(
+ SQLHDBC ConnectionHandle,
+ SQLUSMALLINT Option,
+ SQLPOINTER Value)
+{
+ driver_enter(SQL_API_SQLGETCONNECTOPTION);
+ const char* const sqlFunction = "SQLGetConnectOption";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLGETCONNECTOPTION);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlGetConnectOption(ctx, Option, Value);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETCONNECTOPTION);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLGetCursorName.cpp b/ndb/src/client/odbc/driver/SQLGetCursorName.cpp
new file mode 100644
index 00000000000..d54bdf42005
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetCursorName.cpp
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLGetCursorName(
+ SQLHSTMT StatementHandle,
+ SQLCHAR* CursorName,
+ SQLSMALLINT BufferLength,
+ SQLSMALLINT* NameLength)
+{
+ driver_enter(SQL_API_SQLGETCURSORNAME);
+ const char* const sqlFunction = "SQLGetCursorName";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLGETCURSORNAME);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlGetCursorName(ctx, CursorName, BufferLength, NameLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETCURSORNAME);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLGetData.cpp b/ndb/src/client/odbc/driver/SQLGetData.cpp
new file mode 100644
index 00000000000..3b6987c515d
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetData.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLGetData(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT ColumnNumber,
+ SQLSMALLINT TargetType,
+ SQLPOINTER TargetValue,
+ SQLINTEGER BufferLength,
+ SQLINTEGER* StrLen_or_Ind)
+{
+ driver_enter(SQL_API_SQLGETDATA);
+ const char* const sqlFunction = "SQLGetData";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLGETDATA);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlGetData(ctx, ColumnNumber, TargetType, TargetValue, BufferLength, StrLen_or_Ind);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETDATA);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLGetDescField.cpp b/ndb/src/client/odbc/driver/SQLGetDescField.cpp
new file mode 100644
index 00000000000..6cc390a58ed
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetDescField.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLGetDescField(
+ SQLHDESC DescriptorHandle,
+ SQLSMALLINT RecNumber,
+ SQLSMALLINT FieldIdentifier,
+ SQLPOINTER Value,
+ SQLINTEGER BufferLength,
+ SQLINTEGER* StringLength)
+{
+ driver_enter(SQL_API_SQLGETDESCFIELD);
+ const char* const sqlFunction = "SQLGetDescField";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDesc* pDesc = pRoot->findDesc(DescriptorHandle);
+ if (pDesc == 0) {
+ driver_exit(SQL_API_SQLGETDESCFIELD);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDesc->sqlGetDescField(ctx, RecNumber, FieldIdentifier, Value, BufferLength, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDesc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETDESCFIELD);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLGetDescRec.cpp b/ndb/src/client/odbc/driver/SQLGetDescRec.cpp
new file mode 100644
index 00000000000..c7e9631b075
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetDescRec.cpp
@@ -0,0 +1,55 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLGetDescRec(
+ SQLHDESC DescriptorHandle,
+ SQLSMALLINT RecNumber,
+ SQLCHAR* Name,
+ SQLSMALLINT BufferLength,
+ SQLSMALLINT* StringLength,
+ SQLSMALLINT* Type,
+ SQLSMALLINT* SubType,
+ SQLINTEGER* Length,
+ SQLSMALLINT* Precision,
+ SQLSMALLINT* Scale,
+ SQLSMALLINT* Nullable)
+{
+ driver_enter(SQL_API_SQLGETDESCREC);
+ const char* const sqlFunction = "SQLGetDescRec";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDesc* pDesc = pRoot->findDesc(DescriptorHandle);
+ if (pDesc == 0) {
+ driver_exit(SQL_API_SQLGETDESCREC);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDesc->sqlGetDescRec(ctx, RecNumber, Name, BufferLength, StringLength, Type, SubType, Length, Precision, Scale, Nullable);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDesc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETDESCREC);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLGetDiagField.cpp b/ndb/src/client/odbc/driver/SQLGetDiagField.cpp
new file mode 100644
index 00000000000..3eb34f7ebf6
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetDiagField.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLGetDiagField(
+ SQLSMALLINT HandleType,
+ SQLHANDLE Handle,
+ SQLSMALLINT RecNumber,
+ SQLSMALLINT DiagIdentifier,
+ SQLPOINTER DiagInfo,
+ SQLSMALLINT BufferLength,
+ SQLSMALLINT* StringLength)
+{
+ driver_enter(SQL_API_SQLGETDIAGFIELD);
+ const char* const sqlFunction = "SQLGetDiagField";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleBase* pBase = pRoot->findBase(HandleType, Handle);
+ if (pBase == 0) {
+ driver_exit(SQL_API_SQLGETDIAGFIELD);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx ctx; // discard diagnostics
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pBase->sqlGetDiagField(ctx, RecNumber, DiagIdentifier, DiagInfo, BufferLength, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETDIAGFIELD);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLGetDiagRec.cpp b/ndb/src/client/odbc/driver/SQLGetDiagRec.cpp
new file mode 100644
index 00000000000..448c5206d76
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetDiagRec.cpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLGetDiagRec(
+ SQLSMALLINT HandleType,
+ SQLHANDLE Handle,
+ SQLSMALLINT RecNumber,
+ SQLCHAR* Sqlstate,
+ SQLINTEGER* NativeError,
+ SQLCHAR* MessageText,
+ SQLSMALLINT BufferLength,
+ SQLSMALLINT* TextLength)
+{
+ driver_enter(SQL_API_SQLGETDIAGREC);
+ const char* const sqlFunction = "SQLGetDiagRec";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleBase* pBase = pRoot->findBase(HandleType, Handle);
+ if (pBase == 0) {
+ driver_exit(SQL_API_SQLGETDIAGREC);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx ctx; // discard diagnostics
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pBase->sqlGetDiagRec(ctx, RecNumber, Sqlstate, NativeError, MessageText, BufferLength, TextLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETDIAGREC);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLGetEnvAttr.cpp b/ndb/src/client/odbc/driver/SQLGetEnvAttr.cpp
new file mode 100644
index 00000000000..c93870326e4
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetEnvAttr.cpp
@@ -0,0 +1,66 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLGetEnvAttr(
+ SQLHENV EnvironmentHandle,
+ SQLINTEGER Attribute,
+ SQLPOINTER Value,
+ SQLINTEGER BufferLength,
+ SQLINTEGER* StringLength)
+{
+ driver_enter(SQL_API_SQLGETENVATTR);
+ const char* const sqlFunction = "SQLGetEnvAttr";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ if (EnvironmentHandle == 0) {
+ // process-level attributes
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pRoot->sqlGetRootAttr(ctx, Attribute, Value, BufferLength, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pRoot->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETENVATTR);
+ return ret;
+ } else {
+ HandleEnv* pEnv = pRoot->findEnv(EnvironmentHandle);
+ if (pEnv == 0) {
+ driver_exit(SQL_API_SQLGETENVATTR);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pEnv->sqlGetEnvAttr(ctx, Attribute, Value, BufferLength, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pEnv->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETENVATTR);
+ return ret;
+ }
+ return SQL_ERROR; // not reached
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLGetFunctions.cpp b/ndb/src/client/odbc/driver/SQLGetFunctions.cpp
new file mode 100644
index 00000000000..68416fab1a6
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetFunctions.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLGetFunctions(
+ SQLHDBC ConnectionHandle,
+ SQLUSMALLINT FunctionId,
+ SQLUSMALLINT* Supported)
+{
+ driver_enter(SQL_API_SQLGETFUNCTIONS);
+ const char* const sqlFunction = "SQLGetFunctions";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLGETFUNCTIONS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlGetFunctions(ctx, FunctionId, Supported);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETFUNCTIONS);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLGetInfo.cpp b/ndb/src/client/odbc/driver/SQLGetInfo.cpp
new file mode 100644
index 00000000000..8f0a0d67cfa
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetInfo.cpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLGetInfo(
+ SQLHDBC ConnectionHandle,
+ SQLUSMALLINT InfoType,
+ SQLPOINTER InfoValue,
+ SQLSMALLINT BufferLength,
+ SQLSMALLINT* StringLength)
+{
+ driver_enter(SQL_API_SQLGETINFO);
+ const char* const sqlFunction = "SQLGetInfo";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLGETINFO);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlGetInfo(ctx, InfoType, InfoValue, BufferLength, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETINFO);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLGetStmtAttr.cpp b/ndb/src/client/odbc/driver/SQLGetStmtAttr.cpp
new file mode 100644
index 00000000000..990ab68808a
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetStmtAttr.cpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLGetStmtAttr(
+ SQLHSTMT StatementHandle,
+ SQLINTEGER Attribute,
+ SQLPOINTER Value,
+ SQLINTEGER BufferLength,
+ SQLINTEGER* StringLength)
+{
+ driver_enter(SQL_API_SQLGETSTMTATTR);
+ const char* const sqlFunction = "SQLGetStmtAttr";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLGETSTMTATTR);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlGetStmtAttr(ctx, Attribute, Value, BufferLength, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETSTMTATTR);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLGetStmtOption.cpp b/ndb/src/client/odbc/driver/SQLGetStmtOption.cpp
new file mode 100644
index 00000000000..0b5758b1212
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetStmtOption.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLGetStmtOption(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT Option,
+ SQLPOINTER Value)
+{
+ driver_enter(SQL_API_SQLGETSTMTOPTION);
+ const char* const sqlFunction = "SQLGetStmtOption";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLGETSTMTOPTION);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlGetStmtOption(ctx, Option, Value);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETSTMTOPTION);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLGetTypeInfo.cpp b/ndb/src/client/odbc/driver/SQLGetTypeInfo.cpp
new file mode 100644
index 00000000000..e6a016cc400
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLGetTypeInfo.cpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLGetTypeInfo(
+ SQLHSTMT StatementHandle,
+ SQLSMALLINT DataType)
+{
+ driver_enter(SQL_API_SQLGETTYPEINFO);
+ const char* const sqlFunction = "SQLGetTypeInfo";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLGETTYPEINFO);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlGetTypeInfo(ctx, DataType);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLGETTYPEINFO);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLMoreResults.cpp b/ndb/src/client/odbc/driver/SQLMoreResults.cpp
new file mode 100644
index 00000000000..d23d653a319
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLMoreResults.cpp
@@ -0,0 +1,42 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLMoreResults(
+ SQLHSTMT hstmt)
+{
+ driver_enter(SQL_API_SQLMORERESULTS);
+ const char* const sqlFunction = "SQLMoreResults";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLMORERESULTS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlMoreResults(ctx);
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLMORERESULTS);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLNativeSql.cpp b/ndb/src/client/odbc/driver/SQLNativeSql.cpp
new file mode 100644
index 00000000000..fb8a9bbf3d9
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLNativeSql.cpp
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLNativeSql(
+ SQLHDBC hdbc,
+ SQLCHAR* szSqlStrIn,
+ SQLINTEGER cbSqlStrIn,
+ SQLCHAR* szSqlStr,
+ SQLINTEGER cbSqlStrMax,
+ SQLINTEGER* pcbSqlStr)
+{
+#ifndef auto_SQLNativeSql
+ const char* const sqlFunction = "SQLNativeSql";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLNATIVESQL);
+ const char* const sqlFunction = "SQLNativeSql";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(hdbc);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLNATIVESQL);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pDbc->sqlNativeSql(
+ ctx,
+ &szSqlStrIn,
+ cbSqlStrIn,
+ &szSqlStr,
+ cbSqlStrMax,
+ &pcbSqlStr
+ );
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLNATIVESQL);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLNumParams.cpp b/ndb/src/client/odbc/driver/SQLNumParams.cpp
new file mode 100644
index 00000000000..7b1a6a07aec
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLNumParams.cpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLNumParams(
+ SQLHSTMT StatementHandle,
+ SQLSMALLINT* ParameterCountPtr)
+{
+ driver_enter(SQL_API_SQLNUMPARAMS);
+ const char* const sqlFunction = "SQLNumParams";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLNUMPARAMS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlNumParams(ctx, ParameterCountPtr);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLNUMPARAMS);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLNumResultCols.cpp b/ndb/src/client/odbc/driver/SQLNumResultCols.cpp
new file mode 100644
index 00000000000..2e70897a9a2
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLNumResultCols.cpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLNumResultCols(
+ SQLHSTMT StatementHandle,
+ SQLSMALLINT* ColumnCount)
+{
+ driver_enter(SQL_API_SQLNUMRESULTCOLS);
+ const char* const sqlFunction = "SQLNumResultCols";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLNUMRESULTCOLS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlNumResultCols(ctx, ColumnCount);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLNUMRESULTCOLS);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLParamData.cpp b/ndb/src/client/odbc/driver/SQLParamData.cpp
new file mode 100644
index 00000000000..4eb38a010f4
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLParamData.cpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLParamData(
+ SQLHSTMT StatementHandle,
+ SQLPOINTER* Value)
+{
+ driver_enter(SQL_API_SQLPARAMDATA);
+ const char* const sqlFunction = "SQLParamData";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLPARAMDATA);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlParamData(ctx, Value);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLPARAMDATA);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLParamOptions.cpp b/ndb/src/client/odbc/driver/SQLParamOptions.cpp
new file mode 100644
index 00000000000..59b7dcf7fa9
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLParamOptions.cpp
@@ -0,0 +1,55 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLParamOptions(
+ SQLHSTMT hstmt,
+ SQLUINTEGER crow,
+ SQLUINTEGER* pirow)
+{
+#ifndef auto_SQLParamOptions
+ const char* const sqlFunction = "SQLParamOptions";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLPARAMOPTIONS);
+ const char* const sqlFunction = "SQLParamOptions";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLPARAMOPTIONS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlParamOptions(
+ ctx,
+ crow,
+ &pirow
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLPARAMOPTIONS);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLPrepare.cpp b/ndb/src/client/odbc/driver/SQLPrepare.cpp
new file mode 100644
index 00000000000..b1205fa6e3a
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLPrepare.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLPrepare(
+ SQLHSTMT StatementHandle,
+ SQLCHAR* StatementText,
+ SQLINTEGER TextLength)
+{
+ driver_enter(SQL_API_SQLPREPARE);
+ const char* const sqlFunction = "SQLPrepare";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLPREPARE);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlPrepare(ctx, StatementText, TextLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLPREPARE);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLPrimaryKeys.cpp b/ndb/src/client/odbc/driver/SQLPrimaryKeys.cpp
new file mode 100644
index 00000000000..2d562ae3e19
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLPrimaryKeys.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLPrimaryKeys(
+ SQLHSTMT hstmt,
+ SQLCHAR* szCatalogName,
+ SQLSMALLINT cbCatalogName,
+ SQLCHAR* szSchemaName,
+ SQLSMALLINT cbSchemaName,
+ SQLCHAR* szTableName,
+ SQLSMALLINT cbTableName)
+{
+ driver_enter(SQL_API_SQLPRIMARYKEYS);
+ const char* const sqlFunction = "SQLPrimaryKeys";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLPRIMARYKEYS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ pStmt->sqlPrimaryKeys(ctx, szCatalogName, cbCatalogName, szSchemaName, cbSchemaName, szTableName, cbTableName);
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLPRIMARYKEYS);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLProcedureColumns.cpp b/ndb/src/client/odbc/driver/SQLProcedureColumns.cpp
new file mode 100644
index 00000000000..2e42e428b87
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLProcedureColumns.cpp
@@ -0,0 +1,67 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLProcedureColumns(
+ SQLHSTMT hstmt,
+ SQLCHAR* szCatalogName,
+ SQLSMALLINT cbCatalogName,
+ SQLCHAR* szSchemaName,
+ SQLSMALLINT cbSchemaName,
+ SQLCHAR* szProcName,
+ SQLSMALLINT cbProcName,
+ SQLCHAR* szColumnName,
+ SQLSMALLINT cbColumnName)
+{
+#ifndef auto_SQLProcedureColumns
+ const char* const sqlFunction = "SQLProcedureColumns";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLPROCEDURECOLUMNS);
+ const char* const sqlFunction = "SQLProcedureColumns";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLPROCEDURECOLUMNS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlProcedureColumns(
+ ctx,
+ &szCatalogName,
+ cbCatalogName,
+ &szSchemaName,
+ cbSchemaName,
+ &szProcName,
+ cbProcName,
+ &szColumnName,
+ cbColumnName
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLPROCEDURECOLUMNS);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLProcedures.cpp b/ndb/src/client/odbc/driver/SQLProcedures.cpp
new file mode 100644
index 00000000000..1f3a9f89073
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLProcedures.cpp
@@ -0,0 +1,63 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLProcedures(
+ SQLHSTMT hstmt,
+ SQLCHAR* szCatalogName,
+ SQLSMALLINT cbCatalogName,
+ SQLCHAR* szSchemaName,
+ SQLSMALLINT cbSchemaName,
+ SQLCHAR* szProcName,
+ SQLSMALLINT cbProcName)
+{
+#ifndef auto_SQLProcedures
+ const char* const sqlFunction = "SQLProcedures";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLPROCEDURES);
+ const char* const sqlFunction = "SQLProcedures";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLPROCEDURES);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlProcedures(
+ ctx,
+ &szCatalogName,
+ cbCatalogName,
+ &szSchemaName,
+ cbSchemaName,
+ &szProcName,
+ cbProcName
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLPROCEDURES);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLPutData.cpp b/ndb/src/client/odbc/driver/SQLPutData.cpp
new file mode 100644
index 00000000000..a4715a836d2
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLPutData.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLPutData(
+ SQLHSTMT StatementHandle,
+ SQLPOINTER Data,
+ SQLINTEGER StrLen_or_Ind)
+{
+ driver_enter(SQL_API_SQLPUTDATA);
+ const char* const sqlFunction = "SQLPutData";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLPUTDATA);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlPutData(ctx, Data, StrLen_or_Ind);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLPUTDATA);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLRowCount.cpp b/ndb/src/client/odbc/driver/SQLRowCount.cpp
new file mode 100644
index 00000000000..d03f954386a
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLRowCount.cpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLRowCount(
+ SQLHSTMT StatementHandle,
+ SQLINTEGER* RowCount)
+{
+ driver_enter(SQL_API_SQLROWCOUNT);
+ const char* const sqlFunction = "SQLRowCount";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLROWCOUNT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlRowCount(ctx, RowCount);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLROWCOUNT);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLSetConnectAttr.cpp b/ndb/src/client/odbc/driver/SQLSetConnectAttr.cpp
new file mode 100644
index 00000000000..05bfce5e9cd
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetConnectAttr.cpp
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLSetConnectAttr(
+ SQLHDBC ConnectionHandle,
+ SQLINTEGER Attribute,
+ SQLPOINTER Value,
+ SQLINTEGER StringLength)
+{
+ driver_enter(SQL_API_SQLSETCONNECTATTR);
+ const char* const sqlFunction = "SQLSetConnectAttr";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLSETCONNECTATTR);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlSetConnectAttr(ctx, Attribute, Value, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETCONNECTATTR);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLSetConnectOption.cpp b/ndb/src/client/odbc/driver/SQLSetConnectOption.cpp
new file mode 100644
index 00000000000..a4794316971
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetConnectOption.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLSetConnectOption(
+ SQLHDBC ConnectionHandle,
+ SQLUSMALLINT Option,
+ SQLUINTEGER Value)
+{
+ driver_enter(SQL_API_SQLSETCONNECTOPTION);
+ const char* const sqlFunction = "SQLSetConnectOption";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLSETCONNECTOPTION);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlSetConnectOption(ctx, Option, Value);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETCONNECTOPTION);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLSetCursorName.cpp b/ndb/src/client/odbc/driver/SQLSetCursorName.cpp
new file mode 100644
index 00000000000..291ad817d42
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetCursorName.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLSetCursorName(
+ SQLHSTMT StatementHandle,
+ SQLCHAR* CursorName,
+ SQLSMALLINT NameLength)
+{
+ driver_enter(SQL_API_SQLSETCURSORNAME);
+ const char* const sqlFunction = "SQLSetCursorName";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLSETCURSORNAME);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlSetCursorName(ctx, CursorName, NameLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETCURSORNAME);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLSetDescField.cpp b/ndb/src/client/odbc/driver/SQLSetDescField.cpp
new file mode 100644
index 00000000000..19d34c2f46d
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetDescField.cpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLSetDescField(
+ SQLHDESC DescriptorHandle,
+ SQLSMALLINT RecNumber,
+ SQLSMALLINT FieldIdentifier,
+ SQLPOINTER Value,
+ SQLINTEGER BufferLength)
+{
+ driver_enter(SQL_API_SQLSETDESCFIELD);
+ const char* const sqlFunction = "SQLSetDescField";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDesc* pDesc = pRoot->findDesc(DescriptorHandle);
+ if (pDesc == 0) {
+ driver_exit(SQL_API_SQLSETDESCFIELD);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDesc->sqlSetDescField(ctx, RecNumber, FieldIdentifier, Value, BufferLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDesc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETDESCFIELD);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLSetDescRec.cpp b/ndb/src/client/odbc/driver/SQLSetDescRec.cpp
new file mode 100644
index 00000000000..80a00514a51
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetDescRec.cpp
@@ -0,0 +1,54 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLSetDescRec(
+ SQLHDESC DescriptorHandle,
+ SQLSMALLINT RecNumber,
+ SQLSMALLINT Type,
+ SQLSMALLINT SubType,
+ SQLINTEGER Length,
+ SQLSMALLINT Precision,
+ SQLSMALLINT Scale,
+ SQLPOINTER Data,
+ SQLINTEGER* StringLength,
+ SQLINTEGER* Indicator)
+{
+ driver_enter(SQL_API_SQLSETDESCREC);
+ const char* const sqlFunction = "SQLSetDescRec";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleDesc* pDesc = pRoot->findDesc(DescriptorHandle);
+ if (pDesc == 0) {
+ driver_exit(SQL_API_SQLSETDESCREC);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDesc->sqlSetDescRec(ctx, RecNumber, Type, SubType, Length, Precision, Scale, Data, StringLength, Indicator);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDesc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETDESCREC);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLSetEnvAttr.cpp b/ndb/src/client/odbc/driver/SQLSetEnvAttr.cpp
new file mode 100644
index 00000000000..86364eac5e8
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetEnvAttr.cpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLSetEnvAttr(
+ SQLHENV EnvironmentHandle,
+ SQLINTEGER Attribute,
+ SQLPOINTER Value,
+ SQLINTEGER StringLength)
+{
+ driver_enter(SQL_API_SQLSETENVATTR);
+ const char* const sqlFunction = "SQLSetEnvAttr";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ if (EnvironmentHandle == 0) {
+ // process-level attributes
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pRoot->sqlSetRootAttr(ctx, Attribute, Value, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pRoot->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETENVATTR);
+ return ret;
+ } else {
+ HandleEnv* pEnv = pRoot->findEnv(EnvironmentHandle);
+ if (pEnv == 0) {
+ driver_exit(SQL_API_SQLSETENVATTR);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pEnv->sqlSetEnvAttr(ctx, Attribute, Value, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pEnv->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETENVATTR);
+ return ret;
+ }
+ return SQL_ERROR; // not reached
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLSetParam.cpp b/ndb/src/client/odbc/driver/SQLSetParam.cpp
new file mode 100644
index 00000000000..03bde1076d8
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetParam.cpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLSetParam(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT ParameterNumber,
+ SQLSMALLINT ValueType,
+ SQLSMALLINT ParameterType,
+ SQLUINTEGER LengthPrecision,
+ SQLSMALLINT ParameterScale,
+ SQLPOINTER ParameterValue,
+ SQLINTEGER* StrLen_or_Ind)
+{
+ driver_enter(SQL_API_SQLSETPARAM);
+ const char* const sqlFunction = "SQLSetParam";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLSETPARAM);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlSetParam(ctx, ParameterNumber, ValueType, ParameterType, LengthPrecision, ParameterScale, ParameterValue, StrLen_or_Ind);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETPARAM);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLSetPos.cpp b/ndb/src/client/odbc/driver/SQLSetPos.cpp
new file mode 100644
index 00000000000..653030f90bc
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetPos.cpp
@@ -0,0 +1,57 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLSetPos(
+ SQLHSTMT hstmt,
+ SQLUSMALLINT irow,
+ SQLUSMALLINT fOption,
+ SQLUSMALLINT fLock)
+{
+#ifndef auto_SQLSetPos
+ const char* const sqlFunction = "SQLSetPos";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLSETPOS);
+ const char* const sqlFunction = "SQLSetPos";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLSETPOS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlSetPos(
+ ctx,
+ irow,
+ fOption,
+ fLock
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETPOS);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLSetScrollOptions.cpp b/ndb/src/client/odbc/driver/SQLSetScrollOptions.cpp
new file mode 100644
index 00000000000..a5e89d8568b
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetScrollOptions.cpp
@@ -0,0 +1,57 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLSetScrollOptions(
+ SQLHSTMT hstmt,
+ SQLUSMALLINT fConcurrency,
+ SQLINTEGER crowKeyset,
+ SQLUSMALLINT crowRowset)
+{
+#ifndef auto_SQLSetScrollOptions
+ const char* const sqlFunction = "SQLSetScrollOptions";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLSETSCROLLOPTIONS);
+ const char* const sqlFunction = "SQLSetScrollOptions";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLSETSCROLLOPTIONS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlSetScrollOptions(
+ ctx,
+ fConcurrency,
+ crowKeyset,
+ crowRowset
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETSCROLLOPTIONS);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLSetStmtAttr.cpp b/ndb/src/client/odbc/driver/SQLSetStmtAttr.cpp
new file mode 100644
index 00000000000..9ed6a83b563
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetStmtAttr.cpp
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0300
+SQLRETURN SQL_API
+SQLSetStmtAttr(
+ SQLHSTMT StatementHandle,
+ SQLINTEGER Attribute,
+ SQLPOINTER Value,
+ SQLINTEGER StringLength)
+{
+ driver_enter(SQL_API_SQLSETSTMTATTR);
+ const char* const sqlFunction = "SQLSetStmtAttr";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLSETSTMTATTR);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlSetStmtAttr(ctx, Attribute, Value, StringLength);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETSTMTATTR);
+ return ret;
+}
+#endif // ODBCVER >= 0x0300
diff --git a/ndb/src/client/odbc/driver/SQLSetStmtOption.cpp b/ndb/src/client/odbc/driver/SQLSetStmtOption.cpp
new file mode 100644
index 00000000000..b403fc8408c
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSetStmtOption.cpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLSetStmtOption(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT Option,
+ SQLUINTEGER Value)
+{
+ driver_enter(SQL_API_SQLSETSTMTOPTION);
+ const char* const sqlFunction = "SQLSetStmtOption";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLSETSTMTOPTION);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pStmt->sqlSetStmtOption(ctx, Option, Value);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSETSTMTOPTION);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLSpecialColumns.cpp b/ndb/src/client/odbc/driver/SQLSpecialColumns.cpp
new file mode 100644
index 00000000000..5dd92c86053
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLSpecialColumns.cpp
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLSpecialColumns(
+ SQLHSTMT StatementHandle,
+ SQLUSMALLINT IdentifierType,
+ SQLCHAR* CatalogName,
+ SQLSMALLINT NameLength1,
+ SQLCHAR* SchemaName,
+ SQLSMALLINT NameLength2,
+ SQLCHAR* TableName,
+ SQLSMALLINT NameLength3,
+ SQLUSMALLINT Scope,
+ SQLUSMALLINT Nullable)
+{
+#ifndef auto_SQLSpecialColumns
+ const char* const sqlFunction = "SQLSpecialColumns";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLSPECIALCOLUMNS);
+ const char* const sqlFunction = "SQLSpecialColumns";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLSPECIALCOLUMNS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlSpecialColumns(
+ ctx,
+ IdentifierType,
+ &CatalogName,
+ NameLength1,
+ &SchemaName,
+ NameLength2,
+ &TableName,
+ NameLength3,
+ Scope,
+ Nullable
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSPECIALCOLUMNS);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLStatistics.cpp b/ndb/src/client/odbc/driver/SQLStatistics.cpp
new file mode 100644
index 00000000000..941fb6249a5
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLStatistics.cpp
@@ -0,0 +1,67 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLStatistics(
+ SQLHSTMT StatementHandle,
+ SQLCHAR* CatalogName,
+ SQLSMALLINT NameLength1,
+ SQLCHAR* SchemaName,
+ SQLSMALLINT NameLength2,
+ SQLCHAR* TableName,
+ SQLSMALLINT NameLength3,
+ SQLUSMALLINT Unique,
+ SQLUSMALLINT Reserved)
+{
+#ifndef auto_SQLStatistics
+ const char* const sqlFunction = "SQLStatistics";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLSTATISTICS);
+ const char* const sqlFunction = "SQLStatistics";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLSTATISTICS);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlStatistics(
+ ctx,
+ &CatalogName,
+ NameLength1,
+ &SchemaName,
+ NameLength2,
+ &TableName,
+ NameLength3,
+ Unique,
+ Reserved
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLSTATISTICS);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLTablePrivileges.cpp b/ndb/src/client/odbc/driver/SQLTablePrivileges.cpp
new file mode 100644
index 00000000000..23c6ad9fc4b
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLTablePrivileges.cpp
@@ -0,0 +1,63 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLTablePrivileges(
+ SQLHSTMT hstmt,
+ SQLCHAR* szCatalogName,
+ SQLSMALLINT cbCatalogName,
+ SQLCHAR* szSchemaName,
+ SQLSMALLINT cbSchemaName,
+ SQLCHAR* szTableName,
+ SQLSMALLINT cbTableName)
+{
+#ifndef auto_SQLTablePrivileges
+ const char* const sqlFunction = "SQLTablePrivileges";
+ Ctx ctx;
+ ctx_log1(("*** not implemented: %s", sqlFunction));
+ return SQL_ERROR;
+#else
+ driver_enter(SQL_API_SQLTABLEPRIVILEGES);
+ const char* const sqlFunction = "SQLTablePrivileges";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(hstmt);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLTABLEPRIVILEGES);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ if (ctx.ok())
+ pStmt->sqlTablePrivileges(
+ ctx,
+ &szCatalogName,
+ cbCatalogName,
+ &szSchemaName,
+ cbSchemaName,
+ &szTableName,
+ cbTableName
+ );
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLTABLEPRIVILEGES);
+ return ret;
+#endif
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLTables.cpp b/ndb/src/client/odbc/driver/SQLTables.cpp
new file mode 100644
index 00000000000..b2496bfba87
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLTables.cpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLTables(
+ SQLHSTMT StatementHandle,
+ SQLCHAR* CatalogName,
+ SQLSMALLINT NameLength1,
+ SQLCHAR* SchemaName,
+ SQLSMALLINT NameLength2,
+ SQLCHAR* TableName,
+ SQLSMALLINT NameLength3,
+ SQLCHAR* TableType,
+ SQLSMALLINT NameLength4)
+{
+ driver_enter(SQL_API_SQLTABLES);
+ const char* const sqlFunction = "SQLTables";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ HandleStmt* pStmt = pRoot->findStmt(StatementHandle);
+ if (pStmt == 0) {
+ driver_exit(SQL_API_SQLTABLES);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ pStmt->sqlTables(ctx, CatalogName, NameLength1, SchemaName, NameLength2, TableName, NameLength3, TableType, NameLength4);
+ pStmt->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLTABLES);
+ return ret;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/SQLTransact.cpp b/ndb/src/client/odbc/driver/SQLTransact.cpp
new file mode 100644
index 00000000000..da8b46b1596
--- /dev/null
+++ b/ndb/src/client/odbc/driver/SQLTransact.cpp
@@ -0,0 +1,71 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "driver.hpp"
+
+#if ODBCVER >= 0x0000
+SQLRETURN SQL_API
+SQLTransact(
+ SQLHENV EnvironmentHandle,
+ SQLHDBC ConnectionHandle,
+ SQLUSMALLINT CompletionType)
+{
+ driver_enter(SQL_API_SQLTRANSACT);
+ const char* const sqlFunction = "SQLTransact";
+ HandleRoot* const pRoot = HandleRoot::instance();
+ // check connection first and ignore environment
+ if (ConnectionHandle != SQL_NULL_HANDLE) {
+ HandleDbc* pDbc = pRoot->findDbc(ConnectionHandle);
+ if (pDbc == 0) {
+ driver_exit(SQL_API_SQLTRANSACT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pDbc->sqlTransact(ctx, CompletionType);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pDbc->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLTRANSACT);
+ return ret;
+ }
+ if (EnvironmentHandle != SQL_NULL_HANDLE) {
+ HandleEnv* pEnv = pRoot->findEnv(EnvironmentHandle);
+ if (pEnv == 0) {
+ driver_exit(SQL_API_SQLTRANSACT);
+ return SQL_INVALID_HANDLE;
+ }
+ Ctx& ctx = *new Ctx;
+ ctx.logSqlEnter(sqlFunction);
+ try {
+ pEnv->sqlTransact(ctx, CompletionType);
+ } catch (CtxAssert& ctxAssert) {
+ ctx.handleEx(ctxAssert);
+ }
+ pEnv->saveCtx(ctx);
+ ctx.logSqlExit();
+ SQLRETURN ret = ctx.getCode();
+ driver_exit(SQL_API_SQLTRANSACT);
+ return ret;
+ }
+ driver_exit(SQL_API_SQLTRANSACT);
+ return SQL_INVALID_HANDLE;
+}
+#endif // ODBCVER >= 0x0000
diff --git a/ndb/src/client/odbc/driver/driver.cpp b/ndb/src/client/odbc/driver/driver.cpp
new file mode 100644
index 00000000000..dabfd5f855b
--- /dev/null
+++ b/ndb/src/client/odbc/driver/driver.cpp
@@ -0,0 +1,150 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbMutex.h>
+#include "driver.hpp"
+
+#undef NDB_ODBC_SIG_DFL
+#ifdef NDB_ODBC_SIG_DFL
+#include <signal.h>
+#endif
+
+// The big mutex (just in case).
+
+#ifdef NDB_WIN32
+static NdbMutex & driver_mutex = * NdbMutex_Create();
+#else
+static NdbMutex driver_mutex = NDB_MUTEX_INITIALIZER;
+#endif
+
+static void
+driver_lock()
+{
+ NdbMutex_Lock(&driver_mutex);
+}
+
+static void
+driver_unlock()
+{
+ NdbMutex_Unlock(&driver_mutex);
+}
+
+// Hooks for function entry and exit.
+
+static inline void
+driver_enter(SQLUSMALLINT functionId)
+{
+ switch (functionId) {
+ default:
+ break;
+ }
+#ifdef NDB_ODBC_SIG_DFL
+ // XXX need to restore old sig
+ for (int i = 1; i <= 30; i++)
+ signal(i, SIG_DFL);
+#endif
+}
+
+static inline void
+driver_exit(SQLUSMALLINT functionId)
+{
+ switch (functionId) {
+ default:
+ break;
+ }
+}
+
+// Some C++ compilers (like gcc) cannot merge template code
+// in different files. So compile all in one file.
+
+#include "SQLAllocConnect.cpp"
+#include "SQLAllocEnv.cpp"
+#include "SQLAllocHandle.cpp"
+#include "SQLAllocHandleStd.cpp"
+#include "SQLAllocStmt.cpp"
+#include "SQLBindCol.cpp"
+#include "SQLBindParam.cpp"
+#include "SQLBindParameter.cpp"
+#include "SQLBrowseConnect.cpp"
+#include "SQLBulkOperations.cpp"
+#include "SQLCancel.cpp"
+#include "SQLCloseCursor.cpp"
+#include "SQLColAttribute.cpp"
+#include "SQLColAttributes.cpp"
+#include "SQLColumnPrivileges.cpp"
+#include "SQLColumns.cpp"
+#include "SQLConnect.cpp"
+#include "SQLCopyDesc.cpp"
+#include "SQLDataSources.cpp"
+#include "SQLDescribeCol.cpp"
+#include "SQLDescribeParam.cpp"
+#include "SQLDisconnect.cpp"
+#include "SQLDriverConnect.cpp"
+#include "SQLDrivers.cpp"
+#include "SQLEndTran.cpp"
+#include "SQLError.cpp"
+#include "SQLExecDirect.cpp"
+#include "SQLExecute.cpp"
+#include "SQLExtendedFetch.cpp"
+#include "SQLFetch.cpp"
+#include "SQLFetchScroll.cpp"
+#include "SQLForeignKeys.cpp"
+#include "SQLFreeConnect.cpp"
+#include "SQLFreeEnv.cpp"
+#include "SQLFreeHandle.cpp"
+#include "SQLFreeStmt.cpp"
+#include "SQLGetConnectAttr.cpp"
+#include "SQLGetConnectOption.cpp"
+#include "SQLGetCursorName.cpp"
+#include "SQLGetData.cpp"
+#include "SQLGetDescField.cpp"
+#include "SQLGetDescRec.cpp"
+#include "SQLGetDiagField.cpp"
+#include "SQLGetDiagRec.cpp"
+#include "SQLGetEnvAttr.cpp"
+#include "SQLGetFunctions.cpp"
+#include "SQLGetInfo.cpp"
+#include "SQLGetStmtAttr.cpp"
+#include "SQLGetStmtOption.cpp"
+#include "SQLGetTypeInfo.cpp"
+#include "SQLMoreResults.cpp"
+#include "SQLNativeSql.cpp"
+#include "SQLNumParams.cpp"
+#include "SQLNumResultCols.cpp"
+#include "SQLParamData.cpp"
+#include "SQLParamOptions.cpp"
+#include "SQLPrepare.cpp"
+#include "SQLPrimaryKeys.cpp"
+#include "SQLProcedureColumns.cpp"
+#include "SQLProcedures.cpp"
+#include "SQLPutData.cpp"
+#include "SQLRowCount.cpp"
+#include "SQLSetConnectAttr.cpp"
+#include "SQLSetConnectOption.cpp"
+#include "SQLSetCursorName.cpp"
+#include "SQLSetDescField.cpp"
+#include "SQLSetDescRec.cpp"
+#include "SQLSetEnvAttr.cpp"
+#include "SQLSetParam.cpp"
+#include "SQLSetPos.cpp"
+#include "SQLSetScrollOptions.cpp"
+#include "SQLSetStmtAttr.cpp"
+#include "SQLSetStmtOption.cpp"
+#include "SQLSpecialColumns.cpp"
+#include "SQLStatistics.cpp"
+#include "SQLTablePrivileges.cpp"
+#include "SQLTables.cpp"
+#include "SQLTransact.cpp"
diff --git a/ndb/src/client/odbc/driver/driver.hpp b/ndb/src/client/odbc/driver/driver.hpp
new file mode 100644
index 00000000000..96d2e052c0d
--- /dev/null
+++ b/ndb/src/client/odbc/driver/driver.hpp
@@ -0,0 +1,28 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_DRIVER_driver_hpp
+#define ODBC_DRIVER_driver_hpp
+
+#include <common/common.hpp>
+#include <common/OdbcData.hpp>
+#include <handles/handles.hpp>
+
+#ifndef NDB_WIN32
+#define SQL_API
+#endif
+
+#endif
diff --git a/ndb/src/client/odbc/executor/Exec_comp_op.cpp b/ndb/src/client/odbc/executor/Exec_comp_op.cpp
new file mode 100644
index 00000000000..40d3950a592
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_comp_op.cpp
@@ -0,0 +1,518 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbScanFilter.hpp>
+#include <NdbSqlUtil.hpp>
+#include <codegen/Code_comp_op.hpp>
+
+void
+Exec_comp_op::execInterp(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ const unsigned arity = code.m_op.arity();
+ const Comp_op::Opcode opcode = code.m_op.m_opcode;
+ Data& data = getData();
+ ctx_assert(ctl.m_scanFilter != 0);
+ NdbScanFilter& scanFilter = *ctl.m_scanFilter;
+ if (code.m_interpColumn == 0) {
+ // args are constant on this level so evaluate entire predicate
+ evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ if (data.m_value == Pred_value_true)
+ scanFilter.istrue();
+ else
+ scanFilter.isfalse();
+ return;
+ }
+ const NdbAttrId interpAttrId = code.m_interpAttrId;
+ if (arity == 1) {
+ ctx_assert(m_expr[1] != 0);
+ const SqlType& t1 = m_expr[1]->getCode().sqlSpec().sqlType();
+ const SqlField& f1 = m_expr[1]->getData().sqlField();
+ switch (code.m_op.m_opcode) {
+ case Comp_op::Isnull:
+ scanFilter.isnull(interpAttrId);
+ break;
+ case Comp_op::Isnotnull:
+ scanFilter.isnotnull(interpAttrId);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else if (arity == 2) {
+ ctx_assert(m_expr[1] != 0 && m_expr[2] != 0);
+ // one is column and the other is constant at this level
+ ctx_assert(code.m_interpColumn == 1 || code.m_interpColumn == 2);
+ const unsigned i = code.m_interpColumn;
+ const unsigned j = 3 - i;
+ // evaluate the constant
+ m_expr[j]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ const SqlType& t1 = m_expr[i]->getCode().sqlSpec().sqlType();
+ const SqlField& f1 = m_expr[i]->getData().sqlField();
+ const SqlType& t2 = m_expr[j]->getCode().sqlSpec().sqlType();
+ const SqlField& f2 = m_expr[j]->getData().sqlField();
+ // handle null constant
+ if (f2.sqlNull()) {
+ scanFilter.isfalse();
+ return;
+ }
+ // handle null in interpreter
+ scanFilter.begin(NdbScanFilter::AND);
+ scanFilter.isnotnull(interpAttrId);
+ if (t1.type() == SqlType::Char || t1.type() == SqlType::Varchar) {
+ const char* v2 = 0;
+ unsigned n2 = 0;
+ bool nopad = false;
+ if (t1.type() == SqlType::Char && t2.type() == SqlType::Char) {
+ v2 = reinterpret_cast<const char*>(f2.sqlChar());
+ n2 = t2.length();
+ nopad = false;
+ } else if (t1.type() == SqlType::Char && t2.type() == SqlType::Varchar) {
+ v2 = reinterpret_cast<const char*>(f2.sqlVarchar(&n2));
+ nopad = true;
+ } else if (t1.type() == SqlType::Varchar && t2.type() == SqlType::Char) {
+ v2 = reinterpret_cast<const char*>(f2.sqlChar());
+ n2 = t2.length();
+ nopad = true;
+ } else if (t1.type() == SqlType::Varchar && t2.type() == SqlType::Varchar) {
+ v2 = reinterpret_cast<const char*>(f2.sqlVarchar(&n2));
+ nopad = true;
+ } else {
+ ctx_assert(false);
+ }
+ switch (opcode) {
+ case Comp_op::Eq:
+ scanFilter.eq(interpAttrId, v2, n2, nopad);
+ break;
+ case Comp_op::Noteq:
+ scanFilter.ne(interpAttrId, v2, n2, nopad);
+ break;
+ case Comp_op::Lt:
+ if (i == 1) {
+ scanFilter.lt(interpAttrId, v2, n2, nopad);
+ } else {
+ scanFilter.gt(interpAttrId, v2, n2, nopad);
+ }
+ break;
+ case Comp_op::Lteq:
+ if (i == 1) {
+ scanFilter.le(interpAttrId, v2, n2, nopad);
+ } else {
+ scanFilter.ge(interpAttrId, v2, n2, nopad);
+ }
+ break;
+ case Comp_op::Gt:
+ if (i == 1) {
+ scanFilter.gt(interpAttrId, v2, n2, nopad);
+ } else {
+ scanFilter.lt(interpAttrId, v2, n2, nopad);
+ }
+ break;
+ case Comp_op::Gteq:
+ if (i == 1) {
+ scanFilter.ge(interpAttrId, v2, n2, nopad);
+ } else {
+ scanFilter.le(interpAttrId, v2, n2, nopad);
+ }
+ break;
+ case Comp_op::Like:
+ scanFilter.like(interpAttrId, v2, n2, nopad);
+ break;
+ case Comp_op::Notlike:
+ scanFilter.notlike(interpAttrId, v2, n2, nopad);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else if (t1.type() == SqlType::Smallint || t1.type() == SqlType::Integer || t1.type() == SqlType::Bigint) {
+ ctx_assert(t1.unSigned());
+ bool s2 = ! t2.unSigned();
+ SqlBigint v2;
+ SqlUbigint uv2;
+ if (s2) {
+ v2 =
+ t2.type() == SqlType::Smallint ? f2.sqlSmallint() :
+ t2.type() == SqlType::Integer ? f2.sqlInteger() : f2.sqlBigint();
+ uv2 = v2;
+ } else {
+ uv2 =
+ t2.type() == SqlType::Smallint ? (SqlUsmallint)f2.sqlSmallint() :
+ t2.type() == SqlType::Integer ? (SqlUinteger)f2.sqlInteger() : (SqlUbigint)f2.sqlBigint();
+ v2 = uv2;
+ }
+ switch (code.m_op.m_opcode) {
+ case Comp_op::Eq:
+ if (s2 && v2 < 0)
+ scanFilter.isfalse();
+ else
+ scanFilter.eq(interpAttrId, uv2);
+ break;
+ case Comp_op::Noteq:
+ if (s2 && v2 < 0)
+ scanFilter.istrue();
+ else
+ scanFilter.ne(interpAttrId, uv2);
+ break;
+ case Comp_op::Lt:
+ if (i == 1) {
+ if (s2 && v2 < 0)
+ scanFilter.isfalse();
+ else
+ scanFilter.lt(interpAttrId, uv2);
+ } else {
+ if (s2 && v2 < 0)
+ scanFilter.istrue();
+ else
+ scanFilter.gt(interpAttrId, uv2);
+ }
+ break;
+ case Comp_op::Lteq:
+ if (i == 1) {
+ if (s2 && v2 < 0)
+ scanFilter.isfalse();
+ else
+ scanFilter.le(interpAttrId, uv2);
+ } else {
+ if (s2 && v2 < 0)
+ scanFilter.istrue();
+ else
+ scanFilter.ge(interpAttrId, uv2);
+ }
+ break;
+ case Comp_op::Gt:
+ if (i == 1) {
+ if (s2 && v2 < 0)
+ scanFilter.istrue();
+ else
+ scanFilter.gt(interpAttrId, uv2);
+ } else {
+ if (s2 && v2 < 0)
+ scanFilter.isfalse();
+ else
+ scanFilter.lt(interpAttrId, uv2);
+ }
+ break;
+ case Comp_op::Gteq:
+ if (i == 1) {
+ if (s2 && v2 < 0)
+ scanFilter.istrue();
+ else
+ scanFilter.ge(interpAttrId, uv2);
+ } else {
+ if (s2 && v2 < 0)
+ scanFilter.isfalse();
+ else
+ scanFilter.le(interpAttrId, uv2);
+ }
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else {
+ ctx_assert(false);
+ }
+ // end null guard
+ scanFilter.end();
+ } else {
+ ctx_assert(false);
+ }
+}
+
+static bool
+do_sqlchar_comp(Comp_op::Opcode opcode, const SqlChar* s1, unsigned n1, const SqlChar* s2, unsigned n2, bool padded)
+{
+ int ret = NdbSqlUtil::char_compare(reinterpret_cast<const char*>(s1), n1, reinterpret_cast<const char *>(s2), n2, padded);
+ switch (opcode) {
+ case Comp_op::Eq:
+ return ret == 0;
+ case Comp_op::Noteq:
+ return ret != 0;
+ case Comp_op::Lt:
+ return ret < 0;
+ case Comp_op::Lteq:
+ return ret <= 0;
+ case Comp_op::Gt:
+ return ret > 0;
+ case Comp_op::Gteq:
+ return ret >= 0;
+ default:
+ break;
+ }
+ ctx_assert(false);
+ return false;
+}
+
+static bool
+do_sqlchar_like(const SqlChar* s1, unsigned n1, const SqlChar* s2, unsigned n2, bool padded)
+{
+ bool ret = NdbSqlUtil::char_like(reinterpret_cast<const char*>(s1), n1, reinterpret_cast<const char *>(s2), n2, padded);
+ return ret;
+}
+
+static bool
+do_datetime_comp(Comp_op::Opcode opcode, SqlDatetime v1, SqlDatetime v2)
+{
+ int k = v1.less(v2) ? -1 : v2.less(v1) ? 1 : 0;
+ switch (opcode) {
+ case Comp_op::Eq:
+ return k == 0;
+ case Comp_op::Noteq:
+ return k != 0;
+ case Comp_op::Lt:
+ return k < 0;
+ case Comp_op::Lteq:
+ return k <= 0;
+ case Comp_op::Gt:
+ return k > 0;
+ case Comp_op::Gteq:
+ return k >= 0;
+ default:
+ break;
+ }
+ ctx_assert(false);
+ return false;
+}
+
+void
+Exec_comp_op::evaluate(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ const unsigned arity = code.m_op.arity();
+ const Comp_op::Opcode opcode = code.m_op.m_opcode;
+ Data& data = getData();
+ Pred_value v = Pred_value_unknown;
+ if (arity == 1) {
+ // evaluate sub-expression
+ ctx_assert(m_expr[1] != 0);
+ m_expr[1]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ if (ctl.m_postEval)
+ return;
+ // get type and value
+ const SqlType& t1 = m_expr[1]->getCode().sqlSpec().sqlType();
+ const SqlField& f1 = ctl.m_groupIndex == 0 ? m_expr[1]->getData().sqlField() : m_expr[1]->getData().groupField(ctl.m_groupIndex);
+ switch (code.m_op.m_opcode) {
+ case Comp_op::Isnull:
+ v = f1.sqlNull() ? Pred_value_true : Pred_value_false;
+ break;
+ case Comp_op::Isnotnull:
+ v = f1.sqlNull() ? Pred_value_false : Pred_value_true;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else if (arity == 2) {
+ // evaluate sub-expressions
+ ctx_assert(m_expr[1] != 0 && m_expr[2] != 0);
+ m_expr[1]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ m_expr[2]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ if (ctl.m_postEval)
+ return;
+ // get types and values
+ const SqlType& t1 = m_expr[1]->getCode().sqlSpec().sqlType();
+ const SqlType& t2 = m_expr[2]->getCode().sqlSpec().sqlType();
+ const SqlField& f1 = ctl.m_groupIndex == 0 ? m_expr[1]->getData().sqlField() : m_expr[1]->getData().groupField(ctl.m_groupIndex);
+ const SqlField& f2 = ctl.m_groupIndex == 0 ? m_expr[2]->getData().sqlField() : m_expr[2]->getData().groupField(ctl.m_groupIndex);
+ // handle null
+ if (f1.sqlNull() || f2.sqlNull()) {
+ v = Pred_value_unknown;
+ } else if (t1.type() == SqlType::Char) {
+ const SqlChar* v1 = f1.sqlChar();
+ unsigned n1 = t1.length();
+ if (t2.type() == SqlType::Char) {
+ unsigned n2 = t2.length();
+ const SqlChar* v2 = f2.sqlChar();
+ bool b;
+ switch (opcode) {
+ case Comp_op::Like:
+ b = do_sqlchar_like(v1, n1, v2, n2, true);
+ break;
+ case Comp_op::Notlike:
+ b = ! do_sqlchar_like(v1, n1, v2, n2, true);
+ break;
+ default:
+ b = do_sqlchar_comp(opcode, v1, n1, v2, n2, true);
+ break;
+ }
+ v = b ? Pred_value_true : Pred_value_false;
+ } else if (t2.type() == SqlType::Varchar) {
+ unsigned n2 = 0;
+ const SqlChar* v2 = f2.sqlVarchar(&n2);
+ bool b;
+ switch (opcode) {
+ case Comp_op::Like:
+ b = do_sqlchar_like(v1, n1, v2, n2, true);
+ break;
+ case Comp_op::Notlike:
+ b = ! do_sqlchar_like(v1, n1, v2, n2, true);
+ break;
+ default:
+ b = do_sqlchar_comp(opcode, v1, n1, v2, n2, false);
+ break;
+ }
+ v = b ? Pred_value_true : Pred_value_false;
+ } else {
+ ctx_assert(false);
+ }
+ } else if (t1.type() == SqlType::Varchar) {
+ unsigned n1 = 0;
+ const SqlChar* v1 = f1.sqlVarchar(&n1);
+ if (t2.type() == SqlType::Char) {
+ unsigned n2 = t2.length();
+ const SqlChar* v2 = f2.sqlChar();
+ bool b;
+ switch (opcode) {
+ case Comp_op::Like:
+ b = do_sqlchar_like(v1, n1, v2, n2, false);
+ break;
+ case Comp_op::Notlike:
+ b = ! do_sqlchar_like(v1, n1, v2, n2, false);
+ break;
+ default:
+ b = do_sqlchar_comp(opcode, v1, n1, v2, n2, false);
+ break;
+ }
+ v = b ? Pred_value_true : Pred_value_false;
+ } else if (t2.type() == SqlType::Varchar) {
+ unsigned n2 = 0;
+ const SqlChar* v2 = f2.sqlVarchar(&n2);
+ bool b;
+ switch (opcode) {
+ case Comp_op::Like:
+ b = do_sqlchar_like(v1, n1, v2, n2, false);
+ break;
+ case Comp_op::Notlike:
+ b = ! do_sqlchar_like(v1, n1, v2, n2, false);
+ break;
+ default:
+ b = do_sqlchar_comp(opcode, v1, n1, v2, n2, false);
+ break;
+ }
+ v = b ? Pred_value_true : Pred_value_false;
+ } else {
+ ctx_assert(false);
+ }
+ } else if (t1.type() == SqlType::Smallint || t1.type() == SqlType::Integer || t1.type() == SqlType::Bigint) {
+ // convert to bigint
+ bool s1 = ! t1.unSigned();
+ bool s2 = ! t2.unSigned();
+ SqlBigint v1, v2;
+ SqlUbigint uv1, uv2;
+ if (s1) {
+ v1 =
+ t1.type() == SqlType::Smallint ? f1.sqlSmallint() :
+ t1.type() == SqlType::Integer ? f1.sqlInteger() : f1.sqlBigint();
+ uv1 = v1;
+ } else {
+ uv1 =
+ t1.type() == SqlType::Smallint ? (SqlUsmallint)f1.sqlSmallint() :
+ t1.type() == SqlType::Integer ? (SqlUinteger)f1.sqlInteger() : (SqlUbigint)f1.sqlBigint();
+ v1 = uv1;
+ }
+ if (s2) {
+ v2 =
+ t2.type() == SqlType::Smallint ? f2.sqlSmallint() :
+ t2.type() == SqlType::Integer ? f2.sqlInteger() : f2.sqlBigint();
+ uv2 = v2;
+ } else {
+ uv2 =
+ t2.type() == SqlType::Smallint ? (SqlUsmallint)f2.sqlSmallint() :
+ t2.type() == SqlType::Integer ? (SqlUinteger)f2.sqlInteger() : (SqlUbigint)f2.sqlBigint();
+ v2 = uv2;
+ }
+ bool b;
+ switch (opcode) {
+ case Comp_op::Eq:
+ b = s1 && s2 ? (v1 == v2) : s1 ? (v1 < 0 ? false : uv1 == uv2) : s2 ? (v2 < 0 ? false : uv1 == uv2) : (uv1 == uv2);
+ break;
+ case Comp_op::Noteq:
+ b = s1 && s2 ? (v1 == v2) : s1 ? (v1 < 0 ? true : uv1 != uv2) : s2 ? (v2 < 0 ? true : uv1 != uv2) : (uv1 != uv2);
+ break;
+ case Comp_op::Lt:
+ b = s1 && s2 ? (v1 < v2) : s1 ? (v1 < 0 ? true : uv1 < uv2) : s2 ? (v2 < 0 ? false : uv1 < uv2) : (uv1 < uv2);
+ break;
+ case Comp_op::Lteq:
+ b = s1 && s2 ? (v1 <= v2) : s1 ? (v1 < 0 ? true : uv1 <= uv2) : s2 ? (v2 < 0 ? false : uv1 <= uv2) : (uv1 <= uv2);
+ break;
+ case Comp_op::Gt:
+ b = s1 && s2 ? (v1 > v2) : s1 ? (v1 < 0 ? false : uv1 > uv2) : s2 ? (v2 < 0 ? true : uv1 > uv2) : (uv1 > uv2);
+ break;
+ case Comp_op::Gteq:
+ b = s1 && s2 ? (v1 >= v2) : s1 ? (v1 < 0 ? false : uv1 >= uv2) : s2 ? (v2 < 0 ? true : uv1 >= uv2) : (uv1 >= uv2);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ v = b ? Pred_value_true : Pred_value_false;
+ } else if (t1.type() == SqlType::Double) {
+ SqlDouble v1 = f1.sqlDouble();
+ SqlDouble v2 = f2.sqlDouble();
+ bool b;
+ switch (opcode) {
+ case Comp_op::Eq:
+ b = (v1 == v2);
+ break;
+ case Comp_op::Noteq:
+ b = (v1 != v2);
+ break;
+ case Comp_op::Lt:
+ b = (v1 < v2);
+ break;
+ case Comp_op::Lteq:
+ b = (v1 <= v2);
+ break;
+ case Comp_op::Gt:
+ b = (v1 > v2);
+ break;
+ case Comp_op::Gteq:
+ b = (v1 >= v2);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ v = b ? Pred_value_true : Pred_value_false;
+ } else if (t1.type() == SqlType::Datetime) {
+ SqlDatetime v1 = f1.sqlDatetime();
+ SqlDatetime v2 = f2.sqlDatetime();
+ bool b;
+ b = do_datetime_comp(opcode, v1, v2);
+ v = b ? Pred_value_true : Pred_value_false;
+ } else {
+ ctx_assert(false);
+ }
+ } else {
+ ctx_assert(false);
+ }
+ // set result
+ if (ctl.m_groupIndex == 0)
+ data.m_value = v;
+ else
+ data.groupValue(ctl.m_groupIndex, ctl.m_groupInit) = v;
+}
diff --git a/ndb/src/client/odbc/executor/Exec_create_index.cpp b/ndb/src/client/odbc/executor/Exec_create_index.cpp
new file mode 100644
index 00000000000..3966c6d5db2
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_create_index.cpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <dictionary/DictSchema.hpp>
+#include <codegen/Code_create_index.hpp>
+
+void
+Exec_create_index::execute(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbDictionary::Dictionary* ndbDictionary = ndb->getDictionary();
+ if (ndbDictionary == 0) {
+ ctx.pushStatus(ndb, "getDictionary");
+ return;
+ }
+ NdbDictionary::Index ndbIndex(code.m_indexName.c_str());
+ ndbIndex.setTable(code.m_tableName.c_str());
+ ndbIndex.setType((NdbDictionary::Index::Type)code.m_type);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ ndbIndex.addIndexColumn(code.m_attrList[i]);
+ }
+ // setting fragment type not supported in ndb api
+ ndbIndex.setLogging(code.m_logging);
+ dictSchema().deleteTable(ctx, code.m_tableName);
+ if (ndbDictionary->createIndex(ndbIndex) == -1) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "createIndex %s", ndbIndex.getName());
+ return;
+ }
+ ctx_log1(("index %s on %s created", ndbIndex.getName(), ndbIndex.getTable()));
+}
diff --git a/ndb/src/client/odbc/executor/Exec_create_table.cpp b/ndb/src/client/odbc/executor/Exec_create_table.cpp
new file mode 100644
index 00000000000..d6274119371
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_create_table.cpp
@@ -0,0 +1,83 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <dictionary/DictSchema.hpp>
+#include <codegen/Code_create_table.hpp>
+
+void
+Exec_create_table::execute(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbDictionary::Dictionary* ndbDictionary = ndb->getDictionary();
+ if (ndbDictionary == 0) {
+ ctx.pushStatus(ndb, "getDictionary");
+ return;
+ }
+ NdbDictionary::Table ndbTable(code.m_tableName.c_str());
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ const Code::Attr& attr = code.m_attrList[i];
+ NdbDictionary::Column ndbColumn(attr.m_attrName.c_str());
+ if (i == code.m_tupleId)
+ ndbColumn.setTupleKey(true); // XXX setTupleId()
+ if (ctx.logLevel() >= 3) {
+ char buf[100];
+ attr.m_sqlType.print(buf, sizeof(buf));
+ ctx_log3(("attr %s type %s", ndbColumn.getName(), buf));
+ }
+ if (attr.m_tupleKey)
+ ndbColumn.setPrimaryKey(true);
+ attr.m_sqlType.getType(ctx, &ndbColumn);
+ if (! ctx.ok())
+ return;
+ if (attr.m_autoIncrement)
+ ndbColumn.setAutoIncrement(true);
+ char defaultValue[MAX_ATTR_DEFAULT_VALUE_SIZE];
+ defaultValue[0] = 0;
+ if (attr.m_defaultValue != 0) {
+ // XXX obviously should evalute it at insert time too
+ attr.m_defaultValue->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ const SqlField& f = attr.m_defaultValue->getData().sqlField();
+ const SqlType& t = f.sqlSpec().sqlType();
+ // XXX use SqlField cast method instead
+ SQLINTEGER ind = 0;
+ ExtType extType(ExtType::Char);
+ ExtSpec extSpec(extType);
+ ExtField extField(extSpec, (SQLPOINTER)defaultValue, sizeof(defaultValue), &ind);
+ f.copyout(ctx, extField);
+ if (! ctx.ok())
+ return;
+ if (ind == SQL_NULL_DATA) // do not store NULL default
+ defaultValue[0] = 0;
+ }
+ if (defaultValue[0] != 0)
+ ndbColumn.setDefaultValue(defaultValue);
+ ndbTable.addColumn(ndbColumn);
+ }
+ if (code.m_fragmentType != NdbDictionary::Object::FragUndefined)
+ ndbTable.setFragmentType(code.m_fragmentType);
+ ndbTable.setLogging(code.m_logging);
+ dictSchema().deleteTable(ctx, code.m_tableName);
+ if (ndbDictionary->createTable(ndbTable) == -1) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "createTable %s", ndbTable.getName());
+ return;
+ }
+ ctx_log1(("table %s created", ndbTable.getName()));
+}
diff --git a/ndb/src/client/odbc/executor/Exec_delete_index.cpp b/ndb/src/client/odbc/executor/Exec_delete_index.cpp
new file mode 100644
index 00000000000..10814654a58
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_delete_index.cpp
@@ -0,0 +1,82 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_expr.hpp>
+#include <codegen/Code_delete_index.hpp>
+#include <codegen/Code_query.hpp>
+
+void
+Exec_delete_index::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ // execute subquery
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // delete each row from the query
+ data.setCount(0);
+ while (m_query->fetch(ctx, ctl)) {
+ NdbOperation* op = tcon->getNdbIndexOperation(code.m_indexName, code.m_tableName);
+ if (op == 0) {
+ ctx.pushStatus(ndb, tcon, 0, "getIndexNdbOperation");
+ return;
+ }
+ if (op->deleteTuple() == -1) {
+ ctx.pushStatus(ndb, tcon, op, "deleteTuple");
+ return;
+ }
+ bool done = false;
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* exprMatch = code.m_keyMatch[k];
+ ctx_assert(exprMatch != 0);
+ exprMatch->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ const SqlField& keyMatch = exprMatch->getData().sqlField();
+ SqlField f(code.m_keySpecs.getEntry(k));
+ if (! keyMatch.cast(ctx, f)) {
+ done = true; // match is not possible
+ break;
+ }
+ const NdbAttrId keyId = code.m_keyId[k];
+ const void* addr = f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (op->equal(keyId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "equal attrId=%u", (unsigned)keyId);
+ return;
+ }
+ }
+ if (done)
+ continue;
+ if (tcon->execute(NoCommit) == -1) {
+ // XXX when did 626 move to connection level
+ if (tcon->getNdbError().code != 626 && op->getNdbError().code != 626) {
+ ctx.pushStatus(ndb, tcon, op, "execute without commit");
+ return;
+ }
+ } else {
+ data.addCount();
+ }
+ }
+ stmtArea().setRowCount(ctx, data.getCount());
+}
diff --git a/ndb/src/client/odbc/executor/Exec_delete_lookup.cpp b/ndb/src/client/odbc/executor/Exec_delete_lookup.cpp
new file mode 100644
index 00000000000..d0795286122
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_delete_lookup.cpp
@@ -0,0 +1,82 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_expr.hpp>
+#include <codegen/Code_delete_lookup.hpp>
+#include <codegen/Code_query.hpp>
+
+void
+Exec_delete_lookup::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ // execute subquery
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // delete each row from the query
+ data.setCount(0);
+ while (m_query->fetch(ctx, ctl)) {
+ NdbOperation* op = tcon->getNdbOperation(code.m_tableName);
+ if (op == 0) {
+ ctx.pushStatus(ndb, tcon, 0, "getNdbOperation");
+ return;
+ }
+ if (op->deleteTuple() == -1) {
+ ctx.pushStatus(ndb, tcon, op, "deleteTuple");
+ return;
+ }
+ bool done = false;
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* exprMatch = code.m_keyMatch[k];
+ ctx_assert(exprMatch != 0);
+ exprMatch->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ const SqlField& keyMatch = exprMatch->getData().sqlField();
+ SqlField f(code.m_keySpecs.getEntry(k));
+ if (! keyMatch.cast(ctx, f)) {
+ done = true; // match is not possible
+ break;
+ }
+ const NdbAttrId keyId = code.m_keyId[k];
+ const void* addr = f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (op->equal(keyId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "equal attrId=%u", (unsigned)keyId);
+ return;
+ }
+ }
+ if (done)
+ continue;
+ if (tcon->execute(NoCommit) == -1) {
+ // XXX when did 626 move to connection level
+ if (tcon->getNdbError().code != 626 && op->getNdbError().code != 626) {
+ ctx.pushStatus(ndb, tcon, op, "execute without commit");
+ return;
+ }
+ } else {
+ data.addCount();
+ }
+ }
+ stmtArea().setRowCount(ctx, data.getCount());
+}
diff --git a/ndb/src/client/odbc/executor/Exec_delete_scan.cpp b/ndb/src/client/odbc/executor/Exec_delete_scan.cpp
new file mode 100644
index 00000000000..a0b3b8314b8
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_delete_scan.cpp
@@ -0,0 +1,54 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_delete_scan.hpp>
+#include <codegen/Code_query.hpp>
+
+void
+Exec_delete_scan::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ // execute subquery
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ ctx_assert(ctl.m_scanOp != 0);
+ // delete each row from the scan
+ data.setCount(0);
+ while (m_query->fetch(ctx, ctl)) {
+ NdbOperation* toOp = ctl.m_scanOp->takeOverForDelete(tcon);
+ if (toOp == 0) {
+ ctx.pushStatus(ndb, tcon, ctl.m_scanOp, "takeOverScanOp");
+ return;
+ }
+ if (tcon->execute(NoCommit) == -1) {
+ if (toOp->getNdbError().code != 626) {
+ ctx.pushStatus(ndb, tcon, toOp, "execute without commit");
+ return;
+ }
+ } else {
+ data.addCount();
+ }
+ }
+ stmtArea().setRowCount(ctx, data.getCount());
+}
diff --git a/ndb/src/client/odbc/executor/Exec_drop_index.cpp b/ndb/src/client/odbc/executor/Exec_drop_index.cpp
new file mode 100644
index 00000000000..6bf451f6911
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_drop_index.cpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <dictionary/DictSchema.hpp>
+#include <codegen/Code_drop_index.hpp>
+
+void
+Exec_drop_index::execute(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbDictionary::Dictionary* ndbDictionary = ndb->getDictionary();
+ if (ndbDictionary == 0) {
+ ctx.pushStatus(ndb, "getDictionary");
+ return;
+ }
+ dictSchema().deleteTableByIndex(ctx, code.m_indexName);
+ if (ndbDictionary->dropIndex(code.m_indexName.c_str(), code.m_tableName.c_str()) == -1) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "dropIndex %s on %s", code.m_indexName.c_str(), code.m_tableName.c_str());
+ return;
+ }
+ ctx_log1(("index %s on %s dropped", code.m_indexName.c_str(), code.m_tableName.c_str()));
+}
diff --git a/ndb/src/client/odbc/executor/Exec_drop_table.cpp b/ndb/src/client/odbc/executor/Exec_drop_table.cpp
new file mode 100644
index 00000000000..40d1d42fc61
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_drop_table.cpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <dictionary/DictSchema.hpp>
+#include <codegen/Code_drop_table.hpp>
+
+void
+Exec_drop_table::execute(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbDictionary::Dictionary* ndbDictionary = ndb->getDictionary();
+ if (ndbDictionary == 0) {
+ ctx.pushStatus(ndb, "getDictionary");
+ return;
+ }
+ dictSchema().deleteTable(ctx, code.m_tableName);
+ if (ndbDictionary->dropTable(code.m_tableName.c_str()) == -1) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "dropTable %s", code.m_tableName.c_str());
+ return;
+ }
+ ctx_log1(("table %s dropped", code.m_tableName.c_str()));
+}
diff --git a/ndb/src/client/odbc/executor/Exec_expr_conv.cpp b/ndb/src/client/odbc/executor/Exec_expr_conv.cpp
new file mode 100644
index 00000000000..636bfda7d59
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_expr_conv.cpp
@@ -0,0 +1,54 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <codegen/Code_expr_conv.hpp>
+
+void
+Exec_expr_conv::evaluate(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ const SqlType& t1 = code.sqlSpec().sqlType();
+ SqlField& f1 = ctl.m_groupIndex == 0 ? data.m_sqlField : data.groupField(code.sqlSpec().sqlType() , ctl.m_groupIndex, ctl.m_groupInit);
+ // evaluate the subexpression
+ ctx_assert(m_expr != 0);
+ m_expr->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ if (ctl.m_postEval)
+ return;
+ const SqlType& t2 = m_expr->getCode().sqlSpec().sqlType();
+ const SqlField& f2 = ctl.m_groupIndex == 0 ? m_expr->getData().sqlField() : m_expr->getData().groupField(ctl.m_groupIndex);
+ // conversion to null type
+ if (t1.type() == SqlType::Null) {
+ f1.sqlNull(true);
+ return;
+ }
+ // conversion of null data
+ if (f2.sqlNull()) {
+ f1.sqlNull(true);
+ return;
+ }
+ // try to convert
+ if (! f2.cast(ctx, f1)) {
+ char b1[40], b2[40], b3[40];
+ t1.print(b1, sizeof(b1));
+ t2.print(b2, sizeof(b2));
+ f2.print(b3, sizeof(b3));
+ ctx.pushStatus(Sqlstate::_22003, Error::Gen, "cannot convert %s [%s] to %s", b2, b3, b1);
+ return;
+ }
+}
diff --git a/ndb/src/client/odbc/executor/Exec_expr_func.cpp b/ndb/src/client/odbc/executor/Exec_expr_func.cpp
new file mode 100644
index 00000000000..093d15c6e2b
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_expr_func.cpp
@@ -0,0 +1,284 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <time.h>
+#include <NdbTick.h>
+#include <codegen/Code_expr_func.hpp>
+
+void
+Exec_expr_func::init(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ const SqlType& t = code.sqlSpec().sqlType();
+ SqlField& f = ctl.m_groupIndex == 0 ? data.m_sqlField : data.groupField(code.sqlSpec().sqlType(), ctl.m_groupIndex, ctl.m_groupInit);
+ if (ctl.m_groupIndex >= data.m_groupAcc.size())
+ data.m_groupAcc.resize(1 + ctl.m_groupIndex);
+ Data::Acc& acc = data.m_groupAcc[ctl.m_groupIndex];
+ acc.m_count = 0;
+ Expr_func::Code fc = code.m_func.m_code;
+ if (fc == Expr_func::Substr || fc == Expr_func::Left || fc == Expr_func::Right) {
+ } else if (fc == Expr_func::Count) {
+ f.sqlBigint(0);
+ } else if (fc == Expr_func::Min || fc == Expr_func::Max) {
+ f.sqlNull(true);
+ } else if (fc == Expr_func::Sum) {
+ f.sqlNull(true);
+ switch (t.type()) {
+ case SqlType::Bigint:
+ acc.m_bigint = 0;
+ break;
+ case SqlType::Double:
+ acc.m_double = 0;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else if (fc == Expr_func::Avg) {
+ f.sqlNull(true);
+ switch (t.type()) {
+ case SqlType::Double:
+ acc.m_double = 0.0;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else if (fc == Expr_func::Rownum) {
+ // uses only m_count
+ } else if (fc == Expr_func::Sysdate) {
+ // set time once
+ NDB_TICKS secs = 0;
+ Uint32 micros = 0;
+ NdbTick_CurrentMicrosecond(&secs, &micros);
+ time_t clock = secs;
+ struct tm* t = gmtime(&clock);
+ SqlDatetime& d = acc.m_sysdate;
+ d.cc((1900 + t->tm_year) / 100);
+ d.yy((1900 + t->tm_year) % 100);
+ d.mm(1 + t->tm_mon);
+ d.dd(t->tm_mday);
+ d.HH(t->tm_hour);
+ d.MM(t->tm_min);
+ d.SS(t->tm_sec);
+ d.ff(1000 * micros);
+ } else {
+ ctx_assert(false);
+ }
+}
+
+void
+Exec_expr_func::evaluate(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ const SqlType& t = code.sqlSpec().sqlType();
+ if (ctl.m_groupInit)
+ init(ctx, ctl);
+ SqlField& f = ctl.m_groupIndex == 0 ? data.m_sqlField : data.groupField(code.sqlSpec().sqlType(), ctl.m_groupIndex, false);
+ Data::Acc& acc = data.m_groupAcc[ctl.m_groupIndex];
+ Expr_func::Code fc = code.m_func.m_code;
+ const unsigned narg = code.m_narg;
+ Exec_expr** args = code.m_args;
+ ctx_assert(args != 0);
+ // evaluate arguments
+ for (unsigned i = 1; i <= narg; i++) {
+ ctx_assert(args[i] != 0);
+ unsigned save = ctl.m_groupIndex;
+ if (code.m_func.m_aggr)
+ ctl.m_groupIndex = 0;
+ args[i]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ ctl.m_groupIndex = save;
+ }
+ if (fc == Expr_func::Substr || fc == Expr_func::Left || fc == Expr_func::Right) {
+ ctx_assert((narg == (unsigned)2) || (narg == (unsigned)(fc == Expr_func::Substr ? 3 : 2)));
+ const SqlType& t1 = args[1]->getCode().sqlSpec().sqlType();
+ const SqlField& f1 = args[1]->getData().sqlField();
+ int pos, len;
+ for (unsigned i = 2; i <= narg; i++) {
+ int& num = (fc == Expr_func::Substr ? (i == 2 ? pos : len) : len);
+ const SqlType& t2 = args[i]->getCode().sqlSpec().sqlType();
+ const SqlField& f2 = args[i]->getData().sqlField();
+ switch (t2.type()) {
+ case SqlType::Smallint:
+ num = static_cast<int>(f2.sqlSmallint());
+ break;
+ case SqlType::Integer:
+ num = static_cast<int>(f2.sqlInteger());
+ break;
+ case SqlType::Bigint:
+ num = static_cast<int>(f2.sqlBigint());
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ }
+ int length = 0;
+ const SqlChar* data = 0;
+ switch (t1.type()) {
+ case SqlType::Char:
+ length = t1.length();
+ data = f1.sqlChar();
+ break;
+ case SqlType::Varchar:
+ unsigned ulength;
+ data = f1.sqlVarchar(&ulength);
+ length = ulength;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ if (fc == Expr_func::Left)
+ pos = 1;
+ else if (fc == Expr_func::Right)
+ pos = len > length ? 1 : length - len + 1;
+ else if (pos < 0)
+ pos += length + 1;
+ if (pos <= 0 || pos > length || len <= 0) {
+ f.sqlNull(true); // oracle-ish
+ return;
+ }
+ if (len > length - pos + 1)
+ len = length - pos + 1;
+ switch (t1.type()) {
+ case SqlType::Char:
+ f.sqlChar(data + (pos - 1), len);
+ break;
+ case SqlType::Varchar:
+ f.sqlVarchar(data + (pos - 1), len);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else if (fc == Expr_func::Count) {
+ ctx_assert(narg == 0 || narg == 1);
+ if (ctl.m_postEval)
+ return;
+ if (narg == 1) {
+ const SqlField& f1 = args[1]->getData().sqlField();
+ if (f1.sqlNull())
+ return;
+ }
+ f.sqlBigint(++acc.m_count);
+ } else if (fc == Expr_func::Min) {
+ ctx_assert(narg == 1);
+ if (ctl.m_postEval)
+ return;
+ const SqlField& f1 = args[1]->getData().sqlField();
+ if (f1.sqlNull())
+ return;
+ if (f.sqlNull() || f1.less(f))
+ f1.copy(ctx, f);
+ } else if (fc == Expr_func::Max) {
+ ctx_assert(narg == 1);
+ if (ctl.m_postEval)
+ return;
+ const SqlField& f1 = args[1]->getData().sqlField();
+ if (f1.sqlNull())
+ return;
+ if (f.sqlNull() || f.less(f1))
+ f1.copy(ctx, f);
+ } else if (fc == Expr_func::Sum) {
+ ctx_assert(narg == 1);
+ if (ctl.m_postEval)
+ return;
+ const SqlType& t1 = args[1]->getCode().sqlSpec().sqlType();
+ const SqlField& f1 = args[1]->getData().sqlField();
+ if (f1.sqlNull())
+ return;
+ switch (t.type()) {
+ case SqlType::Bigint:
+ switch (t1.type()) {
+ case SqlType::Integer:
+ acc.m_bigint += f1.sqlInteger();
+ break;
+ case SqlType::Bigint:
+ acc.m_bigint += f1.sqlBigint();
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ f.sqlBigint(acc.m_bigint);
+ break;
+ case SqlType::Double:
+ switch (t1.type()) {
+ case SqlType::Real:
+ acc.m_double += f1.sqlReal();
+ break;
+ case SqlType::Double:
+ acc.m_double += f1.sqlDouble();
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ f.sqlDouble(acc.m_double);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else if (fc == Expr_func::Avg) {
+ ctx_assert(narg == 1);
+ if (ctl.m_postEval)
+ return;
+ const SqlType& t1 = args[1]->getCode().sqlSpec().sqlType();
+ const SqlField& f1 = args[1]->getData().sqlField();
+ if (f1.sqlNull())
+ return;
+ switch (t1.type()) {
+ case SqlType::Smallint:
+ acc.m_bigint += f1.sqlSmallint();
+ break;
+ case SqlType::Integer:
+ acc.m_bigint += f1.sqlInteger();
+ break;
+ case SqlType::Bigint:
+ acc.m_bigint += f1.sqlBigint();
+ break;
+ case SqlType::Real:
+ acc.m_double += f1.sqlReal();
+ break;
+ case SqlType::Double:
+ acc.m_double += f1.sqlDouble();
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ f.sqlDouble(acc.m_double / (SqlDouble)++acc.m_count);
+ } else if (fc == Expr_func::Rownum) {
+ ctx_assert(narg == 0);
+ if (! ctl.m_postEval)
+ f.sqlBigint(1 + acc.m_count);
+ else
+ acc.m_count++;
+ } else if (fc == Expr_func::Sysdate) {
+ ctx_assert(narg == 0);
+ if (ctl.m_postEval)
+ return;
+ f.sqlDatetime(acc.m_sysdate);
+ } else {
+ ctx_assert(false);
+ }
+}
diff --git a/ndb/src/client/odbc/executor/Exec_expr_op.cpp b/ndb/src/client/odbc/executor/Exec_expr_op.cpp
new file mode 100644
index 00000000000..fc8b6df9f5b
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_expr_op.cpp
@@ -0,0 +1,147 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <codegen/Code_expr_op.hpp>
+
+void
+Exec_expr_op::evaluate(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ const SqlType& t = code.sqlSpec().sqlType();
+ SqlField& f = ctl.m_groupIndex == 0 ? data.m_sqlField : data.groupField(code.sqlSpec().sqlType(), ctl.m_groupIndex, ctl.m_groupInit);
+ if (code.m_op.arity() == 1) {
+ // evaluate sub-expression
+ ctx_assert(m_expr[1] != 0);
+ m_expr[1]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ if (ctl.m_postEval)
+ return;
+ const SqlField& f1 = ctl.m_groupIndex == 0 ? m_expr[1]->getData().sqlField() : m_expr[1]->getData().groupField(ctl.m_groupIndex);
+ // handle null
+ if (f1.sqlNull()) {
+ f.sqlNull(true);
+ return;
+ }
+ if (t.type() == SqlType::Bigint) {
+ SqlBigint v = 0;
+ SqlBigint v1 = f1.sqlBigint();
+ switch (code.m_op.m_opcode) {
+ case Expr_op::Plus:
+ v = v1;
+ break;
+ case Expr_op::Minus:
+ v = - v1;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ f.sqlBigint(v);
+ } else if (t.type() == SqlType::Double) {
+ SqlDouble v = 0;
+ SqlDouble v1 = f1.sqlDouble();
+ switch (code.m_op.m_opcode) {
+ case Expr_op::Plus:
+ v = v1;
+ break;
+ case Expr_op::Minus:
+ v = - v1;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ f.sqlDouble(v);
+ } else {
+ ctx_assert(false);
+ }
+ } else if (code.m_op.arity() == 2) {
+ // evaluate sub-expressions
+ ctx_assert(m_expr[1] != 0 && m_expr[2] != 0);
+ m_expr[1]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ m_expr[2]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ if (ctl.m_postEval)
+ return;
+ const SqlField& f1 = ctl.m_groupIndex == 0 ? m_expr[1]->getData().sqlField() : m_expr[1]->getData().groupField(ctl.m_groupIndex);
+ const SqlField& f2 = ctl.m_groupIndex == 0 ? m_expr[2]->getData().sqlField() : m_expr[2]->getData().groupField(ctl.m_groupIndex);
+ // handle null
+ if (f1.sqlNull() || f2.sqlNull()) {
+ f.sqlNull(true);
+ return;
+ }
+ if (t.type() == SqlType::Bigint) {
+ SqlBigint v = 0;
+ SqlBigint v1 = f1.sqlBigint();
+ SqlBigint v2 = f2.sqlBigint();
+ switch (code.m_op.m_opcode) {
+ case Expr_op::Add:
+ v = v1 + v2;
+ break;
+ case Expr_op::Subtract:
+ v = v1 - v2;
+ break;
+ case Expr_op::Multiply:
+ v = v1 * v2;
+ break;
+ case Expr_op::Divide:
+ if (v2 == 0) {
+ ctx.pushStatus(Sqlstate::_22012, Error::Gen, "integer division by zero");
+ return;
+ }
+ v = v1 / v2;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ f.sqlBigint(v);
+ } else if (t.type() == SqlType::Double) {
+ SqlDouble v = 0;
+ SqlDouble v1 = f1.sqlDouble();
+ SqlDouble v2 = f2.sqlDouble();
+ switch (code.m_op.m_opcode) {
+ case Expr_op::Add:
+ v = v1 + v2;
+ break;
+ case Expr_op::Subtract:
+ v = v1 - v2;
+ break;
+ case Expr_op::Multiply:
+ v = v1 * v2;
+ break;
+ case Expr_op::Divide:
+ v = v1 / v2;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ f.sqlDouble(v); // XXX isnan()
+ } else {
+ ctx_assert(false);
+ }
+ } else {
+ ctx_assert(false);
+ }
+ // result is not null
+ f.sqlNull(false);
+}
diff --git a/ndb/src/client/odbc/executor/Exec_insert.cpp b/ndb/src/client/odbc/executor/Exec_insert.cpp
new file mode 100644
index 00000000000..c2612c6aaab
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_insert.cpp
@@ -0,0 +1,144 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_insert.hpp>
+#include <codegen/Code_query.hpp>
+
+#ifdef NDB_WIN32
+#define FMT_I64 "%I64d"
+#else
+#define FMT_I64 "%lld"
+#endif
+
+void
+Exec_insert::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ // execute subquery
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // insert each row from query
+ data.setCount(0);
+ while (m_query->fetch(ctx, ctl)) {
+ NdbOperation* op = tcon->getNdbOperation(code.m_tableName);
+ if (op == 0) {
+ ctx.pushStatus(ndb, tcon, 0, "getNdbOperation");
+ return;
+ }
+ if (code.m_insertOp == Insert_op_insert) {
+ if (op->insertTuple() == -1) {
+ ctx.pushStatus(ndb, tcon, op, "insertTuple");
+ return;
+ }
+ } else if (code.m_insertOp == Insert_op_write) {
+ if (op->writeTuple() == -1) {
+ ctx.pushStatus(ndb, tcon, op, "writeTuple");
+ return;
+ }
+ } else {
+ ctx_assert(false);
+ }
+ const SqlRow& sqlRow = m_query->getData().sqlRow();
+ if (code.m_tupleId != 0) {
+ Uint64 tid = op->setTupleId();
+ if (tid == 0) {
+ ctx.pushStatus(ndb, tcon, op, "setTupleId attrId=%u", (unsigned)code.m_tupleId);
+ return;
+ }
+ ctx_log3(("generated TupleId " FMT_I64, tid));
+ } else if (code.m_autoIncrement != 0) {
+ // XXX use cache size 1 for birdies and fishies
+ Uint64 tupleId = ndb->getAutoIncrementValue(code.m_tableName, 1);
+ if (tupleId == 0) {
+ ctx.pushStatus(ndb, "getTupleIdFromNdb");
+ return;
+ }
+ NdbAttrId attrId = code.m_autoIncrement - 1;
+ SqlSmallint sqlSmallint = 0;
+ SqlInteger sqlInteger = 0;
+ SqlBigint sqlBigint = 0;
+ const char* value = 0;
+ if (code.m_idType.type() == SqlType::Smallint) {
+ sqlSmallint = tupleId;
+ value = (const char*)&sqlSmallint;
+ } else if (code.m_idType.type() == SqlType::Integer) {
+ sqlInteger = tupleId;
+ value = (const char*)&sqlInteger;
+ } else if (code.m_idType.type() == SqlType::Bigint) {
+ sqlBigint = tupleId;
+ value = (const char*)&sqlBigint;
+ } else {
+ ctx_assert(false);
+ }
+ if (op->equal(attrId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "equal attrId=%u", (unsigned)attrId);
+ return;
+ }
+ } else {
+ // normal key attributes
+ for (unsigned i = 1; i <= sqlRow.count(); i++) {
+ if (! code.m_isKey[i])
+ continue;
+ NdbAttrId attrId = code.m_attrId[i];
+ const SqlField& f = sqlRow.getEntry(i);
+ const void* addr = f.sqlNull() ? 0 : f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (op->equal(attrId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "equal attrId=%u", (unsigned)attrId);
+ return;
+ }
+ }
+ }
+ // non-key attributes
+ for (unsigned i = 1; i <= sqlRow.count(); i++) {
+ if (code.m_isKey[i])
+ continue;
+ NdbAttrId attrId = code.m_attrId[i];
+ const SqlField& f = sqlRow.getEntry(i);
+ const void* addr = f.sqlNull() ? 0 : f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (op->setValue(attrId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "setValue attrId=%u", (unsigned)attrId);
+ return;
+ }
+ }
+ // default non-key values
+ for (unsigned i = 1; i <= code.m_defaultCount; i++) {
+ NdbAttrId attrId = code.m_defaultId[i];
+ const SqlField& f = code.m_defaultValue->getEntry(i);
+ const void* addr = f.sqlNull() ? 0 : f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (op->setValue(attrId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "setValue attrId=%u", (unsigned)attrId);
+ return;
+ }
+ }
+ if (tcon->execute(NoCommit) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "execute without commit");
+ return;
+ }
+ data.addCount();
+ }
+ stmtArea().setRowCount(ctx, data.getCount());
+}
diff --git a/ndb/src/client/odbc/executor/Exec_pred_op.cpp b/ndb/src/client/odbc/executor/Exec_pred_op.cpp
new file mode 100644
index 00000000000..7caa4656473
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_pred_op.cpp
@@ -0,0 +1,197 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <codegen/Code_pred_op.hpp>
+#include <NdbScanFilter.hpp>
+
+struct TableUnary {
+ Pred_value value1;
+ Pred_value result;
+};
+
+struct TableBinary {
+ Pred_value value1;
+ Pred_value value2;
+ Pred_value result;
+};
+
+static TableUnary
+tableNot[] = {
+ { Pred_value_unknown, Pred_value_unknown },
+ { Pred_value_false, Pred_value_true },
+ { Pred_value_true, Pred_value_false },
+};
+
+static TableBinary
+tableAnd[] = {
+ { Pred_value_unknown, Pred_value_unknown, Pred_value_unknown },
+ { Pred_value_unknown, Pred_value_false, Pred_value_false },
+ { Pred_value_unknown, Pred_value_true, Pred_value_unknown },
+ { Pred_value_false, Pred_value_unknown, Pred_value_false },
+ { Pred_value_false, Pred_value_false, Pred_value_false },
+ { Pred_value_false, Pred_value_true, Pred_value_false },
+ { Pred_value_true, Pred_value_unknown, Pred_value_unknown },
+ { Pred_value_true, Pred_value_false, Pred_value_false },
+ { Pred_value_true, Pred_value_true, Pred_value_true }
+};
+
+static TableBinary
+tableOr[] = {
+ { Pred_value_unknown, Pred_value_unknown, Pred_value_unknown },
+ { Pred_value_unknown, Pred_value_false, Pred_value_unknown },
+ { Pred_value_unknown, Pred_value_true, Pred_value_true },
+ { Pred_value_false, Pred_value_unknown, Pred_value_unknown },
+ { Pred_value_false, Pred_value_false, Pred_value_false },
+ { Pred_value_false, Pred_value_true, Pred_value_true },
+ { Pred_value_true, Pred_value_unknown, Pred_value_true },
+ { Pred_value_true, Pred_value_false, Pred_value_true },
+ { Pred_value_true, Pred_value_true, Pred_value_true }
+};
+
+void
+Exec_pred_op::execInterp(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ ctx_assert(ctl.m_scanFilter != 0);
+ NdbScanFilter& scanFilter = *ctl.m_scanFilter;
+ if (code.m_op.arity() == 1) {
+ ctx_assert(m_pred[1] != 0);
+ switch (code.m_op.m_opcode) {
+ case Pred_op::Not:
+ scanFilter.begin(NdbScanFilter::NAND);
+ m_pred[1]-> execInterp(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ scanFilter.end();
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else if (code.m_op.arity() == 2) {
+ ctx_assert(m_pred[1] != 0 && m_pred[2] != 0);
+ switch (code.m_op.m_opcode) {
+ case Pred_op::And:
+ scanFilter.begin(NdbScanFilter::AND);
+ m_pred[1]-> execInterp(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ m_pred[2]-> execInterp(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ scanFilter.end();
+ break;
+ case Pred_op::Or:
+ scanFilter.begin(NdbScanFilter::OR);
+ m_pred[1]-> execInterp(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ m_pred[2]-> execInterp(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ scanFilter.end();
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ } else {
+ ctx_assert(false);
+ }
+}
+
+void
+Exec_pred_op::evaluate(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Pred_value v = Pred_value_unknown;
+ if (code.m_op.arity() == 1) {
+ // evaluate sub-expression
+ ctx_assert(m_pred[1] != 0);
+ m_pred[1]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ if (ctl.m_postEval)
+ return;
+ Pred_value v1 = ctl.m_groupIndex == 0 ? m_pred[1]->getData().getValue() : m_pred[1]->getData().groupValue(ctl.m_groupIndex);
+ // look up result
+ TableUnary* table = 0;
+ unsigned size = 0;
+ switch (code.m_op.m_opcode) {
+ case Pred_op::Not:
+ table = tableNot;
+ size = sizeof(tableNot) / sizeof(tableNot[0]);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ unsigned i;
+ for (i = 0; i < size; i++) {
+ if (table[i].value1 == v1) {
+ v = table[i].result;
+ break;
+ }
+ }
+ ctx_assert(i < size);
+ } else if (code.m_op.arity() == 2) {
+ // evaluate sub-expressions
+ ctx_assert(m_pred[1] != 0 && m_pred[2] != 0);
+ m_pred[1]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ m_pred[2]->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ if (ctl.m_postEval)
+ return;
+ Pred_value v1 = ctl.m_groupIndex == 0 ? m_pred[1]->getData().getValue() : m_pred[1]->getData().groupValue(ctl.m_groupIndex);
+ Pred_value v2 = ctl.m_groupIndex == 0 ? m_pred[2]->getData().getValue() : m_pred[2]->getData().groupValue(ctl.m_groupIndex);
+ // look up result
+ TableBinary* table = 0;
+ unsigned size = 0;
+ switch (code.m_op.m_opcode) {
+ case Pred_op::And:
+ table = tableAnd;
+ size = sizeof(tableAnd) / sizeof(tableAnd[0]);
+ break;
+ case Pred_op::Or:
+ table = tableOr;
+ size = sizeof(tableOr) / sizeof(tableOr[0]);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ unsigned i;
+ for (i = 0; i < size; i++) {
+ if (table[i].value1 == v1 && table[i].value2 == v2) {
+ v = table[i].result;
+ break;
+ }
+ }
+ ctx_assert(i < size);
+ } else {
+ ctx_assert(false);
+ }
+ // set result
+ if (ctl.m_groupIndex == 0)
+ data.m_value = v;
+ else
+ data.groupValue(ctl.m_groupIndex, ctl.m_groupInit) = v;
+}
diff --git a/ndb/src/client/odbc/executor/Exec_query_index.cpp b/ndb/src/client/odbc/executor/Exec_query_index.cpp
new file mode 100644
index 00000000000..919743beac2
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_query_index.cpp
@@ -0,0 +1,136 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_expr.hpp>
+#include <codegen/Code_query_index.hpp>
+#include <codegen/Code_table.hpp>
+
+void
+Exec_query_index::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ data.m_done = false;
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ if (data.m_con != 0) {
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ ctx_log2(("lookup closed at re-execute"));
+ }
+ const bool unco = connArea().uncommitted();
+ if (! unco) {
+ // use new transaction to not run out of operations
+ data.m_con = ndb->startTransaction();
+ if (data.m_con == 0) {
+ ctx.pushStatus(ndb, "startTransaction");
+ return;
+ }
+ } else {
+ ctx_log3(("lookup using main transaction"));
+ }
+ data.m_op = (unco ? tcon : data.m_con)->getNdbIndexOperation(code.m_indexName, code.m_tableName);
+ if (data.m_op == 0) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), 0, "getNdbIndexOperation");
+ return;
+ }
+ if (data.m_op->readTuple() == -1) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), data.m_op, "readTuple");
+ return;
+ }
+ // key attributes
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* exprMatch = code.m_keyMatch[k];
+ ctx_assert(exprMatch != 0);
+ exprMatch->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ const SqlField& keyMatch = exprMatch->getData().sqlField();
+ SqlField f(code.m_keySpecs.getEntry(k));
+ if (! keyMatch.cast(ctx, f)) {
+ data.m_done = true; // match is not possible
+ return;
+ }
+ const NdbAttrId keyId = code.m_keyId[k];
+ const void* addr = f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (data.m_op->equal(keyId, value) == -1) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), data.m_op, "equal attrId=%u", (unsigned)keyId);
+ return;
+ }
+ }
+ // queried attributes
+ const SqlRow& sqlRow = data.sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ const NdbAttrId attrId = code.m_attrId[i];
+ SqlField& f = sqlRow.getEntry(i);
+ char* addr = static_cast<char*>(f.addr());
+ NdbRecAttr* recAttr = data.m_op->getValue(attrId, addr);
+ if (recAttr == 0) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), data.m_op, "getValue attrId=%u", (unsigned)attrId);
+ return;
+ }
+ data.m_recAttr[i] = recAttr;
+ }
+ if (code.m_attrCount == 0) { // NDB requires one
+ (void)data.m_op->getValue((NdbAttrId)0);
+ }
+ data.setCount(0);
+ if ((unco ? tcon : data.m_con)->execute(unco ? NoCommit : Commit) == -1) {
+ // XXX when did 626 move to connection level
+ if ((unco ? tcon : data.m_con)->getNdbError().code != 626 && data.m_op->getNdbError().code != 626) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), data.m_op, "execute xxx");
+ return;
+ }
+ data.m_done = true;
+ } else {
+ stmtArea().incTuplesFetched();
+ data.m_done = false;
+ }
+ if (! unco) {
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ ctx_log3(("index lookup closed at execute"));
+ }
+}
+
+bool
+Exec_query_index::fetchImpl(Ctx &ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ // returns at most one row
+ if (data.m_done)
+ return false;
+ // set null bits
+ const SqlRow& sqlRow = data.sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ NdbRecAttr* recAttr = data.m_recAttr[i];
+ int isNULL = recAttr->isNULL();
+ SqlField& f = sqlRow.getEntry(i);
+ ctx_assert(isNULL == 0 || isNULL == 1);
+ f.sqlNull(isNULL == 1);
+ }
+ data.m_done = true;
+ return true;
+}
diff --git a/ndb/src/client/odbc/executor/Exec_query_lookup.cpp b/ndb/src/client/odbc/executor/Exec_query_lookup.cpp
new file mode 100644
index 00000000000..599e1a36461
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_query_lookup.cpp
@@ -0,0 +1,136 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_expr.hpp>
+#include <codegen/Code_query_lookup.hpp>
+#include <codegen/Code_table.hpp>
+
+void
+Exec_query_lookup::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ data.m_done = false;
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ if (data.m_con != 0) {
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ ctx_log2(("lookup closed at re-execute"));
+ }
+ const bool unco = connArea().uncommitted();
+ if (! unco) {
+ // use new transaction to not run out of operations
+ data.m_con = ndb->startTransaction();
+ if (data.m_con == 0) {
+ ctx.pushStatus(ndb, "startTransaction");
+ return;
+ }
+ } else {
+ ctx_log3(("lookup using main transaction"));
+ }
+ data.m_op = (unco ? tcon : data.m_con)->getNdbOperation(code.m_tableName);
+ if (data.m_op == 0) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), 0, "getNdbOperation");
+ return;
+ }
+ if (data.m_op->readTuple() == -1) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), data.m_op, "readTuple");
+ return;
+ }
+ // key attributes
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* exprMatch = code.m_keyMatch[k];
+ ctx_assert(exprMatch != 0);
+ exprMatch->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ const SqlField& keyMatch = exprMatch->getData().sqlField();
+ SqlField f(code.m_keySpecs.getEntry(k));
+ if (! keyMatch.cast(ctx, f)) {
+ data.m_done = true; // match is not possible
+ return;
+ }
+ const NdbAttrId keyId = code.m_keyId[k];
+ const void* addr = f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (data.m_op->equal(keyId, value) == -1) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), data.m_op, "equal attrId=%u", (unsigned)keyId);
+ return;
+ }
+ }
+ // queried attributes
+ const SqlRow& sqlRow = data.sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ const NdbAttrId attrId = code.m_attrId[i];
+ SqlField& f = sqlRow.getEntry(i);
+ char* addr = static_cast<char*>(f.addr());
+ NdbRecAttr* recAttr = data.m_op->getValue(attrId, addr);
+ if (recAttr == 0) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), data.m_op, "getValue attrId=%u", (unsigned)attrId);
+ return;
+ }
+ data.m_recAttr[i] = recAttr;
+ }
+ if (code.m_attrCount == 0) { // NDB requires one
+ (void)data.m_op->getValue((NdbAttrId)0);
+ }
+ data.setCount(0);
+ if ((unco ? tcon : data.m_con)->execute(unco ? NoCommit : Commit) == -1) {
+ // XXX when did 626 move to connection level
+ if ((unco ? tcon : data.m_con)->getNdbError().code != 626 && data.m_op->getNdbError().code != 626) {
+ ctx.pushStatus(ndb, (unco ? tcon : data.m_con), data.m_op, "execute xxx");
+ return;
+ }
+ data.m_done = true;
+ } else {
+ stmtArea().incTuplesFetched();
+ data.m_done = false;
+ }
+ if (! unco) {
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ ctx_log3(("lookup closed at execute"));
+ }
+}
+
+bool
+Exec_query_lookup::fetchImpl(Ctx &ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ // returns at most one row
+ if (data.m_done)
+ return false;
+ // set null bits
+ const SqlRow& sqlRow = data.sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ NdbRecAttr* recAttr = data.m_recAttr[i];
+ int isNULL = recAttr->isNULL();
+ SqlField& f = sqlRow.getEntry(i);
+ ctx_assert(isNULL == 0 || isNULL == 1);
+ f.sqlNull(isNULL == 1);
+ }
+ data.m_done = true;
+ return true;
+}
diff --git a/ndb/src/client/odbc/executor/Exec_query_range.cpp b/ndb/src/client/odbc/executor/Exec_query_range.cpp
new file mode 100644
index 00000000000..0bc878d760d
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_query_range.cpp
@@ -0,0 +1,143 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_expr.hpp>
+#include <codegen/Code_query_range.hpp>
+#include <codegen/Code_table.hpp>
+
+#define startBuddyTransaction(x) hupp(x)
+
+void
+Exec_query_range::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ data.m_done = false;
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ if (data.m_con != 0) {
+ data.m_con->stopScan();
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ ctx_log2(("range scan closed at re-execute"));
+ }
+ data.m_con = ndb->startBuddyTransaction(tcon);
+ if (data.m_con == 0) {
+ ctx.pushStatus(ndb, tcon, 0, "startBuddyTransaction");
+ return;
+ }
+ data.m_op = data.m_con->getNdbOperation(code.m_indexName, code.m_tableName);
+ if (data.m_op == 0) {
+ ctx.pushStatus(ndb, data.m_con, 0, "getNdbOperation");
+ return;
+ }
+ if (! code.m_exclusive) {
+ if (data.m_op->openScanReadCommitted(data.m_parallel) == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "openScanReadCommitted");
+ return;
+ }
+ } else {
+ if (data.m_op->openScanExclusive(data.m_parallel) == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "openScanExclusive");
+ return;
+ }
+ }
+ // set bounds
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* exprMatch = code.m_keyMatch[k];
+ ctx_assert(exprMatch != 0);
+ exprMatch->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ const SqlField& keyMatch = exprMatch->getData().sqlField();
+ SqlField f(code.m_keySpecs.getEntry(k));
+ if (! keyMatch.cast(ctx, f)) {
+ data.m_done = true; // match is not possible
+ return;
+ }
+ const NdbAttrId keyId = code.m_keyId[k];
+ const void* addr = f.addr();
+ const char* value = static_cast<const char*>(addr);
+ const unsigned len = f.allocSize();
+ if (data.m_op->setBound(keyId, NdbOperation::BoundEQ, value, len) == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "setBound attrId=%u", (unsigned)keyId);
+ return;
+ }
+ }
+ // queried attributes
+ const SqlRow& sqlRow = data.sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ const NdbAttrId attrId = code.m_attrId[i];
+ SqlField& f = sqlRow.getEntry(i);
+ char* addr = static_cast<char*>(f.addr());
+ NdbRecAttr* recAttr = data.m_op->getValue(attrId, addr);
+ if (recAttr == 0) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "getValue attrId=%u", (unsigned)attrId);
+ return;
+ }
+ data.m_recAttr[i] = recAttr;
+ }
+ if (code.m_attrCount == 0) { // NDB requires one
+ (void)data.m_op->getValue((NdbAttrId)0);
+ }
+ data.setCount(0);
+ if (data.m_con->executeScan() == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "executeScan");
+ return;
+ }
+ ctx_log2(("range scan %s [%08x] started", ! code.m_exclusive ? "read" : "exclusive", (unsigned)this));
+ ctl.m_scanOp = data.m_op;
+}
+
+bool
+Exec_query_range::fetchImpl(Ctx &ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ // if never started
+ if (data.m_done)
+ return false;
+ int ret = data.m_con->nextScanResult();
+ if (ret != 0) {
+ if (ret == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "nextScanResult");
+ }
+ data.m_con->stopScan();
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ ctx_log2(("range scan [%08x] closed at last row", (unsigned)this));
+ return false;
+ }
+ // set null bits
+ const SqlRow& sqlRow = data.sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ NdbRecAttr* recAttr = data.m_recAttr[i];
+ int isNULL = recAttr->isNULL();
+ SqlField& f = sqlRow.getEntry(i);
+ ctx_assert(isNULL == 0 || isNULL == 1);
+ f.sqlNull(isNULL == 1);
+ }
+ stmtArea().incTuplesFetched();
+ return true;
+}
diff --git a/ndb/src/client/odbc/executor/Exec_query_scan.cpp b/ndb/src/client/odbc/executor/Exec_query_scan.cpp
new file mode 100644
index 00000000000..213dfdd616d
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_query_scan.cpp
@@ -0,0 +1,130 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <NdbScanFilter.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_query_scan.hpp>
+#include <codegen/Code_table.hpp>
+
+#define startBuddyTransaction(x) hupp(x)
+
+void
+Exec_query_scan::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ if (data.m_con != 0) {
+ data.m_con->stopScan();
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ ctx_log2(("scan closed at re-execute"));
+ }
+ data.m_con = ndb->startBuddyTransaction(tcon);
+ if (data.m_con == 0) {
+ ctx.pushStatus(ndb, tcon, 0, "startBuddyTransaction");
+ return;
+ }
+ data.m_op = data.m_con->getNdbOperation(code.m_tableName);
+ if (data.m_op == 0) {
+ ctx.pushStatus(ndb, data.m_con, 0, "getNdbOperation");
+ return;
+ }
+ if (! code.m_exclusive) {
+ if (data.m_op->openScanReadCommitted(data.m_parallel) == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "openScanReadCommitted");
+ return;
+ }
+ } else {
+ if (data.m_op->openScanExclusive(data.m_parallel) == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "openScanExclusive");
+ return;
+ }
+ }
+ if (m_interp == 0) {
+ // XXX unnecessary
+ if (data.m_op->interpret_exit_ok() == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "interpret_exit_ok");
+ return;
+ }
+ } else {
+ NdbScanFilter scanFilter(data.m_op);
+ scanFilter.begin(NdbScanFilter::AND);
+ ctl.m_scanFilter = &scanFilter;
+ m_interp->execInterp(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ scanFilter.end();
+ }
+ const SqlRow& sqlRow = data.sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ const NdbAttrId attrId = code.m_attrId[i];
+ SqlField& sqlField = sqlRow.getEntry(i);
+ char* addr = static_cast<char*>(sqlField.addr());
+ NdbRecAttr* recAttr = data.m_op->getValue(attrId, addr);
+ if (recAttr == 0) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "getValue attrId=%u", (unsigned)attrId);
+ return;
+ }
+ data.m_recAttr[i] = recAttr;
+ }
+ if (code.m_attrCount == 0) { // NDB requires one
+ (void)data.m_op->getValue((NdbAttrId)0);
+ }
+ if (data.m_con->executeScan() == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "executeScan");
+ return;
+ }
+ ctx_log2(("scan %s [%08x] started", ! code.m_exclusive ? "read" : "exclusive", (unsigned)this));
+ ctl.m_scanOp = data.m_op;
+}
+
+bool
+Exec_query_scan::fetchImpl(Ctx &ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ int ret = data.m_con->nextScanResult();
+ if (ret != 0) {
+ if (ret == -1) {
+ ctx.pushStatus(ndb, data.m_con, data.m_op, "nextScanResult");
+ }
+ data.m_con->stopScan();
+ ndb->closeTransaction(data.m_con);
+ data.m_con = 0;
+ data.m_op = 0;
+ ctx_log2(("scan [%08x] closed at last row", (unsigned)this));
+ return false;
+ }
+ // set null bits
+ const SqlRow& sqlRow = data.sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ NdbRecAttr* recAttr = data.m_recAttr[i];
+ int isNULL = recAttr->isNULL();
+ SqlField& sqlField = sqlRow.getEntry(i);
+ ctx_assert(isNULL == 0 || isNULL == 1);
+ sqlField.sqlNull(isNULL == 1);
+ }
+ stmtArea().incTuplesFetched();
+ return true;
+}
diff --git a/ndb/src/client/odbc/executor/Exec_query_sys.cpp b/ndb/src/client/odbc/executor/Exec_query_sys.cpp
new file mode 100644
index 00000000000..acdc120e609
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_query_sys.cpp
@@ -0,0 +1,761 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_query_sys.hpp>
+#include <codegen/Code_table.hpp>
+
+#define NULL_CHAR ((char*)0)
+#define NULL_INT (-2147483647)
+
+struct Typeinfo {
+ const char* m_type_name; // 1
+ int m_data_type; // 2
+ int m_column_size; // 3
+ const char* m_literal_prefix; // 4
+ const char* m_literal_suffix; // 5
+ const char* m_create_params; // 6
+ int m_nullable; // 7
+ int m_case_sensitive; // 8
+ int m_searchable; // 9
+ int m_unsigned_attribute; // 10
+ int m_fixed_prec_scale; // 11
+ int m_auto_unique_value; // 12
+ const char* m_local_type_name; // 13
+ int m_minimum_scale; // 14
+ int m_maximum_scale; // 15
+ int m_sql_data_type; // 16
+ int m_sql_datetime_sub; // 17
+ int m_num_prec_radix; // 18
+ int m_interval_precision; // 19
+};
+
+static const Typeinfo
+typeinfoList[] = {
+ { "CHAR", // 1
+ SQL_CHAR, // 2
+ 8000, // 3
+ "'", // 4
+ "'", // 5
+ "length", // 6
+ SQL_NULLABLE, // 7
+ SQL_TRUE, // 8
+ SQL_SEARCHABLE, // 9
+ NULL_INT, // 10
+ SQL_FALSE, // 11
+ NULL_INT, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_CHAR, // 16
+ NULL_INT, // 17
+ NULL_INT, // 18
+ NULL_INT // 19
+ },
+ { "VARCHAR", // 1
+ SQL_VARCHAR, // 2
+ 8000, // 3
+ "'", // 4
+ "'", // 5
+ "length", // 6
+ SQL_NULLABLE, // 7
+ SQL_TRUE, // 8
+ SQL_SEARCHABLE, // 9
+ NULL_INT, // 10
+ SQL_FALSE, // 11
+ NULL_INT, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_VARCHAR, // 16
+ NULL_INT, // 17
+ NULL_INT, // 18
+ NULL_INT // 19
+ },
+ { "BINARY", // 1
+ SQL_BINARY, // 2
+ 8000, // 3
+ "'", // 4
+ "'", // 5
+ "length", // 6
+ SQL_NULLABLE, // 7
+ SQL_TRUE, // 8
+ SQL_SEARCHABLE, // 9
+ NULL_INT, // 10
+ SQL_FALSE, // 11
+ NULL_INT, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_BINARY, // 16
+ NULL_INT, // 17
+ NULL_INT, // 18
+ NULL_INT // 19
+ },
+ { "VARBINARY", // 1
+ SQL_VARBINARY, // 2
+ 8000, // 3
+ "'", // 4
+ "'", // 5
+ "length", // 6
+ SQL_NULLABLE, // 7
+ SQL_TRUE, // 8
+ SQL_SEARCHABLE, // 9
+ NULL_INT, // 10
+ SQL_FALSE, // 11
+ NULL_INT, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_VARBINARY, // 16
+ NULL_INT, // 17
+ NULL_INT, // 18
+ NULL_INT // 19
+ },
+ { "SMALLINT", // 1
+ SQL_SMALLINT, // 2
+ 4, // 3
+ NULL_CHAR, // 4
+ NULL_CHAR, // 5
+ NULL_CHAR, // 6
+ SQL_NULLABLE, // 7
+ SQL_FALSE, // 8
+ SQL_SEARCHABLE, // 9
+ SQL_FALSE, // 10
+ SQL_TRUE, // 11
+ SQL_FALSE, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_SMALLINT, // 16
+ NULL_INT, // 17
+ 10, // 18
+ NULL_INT // 19
+ },
+ { "SMALLINT UNSIGNED", // 1
+ SQL_SMALLINT, // 2
+ 4, // 3
+ NULL_CHAR, // 4
+ NULL_CHAR, // 5
+ NULL_CHAR, // 6
+ SQL_NULLABLE, // 7
+ SQL_FALSE, // 8
+ SQL_SEARCHABLE, // 9
+ SQL_TRUE, // 10
+ SQL_TRUE, // 11
+ SQL_FALSE, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_SMALLINT, // 16
+ NULL_INT, // 17
+ 10, // 18
+ NULL_INT // 19
+ },
+ { "INT", // 1
+ SQL_INTEGER, // 2
+ 9, // 3
+ NULL_CHAR, // 4
+ NULL_CHAR, // 5
+ NULL_CHAR, // 6
+ SQL_NULLABLE, // 7
+ SQL_FALSE, // 8
+ SQL_SEARCHABLE, // 9
+ SQL_FALSE, // 10
+ SQL_TRUE, // 11
+ SQL_FALSE, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_INTEGER, // 16
+ NULL_INT, // 17
+ 10, // 18
+ NULL_INT // 19
+ },
+ { "INT UNSIGNED", // 1
+ SQL_INTEGER, // 2
+ 9, // 3
+ NULL_CHAR, // 4
+ NULL_CHAR, // 5
+ NULL_CHAR, // 6
+ SQL_NULLABLE, // 7
+ SQL_FALSE, // 8
+ SQL_SEARCHABLE, // 9
+ SQL_TRUE, // 10
+ SQL_TRUE, // 11
+ SQL_FALSE, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_INTEGER, // 16
+ NULL_INT, // 17
+ 10, // 18
+ NULL_INT // 19
+ },
+ { "BIGINT", // 1
+ SQL_BIGINT, // 2
+ 19, // 3
+ NULL_CHAR, // 4
+ NULL_CHAR, // 5
+ NULL_CHAR, // 6
+ SQL_NULLABLE, // 7
+ SQL_FALSE, // 8
+ SQL_SEARCHABLE, // 9
+ SQL_FALSE, // 10
+ SQL_TRUE, // 11
+ SQL_FALSE, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_BIGINT, // 16
+ NULL_INT, // 17
+ 10, // 18
+ NULL_INT // 19
+ },
+ { "BIGINT UNSIGNED", // 1
+ SQL_BIGINT, // 2
+ 19, // 3
+ NULL_CHAR, // 4
+ NULL_CHAR, // 5
+ NULL_CHAR, // 6
+ SQL_NULLABLE, // 7
+ SQL_FALSE, // 8
+ SQL_SEARCHABLE, // 9
+ SQL_TRUE, // 10
+ SQL_TRUE, // 11
+ SQL_FALSE, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_BIGINT, // 16
+ NULL_INT, // 17
+ 10, // 18
+ NULL_INT // 19
+ },
+ { "REAL", // 1
+ SQL_REAL, // 2
+ 31, // 3
+ NULL_CHAR, // 4
+ NULL_CHAR, // 5
+ NULL_CHAR, // 6
+ SQL_NULLABLE, // 7
+ SQL_FALSE, // 8
+ SQL_SEARCHABLE, // 9
+ SQL_FALSE, // 10
+ SQL_TRUE, // 11
+ SQL_FALSE, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_REAL, // 16
+ NULL_INT, // 17
+ 2, // 18
+ NULL_INT // 19
+ },
+ { "FLOAT", // 1
+ SQL_DOUBLE, // 2
+ 63, // 3
+ NULL_CHAR, // 4
+ NULL_CHAR, // 5
+ NULL_CHAR, // 6
+ SQL_NULLABLE, // 7
+ SQL_FALSE, // 8
+ SQL_SEARCHABLE, // 9
+ SQL_FALSE, // 10
+ SQL_TRUE, // 11
+ SQL_FALSE, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_DOUBLE, // 16
+ NULL_INT, // 17
+ 2, // 18
+ NULL_INT // 19
+ },
+ { "DATETIME", // 1
+ SQL_TYPE_TIMESTAMP, // 2
+ 30, // 3
+ NULL_CHAR, // 4
+ NULL_CHAR, // 5
+ NULL_CHAR, // 6
+ SQL_NULLABLE, // 7
+ SQL_FALSE, // 8
+ SQL_SEARCHABLE, // 9
+ SQL_FALSE, // 10
+ SQL_TRUE, // 11
+ SQL_FALSE, // 12
+ NULL_CHAR, // 13
+ NULL_INT, // 14
+ NULL_INT, // 15
+ SQL_DATETIME, // 16 XXX
+ NULL_INT, // 17
+ 2, // 18
+ NULL_INT // 19
+ }
+};
+
+const unsigned
+typeinfoCount = sizeof(typeinfoList)/sizeof(typeinfoList[0]);
+
+void
+Exec_query_sys::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbDictionary::Dictionary* ndbDictionary = ndb->getDictionary();
+ if (ndbDictionary == 0) {
+ ctx.pushStatus(ndb, "getDictionary");
+ return;
+ }
+ if (code.m_sysId == DictSys::OdbcTypeinfo || code.m_sysId == DictSys::Dual) {
+ data.m_rowPos = 0; // at first entry
+ return;
+ }
+ if (code.m_sysId == DictSys::OdbcTables || code.m_sysId == DictSys::OdbcColumns || code.m_sysId == DictSys::OdbcPrimarykeys) {
+ // take all objects
+ if (ndbDictionary->listObjects(data.m_objectList) == -1) {
+ ctx.pushStatus(ndb, "listObjects");
+ return;
+ }
+ data.m_tablePos = 0; // at first entry
+ data.m_attrPos = 0;
+ data.m_keyPos = 0;
+ return;
+ }
+ ctx_assert(false);
+}
+
+static bool
+isNdbTable(const NdbDictionary::Dictionary::List::Element& element)
+{
+ switch (element.type) {
+ //case NdbDictionary::Object::SystemTable:
+ case NdbDictionary::Object::UserTable:
+ case NdbDictionary::Object::UniqueHashIndex:
+ case NdbDictionary::Object::HashIndex:
+ case NdbDictionary::Object::UniqueOrderedIndex:
+ case NdbDictionary::Object::OrderedIndex:
+ return true;
+ default:
+ break;
+ }
+ return false;
+}
+
+
+bool
+Exec_query_sys::fetchImpl(Ctx &ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbDictionary::Dictionary* ndbDictionary = ndb->getDictionary();
+ if (ndbDictionary == 0) {
+ ctx.pushStatus(ndb, "getDictionary");
+ return false;
+ }
+ if (code.m_sysId == DictSys::OdbcTypeinfo) {
+ if (data.m_rowPos >= typeinfoCount) {
+ return false;
+ }
+ SqlRow& row = data.m_sqlRow;
+ const Typeinfo& typeinfo = typeinfoList[data.m_rowPos++];
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ SqlField& f = data.m_sqlRow.getEntry(i);
+ switch (1 + code.m_attrId[i]) {
+ case 1:
+ if (typeinfo.m_type_name == NULL_CHAR)
+ f.sqlNull(true);
+ else
+ f.sqlVarchar(typeinfo.m_type_name, SQL_NTS);
+ break;
+ case 2:
+ if (typeinfo.m_data_type == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_data_type);
+ break;
+ case 3:
+ if (typeinfo.m_column_size == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_column_size);
+ break;
+ case 4:
+ if (typeinfo.m_literal_prefix == NULL_CHAR)
+ f.sqlNull(true);
+ else
+ f.sqlVarchar(typeinfo.m_literal_prefix, SQL_NTS);
+ break;
+ case 5:
+ if (typeinfo.m_literal_suffix == NULL_CHAR)
+ f.sqlNull(true);
+ else
+ f.sqlVarchar(typeinfo.m_literal_suffix, SQL_NTS);
+ break;
+ case 6:
+ if (typeinfo.m_create_params == NULL_CHAR)
+ f.sqlNull(true);
+ else
+ f.sqlVarchar(typeinfo.m_create_params, SQL_NTS);
+ break;
+ case 7:
+ if (typeinfo.m_nullable == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_nullable);
+ break;
+ case 8:
+ if (typeinfo.m_case_sensitive == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_case_sensitive);
+ break;
+ case 9:
+ if (typeinfo.m_searchable == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_searchable);
+ break;
+ case 10:
+ if (typeinfo.m_unsigned_attribute == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_unsigned_attribute);
+ break;
+ case 11:
+ if (typeinfo.m_fixed_prec_scale == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_fixed_prec_scale);
+ break;
+ case 12:
+ if (typeinfo.m_auto_unique_value == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_auto_unique_value);
+ break;
+ case 13:
+ if (typeinfo.m_local_type_name == NULL_CHAR)
+ f.sqlNull(true);
+ else
+ f.sqlVarchar(typeinfo.m_local_type_name, SQL_NTS);
+ break;
+ case 14:
+ if (typeinfo.m_minimum_scale == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_minimum_scale);
+ break;
+ case 15:
+ if (typeinfo.m_maximum_scale == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_maximum_scale);
+ break;
+ case 16:
+ if (typeinfo.m_sql_data_type == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_sql_data_type);
+ break;
+ case 17:
+ if (typeinfo.m_sql_datetime_sub == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_sql_datetime_sub);
+ break;
+ case 18:
+ if (typeinfo.m_sql_datetime_sub == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_sql_datetime_sub);
+ break;
+ case 19:
+ if (typeinfo.m_interval_precision == NULL_INT)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(typeinfo.m_interval_precision);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ }
+ return true;
+ }
+ if (code.m_sysId == DictSys::OdbcTables) {
+ if (data.m_tablePos >= data.m_objectList.count) {
+ return false;
+ }
+ NdbDictionary::Dictionary::List::Element& element = data.m_objectList.elements[data.m_tablePos++];
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ SqlField& f = data.m_sqlRow.getEntry(i);
+ switch (1 + code.m_attrId[i]) {
+ case 1: // TABLE_CAT
+ f.sqlNull(true);
+ break;
+ case 2: // TABLE_SCHEM
+ f.sqlNull(true);
+ break;
+ case 3: // TABLE_NAME
+ f.sqlVarchar(element.name, SQL_NTS);
+ break;
+ case 4: { // TABLE_TYPE
+ if (element.type == NdbDictionary::Object::SystemTable)
+ f.sqlVarchar("SYSTEM TABLE", SQL_NTS);
+ else if (element.type == NdbDictionary::Object::UserTable)
+ f.sqlVarchar("TABLE", SQL_NTS);
+ else if (element.type == NdbDictionary::Object::UniqueHashIndex)
+ f.sqlVarchar("UNIQUE HASH INDEX", SQL_NTS);
+ else if (element.type == NdbDictionary::Object::HashIndex)
+ f.sqlVarchar("HASH INDEX", SQL_NTS);
+ else if (element.type == NdbDictionary::Object::UniqueOrderedIndex)
+ f.sqlVarchar("UNIQUE INDEX", SQL_NTS);
+ else if (element.type == NdbDictionary::Object::OrderedIndex)
+ f.sqlVarchar("INDEX", SQL_NTS);
+ else if (element.type == NdbDictionary::Object::IndexTrigger)
+ f.sqlVarchar("INDEX TRIGGER", SQL_NTS);
+ else if (element.type == NdbDictionary::Object::SubscriptionTrigger)
+ f.sqlVarchar("SUBSCRIPTION TRIGGER", SQL_NTS);
+ else if (element.type == NdbDictionary::Object::ReadOnlyConstraint)
+ f.sqlVarchar("READ ONLY CONSTRAINT", SQL_NTS);
+ else
+ f.sqlVarchar("UNKNOWN", SQL_NTS);
+ }
+ break;
+ case 5: // REMARKS
+ f.sqlNull(true);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ }
+ return true;
+ }
+ if (code.m_sysId == DictSys::OdbcColumns) {
+ if (data.m_tablePos >= data.m_objectList.count) {
+ return false;
+ }
+ const NdbDictionary::Dictionary::List::Element& element = data.m_objectList.elements[data.m_tablePos];
+ unsigned nattr;
+ const NdbDictionary::Table* ndbTable;
+ nattr = 0;
+ if (isNdbTable(element)) {
+ ndbTable = ndbDictionary->getTable(element.name);
+ if (ndbTable == 0) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "getTable %s", element.name);
+ return false;
+ }
+ nattr = ndbTable->getNoOfColumns();
+ }
+ while (data.m_attrPos >= nattr) {
+ if (++data.m_tablePos >= data.m_objectList.count) {
+ return false;
+ }
+ const NdbDictionary::Dictionary::List::Element& element = data.m_objectList.elements[data.m_tablePos];
+ nattr = 0;
+ if (isNdbTable(element)) {
+ ndbTable = ndbDictionary->getTable(element.name);
+ if (ndbTable == 0) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "getTable %s", element.name);
+ return false;
+ }
+ data.m_attrPos = 0;
+ nattr = ndbTable->getNoOfColumns();
+ }
+ }
+ int attrId = data.m_attrPos++;
+ const NdbDictionary::Column* ndbColumn = ndbTable->getColumn(attrId);
+ if (ndbColumn == 0) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "getColumn %s.%d", ndbTable->getName(), attrId);
+ return false;
+ }
+ SqlType sqlType(ctx, ndbColumn);
+ if (! ctx.ok())
+ return false;
+ const char* p;
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ SqlField& f = data.m_sqlRow.getEntry(i);
+ switch (1 + code.m_attrId[i]) {
+ case 1: // TABLE_CAT
+ f.sqlNull(true);
+ break;
+ case 2: // TABLE_SCHEM
+ f.sqlNull(true);
+ break;
+ case 3: // TABLE_NAME
+ f.sqlVarchar(ndbTable->getName(), SQL_NTS);
+ break;
+ case 4: // COLUMN_NAME
+ f.sqlVarchar(ndbColumn->getName(), SQL_NTS);
+ break;
+ case 5: // DATA_TYPE
+ f.sqlInteger(sqlType.type());
+ break;
+ case 6: // TYPE_NAME
+ f.sqlVarchar(sqlType.typeName(), SQL_NTS);
+ break;
+ case 7: // COLUMN_SIZE
+ f.sqlInteger(sqlType.displaySize());
+ break;
+ case 8: // BUFFER_LENGTH
+ f.sqlInteger(sqlType.size());
+ break;
+ case 9: // DECIMAL_DIGITS
+ if (sqlType.type() == SqlType::Char)
+ f.sqlNull(true);
+ else
+ f.sqlInteger(0);
+ break;
+ case 10: // NUM_PREC_RADIX
+ if (sqlType.type() == SqlType::Integer || sqlType.type() == SqlType::Bigint)
+ f.sqlInteger(10);
+ else
+ f.sqlNull(true);
+ break;
+ case 11: // NULLABLE
+ if (sqlType.nullable())
+ f.sqlInteger(SQL_NULLABLE);
+ else
+ f.sqlInteger(SQL_NO_NULLS);
+ break;
+ case 12: // REMARKS
+ f.sqlNull(true);
+ break;
+ case 13: // COLUMN_DEF
+ if ((p = ndbColumn->getDefaultValue()) != 0)
+ f.sqlVarchar(p, SQL_NTS);
+ else
+ f.sqlNull(true);
+ break;
+ case 14: // SQL_DATA_TYPE
+ f.sqlInteger(sqlType.type());
+ break;
+ case 15: // SQL_DATETIME_SUB
+ f.sqlNull(true);
+ break;
+ case 16: // CHAR_OCTET_LENGTH
+ if (sqlType.type() == SqlType::Char)
+ f.sqlInteger(sqlType.length());
+ else
+ f.sqlNull(true);
+ break;
+ case 17: // ORDINAL_POSITION
+ f.sqlInteger(1 + attrId);
+ break;
+ case 18: // IS_NULLABLE
+ if (sqlType.nullable())
+ f.sqlVarchar("YES", SQL_NTS);
+ else
+ f.sqlVarchar("NO", SQL_NTS);
+ break;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ }
+ return true;
+ }
+ if (code.m_sysId == DictSys::OdbcPrimarykeys) {
+ if (data.m_tablePos >= data.m_objectList.count) {
+ return false;
+ }
+ NdbDictionary::Dictionary::List::Element& element = data.m_objectList.elements[data.m_tablePos];
+ unsigned nkeys;
+ const NdbDictionary::Table* ndbTable;
+ nkeys = 0;
+ if (isNdbTable(element)) {
+ ndbTable = ndbDictionary->getTable(element.name);
+ if (ndbTable == 0) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "getTable %s", element.name);
+ return false;
+ }
+ nkeys = ndbTable->getNoOfPrimaryKeys();
+ }
+ while (data.m_keyPos >= nkeys) {
+ if (++data.m_tablePos >= data.m_objectList.count) {
+ return false;
+ }
+ NdbDictionary::Dictionary::List::Element& element = data.m_objectList.elements[data.m_tablePos];
+ nkeys = 0;
+ if (isNdbTable(element)) {
+ ndbTable = ndbDictionary->getTable(element.name);
+ if (ndbTable == 0) {
+ ctx.pushStatus(ndbDictionary->getNdbError(), "getTable %s", element.name);
+ return false;
+ }
+ data.m_keyPos = 0;
+ nkeys = ndbTable->getNoOfPrimaryKeys();
+ }
+ }
+ unsigned keyPos = data.m_keyPos++;
+ const char* keyName = ndbTable->getPrimaryKey(keyPos);
+ if (keyName == 0)
+ keyName = "?";
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ SqlField& f = data.m_sqlRow.getEntry(i);
+ switch (1 + code.m_attrId[i]) {
+ case 1: // TABLE_CAT
+ f.sqlNull(true);
+ break;
+ case 2: // TABLE_SCHEM
+ f.sqlNull(true);
+ break;
+ case 3: // TABLE_NAME
+ f.sqlVarchar(ndbTable->getName(), SQL_NTS);
+ break;
+ case 4: // COLUMN_NAME
+ f.sqlVarchar(keyName, SQL_NTS);
+ break;
+ case 5: // KEY_SEQ
+ f.sqlInteger(1 + keyPos);
+ break;
+ case 6: // PK_NAME
+ f.sqlNull(true);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ }
+ return true;
+ }
+ if (code.m_sysId == DictSys::Dual) {
+ if (data.m_rowPos > 0) {
+ return false;
+ }
+ data.m_rowPos++;
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ SqlField& f = data.m_sqlRow.getEntry(i);
+ switch (1 + code.m_attrId[i]) {
+ case 1: // DUMMY
+ f.sqlVarchar("X", 1);
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ }
+ return true;
+ }
+ ctx_assert(false);
+ return false;
+}
diff --git a/ndb/src/client/odbc/executor/Exec_update_index.cpp b/ndb/src/client/odbc/executor/Exec_update_index.cpp
new file mode 100644
index 00000000000..35b6159d8ca
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_update_index.cpp
@@ -0,0 +1,96 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_expr.hpp>
+#include <codegen/Code_update_index.hpp>
+#include <codegen/Code_query.hpp>
+
+void
+Exec_update_index::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ // execute subquery
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // update each row from the query
+ while (m_query->fetch(ctx, ctl)) {
+ NdbOperation* op = tcon->getNdbIndexOperation(code.m_indexName, code.m_tableName);
+ if (op == 0) {
+ ctx.pushStatus(ndb, tcon, 0, "getNdbIndexOperation");
+ return;
+ }
+ if (op->updateTuple() == -1) {
+ ctx.pushStatus(ndb, tcon, op, "updateTuple");
+ return;
+ }
+ // key attributes
+ bool done = false;
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* exprMatch = code.m_keyMatch[k];
+ ctx_assert(exprMatch != 0);
+ exprMatch->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ const SqlField& keyMatch = exprMatch->getData().sqlField();
+ SqlField f(code.m_keySpecs.getEntry(k));
+ if (! keyMatch.cast(ctx, f)) {
+ done = true; // match is not possible
+ break;
+ }
+ const NdbAttrId keyId = code.m_keyId[k];
+ const void* addr = f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (op->equal(keyId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "equal attrId=%u", (unsigned)keyId);
+ return;
+ }
+ }
+ if (done)
+ continue;
+ // updated attributes
+ const SqlRow& sqlRow = m_query->getData().sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ const NdbAttrId attrId = code.m_attrId[i];
+ const SqlField& f = sqlRow.getEntry(i);
+ const void* addr = f.sqlNull() ? 0 : f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (op->setValue(attrId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "setValue attrId=%u", (unsigned)attrId);
+ return;
+ }
+ }
+ data.setCount(0);
+ if (tcon->execute(NoCommit) == -1) {
+ // XXX when did 626 move to connection level
+ if (tcon->getNdbError().code != 626 && op->getNdbError().code != 626) {
+ ctx.pushStatus(ndb, tcon, op, "execute without commit");
+ return;
+ }
+ } else {
+ data.addCount();
+ }
+ }
+ stmtArea().setRowCount(ctx, data.getCount());
+}
diff --git a/ndb/src/client/odbc/executor/Exec_update_lookup.cpp b/ndb/src/client/odbc/executor/Exec_update_lookup.cpp
new file mode 100644
index 00000000000..2c801372de3
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_update_lookup.cpp
@@ -0,0 +1,96 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_expr.hpp>
+#include <codegen/Code_update_lookup.hpp>
+#include <codegen/Code_query.hpp>
+
+void
+Exec_update_lookup::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ // execute subquery
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ // update each row from the query
+ while (m_query->fetch(ctx, ctl)) {
+ NdbOperation* op = tcon->getNdbOperation(code.m_tableName);
+ if (op == 0) {
+ ctx.pushStatus(ndb, tcon, 0, "getNdbOperation");
+ return;
+ }
+ if (op->updateTuple() == -1) {
+ ctx.pushStatus(ndb, tcon, op, "updateTuple");
+ return;
+ }
+ // key attributes
+ bool done = false;
+ for (unsigned k = 1; k <= code.m_keyCount; k++) {
+ Exec_expr* exprMatch = code.m_keyMatch[k];
+ ctx_assert(exprMatch != 0);
+ exprMatch->evaluate(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ const SqlField& keyMatch = exprMatch->getData().sqlField();
+ SqlField f(code.m_keySpecs.getEntry(k));
+ if (! keyMatch.cast(ctx, f)) {
+ done = true; // match is not possible
+ break;
+ }
+ const NdbAttrId keyId = code.m_keyId[k];
+ const void* addr = f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (op->equal(keyId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "equal attrId=%u", (unsigned)keyId);
+ return;
+ }
+ }
+ if (done)
+ continue;
+ // updated attributes
+ const SqlRow& sqlRow = m_query->getData().sqlRow();
+ ctx_assert(sqlRow.count() == code.m_attrCount);
+ for (unsigned i = 1; i <= code.m_attrCount; i++) {
+ const NdbAttrId attrId = code.m_attrId[i];
+ const SqlField& f = sqlRow.getEntry(i);
+ const void* addr = f.sqlNull() ? 0 : f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (op->setValue(attrId, value) == -1) {
+ ctx.pushStatus(ndb, tcon, op, "setValue attrId=%u", (unsigned)attrId);
+ return;
+ }
+ }
+ data.setCount(0);
+ if (tcon->execute(NoCommit) == -1) {
+ // XXX when did 626 move to connection level
+ if (tcon->getNdbError().code != 626 && op->getNdbError().code != 626) {
+ ctx.pushStatus(ndb, tcon, op, "execute without commit");
+ return;
+ }
+ } else {
+ data.addCount();
+ }
+ }
+ stmtArea().setRowCount(ctx, data.getCount());
+}
diff --git a/ndb/src/client/odbc/executor/Exec_update_scan.cpp b/ndb/src/client/odbc/executor/Exec_update_scan.cpp
new file mode 100644
index 00000000000..a36fdd27142
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Exec_update_scan.cpp
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <common/StmtArea.hpp>
+#include <common/ResultArea.hpp>
+#include <codegen/Code_update_scan.hpp>
+#include <codegen/Code_query.hpp>
+
+void
+Exec_update_scan::execImpl(Ctx& ctx, Ctl& ctl)
+{
+ const Code& code = getCode();
+ Data& data = getData();
+ Ndb* const ndb = ndbObject();
+ NdbConnection* const tcon = ndbConnection();
+ // execute subquery
+ ctx_assert(m_query != 0);
+ m_query->execute(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ ctx_assert(ctl.m_scanOp != 0);
+ // update each row from query
+ data.setCount(0);
+ while (m_query->fetch(ctx, ctl)) {
+ NdbOperation* toOp = ctl.m_scanOp->takeOverForUpdate(tcon);
+ if (toOp == 0) {
+ ctx.pushStatus(ndb, tcon, ctl.m_scanOp, "takeOverScanOp");
+ return;
+ }
+ const SqlRow& sqlRow = m_query->getData().sqlRow();
+ for (unsigned i = 1; i <= sqlRow.count(); i++) {
+ const SqlField& f = sqlRow.getEntry(i);
+ const void* addr = f.sqlNull() ? 0 : f.addr();
+ const char* value = static_cast<const char*>(addr);
+ if (toOp->setValue(code.m_attrId[i], value) == -1) {
+ ctx.pushStatus(ndb, tcon, toOp, "setValue");
+ return;
+ }
+ }
+ if (tcon->execute(NoCommit) == -1) {
+ ctx.pushStatus(ndb, tcon, toOp, "execute without commit");
+ return;
+ }
+ data.addCount();
+ }
+ stmtArea().setRowCount(ctx, data.getCount());
+}
diff --git a/ndb/src/client/odbc/executor/Executor.cpp b/ndb/src/client/odbc/executor/Executor.cpp
new file mode 100644
index 00000000000..adabb28a4a5
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Executor.cpp
@@ -0,0 +1,68 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/common.hpp>
+#include <common/CodeTree.hpp>
+#include <common/StmtArea.hpp>
+#include <codegen/Code_root.hpp>
+#include "Executor.hpp"
+
+void
+Executor::execute(Ctx& ctx)
+{
+ Exec_base::Ctl ctl(0);
+ Exec_root* execRoot = static_cast<Exec_root*>(m_stmtArea.m_execTree);
+ ctx_assert(execRoot != 0);
+ rebind(ctx);
+ if (! ctx.ok())
+ return;
+ execRoot->execute(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ ctx_log2(("Executor: execute done"));
+}
+
+void
+Executor::fetch(Ctx& ctx)
+{
+ Exec_base::Ctl ctl(0);
+ Exec_root* execRoot = static_cast<Exec_root*>(m_stmtArea.m_execTree);
+ ctx_assert(execRoot != 0);
+ rebind(ctx);
+ if (! ctx.ok())
+ return;
+ execRoot->fetch(ctx, ctl);
+ if (! ctx.ok())
+ return;
+ ctx_log2(("Executor: fetch done"));
+}
+
+void
+Executor::rebind(Ctx& ctx)
+{
+ Exec_root* execRoot = static_cast<Exec_root*>(m_stmtArea.m_execTree);
+ ctx_assert(execRoot != 0);
+ DescArea& apd = m_stmtArea.descArea(Desc_usage_APD);
+ DescArea& ard = m_stmtArea.descArea(Desc_usage_ARD);
+ if (! apd.isBound() || ! ard.isBound()) {
+ ctx_log2(("Executor: rebind required"));
+ execRoot->bind(ctx);
+ if (! ctx.ok())
+ return;
+ apd.setBound(true);
+ ard.setBound(true);
+ }
+}
diff --git a/ndb/src/client/odbc/executor/Executor.hpp b/ndb/src/client/odbc/executor/Executor.hpp
new file mode 100644
index 00000000000..5edb9d509ac
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Executor.hpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_EXECUTOR_Executor_hpp
+#define ODBC_EXECUTOR_Executor_hpp
+
+#include <common/common.hpp>
+class StmtArea;
+
+/**
+ * @class Executor
+ * @brief Executes an ExecTree
+ */
+class Executor {
+public:
+ Executor(StmtArea& stmtArea);
+ ~Executor();
+ // execute prepared statement
+ void execute(Ctx& ctx);
+ // fetch next row from query
+ void fetch(Ctx& ctx);
+private:
+ // rebind if necessary
+ void rebind(Ctx& ctx);
+ StmtArea m_stmtArea;
+};
+
+inline
+Executor::Executor(StmtArea& stmtArea) :
+ m_stmtArea(stmtArea)
+{
+}
+
+inline
+Executor::~Executor()
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/executor/Makefile b/ndb/src/client/odbc/executor/Makefile
new file mode 100644
index 00000000000..d86781e212c
--- /dev/null
+++ b/ndb/src/client/odbc/executor/Makefile
@@ -0,0 +1,36 @@
+include .defs.mk
+
+TYPE = *
+
+NONPIC_ARCHIVE = N
+
+PIC_ARCHIVE = Y
+
+ARCHIVE_TARGET = odbcexecutor
+
+SOURCES = \
+ Executor.cpp \
+ Exec_query_lookup.cpp \
+ Exec_query_index.cpp \
+ Exec_query_scan.cpp \
+ Exec_query_range.cpp \
+ Exec_query_sys.cpp \
+ Exec_pred_op.cpp \
+ Exec_comp_op.cpp \
+ Exec_expr_op.cpp \
+ Exec_expr_func.cpp \
+ Exec_expr_conv.cpp \
+ Exec_insert.cpp \
+ Exec_update_lookup.cpp \
+ Exec_update_index.cpp \
+ Exec_update_scan.cpp \
+ Exec_delete_lookup.cpp \
+ Exec_delete_index.cpp \
+ Exec_delete_scan.cpp \
+ Exec_create_table.cpp \
+ Exec_create_index.cpp \
+ Exec_drop_table.cpp \
+ Exec_drop_index.cpp
+
+include ../Extra.mk
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/client/odbc/handles/AttrDbc.cpp b/ndb/src/client/odbc/handles/AttrDbc.cpp
new file mode 100644
index 00000000000..4768a8995a2
--- /dev/null
+++ b/ndb/src/client/odbc/handles/AttrDbc.cpp
@@ -0,0 +1,473 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HandleDbc.hpp"
+
+static void
+callback_SQL_ATTR_ACCESS_MODE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_MODE_READ_ONLY:
+ break;
+ case SQL_MODE_READ_WRITE:
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid access mode value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_ACCESS_MODE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = SQL_MODE_READ_WRITE;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ASYNC_ENABLE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_ASYNC_ENABLE_OFF:
+ break;
+ case SQL_ASYNC_ENABLE_ON:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "async enable on not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid async enable value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_ASYNC_ENABLE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = SQL_ASYNC_ENABLE_OFF;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_AUTO_IPD_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+ ctx_assert(false); // read-only
+}
+
+static void
+callback_SQL_ATTR_AUTO_IPD_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = SQL_FALSE;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_AUTOCOMMIT_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_AUTOCOMMIT_OFF:
+ if (pDbc->autocommit()) {
+ pDbc->autocommit(false);
+ pDbc->useConnection(ctx, true);
+ }
+ break;
+ case SQL_AUTOCOMMIT_ON:
+ if (! pDbc->autocommit()) {
+ pDbc->autocommit(true);
+ pDbc->sqlEndTran(ctx, SQL_COMMIT);
+ pDbc->useConnection(ctx, false);
+ }
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid autocommit value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_AUTOCOMMIT_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = pDbc->autocommit() ? SQL_AUTOCOMMIT_ON : SQL_AUTOCOMMIT_OFF;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_CONNECTION_DEAD_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+ ctx_assert(false); // read-only
+}
+
+static void
+callback_SQL_ATTR_CONNECTION_DEAD_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = pDbc->getState() == HandleDbc::Free ? SQL_CD_TRUE : SQL_CD_FALSE;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_CONNECTION_TIMEOUT_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+}
+
+static void
+callback_SQL_ATTR_CONNECTION_TIMEOUT_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_CURRENT_CATALOG_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_ATTR_CURRENT_CATALOG_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ const char* value = "DEFAULT";
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_LOGIN_TIMEOUT_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ callback_SQL_ATTR_CONNECTION_TIMEOUT_set(ctx, self, data);
+}
+
+static void
+callback_SQL_ATTR_LOGIN_TIMEOUT_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ callback_SQL_ATTR_CONNECTION_TIMEOUT_default(ctx, self, data);
+}
+
+static void
+callback_SQL_ATTR_METADATA_ID_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_FALSE:
+ break;
+ case SQL_TRUE:
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid metadata id value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_METADATA_ID_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = SQL_FALSE;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ODBC_CURSORS_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_CUR_USE_DRIVER:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "cur use driver not supported");
+ break;
+ case SQL_CUR_USE_IF_NEEDED:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "cur use if needed not supported");
+ break;
+ case SQL_CUR_USE_ODBC:
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid odbc cursors value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_ODBC_CURSORS_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = SQL_CUR_USE_ODBC;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_PACKET_SIZE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "packet size (%u) not supported", (unsigned)value);
+}
+
+static void
+callback_SQL_ATTR_PACKET_SIZE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_QUIET_MODE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Pointer);
+}
+
+static void
+callback_SQL_ATTR_QUIET_MODE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_ATTR_TRACE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+}
+
+static void
+callback_SQL_ATTR_TRACE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_ATTR_TRACEFILE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_ATTR_TRACEFILE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_ATTR_TRANSLATE_LIB_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_ATTR_TRANSLATE_LIB_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_ATTR_TRANSLATE_OPTION_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+}
+
+static void
+callback_SQL_ATTR_TRANSLATE_OPTION_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_ATTR_TXN_ISOLATION_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0 && data.type() == OdbcData::Uinteger);
+ if (pDbc->getState() == HandleDbc::Free) {
+ ctx.pushStatus(Sqlstate::_08003, Error::Gen, "not connected");
+ return;
+ }
+ if (pDbc->getState() == HandleDbc::Transacting) {
+ ctx.pushStatus(Sqlstate::_HY011, Error::Gen, "transaction is open");
+ return;
+ }
+ SQLUINTEGER value = data.uinteger();
+ SQLUINTEGER mask = SQL_TXN_READ_COMMITTED;
+ if (! (value & mask)) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "txn isolation level %u not supported", (unsigned)value);
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_TXN_ISOLATION_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDbc* pDbc = static_cast<HandleDbc*>(self);
+ ctx_assert(pDbc != 0);
+ SQLUINTEGER value = SQL_TXN_READ_COMMITTED;
+ data.setValue(value);
+}
+
+AttrSpec HandleDbc::m_attrSpec[] = {
+ { SQL_ATTR_ACCESS_MODE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ACCESS_MODE_set,
+ callback_SQL_ATTR_ACCESS_MODE_default,
+ },
+ { SQL_ATTR_ASYNC_ENABLE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ASYNC_ENABLE_set,
+ callback_SQL_ATTR_ASYNC_ENABLE_default,
+ },
+ { SQL_ATTR_AUTO_IPD,
+ OdbcData::Uinteger,
+ Attr_mode_readonly,
+ callback_SQL_ATTR_AUTO_IPD_set,
+ callback_SQL_ATTR_AUTO_IPD_default,
+ },
+ { SQL_ATTR_AUTOCOMMIT,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_AUTOCOMMIT_set,
+ callback_SQL_ATTR_AUTOCOMMIT_default,
+ },
+ { SQL_ATTR_CONNECTION_DEAD,
+ OdbcData::Uinteger,
+ Attr_mode_readonly,
+ callback_SQL_ATTR_CONNECTION_DEAD_set,
+ callback_SQL_ATTR_CONNECTION_DEAD_default,
+ },
+ { SQL_ATTR_CONNECTION_TIMEOUT,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_CONNECTION_TIMEOUT_set,
+ callback_SQL_ATTR_CONNECTION_TIMEOUT_default,
+ },
+ { SQL_ATTR_CURRENT_CATALOG,
+ OdbcData::Sqlchar,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_CURRENT_CATALOG_set,
+ callback_SQL_ATTR_CURRENT_CATALOG_default,
+ },
+ { SQL_ATTR_LOGIN_TIMEOUT,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_LOGIN_TIMEOUT_set,
+ callback_SQL_ATTR_LOGIN_TIMEOUT_default,
+ },
+ { SQL_ATTR_METADATA_ID,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_METADATA_ID_set,
+ callback_SQL_ATTR_METADATA_ID_default,
+ },
+ { SQL_ATTR_ODBC_CURSORS,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ODBC_CURSORS_set,
+ callback_SQL_ATTR_ODBC_CURSORS_default,
+ },
+ { SQL_ATTR_PACKET_SIZE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_PACKET_SIZE_set,
+ callback_SQL_ATTR_PACKET_SIZE_default,
+ },
+ { SQL_ATTR_QUIET_MODE,
+ OdbcData::Pointer,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_QUIET_MODE_set,
+ callback_SQL_ATTR_QUIET_MODE_default,
+ },
+ { SQL_ATTR_TRACE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_TRACE_set,
+ callback_SQL_ATTR_TRACE_default,
+ },
+ { SQL_ATTR_TRACEFILE,
+ OdbcData::Sqlchar,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_TRACEFILE_set,
+ callback_SQL_ATTR_TRACEFILE_default,
+ },
+ { SQL_ATTR_TRANSLATE_LIB,
+ OdbcData::Sqlchar,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_TRANSLATE_LIB_set,
+ callback_SQL_ATTR_TRANSLATE_LIB_default,
+ },
+ { SQL_ATTR_TRANSLATE_OPTION,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_TRANSLATE_OPTION_set,
+ callback_SQL_ATTR_TRANSLATE_OPTION_default,
+ },
+ { SQL_ATTR_TXN_ISOLATION,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_TXN_ISOLATION_set,
+ callback_SQL_ATTR_TXN_ISOLATION_default,
+ },
+ { 0,
+ OdbcData::Undef,
+ Attr_mode_undef,
+ 0,
+ 0,
+ },
+};
diff --git a/ndb/src/client/odbc/handles/AttrEnv.cpp b/ndb/src/client/odbc/handles/AttrEnv.cpp
new file mode 100644
index 00000000000..3d57fddeb57
--- /dev/null
+++ b/ndb/src/client/odbc/handles/AttrEnv.cpp
@@ -0,0 +1,123 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HandleEnv.hpp"
+
+static void
+callback_SQL_ATTR_CP_MATCH_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleEnv* pEnv = static_cast<HandleEnv*>(self);
+ ctx_assert(pEnv != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_CP_STRICT_MATCH:
+ break;
+ case SQL_CP_RELAXED_MATCH:
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid cp match value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_CP_MATCH_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleEnv* pEnv = static_cast<HandleEnv*>(self);
+ ctx_assert(pEnv != 0);
+ SQLUINTEGER value = SQL_CP_STRICT_MATCH;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ODBC_VERSION_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleEnv* pEnv = static_cast<HandleEnv*>(self);
+ ctx_assert(pEnv != 0 && data.type() == OdbcData::Integer);
+ int version = data.integer();
+ switch (version) {
+ case SQL_OV_ODBC2:
+ case SQL_OV_ODBC3:
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid ODBC version %d", version);
+ return;
+ }
+ ctx_log1(("odbc version set to %d", version));
+}
+
+static void
+callback_SQL_ATTR_ODBC_VERSION_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleEnv* pEnv = static_cast<HandleEnv*>(self);
+ ctx_assert(pEnv != 0);
+ // no default
+ ctx_log1(("odbc version has not been set"));
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "odbc version has not been set");
+}
+
+static void
+callback_SQL_ATTR_OUTPUT_NTS_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleEnv* pEnv = static_cast<HandleEnv*>(self);
+ ctx_assert(pEnv != 0 && data.type() == OdbcData::Integer);
+ SQLINTEGER value = data.integer();
+ switch (value) {
+ case SQL_TRUE:
+ break;
+ case SQL_FALSE:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "output nts FALSE not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid output nts value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_OUTPUT_NTS_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleEnv* pEnv = static_cast<HandleEnv*>(self);
+ ctx_assert(pEnv != 0);
+ data.setValue();
+}
+
+AttrSpec HandleEnv::m_attrSpec[] = {
+ { SQL_ATTR_CP_MATCH,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_CP_MATCH_set,
+ callback_SQL_ATTR_CP_MATCH_default,
+ },
+ { SQL_ATTR_ODBC_VERSION,
+ OdbcData::Integer,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ODBC_VERSION_set,
+ callback_SQL_ATTR_ODBC_VERSION_default,
+ },
+ { SQL_ATTR_OUTPUT_NTS,
+ OdbcData::Integer,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_OUTPUT_NTS_set,
+ callback_SQL_ATTR_OUTPUT_NTS_default,
+ },
+ { 0,
+ OdbcData::Undef,
+ Attr_mode_undef,
+ 0,
+ 0,
+ },
+};
diff --git a/ndb/src/client/odbc/handles/AttrRoot.cpp b/ndb/src/client/odbc/handles/AttrRoot.cpp
new file mode 100644
index 00000000000..d1b264835b6
--- /dev/null
+++ b/ndb/src/client/odbc/handles/AttrRoot.cpp
@@ -0,0 +1,92 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HandleRoot.hpp"
+
+static void
+callback_SQL_ATTR_CONNECTION_POOLING_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleRoot* pRoot = static_cast<HandleRoot*>(self);
+ ctx_assert(pRoot != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_CP_OFF:
+ break;
+ case SQL_CP_ONE_PER_DRIVER:
+ case SQL_CP_ONE_PER_HENV:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "connection pooling not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid connection pooling value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_CONNECTION_POOLING_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleRoot* pRoot = static_cast<HandleRoot*>(self);
+ ctx_assert(pRoot != 0);
+ SQLUINTEGER value = SQL_CP_OFF;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_CP_MATCH_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleRoot* pRoot = static_cast<HandleRoot*>(self);
+ ctx_assert(pRoot != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_CP_STRICT_MATCH:
+ break;
+ case SQL_CP_RELAXED_MATCH:
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid cp match value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_CP_MATCH_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleRoot* pRoot = static_cast<HandleRoot*>(self);
+ ctx_assert(pRoot != 0);
+ SQLUINTEGER value = SQL_CP_STRICT_MATCH;
+ data.setValue(value);
+}
+
+AttrSpec HandleRoot::m_attrSpec[] = {
+ { SQL_ATTR_CONNECTION_POOLING,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_CONNECTION_POOLING_set,
+ callback_SQL_ATTR_CONNECTION_POOLING_default,
+ },
+ { SQL_ATTR_CP_MATCH,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_CP_MATCH_set,
+ callback_SQL_ATTR_CP_MATCH_default,
+ },
+ { 0,
+ OdbcData::Undef,
+ Attr_mode_undef,
+ 0,
+ 0,
+ },
+};
diff --git a/ndb/src/client/odbc/handles/AttrStmt.cpp b/ndb/src/client/odbc/handles/AttrStmt.cpp
new file mode 100644
index 00000000000..ce9a9c03fd1
--- /dev/null
+++ b/ndb/src/client/odbc/handles/AttrStmt.cpp
@@ -0,0 +1,1005 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HandleDbc.hpp"
+#include "HandleStmt.hpp"
+#include "HandleDesc.hpp"
+
+static void
+callback_SQL_ATTR_APP_PARAM_DESC_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Pointer);
+ pStmt->setHandleDesc(ctx, Desc_usage_APD, data.pointer());
+}
+
+static void
+callback_SQL_ATTR_APP_PARAM_DESC_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ HandleDesc* apd = pStmt->getHandleDesc(ctx, Desc_usage_APD);
+ OdbcData value(reinterpret_cast<SQLPOINTER>(apd));
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_APP_ROW_DESC_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Pointer);
+ pStmt->setHandleDesc(ctx, Desc_usage_ARD, data.pointer());
+}
+
+static void
+callback_SQL_ATTR_APP_ROW_DESC_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ HandleDesc* ard = pStmt->getHandleDesc(ctx, Desc_usage_ARD);
+ OdbcData value(reinterpret_cast<SQLPOINTER>(ard));
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ASYNC_ENABLE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_ASYNC_ENABLE_OFF:
+ break;
+ case SQL_ASYNC_ENABLE_ON:
+// ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "async enable ON not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid async enable value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_ASYNC_ENABLE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_ASYNC_ENABLE_OFF;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_CONCURRENCY_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_CONCUR_READ_ONLY:
+ break;
+ case SQL_CONCUR_LOCK:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "concur lock not supported");
+ break;
+ case SQL_CONCUR_ROWVER:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "concur rowver not supported");
+ break;
+ case SQL_CONCUR_VALUES:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "concur values not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid concurrency value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_CONCURRENCY_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_CONCUR_READ_ONLY;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_CURSOR_SCROLLABLE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_NONSCROLLABLE:
+ break;
+ case SQL_SCROLLABLE:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "cursor scrollable not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid concurrency value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_CURSOR_SCROLLABLE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_NONSCROLLABLE;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_CURSOR_SENSITIVITY_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_UNSPECIFIED:
+ case SQL_INSENSITIVE:
+ break;
+ case SQL_SENSITIVE:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "cursor sensitive not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid cursor sensitivity value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_CURSOR_SENSITIVITY_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_INSENSITIVE;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_CURSOR_TYPE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_CURSOR_FORWARD_ONLY:
+ break;
+ case SQL_CURSOR_STATIC:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "cursor static not supported");
+ break;
+ case SQL_CURSOR_KEYSET_DRIVEN:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "cursor keyset driven not supported");
+ break;
+ case SQL_CURSOR_DYNAMIC:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "cursor dynamic not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid cursor type value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_CURSOR_TYPE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_CURSOR_FORWARD_ONLY;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ENABLE_AUTO_IPD_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_FALSE:
+ break;
+ case SQL_TRUE:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "enable auto IPD not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid enable auto IPD value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_ENABLE_AUTO_IPD_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_FALSE;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_FETCH_BOOKMARK_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Pointer);
+ SQLPOINTER value = data.pointer();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "fetch bookmark ptr not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_FETCH_BOOKMARK_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLPOINTER value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_IMP_PARAM_DESC_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Pointer);
+ ctx_assert(false); // read-only
+}
+
+static void
+callback_SQL_ATTR_IMP_PARAM_DESC_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ HandleDesc* ipd = pStmt->getHandleDesc(ctx, Desc_usage_IPD);
+ OdbcData value(reinterpret_cast<SQLPOINTER>(ipd));
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_IMP_ROW_DESC_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Pointer);
+ ctx_assert(false); // read-only
+}
+
+static void
+callback_SQL_ATTR_IMP_ROW_DESC_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ HandleDesc* ird = pStmt->getHandleDesc(ctx, Desc_usage_IRD);
+ OdbcData value(reinterpret_cast<SQLPOINTER>(ird));
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_KEYSET_SIZE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "keyset size not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_KEYSET_SIZE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_MAX_LENGTH_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "max length not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_MAX_LENGTH_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_MAX_ROWS_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "max rows not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_MAX_ROWS_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_METADATA_ID_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_FALSE:
+ break;
+ case SQL_TRUE:
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid metadata id value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_METADATA_ID_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_FALSE;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_NOSCAN_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_NOSCAN_OFF:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "noscan OFF not supported");
+ break;
+ case SQL_NOSCAN_ON:
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid no scan value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_NOSCAN_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_NOSCAN_ON;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_PARAM_BIND_OFFSET_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::UintegerPtr);
+ SQLUINTEGER* value = data.uintegerPtr();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "param bind offset ptr not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_PARAM_BIND_OFFSET_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER* value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_PARAM_BIND_TYPE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ if (value != SQL_PARAM_BIND_BY_COLUMN) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "row-wise param binding not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_PARAM_BIND_TYPE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_PARAM_BIND_BY_COLUMN;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_PARAM_OPERATION_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::UsmallintPtr);
+ SQLUSMALLINT* value = data.usmallintPtr();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "param operation ptr not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_PARAM_OPERATION_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUSMALLINT* value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_PARAM_STATUS_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::UsmallintPtr);
+ SQLUSMALLINT* value = data.usmallintPtr();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "param status ptr not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_PARAM_STATUS_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUSMALLINT* value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_PARAMS_PROCESSED_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::UintegerPtr);
+ SQLUINTEGER* value = data.uintegerPtr();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "params processed ptr not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_PARAMS_PROCESSED_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER* value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_PARAMSET_SIZE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ if (value != 1) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "paramset size %u not supported", (unsigned)value);
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_PARAMSET_SIZE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = 1;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_QUERY_TIMEOUT_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_01S02, Error::Gen, "query timeout %u replaced by 0", (unsigned)value);
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_QUERY_TIMEOUT_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_RETRIEVE_DATA_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_RD_ON:
+ break;
+ case SQL_RD_OFF:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "retrieve data OFF not supported");
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid retrieve data value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_RETRIEVE_DATA_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_RD_ON;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ROW_ARRAY_SIZE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ if (value != 1) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "row array size %u != 1 not supported", (unsigned)value);
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_ROW_ARRAY_SIZE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = 1;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ROW_BIND_OFFSET_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::UintegerPtr);
+ SQLUINTEGER* value = data.uintegerPtr();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "row bind offset ptr != 0 not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_ROW_BIND_OFFSET_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER* value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ROW_BIND_TYPE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ if (value != SQL_BIND_BY_COLUMN) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "row-wise binding not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_ROW_BIND_TYPE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_BIND_BY_COLUMN;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ROW_NUMBER_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ ctx_assert(false); // read-only
+}
+
+static void
+callback_SQL_ATTR_ROW_NUMBER_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = pStmt->getRowCount();
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ROW_OPERATION_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::UsmallintPtr);
+ SQLUSMALLINT* value = data.usmallintPtr();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "row operation ptr not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_ROW_OPERATION_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUSMALLINT* value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ROW_STATUS_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::UsmallintPtr);
+ SQLUSMALLINT* value = data.usmallintPtr();
+ if (value != 0) {
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "row status ptr not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_ROW_STATUS_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUSMALLINT* value = 0;
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_ROWS_FETCHED_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::UintegerPtr);
+ SQLUINTEGER* value = data.uintegerPtr();
+ HandleDesc* ird = pStmt->getHandleDesc(ctx, Desc_usage_IRD);
+ ird->sqlSetDescField(ctx, 0, SQL_DESC_ROWS_PROCESSED_PTR, static_cast<SQLPOINTER>(value), SQL_IS_POINTER);
+}
+
+static void
+callback_SQL_ATTR_ROWS_FETCHED_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER* value = 0;
+ HandleDesc* ird = pStmt->getHandleDesc(ctx, Desc_usage_IRD);
+ ird->sqlGetDescField(ctx, 0, SQL_DESC_ROWS_PROCESSED_PTR, &value, SQL_IS_POINTER, 0);
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_SIMULATE_CURSOR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_SC_NON_UNIQUE:
+ break;
+ case SQL_SC_TRY_UNIQUE:
+ break;
+ case SQL_SC_UNIQUE:
+ break;
+ default:
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "invalid simulate cursor value %u", (unsigned)value);
+ break;
+ }
+}
+
+static void
+callback_SQL_ATTR_SIMULATE_CURSOR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_SC_UNIQUE; // XXX if we did
+ data.setValue(value);
+}
+
+static void
+callback_SQL_ATTR_USE_BOOKMARKS_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ SQLUINTEGER value = data.uinteger();
+ switch (value) {
+ case SQL_UB_OFF:
+ break;
+ case SQL_UB_VARIABLE:
+ case SQL_UB_FIXED:
+ ctx.pushStatus(Sqlstate::_HYC00, Error::Gen, "bookmarks not supported");
+ return;
+ }
+}
+
+static void
+callback_SQL_ATTR_USE_BOOKMARKS_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = SQL_UB_OFF;
+ data.setValue(value);
+}
+
+// driver specific
+
+static void
+callback_SQL_ATTR_NDB_TUPLES_FETCHED_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0 && data.type() == OdbcData::Uinteger);
+ ctx_assert(false); // read-only
+}
+
+static void
+callback_SQL_ATTR_NDB_TUPLES_FETCHED_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleStmt* pStmt = static_cast<HandleStmt*>(self);
+ ctx_assert(pStmt != 0);
+ SQLUINTEGER value = pStmt->getTuplesFetched();
+ data.setValue(value);
+}
+
+AttrSpec HandleStmt::m_attrSpec[] = {
+ { SQL_ATTR_APP_PARAM_DESC,
+ OdbcData::Pointer,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_APP_PARAM_DESC_set,
+ callback_SQL_ATTR_APP_PARAM_DESC_default,
+ },
+ { SQL_ATTR_APP_ROW_DESC,
+ OdbcData::Pointer,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_APP_ROW_DESC_set,
+ callback_SQL_ATTR_APP_ROW_DESC_default,
+ },
+ { SQL_ATTR_ASYNC_ENABLE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ASYNC_ENABLE_set,
+ callback_SQL_ATTR_ASYNC_ENABLE_default,
+ },
+ { SQL_ATTR_CONCURRENCY,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_CONCURRENCY_set,
+ callback_SQL_ATTR_CONCURRENCY_default,
+ },
+ { SQL_ATTR_CURSOR_SCROLLABLE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_CURSOR_SCROLLABLE_set,
+ callback_SQL_ATTR_CURSOR_SCROLLABLE_default,
+ },
+ { SQL_ATTR_CURSOR_SENSITIVITY,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_CURSOR_SENSITIVITY_set,
+ callback_SQL_ATTR_CURSOR_SENSITIVITY_default,
+ },
+ { SQL_ATTR_CURSOR_TYPE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_CURSOR_TYPE_set,
+ callback_SQL_ATTR_CURSOR_TYPE_default,
+ },
+ { SQL_ATTR_ENABLE_AUTO_IPD,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ENABLE_AUTO_IPD_set,
+ callback_SQL_ATTR_ENABLE_AUTO_IPD_default,
+ },
+ { SQL_ATTR_FETCH_BOOKMARK_PTR,
+ OdbcData::Pointer,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_FETCH_BOOKMARK_PTR_set,
+ callback_SQL_ATTR_FETCH_BOOKMARK_PTR_default,
+ },
+ { SQL_ATTR_IMP_PARAM_DESC,
+ OdbcData::Pointer,
+ Attr_mode_readonly,
+ callback_SQL_ATTR_IMP_PARAM_DESC_set,
+ callback_SQL_ATTR_IMP_PARAM_DESC_default,
+ },
+ { SQL_ATTR_IMP_ROW_DESC,
+ OdbcData::Pointer,
+ Attr_mode_readonly,
+ callback_SQL_ATTR_IMP_ROW_DESC_set,
+ callback_SQL_ATTR_IMP_ROW_DESC_default,
+ },
+ { SQL_ATTR_KEYSET_SIZE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_KEYSET_SIZE_set,
+ callback_SQL_ATTR_KEYSET_SIZE_default,
+ },
+ { SQL_ATTR_MAX_LENGTH,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_MAX_LENGTH_set,
+ callback_SQL_ATTR_MAX_LENGTH_default,
+ },
+ { SQL_ATTR_MAX_ROWS,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_MAX_ROWS_set,
+ callback_SQL_ATTR_MAX_ROWS_default,
+ },
+ { SQL_ATTR_METADATA_ID,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_METADATA_ID_set,
+ callback_SQL_ATTR_METADATA_ID_default,
+ },
+ { SQL_ATTR_NOSCAN,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_NOSCAN_set,
+ callback_SQL_ATTR_NOSCAN_default,
+ },
+ { SQL_ATTR_PARAM_BIND_OFFSET_PTR,
+ OdbcData::UintegerPtr,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_PARAM_BIND_OFFSET_PTR_set,
+ callback_SQL_ATTR_PARAM_BIND_OFFSET_PTR_default,
+ },
+ { SQL_ATTR_PARAM_BIND_TYPE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_PARAM_BIND_TYPE_set,
+ callback_SQL_ATTR_PARAM_BIND_TYPE_default,
+ },
+ { SQL_ATTR_PARAM_OPERATION_PTR,
+ OdbcData::UsmallintPtr,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_PARAM_OPERATION_PTR_set,
+ callback_SQL_ATTR_PARAM_OPERATION_PTR_default,
+ },
+ { SQL_ATTR_PARAM_STATUS_PTR,
+ OdbcData::UsmallintPtr,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_PARAM_STATUS_PTR_set,
+ callback_SQL_ATTR_PARAM_STATUS_PTR_default,
+ },
+ { SQL_ATTR_PARAMS_PROCESSED_PTR,
+ OdbcData::UintegerPtr,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_PARAMS_PROCESSED_PTR_set,
+ callback_SQL_ATTR_PARAMS_PROCESSED_PTR_default,
+ },
+ { SQL_ATTR_PARAMSET_SIZE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_PARAMSET_SIZE_set,
+ callback_SQL_ATTR_PARAMSET_SIZE_default,
+ },
+ { SQL_ATTR_QUERY_TIMEOUT,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_QUERY_TIMEOUT_set,
+ callback_SQL_ATTR_QUERY_TIMEOUT_default,
+ },
+ { SQL_ATTR_RETRIEVE_DATA,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_RETRIEVE_DATA_set,
+ callback_SQL_ATTR_RETRIEVE_DATA_default,
+ },
+ { SQL_ATTR_ROW_ARRAY_SIZE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ROW_ARRAY_SIZE_set,
+ callback_SQL_ATTR_ROW_ARRAY_SIZE_default,
+ },
+ { SQL_ATTR_ROW_BIND_OFFSET_PTR,
+ OdbcData::UintegerPtr,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ROW_BIND_OFFSET_PTR_set,
+ callback_SQL_ATTR_ROW_BIND_OFFSET_PTR_default,
+ },
+ { SQL_ATTR_ROW_BIND_TYPE,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ROW_BIND_TYPE_set,
+ callback_SQL_ATTR_ROW_BIND_TYPE_default,
+ },
+ { SQL_ATTR_ROW_NUMBER,
+ OdbcData::Uinteger,
+ Attr_mode_readonly,
+ callback_SQL_ATTR_ROW_NUMBER_set,
+ callback_SQL_ATTR_ROW_NUMBER_default,
+ },
+ { SQL_ATTR_ROW_OPERATION_PTR,
+ OdbcData::UsmallintPtr,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ROW_OPERATION_PTR_set,
+ callback_SQL_ATTR_ROW_OPERATION_PTR_default,
+ },
+ { SQL_ATTR_ROW_STATUS_PTR,
+ OdbcData::UsmallintPtr,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ROW_STATUS_PTR_set,
+ callback_SQL_ATTR_ROW_STATUS_PTR_default,
+ },
+ { SQL_ATTR_ROWS_FETCHED_PTR,
+ OdbcData::UintegerPtr,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_ROWS_FETCHED_PTR_set,
+ callback_SQL_ATTR_ROWS_FETCHED_PTR_default,
+ },
+ { SQL_ATTR_SIMULATE_CURSOR,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_SIMULATE_CURSOR_set,
+ callback_SQL_ATTR_SIMULATE_CURSOR_default,
+ },
+ { SQL_ATTR_USE_BOOKMARKS,
+ OdbcData::Uinteger,
+ Attr_mode_readwrite,
+ callback_SQL_ATTR_USE_BOOKMARKS_set,
+ callback_SQL_ATTR_USE_BOOKMARKS_default,
+ },
+ // driver specific
+ { SQL_ATTR_NDB_TUPLES_FETCHED,
+ OdbcData::Uinteger,
+ Attr_mode_readonly,
+ callback_SQL_ATTR_NDB_TUPLES_FETCHED_set,
+ callback_SQL_ATTR_NDB_TUPLES_FETCHED_default,
+ },
+ { 0,
+ OdbcData::Undef,
+ Attr_mode_undef,
+ 0,
+ 0,
+ },
+};
diff --git a/ndb/src/client/odbc/handles/DescSpec.cpp b/ndb/src/client/odbc/handles/DescSpec.cpp
new file mode 100644
index 00000000000..83905cf9822
--- /dev/null
+++ b/ndb/src/client/odbc/handles/DescSpec.cpp
@@ -0,0 +1,1140 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HandleDbc.hpp"
+#include "HandleDesc.hpp"
+
+static void
+callback_SQL_DESC_ALLOC_TYPE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_ALLOC_TYPE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_ARRAY_SIZE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Uinteger);
+}
+
+static void
+callback_SQL_DESC_ARRAY_SIZE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_ARRAY_STATUS_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::UsmallintPtr);
+}
+
+static void
+callback_SQL_DESC_ARRAY_STATUS_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_BIND_OFFSET_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::IntegerPtr);
+}
+
+static void
+callback_SQL_DESC_BIND_OFFSET_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_BIND_TYPE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Integer);
+}
+
+static void
+callback_SQL_DESC_BIND_TYPE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_COUNT_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_COUNT_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_ROWS_PROCESSED_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::UintegerPtr);
+}
+
+static void
+callback_SQL_DESC_ROWS_PROCESSED_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_AUTO_UNIQUE_VALUE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Integer);
+}
+
+static void
+callback_SQL_DESC_AUTO_UNIQUE_VALUE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_BASE_COLUMN_NAME_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_BASE_COLUMN_NAME_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_BASE_TABLE_NAME_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_BASE_TABLE_NAME_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_CASE_SENSITIVE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Integer);
+}
+
+static void
+callback_SQL_DESC_CASE_SENSITIVE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_CATALOG_NAME_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_CATALOG_NAME_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_CONCISE_TYPE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_CONCISE_TYPE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_DATA_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Pointer);
+}
+
+static void
+callback_SQL_DESC_DATA_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_DATETIME_INTERVAL_CODE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_DATETIME_INTERVAL_CODE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_DATETIME_INTERVAL_PRECISION_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Integer);
+}
+
+static void
+callback_SQL_DESC_DATETIME_INTERVAL_PRECISION_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_DISPLAY_SIZE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Integer);
+}
+
+static void
+callback_SQL_DESC_DISPLAY_SIZE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_FIXED_PREC_SCALE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_FIXED_PREC_SCALE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_INDICATOR_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::IntegerPtr);
+}
+
+static void
+callback_SQL_DESC_INDICATOR_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_LABEL_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_LABEL_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_LENGTH_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Uinteger);
+}
+
+static void
+callback_SQL_DESC_LENGTH_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_LITERAL_PREFIX_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_LITERAL_PREFIX_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_LITERAL_SUFFIX_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_LITERAL_SUFFIX_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_LOCAL_TYPE_NAME_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_LOCAL_TYPE_NAME_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_NAME_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_NAME_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_NULLABLE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_NULLABLE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_NUM_PREC_RADIX_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Integer);
+}
+
+static void
+callback_SQL_DESC_NUM_PREC_RADIX_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_OCTET_LENGTH_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Integer);
+}
+
+static void
+callback_SQL_DESC_OCTET_LENGTH_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_OCTET_LENGTH_PTR_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::IntegerPtr);
+}
+
+static void
+callback_SQL_DESC_OCTET_LENGTH_PTR_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_PARAMETER_TYPE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_PARAMETER_TYPE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_PRECISION_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_PRECISION_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_ROWVER_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_ROWVER_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_SCALE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_SCALE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_SCHEMA_NAME_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_SCHEMA_NAME_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_SEARCHABLE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_SEARCHABLE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_TABLE_NAME_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_TABLE_NAME_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_TYPE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_TYPE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_TYPE_NAME_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Sqlchar);
+}
+
+static void
+callback_SQL_DESC_TYPE_NAME_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_UNNAMED_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_UNNAMED_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_UNSIGNED_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_UNSIGNED_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+static void
+callback_SQL_DESC_UPDATABLE_set(Ctx& ctx, HandleBase* self, const OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0 && data.type() == OdbcData::Smallint);
+}
+
+static void
+callback_SQL_DESC_UPDATABLE_default(Ctx& ctx, HandleBase* self, OdbcData& data)
+{
+ HandleDesc* pDesc = static_cast<HandleDesc*>(self);
+ ctx_assert(pDesc != 0);
+ data.setValue();
+}
+
+DescSpec HandleDesc::m_descSpec[] = {
+ { Desc_pos_header,
+ SQL_DESC_ALLOC_TYPE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readonly,
+ Desc_mode_readonly,
+ Desc_mode_readonly,
+ Desc_mode_readonly,
+ },
+ callback_SQL_DESC_ALLOC_TYPE_set,
+ callback_SQL_DESC_ALLOC_TYPE_default,
+ },
+ { Desc_pos_header,
+ SQL_DESC_ARRAY_SIZE,
+ OdbcData::Uinteger,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_ARRAY_SIZE_set,
+ callback_SQL_DESC_ARRAY_SIZE_default,
+ },
+ { Desc_pos_header,
+ SQL_DESC_ARRAY_STATUS_PTR,
+ OdbcData::UsmallintPtr,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_ARRAY_STATUS_PTR_set,
+ callback_SQL_DESC_ARRAY_STATUS_PTR_default,
+ },
+ { Desc_pos_header,
+ SQL_DESC_BIND_OFFSET_PTR,
+ OdbcData::IntegerPtr,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_BIND_OFFSET_PTR_set,
+ callback_SQL_DESC_BIND_OFFSET_PTR_default,
+ },
+ { Desc_pos_header,
+ SQL_DESC_BIND_TYPE,
+ OdbcData::Integer,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_BIND_TYPE_set,
+ callback_SQL_DESC_BIND_TYPE_default,
+ },
+ { Desc_pos_header,
+ SQL_DESC_COUNT,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_COUNT_set,
+ callback_SQL_DESC_COUNT_default,
+ },
+ { Desc_pos_header,
+ SQL_DESC_ROWS_PROCESSED_PTR,
+ OdbcData::UintegerPtr,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_readwrite,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_ROWS_PROCESSED_PTR_set,
+ callback_SQL_DESC_ROWS_PROCESSED_PTR_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_AUTO_UNIQUE_VALUE,
+ OdbcData::Integer,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_AUTO_UNIQUE_VALUE_set,
+ callback_SQL_DESC_AUTO_UNIQUE_VALUE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_BASE_COLUMN_NAME,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_BASE_COLUMN_NAME_set,
+ callback_SQL_DESC_BASE_COLUMN_NAME_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_BASE_TABLE_NAME,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_BASE_TABLE_NAME_set,
+ callback_SQL_DESC_BASE_TABLE_NAME_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_CASE_SENSITIVE,
+ OdbcData::Integer,
+ { Desc_mode_undef,
+ Desc_mode_readonly,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_CASE_SENSITIVE_set,
+ callback_SQL_DESC_CASE_SENSITIVE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_CATALOG_NAME,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_CATALOG_NAME_set,
+ callback_SQL_DESC_CATALOG_NAME_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_CONCISE_TYPE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_CONCISE_TYPE_set,
+ callback_SQL_DESC_CONCISE_TYPE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_DATA_PTR,
+ OdbcData::Pointer,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_DATA_PTR_set,
+ callback_SQL_DESC_DATA_PTR_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_DATETIME_INTERVAL_CODE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_DATETIME_INTERVAL_CODE_set,
+ callback_SQL_DESC_DATETIME_INTERVAL_CODE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_DATETIME_INTERVAL_PRECISION,
+ OdbcData::Integer,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_DATETIME_INTERVAL_PRECISION_set,
+ callback_SQL_DESC_DATETIME_INTERVAL_PRECISION_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_DISPLAY_SIZE,
+ OdbcData::Integer,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_DISPLAY_SIZE_set,
+ callback_SQL_DESC_DISPLAY_SIZE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_FIXED_PREC_SCALE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readonly,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_FIXED_PREC_SCALE_set,
+ callback_SQL_DESC_FIXED_PREC_SCALE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_INDICATOR_PTR,
+ OdbcData::IntegerPtr,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_INDICATOR_PTR_set,
+ callback_SQL_DESC_INDICATOR_PTR_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_LABEL,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_LABEL_set,
+ callback_SQL_DESC_LABEL_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_LENGTH,
+ OdbcData::Uinteger,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_LENGTH_set,
+ callback_SQL_DESC_LENGTH_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_LITERAL_PREFIX,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_LITERAL_PREFIX_set,
+ callback_SQL_DESC_LITERAL_PREFIX_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_LITERAL_SUFFIX,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_LITERAL_SUFFIX_set,
+ callback_SQL_DESC_LITERAL_SUFFIX_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_LOCAL_TYPE_NAME,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_readonly,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_LOCAL_TYPE_NAME_set,
+ callback_SQL_DESC_LOCAL_TYPE_NAME_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_NAME,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_NAME_set,
+ callback_SQL_DESC_NAME_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_NULLABLE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readonly,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_NULLABLE_set,
+ callback_SQL_DESC_NULLABLE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_NUM_PREC_RADIX,
+ OdbcData::Integer,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_NUM_PREC_RADIX_set,
+ callback_SQL_DESC_NUM_PREC_RADIX_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_OCTET_LENGTH,
+ OdbcData::Integer,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_OCTET_LENGTH_set,
+ callback_SQL_DESC_OCTET_LENGTH_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_OCTET_LENGTH_PTR,
+ OdbcData::IntegerPtr,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_OCTET_LENGTH_PTR_set,
+ callback_SQL_DESC_OCTET_LENGTH_PTR_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_PARAMETER_TYPE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_PARAMETER_TYPE_set,
+ callback_SQL_DESC_PARAMETER_TYPE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_PRECISION,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_PRECISION_set,
+ callback_SQL_DESC_PRECISION_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_ROWVER,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readonly,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_ROWVER_set,
+ callback_SQL_DESC_ROWVER_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_SCALE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_SCALE_set,
+ callback_SQL_DESC_SCALE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_SCHEMA_NAME,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_SCHEMA_NAME_set,
+ callback_SQL_DESC_SCHEMA_NAME_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_SEARCHABLE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_SEARCHABLE_set,
+ callback_SQL_DESC_SEARCHABLE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_TABLE_NAME,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_TABLE_NAME_set,
+ callback_SQL_DESC_TABLE_NAME_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_TYPE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_readwrite,
+ Desc_mode_readonly,
+ Desc_mode_readwrite
+ },
+ callback_SQL_DESC_TYPE_set,
+ callback_SQL_DESC_TYPE_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_TYPE_NAME,
+ OdbcData::Sqlchar,
+ { Desc_mode_undef,
+ Desc_mode_readonly,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_TYPE_NAME_set,
+ callback_SQL_DESC_TYPE_NAME_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_UNNAMED,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readwrite,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_UNNAMED_set,
+ callback_SQL_DESC_UNNAMED_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_UNSIGNED,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_readonly,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_UNSIGNED_set,
+ callback_SQL_DESC_UNSIGNED_default,
+ },
+ { Desc_pos_record,
+ SQL_DESC_UPDATABLE,
+ OdbcData::Smallint,
+ { Desc_mode_undef,
+ Desc_mode_unused,
+ Desc_mode_unused,
+ Desc_mode_readonly,
+ Desc_mode_unused
+ },
+ callback_SQL_DESC_UPDATABLE_set,
+ callback_SQL_DESC_UPDATABLE_default,
+ },
+ { Desc_pos_end,
+ 0,
+ OdbcData::Undef,
+ { Desc_mode_undef,
+ Desc_mode_undef,
+ Desc_mode_undef,
+ Desc_mode_undef,
+ Desc_mode_undef
+ },
+ 0,
+ 0
+ },
+};
diff --git a/ndb/src/client/odbc/handles/FuncTab.cpp b/ndb/src/client/odbc/handles/FuncTab.cpp
new file mode 100644
index 00000000000..6bd744d7a7f
--- /dev/null
+++ b/ndb/src/client/odbc/handles/FuncTab.cpp
@@ -0,0 +1,100 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HandleDbc.hpp"
+
+HandleDbc::FuncTab
+HandleDbc::m_funcTab[] = {
+ { SQL_API_SQLALLOCCONNECT , 1 },
+ { SQL_API_SQLALLOCENV , 1 },
+ { SQL_API_SQLALLOCHANDLE , 1 },
+ { SQL_API_SQLALLOCHANDLESTD , 0 },
+ { SQL_API_SQLALLOCSTMT , 1 },
+ { SQL_API_SQLBINDCOL , 1 },
+ { SQL_API_SQLBINDPARAM , 1 },
+ { SQL_API_SQLBINDPARAMETER , 1 },
+ { SQL_API_SQLBROWSECONNECT , 0 },
+ { SQL_API_SQLBULKOPERATIONS , 0 },
+ { SQL_API_SQLCANCEL , 1 },
+ { SQL_API_SQLCLOSECURSOR , 1 },
+ { SQL_API_SQLCOLATTRIBUTE , 1 },
+ { SQL_API_SQLCOLATTRIBUTES , 1 },
+ { SQL_API_SQLCOLUMNPRIVILEGES , 0 },
+ { SQL_API_SQLCOLUMNS , 1 },
+ { SQL_API_SQLCONNECT , 1 },
+ { SQL_API_SQLCOPYDESC , 0 },
+ { SQL_API_SQLDATASOURCES , 0 },
+ { SQL_API_SQLDESCRIBECOL , 1 },
+ { SQL_API_SQLDESCRIBEPARAM , 0 },
+ { SQL_API_SQLDISCONNECT , 1 },
+ { SQL_API_SQLDRIVERCONNECT , 1 },
+ { SQL_API_SQLDRIVERS , 0 },
+ { SQL_API_SQLENDTRAN , 1 },
+ { SQL_API_SQLERROR , 1 },
+ { SQL_API_SQLEXECDIRECT , 1 },
+ { SQL_API_SQLEXECUTE , 1 },
+ { SQL_API_SQLEXTENDEDFETCH , 0 },
+ { SQL_API_SQLFETCH , 1 },
+ { SQL_API_SQLFETCHSCROLL , 0 },
+ { SQL_API_SQLFOREIGNKEYS , 0 },
+ { SQL_API_SQLFREECONNECT , 1 },
+ { SQL_API_SQLFREEENV , 1 },
+ { SQL_API_SQLFREEHANDLE , 1 },
+ { SQL_API_SQLFREESTMT , 1 },
+ { SQL_API_SQLGETCONNECTATTR , 1 },
+ { SQL_API_SQLGETCONNECTOPTION , 1 },
+ { SQL_API_SQLGETCURSORNAME , 1 },
+ { SQL_API_SQLGETDATA , 1 },
+ { SQL_API_SQLGETDESCFIELD , 1 },
+ { SQL_API_SQLGETDESCREC , 1 },
+ { SQL_API_SQLGETDIAGFIELD , 1 },
+ { SQL_API_SQLGETDIAGREC , 1 },
+ { SQL_API_SQLGETENVATTR , 1 },
+ { SQL_API_SQLGETFUNCTIONS , 1 },
+ { SQL_API_SQLGETINFO , 1 },
+ { SQL_API_SQLGETSTMTATTR , 1 },
+ { SQL_API_SQLGETSTMTOPTION , 1 },
+ { SQL_API_SQLGETTYPEINFO , 1 },
+ { SQL_API_SQLMORERESULTS , 1 },
+ { SQL_API_SQLNATIVESQL , 0 },
+ { SQL_API_SQLNUMPARAMS , 1 },
+ { SQL_API_SQLNUMRESULTCOLS , 1 },
+ { SQL_API_SQLPARAMDATA , 1 },
+ { SQL_API_SQLPARAMOPTIONS , 0 },
+ { SQL_API_SQLPREPARE , 1 },
+ { SQL_API_SQLPRIMARYKEYS , 1 },
+ { SQL_API_SQLPROCEDURECOLUMNS , 0 },
+ { SQL_API_SQLPROCEDURES , 0 },
+ { SQL_API_SQLPUTDATA , 1 },
+ { SQL_API_SQLROWCOUNT , 1 },
+ { SQL_API_SQLSETCONNECTATTR , 1 },
+ { SQL_API_SQLSETCONNECTOPTION , 1 },
+ { SQL_API_SQLSETCURSORNAME , 1 },
+ { SQL_API_SQLSETDESCFIELD , 1 },
+ { SQL_API_SQLSETDESCREC , 1 },
+ { SQL_API_SQLSETENVATTR , 1 },
+ { SQL_API_SQLSETPARAM , 1 },
+ { SQL_API_SQLSETPOS , 0 },
+ { SQL_API_SQLSETSCROLLOPTIONS , 0 },
+ { SQL_API_SQLSETSTMTATTR , 1 },
+ { SQL_API_SQLSETSTMTOPTION , 1 },
+ { SQL_API_SQLSPECIALCOLUMNS , 0 },
+ { SQL_API_SQLSTATISTICS , 0 },
+ { SQL_API_SQLTABLEPRIVILEGES , 0 },
+ { SQL_API_SQLTABLES , 1 },
+ { SQL_API_SQLTRANSACT , 1 },
+ { 0 , -1 }
+};
diff --git a/ndb/src/client/odbc/handles/HandleBase.cpp b/ndb/src/client/odbc/handles/HandleBase.cpp
new file mode 100644
index 00000000000..27379cdc3f8
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleBase.cpp
@@ -0,0 +1,162 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HandleBase.hpp"
+
+HandleBase::~HandleBase()
+{
+ delete m_ctx;
+ m_ctx = 0;
+}
+
+void
+HandleBase::saveCtx(Ctx& ctx)
+{
+ delete m_ctx;
+ m_ctx = &ctx;
+}
+
+// get diagnostics
+
+void
+HandleBase::sqlGetDiagField(Ctx& ctx, SQLSMALLINT recNumber, SQLSMALLINT diagIdentifier, SQLPOINTER diagInfo, SQLSMALLINT bufferLength, SQLSMALLINT* stringLength)
+{
+ const DiagSpec& spec = DiagSpec::find(diagIdentifier);
+ if (spec.m_pos == Diag_pos_end) {
+ ctx.setCode(SQL_ERROR);
+ return;
+ }
+ const bool header = (spec.m_pos == Diag_pos_header);
+ const bool status = (spec.m_pos == Diag_pos_status);
+ ctx_assert(header || status);
+ if (! (spec.m_handles & odbcHandle())) {
+ ctx.setCode(SQL_ERROR);
+ return;
+ }
+ if (header) {
+ recNumber = 0; // ignored for header fields, so fix it
+ if (m_ctx == 0) {
+ if (diagIdentifier == SQL_DIAG_NUMBER) {
+ SQLINTEGER n = 0;
+ OdbcData data(n);
+ data.copyout(ctx, diagInfo, bufferLength, 0, stringLength);
+ return;
+ }
+ if (diagIdentifier == SQL_DIAG_RETURNCODE) {
+ SQLSMALLINT n = 0;
+ OdbcData data(n);
+ data.copyout(ctx, diagInfo, bufferLength, 0, stringLength);
+ return;
+ }
+ return;
+ }
+ }
+ if (status) {
+ if (recNumber <= 0) {
+ ctx.setCode(SQL_ERROR);
+ return;
+ }
+ if (m_ctx == 0) {
+ if ((unsigned)recNumber > 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ return;
+ }
+ if ((unsigned)recNumber > m_ctx->diagArea().numStatus()) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ }
+ OdbcData data;
+ ctx_assert(m_ctx != 0);
+ m_ctx->diagArea().getRecord(ctx, recNumber, diagIdentifier, data);
+ if (data.type() != OdbcData::Undef)
+ data.copyout(ctx, diagInfo, bufferLength, 0, stringLength);
+}
+
+void
+HandleBase::sqlGetDiagRec(Ctx& ctx, SQLSMALLINT recNumber, SQLCHAR* sqlstate, SQLINTEGER* nativeError, SQLCHAR* messageText, SQLSMALLINT bufferLength, SQLSMALLINT* textLength)
+{
+ sqlGetDiagField(ctx, recNumber, SQL_DIAG_SQLSTATE, static_cast<SQLPOINTER>(sqlstate), 5 + 1, 0);
+ sqlGetDiagField(ctx, recNumber, SQL_DIAG_NATIVE, static_cast<SQLPOINTER>(nativeError), -1, 0);
+ sqlGetDiagField(ctx, recNumber, SQL_DIAG_MESSAGE_TEXT, static_cast<SQLPOINTER>(messageText), bufferLength, textLength);
+}
+
+void
+HandleBase::sqlError(Ctx& ctx, SQLCHAR* sqlstate, SQLINTEGER* nativeError, SQLCHAR* messageText, SQLSMALLINT bufferLength, SQLSMALLINT* textLength)
+{
+ if (m_ctx == 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ const SQLSMALLINT recNumber = m_ctx->diagArea().nextRecNumber();
+ if (recNumber == 0) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ sqlGetDiagField(ctx, recNumber, SQL_DIAG_SQLSTATE, static_cast<SQLPOINTER>(sqlstate), 5 + 1, 0);
+ sqlGetDiagField(ctx, recNumber, SQL_DIAG_NATIVE, static_cast<SQLPOINTER>(nativeError), -1, 0);
+ sqlGetDiagField(ctx, recNumber, SQL_DIAG_MESSAGE_TEXT, static_cast<SQLPOINTER>(messageText), bufferLength, textLength);
+}
+
+// common code for attributes
+
+void
+HandleBase::baseSetHandleAttr(Ctx& ctx, AttrArea& attrArea, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength)
+{
+ const AttrSpec& spec = attrArea.findSpec(attribute);
+ if (spec.m_mode == Attr_mode_undef) { // not found
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "undefined attribute id %d", (int)attribute);
+ return;
+ }
+ if (spec.m_mode == Attr_mode_readonly) { // read only
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "read-only attribute id %d", (int)attribute);
+ return;
+ }
+ OdbcData data;
+ data.copyin(ctx, spec.m_type, value, stringLength);
+ if (! ctx.ok())
+ return;
+ attrArea.setAttr(ctx, attribute, data);
+}
+
+void
+HandleBase::baseGetHandleAttr(Ctx& ctx, AttrArea& attrArea, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength)
+{
+ const AttrSpec& spec = attrArea.findSpec(attribute);
+ if (spec.m_mode == Attr_mode_undef) { // not found
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "undefined attribute id %d", (int)attribute);
+ return;
+ }
+ OdbcData data;
+ attrArea.getAttr(ctx, attribute, data);
+ if (! ctx.ok())
+ return;
+ data.copyout(ctx, value, bufferLength, stringLength);
+}
+
+void
+HandleBase::baseSetHandleOption(Ctx& ctx, AttrArea& attrArea, SQLUSMALLINT option, SQLUINTEGER value)
+{
+ baseSetHandleAttr(ctx, attrArea, static_cast<SQLINTEGER>(option), reinterpret_cast<SQLPOINTER>(value), 0);
+}
+
+void
+HandleBase::baseGetHandleOption(Ctx& ctx, AttrArea& attrArea, SQLUSMALLINT option, SQLPOINTER value)
+{
+ baseGetHandleAttr(ctx, attrArea, static_cast<SQLINTEGER>(option), value, SQL_NTS, 0);
+}
diff --git a/ndb/src/client/odbc/handles/HandleBase.hpp b/ndb/src/client/odbc/handles/HandleBase.hpp
new file mode 100644
index 00000000000..fc35c2b559b
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleBase.hpp
@@ -0,0 +1,67 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_HANDLES_HandleBase_hpp
+#define ODBC_HANDLES_HandleBase_hpp
+
+#include <common/common.hpp>
+#include <common/OdbcData.hpp>
+#include <common/DiagArea.hpp>
+#include <common/AttrArea.hpp>
+
+/**
+ * @class HandleBase
+ * @brief Base class for handles
+ *
+ * Following types of handles exist:
+ * - HandleRoot : root node
+ * - HandleEnv : environment handle (SQLHENV)
+ * - HandleDbc : connection handle (SQLHDBC)
+ * - HandleStmt : statement handle (SQLHSTMT)
+ * - HandleDesc : descriptor handle (SQLHDESC)
+ */
+class HandleRoot;
+class HandleBase {
+public:
+ HandleBase();
+ virtual ~HandleBase() = 0;
+ virtual HandleBase* getParent() = 0;
+ virtual HandleRoot* getRoot() = 0;
+ virtual OdbcHandle odbcHandle() = 0;
+ void saveCtx(Ctx& ctx);
+ // allocate and free handles
+ virtual void sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild) = 0;
+ virtual void sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* pChild) = 0;
+ // get diagnostics
+ void sqlGetDiagField(Ctx& ctx, SQLSMALLINT recNumber, SQLSMALLINT diagIdentifier, SQLPOINTER diagInfo, SQLSMALLINT bufferLength, SQLSMALLINT* stringLength);
+ void sqlGetDiagRec(Ctx& ctx, SQLSMALLINT recNumber, SQLCHAR* sqlstate, SQLINTEGER* nativeError, SQLCHAR* messageText, SQLSMALLINT bufferLength, SQLSMALLINT* textLength);
+ void sqlError(Ctx& ctx, SQLCHAR* sqlstate, SQLINTEGER* nativeError, SQLCHAR* messageText, SQLSMALLINT bufferLength, SQLSMALLINT* textLength); // odbc2.0
+ // common code for attributes
+ void baseSetHandleAttr(Ctx& ctx, AttrArea& attrArea, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength);
+ void baseGetHandleAttr(Ctx& ctx, AttrArea& attrArea, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength);
+ void baseSetHandleOption(Ctx& ctx, AttrArea& attrArea, SQLUSMALLINT option, SQLUINTEGER value); // odbc2.0
+ void baseGetHandleOption(Ctx& ctx, AttrArea& attrArea, SQLUSMALLINT option, SQLPOINTER value); // odbc2.0
+protected:
+ Ctx* m_ctx; // saved from last ODBC function
+};
+
+inline
+HandleBase::HandleBase() :
+ m_ctx(0)
+{
+}
+
+#endif
diff --git a/ndb/src/client/odbc/handles/HandleDbc.cpp b/ndb/src/client/odbc/handles/HandleDbc.cpp
new file mode 100644
index 00000000000..2d5ded2cc21
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleDbc.cpp
@@ -0,0 +1,419 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <limits.h>
+#include <NdbApi.hpp>
+#include <common/common.hpp>
+#include <common/DiagArea.hpp>
+#include <common/StmtArea.hpp>
+#include "HandleRoot.hpp"
+#include "HandleEnv.hpp"
+#include "HandleDbc.hpp"
+#include "HandleStmt.hpp"
+#include "HandleDesc.hpp"
+#include "PoolNdb.hpp"
+
+#ifndef INT_MAX
+#define INT_MAX 2147483647
+#endif
+
+HandleDbc::HandleDbc(HandleEnv* pEnv) :
+ m_env(pEnv),
+ m_attrArea(m_attrSpec)
+{
+ m_attrArea.setHandle(this);
+}
+
+HandleDbc::~HandleDbc()
+{
+}
+
+void
+HandleDbc::ctor(Ctx& ctx)
+{
+}
+
+void
+HandleDbc::dtor(Ctx& ctx)
+{
+ if (m_state == Connected) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "cannot delete connection handle - connection is open");
+ return;
+ }
+ if (m_state == Transacting) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "cannot delete connection handle - transaction is active");
+ return;
+ }
+}
+
+// allocate and free handles
+
+void
+HandleDbc::sqlAllocStmt(Ctx& ctx, HandleStmt** ppStmt)
+{
+ if (ppStmt == 0) {
+ ctx.pushStatus(Sqlstate::_HY009, Error::Gen, "cannot allocate statement handle - null return address");
+ return;
+ }
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_08003, Error::Gen, "cannot allocate statement handle - not connected to database");
+ return;
+ }
+ HandleStmt* pStmt = new HandleStmt(this);
+ pStmt->ctor(ctx);
+ if (! ctx.ok()) {
+ pStmt->dtor(ctx);
+ delete pStmt;
+ return;
+ }
+ m_listStmt.push_back(pStmt);
+ getRoot()->record(SQL_HANDLE_STMT, pStmt, true);
+ *ppStmt = pStmt;
+}
+
+void
+HandleDbc::sqlAllocDesc(Ctx& ctx, HandleDesc** ppDesc)
+{
+ if (ppDesc == 0) {
+ ctx.pushStatus(Sqlstate::_HY009, Error::Gen, "cannot allocate descriptor handle - null return address");
+ return;
+ }
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_08003, Error::Gen, "cannot allocate descriptor handle - not connected to database");
+ return;
+ }
+ HandleDesc* pDesc = new HandleDesc(this);
+ pDesc->ctor(ctx);
+ if (! ctx.ok()) {
+ pDesc->dtor(ctx);
+ delete pDesc;
+ return;
+ }
+ m_listDesc.push_back(pDesc);
+ getRoot()->record(SQL_HANDLE_DESC, pDesc, true);
+ *ppDesc = pDesc;
+}
+
+void
+HandleDbc::sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild)
+{
+ switch (childType) {
+ case SQL_HANDLE_STMT:
+ sqlAllocStmt(ctx, (HandleStmt**)ppChild);
+ return;
+ case SQL_HANDLE_DESC:
+ sqlAllocDesc(ctx, (HandleDesc**)ppChild);
+ return;
+ }
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "invalid child handle type %d", (int)childType);
+}
+
+void
+HandleDbc::sqlFreeStmt(Ctx& ctx, HandleStmt* pStmt, SQLUSMALLINT iOption)
+{
+ switch (iOption) {
+ case SQL_CLOSE:
+ // no error if not open
+ if (pStmt->getState() == HandleStmt::Open)
+ pStmt->sqlCloseCursor(ctx);
+ return;
+ case SQL_DROP:
+ pStmt->dtor(ctx);
+ if (! ctx.ok())
+ return;
+ m_listStmt.remove(pStmt);
+ getRoot()->record(SQL_HANDLE_STMT, pStmt, false);
+ delete pStmt;
+ return;
+ case SQL_UNBIND: {
+ DescArea& ard = pStmt->getHandleDesc(ctx, Desc_usage_ARD)->descArea();
+ ard.setCount(ctx, 0);
+ return;
+ }
+ case SQL_RESET_PARAMS: {
+ DescArea& apd = pStmt->getHandleDesc(ctx, Desc_usage_APD)->descArea();
+ apd.setCount(ctx, 0);
+ // SQLFreeStmt doc misses this part
+ DescArea& ipd = pStmt->getHandleDesc(ctx, Desc_usage_IPD)->descArea();
+ ipd.setCount(ctx, 0);
+ return;
+ }
+ }
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "invalid free statement option %u", (unsigned)iOption);
+}
+
+void
+HandleDbc::sqlFreeDesc(Ctx& ctx, HandleDesc* pDesc)
+{
+ pDesc->dtor(ctx);
+ if (! ctx.ok())
+ return;
+ m_listDesc.remove(pDesc);
+ getRoot()->record(SQL_HANDLE_DESC, pDesc, false);
+ delete pDesc;
+}
+
+void
+HandleDbc::sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* pChild)
+{
+ switch (childType) {
+ case SQL_HANDLE_STMT:
+ sqlFreeStmt(ctx, (HandleStmt*)pChild, SQL_DROP);
+ return;
+ case SQL_HANDLE_DESC:
+ sqlFreeDesc(ctx, (HandleDesc*)pChild);
+ return;
+ }
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "invalid child handle type %d", (int)childType);
+}
+
+// attributes and info functions
+
+static bool
+ignore_attr(Ctx& ctx, SQLINTEGER attribute)
+{
+ switch (attribute) {
+ case 1246:
+ ctx_log2(("ignore unknown ADO.NET connect attribute %d", (int)attribute));
+ return true;
+ }
+ return false;
+}
+
+void
+HandleDbc::sqlSetConnectAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength)
+{
+ if (ignore_attr(ctx, attribute))
+ return;
+ baseSetHandleAttr(ctx, m_attrArea, attribute, value, stringLength);
+}
+
+void
+HandleDbc::sqlGetConnectAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength)
+{
+ if (ignore_attr(ctx, attribute))
+ return;
+ baseGetHandleAttr(ctx, m_attrArea, attribute, value, bufferLength, stringLength);
+}
+
+void
+HandleDbc::sqlSetConnectOption(Ctx& ctx, SQLUSMALLINT option, SQLUINTEGER value)
+{
+ if (ignore_attr(ctx, option))
+ return;
+ baseSetHandleOption(ctx, m_attrArea, option, value);
+}
+
+void
+HandleDbc::sqlGetConnectOption(Ctx& ctx, SQLUSMALLINT option, SQLPOINTER value)
+{
+ if (ignore_attr(ctx, option))
+ return;
+ baseGetHandleOption(ctx, m_attrArea, option, value);
+}
+
+void
+HandleDbc::sqlGetFunctions(Ctx& ctx, SQLUSMALLINT functionId, SQLUSMALLINT* supported)
+{
+ if (functionId == SQL_API_ALL_FUNCTIONS) {
+ for (int i = 0; i < 100; i++)
+ supported[i] = SQL_FALSE;
+ FuncTab* f;
+ for (f = m_funcTab; f->m_supported != -1; f++) {
+ SQLUSMALLINT id = f->m_functionId;
+ if (id < 100 && f->m_supported)
+ supported[id] = SQL_TRUE;
+ }
+ } else if (functionId == SQL_API_ODBC3_ALL_FUNCTIONS) {
+ for (int i = 0; i < SQL_API_ODBC3_ALL_FUNCTIONS_SIZE; i++)
+ supported[i] = 0;
+ FuncTab* f;
+ for (f = m_funcTab; f->m_supported != -1; f++) {
+ SQLUSMALLINT id = f->m_functionId;
+ ctx_assert((id >> 4) < SQL_API_ODBC3_ALL_FUNCTIONS_SIZE);
+ if (f->m_supported)
+ supported[id >> 4] |= (1 << (id & 0xf));
+ }
+ } else {
+ FuncTab* f;
+ for (f = m_funcTab; f->m_supported != -1; f++) {
+ if (f->m_functionId == functionId)
+ break;
+ }
+ if (f->m_supported != -1)
+ supported[0] = f->m_supported ? SQL_TRUE : SQL_FALSE;
+ else
+ ctx.pushStatus(Sqlstate::_HY095, Error::Gen, "invalid function id %u", (unsigned)functionId);
+ }
+}
+
+void
+HandleDbc::sqlGetInfo(Ctx& ctx, SQLUSMALLINT infoType, SQLPOINTER infoValue, SQLSMALLINT bufferLength, SQLSMALLINT* stringLength)
+{
+ InfoTab* f;
+ for (f = m_infoTab; f->m_format != InfoTab::End; f++) {
+ if (f->m_id == infoType)
+ break;
+ }
+ if (f->m_format == InfoTab::End) {
+ ctx.pushStatus(Sqlstate::_HY096, Error::Gen, "invalid info type %u", (unsigned)infoType);
+ return;
+ }
+ if (f->m_format == InfoTab::Char || f->m_format == InfoTab::YesNo) {
+ ctx_log3(("SQLGetInfo: type=%u value='%s'", (unsigned)infoType, f->m_str));
+ OdbcData data(f->m_str);
+ data.copyout(ctx, infoValue, bufferLength, 0, stringLength);
+ return;
+ }
+ if (f->m_format == InfoTab::Short) {
+ ctx_log3(("SQLGetInfo: type=%u value=%d", (unsigned)infoType, (int)f->m_int));
+ OdbcData data((SQLUSMALLINT)f->m_int);
+ data.copyout(ctx, infoValue, 0, 0);
+ return;
+ }
+ if (f->m_format == InfoTab::Long || f->m_format == InfoTab::Bitmask) {
+ ctx_log3(("SQLGetInfo: type=%u value=0x%x", (unsigned)infoType, (int)f->m_int));
+ OdbcData data((SQLUINTEGER)f->m_int);
+ data.copyout(ctx, infoValue, 0, 0);
+ return;
+ }
+ ctx_assert(false);
+}
+
+int
+HandleDbc::getOdbcVersion(Ctx& ctx)
+{
+ return m_env->getOdbcVersion(ctx);
+}
+
+// connect and transactions
+
+void
+HandleDbc::sqlConnect(Ctx& ctx, SQLCHAR* serverName, SQLSMALLINT nameLength1, SQLCHAR* userName, SQLSMALLINT nameLength2, SQLCHAR* authentication, SQLSMALLINT nameLength3)
+{
+ if (m_state != Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "already connected");
+ return;
+ }
+ OdbcData data;
+ m_attrArea.getAttr(ctx, SQL_ATTR_CONNECTION_TIMEOUT, data);
+ int timeout = data.uinteger();
+ if (timeout <= 0)
+ timeout = INT_MAX;
+ PoolNdb* poolNdb = getRoot()->getPoolNdb();
+ Ndb* pNdb = poolNdb->allocate(ctx, timeout);
+ if (pNdb == 0) {
+ return;
+ }
+ m_ndbObject = pNdb;
+ m_state = Connected;
+ m_autocommit = true;
+}
+
+void
+HandleDbc::sqlDriverConnect(Ctx& ctx, SQLHWND hwnd, SQLCHAR* szConnStrIn, SQLSMALLINT cbConnStrIn, SQLCHAR* szConnStrOut, SQLSMALLINT cbConnStrOutMax, SQLSMALLINT* pcbConnStrOut, SQLUSMALLINT fDriverCompletion)
+{
+ ctx_log2(("driver connect %.*s", cbConnStrIn, szConnStrIn == 0 ? "" : (char*)szConnStrIn));
+ sqlConnect(ctx, (SQLCHAR*)"", 0, (SQLCHAR*)"", 0, (SQLCHAR*)"", 0);
+ if (! ctx.ok())
+ return;
+ OdbcData data("DNS=DEFAULT");
+ if (szConnStrOut != 0) // ADO NET
+ data.copyout(ctx, static_cast<SQLPOINTER>(szConnStrOut), cbConnStrOutMax, 0, pcbConnStrOut);
+}
+
+void
+HandleDbc::sqlDisconnect(Ctx& ctx)
+{
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "already disconnected");
+ return;
+ }
+ // XXX missing check for uncommited changes
+ ListStmt::iterator i = m_listStmt.begin();
+ while (i != m_listStmt.end()) {
+ HandleStmt* pStmt = *i;
+ ListStmt::iterator j = i++;
+ pStmt->dtor(ctx);
+ getRoot()->record(SQL_HANDLE_STMT, pStmt, false);
+ delete pStmt;
+ m_listStmt.erase(j);
+ }
+ PoolNdb* poolNdb = getRoot()->getPoolNdb();
+ poolNdb->release(ctx, m_ndbObject);
+ m_ndbObject = 0;
+ m_state = Free;
+ m_autocommit = true;
+}
+
+void
+HandleDbc::sqlEndTran(Ctx& ctx, SQLSMALLINT completionType)
+{
+ if (completionType != SQL_COMMIT && completionType != SQL_ROLLBACK) {
+ ctx.pushStatus(Sqlstate::_HY012, Error::Gen, "invalid completion type %d", (int)completionType);
+ return;
+ }
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_08003, Error::Gen, "not connected");
+ return;
+ }
+ Ndb* ndb = m_ndbObject;
+ ctx_assert(ndb != 0);
+ if (m_state == Connected) {
+ ctx_log2(("sqlEndTran: no transaction active"));
+ return;
+ }
+ NdbConnection* tcon = m_ndbConnection;
+ ctx_assert(tcon != 0);
+ if (completionType == SQL_COMMIT) {
+ if (tcon->execute(Commit) == -1) {
+ if (tcon->getNdbError().code != 626)
+ ctx.pushStatus(ndb, tcon, 0, "execute commit");
+ else
+ ctx_log1(("ignore no data (626) at execute commit"));
+ } else {
+ ctx_log2(("sqlEndTran: transaction commit done"));
+ m_uncommitted = false;
+ }
+ } else {
+ if (tcon->execute(Rollback) == -1) {
+ if (tcon->getNdbError().code != 626)
+ ctx.pushStatus(ndb, tcon, 0, "execute rollback");
+ else
+ ctx_log1(("ignore no data (626) at execute rollback"));
+ } else {
+ ctx_log2(("sqlEndTran: transaction rollback done"));
+ m_uncommitted = false;
+ }
+ }
+ for (ListStmt::iterator i = m_listStmt.begin(); i != m_listStmt.end(); i++) {
+ HandleStmt* pStmt = *i;
+ if (pStmt->getState() == HandleStmt::Open) {
+ pStmt->sqlCloseCursor(ctx); // SQL_CB_CLOSE behaviour
+ }
+ pStmt->useConnection(ctx, false);
+ }
+ if (! m_autocommit) {
+ useConnection(ctx, false);
+ useConnection(ctx, true);
+ }
+}
+
+void
+HandleDbc::sqlTransact(Ctx& ctx, SQLUSMALLINT completionType)
+{
+ sqlEndTran(ctx, completionType);
+}
diff --git a/ndb/src/client/odbc/handles/HandleDbc.hpp b/ndb/src/client/odbc/handles/HandleDbc.hpp
new file mode 100644
index 00000000000..130df08d02c
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleDbc.hpp
@@ -0,0 +1,111 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_HANDLES_HandleDbc_hpp
+#define ODBC_HANDLES_HandleDbc_hpp
+
+#include <list>
+#include <common/common.hpp>
+#include <common/ConnArea.hpp>
+#include "HandleBase.hpp"
+
+class HandleRoot;
+class HandleEnv;
+class HandleStmt;
+class HandleDesc;
+
+/**
+ * @class HandleDbc
+ * @brief Connection handle (SQLHDBC).
+ */
+class HandleDbc : public HandleBase, public ConnArea {
+public:
+ HandleDbc(HandleEnv* pEnv);
+ ~HandleDbc();
+ void ctor(Ctx& ctx);
+ void dtor(Ctx& ctx);
+ HandleEnv* getEnv();
+ HandleBase* getParent();
+ HandleRoot* getRoot();
+ OdbcHandle odbcHandle();
+ // allocate and free handles
+ void sqlAllocStmt(Ctx& ctx, HandleStmt** ppStmt);
+ void sqlAllocDesc(Ctx& ctx, HandleDesc** ppDesc);
+ void sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild);
+ void sqlFreeStmt(Ctx& ctx, HandleStmt* pStmt, SQLUSMALLINT iOption);
+ void sqlFreeDesc(Ctx& ctx, HandleDesc* pDesc);
+ void sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* pChild);
+ // attributes and info functions
+ void sqlSetConnectAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength);
+ void sqlGetConnectAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength);
+ void sqlSetConnectOption(Ctx& ctx, SQLUSMALLINT option, SQLUINTEGER value); // odbc2.0
+ void sqlGetConnectOption(Ctx& ctx, SQLUSMALLINT option, SQLPOINTER value); // odbc2.0
+ void sqlGetFunctions(Ctx& ctx, SQLUSMALLINT functionId, SQLUSMALLINT* supported);
+ void sqlGetInfo(Ctx& ctx, SQLUSMALLINT infoType, SQLPOINTER infoValue, SQLSMALLINT bufferLength, SQLSMALLINT* stringLength);
+ int getOdbcVersion(Ctx& ctx);
+ // connect and transactions
+ void sqlConnect(Ctx& ctx, SQLCHAR* serverName, SQLSMALLINT nameLength1, SQLCHAR* userName, SQLSMALLINT nameLength2, SQLCHAR* authentication, SQLSMALLINT nameLength3);
+ void sqlDriverConnect(Ctx& ctx, SQLHWND hwnd, SQLCHAR* szConnStrIn, SQLSMALLINT cbConnStrIn, SQLCHAR* szConnStrOut, SQLSMALLINT cbConnStrOutMax, SQLSMALLINT* pcbConnStrOut, SQLUSMALLINT fDriverCompletion);
+ void sqlDisconnect(Ctx& ctx);
+ void sqlEndTran(Ctx& ctx, SQLSMALLINT completionType);
+ void sqlTransact(Ctx& ctx, SQLUSMALLINT completionType); // odbc2.0
+private:
+ HandleEnv* const m_env;
+ typedef std::list<HandleStmt*> ListStmt;
+ ListStmt m_listStmt;
+ typedef std::list<HandleDesc*> ListDesc;
+ ListDesc m_listDesc;
+ static AttrSpec m_attrSpec[];
+ AttrArea m_attrArea;
+ struct FuncTab {
+ SQLUSMALLINT m_functionId;
+ int m_supported;
+ };
+ static FuncTab m_funcTab[];
+ struct InfoTab {
+ SQLUSMALLINT m_id;
+ enum { Char, YesNo, Short, Long, Bitmask, End } m_format;
+ SQLUINTEGER m_int;
+ const char* m_str;
+ };
+ static InfoTab m_infoTab[];
+};
+
+inline HandleEnv*
+HandleDbc::getEnv()
+{
+ return m_env;
+}
+
+inline HandleBase*
+HandleDbc::getParent()
+{
+ return (HandleBase*)getEnv();
+}
+
+inline HandleRoot*
+HandleDbc::getRoot()
+{
+ return getParent()->getRoot();
+}
+
+inline OdbcHandle
+HandleDbc::odbcHandle()
+{
+ return Odbc_handle_dbc;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/handles/HandleDesc.cpp b/ndb/src/client/odbc/handles/HandleDesc.cpp
new file mode 100644
index 00000000000..4cff1bb8892
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleDesc.cpp
@@ -0,0 +1,254 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/OdbcData.hpp>
+#include <common/DiagArea.hpp>
+#include <common/DataType.hpp>
+#include "HandleRoot.hpp"
+#include "HandleDbc.hpp"
+#include "HandleDesc.hpp"
+
+HandleDesc::HandleDesc(HandleDbc* pDbc) :
+ m_dbc(pDbc),
+ m_descArea(this, m_descSpec)
+{
+}
+
+HandleDesc::~HandleDesc()
+{
+}
+
+void
+HandleDesc::ctor(Ctx& ctx)
+{
+}
+
+void
+HandleDesc::dtor(Ctx& ctx)
+{
+}
+
+// allocate and free handles (no valid case)
+
+void
+HandleDesc::sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild)
+{
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "inappropriate handle type");
+}
+
+void
+HandleDesc::sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* ppChild)
+{
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "inappropriate handle type");
+}
+
+// set and get descriptor values
+
+void
+HandleDesc::sqlSetDescField(Ctx& ctx, SQLSMALLINT recNumber, SQLSMALLINT fieldIdentifier, SQLPOINTER value, SQLINTEGER bufferLength)
+{
+ const DescSpec& spec = m_descArea.findSpec(fieldIdentifier);
+ if (spec.m_pos == Desc_pos_end) {
+ ctx.pushStatus(Sqlstate::_HY091, Error::Gen, "invalid descriptor id %d", (int)fieldIdentifier);
+ return;
+ }
+ OdbcData data;
+ data.copyin(ctx, spec.m_type, value, bufferLength);
+ if (! ctx.ok())
+ return;
+ const bool header = (spec.m_pos == Desc_pos_header);
+ const bool record = (spec.m_pos == Desc_pos_record);
+ ctx_assert(header || record);
+ DescArea& area = m_descArea;
+ if (header) {
+ area.getHeader().setField(ctx, fieldIdentifier, data);
+ }
+ if (record) {
+ if (recNumber < 0) {
+ ctx.pushStatus(Sqlstate::_07009, Error::Gen, "invalid record number %d", (int)recNumber);
+ return;
+ }
+ if (recNumber == 0) { // bookmark record
+ if (area.getUsage() == Desc_usage_IPD) {
+ ctx.pushStatus(Sqlstate::_07009, Error::Gen, "cannot set bookmark IPD");
+ return;
+ }
+ if (area.getUsage() == Desc_usage_APD) {
+ ctx.pushStatus(Sqlstate::_07009, Error::Gen, "cannot set bookmark APD");
+ return;
+ }
+ }
+ area.getRecord(recNumber).setField(ctx, fieldIdentifier, data);
+ }
+}
+
+void
+HandleDesc::sqlGetDescField(Ctx& ctx, SQLSMALLINT recNumber, SQLSMALLINT fieldIdentifier, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength, SQLSMALLINT* stringLength2)
+{
+ const DescSpec& spec = m_descArea.findSpec(fieldIdentifier);
+ if (spec.m_pos == Desc_pos_end) {
+ ctx.pushStatus(Sqlstate::_HY091, Error::Gen, "invalid descriptor id %d", (int)fieldIdentifier);
+ return;
+ }
+ const bool header = (spec.m_pos == Desc_pos_header);
+ const bool record = (spec.m_pos == Desc_pos_record);
+ ctx_assert(header || record);
+ DescArea& area = m_descArea;
+ OdbcData data;
+ if (header) {
+ area.getHeader().getField(ctx, fieldIdentifier, data);
+ if (! ctx.ok())
+ return;
+ }
+ if (record) {
+ if (recNumber < 0) {
+ ctx.pushStatus(Sqlstate::_07009, Error::Gen, "invalid record number %d", (int)recNumber);
+ return;
+ }
+ if (recNumber == 0) { // bookmark record
+ if (area.getUsage() == Desc_usage_IPD) {
+ ctx.pushStatus(Sqlstate::_07009, Error::Gen, "cannot get bookmark IPD");
+ return;
+ }
+ if (area.getUsage() == Desc_usage_IRD) {
+ // XXX check SQL_ATTR_USE_BOOKMARK != SQL_UB_OFF
+ }
+ }
+ if ((unsigned)recNumber > area.getCount()) {
+ ctx.setCode(SQL_NO_DATA);
+ return;
+ }
+ area.getRecord(recNumber).getField(ctx, fieldIdentifier, data);
+ if (! ctx.ok())
+ return;
+ }
+ // if no data, return success and undefined value
+ if (data.type() == OdbcData::Undef)
+ return;
+ data.copyout(ctx, value, bufferLength, stringLength, stringLength2);
+}
+
+void
+HandleDesc::sqlColAttribute(Ctx& ctx, SQLUSMALLINT columnNumber, SQLUSMALLINT fieldIdentifier, SQLPOINTER characterAttribute, SQLSMALLINT bufferLength, SQLSMALLINT* stringLength, SQLPOINTER numericAttribute)
+{
+ ctx_log3(("sqlColAttribute col=%d id=%d", columnNumber, fieldIdentifier));
+ if (fieldIdentifier == SQL_COLUMN_LENGTH) { // XXX iODBC workaround
+ fieldIdentifier = SQL_DESC_LENGTH;
+ }
+ if (fieldIdentifier == 1205 || fieldIdentifier == 1206) {
+ ctx_log2(("ignore unknown OSQL fieldIdentifier %d", (int)fieldIdentifier));
+ if (characterAttribute != 0)
+ *(char*)characterAttribute = 0;
+ if (stringLength != 0)
+ *stringLength = 0;
+ return;
+ }
+ const DescSpec& spec = m_descArea.findSpec(fieldIdentifier);
+ if (spec.m_pos == Desc_pos_end) {
+ ctx.pushStatus(Sqlstate::_HY091, Error::Gen, "invalid descriptor id %d", (int)fieldIdentifier);
+ return;
+ }
+ if (spec.m_type == OdbcData::Sqlchar || spec.m_type == OdbcData::Sqlstate)
+ sqlGetDescField(ctx, columnNumber, fieldIdentifier, characterAttribute, bufferLength, 0, stringLength);
+ else {
+ sqlGetDescField(ctx, columnNumber, fieldIdentifier, numericAttribute, -1, 0);
+ }
+ if (ctx.getCode() == SQL_NO_DATA) {
+ ctx.setCode(SQL_SUCCESS);
+ ctx.pushStatus(Sqlstate::_07009, Error::Gen, "invalid column number %d", (int)columnNumber);
+ }
+}
+
+void
+HandleDesc::sqlColAttributes(Ctx& ctx, SQLUSMALLINT icol, SQLUSMALLINT fdescType, SQLPOINTER rgbDesc, SQLSMALLINT cbDescMax, SQLSMALLINT* pcbDesc, SQLINTEGER* pfDesc)
+{
+ ctx_log3(("sqlColAttributes col=%hu id=%hu", icol, fdescType));
+ SQLUSMALLINT columnNumber = icol;
+ SQLUSMALLINT fieldIdentifier;
+ // XXX incomplete
+ if (fdescType == SQL_COLUMN_TYPE) {
+ fieldIdentifier = SQL_DESC_TYPE;
+ } else if (fdescType == SQL_COLUMN_PRECISION) {
+ SQLSMALLINT type;
+ sqlGetDescField(ctx, columnNumber, SQL_DESC_TYPE, static_cast<SQLPOINTER>(&type), -1, 0);
+ if (! ctx.ok())
+ return;
+ switch (type) {
+ case SQL_CHAR:
+ case SQL_VARCHAR:
+ case SQL_BINARY:
+ case SQL_VARBINARY:
+ case SQL_LONGVARCHAR:
+ case SQL_LONGVARBINARY:
+ case SQL_DATE:
+ fieldIdentifier = SQL_DESC_LENGTH;
+ break;
+ default:
+ fieldIdentifier = SQL_DESC_PRECISION;
+ break;
+ }
+ } else if (fdescType == SQL_COLUMN_SCALE) {
+ SQLSMALLINT type;
+ sqlGetDescField(ctx, columnNumber, SQL_DESC_TYPE, static_cast<SQLPOINTER>(&type), -1, 0);
+ if (! ctx.ok())
+ return;
+ switch (type) {
+ default:
+ fieldIdentifier = SQL_DESC_SCALE;
+ break;
+ }
+ } else if (fdescType == SQL_COLUMN_LENGTH) {
+ SQLSMALLINT type;
+ sqlGetDescField(ctx, columnNumber, SQL_DESC_TYPE, static_cast<SQLPOINTER>(&type), -1, 0);
+ if (! ctx.ok())
+ return;
+ switch (type) {
+ default:
+ fieldIdentifier = SQL_DESC_LENGTH;
+ break;
+ }
+ } else {
+ fieldIdentifier = fdescType;
+ }
+ sqlColAttribute(ctx, columnNumber, fieldIdentifier, rgbDesc, cbDescMax, pcbDesc, pfDesc);
+}
+
+// set and get several common descriptor values
+
+void
+HandleDesc::sqlSetDescRec(Ctx& ctx, SQLSMALLINT recNumber, SQLSMALLINT type, SQLSMALLINT subType, SQLINTEGER length, SQLSMALLINT precision, SQLSMALLINT scale, SQLPOINTER data, SQLINTEGER* stringLength, SQLINTEGER* indicator)
+{
+ sqlSetDescField(ctx, recNumber, SQL_DESC_TYPE, reinterpret_cast<SQLPOINTER>(type), -1);
+ sqlSetDescField(ctx, recNumber, SQL_DESC_DATETIME_INTERVAL_CODE, reinterpret_cast<SQLPOINTER>(subType), -1);
+ sqlSetDescField(ctx, recNumber, SQL_DESC_OCTET_LENGTH, reinterpret_cast<SQLPOINTER>(length), -1);
+ sqlSetDescField(ctx, recNumber, SQL_DESC_PRECISION, reinterpret_cast<SQLPOINTER>(precision), -1);
+ sqlSetDescField(ctx, recNumber, SQL_DESC_SCALE, reinterpret_cast<SQLPOINTER>(scale), -1);
+ sqlSetDescField(ctx, recNumber, SQL_DESC_DATA_PTR, data, -1);
+ sqlSetDescField(ctx, recNumber, SQL_DESC_OCTET_LENGTH_PTR, reinterpret_cast<SQLPOINTER>(stringLength), -1);
+ sqlSetDescField(ctx, recNumber, SQL_DESC_INDICATOR_PTR, reinterpret_cast<SQLPOINTER>(indicator), -1);
+}
+
+void
+HandleDesc::sqlGetDescRec(Ctx& ctx, SQLSMALLINT recNumber, SQLCHAR* name, SQLSMALLINT bufferLength, SQLSMALLINT* stringLength, SQLSMALLINT* type, SQLSMALLINT* subType, SQLINTEGER* length, SQLSMALLINT* precision, SQLSMALLINT* scale, SQLSMALLINT* nullable)
+{
+ sqlGetDescField(ctx, recNumber, SQL_DESC_NAME, reinterpret_cast<SQLPOINTER>(name), bufferLength, 0, stringLength);
+ sqlGetDescField(ctx, recNumber, SQL_DESC_TYPE, reinterpret_cast<SQLPOINTER>(type), -1, 0);
+ sqlGetDescField(ctx, recNumber, SQL_DESC_DATETIME_INTERVAL_CODE, reinterpret_cast<SQLPOINTER>(subType), -1, 0);
+ sqlGetDescField(ctx, recNumber, SQL_DESC_OCTET_LENGTH, reinterpret_cast<SQLPOINTER>(length), -1, 0);
+ sqlGetDescField(ctx, recNumber, SQL_DESC_PRECISION, reinterpret_cast<SQLPOINTER>(precision), -1, 0);
+ sqlGetDescField(ctx, recNumber, SQL_DESC_SCALE, reinterpret_cast<SQLPOINTER>(scale), -1, 0);
+ sqlGetDescField(ctx, recNumber, SQL_DESC_NULLABLE, reinterpret_cast<SQLPOINTER>(nullable), -1, 0);
+}
diff --git a/ndb/src/client/odbc/handles/HandleDesc.hpp b/ndb/src/client/odbc/handles/HandleDesc.hpp
new file mode 100644
index 00000000000..9419697f134
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleDesc.hpp
@@ -0,0 +1,89 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_HANDLES_HandleDesc_hpp
+#define ODBC_HANDLES_HandleDesc_hpp
+
+#include <common/common.hpp>
+#include <common/DescArea.hpp>
+#include "HandleBase.hpp"
+
+class HandleRoot;
+class HandleDbc;
+
+/**
+ * @class HandleDesc
+ * @brief Descriptor handle (SQLHDESC).
+ */
+class HandleDesc : public HandleBase {
+public:
+ HandleDesc(HandleDbc* pDbc);
+ ~HandleDesc();
+ void ctor(Ctx& ctx);
+ void dtor(Ctx& ctx);
+ HandleDbc* getDbc();
+ HandleBase* getParent();
+ HandleRoot* getRoot();
+ OdbcHandle odbcHandle();
+ DescArea& descArea();
+ // allocate and free handles (no valid case)
+ void sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild);
+ void sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* pChild);
+ // set and get descriptor values (internal use)
+ void sqlSetDescField(Ctx& ctx, SQLSMALLINT recNumber, SQLSMALLINT fieldIdentifier, SQLPOINTER value, SQLINTEGER bufferLength);
+ void sqlGetDescField(Ctx& ctx, SQLSMALLINT recNumber, SQLSMALLINT fieldIdentifier, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength, SQLSMALLINT* stringLength2 = 0);
+ void sqlColAttribute(Ctx& ctx, SQLUSMALLINT columnNumber, SQLUSMALLINT fieldIdentifier, SQLPOINTER characterAttribute, SQLSMALLINT bufferLength, SQLSMALLINT* stringLength, SQLPOINTER numericAttribute);
+ void sqlColAttributes(Ctx& ctx, SQLUSMALLINT icol, SQLUSMALLINT fdescType, SQLPOINTER rgbDesc, SQLSMALLINT cbDescMax, SQLSMALLINT* pcbDesc, SQLINTEGER* pfDesc);
+ // set and get several common descriptor values
+ void sqlSetDescRec(Ctx& ctx, SQLSMALLINT recNumber, SQLSMALLINT Type, SQLSMALLINT SubType, SQLINTEGER Length, SQLSMALLINT Precision, SQLSMALLINT Scale, SQLPOINTER Data, SQLINTEGER* StringLength, SQLINTEGER* Indicator);
+ void sqlGetDescRec(Ctx& ctx, SQLSMALLINT recNumber, SQLCHAR* Name, SQLSMALLINT BufferLength, SQLSMALLINT* StringLength, SQLSMALLINT* Type, SQLSMALLINT* SubType, SQLINTEGER* Length, SQLSMALLINT* Precision, SQLSMALLINT* Scale, SQLSMALLINT* Nullable);
+private:
+ HandleDbc* const m_dbc;
+ static DescSpec m_descSpec[];
+ DescArea m_descArea;
+};
+
+inline HandleDbc*
+HandleDesc::getDbc()
+{
+ return m_dbc;
+}
+
+inline HandleBase*
+HandleDesc::getParent()
+{
+ return (HandleDbc*)getDbc();
+}
+
+inline HandleRoot*
+HandleDesc::getRoot()
+{
+ return getParent()->getRoot();
+}
+
+inline OdbcHandle
+HandleDesc::odbcHandle()
+{
+ return Odbc_handle_desc;
+}
+
+inline DescArea&
+HandleDesc::descArea()
+{
+ return m_descArea;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/handles/HandleEnv.cpp b/ndb/src/client/odbc/handles/HandleEnv.cpp
new file mode 100644
index 00000000000..bc9d8b420a6
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleEnv.cpp
@@ -0,0 +1,144 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/DiagArea.hpp>
+#include "HandleRoot.hpp"
+#include "HandleEnv.hpp"
+#include "HandleDbc.hpp"
+
+HandleEnv::HandleEnv(HandleRoot* pRoot) :
+ m_root(pRoot),
+ m_attrArea(m_attrSpec)
+{
+ m_attrArea.setHandle(this);
+}
+
+HandleEnv::~HandleEnv() {
+}
+
+void
+HandleEnv::ctor(Ctx& ctx)
+{
+}
+
+void
+HandleEnv::dtor(Ctx& ctx)
+{
+ if (! m_listDbc.empty()) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "cannot delete environment handle - has %u associated connection handles", (unsigned)m_listDbc.size());
+ return;
+ }
+}
+
+// allocate and free handles
+
+void
+HandleEnv::sqlAllocConnect(Ctx& ctx, HandleDbc** ppDbc)
+{
+ if (getOdbcVersion(ctx) == -1)
+ return;
+ if (ppDbc == 0) {
+ ctx.pushStatus(Sqlstate::_HY009, Error::Gen, "cannot allocate connection handle - null return address");
+ return;
+ }
+ HandleDbc* pDbc = new HandleDbc(this);
+ pDbc->ctor(ctx);
+ if (! ctx.ok()) {
+ pDbc->dtor(ctx);
+ delete pDbc;
+ return;
+ }
+ m_listDbc.push_back(pDbc);
+ getRoot()->record(SQL_HANDLE_DBC, pDbc, true);
+ *ppDbc = pDbc;
+}
+
+void
+HandleEnv::sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild)
+{
+ if (getOdbcVersion(ctx) == -1)
+ return;
+ switch (childType) {
+ case SQL_HANDLE_DBC:
+ sqlAllocConnect(ctx, (HandleDbc**)ppChild);
+ return;
+ }
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "invalid child handle type %d", (int)childType);
+}
+
+void
+HandleEnv::sqlFreeConnect(Ctx& ctx, HandleDbc* pDbc)
+{
+ if (getOdbcVersion(ctx) == -1)
+ return;
+ pDbc->dtor(ctx);
+ if (! ctx.ok())
+ return;
+ m_listDbc.remove(pDbc);
+ getRoot()->record(SQL_HANDLE_DBC, pDbc, false);
+ delete pDbc;
+}
+
+void
+HandleEnv::sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* pChild)
+{
+ if (getOdbcVersion(ctx) == -1)
+ return;
+ switch (childType) {
+ case SQL_HANDLE_DBC:
+ sqlFreeConnect(ctx, (HandleDbc*)pChild);
+ return;
+ }
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "invalid child handle type %d", (int)childType);
+}
+
+// attributes
+
+void
+HandleEnv::sqlSetEnvAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength)
+{
+ baseSetHandleAttr(ctx, m_attrArea, attribute, value, stringLength);
+}
+
+void
+HandleEnv::sqlGetEnvAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength)
+{
+ baseGetHandleAttr(ctx, m_attrArea, attribute, value, bufferLength, stringLength);
+}
+
+int
+HandleEnv::getOdbcVersion(Ctx& ctx)
+{
+ OdbcData data;
+ m_attrArea.getAttr(ctx, SQL_ATTR_ODBC_VERSION, data);
+ if (! ctx.ok())
+ return -1;
+ return data.integer();
+}
+
+// transactions
+
+void
+HandleEnv::sqlEndTran(Ctx& ctx, SQLSMALLINT completionType)
+{
+ ctx_assert(false); // XXX
+}
+
+void
+HandleEnv::sqlTransact(Ctx& ctx, SQLUSMALLINT completionType)
+{
+ ctx_assert(false); // XXX
+}
diff --git a/ndb/src/client/odbc/handles/HandleEnv.hpp b/ndb/src/client/odbc/handles/HandleEnv.hpp
new file mode 100644
index 00000000000..2b13b0256bc
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleEnv.hpp
@@ -0,0 +1,77 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_HANDLES_HandleEnv_hpp
+#define ODBC_HANDLES_HandleEnv_hpp
+
+#include <list>
+#include <common/common.hpp>
+#include "HandleBase.hpp"
+
+class HandleRoot;
+class HandleDbc;
+
+/**
+ * @class HandleEnv
+ * @brief Environment handle (SQLHENV).
+ */
+class HandleEnv : public HandleBase {
+public:
+ HandleEnv(HandleRoot* pRoot);
+ ~HandleEnv();
+ void ctor(Ctx& ctx);
+ void dtor(Ctx& ctx);
+ HandleRoot* getRoot();
+ HandleBase* getParent();
+ OdbcHandle odbcHandle();
+ // allocate and free handles
+ void sqlAllocConnect(Ctx& ctx, HandleDbc** ppDbc);
+ void sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild);
+ void sqlFreeConnect(Ctx& ctx, HandleDbc* pDbc);
+ void sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* pChild);
+ // attributes
+ void sqlSetEnvAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength);
+ void sqlGetEnvAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength);
+ int getOdbcVersion(Ctx& ctx); // returns -1 if not set
+ // transactions
+ void sqlEndTran(Ctx& ctx, SQLSMALLINT completionType);
+ void sqlTransact(Ctx& ctx, SQLUSMALLINT completionType); // odbc2.0
+private:
+ HandleRoot* const m_root;
+ std::list<HandleDbc*> m_listDbc;
+ static AttrSpec m_attrSpec[];
+ AttrArea m_attrArea;
+};
+
+inline HandleRoot*
+HandleEnv::getRoot()
+{
+ return m_root;
+}
+
+inline HandleBase*
+HandleEnv::getParent()
+{
+ return (HandleBase*)getRoot();
+}
+
+inline OdbcHandle
+HandleEnv::odbcHandle()
+{
+ return Odbc_handle_env;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/handles/HandleRoot.cpp b/ndb/src/client/odbc/handles/HandleRoot.cpp
new file mode 100644
index 00000000000..d901cb5cb7f
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleRoot.cpp
@@ -0,0 +1,270 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbMutex.h>
+#include <common/DiagArea.hpp>
+#include "HandleRoot.hpp"
+#include "HandleEnv.hpp"
+#include "HandleDbc.hpp"
+#include "HandleStmt.hpp"
+#include "HandleDesc.hpp"
+#include "PoolNdb.hpp"
+
+HandleRoot::HandleRoot() :
+ m_attrArea(m_attrSpec)
+{
+ m_attrArea.setHandle(this);
+ m_poolNdb = new PoolNdb();
+}
+
+HandleRoot::~HandleRoot()
+{
+}
+
+#ifdef NDB_WIN32
+static NdbMutex & root_mutex = * NdbMutex_Create();
+#else
+static NdbMutex root_mutex = NDB_MUTEX_INITIALIZER;
+#endif
+
+HandleRoot*
+HandleRoot::instance()
+{
+ NdbMutex_Lock(&root_mutex);
+ if (m_instance == 0)
+ m_instance = new HandleRoot();
+ NdbMutex_Unlock(&root_mutex);
+ return m_instance;
+}
+
+void
+HandleRoot::lockHandle()
+{
+ NdbMutex_Lock(&root_mutex);
+}
+
+void
+HandleRoot::unlockHandle()
+{
+ NdbMutex_Unlock(&root_mutex);
+}
+
+// check and find handle types and handles
+
+SQLSMALLINT
+HandleRoot::findParentType(SQLSMALLINT childType)
+{
+ switch (childType) {
+ case SQL_HANDLE_ENV:
+ return SQL_HANDLE_ROOT;
+ case SQL_HANDLE_DBC:
+ return SQL_HANDLE_ENV;
+ case SQL_HANDLE_STMT:
+ return SQL_HANDLE_DBC;
+ case SQL_HANDLE_DESC:
+ return SQL_HANDLE_DBC;
+ }
+ return -1;
+}
+
+HandleBase*
+HandleRoot::findBase(SQLSMALLINT handleType, void* pHandle)
+{
+ switch (handleType) {
+ case SQL_HANDLE_ROOT:
+ return getRoot();
+ case SQL_HANDLE_ENV:
+ return findEnv(pHandle);
+ case SQL_HANDLE_DBC:
+ return findDbc(pHandle);
+ case SQL_HANDLE_STMT:
+ return findStmt(pHandle);
+ case SQL_HANDLE_DESC:
+ return findDesc(pHandle);
+ }
+ return 0;
+}
+
+HandleEnv*
+HandleRoot::findEnv(void* pHandle)
+{
+ lockHandle();
+ ValidList::iterator i = m_validList.find(pHandle);
+ if (i == m_validList.end() || (*i).second != SQL_HANDLE_ENV) {
+ unlockHandle();
+ return 0;
+ }
+ unlockHandle();
+ ctx_assert(pHandle != 0);
+ return static_cast<HandleEnv*>(pHandle);
+}
+
+HandleDbc*
+HandleRoot::findDbc(void* pHandle)
+{
+ lockHandle();
+ ValidList::iterator i = m_validList.find(pHandle);
+ if (i == m_validList.end() || (*i).second != SQL_HANDLE_DBC) {
+ unlockHandle();
+ return 0;
+ }
+ unlockHandle();
+ ctx_assert(pHandle != 0);
+ return static_cast<HandleDbc*>(pHandle);
+}
+
+HandleStmt*
+HandleRoot::findStmt(void* pHandle)
+{
+ lockHandle();
+ ValidList::iterator i = m_validList.find(pHandle);
+ if (i == m_validList.end() || (*i).second != SQL_HANDLE_STMT) {
+ unlockHandle();
+ return 0;
+ }
+ unlockHandle();
+ ctx_assert(pHandle != 0);
+ return static_cast<HandleStmt*>(pHandle);
+}
+
+HandleDesc*
+HandleRoot::findDesc(void* pHandle)
+{
+ lockHandle();
+ ValidList::iterator i = m_validList.find(pHandle);
+ if (i == m_validList.end() || (*i).second != SQL_HANDLE_DESC) {
+ unlockHandle();
+ return 0;
+ }
+ unlockHandle();
+ ctx_assert(pHandle != 0);
+ return static_cast<HandleDesc*>(pHandle);
+}
+
+// add or remove handle from validation list
+
+void
+HandleRoot::record(SQLSMALLINT handleType, HandleBase* pHandle, bool add)
+{
+ switch (handleType) {
+ case SQL_HANDLE_ENV:
+ case SQL_HANDLE_DBC:
+ case SQL_HANDLE_STMT:
+ case SQL_HANDLE_DESC:
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ ctx_assert(pHandle != 0);
+ lockHandle();
+ ValidList::iterator i = m_validList.find(pHandle);
+ if (add) {
+ if (i != m_validList.end()) {
+ unlockHandle();
+ ctx_assert(false);
+ }
+ m_validList.insert(ValidList::value_type(pHandle, handleType));
+ } else {
+ if (i == m_validList.end() || (*i).second != handleType) {
+ unlockHandle();
+ ctx_assert(false);
+ }
+ m_validList.erase(i);
+ }
+ unlockHandle();
+}
+
+// allocate and free handles
+
+void
+HandleRoot::sqlAllocEnv(Ctx& ctx, HandleEnv** ppEnv)
+{
+ if (ppEnv == 0) {
+ ctx.pushStatus(Sqlstate::_HY009, Error::Gen, "cannot allocate environment handle - null return address");
+ return;
+ }
+ HandleEnv* pEnv = new HandleEnv(this);
+ pEnv->ctor(ctx);
+ if (! ctx.ok()) {
+ pEnv->dtor(ctx);
+ delete pEnv;
+ return;
+ }
+ lockHandle();
+ m_listEnv.push_back(pEnv);
+ unlockHandle();
+ getRoot()->record(SQL_HANDLE_ENV, pEnv, true);
+ *ppEnv = pEnv;
+}
+
+void
+HandleRoot::sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild)
+{
+ switch (childType) {
+ case SQL_HANDLE_ENV:
+ sqlAllocEnv(ctx, (HandleEnv**)ppChild);
+ return;
+ }
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "invalid child handle type %d", (int)childType);
+}
+
+void
+HandleRoot::sqlFreeEnv(Ctx& ctx, HandleEnv* pEnv)
+{
+ pEnv->dtor(ctx);
+ if (! ctx.ok()) {
+ return;
+ }
+ lockHandle();
+ m_listEnv.remove(pEnv);
+ unlockHandle();
+ getRoot()->record(SQL_HANDLE_ENV, pEnv, false);
+ delete pEnv;
+}
+
+void
+HandleRoot::sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* pChild)
+{
+ switch (childType) {
+ case SQL_HANDLE_ENV:
+ sqlFreeEnv(ctx, (HandleEnv*)pChild);
+ return;
+ }
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "invalid child handle type %d", (int)childType);
+}
+
+// process-level attributes
+
+void
+HandleRoot::sqlSetRootAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength)
+{
+ lockHandle();
+ baseSetHandleAttr(ctx, m_attrArea, attribute, value, stringLength);
+ unlockHandle();
+}
+
+void
+HandleRoot::sqlGetRootAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength)
+{
+ lockHandle();
+ baseGetHandleAttr(ctx, m_attrArea, attribute, value, bufferLength, stringLength);
+ unlockHandle();
+}
+
+// the instance
+
+HandleRoot* HandleRoot::m_instance = 0;
diff --git a/ndb/src/client/odbc/handles/HandleRoot.hpp b/ndb/src/client/odbc/handles/HandleRoot.hpp
new file mode 100644
index 00000000000..08a22b3e400
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleRoot.hpp
@@ -0,0 +1,103 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_HANDLES_HandleRoot_hpp
+#define ODBC_HANDLES_HandleRoot_hpp
+
+#include <list>
+#include <map>
+#include <common/common.hpp>
+#include "HandleBase.hpp"
+
+class HandleEnv;
+class HandleDbc;
+class HandleStmt;
+class HandleDesc;
+class PoolNdb;
+
+/**
+ * @class HandleRoot
+ * @brief The singleton root handle.
+ *
+ * This class is the level above HandleEnv. It has a single
+ * instance. The instance is the root node of dynamically
+ * allocated handles. The instance is also used to call methods
+ * not tied to any handle.
+ */
+class HandleRoot : public HandleBase {
+protected:
+ HandleRoot();
+ ~HandleRoot();
+public:
+ static HandleRoot* instance();
+ HandleRoot* getRoot();
+ HandleBase* getParent();
+ PoolNdb* getPoolNdb();
+ OdbcHandle odbcHandle();
+ void lockHandle();
+ void unlockHandle();
+ // check and find handle types and handles
+ SQLSMALLINT findParentType(SQLSMALLINT childType);
+ HandleBase* findBase(SQLSMALLINT handleType, void* pHandle);
+ HandleEnv* findEnv(void* pHandle);
+ HandleDbc* findDbc(void* pHandle);
+ HandleStmt* findStmt(void* pHandle);
+ HandleDesc* findDesc(void* pHandle);
+ // add or remove handle from validation list
+ void record(SQLSMALLINT handleType, HandleBase* pHandle, bool add);
+ // allocate and free handles
+ void sqlAllocEnv(Ctx& ctx, HandleEnv** ppEnv);
+ void sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild);
+ void sqlFreeEnv(Ctx& ctx, HandleEnv* pEnv);
+ void sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* pChild);
+ // process-level attributes
+ void sqlSetRootAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength);
+ void sqlGetRootAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength);
+private:
+ static HandleRoot* m_instance; // the instance
+ std::list<HandleEnv*> m_listEnv;
+ PoolNdb* m_poolNdb;
+ typedef std::map<void*, SQLSMALLINT> ValidList;
+ ValidList m_validList;
+ static AttrSpec m_attrSpec[];
+ AttrArea m_attrArea;
+};
+
+inline HandleRoot*
+HandleRoot::getRoot()
+{
+ return this;
+}
+
+inline HandleBase*
+HandleRoot::getParent()
+{
+ return 0;
+}
+
+inline PoolNdb*
+HandleRoot::getPoolNdb()
+{
+ return m_poolNdb;
+}
+
+inline OdbcHandle
+HandleRoot::odbcHandle()
+{
+ return Odbc_handle_root;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/handles/HandleStmt.cpp b/ndb/src/client/odbc/handles/HandleStmt.cpp
new file mode 100644
index 00000000000..d33d33dbd5b
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleStmt.cpp
@@ -0,0 +1,823 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common/OdbcData.hpp>
+#include <codegen/CodeGen.hpp>
+#include "HandleRoot.hpp"
+#include "HandleDbc.hpp"
+#include "HandleStmt.hpp"
+#include "HandleDesc.hpp"
+
+HandleStmt::HandleStmt(HandleDbc* pDbc) :
+ StmtArea(*pDbc),
+ m_dbc(pDbc),
+ m_attrArea(m_attrSpec)
+{
+ m_attrArea.setHandle(this);
+ for (unsigned i = 0; i <= 1; i++) {
+ for (unsigned u = 0; u <= 4; u++) {
+ m_handleDesc[i][u] = 0;
+ }
+ }
+}
+
+HandleStmt::~HandleStmt()
+{
+}
+
+void
+HandleStmt::ctor(Ctx& ctx)
+{
+ for (unsigned u = 1; u <= 4; u++) {
+ HandleDesc** ppDesc = &m_handleDesc[0][u];
+ m_dbc->sqlAllocDesc(ctx, ppDesc);
+ if (! ctx.ok())
+ return;
+ m_descArea[u] = &(*ppDesc)->descArea();
+ m_descArea[u]->setAlloc(Desc_alloc_auto);
+ m_descArea[u]->setUsage((DescUsage)u);
+ }
+}
+
+void
+HandleStmt::dtor(Ctx& ctx)
+{
+ free(ctx);
+ for (unsigned u = 1; u <= 4; u++) {
+ HandleDesc** ppDesc = &m_handleDesc[0][u];
+ if (*ppDesc != 0)
+ m_dbc->sqlFreeDesc(ctx, *ppDesc);
+ *ppDesc = 0;
+ }
+}
+
+// descriptor handles
+
+HandleDesc*
+HandleStmt::getHandleDesc(Ctx& ctx, DescUsage u) const
+{
+ ctx_assert(1 <= u && u <= 4);
+ if (m_handleDesc[1][u] != 0)
+ return m_handleDesc[1][u];
+ return m_handleDesc[0][u];
+}
+
+void
+HandleStmt::setHandleDesc(Ctx& ctx, DescUsage u, SQLPOINTER handle)
+{
+ ctx_assert(1 <= u && u <= 4);
+ if (handle == SQL_NULL_HDESC) {
+ m_handleDesc[1][u] = 0; // revert to implicit
+ m_descArea[u] = &m_handleDesc[0][u]->descArea();
+ return;
+ }
+ HandleDesc* pDesc = getRoot()->findDesc(handle);
+ if (pDesc == 0) {
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "cannot set %s handle to invalid descriptor handle %x", DescArea::nameUsage(u), (unsigned)handle);
+ return;
+ }
+ if (pDesc == m_handleDesc[0][u]) {
+ m_handleDesc[1][u] = 0; // revert to implicit
+ m_descArea[u] = &m_handleDesc[0][u]->descArea();
+ return;
+ }
+ // XXX should check implicit handles on all statements
+ for (unsigned v = 1; v <= 4; v++) {
+ if (pDesc == m_handleDesc[0][v]) {
+ ctx.pushStatus(Sqlstate::_HY024, Error::Gen, "cannot set %s handle to implicitly allocated %s handle", DescArea::nameUsage(u), DescArea::nameUsage((DescUsage)v));
+ return;
+ }
+ }
+ m_handleDesc[1][u] = pDesc;
+ m_descArea[u] = &m_handleDesc[1][u]->descArea();
+}
+
+// allocate and free handles (no valid case)
+
+void
+HandleStmt::sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild)
+{
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "inappropriate handle type");
+}
+
+void
+HandleStmt::sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* ppChild)
+{
+ ctx.pushStatus(Sqlstate::_HY092, Error::Gen, "inappropriate handle type");
+}
+
+// attributes and info
+
+static bool
+ignore_attr(Ctx& ctx, SQLINTEGER attribute)
+{
+ switch (attribute) {
+ case 1211:
+ case 1227:
+ case 1228:
+ ctx_log2(("ignore unknown ADO.NET stmt attribute %d", (int)attribute));
+ return true;
+ }
+ return false;
+}
+
+void
+HandleStmt::sqlSetStmtAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength)
+{
+ if (ignore_attr(ctx, attribute))
+ return;
+ baseSetHandleAttr(ctx, m_attrArea, attribute, value, stringLength);
+}
+
+void
+HandleStmt::sqlGetStmtAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength)
+{
+ if (ignore_attr(ctx, attribute))
+ return;
+ baseGetHandleAttr(ctx, m_attrArea, attribute, value, bufferLength, stringLength);
+}
+
+void
+HandleStmt::sqlSetStmtOption(Ctx& ctx, SQLUSMALLINT option, SQLUINTEGER value)
+{
+ if (ignore_attr(ctx, option))
+ return;
+ baseSetHandleOption(ctx, m_attrArea, option, value);
+}
+
+void
+HandleStmt::sqlGetStmtOption(Ctx& ctx, SQLUSMALLINT option, SQLPOINTER value)
+{
+ if (ignore_attr(ctx, option))
+ return;
+ baseGetHandleOption(ctx, m_attrArea, option, value);
+}
+
+void
+HandleStmt::sqlGetTypeInfo(Ctx& ctx, SQLSMALLINT dataType)
+{
+ BaseString text;
+ // odbc$typeinfo is a (possible unordered) view matching SQLGetTypeInfo result set
+ text.append("SELECT * FROM odbc$typeinfo");
+ if (dataType != SQL_ALL_TYPES) {
+ switch (dataType) {
+ case SQL_CHAR:
+ case SQL_VARCHAR:
+ case SQL_BINARY:
+ case SQL_VARBINARY:
+ case SQL_SMALLINT:
+ case SQL_INTEGER:
+ case SQL_BIGINT:
+ case SQL_REAL:
+ case SQL_DOUBLE:
+ break;
+ default:
+ // XXX unsupported vs illegal
+ ctx_log1(("sqlGetTypeInfo: unknown data type %d", (int)dataType));
+ break;
+ }
+ text.appfmt(" WHERE data_type = %d", (int)dataType);
+ }
+ // sort signed before unsigned
+ text.append(" ORDER BY data_type, unsigned_attribute, type_name");
+ sqlExecDirect(ctx, (SQLCHAR*)text.c_str(), text.length());
+}
+
+void
+HandleStmt::sqlTables(Ctx& ctx, SQLCHAR* catalogName, SQLSMALLINT nameLength1, SQLCHAR* schemaName, SQLSMALLINT nameLength2, SQLCHAR* tableName, SQLSMALLINT nameLength3, SQLCHAR* tableType, SQLSMALLINT nameLength4)
+{
+ BaseString text;
+ // odbc$tables is a (possibly unordered) view matching SQLTables result set
+ text.append("SELECT * FROM odbc$tables");
+ SQLUINTEGER metadata_id = SQL_FALSE;
+ sqlGetStmtAttr(ctx, SQL_ATTR_METADATA_ID, (SQLPOINTER)&metadata_id, SQL_IS_POINTER, 0);
+ if (! ctx.ok())
+ return;
+ unsigned count = 0;
+ if (catalogName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)catalogName, nameLength1);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (getOdbcVersion(ctx) == 2)
+ text.appfmt(" table_cat = '%s'", data.sqlchar());
+ else if (metadata_id == SQL_TRUE)
+ text.appfmt(" table_cat = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" table_cat LIKE '%s'", data.sqlchar());
+ }
+ if (schemaName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)schemaName, nameLength2);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (metadata_id == SQL_TRUE)
+ text.appfmt(" table_schem = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" table_schem LIKE '%s'", data.sqlchar());
+ }
+ if (tableName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)tableName, nameLength3);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (metadata_id == SQL_TRUE)
+ text.appfmt(" table_name = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" table_name LIKE '%s'", data.sqlchar());
+ }
+ if (tableType != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)tableType, nameLength4);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ text.appfmt(" table_type IN (%s)", data.sqlchar()); // XXX UPPER, quotes
+ }
+ text.append(" ORDER BY table_type, table_cat, table_schem, table_name");
+ sqlExecDirect(ctx, (SQLCHAR*)text.c_str(), text.length());
+}
+
+void
+HandleStmt::sqlColumns(Ctx& ctx, SQLCHAR* catalogName, SQLSMALLINT nameLength1, SQLCHAR* schemaName, SQLSMALLINT nameLength2, SQLCHAR* tableName, SQLSMALLINT nameLength3, SQLCHAR* columnName, SQLSMALLINT nameLength4)
+{
+ BaseString text;
+ // odbc$columns is a (possibly unordered) view matching SQLColumns result set
+ text.append("SELECT * FROM odbc$columns");
+ SQLUINTEGER metadata_id = SQL_FALSE;
+ sqlGetStmtAttr(ctx, SQL_ATTR_METADATA_ID, (SQLPOINTER)&metadata_id, SQL_IS_POINTER, 0);
+ if (! ctx.ok())
+ return;
+ unsigned count = 0;
+ if (catalogName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)catalogName, nameLength1);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (getOdbcVersion(ctx) == 2)
+ text.appfmt(" table_cat = '%s'", data.sqlchar());
+ else if (metadata_id == SQL_TRUE)
+ text.appfmt(" table_cat = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" table_cat LIKE '%s'", data.sqlchar());
+ }
+ if (schemaName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)schemaName, nameLength2);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (metadata_id == SQL_TRUE)
+ text.appfmt(" table_schem = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" table_schem LIKE '%s'", data.sqlchar());
+ }
+ if (tableName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)tableName, nameLength3);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (metadata_id == SQL_TRUE)
+ text.appfmt(" table_name = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" table_name LIKE '%s'", data.sqlchar());
+ }
+ if (columnName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)columnName, nameLength4);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (metadata_id == SQL_TRUE)
+ text.appfmt(" column_name = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" column_name LIKE '%s'", data.sqlchar());
+ }
+ text.append(" ORDER BY table_cat, table_schem, table_name, ordinal_position");
+ sqlExecDirect(ctx, (SQLCHAR*)text.c_str(), text.length());
+}
+
+void
+HandleStmt::sqlPrimaryKeys(Ctx& ctx, SQLCHAR* szCatalogName, SQLSMALLINT cbCatalogName, SQLCHAR* szSchemaName, SQLSMALLINT cbSchemaName, SQLCHAR* szTableName, SQLSMALLINT cbTableName)
+{
+ BaseString text;
+ // odbc$primarykeys is a (possible unordered) view matching SQLPrimaryKeys result set
+ text.append("SELECT * FROM odbc$primarykeys");
+ SQLUINTEGER metadata_id = SQL_FALSE;
+ sqlGetStmtAttr(ctx, SQL_ATTR_METADATA_ID, (SQLPOINTER)&metadata_id, SQL_IS_POINTER, 0);
+ if (! ctx.ok())
+ return;
+ unsigned count = 0;
+ if (szCatalogName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)szCatalogName, cbCatalogName);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (getOdbcVersion(ctx) == 2)
+ text.appfmt(" table_cat = '%s'", data.sqlchar());
+ else if (metadata_id == SQL_TRUE)
+ text.appfmt(" table_cat = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" table_cat = '%s'", data.sqlchar()); // no pattern
+ }
+ if (szSchemaName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)szSchemaName, cbSchemaName);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (metadata_id == SQL_TRUE)
+ text.appfmt(" table_schem = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" table_schem = '%s'", data.sqlchar()); // no pattern
+ }
+ if (szTableName != 0) {
+ OdbcData data;
+ data.copyin(ctx, OdbcData::Sqlchar, (SQLPOINTER)szTableName, cbTableName);
+ if (! ctx.ok())
+ return;
+ text.append(++count == 1 ? " WHERE" : " AND");
+ if (metadata_id == SQL_TRUE)
+ text.appfmt(" table_name = '%s'", data.sqlchar()); // XXX UPPER
+ else
+ text.appfmt(" table_name = '%s'", data.sqlchar()); // no pattern
+ } else { // no null
+ ctx.pushStatus(Sqlstate::_HY009, Error::Gen, "null table name");
+ return;
+ }
+ text.append(" ORDER BY table_cat, table_schem, table_name, key_seq");
+ sqlExecDirect(ctx, (SQLCHAR*)text.c_str(), text.length());
+}
+
+int
+HandleStmt::getOdbcVersion(Ctx& ctx)
+{
+ return m_dbc->getOdbcVersion(ctx);
+}
+
+// prepare and execute
+
+void
+HandleStmt::sqlPrepare(Ctx& ctx, SQLCHAR* statementText, SQLINTEGER textLength)
+{
+ if (m_state == Open) {
+ ctx.pushStatus(Sqlstate::_24000, Error::Gen, "cursor is open");
+ return;
+ }
+ free(ctx);
+ const char* text = reinterpret_cast<char*>(statementText);
+ if (textLength == SQL_NTS) {
+ m_sqlText.assign(text);
+ } else if (textLength >= 0) {
+ m_sqlText.assign(text, textLength);
+ } else {
+ ctx.pushStatus(Sqlstate::_HY090, Error::Gen, "missing SQL text");
+ return;
+ }
+ if (! useSchemaCon(ctx, true))
+ return;
+ CodeGen codegen(*this);
+ codegen.prepare(ctx);
+ useSchemaCon(ctx, false);
+ if (! ctx.ok()) {
+ free(ctx);
+ return;
+ }
+ ctx_log2(("prepared %s statement", m_stmtInfo.getDesc()));
+ m_state = Prepared;
+}
+
+void
+HandleStmt::sqlExecute(Ctx& ctx)
+{
+ if (m_state == Open) {
+ ctx.pushStatus(Sqlstate::_24000, Error::Gen, "cursor is open");
+ return;
+ }
+ StmtType stmtType = m_stmtInfo.getType();
+ switch (stmtType) {
+ case Stmt_type_DDL:
+ if (! useSchemaCon(ctx, true))
+ return;
+ break;
+ case Stmt_type_query:
+ case Stmt_type_DML:
+ if (! useConnection(ctx, true))
+ return;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+ CodeGen codegen(*this);
+ codegen.execute(ctx);
+ // valid only after execute says M$ XXX create this diag only on demand
+ setFunction(ctx);
+ if (ctx.getCode() == SQL_NEED_DATA) {
+ m_state = NeedData;
+ return;
+ }
+ ctx_log2(("execute: fetched %u tuples from NDB", (unsigned)m_tuplesFetched));
+ switch (stmtType) {
+ case Stmt_type_DDL:
+ codegen.close(ctx);
+ useSchemaCon(ctx, false);
+ m_state = Prepared;
+ break;
+ case Stmt_type_query:
+ if (! ctx.ok()) {
+ codegen.close(ctx);
+ useConnection(ctx, false);
+ m_state = Prepared;
+ } else {
+ m_state = Open;
+ }
+ break;
+ case Stmt_type_DML:
+ codegen.close(ctx);
+ if (m_dbc->autocommit()) {
+ // even if error
+ m_dbc->sqlEndTran(ctx, SQL_COMMIT);
+ } else {
+ m_dbc->uncommitted(true); // uncommitted changes possible
+ }
+ useConnection(ctx, false);
+ m_state = Prepared;
+ break;
+ default:
+ ctx_assert(false);
+ break;
+ }
+}
+
+void
+HandleStmt::sqlExecDirect(Ctx& ctx, SQLCHAR* statementText, SQLINTEGER textLength)
+{
+ sqlPrepare(ctx, statementText, textLength);
+ if (! ctx.ok())
+ return;
+ sqlExecute(ctx);
+}
+
+void
+HandleStmt::sqlFetch(Ctx& ctx)
+{
+ if (m_state != Open) {
+ ctx.pushStatus(Sqlstate::_24000, Error::Gen, "cursor is not open");
+ return;
+ }
+ StmtType stmtType = m_stmtInfo.getType();
+ switch (stmtType) {
+ case Stmt_type_query: {
+ CodeGen codegen(*this);
+ codegen.fetch(ctx);
+ if (! ctx.ok()) {
+ codegen.close(ctx);
+ useConnection(ctx, false);
+ }
+ break;
+ }
+ default:
+ ctx_assert(false);
+ break;
+ }
+}
+
+void
+HandleStmt::sqlRowCount(Ctx& ctx, SQLINTEGER* rowCount)
+{
+ *rowCount = m_rowCount;
+}
+
+void
+HandleStmt::sqlMoreResults(Ctx& ctx)
+{
+ if (m_state == Open) {
+ sqlCloseCursor(ctx);
+ if (! ctx.ok())
+ return;
+ }
+ ctx.setCode(SQL_NO_DATA);
+}
+
+void
+HandleStmt::sqlCancel(Ctx& ctx)
+{
+ if (m_state == NeedData) {
+ CodeGen codegen(*this);
+ codegen.close(ctx);
+ m_state = Prepared;
+ }
+}
+
+void
+HandleStmt::sqlCloseCursor(Ctx& ctx)
+{
+ if (m_state != Open) {
+ ctx.pushStatus(Sqlstate::_24000, Error::Gen, "cursor is not open");
+ return;
+ }
+ ctx_log2(("execute: fetched %u tuples from NDB", (unsigned)m_tuplesFetched));
+ StmtType stmtType = m_stmtInfo.getType();
+ switch (stmtType) {
+ case Stmt_type_query: {
+ CodeGen codegen(*this);
+ codegen.close(ctx);
+ useConnection(ctx, false);
+ m_state = Prepared;
+ m_rowCount = 0;
+ m_tuplesFetched = 0;
+ break;
+ }
+ default:
+ ctx_assert(false);
+ break;
+ }
+}
+
+void
+HandleStmt::sqlGetCursorName(Ctx& ctx, SQLCHAR* cursorName, SQLSMALLINT bufferLength, SQLSMALLINT* nameLength)
+{
+ OdbcData name("SQL_CUR_DUMMY");
+ name.copyout(ctx, cursorName, bufferLength, 0, nameLength);
+}
+
+void
+HandleStmt::sqlSetCursorName(Ctx& ctx, SQLCHAR* cursorName, SQLSMALLINT nameLength)
+{
+}
+
+// special data access
+
+void
+HandleStmt::sqlGetData(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind)
+{
+ if (m_state != Open) {
+ ctx.pushStatus(Sqlstate::_24000, Error::Gen, "cursor is not open");
+ return;
+ }
+ if (bufferLength < 0) {
+ ctx.pushStatus(Sqlstate::_HY090, Error::Gen, "invalid buffer length %d", (int)bufferLength);
+ return;
+ }
+ CodeGen codegen(*this);
+ codegen.sqlGetData(ctx, columnNumber, targetType, targetValue, bufferLength, strlen_or_Ind);
+}
+
+void
+HandleStmt::sqlParamData(Ctx& ctx, SQLPOINTER* value)
+{
+ if (m_state != NeedData) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "not expecting data-at-exec");
+ return;
+ }
+ CodeGen codegen(*this);
+ codegen.sqlParamData(ctx, value);
+ if (! ctx.ok())
+ return;
+ sqlExecute(ctx);
+}
+
+void
+HandleStmt::sqlPutData(Ctx& ctx, SQLPOINTER data, SQLINTEGER strlen_or_Ind)
+{
+ if (m_state != NeedData) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "not expecting data-at-exec");
+ return;
+ }
+ CodeGen codegen(*this);
+ codegen.sqlPutData(ctx, data, strlen_or_Ind);
+}
+
+// describe statement
+
+void
+HandleStmt::sqlNumParams(Ctx& ctx, SQLSMALLINT* parameterCountPtr)
+{
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "statement is not prepared");
+ return;
+ }
+ HandleDesc* ipd = getHandleDesc(ctx, Desc_usage_IPD);
+ ipd->sqlGetDescField(ctx, 0, SQL_DESC_COUNT, static_cast<SQLPOINTER>(parameterCountPtr), -1, 0);
+}
+
+void
+HandleStmt::sqlDescribeParam(Ctx& ctx, SQLUSMALLINT ipar, SQLSMALLINT* pfSqlType, SQLUINTEGER* pcbParamDef, SQLSMALLINT* pibScale, SQLSMALLINT* pfNullable)
+{
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "statement is not prepared");
+ return;
+ }
+ HandleDesc* ipd = getHandleDesc(ctx, Desc_usage_IPD);
+ ipd->sqlGetDescField(ctx, ipar, SQL_DESC_CONCISE_TYPE, static_cast<SQLPOINTER>(pfSqlType), -1, 0);
+ { // XXX fix it
+ OdbcData data((SQLUINTEGER)0);
+ data.copyout(ctx, (SQLPOINTER)pcbParamDef, -1, 0);
+ }
+ { // XXX fix it
+ OdbcData data((SQLSMALLINT)0);
+ data.copyout(ctx, (SQLPOINTER)pibScale, -1, 0);
+ }
+ ipd->sqlGetDescField(ctx, ipar, SQL_DESC_NULLABLE, static_cast<SQLPOINTER>(pfNullable), -1, 0);
+}
+
+void
+HandleStmt::sqlNumResultCols(Ctx& ctx, SQLSMALLINT* columnCount)
+{
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "statement is not prepared");
+ return;
+ }
+ HandleDesc* const ird = getHandleDesc(ctx, Desc_usage_IRD);
+ ird->sqlGetDescField(ctx, 0, SQL_DESC_COUNT, static_cast<SQLPOINTER>(columnCount), -1, 0);
+ setFunction(ctx); // unixODBC workaround
+}
+
+void
+HandleStmt::sqlDescribeCol(Ctx& ctx, SQLUSMALLINT columnNumber, SQLCHAR* columnName, SQLSMALLINT bufferLength, SQLSMALLINT* nameLength, SQLSMALLINT* dataType, SQLUINTEGER* columnSize, SQLSMALLINT* decimalDigits, SQLSMALLINT* nullable)
+{
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "statement is not prepared");
+ return;
+ }
+ HandleDesc* const ird = getHandleDesc(ctx, Desc_usage_IRD);
+ ird->sqlGetDescField(ctx, columnNumber, SQL_DESC_NAME, static_cast<SQLPOINTER>(columnName), bufferLength, 0, nameLength);
+ ird->sqlGetDescField(ctx, columnNumber, SQL_DESC_CONCISE_TYPE, static_cast<SQLPOINTER>(dataType), -1, 0);
+ if (! ctx.ok())
+ return;
+ if (columnSize != 0) {
+ switch (*dataType) {
+ case SQL_CHAR:
+ case SQL_VARCHAR:
+ case SQL_BINARY:
+ case SQL_VARBINARY:
+ ird->sqlGetDescField(ctx, columnNumber, SQL_DESC_LENGTH, static_cast<SQLPOINTER>(columnSize), -1, 0);
+ break;
+ case SQL_SMALLINT:
+ *columnSize = 5;
+ break;
+ case SQL_INTEGER:
+ *columnSize = 10;
+ break;
+ case SQL_BIGINT:
+ *columnSize = 20; // XXX 19 for signed
+ break;
+ case SQL_REAL:
+ *columnSize = 7;
+ break;
+ case SQL_DOUBLE:
+ *columnSize = 15;
+ break;
+ case SQL_TYPE_TIMESTAMP:
+ *columnSize = 30;
+ break;
+ default:
+ *columnSize = 0;
+ break;
+ }
+ }
+ if (decimalDigits != 0) {
+ switch (*dataType) {
+ case SQL_SMALLINT:
+ case SQL_INTEGER:
+ case SQL_BIGINT:
+ *decimalDigits = 0;
+ break;
+ case SQL_TYPE_TIMESTAMP:
+ *decimalDigits = 10;
+ break;
+ default:
+ *decimalDigits = 0;
+ break;
+ }
+ }
+ ird->sqlGetDescField(ctx, columnNumber, SQL_DESC_NULLABLE, static_cast<SQLPOINTER>(nullable), -1, 0);
+}
+
+void
+HandleStmt::sqlColAttribute(Ctx& ctx, SQLUSMALLINT columnNumber, SQLUSMALLINT fieldIdentifier, SQLPOINTER characterAttribute, SQLSMALLINT bufferLength, SQLSMALLINT* stringLength, SQLPOINTER numericAttribute)
+{
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "statement is not prepared");
+ return;
+ }
+ HandleDesc* const ird = getHandleDesc(ctx, Desc_usage_IRD);
+ ird->sqlColAttribute(ctx, columnNumber, fieldIdentifier, characterAttribute, bufferLength, stringLength, numericAttribute);
+}
+
+void
+HandleStmt::sqlColAttributes(Ctx& ctx, SQLUSMALLINT icol, SQLUSMALLINT fdescType, SQLPOINTER rgbDesc, SQLSMALLINT cbDescMax, SQLSMALLINT* pcbDesc, SQLINTEGER* pfDesc)
+{
+ if (m_state == Free) {
+ ctx.pushStatus(Sqlstate::_HY010, Error::Gen, "statement is nor prepared");
+ return;
+ }
+ HandleDesc* const ird = getHandleDesc(ctx, Desc_usage_IRD);
+ ird->sqlColAttributes(ctx, icol, fdescType, rgbDesc, cbDescMax, pcbDesc, pfDesc);
+}
+
+// descriptor manipulation
+
+void
+HandleStmt::sqlBindCol(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind)
+{
+ HandleDesc* const ard = getHandleDesc(ctx, Desc_usage_ARD);
+ DescArea& desc = ard->descArea();
+ if (desc.getCount() < columnNumber) {
+ desc.setCount(ctx, columnNumber);
+ }
+ DescRec& rec = desc.getRecord(columnNumber);
+ rec.setField(ctx, SQL_DESC_TYPE, targetType);
+ rec.setField(ctx, SQL_DESC_CONCISE_TYPE, targetType);
+ rec.setField(ctx, SQL_DESC_DATA_PTR, targetValue);
+ rec.setField(ctx, SQL_DESC_OCTET_LENGTH, bufferLength);
+ rec.setField(ctx, SQL_DESC_OCTET_LENGTH_PTR, strlen_or_Ind);
+ rec.setField(ctx, SQL_DESC_INDICATOR_PTR, strlen_or_Ind);
+}
+
+void
+HandleStmt::sqlBindParameter(Ctx& ctx, SQLUSMALLINT ipar, SQLSMALLINT fParamType, SQLSMALLINT fCType, SQLSMALLINT fSqlType, SQLUINTEGER cbColDef, SQLSMALLINT ibScale, SQLPOINTER rgbValue, SQLINTEGER cbValueMax, SQLINTEGER* pcbValue)
+{
+ HandleDesc* const ipd = getHandleDesc(ctx, Desc_usage_IPD);
+ HandleDesc* const apd = getHandleDesc(ctx, Desc_usage_APD);
+ DescArea& descIPD = ipd->descArea();
+ DescArea& descAPD = apd->descArea();
+ if (ipar < 1) {
+ ctx.pushStatus(Sqlstate::_07009, Error::Gen, "invalid parameter index %u", (unsigned)ipar);
+ return;
+ }
+ if (descIPD.getCount() < ipar) {
+ descIPD.setCount(ctx, ipar);
+ }
+ if (descAPD.getCount() < ipar) {
+ descAPD.setCount(ctx, ipar);
+ }
+ DescRec& recIPD = descIPD.getRecord(ipar);
+ DescRec& recAPD = descAPD.getRecord(ipar);
+ recIPD.setField(ctx, SQL_DESC_PARAMETER_TYPE, fParamType);
+ recAPD.setField(ctx, SQL_DESC_TYPE, fCType);
+ recAPD.setField(ctx, SQL_DESC_CONCISE_TYPE, fCType);
+ recIPD.setField(ctx, SQL_DESC_TYPE, fSqlType);
+ recIPD.setField(ctx, SQL_DESC_CONCISE_TYPE, fSqlType);
+ switch (fSqlType) {
+ case SQL_CHAR: // XXX not complete
+ case SQL_VARCHAR:
+ case SQL_BINARY:
+ case SQL_VARBINARY:
+ recIPD.setField(ctx, SQL_DESC_LENGTH, cbColDef);
+ break;
+ case SQL_DECIMAL:
+ case SQL_NUMERIC:
+ case SQL_FLOAT:
+ case SQL_REAL:
+ case SQL_DOUBLE:
+ recIPD.setField(ctx, SQL_DESC_PRECISION, cbColDef);
+ break;
+ }
+ switch (fSqlType) {
+ case SQL_TYPE_TIME: // XXX not complete
+ case SQL_TYPE_TIMESTAMP:
+ recIPD.setField(ctx, SQL_DESC_PRECISION, ibScale);
+ break;
+ case SQL_NUMERIC:
+ case SQL_DECIMAL:
+ recIPD.setField(ctx, SQL_DESC_SCALE, ibScale);
+ break;
+ }
+ recAPD.setField(ctx, SQL_DESC_DATA_PTR, rgbValue);
+ recAPD.setField(ctx, SQL_DESC_OCTET_LENGTH, cbValueMax);
+ recAPD.setField(ctx, SQL_DESC_OCTET_LENGTH_PTR, pcbValue);
+ recAPD.setField(ctx, SQL_DESC_INDICATOR_PTR, pcbValue);
+}
+
+void
+HandleStmt::sqlBindParam(Ctx& ctx, SQLUSMALLINT parameterNumber, SQLSMALLINT valueType, SQLSMALLINT parameterType, SQLUINTEGER lengthPrecision, SQLSMALLINT parameterScale, SQLPOINTER parameterValue, SQLINTEGER* strLen_or_Ind)
+{
+ sqlBindParameter(ctx, parameterNumber, SQL_PARAM_INPUT_OUTPUT, valueType, parameterType, lengthPrecision, parameterScale, parameterValue, SQL_SETPARAM_VALUE_MAX, strLen_or_Ind);
+}
+
+void
+HandleStmt::sqlSetParam(Ctx& ctx, SQLUSMALLINT parameterNumber, SQLSMALLINT valueType, SQLSMALLINT parameterType, SQLUINTEGER lengthPrecision, SQLSMALLINT parameterScale, SQLPOINTER parameterValue, SQLINTEGER* strLen_or_Ind)
+{
+ sqlBindParameter(ctx, parameterNumber, SQL_PARAM_INPUT_OUTPUT, valueType, parameterType, lengthPrecision, parameterScale, parameterValue, SQL_SETPARAM_VALUE_MAX, strLen_or_Ind);
+}
diff --git a/ndb/src/client/odbc/handles/HandleStmt.hpp b/ndb/src/client/odbc/handles/HandleStmt.hpp
new file mode 100644
index 00000000000..0bee138bfc6
--- /dev/null
+++ b/ndb/src/client/odbc/handles/HandleStmt.hpp
@@ -0,0 +1,117 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_HANDLES_HandleStmt_hpp
+#define ODBC_HANDLES_HandleStmt_hpp
+
+#include <common/common.hpp>
+#include <common/DescArea.hpp>
+#include <common/StmtArea.hpp>
+#include "HandleBase.hpp"
+
+class HandleDbc;
+class HandleDesc;
+
+/**
+ * @class HandleStmt
+ * @brief Statement handle (SQLHSTMT).
+ */
+class HandleStmt : public HandleBase, public StmtArea {
+public:
+ HandleStmt(HandleDbc* pDbc);
+ ~HandleStmt();
+ void ctor(Ctx& ctx);
+ void dtor(Ctx& ctx);
+ HandleDbc* getDbc();
+ HandleBase* getParent();
+ HandleRoot* getRoot();
+ OdbcHandle odbcHandle();
+ // descriptor handles
+ HandleDesc* getHandleDesc(Ctx& ctx, DescUsage u) const;
+ void setHandleDesc(Ctx& ctx, DescUsage u, SQLPOINTER handle);
+ // allocate and free handles (no valid case)
+ void sqlAllocHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase** ppChild);
+ void sqlFreeHandle(Ctx& ctx, SQLSMALLINT childType, HandleBase* pChild);
+ // attributes and info
+ void sqlSetStmtAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER stringLength);
+ void sqlGetStmtAttr(Ctx& ctx, SQLINTEGER attribute, SQLPOINTER value, SQLINTEGER bufferLength, SQLINTEGER* stringLength);
+ void sqlSetStmtOption(Ctx& ctx, SQLUSMALLINT option, SQLUINTEGER value); // odbc2.0
+ void sqlGetStmtOption(Ctx& ctx, SQLUSMALLINT option, SQLPOINTER value); // odbc2.0
+ void sqlGetTypeInfo(Ctx& ctx, SQLSMALLINT dataType);
+ void sqlTables(Ctx& ctx, SQLCHAR* catalogName, SQLSMALLINT nameLength1, SQLCHAR* schemaName, SQLSMALLINT nameLength2, SQLCHAR* tableName, SQLSMALLINT nameLength3, SQLCHAR* tableType, SQLSMALLINT nameLength4);
+ void sqlColumns(Ctx& ctx, SQLCHAR* catalogName, SQLSMALLINT nameLength1, SQLCHAR* schemaName, SQLSMALLINT nameLength2, SQLCHAR* tableName, SQLSMALLINT nameLength3, SQLCHAR* columnName, SQLSMALLINT nameLength4);
+ void sqlPrimaryKeys(Ctx& ctx, SQLCHAR* szCatalogName, SQLSMALLINT cbCatalogName, SQLCHAR* szSchemaName, SQLSMALLINT cbSchemaName, SQLCHAR* szTableName, SQLSMALLINT cbTableName);
+ int getOdbcVersion(Ctx& ctx);
+ // prepare and execute
+ void sqlPrepare(Ctx& ctx, SQLCHAR* statementText, SQLINTEGER textLength);
+ void sqlExecute(Ctx& ctx);
+ void sqlExecDirect(Ctx& ctx, SQLCHAR* statementText, SQLINTEGER textLength);
+ void sqlFetch(Ctx& ctx);
+ void sqlRowCount(Ctx& ctx, SQLINTEGER* rowCount);
+ void sqlMoreResults(Ctx& ctx);
+ void sqlCancel(Ctx& ctx);
+ void sqlCloseCursor(Ctx& ctx);
+ void sqlGetCursorName(Ctx& ctx, SQLCHAR* cursorName, SQLSMALLINT bufferLength, SQLSMALLINT* nameLength);
+ void sqlSetCursorName(Ctx& ctx, SQLCHAR* cursorName, SQLSMALLINT nameLength);
+ // special data access
+ void sqlGetData(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind);
+ void sqlParamData(Ctx& ctx, SQLPOINTER* value);
+ void sqlPutData(Ctx& ctx, SQLPOINTER data, SQLINTEGER strlen_or_Ind);
+ // describe statement
+ void sqlNumParams(Ctx& ctx, SQLSMALLINT* ParameterCountPtr);
+ void sqlDescribeParam(Ctx& ctx, SQLUSMALLINT ipar, SQLSMALLINT* pfSqlType, SQLUINTEGER* pcbParamDef, SQLSMALLINT* pibScale, SQLSMALLINT* pfNullable);
+ void sqlNumResultCols(Ctx& ctx, SQLSMALLINT* columnCount);
+ void sqlDescribeCol(Ctx& ctx, SQLUSMALLINT columnNumber, SQLCHAR* columnName, SQLSMALLINT bufferLength, SQLSMALLINT* nameLength, SQLSMALLINT* dataType, SQLUINTEGER* columnSize, SQLSMALLINT* decimalDigits, SQLSMALLINT* nullable);
+ void sqlColAttribute(Ctx& ctx, SQLUSMALLINT columnNumber, SQLUSMALLINT fieldIdentifier, SQLPOINTER characterAttribute, SQLSMALLINT bufferLength, SQLSMALLINT* stringLength, SQLPOINTER numericAttribute);
+ void sqlColAttributes(Ctx& ctx, SQLUSMALLINT icol, SQLUSMALLINT fdescType, SQLPOINTER rgbDesc, SQLSMALLINT cbDescMax, SQLSMALLINT* pcbDesc, SQLINTEGER* pfDesc); // odbc2.0
+ // descriptor manipulation
+ void sqlBindCol(Ctx& ctx, SQLUSMALLINT columnNumber, SQLSMALLINT targetType, SQLPOINTER targetValue, SQLINTEGER bufferLength, SQLINTEGER* strlen_or_Ind);
+ void sqlBindParameter(Ctx& ctx, SQLUSMALLINT ipar, SQLSMALLINT fParamType, SQLSMALLINT fCType, SQLSMALLINT fSqlType, SQLUINTEGER cbColDef, SQLSMALLINT ibScale, SQLPOINTER rgbValue, SQLINTEGER cbValueMax, SQLINTEGER* pcbValue);
+ void sqlBindParam(Ctx& ctx, SQLUSMALLINT parameterNumber, SQLSMALLINT valueType, SQLSMALLINT parameterType, SQLUINTEGER lengthPrecision, SQLSMALLINT parameterScale, SQLPOINTER parameterValue, SQLINTEGER* strLen_or_Ind);
+ void sqlSetParam(Ctx& ctx, SQLUSMALLINT parameterNumber, SQLSMALLINT valueType, SQLSMALLINT parameterType, SQLUINTEGER lengthPrecision, SQLSMALLINT parameterScale, SQLPOINTER parameterValue, SQLINTEGER* strLen_or_Ind);
+private:
+ HandleDbc* const m_dbc;
+ static AttrSpec m_attrSpec[];
+ AttrArea m_attrArea;
+ // descriptor handles (0-automatic 1-application)
+ HandleDesc* m_handleDesc[2][1+4];
+};
+
+inline HandleDbc*
+HandleStmt::getDbc()
+{
+ return m_dbc;
+}
+
+inline HandleBase*
+HandleStmt::getParent()
+{
+ return (HandleBase*)getDbc();
+}
+
+inline HandleRoot*
+HandleStmt::getRoot()
+{
+ return getParent()->getRoot();
+}
+
+inline OdbcHandle
+HandleStmt::odbcHandle()
+{
+ return Odbc_handle_stmt;
+}
+
+#endif
diff --git a/ndb/src/client/odbc/handles/InfoTab.cpp b/ndb/src/client/odbc/handles/InfoTab.cpp
new file mode 100644
index 00000000000..1a93c4da264
--- /dev/null
+++ b/ndb/src/client/odbc/handles/InfoTab.cpp
@@ -0,0 +1,878 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HandleDbc.hpp"
+
+HandleDbc::InfoTab
+HandleDbc::m_infoTab[] = {
+ { SQL_ACCESSIBLE_PROCEDURES,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_ACCESSIBLE_TABLES,
+ InfoTab::YesNo,
+ 0L,
+ "Y"
+ },
+ { SQL_ACTIVE_ENVIRONMENTS,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_AGGREGATE_FUNCTIONS,
+ InfoTab::Bitmask,
+ SQL_AF_AVG | SQL_AF_COUNT | SQL_AF_MAX | SQL_AF_MIN | SQL_AF_SUM,
+ 0
+ },
+ { SQL_ALTER_DOMAIN,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_ALTER_TABLE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_ASYNC_MODE,
+ InfoTab::Long,
+ SQL_AM_NONE,
+ 0
+ },
+ { SQL_BATCH_ROW_COUNT,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_BATCH_SUPPORT,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_BOOKMARK_PERSISTENCE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CATALOG_LOCATION,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_CATALOG_NAME,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_CATALOG_NAME_SEPARATOR,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_CATALOG_TERM,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_CATALOG_USAGE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_COLLATION_SEQ,
+ InfoTab::Char,
+ 0L,
+ "ISO 8859-1"
+ },
+ { SQL_COLUMN_ALIAS,
+ InfoTab::YesNo,
+ 0L,
+ "Y"
+ },
+ { SQL_CONCAT_NULL_BEHAVIOR,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_BIGINT,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_BINARY,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_BIT,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_CHAR,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_DATE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_DECIMAL,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_DOUBLE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_FLOAT,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_FUNCTIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+#if 0
+ { SQL_CONVERT_GUID,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+#endif
+ { SQL_CONVERT_INTEGER,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_INTERVAL_DAY_TIME,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_INTERVAL_YEAR_MONTH,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_LONGVARBINARY,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_LONGVARCHAR,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_NUMERIC,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_REAL,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_SMALLINT,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_TIME,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_TIMESTAMP,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_TINYINT,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_VARBINARY,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CONVERT_VARCHAR,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CORRELATION_NAME,
+ InfoTab::Bitmask,
+ SQL_CN_ANY,
+ 0
+ },
+ { SQL_CREATE_ASSERTION,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CREATE_CHARACTER_SET,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CREATE_COLLATION,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CREATE_DOMAIN,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CREATE_SCHEMA,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CREATE_TABLE,
+ InfoTab::Bitmask,
+ SQL_CT_CREATE_TABLE,
+ 0
+ },
+ { SQL_CREATE_TRANSLATION,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CREATE_VIEW,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_CURSOR_COMMIT_BEHAVIOR,
+ InfoTab::Short,
+ SQL_CB_CLOSE,
+ 0
+ },
+ { SQL_CURSOR_ROLLBACK_BEHAVIOR,
+ InfoTab::Short,
+ SQL_CB_CLOSE,
+ 0
+ },
+ { SQL_CURSOR_SENSITIVITY,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_DATABASE_NAME,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_DATA_SOURCE_NAME,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_DATA_SOURCE_READ_ONLY,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_DATETIME_LITERALS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DBMS_NAME,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_DBMS_VER,
+ InfoTab::Char,
+ 0L,
+ "01.43.0000"
+ },
+ { SQL_DDL_INDEX,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_DEFAULT_TXN_ISOLATION,
+ InfoTab::Long,
+ SQL_TXN_READ_COMMITTED,
+ 0
+ },
+ { SQL_DESCRIBE_PARAMETER,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_DM_VER,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_DRIVER_HDBC,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_DRIVER_HDESC,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_DRIVER_HLIB,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_DRIVER_HSTMT,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_DRIVER_NAME,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_DRIVER_ODBC_VER,
+ InfoTab::Char,
+ 0L,
+ "03.00"
+ },
+ { SQL_DRIVER_VER,
+ InfoTab::Char,
+ 0L,
+ "00.10.0000"
+ },
+ { SQL_DROP_ASSERTION,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DROP_CHARACTER_SET,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DROP_COLLATION,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DROP_DOMAIN,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DROP_SCHEMA,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DROP_TABLE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DROP_TRANSLATION,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DROP_VIEW,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DTC_TRANSITION_COST, // not in older MS docs
+ InfoTab::Bitmask,
+ 0L,
+ 0 // SQL_DTC_ENLIST_EXPENSIVE | SQL_DTC_UNENLIST_EXPENSIVE
+ },
+ { SQL_DYNAMIC_CURSOR_ATTRIBUTES1,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_DYNAMIC_CURSOR_ATTRIBUTES2,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_EXPRESSIONS_IN_ORDERBY,
+ InfoTab::Char,
+ 0L,
+ "Y"
+ },
+ { SQL_FILE_USAGE,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES1,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_FORWARD_ONLY_CURSOR_ATTRIBUTES2,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_GETDATA_EXTENSIONS,
+ InfoTab::Bitmask,
+ SQL_GD_ANY_COLUMN | SQL_GD_ANY_ORDER | SQL_GD_BOUND,
+ 0
+ },
+ { SQL_GROUP_BY,
+ InfoTab::Short,
+ SQL_GB_NOT_SUPPORTED,
+ 0
+ },
+ { SQL_IDENTIFIER_CASE,
+ InfoTab::Short,
+ SQL_IC_UPPER,
+ 0
+ },
+ { SQL_IDENTIFIER_QUOTE_CHAR,
+ InfoTab::Char,
+ 0L,
+ "\""
+ },
+ { SQL_INDEX_KEYWORDS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_INFO_SCHEMA_VIEWS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_INSERT_STATEMENT,
+ InfoTab::Bitmask,
+ SQL_IS_INSERT_LITERALS | SQL_IS_SELECT_INTO,
+ 0
+ },
+ { SQL_INTEGRITY,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_KEYSET_CURSOR_ATTRIBUTES1,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_KEYSET_CURSOR_ATTRIBUTES2,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_KEYWORDS,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_LIKE_ESCAPE_CLAUSE,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_MAX_ASYNC_CONCURRENT_STATEMENTS,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_MAX_BINARY_LITERAL_LEN,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_MAX_CATALOG_NAME_LEN,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_CHAR_LITERAL_LEN,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_MAX_COLUMN_NAME_LEN,
+ InfoTab::Short,
+ 16,
+ 0
+ },
+ { SQL_MAX_COLUMNS_IN_GROUP_BY,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_COLUMNS_IN_INDEX,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_COLUMNS_IN_ORDER_BY,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_COLUMNS_IN_SELECT,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_COLUMNS_IN_TABLE,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_CONCURRENT_ACTIVITIES,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_CURSOR_NAME_LEN,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_DRIVER_CONNECTIONS,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_IDENTIFIER_LEN,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_INDEX_SIZE,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_MAX_PROCEDURE_NAME_LEN,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_ROW_SIZE,
+ InfoTab::Long,
+ 8000,
+ 0
+ },
+ { SQL_MAX_ROW_SIZE_INCLUDES_LONG,
+ InfoTab::YesNo,
+ 0L,
+ "Y"
+ },
+ { SQL_MAX_SCHEMA_NAME_LEN,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_STATEMENT_LEN,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_MAX_TABLE_NAME_LEN,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_TABLES_IN_SELECT,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MAX_USER_NAME_LEN,
+ InfoTab::Short,
+ 0L,
+ 0
+ },
+ { SQL_MULTIPLE_ACTIVE_TXN,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_MULT_RESULT_SETS,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_NEED_LONG_DATA_LEN,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_NON_NULLABLE_COLUMNS,
+ InfoTab::Short,
+ SQL_NNC_NON_NULL,
+ 0
+ },
+ { SQL_NULL_COLLATION,
+ InfoTab::Short,
+ SQL_NC_HIGH,
+ 0
+ },
+ { SQL_NUMERIC_FUNCTIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_ODBC_INTERFACE_CONFORMANCE,
+ InfoTab::Long,
+ SQL_OIC_CORE,
+ 0
+ },
+ { SQL_ODBC_VER,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_OJ_CAPABILITIES,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_ORDER_BY_COLUMNS_IN_SELECT,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_PARAM_ARRAY_ROW_COUNTS,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_PARAM_ARRAY_SELECTS,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_POS_OPERATIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_PROCEDURES,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_PROCEDURE_TERM,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_QUOTED_IDENTIFIER_CASE,
+ InfoTab::Short,
+ SQL_IC_SENSITIVE,
+ 0
+ },
+ { SQL_ROW_UPDATES,
+ InfoTab::YesNo,
+ 0L,
+ "N"
+ },
+ { SQL_SCHEMA_TERM,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_SCHEMA_USAGE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SCROLL_OPTIONS,
+ InfoTab::Bitmask,
+ SQL_SO_FORWARD_ONLY,
+ 0
+ },
+ { SQL_SEARCH_PATTERN_ESCAPE,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_SERVER_NAME,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_SPECIAL_CHARACTERS,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_SQL92_DATETIME_FUNCTIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SQL92_FOREIGN_KEY_DELETE_RULE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SQL92_FOREIGN_KEY_UPDATE_RULE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SQL92_GRANT,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SQL92_NUMERIC_VALUE_FUNCTIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SQL92_PREDICATES,
+ InfoTab::Bitmask,
+ SQL_SP_COMPARISON | SQL_SP_IN | SQL_SP_ISNOTNULL | SQL_SP_ISNULL | SQL_SP_LIKE,
+ 0
+ },
+ { SQL_SQL92_RELATIONAL_JOIN_OPERATORS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SQL92_REVOKE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SQL92_ROW_VALUE_CONSTRUCTOR,
+ InfoTab::Bitmask,
+ SQL_SRVC_VALUE_EXPRESSION,
+ 0
+ },
+ { SQL_SQL92_STRING_FUNCTIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SQL92_VALUE_EXPRESSIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SQL_CONFORMANCE,
+ InfoTab::Long,
+ 0L,
+ 0
+ },
+ { SQL_STANDARD_CLI_CONFORMANCE,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_STATIC_CURSOR_ATTRIBUTES1,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_STATIC_CURSOR_ATTRIBUTES2,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_STRING_FUNCTIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SUBQUERIES,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_SYSTEM_FUNCTIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_TABLE_TERM,
+ InfoTab::Char,
+ 0L,
+ "TABLE"
+ },
+ { SQL_TIMEDATE_ADD_INTERVALS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_TIMEDATE_DIFF_INTERVALS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_TIMEDATE_FUNCTIONS,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_TXN_CAPABLE,
+ InfoTab::Short,
+ SQL_TC_DDL_COMMIT, // XXX do it
+ 0
+ },
+ { SQL_TXN_ISOLATION_OPTION,
+ InfoTab::Bitmask,
+ SQL_TXN_READ_COMMITTED,
+ 0
+ },
+ { SQL_UNION,
+ InfoTab::Bitmask,
+ 0L,
+ 0
+ },
+ { SQL_USER_NAME,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { SQL_XOPEN_CLI_YEAR,
+ InfoTab::Char,
+ 0L,
+ ""
+ },
+ { 0,
+ InfoTab::End,
+ 0L,
+ 0
+ }
+};
diff --git a/ndb/src/client/odbc/handles/Makefile b/ndb/src/client/odbc/handles/Makefile
new file mode 100644
index 00000000000..d37e7d286ba
--- /dev/null
+++ b/ndb/src/client/odbc/handles/Makefile
@@ -0,0 +1,28 @@
+include .defs.mk
+
+TYPE = *
+
+NONPIC_ARCHIVE = N
+
+PIC_ARCHIVE = Y
+
+ARCHIVE_TARGET = odbchandles
+
+SOURCES = \
+ HandleBase.cpp \
+ HandleRoot.cpp \
+ HandleEnv.cpp \
+ HandleDbc.cpp \
+ HandleStmt.cpp \
+ HandleDesc.cpp \
+ AttrRoot.cpp \
+ AttrEnv.cpp \
+ AttrDbc.cpp \
+ AttrStmt.cpp \
+ PoolNdb.cpp \
+ DescSpec.cpp \
+ FuncTab.cpp \
+ InfoTab.cpp
+
+include ../Extra.mk
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/client/odbc/handles/PoolNdb.cpp b/ndb/src/client/odbc/handles/PoolNdb.cpp
new file mode 100644
index 00000000000..c487ca2b976
--- /dev/null
+++ b/ndb/src/client/odbc/handles/PoolNdb.cpp
@@ -0,0 +1,80 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbMutex.h>
+#include <NdbApi.hpp>
+#include "PoolNdb.hpp"
+
+#ifdef NDB_WIN32
+static NdbMutex & ndb_mutex = * NdbMutex_Create();
+#else
+static NdbMutex ndb_mutex = NDB_MUTEX_INITIALIZER;
+#endif
+
+PoolNdb::PoolNdb() :
+ m_cntUsed(0),
+ m_cntFree(0)
+{
+}
+
+PoolNdb::~PoolNdb()
+{
+}
+
+Ndb*
+PoolNdb::allocate(Ctx& ctx, int timeout)
+{
+ NdbMutex_Lock(&ndb_mutex);
+ Ndb* pNdb;
+ if (m_cntFree == 0) {
+ pNdb = new Ndb("TEST_DB");
+ pNdb->useFullyQualifiedNames(true);
+ if (pNdb->init(64) < 0) {
+ ctx.pushStatus(pNdb, "init");
+ delete pNdb;
+ NdbMutex_Unlock(&ndb_mutex);
+ return 0;
+ }
+ if (pNdb->waitUntilReady(timeout) < 0) {
+ ctx.pushStatus(Sqlstate::_HYT00, Error::Gen, "connection timeout after %d seconds", timeout);
+ ctx.pushStatus(pNdb, "waitUntilReady");
+ delete pNdb;
+ NdbMutex_Unlock(&ndb_mutex);
+ return 0;
+ }
+ m_listFree.push_back(pNdb);
+ m_cntFree++;
+ }
+ pNdb = m_listFree.front();
+ m_listFree.pop_front();
+ m_cntFree--;
+ m_cntUsed++;
+ ctx_log1(("alloc Ndb: used=%u free=%u", m_cntUsed, m_cntFree));
+ NdbMutex_Unlock(&ndb_mutex);
+ return pNdb;
+}
+
+void
+PoolNdb::release(Ctx& ctx, Ndb* pNdb)
+{
+ NdbMutex_Lock(&ndb_mutex);
+ m_listUsed.remove(pNdb);
+ m_listFree.push_back(pNdb);
+ m_cntFree++;
+ m_cntUsed--;
+ ctx_log1(("free Ndb: used=%u free=%u", m_cntUsed, m_cntFree));
+ NdbMutex_Unlock(&ndb_mutex);
+}
diff --git a/ndb/src/client/odbc/handles/PoolNdb.hpp b/ndb/src/client/odbc/handles/PoolNdb.hpp
new file mode 100644
index 00000000000..bea7e72a59f
--- /dev/null
+++ b/ndb/src/client/odbc/handles/PoolNdb.hpp
@@ -0,0 +1,44 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ODBC_HANDLES_PoolNdb_hpp
+#define ODBC_HANDLES_PoolNdb_hpp
+
+#include <list>
+#include <common/common.hpp>
+
+class Ndb;
+
+/**
+ * @class PoolNdb
+ * @brief Pool of Ndb objects.
+ *
+ * A class implementing pool of Ndb objects.
+ */
+class PoolNdb {
+public:
+ PoolNdb();
+ ~PoolNdb();
+ Ndb* allocate(Ctx& ctx, int timeout);
+ void release(Ctx& ctx, Ndb* pNdb);
+private:
+ std::list<Ndb*> m_listUsed;
+ std::list<Ndb*> m_listFree;
+ unsigned m_cntUsed;
+ unsigned m_cntFree;
+};
+
+#endif
diff --git a/ndb/src/client/odbc/handles/handles.hpp b/ndb/src/client/odbc/handles/handles.hpp
new file mode 100644
index 00000000000..a9f0fcae888
--- /dev/null
+++ b/ndb/src/client/odbc/handles/handles.hpp
@@ -0,0 +1,28 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef OBDC_HANDLES_handles_hpp
+#define OBDC_HANDLES_handles_hpp
+
+#include <common/common.hpp>
+#include "HandleBase.hpp"
+#include "HandleRoot.hpp"
+#include "HandleEnv.hpp"
+#include "HandleDbc.hpp"
+#include "HandleStmt.hpp"
+#include "HandleDesc.hpp"
+
+#endif
diff --git a/ndb/src/common/Makefile b/ndb/src/common/Makefile
new file mode 100644
index 00000000000..ebde75bf3ec
--- /dev/null
+++ b/ndb/src/common/Makefile
@@ -0,0 +1,15 @@
+include .defs.mk
+
+LIB_DIRS := \
+ portlib \
+ debugger \
+ util \
+ logger
+
+ifneq ($(USE_EDITLINE), N)
+LIB_DIRS += editline
+endif
+
+DIRS := transporter mgmcommon
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/common/debugger/BlockNames.cpp b/ndb/src/common/debugger/BlockNames.cpp
new file mode 100644
index 00000000000..44650b84c5c
--- /dev/null
+++ b/ndb/src/common/debugger/BlockNames.cpp
@@ -0,0 +1,39 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <BlockNumbers.h>
+
+const BlockName BlockNames[] = {
+ { "CMVMI", CMVMI },
+ { "DBACC", DBACC },
+ { "DBDICT", DBDICT },
+ { "DBDIH", DBDIH },
+ { "DBLQH", DBLQH },
+ { "DBTC", DBTC },
+ { "DBTUP", DBTUP },
+ { "NDBFS", NDBFS },
+ { "NDBCNTR", NDBCNTR },
+ { "QMGR", QMGR },
+ { "TRIX", TRIX },
+ { "BACKUP", BACKUP },
+ { "DBUTIL", DBUTIL },
+ { "SUMA", SUMA },
+ { "GREP", GREP },
+ { "DBTUX", DBTUX }
+};
+
+const BlockNumber NO_OF_BLOCK_NAMES = sizeof(BlockNames) / sizeof(BlockName);
diff --git a/ndb/src/common/debugger/DebuggerNames.cpp b/ndb/src/common/debugger/DebuggerNames.cpp
new file mode 100644
index 00000000000..fdee978ab54
--- /dev/null
+++ b/ndb/src/common/debugger/DebuggerNames.cpp
@@ -0,0 +1,154 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "DebuggerNames.hpp"
+
+#include <BlockNumbers.h>
+#include <GlobalSignalNumbers.h>
+#include <signaldata/SignalDataPrint.hpp>
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+static const char * localSignalNames[MAX_GSN+1];
+static SignalDataPrintFunction localPrintFunctions[MAX_GSN+1];
+static const char * localBlockNames[NO_OF_BLOCKS];
+
+static
+int
+initSignalNames(const char * dst[], const GsnName src[], unsigned short len){
+ for(int i = 0; i<=MAX_GSN; i++)
+ dst[i] = 0;
+
+ for(int i = 0; i<len; i++){
+ unsigned short gsn = src[i].gsn;
+ const char * name = src[i].name;
+
+ if(dst[gsn] != 0 && name != 0){
+ if(strcmp(dst[gsn], name) != 0){
+ fprintf(stderr,
+ "Multiple definition of signal name for gsn: %d (%s, %s)\n",
+ gsn, dst[gsn], name);
+ exit(0);
+ }
+ }
+ dst[gsn] = name;
+ }
+ return 0;
+}
+
+static
+int
+initSignalPrinters(SignalDataPrintFunction dst[],
+ const NameFunctionPair src[],
+ unsigned short len){
+ for(int i = 0; i<=MAX_GSN; i++)
+ dst[i] = 0;
+
+ for(int i = 0; i<len; i++){
+ unsigned short gsn = src[i].gsn;
+ SignalDataPrintFunction fun = src[i].function;
+
+ if(dst[gsn] != 0 && fun != 0){
+ if(dst[gsn] != fun){
+ fprintf(stderr,
+ "Multiple definition of signal print function for gsn: %d\n",
+ gsn);
+ exit(0);
+ }
+ }
+ dst[gsn] = fun;
+ }
+ return 0;
+}
+
+static
+int
+initBlockNames(const char * dst[],
+ const BlockName src[],
+ unsigned len){
+ for(int i = 0; i<NO_OF_BLOCKS; i++)
+ dst[i] = 0;
+
+ for(unsigned i = 0; i<len; i++){
+ const int index = src[i].number - MIN_BLOCK_NO;
+ if(index < 0 && index >= NO_OF_BLOCKS || dst[index] != 0){
+ fprintf(stderr,
+ "Invalid block name definition: %d %s\n",
+ src[i].number, src[i].name);
+ exit(0);
+ }
+ dst[index] = src[i].name;
+ }
+ return 0;
+}
+
+/**
+ * Run static initializer
+ */
+static const int
+xxx_DUMMY_SIGNAL_NAMES_xxx = initSignalNames(localSignalNames,
+ SignalNames,
+ NO_OF_SIGNAL_NAMES);
+static const int
+xxx_DUMMY_PRINT_FUNCTIONS_xxx = initSignalPrinters(localPrintFunctions,
+ SignalDataPrintFunctions,
+ NO_OF_PRINT_FUNCTIONS);
+
+static const int
+xxx_DUMMY_BLOCK_NAMES_xxx = initBlockNames(localBlockNames,
+ BlockNames,
+ NO_OF_BLOCK_NAMES);
+
+const char *
+getSignalName(unsigned short gsn, const char * defVal){
+ if(gsn > 0 && gsn <= MAX_GSN)
+ return (localSignalNames[gsn] ? localSignalNames[gsn] : defVal);
+ return defVal;
+}
+
+unsigned short
+getGsn(const char * signalName){
+ return 0;
+}
+
+const char *
+getBlockName(unsigned short blockNo, const char * ret){
+ if(blockNo >= MIN_BLOCK_NO && blockNo <= MAX_BLOCK_NO)
+ return localBlockNames[blockNo-MIN_BLOCK_NO];
+ if (ret == 0) {
+ static char buf[20];
+ snprintf(buf, sizeof(buf), "BLOCK#%d", (int)blockNo);
+ return buf;
+ }
+ return ret;
+}
+
+unsigned short
+getBlockNo(const char * blockName){
+ for(int i = 0; i<NO_OF_BLOCKS; i++)
+ if(localBlockNames[i] != 0 && strcmp(localBlockNames[i], blockName) == 0)
+ return i + MIN_BLOCK_NO;
+ return 0;
+}
+
+SignalDataPrintFunction
+findPrintFunction(unsigned short gsn){
+ if(gsn > 0 && gsn <= MAX_GSN)
+ return localPrintFunctions[gsn];
+ return 0;
+}
diff --git a/ndb/src/common/debugger/EventLogger.cpp b/ndb/src/common/debugger/EventLogger.cpp
new file mode 100644
index 00000000000..12f01890c54
--- /dev/null
+++ b/ndb/src/common/debugger/EventLogger.cpp
@@ -0,0 +1,1454 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "EventLogger.hpp"
+
+#include <NdbConfig.h>
+#include <kernel/BlockNumbers.h>
+#include <signaldata/ArbitSignalData.hpp>
+#include <GrepEvent.hpp>
+#include <NodeState.hpp>
+#include <version.h>
+#include <NdbStdio.h>
+
+#include <string.h>
+#include <ctype.h>
+
+//
+// PUBLIC
+//
+
+/**
+ * This matrix defines which event should be printed when
+ *
+ * threshold - is in range [0-15]
+ * severity - DEBUG to ALERT (Type of log message)
+ */
+const EventLogger::EventRepLogLevelMatrix EventLogger::matrix[] = {
+ // CONNECTION
+ { EventReport::Connected, LogLevel::llConnection, 8, LL_INFO },
+ { EventReport::Disconnected, LogLevel::llConnection, 8, LL_ALERT },
+ { EventReport::CommunicationClosed, LogLevel::llConnection, 8, LL_INFO },
+ { EventReport::CommunicationOpened, LogLevel::llConnection, 8, LL_INFO },
+ { EventReport::ConnectedApiVersion, LogLevel::llConnection, 8, LL_INFO },
+ // CHECKPOINT
+ { EventReport::GlobalCheckpointStarted, LogLevel::llCheckpoint, 9, LL_INFO },
+ { EventReport::GlobalCheckpointCompleted,LogLevel::llCheckpoint,10, LL_INFO },
+ { EventReport::LocalCheckpointStarted, LogLevel::llCheckpoint, 7, LL_INFO },
+ { EventReport::LocalCheckpointCompleted,LogLevel::llCheckpoint, 8, LL_INFO },
+ { EventReport::LCPStoppedInCalcKeepGci, LogLevel::llCheckpoint, 0, LL_ALERT },
+ { EventReport::LCPFragmentCompleted, LogLevel::llCheckpoint, 11, LL_INFO },
+ { EventReport::UndoLogBlocked, LogLevel::llCheckpoint, 7, LL_INFO },
+
+ // STARTUP
+ { EventReport::NDBStartStarted, LogLevel::llStartUp, 1, LL_INFO },
+ { EventReport::NDBStartCompleted, LogLevel::llStartUp, 1, LL_INFO },
+ { EventReport::STTORRYRecieved, LogLevel::llStartUp,15, LL_INFO },
+ { EventReport::StartPhaseCompleted, LogLevel::llStartUp, 4, LL_INFO },
+ { EventReport::CM_REGCONF, LogLevel::llStartUp, 3, LL_INFO },
+ { EventReport::CM_REGREF, LogLevel::llStartUp, 8, LL_INFO },
+ { EventReport::FIND_NEIGHBOURS, LogLevel::llStartUp, 8, LL_INFO },
+ { EventReport::NDBStopStarted, LogLevel::llStartUp, 1, LL_INFO },
+ { EventReport::NDBStopAborted, LogLevel::llStartUp, 1, LL_INFO },
+ { EventReport::StartREDOLog, LogLevel::llStartUp, 10, LL_INFO },
+ { EventReport::StartLog, LogLevel::llStartUp, 10, LL_INFO },
+ { EventReport::UNDORecordsExecuted, LogLevel::llStartUp, 15, LL_INFO },
+
+ // NODERESTART
+ { EventReport::NR_CopyDict, LogLevel::llNodeRestart, 8, LL_INFO },
+ { EventReport::NR_CopyDistr, LogLevel::llNodeRestart, 8, LL_INFO },
+ { EventReport::NR_CopyFragsStarted, LogLevel::llNodeRestart, 8, LL_INFO },
+ { EventReport::NR_CopyFragDone, LogLevel::llNodeRestart, 10, LL_INFO },
+ { EventReport::NR_CopyFragsCompleted, LogLevel::llNodeRestart, 8, LL_INFO },
+
+ { EventReport::NodeFailCompleted, LogLevel::llNodeRestart, 8, LL_ALERT},
+ { EventReport::NODE_FAILREP, LogLevel::llNodeRestart, 8, LL_ALERT},
+ { EventReport::ArbitState, LogLevel::llNodeRestart, 6, LL_INFO },
+ { EventReport::ArbitResult, LogLevel::llNodeRestart, 2, LL_ALERT},
+ { EventReport::GCP_TakeoverStarted, LogLevel::llNodeRestart, 7, LL_INFO },
+ { EventReport::GCP_TakeoverCompleted, LogLevel::llNodeRestart, 7, LL_INFO },
+ { EventReport::LCP_TakeoverStarted, LogLevel::llNodeRestart, 7, LL_INFO },
+ { EventReport::LCP_TakeoverCompleted, LogLevel::llNodeRestart, 7, LL_INFO },
+
+ // STATISTIC
+ { EventReport::TransReportCounters, LogLevel::llStatistic, 8, LL_INFO },
+ { EventReport::OperationReportCounters, LogLevel::llStatistic, 8, LL_INFO },
+ { EventReport::TableCreated, LogLevel::llStatistic, 7, LL_INFO },
+ { EventReport::JobStatistic, LogLevel::llStatistic, 9, LL_INFO },
+ { EventReport::SendBytesStatistic, LogLevel::llStatistic, 9, LL_INFO },
+ { EventReport::ReceiveBytesStatistic, LogLevel::llStatistic, 9, LL_INFO },
+ { EventReport::MemoryUsage, LogLevel::llStatistic, 5, LL_INFO },
+
+ // ERROR
+ { EventReport::TransporterError, LogLevel::llError, 2, LL_ERROR },
+ { EventReport::TransporterWarning, LogLevel::llError, 8, LL_WARNING },
+ { EventReport::MissedHeartbeat, LogLevel::llError, 8, LL_WARNING },
+ { EventReport::DeadDueToHeartbeat, LogLevel::llError, 8, LL_ALERT },
+ { EventReport::WarningEvent, LogLevel::llError, 2, LL_WARNING },
+ // INFO
+ { EventReport::SentHeartbeat, LogLevel::llInfo, 12, LL_INFO },
+ { EventReport::CreateLogBytes, LogLevel::llInfo, 11, LL_INFO },
+ { EventReport::InfoEvent, LogLevel::llInfo, 2, LL_INFO },
+
+ //Global replication
+ { EventReport::GrepSubscriptionInfo, LogLevel::llGrep, 7, LL_INFO},
+ { EventReport::GrepSubscriptionAlert, LogLevel::llGrep, 7, LL_ALERT}
+};
+
+const Uint32 EventLogger::matrixSize = sizeof(EventLogger::matrix)/
+ sizeof(EventRepLogLevelMatrix);
+
+/**
+ * Default log levels for management nodes.
+ *
+ * threshold - is in range [0-15]
+ */
+const EventLogger::EventLogMatrix EventLogger::defEventLogMatrix[] = {
+ { LogLevel::llStartUp, 7 },
+ { LogLevel::llShutdown, 7 },
+ { LogLevel::llStatistic, 7 },
+ { LogLevel::llCheckpoint, 7 },
+ { LogLevel::llNodeRestart, 7 },
+ { LogLevel::llConnection, 7 },
+ { LogLevel::llError, 15 },
+ { LogLevel::llInfo, 7 },
+ { LogLevel::llGrep, 7 }
+};
+
+const Uint32
+EventLogger::defEventLogMatrixSize = sizeof(EventLogger::defEventLogMatrix)/
+ sizeof(EventLogMatrix);
+/**
+ * Specifies allowed event categories/log levels that can be set from
+ * the Management API/interactive shell.
+ */
+const EventLogger::EventCategoryName EventLogger::eventCategoryNames[] = {
+ { LogLevel::llStartUp, "STARTUP" },
+ { LogLevel::llStatistic, "STATISTIC" },
+ { LogLevel::llCheckpoint, "CHECKPOINT" },
+ { LogLevel::llNodeRestart, "NODERESTART" },
+ { LogLevel::llConnection, "CONNECTION" },
+ { LogLevel::llInfo, "INFO" },
+ { LogLevel::llGrep, "GREP" }
+};
+
+const Uint32
+EventLogger::noOfEventCategoryNames = sizeof(EventLogger::eventCategoryNames)/
+ sizeof(EventLogger::EventCategoryName);
+
+char EventLogger::m_text[MAX_TEXT_LENGTH];
+
+const char*
+EventLogger::getText(int type,
+ const Uint32* theData, NodeId nodeId)
+{
+ // TODO: Change the switch implementation...
+ char theNodeId[32];
+ if (nodeId != 0){
+ ::snprintf(theNodeId, 32, "Node %u: ", nodeId);
+ } else {
+ theNodeId[0] = 0;
+ }
+
+ EventReport::EventType eventType = (EventReport::EventType)type;
+ switch (eventType){
+ case EventReport::Connected:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sNode %u Connected",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::ConnectedApiVersion:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sNode %u: API version %d.%d.%d",
+ theNodeId,
+ theData[1],
+ getMajor(theData[2]),
+ getMinor(theData[2]),
+ getBuild(theData[2]));
+ break;
+ case EventReport::Disconnected:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sNode %u Disconnected",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::CommunicationClosed:
+ //-----------------------------------------------------------------------
+ // REPORT communication to node closed.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text, sizeof(m_text),
+ "%sCommunication to Node %u closed",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::CommunicationOpened:
+ //-----------------------------------------------------------------------
+ // REPORT communication to node opened.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text, sizeof(m_text),
+ "%sCommunication to Node %u opened",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::NDBStartStarted:
+ //-----------------------------------------------------------------------
+ // Start of NDB has been initiated.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text, sizeof(m_text),
+ "%sStart initiated (version %d.%d.%d)",
+ theNodeId ,
+ getMajor(theData[1]),
+ getMinor(theData[1]),
+ getBuild(theData[1]));
+ break;
+ case EventReport::NDBStopStarted:
+ ::snprintf(m_text, sizeof(m_text),
+ "%s%s shutdown initiated",
+ theNodeId,
+ (theData[1] == 1 ? "Cluster" : "Node"));
+ break;
+ case EventReport::NDBStopAborted:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sNode shutdown aborted",
+ theNodeId);
+ break;
+ case EventReport::NDBStartCompleted:
+ //-----------------------------------------------------------------------
+ // Start of NDB has been completed.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text, sizeof(m_text),
+ "%sStarted (version %d.%d.%d)",
+ theNodeId ,
+ getMajor(theData[1]),
+ getMinor(theData[1]),
+ getBuild(theData[1]));
+
+ break;
+ case EventReport::STTORRYRecieved:
+ //-----------------------------------------------------------------------
+ // STTORRY recevied after restart finished.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text, sizeof(m_text),
+ "%sSTTORRY received after restart finished",
+ theNodeId);
+ break;
+ case EventReport::StartPhaseCompleted:{
+ //-----------------------------------------------------------------------
+ // REPORT Start phase completed.
+ //-----------------------------------------------------------------------
+ const char * type = "<Unknown>";
+ switch((NodeState::StartType)theData[2]){
+ case NodeState::ST_INITIAL_START:
+ type = "(initial start)";
+ break;
+ case NodeState::ST_SYSTEM_RESTART:
+ type = "(system restart)";
+ break;
+ case NodeState::ST_NODE_RESTART:
+ type = "(node restart)";
+ break;
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ type = "(initial node restart)";
+ break;
+ case NodeState::ST_ILLEGAL_TYPE:
+ type = "";
+ break;
+ default:{
+ ::snprintf(m_text, sizeof(m_text),
+ "%sStart phase %u completed (unknown = %d)",
+ theNodeId,
+ theData[1],
+ theData[2]);
+ return m_text;
+ }
+ }
+ ::snprintf(m_text, sizeof(m_text),
+ "%sStart phase %u completed %s",
+ theNodeId,
+ theData[1],
+ type);
+ return m_text;
+ break;
+ }
+ case EventReport::CM_REGCONF:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sCM_REGCONF president = %u, own Node = %u, our dynamic id = %u"
+ ,
+ theNodeId,
+ theData[2],
+ theData[1],
+ theData[3]);
+ break;
+ case EventReport::CM_REGREF:
+ {
+ const char* line = "";
+ switch (theData[3]) {
+ case 0:
+ line = "Busy";
+ break;
+ case 1:
+ line = "Election with wait = false";
+ break;
+ case 2:
+ line = "Election with wait = false";
+ break;
+ case 3:
+ line = "Not president";
+ break;
+ case 4:
+ line = "Election without selecting new candidate";
+ break;
+ default:
+ line = "No such cause";
+ break;
+ }//switch
+
+ ::snprintf(m_text, sizeof(m_text),
+ "%sCM_REGREF from Node %u to our Node %u. Cause = %s",
+ theNodeId,
+ theData[2],
+ theData[1],
+ line);
+ }
+ break;
+ case EventReport::FIND_NEIGHBOURS:
+ //-----------------------------------------------------------------------
+ // REPORT Node Restart copied a fragment.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sWe are Node %u with dynamic ID %u, our left neighbour "
+ "is Node %u, our right is Node %u",
+ theNodeId,
+ theData[1],
+ theData[4],
+ theData[2],
+ theData[3]);
+ break;
+ case EventReport::NodeFailCompleted:
+ //-----------------------------------------------------------------------
+ // REPORT Node failure phase completed.
+ //-----------------------------------------------------------------------
+ if (theData[1] == 0)
+ {
+ if (theData[3] != 0) {
+ ::snprintf(m_text, sizeof(m_text),
+ "%sNode %u completed failure of Node %u",
+ theNodeId,
+ theData[3],
+ theData[2]);
+ } else {
+ ::snprintf(m_text, sizeof(m_text),
+ "%sAll nodes completed failure of Node %u",
+ theNodeId,
+ theData[2]);
+ }//if
+ } else {
+ const char* line = "";
+ if (theData[1] == DBTC){
+ line = "DBTC";
+ }else if (theData[1] == DBDICT){
+ line = "DBDICT";
+ }else if (theData[1] == DBDIH){
+ line = "DBDIH";
+ }else if (theData[1] == DBLQH){
+ line = "DBLQH";
+ }
+
+ ::snprintf(m_text, sizeof(m_text),
+ "%sNode failure of %u %s completed",
+ theNodeId,
+ theData[2],
+ line);
+ }
+ break;
+ case EventReport::NODE_FAILREP:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sNode %u has failed. The Node state at failure "
+ "was %u",
+ theNodeId,
+ theData[1],
+ theData[2]);
+
+ break;
+ case EventReport::ArbitState:
+ //-----------------------------------------------------------------------
+ // REPORT arbitrator found or lost.
+ //-----------------------------------------------------------------------
+ { const ArbitSignalData* sd = (ArbitSignalData*)theData;
+ char ticketText[ArbitTicket::TextLength + 1];
+ char errText[ArbitCode::ErrTextLength + 1];
+ const unsigned code = sd->code & 0xFFFF;
+ const unsigned state = sd->code >> 16;
+ switch (code) {
+ case ArbitCode::ThreadStart:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sPresident restarts arbitration thread [state=%u]",
+ theNodeId, state);
+ break;
+ case ArbitCode::PrepPart2:
+ sd->ticket.getText(ticketText, sizeof(ticketText));
+ ::snprintf(m_text, sizeof(m_text),
+ "%sPrepare arbitrator node %u [ticket=%s]",
+ theNodeId, sd->node, ticketText);
+ break;
+ case ArbitCode::PrepAtrun:
+ sd->ticket.getText(ticketText, sizeof(ticketText));
+ ::snprintf(m_text, sizeof(m_text),
+ "%sReceive arbitrator node %u [ticket=%s]",
+ theNodeId, sd->node, ticketText);
+ break;
+ case ArbitCode::ApiStart:
+ sd->ticket.getText(ticketText, sizeof(ticketText));
+ ::snprintf(m_text, sizeof(m_text),
+ "%sStarted arbitrator node %u [ticket=%s]",
+ theNodeId, sd->node, ticketText);
+ break;
+ case ArbitCode::ApiFail:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sLost arbitrator node %u - process failure [state=%u]",
+ theNodeId, sd->node, state);
+ break;
+ case ArbitCode::ApiExit:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sLost arbitrator node %u - process exit [state=%u]",
+ theNodeId, sd->node, state);
+ break;
+ default:
+ ArbitCode::getErrText(code, errText, sizeof(errText));
+ ::snprintf(m_text, sizeof(m_text),
+ "%sLost arbitrator node %u - %s [state=%u]",
+ theNodeId, sd->node, errText, state);
+ break;
+ }
+ }
+ break;
+ case EventReport::ArbitResult:
+ //-----------------------------------------------------------------------
+ // REPORT arbitration result (the failures may not reach us).
+ //-----------------------------------------------------------------------
+ { const ArbitSignalData* sd = (ArbitSignalData*)theData;
+ char errText[ArbitCode::ErrTextLength + 1];
+ const unsigned code = sd->code & 0xFFFF;
+ const unsigned state = sd->code >> 16;
+ switch (code) {
+ case ArbitCode::LoseNodes:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sArbitration check lost - less than 1/2 nodes left",
+ theNodeId);
+ break;
+ case ArbitCode::WinGroups:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sArbitration check won - node group majority",
+ theNodeId);
+ break;
+ case ArbitCode::LoseGroups:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sArbitration check lost - missing node group",
+ theNodeId);
+ break;
+ case ArbitCode::Partitioning:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sNetwork partitioning - arbitration required",
+ theNodeId);
+ break;
+ case ArbitCode::WinChoose:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sArbitration won - positive reply from node %u",
+ theNodeId, sd->node);
+ break;
+ case ArbitCode::LoseChoose:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sArbitration lost - negative reply from node %u",
+ theNodeId, sd->node);
+ break;
+ case ArbitCode::LoseNorun:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sNetwork partitioning - no arbitrator available",
+ theNodeId);
+ break;
+ case ArbitCode::LoseNocfg:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sNetwork partitioning - no arbitrator configured",
+ theNodeId);
+ break;
+ default:
+ ArbitCode::getErrText(code, errText, sizeof(errText));
+ ::snprintf(m_text, sizeof(m_text),
+ "%sArbitration failure - %s [state=%u]",
+ theNodeId, errText, state);
+ break;
+ }
+ }
+ break;
+ case EventReport::GlobalCheckpointStarted:
+ //-----------------------------------------------------------------------
+ // This event reports that a global checkpoint has been started and this
+ // node is the master of this global checkpoint.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sGlobal checkpoint %u started",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::GlobalCheckpointCompleted:
+ //-----------------------------------------------------------------------
+ // This event reports that a global checkpoint has been completed on this
+ // node and the node is the master of this global checkpoint.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text, sizeof(m_text),
+ "%sGlobal checkpoint %u completed",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::LocalCheckpointStarted:
+ //-----------------------------------------------------------------------
+ // This event reports that a local checkpoint has been started and this
+ // node is the master of this local checkpoint.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sLocal checkpoint %u started. "
+ "Keep GCI = %u oldest restorable GCI = %u",
+ theNodeId,
+ theData[1],
+ theData[2],
+ theData[3]);
+ break;
+ case EventReport::LocalCheckpointCompleted:
+ //-----------------------------------------------------------------------
+ // This event reports that a local checkpoint has been completed on this
+ // node and the node is the master of this local checkpoint.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sLocal checkpoint %u completed",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::TableCreated:
+ //-----------------------------------------------------------------------
+ // This event reports that a table has been created.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text, sizeof(m_text),
+ "%sTable with ID = %u created",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::LCPStoppedInCalcKeepGci:
+ if (theData[1] == 0)
+ ::snprintf(m_text, sizeof(m_text),
+ "%sLocal Checkpoint stopped in CALCULATED_KEEP_GCI",
+ theNodeId);
+ break;
+ case EventReport::NR_CopyDict:
+ //-----------------------------------------------------------------------
+ // REPORT Node Restart completed copy of dictionary information.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sNode restart completed copy of dictionary information",
+ theNodeId);
+ break;
+ case EventReport::NR_CopyDistr:
+ //-----------------------------------------------------------------------
+ // REPORT Node Restart completed copy of distribution information.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sNode restart completed copy of distribution information",
+ theNodeId);
+ break;
+ case EventReport::NR_CopyFragsStarted:
+ //-----------------------------------------------------------------------
+ // REPORT Node Restart is starting to copy the fragments.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sNode restart starting to copy the fragments "
+ "to Node %u",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::NR_CopyFragDone:
+ //-----------------------------------------------------------------------
+ // REPORT Node Restart copied a fragment.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sTable ID = %u, fragment ID = %u have been copied "
+ "to Node %u",
+ theNodeId,
+ theData[2],
+ theData[3],
+ theData[1]);
+ break;
+ case EventReport::NR_CopyFragsCompleted:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sNode restart completed copying the fragments "
+ "to Node %u",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::LCPFragmentCompleted:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sTable ID = %u, fragment ID = %u has completed LCP "
+ "on Node %u",
+ theNodeId,
+ theData[2],
+ theData[3],
+ theData[1]);
+ break;
+ case EventReport::TransReportCounters:
+ // -------------------------------------------------------------------
+ // Report information about transaction activity once per 10 seconds.
+ // -------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sTrans. Count = %u, Commit Count = %u, "
+ "Read Count = %u, Simple Read Count = %u,\n"
+ "Write Count = %u, AttrInfo Count = %u, "
+ "Concurrent Operations = %u, Abort Count = %u",
+ theNodeId,
+ theData[1],
+ theData[2],
+ theData[3],
+ theData[4],
+ theData[5],
+ theData[6],
+ theData[7],
+ theData[8]);
+ break;
+ case EventReport::OperationReportCounters:
+ ::snprintf(m_text, sizeof(m_text),
+ "%sOperations=%u",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::UndoLogBlocked:
+ //-----------------------------------------------------------------------
+ // REPORT Undo Logging blocked due to buffer near to overflow.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sACC Blocked %u and TUP Blocked %u times last second",
+ theNodeId,
+ theData[1],
+ theData[2]);
+ break;
+ case EventReport::TransporterError:
+ case EventReport::TransporterWarning:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sTransporter to node %d reported error 0x%x",
+ theNodeId,
+ theData[1],
+ theData[2]);
+ break;
+ case EventReport::MissedHeartbeat:
+ //-----------------------------------------------------------------------
+ // REPORT Undo Logging blocked due to buffer near to overflow.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sNode %d missed heartbeat %d",
+ theNodeId,
+ theData[1],
+ theData[2]);
+ break;
+ case EventReport::DeadDueToHeartbeat:
+ //-----------------------------------------------------------------------
+ // REPORT Undo Logging blocked due to buffer near to overflow.
+ //-----------------------------------------------------------------------
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sNode %d declared dead due to missed heartbeat",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::JobStatistic:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sMean loop Counter in doJob last 8192 times = %u",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::SendBytesStatistic:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sMean send size to Node = %d last 4096 sends = %u bytes",
+ theNodeId,
+ theData[1],
+ theData[2]);
+ break;
+ case EventReport::ReceiveBytesStatistic:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sMean receive size to Node = %d last 4096 sends = %u bytes",
+ theNodeId,
+ theData[1],
+ theData[2]);
+ break;
+ case EventReport::SentHeartbeat:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sNode Sent Heartbeat to node = %d",
+ theNodeId,
+ theData[1]);
+ break;
+ case EventReport::CreateLogBytes:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sLog part %u, log file %u, MB %u",
+ theNodeId,
+ theData[1],
+ theData[2],
+ theData[3]);
+ break;
+ case EventReport::StartLog:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sLog part %u, start MB %u, stop MB %u, last GCI, log exec %u",
+ theNodeId,
+ theData[1],
+ theData[2],
+ theData[3],
+ theData[4]);
+ break;
+ case EventReport::StartREDOLog:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sNode: %d StartLog: [GCI Keep: %d LastCompleted: %d NewestRestorable: %d]",
+ theNodeId,
+ theData[1],
+ theData[2],
+ theData[3],
+ theData[4]);
+ break;
+ case EventReport::UNDORecordsExecuted:{
+ const char* line = "";
+ if (theData[1] == DBTUP){
+ line = "DBTUP";
+ }else if (theData[1] == DBACC){
+ line = "DBACC";
+ }
+
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%s UNDO %s %d [%d %d %d %d %d %d %d %d %d]",
+ theNodeId,
+ line,
+ theData[2],
+ theData[3],
+ theData[4],
+ theData[5],
+ theData[6],
+ theData[7],
+ theData[8],
+ theData[9],
+ theData[10],
+ theData[11]);
+ }
+ break;
+ case EventReport::InfoEvent:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%s%s",
+ theNodeId,
+ (char *)&theData[1]);
+ break;
+ case EventReport::WarningEvent:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%s%s",
+ theNodeId,
+ (char *)&theData[1]);
+ break;
+ case EventReport::GCP_TakeoverStarted:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sGCP Take over started", theNodeId);
+ break;
+ case EventReport::GCP_TakeoverCompleted:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sGCP Take over completed", theNodeId);
+ break;
+ case EventReport::LCP_TakeoverStarted:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sLCP Take over started", theNodeId);
+ break;
+ case EventReport::LCP_TakeoverCompleted:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sLCP Take over completed (state = %d)",
+ theNodeId, theData[1]);
+ break;
+ case EventReport::MemoryUsage:{
+ const int gth = theData[1];
+ const int size = theData[2];
+ const int used = theData[3];
+ const int total = theData[4];
+ const int block = theData[5];
+ const int percent = (used*100)/total;
+
+ ::snprintf(m_text, sizeof(m_text),
+ "%s%s usage %s %d%s(%d %dK pages of total %d)",
+ theNodeId,
+ (block==DBACC ? "Index" : (block == DBTUP ?"Data":"<unknown>")),
+ (gth == 0 ? "is" : (gth > 0 ? "increased to" : "decreased to")),
+ percent, "%",
+ used, size/1024, total
+ );
+ break;
+ }
+
+
+ case EventReport::GrepSubscriptionInfo :
+ {
+ GrepEvent::Subscription event = (GrepEvent::Subscription)theData[1];
+ switch(event) {
+ case GrepEvent::GrepSS_CreateSubIdConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Created subscription id"
+ " (subId=%d,SubKey=%d)"
+ " Return code: %d.",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepPS_CreateSubIdConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: Created subscription id"
+ " (subId=%d,SubKey=%d)"
+ " Return code: %d.",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepSS_SubCreateConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ const int nodegrp = theData[5];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Created subscription using"
+ " (subId=%d,SubKey=%d)"
+ " in primary system. Primary system has %d nodegroup(s)."
+ " Return code: %d",
+ subId,
+ subKey,
+ nodegrp,
+ err);
+ break;
+ }
+ case GrepEvent::GrepPS_SubCreateConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: All participants have created "
+ "subscriptions"
+ " using (subId=%d,SubKey=%d)."
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepSS_SubStartMetaConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Logging started on meta data changes."
+ " using (subId=%d,SubKey=%d)"
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepPS_SubStartMetaConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: All participants have started "
+ "logging meta data"
+ " changes on the subscription subId=%d,SubKey=%d) "
+ "(N.I yet)."
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepSS_SubStartDataConf: {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Logging started on table data changes "
+ " using (subId=%d,SubKey=%d)"
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepPS_SubStartDataConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: All participants have started logging "
+ "table data changes on the subscription "
+ "subId=%d,SubKey=%d)."
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepPS_SubSyncMetaConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: All participants have started "
+ " synchronization on meta data (META SCAN) using "
+ "(subId=%d,SubKey=%d)."
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepSS_SubSyncMetaConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Synchronization started (META SCAN) on "
+ " meta data using (subId=%d,SubKey=%d)"
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepPS_SubSyncDataConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: All participants have started "
+ "synchronization "
+ " on table data (DATA SCAN) using (subId=%d,SubKey=%d)."
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepSS_SubSyncDataConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ const int gci = theData[5];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Synchronization started (DATA SCAN) on "
+ "table data using (subId=%d,SubKey=%d). GCI = %d"
+ " Return code: %d",
+ subId,
+ subKey,
+ gci,
+ err);
+ break;
+ }
+ case GrepEvent::GrepPS_SubRemoveConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: All participants have removed "
+ "subscription (subId=%d,SubKey=%d). I have cleaned "
+ "up resources I've used."
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ case GrepEvent::GrepSS_SubRemoveConf:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Removed subscription "
+ "(subId=%d,SubKey=%d)"
+ " Return code: %d",
+ subId,
+ subKey,
+ err);
+ break;
+ }
+ default:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sUnknown GrepSubscriptonInfo event: %d",
+ theNodeId,
+ theData[1]);
+ }
+ break;
+ }
+
+ case EventReport::GrepSubscriptionAlert :
+ {
+ GrepEvent::Subscription event = (GrepEvent::Subscription)theData[1];
+ switch(event)
+ {
+ case GrepEvent::GrepSS_CreateSubIdRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord:Error code: %d Error message: %s"
+ " (subId=%d,SubKey=%d)",
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err),
+ subId,
+ subKey);
+ break;
+ }
+ case GrepEvent::GrepSS_SubCreateRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: FAILED to Created subscription using"
+ " (subId=%d,SubKey=%d)in primary system."
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepSS_SubStartMetaRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Logging failed to start on meta "
+ "data changes."
+ " using (subId=%d,SubKey=%d)"
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepSS_SubStartDataRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Logging FAILED to start on table data "
+ " changes using (subId=%d,SubKey=%d)"
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepSS_SubSyncMetaRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Synchronization FAILED (META SCAN) on "
+ " meta data using (subId=%d,SubKey=%d)"
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepSS_SubSyncDataRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ const int gci = theData[5];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Synchronization FAILED (DATA SCAN) on "
+ "table data using (subId=%d,SubKey=%d). GCI = %d"
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ gci,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepSS_SubRemoveRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::SSCoord: Failed to remove subscription "
+ "(subId=%d,SubKey=%d). "
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err)
+ );
+ break;
+ }
+
+ case GrepEvent::GrepPS_CreateSubIdRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: Error code: %d Error Message: %s"
+ " (subId=%d,SubKey=%d)",
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err),
+ subId,
+ subKey);
+ break;
+ }
+ case GrepEvent::GrepPS_SubCreateRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: FAILED to Created subscription using"
+ " (subId=%d,SubKey=%d)in primary system."
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepPS_SubStartMetaRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: Logging failed to start on meta "
+ "data changes."
+ " using (subId=%d,SubKey=%d)"
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepPS_SubStartDataRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: Logging FAILED to start on table data "
+ " changes using (subId=%d,SubKey=%d)"
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepPS_SubSyncMetaRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: Synchronization FAILED (META SCAN) on "
+ " meta data using (subId=%d,SubKey=%d)"
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepPS_SubSyncDataRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ const int gci = theData[5];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: Synchronization FAILED (DATA SCAN) on "
+ "table data using (subId=%d,SubKey=%d). GCI = %d. "
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ gci,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::GrepPS_SubRemoveRef:
+ {
+ const int subId = theData[2];
+ const int subKey = theData[3];
+ const int err = theData[4];
+ ::snprintf(m_text, sizeof(m_text),
+ "Grep::PSCoord: Failed to remove subscription "
+ "(subId=%d,SubKey=%d)."
+ " Error code: %d Error Message: %s",
+ subId,
+ subKey,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+ case GrepEvent::Rep_Disconnect:
+ {
+ const int err = theData[4];
+ const int nodeId = theData[5];
+ ::snprintf(m_text, sizeof(m_text),
+ "Rep: Node %d."
+ " Error code: %d Error Message: %s",
+ nodeId,
+ err,
+ GrepError::getErrorDesc((GrepError::Code)err));
+ break;
+ }
+
+
+ default:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sUnknown GrepSubscriptionAlert event: %d",
+ theNodeId,
+ theData[1]);
+ break;
+ }
+ break;
+ }
+
+ default:
+ ::snprintf(m_text,
+ sizeof(m_text),
+ "%sUnknown event: %d",
+ theNodeId,
+ theData[0]);
+
+ }
+ return m_text;
+}
+
+bool
+EventLogger::matchEventCategory(const char * str,
+ LogLevel::EventCategory * cat,
+ bool exactMatch){
+ if(cat == 0 || str == 0)
+ return false;
+
+ char * tmp = strdup(str);
+ for(size_t i = 0; i<strlen(tmp); i++)
+ tmp[i] = toupper(tmp[i]);
+
+ for(Uint32 i = 0; i<noOfEventCategoryNames; i++){
+ if(strcmp(tmp, eventCategoryNames[i].name) == 0){
+ * cat = eventCategoryNames[i].category;
+ free(tmp);
+ return true;
+ }
+ }
+ free(tmp);
+ return false;
+}
+
+const char *
+EventLogger::getEventCategoryName(LogLevel::EventCategory cat){
+
+ for(unsigned i = 0; i<noOfEventCategoryNames; i++){
+ if(cat == eventCategoryNames[i].category){
+ return eventCategoryNames[i].name;
+ }
+ }
+ return 0;
+}
+
+
+EventLogger::EventLogger() : Logger(), m_logLevel(), m_filterLevel(15)
+{
+ setCategory("EventLogger");
+ m_logLevel.setLogLevel(LogLevel::llStartUp, m_filterLevel);
+ m_logLevel.setLogLevel(LogLevel::llShutdown, m_filterLevel);
+ // m_logLevel.setLogLevel(LogLevel::llStatistic, m_filterLevel);
+ // m_logLevel.setLogLevel(LogLevel::llCheckpoint, m_filterLevel);
+ m_logLevel.setLogLevel(LogLevel::llNodeRestart, m_filterLevel);
+ m_logLevel.setLogLevel(LogLevel::llConnection, m_filterLevel);
+ m_logLevel.setLogLevel(LogLevel::llError, m_filterLevel);
+ m_logLevel.setLogLevel(LogLevel::llInfo, m_filterLevel);
+ enable(Logger::LL_INFO, Logger::LL_ALERT); // Log INFO to ALERT
+
+}
+
+EventLogger::~EventLogger()
+{
+
+}
+
+bool
+EventLogger::open()
+{
+ char clusterLog[128];
+ NdbConfig_ClusterLogFileName(clusterLog, 128);
+ return open(clusterLog);
+}
+
+bool
+EventLogger::open(const char* logFileName, int maxNoFiles, long maxFileSize,
+ unsigned int maxLogEntries)
+{
+ return addHandler(new FileLogHandler(logFileName, maxNoFiles, maxFileSize,
+ maxLogEntries));
+}
+
+void
+EventLogger::close()
+{
+ removeAllHandlers();
+}
+
+void
+EventLogger::log(NodeId nodeId, int eventType, const Uint32* theData)
+{
+ log(eventType, theData, nodeId);
+}
+
+void
+EventLogger::log(int eventType, const Uint32* theData, NodeId nodeId)
+{
+ Uint32 threshold = 0;
+ Logger::LoggerLevel severity = LL_WARNING;
+
+ for(unsigned i = 0; i<EventLogger::matrixSize; i++){
+ if(EventLogger::matrix[i].eventType == eventType){
+ const LogLevel::EventCategory cat = EventLogger::matrix[i].eventCategory;
+ threshold = m_logLevel.getLogLevel(cat);
+ severity = EventLogger::matrix[i].severity;
+ break;
+ }
+ }
+
+ if (threshold <= m_filterLevel){
+ switch (severity){
+ case LL_ALERT:
+ alert(EventLogger::getText(eventType, theData, nodeId));
+ break;
+
+ case LL_CRITICAL:
+ critical(EventLogger::getText(eventType, theData, nodeId));
+ break;
+
+ case LL_WARNING:
+ warning(EventLogger::getText(eventType, theData, nodeId));
+ break;
+
+ case LL_ERROR:
+ error(EventLogger::getText(eventType, theData, nodeId));
+ break;
+
+ case LL_INFO:
+ info(EventLogger::getText(eventType, theData, nodeId));
+ break;
+
+ case LL_DEBUG:
+ debug(EventLogger::getText(eventType, theData, nodeId));
+ break;
+
+ default:
+ info(EventLogger::getText(eventType, theData, nodeId));
+ break;
+ }
+ } // if (..
+}
+
+LogLevel&
+EventLogger::getLoglevel()
+{
+ return m_logLevel;
+}
+
+int
+EventLogger::getFilterLevel() const
+{
+ return m_filterLevel;
+}
+
+void
+EventLogger::setFilterLevel(int filterLevel)
+{
+ m_filterLevel = filterLevel;
+}
+
+//
+// PRIVATE
+//
diff --git a/ndb/src/common/debugger/GrepError.cpp b/ndb/src/common/debugger/GrepError.cpp
new file mode 100644
index 00000000000..ec0c26a5855
--- /dev/null
+++ b/ndb/src/common/debugger/GrepError.cpp
@@ -0,0 +1,133 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <GrepError.hpp>
+
+/**
+ * Error descriptions.
+ */
+
+const GrepError::ErrorDescription GrepError::errorDescriptions[] = {
+ { GrepError::NO_ERROR,
+ "No error" },
+ { GrepError::SUBSCRIPTION_ID_NOMEM,
+ "Not enough resources to allocate the subscription" },
+ { GrepError::SUBSCRIPTION_ID_NOT_FOUND,
+ "The requested subscription (id, key) does not exist"},
+ { GrepError::SUBSCRIPTION_ID_NOT_UNIQUE,
+ "A subscription with (id, key) does already exist"},
+ { GrepError::SUBSCRIPTION_ID_SUMA_FAILED_CREATE,
+ "Suma failed to create a new subscription id"},
+ { GrepError::NULL_VALUE,
+ "NULL"},
+ { GrepError::SEQUENCE_ERROR,
+ "Error when creating or using sequence."},
+ { GrepError::NOSPACE_IN_POOL,
+ "No space left in pool when trying to seize data"},
+ { GrepError::SUBSCRIPTION_ID_ALREADY_EXIST,
+ "A subscription for this replication channel does already exist"},
+ { GrepError::SUBSCRIPTION_NOT_STARTED,
+ "No subscription is started"},
+ { GrepError::SUBSCRIBER_NOT_FOUND,
+ "The subscriber does not exist in SUMA."},
+ { GrepError::WRONG_NO_OF_SECTIONS,
+ "Something is wrong with the supplied arguments"},
+ { GrepError::ILLEGAL_ACTION_WHEN_STOPPING,
+ "Action can not be performed while channel is in stopping state"},
+ { GrepError::SELECTED_TABLE_NOT_FOUND,
+ "The selected table was not found. "},
+ { GrepError::REP_APPLY_LOGRECORD_FAILED,
+ "Failed applying a log record (permanent error)"},
+ { GrepError::REP_APPLY_METARECORD_FAILED,
+ "Failed applying a meta record (permanent error)"},
+ { GrepError::REP_DELETE_NEGATIVE_EPOCH,
+ "Trying to delete a GCI Buffer using a negative epoch."},
+ { GrepError::REP_DELETE_NONEXISTING_EPOCH,
+ "Trying to delete a non-existing GCI Buffer."},
+ { GrepError::REP_NO_CONNECTED_NODES,
+ "There are no connected nodes in the node group."},
+ { GrepError::REP_DISCONNECT,
+ "Global Replication Server disconnected."},
+ { GrepError::COULD_NOT_ALLOCATE_MEM_FOR_SIGNAL,
+ "Could not allocate memory for signal."},
+ { GrepError::REP_NOT_PROPER_TABLE,
+ "Specified table is not a valid table. "
+ "Either the format is not <db>/<schema>/<tablename> or "
+ "the table name is too long "},
+ { GrepError::REP_TABLE_ALREADY_SELECTED,
+ "The specified table is already selected for replication" },
+ { GrepError::REP_TABLE_NOT_FOUND,
+ "The specified table was not found" },
+ { GrepError::START_OF_COMPONENT_IN_WRONG_STATE,
+ "Component or protocol can not be started in the current state."},
+ { GrepError::START_ALREADY_IN_PROGRESS,
+ "Start of replication protocol is already in progress."},
+ { GrepError::ILLEGAL_STOP_EPOCH_ID,
+ "It is not possible to stop on the requested epoch id."},
+ { GrepError::ILLEGAL_USE_OF_COMMAND,
+ "The command cannot be executed in this state."},
+ { GrepError::CHANNEL_NOT_STOPPABLE,
+ "It is not possible to stop the in this state."},
+
+ /**
+ * Applier stuff
+ */
+ { GrepError::REP_APPLY_NONCOMPLETE_GCIBUFFER,
+ "Applier: Ordered to apply an incomplete GCI Buffer."},
+ { GrepError::REP_APPLY_NULL_GCIBUFFER,
+ "Applier: Tried to apply a NULL GCI Buffer."},
+ { GrepError::REP_APPLIER_START_TRANSACTION,
+ "Applier: Could not start a transaction."},
+ { GrepError::REP_APPLIER_NO_TABLE,
+ "Applier: Table does not exist"},
+ { GrepError::REP_APPLIER_NO_OPERATION,
+ "Applier: Cannot get NdbOperation record."},
+ { GrepError::REP_APPLIER_EXECUTE_TRANSACTION,
+ "Applier: Execute transaction failed."},
+ { GrepError::REP_APPLIER_CREATE_TABLE,
+ "Applier: Create table failed."},
+ { GrepError::REP_APPLIER_PREPARE_TABLE,
+ "Applier: Prepare table for create failed."},
+
+ { GrepError::NOT_YET_IMPLEMENTED,
+ "Command or event not yet implemented."}
+};
+
+
+
+
+
+const Uint32
+GrepError::noOfErrorDescs = sizeof(GrepError::errorDescriptions) /
+ sizeof(GrepError::ErrorDescription);
+
+
+/**
+ * gets the corresponding error message to an err code
+ */
+const char *
+GrepError::getErrorDesc(GrepError::Code err) {
+
+ for(Uint32 i = 0; i<noOfErrorDescs; i++){
+ if(err == errorDescriptions[i].errCode){
+ return errorDescriptions[i].name;
+ }
+ }
+ return 0;
+}
+
+
+
diff --git a/ndb/src/common/debugger/LogLevel.cpp b/ndb/src/common/debugger/LogLevel.cpp
new file mode 100644
index 00000000000..5348924bbbb
--- /dev/null
+++ b/ndb/src/common/debugger/LogLevel.cpp
@@ -0,0 +1,29 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <LogLevel.hpp>
+
+const LogLevel::LogLevelCategoryName LogLevel::LOGLEVEL_CATEGORY_NAME[] = {
+ {"LogLevelStartup"},
+ {"LogLevelShutdown"},
+ {"LogLevelStatistic"},
+ {"LogLevelCheckpoint"},
+ {"LogLevelNodeRestart"},
+ {"LogLevelConnection"},
+ {"LogLevelError"},
+ {"LogLevelInfo"},
+ {"LogLevelGrep"}
+};
diff --git a/ndb/src/common/debugger/Makefile b/ndb/src/common/debugger/Makefile
new file mode 100644
index 00000000000..ac3a4475a54
--- /dev/null
+++ b/ndb/src/common/debugger/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := kernel
+DIRS := signaldata
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := trace
+
+SOURCES = SignalLoggerManager.cpp DebuggerNames.cpp BlockNames.cpp LogLevel.cpp EventLogger.cpp GrepError.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/common/debugger/SignalLoggerManager.cpp b/ndb/src/common/debugger/SignalLoggerManager.cpp
new file mode 100644
index 00000000000..e51edbba169
--- /dev/null
+++ b/ndb/src/common/debugger/SignalLoggerManager.cpp
@@ -0,0 +1,513 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "SignalLoggerManager.hpp"
+#include <LongSignal.hpp>
+
+#include <stdlib.h>
+#include <string.h>
+#include <assert.h>
+#include <time.h>
+#include <NdbString.h>
+
+#include <DebuggerNames.hpp>
+
+SignalLoggerManager::SignalLoggerManager()
+{
+ for (int i = 0; i < NO_OF_BLOCKS; i++){
+ logModes[i] = 0;
+ }
+ outputStream = 0;
+}
+
+SignalLoggerManager::~SignalLoggerManager()
+{
+ if(outputStream != 0){
+ fflush(outputStream);
+ fclose(outputStream);
+ outputStream = 0;
+ }
+}
+
+FILE *
+SignalLoggerManager::setOutputStream(FILE * output)
+{
+ if(outputStream != 0){
+ fflush(outputStream);
+ }
+
+ FILE * out = outputStream;
+ outputStream = output;
+ return out;
+}
+
+FILE *
+SignalLoggerManager::getOutputStream() const
+{
+ return outputStream;
+}
+
+void
+SignalLoggerManager::flushSignalLog()
+{
+ if(outputStream != 0)
+ fflush(outputStream);
+}
+
+void
+SignalLoggerManager::setTrace(unsigned long trace)
+{
+ traceId = trace;
+}
+
+unsigned long
+SignalLoggerManager::getTrace() const
+{
+ return traceId;
+}
+
+int
+getParameter(char *blocks[NO_OF_BLOCKS], const char * par, const char * line)
+{
+ const char * loc = strstr(line, par);
+ if(loc == NULL)
+ return 0;
+
+ loc += strlen(par);
+
+ int found = 0;
+
+ char * copy = strdup(loc);
+ char * tmp = copy;
+ bool done = false;
+ while(!done){
+ int len = strcspn(tmp, ", ;:\0");
+ if(len == 0)
+ done = true;
+ else {
+ if(* (tmp + len) != ',')
+ done = true;
+ * (tmp + len) = 0;
+ blocks[found] = strdup(tmp);
+ found ++;
+ tmp += (len + 1);
+ }
+ }
+ free(copy);
+ return found;
+}
+
+
+#define SLM_OFF 0
+#define SLM_ON 1
+#define SLM_TOGGLE 2
+
+int
+SignalLoggerManager::log(LogMode logMode, const char * params)
+{
+ char * blocks[NO_OF_BLOCKS];
+ const int count = getParameter(blocks, "BLOCK=", params);
+
+ int cnt = 0;
+ if((count == 1 && blocks[0] == "ALL") ||
+ count == 0){
+
+ for (int number = 0; number < NO_OF_BLOCKS; ++number){
+ cnt += log(SLM_ON, number, logMode);
+ }
+ } else {
+ for (int i = 0; i < count; ++i){
+ BlockNumber number = getBlockNo(blocks[i]);
+ cnt += log(SLM_ON, number-MIN_BLOCK_NO, logMode);
+ }
+ }
+ for(int i = 0; i<count; i++){
+ free(blocks[i]);
+ }
+
+ return cnt;
+}
+
+int
+SignalLoggerManager::log(int cmd, BlockNumber bno, LogMode logMode)
+{
+ // Normalise blocknumber for use in logModes array
+ const BlockNumber bno2 = bno-MIN_BLOCK_NO;
+ assert(bno2<NO_OF_BLOCKS);
+ switch(cmd){
+ case SLM_ON:
+ logModes[bno2] |= logMode;
+ return 1;
+ break;
+ case SLM_OFF:
+ logModes[bno2] &= (~logMode);
+ return 1;
+ break;
+ case SLM_TOGGLE:
+ logModes[bno2] ^= logMode;
+ return 1;
+ break;
+ }
+ return 0;
+}
+
+int
+SignalLoggerManager::logOn(bool allBlocks, BlockNumber bno, LogMode logMode)
+{
+ if(!allBlocks){
+ return log(SLM_ON, bno, logMode);
+ }
+ int cnt = 0;
+ for(unsigned int i = MIN_BLOCK_NO; i <= MAX_BLOCK_NO; i++)
+ cnt += log(SLM_ON, i, logMode);
+ return cnt;
+}
+
+int
+SignalLoggerManager::logOff(bool allBlocks, BlockNumber bno, LogMode logMode)
+{
+ if(!allBlocks){
+ return log(SLM_OFF, bno, logMode);
+ }
+ int cnt = 0;
+ for(unsigned int i = MIN_BLOCK_NO; i <= MAX_BLOCK_NO; i++)
+ cnt += log(SLM_OFF, i, logMode);
+ return cnt;
+
+}
+
+int
+SignalLoggerManager::logToggle(bool allBlocks, BlockNumber bno, LogMode logMode)
+{
+ if(!allBlocks){
+ return log(SLM_TOGGLE, bno, logMode);
+ }
+ int cnt = 0;
+ for(unsigned int i = MIN_BLOCK_NO; i <= MAX_BLOCK_NO; i++)
+ cnt += log(SLM_TOGGLE, i, logMode);
+ return cnt;
+}
+
+void
+SignalLoggerManager::executeDirect(const SignalHeader& sh,
+ Uint8 prio, // in-out flag
+ const Uint32 * theData, Uint32 node)
+{
+ Uint32 trace = sh.theTrace;
+ Uint32 senderBlockNo = refToBlock(sh.theSendersBlockRef);
+ Uint32 receiverBlockNo = sh.theReceiversBlockNumber;
+
+ if(outputStream != 0 &&
+ (traceId == 0 || traceId == trace) &&
+ (logMatch(senderBlockNo, LogOut) || logMatch(receiverBlockNo, LogIn))){
+ const char* inOutStr = prio == 0 ? "In" : "Out";
+#ifdef VM_TRACE_TIME
+ fprintf(outputStream, "---- Direct --- Signal --- %s - %d ----\n", inOutStr, time(0));
+#else
+ fprintf(outputStream, "---- Direct --- Signal --- %s ----------------\n", inOutStr);
+#endif
+ // XXX pass in/out to print* function somehow
+ printSignalHeader(outputStream, sh, 0, node, true);
+ printSignalData(outputStream, sh, theData);
+ }
+}
+
+/**
+ * For input signals
+ */
+void
+SignalLoggerManager::executeSignal(const SignalHeader& sh, Uint8 prio,
+ const Uint32 * theData, Uint32 node,
+ const SegmentedSectionPtr ptr[3], Uint32 secs)
+{
+ Uint32 trace = sh.theTrace;
+ //Uint32 senderBlockNo = refToBlock(sh.theSendersBlockRef);
+ Uint32 receiverBlockNo = sh.theReceiversBlockNumber;
+
+ if(outputStream != 0 &&
+ (traceId == 0 || traceId == trace) &&
+ logMatch(receiverBlockNo, LogIn)){
+#ifdef VM_TRACE_TIME
+ fprintf(outputStream, "---- Received - Signal - %d ----\n", time(0));
+#else
+ fprintf(outputStream, "---- Received - Signal ----------------\n");
+#endif
+
+ printSignalHeader(outputStream, sh, prio, node, true);
+ printSignalData(outputStream, sh, theData);
+ for (unsigned i = 0; i < secs; i++)
+ printSegmentedSection(outputStream, sh, ptr, i);
+ }
+}
+
+void
+SignalLoggerManager::executeSignal(const SignalHeader& sh, Uint8 prio,
+ const Uint32 * theData, Uint32 node,
+ const LinearSectionPtr ptr[3], Uint32 secs)
+{
+ Uint32 trace = sh.theTrace;
+ //Uint32 senderBlockNo = refToBlock(sh.theSendersBlockRef);
+ Uint32 receiverBlockNo = sh.theReceiversBlockNumber;
+
+ if(outputStream != 0 &&
+ (traceId == 0 || traceId == trace) &&
+ logMatch(receiverBlockNo, LogIn)){
+#ifdef VM_TRACE_TIME
+ fprintf(outputStream, "---- Received - Signal - %d ----\n", time(0));
+#else
+ fprintf(outputStream, "---- Received - Signal ----------------\n");
+#endif
+
+ printSignalHeader(outputStream, sh, prio, node, true);
+ printSignalData(outputStream, sh, theData);
+ for (unsigned i = 0; i < secs; i++)
+ printLinearSection(outputStream, sh, ptr, i);
+ }
+}
+
+/**
+ * For output signals
+ */
+void
+SignalLoggerManager::sendSignal(const SignalHeader& sh,
+ Uint8 prio,
+ const Uint32 * theData, Uint32 node,
+ const LinearSectionPtr ptr[3], Uint32 secs)
+{
+ Uint32 trace = sh.theTrace;
+ Uint32 senderBlockNo = refToBlock(sh.theSendersBlockRef);
+ //Uint32 receiverBlockNo = sh.theReceiversBlockNumber;
+
+ if(outputStream != 0 &&
+ (traceId == 0 || traceId == trace) &&
+ logMatch(senderBlockNo, LogOut)){
+#ifdef VM_TRACE_TIME
+ fprintf(outputStream, "---- Send ----- Signal - %d ----\n", time(0));
+#else
+ fprintf(outputStream, "---- Send ----- Signal ----------------\n");
+#endif
+
+ printSignalHeader(outputStream, sh, prio, node, false);
+ printSignalData(outputStream, sh, theData);
+ for (unsigned i = 0; i < secs; i++)
+ printLinearSection(outputStream, sh, ptr, i);
+ }
+}
+
+/**
+ * For output signals
+ */
+void
+SignalLoggerManager::sendSignal(const SignalHeader& sh, Uint8 prio,
+ const Uint32 * theData, Uint32 node,
+ const SegmentedSectionPtr ptr[3], Uint32 secs)
+{
+ Uint32 trace = sh.theTrace;
+ Uint32 senderBlockNo = refToBlock(sh.theSendersBlockRef);
+ //Uint32 receiverBlockNo = sh.theReceiversBlockNumber;
+
+ if(outputStream != 0 &&
+ (traceId == 0 || traceId == trace) &&
+ logMatch(senderBlockNo, LogOut)){
+#ifdef VM_TRACE_TIME
+ fprintf(outputStream, "---- Send ----- Signal - %d ----\n", time(0));
+#else
+ fprintf(outputStream, "---- Send ----- Signal ----------------\n");
+#endif
+
+ printSignalHeader(outputStream, sh, prio, node, false);
+ printSignalData(outputStream, sh, theData);
+ for (unsigned i = 0; i < secs; i++)
+ printSegmentedSection(outputStream, sh, ptr, i);
+ }
+}
+
+void
+SignalLoggerManager::sendSignalWithDelay(Uint32 delayInMilliSeconds,
+ const SignalHeader & sh, Uint8 prio,
+ const Uint32 * theData, Uint32 node,
+ const SegmentedSectionPtr ptr[3], Uint32 secs)
+{
+ Uint32 trace = sh.theTrace;
+ Uint32 senderBlockNo = refToBlock(sh.theSendersBlockRef);
+ //Uint32 receiverBlockNo = sh.theReceiversBlockNumber;
+
+ if(outputStream != 0 &&
+ (traceId == 0 || traceId == trace) &&
+ logMatch(senderBlockNo, LogOut)){
+#ifdef VM_TRACE_TIME
+ fprintf(outputStream,
+ "---- Send ----- Signal (%d ms) %d\n",
+ delayInMilliSeconds,
+ time(0));
+#else
+ fprintf(outputStream, "---- Send delay Signal (%d ms) ----------\n",
+ delayInMilliSeconds);
+#endif
+
+ printSignalHeader(outputStream, sh, prio, node, false);
+ printSignalData(outputStream, sh, theData);
+ for (unsigned i = 0; i < secs; i++)
+ printSegmentedSection(outputStream, sh, ptr, i);
+ }
+}
+
+/**
+ * Generic messages in the signal log
+ */
+void
+SignalLoggerManager::log(BlockNumber bno, const char * msg)
+{
+ // Normalise blocknumber for use in logModes array
+ const BlockNumber bno2 = bno - MIN_BLOCK_NO;
+ assert(bno2<NO_OF_BLOCKS);
+
+ if(outputStream != 0 &&
+ logModes[bno2] != LogOff){
+ fprintf(outputStream, "%s: %s\n", getBlockName(bno, "API"), msg);
+ }
+}
+
+
+void
+SignalLoggerManager::printSignalHeader(FILE * output,
+ const SignalHeader & sh,
+ Uint8 prio,
+ Uint32 node,
+ bool printReceiversSignalId)
+{
+ Uint32 receiverBlockNo = sh.theReceiversBlockNumber;
+ Uint32 receiverProcessor = node;
+ Uint32 gsn = sh.theVerId_signalNumber;
+ Uint32 senderBlockNo = refToBlock(sh.theSendersBlockRef);
+ Uint32 senderProcessor = refToNode(sh.theSendersBlockRef);
+ Uint32 length = sh.theLength;
+ Uint32 trace = sh.theTrace;
+ Uint32 rSigId = sh.theSignalId;
+ Uint32 sSigId = sh.theSendersSignalId;
+
+ const char * signalName = getSignalName(gsn);
+ const char * rBlockName = getBlockName(receiverBlockNo, "API");
+ const char * sBlockName = getBlockName(senderBlockNo, "API");
+
+ if(printReceiversSignalId)
+ fprintf(output,
+ "r.bn: %d \"%s\", r.proc: %d, r.sigId: %d gsn: %d \"%s\" prio: %d\n"
+ ,receiverBlockNo, rBlockName, receiverProcessor, rSigId,
+ gsn, signalName, prio);
+ else
+ fprintf(output,
+ "r.bn: %d \"%s\", r.proc: %d, gsn: %d \"%s\" prio: %d\n",
+ receiverBlockNo, rBlockName, receiverProcessor, gsn,
+ signalName, prio);
+
+ fprintf(output,
+ "s.bn: %d \"%s\", s.proc: %d, s.sigId: %d length: %d trace: %d "
+ "#sec: %d fragInf: %d\n",
+ senderBlockNo, sBlockName, senderProcessor, sSigId, length, trace,
+ sh.m_noOfSections, sh.m_fragmentInfo);
+}
+
+void
+SignalLoggerManager::printSignalData(FILE * output,
+ const SignalHeader & sh,
+ const Uint32 * signalData)
+{
+ Uint32 len = sh.theLength;
+ SignalDataPrintFunction printFunction =
+ findPrintFunction(sh.theVerId_signalNumber);
+
+ bool ok = false; // done with printing
+ if(printFunction != 0){
+ ok = (* printFunction)(output, signalData, len, sh.theReceiversBlockNumber);
+ }
+ if(!ok){
+ while(len >= 7){
+ fprintf(output,
+ " H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x\n",
+ signalData[0], signalData[1], signalData[2], signalData[3],
+ signalData[4], signalData[5], signalData[6]);
+ len -= 7;
+ signalData += 7;
+ }
+ if(len > 0){
+ for(Uint32 i = 0; i<len; i++)
+ fprintf(output, " H\'%.8x", signalData[i]);
+ fprintf(output, "\n");
+ }
+ }
+}
+
+void
+SignalLoggerManager::printLinearSection(FILE * output,
+ const SignalHeader & sh,
+ const LinearSectionPtr ptr[3],
+ unsigned i)
+{
+ fprintf(output, "SECTION %u type=linear", i);
+ if (i >= 3) {
+ fprintf(output, " *** invalid ***\n");
+ return;
+ }
+ const Uint32 len = ptr[i].sz;
+ const Uint32 * data = ptr[i].p;
+ Uint32 pos = 0;
+ fprintf(output, " size=%u\n", (unsigned)len);
+ while (pos < len) {
+ printDataWord(output, pos, data[pos]);
+ }
+ if (len > 0)
+ putc('\n', output);
+}
+
+void
+SignalLoggerManager::printSegmentedSection(FILE * output,
+ const SignalHeader & sh,
+ const SegmentedSectionPtr ptr[3],
+ unsigned i)
+{
+ fprintf(output, "SECTION %u type=segmented", i);
+ if (i >= 3) {
+ fprintf(output, " *** invalid ***\n");
+ return;
+ }
+ const Uint32 len = ptr[i].sz;
+ SectionSegment * ssp = ptr[i].p;
+ Uint32 pos = 0;
+ fprintf(output, " size=%u\n", (unsigned)len);
+ while (pos < len) {
+ if (pos > 0 && pos % SectionSegment::DataLength == 0) {
+ ssp = g_sectionSegmentPool.getPtr(ssp->m_nextSegment);
+ }
+ printDataWord(output, pos, ssp->theData[pos % SectionSegment::DataLength]);
+ }
+ if (len > 0)
+ putc('\n', output);
+}
+
+void
+SignalLoggerManager::printDataWord(FILE * output, Uint32 & pos, const Uint32 data)
+{
+ const char* const hex = "0123456789abcdef";
+ if (pos > 0 && pos % 7 == 0)
+ putc('\n', output);
+ putc(' ', output);
+ putc('H', output);
+ putc('\'', output);
+ for (int i = 7; i >= 0; i--)
+ putc(hex[(data >> (i << 2)) & 0xf], output);
+ pos++;
+}
diff --git a/ndb/src/common/debugger/signaldata/AccLock.cpp b/ndb/src/common/debugger/signaldata/AccLock.cpp
new file mode 100644
index 00000000000..affed431957
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/AccLock.cpp
@@ -0,0 +1,75 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/AccLock.hpp>
+#include <SignalLoggerManager.hpp>
+
+bool
+printACC_LOCKREQ(FILE* output, const Uint32* theData, Uint32 len, Uint16 rbn)
+{
+ const AccLockReq* const sig = (const AccLockReq*)theData;
+ Uint32 reqtype = sig->requestInfo & 0xFF;
+ switch (sig->returnCode) {
+ case RNIL:
+ fprintf(output, " returnCode=RNIL");
+ break;
+ case AccLockReq::Success:
+ fprintf(output, " returnCode=Success");
+ break;
+ case AccLockReq::IsBlocked:
+ fprintf(output, " returnCode=IsBlocked");
+ break;
+ case AccLockReq::WouldBlock:
+ fprintf(output, " returnCode=WouldBlock");
+ break;
+ case AccLockReq::Refused:
+ fprintf(output, " returnCode=Refused");
+ break;
+ case AccLockReq::NoFreeOp:
+ fprintf(output, " returnCode=NoFreeOp");
+ break;
+ default:
+ fprintf(output, " returnCode=%u?", sig->returnCode);
+ break;
+ }
+ switch (reqtype) {
+ case AccLockReq::LockShared:
+ fprintf(output, " req=LockShared\n");
+ break;
+ case AccLockReq::LockExclusive:
+ fprintf(output, " req=LockExclusive\n");
+ break;
+ case AccLockReq::Unlock:
+ fprintf(output, " req=Unlock\n");
+ break;
+ case AccLockReq::Abort:
+ fprintf(output, " req=Abort\n");
+ break;
+ default:
+ fprintf(output, " req=%u\n", reqtype);
+ break;
+ }
+ fprintf(output, " accOpPtr: 0x%x\n", sig->accOpPtr);
+ if (reqtype == AccLockReq::LockShared ||
+ reqtype == AccLockReq::LockExclusive) {
+ fprintf(output, " userPtr: 0x%x userRef: 0x%x\n", sig->userPtr, sig->userRef);
+ fprintf(output, " table: id=%u", sig->tableId);
+ fprintf(output, " fragment: id=%u ptr=0x%x\n", sig->fragId, sig->fragPtrI);
+ fprintf(output, " tuple: addr=0x%x hashValue=%x\n", sig->tupAddr, sig->hashValue);
+ fprintf(output, " transid: %08x %08x\n", sig->transId1, sig->transId2);
+ }
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/AlterIndx.cpp b/ndb/src/common/debugger/signaldata/AlterIndx.cpp
new file mode 100644
index 00000000000..e1865136fc3
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/AlterIndx.cpp
@@ -0,0 +1,35 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/AlterIndx.hpp>
+
+bool printALTER_INDX_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const AlterIndxReq * const sig = (AlterIndxReq *) theData;
+ return false;
+}
+
+bool printALTER_INDX_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const AlterIndxConf * const sig = (AlterIndxConf *) theData;
+ return false;
+}
+
+bool printALTER_INDX_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const AlterIndxRef * const sig = (AlterIndxRef *) theData;
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/AlterTab.cpp b/ndb/src/common/debugger/signaldata/AlterTab.cpp
new file mode 100644
index 00000000000..f9521984095
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/AlterTab.cpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/AlterTab.hpp>
+
+bool printALTER_TAB_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const AlterTabReq * const sig = (AlterTabReq *) theData;
+
+ return false;
+}
+
+bool printALTER_TAB_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const AlterTabConf * const sig = (AlterTabConf *) theData;
+
+ return false;
+}
+
+bool printALTER_TAB_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const AlterTabRef * const sig = (AlterTabRef *) theData;
+
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/AlterTable.cpp b/ndb/src/common/debugger/signaldata/AlterTable.cpp
new file mode 100644
index 00000000000..59909c8e490
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/AlterTable.cpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/AlterTable.hpp>
+
+bool printALTER_TABLE_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const AlterTableReq * const sig = (AlterTableReq *) theData;
+
+ return false;
+}
+
+bool printALTER_TABLE_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const AlterTableConf * const sig = (AlterTableConf *) theData;
+
+ return false;
+}
+
+bool printALTER_TABLE_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const AlterTableRef * const sig = (AlterTableRef *) theData;
+
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/AlterTrig.cpp b/ndb/src/common/debugger/signaldata/AlterTrig.cpp
new file mode 100644
index 00000000000..d488fd6e348
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/AlterTrig.cpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/AlterTrig.hpp>
+
+bool printALTER_TRIG_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const AlterTrigReq * const sig = (AlterTrigReq *) theData;
+
+ fprintf(output, "User: %u, ", sig->getUserRef());
+ fprintf(output, "Trigger id: %u, ", sig->getTriggerId());
+ fprintf(output, "\n");
+
+ return false;
+}
+
+bool printALTER_TRIG_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const AlterTrigConf * const sig = (AlterTrigConf *) theData;
+
+ fprintf(output, "User: %u, ", sig->getUserRef());
+ fprintf(output, "Trigger id: %u, ", sig->getTriggerId());
+ fprintf(output, "\n");
+
+ return false;
+}
+
+bool printALTER_TRIG_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const AlterTrigRef * const sig = (AlterTrigRef *) theData;
+
+ fprintf(output, "User: %u, ", sig->getUserRef());
+ fprintf(output, "Trigger id: %u, ", sig->getTriggerId());
+ fprintf(output, "Error code: %u, ", sig->getErrorCode());
+ fprintf(output, "\n");
+
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/BackupImpl.cpp b/ndb/src/common/debugger/signaldata/BackupImpl.cpp
new file mode 100644
index 00000000000..be9e43e3df1
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/BackupImpl.cpp
@@ -0,0 +1,142 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <trigger_definitions.h>
+#include <signaldata/BackupImpl.hpp>
+
+bool
+printDEFINE_BACKUP_REQ(FILE * out, const Uint32 * data, Uint32 len, Uint16 bno){
+ DefineBackupReq* sig = (DefineBackupReq*)data;
+ fprintf(out, " backupPtr: %d backupId: %d clientRef: %d clientData: %d\n",
+ sig->backupPtr, sig->backupId, sig->clientRef, sig->clientData);
+ fprintf(out, " backupKey: [ %08x%08x ] DataLength: %d\n",
+ sig->backupKey[0], sig->backupKey[1], sig->backupDataLen);
+ char buf[sig->nodes.TextLength + 1];
+ fprintf(out, " Nodes: %s\n", sig->nodes.getText(buf));
+ return true;
+}
+
+bool
+printDEFINE_BACKUP_REF(FILE * out, const Uint32 * data, Uint32 len, Uint16 bno){
+ DefineBackupRef* sig = (DefineBackupRef*)data;
+ fprintf(out, " backupPtr: %d backupId: %d errorCode: %d\n",
+ sig->backupPtr, sig->backupId, sig->errorCode);
+ return true;
+}
+
+bool
+printDEFINE_BACKUP_CONF(FILE * out, const Uint32 * data, Uint32 l, Uint16 bno){
+ DefineBackupConf* sig = (DefineBackupConf*)data;
+ fprintf(out, " backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+}
+
+bool
+printSTART_BACKUP_REQ(FILE * out, const Uint32 * data, Uint32 l, Uint16 bno){
+ StartBackupReq* sig = (StartBackupReq*)data;
+ fprintf(out, " backupPtr: %d backupId: %d signalNo: %d of %d\n",
+ sig->backupPtr, sig->backupId,
+ sig->signalNo + 1, sig->noOfSignals);
+ for(Uint32 i = 0; i<sig->noOfTableTriggers; i++)
+ fprintf(out,
+ " Table: %d Triggers = [ insert: %d update: %d delete: %d ]\n",
+ sig->tableTriggers[i].tableId,
+ sig->tableTriggers[i].triggerIds[TriggerEvent::TE_INSERT],
+ sig->tableTriggers[i].triggerIds[TriggerEvent::TE_UPDATE],
+ sig->tableTriggers[i].triggerIds[TriggerEvent::TE_DELETE]);
+ return true;
+}
+
+bool
+printSTART_BACKUP_REF(FILE * out, const Uint32 * data, Uint32 len, Uint16 bno){
+ StartBackupRef* sig = (StartBackupRef*)data;
+ fprintf(out, " backupPtr: %d backupId: %d errorCode: %d\n",
+ sig->backupPtr, sig->backupId, sig->errorCode);
+ return true;
+}
+
+bool
+printSTART_BACKUP_CONF(FILE * out, const Uint32 * data, Uint32 l, Uint16 bno){
+ StartBackupConf* sig = (StartBackupConf*)data;
+ fprintf(out, " backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+}
+
+bool
+printBACKUP_FRAGMENT_REQ(FILE * out, const Uint32 * data, Uint32 l, Uint16 bno){
+ BackupFragmentReq* sig = (BackupFragmentReq*)data;
+ fprintf(out, " backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ fprintf(out, " tableId: %d fragmentNo: %d (count = %d)\n",
+ sig->tableId, sig->fragmentNo, sig->count);
+ return true;
+}
+
+bool
+printBACKUP_FRAGMENT_REF(FILE * out, const Uint32 * data, Uint32 l, Uint16 bno){
+ BackupFragmentRef* sig = (BackupFragmentRef*)data;
+ fprintf(out, " backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ fprintf(out, " tableId: %d fragmentNo: %d errorCode: %d\n",
+ sig->tableId, sig->fragmentNo, sig->errorCode);
+ return true;
+}
+
+bool
+printBACKUP_FRAGMENT_CONF(FILE * out, const Uint32 * data, Uint32 l, Uint16 b){
+ BackupFragmentConf* sig = (BackupFragmentConf*)data;
+ fprintf(out, " backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ fprintf(out, " tableId: %d fragmentNo: %d records: %d bytes: %d\n",
+ sig->tableId, sig->fragmentNo, sig->noOfRecords, sig->noOfBytes);
+ return true;
+}
+
+bool
+printSTOP_BACKUP_REQ(FILE * out, const Uint32 * data, Uint32 l, Uint16 bno){
+ StopBackupReq* sig = (StopBackupReq*)data;
+ fprintf(out, " backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+}
+
+bool
+printSTOP_BACKUP_REF(FILE * out, const Uint32 * data, Uint32 len, Uint16 bno){
+ StopBackupRef* sig = (StopBackupRef*)data;
+ fprintf(out, " backupPtr: %d backupId: %d errorCode: %d\n",
+ sig->backupPtr, sig->backupId, sig->errorCode);
+ return true;
+}
+
+bool
+printSTOP_BACKUP_CONF(FILE * out, const Uint32 * data, Uint32 l, Uint16 bno){
+ StopBackupConf* sig = (StopBackupConf*)data;
+ fprintf(out, " backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+}
+
+bool
+printBACKUP_STATUS_REQ(FILE *, const Uint32 *, Uint32, Uint16){
+ return false;
+}
+
+bool
+printBACKUP_STATUS_CONF(FILE *, const Uint32 *, Uint32, Uint16){
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/BackupSignalData.cpp b/ndb/src/common/debugger/signaldata/BackupSignalData.cpp
new file mode 100644
index 00000000000..4b0a0e07b66
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/BackupSignalData.cpp
@@ -0,0 +1,129 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <signaldata/BackupSignalData.hpp>
+
+bool
+printBACKUP_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 bno){
+ BackupReq* sig = (BackupReq*)theData;
+ fprintf(output, " senderData: %d DataLength: %d\n",
+ sig->senderData,
+ sig->backupDataLen);
+ return true;
+}
+
+bool
+printBACKUP_DATA(FILE * output, const Uint32 * theData, Uint32 len, Uint16 bno){
+ BackupData * sig = (BackupData*)theData;
+ if(sig->requestType == BackupData::ClientToMaster){
+ fprintf(output, " ClientToMaster: senderData: %d backupId: %d\n",
+ sig->senderData, sig->backupId);
+ } else if(sig->requestType == BackupData::MasterToSlave){
+ fprintf(output, " MasterToSlave: backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ }
+ return false;
+}
+
+bool
+printBACKUP_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 bno){
+
+ BackupRef* sig = (BackupRef*)theData;
+ fprintf(output, " senderData: %d errorCode: %d masterRef: %d\n",
+ sig->senderData,
+ sig->errorCode,
+ sig->masterRef);
+ return true;
+}
+
+bool
+printBACKUP_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 bno){
+ BackupConf* sig = (BackupConf*)theData;
+ fprintf(output, " senderData: %d backupId: %d\n",
+ sig->senderData,
+ sig->backupId);
+ return true;
+}
+
+bool
+printBACKUP_ABORT_REP(FILE * out, const Uint32 * data, Uint32 len, Uint16 bno){
+ BackupAbortRep* sig = (BackupAbortRep*)data;
+ fprintf(out, " senderData: %d backupId: %d reason: %d\n",
+ sig->senderData,
+ sig->backupId,
+ sig->reason);
+ return true;
+}
+
+bool
+printBACKUP_COMPLETE_REP(FILE * out, const Uint32 * data, Uint32 len, Uint16 b){
+ BackupCompleteRep* sig = (BackupCompleteRep*)data;
+ fprintf(out, " senderData: %d backupId: %d records: %d bytes: %d\n",
+ sig->senderData,
+ sig->backupId,
+ sig->noOfRecords,
+ sig->noOfBytes);
+ return true;
+}
+
+bool
+printBACKUP_NF_COMPLETE_REP(FILE*, const Uint32*, Uint32, Uint16){
+ return false;
+}
+
+bool
+printABORT_BACKUP_ORD(FILE * out, const Uint32 * data, Uint32 len, Uint16 b){
+ AbortBackupOrd* sig = (AbortBackupOrd*)data;
+
+ AbortBackupOrd::RequestType rt =(AbortBackupOrd::RequestType)sig->requestType;
+ switch(rt){
+ case AbortBackupOrd::ClientAbort:
+ fprintf(out, " ClientAbort: senderData: %d backupId: %d\n",
+ sig->senderData, sig->backupId);
+ return true;
+ break;
+ case AbortBackupOrd::BackupComplete:
+ fprintf(out, " BackupComplete: backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+ case AbortBackupOrd::BackupFailure:
+ fprintf(out, " BackupFailure: backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+ case AbortBackupOrd::LogBufferFull:
+ fprintf(out, " LogBufferFull: backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+ break;
+ case AbortBackupOrd::FileOrScanError:
+ fprintf(out, " FileOrScanError: backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+ break;
+ case AbortBackupOrd::BackupFailureDueToNodeFail:
+ fprintf(out, " BackupFailureDueToNodeFail: backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+ break;
+ case AbortBackupOrd::OkToClean:
+ fprintf(out, " OkToClean: backupPtr: %d backupId: %d\n",
+ sig->backupPtr, sig->backupId);
+ return true;
+ break;
+ }
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/CloseComReqConf.cpp b/ndb/src/common/debugger/signaldata/CloseComReqConf.cpp
new file mode 100644
index 00000000000..11ee0948c17
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/CloseComReqConf.cpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <kernel_types.h>
+#include <BlockNumbers.h>
+#include <signaldata/CloseComReqConf.hpp>
+
+bool
+printCLOSECOMREQCONF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo){
+
+ CloseComReqConf * cc = (CloseComReqConf*)theData;
+
+ fprintf(output, " xxxBlockRef = (%d, %d) failNo = %d noOfNodes = %d\n",
+ refToBlock(cc->xxxBlockRef), refToNode(cc->xxxBlockRef),
+ cc->failNo, cc->noOfNodes);
+
+ int hits = 0;
+ fprintf(output, " Nodes: ");
+ for(int i = 0; i<MAX_NODES; i++){
+ if(NodeBitmask::get(cc->theNodes, i)){
+ hits++;
+ fprintf(output, " %d", i);
+ }
+ if(hits == 16){
+ fprintf(output, "\n Nodes: ");
+ hits = 0;
+ }
+ }
+ if(hits != 0)
+ fprintf(output, "\n");
+
+ return true;
+}
+
+
diff --git a/ndb/src/common/debugger/signaldata/ContinueB.cpp b/ndb/src/common/debugger/signaldata/ContinueB.cpp
new file mode 100644
index 00000000000..054909d961e
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/ContinueB.cpp
@@ -0,0 +1,36 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <kernel_types.h>
+#include <BlockNumbers.h>
+#include <signaldata/DihContinueB.hpp>
+#include <signaldata/NdbfsContinueB.hpp>
+
+bool
+printCONTINUEB(FILE * output, const Uint32 * theData, Uint32 len,
+ Uint16 receiverBlockNo){
+ if(receiverBlockNo == DBDIH){
+ return printCONTINUEB_DBDIH(output, theData, len);
+ } else if(receiverBlockNo == NDBFS) {
+ return printCONTINUEB_NDBFS(output, theData, len);
+ }
+
+ return false;
+}
+
+
diff --git a/ndb/src/common/debugger/signaldata/CopyGCI.cpp b/ndb/src/common/debugger/signaldata/CopyGCI.cpp
new file mode 100644
index 00000000000..96186e82525
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/CopyGCI.cpp
@@ -0,0 +1,58 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/CopyGCIReq.hpp>
+
+static
+void
+print(char * buf, size_t buf_len, CopyGCIReq::CopyReason r){
+ switch(r){
+ case CopyGCIReq::IDLE:
+ snprintf(buf, buf_len, "IDLE");
+ break;
+ case CopyGCIReq::LOCAL_CHECKPOINT:
+ snprintf(buf, buf_len, "LOCAL_CHECKPOINT");
+ break;
+ case CopyGCIReq::RESTART:
+ snprintf(buf, buf_len, "RESTART");
+ break;
+ case CopyGCIReq::GLOBAL_CHECKPOINT:
+ snprintf(buf, buf_len, "GLOBAL_CHECKPOINT");
+ break;
+ case CopyGCIReq::INITIAL_START_COMPLETED:
+ snprintf(buf, buf_len, "INITIAL_START_COMPLETED");
+ break;
+ default:
+ snprintf(buf, buf_len, "<Unknown>");
+ }
+}
+
+bool
+printCOPY_GCI_REQ(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 recBlockNo){
+ CopyGCIReq * sig = (CopyGCIReq*)theData;
+
+ static char buf[255];
+ print(buf, sizeof(buf), (CopyGCIReq::CopyReason)sig->copyReason);
+
+ fprintf(output, " SenderData: %d CopyReason: %s StartWord: %d\n",
+ sig->anyData,
+ buf,
+ sig->startWord);
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/CreateEvnt.cpp b/ndb/src/common/debugger/signaldata/CreateEvnt.cpp
new file mode 100644
index 00000000000..7b497d6a974
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/CreateEvnt.cpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/CreateEvnt.hpp>
+
+bool printCREATE_EVNT_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const CreateEvntReq * const sig = (CreateEvntReq *) theData;
+
+ return false;
+}
+
+bool printCREATE_EVNT_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const CreateEvntConf * const sig = (CreateEvntConf *) theData;
+
+ return false;
+}
+
+bool printCREATE_EVNT_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const CreateEvntRef * const sig = (CreateEvntRef *) theData;
+
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/CreateFragmentation.cpp b/ndb/src/common/debugger/signaldata/CreateFragmentation.cpp
new file mode 100644
index 00000000000..6685345f17a
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/CreateFragmentation.cpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/CreateFragmentation.hpp>
+
+bool
+printCREATE_FRAGMENTATION_REQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const CreateFragmentationReq * const sig = (CreateFragmentationReq *)theData;
+ fprintf(output, " senderRef: %x\n", sig->senderRef);
+ fprintf(output, " senderData: %x\n", sig->senderData);
+ fprintf(output, " fragmentationType: %x\n", sig->fragmentationType);
+ fprintf(output, " noOfFragments: %x\n", sig->noOfFragments);
+ fprintf(output, " fragmentNode: %x\n", sig->fragmentNode);
+ if (sig->primaryTableId == RNIL)
+ fprintf(output, " primaryTableId: none\n", sig->primaryTableId);
+ else
+ fprintf(output, " primaryTableId: %x\n", sig->primaryTableId);
+ return true;
+}
+
+bool
+printCREATE_FRAGMENTATION_REF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const CreateFragmentationRef * const sig = (CreateFragmentationRef *)theData;
+ fprintf(output, " senderRef: %x\n", sig->senderRef);
+ fprintf(output, " senderData: %x\n", sig->senderData);
+ fprintf(output, " errorCode: %x\n", sig->errorCode);
+ return true;
+}
+
+bool
+printCREATE_FRAGMENTATION_CONF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const CreateFragmentationConf * const sig =
+ (CreateFragmentationConf *)theData;
+ fprintf(output, " senderRef: %x\n", sig->senderRef);
+ fprintf(output, " senderData: %x\n", sig->senderData);
+ fprintf(output, " noOfReplicas: %x\n", sig->noOfReplicas);
+ fprintf(output, " noOfFragments: %x\n", sig->noOfFragments);
+ return true;
+}
+
diff --git a/ndb/src/common/debugger/signaldata/CreateIndx.cpp b/ndb/src/common/debugger/signaldata/CreateIndx.cpp
new file mode 100644
index 00000000000..8fcbb9279ed
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/CreateIndx.cpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/CreateIndx.hpp>
+
+bool printCREATE_INDX_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const CreateIndxReq * const sig = (CreateIndxReq *) theData;
+
+ return false;
+}
+
+bool printCREATE_INDX_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const CreateIndxConf * const sig = (CreateIndxConf *) theData;
+
+ return false;
+}
+
+bool printCREATE_INDX_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const CreateIndxRef * const sig = (CreateIndxRef *) theData;
+
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/CreateTrig.cpp b/ndb/src/common/debugger/signaldata/CreateTrig.cpp
new file mode 100644
index 00000000000..d8360dec4d5
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/CreateTrig.cpp
@@ -0,0 +1,120 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/CreateTrig.hpp>
+
+bool printCREATE_TRIG_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const CreateTrigReq * const sig = (CreateTrigReq *) theData;
+
+ //char triggerName[MAX_TAB_NAME_SIZE];
+ char triggerType[32];
+ char triggerActionTime[32];
+ char triggerEvent[32];
+
+ //sig->getTriggerName((char *) &triggerName);
+ switch (sig->getTriggerType()) {
+ case(TriggerType::SECONDARY_INDEX):
+ snprintf(triggerType, sizeof(triggerType), "SECONDARY_INDEX");
+ break;
+ case(TriggerType::SUBSCRIPTION):
+ snprintf(triggerType, sizeof(triggerType), "SUBSCRIPTION");
+ break;
+ case(TriggerType::ORDERED_INDEX):
+ snprintf(triggerType, sizeof(triggerType), "ORDERED_INDEX");
+ break;
+ default:
+ snprintf(triggerType, sizeof(triggerType), "UNKNOWN [%d]", (int)sig->getTriggerType());
+ break;
+ }
+ switch (sig->getTriggerActionTime()) {
+ case (TriggerActionTime::TA_BEFORE):
+ snprintf(triggerActionTime, sizeof(triggerActionTime), "BEFORE");
+ break;
+ case(TriggerActionTime::TA_AFTER):
+ snprintf(triggerActionTime, sizeof(triggerActionTime), "AFTER");
+ break;
+ case (TriggerActionTime::TA_DEFERRED):
+ snprintf(triggerActionTime, sizeof(triggerActionTime), "DEFERRED");
+ break;
+ case (TriggerActionTime::TA_DETACHED):
+ snprintf(triggerActionTime, sizeof(triggerActionTime), "DETACHED");
+ break;
+ default:
+ snprintf(triggerActionTime, sizeof(triggerActionTime),
+ "UNKNOWN [%d]", (int)sig->getTriggerActionTime());
+ break;
+ }
+ switch (sig->getTriggerEvent()) {
+ case (TriggerEvent::TE_INSERT):
+ snprintf(triggerEvent, sizeof(triggerEvent), "INSERT");
+ break;
+ case(TriggerEvent::TE_DELETE):
+ snprintf(triggerEvent, sizeof(triggerEvent), "DELETE");
+ break;
+ case(TriggerEvent::TE_UPDATE):
+ snprintf(triggerEvent, sizeof(triggerEvent), "UPDATE");
+ break;
+ case(TriggerEvent::TE_CUSTOM):
+ snprintf(triggerEvent, sizeof(triggerEvent), "CUSTOM");
+ break;
+ default:
+ snprintf(triggerEvent, sizeof(triggerEvent), "UNKNOWN [%d]", (int)sig->getTriggerEvent());
+ break;
+ }
+
+ fprintf(output, "User: %u, ", sig->getUserRef());
+ //fprintf(output, "Trigger name: \"%s\"\n", triggerName);
+ fprintf(output, "Type: %s, ", triggerType);
+ fprintf(output, "Action: %s, ", triggerActionTime);
+ fprintf(output, "Event: %s, ", triggerEvent);
+ fprintf(output, "Trigger id: %u, ", sig->getTriggerId());
+ fprintf(output, "Table id: %u, ", sig->getTableId());
+ fprintf(output, "Monitor replicas: %s ", (sig->getMonitorReplicas())?"true":"false");
+ fprintf(output, "Monitor all attributes: %s ", (sig->getMonitorAllAttributes())?"true":"false");
+ const AttributeMask& attributeMask = sig->getAttributeMask();
+
+ char buf[attributeMask.TextLength + 1];
+ fprintf(output, "Attribute mask: %s", attributeMask.getText(buf));
+ fprintf(output, "\n");
+
+ return false;
+}
+
+bool printCREATE_TRIG_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const CreateTrigConf * const sig = (CreateTrigConf *) theData;
+
+ fprintf(output, "User: %u, ", sig->getUserRef());
+ fprintf(output, "Trigger id: %u, ", sig->getTriggerId());
+ fprintf(output, "Table id: %u, ", sig->getTableId());
+ fprintf(output, "\n");
+
+ return false;
+}
+
+bool printCREATE_TRIG_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const CreateTrigRef * const sig = (CreateTrigRef *) theData;
+
+ fprintf(output, "User: %u, ", sig->getUserRef());
+ fprintf(output, "Trigger id: %u, ", sig->getTriggerId());
+ fprintf(output, "Table id: %u, ", sig->getTableId());
+ fprintf(output, "Error code: %u, ", sig->getErrorCode());
+ fprintf(output, "\n");
+
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/DictTabInfo.cpp b/ndb/src/common/debugger/signaldata/DictTabInfo.cpp
new file mode 100644
index 00000000000..a0e0195adad
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/DictTabInfo.cpp
@@ -0,0 +1,153 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/DictTabInfo.hpp>
+#include <ndb_limits.h>
+
+//static
+const
+SimpleProperties::SP2StructMapping
+DictTabInfo::TableMapping[] = {
+ DTIMAPS(Table, TableName, TableName, 0, MAX_TAB_NAME_SIZE),
+ DTIMAP(Table, TableId, TableId),
+ DTIMAP(Table, SecondTableId, SecondTableId),
+ DTIMAPS(Table, PrimaryTable, PrimaryTable, 0, MAX_TAB_NAME_SIZE),
+ DTIMAP(Table, PrimaryTableId, PrimaryTableId),
+ DTIMAP2(Table, TableLoggedFlag, TableLoggedFlag, 0, 1),
+ DTIMAP2(Table, TableKValue, TableKValue, 6, 6),
+ DTIMAP2(Table, MinLoadFactor, MinLoadFactor, 0, 90),
+ DTIMAP2(Table, MaxLoadFactor, MaxLoadFactor, 25, 110),
+ DTIMAP2(Table, FragmentTypeVal, FragmentType, 0, 3),
+ DTIMAP2(Table, TableStorageVal, TableStorage, 0, 0),
+ DTIMAP2(Table, ScanOptimised, ScanOptimised, 0, 0),
+ DTIMAP2(Table, FragmentKeyTypeVal, FragmentKeyType, 0, 2),
+ DTIMAP2(Table, TableTypeVal, TableType, 1, 3),
+ DTIMAP(Table, NoOfKeyAttr, NoOfKeyAttr),
+ DTIMAP2(Table, NoOfAttributes, NoOfAttributes, 1, MAX_ATTRIBUTES_IN_TABLE),
+ DTIMAP(Table, NoOfNullable, NoOfNullable),
+ DTIMAP2(Table, NoOfVariable, NoOfVariable, 0, 0),
+ DTIMAP(Table, KeyLength, KeyLength),
+ DTIMAP(Table, TableVersion, TableVersion),
+ DTIMAP(Table, IndexState, IndexState),
+ DTIMAP(Table, InsertTriggerId, InsertTriggerId),
+ DTIMAP(Table, UpdateTriggerId, UpdateTriggerId),
+ DTIMAP(Table, DeleteTriggerId, DeleteTriggerId),
+ DTIMAP(Table, CustomTriggerId, CustomTriggerId),
+ DTIMAP2(Table, FrmLen, FrmLen, 0, MAX_FRM_DATA_SIZE),
+ DTIMAPB(Table, FrmData, FrmData, 0, MAX_FRM_DATA_SIZE, FrmLen),
+ DTIBREAK(AttributeName)
+};
+
+//static
+const Uint32 DictTabInfo::TableMappingSize =
+sizeof(DictTabInfo::TableMapping) / sizeof(SimpleProperties::SP2StructMapping);
+
+//static
+const
+SimpleProperties::SP2StructMapping
+DictTabInfo::AttributeMapping[] = {
+ DTIMAPS(Attribute, AttributeName, AttributeName, 0, MAX_ATTR_NAME_SIZE),
+ DTIMAP(Attribute, AttributeId, AttributeId),
+ DTIMAP2(Attribute, AttributeType, AttributeType, 0, 3),
+ DTIMAP2(Attribute, AttributeSize, AttributeSize, 3, 7),
+ DTIMAP2(Attribute, AttributeArraySize, AttributeArraySize, 0, 65535),
+ DTIMAP2(Attribute, AttributeKeyFlag, AttributeKeyFlag, 0, 1),
+ DTIMAP2(Attribute, AttributeStorage, AttributeStorage, 0, 0),
+ DTIMAP2(Attribute, AttributeNullableFlag, AttributeNullableFlag, 0, 1),
+ DTIMAP2(Attribute, AttributeDGroup, AttributeDGroup, 0, 1),
+ DTIMAP2(Attribute, AttributeDKey, AttributeDKey, 0, 1),
+ DTIMAP2(Attribute, AttributeStoredInd, AttributeStoredInd, 0, 1),
+ DTIMAP2(Attribute, AttributeGroup, AttributeGroup, 0, 0),
+ DTIMAP(Attribute, AttributeExtType, AttributeExtType),
+ DTIMAP(Attribute, AttributeExtPrecision, AttributeExtPrecision),
+ DTIMAP(Attribute, AttributeExtScale, AttributeExtScale),
+ DTIMAP(Attribute, AttributeExtLength, AttributeExtLength),
+ DTIMAP2(Attribute, AttributeAutoIncrement, AttributeAutoIncrement, 0, 1),
+ DTIMAPS(Attribute, AttributeDefaultValue, AttributeDefaultValue,
+ 0, MAX_ATTR_DEFAULT_VALUE_SIZE),
+ DTIBREAK(AttributeEnd)
+};
+
+//static
+const Uint32 DictTabInfo::AttributeMappingSize =
+sizeof(DictTabInfo::AttributeMapping) /
+sizeof(SimpleProperties::SP2StructMapping);
+
+bool printDICTTABINFO(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+// const DictTabInfo * const sig = (DictTabInfo *) theData;
+
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ return true;
+}
+
+void
+DictTabInfo::Table::init(){
+ memset(TableName, 0, sizeof(TableName));//TableName[0] = 0;
+ TableId = ~0;
+ SecondTableId = ~0;
+ memset(PrimaryTable, 0, sizeof(PrimaryTable));//PrimaryTable[0] = 0; // Only used when "index"
+ PrimaryTableId = RNIL;
+ TableLoggedFlag = 1;
+ NoOfKeyAttr = 0;
+ NoOfAttributes = 0;
+ NoOfNullable = 0;
+ NoOfVariable = 0;
+ TableKValue = 6;
+ MinLoadFactor = 78;
+ MaxLoadFactor = 80;
+ KeyLength = 0;
+ FragmentType = DictTabInfo::AllNodesSmallTable;
+ TableStorage = 0;
+ ScanOptimised = 0;
+ FragmentKeyType = DictTabInfo::PrimaryKey;
+ TableType = DictTabInfo::UndefTableType;
+ TableVersion = 0;
+ IndexState = ~0;
+ InsertTriggerId = RNIL;
+ UpdateTriggerId = RNIL;
+ DeleteTriggerId = RNIL;
+ CustomTriggerId = RNIL;
+ FrmLen = 0;
+ memset(FrmData, 0, sizeof(FrmData));
+}
+
+void
+DictTabInfo::Attribute::init(){
+ memset(AttributeName, 0, sizeof(AttributeName));//AttributeName[0] = 0;
+ AttributeId = 0;
+ AttributeType = DictTabInfo::UnSignedType;
+ AttributeSize = DictTabInfo::a32Bit;
+ AttributeArraySize = 1;
+ AttributeKeyFlag = 0;
+ AttributeStorage = 1;
+ AttributeNullableFlag = 0;
+ AttributeDGroup = 0;
+ AttributeDKey = 0;
+ AttributeStoredInd = 1;
+ AttributeGroup = 0;
+ AttributeExtType = 0,
+ AttributeExtPrecision = 0,
+ AttributeExtScale = 0,
+ AttributeExtLength = 0,
+ AttributeAutoIncrement = false;
+ memset(AttributeDefaultValue, 0, sizeof(AttributeDefaultValue));//AttributeDefaultValue[0] = 0;
+};
diff --git a/ndb/src/common/debugger/signaldata/DihContinueB.cpp b/ndb/src/common/debugger/signaldata/DihContinueB.cpp
new file mode 100644
index 00000000000..94453e76d72
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/DihContinueB.cpp
@@ -0,0 +1,217 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <signaldata/DihContinueB.hpp>
+
+bool
+printCONTINUEB_DBDIH(FILE * output, const Uint32 * theData, Uint32 len){
+
+ switch (theData[0]) {
+ case DihContinueB::ZPACK_TABLE_INTO_PAGES:
+ fprintf(output, " Pack Table Into Pages: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZPACK_FRAG_INTO_PAGES:
+ fprintf(output, " Pack Frag Into Pages: Table: %d Fragment: %d PageIndex: %d WordIndex: %d\n",
+ theData[1], theData[2], theData[3], theData[4]);
+ return true;
+ break;
+ case DihContinueB::ZREAD_PAGES_INTO_TABLE:
+ fprintf(output, " Read Pages Into Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZREAD_PAGES_INTO_FRAG:
+ fprintf(output, " Read Pages Into Frag: Table: %d Fragment: %d PageIndex: %d WordIndex: %d\n",
+ theData[1], theData[2], theData[3], theData[4]);
+ return true;
+ break;
+#if 0
+ case DihContinueB::ZREAD_TAB_DESCRIPTION:
+ fprintf(output, " Read Table description: %d\n", theData[1]);
+ return true;
+ break;
+#endif
+ case DihContinueB::ZCOPY_TABLE:
+ fprintf(output, " Copy Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZCOPY_TABLE_NODE:
+ fprintf(output, " Copy table node: TableId: %d NodeId: %d\n",
+ theData[1], theData[2]);
+ fprintf(output, "PageIndex: %d WordIndex: %d NoOfWords: %d\n",
+ theData[3], theData[4], theData[5]);
+ return true;
+ break;
+ case DihContinueB::ZSTART_FRAGMENT:
+ fprintf(output, " Start fragment: Table: %d Fragment: %d\n",
+ theData[1], theData[2]);
+ return true;
+ break;
+ case DihContinueB::ZCOMPLETE_RESTART:
+ fprintf(output, "Complete Restart\n");
+ return true;
+ break;
+ case DihContinueB::ZREAD_TABLE_FROM_PAGES:
+ fprintf(output, " Read Table From Pages: Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZSR_PHASE2_READ_TABLE:
+ fprintf(output, " Phase 2 Read Table: Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZCHECK_TC_COUNTER:
+ fprintf(output, " Check Tc Counter from place %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZCALCULATE_KEEP_GCI:
+ fprintf(output, " Calc Keep GCI: Table: %d Fragment: %d\n",
+ theData[1], theData[2]);
+ return true;
+ break;
+ case DihContinueB::ZSTORE_NEW_LCP_ID:
+ fprintf(output, " Store New LCP Id\n");
+ return true;
+ break;
+ case DihContinueB::ZTABLE_UPDATE:
+ fprintf(output, " Table Update: Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZCHECK_LCP_COMPLETED:
+ fprintf(output, " Check LCP Completed: TableId %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZINIT_LCP:
+ fprintf(output, " Init LCP: Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZADD_TABLE_MASTER_PAGES:
+ fprintf(output, " Add Table Master Pages: Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZDIH_ADD_TABLE_MASTER:
+ fprintf(output, " Dih Add Table Master: Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZADD_TABLE_SLAVE_PAGES:
+ fprintf(output, " Add Table Slave Pages: Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZDIH_ADD_TABLE_SLAVE:
+ fprintf(output, " Add Table Slave: Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZSTART_GCP:
+ fprintf(output, " Start GCP\n");
+ return true;
+ break;
+ case DihContinueB::ZCOPY_GCI:
+ fprintf(output, " Copy GCI\n");
+ return true;
+ break;
+ case DihContinueB::ZEMPTY_VERIFY_QUEUE:
+ fprintf(output, " Empty Verify Queue\n");
+ return true;
+ break;
+ case DihContinueB::ZCHECK_GCP_STOP:
+ fprintf(output, " Check GCP Stop\n");
+ if (len == 6){
+ fprintf(output, "coldGcpStatus = %d\n", theData[1]);
+ fprintf(output, "cgcpStatus = %d\n", theData[2]);
+ fprintf(output, "coldGcpId = %d\n", theData[3]);
+ fprintf(output, "cnewgcp = %d\n", theData[4]);
+ fprintf(output, "cgcpSameCounter = %d\n", theData[5]);
+ }
+ return true;
+ break;
+ case DihContinueB::ZREMOVE_NODE_FROM_TABLE:
+ fprintf(output, " Remove Node From Table: Node: %d Table: %d\n",
+ theData[1], theData[2]);
+ return true;
+ break;
+ case DihContinueB::ZCOPY_NODE:
+ fprintf(output, " Copy Node: Table: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZSTART_TAKE_OVER:
+ fprintf(output, " Start Take Over: TakeOverPtr: %d, startNode: %d, toNode: %d\n",
+ theData[1], theData[2], theData[3]);
+ return true;
+ break;
+ case DihContinueB::ZCHECK_START_TAKE_OVER:
+ fprintf(output, " Check Start Take Over\n");
+ return true;
+ break;
+ case DihContinueB::ZTO_START_COPY_FRAG:
+ fprintf(output, " To Start Copy Frag: TakeOverPtr: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZINVALIDATE_NODE_LCP:
+ fprintf(output, " Invalide LCP: NodeId: %d TableId %d\n",
+ theData[1], theData[2]);
+ return true;
+ break;
+ case DihContinueB::ZINITIALISE_RECORDS:
+ fprintf(output, " Initialise Records: tdata0: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::ZSTART_PERMREQ_AGAIN:
+ fprintf(output, " START_PERMREQ again for node: %d\n", theData[1]);
+ return true;
+ break;
+ case DihContinueB::SwitchReplica:
+ fprintf(output, " NodeId = %d TableId = %d FragNo = %d\n",
+ theData[1], theData[2], theData[3]);
+ return true;
+ break;
+ case DihContinueB::ZSEND_START_TO:
+ fprintf(output, " Send Start Take Over: TakeOverPtr: %d, startNode: %d, toNode: %d\n",
+ theData[1], theData[2], theData[3]);
+ return true;
+ break;
+ case DihContinueB::ZSEND_UPDATE_TO:
+ fprintf(output, " Send Update Take Over: TakeOverPtr: %d, startNode: %d, toNode: %d\n",
+ theData[1], theData[2], theData[3]);
+ return true;
+ break;
+ case DihContinueB::ZSEND_END_TO:
+ fprintf(output, " Send End Take Over: TakeOverPtr: %d, startNode: %d, toNode: %d\n",
+ theData[1], theData[2], theData[3]);
+ return true;
+ break;
+ case DihContinueB::ZSEND_ADD_FRAG:
+ fprintf(output, " Send Add Fragment: TakeOverPtr: %d, startNode: %d, toNode: %d\n",
+ theData[1], theData[2], theData[3]);
+ return true;
+ break;
+ case DihContinueB::ZSEND_CREATE_FRAG:
+ fprintf(output, " Send Create Fragment: TakeOverPtr: %d, storedType: %d, start Gci: %d, startNode: %d, toNode: %d\n",
+ theData[1], theData[2], theData[3], theData[4], theData[5]);
+ return true;
+ break;
+ case DihContinueB::WAIT_DROP_TAB_WRITING_TO_FILE:
+ fprintf(output, " Wait drop tab writing to file TableId: %d\n", theData[1]);
+ return true;
+ case DihContinueB::CHECK_WAIT_DROP_TAB_FAILED_LQH:
+ fprintf(output, " Wait drop tab FailedNodeId: %d TableId: %d\n",
+ theData[1], theData[2]);
+ return true;
+ default:
+ fprintf(output, " Default system error lab...\n");
+ break;
+ }//switch
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/DihSwitchReplicaReq.cpp b/ndb/src/common/debugger/signaldata/DihSwitchReplicaReq.cpp
new file mode 100644
index 00000000000..2e4318f4033
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/DihSwitchReplicaReq.cpp
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <signaldata/DihSwitchReplicaReq.hpp>
+
+bool
+printDIH_SWITCH_REPLICA_REQ(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 recBlockNo){
+
+ DihSwitchReplicaReq * req = (DihSwitchReplicaReq *)&theData[0];
+
+ const Uint32 requestInfo = req->requestInfo;
+
+ switch(DihSwitchReplicaReq::getRequestType(requestInfo)){
+ case DihSwitchReplicaReq::RemoveNodeAsPrimary:{
+ fprintf(output, " RemoveNodeAsPrimary: Node=%d", req->nodeId);
+ if(DihSwitchReplicaReq::getAllTables(requestInfo))
+ fprintf(output, " All Tables");
+ else
+ fprintf(output, " TableId=%d", req->tableId);
+
+ if(DihSwitchReplicaReq::getDistribute(requestInfo))
+ fprintf(output, " Distribute");
+ fprintf(output, "\n");
+ return true;
+ }
+ break;
+ default:
+ fprintf(output, " Unknown request type:\n");
+ }
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/DisconnectRep.cpp b/ndb/src/common/debugger/signaldata/DisconnectRep.cpp
new file mode 100644
index 00000000000..3a73747a978
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/DisconnectRep.cpp
@@ -0,0 +1,30 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <signaldata/DisconnectRep.hpp>
+
+bool
+printDISCONNECT_REP(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const DisconnectRep * const sig = (DisconnectRep *) theData;
+
+ fprintf(output, " NodeId: %d, ErrorCode: %d\n",
+ sig->nodeId, sig->err);
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/DropIndx.cpp b/ndb/src/common/debugger/signaldata/DropIndx.cpp
new file mode 100644
index 00000000000..0d59a981a18
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/DropIndx.cpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/DropIndx.hpp>
+
+bool printDROP_INDX_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const DropIndxReq * const sig = (DropIndxReq *) theData;
+
+ return false;
+}
+
+bool printDROP_INDX_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const DropIndxConf * const sig = (DropIndxConf *) theData;
+
+ return false;
+}
+
+bool printDROP_INDX_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+// const DropIndxRef * const sig = (DropIndxRef *) theData;
+
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/DropTab.cpp b/ndb/src/common/debugger/signaldata/DropTab.cpp
new file mode 100644
index 00000000000..83c95b0e344
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/DropTab.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/DropTab.hpp>
+
+bool
+printDROP_TAB_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const DropTabReq * const sig = (DropTabReq *) theData;
+
+ fprintf(output,
+ " senderRef: %x senderData: %d TableId: %d requestType: %d\n",
+ sig->senderRef, sig->senderData, sig->tableId, sig->requestType);
+ return true;
+}
+
+bool printDROP_TAB_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const DropTabConf * const sig = (DropTabConf *) theData;
+
+ fprintf(output,
+ " senderRef: %x senderData: %d TableId: %d\n",
+ sig->senderRef, sig->senderData, sig->tableId);
+
+ return true;
+}
+
+bool printDROP_TAB_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const DropTabRef * const sig = (DropTabRef *) theData;
+
+ fprintf(output,
+ " senderRef: %x senderData: %d TableId: %d errorCode: %d\n",
+ sig->senderRef, sig->senderData, sig->tableId, sig->errorCode);
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/DropTrig.cpp b/ndb/src/common/debugger/signaldata/DropTrig.cpp
new file mode 100644
index 00000000000..54e8734439f
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/DropTrig.cpp
@@ -0,0 +1,89 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/DropTrig.hpp>
+
+bool printDROP_TRIG_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const DropTrigReq * const sig = (DropTrigReq *) theData;
+
+ //char triggerName[MAX_TAB_NAME_SIZE];
+ //char triggerType[32];
+ //char triggerActionTime[32];
+ //char triggerEvent[32];
+
+ //sig->getTriggerName((char *) &triggerName);
+ //switch(sig->getTriggerType()) {
+ //case(TriggerType::SECONDARY_INDEX):
+ //strcpy(triggerType, "SECONDARY_INDEX");
+ //break;
+ //case(TriggerType::SUBSCRIPTION):
+ //strcpy(triggerType, "SUBSCRIPTION");
+ //break;
+ //default:
+ //strcpy(triggerType, "UNSUPPORTED");
+ //}
+ //strcpy(triggerActionTime,
+ //(sig->getTriggerActionTime() == TriggerActionTime::BEFORE)?
+ //"BEFORE":"AFTER");
+ //switch(sig->getTriggerEvent()) {
+ //case (TriggerEvent::TE_INSERT):
+ //strcpy(triggerEvent, "INSERT");
+ //break;
+ //case(TriggerEvent::TE_DELETE):
+ //strcpy(triggerEvent, "DELETE");
+ //break;
+ //case(TriggerEvent::TE_UPDATE):
+ //strcpy(triggerEvent, "UPDATE");
+ //break;
+ //}
+
+ fprintf(output, "User: %u, ", sig->getUserRef());
+ //fprintf(output, "Trigger name: \"%s\"\n", triggerName);
+ //fprintf(output, "Type: %s, ", triggerType);
+ //fprintf(output, "Action: %s, ", triggerActionTime);
+ //fprintf(output, "Event: %s, ", triggerEvent);
+ fprintf(output, "Trigger id: %u, ", sig->getTriggerId());
+ fprintf(output, "Table id: %u, ", sig->getTableId());
+ fprintf(output, "\n");
+
+ return false;
+}
+
+bool printDROP_TRIG_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const DropTrigConf * const sig = (DropTrigConf *) theData;
+
+ fprintf(output, "User: %u, ", sig->getUserRef());
+ fprintf(output, "Trigger id: %u, ", sig->getTriggerId());
+ fprintf(output, "Table id: %u, ", sig->getTableId());
+ fprintf(output, "\n");
+
+ return false;
+}
+
+bool printDROP_TRIG_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const DropTrigRef * const sig = (DropTrigRef *) theData;
+
+ fprintf(output, "User: %u, ", sig->getUserRef());
+ fprintf(output, "Trigger id: %u, ", sig->getTriggerId());
+ fprintf(output, "Table id: %u, ", sig->getTableId());
+ fprintf(output, "Error code: %u, ", sig->getErrorCode());
+ fprintf(output, "\n");
+
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/FailRep.cpp b/ndb/src/common/debugger/signaldata/FailRep.cpp
new file mode 100644
index 00000000000..d70912fe8c7
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/FailRep.cpp
@@ -0,0 +1,31 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <signaldata/FailRep.hpp>
+
+bool
+printFAIL_REP(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const FailRep * const sig = (FailRep *) theData;
+
+ fprintf(output, " FailedNode: %d, FailCause: %d\n",
+ sig->failNodeId, sig->failCause);
+
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/FireTrigOrd.cpp b/ndb/src/common/debugger/signaldata/FireTrigOrd.cpp
new file mode 100644
index 00000000000..d86aa2e06de
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/FireTrigOrd.cpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/FireTrigOrd.hpp>
+#include <RefConvert.hpp>
+
+static
+const char *
+trigEvent(Uint32 i){
+ switch(i){
+ case TriggerEvent::TE_INSERT:
+ return "insert";
+ break;
+ case TriggerEvent::TE_UPDATE:
+ return "update";
+ break;
+ case TriggerEvent::TE_DELETE:
+ return "delete";
+ break;
+ }
+ return "UNKNOWN";
+}
+
+bool
+printFIRE_TRIG_ORD(FILE * output, const Uint32 * theData, Uint32 len,
+ Uint16 receiverBlockNo)
+{
+ const FireTrigOrd * const sig = (FireTrigOrd *) theData;
+
+ fprintf(output, " TriggerId: %d TriggerEvent: %s\n",
+ sig->getTriggerId(),
+ trigEvent(sig->getTriggerEvent()));
+ fprintf(output, " UserRef: (%d, %d) User data: %x\n",
+ refToNode(sig->getUserRef()),
+ refToBlock(sig->getUserRef()),
+ sig->getConnectionPtr());
+ fprintf(output, " Signal: PK=%d BEFORE=%d AFTER=%d\n",
+ sig->getNoOfPrimaryKeyWords(),
+ sig->getNoOfBeforeValueWords(),
+ sig->getNoOfAfterValueWords());
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/FsAppendReq.cpp b/ndb/src/common/debugger/signaldata/FsAppendReq.cpp
new file mode 100644
index 00000000000..6e443ffe5fc
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/FsAppendReq.cpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <signaldata/FsAppendReq.hpp>
+
+bool
+printFSAPPENDREQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo){
+
+ bool ret = true;
+
+ const FsAppendReq * const sig = (FsAppendReq *) theData;
+
+ fprintf(output, " FilePointer: %d\n", sig->filePointer);
+ fprintf(output, " UserReference: H\'%.8x, UserPointer: H\'%.8x\n",
+ sig->userReference, sig->userPointer);
+
+ fprintf(output, " varIndex: %d offset: %d size: %d\n",
+ sig->varIndex,
+ sig->offset,
+ sig->size);
+ return ret;
+}
diff --git a/ndb/src/common/debugger/signaldata/FsCloseReq.cpp b/ndb/src/common/debugger/signaldata/FsCloseReq.cpp
new file mode 100644
index 00000000000..143250b7db1
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/FsCloseReq.cpp
@@ -0,0 +1,40 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <signaldata/FsCloseReq.hpp>
+
+bool
+printFSCLOSEREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const FsCloseReq * const sig = (FsCloseReq *) theData;
+
+ fprintf(output, " UserPointer: %d\n",
+ sig->userPointer);
+ fprintf(output, " FilePointer: %d\n",
+ sig->filePointer);
+ fprintf(output, " UserReference: H\'%.8x\n",
+ sig->userReference);
+
+ fprintf(output, " Flags: H\'%.8x, ", sig->fileFlag);
+ if (sig->getRemoveFileFlag(sig->fileFlag) == true)
+ fprintf(output, "Remove file");
+ else
+ fprintf(output, "Don't remove file");
+ fprintf(output, "\n");
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/FsConf.cpp b/ndb/src/common/debugger/signaldata/FsConf.cpp
new file mode 100644
index 00000000000..f0ab57aadcf
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/FsConf.cpp
@@ -0,0 +1,33 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <signaldata/FsConf.hpp>
+
+bool
+printFSCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const FsConf * const sig = (FsConf *) theData;
+
+ fprintf(output, " UserPointer: %d\n", sig->userPointer);
+
+ if (len > 1){
+ // Only valid if this is a FSOPENCONF
+ fprintf(output, " FilePointer: %d\n", sig->filePointer);
+ }
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/FsOpenReq.cpp b/ndb/src/common/debugger/signaldata/FsOpenReq.cpp
new file mode 100644
index 00000000000..31d351a8a84
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/FsOpenReq.cpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <signaldata/FsOpenReq.hpp>
+
+bool
+printFSOPENREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const FsOpenReq * const sig = (FsOpenReq *) theData;
+
+
+ fprintf(output, " UserReference: H\'%.8x, userPointer: H\'%.8x\n",
+ sig->userReference, sig->userPointer);
+ fprintf(output, " FileNumber[1-4]: H\'%.8x H\'%.8x H\'%.8x H\'%.8x\n",
+ sig->fileNumber[0], sig->fileNumber[1], sig->fileNumber[2], sig->fileNumber[3]);
+ fprintf(output, " FileFlags: H\'%.8x ",
+ sig->fileFlags);
+
+ // File open mode must be one of ReadOnly, WriteOnly or ReadWrite
+ const Uint32 flags = sig->fileFlags;
+ switch(flags & 3){
+ case FsOpenReq::OM_READONLY:
+ fprintf(output, "Open read only");
+ break;
+ case FsOpenReq::OM_WRITEONLY:
+ fprintf(output, "Open write only");
+ break;
+ case FsOpenReq::OM_READWRITE:
+ fprintf(output, "Open read and write");
+ break;
+ default:
+ fprintf(output, "Open mode unknown!");
+ }
+
+ if (flags & FsOpenReq::OM_CREATE)
+ fprintf(output, ", Create new file");
+ if (flags & FsOpenReq::OM_TRUNCATE)
+ fprintf(output, ", Truncate existing file");
+ if (flags & FsOpenReq::OM_APPEND)
+ fprintf(output, ", Append");
+
+ fprintf(output, "\n");
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/FsReadWriteReq.cpp b/ndb/src/common/debugger/signaldata/FsReadWriteReq.cpp
new file mode 100644
index 00000000000..ad9cb623c17
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/FsReadWriteReq.cpp
@@ -0,0 +1,85 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <signaldata/FsReadWriteReq.hpp>
+
+bool
+printFSREADWRITEREQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo){
+
+ bool ret = true;
+
+ const FsReadWriteReq * const sig = (FsReadWriteReq *) theData;
+
+ fprintf(output, " UserPointer: %d\n", sig->userPointer);
+ fprintf(output, " FilePointer: %d\n", sig->filePointer);
+ fprintf(output, " UserReference: H\'%.8x", sig->userReference);
+
+ fprintf(output, " Operation flag: H\'%.8x (", sig->operationFlag);
+ if (sig->getSyncFlag(sig->operationFlag) == true)
+ fprintf(output, "Sync,");
+ else
+ fprintf(output, "No sync,");
+
+ fprintf(output, " Format=");
+ switch(sig->getFormatFlag(sig->operationFlag)){
+ case FsReadWriteReq::fsFormatListOfPairs:
+ fprintf(output, "List of pairs)\n");
+ break;
+ case FsReadWriteReq::fsFormatArrayOfPages:
+ fprintf(output, "Array of pages)\n");
+ break;
+ case FsReadWriteReq::fsFormatListOfMemPages:
+ fprintf(output, "List of mem pages)\n");
+ break;
+ default:
+ fprintf(output, "fsFormatMax not handled\n");
+ ret = false;
+ break;
+ }
+
+ fprintf(output, " varIndex: %d\n",
+ sig->varIndex);
+ fprintf(output, " numberOfPages: %d\n",
+ sig->numberOfPages);
+ fprintf(output, " pageData: ");
+
+
+ switch(sig->getFormatFlag(sig->operationFlag)){
+ case FsReadWriteReq::fsFormatListOfPairs:
+ for (unsigned int i = 0; i < sig->numberOfPages*2; i += 2){
+ fprintf(output, " H\'%.8x, H\'%.8x\n", sig->data.pageData[i],
+ sig->data.pageData[i + 1]);
+ }
+ break;
+ case FsReadWriteReq::fsFormatArrayOfPages:
+ fprintf(output, " H\'%.8x, H\'%.8x\n", sig->data.pageData[0],
+ sig->data.pageData[1]);
+ break;
+ case FsReadWriteReq::fsFormatListOfMemPages:
+ for (unsigned int i = 0; i < (sig->numberOfPages + 1); i++){
+ fprintf(output, " H\'%.8x, ", sig->data.pageData[i]);
+ }
+ break;
+ default:
+ fprintf(output, "Impossible event\n");
+ }
+
+ fprintf(output, "\n");
+ return ret;
+}
diff --git a/ndb/src/common/debugger/signaldata/FsRef.cpp b/ndb/src/common/debugger/signaldata/FsRef.cpp
new file mode 100644
index 00000000000..ccf3d6da9c8
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/FsRef.cpp
@@ -0,0 +1,75 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <signaldata/FsRef.hpp>
+
+bool
+printFSREF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo){
+
+ bool ret = true;
+
+ const FsRef * const sig = (FsRef *) theData;
+
+ fprintf(output, " UserPointer: %d\n",
+ sig->userPointer);
+
+ fprintf(output, " ErrorCode: %d, ", sig->errorCode);
+ switch (sig->getErrorCode(sig->errorCode)){
+ case FsRef::fsErrNone:
+ fprintf(output, "No error");
+ break;
+ case FsRef::fsErrHardwareFailed:
+ fprintf(output, "Hardware failure!");
+ break;
+ case FsRef::fsErrUserError:
+ fprintf(output, "User error!");
+ break;
+ case FsRef::fsErrEnvironmentError:
+ fprintf(output, "Environment error!");
+ break;
+ case FsRef::fsErrTemporaryNotAccessible:
+ fprintf(output, "Temporary not accesible!");
+ break;
+ case FsRef::fsErrNoSpaceLeftOnDevice:
+ fprintf(output, "No space left on device!");
+ break;
+ case FsRef::fsErrPermissionDenied:
+ fprintf(output, "Permission denied!");
+ break;
+ case FsRef::fsErrInvalidParameters:
+ fprintf(output, "Invalid parameters!");
+ break;
+ case FsRef::fsErrNoMoreResources:
+ fprintf(output, "No more resources!");
+ break;
+ case FsRef::fsErrFileDoesNotExist:
+ fprintf(output, "File does not exist!");
+ break;
+
+ case FsRef::fsErrUnknown:
+ default:
+ fprintf(output, "Unknown!");
+ ret = false;
+ break;
+ }
+ fprintf(output, "\n");
+ fprintf(output, " OS ErrorCode: %d \n", sig->osErrorCode);
+
+ return ret;
+}
diff --git a/ndb/src/common/debugger/signaldata/GCPSave.cpp b/ndb/src/common/debugger/signaldata/GCPSave.cpp
new file mode 100644
index 00000000000..7566f004bfd
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/GCPSave.cpp
@@ -0,0 +1,78 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <signaldata/GCPSave.hpp>
+#include <RefConvert.hpp>
+
+bool
+printGCPSaveReq(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo){
+
+ GCPSaveReq * sr = (GCPSaveReq*)theData;
+
+ fprintf(output, " dihBlockRef = (%d, %d) dihPtr = %d gci = %d\n",
+ refToBlock(sr->dihBlockRef), refToNode(sr->dihBlockRef),
+ sr->dihPtr, sr->gci);
+
+ return true;
+}
+
+bool
+printGCPSaveRef(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo){
+
+ GCPSaveRef * sr = (GCPSaveRef*)theData;
+
+ fprintf(output, " nodeId = %d dihPtr = %d gci = %d reason: ",
+ sr->nodeId,
+ sr->dihPtr, sr->gci);
+
+ switch(sr->errorCode){
+ case GCPSaveRef::NodeShutdownInProgress:
+ fprintf(output, "NodeShutdownInProgress\n");
+ break;
+ case GCPSaveRef::FakedSignalDueToNodeFailure:
+ fprintf(output, "FakedSignalDueToNodeFailure\n");
+ break;
+ default:
+ fprintf(output, "Unknown reason: %d\n", sr->errorCode);
+ return false;
+ }
+
+ return true;
+}
+
+bool
+printGCPSaveConf(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo){
+
+ GCPSaveConf * sr = (GCPSaveConf*)theData;
+
+ fprintf(output, " nodeId = %d dihPtr = %d gci = %d\n",
+ sr->nodeId,
+ sr->dihPtr, sr->gci);
+
+ return true;
+}
+
+
diff --git a/ndb/src/common/debugger/signaldata/IndxAttrInfo.cpp b/ndb/src/common/debugger/signaldata/IndxAttrInfo.cpp
new file mode 100755
index 00000000000..2ef5feaada7
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/IndxAttrInfo.cpp
@@ -0,0 +1,31 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/IndxAttrInfo.hpp>
+
+bool
+printINDXATTRINFO(FILE * output, const Uint32 * theData, Uint32 len,
+ Uint16 receiverBlockNo)
+{
+// const IndxAttrInfo * const sig = (IndxAttrInfo *) theData;
+
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, " H\'%.8x", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/IndxKeyInfo.cpp b/ndb/src/common/debugger/signaldata/IndxKeyInfo.cpp
new file mode 100755
index 00000000000..6fe5567188d
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/IndxKeyInfo.cpp
@@ -0,0 +1,31 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/IndxKeyInfo.hpp>
+
+bool
+printINDXKEYINFO(FILE * output, const Uint32 * theData, Uint32 len,
+ Uint16 receiverBlockNo)
+{
+// const IndxKeyInfo * const sig = (IndxKeyInfo *) theData;
+
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, " H\'%.8x", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/LCP.cpp b/ndb/src/common/debugger/signaldata/LCP.cpp
new file mode 100644
index 00000000000..825659d13b3
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/LCP.cpp
@@ -0,0 +1,88 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <RefConvert.hpp>
+#include <signaldata/LCP.hpp>
+#include <DebuggerNames.hpp>
+
+bool
+printSTART_LCP_REQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo){
+
+ const StartLcpReq * const sig = (StartLcpReq *) theData;
+
+ char buf1[sig->participatingDIH.TextLength+1], buf2[sig->participatingLQH.TextLength+1];
+ fprintf(output,
+ " Sender: %d LcpId: %d\n"
+ " ParticipatingDIH = %s\n"
+ " ParticipatingLQH = %s\n",
+ refToNode(sig->senderRef), sig->lcpId,
+ sig->participatingDIH.getText(buf1),
+ sig->participatingLQH.getText(buf2));
+
+ return true;
+}
+
+bool
+printSTART_LCP_CONF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo){
+
+ const StartLcpConf * const sig = (StartLcpConf *) theData;
+
+ fprintf(output, " Sender: %d LcpId: %d\n",
+ refToNode(sig->senderRef), sig->lcpId);
+
+ return true;
+}
+
+bool
+printLCP_FRAG_ORD(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo){
+
+ const LcpFragOrd * const sig = (LcpFragOrd *) theData;
+
+ fprintf(output, " LcpId: %d LcpNo: %d Table: %d Fragment: %d\n",
+ sig->lcpId, sig->lcpNo, sig->tableId, sig->fragmentId);
+
+ fprintf(output, " KeepGCI: %d LastFragmentFlag: %d\n",
+ sig->keepGci, sig->lastFragmentFlag);
+ return true;
+}
+
+bool
+printLCP_FRAG_REP(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo){
+
+ const LcpFragRep * const sig = (LcpFragRep *) theData;
+
+ fprintf(output, " LcpId: %d LcpNo: %d NodeId: %d Table: %d Fragment: %d\n",
+ sig->lcpId, sig->lcpNo, sig->nodeId, sig->tableId, sig->fragId);
+ fprintf(output, " Max GCI Started: %d Max GCI Completed: %d\n",
+ sig->maxGciStarted, sig->maxGciCompleted);
+ return true;
+}
+
+bool
+printLCP_COMPLETE_REP(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo){
+
+ const LcpCompleteRep * const sig = (LcpCompleteRep *) theData;
+
+ fprintf(output, " LcpId: %d NodeId: %d Block: %s\n",
+ sig->lcpId, sig->nodeId, getBlockName(sig->blockNo));
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/LqhFrag.cpp b/ndb/src/common/debugger/signaldata/LqhFrag.cpp
new file mode 100644
index 00000000000..6d727959a67
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/LqhFrag.cpp
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <signaldata/LqhFrag.hpp>
+
+bool
+printLQH_FRAG_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 recB){
+ LqhFragReq* sig = (LqhFragReq*)theData;
+
+ fprintf(output, " senderData: %d senderRef: %x",
+ sig->senderData, sig->senderRef);
+ fprintf(output, " tableId: %d fragmentId: %d tableType: %d",
+ sig->tableId, sig->fragmentId, sig->tableType);
+ if (sig->primaryTableId == RNIL)
+ fprintf(output, " primaryTableId: RNIL\n");
+ else
+ fprintf(output, " primaryTableId: %d\n", sig->primaryTableId);
+ fprintf(output, " localKeyLength: %d maxLoadFactor: %d minLoadFactor: %d\n",
+ sig->localKeyLength, sig->maxLoadFactor, sig->minLoadFactor);
+ fprintf(output, " kValue: %d lh3DistrBits: %d lh3PageBits: %d\n",
+ sig->kValue, sig->lh3DistrBits, sig->lh3PageBits);
+
+ fprintf(output, " noOfAttributes: %d noOfNullAttributes: %d keyLength: %d\n",
+ sig->noOfAttributes, sig->noOfNullAttributes, sig->keyLength);
+
+ fprintf(output, " noOfPagesToPreAllocate: %d schemaVersion: %d nextLCP: %d\n",
+ sig->noOfPagesToPreAllocate, sig->schemaVersion, sig->nextLCP);
+
+ return true;
+}
+bool
+printLQH_FRAG_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 rec){
+ LqhFragConf* sig = (LqhFragConf*)theData;
+
+ fprintf(output, " senderData: %d lqhFragPtr: %d\n",
+ sig->senderData, sig->lqhFragPtr);
+ return true;
+}
+
+bool
+printLQH_FRAG_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 rec){
+ LqhFragRef* sig = (LqhFragRef*)theData;
+
+ fprintf(output, " senderData: %d errorCode: %d\n",
+ sig->senderData, sig->errorCode);
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/LqhKey.cpp b/ndb/src/common/debugger/signaldata/LqhKey.cpp
new file mode 100644
index 00000000000..2796437fd8b
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/LqhKey.cpp
@@ -0,0 +1,161 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/LqhKey.hpp>
+
+bool
+printLQHKEYREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const LqhKeyReq * const sig = (LqhKeyReq *) theData;
+
+ fprintf(output,
+ " ClientPtr = H\'%.8x hashValue = H\'%.8x tcBlockRef = H\'%.8x\n"
+ " transId1 = H\'%.8x transId2 = H\'%.8x savePointId = H\'%.8x\n",
+ sig->clientConnectPtr, // DATA 0
+ sig->hashValue, // DATA 2
+ sig->tcBlockref, // DATA 4
+ sig->transId1, // DATA 7
+ sig->transId2, // DATA 8
+ sig->savePointId // DATA 9
+ );
+
+ const Uint32 reqInfo = sig->requestInfo;
+ const Uint32 attrLen = sig->attrLen;
+
+ fprintf(output,
+ " Op: %d Lock: %d Flags: ",
+ LqhKeyReq::getOperation(reqInfo),
+ LqhKeyReq::getLockType(reqInfo));
+ if(LqhKeyReq::getSimpleFlag(reqInfo))
+ fprintf(output, "Simple ");
+ if(LqhKeyReq::getDirtyFlag(reqInfo))
+ fprintf(output, "Dirty ");
+ if(LqhKeyReq::getInterpretedFlag(reqInfo))
+ fprintf(output, "Interpreted ");
+ if(LqhKeyReq::getScanTakeOverFlag(attrLen))
+ fprintf(output, "ScanTakeOver ");
+ if(LqhKeyReq::getMarkerFlag(reqInfo))
+ fprintf(output, "CommitAckMarker ");
+
+ fprintf(output, "ScanInfo/noFiredTriggers: H\'%x\n", sig->scanInfo);
+
+ fprintf(output,
+ " AttrLen: %d (%d in this) KeyLen: %d TableId: %d SchemaVer: %d\n",
+ LqhKeyReq::getAttrLen(attrLen),
+ LqhKeyReq::getAIInLqhKeyReq(reqInfo),
+ LqhKeyReq::getKeyLen(reqInfo),
+ LqhKeyReq::getTableId(sig->tableSchemaVersion),
+ LqhKeyReq::getSchemaVersion(sig->tableSchemaVersion));
+
+ fprintf(output,
+ " FragId: %d ReplicaNo: %d LastReplica: %d NextNodeId: %d\n",
+ LqhKeyReq::getFragmentId(sig->fragmentData),
+ LqhKeyReq::getSeqNoReplica(reqInfo),
+ LqhKeyReq::getLastReplicaNo(reqInfo),
+ LqhKeyReq::getNextReplicaNodeId(sig->fragmentData));
+
+ bool printed = false;
+ Uint32 nextPos = LqhKeyReq::getApplicationAddressFlag(reqInfo) << 1;
+ if(nextPos != 0){
+ fprintf(output,
+ " ApiRef: H\'%.8x ApiOpRef: H\'%.8x",
+ sig->variableData[0],
+ sig->variableData[1]);
+ printed = true;
+ }
+
+ if(LqhKeyReq::getSameClientAndTcFlag(reqInfo)){
+ fprintf(output, " TcOpRec: H\'%.8x", sig->variableData[nextPos]);
+ nextPos++;
+ printed = true;
+ }
+
+ Uint32 tmp = LqhKeyReq::getLastReplicaNo(reqInfo) -
+ LqhKeyReq::getSeqNoReplica(reqInfo);
+ if(tmp > 1){
+ NodeId node2 = sig->variableData[nextPos] & 0xffff;
+ NodeId node3 = sig->variableData[nextPos] >> 16;
+ fprintf(output, " NextNodeId2: %d NextNodeId3: %d",
+ node2, node3);
+ nextPos ++;
+ printed = true;
+ }
+ if(printed)
+ fprintf(output, "\n");
+
+ printed = false;
+ if(LqhKeyReq::getStoredProcFlag(attrLen)){
+ fprintf(output, " StoredProcId: %d", sig->variableData[nextPos]);
+ nextPos++;
+ printed = true;
+ }
+
+ if(LqhKeyReq::getReturnedReadLenAIFlag(reqInfo)){
+ fprintf(output, " ReturnedReadLenAI: %d",
+ sig->variableData[nextPos]);
+ nextPos++;
+ printed = true;
+ }
+
+ const UintR keyLen = LqhKeyReq::getKeyLen(reqInfo);
+ if(keyLen > 0){
+ fprintf(output, " KeyInfo: ");
+ for(UintR i = 0; i<keyLen && i<4; i++, nextPos++)
+ fprintf(output, "H\'%.8x ", sig->variableData[nextPos]);
+ fprintf(output, "\n");
+ }
+
+ if(!LqhKeyReq::getInterpretedFlag(reqInfo)){
+ fprintf(output, " AttrInfo: ");
+ for(int i = 0; i<LqhKeyReq::getAIInLqhKeyReq(reqInfo); i++, nextPos++)
+ fprintf(output, "H\'%.8x ", sig->variableData[nextPos]);
+ fprintf(output, "\n");
+ } else {
+ fprintf(output, " InitialReadSize: %d InterpretedSize: %d "
+ "FinalUpdateSize: %d FinalReadSize: %d SubroutineSize: %d\n",
+ sig->variableData[nextPos+0], sig->variableData[nextPos+1],
+ sig->variableData[nextPos+2], sig->variableData[nextPos+3],
+ sig->variableData[nextPos+4]);
+ nextPos += 5;
+ }
+ return true;
+}
+
+bool
+printLQHKEYCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+// const LqhKeyConf * const sig = (LqhKeyConf *) theData;
+
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
+
+bool
+printLQHKEYREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+// const LqhKeyRef * const sig = (LqhKeyRef *) theData;
+
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/LqhTrans.cpp b/ndb/src/common/debugger/signaldata/LqhTrans.cpp
new file mode 100644
index 00000000000..8282530cae6
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/LqhTrans.cpp
@@ -0,0 +1,40 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/LqhTransConf.hpp>
+
+bool
+printLQH_TRANSCONF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const LqhTransConf * const sig = (LqhTransConf *)theData;
+ fprintf(output, " tcRef: %x\n", sig->tcRef);
+ fprintf(output, " lqhNodeId: %x\n", sig->lqhNodeId);
+ fprintf(output, " operationStatus: %x\n", sig->operationStatus);
+ fprintf(output, " transId1: %x\n", sig->transId1);
+ fprintf(output, " transId2: %x\n", sig->transId2);
+ fprintf(output, " apiRef: %x\n", sig->apiRef);
+ fprintf(output, " apiOpRec: %x\n", sig->apiOpRec);
+ fprintf(output, " lqhConnectPtr: %x\n", sig->lqhConnectPtr);
+ fprintf(output, " oldTcOpRec: %x\n", sig->oldTcOpRec);
+ fprintf(output, " requestInfo: %x\n", sig->requestInfo);
+ fprintf(output, " gci: %x\n", sig->gci);
+ fprintf(output, " nextNodeId1: %x\n", sig->nextNodeId1);
+ fprintf(output, " nextNodeId2: %x\n", sig->nextNodeId2);
+ fprintf(output, " nextNodeId3: %x\n", sig->nextNodeId3);
+ fprintf(output, " tableId: %x\n", sig->tableId);
+ return true;
+}
+
diff --git a/ndb/src/common/debugger/signaldata/Makefile b/ndb/src/common/debugger/signaldata/Makefile
new file mode 100644
index 00000000000..5e86aaf97c0
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/Makefile
@@ -0,0 +1,32 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := signaldataprint
+
+SOURCES = TcKeyReq.cpp TcKeyConf.cpp TcKeyRef.cpp \
+ TcRollbackRep.cpp \
+ TupKey.cpp TupCommit.cpp LqhKey.cpp \
+ FsOpenReq.cpp FsCloseReq.cpp FsRef.cpp FsConf.cpp FsReadWriteReq.cpp\
+ SignalDataPrint.cpp SignalNames.cpp \
+ ContinueB.cpp DihContinueB.cpp NdbfsContinueB.cpp \
+ CloseComReqConf.cpp PackedSignal.cpp PrepFailReqRef.cpp \
+ GCPSave.cpp DictTabInfo.cpp \
+ AlterTable.cpp AlterTab.cpp \
+ CreateTrig.cpp AlterTrig.cpp DropTrig.cpp \
+ FireTrigOrd.cpp TrigAttrInfo.cpp \
+ CreateIndx.cpp AlterIndx.cpp DropIndx.cpp TcIndx.cpp \
+ IndxKeyInfo.cpp IndxAttrInfo.cpp \
+ FsAppendReq.cpp ScanTab.cpp \
+ BackupImpl.cpp BackupSignalData.cpp \
+ UtilSequence.cpp UtilPrepare.cpp UtilDelete.cpp UtilExecute.cpp \
+ LqhFrag.cpp DropTab.cpp PrepDropTab.cpp LCP.cpp MasterLCP.cpp \
+ CopyGCI.cpp SystemError.cpp StartRec.cpp NFCompleteRep.cpp \
+ FailRep.cpp DisconnectRep.cpp SignalDroppedRep.cpp \
+ SumaImpl.cpp NdbSttor.cpp CreateFragmentation.cpp \
+ UtilLock.cpp TuxMaint.cpp TupAccess.cpp AccLock.cpp \
+ LqhTrans.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/common/debugger/signaldata/MasterLCP.cpp b/ndb/src/common/debugger/signaldata/MasterLCP.cpp
new file mode 100644
index 00000000000..aa30404524f
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/MasterLCP.cpp
@@ -0,0 +1,87 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <signaldata/MasterLCP.hpp>
+#include <RefConvert.hpp>
+
+static
+void
+print(char *buf, size_t buf_len, MasterLCPConf::State s){
+ switch(s){
+ case MasterLCPConf::LCP_STATUS_IDLE:
+ snprintf(buf, buf_len, "LCP_STATUS_IDLE");
+ break;
+ case MasterLCPConf::LCP_STATUS_ACTIVE:
+ snprintf(buf, buf_len, "LCP_STATUS_ACTIVE");
+ break;
+ case MasterLCPConf::LCP_TAB_COMPLETED:
+ snprintf(buf, buf_len, "LCP_TAB_COMPLETED");
+ break;
+ case MasterLCPConf::LCP_TAB_SAVED:
+ snprintf(buf, buf_len, "LCP_TAB_SAVED");
+ break;
+ }
+}
+
+NdbOut &
+operator<<(NdbOut& out, const MasterLCPConf::State& s){
+ static char buf[255];
+ print(buf, sizeof(buf), s);
+ out << buf;
+ return out;
+}
+
+bool
+printMASTER_LCP_CONF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 recBlockNo){
+
+ MasterLCPConf * sig = (MasterLCPConf *)&theData[0];
+
+ static char buf[255];
+ print(buf, sizeof(buf), (MasterLCPConf::State)sig->lcpState);
+ fprintf(output, " senderNode=%d failedNode=%d SenderState=%s\n",
+ sig->senderNodeId, sig->failedNodeId, buf);
+ return true;
+}
+
+bool
+printMASTER_LCP_REQ(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 recBlockNo){
+
+ MasterLCPReq * sig = (MasterLCPReq *)&theData[0];
+
+ fprintf(output, " masterRef=(node=%d, block=%d), failedNode=%d\n",
+ refToNode(sig->masterRef), refToBlock(sig->masterRef),
+ sig->failedNodeId);
+ return true;
+}
+
+bool
+printMASTER_LCP_REF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 recBlockNo){
+
+ MasterLCPRef * sig = (MasterLCPRef *)&theData[0];
+ fprintf(output, " senderNode=%d failedNode=%d\n",
+ sig->senderNodeId, sig->failedNodeId);
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/NFCompleteRep.cpp b/ndb/src/common/debugger/signaldata/NFCompleteRep.cpp
new file mode 100644
index 00000000000..20f7ea99871
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/NFCompleteRep.cpp
@@ -0,0 +1,44 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <DebuggerNames.hpp>
+#include <signaldata/NFCompleteRep.hpp>
+
+bool
+printNF_COMPLETE_REP(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 recBlockNo){
+
+ NFCompleteRep * sig = (NFCompleteRep*)theData;
+ const char * who = getBlockName(sig->blockNo, 0);
+
+ if(who == 0){
+ fprintf(output,
+ " Node: %d has completed failure of node %d\n",
+ sig->nodeId, sig->failedNodeId);
+ } else {
+ fprintf(output,
+ " Node: %d block: %s has completed failure of node %d\n",
+ sig->nodeId, who, sig->failedNodeId);
+ }
+
+ fprintf(output, "Sent from line: %d\n",
+ sig->from);
+
+ return true;
+};
diff --git a/ndb/src/common/debugger/signaldata/NdbSttor.cpp b/ndb/src/common/debugger/signaldata/NdbSttor.cpp
new file mode 100644
index 00000000000..9fd081313be
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/NdbSttor.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/NdbSttor.hpp>
+
+bool
+printNDB_STTOR(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const NdbSttor * const sig = (NdbSttor *)theData;
+ fprintf(output, " senderRef: %x\n", sig->senderRef);
+ fprintf(output, " nodeId: %x\n", sig->nodeId);
+ fprintf(output, " internalStartPhase: %x\n", sig->internalStartPhase);
+ fprintf(output, " typeOfStart: %x\n", sig->typeOfStart);
+ fprintf(output, " masterNodeId: %x\n", sig->masterNodeId);
+
+ int left = len - NdbSttor::SignalLength;
+ if(left > 0){
+ fprintf(output, " config: ");
+ for(int i = 0; i<left; i++){
+ fprintf(output, "%x ", sig->config[i]);
+ if(((i + 1) % 7) == 0 && (i+1) < left){
+ fprintf(output, "\n config: ");
+ }
+ }
+ fprintf(output, "\n");
+ }
+ return true;
+}
+
+bool
+printNDB_STTORRY(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const NdbSttorry * const sig = (NdbSttorry *)theData;
+ fprintf(output, " senderRef: %x\n", sig->senderRef);
+ return true;
+}
+
diff --git a/ndb/src/common/debugger/signaldata/NdbfsContinueB.cpp b/ndb/src/common/debugger/signaldata/NdbfsContinueB.cpp
new file mode 100644
index 00000000000..b3c7a61136e
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/NdbfsContinueB.cpp
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <signaldata/NdbfsContinueB.hpp>
+
+bool
+printCONTINUEB_NDBFS(FILE * output, const Uint32 * theData, Uint32 len){
+
+ switch (theData[0]) {
+ case NdbfsContinueB::ZSCAN_MEMORYCHANNEL_10MS_DELAY:
+ fprintf(output, " Scanning the memory channel every 10ms\n");
+ return true;
+ break;
+ case NdbfsContinueB::ZSCAN_MEMORYCHANNEL_NO_DELAY:
+ fprintf(output, " Scanning the memory channel again with no delay\n");
+ return true;
+ break;
+ default:
+ fprintf(output, " Default system error lab...\n");
+ return false;
+ break;
+ }//switch
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/PackedSignal.cpp b/ndb/src/common/debugger/signaldata/PackedSignal.cpp
new file mode 100644
index 00000000000..f0f7aee74e4
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/PackedSignal.cpp
@@ -0,0 +1,104 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/PackedSignal.hpp>
+#include <signaldata/LqhKey.hpp>
+#include <debugger/DebuggerNames.hpp>
+
+bool
+printPACKED_SIGNAL(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ fprintf(output, "--------- Begin Packed Signals --------\n");
+ // Print each signal separately
+ for (i = 0; i < len;) {
+ switch (PackedSignal::getSignalType(theData[i])) {
+ case ZCOMMIT: {
+ Uint32 signalLength = 4;
+ fprintf(output, "--------------- Signal ----------------\n");
+ fprintf(output, "r.bn: %u \"%s\", length: %u \"COMMIT\"\n",
+ receiverBlockNo, getBlockName(receiverBlockNo,""), signalLength);
+ fprintf(output, "Signal data: ");
+ for(Uint32 j = 0; j < signalLength; j++)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ break;
+ }
+ case ZCOMPLETE: {
+ Uint32 signalLength = 3;
+ fprintf(output, "--------------- Signal ----------------\n");
+ fprintf(output, "r.bn: %u \"%s\", length: %u \"COMPLETE\"\n",
+ receiverBlockNo, getBlockName(receiverBlockNo,""), signalLength);
+ fprintf(output, "Signal data: ");
+ for(Uint32 j = 0; j < signalLength; j++)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ break;
+ }
+ case ZCOMMITTED: {
+ Uint32 signalLength = 3;
+ fprintf(output, "--------------- Signal ----------------\n");
+ fprintf(output, "r.bn: %u \"%s\", length: %u \"COMMITTED\"\n",
+ receiverBlockNo, getBlockName(receiverBlockNo,""), signalLength);
+ fprintf(output, "Signal data: ");
+ for(Uint32 j = 0; j < signalLength; j++)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ break;
+ }
+ case ZCOMPLETED: {
+ Uint32 signalLength = 3;
+ fprintf(output, "--------------- Signal ----------------\n");
+ fprintf(output, "r.bn: %u \"%s\", length: %u \"COMPLETED\"\n",
+ receiverBlockNo, getBlockName(receiverBlockNo,""), signalLength);
+ fprintf(output, "Signal data: ");
+ for(Uint32 j = 0; j < signalLength; j++)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ break;
+ }
+ case ZLQHKEYCONF: {
+ Uint32 signalLength = LqhKeyConf::SignalLength;
+
+ fprintf(output, "--------------- Signal ----------------\n");
+ fprintf(output, "r.bn: %u \"%s\", length: %u \"LQHKEYCONF\"\n",
+ receiverBlockNo, getBlockName(receiverBlockNo,""), signalLength);
+ printLQHKEYCONF(output, theData + i, signalLength, receiverBlockNo);
+ i += signalLength;
+ break;
+ }
+ case ZREMOVE_MARKER: {
+ Uint32 signalLength = 2;
+ fprintf(output, "--------------- Signal ----------------\n");
+ fprintf(output, "r.bn: %u \"%s\", length: %u \"REMOVE_MARKER\"\n",
+ receiverBlockNo, getBlockName(receiverBlockNo,""), signalLength);
+ fprintf(output, "Signal data: ");
+ i++; // Skip first word!
+ for(Uint32 j = 0; j < signalLength; j++)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ break;
+ }
+ default:
+ fprintf(output, "Unknown signal type\n");
+ }
+ }//for
+ fprintf(output, "--------- End Packed Signals ----------\n");
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/PrepDropTab.cpp b/ndb/src/common/debugger/signaldata/PrepDropTab.cpp
new file mode 100644
index 00000000000..59001bcd6f6
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/PrepDropTab.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/PrepDropTab.hpp>
+
+bool
+printPREP_DROP_TAB_REQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const PrepDropTabReq * const sig = (PrepDropTabReq *) theData;
+
+ fprintf(output,
+ " senderRef: %x senderData: %d TableId: %d\n",
+ sig->senderRef, sig->senderData, sig->tableId);
+ return true;
+}
+
+bool printPREP_DROP_TAB_CONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const PrepDropTabConf * const sig = (PrepDropTabConf *) theData;
+
+ fprintf(output,
+ " senderRef: %x senderData: %d TableId: %d\n",
+ sig->senderRef, sig->senderData, sig->tableId);
+
+ return true;
+}
+
+bool printPREP_DROP_TAB_REF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo)
+{
+ const PrepDropTabRef * const sig = (PrepDropTabRef *) theData;
+
+ fprintf(output,
+ " senderRef: %x senderData: %d TableId: %d errorCode: %d\n",
+ sig->senderRef, sig->senderData, sig->tableId, sig->errorCode);
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/PrepFailReqRef.cpp b/ndb/src/common/debugger/signaldata/PrepFailReqRef.cpp
new file mode 100644
index 00000000000..f3b4b97f0fd
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/PrepFailReqRef.cpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <kernel_types.h>
+#include <BlockNumbers.h>
+#include <signaldata/PrepFailReqRef.hpp>
+
+bool
+printPREPFAILREQREF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 receiverBlockNo){
+
+ PrepFailReqRef * cc = (PrepFailReqRef*)theData;
+
+ fprintf(output, " xxxBlockRef = (%d, %d) failNo = %d noOfNodes = %d\n",
+ refToBlock(cc->xxxBlockRef), refToNode(cc->xxxBlockRef),
+ cc->failNo, cc->noOfNodes);
+
+ int hits = 0;
+ fprintf(output, " Nodes: ");
+ for(int i = 0; i<MAX_NODES; i++){
+ if(NodeBitmask::get(cc->theNodes, i)){
+ hits++;
+ fprintf(output, " %d", i);
+ }
+ if(hits == 16){
+ fprintf(output, "\n Nodes: ");
+ hits = 0;
+ }
+ }
+ if(hits != 0)
+ fprintf(output, "\n");
+
+ return true;
+}
+
+
diff --git a/ndb/src/common/debugger/signaldata/ScanTab.cpp b/ndb/src/common/debugger/signaldata/ScanTab.cpp
new file mode 100644
index 00000000000..b4246059f6a
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/ScanTab.cpp
@@ -0,0 +1,163 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <BlockNumbers.h>
+#include <signaldata/ScanTab.hpp>
+#include <signaldata/ScanFrag.hpp>
+
+bool
+printSCANTABREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const ScanTabReq * const sig = (ScanTabReq *) theData;
+
+ const UintR requestInfo = sig->requestInfo;
+
+ fprintf(output, " apiConnectPtr: H\'%.8x\n",
+ sig->apiConnectPtr);
+ fprintf(output, " requestInfo: H\'%.8x:\n", requestInfo);
+ fprintf(output, " Parallellism: %u, LockMode: %u, Holdlock: %u, RangeScan: %u\n",
+ sig->getParallelism(requestInfo), sig->getLockMode(requestInfo), sig->getHoldLockFlag(requestInfo), sig->getRangeScanFlag(requestInfo));
+
+ fprintf(output, " attrLen: %d, tableId: %d, tableSchemaVer: %d\n",
+ sig->attrLen, sig->tableId, sig->tableSchemaVersion);
+
+ fprintf(output, " transId(1, 2): (H\'%.8x, H\'%.8x) storedProcId: H\'%.8x\n",
+ sig->transId1, sig->transId2, sig->storedProcId);
+
+ fprintf(output, " OperationPtr(s):\n");
+ for(int i = 0; i<16; i=i+4){
+ fprintf(output, " H\'%.8x, H\'%.8x, H\'%.8x, H\'%.8x\n",
+ sig->apiOperationPtr[i], sig->apiOperationPtr[i+1],
+ sig->apiOperationPtr[i+2], sig->apiOperationPtr[i+3]);
+ }
+ return false;
+}
+
+bool
+printSCANTABCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const ScanTabConf * const sig = (ScanTabConf *) theData;
+
+ const UintR requestInfo = sig->requestInfo;
+
+ fprintf(output, " apiConnectPtr: H\'%.8x\n",
+ sig->apiConnectPtr);
+ fprintf(output, " transId(1, 2): (H\'%.8x, H\'%.8x)\n",
+ sig->transId1, sig->transId2);
+
+ fprintf(output, " requestInfo: H\'%.8x(Operations: %u, ScanStatus: %u(\"",
+ requestInfo, sig->getOperations(requestInfo), sig->getScanStatus(requestInfo));
+ switch(sig->getScanStatus(requestInfo)){
+ case 0:
+ fprintf(output, "ZFALSE");
+ break;
+ case 1:
+ fprintf(output, "ZTRUE");
+ break;
+ case 2:
+ fprintf(output, "ZCLOSED");
+ break;
+ default:
+ fprintf(output, "UNKNOWN");
+ break;
+ }
+ fprintf(output, "\"))\n");
+#if 0
+ fprintf(output, " Operation(s):\n");
+ for(int i = 0; i<16; i++){
+ fprintf(output, " [%.2u]ix=%d l=%.2d,",
+ i, sig->getIdx(sig->operLenAndIdx[i]), sig->getLen(sig->operLenAndIdx[i]));
+ if (((i+1) % 4) == 0)
+ fprintf(output, "\n");
+ }
+#endif
+ return false;
+}
+
+bool
+printSCANTABINFO(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const ScanTabInfo * const sig = (ScanTabInfo *) theData;
+
+ fprintf(output, " apiConnectPtr: H\'%.8x\n",
+ sig->apiConnectPtr);
+
+ fprintf(output, " Operation(s):\n");
+ for(int i = 0; i<16; i++){
+ fprintf(output, " [%.2u]ix=%d l=%.2d,",
+ i, sig->getIdx(sig->operLenAndIdx[i]), sig->getLen(sig->operLenAndIdx[i]));
+ if (((i+1) % 4) == 0)
+ fprintf(output, "\n");
+ }
+
+ return false;
+}
+
+bool
+printSCANTABREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const ScanTabRef * const sig = (ScanTabRef *) theData;
+
+ fprintf(output, " apiConnectPtr: H\'%.8x\n",
+ sig->apiConnectPtr);
+
+ fprintf(output, " transId(1, 2): (H\'%.8x, H\'%.8x)\n",
+ sig->transId1, sig->transId2);
+
+ fprintf(output, " Errorcode: %u\n", sig->errorCode);
+
+ // fprintf(output, " sendScanNextReqWithClose: %u\n", sig->sendScanNextReqWithClose);
+ return false;
+}
+
+
+bool
+printSCANFRAGNEXTREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+ const ScanFragNextReq * const sig = (ScanFragNextReq *) theData;
+
+ fprintf(output, " senderData: H\'%.8x\n",
+ sig->senderData);
+
+ fprintf(output, " transId(1, 2): (H\'%.8x, H\'%.8x)\n",
+ sig->transId1, sig->transId2);
+
+ fprintf(output, " Close scan: %u\n", sig->closeFlag);
+
+ return false;
+}
+
+bool
+printSCANNEXTREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ if(receiverBlockNo == DBTC){
+ const ScanNextReq * const sig = (ScanNextReq *) theData;
+
+ fprintf(output, " aipConnectPtr: H\'%.8x\n",
+ sig->apiConnectPtr);
+
+ fprintf(output, " transId(1, 2): (H\'%.8x, H\'%.8x)\n",
+ sig->transId1, sig->transId2);
+
+ fprintf(output, " Stop this scan: %u\n", sig->stopScan);
+ }
+ if (receiverBlockNo == DBLQH){
+ return printSCANFRAGNEXTREQ(output, theData, len, receiverBlockNo);
+ }
+ return false;
+}
+
diff --git a/ndb/src/common/debugger/signaldata/SignalDataPrint.cpp b/ndb/src/common/debugger/signaldata/SignalDataPrint.cpp
new file mode 100644
index 00000000000..2236d0c0af1
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/SignalDataPrint.cpp
@@ -0,0 +1,254 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include "GlobalSignalNumbers.h"
+#include "signaldata/SignalDataPrint.hpp"
+#include "signaldata/TcKeyReq.hpp"
+#include "signaldata/TcKeyConf.hpp"
+#include "signaldata/TcKeyRef.hpp"
+#include "signaldata/LqhKey.hpp"
+#include "signaldata/TupKey.hpp"
+#include "signaldata/TupCommit.hpp"
+#include "signaldata/FsOpenReq.hpp"
+#include "signaldata/FsCloseReq.hpp"
+#include "signaldata/FsReadWriteReq.hpp"
+#include "signaldata/FsRef.hpp"
+#include "signaldata/FsConf.hpp"
+#include "signaldata/CloseComReqConf.hpp"
+#include "signaldata/PackedSignal.hpp"
+#include "signaldata/PrepFailReqRef.hpp"
+#include "signaldata/DictTabInfo.hpp"
+#include "signaldata/AlterTable.hpp"
+#include "signaldata/AlterTab.hpp"
+#include "signaldata/CreateTrig.hpp"
+#include "signaldata/AlterTrig.hpp"
+#include "signaldata/DropTrig.hpp"
+#include "signaldata/FireTrigOrd.hpp"
+#include "signaldata/TrigAttrInfo.hpp"
+#include "signaldata/CreateIndx.hpp"
+#include "signaldata/AlterIndx.hpp"
+#include "signaldata/DropIndx.hpp"
+#include "signaldata/TcIndx.hpp"
+#include "signaldata/IndxKeyInfo.hpp"
+#include "signaldata/IndxAttrInfo.hpp"
+#include <signaldata/FsAppendReq.hpp>
+#include <signaldata/BackupSignalData.hpp>
+#include <signaldata/BackupImpl.hpp>
+#include <signaldata/UtilSequence.hpp>
+#include <signaldata/UtilPrepare.hpp>
+#include <signaldata/UtilExecute.hpp>
+#include <signaldata/ScanTab.hpp>
+#include <signaldata/LqhFrag.hpp>
+#include <signaldata/LqhTransConf.hpp>
+#include <signaldata/DropTab.hpp>
+#include <signaldata/PrepDropTab.hpp>
+#include <signaldata/LCP.hpp>
+#include <signaldata/MasterLCP.hpp>
+#include <signaldata/CopyGCIReq.hpp>
+#include <signaldata/SystemError.hpp>
+#include <signaldata/StartRec.hpp>
+#include <signaldata/NFCompleteRep.hpp>
+#include <signaldata/SignalDroppedRep.hpp>
+#include <signaldata/FailRep.hpp>
+#include <signaldata/DisconnectRep.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <signaldata/NdbSttor.hpp>
+#include <signaldata/CreateFragmentation.hpp>
+#include <signaldata/UtilLock.hpp>
+#include <signaldata/TuxMaint.hpp>
+#include <signaldata/TupAccess.hpp>
+#include <signaldata/AccLock.hpp>
+
+bool printCONTINUEB(FILE *, const Uint32 *, Uint32, Uint16);
+
+/**
+ * This is the register
+ */
+const NameFunctionPair
+SignalDataPrintFunctions[] = {
+ { GSN_TCKEYREQ, printTCKEYREQ },
+ { GSN_TCKEYCONF, printTCKEYCONF },
+ { GSN_TCKEYREF, printTCKEYREF },
+ { GSN_LQHKEYREQ, printLQHKEYREQ },
+ { GSN_LQHKEYCONF, printLQHKEYCONF },
+ { GSN_LQHKEYREF, printLQHKEYREF },
+ { GSN_TUPKEYREQ, printTUPKEYREQ },
+ { GSN_TUPKEYCONF, printTUPKEYCONF },
+ { GSN_TUPKEYREF, printTUPKEYREF },
+ { GSN_TUP_COMMITREQ, printTUPCOMMITREQ },
+ { GSN_CONTINUEB, printCONTINUEB },
+ { GSN_FSOPENREQ, printFSOPENREQ },
+ { GSN_FSCLOSEREQ, printFSCLOSEREQ },
+ { GSN_FSREADREQ, printFSREADWRITEREQ },
+ { GSN_FSWRITEREQ, printFSREADWRITEREQ },
+ { GSN_FSCLOSEREF, printFSREF },
+ { GSN_FSOPENREF, printFSREF },
+ { GSN_FSWRITEREF, printFSREF },
+ { GSN_FSREADREF, printFSREF },
+ { GSN_FSSYNCREF, printFSREF },
+ { GSN_FSCLOSECONF, printFSCONF },
+ { GSN_FSOPENCONF, printFSCONF },
+ { GSN_FSWRITECONF, printFSCONF },
+ { GSN_FSREADCONF, printFSCONF },
+ { GSN_FSSYNCCONF, printFSCONF },
+ { GSN_CLOSE_COMREQ, printCLOSECOMREQCONF },
+ { GSN_CLOSE_COMCONF, printCLOSECOMREQCONF },
+ { GSN_PACKED_SIGNAL, printPACKED_SIGNAL },
+ { GSN_PREP_FAILREQ, printPREPFAILREQREF },
+ { GSN_PREP_FAILREF, printPREPFAILREQREF },
+ { GSN_ALTER_TABLE_REQ, printALTER_TABLE_REQ },
+ { GSN_ALTER_TABLE_CONF, printALTER_TABLE_CONF },
+ { GSN_ALTER_TABLE_REF, printALTER_TABLE_REF },
+ { GSN_ALTER_TAB_REQ, printALTER_TAB_REQ },
+ { GSN_ALTER_TAB_CONF, printALTER_TAB_CONF },
+ { GSN_ALTER_TAB_REF, printALTER_TAB_REF },
+ { GSN_CREATE_TRIG_REQ, printCREATE_TRIG_REQ },
+ { GSN_CREATE_TRIG_CONF, printCREATE_TRIG_CONF },
+ { GSN_CREATE_TRIG_REF, printCREATE_TRIG_REF },
+ { GSN_ALTER_TRIG_REQ, printALTER_TRIG_REQ },
+ { GSN_ALTER_TRIG_CONF, printALTER_TRIG_CONF },
+ { GSN_ALTER_TRIG_REF, printALTER_TRIG_REF },
+ { GSN_DROP_TRIG_REQ, printDROP_TRIG_REQ },
+ { GSN_DROP_TRIG_CONF, printDROP_TRIG_CONF },
+ { GSN_DROP_TRIG_REF, printDROP_TRIG_REF },
+ { GSN_FIRE_TRIG_ORD, printFIRE_TRIG_ORD },
+ { GSN_TRIG_ATTRINFO, printTRIG_ATTRINFO },
+ { GSN_CREATE_INDX_REQ, printCREATE_INDX_REQ },
+ { GSN_CREATE_INDX_CONF, printCREATE_INDX_CONF },
+ { GSN_CREATE_INDX_REF, printCREATE_INDX_REF },
+ { GSN_DROP_INDX_REQ, printDROP_INDX_REQ },
+ { GSN_DROP_INDX_CONF, printDROP_INDX_CONF },
+ { GSN_DROP_INDX_REF, printDROP_INDX_REF },
+ { GSN_ALTER_INDX_REQ, printALTER_INDX_REQ },
+ { GSN_ALTER_INDX_CONF, printALTER_INDX_CONF },
+ { GSN_ALTER_INDX_REF, printALTER_INDX_REF },
+ { GSN_TCINDXREQ, printTCINDXREQ },
+ { GSN_TCINDXCONF, printTCINDXCONF },
+ { GSN_TCINDXREF, printTCINDXREF },
+ { GSN_INDXKEYINFO, printINDXKEYINFO },
+ { GSN_INDXATTRINFO, printINDXATTRINFO },
+ //{ GSN_TCINDXNEXTREQ, printTCINDXNEXTREQ },
+ //{ GSN_TCINDEXNEXTCONF, printTCINDEXNEXTCONF },
+ //{ GSN_TCINDEXNEXREF, printTCINDEXNEXREF },
+ { GSN_FSAPPENDREQ, printFSAPPENDREQ },
+ { GSN_BACKUP_REQ, printBACKUP_REQ },
+ { GSN_BACKUP_DATA, printBACKUP_DATA },
+ { GSN_BACKUP_REF, printBACKUP_REF },
+ { GSN_BACKUP_CONF, printBACKUP_CONF },
+ { GSN_ABORT_BACKUP_ORD, printABORT_BACKUP_ORD },
+ { GSN_BACKUP_ABORT_REP, printBACKUP_ABORT_REP },
+ { GSN_BACKUP_COMPLETE_REP, printBACKUP_COMPLETE_REP },
+ { GSN_BACKUP_NF_COMPLETE_REP, printBACKUP_NF_COMPLETE_REP },
+ { GSN_DEFINE_BACKUP_REQ, printDEFINE_BACKUP_REQ },
+ { GSN_DEFINE_BACKUP_REF, printDEFINE_BACKUP_REF },
+ { GSN_DEFINE_BACKUP_CONF, printDEFINE_BACKUP_CONF },
+ { GSN_START_BACKUP_REQ, printSTART_BACKUP_REQ },
+ { GSN_START_BACKUP_REF, printSTART_BACKUP_REF },
+ { GSN_START_BACKUP_CONF, printSTART_BACKUP_CONF },
+ { GSN_BACKUP_FRAGMENT_REQ, printBACKUP_FRAGMENT_REQ },
+ { GSN_BACKUP_FRAGMENT_REF, printBACKUP_FRAGMENT_REF },
+ { GSN_BACKUP_FRAGMENT_CONF, printBACKUP_FRAGMENT_CONF },
+ { GSN_STOP_BACKUP_REQ, printSTOP_BACKUP_REQ },
+ { GSN_STOP_BACKUP_REF, printSTOP_BACKUP_REF },
+ { GSN_STOP_BACKUP_CONF, printSTOP_BACKUP_CONF },
+ { GSN_BACKUP_STATUS_REQ, printBACKUP_STATUS_REQ },
+ //{ GSN_BACKUP_STATUS_REF, printBACKUP_STATUS_REF },
+ { GSN_BACKUP_STATUS_CONF, printBACKUP_STATUS_CONF },
+ { GSN_UTIL_SEQUENCE_REQ, printUTIL_SEQUENCE_REQ },
+ { GSN_UTIL_SEQUENCE_REF, printUTIL_SEQUENCE_REF },
+ { GSN_UTIL_SEQUENCE_CONF, printUTIL_SEQUENCE_CONF },
+ { GSN_UTIL_PREPARE_REQ, printUTIL_PREPARE_REQ },
+ { GSN_UTIL_PREPARE_REF, printUTIL_PREPARE_REF },
+ { GSN_UTIL_PREPARE_CONF, printUTIL_PREPARE_CONF },
+ { GSN_UTIL_EXECUTE_REQ, printUTIL_EXECUTE_REQ },
+ { GSN_UTIL_EXECUTE_REF, printUTIL_EXECUTE_REF },
+ { GSN_UTIL_EXECUTE_CONF, printUTIL_EXECUTE_CONF },
+ { GSN_SCAN_TABREQ, printSCANTABREQ },
+ { GSN_SCAN_TABCONF, printSCANTABCONF },
+ { GSN_SCAN_TABREF, printSCANTABREF },
+ { GSN_SCAN_NEXTREQ, printSCANNEXTREQ },
+ { GSN_LQHFRAGREQ, printLQH_FRAG_REQ },
+ { GSN_LQHFRAGREF, printLQH_FRAG_REF },
+ { GSN_LQHFRAGCONF, printLQH_FRAG_CONF },
+ { GSN_PREP_DROP_TAB_REQ, printPREP_DROP_TAB_REQ },
+ { GSN_PREP_DROP_TAB_REF, printPREP_DROP_TAB_REF },
+ { GSN_PREP_DROP_TAB_CONF, printPREP_DROP_TAB_CONF },
+ { GSN_DROP_TAB_REQ, printDROP_TAB_REQ },
+ { GSN_DROP_TAB_REF, printDROP_TAB_REF },
+ { GSN_DROP_TAB_CONF, printDROP_TAB_CONF },
+ { GSN_LCP_FRAG_ORD, printLCP_FRAG_ORD },
+ { GSN_LCP_FRAG_REP, printLCP_FRAG_REP },
+ { GSN_LCP_COMPLETE_REP, printLCP_COMPLETE_REP },
+ { GSN_START_LCP_REQ, printSTART_LCP_REQ },
+ { GSN_START_LCP_CONF, printSTART_LCP_CONF },
+ { GSN_MASTER_LCPREQ, printMASTER_LCP_REQ },
+ { GSN_MASTER_LCPREF, printMASTER_LCP_REF },
+ { GSN_MASTER_LCPCONF, printMASTER_LCP_CONF },
+ { GSN_COPY_GCIREQ, printCOPY_GCI_REQ },
+ { GSN_SYSTEM_ERROR, printSYSTEM_ERROR },
+ { GSN_START_RECREQ, printSTART_REC_REQ },
+ { GSN_START_RECCONF, printSTART_REC_CONF },
+ { GSN_NF_COMPLETEREP, printNF_COMPLETE_REP },
+ { GSN_SIGNAL_DROPPED_REP, printSIGNAL_DROPPED_REP },
+ { GSN_FAIL_REP, printFAIL_REP },
+ { GSN_DISCONNECT_REP, printDISCONNECT_REP },
+
+ { GSN_SUB_CREATE_REQ, printSUB_CREATE_REQ },
+ //{ GSN_SUB_CREATE_REF, printSUB_CREATE_REF },
+ { GSN_SUB_CREATE_CONF, printSUB_CREATE_CONF },
+ { GSN_SUB_START_REQ, printSUB_START_REQ },
+ { GSN_SUB_START_REF, printSUB_START_REF },
+ { GSN_SUB_START_CONF, printSUB_START_CONF },
+ { GSN_SUB_SYNC_REQ, printSUB_SYNC_REQ },
+ { GSN_SUB_SYNC_REF, printSUB_SYNC_REF },
+ { GSN_SUB_SYNC_CONF, printSUB_SYNC_CONF },
+ { GSN_SUB_META_DATA, printSUB_META_DATA },
+ { GSN_SUB_TABLE_DATA, printSUB_TABLE_DATA },
+ { GSN_SUB_SYNC_CONTINUE_REQ, printSUB_SYNC_CONTINUE_REQ },
+ { GSN_SUB_SYNC_CONTINUE_REF, printSUB_SYNC_CONTINUE_REF },
+ { GSN_SUB_SYNC_CONTINUE_CONF, printSUB_SYNC_CONTINUE_CONF },
+ { GSN_SUB_GCP_COMPLETE_REP, printSUB_GCP_COMPLETE_REP }
+
+ ,{ GSN_CREATE_FRAGMENTATION_REQ, printCREATE_FRAGMENTATION_REQ }
+ ,{ GSN_CREATE_FRAGMENTATION_REF, printCREATE_FRAGMENTATION_REF }
+ ,{ GSN_CREATE_FRAGMENTATION_CONF, printCREATE_FRAGMENTATION_CONF }
+
+ ,{ GSN_UTIL_CREATE_LOCK_REQ, printUTIL_CREATE_LOCK_REQ }
+ ,{ GSN_UTIL_CREATE_LOCK_REF, printUTIL_CREATE_LOCK_REF }
+ ,{ GSN_UTIL_CREATE_LOCK_CONF, printUTIL_CREATE_LOCK_CONF }
+ ,{ GSN_UTIL_DESTROY_LOCK_REQ, printUTIL_DESTROY_LOCK_REQ }
+ ,{ GSN_UTIL_DESTROY_LOCK_REF, printUTIL_DESTROY_LOCK_REF }
+ ,{ GSN_UTIL_DESTROY_LOCK_CONF, printUTIL_DESTROY_LOCK_CONF }
+ ,{ GSN_UTIL_LOCK_REQ, printUTIL_LOCK_REQ }
+ ,{ GSN_UTIL_LOCK_REF, printUTIL_LOCK_REF }
+ ,{ GSN_UTIL_LOCK_CONF, printUTIL_LOCK_CONF }
+ ,{ GSN_UTIL_UNLOCK_REQ, printUTIL_UNLOCK_REQ }
+ ,{ GSN_UTIL_UNLOCK_REF, printUTIL_UNLOCK_REF }
+ ,{ GSN_UTIL_UNLOCK_CONF, printUTIL_UNLOCK_CONF }
+ ,{ GSN_TUX_MAINT_REQ, printTUX_MAINT_REQ }
+ ,{ GSN_TUP_READ_ATTRS, printTUP_READ_ATTRS }
+ ,{ GSN_TUP_QUERY_TH, printTUP_QUERY_TH }
+ ,{ GSN_TUP_STORE_TH, printTUP_STORE_TH }
+ ,{ GSN_ACC_LOCKREQ, printACC_LOCKREQ }
+ ,{ GSN_LQH_TRANSCONF, printLQH_TRANSCONF }
+};
+
+const unsigned short NO_OF_PRINT_FUNCTIONS = sizeof(SignalDataPrintFunctions)/sizeof(NameFunctionPair);
+
+
+
diff --git a/ndb/src/common/debugger/signaldata/SignalDroppedRep.cpp b/ndb/src/common/debugger/signaldata/SignalDroppedRep.cpp
new file mode 100644
index 00000000000..be31b4edb22
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/SignalDroppedRep.cpp
@@ -0,0 +1,34 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <DebuggerNames.hpp>
+#include <signaldata/SignalDroppedRep.hpp>
+
+bool
+printSIGNAL_DROPPED_REP(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 recBlockNo){
+ SignalDroppedRep * sig = (SignalDroppedRep*)theData;
+
+ fprintf(output, " originalGsn: %s(%d) Length: %d SectionCount: %d\n",
+ getSignalName(sig->originalGsn),
+ sig->originalGsn,
+ sig->originalLength,
+ sig->originalSectionCount);
+ return false;
+}
diff --git a/ndb/src/common/debugger/signaldata/SignalNames.cpp b/ndb/src/common/debugger/signaldata/SignalNames.cpp
new file mode 100644
index 00000000000..4e5c8603db1
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/SignalNames.cpp
@@ -0,0 +1,673 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "GlobalSignalNumbers.h"
+
+const GsnName SignalNames [] = {
+ { GSN_API_REGCONF, "API_REGCONF" }
+ ,{ GSN_API_REGREF, "API_REGREF" }
+ ,{ GSN_API_REGREQ, "API_REGREQ" }
+ ,{ GSN_ATTRINFO, "ATTRINFO" }
+ ,{ GSN_SCHEMA_INFO, "SCHEMA_INFO" }
+ ,{ GSN_SCHEMA_INFOCONF, "SCHEMA_INFOCONF" }
+ ,{ GSN_GET_SCHEMA_INFOREQ, "GET_SCHEMA_INFOREQ" }
+ ,{ GSN_DIHNDBTAMPER, "DIHNDBTAMPER" }
+ ,{ GSN_KEYINFO, "KEYINFO" }
+ ,{ GSN_KEYINFO20, "KEYINFO20" }
+ ,{ GSN_KEYINFO20_R, "KEYINFO20_R" }
+ ,{ GSN_NODE_FAILREP, "NODE_FAILREP" }
+ ,{ GSN_READCONF, "READCONF" }
+ ,{ GSN_SCAN_NEXTREQ, "SCAN_NEXTREQ" }
+ ,{ GSN_SCAN_TABCONF, "SCAN_TABCONF" }
+ ,{ GSN_SCAN_TABINFO, "SCAN_TABINFO" }
+ ,{ GSN_SCAN_TABREF, "SCAN_TABREF" }
+ ,{ GSN_SCAN_TABREQ, "SCAN_TABREQ" }
+ ,{ GSN_TC_COMMITCONF, "TC_COMMITCONF" }
+ ,{ GSN_TC_COMMITREF, "TC_COMMITREF" }
+ ,{ GSN_TC_COMMITREQ, "TC_COMMITREQ" }
+ ,{ GSN_TCKEY_FAILCONF, "TCKEY_FAILCONF" }
+ ,{ GSN_TCKEY_FAILREF, "TCKEY_FAILREF" }
+ ,{ GSN_TCKEYCONF, "TCKEYCONF" }
+ ,{ GSN_TCKEYREF, "TCKEYREF" }
+ ,{ GSN_TCKEYREQ, "TCKEYREQ" }
+ ,{ GSN_TCRELEASECONF, "TCRELEASECONF" }
+ ,{ GSN_TCRELEASEREF, "TCRELEASEREF" }
+ ,{ GSN_TCRELEASEREQ, "TCRELEASEREQ" }
+ ,{ GSN_TCROLLBACKCONF, "TCROLLBACKCONF" }
+ ,{ GSN_TCROLLBACKREF, "TCROLLBACKREF" }
+ ,{ GSN_TCROLLBACKREQ, "TCROLLBACKREQ" }
+ ,{ GSN_TCROLLBACKREP, "TCROLLBACKREP" }
+ ,{ GSN_TCSEIZECONF, "TCSEIZECONF" }
+ ,{ GSN_TCSEIZEREF, "TCSEIZEREF" }
+ ,{ GSN_TCSEIZEREQ, "TCSEIZEREQ" }
+ ,{ GSN_TRANSID_AI, "TRANSID_AI" }
+ ,{ GSN_TRANSID_AI_R, "TRANSID_AI_R" }
+ ,{ GSN_ABORT, "ABORT" }
+ ,{ GSN_ABORTCONF, "ABORTCONF" }
+ ,{ GSN_ABORTED, "ABORTED" }
+ ,{ GSN_ABORTREQ, "ABORTREQ" }
+ ,{ GSN_ACC_ABORTCONF, "ACC_ABORTCONF" }
+ ,{ GSN_ACC_ABORTREQ, "ACC_ABORTREQ" }
+ ,{ GSN_ACC_CHECK_SCAN, "ACC_CHECK_SCAN" }
+ ,{ GSN_ACC_COMMITCONF, "ACC_COMMITCONF" }
+ ,{ GSN_ACC_COMMITREQ, "ACC_COMMITREQ" }
+ ,{ GSN_ACC_CONTOPCONF, "ACC_CONTOPCONF" }
+ ,{ GSN_ACC_CONTOPREQ, "ACC_CONTOPREQ" }
+ ,{ GSN_ACC_LCPCONF, "ACC_LCPCONF" }
+ ,{ GSN_ACC_LCPREF, "ACC_LCPREF" }
+ ,{ GSN_ACC_LCPREQ, "ACC_LCPREQ" }
+ ,{ GSN_ACC_LCPSTARTED, "ACC_LCPSTARTED" }
+ ,{ GSN_ACC_OVER_REC, "ACC_OVER_REC" }
+ ,{ GSN_ACC_SAVE_PAGES, "ACC_SAVE_PAGES" }
+ ,{ GSN_ACC_SCAN_INFO, "ACC_SCAN_INFO" }
+ ,{ GSN_ACC_SCAN_INFO24, "ACC_SCAN_INFO24" }
+ ,{ GSN_ACC_SCANCONF, "ACC_SCANCONF" }
+ ,{ GSN_ACC_SCANREF, "ACC_SCANREF" }
+ ,{ GSN_ACC_SCANREQ, "ACC_SCANREQ" }
+ ,{ GSN_ACC_SRCONF, "ACC_SRCONF" }
+ ,{ GSN_ACC_SRREF, "ACC_SRREF" }
+ ,{ GSN_ACC_SRREQ, "ACC_SRREQ" }
+ ,{ GSN_ACC_TO_CONF, "ACC_TO_CONF" }
+ ,{ GSN_ACC_TO_REF, "ACC_TO_REF" }
+ ,{ GSN_ACC_TO_REQ, "ACC_TO_REQ" }
+ ,{ GSN_ACCFRAGCONF, "ACCFRAGCONF" }
+ ,{ GSN_ACCFRAGREF, "ACCFRAGREF" }
+ ,{ GSN_ACCFRAGREQ, "ACCFRAGREQ" }
+ ,{ GSN_ACCKEYCONF, "ACCKEYCONF" }
+ ,{ GSN_ACCKEYREF, "ACCKEYREF" }
+ ,{ GSN_ACCKEYREQ, "ACCKEYREQ" }
+ ,{ GSN_ACCMINUPDATE, "ACCMINUPDATE" }
+ ,{ GSN_ACCSEIZECONF, "ACCSEIZECONF" }
+ ,{ GSN_ACCSEIZEREF, "ACCSEIZEREF" }
+ ,{ GSN_ACCSEIZEREQ, "ACCSEIZEREQ" }
+ ,{ GSN_ACCUPDATECONF, "ACCUPDATECONF" }
+ ,{ GSN_ACCUPDATEKEY, "ACCUPDATEKEY" }
+ ,{ GSN_ACCUPDATEREF, "ACCUPDATEREF" }
+ ,{ GSN_ADD_FRAGCONF, "ADD_FRAGCONF" }
+ ,{ GSN_ADD_FRAGREF, "ADD_FRAGREF" }
+ ,{ GSN_ADD_FRAGREQ, "ADD_FRAGREQ" }
+ ,{ GSN_API_FAILCONF, "API_FAILCONF" }
+ ,{ GSN_API_FAILREQ, "API_FAILREQ" }
+ ,{ GSN_APPL_CHANGEREP, "APPL_CHANGEREP" }
+ // ,{ GSN_APPL_ERROR, "APPL_ERROR" }
+ ,{ GSN_APPL_HB, "APPL_HB" }
+ ,{ GSN_APPL_HBREQ, "APPL_HBREQ" }
+ ,{ GSN_APPL_REGCONF, "APPL_REGCONF" }
+ ,{ GSN_APPL_REGREF, "APPL_REGREF" }
+ ,{ GSN_APPL_REGREQ, "APPL_REGREQ" }
+ ,{ GSN_APPL_RUN, "APPL_RUN" }
+ ,{ GSN_APPL_STARTCONF, "APPL_STARTCONF" }
+ ,{ GSN_APPL_STARTREG, "APPL_STARTREG" }
+ ,{ GSN_CHECK_LCP_STOP, "CHECK_LCP_STOP" }
+ ,{ GSN_CLOSE_COMCONF, "CLOSE_COMCONF" }
+ ,{ GSN_CLOSE_COMREQ, "CLOSE_COMREQ" }
+ ,{ GSN_CM_ACKADD, "CM_ACKADD" }
+ ,{ GSN_CM_ACKALARM, "CM_ACKALARM" }
+ ,{ GSN_CM_ADD, "CM_ADD" }
+ ,{ GSN_CM_APPCHG, "CM_APPCHG" }
+ ,{ GSN_CM_HEARTBEAT, "CM_HEARTBEAT" }
+ ,{ GSN_CM_INFOCONF, "CM_INFOCONF" }
+ ,{ GSN_CM_INFOREQ, "CM_INFOREQ" }
+ ,{ GSN_CM_INIT, "CM_INIT" }
+ ,{ GSN_CM_NODEINFOCONF, "CM_NODEINFOCONF" }
+ ,{ GSN_CM_NODEINFOREF, "CM_NODEINFOREF" }
+ ,{ GSN_CM_NODEINFOREQ, "CM_NODEINFOREQ" }
+ ,{ GSN_CM_REGCONF, "CM_REGCONF" }
+ ,{ GSN_CM_REGREF, "CM_REGREF" }
+ ,{ GSN_CM_REGREQ, "CM_REGREQ" }
+ ,{ GSN_CM_RUN, "CM_RUN" }
+ ,{ GSN_CMVMI_CFGCONF, "CMVMI_CFGCONF" }
+ ,{ GSN_CMVMI_CFGREQ, "CMVMI_CFGREQ" }
+ ,{ GSN_CNTR_CHANGEREP, "CNTR_CHANGEREP" }
+ ,{ GSN_CNTR_MASTERCONF, "CNTR_MASTERCONF" }
+ ,{ GSN_CNTR_MASTERREF, "CNTR_MASTERREF" }
+ ,{ GSN_CNTR_MASTERREQ, "CNTR_MASTERREQ" }
+ ,{ GSN_CNTR_WAITREP, "CNTR_WAITREP" }
+ ,{ GSN_COMMIT, "COMMIT" }
+ ,{ GSN_COMMIT_FAILCONF, "COMMIT_FAILCONF" }
+ ,{ GSN_COMMIT_FAILREQ, "COMMIT_FAILREQ" }
+ ,{ GSN_COMMITCONF, "COMMITCONF" }
+ ,{ GSN_COMMITREQ, "COMMITREQ" }
+ ,{ GSN_COMMITTED, "COMMITTED" }
+ ,{ GSN_LCP_FRAG_ORD, "LCP_FRAG_ORD" }
+ ,{ GSN_LCP_FRAG_REP, "LCP_FRAG_REP" }
+ ,{ GSN_LCP_COMPLETE_REP, "LCP_COMPLETE_REP" }
+ ,{ GSN_START_LCP_REQ, "START_LCP_REQ" }
+ ,{ GSN_START_LCP_CONF, "START_LCP_CONF" }
+ ,{ GSN_COMPLETE, "COMPLETE" }
+ ,{ GSN_COMPLETECONF, "COMPLETECONF" }
+ ,{ GSN_COMPLETED, "COMPLETED" }
+ ,{ GSN_COMPLETEREQ, "COMPLETEREQ" }
+ ,{ GSN_CONNECT_REP, "CONNECT_REP" }
+ ,{ GSN_CONTINUEB, "CONTINUEB" }
+ ,{ GSN_COPY_ACTIVECONF, "COPY_ACTIVECONF" }
+ ,{ GSN_COPY_ACTIVEREF, "COPY_ACTIVEREF" }
+ ,{ GSN_COPY_ACTIVEREQ, "COPY_ACTIVEREQ" }
+ ,{ GSN_COPY_FRAGCONF, "COPY_FRAGCONF" }
+ ,{ GSN_COPY_FRAGREF, "COPY_FRAGREF" }
+ ,{ GSN_COPY_FRAGREQ, "COPY_FRAGREQ" }
+ ,{ GSN_COPY_GCICONF, "COPY_GCICONF" }
+ ,{ GSN_COPY_GCIREQ, "COPY_GCIREQ" }
+ ,{ GSN_COPY_STATECONF, "COPY_STATECONF" }
+ ,{ GSN_COPY_STATEREQ, "COPY_STATEREQ" }
+ ,{ GSN_COPY_TABCONF, "COPY_TABCONF" }
+ ,{ GSN_COPY_TABREQ, "COPY_TABREQ" }
+ ,{ GSN_CREATE_FRAGCONF, "CREATE_FRAGCONF" }
+ ,{ GSN_CREATE_FRAGREF, "CREATE_FRAGREF" }
+ ,{ GSN_CREATE_FRAGREQ, "CREATE_FRAGREQ" }
+ ,{ GSN_DEBUG_SIG, "DEBUG_SIG" }
+ ,{ GSN_DI_FCOUNTCONF, "DI_FCOUNTCONF" }
+ ,{ GSN_DI_FCOUNTREF, "DI_FCOUNTREF" }
+ ,{ GSN_DI_FCOUNTREQ, "DI_FCOUNTREQ" }
+ ,{ GSN_DIADDTABCONF, "DIADDTABCONF" }
+ ,{ GSN_DIADDTABREF, "DIADDTABREF" }
+ ,{ GSN_DIADDTABREQ, "DIADDTABREQ" }
+ ,{ GSN_DICTSTARTCONF, "DICTSTARTCONF" }
+ ,{ GSN_DICTSTARTREQ, "DICTSTARTREQ" }
+ ,{ GSN_LIST_TABLES_REQ, "LIST_TABLES_REQ" }
+ ,{ GSN_LIST_TABLES_CONF, "LIST_TABLES_CONF" }
+ ,{ GSN_DIGETNODESCONF, "DIGETNODESCONF" }
+ ,{ GSN_DIGETNODESREF, "DIGETNODESREF" }
+ ,{ GSN_DIGETNODESREQ, "DIGETNODESREQ" }
+ ,{ GSN_DIGETPRIMCONF, "DIGETPRIMCONF" }
+ ,{ GSN_DIGETPRIMREF, "DIGETPRIMREF" }
+ ,{ GSN_DIGETPRIMREQ, "DIGETPRIMREQ" }
+ ,{ GSN_DIH_RESTARTCONF, "DIH_RESTARTCONF" }
+ ,{ GSN_DIH_RESTARTREF, "DIH_RESTARTREF" }
+ ,{ GSN_DIH_RESTARTREQ, "DIH_RESTARTREQ" }
+
+ ,{ GSN_DIRELEASECONF, "DIRELEASECONF" }
+ ,{ GSN_DIRELEASEREF, "DIRELEASEREF" }
+ ,{ GSN_DIRELEASEREQ, "DIRELEASEREQ" }
+ ,{ GSN_DISCONNECT_REP, "DISCONNECT_REP" }
+ ,{ GSN_DISEIZECONF, "DISEIZECONF" }
+ ,{ GSN_DISEIZEREF, "DISEIZEREF" }
+ ,{ GSN_DISEIZEREQ, "DISEIZEREQ" }
+ ,{ GSN_DIVERIFYCONF, "DIVERIFYCONF" }
+ ,{ GSN_DIVERIFYREF, "DIVERIFYREF" }
+ ,{ GSN_DIVERIFYREQ, "DIVERIFYREQ" }
+ ,{ GSN_EMPTY_LCP_REQ, "EMPTY_LCP_REQ" }
+ ,{ GSN_EMPTY_LCP_CONF, "EMPTY_LCP_CONF" }
+ ,{ GSN_ENABLE_COMORD, "ENABLE_COMORD" }
+ ,{ GSN_END_LCPCONF, "END_LCPCONF" }
+ ,{ GSN_END_LCPREQ, "END_LCPREQ" }
+ ,{ GSN_END_TOCONF, "END_TOCONF" }
+ ,{ GSN_END_TOREQ, "END_TOREQ" }
+ ,{ GSN_EVENT_REP, "EVENT_REP" }
+ ,{ GSN_EXEC_FRAGCONF, "EXEC_FRAGCONF" }
+ ,{ GSN_EXEC_FRAGREF, "EXEC_FRAGREF" }
+ ,{ GSN_EXEC_FRAGREQ, "EXEC_FRAGREQ" }
+ ,{ GSN_EXEC_SRCONF, "EXEC_SRCONF" }
+ ,{ GSN_EXEC_SRREQ, "EXEC_SRREQ" }
+ ,{ GSN_EXPANDCHECK2, "EXPANDCHECK2" }
+ ,{ GSN_FAIL_REP, "FAIL_REP" }
+ ,{ GSN_FSCLOSECONF, "FSCLOSECONF" }
+ ,{ GSN_FSCLOSEREF, "FSCLOSEREF" }
+ ,{ GSN_FSCLOSEREQ, "FSCLOSEREQ" }
+ ,{ GSN_FSOPENCONF, "FSOPENCONF" }
+ ,{ GSN_FSOPENREF, "FSOPENREF" }
+ ,{ GSN_FSOPENREQ, "FSOPENREQ" }
+ ,{ GSN_FSREADCONF, "FSREADCONF" }
+ ,{ GSN_FSREADREF, "FSREADREF" }
+ ,{ GSN_FSREADREQ, "FSREADREQ" }
+ ,{ GSN_FSSYNCCONF, "FSSYNCCONF" }
+ ,{ GSN_FSSYNCREF, "FSSYNCREF" }
+ ,{ GSN_FSSYNCREQ, "FSSYNCREQ" }
+ ,{ GSN_FSWRITECONF, "FSWRITECONF" }
+ ,{ GSN_FSWRITEREF, "FSWRITEREF" }
+ ,{ GSN_FSWRITEREQ, "FSWRITEREQ" }
+ ,{ GSN_FSAPPENDCONF, "FSAPPENDCONF" }
+ ,{ GSN_FSAPPENDREF, "FSAPPENDREF" }
+ ,{ GSN_FSAPPENDREQ, "FSAPPENDREQ" }
+ ,{ GSN_FSREMOVECONF, "FSREMOVECONF" }
+ ,{ GSN_FSREMOVEREF, "FSREMOVEREF" }
+ ,{ GSN_FSREMOVEREQ, "FSREMOVEREQ" }
+ ,{ GSN_GCP_ABORT, "GCP_ABORT" }
+ ,{ GSN_GCP_ABORTED, "GCP_ABORTED" }
+ ,{ GSN_GCP_COMMIT, "GCP_COMMIT" }
+ ,{ GSN_GCP_NODEFINISH, "GCP_NODEFINISH" }
+ ,{ GSN_GCP_NOMORETRANS, "GCP_NOMORETRANS" }
+ ,{ GSN_GCP_PREPARE, "GCP_PREPARE" }
+ ,{ GSN_GCP_PREPARECONF, "GCP_PREPARECONF" }
+ ,{ GSN_GCP_PREPAREREF, "GCP_PREPAREREF" }
+ ,{ GSN_GCP_SAVECONF, "GCP_SAVECONF" }
+ ,{ GSN_GCP_SAVEREF, "GCP_SAVEREF" }
+ ,{ GSN_GCP_SAVEREQ, "GCP_SAVEREQ" }
+ ,{ GSN_GCP_TCFINISHED, "GCP_TCFINISHED" }
+ ,{ GSN_GET_TABINFOREF, "GET_TABINFOREF" }
+ ,{ GSN_GET_TABINFOREQ, "GET_TABINFOREQ" }
+ ,{ GSN_GET_TABINFO_CONF, "GET_TABINFO_CONF" }
+ ,{ GSN_GETGCICONF, "GETGCICONF" }
+ ,{ GSN_GETGCIREQ, "GETGCIREQ" }
+ ,{ GSN_HOT_SPAREREP, "HOT_SPAREREP" }
+ ,{ GSN_INCL_NODECONF, "INCL_NODECONF" }
+ ,{ GSN_INCL_NODEREF, "INCL_NODEREF" }
+ ,{ GSN_INCL_NODEREQ, "INCL_NODEREQ" }
+ ,{ GSN_LCP_FRAGIDCONF, "LCP_FRAGIDCONF" }
+ ,{ GSN_LCP_FRAGIDREF, "LCP_FRAGIDREF" }
+ ,{ GSN_LCP_FRAGIDREQ, "LCP_FRAGIDREQ" }
+ ,{ GSN_LCP_HOLDOPCONF, "LCP_HOLDOPCONF" }
+ ,{ GSN_LCP_HOLDOPREF, "LCP_HOLDOPREF" }
+ ,{ GSN_LCP_HOLDOPREQ, "LCP_HOLDOPREQ" }
+ ,{ GSN_LQH_RESTART_OP, "LQH_RESTART_OP" }
+ ,{ GSN_LQH_TRANSCONF, "LQH_TRANSCONF" }
+ ,{ GSN_LQH_TRANSREQ, "LQH_TRANSREQ" }
+ ,{ GSN_LQHADDATTCONF, "LQHADDATTCONF" }
+ ,{ GSN_LQHADDATTREF, "LQHADDATTREF" }
+ ,{ GSN_LQHADDATTREQ, "LQHADDATTREQ" }
+ ,{ GSN_LQHFRAGCONF, "LQHFRAGCONF" }
+ ,{ GSN_LQHFRAGREF, "LQHFRAGREF" }
+ ,{ GSN_LQHFRAGREQ, "LQHFRAGREQ" }
+ ,{ GSN_LQHKEYCONF, "LQHKEYCONF" }
+ ,{ GSN_LQHKEYREF, "LQHKEYREF" }
+ ,{ GSN_LQHKEYREQ, "LQHKEYREQ" }
+ ,{ GSN_MASTER_GCPCONF, "MASTER_GCPCONF" }
+ ,{ GSN_MASTER_GCPREF, "MASTER_GCPREF" }
+ ,{ GSN_MASTER_GCPREQ, "MASTER_GCPREQ" }
+ ,{ GSN_MASTER_LCPCONF, "MASTER_LCPCONF" }
+ ,{ GSN_MASTER_LCPREF, "MASTER_LCPREF" }
+ ,{ GSN_MASTER_LCPREQ, "MASTER_LCPREQ" }
+ ,{ GSN_MEMCHECKCONF, "MEMCHECKCONF" }
+ ,{ GSN_MEMCHECKREQ, "MEMCHECKREQ" }
+ ,{ GSN_NDB_FAILCONF, "NDB_FAILCONF" }
+ ,{ GSN_NDB_STARTCONF, "NDB_STARTCONF" }
+ ,{ GSN_NDB_STARTREF, "NDB_STARTREF" }
+ ,{ GSN_NDB_STARTREQ, "NDB_STARTREQ" }
+ ,{ GSN_NDB_STTOR, "NDB_STTOR" }
+ ,{ GSN_NDB_STTORRY, "NDB_STTORRY" }
+ ,{ GSN_NDB_TAMPER, "NDB_TAMPER" }
+ ,{ GSN_NEXT_SCANCONF, "NEXT_SCANCONF" }
+ ,{ GSN_NEXT_SCANREF, "NEXT_SCANREF" }
+ ,{ GSN_NEXT_SCANREQ, "NEXT_SCANREQ" }
+ ,{ GSN_NEXTOPERATION, "NEXTOPERATION" }
+ ,{ GSN_NF_COMPLETEREP, "NF_COMPLETEREP" }
+ ,{ GSN_NODE_STATESCONF, "NODE_STATESCONF" }
+ ,{ GSN_NODE_STATESREF, "NODE_STATESREF" }
+ ,{ GSN_NODE_STATESREQ, "NODE_STATESREQ" }
+ ,{ GSN_OPEN_COMCONF, "OPEN_COMCONF" }
+ ,{ GSN_OPEN_COMREF, "OPEN_COMREF" }
+ ,{ GSN_OPEN_COMREQ, "OPEN_COMREQ" }
+ ,{ GSN_PACKED_SIGNAL, "PACKED_SIGNAL" }
+ ,{ GSN_PREP_FAILCONF, "PREP_FAILCONF" }
+ ,{ GSN_PREP_FAILREF, "PREP_FAILREF" }
+ ,{ GSN_PREP_FAILREQ, "PREP_FAILREQ" }
+ ,{ GSN_PRES_TOCONF, "PRES_TOCONF" }
+ ,{ GSN_PRES_TOREQ, "PRES_TOREQ" }
+ ,{ GSN_READ_NODESCONF, "READ_NODESCONF" }
+ ,{ GSN_READ_NODESREF, "READ_NODESREF" }
+ ,{ GSN_READ_NODESREQ, "READ_NODESREQ" }
+ ,{ GSN_SCAN_FRAGCONF, "SCAN_FRAGCONF" }
+ ,{ GSN_SCAN_FRAGREF, "SCAN_FRAGREF" }
+ ,{ GSN_SCAN_FRAGREQ, "SCAN_FRAGREQ" }
+ ,{ GSN_SCAN_HBREP, "SCAN_HBREP" }
+ ,{ GSN_SCAN_PROCCONF, "SCAN_PROCCONF" }
+ ,{ GSN_SCAN_PROCREQ, "SCAN_PROCREQ" }
+ ,{ GSN_SEND_PACKED, "SEND_PACKED" }
+ ,{ GSN_SET_LOGLEVELORD, "SET_LOGLEVELORD" }
+ ,{ GSN_SHRINKCHECK2, "SHRINKCHECK2" }
+ ,{ GSN_SIZEALT_ACK, "SIZEALT_ACK" }
+ ,{ GSN_SIZEALT_REP, "SIZEALT_REP" }
+ ,{ GSN_SR_FRAGIDCONF, "SR_FRAGIDCONF" }
+ ,{ GSN_SR_FRAGIDREF, "SR_FRAGIDREF" }
+ ,{ GSN_SR_FRAGIDREQ, "SR_FRAGIDREQ" }
+ ,{ GSN_START_COPYCONF, "START_COPYCONF" }
+ ,{ GSN_START_COPYREF, "START_COPYREF" }
+ ,{ GSN_START_COPYREQ, "START_COPYREQ" }
+ ,{ GSN_START_EXEC_SR, "START_EXEC_SR" }
+ ,{ GSN_START_FRAGCONF, "START_FRAGCONF" }
+ ,{ GSN_START_FRAGREF, "START_FRAGREF" }
+ ,{ GSN_START_FRAGREQ, "START_FRAGREQ" }
+ ,{ GSN_START_LCP_REF, "START_LCP_REF" }
+ ,{ GSN_START_LCP_ROUND, "START_LCP_ROUND" }
+ ,{ GSN_START_MECONF, "START_MECONF" }
+ ,{ GSN_START_MEREF, "START_MEREF" }
+ ,{ GSN_START_MEREQ, "START_MEREQ" }
+ ,{ GSN_START_PERMCONF, "START_PERMCONF" }
+ ,{ GSN_START_PERMREF, "START_PERMREF" }
+ ,{ GSN_START_PERMREQ, "START_PERMREQ" }
+ ,{ GSN_START_RECCONF, "START_RECCONF" }
+ ,{ GSN_START_RECREF, "START_RECREF" }
+ ,{ GSN_START_RECREQ, "START_RECREQ" }
+ ,{ GSN_START_TOCONF, "START_TOCONF" }
+ ,{ GSN_START_TOREQ, "START_TOREQ" }
+ ,{ GSN_STORED_PROCCONF, "STORED_PROCCONF" }
+ ,{ GSN_STORED_PROCREF, "STORED_PROCREF" }
+ ,{ GSN_STORED_PROCREQ, "STORED_PROCREQ" }
+ ,{ GSN_STTOR, "STTOR" }
+ ,{ GSN_STTORRY, "STTORRY" }
+ ,{ GSN_SYSTEM_ERROR, "SYSTEM_ERROR" }
+ ,{ GSN_TAB_COMMITCONF, "TAB_COMMITCONF" }
+ ,{ GSN_TAB_COMMITREF, "TAB_COMMITREF" }
+ ,{ GSN_TAB_COMMITREQ, "TAB_COMMITREQ" }
+ ,{ GSN_TAKE_OVERTCCONF, "TAKE_OVERTCCONF" }
+ ,{ GSN_TAKE_OVERTCREQ, "TAKE_OVERTCREQ" }
+ ,{ GSN_TC_CLOPSIZECONF, "TC_CLOPSIZECONF" }
+ ,{ GSN_TC_CLOPSIZEREQ, "TC_CLOPSIZEREQ" }
+ ,{ GSN_TC_SCHVERCONF, "TC_SCHVERCONF" }
+ ,{ GSN_TC_SCHVERREQ, "TC_SCHVERREQ" }
+ ,{ GSN_TCGETOPSIZECONF, "TCGETOPSIZECONF" }
+ ,{ GSN_TCGETOPSIZEREQ, "TCGETOPSIZEREQ" }
+ ,{ GSN_TEST_ORD, "TEST_ORD" }
+ ,{ GSN_TESTSIG, "TESTSIG" }
+ ,{ GSN_TIME_SIGNAL, "TIME_SIGNAL" }
+ ,{ GSN_TUP_ABORTREQ, "TUP_ABORTREQ" }
+ ,{ GSN_TUP_ADD_ATTCONF, "TUP_ADD_ATTCONF" }
+ ,{ GSN_TUP_ADD_ATTRREF, "TUP_ADD_ATTRREF" }
+ ,{ GSN_TUP_ADD_ATTRREQ, "TUP_ADD_ATTRREQ" }
+ ,{ GSN_TUP_ATTRINFO, "TUP_ATTRINFO" }
+ ,{ GSN_TUP_COMMITREQ, "TUP_COMMITREQ" }
+ ,{ GSN_TUP_LCPCONF, "TUP_LCPCONF" }
+ ,{ GSN_TUP_LCPREF, "TUP_LCPREF" }
+ ,{ GSN_TUP_LCPREQ, "TUP_LCPREQ" }
+ ,{ GSN_TUP_LCPSTARTED, "TUP_LCPSTARTED" }
+ ,{ GSN_TUP_PREPLCPCONF, "TUP_PREPLCPCONF" }
+ ,{ GSN_TUP_PREPLCPREF, "TUP_PREPLCPREF" }
+ ,{ GSN_TUP_PREPLCPREQ, "TUP_PREPLCPREQ" }
+ ,{ GSN_TUP_SRCONF, "TUP_SRCONF" }
+ ,{ GSN_TUP_SRREF, "TUP_SRREF" }
+ ,{ GSN_TUP_SRREQ, "TUP_SRREQ" }
+ ,{ GSN_TUPFRAGCONF, "TUPFRAGCONF" }
+ ,{ GSN_TUPFRAGREF, "TUPFRAGREF" }
+ ,{ GSN_TUPFRAGREQ, "TUPFRAGREQ" }
+ ,{ GSN_TUPKEYCONF, "TUPKEYCONF" }
+ ,{ GSN_TUPKEYREF, "TUPKEYREF" }
+ ,{ GSN_TUPKEYREQ, "TUPKEYREQ" }
+ ,{ GSN_TUPRELEASECONF, "TUPRELEASECONF" }
+ ,{ GSN_TUPRELEASEREF, "TUPRELEASEREF" }
+ ,{ GSN_TUPRELEASEREQ, "TUPRELEASEREQ" }
+ ,{ GSN_TUPSEIZECONF, "TUPSEIZECONF" }
+ ,{ GSN_TUPSEIZEREF, "TUPSEIZEREF" }
+ ,{ GSN_TUPSEIZEREQ, "TUPSEIZEREQ" }
+ ,{ GSN_UNBLO_DICTCONF, "UNBLO_DICTCONF" }
+ ,{ GSN_UNBLO_DICTREQ, "UNBLO_DICTREQ" }
+ ,{ GSN_UPDATE_TOCONF, "UPDATE_TOCONF" }
+ ,{ GSN_UPDATE_TOREF, "UPDATE_TOREF" }
+ ,{ GSN_UPDATE_TOREQ, "UPDATE_TOREQ" }
+ ,{ GSN_VOTE_MASTERORD, "VOTE_MASTERORD" }
+ ,{ GSN_TUP_ALLOCREQ, "TUP_ALLOCREQ" }
+ ,{ GSN_LQH_ALLOCREQ, "LQH_ALLOCREQ" }
+ ,{ GSN_TUP_DEALLOCREQ, "TUP_DEALLOCREQ" }
+ ,{ GSN_TUP_WRITELOG_REQ, "TUP_WRITELOG_REQ" }
+ ,{ GSN_LQH_WRITELOG_REQ, "LQH_WRITELOG_REQ" }
+
+ ,{ GSN_STATISTICS_REQ, "STATISTICS_REQ" }
+ ,{ GSN_START_ORD, "START_ORD" }
+ ,{ GSN_STOP_ORD, "STOP_ORD" }
+ ,{ GSN_TAMPER_ORD, "TAMPER_ORD" }
+ ,{ GSN_SET_VAR_REQ, "SET_VAR_REQ" }
+ ,{ GSN_SET_VAR_CONF, "SET_VAR_CONF" }
+ ,{ GSN_SET_VAR_REF, "SET_VAR_REF" }
+ ,{ GSN_STATISTICS_CONF, "STATISTICS_CONF" }
+
+ ,{ GSN_EVENT_SUBSCRIBE_REQ, "EVENT_SUBSCRIBE_REQ" }
+ ,{ GSN_EVENT_SUBSCRIBE_CONF, "EVENT_SUBSCRIBE_CONF" }
+ ,{ GSN_EVENT_SUBSCRIBE_REF, "EVENT_SUBSCRIBE_REF" }
+ ,{ GSN_ACC_COM_BLOCK, "ACC_COM_BLOCK" }
+ ,{ GSN_ACC_COM_UNBLOCK, "ACC_COM_UNBLOCK" }
+ ,{ GSN_TUP_COM_BLOCK, "TUP_COM_BLOCK" }
+ ,{ GSN_TUP_COM_UNBLOCK, "TUP_COM_UNBLOCK" }
+ ,{ GSN_DUMP_STATE_ORD, "DUMP_STATE_ORD" }
+
+ ,{ GSN_START_INFOREQ, "START_INFOREQ" }
+ ,{ GSN_START_INFOREF, "START_INFOREF" }
+ ,{ GSN_START_INFOCONF, "START_INFOCONF" }
+
+ ,{ GSN_CHECKNODEGROUPSREQ, "CHECKNODEGROUPSREQ" }
+ ,{ GSN_CHECKNODEGROUPSCONF, "CHECKNODEGROUPSCONF" }
+
+ ,{ GSN_ARBIT_CFG, "ARBIT_CFG" }
+ ,{ GSN_ARBIT_PREPREQ, "ARBIT_PREPREQ" }
+ ,{ GSN_ARBIT_PREPCONF, "ARBIT_PREPCONF" }
+ ,{ GSN_ARBIT_PREPREF, "ARBIT_PREPREF" }
+ ,{ GSN_ARBIT_STARTREQ, "ARBIT_STARTREQ" }
+ ,{ GSN_ARBIT_STARTCONF, "ARBIT_STARTCONF" }
+ ,{ GSN_ARBIT_STARTREF, "ARBIT_STARTREF" }
+ ,{ GSN_ARBIT_CHOOSEREQ, "ARBIT_CHOOSEREQ" }
+ ,{ GSN_ARBIT_CHOOSECONF, "ARBIT_CHOOSECONF" }
+ ,{ GSN_ARBIT_CHOOSEREF, "ARBIT_CHOOSEREF" }
+ ,{ GSN_ARBIT_STOPORD, "ARBIT_STOPORD" }
+ ,{ GSN_ARBIT_STOPREP, "ARBIT_STOPREP" }
+
+ ,{ GSN_TC_COMMIT_ACK, "TC_COMMIT_ACK" }
+ ,{ GSN_REMOVE_MARKER_ORD, "REMOVE_MARKER_ORD" }
+
+ ,{ GSN_NODE_STATE_REP, "NODE_STATE_REP" }
+ ,{ GSN_CHANGE_NODE_STATE_REQ, "CHANGE_NODE_STATE_REQ" }
+ ,{ GSN_CHANGE_NODE_STATE_CONF, "CHANGE_NODE_STATE_CONF" }
+
+ ,{ GSN_BLOCK_COMMIT_ORD, "BLOCK_COMMIT_ORD" }
+ ,{ GSN_UNBLOCK_COMMIT_ORD, "UNBLOCK_COMMIT_ORD" }
+
+ ,{ GSN_DIH_SWITCH_REPLICA_REQ, "DIH_SWITCH_REPLICA_REQ" }
+ ,{ GSN_DIH_SWITCH_REPLICA_REF, "DIH_SWITCH_REPLICA_REF" }
+ ,{ GSN_DIH_SWITCH_REPLICA_CONF, "DIH_SWITCH_REPLICA_CONF" }
+
+ ,{ GSN_STOP_PERM_REQ, "STOP_PERM_REQ" }
+ ,{ GSN_STOP_PERM_REF, "STOP_PERM_REF" }
+ ,{ GSN_STOP_PERM_CONF, "STOP_PERM_CONF" }
+
+ ,{ GSN_STOP_ME_REQ, "STOP_ME_REQ" }
+ ,{ GSN_STOP_ME_REF, "STOP_ME_REF" }
+ ,{ GSN_STOP_ME_CONF, "STOP_ME_CONF" }
+
+ ,{ GSN_WAIT_GCP_REQ, "WAIT_GCP_REQ" }
+ ,{ GSN_WAIT_GCP_REF, "WAIT_GCP_REF" }
+ ,{ GSN_WAIT_GCP_CONF, "WAIT_GCP_CONF" }
+
+ ,{ GSN_STOP_REQ, "STOP_REQ" }
+ ,{ GSN_STOP_REF, "STOP_REF" }
+
+ ,{ GSN_ABORT_ALL_REQ, "ABORT_ALL_REQ" }
+ ,{ GSN_ABORT_ALL_REF, "ABORT_ALL_REF" }
+ ,{ GSN_ABORT_ALL_CONF, "ABORT_ALL_CONF" }
+
+ ,{ GSN_DROP_TABLE_REQ, "DROP_TABLE_REQ" }
+ ,{ GSN_DROP_TABLE_REF, "DROP_TABLE_REF" }
+ ,{ GSN_DROP_TABLE_CONF, "DROP_TABLE_CONF" }
+
+ ,{ GSN_DROP_TAB_REQ, "DROP_TAB_REQ" }
+ ,{ GSN_DROP_TAB_REF, "DROP_TAB_REF" }
+ ,{ GSN_DROP_TAB_CONF, "DROP_TAB_CONF" }
+
+ ,{ GSN_PREP_DROP_TAB_REQ, "PREP_DROP_TAB_REQ" }
+ ,{ GSN_PREP_DROP_TAB_REF, "PREP_DROP_TAB_REF" }
+ ,{ GSN_PREP_DROP_TAB_CONF, "PREP_DROP_TAB_CONF" }
+
+ ,{ GSN_WAIT_DROP_TAB_REQ, "WAIT_DROP_TAB_REQ" }
+ ,{ GSN_WAIT_DROP_TAB_REF, "WAIT_DROP_TAB_REF" }
+ ,{ GSN_WAIT_DROP_TAB_CONF, "WAIT_DROP_TAB_CONF" }
+
+ ,{ GSN_CREATE_TRIG_REQ, "CREATE_TRIG_REQ" }
+ ,{ GSN_CREATE_TRIG_CONF, "CREATE_TRIG_CONF" }
+ ,{ GSN_CREATE_TRIG_REF, "CREATE_TRIG_REF" }
+ ,{ GSN_ALTER_TRIG_REQ, "ALTER_TRIG_REQ" }
+ ,{ GSN_ALTER_TRIG_CONF, "ALTER_TRIG_CONF" }
+ ,{ GSN_ALTER_TRIG_REF, "ALTER_TRIG_REF" }
+ ,{ GSN_DROP_TRIG_REQ, "DROP_TRIG_REQ" }
+ ,{ GSN_DROP_TRIG_CONF, "DROP_TRIG_CONF" }
+ ,{ GSN_DROP_TRIG_REF, "DROP_TRIG_REF" }
+ ,{ GSN_FIRE_TRIG_ORD, "FIRE_TRIG_ORD" }
+ ,{ GSN_TRIG_ATTRINFO, "TRIG_ATTRINFO" }
+
+ ,{ GSN_CREATE_INDX_REQ, "CREATE_INDX_REQ" }
+ ,{ GSN_CREATE_INDX_CONF, "CREATE_INDX_CONF" }
+ ,{ GSN_CREATE_INDX_REF, "CREATE_INDX_REF" }
+ ,{ GSN_DROP_INDX_REQ, "DROP_INDX_REQ" }
+ ,{ GSN_DROP_INDX_CONF, "DROP_INDX_CONF" }
+ ,{ GSN_DROP_INDX_REF, "DROP_INDX_REF" }
+ ,{ GSN_ALTER_INDX_REQ, "ALTER_INDX_REQ" }
+ ,{ GSN_ALTER_INDX_CONF, "ALTER_INDX_CONF" }
+ ,{ GSN_ALTER_INDX_REF, "ALTER_INDX_REF" }
+ ,{ GSN_TCINDXREQ, "TCINDXREQ" }
+ ,{ GSN_TCINDXCONF, "TCINDXCONF" }
+ ,{ GSN_TCINDXREF, "TCINDXREF" }
+ ,{ GSN_INDXKEYINFO, "INDXKEYINFO" }
+ ,{ GSN_INDXATTRINFO, "INDXATTRINFO" }
+ ,{ GSN_BUILDINDXREQ, "BUILDINDXREQ" }
+ ,{ GSN_BUILDINDXCONF, "BUILDINDXCONF" }
+ ,{ GSN_BUILDINDXREF, "BUILDINDXREF" }
+ //,{ GSN_TCINDXNEXTREQ, "TCINDXNEXTREQ" }
+ //,{ GSN_TCINDEXNEXTCONF, "TCINDEXNEXTCONF" }
+ //,{ GSN_TCINDEXNEXREF, "TCINDEXNEXREF" }
+
+ ,{ GSN_CREATE_EVNT_REQ, "CREATE_EVNT_REQ" }
+ ,{ GSN_CREATE_EVNT_CONF, "CREATE_EVNT_CONF" }
+ ,{ GSN_CREATE_EVNT_REF, "CREATE_EVNT_REF" }
+
+ ,{ GSN_SUMA_START_ME, "SUMA_START_ME" }
+ ,{ GSN_SUMA_HANDOVER_REQ, "SUMA_HANDOVER_REQ"}
+ ,{ GSN_SUMA_HANDOVER_CONF, "SUMA_HANDOVER_CONF"}
+
+ ,{ GSN_DROP_EVNT_REQ, "DROP_EVNT_REQ" }
+ ,{ GSN_DROP_EVNT_CONF, "DROP_EVNT_CONF" }
+ ,{ GSN_DROP_EVNT_REF, "DROP_EVNT_REF" }
+
+ ,{ GSN_BACKUP_TRIG_REQ, "BACKUP_TRIG_REQ" }
+ ,{ GSN_BACKUP_REQ, "BACKUP_REQ" }
+ ,{ GSN_BACKUP_DATA, "BACKUP_DATA" }
+ ,{ GSN_BACKUP_REF, "BACKUP_REF" }
+ ,{ GSN_BACKUP_CONF, "BACKUP_CONF" }
+ ,{ GSN_ABORT_BACKUP_ORD, "ABORT_BACKUP_ORD" }
+ ,{ GSN_BACKUP_ABORT_REP, "BACKUP_ABORT_REP" }
+ ,{ GSN_BACKUP_COMPLETE_REP, "BACKUP_COMPLETE_REP" }
+ ,{ GSN_BACKUP_NF_COMPLETE_REP, "BACKUP_NF_COMPLETE_REP" }
+ ,{ GSN_DEFINE_BACKUP_REQ, "DEFINE_BACKUP_REQ" }
+ ,{ GSN_DEFINE_BACKUP_REF, "DEFINE_BACKUP_REF" }
+ ,{ GSN_DEFINE_BACKUP_CONF, "DEFINE_BACKUP_CONF" }
+ ,{ GSN_START_BACKUP_REQ, "START_BACKUP_REQ" }
+ ,{ GSN_START_BACKUP_REF, "START_BACKUP_REF" }
+ ,{ GSN_START_BACKUP_CONF, "START_BACKUP_CONF" }
+ ,{ GSN_BACKUP_FRAGMENT_REQ, "BACKUP_FRAGMENT_REQ" }
+ ,{ GSN_BACKUP_FRAGMENT_REF, "BACKUP_FRAGMENT_REF" }
+ ,{ GSN_BACKUP_FRAGMENT_CONF, "BACKUP_FRAGMENT_CONF" }
+ ,{ GSN_STOP_BACKUP_REQ, "STOP_BACKUP_REQ" }
+ ,{ GSN_STOP_BACKUP_REF, "STOP_BACKUP_REF" }
+ ,{ GSN_STOP_BACKUP_CONF, "STOP_BACKUP_CONF" }
+ ,{ GSN_BACKUP_STATUS_REQ, "BACKUP_STATUS_REQ" }
+ ,{ GSN_BACKUP_STATUS_REF, "BACKUP_STATUS_REF" }
+ ,{ GSN_BACKUP_STATUS_CONF, "BACKUP_STATUS_CONF" }
+ ,{ GSN_SIGNAL_DROPPED_REP, "SIGNAL_DROPPED_REP" }
+ ,{ GSN_CONTINUE_FRAGMENTED, "CONTINUE_FRAGMENTED" }
+
+ /** Util Block Services **/
+ ,{ GSN_UTIL_SEQUENCE_REQ, "UTIL_SEQUENCE_REQ" }
+ ,{ GSN_UTIL_SEQUENCE_REF, "UTIL_SEQUENCE_REF" }
+ ,{ GSN_UTIL_SEQUENCE_CONF, "UTIL_SEQUENCE_CONF" }
+ ,{ GSN_UTIL_PREPARE_REQ, "UTIL_PREPARE_REQ" }
+ ,{ GSN_UTIL_PREPARE_CONF, "UTIL_PREPARE_CONF" }
+ ,{ GSN_UTIL_PREPARE_REF, "UTIL_PREPARE_REF" }
+ ,{ GSN_UTIL_EXECUTE_REQ, "UTIL_EXECUTE_REQ" }
+ ,{ GSN_UTIL_EXECUTE_CONF, "UTIL_EXECUTE_CONF" }
+ ,{ GSN_UTIL_EXECUTE_REF, "UTIL_EXECUTE_REF" }
+ ,{ GSN_UTIL_RELEASE_REQ, "UTIL_RELEASE_REQ" }
+ ,{ GSN_UTIL_RELEASE_CONF, "UTIL_RELEASE_CONF" }
+ ,{ GSN_UTIL_RELEASE_REF, "UTIL_RELASE_REF" }
+
+ ,{ GSN_GREP_CREATE_REQ, "GREP_CREATE_REQ" },
+ { GSN_GREP_CREATE_REF, "GREP_CREATE_REF" },
+ { GSN_GREP_CREATE_CONF, "GREP_CREATE_CONF" },
+ { GSN_GREP_START_REQ, "GREP_START_REQ" },
+ { GSN_GREP_START_REF, "GREP_START_REF" },
+ { GSN_GREP_START_CONF, "GREP_START_CONF" },
+ { GSN_GREP_SYNC_REQ, "GREP_SYNC_REQ" },
+ { GSN_GREP_SYNC_REF, "GREP_SYNC_REF" },
+ { GSN_GREP_SYNC_CONF, "GREP_SYNC_CONF" },
+ //{ GSN_REP_CONNECT_REQ, "REP_CONNECT_REQ" }, Not used
+ //{ GSN_REP_CONNECT_REF, "REP_CONNECT_REF" }, Not used
+ //{ GSN_REP_CONNECT_CONF, "REP_CONNECT_CONF" }, Not used
+ { GSN_REP_WAITGCP_REQ, "REP_WAIT_GCP_REQ" },
+ { GSN_REP_WAITGCP_REF, "REP_WAIT_GCP_REF" },
+ { GSN_REP_WAITGCP_CONF, "REP_WAIT_GCP_CONF" },
+ { GSN_GREP_WAITGCP_REQ, "GREP_WAIT_GCP_REQ" },
+ { GSN_GREP_WAITGCP_REF, "GREP_WAIT_GCP_REF" },
+ { GSN_GREP_WAITGCP_CONF, "GREP_WAIT_GCP_CONF" }
+
+ /* Suma Block Services **/
+ ,{ GSN_SUB_CREATE_REQ, "SUB_CREATE_REQ" }
+ ,{ GSN_SUB_CREATE_REF, "SUB_CREATE_REF" }
+ ,{ GSN_SUB_CREATE_CONF, "SUB_CREATE_CONF" }
+ ,{ GSN_SUB_START_REQ, "SUB_START_REQ" }
+ ,{ GSN_SUB_START_REF, "SUB_START_REF" }
+ ,{ GSN_SUB_START_CONF, "SUB_START_CONF" }
+ ,{ GSN_SUB_STOP_REQ, "SUB_STOP_REQ" }
+ ,{ GSN_SUB_STOP_REF, "SUB_STOP_REF" }
+ ,{ GSN_SUB_STOP_CONF, "SUB_STOP_CONF" }
+ ,{ GSN_SUB_SYNC_REQ, "SUB_SYNC_REQ" }
+ ,{ GSN_SUB_SYNC_REF, "SUB_SYNC_REF" }
+ ,{ GSN_SUB_SYNC_CONF, "SUB_SYNC_CONF" }
+ ,{ GSN_SUB_META_DATA, "SUB_META_DATA" }
+ ,{ GSN_SUB_TABLE_DATA, "SUB_TABLE_DATA" }
+ ,{ GSN_SUB_SYNC_CONTINUE_REQ, "SUB_SYNC_CONTINUE_REQ" }
+ ,{ GSN_SUB_SYNC_CONTINUE_REF, "SUB_SYNC_CONTINUE_REF" }
+ ,{ GSN_SUB_SYNC_CONTINUE_CONF, "SUB_SYNC_CONTINUE_CONF" }
+ ,{ GSN_SUB_GCP_COMPLETE_REP, "SUB_GCP_COMPLETE_REP" }
+ ,{ GSN_SUB_GCP_COMPLETE_ACC, "SUB_GCP_COMPLETE_ACC" }
+
+ ,{ GSN_CREATE_SUBID_REQ, "CREATE_SUBID_REQ" }
+ ,{ GSN_CREATE_SUBID_REF, "CREATE_SUBID_REF" }
+ ,{ GSN_CREATE_SUBID_CONF, "CREATE_SUBID_CONF" }
+
+ ,{ GSN_CREATE_TABLE_REQ, "CREATE_TABLE_REQ" }
+ ,{ GSN_CREATE_TABLE_REF, "CREATE_TABLE_REF" }
+ ,{ GSN_CREATE_TABLE_CONF, "CREATE_TABLE_CONF" }
+
+ ,{ GSN_CREATE_TAB_REQ, "CREATE_TAB_REQ" }
+ ,{ GSN_CREATE_TAB_REF, "CREATE_TAB_REF" }
+ ,{ GSN_CREATE_TAB_CONF, "CREATE_TAB_CONF" }
+
+ ,{ GSN_ALTER_TABLE_REQ, "ALTER_TABLE_REQ" }
+ ,{ GSN_ALTER_TABLE_REF, "ALTER_TABLE_REF" }
+ ,{ GSN_ALTER_TABLE_CONF, "ALTER_TABLE_CONF" }
+
+ ,{ GSN_ALTER_TAB_REQ, "ALTER_TAB_REQ" }
+ ,{ GSN_ALTER_TAB_REF, "ALTER_TAB_REF" }
+ ,{ GSN_ALTER_TAB_CONF, "ALTER_TAB_CONF" }
+
+ ,{ GSN_CREATE_FRAGMENTATION_REQ, "CREATE_FRAGMENTATION_REQ" }
+ ,{ GSN_CREATE_FRAGMENTATION_REF, "CREATE_FRAGMENTATION_REF" }
+ ,{ GSN_CREATE_FRAGMENTATION_CONF, "CREATE_FRAGMENTATION_CONF" }
+
+ ,{ GSN_UTIL_CREATE_LOCK_REQ, "UTIL_CREATE_LOCK_REQ" }
+ ,{ GSN_UTIL_CREATE_LOCK_REF, "UTIL_CREATE_LOCK_REF" }
+ ,{ GSN_UTIL_CREATE_LOCK_CONF, "UTIL_CREATE_LOCK_CONF" }
+ ,{ GSN_UTIL_DESTROY_LOCK_REQ, "UTIL_DESTROY_LOCK_REQ" }
+ ,{ GSN_UTIL_DESTROY_LOCK_REF, "UTIL_DESTROY_LOCK_REF" }
+ ,{ GSN_UTIL_DESTROY_LOCK_CONF, "UTIL_DESTROY_LOCK_CONF" }
+ ,{ GSN_UTIL_LOCK_REQ, "UTIL_LOCK_REQ" }
+ ,{ GSN_UTIL_LOCK_REF, "UTIL_LOCK_REF" }
+ ,{ GSN_UTIL_LOCK_CONF, "UTIL_LOCK_CONF" }
+ ,{ GSN_UTIL_UNLOCK_REQ, "UTIL_UNLOCK_REQ" }
+ ,{ GSN_UTIL_UNLOCK_REF, "UTIL_UNLOCK_REF" }
+ ,{ GSN_UTIL_UNLOCK_CONF, "UTIL_UNLOCK_CONF" }
+
+ /* TUX */
+ ,{ GSN_TUXFRAGREQ, "TUXFRAGREQ" }
+ ,{ GSN_TUXFRAGCONF, "TUXFRAGCONF" }
+ ,{ GSN_TUXFRAGREF, "TUXFRAGREF" }
+ ,{ GSN_TUX_ADD_ATTRREQ, "TUX_ADD_ATTRREQ" }
+ ,{ GSN_TUX_ADD_ATTRCONF, "TUX_ADD_ATTRCONF" }
+ ,{ GSN_TUX_ADD_ATTRREF, "TUX_ADD_ATTRREF" }
+ ,{ GSN_TUX_MAINT_REQ, "TUX_MAINT_REQ" }
+ ,{ GSN_TUX_MAINT_CONF, "TUX_MAINT_CONF" }
+ ,{ GSN_TUX_MAINT_REF, "TUX_MAINT_REF" }
+ ,{ GSN_TUP_READ_ATTRS, "TUP_READ_ATTRS" }
+ ,{ GSN_TUP_QUERY_TH, "TUP_QUERY_TH" }
+ ,{ GSN_TUP_STORE_TH, "TUP_STORE_TH" }
+ ,{ GSN_TUX_BOUND_INFO, "TUX_BOUND_INFO" }
+ ,{ GSN_ACC_LOCKREQ, "ACC_LOCKREQ" }
+
+};
+const unsigned short NO_OF_SIGNAL_NAMES = sizeof(SignalNames)/sizeof(GsnName);
diff --git a/ndb/src/common/debugger/signaldata/StartRec.cpp b/ndb/src/common/debugger/signaldata/StartRec.cpp
new file mode 100644
index 00000000000..482e3cb0728
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/StartRec.cpp
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <RefConvert.hpp>
+#include <signaldata/StartRec.hpp>
+
+bool
+printSTART_REC_REQ(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 recBlockNo){
+ StartRecReq * sig = (StartRecReq *) theData;
+
+ fprintf(output, " receivingNodeId: %d senderRef: (%d, %d)\n",
+ sig->receivingNodeId,
+ refToNode(sig->senderRef),
+ refToBlock(sig->senderRef));
+
+ fprintf(output, " keepGci: %d lastCompletedGci: %d newestGci: %d\n",
+ sig->keepGci,
+ sig->lastCompletedGci,
+ sig->newestGci);
+
+ return true;
+}
+
+bool
+printSTART_REC_CONF(FILE * output,
+ const Uint32 * theData,
+ Uint32 len,
+ Uint16 recBlockNo){
+ StartRecConf * sig = (StartRecConf *) theData;
+
+ fprintf(output, " startingNodeId: %d\n",
+ sig->startingNodeId);
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/SumaImpl.cpp b/ndb/src/common/debugger/signaldata/SumaImpl.cpp
new file mode 100644
index 00000000000..558842ed2ba
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/SumaImpl.cpp
@@ -0,0 +1,167 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/SumaImpl.hpp>
+
+bool
+printSUB_CREATE_REQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubCreateReq * const sig = (SubCreateReq *)theData;
+ fprintf(output, " subscriberRef: %x\n", sig->subscriberRef);
+ fprintf(output, " subscriberData: %x\n", sig->subscriberData);
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ fprintf(output, " subscriptionType: %x\n", sig->subscriptionType);
+ fprintf(output, " tableId: %x\n", sig->tableId);
+ return false;
+}
+
+bool
+printSUB_CREATE_CONF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubCreateConf * const sig = (SubCreateConf *)theData;
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ fprintf(output, " subscriberData: %x\n", sig->subscriberData);
+ return false;
+}
+
+bool
+printSUB_START_REQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubStartReq * const sig = (SubStartReq *)theData;
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ fprintf(output, " startPart: %x\n", sig->part);
+ return false;
+}
+
+bool
+printSUB_START_REF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubStartRef * const sig = (SubStartRef *)theData;
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ fprintf(output, " startPart: %x\n", sig->part);
+ fprintf(output, " subscriberData: %x\n", sig->subscriberData);
+ fprintf(output, " err: %x\n", sig->err);
+ return false;
+}
+
+bool
+printSUB_START_CONF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubStartConf * const sig = (SubStartConf *)theData;
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ fprintf(output, " startPart: %x\n", sig->part);
+ fprintf(output, " subscriberData: %x\n", sig->subscriberData);
+ return false;
+}
+
+bool
+printSUB_SYNC_REQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubSyncReq * const sig = (SubSyncReq *)theData;
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ fprintf(output, " syncPart: %x\n", sig->part);
+ return false;
+}
+
+bool
+printSUB_SYNC_REF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubSyncRef * const sig = (SubSyncRef *)theData;
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ fprintf(output, " syncPart: %x\n", sig->part);
+ fprintf(output, " subscriberData: %x\n", sig->subscriberData);
+ fprintf(output, " err: %x\n", sig->err);
+ return false;
+}
+
+bool
+printSUB_SYNC_CONF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubSyncConf * const sig = (SubSyncConf *)theData;
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ fprintf(output, " syncPart: %x\n", sig->part);
+ fprintf(output, " subscriberData: %x\n", sig->subscriberData);
+ return false;
+}
+
+bool
+printSUB_META_DATA(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubMetaData * const sig = (SubMetaData *)theData;
+ fprintf(output, " gci: %x\n", sig->gci);
+ fprintf(output, " senderData: %x\n", sig->senderData);
+ fprintf(output, " subscriberData: %x\n", sig->subscriberData);
+ fprintf(output, " tableId: %x\n", sig->tableId);
+ return false;
+}
+
+bool
+printSUB_TABLE_DATA(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubTableData * const sig = (SubTableData *)theData;
+ fprintf(output, " senderData: %x\n", sig->senderData);
+ fprintf(output, " subscriberData: %x\n", sig->subscriberData);
+ fprintf(output, " gci: %x\n", sig->gci);
+ fprintf(output, " tableId: %x\n", sig->tableId);
+ fprintf(output, " operation: %x\n", sig->operation);
+ fprintf(output, " noOfAttributes: %x\n", sig->noOfAttributes);
+ fprintf(output, " dataSize: %x\n", sig->dataSize);
+ return false;
+}
+
+bool
+printSUB_SYNC_CONTINUE_REQ(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubSyncContinueReq * const sig = (SubSyncContinueReq *)theData;
+ fprintf(output, " subscriberData: %x\n", sig->subscriberData);
+ fprintf(output, " noOfRowsSent: %x\n", sig->noOfRowsSent);
+ return false;
+}
+
+bool
+printSUB_SYNC_CONTINUE_REF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubSyncContinueRef * const sig = (SubSyncContinueRef *)theData;
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ return false;
+}
+
+bool
+printSUB_SYNC_CONTINUE_CONF(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubSyncContinueConf * const sig = (SubSyncContinueConf *)theData;
+ fprintf(output, " subscriptionId: %x\n", sig->subscriptionId);
+ fprintf(output, " subscriptionKey: %x\n", sig->subscriptionKey);
+ return false;
+}
+
+bool
+printSUB_GCP_COMPLETE_REP(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo) {
+ const SubGcpCompleteRep * const sig = (SubGcpCompleteRep *)theData;
+ fprintf(output, " gci: %x\n", sig->gci);
+ return false;
+}
+
diff --git a/ndb/src/common/debugger/signaldata/SystemError.cpp b/ndb/src/common/debugger/signaldata/SystemError.cpp
new file mode 100644
index 00000000000..5ed7dc6b18d
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/SystemError.cpp
@@ -0,0 +1,41 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <kernel_types.h>
+#include <BlockNumbers.h>
+#include <signaldata/SystemError.hpp>
+
+bool
+printSYSTEM_ERROR(FILE * output, const Uint32 * theData, Uint32 len,
+ Uint16 receiverBlockNo){
+
+ const SystemError * const sig = (SystemError *) theData;
+
+ fprintf(output, "errorRef: H\'%.8x\n",
+ sig->errorRef);
+ fprintf(output, "errorCode: %d\n",
+ sig->errorCode);
+ fprintf(output, "data1: H\'%.8x\n",
+ sig->data1);
+ fprintf(output, "data2: H\'%.8x\n",
+ sig->data2);
+
+ return true;
+}
+
+
diff --git a/ndb/src/common/debugger/signaldata/TcIndx.cpp b/ndb/src/common/debugger/signaldata/TcIndx.cpp
new file mode 100644
index 00000000000..6bfa29eff15
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TcIndx.cpp
@@ -0,0 +1,159 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TcIndx.hpp>
+#include <signaldata/TcKeyReq.hpp>
+#include <BlockNumbers.h>
+
+bool
+printTCINDXREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const TcIndxReq * const sig = (TcIndxReq *) theData;
+
+ UintR requestInfo = sig->requestInfo;
+ UintR scanInfo = sig->scanInfo;
+
+ fprintf(output, " apiConnectPtr: H\'%.8x, senderData: H\'%.8x\n",
+ sig->apiConnectPtr, sig->senderData);
+
+ fprintf(output, " Operation: %s, Flags: ",
+ sig->getOperationType(requestInfo) == ZREAD ? "Read" :
+ sig->getOperationType(requestInfo) == ZREAD_EX ? "Read-Ex" :
+ sig->getOperationType(requestInfo) == ZUPDATE ? "Update" :
+ sig->getOperationType(requestInfo) == ZINSERT ? "Insert" :
+ sig->getOperationType(requestInfo) == ZDELETE ? "Delete" :
+ sig->getOperationType(requestInfo) == ZWRITE ? "Write" :
+ "Unknown");
+
+ {
+ if(sig->getDirtyFlag(requestInfo)){
+ fprintf(output, "Dirty ");
+ }
+ if(sig->getStartFlag(requestInfo)){
+ fprintf(output, "Start ");
+ }
+ if (TcKeyReq::getExecuteFlag(sig->requestInfo)) {
+ fprintf(output, "Execute ");
+ }
+ if(sig->getCommitFlag(requestInfo)){
+ fprintf(output, "Commit, Type = ");
+ UintR TcommitType = sig->getCommitType(requestInfo);
+ if (TcommitType == TcIndxReq::CommitIfFailFree) {
+ fprintf(output, "FailFree ");
+ } else if (TcommitType == TcIndxReq::TryCommit) {
+ fprintf(output, "TryCommit ");
+ } else if (TcommitType == TcIndxReq::CommitAsMuchAsPossible) {
+ fprintf(output, "Always ");
+ }//if
+ }
+ if(sig->getSimpleFlag(requestInfo)){
+ fprintf(output, "Simple ");
+ }
+ if(sig->getInterpretedFlag(requestInfo)){
+ fprintf(output, "Interpreted ");
+ }
+ if(sig->getDistributionGroupFlag(requestInfo)){
+ fprintf(output, "DGroup = %d ", sig->distrGroupHashValue);
+ }
+ if(sig->getDistributionKeyFlag(sig->requestInfo)){
+ fprintf(output, "DKey = %d ", sig->distributionKeySize);
+ }
+ fprintf(output, "\n");
+ }
+
+ const int indexLen = sig->getIndexLength(requestInfo);
+ const int attrInThis = sig->getAIInTcIndxReq(requestInfo);
+ fprintf(output,
+ " indexLen: %d, attrLen: %d, AI in this: %d, indexId: %d, "
+ "indexSchemaVer: %d, API Ver: %d\n",
+ indexLen, sig->attrLen, attrInThis,
+ sig->indexId, sig->indexSchemaVersion, sig->getAPIVersion(scanInfo));
+
+ fprintf(output, " transId(1, 2): (H\'%.8x, H\'%.8x)\n -- Variable Data --\n",
+ sig->transId1, sig->transId2);
+
+ Uint32 restLen = (len - 8);
+ const Uint32 * rest = &sig->scanInfo;
+ while(restLen >= 7){
+ fprintf(output,
+ " H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x\n",
+ rest[0], rest[1], rest[2], rest[3],
+ rest[4], rest[5], rest[6]);
+ restLen -= 7;
+ rest += 7;
+ }
+ if(restLen > 0){
+ for(Uint32 i = 0; i<restLen; i++)
+ fprintf(output, " H\'%.8x", rest[i]);
+ fprintf(output, "\n");
+ }
+
+ return true;
+}
+
+bool
+printTCINDXCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ if (receiverBlockNo == API_PACKED) {
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ }
+ else {
+ const TcIndxConf * const sig = (TcIndxConf *) theData;
+
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ Uint32 confInfo = sig->confInfo;
+ Uint32 noOfOp = TcIndxConf::getNoOfOperations(confInfo);
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ fprintf(output, "apiConnectPtr: H'%.8x, gci: %u, transId:(H'%.8x, H'%.8x)\n",
+ sig->apiConnectPtr, sig->gci, sig->transId1, sig->transId2);
+
+ fprintf(output, "noOfOperations: %u, commitFlag: %s, markerFlag: %s\n",
+ noOfOp,
+ (TcIndxConf::getCommitFlag(confInfo) == 0)?"false":"true",
+ (TcIndxConf::getMarkerFlag(confInfo) == 0)?"false":"true");
+ fprintf(output, "Operations:\n");
+ for(i = 0; i < noOfOp; i++) {
+ fprintf(output,
+ "apiOperationPtr: H'%.8x, attrInfoLen: %u\n",
+ sig->operations[i].apiOperationPtr,
+ sig->operations[i].attrInfoLen);
+ }
+ }
+
+ return true;
+}
+
+bool
+printTCINDXREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+// const TcIndxRef * const sig = (TcIndxRef *) theData;
+
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
+
diff --git a/ndb/src/common/debugger/signaldata/TcKeyConf.cpp b/ndb/src/common/debugger/signaldata/TcKeyConf.cpp
new file mode 100644
index 00000000000..727e097a464
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TcKeyConf.cpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TcKeyConf.hpp>
+#include <BlockNumbers.h>
+
+bool
+printTCKEYCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+
+ if (receiverBlockNo == API_PACKED) {
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ }
+ else {
+ const TcKeyConf * const sig = (TcKeyConf *) theData;
+
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ Uint32 confInfo = sig->confInfo;
+ Uint32 noOfOp = TcKeyConf::getNoOfOperations(confInfo);
+ if (noOfOp > 10) noOfOp = 10;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+ fprintf(output, "apiConnectPtr: H'%.8x, gci: %u, transId:(H'%.8x, H'%.8x)\n",
+ sig->apiConnectPtr, sig->gci, sig->transId1, sig->transId2);
+
+ fprintf(output, "noOfOperations: %u, commitFlag: %s, markerFlag: %s\n",
+ noOfOp,
+ (TcKeyConf::getCommitFlag(confInfo) == 0)?"false":"true",
+ (TcKeyConf::getMarkerFlag(confInfo) == 0)?"false":"true");
+ fprintf(output, "Operations:\n");
+ for(i = 0; i < noOfOp; i++) {
+ fprintf(output,
+ "apiOperationPtr: H'%.8x, attrInfoLen: %u\n",
+ sig->operations[i].apiOperationPtr,
+ sig->operations[i].attrInfoLen);
+ }
+ }
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/TcKeyRef.cpp b/ndb/src/common/debugger/signaldata/TcKeyRef.cpp
new file mode 100644
index 00000000000..0dba9909caf
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TcKeyRef.cpp
@@ -0,0 +1,28 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TcKeyRef.hpp>
+
+bool
+printTCKEYREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/TcKeyReq.cpp b/ndb/src/common/debugger/signaldata/TcKeyReq.cpp
new file mode 100644
index 00000000000..7304872ff9c
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TcKeyReq.cpp
@@ -0,0 +1,114 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <signaldata/TcKeyReq.hpp>
+
+bool
+printTCKEYREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+
+ const TcKeyReq * const sig = (TcKeyReq *) theData;
+
+ UintR requestInfo = sig->requestInfo;
+
+ fprintf(output, " apiConnectPtr: H\'%.8x, apiOperationPtr: H\'%.8x\n",
+ sig->apiConnectPtr, sig->apiOperationPtr);
+ fprintf(output, " Operation: %s, Flags: ",
+ sig->getOperationType(requestInfo) == ZREAD ? "Read" :
+ sig->getOperationType(requestInfo) == ZREAD_EX ? "Read-Ex" :
+ sig->getOperationType(requestInfo) == ZUPDATE ? "Update" :
+ sig->getOperationType(requestInfo) == ZINSERT ? "Insert" :
+ sig->getOperationType(requestInfo) == ZDELETE ? "Delete" :
+ sig->getOperationType(requestInfo) == ZWRITE ? "Write" :
+ "Unknown");
+ {
+ if(sig->getDirtyFlag(requestInfo)){
+ fprintf(output, "Dirty ");
+ }
+ if(sig->getStartFlag(requestInfo)){
+ fprintf(output, "Start ");
+ }
+ if(sig->getExecuteFlag(requestInfo)){
+ fprintf(output, "Execute ");
+ }
+ if(sig->getCommitFlag(requestInfo)){
+ fprintf(output, "Commit ");
+ }
+ if (sig->getExecutingTrigger(requestInfo)) {
+ fprintf(output, "Trigger ");
+ }
+
+ UintR TcommitType = sig->getAbortOption(requestInfo);
+ if (TcommitType == TcKeyReq::AbortOnError) {
+ fprintf(output, "AbortOnError ");
+ } else if (TcommitType == TcKeyReq::IgnoreError) {
+ fprintf(output, "IgnoreError ");
+ }//if
+
+ if(sig->getSimpleFlag(requestInfo)){
+ fprintf(output, "Simple ");
+ }
+ if(sig->getScanIndFlag(requestInfo)){
+ fprintf(output, "ScanInd ");
+ }
+ if(sig->getInterpretedFlag(requestInfo)){
+ fprintf(output, "Interpreted ");
+ }
+ if(sig->getDistributionGroupFlag(requestInfo)){
+ fprintf(output, "DGroup = %d ", sig->distrGroupHashValue);
+ }
+ if(sig->getDistributionKeyFlag(sig->requestInfo)){
+ fprintf(output, "DKey = %d ", sig->distributionKeySize);
+ }
+ fprintf(output, "\n");
+ }
+
+ const int keyLen = sig->getKeyLength(requestInfo);
+ const int attrInThis = sig->getAIInTcKeyReq(requestInfo);
+ const int attrLen = sig->getAttrinfoLen(sig->attrLen);
+ const int apiVer = sig->getAPIVersion(sig->attrLen);
+ fprintf(output,
+ " keyLen: %d, attrLen: %d, AI in this: %d, tableId: %d, "
+ "tableSchemaVer: %d, API Ver: %d\n",
+ keyLen, attrLen, attrInThis,
+ sig->tableId, sig->tableSchemaVersion, apiVer);
+
+ fprintf(output, " transId(1, 2): (H\'%.8x, H\'%.8x)\n -- Variable Data --\n",
+ sig->transId1, sig->transId2);
+
+ if (len >= TcKeyReq::StaticLength) {
+ Uint32 restLen = (len - TcKeyReq::StaticLength);
+ const Uint32 * rest = &sig->scanInfo;
+ while(restLen >= 7){
+ fprintf(output,
+ " H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x\n",
+ rest[0], rest[1], rest[2], rest[3],
+ rest[4], rest[5], rest[6]);
+ restLen -= 7;
+ rest += 7;
+ }
+ if(restLen > 0){
+ for(Uint32 i = 0; i<restLen; i++)
+ fprintf(output, " H\'%.8x", rest[i]);
+ fprintf(output, "\n");
+ }
+ } else {
+ fprintf(output, "*** invalid len %u ***\n", len);
+ }
+ return true;
+}
+
diff --git a/ndb/src/common/debugger/signaldata/TcRollbackRep.cpp b/ndb/src/common/debugger/signaldata/TcRollbackRep.cpp
new file mode 100644
index 00000000000..961f0c3619d
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TcRollbackRep.cpp
@@ -0,0 +1,28 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TcRollbackRep.hpp>
+
+bool
+printTCROLLBACKREP(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/TrigAttrInfo.cpp b/ndb/src/common/debugger/signaldata/TrigAttrInfo.cpp
new file mode 100644
index 00000000000..7a8d176ec61
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TrigAttrInfo.cpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TrigAttrInfo.hpp>
+
+static
+const char *
+tatype(Uint32 i){
+ switch(i){
+ case TrigAttrInfo::PRIMARY_KEY:
+ return "PK";
+ break;
+ case TrigAttrInfo::BEFORE_VALUES:
+ return "BEFORE";
+ break;
+ case TrigAttrInfo::AFTER_VALUES:
+ return "AFTER";
+ break;
+ }
+ return "UNKNOWN";
+}
+
+bool
+printTRIG_ATTRINFO(FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const TrigAttrInfo * const sig = (TrigAttrInfo *) theData;
+
+ fprintf(output, " TriggerId: %d Type: %s ConnectPtr: %x\n",
+ sig->getTriggerId(),
+ tatype(sig->getAttrInfoType()),
+ sig->getConnectionPtr());
+
+ Uint32 i = 0;
+ while (i < len - TrigAttrInfo::StaticLength)
+ fprintf(output, " H\'%.8x", sig->getData()[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/TupAccess.cpp b/ndb/src/common/debugger/signaldata/TupAccess.cpp
new file mode 100644
index 00000000000..e94d4636cf5
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TupAccess.cpp
@@ -0,0 +1,131 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TupAccess.hpp>
+#include <SignalLoggerManager.hpp>
+#include <AttributeHeader.hpp>
+
+bool
+printTUP_READ_ATTRS(FILE* output, const Uint32* theData, Uint32 len, Uint16 rbn)
+{
+ const TupReadAttrs* const sig = (const TupReadAttrs*)theData;
+ if (sig->errorCode == RNIL)
+ fprintf(output, " errorCode=RNIL flags=%x\n", sig->requestInfo);
+ else
+ fprintf(output, " errorCode=%u flags=%x\n", sig->errorCode, sig->requestInfo);
+ fprintf(output, " table: id=%u", sig->tableId);
+ fprintf(output, " fragment: id=%u ptr=0x%x\n", sig->fragId, sig->fragPtrI);
+ fprintf(output, " tuple: addr=0x%x version=%u", sig->tupAddr, sig->tupVersion);
+ fprintf(output, " realPage=0x%x offset=%u\n", sig->pageId, sig->pageOffset);
+ const Uint32* buffer = (const Uint32*)sig + TupReadAttrs::SignalLength;
+ Uint32 attrCount = buffer[0];
+ bool readKeys = (sig->requestInfo & TupReadAttrs::ReadKeys);
+ if (sig->errorCode == RNIL && ! readKeys ||
+ sig->errorCode == 0 && readKeys) {
+ fprintf(output, " input: attrCount=%u\n", attrCount);
+ for (unsigned i = 0; i < attrCount; i++) {
+ AttributeHeader ah(buffer[1 + i]);
+ fprintf(output, " %u: attrId=%u\n", i, ah.getAttributeId());
+ }
+ }
+ if (sig->errorCode == 0) {
+ fprintf(output, " output: attrCount=%u\n", attrCount);
+ Uint32 pos = 1 + attrCount;
+ for (unsigned i = 0; i < attrCount; i++) {
+ AttributeHeader ah(buffer[pos++]);
+ fprintf(output, " %u: attrId=%u dataSize=%u\n", i, ah.getAttributeId(), ah.getDataSize());
+ Uint32 next = pos + ah.getDataSize();
+ Uint32 printpos = 0;
+ while (pos < next) {
+ SignalLoggerManager::printDataWord(output, printpos, buffer[pos]);
+ pos++;
+ }
+ if (ah.getDataSize() > 0)
+ fprintf(output, "\n");
+ }
+ }
+ return true;
+}
+
+bool
+printTUP_QUERY_TH(FILE* output, const Uint32* theData, Uint32 len, Uint16 rbn)
+{
+ const TupQueryTh* const sig = (const TupQueryTh*)theData;
+ fprintf(output, "tableId = %u, fragId = %u ", sig->tableId, sig->fragId);
+ fprintf(output, "tuple: addr = 0x%x version = %u\n", sig->tupAddr,
+ sig->tupVersion);
+ fprintf(output, "transId1 = 0x%x, transId2 = 0x%x, savePointId = %u\n",
+ sig->transId1, sig->transId2, sig->savePointId);
+ return true;
+}
+
+bool
+printTUP_STORE_TH(FILE* output, const Uint32* theData, Uint32 len, Uint16 rbn)
+{
+ const TupStoreTh* const sig = (const TupStoreTh*)theData;
+ if (sig->errorCode == RNIL)
+ fprintf(output, " errorCode=RNIL\n");
+ else
+ fprintf(output, " errorCode=%u\n", sig->errorCode);
+ fprintf(output, " table: id=%u", sig->tableId);
+ fprintf(output, " fragment: id=%u ptr=0x%x\n", sig->fragId, sig->fragPtrI);
+ fprintf(output, " tuple: addr=0x%x", sig->tupAddr);
+ if ((sig->tupAddr & 0x1) == 0) {
+ fprintf(output, " fragPage=0x%x index=%u",
+ sig->tupAddr >> MAX_TUPLES_BITS,
+ (sig->tupAddr & ((1 <<MAX_TUPLES_BITS) - 1)) >> 1);
+ fprintf(output, " realPage=0x%x offset=%u\n", sig->pageId, sig->pageOffset);
+ } else {
+ fprintf(output, " cacheId=%u\n",
+ sig->tupAddr >> 1);
+ }
+ if (sig->tupVersion != 0) {
+ fprintf(output, " version=%u ***invalid***\n", sig->tupVersion);
+ }
+ bool showdata = true;
+ switch (sig->opCode) {
+ case TupStoreTh::OpRead:
+ fprintf(output, " operation=Read\n");
+ showdata = false;
+ break;
+ case TupStoreTh::OpInsert:
+ fprintf(output, " operation=Insert\n");
+ break;
+ case TupStoreTh::OpUpdate:
+ fprintf(output, " operation=Update\n");
+ break;
+ case TupStoreTh::OpDelete:
+ fprintf(output, " operation=Delete\n");
+ showdata = false;
+ break;
+ default:
+ fprintf(output, " operation=%u ***invalid***\n", sig->opCode);
+ break;
+ }
+ fprintf(output, " data: offset=%u size=%u", sig->dataOffset, sig->dataSize);
+ if (! showdata) {
+ fprintf(output, " [not printed]\n");
+ } else {
+ fprintf(output, "\n");
+ const Uint32* buffer = (const Uint32*)sig + TupStoreTh::SignalLength;
+ Uint32 pos = 0;
+ while (pos < sig->dataSize)
+ SignalLoggerManager::printDataWord(output, pos, buffer[sig->dataOffset + pos]);
+ if (sig->dataSize > 0)
+ fprintf(output, "\n");
+ }
+ return true;
+};
diff --git a/ndb/src/common/debugger/signaldata/TupCommit.cpp b/ndb/src/common/debugger/signaldata/TupCommit.cpp
new file mode 100644
index 00000000000..d0391b2a8e6
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TupCommit.cpp
@@ -0,0 +1,28 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TupCommit.hpp>
+
+bool
+printTUPCOMMITREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/TupKey.cpp b/ndb/src/common/debugger/signaldata/TupKey.cpp
new file mode 100644
index 00000000000..134b5fde8bc
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TupKey.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TupKey.hpp>
+
+bool
+printTUPKEYREQ(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
+
+bool
+printTUPKEYCONF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
+
+bool
+printTUPKEYREF(FILE * output, const Uint32 * theData, Uint32 len, Uint16 receiverBlockNo){
+ fprintf(output, "Signal data: ");
+ Uint32 i = 0;
+ while (i < len)
+ fprintf(output, "H\'%.8x ", theData[i++]);
+ fprintf(output,"\n");
+
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/TuxMaint.cpp b/ndb/src/common/debugger/signaldata/TuxMaint.cpp
new file mode 100644
index 00000000000..06ac475382c
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/TuxMaint.cpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TuxMaint.hpp>
+#include <SignalLoggerManager.hpp>
+#include <AttributeHeader.hpp>
+
+bool
+printTUX_MAINT_REQ(FILE* output, const Uint32* theData, Uint32 len, Uint16 rbn)
+{
+ //const bool inOut = rbn & (1 << 15);
+ const TuxMaintReq* const sig = (const TuxMaintReq*)theData;
+ fprintf(output, " errorCode=%d\n", sig->errorCode);
+ fprintf(output, " table: id=%d", sig->tableId);
+ fprintf(output, " index: id=%d", sig->indexId);
+ fprintf(output, " fragment: id=%d\n", sig->fragId);
+ fprintf(output, " tuple: addr=0x%x version=%d\n", sig->tupAddr, sig->tupVersion);
+ const Uint32 opCode = sig->opInfo & 0xFF;
+ const Uint32 opFlag = sig->opInfo >> 8;
+ switch (opCode ) {
+ case TuxMaintReq::OpAdd:
+ fprintf(output, " opCode=Add opFlag=%u\n", opFlag);
+ break;
+ case TuxMaintReq::OpRemove:
+ fprintf(output, " opCode=Remove opFlag=%u\n", opFlag);
+ break;
+ default:
+ fprintf(output, " opInfo=%x ***invalid***\n", sig->opInfo);
+ break;
+ }
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/UtilDelete.cpp b/ndb/src/common/debugger/signaldata/UtilDelete.cpp
new file mode 100644
index 00000000000..b6ba53559ac
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/UtilDelete.cpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/UtilDelete.hpp>
+
+bool
+printUTIL_DELETE_REQ(FILE * out, const Uint32 * data, Uint32 l, Uint16 b){
+ (void)l; // Don't want compiler warning
+ (void)b; // Don't want compiler warning
+
+ UtilDeleteReq* sig = (UtilDeleteReq*)data;
+ fprintf(out, " senderData: %d prepareId: %d totalDataLen: %d\n",
+ sig->senderData,
+ sig->prepareId,
+ sig->totalDataLen);
+ fprintf(out,
+ " H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x\n"
+ " H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x\n"
+ " H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x\n",
+ sig->attrData[0], sig->attrData[1], sig->attrData[2],
+ sig->attrData[3], sig->attrData[4], sig->attrData[5],
+ sig->attrData[6], sig->attrData[7], sig->attrData[8],
+ sig->attrData[9], sig->attrData[10], sig->attrData[11],
+ sig->attrData[12], sig->attrData[13], sig->attrData[14],
+ sig->attrData[15], sig->attrData[16], sig->attrData[17],
+ sig->attrData[18], sig->attrData[19], sig->attrData[20],
+ sig->attrData[21]
+ );
+
+ return true;
+}
+
+bool
+printUTIL_DELETE_CONF(FILE * out, const Uint32 * data, Uint32 l, Uint16 b){
+ (void)l; // Don't want compiler warning
+ (void)b; // Don't want compiler warning
+
+ UtilDeleteConf* sig = (UtilDeleteConf*)data;
+ fprintf(out, " senderData: %d\n", sig->senderData);
+ return true;
+}
+
+bool
+printUTIL_DELETE_REF(FILE * out, const Uint32 * data, Uint32 l, Uint16 b){
+ (void)l; // Don't want compiler warning
+ (void)b; // Don't want compiler warning
+
+ UtilDeleteRef* sig = (UtilDeleteRef*)data;
+ fprintf(out, " senderData: %d\n", sig->senderData);
+ fprintf(out, " errorCode: %d\n", sig->errorCode);
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/UtilExecute.cpp b/ndb/src/common/debugger/signaldata/UtilExecute.cpp
new file mode 100644
index 00000000000..2c88fa174d4
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/UtilExecute.cpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/UtilExecute.hpp>
+
+bool
+printUTIL_EXECUTE_REQ(FILE* out, const Uint32 * data, Uint32 len, Uint16 rec)
+{
+ const UtilExecuteReq* const sig = (UtilExecuteReq*)data;
+ fprintf(out, " senderRef: H'%.8x, senderData: H'%.8x prepareId: %d\n",
+ sig->senderRef,
+ sig->senderData,
+ sig->prepareId);
+ return true;
+}
+
+bool
+printUTIL_EXECUTE_CONF(FILE* out, const Uint32 * data, Uint32 len, Uint16 rec)
+{
+ UtilExecuteConf* sig = (UtilExecuteConf*)data;
+ fprintf(out, " senderData: H'%.8x\n",
+ sig->senderData);
+ return true;
+}
+
+bool
+printUTIL_EXECUTE_REF(FILE* out, const Uint32 * data, Uint32 len, Uint16 rec)
+{
+ UtilExecuteRef* sig = (UtilExecuteRef*)data;
+ fprintf(out, " senderData: H'%.8x, ", sig->senderData);
+ fprintf(out, " errorCode: %s, ",
+ sig->errorCode == UtilExecuteRef::IllegalKeyNumber ?
+ "IllegalKeyNumber" :
+ sig->errorCode == UtilExecuteRef::IllegalAttrNumber ?
+ "IllegalAttrNumber" :
+ sig->errorCode == UtilExecuteRef::TCError ?
+ "TCError" :
+ sig->errorCode == UtilExecuteRef::IllegalPrepareId ?
+ "IllegalPrepareId" :
+ sig->errorCode == UtilExecuteRef::AllocationError ?
+ "AllocationError" :
+ "Unknown");
+ fprintf(out, " TCErrorCode: %d\n",
+ sig->TCErrorCode);
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/UtilLock.cpp b/ndb/src/common/debugger/signaldata/UtilLock.cpp
new file mode 100644
index 00000000000..34e37c3e2d8
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/UtilLock.cpp
@@ -0,0 +1,158 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/UtilLock.hpp>
+
+bool
+printUTIL_LOCK_REQ (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilLockReq *const sig = (UtilLockReq *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ fprintf (output, " requestInfo: %x\n", sig->requestInfo);
+ return true;
+}
+
+bool
+printUTIL_LOCK_CONF (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilLockConf *const sig = (UtilLockConf *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ fprintf (output, " lockKey: %x\n", sig->lockKey);
+ return true;
+}
+
+bool
+printUTIL_LOCK_REF (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilLockRef *const sig = (UtilLockRef *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ fprintf (output, " errorCode: %x\n", sig->errorCode);
+ return true;
+}
+
+bool
+printUTIL_UNLOCK_REQ (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilUnlockReq *const sig = (UtilUnlockReq *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ fprintf (output, " lockKey: %x\n", sig->lockKey);
+ return true;
+}
+
+bool
+printUTIL_UNLOCK_CONF (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilUnlockConf *const sig = (UtilUnlockConf *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ return true;
+}
+
+bool
+printUTIL_UNLOCK_REF (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilUnlockRef *const sig = (UtilUnlockRef *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ fprintf (output, " errorCode: %x\n", sig->errorCode);
+ return true;
+}
+
+bool
+printUTIL_CREATE_LOCK_REQ (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilCreateLockReq *const sig = (UtilCreateLockReq *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ fprintf (output, " lockType: %x\n", sig->lockType);
+ return true;
+}
+
+bool
+printUTIL_CREATE_LOCK_REF (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilCreateLockRef *const sig = (UtilCreateLockRef *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ fprintf (output, " errorCode: %x\n", sig->errorCode);
+ return true;
+}
+
+bool
+printUTIL_CREATE_LOCK_CONF (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilCreateLockConf *const sig = (UtilCreateLockConf *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ return true;
+}
+
+bool
+printUTIL_DESTROY_LOCK_REQ (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilDestroyLockReq *const sig = (UtilDestroyLockReq *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ fprintf (output, " lockKey: %x\n", sig->lockKey);
+ return true;
+}
+
+bool
+printUTIL_DESTROY_LOCK_REF (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilDestroyLockRef *const sig = (UtilDestroyLockRef *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ fprintf (output, " errorCode: %x\n", sig->errorCode);
+ return true;
+}
+
+bool
+printUTIL_DESTROY_LOCK_CONF (FILE * output, const Uint32 * theData,
+ Uint32 len, Uint16 receiverBlockNo)
+{
+ const UtilDestroyLockConf *const sig = (UtilDestroyLockConf *) theData;
+ fprintf (output, " senderData: %x\n", sig->senderData);
+ fprintf (output, " senderRef: %x\n", sig->senderRef);
+ fprintf (output, " lockId: %x\n", sig->lockId);
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/UtilPrepare.cpp b/ndb/src/common/debugger/signaldata/UtilPrepare.cpp
new file mode 100644
index 00000000000..adc2e299380
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/UtilPrepare.cpp
@@ -0,0 +1,64 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/UtilPrepare.hpp>
+
+bool
+printUTIL_PREPARE_REQ(FILE* out, const Uint32 * data, Uint32 len, Uint16 rec)
+{
+ UtilPrepareReq* sig = (UtilPrepareReq*)data;
+ fprintf(out, " senderRef: H'%.8x senderData: H'%.8x\n",
+ sig->senderRef,
+ sig->senderData);
+
+ return true;
+}
+
+bool
+printUTIL_PREPARE_CONF(FILE* out, const Uint32 * data, Uint32 len, Uint16 rec)
+{
+ UtilPrepareConf* sig = (UtilPrepareConf*)data;
+ fprintf(out, " senderData: H'%.8x prepareId: %d\n",
+ sig->senderData,
+ sig->prepareId);
+ return true;
+}
+
+bool
+printUTIL_PREPARE_REF(FILE* out, const Uint32 * data, Uint32 len, Uint16 rec)
+{
+ UtilPrepareRef* sig = (UtilPrepareRef*)data;
+ fprintf(out, " senderData: H'%.8x, ", sig->senderData);
+ fprintf(out, " error: %d, ", sig->errorCode);
+
+ fprintf(out, " errorMsg: ");
+ switch(sig->errorCode) {
+ case UtilPrepareRef::NO_ERROR:
+ fprintf(out, "No error");
+ break;
+ case UtilPrepareRef::PREPARE_SEIZE_ERROR:
+ fprintf(out, "Failed to seize Prepare record");
+ break;
+ case UtilPrepareRef::PREPARED_OPERATION_SEIZE_ERROR:
+ fprintf(out, "Failed to seize PreparedOperation record");
+ break;
+ case UtilPrepareRef::DICT_TAB_INFO_ERROR:
+ fprintf(out, "Failed to get table info from DICT");
+ break;
+ }
+ fprintf(out, "\n");
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/UtilSequence.cpp b/ndb/src/common/debugger/signaldata/UtilSequence.cpp
new file mode 100644
index 00000000000..e91999d9abf
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/UtilSequence.cpp
@@ -0,0 +1,67 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/UtilSequence.hpp>
+
+inline
+const char *
+type2string(UtilSequenceReq::RequestType type){
+ switch(type){
+ case UtilSequenceReq::NextVal:
+ return "NextVal";
+ case UtilSequenceReq::CurrVal:
+ return "CurrVal";
+ case UtilSequenceReq::Create:
+ return "Create";
+ default:
+ return "Unknown";
+ }
+}
+
+bool
+printUTIL_SEQUENCE_REQ(FILE * out, const Uint32 * data, Uint32 l, Uint16 b){
+ UtilSequenceReq* sig = (UtilSequenceReq*)data;
+ fprintf(out, " senderData: %d sequenceId: %d RequestType: %s\n",
+ sig->senderData,
+ sig->sequenceId,
+ type2string((UtilSequenceReq::RequestType)sig->requestType));
+ return true;
+}
+
+bool
+printUTIL_SEQUENCE_CONF(FILE * out, const Uint32 * data, Uint32 l, Uint16 b){
+ UtilSequenceConf* sig = (UtilSequenceConf*)data;
+ fprintf(out, " senderData: %d sequenceId: %d RequestType: %s\n",
+ sig->senderData,
+ sig->sequenceId,
+ type2string((UtilSequenceReq::RequestType)sig->requestType));
+ fprintf(out, " val: [ %d %d ]\n",
+ sig->sequenceValue[0],
+ sig->sequenceValue[1]);
+ return true;
+}
+
+bool
+printUTIL_SEQUENCE_REF(FILE * out, const Uint32 * data, Uint32 l, Uint16 b){
+ UtilSequenceRef* sig = (UtilSequenceRef*)data;
+ fprintf(out, " senderData: %d sequenceId: %d RequestType: %s\n",
+ sig->senderData,
+ sig->sequenceId,
+ type2string((UtilSequenceReq::RequestType)sig->requestType));
+ fprintf(out, " errorCode: %d, TCErrorCode: %d\n",
+ sig->errorCode, sig->TCErrorCode);
+ return true;
+}
diff --git a/ndb/src/common/debugger/signaldata/print.awk b/ndb/src/common/debugger/signaldata/print.awk
new file mode 100644
index 00000000000..9730fb4a236
--- /dev/null
+++ b/ndb/src/common/debugger/signaldata/print.awk
@@ -0,0 +1,55 @@
+BEGIN {
+ m_curr="";
+ m_count=0;
+ m_level=0;
+}
+/^[ ]*class[ ]+.*{/ {
+ if(m_curr != ""){
+ print;
+ print "ERROR: " m_curr;
+ exit;
+ }
+ m_curr = $2;
+}
+/{/ {
+ m_level++;
+}
+/bool print/{
+ m_print=$3;
+ i=index($3, "(");
+ if(i > 0){
+ m_print=substr($3, 0, i-1);
+ }
+}
+
+/[ ]+Uint32[ ]+[^)]*;/ {
+ if(m_level >= 0){
+ m=$2;
+ i=index($2, ";");
+ if(i > 0){
+ m=substr($2, 0, i-1);
+ }
+ m_members[m_count]=m;
+ m_count++;
+ }
+}
+/^[ ]*}[ ]*;/ {
+ m_level--;
+ if(m_level == 0){
+ if(m_count > 0 && m_print != ""){
+ print "bool";
+ print m_print "(FILE * output, const Uint32 * theData, ";
+ print "Uint32 len, Uint16 receiverBlockNo) {";
+ print "const " m_curr " * const sig = (" m_curr " *)theData;";
+ for(i = 0; i<m_count; i++){
+ print "fprintf(output, \" " m_members[i] ": %x\\n\", sig->" m_members[i] ");";
+ }
+ print "return true;";
+ print "}";
+ print "";
+ }
+ m_curr="";
+ m_print="";
+ m_count=0;
+ }
+}
diff --git a/ndb/src/common/editline/MANIFEST b/ndb/src/common/editline/MANIFEST
new file mode 100644
index 00000000000..dc8b4b36f88
--- /dev/null
+++ b/ndb/src/common/editline/MANIFEST
@@ -0,0 +1,15 @@
+File Name Description
+--------------------------------------------------------
+README Release notes and copyright
+MANIFEST This shipping list
+Make.os9 OS-9 makefile
+Makefile Unix makefile
+complete.c Filename completion routines
+editline.3 Manual page for editline library
+editline.c Line-editing routines
+editline_internal.h Internal library header file
+os9.h OS-9-specific declarations
+sysos9.c OS-9-specific routines
+sysunix.c Unix-specific routines
+testit.c Test driver
+unix.h Unix-specific declarations
diff --git a/ndb/src/common/editline/Makefile b/ndb/src/common/editline/Makefile
new file mode 100644
index 00000000000..800df8f0f31
--- /dev/null
+++ b/ndb/src/common/editline/Makefile
@@ -0,0 +1,18 @@
+include .defs.mk
+
+TYPE :=
+
+ARCHIVE_TARGET := editline
+
+CFLAGS += -DANSI_ARROWS -DHAVE_TCGETATTR -DSYS_UNIX
+
+ifeq ($(NDB_OS), WIN32)
+SOURCES = editline_win32.c
+else
+SOURCES = complete.c editline.c sysunix.c
+endif
+
+DIRS := test
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/common/editline/README b/ndb/src/common/editline/README
new file mode 100644
index 00000000000..537c7bd8611
--- /dev/null
+++ b/ndb/src/common/editline/README
@@ -0,0 +1,53 @@
+--
+NOTE: This version has been modified by Ericsson/Alzato. Please
+see the cvs changelog for more details.
+--
+
+$Revision: 1.2 $
+
+This is a line-editing library. It can be linked into almost any
+program to provide command-line editing and recall.
+
+It is call-compatible with the FSF readline library, but it is a
+fraction of the size (and offers fewer features). It does not use
+standard I/O. It is distributed under a "C News-like" copyright.
+
+Configuration is done in the Makefile. Type "make testit" to get
+a small slow shell for testing.
+
+This contains some changes since the posting to comp.sources.misc:
+ - Bugfix for completion on absolute pathnames.
+ - Better handling of M-n versus showing raw 8bit chars.
+ - Better signal handling.
+ - Now supports termios/termio/sgttyb ioctl's.
+ - Add M-m command to toggle how 8bit data is displayed.
+The following changes, made since the last public release, come from
+J.G. Vons <vons@cesar.crbca1.sinet.slb.com>:
+ - History-searching no longer redraws the line wrong
+ - Added ESC-ESC as synonym for ESC-?
+ - SIGQUIT (normally ^\) now sends a signal, not indicating EOF.
+ - Fixed some typo's and unclear wording in the manpage.
+ - Fixed completion when all entries shared a common prefix.
+ - Fixed some meta-char line-redrawing bugs.
+
+Enjoy,
+ Rich $alz
+ <rsalz@osf.org>
+
+ Copyright 1992,1993 Simmule Turner and Rich Salz. All rights reserved.
+
+ This software is not subject to any license of the American Telephone
+ and Telegraph Company or of the Regents of the University of California.
+
+ Permission is granted to anyone to use this software for any purpose on
+ any computer system, and to alter it and redistribute it freely, subject
+ to the following restrictions:
+ 1. The authors are not responsible for the consequences of use of this
+ software, no matter how awful, even if they arise from flaws in it.
+ 2. The origin of this software must not be misrepresented, either by
+ explicit claim or by omission. Since few users ever read sources,
+ credits must appear in the documentation.
+ 3. Altered versions must be plainly marked as such, and must not be
+ misrepresented as being the original software. Since few users
+ ever read sources, credits must appear in the documentation.
+ 4. This notice may not be removed or altered.
diff --git a/ndb/src/common/editline/complete.c b/ndb/src/common/editline/complete.c
new file mode 100644
index 00000000000..d1f8b1d3ff4
--- /dev/null
+++ b/ndb/src/common/editline/complete.c
@@ -0,0 +1,211 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* -*- c-basic-offset: 4; -*-
+** $Revision: 1.8 $
+**
+** History and file completion functions for editline library.
+*/
+#include "editline_internal.h"
+
+
+/*
+** strcmp-like sorting predicate for qsort.
+*/
+static int
+compare(const void *p1, const void *p2)
+{
+ const char **v1;
+ const char **v2;
+
+ v1 = (const char **)p1;
+ v2 = (const char **)p2;
+ return strcmp(*v1, *v2);
+}
+
+/*
+** Fill in *avp with an array of names that match file, up to its length.
+** Ignore . and .. .
+*/
+static int
+FindMatches(char *dir, char *file, char ***avp)
+{
+ char **av;
+ char **new;
+ char *p;
+ DIR *dp;
+ struct dirent *ep;
+ size_t ac;
+ size_t len;
+
+ if ((dp = opendir(dir)) == NULL)
+ return 0;
+
+ av = NULL;
+ ac = 0;
+ len = strlen(file);
+ while ((ep = readdir(dp)) != NULL) {
+ p = ep->d_name;
+ if (p[0] == '.' && (p[1] == '\0' || (p[1] == '.' && p[2] == '\0')))
+ continue;
+ if (len && strncmp(p, file, len) != 0)
+ continue;
+
+ if ((ac % MEM_INC) == 0) {
+ if ((new = malloc(sizeof(char*) * (ac + MEM_INC))) == NULL)
+ break;
+ if (ac) {
+ memcpy(new, av, ac * sizeof (char **));
+ free(av);
+ }
+ *avp = av = new;
+ }
+
+ if ((av[ac] = strdup(p)) == NULL) {
+ if (ac == 0)
+ free(av);
+ break;
+ }
+ ac++;
+ }
+
+ /* Clean up and return. */
+ (void)closedir(dp);
+ if (ac)
+ qsort(av, ac, sizeof (char **), compare);
+ return ac;
+}
+
+/*
+** Split a pathname into allocated directory and trailing filename parts.
+*/
+static int
+SplitPath(char *path, char **dirpart, char ** filepart)
+{
+ static char DOT[] = ".";
+ char *dpart;
+ char *fpart;
+
+ if ((fpart = strrchr(path, '/')) == NULL) {
+ if ((dpart = strdup(DOT)) == NULL)
+ return -1;
+ if ((fpart = strdup(path)) == NULL) {
+ free(dpart);
+ return -1;
+ }
+ }
+ else {
+ if ((dpart = strdup(path)) == NULL)
+ return -1;
+ dpart[fpart - path + 1] = '\0';
+ if ((fpart = strdup(++fpart)) == NULL) {
+ free(dpart);
+ return -1;
+ }
+ }
+ *dirpart = dpart;
+ *filepart = fpart;
+ return 0;
+}
+
+/*
+** Attempt to complete the pathname, returning an allocated copy.
+** Fill in *unique if we completed it, or set it to 0 if ambiguous.
+*/
+char *
+rl_complete(char *pathname,int *unique)
+{
+ char **av;
+ char *dir;
+ char *file;
+ char *new;
+ char *p;
+ size_t ac;
+ size_t end;
+ size_t i;
+ size_t j;
+ size_t len;
+ size_t new_len;
+ size_t p_len;
+
+ if (SplitPath(pathname, &dir, &file) < 0)
+ return NULL;
+ if ((ac = FindMatches(dir, file, &av)) == 0) {
+ free(dir);
+ free(file);
+ return NULL;
+ }
+
+ p = NULL;
+ len = strlen(file);
+ if (ac == 1) {
+ /* Exactly one match -- finish it off. */
+ *unique = 1;
+ j = strlen(av[0]) - len + 2;
+ p_len = sizeof(char) * (j + 1);
+ if ((p = malloc(p_len)) != NULL) {
+ memcpy(p, av[0] + len, j);
+ new_len = sizeof(char) * (strlen(dir) + strlen(av[0]) + 2);
+ new = malloc(new_len);
+ if(new != NULL) {
+ snprintf(new, new_len, "%s/%s", dir, av[0]);
+ rl_add_slash(new, p, p_len);
+ free(new);
+ }
+ }
+ }
+ else {
+ /* Find largest matching substring. */
+ for (*unique = 0, i = len, end = strlen(av[0]); i < end; i++)
+ for (j = 1; j < ac; j++)
+ if (av[0][i] != av[j][i])
+ goto breakout;
+breakout:
+ if (i > len) {
+ j = i - len + 1;
+ if ((p = malloc(sizeof(char) * j)) != NULL) {
+ memcpy(p, av[0] + len, j);
+ p[j - 1] = '\0';
+ }
+ }
+ }
+
+ /* Clean up and return. */
+ free(dir);
+ free(file);
+ for (i = 0; i < ac; i++)
+ free(av[i]);
+ free(av);
+ return p;
+}
+
+/*
+** Return all possible completions.
+*/
+int
+rl_list_possib(char *pathname, char ***avp)
+{
+ char *dir;
+ char *file;
+ int ac;
+
+ if (SplitPath(pathname, &dir, &file) < 0)
+ return 0;
+ ac = FindMatches(dir, file, avp);
+ free(dir);
+ free(file);
+ return ac;
+}
diff --git a/ndb/src/common/editline/editline.3 b/ndb/src/common/editline/editline.3
new file mode 100644
index 00000000000..159cc7f87bb
--- /dev/null
+++ b/ndb/src/common/editline/editline.3
@@ -0,0 +1,178 @@
+.\" $Revision: 1.1 $
+.TH EDITLINE 3
+.SH NAME
+editline \- command-line editing library with history
+.SH SYNOPSIS
+.nf
+.B "char *"
+.B "readline(prompt)"
+.B " char *prompt;"
+
+.B "void"
+.B "add_history(line)"
+.B " char *line;"
+.fi
+.SH DESCRIPTION
+.I Editline
+is a library that provides an line-editing interface with text recall.
+It is intended to be compatible with the
+.I readline
+library provided by the Free Software Foundation, but much smaller.
+The bulk of this manual page describes the user interface.
+.PP
+The
+.I readline
+routine returns a line of text with the trailing newline removed.
+The data is returned in a buffer allocated with
+.IR malloc (3),
+so the space should be released with
+.IR free (3)
+when the calling program is done with it.
+Before accepting input from the user, the specified
+.I prompt
+is displayed on the terminal.
+.PP
+The
+.I add_history
+routine makes a copy of the specified
+.I line
+and adds it to the internal history list.
+.SS "User Interface"
+A program that uses this library provides a simple emacs-like editing
+interface to its users.
+A line may be edited before it is sent to the calling program by typing either
+control characters or escape sequences.
+A control character, shown as a caret followed by a letter, is typed by
+holding down the ``control'' key while the letter is typed.
+For example, ``^A'' is a control-A.
+An escape sequence is entered by typing the ``escape'' key followed by one or
+more characters.
+The escape key is abbreviated as ``ESC''.
+Note that unlike control keys, case matters in escape sequences; ``ESC\ F''
+is not the same as ``ESC\ f''.
+.PP
+An editing command may be typed anywhere on the line, not just at the
+beginning.
+In addition, a return may also be typed anywhere on the line, not just at
+the end.
+.PP
+Most editing commands may be given a repeat count,
+.IR n ,
+where
+.I n
+is a number.
+To enter a repeat count, type the escape key, the number, and then
+the command to execute.
+For example, ``ESC\ 4\ ^f'' moves forward four characters.
+If a command may be given a repeat count then the text ``[n]'' is given at the
+end of its description.
+.PP
+The following control characters are accepted:
+.RS
+.nf
+.ta \w'ESC DEL 'u
+^A Move to the beginning of the line
+^B Move left (backwards) [n]
+^D Delete character [n]
+^E Move to end of line
+^F Move right (forwards) [n]
+^G Ring the bell
+^H Delete character before cursor (backspace key) [n]
+^I Complete filename (tab key); see below
+^J Done with line (return key)
+^K Kill to end of line (or column [n])
+^L Redisplay line
+^M Done with line (alternate return key)
+^N Get next line from history [n]
+^P Get previous line from history [n]
+^R Search backward (forward if [n]) through history for text;
+\& prefixing the string with a caret (^) forces it to
+\& match only at the beginning of a history line
+^T Transpose characters
+^V Insert next character, even if it is an edit command
+^W Wipe to the mark
+^X^X Exchange current location and mark
+^Y Yank back last killed text
+^[ Start an escape sequence (escape key)
+^]c Move forward to next character ``c''
+^? Delete character before cursor (delete key) [n]
+.fi
+.RE
+.PP
+The following escape sequences are provided.
+.RS
+.nf
+.ta \w'ESC DEL 'u
+ESC\ ^H Delete previous word (backspace key) [n]
+ESC\ DEL Delete previous word (delete key) [n]
+ESC\ ESC Show possible completions; see below
+ESC\ SP Set the mark (space key); see ^X^X and ^Y above
+ESC\ . Get the last (or [n]'th) word from previous line
+ESC\ ? Show possible completions; see below
+ESC\ < Move to start of history
+ESC\ > Move to end of history
+ESC\ b Move backward a word [n]
+ESC\ d Delete word under cursor [n]
+ESC\ f Move forward a word [n]
+ESC\ l Make word lowercase [n]
+ESC\ m Toggle if 8bit chars display as themselves or with
+\& an ``M\-'' prefix
+ESC\ u Make word uppercase [n]
+ESC\ y Yank back last killed text
+ESC\ w Make area up to mark yankable
+ESC\ nn Set repeat count to the number nn
+ESC\ C Read from environment variable ``_C_'', where C is
+\& an uppercase letter
+.fi
+.RE
+.PP
+The
+.I editline
+library has a small macro facility.
+If you type the escape key followed by an uppercase letter,
+.IR C ,
+then the contents of the environment variable
+.I _C_
+are read in as if you had typed them at the keyboard.
+For example, if the variable
+.I _L_
+contains the following:
+.RS
+^A^Kecho '^V^[[H^V^[[2J'^M
+.RE
+Then typing ``ESC L'' will move to the beginning of the line, kill the
+entire line, enter the echo command needed to clear the terminal (if your
+terminal is like a VT-100), and send the line back to the shell.
+.PP
+The
+.I editline
+library also does filename completion.
+Suppose the root directory has the following files in it:
+.RS
+.nf
+.ta \w'core 'u
+bin vmunix
+core vmunix.old
+.fi
+.RE
+If you type ``rm\ /v'' and then the tab key.
+.I Editline
+will then finish off as much of the name as possible by adding ``munix''.
+Because the name is not unique, it will then beep.
+If you type the escape key followed by either a question mark or another
+escape, it will display the two choices.
+If you then type a period and a tab, the library will finish off the filename
+for you:
+.RS
+.nf
+.RI "rm /v[TAB]" munix ".[TAB]" old
+.fi
+.RE
+The tab key is shown by ``[TAB]'' and the automatically-entered text
+is shown in italics.
+.SH "BUGS AND LIMITATIONS"
+Cannot handle lines more than 80 columns.
+.SH AUTHORS
+Simmule R. Turner <uunet.uu.net!capitol!sysgo!simmy>
+and Rich $alz <rsalz@osf.org>.
+Original manual page by DaviD W. Sanderson <dws@ssec.wisc.edu>.
diff --git a/ndb/src/common/editline/editline.c b/ndb/src/common/editline/editline.c
new file mode 100644
index 00000000000..0529d18b952
--- /dev/null
+++ b/ndb/src/common/editline/editline.c
@@ -0,0 +1,1514 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* -*- c-basic-offset: 4; -*-
+** $Revision: 1.6 $
+**
+** Main editing routines for editline library.
+*/
+#include "editline_internal.h"
+#include <signal.h>
+#include <ctype.h>
+#include <unistd.h>
+
+/*
+** Manifest constants.
+*/
+#define SCREEN_WIDTH 80
+#define SCREEN_ROWS 24
+#define NO_ARG (-1)
+#define DEL 127
+#define TAB '\t'
+#define CTL(x) ((x) & 0x1F)
+#define ISCTL(x) ((x) && (x) < ' ')
+#define UNCTL(x) ((x) + 64)
+#define META(x) ((x) | 0x80)
+#define ISMETA(x) ((x) & 0x80)
+#define UNMETA(x) ((x) & 0x7F)
+#define MAPSIZE 32
+#define METAMAPSIZE 16
+#if !defined(HIST_SIZE)
+#define HIST_SIZE 20
+#endif /* !defined(HIST_SIZE) */
+
+/*
+** Command status codes.
+*/
+typedef enum _STATUS {
+ CSdone, CSeof, CSmove, CSdispatch, CSstay, CSsignal
+} STATUS;
+
+/*
+** The type of case-changing to perform.
+*/
+typedef enum _CASE {
+ TOupper, TOlower
+} CASE;
+
+/*
+** Key to command mapping.
+*/
+typedef struct _KEYMAP {
+ char Key;
+ char Active;
+ STATUS (*Function)();
+} KEYMAP;
+
+/*
+** Command history structure.
+*/
+typedef struct _HISTORY {
+ int Size;
+ int Pos;
+ char *Lines[HIST_SIZE];
+} HISTORY;
+
+/*
+** Globals.
+*/
+int rl_eof;
+int rl_erase;
+int rl_intr;
+int rl_kill;
+int rl_quit;
+#if defined(DO_SIGTSTP)
+int rl_susp;
+#endif /* defined(DO_SIGTSTP) */
+
+static char NIL[] = "";
+static const char *Input = NIL;
+static char *Line;
+static const char *Prompt;
+static char *Yanked;
+static char *Screen;
+static char NEWLINE[]= CRLF;
+static HISTORY H;
+static int Repeat;
+static int End;
+static int Mark;
+static int OldPoint;
+static int Point;
+static int PushBack;
+static int Pushed;
+static int Signal;
+static KEYMAP Map[MAPSIZE];
+static KEYMAP MetaMap[METAMAPSIZE];
+static size_t Length;
+static size_t ScreenCount;
+static size_t ScreenSize;
+static char *backspace;
+static int TTYwidth;
+static int TTYrows;
+
+/* Display print 8-bit chars as `M-x' or as the actual 8-bit char? */
+int rl_meta_chars = 1;
+
+/*
+** Declarations.
+*/
+static char *editinput();
+
+#if defined(USE_TERMCAP)
+extern char *getenv();
+extern char *tgetstr();
+extern int tgetent();
+extern int tgetnum();
+#endif /* defined(USE_TERMCAP) */
+
+/*
+** TTY input/output functions.
+*/
+
+static void
+TTYflush()
+{
+ if (ScreenCount) {
+ (void)write(1, Screen, ScreenCount);
+ ScreenCount = 0;
+ }
+}
+
+static void
+TTYput(const char c)
+{
+ Screen[ScreenCount] = c;
+ if (++ScreenCount >= ScreenSize - 1) {
+ ScreenSize += SCREEN_INC;
+ Screen = realloc(Screen, sizeof(char) * ScreenSize);
+ /* XXX what to do if realloc failes? */
+ }
+}
+
+static void
+TTYputs(const char *p)
+{
+ while (*p)
+ TTYput(*p++);
+}
+
+static void
+TTYshow(char c)
+{
+ if (c == DEL) {
+ TTYput('^');
+ TTYput('?');
+ }
+ else if (c == TAB) {
+ /* XXX */
+ }
+ else if (ISCTL(c)) {
+ TTYput('^');
+ TTYput(UNCTL(c));
+ }
+ else if (rl_meta_chars && ISMETA(c)) {
+ TTYput('M');
+ TTYput('-');
+ TTYput(UNMETA(c));
+ }
+ else
+ TTYput(c);
+}
+
+static void
+TTYstring(char *p)
+{
+ while (*p)
+ TTYshow(*p++);
+}
+
+static int
+TTYget()
+{
+ char c;
+
+ TTYflush();
+ if (Pushed) {
+ Pushed = 0;
+ return PushBack;
+ }
+ if (*Input)
+ return *Input++;
+ return read(0, &c, (size_t)1) == 1 ? c : EOF;
+}
+
+#define TTYback() (backspace ? TTYputs((const char *)backspace) : TTYput('\b'))
+
+static void
+TTYbackn(int n)
+{
+ while (--n >= 0)
+ TTYback();
+}
+
+static void
+TTYinfo()
+{
+ static int init;
+#if defined(USE_TERMCAP)
+ char *term;
+ char buff[2048];
+ char *bp;
+ char *p;
+#endif /* defined(USE_TERMCAP) */
+#if defined(TIOCGWINSZ)
+ struct winsize W;
+#endif /* defined(TIOCGWINSZ) */
+
+ if (init) {
+#if defined(TIOCGWINSZ)
+ /* Perhaps we got resized. */
+ if (ioctl(0, TIOCGWINSZ, &W) >= 0
+ && W.ws_col > 0 && W.ws_row > 0) {
+ TTYwidth = (int)W.ws_col;
+ TTYrows = (int)W.ws_row;
+ }
+#endif /* defined(TIOCGWINSZ) */
+ return;
+ }
+ init++;
+
+ TTYwidth = TTYrows = 0;
+#if defined(USE_TERMCAP)
+ bp = &buff[0];
+ if ((term = getenv("TERM")) == NULL)
+ term = "dumb";
+ if (tgetent(buff, term) < 0) {
+ TTYwidth = SCREEN_WIDTH;
+ TTYrows = SCREEN_ROWS;
+ return;
+ }
+ p = tgetstr("le", &bp);
+ backspace = p ? strdup(p) : NULL;
+ TTYwidth = tgetnum("co");
+ TTYrows = tgetnum("li");
+#endif /* defined(USE_TERMCAP) */
+
+#if defined(TIOCGWINSZ)
+ if (ioctl(0, TIOCGWINSZ, &W) >= 0) {
+ TTYwidth = (int)W.ws_col;
+ TTYrows = (int)W.ws_row;
+ }
+#endif /* defined(TIOCGWINSZ) */
+
+ if (TTYwidth <= 0 || TTYrows <= 0) {
+ TTYwidth = SCREEN_WIDTH;
+ TTYrows = SCREEN_ROWS;
+ }
+}
+
+
+/*
+** Print an array of words in columns.
+*/
+static void
+columns(int ac, char **av)
+{
+ char *p;
+ int i;
+ int j;
+ int k;
+ int len;
+ int skip;
+ int longest;
+ int cols;
+
+ /* Find longest name, determine column count from that. */
+ for (longest = 0, i = 0; i < ac; i++)
+ if ((j = strlen((char *)av[i])) > longest)
+ longest = j;
+ cols = TTYwidth / (longest + 3);
+
+ TTYputs((const char *)NEWLINE);
+ for (skip = ac / cols + 1, i = 0; i < skip; i++) {
+ for (j = i; j < ac; j += skip) {
+ for (p = av[j], len = strlen((char *)p), k = len; --k >= 0; p++)
+ TTYput(*p);
+ if (j + skip < ac)
+ while (++len < longest + 3)
+ TTYput(' ');
+ }
+ TTYputs((const char *)NEWLINE);
+ }
+}
+
+static void
+reposition()
+{
+ int i;
+ char *p;
+
+ TTYput('\r');
+ TTYputs((const char *)Prompt);
+ for (i = Point, p = Line; --i >= 0; p++)
+ TTYshow(*p);
+}
+
+static void
+left(STATUS Change)
+{
+ char c;
+
+ TTYback();
+ if (Point) {
+ c = Line[Point - 1];
+ if (c == TAB) {
+ /* XXX */
+ }
+ else if (ISCTL(c))
+ TTYback();
+ else if (rl_meta_chars && ISMETA(c)) {
+ TTYback();
+ TTYback();
+ }
+ }
+ if (Change == CSmove)
+ Point--;
+}
+
+static void
+right(STATUS Change)
+{
+ TTYshow(Line[Point]);
+ if (Change == CSmove)
+ Point++;
+}
+
+static STATUS
+ring_bell()
+{
+ TTYput('\07');
+ TTYflush();
+ return CSstay;
+}
+
+static STATUS
+do_macro(int c)
+{
+ char name[4];
+
+ name[0] = '_';
+ name[1] = c;
+ name[2] = '_';
+ name[3] = '\0';
+
+ if ((Input = (char *)getenv((char *)name)) == NULL) {
+ Input = NIL;
+ return ring_bell();
+ }
+ return CSstay;
+}
+
+static STATUS
+do_forward(STATUS move)
+{
+ int i;
+ char *p;
+
+ i = 0;
+ do {
+ p = &Line[Point];
+ for ( ; Point < End && (*p == ' ' || !isalnum((int)*p)); Point++, p++)
+ if (move == CSmove)
+ right(CSstay);
+
+ for (; Point < End && isalnum((int)*p); Point++, p++)
+ if (move == CSmove)
+ right(CSstay);
+
+ if (Point == End)
+ break;
+ } while (++i < Repeat);
+
+ return CSstay;
+}
+
+static STATUS
+do_case(CASE type)
+{
+ int i;
+ int end;
+ int count;
+ char *p;
+
+ (void)do_forward(CSstay);
+ if (OldPoint != Point) {
+ if ((count = Point - OldPoint) < 0)
+ count = -count;
+ Point = OldPoint;
+ if ((end = Point + count) > End)
+ end = End;
+ for (i = Point, p = &Line[i]; i < end; i++, p++) {
+ if (type == TOupper) {
+ if (islower((int)*p))
+ *p = toupper((int)*p);
+ }
+ else if (isupper((int)*p))
+ *p = tolower((int)*p);
+ right(CSmove);
+ }
+ }
+ return CSstay;
+}
+
+static STATUS
+case_down_word()
+{
+ return do_case(TOlower);
+}
+
+static STATUS
+case_up_word()
+{
+ return do_case(TOupper);
+}
+
+static void
+ceol()
+{
+ int extras;
+ int i;
+ char *p;
+
+ for (extras = 0, i = Point, p = &Line[i]; i <= End; i++, p++) {
+ TTYput(' ');
+ if (*p == TAB) {
+ /* XXX */
+ }
+ else if (ISCTL(*p)) {
+ TTYput(' ');
+ extras++;
+ }
+ else if (rl_meta_chars && ISMETA(*p)) {
+ TTYput(' ');
+ TTYput(' ');
+ extras += 2;
+ }
+ }
+
+ for (i += extras; i > Point; i--)
+ TTYback();
+}
+
+static void
+clear_line()
+{
+ Point = -strlen(Prompt);
+ TTYput('\r');
+ ceol();
+ Point = 0;
+ End = 0;
+ Line[0] = '\0';
+}
+
+static STATUS
+insert_string(char *p)
+{
+ size_t len;
+ int i;
+ char *new;
+ char *q;
+
+ len = strlen((char *)p);
+ if (End + len >= Length) {
+ if ((new = malloc(sizeof(char) * (Length + len + MEM_INC))) == NULL)
+ return CSstay;
+ if (Length) {
+ memcpy(new, Line, Length);
+ free(Line);
+ }
+ Line = new;
+ Length += len + MEM_INC;
+ }
+
+ for (q = &Line[Point], i = End - Point; --i >= 0; )
+ q[len + i] = q[i];
+ memcpy(&Line[Point], p, len);
+ End += len;
+ Line[End] = '\0';
+ TTYstring(&Line[Point]);
+ Point += len;
+
+ return Point == End ? CSstay : CSmove;
+}
+
+static STATUS
+redisplay()
+{
+ TTYputs((const char *)NEWLINE);
+ TTYputs((const char *)Prompt);
+ TTYstring(Line);
+ return CSmove;
+}
+
+static STATUS
+redisplay_no_nl()
+{
+ TTYput('\r');
+ TTYputs((const char *)Prompt);
+ TTYstring(Line);
+ return CSmove;
+}
+
+static STATUS
+toggle_meta_mode()
+{
+ rl_meta_chars = !rl_meta_chars;
+ return redisplay();
+}
+
+
+static char *
+next_hist()
+{
+ return H.Pos >= H.Size - 1 ? NULL : H.Lines[++H.Pos];
+}
+
+static char *
+prev_hist()
+{
+ return H.Pos == 0 ? NULL : H.Lines[--H.Pos];
+}
+
+static STATUS
+do_insert_hist(char *p)
+{
+ if (p == NULL)
+ return ring_bell();
+ Point = 0;
+ reposition();
+ ceol();
+ End = 0;
+ return insert_string(p);
+}
+
+static STATUS
+do_hist(char *(*move)())
+{
+ char *p;
+ int i;
+
+ i = 0;
+ do {
+ if ((p = (*move)()) == NULL)
+ return ring_bell();
+ } while (++i < Repeat);
+ return do_insert_hist(p);
+}
+
+static STATUS
+h_next()
+{
+ return do_hist(next_hist);
+}
+
+static STATUS
+h_prev()
+{
+ return do_hist(prev_hist);
+}
+
+static STATUS
+h_first()
+{
+ return do_insert_hist(H.Lines[H.Pos = 0]);
+}
+
+static STATUS
+h_last()
+{
+ return do_insert_hist(H.Lines[H.Pos = H.Size - 1]);
+}
+
+/*
+** Return zero if pat appears as a substring in text.
+*/
+static int
+substrcmp(char *text, char *pat,int len)
+{
+ char c;
+
+ if ((c = *pat) == '\0')
+ return *text == '\0';
+ for ( ; *text; text++)
+ if (*text == c && strncmp(text, pat, len) == 0)
+ return 0;
+ return 1;
+}
+
+static char *
+search_hist(char *search,char *(*move)())
+{
+ static char *old_search;
+ int len;
+ int pos;
+ int (*match)();
+ char *pat;
+
+ /* Save or get remembered search pattern. */
+ if (search && *search) {
+ if (old_search)
+ free(old_search);
+ old_search = strdup(search);
+ }
+ else {
+ if (old_search == NULL || *old_search == '\0')
+ return NULL;
+ search = old_search;
+ }
+
+ /* Set up pattern-finder. */
+ if (*search == '^') {
+ match = strncmp;
+ pat = (char *)(search + 1);
+ }
+ else {
+ match = substrcmp;
+ pat = (char *)search;
+ }
+ len = strlen(pat);
+
+ for (pos = H.Pos; (*move)() != NULL; )
+ if ((*match)((char *)H.Lines[H.Pos], pat, len) == 0)
+ return H.Lines[H.Pos];
+ H.Pos = pos;
+ return NULL;
+}
+
+static STATUS
+h_search()
+{
+ static int Searching;
+ const char *old_prompt;
+ char *(*move)();
+ char *p;
+
+ if (Searching)
+ return ring_bell();
+ Searching = 1;
+
+ clear_line();
+ old_prompt = Prompt;
+ Prompt = "Search: ";
+ TTYputs((const char *)Prompt);
+ move = Repeat == NO_ARG ? prev_hist : next_hist;
+ p = editinput();
+ Searching = 0;
+ if (p == NULL && Signal > 0) {
+ Signal = 0;
+ clear_line();
+ Prompt = old_prompt;
+ return redisplay_no_nl();
+ }
+ p = search_hist(p, move);
+ clear_line();
+ Prompt = old_prompt;
+ if (p == NULL) {
+ (void)ring_bell();
+ return redisplay_no_nl();
+ }
+ return do_insert_hist(p);
+}
+
+static STATUS
+fd_char()
+{
+ int i;
+
+ i = 0;
+ do {
+ if (Point >= End)
+ break;
+ right(CSmove);
+ } while (++i < Repeat);
+ return CSstay;
+}
+
+static void
+save_yank(int begin, int i)
+{
+ if (Yanked) {
+ free(Yanked);
+ Yanked = NULL;
+ }
+
+ if (i < 1)
+ return;
+
+ if ((Yanked = malloc(sizeof(char) * (i + 1))) != NULL) {
+ memcpy(Yanked, &Line[begin], i);
+ Yanked[i] = '\0';
+ }
+}
+
+static STATUS
+delete_string(int count)
+{
+ int i;
+ char *p;
+
+ if (count <= 0 || End == Point)
+ return ring_bell();
+
+ if (count == 1 && Point == End - 1) {
+ /* Optimize common case of delete at end of line. */
+ End--;
+ p = &Line[Point];
+ i = 1;
+ TTYput(' ');
+ if (*p == TAB) {
+ /* XXX */
+ }
+ else if (ISCTL(*p)) {
+ i = 2;
+ TTYput(' ');
+ }
+ else if (rl_meta_chars && ISMETA(*p)) {
+ i = 3;
+ TTYput(' ');
+ TTYput(' ');
+ }
+ TTYbackn(i);
+ *p = '\0';
+ return CSmove;
+ }
+ if (Point + count > End && (count = End - Point) <= 0)
+ return CSstay;
+
+ if (count > 1)
+ save_yank(Point, count);
+
+ ceol();
+ for (p = &Line[Point], i = End - (Point + count) + 1; --i >= 0; p++)
+ p[0] = p[count];
+ End -= count;
+ TTYstring(&Line[Point]);
+ return CSmove;
+}
+
+static STATUS
+bk_char()
+{
+ int i;
+
+ i = 0;
+ do {
+ if (Point == 0)
+ break;
+ left(CSmove);
+ } while (++i < Repeat);
+
+ return CSstay;
+}
+
+static STATUS
+bk_del_char()
+{
+ int i;
+
+ i = 0;
+ do {
+ if (Point == 0)
+ break;
+ left(CSmove);
+ } while (++i < Repeat);
+
+ return delete_string(i);
+}
+
+static STATUS
+kill_line()
+{
+ int i;
+
+ if (Repeat != NO_ARG) {
+ if (Repeat < Point) {
+ i = Point;
+ Point = Repeat;
+ reposition();
+ (void)delete_string(i - Point);
+ }
+ else if (Repeat > Point) {
+ right(CSmove);
+ (void)delete_string(Repeat - Point - 1);
+ }
+ return CSmove;
+ }
+
+ save_yank(Point, End - Point);
+ ceol();
+ Line[Point] = '\0';
+ End = Point;
+ return CSstay;
+}
+
+static STATUS
+insert_char(int c)
+{
+ STATUS s;
+ char buff[2];
+ char *p;
+ char *q;
+ int i;
+
+ if (Repeat == NO_ARG || Repeat < 2) {
+ buff[0] = c;
+ buff[1] = '\0';
+ return insert_string(buff);
+ }
+
+ if ((p = malloc(sizeof(char) * (Repeat + 1))) == NULL)
+ return CSstay;
+ for (i = Repeat, q = p; --i >= 0; )
+ *q++ = c;
+ *q = '\0';
+ Repeat = 0;
+ s = insert_string(p);
+ free(p);
+ return s;
+}
+
+static STATUS
+meta()
+{
+ int c;
+ KEYMAP *kp;
+
+ if ((c = TTYget()) == EOF)
+ return CSeof;
+#if defined(ANSI_ARROWS)
+ /* Also include VT-100 arrows. */
+ if (c == '[' || c == 'O')
+ switch ((int)(c = TTYget())) {
+ default: return ring_bell();
+ case EOF: return CSeof;
+ case 'A': return h_prev();
+ case 'B': return h_next();
+ case 'C': return fd_char();
+ case 'D': return bk_char();
+ }
+#endif /* defined(ANSI_ARROWS) */
+
+ if (isdigit(c)) {
+ for (Repeat = c - '0'; (c = TTYget()) != EOF && isdigit(c); )
+ Repeat = Repeat * 10 + c - '0';
+ Pushed = 1;
+ PushBack = c;
+ return CSstay;
+ }
+
+ if (isupper(c))
+ return do_macro(c);
+ for (OldPoint = Point, kp = MetaMap; kp < &MetaMap[METAMAPSIZE]; kp++)
+ if (kp->Key == c && kp->Active)
+ return (*kp->Function)();
+
+ return ring_bell();
+}
+
+static STATUS
+emacs(int c)
+{
+ STATUS s;
+ KEYMAP *kp;
+
+#if 0
+ /* This test makes it impossible to enter eight-bit characters when
+ * meta-char mode is enabled. */
+ if (rl_meta_chars && ISMETA(c)) {
+ Pushed = 1;
+ PushBack = UNMETA(c);
+ return meta();
+ }
+#endif /* 0 */
+ for (kp = Map; kp < &Map[MAPSIZE]; kp++)
+ if (kp->Key == c && kp->Active)
+ break;
+ s = kp < &Map[MAPSIZE] ? (*kp->Function)() : insert_char((int)c);
+ if (!Pushed)
+ /* No pushback means no repeat count; hacky, but true. */
+ Repeat = NO_ARG;
+ return s;
+}
+
+static STATUS
+TTYspecial(int c)
+{
+ if (rl_meta_chars && ISMETA(c))
+ return CSdispatch;
+
+ if (c == rl_erase || c == DEL)
+ return bk_del_char();
+ if (c == rl_kill) {
+ if (Point != 0) {
+ Point = 0;
+ reposition();
+ }
+ Repeat = NO_ARG;
+ return kill_line();
+ }
+ if (c == rl_eof && Point == 0 && End == 0)
+ return CSeof;
+ if (c == rl_intr) {
+ Signal = SIGINT;
+ return CSsignal;
+ }
+ if (c == rl_quit) {
+ Signal = SIGQUIT;
+ return CSsignal;
+ }
+#if defined(DO_SIGTSTP)
+ if (c == rl_susp) {
+ Signal = SIGTSTP;
+ return CSsignal;
+ }
+#endif /* defined(DO_SIGTSTP) */
+
+ return CSdispatch;
+}
+
+static char *
+editinput()
+{
+ int c;
+
+ Repeat = NO_ARG;
+ OldPoint = Point = Mark = End = 0;
+ Line[0] = '\0';
+
+ Signal = -1;
+ while ((c = TTYget()) != EOF)
+ switch (TTYspecial(c)) {
+ case CSdone:
+ return Line;
+ case CSeof:
+ return NULL;
+ case CSsignal:
+ return (char *)"";
+ case CSmove:
+ reposition();
+ break;
+ case CSdispatch:
+ switch (emacs(c)) {
+ case CSdone:
+ return Line;
+ case CSeof:
+ return NULL;
+ case CSsignal:
+ return (char *)"";
+ case CSmove:
+ reposition();
+ break;
+ case CSdispatch:
+ case CSstay:
+ break;
+ }
+ break;
+ case CSstay:
+ break;
+ }
+ return NULL;
+}
+
+static void
+hist_add(char *p)
+{
+ int i;
+
+ if ((p = strdup(p)) == NULL)
+ return;
+ if (H.Size < HIST_SIZE)
+ H.Lines[H.Size++] = p;
+ else {
+ free(H.Lines[0]);
+ for (i = 0; i < HIST_SIZE - 1; i++)
+ H.Lines[i] = H.Lines[i + 1];
+ H.Lines[i] = p;
+ }
+ H.Pos = H.Size - 1;
+}
+
+static char *
+read_redirected()
+{
+ int size;
+ char *p;
+ char *line;
+ char *end;
+
+ for (size = MEM_INC, p = line = malloc(sizeof(char) * size), end = p + size; ; p++) {
+ if (p == end) {
+ size += MEM_INC;
+ p = line = realloc(line, size);
+ end = p + size;
+ }
+ if (read(0, p, 1) <= 0) {
+ /* Ignore "incomplete" lines at EOF, just like we do for a tty. */
+ free(line);
+ return NULL;
+ }
+ if (*p == '\n')
+ break;
+ }
+ *p = '\0';
+ return line;
+}
+
+/*
+** For compatibility with FSF readline.
+*/
+/* ARGSUSED0 */
+void
+rl_reset_terminal(char *p)
+{
+ (void)p; /* Suppress warning */
+}
+
+void
+rl_initialize()
+{
+}
+
+int
+rl_insert(int count, int c)
+{
+ if (count > 0) {
+ Repeat = count;
+ (void)insert_char(c);
+ (void)redisplay_no_nl();
+ }
+ return 0;
+}
+
+int (*rl_event_hook)();
+
+int
+rl_key_action(int c, char flag)
+{
+ KEYMAP *kp;
+ int size;
+
+ (void)flag; /* Suppress warning */
+
+ if (ISMETA(c)) {
+ kp = MetaMap;
+ size = METAMAPSIZE;
+ }
+ else {
+ kp = Map;
+ size = MAPSIZE;
+ }
+ for ( ; --size >= 0; kp++)
+ if (kp->Key == c) {
+ kp->Active = c ? 1 : 0;
+ return 1;
+ }
+ return -1;
+}
+
+char *
+readline(const char *prompt)
+{
+ char *line;
+ int s;
+
+ if (!isatty(0)) {
+ TTYflush();
+ return read_redirected();
+ }
+
+ if (Line == NULL) {
+ Length = MEM_INC;
+ if ((Line = malloc(sizeof(char) * Length)) == NULL)
+ return NULL;
+ }
+
+ TTYinfo();
+ rl_ttyset(0);
+ hist_add(NIL);
+ ScreenSize = SCREEN_INC;
+ Screen = malloc(sizeof(char) * ScreenSize);
+ Prompt = prompt ? prompt : (char *)NIL;
+ TTYputs((const char *)Prompt);
+ if ((line = editinput()) != NULL) {
+ line = strdup(line);
+ TTYputs((const char *)NEWLINE);
+ TTYflush();
+ }
+ rl_ttyset(1);
+ free(Screen);
+ free(H.Lines[--H.Size]);
+ if (Signal > 0) {
+ s = Signal;
+ Signal = 0;
+ (void)kill(getpid(), s);
+ }
+ return (char *)line;
+}
+
+void
+add_history(char *p)
+{
+ if (p == NULL || *p == '\0')
+ return;
+
+#if defined(UNIQUE_HISTORY)
+ if (H.Size && strcmp(p, H.Lines[H.Size - 1]) == 0)
+ return;
+#endif /* defined(UNIQUE_HISTORY) */
+ hist_add((char *)p);
+}
+
+
+static STATUS
+beg_line()
+{
+ if (Point) {
+ Point = 0;
+ return CSmove;
+ }
+ return CSstay;
+}
+
+static STATUS
+del_char()
+{
+ return delete_string(Repeat == NO_ARG ? 1 : Repeat);
+}
+
+static STATUS
+end_line()
+{
+ if (Point != End) {
+ Point = End;
+ return CSmove;
+ }
+ return CSstay;
+}
+
+/*
+** Return allocated copy of word under cursor, moving cursor after the
+** word.
+*/
+static char *
+find_word()
+{
+ static char SEPS[] = "\"#;&|^$=`'{}()<>\n\t ";
+ char *p;
+ char *new;
+ size_t len;
+
+ /* Move forward to end of word. */
+ p = &Line[Point];
+ for ( ; Point < End && strchr(SEPS, (char)*p) == NULL; Point++, p++)
+ right(CSstay);
+
+ /* Back up to beginning of word. */
+ for (p = &Line[Point]; p > Line && strchr(SEPS, (char)p[-1]) == NULL; p--)
+ continue;
+ len = Point - (p - Line) + 1;
+ if ((new = malloc(sizeof(char) * len)) == NULL)
+ return NULL;
+ memcpy(new, p, len);
+ new[len - 1] = '\0';
+ return new;
+}
+
+static STATUS
+c_complete()
+{
+ char *p;
+ char *word;
+ int unique;
+
+ word = find_word();
+ p = (char *)rl_complete((char *)word, &unique);
+ if (word)
+ free(word);
+ if (p && *p) {
+ (void)insert_string(p);
+ if (!unique)
+ (void)ring_bell();
+ free(p);
+ return redisplay_no_nl();
+ }
+ return ring_bell();
+}
+
+static STATUS
+c_possible()
+{
+ char **av;
+ char *word;
+ int ac;
+
+ word = find_word();
+ ac = rl_list_possib((char *)word, (char ***)&av);
+ if (word)
+ free(word);
+ if (ac) {
+ columns(ac, av);
+ while (--ac >= 0)
+ free(av[ac]);
+ free(av);
+ return redisplay_no_nl();
+ }
+ return ring_bell();
+}
+
+static STATUS
+accept_line()
+{
+ Line[End] = '\0';
+ return CSdone;
+}
+
+static STATUS
+transpose()
+{
+ char c;
+
+ if (Point) {
+ if (Point == End)
+ left(CSmove);
+ c = Line[Point - 1];
+ left(CSstay);
+ Line[Point - 1] = Line[Point];
+ TTYshow(Line[Point - 1]);
+ Line[Point++] = c;
+ TTYshow(c);
+ }
+ return CSstay;
+}
+
+static STATUS
+quote()
+{
+ int c;
+
+ return (c = TTYget()) == EOF ? CSeof : insert_char((int)c);
+}
+
+static STATUS
+wipe()
+{
+ int i;
+
+ if (Mark > End)
+ return ring_bell();
+
+ if (Point > Mark) {
+ i = Point;
+ Point = Mark;
+ Mark = i;
+ reposition();
+ }
+
+ return delete_string(Mark - Point);
+}
+
+static STATUS
+mk_set()
+{
+ Mark = Point;
+ return CSstay;
+}
+
+static STATUS
+exchange()
+{
+ int c;
+
+ if ((c = TTYget()) != CTL('X'))
+ return c == EOF ? CSeof : ring_bell();
+
+ if ((c = Mark) <= End) {
+ Mark = Point;
+ Point = c;
+ return CSmove;
+ }
+ return CSstay;
+}
+
+static STATUS
+yank()
+{
+ if (Yanked && *Yanked)
+ return insert_string(Yanked);
+ return CSstay;
+}
+
+static STATUS
+copy_region()
+{
+ if (Mark > End)
+ return ring_bell();
+
+ if (Point > Mark)
+ save_yank(Mark, Point - Mark);
+ else
+ save_yank(Point, Mark - Point);
+
+ return CSstay;
+}
+
+static STATUS
+move_to_char()
+{
+ int c;
+ int i;
+ char *p;
+
+ if ((c = TTYget()) == EOF)
+ return CSeof;
+ for (i = Point + 1, p = &Line[i]; i < End; i++, p++)
+ if (*p == c) {
+ Point = i;
+ return CSmove;
+ }
+ return CSstay;
+}
+
+static STATUS
+fd_word()
+{
+ return do_forward(CSmove);
+}
+
+static STATUS
+fd_kill_word()
+{
+ int i;
+
+ (void)do_forward(CSstay);
+ if (OldPoint != Point) {
+ i = Point - OldPoint;
+ Point = OldPoint;
+ return delete_string(i);
+ }
+ return CSstay;
+}
+
+static STATUS
+bk_word()
+{
+ int i;
+ char *p;
+
+ i = 0;
+ do {
+ for (p = &Line[Point]; p > Line && !isalnum((int)p[-1]); p--)
+ left(CSmove);
+
+ for (; p > Line && p[-1] != ' ' && isalnum((int)p[-1]); p--)
+ left(CSmove);
+
+ if (Point == 0)
+ break;
+ } while (++i < Repeat);
+
+ return CSstay;
+}
+
+static STATUS
+bk_kill_word()
+{
+ (void)bk_word();
+ if (OldPoint != Point)
+ return delete_string(OldPoint - Point);
+ return CSstay;
+}
+
+static int
+argify(char *line, char ***avp)
+{
+ char *c;
+ char **p;
+ char **new;
+ int ac;
+ int i;
+
+ i = MEM_INC;
+ if ((*avp = p = malloc(sizeof(char*) * i))== NULL)
+ return 0;
+
+ for (c = line; isspace((int)*c); c++)
+ continue;
+ if (*c == '\n' || *c == '\0')
+ return 0;
+
+ for (ac = 0, p[ac++] = c; *c && *c != '\n'; ) {
+ if (isspace((int)*c)) {
+ *c++ = '\0';
+ if (*c && *c != '\n') {
+ if (ac + 1 == i) {
+ new = malloc(sizeof(char*) * (i + MEM_INC));
+ if (new == NULL) {
+ p[ac] = NULL;
+ return ac;
+ }
+ memcpy(new, p, i * sizeof (char **));
+ i += MEM_INC;
+ free(p);
+ *avp = p = new;
+ }
+ p[ac++] = c;
+ }
+ }
+ else
+ c++;
+ }
+ *c = '\0';
+ p[ac] = NULL;
+ return ac;
+}
+
+static STATUS
+last_argument()
+{
+ char **av;
+ char *p;
+ STATUS s;
+ int ac;
+
+ if (H.Size == 1 || (p = H.Lines[H.Size - 2]) == NULL)
+ return ring_bell();
+
+ if ((p = strdup(p)) == NULL)
+ return CSstay;
+ ac = argify(p, &av);
+
+ if (Repeat != NO_ARG)
+ s = Repeat < ac ? insert_string(av[Repeat]) : ring_bell();
+ else
+ s = ac ? insert_string(av[ac - 1]) : CSstay;
+
+ if (ac)
+ free(av);
+ free(p);
+ return s;
+}
+
+static KEYMAP Map[MAPSIZE] = {
+ { CTL('@'), 1, ring_bell },
+ { CTL('A'), 1, beg_line },
+ { CTL('B'), 1, bk_char },
+ { CTL('D'), 1, del_char },
+ { CTL('E'), 1, end_line },
+ { CTL('F'), 1, fd_char },
+ { CTL('G'), 1, ring_bell },
+ { CTL('H'), 1, bk_del_char },
+ { CTL('I'), 1, c_complete },
+ { CTL('J'), 1, accept_line },
+ { CTL('K'), 1, kill_line },
+ { CTL('L'), 1, redisplay },
+ { CTL('M'), 1, accept_line },
+ { CTL('N'), 1, h_next },
+ { CTL('O'), 1, ring_bell },
+ { CTL('P'), 1, h_prev },
+ { CTL('Q'), 1, ring_bell },
+ { CTL('R'), 1, h_search },
+ { CTL('S'), 1, ring_bell },
+ { CTL('T'), 1, transpose },
+ { CTL('U'), 1, ring_bell },
+ { CTL('V'), 1, quote },
+ { CTL('W'), 1, wipe },
+ { CTL('X'), 1, exchange },
+ { CTL('Y'), 1, yank },
+ { CTL('Z'), 1, ring_bell },
+ { CTL('['), 1, meta },
+ { CTL(']'), 1, move_to_char },
+ { CTL('^'), 1, ring_bell },
+ { CTL('_'), 1, ring_bell },
+};
+
+static KEYMAP MetaMap[16]= {
+ { CTL('H'), 1, bk_kill_word },
+ { CTL('['), 1, c_possible },
+ { DEL, 1, bk_kill_word },
+ { ' ', 1, mk_set },
+ { '.', 1, last_argument },
+ { '<', 1, h_first },
+ { '>', 1, h_last },
+ { '?', 1, c_possible },
+ { 'b', 1, bk_word },
+ { 'd', 1, fd_kill_word },
+ { 'f', 1, fd_word },
+ { 'l', 1, case_down_word },
+ { 'm', 1, toggle_meta_mode},
+ { 'u', 1, case_up_word },
+ { 'y', 1, yank },
+ { 'w', 1, copy_region },
+};
diff --git a/ndb/src/common/editline/editline_internal.h b/ndb/src/common/editline/editline_internal.h
new file mode 100644
index 00000000000..93c13e55edc
--- /dev/null
+++ b/ndb/src/common/editline/editline_internal.h
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* $Revision: 1.2 $
+**
+** Internal header file for editline library.
+*/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#if defined(SYS_UNIX)
+#include "unix.h"
+#endif /* defined(SYS_UNIX) */
+
+#define MEM_INC 64
+#define SCREEN_INC 256
+
+/*
+** Variables and routines internal to this package.
+*/
+extern int rl_eof;
+extern int rl_erase;
+extern int rl_intr;
+extern int rl_kill;
+extern int rl_quit;
+#if defined(DO_SIGTSTP)
+extern int rl_susp;
+#endif /* defined(DO_SIGTSTP) */
+extern char *rl_complete();
+extern int rl_list_possib();
+extern void rl_ttyset();
+extern void rl_add_slash();
+
diff --git a/ndb/src/common/editline/editline_win32.c b/ndb/src/common/editline/editline_win32.c
new file mode 100644
index 00000000000..feef0108523
--- /dev/null
+++ b/ndb/src/common/editline/editline_win32.c
@@ -0,0 +1,33 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <stdio.h>
+#include <malloc.h>
+
+
+char* readline(const char* prompt)
+{
+ char* szBuf;
+ printf(prompt);
+ szBuf = (char*)malloc(256);
+ return gets(szBuf);
+}
+
+void add_history(char* pch)
+{
+}
+
diff --git a/ndb/src/common/editline/sysunix.c b/ndb/src/common/editline/sysunix.c
new file mode 100644
index 00000000000..000bca78dfc
--- /dev/null
+++ b/ndb/src/common/editline/sysunix.c
@@ -0,0 +1,143 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* $Revision: 1.4 $
+**
+** Unix system-dependant routines for editline library.
+*/
+#include "editline_internal.h"
+
+#if defined(HAVE_TCGETATTR)
+#include <termios.h>
+
+void
+rl_ttyset(int Reset)
+{
+ static struct termios old;
+ struct termios new;
+
+ if (Reset == 0) {
+ if (tcgetattr(0, &old) < 0) perror("tcgetattr");
+ rl_erase = old.c_cc[VERASE];
+ rl_kill = old.c_cc[VKILL];
+ rl_eof = old.c_cc[VEOF];
+ rl_intr = old.c_cc[VINTR];
+ rl_quit = old.c_cc[VQUIT];
+#if defined(DO_SIGTSTP)
+ rl_susp = old.c_cc[VSUSP];
+#endif /* defined(DO_SIGTSTP) */
+
+ new = old;
+ new.c_lflag &= ~(ECHO | ICANON | ISIG);
+ new.c_iflag &= ~(ISTRIP | INPCK);
+ new.c_cc[VMIN] = 1;
+ new.c_cc[VTIME] = 0;
+ if (tcsetattr(0, TCSADRAIN, &new) < 0) perror("tcsetattr");
+ }
+ else
+ (void)tcsetattr(0, TCSADRAIN, &old);
+}
+
+#else
+#if defined(HAVE_TERMIO)
+#include <termio.h>
+
+void
+rl_ttyset(int Reset)
+{
+ static struct termio old;
+ struct termio new;
+
+ if (Reset == 0) {
+ (void)ioctl(0, TCGETA, &old);
+ rl_erase = old.c_cc[VERASE];
+ rl_kill = old.c_cc[VKILL];
+ rl_eof = old.c_cc[VEOF];
+ rl_intr = old.c_cc[VINTR];
+ rl_quit = old.c_cc[VQUIT];
+#if defined(DO_SIGTSTP)
+ rl_susp = old.c_cc[VSUSP];
+#endif /* defined(DO_SIGTSTP) */
+
+ new = old;
+ new.c_lflag &= ~(ECHO | ICANON | ISIG);
+ new.c_iflag &= ~(ISTRIP | INPCK);
+ new.c_cc[VMIN] = 1;
+ new.c_cc[VTIME] = 0;
+ (void)ioctl(0, TCSETAW, &new);
+ }
+ else
+ (void)ioctl(0, TCSETAW, &old);
+}
+
+#else
+#include <sgtty.h>
+
+void
+rl_ttyset(int Reset)
+{
+ static struct sgttyb old_sgttyb;
+ static struct tchars old_tchars;
+ struct sgttyb new_sgttyb;
+ struct tchars new_tchars;
+#if defined(DO_SIGTSTP)
+ struct ltchars old_ltchars;
+#endif /* defined(DO_SIGTSTP) */
+
+ if (Reset == 0) {
+ (void)ioctl(0, TIOCGETP, &old_sgttyb);
+ rl_erase = old_sgttyb.sg_erase;
+ rl_kill = old_sgttyb.sg_kill;
+
+ (void)ioctl(0, TIOCGETC, &old_tchars);
+ rl_eof = old_tchars.t_eofc;
+ rl_intr = old_tchars.t_intrc;
+ rl_quit = old_tchars.t_quitc;
+
+#if defined(DO_SIGTSTP)
+ (void)ioctl(0, TIOCGLTC, &old_ltchars);
+ rl_susp = old_ltchars.t_suspc;
+#endif /* defined(DO_SIGTSTP) */
+
+ new_sgttyb = old_sgttyb;
+ new_sgttyb.sg_flags &= ~ECHO;
+ new_sgttyb.sg_flags |= RAW;
+#if defined(PASS8)
+ new_sgttyb.sg_flags |= PASS8;
+#endif /* defined(PASS8) */
+ (void)ioctl(0, TIOCSETP, &new_sgttyb);
+
+ new_tchars = old_tchars;
+ new_tchars.t_intrc = -1;
+ new_tchars.t_quitc = -1;
+ (void)ioctl(0, TIOCSETC, &new_tchars);
+ }
+ else {
+ (void)ioctl(0, TIOCSETP, &old_sgttyb);
+ (void)ioctl(0, TIOCSETC, &old_tchars);
+ }
+}
+#endif /* defined(HAVE_TERMIO) */
+#endif /* defined(HAVE_TCGETATTR) */
+
+void
+rl_add_slash(char *path, char *p, size_t p_len)
+{
+ struct stat Sb;
+
+ if (stat(path, &Sb) >= 0)
+ (void)strlcat(p, S_ISDIR(Sb.st_mode) ? "/" : " ", p_len);
+}
diff --git a/ndb/src/common/editline/test/Makefile b/ndb/src/common/editline/test/Makefile
new file mode 100644
index 00000000000..20229d0aa62
--- /dev/null
+++ b/ndb/src/common/editline/test/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := util
+
+BIN_TARGET := editline_test
+BIN_TARGET_ARCHIVES := editline
+
+SOURCES = testit.c
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/common/editline/test/testit.c b/ndb/src/common/editline/test/testit.c
new file mode 100644
index 00000000000..9a7dfb7bbdf
--- /dev/null
+++ b/ndb/src/common/editline/test/testit.c
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* -*- c-basic-offset: 4; -*-
+** $Revision: 1.5 $
+**
+** A "micro-shell" to test editline library.
+** If given any arguments, commands aren't executed.
+*/
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+
+#include <NdbString.h>
+#include <editline/editline.h>
+
+int
+main(int argc, char **argv)
+{
+ char *prompt;
+ char *p;
+ int doit;
+
+ (void)argv; /* Suppress warning */
+
+ doit = argc == 1;
+ if ((prompt = getenv("TESTPROMPT")) == NULL)
+ prompt = "testit> ";
+
+ while ((p = readline(prompt)) != NULL) {
+ (void)printf("\t\t\t|%s|\n", p);
+ if (doit) {
+ if (strncmp(p, "cd ", 3) == 0) {
+ if (chdir(&p[3]) < 0)
+ perror(&p[3]);
+ } else {
+ if (system(p) != 0)
+ perror(p);
+ }
+ }
+ add_history(p);
+ free(p);
+ }
+
+ return 0;
+}
diff --git a/ndb/src/common/editline/unix.h b/ndb/src/common/editline/unix.h
new file mode 100644
index 00000000000..582c4888856
--- /dev/null
+++ b/ndb/src/common/editline/unix.h
@@ -0,0 +1,26 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* $Revision: 1.3 $
+**
+** Editline system header file for Unix.
+*/
+
+#define CRLF "\r\n"
+
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <dirent.h>
diff --git a/ndb/src/common/logger/ConsoleLogHandler.cpp b/ndb/src/common/logger/ConsoleLogHandler.cpp
new file mode 100644
index 00000000000..8f6a45fe5dd
--- /dev/null
+++ b/ndb/src/common/logger/ConsoleLogHandler.cpp
@@ -0,0 +1,68 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "ConsoleLogHandler.hpp"
+
+#include <NdbOut.hpp>
+
+ConsoleLogHandler::ConsoleLogHandler() : LogHandler()
+{
+}
+
+ConsoleLogHandler::~ConsoleLogHandler()
+{
+
+}
+
+bool
+ConsoleLogHandler::open()
+{
+ return true;
+}
+
+bool
+ConsoleLogHandler::close()
+{
+ return true;
+}
+
+//
+// PROTECTED
+//
+void
+ConsoleLogHandler::writeHeader(const char* pCategory, Logger::LoggerLevel level)
+{
+ char str[LogHandler::MAX_HEADER_LENGTH];
+ ndbout << getDefaultHeader(str, pCategory, level);
+}
+
+void
+ConsoleLogHandler::writeMessage(const char* pMsg)
+{
+ ndbout << pMsg;
+}
+
+void
+ConsoleLogHandler::writeFooter()
+{
+ ndbout << getDefaultFooter();
+}
+
+
+bool
+ConsoleLogHandler::setParam(const BaseString &param, const BaseString &value) {
+ return false;
+}
diff --git a/ndb/src/common/logger/FileLogHandler.cpp b/ndb/src/common/logger/FileLogHandler.cpp
new file mode 100644
index 00000000000..f3d547b4fe7
--- /dev/null
+++ b/ndb/src/common/logger/FileLogHandler.cpp
@@ -0,0 +1,241 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "FileLogHandler.hpp"
+
+#include <File.hpp>
+
+#include <NdbStdio.h>
+#include <sys/param.h>
+
+#include <errno.h>
+#include <string.h>
+
+//
+// PUBLIC
+//
+
+FileLogHandler::FileLogHandler() :
+ LogHandler(),
+ m_maxNoFiles(MAX_NO_FILES),
+ m_maxFileSize(MAX_FILE_SIZE),
+ m_maxLogEntries(MAX_LOG_ENTRIES)
+
+{
+ m_pLogFile = new File("logger.log", "a+");
+}
+
+FileLogHandler::FileLogHandler(const char* aFileName,
+ int maxNoFiles,
+ long maxFileSize,
+ unsigned int maxLogEntries) :
+ LogHandler(),
+ m_maxNoFiles(maxNoFiles),
+ m_maxFileSize(maxFileSize),
+ m_maxLogEntries(maxLogEntries)
+{
+ m_pLogFile = new File(aFileName, "a+");
+}
+
+FileLogHandler::~FileLogHandler()
+{
+ delete m_pLogFile;
+}
+
+bool
+FileLogHandler::open()
+{
+ bool rc = true;
+
+ if (m_pLogFile->open())
+ {
+ if (isTimeForNewFile())
+ {
+ if (!createNewFile())
+ {
+ setErrorCode(errno);
+ rc = false;
+ }
+ }
+ }
+ else
+ {
+ setErrorCode(errno);
+ rc = false;
+ }
+
+ return rc;
+}
+
+bool
+FileLogHandler::close()
+{
+ bool rc = true;
+ if (!m_pLogFile->close())
+ {
+ setErrorCode(errno);
+ rc = false;
+ }
+
+ return rc;
+}
+
+void
+FileLogHandler::writeHeader(const char* pCategory, Logger::LoggerLevel level)
+{
+ char str[LogHandler::MAX_HEADER_LENGTH];
+ m_pLogFile->writeChar(getDefaultHeader(str, pCategory, level));
+}
+
+void
+FileLogHandler::writeMessage(const char* pMsg)
+{
+ m_pLogFile->writeChar(pMsg);
+}
+
+void
+FileLogHandler::writeFooter()
+{
+ static int callCount = 0;
+ m_pLogFile->writeChar(getDefaultFooter());
+ /**
+ * The reason I also check the number of log entries instead of
+ * only the log size, is that I do not want to check the file size
+ * after each log entry which requires system calls and is quite slow.
+ * TODO: Any better way?
+ */
+ if (callCount % m_maxLogEntries != 0) // Check every m_maxLogEntries
+ {
+ if (isTimeForNewFile())
+ {
+ if (!createNewFile())
+ {
+ // Baby one more time...
+ createNewFile();
+ }
+ }
+ callCount = 0;
+ }
+ callCount++;
+
+ // Needed on Cello since writes to the flash disk does not happen until
+ // we flush and fsync.
+ m_pLogFile->flush();
+}
+
+
+//
+// PRIVATE
+//
+
+bool
+FileLogHandler::isTimeForNewFile()
+{
+ return (m_pLogFile->size() >= m_maxFileSize);
+}
+
+bool
+FileLogHandler::createNewFile()
+{
+ bool rc = true;
+ int fileNo = 1;
+ char newName[MAXPATHLEN];
+
+ do
+ {
+ if (fileNo >= m_maxNoFiles)
+ {
+ fileNo = 1;
+ ::snprintf(newName, sizeof(newName),
+ "%s.%d", m_pLogFile->getName(), fileNo);
+ break;
+ }
+ ::snprintf(newName, sizeof(newName),
+ "%s.%d", m_pLogFile->getName(), fileNo++);
+
+ } while (File::exists(newName));
+
+ m_pLogFile->close();
+ if (!File::rename(m_pLogFile->getName(), newName))
+ {
+ setErrorCode(errno);
+ rc = false;
+ }
+
+ // Open again
+ if (!m_pLogFile->open())
+ {
+ setErrorCode(errno);
+ rc = false;
+ }
+
+ return rc;
+}
+
+bool
+FileLogHandler::setParam(const BaseString &param, const BaseString &value){
+ if(param == "filename")
+ return setFilename(value);
+ if(param == "maxsize")
+ return setMaxSize(value);
+ if(param == "maxfiles")
+ return setMaxFiles(value);
+ return false;
+}
+
+bool
+FileLogHandler::setFilename(const BaseString &filename) {
+ close();
+ if(m_pLogFile)
+ delete m_pLogFile;
+ m_pLogFile = new File(filename.c_str(), "a+");
+ open();
+ return true;
+};
+
+bool
+FileLogHandler::setMaxSize(const BaseString &size) {
+ char *end;
+ long val = strtol(size.c_str(), &end, 0); /* XXX */
+ if(size.c_str() == end)
+ return false;
+ if(strncasecmp("M", end, 1) == 0)
+ val *= 1024*1024;
+ if(strncasecmp("k", end, 1) == 0)
+ val *= 1024;
+
+ m_maxFileSize = val;
+
+ return true;
+};
+
+bool
+FileLogHandler::setMaxFiles(const BaseString &files) {
+ char *end;
+ long val = strtol(files.c_str(), &end, 0);
+ if(files.c_str() == end)
+ return false;
+ m_maxNoFiles = val;
+
+ return true;
+};
+
+bool
+FileLogHandler::checkParams() {
+ if(m_pLogFile == NULL)
+ return false;
+ return true;
+}
diff --git a/ndb/src/common/logger/LogHandler.cpp b/ndb/src/common/logger/LogHandler.cpp
new file mode 100644
index 00000000000..d1445555e87
--- /dev/null
+++ b/ndb/src/common/logger/LogHandler.cpp
@@ -0,0 +1,142 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "LogHandler.hpp"
+
+#include <NdbTick.h>
+#include <NdbString.h>
+
+#include <time.h>
+
+//
+// PUBLIC
+//
+LogHandler::LogHandler() :
+ m_pDateTimeFormat("%d-%.2d-%.2d %.2d:%.2d:%.2d"),
+ m_errorCode(0)
+{
+}
+
+LogHandler::~LogHandler()
+{
+}
+
+void
+LogHandler::append(const char* pCategory, Logger::LoggerLevel level,
+ const char* pMsg)
+{
+ writeHeader(pCategory, level);
+ writeMessage(pMsg);
+ writeFooter();
+}
+
+const char*
+LogHandler::getDefaultHeader(char* pStr, const char* pCategory,
+ Logger::LoggerLevel level) const
+{
+ char time[MAX_DATE_TIME_HEADER_LENGTH];
+ ::snprintf(pStr, MAX_HEADER_LENGTH, "%s [%s] %s -- ",
+ getTimeAsString((char*)time),
+ pCategory,
+ Logger::LoggerLevelNames[level]);
+
+ return pStr;
+}
+
+
+const char*
+LogHandler::getDefaultFooter() const
+{
+ return "\n";
+}
+
+const char*
+LogHandler::getDateTimeFormat() const
+{
+ return m_pDateTimeFormat;
+}
+
+void
+LogHandler::setDateTimeFormat(const char* pFormat)
+{
+ m_pDateTimeFormat = (char*)pFormat;
+}
+
+char*
+LogHandler::getTimeAsString(char* pStr) const
+{
+ struct tm* tm_now;
+ time_t now;
+ now = ::time((time_t*)NULL);
+#ifdef NDB_WIN32
+ tm_now = localtime(&now);
+#else
+ tm_now = ::localtime(&now); //uses the "current" timezone
+#endif
+
+ ::snprintf(pStr, MAX_DATE_TIME_HEADER_LENGTH,
+ m_pDateTimeFormat,
+ tm_now->tm_year + 1900,
+ tm_now->tm_mon + 1, //month is [0,11]. +1 -> [1,12]
+ tm_now->tm_mday,
+ tm_now->tm_hour,
+ tm_now->tm_min,
+ tm_now->tm_sec);
+
+ return pStr;
+}
+
+int
+LogHandler::getErrorCode() const
+{
+ return m_errorCode;
+}
+
+void
+LogHandler::setErrorCode(int code)
+{
+ m_errorCode = code;
+}
+
+bool
+LogHandler::parseParams(const BaseString &_params) {
+ Vector<BaseString> v_args;
+
+ bool ret = true;
+
+ _params.split(v_args, ",");
+ for(size_t i=0; i < v_args.size(); i++) {
+ Vector<BaseString> v_param_value;
+
+ v_args[i].split(v_param_value, "=", 2);
+ if(v_param_value.size() == 2 &&
+ !setParam(v_param_value[0], v_param_value[1]))
+ ret = false;
+ }
+
+ if(!checkParams())
+ ret = false;
+ return ret;
+}
+
+bool
+LogHandler::checkParams() {
+ return true;
+}
+
+//
+// PRIVATE
+//
diff --git a/ndb/src/common/logger/LogHandlerList.cpp b/ndb/src/common/logger/LogHandlerList.cpp
new file mode 100644
index 00000000000..f020ad23e56
--- /dev/null
+++ b/ndb/src/common/logger/LogHandlerList.cpp
@@ -0,0 +1,183 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "LogHandlerList.hpp"
+
+#include <LogHandler.hpp>
+#include <NdbStdio.h>
+#include <ndb_types.h>
+
+//
+// PUBLIC
+//
+
+LogHandlerList::LogHandlerList() :
+ m_size(0),
+ m_pHeadNode(NULL),
+ m_pTailNode(NULL),
+ m_pCurrNode(NULL)
+{
+}
+
+LogHandlerList::~LogHandlerList()
+{
+ removeAll();
+}
+
+void
+LogHandlerList::add(LogHandler* pNewHandler)
+{
+ LogHandlerNode* pNode = new LogHandlerNode();
+
+ if (m_pHeadNode == NULL)
+ {
+ m_pHeadNode = pNode;
+ pNode->pPrev = NULL;
+ }
+ else
+ {
+ m_pTailNode->pNext = pNode;
+ pNode->pPrev = m_pTailNode;
+ }
+ m_pTailNode = pNode;
+ pNode->pNext = NULL;
+ pNode->pHandler = pNewHandler;
+
+ m_size++;
+}
+
+bool
+LogHandlerList::remove(LogHandler* pRemoveHandler)
+{
+ LogHandlerNode* pNode = m_pHeadNode;
+ bool removed = false;
+ do
+ {
+ if (pNode->pHandler == pRemoveHandler)
+ {
+ removeNode(pNode);
+ removed = true;
+ break;
+ }
+ } while ( (pNode = next(pNode)) != NULL);
+
+ return removed;
+}
+
+void
+LogHandlerList::removeAll()
+{
+ while (m_pHeadNode != NULL)
+ {
+ removeNode(m_pHeadNode);
+ }
+}
+
+LogHandler*
+LogHandlerList::next()
+{
+ LogHandler* pHandler = NULL;
+ if (m_pCurrNode == NULL)
+ {
+ m_pCurrNode = m_pHeadNode;
+ if (m_pCurrNode != NULL)
+ {
+ pHandler = m_pCurrNode->pHandler;
+ }
+ }
+ else
+ {
+ m_pCurrNode = next(m_pCurrNode); // Next node
+ if (m_pCurrNode != NULL)
+ {
+ pHandler = m_pCurrNode->pHandler;
+ }
+ }
+
+ return pHandler;
+}
+
+int
+LogHandlerList::size() const
+{
+ return m_size;
+}
+
+//
+// PRIVATE
+//
+
+LogHandlerList::LogHandlerNode*
+LogHandlerList::next(LogHandlerNode* pNode)
+{
+ LogHandlerNode* pCurr = pNode;
+ if (pNode->pNext != NULL)
+ {
+ pCurr = pNode->pNext;
+ }
+ else
+ {
+ // Tail
+ pCurr = NULL;
+ }
+ return pCurr;
+}
+
+LogHandlerList::LogHandlerNode*
+LogHandlerList::prev(LogHandlerNode* pNode)
+{
+ LogHandlerNode* pCurr = pNode;
+ if (pNode->pPrev != NULL) // head
+ {
+ pCurr = pNode->pPrev;
+ }
+ else
+ {
+ // Head
+ pCurr = NULL;
+ }
+
+ return pCurr;
+}
+
+void
+LogHandlerList::removeNode(LogHandlerNode* pNode)
+{
+ if (pNode->pPrev == NULL) // If head
+ {
+ m_pHeadNode = pNode->pNext;
+ }
+ else
+ {
+ pNode->pPrev->pNext = pNode->pNext;
+ }
+
+ if (pNode->pNext == NULL) // if tail
+ {
+ m_pTailNode = pNode->pPrev;
+ }
+ else
+ {
+ pNode->pNext->pPrev = pNode->pPrev;
+ }
+
+ pNode->pNext = NULL;
+ pNode->pPrev = NULL;
+ delete pNode->pHandler; // Delete log handler
+ delete pNode;
+
+ m_size--;
+}
diff --git a/ndb/src/common/logger/LogHandlerList.hpp b/ndb/src/common/logger/LogHandlerList.hpp
new file mode 100644
index 00000000000..7c001a28a76
--- /dev/null
+++ b/ndb/src/common/logger/LogHandlerList.hpp
@@ -0,0 +1,93 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LOGHANDLERLIST_H
+#define LOGHANDLERLIST_H
+
+class LogHandler;
+
+/**
+ * Provides a simple linked list of log handlers.
+ *
+ * @see LogHandler
+ * @version #@ $Id: LogHandlerList.hpp,v 1.2 2002/03/14 13:07:21 eyualex Exp $
+ */
+class LogHandlerList
+{
+public:
+ /**
+ * Default Constructor.
+ */
+ LogHandlerList();
+
+ /**
+ * Destructor.
+ */
+ ~LogHandlerList();
+
+ /**
+ * Adds a new log handler.
+ *
+ * @param pNewHandler log handler.
+ */
+ void add(LogHandler* pNewHandler);
+
+ /**
+ * Removes a log handler from the list and call its destructor.
+ *
+ * @param pRemoveHandler the handler to remove
+ */
+ bool remove(LogHandler* pRemoveHandler);
+
+ /**
+ * Removes all log handlers.
+ */
+ void removeAll();
+
+ /**
+ * Returns the next log handler in the list.
+ * returns a log handler or NULL.
+ */
+ LogHandler* next();
+
+ /**
+ * Returns the size of the list.
+ */
+ int size() const;
+private:
+ /** List node */
+ struct LogHandlerNode
+ {
+ LogHandlerNode* pPrev;
+ LogHandlerNode* pNext;
+ LogHandler* pHandler;
+ };
+
+ LogHandlerNode* next(LogHandlerNode* pNode);
+ LogHandlerNode* prev(LogHandlerNode* pNode);
+
+ void removeNode(LogHandlerNode* pNode);
+
+ int m_size;
+
+ LogHandlerNode* m_pHeadNode;
+ LogHandlerNode* m_pTailNode;
+ LogHandlerNode* m_pCurrNode;
+};
+
+#endif
+
+
diff --git a/ndb/src/common/logger/Logger.cpp b/ndb/src/common/logger/Logger.cpp
new file mode 100644
index 00000000000..6eaafd91854
--- /dev/null
+++ b/ndb/src/common/logger/Logger.cpp
@@ -0,0 +1,358 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdarg.h>
+
+#include "Logger.hpp"
+
+#include <LogHandler.hpp>
+#include <ConsoleLogHandler.hpp>
+#include <FileLogHandler.hpp>
+#include "LogHandlerList.hpp"
+
+#if !defined NDB_OSE || !defined NDB_SOFTOSE || !defined NDB_WIN32
+#include <SysLogHandler.hpp>
+#endif
+
+#include <NdbStdio.h>
+#include <ndb_types.h>
+#include <NdbString.h>
+
+#include <assert.h>
+
+//
+// PUBLIC
+//
+const char* Logger::LoggerLevelNames[] = { "OFF ",
+ "DEBUG ",
+ "INFO ",
+ "WARNING ",
+ "ERROR ",
+ "CRITICAL",
+ "ALERT ",
+ "ALL "
+ };
+Logger::Logger() :
+ m_pCategory("Logger"),
+ m_pConsoleHandler(NULL),
+ m_pFileHandler(NULL),
+ m_pSyslogHandler(NULL)
+{
+ m_pHandlerList = new LogHandlerList();
+ m_logLevels[LL_INFO] = true;
+}
+
+Logger::~Logger()
+{
+ removeAllHandlers();
+ delete m_pHandlerList;
+}
+
+void
+Logger::setCategory(const char* pCategory)
+{
+ m_pCategory = pCategory;
+}
+
+bool
+Logger::createConsoleHandler()
+{
+ bool rc = true;
+ if (m_pConsoleHandler == NULL)
+ {
+ m_pConsoleHandler = new ConsoleLogHandler();
+ if (!addHandler(m_pConsoleHandler)) // TODO: check error code
+ {
+ rc = false;
+ delete m_pConsoleHandler;
+ m_pConsoleHandler = NULL;
+ }
+ }
+
+ return rc;
+}
+
+void
+Logger::removeConsoleHandler()
+{
+ if (removeHandler(m_pConsoleHandler))
+ {
+ m_pConsoleHandler = NULL;
+ }
+}
+
+bool
+Logger::createFileHandler()
+{
+ bool rc = true;
+ if (m_pFileHandler == NULL)
+ {
+ m_pFileHandler = new FileLogHandler();
+ if (!addHandler(m_pFileHandler)) // TODO: check error code
+ {
+ rc = false;
+ delete m_pFileHandler;
+ m_pFileHandler = NULL;
+ }
+ }
+
+ return rc;
+}
+
+void
+Logger::removeFileHandler()
+{
+ if (removeHandler(m_pFileHandler))
+ {
+ m_pFileHandler = NULL;
+ }
+}
+
+bool
+Logger::createSyslogHandler()
+{
+ bool rc = true;
+ if (m_pSyslogHandler == NULL)
+ {
+#if defined NDB_OSE || defined NDB_SOFTOSE || defined NDB_WIN32
+ m_pSyslogHandler = new ConsoleLogHandler();
+#else
+ m_pSyslogHandler = new SysLogHandler();
+#endif
+ if (!addHandler(m_pSyslogHandler)) // TODO: check error code
+ {
+ rc = false;
+ delete m_pSyslogHandler;
+ m_pSyslogHandler = NULL;
+ }
+ }
+
+ return rc;
+}
+
+void
+Logger::removeSyslogHandler()
+{
+ if (removeHandler(m_pSyslogHandler))
+ {
+ m_pSyslogHandler = NULL;
+ }
+}
+
+bool
+Logger::addHandler(LogHandler* pHandler)
+{
+ assert(pHandler != NULL);
+
+ bool rc = pHandler->open();
+ if (rc)
+ {
+ m_pHandlerList->add(pHandler);
+ }
+ else
+ {
+ delete pHandler;
+ }
+
+ return rc;
+}
+
+bool
+Logger::addHandler(const BaseString &logstring) {
+ size_t i;
+ Vector<BaseString> logdest;
+ Vector<LogHandler *>loghandlers;
+
+ logstring.split(logdest, ";");
+
+ for(i = 0; i < logdest.size(); i++) {
+ Vector<BaseString> v_type_args;
+ logdest[i].split(v_type_args, ":", 2);
+
+ BaseString type(v_type_args[0]);
+ BaseString params;
+ if(v_type_args.size() >= 2)
+ params = v_type_args[1];
+
+ LogHandler *handler = NULL;
+
+ if(type == "SYSLOG") {
+ handler = new SysLogHandler();
+ } else if(type == "FILE")
+ handler = new FileLogHandler();
+ else if(type == "CONSOLE")
+ handler = new ConsoleLogHandler();
+
+ if(handler == NULL)
+ return false;
+ if(!handler->parseParams(params))
+ return false;
+ loghandlers.push_back(handler);
+ }
+
+ for(i = 0; i < loghandlers.size(); i++)
+ addHandler(loghandlers[i]);
+
+ return true; /* @todo handle errors */
+}
+
+bool
+Logger::removeHandler(LogHandler* pHandler)
+{
+ int rc = false;
+ if (pHandler != NULL)
+ {
+ rc = m_pHandlerList->remove(pHandler);
+ }
+
+ return rc;
+}
+
+void
+Logger::removeAllHandlers()
+{
+ m_pHandlerList->removeAll();
+}
+
+bool
+Logger::isEnable(LoggerLevel logLevel) const
+{
+ return m_logLevels[logLevel];
+}
+
+void
+Logger::enable(LoggerLevel logLevel)
+{
+ if (logLevel == LL_ALL)
+ {
+ for (int i = 1; i < MAX_LOG_LEVELS; i++)
+ {
+ m_logLevels[i] = true;
+ }
+ }
+ else
+ {
+ m_logLevels[logLevel] = true;
+ }
+}
+
+void
+Logger::enable(LoggerLevel fromLogLevel, LoggerLevel toLogLevel)
+{
+ if (fromLogLevel > toLogLevel)
+ {
+ LoggerLevel tmp = toLogLevel;
+ toLogLevel = fromLogLevel;
+ fromLogLevel = tmp;
+ }
+
+ for (int i = fromLogLevel; i <= toLogLevel; i++)
+ {
+ m_logLevels[i] = true;
+ }
+}
+
+void
+Logger::disable(LoggerLevel logLevel)
+{
+ if (logLevel == LL_ALL)
+ {
+ for (int i = 0; i < MAX_LOG_LEVELS; i++)
+ {
+ m_logLevels[i] = false;
+ }
+ }
+ else
+ {
+ m_logLevels[logLevel] = false;
+ }
+}
+
+void
+Logger::alert(const char* pMsg, ...) const
+{
+ va_list ap;
+ va_start(ap, pMsg);
+ log(LL_ALERT, pMsg, ap);
+ va_end(ap);
+}
+
+void
+Logger::critical(const char* pMsg, ...) const
+{
+ va_list ap;
+ va_start(ap, pMsg);
+ log(LL_CRITICAL, pMsg, ap);
+ va_end(ap);
+}
+void
+Logger::error(const char* pMsg, ...) const
+{
+ va_list ap;
+ va_start(ap, pMsg);
+ log(LL_ERROR, pMsg, ap);
+ va_end(ap);
+}
+void
+Logger::warning(const char* pMsg, ...) const
+{
+ va_list ap;
+ va_start(ap, pMsg);
+ log(LL_WARNING, pMsg, ap);
+ va_end(ap);
+}
+
+void
+Logger::info(const char* pMsg, ...) const
+{
+ va_list ap;
+ va_start(ap, pMsg);
+ log(LL_INFO, pMsg, ap);
+ va_end(ap);
+}
+
+void
+Logger::debug(const char* pMsg, ...) const
+{
+ va_list ap;
+ va_start(ap, pMsg);
+ log(LL_DEBUG, pMsg, ap);
+ va_end(ap);
+}
+
+//
+// PROTECTED
+//
+
+void
+Logger::log(LoggerLevel logLevel, const char* pMsg, va_list ap) const
+{
+ if (m_logLevels[LL_OFF] == false && m_logLevels[logLevel])
+ {
+ LogHandler* pHandler = NULL;
+ while ( (pHandler = m_pHandlerList->next()) != NULL)
+ {
+ char buf[1024];
+ vsnprintf(buf, sizeof(buf), pMsg, ap);
+ pHandler->append(m_pCategory, logLevel, buf);
+ }
+ }
+}
+
+//
+// PRIVATE
+//
+
diff --git a/ndb/src/common/logger/Makefile b/ndb/src/common/logger/Makefile
new file mode 100644
index 00000000000..994eb86ba35
--- /dev/null
+++ b/ndb/src/common/logger/Makefile
@@ -0,0 +1,27 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := logger
+
+DIRS := loggertest
+
+SOURCES := Logger.cpp LogHandlerList.cpp LogHandler.cpp \
+ ConsoleLogHandler.cpp FileLogHandler.cpp
+
+ifeq ($(NDB_OS), OSE)
+NO_SYSLOG := Y
+endif
+
+ifeq ($(NDB_OS), WIN32)
+NO_SYSLOG := Y
+endif
+
+ifneq ($(NO_SYSLOG), Y)
+SOURCES += SysLogHandler.cpp
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/src/common/logger/SysLogHandler.cpp b/ndb/src/common/logger/SysLogHandler.cpp
new file mode 100644
index 00000000000..f3511bf5638
--- /dev/null
+++ b/ndb/src/common/logger/SysLogHandler.cpp
@@ -0,0 +1,159 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "SysLogHandler.hpp"
+
+#include <syslog.h>
+#include <NdbString.h>
+
+//
+// PUBLIC
+//
+
+SysLogHandler::SysLogHandler() :
+ m_severity(LOG_INFO),
+ m_pIdentity("NDB"),
+ m_facility(LOG_USER)
+{
+}
+
+SysLogHandler::SysLogHandler(const char* pIdentity, int facility) :
+ m_severity(LOG_INFO),
+ m_pIdentity(pIdentity),
+ m_facility(facility)
+{
+
+}
+
+SysLogHandler::~SysLogHandler()
+{
+}
+
+bool
+SysLogHandler::open()
+{
+ ::setlogmask(LOG_UPTO(LOG_DEBUG)); // Log from EMERGENCY down to DEBUG
+ ::openlog(m_pIdentity, LOG_PID|LOG_CONS|LOG_ODELAY, m_facility); // PID, CONSOLE delay openlog
+
+ return true;
+}
+
+bool
+SysLogHandler::close()
+{
+ ::closelog();
+
+ return true;
+}
+
+void
+SysLogHandler::writeHeader(const char* pCategory, Logger::LoggerLevel level)
+{
+ // Save category to be used by writeMessage...
+ m_pCategory = pCategory;
+ // Map LogLevel to syslog severity
+ switch (level)
+ {
+ case Logger::LL_ALERT:
+ m_severity = LOG_ALERT;
+ break;
+ case Logger::LL_CRITICAL:
+ m_severity = LOG_CRIT;
+ break;
+ case Logger::LL_ERROR:
+ m_severity = LOG_ERR;
+ break;
+ case Logger::LL_WARNING:
+ m_severity = LOG_WARNING;
+ break;
+ case Logger::LL_INFO:
+ m_severity = LOG_INFO;
+ break;
+ case Logger::LL_DEBUG:
+ m_severity = LOG_DEBUG;
+ break;
+ default:
+ m_severity = LOG_INFO;
+ break;
+ }
+
+}
+
+void
+SysLogHandler::writeMessage(const char* pMsg)
+{
+ ::syslog(m_facility | m_severity, "[%s] %s", m_pCategory, pMsg);
+}
+
+void
+SysLogHandler::writeFooter()
+{
+ // Need to close it everytime? Do we run out of file descriptors?
+ //::closelog();
+}
+
+bool
+SysLogHandler::setParam(const BaseString &param, const BaseString &value) {
+ if(param == "facility") {
+ return setFacility(value);
+ }
+ return false;
+}
+
+static const struct syslog_facility {
+ char *name;
+ int value;
+} facilitynames[] = {
+ { "auth", LOG_AUTH },
+#ifdef LOG_AUTHPRIV
+ { "authpriv", LOG_AUTHPRIV },
+#endif
+ { "cron", LOG_CRON },
+ { "daemon", LOG_DAEMON },
+#ifdef LOG_FTP
+ { "ftp", LOG_FTP },
+#endif
+ { "kern", LOG_KERN },
+ { "lpr", LOG_LPR },
+ { "mail", LOG_MAIL },
+ { "news", LOG_NEWS },
+ { "syslog", LOG_SYSLOG },
+ { "user", LOG_USER },
+ { "uucp", LOG_UUCP },
+ { "local0", LOG_LOCAL0 },
+ { "local1", LOG_LOCAL1 },
+ { "local2", LOG_LOCAL2 },
+ { "local3", LOG_LOCAL3 },
+ { "local4", LOG_LOCAL4 },
+ { "local5", LOG_LOCAL5 },
+ { "local6", LOG_LOCAL6 },
+ { "local7", LOG_LOCAL7 },
+ { NULL, -1 }
+};
+
+bool
+SysLogHandler::setFacility(const BaseString &facility) {
+ const struct syslog_facility *c;
+ for(c = facilitynames; c->name != NULL; c++) {
+ if(facility == c->name) {
+ m_facility = c->value;
+ close();
+ open();
+ return true;
+ }
+ }
+ return false;
+}
diff --git a/ndb/src/common/logger/listtest/LogHandlerListUnitTest.cpp b/ndb/src/common/logger/listtest/LogHandlerListUnitTest.cpp
new file mode 100644
index 00000000000..22f67d15659
--- /dev/null
+++ b/ndb/src/common/logger/listtest/LogHandlerListUnitTest.cpp
@@ -0,0 +1,165 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "LogHandlerListUnitTest.hpp"
+
+#include <ConsoleLogHandler.hpp>
+#include <FileLogHandler.hpp>
+#include <SysLogHandler.hpp>
+
+#include <NdbOut.hpp>
+
+#include <stdio.h>
+#include <assert.h>
+
+typedef bool (*TESTFUNC)(const char*);
+typedef struct
+{
+ const char* name;
+ TESTFUNC test;
+}Tests;
+
+static Tests testCases[] = { {"Add", &LogHandlerListUnitTest::testAdd},
+ {"Remove", &LogHandlerListUnitTest::testRemove},
+ {"Traverse Next", &LogHandlerListUnitTest::testTraverseNext}
+ };
+
+
+int testFailed = 0;
+
+int main(int argc, char* argv[])
+{
+ char str[256];
+ int testCount = (sizeof(testCases) / sizeof(Tests));
+ ndbout << "Starting " << testCount << " tests..." << endl;
+ for (int i = 0; i < testCount; i++)
+ {
+ ndbout << "-- " << " Test " << i + 1
+ << " [" << testCases[i].name << "] --" << endl;
+ snprintf(str, 256, "%s %s %s %d", "Logging ",
+ testCases[i].name, " message ", i);
+ if (testCases[i].test(str))
+ {
+ ndbout << "-- Passed --" << endl;
+ }
+ else
+ {
+ ndbout << "-- Failed -- " << endl;
+ }
+
+ }
+ ndbout << endl << "-- " << testCount - testFailed << " passed, "
+ << testFailed << " failed --" << endl;
+
+ return 0;
+}
+
+bool
+LogHandlerListUnitTest::testAdd(const char* msg)
+{
+ bool rc = true;
+ LogHandlerList list;
+ int size = 10;
+ for (int i = 0; i < size; i++)
+ {
+ list.add(new ConsoleLogHandler());
+ }
+ if (list.size() != size)
+ {
+ rc = false;
+ }
+ ndbout << "List size: " << list.size() << endl;
+
+
+ return rc;
+}
+bool
+LogHandlerListUnitTest::testRemove(const char* msg)
+{
+ bool rc = true;
+
+ LogHandlerList list;
+ int size = 10;
+ LogHandler* pHandlers[10];
+ for (int i = 0; i < size; i++)
+ {
+ pHandlers[i] = new ConsoleLogHandler();
+ list.add(pHandlers[i]);
+ }
+
+ // Remove
+
+ for (int i = 0; i < size; i++)
+ {
+ if (!list.remove(pHandlers[i]))
+ {
+ ndbout << "Could not remove handler!" << endl;
+ }
+ else
+ {
+ ndbout << "List size: " << list.size() << endl;
+ }
+ }
+
+ return rc;
+
+}
+bool
+LogHandlerListUnitTest::testTraverseNext(const char* msg)
+{
+ bool rc = true;
+ LogHandlerList list;
+ int size = 10;
+ LogHandler* pHandlers[10];
+
+ for (int i = 0; i < size; i++)
+ {
+ char* str = new char[3];
+ pHandlers[i] = new ConsoleLogHandler();
+ ::snprintf(str, 3, "%d", i);
+ pHandlers[i]->setDateTimeFormat(str);
+ list.add(pHandlers[i]);
+ }
+
+ ndbout << "List size: " << list.size() << endl;
+
+ LogHandler* pHandler = NULL;
+ int i = 0;
+ while ((pHandler = list.next()) != NULL)
+ {
+ ndbout << "Handler[" << i++ << "]:dateformat = "
+ << pHandler->getDateTimeFormat() << endl;
+ }
+
+ list.removeAll();
+
+ return rc;
+
+}
+
+void
+LogHandlerListUnitTest::error(const char* msg)
+{
+ testFailed++;
+ ndbout << "Test failed: " << msg << endl;
+}
+
+LogHandlerListUnitTest::LogHandlerListUnitTest()
+{
+}
+LogHandlerListUnitTest::~LogHandlerListUnitTest()
+{
+}
diff --git a/ndb/src/common/logger/listtest/LogHandlerListUnitTest.hpp b/ndb/src/common/logger/listtest/LogHandlerListUnitTest.hpp
new file mode 100644
index 00000000000..e98a2722b8d
--- /dev/null
+++ b/ndb/src/common/logger/listtest/LogHandlerListUnitTest.hpp
@@ -0,0 +1,40 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LOGHANDLERLISTUNITTEST_H
+#define LOGHANDLERLISTUNITTEST_H
+
+#include "LogHandlerList.hpp"
+
+/**
+ * Unit test of LogHandlerList.
+ *
+ * @version #@ $Id: LogHandlerListUnitTest.hpp,v 1.1 2002/03/13 17:59:15 eyualex Exp $
+ */
+class LogHandlerListUnitTest
+{
+public:
+
+ static bool testAdd(const char* msg);
+ static bool testRemove(const char* msg);
+ static bool testTraverseNext(const char* msg);
+
+ void error(const char* msg);
+
+ LogHandlerListUnitTest();
+ ~LogHandlerListUnitTest();
+};
+#endif
diff --git a/ndb/src/common/logger/listtest/Makefile b/ndb/src/common/logger/listtest/Makefile
new file mode 100644
index 00000000000..4688a5e5a2f
--- /dev/null
+++ b/ndb/src/common/logger/listtest/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE :=
+
+BIN_TARGET := listtest
+BIN_TARGET_ARCHIVES := portlib logger general
+
+SOURCES := LogHandlerListUnitTest.cpp
+
+CCFLAGS_LOC += -I../ -I$(NDB_TOP)/include/logger -I$(NDB_TOP)/include/portlib
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/src/common/logger/loggertest/LoggerUnitTest.cpp b/ndb/src/common/logger/loggertest/LoggerUnitTest.cpp
new file mode 100644
index 00000000000..4b0241a0b03
--- /dev/null
+++ b/ndb/src/common/logger/loggertest/LoggerUnitTest.cpp
@@ -0,0 +1,201 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "LoggerUnitTest.hpp"
+
+#include <Logger.hpp>
+#include <ConsoleLogHandler.hpp>
+#include <FileLogHandler.hpp>
+
+#if !defined NDB_OSE || !defined NDB_SOFTOSE
+#include <SysLogHandler.hpp>
+#endif
+
+#include <NdbOut.hpp>
+#include <NdbStdio.h>
+#include <NdbMain.h>
+
+#include <string.h>
+#include <assert.h>
+
+typedef bool (*TESTFUNC)(const char*);
+typedef struct
+{
+ const char* name;
+ TESTFUNC test;
+}Tests;
+
+static Tests testCases[] = { {"Alert", &LoggerUnitTest::testAlert},
+ {"Critical", &LoggerUnitTest::testCritical},
+ {"Error", &LoggerUnitTest::testError},
+ {"Warning", &LoggerUnitTest::testWarning},
+ {"Info", &LoggerUnitTest::testInfo},
+ {"Debug", &LoggerUnitTest::testDebug},
+ {"Info to Critical", &LoggerUnitTest::testInfoCritical},
+ {"All", &LoggerUnitTest::testAll},
+ {"Off", &LoggerUnitTest::testOff}
+ };
+
+static Logger logger;
+int testFailed = 0;
+
+NDB_COMMAND(loggertest, "loggertest", "loggertest -console | -file",
+ "loggertest", 16384)
+{
+ if (argc < 2)
+ {
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ ndbout << "Usage: loggertest -console | -file" << endl;
+#else
+ ndbout << "Usage: loggertest -console | -file | -syslog" << endl;
+#endif
+ return 0;
+ }
+
+ if (strcmp(argv[1], "-console") == 0)
+ {
+ logger.createConsoleHandler();
+ }
+ else if (strcmp(argv[1], "-file") == 0)
+ {
+ logger.createFileHandler();
+ //logger.addHandler(new FileLogHandler(argv[2]));
+ }
+#if !defined NDB_OSE || !defined NDB_SOFTOSE
+ else if (strcmp(argv[1], "-syslog") == 0)
+ {
+ logger.createSyslogHandler();
+ }
+#endif
+
+ logger.disable(Logger::LL_ALL);
+
+ char str[256];
+ int testCount = (sizeof(testCases) / sizeof(Tests));
+ ndbout << "Starting " << testCount << " tests..." << endl;
+ for (int i = 0; i < testCount; i++)
+ {
+ ndbout << "-- " << " Test " << i + 1
+ << " [" << testCases[i].name << "] --" << endl;
+ ::snprintf(str, 256, "%s %s %s %d", "Logging ",
+ testCases[i].name, " message ", i);
+ if (testCases[i].test(str))
+ {
+ ndbout << "-- Passed --" << endl;
+ }
+ else
+ {
+ ndbout << "-- Failed -- " << endl;
+ }
+
+ }
+ ndbout << endl << "-- " << testCount - testFailed << " passed, "
+ << testFailed << " failed --" << endl;
+
+ logger.removeAllHandlers(); // Need to remove all for OSE,
+ // because logger is global
+ return 0;
+}
+
+bool
+LoggerUnitTest::logTo(Logger::LoggerLevel from, Logger::LoggerLevel to, const char* msg)
+{
+ logger.enable(from, to);
+ return logTo(from, msg);
+}
+
+bool
+LoggerUnitTest::logTo(Logger::LoggerLevel level, const char* msg)
+{
+ logger.enable(level);
+ logger.alert(msg);
+ logger.critical(msg);
+ logger.error(msg);
+ logger.warning(msg);
+ logger.info(msg);
+ logger.debug(msg);
+ logger.disable(level);
+ return true;
+}
+
+bool
+LoggerUnitTest::testAll(const char* msg)
+{
+ return logTo(Logger::LL_ALL, msg);
+}
+
+bool
+LoggerUnitTest::testOff(const char* msg)
+{
+ return logTo(Logger::LL_OFF, msg);
+
+}
+
+bool
+LoggerUnitTest::testAlert(const char* msg)
+{
+ return logTo(Logger::LL_ALERT, msg);
+}
+
+bool
+LoggerUnitTest::testCritical(const char* msg)
+{
+ return logTo(Logger::LL_CRITICAL, msg);
+}
+
+bool
+LoggerUnitTest::testError(const char* msg)
+{
+ return logTo(Logger::LL_ERROR, msg);
+}
+
+bool
+LoggerUnitTest::testWarning(const char* msg)
+{
+ return logTo(Logger::LL_WARNING, msg);
+}
+
+bool
+LoggerUnitTest::testInfo(const char* msg)
+{
+ return logTo(Logger::LL_INFO, msg);
+}
+
+bool
+LoggerUnitTest::testDebug(const char* msg)
+{
+ return logTo(Logger::LL_DEBUG, msg);
+}
+
+bool
+LoggerUnitTest::testInfoCritical(const char* msg)
+{
+ return logTo(Logger::LL_CRITICAL, Logger::LL_INFO, msg);
+}
+
+void
+LoggerUnitTest::error(const char* msg)
+{
+ testFailed++;
+ ndbout << "Test failed: " << msg << endl;
+}
+
+LoggerUnitTest::LoggerUnitTest()
+{
+}
+LoggerUnitTest::~LoggerUnitTest()
+{
+}
diff --git a/ndb/src/common/logger/loggertest/LoggerUnitTest.hpp b/ndb/src/common/logger/loggertest/LoggerUnitTest.hpp
new file mode 100644
index 00000000000..79f560750d5
--- /dev/null
+++ b/ndb/src/common/logger/loggertest/LoggerUnitTest.hpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LOGGERUNITTEST_H
+#define LOGGERUNITTEST_H
+
+#include "Logger.hpp"
+
+/**
+ * Unit test of Logger.
+ *
+ * @version #@ $Id: LoggerUnitTest.hpp,v 1.1 2002/03/13 17:55:31 eyualex Exp $
+ */
+class LoggerUnitTest
+{
+public:
+
+ static bool testAll(const char* msg);
+ static bool testOff(const char* msg);
+ static bool testAlert(const char* msg);
+ static bool testCritical(const char* msg);
+ static bool testError(const char* msg);
+ static bool testWarning(const char* msg);
+ static bool testInfo(const char* msg);
+ static bool testDebug(const char* msg);
+ static bool testInfoCritical(const char* msg);
+
+ static bool logTo(Logger::LoggerLevel level, const char* msg);
+ static bool logTo(Logger::LoggerLevel from, Logger::LoggerLevel to, const char* msg);
+
+ void error(const char* msg);
+
+ LoggerUnitTest();
+ ~LoggerUnitTest();
+};
+#endif
diff --git a/ndb/src/common/logger/loggertest/Makefile b/ndb/src/common/logger/loggertest/Makefile
new file mode 100644
index 00000000000..0aef0ca2bce
--- /dev/null
+++ b/ndb/src/common/logger/loggertest/Makefile
@@ -0,0 +1,16 @@
+include .defs.mk
+
+TYPE :=
+
+BIN_TARGET := loggertest
+BIN_TARGET_ARCHIVES := logger portlib general
+
+SOURCES := LoggerUnitTest.cpp
+
+CCFLAGS_LOC += -I$(NDB_TOP)/include/logger \
+ -I$(NDB_TOP)/include/util \
+ -I$(NDB_TOP)/include/portlib
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/src/common/mgmcommon/Config.cpp b/ndb/src/common/mgmcommon/Config.cpp
new file mode 100644
index 00000000000..5492394ee4a
--- /dev/null
+++ b/ndb/src/common/mgmcommon/Config.cpp
@@ -0,0 +1,255 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Config.hpp"
+#include <ctype.h>
+#include <string.h>
+#include "MgmtErrorReporter.hpp"
+#include <Properties.hpp>
+#include "ConfigInfo.hpp"
+
+//*****************************************************************************
+// Ctor / Dtor
+//*****************************************************************************
+
+Config::Config() {
+ m_info = new ConfigInfo();
+}
+
+Config::Config(const Config & org) :
+ Properties(org) {
+
+ m_info = new ConfigInfo();
+}
+
+Config::Config(const Properties & org) :
+ Properties(org) {
+
+ m_info = new ConfigInfo();
+}
+
+Config::~Config() {
+ delete m_info;
+}
+
+/*****************************************************************************/
+
+void
+Config::printAllNameValuePairs(NdbOut &out,
+ const Properties *prop,
+ const char* s) const {
+ Properties::Iterator it(prop);
+ const Properties * section = m_info->getInfo(s);
+ for (const char* n = it.first(); n != NULL; n = it.next()) {
+ Uint32 int_value;
+ const char* str_value;
+
+ if (m_info->getStatus(section, n) == ConfigInfo::INTERNAL)
+ continue;
+ if (m_info->getStatus(section, n) == ConfigInfo::DEPRICATED)
+ continue;
+ if (m_info->getStatus(section, n) == ConfigInfo::NOTIMPLEMENTED)
+ continue;
+
+ out << n << ": ";
+
+ switch (m_info->getType(section, n)) {
+ case ConfigInfo::INT:
+ MGM_REQUIRE(prop->get(n, &int_value));
+ out << int_value;
+ break;
+
+ case ConfigInfo::BOOL:
+ MGM_REQUIRE(prop->get(n, &int_value));
+ if (int_value) {
+ out << "Y";
+ } else {
+ out << "N";
+ }
+ break;
+ case ConfigInfo::STRING:
+ MGM_REQUIRE(prop->get(n, &str_value));
+ out << str_value;
+ break;
+ }
+ out << endl;
+ }
+}
+
+/*****************************************************************************/
+
+void Config::printConfigFile(NdbOut &out) const {
+ Uint32 noOfNodes, noOfConnections, noOfComputers;
+ MGM_REQUIRE(get("NoOfNodes", &noOfNodes));
+ MGM_REQUIRE(get("NoOfConnections", &noOfConnections));
+ MGM_REQUIRE(get("NoOfComputers", &noOfComputers));
+
+ out <<
+ "######################################################################" <<
+ endl <<
+ "#" << endl <<
+ "# NDB Cluster System configuration" << endl <<
+ "#" << endl <<
+ "######################################################################" <<
+ endl <<
+ "# No of nodes (DB, API or MGM): " << noOfNodes << endl <<
+ "# No of connections: " << noOfConnections << endl <<
+ "######################################################################" <<
+ endl;
+
+ /**************************
+ * Print COMPUTER configs *
+ **************************/
+ const char * name;
+ Properties::Iterator it(this);
+ for(name = it.first(); name != NULL; name = it.next()){
+ if(strncasecmp("Computer_", name, 9) == 0){
+
+ const Properties *prop;
+ out << endl << "[COMPUTER]" << endl;
+ MGM_REQUIRE(get(name, &prop));
+ printAllNameValuePairs(out, prop, "COMPUTER");
+
+ out << endl <<
+ "###################################################################" <<
+ endl;
+
+ } else if(strncasecmp("Node_", name, 5) == 0){
+ /**********************
+ * Print NODE configs *
+ **********************/
+ const Properties *prop;
+ const char *s;
+
+ MGM_REQUIRE(get(name, &prop));
+ MGM_REQUIRE(prop->get("Type", &s));
+ out << endl << "[" << s << "]" << endl;
+ printAllNameValuePairs(out, prop, s);
+
+ out << endl <<
+ "###################################################################" <<
+ endl;
+ } else if(strncasecmp("Connection_", name, 11) == 0){
+ /****************************
+ * Print CONNECTION configs *
+ ****************************/
+ const Properties *prop;
+ const char *s;
+
+ MGM_REQUIRE(get(name, &prop));
+ MGM_REQUIRE(prop->get("Type", &s));
+ out << endl << "[" << s << "]" << endl;
+ printAllNameValuePairs(out, prop, s);
+
+ out << endl <<
+ "###################################################################" <<
+ endl;
+ } else if(strncasecmp("SYSTEM", name, strlen("SYSTEM")) == 0) {
+ /************************
+ * Print SYSTEM configs *
+ ************************/
+ const Properties *prop;
+
+ MGM_REQUIRE(get(name, &prop));
+ out << endl << "[SYSTEM]" << endl;
+ printAllNameValuePairs(out, prop, "SYSTEM");
+
+ out << endl <<
+ "###################################################################" <<
+ endl;
+ }
+ }
+}
+
+const
+ConfigInfo* Config::getConfigInfo() const {
+ return m_info;
+}
+
+Uint32
+Config::getGenerationNumber() const {
+ Uint32 ret;
+ const Properties *prop = NULL;
+
+ get("SYSTEM", &prop);
+
+ if(prop != NULL)
+ if(prop->get("ConfigGenerationNumber", &ret))
+ return ret;
+
+ return 0;
+}
+
+int
+Config::setGenerationNumber(Uint32 gen) {
+ Properties *prop = NULL;
+
+ getCopy("SYSTEM", &prop);
+
+ if(prop != NULL) {
+ MGM_REQUIRE(prop->put("ConfigGenerationNumber", gen, true));
+ MGM_REQUIRE(put("SYSTEM", prop, true));
+ return 0;
+ }
+ return -1;
+}
+
+bool
+Config::change(const BaseString &section,
+ const BaseString &param,
+ const BaseString &value) {
+ const char *name;
+ Properties::Iterator it(this);
+
+ for(name = it.first(); name != NULL; name = it.next()) {
+ Properties *prop = NULL;
+ if(strcasecmp(section.c_str(), name) == 0) {
+ getCopy(name, &prop);
+ if(prop == NULL) /* doesn't exist */
+ return false;
+ if(value == "") {
+ prop->remove(param.c_str());
+ put(section.c_str(), prop, true);
+ } else {
+ PropertiesType t;
+ if(!prop->getTypeOf(param.c_str(), &t)) /* doesn't exist */
+ return false;
+ switch(t) {
+ case PropertiesType_Uint32:
+ long val;
+ char *ep;
+ errno = 0;
+ val = strtol(value.c_str(), &ep, 0);
+ if(value.length() == 0 || *ep != '\0') /* not a number */
+ return false;
+ if(errno == ERANGE)
+ return false;
+ prop->put(param.c_str(), (unsigned int)val, true);
+ put(section.c_str(), prop, true);
+ break;
+ case PropertiesType_char:
+ prop->put(param.c_str(), value.c_str(), true);
+ put(section.c_str(), prop, true);
+ break;
+ default:
+ return false;
+ }
+ }
+ break;
+ }
+ }
+ return true;
+}
diff --git a/ndb/src/common/mgmcommon/Config.hpp b/ndb/src/common/mgmcommon/Config.hpp
new file mode 100644
index 00000000000..1314abe004a
--- /dev/null
+++ b/ndb/src/common/mgmcommon/Config.hpp
@@ -0,0 +1,86 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef Config_H
+#define Config_H
+
+#include <signaldata/ConfigParamId.hpp>
+#include <LogLevel.hpp>
+
+#include <kernel_types.h>
+
+#include <NdbOut.hpp>
+#include <NdbStdio.h>
+#include <ndb_limits.h>
+#include <NdbConstant.hpp>
+#include <Properties.hpp>
+
+/**
+ * @class Config
+ * @brief Cluster Configuration (corresponds to initial configuration file)
+ *
+ * Contains all cluster configuration parameters.
+ *
+ * The information includes all configurable parameters for a NDB cluster:
+ * - DB, API and MGM nodes with all their properties,
+ * - Connections between nodes and computers the nodes will execute on.
+ *
+ * The following categories (sections) of configuration parameters exists:
+ * - COMPUTER, DB, MGM, API, TCP, SCI, SHM, OSE
+ */
+class Config : public Properties {
+public:
+ /**
+ * Constructor which loads the object with an Properties object
+ */
+ Config(const Config & org);
+ Config(const Properties & org);
+ Config();
+ virtual ~Config();
+
+ /**
+ * Prints the configuration in configfile format
+ */
+ void printConfigFile(NdbOut &out = ndbout) const;
+ void printConfigFile(OutputStream &out) const {
+ NdbOut ndb(out);
+ printConfigFile(ndb);
+ }
+
+ const class ConfigInfo* getConfigInfo() const;
+
+ Uint32 getGenerationNumber() const;
+ int setGenerationNumber(Uint32);
+
+ /** Change configuration
+ */
+ bool change(const BaseString &section,
+ const BaseString &param,
+ const BaseString &value);
+
+private:
+
+ void printAllNameValuePairs(NdbOut &out,
+ const Properties *prop,
+ const char* section) const;
+
+ /**
+ * Information about parameters (min, max values etc)
+ */
+ const class ConfigInfo* m_info;
+};
+
+#endif // Config_H
diff --git a/ndb/src/common/mgmcommon/ConfigInfo.cpp b/ndb/src/common/mgmcommon/ConfigInfo.cpp
new file mode 100644
index 00000000000..da6024d946f
--- /dev/null
+++ b/ndb/src/common/mgmcommon/ConfigInfo.cpp
@@ -0,0 +1,2629 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "ConfigInfo.hpp"
+#define MAX_LINE_LENGTH 255
+
+/****************************************************************************
+ * Section names
+ ****************************************************************************/
+const char*
+ConfigInfo::m_sectionNames[]={
+ "SYSTEM",
+ "EXTERNAL SYSTEM",
+ "COMPUTER",
+
+ "DB",
+ "MGM",
+ "API",
+ "REP",
+ "EXTERNAL REP",
+
+ "TCP",
+ "SCI",
+ "SHM",
+ "OSE"
+};
+const int ConfigInfo::m_noOfSectionNames =
+sizeof(m_sectionNames)/sizeof(char*);
+
+
+/****************************************************************************
+ * Section Rules declarations
+ ****************************************************************************/
+bool transformComputer(InitConfigFileParser::Context & ctx, const char *);
+bool transformSystem(InitConfigFileParser::Context & ctx, const char *);
+bool transformExternalSystem(InitConfigFileParser::Context & ctx, const char *);
+bool transformNode(InitConfigFileParser::Context & ctx, const char *);
+bool transformExtNode(InitConfigFileParser::Context & ctx, const char *);
+bool transformConnection(InitConfigFileParser::Context & ctx, const char *);
+bool applyDefaultValues(InitConfigFileParser::Context & ctx, const char *);
+bool checkMandatory(InitConfigFileParser::Context & ctx, const char *);
+bool fixPortNumber(InitConfigFileParser::Context & ctx, const char *);
+bool fixShmkey(InitConfigFileParser::Context & ctx, const char *);
+bool checkDbConstraints(InitConfigFileParser::Context & ctx, const char *);
+bool checkConnectionConstraints(InitConfigFileParser::Context &, const char *);
+bool fixHostname(InitConfigFileParser::Context & ctx, const char * data);
+bool fixNodeId(InitConfigFileParser::Context & ctx, const char * data);
+bool fixExtConnection(InitConfigFileParser::Context & ctx, const char * data);
+
+const ConfigInfo::SectionRule
+ConfigInfo::m_SectionRules[] = {
+ { "SYSTEM", transformSystem, 0 },
+ { "EXTERNAL SYSTEM", transformExternalSystem, 0 },
+ { "COMPUTER", transformComputer, 0 },
+
+ { "DB", transformNode, 0 },
+ { "API", transformNode, 0 },
+ { "MGM", transformNode, 0 },
+ { "REP", transformNode, 0 },
+ { "EXTERNAL REP", transformExtNode, 0 },
+
+ { "TCP", transformConnection, 0 },
+ { "SHM", transformConnection, 0 },
+ { "SCI", transformConnection, 0 },
+ { "OSE", transformConnection, 0 },
+
+ { "TCP", fixPortNumber, 0 },
+ //{ "SHM", fixShmKey, 0 },
+
+ { "COMPUTER", applyDefaultValues, 0 },
+
+ { "DB", applyDefaultValues, 0 },
+ { "API", applyDefaultValues, 0 },
+ { "MGM", applyDefaultValues, 0 },
+ { "REP", applyDefaultValues, 0 },
+ { "EXTERNAL REP", applyDefaultValues, 0 },
+
+ { "TCP", applyDefaultValues, 0 },
+ { "SHM", applyDefaultValues, 0 },
+ { "SCI", applyDefaultValues, 0 },
+ { "OSE", applyDefaultValues, 0 },
+
+ { "DB", checkDbConstraints, 0 },
+
+ { "TCP", fixNodeId, "NodeId1" },
+ { "TCP", fixNodeId, "NodeId2" },
+ { "SHM", fixNodeId, "NodeId1" },
+ { "SHM", fixNodeId, "NodeId2" },
+ { "SCI", fixNodeId, "NodeId1" },
+ { "SCI", fixNodeId, "NodeId2" },
+ { "OSE", fixNodeId, "NodeId1" },
+ { "OSE", fixNodeId, "NodeId2" },
+
+ /**
+ * fixExtConnection must be after fixNodeId
+ */
+ { "TCP", fixExtConnection, 0 },
+ { "SHM", fixExtConnection, 0 },
+ { "SCI", fixExtConnection, 0 },
+ { "OSE", fixExtConnection, 0 },
+
+ /**
+ * checkConnectionConstraints must be after fixExtConnection
+ */
+ { "TCP", checkConnectionConstraints, 0 },
+ { "SHM", checkConnectionConstraints, 0 },
+ { "SCI", checkConnectionConstraints, 0 },
+ { "OSE", checkConnectionConstraints, 0 },
+
+ { "COMPUTER", checkMandatory, 0 },
+
+ { "DB", checkMandatory, 0 },
+ { "API", checkMandatory, 0 },
+ { "MGM", checkMandatory, 0 },
+ { "REP", checkMandatory, 0 },
+
+ { "TCP", checkMandatory, 0 },
+ { "SHM", checkMandatory, 0 },
+ { "SCI", checkMandatory, 0 },
+ { "OSE", checkMandatory, 0 },
+
+ { "TCP", fixHostname, "HostName1" },
+ { "TCP", fixHostname, "HostName2" },
+ { "OSE", fixHostname, "HostName1" },
+ { "OSE", fixHostname, "HostName2" },
+};
+const int ConfigInfo::m_NoOfRules = sizeof(m_SectionRules)/sizeof(SectionRule);
+
+/**
+ * The default constructors create objects with suitable values for the
+ * configuration parameters.
+ *
+ * Some are however given the value MANDATORY which means that the value
+ * must be specified in the configuration file.
+ *
+ * Min and max values are also given for some parameters.
+ * - Attr1: Name in file (initial config file)
+ * - Attr2: Name in prop (properties object)
+ * - Attr3: Name of Section (in init config file)
+ * - Attr4: Updateable
+ * - Attr5: Type of parameter (INT or BOOL)
+ * - Attr6: Default Value (number only)
+ * - Attr7: Min value
+ * - Attr8: Max value
+ *
+ * Parameter constraints are coded in file Config.cpp.
+ *
+ * *******************************************************************
+ * Parameters used under development should be marked "NOTIMPLEMENTED"
+ * *******************************************************************
+ */
+const ConfigInfo::ParamInfo ConfigInfo::m_ParamInfo[] = {
+
+ /****************************************************************************
+ * COMPUTER
+ ****************************************************************************/
+
+ {"Id",
+ "Id",
+ "COMPUTER",
+ "Name of computer",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0},
+
+ {"HostName",
+ "HostName",
+ "COMPUTER",
+ "Hostname of computer (e.g. mysql.com)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ByteOrder",
+ "ByteOrder",
+ "COMPUTER",
+ "Not yet implemented",
+ ConfigInfo::USED, // Actually not used, but since it is MANDATORY,
+ // we don't want any warning message
+ false,
+ ConfigInfo::STRING,
+ MANDATORY, // Big == 0, Little == 1, NotSet == 2 (?)
+ 0,
+ 0x7FFFFFFF},
+
+ /****************************************************************************
+ * DB
+ ****************************************************************************/
+
+ {"Id",
+ "Id",
+ "DB",
+ "Number identifying the database node (DB)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 1,
+ (MAX_NODES - 1)},
+
+ {"Type",
+ "Type",
+ "DB",
+ "Type of node (Should have value DB)",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0},
+
+ {"NoOfReplicas",
+ "NoOfReplicas",
+ "DB",
+ "Number of copies of all data in the database (1-4)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 1,
+ 4},
+
+ {"MaxNoOfAttributes",
+ "MaxNoOfAttributes",
+ "DB",
+ "Total number of attributes stored in database. I.e. sum over all tables",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 1000,
+ 32,
+ 4096},
+
+ {"MaxNoOfTables",
+ "MaxNoOfTables",
+ "DB",
+ "Total number of tables stored in the database",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 32,
+ 8,
+ 128},
+
+ {"MaxNoOfIndexes",
+ "MaxNoOfIndexes",
+ "DB",
+ "Total number of indexes that can be defined in the system",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 128,
+ 0,
+ 2048},
+
+ {"MaxNoOfConcurrentIndexOperations",
+ "MaxNoOfConcurrentIndexOperations",
+ "DB",
+ "Total number of index operations that can execute simultaneously on one DB node",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 8192,
+ 0,
+ 1000000
+ },
+
+ {"MaxNoOfTriggers",
+ "MaxNoOfTriggers",
+ "DB",
+ "Total number of triggers that can be defined in the system",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 768,
+ 0,
+ 2432},
+
+ {"MaxNoOfFiredTriggers",
+ "MaxNoOfFiredTriggers",
+ "DB",
+ "Total number of triggers that can fire simultaneously in one DB node",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 1000,
+ 0,
+ 1000000},
+
+ {"ExecuteOnComputer",
+ "ExecuteOnComputer",
+ "DB",
+ "String referencing an earlier defined COMPUTER",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"MaxNoOfSavedMessages",
+ "MaxNoOfSavedMessages",
+ "DB",
+ "Max number of error messages in error log and max number of trace files",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 25,
+ 0,
+ 0x7FFFFFFF},
+
+ {"LockPagesInMainMemory",
+ "LockPagesInMainMemory",
+ "DB",
+ "If set to yes, then NDB Cluster data will not be swapped out to disk",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::BOOL,
+ false,
+ 0,
+ 0x7FFFFFFF},
+
+ {"SleepWhenIdle",
+ "SleepWhenIdle",
+ "DB",
+ "?",
+ ConfigInfo::DEPRICATED,
+ true,
+ ConfigInfo::BOOL,
+ true,
+ 0,
+ 0x7FFFFFFF},
+
+ {"NoOfSignalsToExecuteBetweenCommunicationInterfacePoll",
+ "NoOfSignalsToExecuteBetweenCommunicationInterfacePoll",
+ "DB",
+ "?",
+ ConfigInfo::DEPRICATED,
+ true,
+ ConfigInfo::INT,
+ 20,
+ 1,
+ 0x7FFFFFFF},
+
+ {"TimeBetweenWatchDogCheck",
+ "TimeBetweenWatchDogCheck",
+ "DB",
+ "Time between execution checks inside a database node",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 4000,
+ 70,
+ 0x7FFFFFFF},
+
+ {"StopOnError",
+ "StopOnError",
+ "DB",
+ "If set to N, the DB automatically restarts/recovers in case of node failure",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::BOOL,
+ true,
+ 0,
+ 0x7FFFFFFF},
+
+ { "RestartOnErrorInsert",
+ "RestartOnErrorInsert",
+ "DB",
+ "See src/kernel/vm/Emulator.hpp NdbRestartType for details",
+ ConfigInfo::INTERNAL,
+ true,
+ ConfigInfo::INT,
+ 2,
+ 0,
+ 4 },
+
+ {"MaxNoOfConcurrentOperations",
+ "MaxNoOfConcurrentOperations",
+ "DB",
+ "Max no of op:s on DB (op:s within a transaction are concurrently executed)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 8192,
+ 32,
+ 1000000},
+
+ {"MaxNoOfConcurrentTransactions",
+ "MaxNoOfConcurrentTransactions",
+ "DB",
+ "Max number of transaction executing concurrently on the DB node",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 4096,
+ 32,
+ 1000000},
+
+ {"MaxNoOfConcurrentScans",
+ "MaxNoOfConcurrentScans",
+ "DB",
+ "Max number of scans executing concurrently on the DB node",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 25,
+ 2,
+ 500},
+
+ {"TransactionBufferMemory",
+ "TransactionBufferMemory",
+ "DB",
+ "Dynamic buffer space (in bytes) for key and attribute data allocated for each DB node",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 1024000,
+ 1024,
+ 0x7FFFFFFF},
+
+ {"NoOfIndexPages",
+ "NoOfIndexPages",
+ "DB",
+ "Number of 8k byte pages on each DB node for storing indexes",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 3000,
+ 128,
+ 192000},
+
+ {"MemorySpaceIndexes",
+ "NoOfIndexPages",
+ "DB",
+ "Depricated",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 128,
+ 192000},
+
+ {"NoOfDataPages",
+ "NoOfDataPages",
+ "DB",
+ "Number of 8k byte pages on each DB node for storing data",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 10000,
+ 128,
+ 400000},
+
+ {"MemorySpaceTuples",
+ "NoOfDataPages",
+ "DB",
+ "Depricated",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 128,
+ 400000},
+
+ {"NoOfDiskBufferPages",
+ "NoOfDiskBufferPages",
+ "DB",
+ "?",
+ ConfigInfo::NOTIMPLEMENTED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 0},
+
+ {"MemoryDiskPages",
+ "NoOfDiskBufferPages",
+ "DB",
+ "?",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0},
+
+ {"NoOfFreeDiskClusters",
+ "NoOfFreeDiskClusters",
+ "DB",
+ "?",
+ ConfigInfo::NOTIMPLEMENTED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 0},
+
+ {"NoOfDiskClusters",
+ "NoOfDiskClusters",
+ "DB",
+ "?",
+ ConfigInfo::NOTIMPLEMENTED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ {"TimeToWaitAlive",
+ "TimeToWaitAlive",
+ "DB",
+ "Time to wait for other nodes to become alive during initial system start",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 25,
+ 2,
+ 4000},
+
+ {"HeartbeatIntervalDbDb",
+ "HeartbeatIntervalDbDb",
+ "DB",
+ "Time between DB-to-DB heartbeats. DB considered dead after 3 missed HBs",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 1500,
+ 10,
+ 0x7FFFFFFF},
+
+ {"HeartbeatIntervalDbApi",
+ "HeartbeatIntervalDbApi",
+ "DB",
+ "Time between API-to-DB heartbeats. API connection closed after 3 missed HBs",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 1500,
+ 100,
+ 0x7FFFFFFF},
+
+ {"TimeBetweenLocalCheckpoints",
+ "TimeBetweenLocalCheckpoints",
+ "DB",
+ "Time between taking snapshots of the database (expressed in 2log of bytes)",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 20,
+ 0,
+ 31},
+
+ {"TimeBetweenGlobalCheckpoints",
+ "TimeBetweenGlobalCheckpoints",
+ "DB",
+ "Time between doing group commit of transactions to disk",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 2000,
+ 10,
+ 32000},
+
+ {"NoOfFragmentLogFiles",
+ "NoOfFragmentLogFiles",
+ "DB",
+ "No of 16 Mbyte Redo log files in each of 4 file sets belonging to DB node",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 8,
+ 1,
+ 0x7FFFFFFF},
+
+ {"MaxNoOfOpenFiles",
+ "MaxNoOfOpenFiles",
+ "DB",
+ "Max number of files open per DB node.(One thread is created per file)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 40,
+ 20,
+ 256},
+
+ {"NoOfConcurrentCheckpointsDuringRestart",
+ "NoOfConcurrentCheckpointsDuringRestart",
+ "DB",
+ "?",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 1,
+ 1,
+ 4},
+
+ {"TimeBetweenInactiveTransactionAbortCheck",
+ "TimeBetweenInactiveTransactionAbortCheck",
+ "DB",
+ "Time between inactive transaction checks",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 1000,
+ 1000,
+ 0x7FFFFFFF},
+
+ {"TransactionInactiveTimeout",
+ "TransactionInactiveTimeout",
+ "DB",
+ "Time application can wait before executing another transaction part (ms).\n"
+ "This is the time the transaction coordinator waits for the application\n"
+ "to execute or send another part (query, statement) of the transaction.\n"
+ "If the application takes too long time, the transaction gets aborted.\n"
+ "Timeout set to 0 means that we don't timeout at all on application wait.",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 3000,
+ 0,
+ 0x7FFFFFFF},
+
+ {"TransactionDeadlockDetectionTimeout",
+ "TransactionDeadlockDetectionTimeout",
+ "DB",
+ "Time transaction can be executing in a DB node (ms).\n"
+ "This is the time the transaction coordinator waits for each database node\n"
+ "of the transaction to execute a request. If the database node takes too\n"
+ "long time, the transaction gets aborted.",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 3000,
+ 50,
+ 0x7FFFFFFF},
+
+ {"TransactionInactiveTimeBeforeAbort",
+ "TransactionInactiveTimeBeforeAbort",
+ "DB",
+ "Time a transaction can be inactive before getting aborted (ms)",
+ ConfigInfo::DEPRICATED,
+ true,
+ ConfigInfo::INT,
+ 3000,
+ 20,
+ 0x7FFFFFFF},
+
+ {"NoOfConcurrentProcessesHandleTakeover",
+ "NoOfConcurrentProcessesHandleTakeover",
+ "DB",
+ "?",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 1,
+ 1,
+ 15},
+
+ {"NoOfConcurrentCheckpointsAfterRestart",
+ "NoOfConcurrentCheckpointsAfterRestart",
+ "DB",
+ "?",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 1,
+ 1,
+ 4},
+
+ {"NoOfDiskPagesToDiskDuringRestartTUP",
+ "NoOfDiskPagesToDiskDuringRestartTUP",
+ "DB",
+ "?",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 50,
+ 1,
+ 0x7FFFFFFF},
+
+ {"NoOfDiskPagesToDiskAfterRestartTUP",
+ "NoOfDiskPagesToDiskAfterRestartTUP",
+ "DB",
+ "?",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 10,
+ 1,
+ 0x7FFFFFFF},
+
+ {"NoOfDiskPagesToDiskDuringRestartACC",
+ "NoOfDiskPagesToDiskDuringRestartACC",
+ "DB",
+ "?",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 25,
+ 1,
+ 0x7FFFFFFF},
+
+ {"NoOfDiskPagesToDiskAfterRestartACC",
+ "NoOfDiskPagesToDiskAfterRestartACC",
+ "DB",
+ "?",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 5,
+ 1,
+ 0x7FFFFFFF},
+
+ {"NoOfDiskClustersPerDiskFile",
+ "NoOfDiskClustersPerDiskFile",
+ "DB",
+ "?",
+ ConfigInfo::NOTIMPLEMENTED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ {"NoOfDiskFiles",
+ "NoOfDiskFiles",
+ "DB",
+ "?",
+ ConfigInfo::NOTIMPLEMENTED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ArbitrationTimeout",
+ "ArbitrationTimeout",
+ "DB",
+ "Max time (milliseconds) database partion waits for arbitration signal",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 1000,
+ 10,
+ 0x7FFFFFFF},
+
+ {"FileSystemPath",
+ "FileSystemPath",
+ "DB",
+ "Path to directory where the DB node stores its data (directory must exist)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"LogLevelStartup",
+ "LogLevelStartup",
+ "DB",
+ "Node startup info printed on stdout",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 1,
+ 0,
+ 15},
+
+ {"LogLevelShutdown",
+ "LogLevelShutdown",
+ "DB",
+ "Node shutdown info printed on stdout",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 15},
+
+ {"LogLevelStatistic",
+ "LogLevelStatistic",
+ "DB",
+ "Transaction, operation, transporter info printed on stdout",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 15},
+
+ {"LogLevelCheckpoint",
+ "LogLevelCheckpoint",
+ "DB",
+ "Local and Global checkpoint info printed on stdout",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 15},
+
+ {"LogLevelNodeRestart",
+ "LogLevelNodeRestart",
+ "DB",
+ "Node restart, node failure info printed on stdout",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 15},
+
+ {"LogLevelConnection",
+ "LogLevelConnection",
+ "DB",
+ "Node connect/disconnect info printed on stdout",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 15},
+
+ {"LogLevelError",
+ "LogLevelError",
+ "DB",
+ "Transporter, heartbeat errors printed on stdout",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 15},
+
+ {"LogLevelInfo",
+ "LogLevelInfo",
+ "DB",
+ "Heartbeat and log info printed on stdout",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 15},
+
+ /**
+ * Backup
+ */
+ { "ParallelBackups",
+ "ParallelBackups",
+ "DB",
+ "Maximum number of parallel backups",
+ ConfigInfo::NOTIMPLEMENTED,
+ false,
+ ConfigInfo::INT,
+ 1,
+ 1,
+ 1 },
+
+ { "BackupMemory",
+ "BackupMemory",
+ "DB",
+ "Total memory allocated for backups per node (in bytes)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ (2 * 1024 * 1024) + (2 * 1024 * 1024), // sum of BackupDataBufferSize and BackupLogBufferSize
+ 0,
+ 0x7FFFFFFF },
+
+ { "BackupDataBufferSize",
+ "BackupDataBufferSize",
+ "DB",
+ "Default size of databuffer for a backup (in bytes)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ (2 * 1024 * 1024), // remember to change BackupMemory
+ 0,
+ 0x7FFFFFFF },
+
+ { "BackupLogBufferSize",
+ "BackupLogBufferSize",
+ "DB",
+ "Default size of logbuffer for a backup (in bytes)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ (2 * 1024 * 1024), // remember to change BackupMemory
+ 0,
+ 0x7FFFFFFF },
+
+ { "BackupWriteSize",
+ "BackupWriteSize",
+ "DB",
+ "Default size of filesystem writes made by backup (in bytes)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 32768,
+ 0,
+ 0x7FFFFFFF },
+
+ /****************************************************************************
+ * REP
+ ****************************************************************************/
+
+ {"Id",
+ "Id",
+ "REP",
+ "Number identifying replication node (REP)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 1,
+ (MAX_NODES - 1)},
+
+ {"Type",
+ "Type",
+ "REP",
+ "Type of node (Should have value REP)",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0},
+
+ {"ExecuteOnComputer",
+ "ExecuteOnComputer",
+ "REP",
+ "String referencing an earlier defined COMPUTER",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ /****************************************************************************
+ * EXTERNAL REP
+ ****************************************************************************/
+
+ {"Id",
+ "Id",
+ "EXTERNAL REP",
+ "Number identifying external (i.e. in another NDB Cluster) replication node (REP)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 1,
+ (MAX_NODES - 1)},
+
+ {"Type",
+ "Type",
+ "EXTERNAL REP",
+ "Type of node (Should have value REP)",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0},
+
+ {"System",
+ "System",
+ "EXTERNAL REP",
+ "System name of system hosting node",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0},
+
+ {"HeartbeatIntervalRepRep",
+ "HeartbeatIntervalRepRep",
+ "EXTERNAL REP",
+ "Time between REP-REP heartbeats. Connection closed after 3 missed HBs",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 3000,
+ 100,
+ 0x7FFFFFFF},
+
+ /****************************************************************************
+ * API
+ ****************************************************************************/
+
+ {"Id",
+ "Id",
+ "API",
+ "Number identifying application node (API)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 1,
+ (MAX_NODES - 1)},
+
+ {"Type",
+ "Type",
+ "API",
+ "Type of node (Should have value API)",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0},
+
+ {"ExecuteOnComputer",
+ "ExecuteOnComputer",
+ "API",
+ "String referencing an earlier defined COMPUTER",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"MaxNoOfSavedMessages",
+ "MaxNoOfSavedMessages",
+ "API",
+ "Max number of error messages in error log and max number of trace files",
+ ConfigInfo::USED,
+ true,
+ ConfigInfo::INT,
+ 25,
+ 0,
+ 0x7FFFFFFF},
+
+ {"SleepWhenIdle",
+ "SleepWhenIdle",
+ "API",
+ "?",
+ ConfigInfo::DEPRICATED,
+ true,
+ ConfigInfo::BOOL,
+ true,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ArbitrationRank",
+ "ArbitrationRank",
+ "API",
+ "If 0, then API is not arbitrator. Kernel selects arbitrators in order 1, 2",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 2,
+ 0,
+ 2},
+
+ {"ArbitrationDelay",
+ "ArbitrationDelay",
+ "API",
+ "When asked to arbitrate, arbitrator waits this long before voting (msec)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ /****************************************************************************
+ * MGM
+ ****************************************************************************/
+
+ {"Id",
+ "Id",
+ "MGM",
+ "Number identifying the management server node (MGM)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 1,
+ (MAX_NODES - 1)},
+
+ {"Type",
+ "Type",
+ "MGM",
+ "Type of node (Should have value MGM)",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0},
+
+ {"ExecuteOnComputer",
+ "ExecuteOnComputer",
+ "MGM",
+ "String referencing an earlier defined COMPUTER",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ // SHOULD THIS REALLY BE DEFINABLE FOR MGM ???
+ {"MaxNoOfSavedMessages",
+ "MaxNoOfSavedMessages",
+ "MGM",
+ "Max number of error messages in error log and max number of trace files",
+ ConfigInfo::DEPRICATED,
+ true,
+ ConfigInfo::INT,
+ 25,
+ 0,
+ 0x7FFFFFFF},
+
+ {"MaxNoOfSavedEvents",
+ "MaxNoOfSavedEvents",
+ "MGM",
+ "",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 100,
+ 0,
+ 0x7FFFFFFF},
+
+ {"PortNumber",
+ "PortNumber",
+ "MGM",
+ "Port number to give commands to/fetch configurations from management server",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 2200,
+ 0,
+ 0x7FFFFFFF},
+
+ {"PortNumberStats",
+ "PortNumberStats",
+ "MGM",
+ "Port number used to get statistical information from a management server",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 2199,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ArbitrationRank",
+ "ArbitrationRank",
+ "MGM",
+ "If 0, then MGM is not arbitrator. Kernel selects arbitrators in order 1, 2",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 2,
+ 0,
+ 2},
+
+ {"ArbitrationDelay",
+ "ArbitrationDelay",
+ "MGM",
+ "",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ /*****************************************************************************
+ * SYSTEM
+ ****************************************************************************/
+
+ {"Name",
+ "Name",
+ "SYSTEM",
+ "Name of system (NDB Cluster)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0},
+
+ {"ReplicationRole",
+ "ReplicationRole",
+ "SYSTEM",
+ "Role in Global Replication (None, Primary, or Standby)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0},
+
+ {"LogDestination",
+ "LogDestination",
+ "MGM",
+ "String describing where logmessages are sent",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ {"PrimaryMGMNode",
+ "PrimaryMGMNode",
+ "SYSTEM",
+ "Node id of Primary MGM node",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ConfigGenerationNumber",
+ "ConfigGenerationNumber",
+ "SYSTEM",
+ "Configuration generation number",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Name",
+ "Name",
+ "EXTERNAL SYSTEM",
+ "Name of external system (another NDB Cluster)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0},
+
+ /*****************************************************************************
+ * TCP
+ ****************************************************************************/
+
+ {"Type",
+ "Type",
+ "TCP",
+ "",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ {"HostName1",
+ "HostName1",
+ "TCP",
+ "Name of computer on one side of the connection",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"HostName2",
+ "HostName2",
+ "TCP",
+ "Name of computer on one side of the connection",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"NodeId1",
+ "NodeId1",
+ "TCP",
+ "Id of node (DB, API or MGM) on one side of the connection",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ProcessId1",
+ "NodeId1",
+ "TCP",
+ "Depricated",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"NodeId2",
+ "NodeId2",
+ "TCP",
+ "Id of node (DB, API or MGM) on one side of the connection",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ProcessId2",
+ "NodeId2",
+ "TCP",
+ "Depricated",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"IpAddress1",
+ "HostName1",
+ "TCP",
+ "IP address of first node in connection.",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"IpAddress2",
+ "HostName2",
+ "TCP",
+ "IP address of second node in connection.",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0},
+
+ {"SendSignalId",
+ "SendSignalId",
+ "TCP",
+ "Sends id in each signal. Used in trace files.",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ true,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Compression",
+ "Compression",
+ "TCP",
+ "If compression is enabled, then all signals between nodes are compressed",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ false,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Checksum",
+ "Checksum",
+ "TCP",
+ "If checksum is enabled, all signals between nodes are checked for errors",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ false,
+ 0,
+ 0x7FFFFFFF},
+
+ {"PortNumber",
+ "PortNumber",
+ "TCP",
+ "Port used for this transporter",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"SendBufferSize",
+ "SendBufferSize",
+ "TCP",
+ "Size of buffer for signals sent from this node (in no of signals)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 16,
+ 1,
+ 0x7FFFFFFF},
+
+ {"MaxReceiveSize",
+ "MaxReceiveSize",
+ "TCP",
+ "Size of buffer for signals received by this node (in no of signals)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 4,
+ 1,
+ 0x7FFFFFFF},
+
+ {"Proxy",
+ "Proxy",
+ "TCP",
+ "",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0},
+
+ /*****************************************************************************
+ * SHM
+ ****************************************************************************/
+
+ {"Type",
+ "Type",
+ "SHM",
+ "",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ {"NodeId1",
+ "NodeId1",
+ "SHM",
+ "Id of node (DB, API or MGM) on one side of the connection",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ProcessId1",
+ "NodeId1",
+ "SHM",
+ "Depricated",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"NodeId2",
+ "NodeId2",
+ "SHM",
+ "Id of node (DB, API or MGM) on one side of the connection",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ProcessId2",
+ "NodeId2",
+ "SHM",
+ "Depricated",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"SendSignalId",
+ "SendSignalId",
+ "SHM",
+ "Sends id in each signal. Used in trace files.",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ false,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Compression",
+ "Compression",
+ "SHM",
+ "If compression is enabled, then all signals between nodes are compressed",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ false,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Checksum",
+ "Checksum",
+ "SHM",
+ "If checksum is enabled, all signals between nodes are checked for errors",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ true,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ShmKey",
+ "ShmKey",
+ "SHM",
+ "A shared memory key",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF },
+
+ {"ShmSize",
+ "ShmSize",
+ "SHM",
+ "Size of shared memory segment",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 1048576,
+ 4096,
+ 0x7FFFFFFF},
+
+ /*****************************************************************************
+ * SCI
+ ****************************************************************************/
+
+ {"NodeId1",
+ "NodeId1",
+ "SCI",
+ "Id of node (DB, API or MGM) on one side of the connection",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ProcessId1",
+ "NodeId1",
+ "SCI",
+ "Depricated",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"NodeId2",
+ "NodeId2",
+ "SCI",
+ "Id of node (DB, API or MGM) on one side of the connection",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ProcessId2",
+ "NodeId2",
+ "SCI",
+ "Depricated",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"SciId0",
+ "SciId0",
+ "SCI",
+ "Local SCI-node id for adapter 0 (a computer can have two adapters)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"SciId1",
+ "SciId1",
+ "SCI",
+ "Local SCI-node id for adapter 1 (a computer can have two adapters)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"SendSignalId",
+ "SendSignalId",
+ "SCI",
+ "Sends id in each signal. Used in trace files.",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ true,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Compression",
+ "Compression",
+ "SCI",
+ "If compression is enabled, then all signals between nodes are compressed",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ false,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Checksum",
+ "Checksum",
+ "SCI",
+ "If checksum is enabled, all signals between nodes are checked for errors",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ false,
+ 0,
+ 0x7FFFFFFF},
+
+ {"SendLimit",
+ "SendLimit",
+ "SCI",
+ "Transporter send buffer contents are sent when this no of bytes is buffered",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 2048,
+ 512,
+ 0x7FFFFFFF},
+
+ {"SharedBufferSize",
+ "SharedBufferSize",
+ "SCI",
+ "Size of shared memory segment",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 1048576,
+ 262144,
+ 0x7FFFFFFF},
+
+ {"Node1_NoOfAdapters",
+ "Node1_NoOfAdapters",
+ "SCI",
+ "",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Node2_NoOfAdapters",
+ "Node2_NoOfAdapters",
+ "SCI",
+ "",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Node1_Adapter",
+ "Node1_Adapter",
+ "SCI",
+ "",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Node2_Adapter",
+ "Node2_Adapter",
+ "SCI",
+ "",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ /*****************************************************************************
+ * OSE
+ ****************************************************************************/
+
+ {"Type",
+ "Type",
+ "OSE",
+ "",
+ ConfigInfo::INTERNAL,
+ false,
+ ConfigInfo::STRING,
+ 0,
+ 0,
+ 0x7FFFFFFF},
+
+ {"HostName1",
+ "HostName1",
+ "OSE",
+ "Name of computer on one side of the connection",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"HostName2",
+ "HostName2",
+ "OSE",
+ "Name of computer on one side of the connection",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::STRING,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"NodeId1",
+ "NodeId1",
+ "OSE",
+ "Id of node (DB, API or MGM) on one side of the connection",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ProcessId1",
+ "NodeId1",
+ "OSE",
+ "Depricated",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"NodeId2",
+ "NodeId2",
+ "OSE",
+ "Id of node (DB, API or MGM) on one side of the connection",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ProcessId2",
+ "NodeId2",
+ "OSE",
+ "Depricated",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ MANDATORY,
+ 0,
+ 0x7FFFFFFF},
+
+ {"SendSignalId",
+ "SendSignalId",
+ "OSE",
+ "Sends id in each signal. Used in trace files.",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ true,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Compression",
+ "Compression",
+ "OSE",
+ "If compression is enabled, then all signals between nodes are compressed",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ false,
+ 0,
+ 0x7FFFFFFF},
+
+ {"Checksum",
+ "Checksum",
+ "OSE",
+ "If checksum is enabled, all signals between nodes are checked for errors",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::BOOL,
+ false,
+ 0,
+ 0x7FFFFFFF},
+
+ // Should not be part of OSE ?
+ {"SharedBufferSize",
+ "SharedBufferSize",
+ "OSE",
+ "?",
+ ConfigInfo::DEPRICATED,
+ false,
+ ConfigInfo::INT,
+ UNDEFINED,
+ 2000,
+ 0x7FFFFFFF},
+
+ {"PrioASignalSize",
+ "PrioASignalSize",
+ "OSE",
+ "Size of priority A signals (in bytes)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 1000,
+ 0,
+ 0x7FFFFFFF},
+
+ {"PrioBSignalSize",
+ "PrioBSignalSize",
+ "OSE",
+ "Size of priority B signals (in bytes)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 1000,
+ 0,
+ 0x7FFFFFFF},
+
+ {"ReceiveArraySize",
+ "ReceiveArraySize",
+ "OSE",
+ "Number of OSE signals checked for correct ordering (in no of OSE signals)",
+ ConfigInfo::USED,
+ false,
+ ConfigInfo::INT,
+ 10,
+ 0,
+ 0x7FFFFFFF}
+};
+
+const int ConfigInfo::m_NoOfParams = sizeof(m_ParamInfo) / sizeof(ParamInfo);
+
+
+/****************************************************************************
+ * Ctor
+ ****************************************************************************/
+inline void require(bool v) { if(!v) abort();}
+
+ConfigInfo::ConfigInfo() {
+ Properties *section;
+ const Properties *oldpinfo;
+
+ m_info.setCaseInsensitiveNames(true);
+ m_systemDefaults.setCaseInsensitiveNames(true);
+
+ for (int i=0; i<m_NoOfParams; i++) {
+ const ParamInfo & param = m_ParamInfo[i];
+
+ // Create new section if it did not exist
+ if (!m_info.getCopy(param._section, &section)) {
+ Properties newsection;
+ newsection.setCaseInsensitiveNames(true);
+ m_info.put(param._section, &newsection);
+ }
+
+ // Get copy of section
+ m_info.getCopy(param._section, &section);
+
+ // Create pinfo (parameter info) entry
+ Properties pinfo;
+ pinfo.put("Fname", param._fname);
+ pinfo.put("Pname", param._pname);
+ pinfo.put("Description", param._description);
+ pinfo.put("Updateable", param._updateable);
+ pinfo.put("Type", param._type);
+ pinfo.put("Status", param._status);
+ pinfo.put("Default", param._default);
+ pinfo.put("Min", param._min);
+ pinfo.put("Max", param._max);
+
+ // Check that pinfo is really new
+ if (section->get(param._fname, &oldpinfo)) {
+ ndbout << "Error: Parameter " << param._fname
+ << " defined twice in section " << param._section
+ << "." << endl;
+ exit(-1);
+ }
+
+ // Add new pinfo to section
+ section->put(param._fname, &pinfo);
+
+ // Replace section with modified section
+ m_info.put(param._section, section, true);
+
+ {
+ Properties * p;
+ if(!m_systemDefaults.getCopy(param._section, &p)){
+ p = new Properties();
+ p->setCaseInsensitiveNames(true);
+ }
+ if(param._type != STRING &&
+ param._default != UNDEFINED &&
+ param._default != MANDATORY){
+ require(p->put(param._pname, param._default));
+ }
+ require(m_systemDefaults.put(param._section, p, true));
+ delete p;
+ }
+ }
+
+ for (int i=0; i<m_NoOfParams; i++) {
+ if(m_ParamInfo[i]._section == NULL){
+ ndbout << "Check that each pname has an fname failed." << endl;
+ ndbout << "Parameter \"" << m_ParamInfo[i]._pname
+ << "\" does not exist in section \""
+ << m_ParamInfo[i]._section << "\"." << endl;
+ ndbout << "Edit file " << __FILE__ << "." << endl;
+ exit(-1);
+ }
+ const Properties * p = getInfo(m_ParamInfo[i]._section);
+ if (!p || !p->contains(m_ParamInfo[i]._pname)) {
+ ndbout << "Check that each pname has an fname failed." << endl;
+ ndbout << "Parameter \"" << m_ParamInfo[i]._pname
+ << "\" does not exist in section \""
+ << m_ParamInfo[i]._section << "\"." << endl;
+ ndbout << "Edit file " << __FILE__ << "." << endl;
+ exit(-1);
+ }
+ }
+}
+
+/****************************************************************************
+ * Getters
+ ****************************************************************************/
+inline void warning(const char * src, const char * arg){
+ ndbout << "Illegal call to ConfigInfo::" << src << "() - " << arg << endl;
+ abort();
+}
+
+const Properties *
+ConfigInfo::getInfo(const char * section) const {
+ const Properties * p;
+ if(!m_info.get(section, &p)){
+ warning("getInfo", section);
+ }
+ return p;
+}
+
+const Properties *
+ConfigInfo::getDefaults(const char * section) const {
+ const Properties * p;
+ if(!m_systemDefaults.get(section, &p)){
+ warning("getDefaults", section);
+ }
+ return p;
+}
+
+static
+Uint32
+getInfoInt(const Properties * section,
+ const char* fname, const char * type){
+ Uint32 val;
+ const Properties * p;
+ if (section->get(fname, &p) && p->get(type, &val)) {
+ return val;
+ }
+ warning(type, fname);
+ return val;
+}
+
+static
+const char *
+getInfoString(const Properties * section,
+ const char* fname, const char * type){
+ const char* val;
+ const Properties * p;
+ if (section->get(fname, &p) && p->get(type, &val)) {
+ return val;
+ }
+ warning(type, fname);
+ return val;
+}
+
+Uint32
+ConfigInfo::getMax(const Properties * section, const char* fname) const {
+ return getInfoInt(section, fname, "Max");
+}
+
+Uint32
+ConfigInfo::getMin(const Properties * section, const char* fname) const {
+ return getInfoInt(section, fname, "Min");
+}
+
+Uint32
+ConfigInfo::getDefault(const Properties * section, const char* fname) const {
+ return getInfoInt(section, fname, "Default");
+}
+
+const char*
+ConfigInfo::getPName(const Properties * section, const char* fname) const {
+ return getInfoString(section, fname, "Pname");
+}
+
+const char*
+ConfigInfo::getDescription(const Properties * section,
+ const char* fname) const {
+ return getInfoString(section, fname, "Description");
+}
+
+bool
+ConfigInfo::isSection(const char * section) const {
+ for (int i = 0; i<m_noOfSectionNames; i++) {
+ if(!strcmp(section, m_sectionNames[i])) return true;
+ }
+ return false;
+}
+
+bool
+ConfigInfo::verify(const Properties * section, const char* fname,
+ Uint32 value) const {
+ Uint32 min, max; min = max + 1;
+
+ min = getInfoInt(section, fname, "Min");
+ max = getInfoInt(section, fname, "Max");
+ if(min > max){
+ warning("verify", fname);
+ }
+ if (value >= min && value <= max)
+ return true;
+ else
+ return false;
+}
+
+ConfigInfo::Type
+ConfigInfo::getType(const Properties * section, const char* fname) const {
+ return (ConfigInfo::Type) getInfoInt(section, fname, "Type");
+}
+
+ConfigInfo::Status
+ConfigInfo::getStatus(const Properties * section, const char* fname) const {
+ return (ConfigInfo::Status) getInfoInt(section, fname, "Status");
+}
+
+/****************************************************************************
+ * Printers
+ ****************************************************************************/
+
+void ConfigInfo::print() const {
+ Properties::Iterator it(&m_info);
+ for (const char* n = it.first(); n != NULL; n = it.next()) {
+ print(n);
+ }
+}
+
+void ConfigInfo::print(const char* section) const {
+ ndbout << "****** " << section << " ******" << endl << endl;
+ const Properties * sec = getInfo(section);
+ Properties::Iterator it(sec);
+ for (const char* n = it.first(); n != NULL; n = it.next()) {
+ // Skip entries with different F- and P-names
+ if (strcmp(n, getPName(sec, n))) continue;
+ if (getStatus(sec, n) == ConfigInfo::INTERNAL) continue;
+ if (getStatus(sec, n) == ConfigInfo::DEPRICATED) continue;
+ if (getStatus(sec, n) == ConfigInfo::NOTIMPLEMENTED) continue;
+ print(sec, n);
+ }
+}
+
+void ConfigInfo::print(const Properties * section,
+ const char* parameter) const {
+ ndbout << getPName(section, parameter);
+ // ndbout << getDescription(section, parameter) << endl;
+ switch (getType(section, parameter)) {
+ case ConfigInfo::BOOL:
+ ndbout << " (Boolean value)" << endl;
+ ndbout << getDescription(section, parameter) << endl;
+ if (getDefault(section, parameter) == false) {
+ ndbout << "Default: N (Legal values: Y, N)" << endl;
+ } else if (getDefault(section, parameter) == true) {
+ ndbout << "Default: Y (Legal values: Y, N)" << endl;
+ } else if (getDefault(section, parameter) == MANDATORY) {
+ ndbout << "MANDATORY (Legal values: Y, N)" << endl;
+ } else {
+ ndbout << "UNKNOWN" << endl;
+ }
+ ndbout << endl;
+ break;
+
+ case ConfigInfo::INT:
+ ndbout << " (Non-negative Integer)" << endl;
+ ndbout << getDescription(section, parameter) << endl;
+ if (getDefault(section, parameter) == MANDATORY) {
+ ndbout << "MANDATORY (";
+ } else if (getDefault(section, parameter) == UNDEFINED) {
+ ndbout << "UNDEFINED (";
+ } else {
+ ndbout << "Default: " << getDefault(section, parameter) << " (";
+ }
+ ndbout << "Min: " << getMin(section, parameter) << ", ";
+ ndbout << "Max: " << getMax(section, parameter) << ")" << endl;
+ ndbout << endl;
+ break;
+
+ case ConfigInfo::STRING:
+ ndbout << " (String)" << endl;
+ ndbout << getDescription(section, parameter) << endl;
+ if (getDefault(section, parameter) == MANDATORY) {
+ ndbout << "MANDATORY" << endl;
+ } else {
+ ndbout << "No default value" << endl;
+ }
+ ndbout << endl;
+ break;
+ }
+}
+
+/****************************************************************************
+ * Section Rules
+ ****************************************************************************/
+
+/**
+ * Node rule: Add "Type" and update "NoOfNodes"
+ */
+bool
+transformNode(InitConfigFileParser::Context & ctx, const char * data){
+
+ Uint32 id;
+ if(!ctx.m_currentSection->get("Id", &id)){
+ ctx.reportError("Mandatory parameter Id missing from section "
+ "[%s] starting at line: %d",
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+ snprintf(ctx.pname, sizeof(ctx.pname), "Node_%d", id);
+
+ ctx.m_currentSection->put("Type", ctx.fname);
+
+ Uint32 nodes = 0;
+ ctx.m_userProperties.get("NoOfNodes", &nodes);
+ ctx.m_userProperties.put("NoOfNodes", ++nodes, true);
+
+ return true;
+}
+
+bool
+transformExtNode(InitConfigFileParser::Context & ctx, const char * data){
+
+ Uint32 id;
+ const char * systemName;
+
+ if(!ctx.m_currentSection->get("Id", &id)){
+ ctx.reportError("Mandatory parameter 'Id' missing from section "
+ "[%s] starting at line: %d",
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+
+ if(!ctx.m_currentSection->get("System", &systemName)){
+ ctx.reportError("Mandatory parameter 'System' missing from section "
+ "[%s] starting at line: %d",
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+
+ ctx.m_currentSection->put("Type", ctx.fname);
+
+ Uint32 nodes = 0;
+ ctx.m_userProperties.get("ExtNoOfNodes", &nodes);
+ require(ctx.m_userProperties.put("ExtNoOfNodes",++nodes, true));
+
+ snprintf(ctx.pname, sizeof(ctx.pname), "EXTERNAL SYSTEM_%s:Node_%d",
+ systemName, id);
+
+ return true;
+}
+
+/**
+ * Connection rule: Update "NoOfConnections"
+ */
+bool
+transformConnection(InitConfigFileParser::Context & ctx, const char * data){
+
+ Uint32 connections = 0;
+ ctx.m_userProperties.get("NoOfConnections", &connections);
+ snprintf(ctx.pname, sizeof(ctx.pname), "Connection_%d", connections);
+ ctx.m_userProperties.put("NoOfConnections", ++connections, true);
+
+ ctx.m_currentSection->put("Type", ctx.fname);
+ return true;
+}
+
+/**
+ * System rule: Just add it
+ */
+bool
+transformSystem(InitConfigFileParser::Context & ctx, const char * data){
+
+ const char * name;
+ if(!ctx.m_currentSection->get("Name", &name)){
+ ctx.reportError("Mandatory parameter Name missing from section "
+ "[%s] starting at line: %d",
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+ snprintf(ctx.pname, sizeof(ctx.pname), "SYSTEM_%s", name);
+
+ return true;
+}
+
+/**
+ * External system rule: Just add it
+ */
+bool
+transformExternalSystem(InitConfigFileParser::Context & ctx, const char * data){
+ const char * name;
+ if(!ctx.m_currentSection->get("Name", &name)){
+ ctx.reportError("Mandatory parameter Name missing from section "
+ "[%s] starting at line: %d",
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+ snprintf(ctx.pname, sizeof(ctx.pname), "EXTERNAL SYSTEM_%s", name);
+
+ return true;
+}
+
+/**
+ * Computer rule: Update "NoOfComputers", add "Type"
+ */
+bool
+transformComputer(InitConfigFileParser::Context & ctx, const char * data){
+ const char * id;
+ if(!ctx.m_currentSection->get("Id", &id)){
+ ctx.reportError("Mandatory parameter Id missing from section "
+ "[%s] starting at line: %d",
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+ snprintf(ctx.pname, sizeof(ctx.pname), "Computer_%s", id);
+
+ Uint32 computers = 0;
+ ctx.m_userProperties.get("NoOfComputers", &computers);
+ ctx.m_userProperties.put("NoOfComputers", ++computers, true);
+
+ return true;
+}
+
+/**
+ * Apply default values
+ */
+void
+applyDefaultValues(InitConfigFileParser::Context & ctx,
+ const Properties * defaults){
+ if(defaults != NULL){
+ Properties::Iterator it(defaults);
+
+ for(const char * name = it.first(); name != NULL; name = it.next()){
+ if(!ctx.m_currentSection->contains(name)){
+ switch (ctx.m_info->getType(ctx.m_currentInfo, name)){
+ case ConfigInfo::INT:
+ case ConfigInfo::BOOL:{
+ Uint32 val = 0;
+ ::require(defaults->get(name, &val));
+ ctx.m_currentSection->put(name, val);
+ break;
+ }
+ case ConfigInfo::STRING:{
+ const char * val;
+ ::require(defaults->get(name, &val));
+ ctx.m_currentSection->put(name, val);
+ break;
+ }
+ }
+ }
+ }
+ }
+}
+
+bool
+applyDefaultValues(InitConfigFileParser::Context & ctx, const char * data){
+
+ applyDefaultValues(ctx, ctx.m_userDefaults);
+ applyDefaultValues(ctx, ctx.m_systemDefaults);
+
+ return true;
+}
+
+/**
+ * Check that a section contains all MANDATORY parameters
+ */
+bool
+checkMandatory(InitConfigFileParser::Context & ctx, const char * data){
+
+ Properties::Iterator it(ctx.m_currentInfo);
+ for(const char * name = it.first(); name != NULL; name = it.next()){
+ const Properties * info = NULL;
+ ::require(ctx.m_currentInfo->get(name, &info));
+ Uint32 val;
+ if(info->get("Default", &val) && val == MANDATORY){
+ const char * pname;
+ const char * fname;
+ ::require(info->get("Pname", &pname));
+ ::require(info->get("Fname", &fname));
+ if(!ctx.m_currentSection->contains(pname)){
+ ctx.reportError("Mandatory parameter %s missing from section "
+ "[%s] starting at line: %d",
+ fname, ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+ }
+ }
+ return true;
+}
+
+/**
+ * Connection rule: Fix node id
+ *
+ * Transform a string "NodeidX" (e.g. "uppsala.32")
+ * into a Uint32 "NodeIdX" (e.g. 32) and a string "SystemX" (e.g. "uppsala").
+ */
+bool fixNodeId(InitConfigFileParser::Context & ctx, const char * data){
+
+ char buf[] = "NodeIdX"; buf[6] = data[sizeof("NodeI")];
+ char sysbuf[] = "SystemX"; sysbuf[6] = data[sizeof("NodeI")];
+ const char* nodeId;
+ require(ctx.m_currentSection->get(buf, &nodeId));
+
+ char tmpLine[MAX_LINE_LENGTH];
+ strncpy(tmpLine, nodeId, MAX_LINE_LENGTH);
+ char* token1 = strtok(tmpLine, ".");
+ char* token2 = strtok(NULL, ".");
+ Uint32 id;
+
+ if (token2 == NULL) { // Only a number given
+ errno = 0;
+ char* p;
+ id = strtol(token1, &p, 10);
+ if (errno != 0) warning("STRTOK1", nodeId);
+ require(ctx.m_currentSection->put(buf, id, true));
+ } else { // A pair given (e.g. "uppsala.32")
+ errno = 0;
+ char* p;
+ id = strtol(token2, &p, 10);
+ if (errno != 0) warning("STRTOK2", nodeId);
+ require(ctx.m_currentSection->put(buf, id, true));
+ require(ctx.m_currentSection->put(sysbuf, token1));
+ }
+
+ return true;
+}
+
+/**
+ * @returns true if connection is external (one node is external)
+ * Also returns:
+ * - name of external system in parameter extSystemName, and
+ * - nodeId of external node in parameter extSystemNodeId.
+ */
+bool
+isExtConnection(InitConfigFileParser::Context & ctx,
+ const char **extSystemName, Uint32 * extSystemNodeId){
+
+ Uint32 nodeId1, nodeId2;
+
+ if (ctx.m_currentSection->contains("System1") &&
+ ctx.m_currentSection->get("System1", extSystemName) &&
+ ctx.m_currentSection->get("NodeId1", &nodeId1)) {
+ *extSystemNodeId = nodeId1;
+ return true;
+ }
+
+ if (ctx.m_currentSection->contains("System2") &&
+ ctx.m_currentSection->get("System2", extSystemName) &&
+ ctx.m_currentSection->get("NodeId2", &nodeId2)) {
+ *extSystemNodeId = nodeId2;
+ return true;
+ }
+
+ return false;
+}
+
+/**
+ * External Connection Rule:
+ * If connection is to an external system, then move connection into
+ * external system configuration (i.e. a sub-property).
+ */
+bool
+fixExtConnection(InitConfigFileParser::Context & ctx, const char * data){
+
+ const char * extSystemName;
+ Uint32 extSystemNodeId;
+
+ if (isExtConnection(ctx, &extSystemName, &extSystemNodeId)) {
+
+ Uint32 connections = 0;
+ ctx.m_userProperties.get("ExtNoOfConnections", &connections);
+ require(ctx.m_userProperties.put("ExtNoOfConnections",++connections, true));
+
+ char tmpLine1[MAX_LINE_LENGTH];
+ snprintf(tmpLine1, MAX_LINE_LENGTH, "Connection_%d", connections-1);
+
+ /**
+ * Section: EXTERNAL SYSTEM_<Ext System Name>
+ */
+ char extSystemPropName[MAX_LINE_LENGTH];
+ strncpy(extSystemPropName, "EXTERNAL SYSTEM_", MAX_LINE_LENGTH);
+ strncat(extSystemPropName, extSystemName, MAX_LINE_LENGTH);
+ strncat(extSystemPropName, ":", MAX_LINE_LENGTH);
+ strncat(extSystemPropName, tmpLine1, MAX_LINE_LENGTH);
+
+ /**
+ * Increase number of external connections for the system
+ *
+ * @todo Limitation: Only one external system is allowed
+ */
+ require(ctx.m_userProperties.put("ExtSystem", extSystemName, true));
+
+ /**
+ * Make sure section is stored in right place
+ */
+ strncpy(ctx.pname, extSystemPropName, MAX_LINE_LENGTH);
+
+ /**
+ * Since this is an external connection,
+ * decrease number of internal connections
+ */
+ require(ctx.m_userProperties.get("NoOfConnections", &connections));
+ require(ctx.m_userProperties.put("NoOfConnections", --connections, true));
+ }
+
+ return true;
+}
+
+/**
+ * Connection rule: Fix hostname
+ *
+ * Unless Hostname is not already specified, do steps:
+ * -# Via Connection's NodeId lookup Node
+ * -# Via Node's ExecuteOnComputer lookup Hostname
+ * -# Add HostName to Connection
+ */
+bool
+fixHostname(InitConfigFileParser::Context & ctx, const char * data){
+
+ char buf[] = "NodeIdX"; buf[6] = data[sizeof("HostNam")];
+ char sysbuf[] = "SystemX"; sysbuf[6] = data[sizeof("HostNam")];
+
+ if(!ctx.m_currentSection->contains(data)){
+ Uint32 id = 0;
+ require(ctx.m_currentSection->get(buf, &id));
+
+ const Properties * node;
+ require(ctx.m_config->get("Node", id, &node));
+
+ const char * compId;
+ require(node->get("ExecuteOnComputer", &compId));
+
+ const Properties * computer;
+ char tmp[255];
+ snprintf(tmp, sizeof(tmp), "Computer_%s", compId);
+ require(ctx.m_config->get(tmp, &computer));
+
+ const char * hostname;
+ require(computer->get("HostName", &hostname));
+ require(ctx.m_currentSection->put(data, hostname));
+ }
+ return true;
+}
+
+/**
+ * Connection rule: Fix port number (using a port number adder)
+ */
+bool
+fixPortNumber(InitConfigFileParser::Context & ctx, const char * data){
+
+ if(!ctx.m_currentSection->contains("PortNumber")){
+ Uint32 adder = 0;
+ ctx.m_userProperties.get("PortNumberAdder", &adder);
+ Uint32 base = 0;
+ if(!ctx.m_userDefaults->get("PortNumber", &base) &&
+ !ctx.m_systemDefaults->get("PortNumber", &base)){
+ return true;
+ }
+ ctx.m_currentSection->put("PortNumber", base + adder);
+ adder++;
+ ctx.m_userProperties.put("PortNumberAdder", adder, true);
+ }
+ return true;
+}
+
+/**
+ * DB Node rule: Check various constraints
+ */
+bool
+checkDbConstraints(InitConfigFileParser::Context & ctx, const char *){
+
+ Uint32 t1 = 0, t2 = 0;
+ ctx.m_currentSection->get("MaxNoOfConcurrentOperations", &t1);
+ ctx.m_currentSection->get("MaxNoOfConcurrentTransactions", &t2);
+
+ if (t1 < t2) {
+ ctx.reportError("MaxNoOfConcurrentOperations must be greater than "
+ "MaxNoOfConcurrentTransactions - [%s] starting at line: %d",
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+
+ Uint32 replicas = 0, otherReplicas;
+ ctx.m_currentSection->get("NoOfReplicas", &replicas);
+ if(ctx.m_userProperties.get("NoOfReplicas", &otherReplicas)){
+ if(replicas != otherReplicas){
+ ctx.reportError("NoOfReplicas defined differently on different nodes"
+ " - [%s] starting at line: %d",
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+ } else {
+ ctx.m_userProperties.put("NoOfReplicas", replicas);
+ }
+
+ return true;
+}
+
+/**
+ * Connection rule: Check varius constraints
+ */
+bool
+checkConnectionConstraints(InitConfigFileParser::Context & ctx, const char *){
+
+ Uint32 id1 = 0, id2 = 0;
+ ctx.m_currentSection->get("NodeId1", &id1);
+ ctx.m_currentSection->get("NodeId2", &id2);
+
+ // If external connection, just accept it
+ if (ctx.m_currentSection->contains("System1") ||
+ ctx.m_currentSection->contains("System2"))
+ return true;
+
+ if(id1 == id2){
+ ctx.reportError("Illegal connection from node to itself"
+ " - [%s] starting at line: %d",
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+
+ const Properties * node1;
+ if(!ctx.m_config->get("Node", id1, &node1)){
+ ctx.reportError("Connection refering to undefined node: %d"
+ " - [%s] starting at line: %d",
+ id1, ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+
+ const Properties * node2;
+ if(!ctx.m_config->get("Node", id2, &node2)){
+ ctx.reportError("Connection refering to undefined node: %d"
+ " - [%s] starting at line: %d",
+ id2, ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+
+ const char * type1;
+ const char * type2;
+ require(node1->get("Type", &type1));
+ require(node2->get("Type", &type2));
+
+ /**
+ * Report error if the following are true
+ * -# None of the nodes is of type DB
+ * -# Not both of them are MGMs
+ * -# None of them contain a "SystemX" name
+ */
+ if((strcmp(type1, "DB") != 0 && strcmp(type2, "DB") != 0) &&
+ !(strcmp(type1, "MGM") == 0 && strcmp(type2, "MGM") == 0) &&
+ !ctx.m_currentSection->contains("System1") &&
+ !ctx.m_currentSection->contains("System2")){
+ ctx.reportError("Invalid connection between node %d (%s) and node %d (%s)"
+ " - [%s] starting at line: %d",
+ id1, type1, id2, type2,
+ ctx.fname, ctx.m_sectionLineno);
+ return false;
+ }
+ return true;
+}
diff --git a/ndb/src/common/mgmcommon/ConfigInfo.hpp b/ndb/src/common/mgmcommon/ConfigInfo.hpp
new file mode 100644
index 00000000000..43041a3f772
--- /dev/null
+++ b/ndb/src/common/mgmcommon/ConfigInfo.hpp
@@ -0,0 +1,120 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ConfigInfo_H
+#define ConfigInfo_H
+
+#include <kernel_types.h>
+#include <Properties.hpp>
+#include <ndb_limits.h>
+#include <NdbOut.hpp>
+#include "InitConfigFileParser.hpp"
+
+/**
+ * A MANDATORY parameters must be specified in the config file
+ * An UNDEFINED parameter may or may not be specified in the config file
+ */
+static const Uint32 MANDATORY = ~0; // Default value for mandatory params.
+static const Uint32 UNDEFINED = (~0)-1; // Default value for undefined params.
+
+/**
+ * @class ConfigInfo
+ * @brief Metainformation about ALL cluster configuration parameters
+ *
+ * Use the getters to find out metainformation about parameters.
+ */
+class ConfigInfo {
+public:
+ enum Type {BOOL, INT, STRING};
+ enum Status {USED, ///< Active
+ DEPRICATED, ///< Can be, but should not be used anymore
+ NOTIMPLEMENTED, ///< Can not be used currently. Is ignored.
+ INTERNAL ///< Not configurable by the user
+ };
+
+ /**
+ * Entry for one configuration parameter
+ */
+ struct ParamInfo {
+ const char* _fname;
+ const char* _pname;
+ const char* _section;
+ const char* _description;
+ Status _status;
+ bool _updateable;
+ Type _type;
+ Uint32 _default;
+ Uint32 _min;
+ Uint32 _max;
+ };
+
+ /**
+ * Entry for one section rule
+ */
+ struct SectionRule {
+ const char * m_section;
+ bool (* m_sectionRule)(struct InitConfigFileParser::Context &,
+ const char * m_ruleData);
+ const char * m_ruleData;
+ };
+
+ ConfigInfo();
+
+ /**
+ * Checks if the suggested value is valid for the suggested parameter
+ * (i.e. if it is >= than min and <= than max).
+ *
+ * @param section Init Config file section name
+ * @param fname Name of parameter
+ * @param value Value to check
+ * @return true if parameter value is valid.
+ *
+ * @note Result is not defined if section/name are wrong!
+ */
+ bool verify(const Properties* section, const char* fname, Uint32 value) const;
+ bool isSection(const char*) const;
+
+ const char* getPName(const Properties * section, const char* fname) const;
+ const char* getDescription(const Properties * section, const char* fname) const;
+ Type getType(const Properties * section, const char* fname) const;
+ Status getStatus(const Properties* section, const char* fname) const;
+ Uint32 getMin(const Properties * section, const char* fname) const;
+ Uint32 getMax(const Properties * section, const char* fname) const;
+ Uint32 getDefault(const Properties * section, const char* fname) const;
+
+ const Properties * getInfo(const char * section) const;
+ const Properties * getDefaults(const char * section) const;
+
+ void print() const;
+ void print(const char* section) const;
+ void print(const Properties * section, const char* parameter) const;
+
+private:
+ Properties m_info;
+ Properties m_systemDefaults;
+
+ static const ParamInfo m_ParamInfo[];
+ static const int m_NoOfParams;
+
+ static const char* m_sectionNames[];
+ static const int m_noOfSectionNames;
+
+public:
+ static const SectionRule m_SectionRules[];
+ static const int m_NoOfRules;
+};
+
+#endif // ConfigInfo_H
diff --git a/ndb/src/common/mgmcommon/ConfigRetriever.cpp b/ndb/src/common/mgmcommon/ConfigRetriever.cpp
new file mode 100644
index 00000000000..a3f26454df6
--- /dev/null
+++ b/ndb/src/common/mgmcommon/ConfigRetriever.cpp
@@ -0,0 +1,514 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <ConfigRetriever.hpp>
+
+#include "LocalConfig.hpp"
+#include <NdbSleep.h>
+#include <NdbOut.hpp>
+
+#include <NdbTCP.h>
+#include <string.h>
+#include <NdbStdio.h>
+#include <NdbEnv.h>
+#include "MgmtErrorReporter.hpp"
+
+#include <uucode.h>
+#include <Properties.hpp>
+
+#include <stdio.h>
+#include <NdbString.h>
+#include <sys/stat.h>
+#include <socket_io.h>
+#include <NdbConfig.h>
+
+#include <ndb_version.h>
+
+//****************************************************************************
+//****************************************************************************
+
+ConfigRetriever::ConfigRetriever() {
+
+ _localConfigFileName = NULL;
+ m_defaultConnectString = NULL;
+
+
+ errorString = 0;
+ _localConfig = new LocalConfig();
+ m_connectString = NULL;
+}
+
+ConfigRetriever::~ConfigRetriever(){
+ if(_localConfigFileName != 0)
+ free(_localConfigFileName);
+
+ if(m_defaultConnectString != 0)
+ free(m_defaultConnectString);
+
+ if(m_connectString != 0)
+ free(m_connectString);
+
+ if(errorString != 0)
+ free(errorString);
+
+ delete _localConfig;
+}
+
+
+//****************************************************************************
+//****************************************************************************
+
+int
+ConfigRetriever::init(bool onlyNodeId) {
+ if (_localConfig->init(onlyNodeId, m_connectString, _localConfigFileName, m_defaultConnectString)) {
+ return _ownNodeId = (*_localConfig)._ownNodeId;
+ }
+
+ setError(CR_ERROR, "error in retrieving contact info for mgmtsrvr");
+ _localConfig->printError();
+ _localConfig->printUsage();
+
+ return -1;
+}
+
+//****************************************************************************
+//****************************************************************************
+
+Properties *
+ConfigRetriever::getConfig(const char * nodeType, int versionId) {
+ Properties * p = getConfig(versionId);
+
+ if (p == 0) {
+ char err_buf[255];
+ snprintf(err_buf, sizeof(err_buf),
+ "No configuration retrieved for this %s node ", nodeType);
+ setError(CR_ERROR, err_buf);
+ return 0;
+ }
+
+ const Uint32 nodeId = _ownNodeId;
+
+ if (strcmp(nodeType, "DB") == 0) {
+ if (!verifyProperties("DB", p, nodeId, versionId)) return 0;
+ } else if (strcmp(nodeType, "API") == 0) {
+ if (!verifyProperties("API", p, nodeId, versionId)) return 0;
+ } else if (strcmp(nodeType, "REP") == 0) {
+ if (!verifyProperties("REP", p, nodeId, versionId)) return 0;
+ } else if (strcmp(nodeType, "MGM") == 0) {
+ if (!verifyProperties("MGM", p, nodeId, versionId)) return 0;
+ } else {
+ return 0;
+ }
+
+ return p;
+}
+
+
+//****************************************************************************
+//****************************************************************************
+Properties *
+ConfigRetriever::getConfig(int verId) {
+
+ int res = init();
+ if (res == -1) {
+ return 0;
+ }
+
+ if (_localConfig->items == 0){
+ setError(CR_ERROR, "No Management Servers configured in local config file");
+ return 0;
+ }
+
+ int retry = 1;
+ int retry_max = 12; // Max number of retry attempts
+ int retry_interval= 5; // Seconds between each retry
+ do {
+ Uint32 type = CR_ERROR;
+ for (int i = 0; i<_localConfig->items; i++){
+ MgmtSrvrId * m = _localConfig->ids[i];
+ Properties * p = 0;
+ const Uint32 nodeId = _ownNodeId;
+ switch(m->type){
+ case MgmId_TCP:
+ p = getConfig(m->data.tcp.remoteHost, m->data.tcp.port, nodeId, verId);
+ break;
+ case MgmId_File:
+ p = getConfig(m->data.file.filename, nodeId, verId);
+ break;
+ default:
+ setError(CR_ERROR, "Unknown error type");
+ break;
+ }
+
+ if (p != 0) {
+ return p;
+ }
+ if(latestErrorType == CR_RETRY)
+ type = CR_RETRY;
+ } // for
+
+ if(type == CR_RETRY){
+ REPORT_WARNING("Failed to retrieve cluster configuration");
+ ndbout << "(Cause of failure: " << getErrorString() << ")" << endl;
+ ndbout << "Attempt " << retry << " of " << retry_max << ". "
+ << "Trying again in "<<retry_interval<<" seconds..." << endl << endl;
+ NdbSleep_SecSleep(retry_interval);
+ } else {
+ break;
+ }
+ retry++;
+
+ } while (retry <= retry_max);
+
+ return 0;
+}
+
+int global_ndb_check = 0; // set to one in ndb main;
+Properties *
+ConfigRetriever::getConfig(const char * mgmhost,
+ unsigned int port,
+ Uint32 nodeId,
+ int versionId){
+ const int socketTimeout = 10000;
+ int result;
+ const NDB_SOCKET_TYPE sockfd = socket(AF_INET, SOCK_STREAM, 0);
+ if (sockfd == NDB_INVALID_SOCKET) {
+ setError(CR_RETRY, "Could not create socket to Management Server");
+ return 0;
+ }
+
+ char err_buf[255];
+ struct sockaddr_in servaddr;
+ memset(&servaddr, 0, sizeof(servaddr));
+ servaddr.sin_family = AF_INET;
+ servaddr.sin_port = htons(port);
+ // Convert ip address presentation format to numeric format
+ result = Ndb_getInAddr(&servaddr.sin_addr, mgmhost);
+ if (result != 0) {
+ snprintf(err_buf, sizeof(err_buf),
+ "Name lookup failed: host \"%s\"", mgmhost);
+ setError(CR_ERROR, err_buf);
+ return 0;
+ }
+
+ result = connect(sockfd, (struct sockaddr*) &servaddr, sizeof(servaddr));
+ if (result == -1) {
+ snprintf(err_buf, sizeof(err_buf),
+ "Failed to connect to \"%s:%d\"", mgmhost, port);
+ setError(CR_RETRY, err_buf);
+ NDB_CLOSE_SOCKET(sockfd);
+ return 0;
+ }
+
+ if(println_socket(sockfd, 1000, "GET CONFIG %d %d" ,
+ versionId, nodeId) != 0){
+ NDB_CLOSE_SOCKET(sockfd);
+ setError(CR_ERROR, "IO error, write");
+ return 0;
+ }
+
+ char buf[255];
+ {
+ const int tmp = readln_socket(sockfd, socketTimeout, buf, 255);
+ if(tmp == -1){
+ NDB_CLOSE_SOCKET(sockfd);
+ setError(CR_ERROR, "IO error, read");
+ return 0;
+ }
+
+ if(tmp == 0){
+ snprintf(err_buf, 256,
+ "IO error, failed request: "
+ "GET CONFIG %d %d", versionId, nodeId);
+ NDB_CLOSE_SOCKET(sockfd);
+ setError(CR_ERROR, err_buf);
+ return 0;
+ }
+ }
+
+ int status, version, node, bytes, bytesUU;
+ if(sscanf(buf, "GET CONFIG %d %d %d %d %d",
+ &status, &version, &node, &bytes, &bytesUU) != 5){
+ NDB_CLOSE_SOCKET(sockfd);
+ snprintf(err_buf, sizeof(err_buf),
+ "Invalid response: %s", buf);
+ setError(CR_ERROR, err_buf);
+ return 0;
+ }
+
+ if(status != 0){
+ NDB_CLOSE_SOCKET(sockfd);
+ if (status == 1){
+ snprintf(err_buf, sizeof(err_buf),
+ "Management Server: Requested version id is invalid");
+ } else if (status == 2){
+ snprintf(err_buf, sizeof(err_buf),
+ "Management Server: Node with id %d has not been specified",
+ nodeId);
+ } else if (status == 3){
+ snprintf(err_buf, sizeof(err_buf), "Management Server: Internal error");
+ } else {
+ snprintf(err_buf, sizeof(err_buf),
+ "Management Server returned unknown error: %d", status);
+ }
+ setError(CR_ERROR, err_buf);
+ return 0;
+ }
+
+ bool compatible;
+ if (global_ndb_check)
+ compatible = ndbCompatible_ndb_mgmt(versionId, version);
+ else
+ compatible = ndbCompatible_api_mgmt(versionId, version);
+
+ if(!compatible){ // if(version != versionId){
+ NDB_CLOSE_SOCKET(sockfd);
+ snprintf(err_buf, sizeof(err_buf), "Management Server: Invalid version. "
+ "Version from server: %d Own version: %d", version, versionId);
+ setError(CR_ERROR, err_buf);
+ return 0;
+ }
+
+ if(node != (int)nodeId){
+ NDB_CLOSE_SOCKET(sockfd);
+ snprintf(err_buf, sizeof(err_buf), "Management Server: Invalid node id. "
+ "Node id from server: %d Own node id: %d", node, nodeId);
+ setError(CR_ERROR, err_buf);
+ return 0;
+ }
+
+ if(readln_socket(sockfd, socketTimeout, buf, 255) == -1){
+ NDB_CLOSE_SOCKET(sockfd);
+ setError(CR_ERROR, "IO error, read");
+ return 0;
+ }
+
+ if(strncmp("begin", buf, strlen("begin")) != 0){
+ NDB_CLOSE_SOCKET(sockfd);
+ snprintf(err_buf, sizeof(err_buf),
+ "Invalid response: %s", buf);
+ setError(CR_ERROR, err_buf);
+ return 0;
+ }
+
+ char* bufUU = new char[bytesUU];
+ int read = 0;
+ int start = 0;
+ do {
+ if((read = read_socket(sockfd, socketTimeout, &bufUU[start], bytesUU-start)) == -1){
+ delete[] bufUU;
+ NDB_CLOSE_SOCKET(sockfd);
+ setError(CR_ERROR, "IO error, read(bufUU)");
+ return 0;
+ }
+ start += read;
+ } while(start < bytesUU);
+
+ Uint32 * buf2 = new Uint32[bytes/4+1]; // Properties byte size
+ char * dst = (char *)buf2;
+ int sz = 0;
+ start = 0;
+
+ for (int i = 0; i < bytesUU; i++) {
+ if (bufUU[i] == '\n') {
+ bufUU[i] = 0;
+ if (bufUU[i-1] == '\r') {
+ bufUU[i-1] = 0;
+ }
+ sz = uudecode_mem(dst, bytes, &bufUU[start]);
+ dst += sz;
+ start = i + 1; // Next row
+ }
+ }
+
+ delete[] bufUU;
+
+ if(sz < 0){
+ delete []buf2;
+ NDB_CLOSE_SOCKET(sockfd);
+ setError(CR_ERROR, "IO error, sz < 0");
+ return 0;
+ }
+
+ Properties * p = new Properties();
+ if(!p->unpack(buf2, bytes+4)){
+ snprintf(buf, sizeof(buf), "Error while unpacking %d,%d",
+ p->getPropertiesErrno(),
+ p->getOSErrno());
+ setError(CR_ERROR, buf);
+ delete []buf2;
+ delete p;
+ return 0;
+ }
+ delete []buf2;
+
+ NDB_CLOSE_SOCKET(sockfd);
+
+ return p;
+
+}
+
+Properties *
+ConfigRetriever::getConfig(const char * filename,
+ Uint32 nodeId,
+ int versionId){
+
+ struct stat sbuf;
+ const int res = stat(filename, &sbuf);
+ if(res != 0){
+ char buf[255];
+ snprintf(buf, sizeof(buf), "Could not find file: \"%s\"", filename);
+ setError(CR_ERROR, buf);
+ return 0;
+ }
+ const Uint32 bytes = sbuf.st_size;
+
+ Uint32 * buf2 = new Uint32[bytes/4+1];
+
+ FILE * f = fopen(filename, "rb");
+ if(f == 0){
+ setError(CR_ERROR, "Failed to open file");
+ delete []buf2;
+ return 0;
+ }
+ Uint32 sz = fread(buf2, 1, bytes, f);
+ fclose(f);
+ if(sz != bytes){
+ setError(CR_ERROR, "Failed to read file");
+ delete []buf2;
+ return 0;
+ }
+
+ Properties * p = new Properties();
+ if(!p->unpack(buf2, bytes+4)){
+ char buf[255];
+ snprintf(buf, sizeof(buf), "Error while unpacking %d,%d",
+ p->getPropertiesErrno(),
+ p->getOSErrno());
+ setError(CR_ERROR, buf);
+ delete []buf2;
+ delete p;
+ return 0;
+ }
+ delete [] buf2;
+
+ return p;
+}
+
+bool
+ConfigRetriever::verifyProperties(const char* nodeType, Properties * p,
+ Uint32 nodeId, int versionId){
+
+ Uint32 t = 0;
+ const Properties *tmp;
+ const char *type;
+
+ if (p == 0) return false;
+
+ bool compatible = false;
+ if (p->get("Version", &t))
+ if (global_ndb_check)
+ compatible = ndbCompatible_ndb_mgmt(versionId, t);
+ else
+ compatible = ndbCompatible_api_mgmt(versionId, t);
+
+ if(!compatible){ // if(!p->get("Version", &t) || versionId != (int)t){
+ setError(CR_ERROR, "Invalid configuration version");
+ delete p;
+ return false;
+ }
+
+ if(!p->get("LocalNodeId", &t) || nodeId != t){
+ setError(CR_ERROR, "Invalid node identity in configuration");
+ delete p;
+ return false;
+ }
+
+ if(!p->get("Node", nodeId, &tmp)){
+ setError(CR_ERROR, "Internal error while processing configuration");
+ ndbout_c("nodeId = %d", nodeId);
+ p->print();
+ delete p;
+ return false;
+ }
+
+ if(!tmp->get("Type", &type) || strcmp(type, nodeType)) {
+ if (!(!strcmp(type, "REP") && !strcmp(nodeType, "API"))) {
+ char buf[1024];
+ snprintf(buf, sizeof(buf),
+ "Configuration error: Node with id %d is not of type %s.\n"
+ "Check local config file: %s", nodeId, nodeType,
+ _localConfigFileName);
+ setError(CR_ERROR, buf);
+ return false;
+ }
+ }
+
+ return true;
+}
+
+void
+ConfigRetriever::setError(ErrorType et, const char * s){
+ if(errorString != 0){
+ free(errorString);
+ }
+ if(s == 0)
+ errorString = 0;
+ else
+ errorString = strdup(s);
+ latestErrorType = et;
+}
+
+
+const char *
+ConfigRetriever::getErrorString(){
+ return errorString;
+}
+
+void
+ConfigRetriever::setLocalConfigFileName(const char * localConfigFileName) {
+ if(_localConfigFileName != 0)
+ free(_localConfigFileName);
+ if(localConfigFileName != 0)
+ _localConfigFileName = strdup(localConfigFileName);
+ else
+ _localConfigFileName = 0;
+}
+
+void
+ConfigRetriever::setConnectString(const char * connectString) {
+ if(m_connectString != 0)
+ free(m_connectString);
+ if (connectString != 0) {
+ m_connectString = strdup(connectString);
+ } else {
+ m_connectString = 0;
+ }
+}
+
+/**
+ * @note Do not use! Use the one above if possible. /elathal
+ */
+void
+ConfigRetriever::setDefaultConnectString(const char * defaultConnectString) {
+ if(m_defaultConnectString != 0)
+ free(m_defaultConnectString);
+ if (defaultConnectString != 0) {
+ m_defaultConnectString = strdup(defaultConnectString);
+ } else {
+ m_defaultConnectString = 0;
+ }
+}
diff --git a/ndb/src/common/mgmcommon/IPCConfig.cpp b/ndb/src/common/mgmcommon/IPCConfig.cpp
new file mode 100644
index 00000000000..f75cf806cc0
--- /dev/null
+++ b/ndb/src/common/mgmcommon/IPCConfig.cpp
@@ -0,0 +1,336 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "IPCConfig.hpp"
+#include <NdbOut.hpp>
+#include <NdbHost.h>
+#include <TransporterDefinitions.hpp>
+#include <TransporterRegistry.hpp>
+#include <Properties.hpp>
+
+#if defined DEBUG_TRANSPORTER
+#define DEBUG(t) ndbout << __FILE__ << ":" << __LINE__ << ":" << t << endl;
+#else
+#define DEBUG(t)
+#endif
+
+IPCConfig::IPCConfig(Properties * p)
+{
+ theNoOfRemoteNodes = 0;
+ the_ownId = 0;
+ if(p != 0)
+ props = new Properties(* p);
+ else
+ props = 0;
+}
+
+
+IPCConfig::~IPCConfig()
+{
+ if(props != 0){
+ delete props;
+ }
+}
+
+int
+IPCConfig::init(){
+ Uint32 nodeId;
+
+ if(props == 0) return -1;
+ if(!props->get("LocalNodeId", &nodeId)) {
+ DEBUG( "Did not find local node id." );
+ return -1;
+ }
+ the_ownId = nodeId;
+
+ Uint32 noOfConnections;
+ if(!props->get("NoOfConnections", &noOfConnections)) {
+ DEBUG( "Did not find noOfConnections." );
+ return -1;
+ }
+
+ for(Uint32 i = 0; i<noOfConnections; i++){
+ const Properties * tmp;
+ Uint32 node1, node2;
+
+ if(!props->get("Connection", i, &tmp)) {
+ DEBUG( "Did not find Connection." );
+ return -1;
+ }
+ if(!tmp->get("NodeId1", &node1)) {
+ DEBUG( "Did not find NodeId1." );
+ return -1;
+ }
+ if(!tmp->get("NodeId2", &node2)) {
+ DEBUG( "Did not find NodeId2." );
+ return -1;
+ }
+
+ if(node1 == the_ownId && node2 != the_ownId)
+ if(!addRemoteNodeId(node2)) {
+ DEBUG( "addRemoteNodeId(node2) failed." );
+ return -1;
+ }
+
+ if(node1 != the_ownId && node2 == the_ownId)
+ if(!addRemoteNodeId(node1)) {
+ DEBUG( "addRemoteNodeId(node2) failed." );
+ return -1;
+ }
+ }
+ return 0;
+}
+
+bool
+IPCConfig::addRemoteNodeId(NodeId nodeId){
+ for(int i = 0; i<theNoOfRemoteNodes; i++)
+ if(theRemoteNodeIds[i] == nodeId)
+ return false;
+ theRemoteNodeIds[theNoOfRemoteNodes] = nodeId;
+ theNoOfRemoteNodes++;
+ return true;
+}
+
+/**
+ * Returns no of transporters configured
+ */
+int
+IPCConfig::configureTransporters(TransporterRegistry * theTransporterRegistry){
+ int noOfTransportersCreated = 0;
+
+ Uint32 noOfConnections;
+ if(!props->get("NoOfConnections", &noOfConnections)) return -1;
+
+ for (Uint32 i = 0; i < noOfConnections; i++){
+ const Properties * tmp;
+ Uint32 nodeId1, nodeId2;
+ const char * host1;
+ const char * host2;
+
+ if(!props->get("Connection", i, &tmp)) continue;
+ if(!tmp->get("NodeId1", &nodeId1)) continue;
+ if(!tmp->get("NodeId2", &nodeId2)) continue;
+ if(nodeId1 != the_ownId && nodeId2 != the_ownId) continue;
+
+ Uint32 sendSignalId;
+ Uint32 compression;
+ Uint32 checksum;
+ if(!tmp->get("SendSignalId", &sendSignalId)) continue;
+ if(!tmp->get("Compression", &compression)) continue;
+ if(!tmp->get("Checksum", &checksum)) continue;
+
+ const char * type;
+ if(!tmp->get("Type", &type)) continue;
+
+ if(strcmp("SHM", type) == 0){
+ SHM_TransporterConfiguration conf;
+ conf.localNodeId = the_ownId;
+ conf.remoteNodeId = (nodeId1 != the_ownId ? nodeId1 : nodeId2);
+ conf.byteOrder = 0;
+ conf.compression = compression;
+ conf.checksum = checksum;
+ conf.signalId = sendSignalId;
+
+ if(!tmp->get("ShmKey", &conf.shmKey)) continue;
+ if(!tmp->get("ShmSize", &conf.shmSize)) continue;
+
+ if(!theTransporterRegistry->createTransporter(&conf)){
+ ndbout << "Failed to create SHM Transporter from: "
+ << conf.localNodeId << " to: " << conf.remoteNodeId << endl;
+ continue;
+ } else {
+ noOfTransportersCreated++;
+ continue;
+ }
+
+ } else if(strcmp("SCI", type) == 0){
+ SCI_TransporterConfiguration conf;
+ conf.localNodeId = the_ownId;
+ conf.remoteNodeId = (nodeId1 != the_ownId ? nodeId1 : nodeId2);
+ conf.byteOrder = 0;
+ conf.compression = compression;
+ conf.checksum = checksum;
+ conf.signalId = sendSignalId;
+
+ if(!tmp->get("SendLimit", &conf.sendLimit)) continue;
+ if(!tmp->get("SharedBufferSize", &conf.bufferSize)) continue;
+
+ if(the_ownId == nodeId1){
+ if(!tmp->get("Node1_NoOfAdapters", &conf.nLocalAdapters)) continue;
+ if(!tmp->get("Node2_NoOfAdapters", &conf.nRemoteAdapters)) continue;
+ if(!tmp->get("Node2_Adapter", 0, &conf.remoteSciNodeId0)) continue;
+
+ if(conf.nRemoteAdapters > 1){
+ if(!tmp->get("Node2_Adapter", 1, &conf.remoteSciNodeId1)) continue;
+ }
+ } else {
+ if(!tmp->get("Node2_NoOfAdapters", &conf.nLocalAdapters)) continue;
+ if(!tmp->get("Node1_NoOfAdapters", &conf.nRemoteAdapters)) continue;
+ if(!tmp->get("Node1_Adapter", 0, &conf.remoteSciNodeId0)) continue;
+
+ if(conf.nRemoteAdapters > 1){
+ if(!tmp->get("Node1_Adapter", 1, &conf.remoteSciNodeId1)) continue;
+ }
+ }
+
+ if(!theTransporterRegistry->createTransporter(&conf)){
+ ndbout << "Failed to create SCI Transporter from: "
+ << conf.localNodeId << " to: " << conf.remoteNodeId << endl;
+ continue;
+ } else {
+ noOfTransportersCreated++;
+ continue;
+ }
+ }
+
+ if(!tmp->get("HostName1", &host1)) continue;
+ if(!tmp->get("HostName2", &host2)) continue;
+
+ Uint32 ownNodeId;
+ Uint32 remoteNodeId;
+ const char * ownHostName;
+ const char * remoteHostName;
+
+ if(nodeId1 == the_ownId){
+ ownNodeId = nodeId1;
+ ownHostName = host1;
+ remoteNodeId = nodeId2;
+ remoteHostName = host2;
+ } else if(nodeId2 == the_ownId){
+ ownNodeId = nodeId2;
+ ownHostName = host2;
+ remoteNodeId = nodeId1;
+ remoteHostName = host1;
+ } else {
+ continue;
+ }
+
+ if(strcmp("TCP", type) == 0){
+ TCP_TransporterConfiguration conf;
+
+ if(!tmp->get("PortNumber", &conf.port)) continue;
+ if(!tmp->get("SendBufferSize", &conf.sendBufferSize)) continue;
+ if(!tmp->get("MaxReceiveSize", &conf.maxReceiveSize)) continue;
+
+ const char * proxy;
+ if (tmp->get("Proxy", &proxy)) {
+ if (strlen(proxy) > 0 && nodeId2 == the_ownId) {
+ // TODO handle host:port
+ conf.port = atoi(proxy);
+ }
+ }
+ conf.sendBufferSize *= MAX_MESSAGE_SIZE;
+ conf.maxReceiveSize *= MAX_MESSAGE_SIZE;
+
+ conf.remoteHostName = remoteHostName;
+ conf.localHostName = ownHostName;
+ conf.remoteNodeId = remoteNodeId;
+ conf.localNodeId = ownNodeId;
+ conf.byteOrder = 0;
+ conf.compression = compression;
+ conf.checksum = checksum;
+ conf.signalId = sendSignalId;
+
+ if(!theTransporterRegistry->createTransporter(&conf)){
+ ndbout << "Failed to create TCP Transporter from: "
+ << ownNodeId << " to: " << remoteNodeId << endl;
+ } else {
+ noOfTransportersCreated++;
+ }
+
+ } else if(strcmp("OSE", type) == 0){
+
+ OSE_TransporterConfiguration conf;
+
+ if(!tmp->get("PrioASignalSize", &conf.prioASignalSize))
+ continue;
+ if(!tmp->get("PrioBSignalSize", &conf.prioBSignalSize))
+ continue;
+ if(!tmp->get("ReceiveArraySize", &conf.receiveBufferSize))
+ continue;
+
+ conf.remoteHostName = remoteHostName;
+ conf.localHostName = ownHostName;
+ conf.remoteNodeId = remoteNodeId;
+ conf.localNodeId = ownNodeId;
+ conf.byteOrder = 0;
+ conf.compression = compression;
+ conf.checksum = checksum;
+ conf.signalId = sendSignalId;
+
+ if(!theTransporterRegistry->createTransporter(&conf)){
+ ndbout << "Failed to create OSE Transporter from: "
+ << ownNodeId << " to: " << remoteNodeId << endl;
+ } else {
+ noOfTransportersCreated++;
+ }
+ } else {
+ continue;
+ }
+ }
+ return noOfTransportersCreated;
+}
+
+/**
+ * Supply a nodeId,
+ * and get next higher node id
+ * Returns false if none found
+ */
+bool
+IPCConfig::getNextRemoteNodeId(NodeId & nodeId) const {
+ NodeId returnNode = MAX_NODES + 1;
+ for(int i = 0; i<theNoOfRemoteNodes; i++)
+ if(theRemoteNodeIds[i] > nodeId){
+ if(theRemoteNodeIds[i] < returnNode){
+ returnNode = theRemoteNodeIds[i];
+ }
+ }
+ if(returnNode == (MAX_NODES + 1))
+ return false;
+ nodeId = returnNode;
+ return true;
+}
+
+
+Uint32
+IPCConfig::getREPHBFrequency(NodeId id) const {
+ const Properties * tmp;
+ Uint32 out;
+
+ /**
+ * Todo: Fix correct heartbeat
+ */
+ if (!props->get("Node", id, &tmp) ||
+ !tmp->get("HeartbeatIntervalRepRep", &out)) {
+ DEBUG("Illegal Node or HeartbeatIntervalRepRep in config.");
+ out = 10000;
+ }
+
+ return out;
+}
+
+const char*
+IPCConfig::getNodeType(NodeId id) const {
+ const char * out;
+ const Properties * tmp;
+
+ if (!props->get("Node", id, &tmp) || !tmp->get("Type", &out)) {
+ DEBUG("Illegal Node or NodeType in config.");
+ out = "Unknown";
+ }
+
+ return out;
+}
diff --git a/ndb/src/common/mgmcommon/InitConfigFileParser.cpp b/ndb/src/common/mgmcommon/InitConfigFileParser.cpp
new file mode 100644
index 00000000000..33652fa472c
--- /dev/null
+++ b/ndb/src/common/mgmcommon/InitConfigFileParser.cpp
@@ -0,0 +1,544 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "InitConfigFileParser.hpp"
+#include <string.h>
+#include <errno.h>
+#include "Config.hpp"
+#include "MgmtErrorReporter.hpp"
+#include <NdbOut.hpp>
+#include "ConfigInfo.hpp"
+#include <stdarg.h>
+#include <ctype.h>
+#include <NdbString.h>
+
+const int MAX_LINE_LENGTH = 120; // Max length of line of text in config file
+static void trim(char *);
+
+static void require(bool v) { if(!v) abort();}
+
+//****************************************************************************
+// Ctor / Dtor
+//****************************************************************************
+InitConfigFileParser::InitConfigFileParser(const char* initialConfigFileName){
+
+ m_initConfigStream = fopen(initialConfigFileName, "r");
+
+ m_info = new ConfigInfo();
+ m_config = new Config();
+ m_defaults = new Properties();
+ m_defaults->setCaseInsensitiveNames(true);
+}
+
+InitConfigFileParser::~InitConfigFileParser() {
+ if (m_initConfigStream != NULL) fclose(m_initConfigStream);
+
+ delete m_info;
+ delete m_config;
+ delete m_defaults;
+}
+
+//****************************************************************************
+// Read Config File
+//****************************************************************************
+bool InitConfigFileParser::readConfigFile() {
+
+ char line[MAX_LINE_LENGTH];
+
+ Context ctx;
+ ctx.m_lineno = 0;
+ ctx.m_currentSection = 0;
+
+ ctx.m_info = m_info;
+ ctx.m_config = m_config;
+ ctx.m_defaults = m_defaults;
+
+ /*************
+ * Open file *
+ *************/
+ if (m_initConfigStream == NULL) {
+ ctx.reportError("Could not open file.");
+ return false;
+ }
+
+ /***********************
+ * While lines to read *
+ ***********************/
+ while (fgets(line, MAX_LINE_LENGTH, m_initConfigStream)) {
+ ctx.m_lineno++;
+
+ trim(line);
+
+ if (isEmptyLine(line)) // Skip if line is empty or comment
+ continue;
+
+ // End with NULL instead of newline
+ if (line[strlen(line)-1] == '\n')
+ line[strlen(line)-1] = '\0';
+
+ /********************************
+ * 1. Parse new default section *
+ ********************************/
+ if (char* section = parseDefaultSectionHeader(line)) {
+ if(!storeSection(ctx)){
+ free(section);
+ ctx.reportError("Could not store previous default section "
+ "of configuration file.");
+ return false;
+ }
+
+ snprintf(ctx.fname, sizeof(ctx.fname), section); free(section);
+ ctx.type = InitConfigFileParser::DefaultSection;
+ ctx.m_sectionLineno = ctx.m_lineno;
+ ctx.m_currentSection = new Properties();
+ ctx.m_userDefaults = NULL;
+ ctx.m_currentInfo = m_info->getInfo(ctx.fname);
+ ctx.m_systemDefaults = m_info->getDefaults(ctx.fname);
+ continue;
+ }
+
+ /************************
+ * 2. Parse new section *
+ ************************/
+ if (char* section = parseSectionHeader(line)) {
+ if(!storeSection(ctx)){
+ free(section);
+ ctx.reportError("Could not store previous section "
+ "of configuration file.");
+ return false;
+ }
+
+ snprintf(ctx.fname, sizeof(ctx.fname), section);
+ free(section);
+ ctx.type = InitConfigFileParser::Section;
+ ctx.m_sectionLineno = ctx.m_lineno;
+ ctx.m_currentSection = new Properties();
+ ctx.m_userDefaults = getSection(ctx.fname, m_defaults);
+ ctx.m_currentInfo = m_info->getInfo(ctx.fname);
+ ctx.m_systemDefaults = m_info->getDefaults(ctx.fname);
+ continue;
+ }
+
+ /****************************
+ * 3. Parse name-value pair *
+ ****************************/
+ if (!parseNameValuePair(ctx, line)) {
+ ctx.reportError("Could not parse name-value pair in config file.");
+ return false;
+ }
+ }
+
+ if(!storeSection(ctx)) {
+ ctx.reportError("Could not store section of configuration file.");
+ return false;
+ }
+
+ Uint32 nConnections = 0;
+ Uint32 nComputers = 0;
+ Uint32 nNodes = 0;
+ Uint32 nExtConnections = 0;
+ const char * system = "?";
+ ctx.m_userProperties.get("NoOfConnections", &nConnections);
+ ctx.m_userProperties.get("NoOfComputers", &nComputers);
+ ctx.m_userProperties.get("NoOfNodes", &nNodes);
+ ctx.m_userProperties.get("ExtNoOfConnections", &nExtConnections);
+ ctx.m_userProperties.get("ExtSystem", &system);
+ m_config->put("NoOfConnections", nConnections);
+ m_config->put("NoOfComputers", nComputers);
+ m_config->put("NoOfNodes", nNodes);
+
+ char tmpLine[MAX_LINE_LENGTH];
+ snprintf(tmpLine, MAX_LINE_LENGTH, "EXTERNAL SYSTEM_");
+ strncat(tmpLine, system, MAX_LINE_LENGTH);
+ strncat(tmpLine, ":NoOfConnections", MAX_LINE_LENGTH);
+ m_config->put(tmpLine, nExtConnections);
+
+ if (ferror(m_initConfigStream)) {
+ ctx.reportError("Failure in reading");
+ return false;
+ }
+ return true;
+}
+
+//****************************************************************************
+// Parse Name-Value Pair
+//****************************************************************************
+
+bool InitConfigFileParser::parseNameValuePair(Context& ctx, const char* line) {
+
+ char tmpLine[MAX_LINE_LENGTH];
+ char fname[MAX_LINE_LENGTH], rest[MAX_LINE_LENGTH];
+ char* t;
+
+ if (ctx.m_currentSection == NULL){
+ ctx.reportError("Value specified outside section");
+ return false;
+ }
+
+ strncpy(tmpLine, line, MAX_LINE_LENGTH);
+
+ // *************************************
+ // Check if a separator exists in line
+ // *************************************
+ if (!strchr(tmpLine, ':')) {
+ ctx.reportError("Parse error");
+ return false;
+ }
+
+ // *******************************************
+ // Get pointer to substring before separator
+ // *******************************************
+ t = strtok(tmpLine, ":");
+
+ // *****************************************
+ // Count number of tokens before separator
+ // *****************************************
+ if (sscanf(t, "%120s%120s", fname, rest) != 1) {
+ ctx.reportError("Multiple names before \':\'");
+ return false;
+ }
+ if (!ctx.m_currentInfo->contains(fname)) {
+ ctx.reportError("[%s] Unknown parameter: %s", ctx.fname, fname);
+ return false;
+ }
+ ConfigInfo::Status status = m_info->getStatus(ctx.m_currentInfo, fname);
+ if (status == ConfigInfo::NOTIMPLEMENTED) {
+ ctx.reportWarning("[%s] %s not yet implemented", ctx.fname, fname);
+ }
+ if (status == ConfigInfo::DEPRICATED) {
+ ctx.reportWarning("[%s] %s is depricated", ctx.fname, fname);
+ }
+
+ // ******************************************
+ // Get pointer to substring after separator
+ // ******************************************
+ t = strtok(NULL, "\0");
+ if (t == NULL) {
+ ctx.reportError("No value for parameter");
+ return false;
+ }
+
+ // ******************************************
+ // Remove prefix and postfix spaces and tabs
+ // *******************************************
+ trim(t);
+
+ // ***********************
+ // Store name-value pair
+ // ***********************
+ return storeNameValuePair(ctx, fname, t);
+}
+
+
+//****************************************************************************
+// STORE NAME-VALUE pair in properties section
+//****************************************************************************
+
+bool
+InitConfigFileParser::storeNameValuePair(Context& ctx,
+ const char* fname,
+ const char* value) {
+
+ const char * pname = m_info->getPName(ctx.m_currentInfo, fname);
+
+ if (ctx.m_currentSection->contains(pname)) {
+ ctx.reportError("[%s] Parameter %s specified twice", ctx.fname, fname);
+ return false;
+ }
+
+ // ***********************
+ // Store name-value pair
+ // ***********************
+
+ switch(m_info->getType(ctx.m_currentInfo, fname)){
+ case ConfigInfo::BOOL: {
+ bool value_bool;
+ if (!convertStringToBool(value, value_bool)) {
+ ctx.reportError("Illegal boolean value for parameter %s", fname);
+ return false;
+ }
+ MGM_REQUIRE(ctx.m_currentSection->put(pname, value_bool));
+ break;
+ }
+ case ConfigInfo::INT:{
+ Uint32 value_int;
+ if (!convertStringToUint32(value, value_int)) {
+ ctx.reportError("Illegal integer value for parameter %s", fname);
+ return false;
+ }
+ if (!m_info->verify(ctx.m_currentInfo, fname, value_int)) {
+ ctx.reportError("Illegal value %s for parameter %s.\n"
+ "Legal values are between %d and %d", value, fname,
+ m_info->getMin(ctx.m_currentInfo, fname),
+ m_info->getMax(ctx.m_currentInfo, fname));
+ return false;
+ }
+ MGM_REQUIRE(ctx.m_currentSection->put(pname, value_int));
+ break;
+ }
+ case ConfigInfo::STRING:
+ MGM_REQUIRE(ctx.m_currentSection->put(pname, value));
+ break;
+ }
+ return true;
+}
+
+//****************************************************************************
+// Is Empty Line
+//****************************************************************************
+
+bool InitConfigFileParser::isEmptyLine(const char* line) const {
+ int i;
+
+ // Check if it is a comment line
+ if (line[0] == '#') return true;
+
+ // Check if it is a line with only spaces
+ for (i = 0; i < MAX_LINE_LENGTH && line[i] != '\n' && line[i] != '\0'; i++) {
+ if (line[i] != ' ' && line[i] != '\t') return false;
+ }
+ return true;
+}
+
+//****************************************************************************
+// Convert String to Int
+//****************************************************************************
+bool InitConfigFileParser::convertStringToUint32(const char* s,
+ Uint32& val,
+ Uint32 log10base) {
+ if (s == NULL)
+ return false;
+ if (strlen(s) == 0)
+ return false;
+
+ errno = 0;
+ char* p;
+ long v = strtol(s, &p, 10);
+ if (errno != 0)
+ return false;
+
+ long mul = 0;
+ if (p != &s[strlen(s)]){
+ char * tmp = strdup(p);
+ trim(tmp);
+ switch(tmp[0]){
+ case 'k':
+ case 'K':
+ mul = 10;
+ break;
+ case 'M':
+ mul = 20;
+ break;
+ case 'G':
+ mul = 30;
+ break;
+ default:
+ free(tmp);
+ return false;
+ }
+ free(tmp);
+ }
+
+ val = (v << mul);
+ return true;
+}
+
+bool InitConfigFileParser::convertStringToBool(const char* s, bool& val) {
+ if (s == NULL) return false;
+ if (strlen(s) == 0) return false;
+
+ if (!strcmp(s, "Y") || !strcmp(s, "y") ||
+ !strcmp(s, "Yes") || !strcmp(s, "YES") || !strcmp(s, "yes") ||
+ !strcmp(s, "True") || !strcmp(s, "TRUE") || !strcmp(s, "true")) {
+ val = true;
+ return true;
+ }
+
+ if (!strcmp(s, "N") || !strcmp(s, "n") ||
+ !strcmp(s, "No") || !strcmp(s, "NO") || !strcmp(s, "no") ||
+ !strcmp(s, "False") || !strcmp(s, "FALSE") || !strcmp(s, "false")) {
+ val = false;
+ return true;
+ }
+
+ return false; // Failure to convert
+}
+
+//****************************************************************************
+// Get Config
+//****************************************************************************
+const Config* InitConfigFileParser::getConfig() {
+ Uint32 nConnections = 0;
+ Uint32 nComputers = 0;
+ Uint32 nNodes = 0;
+ m_config->get("NoOfConnections", &nConnections);
+ m_config->get("NoOfComputers", &nComputers);
+ m_config->get("NoOfNodes", &nNodes);
+
+ return m_config;
+}
+
+
+//****************************************************************************
+// Parse Section Header
+//****************************************************************************
+static void
+trim(char * str){
+ int len = strlen(str);
+ for(len--;
+ (str[len] == '\r' || str[len] == '\n' ||
+ str[len] == ' ' || str[len] == '\t') &&
+ len > 0;
+ len--)
+ str[len] = 0;
+
+ int pos = 0;
+ while(str[pos] == ' ' || str[pos] == '\t')
+ pos++;
+
+ if(str[pos] == '\"' && str[len] == '\"') {
+ pos++;
+ str[len] = 0;
+ len--;
+ }
+
+ memmove(str, &str[pos], len - pos + 2);
+}
+
+char*
+InitConfigFileParser::parseSectionHeader(const char* line) const {
+ char * tmp = strdup(line);
+
+ if(tmp[0] != '['){
+ free(tmp);
+ return NULL;
+ }
+
+ if(tmp[strlen(tmp)-1] != ']'){
+ free(tmp);
+ return NULL;
+ }
+ tmp[strlen(tmp)-1] = 0;
+
+ tmp[0] = ' ';
+ trim(tmp);
+
+ // Lookup token among sections
+ if(!m_info->isSection(tmp)) return NULL;
+ if(m_info->getInfo(tmp)) return tmp;
+
+ free(tmp);
+ return NULL;
+}
+
+//****************************************************************************
+// Parse Default Section Header
+//****************************************************************************
+
+char*
+InitConfigFileParser::parseDefaultSectionHeader(const char* line) const {
+ static char token1[MAX_LINE_LENGTH], token2[MAX_LINE_LENGTH];
+
+ int no = sscanf(line, "[%120[A-Za-z] %120[A-Za-z]]", token1, token2);
+
+ // Not correct no of tokens
+ if (no != 2) return NULL;
+
+ // Not correct keyword at end
+ if (!strcmp(token2, "DEFAULT") == 0) return NULL;
+
+ if(m_info->getInfo(token1)){
+ return strdup(token1);
+ }
+
+ // Did not find section
+ return NULL;
+}
+
+const Properties *
+InitConfigFileParser::getSection(const char * name, const Properties * src){
+ const Properties * p;
+ if(src && src->get(name, &p))
+ return p;
+
+ return 0;
+}
+
+//****************************************************************************
+// STORE section
+//****************************************************************************
+bool
+InitConfigFileParser::storeSection(Context& ctx){
+ if(ctx.m_currentSection == NULL)
+ return true;
+
+ for(int i = strlen(ctx.fname) - 1; i>=0; i--){
+ ctx.fname[i] = toupper(ctx.fname[i]);
+ }
+
+ snprintf(ctx.pname, sizeof(ctx.pname), ctx.fname);
+
+ char buf[255];
+ if(ctx.type == InitConfigFileParser::Section)
+ snprintf(buf, sizeof(buf), "%s", ctx.fname);
+ if(ctx.type == InitConfigFileParser::DefaultSection)
+ snprintf(buf, sizeof(buf), "%s DEFAULT", ctx.fname);
+
+ snprintf(ctx.fname, sizeof(ctx.fname), buf);
+ for(int i = 0; i<m_info->m_NoOfRules; i++){
+ const ConfigInfo::SectionRule & rule = m_info->m_SectionRules[i];
+ if(strcmp(rule.m_section, ctx.fname) == 0)
+ if(!(* rule.m_sectionRule)(ctx, rule.m_ruleData)){
+ return false;
+ }
+ }
+
+ if(ctx.type == InitConfigFileParser::DefaultSection)
+ require(m_defaults->put(ctx.pname, ctx.m_currentSection));
+
+ if(ctx.type == InitConfigFileParser::Section)
+ require(m_config->put(ctx.pname, ctx.m_currentSection));
+
+ delete ctx.m_currentSection; ctx.m_currentSection = NULL;
+
+ return true;
+}
+
+void
+InitConfigFileParser::Context::reportError(const char * fmt, ...){
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ ndbout << "Error line " << m_lineno << ": " << buf << endl;
+ va_end(ap);
+
+ //m_currentSection->print();
+}
+
+void
+InitConfigFileParser::Context::reportWarning(const char * fmt, ...){
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ ndbout << "Warning line " << m_lineno << ": " << buf << endl;
+ va_end(ap);
+}
diff --git a/ndb/src/common/mgmcommon/InitConfigFileParser.hpp b/ndb/src/common/mgmcommon/InitConfigFileParser.hpp
new file mode 100644
index 00000000000..1e85067396c
--- /dev/null
+++ b/ndb/src/common/mgmcommon/InitConfigFileParser.hpp
@@ -0,0 +1,142 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef InitConfigFileParser_H
+#define InitConfigFileParser_H
+
+#include <stdio.h>
+#include <Properties.hpp>
+
+class Config;
+class ConfigInfo;
+
+/**
+ * @class InitConfigFileParser
+ * @brief Reads initial config file and returns Config object
+ *
+ * This class contains one public method InitConfigFileParser::getConfig,
+ * which reads an initial configuration file and returns a Config
+ * object if the config file has correct syntax and semantic.
+ */
+class InitConfigFileParser {
+public:
+ enum ContextSectionType { Undefined, Section, DefaultSection };
+
+ /**
+ * Context = Which section in init config file we are currently parsing
+ */
+ struct Context {
+ ContextSectionType type; ///< Section type (e.g. default section,section)
+ char fname[256]; ///< Section name occuring in init config file
+ char pname[256]; ///< Section name stored in properties object
+ Uint32 m_lineno; ///< Current line no in config file
+ Uint32 m_sectionLineno; ///< Where did current section start
+
+ const ConfigInfo * m_info; // The config info
+ const Properties * m_config; // The config object
+ const Properties * m_defaults; // The user defaults
+
+ Properties * m_currentSection; // The current section I'm in
+ const Properties * m_userDefaults; // The defaults of this section
+ const Properties * m_systemDefaults; // The syst. defaults for this section
+ const Properties * m_currentInfo; // The "info" for this section
+
+ Properties m_userProperties; // User properties (temporary values)
+
+ public:
+ void reportError(const char * msg, ...);
+ void reportWarning(const char * msg, ...);
+ };
+
+
+ /**
+ * Constructor
+ * @param initialConfigFileName: Name of the initial configuration file
+ */
+ InitConfigFileParser(const char* initialConfigFileName);
+ ~InitConfigFileParser();
+
+ /**
+ * Reads the initial configuration file, checks syntax and semantic
+ * and stores internally the values of all parameters.
+ *
+ * @returns true if succeeded, o/w false (e.g. incorrect config file)
+ */
+ bool readConfigFile();
+
+ /**
+ * Get config. Must execute InitConfigFileParser::readConfigFile first.
+ *
+ * @returns Config if succeeded, o/w NULL
+ */
+ const Config* getConfig();
+
+
+private:
+ /**
+ * Check if line only contains space/comments
+ * @param line: The line to check
+ * @return true if spaces/comments only, false otherwise
+ */
+ bool isEmptyLine(const char* line) const;
+
+ /**
+ * Checks if line contains a section header
+ * @param line: String to search
+ * @return section header if matching some section header, NULL otherwise
+ */
+ char* parseSectionHeader(const char* line) const;
+
+ /**
+ * Checks if line contains a default header
+ * @param line: String to search
+ * @return section header if matching some section header, NULL otherwise
+ */
+ char* parseDefaultSectionHeader(const char* line) const;
+
+ bool parseNameValuePair(Context&, const char* line);
+ bool storeNameValuePair(Context&, const char* fname, const char* value);
+
+ bool convertStringToUint32(const char* s, Uint32& val, Uint32 log10base = 0);
+ bool convertStringToBool(const char* s, bool& val);
+
+ const Properties* getSection(const char * name, const Properties* src);
+
+ /***************************************************************************
+ * VARIABLES
+ ***************************************************************************/
+ FILE* m_initConfigStream;
+
+ /**
+ * Information about parameters (min, max values etc)
+ */
+ const ConfigInfo* m_info;
+
+ /**
+ * Configuration from initial configuration file
+ * (returned by InitConfigFileParser::readConfigFile)
+ */
+ Config* m_config;
+
+ /**
+ * Default values specified in default sections
+ */
+ Properties* m_defaults;
+
+ bool storeSection(Context&);
+};
+
+#endif // InitConfigFileParser_H
diff --git a/ndb/src/common/mgmcommon/LocalConfig.cpp b/ndb/src/common/mgmcommon/LocalConfig.cpp
new file mode 100644
index 00000000000..12e685ced34
--- /dev/null
+++ b/ndb/src/common/mgmcommon/LocalConfig.cpp
@@ -0,0 +1,308 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "LocalConfig.hpp"
+#include <NdbEnv.h>
+#include <NdbConfig.h>
+
+LocalConfig::LocalConfig(){
+ ids = 0; size = 0; items = 0;
+ error_line = 0; error_msg[0] = 0;
+}
+
+bool
+LocalConfig::init(bool onlyNodeId,
+ const char *connectString,
+ const char *fileName,
+ const char *defaultConnectString) {
+ /**
+ * Escalation:
+ * 1. Check connectString
+ * 2. Check given filename
+ * 3. Check environment variable NDB_CONNECTSTRING
+ * 4. Check Ndb.cfg in NDB_HOME
+ * 5. Check Ndb.cfg in cwd
+ * 6. Check defaultConnectString
+ */
+
+ //1. Check connectString
+ if(connectString != 0) {
+ if(readConnectString(connectString, onlyNodeId)){
+ return true;
+ }
+ return false;
+ }
+
+ //2. Check given filename
+ if (fileName && strlen(fileName) > 0) {
+ bool fopenError;
+ if(readFile(fileName, fopenError, onlyNodeId)){
+ return true;
+ }
+ return false;
+ }
+
+ //3. Check environment variable
+ char buf[255];
+ if(NdbEnv_GetEnv("NDB_CONNECTSTRING", buf, sizeof(buf)) &&
+ strlen(buf) != 0){
+ if(readConnectString(buf, onlyNodeId)){
+ return true;
+ }
+ return false;
+ }
+
+ //4. Check Ndb.cfg in NDB_HOME
+ {
+ bool fopenError;
+ char buf[256];
+ if(readFile(NdbConfig_NdbCfgName(buf, sizeof(buf), 1 /*true*/), fopenError, onlyNodeId)){
+ return true;
+ }
+ if (!fopenError)
+ return false;
+ }
+
+ //5. Check Ndb.cfg in cwd
+ {
+ bool fopenError;
+ char buf[256];
+ if(readFile(NdbConfig_NdbCfgName(buf, sizeof(buf), 0 /*false*/), fopenError, onlyNodeId)){
+ return true;
+ }
+ if (!fopenError)
+ return false;
+ }
+
+ //6. Check defaultConnectString
+ if(defaultConnectString != 0) {
+ if(readConnectString(defaultConnectString, onlyNodeId)){
+ return true;
+ }
+ return false;
+ }
+
+ setError(0, "");
+
+ return false;
+}
+
+LocalConfig::~LocalConfig(){
+ for(int i = 0; i<items; i++){
+ if(ids[i]->type == MgmId_TCP)
+ free(ids[i]->data.tcp.remoteHost);
+ else if(ids[i]->type == MgmId_File)
+ free(ids[i]->data.file.filename);
+ delete ids[i];
+ }
+ if(ids != 0)
+ delete[] ids;
+}
+
+void LocalConfig::add(MgmtSrvrId * i){
+ if(items == size){
+ MgmtSrvrId ** tmp = new MgmtSrvrId * [size+10];
+ if(ids != 0){
+ memcpy(tmp, ids, items*sizeof(MgmtSrvrId *));
+ delete []ids;
+ }
+ ids = tmp;
+ }
+ ids[items] = i;
+ items++;
+}
+
+void LocalConfig::setError(int lineNumber, const char * _msg) {
+ error_line = lineNumber;
+ strncpy(error_msg, _msg, sizeof(error_msg));
+}
+
+void LocalConfig::printError() const {
+ ndbout << "Local configuration error"<< endl
+ << "Line: "<< error_line << ", " << error_msg << endl << endl;
+}
+
+void LocalConfig::printUsage() const {
+ ndbout << "This node needs information on how to connect"<<endl
+ << "to the NDB Management Server."<<endl
+ << "The information can be supplied in one of the following ways:"
+ << endl;
+
+ ndbout << "1. Put a Ndb.cfg file in the directory where you start"<<endl
+ << " the node. "<< endl
+ << " Ex: Ndb.cfg" << endl
+ << " | nodeid=11;host=localhost:2200"<<endl<<endl;
+
+ ndbout << "2. Use the environment variable NDB_CONNECTSTRING to "<<endl
+ << " provide this information." <<endl
+ << " Ex: " << endl
+ << " >export NDB_CONNECTSTRING=\"nodeid=11;host=localhost:2200\""
+ <<endl<<endl;
+}
+
+char *nodeIdTokens[] = {
+ "OwnProcessId %i",
+ "nodeid=%i",
+ 0
+};
+
+char *hostNameTokens[] = {
+ "host://%[^:]:%i",
+ "host=%[^:]:%i",
+ "%[^:]:%i",
+ "%s %i",
+ 0
+};
+
+char *fileNameTokens[] = {
+ "file://%s",
+ "file=%s",
+ 0
+};
+
+bool
+LocalConfig::parseNodeId(const char * buf){
+ for(int i = 0; nodeIdTokens[i] != 0; i++)
+ if (sscanf(buf, nodeIdTokens[i], &_ownNodeId) == 1)
+ return true;
+ return false;
+}
+
+bool
+LocalConfig::parseHostName(const char * buf){
+ char tempString[100];
+ int port;
+ for(int i = 0; hostNameTokens[i] != 0; i++) {
+ if (sscanf(buf, hostNameTokens[i], tempString, &port) == 2) {
+ MgmtSrvrId* mgmtSrvrId = new MgmtSrvrId();
+ mgmtSrvrId->type = MgmId_TCP;
+ mgmtSrvrId->data.tcp.remoteHost = strdup(tempString);
+ mgmtSrvrId->data.tcp.port = port;
+ add(mgmtSrvrId);
+ return true;
+ }
+ }
+ return false;
+}
+
+bool
+LocalConfig::parseFileName(const char * buf){
+ char tempString[100];
+ for(int i = 0; fileNameTokens[i] != 0; i++) {
+ if (sscanf(buf, fileNameTokens[i], tempString) == 1) {
+ MgmtSrvrId* mgmtSrvrId = new MgmtSrvrId();
+ mgmtSrvrId->type = MgmId_File;
+ mgmtSrvrId->data.file.filename = strdup(tempString);
+ add(mgmtSrvrId);
+ return true;
+ }
+ }
+ return false;
+}
+
+bool
+LocalConfig::parseString(const char * connectString, bool onlyNodeId, char *line){
+ bool return_value = true;
+
+ char * for_strtok;
+ char * copy = strdup(connectString);
+
+ bool b_nodeId = false;
+ bool found_other = false;
+
+ for (char *tok = strtok_r(copy,";",&for_strtok);
+ tok != 0 && !(onlyNodeId && b_nodeId);
+ tok = strtok_r(NULL, ";", &for_strtok)) {
+
+ if (tok[0] == '#') continue;
+
+ if (!b_nodeId) // only one nodeid definition allowed
+ if (b_nodeId = parseNodeId(tok))
+ continue;
+ if (onlyNodeId)
+ continue;
+ if (found_other = parseHostName(tok))
+ continue;
+ if (found_other = parseFileName(tok))
+ continue;
+
+ snprintf(line, 150, "Unexpected entry: \"%s\"", tok);
+ return_value = false;
+ break;
+ }
+
+ if (return_value && !onlyNodeId && !found_other) {
+ return_value = false;
+ snprintf(line, 150, "Missing host/file name extry in \"%s\"", connectString);
+ }
+
+ free(copy);
+ return return_value;
+}
+
+bool LocalConfig::readFile(const char * filename, bool &fopenError, bool onlyNodeId)
+{
+ char line[150], line2[150];
+
+ fopenError = false;
+
+ FILE * file = fopen(filename, "r");
+ if(file == 0){
+ snprintf(line, 150, "Unable to open local config file: %s", filename);
+ setError(0, line);
+ fopenError = true;
+ return false;
+ }
+
+ int sz = 1024;
+ char* theString = (char*)malloc(sz);
+ theString[0] = 0;
+
+ fgets(theString, sz, file);
+ while (fgets(line+1, 100, file)) {
+ line[0] = ';';
+ while (strlen(theString) + strlen(line) >= sz) {
+ sz = sz*2;
+ char *newString = (char*)malloc(sz);
+ strcpy(newString, theString);
+ free(theString);
+ theString = newString;
+ }
+ strcat(theString, line);
+ }
+
+ bool return_value = parseString(theString, onlyNodeId, line);
+
+ if (!return_value) {
+ snprintf(line2, 150, "Reading %s: %s", filename, line);
+ setError(0,line2);
+ }
+
+ free(theString);
+ fclose(file);
+ return return_value;
+}
+
+bool
+LocalConfig::readConnectString(const char * connectString, bool onlyNodeId){
+ char line[150], line2[150];
+ bool return_value = parseString(connectString, onlyNodeId, line);
+ if (!return_value) {
+ snprintf(line2, 150, "Reading NDB_CONNECTSTRING \"%s\": %s", connectString, line);
+ setError(0,line2);
+ }
+ return return_value;
+}
diff --git a/ndb/src/common/mgmcommon/LocalConfig.hpp b/ndb/src/common/mgmcommon/LocalConfig.hpp
new file mode 100644
index 00000000000..ec7b572e92d
--- /dev/null
+++ b/ndb/src/common/mgmcommon/LocalConfig.hpp
@@ -0,0 +1,83 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LocalConfig_H
+#define LocalConfig_H
+
+#include <stdlib.h>
+#include <string.h>
+#include <NdbOut.hpp>
+#include <NdbStdio.h>
+
+//****************************************************************************
+// Description: The class LocalConfig corresponds to the information possible
+// to give in the local configuration file.
+//*****************************************************************************
+
+enum MgmtSrvrId_Type {
+ MgmId_TCP = 0,
+ MgmId_File = 1
+};
+
+struct MgmtSrvrId {
+ MgmtSrvrId_Type type;
+ union {
+ struct {
+ char * remoteHost;
+ unsigned int port;
+ } tcp;
+ struct {
+ char * filename;
+ } file;
+ } data;
+};
+
+struct LocalConfig {
+
+ int _ownNodeId;
+
+ int size;
+ int items;
+ MgmtSrvrId ** ids;
+
+ int error_line;
+ char error_msg[256];
+
+ LocalConfig();
+ ~LocalConfig();
+ bool init(bool onlyNodeId = false,
+ const char *connectString = 0,
+ const char *fileName = 0,
+ const char *defaultConnectString = 0);
+
+ void add(MgmtSrvrId *i);
+
+ void printError() const;
+ void printUsage() const;
+
+ void setError(int lineNumber, const char * _msg);
+ bool readConnectString(const char * connectString, bool onlyNodeId = false);
+ bool readFile(const char * filename, bool &fopenError, bool onlyNodeId = false);
+ bool parseLine(char * line, int lineNumber);
+
+ bool parseNodeId(const char *buf);
+ bool parseHostName(const char *buf);
+ bool parseFileName(const char *buf);
+ bool parseString(const char *buf, bool onlyNodeId, char *line);
+};
+
+#endif // LocalConfig_H
+
diff --git a/ndb/src/common/mgmcommon/Makefile b/ndb/src/common/mgmcommon/Makefile
new file mode 100644
index 00000000000..2db7be01d60
--- /dev/null
+++ b/ndb/src/common/mgmcommon/Makefile
@@ -0,0 +1,26 @@
+include .defs.mk
+
+TYPE := ndbapi mgmapiclient
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := mgmsrvcommon
+
+DIRS := printConfig
+
+SOURCES = \
+ LocalConfig.cpp \
+ Config.cpp \
+ ConfigInfo.cpp \
+ ConfigRetriever.cpp \
+ InitConfigFileParser.cpp \
+ IPCConfig.cpp
+
+SOURCES.c = NdbConfig.c
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
+
diff --git a/ndb/src/common/mgmcommon/NdbConfig.c b/ndb/src/common/mgmcommon/NdbConfig.c
new file mode 100644
index 00000000000..b12d9fcfaf9
--- /dev/null
+++ b/ndb/src/common/mgmcommon/NdbConfig.c
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbConfig.h>
+#include <NdbEnv.h>
+#include <stdlib.h>
+#include <string.h>
+
+const char*
+NdbConfig_HomePath(char* buf, int buflen){
+ const char* p;
+ p = NdbEnv_GetEnv("NDB_HOME", buf, buflen);
+ if (p == NULL){
+ strlcpy(buf, "", buflen);
+ p = buf;
+ } else {
+ const int len = strlen(buf);
+ if(len != 0 && buf[len-1] != '/'){
+ buf[len] = '/';
+ buf[len+1] = 0;
+ }
+ }
+ return p;
+}
+
+const char*
+NdbConfig_NdbCfgName(char* buf, int buflen, int with_ndb_home){
+ if (with_ndb_home)
+ NdbConfig_HomePath(buf, buflen);
+ else
+ buf[0] = 0;
+ strlcat(buf, "Ndb.cfg", buflen);
+ return buf;
+}
+
+const char*
+NdbConfig_ErrorFileName(char* buf, int buflen){
+ NdbConfig_HomePath(buf, buflen);
+ strlcat(buf, "error.log", buflen);
+ return buf;
+}
+
+const char*
+NdbConfig_ClusterLogFileName(char* buf, int buflen){
+ NdbConfig_HomePath(buf, buflen);
+ strlcat(buf, "cluster.log", buflen);
+ return buf;
+}
diff --git a/ndb/src/common/mgmcommon/printConfig/Makefile b/ndb/src/common/mgmcommon/printConfig/Makefile
new file mode 100644
index 00000000000..9194316da87
--- /dev/null
+++ b/ndb/src/common/mgmcommon/printConfig/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE := ndbapi mgmapiclient
+
+BIN_TARGET := printConfig
+BIN_TARGET_ARCHIVES := general portlib
+
+CCFLAGS_LOC += -I..
+
+SOURCES := printConfig.cpp
+
+SOURCES.c := ../ConfigRetriever.c ../NdbConfig.c ../LocalConfig.c
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/common/mgmcommon/printConfig/printConfig.cpp b/ndb/src/common/mgmcommon/printConfig/printConfig.cpp
new file mode 100644
index 00000000000..7260a84ce7a
--- /dev/null
+++ b/ndb/src/common/mgmcommon/printConfig/printConfig.cpp
@@ -0,0 +1,89 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbMain.h>
+#include <ConfigRetriever.hpp>
+#include <Properties.hpp>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+void usage(const char * prg){
+ ndbout << "Usage " << prg
+ << " host <mgm host> <mgm port> <node id> [<ver id>]" << endl;
+
+ char buf[255];
+ for(unsigned i = 0; i<strlen(prg); i++)
+ buf[i] = ' ';
+ buf[strlen(prg)] = 0;
+ ndbout << " " << buf << " file <filename> <node id> [<ver id>]"
+ << endl;
+}
+
+NDB_COMMAND(printConfig,
+ "printConfig", "printConfig", "Prints configuration", 16384){
+ if(argc < 4){
+ usage(argv[0]);
+ return 0;
+ }
+ if(strcmp("file", argv[1]) != 0 && strcmp("host", argv[1]) != 0){
+ usage(argv[0]);
+ return 0;
+ }
+
+ if(strcmp("host", argv[1]) == 0 && argc < 5){
+ usage(argv[0]);
+ return 0;
+ }
+
+ Properties * p = 0;
+ ConfigRetriever c;
+
+ if(strcmp("host", argv[1]) == 0){
+ int verId = 0;
+ if(argc > 5)
+ verId = atoi(argv[5]);
+
+ ndbout << "Getting config from: " << argv[2] << ":" << atoi(argv[3])
+ << " NodeId =" << atoi(argv[4])
+ << " VersionId = " << verId << endl;
+
+ p = c.getConfig(argv[2],
+ atoi(argv[3]),
+ atoi(argv[4]),
+ verId);
+ } else if (strcmp("file", argv[1]) == 0){
+ int verId = 0;
+ if(argc > 4)
+ verId = atoi(argv[4]);
+
+ ndbout << "Getting config from: " << argv[2]
+ << " NodeId =" << atoi(argv[3])
+ << " VersionId = " << verId << endl;
+
+ p = c.getConfig(argv[2], atoi(argv[3]), verId);
+ }
+
+ if(p != 0){
+ p->print(stdout);
+ } else {
+ ndbout << "Configuration not found: " << c.getErrorString() << endl;
+ }
+
+ delete p;
+
+ return 0;
+}
diff --git a/ndb/src/common/portlib/Makefile b/ndb/src/common/portlib/Makefile
new file mode 100644
index 00000000000..a928fc1e6d7
--- /dev/null
+++ b/ndb/src/common/portlib/Makefile
@@ -0,0 +1,43 @@
+include .defs.mk
+
+DIRS :=
+
+ifeq ($(NDB_OS), SOFTOSE)
+DIRS += ose
+endif
+
+ifeq ($(NDB_OS), OSE)
+DIRS += ose
+endif
+
+ifeq ($(NDB_OS), SIMCELLO)
+DIRS += ose
+endif
+
+ifeq ($(NDB_OS), LINUX)
+DIRS += unix
+endif
+
+ifeq ($(NDB_OS), MACOSX)
+DIRS += unix
+endif
+
+ifeq ($(NDB_OS), SOLARIS)
+DIRS += unix
+endif
+
+ifeq ($(NDB_OS), SOLARIS6)
+DIRS += unix
+endif
+
+ifeq ($(NDB_OS), HPUX)
+DIRS += unix
+endif
+
+ifeq ($(NDB_OS), WIN32)
+DIRS += win32
+endif
+
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/common/portlib/memtest/Makefile b/ndb/src/common/portlib/memtest/Makefile
new file mode 100644
index 00000000000..716cdbdea82
--- /dev/null
+++ b/ndb/src/common/portlib/memtest/Makefile
@@ -0,0 +1,12 @@
+CC=gcc
+LD=$(CC)
+SOURCES=memtest.c
+OUTPUT=memtest
+all:
+ $(CC) $(SOURCES) -o $(OUTPUT)
+
+debug:
+ $(CC) -g $(SOURCES) -o $(OUTPUT)
+
+clean: rm -rf *.o
+ rm -rf core*
diff --git a/ndb/src/common/portlib/memtest/memtest.c b/ndb/src/common/portlib/memtest/memtest.c
new file mode 100644
index 00000000000..d23235b7aa2
--- /dev/null
+++ b/ndb/src/common/portlib/memtest/memtest.c
@@ -0,0 +1,245 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <stdio.h>
+#include <unistd.h>
+#include <sys/time.h>
+#include <sys/mman.h>
+long long getMilli();
+long long getMicro();
+void malloctest(int loopcount, int memsize, int touch);
+void freetest(int loopcount, int memsize);
+void mmaptest(int loopcount, int memsize, int touch);
+void unmaptest(int loopcount, int memsize);
+
+
+main(int argc, char ** argv)
+{
+
+ int loopcount;
+ int memsize;
+ if(argc < 4) {
+ printf("Usage: memtest X loopcount memsize(MB)\n");
+ printf("where X = \n");
+ printf("1 : malloc test \n");
+ printf("2 : mmap test \n");
+ printf("3 : malloc test + touch pages\n");
+ printf("4 : mmap test + touch pages\n");
+ printf("5 : malloc/free test \n");
+ printf("6 : mmap/munmap test \n");
+ printf("loopcount - number of loops\n");
+ printf("memsize - memory segment size to allocate in MB.\n");
+ exit(1);
+ }
+
+
+ loopcount = atoi(argv[2]);
+ memsize = atoi(argv[3]);
+ switch(atoi(argv[1])) {
+ case 1: malloctest(loopcount, memsize , 0 );
+ break;
+ case 2: mmaptest(loopcount, memsize,0);
+ break;
+ case 3: malloctest(loopcount, memsize,1);
+ break;
+ case 4: mmaptest(loopcount, memsize,1);
+ break;
+ case 5: freetest(loopcount, memsize);
+ break;
+ case 6: unmaptest(loopcount, memsize);
+ break;
+ default:
+ break;
+ }
+}
+
+long long getMilli() {
+ struct timeval tick_time;
+ gettimeofday(&tick_time, 0);
+
+ return
+ ((long long)tick_time.tv_sec) * ((long long)1000) +
+ ((long long)tick_time.tv_usec) / ((long long)1000);
+}
+
+long long getMicro(){
+ struct timeval tick_time;
+ int res = gettimeofday(&tick_time, 0);
+
+ long long secs = tick_time.tv_sec;
+ long long micros = tick_time.tv_usec;
+
+ micros = secs*1000000+micros;
+ return micros;
+}
+
+void malloctest(int loopcount, int memsize, int touch) {
+ long long start=0;
+ int total=0;
+ int i=0, j=0;
+ int size=memsize*1024*1024; //bytes;
+ float mean;
+ char * ptr =0;
+
+ printf("Staring malloctest ");
+ if(touch)
+ printf("with touch\n");
+ else
+ printf("\n");
+
+ start=getMicro();
+
+ for(i=0; i<loopcount; i++){
+ ptr=(char *)malloc((size_t)(size));
+ if(ptr==0) {
+ printf("failed to malloc!\n");
+ return;
+ }
+ if(touch) {
+ for(j=0; j<size; j=j+4096)
+ ptr[j]=1;
+ }
+ }
+ total=(int)(getMicro()-start);
+
+ mean=(float)((float)total/(float)loopcount);
+ printf("Total time malloc %d bytes: %2.3f microsecs loopcount %d touch %d \n",
+ size, mean,loopcount, touch);
+}
+
+
+void mmaptest(int loopcount, int memsize, int touch) {
+ long long start=0;
+ int total=0;
+ int i=0, j=0;
+ char * ptr;
+ int size=memsize*1024*1024; //bytes;
+ float mean;
+
+ printf("Staring mmaptest ");
+ if(touch)
+ printf("with touch \n");
+ else
+ printf("\n");
+
+ start=getMicro();
+ for(i=0; i<loopcount; i++){
+ ptr = mmap(0,
+ size,
+ PROT_READ|PROT_WRITE,
+ MAP_PRIVATE|MAP_ANONYMOUS,
+ 0,
+ 0);
+ if(ptr<0) {
+ printf("failed to mmap!\n");
+ return;
+ }
+
+ if(touch) {
+ for(j=0; j<size; j=j+4096)
+ ptr[j]=1;
+ }
+ }
+ total=(int)(getMicro()-start);
+ mean=(float)((float)total/(float)loopcount);
+ printf("Total time mmap %d bytes: %2.3f microsecs \n",size, mean);
+}
+
+
+void unmaptest(loopcount, memsize)
+{
+ long long start=0;
+ int total=0;
+ int i=0, j=0;
+ char * ptr;
+ int size=memsize*1024*1024; //bytes;
+ float mean;
+
+ printf("Staring munmap test (loopcount = 1 no matter what you prev. set)\n");
+
+ loopcount = 1;
+
+
+ for(i=0; i<loopcount; i++){
+ ptr =(char*) mmap(0,
+ size,
+ PROT_READ|PROT_WRITE,
+ MAP_PRIVATE|MAP_ANONYMOUS,
+ 0,
+ 0);
+ if(ptr<0) {
+ printf("failed to mmap!\n");
+ return;
+ }
+
+
+ for(j=0; j<size; j=j+1)
+ ptr[j]='1';
+ start=getMicro();
+ if(munmap(ptr, size)<0) {
+ printf("failed to munmap!\n");
+ return;
+ }
+
+ total=(int)(getMicro()-start);
+ /*
+ for(j=8192; j<size; j=j+4096) {
+
+ *(ptr+j)='1';
+ }
+
+ for(j=0; j<4096; j=j+4096) {
+ *(ptr+j)='1';
+ }
+
+ */
+ }
+ mean=(float)((float)total/(float)loopcount);
+ printf("Total time unmap %d bytes: %2.3f microsecs \n",size, mean);
+}
+
+void freetest(int loopcount, int memsize) {
+ long long start=0;
+ int total=0;
+ int i=0, j=0;
+ int size=memsize*1024*1024; //bytes;
+ float mean;
+ char * ptr =0;
+
+ loopcount = 1;
+ printf("Staring free test (loopcount = 1 no matter what you prev. set)\n");
+
+
+ for(i=0; i<loopcount; i++){
+ ptr=(char*)malloc((size_t)(size));
+ if(ptr==0) {
+ printf("failed to malloc!\n");
+ return;
+ }
+ for(j=0; j<size; j=j+4096)
+ ptr[j]='1';
+ start=getMicro();
+ free(ptr);
+ total=(int)(getMicro()-start);
+ }
+
+
+ mean=(float)((float)total/(float)loopcount);
+ printf("Total time free %d bytes: %2.3f microsecs loopcount %d \n",
+ size, mean,loopcount);
+}
diff --git a/ndb/src/common/portlib/memtest/munmaptest/Makefile b/ndb/src/common/portlib/memtest/munmaptest/Makefile
new file mode 100644
index 00000000000..ea8c5238d1c
--- /dev/null
+++ b/ndb/src/common/portlib/memtest/munmaptest/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE := ndbapitest
+BIN_TARGET := munmaptest
+
+
+SOURCES = munmaptest.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
diff --git a/ndb/src/common/portlib/memtest/munmaptest/munmaptest.cpp b/ndb/src/common/portlib/memtest/munmaptest/munmaptest.cpp
new file mode 100644
index 00000000000..9e396cd98ee
--- /dev/null
+++ b/ndb/src/common/portlib/memtest/munmaptest/munmaptest.cpp
@@ -0,0 +1,251 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+
+#include <NdbOut.hpp>
+#include <NdbThread.h>
+#include <NdbMutex.h>
+#include <NdbCondition.h>
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#include <NdbEnv.h>
+#include <NdbHost.h>
+#include <NdbMain.h>
+#include <getarg.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <string.h>
+#include <sys/time.h>
+#include <sys/mman.h>
+
+
+struct ThreadData
+{
+ char * mapAddr;
+ Uint32 mapSize;
+ Uint32 chunk;
+ Uint32 idx;
+
+};
+
+long long getMilli();
+long long getMicro();
+
+
+void* mapSegment(void * arg);
+void* unmapSegment(void * arg);
+
+
+void* mapSegment(void * arg) {
+
+ ThreadData * threadArgs;
+ long long start=0;
+ int total=0;
+ int id = *(int *)arg;
+ threadArgs = new ThreadData [1];
+ Uint32 size=5*1024*1024;
+ struct NdbThread* unmapthread_var;
+ void *status = 0;
+ int run = 1;
+ int max=0, min =100000000, sum=0;
+ while(run < 1001) {
+ start=getMicro();
+ char * ptr =(char*) mmap(0,
+ size,
+ PROT_READ|PROT_WRITE,
+ MAP_PRIVATE|MAP_ANONYMOUS,
+ 0,
+ 0);
+
+ total=(int)(getMicro()-start);
+
+ ndbout << "T" << id << ": mmap took : " << total << " microsecs. "
+ << " Run: " << run ;
+ ndbout_c(" mapped @ %p \n", ptr);
+
+ if(total>max)
+ max = total;
+ if(total<min)
+ min=total;
+
+ sum+=total;
+
+ if(ptr<0) {
+ ndbout << "failed to mmap!" << endl;
+ exit(1);
+ }
+
+
+ threadArgs[0].mapAddr = (char *)ptr;
+ threadArgs[0].mapSize = size;
+ threadArgs[0].chunk = 4096;
+ threadArgs[0].idx = 0;
+
+
+ for(Uint32 j=0; j<size; j=j+4096)
+ ptr[j]='1';
+
+ unmapthread_var = NdbThread_Create(unmapSegment, // Function
+ (void**)&threadArgs[0],// Arg
+ 32768, // Stacksize
+ (char*)"unmapthread", // Thread name
+ NDB_THREAD_PRIO_MEAN); // Thread prio
+
+
+ if(NdbThread_WaitFor(unmapthread_var, &status) != 0) {
+ ndbout << "test failed - exitting " << endl;
+ exit(1);
+ }
+ run++;
+ }
+
+ ndbout << "MAX: " << max << " MIN: " << min;
+ float mean = (float) ((float)sum/(float)run);
+ ndbout_c(" AVERAGE: %2.5f\n",mean);
+}
+
+
+
+void* unmapSegment(void * arg)
+{
+
+ char * freeAddr;
+ char * mapAddr;
+ ThreadData * threadData = (ThreadData*) arg;
+ int start=0;
+ int total=0;
+ Uint32 mapSize = threadData->mapSize;
+ Uint32 chunk = threadData->chunk;
+ mapAddr = threadData->mapAddr;
+
+
+
+ freeAddr = mapAddr+mapSize-chunk;
+ NdbSleep_MilliSleep(100);
+ for(Uint32 i=0;i<mapSize; i = i+chunk) {
+ start=getMicro();
+ if(munmap(freeAddr, chunk) < 0){
+ ndbout << "munmap failed" << endl;
+ exit(1);
+ }
+ total=(int)(getMicro()-start);
+ freeAddr = freeAddr - chunk;
+ NdbSleep_MilliSleep(10);
+ ndbout << "unmap 4096 bytes : " << total << "microsecs" << endl;
+ }
+ return NULL;
+}
+
+
+static int trash;
+static int segmentsize=1;
+
+
+static struct getargs args[] = {
+ { "trash", 't', arg_integer, &trash,
+ "trash the memory before (1 to trash 0 to not trash)", "trash"},
+ { "segment", 's', arg_integer, &segmentsize,
+ "segment size (in MB)", "segment"},
+};
+
+
+static const int num_args = sizeof(args) / sizeof(args[0]);
+
+NDB_MAIN(munmaptest) {
+
+ const char *progname = "munmaptest";
+ int optind = 0;
+
+ if(getarg(args, num_args, argc, argv, &optind)) {
+ arg_printusage(args, num_args, progname, "");
+ exit(1);
+ }
+
+ int size;
+ char * ptr;
+ if(trash) {
+ for(int i=0; i<100; i++) {
+ size=1+(int) (10.0*rand()/(RAND_MAX+1.0));
+ NdbSleep_MilliSleep(10);
+ ptr =(char*) mmap(0,
+ size*1024*1024,
+ PROT_READ|PROT_WRITE,
+ MAP_PRIVATE|MAP_ANONYMOUS,
+ 0,
+ 0);
+ for(int i=0;i<(size*1024*1024); i=i+4096) {
+ *(ptr+i)='1';
+ }
+ NdbSleep_MilliSleep(10);
+
+ munmap(ptr,size);
+
+ }
+
+
+ }
+
+ int noThreads = 1;
+ struct NdbThread* mapthread_var;
+ int id[noThreads];
+ void *status=0;
+
+ ThreadData * threadArgs = new ThreadData[noThreads];
+
+
+
+
+ for(int i=0; i < noThreads; i++) {
+ threadArgs[i].mapSize = segmentsize*1024*1024;
+ threadArgs[i].idx = i;
+ mapthread_var = NdbThread_Create(mapSegment, // Function
+ (void**)&threadArgs[i],// Arg
+ 32768, // Stacksize
+ (char*)"mapthread", // Thread name
+ NDB_THREAD_PRIO_MEAN); // Thread prio
+
+ }
+
+
+ if(NdbThread_WaitFor(mapthread_var, &status) != 0) {
+ ndbout << "test failed - exitting " << endl;
+ exit(1);
+ }
+
+}
+
+long long getMilli() {
+ struct timeval tick_time;
+ gettimeofday(&tick_time, 0);
+
+ return
+ ((long long)tick_time.tv_sec) * ((long long)1000) +
+ ((long long)tick_time.tv_usec) / ((long long)1000);
+}
+
+long long getMicro(){
+ struct timeval tick_time;
+ int res = gettimeofday(&tick_time, 0);
+
+ long long secs = tick_time.tv_sec;
+ long long micros = tick_time.tv_usec;
+
+ micros = secs*1000000+micros;
+ return micros;
+}
diff --git a/ndb/src/common/portlib/mmstest/mmslist.cpp b/ndb/src/common/portlib/mmstest/mmslist.cpp
new file mode 100644
index 00000000000..bd00211445c
--- /dev/null
+++ b/ndb/src/common/portlib/mmstest/mmslist.cpp
@@ -0,0 +1,103 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <NdbMain.h>
+
+#include <ose.h>
+#include <mms.sig>
+#include <mms_err.h>
+#include <string.h>
+#include <stdio.h>
+#include <NdbOut.hpp>
+
+/**
+ * NOTE: To use NdbMem from a OSE system ose_mms has to be defined
+ * as a "Required External Process"(see OSE Kernel User's Guide/R1.1(p. 148)),
+ * like this:
+ * EXT_PROC(ose_mms, ose_mms, 50000)
+ * This will create a global variable ose_mms_ that is used from here.
+ */
+
+union SIGNAL
+{
+ SIGSELECT sigNo;
+ struct MmsListDomainRequest mmsListDomainRequest;
+ struct MmsListDomainReply mmsListDomainReply;
+}; /* union SIGNAL */
+
+extern PROCESS ose_mms_;
+
+struct ARegion
+{
+ unsigned long int address;
+ unsigned long int size;
+ char name[32];
+
+ U32 resident; /* Boolean, nonzero if resident. */
+ U32 access; /* See values for AccessType (above) .*/
+ U32 type; /* either RAM-mem (1) or Io-mem (2) */
+ U32 cache; /* 0-copyback,1-writethrough, 2-CacheInhibit.*/
+};
+
+NDB_COMMAND(mmslist, "mmslist", "mmslist", "LIst the MMS memory segments", 4096){
+ if (argc == 1){
+
+ static SIGSELECT allocate_sig[] = {1,MMS_LIST_DOMAIN_REPLY};
+ union SIGNAL *sig;
+
+ /* Send request to list all segments and regions. */
+ sig = alloc(sizeof(struct MmsListDomainRequest),
+ MMS_LIST_DOMAIN_REQUEST);
+ send(&sig, ose_mms_);
+
+ while (true){
+ sig = receive(allocate_sig);
+ if (sig != NIL){
+ if (sig->mmsListDomainReply.status == MMS_SUCCESS){
+ /* Print domain info */
+ ndbout << "=================================" << endl;
+ ndbout << "domain: " << sig->mmsListDomainReply.domain << endl;
+ ndbout << "name : " << sig->mmsListDomainReply.name << endl;
+ ndbout << "used : " << sig->mmsListDomainReply.used << endl;
+ ndbout << "lock : " << sig->mmsListDomainReply.lock << endl;
+ ndbout << "numOfRegions:" << sig->mmsListDomainReply.numOfRegions << endl;
+ struct ARegion * tmp = (struct ARegion*)&sig->mmsListDomainReply.regions[0];
+ for (int i = 0; i < sig->mmsListDomainReply.numOfRegions && i < 256; i++){
+ ndbout << i << ": adress=" << tmp->address <<
+ ", size=" << tmp->size <<
+ ", name=" << tmp->name <<
+ ", resident=" << tmp->resident <<
+ ", access=" << tmp->access <<
+ ", type=" << tmp->type <<
+ ", cache=" << tmp->cache << endl;
+ tmp++;
+ }
+
+ free_buf(&sig);
+ }else{
+ free_buf(&sig);
+ break;
+ }
+ }
+
+ }
+
+ }else{
+ ndbout << "Usage: mmslist" << endl;
+ }
+ return NULL;
+}
diff --git a/ndb/src/common/portlib/mmstest/mmstest.cpp b/ndb/src/common/portlib/mmstest/mmstest.cpp
new file mode 100644
index 00000000000..6ebb5064aaf
--- /dev/null
+++ b/ndb/src/common/portlib/mmstest/mmstest.cpp
@@ -0,0 +1,76 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include "NdbThread.h"
+#include <NdbMem.h>
+#include <NdbMain.h>
+
+#include <stdlib.h>
+
+NDB_COMMAND(ndbmem, "ndbmem", "ndbmem", "Test the ndbmem functionality", 4096){
+
+ ndbout << "Starting test of NdbMem" << endl;
+ ndbout << "=======================" << endl;
+
+ ndbout << "Creating NdbMem" << endl;
+ NdbMem_Create();
+
+
+ ndbout << "NdbMem - test 1" << endl;
+ if (argc == 2){
+ int size1 = atoi(argv[1]);
+ ndbout << "Allocate and test "<<size1<<" bytes of memory" << endl;
+ char* mem1 = (char*)NdbMem_Allocate(size1);
+ ndbout << "mem1 = " << hex << (int)mem1 << endl;
+ if (mem1 != NULL){
+ char* p1;
+
+ // Write to the memory allocated
+ p1 = mem1;
+ for(int i = 0; i < size1; i++){
+ *p1 = (char)(i%256);
+ p1++;
+ }
+
+ // Read from the memory and check value
+ char read1;
+ char* pread1;
+ pread1 = mem1;
+ for(int i = 0; i < size1; i++){
+ read1 = *pread1;
+ //ndbout << i << "=" << read1 << endl;
+ if (read1 != (i%256))
+ ndbout << "Byte " << i << " was not correct, read1=" << read1 << endl;
+ pread1++;
+ }
+
+ ndbout << "Freeing NdbMem" << endl;
+ NdbMem_Free(mem1);
+ }
+
+ ndbout << "Destroying NdbMem" << endl;
+ NdbMem_Destroy();
+ }else{
+ ndbout << "Usage: ndbmem <size(bytes)>"<< endl;
+ }
+
+ return NULL;
+
+}
+
+
+
diff --git a/ndb/src/common/portlib/ose/Makefile b/ndb/src/common/portlib/ose/Makefile
new file mode 100644
index 00000000000..4ef93b7824a
--- /dev/null
+++ b/ndb/src/common/portlib/ose/Makefile
@@ -0,0 +1,31 @@
+include .defs.mk
+
+TYPE :=
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := portlib
+
+SOURCES = NdbOut.cpp
+
+SOURCES.c = NdbCondition.c \
+ NdbMutex.c \
+ NdbSleep.c \
+ NdbTick.c \
+ NdbEnv.c \
+ NdbThread.c \
+ NdbHost.c \
+ NdbTCP.c
+
+ifeq ($(NDB_OS), SOFTOSE)
+ SOURCES += NdbMem_SoftOse.cpp
+else
+ SOURCES.c += NdbMem.c
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
+
diff --git a/ndb/src/common/portlib/ose/NdbCondition.c b/ndb/src/common/portlib/ose/NdbCondition.c
new file mode 100644
index 00000000000..2ab6e49006b
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbCondition.c
@@ -0,0 +1,244 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbCondition.h"
+#include <pthread.h>
+#include <assert.h>
+#include <sys/types.h>
+#include <malloc.h>
+
+#include <NdbMutex.h>
+
+#include "NdbConditionOSE.h"
+struct NdbCondition
+{
+ PROCESS condserv_pid;
+};
+
+
+OS_PROCESS(ndbcond_serv){
+
+ union SIGNAL* sig;
+ union SIGNAL* sig2;
+
+ static const SIGSELECT sel_signal[] = {2, NDBCOND_SIGNAL, NDBCOND_BROADCAST};
+ static const SIGSELECT sel_cond[] = {2, NDBCOND_WAIT, NDBCOND_WAITTIMEOUT};
+
+ for(;;){
+ /* Receive condition wait signal */
+ sig = receive((SIGSELECT*)sel_cond);
+ if (sig != NIL){
+ switch (sig->sigNo){
+
+ case NDBCOND_WAIT:
+ /* Wait for a SIGNAL or BROADCAST from anyone */
+ sig2 = receive((SIGSELECT*)sel_signal);
+ if (sig2 != NIL){
+ switch(sig2->sigNo){
+
+ case NDBCOND_SIGNAL:
+ ((struct NdbCondWait*)sig)->status = NDBCOND_SIGNALED;
+ /* Send signal back to the one waiting for this condition */
+ send(&sig, sender(&sig));
+ break;
+ case NDBCOND_BROADCAST:
+ /* Not handled yet */
+ assert(1==0);
+ break;
+ default:
+ assert(1==0);
+ break;
+ }
+ free_buf(&sig2);
+ }
+ break;
+
+ case NDBCOND_WAITTIMEOUT:
+ /* Wait for a SIGNAL or BROADCAST from anyone */
+ sig2 = receive_w_tmo(((struct NdbCondWaitTimeout*)sig)->timeout, (SIGSELECT*)sel_signal);
+ if (sig2 != NIL){
+ switch(sig2->sigNo){
+
+ case NDBCOND_SIGNAL:
+ ((struct NdbCondWaitTimeout*)sig)->status = NDBCOND_SIGNALED;
+ /* Send signal back to the one waiting for this condition */
+ send(&sig, sender(&sig));
+ break;
+ case NDBCOND_BROADCAST:
+ /* Not handled yet */
+ assert(1==0);
+ break;
+ default:
+ assert(1==0);
+ break;
+ }
+ free_buf(&sig2);
+ }else{
+ ((struct NdbCondWaitTimeout*)sig)->status = NDBCOND_TIMEOUT;
+ send(&sig, sender(&sig));
+ }
+ break;
+
+ default:
+ assert(1==0);
+ break;
+
+ }
+ }
+
+ }
+}
+
+
+struct NdbCondition*
+NdbCondition_Create(void)
+{
+ struct NdbCondition* tmpCond;
+
+
+ tmpCond = (struct NdbCondition*)malloc(sizeof(struct NdbCondition));
+
+ if (tmpCond == NULL)
+ return NULL;
+
+ /**
+ * Start this process with a quite high
+ * priority, we want it to be responsive
+ */
+ tmpCond->condserv_pid = create_process(OS_PRI_PROC, /* Process type */
+ "ndbcond_serv", /* Name */
+ ndbcond_serv, /* Entry point */
+ 2048, /* Stack size */
+ 10, /* Priority */
+ 0, /* Time slice */
+ get_bid(current_process()), /* Block */
+ NULL, /* Redir table */
+ 0,
+ 0);
+
+ start(tmpCond->condserv_pid);
+
+ return tmpCond;
+}
+
+
+int
+NdbCondition_Wait(struct NdbCondition* p_cond,
+ NdbMutex* p_mutex)
+{
+ static const SIGSELECT sel_cond[] = {1, NDBCOND_WAIT};
+ union SIGNAL* sig;
+ int result;
+ if (p_cond == NULL || p_mutex == NULL)
+ return 0;
+
+ sig = alloc(sizeof(struct NdbCondWait), NDBCOND_WAIT);
+ send(&sig, p_cond->condserv_pid);
+
+ NdbMutex_Unlock(p_mutex);
+
+ result = 1;
+ while(NIL == (sig = receive_from((OSTIME)-1, (SIGSELECT*)sel_cond, p_cond->condserv_pid)));
+ if (sig != NIL){
+ if (sig->sigNo == NDBCOND_WAIT){
+ /* Condition is signaled */
+ result = 0;
+ }else{
+ assert(1==0);
+ }
+ free_buf(&sig);
+
+ }
+ NdbMutex_Lock(p_mutex);
+
+ return result;
+}
+
+
+int
+NdbCondition_WaitTimeout(struct NdbCondition* p_cond,
+ NdbMutex* p_mutex,
+ int msecs){
+ static const SIGSELECT sel_cond[] = {1, NDBCOND_WAITTIMEOUT};
+ union SIGNAL* sig;
+ int result;
+ if (p_cond == NULL || p_mutex == NULL)
+ return 0;
+
+ sig = alloc(sizeof(struct NdbCondWaitTimeout), NDBCOND_WAITTIMEOUT);
+ ((struct NdbCondWaitTimeout*)sig)->timeout = msecs;
+ send(&sig, p_cond->condserv_pid);
+
+ NdbMutex_Unlock(p_mutex);
+
+ result = 1;
+ while(NIL == (sig = receive_from((OSTIME)-1, (SIGSELECT*)sel_cond, p_cond->condserv_pid)));
+ if (sig != NIL){
+ if (sig->sigNo == NDBCOND_WAITTIMEOUT){
+ /* Condition is signaled */
+ result = 0;
+ }else{
+ assert(1==0);
+ }
+ free_buf(&sig);
+
+ }
+
+ NdbMutex_Lock(p_mutex);
+
+ return result;
+}
+
+
+int
+NdbCondition_Signal(struct NdbCondition* p_cond){
+
+ union SIGNAL* sig;
+ if (p_cond == NULL)
+ return 1;
+
+ sig = alloc(sizeof(struct NdbCondSignal), NDBCOND_SIGNAL);
+ send(&sig, p_cond->condserv_pid);
+
+ return 0;
+}
+
+
+int NdbCondition_Broadcast(struct NdbCondition* p_cond)
+{
+ union SIGNAL* sig;
+ if (p_cond == NULL)
+ return 1;
+
+ sig = alloc(sizeof(struct NdbCondBroadcast), NDBCOND_BROADCAST);
+ send(&sig, p_cond->condserv_pid);
+
+ return 0;
+}
+
+
+int NdbCondition_Destroy(struct NdbCondition* p_cond)
+{
+ if (p_cond == NULL)
+ return 1;
+
+ kill_proc(p_cond->condserv_pid);
+ free(p_cond);
+
+ return 0;
+}
+
diff --git a/ndb/src/common/portlib/ose/NdbConditionOSE.h b/ndb/src/common/portlib/ose/NdbConditionOSE.h
new file mode 100644
index 00000000000..bd0306261cc
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbConditionOSE.h
@@ -0,0 +1,103 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_CONDITIONOSE_H
+#define NDB_CONDITIONOSE_H
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+#define NDBCOND_SIGBASE 4000
+
+#define NDBCOND_WAIT (NDBCOND_SIGBASE + 1) /* !-SIGNO(struct NdbCondWait)-! */
+#define NDBCOND_WAITTIMEOUT (NDBCOND_SIGBASE + 2) /* !-SIGNO(struct NdbCondWaitTimeOut)-! */
+#define NDBCOND_SIGNAL (NDBCOND_SIGBASE + 3) /* !-SIGNO(struct NdbCondSignal)-! */
+#define NDBCOND_BROADCAST (NDBCOND_SIGBASE + 4) /* !-SIGNO(struct NdbCondBroadcast)-! */
+
+
+const char *
+sigNo2String(SIGSELECT sigNo){
+ switch(sigNo){
+ case NDBCOND_WAIT:
+ return "NDBCOND_WAIT";
+ break;
+ case NDBCOND_WAITTIMEOUT:
+ return "NDBCOND_WAITTIMEOUT";
+ break;
+ case NDBCOND_SIGNAL:
+ return "NDBCOND_SIGNAL";
+ break;
+ case NDBCOND_BROADCAST:
+ return "NDBCOND_BROADCAST";
+ break;
+ }
+ return "UNKNOWN";
+}
+
+struct NdbCondWait
+{
+ SIGSELECT sigNo;
+ int status;
+};
+
+/**
+ * Signal received
+ */
+#define NDBCOND_SIGNALED 1
+
+/**
+ * Timeout occured
+ */
+#define NDBCOND_TIMEOUT 2
+
+struct NdbCondWaitTimeout
+{
+ SIGSELECT sigNo;
+ int timeout;
+ int status;
+
+};
+
+struct NdbCondSignal
+{
+ SIGSELECT sigNo;
+};
+
+struct NdbCondBroadcast
+{
+ SIGSELECT sigNo;
+};
+
+
+union SIGNAL
+{
+ SIGSELECT sigNo;
+ struct NdbCondWait condWait;
+ struct NdbCondWaitTimeout condWaitTimeout;
+ struct NdbCondSignal condSignal;
+ struct NdbCondBroadcast condBroadcast;
+};
+
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/ndb/src/common/portlib/ose/NdbEnv.c b/ndb/src/common/portlib/ose/NdbEnv.c
new file mode 100644
index 00000000000..e2ac4d879d2
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbEnv.c
@@ -0,0 +1,55 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbEnv.h"
+#include <string.h>
+#include <stdlib.h>
+
+const char* NdbEnv_GetEnv(const char* name, char * buf, int buflen)
+{
+ /**
+ * All environment variables are associated with a process
+ * it's important to read env from the correct process
+ * for now read from own process, own block and last the "ose_shell" process.
+ *
+ * TODO! What process should this be read from in the future?
+ *
+ */
+ PROCESS proc_;
+ char* p = NULL;
+ /* Look in own process */
+ p = get_env(current_process(), (char*)name);
+ if (p == NULL){
+ /* Look in block process */
+ p = get_env(get_bid(current_process()), (char*)name);
+ if (p == NULL){
+ /* Look in ose_shell process */
+ if (hunt("ose_shell", 0, &proc_, NULL)){
+ p = get_env(proc_, (char*)name);
+ }
+ }
+ }
+
+ if (p != NULL){
+ strncpy(buf, p, buflen);
+ buf[buflen-1] = 0;
+ free_buf((union SIGNAL **)&p);
+ p = buf;
+ }
+ return p;
+}
+
diff --git a/ndb/src/common/portlib/ose/NdbHost.c b/ndb/src/common/portlib/ose/NdbHost.c
new file mode 100644
index 00000000000..f5e1e511c16
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbHost.c
@@ -0,0 +1,55 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbHost.h"
+#include <unistd.h>
+
+
+#include <inet.sig>
+#include <string.h>
+
+union SIGNAL
+{
+ SIGSELECT sigNo;
+ struct InetIfUp inetIfUp;
+};
+
+int NdbHost_GetHostName(char* buf)
+{
+#if 0
+ extern PROCESS ose_inet_;
+ union SIGNAL *signal;
+ static const SIGSELECT select_if_up_reply[] = { 1, INET_IF_UP_REPLY };
+
+ signal = alloc(sizeof(struct InetIfUp), INET_IF_UP_REQUEST);
+ strcpy(signal->inetIfUp.ifName, "*");
+ send((union SIGNAL **)&signal, ose_inet_);
+ signal = receive((SIGSELECT *)select_if_up_reply);
+ strcpy(buf, signal->inetIfUp.ifName);
+ free_buf(&signal);
+ return 0;
+#else
+ return -1;
+#endif
+}
+
+
+int NdbHost_GetProcessId(void)
+{
+ return current_process();
+}
+
diff --git a/ndb/src/common/portlib/ose/NdbMem.c b/ndb/src/common/portlib/ose/NdbMem.c
new file mode 100644
index 00000000000..6d922e4c073
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbMem.c
@@ -0,0 +1,183 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbMem.h"
+
+#include <assert.h>
+
+#if defined NDB_OSE
+#include <ose.h>
+#include <mms.sig>
+#include <mms_err.h>
+#include <string.h>
+#include <stdio.h>
+#include <NdbOut.hpp>
+
+// Page size for mp750 is 4096 bytes.
+#define PAGE_SIZE 4096
+
+/**
+ * NOTE: To use NdbMem from a OSE system ose_mms has to be defined
+ * as a "Required External Process"(see OSE Kernel User's Guide/R1.1(p. 148)),
+ * like this in osemain.con:
+ * EXT_PROC(ose_mms, ose_mms, 50000)
+ * This will create a global variable ose_mms_ that is used from here.
+ */
+
+union SIGNAL
+{
+ SIGSELECT sigNo;
+ struct MmsAllocateRegionRequest mmsAllocateRegionRequest;
+ struct MmsAllocateRegionReply mmsAllocateRegionReply;
+ struct MmsFreeRegionRequest mmsFreeRegionRequest;
+ struct MmsFreeRegionReply mmsFreeRegionReply;
+}; /* union SIGNAL */
+
+extern PROCESS ose_mms_;
+
+void NdbMem_Create(void)
+{
+ /* Do nothing */
+ return;
+}
+
+void NdbMem_Destroy(void)
+{
+ /* Do nothing */
+ return;
+}
+
+void* NdbMem_Allocate(size_t size)
+{
+ static SIGSELECT allocate_sig[] = {1,MMS_ALLOCATE_REGION_REPLY};
+ union SIGNAL *sig;
+ U32 allocatedAdress;
+
+ assert(size > 0);
+
+ // Only allowed to allocate multiples of the page size.
+ if(size % PAGE_SIZE != 0) {
+ size += PAGE_SIZE - size%PAGE_SIZE;
+ }
+
+ /* Allocate a new region in the callers memory segment. */
+ sig = alloc(sizeof(struct MmsAllocateRegionRequest),
+ MMS_ALLOCATE_REGION_REQUEST);
+ /* -1: The callers domain is used */
+ sig->mmsAllocateRegionRequest.domain = (MemoryDomain)-1;
+ sig->mmsAllocateRegionRequest.useAddr = False;
+ sig->mmsAllocateRegionRequest.size = size;
+ sig->mmsAllocateRegionRequest.access = SuperRW_UserRW;
+ sig->mmsAllocateRegionRequest.resident = False;
+ sig->mmsAllocateRegionRequest.memory = CodeData;
+ sig->mmsAllocateRegionRequest.cache = CopyBack;
+ strcpy(sig->mmsAllocateRegionRequest.name, "NDB_DATA");
+ send(&sig, ose_mms_);
+ sig = receive(allocate_sig);
+
+ if (sig->mmsAllocateRegionReply.status != MMS_SUCCESS){
+ /* Memory allocation failed, make sure this function returns NULL */
+ allocatedAdress = NULL;
+ }
+ else{
+ allocatedAdress = sig->mmsAllocateRegionReply.address;
+ }
+ free_buf(&sig);
+ return (void*)allocatedAdress;
+}
+
+void* NdbMem_AllocateAlign(size_t size, size_t alignment)
+{
+ return NdbMem_Allocate(size);
+}
+
+
+void NdbMem_Free(void* ptr)
+{
+ static SIGSELECT free_sig[] = {1,MMS_FREE_REGION_REPLY};
+ union SIGNAL *sig;
+
+ /* Free a region in the callers domain. */
+ sig = alloc(sizeof(struct MmsFreeRegionRequest),
+ MMS_FREE_REGION_REQUEST);
+ sig->mmsFreeRegionRequest.address = (U32)ptr;
+ send(&sig, ose_mms_);
+ sig = receive(free_sig);
+
+ if (sig->mmsFreeRegionReply.status != MMS_SUCCESS){
+ ndbout_c("The MMS Region could not be deallocated.\r\n");
+ error(sig->mmsFreeRegionReply.status);
+ };
+ free_buf(&sig);
+}
+
+int NdbMem_MemLockAll(){
+ return -1;
+}
+
+int NdbMem_MemUnlockAll(){
+ return -1;
+}
+
+#else
+#include <assert.h>
+#include <stdlib.h>
+
+
+void NdbMem_Create()
+{
+ /* Do nothing */
+ return;
+}
+
+void NdbMem_Destroy()
+{
+ /* Do nothing */
+ return;
+}
+
+void* NdbMem_Allocate(size_t size)
+{
+ assert(size > 0);
+ return (void*)malloc(size);
+}
+
+void* NdbMem_AllocateAlign(size_t size, size_t alignment)
+{
+ /*
+ return (void*)memalign(alignment, size);
+ TEMP fix
+ */
+ return (void*)malloc(size);
+}
+
+
+void NdbMem_Free(void* ptr)
+{
+ free(ptr);
+}
+
+
+int NdbMem_MemLockAll(){
+ return -1;
+}
+
+int NdbMem_MemUnlockAll(){
+ return -1;
+}
+
+#endif
diff --git a/ndb/src/common/portlib/ose/NdbMem_SoftOse.cpp b/ndb/src/common/portlib/ose/NdbMem_SoftOse.cpp
new file mode 100644
index 00000000000..cad22c0474b
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbMem_SoftOse.cpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbMem.h"
+
+extern "C"
+void NdbMem_Create()
+{
+}
+extern "C"
+void NdbMem_Destroy()
+{
+}
+
+extern "C"
+void* NdbMem_Allocate(size_t size)
+{
+ return new char[size];
+}
+
+extern "C"
+void* NdbMem_AllocateAlign(size_t size, size_t alignment)
+{
+ return NdbMem_Allocate(size);
+}
+
+extern "C"
+void NdbMem_Free(void* ptr)
+{
+ delete [] (char *)(ptr);
+}
+
+int NdbMem_MemLockAll(){
+ return -1;
+}
+
+int NdbMem_MemUnlockAll(){
+ return -1;
+}
+
diff --git a/ndb/src/common/portlib/ose/NdbMutex.c b/ndb/src/common/portlib/ose/NdbMutex.c
new file mode 100644
index 00000000000..859ddefd536
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbMutex.c
@@ -0,0 +1,86 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbMutex.h"
+
+#include <pthread.h>
+#include <stdlib.h>
+#include <assert.h>
+
+
+NdbMutex* NdbMutex_Create(void)
+{
+ NdbMutex* pNdbMutex;
+
+ pNdbMutex = create_sem(1);
+
+ return pNdbMutex;
+}
+
+
+int NdbMutex_Destroy(NdbMutex* p_mutex)
+{
+
+ if (p_mutex == NULL)
+ return -1;
+
+ kill_sem(p_mutex);
+
+ return 0;
+
+}
+
+
+int NdbMutex_Lock(NdbMutex* p_mutex)
+{
+ if (p_mutex == NULL)
+ return -1;
+
+ wait_sem(p_mutex);
+
+ return 0;
+}
+
+
+int NdbMutex_Unlock(NdbMutex* p_mutex)
+{
+
+ if (p_mutex == NULL)
+ return -1;
+
+ signal_sem(p_mutex);
+
+ return 0;
+}
+
+
+int NdbMutex_Trylock(NdbMutex* p_mutex)
+{
+ int result = -1;
+
+ if (p_mutex != NULL) {
+ OSSEMVAL semvalue = get_sem(p_mutex);
+ if (semvalue > 0) {
+ wait_sem(p_mutex);
+ result = 0;
+ }
+ }
+
+ return result;
+
+}
+
diff --git a/ndb/src/common/portlib/ose/NdbOut.cpp b/ndb/src/common/portlib/ose/NdbOut.cpp
new file mode 100644
index 00000000000..0ee12249ff5
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbOut.cpp
@@ -0,0 +1,99 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbOut.hpp"
+#include <NdbStdio.h>
+#include <stdarg.h>
+#include <NdbUnistd.h>
+#include <string.h>
+
+
+#if defined NDB_SOFTOSE
+#include <dbgprintf.h>
+#define printfunc dbgprintf
+#else
+#define printfunc printf
+#endif
+
+static char const* const endlineString = "\r\n";
+
+static int CtrlC = 0;
+NdbOut ndbout;
+
+
+NdbOut& NdbOut::operator<<(int aVal)
+{
+ char* format;
+ char HexFormat[] = "0x%08x";
+ char DecFormat[] = "%d";
+ if (isHexFormat == 1)
+ format = HexFormat;
+ else
+ format = DecFormat;
+
+ printfunc(format, aVal);
+ return *this;
+}
+
+NdbOut& NdbOut::operator<<(char* pVal)
+{
+ printfunc("%s", pVal);
+ return *this;
+}
+
+NdbOut& NdbOut::endline()
+{
+ isHexFormat = 0; // Reset hex to normal, if user forgot this
+ printfunc(endlineString);
+ return *this;
+}
+
+NdbOut& NdbOut::flushline()
+{
+ isHexFormat = 0; // Reset hex to normal, if user forgot this
+ return *this;
+}
+
+NdbOut& NdbOut::setHexFormat(int _format)
+{
+ isHexFormat = _format;
+ return *this;
+}
+
+NdbOut::NdbOut()
+{
+ CtrlC = 0;
+ isHexFormat = 0;
+}
+
+NdbOut::~NdbOut()
+{
+}
+
+
+
+extern "C"
+void
+ndbout_c(const char * fmt, ...){
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ ndbout << buf << endl;
+ va_end(ap);
+}
diff --git a/ndb/src/common/portlib/ose/NdbSleep.c b/ndb/src/common/portlib/ose/NdbSleep.c
new file mode 100644
index 00000000000..70fd83117ef
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbSleep.c
@@ -0,0 +1,36 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbSleep.h"
+
+#include <ose.h>
+
+
+int
+NdbSleep_MilliSleep(int milliseconds){
+ const OSTIME millisecond_delay = milliseconds;
+ delay(millisecond_delay);
+ return 0;
+}
+
+int
+NdbSleep_SecSleep(int seconds){
+ const OSTIME millisecond_delay = seconds*1000;
+ delay(millisecond_delay);
+ return 0;
+}
+
diff --git a/ndb/src/common/portlib/ose/NdbTCP.c b/ndb/src/common/portlib/ose/NdbTCP.c
new file mode 100644
index 00000000000..9994697b3f8
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbTCP.c
@@ -0,0 +1,38 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbTCP.h"
+
+
+int
+Ndb_getInAddr(struct in_addr * dst, const char *address) {
+ struct hostent * host;
+ host = gethostbyname_r(address);
+ if(host != 0){
+ dst->s_addr = ((struct in_addr *) *host->h_addr_list)->s_addr;
+ free_buf((union SIGNAL **)&host);
+ return 0;
+ }
+ /* Try it as aaa.bbb.ccc.ddd. */
+ dst->s_addr = inet_addr(address);
+ if (dst->s_addr != INADDR_NONE) {
+ return 0;
+ }
+ return -1;
+}
+
+
diff --git a/ndb/src/common/portlib/ose/NdbThread.c b/ndb/src/common/portlib/ose/NdbThread.c
new file mode 100644
index 00000000000..41a5f181c40
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbThread.c
@@ -0,0 +1,184 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbThread.h"
+#include <pthread.h>
+#include <malloc.h>
+#include <assert.h>
+#include <string.h>
+#include <NdbOut.hpp>
+
+#define MAX_THREAD_NAME 16
+
+
+struct NdbThread
+{
+ PROCESS pid;
+ char thread_name[MAX_THREAD_NAME];
+};
+
+#define NDBTHREAD_SIGBASE 4010
+
+#define NDBTHREAD_START (NDBTHREAD_SIGBASE + 1) /* !-SIGNO(struct NdbThreadStart)-! */
+
+struct NdbThreadStart
+{
+ SIGSELECT sigNo;
+ NDB_THREAD_FUNC* func;
+ NDB_THREAD_ARG arg;
+};
+
+struct NdbThreadStopped
+{
+ SIGSELECT sigNo;
+};
+
+union SIGNAL
+{
+ SIGSELECT sigNo;
+ struct NdbThreadStart threadStart;
+ struct NdbThreadStopped threadStopped;
+};
+
+OS_PROCESS(thread_starter){
+ static const SIGSELECT sel_start[] = {1, NDBTHREAD_START};
+ struct NdbThreadStart* sigstart;
+ union SIGNAL* sig;
+
+ /* Receive function adress and params */
+ sig = receive((SIGSELECT*)sel_start);
+ if (sig != NIL){
+ if (sig->sigNo == NDBTHREAD_START){
+ sigstart = ((struct NdbThreadStart*)sig);
+ /* Execute function with arg */
+ (*sigstart->func)(sigstart->arg);
+ }else{
+ assert(1==0);
+ }
+ free_buf(&sig);
+ }
+}
+
+struct NdbThread* NdbThread_Create(NDB_THREAD_FUNC* p_thread_func,
+ NDB_THREAD_ARG *p_thread_arg,
+ const NDB_THREAD_STACKSIZE thread_stack_size,
+ const char* p_thread_name,
+ NDB_THREAD_PRIO thread_prio)
+{
+ struct NdbThread* tmpThread;
+ union SIGNAL* sig;
+ int ose_prio;
+
+ if (p_thread_func == NULL)
+ return 0;
+
+ tmpThread = (struct NdbThread*)malloc(sizeof(struct NdbThread));
+ if (tmpThread == NULL)
+ return NULL;
+
+ strncpy((char*)&tmpThread->thread_name, p_thread_name, MAX_THREAD_NAME);
+
+ switch(thread_prio){
+ case NDB_THREAD_PRIO_HIGHEST:
+ ose_prio = 1;
+ break;
+ case NDB_THREAD_PRIO_HIGH:
+ ose_prio = 10;
+ break;
+ case NDB_THREAD_PRIO_MEAN:
+ ose_prio = 16;
+ break;
+ case NDB_THREAD_PRIO_LOW:
+ ose_prio = 23;
+ break;
+ case NDB_THREAD_PRIO_LOWEST:
+ ose_prio = 31;
+ break;
+ default:
+ return NULL;
+ break;
+ }
+
+ /* Create process */
+ tmpThread->pid = create_process(OS_PRI_PROC, /* Process type */
+ (char*)p_thread_name, /* Name */
+ thread_starter, /* Entry point */
+ thread_stack_size, /* Stack size */
+ ose_prio, /* Priority */
+ 0, /* Time slice */
+ get_bid(current_process()), /* Block */
+ NULL, /* Redir table */
+ 0,
+ 0);
+
+ /* Send params to process */
+ sig = alloc(sizeof(struct NdbThreadStart), NDBTHREAD_START);
+ ((struct NdbThreadStart*)sig)->func = p_thread_func;
+ ((struct NdbThreadStart*)sig)->arg = p_thread_arg;
+ send(&sig, tmpThread->pid);
+
+ /* Enable NDB_HOME environment variable for the thread */
+ {
+ /* Hardcoded NDB_HOME...*/
+ char* ndb_home_env = get_env(current_process(), "NDB_HOME");
+ if (ndb_home_env != NULL)
+ {
+ /* Set NDB_HOME */
+ int rc = set_env(tmpThread->pid, "NDB_HOME", ndb_home_env);
+ if (rc != 0)
+ {
+ /* Not really a problem */
+ }
+ } /* Enable NDB_HOME */
+ }
+
+ /* Start process */
+ start(tmpThread->pid);
+
+ return tmpThread;
+}
+
+
+
+void NdbThread_Destroy(struct NdbThread** p_thread)
+{
+ free(* p_thread); * p_thread = 0;
+}
+
+
+int NdbThread_WaitFor(struct NdbThread* p_wait_thread, void** status)
+{
+ while(hunt(p_wait_thread->thread_name, 0, NULL, NULL) != 0)
+ delay(1000);
+
+ * status = 0;
+
+ return 0;
+}
+
+
+void NdbThread_Exit(int a)
+{
+ kill_proc(current_process());
+}
+
+
+int NdbThread_SetConcurrencyLevel(int level)
+{
+ return 0;
+}
+
diff --git a/ndb/src/common/portlib/ose/NdbTick.c b/ndb/src/common/portlib/ose/NdbTick.c
new file mode 100644
index 00000000000..c3deae2bec3
--- /dev/null
+++ b/ndb/src/common/portlib/ose/NdbTick.c
@@ -0,0 +1,64 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbTick.h"
+#include <time.h>
+
+#define NANOSEC_PER_SEC 1000000000
+#define MICROSEC_PER_SEC 1000000
+#define MILLISEC_PER_SEC 1000
+#define MICROSEC_PER_MILLISEC 1000
+#define MILLISEC_PER_NANOSEC 1000000
+
+#ifdef NDB_OSE
+NDB_TICKS NdbTick_CurrentMillisecond(void)
+{
+ return get_ticks()*4;
+}
+#include <rtc.h>
+int
+NdbTick_CurrentMicrosecond(NDB_TICKS * secs, Uint32 * micros){
+ struct TimePair tvp;
+ rtc_get_time(&tvp);
+ * secs = tvp.seconds;
+ * micros = tvp.micros;
+ return 0;
+}
+
+#endif
+
+#if defined NDB_SOFTOSE
+NDB_TICKS NdbTick_CurrentMillisecond(void)
+{
+ /**
+ * Depends on the interval counter in solaris
+ * that means each "tick" in OSE is really 10 milliseconds
+ */
+ return get_ticks()*10;
+}
+
+#include <rtc.h>
+int
+NdbTick_CurrentMicrosecond(NDB_TICKS * secs, Uint32 * micros){
+ struct TimePair tvp;
+ rtc_get_time(&tvp);
+ * secs = tvp.seconds;
+ * micros = tvp.micros;
+ return 0;
+}
+#endif
+
diff --git a/ndb/src/common/portlib/test/Makefile b/ndb/src/common/portlib/test/Makefile
new file mode 100644
index 00000000000..4edc98ede75
--- /dev/null
+++ b/ndb/src/common/portlib/test/Makefile
@@ -0,0 +1,15 @@
+include .defs.mk
+
+TYPE := kernel
+
+BIN_TARGET := PortLibTest
+BIN_TARGET_ARCHIVES := portlib general
+
+SOURCES = NdbPortLibTest.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
diff --git a/ndb/src/common/portlib/test/NdbPortLibTest.cpp b/ndb/src/common/portlib/test/NdbPortLibTest.cpp
new file mode 100644
index 00000000000..8a5c8f4a878
--- /dev/null
+++ b/ndb/src/common/portlib/test/NdbPortLibTest.cpp
@@ -0,0 +1,621 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * NdbPortLibTest.cpp
+ * Test the functionality of portlib
+ * TODO - Add tests for NdbMem
+ */
+
+
+
+#include "NdbOut.hpp"
+#include "NdbThread.h"
+#include "NdbMutex.h"
+#include "NdbCondition.h"
+#include "NdbSleep.h"
+#include "NdbTick.h"
+#include "NdbEnv.h"
+#include "NdbHost.h"
+#include "NdbMain.h"
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <string.h>
+
+int TestHasFailed;
+int verbose = 0;
+
+static void fail(const char* test, const char* cause)
+{
+ TestHasFailed = 1;
+ ndbout << test << " failed, " << cause << endl;
+}
+
+// test 1 variables and funcs
+
+extern "C" void* thread1func(void* arg)
+{
+ int arg1;
+ int returnvalue = 8;
+ arg1 = *(int*)arg;
+ ndbout << "thread1: thread1func called with arg = " << arg1 << endl;
+
+ // delay(1000);
+ if (arg1 != 7)
+ fail("TEST1", "Wrong arg");
+
+ NdbThread_Exit(returnvalue);
+
+ return NULL;
+
+}
+
+// test 2 variables and funcs
+
+NdbMutex* test2mutex;
+
+extern "C" void* test2func(void* arg)
+{
+
+ int arg1;
+ arg1 = *(int*)arg;
+ ndbout << "thread" << arg1 << " started in test2func" << endl;
+
+ if (NdbMutex_Lock(test2mutex) != 0)
+ fail("TEST2", "Failed to lock mutex");
+
+ ndbout << "thread" << arg1 << ", test2func " << endl;
+
+ if (NdbMutex_Unlock(test2mutex) != 0)
+ fail("TEST2", "Failed to unlock mutex");
+
+ int returnvalue = arg1;
+ NdbThread_Exit(returnvalue);
+
+ return NULL;
+
+}
+
+
+// test 3 and 7 variables and funcs
+
+NdbMutex* testmutex;
+NdbCondition* testcond;
+int testthreadsdone;
+
+extern "C" void* testfunc(void* arg)
+{
+ int tmpVar;
+ int threadno;
+ int result;
+
+ threadno = *(int*)arg;
+
+ ndbout << "Thread" << threadno << " started in testfunc" << endl;
+ do
+ {
+
+ if ((threadno % 2) == 0)
+ result = NdbSleep_SecSleep(1);
+ else
+ result = NdbSleep_MilliSleep(100);
+
+ if (result != 0)
+ fail("TEST3", "Wrong result from sleep function");
+
+ if (NdbMutex_Lock(testmutex) != 0)
+ fail("TEST3", "Wrong result from NdbMutex_Lock function");
+
+ ndbout << "thread" << threadno << ", testfunc " << endl;
+ testthreadsdone++;
+ tmpVar = testthreadsdone;
+
+ if (NdbCondition_Signal(testcond) != 0)
+ fail("TEST3", "Wrong result from NdbCondition_Signal function");
+
+ if (NdbMutex_Unlock(testmutex) != 0)
+ fail("TEST3", "Wrong result from NdbMutex_Unlock function");
+
+ }
+ while(tmpVar<100);
+
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+extern "C" void* testTryLockfunc(void* arg)
+{
+ int tmpVar = 0;
+ int threadno;
+ int result;
+
+ threadno = *(int*)arg;
+
+ ndbout << "Thread" << threadno << " started" << endl;
+ do
+ {
+
+ if ((threadno % 2) == 0)
+ result = NdbSleep_SecSleep(1);
+ else
+ result = NdbSleep_MilliSleep(100);
+
+ if (result != 0)
+ fail("TEST3", "Wrong result from sleep function");
+
+ if (NdbMutex_Trylock(testmutex) == 0){
+
+ ndbout << "thread" << threadno << ", testTryLockfunc locked" << endl;
+ testthreadsdone++;
+ tmpVar = testthreadsdone;
+
+ if (NdbCondition_Signal(testcond) != 0)
+ fail("TEST3", "Wrong result from NdbCondition_Signal function");
+
+ if (NdbMutex_Unlock(testmutex) != 0)
+ fail("TEST3", "Wrong result from NdbMutex_Unlock function");
+ }
+
+ }
+ while(tmpVar<100);
+
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+
+
+void testMicros(int count);
+Uint64 time_diff(Uint64 s1, Uint64 s2, Uint32 m1, Uint32 m2);
+
+NDB_COMMAND(PortLibTest, "portlibtest", "portlibtest", "Test the portable function layer", 4096){
+
+ ndbout << "= TESTING ARGUMENT PASSING ============" << endl;
+ ndbout << "ARGC: " << argc << endl;
+ for(int i = 1; i < argc; i++){
+ ndbout << " ARGV"<<i<<": " << (char*)argv[i] << endl;
+ }
+ ndbout << endl << endl;
+
+
+ struct NdbThread* thread1var;
+ void *status = 0;
+ int arg = 7;
+
+ TestHasFailed = 0;
+ // create one thread and wait for it to return
+ ndbout << "= TEST1 ===============================" << endl;
+
+ thread1var = NdbThread_Create(thread1func, // Function
+ (void**)&arg,// Arg
+ 2048, // Stacksize
+ (char*)"thread1", // Thread name
+ NDB_THREAD_PRIO_MEAN); // Thread priority
+
+
+ if(NdbThread_WaitFor(thread1var, &status) != 0)
+ fail("TEST1", "NdbThread_WaitFor failed");
+ // NOTE! thread return value is not yet used in Ndb and thus not tested(does not work)
+ //ndbout << "thread1 returned, status = " << status << endl;
+ //if (status != 8)
+ // fail("TEST1", "Wrong status");
+ ndbout << "TEST1 completed" << endl;
+
+
+ NdbThread_Destroy(&thread1var);
+
+ // Create 10 threads that will wait for a mutex before printing it's message to screen
+ ndbout << "= TEST2 ===============================" << endl;
+#define T2_THREADS 10
+ NdbThread* threads[T2_THREADS];
+ int args[T2_THREADS];
+ void *status2 = 0;
+ test2mutex = NdbMutex_Create();
+ NdbMutex_Lock(test2mutex);
+
+ for (int i = 0; i < T2_THREADS; i++)
+ {
+ args[i] = i;
+ threads[i] = NdbThread_Create(test2func, // Function
+ (void**)&args[i],// Arg
+ 2048, // Stacksize
+ (char*)"test2thread", // Thread name
+ NDB_THREAD_PRIO_MEAN); // Thread priority
+ if (threads[i] == NULL)
+ fail("TEST2", "NdbThread_Create failed");
+ }
+
+ ndbout << "All threads created" << endl;
+
+ NdbMutex_Unlock(test2mutex);
+
+ for (int i = 0; i < T2_THREADS; i++)
+ {
+ if (NdbThread_WaitFor(threads[i], &status2))
+ fail("TEST2", "NdbThread_WaitFor failed");
+
+ NdbThread_Destroy(&threads[i]);
+ // Don't test return values
+ // ndbout << "thread" << i << " returned, status = " << status2 << endl;
+ // if (status2 != i)
+ // fail("TEST2", "Wrong status");
+ }
+
+ if (NdbMutex_Lock(test2mutex) != 0)
+ fail("TEST2", "NdbMutex_Lock failed");
+ if (NdbMutex_Unlock(test2mutex) != 0)
+ fail("TEST2", "NdbMutex_Unlock failed");
+ if (NdbMutex_Destroy(test2mutex) != 0)
+ fail("TEST2", "NdbMutex_Destroy failed");
+ ndbout << "TEST2 completed" << endl;
+
+ ndbout << "= TEST3 ===============================" << endl;
+ // Create 10 threads that will by synchronised by a condition
+ // When they are awakened and have the mutex they will increment a global variable
+#define T3_THREADS 10
+ NdbThread* t3threads[T3_THREADS];
+ int t3args[T3_THREADS];
+ void *status3 = 0;
+
+ testmutex = NdbMutex_Create();
+ testcond = NdbCondition_Create();
+ testthreadsdone = 0;
+
+ for (int i = 0; i < T3_THREADS; i++)
+ {
+ t3args[i] = i;
+ t3threads[i] = NdbThread_Create(testfunc, // Function
+ (void**)&t3args[i],// Arg
+ 2048, // Stacksize
+ (char*)"test3thread", // Thread name
+ NDB_THREAD_PRIO_MEAN); // Thread priority
+ }
+
+ ndbout << "All threads created" << endl;
+
+ if (NdbMutex_Lock(testmutex) != 0)
+ fail("TEST3", "NdbMutex_Lock failed");
+
+ while (testthreadsdone < T3_THREADS*10)
+ {
+ if(NdbCondition_Wait(testcond, testmutex) != 0)
+ fail("TEST3", "NdbCondition_Wait failed");
+ ndbout << "Condition signaled, there are " << testthreadsdone << " completed threads" << endl;
+ }
+ if (NdbMutex_Unlock(testmutex) != 0)
+ fail("TEST3", "NdbMutex_Unlock failed");
+
+ for (int i = 0; i < T3_THREADS; i++)
+ {
+ if (NdbThread_WaitFor(t3threads[i], &status3) != 0)
+ fail("TEST3", "NdbThread_WaitFor failed");
+
+ NdbThread_Destroy(&t3threads[i]);
+ //ndbout << "thread" << i << " returned, status = " << status3 << endl;
+ //if (status3 != i)
+ // fail("TEST3", "Wrong status");
+ }
+
+ NdbMutex_Destroy(testmutex);
+ NdbCondition_Destroy(testcond);
+ ndbout << "TEST3 completed" << endl;
+
+ ndbout << "= TEST4 ===============================" << endl;
+ // Check tick functions
+
+ //#if 0
+
+ int sleeptimes[] = {78, 12, 199, 567, 899};
+
+
+ for (int i = 0; i < 5; i++)
+ {
+ ndbout << "*------------------------------- Measure" << i << endl;
+
+ NDB_TICKS millisec_now;
+ NDB_TICKS millisec_now2;
+
+ millisec_now = NdbTick_CurrentMillisecond();
+ NdbSleep_MilliSleep(sleeptimes[i]);
+ millisec_now2 = NdbTick_CurrentMillisecond();
+
+ ndbout << " Time before sleep = " << millisec_now << endl;
+ ndbout << " Time after sleep = " << millisec_now2 << endl;
+ ndbout << " Tried to sleep "<<sleeptimes[i]<<" milliseconds." << endl;
+ ndbout << " Sleep time was " << millisec_now2 -millisec_now <<" milliseconds." << endl;
+
+ }
+
+ ndbout << "TEST4 completed" << endl;
+
+ ndbout << "= TEST5 ===============================" << endl;
+ // Check NdbOut
+
+ ndbout << "Testing hex and dec functions of NdbOut" << endl;
+
+ for (int i = 0; i<= 0xFF; i++)
+ {
+ ndbout << i << "=" <<hex << i << "="<<dec << i << ", ";
+ }
+
+ ndbout << endl<< "Testing that hex is reset to dec by endl" << endl;
+ ndbout << hex << 67 << endl;
+ ndbout << 67 << endl;
+
+ ndbout << "TEST5 completed" << endl;
+
+
+ ndbout << "= TEST6 ===============================" << endl;
+ const char* theEnvHostNamePtr;
+ char buf[255];
+ char theHostHostName[256];
+ theEnvHostNamePtr = NdbEnv_GetEnv("HOSTNAME", buf, 255);
+ if(theEnvHostNamePtr == NULL)
+ fail("TEST6", "Could not get HOSTNAME from env");
+ else{
+ ndbout << "HOSTNAME from GetEnv" << theEnvHostNamePtr << endl;
+
+ NdbHost_GetHostName(theHostHostName);
+
+ ndbout << "HOSTNAME from GetHostName" <<theHostHostName << endl;
+
+ if (strcmp(theEnvHostNamePtr, theHostHostName) != 0)
+ fail("TEST6", "NdbHost_GetHostName or NdbEnv_GetEnv failed");
+ }
+
+ ndbout << "= TEST7 ===============================" << endl;
+
+ testmutex = NdbMutex_Create();
+ testcond = NdbCondition_Create();
+ testthreadsdone = 0;
+
+ for (int i = 0; i < T3_THREADS; i++)
+ {
+ t3args[i] = i;
+ t3threads[i] = NdbThread_Create(testfunc, // Function
+ (void**)&t3args[i],// Arg
+ 2048, // Stacksize
+ (char*)"test7thread", // Thread name
+ NDB_THREAD_PRIO_MEAN); // Thread priority
+ }
+
+ ndbout << "All threads created" << endl;
+
+ if (NdbMutex_Lock(testmutex) != 0)
+ fail("TEST7", "NdbMutex_Lock failed");
+
+ while (testthreadsdone < T3_THREADS*10)
+ {
+ // just testing the functionality without timing out, therefor 20 sec.
+ if(NdbCondition_WaitTimeout(testcond, testmutex, 20000) != 0)
+ fail("TEST7", "NdbCondition_WaitTimeout failed");
+ ndbout << "Condition signaled, there are " << testthreadsdone << " completed threads" << endl;
+ }
+ if (NdbMutex_Unlock(testmutex) != 0)
+ fail("TEST7", "NdbMutex_Unlock failed");
+
+ for (int i = 0; i < T3_THREADS; i++)
+ {
+ if (NdbThread_WaitFor(t3threads[i], &status3) != 0)
+ fail("TEST7", "NdbThread_WaitFor failed");
+
+ NdbThread_Destroy(&t3threads[i]);
+ }
+
+ NdbMutex_Destroy(testmutex);
+ NdbCondition_Destroy(testcond);
+
+ ndbout << "TEST7 completed" << endl;
+
+
+ ndbout << "= TEST8 ===============================" << endl;
+ ndbout << " NdbCondition_WaitTimeout" << endl;
+ testmutex = NdbMutex_Create();
+ testcond = NdbCondition_Create();
+
+ for (int i = 0; i < 5; i++)
+ {
+ ndbout << "*------------------------------- Measure" << i << endl;
+
+ NDB_TICKS millisec_now;
+ NDB_TICKS millisec_now2;
+
+ millisec_now = NdbTick_CurrentMillisecond();
+ if (NdbCondition_WaitTimeout(testcond, testmutex, sleeptimes[i]) != 0)
+ fail("TEST8", "NdbCondition_WaitTimeout failed");
+ millisec_now2 = NdbTick_CurrentMillisecond();
+
+ ndbout << " Time before WaitTimeout = " << millisec_now << endl;
+ ndbout << " Time after WaitTimeout = " << millisec_now2 << endl;
+ ndbout << " Tried to wait "<<sleeptimes[i]<<" milliseconds." << endl;
+ ndbout << " Wait time was " << millisec_now2 -millisec_now <<" milliseconds." << endl;
+
+ }
+
+ ndbout << "TEST8 completed" << endl;
+
+
+ ndbout << "= TEST9 ===============================" << endl;
+ ndbout << " NdbTick_CurrentXXXXXsecond compare" << endl;
+
+ for (int i = 0; i < 5; i++)
+ {
+ ndbout << "*------------------------------- Measure" << i << endl;
+
+ NDB_TICKS millisec_now;
+ NDB_TICKS millisec_now2;
+ Uint32 usec_now, usec_now2;
+ Uint64 msec_now, msec_now2;
+
+
+ millisec_now = NdbTick_CurrentMillisecond();
+ NdbTick_CurrentMicrosecond( &msec_now, &usec_now);
+
+ NdbSleep_MilliSleep(sleeptimes[i]);
+
+ millisec_now2 = NdbTick_CurrentMillisecond();
+ NdbTick_CurrentMicrosecond( &msec_now2, &usec_now2);
+
+ Uint64 usecdiff = time_diff(msec_now,msec_now2,usec_now,usec_now2);
+ NDB_TICKS msecdiff = millisec_now2 -millisec_now;
+
+ ndbout << " Slept "<<sleeptimes[i]<<" milliseconds." << endl;
+ ndbout << " Measured " << msecdiff <<" milliseconds with milli function ." << endl;
+ ndbout << " Measured " << usecdiff/1000 << "," << usecdiff%1000<<" milliseconds with micro function ." << endl;
+ }
+
+ ndbout << "TEST9 completed" << endl;
+
+
+ const int iter = 20;
+ ndbout << "Testing microsecond timer - " << iter << " iterations" << endl;
+ testMicros(iter);
+ ndbout << "Testing microsecond timer - COMPLETED" << endl;
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ ndbout << "system_tick() = " << system_tick() << " us per tick" << endl;
+#endif
+
+
+ ndbout << "= TEST10 ===============================" << endl;
+
+ testmutex = NdbMutex_Create();
+ testcond = NdbCondition_Create();
+ testthreadsdone = 0;
+
+ for (int i = 0; i < T3_THREADS; i++)
+ {
+ t3args[i] = i;
+ t3threads[i] = NdbThread_Create(testTryLockfunc, // Function
+ (void**)&t3args[i],// Arg
+ 2048, // Stacksize
+ (char*)"test10thread", // Thread name
+ NDB_THREAD_PRIO_MEAN); // Thread priority
+ }
+
+ ndbout << "All threads created" << endl;
+
+ if (NdbMutex_Lock(testmutex) != 0)
+ fail("TEST10", "NdbMutex_Lock failed");
+
+ while (testthreadsdone < T3_THREADS*10)
+ {
+ if(NdbCondition_Wait(testcond, testmutex) != 0)
+ fail("TEST10", "NdbCondition_WaitTimeout failed");
+ ndbout << "Condition signaled, there are " << testthreadsdone << " completed threads" << endl;
+ }
+ if (NdbMutex_Unlock(testmutex) != 0)
+ fail("TEST10", "NdbMutex_Unlock failed");
+
+ for (int i = 0; i < T3_THREADS; i++)
+ {
+ if (NdbThread_WaitFor(t3threads[i], &status3) != 0)
+ fail("TEST10", "NdbThread_WaitFor failed");
+
+ NdbThread_Destroy(&t3threads[i]);
+ }
+
+ NdbMutex_Destroy(testmutex);
+ NdbCondition_Destroy(testcond);
+
+ ndbout << "TEST10 completed" << endl;
+
+
+ // Check total status of test
+
+ if (TestHasFailed == 1)
+ ndbout << endl << "TEST FAILED!" << endl;
+ else
+ ndbout << endl << "TEST PASSED!" << endl;
+
+ return TestHasFailed;
+
+};
+
+Uint64 time_diff(Uint64 s1, Uint64 s2, Uint32 m1, Uint32 m2){
+
+ Uint64 diff = 0;
+ diff += (s2 - s1) * 1000000;
+ if(m2 >= m1)
+ diff += (m2 - m1);
+ else {
+ diff += m2;
+ diff -= m1;
+ }
+
+ // if(0)
+ // ndbout("(s1,m1) = (%d, %d) (s2,m2) = (%d, %d) -> diff = %d\n",
+ // (Uint32)s1,m1,(Uint32)s2,m2, (Uint32)diff);
+
+ return diff;
+};
+
+void
+testMicros(int count){
+ Uint32 avg = 0;
+ Uint32 sum2 = 0;
+
+ for(int i = 0; i<count; i++){
+ Uint64 s1, s2;
+ Uint32 m1, m2;
+ if(NdbTick_CurrentMicrosecond(&s1, &m1) != 0){
+ ndbout << "Failed to get current micro" << endl;
+ TestHasFailed = 1;
+ return;
+ }
+ Uint32 r = (rand() % 1000) + 1;
+ NdbSleep_MilliSleep(r);
+ if(NdbTick_CurrentMicrosecond(&s2, &m2) != 0){
+ ndbout << "Failed to get current micro" << endl;
+ TestHasFailed = 1;
+ return;
+ }
+ Uint64 m = time_diff(s1,s2,m1,m2);
+ if(verbose)
+ ndbout << "Slept for " << r << " ms"
+ << " - Measured " << m << " us" << endl;
+
+ if(m > (r*1000)){
+ avg += (m - (r*1000));
+ sum2 += (m - (r*1000)) * (m - (r*1000));
+ } else {
+ avg += ((r*1000) - m);
+ sum2 += ((r*1000) - m) * ((r*1000) - m);
+ }
+#if 0
+ m /= 1000;
+ if(m > r && ((m - r) > 10)){
+ ndbout << "Difference to big: " << (m - r) << " - Test failed" << endl;
+ TestHasFailed = 1;
+ }
+ if(m < r && ((r - m) > 10)){
+ ndbout << "Difference to big: " << (r - m) << " - Test failed" << endl;
+ TestHasFailed = 1;
+ }
+#endif
+ }
+
+ Uint32 dev = (avg * avg - sum2) / count; dev /= count;
+ avg /= count;
+
+ Uint32 t = 0;
+ while((t*t)<dev) t++;
+ ndbout << "NOTE - measure are compared to NdbSleep_MilliSleep(...)" << endl;
+ ndbout << "Average error = " << avg << " us" << endl;
+ ndbout << "Stddev error = " << t << " us" << endl;
+}
diff --git a/ndb/src/common/portlib/unix/Makefile b/ndb/src/common/portlib/unix/Makefile
new file mode 100644
index 00000000000..452196d9f08
--- /dev/null
+++ b/ndb/src/common/portlib/unix/Makefile
@@ -0,0 +1,27 @@
+include .defs.mk
+
+TYPE := util
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := portlib
+
+SOURCES.c = NdbCondition.c \
+ NdbMutex.c \
+ NdbSleep.c \
+ NdbTick.c \
+ NdbEnv.c \
+ NdbThread.c \
+ NdbHost.c \
+ NdbTCP.c \
+ NdbDaemon.c
+
+ifeq ($(NDB_OS), SOFTOSE)
+ SOURCES += NdbMem_SoftOse.cpp
+else
+ SOURCES.c += NdbMem.c
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+testNdbDaemon: NdbDaemon.c
+ $(CC) -o $@ NdbDaemon.c $(CCFLAGS) -DNDB_DAEMON_TEST -L$(NDB_TOP)/lib
diff --git a/ndb/src/common/portlib/unix/NdbCondition.c b/ndb/src/common/portlib/unix/NdbCondition.c
new file mode 100644
index 00000000000..35b80821052
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbCondition.c
@@ -0,0 +1,179 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbCondition.h>
+#include <pthread.h>
+#include <assert.h>
+#include <sys/types.h>
+#if defined NDB_MACOSX
+#include <stdlib.h>
+#else
+#include <malloc.h>
+#endif
+
+#include <NdbMutex.h>
+
+struct NdbCondition
+{
+ pthread_cond_t cond;
+};
+
+
+
+struct NdbCondition*
+NdbCondition_Create(void)
+{
+ struct NdbCondition* tmpCond;
+ int result;
+
+ tmpCond = (struct NdbCondition*)malloc(sizeof(struct NdbCondition));
+
+ if (tmpCond == NULL)
+ return NULL;
+
+ result = pthread_cond_init(&tmpCond->cond, NULL);
+
+ assert(result==0);
+ return tmpCond;
+}
+
+
+
+int
+NdbCondition_Wait(struct NdbCondition* p_cond,
+ NdbMutex* p_mutex)
+{
+ int result;
+
+ if (p_cond == NULL || p_mutex == NULL)
+ return 1;
+
+ result = pthread_cond_wait(&p_cond->cond, p_mutex);
+
+ return result;
+}
+
+#if defined NDB_SOLARIS || defined NDB_HPUX
+#include <time.h>
+int
+NdbCondition_WaitTimeout(struct NdbCondition* p_cond,
+ NdbMutex* p_mutex,
+ int msecs){
+ int result;
+ struct timespec abstime;
+ int secs = 0;
+
+ if (p_cond == NULL || p_mutex == NULL)
+ return 1;
+
+ clock_gettime(CLOCK_REALTIME, &abstime);
+
+ if(msecs >= 1000){
+ secs = msecs / 1000;
+ msecs = msecs % 1000;
+ }
+
+ abstime.tv_sec += secs;
+ abstime.tv_nsec += msecs * 1000000;
+ if (abstime.tv_nsec >= 1000000000) {
+ abstime.tv_sec += 1;
+ abstime.tv_nsec -= 1000000000;
+ }
+
+ result = pthread_cond_timedwait(&p_cond->cond, p_mutex, &abstime);
+
+ return result;
+}
+#endif
+
+#if defined NDB_LINUX || defined NDB_MACOSX
+#include <unistd.h>
+#include <sys/time.h>
+
+int
+NdbCondition_WaitTimeout(struct NdbCondition* p_cond,
+ NdbMutex* p_mutex,
+ int msecs){
+ int result;
+ struct timespec abstime;
+ struct timeval tick_time;
+ int secs = 0;
+
+ if (p_cond == NULL || p_mutex == NULL)
+ return 1;
+
+ gettimeofday(&tick_time, 0);
+
+ if(msecs >= 1000){
+ secs = msecs / 1000;
+ msecs = msecs % 1000;
+ }
+
+
+ abstime.tv_sec = tick_time.tv_sec + secs;
+ abstime.tv_nsec = tick_time.tv_usec * 1000 + msecs * 1000000;
+ if (abstime.tv_nsec >= 1000000000) {
+ abstime.tv_sec += 1;
+ abstime.tv_nsec -= 1000000000;
+ }
+
+ result = pthread_cond_timedwait(&p_cond->cond, p_mutex, &abstime);
+
+ return result;
+}
+#endif
+
+
+int
+NdbCondition_Signal(struct NdbCondition* p_cond){
+ int result;
+
+ if (p_cond == NULL)
+ return 1;
+
+ result = pthread_cond_signal(&p_cond->cond);
+
+ return result;
+}
+
+
+int NdbCondition_Broadcast(struct NdbCondition* p_cond)
+{
+ int result;
+
+ if (p_cond == NULL)
+ return 1;
+
+ result = pthread_cond_broadcast(&p_cond->cond);
+
+ return result;
+}
+
+
+int NdbCondition_Destroy(struct NdbCondition* p_cond)
+{
+ int result;
+
+ if (p_cond == NULL)
+ return 1;
+
+ result = pthread_cond_destroy(&p_cond->cond);
+ free(p_cond);
+
+ return 0;
+}
+
diff --git a/ndb/src/common/portlib/unix/NdbDaemon.c b/ndb/src/common/portlib/unix/NdbDaemon.c
new file mode 100644
index 00000000000..fc114266c9d
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbDaemon.c
@@ -0,0 +1,170 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbDaemon.h"
+#include <assert.h>
+
+#ifdef NDB_LINUX
+#include <sys/types.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include <string.h>
+#include <stdio.h>
+#include <fcntl.h>
+#include <errno.h>
+#endif
+
+#ifdef NDB_SOLARIS
+#include <sys/types.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include <string.h>
+#include <stdio.h>
+#include <fcntl.h>
+#include <errno.h>
+#endif
+
+#define NdbDaemon_ErrorSize 500
+#if defined(NDB_LINUX) || defined(NDB_SOLARIS)
+long NdbDaemon_DaemonPid;
+int NdbDaemon_ErrorCode;
+char NdbDaemon_ErrorText[NdbDaemon_ErrorSize];
+#endif
+int
+NdbDaemon_Make(const char* lockfile, const char* logfile, unsigned flags)
+{
+ /* XXX fix other unixes */
+#if defined(NDB_LINUX) || defined(NDB_SOLARIS)
+ int lockfd = -1, logfd = -1, n;
+ char buf[64];
+
+ /* Check that we have write access to lock file */
+ assert(lockfile != NULL);
+ lockfd = open(lockfile, O_CREAT|O_RDWR, 0644);
+ if (lockfd == -1) {
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "%s: open for write failed: %s", lockfile, strerror(errno));
+ return -1;
+ }
+ /* Read any old pid from lock file */
+ buf[0] = 0;
+ n = read(lockfd, buf, sizeof(buf));
+ if (n < 0) {
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "%s: read failed: %s", lockfile, strerror(errno));
+ return -1;
+ }
+ NdbDaemon_DaemonPid = atol(buf);
+ if (lseek(lockfd, 0, SEEK_SET) == -1) {
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "%s: lseek failed: %s", lockfile, strerror(errno));
+ return -1;
+ }
+ /* Test for lock before becoming daemon */
+ if (lockf(lockfd, F_TEST, 0) == -1) {
+ if (errno == EACCES || errno == EAGAIN) { /* results may vary */
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "%s: already locked by pid=%ld", lockfile, NdbDaemon_DaemonPid);
+ return -1;
+ }
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "%s: lock test failed: %s", lockfile, strerror(errno));
+ return -1;
+ }
+ /* Test open log file before becoming daemon */
+ if (logfile != NULL) {
+ logfd = open(logfile, O_CREAT|O_WRONLY|O_APPEND, 0644);
+ if (logfd == -1) {
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "%s: open for write failed: %s", logfile, strerror(errno));
+ return -1;
+ }
+ }
+ /* Fork */
+ n = fork();
+ if (n == -1) {
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "fork failed: %s", strerror(errno));
+ return -1;
+ }
+ /* Exit if we are the parent */
+ if (n != 0) {
+ exit(0);
+ }
+ /* Running in child process */
+ NdbDaemon_DaemonPid = getpid();
+ /* Lock the lock file (likely to succeed due to test above) */
+ if (lockf(lockfd, F_LOCK, 0) == -1) {
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "%s: lock failed: %s", lockfile, strerror(errno));
+ return -1;
+ }
+ /* Become process group leader */
+ if (setsid() == -1) {
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "setsid failed: %s", strerror(errno));
+ return -1;
+ }
+ /* Write pid to lock file */
+ if (ftruncate(lockfd, 0) == -1) {
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "%s: ftruncate failed: %s", lockfile, strerror(errno));
+ return -1;
+ }
+ sprintf(buf, "%ld\n", NdbDaemon_DaemonPid);
+ n = strlen(buf);
+ if (write(lockfd, buf, n) != n) {
+ NdbDaemon_ErrorCode = errno;
+ snprintf(NdbDaemon_ErrorText, NdbDaemon_ErrorSize,
+ "%s: write failed: %s", lockfile, strerror(errno));
+ return -1;
+ }
+ /* Do input/output redirections (assume fd 0,1,2 not in use) */
+ close(0);
+ open("/dev/null", O_RDONLY);
+ if (logfile != 0) {
+ dup2(logfd, 1);
+ dup2(logfd, 2);
+ close(logfd);
+ }
+#endif
+ /* Success */
+ return 0;
+}
+
+#ifdef NDB_DAEMON_TEST
+
+int
+main()
+{
+ if (NdbDaemon_Make("test.pid", "test.log", 0) == -1) {
+ fprintf(stderr, "NdbDaemon_Make: %s\n", NdbDaemon_ErrorText);
+ return 1;
+ }
+ sleep(10);
+ return 0;
+}
+
+#endif
diff --git a/ndb/src/common/portlib/unix/NdbEnv.c b/ndb/src/common/portlib/unix/NdbEnv.c
new file mode 100644
index 00000000000..b01e3b239ca
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbEnv.c
@@ -0,0 +1,34 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbEnv.h"
+#include <string.h>
+#include <stdlib.h>
+
+const char* NdbEnv_GetEnv(const char* name, char * buf, int buflen)
+{
+ char* p = NULL;
+ p = getenv(name);
+
+ if (p != NULL && buf != NULL){
+ strncpy(buf, p, buflen);
+ buf[buflen-1] = 0;
+ }
+ return p;
+
+}
+
diff --git a/ndb/src/common/portlib/unix/NdbHost.c b/ndb/src/common/portlib/unix/NdbHost.c
new file mode 100644
index 00000000000..8d2a23fccda
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbHost.c
@@ -0,0 +1,34 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbHost.h"
+#include <unistd.h>
+
+int NdbHost_GetHostName(char* buf)
+{
+ if (gethostname(buf, MAXHOSTNAMELEN) != 0)
+ {
+ return -1;
+ }
+ return 0;
+}
+
+int NdbHost_GetProcessId(void)
+{
+ return getpid();
+}
+
diff --git a/ndb/src/common/portlib/unix/NdbMem.c b/ndb/src/common/portlib/unix/NdbMem.c
new file mode 100644
index 00000000000..a18cf30cc8a
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbMem.c
@@ -0,0 +1,76 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbMem.h"
+
+#include <assert.h>
+#include <assert.h>
+#include <stdlib.h>
+#ifndef NDB_MACOSX
+#include <sys/mman.h>
+#endif
+
+void NdbMem_Create()
+{
+ /* Do nothing */
+ return;
+}
+
+void NdbMem_Destroy()
+{
+ /* Do nothing */
+ return;
+}
+
+void* NdbMem_Allocate(size_t size)
+{
+ assert(size > 0);
+ return (void*)malloc(size);
+}
+
+void* NdbMem_AllocateAlign(size_t size, size_t alignment)
+{
+ /*
+ return (void*)memalign(alignment, size);
+ TEMP fix
+ */
+ return (void*)malloc(size);
+}
+
+
+void NdbMem_Free(void* ptr)
+{
+ free(ptr);
+}
+
+
+int NdbMem_MemLockAll(){
+#if defined NDB_MACOSX
+ return 0;
+#else
+ return mlockall(MCL_CURRENT | MCL_FUTURE);
+#endif
+}
+
+int NdbMem_MemUnlockAll(){
+#if defined NDB_MACOSX
+ return 0;
+#else
+ return munlockall();
+#endif
+}
+
diff --git a/ndb/src/common/portlib/unix/NdbMutex.c b/ndb/src/common/portlib/unix/NdbMutex.c
new file mode 100644
index 00000000000..3cadc0667e7
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbMutex.c
@@ -0,0 +1,93 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbMutex.h"
+
+#include <pthread.h>
+#include <stdlib.h>
+#include <assert.h>
+
+NdbMutex* NdbMutex_Create(void)
+{
+ NdbMutex* pNdbMutex;
+ int result;
+
+ pNdbMutex = (NdbMutex*)malloc(sizeof(NdbMutex));
+
+ if (pNdbMutex == NULL)
+ return NULL;
+
+ result = pthread_mutex_init(pNdbMutex, NULL);
+ assert(result == 0);
+
+ return pNdbMutex;
+
+}
+
+
+int NdbMutex_Destroy(NdbMutex* p_mutex)
+{
+ int result;
+
+ if (p_mutex == NULL)
+ return -1;
+
+ result = pthread_mutex_destroy(p_mutex);
+ free(p_mutex);
+
+ return result;
+
+}
+
+
+int NdbMutex_Lock(NdbMutex* p_mutex)
+{
+ int result;
+
+ if (p_mutex == NULL)
+ return -1;
+
+ result = pthread_mutex_lock(p_mutex);
+
+ return result;
+}
+
+
+int NdbMutex_Unlock(NdbMutex* p_mutex)
+{
+ int result;
+
+ if (p_mutex == NULL)
+ return -1;
+
+ result = pthread_mutex_unlock(p_mutex);
+
+ return result;
+}
+
+
+int NdbMutex_Trylock(NdbMutex* p_mutex)
+{
+ int result = -1;
+
+ if (p_mutex != NULL) {
+ result = pthread_mutex_trylock(p_mutex);
+ }
+
+ return result;
+}
+
diff --git a/ndb/src/common/portlib/unix/NdbSleep.c b/ndb/src/common/portlib/unix/NdbSleep.c
new file mode 100644
index 00000000000..35132d7f9c7
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbSleep.c
@@ -0,0 +1,48 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbSleep.h"
+
+
+#ifdef NDB_SOLARIS
+#include <sys/types.h>
+#include <unistd.h>
+#endif
+
+#if defined NDB_LINUX || defined NDB_HPUX || defined NDB_MACOSX
+#include <time.h>
+#include <unistd.h>
+#endif
+
+int
+NdbSleep_MilliSleep(int milliseconds){
+ int result = 0;
+ struct timespec sleeptime;
+ sleeptime.tv_sec = milliseconds / 1000;
+ sleeptime.tv_nsec = (milliseconds - (sleeptime.tv_sec * 1000)) * 1000000;
+ result = nanosleep(&sleeptime, NULL);
+ return result;
+}
+
+int
+NdbSleep_SecSleep(int seconds){
+ int result = 0;
+ result = sleep(seconds);
+ return result;
+}
+
+
diff --git a/ndb/src/common/portlib/unix/NdbTCP.c b/ndb/src/common/portlib/unix/NdbTCP.c
new file mode 100644
index 00000000000..c2613c211c5
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbTCP.c
@@ -0,0 +1,60 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbTCP.h"
+
+
+#ifdef NDB_SOLARIS
+int
+Ndb_getInAddr(struct in_addr * dst, const char *address) {
+ struct hostent host, * hostPtr;
+ char buf[1024];
+ int h_errno;
+ hostPtr = gethostbyname_r(address, &host, &buf[0], 1024, &h_errno);
+ if (hostPtr != NULL) {
+ dst->s_addr = ((struct in_addr *) *hostPtr->h_addr_list)->s_addr;
+ return 0;
+ }
+
+ /* Try it as aaa.bbb.ccc.ddd. */
+ dst->s_addr = inet_addr(address);
+ if (dst->s_addr != -1) {
+ return 0;
+ }
+ return -1;
+}
+#endif
+
+#if defined NDB_LINUX || defined NDB_HPUX || defined NDB_MACOSX
+int
+Ndb_getInAddr(struct in_addr * dst, const char *address) {
+ struct hostent * hostPtr;
+ hostPtr = gethostbyname(address);
+ if (hostPtr != NULL) {
+ dst->s_addr = ((struct in_addr *) *hostPtr->h_addr_list)->s_addr;
+ return 0;
+ }
+
+ /* Try it as aaa.bbb.ccc.ddd. */
+ dst->s_addr = inet_addr(address);
+ if (dst->s_addr != -1) {
+ return 0;
+ }
+ return -1;
+}
+#endif
+
diff --git a/ndb/src/common/portlib/unix/NdbThread.c b/ndb/src/common/portlib/unix/NdbThread.c
new file mode 100644
index 00000000000..3665c4c9159
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbThread.c
@@ -0,0 +1,119 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbThread.h"
+#include <pthread.h>
+#ifdef NDB_MACOSX
+#include <stdlib.h>
+#else
+#include <malloc.h>
+#endif
+#include <assert.h>
+#include <string.h>
+#include <NdbStdio.h>
+
+#define MAX_THREAD_NAME 16
+
+
+struct NdbThread
+{
+ pthread_t thread;
+ char thread_name[MAX_THREAD_NAME];
+};
+
+
+
+struct NdbThread* NdbThread_Create(NDB_THREAD_FUNC *p_thread_func,
+ NDB_THREAD_ARG *p_thread_arg,
+ const NDB_THREAD_STACKSIZE thread_stack_size,
+ const char* p_thread_name,
+ NDB_THREAD_PRIO thread_prio)
+{
+ struct NdbThread* tmpThread;
+ int result;
+ pthread_attr_t thread_attr;
+
+ if (p_thread_func == NULL)
+ return 0;
+
+ tmpThread = (struct NdbThread*)malloc(sizeof(struct NdbThread));
+ if (tmpThread == NULL)
+ return NULL;
+
+ snprintf(tmpThread->thread_name, sizeof(tmpThread->thread_name),
+ "%s", p_thread_name);
+
+ pthread_attr_init(&thread_attr);
+ pthread_attr_setstacksize(&thread_attr, thread_stack_size);
+#if defined NDB_SOLARIS
+#if !defined NDB_SOLARIS6
+ /* Guard stack overflow with a 2k databuffer */
+ pthread_attr_setguardsize(&thread_attr, 2048);
+#endif
+#endif
+
+ pthread_attr_setdetachstate(&thread_attr, PTHREAD_CREATE_JOINABLE);
+ result = pthread_create(&tmpThread->thread,
+ &thread_attr,
+ p_thread_func,
+ p_thread_arg);
+ assert(result==0);
+
+ pthread_attr_destroy(&thread_attr);
+ return tmpThread;
+}
+
+
+void NdbThread_Destroy(struct NdbThread** p_thread)
+{
+ if (*p_thread != NULL){
+ free(* p_thread);
+ * p_thread = 0;
+ }
+}
+
+
+int NdbThread_WaitFor(struct NdbThread* p_wait_thread, void** status)
+{
+ int result;
+
+ if (p_wait_thread == NULL)
+ return 0;
+
+ if (p_wait_thread->thread == NULL)
+ return 0;
+
+ result = pthread_join(p_wait_thread->thread, status);
+
+ return result;
+}
+
+
+void NdbThread_Exit(int status)
+{
+ pthread_exit(&status);
+}
+
+
+int NdbThread_SetConcurrencyLevel(int level)
+{
+#ifndef NDB_SOLARIS6
+ return pthread_setconcurrency(level);
+#else
+ return 0;
+#endif
+}
diff --git a/ndb/src/common/portlib/unix/NdbTick.c b/ndb/src/common/portlib/unix/NdbTick.c
new file mode 100644
index 00000000000..5adb4ec80c2
--- /dev/null
+++ b/ndb/src/common/portlib/unix/NdbTick.c
@@ -0,0 +1,110 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbTick.h"
+#include <time.h>
+
+#define NANOSEC_PER_SEC 1000000000
+#define MICROSEC_PER_SEC 1000000
+#define MILLISEC_PER_SEC 1000
+#define MICROSEC_PER_MILLISEC 1000
+#define MILLISEC_PER_NANOSEC 1000000
+
+
+#if defined NDB_SOLARIS || NDB_HPUX
+NDB_TICKS NdbTick_CurrentMillisecond(void)
+{
+ struct timespec tick_time;
+ clock_gettime(CLOCK_REALTIME, &tick_time);
+
+ return
+ ((NDB_TICKS)tick_time.tv_sec) * ((NDB_TICKS)MILLISEC_PER_SEC) +
+ ((NDB_TICKS)tick_time.tv_nsec) / ((NDB_TICKS)MILLISEC_PER_NANOSEC);
+}
+
+int
+NdbTick_CurrentMicrosecond(NDB_TICKS * secs, Uint32 * micros){
+ struct timespec t;
+ int res = clock_gettime(CLOCK_REALTIME, &t);
+ * secs = t.tv_sec;
+ * micros = t.tv_nsec / 1000;
+ return res;
+}
+#endif
+
+#if defined NDB_LINUX || NDB_MACOSX
+#include <unistd.h>
+#include <sys/time.h>
+NDB_TICKS NdbTick_CurrentMillisecond(void)
+{
+ struct timeval tick_time;
+ gettimeofday(&tick_time, 0);
+
+ return
+ ((NDB_TICKS)tick_time.tv_sec) * ((NDB_TICKS)MILLISEC_PER_SEC) +
+ ((NDB_TICKS)tick_time.tv_usec) / ((NDB_TICKS)MICROSEC_PER_MILLISEC);
+}
+
+int
+NdbTick_CurrentMicrosecond(NDB_TICKS * secs, Uint32 * micros){
+ struct timeval tick_time;
+ int res = gettimeofday(&tick_time, 0);
+
+ if(secs==0) {
+ NDB_TICKS secs = tick_time.tv_sec;
+ *micros = tick_time.tv_usec;
+ *micros = secs*1000000+*micros;
+ } else {
+ * secs = tick_time.tv_sec;
+ * micros = tick_time.tv_usec;
+ }
+ return res;
+}
+
+#endif
+#ifdef TIME_MEASUREMENT
+int
+NdbTick_getMicroTimer(struct MicroSecondTimer* input_timer)
+{
+ NDB_TICKS secs;
+ Uint32 mics;
+ int ret_value;
+ ret_value = NdbTick_CurrentMicrosecond(&secs, &mics);
+ input_timer->seconds = secs;
+ input_timer->micro_seconds = (NDB_TICKS)mics;
+ return ret_value;
+}
+
+NDB_TICKS
+NdbTick_getMicrosPassed(struct MicroSecondTimer start,
+ struct MicroSecondTimer stop)
+{
+ NDB_TICKS ret_value = (NDB_TICKS)0;
+ if (start.seconds < stop.seconds) {
+ NDB_TICKS sec_passed = stop.seconds - start.seconds;
+ ret_value = ((NDB_TICKS)MICROSEC_PER_SEC) * sec_passed;
+ } else if (start.seconds > stop.seconds) {
+ return ret_value;
+ }//if
+ if (start.micro_seconds < stop.micro_seconds) {
+ ret_value += (stop.micro_seconds - start.micro_seconds);
+ } else if (ret_value != (NDB_TICKS)0) {
+ ret_value -= (start.micro_seconds - stop.micro_seconds);
+ }//if
+ return ret_value;
+}
+#endif
diff --git a/ndb/src/common/portlib/win32/Makefile b/ndb/src/common/portlib/win32/Makefile
new file mode 100644
index 00000000000..bb29ac5547e
--- /dev/null
+++ b/ndb/src/common/portlib/win32/Makefile
@@ -0,0 +1,30 @@
+include .defs.mk
+
+TYPE := util
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := portlib
+
+SOURCES.c = NdbCondition.c \
+ NdbMutex.c \
+ NdbSleep.c \
+ NdbTick.c \
+ NdbEnv.c \
+ NdbThread.c \
+ NdbHost.c \
+ NdbTCP.c \
+ NdbDaemon.c
+
+ifeq ($(NDB_OS), SOFTOSE)
+ SOURCES += NdbMem_SoftOse.cpp
+else
+ SOURCES.c += NdbMem.c
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
+
diff --git a/ndb/src/common/portlib/win32/NdbCondition.c b/ndb/src/common/portlib/win32/NdbCondition.c
new file mode 100644
index 00000000000..12b508cf33b
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbCondition.c
@@ -0,0 +1,184 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <winsock2.h>
+#include <ws2tcpip.h>
+#include <windows.h>
+#include <assert.h>
+#include <sys/types.h>
+
+#include "NdbCondition.h"
+#include <NdbMutex.h>
+
+
+struct NdbCondition
+{
+ long nWaiters;
+ NdbMutex* pNdbMutexWaitersLock;
+ HANDLE hSemaphore;
+ HANDLE hEventWaitersDone;
+ int bWasBroadcast;
+};
+
+
+struct NdbCondition*
+NdbCondition_Create(void)
+{
+ int result = 0;
+ struct NdbCondition* pNdbCondition = (struct NdbCondition*)malloc(sizeof(struct NdbCondition));
+ if(!pNdbCondition)
+ return 0;
+
+ pNdbCondition->nWaiters = 0;
+ pNdbCondition->bWasBroadcast = 0;
+ if(!(pNdbCondition->hSemaphore = CreateSemaphore(0, 0, MAXLONG, 0)))
+ result = -1;
+ else if(!(pNdbCondition->pNdbMutexWaitersLock = NdbMutex_Create()))
+ result = -1;
+ else if(!(pNdbCondition->hEventWaitersDone = CreateEvent(0, 0, 0, 0)))
+ result = -1;
+
+ assert(!result);
+ return pNdbCondition;
+}
+
+
+int
+NdbCondition_Wait(struct NdbCondition* p_cond,
+ NdbMutex* p_mutex)
+{
+ int result;
+ int bLastWaiter;
+ if(!p_cond || !p_mutex)
+ return 1;
+
+ NdbMutex_Lock(p_cond->pNdbMutexWaitersLock);
+ p_cond->nWaiters++;
+ NdbMutex_Unlock(p_cond->pNdbMutexWaitersLock);
+
+ if(NdbMutex_Unlock(p_mutex))
+ return -1;
+ result = WaitForSingleObject (p_cond->hSemaphore, INFINITE);
+
+ NdbMutex_Lock(p_cond->pNdbMutexWaitersLock);
+ p_cond->nWaiters--;
+ bLastWaiter = (p_cond->bWasBroadcast && p_cond->nWaiters==0);
+ NdbMutex_Unlock(p_cond->pNdbMutexWaitersLock);
+
+ if(result==WAIT_OBJECT_0 && bLastWaiter)
+ SetEvent(p_cond->hEventWaitersDone);
+
+ NdbMutex_Lock(p_mutex);
+ return result;
+}
+
+
+int
+NdbCondition_WaitTimeout(struct NdbCondition* p_cond,
+ NdbMutex* p_mutex,
+ int msecs)
+{
+ int result;
+ int bLastWaiter;
+ if (!p_cond || !p_mutex)
+ return 1;
+
+ NdbMutex_Lock(p_cond->pNdbMutexWaitersLock);
+ p_cond->nWaiters++;
+ NdbMutex_Unlock(p_cond->pNdbMutexWaitersLock);
+ if(msecs<0)
+ msecs = 0;
+
+ if(NdbMutex_Unlock(p_mutex))
+ return -1;
+ result = WaitForSingleObject(p_cond->hSemaphore, msecs);
+
+ NdbMutex_Lock(p_cond->pNdbMutexWaitersLock);
+ p_cond->nWaiters--;
+ bLastWaiter = (p_cond->bWasBroadcast && p_cond->nWaiters==0);
+ NdbMutex_Unlock(p_cond->pNdbMutexWaitersLock);
+
+ if(result!=WAIT_OBJECT_0)
+ result = -1;
+
+ if(bLastWaiter)
+ SetEvent(p_cond->hEventWaitersDone);
+
+ NdbMutex_Lock(p_mutex);
+ return result;
+}
+
+
+int
+NdbCondition_Signal(struct NdbCondition* p_cond)
+{
+ int bHaveWaiters;
+ if(!p_cond)
+ return 1;
+
+ NdbMutex_Lock(p_cond->pNdbMutexWaitersLock);
+ bHaveWaiters = (p_cond->nWaiters > 0);
+ NdbMutex_Unlock(p_cond->pNdbMutexWaitersLock);
+
+ if(bHaveWaiters)
+ return (ReleaseSemaphore(p_cond->hSemaphore, 1, 0) ? 0 : -1);
+ else
+ return 0;
+}
+
+
+int NdbCondition_Broadcast(struct NdbCondition* p_cond)
+{
+ int bHaveWaiters;
+ int result = 0;
+ if(!p_cond)
+ return 1;
+
+ NdbMutex_Lock(p_cond->pNdbMutexWaitersLock);
+ bHaveWaiters = 0;
+ if(p_cond->nWaiters > 0)
+ {
+ p_cond->bWasBroadcast = !0;
+ bHaveWaiters = 1;
+ }
+ NdbMutex_Unlock(p_cond->pNdbMutexWaitersLock);
+ if(bHaveWaiters)
+ {
+ if(!ReleaseSemaphore(p_cond->hSemaphore, p_cond->nWaiters, 0))
+ result = -1;
+ else if(WaitForSingleObject (p_cond->hEventWaitersDone, INFINITE) != WAIT_OBJECT_0)
+ result = -1;
+ p_cond->bWasBroadcast = 0;
+ }
+ return result;
+}
+
+
+int NdbCondition_Destroy(struct NdbCondition* p_cond)
+{
+ int result;
+ if(!p_cond)
+ return 1;
+
+ CloseHandle(p_cond->hEventWaitersDone);
+ NdbMutex_Destroy(p_cond->pNdbMutexWaitersLock);
+ result = (CloseHandle(p_cond->hSemaphore) ? 0 : -1);
+
+ free(p_cond);
+ return 0;
+}
+
diff --git a/ndb/src/common/portlib/win32/NdbDaemon.c b/ndb/src/common/portlib/win32/NdbDaemon.c
new file mode 100644
index 00000000000..b96d4c20260
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbDaemon.c
@@ -0,0 +1,44 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbDaemon.h"
+
+#define NdbDaemon_ErrorSize 500
+long NdbDaemon_DaemonPid;
+int NdbDaemon_ErrorCode;
+char NdbDaemon_ErrorText[NdbDaemon_ErrorSize];
+
+int
+NdbDaemon_Make(const char* lockfile, const char* logfile, unsigned flags)
+{
+ // XXX do something
+ return 0;
+}
+
+#ifdef NDB_DAEMON_TEST
+
+int
+main()
+{
+ if (NdbDaemon_Make("test.pid", "test.log", 0) == -1) {
+ fprintf(stderr, "NdbDaemon_Make: %s\n", NdbDaemon_ErrorText);
+ return 1;
+ }
+ sleep(10);
+ return 0;
+}
+
+#endif
diff --git a/ndb/src/common/portlib/win32/NdbEnv.c b/ndb/src/common/portlib/win32/NdbEnv.c
new file mode 100644
index 00000000000..0df703a5e97
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbEnv.c
@@ -0,0 +1,33 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbEnv.h"
+#include <string.h>
+#include <stdlib.h>
+
+const char* NdbEnv_GetEnv(const char* name, char * buf, int buflen)
+{
+ char* p = NULL;
+ p = getenv(name);
+
+ if (p != NULL && buf != NULL){
+ strncpy(buf, p, buflen);
+ buf[buflen-1] = 0;
+ }
+ return p;
+}
+
diff --git a/ndb/src/common/portlib/win32/NdbHost.c b/ndb/src/common/portlib/win32/NdbHost.c
new file mode 100644
index 00000000000..f91dd1a531c
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbHost.c
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbHost.h"
+#include <windows.h>
+#include <process.h>
+
+
+int NdbHost_GetHostName(char* buf)
+{
+ /* We must initialize TCP/IP if we want to call gethostname */
+ WORD wVersionRequested;
+ WSADATA wsaData;
+ int err;
+
+ wVersionRequested = MAKEWORD( 2, 0 );
+ err = WSAStartup( wVersionRequested, &wsaData );
+ if ( err != 0 ) {
+ /**
+ * Tell the user that we couldn't find a usable
+ * WinSock DLL.
+ */
+ return -1;
+ }
+
+ /* Get host name */
+ if(gethostname(buf, MAXHOSTNAMELEN))
+ {
+ return -1;
+ }
+ return 0;
+}
+
+
+int NdbHost_GetProcessId(void)
+{
+ return _getpid();
+}
+
diff --git a/ndb/src/common/portlib/win32/NdbMem.c b/ndb/src/common/portlib/win32/NdbMem.c
new file mode 100644
index 00000000000..274dc31353f
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbMem.c
@@ -0,0 +1,237 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <windows.h>
+#include <assert.h>
+#include <NdbStdio.h>
+
+#include "NdbMem.h"
+
+
+struct AWEINFO
+{
+ SIZE_T dwSizeInBytesRequested;
+ ULONG_PTR nNumberOfPagesRequested;
+ ULONG_PTR nNumberOfPagesActual;
+ ULONG_PTR nNumberOfPagesFreed;
+ ULONG_PTR* pnPhysicalMemoryPageArray;
+ void* pRegionReserved;
+};
+
+const size_t cNdbMem_nMaxAWEinfo = 256;
+size_t gNdbMem_nAWEinfo = 0;
+
+struct AWEINFO* gNdbMem_pAWEinfo = 0;
+
+
+void ShowLastError(const char* szContext, const char* szFunction)
+{
+ DWORD dwError = GetLastError();
+ LPVOID lpMsgBuf;
+ FormatMessage(
+ FORMAT_MESSAGE_ALLOCATE_BUFFER |
+ FORMAT_MESSAGE_FROM_SYSTEM |
+ FORMAT_MESSAGE_IGNORE_INSERTS,
+ NULL,
+ dwError,
+ MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
+ (LPTSTR)&lpMsgBuf,
+ 0,
+ NULL
+ );
+ printf("%s : %s failed : %lu : %s\n", szContext, szFunction, dwError, (char*)lpMsgBuf);
+ LocalFree(lpMsgBuf);
+}
+
+
+
+void NdbMem_Create()
+{
+ // Address Windowing Extensions
+ struct PRIVINFO
+ {
+ DWORD Count;
+ LUID_AND_ATTRIBUTES Privilege[1];
+ } Info;
+
+ HANDLE hProcess = GetCurrentProcess();
+ HANDLE hToken;
+ if(!OpenProcessToken(hProcess, TOKEN_ADJUST_PRIVILEGES, &hToken))
+ {
+ ShowLastError("NdbMem_Create", "OpenProcessToken");
+ }
+
+ Info.Count = 1;
+ Info.Privilege[0].Attributes = SE_PRIVILEGE_ENABLED;
+ if(!LookupPrivilegeValue(0, SE_LOCK_MEMORY_NAME, &(Info.Privilege[0].Luid)))
+ {
+ ShowLastError("NdbMem_Create", "LookupPrivilegeValue");
+ }
+
+ if(!AdjustTokenPrivileges(hToken, FALSE, (PTOKEN_PRIVILEGES)&Info, 0, 0, 0))
+ {
+ ShowLastError("NdbMem_Create", "AdjustTokenPrivileges");
+ }
+
+ if(!CloseHandle(hToken))
+ {
+ ShowLastError("NdbMem_Create", "CloseHandle");
+ }
+
+ return;
+}
+
+void NdbMem_Destroy()
+{
+ /* Do nothing */
+ return;
+}
+
+void* NdbMem_Allocate(size_t size)
+{
+ // Address Windowing Extensions
+ struct AWEINFO* pAWEinfo;
+ HANDLE hProcess;
+ SYSTEM_INFO sysinfo;
+
+ if(!gNdbMem_pAWEinfo)
+ {
+ gNdbMem_pAWEinfo = VirtualAlloc(0,
+ sizeof(struct AWEINFO)*cNdbMem_nMaxAWEinfo,
+ MEM_COMMIT|MEM_RESERVE, PAGE_READWRITE);
+ }
+
+ assert(gNdbMem_nAWEinfo < cNdbMem_nMaxAWEinfo);
+ pAWEinfo = gNdbMem_pAWEinfo+gNdbMem_nAWEinfo++;
+
+ hProcess = GetCurrentProcess();
+ GetSystemInfo(&sysinfo);
+ pAWEinfo->nNumberOfPagesRequested = (size+sysinfo.dwPageSize-1)/sysinfo.dwPageSize;
+ pAWEinfo->pnPhysicalMemoryPageArray = VirtualAlloc(0,
+ sizeof(ULONG_PTR)*pAWEinfo->nNumberOfPagesRequested,
+ MEM_COMMIT|MEM_RESERVE, PAGE_READWRITE);
+ pAWEinfo->nNumberOfPagesActual = pAWEinfo->nNumberOfPagesRequested;
+ if(!AllocateUserPhysicalPages(hProcess, &(pAWEinfo->nNumberOfPagesActual), pAWEinfo->pnPhysicalMemoryPageArray))
+ {
+ ShowLastError("NdbMem_Allocate", "AllocateUserPhysicalPages");
+ return 0;
+ }
+ if(pAWEinfo->nNumberOfPagesRequested != pAWEinfo->nNumberOfPagesActual)
+ {
+ ShowLastError("NdbMem_Allocate", "nNumberOfPagesRequested != nNumberOfPagesActual");
+ return 0;
+ }
+
+ pAWEinfo->dwSizeInBytesRequested = size;
+ pAWEinfo->pRegionReserved = VirtualAlloc(0, pAWEinfo->dwSizeInBytesRequested, MEM_RESERVE | MEM_PHYSICAL, PAGE_READWRITE);
+ if(!pAWEinfo->pRegionReserved)
+ {
+ ShowLastError("NdbMem_Allocate", "VirtualAlloc");
+ return 0;
+ }
+
+ if(!MapUserPhysicalPages(pAWEinfo->pRegionReserved, pAWEinfo->nNumberOfPagesActual, pAWEinfo->pnPhysicalMemoryPageArray))
+ {
+ ShowLastError("NdbMem_Allocate", "MapUserPhysicalPages");
+ return 0;
+ }
+
+ /*
+ printf("allocate AWE memory: %lu bytes, %lu pages, address %lx\n",
+ pAWEinfo->dwSizeInBytesRequested,
+ pAWEinfo->nNumberOfPagesActual,
+ pAWEinfo->pRegionReserved);
+ */
+ return pAWEinfo->pRegionReserved;
+}
+
+
+void* NdbMem_AllocateAlign(size_t size, size_t alignment)
+{
+ /*
+ return (void*)memalign(alignment, size);
+ TEMP fix
+ */
+ return NdbMem_Allocate(size);
+}
+
+
+void NdbMem_Free(void* ptr)
+{
+ // VirtualFree(ptr, 0, MEM_DECOMMIT|MEM_RELEASE);
+
+ // Address Windowing Extensions
+ struct AWEINFO* pAWEinfo = 0;
+ size_t i;
+ HANDLE hProcess;
+
+ for(i=0; i<gNdbMem_nAWEinfo; ++i)
+ {
+ if(ptr==gNdbMem_pAWEinfo[i].pRegionReserved)
+ {
+ pAWEinfo = gNdbMem_pAWEinfo+i;
+ }
+ }
+ if(!pAWEinfo)
+ {
+ ShowLastError("NdbMem_Free", "ptr is not AWE memory");
+ }
+
+ hProcess = GetCurrentProcess();
+ if(!MapUserPhysicalPages(ptr, pAWEinfo->nNumberOfPagesActual, 0))
+ {
+ ShowLastError("NdbMem_Free", "MapUserPhysicalPages");
+ }
+
+ if(!VirtualFree(ptr, 0, MEM_RELEASE))
+ {
+ ShowLastError("NdbMem_Free", "VirtualFree");
+ }
+
+ pAWEinfo->nNumberOfPagesFreed = pAWEinfo->nNumberOfPagesActual;
+ if(!FreeUserPhysicalPages(hProcess, &(pAWEinfo->nNumberOfPagesFreed), pAWEinfo->pnPhysicalMemoryPageArray))
+ {
+ ShowLastError("NdbMem_Free", "FreeUserPhysicalPages");
+ }
+
+ VirtualFree(pAWEinfo->pnPhysicalMemoryPageArray, 0, MEM_DECOMMIT|MEM_RELEASE);
+}
+
+
+int NdbMem_MemLockAll()
+{
+ /*
+ HANDLE hProcess = GetCurrentProcess();
+ SIZE_T nMinimumWorkingSetSize;
+ SIZE_T nMaximumWorkingSetSize;
+ GetProcessWorkingSetSize(hProcess, &nMinimumWorkingSetSize, &nMaximumWorkingSetSize);
+ ndbout << "nMinimumWorkingSetSize=" << nMinimumWorkingSetSize << ", nMaximumWorkingSetSize=" << nMaximumWorkingSetSize << endl;
+
+ SetProcessWorkingSetSize(hProcess, 50000000, 100000000);
+
+ GetProcessWorkingSetSize(hProcess, &nMinimumWorkingSetSize, &nMaximumWorkingSetSize);
+ ndbout << "nMinimumWorkingSetSize=" << nMinimumWorkingSetSize << ", nMaximumWorkingSetSize=" << nMaximumWorkingSetSize << endl;
+ */
+ return -1;
+}
+
+int NdbMem_MemUnlockAll()
+{
+ //VirtualUnlock();
+ return -1;
+}
+
diff --git a/ndb/src/common/portlib/win32/NdbMutex.c b/ndb/src/common/portlib/win32/NdbMutex.c
new file mode 100644
index 00000000000..c93384d91db
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbMutex.c
@@ -0,0 +1,78 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <winsock2.h>
+#include <ws2tcpip.h>
+#include <windows.h>
+#include <time.h>
+#include <assert.h>
+
+#include "NdbMutex.h"
+
+
+NdbMutex* NdbMutex_Create(void)
+{
+ NdbMutex* pNdbMutex = (NdbMutex*)malloc(sizeof(NdbMutex));
+ if(!pNdbMutex)
+ return 0;
+
+ InitializeCriticalSection(pNdbMutex);
+ return pNdbMutex;
+}
+
+
+int NdbMutex_Destroy(NdbMutex* p_mutex)
+{
+ if(!p_mutex)
+ return -1;
+
+ DeleteCriticalSection(p_mutex);
+ free(p_mutex);
+ return 0;
+}
+
+
+int NdbMutex_Lock(NdbMutex* p_mutex)
+{
+ if(!p_mutex)
+ return -1;
+
+ EnterCriticalSection (p_mutex);
+ return 0;
+}
+
+
+int NdbMutex_Unlock(NdbMutex* p_mutex)
+{
+ if(!p_mutex)
+ return -1;
+
+ LeaveCriticalSection(p_mutex);
+ return 0;
+}
+
+
+int NdbMutex_Trylock(NdbMutex* p_mutex)
+{
+ int result = -1;
+ if(p_mutex)
+ {
+ result = (TryEnterCriticalSection(p_mutex) ? 0 : -1);
+ }
+ return result;
+}
+
diff --git a/ndb/src/common/portlib/win32/NdbSleep.c b/ndb/src/common/portlib/win32/NdbSleep.c
new file mode 100644
index 00000000000..ac0f44dd07f
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbSleep.c
@@ -0,0 +1,35 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbSleep.h"
+
+#include <windows.h>
+
+
+int
+NdbSleep_MilliSleep(int milliseconds)
+{
+ Sleep(milliseconds);
+ return 0;
+}
+
+int
+NdbSleep_SecSleep(int seconds)
+{
+ return NdbSleep_MilliSleep(seconds*1000);
+}
+
diff --git a/ndb/src/common/portlib/win32/NdbTCP.c b/ndb/src/common/portlib/win32/NdbTCP.c
new file mode 100644
index 00000000000..483a53bd606
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbTCP.c
@@ -0,0 +1,39 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbTCP.h"
+
+int
+Ndb_getInAddr(struct in_addr * dst, const char *address)
+{
+ struct hostent * hostPtr;
+
+ /* Try it as aaa.bbb.ccc.ddd. */
+ dst->s_addr = inet_addr(address);
+ if (dst->s_addr != -1) {
+ return 0;
+ }
+
+ hostPtr = gethostbyname(address);
+ if (hostPtr != NULL) {
+ dst->s_addr = ((struct in_addr *) *hostPtr->h_addr_list)->s_addr;
+ return 0;
+ }
+
+ return -1;
+}
+
diff --git a/ndb/src/common/portlib/win32/NdbThread.c b/ndb/src/common/portlib/win32/NdbThread.c
new file mode 100644
index 00000000000..ae3c74be70d
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbThread.c
@@ -0,0 +1,118 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <windows.h>
+#include <process.h>
+#include <assert.h>
+
+#include "NdbThread.h"
+
+
+#define MAX_THREAD_NAME 16
+
+typedef unsigned (WINAPI* NDB_WIN32_THREAD_FUNC)(void*);
+
+
+struct NdbThread
+{
+ HANDLE hThread;
+ unsigned nThreadId;
+ char thread_name[MAX_THREAD_NAME];
+};
+
+
+struct NdbThread* NdbThread_Create(NDB_THREAD_FUNC *p_thread_func,
+ NDB_THREAD_ARG *p_thread_arg,
+ const NDB_THREAD_STACKSIZE thread_stack_size,
+ const char* p_thread_name,
+ NDB_THREAD_PRIO thread_prio)
+{
+ struct NdbThread* tmpThread;
+ unsigned initflag;
+ int nPriority = 0;
+
+ if(!p_thread_func)
+ return 0;
+
+ tmpThread = (struct NdbThread*)malloc(sizeof(struct NdbThread));
+ if(!tmpThread)
+ return 0;
+
+ strncpy((char*)&tmpThread->thread_name, p_thread_name, MAX_THREAD_NAME);
+
+ switch(thread_prio)
+ {
+ case NDB_THREAD_PRIO_HIGHEST: nPriority=THREAD_PRIORITY_HIGHEST; break;
+ case NDB_THREAD_PRIO_HIGH: nPriority=THREAD_PRIORITY_ABOVE_NORMAL; break;
+ case NDB_THREAD_PRIO_MEAN: nPriority=THREAD_PRIORITY_NORMAL; break;
+ case NDB_THREAD_PRIO_LOW: nPriority=THREAD_PRIORITY_BELOW_NORMAL; break;
+ case NDB_THREAD_PRIO_LOWEST: nPriority=THREAD_PRIORITY_LOWEST; break;
+ }
+ initflag = (nPriority ? CREATE_SUSPENDED : 0);
+
+ tmpThread->hThread = (HANDLE)_beginthreadex(0, thread_stack_size,
+ (NDB_WIN32_THREAD_FUNC)p_thread_func, p_thread_arg,
+ initflag, &tmpThread->nThreadId);
+
+ if(nPriority && tmpThread->hThread)
+ {
+ SetThreadPriority(tmpThread->hThread, nPriority);
+ ResumeThread (tmpThread->hThread);
+ }
+
+ assert(tmpThread->hThread);
+ return tmpThread;
+}
+
+
+void NdbThread_Destroy(struct NdbThread** p_thread)
+{
+ CloseHandle((*p_thread)->hThread);
+ (*p_thread)->hThread = 0;
+ free(*p_thread);
+ *p_thread = 0;
+}
+
+
+int NdbThread_WaitFor(struct NdbThread* p_wait_thread, void** status)
+{
+ void *local_status = 0;
+ if (status == 0)
+ status = &local_status;
+
+ if(WaitForSingleObject(p_wait_thread->hThread, INFINITE) == WAIT_OBJECT_0
+ && GetExitCodeThread(p_wait_thread->hThread, (LPDWORD)status))
+ {
+ CloseHandle(p_wait_thread->hThread);
+ p_wait_thread->hThread = 0;
+ return 0;
+ }
+ return -1;
+}
+
+
+void NdbThread_Exit(int status)
+{
+ _endthreadex((DWORD) status);
+}
+
+
+int NdbThread_SetConcurrencyLevel(int level)
+{
+ return 0;
+}
+
diff --git a/ndb/src/common/portlib/win32/NdbTick.c b/ndb/src/common/portlib/win32/NdbTick.c
new file mode 100644
index 00000000000..e3a67d8437d
--- /dev/null
+++ b/ndb/src/common/portlib/win32/NdbTick.c
@@ -0,0 +1,64 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <windows.h>
+#include "NdbTick.h"
+
+/*
+#define FILETIME_PER_MICROSEC 10
+#define FILETIME_PER_MILLISEC 10000
+#define FILETIME_PER_SEC 10000000
+
+
+NDB_TICKS NdbTick_CurrentMillisecond(void)
+{
+ ULONGLONG ullTime;
+ GetSystemTimeAsFileTime((LPFILETIME)&ullTime);
+ return (ullTime / FILETIME_PER_MILLISEC);
+}
+
+int
+NdbTick_CurrentMicrosecond(NDB_TICKS * secs, Uint32 * micros)
+{
+ ULONGLONG ullTime;
+ GetSystemTimeAsFileTime((LPFILETIME)&ullTime);
+ *secs = (ullTime / FILETIME_PER_SEC);
+ *micros = (Uint32)((ullTime % FILETIME_PER_SEC) / FILETIME_PER_MICROSEC);
+ return 0;
+}
+*/
+
+
+NDB_TICKS NdbTick_CurrentMillisecond(void)
+{
+ LARGE_INTEGER liCount, liFreq;
+ QueryPerformanceCounter(&liCount);
+ QueryPerformanceFrequency(&liFreq);
+ return (liCount.QuadPart*1000) / liFreq.QuadPart;
+}
+
+int
+NdbTick_CurrentMicrosecond(NDB_TICKS * secs, Uint32 * micros)
+{
+ LARGE_INTEGER liCount, liFreq;
+ QueryPerformanceCounter(&liCount);
+ QueryPerformanceFrequency(&liFreq);
+ *secs = liCount.QuadPart / liFreq.QuadPart;
+ liCount.QuadPart -= *secs * liFreq.QuadPart;
+ *micros = (liCount.QuadPart*1000000) / liFreq.QuadPart;
+ return 0;
+}
diff --git a/ndb/src/common/transporter/Makefile b/ndb/src/common/transporter/Makefile
new file mode 100644
index 00000000000..3bd23b627d3
--- /dev/null
+++ b/ndb/src/common/transporter/Makefile
@@ -0,0 +1,62 @@
+include .defs.mk
+
+TYPE := util
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := transporter
+DIRS := basictest perftest
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = \
+ Transporter.cpp \
+ SendBuffer.cpp \
+ SHM_Transporter.cpp \
+ TCP_Transporter.cpp \
+ TransporterRegistry.cpp \
+ Packer.cpp
+
+DIRS := basictest perftest
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include/kernel) \
+ -I$(call fixpath,$(NDB_TOP)/include/transporter)
+
+ifeq ($(NDB_OS), WIN32)
+SOURCES += SHM_Transporter.win32.cpp
+endif
+
+ifeq ($(NDB_OS), SOLARIS)
+SOURCES += SHM_Transporter.unix.cpp
+endif
+
+ifeq ($(NDB_OS), HPUX)
+SOURCES += SHM_Transporter.unix.cpp
+endif
+
+ifeq ($(NDB_OS), MACOSX)
+SOURCES += SHM_Transporter.unix.cpp
+endif
+
+ifeq ($(NDB_OS), IBMAIX)
+SOURCES += SHM_Transporter.unix.cpp
+endif
+
+ifeq ($(NDB_OS), TRU64X)
+SOURCES += SHM_Transporter.unix.cpp
+endif
+
+ifeq ($(NDB_OS), LINUX)
+SOURCES += SHM_Transporter.unix.cpp
+ifeq ($(NDB_SCI), Y)
+SOURCES += SCI_Transporter.cpp
+endif
+endif
+
+
+ifneq ($(findstring OSE, $(NDB_OS)),)
+ SOURCES += OSE_Transporter.cpp
+ SOURCES += OSE_Receiver.cpp
+endif
+
+
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/common/transporter/OSE_Receiver.cpp b/ndb/src/common/transporter/OSE_Receiver.cpp
new file mode 100644
index 00000000000..558dee92d8d
--- /dev/null
+++ b/ndb/src/common/transporter/OSE_Receiver.cpp
@@ -0,0 +1,360 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include "OSE_Receiver.hpp"
+#include "OSE_Transporter.hpp"
+#include "TransporterCallback.hpp"
+#include <TransporterRegistry.hpp>
+#include "TransporterInternalDefinitions.hpp"
+#include <NdbStdio.h>
+
+OSE_Receiver::OSE_Receiver(TransporterRegistry * tr,
+ int _recBufSize,
+ NodeId _localNodeId) {
+ theTransporterRegistry = tr;
+
+ recBufSize = _recBufSize;
+ recBufReadIndex = 0;
+ recBufWriteIndex = 0;
+ receiveBuffer = new union SIGNAL * [recBufSize];
+
+ waitStackCount = 0;
+ waitStackSize = _recBufSize;
+ waitStack = new union SIGNAL * [waitStackSize];
+
+ nextSigId = new Uint32[MAX_NTRANSPORTERS];
+ for (int i = 0; i < MAX_NTRANSPORTERS; i++)
+ nextSigId[i] = 0;
+
+ phantomCreated = false;
+ localNodeId = _localNodeId;
+ snprintf(localHostName, sizeof(localHostName),
+ "ndb_node%d", localNodeId);
+
+ DEBUG("localNodeId = " << localNodeId << " -> localHostName = "
+ << localHostName);
+}
+
+OSE_Receiver::~OSE_Receiver(){
+ while(recBufReadIndex != recBufWriteIndex){
+ free_buf(&receiveBuffer[recBufReadIndex]);
+ recBufReadIndex = (recBufReadIndex + 1) % recBufSize;
+ }
+ delete [] receiveBuffer;
+ destroyPhantom();
+}
+
+PROCESS
+OSE_Receiver::createPhantom(){
+ redir.sig = 1;
+ redir.pid = current_process();
+
+ if(!phantomCreated){
+ phantomPid = create_process
+ (OS_PHANTOM, // Type
+ localHostName, // Name
+ NULL, // Entry point
+ 0, // Stack size
+ 0, // Prio - Not used
+ (OSTIME)0, // Timeslice - Not used
+ 0, // Block - current block
+ &redir,
+ (OSVECTOR)0, // vector
+ (OSUSER)0); // user
+ phantomCreated = true;
+ DEBUG("Created phantom pid: " << hex << phantomPid);
+ }
+ return phantomPid;
+}
+
+void
+OSE_Receiver::destroyPhantom(){
+ if(phantomCreated){
+ DEBUG("Destroying phantom pid: " << hex << phantomPid);
+ kill_proc(phantomPid);
+ phantomCreated = false;
+ }
+}
+
+static SIGSELECT PRIO_A_SIGNALS[] = { 6,
+ NDB_TRANSPORTER_PRIO_A,
+ NDB_TRANSPORTER_HUNT,
+ NDB_TRANSPORTER_CONNECT_REQ,
+ NDB_TRANSPORTER_CONNECT_REF,
+ NDB_TRANSPORTER_CONNECT_CONF,
+ NDB_TRANSPORTER_DISCONNECT_ORD
+};
+
+static SIGSELECT PRIO_B_SIGNALS[] = { 1,
+ NDB_TRANSPORTER_DATA
+};
+
+/**
+ * Check waitstack for signals that are next in sequence
+ * Put any found signal in receive buffer
+ * Returns true if one signal is found
+ */
+bool
+OSE_Receiver::checkWaitStack(NodeId _nodeId){
+
+ for(int i = 0; i < waitStackCount; i++){
+ if (waitStack[i]->dataSignal.senderNodeId == _nodeId &&
+ waitStack[i]->dataSignal.sigId == nextSigId[_nodeId]){
+
+ ndbout_c("INFO: signal popped from waitStack, sigId = %d",
+ waitStack[i]->dataSignal.sigId);
+
+ if(isFull()){
+ ndbout_c("ERROR: receiveBuffer is full");
+ reportError(callbackObj, _nodeId, TE_RECEIVE_BUFFER_FULL);
+ return false;
+ }
+
+ // The next signal was found, put it in the receive buffer
+ insertReceiveBuffer(waitStack[i]);
+
+ // Increase sequence id, set it to the next expected id
+ nextSigId[_nodeId]++;
+
+ // Move signals below up one step
+ for(int j = i; j < waitStackCount-1; j++)
+ waitStack[j] = waitStack[j+1];
+ waitStack[waitStackCount] = NULL;
+ waitStackCount--;
+
+ // return true since signal was found
+ return true;
+ }
+ }
+ return false;
+}
+
+/**
+ * Clear waitstack for signals from node with _nodeId
+ */
+void
+OSE_Receiver::clearWaitStack(NodeId _nodeId){
+
+ for(int i = 0; i < waitStackCount; i++){
+ if (waitStack[i]->dataSignal.senderNodeId == _nodeId){
+
+ // Free signal buffer
+ free_buf(&waitStack[i]);
+
+ // Move signals below up one step
+ for(int j = i; j < waitStackCount-1; j++)
+ waitStack[j] = waitStack[j+1];
+ waitStack[waitStackCount] = NULL;
+ waitStackCount--;
+ }
+ }
+ nextSigId[_nodeId] = 0;
+}
+
+
+inline
+void
+OSE_Receiver::insertWaitStack(union SIGNAL* _sig){
+ if (waitStackCount <= waitStackSize){
+ waitStack[waitStackCount] = _sig;
+ waitStackCount++;
+ } else {
+ ndbout_c("ERROR: waitStack is full");
+ reportError(callbackObj, localNodeId, TE_WAIT_STACK_FULL);
+ }
+}
+
+bool
+OSE_Receiver::doReceive(Uint32 timeOutMillis) {
+ if(isFull())
+ return false;
+
+ union SIGNAL * sig = receive_w_tmo(0,
+ PRIO_A_SIGNALS);
+ if(sig == NIL){
+ sig = receive_w_tmo(timeOutMillis,
+ PRIO_B_SIGNALS);
+ if(sig == NIL)
+ return false;
+ }
+
+ DEBUG("Received signal: " << sig->sigNo << " "
+ << sigNo2String(sig->sigNo));
+
+ switch(sig->sigNo){
+ case NDB_TRANSPORTER_PRIO_A:
+ {
+ OSE_Transporter * t = getTransporter(sig->dataSignal.senderNodeId);
+ if (t != 0 && t->isConnected()){
+ insertReceiveBuffer(sig);
+ } else {
+ free_buf(&sig);
+ }
+ }
+ break;
+ case NDB_TRANSPORTER_DATA:
+ {
+ OSE_Transporter * t = getTransporter(sig->dataSignal.senderNodeId);
+ if (t != 0 && t->isConnected()){
+ int nodeId = sig->dataSignal.senderNodeId;
+ Uint32 currSigId = sig->dataSignal.sigId;
+
+ /**
+ * Check if signal is the next in sequence
+ * nextSigId is always set to the next sigId to wait for
+ */
+ if (nextSigId[nodeId] == currSigId){
+
+ // Insert in receive buffer
+ insertReceiveBuffer(sig);
+
+ // Increase sequence id, set it to the next expected id
+ nextSigId[nodeId]++;
+
+ // Check if there are any signal in the wait stack
+ if (waitStackCount > 0){
+ while(checkWaitStack(nodeId));
+ }
+ } else {
+ // Signal was not received in correct order
+ // Check values and put it in the waitStack
+ ndbout_c("WARNING: sigId out of order,"
+ " currSigId = %d, nextSigId = %d",
+ currSigId, nextSigId[nodeId]);
+
+ if (currSigId < nextSigId[nodeId]){
+ // Current recieved sigId was smaller than nextSigId
+ // There is no use to put it in the waitStack
+ ndbout_c("ERROR: recieved sigId was smaller than nextSigId");
+ reportError(callbackObj, nodeId, TE_TOO_SMALL_SIGID);
+ return false;
+ }
+
+ if (currSigId > (nextSigId[nodeId] + waitStackSize)){
+ // Current sigId was larger than nextSigId + size of waitStack
+ // we can never "save" so many signal's on the stack
+ ndbout_c("ERROR: currSigId > (nextSigId + size of waitStack)");
+ reportError(callbackObj, nodeId, TE_TOO_LARGE_SIGID);
+ return false;
+ }
+
+ // Insert in wait stack
+ insertWaitStack(sig);
+ }
+ } else {
+ free_buf(&sig);
+ }
+ }
+ break;
+ case NDB_TRANSPORTER_HUNT:
+ {
+ NdbTransporterHunt * s = (NdbTransporterHunt*)sig;
+ OSE_Transporter * t = getTransporter(s->remoteNodeId);
+ if(t != 0)
+ t->huntReceived(s);
+ free_buf(&sig);
+ }
+ break;
+ case NDB_TRANSPORTER_CONNECT_REQ:
+ {
+ NdbTransporterConnectReq * s = (NdbTransporterConnectReq*)sig;
+ OSE_Transporter * t = getTransporter(s->senderNodeId);
+ if(t != 0){
+ if(t->connectReq(s)){
+ clearWaitStack(s->senderNodeId);
+ clearRecvBuffer(s->senderNodeId);
+ }
+ }
+ free_buf(&sig);
+ }
+ break;
+ case NDB_TRANSPORTER_CONNECT_REF:
+ {
+ NdbTransporterConnectRef * s = (NdbTransporterConnectRef*)sig;
+ OSE_Transporter * t = getTransporter(s->senderNodeId);
+ if(t != 0){
+ if(t->connectRef(s)){
+ clearWaitStack(s->senderNodeId);
+ clearRecvBuffer(s->senderNodeId);
+ }
+ }
+ free_buf(&sig);
+ }
+ break;
+ case NDB_TRANSPORTER_CONNECT_CONF:
+ {
+ NdbTransporterConnectConf * s = (NdbTransporterConnectConf*)sig;
+ OSE_Transporter * t = getTransporter(s->senderNodeId);
+ if(t != 0){
+ if(t->connectConf(s)){
+ clearWaitStack(s->senderNodeId);
+ clearRecvBuffer(s->senderNodeId);
+ }
+ }
+ free_buf(&sig);
+ }
+ break;
+ case NDB_TRANSPORTER_DISCONNECT_ORD:
+ {
+ NdbTransporterDisconnectOrd * s = (NdbTransporterDisconnectOrd*)sig;
+ OSE_Transporter * t = getTransporter(s->senderNodeId);
+ if(t != 0){
+ if(t->disconnectOrd(s)){
+ clearWaitStack(s->senderNodeId);
+ clearRecvBuffer(s->senderNodeId);
+ }
+ }
+ free_buf(&sig);
+ }
+ }
+ return true;
+}
+
+OSE_Transporter *
+OSE_Receiver::getTransporter(NodeId nodeId){
+ if(theTransporterRegistry->theTransporterTypes[nodeId] != tt_OSE_TRANSPORTER)
+ return 0;
+ return (OSE_Transporter *)
+ theTransporterRegistry->theTransporters[nodeId];
+}
+
+void
+OSE_Receiver::clearRecvBuffer(NodeId nodeId){
+ int tmpIndex = 0;
+ union SIGNAL** tmp = new union SIGNAL * [recBufSize];
+
+ /**
+ * Put all signal that I want to keep into tmp
+ */
+ while(recBufReadIndex != recBufWriteIndex){
+ if(receiveBuffer[recBufReadIndex]->dataSignal.senderNodeId != nodeId){
+ tmp[tmpIndex] = receiveBuffer[recBufReadIndex];
+ tmpIndex++;
+ } else {
+ free_buf(&receiveBuffer[recBufReadIndex]);
+ }
+ recBufReadIndex = (recBufReadIndex + 1) % recBufSize;
+ }
+
+ /**
+ * Put all signals that I kept back into receiveBuffer
+ */
+ for(int i = 0; i<tmpIndex; i++)
+ insertReceiveBuffer(tmp[i]);
+
+ delete [] tmp;
+}
diff --git a/ndb/src/common/transporter/OSE_Receiver.hpp b/ndb/src/common/transporter/OSE_Receiver.hpp
new file mode 100644
index 00000000000..1812ab51065
--- /dev/null
+++ b/ndb/src/common/transporter/OSE_Receiver.hpp
@@ -0,0 +1,119 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef OSE_RECEIVER_HPP
+#define OSE_RECEIVER_HPP
+
+#include "ose.h"
+#include "OSE_Signals.hpp"
+#include <kernel_types.h>
+
+class OSE_Receiver {
+public:
+ OSE_Receiver(class TransporterRegistry *,
+ int recBufSize,
+ NodeId localNodeId);
+
+ ~OSE_Receiver();
+
+ bool hasData() const ;
+ bool isFull() const ;
+
+ Uint32 getReceiveData(NodeId * remoteNodeId,
+ Uint32 ** readPtr);
+
+ void updateReceiveDataPtr(Uint32 szRead);
+
+ bool doReceive(Uint32 timeOutMillis);
+
+ PROCESS createPhantom();
+ void destroyPhantom();
+
+private:
+ class TransporterRegistry * theTransporterRegistry;
+
+ NodeId localNodeId;
+ char localHostName[255];
+
+ bool phantomCreated;
+ PROCESS phantomPid;
+ struct OS_redir_entry redir;
+
+ int recBufReadIndex;
+ int recBufWriteIndex;
+ int recBufSize;
+ union SIGNAL **receiveBuffer;
+
+ // Stack for signals that are received out of order
+ int waitStackCount;
+ int waitStackSize;
+ union SIGNAL** waitStack;
+
+ // Counters for the next signal id
+ Uint32* nextSigId;
+
+ class OSE_Transporter * getTransporter(NodeId nodeId);
+
+ void insertReceiveBuffer(union SIGNAL * _sig);
+ void clearRecvBuffer(NodeId _nodeId);
+ bool checkWaitStack(NodeId _nodeId);
+ void clearWaitStack(NodeId _nodeId);
+ void insertWaitStack(union SIGNAL* _sig);
+};
+
+inline
+bool
+OSE_Receiver::hasData () const {
+ return recBufReadIndex != recBufWriteIndex;
+}
+
+inline
+bool
+OSE_Receiver::isFull () const {
+ return ((recBufWriteIndex + 1) % recBufSize) == recBufWriteIndex;
+}
+
+inline
+Uint32
+OSE_Receiver::getReceiveData(NodeId * remoteNodeId,
+ Uint32 ** readPtr){
+ NdbTransporterData *s = (NdbTransporterData *)receiveBuffer[recBufReadIndex];
+ if(recBufReadIndex != recBufWriteIndex){
+ * remoteNodeId = s->senderNodeId;
+ * readPtr = &s->data[0];
+ return s->length;
+ }
+ return 0;
+}
+
+inline
+void
+OSE_Receiver::updateReceiveDataPtr(Uint32 bytesRead){
+ if(bytesRead != 0){
+ free_buf(&receiveBuffer[recBufReadIndex]);
+ recBufReadIndex = (recBufReadIndex + 1) % recBufSize;
+ }
+}
+
+inline
+void
+OSE_Receiver::insertReceiveBuffer(union SIGNAL * _sig){
+ receiveBuffer[recBufWriteIndex] = _sig;
+ recBufWriteIndex = (recBufWriteIndex + 1) % recBufSize;
+}
+
+
+#endif
diff --git a/ndb/src/common/transporter/OSE_Signals.hpp b/ndb/src/common/transporter/OSE_Signals.hpp
new file mode 100644
index 00000000000..3f6cc07b473
--- /dev/null
+++ b/ndb/src/common/transporter/OSE_Signals.hpp
@@ -0,0 +1,144 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef OSE_SIGNALS_HPP
+#define OSE_SIGNALS_HPP
+
+#include <ose.h>
+#include <kernel_types.h>
+
+#define NDB_TRANSPORTER_SIGBASE 3000
+
+#define NDB_TRANSPORTER_DATA (NDB_TRANSPORTER_SIGBASE + 1) /* !-SIGNO(struct NdbTransporterData)-! */
+#define NDB_TRANSPORTER_HUNT (NDB_TRANSPORTER_SIGBASE + 2) /* !-SIGNO(struct NdbTransporterHunt)-! */
+#define NDB_TRANSPORTER_CONNECT_REQ (NDB_TRANSPORTER_SIGBASE + 3) /* !-SIGNO(struct NdbTransporterConnectReq)-! */
+#define NDB_TRANSPORTER_CONNECT_REF (NDB_TRANSPORTER_SIGBASE + 4) /* !-SIGNO(struct NdbTransporterConnectRef)-! */
+#define NDB_TRANSPORTER_CONNECT_CONF (NDB_TRANSPORTER_SIGBASE + 5) /* !-SIGNO(struct NdbTransporterConnectConf)-! */
+#define NDB_TRANSPORTER_DISCONNECT_ORD (NDB_TRANSPORTER_SIGBASE + 6) /* !-SIGNO(struct NdbTransporterDisconnectOrd)-! */
+#define NDB_TRANSPORTER_PRIO_A (NDB_TRANSPORTER_SIGBASE + 7)
+
+inline
+const char *
+sigNo2String(SIGSELECT sigNo){
+ switch(sigNo){
+ case NDB_TRANSPORTER_PRIO_A:
+ return "PRIO_A_DATA";
+ break;
+ case NDB_TRANSPORTER_DATA:
+ return "PRIO_B_DATA";
+ break;
+ case NDB_TRANSPORTER_HUNT:
+ return "HUNT";
+ break;
+ case NDB_TRANSPORTER_CONNECT_REQ:
+ return "CONNECT_REQ";
+ break;
+ case NDB_TRANSPORTER_CONNECT_REF:
+ return "CONNECT_REF";
+ break;
+ case NDB_TRANSPORTER_CONNECT_CONF:
+ return "CONNECT_CONF";
+ break;
+ case NDB_TRANSPORTER_DISCONNECT_ORD:
+ return "DISCONNECT_ORD";
+ break;
+ }
+ return "UNKNOWN";
+}
+
+struct NdbTransporterData
+{
+ SIGSELECT sigNo;
+ Uint32 sigId; // Sequence number for this signal
+ Uint32 senderNodeId;
+ Uint32 length;
+ Uint32 data[1];
+};
+
+struct NdbTransporterData_PrioA
+{
+ SIGSELECT sigNo;
+ Uint32 sigId; // Sequence number for this signal
+ Uint32 senderNodeId;
+ Uint32 length;
+ Uint32 data[1];
+};
+
+struct NdbTransporterHunt
+{
+ SIGSELECT sigNo;
+ NodeId remoteNodeId;
+};
+
+
+struct NdbTransporterConnectReq
+{
+ SIGSELECT sigNo;
+ NodeId remoteNodeId;
+ NodeId senderNodeId;
+};
+
+
+struct NdbTransporterConnectConf
+{
+ SIGSELECT sigNo;
+ NodeId remoteNodeId;
+ NodeId senderNodeId;
+};
+
+struct NdbTransporterConnectRef
+{
+ SIGSELECT sigNo;
+ NodeId remoteNodeId;
+ NodeId senderNodeId;
+ Uint32 reason;
+
+ /**
+ * Node is not accepting connections
+ */
+ static const Uint32 INVALID_STATE = 1;
+};
+
+struct NdbTransporterDisconnectOrd
+{
+ SIGSELECT sigNo;
+ NodeId senderNodeId;
+ Uint32 reason;
+
+ /**
+ * Process died
+ */
+ static const Uint32 PROCESS_DIED = 1;
+
+ /**
+ * Ndb disconnected
+ */
+ static const Uint32 NDB_DISCONNECT = 2;
+};
+
+union SIGNAL
+{
+ SIGSELECT sigNo;
+ struct NdbTransporterData dataSignal;
+ struct NdbTransporterData prioAData;
+ struct NdbTransporterHunt ndbHunt;
+ struct NdbTransporterConnectReq ndbConnectReq;
+ struct NdbTransporterConnectRef ndbConnectRef;
+ struct NdbTransporterConnectConf ndbConnectConf;
+ struct NdbTransporterDisconnectOrd ndbDisconnect;
+};
+
+#endif
diff --git a/ndb/src/common/transporter/OSE_Transporter.cpp b/ndb/src/common/transporter/OSE_Transporter.cpp
new file mode 100644
index 00000000000..a7a5ed81ce2
--- /dev/null
+++ b/ndb/src/common/transporter/OSE_Transporter.cpp
@@ -0,0 +1,487 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <ose.h>
+#include "OSE_Transporter.hpp"
+#include "OSE_Signals.hpp"
+
+#include <TransporterCallback.hpp>
+#include "TransporterInternalDefinitions.hpp"
+
+#include <NdbMutex.h>
+
+#include <NdbStdio.h>
+#include <NdbHost.h>
+#include <NdbOut.hpp>
+#include <time.h>
+#include <assert.h>
+
+OSE_Transporter::OSE_Transporter(int _prioASignalSize,
+ int _prioBSignalSize,
+ NodeId localNodeId,
+ const char * lHostName,
+ NodeId remoteNodeId,
+ const char * rHostName,
+ int byteorder,
+ bool compression,
+ bool checksum,
+ bool signalId,
+ Uint32 reportFreq) :
+ Transporter(localNodeId,
+ remoteNodeId,
+ byteorder,
+ compression,
+ checksum,
+ signalId),
+ isServer(localNodeId < remoteNodeId)
+{
+
+ signalIdCounter = 0;
+ prioBSignalSize = _prioBSignalSize;
+
+ if (strcmp(lHostName, rHostName) == 0){
+ snprintf(remoteNodeName, sizeof(remoteNodeName),
+ "ndb_node%d", remoteNodeId);
+ } else {
+ snprintf(remoteNodeName, sizeof(remoteNodeName),
+ "%s/ndb_node%d", rHostName, remoteNodeId);
+ }
+
+ prioBSignal = NIL;
+}
+
+OSE_Transporter::~OSE_Transporter() {
+
+#if 0
+ /**
+ * Don't free these buffers since they have already been freed
+ * when the process allocating them died (wild pointers)
+ */
+ if(prioBSignal != NIL)
+ free_buf(&prioBSignal);
+#endif
+}
+
+bool
+OSE_Transporter::initTransporter() {
+
+ struct OS_pcb * pcb = get_pcb(current_process());
+ if(pcb != NULL){
+ if(pcb->type != OS_ILLEGAL){
+ if(prioBSignalSize > pcb->max_sigsize){
+ DEBUG("prioBSignalSize(" << prioBSignalSize << ") > max_sigsize("
+ << pcb->max_sigsize << ") using max_sigsize");
+ prioBSignalSize = pcb->max_sigsize;
+ }
+ }
+ free_buf((union SIGNAL **)&pcb);
+ }
+
+ maxPrioBDataSize = prioBSignalSize;
+ maxPrioBDataSize -= (sizeof(NdbTransporterData) + MAX_MESSAGE_SIZE - 4);
+
+ if(maxPrioBDataSize < 0){
+
+#ifdef DEBUG_TRANSPORTER
+ printf("maxPrioBDataSize < 0 %d\n",
+ maxPrioBDataSize);
+#endif
+ return false;
+ }
+
+ initSignals();
+
+ return true;
+}
+
+void
+OSE_Transporter::initSignals(){
+ if(prioBSignal == NIL){
+ prioBSignal = alloc(prioBSignalSize, NDB_TRANSPORTER_DATA);
+ prioBInsertPtr = &prioBSignal->dataSignal.data[0];
+
+ prioBSignal->dataSignal.length = 0;
+ prioBSignal->dataSignal.senderNodeId = localNodeId;
+ }
+ dataToSend = 0;
+}
+
+NdbTransporterData *
+OSE_Transporter::allocPrioASignal(Uint32 messageLenBytes) const
+{
+
+ const Uint32 lenBytes = messageLenBytes + sizeof(NdbTransporterData) - 4;
+
+ NdbTransporterData * sig =
+ (NdbTransporterData*)alloc(lenBytes, NDB_TRANSPORTER_PRIO_A);
+
+ sig->length = 0;
+ sig->senderNodeId = localNodeId;
+
+ return sig;
+}
+
+Uint32 *
+OSE_Transporter::getWritePtr(Uint32 lenBytes, Uint32 prio){
+ if(prio >= 1){
+ prio = 1;
+ insertPtr = prioBInsertPtr;
+ signal = (NdbTransporterData*)prioBSignal;
+ } else {
+ signal = allocPrioASignal(lenBytes);
+ insertPtr = &signal->data[0];
+ }
+ return insertPtr;
+}
+
+void
+OSE_Transporter::updateWritePtr(Uint32 lenBytes, Uint32 prio){
+
+ Uint32 bufferSize = signal->length;
+ bufferSize += lenBytes;
+ signal->length = bufferSize;
+ if(prio >= 1){
+ prioBInsertPtr += (lenBytes / 4);
+ if(bufferSize >= maxPrioBDataSize)
+ doSend();
+ } else {
+ /**
+ * Prio A signal are sent directly
+ */
+ signal->sigId = 0;
+
+ ::send((union SIGNAL**)&signal, remoteNodePid);
+ }
+}
+
+#if 0
+int getSeq(int _seq){
+ if (_seq > 0){
+ switch (_seq % 100){
+ case 10:
+ return _seq - 1;
+ case 9:
+ return _seq + 1;
+ default:
+ return _seq;
+ }
+ }else{
+ return _seq;
+ }
+}
+int getSeq(int _seq){
+
+ switch (_seq % 40){
+ case 10:
+ return _seq-4;
+ case 9:
+ return _seq-2;
+ case 8:
+ return _seq;
+ case 7:
+ return _seq+2;
+ case 6:
+ return _seq+4;
+
+
+ case 30:
+ return _seq-9;
+ case 29:
+ return _seq-7;
+ case 28:
+ return _seq-5;
+ case 27:
+ return _seq-3;
+ case 26:
+ return _seq-1;
+ case 25:
+ return _seq+1;
+ case 24:
+ return _seq+3;
+ case 23:
+ return _seq+5;
+ case 22:
+ return _seq+7;
+ case 21:
+ return _seq+9;
+
+ default:
+ return _seq;
+
+ }
+}
+#endif
+
+void
+OSE_Transporter::doSend() {
+ /**
+ * restore is always called to make sure the signal buffer is taken over
+ * by a process that is alive, this will otherwise lead to that these buffers
+ * are removed when the process that allocated them dies
+ */
+ restore(prioBSignal);
+ if(prioBSignal->dataSignal.length > 0){
+
+ prioBSignal->dataSignal.sigId = signalIdCounter;
+ signalIdCounter++;
+
+ ::send(&prioBSignal, remoteNodePid);
+ }
+
+ initSignals();
+}
+
+void
+OSE_Transporter::doConnect() {
+
+ NdbMutex_Lock(theMutexPtr);
+ if(_connecting || _disconnecting || _connected){
+ NdbMutex_Unlock(theMutexPtr);
+ return;
+ }
+
+ _connecting = true;
+ signalIdCounter = 0;
+
+ if(isServer){
+ DEBUG("Waiting for connect req: ");
+ state = WAITING_FOR_CONNECT_REQ;
+ } else {
+ state = WAITING_FOR_HUNT;
+
+ DEBUG("Hunting for: " << remoteNodeName);
+
+ union SIGNAL* huntsig;
+ huntsig = alloc(sizeof(NdbTransporterHunt), NDB_TRANSPORTER_HUNT);
+ huntsig->ndbHunt.remoteNodeId = remoteNodeId;
+ hunt(remoteNodeName, 0, NULL, &huntsig);
+ }
+ NdbMutex_Unlock(theMutexPtr);
+}
+
+void
+OSE_Transporter::doDisconnect() {
+ NdbMutex_Lock(theMutexPtr);
+
+ switch(state){
+ case DISCONNECTED:
+ case WAITING_FOR_HUNT:
+ case WAITING_FOR_CONNECT_REQ:
+ case WAITING_FOR_CONNECT_CONF:
+ break;
+ case CONNECTED:
+ {
+#if 0
+ /**
+ * There should not be anything in the buffer that needs to be sent here
+ */
+ DEBUG("Doing send before disconnect");
+ doSend();
+#endif
+ union SIGNAL * sig = alloc(sizeof(NdbTransporterDisconnectOrd),
+ NDB_TRANSPORTER_DISCONNECT_ORD);
+ sig->ndbDisconnect.senderNodeId = localNodeId;
+ sig->ndbDisconnect.reason = NdbTransporterDisconnectOrd::NDB_DISCONNECT;
+ ::send(&sig, remoteNodePid);
+ detach(&remoteNodeRef);
+
+ }
+ break;
+ }
+ state = DISCONNECTED;
+
+ _connected = false;
+ _connecting = false;
+ _disconnecting = false;
+
+ NdbMutex_Unlock(theMutexPtr);
+}
+
+void
+OSE_Transporter::huntReceived(struct NdbTransporterHunt * sig){
+ if(isServer){
+ WARNING("Hunt received for server: remoteNodeId: " <<
+ sig->remoteNodeId);
+ return;
+ }
+
+ if(state != WAITING_FOR_HUNT){
+ WARNING("Hunt received while in state: " << state);
+ return;
+ }
+ remoteNodePid = sender((union SIGNAL**)&sig);
+ union SIGNAL * signal = alloc(sizeof(NdbTransporterConnectReq),
+ NDB_TRANSPORTER_CONNECT_REQ);
+ signal->ndbConnectReq.remoteNodeId = remoteNodeId;
+ signal->ndbConnectReq.senderNodeId = localNodeId;
+
+ DEBUG("Sending connect req to pid: " << hex << remoteNodePid);
+
+ ::send(&signal, remoteNodePid);
+ state = WAITING_FOR_CONNECT_CONF;
+ return;
+}
+
+bool
+OSE_Transporter::connectReq(struct NdbTransporterConnectReq * sig){
+ if(!isServer){
+ WARNING("OSE Connect Req received for client: senderNodeId: " <<
+ sig->senderNodeId);
+ return false;
+ }
+
+ if(state != WAITING_FOR_CONNECT_REQ){
+ PROCESS pid = sender((union SIGNAL**)&sig);
+ union SIGNAL * signal = alloc(sizeof(NdbTransporterConnectRef),
+ NDB_TRANSPORTER_CONNECT_REF);
+ signal->ndbConnectRef.senderNodeId = localNodeId;
+ signal->ndbConnectRef.reason = NdbTransporterConnectRef::INVALID_STATE;
+
+ DEBUG("Sending connect ref to pid: " << hex << pid);
+
+ ::send(&signal, pid);
+ return false;
+ }
+
+ NdbMutex_Lock(theMutexPtr);
+
+ if(prioBSignal != NIL){
+ restore(prioBSignal);
+ free_buf(&prioBSignal);
+ }
+ initSignals();
+
+ remoteNodePid = sender((union SIGNAL**)&sig);
+ union SIGNAL * signal = alloc(sizeof(NdbTransporterConnectRef),
+ NDB_TRANSPORTER_CONNECT_CONF);
+ signal->ndbConnectConf.senderNodeId = localNodeId;
+ signal->ndbConnectConf.remoteNodeId = remoteNodeId;
+
+ union SIGNAL * discon = alloc(sizeof(NdbTransporterDisconnectOrd),
+ NDB_TRANSPORTER_DISCONNECT_ORD);
+ discon->ndbDisconnect.senderNodeId = remoteNodeId;
+ discon->ndbDisconnect.reason = NdbTransporterDisconnectOrd::PROCESS_DIED;
+
+ DEBUG("Attaching to pid: " << hex << remoteNodePid);
+
+ remoteNodeRef = attach(&discon, remoteNodePid);
+
+ DEBUG("Sending connect conf to pid: " << hex << remoteNodePid);
+
+ ::send(&signal, remoteNodePid);
+ state = CONNECTED;
+
+ _connected = true;
+ _connecting = false;
+ _disconnecting = false;
+
+ NdbMutex_Unlock(theMutexPtr);
+
+ return true;
+}
+
+bool
+OSE_Transporter::connectRef(struct NdbTransporterConnectRef * sig){
+ if(isServer){
+ WARNING("OSE Connect Ref received for server: senderNodeId: " <<
+ sig->senderNodeId);
+ return false;
+ }
+ if(state != WAITING_FOR_CONNECT_CONF){
+ WARNING("OSE Connect Ref received for client while in state: " <<
+ state << " senderNodeId: " << sig->senderNodeId);
+ return false;
+ }
+ doDisconnect();
+#if 0
+ /**
+ * Don't call connect directly, wait until the next time
+ * checkConnections is called which will trigger a new connect attempt
+ */
+ doConnect();
+#endif
+ return true;
+}
+
+
+bool
+OSE_Transporter::connectConf(struct NdbTransporterConnectConf * sig){
+ if(isServer){
+ WARNING("OSE Connect Conf received for server: senderNodeId: " <<
+ sig->senderNodeId);
+ return false;
+ }
+ if(state != WAITING_FOR_CONNECT_CONF){
+ WARNING("OSE Connect Conf received while in state: " <<
+ state);
+ return false;
+ }
+ NdbMutex_Lock(theMutexPtr);
+
+ // Free the buffers to get rid of any "junk" that they might contain
+ if(prioBSignal != NIL){
+ restore(prioBSignal);
+ free_buf(&prioBSignal);
+ }
+ initSignals();
+
+ union SIGNAL * discon = alloc(sizeof(NdbTransporterDisconnectOrd),
+ NDB_TRANSPORTER_DISCONNECT_ORD);
+ discon->ndbDisconnect.senderNodeId = remoteNodeId;
+ discon->ndbDisconnect.reason= NdbTransporterDisconnectOrd::PROCESS_DIED;
+
+ remoteNodeRef = attach(&discon, remoteNodePid);
+
+ state = CONNECTED;
+ _connected = true;
+ _connecting = false;
+ _disconnecting = false;
+
+ // Free the buffers to get rid of any "junk" that they might contain
+ if(prioBSignal != NIL){
+ restore(prioBSignal);
+ free_buf(&prioBSignal);
+ }
+ initSignals();
+
+ NdbMutex_Unlock(theMutexPtr);
+ return true;
+}
+
+
+bool
+OSE_Transporter::disconnectOrd(struct NdbTransporterDisconnectOrd * sig){
+ if(state != CONNECTED){
+ WARNING("OSE Disconnect Ord received while in state: " << state <<
+ " reason: " << sig->reason);
+ return false;
+ }
+
+ if(sig->reason == NdbTransporterDisconnectOrd::PROCESS_DIED){
+ state = DISCONNECTED;
+ }
+
+ doDisconnect();
+ reportDisconnect(callbackObj, remoteNodeId,0);
+ return true;
+}
+
+
+
+
+
+
+
diff --git a/ndb/src/common/transporter/OSE_Transporter.hpp b/ndb/src/common/transporter/OSE_Transporter.hpp
new file mode 100644
index 00000000000..4fd06130477
--- /dev/null
+++ b/ndb/src/common/transporter/OSE_Transporter.hpp
@@ -0,0 +1,158 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//****************************************************************************
+//
+// AUTHOR
+// Magnus Svensson
+//
+// NAME
+// OSE_Transporter
+//
+// DESCRIPTION
+// A OSE_Transporter instance is created when OSE-signal communication
+// shall be used (user specified). It handles connect, disconnect,
+// send and receive.
+//
+//
+//
+//***************************************************************************/
+#ifndef OSE_Transporter_H
+#define OSE_Transporter_H
+
+#include "Transporter.hpp"
+
+#include "ose.h"
+
+class OSE_Transporter : public Transporter {
+ friend class OSE_Receiver;
+ friend class TransporterRegistry;
+public:
+
+ // Initialize member variables
+ OSE_Transporter(int prioASignalSize,
+ int prioBSignalSize,
+ NodeId localNodeId,
+ const char * lHostName,
+ NodeId remoteNodeId,
+ const char * rHostName,
+ int byteorder,
+ bool compression,
+ bool checksum,
+ bool signalId,
+ Uint32 reportFreq = 4096);
+
+ // Disconnect, delete send buffers and receive buffer
+ ~OSE_Transporter();
+
+ /**
+ * Allocate buffers for sending and receiving
+ */
+ bool initTransporter();
+
+ /**
+ * Connect
+ */
+ virtual void doConnect();
+
+ /**
+ * Disconnect
+ */
+ virtual void doDisconnect();
+
+ Uint32 * getWritePtr(Uint32 lenBytes, Uint32 prio);
+ void updateWritePtr(Uint32 lenBytes, Uint32 prio);
+
+ /**
+ * Retrieves the contents of the send buffers, copies it into
+ * an OSE signal and sends it. Until the send buffers are empty
+ */
+ void doSend();
+
+ bool hasDataToSend() const {
+ return prioBSignal->dataSignal.length > 0;
+ }
+
+protected:
+ /**
+ * Not implemented
+ * OSE uses async connect/disconnect
+ */
+ virtual bool connectImpl(Uint32 timeOut){
+ return false;
+ }
+
+ /**
+ * Not implemented
+ * OSE uses async connect/disconnect
+ */
+ virtual void disconnectImpl(){
+ }
+
+private:
+ const bool isServer;
+
+ int maxPrioBDataSize;
+
+ /**
+ * Remote node name
+ * On same machine: ndb_node1
+ * On remote machine: rhost/ndb_node1
+ **/
+ PROCESS remoteNodePid;
+ OSATTREF remoteNodeRef;
+ char remoteNodeName[256];
+
+ Uint32 signalIdCounter;
+
+ int prioBSignalSize;
+
+ Uint32 * prioBInsertPtr;
+ union SIGNAL * prioBSignal;
+
+ struct NdbTransporterData * allocPrioASignal(Uint32 lenBytes) const;
+
+ /**
+ * Statistics
+ */
+ Uint32 reportFreq;
+ Uint32 receiveCount;
+ Uint64 receiveSize;
+ Uint32 sendCount;
+ Uint64 sendSize;
+
+ void initSignals();
+
+ /**
+ * OSE Receiver callbacks
+ */
+ void huntReceived(struct NdbTransporterHunt * sig);
+ bool connectReq(struct NdbTransporterConnectReq * sig);
+ bool connectRef(struct NdbTransporterConnectRef * sig);
+ bool connectConf(struct NdbTransporterConnectConf * sig);
+ bool disconnectOrd(struct NdbTransporterDisconnectOrd * sig);
+
+ enum OSETransporterState {
+ DISCONNECTED = 0,
+ WAITING_FOR_HUNT = 1,
+ WAITING_FOR_CONNECT_REQ = 2,
+ WAITING_FOR_CONNECT_CONF = 3,
+ CONNECTED = 4
+ } state;
+};
+
+// Define of OSE_Transporter_H
+#endif
diff --git a/ndb/src/common/transporter/Packer.cpp b/ndb/src/common/transporter/Packer.cpp
new file mode 100644
index 00000000000..77bd66d1ba9
--- /dev/null
+++ b/ndb/src/common/transporter/Packer.cpp
@@ -0,0 +1,502 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Packer.hpp"
+#include <TransporterRegistry.hpp>
+#include <TransporterCallback.hpp>
+#include <RefConvert.hpp>
+
+#include <stdio.h>
+
+Uint32
+TransporterRegistry::unpack(Uint32 * readPtr,
+ Uint32 sizeOfData,
+ NodeId remoteNodeId,
+ IOState state) {
+ SignalHeader signalHeader;
+ LinearSectionPtr ptr[3];
+
+ Uint32 usedData = 0;
+
+ if(state == NoHalt || state == HaltOutput){
+ while(sizeOfData >= 4 + sizeof(Protocol6)){
+ Uint32 word1 = readPtr[0];
+ Uint32 word2 = readPtr[1];
+ Uint32 word3 = readPtr[2];
+
+#if 0
+ if(Protocol6::getByteOrder(word1) != MY_OWN_BYTE_ORDER){
+ //Do funky stuff
+ }
+#endif
+
+ const Uint16 messageLen32 = Protocol6::getMessageLength(word1);
+ const Uint32 messageLenBytes = ((Uint32)messageLen32) << 2;
+
+ if(messageLen32 == 0 || messageLen32 > MAX_MESSAGE_SIZE){
+ DEBUG("Message Size = " << messageLenBytes);
+ reportError(callbackObj, remoteNodeId, TE_INVALID_MESSAGE_LENGTH);
+ return usedData;
+ }//if
+
+ if (sizeOfData < messageLenBytes) {
+ break;
+ }//if
+
+ if(Protocol6::getCheckSumIncluded(word1)){
+ const Uint32 tmpLen = messageLen32 - 1;
+ const Uint32 checkSumSent = readPtr[tmpLen];
+ const Uint32 checkSumComputed = computeChecksum(&readPtr[0], tmpLen);
+
+ if(checkSumComputed != checkSumSent){
+ reportError(callbackObj, remoteNodeId, TE_INVALID_CHECKSUM);
+ return usedData;
+ }//if
+ }//if
+
+#if 0
+ if(Protocol6::getCompressed(word1)){
+ //Do funky stuff
+ }//if
+#endif
+
+ Protocol6::createSignalHeader(&signalHeader, word1, word2, word3);
+
+ Uint32 sBlockNum = signalHeader.theSendersBlockRef;
+ sBlockNum = numberToRef(sBlockNum, remoteNodeId);
+ signalHeader.theSendersBlockRef = sBlockNum;
+
+ Uint8 prio = Protocol6::getPrio(word1);
+
+ Uint32 * signalData = &readPtr[3];
+
+ if(Protocol6::getSignalIdIncluded(word1) == 0){
+ signalHeader.theSendersSignalId = ~0;
+ } else {
+ signalHeader.theSendersSignalId = * signalData;
+ signalData ++;
+ }//if
+
+ Uint32 * sectionPtr = signalData + signalHeader.theLength;
+ Uint32 * sectionData = sectionPtr + signalHeader.m_noOfSections;
+ for(Uint32 i = 0; i<signalHeader.m_noOfSections; i++){
+ Uint32 sz = * sectionPtr;
+ ptr[i].sz = sz;
+ ptr[i].p = sectionData;
+
+ sectionPtr ++;
+ sectionData += sz;
+ }
+
+ execute(callbackObj, &signalHeader, prio, signalData, ptr);
+
+ readPtr += messageLen32;
+ sizeOfData -= messageLenBytes;
+ usedData += messageLenBytes;
+ }//while
+
+ return usedData;
+ } else {
+ /** state = HaltIO || state == HaltInput */
+
+ while(sizeOfData >= 4 + sizeof(Protocol6)){
+ Uint32 word1 = readPtr[0];
+ Uint32 word2 = readPtr[1];
+ Uint32 word3 = readPtr[2];
+
+#if 0
+ if(Protocol6::getByteOrder(word1) != MY_OWN_BYTE_ORDER){
+ //Do funky stuff
+ }//if
+#endif
+
+ const Uint16 messageLen32 = Protocol6::getMessageLength(word1);
+ const Uint32 messageLenBytes = ((Uint32)messageLen32) << 2;
+ if(messageLen32 == 0 || messageLen32 > MAX_MESSAGE_SIZE){
+ DEBUG("Message Size = " << messageLenBytes);
+ reportError(callbackObj, remoteNodeId, TE_INVALID_MESSAGE_LENGTH);
+ return usedData;
+ }//if
+
+ if (sizeOfData < messageLenBytes) {
+ break;
+ }//if
+
+ if(Protocol6::getCheckSumIncluded(word1)){
+ const Uint32 tmpLen = messageLen32 - 1;
+ const Uint32 checkSumSent = readPtr[tmpLen];
+ const Uint32 checkSumComputed = computeChecksum(&readPtr[0], tmpLen);
+
+ if(checkSumComputed != checkSumSent){
+
+ //theTransporters[remoteNodeId]->disconnect();
+ reportError(callbackObj, remoteNodeId, TE_INVALID_CHECKSUM);
+ return usedData;
+ }//if
+ }//if
+
+#if 0
+ if(Protocol6::getCompressed(word1)){
+ //Do funky stuff
+ }//if
+#endif
+
+ Protocol6::createSignalHeader(&signalHeader, word1, word2, word3);
+
+ Uint32 rBlockNum = signalHeader.theReceiversBlockNumber;
+
+ if(rBlockNum == 252){
+ Uint32 sBlockNum = signalHeader.theSendersBlockRef;
+ sBlockNum = numberToRef(sBlockNum, remoteNodeId);
+ signalHeader.theSendersBlockRef = sBlockNum;
+
+ Uint8 prio = Protocol6::getPrio(word1);
+
+ Uint32 * signalData = &readPtr[3];
+
+ if(Protocol6::getSignalIdIncluded(word1) == 0){
+ signalHeader.theSendersSignalId = ~0;
+ } else {
+ signalHeader.theSendersSignalId = * signalData;
+ signalData ++;
+ }//if
+
+ Uint32 * sectionPtr = signalData + signalHeader.theLength;
+ Uint32 * sectionData = sectionPtr + signalHeader.m_noOfSections;
+ for(Uint32 i = 0; i<signalHeader.m_noOfSections; i++){
+ Uint32 sz = * sectionPtr;
+ ptr[i].sz = sz;
+ ptr[i].p = sectionData;
+
+ sectionPtr ++;
+ sectionData += sz;
+ }
+
+ execute(callbackObj, &signalHeader, prio, signalData, ptr);
+ } else {
+ DEBUG("prepareReceive(...) - Discarding message to block: "
+ << rBlockNum << " from Node: " << remoteNodeId);
+ }//if
+
+ readPtr += messageLen32;
+ sizeOfData -= messageLenBytes;
+ usedData += messageLenBytes;
+ }//while
+
+
+ return usedData;
+ }//if
+}
+
+Uint32 *
+TransporterRegistry::unpack(Uint32 * readPtr,
+ Uint32 * eodPtr,
+ NodeId remoteNodeId,
+ IOState state) {
+ static SignalHeader signalHeader;
+ static LinearSectionPtr ptr[3];
+ if(state == NoHalt || state == HaltOutput){
+ while(readPtr < eodPtr){
+ Uint32 word1 = readPtr[0];
+ Uint32 word2 = readPtr[1];
+ Uint32 word3 = readPtr[2];
+
+#if 0
+ if(Protocol6::getByteOrder(word1) != MY_OWN_BYTE_ORDER){
+ //Do funky stuff
+ }
+#endif
+
+ const Uint16 messageLen32 = Protocol6::getMessageLength(word1);
+
+ if(messageLen32 == 0 || messageLen32 > MAX_MESSAGE_SIZE){
+ DEBUG("Message Size(words) = " << messageLen32);
+ reportError(callbackObj, remoteNodeId, TE_INVALID_MESSAGE_LENGTH);
+ return readPtr;
+ }//if
+
+ if(Protocol6::getCheckSumIncluded(word1)){
+ const Uint32 tmpLen = messageLen32 - 1;
+ const Uint32 checkSumSent = readPtr[tmpLen];
+ const Uint32 checkSumComputed = computeChecksum(&readPtr[0], tmpLen);
+
+ if(checkSumComputed != checkSumSent){
+ reportError(callbackObj, remoteNodeId, TE_INVALID_CHECKSUM);
+ return readPtr;
+ }//if
+ }//if
+
+#if 0
+ if(Protocol6::getCompressed(word1)){
+ //Do funky stuff
+ }//if
+#endif
+
+ Protocol6::createSignalHeader(&signalHeader, word1, word2, word3);
+
+ Uint32 sBlockNum = signalHeader.theSendersBlockRef;
+ sBlockNum = numberToRef(sBlockNum, remoteNodeId);
+ signalHeader.theSendersBlockRef = sBlockNum;
+
+ Uint8 prio = Protocol6::getPrio(word1);
+
+ Uint32 * signalData = &readPtr[3];
+
+ if(Protocol6::getSignalIdIncluded(word1) == 0){
+ signalHeader.theSendersSignalId = ~0;
+ } else {
+ signalHeader.theSendersSignalId = * signalData;
+ signalData ++;
+ }//if
+
+ Uint32 * sectionPtr = signalData + signalHeader.theLength;
+ Uint32 * sectionData = sectionPtr + signalHeader.m_noOfSections;
+ for(Uint32 i = 0; i<signalHeader.m_noOfSections; i++){
+ Uint32 sz = * sectionPtr;
+ ptr[i].sz = sz;
+ ptr[i].p = sectionData;
+
+ sectionPtr ++;
+ sectionData += sz;
+ }
+
+ execute(callbackObj, &signalHeader, prio, signalData, ptr);
+
+ readPtr += messageLen32;
+ }//while
+ } else {
+ /** state = HaltIO || state == HaltInput */
+
+ while(readPtr < eodPtr){
+ Uint32 word1 = readPtr[0];
+ Uint32 word2 = readPtr[1];
+ Uint32 word3 = readPtr[2];
+
+#if 0
+ if(Protocol6::getByteOrder(word1) != MY_OWN_BYTE_ORDER){
+ //Do funky stuff
+ }//if
+#endif
+
+ const Uint16 messageLen32 = Protocol6::getMessageLength(word1);
+ if(messageLen32 == 0 || messageLen32 > MAX_MESSAGE_SIZE){
+ DEBUG("Message Size(words) = " << messageLen32);
+ reportError(callbackObj, remoteNodeId, TE_INVALID_MESSAGE_LENGTH);
+ return readPtr;
+ }//if
+
+ if(Protocol6::getCheckSumIncluded(word1)){
+ const Uint32 tmpLen = messageLen32 - 1;
+ const Uint32 checkSumSent = readPtr[tmpLen];
+ const Uint32 checkSumComputed = computeChecksum(&readPtr[0], tmpLen);
+
+ if(checkSumComputed != checkSumSent){
+
+ //theTransporters[remoteNodeId]->disconnect();
+ reportError(callbackObj, remoteNodeId, TE_INVALID_CHECKSUM);
+ return readPtr;
+ }//if
+ }//if
+
+#if 0
+ if(Protocol6::getCompressed(word1)){
+ //Do funky stuff
+ }//if
+#endif
+
+ Protocol6::createSignalHeader(&signalHeader, word1, word2, word3);
+
+ Uint32 rBlockNum = signalHeader.theReceiversBlockNumber;
+
+ if(rBlockNum == 252){
+ Uint32 sBlockNum = signalHeader.theSendersBlockRef;
+ sBlockNum = numberToRef(sBlockNum, remoteNodeId);
+ signalHeader.theSendersBlockRef = sBlockNum;
+
+ Uint8 prio = Protocol6::getPrio(word1);
+
+ Uint32 * signalData = &readPtr[3];
+
+ if(Protocol6::getSignalIdIncluded(word1) == 0){
+ signalHeader.theSendersSignalId = ~0;
+ } else {
+ signalHeader.theSendersSignalId = * signalData;
+ signalData ++;
+ }//if
+
+ Uint32 * sectionPtr = signalData + signalHeader.theLength;
+ Uint32 * sectionData = sectionPtr + signalHeader.m_noOfSections;
+ for(Uint32 i = 0; i<signalHeader.m_noOfSections; i++){
+ Uint32 sz = * sectionPtr;
+ ptr[i].sz = sz;
+ ptr[i].p = sectionData;
+
+ sectionPtr ++;
+ sectionData += sz;
+ }
+
+ execute(callbackObj, &signalHeader, prio, signalData, ptr);
+ } else {
+ DEBUG("prepareReceive(...) - Discarding message to block: "
+ << rBlockNum << " from Node: " << remoteNodeId);
+ }//if
+
+ readPtr += messageLen32;
+ }//while
+ }//if
+ return readPtr;
+}
+
+Packer::Packer(bool signalId, bool checksum) {
+
+ checksumUsed = (checksum ? 1 : 0);
+ signalIdUsed = (signalId ? 1 : 0);
+
+ // Set the priority
+
+ preComputedWord1 = 0;
+ Protocol6::setByteOrder(preComputedWord1, 0);
+ Protocol6::setSignalIdIncluded(preComputedWord1, signalIdUsed);
+ Protocol6::setCheckSumIncluded(preComputedWord1, checksumUsed);
+ Protocol6::setCompressed(preComputedWord1, 0);
+}
+
+inline
+void
+import(Uint32 * & insertPtr, const LinearSectionPtr & ptr){
+ const Uint32 sz = ptr.sz;
+ memcpy(insertPtr, ptr.p, 4 * sz);
+ insertPtr += sz;
+}
+
+void copy(Uint32 * & insertPtr,
+ class SectionSegmentPool &, const SegmentedSectionPtr & ptr);
+
+void
+Packer::pack(Uint32 * insertPtr,
+ Uint32 prio,
+ const SignalHeader * header,
+ const Uint32 * theData,
+ const LinearSectionPtr ptr[3]) const {
+
+ Uint32 dataLen32 = header->theLength;
+ Uint32 no_segs = header->m_noOfSections;
+
+ Uint32 len32 =
+ dataLen32 + no_segs +
+ checksumUsed + signalIdUsed + (sizeof(Protocol6)/4);
+
+
+ for(Uint32 i = 0; i<no_segs; i++){
+ len32 += ptr[i].sz;
+ }
+
+ /**
+ * Do insert of data
+ */
+ Uint32 word1 = preComputedWord1;
+ Uint32 word2 = 0;
+ Uint32 word3 = 0;
+
+ Protocol6::setPrio(word1, prio);
+ Protocol6::setMessageLength(word1, len32);
+ Protocol6::createProtocol6Header(word1, word2, word3, header);
+
+ insertPtr[0] = word1;
+ insertPtr[1] = word2;
+ insertPtr[2] = word3;
+
+ Uint32 * tmpInserPtr = &insertPtr[3];
+
+ if(signalIdUsed){
+ * tmpInserPtr = header->theSignalId;
+ tmpInserPtr++;
+ }
+
+ memcpy(tmpInserPtr, theData, 4 * dataLen32);
+
+ tmpInserPtr += dataLen32;
+ for(Uint32 i = 0; i<no_segs; i++){
+ tmpInserPtr[i] = ptr[i].sz;
+ }
+
+ tmpInserPtr += no_segs;
+ for(Uint32 i = 0; i<no_segs; i++){
+ import(tmpInserPtr, ptr[i]);
+ }
+
+ if(checksumUsed){
+ * tmpInserPtr = computeChecksum(&insertPtr[0], len32-1);
+ }
+}
+
+void
+Packer::pack(Uint32 * insertPtr,
+ Uint32 prio,
+ const SignalHeader * header,
+ const Uint32 * theData,
+ class SectionSegmentPool & thePool,
+ const SegmentedSectionPtr ptr[3]) const {
+
+ Uint32 dataLen32 = header->theLength;
+ Uint32 no_segs = header->m_noOfSections;
+
+ Uint32 len32 =
+ dataLen32 + no_segs +
+ checksumUsed + signalIdUsed + (sizeof(Protocol6)/4);
+
+ for(Uint32 i = 0; i<no_segs; i++){
+ len32 += ptr[i].sz;
+ }
+
+ /**
+ * Do insert of data
+ */
+ Uint32 word1 = preComputedWord1;
+ Uint32 word2 = 0;
+ Uint32 word3 = 0;
+
+ Protocol6::setPrio(word1, prio);
+ Protocol6::setMessageLength(word1, len32);
+ Protocol6::createProtocol6Header(word1, word2, word3, header);
+
+ insertPtr[0] = word1;
+ insertPtr[1] = word2;
+ insertPtr[2] = word3;
+
+ Uint32 * tmpInserPtr = &insertPtr[3];
+
+ if(signalIdUsed){
+ * tmpInserPtr = header->theSignalId;
+ tmpInserPtr++;
+ }
+
+ memcpy(tmpInserPtr, theData, 4 * dataLen32);
+
+ tmpInserPtr += dataLen32;
+ for(Uint32 i = 0; i<no_segs; i++){
+ tmpInserPtr[i] = ptr[i].sz;
+ }
+
+ tmpInserPtr += no_segs;
+ for(Uint32 i = 0; i<no_segs; i++){
+ copy(tmpInserPtr, thePool, ptr[i]);
+ }
+
+ if(checksumUsed){
+ * tmpInserPtr = computeChecksum(&insertPtr[0], len32-1);
+ }
+}
diff --git a/ndb/src/common/transporter/Packer.hpp b/ndb/src/common/transporter/Packer.hpp
new file mode 100644
index 00000000000..5c191203201
--- /dev/null
+++ b/ndb/src/common/transporter/Packer.hpp
@@ -0,0 +1,85 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PACKER_HPP
+#define PACKER_HPP
+
+#include <TransporterDefinitions.hpp>
+#include "TransporterInternalDefinitions.hpp"
+
+class Packer {
+ Uint32 preComputedWord1;
+ Uint32 checksumUsed; // Checksum shall be included in the message
+ Uint32 signalIdUsed; // Senders signal id shall be included in the message
+public:
+ Packer(bool signalId, bool checksum);
+
+ Uint32 getMessageLength(const SignalHeader* header,
+ const LinearSectionPtr ptr[3]) const ;
+
+
+ Uint32 getMessageLength(const SignalHeader* header,
+ const SegmentedSectionPtr ptr[3]) const ;
+
+ void pack(Uint32 * insertPtr,
+ Uint32 prio,
+ const SignalHeader* header,
+ const Uint32* data,
+ const LinearSectionPtr ptr[3]) const ;
+
+ void pack(Uint32 * insertPtr,
+ Uint32 prio,
+ const SignalHeader* header,
+ const Uint32* data,
+ class SectionSegmentPool & thePool,
+ const SegmentedSectionPtr ptr[3]) const ;
+};
+
+inline
+Uint32
+Packer::getMessageLength(const SignalHeader* header,
+ const LinearSectionPtr ptr[3]) const {
+ Uint32 tLen32 = header->theLength;
+ Uint32 no_seg = header->m_noOfSections;
+ tLen32 += checksumUsed;
+ tLen32 += signalIdUsed;
+ tLen32 += no_seg;
+
+ for(Uint32 i = 0; i<no_seg; i++){
+ tLen32 += ptr[i].sz;
+ }
+
+ return (tLen32 * 4) + sizeof(Protocol6);
+}
+
+inline
+Uint32
+Packer::getMessageLength(const SignalHeader* header,
+ const SegmentedSectionPtr ptr[3]) const {
+ Uint32 tLen32 = header->theLength;
+ Uint32 no_seg = header->m_noOfSections;
+ tLen32 += checksumUsed;
+ tLen32 += signalIdUsed;
+ tLen32 += no_seg;
+
+ for(Uint32 i = 0; i<no_seg; i++){
+ tLen32 += ptr[i].sz;
+ }
+
+ return (tLen32 * 4) + sizeof(Protocol6);
+}
+
+#endif
diff --git a/ndb/src/common/transporter/SCI_Transporter.cpp b/ndb/src/common/transporter/SCI_Transporter.cpp
new file mode 100644
index 00000000000..2be857e8115
--- /dev/null
+++ b/ndb/src/common/transporter/SCI_Transporter.cpp
@@ -0,0 +1,1006 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "SCI_Transporter.hpp"
+#include <NdbStdio.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#include <stdlib.h>
+#include <NdbTick.h>
+#if 0
+#include <malloc.h>
+#include <sys/time.h>
+#endif
+#include "TransporterInternalDefinitions.hpp"
+#include <TransporterCallback.hpp>
+
+#define FLAGS 0
+
+SCI_Transporter::SCI_Transporter(Uint32 packetSize,
+ Uint32 bufferSize,
+ Uint32 nAdapters,
+ Uint16 remoteSciNodeId0,
+ Uint16 remoteSciNodeId1,
+ NodeId _localNodeId,
+ NodeId _remoteNodeId,
+ int byte_order,
+ bool compr,
+ bool chksm,
+ bool signalId,
+ Uint32 reportFreq) :
+ Transporter(_localNodeId, _remoteNodeId, byte_order, compr, chksm, signalId)
+{
+ m_PacketSize = (packetSize + 3)/4 ;
+ m_BufferSize = bufferSize;
+ m_sendBuffer.m_buffer = NULL;
+
+ m_RemoteSciNodeId = remoteSciNodeId0;
+
+ if(remoteSciNodeId0 == 0 || remoteSciNodeId1 == 0)
+ m_numberOfRemoteNodes=1;
+ else
+ m_numberOfRemoteNodes=2;
+
+ m_RemoteSciNodeId1 = remoteSciNodeId1;
+
+
+ m_initLocal=false;
+ m_remoteNodes= new Uint16[m_numberOfRemoteNodes];
+ if(m_remoteNodes == NULL) {
+ //DO WHAT??
+ }
+ m_swapCounter=0;
+ m_failCounter=0;
+ m_remoteNodes[0]=remoteSciNodeId0;
+ m_remoteNodes[1]=remoteSciNodeId1;
+ m_adapters = nAdapters;
+ // The maximum number of times to try and create,
+ // start and destroy a sequence
+ m_ActiveAdapterId=0;
+ m_StandbyAdapterId=1;
+
+ m_mapped = false;
+ m_sciinit=false;
+
+ sciAdapters= new SciAdapter[nAdapters* (sizeof (SciAdapter))];
+ if(sciAdapters==NULL) {
+ }
+ m_SourceSegm= new sourceSegm[nAdapters* (sizeof (sourceSegm))];
+ if(m_SourceSegm==NULL) {
+ }
+ m_TargetSegm= new targetSegm[nAdapters* (sizeof (targetSegm))];
+ if(m_TargetSegm==NULL) {
+ }
+ m_reportFreq= reportFreq;
+
+ //reset all statistic counters.
+#ifdef DEBUG_TRANSPORTER
+ i1024=0;
+ i2048=0;
+ i2049=0;
+ i10242048=0;
+ i20484096=0;
+ i4096=0;
+ i4097=0;
+#endif
+
+}
+
+
+
+void SCI_Transporter::disconnectImpl()
+{
+ sci_error_t err;
+ if(m_mapped){
+ setDisconnect();
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "DisconnectImpl " << getConnectionStatus() << endl;
+ ndbout << "remote node " << remoteNodeId << endl;
+#endif
+ disconnectRemote();
+ disconnectLocal();
+ }
+
+ // Empty send buffer
+
+ m_sendBuffer.m_dataSize = 0;
+
+ m_initLocal=false;
+ m_mapped = false;
+
+ if(m_sciinit) {
+ for(Uint32 i=0; i<m_adapters ; i++) {
+ SCIClose(sciAdapters[i].scidesc, FLAGS, &err);
+
+ if(err != SCI_ERR_OK) {
+ reportError(callbackObj, localNodeId, TE_SCI_UNABLE_TO_CLOSE_CHANNEL);
+#ifdef DEBUG_TRANSPORTER
+ fprintf(stderr,
+ "\nCannot close channel to the driver. Error code 0x%x",
+ err);
+#endif
+ }
+ }
+ }
+ m_sciinit=false;
+
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "total: " << i1024+ i10242048 + i2048+i2049 << endl;
+ ndbout << "<1024: " << i1024 << endl;
+ ndbout << "1024-2047: " << i10242048 << endl;
+ ndbout << "==2048: " << i2048 << endl;
+ ndbout << "2049-4096: " << i20484096 << endl;
+ ndbout << "==4096: " << i4096 << endl;
+ ndbout << ">4096: " << i4097 << endl;
+
+#endif
+
+}
+
+
+bool SCI_Transporter::initTransporter() {
+ if(m_BufferSize < (2*MAX_MESSAGE_SIZE)){
+ m_BufferSize = 2 * MAX_MESSAGE_SIZE;
+ }
+
+ // Allocate buffers for sending
+ Uint32 sz = 0;
+ if(m_BufferSize < (m_PacketSize * 4)){
+ sz = m_BufferSize + MAX_MESSAGE_SIZE;
+ } else {
+ /**
+ * 3 packages
+ */
+ sz = (m_PacketSize * 4) * 3 + MAX_MESSAGE_SIZE;
+ }
+
+ m_sendBuffer.m_bufferSize = 4 * ((sz + 3) / 4);
+ m_sendBuffer.m_buffer = new Uint32[m_sendBuffer.m_bufferSize / 4];
+ m_sendBuffer.m_dataSize = 0;
+
+ if(!getLinkStatus(m_ActiveAdapterId) ||
+ !getLinkStatus(m_StandbyAdapterId)) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "The link is not fully operational. " << endl;
+ ndbout << "Check the cables and the switches" << endl;
+#endif
+ //reportDisconnect(remoteNodeId, 0);
+ //doDisconnect();
+ //NDB should terminate
+ reportError(callbackObj, localNodeId, TE_SCI_LINK_ERROR);
+ return false;
+ }
+
+ return true;
+} // initTransporter()
+
+
+
+Uint32 SCI_Transporter::getLocalNodeId(Uint32 adapterNo)
+{
+ sci_query_adapter_t queryAdapter;
+ sci_error_t error;
+ Uint32 _localNodeId;
+
+ queryAdapter.subcommand = SCI_Q_ADAPTER_NODEID;
+ queryAdapter.localAdapterNo = adapterNo;
+ queryAdapter.data = &_localNodeId;
+
+ SCIQuery(SCI_Q_ADAPTER,(void*)(&queryAdapter),(Uint32)NULL,&error);
+
+ if(error != SCI_ERR_OK)
+ return 0;
+ return _localNodeId;
+}
+
+
+bool SCI_Transporter::getLinkStatus(Uint32 adapterNo)
+{
+ sci_query_adapter_t queryAdapter;
+ sci_error_t error;
+ int linkstatus;
+ queryAdapter.subcommand = SCI_Q_ADAPTER_LINK_OPERATIONAL;
+
+ queryAdapter.localAdapterNo = adapterNo;
+ queryAdapter.data = &linkstatus;
+
+ SCIQuery(SCI_Q_ADAPTER,(void*)(&queryAdapter),(Uint32)NULL,&error);
+
+ if(error != SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "error querying adapter " << endl;
+#endif
+ return false;
+ }
+ if(linkstatus<=0)
+ return false;
+ return true;
+}
+
+
+
+sci_error_t SCI_Transporter::initLocalSegment() {
+ Uint32 segmentSize = m_BufferSize;
+ Uint32 offset = 0;
+ sci_error_t err;
+ if(!m_sciinit) {
+ for(Uint32 i=0; i<m_adapters ; i++) {
+ SCIOpen(&(sciAdapters[i].scidesc), FLAGS, &err);
+ sciAdapters[i].localSciNodeId=getLocalNodeId(i);
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("SCInode iD %d adapter %d\n",
+ sciAdapters[i].localSciNodeId, i);
+#endif
+ if(err != SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("\nCannot open an SCI virtual device. Error code 0x%x",
+ err);
+#endif
+ return err;
+ }
+ }
+ }
+
+ m_sciinit=true;
+
+ SCICreateSegment(sciAdapters[0].scidesc,
+ &(m_SourceSegm[0].localHandle),
+ hostSegmentId(localNodeId, remoteNodeId),
+ segmentSize,
+ 0,
+ 0,
+ 0,
+ &err);
+
+ if(err != SCI_ERR_OK) {
+ return err;
+ } else {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "created segment id : "
+ << hostSegmentId(localNodeId, remoteNodeId) << endl;
+#endif
+ }
+
+ /** Prepare the segment*/
+ for(Uint32 i=0; i < m_adapters; i++) {
+ SCIPrepareSegment((m_SourceSegm[0].localHandle),
+ i,
+ FLAGS,
+ &err);
+
+ if(err != SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("Local Segment is not accessible by an SCI adapter.");
+ ndbout_c("Error code 0x%x\n", err);
+#endif
+ return err;
+ }
+ }
+
+
+ m_SourceSegm[0].mappedMemory =
+ SCIMapLocalSegment((m_SourceSegm[0].localHandle),
+ &(m_SourceSegm[0].lhm[0].map),
+ offset,
+ segmentSize,
+ NULL,
+ FLAGS,
+ &err);
+
+
+
+ if(err != SCI_ERR_OK) {
+
+#ifdef DEBUG_TRANSPORTER
+ fprintf(stderr, "\nCannot map area of size %d. Error code 0x%x",
+ segmentSize,err);
+ ndbout << "initLocalSegment does a disConnect" << endl;
+#endif
+ doDisconnect();
+ return err;
+ }
+
+
+ /** Make the local segment available*/
+ for(Uint32 i=0; i < m_adapters; i++) {
+ SCISetSegmentAvailable((m_SourceSegm[0].localHandle),
+ i,
+ FLAGS,
+ &err);
+
+ if(err != SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("\nLocal Segment is not available for remote connections.");
+ ndbout_c("Error code 0x%x\n", err);
+#endif
+ return err;
+ }
+ }
+
+
+ setupLocalSegment();
+
+ return err;
+
+} // initLocalSegment()
+
+
+bool SCI_Transporter::doSend() {
+#ifdef DEBUG_TRANSPORTER
+ NDB_TICKS startSec=0, stopSec=0;
+ Uint32 startMicro=0, stopMicro=0, totalMicro=0;
+#endif
+ sci_error_t err;
+ Uint32 retry=0;
+
+ const char * const sendPtr = (char*)m_sendBuffer.m_buffer;
+ const Uint32 sizeToSend = m_sendBuffer.m_dataSize;
+
+ if (sizeToSend > 0){
+#ifdef DEBUG_TRANSPORTER
+ if(sizeToSend < 1024 )
+ i1024++;
+ if(sizeToSend > 1024 && sizeToSend < 2048 )
+ i10242048++;
+ if(sizeToSend==2048)
+ i2048++;
+ if(sizeToSend>2048 && sizeToSend < 4096)
+ i20484096++;
+ if(sizeToSend==4096)
+ i4096++;
+ if(sizeToSend==4097)
+ i4097++;
+#endif
+ if(startSequence(m_ActiveAdapterId)!=SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "Start sequence failed" << endl;
+#endif
+ reportError(callbackObj, remoteNodeId, TE_SCI_UNABLE_TO_START_SEQUENCE);
+ return false;
+ }
+
+
+ tryagain:
+ Uint32 * insertPtr = (Uint32 *)
+ (m_TargetSegm[m_ActiveAdapterId].writer)->getWritePtr(sizeToSend);
+
+ if(insertPtr != 0) {
+
+ const Uint32 remoteOffset=(Uint32)
+ ((char*)insertPtr -
+ (char*)(m_TargetSegm[m_ActiveAdapterId].mappedMemory));
+
+ SCIMemCpy(m_TargetSegm[m_ActiveAdapterId].sequence,
+ (void*)sendPtr,
+ m_TargetSegm[m_ActiveAdapterId].rhm[m_ActiveAdapterId].map,
+ remoteOffset,
+ sizeToSend,
+ SCI_FLAG_ERROR_CHECK,
+ &err);
+
+
+ if(err == SCI_ERR_OUT_OF_RANGE) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "Data transfer : out of range error \n" << endl;
+#endif
+ goto tryagain;
+ }
+ if(err == SCI_ERR_SIZE_ALIGNMENT) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "Data transfer : aligne\n" << endl;
+#endif
+ goto tryagain;
+ }
+ if(err == SCI_ERR_OFFSET_ALIGNMENT) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "Data transfer : offset alignment\n" << endl;
+#endif
+ goto tryagain;
+ }
+ if(err == SCI_ERR_TRANSFER_FAILED) {
+ //(m_TargetSegm[m_StandbyAdapterId].writer)->heavyLock();
+ if(getLinkStatus(m_ActiveAdapterId)) {
+ retry++;
+ if(retry>3) {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNRECOVERABLE_DATA_TFX_ERROR);
+ return false;
+ }
+ goto tryagain;
+ }
+ m_failCounter++;
+ Uint32 temp=m_ActiveAdapterId;
+ switch(m_swapCounter) {
+ case 0:
+ /**swap from active (0) to standby (1)*/
+ if(getLinkStatus(m_StandbyAdapterId)) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "Swapping from 0 to 1 " << endl;
+#endif
+ failoverShmWriter();
+ SCIStoreBarrier(m_TargetSegm[m_StandbyAdapterId].sequence,0);
+ m_ActiveAdapterId=m_StandbyAdapterId;
+ m_StandbyAdapterId=temp;
+ SCIRemoveSequence((m_TargetSegm[m_StandbyAdapterId].sequence),
+ FLAGS,
+ &err);
+ if(err!=SCI_ERR_OK) {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNABLE_TO_REMOVE_SEQUENCE);
+ return false;
+ }
+ if(startSequence(m_ActiveAdapterId)!=SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "Start sequence failed" << endl;
+#endif
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNABLE_TO_START_SEQUENCE);
+ return false;
+ }
+ m_swapCounter++;
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "failover complete.." << endl;
+#endif
+ goto tryagain;
+ } else {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNRECOVERABLE_DATA_TFX_ERROR);
+ return false;
+ }
+ return false;
+ break;
+ case 1:
+ /** swap back from 1 to 0
+ must check that the link is up */
+
+ if(getLinkStatus(m_StandbyAdapterId)) {
+ failoverShmWriter();
+ m_ActiveAdapterId=m_StandbyAdapterId;
+ m_StandbyAdapterId=temp;
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "Swapping from 1 to 0 " << endl;
+#endif
+ if(createSequence(m_ActiveAdapterId)!=SCI_ERR_OK) {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNABLE_TO_CREATE_SEQUENCE);
+ return false;
+ }
+ if(startSequence(m_ActiveAdapterId)!=SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "startSequence failed... disconnecting" << endl;
+#endif
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNABLE_TO_START_SEQUENCE);
+ return false;
+ }
+
+ SCIRemoveSequence((m_TargetSegm[m_StandbyAdapterId].sequence)
+ , FLAGS,
+ &err);
+ if(err!=SCI_ERR_OK) {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNABLE_TO_REMOVE_SEQUENCE);
+ return false;
+ }
+
+ if(createSequence(m_StandbyAdapterId)!=SCI_ERR_OK) {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNABLE_TO_CREATE_SEQUENCE);
+ return false;
+ }
+
+ m_swapCounter=0;
+
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "failover complete.." << endl;
+#endif
+ goto tryagain;
+
+ } else {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNRECOVERABLE_DATA_TFX_ERROR);
+ return false;
+ }
+
+ break;
+ default:
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNRECOVERABLE_DATA_TFX_ERROR);
+ return false;
+ break;
+ }
+ } else {
+ SHM_Writer * writer = (m_TargetSegm[m_ActiveAdapterId].writer);
+ writer->updateWritePtr(sizeToSend);
+
+ Uint32 sendLimit = writer->getBufferSize();
+ sendLimit -= writer->getWriteIndex();
+
+ m_sendBuffer.m_dataSize = 0;
+ m_sendBuffer.m_forceSendLimit = sendLimit;
+ }
+
+ } else {
+ /**
+ * If we end up here, the SCI segment is full.
+ */
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "the segment is full for some reason" << endl;
+#endif
+ return false;
+ } //if
+ }
+
+ return true;
+} // doSend()
+
+
+
+void SCI_Transporter::failoverShmWriter() {
+#if 0
+ (m_TargetSegm[m_StandbyAdapterId].writer)
+ ->copyIndexes((m_TargetSegm[m_StandbyAdapterId].writer));
+#endif
+} //failoverShm
+
+
+void SCI_Transporter::setupLocalSegment()
+{
+
+ Uint32 sharedSize = 0;
+ sharedSize += 16; //SHM_Reader::getSharedSize();
+ sharedSize += 16; //SHM_Writer::getSharedSize();
+ sharedSize += 32; //SHM_Writer::getSharedSize();
+ sharedSize =4096; //start of the buffer is page aligend
+
+ Uint32 sizeOfBuffer = m_BufferSize;
+
+ sizeOfBuffer -= sharedSize;
+
+ Uint32 * localReadIndex =
+ (Uint32*)m_SourceSegm[m_ActiveAdapterId].mappedMemory;
+ Uint32 * localWriteIndex =
+ (Uint32*)(localReadIndex+ 1);
+
+ Uint32 * localEndOfDataIndex = (Uint32*)
+ (localReadIndex + 2);
+
+ m_localStatusFlag = (Uint32*)(localReadIndex + 3);
+
+ Uint32 * sharedLockIndex = (Uint32*)
+ (localReadIndex + 4);
+
+ Uint32 * sharedHeavyLock = (Uint32*)
+ (localReadIndex + 5);
+
+ char * localStartOfBuf = (char*)
+ ((char*)m_SourceSegm[m_ActiveAdapterId].mappedMemory+sharedSize);
+
+
+ * localReadIndex = * localWriteIndex = 0;
+ * localEndOfDataIndex = sizeOfBuffer -1;
+
+ const Uint32 slack = MAX_MESSAGE_SIZE;
+
+ reader = new SHM_Reader(localStartOfBuf,
+ sizeOfBuffer,
+ slack,
+ localReadIndex,
+ localWriteIndex);
+
+ * localReadIndex = 0;
+ * localWriteIndex = 0;
+
+ reader->clear();
+} //setupLocalSegment
+
+
+
+void SCI_Transporter::setupRemoteSegment()
+{
+ Uint32 sharedSize = 0;
+ sharedSize += 16; //SHM_Reader::getSharedSize();
+ sharedSize += 16; //SHM_Writer::getSharedSize();
+ sharedSize += 32;
+ sharedSize =4096; //start of the buffer is page aligend
+
+
+ Uint32 sizeOfBuffer = m_BufferSize;
+ sizeOfBuffer -= sharedSize;
+ Uint32 * segPtr = (Uint32*) m_TargetSegm[m_StandbyAdapterId].mappedMemory ;
+
+ Uint32 * remoteReadIndex2 = (Uint32*)segPtr;
+ Uint32 * remoteWriteIndex2 = (Uint32*) (segPtr + 1);
+ Uint32 * remoteEndOfDataIndex2 = (Uint32*) (segPtr + 2);
+ Uint32 * sharedLockIndex2 = (Uint32*) (segPtr + 3);
+ m_remoteStatusFlag2 = (Uint32*)(segPtr + 4);
+ Uint32 * sharedHeavyLock2 = (Uint32*) (segPtr + 5);
+
+
+ char * remoteStartOfBuf2 = ( char*)((char *)segPtr+sharedSize);
+
+ segPtr = (Uint32*) m_TargetSegm[m_ActiveAdapterId].mappedMemory ;
+
+ Uint32 * remoteReadIndex = (Uint32*)segPtr;
+ Uint32 * remoteWriteIndex = (Uint32*) (segPtr + 1);
+ Uint32 * remoteEndOfDataIndex = (Uint32*) (segPtr + 2);
+ Uint32 * sharedLockIndex = (Uint32*) (segPtr + 3);
+ m_remoteStatusFlag = (Uint32*)(segPtr + 4);
+ Uint32 * sharedHeavyLock = (Uint32*) (segPtr + 5);
+
+ char * remoteStartOfBuf = ( char*)((char*)segPtr+(sharedSize));
+
+ * remoteReadIndex = * remoteWriteIndex = 0;
+ * remoteReadIndex2 = * remoteWriteIndex2 = 0;
+ * remoteEndOfDataIndex = sizeOfBuffer - 1;
+ * remoteEndOfDataIndex2 = sizeOfBuffer - 1;
+
+ /**
+ * setup two writers. writer2 is used to mirror the changes of
+ * writer on the standby
+ * segment, so that in the case of a failover, we can switch
+ * to the stdby seg. quickly.*
+ */
+ const Uint32 slack = MAX_MESSAGE_SIZE;
+
+ writer = new SHM_Writer(remoteStartOfBuf,
+ sizeOfBuffer,
+ slack,
+ remoteReadIndex,
+ remoteWriteIndex);
+
+ writer2 = new SHM_Writer(remoteStartOfBuf2,
+ sizeOfBuffer,
+ slack,
+ remoteReadIndex2,
+ remoteWriteIndex2);
+
+ * remoteReadIndex = 0;
+ * remoteWriteIndex = 0;
+
+ writer->clear();
+ writer2->clear();
+
+ m_TargetSegm[0].writer=writer;
+ m_TargetSegm[1].writer=writer2;
+
+ m_sendBuffer.m_forceSendLimit = writer->getBufferSize();
+
+ if(createSequence(m_ActiveAdapterId)!=SCI_ERR_OK) {
+ reportThreadError(remoteNodeId, TE_SCI_UNABLE_TO_CREATE_SEQUENCE);
+ doDisconnect();
+ }
+ if(createSequence(m_StandbyAdapterId)!=SCI_ERR_OK) {
+ reportThreadError(remoteNodeId, TE_SCI_UNABLE_TO_CREATE_SEQUENCE);
+ doDisconnect();
+ }
+
+
+} //setupRemoteSegment
+
+
+bool SCI_Transporter::connectImpl(Uint32 timeout) {
+
+ sci_error_t err;
+ Uint32 offset = 0;
+
+ if(!m_initLocal) {
+ if(initLocalSegment()!=SCI_ERR_OK){
+ NdbSleep_MilliSleep(timeout);
+ //NDB SHOULD TERMINATE AND COMPUTER REBOOTED!
+ reportThreadError(localNodeId, TE_SCI_CANNOT_INIT_LOCALSEGMENT);
+ return false;
+ }
+ m_initLocal=true;
+ }
+
+ if(!m_mapped ) {
+
+ for(Uint32 i=0; i < m_adapters ; i++) {
+ m_TargetSegm[i].rhm[i].remoteHandle=0;
+ SCIConnectSegment(sciAdapters[i].scidesc,
+ &(m_TargetSegm[i].rhm[i].remoteHandle),
+ m_remoteNodes[i],
+ remoteSegmentId(localNodeId, remoteNodeId),
+ i,
+ 0,
+ 0,
+ 0,
+ 0,
+ &err);
+
+ if(err != SCI_ERR_OK) {
+ NdbSleep_MilliSleep(timeout);
+ return false;
+ }
+
+ }
+
+
+ // Map the remote memory segment into program space
+ for(Uint32 i=0; i < m_adapters ; i++) {
+ m_TargetSegm[i].mappedMemory =
+ SCIMapRemoteSegment((m_TargetSegm[i].rhm[i].remoteHandle),
+ &(m_TargetSegm[i].rhm[i].map),
+ offset,
+ m_BufferSize,
+ NULL,
+ FLAGS,
+ &err);
+
+
+ if(err!= SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("\nCannot map a segment to the remote node %d.");
+ ndbout_c("Error code 0x%x",m_RemoteSciNodeId, err);
+#endif
+ //NDB SHOULD TERMINATE AND COMPUTER REBOOTED!
+ reportThreadError(remoteNodeId, TE_SCI_CANNOT_MAP_REMOTESEGMENT);
+ return false;
+ }
+
+
+ }
+ m_mapped=true;
+ setupRemoteSegment();
+ setConnected();
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "connected and mapped to segment : " << endl;
+ ndbout << "remoteNode: " << m_remoteNodes[0] << endl;
+ ndbout << "remoteNode: " << m_remotenodes[1] << endl;
+ ndbout << "remoteSegId: "
+ << remoteSegmentId(localNodeId, remoteNodeId)
+ << endl;
+#endif
+ return true;
+ }
+ else {
+ return getConnectionStatus();
+ }
+} // connectImpl()
+
+
+sci_error_t SCI_Transporter::createSequence(Uint32 adapterid) {
+ sci_error_t err;
+ SCICreateMapSequence((m_TargetSegm[adapterid].rhm[adapterid].map),
+ &(m_TargetSegm[adapterid].sequence),
+ SCI_FLAG_FAST_BARRIER,
+ &err);
+
+
+ return err;
+} // createSequence()
+
+
+sci_error_t SCI_Transporter::startSequence(Uint32 adapterid) {
+
+ sci_error_t err;
+ /** Perform preliminary error check on an SCI adapter before starting a
+ * sequence of read and write operations on the mapped segment.
+ */
+ m_SequenceStatus = SCIStartSequence(
+ (m_TargetSegm[adapterid].sequence),
+ FLAGS, &err);
+
+
+ // If there still is an error then data cannot be safely send
+ return err;
+} // startSequence()
+
+
+
+bool SCI_Transporter::disconnectLocal()
+{
+ sci_error_t err;
+ m_ActiveAdapterId=0;
+
+ /** Free resources used by a local segment
+ */
+
+ SCIUnmapSegment(m_SourceSegm[0].lhm[0].map,0,&err);
+ if(err!=SCI_ERR_OK) {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNABLE_TO_UNMAP_SEGMENT);
+ return false;
+ }
+
+ SCIRemoveSegment((m_SourceSegm[m_ActiveAdapterId].localHandle),
+ FLAGS,
+ &err);
+
+ if(err!=SCI_ERR_OK) {
+ reportError(callbackObj, remoteNodeId, TE_SCI_UNABLE_TO_REMOVE_SEGMENT);
+ return false;
+ }
+
+ if(err == SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ printf("Local memory segment is unmapped and removed\n" );
+#endif
+ }
+ return true;
+} // disconnectLocal()
+
+
+bool SCI_Transporter::disconnectRemote() {
+ sci_error_t err;
+ for(Uint32 i=0; i<m_adapters; i++) {
+ /**
+ * Segment unmapped, disconnect from the remotely connected segment
+ */
+ SCIUnmapSegment(m_TargetSegm[i].rhm[i].map,0,&err);
+ if(err!=SCI_ERR_OK) {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNABLE_TO_DISCONNECT_SEGMENT);
+ return false;
+ }
+
+ SCIDisconnectSegment(m_TargetSegm[i].rhm[i].remoteHandle,
+ FLAGS,
+ &err);
+ if(err!=SCI_ERR_OK) {
+ reportError(callbackObj,
+ remoteNodeId, TE_SCI_UNABLE_TO_DISCONNECT_SEGMENT);
+ return false;
+ }
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("Remote memory segment is unmapped and disconnected\n" );
+#endif
+ }
+ return true;
+} // disconnectRemote()
+
+
+SCI_Transporter::~SCI_Transporter() {
+ // Close channel to the driver
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "~SCITransporter does a disConnect" << endl;
+#endif
+ doDisconnect();
+ if(m_sendBuffer.m_buffer != NULL)
+ delete[] m_sendBuffer.m_buffer;
+} // ~SCI_Transporter()
+
+
+
+
+void SCI_Transporter::closeSCI() {
+ // Termination of SCI
+ sci_error_t err;
+ printf("\nClosing SCI Transporter...\n");
+
+ // Disconnect and remove remote segment
+ disconnectRemote();
+
+ // Unmap and remove local segment
+
+ disconnectLocal();
+
+ // Closes an SCI virtual device
+ SCIClose(activeSCIDescriptor, FLAGS, &err);
+
+ if(err != SCI_ERR_OK)
+ fprintf(stderr,
+ "\nCannot close SCI channel to the driver. Error code 0x%x",
+ err);
+ SCITerminate();
+} // closeSCI()
+
+Uint32 *
+SCI_Transporter::getWritePtr(Uint32 lenBytes, Uint32 prio){
+
+ if(m_sendBuffer.full()){
+ /**-------------------------------------------------
+ * Buffer was completely full. We have severe problems.
+ * -------------------------------------------------
+ */
+ if(!doSend()){
+ return 0;
+ }
+ }
+
+ Uint32 sz = m_sendBuffer.m_dataSize;
+ return &m_sendBuffer.m_buffer[sz];
+}
+
+void
+SCI_Transporter::updateWritePtr(Uint32 lenBytes, Uint32 prio){
+
+ Uint32 sz = m_sendBuffer.m_dataSize;
+ sz += (lenBytes / 4);
+ m_sendBuffer.m_dataSize = sz;
+
+ if(sz > m_PacketSize) {
+ /**-------------------------------------------------
+ * Buffer is full and we are ready to send. We will
+ * not wait since the signal is already in the buffer.
+ * Force flag set has the same indication that we
+ * should always send. If it is not possible to send
+ * we will not worry since we will soon be back for
+ * a renewed trial.
+ *-------------------------------------------------
+ */
+ doSend();
+ }
+}
+
+enum SciStatus {
+ SCIDISCONNECT = 1,
+ SCICONNECTED = 2
+};
+
+bool
+SCI_Transporter::getConnectionStatus() {
+ if(*m_localStatusFlag == SCICONNECTED &&
+ (*m_remoteStatusFlag == SCICONNECTED ||
+ *m_remoteStatusFlag2 == SCICONNECTED))
+ return true;
+ else
+ return false;
+}
+
+
+void
+SCI_Transporter::setConnected() {
+ *m_remoteStatusFlag = SCICONNECTED;
+ *m_remoteStatusFlag2 = SCICONNECTED;
+ *m_localStatusFlag = SCICONNECTED;
+}
+
+
+void
+SCI_Transporter::setDisconnect() {
+ if(getLinkStatus(m_ActiveAdapterId))
+ *m_remoteStatusFlag = SCIDISCONNECT;
+ if(getLinkStatus(m_StandbyAdapterId))
+ *m_remoteStatusFlag2 = SCIDISCONNECT;
+}
+
+
+bool
+SCI_Transporter::checkConnected() {
+ if (*m_localStatusFlag == SCIDISCONNECT) {
+ return false;
+ }
+ else
+ return true;
+}
+
+static bool init = false;
+
+bool
+SCI_Transporter::initSCI() {
+ if(!init){
+ sci_error_t error;
+ // Initialize SISCI library
+ SCIInitialize(0, &error);
+ if(error != SCI_ERR_OK) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("\nCannot initialize SISCI library.");
+ ndbout_c("\nInconsistency between SISCI library and SISCI driver.Error code 0x%x", error);
+#endif
+ return false;
+ }
+ init = true;
+ }
+ return true;
+}
+
+
+
+
+
diff --git a/ndb/src/common/transporter/SCI_Transporter.hpp b/ndb/src/common/transporter/SCI_Transporter.hpp
new file mode 100644
index 00000000000..03496c2ce21
--- /dev/null
+++ b/ndb/src/common/transporter/SCI_Transporter.hpp
@@ -0,0 +1,390 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SCI_Transporter_H
+#define SCI_Transporter_H
+#include "Transporter.hpp"
+#include "SHM_Buffer.hpp"
+
+
+#include <sisci_api.h>
+#include <sisci_error.h>
+#include <sisci_types.h>
+
+#include <ndb_types.h>
+
+ /**
+ * The SCI Transporter
+ *
+ * The design goal of the SCI transporter is to deliver high performance
+ * data transfers (low latency, high bandwidth) combined with very high
+ * availability (failover support).
+ * High performance is an inherit feature of SCI and the, whereas failover
+ * support is implemented at the application level.
+ * In SCI the programming model is similar to the shared memory paradigm.
+ * A process on one node (A) allocates a memory segment and import the
+ * segment to its virtual address space. Another node (B) can connect to
+ * the segment and map this segment into its virtual address space.
+ * If A writes data to the segment, then B can read it and vice versa, through
+ * ordinary loads and stores. This is also called PIO (programmable IO), and
+ * is one thing that distinguish SCI from other interconnects such as,
+ * ethernet, Gig-e, Myrinet, and Infiniband. By using PIO, lower network
+ * latency is achieved, compared to the interconnects mentioned above.
+ * In order for NDB to utilize SCI, the SCI transporter relies on the
+ * SISCI api. The SISCI api provides a high level abstraction to the low
+ * level SCI driver called PCISCI driver.
+ * The SISCI api provides functions to setup, export, and import
+ * memory segments in a process virtual address space, and also functions to
+ * guarantee the correctness of data transfers between nodes. Basically, the
+ *
+ * In NDB Cluster, each SCI transporter creates a local segment
+ * that is mapped into the virtual address space. After the creation of the
+ * local segment, the SCI transporter connects to a segment created by another
+ * transporter at a remote node, and the maps the remote segment into its
+ * virtual address space. However, since NDB Cluster relies on redundancy
+ * at the network level, by using dual SCI adapters communica
+ *
+ *
+ */
+
+
+/**
+ * class SCITransporter
+ * @brief - main class for the SCI transporter.
+ */
+class SCI_Transporter : public Transporter {
+ friend class TransporterRegistry;
+public:
+
+ /**
+ * Init the transporter. Allocate sendbuffers and open a SCI virtual device
+ * for each adapter.
+ * @return true if successful, otherwize false
+ */
+ bool initTransporter();
+
+
+ /**
+ * Creates a sequence for error checking.
+ * @param adapterid the adapter on which to create a new sequence.
+ * @return SCI_ERR_OK if ok, otherwize something else.
+ */
+ sci_error_t createSequence(Uint32 adapterid);
+
+
+ /**
+ * starts a sequence for error checking.
+ * The actual checking that a sequence is correct is done implicitly
+ * in SCIMemCpy (in doSend).
+ * @param adapterid the adapter on which to start the sequence.
+ * @return SCI_ERR_OK if ok, otherwize something else.
+ */
+ sci_error_t startSequence(Uint32 adapterid);
+
+
+ /** Initiate Local Segment: create a memory segment,
+ * prepare a memory segment, map the local segment
+ * into memory space and make segment available.
+ * @return SCI_ERR_OK if ok, otherwize something else.
+ */
+ sci_error_t initLocalSegment();
+
+ /**
+ * Calculate the segment id for the remote segment
+ * @param localNodeId - local id (e.g. 1 = mgm , 2 = ndb.2 etc.)
+ * @param remoteNodeId - remote id (e.g. 1 = mgm , 2 = ndb.2 etc.)
+ * @return a segment id
+ */
+ Uint32 remoteSegmentId(Uint16 localNodeId, Uint16 remoteNodeId);
+
+ // Get local segment id (inline)
+ Uint32 hostSegmentId(Uint16 localNodeId, Uint16 remoteNodeId);
+
+ /**
+ * closeSCI closes the SCI virtual device
+ */
+ void closeSCI();
+
+
+ /**
+ * Check the status of the remote node,
+ * if it is connected or has disconnected
+ * @return true if connected, otherwize false.
+ */
+ bool checkConnected();
+
+ /**
+ * Check if the segment are properly connected to each other (remotely
+ * and locally).
+ * @return True if the both the local segment is mapped and the
+ * remote segment is mapped. Otherwize false.
+ */
+ bool getConnectionStatus();
+
+private:
+ SCI_Transporter(Uint32 packetSize,
+ Uint32 bufferSize,
+ Uint32 nAdapters,
+ Uint16 remoteSciNodeId0,
+ Uint16 remoteSciNodeId1,
+ NodeId localNodeID,
+ NodeId remoteNodeID,
+ int byteorder,
+ bool compression,
+ bool checksum,
+ bool signalId,
+ Uint32 reportFreq = 4096);
+
+ /**
+ * Destructor. Disconnects the transporter.
+ */
+ ~SCI_Transporter();
+ bool m_mapped;
+ bool m_initLocal;
+ bool m_sciinit;
+ Uint32 m_swapCounter;
+ Uint32 m_failCounter;
+ /**
+ * For statistics on transfered packets
+ */
+#ifdef DEBUG_TRANSPORTER
+ Uint32 i1024;
+ Uint32 i2048;
+ Uint32 i2049;
+ Uint32 i10242048;
+ Uint32 i20484096;
+ Uint32 i4096;
+ Uint32 i4097;
+#endif
+
+ volatile Uint32 * m_localStatusFlag;
+ volatile Uint32 * m_remoteStatusFlag;
+ volatile Uint32 * m_remoteStatusFlag2;
+
+ struct {
+ Uint32 * m_buffer; // The buffer
+ Uint32 m_dataSize; // No of words in buffer
+ Uint32 m_bufferSize; // Buffer size
+ Uint32 m_forceSendLimit; // Send when buffer is this full
+
+ bool full() const { return (m_dataSize * 4) > m_forceSendLimit ;}
+ } m_sendBuffer;
+
+ SHM_Reader * reader;
+ SHM_Writer * writer;
+ SHM_Writer * writer2;
+
+ /**
+ * Statistics
+ */
+ Uint32 m_reportFreq;
+
+
+ Uint32 m_adapters;
+ Uint32 m_numberOfRemoteNodes;
+
+ Uint16* m_remoteNodes;
+
+ typedef struct SciAdapter {
+ sci_desc_t scidesc;
+ Uint32 localSciNodeId;
+ bool linkStatus;
+ } SciAdapter;
+
+ SciAdapter* sciAdapters;
+ Uint32 m_ActiveAdapterId;
+ Uint32 m_StandbyAdapterId;
+
+ typedef struct sourceSegm {
+ sci_local_segment_t localHandle; // Handle to local segment to be mapped
+ struct localHandleMap {
+ sci_map_t map; // Handle to the new mapped segment.
+ // 2 = max adapters in one node
+ } lhm[2];
+
+ volatile void *mappedMemory; // Used when reading
+ } sourceSegm;
+
+ typedef struct targetSegm {
+ struct remoteHandleMap {
+ sci_remote_segment_t remoteHandle; //Handle to local segment to be mapped
+ sci_map_t map; //Handle to the new mapped segment
+ } rhm[2];
+
+ sci_sequence_status_t m_SequenceStatus; // Used for error checking
+ sci_sequence_t sequence;
+ volatile void * mappedMemory; // Used when writing
+ SHM_Writer * writer;
+ } targetSegm;
+
+ sci_sequence_status_t m_SequenceStatus; // Used for error checking
+
+
+ // Shared between all SCI users active=(either prim or second)
+ sci_desc_t activeSCIDescriptor;
+
+ sourceSegm* m_SourceSegm; // Local segment reference
+ targetSegm* m_TargetSegm; // Remote segment reference
+
+ Uint32 m_LocalAdapterId; // Adapter Id
+ Uint16 m_LocalSciNodeId; // The SCI-node Id of this machine (adapter 0)
+ Uint16 m_LocalSciNodeId1; // The SCI-node Id of this machine (adapter 1)
+ Uint16 m_RemoteSciNodeId; // The SCI-node Id of remote machine (adapter 0)
+ Uint16 m_RemoteSciNodeId1; // The SCI-node Id of remote machine (adapter 1)
+
+ Uint32 m_PacketSize; // The size of each data packet
+ Uint32 m_BufferSize; // Mapped SCI buffer size
+
+ Uint32 * getWritePtr(Uint32 lenBytes, Uint32 prio);
+ void updateWritePtr(Uint32 lenBytes, Uint32 prio);
+
+ /**
+ * doSend. Copies the data from the source (the send buffer) to the
+ * shared mem. segment.
+ * Sequences are used for error checking.
+ * If an error occurs, the transfer is retried.
+ * If the link that we need to swap to is broken, we will disconnect.
+ * @return Returns true if datatransfer ok. If not retriable
+ * then false is returned.
+ */
+ bool doSend();
+
+ /**
+ * @param adapterNo the adapter for which to retrieve the node id.
+ * @return Returns the node id for an adapter.
+ */
+ Uint32 getLocalNodeId(Uint32 adapterNo);
+
+ bool hasDataToRead() const {
+ return reader->empty() == false;
+ }
+
+ bool hasDataToSend() const {
+ return m_sendBuffer.m_dataSize > 0;
+ }
+
+ /**
+ * Make the local segment unavailable, no new connections will be accepted.
+ * @return Returns true if the segment was successfully disconnected.
+ */
+ bool disconnectLocal();
+
+ /**
+ * Make the local segment unavailable, no new connections will be accepted.
+ * @return Returns true if the segment was successfully disconnected.
+ */
+ bool disconnectRemote();
+
+ void resetToInitialState();
+
+ /**
+ * It is always possible to send data with SCI!
+ * @return True (always)
+ */
+ bool sendIsPossible(struct timeval * timeout);
+
+
+ void getReceivePtr(Uint32 ** ptr, Uint32 ** eod){
+ reader->getReadPtr(* ptr, * eod);
+ }
+
+ void updateReceivePtr(Uint32 * ptr){
+ reader->updateReadPtr(ptr);
+ }
+
+ /**
+ * Corresponds to SHM_Transporter::setupBuffers()
+ * Initiates the start pointer of the buffer and read pointers.
+ * Initiate the localSegment for the SHM reader.
+ */
+ void setupLocalSegment();
+
+ /**
+ * Initiate the remoteSegment for the SHM writer
+ */
+ void setupRemoteSegment();
+
+ /**
+ * Set the connect flag in the remote memory segment (write through)
+ */
+ void setConnected();
+
+ /**
+ * Set the disconnect flag in the remote memory segment (write through)
+ */
+ void setDisconnect();
+
+ /**
+ * Check if there is a link between the adapter and the switch
+ * @param adapterNo the adapter for which to retrieve the link status.
+ * @return Returns true if there is a link between adapter and switch.
+ * Otherwize false is returned and the cables must be checked.
+ */
+ bool getLinkStatus(Uint32 adapterNo);
+
+ /**
+ * failoverShmWriter takes the state of the active writer and inserts into
+ * the standby writer.
+ */
+ void failoverShmWriter();
+
+
+protected:
+
+ /** Perform a connection between segment
+ * This is a client node, trying to connect to a remote segment.
+ * @param timeout, the time the connect thread sleeps before
+ * retrying.
+ * @return Returns true on success, otherwize falser
+ */
+ bool connectImpl(Uint32 timeOutMillis);
+
+ /**
+ * We will disconnect if:
+ * -# the other node has disconnected from us
+ * -# unrecoverable error in transmission, on both adapters
+ * -# if we are shutdown properly
+ */
+ void disconnectImpl();
+
+ static bool initSCI();
+};
+
+
+/** The theLocalAdapterId combined with the theRemoteNodeId constructs
+ * (SCI ids)* a unique identifier for the local segment
+ */
+inline
+Uint32
+SCI_Transporter::hostSegmentId(Uint16 SciLocalNodeId,
+ Uint16 SciRemoteNodeId) {
+
+ return (SciLocalNodeId << 16) | SciRemoteNodeId;
+}
+
+/** The theLocalAdapterId combined with the theRemoteNodeId constructs
+ * (SCI ids)* a unique identifier for the remote segment
+ */
+inline
+Uint32
+SCI_Transporter::remoteSegmentId(Uint16 SciLocalNodeId,
+ Uint16 SciRemoteNodeId) {
+
+ return (SciRemoteNodeId << 16) | SciLocalNodeId;
+}
+
+
+#endif
diff --git a/ndb/src/common/transporter/SHM_Buffer.hpp b/ndb/src/common/transporter/SHM_Buffer.hpp
new file mode 100644
index 00000000000..43250853fee
--- /dev/null
+++ b/ndb/src/common/transporter/SHM_Buffer.hpp
@@ -0,0 +1,217 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SHM_BUFFER_HPP
+#define SHM_BUFFER_HPP
+
+#include <stdio.h>
+#include <ndb_types.h>
+#include <NdbSleep.h>
+#include <assert.h>
+
+/**
+ * These classes implement a circular buffer
+ *
+ * One reader and one writer
+ */
+
+/**
+ * SHM_Reader
+ *
+ * Use as follows:
+ * getReadPtr(ptr, sz);
+ * for(int i = 0; i<sz; i++)
+ * printf("%c\n", ptr[i]);
+ * updateReadPtr(sz);
+ */
+class SHM_Reader {
+public:
+ SHM_Reader(char * const _startOfBuffer,
+ Uint32 _sizeOfBuffer,
+ Uint32 _slack,
+ Uint32 * _readIndex,
+ Uint32 * _writeIndex) :
+ m_startOfBuffer(_startOfBuffer),
+ m_totalBufferSize(_sizeOfBuffer),
+ m_bufferSize(_sizeOfBuffer - _slack),
+ m_sharedReadIndex(_readIndex),
+ m_sharedWriteIndex(_writeIndex)
+ {
+ }
+
+ void clear() {
+ m_readIndex = * m_sharedReadIndex;
+ }
+
+ /**
+ *
+ */
+ inline bool empty() const;
+
+ /**
+ * Get read pointer
+ *
+ * returns ptr - where to start reading
+ * sz - how much can I read
+ */
+ inline void getReadPtr(Uint32 * & ptr, Uint32 * & eod);
+
+ /**
+ * Update read ptr
+ */
+ inline void updateReadPtr(Uint32 * readPtr);
+
+private:
+ char * const m_startOfBuffer;
+ Uint32 m_totalBufferSize;
+ Uint32 m_bufferSize;
+ Uint32 m_readIndex;
+
+ Uint32 * m_sharedReadIndex;
+ Uint32 * m_sharedWriteIndex;
+};
+
+inline
+bool
+SHM_Reader::empty() const{
+ bool ret = (m_readIndex == * m_sharedWriteIndex);
+ return ret;
+}
+
+/**
+ * Get read pointer
+ *
+ * returns ptr - where to start reading
+ * sz - how much can I read
+ */
+inline
+void
+SHM_Reader::getReadPtr(Uint32 * & ptr, Uint32 * & eod){
+
+ Uint32 tReadIndex = m_readIndex;
+ Uint32 tWriteIndex = * m_sharedWriteIndex;
+
+ ptr = (Uint32*)&m_startOfBuffer[tReadIndex];
+
+ if(tReadIndex <= tWriteIndex){
+ eod = (Uint32*)&m_startOfBuffer[tWriteIndex];
+ } else {
+ eod = (Uint32*)&m_startOfBuffer[m_bufferSize];
+ }
+}
+
+/**
+ * Update read ptr
+ */
+inline
+void
+SHM_Reader::updateReadPtr(Uint32 * ptr){
+
+ Uint32 tReadIndex = ((char *)ptr) - m_startOfBuffer;
+
+ assert(tReadIndex < m_totalBufferSize);
+
+ if(tReadIndex >= m_bufferSize){
+ tReadIndex = 0; //-= m_bufferSize;
+ }
+
+ m_readIndex = tReadIndex;
+ * m_sharedReadIndex = tReadIndex;
+}
+
+#define WRITER_SLACK 4
+
+class SHM_Writer {
+public:
+ SHM_Writer(char * const _startOfBuffer,
+ Uint32 _sizeOfBuffer,
+ Uint32 _slack,
+ Uint32 * _readIndex,
+ Uint32 * _writeIndex) :
+ m_startOfBuffer(_startOfBuffer),
+ m_totalBufferSize(_sizeOfBuffer),
+ m_bufferSize(_sizeOfBuffer - _slack),
+ m_sharedReadIndex(_readIndex),
+ m_sharedWriteIndex(_writeIndex)
+ {
+ }
+
+ void clear() {
+ m_writeIndex = * m_sharedWriteIndex;
+ }
+
+ inline char * getWritePtr(Uint32 sz);
+ inline void updateWritePtr(Uint32 sz);
+
+ inline Uint32 getWriteIndex() const { return m_writeIndex;}
+ inline Uint32 getBufferSize() const { return m_bufferSize;}
+
+ inline void copyIndexes(SHM_Writer * standbyWriter);
+
+private:
+ char * const m_startOfBuffer;
+ Uint32 m_totalBufferSize;
+ Uint32 m_bufferSize;
+
+ Uint32 m_writeIndex;
+
+ Uint32 * m_sharedReadIndex;
+ Uint32 * m_sharedWriteIndex;
+};
+
+inline
+char *
+SHM_Writer::getWritePtr(Uint32 sz){
+ Uint32 tReadIndex = * m_sharedReadIndex;
+ Uint32 tWriteIndex = m_writeIndex;
+
+ char * ptr = &m_startOfBuffer[tWriteIndex];
+
+ Uint32 free;
+ if(tReadIndex <= tWriteIndex){
+ free = m_bufferSize + tReadIndex - tWriteIndex;
+ } else {
+ free = tReadIndex - tWriteIndex;
+ }
+
+ sz += 4;
+ if(sz < free){
+ return ptr;
+ }
+
+ return 0;
+}
+
+inline
+void
+SHM_Writer::updateWritePtr(Uint32 sz){
+
+ assert(m_writeIndex == * m_sharedWriteIndex);
+
+ Uint32 tWriteIndex = m_writeIndex;
+ tWriteIndex += sz;
+
+ assert(tWriteIndex < m_totalBufferSize);
+
+ if(tWriteIndex >= m_bufferSize){
+ tWriteIndex = 0; //-= m_bufferSize;
+ }
+
+ m_writeIndex = tWriteIndex;
+ * m_sharedWriteIndex = tWriteIndex;
+}
+
+#endif
diff --git a/ndb/src/common/transporter/SHM_Transporter.cpp b/ndb/src/common/transporter/SHM_Transporter.cpp
new file mode 100644
index 00000000000..f18b775efa4
--- /dev/null
+++ b/ndb/src/common/transporter/SHM_Transporter.cpp
@@ -0,0 +1,238 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "SHM_Transporter.hpp"
+#include "TransporterInternalDefinitions.hpp"
+#include <TransporterCallback.hpp>
+#include <NdbSleep.h>
+#include <NdbOut.hpp>
+
+#include <stdio.h>
+#include <stdlib.h>
+
+#ifdef NDB_WIN32
+#include <windows.h>
+#else
+#include <sys/ipc.h>
+#include <sys/shm.h>
+#endif
+
+
+SHM_Transporter::SHM_Transporter(NodeId lNodeId,
+ NodeId rNodeId,
+ key_t _shmKey,
+ Uint32 _shmSize,
+ bool compression,
+ bool checksum,
+ bool signalId) :
+ Transporter(lNodeId,
+ rNodeId,
+ 0,
+ compression,
+ checksum,
+ signalId),
+ isServer(lNodeId < rNodeId),
+ shmKey(_shmKey),
+ shmSize(_shmSize)
+{
+ _shmSegCreated = false;
+ _attached = false;
+
+ shmBuf = 0;
+ reader = 0;
+ writer = 0;
+
+ setupBuffersDone=false;
+#ifdef DEBUG_TRANSPORTER
+ printf("shm key (%d - %d) = %d\n", lNodeId, rNodeId, shmKey);
+#endif
+}
+
+SHM_Transporter::~SHM_Transporter(){
+ doDisconnect();
+}
+
+bool
+SHM_Transporter::initTransporter(){
+ return true;
+}
+
+bool
+SHM_Transporter::connectImpl(Uint32 timeOutMillis){
+ bool res;
+ if(isServer)
+ res = connectServer(timeOutMillis);
+ else
+ res = connectClient(timeOutMillis);
+ return res;
+}
+
+void
+SHM_Transporter::setupBuffers(){
+ Uint32 sharedSize = 0;
+ sharedSize += 28; //SHM_Reader::getSharedSize();
+ sharedSize += 28; //SHM_Writer::getSharedSize();
+
+ const Uint32 slack = MAX_MESSAGE_SIZE;
+
+ /**
+ * NOTE: There is 7th shared variable in Win2k (sharedCountAttached).
+ */
+ Uint32 sizeOfBuffer = shmSize;
+ sizeOfBuffer -= 2*sharedSize;
+ sizeOfBuffer /= 2;
+
+ Uint32 * base1 = (Uint32*)shmBuf;
+
+ Uint32 * sharedReadIndex1 = base1;
+ Uint32 * sharedWriteIndex1 = base1 + 1;
+ serverStatusFlag = base1 + 4;
+ char * startOfBuf1 = shmBuf+sharedSize;
+
+ Uint32 * base2 = (Uint32*)(shmBuf + sizeOfBuffer + sharedSize);
+ Uint32 * sharedReadIndex2 = base2;
+ Uint32 * sharedWriteIndex2 = base2 + 1;
+ clientStatusFlag = base2 + 4;
+ char * startOfBuf2 = ((char *)base2)+sharedSize;
+
+ * sharedReadIndex2 = * sharedWriteIndex2 = 0;
+
+ if(isServer){
+ * serverStatusFlag = 0;
+ reader = new SHM_Reader(startOfBuf1,
+ sizeOfBuffer,
+ slack,
+ sharedReadIndex1,
+ sharedWriteIndex1);
+
+ writer = new SHM_Writer(startOfBuf2,
+ sizeOfBuffer,
+ slack,
+ sharedReadIndex2,
+ sharedWriteIndex2);
+
+ * sharedReadIndex1 = 0;
+ * sharedWriteIndex2 = 0;
+
+ * sharedReadIndex2 = 0;
+ * sharedWriteIndex1 = 0;
+
+ reader->clear();
+ writer->clear();
+
+ * serverStatusFlag = 1;
+
+#ifdef DEBUG_TRANSPORTER
+ printf("-- (%d - %d) - Server -\n", localNodeId, remoteNodeId);
+ printf("Reader at: %d (%p)\n", startOfBuf1 - shmBuf, startOfBuf1);
+ printf("sharedReadIndex1 at %d (%p) = %d\n",
+ (char*)sharedReadIndex1-shmBuf,
+ sharedReadIndex1, *sharedReadIndex1);
+ printf("sharedWriteIndex1 at %d (%p) = %d\n",
+ (char*)sharedWriteIndex1-shmBuf,
+ sharedWriteIndex1, *sharedWriteIndex1);
+
+ printf("Writer at: %d (%p)\n", startOfBuf2 - shmBuf, startOfBuf2);
+ printf("sharedReadIndex2 at %d (%p) = %d\n",
+ (char*)sharedReadIndex2-shmBuf,
+ sharedReadIndex2, *sharedReadIndex2);
+ printf("sharedWriteIndex2 at %d (%p) = %d\n",
+ (char*)sharedWriteIndex2-shmBuf,
+ sharedWriteIndex2, *sharedWriteIndex2);
+
+ printf("sizeOfBuffer = %d\n", sizeOfBuffer);
+#endif
+ } else {
+ * clientStatusFlag = 0;
+ reader = new SHM_Reader(startOfBuf2,
+ sizeOfBuffer,
+ slack,
+ sharedReadIndex2,
+ sharedWriteIndex2);
+
+ writer = new SHM_Writer(startOfBuf1,
+ sizeOfBuffer,
+ slack,
+ sharedReadIndex1,
+ sharedWriteIndex1);
+
+ * sharedReadIndex2 = 0;
+ * sharedWriteIndex1 = 0;
+
+ reader->clear();
+ writer->clear();
+ * clientStatusFlag = 1;
+#ifdef DEBUG_TRANSPORTER
+ printf("-- (%d - %d) - Client -\n", localNodeId, remoteNodeId);
+ printf("Reader at: %d (%p)\n", startOfBuf2 - shmBuf, startOfBuf2);
+ printf("sharedReadIndex2 at %d (%p) = %d\n",
+ (char*)sharedReadIndex2-shmBuf,
+ sharedReadIndex2, *sharedReadIndex2);
+ printf("sharedWriteIndex2 at %d (%p) = %d\n",
+ (char*)sharedWriteIndex2-shmBuf,
+ sharedWriteIndex2, *sharedWriteIndex2);
+
+ printf("Writer at: %d (%p)\n", startOfBuf1 - shmBuf, startOfBuf1);
+ printf("sharedReadIndex1 at %d (%p) = %d\n",
+ (char*)sharedReadIndex1-shmBuf,
+ sharedReadIndex1, *sharedReadIndex1);
+ printf("sharedWriteIndex1 at %d (%p) = %d\n",
+ (char*)sharedWriteIndex1-shmBuf,
+ sharedWriteIndex1, *sharedWriteIndex1);
+
+ printf("sizeOfBuffer = %d\n", sizeOfBuffer);
+#endif
+ }
+#ifdef DEBUG_TRANSPORTER
+ printf("Mapping from %p to %p\n", shmBuf, shmBuf+shmSize);
+#endif
+}
+
+#if 0
+SendStatus
+SHM_Transporter::prepareSend(const SignalHeader * const signalHeader,
+ Uint8 prio,
+ const Uint32 * const signalData,
+ const LinearSegmentPtr ptr[3],
+ bool force){
+
+ if(isConnected()){
+
+ const Uint32 lenBytes = m_packer.getMessageLength(signalHeader, ptr);
+
+ Uint32 * insertPtr = (Uint32 *)writer->getWritePtr(lenBytes);
+
+ if(insertPtr != 0){
+
+ m_packer.pack(insertPtr, prio, signalHeader, signalData, ptr);
+
+ /**
+ * Do funky membar stuff
+ */
+
+ writer->updateWritePtr(lenBytes);
+ return SEND_OK;
+
+ } else {
+ // NdbSleep_MilliSleep(3);
+ //goto tryagain;
+ return SEND_BUFFER_FULL;
+ }
+ }
+ return SEND_DISCONNECTED;
+}
+#endif
diff --git a/ndb/src/common/transporter/SHM_Transporter.hpp b/ndb/src/common/transporter/SHM_Transporter.hpp
new file mode 100644
index 00000000000..da4566515e3
--- /dev/null
+++ b/ndb/src/common/transporter/SHM_Transporter.hpp
@@ -0,0 +1,156 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SHM_Transporter_H
+#define SHM_Transporter_H
+
+#include "Transporter.hpp"
+#include "SHM_Buffer.hpp"
+
+#ifdef NDB_WIN32
+typedef Uint32 key_t;
+#endif
+
+/**
+ * class SHMTransporter
+ * @brief - main class for the SHM transporter.
+ */
+
+class SHM_Transporter : public Transporter {
+ friend class TransporterRegistry;
+public:
+ SHM_Transporter(NodeId lNodeId,
+ NodeId rNodeId,
+ key_t shmKey,
+ Uint32 shmSize,
+ bool compression,
+ bool checksum,
+ bool signalId);
+
+ /**
+ * SHM destructor
+ */
+ virtual ~SHM_Transporter();
+
+ /**
+ * Do initialization
+ */
+ bool initTransporter();
+
+ Uint32 * getWritePtr(Uint32 lenBytes, Uint32 prio){
+ return (Uint32 *)writer->getWritePtr(lenBytes);
+ }
+
+ void updateWritePtr(Uint32 lenBytes, Uint32 prio){
+ writer->updateWritePtr(lenBytes);
+ }
+
+ void getReceivePtr(Uint32 ** ptr, Uint32 ** eod){
+ reader->getReadPtr(* ptr, * eod);
+ }
+
+ void updateReceivePtr(Uint32 * ptr){
+ reader->updateReadPtr(ptr);
+ }
+
+protected:
+ /**
+ * disconnect a segmnet
+ * -# deletes the shm buffer associated with a segment
+ * -# marks the segment for removal
+ */
+ void disconnectImpl();
+
+ /**
+ * Invokes the connectServer or connectClient.
+ * @param timeOutMillis - the timeout the connect thread waits before
+ * retrying.
+ * @return True if connectImpl successful, otherwise false.
+ */
+ bool connectImpl(Uint32 timeOutMillis);
+
+ /**
+ * Blocking
+ *
+ * -# Create shm segment
+ * -# Attach to it
+ * -# Wait for someone to attach (max wait = timeout), then rerun again
+ * until connection established.
+ * @param timeOutMillis - the time to sleep before (ms) trying again.
+ * @returns - True if the server managed to hook up with the client,
+ * i.e., both agrees that the other one has setup the segment.
+ * Otherwise false.
+ */
+ bool connectServer(Uint32 timeOutMillis);
+
+ /**
+ * Blocking
+ *
+ * -# Attach to shm segment
+ * -# Check if the segment is setup
+ * -# Check if the server set it up
+ * -# If all clear, return.
+ * @param timeOutMillis - the time to sleep before (ms) trying again.
+ * @returns - True if the client managed to hook up with the server,
+ * i.e., both agrees that the other one has setup the segment.
+ * Otherwise false.
+ */
+ bool connectClient(Uint32 timeOutMillis);
+
+
+ /**
+ * Check if there are two processes attached to the segment (a connection)
+ * @return - True if the above holds. Otherwise false.
+ */
+ bool checkConnected();
+
+
+ /**
+ * Initialises the SHM_Reader and SHM_Writer on the segment
+ */
+ void setupBuffers();
+
+private:
+ bool _shmSegCreated;
+ bool _attached;
+
+ const bool isServer;
+ key_t shmKey;
+ volatile Uint32 * serverStatusFlag;
+ volatile Uint32 * clientStatusFlag;
+ bool setupBuffersDone;
+
+#ifdef NDB_WIN32
+ HANDLE hFileMapping;
+#else
+ int shmId;
+#endif
+
+ int shmSize;
+ char * shmBuf;
+
+ SHM_Reader * reader;
+ SHM_Writer * writer;
+
+ /**
+ * @return - True if the reader has data to read on its segment.
+ */
+ bool hasDataToRead() const {
+ return reader->empty() == false;
+ }
+};
+
+#endif
diff --git a/ndb/src/common/transporter/SHM_Transporter.unix.cpp b/ndb/src/common/transporter/SHM_Transporter.unix.cpp
new file mode 100644
index 00000000000..975c1191aea
--- /dev/null
+++ b/ndb/src/common/transporter/SHM_Transporter.unix.cpp
@@ -0,0 +1,179 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "SHM_Transporter.hpp"
+#include "TransporterInternalDefinitions.hpp"
+#include <TransporterCallback.hpp>
+#include <NdbSleep.h>
+#include <NdbOut.hpp>
+
+#include <stdio.h>
+
+
+#include <sys/ipc.h>
+#include <sys/shm.h>
+
+
+
+bool
+SHM_Transporter::connectServer(Uint32 timeOutMillis){
+ if(!_shmSegCreated){
+ shmId = shmget(shmKey, shmSize, IPC_CREAT | 960);
+ if(shmId == -1){
+ perror("shmget: ");
+ reportThreadError(remoteNodeId, TE_SHM_UNABLE_TO_CREATE_SEGMENT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+ _shmSegCreated = true;
+ }
+
+ if(!_attached){
+ shmBuf = (char *)shmat(shmId, 0, 0);
+ if(shmBuf == 0){
+ perror("shmat: ");
+ reportThreadError(remoteNodeId, TE_SHM_UNABLE_TO_ATTACH_SEGMENT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+ _attached = true;
+ }
+
+ struct shmid_ds info;
+ const int res = shmctl(shmId, IPC_STAT, &info);
+ if(res == -1){
+ perror("shmctl: ");
+ reportThreadError(remoteNodeId, TE_SHM_IPC_STAT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+
+ if(info.shm_nattch == 2 && !setupBuffersDone) {
+ setupBuffers();
+ setupBuffersDone=true;
+ }
+
+ if(setupBuffersDone) {
+ NdbSleep_MilliSleep(timeOutMillis);
+ if(*serverStatusFlag==1 && *clientStatusFlag==1)
+ return true;
+ }
+
+
+ if(info.shm_nattch > 2){
+ reportThreadError(remoteNodeId, TE_SHM_DISCONNECT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+}
+
+bool
+SHM_Transporter::connectClient(Uint32 timeOutMillis){
+ if(!_shmSegCreated){
+
+ shmId = shmget(shmKey, shmSize, 0);
+ if(shmId == -1){
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+ _shmSegCreated = true;
+ }
+
+ if(!_attached){
+ shmBuf = (char *)shmat(shmId, 0, 0);
+ if(shmBuf == 0){
+ reportThreadError(remoteNodeId, TE_SHM_UNABLE_TO_ATTACH_SEGMENT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+ _attached = true;
+ }
+
+ struct shmid_ds info;
+
+ const int res = shmctl(shmId, IPC_STAT, &info);
+ if(res == -1){
+ reportThreadError(remoteNodeId, TE_SHM_IPC_STAT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+
+
+ if(info.shm_nattch == 2 && !setupBuffersDone) {
+ setupBuffers();
+ setupBuffersDone=true;
+ }
+
+ if(setupBuffersDone) {
+ NdbSleep_MilliSleep(timeOutMillis);
+ if(*serverStatusFlag==1 && *clientStatusFlag==1)
+ return true;
+ }
+
+ if(info.shm_nattch > 2){
+ reportThreadError(remoteNodeId, TE_SHM_DISCONNECT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+}
+
+bool
+SHM_Transporter::checkConnected(){
+ struct shmid_ds info;
+ const int res = shmctl(shmId, IPC_STAT, &info);
+ if(res == -1){
+ reportError(callbackObj, remoteNodeId, TE_SHM_IPC_STAT);
+ return false;
+ }
+
+ if(info.shm_nattch != 2){
+ reportError(callbackObj, remoteNodeId, TE_SHM_DISCONNECT);
+ return false;
+ }
+ return true;
+}
+
+void
+SHM_Transporter::disconnectImpl(){
+ if(_attached){
+ const int res = shmdt(shmBuf);
+ if(res == -1){
+ perror("shmdelete: ");
+ return;
+ }
+ _attached = false;
+ if(!isServer && _shmSegCreated)
+ _shmSegCreated = false;
+ }
+
+ if(isServer && _shmSegCreated){
+ const int res = shmctl(shmId, IPC_RMID, 0);
+ if(res == -1){
+ reportError(callbackObj, remoteNodeId, TE_SHM_UNABLE_TO_REMOVE_SEGMENT);
+ return;
+ }
+ _shmSegCreated = false;
+ }
+ setupBuffersDone=false;
+}
+
diff --git a/ndb/src/common/transporter/SHM_Transporter.win32.cpp b/ndb/src/common/transporter/SHM_Transporter.win32.cpp
new file mode 100644
index 00000000000..4ba52c9179d
--- /dev/null
+++ b/ndb/src/common/transporter/SHM_Transporter.win32.cpp
@@ -0,0 +1,172 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "SHM_Transporter.hpp"
+#include "TransporterInternalDefinitions.hpp"
+#include <TransporterCallback.hpp>
+#include <NdbSleep.h>
+#include <NdbOut.hpp>
+
+#include <stdio.h>
+
+#include <windows.h>
+
+
+bool
+SHM_Transporter::connectServer(Uint32 timeOutMillis){
+ if(!_shmSegCreated)
+ {
+ char szName[32];
+ sprintf(szName, "ndb%lu", shmKey);
+ hFileMapping = CreateFileMapping(INVALID_HANDLE_VALUE,
+ 0,
+ PAGE_READWRITE,
+ 0,
+ shmSize,
+ szName);
+
+ if(!hFileMapping)
+ {
+ reportThreadError(remoteNodeId, TE_SHM_UNABLE_TO_CREATE_SEGMENT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+ _shmSegCreated = true;
+ }
+
+ if(!_attached){
+ shmBuf = (char*)MapViewOfFile(hFileMapping, FILE_MAP_ALL_ACCESS, 0, 0, 0);
+ if(shmBuf == 0){
+ reportThreadError(remoteNodeId, TE_SHM_UNABLE_TO_ATTACH_SEGMENT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+ volatile Uint32 * sharedCountAttached =
+ (volatile Uint32*)(shmBuf + 6*sizeof(Uint32*));
+ ++*sharedCountAttached;
+ _attached = true;
+ }
+
+ volatile Uint32 * sharedCountAttached =
+ (volatile Uint32*)(shmBuf + 6*sizeof(Uint32*));
+
+ if(*sharedCountAttached == 2 && !setupBuffersDone) {
+ setupBuffers();
+ setupBuffersDone=true;
+ }
+ if(*sharedCountAttached > 2) {
+ reportThreadError(remoteNodeId, TE_SHM_DISCONNECT);
+ return false;
+ }
+
+ if(setupBuffersDone) {
+ NdbSleep_MilliSleep(timeOutMillis);
+ if(*serverStatusFlag==1 && *clientStatusFlag==1)
+ return true;
+ }
+
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+}
+
+bool
+SHM_Transporter::connectClient(Uint32 timeOutMillis){
+ if(!_shmSegCreated)
+ {
+ char szName[32];
+ sprintf(szName, "ndb%lu", shmKey);
+ hFileMapping = OpenFileMapping(FILE_MAP_ALL_ACCESS, FALSE, szName);
+
+ if(!hFileMapping)
+ {
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+ _shmSegCreated = true;
+ }
+
+ if(!_attached){
+ shmBuf = (char*)MapViewOfFile(hFileMapping, FILE_MAP_ALL_ACCESS, 0, 0, 0);
+ if(shmBuf == 0){
+ reportThreadError(remoteNodeId, TE_SHM_UNABLE_TO_ATTACH_SEGMENT);
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+ volatile Uint32 * sharedCountAttached =
+ (volatile Uint32*)(shmBuf + 6*sizeof(Uint32*));
+ ++*sharedCountAttached;
+ _attached = true;
+ }
+
+ volatile Uint32 * sharedCountAttached =
+ (volatile Uint32*)(shmBuf + 6*sizeof(Uint32*));
+
+ if(*sharedCountAttached == 2 && !setupBuffersDone) {
+ setupBuffers();
+ setupBuffersDone=true;
+ }
+
+ if(setupBuffersDone) {
+ if(*serverStatusFlag==1 && *clientStatusFlag==1)
+ return true;
+ }
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+
+}
+
+
+bool
+SHM_Transporter::checkConnected(){
+ volatile Uint32 * sharedCountAttached =
+ (volatile Uint32*)(shmBuf + 6*sizeof(Uint32*));
+ if(*sharedCountAttached != 2) {
+ reportError(callbackObj, remoteNodeId, TE_SHM_DISCONNECT);
+ return false;
+ }
+ return true;
+}
+
+void
+SHM_Transporter::disconnectImpl(){
+ if(_attached) {
+ volatile Uint32 * sharedCountAttached =
+ (volatile Uint32*)(shmBuf + 6*sizeof(Uint32*));
+
+ --*sharedCountAttached;
+
+ if(!UnmapViewOfFile(shmBuf)) {
+ reportError(callbackObj, remoteNodeId, TE_SHM_UNABLE_TO_REMOVE_SEGMENT);
+ return;
+ }
+
+ _attached = false;
+ if(!isServer && _shmSegCreated)
+ _shmSegCreated = false;
+ }
+
+ if(_shmSegCreated){
+ if(!CloseHandle(hFileMapping)) {
+ reportError(callbackObj, remoteNodeId, TE_SHM_UNABLE_TO_REMOVE_SEGMENT);
+ return;
+ }
+ _shmSegCreated = false;
+ }
+ setupBuffersDone=false;
+
+}
+
diff --git a/ndb/src/common/transporter/SendBuffer.cpp b/ndb/src/common/transporter/SendBuffer.cpp
new file mode 100644
index 00000000000..58cad96931f
--- /dev/null
+++ b/ndb/src/common/transporter/SendBuffer.cpp
@@ -0,0 +1,89 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "SendBuffer.hpp"
+#include "TransporterInternalDefinitions.hpp"
+
+SendBuffer::SendBuffer(Uint32 bufSize) {
+
+ sizeOfBuffer = bufSize;
+ if(sizeOfBuffer < MAX_MESSAGE_SIZE)
+ sizeOfBuffer = 2 * MAX_MESSAGE_SIZE;
+ startOfBuffer = NULL;
+
+ // Initalise pointers
+ endOfBuffer = NULL;
+ insertPtr = NULL;
+ sendPtr = NULL;
+ sendDataSize = 0;
+ dataSize = 0;
+}
+
+bool
+SendBuffer::initBuffer(Uint32 aRemoteNodeId) {
+
+ // Allocate memory for the buffer
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "Allocating " << sizeOfBuffer << " bytes for send buffer" << endl;
+#endif
+
+ startOfBuffer = new Uint32[(sizeOfBuffer >> 2) + 1];
+ endOfBuffer = startOfBuffer + (sizeOfBuffer >> 2);
+
+ emptyBuffer();
+ theRemoteNodeId = aRemoteNodeId;
+ return true;
+}
+
+SendBuffer::~SendBuffer() {
+ // Deallocate the buffer memory
+ if(startOfBuffer != NULL)
+ delete[] startOfBuffer;
+}
+
+int
+SendBuffer::bufferSize() {
+ return dataSize;
+}
+
+Uint32
+SendBuffer::bufferSizeRemaining() {
+ return (sizeOfBuffer - dataSize);
+}
+
+void
+SendBuffer::emptyBuffer() {
+ insertPtr = startOfBuffer;
+ sendPtr = (char*)startOfBuffer;
+ dataSize = 0;
+ sendDataSize = 0;
+}
+
+#ifdef DEBUG_TRANSPORTER
+void
+SendBuffer::print() {
+
+ printf("SendBuffer status printouts\n");
+
+ printf( "sizeOfBuffer: %d\n", sizeOfBuffer);
+ printf( "startOfBuffer: %.8x\n", startOfBuffer);
+ printf( "endOfBuffer: %.8x\n", endOfBuffer);
+ printf( "insertPtr: %.8x\n", insertPtr);
+ printf( "sendPtr: %.8x\n", sendPtr);
+ printf( "sendDataSize: %d\n", sendDataSize);
+ printf( "dataSize: %d\n", dataSize);
+}
+#endif
diff --git a/ndb/src/common/transporter/SendBuffer.hpp b/ndb/src/common/transporter/SendBuffer.hpp
new file mode 100644
index 00000000000..75ef0708e83
--- /dev/null
+++ b/ndb/src/common/transporter/SendBuffer.hpp
@@ -0,0 +1,191 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//****************************************************************************
+//
+// NAME
+// SendBuffer
+//
+// DESCRIPTION
+// The SendBuffer is a circular buffer storing signals waiting to be sent.
+// The signals can be of variable size and are copied into the buffer
+// in Protocol 6 format. There will be two SendBuffer instances
+// (priority level A and B) for each transporter using a buffer for
+// sending. The buffering will in most cases be done to send as big
+// packages as possible over TCP/IP.
+//
+//***************************************************************************/
+#ifndef SendBuffer_H
+#define SendBuffer_H
+
+#include "TransporterDefinitions.hpp"
+#include <TransporterCallback.hpp>
+#include <stdlib.h>
+
+#ifdef DEBUG_TRANSPORTER
+#include <stdio.h>
+#endif
+
+class SendBuffer {
+ friend class TCP_Transporter;
+public:
+ // Set member variables
+ SendBuffer(Uint32 bufSize);
+
+ // Deallocate the buffer memory
+ ~SendBuffer();
+
+ // Allocate memory for the buffer and initialize the buffer pointers
+ bool initBuffer(Uint32 aRemoteNodeId);
+
+ // Number of bytes remaining in the buffer
+ Uint32 bufferSizeRemaining();
+
+ // Number of bytes of data in the buffer
+ int bufferSize();
+
+ // Empty the buffer
+ void emptyBuffer();
+
+ /**
+ * The transporter calls updateBuffer after a retrieve followed by
+ * a successful send, to update the cirkular buffer pointers.
+ * updateBuffer is called with the number of bytes really sent,
+ * it may be that it is less than what was retrived from the buffer.
+ * If that is the case there will be an incomplete message (slack)
+ * in the SendBuffer.
+ *
+ * Returns 0 if buffer empty
+ * else ~0
+ */
+ Uint32 bytesSent(Uint32 len);
+
+#ifdef DEBUG_TRANSPORTER
+ // Prints the buffer status on the screen. Can be used for testing purposes.
+ void print();
+#endif
+
+ Uint32* getInsertPtr(Uint32 bytes);
+ void updateInsertPtr(Uint32 bytes);
+
+private:
+
+ Uint32 sizeOfBuffer; // Length, in number of bytes, of the buffer memory
+ Uint32 dataSize; // Number of bytes in buffer
+
+ Uint32 * startOfBuffer; // Pointer to the start of the buffer memory
+ Uint32 * endOfBuffer; // Pointer to end of buffer
+
+ Uint32 * insertPtr; // Where to insert next
+
+ char * sendPtr; // Where data to send starts
+ Uint32 sendDataSize; // Num bytes to send
+
+ Uint32 theRemoteNodeId;
+};
+
+inline
+Uint32
+SendBuffer::bytesSent(Uint32 bytes) {
+
+ if(bytes > dataSize){
+#ifdef DEBUG_TRANSPORTER
+ printf("bytes(%d) > dataSize(%d)\n", bytes, dataSize);
+#endif
+ abort();
+ // reportError(0 ,theRemoteNodeId, TE_INVALID_MESSAGE_LENGTH);
+ return 0;
+ }//if
+
+ if(bytes > sendDataSize){
+#ifdef DEBUG_TRANSPORTER
+ printf("bytes(%d) > sendDataSize(%d)\n", bytes, sendDataSize);
+#endif
+ abort();
+ //reportError(0,theRemoteNodeId, TE_INVALID_MESSAGE_LENGTH);
+ return 0;
+ }//if
+
+ dataSize -= bytes;
+ sendPtr += bytes;
+ sendDataSize -= bytes;
+
+ if(sendDataSize == 0){
+ if(sendPtr > (char*)insertPtr){
+ sendPtr = (char *)startOfBuffer;
+ sendDataSize = dataSize;
+ } else {
+ sendPtr = ((char*)insertPtr) - dataSize;
+ sendDataSize = dataSize;
+ }
+ }
+
+ if(dataSize == 0)
+ return 0;
+ return ~0;
+}
+
+inline
+Uint32*
+SendBuffer::getInsertPtr(Uint32 len){
+ if (bufferSizeRemaining() < len){
+ return 0;
+ }
+
+ const char * const tmpInsertPtr = (char *) insertPtr;
+
+ if(tmpInsertPtr >= sendPtr){
+ // Is there enough space at the end of the buffer?
+ if ((tmpInsertPtr + len) < (char*)endOfBuffer){
+ sendDataSize += len;
+ return insertPtr;
+ } else {
+ // We have passed the end of the cirkular buffer,
+ // must start from the beginning
+ // Is there enough space in the beginning of the buffer?
+ if ((Uint32)(sendPtr - (char *)startOfBuffer) <= len){
+ // Not enough space available, insert failed
+ return 0;
+ } else {
+ // There is space available at the beginning of the buffer
+ // We start from the beginning, set endOfData and insertPtr
+ insertPtr = startOfBuffer;
+ if(sendDataSize != 0){
+ return insertPtr;
+ }
+ sendPtr = (char *)startOfBuffer;
+ sendDataSize = len;
+ return insertPtr;
+ }
+ }
+ } else {
+ // sendPtr > insertPtr
+ // Is there enought room
+ if((tmpInsertPtr + len) < sendPtr){
+ return insertPtr;
+ }
+ return 0;
+ }
+}
+
+inline
+void
+SendBuffer::updateInsertPtr(Uint32 lenBytes){
+ dataSize += lenBytes;
+ insertPtr += (lenBytes / 4);
+}
+
+#endif // Define of SendBuffer_H
diff --git a/ndb/src/common/transporter/TCP_Transporter.cpp b/ndb/src/common/transporter/TCP_Transporter.cpp
new file mode 100644
index 00000000000..8a7d1741636
--- /dev/null
+++ b/ndb/src/common/transporter/TCP_Transporter.cpp
@@ -0,0 +1,603 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbTCP.h>
+#include "TCP_Transporter.hpp"
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+// End of stuff to be moved
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+#define inet_send inet_send
+#else
+#include <NdbStdio.h>
+#define inet_send send
+#endif
+
+#include <stdlib.h>
+
+
+#ifdef NDB_WIN32
+class ndbstrerror
+{
+public:
+ ndbstrerror(int iError);
+ ~ndbstrerror(void);
+ operator char*(void) { return m_szError; };
+
+private:
+ int m_iError;
+ char* m_szError;
+};
+
+ndbstrerror::ndbstrerror(int iError)
+: m_iError(iError)
+{
+ FormatMessage(
+ FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
+ 0,
+ iError,
+ MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
+ (LPTSTR)&m_szError,
+ 0,
+ 0);
+}
+
+ndbstrerror::~ndbstrerror(void)
+{
+ LocalFree( m_szError );
+ m_szError = 0;
+}
+#else
+#define ndbstrerror strerror
+#endif
+
+TCP_Transporter::TCP_Transporter(int sendBufSize, int maxRecvSize,
+ int portNo,
+ const char *rHostName,
+ const char *lHostName,
+ NodeId rNodeId, NodeId lNodeId,
+ int byte_order,
+ bool compr, bool chksm, bool signalId,
+ Uint32 _reportFreq) :
+ Transporter(lNodeId, rNodeId, byte_order, compr, chksm, signalId),
+ m_sendBuffer(sendBufSize),
+ isServer(lNodeId < rNodeId),
+ port(portNo)
+{
+ maxReceiveSize = maxRecvSize;
+
+ strncpy(remoteHostName, rHostName, sizeof(remoteHostName));
+
+ // Initialize member variables
+ Ndb_getInAddr(&remoteHostAddress, rHostName);
+
+ Ndb_getInAddr(&localHostAddress, lHostName);
+ theSocket = NDB_INVALID_SOCKET;
+
+ sendCount = receiveCount = 0;
+ sendSize = receiveSize = 0;
+ reportFreq = _reportFreq;
+
+ sockOptRcvBufSize = 70080;
+ sockOptSndBufSize = 71540;
+ sockOptNodelay = 1;
+ sockOptTcpMaxSeg = 4096;
+}
+
+TCP_Transporter::~TCP_Transporter() {
+
+ // Disconnect
+ if (theSocket != NDB_INVALID_SOCKET)
+ doDisconnect();
+
+ // Delete send buffers
+
+ // Delete receive buffer!!
+ receiveBuffer.destroy();
+}
+
+bool
+TCP_Transporter::initTransporter() {
+
+ // Allocate buffer for receiving
+ // Let it be the maximum size we receive plus 8 kB for any earlier received
+ // incomplete messages (slack)
+ Uint32 recBufSize = maxReceiveSize;
+ if(recBufSize < MAX_MESSAGE_SIZE){
+ recBufSize = MAX_MESSAGE_SIZE;
+ }
+
+ if(!receiveBuffer.init(recBufSize+MAX_MESSAGE_SIZE)){
+ return false;
+ }
+
+ // Allocate buffers for sending
+ if (!m_sendBuffer.initBuffer(remoteNodeId)) {
+ // XXX What shall be done here?
+ // The same is valid for the other init-methods
+ return false;
+ }
+
+ return true;
+}
+
+void
+TCP_Transporter::setSocketOptions(){
+ if (setsockopt(theSocket, SOL_SOCKET, SO_RCVBUF,
+ (char*)&sockOptRcvBufSize, sizeof(sockOptRcvBufSize)) < 0) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("The setsockopt SO_RCVBUF error code = %d", InetErrno);
+#endif
+ }//if
+
+ if (setsockopt(theSocket, SOL_SOCKET, SO_SNDBUF,
+ (char*)&sockOptSndBufSize, sizeof(sockOptSndBufSize)) < 0) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("The setsockopt SO_SNDBUF error code = %d", InetErrno);
+#endif
+ }//if
+
+ //-----------------------------------------------
+ // Set the TCP_NODELAY option so also small packets are sent
+ // as soon as possible
+ //-----------------------------------------------
+ if (setsockopt(theSocket, IPPROTO_TCP, TCP_NODELAY,
+ (char*)&sockOptNodelay, sizeof(sockOptNodelay)) < 0) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("The setsockopt TCP_NODELAY error code = %d", InetErrno);
+#endif
+ }//if
+}
+
+
+#ifdef NDB_WIN32
+
+bool
+TCP_Transporter::setSocketNonBlocking(NDB_SOCKET_TYPE socket){
+ unsigned long ul = 1;
+ if(ioctlsocket(socket, FIONBIO, &ul))
+ {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("Set non-blocking server error3: %d", InetErrno);
+#endif
+ }//if
+ return true;
+}
+
+#else
+
+bool
+TCP_Transporter::setSocketNonBlocking(NDB_SOCKET_TYPE socket){
+ int flags;
+ flags = fcntl(socket, F_GETFL, 0);
+ if (flags < 0) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("Set non-blocking server error1: %s", strerror(InetErrno));
+#endif
+ }//if
+ flags |= NDB_NONBLOCK;
+ if (fcntl(socket, F_SETFL, flags) == -1) {
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("Set non-blocking server error2: %s", strerror(InetErrno));
+#endif
+ }//if
+ return true;
+}
+
+#endif
+
+bool
+TCP_Transporter::sendIsPossible(struct timeval * timeout) {
+#ifdef NDB_OSE
+ /**
+ * In OSE you cant do select without owning a socket,
+ * and since this method might be called by any thread in the api
+ * we choose not to implementet and always return true after sleeping
+ * a while.
+ *
+ * Note that this only sensible as long as the sockets are non blocking
+ */
+ if(theSocket >= 0){
+ Uint32 timeOutMillis = timeout->tv_sec * 1000 + timeout->tv_usec / 1000;
+ NdbSleep_MilliSleep(timeOutMillis);
+ return true;
+ }
+ return false;
+#else
+ if(theSocket != NDB_INVALID_SOCKET){
+ fd_set writeset;
+ FD_ZERO(&writeset);
+ FD_SET(theSocket, &writeset);
+
+ int selectReply = select(theSocket + 1, NULL, &writeset, NULL, timeout);
+
+ if ((selectReply > 0) && FD_ISSET(theSocket, &writeset))
+ return true;
+ else
+ return false;
+ }
+ return false;
+#endif
+}
+
+
+Uint32 *
+TCP_Transporter::getWritePtr(Uint32 lenBytes, Uint32 prio){
+
+ Uint32 * insertPtr = m_sendBuffer.getInsertPtr(lenBytes);
+
+ struct timeval timeout = {0, 10000};
+
+ if (insertPtr == 0) {
+ //-------------------------------------------------
+ // Buffer was completely full. We have severe problems.
+ // We will attempt to wait for a small time
+ //-------------------------------------------------
+ if(sendIsPossible(&timeout)) {
+ //-------------------------------------------------
+ // Send is possible after the small timeout.
+ //-------------------------------------------------
+ if(!doSend()){
+ return 0;
+ } else {
+ //-------------------------------------------------
+ // Since send was successful we will make a renewed
+ // attempt at inserting the signal into the buffer.
+ //-------------------------------------------------
+ insertPtr = m_sendBuffer.getInsertPtr(lenBytes);
+ }//if
+ } else {
+ return 0;
+ }//if
+ }
+ return insertPtr;
+}
+
+void
+TCP_Transporter::updateWritePtr(Uint32 lenBytes, Uint32 prio){
+ m_sendBuffer.updateInsertPtr(lenBytes);
+
+ const int bufsize = m_sendBuffer.bufferSize();
+ if(bufsize > TCP_SEND_LIMIT) {
+ //-------------------------------------------------
+ // Buffer is full and we are ready to send. We will
+ // not wait since the signal is already in the buffer.
+ // Force flag set has the same indication that we
+ // should always send. If it is not possible to send
+ // we will not worry since we will soon be back for
+ // a renewed trial.
+ //-------------------------------------------------
+ struct timeval no_timeout = {0,0};
+ if(sendIsPossible(&no_timeout)) {
+ //-------------------------------------------------
+ // Send was possible, attempt at a send.
+ //-------------------------------------------------
+ doSend();
+ }//if
+ }
+}
+
+#define DISCONNECT_ERRNO(e, sz) ((sz == 0) || \
+ (!((sz == -1) && (e == EAGAIN) || (e == EWOULDBLOCK) || (e == EINTR))))
+
+
+bool
+TCP_Transporter::doSend() {
+ // If no sendbuffers are used nothing is done
+ // Sends the contents of the SendBuffers until they are empty
+ // or until select does not select the socket for write.
+ // Before calling send, the socket must be selected for write
+ // using "select"
+ // It writes on the external TCP/IP interface until the send buffer is empty
+ // and as long as write is possible (test it using select)
+
+ // Empty the SendBuffers
+
+ const char * const sendPtr = m_sendBuffer.sendPtr;
+ const Uint32 sizeToSend = m_sendBuffer.sendDataSize;
+ if (sizeToSend > 0){
+ const int nBytesSent = inet_send(theSocket, sendPtr, sizeToSend, 0);
+
+ if (nBytesSent > 0) {
+ m_sendBuffer.bytesSent(nBytesSent);
+
+ sendCount ++;
+ sendSize += nBytesSent;
+ if(sendCount == reportFreq){
+ reportSendLen(callbackObj,remoteNodeId, sendCount, sendSize);
+ sendCount = 0;
+ sendSize = 0;
+ }
+ } else {
+ // Send failed
+#if defined DEBUG_TRANSPORTER
+ ndbout_c("Send Failure(disconnect==%d) to node = %d nBytesSent = %d "
+ "errno = %d strerror = %s",
+ DISCONNECT_ERRNO(InetErrno, nBytesSent),
+ remoteNodeId, nBytesSent, InetErrno,
+ (char*)ndbstrerror(InetErrno));
+#endif
+ if(DISCONNECT_ERRNO(InetErrno, nBytesSent)){
+ doDisconnect();
+ reportDisconnect(callbackObj, remoteNodeId, InetErrno);
+ }
+
+ return false;
+ }
+ }
+ return true;
+}
+
+int
+TCP_Transporter::doReceive() {
+ // Select-function must return the socket for read
+ // before this method is called
+ // It reads the external TCP/IP interface once
+
+ const int nBytesRead = recv(theSocket,
+ receiveBuffer.insertPtr, maxReceiveSize, 0);
+
+ if (nBytesRead > 0) {
+ receiveBuffer.sizeOfData += nBytesRead;
+ receiveBuffer.insertPtr += nBytesRead;
+
+ if(receiveBuffer.sizeOfData > receiveBuffer.sizeOfBuffer){
+#ifdef DEBUG_TRANSPORTER
+ ndbout_c("receiveBuffer.sizeOfData(%d) > receiveBuffer.sizeOfBuffer(%d)",
+ receiveBuffer.sizeOfData, receiveBuffer.sizeOfBuffer);
+ ndbout_c("nBytesRead = %d", nBytesRead);
+#endif
+ ndbout_c("receiveBuffer.sizeOfData(%d) > receiveBuffer.sizeOfBuffer(%d)",
+ receiveBuffer.sizeOfData, receiveBuffer.sizeOfBuffer);
+ reportError(callbackObj, remoteNodeId, TE_INVALID_MESSAGE_LENGTH);
+ return 0;
+ }
+
+ receiveCount ++;
+ receiveSize += nBytesRead;
+ if(receiveCount == reportFreq){
+ reportReceiveLen(callbackObj, remoteNodeId, receiveCount, receiveSize);
+ receiveCount = 0;
+ receiveSize = 0;
+ }
+ return nBytesRead;
+ } else {
+#if defined DEBUG_TRANSPORTER
+ ndbout_c("Receive Failure(disconnect==%d) to node = %d nBytesSent = %d "
+ "errno = %d strerror = %s",
+ DISCONNECT_ERRNO(InetErrno, nBytesRead),
+ remoteNodeId, nBytesRead, InetErrno,
+ (char*)ndbstrerror(InetErrno));
+#endif
+ if(DISCONNECT_ERRNO(InetErrno, nBytesRead)){
+ // The remote node has closed down
+ doDisconnect();
+ reportDisconnect(callbackObj, remoteNodeId,InetErrno);
+ }
+ }
+ return nBytesRead;
+}
+
+bool
+TCP_Transporter::connectImpl(Uint32 timeOutMillis){
+ struct timeval timeout = {0, 0};
+ timeout.tv_sec = timeOutMillis / 1000;
+ timeout.tv_usec = (timeOutMillis % 1000)*1000;
+
+ bool retVal = false;
+
+ if(isServer){
+ if(theSocket == NDB_INVALID_SOCKET){
+ startTCPServer();
+ }
+ if(theSocket == NDB_INVALID_SOCKET)
+ {
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+ retVal = acceptClient(&timeout);
+ } else {
+ // Is client
+ retVal = connectClient(&timeout);
+ }
+
+ if(!retVal) {
+ NdbSleep_MilliSleep(timeOutMillis);
+ return false;
+ }
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ if(setsockopt(theSocket, SOL_SOCKET, SO_OSEOWNER,
+ &theReceiverPid, sizeof(PROCESS)) != 0){
+
+ ndbout << "Failed to transfer ownership of socket" << endl;
+ NDB_CLOSE_SOCKET(theSocket);
+ theSocket = -1;
+ return false;
+ }
+#endif
+
+ return true;
+}
+
+
+void
+TCP_Transporter::disconnectImpl() {
+ if(theSocket != NDB_INVALID_SOCKET){
+ if(NDB_CLOSE_SOCKET(theSocket) < 0){
+ reportError(callbackObj, remoteNodeId, TE_ERROR_CLOSING_SOCKET);
+ }
+ }
+
+ // Empty send och receive buffers
+ receiveBuffer.clear();
+ m_sendBuffer.emptyBuffer();
+
+ theSocket = NDB_INVALID_SOCKET;
+}
+
+bool
+TCP_Transporter::startTCPServer() {
+
+ int bindResult, listenResult;
+
+ // The server variable is the remote server when we are a client
+ // htonl and htons returns the parameter in network byte order
+ // INADDR_ANY tells the OS kernel to choose the IP address
+ struct sockaddr_in server;
+ memset((void*)&server, 0, sizeof(server));
+ server.sin_family = AF_INET;
+ server.sin_addr.s_addr = localHostAddress.s_addr;
+ server.sin_port = htons(port);
+
+ if (theSocket != NDB_INVALID_SOCKET) {
+ return true; // Server socket is already initialized
+ }
+
+ // Create the socket
+ theSocket = socket(AF_INET, SOCK_STREAM, 0);
+ if (theSocket == NDB_INVALID_SOCKET) {
+ reportThreadError(remoteNodeId, TE_COULD_NOT_CREATE_SOCKET);
+ return false;
+ }
+
+ // Set the socket reuse addr to true, so we are sure we can bind the
+ // socket
+ int reuseAddr = 1;
+ setsockopt(theSocket, SOL_SOCKET, SO_REUSEADDR,
+ (char*)&reuseAddr, sizeof(reuseAddr));
+
+ // Set the TCP_NODELAY option so also small packets are sent
+ // as soon as possible
+ int nodelay = 1;
+ setsockopt(theSocket, IPPROTO_TCP, TCP_NODELAY,
+ (char*)&nodelay, sizeof(nodelay));
+
+ // Bind the socket
+ bindResult = bind(theSocket, (struct sockaddr *) &server,
+ sizeof(server));
+ if (bindResult < 0) {
+ reportThreadError(remoteNodeId, TE_COULD_NOT_BIND_SOCKET);
+ NDB_CLOSE_SOCKET(theSocket);
+ theSocket = NDB_INVALID_SOCKET;
+ return false;
+ }
+
+ // Perform listen.
+ listenResult = listen(theSocket, 1);
+ if (listenResult == 1) {
+ reportThreadError(remoteNodeId, TE_LISTEN_FAILED);
+ NDB_CLOSE_SOCKET(theSocket);
+ theSocket = NDB_INVALID_SOCKET;
+ return false;
+ }
+
+ return true;
+}
+
+
+bool
+TCP_Transporter::acceptClient (struct timeval * timeout){
+
+ struct sockaddr_in clientAddress;
+
+ fd_set readset;
+ FD_ZERO(&readset);
+ FD_SET(theSocket, &readset);
+ const int res = select(theSocket + 1, &readset, 0, 0, timeout);
+ if(res == 0)
+ return false;
+
+ if(res < 0){
+ reportThreadError(remoteNodeId, TE_ERROR_IN_SELECT_BEFORE_ACCEPT);
+ return false;
+ }
+
+ NDB_SOCKLEN_T clientAddressLen = sizeof(clientAddress);
+ const NDB_SOCKET_TYPE clientSocket = accept(theSocket,
+ (struct sockaddr*)&clientAddress,
+ &clientAddressLen);
+ if (clientSocket == NDB_INVALID_SOCKET) {
+ reportThreadError(remoteNodeId, TE_ACCEPT_RETURN_ERROR);
+ return false;
+ }
+
+ if (clientAddress.sin_addr.s_addr != remoteHostAddress.s_addr) {
+ ndbout_c("Wrong client connecting!");
+ ndbout_c("connecting address: %s", inet_ntoa(clientAddress.sin_addr));
+ ndbout_c("expecting address: %s", inet_ntoa(remoteHostAddress));
+ // The newly connected host is not the remote host
+ // we wanted to connect to. Disconnect it.
+ // XXX This is not valid. We cannot disconnect it.
+ NDB_CLOSE_SOCKET(clientSocket);
+ return false;
+ } else {
+ NDB_CLOSE_SOCKET(theSocket);
+ theSocket = clientSocket;
+ setSocketOptions();
+ setSocketNonBlocking(theSocket);
+ return true;
+ }
+}
+
+bool
+TCP_Transporter::connectClient (struct timeval * timeout){
+
+ // Create the socket
+ theSocket = socket(AF_INET, SOCK_STREAM, 0);
+ if (theSocket == NDB_INVALID_SOCKET) {
+ reportThreadError(remoteNodeId, TE_COULD_NOT_CREATE_SOCKET);
+ return false;
+ }
+
+ struct sockaddr_in server;
+ memset((void*)&server, 0, sizeof(server));
+ server.sin_family = AF_INET;
+ server.sin_addr = remoteHostAddress;
+ server.sin_port = htons(port);
+
+ struct sockaddr_in client;
+ memset((void*)&client, 0, sizeof(client));
+ client.sin_family = AF_INET;
+ client.sin_addr = localHostAddress;
+ client.sin_port = 0; // Any port
+
+ // Bind the socket
+ const int bindResult = bind(theSocket, (struct sockaddr *) &client,
+ sizeof(client));
+ if (bindResult < 0) {
+ reportThreadError(remoteNodeId, TE_COULD_NOT_BIND_SOCKET);
+ NDB_CLOSE_SOCKET(theSocket);
+ theSocket = NDB_INVALID_SOCKET;
+ return false;
+ }
+
+ const int connectRes = ::connect(theSocket, (struct sockaddr *) &server,
+ sizeof(server));
+ if(connectRes == 0){
+ setSocketOptions();
+ setSocketNonBlocking(theSocket);
+ return true;
+ }
+
+ NDB_CLOSE_SOCKET(theSocket);
+ theSocket = NDB_INVALID_SOCKET;
+ return false;
+}
+
+
+
diff --git a/ndb/src/common/transporter/TCP_Transporter.hpp b/ndb/src/common/transporter/TCP_Transporter.hpp
new file mode 100644
index 00000000000..30b730a5b1c
--- /dev/null
+++ b/ndb/src/common/transporter/TCP_Transporter.hpp
@@ -0,0 +1,290 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//****************************************************************************
+//
+// AUTHOR
+// Åsa Fransson
+//
+// NAME
+// TCP_Transporter
+//
+// DESCRIPTION
+// A TCP_Transporter instance is created when TCP/IP-communication
+// shall be used (user specified). It handles connect, disconnect,
+// send and receive.
+//
+//
+//
+//***************************************************************************/
+#ifndef TCP_Transporter_H
+#define TCP_Transporter_H
+
+#include "Transporter.hpp"
+#include "SendBuffer.hpp"
+
+#include <NdbTCP.h>
+
+struct ReceiveBuffer {
+ Uint32 *startOfBuffer; // Pointer to start of the receive buffer
+ Uint32 *readPtr; // Pointer to start reading data
+
+ char *insertPtr; // Pointer to first position in the receiveBuffer
+ // in which to insert received data. Earlier
+ // received incomplete messages (slack) are
+ // copied into the first part of the receiveBuffer
+
+ Uint32 sizeOfData; // In bytes
+ Uint32 sizeOfBuffer;
+
+ bool init(int bytes);
+ void destroy();
+
+ void clear();
+ void incompleteMessage();
+};
+
+class TCP_Transporter : public Transporter {
+ friend class TransporterRegistry;
+private:
+ // Initialize member variables
+ TCP_Transporter(int sendBufferSize, int maxReceiveSize,
+ int port,
+ const char *rHostName,
+ const char *lHostName,
+ NodeId rHostId, NodeId lHostId,
+ int byteorder,
+ bool compression, bool checksum, bool signalId,
+ Uint32 reportFreq = 4096);
+
+ // Disconnect, delete send buffers and receive buffer
+ virtual ~TCP_Transporter();
+
+ /**
+ * Allocate buffers for sending and receiving
+ */
+ bool initTransporter();
+
+ Uint32 * getWritePtr(Uint32 lenBytes, Uint32 prio);
+ void updateWritePtr(Uint32 lenBytes, Uint32 prio);
+
+ bool hasDataToSend() const ;
+
+ /**
+ * Retrieves the contents of the send buffers and writes it on
+ * the external TCP/IP interface until the send buffers are empty
+ * and as long as write is possible.
+ */
+ bool doSend();
+
+ /**
+ * It reads the external TCP/IP interface once
+ * and puts the data in the receiveBuffer
+ */
+ int doReceive();
+
+ /**
+ * Returns socket (used for select)
+ */
+ NDB_SOCKET_TYPE getSocket() const;
+
+ /**
+ * Get Receive Data
+ *
+ * Returns - no of bytes to read
+ * and set ptr
+ */
+ virtual Uint32 getReceiveData(Uint32 ** ptr);
+
+ /**
+ * Update receive data ptr
+ */
+ virtual void updateReceiveDataPtr(Uint32 bytesRead);
+
+protected:
+ /**
+ * Setup client/server and perform connect/accept
+ * Is used both by clients and servers
+ * A client connects to the remote server
+ * A server accepts any new connections
+ */
+ bool connectImpl(Uint32 timeOutMillis);
+
+ /**
+ * Disconnects a TCP/IP node. Empty send and receivebuffer.
+ */
+ void disconnectImpl();
+
+private:
+ /**
+ * Send buffers
+ */
+ SendBuffer m_sendBuffer;
+
+ const bool isServer;
+ const unsigned int port;
+
+ // Sending/Receiving socket used by both client and server
+ NDB_SOCKET_TYPE theSocket;
+
+ Uint32 maxReceiveSize;
+
+ /**
+ * Remote host name/and address
+ */
+ char remoteHostName[256];
+ struct in_addr remoteHostAddress;
+ struct in_addr localHostAddress;
+
+ /**
+ * Socket options
+ */
+ int sockOptRcvBufSize;
+ int sockOptSndBufSize;
+ int sockOptNodelay;
+ int sockOptTcpMaxSeg;
+
+ void setSocketOptions();
+
+ static bool setSocketNonBlocking(NDB_SOCKET_TYPE aSocket);
+
+ bool sendIsPossible(struct timeval * timeout);
+
+ /**
+ * startTCPServer - None blocking
+ *
+ * create a server socket
+ * bind
+ * listen
+ *
+ * Note: Does not call accept
+ */
+ bool startTCPServer();
+
+ /**
+ * acceptClient - Blocking
+ *
+ * Accept a connection
+ * checks if "right" client has connected
+ * if so
+ * close server socket
+ * else
+ * close newly created socket and goto begin
+ */
+ bool acceptClient(struct timeval * timeout);
+
+ /**
+ * Creates a client socket
+ *
+ * Note does not call connect
+ */
+ bool createClientSocket();
+
+ /**
+ * connectClient - Blocking
+ *
+ * connects to remote host
+ */
+ bool connectClient(struct timeval * timeout);
+
+ /**
+ * Statistics
+ */
+ Uint32 reportFreq;
+ Uint32 receiveCount;
+ Uint64 receiveSize;
+ Uint32 sendCount;
+ Uint64 sendSize;
+
+ ReceiveBuffer receiveBuffer;
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ PROCESS theReceiverPid;
+#endif
+};
+
+inline
+NDB_SOCKET_TYPE
+TCP_Transporter::getSocket() const {
+ return theSocket;
+}
+
+inline
+Uint32
+TCP_Transporter::getReceiveData(Uint32 ** ptr){
+ (* ptr) = receiveBuffer.readPtr;
+ return receiveBuffer.sizeOfData;
+}
+
+inline
+void
+TCP_Transporter::updateReceiveDataPtr(Uint32 bytesRead){
+ char * ptr = (char *)receiveBuffer.readPtr;
+ ptr += bytesRead;
+ receiveBuffer.readPtr = (Uint32*)ptr;
+ receiveBuffer.sizeOfData -= bytesRead;
+ receiveBuffer.incompleteMessage();
+}
+
+inline
+bool
+TCP_Transporter::hasDataToSend() const {
+ return m_sendBuffer.dataSize > 0;
+}
+
+inline
+bool
+ReceiveBuffer::init(int bytes){
+#ifdef DEBUG_TRANSPORTER
+ ndbout << "Allocating " << bytes << " bytes as receivebuffer" << endl;
+#endif
+
+ startOfBuffer = new Uint32[((bytes + 0) >> 2) + 1];
+ sizeOfBuffer = bytes + sizeof(Uint32);
+ clear();
+ return true;
+}
+
+inline
+void
+ReceiveBuffer::destroy(){
+ delete[] startOfBuffer;
+ sizeOfBuffer = 0;
+ startOfBuffer = 0;
+ clear();
+}
+
+inline
+void
+ReceiveBuffer::clear(){
+ readPtr = startOfBuffer;
+ insertPtr = (char *)startOfBuffer;
+ sizeOfData = 0;
+}
+
+inline
+void
+ReceiveBuffer::incompleteMessage() {
+ if(startOfBuffer != readPtr){
+ if(sizeOfData != 0)
+ memmove(startOfBuffer, readPtr, sizeOfData);
+ readPtr = startOfBuffer;
+ insertPtr = ((char *)startOfBuffer) + sizeOfData;
+ }
+}
+
+
+#endif // Define of TCP_Transporter_H
diff --git a/ndb/src/common/transporter/Transporter.cpp b/ndb/src/common/transporter/Transporter.cpp
new file mode 100644
index 00000000000..1e19a8375ba
--- /dev/null
+++ b/ndb/src/common/transporter/Transporter.cpp
@@ -0,0 +1,147 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "Transporter.hpp"
+#include "TransporterInternalDefinitions.hpp"
+#include <NdbStdio.h>
+#include <NdbSleep.h>
+
+Transporter::Transporter(NodeId lNodeId, NodeId rNodeId,
+ int _byteorder,
+ bool _compression, bool _checksum, bool _signalId)
+ : localNodeId(lNodeId), remoteNodeId(rNodeId),
+ m_packer(_signalId, _checksum)
+{
+ byteOrder = _byteorder;
+ compressionUsed = _compression;
+ checksumUsed = _checksum;
+ signalIdUsed = _signalId;
+
+ _threadError = TE_NO_ERROR;
+
+ _connecting = false;
+ _disconnecting = false;
+ _connected = false;
+ _timeOutMillis = 1000;
+ theThreadPtr = NULL;
+ theMutexPtr = NdbMutex_Create();
+}
+
+Transporter::~Transporter(){
+ NdbMutex_Destroy(theMutexPtr);
+
+ if(theThreadPtr != 0){
+ void * retVal;
+ NdbThread_WaitFor(theThreadPtr, &retVal);
+ NdbThread_Destroy(&theThreadPtr);
+ }
+}
+
+extern "C"
+void *
+runConnect_C(void * me)
+{
+ runConnect(me);
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+void *
+runConnect(void * me){
+ Transporter * t = (Transporter *) me;
+
+ DEBUG("Connect thread to " << t->remoteNodeId << " started");
+
+ while(true){
+ NdbMutex_Lock(t->theMutexPtr);
+ if(t->_disconnecting){
+ t->_connecting = false;
+ NdbMutex_Unlock(t->theMutexPtr);
+ DEBUG("Connect Thread " << t->remoteNodeId << " stop due to disconnect");
+ return 0;
+ }
+ NdbMutex_Unlock(t->theMutexPtr);
+
+ bool res = t->connectImpl(t->_timeOutMillis); // 1000 ms
+ DEBUG("Waiting for " << t->remoteNodeId << "...");
+ if(res){
+ t->_connected = true;
+ t->_connecting = false;
+ t->_errorCount = 0;
+ t->_threadError = TE_NO_ERROR;
+ DEBUG("Connect Thread " << t->remoteNodeId << " stop due to connect");
+ return 0;
+ }
+ }
+}
+
+void
+Transporter::doConnect() {
+
+ NdbMutex_Lock(theMutexPtr);
+ if(_connecting || _disconnecting || _connected){
+ NdbMutex_Unlock(theMutexPtr);
+ return;
+ }
+
+ _connecting = true;
+
+ _threadError = TE_NO_ERROR;
+
+ // Start thread
+
+ char buf[16];
+ snprintf(buf, sizeof(buf), "ndb_con_%d", remoteNodeId);
+
+ if(theThreadPtr != 0){
+ void * retVal;
+ NdbThread_WaitFor(theThreadPtr, &retVal);
+ NdbThread_Destroy(&theThreadPtr);
+ }
+
+ theThreadPtr = NdbThread_Create(runConnect_C,
+ (void**)this,
+ 32768,
+ buf,
+ NDB_THREAD_PRIO_LOW);
+
+ NdbSleep_MilliSleep(100); // Let thread start
+
+ NdbMutex_Unlock(theMutexPtr);
+}
+
+void
+Transporter::doDisconnect() {
+
+ NdbMutex_Lock(theMutexPtr);
+ _disconnecting = true;
+ while(_connecting){
+ DEBUG("Waiting for connect to finish...");
+
+ NdbMutex_Unlock(theMutexPtr);
+ NdbSleep_MilliSleep(500);
+ NdbMutex_Lock(theMutexPtr);
+ }
+
+ _connected = false;
+
+ disconnectImpl();
+ _threadError = TE_NO_ERROR;
+ _disconnecting = false;
+
+ NdbMutex_Unlock(theMutexPtr);
+}
diff --git a/ndb/src/common/transporter/Transporter.hpp b/ndb/src/common/transporter/Transporter.hpp
new file mode 100644
index 00000000000..c562451a1b0
--- /dev/null
+++ b/ndb/src/common/transporter/Transporter.hpp
@@ -0,0 +1,177 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef Transporter_H
+#define Transporter_H
+
+#include <TransporterCallback.hpp>
+#include "TransporterDefinitions.hpp"
+#include "Packer.hpp"
+
+#include <NdbMutex.h>
+#include <NdbThread.h>
+
+class Transporter {
+ friend class TransporterRegistry;
+public:
+ virtual bool initTransporter() = 0;
+
+ /**
+ * Destructor
+ */
+ virtual ~Transporter();
+
+ /**
+ * None blocking
+ * Use isConnected() to check status
+ */
+ virtual void doConnect();
+
+ /**
+ * Blocking
+ */
+ virtual void doDisconnect();
+
+ virtual Uint32 * getWritePtr(Uint32 lenBytes, Uint32 prio) = 0;
+ virtual void updateWritePtr(Uint32 lenBytes, Uint32 prio) = 0;
+
+ /**
+ * Are we currently connected
+ */
+ bool isConnected() const;
+
+ /**
+ * Remote Node Id
+ */
+ NodeId getRemoteNodeId() const;
+
+
+ /**
+ * Set callback object
+ */
+ void setCallbackObject(void * callback);
+
+protected:
+ Transporter(NodeId lNodeId,
+ NodeId rNodeId,
+ int byteorder,
+ bool compression,
+ bool checksum,
+ bool signalId);
+
+ /**
+ * Blocking, for max timeOut milli seconds
+ * Returns true if connect succeded
+ */
+ virtual bool connectImpl(Uint32 timeOut) = 0;
+
+ /**
+ * Blocking
+ */
+ virtual void disconnectImpl() = 0;
+
+ const NodeId localNodeId;
+ const NodeId remoteNodeId;
+
+ unsigned createIndex;
+
+ int byteOrder;
+ bool compressionUsed;
+ bool checksumUsed;
+ bool signalIdUsed;
+ Packer m_packer;
+
+
+private:
+ /**
+ * Thread and mutex for connect
+ */
+ NdbThread* theThreadPtr;
+ friend void* runConnect(void * me);
+
+protected:
+ /**
+ * Error reporting from connect thread(s)
+ */
+ void reportThreadError(NodeId nodeId,
+ TransporterError errorCode);
+ Uint32 getErrorCount();
+ TransporterError getThreadError();
+ void resetThreadError();
+ TransporterError _threadError;
+ Uint32 _timeOutMillis;
+ Uint32 _errorCount;
+
+protected:
+ NdbMutex* theMutexPtr;
+ bool _connected; // Are we connected
+ bool _connecting; // Connect thread is running
+ bool _disconnecting; // We are disconnecting
+
+ void * callbackObj;
+};
+
+inline
+bool
+Transporter::isConnected() const {
+ return _connected;
+}
+
+inline
+NodeId
+Transporter::getRemoteNodeId() const {
+ return remoteNodeId;
+}
+
+inline
+void
+Transporter::reportThreadError(NodeId nodeId, TransporterError errorCode)
+{
+#if 0
+ ndbout_c("Transporter::reportThreadError (NodeId: %d, Error code: %d)",
+ nodeId, errorCode);
+#endif
+ _threadError = errorCode;
+ _errorCount++;
+}
+
+inline
+TransporterError
+Transporter::getThreadError(){
+ return _threadError;
+}
+
+inline
+Uint32
+Transporter::getErrorCount()
+{
+ return _errorCount;
+}
+
+inline
+void
+Transporter::resetThreadError()
+{
+ _threadError = TE_NO_ERROR;
+}
+
+inline
+void
+Transporter::setCallbackObject(void * callback) {
+ callbackObj = callback;
+}
+
+#endif // Define of Transporter_H
diff --git a/ndb/src/common/transporter/TransporterInternalDefinitions.hpp b/ndb/src/common/transporter/TransporterInternalDefinitions.hpp
new file mode 100644
index 00000000000..18d54ca1e89
--- /dev/null
+++ b/ndb/src/common/transporter/TransporterInternalDefinitions.hpp
@@ -0,0 +1,323 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TransporterInternalDefinitions_H
+#define TransporterInternalDefinitions_H
+
+#if defined DEBUG_TRANSPORTER || defined VM_TRACE
+#include <NdbOut.hpp>
+#endif
+
+#ifdef NDB_SOLARIS
+#define NDB_TCP_TRANSPORTER
+//#define NDB_SCI_TRANSPORTER
+#define NDB_SHM_TRANSPORTER
+#elif defined NDB_OSE || defined NDB_SOFTOSE
+#define NDB_TCP_TRANSPORTER
+#define NDB_OSE_TRANSPORTER
+#elif defined NDB_LINUX
+#define NDB_TCP_TRANSPORTER
+#define NDB_SCI_TRANSPORTER
+#define NDB_SHM_TRANSPORTER
+#elif defined NDB_WIN32
+#define NDB_TCP_TRANSPORTER
+#elif defined NDB_HPUX
+#define NDB_TCP_TRANSPORTER
+#define NDB_SHM_TRANSPORTER
+#elif defined NDB_MACOSX
+#define NDB_TCP_TRANSPORTER
+#define NDB_SHM_TRANSPORTER
+#elif defined NDB_IBMAIX
+#define NDB_TCP_TRANSPORTER
+#define NDB_SHM_TRANSPORTER
+#elif defined NDB_TRU64X
+#define NDB_TCP_TRANSPORTER
+#define NDB_SHM_TRANSPORTER
+#else
+#error unsupported platform
+#endif
+
+#ifndef HAVE_SCI
+#ifdef NDB_SCI_TRANSPORTER
+#undef NDB_SCI_TRANSPORTER
+#endif
+#endif
+
+#ifdef DEBUG_TRANSPORTER
+#define DEBUG(x) ndbout << x << endl
+#else
+#define DEBUG(x)
+#endif
+
+#if defined VM_TRACE || defined DEBUG_TRANSPORTER
+#define WARNING(X) ndbout << X << endl;
+#else
+#define WARNING(X)
+#endif
+
+// Calculate a checksum
+inline
+Uint32
+computeChecksum(const Uint32 * const startOfData, int nWords) {
+ Uint32 chksum = startOfData[0];
+ for (int i=1; i < nWords; i++)
+ chksum ^= startOfData[i];
+ return chksum;
+}
+
+struct Protocol6 {
+ Uint32 word1;
+ Uint32 word2;
+ Uint32 word3;
+
+/**
+ *
+ * b = Byte order - 4 Bits (Note 1 significant bit)
+ * g = GSN - 16 Bits
+ * p = Prio - 2 Bits
+ * c = Checksum included - 1 Bit
+ * z = Compression - 1 Bit
+ * v = Version id - 4 Bits
+ * i = Signal id included - 1 Bit
+ * m = Message length - 16 Bits (0-65536) (In word -> 0-256k bytes)
+ * d = Signal data length - 5 Bits (0-31)
+ * t = trace - 6 Bits (0-63)
+ * r = Recievers block no - 16 Bits
+ * s = Senders block no - 16 Bits
+ * u = Unused - 7 Bits
+ * f = FragmentInfo1 - 1 Bit
+ * h = FragmentInfo2 - 1 bit
+ * n = No of segments - 2 Bits
+
+ * Word 1
+ *
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * bfizcppbmmmmmmmmmmmmmmmmbhdddddb
+
+ **
+ * Word 2
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * ggggggggggggggggvvvvttttttnn
+
+ **
+ * Word 3
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * rrrrrrrrrrrrrrrrssssssssssssssss
+
+ **
+ * Word 4 (Optional Signal Id)
+ */
+
+ /**
+ * 0 = Big endian (Sparc), 1 = Little endian (Intel)
+ */
+ static Uint32 getByteOrder (const Uint32 & word1);
+ static Uint32 getCompressed (const Uint32 & word1);
+ static Uint32 getSignalIdIncluded(const Uint32 & word1);
+ static Uint32 getCheckSumIncluded(const Uint32 & word1);
+ static Uint32 getPrio (const Uint32 & word1);
+ static Uint32 getMessageLength (const Uint32 & word1);
+
+ static void setByteOrder (Uint32 & word1, Uint32 byteOrder);
+ static void setCompressed (Uint32 & word1, Uint32 compressed);
+ static void setSignalIdIncluded(Uint32 & word1, Uint32 signalId);
+ static void setCheckSumIncluded(Uint32 & word1, Uint32 checkSum);
+ static void setPrio (Uint32 & word1, Uint32 prio);
+ static void setMessageLength (Uint32 & word1, Uint32 messageLen);
+
+ static void createSignalHeader(SignalHeader * const dst,
+ const Uint32 & word1,
+ const Uint32 & word2,
+ const Uint32 & word3);
+
+ static void createProtocol6Header(Uint32 & word1,
+ Uint32 & word2,
+ Uint32 & word3,
+ const SignalHeader * const src);
+};
+
+#define WORD1_BYTEORDER_MASK (0x81000081)
+#define WORD1_SIGNALID_MASK (0x00000004)
+#define WORD1_COMPRESSED_MASK (0x00000008)
+#define WORD1_CHECKSUM_MASK (0x00000010)
+#define WORD1_PRIO_MASK (0x00000060)
+#define WORD1_MESSAGELEN_MASK (0x00FFFF00)
+#define WORD1_SIGNAL_LEN_MASK (0x7C000000)
+#define WORD1_FRAG_INF_MASK (0x00000002)
+#define WORD1_FRAG_INF2_MASK (0x02000000)
+
+#define WORD1_FRAG_INF_SHIFT (1)
+#define WORD1_SIGNALID_SHIFT (2)
+#define WORD1_COMPRESSED_SHIFT (3)
+#define WORD1_CHECKSUM_SHIFT (4)
+#define WORD1_PRIO_SHIFT (5)
+#define WORD1_MESSAGELEN_SHIFT (8)
+#define WORD1_FRAG_INF2_SHIFT (25)
+#define WORD1_SIGNAL_LEN_SHIFT (26)
+
+#define WORD2_VERID_GSN_MASK (0x000FFFFF)
+#define WORD2_TRACE_MASK (0x03f00000)
+#define WORD2_SEC_COUNT_MASK (0x0c000000)
+
+#define WORD2_TRACE_SHIFT (20)
+#define WORD2_SEC_COUNT_SHIFT (26)
+
+#define WORD3_SENDER_MASK (0x0000FFFF)
+#define WORD3_RECEIVER_MASK (0xFFFF0000)
+
+#define WORD3_RECEIVER_SHIFT (16)
+
+inline
+Uint32
+Protocol6::getByteOrder(const Uint32 & word1){
+ return word1 & 1;
+}
+
+inline
+Uint32
+Protocol6::getCompressed(const Uint32 & word1){
+ return (word1 & WORD1_COMPRESSED_MASK) >> WORD1_COMPRESSED_SHIFT;
+}
+
+inline
+Uint32
+Protocol6::getSignalIdIncluded(const Uint32 & word1){
+ return (word1 & WORD1_SIGNALID_MASK) >> WORD1_SIGNALID_SHIFT;
+}
+
+inline
+Uint32
+Protocol6::getCheckSumIncluded(const Uint32 & word1){
+ return (word1 & WORD1_CHECKSUM_MASK) >> WORD1_CHECKSUM_SHIFT;
+}
+
+inline
+Uint32
+Protocol6::getMessageLength(const Uint32 & word1){
+ return (word1 & WORD1_MESSAGELEN_MASK) >> WORD1_MESSAGELEN_SHIFT;
+}
+
+inline
+Uint32
+Protocol6::getPrio(const Uint32 & word1){
+ return (word1 & WORD1_PRIO_MASK) >> WORD1_PRIO_SHIFT;
+}
+
+inline
+void
+Protocol6::setByteOrder(Uint32 & word1, Uint32 byteOrder){
+ Uint32 tmp = byteOrder;
+ tmp |= (tmp << 7);
+ tmp |= (tmp << 24);
+ word1 |= (tmp & WORD1_BYTEORDER_MASK);
+}
+
+inline
+void
+Protocol6::setCompressed(Uint32 & word1, Uint32 compressed){
+ word1 |= ((compressed << WORD1_COMPRESSED_SHIFT) & WORD1_COMPRESSED_MASK);
+}
+
+inline
+void
+Protocol6::setSignalIdIncluded(Uint32 & word1, Uint32 signalId){
+ word1 |= ((signalId << WORD1_SIGNALID_SHIFT) & WORD1_SIGNALID_MASK);
+}
+
+inline
+void
+Protocol6::setCheckSumIncluded(Uint32 & word1, Uint32 checkSum){
+ word1 |= ((checkSum << WORD1_CHECKSUM_SHIFT) & WORD1_CHECKSUM_MASK);
+}
+
+inline
+void
+Protocol6::setMessageLength(Uint32 & word1, Uint32 messageLen){
+ word1 |= ((messageLen << WORD1_MESSAGELEN_SHIFT) & WORD1_MESSAGELEN_MASK);
+}
+
+inline
+void
+Protocol6::setPrio(Uint32 & word1, Uint32 prio){
+ word1 |= ((prio << WORD1_PRIO_SHIFT) & WORD1_PRIO_MASK);
+}
+
+inline
+void
+Protocol6::createSignalHeader(SignalHeader * const dst,
+ const Uint32 & word1,
+ const Uint32 & word2,
+ const Uint32 & word3){
+
+ Uint32 signal_len = (word1 & WORD1_SIGNAL_LEN_MASK)>> WORD1_SIGNAL_LEN_SHIFT;
+ Uint32 fragInfo1 = (word1 & WORD1_FRAG_INF_MASK) >> (WORD1_FRAG_INF_SHIFT-1);
+ Uint32 fragInfo2 = (word1 & WORD1_FRAG_INF2_MASK) >> (WORD1_FRAG_INF2_SHIFT);
+ Uint32 trace = (word2 & WORD2_TRACE_MASK) >> WORD2_TRACE_SHIFT;
+ Uint32 verid_gsn = (word2 & WORD2_VERID_GSN_MASK);
+ Uint32 secCount = (word2 & WORD2_SEC_COUNT_MASK) >> WORD2_SEC_COUNT_SHIFT;
+
+ dst->theTrace = trace;
+ dst->m_noOfSections = secCount;
+ dst->m_fragmentInfo = fragInfo1 | fragInfo2;
+
+ dst->theLength = signal_len;
+ dst->theVerId_signalNumber = verid_gsn;
+
+ Uint32 sBlockNum = (word3 & WORD3_SENDER_MASK);
+ Uint32 rBlockNum = (word3 & WORD3_RECEIVER_MASK) >> WORD3_RECEIVER_SHIFT;
+
+ dst->theSendersBlockRef = sBlockNum;
+ dst->theReceiversBlockNumber = rBlockNum;
+}
+
+inline
+void
+Protocol6::createProtocol6Header(Uint32 & word1,
+ Uint32 & word2,
+ Uint32 & word3,
+ const SignalHeader * const src){
+ const Uint32 signal_len = src->theLength;
+ const Uint32 fragInfo = src->m_fragmentInfo;
+ const Uint32 fragInfo1 = (fragInfo & 2);
+ const Uint32 fragInfo2 = (fragInfo & 1);
+
+ const Uint32 trace = src->theTrace;
+ const Uint32 verid_gsn = src->theVerId_signalNumber;
+ const Uint32 secCount = src->m_noOfSections;
+
+ word1 |= ((signal_len << WORD1_SIGNAL_LEN_SHIFT) & WORD1_SIGNAL_LEN_MASK);
+ word1 |= ((fragInfo1 << (WORD1_FRAG_INF_SHIFT-1)) & WORD1_FRAG_INF_MASK);
+ word1 |= ((fragInfo2 << WORD1_FRAG_INF2_SHIFT) & WORD1_FRAG_INF2_MASK);
+
+ word2 |= ((trace << WORD2_TRACE_SHIFT) & WORD2_TRACE_MASK);
+ word2 |= (verid_gsn & WORD2_VERID_GSN_MASK);
+ word2 |= ((secCount << WORD2_SEC_COUNT_SHIFT) & WORD2_SEC_COUNT_MASK);
+
+ Uint32 sBlockNum = src->theSendersBlockRef ;
+ Uint32 rBlockNum = src->theReceiversBlockNumber ;
+
+ word3 |= (sBlockNum & WORD3_SENDER_MASK);
+ word3 |= ((rBlockNum << WORD3_RECEIVER_SHIFT) & WORD3_RECEIVER_MASK);
+}
+
+// Define of TransporterInternalDefinitions_H
+#endif
diff --git a/ndb/src/common/transporter/TransporterRegistry.cpp b/ndb/src/common/transporter/TransporterRegistry.cpp
new file mode 100644
index 00000000000..dcd957f40ce
--- /dev/null
+++ b/ndb/src/common/transporter/TransporterRegistry.cpp
@@ -0,0 +1,1188 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "TransporterRegistry.hpp"
+#include "TransporterInternalDefinitions.hpp"
+
+#include "Transporter.hpp"
+
+#include <assert.h>
+#include <stdlib.h>
+
+#ifdef NDB_TCP_TRANSPORTER
+#include "TCP_Transporter.hpp"
+#endif
+
+#ifdef NDB_OSE_TRANSPORTER
+#include "OSE_Receiver.hpp"
+#include "OSE_Transporter.hpp"
+#endif
+
+#ifdef NDB_SCI_TRANSPORTER
+#include "SCI_Transporter.hpp"
+#endif
+
+#ifdef NDB_SHM_TRANSPORTER
+#include "SHM_Transporter.hpp"
+#endif
+
+#include "TransporterCallback.hpp"
+#include "NdbOut.hpp"
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#define STEPPING 1
+
+TransporterRegistry::TransporterRegistry(void * callback,
+ unsigned _maxTransporters,
+ unsigned sizeOfLongSignalMemory) {
+
+ nodeIdSpecified = false;
+ maxTransporters = _maxTransporters;
+ sendCounter = 1;
+ m_ccCount = 0;
+ m_ccIndex = 0;
+ m_ccStep = STEPPING;
+ m_ccReady = false;
+ m_nTransportersPerformConnect=0;
+
+ callbackObj=callback;
+
+ theTCPTransporters = new TCP_Transporter * [maxTransporters];
+ theSCITransporters = new SCI_Transporter * [maxTransporters];
+ theSHMTransporters = new SHM_Transporter * [maxTransporters];
+ theOSETransporters = new OSE_Transporter * [maxTransporters];
+ theTransporterTypes = new TransporterType [maxTransporters];
+ theTransporters = new Transporter * [maxTransporters];
+ performStates = new PerformState [maxTransporters];
+ ioStates = new IOState [maxTransporters];
+
+ // Initialize member variables
+ nTransporters = 0;
+ nTCPTransporters = 0;
+ nSCITransporters = 0;
+ nSHMTransporters = 0;
+ nOSETransporters = 0;
+
+ // Initialize the transporter arrays
+ for (unsigned i=0; i<maxTransporters; i++) {
+ theTCPTransporters[i] = NULL;
+ theSCITransporters[i] = NULL;
+ theSHMTransporters[i] = NULL;
+ theOSETransporters[i] = NULL;
+ theTransporters[i] = NULL;
+ performStates[i] = PerformNothing;
+ ioStates[i] = NoHalt;
+ }
+ theOSEReceiver = 0;
+ theOSEJunkSocketSend = 0;
+ theOSEJunkSocketRecv = 0;
+}
+
+TransporterRegistry::~TransporterRegistry() {
+
+ removeAll();
+
+ delete[] theTCPTransporters;
+ delete[] theSCITransporters;
+ delete[] theSHMTransporters;
+ delete[] theOSETransporters;
+ delete[] theTransporterTypes;
+ delete[] theTransporters;
+ delete[] performStates;
+ delete[] ioStates;
+
+#ifdef NDB_OSE_TRANSPORTER
+ if(theOSEReceiver != NULL){
+ theOSEReceiver->destroyPhantom();
+ delete theOSEReceiver;
+ theOSEReceiver = 0;
+ }
+#endif
+}
+
+void
+TransporterRegistry::removeAll(){
+ for(unsigned i = 0; i<maxTransporters; i++){
+ if(theTransporters[i] != NULL)
+ removeTransporter(theTransporters[i]->getRemoteNodeId());
+ }
+}
+
+void
+TransporterRegistry::disconnectAll(){
+ for(unsigned i = 0; i<maxTransporters; i++){
+ if(theTransporters[i] != NULL)
+ theTransporters[i]->doDisconnect();
+ }
+}
+
+bool
+TransporterRegistry::init(NodeId nodeId) {
+ nodeIdSpecified = true;
+ localNodeId = nodeId;
+
+ DEBUG("TransporterRegistry started node: " << localNodeId);
+
+ // return allocateLongSignalMemoryPool(nLargeSegments);
+ return true;
+}
+
+bool
+TransporterRegistry::createTransporter(TCP_TransporterConfiguration *config) {
+#ifdef NDB_TCP_TRANSPORTER
+
+ if(!nodeIdSpecified){
+ init(config->localNodeId);
+ }
+
+ if(config->localNodeId != localNodeId)
+ return false;
+
+ if(theTransporters[config->remoteNodeId] != NULL)
+ return false;
+
+
+ TCP_Transporter * t = new TCP_Transporter(config->sendBufferSize,
+ config->maxReceiveSize,
+ config->port,
+ config->remoteHostName,
+ config->localHostName,
+ config->remoteNodeId,
+ localNodeId,
+ config->byteOrder,
+ config->compression,
+ config->checksum,
+ config->signalId);
+ if (t == NULL)
+ return false;
+ else if (!t->initTransporter()) {
+ delete t;
+ return false;
+ }
+
+ t->setCallbackObject(callbackObj);
+
+ // Put the transporter in the transporter arrays
+ theTCPTransporters[nTCPTransporters] = t;
+ theTransporters[t->getRemoteNodeId()] = t;
+ theTransporterTypes[t->getRemoteNodeId()] = tt_TCP_TRANSPORTER;
+ performStates[t->getRemoteNodeId()] = PerformNothing;
+ nTransporters++;
+ nTCPTransporters++;
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ t->theReceiverPid = theReceiverPid;
+#endif
+
+ return true;
+#else
+ return false;
+#endif
+}
+
+bool
+TransporterRegistry::createTransporter(OSE_TransporterConfiguration *conf) {
+#ifdef NDB_OSE_TRANSPORTER
+
+ if(!nodeIdSpecified){
+ init(conf->localNodeId);
+ }
+
+ if(conf->localNodeId != localNodeId)
+ return false;
+
+ if(theTransporters[conf->remoteNodeId] != NULL)
+ return false;
+
+ if(theOSEReceiver == NULL){
+ theOSEReceiver = new OSE_Receiver(this,
+ 10,
+ localNodeId);
+ }
+
+ OSE_Transporter * t = new OSE_Transporter(conf->prioASignalSize,
+ conf->prioBSignalSize,
+ localNodeId,
+ conf->localHostName,
+ conf->remoteNodeId,
+ conf->remoteHostName,
+ conf->byteOrder,
+ conf->compression,
+ conf->checksum,
+ conf->signalId);
+ if (t == NULL)
+ return false;
+ else if (!t->initTransporter()) {
+ delete t;
+ return false;
+ }
+ t->setCallbackObject(callbackObj);
+ // Put the transporter in the transporter arrays
+ theOSETransporters[nOSETransporters] = t;
+ theTransporters[t->getRemoteNodeId()] = t;
+ theTransporterTypes[t->getRemoteNodeId()] = tt_OSE_TRANSPORTER;
+ performStates[t->getRemoteNodeId()] = PerformNothing;
+
+ nTransporters++;
+ nOSETransporters++;
+
+ return true;
+#else
+ return false;
+#endif
+}
+
+bool
+TransporterRegistry::createTransporter(SCI_TransporterConfiguration *config) {
+#ifdef NDB_SCI_TRANSPORTER
+
+ if(!SCI_Transporter::initSCI())
+ abort();
+
+ if(!nodeIdSpecified){
+ init(config->localNodeId);
+ }
+
+ if(config->localNodeId != localNodeId)
+ return false;
+
+ if(theTransporters[config->remoteNodeId] != NULL)
+ return false;
+
+ SCI_Transporter * t = new SCI_Transporter(config->sendLimit,
+ config->bufferSize,
+ config->nLocalAdapters,
+ config->remoteSciNodeId0,
+ config->remoteSciNodeId1,
+ localNodeId,
+ config->remoteNodeId,
+ config->byteOrder,
+ config->compression,
+ config->checksum,
+ config->signalId);
+
+ if (t == NULL)
+ return false;
+ else if (!t->initTransporter()) {
+ delete t;
+ return false;
+ }
+ t->setCallbackObject(callbackObj);
+ // Put the transporter in the transporter arrays
+ theSCITransporters[nSCITransporters] = t;
+ theTransporters[t->getRemoteNodeId()] = t;
+ theTransporterTypes[t->getRemoteNodeId()] = tt_SCI_TRANSPORTER;
+ performStates[t->getRemoteNodeId()] = PerformNothing;
+ nTransporters++;
+ nSCITransporters++;
+
+ return true;
+#else
+ return false;
+#endif
+}
+
+bool
+TransporterRegistry::createTransporter(SHM_TransporterConfiguration *config) {
+#ifdef NDB_SHM_TRANSPORTER
+ if(!nodeIdSpecified){
+ init(config->localNodeId);
+ }
+
+ if(config->localNodeId != localNodeId)
+ return false;
+
+ if(theTransporters[config->remoteNodeId] != NULL)
+ return false;
+
+ SHM_Transporter * t = new SHM_Transporter(config->localNodeId,
+ config->remoteNodeId,
+ config->shmKey,
+ config->shmSize,
+ config->compression,
+ config->checksum,
+ config->signalId
+ );
+ if (t == NULL)
+ return false;
+ else if (!t->initTransporter()) {
+ delete t;
+ return false;
+ }
+ t->setCallbackObject(callbackObj);
+ // Put the transporter in the transporter arrays
+ theSHMTransporters[nSHMTransporters] = t;
+ theTransporters[t->getRemoteNodeId()] = t;
+ theTransporterTypes[t->getRemoteNodeId()] = tt_SHM_TRANSPORTER;
+ performStates[t->getRemoteNodeId()] = PerformNothing;
+
+ nTransporters++;
+ nSHMTransporters++;
+
+ return true;
+#else
+ return false;
+#endif
+}
+
+
+void
+TransporterRegistry::removeTransporter(NodeId nodeId) {
+
+ DEBUG("Removing transporter from " << localNodeId
+ << " to " << nodeId);
+
+ if(theTransporters[nodeId] == NULL)
+ return;
+
+ theTransporters[nodeId]->doDisconnect();
+
+ const TransporterType type = theTransporterTypes[nodeId];
+
+ int ind = 0;
+ switch(type){
+ case tt_TCP_TRANSPORTER:
+#ifdef NDB_TCP_TRANSPORTER
+ for(; ind < nTCPTransporters; ind++)
+ if(theTCPTransporters[ind]->getRemoteNodeId() == nodeId)
+ break;
+ ind++;
+ for(; ind<nTCPTransporters; ind++)
+ theTCPTransporters[ind-1] = theTCPTransporters[ind];
+ nTCPTransporters --;
+#endif
+ break;
+ case tt_SCI_TRANSPORTER:
+#ifdef NDB_SCI_TRANSPORTER
+ for(; ind < nSCITransporters; ind++)
+ if(theSCITransporters[ind]->getRemoteNodeId() == nodeId)
+ break;
+ ind++;
+ for(; ind<nSCITransporters; ind++)
+ theSCITransporters[ind-1] = theSCITransporters[ind];
+ nSCITransporters --;
+#endif
+ break;
+ case tt_SHM_TRANSPORTER:
+#ifdef NDB_SHM_TRANSPORTER
+ for(; ind < nSHMTransporters; ind++)
+ if(theSHMTransporters[ind]->getRemoteNodeId() == nodeId)
+ break;
+ ind++;
+ for(; ind<nSHMTransporters; ind++)
+ theSHMTransporters[ind-1] = theSHMTransporters[ind];
+ nSHMTransporters --;
+#endif
+ break;
+ case tt_OSE_TRANSPORTER:
+#ifdef NDB_OSE_TRANSPORTER
+ for(; ind < nOSETransporters; ind++)
+ if(theOSETransporters[ind]->getRemoteNodeId() == nodeId)
+ break;
+ ind++;
+ for(; ind<nOSETransporters; ind++)
+ theOSETransporters[ind-1] = theOSETransporters[ind];
+ nOSETransporters --;
+#endif
+ break;
+ }
+
+ nTransporters--;
+
+ // Delete the transporter and remove it from theTransporters array
+ delete theTransporters[nodeId];
+ theTransporters[nodeId] = NULL;
+}
+
+SendStatus
+TransporterRegistry::prepareSend(const SignalHeader * const signalHeader,
+ Uint8 prio,
+ const Uint32 * const signalData,
+ NodeId nodeId,
+ const LinearSectionPtr ptr[3]){
+
+
+ Transporter *t = theTransporters[nodeId];
+ if(t != NULL &&
+ (((ioStates[nodeId] != HaltOutput) && (ioStates[nodeId] != HaltIO)) ||
+ (signalHeader->theReceiversBlockNumber == 252))) {
+
+ if(t->isConnected()){
+ Uint32 lenBytes = t->m_packer.getMessageLength(signalHeader, ptr);
+ if(lenBytes <= MAX_MESSAGE_SIZE){
+ Uint32 * insertPtr = t->getWritePtr(lenBytes, prio);
+ if(insertPtr != 0){
+ t->m_packer.pack(insertPtr, prio, signalHeader, signalData, ptr);
+ t->updateWritePtr(lenBytes, prio);
+ return SEND_OK;
+ }
+
+ int sleepTime = 2;
+
+ /**
+ * @note: on linux/i386 the granularity is 10ms
+ * so sleepTime = 2 generates a 10 ms sleep.
+ */
+ for(int i = 0; i<50; i++){
+ if((nSHMTransporters+nSCITransporters) == 0)
+ NdbSleep_MilliSleep(sleepTime);
+ insertPtr = t->getWritePtr(lenBytes, prio);
+ if(insertPtr != 0){
+ t->m_packer.pack(insertPtr, prio, signalHeader, signalData, ptr);
+ t->updateWritePtr(lenBytes, prio);
+ break;
+ }
+ }
+
+ if(insertPtr != 0){
+ /**
+ * Send buffer full, but resend works
+ */
+ reportError(callbackObj, nodeId, TE_SEND_BUFFER_FULL);
+ return SEND_OK;
+ }
+
+ WARNING("Signal to " << nodeId << " lost(buffer)");
+ reportError(callbackObj, nodeId, TE_SIGNAL_LOST_SEND_BUFFER_FULL);
+ return SEND_BUFFER_FULL;
+ } else {
+ return SEND_MESSAGE_TOO_BIG;
+ }
+ } else {
+ DEBUG("Signal to " << nodeId << " lost(disconnect) ");
+ return SEND_DISCONNECTED;
+ }
+ } else {
+ DEBUG("Discarding message to block: "
+ << signalHeader->theReceiversBlockNumber
+ << " node: " << nodeId);
+
+ if(t == NULL)
+ return SEND_UNKNOWN_NODE;
+
+ return SEND_BLOCKED;
+ }
+}
+
+SendStatus
+TransporterRegistry::prepareSend(const SignalHeader * const signalHeader,
+ Uint8 prio,
+ const Uint32 * const signalData,
+ NodeId nodeId,
+ class SectionSegmentPool & thePool,
+ const SegmentedSectionPtr ptr[3]){
+
+
+ Transporter *t = theTransporters[nodeId];
+ if(t != NULL &&
+ (((ioStates[nodeId] != HaltOutput) && (ioStates[nodeId] != HaltIO)) ||
+ (signalHeader->theReceiversBlockNumber == 252))) {
+
+ if(t->isConnected()){
+ Uint32 lenBytes = t->m_packer.getMessageLength(signalHeader, ptr);
+ if(lenBytes <= MAX_MESSAGE_SIZE){
+ Uint32 * insertPtr = t->getWritePtr(lenBytes, prio);
+ if(insertPtr != 0){
+ t->m_packer.pack(insertPtr, prio, signalHeader, signalData, thePool, ptr);
+ t->updateWritePtr(lenBytes, prio);
+ return SEND_OK;
+ }
+
+
+ /**
+ * @note: on linux/i386 the granularity is 10ms
+ * so sleepTime = 2 generates a 10 ms sleep.
+ */
+ int sleepTime = 2;
+ for(int i = 0; i<50; i++){
+ if((nSHMTransporters+nSCITransporters) == 0)
+ NdbSleep_MilliSleep(sleepTime);
+ insertPtr = t->getWritePtr(lenBytes, prio);
+ if(insertPtr != 0){
+ t->m_packer.pack(insertPtr, prio, signalHeader, signalData, thePool, ptr);
+ t->updateWritePtr(lenBytes, prio);
+ break;
+ }
+ }
+
+ if(insertPtr != 0){
+ /**
+ * Send buffer full, but resend works
+ */
+ reportError(callbackObj, nodeId, TE_SEND_BUFFER_FULL);
+ return SEND_OK;
+ }
+
+ WARNING("Signal to " << nodeId << " lost(buffer)");
+ reportError(callbackObj, nodeId, TE_SIGNAL_LOST_SEND_BUFFER_FULL);
+ return SEND_BUFFER_FULL;
+ } else {
+ return SEND_MESSAGE_TOO_BIG;
+ }
+ } else {
+ DEBUG("Signal to " << nodeId << " lost(disconnect) ");
+ return SEND_DISCONNECTED;
+ }
+ } else {
+ DEBUG("Discarding message to block: "
+ << signalHeader->theReceiversBlockNumber
+ << " node: " << nodeId);
+
+ if(t == NULL)
+ return SEND_UNKNOWN_NODE;
+
+ return SEND_BLOCKED;
+ }
+}
+
+void
+TransporterRegistry::external_IO(Uint32 timeOutMillis) {
+ //-----------------------------------------------------------
+ // Most of the time we will send the buffers here and then wait
+ // for new signals. Thus we start by sending without timeout
+ // followed by the receive part where we expect to sleep for
+ // a while.
+ //-----------------------------------------------------------
+ if(pollReceive(timeOutMillis)){
+ performReceive();
+ }
+ performSend();
+}
+
+Uint32
+TransporterRegistry::pollReceive(Uint32 timeOutMillis){
+ Uint32 retVal = 0;
+#ifdef NDB_OSE_TRANSPORTER
+ retVal |= poll_OSE(timeOutMillis);
+ retVal |= poll_TCP(0);
+ return retVal;
+#endif
+
+ if((nSHMTransporters+nSCITransporters) > 0)
+ timeOutMillis=0;
+#ifdef NDB_TCP_TRANSPORTER
+ if(nTCPTransporters > 0)
+ retVal |= poll_TCP(timeOutMillis);
+ else
+ tcpReadSelectReply = 0;
+#endif
+#ifdef NDB_SCI_TRANSPORTER
+ if(nSCITransporters > 0)
+ retVal |= poll_SCI(timeOutMillis);
+#endif
+#ifdef NDB_SHM_TRANSPORTER
+ if(nSHMTransporters > 0)
+ retVal |= poll_SHM(timeOutMillis);
+#endif
+ return retVal;
+}
+
+
+#ifdef NDB_SCI_TRANSPORTER
+Uint32
+TransporterRegistry::poll_SCI(Uint32 timeOutMillis){
+
+ for (int i=0; i<nSCITransporters; i++) {
+ SCI_Transporter * t = theSCITransporters[i];
+ if (t->isConnected()) {
+ if(t->hasDataToRead())
+ return 1;
+ }
+ }
+ return 0;
+}
+#endif
+
+
+#ifdef NDB_SHM_TRANSPORTER
+Uint32
+TransporterRegistry::poll_SHM(Uint32 timeOutMillis)
+{
+ for(int j=0; j < 20; j++)
+ for (int i=0; i<nSHMTransporters; i++) {
+ SHM_Transporter * t = theSHMTransporters[i];
+ if (t->isConnected()) {
+ if(t->hasDataToRead()) {
+ return 1;
+ }
+ }
+ }
+ /**
+ * @note: granularity of linux/i386 timer is not good enough.
+ * Can't sleep if using SHM as it is now.
+ */
+ /*
+ if(timeOutMillis > 0)
+ NdbSleep_MilliSleep(timeOutMillis);
+ else
+ NdbSleep_MilliSleep(1);
+ */
+ return 0;
+#if 0
+ NDB_TICKS startTime = NdbTick_CurrentMillisecond();
+ for(int i=0; i<100; i++) {
+ for (int i=0; i<nSHMTransporters; i++) {
+ SHM_Transporter * t = theSHMTransporters[i];
+ if (t->isConnected()) {
+ if(t->hasDataToRead()){
+ return 1;
+ }
+ else
+ continue;
+ }
+ else
+ continue;
+ }
+
+ if(NdbTick_CurrentMillisecond() > (startTime +timeOutMillis))
+ return 0;
+ }
+ NdbSleep_MilliSleep(5);
+ return 0;
+
+#endif
+#if 0
+
+ for(int j=0; j < 100; j++) {
+ for (int i=0; i<nSHMTransporters; i++) {
+ SHM_Transporter * t = theSHMTransporters[i];
+ if (t->isConnected()) {
+ if(t->hasDataToRead())
+ return 1;
+ }
+ }
+ }
+ return 0;
+#endif
+}
+
+
+#endif
+
+#ifdef NDB_OSE_TRANSPORTER
+Uint32
+TransporterRegistry::poll_OSE(Uint32 timeOutMillis){
+ if(theOSEReceiver != NULL){
+ return theOSEReceiver->doReceive(timeOutMillis);
+ }
+ NdbSleep_MilliSleep(timeOutMillis);
+ return 0;
+}
+#endif
+
+#ifdef NDB_TCP_TRANSPORTER
+Uint32
+TransporterRegistry::poll_TCP(Uint32 timeOutMillis){
+
+ if (nTCPTransporters == 0){
+ tcpReadSelectReply = 0;
+ return 0;
+ }
+
+ struct timeval timeout;
+#ifdef NDB_OSE
+
+ // Return directly if there are no TCP transporters configured
+
+ if(timeOutMillis <= 1){
+ timeout.tv_sec = 0;
+ timeout.tv_usec = 1025;
+ } else {
+ timeout.tv_sec = timeOutMillis / 1000;
+ timeout.tv_usec = (timeOutMillis % 1000) * 1000;
+ }
+#else
+ timeout.tv_sec = timeOutMillis / 1000;
+ timeout.tv_usec = (timeOutMillis % 1000) * 1000;
+#endif
+
+ NDB_SOCKET_TYPE maxSocketValue = 0;
+
+ // Needed for TCP/IP connections
+ // The read- and writeset are used by select
+
+ FD_ZERO(&tcpReadset);
+
+ // Prepare for sending and receiving
+ for (int i = 0; i < nTCPTransporters; i++) {
+ TCP_Transporter * t = theTCPTransporters[i];
+
+ // If the transporter is connected
+ if (t->isConnected()) {
+
+ const NDB_SOCKET_TYPE socket = t->getSocket();
+ // Find the highest socket value. It will be used by select
+ if (socket > maxSocketValue)
+ maxSocketValue = socket;
+
+ // Put the connected transporters in the socket read-set
+ FD_SET(socket, &tcpReadset);
+ }
+ }
+
+ // The highest socket value plus one
+ maxSocketValue++;
+
+ tcpReadSelectReply = select(maxSocketValue, &tcpReadset, 0, 0, &timeout);
+#ifdef NDB_WIN32
+ if(tcpReadSelectReply == SOCKET_ERROR)
+ {
+ NdbSleep_MilliSleep(timeOutMillis);
+ }
+#endif
+
+ return tcpReadSelectReply;
+}
+#endif
+
+
+void
+TransporterRegistry::performReceive(){
+#ifdef NDB_OSE_TRANSPORTER
+ if(theOSEReceiver != 0){
+ while(theOSEReceiver->hasData()){
+ NodeId remoteNodeId;
+ Uint32 * readPtr;
+ Uint32 sz = theOSEReceiver->getReceiveData(&remoteNodeId, &readPtr);
+ Uint32 szUsed = unpack(readPtr,
+ sz,
+ remoteNodeId,
+ ioStates[remoteNodeId]);
+#ifdef DEBUG_TRANSPORTER
+ /**
+ * OSE transporter can handle executions of
+ * half signals
+ */
+ assert(sz == szUsed);
+#endif
+ theOSEReceiver->updateReceiveDataPtr(szUsed);
+ theOSEReceiver->doReceive(0);
+ // checkJobBuffer();
+ }
+ }
+#endif
+
+#ifdef NDB_TCP_TRANSPORTER
+ if(tcpReadSelectReply > 0){
+ for (int i=0; i<nTCPTransporters; i++) {
+ checkJobBuffer();
+ TCP_Transporter *t = theTCPTransporters[i];
+ const NodeId nodeId = t->getRemoteNodeId();
+ const NDB_SOCKET_TYPE socket = t->getSocket();
+ if(performStates[nodeId] == PerformIO){
+ if(t->isConnected() && FD_ISSET(socket, &tcpReadset)) {
+ const int receiveSize = t->doReceive();
+ if(receiveSize > 0){
+ Uint32 * ptr;
+ Uint32 sz = t->getReceiveData(&ptr);
+ Uint32 szUsed = unpack(ptr, sz, nodeId, ioStates[nodeId]);
+ t->updateReceiveDataPtr(szUsed);
+ }
+ }
+ }
+ }
+ }
+#endif
+
+
+#ifdef NDB_SCI_TRANSPORTER
+ //performReceive
+ //do prepareReceive on the SCI transporters (prepareReceive(t,,,,))
+ for (int i=0; i<nSCITransporters; i++) {
+ checkJobBuffer();
+ SCI_Transporter *t = theSCITransporters[i];
+ const NodeId nodeId = t->getRemoteNodeId();
+ if(performStates[nodeId] == PerformIO){
+ if(t->isConnected() && t->checkConnected()){
+ Uint32 * readPtr, * eodPtr;
+ t->getReceivePtr(&readPtr, &eodPtr);
+ readPtr = unpack(readPtr, eodPtr, nodeId, ioStates[nodeId]);
+ t->updateReceivePtr(readPtr);
+ }
+ }
+ }
+#endif
+#ifdef NDB_SHM_TRANSPORTER
+ for (int i=0; i<nSHMTransporters; i++) {
+ checkJobBuffer();
+ SHM_Transporter *t = theSHMTransporters[i];
+ const NodeId nodeId = t->getRemoteNodeId();
+ if(performStates[nodeId] == PerformIO){
+ if(t->isConnected() && t->checkConnected()){
+ Uint32 * readPtr, * eodPtr;
+ t->getReceivePtr(&readPtr, &eodPtr);
+ readPtr = unpack(readPtr, eodPtr, nodeId, ioStates[nodeId]);
+ t->updateReceivePtr(readPtr);
+ }
+ }
+ }
+#endif
+}
+
+static int x = 0;
+void
+TransporterRegistry::performSend(){
+
+ sendCounter = 1;
+
+#ifdef NDB_OSE_TRANSPORTER
+ for (int i = 0; i < nOSETransporters; i++){
+ OSE_Transporter *t = theOSETransporters[i];
+ if((performStates[t->getRemoteNodeId()] == PerformIO) &&
+ (t->isConnected())) {
+ t->doSend();
+ }//if
+ }//for
+#endif
+
+#ifdef NDB_TCP_TRANSPORTER
+#ifdef NDB_OSE
+ {
+ int maxSocketValue = 0;
+
+ // Needed for TCP/IP connections
+ // The writeset are used by select
+ fd_set writeset;
+ FD_ZERO(&writeset);
+
+ // Prepare for sending and receiving
+ for (int i = 0; i < nTCPTransporters; i++) {
+ TCP_Transporter * t = theTCPTransporters[i];
+
+ // If the transporter is connected
+ if ((t->hasDataToSend()) && (t->isConnected())) {
+ const int socket = t->getSocket();
+ // Find the highest socket value. It will be used by select
+ if (socket > maxSocketValue) {
+ maxSocketValue = socket;
+ }//if
+ FD_SET(socket, &writeset);
+ }//if
+ }//for
+
+ // The highest socket value plus one
+ if(maxSocketValue == 0)
+ return;
+
+ maxSocketValue++;
+ struct timeval timeout = { 0, 1025 };
+ Uint32 tmp = select(maxSocketValue, 0, &writeset, 0, &timeout);
+
+ if (tmp == 0) {
+ return;
+ }//if
+ for (int i = 0; i < nTCPTransporters; i++) {
+ TCP_Transporter *t = theTCPTransporters[i];
+ const NodeId nodeId = t->getRemoteNodeId();
+ const int socket = t->getSocket();
+ if(performStates[nodeId] == PerformIO){
+ if(t->isConnected() && FD_ISSET(socket, &writeset)) {
+ t->doSend();
+ }//if
+ }//if
+ }//for
+ }
+#endif
+#ifdef NDB_TCP_TRANSPORTER
+ for (int i = x; i < nTCPTransporters; i++) {
+ TCP_Transporter *t = theTCPTransporters[i];
+ if (t &&
+ (t->hasDataToSend()) &&
+ (t->isConnected()) &&
+ (performStates[t->getRemoteNodeId()] == PerformIO)) {
+ t->doSend();
+ }//if
+ }//for
+ for (int i = 0; i < x && i < nTCPTransporters; i++) {
+ TCP_Transporter *t = theTCPTransporters[i];
+ if (t &&
+ (t->hasDataToSend()) &&
+ (t->isConnected()) &&
+ (performStates[t->getRemoteNodeId()] == PerformIO)) {
+ t->doSend();
+ }//if
+ }//for
+ x++;
+ if (x == nTCPTransporters) x = 0;
+#endif
+#endif
+#ifdef NDB_SCI_TRANSPORTER
+ //scroll through the SCI transporters,
+ // get each transporter, check if connected, send data
+ for (int i=0; i<nSCITransporters; i++) {
+ SCI_Transporter *t = theSCITransporters[i];
+ const NodeId nodeId = t->getRemoteNodeId();
+
+ if(performStates[nodeId] == PerformIO){
+ if(t->isConnected() && t->hasDataToSend()) {
+ t->doSend();
+ } //if
+ } //if
+ } //if
+#endif
+}
+
+int
+TransporterRegistry::forceSendCheck(int sendLimit){
+ int tSendCounter = sendCounter;
+ sendCounter = tSendCounter + 1;
+ if (tSendCounter >= sendLimit) {
+ performSend();
+ sendCounter = 1;
+ return 1;
+ }//if
+ return 0;
+}//TransporterRegistry::forceSendCheck()
+
+#ifdef DEBUG_TRANSPORTER
+void
+TransporterRegistry::printState(){
+ ndbout << "-- TransporterRegistry -- " << endl << endl
+ << "Transporters = " << nTransporters << endl;
+ for(int i = 0; i<maxTransporters; i++)
+ if(theTransporters[i] != NULL){
+ const NodeId remoteNodeId = theTransporters[i]->getRemoteNodeId();
+ ndbout << "Transporter: " << remoteNodeId
+ << " PerformState: " << performStates[remoteNodeId]
+ << " IOState: " << ioStates[remoteNodeId] << endl;
+ }
+}
+#endif
+
+PerformState
+TransporterRegistry::performState(NodeId nodeId) {
+ return performStates[nodeId];
+}
+
+#ifdef DEBUG_TRANSPORTER
+const char *
+performStateString(PerformState state){
+ switch(state){
+ case PerformNothing:
+ return "PerformNothing";
+ break;
+ case PerformIO:
+ return "PerformIO";
+ break;
+ case PerformConnect:
+ return "PerformConnect";
+ break;
+ case PerformDisconnect:
+ return "PerformDisconnect";
+ break;
+ case RemoveTransporter:
+ return "RemoveTransporter";
+ break;
+ }
+ return "Unknown";
+}
+#endif
+
+void
+TransporterRegistry::setPerformState(NodeId nodeId, PerformState state) {
+ DEBUG("TransporterRegistry::setPerformState("
+ << nodeId << ", " << performStateString(state) << ")");
+
+ performStates[nodeId] = state;
+}
+
+void
+TransporterRegistry::setPerformState(PerformState state) {
+ int count = 0;
+ int index = 0;
+ while(count < nTransporters){
+ if(theTransporters[index] != 0){
+ setPerformState(theTransporters[index]->getRemoteNodeId(), state);
+ count ++;
+ }
+ index ++;
+ }
+}
+
+IOState
+TransporterRegistry::ioState(NodeId nodeId) {
+ return ioStates[nodeId];
+}
+
+void
+TransporterRegistry::setIOState(NodeId nodeId, IOState state) {
+ DEBUG("TransporterRegistry::setIOState("
+ << nodeId << ", " << state << ")");
+ ioStates[nodeId] = state;
+}
+
+void
+TransporterRegistry::startReceiving(){
+#ifdef NDB_OSE_TRANSPORTER
+ if(theOSEReceiver != NULL){
+ theOSEReceiver->createPhantom();
+ }
+#endif
+
+#ifdef NDB_OSE
+ theOSEJunkSocketRecv = socket(AF_INET, SOCK_STREAM, 0);
+#endif
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ theReceiverPid = current_process();
+ for(int i = 0; i<nTCPTransporters; i++)
+ theTCPTransporters[i]->theReceiverPid = theReceiverPid;
+#endif
+}
+
+void
+TransporterRegistry::stopReceiving(){
+#ifdef NDB_OSE_TRANSPORTER
+ if(theOSEReceiver != NULL){
+ theOSEReceiver->destroyPhantom();
+ }
+#endif
+
+ /**
+ * Disconnect all transporters, this includes detach from remote node
+ * and since that must be done from the same process that called attach
+ * it's done here in the receive thread
+ */
+ disconnectAll();
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ if(theOSEJunkSocketRecv > 0)
+ close(theOSEJunkSocketRecv);
+ theOSEJunkSocketRecv = -1;
+#endif
+
+}
+
+void
+TransporterRegistry::startSending(){
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ theOSEJunkSocketSend = socket(AF_INET, SOCK_STREAM, 0);
+#endif
+}
+
+void
+TransporterRegistry::stopSending(){
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ if(theOSEJunkSocketSend > 0)
+ close(theOSEJunkSocketSend);
+ theOSEJunkSocketSend = -1;
+#endif
+}
+
+/**
+ * The old implementation did not scale with a large
+ * number of nodes. (Watchdog killed NDB because
+ * it took too long time to allocated threads in
+ * doConnect.
+ *
+ * The new implementation only checks the connection
+ * for a number of transporters (STEPPING), until to
+ * the point where all transporters has executed
+ * doConnect once. After that, the behaviour is as
+ * in the old implemenation, i.e, checking the connection
+ * for all transporters.
+ * @todo: instead of STEPPING, maybe we should only
+ * allow checkConnections to execute for a certain
+ * time that somehow factors in heartbeat times and
+ * watchdog times.
+ *
+ */
+
+void
+TransporterRegistry::checkConnections(){
+ if(m_ccStep > nTransporters)
+ m_ccStep = nTransporters;
+
+ while(m_ccCount < m_ccStep){
+ if(theTransporters[m_ccIndex] != 0){
+ Transporter * t = theTransporters[m_ccIndex];
+ const NodeId nodeId = t->getRemoteNodeId();
+ if(t->getThreadError() != 0) {
+ reportError(callbackObj, nodeId, t->getThreadError());
+ t->resetThreadError();
+ }
+
+ switch(performStates[nodeId]){
+ case PerformConnect:
+ if(!t->isConnected()){
+ t->doConnect();
+ if(m_nTransportersPerformConnect!=nTransporters)
+ m_nTransportersPerformConnect++;
+
+ } else {
+ performStates[nodeId] = PerformIO;
+ reportConnect(callbackObj, nodeId);
+ }
+ break;
+ case PerformDisconnect:
+ {
+ bool wasConnected = t->isConnected();
+ t->doDisconnect();
+ performStates[nodeId] = PerformNothing;
+ if(wasConnected){
+ reportDisconnect(callbackObj, nodeId,0);
+ }
+ }
+ break;
+ case RemoveTransporter:
+ removeTransporter(nodeId);
+ break;
+ case PerformNothing:
+ case PerformIO:
+ break;
+ }
+ m_ccCount ++;
+ }
+ m_ccIndex ++;
+ }
+
+ if(!m_ccReady) {
+ if(m_ccCount < nTransporters) {
+ if(nTransporters - m_ccStep < STEPPING)
+ m_ccStep += nTransporters-m_ccStep;
+ else
+ m_ccStep += STEPPING;
+
+ // ndbout_c("count %d step %d ", m_ccCount, m_ccStep);
+ }
+ else {
+ m_ccCount = 0;
+ m_ccIndex = 0;
+ m_ccStep = STEPPING;
+ // ndbout_c("count %d step %d ", m_ccCount, m_ccStep);
+ }
+ }
+ if((nTransporters == m_nTransportersPerformConnect) || m_ccReady) {
+ m_ccReady = true;
+ m_ccCount = 0;
+ m_ccIndex = 0;
+ m_ccStep = nTransporters;
+ // ndbout_c("alla count %d step %d ", m_ccCount, m_ccStep);
+ }
+
+}//TransporterRegistry::checkConnections()
+
+NdbOut & operator <<(NdbOut & out, SignalHeader & sh){
+ out << "-- Signal Header --" << endl;
+ out << "theLength: " << sh.theLength << endl;
+ out << "gsn: " << sh.theVerId_signalNumber << endl;
+ out << "recBlockNo: " << sh.theReceiversBlockNumber << endl;
+ out << "sendBlockRef: " << sh.theSendersBlockRef << endl;
+ out << "sendersSig: " << sh.theSendersSignalId << endl;
+ out << "theSignalId: " << sh.theSignalId << endl;
+ out << "trace: " << (int)sh.theTrace << endl;
+ return out;
+}
diff --git a/ndb/src/common/transporter/basictest/Makefile b/ndb/src/common/transporter/basictest/Makefile
new file mode 100644
index 00000000000..d86af360408
--- /dev/null
+++ b/ndb/src/common/transporter/basictest/Makefile
@@ -0,0 +1,15 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := basicTransporterTest
+BIN_TARGET_ARCHIVES := transporter portlib general
+
+SOURCES = basicTransporterTest.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
diff --git a/ndb/src/common/transporter/basictest/basicTransporterTest.cpp b/ndb/src/common/transporter/basictest/basicTransporterTest.cpp
new file mode 100644
index 00000000000..5d8186badb8
--- /dev/null
+++ b/ndb/src/common/transporter/basictest/basicTransporterTest.cpp
@@ -0,0 +1,536 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "TransporterRegistry.hpp"
+#include "TransporterDefinitions.hpp"
+#include "TransporterCallback.hpp"
+#include <RefConvert.hpp>
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbString.h>
+
+int basePortTCP = 17000;
+
+SCI_TransporterConfiguration sciTemplate = {
+ 8000,
+ // Packet size
+ 2500000, // Buffer size
+ 2, // number of adapters
+ 1, // remote node id SCI
+ 2, // Remote node Id SCI
+ 0, // local ndb node id (server)
+ 0, // remote ndb node id (client)
+ 0, // byteOrder;
+ false, // compression;
+ true, // checksum;
+ true // signalId;
+};
+
+TCP_TransporterConfiguration tcpTemplate = {
+ 17000, // port;
+ "", // remoteHostName;
+ "", // localhostname
+ 2, // remoteNodeId;
+ 1, // localNodeId;
+ 10000, // sendBufferSize - Size of SendBuffer of priority B
+ 10000, // maxReceiveSize - Maximum no of bytes to receive
+ 0, // byteOrder;
+ false, // compression;
+ true, // checksum;
+ true // signalId;
+};
+
+OSE_TransporterConfiguration oseTemplate = {
+ "", // remoteHostName;
+ "", // localHostName;
+ 0, // remoteNodeId;
+ 0, // localNodeId;
+ false, // compression;
+ true, // checksum;
+ true, // signalId;
+ 0, // byteOrder;
+
+ 2000, // prioASignalSize;
+ 1000, // prioBSignalSize;
+ 10
+};
+
+SHM_TransporterConfiguration shmTemplate = {
+ 0, //remoteNodeId
+ 0, //localNodeId;
+ false, //compression
+ true, //checksum;
+ true, //signalId;
+ 0, //byteOrder;
+ 123, //shmKey;
+ 2500000 //shmSize;
+};
+
+TransporterRegistry *tReg = 0;
+
+#ifndef OSE_DELTA
+#include <signal.h>
+#endif
+
+extern "C"
+void
+signalHandler(int signo){
+#ifndef OSE_DELTA
+ ::signal(13, signalHandler);
+#endif
+ char buf[255];
+ sprintf(buf,"Signal: %d\n", signo);
+ ndbout << buf << endl;
+}
+
+void
+usage(const char * progName){
+ ndbout << "Usage: " << progName << " <type> localNodeId localHostName"
+ << " remoteHostName1 remoteHostName2" << endl;
+ ndbout << " type = shm tcp ose sci" << endl;
+ ndbout << " localNodeId - 1 to 3" << endl;
+}
+
+typedef void (* CreateTransporterFunc)(void * conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName);
+
+void createOSETransporter(void *, NodeId, NodeId, const char *, const char *);
+void createSCITransporter(void *, NodeId, NodeId, const char *, const char *);
+void createTCPTransporter(void *, NodeId, NodeId, const char *, const char *);
+void createSHMTransporter(void *, NodeId, NodeId, const char *, const char *);
+
+int signalReceived[4];
+
+int
+main(int argc, const char **argv){
+
+ signalHandler(0);
+
+ for(int i = 0; i<4; i++)
+ signalReceived[i] = 0;
+
+ if(argc < 5){
+ usage(argv[0]);
+ return 0;
+ }
+
+ Uint32 noOfConnections = 0;
+ const char * progName = argv[0];
+ const char * type = argv[1];
+ const NodeId localNodeId = atoi(argv[2]);
+ const char * localHostName = argv[3];
+ const char * remoteHost1 = argv[4];
+ const char * remoteHost2 = NULL;
+
+ if(argc == 5)
+ noOfConnections = 1;
+ else {
+ noOfConnections = 2;
+ remoteHost2 = argv[5];
+ }
+
+ if(localNodeId < 1 || localNodeId > 3){
+ ndbout << "localNodeId = " << localNodeId << endl << endl;
+ usage(progName);
+ return 0;
+ }
+
+ ndbout << "-----------------" << endl;
+ ndbout << "localNodeId: " << localNodeId << endl;
+ ndbout << "localHostName: " << localHostName << endl;
+ ndbout << "remoteHost1 (node " << (localNodeId == 1?2:1) << "): "
+ << remoteHost1 << endl;
+ if(noOfConnections == 2){
+ ndbout << "remoteHost2 (node " << (localNodeId == 3?2:3) << "): "
+ << remoteHost2 << endl;
+ }
+ ndbout << "-----------------" << endl;
+
+ void * confTemplate = 0;
+ CreateTransporterFunc func = 0;
+
+ if(strcasecmp(type, "tcp") == 0){
+ func = createTCPTransporter;
+ confTemplate = &tcpTemplate;
+ } else if(strcasecmp(type, "ose") == 0){
+ func = createOSETransporter;
+ confTemplate = &oseTemplate;
+ } else if(strcasecmp(type, "sci") == 0){
+ func = createSCITransporter;
+ confTemplate = &sciTemplate;
+ } else if(strcasecmp(type, "shm") == 0){
+ func = createSHMTransporter;
+ confTemplate = &shmTemplate;
+ } else {
+ ndbout << "Unsupported transporter type" << endl;
+ return 0;
+ }
+
+ ndbout << "Creating transporter registry" << endl;
+ tReg = new TransporterRegistry;
+ tReg->init(localNodeId);
+
+ switch(localNodeId){
+ case 1:
+ (* func)(confTemplate, 1, 2, localHostName, remoteHost1);
+ if(noOfConnections == 2)
+ (* func)(confTemplate, 1, 3, localHostName, remoteHost2);
+ break;
+ case 2:
+ (* func)(confTemplate, 2, 1, localHostName, remoteHost1);
+ if(noOfConnections == 2)
+ (* func)(confTemplate, 2, 3, localHostName, remoteHost2);
+ break;
+ case 3:
+ (* func)(confTemplate, 3, 1, localHostName, remoteHost1);
+ if(noOfConnections == 2)
+ (* func)(confTemplate, 3, 2, localHostName, remoteHost2);
+ break;
+ }
+
+ ndbout << "Doing startSending/startReceiving" << endl;
+ tReg->startSending();
+ tReg->startReceiving();
+
+ ndbout << "Connecting" << endl;
+ tReg->setPerformState(PerformConnect);
+ tReg->checkConnections();
+
+ unsigned sum = 0;
+ do {
+ sum = 0;
+ for(int i = 0; i<4; i++)
+ sum += signalReceived[i];
+
+ tReg->checkConnections();
+
+ tReg->external_IO(500);
+ NdbSleep_MilliSleep(500);
+
+ ndbout << "In main loop" << endl;
+ } while(sum != 2*noOfConnections);
+
+ ndbout << "Doing setPerformState(Disconnect)" << endl;
+ tReg->setPerformState(PerformDisconnect);
+
+ ndbout << "Doing checkConnections()" << endl;
+ tReg->checkConnections();
+
+ ndbout << "Sleeping 3 secs" << endl;
+ NdbSleep_SecSleep(3);
+
+ ndbout << "Deleting transporter registry" << endl;
+ delete tReg; tReg = 0;
+
+ return 0;
+}
+
+void
+checkData(SignalHeader * const header, Uint8 prio, Uint32 * const theData,
+ LinearSectionPtr ptr[3]){
+ Uint32 expectedLength = 0;
+ if(prio == 0)
+ expectedLength = 17;
+ else
+ expectedLength = 19;
+
+ if(header->theLength != expectedLength){
+ ndbout << "Unexpected signal length: " << header->theLength
+ << " expected: " << expectedLength << endl;
+ abort();
+ }
+
+ if(header->theVerId_signalNumber != expectedLength + 1)
+ abort();
+
+ if(header->theReceiversBlockNumber != expectedLength + 2)
+ abort();
+
+ if(refToBlock(header->theSendersBlockRef) != expectedLength + 3)
+ abort();
+
+ if(header->theSendersSignalId != expectedLength + 5)
+ abort();
+
+ if(header->theTrace != expectedLength + 6)
+ abort();
+
+ if(header->m_noOfSections != (prio == 0 ? 0 : 1))
+ abort();
+
+ if(header->m_fragmentInfo != (prio + 1))
+ abort();
+
+ Uint32 dataWordStart = header->theLength ;
+ for(unsigned i = 0; i<header->theLength; i++){
+ if(theData[i] != i){ //dataWordStart){
+ ndbout << "data corrupt!\n" << endl;
+ abort();
+ }
+ dataWordStart ^= (~i*i);
+ }
+
+ if(prio != 0){
+ ndbout_c("Found section");
+ if(ptr[0].sz != header->theLength)
+ abort();
+
+ if(memcmp(ptr[0].p, theData, (ptr[0].sz * 4)) != 0)
+ abort();
+ }
+}
+
+void
+sendSignalTo(NodeId nodeId, int prio){
+ SignalHeader sh;
+ sh.theLength = (prio == 0 ? 17 : 19);
+ sh.theVerId_signalNumber = sh.theLength + 1;
+ sh.theReceiversBlockNumber = sh.theLength + 2;
+ sh.theSendersBlockRef = sh.theLength + 3;
+ sh.theSendersSignalId = sh.theLength + 4;
+ sh.theSignalId = sh.theLength + 5;
+ sh.theTrace = sh.theLength + 6;
+ sh.m_noOfSections = (prio == 0 ? 0 : 1);
+ sh.m_fragmentInfo = prio + 1;
+
+ Uint32 theData[25];
+
+ Uint32 dataWordStart = sh.theLength;
+ for(unsigned i = 0; i<sh.theLength; i++){
+ theData[i] = i;
+ dataWordStart ^= (~i*i);
+ }
+ ndbout << "Sending prio " << (int)prio << " signal to node: "
+ << nodeId
+ << " gsn = " << sh.theVerId_signalNumber << endl;
+
+ LinearSectionPtr ptr[3];
+ ptr[0].p = &theData[0];
+ ptr[0].sz = sh.theLength;
+
+ SendStatus s = tReg->prepareSend(&sh, prio, theData, nodeId, ptr);
+ if(s != SEND_OK){
+ ndbout << "Send was not ok. Send was: " << s << endl;
+ }
+}
+
+void
+execute(void* callbackObj,
+ SignalHeader * const header, Uint8 prio, Uint32 * const theData,
+ LinearSectionPtr ptr[3]){
+ const NodeId nodeId = refToNode(header->theSendersBlockRef);
+
+ ndbout << "Recieved prio " << (int)prio << " signal from node: "
+ << nodeId
+ << " gsn = " << header->theVerId_signalNumber << endl;
+ checkData(header, prio, theData, ptr);
+ ndbout << " Data is ok!\n" << endl;
+
+ signalReceived[nodeId]++;
+
+ if(prio == 0)
+ sendSignalTo(nodeId, 1);
+ else
+ tReg->setPerformState(nodeId, PerformDisconnect);
+}
+
+void
+copy(Uint32 * & insertPtr,
+ class SectionSegmentPool & thePool, const SegmentedSectionPtr & _ptr){
+ abort();
+}
+
+void
+reportError(void* callbackObj, NodeId nodeId, TransporterError errorCode){
+ char buf[255];
+ sprintf(buf, "reportError (%d, %x)", nodeId, errorCode);
+ ndbout << buf << endl;
+ if(errorCode & 0x8000){
+ tReg->setPerformState(nodeId, PerformDisconnect);
+ abort();
+ }
+}
+
+/**
+ * Report average send theLength in bytes (4096 last sends)
+ */
+void
+reportSendLen(void* callbackObj, NodeId nodeId, Uint32 count, Uint64 bytes){
+ char buf[255];
+ sprintf(buf, "reportSendLen(%d, %d)", nodeId, (Uint32)(bytes/count));
+ ndbout << buf << endl;
+}
+
+/**
+ * Report average receive theLength in bytes (4096 last receives)
+ */
+void
+reportReceiveLen(void* callbackObj, NodeId nodeId, Uint32 count, Uint64 bytes){
+ char buf[255];
+ sprintf(buf, "reportReceiveLen(%d, %d)", nodeId, (Uint32)(bytes/count));
+ ndbout << buf << endl;
+}
+
+/**
+ * Report connection established
+ */
+void
+reportConnect(void* callbackObj, NodeId nodeId){
+ char buf[255];
+ sprintf(buf, "reportConnect(%d)", nodeId);
+ ndbout << buf << endl;
+ tReg->setPerformState(nodeId, PerformIO);
+
+ sendSignalTo(nodeId, 0);
+}
+
+/**
+ * Report connection broken
+ */
+void
+reportDisconnect(void* callbackObj, NodeId nodeId, Uint32 errNo){
+ char buf[255];
+ sprintf(buf, "reportDisconnect(%d)", nodeId);
+ ndbout << buf << endl;
+ if(signalReceived[nodeId] < 2)
+ tReg->setPerformState(nodeId, PerformConnect);
+}
+
+int
+checkJobBuffer() {
+ /**
+ * Check to see if jobbbuffers are starting to get full
+ * and if so call doJob
+ */
+ return 0;
+}
+
+void
+createOSETransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName){
+ ndbout << "Creating OSE transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+ OSE_TransporterConfiguration * conf = (OSE_TransporterConfiguration*)_conf;
+
+ conf->localNodeId = localNodeId;
+ conf->localHostName = localHostName;
+ conf->remoteNodeId = remoteNodeId;
+ conf->remoteHostName = remoteHostName;
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+void
+createTCPTransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName){
+ ndbout << "Creating TCP transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+ TCP_TransporterConfiguration * conf = (TCP_TransporterConfiguration*)_conf;
+
+ int port;
+ if(localNodeId == 1 && remoteNodeId == 2) port = basePortTCP + 0;
+ if(localNodeId == 1 && remoteNodeId == 3) port = basePortTCP + 1;
+ if(localNodeId == 2 && remoteNodeId == 1) port = basePortTCP + 0;
+ if(localNodeId == 2 && remoteNodeId == 3) port = basePortTCP + 2;
+ if(localNodeId == 3 && remoteNodeId == 1) port = basePortTCP + 1;
+ if(localNodeId == 3 && remoteNodeId == 2) port = basePortTCP + 2;
+
+ conf->localNodeId = localNodeId;
+ conf->localHostName = localHostName;
+ conf->remoteNodeId = remoteNodeId;
+ conf->remoteHostName = remoteHostName;
+ conf->port = port;
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+void
+createSCITransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName){
+
+
+ ndbout << "Creating SCI transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+
+ SCI_TransporterConfiguration * conf = (SCI_TransporterConfiguration*)_conf;
+
+ conf->remoteSciNodeId0= (Uint16)atoi(localHostName);
+ conf->remoteSciNodeId1= (Uint16)atoi(remoteHostName);
+
+
+ conf->localNodeId = localNodeId;
+ conf->remoteNodeId = remoteNodeId;
+
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+void
+createSHMTransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName){
+
+
+ ndbout << "Creating SHM transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+
+ SHM_TransporterConfiguration * conf = (SHM_TransporterConfiguration*)_conf;
+
+ conf->localNodeId = localNodeId;
+ conf->remoteNodeId = remoteNodeId;
+
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
diff --git a/ndb/src/common/transporter/buddy.cpp b/ndb/src/common/transporter/buddy.cpp
new file mode 100644
index 00000000000..c65aad1df2c
--- /dev/null
+++ b/ndb/src/common/transporter/buddy.cpp
@@ -0,0 +1,328 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "buddy.hpp"
+#include <stdlib.h>
+#include <math.h>
+
+
+void Chunk256::setFree(bool free){
+ // Bit 0 of allocationTimeStamp represents if the segment is free or not
+ Uint32 offMask = 0x0; // A mask to set the 0 bit to 0
+ allocationTimeStamp = 0x0;
+ if(free)
+ // Set this bit to 0, if segment should be free
+ allocationTimeStamp = allocationTimeStamp & offMask;
+}
+
+bool Chunk256::getFree(){
+ Uint32 offMask = 0x0;
+ return ((allocationTimeStamp | offMask) == offMask ? true : false);
+}
+
+void Chunk256::setAllocationTimeStamp(Uint32 cTime){
+ // Bits 1-31 of allocationTimeStamp represent the allocation time for segment
+
+ // printf("\nSet allocation time. Current time %d", cTime);
+ Uint32 onMask = 0x80000000; // A mask to set the 0 bit to 1
+ allocationTimeStamp = 0x0;
+ allocationTimeStamp = onMask | cTime;
+}
+
+Uint32 Chunk256::getAllocationTimeStamp(){
+ Uint32 onMask = 0x80000000;
+ allocationTimeStamp = allocationTimeStamp ^ onMask;
+ printf("\nGet allocation time. Time is %d", allocationTimeStamp);
+ return allocationTimeStamp;
+};
+
+bool BuddyMemory::allocate(int nChunksToAllocate) {
+
+ // Allocate the memory block needed. This memory is deallocated in the
+ // destructor of TransporterRegistry.
+
+ printf("\nAllocating %d chunks...", nChunksToAllocate);
+
+ startOfMemoryBlock = (Uint32*) malloc(256 * nChunksToAllocate);
+
+ if (startOfMemoryBlock == NULL)
+ return false;
+
+ // Allocate the array of 256-byte chunks
+ chunk = new Chunk256[nChunksToAllocate];
+
+ // Initialize the chunk-array. Every 8 kB segment consists of 32 chunks.
+ // Set all chunks to free and set the prev and next pointer
+ for (int i=0; i < nChunksToAllocate; i++) {
+ chunk[i].setFree(true);
+ if (i%32 == 0) {
+ // The first chunk in every segment will point to the prev and next segment
+ chunk[i].prevSegmentOfSameSize = i-32;
+ chunk[i].nextSegmentOfSameSize = i + 32;
+ chunk[0].prevSegmentOfSameSize = END_OF_CHUNK_LIST;
+ chunk[totalNoOfChunks-32].nextSegmentOfSameSize = END_OF_CHUNK_LIST;
+ } else {
+ // The rest of the chunks in the segments have undefined prev and next pointers
+ chunk[i].prevSegmentOfSameSize = UNDEFINED_CHUNK;
+ chunk[i].nextSegmentOfSameSize = UNDEFINED_CHUNK;
+ }
+ }
+
+ // Initialize the freeSegment-pointers
+ for (int i=0; i<sz_MAX; i++)
+ freeSegment[i] = UNDEFINED_CHUNK;
+
+ // There are only 8 kB segments at startup
+ freeSegment[sz_8192] = 0;
+
+ for (int i=0; i<sz_MAX; i++)
+ printf("\nPointers: %d", freeSegment[i]);
+
+ return true;
+}
+
+
+bool BuddyMemory::getSegment(Uint32 size, Segment * dst) {
+
+ // The no of chunks the user asked for
+ Uint32 nChunksAskedFor = ceil((double(size)/double(256)));
+ int segm;
+
+ printf("\n%d chunks asked for", nChunksAskedFor);
+
+ // It may be that the closest segment size above
+ // nChunksAskedFor*256 is not a size that is available in
+ // the freeSegment-list, i.e. it may not be of FreeSegmentSize.
+ int nChunksToAllocate = nChunksAskedFor;
+
+ // Find the FreeSegmentSize closest above nChunksAskedFor
+ if ((nChunksToAllocate != 1) && (nChunksToAllocate % 2 != 0))
+ nChunksToAllocate++;
+
+ printf("\n%d chunks to allocate", nChunksToAllocate);
+ int segmSize = logTwoPlus(nChunksToAllocate) - 1;
+ if (size-pow(2,segmSize) > 256)
+ segmSize ++;
+ printf("\nSegment size: %f", pow(2,int(8+segmSize)));
+
+ while ((segmSize <= sz_GET_MAX) && (freeSegment[segmSize] == UNDEFINED_CHUNK))
+ segmSize++;
+
+ segm = freeSegment[segmSize];
+ if (segm != UNDEFINED_CHUNK){
+ // Free segment of asked size or larger is found
+
+ // Remove the found segment from the freeSegment-list
+ removeFromFreeSegmentList(segmSize, segm);
+
+ // Set all chunks to allocated (not free) and set the allocation time
+ // for the segment we are about to allocate
+ for (int i = segm; i <= segm+nChunksToAllocate; i++) {
+ chunk[i].setFree(false);
+ chunk[i].setAllocationTimeStamp(currentTime);
+ }
+
+ // Before returning the segment, check if it is larger than the segment asked for
+ if (nChunksAskedFor < nChunksToAllocate)
+ release(nChunksAskedFor, nChunksToAllocate - nChunksAskedFor - 1);
+
+ Segment segment;
+ segment.segmentAddress = startOfMemoryBlock+(segm * 256);
+ segment.segmentSize = 256 * nChunksAskedFor;
+ segment.releaseId = segm;
+
+ printf("\nSegment: segment address = %d, segment size = %d, release Id = %d",
+ segment.segmentAddress, segment.segmentSize, segment.releaseId);
+
+ return true;
+ }
+ printf("\nNo segments of asked size or larger are found");
+ return false;
+}
+
+void BuddyMemory::removeFromFreeSegmentList(int sz, int index) {
+ // Remove the segment from the freeSegment list
+
+ printf("\nRemoving segment from list...");
+ if (index != UNDEFINED_CHUNK) {
+ Chunk256 prevChunk;
+ Chunk256 nextChunk;
+ int prevChunkIndex = chunk[index].prevSegmentOfSameSize;
+ int nextChunkIndex = chunk[index].nextSegmentOfSameSize;
+
+ if (prevChunkIndex == END_OF_CHUNK_LIST) {
+ if (nextChunkIndex == END_OF_CHUNK_LIST)
+ // We are about to remove the only element in the list
+ freeSegment[sz] = UNDEFINED_CHUNK;
+ else {
+ // We are about to remove the first element in the list
+ nextChunk = chunk[nextChunkIndex];
+ nextChunk.prevSegmentOfSameSize = END_OF_CHUNK_LIST;
+ freeSegment[sz] = nextChunkIndex;
+ }
+ } else {
+ if (nextChunkIndex == END_OF_CHUNK_LIST) {
+ // We are about to remove the last element in the list
+ prevChunk = chunk[prevChunkIndex];
+ prevChunk.nextSegmentOfSameSize = END_OF_CHUNK_LIST;
+ } else {
+ // We are about to remove an element in the middle of the list
+ prevChunk = chunk[prevChunkIndex];
+ nextChunk = chunk[nextChunkIndex];
+ prevChunk.nextSegmentOfSameSize = nextChunkIndex;
+ nextChunk.prevSegmentOfSameSize = prevChunkIndex;
+ }
+ }
+ }
+ for (int i=0; i<sz_MAX; i++)
+ printf("\nPointers: %d", freeSegment[i]);
+}
+
+void BuddyMemory::release(int releaseId, int size) {
+
+ int nChunksToRelease = (size == 0 ? 1 : ceil(double(size)/double(256)));
+ //nChunksToRelease = ceil(double(size)/double(256));
+ int startChunk = releaseId;
+ int endChunk = releaseId + nChunksToRelease - 1;
+
+ printf("\n%d chunks to release (initially)", nChunksToRelease);
+
+ // Set the chunks we are about to release to free
+ for (int i = startChunk; i <= endChunk; i++){
+ chunk[i].setFree(true);
+ }
+
+ // Look at the chunks before the segment we are about to release
+ for (int i = releaseId-1; i >= 0; i--) {
+ if (!chunk[i].getFree())
+ break;
+ else {
+ startChunk = i;
+ nChunksToRelease++;
+ // Look at the next-pointer. If it is valid, we have a
+ // chunk that is the start of a free segment. Remove it
+ // from the freeSegment-list.
+ if (chunk[i].nextSegmentOfSameSize != UNDEFINED_CHUNK)
+ removeFromFreeSegmentList(size, i);
+ }
+ }
+
+ // Look at the chunks after the segment we are about to release
+ for (int i = endChunk+1; i <= totalNoOfChunks; i++) {
+ if (!chunk[i].getFree())
+ break;
+ else {
+ endChunk = i;
+ nChunksToRelease++;
+ // Look at the next-pointer. If it is valid, we have a
+ // chunk that is the start of a free segment. Remove it
+ // from the free segment list
+ if (chunk[i].nextSegmentOfSameSize != UNDEFINED_CHUNK)
+ removeFromFreeSegmentList(size, i);
+ }
+ }
+
+ // We have the start and end indexes and total no of free chunks.
+ // Separate the chunks into segments that can be added to the
+ // freeSegments-list.
+ int restChunk = 0;
+ int segmSize;
+
+ printf("\n%d chunks to release (finally)", nChunksToRelease);
+
+ segmSize = logTwoPlus(nChunksToRelease) - 1;
+ if (segmSize > sz_MAX) {
+ segmSize = sz_MAX;
+ }
+
+ nChunksToRelease = pow(2,segmSize);
+ addToFreeSegmentList(nChunksToRelease*256, startChunk);
+}
+
+void BuddyMemory::addToFreeSegmentList(int sz, int index) {
+ // Add a segment to the freeSegment list
+
+ printf("\nAsked to add segment of size %d", sz);
+
+ // Get an index in freeSegment list corresponding to sz size
+ int segmSize = logTwoPlus(sz) - 1;
+ if (sz - pow(2,segmSize) >= 256)
+ segmSize ++;
+ sz = segmSize - 8;
+
+ int nextSegm = freeSegment[sz];
+
+ printf("\nAdding a segment of size %f", pow(2,(8 + sz)));
+
+ freeSegment[sz] = index;
+ if (nextSegm == UNDEFINED_CHUNK) {
+ // We are about to add a segment to an empty list
+ chunk[index].prevSegmentOfSameSize = END_OF_CHUNK_LIST;
+ chunk[index].nextSegmentOfSameSize = END_OF_CHUNK_LIST;
+ }
+ else {
+ // Add the segment first in the list
+ chunk[index].prevSegmentOfSameSize = END_OF_CHUNK_LIST;
+ chunk[index].nextSegmentOfSameSize = nextSegm;
+ chunk[nextSegm].prevSegmentOfSameSize = index;
+ }
+
+ for (int i=0; i<sz_MAX; i++)
+ printf("\nPointers: %d", freeSegment[i]);
+
+}
+
+Uint32 BuddyMemory::logTwoPlus(Uint32 arg) {
+ // Calculate log2(arg) + 1
+
+ Uint32 resValue;
+
+ arg = arg | (arg >> 8);
+ arg = arg | (arg >> 4);
+ arg = arg | (arg >> 2);
+ arg = arg | (arg >> 1);
+ resValue = (arg & 0x5555) + ((arg >> 1) & 0x5555);
+ resValue = (resValue & 0x3333) + ((resValue >> 2) & 0x3333);
+ resValue = resValue + (resValue >> 4);
+ resValue = (resValue & 0xf) + ((resValue >> 8) & 0xf);
+
+ return resValue;
+}
+
+bool BuddyMemory::memoryAvailable() {
+ // Return true if there is at least 8 kB memory available
+ for (int i = sz_8192; i < sz_MAX; i++)
+ if (freeSegment[i] != UNDEFINED_CHUNK)
+ return true;
+ return false;
+}
+
+
+void BuddyMemory::refreshTime(Uint32 time) {
+ if (time - currentTime > 1000) {
+ // Update current time
+ currentTime = time;
+ // Go through the chunk-list every second and release
+ // any chunks that have been allocated for too long
+ for (int i=0; i<totalNoOfChunks; i++) {
+ if ((!chunk[i].getFree()) &&
+ (currentTime-chunk[i].getAllocationTimeStamp() > ALLOCATION_TIMEOUT)) {
+ release(i, 256);
+ printf("\nChunks hve been allocated for too long");
+ }
+ }
+ }
+}
diff --git a/ndb/src/common/transporter/buddy.hpp b/ndb/src/common/transporter/buddy.hpp
new file mode 100644
index 00000000000..7272ac884ec
--- /dev/null
+++ b/ndb/src/common/transporter/buddy.hpp
@@ -0,0 +1,173 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BUDDY_H
+#define BUDDY_H
+
+#include <stdlib.h>
+#include <stdio.h>
+
+typedef unsigned int Uint32;
+typedef unsigned short Uint16;
+typedef unsigned long long Uint64;
+
+//
+const int UNDEFINED_CHUNK = -2; // XXX Set to hex
+
+//
+const int END_OF_CHUNK_LIST = -1; // XXX Set to hex
+
+// A timeout (no of seconds) for the memory segments in the TransporterRegistry
+// memory pool. If a segment has been occupied (free=false) for a longer period
+// than this timeout, it will be released.
+const int ALLOCATION_TIMEOUT = 10000;
+
+// Free segments should always be as large as possible
+// and are only allowed to be in any of these sizes
+enum FreeSegmentSize {
+ sz_256 = 0,
+ sz_512 = 1,
+ sz_1024 = 2,
+ sz_2048 = 3,
+ sz_4096 = 4,
+ sz_8192 = 5,
+ sz_16384 = 6,
+ sz_32768 = 7,
+ sz_65536 = 8,
+ sz_131072 = 9,
+ sz_GET_MAX = 5,
+ sz_MAX = 9
+};
+
+struct Segment;
+
+class BuddyMemory {
+public:
+
+ // Return true if there is at least 8 kB memory available
+ bool memoryAvailable();
+
+ //
+ bool allocate(int nChunksToAllocate);
+
+ // Remove the segment from the freeSegment list
+ void removeFromFreeSegmentList(int sz, int index);
+
+ // Release the segment of size
+ void release(int releaseId, int size);
+
+ // Add a segment to the freeSegment list
+ void addToFreeSegmentList(int sz, int index);
+
+ bool getSegment(Uint32 size, Segment * dst);
+
+ void refreshTime(Uint32 time);
+
+ //Calculate log2(arg) + 1
+ Uint32 logTwoPlus(Uint32 arg);
+
+ // The current time
+ Uint32 currentTime;
+
+ // Pointer to the first free segment of size FreeSegmentSize
+ Uint32 freeSegment[sz_MAX];
+
+ // Start address of the memory block allocated
+ Uint32* startOfMemoryBlock;
+
+ // Total number of 256 byte chunks.
+ Uint32 totalNoOfChunks;
+
+ // Array of 256-byte chunks
+ struct Chunk256* chunk;
+};
+
+struct Segment {
+ Uint32 segmentSize; // Size of the segment in no of words
+ Uint16 index; // Index in the array of SegmentListElements
+ Uint16 releaseId; // Unique no used when releasing the segment
+ // Undefined if Long_signal.deallocIndicator==0
+ union {
+ Uint32* segmentAddress; // Address to the memory segment
+ Uint64 _padding_NOT_TO_BE_USED_;
+ };
+};
+
+struct Chunk256 {
+ Uint32 allocationTimeStamp; // Bit 0 represents if the segment is free or not
+ // Bit 1-31 is the allocation time for the segment
+ // Bit 1-31 are undefined if the segment is free
+ Uint32 nextSegmentOfSameSize; // Undefined if allocated.
+ // The first chunk in a free segment has a valid
+ // next-pointer. In the rest of the chunks
+ // belonging to the segment it is UNDEFINED_CHUNK.
+ Uint32 prevSegmentOfSameSize; // Undefined if allocated
+ // The first chunk in a free segment has a valid
+ // prev-pointer. In the rest of the chunks
+ // belonging to the segment it is UNDEFINED_CHUNK.
+
+ void setFree(bool free);
+
+ bool getFree();
+
+ void setAllocationTimeStamp(Uint32 cTime);
+
+ Uint32 getAllocationTimeStamp();
+};
+
+// inline void Chunk256::setFree(bool free){
+// // Bit 0 of allocationTimeStamp represents if the segment is free or not
+// allocationTimeStamp = 0x0;
+
+// printf("\nSet free segment");
+// Uint32 offMask = 0x0; // A mask to set the 0 bit to 0
+// if(free)
+// // Set this bit to 0, if segment should be free
+// allocationTimeStamp = allocationTimeStamp & offMask;
+// }
+
+// inline bool Chunk256::getFree(){
+// // Get free segment
+
+// allocationTimeStamp = 0x0;
+// Uint32 offMask = 0x0;
+
+// printf("\nGet free segment");
+// return ((allocationTimeStamp | offMask) == offMask ? true : false);
+// }
+
+// inline void Chunk256::setAllocationTimeStamp(Uint32 cTime){
+// // Bits 1-31 of allocationTimeStamp represent the allocation time for segment
+
+// Uint32 onMask = 0x80000000; // A mask to set the 0 bit to 1
+// allocationTimeStamp = 0x0;
+
+// printf("\nSet allocation time");
+
+// allocationTimeStamp = onMask | cTime;
+// }
+
+// inline Uint32 Chunk256::getAllocationTimeStamp(){
+
+// Uint32 onMask = 0x80000000; // A mask to set the 0 bit to 1
+// allocationTimeStamp = 0x0;
+
+// printf("\nGet allocation time");
+// allocationTimeStamp = allocationTimeStamp ^ onMask;
+// return allocationTimeStamp;
+// };
+
+#endif
diff --git a/ndb/src/common/transporter/failoverSCI/Makefile b/ndb/src/common/transporter/failoverSCI/Makefile
new file mode 100644
index 00000000000..1e3d5f4a4b7
--- /dev/null
+++ b/ndb/src/common/transporter/failoverSCI/Makefile
@@ -0,0 +1,18 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := failoverSCI
+BIN_TARGET_LIBS := sisci
+BIN_TARGET_ARCHIVES := portlib
+
+CCFLAGS_LOC += -I..
+
+SOURCES = failoverSCI.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
diff --git a/ndb/src/common/transporter/failoverSCI/failoverSCI.cpp b/ndb/src/common/transporter/failoverSCI/failoverSCI.cpp
new file mode 100644
index 00000000000..03ce7ea6df3
--- /dev/null
+++ b/ndb/src/common/transporter/failoverSCI/failoverSCI.cpp
@@ -0,0 +1,866 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include "sisci_types.h"
+#include "sisci_api.h"
+#include "sisci_error.h"
+//#include "sisci_demolib.h"
+#include <strings.h>
+#include <ndb_types.h>
+#include <NdbTick.h>
+#include "NdbSleep.h"
+#define NO_CALLBACK NULL
+#define NO_FLAGS 0
+#define DATA_TRANSFER_READY 8
+
+sci_error_t error;
+sci_desc_t sdOne;
+sci_desc_t sdTwo;
+sci_local_segment_t localSegmentOne;
+sci_local_segment_t localSegmentTwo;
+sci_remote_segment_t remoteSegmentOne;
+sci_remote_segment_t remoteSegmentTwo;
+sci_map_t localMapOne;
+sci_map_t localMapTwo;
+sci_map_t remoteMapOne;
+sci_map_t remoteMapTwo;
+unsigned int localAdapterNo = 0;
+unsigned int standbyAdapterNo = 1;
+unsigned int localNodeId1;
+unsigned int localNodeId2;
+unsigned int remoteNodeId1 = 0;
+unsigned int remoteNodeId2 = 0;
+unsigned int localSegmentId;
+unsigned int remoteSegmentId1;
+unsigned int remoteSegmentId2;
+unsigned int segmentSize = 8192;
+unsigned int offset = 0;
+unsigned int client = 0;
+unsigned int server = 0;
+unsigned int *localbufferPtr;
+static int data;
+static int interruptConnected=0;
+
+/*********************************************************************************/
+/* U S A G E */
+/* */
+/*********************************************************************************/
+
+void Usage()
+{
+ printf("Usage of shmem\n");
+ printf("shmem -rn <remote node-id> -client/server [ -adapterno <adapter no> -size <segment size> ] \n\n");
+ printf(" -rn : Remote node-id\n");
+ printf(" -client : The local node is client\n");
+ printf(" -server : The local node is server\n");
+ printf(" -adapterno : Local adapter number (default %d)\n", localAdapterNo);
+ printf(" -size : Segment block size (default %d)\n", segmentSize);
+ printf(" -help : This helpscreen\n");
+
+ printf("\n");
+}
+
+
+/*********************************************************************************/
+/* P R I N T P A R A M E T E R S */
+/* */
+/*********************************************************************************/
+void PrintParameters(void)
+{
+
+ printf("Test parameters for %s \n",(client) ? "client" : "server" );
+ printf("----------------------------\n\n");
+ printf("Local node-id1 : %d\n",localNodeId1);
+ printf("Local node-id2 : %d\n",localNodeId2);
+ // printf("Remote node-id : %d\n",remoteNodeId);
+ printf("Local adapter no. : %d\n",localAdapterNo);
+ printf("Segment size : %d\n",segmentSize);
+ printf("----------------------------\n\n");
+
+}
+
+
+/*********************************************************************************/
+/* F I L L S E G M E N T W I T H D A T A */
+/* */
+/*********************************************************************************/
+
+sci_error_t GetLocalNodeId(Uint32 localAdapterNo, Uint32* localNodeId)
+{
+ sci_query_adapter_t queryAdapter;
+ sci_error_t error;
+ unsigned int _localNodeId;
+
+ queryAdapter.subcommand = SCI_Q_ADAPTER_NODEID;
+ queryAdapter.localAdapterNo = localAdapterNo;
+ queryAdapter.data = &_localNodeId;
+
+ SCIQuery(SCI_Q_ADAPTER,&queryAdapter,NO_FLAGS,&error);
+
+ *localNodeId=_localNodeId;
+
+ return error;
+}
+
+
+
+
+
+
+sci_error_t SendInterrupt(sci_desc_t sd,
+ Uint32 localAdapterNo,
+ Uint32 localSciNodeId,
+ Uint32 remoteSciNodeId,
+ Uint32 interruptNo){
+
+ sci_error_t error;
+ sci_remote_interrupt_t remoteInterrupt;
+ Uint32 timeOut = SCI_INFINITE_TIMEOUT;
+
+ // Now connect to the other sides interrupt flag
+ do {
+ SCIConnectInterrupt(sd, &remoteInterrupt, remoteSciNodeId, localAdapterNo,
+ interruptNo, timeOut, NO_FLAGS, &error);
+ } while (error != SCI_ERR_OK);
+
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr, "SCIConnectInterrupt failed - Error code 0x%x\n", error);
+ return error;
+ }
+
+ // Trigger interrupt
+ printf("\nNode %u sent interrupt (0x%x) to node %d\n",localSciNodeId, interruptNo, remoteSciNodeId);
+ SCITriggerInterrupt(remoteInterrupt, NO_FLAGS, &error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr, "SCITriggerInterrupt failed - Error code 0x%x\n", error);
+ return error;
+ }
+
+
+ // Disconnect and remove interrupts
+ SCIDisconnectInterrupt(remoteInterrupt, NO_FLAGS, &error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr, "SCIDisconnectInterrupt failed - Error code 0x%x\n", error);
+ return error;
+ }
+
+ return error;
+}
+
+
+sci_error_t ReceiveInterrupt(sci_desc_t sd,
+ Uint32 localAdapterNo,
+ Uint32 localSciNodeId,
+ Uint32 interruptNo,
+ Uint32 timeout) {
+
+ sci_error_t error;
+ sci_local_interrupt_t localInterrupt;
+ Uint32 timeOut = SCI_INFINITE_TIMEOUT;
+
+ // Create an interrupt
+ SCICreateInterrupt(sd, &localInterrupt, localAdapterNo,
+ &interruptNo, 0, NULL, SCI_FLAG_FIXED_INTNO, &error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr, "SCICreateInterrupt failed - Error code 0x%x\n", error);
+ return error;
+ }
+
+
+ // Wait for an interrupt
+ SCIWaitForInterrupt(localInterrupt, timeOut, NO_FLAGS, &error);
+
+ printf("\nNode %u received interrupt (0x%x)\n", localSciNodeId, interruptNo);
+
+ // Remove interrupt
+
+ SCIRemoveInterrupt(localInterrupt, NO_FLAGS, &error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr, "SCIRemoveInterrupt failed - Error code 0x%x\n", error);
+ return error;
+ }
+ return error;
+}
+
+
+sci_error_t FillSegmentWithData(unsigned int segmentSize, int reverse)
+{
+ unsigned int i;
+ unsigned int nostores;
+
+
+ nostores = (segmentSize) / sizeof(unsigned int);
+
+ /* Allocate buffer */
+
+ localbufferPtr = (unsigned int*)malloc( segmentSize );
+ if ( localbufferPtr == NULL ) {
+ /*
+ * Unable to create local buffer - Insufficient memory available
+ */
+ return SCI_ERR_NOSPC;
+ }
+ if(reverse) {
+ /* Fill in the data into a local buffer */
+ printf("Filling forward order \n");
+ for (i=0;i<nostores;i++) {
+ localbufferPtr[i] = i;
+ }
+ }
+ else {
+ int temp=nostores;
+ printf("Filling reverse order \n");
+ for (i=0;i<nostores;i++) {
+ localbufferPtr[i] = temp-- ;
+
+ }
+
+ }
+
+ return SCI_ERR_OK;
+}
+
+
+
+
+/*********************************************************************************/
+/* P R I N T C L I E N T D A T A */
+/* */
+/*********************************************************************************/
+
+void PrintClientData(void)
+{
+ unsigned int i;
+
+ printf("\nClient data: ");
+ /* Print the first 20 entries in the segment */
+ for (i=0;i<20;i++) {
+ printf("%d ",localbufferPtr[i]);
+ }
+
+ printf("\n");
+}
+
+
+/*********************************************************************************/
+/* P R I N T S E R V E R D A T A */
+/* */
+/*********************************************************************************/
+
+void PrintServerData(volatile unsigned int *localMapAddr)
+{
+
+ unsigned int *buffer;
+ int i;
+
+ // printf("\nServer data: ");
+ buffer = (unsigned int *)localMapAddr;
+
+ /* Print the first 20 entries in the segment */
+ for (i=0; i< 20; i++) {
+
+ printf("%d ",buffer[i]);
+ }
+ printf("\n");
+
+}
+
+
+
+/*********************************************************************************/
+/* T R A N S F E R D A T A */
+/* */
+/*********************************************************************************/
+
+unsigned int TransferData(sci_map_t remoteMap,
+ volatile unsigned int *remoteSegmentAddr1,
+ volatile unsigned int *remoteSegmentAddr2,
+ unsigned int segmentSize)
+
+{
+
+ volatile unsigned int *remoteBuffer1;
+ volatile unsigned int *remoteBuffer;
+ volatile unsigned int *remoteBuffer2;
+ static int times = 0;
+ sci_sequence_t sequence;
+ sci_error_t error;
+ unsigned int nostores;
+ unsigned int j;
+ sci_sequence_status_t sequenceStatus;
+
+
+ remoteBuffer1 = (volatile unsigned int *)remoteSegmentAddr1;
+ remoteBuffer2 = (volatile unsigned int *)remoteSegmentAddr2;
+ remoteBuffer=remoteBuffer1;
+
+ /* 4-byte test only */
+ nostores = (segmentSize) / sizeof(unsigned int);
+
+ /* Create a sequence for data error checking */
+ SCICreateMapSequence(remoteMapOne,&sequence,NO_FLAGS,&error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr,"SCICreateMapSequence failed - Error code 0x%x\n",error);
+ return error;
+ }
+
+
+
+ /* Fill in the data into a local buffer */
+ error = SendInterrupt(sdOne,localAdapterNo,localNodeId1,remoteNodeId1, DATA_TRANSFER_READY);
+
+ error = FillSegmentWithData(segmentSize, 0);
+
+ tryagain:
+ PrintServerData(localbufferPtr);
+ fprintf(stderr,"After recover \n");
+ while(1){
+
+
+ //data=0;
+
+ if (error != SCI_ERR_OK) {
+ /*
+ * Unable to create local buffer - Insufficient memory available
+ */
+ printf( "Unable to create local buffer - Insufficient memory available\n" );
+
+ return error;
+ }
+
+ do {
+ /* Start data error checking */
+ sequenceStatus = SCIStartSequence(sequence,NO_FLAGS,&error);
+ } while (sequenceStatus != SCI_SEQ_OK) ;
+
+
+ /* Transfer data to remote node */
+ for (j=0;j<nostores;j++) {
+ remoteBuffer[j] = localbufferPtr[j];
+ }
+
+ /* Check for error after data transfer */
+ sequenceStatus = SCICheckSequence(sequence,NO_FLAGS,&error);
+ if (sequenceStatus != SCI_SEQ_OK) {
+ fprintf(stderr,"Data transfer failed\n");
+ if(times==0) {
+ error = FillSegmentWithData(segmentSize, 1);
+
+ SCICreateMapSequence(remoteMapTwo,&sequence,NO_FLAGS,&error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr,"SCICreateMapSequence failed - Error code 0x%x\n",error);
+ return error;
+ return SCI_ERR_TRANSFER_FAILED;
+ }
+ }
+ else
+ {
+ error = FillSegmentWithData(segmentSize, 0);
+ /* Create a sequence for data error checking */
+ SCICreateMapSequence(remoteMapOne,&sequence,NO_FLAGS,&error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr,"SCICreateMapSequence failed - Error code 0x%x\n",error);
+ return error;
+ return SCI_ERR_TRANSFER_FAILED;
+ }
+
+ }
+ fprintf(stderr,"Recovery \n");
+ if(times==0)
+ remoteBuffer=remoteBuffer2;
+ else
+ remoteBuffer=remoteBuffer1;
+ times++;
+ printf("remotebuffer %p times %d\n", remoteBuffer, times);
+ goto tryagain;
+
+ }
+ int timeout=0;
+ // error = SendInterrupt(sdOne,localAdapterNo,localNodeId1,remoteNodeId1, DATA_TRANSFER_READY);
+ // NdbSleep_MilliSleep(100);
+ //error = ReceiveInterrupt(sdOne,localAdapterNo,localNodeId1,DATA_TRANSFER_READY, timeout);
+
+ }
+ /* Remove the Sequence */
+ SCIRemoveSequence(sequence,NO_FLAGS, &error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr,"SCIRemoveSequence failed - Error code 0x%x\n",error);
+ return error;
+ }
+
+ return SCI_ERR_OK;
+}
+
+
+/*********************************************************************************/
+/* S H M E M C L I E N T N O D E */
+/* */
+/*********************************************************************************/
+
+unsigned int ShmemClientNode(void)
+{
+
+ volatile unsigned int *remoteMapAddr1;
+ volatile unsigned int *remoteMapAddr2;
+ printf("here?\n");
+
+
+ /* Create a segmentId */
+ remoteSegmentId1 = 1;//(remoteNodeId1 << 16) | localNodeId1;
+
+ /* Connect to remote segment */
+
+ printf("Connect to remote segment .... \n");
+ printf("segid = %d node %d \n",remoteSegmentId1, remoteNodeId1 );
+
+ do {
+ SCIConnectSegment(sdOne,
+ &remoteSegmentOne,
+ remoteNodeId1,
+ remoteSegmentId1,
+ localAdapterNo,
+ NO_CALLBACK,
+ NULL,
+ SCI_INFINITE_TIMEOUT,
+ NO_FLAGS,
+ &error);
+
+ } while (error != SCI_ERR_OK);
+
+
+ printf("connected\n");
+
+ // remoteSegmentId2 = (remoteNodeId2 << 16) | localNodeId2;
+ // printf("segid = %d\n",remoteSegmentId2 );
+ printf("segid = %d node %d \n",remoteSegmentId1, remoteNodeId1 );
+ do {
+ SCIConnectSegment(sdTwo,
+ &remoteSegmentTwo,
+ remoteNodeId2,
+ remoteSegmentId1,
+ standbyAdapterNo,
+ NO_CALLBACK,
+ NULL,
+ SCI_INFINITE_TIMEOUT,
+ NO_FLAGS,
+ &error);
+
+ } while (error != SCI_ERR_OK);
+
+
+
+ printf("connected 3\n");
+ printf("Remote segment (id=0x%x) is connected.\n", remoteSegmentId2);
+
+
+ /* Map remote segment to user space */
+ remoteMapAddr1 = (unsigned int*)SCIMapRemoteSegment(remoteSegmentOne,&remoteMapOne,offset,segmentSize,NULL,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("Remote segment (id=0x%x) is mapped to user space @ 0x%x. \n", remoteSegmentId1, remoteMapAddr1);
+ } else {
+ fprintf(stderr,"SCIMapRemoteSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+ remoteMapAddr2 = (unsigned int *)SCIMapRemoteSegment(remoteSegmentTwo,&remoteMapTwo,offset,segmentSize,NULL,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("Remote segment (id=0x%x) is mapped to user space @ 0x%x. \n", remoteSegmentId2, remoteMapAddr2);
+ } else {
+ fprintf(stderr,"SCIMapRemoteSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+
+ /* Start data transfer and error checking */
+ error = (sci_error_t)TransferData(remoteMapOne,remoteMapAddr1, remoteMapAddr2,segmentSize);
+ if (error == SCI_ERR_OK) {
+ printf("Data transfer done!\n\n");
+ } else {
+ fprintf(stderr,"Data transfer failed - Error code 0x%x\n\n",error);
+ return 0;
+ }
+
+ /* Send an interrupt to remote node telling that the data transfer is ready */
+ error = SendInterrupt(sdOne,localAdapterNo,localNodeId1,remoteNodeId1, DATA_TRANSFER_READY);
+ if (error == SCI_ERR_OK) {
+ printf("\nInterrupt message sent to remote node\n");
+ } else {
+ printf("\nInterrupt synchronization failed\n");
+ return 0;
+ }
+
+ PrintClientData();
+
+ /* Unmap remote segment */
+ SCIUnmapSegment(remoteMapOne,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("The remote segment is unmapped\n");
+ } else {
+ fprintf(stderr,"SCIUnmapSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+ SCIUnmapSegment(remoteMapTwo,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("The remote segment is unmapped\n");
+ } else {
+ fprintf(stderr,"SCIUnmapSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+ /* Disconnect segment */
+ SCIDisconnectSegment(remoteSegmentOne,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("The segment is disconnected\n");
+ } else {
+ fprintf(stderr,"SCIDisconnectSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+ SCIDisconnectSegment(remoteSegmentTwo,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("The segment is disconnected\n");
+ } else {
+ fprintf(stderr,"SCIDisconnectSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+
+ return 1;
+}
+
+
+/*********************************************************************************/
+/* S H M E M S E R V E R N O D E */
+/* */
+/*********************************************************************************/
+
+unsigned int ShmemServerNode(void)
+{
+
+ unsigned int *localMapAddr;
+
+ /* Create a segmentId */
+ localSegmentId =1;// (localNodeId1 << 16) | remoteNodeId1;
+
+ /* Create local segment */
+ SCICreateSegment(sdOne,&localSegmentOne,localSegmentId, segmentSize, NO_CALLBACK, NULL, NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("Local segment (id=%d, size=%d) is created. \n", localSegmentId, segmentSize);
+ } else {
+ fprintf(stderr,"SCICreateSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+ //localSegmentId = (localNodeId2 << 16) | remoteNodeId2;
+ /*
+ SCICreateSegment(sdTwo,&localSegmentTwo,localSegmentId+1, segmentSize, NO_CALLBACK, NULL, NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("Local segment (id=%d, size=%d) is created (2). \n", localSegmentId, segmentSize);
+ } else {
+ fprintf(stderr,"SCICreateSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+ printf("segment one %p segment 2 %p\n", localSegmentOne, localSegmentTwo);
+ */
+ /* Prepare the segment */
+ SCIPrepareSegment(localSegmentOne,localAdapterNo,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("Local segment (id=%d, size=%d) is prepared. \n", localSegmentId, segmentSize);
+ } else {
+ fprintf(stderr,"SCIPrepareSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+
+ /* Prepare the segment */
+
+ SCIPrepareSegment(localSegmentOne,standbyAdapterNo,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("Local segment (id=%d, size=%d) is created. \n", localSegmentId, segmentSize);
+ } else {
+ fprintf(stderr,"SCIPrepareSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+
+ /* Map local segment to user space */
+ localMapAddr = (unsigned int *)SCIMapLocalSegment(localSegmentOne,&localMapOne, offset,segmentSize, NULL,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("Local segment (id=0x%x) is mapped to user space @ 0x%x.\n", localSegmentId, localMapAddr);
+ } else {
+ fprintf(stderr,"SCIMapLocalSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+
+ /* Map local segment to user space */
+ /*
+ localMapAddr = (unsigned int *)SCIMapLocalSegment(localSegmentTwo,&localMapTwo, offset,segmentSize, NULL,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("Local segment (id=0x%x) is mapped to user space @ 0x%x.\n", localSegmentId, localMapAddr);
+ printf("Local segment (id=%d) is mapped to user space.\n", localSegmentId);
+ } else {
+ fprintf(stderr,"SCIMapLocalSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+ */
+
+ /* Set the segment available */
+ SCISetSegmentAvailable(localSegmentOne, localAdapterNo, NO_FLAGS, &error);
+ if (error == SCI_ERR_OK) {
+ printf("Local segment (id=0x%x) is available for remote connections. \n", localSegmentId);
+ } else {
+ fprintf(stderr,"SCISetSegmentAvailable failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+
+ SCISetSegmentAvailable(localSegmentOne, standbyAdapterNo, NO_FLAGS, &error);
+ if (error == SCI_ERR_OK) {
+ printf("Local segment (id=0x%x) is available for remote connections. \n", localSegmentId);
+ } else {
+ fprintf(stderr,"SCISetSegmentAvailable failed - Error code 0x%x\n",error);
+ return 0;
+ }
+ int timeout=0;
+ error = ReceiveInterrupt(sdOne,localAdapterNo,localNodeId1,DATA_TRANSFER_READY, timeout);
+
+ if (error == SCI_ERR_OK) {
+ printf("\nThe data transfer is ready\n");
+ } else {
+ printf("\nInterrupt synchronization failed\n");
+ return 0;
+ }
+
+
+ again:
+
+ // printf("Wait for the shared memory data transfer .....");
+ /* Wait for interrupt signal telling that block transfer is ready */
+
+ //printf("\nData transfer done!\n");
+ //PrintClientData()
+ PrintServerData(localMapAddr);
+ /*Uint32 micros;
+ Uint32 micros2;
+ NDB_TICKS secs;
+ NdbTick_CurrentMicrosecond(&secs, &micros);
+ error = SendInterrupt(sdOne,localAdapterNo,localNodeId1,remoteNodeId1, DATA_TRANSFER_READY);
+ NdbTick_CurrentMicrosecond(&secs, &micros2);
+ printf("TIME ELAPSED %d \n", micros2-micros);
+// NdbSleep_MilliSleep(100);
+ */
+ goto again;
+
+ /* Unmap local segment */
+ SCIUnmapSegment(localMapTwo,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("The local segment is unmapped\n");
+ } else {
+ fprintf(stderr,"SCIUnmapSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+ /* Unmap local segment */
+ SCIUnmapSegment(localMapOne,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("The local segment is unmapped\n");
+ } else {
+ fprintf(stderr,"SCIUnmapSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+ /* Remove local segment */
+ SCIRemoveSegment(localSegmentOne,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("The local segment is removed\n");
+ } else {
+ fprintf(stderr,"SCIRemoveSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+ /* Remove local segment */
+ SCIRemoveSegment(localSegmentTwo,NO_FLAGS,&error);
+ if (error == SCI_ERR_OK) {
+ printf("The local segment is removed\n");
+ } else {
+ fprintf(stderr,"SCIRemoveSegment failed - Error code 0x%x\n",error);
+ return 0;
+ }
+
+
+
+
+ return 1;
+}
+
+
+
+/*********************************************************************************/
+/* M A I N */
+/* */
+/*********************************************************************************/
+
+int main(int argc,char *argv[])
+{
+
+ int counter;
+
+ printf("\n %s compiled %s : %s\n\n",argv[0],__DATE__,__TIME__);
+
+ if (argc<3) {
+ Usage();
+ exit(-1);
+ }
+
+
+ /* Get the parameters */
+ for (counter=1; counter<argc; counter++) {
+
+ if (!strcmp("-rn",argv[counter])) {
+ // remoteNodeId = strtol(argv[counter+1],(char **) NULL,10);
+ continue;
+ }
+
+ if (!strcmp("-size",argv[counter])) {
+ segmentSize = strtol(argv[counter+1],(char **) NULL,10);
+ continue;
+ }
+
+ if (!strcmp("-adapterno",argv[counter])) {
+ localAdapterNo = strtol(argv[counter+1],(char **) NULL,10);
+ continue;
+ }
+
+ if (!strcmp("-client",argv[counter])) {
+ client = 1;
+ continue;
+ }
+
+ if (!strcmp("-server",argv[counter])) {
+ server = 1;
+ continue;
+ }
+
+ if (!strcmp("-help",argv[counter])) {
+ Usage();
+ exit(0);
+ }
+ }
+
+ // if (remoteNodeId == 0) {
+ // fprintf(stderr,"Remote node-id is not specified. Use -rn <remote node-id>\n");
+ // exit(-1);
+ //}
+
+ if (server == 0 && client == 0) {
+ fprintf(stderr,"You must specify a client node or a server node\n");
+ exit(-1);
+ }
+
+ if (server == 1 && client == 1) {
+ fprintf(stderr,"Both server node and client node is selected.\n");
+ fprintf(stderr,"You must specify either a client or a server node\n");
+ exit(-1);
+ }
+
+
+ /* Initialize the SISCI library */
+ SCIInitialize(NO_FLAGS, &error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr,"SCIInitialize failed - Error code: 0x%x\n",error);
+ exit(error);
+ }
+
+
+ /* Open a file descriptor */
+ SCIOpen(&sdOne,NO_FLAGS,&error);
+ if (error != SCI_ERR_OK) {
+ if (error == SCI_ERR_INCONSISTENT_VERSIONS) {
+ fprintf(stderr,"Version mismatch between SISCI user library and SISCI driver\n");
+ }
+ fprintf(stderr,"SCIOpen failed - Error code 0x%x\n",error);
+ exit(error);
+ }
+
+ /* Open a file descriptor */
+ SCIOpen(&sdTwo,NO_FLAGS,&error);
+ if (error != SCI_ERR_OK) {
+ if (error == SCI_ERR_INCONSISTENT_VERSIONS) {
+ fprintf(stderr,"Version mismatch between SISCI user library and SISCI driver\n");
+ }
+ fprintf(stderr,"SCIOpen failed - Error code 0x%x\n",error);
+ exit(error);
+ }
+
+
+ /* Get local node-id */
+ error = GetLocalNodeId(localAdapterNo, &localNodeId1);
+ error = GetLocalNodeId(standbyAdapterNo, &localNodeId2);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr,"Could not find the local adapter %d\n", localAdapterNo);
+ SCIClose(sdOne,NO_FLAGS,&error);
+ SCIClose(sdTwo,NO_FLAGS,&error);
+ exit(-1);
+ }
+
+
+ /* Print parameters */
+ PrintParameters();
+
+ if (client) {
+ remoteNodeId1=324;
+ remoteNodeId2=328;
+ ShmemClientNode();
+ } else {
+ remoteNodeId1=452;
+ remoteNodeId2=456;
+ ShmemServerNode();
+ }
+
+ /* Close the file descriptor */
+ SCIClose(sdOne,NO_FLAGS,&error);
+ SCIClose(sdTwo,NO_FLAGS,&error);
+ if (error != SCI_ERR_OK) {
+ fprintf(stderr,"SCIClose failed - Error code: 0x%x\n",error);
+ }
+
+
+ /* Free allocated resources */
+ SCITerminate();
+
+ return SCI_ERR_OK;
+}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/ndb/src/common/transporter/perftest/Makefile b/ndb/src/common/transporter/perftest/Makefile
new file mode 100644
index 00000000000..01869e1acf9
--- /dev/null
+++ b/ndb/src/common/transporter/perftest/Makefile
@@ -0,0 +1,15 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := perfTransporterTest
+BIN_TARGET_ARCHIVES := transporter portlib general
+
+SOURCES = perfTransporterTest.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
diff --git a/ndb/src/common/transporter/perftest/perfTransporterTest.cpp b/ndb/src/common/transporter/perftest/perfTransporterTest.cpp
new file mode 100644
index 00000000000..6d7f7083a48
--- /dev/null
+++ b/ndb/src/common/transporter/perftest/perfTransporterTest.cpp
@@ -0,0 +1,774 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "TransporterRegistry.hpp"
+#include "TransporterDefinitions.hpp"
+#include "TransporterCallback.hpp"
+#include <RefConvert.hpp>
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbString.h>
+
+int basePortTCP = 17000;
+
+SCI_TransporterConfiguration sciTemplate = {
+ 2000,
+ // Packet size
+ 2000000, // Buffer size
+ 2, // number of adapters
+ 1, // remote node id SCI
+ 2, // Remote node Id SCI
+ 0, // local ndb node id (server)
+ 0, // remote ndb node id (client)
+ 0, // byteOrder;
+ false, // compression;
+ true, // checksum;
+ true // signalId;
+};
+
+
+SHM_TransporterConfiguration shmTemplate = {
+ 0, //remoteNodeId
+ 0, //localNodeId;
+ false, //compression
+ true, //checksum;
+ true, //signalId;
+ 0, //byteOrder;
+ 123, //shmKey;
+ 25000000 //shmSize;
+};
+
+
+TCP_TransporterConfiguration tcpTemplate = {
+ 17000, // port;
+ "", // remoteHostName;
+ "", // localhostname
+ 2, // remoteNodeId;
+ 1, // localNodeId;
+ 25000000, // sendBufferSize - Size of SendBuffer of priority B
+ 5000000, // maxReceiveSize - Maximum no of bytes to receive
+ 0, // byteOrder;
+ false, // compression;
+ true, // checksum;
+ true // signalId;
+};
+
+OSE_TransporterConfiguration oseTemplate = {
+ "", // remoteHostName;
+ "", // localHostName;
+ 0, // remoteNodeId;
+ 0, // localNodeId;
+ false, // compression;
+ true, // checksum;
+ true, // signalId;
+ 0, // byteOrder;
+
+ 2000, // prioASignalSize;
+ 2000, // prioBSignalSize;
+ 10 // Recv buf size
+};
+
+TransporterRegistry *tReg = 0;
+
+#ifndef OSE_DELTA
+#include <signal.h>
+#endif
+
+extern "C"
+void
+signalHandler(int signo){
+#ifndef OSE_DELTA
+ ::signal(13, signalHandler);
+#endif
+ char buf[255];
+ sprintf(buf,"Signal: %d\n", signo);
+ ndbout << buf << endl;
+}
+
+void
+usage(const char * progName){
+ ndbout << "Usage: " << progName << " <type> localNodeId localHostName"
+ << " remoteHostName"
+ << " [<loop count>] [<send buf size>] [<recv buf size>]" << endl;
+ ndbout << " type = shm tcp ose sci" << endl;
+ ndbout << " localNodeId - {1,2}" << endl;
+}
+
+typedef void (* CreateTransporterFunc)(void * conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendBuf,
+ int recvBuf);
+
+void
+createOSETransporter(void*, NodeId, NodeId, const char*, const char*, int, int);
+void
+createTCPTransporter(void*, NodeId, NodeId, const char*, const char*, int, int);
+void
+createSHMTransporter(void*, NodeId, NodeId, const char*, const char*, int, int);
+void
+createSCITransporter(void*, NodeId, NodeId, const char*, const char*, int, int);
+
+struct TestPhase {
+ int signalSize;
+ int noOfSignals;
+ int noOfSignalSent;
+ int noOfSignalReceived;
+ NDB_TICKS startTime;
+ NDB_TICKS stopTime;
+ NDB_TICKS accTime;
+ int loopCount;
+ Uint64 sendLenBytes, sendCount;
+ Uint64 recvLenBytes, recvCount;
+};
+
+TestPhase testSpec[] = {
+ { 1, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of size 1 word
+ ,{ 1, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals of size 1 word
+ ,{ 1, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of size 1 word
+ ,{ 1, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of size 1 word
+
+ ,{ 8, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of size 1 word
+ ,{ 8, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals of size 1 word
+ ,{ 8, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of size 1 word
+ ,{ 8, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of size 1 word
+
+ ,{ 16, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of size 1 word
+ ,{ 16, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals of size 1 word
+ ,{ 16, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of size 1 word
+ ,{ 16, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of size 1 word
+
+ ,{ 24, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of size 1 word
+ ,{ 24, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals of size 1 word
+ ,{ 24, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of size 1 word
+ ,{ 24, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of size 1 word
+
+ ,{ 0, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of random size
+ ,{ 0, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals of random size
+ ,{ 0, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of random size
+ ,{ 0, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of random size
+
+ ,{ 100, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals
+ ,{ 100, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals
+ ,{ 100, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals
+ ,{ 100, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals
+
+ ,{ 500, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals
+ ,{ 500, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals
+ ,{ 500, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals
+ ,{ 500, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals
+
+ ,{ 1000, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals
+ ,{ 1000, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals
+ ,{ 1000, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals
+ ,{ 1000, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals
+};
+
+const int noOfTests = sizeof(testSpec)/sizeof(TestPhase);
+
+Uint32 StaticBuffer[1000];
+
+SendStatus
+sendSignalTo(NodeId nodeId, int signalSize, Uint32 count){
+ if(signalSize == 0)
+ signalSize = (rand() % 25) + 1;
+
+ SignalHeader sh;
+ sh.theLength = (signalSize > 25 ? 25 : signalSize);
+ sh.theVerId_signalNumber = count;
+ sh.theReceiversBlockNumber = rand();
+ sh.theSendersBlockRef = rand();
+ sh.theSendersSignalId = rand();
+ sh.theSignalId = rand();
+ sh.theTrace = rand();
+
+ Uint32 theData[25];
+ for(int i = 0; i<25; i++)
+ theData[i] = (i+1) * (Uint32)(&theData[i]);
+
+ theData[0] = count;
+ LinearSectionPtr ptr[3];
+
+ if(signalSize <= 25){
+ sh.m_noOfSections = 0;
+ } else {
+ sh.m_noOfSections = 1;
+ ptr[0].sz = signalSize - 25;
+ ptr[0].p = &StaticBuffer[0];
+ }
+
+ return tReg->prepareSend(&sh, 1, theData, nodeId, ptr);
+}
+
+void
+reportHeader(){
+ ndbout << "#Sigs\tSz\tTime\tSig/sec\tBps\tBps-tot\t"
+ << "s len\tr len" << endl;
+}
+
+void
+print(char * dst, int i){
+ if(i > 1000000){
+ const int d = i / 1000000;
+ const int r = (i - (d * 1000000)) / 100000;
+ if(d < 100)
+ sprintf(dst, "%d.%dM", d, r);
+ else
+ sprintf(dst, "%dM", d);
+ } else if(i > 1000){
+ const int d = i / 1000;
+ const int r = (i - (d * 1000)) / 100;
+ if(d < 100)
+ sprintf(dst, "%d.%dk", d, r);
+ else
+ sprintf(dst, "%dk", d);
+ } else {
+ sprintf(dst, "%d", i);
+ }
+}
+
+void
+printReport(TestPhase & p){
+ if(p.accTime > 0) {
+ Uint32 secs = (p.accTime/p.loopCount)/1000;
+ Uint32 mill = (p.accTime/p.loopCount)%1000;
+ char st[255];
+ if(secs > 0){
+ sprintf(st, "%d.%.2ds", secs, (mill/10));
+ } else {
+ sprintf(st, "%dms", mill);
+ }
+
+ Uint32 sps = (1000*p.noOfSignals*p.loopCount)/p.accTime;
+ Uint32 dps = ((4000*p.noOfSignals)/p.accTime)*(p.loopCount*p.signalSize);
+ Uint32 bps = ((4000*p.noOfSignals)/p.accTime)*(p.loopCount*(p.signalSize+3));
+ if(p.signalSize == 0){
+ dps = ((4000*p.noOfSignals)/p.accTime)*(p.loopCount*(13));
+ bps = ((4000*p.noOfSignals)/p.accTime)*(p.loopCount*(13+3));
+ }
+ char ssps[255];
+ char sbps[255];
+ char sdps[255];
+
+ print(ssps, sps);
+ print(sbps, bps);
+ print(sdps, dps);
+
+
+ char buf[255];
+ if(p.signalSize != 0){
+ snprintf(buf, 255,
+ "%d\t%d\t%s\t%s\t%s\t%s\t%d\t%d",
+ p.noOfSignals,
+ 4*p.signalSize,
+ st,
+ ssps,
+ sdps,
+ sbps,
+ (int)(p.sendLenBytes / (p.sendCount == 0 ? 1 : p.sendCount)),
+ (int)(p.recvLenBytes / (p.recvCount == 0 ? 1 : p.recvCount)));
+ } else {
+ snprintf(buf, 255,
+ "%d\trand\t%s\t%s\t%s\t%s\t%d\t%d",
+ p.noOfSignals,
+ st,
+ ssps,
+ sdps,
+ sbps,
+ (int)(p.sendLenBytes / (p.sendCount == 0 ? 1 : p.sendCount)),
+ (int)(p.recvLenBytes / (p.recvCount == 0 ? 1 : p.recvCount)));
+
+ }
+ ndbout << buf << endl;
+ }
+}
+
+int loopCount = 1;
+int sendBufSz = -1;
+int recvBufSz = -1;
+
+bool isClient = false;
+bool isConnected = false;
+bool isStarted = false;
+int currentPhase = 0;
+TestPhase allPhases[noOfTests];
+Uint32 signalToEcho;
+Uint32 signalsEchoed;
+NDB_TICKS startTime, stopTime;
+
+void
+client(NodeId remoteNodeId){
+ isClient = true;
+
+ currentPhase = 0;
+ memcpy(allPhases, testSpec, sizeof(testSpec));
+
+ int counter = 0;
+ int sigCounter = 0;
+
+ while(true){
+ TestPhase * current = &allPhases[currentPhase];
+ if(current->noOfSignals == current->noOfSignalSent &&
+ current->noOfSignals == current->noOfSignalReceived){
+
+ /**
+ * Test phase done
+ */
+ current->stopTime = NdbTick_CurrentMillisecond();
+ current->accTime += (current->stopTime - current->startTime);
+
+ NdbSleep_MilliSleep(500 / loopCount);
+
+ current->startTime = NdbTick_CurrentMillisecond();
+
+ current->noOfSignalSent = 0;
+ current->noOfSignalReceived = 0;
+
+ current->loopCount ++;
+ if(current->loopCount == loopCount){
+
+ printReport(allPhases[currentPhase]);
+
+ currentPhase ++;
+ if(currentPhase == noOfTests){
+ /**
+ * Now we are done
+ */
+ break;
+ }
+ NdbSleep_MilliSleep(500);
+ current = &allPhases[currentPhase];
+ current->startTime = NdbTick_CurrentMillisecond();
+ }
+ }
+
+ int signalsLeft = current->noOfSignals - current->noOfSignalSent;
+ if(signalsLeft > 0){
+ for(; signalsLeft > 0; signalsLeft--){
+ if(sendSignalTo(remoteNodeId,current->signalSize,sigCounter)== SEND_OK){
+ current->noOfSignalSent++;
+ sigCounter++;
+ } else {
+ ndbout << "Failed to send: " << sigCounter << endl;
+ tReg->external_IO(10);
+ break;
+ }
+ }
+ }
+ if(counter % 10 == 0)
+ tReg->checkConnections();
+ tReg->external_IO(0);
+ counter++;
+ }
+}
+
+void
+server(){
+ isClient = false;
+
+ signalToEcho = 0;
+ signalsEchoed = 0;
+ for(int i = 0; i<noOfTests; i++)
+ signalToEcho += testSpec[i].noOfSignals;
+
+ signalToEcho *= loopCount;
+
+ while(signalToEcho > signalsEchoed){
+ tReg->checkConnections();
+ for(int i = 0; i<10; i++)
+ tReg->external_IO(10);
+ }
+}
+
+int
+main(int argc, const char **argv){
+
+ const char * progName = argv[0];
+
+ loopCount = 100;
+ sendBufSz = -1;
+ recvBufSz = -1;
+
+ isClient = false;
+ isConnected = false;
+ isStarted = false;
+ currentPhase = 0;
+
+ signalHandler(0);
+
+ if(argc < 5){
+ usage(progName);
+ return 0;
+ }
+
+ const char * type = argv[1];
+ const NodeId localNodeId = atoi(argv[2]);
+ const char * localHostName = argv[3];
+ const char * remoteHost1 = argv[4];
+
+ if(argc >= 6)
+ loopCount = atoi(argv[5]);
+ if(argc >= 7)
+ sendBufSz = atoi(argv[6]);
+ if(argc >= 8)
+ recvBufSz = atoi(argv[7]);
+
+ if(localNodeId < 1 || localNodeId > 2){
+ ndbout << "localNodeId = " << localNodeId << endl << endl;
+ usage(progName);
+ return 0;
+ }
+
+ if(localNodeId == 1)
+ ndbout << "-- ECHO CLIENT --" << endl;
+ else
+ ndbout << "-- ECHO SERVER --" << endl;
+
+ ndbout << "localNodeId: " << localNodeId << endl;
+ ndbout << "localHostName: " << localHostName << endl;
+ ndbout << "remoteHost1 (node " << (localNodeId == 1?2:1) << "): "
+ << remoteHost1 << endl;
+ ndbout << "Loop count: " << loopCount << endl;
+ ndbout << "-----------------" << endl;
+
+ void * confTemplate = 0;
+ CreateTransporterFunc func = 0;
+ if(strcasecmp(type, "tcp") == 0){
+ func = createTCPTransporter;
+ confTemplate = &tcpTemplate;
+ } else if(strcasecmp(type, "ose") == 0){
+ func = createOSETransporter;
+ confTemplate = &oseTemplate;
+ } else if(strcasecmp(type, "sci") == 0){
+ func = createSCITransporter;
+ confTemplate = &sciTemplate;
+ } else if(strcasecmp(type, "shm") == 0){
+ func = createSHMTransporter;
+ confTemplate = &shmTemplate;
+ } else {
+ ndbout << "Unsupported transporter type" << endl;
+ return 0;
+ }
+
+ ndbout << "Creating transporter registry" << endl;
+ tReg = new TransporterRegistry;
+ tReg->init(localNodeId);
+
+ switch(localNodeId){
+ case 1:
+ (* func)(confTemplate, 1, 2, localHostName, remoteHost1,
+ sendBufSz, recvBufSz);
+ break;
+ case 2:
+ (* func)(confTemplate, 2, 1, localHostName, remoteHost1,
+ sendBufSz, recvBufSz);
+ break;
+ }
+
+ ndbout << "Doing startSending/startReceiving" << endl;
+ tReg->startSending();
+ tReg->startReceiving();
+
+ ndbout << "Connecting" << endl;
+ tReg->setPerformState(PerformConnect);
+ tReg->checkConnections();
+
+ if(localNodeId == 1)
+ client(2);
+ else
+ server();
+
+ isStarted = false;
+
+ ndbout << "Sleep 3 secs" << endl;
+ NdbSleep_SecSleep(3);
+
+ ndbout << "Doing setPerformState(Disconnect)" << endl;
+ tReg->setPerformState(PerformDisconnect);
+
+ ndbout << "Doing checkConnections()" << endl;
+ tReg->checkConnections();
+
+ ndbout << "Deleting transporter registry" << endl;
+ delete tReg; tReg = 0;
+
+ return 0;
+}
+
+void
+execute(void* callbackObj, SignalHeader * const header, Uint8 prio,
+ Uint32 * const theData,
+ LinearSectionPtr ptr[3]){
+ const NodeId nodeId = refToNode(header->theSendersBlockRef);
+
+ if(isClient){
+ allPhases[currentPhase].noOfSignalReceived++;
+ } else {
+ int sleepTime = 10;
+ if(theData[0] != signalsEchoed){
+ ndbout << "Missing signal theData[0] = " << theData[0]
+ << " signalsEchoed = " << signalsEchoed << endl;
+ ndbout << (* header) << endl;
+ abort();
+ }
+ while(tReg->prepareSend(header, prio, theData, nodeId, ptr) != SEND_OK){
+ ndbout << "Failed to echo " << theData[0] << endl;
+ NdbSleep_MilliSleep(sleepTime);
+ // sleepTime += 10;
+ }
+ signalsEchoed++;
+ }
+}
+
+void
+copy(Uint32 * & insertPtr,
+ class SectionSegmentPool & thePool, const SegmentedSectionPtr & _ptr){
+ abort();
+}
+
+void
+reportError(void* callbackObj, NodeId nodeId, TransporterError errorCode){
+ char buf[255];
+ sprintf(buf, "reportError (%d, %x) in perfTest", nodeId, errorCode);
+ ndbout << buf << endl;
+ if(errorCode & 0x8000 && errorCode != 0x8014){
+ abort(); //tReg->setPerformState(nodeId, PerformDisconnect);
+ }
+}
+
+/**
+ * Report average send theLength in bytes (4096 last sends)
+ */
+void
+reportSendLen(void* callbackObj, NodeId nodeId, Uint32 count, Uint64 bytes){
+ allPhases[currentPhase].sendCount += count;
+ allPhases[currentPhase].sendLenBytes += bytes;
+
+ if(!isClient){
+ ndbout << "reportSendLen(" << nodeId << ", "
+ << (bytes/count) << ")" << endl;
+ }
+}
+
+/**
+ * Report average receive theLength in bytes (4096 last receives)
+ */
+void
+reportReceiveLen(void* callbackObj, NodeId nodeId, Uint32 count, Uint64 bytes){
+ allPhases[currentPhase].recvCount += count;
+ allPhases[currentPhase].recvLenBytes += bytes;
+
+ if(!isClient){
+ ndbout << "reportReceiveLen(" << nodeId << ", "
+ << (bytes/count) << ")" << endl;
+ }
+}
+
+/**
+ * Report connection established
+ */
+void
+reportConnect(void* callbackObj, NodeId nodeId){
+ char buf[255];
+ sprintf(buf, "reportConnect(%d)", nodeId);
+ ndbout << buf << endl;
+ tReg->setPerformState(nodeId, PerformIO);
+
+ if(!isStarted){
+ isStarted = true;
+ startTime = NdbTick_CurrentMillisecond();
+ if(isClient){
+ reportHeader();
+ allPhases[0].startTime = startTime;
+ }
+ }
+ else{
+ // Resend signals that were lost when connection failed
+ TestPhase * current = &allPhases[currentPhase];
+ current->noOfSignalSent = current->noOfSignalReceived;
+ }
+}
+
+/**
+ * Report connection broken
+ */
+void
+reportDisconnect(void* callbackObj, NodeId nodeId, Uint32 errNo){
+ char buf[255];
+ sprintf(buf, "reportDisconnect(%d)", nodeId);
+ ndbout << buf << endl;
+
+ if(isStarted)
+ tReg->setPerformState(nodeId, PerformConnect);
+}
+
+
+int
+checkJobBuffer() {
+ /**
+ * Check to see if jobbbuffers are starting to get full
+ * and if so call doJob
+ */
+ return 0;
+}
+
+void
+createOSETransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendBuf,
+ int recvBuf){
+
+ ndbout << "Creating OSE transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+ OSE_TransporterConfiguration * conf = (OSE_TransporterConfiguration*)_conf;
+
+ if(sendBuf != -1){
+ conf->prioBSignalSize = sendBuf;
+ }
+ if(recvBuf != -1){
+ conf->receiveBufferSize = recvBuf;
+ }
+
+ ndbout << "\tSendBufferSize: " << conf->prioBSignalSize << endl;
+ ndbout << "\tReceiveBufferSize: " << conf->receiveBufferSize << endl;
+
+ conf->localNodeId = localNodeId;
+ conf->localHostName = localHostName;
+ conf->remoteNodeId = remoteNodeId;
+ conf->remoteHostName = remoteHostName;
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+
+void
+createSCITransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendbuf,
+ int recvbuf) {
+
+
+ ndbout << "Creating SCI transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+
+ SCI_TransporterConfiguration * conf = (SCI_TransporterConfiguration*)_conf;
+
+ conf->remoteSciNodeId0= (Uint16)atoi(localHostName);
+ conf->remoteSciNodeId1= (Uint16)atoi(remoteHostName);
+
+
+ conf->localNodeId = localNodeId;
+ conf->remoteNodeId = remoteNodeId;
+
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+void
+createSHMTransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendbuf,
+ int recvbuf) {
+
+
+ ndbout << "Creating SHM transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+
+ SHM_TransporterConfiguration * conf = (SHM_TransporterConfiguration*)_conf;
+
+
+ conf->localNodeId = localNodeId;
+ conf->remoteNodeId = remoteNodeId;
+
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+
+void
+createTCPTransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendBuf,
+ int recvBuf){
+ ndbout << "Creating TCP transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+ TCP_TransporterConfiguration * conf = (TCP_TransporterConfiguration*)_conf;
+
+ int port;
+ if(localNodeId == 1 && remoteNodeId == 2) port = basePortTCP + 0;
+ if(localNodeId == 1 && remoteNodeId == 3) port = basePortTCP + 1;
+ if(localNodeId == 2 && remoteNodeId == 1) port = basePortTCP + 0;
+ if(localNodeId == 2 && remoteNodeId == 3) port = basePortTCP + 2;
+ if(localNodeId == 3 && remoteNodeId == 1) port = basePortTCP + 1;
+ if(localNodeId == 3 && remoteNodeId == 2) port = basePortTCP + 2;
+
+ if(sendBuf != -1){
+ conf->sendBufferSize = sendBuf;
+ }
+ if(recvBuf != -1){
+ conf->maxReceiveSize = recvBuf;
+ }
+
+ ndbout << "\tSendBufferSize: " << conf->sendBufferSize << endl;
+ ndbout << "\tReceiveBufferSize: " << conf->maxReceiveSize << endl;
+
+ conf->localNodeId = localNodeId;
+ conf->localHostName = localHostName;
+ conf->remoteNodeId = remoteNodeId;
+ conf->remoteHostName = remoteHostName;
+ conf->port = port;
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
diff --git a/ndb/src/common/transporter/priotest/Makefile b/ndb/src/common/transporter/priotest/Makefile
new file mode 100644
index 00000000000..483fc0f1f07
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/Makefile
@@ -0,0 +1,15 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+SOURCES = prioTransporterTest.cpp
+ARCHIVE_TARGET := libpriotransportertest.a
+
+DIRS := prioTCP prioSHM prioSCI
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
diff --git a/ndb/src/common/transporter/priotest/prioOSE/Makefile b/ndb/src/common/transporter/priotest/prioOSE/Makefile
new file mode 100644
index 00000000000..4df66fa35e0
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/prioOSE/Makefile
@@ -0,0 +1,17 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := perfOSE
+BIN_TARGET_ARCHIVES := perftransportertest transporter portlib
+
+CCFLAGS_LOC += -I..
+
+SOURCES = perfOSE.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
diff --git a/ndb/src/common/transporter/priotest/prioSCI/Makefile b/ndb/src/common/transporter/priotest/prioSCI/Makefile
new file mode 100644
index 00000000000..7d403539bf3
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/prioSCI/Makefile
@@ -0,0 +1,17 @@
+include .defs.mk
+
+TYPE := ndbapi
+BIN_TARGET := prioSCI
+BIN_TARGET_LIBS := sisci
+BIN_TARGET_ARCHIVES := priotransportertest transporter portlib general
+
+CCFLAGS_LOC += -I..
+
+SOURCES = prioSCI.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
diff --git a/ndb/src/common/transporter/priotest/prioSCI/prioSCI.cpp b/ndb/src/common/transporter/priotest/prioSCI/prioSCI.cpp
new file mode 100644
index 00000000000..6218b764e09
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/prioSCI/prioSCI.cpp
@@ -0,0 +1,29 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <prioTransporterTest.hpp>
+#include <NdbMain.h>
+
+NDB_COMMAND(prioSCI, "prioSCI", "prioSCI", "Test the SCI Transporter", 65535)
+{
+ basePortTCP = 17000;
+ return prioTransporterTest(TestSCI, "prioSCI", argc, argv);
+}
+
+
+
+
diff --git a/ndb/src/common/transporter/priotest/prioSHM/Makefile b/ndb/src/common/transporter/priotest/prioSHM/Makefile
new file mode 100644
index 00000000000..a827c6e3f1e
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/prioSHM/Makefile
@@ -0,0 +1,13 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := prioSHM
+BIN_TARGET_ARCHIVES := priotransportertest transporter portlib general
+
+CCFLAGS_LOC += -I..
+
+SOURCES = prioSHM.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/common/transporter/priotest/prioSHM/prioSHM.cpp b/ndb/src/common/transporter/priotest/prioSHM/prioSHM.cpp
new file mode 100644
index 00000000000..4c1701a91e4
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/prioSHM/prioSHM.cpp
@@ -0,0 +1,26 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <prioTransporterTest.hpp>
+#include <NdbMain.h>
+
+NDB_COMMAND(prioSHM, "prioSHM", "prioSHM", "Test the SHM Transporter", 65535)
+{
+ basePortTCP = 17000;
+ return prioTransporterTest(TestSHM, "prioSHM", argc, argv);
+}
+
diff --git a/ndb/src/common/transporter/priotest/prioTCP/Makefile b/ndb/src/common/transporter/priotest/prioTCP/Makefile
new file mode 100644
index 00000000000..92abf3e7424
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/prioTCP/Makefile
@@ -0,0 +1,13 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := prioTCP
+BIN_TARGET_ARCHIVES := priotransportertest transporter portlib general
+
+CCFLAGS_LOC += -I..
+
+SOURCES = prioTCP.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/common/transporter/priotest/prioTCP/prioTCP.cpp b/ndb/src/common/transporter/priotest/prioTCP/prioTCP.cpp
new file mode 100644
index 00000000000..f993dd05ac8
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/prioTCP/prioTCP.cpp
@@ -0,0 +1,26 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <prioTransporterTest.hpp>
+#include <NdbMain.h>
+
+NDB_COMMAND(prioTCP, "prioTCP", "prioTCP", "Test the TCP Transporter", 65535)
+{
+ basePortTCP = 17000;
+ return prioTransporterTest(TestTCP, "prioTCP", argc, argv);
+}
+
diff --git a/ndb/src/common/transporter/priotest/prioTransporterTest.cpp b/ndb/src/common/transporter/priotest/prioTransporterTest.cpp
new file mode 100644
index 00000000000..919cc9d7511
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/prioTransporterTest.cpp
@@ -0,0 +1,768 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "TransporterRegistry.hpp"
+#include "TransporterDefinitions.hpp"
+#include "TransporterCallback.hpp"
+#include <RefConvert.hpp>
+
+#include "prioTransporterTest.hpp"
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+
+int basePortTCP = 17000;
+
+SCI_TransporterConfiguration sciTemplate = {
+ 2000,
+ // Packet size
+ 2000000, // Buffer size
+ 2, // number of adapters
+ 1, // remote node id SCI
+ 2, // Remote node Id SCI
+ 0, // local ndb node id (server)
+ 0, // remote ndb node id (client)
+ 0, // byteOrder;
+ false, // compression;
+ true, // checksum;
+ true // signalId;
+};
+
+
+SHM_TransporterConfiguration shmTemplate = {
+ 100000, // shmSize
+ 0, // shmKey
+ 1, // local ndb node id (server)
+ 2, // remote ndb node id (client)
+ 0, // byteOrder;
+ false, // compression;
+ true, // checksum;
+ true // signalId;
+};
+
+TCP_TransporterConfiguration tcpTemplate = {
+ 17000, // port;
+ "", // remoteHostName;
+ "", // localhostname
+ 2, // remoteNodeId;
+ 1, // localNodeId;
+ 2000000, // sendBufferSize - Size of SendBuffer of priority B
+ 2000, // maxReceiveSize - Maximum no of bytes to receive
+ 0, // byteOrder;
+ false, // compression;
+ true, // checksum;
+ true // signalId;
+};
+
+OSE_TransporterConfiguration oseTemplate = {
+ "", // remoteHostName;
+ "", // localHostName;
+ 0, // remoteNodeId;
+ 0, // localNodeId;
+ false, // compression;
+ true, // checksum;
+ true, // signalId;
+ 0, // byteOrder;
+
+ 2000, // prioASignalSize;
+ 2000, // prioBSignalSize;
+ 10 // Recv buf size
+};
+
+TransporterRegistry *tReg = 0;
+
+#ifndef OSE_DELTA
+#include <signal.h>
+#endif
+
+extern "C"
+void
+signalHandler(int signo){
+#ifndef OSE_DELTA
+ ::signal(13, signalHandler);
+#endif
+ char buf[255];
+ sprintf(buf,"Signal: %d\n", signo);
+ ndbout << buf << endl;
+}
+
+void
+usage(const char * progName){
+ ndbout << "Usage: " << progName << " localNodeId localHostName"
+ << " remoteHostName"
+ << " [<loop count>] [<send buf size>] [<recv buf size>]" << endl;
+ ndbout << " localNodeId - {1,2}" << endl;
+}
+
+typedef void (* CreateTransporterFunc)(void * conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendBuf,
+ int recvBuf);
+
+void
+createOSETransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendBuf,
+ int recvBuf){
+
+ ndbout << "Creating OSE transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+ OSE_TransporterConfiguration * conf = (OSE_TransporterConfiguration*)_conf;
+
+ if(sendBuf != -1){
+ conf->prioBSignalSize = sendBuf;
+ }
+ if(recvBuf != -1){
+ conf->receiveBufferSize = recvBuf;
+ }
+
+ ndbout << "\tSendBufferSize: " << conf->prioBSignalSize << endl;
+ ndbout << "\tReceiveBufferSize: " << conf->receiveBufferSize << endl;
+
+ conf->localNodeId = localNodeId;
+ conf->localHostName = localHostName;
+ conf->remoteNodeId = remoteNodeId;
+ conf->remoteHostName = remoteHostName;
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+
+void
+createSCITransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendbuf,
+ int recvbuf) {
+
+
+ ndbout << "Creating SCI transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+
+ SCI_TransporterConfiguration * conf = (SCI_TransporterConfiguration*)_conf;
+
+ conf->remoteSciNodeId0= (Uint16)atoi(localHostName);
+ conf->remoteSciNodeId1= (Uint16)atoi(remoteHostName);
+
+
+ conf->localNodeId = localNodeId;
+ conf->remoteNodeId = remoteNodeId;
+
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+void
+createSHMTransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendbuf,
+ int recvbuf) {
+
+
+ ndbout << "Creating SHM transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+
+ SHM_TransporterConfiguration * conf = (SHM_TransporterConfiguration*)_conf;
+
+
+ conf->localNodeId = localNodeId;
+ conf->remoteNodeId = remoteNodeId;
+
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+
+void
+createTCPTransporter(void * _conf,
+ NodeId localNodeId,
+ NodeId remoteNodeId,
+ const char * localHostName,
+ const char * remoteHostName,
+ int sendBuf,
+ int recvBuf){
+ ndbout << "Creating TCP transporter from node "
+ << localNodeId << "(" << localHostName << ") to "
+ << remoteNodeId << "(" << remoteHostName << ")..." << endl;;
+
+ TCP_TransporterConfiguration * conf = (TCP_TransporterConfiguration*)_conf;
+
+ int port;
+ if(localNodeId == 1 && remoteNodeId == 2) port = basePortTCP + 0;
+ if(localNodeId == 1 && remoteNodeId == 3) port = basePortTCP + 1;
+ if(localNodeId == 2 && remoteNodeId == 1) port = basePortTCP + 0;
+ if(localNodeId == 2 && remoteNodeId == 3) port = basePortTCP + 2;
+ if(localNodeId == 3 && remoteNodeId == 1) port = basePortTCP + 1;
+ if(localNodeId == 3 && remoteNodeId == 2) port = basePortTCP + 2;
+
+ if(sendBuf != -1){
+ conf->sendBufferSize = sendBuf;
+ }
+ if(recvBuf != -1){
+ conf->maxReceiveSize = recvBuf;
+ }
+
+ ndbout << "\tSendBufferSize: " << conf->sendBufferSize << endl;
+ ndbout << "\tReceiveBufferSize: " << conf->maxReceiveSize << endl;
+
+ conf->localNodeId = localNodeId;
+ conf->localHostName = localHostName;
+ conf->remoteNodeId = remoteNodeId;
+ conf->remoteHostName = remoteHostName;
+ conf->port = port;
+ bool res = tReg->createTransporter(conf);
+ if(res)
+ ndbout << "... -- Success " << endl;
+ else
+ ndbout << "... -- Failure " << endl;
+}
+
+struct TestPhase {
+ int signalSize;
+ int noOfSignals;
+ int noOfSignalSent;
+ int noOfSignalReceived;
+ NDB_TICKS startTime;
+ NDB_TICKS stopTime;
+
+ NDB_TICKS startTimePrioA;
+ NDB_TICKS stopTimePrioA;
+ NDB_TICKS totTimePrioA;
+ int bytesSentBeforePrioA;
+ NDB_TICKS accTime;
+ int loopCount;
+ Uint64 sendLenBytes, sendCount;
+ Uint64 recvLenBytes, recvCount;
+};
+
+TestPhase testSpec[] = {
+ { 1, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of size 1 word
+ ,{ 1, 10000, 0,0, 0,0,0,0,0,0,0 } // 100 signals of size 1 word
+ ,{ 1, 10000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of size 1 word
+ ,{ 1, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of size 1 word
+
+ ,{ 8, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of size 1 word
+ ,{ 8, 10000, 0,0, 0,0,0,0,0,0,0 } // 100 signals of size 1 word
+ ,{ 8, 10000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of size 1 word
+ ,{ 8, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of size 1 word
+
+ ,{ 16, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of size 1 word
+ ,{ 16, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals of size 1 word
+ ,{ 16, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of size 1 word
+ ,{ 16, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of size 1 word
+
+ ,{ 24, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of size 1 word
+ ,{ 24, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals of size 1 word
+ ,{ 24, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of size 1 word
+ ,{ 24, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of size 1 word
+
+ ,{ 0, 10, 0,0, 0,0,0,0,0,0,0 } // 10 signals of random size
+ ,{ 0, 100, 0,0, 0,0,0,0,0,0,0 } // 100 signals of random size
+ ,{ 0, 1000, 0,0, 0,0,0,0,0,0,0 } // 1000 signals of random size
+ ,{ 0, 10000, 0,0, 0,0,0,0,0,0,0 } // 10000 signals of random size
+};
+
+const int noOfTests = sizeof(testSpec)/sizeof(TestPhase);
+
+SendStatus
+sendSignalTo(NodeId nodeId, int signalSize, int prio){
+ if(signalSize == 0)
+ signalSize = (rand() % 25) + 1;
+
+ SignalHeader sh;
+ sh.theLength = signalSize;
+ sh.theVerId_signalNumber = rand();
+ sh.theReceiversBlockNumber = rand();
+ sh.theSendersBlockRef = rand();
+ sh.theSendersSignalId = rand();
+ sh.theSignalId = rand();
+ sh.theTrace = rand();
+
+ Uint32 theData[25];
+ for(int i = 0; i<signalSize; i++)
+ theData[i] = (i+1) * (Uint32)(&theData[i]);
+
+ return tReg->prepareSend(&sh, prio, theData, nodeId);
+}
+
+void
+reportHeader(){
+ ndbout << "#Sigs\tSz\tPayload\tTime\tSig/sec\tBps\t"
+ << "s len\tr len\tprioAtime\tbytesb4pA" << endl;
+}
+
+void
+printReport(TestPhase & p){
+ if(p.accTime > 0) {
+ Uint32 secs = (p.accTime/p.loopCount)/1000;
+ Uint32 mill = (p.accTime/p.loopCount)%1000;
+ char st[255];
+ if(secs > 0){
+ sprintf(st, "%d.%.2ds", secs, (mill/10));
+ } else {
+ sprintf(st, "%dms", mill);
+ }
+
+ Uint32 sps = (1000*p.noOfSignals*p.loopCount)/p.accTime;
+ Uint32 bps = ((4000*p.noOfSignals)/p.accTime)*(p.loopCount*(p.signalSize+3));
+ if(p.signalSize == 0)
+ ((4000*p.noOfSignals)/p.accTime)*(p.loopCount*(13+3));
+
+ char ssps[255];
+ if(sps > 1000000){
+ sps /= 1000000;
+ sprintf(ssps, "%dM", (int)sps);
+ } else if(sps > 1000){
+ sps /= 1000;
+ sprintf(ssps, "%dk", (int)sps);
+ } else {
+ sprintf(ssps, "%d", (int)sps);
+ }
+
+ char sbps[255];
+ if(bps > 1000000){
+ bps /= 1000000;
+ sprintf(sbps, "%dM", bps);
+ } else if(bps>1000){
+ bps /= 1000;
+ sprintf(sbps, "%dk", bps);
+ } else {
+ sprintf(sbps, "%d", bps);
+ }
+
+ char buf[255];
+ if(p.signalSize != 0){
+ snprintf(buf, 255,
+ "%d\t%d\t%d\t%s\t%s\t%s\t%d\t%d\t%d\t%d",
+ p.noOfSignals,
+ p.signalSize,
+ (4*p.signalSize),
+ st,
+ ssps,
+ sbps,
+ (int)(p.sendLenBytes / (p.sendCount == 0 ? 1 : p.sendCount)),
+ (int)(p.recvLenBytes / (p.recvCount == 0 ? 1 : p.recvCount)),
+ (int)(p.totTimePrioA / p.loopCount),
+ (int)(p.bytesSentBeforePrioA));
+ } else {
+ snprintf(buf, 255,
+ "%d\trand\t4*rand\t%s\t%s\t%s\t%d\t%d\t%d\t%d",
+ p.noOfSignals,
+ st,
+ ssps,
+ sbps,
+ (int)(p.sendLenBytes / (p.sendCount == 0 ? 1 : p.sendCount)),
+ (int)(p.recvLenBytes / (p.recvCount == 0 ? 1 : p.recvCount)),
+ (int)(p.totTimePrioA / p.loopCount),
+ (int)(p.bytesSentBeforePrioA));
+
+ }
+ ndbout << buf << endl;
+ }
+}
+
+int loopCount = 1;
+int sendBufSz = -1;
+int recvBufSz = -1;
+
+NDB_TICKS startSec=0;
+NDB_TICKS stopSec=0;
+Uint32 startMicro=0;
+Uint32 stopMicro=0;
+int timerStarted;
+int timerStopped;
+
+bool isClient = false;
+bool isConnected = false;
+bool isStarted = false;
+int currentPhase = 0;
+TestPhase allPhases[noOfTests];
+Uint32 signalToEcho;
+NDB_TICKS startTime, stopTime;
+
+void
+client(NodeId remoteNodeId){
+ isClient = true;
+
+ currentPhase = 0;
+ memcpy(allPhases, testSpec, sizeof(testSpec));
+
+ int counter = 0;
+
+ while(true){
+ TestPhase * current = &allPhases[currentPhase];
+ if(current->noOfSignals == current->noOfSignalSent &&
+ current->noOfSignals == current->noOfSignalReceived){
+
+ /**
+ * Test phase done
+ */
+ current->stopTime = NdbTick_CurrentMillisecond();
+ current->accTime += (current->stopTime - current->startTime);
+
+ NdbSleep_MilliSleep(500 / loopCount);
+
+ current->startTime = NdbTick_CurrentMillisecond();
+
+ current->noOfSignalSent = 0;
+ current->noOfSignalReceived = 0;
+
+ current->loopCount ++;
+ if(current->loopCount == loopCount){
+
+ printReport(allPhases[currentPhase]);
+
+ currentPhase ++;
+ if(currentPhase == noOfTests){
+ /**
+ * Now we are done
+ */
+ break;
+ }
+ NdbSleep_MilliSleep(500);
+ current = &allPhases[currentPhase];
+ current->startTime = NdbTick_CurrentMillisecond();
+ }
+ }
+ int signalsLeft = current->noOfSignals - current->noOfSignalSent;
+ if(signalsLeft > 0){
+ for(; signalsLeft > 1; signalsLeft--){
+ if(sendSignalTo(remoteNodeId, current->signalSize, 1) == SEND_OK) {
+ current->noOfSignalSent++;
+ // ndbout << "sent prio b" << endl;
+ current->bytesSentBeforePrioA += (current->signalSize << 2);
+ }
+ else {
+ tReg->external_IO(10);
+ break;
+ }
+ }
+ //prio A
+ if(signalsLeft==1) {
+ NDB_TICKS sec = 0;
+ Uint32 micro=0;
+ int ret = NdbTick_CurrentMicrosecond(&sec,&micro);
+ if(ret==0)
+ current->startTimePrioA = micro + sec*1000000;
+ if(sendSignalTo(remoteNodeId, current->signalSize, 0) == SEND_OK) {
+ current->noOfSignalSent++;
+ signalsLeft--;
+ }
+ else {
+ tReg->external_IO(10);
+ break;
+ }
+ }
+ }
+
+ if(counter % 10 == 0)
+ tReg->checkConnections();
+ tReg->external_IO(0);
+ counter++;
+ }
+}
+
+void
+server(){
+ isClient = false;
+
+ signalToEcho = 0;
+ for(int i = 0; i<noOfTests; i++)
+ signalToEcho += testSpec[i].noOfSignals;
+
+ signalToEcho *= loopCount;
+
+ while(signalToEcho > 0){
+ tReg->checkConnections();
+ for(int i = 0; i<10; i++)
+ tReg->external_IO(10);
+ }
+}
+
+int
+prioTransporterTest(TestType tt, const char * progName,
+ int argc, const char **argv){
+
+ loopCount = 100;
+ sendBufSz = -1;
+ recvBufSz = -1;
+
+ isClient = false;
+ isConnected = false;
+ isStarted = false;
+ currentPhase = 0;
+
+ signalHandler(0);
+
+ if(argc < 4){
+ usage(progName);
+ return 0;
+ }
+
+ const NodeId localNodeId = atoi(argv[1]);
+ const char * localHostName = argv[2];
+ const char * remoteHost1 = argv[3];
+
+ if(argc >= 5)
+ loopCount = atoi(argv[4]);
+ if(argc >= 6)
+ sendBufSz = atoi(argv[5]);
+ if(argc >= 7)
+ recvBufSz = atoi(argv[6]);
+
+ if(localNodeId < 1 || localNodeId > 2){
+ ndbout << "localNodeId = " << localNodeId << endl << endl;
+ usage(progName);
+ return 0;
+ }
+
+ if(localNodeId == 1)
+ ndbout << "-- ECHO CLIENT --" << endl;
+ else
+ ndbout << "-- ECHO SERVER --" << endl;
+
+ ndbout << "localNodeId: " << localNodeId << endl;
+ ndbout << "localHostName: " << localHostName << endl;
+ ndbout << "remoteHost1 (node " << (localNodeId == 1?2:1) << "): "
+ << remoteHost1 << endl;
+ ndbout << "Loop count: " << loopCount << endl;
+ ndbout << "-----------------" << endl;
+
+ void * confTemplate = 0;
+ CreateTransporterFunc func = 0;
+ switch(tt){
+ case TestTCP:
+ func = createTCPTransporter;
+ confTemplate = &tcpTemplate;
+ break;
+ case TestOSE:
+ func = createOSETransporter;
+ confTemplate = &oseTemplate;
+ break;
+ case TestSCI:
+ func = createSCITransporter;
+ confTemplate = &sciTemplate;
+ break;
+ case TestSHM:
+ func = createSHMTransporter;
+ confTemplate = &shmTemplate;
+ break;
+ default:
+ ndbout << "Unsupported transporter type" << endl;
+ return 0;
+ }
+
+ ndbout << "Creating transporter registry" << endl;
+ tReg = new TransporterRegistry;
+ tReg->init(localNodeId);
+
+ switch(localNodeId){
+ case 1:
+ (* func)(confTemplate, 1, 2, localHostName, remoteHost1,
+ sendBufSz, recvBufSz);
+ break;
+ case 2:
+ (* func)(confTemplate, 2, 1, localHostName, remoteHost1,
+ sendBufSz, recvBufSz);
+ break;
+ }
+
+ ndbout << "Doing startSending/startReceiving" << endl;
+ tReg->startSending();
+ tReg->startReceiving();
+
+ ndbout << "Connecting" << endl;
+ tReg->setPerformState(PerformConnect);
+ tReg->checkConnections();
+
+ if(localNodeId == 1)
+ client(2);
+ else
+ server();
+
+ isStarted = false;
+
+ ndbout << "Sleep 3 secs" << endl;
+ NdbSleep_SecSleep(3);
+
+ ndbout << "Doing setPerformState(Disconnect)" << endl;
+ tReg->setPerformState(PerformDisconnect);
+
+ ndbout << "Doing checkConnections()" << endl;
+ tReg->checkConnections();
+
+ ndbout << "Deleting transporter registry" << endl;
+ delete tReg; tReg = 0;
+
+ return 0;
+}
+
+NdbOut & operator <<(NdbOut & out, SignalHeader & sh){
+ out << "-- Signal Header --" << endl;
+ out << "theLength: " << sh.theLength << endl;
+ out << "gsn: " << sh.theVerId_signalNumber << endl;
+ out << "recBlockNo: " << sh.theReceiversBlockNumber << endl;
+ out << "sendBlockRef: " << sh.theSendersBlockRef << endl;
+ out << "sendersSig: " << sh.theSendersSignalId << endl;
+ out << "theSignalId: " << sh.theSignalId << endl;
+ out << "trace: " << (int)sh.theTrace << endl;
+ return out;
+}
+
+void
+execute(SignalHeader * const header, Uint8 prio, Uint32 * const theData){
+ const NodeId nodeId = refToNode(header->theSendersBlockRef);
+ NDB_TICKS sec = 0;
+ Uint32 micro=0;
+ int ret = NdbTick_CurrentMicrosecond(&sec,&micro);
+ if(prio == 0 && isClient && ret == 0) {
+ allPhases[currentPhase].stopTimePrioA = micro + sec*1000000;
+ allPhases[currentPhase].totTimePrioA +=
+ allPhases[currentPhase].stopTimePrioA -
+ allPhases[currentPhase].startTimePrioA;
+ }
+ if(ret!=0)
+ allPhases[currentPhase].totTimePrioA = -1;
+
+ if(isClient){
+ allPhases[currentPhase].noOfSignalReceived++;
+ } else {
+ int sleepTime = 10;
+ while(tReg->prepareSend(header, prio, theData, nodeId) != SEND_OK){
+ ndbout << "Failed to echo" << sleepTime << endl;
+ NdbSleep_MilliSleep(sleepTime);
+ // sleepTime += 10;
+ }
+
+ signalToEcho--;
+ }
+}
+
+void
+reportError(NodeId nodeId, TransporterError errorCode){
+ char buf[255];
+ sprintf(buf, "reportError (%d, %x) in perfTest", nodeId, errorCode);
+ ndbout << buf << endl;
+ if(errorCode & 0x8000){
+ tReg->setPerformState(nodeId, PerformDisconnect);
+ }
+}
+
+/**
+ * Report average send theLength in bytes (4096 last sends)
+ */
+void
+reportSendLen(NodeId nodeId, Uint32 count, Uint64 bytes){
+ allPhases[currentPhase].sendCount += count;
+ allPhases[currentPhase].sendLenBytes += bytes;
+
+ if(!isClient){
+ ndbout << "reportSendLen(" << nodeId << ", "
+ << (bytes/count) << ")" << endl;
+ }
+}
+
+/**
+ * Report average receive theLength in bytes (4096 last receives)
+ */
+void
+reportReceiveLen(NodeId nodeId, Uint32 count, Uint64 bytes){
+ allPhases[currentPhase].recvCount += count;
+ allPhases[currentPhase].recvLenBytes += bytes;
+
+ if(!isClient){
+ ndbout << "reportReceiveLen(" << nodeId << ", "
+ << (bytes/count) << ")" << endl;
+ }
+}
+
+/**
+ * Report connection established
+ */
+void
+reportConnect(NodeId nodeId){
+ char buf[255];
+ sprintf(buf, "reportConnect(%d)", nodeId);
+ ndbout << buf << endl;
+ tReg->setPerformState(nodeId, PerformIO);
+
+ if(!isStarted){
+ isStarted = true;
+ startTime = NdbTick_CurrentMillisecond();
+ if(isClient){
+ reportHeader();
+ allPhases[0].startTime = startTime;
+ }
+ }
+ else{
+ // Resend signals that were lost when connection failed
+ TestPhase * current = &allPhases[currentPhase];
+ current->noOfSignalSent = current->noOfSignalReceived;
+ }
+}
+
+/**
+ * Report connection broken
+ */
+void
+reportDisconnect(NodeId nodeId, Uint32 errNo){
+ char buf[255];
+ sprintf(buf, "reportDisconnect(%d)", nodeId);
+ ndbout << buf << endl;
+
+ if(isStarted)
+ tReg->setPerformState(nodeId, PerformConnect);
+}
+
+
+int
+checkJobBuffer() {
+ /**
+ * Check to see if jobbbuffers are starting to get full
+ * and if so call doJob
+ */
+ return 0;
+}
diff --git a/ndb/src/common/transporter/priotest/prioTransporterTest.hpp b/ndb/src/common/transporter/priotest/prioTransporterTest.hpp
new file mode 100644
index 00000000000..787a9f46433
--- /dev/null
+++ b/ndb/src/common/transporter/priotest/prioTransporterTest.hpp
@@ -0,0 +1,35 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PRIO_TRANSPORTER_TEST_HPP
+#define PRIO_TRANSPORTER_TEST_HPP
+
+
+enum TestType {
+ TestTCP,
+ TestOSE,
+ TestSCI,
+ TestSHM
+};
+
+extern int basePortTCP;
+
+int prioTransporterTest(TestType tt, const char * pName,
+ int argc, const char **argv);
+
+
+
+#endif
diff --git a/ndb/src/common/util/Base64.cpp b/ndb/src/common/util/Base64.cpp
new file mode 100644
index 00000000000..5f4bbc8645a
--- /dev/null
+++ b/ndb/src/common/util/Base64.cpp
@@ -0,0 +1,111 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <string.h>
+#include <Base64.hpp>
+
+static char base64_table[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
+ "abcdefghijklmnopqrstuvwxyz"
+ "0123456789+/";
+
+int
+base64_encode(UtilBuffer &src, BaseString &dst) {
+ char *s = (char *)src.get_data();
+ int i = 0;
+
+ while(i < src.length()) {
+ int c;
+ c = s[i++];
+ c <<= 8;
+
+ if(i < src.length())
+ c += s[i];
+ c <<= 8;
+ i++;
+
+ if(i < src.length())
+ c += s[i];
+ i++;
+
+ dst.append(base64_table[(c >> 18) & 0x3f]);
+ dst.append(base64_table[(c >> 12) & 0x3f]);
+
+ if(i > (src.length() + 1))
+ dst.append('=');
+ else
+ dst.append(base64_table[(c >> 6) & 0x3f]);
+
+ if(i > src.length())
+ dst.append('=');
+ else
+ dst.append(base64_table[(c >> 0) & 0x3f]);
+ }
+ return 0;
+}
+
+static inline int
+pos(char c) {
+ return strchr(base64_table, c) - base64_table;
+}
+
+
+int
+base64_decode(BaseString &src, UtilBuffer &dst) {
+ size_t size;
+ size = (src.length() * 3) / 4;
+ size_t i = 0;
+ const char *s = src.c_str();
+ while(i < size) {
+ int c = 0;
+ int mark = 0;
+ c += pos(*s++);
+ c <<= 6;
+ i++;
+
+ c += pos(*s++);
+ c <<= 6;
+ i++;
+
+ if(*s != '=')
+ c += pos(*s++);
+ else {
+ size--;
+ mark++;
+ }
+ c <<= 6;
+ i++;
+
+ if(*s != '=')
+ c += pos(*s++);
+ else {
+ size--;
+ mark++;
+ }
+ /* c <<= 6; */
+ i++;
+
+ char b[3];
+
+
+ b[0] = (c >> 16) & 0xff;
+ b[1] = (c >> 8) & 0xff;
+ b[2] = (c >> 0) & 0xff;
+
+ dst.append((void *)b, 3-mark);
+ }
+ return 0;
+}
diff --git a/ndb/src/common/util/BaseString.cpp b/ndb/src/common/util/BaseString.cpp
new file mode 100644
index 00000000000..1b0eaa1b83c
--- /dev/null
+++ b/ndb/src/common/util/BaseString.cpp
@@ -0,0 +1,418 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* -*- c-basic-offset: 4; -*- */
+#include <string.h>
+#include <NdbStdio.h>
+#include <stdarg.h>
+#include "BaseString.hpp"
+#include <stdlib.h>
+
+BaseString::BaseString()
+{
+ m_chr = new char[1];
+ m_chr[0] = 0;
+ m_len = 0;
+}
+
+BaseString::BaseString(const char* s)
+{
+ const size_t n = strlen(s);
+ m_chr = new char[n + 1];
+ memcpy(m_chr, s, n + 1);
+ m_len = n;
+}
+
+BaseString::BaseString(const BaseString& str)
+{
+ const char* const s = str.m_chr;
+ const size_t n = str.m_len;
+ char* t = new char[n + 1];
+ memcpy(t, s, n + 1);
+ m_chr = t;
+ m_len = n;
+}
+
+BaseString::~BaseString()
+{
+ delete[] m_chr;
+}
+
+BaseString&
+BaseString::assign(const char* s)
+{
+ const size_t n = strlen(s);
+ char* t = new char[n + 1];
+ memcpy(t, s, n + 1);
+ delete[] m_chr;
+ m_chr = t;
+ m_len = n;
+ return *this;
+}
+
+BaseString&
+BaseString::assign(const char* s, size_t n)
+{
+ char* t = new char[n + 1];
+ memcpy(t, s, n);
+ t[n] = 0;
+ delete[] m_chr;
+ m_chr = t;
+ m_len = n;
+ return *this;
+}
+
+BaseString&
+BaseString::assign(const BaseString& str, size_t n)
+{
+ if (n > str.m_len)
+ n = str.m_len;
+ return assign(str.m_chr, n);
+}
+
+BaseString&
+BaseString::append(const char* s)
+{
+ const size_t n = strlen(s);
+ char* t = new char[m_len + n + 1];
+ memcpy(t, m_chr, m_len);
+ memcpy(t + m_len, s, n + 1);
+ delete[] m_chr;
+ m_chr = t;
+ m_len += n;
+ return *this;
+}
+
+BaseString&
+BaseString::append(char c) {
+ return appfmt("%c", c);
+}
+
+BaseString&
+BaseString::append(const BaseString& str)
+{
+ return append(str.m_chr);
+}
+
+BaseString&
+BaseString::append(const Vector<BaseString> &vector,
+ const BaseString &separator) {
+ for(size_t i=0;i<vector.size(); i++) {
+ append(vector[i]);
+ if(i<vector.size()-1)
+ append(separator);
+ }
+ return *this;
+}
+
+BaseString&
+BaseString::assfmt(const char *fmt, ...)
+{
+ char buf[1];
+ va_list ap;
+ int l;
+
+ /* Figure out how long the formatted string will be. A small temporary
+ * buffer is used, because I don't trust all implementations to work
+ * when called as vsnprintf(NULL, 0, ...).
+ */
+ va_start(ap, fmt);
+ l = vsnprintf(buf, sizeof(buf), fmt, ap) + 1;
+ va_end(ap);
+ if(l > (int)m_len) {
+ delete[] m_chr;
+ m_chr = new char[l];
+ }
+ va_start(ap, fmt);
+ vsnprintf(m_chr, l, fmt, ap);
+ va_end(ap);
+ m_len = strlen(m_chr);
+ return *this;
+}
+
+BaseString&
+BaseString::appfmt(const char *fmt, ...)
+{
+ char buf[1];
+ va_list ap;
+ int l;
+
+ /* Figure out how long the formatted string will be. A small temporary
+ * buffer is used, because I don't trust all implementations to work
+ * when called as vsnprintf(NULL, 0, ...).
+ */
+ va_start(ap, fmt);
+ l = vsnprintf(buf, sizeof(buf), fmt, ap) + 1;
+ va_end(ap);
+ char *tmp = new char[l];
+ va_start(ap, fmt);
+ vsnprintf(tmp, l, fmt, ap);
+ va_end(ap);
+ append(tmp);
+ delete[] tmp;
+ return *this;
+}
+
+BaseString&
+BaseString::operator=(const BaseString& str)
+{
+ if (this != &str) {
+ this->assign(str);
+ }
+ return *this;
+}
+
+int
+BaseString::split(Vector<BaseString> &v,
+ const BaseString &separator,
+ int maxSize) const {
+ char *str = strdup(m_chr);
+ int i, start, len, num = 0;
+ len = strlen(str);
+ for(start = i = 0;
+ (i <= len) && ( (maxSize<0) || ((int)v.size()<=maxSize-1) );
+ i++) {
+ if(strchr(separator.c_str(), str[i]) || i == len) {
+ if(maxSize < 0 || (int)v.size() < maxSize-1)
+ str[i] = '\0';
+ v.push_back(BaseString(str+start));
+ num++;
+ start = i+1;
+ }
+ }
+ free(str);
+
+ return num;
+}
+
+ssize_t
+BaseString::indexOf(char c) {
+ char *p;
+ p = strchr(m_chr, c);
+ if(p == NULL)
+ return -1;
+ return (ssize_t)(p-m_chr);
+}
+
+ssize_t
+BaseString::lastIndexOf(char c) {
+ char *p;
+ p = strrchr(m_chr, c);
+ if(p == NULL)
+ return -1;
+ return (ssize_t)(p-m_chr);
+}
+
+BaseString
+BaseString::substr(ssize_t start, ssize_t stop) {
+ if(stop < 0)
+ stop = length();
+ ssize_t len = stop-start;
+ if(len <= 0)
+ return BaseString("");
+ BaseString s;
+ s.assign(m_chr+start, len);
+ return s;
+}
+
+static bool
+iswhite(char c) {
+ switch(c) {
+ case ' ':
+ case '\t':
+ return true;
+ default:
+ return false;
+ }
+ /* NOTREACHED */
+}
+
+char **
+BaseString::argify(const char *argv0, const char *src) {
+ Vector<char *> vargv;
+
+ if(argv0 != NULL)
+ vargv.push_back(strdup(argv0));
+
+ char *tmp = new char[strlen(src)+1];
+ char *dst = tmp;
+ const char *end = src + strlen(src);
+ /* Copy characters from src to destination, while compacting them
+ * so that all whitespace is compacted and replaced by a NUL-byte.
+ * At the same time, add pointers to strings in the vargv vector.
+ * When whitespace is detected, the characters '"' and '\' are honored,
+ * to make it possible to give arguments containing whitespace.
+ * The semantics of '"' and '\' match that of most Unix shells.
+ */
+ while(src < end && *src) {
+ /* Skip initial whitespace */
+ while(src < end && *src && iswhite(*src))
+ src++;
+
+ char *begin = dst;
+ while(src < end && *src) {
+ /* Handle '"' quotation */
+ if(*src == '"') {
+ src++;
+ while(src < end && *src && *src != '"') {
+ if(*src == '\\')
+ src++;
+ *dst++ = *src++;
+ }
+ src++;
+ if(src >= end)
+ goto end;
+ }
+
+ /* Handle '\' */
+ if(*src == '\\')
+ src++;
+ else if(iswhite(*src))
+ break;
+
+ /* Actually copy characters */
+ *dst++ = *src++;
+ }
+
+ /* Make sure the string is properly terminated */
+ *dst++ = '\0';
+ src++;
+
+ vargv.push_back(strdup(begin));
+ }
+ end:
+
+ delete[] tmp;
+ vargv.push_back(NULL);
+
+ /* Convert the C++ Vector into a C-vector of strings, suitable for
+ * calling execv().
+ */
+ char **argv = (char **)malloc(sizeof(*argv) * (vargv.size()));
+ if(argv == NULL)
+ return NULL;
+
+ for(size_t i = 0; i < vargv.size(); i++){
+ argv[i] = vargv[i];
+ }
+
+ return argv;
+}
+
+BaseString&
+BaseString::trim(const char * delim){
+ trim(m_chr, delim);
+ m_len = strlen(m_chr);
+ return * this;
+}
+
+char*
+BaseString::trim(char * str, const char * delim){
+ int len = strlen(str) - 1;
+ for(; len > 0 && strchr(delim, str[len]); len--);
+
+ int pos = 0;
+ for(; pos <= len && strchr(delim, str[pos]); pos++);
+
+ if(pos > len){
+ str[0] = 0;
+ return 0;
+ } else {
+ memmove(str, &str[pos], len - pos + 1);
+ str[len-pos+1] = 0;
+ }
+
+ return str;
+}
+
+
+#ifdef TEST_BASE_STRING
+#include <assert.h>
+
+/*
+g++ -g -Wall -o tbs -DTEST_BASE_STRING -I$NDB_TOP/include/util \
+ -I$NDB_TOP/include/portlib BaseString.cpp
+valgrind ./tbs
+*/
+
+int main()
+{
+ BaseString s("abc");
+ BaseString t(s);
+ s.assign("def");
+ t.append("123");
+ assert(s == "def");
+ assert(t == "abc123");
+ s.assign("");
+ t.assign("");
+ for (unsigned i = 0; i < 1000; i++) {
+ s.append("xyz");
+ t.assign(s);
+ assert(strlen(t.c_str()) % 3 == 0);
+ }
+
+ {
+ BaseString s(":123:abc:;:foo:");
+ Vector<BaseString> v;
+ assert(s.split(v, ":;") == 7);
+
+ assert(v[0] == "");
+ assert(v[1] == "123");
+ assert(v[2] == "abc");
+ assert(v[3] == "");
+ assert(v[4] == "");
+ assert(v[5] == "foo");
+ assert(v[6] == "");
+ }
+
+ {
+ BaseString s(":123:abc:foo:bar");
+ Vector<BaseString> v;
+ assert(s.split(v, ":;", 4) == 4);
+
+ assert(v[0] == "");
+ assert(v[1] == "123");
+ assert(v[2] == "abc");
+ assert(v[3] == "foo:bar");
+
+ BaseString n;
+ n.append(v, "()");
+ assert(n == "()123()abc()foo:bar");
+ n = "";
+ n.append(v);
+ assert(n == " 123 abc foo:bar");
+ }
+
+ {
+ assert(BaseString("hamburger").substr(4,2) == "");
+ assert(BaseString("hamburger").substr(3) == "burger");
+ assert(BaseString("hamburger").substr(4,8) == "urge");
+ assert(BaseString("smiles").substr(1,5) == "mile");
+ assert(BaseString("012345").indexOf('2') == 2);
+ assert(BaseString("hej").indexOf('X') == -1);
+ }
+
+ {
+ assert(BaseString(" 1").trim(" ") == "1");
+ assert(BaseString("1 ").trim(" ") == "1");
+ assert(BaseString(" 1 ").trim(" ") == "1");
+ assert(BaseString("abc\t\n\r kalleabc\t\r\n").trim("abc\t\r\n ") == "kalle");
+ assert(BaseString(" ").trim(" ") == "");
+ }
+ return 0;
+}
+
+#endif
diff --git a/ndb/src/common/util/File.cpp b/ndb/src/common/util/File.cpp
new file mode 100644
index 00000000000..ad72b41835d
--- /dev/null
+++ b/ndb/src/common/util/File.cpp
@@ -0,0 +1,207 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <File.hpp>
+
+#include <errno.h>
+#include <string.h>
+#include <sys/stat.h>
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+#include <unistd.h>
+#endif
+
+#include <NdbOut.hpp>
+
+//
+// PUBLIC
+//
+
+bool
+File::exists(const char* aFileName)
+{
+ bool rc = true;
+
+ struct stat stmp;
+ if (::stat(aFileName, &stmp) != 0)
+ {
+ rc = false;
+ }
+
+ /*
+ File f;
+ if (!f.open(aFileName, "r"))
+ {
+ rc = (errno == ENOENT ? false : true);
+ }
+ else
+ {
+ f.close();
+ }
+ */
+ return rc;
+}
+
+long
+File::size(FILE* f)
+{
+ long cur_pos = 0, length = 0;
+
+ cur_pos = ::ftell(f);
+ ::fseek(f, 0, SEEK_END);
+ length = ::ftell(f);
+ ::fseek(f, cur_pos, SEEK_SET); // restore original position
+
+ return length;
+}
+
+bool
+File::rename(const char* currFileName, const char* newFileName)
+{
+ return ::rename(currFileName, newFileName) == 0 ? true : false;
+}
+bool
+File::remove(const char* aFileName)
+{
+ return ::remove(aFileName) == 0 ? true : false;
+}
+
+File::File() :
+ m_file(NULL),
+ m_fileMode("r")
+{
+}
+
+File::File(const char* aFileName, const char* mode) :
+ m_file(NULL),
+ m_fileMode(mode)
+{
+ ::snprintf(m_fileName, MAX_FILE_NAME_SIZE, aFileName);
+}
+
+bool
+File::open()
+{
+ return open(m_fileName, m_fileMode);
+}
+
+bool
+File::open(const char* aFileName, const char* mode)
+{
+ if(m_fileName != aFileName){
+ /**
+ * Only copy if it's not the same string
+ */
+ ::snprintf(m_fileName, MAX_FILE_NAME_SIZE, aFileName);
+ }
+ m_fileMode = mode;
+ bool rc = true;
+ if ((m_file = ::fopen(m_fileName, m_fileMode))== NULL)
+ {
+ rc = false;
+ }
+
+ return rc;
+}
+File::~File()
+{
+ close();
+}
+
+bool
+File::remove()
+{
+ // Close the file first!
+ close();
+ return File::remove(m_fileName);
+}
+
+bool
+File::close()
+{
+ bool rc = true;
+ if (m_file != NULL)
+ {
+ ::fflush(m_file);
+ rc = (::fclose(m_file) == 0 ? true : false);
+ m_file = NULL; // Try again?
+ }
+
+ return rc;
+}
+
+int
+File::read(void* buf, size_t itemSize, size_t nitems) const
+{
+ return ::fread(buf, itemSize, nitems, m_file);
+}
+
+int
+File::readChar(char* buf, long start, long length) const
+{
+ return ::fread((void*)&buf[start], 1, length, m_file);
+}
+
+int
+File::readChar(char* buf)
+{
+ return readChar(buf, 0, strlen(buf));
+}
+
+int
+File::write(const void* buf, size_t size, size_t nitems)
+{
+ return ::fwrite(buf, size, nitems, m_file);
+}
+
+int
+File::writeChar(const char* buf, long start, long length)
+{
+ return ::fwrite((const void*)&buf[start], sizeof(char), length, m_file);
+}
+
+int
+File::writeChar(const char* buf)
+{
+ return writeChar(buf, 0, ::strlen(buf));
+}
+
+long
+File::size() const
+{
+ return File::size(m_file);
+}
+
+const char*
+File::getName() const
+{
+ return m_fileName;
+}
+
+int
+File::flush() const
+{
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ ::fflush(m_file);
+ return ::fsync(::fileno(m_file));
+#else
+ return 0;
+#endif
+}
+
+//
+// PRIVATE
+//
diff --git a/ndb/src/common/util/InputStream.cpp b/ndb/src/common/util/InputStream.cpp
new file mode 100644
index 00000000000..c52b594225d
--- /dev/null
+++ b/ndb/src/common/util/InputStream.cpp
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "InputStream.hpp"
+#include <socket_io.h>
+#include <assert.h>
+#include <string.h>
+
+FileInputStream Stdin(stdin);
+
+FileInputStream::FileInputStream(FILE * file)
+ : f(file) {
+}
+
+char*
+FileInputStream::gets(char * buf, int bufLen){
+ if(!feof(f)){
+ return fgets(buf, bufLen, f);
+ }
+ return 0;
+}
+
+SocketInputStream::SocketInputStream(NDB_SOCKET_TYPE socket,
+ unsigned readTimeout)
+ : m_socket(socket) {
+ m_timeout = readTimeout;
+}
+
+char*
+SocketInputStream::gets(char * buf, int bufLen) {
+ buf[0] = 77;
+ assert(bufLen >= 2);
+ int res = readln_socket(m_socket, m_timeout, buf, bufLen - 1);
+ if(res == -1)
+ return 0;
+ if(res == 0 && buf[0] == 77){ // select return 0
+ buf[0] = 0;
+ } else if(res == 0 && buf[0] == 0){ // only newline
+ buf[0] = '\n';
+ buf[1] = 0;
+ } else {
+ int len = strlen(buf);
+ buf[len + 1] = '\0';
+ buf[len] = '\n';
+ }
+ return buf;
+}
diff --git a/ndb/src/common/util/Makefile b/ndb/src/common/util/Makefile
new file mode 100644
index 00000000000..e400bb12d29
--- /dev/null
+++ b/ndb/src/common/util/Makefile
@@ -0,0 +1,36 @@
+include .defs.mk
+
+TYPE := util
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := general
+
+SOURCES = File.cpp md5_hash.cpp Properties.cpp socket_io.cpp \
+ SimpleProperties.cpp Parser.cpp InputStream.cpp SocketServer.cpp \
+ OutputStream.cpp NdbOut.cpp BaseString.cpp Base64.cpp \
+ NdbSqlUtil.cpp
+
+SOURCES.c = uucode.c random.c getarg.c version.c
+
+ifeq ($(NDB_OS), OSE)
+ SOURCES += NdbErrHnd.cpp
+endif
+ifeq ($(NDB_OS), OSE)
+ SOURCES += NdbErrHnd.cpp
+endif
+ifdef NDB_STRDUP
+ SOURCES.c += strdup.c
+endif
+ifdef NDB_STRLCAT
+ SOURCES.c += strlcat.c
+endif
+ifdef NDB_STRLCPY
+ SOURCES.c += strlcpy.c
+endif
+
+DIRS := testSimpleProperties
+
+include $(NDB_TOP)/Epilogue.mk
+
+testNdbSqlUtil: NdbSqlUtil.cpp
+ $(CC) -o $@ NdbSqlUtil.cpp $(CCFLAGS) -DNDB_SQL_UTIL_TEST -L$(NDB_TOP)/lib -lportlib -lgeneral
diff --git a/ndb/src/common/util/NdbErrHnd.cpp b/ndb/src/common/util/NdbErrHnd.cpp
new file mode 100644
index 00000000000..53df5d702ca
--- /dev/null
+++ b/ndb/src/common/util/NdbErrHnd.cpp
@@ -0,0 +1,493 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+
+#include <NdbOut.hpp>
+#include <NdbStdio.h>
+#include <ndb_types.h>
+
+#include "ose.h"
+#include "ose_err.h"
+#include "osetypes.h"
+
+
+#define BUFSIZE 100
+
+typedef struct {
+ char header1[BUFSIZE];
+ char header2[BUFSIZE];
+ char error_code_line[BUFSIZE];
+ char subcode_line[BUFSIZE];
+ char product_line[BUFSIZE];
+ char header_file_line[BUFSIZE];
+ char extra_line[BUFSIZE];
+ char user_called_line[BUFSIZE];
+ char current_process_id_line[BUFSIZE];
+ char current_process_name_line[BUFSIZE];
+ char file_line[BUFSIZE];
+ char line_line[BUFSIZE];
+ char err_hnd_file[BUFSIZE];
+} Error_message;
+
+char assert_line[BUFSIZE];
+char unknown_signal_line[BUFSIZE];
+char signal_number_line[BUFSIZE];
+char sender_line[BUFSIZE];
+char receiver_line[BUFSIZE];
+
+extern "C" OSBOOLEAN ndb_err_hnd(bool user_called,
+ Uint32 error_code,
+ Uint32 extra)
+{
+ static Error_message error_message;
+ bool error_handled;
+ Uint32 subcode;
+
+ char* subcode_mnemonic;
+ char* product_name;
+ char* file_name;
+
+ /*The subcode (bit 16 - 30) is extracted from error_code */
+ subcode = (error_code & 0x7fff0000) >> 16;
+
+ if (user_called) {
+ switch (subcode) {
+ case 0x0050 :
+ subcode_mnemonic= "OSE_PRH_PLS";
+ product_name= "Program Loader";
+ file_name = "prherr.h";
+ break;
+ case 0x0051 :
+ subcode_mnemonic = "OSE_PRH_START_PRH";
+ product_name= "start_prh";
+ file_name= " start_prh.c";
+ break;
+ case 0x0052 :
+ subcode_mnemonic= "OSE_PRH_ASF";
+ product_name= "Archive Server";
+ file_name = "prherr.h";
+ break;
+ case 0x0058 :
+ case 0x4058 :
+ case 0x3fff :
+ case 0x8058 :
+ subcode_mnemonic= "OSE_MMS_EBASE";
+ product_name= "MMS";
+ file_name= "mms_err.h";
+ break;
+ /*Link Handler G3***************************************/
+ case 0x0060 :
+ case 0x8060 :
+ subcode_mnemonic= "OSE_GLH_EBASE";
+ product_name= "General Link Handler";
+ file_name= "glherr.h";
+ break;
+ case 0x0064 :
+ case 0x8064 :
+ subcode_mnemonic= "OSE_GPL_EBASE";
+ product_name= "General Protocol Link Handler";
+ file_name= "gplerr.h";
+ break;
+ case 0x0066 :
+ case 0x8066 :
+ subcode_mnemonic= "OSE_UDPPDR_EBASE";
+ product_name= "UDP driver for GPL";
+ file_name= "udppdrerr.h";
+ break;
+ case 0x0067 :
+ case 0x8067 :
+ subcode_mnemonic= "OSE_SERPDR_EBASE";
+ product_name= "Serial driver for GPL";
+ file_name= "serpdrerr.h";
+ break;
+ case 0x0068 :
+ case 0x8068 :
+ subcode_mnemonic= "OSE_ETHPDR_EBASE";
+ product_name= "Ethernet driver for GPL";
+ file_name= "ethpdrerr.h";
+ break;
+ /*Link handler G4***************************************/
+ case 0x0061 :
+ subcode_mnemonic= "OSE_OTL_EBASE";
+ product_name= "OSE Transport Layer";
+ file_name= "otlerr.h";
+ break;
+ case 0x0062 :
+ subcode_mnemonic= "OSE_LALUDP_EBASE";
+ product_name= "Link Adaption Layer for UDP";
+ file_name= "header file unknown";
+ break;
+ /*Internet Utilities************************************/
+ case 0x0069 :
+ subcode_mnemonic= "OSE_TFTPD";
+ product_name= "TFTP server";
+ file_name= "inetutilerr.h";
+ break;
+ case 0x006a :
+ subcode_mnemonic= "OSE_TELUDPD";
+ product_name= "TELNET/UDP server";
+ file_name= "inetutilerr.h";
+ break;
+ case 0x006b :
+ subcode_mnemonic= "OSE_FTPD";
+ product_name= "FTP server";
+ file_name= "inetutilerr.h";
+ break;
+ case 0x006c :
+ subcode_mnemonic= "OSE_TELNETD";
+ product_name= "TELNET server";
+ file_name= "inetutilerr.h";
+ break;
+ case 0x006d :
+ subcode_mnemonic= "OSE_SURFER";
+ product_name= "OSE System Surfer";
+ file_name= "inetutilerr.h";
+ break;
+ case 0x006e :
+ subcode_mnemonic= "OSE_BOOTP";
+ product_name= "BOOTP client";
+ file_name= "inetutilerr.h";
+ break;
+ case 0x006f :
+ switch((error_code & 0x0000f000)){
+ case 0x00000000 :
+ subcode_mnemonic= "OSE_RES";
+ product_name= "DNS resolver";
+ file_name= "inetutilerr.h";
+ break;
+ case 0x00001000 :
+ subcode_mnemonic= "OSE_DHCPC";
+ product_name= "DHCP client";
+ file_name= "inetutilerr.h";
+ break;
+ case 0x00002000 :
+ subcode_mnemonic= "OSE_FTP";
+ product_name= "FTP client";
+ file_name= "inetutilerr.h";
+ break;
+ default :
+ subcode_mnemonic= "Unknown error";
+ product_name= "unknown product";
+ file_name = "header file unknown";
+ break;
+ }
+ break;
+ case 0x00c2 :
+ subcode_mnemonic= "OSE_DNS";
+ product_name= "DNS server";
+ file_name= "dns_err.h";
+ break;
+ /*INET**************************/
+ case 0x0070 :
+ subcode_mnemonic= "INET_ERRBASE";
+ product_name= "Internet Protocols (INET)";
+ file_name= "ineterr.h";
+ break;
+ case 0x0071 :
+ subcode_mnemonic= "WEBS_ERRBASE";
+ product_name= "Web Server (WEBS)";
+ file_name= "webserr.h";
+ break;
+ case 0x0072 :
+ subcode_mnemonic= "SNMP";
+ product_name= "SNMP";
+ file_name= "header file unknown";
+ break;
+ case 0x0073 :
+ subcode_mnemonic= "STP_BRIDGE";
+ product_name= "STP bridge";
+ file_name= "header file unknown";
+ break;
+ case 0x0200 :
+ case 0x0201 :
+ case 0x0202 :
+ case 0x0203 :
+ case 0x0204 :
+ case 0x0205 :
+ case 0x0206 :
+ case 0x0207 :
+ case 0x0208 :
+ case 0x0209 :
+ case 0x020a :
+ case 0x020b :
+ case 0x020c :
+ case 0x020d :
+ case 0x020e :
+ case 0x020f :
+ subcode_mnemonic = "INETINIT_ERR_BASE";
+ product_name = "INET";
+ file_name = "startinet.c";
+ break;
+ /*Miscellanous******************************************/
+ case 0x0082 :
+ subcode_mnemonic= "OSE_HEAP_EBASE";
+ product_name= "Heap Manager";
+ file_name= "heap_err.h";
+ break;
+ case 0x0088 :
+ subcode_mnemonic= "OSE_BSP";
+ product_name= "Board Support Package";
+ file_name= "bsperr.h";
+ break;
+ case 0x008a :
+ subcode_mnemonic= "OSE_TOSV_EBASE";
+ product_name= "Time Out Server";
+ file_name= "tosverr.h";
+ break;
+ case 0x008b :
+ subcode_mnemonic= "OSE_RTC_EBASE";
+ product_name= "Real Time Clock";
+ file_name= "rtcerr.h";
+ break;
+ case 0x008d :
+ case 0x808d :
+ subcode_mnemonic= "OSENS_ERR_BASE";
+ product_name= "Name Server";
+ file_name= "osens_err.h";
+ break;
+ case 0x008e :
+ subcode_mnemonic= "PMD_ERR_BASE";
+ product_name= "Post Mortem Dump";
+ file_name= "pmderr.h";
+ break;
+ /*Embedded File System***********************************/
+ case 0x0090 :
+ subcode_mnemonic= "OSE_EFS_COMMON";
+ product_name= "EFS common";
+ file_name= "efs_err.h";
+ break;
+ case 0x0091 :
+ subcode_mnemonic= "OSE_EFS_FLIB";
+ product_name= "EFS function library";
+ file_name= "efs_err.h";
+ break;
+ case 0x0092 :
+ subcode_mnemonic= "OSE_EFS_SERDD";
+ product_name= "EFS serdd";
+ file_name= "efs_err.h";
+ break;
+ case 0x0093 :
+ subcode_mnemonic= "OSE_EFS_SHELL";
+ product_name= "OSE shell";
+ file_name= "efs_err.h";
+ break;
+ case 0x0094 :
+ subcode_mnemonic= "OSE_EFS_STARTEFS";
+ product_name= "EFS startefs.c";
+ file_name= "efs_err.h";
+ break;
+ /*Debugger related***************************************/
+ case 0x00a0 :
+ subcode_mnemonic= "DBGSERVER_ERR_BASE";
+ product_name= "Debug server for Illuminator";
+ file_name= "degservererr.h";
+ break;
+ case 0x00b2 :
+ subcode_mnemonic= "OSE_MDM";
+ product_name= "Multi INDRT monitor";
+ file_name= "header file unknown";
+ break;
+ /*Miscellanous*******************************************/
+ case 0x00c0 :
+ subcode_mnemonic= "OSE_POTS_EBASE";
+ product_name= "POTS tutorial example";
+ file_name= "pots_err.h";
+ break;
+ case 0x00c1 :
+ subcode_mnemonic= "OSE_PTH_ECODE_BASE";
+ product_name= "Pthreads";
+ file_name= "pthread_err.h";
+ break;
+ case 0x00c3 :
+ subcode_mnemonic= "OSE_NTP_EBASE";
+ product_name= "OSE NTP/SNTP";
+ file_name= "ntp_err.h";
+ break;
+ case 0x00c4 :
+ subcode_mnemonic= "TRILLIUM_BASE";
+ product_name= "Trillium OSE port";
+ file_name= "sk_ss.c";
+ break;
+ case 0x00c5 :
+ subcode_mnemonic= "OSE_OSECPP_EBASE";
+ product_name= "C++ Support with libosecpp.a";
+ file_name= "cpp_err.h";
+ break;
+ case 0x00c6 :
+ subcode_mnemonic= "OSE_RIP_ERR_BASE";
+ product_name= "OSE RIP";
+ file_name= "oserip.h";
+ break;
+ /*Unknown error_code*************************************/
+ default :
+ subcode_mnemonic= "Unknown error";
+ product_name= "unknown product";
+ file_name = "header file unknown";
+ break;
+ }
+ } else {
+ /* user_called = 0, i.e. reported by the kernel */
+ subcode_mnemonic= "OSE_KRN";
+ product_name= "Kernel";
+ file_name = "ose_err.h";
+ }
+
+ snprintf (error_message.header1,
+ BUFSIZE,
+ "This is the OSE Example System Error handler\r\n");
+
+ snprintf (error_message.err_hnd_file,
+ BUFSIZE,
+ "located in: " __FILE__ "\r\n");
+
+ snprintf (error_message.header2,
+ BUFSIZE,
+ "An Error has been reported:\r\n");
+
+ if (user_called == (OSBOOLEAN) 0 ) {
+ snprintf(error_message.user_called_line,
+ BUFSIZE,
+ "user_called: 0x%x (Error detected by the kernel)\r\n",
+ user_called);
+ }
+ else {
+ snprintf(error_message.user_called_line,
+ BUFSIZE,
+ "user_called: 0x%x (Error detected by an application)\r\n",
+ user_called);
+ }
+
+ snprintf (error_message.error_code_line,
+ BUFSIZE,
+ "error code: 0x%08x\r\n",
+ error_code);
+
+ snprintf (error_message.subcode_line,
+ BUFSIZE,
+ " subcode: %s (0x%08x)\r\n",
+ subcode_mnemonic,
+ ( subcode << 16));
+
+ snprintf (error_message.product_line,
+ BUFSIZE,
+ " product: %s\r\n",
+ product_name);
+
+ snprintf (error_message.header_file_line,
+ BUFSIZE,
+ " header file: %s\r\n",
+ file_name);
+
+ snprintf (error_message.extra_line,
+ BUFSIZE,
+ "extra: 0x%08x\r\n",
+ extra);
+
+ if (error_code != OSE_ENO_KERN_SPACE || user_called){
+ struct OS_pcb *pcb = get_pcb(current_process());
+ const char *process_name = &pcb->strings[pcb->name];
+
+ snprintf(error_message.current_process_id_line,
+ BUFSIZE,
+ "Current Process: 0x%08x\r\n",
+ current_process());
+
+ snprintf(error_message.current_process_name_line,
+ BUFSIZE,
+ "Process Name: %s\r\n",
+ process_name);
+
+ snprintf(error_message.file_line,
+ BUFSIZE,
+ "File: %s\r\n",
+ &pcb->strings[pcb->file]);
+
+ snprintf(error_message.line_line,
+ BUFSIZE,
+ "Line: %d\r\n",
+ pcb->line);
+
+ free_buf((union SIGNAL **)&pcb);
+ }
+
+ if ( !(((error_code & OSE_EFATAL_MASK) != 0) && (user_called == 0))){
+ /* If the error is reported by the kernel and the fatal flag is set,
+ * dbgprintf can't be trusted */
+ ndbout << error_message.header1;
+ ndbout << error_message.err_hnd_file;
+ ndbout << error_message.header2;
+ ndbout << error_message.user_called_line;
+ ndbout << error_message.error_code_line;
+ ndbout << error_message.subcode_line;
+ ndbout << error_message.product_line;
+ ndbout << error_message.header_file_line;
+ ndbout << error_message.extra_line;
+ ndbout << error_message.current_process_id_line;
+ ndbout << error_message.current_process_name_line;
+ ndbout << error_message.file_line;
+ ndbout << error_message.line_line;
+ ndbout << endl;
+ }
+
+ if(user_called){
+ switch (error_code) {
+ /* Check for assertion failure (see oseassert.h and assert.c). */
+ case (OSERRCODE) 0xffffffff:
+ {
+ if(extra != 0){
+ char *expr = ((char **)extra)[0];
+ char *file = ((char **)extra)[1];
+ unsigned line = ((unsigned *)extra)[2];
+ snprintf(assert_line, BUFSIZE, "Assertion Failed: %s:%u: %s\r\n", file, line, expr);
+ ndbout << assert_line;
+ }
+ }
+ /* Check for unknown signal */
+ case (OSERRCODE) 0xfffffffe:
+ {
+ union SIGNAL *sig = (union SIGNAL *)extra;
+ SIGSELECT signo = *(SIGSELECT*)sig;
+ PROCESS rcv_ = current_process();
+ PROCESS snd_ = sender(&sig);
+ struct OS_pcb *rcv = get_pcb(rcv_);
+ const char *rcv_name = &rcv->strings[rcv->name];
+ struct OS_pcb *snd = get_pcb(snd_);
+ const char *snd_name = &snd->strings[snd->name];
+ snprintf(unknown_signal_line, BUFSIZE,
+ "Unknown Signal Received\r\n");
+ snprintf(unknown_signal_line, BUFSIZE,
+ "Signal Number: 0x%08lx\r\n", signo);
+ snprintf(unknown_signal_line, BUFSIZE,
+ "Sending Process: 0x%08lx (%s))\r\n", snd_, snd_name);
+ snprintf(unknown_signal_line, BUFSIZE,
+ "Receiving Process: 0x%08lx (%s))\r\n", rcv_, rcv_name);
+ free_buf((union SIGNAL **)&rcv);
+ free_buf((union SIGNAL **)&snd); }
+ ndbout << unknown_signal_line;
+ ndbout << signal_number_line;
+ ndbout << sender_line;
+ ndbout << receiver_line;
+ } /* switch */
+ } /* if */
+
+ /* Zero means the error has not been fixed by the error handler. */
+ error_handled = 0;
+ return error_handled;
+}
+
+#endif
diff --git a/ndb/src/common/util/NdbOut.cpp b/ndb/src/common/util/NdbOut.cpp
new file mode 100644
index 00000000000..2624bfa04bd
--- /dev/null
+++ b/ndb/src/common/util/NdbOut.cpp
@@ -0,0 +1,175 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbOut.hpp"
+#include <NdbStdio.h>
+#include <stdarg.h>
+#include <NdbUnistd.h>
+#include <string.h>
+#include <OutputStream.hpp>
+
+static FileOutputStream ndbouts_fileoutputstream(stdout);
+NdbOut ndbout(ndbouts_fileoutputstream);
+
+static const char * fms[] = {
+ "%d", "0x%02x", // Int8
+ "%u", "0x%02x", // Uint8
+ "%d", "0x%04x", // Int16
+ "%u", "0x%04x", // Uint16
+ "%d", "0x%08x", // Int32
+ "%u", "0x%08x", // Uint32
+ "%lld", "0x%016llx", // Int64
+ "%llu", "0x%016llx" // Uint64
+ "%llu", "0x%016llx" // UintPtr
+};
+
+NdbOut&
+NdbOut::operator<<(Int8 v) { m_out->print(fms[0+isHex],(int)v); return *this;}
+NdbOut&
+NdbOut::operator<<(Uint8 v) { m_out->print(fms[2+isHex],(int)v); return *this;}
+NdbOut&
+NdbOut::operator<<(Int16 v) { m_out->print(fms[4+isHex],(int)v); return *this;}
+NdbOut&
+NdbOut::operator<<(Uint16 v) { m_out->print(fms[6+isHex],(int)v); return *this;}
+NdbOut&
+NdbOut::operator<<(Int32 v) { m_out->print(fms[8+isHex], v); return *this;}
+NdbOut&
+NdbOut::operator<<(Uint32 v) { m_out->print(fms[10+isHex], v); return *this;}
+NdbOut&
+NdbOut::operator<<(Int64 v) { m_out->print(fms[12+isHex], v); return *this;}
+NdbOut&
+NdbOut::operator<<(Uint64 v) { m_out->print(fms[14+isHex], v); return *this;}
+NdbOut&
+NdbOut::operator<<(unsigned long int v) { return *this << (Uint64) v; }
+
+NdbOut&
+NdbOut::operator<<(const char* val){ m_out->print("%s", val); return * this; }
+NdbOut&
+NdbOut::operator<<(const void* val){ m_out->print("%p", val); return * this; }
+NdbOut&
+NdbOut::operator<<(BaseString &val){ return *this << val.c_str(); }
+
+NdbOut&
+NdbOut::operator<<(float val){ m_out->print("%f", (double)val); return * this;}
+NdbOut&
+NdbOut::operator<<(double val){ m_out->print("%f", val); return * this; }
+
+NdbOut& NdbOut::endline()
+{
+ isHex = 0; // Reset hex to normal, if user forgot this
+ m_out->println("");
+ m_out->flush();
+ return *this;
+}
+
+NdbOut& NdbOut::flushline()
+{
+ m_out->flush();
+ return *this;
+}
+
+NdbOut& NdbOut::setHexFormat(int _format)
+{
+ isHex = (_format == 0 ? 0 : 1);
+ return *this;
+}
+
+NdbOut::NdbOut(OutputStream & out)
+ : m_out(& out)
+{
+ isHex = 0;
+}
+
+NdbOut::~NdbOut()
+{
+}
+
+void
+NdbOut::print(const char * fmt, ...){
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ ndbout << buf;
+ va_end(ap);
+}
+
+void
+NdbOut::println(const char * fmt, ...){
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ ndbout << buf << endl;
+ va_end(ap);
+}
+
+extern "C"
+void
+ndbout_c(const char * fmt, ...){
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ ndbout << buf << endl;
+ va_end(ap);
+}
+
+FilteredNdbOut::FilteredNdbOut(OutputStream & out,
+ int threshold, int level)
+ : NdbOut(out) {
+ m_level = level;
+ m_threshold = threshold;
+ m_org = &out;
+ m_null = new NullOutputStream();
+ setLevel(level);
+}
+
+FilteredNdbOut::~FilteredNdbOut(){
+ delete m_null;
+}
+
+void
+FilteredNdbOut::setLevel(int i){
+ m_level = i;
+ if(m_level >= m_threshold){
+ m_out = m_org;
+ } else {
+ m_out = m_null;
+ }
+}
+
+void
+FilteredNdbOut::setThreshold(int i){
+ m_threshold = i;
+ setLevel(m_level);
+}
+
+int
+FilteredNdbOut::getLevel() const {
+ return m_level;
+}
+int
+FilteredNdbOut::getThreshold() const {
+ return m_threshold;
+}
+
diff --git a/ndb/src/common/util/NdbSqlUtil.cpp b/ndb/src/common/util/NdbSqlUtil.cpp
new file mode 100644
index 00000000000..dba7012cc0f
--- /dev/null
+++ b/ndb/src/common/util/NdbSqlUtil.cpp
@@ -0,0 +1,351 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbSqlUtil.hpp>
+
+int
+NdbSqlUtil::char_compare(const char* s1, unsigned n1,
+ const char* s2, unsigned n2, bool padded)
+{
+ int c1 = 0;
+ int c2 = 0;
+ unsigned i = 0;
+ while (i < n1 || i < n2) {
+ c1 = i < n1 ? s1[i] : padded ? 0x20 : 0;
+ c2 = i < n2 ? s2[i] : padded ? 0x20 : 0;
+ if (c1 != c2)
+ break;
+ i++;
+ }
+ return c1 - c2;
+}
+
+bool
+NdbSqlUtil::char_like(const char* s1, unsigned n1,
+ const char* s2, unsigned n2, bool padded)
+{
+ int c1 = 0;
+ int c2 = 0;
+ unsigned i1 = 0;
+ unsigned i2 = 0;
+ while (i1 < n1 || i2 < n2) {
+ c1 = i1 < n1 ? s1[i1] : padded ? 0x20 : 0;
+ c2 = i2 < n2 ? s2[i2] : padded ? 0x20 : 0;
+ if (c2 == '%') {
+ while (i2 + 1 < n2 && s2[i2 + 1] == '%') {
+ i2++;
+ }
+ unsigned m = 0;
+ while (m <= n1 - i1) {
+ if (char_like(s1 + i1 + m, n1 -i1 - m,
+ s2 + i2 + 1, n2 - i2 - 1, padded))
+ return true;
+ m++;
+ }
+ return false;
+ }
+ if (c2 == '_') {
+ if (c1 == 0)
+ return false;
+ } else {
+ if (c1 != c2)
+ return false;
+ }
+ i1++;
+ i2++;
+ }
+ return i1 == n2 && i2 == n2;
+}
+
+/**
+ * Data types.
+ */
+
+const NdbSqlUtil::Type
+NdbSqlUtil::m_typeList[] = {
+ {
+ Type::Undefined,
+ NULL
+ },
+ {
+ Type::Tinyint,
+ cmpTinyint
+ },
+ {
+ Type::Tinyunsigned,
+ cmpTinyunsigned
+ },
+ {
+ Type::Smallint,
+ cmpSmallint
+ },
+ {
+ Type::Smallunsigned,
+ cmpSmallunsigned
+ },
+ {
+ Type::Mediumint,
+ NULL // cmpMediumint
+ },
+ {
+ Type::Mediumunsigned,
+ NULL // cmpMediumunsigned
+ },
+ {
+ Type::Int,
+ cmpInt
+ },
+ {
+ Type::Unsigned,
+ cmpUnsigned
+ },
+ {
+ Type::Bigint,
+ cmpBigint
+ },
+ {
+ Type::Bigunsigned,
+ cmpBigunsigned
+ },
+ {
+ Type::Float,
+ cmpFloat
+ },
+ {
+ Type::Double,
+ cmpDouble
+ },
+ {
+ Type::Decimal,
+ NULL // cmpDecimal
+ },
+ {
+ Type::Char,
+ cmpChar
+ },
+ {
+ Type::Varchar,
+ cmpVarchar
+ },
+ {
+ Type::Binary,
+ NULL // cmpBinary
+ },
+ {
+ Type::Varbinary,
+ NULL // cmpVarbinary
+ },
+ {
+ Type::Datetime,
+ cmpDatetime
+ },
+ {
+ Type::Timespec,
+ NULL // cmpTimespec
+ }
+};
+
+const NdbSqlUtil::Type&
+NdbSqlUtil::type(Uint32 typeId)
+{
+ if (typeId < sizeof(m_typeList) / sizeof(m_typeList[0]) &&
+ m_typeList[typeId].m_typeId != Type::Undefined) {
+ return m_typeList[typeId];
+ }
+ return m_typeList[Type::Undefined];
+}
+
+// compare
+
+int
+NdbSqlUtil::cmpTinyint(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Tinyint, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpTinyunsigned(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Tinyunsigned, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpSmallint(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Smallint, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpSmallunsigned(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Smallunsigned, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpMediumint(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Mediumint, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpMediumunsigned(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Mediumunsigned, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpInt(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Int, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpUnsigned(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Unsigned, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpBigint(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Bigint, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpBigunsigned(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Bigunsigned, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpFloat(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Float, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpDouble(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Double, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpDecimal(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Decimal, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpChar(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Char, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpVarchar(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Varchar, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpBinary(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Binary, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpVarbinary(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Varbinary, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpDatetime(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Datetime, p1, p2, full, size);
+}
+
+int
+NdbSqlUtil::cmpTimespec(const Uint32* p1, const Uint32* p2, Uint32 full, Uint32 size)
+{
+ return cmp(Type::Timespec, p1, p2, full, size);
+}
+
+#ifdef NDB_SQL_UTIL_TEST
+
+#include <assert.h>
+#include <NdbTick.h>
+#include <NdbOut.hpp>
+
+struct Testcase {
+ int op; // 1=compare 2=like
+ int res;
+ const char* s1;
+ const char* s2;
+ int pad;
+};
+const Testcase testcase[] = {
+ { 2, 1, "abc", "abc", 0 },
+ { 2, 1, "abc", "abc%", 0 },
+ { 2, 1, "abcdef", "abc%", 0 },
+ { 2, 1, "abcdefabcdefabcdef", "abc%", 0 },
+ { 2, 1, "abcdefabcdefabcdef", "abc%f", 0 },
+ { 2, 0, "abcdefabcdefabcdef", "abc%z", 0 },
+ { 2, 1, "abcdefabcdefabcdef", "%f", 0 },
+ { 2, 1, "abcdef", "a%b%c%d%e%f", 0 },
+ { 0, 0, 0, 0 }
+};
+
+int
+main(int argc, char** argv)
+{
+ unsigned count = argc > 1 ? atoi(argv[1]) : 1000000;
+ ndbout_c("count = %u", count);
+ assert(count != 0);
+ for (const Testcase* t = testcase; t->s1 != 0; t++) {
+ ndbout_c("%d = '%s' %s '%s' pad=%d",
+ t->res, t->s1, t->op == 1 ? "comp" : "like", t->s2);
+ NDB_TICKS x1 = NdbTick_CurrentMillisecond();
+ unsigned n1 = strlen(t->s1);
+ unsigned n2 = strlen(t->s2);
+ for (unsigned i = 0; i < count; i++) {
+ if (t->op == 1) {
+ int res = NdbSqlUtil::char_compare(t->s1, n1, t->s2, n2, t->pad);
+ assert(res == t->res);
+ continue;
+ }
+ if (t->op == 2) {
+ int res = NdbSqlUtil::char_like(t->s1, n1, t->s2, n2, t->pad);
+ assert(res == t->res);
+ continue;
+ }
+ assert(false);
+ }
+ NDB_TICKS x2 = NdbTick_CurrentMillisecond();
+ if (x2 < x1)
+ x2 = x1;
+ double usec = 1000000.0 * double(x2 - x1) / double(count);
+ ndbout_c("time %.0f usec per call", usec);
+ }
+ // quick check
+ for (unsigned i = 0; i < sizeof(m_typeList) / sizeof(m_typeList[0]); i++) {
+ const NdbSqlUtil::Type& t = m_typeList[i];
+ assert(t.m_typeId == i);
+ }
+ return 0;
+}
+
+#endif
diff --git a/ndb/src/common/util/OutputStream.cpp b/ndb/src/common/util/OutputStream.cpp
new file mode 100644
index 00000000000..1143fe00fd1
--- /dev/null
+++ b/ndb/src/common/util/OutputStream.cpp
@@ -0,0 +1,98 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <OutputStream.hpp>
+#include <stdarg.h>
+#include <socket_io.h>
+
+FileOutputStream::FileOutputStream(FILE * file){
+ f = file;
+}
+
+int
+FileOutputStream::print(const char * fmt, ...){
+ va_list ap;
+ va_start(ap, fmt);
+ const int ret = vfprintf(f, fmt, ap);
+ va_end(ap);
+ return ret;
+}
+
+int
+FileOutputStream::println(const char * fmt, ...){
+ va_list ap;
+ va_start(ap, fmt);
+ const int ret = vfprintf(f, fmt, ap);
+ va_end(ap);
+ return ret + fprintf(f, "\n");
+}
+
+SocketOutputStream::SocketOutputStream(NDB_SOCKET_TYPE socket,
+ unsigned timeout){
+ m_socket = socket;
+ m_timeout = timeout;
+}
+
+int
+SocketOutputStream::print(const char * fmt, ...){
+ va_list ap;
+ va_start(ap, fmt);
+ const int ret = vprint_socket(m_socket, m_timeout, fmt, ap);
+ va_end(ap);
+ return ret;
+}
+int
+SocketOutputStream::println(const char * fmt, ...){
+ va_list ap;
+ va_start(ap, fmt);
+ const int ret = vprintln_socket(m_socket, m_timeout, fmt, ap);
+ va_end(ap);
+ return ret;
+}
+
+#ifdef NDB_SOFTOSE
+#include <dbgprintf.h>
+int
+SoftOseOutputStream::print(const char * fmt, ...){
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ else
+ buf[0] = 0;
+ va_end(ap);
+ dbgprintf(buf);
+}
+
+int
+SoftOseOutputStream::println(const char * fmt, ...){
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ else
+ buf[0] = 0;
+ va_end(ap);
+
+ strcat(buf, "\n\r");
+ dbgprintf(buf);
+}
+#endif
diff --git a/ndb/src/common/util/Parser.cpp b/ndb/src/common/util/Parser.cpp
new file mode 100644
index 00000000000..d5c23fe14c1
--- /dev/null
+++ b/ndb/src/common/util/Parser.cpp
@@ -0,0 +1,349 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "Parser.hpp"
+#include <stdio.h>
+#include <NdbOut.hpp>
+#include <string.h>
+#include <Properties.hpp>
+#include <assert.h>
+#include <Base64.hpp>
+
+#define DEBUG(x) ndbout << x << endl;
+
+static void trim(char * str);
+
+class ParseInputStream : public InputStream {
+public:
+ ParseInputStream(InputStream & in, bool trim = true, char eofComment = '#');
+
+ char* gets(char * buf, int bufLen);
+ void push_back(const char *);
+private:
+ InputStream & in;
+ char * buffer;
+};
+
+ParseInputStream::ParseInputStream(InputStream & _in,
+ bool /* unused */,
+ char /* unused */)
+ : in(_in){
+ buffer = 0;
+}
+
+char*
+ParseInputStream::gets(char * buf, int bufLen){
+ if(buffer != 0){
+ strncpy(buf, buffer, bufLen);
+ free(buffer);
+ buffer = 0;
+ return buf;
+ }
+ char *t = in.gets(buf, bufLen);
+ return t;
+}
+
+void
+ParseInputStream::push_back(const char * str){
+ if(buffer != 0)
+ abort();
+ buffer = strdup(str);
+}
+
+ParserImpl::ParserImpl(const DummyRow * rows, InputStream & in,
+ bool b_cmd, bool b_empty, bool b_iarg)
+ : m_rows(rows), input(* new ParseInputStream(in))
+{
+ m_breakOnCmd = b_cmd;
+ m_breakOnEmpty = b_empty;
+ m_breakOnInvalidArg = b_iarg;
+}
+
+ParserImpl::~ParserImpl(){
+ delete & input;
+}
+
+static
+bool
+Empty(const char * str){
+ if(str == 0)
+ return true;
+ const int len = strlen(str);
+ if(len == 0)
+ return false;
+ for(int i = 0; i<len; i++)
+ if(str[i] != ' ' && str[i] != '\t' && str[i] != '\n')
+ return false;
+ return true;
+}
+
+static
+bool
+Eof(const char * str) { return str == 0;}
+
+static
+void
+trim(char * str){
+ if(str == NULL)
+ return;
+ int len = strlen(str);
+ for(len--; str[len] == '\n' || str[len] == ' ' || str[len] == '\t'; len--)
+ str[len] = 0;
+
+ int pos = 0;
+ while(str[pos] == ' ' || str[pos] == '\t')
+ pos++;
+
+ if(str[pos] == '\"' && str[len] == '\"') {
+ pos++;
+ str[len] = 0;
+ len--;
+ }
+
+ memmove(str, &str[pos], len - pos + 2);
+}
+
+static
+bool
+split(char * buf, char ** name, char ** value){
+
+ * value = strchr(buf, ':');
+ if(* value == 0)
+ * value = strchr(buf, '=');
+
+
+ if(* value == 0){
+ return false;
+ }
+ (* value)[0] = 0;
+ * value = (* value + 1);
+ * name = buf;
+
+ trim(* name);
+ trim(* value);
+
+ return true;
+}
+
+bool
+ParserImpl::run(Context * ctx, const class Properties ** pDst,
+ volatile bool * stop) const {
+ * pDst = 0;
+ bool ownStop = false;
+ if(stop == 0)
+ stop = &ownStop;
+
+ ctx->m_aliasUsed.clear();
+
+ const unsigned sz = sizeof(ctx->m_tokenBuffer);
+ ctx->m_currentToken = input.gets(ctx->m_tokenBuffer, sz);
+ if(Eof(ctx->m_currentToken)){
+ ctx->m_status = Parser<Dummy>::Eof;
+ return false;
+ }
+
+ if(ctx->m_currentToken[0] == 0){
+ ctx->m_status = Parser<Dummy>::NoLine;
+ return false;
+ }
+
+ if(Empty(ctx->m_currentToken)){
+ ctx->m_status = Parser<Dummy>::EmptyLine;
+ return false;
+ }
+
+ trim(ctx->m_currentToken);
+ ctx->m_currentCmd = matchCommand(ctx, ctx->m_currentToken, m_rows);
+ if(ctx->m_currentCmd == 0){
+ ctx->m_status = Parser<Dummy>::UnknownCommand;
+ return false;
+ }
+
+ Properties * p = new Properties();
+
+ bool invalidArgument = false;
+ ctx->m_currentToken = input.gets(ctx->m_tokenBuffer, sz);
+
+ while((! * stop) &&
+ !Eof(ctx->m_currentToken) &&
+ !Empty(ctx->m_currentToken)){
+ if(ctx->m_currentToken[0] != 0){
+ trim(ctx->m_currentToken);
+ if(!parseArg(ctx, ctx->m_currentToken, ctx->m_currentCmd + 1, p)){
+ delete p;
+ invalidArgument = true;
+ break;
+ }
+ }
+ ctx->m_currentToken = input.gets(ctx->m_tokenBuffer, sz);
+ }
+
+ if(invalidArgument){
+ char buf[sz];
+ char * tmp;
+ if(!m_breakOnInvalidArg){
+ do {
+ tmp = input.gets(buf, sz);
+ } while((! * stop) && !Eof(tmp) && !Empty(tmp));
+ }
+ return false;
+ }
+
+ if(* stop){
+ delete p;
+ ctx->m_status = Parser<Dummy>::ExternalStop;
+ return false;
+ }
+
+ if(!checkMandatory(ctx, p)){
+ ctx->m_status = Parser<Dummy>::MissingMandatoryArgument;
+ delete p;
+ return false;
+ }
+
+ /**
+ * Add alias to properties
+ */
+ for(unsigned i = 0; i<ctx->m_aliasUsed.size(); i++){
+ const ParserRow<Dummy> * alias = ctx->m_aliasUsed[i];
+ Properties tmp;
+ tmp.put("name", alias->name);
+ tmp.put("realName", alias->realName);
+ p->put("$ALIAS", i, &tmp);
+ }
+ p->put("$ALIAS", ctx->m_aliasUsed.size());
+
+ ctx->m_status = Parser<Dummy>::Ok;
+ * pDst = p;
+ return true;
+}
+
+const ParserImpl::DummyRow*
+ParserImpl::matchCommand(Context* ctx, const char* buf, const DummyRow rows[]){
+ const char * name = buf;
+ const DummyRow * tmp = &rows[0];
+ while(tmp->name != 0 && name != 0){
+ if(strcmp(tmp->name, name) == 0){
+ if(tmp->type == DummyRow::Cmd)
+ return tmp;
+ if(tmp->type == DummyRow::CmdAlias){
+ if(ctx != 0)
+ ctx->m_aliasUsed.push_back(tmp);
+ name = tmp->realName;
+ tmp = &rows[0];
+ continue;
+ }
+ }
+ tmp++;
+ }
+ return 0;
+}
+
+const ParserImpl::DummyRow*
+ParserImpl::matchArg(Context* ctx, const char * buf, const DummyRow rows[]){
+ const char * name = buf;
+ const DummyRow * tmp = &rows[0];
+ while(tmp->name != 0){
+ const DummyRow::Type t = tmp->type;
+ if(t != DummyRow::Arg && t != DummyRow::ArgAlias && t !=DummyRow::CmdAlias)
+ break;
+ if(t !=DummyRow::CmdAlias && strcmp(tmp->name, name) == 0){
+ if(tmp->type == DummyRow::Arg){
+ return tmp;
+ }
+ if(tmp->type == DummyRow::ArgAlias){
+ if(ctx != 0)
+ ctx->m_aliasUsed.push_back(tmp);
+ name = tmp->realName;
+ tmp = &rows[0];
+ continue;
+ }
+ }
+ tmp++;
+ }
+ return 0;
+}
+
+bool
+ParserImpl::parseArg(Context * ctx,
+ char * buf,
+ const DummyRow * rows,
+ Properties * p){
+ char * name;
+ char * value;
+ if(!split(buf, &name, &value)){
+ ctx->m_status = Parser<Dummy>::InvalidArgumentFormat;
+ return false;
+ }
+ const DummyRow * arg = matchArg(ctx, name, rows);
+ if(arg == 0){
+ ctx->m_status = Parser<Dummy>::UnknownArgument;
+ return false;
+ }
+
+ switch(arg->argType){
+ case DummyRow::String:
+ if(p->put(arg->name, value))
+ return true;
+ break;
+ case DummyRow::Int:{
+ Uint32 i;
+ int c = sscanf(value, "%u", &i);
+ if(c != 1){
+ ctx->m_status = Parser<Dummy>::TypeMismatch;
+ return false;
+ }
+ if(p->put(arg->name, i))
+ return true;
+ break;
+ }
+
+ case DummyRow::Properties: {
+ Properties *sp = new Properties();
+ BaseString v(value);
+ UtilBuffer b;
+ base64_decode(v, b);
+ sp->unpack((const Uint32 *)b.get_data(), b.length());
+ break;
+ }
+ default:
+ ctx->m_status = Parser<Dummy>::UnknownArgumentType;
+ return false;
+ }
+ if(p->getPropertiesErrno() == E_PROPERTIES_ELEMENT_ALREADY_EXISTS){
+ ctx->m_status = Parser<Dummy>::ArgumentGivenTwice;
+ return false;
+ }
+
+ abort();
+}
+
+bool
+ParserImpl::checkMandatory(Context* ctx, const Properties* props){
+ const DummyRow * tmp = &ctx->m_currentCmd[1];
+ while(tmp->name != 0 && tmp->type == DummyRow::Arg){
+ if(tmp->argRequired == ParserRow<Dummy>::Mandatory &&
+ !props->contains(tmp->name)){
+ ctx->m_status = Parser<Dummy>::MissingMandatoryArgument;
+ ctx->m_currentArg = tmp;
+ return false;
+ }
+ tmp++;
+ }
+ return true;
+}
+
diff --git a/ndb/src/common/util/Properties.cpp b/ndb/src/common/util/Properties.cpp
new file mode 100644
index 00000000000..4841d6e5e9e
--- /dev/null
+++ b/ndb/src/common/util/Properties.cpp
@@ -0,0 +1,1019 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <Properties.hpp>
+
+#include <NdbTCP.h>
+#include <string.h>
+#include <stdlib.h>
+#include <assert.h>
+#include <NdbString.h>
+
+#include <NdbOut.hpp>
+
+/**
+ * Note has to be a multiple of 4 bytes
+ */
+const char Properties::version[] = { 2, 0, 0, 1, 1, 1, 1, 4 };
+
+/**
+ * PropertyImpl
+ */
+struct PropertyImpl{
+ PropertiesType valueType;
+ const char * name;
+ void * value;
+
+ ~PropertyImpl();
+ PropertyImpl(const char * name, Uint32 value);
+ PropertyImpl(const char * name, const char * value);
+ PropertyImpl(const char * name, const Properties * value);
+
+ static PropertyImpl * copyPropertyImpl(const PropertyImpl &);
+};
+
+/**
+ * PropertiesImpl
+ */
+class PropertiesImpl {
+ PropertiesImpl(const PropertiesImpl &); // Not implemented
+ PropertiesImpl& operator=(const PropertiesImpl&); // Not implemented
+public:
+ PropertiesImpl(Properties *);
+ PropertiesImpl(Properties *, const PropertiesImpl &);
+ ~PropertiesImpl();
+
+ Properties * properties;
+
+ Uint32 size;
+ Uint32 items;
+ PropertyImpl **content;
+
+ bool m_insensitive;
+ int (* compare)(const char *s1, const char *s2);
+
+ void grow(int sizeToAdd);
+
+ PropertyImpl * get(const char * name) const;
+ PropertyImpl * put(PropertyImpl *);
+ void remove(const char * name);
+
+ Uint32 getPackedSize(Uint32 pLen) const;
+ bool pack(Uint32 *& buf, const char * prefix, Uint32 prefixLen) const;
+ bool unpack(const Uint32 * buf, Uint32 &bufLen, Properties * top, int items);
+
+ Uint32 getTotalItems() const;
+
+ void setErrno(Uint32 pErr, Uint32 osErr = 0){
+ properties->setErrno(pErr, osErr);
+ }
+
+ const char * getProps(const char * name, const PropertiesImpl ** impl) const;
+ const char * getPropsPut(const char * name, PropertiesImpl ** impl);
+};
+
+/**
+ * Methods for Property
+ */
+Property::Property(const char * name, Uint32 value){
+ impl = new PropertyImpl(name, value);
+}
+
+Property::Property(const char * name, const char * value){
+ impl = new PropertyImpl(name, value);
+}
+
+Property::Property(const char * name, const class Properties * value){
+ impl = new PropertyImpl(name, value);
+
+ ((Properties*)impl->value)->setCaseInsensitiveNames(value->getCaseInsensitiveNames());
+}
+
+Property::~Property(){
+ delete impl;
+}
+
+/**
+ * Methods for Properties
+ */
+Properties::Properties(){
+ parent = 0;
+ impl = new PropertiesImpl(this);
+}
+
+Properties::Properties(const Properties & org){
+ parent = 0;
+ impl = new PropertiesImpl(this, * org.impl);
+}
+
+Properties::Properties(const Property * anArray, int arrayLen){
+ impl = new PropertiesImpl(this);
+
+ put(anArray, arrayLen);
+}
+
+Properties::~Properties(){
+ clear();
+ delete impl;
+}
+
+void
+Properties::put(const Property * anArray, int arrayLen){
+ if(anArray == 0)
+ return;
+ for(int i = 0; i<arrayLen; i++)
+ impl->put(anArray[i].impl);
+}
+
+template <class T>
+bool
+put(PropertiesImpl * impl, const char * name, T value, bool replace){
+ if(name == 0){
+ impl->setErrno(E_PROPERTIES_INVALID_NAME);
+ return false;
+ }
+
+ PropertiesImpl * tmp = 0;
+ const char * short_name = impl->getPropsPut(name, &tmp);
+
+ if(tmp == 0){
+ impl->setErrno(E_PROPERTIES_NO_SUCH_ELEMENT);
+ return false;
+ }
+
+ if(tmp->get(short_name) != 0){
+ if(replace){
+ tmp->remove(short_name);
+ } else {
+ impl->setErrno(E_PROPERTIES_ELEMENT_ALREADY_EXISTS);
+ return false;
+ }
+ }
+ return tmp->put(new PropertyImpl(short_name, value));
+}
+
+bool
+Properties::put(const char * name, Uint32 value, bool replace){
+ return ::put(impl, name, value, replace);
+}
+
+bool
+Properties::put(const char * name, const char * value, bool replace){
+ return ::put(impl, name, value, replace);
+}
+
+bool
+Properties::put(const char * name, const Properties * value, bool replace){
+ return ::put(impl, name, value, replace);
+}
+
+bool
+Properties::getTypeOf(const char * name, PropertiesType * type) const {
+ PropertyImpl * nvp = impl->get(name);
+ if(nvp == 0){
+ setErrno(E_PROPERTIES_NO_SUCH_ELEMENT);
+ return false;
+ }
+ setErrno(E_PROPERTIES_OK);
+ * type = nvp->valueType;
+ return true;
+}
+
+bool
+Properties::contains(const char * name) const {
+ PropertyImpl * nvp = impl->get(name);
+ return nvp != 0;
+}
+
+bool
+Properties::get(const char * name, Uint32 * value) const {
+ PropertyImpl * nvp = impl->get(name);
+ if(nvp == 0){
+ setErrno(E_PROPERTIES_NO_SUCH_ELEMENT);
+ return false;
+ }
+
+ if(nvp->valueType == PropertiesType_Uint32){
+ * value = * (Uint32 *)nvp->value;
+ setErrno(E_PROPERTIES_OK);
+ return true;
+ }
+ setErrno(E_PROPERTIES_INVALID_TYPE);
+ return false;
+}
+
+bool
+Properties::get(const char * name, const char ** value) const {
+ PropertyImpl * nvp = impl->get(name);
+ if(nvp == 0){
+ setErrno(E_PROPERTIES_NO_SUCH_ELEMENT);
+ return false;
+ }
+
+ if(nvp->valueType == PropertiesType_char){
+ * value = (const char *)nvp->value;
+ setErrno(E_PROPERTIES_OK);
+ return true;
+ }
+ setErrno(E_PROPERTIES_INVALID_TYPE);
+ return false;
+}
+
+bool
+Properties::get(const char * name, BaseString& value) const {
+ const char *tmp = "";
+ bool ret;
+ ret = get(name, &tmp);
+ value.assign(tmp);
+ return ret;
+}
+
+bool
+Properties::get(const char * name, const Properties ** value) const {
+ PropertyImpl * nvp = impl->get(name);
+ if(nvp == 0){
+ setErrno(E_PROPERTIES_NO_SUCH_ELEMENT);
+ return false;
+ }
+ if(nvp->valueType == PropertiesType_Properties){
+ * value = (const Properties *)nvp->value;
+ setErrno(E_PROPERTIES_OK);
+ return true;
+ }
+ setErrno(E_PROPERTIES_INVALID_TYPE);
+ return false;
+}
+
+bool
+Properties::getCopy(const char * name, char ** value) const {
+ PropertyImpl * nvp = impl->get(name);
+ if(nvp == 0){
+ setErrno(E_PROPERTIES_NO_SUCH_ELEMENT);
+ return false;
+ }
+
+ if(nvp->valueType == PropertiesType_char){
+ * value = strdup((const char *)nvp->value);
+ setErrno(E_PROPERTIES_OK);
+ return true;
+ }
+ setErrno(E_PROPERTIES_INVALID_TYPE);
+ return false;
+}
+
+bool
+Properties::getCopy(const char * name, Properties ** value) const {
+ PropertyImpl * nvp = impl->get(name);
+ if(nvp == 0){
+ setErrno(E_PROPERTIES_NO_SUCH_ELEMENT);
+ return false;
+ }
+
+ if(nvp->valueType == PropertiesType_Properties){
+ * value = new Properties(* (const Properties *)nvp->value);
+ setErrno(E_PROPERTIES_OK);
+ return true;
+ }
+ setErrno(E_PROPERTIES_INVALID_TYPE);
+ return false;
+}
+
+void
+Properties::clear(){
+ while(impl->items > 0)
+ impl->remove(impl->content[0]->name);
+}
+
+void
+Properties::remove(const char * name) {
+ impl->remove(name);
+}
+
+void
+Properties::print(FILE * out, const char * prefix) const{
+ char buf[1024];
+ if(prefix == 0)
+ buf[0] = 0;
+ else
+ strncpy(buf, prefix, 1024);
+
+ for(unsigned int i = 0; i<impl->items; i++){
+ switch(impl->content[i]->valueType){
+ case PropertiesType_Uint32:
+ fprintf(out, "%s%s = (Uint32) %d\n", buf, impl->content[i]->name,
+ *(Uint32 *)impl->content[i]->value);
+ break;
+ case PropertiesType_char:
+ fprintf(out, "%s%s = (char*) \"%s\"\n", buf, impl->content[i]->name,
+ (char *)impl->content[i]->value);
+ break;
+ case PropertiesType_Properties:
+ char buf2 [1024];
+ snprintf(buf2, sizeof(buf2), "%s%s%c",buf, impl->content[i]->name,
+ Properties::delimiter);
+ ((Properties *)impl->content[i]->value)->print(out, buf2);
+ break;
+ }
+ }
+}
+
+Properties::Iterator::Iterator(const Properties* prop) :
+ m_prop(prop),
+ m_iterator(0) {
+}
+
+const char*
+Properties::Iterator::first() {
+ m_iterator = 0;
+ return next();
+}
+
+const char*
+Properties::Iterator::next() {
+ if (m_iterator < m_prop->impl->items)
+ return m_prop->impl->content[m_iterator++]->name;
+ else
+ return NULL;
+}
+
+Uint32
+Properties::getPackedSize() const {
+ Uint32 sz = 0;
+
+ sz += sizeof(version); // Version id of properties object
+ sz += 4; // No Of Items
+ sz += 4; // Checksum
+
+ return sz + impl->getPackedSize(0);
+}
+
+static
+Uint32
+computeChecksum(const Uint32 * buf, Uint32 words){
+ Uint32 sum = 0;
+ for(unsigned int i = 0; i<words; i++)
+ sum ^= htonl(buf[i]);
+
+ return sum;
+}
+
+bool
+Properties::pack(Uint32 * buf) const {
+ Uint32 * bufStart = buf;
+
+ memcpy(buf, version, sizeof(version));
+
+ // Note that version must be a multiple of 4
+ buf += (sizeof(version) / 4);
+
+ * buf = htonl(impl->getTotalItems());
+ buf++;
+ bool res = impl->pack(buf, "", 0);
+ if(!res)
+ return res;
+
+ * buf = htonl(computeChecksum(bufStart, (buf - bufStart)));
+
+ return true;
+}
+
+bool
+Properties::unpack(const Uint32 * buf, Uint32 bufLen){
+ const Uint32 * bufStart = buf;
+ Uint32 bufLenOrg = bufLen;
+
+ if(bufLen < sizeof(version)){
+ setErrno(E_PROPERTIES_INVALID_BUFFER_TO_SHORT);
+ return false;
+ }
+
+ if(memcmp(buf, version, sizeof(version)) != 0){
+ setErrno(E_PROPERTIES_INVALID_VERSION_WHILE_UNPACKING);
+ return false;
+ }
+ bufLen -= sizeof(version);
+
+ // Note that version must be a multiple of 4
+ buf += (sizeof(version) / 4);
+
+ if(bufLen < 4){
+ setErrno(E_PROPERTIES_INVALID_BUFFER_TO_SHORT);
+ return false;
+ }
+
+ Uint32 totalItems = ntohl(* buf);
+ buf++; bufLen -= 4;
+ bool res = impl->unpack(buf, bufLen, this, totalItems);
+ if(!res)
+ return res;
+
+ Uint32 sum = computeChecksum(bufStart, (bufLenOrg-bufLen)/4);
+ if(sum != ntohl(bufStart[(bufLenOrg-bufLen)/4])){
+ setErrno(E_PROPERTIES_INVALID_CHECKSUM);
+ return false;
+ }
+ return true;
+}
+
+/**
+ * Methods for PropertiesImpl
+ */
+PropertiesImpl::PropertiesImpl(Properties * p){
+ this->properties = p;
+ items = 0;
+ size = 25;
+ content = new PropertyImpl * [size];
+ this->m_insensitive = false;
+ this->compare = strcmp;
+}
+
+PropertiesImpl::PropertiesImpl(Properties * p, const PropertiesImpl & org){
+ this->properties = p;
+ this->size = org.size;
+ this->items = org.items;
+ this->m_insensitive = org.m_insensitive;
+ this->compare = org.compare;
+ content = new PropertyImpl * [size];
+ for(unsigned int i = 0; i<items; i++){
+ content[i] = PropertyImpl::copyPropertyImpl(* org.content[i]);
+ }
+}
+
+PropertiesImpl::~PropertiesImpl(){
+ for(unsigned int i = 0; i<items; i++)
+ delete content[i];
+ delete [] content;
+}
+
+void
+PropertiesImpl::grow(int sizeToAdd){
+ PropertyImpl ** newContent = new PropertyImpl * [size + sizeToAdd];
+ memcpy(newContent, content, items * sizeof(PropertyImpl *));
+ delete [] content;
+ content = newContent;
+ size += sizeToAdd;
+}
+
+PropertyImpl *
+PropertiesImpl::get(const char * name) const {
+ const PropertiesImpl * tmp = 0;
+ const char * short_name = getProps(name, &tmp);
+ if(tmp == 0){
+ return 0;
+ }
+
+ for(unsigned int i = 0; i<tmp->items; i++)
+ if((* compare)(tmp->content[i]->name, short_name) == 0)
+ return tmp->content[i];
+ return 0;
+}
+
+PropertyImpl *
+PropertiesImpl::put(PropertyImpl * nvp){
+ if(items == size)
+ grow(size);
+ content[items] = nvp;
+
+ items ++;
+
+ if(nvp->valueType == PropertiesType_Properties){
+ ((Properties*)nvp->value)->parent = properties;
+ }
+ return nvp;
+}
+
+void
+PropertiesImpl::remove(const char * name){
+ for(unsigned int i = 0; i<items; i++){
+ if((* compare)(content[i]->name, name) == 0){
+ delete content[i];
+ memmove(&content[i], &content[i+1], (items-i-1)*sizeof(PropertyImpl *));
+ items --;
+ return;
+ }
+ }
+}
+
+Uint32
+PropertiesImpl::getTotalItems() const {
+ int ret = 0;
+ for(unsigned int i = 0; i<items; i++)
+ if(content[i]->valueType == PropertiesType_Properties){
+ ret += ((Properties*)content[i]->value)->impl->getTotalItems();
+ } else {
+ ret ++;
+ }
+ return ret;
+}
+
+const char *
+PropertiesImpl::getProps(const char * name,
+ const PropertiesImpl ** impl) const {
+ const char * ret = name;
+ const char * tmp = strchr(name, Properties::delimiter);
+ if(tmp == 0){
+ * impl = this;
+ return ret;
+ } else {
+ Uint32 sz = tmp - name;
+ char * tmp2 = (char*)malloc(sz + 1);
+ memcpy(tmp2, name, sz);
+ tmp2[sz] = 0;
+
+ PropertyImpl * nvp = get(tmp2);
+
+ free(tmp2);
+
+ if(nvp == 0){
+ * impl = 0;
+ return 0;
+ }
+ if(nvp->valueType != PropertiesType_Properties){
+ * impl = 0;
+ return name;
+ }
+ return ((Properties*)nvp->value)->impl->getProps(tmp+1, impl);
+ }
+}
+
+const char *
+PropertiesImpl::getPropsPut(const char * name,
+ PropertiesImpl ** impl) {
+ const char * ret = name;
+ const char * tmp = strchr(name, Properties::delimiter);
+ if(tmp == 0){
+ * impl = this;
+ return ret;
+ } else {
+ Uint32 sz = tmp - name;
+ char * tmp2 = (char*)malloc(sz + 1);
+ memcpy(tmp2, name, sz);
+ tmp2[sz] = 0;
+
+ PropertyImpl * nvp = get(tmp2);
+
+ if(nvp == 0){
+ Properties * tmpP = new Properties();
+ PropertyImpl * tmpPI = new PropertyImpl(tmp2, tmpP);
+ PropertyImpl * nvp = put(tmpPI);
+
+ delete tmpP;
+ free(tmp2);
+ return ((Properties*)nvp->value)->impl->getPropsPut(tmp+1, impl);
+ }
+ free(tmp2);
+ if(nvp->valueType != PropertiesType_Properties){
+ * impl = 0;
+ return name;
+ }
+ return ((Properties*)nvp->value)->impl->getPropsPut(tmp+1, impl);
+ }
+}
+
+int
+mod4(unsigned int i){
+ int res = i + (4 - (i % 4));
+ return res;
+}
+
+Uint32
+PropertiesImpl::getPackedSize(Uint32 pLen) const {
+ Uint32 sz = 0;
+ for(unsigned int i = 0; i<items; i++){
+ if(content[i]->valueType == PropertiesType_Properties){
+ Properties * p = (Properties*)content[i]->value;
+ sz += p->impl->getPackedSize(pLen+strlen(content[i]->name)+1);
+ } else {
+ sz += 4; // Type
+ sz += 4; // Name Len
+ sz += 4; // Value Len
+ sz += mod4(pLen + strlen(content[i]->name)); // Name
+ if(content[i]->valueType == PropertiesType_char){
+ sz += mod4(strlen((char *)content[i]->value));
+ } else if(content[i]->valueType == PropertiesType_Uint32){
+ sz += mod4(4);
+ } else {
+ assert(0);
+ }
+ }
+ }
+ return sz;
+}
+
+struct CharBuf {
+ char * buffer;
+ Uint32 bufLen;
+ Uint32 contentLen;
+
+ CharBuf(){
+ buffer = 0;
+ bufLen = 0;
+ contentLen = 0;
+ }
+
+ ~CharBuf(){
+ free(buffer);
+ }
+
+ void clear() { contentLen = 0;}
+ bool add(const char * str, Uint32 strLen){
+ if(!expand(contentLen + strLen + 1))
+ return false;
+ memcpy(&buffer[contentLen], str, strLen);
+ contentLen += strLen;
+ buffer[contentLen] = 0;
+ return true;
+ }
+
+ bool add(char c){
+ return add(&c, 1);
+ }
+
+ bool expand(Uint32 newSize){
+ if(newSize >= bufLen){
+
+ char * tmp = (char*)malloc(newSize + 1024);
+ memset(tmp, 0, newSize + 1024);
+ if(tmp == 0)
+ return false;
+ if(contentLen > 0)
+ memcpy(tmp, buffer, contentLen);
+ if(buffer != 0)
+ free(buffer);
+ buffer = tmp;
+ bufLen = newSize + 1024;
+ }
+ return true;
+ }
+};
+
+bool
+PropertiesImpl::pack(Uint32 *& buf, const char * prefix, Uint32 pLen) const {
+ CharBuf charBuf;
+
+ for(unsigned int i = 0; i<items; i++){
+ const int strLenName = strlen(content[i]->name);
+
+ if(content[i]->valueType == PropertiesType_Properties){
+ charBuf.clear();
+ if(!charBuf.add(prefix, pLen)){
+ properties->setErrno(E_PROPERTIES_ERROR_MALLOC_WHILE_PACKING,
+ errno);
+ return false;
+ }
+
+ if(!charBuf.add(content[i]->name, strLenName)){
+ properties->setErrno(E_PROPERTIES_ERROR_MALLOC_WHILE_PACKING,
+ errno);
+ return false;
+ }
+
+ if(!charBuf.add(Properties::delimiter)){
+ properties->setErrno(E_PROPERTIES_ERROR_MALLOC_WHILE_PACKING,
+ errno);
+ return false;
+ }
+
+ if(!((Properties*)(content[i]->value))->impl->pack(buf,
+ charBuf.buffer,
+ charBuf.contentLen)){
+
+ return false;
+ }
+ continue;
+ }
+
+ Uint32 valLenData = 0;
+ Uint32 valLenWrite = 0;
+ Uint32 sz = 4 + 4 + 4 + mod4(pLen + strLenName);
+ switch(content[i]->valueType){
+ case PropertiesType_Uint32:
+ valLenData = 4;
+ break;
+ case PropertiesType_char:
+ valLenData = strlen((char *)content[i]->value);
+ break;
+ case PropertiesType_Properties:
+ assert(0);
+ }
+ valLenWrite = mod4(valLenData);
+ sz += valLenWrite;
+
+ * (buf + 0) = htonl(content[i]->valueType);
+ * (buf + 1) = htonl(pLen + strLenName);
+ * (buf + 2) = htonl(valLenData);
+
+ char * valBuf = (char*)(buf + 3);
+ char * nameBuf = (char*)(buf + 3 + (valLenWrite / 4));
+
+ memset(valBuf, 0, sz-12);
+
+ switch(content[i]->valueType){
+ case PropertiesType_Uint32:
+ * (Uint32 *)valBuf = htonl(* (Uint32 *)content[i]->value);
+ break;
+ case PropertiesType_char:
+ memcpy(valBuf, content[i]->value, strlen((char*)content[i]->value));
+ break;
+ case PropertiesType_Properties:
+ assert(0);
+ }
+ if(pLen > 0)
+ memcpy(nameBuf, prefix, pLen);
+ memcpy(nameBuf + pLen, content[i]->name, strLenName);
+
+ buf += (sz / 4);
+ }
+
+ return true;
+}
+
+bool
+PropertiesImpl::unpack(const Uint32 * buf, Uint32 &bufLen, Properties * top,
+ int _items){
+ CharBuf charBuf;
+ while(_items > 0){
+ Uint32 tmp[3];
+
+ if(bufLen <= 12){
+ top->setErrno(E_PROPERTIES_BUFFER_TO_SMALL_WHILE_UNPACKING);
+ return false;
+ }
+
+ tmp[0] = ntohl(buf[0]);
+ tmp[1] = ntohl(buf[1]);
+ tmp[2] = ntohl(buf[2]);
+ buf += 3;
+ bufLen -= 12;
+
+ PropertiesType pt = (PropertiesType)tmp[0];
+ Uint32 nameLen = tmp[1];
+ Uint32 valueLen = tmp[2];
+ Uint32 nameLenRead = mod4(nameLen);
+ Uint32 valueLenRead = mod4(valueLen);
+
+ Uint32 sz = nameLenRead + valueLenRead;
+ if(bufLen < sz){
+ top->setErrno(E_PROPERTIES_BUFFER_TO_SMALL_WHILE_UNPACKING);
+ return false;
+ }
+
+ if(!charBuf.expand(sz)){
+ top->setErrno(E_PROPERTIES_ERROR_MALLOC_WHILE_UNPACKING, errno);
+ return false;
+ }
+
+ memcpy(charBuf.buffer, buf, sz);
+ buf += (sz / 4);
+ bufLen -= sz ;
+
+ char * valBuf = charBuf.buffer;
+ char * nameBuf = charBuf.buffer + valueLenRead;
+
+ nameBuf[nameLen] = 0;
+ valBuf[valueLen] = 0;
+
+ bool res3 = false;
+ switch(pt){
+ case PropertiesType_Uint32:
+ res3 = top->put(nameBuf, ntohl(* (Uint32 *)valBuf), true);
+ break;
+ case PropertiesType_char:
+ res3 = top->put(nameBuf, valBuf, true);
+ break;
+ case PropertiesType_Properties:
+ assert(0);
+ }
+ if(res3 != true){
+ return false;
+ }
+ _items--;
+ }
+ return true;
+}
+
+PropertyImpl::~PropertyImpl(){
+ free((char*)name);
+ switch(valueType){
+ case PropertiesType_Uint32:
+ delete (Uint32 *)value;
+ break;
+ case PropertiesType_char:
+ free((char *)value);
+ break;
+ case PropertiesType_Properties:
+ delete (Properties *)value;
+ break;
+ }
+}
+
+PropertyImpl *
+PropertyImpl::copyPropertyImpl(const PropertyImpl & org){
+ switch(org.valueType){
+ case PropertiesType_Uint32:
+ return new PropertyImpl(org.name, * (Uint32 *)org.value);
+ break;
+ case PropertiesType_char:
+ return new PropertyImpl(org.name, (char *)org.value);
+ break;
+ case PropertiesType_Properties:
+ return new PropertyImpl(org.name, (Properties *)org.value);
+ break;
+ default:
+ assert(0);
+ }
+ return 0;
+}
+
+PropertyImpl::PropertyImpl(const char * _name, Uint32 _value){
+ this->name = strdup(_name);
+ this->value = new Uint32;
+ * ((Uint32 *)this->value) = _value;
+ this->valueType = PropertiesType_Uint32;
+}
+
+PropertyImpl::PropertyImpl(const char * _name, const char * _value){
+ this->name = strdup(_name);
+ this->value = strdup(_value);
+ this->valueType = PropertiesType_char;
+
+}
+
+PropertyImpl::PropertyImpl(const char * _name, const Properties * _value){
+ this->name = strdup(_name);
+ this->value = new Properties(* _value);
+ this->valueType = PropertiesType_Properties;
+}
+
+const Uint32 E_PROPERTIES_OK = 0;
+const Uint32 E_PROPERTIES_INVALID_NAME = 1;
+const Uint32 E_PROPERTIES_NO_SUCH_ELEMENT = 2;
+const Uint32 E_PROPERTIES_INVALID_TYPE = 3;
+const Uint32 E_PROPERTIES_ELEMENT_ALREADY_EXISTS = 4;
+
+const Uint32 E_PROPERTIES_ERROR_MALLOC_WHILE_PACKING = 5;
+const Uint32 E_PROPERTIES_INVALID_VERSION_WHILE_UNPACKING = 6;
+const Uint32 E_PROPERTIES_INVALID_BUFFER_TO_SHORT = 7;
+const Uint32 E_PROPERTIES_ERROR_MALLOC_WHILE_UNPACKING = 8;
+const Uint32 E_PROPERTIES_INVALID_CHECKSUM = 9;
+const Uint32 E_PROPERTIES_BUFFER_TO_SMALL_WHILE_UNPACKING = 10;
+
+/**
+ * These are methods that used to be inline
+ *
+ * But Diab 4.1f could not compile -release with to many inlines
+ */
+void
+Properties::setErrno(Uint32 pErr, Uint32 osErr) const {
+ if(parent != 0){
+ parent->setErrno(pErr, osErr);
+ return ;
+ }
+
+ /**
+ * propErrno & osErrno used to be mutable,
+ * but diab didn't know what mutable meant.
+ */
+ *((Uint32*)&propErrno) = pErr;
+ *((Uint32*)&osErrno) = osErr;
+}
+
+/**
+ * Inlined get/put(name, no, ...) - methods
+ */
+
+bool
+Properties::put(const char * name, Uint32 no, Uint32 val, bool replace){
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = put(tmp, val, replace);
+ free(tmp);
+ return res;
+}
+
+
+bool
+Properties::put(const char * name, Uint32 no, const char * val, bool replace){
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = put(tmp, val, replace);
+ free(tmp);
+ return res;
+}
+
+
+bool
+Properties::put(const char * name, Uint32 no, const Properties * val,
+ bool replace){
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = put(tmp, val, replace);
+ free(tmp);
+ return res;
+}
+
+
+bool
+Properties::getTypeOf(const char * name, Uint32 no,
+ PropertiesType * type) const {
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = getTypeOf(tmp, type);
+ free(tmp);
+ return res;
+}
+
+bool
+Properties::contains(const char * name, Uint32 no) const {
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = contains(tmp);
+ free(tmp);
+ return res;
+}
+
+bool
+Properties::get(const char * name, Uint32 no, Uint32 * value) const{
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = get(tmp, value);
+ free(tmp);
+ return res;
+}
+
+
+bool
+Properties::get(const char * name, Uint32 no, const char ** value) const {
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = get(tmp, value);
+ free(tmp);
+ return res;
+}
+
+
+bool
+Properties::get(const char * name, Uint32 no, const Properties ** value) const{
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = get(tmp, value);
+ free(tmp);
+ return res;
+}
+
+
+bool
+Properties::getCopy(const char * name, Uint32 no, char ** value) const {
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = getCopy(tmp, value);
+ free(tmp);
+ return res;
+}
+
+
+bool
+Properties::getCopy(const char * name, Uint32 no, Properties ** value) const {
+ size_t tmp_len = strlen(name)+20;
+ char * tmp = (char*)malloc(tmp_len);
+ snprintf(tmp, tmp_len, "%s_%d", name, no);
+ bool res = getCopy(tmp, value);
+ free(tmp);
+ return res;
+}
+
+void
+Properties::setCaseInsensitiveNames(bool value){
+ impl->m_insensitive = value;
+ if(value)
+ impl->compare = strcasecmp;
+ else
+ impl->compare = strcmp;
+}
+
+bool
+Properties::getCaseInsensitiveNames() const {
+ return impl->m_insensitive;
+}
diff --git a/ndb/src/common/util/SimpleProperties.cpp b/ndb/src/common/util/SimpleProperties.cpp
new file mode 100644
index 00000000000..a118478ba6c
--- /dev/null
+++ b/ndb/src/common/util/SimpleProperties.cpp
@@ -0,0 +1,509 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <SimpleProperties.hpp>
+#include <stdlib.h>
+#include <NdbString.h>
+#include <NdbOut.hpp>
+#include <NdbTCP.h>
+#include <assert.h>
+#include <UtilBuffer.hpp>
+
+bool
+SimpleProperties::Writer::first(){
+ return reset();
+}
+
+bool
+SimpleProperties::Writer::add(Uint16 key, Uint32 value){
+ Uint32 head = Uint32Value;
+ head <<= 16;
+ head += key;
+ if(!putWord(htonl(head)))
+ return false;
+
+ return putWord(htonl(value));
+}
+
+bool
+SimpleProperties::Writer::add(Uint16 key, const char * value){
+ Uint32 head = StringValue;
+ head <<= 16;
+ head += key;
+ if(!putWord(htonl(head)))
+ return false;
+ Uint32 strLen = strlen(value) + 1; // Including NULL-byte
+ if(!putWord(htonl(strLen)))
+ return false;
+
+ const Uint32 valLen = (strLen + 3) / 4;
+ return putWords((Uint32*)value, valLen);
+}
+
+bool
+SimpleProperties::Writer::add(Uint16 key, const void* value, int len){
+ Uint32 head = BinaryValue;
+ head <<= 16;
+ head += key;
+ if(!putWord(htonl(head)))
+ return false;
+ if(!putWord(htonl(len)))
+ return false;
+
+ const Uint32 valLen = (len + 3) / 4;
+ return putWords((Uint32*)value, valLen);
+}
+
+SimpleProperties::Reader::Reader(){
+ m_itemLen = 0;
+}
+
+bool
+SimpleProperties::Reader::first(){
+ reset();
+ m_itemLen = 0;
+ return readValue();
+}
+
+bool
+SimpleProperties::Reader::next(){
+ return readValue();
+}
+
+bool
+SimpleProperties::Reader::valid() const {
+ return m_type != InvalidValue;
+}
+
+Uint16
+SimpleProperties::Reader::getKey() const{
+ return m_key;
+}
+
+Uint16
+SimpleProperties::Reader::getValueLen() const {
+ switch(m_type){
+ case Uint32Value:
+ return 4;
+ case StringValue:
+ case BinaryValue:
+ return m_strLen;
+ case InvalidValue:
+ return 0;
+ }
+ return 0;
+}
+
+SimpleProperties::ValueType
+SimpleProperties::Reader::getValueType() const {
+ return m_type;
+}
+
+Uint32
+SimpleProperties::Reader::getUint32() const {
+ return m_ui32_value;
+}
+
+char *
+SimpleProperties::Reader::getString(char * dst) const {
+ if(peekWords((Uint32*)dst, m_itemLen))
+ return dst;
+ return 0;
+}
+
+bool
+SimpleProperties::Reader::readValue(){
+ if(!step(m_itemLen)){
+ m_type = InvalidValue;
+ return false;
+ }
+
+ Uint32 tmp;
+ if(!getWord(&tmp)){
+ m_type = InvalidValue;
+ return false;
+ }
+
+ tmp = ntohl(tmp);
+ m_key = tmp & 0xFFFF;
+ m_type = (SimpleProperties::ValueType)(tmp >> 16);
+ switch(m_type){
+ case Uint32Value:
+ m_itemLen = 1;
+ if(!peekWord(&m_ui32_value))
+ return false;
+ m_ui32_value = ntohl(m_ui32_value);
+ return true;
+ case StringValue:
+ case BinaryValue:
+ if(!getWord(&tmp))
+ return false;
+ m_strLen = ntohl(tmp);
+ m_itemLen = (m_strLen + 3)/4;
+ return true;
+ default:
+ m_itemLen = 0;
+ m_type = InvalidValue;
+ return false;
+ }
+}
+
+SimpleProperties::UnpackStatus
+SimpleProperties::unpack(Reader & it, void * dst,
+ const SP2StructMapping _map[], Uint32 mapSz,
+ bool ignoreMinMax,
+ bool ignoreUnknownKeys){
+ do {
+ if(!it.valid())
+ break;
+
+ bool found = false;
+ Uint16 key = it.getKey();
+ for(Uint32 i = 0; i<mapSz; i++){
+ if(key == _map[i].Key){
+ found = true;
+ if(_map[i].Type == InvalidValue)
+ return Break;
+ if(_map[i].Type != it.getValueType())
+ return TypeMismatch;
+
+ char * _dst = (char *)dst;
+ _dst += _map[i].Offset;
+
+ switch(it.getValueType()){
+ case Uint32Value:{
+ const Uint32 val = it.getUint32();
+ if(!ignoreMinMax){
+ if(val < _map[i].minValue)
+ return ValueTooLow;
+ if(val > _map[i].maxValue)
+ return ValueTooHigh;
+ }
+ * ((Uint32 *)_dst) = val;
+ break;
+ }
+ case BinaryValue:
+ case StringValue:{
+ unsigned len = it.getValueLen();
+ if(len < _map[i].minValue)
+ return ValueTooLow;
+ if(len > _map[i].maxValue)
+ return ValueTooHigh;
+ it.getString(_dst);
+ break;
+ }
+ default:
+ abort();
+ }
+ break;
+ }
+ }
+ if(!found && !ignoreUnknownKeys)
+ return UnknownKey;
+ } while(it.next());
+
+ return Eof;
+}
+
+SimpleProperties::UnpackStatus
+SimpleProperties::pack(Writer & it, const void * __src,
+ const SP2StructMapping _map[], Uint32 mapSz,
+ bool ignoreMinMax){
+
+ const char * _src = (const char *)__src;
+
+ for(Uint32 i = 0; i<mapSz; i++){
+ bool ok = false;
+ const char * src = _src + _map[i].Offset;
+ switch(_map[i].Type){
+ case SimpleProperties::InvalidValue:
+ ok = true;
+ break;
+ case SimpleProperties::Uint32Value:{
+ Uint32 val = * ((Uint32*)src);
+ if(!ignoreMinMax){
+ if(val < _map[i].minValue)
+ return ValueTooLow;
+ if(val > _map[i].maxValue)
+ return ValueTooHigh;
+ }
+ ok = it.add(_map[i].Key, val);
+ }
+ break;
+ case SimpleProperties::BinaryValue:{
+ const char * src_len = _src + _map[i].Length_Offset;
+ Uint32 len = *((Uint32*)src_len);
+ if(!ignoreMinMax){
+ if(len == _map[i].maxValue)
+ return ValueTooHigh;
+ }
+ ok = it.add(_map[i].Key, src, len);
+ break;
+ }
+ case SimpleProperties::StringValue:
+ if(!ignoreMinMax){
+ size_t len = strlen(src);
+ if(len == _map[i].maxValue)
+ return ValueTooHigh;
+ }
+ ok = it.add(_map[i].Key, src);
+ break;
+ }
+ if(!ok)
+ return OutOfMemory;
+ }
+
+ return Eof;
+}
+
+void
+SimpleProperties::Reader::printAll(NdbOut& ndbout){
+ char tmp[1024];
+ for(first(); valid(); next()){
+ switch(getValueType()){
+ case SimpleProperties::Uint32Value:
+ ndbout << "Key: " << getKey()
+ << " value(" << getValueLen() << ") : "
+ << getUint32() << endl;
+ break;
+ case SimpleProperties::BinaryValue:
+ case SimpleProperties::StringValue:
+ if(getValueLen() < 1024){
+ getString(tmp);
+ ndbout << "Key: " << getKey()
+ << " value(" << getValueLen() << ") : "
+ << "\"" << tmp << "\"" << endl;
+ } else {
+ ndbout << "Key: " << getKey()
+ << " value(" << getValueLen() << ") : "
+ << "\"" << "<TOO LONG>" << "\"" << endl;
+
+ }
+ break;
+ default:
+ ndbout << "Unknown type for key: " << getKey()
+ << " type: " << getValueType() << endl;
+ }
+ }
+}
+
+SimplePropertiesLinearReader::SimplePropertiesLinearReader
+(const Uint32 * src, Uint32 len){
+ m_src = src;
+ m_len = len;
+ m_pos = 0;
+ first();
+}
+
+void
+SimplePropertiesLinearReader::reset() {
+ m_pos = 0;
+}
+
+bool
+SimplePropertiesLinearReader::step(Uint32 len){
+ m_pos += len;
+ return m_pos < m_len;
+}
+
+bool
+SimplePropertiesLinearReader::getWord(Uint32 * dst) {
+ if(m_pos<m_len){
+ * dst = m_src[m_pos++];
+ return true;
+ }
+ return false;
+}
+
+bool
+SimplePropertiesLinearReader::peekWord(Uint32 * dst) const {
+ if(m_pos<m_len){
+ * dst = m_src[m_pos];
+ return true;
+ }
+ return false;
+}
+
+bool
+SimplePropertiesLinearReader::peekWords(Uint32 * dst, Uint32 len) const {
+ if(m_pos + len <= m_len){
+ memcpy(dst, &m_src[m_pos], 4 * len);
+ return true;
+ }
+ return false;
+}
+
+LinearWriter::LinearWriter(Uint32 * src, Uint32 len){
+ m_src = src;
+ m_len = len;
+ reset();
+}
+
+bool LinearWriter::reset() { m_pos = 0; return m_len > 0;}
+
+bool
+LinearWriter::putWord(Uint32 val){
+ if(m_pos < m_len){
+ m_src[m_pos++] = val;
+ return true;
+ }
+ return false;
+}
+
+bool
+LinearWriter::putWords(const Uint32 * src, Uint32 len){
+ if(m_pos + len <= m_len){
+ memcpy(&m_src[m_pos], src, 4 * len);
+ m_pos += len;
+ return true;
+ }
+ return false;
+}
+
+Uint32
+LinearWriter::getWordsUsed() const { return m_pos;}
+
+UtilBufferWriter::UtilBufferWriter(UtilBuffer & b)
+ : m_buf(b)
+{
+ reset();
+}
+
+bool UtilBufferWriter::reset() { m_buf.clear(); return true;}
+
+bool
+UtilBufferWriter::putWord(Uint32 val){
+ return (m_buf.append(&val, 4) == 0);
+}
+
+bool
+UtilBufferWriter::putWords(const Uint32 * src, Uint32 len){
+ return (m_buf.append(src, 4 * len) == 0);
+}
+
+Uint32
+UtilBufferWriter::getWordsUsed() const { return m_buf.length() / 4;}
+
+#if 0
+LinearPagesReader::LinearPagesReader(const Uint32 * base,
+ Uint32 pageSize,
+ Uint32 headerSize,
+ Uint32 noOfPages,
+ Uint32 len){
+ m_base = base;
+ m_pageSz = pageSize;
+ m_noOfPages = noOfPages;
+ m_pageHeaderSz = headerSize;
+ m_len = len;
+ reset();
+}
+
+void
+LinearPagesReader::reset() { m_pos = 0;}
+
+bool
+LinearPagesReader::step(Uint32 len){
+ m_pos += len;
+ return m_pos < m_len;
+}
+
+bool
+LinearPagesReader::getWord(Uint32 * dst) {
+ if(m_pos<m_len){
+ * dst = m_base[getPos(m_pos++)];
+ return true;
+ }
+ return false;
+}
+
+bool
+LinearPagesReader::peekWord(Uint32 * dst) const {
+ if(m_pos<m_len){
+ * dst = m_base[getPos(m_pos)];
+ return true;
+ }
+ return false;
+}
+
+bool
+LinearPagesReader::peekWords(Uint32 * dst, Uint32 len) const {
+ if(m_pos + len <= m_len){
+ for(Uint32 i = 0; i<len; i++)
+ * (dst + i) = m_base[getPos(m_pos + i)];
+ return true;
+ }
+ return false;
+}
+
+Uint32
+LinearPagesReader::getPos(Uint32 pos) const {
+ const Uint32 sz = (m_pageSz - m_pageHeaderSz);
+ Uint32 no = pos / sz;
+ Uint32 in = pos % sz;
+ return no * m_pageSz + m_pageHeaderSz + in;
+}
+
+LinearPagesWriter::LinearPagesWriter(Uint32 * base,
+ Uint32 pageSize,
+ Uint32 noOfPages,
+ Uint32 headerSize){
+ m_base = base;
+ m_pageSz = pageSize;
+ m_noOfPages = noOfPages;
+ m_pageHeaderSz = headerSize;
+ m_len = noOfPages * (pageSize - headerSize);
+ reset();
+}
+
+bool
+LinearPagesWriter::putWord(Uint32 val){
+ if(m_pos < m_len){
+ m_base[getPos(m_pos++)] = val;
+ return true;
+ }
+ return false;
+}
+
+bool
+LinearPagesWriter::putWords(const Uint32 * src, Uint32 len){
+ if(m_pos + len <= m_len){
+ for(Uint32 i = 0; i<len; i++)
+ m_base[getPos(m_pos++)] = src[i];
+ return true;
+ }
+ return false;
+}
+
+#if 0
+Uint32
+LinearPagesWriter::getWordsUsed() const {
+ return getPos(m_pos);
+}
+#endif
+
+Uint32
+LinearPagesWriter::getPagesUsed() const {
+ return m_pos / (m_pageSz - m_pageHeaderSz);
+}
+
+Uint32
+LinearPagesWriter::getPos(Uint32 pos) const {
+ const Uint32 sz = (m_pageSz - m_pageHeaderSz);
+ Uint32 no = pos / sz;
+ Uint32 in = pos % sz;
+ return no * m_pageSz + m_pageHeaderSz + in;
+}
+#endif
diff --git a/ndb/src/common/util/SocketServer.cpp b/ndb/src/common/util/SocketServer.cpp
new file mode 100644
index 00000000000..39f46eceed9
--- /dev/null
+++ b/ndb/src/common/util/SocketServer.cpp
@@ -0,0 +1,307 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "SocketServer.hpp"
+
+#include <NdbTCP.h>
+#include <string.h>
+#include <NdbOut.hpp>
+#include <NdbThread.h>
+#include <NdbSleep.h>
+
+#include <stdio.h>
+#include <assert.h>
+
+#define DEBUG(x) ndbout << x << endl;
+
+SocketServer::SocketServer(int maxSessions) :
+ m_sessions(10),
+ m_services(5)
+{
+ m_thread = 0;
+ m_stopThread = false;
+ m_maxSessions = maxSessions;
+}
+
+SocketServer::~SocketServer() {
+ for(unsigned i = 0; i<m_sessions.size(); i++){
+ delete m_sessions[i].m_session;
+ }
+ for(unsigned i = 0; i<m_services.size(); i++){
+ delete m_services[i].m_service;
+ }
+}
+
+bool
+SocketServer::tryBind(unsigned short port, const char * intface) const {
+ struct sockaddr_in servaddr;
+ memset(&servaddr, 0, sizeof(servaddr));
+ servaddr.sin_family = AF_INET;
+ servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
+ servaddr.sin_port = htons(port);
+
+ if(intface != 0){
+ if(Ndb_getInAddr(&servaddr.sin_addr, intface))
+ return false;
+ }
+
+ const NDB_SOCKET_TYPE sock = socket(AF_INET, SOCK_STREAM, 0);
+ if (sock == NDB_INVALID_SOCKET) {
+ return false;
+ }
+
+ const int on = 1;
+ if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR,
+ (const char*)&on, sizeof(on)) == -1) {
+ NDB_CLOSE_SOCKET(sock);
+ return false;
+ }
+
+ if (bind(sock, (struct sockaddr*) &servaddr, sizeof(servaddr)) == -1) {
+ NDB_CLOSE_SOCKET(sock);
+ return false;
+ }
+
+ NDB_CLOSE_SOCKET(sock);
+ return true;
+}
+
+bool
+SocketServer::setup(SocketServer::Service * service,
+ unsigned short port,
+ const char * intface){
+
+ struct sockaddr_in servaddr;
+ memset(&servaddr, 0, sizeof(servaddr));
+ servaddr.sin_family = AF_INET;
+ servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
+ servaddr.sin_port = htons(port);
+
+ if(intface != 0){
+ if(Ndb_getInAddr(&servaddr.sin_addr, intface))
+ return false;
+ }
+
+ const NDB_SOCKET_TYPE sock = socket(AF_INET, SOCK_STREAM, 0);
+ if (sock == NDB_INVALID_SOCKET) {
+ return false;
+ }
+
+ const int on = 1;
+ if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR,
+ (const char*)&on, sizeof(on)) == -1) {
+ NDB_CLOSE_SOCKET(sock);
+ return false;
+ }
+
+ if (bind(sock, (struct sockaddr*) &servaddr, sizeof(servaddr)) == -1) {
+ NDB_CLOSE_SOCKET(sock);
+ return false;
+ }
+
+ if (listen(sock, m_maxSessions) == -1){
+ NDB_CLOSE_SOCKET(sock);
+ return false;
+ }
+
+ ServiceInstance i;
+ i.m_socket = sock;
+ i.m_service = service;
+ m_services.push_back(i);
+ return true;
+}
+
+void
+SocketServer::doAccept(){
+ fd_set readSet, exceptionSet;
+ FD_ZERO(&readSet);
+ FD_ZERO(&exceptionSet);
+
+ m_services.lock();
+ int maxSock = 0;
+ for (unsigned i = 0; i < m_services.size(); i++){
+ const NDB_SOCKET_TYPE s = m_services[i].m_socket;
+ FD_SET(s, &readSet);
+ FD_SET(s, &exceptionSet);
+ maxSock = (maxSock > s ? maxSock : s);
+ }
+ struct timeval timeout;
+ timeout.tv_sec = 1;
+ timeout.tv_usec = 0;
+
+ if(select(maxSock + 1, &readSet, 0, &exceptionSet, &timeout) > 0){
+ for (unsigned i = 0; i < m_services.size(); i++){
+ ServiceInstance & si = m_services[i];
+
+ if(FD_ISSET(si.m_socket, &readSet)){
+
+ NDB_SOCKET_TYPE childSock = accept(si.m_socket, 0, 0);
+ if(childSock == NDB_INVALID_SOCKET){
+ continue;
+ }
+
+ SessionInstance s;
+ s.m_service = si.m_service;
+ s.m_session = si.m_service->newSession(childSock);
+ if(s.m_session != 0){
+ m_sessions.push_back(s);
+ startSession(m_sessions.back());
+ }
+
+ continue;
+ }
+
+ if(FD_ISSET(si.m_socket, &exceptionSet)){
+ DEBUG("socket in the exceptionSet");
+ continue;
+ }
+ }
+ }
+ m_services.unlock();
+}
+
+extern "C"
+void*
+socketServerThread_C(void* _ss){
+ SocketServer * ss = (SocketServer *)_ss;
+
+ ss->doRun();
+
+ NdbThread_Exit(0);
+ return 0;
+}
+
+void
+SocketServer::startServer(){
+ m_threadLock.lock();
+ if(m_thread == 0 && m_stopThread == false){
+ m_thread = NdbThread_Create(socketServerThread_C,
+ (void**)this,
+ 32768,
+ "NdbSockServ",
+ NDB_THREAD_PRIO_LOW);
+ }
+ m_threadLock.unlock();
+}
+
+void
+SocketServer::stopServer(){
+ m_threadLock.lock();
+ if(m_thread != 0){
+ m_stopThread = true;
+
+ void * res;
+ NdbThread_WaitFor(m_thread, &res);
+ NdbThread_Destroy(&m_thread);
+ m_thread = 0;
+ }
+ m_threadLock.unlock();
+}
+
+void
+SocketServer::doRun(){
+
+ while(!m_stopThread){
+ checkSessions();
+ if(m_sessions.size() < m_maxSessions){
+ doAccept();
+ } else {
+ NdbSleep_MilliSleep(200);
+ }
+ }
+}
+
+void
+SocketServer::startSession(SessionInstance & si){
+ si.m_thread = NdbThread_Create(sessionThread_C,
+ (void**)si.m_session,
+ 32768,
+ "NdbSock_Session",
+ NDB_THREAD_PRIO_LOW);
+}
+
+static
+bool
+transfer(NDB_SOCKET_TYPE sock){
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ const PROCESS p = current_process();
+ const size_t ps = sizeof(PROCESS);
+ int res = setsockopt(sock, SOL_SOCKET, SO_OSEOWNER, &p, ps);
+ if(res != 0){
+ ndbout << "Failed to transfer ownership of socket" << endl;
+ return false;
+ }
+#endif
+ return true;
+}
+
+void
+SocketServer::checkSessions(){
+ for(int i = m_sessions.size() - 1; i >= 0; i--){
+ if(m_sessions[i].m_session->m_stopped){
+ if(m_sessions[i].m_thread != 0){
+ void* ret;
+ NdbThread_WaitFor(m_sessions[i].m_thread, &ret);
+ NdbThread_Destroy(&m_sessions[i].m_thread);
+ }
+ m_sessions[i].m_session->stopSession();
+ delete m_sessions[i].m_session;
+ m_sessions.erase(i);
+ }
+ }
+}
+
+void
+SocketServer::stopSessions(bool wait){
+ for(int i = m_sessions.size() - 1; i>=0; i--)
+ m_sessions[i].m_session->m_stop = true;
+
+ for(int i = m_services.size() - 1; i>=0; i--)
+ m_services[i].m_service->stopSessions();
+
+ if(wait){
+ while(m_sessions.size() > 0){
+ checkSessions();
+ NdbSleep_MilliSleep(100);
+ }
+ }
+}
+
+/***** Session code ******/
+
+extern "C"
+void*
+sessionThread_C(void* _sc){
+ SocketServer::Session * si = (SocketServer::Session *)_sc;
+
+ if(!transfer(si->m_socket)){
+ si->m_stopped = true;
+ NdbThread_Exit(0);
+ return 0;
+ }
+
+ if(!si->m_stop){
+ si->m_stopped = false;
+ si->runSession();
+ } else {
+ NDB_CLOSE_SOCKET(si->m_socket);
+ }
+
+ si->m_stopped = true;
+ NdbThread_Exit(0);
+ return 0;
+}
diff --git a/ndb/src/common/util/filetest/FileUnitTest.cpp b/ndb/src/common/util/filetest/FileUnitTest.cpp
new file mode 100644
index 00000000000..ebcca26d3d2
--- /dev/null
+++ b/ndb/src/common/util/filetest/FileUnitTest.cpp
@@ -0,0 +1,238 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "FileUnitTest.hpp"
+#include <File.hpp>
+
+#include <NdbOut.hpp>
+#include <assert.h>
+
+typedef bool (*TESTFUNC)(const char*);
+
+typedef const char TESTNAME;
+typedef struct
+{
+ const char* name;
+ TESTFUNC test;
+}Tests;
+
+static Tests testCases[] = { {"Create/Write", &FileUnitTest::testWrite},
+ {"Read", &FileUnitTest::testRead},
+ {"Exists", &FileUnitTest::testExists},
+ {"File Size", &FileUnitTest::testSize},
+ {"Rename", &FileUnitTest::testRename},
+ {"Remove", &FileUnitTest::testRemove} };
+
+static int testFailed = 0;
+
+int main(int argc, char* argv[])
+{
+ if (argc < 2)
+ {
+ ndbout << "Usage: filetest <filename>" << endl;
+ return 0;
+ }
+ const char* fileName = argv[1];
+
+ int testCount = (sizeof(testCases) / sizeof(Tests));
+ ndbout << "Starting " << testCount << " tests..." << endl;
+ for (int i = 0; i < testCount; i++)
+ {
+ ndbout << "-- " << " Test " << i + 1
+ << " [" << testCases[i].name << "] --" << endl;
+ if (testCases[i].test(fileName))
+ {
+ ndbout << "-- Passed --" << endl;
+ }
+ else
+ {
+ ndbout << "-- Failed -- " << endl;
+ }
+
+ }
+ ndbout << endl << "-- " << testCount - testFailed << " passed, "
+ << testFailed << " failed --" << endl;
+ return 0;
+}
+
+
+bool
+FileUnitTest::testWrite(const char* aFileName)
+{
+ bool rc = true;
+ File f;
+ if (f.open(aFileName, "w"))
+ {
+ f.writeChar("ABABABABABAB ABBABAB ABBABA ABAB JKH KJHA JHHAHAH...");
+ f.writeChar("12129791242 1298371923 912738912 378129837128371128132...\n");
+ f.close();
+ }
+ else
+ {
+ error("testWrite failed: ");
+ rc = false;
+ }
+ return rc;
+}
+
+bool
+FileUnitTest::testRead(const char* aFileName)
+{
+ bool rc = true;
+ // Read file
+ File f;
+ if (f.open(aFileName, "r"))
+ {
+ long size = f.size();
+ ndbout << "File size = " << size << endl;
+ ndbout << "Allocating buf of " << size << " bytes" << endl;
+ char* buf = new char[size];
+ buf[size - 1] = '\0';
+ int r = 0;
+ while ((r = f.readChar(buf, r, size)) > 0)
+ {
+ ndbout << "Read(" << r << "):" << buf << endl;
+ }
+ f.close();
+ delete buf;
+ }
+ else
+ {
+ error("readTest failed: ");
+ rc = false;
+ }
+ return rc;
+}
+
+bool
+FileUnitTest::testExists(const char* aFileName)
+{
+ bool rc = true;
+ if (File::exists(aFileName))
+ {
+ if (File::exists("ThisFileShouldnotbe.txt"))
+ {
+ rc = false;
+ error("testExists failed, the file should NOT be found.");
+ }
+ }
+ else
+ {
+ rc = false;
+ error("testExists failed, the file should exist.");
+ }
+
+ return rc;
+}
+
+
+bool
+FileUnitTest::testSize(const char* aFileName)
+{
+ bool rc = true;
+ File f;
+ if (f.open(aFileName, "r"))
+ {
+ long size = f.size();
+ if (size <= 0)
+ {
+ rc = false;
+ error("testSize failed, size is <= 0");
+ }
+ ndbout << "File size = " << size << endl;
+ }
+ else
+ {
+ rc = false;
+ error("testSize failed, could no open file.");
+ }
+ f.close();
+ return rc;
+}
+
+bool
+FileUnitTest::testRename(const char* aFileName)
+{
+ bool rc = true;
+ if (File::rename(aFileName, "filetest_new.txt"))
+ {
+ if (!File::exists("filetest_new.txt"))
+ {
+ rc = false;
+ error("testRename failed, new file does not exists.");
+ }
+ else
+ {
+ ndbout << "Renamed " << aFileName << " to filetest_new.txt" << endl;
+ }
+ }
+ else
+ {
+ rc = false;
+ error("testRename failed, unable to rename file.");
+ }
+
+ return rc;
+}
+
+bool
+FileUnitTest::testRemove(const char* aFileName)
+{
+ bool rc = true;
+ File f;
+ if (f.open("filetest_new.txt", "r"))
+ {
+ if (!f.remove())
+ {
+ rc = false;
+ error("testRemove failed, could not remove file.");
+ }
+ else
+ {
+ if (File::exists("filetest_new"))
+ {
+ rc = false;
+ error("testRemove failed, file was not removed, it still exists.");
+ }
+ }
+ } // (f.open("filetest_new", "r"))
+ else
+ {
+ rc = false;
+ error("testRemove failed, could not read the file.");
+ }
+
+ return rc;
+}
+
+void
+FileUnitTest::error(const char* msg)
+{
+ testFailed++;
+ ndbout << "Test failed: " << msg << endl;
+ perror("Errno msg");
+}
+
+
+FileUnitTest::FileUnitTest()
+{
+
+}
+
+FileUnitTest::~FileUnitTest()
+{
+
+}
diff --git a/ndb/src/common/util/filetest/FileUnitTest.hpp b/ndb/src/common/util/filetest/FileUnitTest.hpp
new file mode 100644
index 00000000000..a589615e9b2
--- /dev/null
+++ b/ndb/src/common/util/filetest/FileUnitTest.hpp
@@ -0,0 +1,41 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FILEUNITTEST_H
+#define FILEUNITTEST_H
+
+/**
+ * Unit test of File.
+ *
+ * @version #@ $Id: FileUnitTest.hpp,v 1.1 2002/03/13 18:09:03 eyualex Exp $
+ */
+class FileUnitTest
+{
+public:
+ static bool testWrite(const char* aFileName);
+ static bool testRead(const char* aFileName);
+ static bool testExists(const char* aFileName);
+ static bool testSize(const char* aFileName);
+ static bool testRename(const char* aFileName);
+ static bool testRemove(const char* aFileName);
+
+ static void error(const char* msg);
+private:
+ FileUnitTest();
+ ~FileUnitTest();
+
+};
+#endif
diff --git a/ndb/src/common/util/filetest/Makefile b/ndb/src/common/util/filetest/Makefile
new file mode 100644
index 00000000000..fe1842921f9
--- /dev/null
+++ b/ndb/src/common/util/filetest/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE :=
+
+BIN_TARGET := filetest
+BIN_TARGET_ARCHIVES := portlib general
+
+SOURCES := FileUnitTest.cpp
+
+CCFLAGS_LOC += -I$(NDB_TOP)/include/logger -I$(NDB_TOP)/include/portlib
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/src/common/util/getarg.3 b/ndb/src/common/util/getarg.3
new file mode 100644
index 00000000000..43aae5d7b31
--- /dev/null
+++ b/ndb/src/common/util/getarg.3
@@ -0,0 +1,315 @@
+.\" Copyright (c) 1999 Kungliga Tekniska Högskolan
+.\" $KTH: getarg.3,v 1.1.4.1 2001/07/26 19:54:45 lha Exp $
+.Dd September 24, 1999
+.Dt GETARG 3
+.Os ROKEN
+.Sh NAME
+.Nm getarg ,
+.Nm arg_printusage
+.Nd collect command line options
+.Sh SYNOPSIS
+.Fd #include <getarg.h>
+
+.Ft int
+.Fn getarg "struct getargs *args" "size_t num_args" "int argc" "char **argv" "int *optind"
+
+.Ft void
+.Fn arg_printusage "struct getargs *args" "size_t num_args" "const char *progname" "const char *extra_string"
+
+.Sh DESCRIPTION
+.Fn getarg
+collects any command line options given to a program in an easily used way.
+.Fn arg_printusage
+pretty-prints the available options, with a short help text.
+.Pp
+.Fa args
+is the option specification to use, and it's an array of
+.Fa struct getargs
+elements.
+.Fa num_args
+is the size of
+.Fa args
+(in elements).
+.Fa argc
+and
+.Fa argv
+are the argument count and argument vector to extract option from.
+.Fa optind
+is a pointer to an integer where the index to the last processed
+argument is stored, it must be initialised to the first index (minus
+one) to process (normally 0) before the first call.
+.Pp
+.Fa arg_printusage
+take the same
+.Fa args
+and
+.Fa num_args
+as getarg;
+.Fa progname is the name of the program (to be used in the help text), and
+.Fa extra_string
+is a string to print after the actual options to indicate more
+arguments. The usefulness of this function is realised only be people
+who has used programs that has help strings that doesn't match what
+the code does.
+.Pp
+The
+.Fa getargs
+struct has the following elements.
+
+.Bd -literal
+struct getargs{
+ const char *long_name;
+ char short_name;
+ enum { arg_integer,
+ arg_string,
+ arg_flag,
+ arg_negative_flag,
+ arg_strings,
+ arg_double,
+ arg_collect
+ } type;
+ void *value;
+ const char *help;
+ const char *arg_help;
+};
+.Ed
+.Pp
+.Fa long_name
+is the long name of the option, it can be
+.Dv NULL ,
+if you don't want a long name.
+.Fa short_name
+is the characted to use as short option, it can be zero. If the option
+has a value the
+.Fa value
+field gets filled in with that value interpreted as specified by the
+.Fa type
+field.
+.Fa help
+is a longer help string for the option as a whole, if it's
+.Dv NULL
+the help text for the option is omitted (but it's still displayed in
+the synopsis).
+.Fa arg_help
+is a description of the argument, if
+.Dv NULL
+a default value will be used, depending on the type of the option:
+.Pp
+.Bl -hang -width arg_negative_flag
+.It arg_integer
+the argument is a signed integer, and
+.Fa value
+should point to an
+.Fa int .
+.It Fa arg_string
+the argument is a string, and
+.Fa value
+should point to a
+.Fa char* .
+.It Fa arg_flag
+the argument is a flag, and
+.Fa value
+should point to a
+.Fa int .
+It gets filled in with either zero or one, depending on how the option
+is given, the normal case beeing one. Note that if the option isn't
+given, the value isn't altered, so it should be initialised to some
+useful default.
+.It Fa arg_negative_flag
+this is the same as
+.Fa arg_flag
+but it reverses the meaning of the flag (a given short option clears
+the flag), and the synopsis of a long option is negated.
+.It Fa arg_strings
+the argument can be given multiple times, and the values are collected
+in an array;
+.Fa value
+should be a pointer to a
+.Fa struct getarg_strings
+structure, which holds a length and a string pointer.
+.It Fa arg_double
+argument is a double precision floating point value, and
+.Fa value
+should point to a
+.Fa double .
+.It Fa arg_collect
+allows more fine-grained control of the option parsing process.
+.Fa value
+should be a pointer to a
+.Fa getarg_collect_info
+structure:
+.Bd -literal
+typedef int (*getarg_collect_func)(int short_opt,
+ int argc,
+ char **argv,
+ int *optind,
+ int *optarg,
+ void *data);
+
+typedef struct getarg_collect_info {
+ getarg_collect_func func;
+ void *data;
+} getarg_collect_info;
+.Ed
+.Pp
+With the
+.Fa func
+member set to a function to call, and
+.Fa data
+to some application specific data. The parameters to the collect function are:
+.Bl -inset
+.It Fa short_flag
+non-zero if this call is via a short option flag, zero otherwise
+.It Fa argc , argv
+the whole argument list
+.It Fa optind
+pointer to the index in argv where the flag is
+.It Fa optarg
+pointer to the index in argv[*optind] where the flag name starts
+.It Fa data
+application specific data
+.El
+.Pp
+You can modify
+.Fa *optind ,
+and
+.Fa *optarg ,
+but to do this correct you (more or less) have to know about the inner
+workings of getarg.
+
+You can skip parts of arguments by increasing
+.Fa *optarg
+(you could
+implement the
+.Fl z Ns Ar 3
+set of flags from
+.Nm gzip
+with this), or whole argument strings by increasing
+.Fa *optind
+(let's say you want a flag
+.Fl c Ar x y z
+to specify a coordinate); if you also have to set
+.Fa *optarg
+to a sane value.
+.Pp
+The collect function should return one of
+.Dv ARG_ERR_NO_MATCH , ARG_ERR_BAD_ARG , ARG_ERR_NO_ARG
+on error, zero otherwise.
+.Pp
+For your convenience there is a function,
+.Fn getarg_optarg ,
+that returns the traditional argument string, and you pass it all
+arguments, sans data, that where given to the collection function.
+.Pp
+Don't use this more this unless you absolutely have to.
+.El
+.Pp
+Option parsing is similar to what
+.Xr getopt
+uses. Short options without arguments can be compressed
+.Pf ( Fl xyz
+is the same as
+.Fl x y z ) ,
+and short
+options with arguments take these as either the rest of the
+argv-string or as the next option
+.Pf ( Fl o Ns Ar foo ,
+or
+.Fl o Ar foo ) .
+.Pp
+Long option names are prefixed with -- (double dash), and the value
+with a = (equal),
+.Fl -foo= Ns Ar bar .
+Long option flags can either be specified as they are
+.Pf ( Fl -help ) ,
+or with an (boolean parsable) option
+.Pf ( Fl -help= Ns Ar yes ,
+.Fl -help= Ns Ar true ,
+or similar), or they can also be negated
+.Pf ( Fl -no-help
+is the same as
+.Fl -help= Ns no ) ,
+and if you're really confused you can do it multiple times
+.Pf ( Fl -no-no-help= Ns Ar false ,
+or even
+.Fl -no-no-help= Ns Ar maybe ) .
+.Sh EXAMPLE
+.Bd -literal
+#include <stdio.h>
+#include <string.h>
+#include <getarg.h>
+
+char *source = "Ouagadougou";
+char *destination;
+int weight;
+int include_catalog = 1;
+int help_flag;
+
+struct getargs args[] = {
+ { "source", 's', arg_string, &source,
+ "source of shippment", "city" },
+ { "destination", 'd', arg_string, &destination,
+ "destination of shippment", "city" },
+ { "weight", 'w', arg_integer, &weight,
+ "weight of shippment", "tons" },
+ { "catalog", 'c', arg_negative_flag, &include_catalog,
+ "include product catalog" },
+ { "help", 'h', arg_flag, &help_flag }
+};
+
+int num_args = sizeof(args) / sizeof(args[0]); /* number of elements in args */
+
+const char *progname = "ship++";
+
+int
+main(int argc, char **argv)
+{
+ int optind = 0;
+ if (getarg(args, num_args, argc, argv, &optind)) {
+ arg_printusage(args, num_args, progname, "stuff...");
+ exit (1);
+ }
+ if (help_flag) {
+ arg_printusage(args, num_args, progname, "stuff...");
+ exit (0);
+ }
+ if (destination == NULL) {
+ fprintf(stderr, "%s: must specify destination\n", progname);
+ exit(1);
+ }
+ if (strcmp(source, destination) == 0) {
+ fprintf(stderr, "%s: destination must be different from source\n");
+ exit(1);
+ }
+ /* include more stuff here ... */
+ exit(2);
+}
+.Ed
+.Pp
+The output help output from this program looks like this:
+.Bd -literal
+$ ship++ --help
+Usage: ship++ [--source=city] [-s city] [--destination=city] [-d city]
+ [--weight=tons] [-w tons] [--no-catalog] [-c] [--help] [-h] stuff...
+-s city, --source=city source of shippment
+-d city, --destination=city destination of shippment
+-w tons, --weight=tons weight of shippment
+-c, --no-catalog include product catalog
+.Ed
+
+.Sh BUGS
+It should be more flexible, so it would be possible to use other more
+complicated option syntaxes, such as what
+.Xr ps 1 ,
+and
+.Xr tar 1 ,
+uses, or the AFS model where you can skip the flag names as long as
+the options come in the correct order.
+.Pp
+Options with multiple arguments should be handled better.
+.Pp
+Should be integreated with SL.
+.Pp
+It's very confusing that the struct you pass in is called getargS.
+.Sh SEE ALSO
+.Xr getopt 3
diff --git a/ndb/src/common/util/getarg.3.ps b/ndb/src/common/util/getarg.3.ps
new file mode 100644
index 00000000000..146fb8e4961
--- /dev/null
+++ b/ndb/src/common/util/getarg.3.ps
@@ -0,0 +1,458 @@
+%!PS-Adobe-3.0
+%%Creator: groff version 1.15
+%%CreationDate: Thu Nov 7 12:53:13 2002
+%%DocumentNeededResources: font Times-Roman
+%%+ font Times-Bold
+%%+ font Courier-Bold
+%%+ font Courier-Oblique
+%%+ font Symbol
+%%+ font Courier
+%%DocumentSuppliedResources: procset grops 1.15 0
+%%Pages: 4
+%%PageOrder: Ascend
+%%Orientation: Portrait
+%%EndComments
+%%BeginProlog
+%%BeginResource: procset grops 1.15 0
+/setpacking where{
+pop
+currentpacking
+true setpacking
+}if
+/grops 120 dict dup begin
+/SC 32 def
+/A/show load def
+/B{0 SC 3 -1 roll widthshow}bind def
+/C{0 exch ashow}bind def
+/D{0 exch 0 SC 5 2 roll awidthshow}bind def
+/E{0 rmoveto show}bind def
+/F{0 rmoveto 0 SC 3 -1 roll widthshow}bind def
+/G{0 rmoveto 0 exch ashow}bind def
+/H{0 rmoveto 0 exch 0 SC 5 2 roll awidthshow}bind def
+/I{0 exch rmoveto show}bind def
+/J{0 exch rmoveto 0 SC 3 -1 roll widthshow}bind def
+/K{0 exch rmoveto 0 exch ashow}bind def
+/L{0 exch rmoveto 0 exch 0 SC 5 2 roll awidthshow}bind def
+/M{rmoveto show}bind def
+/N{rmoveto 0 SC 3 -1 roll widthshow}bind def
+/O{rmoveto 0 exch ashow}bind def
+/P{rmoveto 0 exch 0 SC 5 2 roll awidthshow}bind def
+/Q{moveto show}bind def
+/R{moveto 0 SC 3 -1 roll widthshow}bind def
+/S{moveto 0 exch ashow}bind def
+/T{moveto 0 exch 0 SC 5 2 roll awidthshow}bind def
+/SF{
+findfont exch
+[exch dup 0 exch 0 exch neg 0 0]makefont
+dup setfont
+[exch/setfont cvx]cvx bind def
+}bind def
+/MF{
+findfont
+[5 2 roll
+0 3 1 roll
+neg 0 0]makefont
+dup setfont
+[exch/setfont cvx]cvx bind def
+}bind def
+/level0 0 def
+/RES 0 def
+/PL 0 def
+/LS 0 def
+/MANUAL{
+statusdict begin/manualfeed true store end
+}bind def
+/PLG{
+gsave newpath clippath pathbbox grestore
+exch pop add exch pop
+}bind def
+/BP{
+/level0 save def
+1 setlinecap
+1 setlinejoin
+72 RES div dup scale
+LS{
+90 rotate
+}{
+0 PL translate
+}ifelse
+1 -1 scale
+}bind def
+/EP{
+level0 restore
+showpage
+}bind def
+/DA{
+newpath arcn stroke
+}bind def
+/SN{
+transform
+.25 sub exch .25 sub exch
+round .25 add exch round .25 add exch
+itransform
+}bind def
+/DL{
+SN
+moveto
+SN
+lineto stroke
+}bind def
+/DC{
+newpath 0 360 arc closepath
+}bind def
+/TM matrix def
+/DE{
+TM currentmatrix pop
+translate scale newpath 0 0 .5 0 360 arc closepath
+TM setmatrix
+}bind def
+/RC/rcurveto load def
+/RL/rlineto load def
+/ST/stroke load def
+/MT/moveto load def
+/CL/closepath load def
+/FL{
+currentgray exch setgray fill setgray
+}bind def
+/BL/fill load def
+/LW/setlinewidth load def
+/RE{
+findfont
+dup maxlength 1 index/FontName known not{1 add}if dict begin
+{
+1 index/FID ne{def}{pop pop}ifelse
+}forall
+/Encoding exch def
+dup/FontName exch def
+currentdict end definefont pop
+}bind def
+/DEFS 0 def
+/EBEGIN{
+moveto
+DEFS begin
+}bind def
+/EEND/end load def
+/CNT 0 def
+/level1 0 def
+/PBEGIN{
+/level1 save def
+translate
+div 3 1 roll div exch scale
+neg exch neg exch translate
+0 setgray
+0 setlinecap
+1 setlinewidth
+0 setlinejoin
+10 setmiterlimit
+[]0 setdash
+/setstrokeadjust where{
+pop
+false setstrokeadjust
+}if
+/setoverprint where{
+pop
+false setoverprint
+}if
+newpath
+/CNT countdictstack def
+userdict begin
+/showpage{}def
+}bind def
+/PEND{
+clear
+countdictstack CNT sub{end}repeat
+level1 restore
+}bind def
+end def
+/setpacking where{
+pop
+setpacking
+}if
+%%EndResource
+%%IncludeResource: font Times-Roman
+%%IncludeResource: font Times-Bold
+%%IncludeResource: font Courier-Bold
+%%IncludeResource: font Courier-Oblique
+%%IncludeResource: font Symbol
+%%IncludeResource: font Courier
+grops begin/DEFS 1 dict def DEFS begin/u{.001 mul}bind def end/RES 72
+def/PL 792 def/LS false def/ENC0[/asciicircum/asciitilde/Scaron/Zcaron
+/scaron/zcaron/Ydieresis/trademark/quotesingle/.notdef/.notdef/.notdef
+/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef
+/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef/.notdef
+/.notdef/.notdef/space/exclam/quotedbl/numbersign/dollar/percent
+/ampersand/quoteright/parenleft/parenright/asterisk/plus/comma/hyphen
+/period/slash/zero/one/two/three/four/five/six/seven/eight/nine/colon
+/semicolon/less/equal/greater/question/at/A/B/C/D/E/F/G/H/I/J/K/L/M/N/O
+/P/Q/R/S/T/U/V/W/X/Y/Z/bracketleft/backslash/bracketright/circumflex
+/underscore/quoteleft/a/b/c/d/e/f/g/h/i/j/k/l/m/n/o/p/q/r/s/t/u/v/w/x/y
+/z/braceleft/bar/braceright/tilde/.notdef/quotesinglbase/guillemotleft
+/guillemotright/bullet/florin/fraction/perthousand/dagger/daggerdbl
+/endash/emdash/ff/fi/fl/ffi/ffl/dotlessi/dotlessj/grave/hungarumlaut
+/dotaccent/breve/caron/ring/ogonek/quotedblleft/quotedblright/oe/lslash
+/quotedblbase/OE/Lslash/.notdef/exclamdown/cent/sterling/currency/yen
+/brokenbar/section/dieresis/copyright/ordfeminine/guilsinglleft
+/logicalnot/minus/registered/macron/degree/plusminus/twosuperior
+/threesuperior/acute/mu/paragraph/periodcentered/cedilla/onesuperior
+/ordmasculine/guilsinglright/onequarter/onehalf/threequarters
+/questiondown/Agrave/Aacute/Acircumflex/Atilde/Adieresis/Aring/AE
+/Ccedilla/Egrave/Eacute/Ecircumflex/Edieresis/Igrave/Iacute/Icircumflex
+/Idieresis/Eth/Ntilde/Ograve/Oacute/Ocircumflex/Otilde/Odieresis
+/multiply/Oslash/Ugrave/Uacute/Ucircumflex/Udieresis/Yacute/Thorn
+/germandbls/agrave/aacute/acircumflex/atilde/adieresis/aring/ae/ccedilla
+/egrave/eacute/ecircumflex/edieresis/igrave/iacute/icircumflex/idieresis
+/eth/ntilde/ograve/oacute/ocircumflex/otilde/odieresis/divide/oslash
+/ugrave/uacute/ucircumflex/udieresis/yacute/thorn/ydieresis]def
+/Courier@0 ENC0/Courier RE/Courier-Oblique@0 ENC0/Courier-Oblique RE
+/Courier-Bold@0 ENC0/Courier-Bold RE/Times-Bold@0 ENC0/Times-Bold RE
+/Times-Roman@0 ENC0/Times-Roman RE
+%%EndProlog
+%%Page: 1 1
+%%BeginPageSetup
+BP
+%%EndPageSetup
+/F0 10/Times-Roman@0 SF(GET)72 48 Q -.834(ARG \( 3 \))-.93 F
+(OpenBSD Programmer')111.062 E 2.5(sM)-.55 G 108.562(anual GET)-2.5 F
+-.834(ARG \( 3 \))-.93 F/F1 10/Times-Bold@0 SF -.2(NA)72 108 S(ME).2 E
+/F2 10/Courier-Bold@0 SF(getarg)102 120 Q F0(,)A F2(arg_printusage)2.5 E
+F0 2.5<ad63>2.5 G(ollect command line options)-2.5 E F1(SYNOPSIS)72 144
+Q F2(#include <getarg.h>)102 156 Q/F3 10/Courier-Oblique@0 SF(int)102
+186 Q F2(getarg)102 198 Q F0(\()A F3(struct getargs)A/F4 10/Symbol SF(*)
+6 E F3(args)A F0(,)1.666 E F3(size_t num_args)4.166 E F0(,)1.666 E F3
+(int argc)4.166 E F0(,)1.666 E F3(char)4.166 E F4(**)6 E F3(argv)A F0(,)
+1.666 E F3(int)151.666 210 Q F4(*)6 E F3(optind)A F0(\);)A F3(void)102
+240 Q F2(arg_printusage)102 252 Q F0(\()A F3(struct getargs)A F4(*)6 E
+F3(args)A F0(,)1.666 E F3(size_t num_args)4.166 E F0(,)1.666 E F3
+(const char)4.166 E F4(*)6 E F3(progname)A F0(,)1.666 E F3(const char)
+151.666 264 Q F4(*)6 E F3(extra_string)A F0(\);)A F1(DESCRIPTION)72 300
+Q F2(getarg)102 312 Q F0 2.721 1.666(\(\) c)D 6.053(ollects an)-1.666 F
+8.553(yc)-.15 G 6.053(ommand line options gi)-8.553 F -.15(ve)-.25 G
+8.552(nt).15 G 8.552(oap)-8.552 G 6.052(rogram in an easily used w)
+-8.552 F(ay)-.1 E(.)-.65 E F2(arg_printusage)102 324 Q F0 -3.332 1.666
+(\(\) p)D(retty-prints the a)-1.666 E -.25(va)-.2 G
+(ilable options, with a short help te).25 E(xt.)-.15 E F3(args)102 342 Q
+F0 .855(is the option speci\214cation to use, and it')3.355 F 3.356(sa)
+-.55 G 3.356(na)-3.356 G .856(rray of)-3.356 F F3 .856(struct getargs)
+3.356 F F0(elements.)3.356 E F3(num_args)5.856 E F0(is)3.356 E .344
+(the size of)102 354 R F3(args)2.844 E F0 .344(\(in elements\).)2.844 F
+F3(argc)5.344 E F0(and)2.844 E F3(argv)2.844 E F0 .344(are the ar)2.844
+F .344(gument count and ar)-.18 F .344(gument v)-.18 F .344(ector to e)
+-.15 F .343(xtract op-)-.15 F 1.127(tion from.)102 366 R F3(optind)6.127
+E F0 1.127(is a pointer to an inte)3.627 F 1.127(ger where the inde)-.15
+F 3.627(xt)-.15 G 3.628(ot)-3.627 G 1.128(he last processed ar)-3.628 F
+1.128(gument is stored, it)-.18 F
+(must be initialised to the \214rst inde)102 378 Q 2.5(x\()-.15 G
+(minus one\) to process \(normally 0\) before the \214rst call.)-2.5 E
+F3(arg_printusage)102 396 Q F0(tak)4.178 E 4.178(et)-.1 G 1.678(he same)
+-4.178 F F3(args)4.178 E F0(and)4.178 E F3(num_args)4.178 E F0 1.678
+(as getar)4.178 F(g;)-.18 E F3 1.677(progname is the name of)4.178 F
+6.381(the program \(to be)102 408 R F0(progname0)12.381 E F3(0)12.381 E
+F0(progname1)A F3(1)12.381 E F0(progname2)A F3(2)12.382 E F0(progname3)A
+F3(3)12.382 E F0(progname4)A F3(4)102 420 Q F0(progname5)A F3
+(extra_string)3.404 E F0 .904
+(is a string to print after the actual options to indicate more ar)3.404
+F .904(guments. The)-.18 F .025(usefulness of this function is realised\
+ only be people who has used programs that has help strings that doesn')
+102 432 R(t)-.18 E(match what the code does.)102 444 Q(The)102 462 Q F3
+(getargs)2.5 E F0(struct has the follo)2.5 E(wing elements.)-.25 E/F5 10
+/Courier@0 SF(struct getargs{)102 504 Q(const char)126 516 Q F4(*)6 E F5
+(long_name;)A(char short_name;)126 528 Q(enum { arg_integer,)126 540 Q
+(arg_string,)165 552 Q(arg_flag,)165 564 Q(arg_negative_flag,)165 576 Q
+(arg_strings,)165 588 Q(arg_double,)165 600 Q(arg_collect)168 612 Q 6
+(}t)126 624 S(ype;)-6 E(void)126 636 Q F4(*)6 E F5(value;)A(const char)
+126 648 Q F4(*)6 E F5(help;)A(const char)126 660 Q F4(*)6 E F5
+(arg_help;)A(};)102 672 Q F3(long_name)102 690 Q F0 .207
+(is the long name of the option, it can be)2.707 F F5(NULL)2.706 E F0
+2.706(,i)C 2.706(fy)-2.706 G .206(ou don')-2.706 F 2.706(tw)-.18 G .206
+(ant a long name.)-2.806 F F3(short_name)5.206 E F0 .397(is the charact\
+ed to use as short option, it can be zero. If the option has a v)102 702
+R .398(alue the)-.25 F F3(value)2.898 E F0 .398
+(\214eld gets \214lled in)2.898 F -.4(RO)77 750 S 152.325(KEN September)
+.4 F(24, 1999)2.5 E(1)188.865 E EP
+%%Page: 2 2
+%%BeginPageSetup
+BP
+%%EndPageSetup
+/F0 10/Times-Roman@0 SF(GET)72 48 Q -.834(ARG \( 3 \))-.93 F
+(OpenBSD Programmer')111.062 E 2.5(sM)-.55 G 108.562(anual GET)-2.5 F
+-.834(ARG \( 3 \))-.93 F .737(with that v)102 96 R .737
+(alue interpreted as speci\214ed by the)-.25 F/F1 10/Courier-Oblique@0
+SF(type)3.237 E F0(\214eld.)3.237 E F1(help)5.737 E F0 .737
+(is a longer help string for the option as a)3.237 F 2.833
+(whole, if it')102 108 R(s)-.55 E/F2 10/Courier@0 SF(NULL)5.333 E F0
+2.833(the help te)5.333 F 2.833(xt for the option is omitted \(b)-.15 F
+2.834(ut it')-.2 F 5.334(ss)-.55 G 2.834
+(till displayed in the synopsis\).)-5.334 F F1(arg_help)102 120 Q F0
+.391(is a description of the ar)2.891 F .391(gument, if)-.18 F F2(NULL)
+2.891 E F0 2.891(ad)2.891 G(ef)-2.891 E .39(ault v)-.1 F .39
+(alue will be used, depending on the type of)-.25 F(the option:)102 132
+Q(ar)102 150 Q(g_inte)-.18 E 59.29(ger the)-.15 F(ar)2.5 E
+(gument is a signed inte)-.18 E(ger)-.15 E 2.5(,a)-.4 G(nd)-2.5 E F1
+(value)2.5 E F0(should point to an)2.5 E F1(int)2.5 E F0(.)A F1
+(arg_string)102 168 Q F0(the ar)47 E(gument is a string, and)-.18 E F1
+(value)2.5 E F0(should point to a)2.5 E F1(char)2.5 E/F3 10/Symbol SF(*)
+A F0(.)A F1(arg_flag)102 186 Q F0 .4(the ar)59 F .4
+(gument is a \215ag, and)-.18 F F1(value)2.9 E F0 .4(should point to a)
+2.9 F F1(int)2.9 E F0 2.9(.I)C 2.9(tg)-2.9 G .4
+(ets \214lled in with ei-)-2.9 F 1.154
+(ther zero or one, depending on ho)209 198 R 3.654(wt)-.25 G 1.153
+(he option is gi)-3.654 F -.15(ve)-.25 G 1.153
+(n, the normal case beeing).15 F .526(one. Note that if the option isn')
+209 210 R 3.026(tg)-.18 G -2.15 -.25(iv e)-3.026 H .526(n, the v).25 F
+.526(alue isn')-.25 F 3.026(ta)-.18 G .527(ltered, so it should be ini-)
+-3.026 F(tialised to some useful def)209 222 Q(ault.)-.1 E F1
+(arg_negative_flag)102 240 Q F0 .058(this is the same as)2.558 F F1
+(arg_flag)2.558 E F0 -.2(bu)2.558 G 2.558(ti).2 G 2.558(tr)-2.558 G
+-2.15 -.25(ev e)-2.558 H .057(rses the meaning of the \215ag \(a gi).25
+F -.15(ve)-.25 G 2.557(ns).15 G(hort)-2.557 E
+(option clears the \215ag\), and the synopsis of a long option is ne)209
+252 Q -.05(ga)-.15 G(ted.).05 E F1(arg_strings)102 270 Q F0 .195(the ar)
+41 F .195(gument can be gi)-.18 F -.15(ve)-.25 G 2.695(nm).15 G .195
+(ultiple times, and the v)-2.695 F .195
+(alues are collected in an array;)-.25 F F1(value)209 282 Q F0 .947
+(should be a pointer to a)3.447 F F1 .947(struct getarg_strings)3.447 F
+F0 .947(structure, which)3.447 F(holds a length and a string pointer)209
+294 Q(.)-.55 E F1(arg_double)102 312 Q F0(ar)47 E .538
+(gument is a double precision \215oating point v)-.18 F .539(alue, and)
+-.25 F F1(value)3.039 E F0 .539(should point to a)3.039 F F1(double)209
+324 Q F0(.)A F1(arg_collect)102 342 Q F0(allo)41 E .345
+(ws more \214ne-grained control of the option parsing process.)-.25 F F1
+(value)5.344 E F0 .344(should be)2.844 F 2.5(ap)209 354 S(ointer to a)
+-2.5 E F1(getarg_collect_info)2.5 E F0(structure:)2.5 E F2
+(typedef int \()209 372 Q F3(*)A F2
+(getarg_collect_func\)\(int short_opt,)A(int argc,)407 384 Q(char)407
+396 Q F3(**)6 E F2(argv,)A(int)407 408 Q F3(*)6 E F2(optind,)A(int)407
+420 Q F3(*)6 E F2(optarg,)A(void)407 432 Q F3(*)6 E F2(data\);)A
+(typedef struct getarg_collect_info {)209 456 Q
+(getarg_collect_func func;)233 468 Q(void)233 480 Q F3(*)6 E F2(data;)A
+6(}g)209 492 S(etarg_collect_info;)-6 E F0 -.4(Wi)209 510 S 1.018
+(th the).4 F F1(func)3.518 E F0 1.019
+(member set to a function to call, and)3.518 F F1(data)3.519 E F0 1.019
+(to some application)3.519 F
+(speci\214c data. The parameters to the collect function are:)209 522 Q
+F1(short_flag)209 540 Q F0
+(non-zero if this call is via a short option \215ag, zero otherwise)2.5
+E F1(argc)209 558 Q F0(,)A F1(argv)6 E F0(the whole ar)2.5 E
+(gument list)-.18 E F1(optind)209 576 Q F0(pointer to the inde)2.5 E 2.5
+(xi)-.15 G 2.5(na)-2.5 G -.18(rg)-2.5 G 2.5(vw).18 G(here the \215ag is)
+-2.5 E F1(optarg)209 594 Q F0(pointer to the inde)2.5 E 2.5(xi)-.15 G
+2.5(na)-2.5 G -.18(rg)-2.5 G(v[).18 E F3(*)A F0
+(optind] where the \215ag name starts)A F1(data)209 612 Q F0
+(application speci\214c data)2.5 E -1.1(Yo)209 630 S 3.915(uc)1.1 G
+1.415(an modify)-3.915 F F3(*)3.915 E F1(optind)A F0 3.915(,a)C(nd)
+-3.915 E F3(*)3.915 E F1(optarg)A F0 3.915(,b)C 1.414
+(ut to do this correct you \(more or)-4.115 F(less\) ha)209 642 Q .3
+-.15(ve t)-.2 H 2.5(ok).15 G(no)-2.5 E 2.5(wa)-.25 G(bout the inner w)
+-2.5 E(orkings of getar)-.1 E(g.)-.18 E -1.1(Yo)209 666 S 3.604(uc)1.1 G
+1.104(an skip parts of ar)-3.604 F 1.105(guments by increasing)-.18 F F3
+(*)3.605 E F1(optarg)A F0 1.105(\(you could implement)3.605 F(the)209
+678 Q/F4 10/Courier-Bold@0 SF<ad7a>4.567 E F1(3)A F0 .401
+(set of \215ags from)2.901 F F4(gzip)2.9 E F0 .4
+(with this\), or whole ar)2.9 F .4(gument strings by increas-)-.18 F
+(ing)209 690 Q F3(*)3.275 E F1(optind)A F0(\(let')3.275 E 3.276(ss)-.55
+G .776(ay you w)-3.276 F .776(ant a \215ag)-.1 F F4<ad63>4.942 E F1
+6.776(xyz)6.776 G F0 .776(to specify a coordinate\); if)-3.5 F
+(you also ha)209 702 Q .3 -.15(ve t)-.2 H 2.5(os).15 G(et)-2.5 E F3(*)
+2.5 E F1(optarg)A F0(to a sane v)2.5 E(alue.)-.25 E -.4(RO)77 750 S
+152.325(KEN September).4 F(24, 1999)2.5 E(2)188.865 E EP
+%%Page: 3 3
+%%BeginPageSetup
+BP
+%%EndPageSetup
+/F0 10/Times-Roman@0 SF(GET)72 48 Q -.834(ARG \( 3 \))-.93 F
+(OpenBSD Programmer')111.062 E 2.5(sM)-.55 G 108.562(anual GET)-2.5 F
+-.834(ARG \( 3 \))-.93 F 9.449
+(The collect function should return one of)209 96 R/F1 10/Courier@0 SF
+(ARG_ERR_NO_MATCH)11.948 E F0(,)A F1(ARG_ERR_BAD_ARG)209 108 Q F0(,)A F1
+(ARG_ERR_NO_ARG)6 E F0(on error)2.5 E 2.5(,z)-.4 G(ero otherwise.)-2.5 E
+-.15(Fo)209 126 S 4.042(ry).15 G 1.542(our con)-4.042 F -.15(ve)-.4 G
+1.542(nience there is a function,).15 F/F2 10/Courier-Bold@0 SF
+(getarg_optarg)4.042 E F0 1.542(\(\), that returns the)B 1.251
+(traditional ar)209 138 R 1.251(gument string, and you pass it all ar)
+-.18 F 1.251(guments, sans data, that where)-.18 F(gi)209 150 Q -.15(ve)
+-.25 G 2.5(nt).15 G 2.5(ot)-2.5 G(he collection function.)-2.5 E(Don')
+209 168 Q 2.5(tu)-.18 G(se this more this unless you absolutely ha)-2.5
+E .3 -.15(ve t)-.2 H(o.).15 E .213(Option parsing is similar to what)102
+186 R F1(getopt)2.713 E F0 .214(uses. Short options without ar)2.714 F
+.214(guments can be compressed \()-.18 F F2(\255xyz)1.666 E F0 .207
+(is the same as)102 198 R F2 1.873<ad7820ad7920ad7a>4.373 F F0 .207
+(\), and short options with ar)B .207(guments tak)-.18 F 2.706(et)-.1 G
+.206(hese as either the rest of the ar)-2.706 F(gv-string)-.18 E
+(or as the ne)102 210 Q(xt option \()-.15 E F2<ad6f>1.666 E/F3 10
+/Courier-Oblique@0 SF(foo)A F0 2.5(,o)C(r)-2.5 E F2<ad6f>4.166 E F3(foo)
+6 E F0(\).)A .78(Long option names are pre\214x)102 228 R .781
+(ed with -- \(double dash\), and the v)-.15 F .781
+(alue with a = \(equal\),)-.25 F F2(\255-foo=)4.947 E F3(bar)A F0 3.281
+(.L)C(ong)-3.281 E 3.815
+(option \215ags can either be speci\214ed as the)102 240 R 6.315(ya)-.15
+G 3.815(re \()-6.315 F F2(\255-help)1.666 E F0 3.815
+(\), or with an \(boolean parsable\) option)B(\()102 252 Q F2
+(\255-help=)1.666 E F3(yes)A F0(,)A F2(\255-help=)5.659 E F3(true)A F0
+3.993(,o)C 3.993(rs)-3.993 G 1.493(imilar\), or the)-3.993 F 3.993(yc)
+-.15 G 1.493(an also be ne)-3.993 F -.05(ga)-.15 G 1.493(ted \().05 F F2
+(\255-no-help)1.666 E F0 1.493(is the same as)3.993 F F2(\255-help=)
+103.666 264 Q F0 1.363(no\), and if you')B 1.362
+(re really confused you can do it multiple times \()-.5 F F2
+(\255-no-no-help=)1.666 E F3(false)A F0 3.862(,o)C(r)-3.862 E -2.15 -.25
+(ev e)102 276 T(n).25 E F2(\255-no-no-help=)4.166 E F3(maybe)A F0(\).)A
+/F4 10/Times-Bold@0 SF(EXAMPLE)72 300 Q F1(#include <stdio.h>)102 330 Q
+(#include <string.h>)102 342 Q(#include <getarg.h>)102 354 Q(char)102
+378 Q/F5 10/Symbol SF(*)6 E F1(source = "Ouagadougou";)A(char)102 390 Q
+F5(*)6 E F1(destination;)A(int weight;)102 402 Q
+(int include_catalog = 1;)102 414 Q(int help_flag;)102 426 Q
+(struct getargs args[] = {)102 450 Q 6({")126 462 S 30(source", 's',)-6
+F 6(arg_string, &source,)6 F("source of shippment", "city" },)138 474 Q
+6({")126 486 S(destination", 'd', arg_string,)-6 E(&destination,)12 E
+("destination of shippment", "city" },)138 498 Q 6({")126 510 S 30
+(weight", 'w',)-6 F(arg_integer, &weight,)6 E
+("weight of shippment", "tons" },)138 522 Q 6({")126 534 S 24
+(catalog", 'c',)-6 F(arg_negative_flag, &include_catalog,)6 E
+("include product catalog" },)138 546 Q 6({")126 558 S 42(help", 'h',)-6
+F(arg_flag, &help_flag })6 E(};)102 570 Q
+(int num_args = sizeof\(args\) / sizeof\(args[0]\); /)102 594 Q F5(*)A
+F1(number of elements in args)6 E F5(*)6 E F1(/)A(const char)102 618 Q
+F5(*)6 E F1(progname = "ship++";)A(int)102 642 Q(main\(int argc, char)
+102 654 Q F5(**)6 E F1(argv\))A({)102 666 Q(int optind = 0;)126 678 Q
+(if \(getarg\(args, num_args, argc, argv, &optind\)\) {)126 690 Q
+(arg_printusage\(args, num_args, progname, "stuff..."\);)147 702 Q F0
+-.4(RO)77 750 S 152.325(KEN September).4 F(24, 1999)2.5 E(3)188.865 E EP
+%%Page: 4 4
+%%BeginPageSetup
+BP
+%%EndPageSetup
+/F0 10/Times-Roman@0 SF(GET)72 48 Q -.834(ARG \( 3 \))-.93 F
+(OpenBSD Programmer')111.062 E 2.5(sM)-.55 G 108.562(anual GET)-2.5 F
+-.834(ARG \( 3 \))-.93 F/F1 10/Courier@0 SF(exit \(1\);)147 96 Q(})126
+108 Q(if \(help_flag\) {)126 120 Q
+(arg_printusage\(args, num_args, progname, "stuff..."\);)147 132 Q
+(exit \(0\);)147 144 Q(})126 156 Q(if \(destination == NULL\) {)126 168
+Q(fprintf\(stderr, "%s: must specify destination0, progname\);)147 180 Q
+(exit\(1\);)147 192 Q(})126 204 Q
+(if \(strcmp\(source, destination\) == 0\) {)126 216 Q
+(fprintf\(stderr, "%s: destination must be different from source0\);)147
+228 Q(exit\(1\);)147 240 Q(})126 252 Q(/)126 264 Q/F2 10/Symbol SF(*)A
+F1(include more stuff here ...)6 E F2(*)6 E F1(/)A(exit\(2\);)126 276 Q
+(})102 288 Q F0(The output help output from this program looks lik)102
+306 Q 2.5(et)-.1 G(his:)-2.5 E F1 6($s)102 324 S(hip++ --help)-6 E
+(Usage: ship++ [--source=city] [-s city] [--destination=city] [-d city])
+102 336 Q
+([--weight=tons] [-w tons] [--no-catalog] [-c] [--help] [-h] stuff...)
+120 348 Q(-s city, --source=city)102 360 Q(source of shippment)36 E
+(-d city, --destination=city destination of shippment)102 372 Q
+(-w tons, --weight=tons)102 384 Q(weight of shippment)36 E
+(-c, --no-catalog)102 396 Q(include product catalog)72 E/F3 10
+/Times-Bold@0 SF -.1(BU)72 432 S(GS).1 E F0 .9(It should be more \215e)
+102 444 R .9(xible, so it w)-.15 F .901
+(ould be possible to use other more complicated option syntax)-.1 F .901
+(es, such as)-.15 F(what)102 456 Q F1(ps)3.167 E F0 .667(\(1\), and)B F1
+(tar)3.167 E F0 .666(\(1\), uses, or the AFS model where you can skip t\
+he \215ag names as long as the options)B(come in the correct order)102
+468 Q(.)-.55 E(Options with multiple ar)102 486 Q
+(guments should be handled better)-.18 E(.)-.55 E(Should be inte)102 504
+Q(greated with SL.)-.15 E(It')102 522 Q 2.5(sv)-.55 G
+(ery confusing that the struct you pass in is called getar)-2.65 E(gS.)
+-.18 E F3 1.666(SEE ALSO)72 546 R F1(getopt)102 558 Q F0(\(3\))A -.4(RO)
+77 750 S 152.325(KEN September).4 F(24, 1999)2.5 E(4)188.865 E EP
+%%Trailer
+end
+%%EOF
diff --git a/ndb/src/common/util/getarg.c b/ndb/src/common/util/getarg.c
new file mode 100644
index 00000000000..5f792437a65
--- /dev/null
+++ b/ndb/src/common/util/getarg.c
@@ -0,0 +1,599 @@
+/* -*- c-basic-offset: 4; -*- */
+/*
+ * Copyright (c) 1997 - 2000 Kungliga Tekniska Högskolan
+ * (Royal Institute of Technology, Stockholm, Sweden).
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of the Institute nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#include <ndb_types.h>
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+RCSID("$KTH: getarg.c,v 1.23 2000/09/01 21:25:54 lha Exp $");
+#endif
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <time.h>
+#include <ctype.h>
+#include "getarg.h"
+
+#define ISFLAG(X) ((X).type == arg_flag || (X).type == arg_negative_flag)
+
+#ifndef HAVE_STRLCPY
+extern size_t strlcpy (char *dst, const char *src, size_t dst_sz);
+#endif /* !HAVE_STRLCPY */
+
+#ifndef HAVE_STRLCAT
+extern size_t strlcat (char *dst, const char *src, size_t dst_sz);
+#endif /* !HAVE_STRLCAT */
+
+#ifndef max
+#define max(a, b) (a) > (b) ? (a) : (b)
+#endif
+
+#ifdef HAVE___PROGNAME
+extern char *__progname;
+#endif
+
+#ifndef TRUE
+#define TRUE 1
+#endif
+
+#ifndef FALSE
+#define FALSE 0
+#endif
+
+char *
+strupr(char *str)
+{
+ char *s;
+
+ for(s = str; *s; s++)
+ *s = toupper(*s);
+ return str;
+}
+
+static size_t
+print_arg (char *string, size_t len, int mdoc, int longp, struct getargs *arg)
+{
+ const char *s;
+
+ *string = '\0';
+
+ if (ISFLAG(*arg) || (!longp && arg->type == arg_counter))
+ return 0;
+
+ if(mdoc){
+ if(longp)
+ strlcat(string, "= Ns", len);
+ strlcat(string, " Ar ", len);
+ }else
+ if (longp)
+ strlcat (string, "=", len);
+ else
+ strlcat (string, " ", len);
+
+ if (arg->arg_help)
+ s = arg->arg_help;
+ else if (arg->type == arg_integer || arg->type == arg_counter)
+ s = "integer";
+ else if (arg->type == arg_string)
+ s = "string";
+ else if (arg->type == arg_double)
+ s = "float";
+ else
+ s = "<undefined>";
+
+ strlcat(string, s, len);
+ return 1 + strlen(s);
+}
+
+#ifdef GETARGMANDOC
+static void
+mandoc_template(struct getargs *args,
+ size_t num_args,
+ const char *progname,
+ const char *extra_string)
+{
+ size_t i;
+ char timestr[64], cmd[64];
+ char buf[128];
+ const char *p;
+ time_t t;
+
+ printf(".\\\" Things to fix:\n");
+ printf(".\\\" * correct section, and operating system\n");
+ printf(".\\\" * remove Op from mandatory flags\n");
+ printf(".\\\" * use better macros for arguments (like .Pa for files)\n");
+ printf(".\\\"\n");
+ t = time(NULL);
+ strftime(timestr, sizeof(timestr), "%B %e, %Y", localtime(&t));
+ printf(".Dd %s\n", timestr);
+ p = strrchr(progname, '/');
+ if(p) p++; else p = progname;
+ strlcpy(cmd, p, sizeof(cmd));
+ strupr(cmd);
+
+ printf(".Dt %s SECTION\n", cmd);
+ printf(".Os OPERATING_SYSTEM\n");
+ printf(".Sh NAME\n");
+ printf(".Nm %s\n", p);
+ printf(".Nd\n");
+ printf("in search of a description\n");
+ printf(".Sh SYNOPSIS\n");
+ printf(".Nm\n");
+ for(i = 0; i < num_args; i++){
+ /* we seem to hit a limit on number of arguments if doing
+ short and long flags with arguments -- split on two lines */
+ if(ISFLAG(args[i]) ||
+ args[i].short_name == 0 || args[i].long_name == NULL) {
+ printf(".Op ");
+
+ if(args[i].short_name) {
+ print_arg(buf, sizeof(buf), 1, 0, args + i);
+ printf("Fl %c%s", args[i].short_name, buf);
+ if(args[i].long_name)
+ printf(" | ");
+ }
+ if(args[i].long_name) {
+ print_arg(buf, sizeof(buf), 1, 1, args + i);
+ printf("Fl -%s%s%s",
+ args[i].type == arg_negative_flag ? "no-" : "",
+ args[i].long_name, buf);
+ }
+ printf("\n");
+ } else {
+ print_arg(buf, sizeof(buf), 1, 0, args + i);
+ printf(".Oo Fl %c%s \\*(Ba Xo\n", args[i].short_name, buf);
+ print_arg(buf, sizeof(buf), 1, 1, args + i);
+ printf(".Fl -%s%s Oc\n.Xc\n", args[i].long_name, buf);
+ }
+ /*
+ if(args[i].type == arg_strings)
+ fprintf (stderr, "...");
+ */
+ }
+ if (extra_string && *extra_string)
+ printf (".Ar %s\n", extra_string);
+ printf(".Sh DESCRIPTION\n");
+ printf("Supported options:\n");
+ printf(".Bl -tag -width Ds\n");
+ for(i = 0; i < num_args; i++){
+ printf(".It Xo\n");
+ if(args[i].short_name){
+ printf(".Fl %c", args[i].short_name);
+ print_arg(buf, sizeof(buf), 1, 0, args + i);
+ printf("%s", buf);
+ if(args[i].long_name)
+ printf(" Ns ,");
+ printf("\n");
+ }
+ if(args[i].long_name){
+ printf(".Fl -%s%s",
+ args[i].type == arg_negative_flag ? "no-" : "",
+ args[i].long_name);
+ print_arg(buf, sizeof(buf), 1, 1, args + i);
+ printf("%s\n", buf);
+ }
+ printf(".Xc\n");
+ if(args[i].help)
+ printf("%s\n", args[i].help);
+ /*
+ if(args[i].type == arg_strings)
+ fprintf (stderr, "...");
+ */
+ }
+ printf(".El\n");
+ printf(".\\\".Sh ENVIRONMENT\n");
+ printf(".\\\".Sh FILES\n");
+ printf(".\\\".Sh EXAMPLES\n");
+ printf(".\\\".Sh DIAGNOSTICS\n");
+ printf(".\\\".Sh SEE ALSO\n");
+ printf(".\\\".Sh STANDARDS\n");
+ printf(".\\\".Sh HISTORY\n");
+ printf(".\\\".Sh AUTHORS\n");
+ printf(".\\\".Sh BUGS\n");
+}
+#endif /* GETARGMANDOC */
+
+static int
+check_column(FILE *f, int col, int len, int columns)
+{
+ if(col + len > columns) {
+ fprintf(f, "\n");
+ col = fprintf(f, " ");
+ }
+ return col;
+}
+
+void
+arg_printusage (struct getargs *args,
+ size_t num_args,
+ const char *progname,
+ const char *extra_string)
+{
+ unsigned int i;
+ size_t max_len = 0;
+ char buf[128];
+ int col = 0, columns;
+
+#ifdef HAVE___PROGNAME
+ if (progname == NULL)
+ progname = __progname;
+#endif
+ if (progname == NULL)
+ progname = "";
+
+#ifdef GETARGMANDOC
+ if(getenv("GETARGMANDOC")){
+ mandoc_template(args, num_args, progname, extra_string);
+ return;
+ }
+#endif
+
+ columns = 80; /* Always assume that the window is 80 chars wide */
+ col = 0;
+ col += fprintf (stderr, "Usage: %s", progname);
+ for (i = 0; i < num_args; ++i) {
+ size_t len = 0;
+
+ if (args[i].long_name) {
+ buf[0] = '\0';
+ strlcat(buf, "[--", sizeof(buf));
+ len += 2;
+ if(args[i].type == arg_negative_flag) {
+ strlcat(buf, "no-", sizeof(buf));
+ len += 3;
+ }
+ strlcat(buf, args[i].long_name, sizeof(buf));
+ len += strlen(args[i].long_name);
+ len += print_arg(buf + strlen(buf), sizeof(buf) - strlen(buf),
+ 0, 1, &args[i]);
+ strlcat(buf, "]", sizeof(buf));
+ if(args[i].type == arg_strings)
+ strlcat(buf, "...", sizeof(buf));
+ col = check_column(stderr, col, strlen(buf) + 1, columns);
+ col += fprintf(stderr, " %s", buf);
+ }
+ if (args[i].short_name) {
+ snprintf(buf, sizeof(buf), "[-%c", args[i].short_name);
+ len += 2;
+ len += print_arg(buf + strlen(buf), sizeof(buf) - strlen(buf),
+ 0, 0, &args[i]);
+ strlcat(buf, "]", sizeof(buf));
+ if(args[i].type == arg_strings)
+ strlcat(buf, "...", sizeof(buf));
+ col = check_column(stderr, col, strlen(buf) + 1, columns);
+ col += fprintf(stderr, " %s", buf);
+ }
+ if (args[i].long_name && args[i].short_name)
+ len += 2; /* ", " */
+ max_len = max(max_len, len);
+ }
+ if (extra_string) {
+ col = check_column(stderr, col, strlen(extra_string) + 1, columns);
+ fprintf (stderr, " %s\n", extra_string);
+ } else
+ fprintf (stderr, "\n");
+ for (i = 0; i < num_args; ++i) {
+ if (args[i].help) {
+ size_t count = 0;
+
+ if (args[i].short_name) {
+ count += fprintf (stderr, "-%c", args[i].short_name);
+ print_arg (buf, sizeof(buf), 0, 0, &args[i]);
+ count += fprintf(stderr, "%s", buf);
+ }
+ if (args[i].short_name && args[i].long_name)
+ count += fprintf (stderr, ", ");
+ if (args[i].long_name) {
+ count += fprintf (stderr, "--");
+ if (args[i].type == arg_negative_flag)
+ count += fprintf (stderr, "no-");
+ count += fprintf (stderr, "%s", args[i].long_name);
+ print_arg (buf, sizeof(buf), 0, 1, &args[i]);
+ count += fprintf(stderr, "%s", buf);
+ }
+ while(count++ <= max_len)
+ putc (' ', stderr);
+ fprintf (stderr, "%s\n", args[i].help);
+ }
+ }
+}
+
+static void
+add_string(getarg_strings *s, char *value)
+{
+ s->strings = realloc(s->strings, (s->num_strings + 1) * sizeof(*s->strings));
+ s->strings[s->num_strings] = value;
+ s->num_strings++;
+}
+
+static int
+arg_match_long(struct getargs *args, size_t num_args,
+ char *argv, int argc, const char **rargv, int *optind)
+{
+ unsigned int i;
+ const char *optarg = NULL;
+ int negate = 0;
+ int partial_match = 0;
+ struct getargs *partial = NULL;
+ struct getargs *current = NULL;
+ int argv_len;
+ char *p;
+
+ argv_len = strlen(argv);
+ p = strchr (argv, '=');
+ if (p != NULL)
+ argv_len = p - argv;
+
+ for (i = 0; i < num_args; ++i) {
+ if(args[i].long_name) {
+ int len = strlen(args[i].long_name);
+ char *p = argv;
+ int p_len = argv_len;
+ negate = 0;
+
+ for (;;) {
+ if (strncmp (args[i].long_name, p, p_len) == 0) {
+ if(p_len == len)
+ current = &args[i];
+ else {
+ ++partial_match;
+ partial = &args[i];
+ }
+ optarg = p + p_len;
+ } else if (ISFLAG(args[i]) && strncmp (p, "no-", 3) == 0) {
+ negate = !negate;
+ p += 3;
+ p_len -= 3;
+ continue;
+ }
+ break;
+ }
+ if (current)
+ break;
+ }
+ }
+ if (current == NULL) {
+ if (partial_match == 1)
+ current = partial;
+ else
+ return ARG_ERR_NO_MATCH;
+ }
+
+ if(*optarg == '\0'
+ && !ISFLAG(*current)
+ && current->type != arg_collect
+ && current->type != arg_counter)
+ return ARG_ERR_NO_MATCH;
+ switch(current->type){
+ case arg_integer:
+ {
+ int tmp;
+ if(sscanf(optarg + 1, "%d", &tmp) != 1)
+ return ARG_ERR_BAD_ARG;
+ *(int*)current->value = tmp;
+ return 0;
+ }
+ case arg_string:
+ {
+ *(char**)current->value = optarg + 1;
+ return 0;
+ }
+ case arg_strings:
+ {
+ add_string((getarg_strings*)current->value, optarg + 1);
+ return 0;
+ }
+ case arg_flag:
+ case arg_negative_flag:
+ {
+ int *flag = current->value;
+ if(*optarg == '\0' ||
+ strcmp(optarg + 1, "yes") == 0 ||
+ strcmp(optarg + 1, "true") == 0){
+ *flag = !negate;
+ return 0;
+ } else if (*optarg && strcmp(optarg + 1, "maybe") == 0) {
+ *flag = rand() & 1;
+ } else {
+ *flag = negate;
+ return 0;
+ }
+ return ARG_ERR_BAD_ARG;
+ }
+ case arg_counter :
+ {
+ int val;
+
+ if (*optarg == '\0')
+ val = 1;
+ else {
+ char *endstr;
+
+ val = strtol (optarg, &endstr, 0);
+ if (endstr == optarg)
+ return ARG_ERR_BAD_ARG;
+ }
+ *(int *)current->value += val;
+ return 0;
+ }
+ case arg_double:
+ {
+ double tmp;
+ if(sscanf(optarg + 1, "%lf", &tmp) != 1)
+ return ARG_ERR_BAD_ARG;
+ *(double*)current->value = tmp;
+ return 0;
+ }
+ case arg_collect:{
+ struct getarg_collect_info *c = current->value;
+ int o = argv - rargv[*optind];
+ return (*c->func)(FALSE, argc, rargv, optind, &o, c->data);
+ }
+
+ default:
+ abort ();
+ }
+}
+
+static int
+arg_match_short (struct getargs *args, size_t num_args,
+ char *argv, int argc, const char **rargv, int *optind)
+{
+ int j, k;
+
+ for(j = 1; j > 0 && j < (int)strlen(rargv[*optind]); j++) {
+ for(k = 0; k < (int)num_args; k++) {
+ char *optarg;
+
+ if(args[k].short_name == 0)
+ continue;
+ if(argv[j] == args[k].short_name) {
+ if(args[k].type == arg_flag) {
+ *(int*)args[k].value = 1;
+ break;
+ }
+ if(args[k].type == arg_negative_flag) {
+ *(int*)args[k].value = 0;
+ break;
+ }
+ if(args[k].type == arg_counter) {
+ ++*(int *)args[k].value;
+ break;
+ }
+ if(args[k].type == arg_collect) {
+ struct getarg_collect_info *c = args[k].value;
+
+ if((*c->func)(TRUE, argc, rargv, optind, &j, c->data))
+ return ARG_ERR_BAD_ARG;
+ break;
+ }
+
+ if(argv[j + 1])
+ optarg = &argv[j + 1];
+ else {
+ ++*optind;
+ optarg = rargv[*optind];
+ }
+ if(optarg == NULL) {
+ --*optind;
+ return ARG_ERR_NO_ARG;
+ }
+ if(args[k].type == arg_integer) {
+ int tmp;
+ if(sscanf(optarg, "%d", &tmp) != 1)
+ return ARG_ERR_BAD_ARG;
+ *(int*)args[k].value = tmp;
+ return 0;
+ } else if(args[k].type == arg_string) {
+ *(char**)args[k].value = optarg;
+ return 0;
+ } else if(args[k].type == arg_strings) {
+ add_string((getarg_strings*)args[k].value, optarg);
+ return 0;
+ } else if(args[k].type == arg_double) {
+ double tmp;
+ if(sscanf(optarg, "%lf", &tmp) != 1)
+ return ARG_ERR_BAD_ARG;
+ *(double*)args[k].value = tmp;
+ return 0;
+ }
+ return ARG_ERR_BAD_ARG;
+ }
+ }
+ if (k == (int)num_args)
+ return ARG_ERR_NO_MATCH;
+ }
+ return 0;
+}
+
+int
+getarg(struct getargs *args, size_t num_args,
+ int argc, const char **argv, int *optind)
+{
+ int i;
+ int ret = 0;
+
+ srand (time(NULL));
+ (*optind)++;
+ for(i = *optind; i < argc; i++) {
+ if(argv[i][0] != '-')
+ break;
+ if(argv[i][1] == '-'){
+ if(argv[i][2] == 0){
+ i++;
+ break;
+ }
+ ret = arg_match_long (args, num_args, argv[i] + 2,
+ argc, argv, &i);
+ } else {
+ ret = arg_match_short (args, num_args, argv[i],
+ argc, argv, &i);
+ }
+ if(ret)
+ break;
+ }
+ *optind = i;
+ return ret;
+}
+
+
+#if TEST
+int foo_flag = 2;
+int flag1 = 0;
+int flag2 = 0;
+int bar_int;
+char *baz_string;
+
+struct getargs args[] = {
+ { NULL, '1', arg_flag, &flag1, "one", NULL },
+ { NULL, '2', arg_flag, &flag2, "two", NULL },
+ { "foo", 'f', arg_negative_flag, &foo_flag, "foo", NULL },
+ { "bar", 'b', arg_integer, &bar_int, "bar", "seconds"},
+ { "baz", 'x', arg_string, &baz_string, "baz", "name" },
+};
+
+int main(int argc, char **argv)
+{
+ int optind = 0;
+ while(getarg(args, 5, argc, argv, &optind))
+ printf("Bad arg: %s\n", argv[optind]);
+ printf("flag1 = %d\n", flag1);
+ printf("flag2 = %d\n", flag2);
+ printf("foo_flag = %d\n", foo_flag);
+ printf("bar_int = %d\n", bar_int);
+ printf("baz_flag = %s\n", baz_string);
+ arg_printusage (args, 5, argv[0], "nothing here");
+}
+#endif
diff --git a/ndb/src/common/util/getarg.cat3 b/ndb/src/common/util/getarg.cat3
new file mode 100644
index 00000000000..31685510537
--- /dev/null
+++ b/ndb/src/common/util/getarg.cat3
@@ -0,0 +1,237 @@
+GETARG(3) OpenBSD Programmer's Manual GETARG(3)
+
+NNAAMMEE
+ ggeettaarrgg, aarrgg__pprriinnttuussaaggee - collect command line options
+
+SSYYNNOOPPSSIISS
+ ##iinncclluuddee <<ggeettaarrgg..hh>>
+
+
+ _i_n_t
+ ggeettaarrgg(_s_t_r_u_c_t _g_e_t_a_r_g_s _*_a_r_g_s, _s_i_z_e___t _n_u_m___a_r_g_s, _i_n_t _a_r_g_c, _c_h_a_r _*_*_a_r_g_v,
+ _i_n_t _*_o_p_t_i_n_d);
+
+
+ _v_o_i_d
+ aarrgg__pprriinnttuussaaggee(_s_t_r_u_c_t _g_e_t_a_r_g_s _*_a_r_g_s, _s_i_z_e___t _n_u_m___a_r_g_s,
+ _c_o_n_s_t _c_h_a_r _*_p_r_o_g_n_a_m_e, _c_o_n_s_t _c_h_a_r _*_e_x_t_r_a___s_t_r_i_n_g);
+
+
+DDEESSCCRRIIPPTTIIOONN
+ ggeettaarrgg() collects any command line options given to a program in an easi­
+ ly used way. aarrgg__pprriinnttuussaaggee() pretty-prints the available options, with
+ a short help text.
+
+ _a_r_g_s is the option specification to use, and it's an array of _s_t_r_u_c_t
+ _g_e_t_a_r_g_s elements. _n_u_m___a_r_g_s is the size of _a_r_g_s (in elements). _a_r_g_c and
+ _a_r_g_v are the argument count and argument vector to extract option from.
+ _o_p_t_i_n_d is a pointer to an integer where the index to the last processed
+ argument is stored, it must be initialised to the first index (minus one)
+ to process (normally 0) before the first call.
+
+ _a_r_g___p_r_i_n_t_u_s_a_g_e take the same _a_r_g_s and _n_u_m___a_r_g_s as getarg; _p_r_o_g_n_a_m_e _i_s _t_h_e
+ _n_a_m_e _o_f _t_h_e _p_r_o_g_r_a_m _(_t_o _b_e progname0 _0progname1 _1progname2 _2progname3
+ _3progname4 _4progname5 _e_x_t_r_a___s_t_r_i_n_g is a string to print after the actual
+ options to indicate more arguments. The usefulness of this function is
+ realised only be people who has used programs that has help strings that
+ doesn't match what the code does.
+
+ The _g_e_t_a_r_g_s struct has the following elements.
+
+
+ struct getargs{
+ const char *long_name;
+ char short_name;
+ enum { arg_integer,
+ arg_string,
+ arg_flag,
+ arg_negative_flag,
+ arg_strings,
+ arg_double,
+ arg_collect
+ } type;
+ void *value;
+ const char *help;
+ const char *arg_help;
+ };
+
+ _l_o_n_g___n_a_m_e is the long name of the option, it can be NULL, if you don't
+ want a long name. _s_h_o_r_t___n_a_m_e is the characted to use as short option, it
+ can be zero. If the option has a value the _v_a_l_u_e field gets filled in
+ with that value interpreted as specified by the _t_y_p_e field. _h_e_l_p is a
+ longer help string for the option as a whole, if it's NULL the help text
+ for the option is omitted (but it's still displayed in the synopsis).
+ _a_r_g___h_e_l_p is a description of the argument, if NULL a default value will
+ be used, depending on the type of the option:
+
+
+ arg_integer the argument is a signed integer, and _v_a_l_u_e should
+ point to an _i_n_t.
+
+ _a_r_g___s_t_r_i_n_g the argument is a string, and _v_a_l_u_e should point to a
+ _c_h_a_r_*.
+
+ _a_r_g___f_l_a_g the argument is a flag, and _v_a_l_u_e should point to a
+ _i_n_t. It gets filled in with either zero or one, de­
+ pending on how the option is given, the normal case
+ beeing one. Note that if the option isn't given, the
+ value isn't altered, so it should be initialised to
+ some useful default.
+
+ _a_r_g___n_e_g_a_t_i_v_e___f_l_a_g this is the same as _a_r_g___f_l_a_g but it reverses the mean­
+ ing of the flag (a given short option clears the
+ flag), and the synopsis of a long option is negated.
+
+ _a_r_g___s_t_r_i_n_g_s the argument can be given multiple times, and the val­
+ ues are collected in an array; _v_a_l_u_e should be a
+ pointer to a _s_t_r_u_c_t _g_e_t_a_r_g___s_t_r_i_n_g_s structure, which
+ holds a length and a string pointer.
+
+ _a_r_g___d_o_u_b_l_e argument is a double precision floating point value,
+ and _v_a_l_u_e should point to a _d_o_u_b_l_e.
+
+ _a_r_g___c_o_l_l_e_c_t allows more fine-grained control of the option parsing
+ process. _v_a_l_u_e should be a pointer to a
+ _g_e_t_a_r_g___c_o_l_l_e_c_t___i_n_f_o structure:
+
+ typedef int (*getarg_collect_func)(int short_opt,
+ int argc,
+ char **argv,
+ int *optind,
+ int *optarg,
+ void *data);
+
+ typedef struct getarg_collect_info {
+ getarg_collect_func func;
+ void *data;
+ } getarg_collect_info;
+
+ With the _f_u_n_c member set to a function to call, and
+ _d_a_t_a to some application specific data. The parameters
+ to the collect function are:
+
+ _s_h_o_r_t___f_l_a_g non-zero if this call is via a short option
+ flag, zero otherwise
+
+ _a_r_g_c, _a_r_g_v the whole argument list
+
+ _o_p_t_i_n_d pointer to the index in argv where the flag is
+
+ _o_p_t_a_r_g pointer to the index in argv[*optind] where the
+ flag name starts
+
+ _d_a_t_a application specific data
+
+ You can modify _*_o_p_t_i_n_d, and _*_o_p_t_a_r_g, but to do this
+ correct you (more or less) have to know about the in­
+ ner workings of getarg.
+
+ You can skip parts of arguments by increasing _*_o_p_t_a_r_g
+ (you could implement the --zz_3 set of flags from ggzziipp
+ with this), or whole argument strings by increasing
+ _*_o_p_t_i_n_d (let's say you want a flag --cc _x _y _z to specify
+ a coordinate); if you also have to set _*_o_p_t_a_r_g to a
+ sane value.
+
+ The collect function should return one of
+ ARG_ERR_NO_MATCH, ARG_ERR_BAD_ARG, ARG_ERR_NO_ARG on
+ error, zero otherwise.
+
+ For your convenience there is a function,
+ ggeettaarrgg__ooppttaarrgg(), that returns the traditional argument
+ string, and you pass it all arguments, sans data, that
+ where given to the collection function.
+
+ Don't use this more this unless you absolutely have
+ to.
+
+ Option parsing is similar to what getopt uses. Short options without ar­
+ guments can be compressed (--xxyyzz is the same as --xx --yy --zz), and short op­
+ tions with arguments take these as either the rest of the argv-string or
+ as the next option (--oo_f_o_o, or --oo _f_o_o).
+
+ Long option names are prefixed with -- (double dash), and the value with
+ a = (equal), ----ffoooo==_b_a_r. Long option flags can either be specified as they
+ are (----hheellpp), or with an (boolean parsable) option (----hheellpp==_y_e_s,
+ ----hheellpp==_t_r_u_e, or similar), or they can also be negated (----nnoo--hheellpp is the
+ same as ----hheellpp==no), and if you're really confused you can do it multiple
+ times (----nnoo--nnoo--hheellpp==_f_a_l_s_e, or even ----nnoo--nnoo--hheellpp==_m_a_y_b_e).
+
+EEXXAAMMPPLLEE
+ #include <stdio.h>
+ #include <string.h>
+ #include <getarg.h>
+
+ char *source = "Ouagadougou";
+ char *destination;
+ int weight;
+ int include_catalog = 1;
+ int help_flag;
+
+ struct getargs args[] = {
+ { "source", 's', arg_string, &source,
+ "source of shippment", "city" },
+ { "destination", 'd', arg_string, &destination,
+ "destination of shippment", "city" },
+ { "weight", 'w', arg_integer, &weight,
+ "weight of shippment", "tons" },
+ { "catalog", 'c', arg_negative_flag, &include_catalog,
+ "include product catalog" },
+ { "help", 'h', arg_flag, &help_flag }
+ };
+
+ int num_args = sizeof(args) / sizeof(args[0]); /* number of elements in args */
+
+ const char *progname = "ship++";
+
+ int
+ main(int argc, char **argv)
+ {
+ int optind = 0;
+ if (getarg(args, num_args, argc, argv, &optind)) {
+ arg_printusage(args, num_args, progname, "stuff...");
+ exit (1);
+ }
+ if (help_flag) {
+ arg_printusage(args, num_args, progname, "stuff...");
+ exit (0);
+ }
+ if (destination == NULL) {
+ fprintf(stderr, "%s: must specify destination0, progname);
+ exit(1);
+ }
+ if (strcmp(source, destination) == 0) {
+ fprintf(stderr, "%s: destination must be different from source0);
+ exit(1);
+ }
+ /* include more stuff here ... */
+ exit(2);
+ }
+
+ The output help output from this program looks like this:
+
+ $ ship++ --help
+ Usage: ship++ [--source=city] [-s city] [--destination=city] [-d city]
+ [--weight=tons] [-w tons] [--no-catalog] [-c] [--help] [-h] stuff...
+ -s city, --source=city source of shippment
+ -d city, --destination=city destination of shippment
+ -w tons, --weight=tons weight of shippment
+ -c, --no-catalog include product catalog
+
+
+BBUUGGSS
+ It should be more flexible, so it would be possible to use other more
+ complicated option syntaxes, such as what ps(1), and tar(1), uses, or the
+ AFS model where you can skip the flag names as long as the options come
+ in the correct order.
+
+ Options with multiple arguments should be handled better.
+
+ Should be integreated with SL.
+
+ It's very confusing that the struct you pass in is called getargS.
+
+SSEEEE AALLSSOO
+ getopt(3)
+
+ ROKEN September 24, 1999 4
diff --git a/ndb/src/common/util/md5_hash.cpp b/ndb/src/common/util/md5_hash.cpp
new file mode 100644
index 00000000000..5e28edcf8fa
--- /dev/null
+++ b/ndb/src/common/util/md5_hash.cpp
@@ -0,0 +1,235 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include "md5_hash.hpp"
+
+#ifdef WORDS_BIGENDIAN
+#define HIGHFIRST 1
+#endif
+
+/*
+ * This code implements the MD5 message-digest algorithm.
+ * The algorithm is due to Ron Rivest. This code was
+ * written by Colin Plumb in 1993, no copyright is claimed.
+ * This code is in the public domain; do with it what you wish.
+ *
+ * Equivalent code is available from RSA Data Security, Inc.
+ * This code has been tested against that, and is equivalent,
+ * except that you don't need to include two pages of legalese
+ * with every copy.
+ *
+ * The code has been modified by Mikael Ronstroem to handle
+ * calculating a hash value of a key that is always a multiple
+ * of 4 bytes long. Word 0 of the calculated 4-word hash value
+ * is returned as the hash value.
+ */
+
+#ifndef HIGHFIRST
+#define byteReverse(buf, len) /* Nothing */
+#else
+void byteReverse(unsigned char *buf, unsigned longs);
+/*
+ * Note: this code is harmless on little-endian machines.
+ */
+void byteReverse(unsigned char *buf, unsigned longs)
+{
+ uint32 t;
+ do {
+ t = (Uint32) ((unsigned) buf[3] << 8 | buf[2]) << 16 |
+ ((unsigned) buf[1] << 8 | buf[0]);
+ *(Uint32 *) buf = t;
+ buf += 4;
+ } while (--longs);
+}
+#endif
+
+/* The four core functions - F1 is optimized somewhat */
+
+/* #define F1(x, y, z) (x & y | ~x & z) */
+#define F1(x, y, z) (z ^ (x & (y ^ z)))
+#define F2(x, y, z) F1(z, x, y)
+#define F3(x, y, z) (x ^ y ^ z)
+#define F4(x, y, z) (y ^ (x | ~z))
+
+/* This is the central step in the MD5 algorithm. */
+#define MD5STEP(f, w, x, y, z, data, s) \
+ ( w += f(x, y, z) + data, w = w<<s | w>>(32-s), w += x )
+
+/*
+ * The core of the MD5 algorithm, this alters an existing MD5 hash to
+ * reflect the addition of 16 longwords of new data. MD5Update blocks
+ * the data and converts bytes into longwords for this routine.
+ */
+void MD5Transform(Uint32 buf[4], Uint32 const in[16])
+{
+ register Uint32 a, b, c, d;
+
+ a = buf[0];
+ b = buf[1];
+ c = buf[2];
+ d = buf[3];
+
+ MD5STEP(F1, a, b, c, d, in[0] + 0xd76aa478, 7);
+ MD5STEP(F1, d, a, b, c, in[1] + 0xe8c7b756, 12);
+ MD5STEP(F1, c, d, a, b, in[2] + 0x242070db, 17);
+ MD5STEP(F1, b, c, d, a, in[3] + 0xc1bdceee, 22);
+ MD5STEP(F1, a, b, c, d, in[4] + 0xf57c0faf, 7);
+ MD5STEP(F1, d, a, b, c, in[5] + 0x4787c62a, 12);
+ MD5STEP(F1, c, d, a, b, in[6] + 0xa8304613, 17);
+ MD5STEP(F1, b, c, d, a, in[7] + 0xfd469501, 22);
+ MD5STEP(F1, a, b, c, d, in[8] + 0x698098d8, 7);
+ MD5STEP(F1, d, a, b, c, in[9] + 0x8b44f7af, 12);
+ MD5STEP(F1, c, d, a, b, in[10] + 0xffff5bb1, 17);
+ MD5STEP(F1, b, c, d, a, in[11] + 0x895cd7be, 22);
+ MD5STEP(F1, a, b, c, d, in[12] + 0x6b901122, 7);
+ MD5STEP(F1, d, a, b, c, in[13] + 0xfd987193, 12);
+ MD5STEP(F1, c, d, a, b, in[14] + 0xa679438e, 17);
+ MD5STEP(F1, b, c, d, a, in[15] + 0x49b40821, 22);
+
+ MD5STEP(F2, a, b, c, d, in[1] + 0xf61e2562, 5);
+ MD5STEP(F2, d, a, b, c, in[6] + 0xc040b340, 9);
+ MD5STEP(F2, c, d, a, b, in[11] + 0x265e5a51, 14);
+ MD5STEP(F2, b, c, d, a, in[0] + 0xe9b6c7aa, 20);
+ MD5STEP(F2, a, b, c, d, in[5] + 0xd62f105d, 5);
+ MD5STEP(F2, d, a, b, c, in[10] + 0x02441453, 9);
+ MD5STEP(F2, c, d, a, b, in[15] + 0xd8a1e681, 14);
+ MD5STEP(F2, b, c, d, a, in[4] + 0xe7d3fbc8, 20);
+ MD5STEP(F2, a, b, c, d, in[9] + 0x21e1cde6, 5);
+ MD5STEP(F2, d, a, b, c, in[14] + 0xc33707d6, 9);
+ MD5STEP(F2, c, d, a, b, in[3] + 0xf4d50d87, 14);
+ MD5STEP(F2, b, c, d, a, in[8] + 0x455a14ed, 20);
+ MD5STEP(F2, a, b, c, d, in[13] + 0xa9e3e905, 5);
+ MD5STEP(F2, d, a, b, c, in[2] + 0xfcefa3f8, 9);
+ MD5STEP(F2, c, d, a, b, in[7] + 0x676f02d9, 14);
+ MD5STEP(F2, b, c, d, a, in[12] + 0x8d2a4c8a, 20);
+
+ MD5STEP(F3, a, b, c, d, in[5] + 0xfffa3942, 4);
+ MD5STEP(F3, d, a, b, c, in[8] + 0x8771f681, 11);
+ MD5STEP(F3, c, d, a, b, in[11] + 0x6d9d6122, 16);
+ MD5STEP(F3, b, c, d, a, in[14] + 0xfde5380c, 23);
+ MD5STEP(F3, a, b, c, d, in[1] + 0xa4beea44, 4);
+ MD5STEP(F3, d, a, b, c, in[4] + 0x4bdecfa9, 11);
+ MD5STEP(F3, c, d, a, b, in[7] + 0xf6bb4b60, 16);
+ MD5STEP(F3, b, c, d, a, in[10] + 0xbebfbc70, 23);
+ MD5STEP(F3, a, b, c, d, in[13] + 0x289b7ec6, 4);
+ MD5STEP(F3, d, a, b, c, in[0] + 0xeaa127fa, 11);
+ MD5STEP(F3, c, d, a, b, in[3] + 0xd4ef3085, 16);
+ MD5STEP(F3, b, c, d, a, in[6] + 0x04881d05, 23);
+ MD5STEP(F3, a, b, c, d, in[9] + 0xd9d4d039, 4);
+ MD5STEP(F3, d, a, b, c, in[12] + 0xe6db99e5, 11);
+ MD5STEP(F3, c, d, a, b, in[15] + 0x1fa27cf8, 16);
+ MD5STEP(F3, b, c, d, a, in[2] + 0xc4ac5665, 23);
+
+ MD5STEP(F4, a, b, c, d, in[0] + 0xf4292244, 6);
+ MD5STEP(F4, d, a, b, c, in[7] + 0x432aff97, 10);
+ MD5STEP(F4, c, d, a, b, in[14] + 0xab9423a7, 15);
+ MD5STEP(F4, b, c, d, a, in[5] + 0xfc93a039, 21);
+ MD5STEP(F4, a, b, c, d, in[12] + 0x655b59c3, 6);
+ MD5STEP(F4, d, a, b, c, in[3] + 0x8f0ccc92, 10);
+ MD5STEP(F4, c, d, a, b, in[10] + 0xffeff47d, 15);
+ MD5STEP(F4, b, c, d, a, in[1] + 0x85845dd1, 21);
+ MD5STEP(F4, a, b, c, d, in[8] + 0x6fa87e4f, 6);
+ MD5STEP(F4, d, a, b, c, in[15] + 0xfe2ce6e0, 10);
+ MD5STEP(F4, c, d, a, b, in[6] + 0xa3014314, 15);
+ MD5STEP(F4, b, c, d, a, in[13] + 0x4e0811a1, 21);
+ MD5STEP(F4, a, b, c, d, in[4] + 0xf7537e82, 6);
+ MD5STEP(F4, d, a, b, c, in[11] + 0xbd3af235, 10);
+ MD5STEP(F4, c, d, a, b, in[2] + 0x2ad7d2bb, 15);
+ MD5STEP(F4, b, c, d, a, in[9] + 0xeb86d391, 21);
+
+ buf[0] += a;
+ buf[1] += b;
+ buf[2] += c;
+ buf[3] += d;
+}
+
+/*
+ * Start MD5 accumulation. Set bit count to 0 and buffer to mysterious
+ * initialization constants.
+ */
+Uint32 md5_hash(const Uint64* keybuf, Uint32 no_of_32_words)
+{
+/*
+ * This is the external interface of the module
+ * It is assumed that keybuf is placed on 8 byte
+ * alignment.
+ */
+ Uint32 i;
+ Uint32 buf[4];
+ Uint64 transform64_buf[8];
+ Uint32* transform32_buf;
+ Uint32 len = no_of_32_words << 2;
+ const Uint64* key64buf = (const Uint64*)keybuf;
+ const Uint32* key32buf = (const Uint32*)keybuf;
+
+ transform32_buf = (Uint32*)&transform64_buf[0];
+ buf[0] = 0x67452301;
+ buf[1] = 0xefcdab89;
+ buf[2] = 0x98badcfe;
+ buf[3] = 0x10325476;
+
+ while (no_of_32_words >= 16) {
+ transform64_buf[0] = key64buf[0];
+ transform64_buf[1] = key64buf[1];
+ transform64_buf[2] = key64buf[2];
+ transform64_buf[3] = key64buf[3];
+ transform64_buf[4] = key64buf[4];
+ transform64_buf[5] = key64buf[5];
+ transform64_buf[6] = key64buf[6];
+ transform64_buf[7] = key64buf[7];
+ no_of_32_words -= 16;
+ key64buf += 8;
+ byteReverse((unsigned char *)transform32_buf, 16);
+ MD5Transform(buf, transform32_buf);
+ }
+
+ key32buf = (const Uint32*)key64buf;
+ transform64_buf[0] = 0;
+ transform64_buf[1] = 0;
+ transform64_buf[2] = 0;
+ transform64_buf[3] = 0;
+ transform64_buf[4] = 0;
+ transform64_buf[5] = 0;
+ transform64_buf[6] = 0;
+ transform64_buf[7] = (Uint64)len;
+
+ for (i = 0; i < no_of_32_words; i++)
+ transform32_buf[i] = key32buf[i];
+ transform32_buf[no_of_32_words] = 0x80000000;
+
+ if (no_of_32_words < 14) {
+ byteReverse((unsigned char *)transform32_buf, 16);
+ MD5Transform(buf, transform32_buf);
+ } else {
+ if (no_of_32_words == 14)
+ transform32_buf[15] = 0;
+ MD5Transform(buf, transform32_buf);
+ transform64_buf[0] = 0;
+ transform64_buf[1] = 0;
+ transform64_buf[2] = 0;
+ transform64_buf[3] = 0;
+ transform64_buf[4] = 0;
+ transform64_buf[5] = 0;
+ transform64_buf[6] = 0;
+ transform64_buf[7] = (Uint64)len;
+ byteReverse((unsigned char *)transform32_buf, 16);
+ MD5Transform(buf, transform32_buf);
+ }
+ return buf[0];
+}
+
diff --git a/ndb/src/common/util/random.c b/ndb/src/common/util/random.c
new file mode 100644
index 00000000000..91da19572e2
--- /dev/null
+++ b/ndb/src/common/util/random.c
@@ -0,0 +1,292 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <limits.h>
+#include <sys/types.h>
+
+#ifndef NDB_WIN32
+#include <sys/time.h>
+#endif
+
+#include <ndb_types.h>
+#include <NdbOut.hpp>
+
+#include <random.h>
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+typedef struct {
+ unsigned short int x[3]; /* Current state. */
+ unsigned short int a[3]; /* Factor in congruential formula. */
+ unsigned short int c; /* Additive const. in congruential formula. */
+ int init; /* Flag for initializing. */
+}DRand48Data;
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+static void shuffleSequence(RandomSequence *seq);
+
+/***************************************************************
+* L O C A L D A T A *
+***************************************************************/
+
+static DRand48Data dRand48Data;
+
+/***************************************************************
+* P U B L I C D A T A *
+***************************************************************/
+
+
+/***************************************************************
+****************************************************************
+* L O C A L F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+static void localRandom48Init(long int seedval, DRand48Data *buffer)
+{
+ /* The standards say we only have 32 bits. */
+ if (sizeof (long int) > 4)
+ seedval &= 0xffffffffl;
+
+#if USHRT_MAX == 0xffffU
+ buffer->x[2] = seedval >> 16;
+ buffer->x[1] = seedval & 0xffffl;
+ buffer->x[0] = 0x330e;
+
+ buffer->a[2] = 0x5;
+ buffer->a[1] = 0xdeec;
+ buffer->a[0] = 0xe66d;
+#else
+ buffer->x[2] = seedval;
+ buffer->x[1] = 0x330e0000UL;
+ buffer->x[0] = 0;
+
+ buffer->a[2] = 0x5deecUL;
+ buffer->a[1] = 0xe66d0000UL;
+ buffer->a[0] = 0;
+#endif
+
+ buffer->c = 0xb;
+ buffer->init = 1;
+}
+
+static void localRandom48(DRand48Data *buffer, long int *result)
+{
+ Uint64 X;
+ Uint64 a;
+ Uint64 loc_result;
+
+ /*--------------------------------------*/
+ /* Initialize buffer, if not yet done. */
+ /*--------------------------------------*/
+ if (!buffer->init) {
+#if (USHRT_MAX == 0xffffU)
+ buffer->a[2] = 0x5;
+ buffer->a[1] = 0xdeec;
+ buffer->a[0] = 0xe66d;
+#else
+ buffer->a[2] = 0x5deecUL;
+ buffer->a[1] = 0xe66d0000UL;
+ buffer->a[0] = 0;
+#endif
+ buffer->c = 0xb;
+ buffer->init = 1;
+ }
+
+ /* Do the real work. We choose a data type which contains at least
+ 48 bits. Because we compute the modulus it does not care how
+ many bits really are computed. */
+
+ if (sizeof (unsigned short int) == 2) {
+ X = (Uint64)buffer->x[2] << 32 |
+ (Uint64)buffer->x[1] << 16 |
+ buffer->x[0];
+ a = ((Uint64)buffer->a[2] << 32 |
+ (Uint64)buffer->a[1] << 16 |
+ buffer->a[0]);
+
+ loc_result = X * a + buffer->c;
+
+ buffer->x[0] = loc_result & 0xffff;
+ buffer->x[1] = (loc_result >> 16) & 0xffff;
+ buffer->x[2] = (loc_result >> 32) & 0xffff;
+ }
+ else {
+ X = (Uint64)buffer->x[2] << 16 |
+ buffer->x[1] >> 16;
+ a = (Uint64)buffer->a[2] << 16 |
+ buffer->a[1] >> 16;
+
+ loc_result = X * a + buffer->c;
+
+ buffer->x[0] = loc_result >> 16 & 0xffffffffl;
+ buffer->x[1] = loc_result << 16 & 0xffff0000l;
+ }
+
+ /*--------------------*/
+ /* Store the result. */
+ /*--------------------*/
+ if (sizeof (unsigned short int) == 2)
+ *result = buffer->x[2] << 15 | buffer->x[1] >> 1;
+ else
+ *result = buffer->x[2] >> 1;
+}
+
+static void shuffleSequence(RandomSequence *seq)
+{
+ int i;
+ int j;
+ unsigned int tmp;
+
+ if( !seq ) return;
+
+ for(i = 0; i < seq->length; i++ ) {
+ j = myRandom48(seq->length);
+ if( i != j ) {
+ tmp = seq->values[i];
+ seq->values[i] = seq->values[j];
+ seq->values[j] = tmp;
+ }
+ }
+}
+
+
+/***************************************************************
+****************************************************************
+* P U B L I C F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+
+double getTps(unsigned int count, double timeValue)
+{
+ double f;
+
+ if( timeValue != 0.0 )
+ f = count / timeValue;
+ else
+ f = 0.0;
+
+ return(f);
+}
+
+/*----------------------------*/
+/* Random Sequences Functions */
+/*----------------------------*/
+int initSequence(RandomSequence *seq, SequenceValues *inputValues)
+{
+ unsigned int i;
+ unsigned int j;
+ unsigned int totalLength;
+ unsigned int index;
+
+ if( !seq || !inputValues ) return(-1);
+
+ /*------------------------------------*/
+ /* Find the total length of the array */
+ /*------------------------------------*/
+ totalLength = 0;
+
+ for(i = 0; inputValues[i].length != 0; i++)
+ totalLength += inputValues[i].length;
+
+ if( totalLength == 0 ) return(-1);
+
+ seq->length = totalLength;
+ seq->values = calloc(totalLength, sizeof(unsigned int));
+
+ if( seq->values == 0 ) return(-1);
+
+ /*----------------------*/
+ /* set the array values */
+ /*----------------------*/
+ index = 0;
+
+ for(i = 0; inputValues[i].length != 0; i++) {
+ for(j = 0; j < inputValues[i].length; j++ ) {
+ seq->values[index] = inputValues[i].value;
+ index++;
+ }
+ }
+
+ shuffleSequence(seq);
+
+ seq->currentIndex = 0;
+
+ return(0);
+}
+
+unsigned int getNextRandom(RandomSequence *seq)
+{
+ unsigned int nextValue;
+
+ nextValue = seq->values[seq->currentIndex];
+
+ seq->currentIndex++;
+
+ if(seq->currentIndex == seq->length){
+ seq->currentIndex = 0;
+ shuffleSequence(seq);
+ }
+
+ return nextValue;
+}
+
+void printSequence(RandomSequence *seq, unsigned int numPerRow)
+{
+ int i;
+
+ if( !seq ) return;
+
+ for(i = 0; i<seq->length; i++) {
+ ndbout_c("%d ", seq->values[i]);
+
+ if((i+1) % numPerRow == 0)
+ ndbout_c("");
+ }
+
+ if(i % numPerRow != 0)
+ ndbout_c("");
+}
+
+void myRandom48Init(long int seedval)
+{
+ localRandom48Init(seedval, &dRand48Data);
+}
+
+long int myRandom48(unsigned int maxValue)
+{
+ long int result;
+
+ localRandom48(&dRand48Data, &result);
+
+ return(result % maxValue);
+}
diff --git a/ndb/src/common/util/socket_io.cpp b/ndb/src/common/util/socket_io.cpp
new file mode 100644
index 00000000000..878a9059512
--- /dev/null
+++ b/ndb/src/common/util/socket_io.cpp
@@ -0,0 +1,284 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbTCP.h>
+#include <socket_io.h>
+#include <stdarg.h>
+#include <string.h>
+#include <NdbStdio.h>
+#include <NdbOut.hpp>
+#include <NdbString.h>
+
+extern "C"
+int
+read_socket(NDB_SOCKET_TYPE socket, int timeout_millis,
+ char * buf, int buflen){
+ if(buflen < 1)
+ return 0;
+
+ fd_set readset;
+ FD_ZERO(&readset);
+ FD_SET(socket, &readset);
+
+ struct timeval timeout;
+ timeout.tv_sec = (timeout_millis / 1000);
+ timeout.tv_usec = (timeout_millis % 1000) * 1000;
+
+ const int selectRes = select(socket + 1, &readset, 0, 0, &timeout);
+ if(selectRes == 0)
+ return 0;
+
+ if(selectRes == -1){
+ return -1;
+ }
+
+ return recv(socket, &buf[0], buflen, 0);
+}
+
+extern "C"
+int
+readln_socket(NDB_SOCKET_TYPE socket, int timeout_millis,
+ char * buf, int buflen){
+ if(buflen <= 1)
+ return 0;
+
+ fd_set readset;
+ FD_ZERO(&readset);
+ FD_SET(socket, &readset);
+
+ struct timeval timeout;
+ timeout.tv_sec = (timeout_millis / 1000);
+ timeout.tv_usec = (timeout_millis % 1000) * 1000;
+
+ const int selectRes = select(socket + 1, &readset, 0, 0, &timeout);
+ if(selectRes == 0)
+ return 0;
+
+ if(selectRes == -1){
+ return -1;
+ }
+
+ int pos = 0; buf[pos] = 0;
+ while(true){
+ const int t = recv(socket, &buf[pos], 1, 0);
+ if(t != 1){
+ return -1;
+ }
+ if(buf[pos] == '\n'){
+ buf[pos] = 0;
+
+ if(pos > 0 && buf[pos-1] == '\r'){
+ pos--;
+ buf[pos] = 0;
+ }
+
+ return pos;
+ }
+ pos++;
+ if(pos == (buflen - 1)){
+ buf[pos] = 0;
+ return buflen;
+ }
+
+ FD_ZERO(&readset);
+ FD_SET(socket, &readset);
+ timeout.tv_sec = 1;
+ timeout.tv_usec = 0; // 1 s
+ const int selectRes = select(socket + 1, &readset, 0, 0, &timeout);
+ if(selectRes != 1){
+ return -1;
+ }
+ }
+}
+
+extern "C"
+int
+write_socket(NDB_SOCKET_TYPE socket, int timeout_millis,
+ const char buf[], int len){
+ fd_set writeset;
+ FD_ZERO(&writeset);
+ FD_SET(socket, &writeset);
+ struct timeval timeout;
+ timeout.tv_sec = (timeout_millis / 1000);
+ timeout.tv_usec = (timeout_millis % 1000) * 1000;
+
+ const int selectRes = select(socket + 1, 0, &writeset, 0, &timeout);
+ if(selectRes != 1){
+ return -1;
+ }
+
+ const char * tmp = &buf[0];
+ while(len > 0){
+ const int w = send(socket, tmp, len, 0);
+ if(w == -1){
+ return -1;
+ }
+ len -= w;
+ tmp += w;
+
+ if(len == 0)
+ break;
+
+ FD_ZERO(&writeset);
+ FD_SET(socket, &writeset);
+ timeout.tv_sec = 1;
+ timeout.tv_usec = 0;
+ const int selectRes = select(socket + 1, 0, &writeset, 0, &timeout);
+ if(selectRes != 1){
+ return -1;
+ }
+ }
+
+ return 0;
+}
+
+extern "C"
+int
+print_socket(NDB_SOCKET_TYPE socket, int timeout_millis,
+ const char * fmt, ...){
+ va_list ap;
+ va_start(ap, fmt);
+ int ret = vprint_socket(socket, timeout_millis, fmt, ap);
+ va_end(ap);
+
+ return ret;
+}
+
+extern "C"
+int
+println_socket(NDB_SOCKET_TYPE socket, int timeout_millis,
+ const char * fmt, ...){
+ va_list ap;
+ va_start(ap, fmt);
+ int ret = vprintln_socket(socket, timeout_millis, fmt, ap);
+ va_end(ap);
+ return ret;
+}
+
+extern "C"
+int
+vprint_socket(NDB_SOCKET_TYPE socket, int timeout_millis,
+ const char * fmt, va_list ap){
+ char buf[1000];
+ char *buf2 = buf;
+ size_t size = sizeof(buf);
+
+ if (fmt != 0) {
+ size = vsnprintf(buf, sizeof(buf), fmt, ap);
+ /* Check if the output was truncated */
+ if(size >= sizeof(buf)) {
+ buf2 = (char *)malloc(size+1);
+ if(buf2 == NULL)
+ return -1;
+ vsnprintf(buf2, size, fmt, ap);
+ } else
+ size = sizeof(buf);
+ } else
+ buf[0] = 0;
+
+ int ret = write_socket(socket, timeout_millis, buf2, strlen(buf2));
+ if(buf2 != buf)
+ free(buf2);
+ return ret;
+}
+
+extern "C"
+int
+vprintln_socket(NDB_SOCKET_TYPE socket, int timeout_millis,
+ const char * fmt, va_list ap){
+ char buf[1000];
+ char *buf2 = buf;
+ size_t size = sizeof(buf);
+
+ if (fmt != 0) {
+ size = vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ /* Check if the output was truncated */
+ if(size >= sizeof(buf)) {
+ buf2 = (char *)malloc(size+2);
+ if(buf2 == NULL)
+ return -1;
+ vsnprintf(buf2, size, fmt, ap);
+ } else
+ size = sizeof(buf);
+ } else
+ buf[0] = 0;
+ strlcat(buf2, "\n", size+2);
+
+ int ret = write_socket(socket, timeout_millis, buf2, strlen(buf2));
+ if(buf2 != buf)
+ free(buf2);
+ return ret;
+}
+
+#ifdef NDB_WIN32
+
+class INIT_WINSOCK2
+{
+public:
+ INIT_WINSOCK2(void);
+ ~INIT_WINSOCK2(void);
+
+private:
+ bool m_bAcceptable;
+};
+
+INIT_WINSOCK2 g_init_winsock2;
+
+INIT_WINSOCK2::INIT_WINSOCK2(void)
+: m_bAcceptable(false)
+{
+ WORD wVersionRequested;
+ WSADATA wsaData;
+ int err;
+
+ wVersionRequested = MAKEWORD( 2, 2 );
+
+ err = WSAStartup( wVersionRequested, &wsaData );
+ if ( err != 0 ) {
+ /* Tell the user that we could not find a usable */
+ /* WinSock DLL. */
+ m_bAcceptable = false;
+ }
+
+ /* Confirm that the WinSock DLL supports 2.2.*/
+ /* Note that if the DLL supports versions greater */
+ /* than 2.2 in addition to 2.2, it will still return */
+ /* 2.2 in wVersion since that is the version we */
+ /* requested. */
+
+ if ( LOBYTE( wsaData.wVersion ) != 2 ||
+ HIBYTE( wsaData.wVersion ) != 2 ) {
+ /* Tell the user that we could not find a usable */
+ /* WinSock DLL. */
+ WSACleanup( );
+ m_bAcceptable = false;
+ }
+
+ /* The WinSock DLL is acceptable. Proceed. */
+ m_bAcceptable = true;
+}
+
+INIT_WINSOCK2::~INIT_WINSOCK2(void)
+{
+ if(m_bAcceptable)
+ {
+ m_bAcceptable = false;
+ WSACleanup();
+ }
+}
+
+#endif
+
diff --git a/ndb/src/common/util/strdup.c b/ndb/src/common/util/strdup.c
new file mode 100644
index 00000000000..5291be86b0f
--- /dev/null
+++ b/ndb/src/common/util/strdup.c
@@ -0,0 +1,28 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <stdlib.h>
+
+#ifndef HAVE_STRDUP
+char *
+strdup(const char *s){
+ void *p2;
+ p2 = malloc(strlen(s)+1);
+ strcpy(p2, s);
+ return p2;
+}
+#endif
diff --git a/ndb/src/common/util/strlcat.c b/ndb/src/common/util/strlcat.c
new file mode 100644
index 00000000000..ccff15da27f
--- /dev/null
+++ b/ndb/src/common/util/strlcat.c
@@ -0,0 +1,52 @@
+/*
+ * Copyright (c) 1995 - 1999 Kungliga Tekniska Högskolan
+ * (Royal Institute of Technology, Stockholm, Sweden).
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of the Institute nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <ndb_types.h>
+
+/* RCSID("$KTH: strlcat.c,v 1.1 2000/08/16 01:23:47 lha Exp $"); */
+
+//#include <NdbString.h>
+
+#ifndef HAVE_STRLCAT
+size_t
+strlcat (char *dst, const char *src, size_t dst_sz)
+{
+ size_t len = strlen(dst);
+
+ return len + strlcpy (dst + len, src, dst_sz - len);
+}
+#endif
diff --git a/ndb/src/common/util/strlcpy.c b/ndb/src/common/util/strlcpy.c
new file mode 100644
index 00000000000..9a3048081ca
--- /dev/null
+++ b/ndb/src/common/util/strlcpy.c
@@ -0,0 +1,65 @@
+/*
+ * Copyright (c) 1995 - 1999 Kungliga Tekniska Högskolan
+ * (Royal Institute of Technology, Stockholm, Sweden).
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * 3. Neither the name of the Institute nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <ndb_types.h>
+
+/* RCSID("$KTH: strlcpy.c,v 1.1 2000/08/16 01:23:48 lha Exp $"); */
+
+#ifndef HAVE_STRLCPY
+
+#ifdef NDB_WIN32
+#include <string.h>
+#endif
+
+size_t
+strlcpy (char *dst, const char *src, size_t dst_sz)
+{
+ size_t n;
+ char *p;
+
+ for (p = dst, n = 0;
+ n + 1 < dst_sz && *src != '\0';
+ ++p, ++src, ++n)
+ *p = *src;
+ *p = '\0';
+ if (*src == '\0')
+ return n;
+ else
+ return n + strlen (src);
+}
+
+#endif
diff --git a/ndb/src/common/util/testProperties/Makefile b/ndb/src/common/util/testProperties/Makefile
new file mode 100644
index 00000000000..00b4465b69d
--- /dev/null
+++ b/ndb/src/common/util/testProperties/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE :=
+
+BIN_TARGET := keso
+BIN_TARGET_ARCHIVES := portlib general
+
+SOURCES := testProperties.cpp
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include/util)
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/common/util/testProperties/testProperties.cpp b/ndb/src/common/util/testProperties/testProperties.cpp
new file mode 100644
index 00000000000..4a2999b89c1
--- /dev/null
+++ b/ndb/src/common/util/testProperties/testProperties.cpp
@@ -0,0 +1,203 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Properties.hpp"
+#include <NdbOut.hpp>
+#include <stdlib.h>
+#include <string.h>
+
+#include "uucode.h"
+
+bool
+writeToFile(const Properties & p, const char * fname, bool uu = true){
+ Uint32 sz = p.getPackedSize();
+ char * buffer = (char*)malloc(sz);
+
+ FILE * f = fopen(fname, "wb");
+ bool res = p.pack((Uint32*)buffer);
+ if(res != true){
+ ndbout << "Error packing" << endl;
+ ndbout << "p.getPropertiesErrno() = " << p.getPropertiesErrno() << endl;
+ ndbout << "p.getOSErrno() = " << p.getOSErrno() << endl;
+ }
+ if(uu)
+ uuencode(buffer, sz, f);
+ else {
+ fwrite(buffer, 1, sz, f);
+ }
+
+ fclose(f);
+ free(buffer);
+ return res;
+}
+
+bool
+readFromFile(Properties & p, const char *fname, bool uu = true){
+ Uint32 sz = 30000;
+ char * buffer = (char*)malloc(sz);
+ FILE * f = fopen(fname, "rb");
+ if(uu)
+ uudecode(f, buffer, sz);
+ else
+ fread(buffer, 1, sz, f);
+ fclose(f);
+ bool res = p.unpack((Uint32*)buffer, sz);
+ if(res != true){
+ ndbout << "Error unpacking" << endl;
+ ndbout << "p.getPropertiesErrno() = " << p.getPropertiesErrno() << endl;
+ ndbout << "p.getOSErrno() = " << p.getOSErrno() << endl;
+ }
+ free(buffer);
+ return res;
+}
+
+Property defs[] = {
+ Property("Rolf", 123)
+ ,Property("Keso", "Kent")
+};
+
+
+void putALot(Properties & tmp){
+ int i = 123;
+ tmp.put("LockPagesInMainMemory", i++);
+ tmp.put("SleepWhenIdle", i++);
+ tmp.put("NoOfSignalsToExecuteBetweenCommunicationInterfacePoll", i++);
+ tmp.put("TimeBetweenWatchDogCheck", i++);
+ tmp.put("StopOnError", i++);
+
+ tmp.put("MaxNoOfConcurrentOperations", i++);
+ tmp.put("MaxNoOfConcurrentTransactions", i++);
+ tmp.put("MemorySpaceIndexes", i++);
+ tmp.put("MemorySpaceTuples", i++);
+ tmp.put("MemoryDiskPages", i++);
+ tmp.put("NoOfFreeDiskClusters", i++);
+ tmp.put("NoOfDiskClusters", i++);
+
+ tmp.put("TimeToWaitAlive", i++);
+ tmp.put("HeartbeatIntervalDbDb", i++);
+ tmp.put("HeartbeatIntervalDbApi", i++);
+ tmp.put("TimeBetweenInactiveTransactionAbortCheck", i++);
+
+ tmp.put("TimeBetweenLocalCheckpoints", i++);
+ tmp.put("TimeBetweenGlobalCheckpoints", i++);
+ tmp.put("NoOfFragmentLogFiles", i++);
+ tmp.put("NoOfConcurrentCheckpointsDuringRestart", i++);
+ tmp.put("TransactionInactiveTimeBeforeAbort", i++);
+ tmp.put("NoOfConcurrentProcessesHandleTakeover", i++);
+
+ tmp.put("NoOfConcurrentCheckpointsAfterRestart", i++);
+
+ tmp.put("NoOfDiskPagesToDiskDuringRestartTUP", i++);
+ tmp.put("NoOfDiskPagesToDiskAfterRestartTUP", i++);
+ tmp.put("NoOfDiskPagesToDiskDuringRestartACC", i++);
+ tmp.put("NoOfDiskPagesToDiskAfterRestartACC", i++);
+
+ tmp.put("NoOfDiskClustersPerDiskFile", i++);
+ tmp.put("NoOfDiskFiles", i++);
+
+ // Always found
+ tmp.put("NoOfReplicas", 33);
+ tmp.put("MaxNoOfAttributes", 34);
+ tmp.put("MaxNoOfTables", 35);
+}
+
+int
+main(void){
+ Properties p;
+
+ p.put("Kalle", 1);
+ p.put("Ank1", "anka");
+ p.put("Ank2", "anka");
+ p.put("Ank3", "anka");
+ p.put("Ank4", "anka");
+ putALot(p);
+
+ //p.put(defs, 2);
+ Properties tmp;
+ tmp.put("Type", "TCP");
+ tmp.put("OwnNodeId", 1);
+ tmp.put("RemoteNodeId", 2);
+ tmp.put("OwnHostName", "local");
+ tmp.put("RemoteHostName", "remote");
+
+ tmp.put("SendSignalId", 1);
+ tmp.put("Compression", (Uint32)false);
+ tmp.put("Checksum", 1);
+
+ tmp.put("SendBufferSize", 2000);
+ tmp.put("MaxReceiveSize", 1000);
+
+ tmp.put("PortNumber", 1233);
+ putALot(tmp);
+
+ p.put("Connection", 1, &tmp);
+
+ p.put("NoOfConnections", 2);
+ p.put("NoOfConnection2", 2);
+
+ p.put("kalle", 3);
+ p.put("anka", "kalle");
+
+ Properties p2;
+ p2.put("kalle", "anka");
+
+ p.put("prop", &p2);
+
+ p.put("Connection", 2, &tmp);
+
+ p.put("Connection", 3, &tmp);
+
+ p.put("Connection", 4, &tmp);
+ /*
+ */
+
+ Uint32 a = 99;
+ const char * b;
+ const Properties * p3;
+ Properties * p4;
+
+ bool bb = p.get("kalle", &a);
+ bool cc = p.get("anka", &b);
+ bool dd = p.get("prop", &p3);
+ if(p.getCopy("prop", &p4))
+ delete p4;
+
+ p2.put("p2", &p2);
+
+ p.put("prop2", &p2);
+ /* */
+
+ p.print(stdout, "testing 1: ");
+
+ writeToFile(p, "A_1");
+ writeToFile(p, "B_1", false);
+
+ Properties r1;
+ readFromFile(r1, "A_1");
+ writeToFile(r1, "A_3");
+
+ //r1.print(stdout, "testing 2: ");
+ Properties r2;
+ readFromFile(r2, "A_1");
+ writeToFile(r2, "A_4");
+
+ Properties r3;
+ readFromFile(r3, "B_1", false);
+ writeToFile(r3, "A_5");
+ r3.print(stdout, "testing 3: ");
+
+ return 0;
+}
diff --git a/ndb/src/common/util/testSimpleProperties/Makefile b/ndb/src/common/util/testSimpleProperties/Makefile
new file mode 100644
index 00000000000..89d33fa8dd8
--- /dev/null
+++ b/ndb/src/common/util/testSimpleProperties/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := util
+
+BIN_TARGET := sp_test
+BIN_TARGET_ARCHIVES := portlib general
+
+SOURCES := sp_test.cpp
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include/util)
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/common/util/testSimpleProperties/sp_test.cpp b/ndb/src/common/util/testSimpleProperties/sp_test.cpp
new file mode 100644
index 00000000000..d6dbe2a1502
--- /dev/null
+++ b/ndb/src/common/util/testSimpleProperties/sp_test.cpp
@@ -0,0 +1,95 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "SimpleProperties.hpp"
+#include <NdbOut.hpp>
+#include <assert.h>
+#include <stdio.h>
+
+Uint32 page[8192];
+
+int writer();
+int reader(Uint32 *, Uint32 len);
+int unpack(Uint32 *, Uint32 len);
+
+int main(){
+ int len = writer();
+ reader(page, len);
+ unpack(page, len);
+
+ return 0;
+}
+
+int
+writer(){
+ LinearWriter w(&page[0], 8192);
+
+ w.first();
+ w.add(1, 2);
+ w.add(7, 3);
+ w.add(3, "jonas");
+ w.add(5, "0123456789");
+ w.add(7, 4);
+ w.add(3, "e cool");
+ w.add(5, "9876543210");
+
+ ndbout_c("WordsUsed = %d", w.getWordsUsed());
+
+ return w.getWordsUsed();
+}
+
+int
+reader(Uint32 * pages, Uint32 len){
+ SimplePropertiesLinearReader it(pages, len);
+
+ it.printAll(ndbout);
+ return 0;
+}
+
+struct Test {
+ Uint32 val1;
+ Uint32 val7;
+ char val3[100];
+ Test() : val1(0xFFFFFFFF), val7(0xFFFFFFFF) { sprintf(val3, "bad");}
+};
+
+static const
+SimpleProperties::SP2StructMapping
+test_map [] = {
+ { 1, offsetof(Test, val1), SimpleProperties::Uint32Value, 0, ~0 },
+ { 7, offsetof(Test, val7), SimpleProperties::Uint32Value, 0, ~0 },
+ { 3, offsetof(Test, val3), SimpleProperties::StringValue, 0, sizeof(100) },
+ { 5, 0, SimpleProperties::InvalidValue, 0, 0 }
+};
+
+static unsigned
+test_map_sz = sizeof(test_map)/sizeof(test_map[0]);
+
+int
+unpack(Uint32 * pages, Uint32 len){
+ Test test;
+ SimplePropertiesLinearReader it(pages, len);
+ SimpleProperties::UnpackStatus status;
+ while((status = SimpleProperties::unpack(it, &test, test_map, test_map_sz,
+ true, false)) == SimpleProperties::Break){
+ ndbout << "test.val1 = " << test.val1 << endl;
+ ndbout << "test.val7 = " << test.val7 << endl;
+ ndbout << "test.val3 = " << test.val3 << endl;
+ it.next();
+ }
+ assert(status == SimpleProperties::Eof);
+ return 0;
+}
diff --git a/ndb/src/common/util/uucode.c b/ndb/src/common/util/uucode.c
new file mode 100644
index 00000000000..f862d982204
--- /dev/null
+++ b/ndb/src/common/util/uucode.c
@@ -0,0 +1,235 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <stdio.h>
+#include <string.h>
+
+/* ENC is the basic 1 character encoding function to make a char printing */
+/* DEC is single character decode */
+#define ENC(c) ((c) ? ((c) & 077) + ' ': '`')
+#define DEC(c) (((c) - ' ') & 077)
+
+/*
+ * copy from in to out, encoding as you go along.
+ */
+void
+uuencode(const char * data, int dataLen, FILE * out)
+{
+ int ch, n;
+ const char *p = data;
+
+ fprintf(out, "begin\n");
+
+ while (dataLen > 0){
+ n = dataLen > 45 ? 45 : dataLen;
+ dataLen -= n;
+ ch = ENC(n);
+ if (putc(ch, out) == EOF)
+ break;
+ for (; n > 0; n -= 3, p += 3) {
+ char p_0 = * p;
+ char p_1 = 0;
+ char p_2 = 0;
+
+ if(n >= 2){
+ p_1 = p[1];
+ }
+ if(n >= 3){
+ p_2 = p[2];
+ }
+
+ ch = p_0 >> 2;
+ ch = ENC(ch);
+ if (putc(ch, out) == EOF)
+ break;
+ ch = ((p_0 << 4) & 060) | ((p_1 >> 4) & 017);
+ ch = ENC(ch);
+ if (putc(ch, out) == EOF)
+ break;
+ ch = ((p_1 << 2) & 074) | ((p_2 >> 6) & 03);
+ ch = ENC(ch);
+ if (putc(ch, out) == EOF)
+ break;
+ ch = p_2 & 077;
+ ch = ENC(ch);
+ if (putc(ch, out) == EOF)
+ break;
+ }
+ if (putc('\n', out) == EOF)
+ break;
+ }
+ ch = ENC('\0');
+ putc(ch, out);
+ putc('\n', out);
+ fprintf(out, "end\n");
+}
+
+int
+uudecode(FILE * input, char * outBuf, int bufLen){
+ int n;
+ char ch, *p, returnCode;
+ char buf[255];
+
+ returnCode = 0;
+ /* search for header line */
+ do {
+ if (!fgets(buf, sizeof(buf), input)) {
+ return 1;
+ }
+ } while (strncmp(buf, "begin", 5));
+
+ /* for each input line */
+ for (;;) {
+ if (!fgets(p = buf, sizeof(buf), input)) {
+ return 1;
+ }
+ /*
+ * `n' is used to avoid writing out all the characters
+ * at the end of the file.
+ */
+ if ((n = DEC(*p)) <= 0)
+ break;
+ if(n >= bufLen){
+ returnCode = 1;
+ break;
+ }
+ for (++p; n > 0; p += 4, n -= 3)
+ if (n >= 3) {
+ ch = DEC(p[0]) << 2 | DEC(p[1]) >> 4;
+ * outBuf = ch; outBuf++; bufLen--;
+ ch = DEC(p[1]) << 4 | DEC(p[2]) >> 2;
+ * outBuf = ch; outBuf++; bufLen--;
+ ch = DEC(p[2]) << 6 | DEC(p[3]);
+ * outBuf = ch; outBuf++; bufLen--;
+ } else {
+ if (n >= 1) {
+ ch = DEC(p[0]) << 2 | DEC(p[1]) >> 4;
+ * outBuf = ch; outBuf++; bufLen--;
+ }
+ if (n >= 2) {
+ ch = DEC(p[1]) << 4 | DEC(p[2]) >> 2;
+ * outBuf = ch; outBuf++; bufLen--;
+ }
+ if (n >= 3) {
+ ch = DEC(p[2]) << 6 | DEC(p[3]);
+ * outBuf = ch; outBuf++; bufLen--;
+ }
+ }
+ }
+ if (!fgets(buf, sizeof(buf), input) || strcmp(buf, "end\n")) {
+ return 1;
+ }
+ return returnCode;
+}
+
+int
+uuencode_mem(char * dst, const char * data, int dataLen)
+{
+ int sz = 0;
+
+ int ch, n;
+ const char *p = data;
+
+ while (dataLen > 0){
+ n = dataLen > 45 ? 45 : dataLen;
+ dataLen -= n;
+ ch = ENC(n);
+ * dst = ch; dst++; sz++;
+ for (; n > 0; n -= 3, p += 3) {
+ char p_0 = * p;
+ char p_1 = 0;
+ char p_2 = 0;
+
+ if(n >= 2){
+ p_1 = p[1];
+ }
+ if(n >= 3){
+ p_2 = p[2];
+ }
+
+ ch = p_0 >> 2;
+ ch = ENC(ch);
+ * dst = ch; dst++; sz++;
+
+ ch = ((p_0 << 4) & 060) | ((p_1 >> 4) & 017);
+ ch = ENC(ch);
+ * dst = ch; dst++; sz++;
+
+ ch = ((p_1 << 2) & 074) | ((p_2 >> 6) & 03);
+ ch = ENC(ch);
+ * dst = ch; dst++; sz++;
+
+ ch = p_2 & 077;
+ ch = ENC(ch);
+ * dst = ch; dst++; sz++;
+ }
+
+ * dst = '\n'; dst++; sz++;
+ }
+ ch = ENC('\0');
+ * dst = ch; dst++; sz++;
+
+ * dst = '\n'; dst++; sz++;
+ * dst = 0; dst++; sz++;
+
+ return sz;
+}
+
+int
+uudecode_mem(char * outBuf, int bufLen, const char * src){
+ int n;
+ char ch;
+ int sz = 0;
+ const char * p = src;
+
+ /*
+ * `n' is used to avoid writing out all the characters
+ * at the end of the file.
+ */
+ if ((n = DEC(*p)) <= 0)
+ return 0;
+ if(n >= bufLen){
+ return -1;
+ }
+ for (++p; n > 0; p += 4, n -= 3){
+ if (n >= 3) {
+ ch = DEC(p[0]) << 2 | DEC(p[1]) >> 4;
+ * outBuf = ch; outBuf++; bufLen--; sz++;
+ ch = DEC(p[1]) << 4 | DEC(p[2]) >> 2;
+ * outBuf = ch; outBuf++; bufLen--; sz++;
+ ch = DEC(p[2]) << 6 | DEC(p[3]);
+ * outBuf = ch; outBuf++; bufLen--; sz++;
+ } else {
+ if (n >= 1) {
+ ch = DEC(p[0]) << 2 | DEC(p[1]) >> 4;
+ * outBuf = ch; outBuf++; bufLen--; sz++;
+ }
+ if (n >= 2) {
+ ch = DEC(p[1]) << 4 | DEC(p[2]) >> 2;
+ * outBuf = ch; outBuf++; bufLen--; sz++;
+ }
+ if (n >= 3) {
+ ch = DEC(p[2]) << 6 | DEC(p[3]);
+ * outBuf = ch; outBuf++; bufLen--; sz++;
+ }
+ }
+ }
+ return sz;
+}
+
+
+
diff --git a/ndb/src/common/util/version.c b/ndb/src/common/util/version.c
new file mode 100644
index 00000000000..d220a06850a
--- /dev/null
+++ b/ndb/src/common/util/version.c
@@ -0,0 +1,224 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <string.h>
+#include <ndb_types.h>
+#include <ndb_version.h>
+#include <version.h>
+
+Uint32 getMajor(Uint32 version) {
+ return (version >> 16) & 0xFF;
+}
+
+Uint32 getMinor(Uint32 version) {
+ return (version >> 8) & 0xFF;
+}
+
+Uint32 getBuild(Uint32 version) {
+ return (version >> 0) & 0xFF;
+}
+
+Uint32 makeVersion(Uint32 major, Uint32 minor, Uint32 build) {
+ return MAKE_VERSION(major, minor, build);
+
+}
+
+char * getVersionString(Uint32 version, char * status) {
+ char buff[100];
+ snprintf(buff, sizeof(buff),
+ "Version %d.%d.%d (%s)",
+ getMajor(version),
+ getMinor(version),
+ getBuild(version),
+ status);
+
+ return strdup(buff);
+}
+
+typedef enum {
+ UG_Null,
+ UG_Range,
+ UG_Exact
+} UG_MatchType;
+
+struct NdbUpGradeCompatible {
+ Uint32 ownVersion;
+ Uint32 otherVersion;
+ UG_MatchType matchType;
+};
+
+//#define TEST_VERSION
+
+#ifndef TEST_VERSION
+struct NdbUpGradeCompatible ndbCompatibleTable_full[] = {
+ { NDB_VERSION_D , MAKE_VERSION(NDB_VERSION_MAJOR,NDB_VERSION_MINOR,2), UG_Range },
+ { 0, 0, UG_Null }
+};
+
+struct NdbUpGradeCompatible ndbCompatibleTable_upgrade[] = {
+ { 0, 0, UG_Null }
+};
+
+void ndbSetOwnVersion() {}
+
+#else // testing purposes
+
+struct NdbUpGradeCompatible ndbCompatibleTable_full[] = {
+ { MAKE_VERSION(4,1,5), MAKE_VERSION(4,1,0), UG_Range },
+ { MAKE_VERSION(3,6,9), MAKE_VERSION(3,6,1), UG_Range },
+ { MAKE_VERSION(3,6,2), MAKE_VERSION(3,6,1), UG_Range },
+ { MAKE_VERSION(3,5,7), MAKE_VERSION(3,5,0), UG_Range },
+ { MAKE_VERSION(3,5,1), MAKE_VERSION(3,5,0), UG_Range },
+ { NDB_VERSION_D , MAKE_VERSION(NDB_VERSION_MAJOR,NDB_VERSION_MINOR,2), UG_Range },
+ { 0, 0, UG_Null }
+};
+
+struct NdbUpGradeCompatible ndbCompatibleTable_upgrade[] = {
+ { MAKE_VERSION(4,1,5), MAKE_VERSION(3,6,9), UG_Exact },
+ { MAKE_VERSION(3,6,2), MAKE_VERSION(3,5,7), UG_Exact },
+ { MAKE_VERSION(3,5,1), NDB_VERSION_D , UG_Exact },
+ { 0, 0, UG_Null }
+};
+
+
+Uint32 ndbOwnVersionTesting = 0;
+void
+ndbSetOwnVersion() {
+ char buf[256];
+ if (NdbEnv_GetEnv("NDB_SETVERSION", buf, sizeof(buf))) {
+ Uint32 _v1,_v2,_v3;
+ if (sscanf(buf, "%u.%u.%u", &_v1, &_v2, &_v3) == 3) {
+ ndbOwnVersionTesting = MAKE_VERSION(_v1,_v2,_v3);
+ ndbout_c("Testing: Version set to 0x%x", ndbOwnVersionTesting);
+ }
+ }
+}
+
+#endif
+
+void ndbPrintVersion()
+{
+ printf("Version: %u.%u.%u\n",
+ getMajor(ndbGetOwnVersion()),
+ getMinor(ndbGetOwnVersion()),
+ getBuild(ndbGetOwnVersion()));
+}
+
+Uint32
+ndbGetOwnVersion()
+{
+#ifndef TEST_VERSION
+ return NDB_VERSION_D;
+#else // testing purposes
+ if (ndbOwnVersionTesting == 0)
+ return NDB_VERSION_D;
+ else
+ return ndbOwnVersionTesting;
+#endif
+}
+
+int
+ndbSearchUpgradeCompatibleTable(Uint32 ownVersion, Uint32 otherVersion,
+ struct NdbUpGradeCompatible table[])
+{
+ int i;
+ for (i = 0; table[i].ownVersion != 0 && table[i].otherVersion != 0; i++) {
+ if (table[i].ownVersion == ownVersion ||
+ table[i].ownVersion == ~0) {
+ switch (table[i].matchType) {
+ case UG_Range:
+ if (otherVersion >= table[i].otherVersion){
+ return 1;
+ }
+ break;
+ case UG_Exact:
+ if (otherVersion == table[i].otherVersion){
+ return 1;
+ }
+ break;
+ default:
+ break;
+ }
+ }
+ }
+ return 0;
+}
+
+int
+ndbCompatible(Uint32 ownVersion, Uint32 otherVersion, struct NdbUpGradeCompatible table[])
+{
+ if (otherVersion >= ownVersion) {
+ return 1;
+ }
+ return ndbSearchUpgradeCompatibleTable(ownVersion, otherVersion, table);
+}
+
+int
+ndbCompatible_full(Uint32 ownVersion, Uint32 otherVersion)
+{
+ return ndbCompatible(ownVersion, otherVersion, ndbCompatibleTable_full);
+}
+
+int
+ndbCompatible_upgrade(Uint32 ownVersion, Uint32 otherVersion)
+{
+ if (ndbCompatible_full(ownVersion, otherVersion))
+ return 1;
+ return ndbCompatible(ownVersion, otherVersion, ndbCompatibleTable_upgrade);
+}
+
+int
+ndbCompatible_mgmt_ndb(Uint32 ownVersion, Uint32 otherVersion)
+{
+ return ndbCompatible_upgrade(ownVersion, otherVersion);
+}
+
+int
+ndbCompatible_mgmt_api(Uint32 ownVersion, Uint32 otherVersion)
+{
+ return ndbCompatible_upgrade(ownVersion, otherVersion);
+}
+
+int
+ndbCompatible_ndb_mgmt(Uint32 ownVersion, Uint32 otherVersion)
+{
+ return ndbCompatible_full(ownVersion, otherVersion);
+}
+
+int
+ndbCompatible_api_mgmt(Uint32 ownVersion, Uint32 otherVersion)
+{
+ return ndbCompatible_full(ownVersion, otherVersion);
+}
+
+int
+ndbCompatible_api_ndb(Uint32 ownVersion, Uint32 otherVersion)
+{
+ return ndbCompatible_full(ownVersion, otherVersion);
+}
+
+int
+ndbCompatible_ndb_api(Uint32 ownVersion, Uint32 otherVersion)
+{
+ return ndbCompatible_upgrade(ownVersion, otherVersion);
+}
+
+int
+ndbCompatible_ndb_ndb(Uint32 ownVersion, Uint32 otherVersion)
+{
+ return ndbCompatible_upgrade(ownVersion, otherVersion);
+}
diff --git a/ndb/src/cw/Makefile b/ndb/src/cw/Makefile
new file mode 100644
index 00000000000..e710c1e244d
--- /dev/null
+++ b/ndb/src/cw/Makefile
@@ -0,0 +1,6 @@
+include .defs.mk
+
+DIRS := cpcd
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/cw/cpcc-win32/C++/CPC_GUI.cpp b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.cpp
new file mode 100644
index 00000000000..59ee3e90451
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.cpp
@@ -0,0 +1,215 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "stdafx.h"
+
+HINSTANCE hInst ;
+TCHAR szTitle[MAX_LOADSTRING] ;
+TCHAR szWindowClass[MAX_LOADSTRING] ;
+
+static CNdbControls controls ;
+
+int APIENTRY WinMain(HINSTANCE hInstance,
+ HINSTANCE hPrevInstance,
+ LPSTR lpCmdLine,
+ int nCmdShow){
+ MSG msg;
+ HACCEL hAccelTable;
+
+ LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING) ;
+ LoadString(hInstance, IDC_CPC_GUI, szWindowClass, MAX_LOADSTRING) ;
+ NdbRegisterClass(hInstance);
+
+ if (!InitInstance (hInstance, nCmdShow)) {
+ return FALSE;
+ }
+
+ hAccelTable = LoadAccelerators(hInstance, (LPCTSTR)IDC_CPC_GUI);
+
+ while (GetMessage(&msg, NULL, 0, 0)){
+
+ if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)){
+
+ TranslateMessage(&msg);
+ DispatchMessage(&msg);
+
+ }
+
+ }
+
+ return msg.wParam;
+}
+
+
+ATOM NdbRegisterClass(HINSTANCE hInstance){
+ WNDCLASSEX wcex;
+
+ wcex.cbSize = sizeof(WNDCLASSEX);
+
+ wcex.style = CS_HREDRAW | CS_VREDRAW ;
+ wcex.lpfnWndProc = (WNDPROC)WndProc;
+ wcex.cbClsExtra = 0;
+ wcex.cbWndExtra = 0;
+ wcex.hInstance = hInstance;
+ wcex.hIcon = LoadIcon(hInstance, (LPCTSTR)IDI_CPC_GUI);
+ wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
+ wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW);
+ wcex.lpszMenuName = (LPCSTR)IDC_CPC_GUI;
+ wcex.lpszClassName = szWindowClass;
+ wcex.hIconSm = LoadIcon(wcex.hInstance, (LPCTSTR)IDI_SMALL);
+
+ return RegisterClassEx(&wcex);
+}
+
+
+BOOL InitInstance(HINSTANCE hInstance, int nCmdShow){
+
+ HWND hWnd;
+
+ hInst = hInstance;
+
+ hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
+ CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
+
+ InitCommonControls();
+
+ if (!hWnd) return FALSE ;
+
+ ShowWindow(hWnd, nCmdShow) ;
+ UpdateWindow(hWnd) ;
+
+ return TRUE;
+}
+
+LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam){
+
+ int wmId, wmEvent;
+ PAINTSTRUCT ps;
+ HDC hdc;
+ int c = 0 ;
+
+ switch (message)
+ {
+
+ case WM_CREATE:
+ _assert(controls.Create(hInst, hWnd)) ;
+ return 0 ;
+
+ case WM_COMMAND:
+ wmId = LOWORD(wParam);
+ wmEvent = HIWORD(wParam);
+
+ switch (wmId){
+ case IDM_ABOUT:
+ DialogBox(hInst, (LPCTSTR)IDD_ABOUTBOX, hWnd, (DLGPROC)About);
+ break;
+ case IDM_EXIT:
+ DestroyWindow(hWnd);
+ break;
+ default:
+ return DefWindowProc(hWnd, message, wParam, lParam);
+ }
+ break;
+
+ case WM_NOTIFY:
+ switch (((LPNMHDR) lParam)->code) {
+ case TTN_GETDISPINFO: {
+
+ LPTOOLTIPTEXT lpttt;
+ lpttt = (LPTOOLTIPTEXT) lParam;
+ lpttt->hinst = hInst;
+
+ int idButton = lpttt->hdr.idFrom;
+
+ switch (idButton){
+ case IDM_NEW:
+ lpttt->lpszText = MAKEINTRESOURCE(IDS_TIP_NEW);
+ break;
+ case IDM_DELETE:
+ lpttt->lpszText = MAKEINTRESOURCE(IDS_TIP_DELETE);
+ break;
+ case IDM_PROPS:
+ lpttt->lpszText = MAKEINTRESOURCE(IDS_TIP_PROPS);
+ break;
+ }
+ break;
+ }
+ case TVN_SELCHANGED: {
+ LPNMTREEVIEW pnmtv ;
+
+ pnmtv = (LPNMTREEVIEW) lParam ;
+ controls.ToggleListViews(pnmtv) ;
+
+ break ;
+ }
+
+ case NM_RCLICK: {
+ LPNMHDR lpnmh ;
+ lpnmh = (LPNMHDR) lParam ;
+ switch(lpnmh->idFrom){
+ case ID_TREEVIEW:
+ break;
+ default:
+ break ;
+ }
+ }
+
+ default:
+ break;
+ }
+
+
+ case WM_PAINT:
+ hdc = BeginPaint(hWnd, &ps) ;
+ EndPaint(hWnd, &ps);
+ break;
+
+ case WM_SIZE:
+ controls.Resize() ;
+ return 0 ;
+
+ case WM_DESTROY:
+ PostQuitMessage(0);
+ break;
+
+ default:
+ return DefWindowProc(hWnd, message, wParam, lParam);
+ }
+ return 0;
+}
+
+
+LRESULT CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam){
+
+ switch (message){
+
+ case WM_INITDIALOG:
+ return TRUE;
+
+ case WM_COMMAND:
+ if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL){
+ EndDialog(hDlg, LOWORD(wParam));
+ return TRUE;
+ }
+ break;
+ }
+ return FALSE;
+}
+
+
+
+
+
diff --git a/ndb/src/cw/cpcc-win32/C++/CPC_GUI.dsp b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.dsp
new file mode 100644
index 00000000000..91007b0a47e
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.dsp
@@ -0,0 +1,216 @@
+# Microsoft Developer Studio Project File - Name="CPC_GUI" - Package Owner=<4>
+# Microsoft Developer Studio Generated Build File, Format Version 6.00
+# ** DO NOT EDIT **
+
+# TARGTYPE "Win32 (x86) Application" 0x0101
+
+CFG=CPC_GUI - Win32 Debug
+!MESSAGE This is not a valid makefile. To build this project using NMAKE,
+!MESSAGE use the Export Makefile command and run
+!MESSAGE
+!MESSAGE NMAKE /f "CPC_GUI.mak".
+!MESSAGE
+!MESSAGE You can specify a configuration when running NMAKE
+!MESSAGE by defining the macro CFG on the command line. For example:
+!MESSAGE
+!MESSAGE NMAKE /f "CPC_GUI.mak" CFG="CPC_GUI - Win32 Debug"
+!MESSAGE
+!MESSAGE Possible choices for configuration are:
+!MESSAGE
+!MESSAGE "CPC_GUI - Win32 Release" (based on "Win32 (x86) Application")
+!MESSAGE "CPC_GUI - Win32 Debug" (based on "Win32 (x86) Application")
+!MESSAGE
+
+# Begin Project
+# PROP AllowPerConfigDependencies 0
+# PROP Scc_ProjName ""
+# PROP Scc_LocalPath ""
+CPP=cl.exe
+MTL=midl.exe
+RSC=rc.exe
+
+!IF "$(CFG)" == "CPC_GUI - Win32 Release"
+
+# PROP BASE Use_MFC 0
+# PROP BASE Use_Debug_Libraries 0
+# PROP BASE Output_Dir "Release"
+# PROP BASE Intermediate_Dir "Release"
+# PROP BASE Target_Dir ""
+# PROP Use_MFC 0
+# PROP Use_Debug_Libraries 0
+# PROP Output_Dir "Release"
+# PROP Intermediate_Dir "Release"
+# PROP Ignore_Export_Lib 0
+# PROP Target_Dir ""
+# ADD BASE CPP /nologo /W3 /GX /O2 /D "WIN32" /D "NDEBUG" /D "_WINDOWS" /D "_MBCS" /Yu"stdafx.h" /FD /c
+# ADD CPP /nologo /W3 /GX /O2 /D "WIN32" /D "NDEBUG" /D "_WINDOWS" /D "_MBCS" /Yu"stdafx.h" /FD /c
+# ADD BASE MTL /nologo /D "NDEBUG" /mktyplib203 /win32
+# ADD MTL /nologo /D "NDEBUG" /mktyplib203 /win32
+# ADD BASE RSC /l 0x409 /d "NDEBUG"
+# ADD RSC /l 0x409 /d "NDEBUG"
+BSC32=bscmake.exe
+# ADD BASE BSC32 /nologo
+# ADD BSC32 /nologo
+LINK32=link.exe
+# ADD BASE LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /subsystem:windows /machine:I386
+# ADD LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib mfc42.lib /nologo /subsystem:windows /machine:I386
+
+!ELSEIF "$(CFG)" == "CPC_GUI - Win32 Debug"
+
+# PROP BASE Use_MFC 0
+# PROP BASE Use_Debug_Libraries 1
+# PROP BASE Output_Dir "Debug"
+# PROP BASE Intermediate_Dir "Debug"
+# PROP BASE Target_Dir ""
+# PROP Use_MFC 0
+# PROP Use_Debug_Libraries 1
+# PROP Output_Dir "Debug"
+# PROP Intermediate_Dir "Debug"
+# PROP Ignore_Export_Lib 0
+# PROP Target_Dir ""
+# ADD BASE CPP /nologo /W3 /Gm /GX /ZI /Od /D "WIN32" /D "_DEBUG" /D "_WINDOWS" /D "_MBCS" /Yu"stdafx.h" /FD /GZ /c
+# ADD CPP /nologo /W3 /Gm /GX /ZI /Od /D "WIN32" /D "_DEBUG" /D "_WINDOWS" /D "_MBCS" /FR /Yu"stdafx.h" /FD /GZ /c
+# ADD BASE MTL /nologo /D "_DEBUG" /mktyplib203 /win32
+# ADD MTL /nologo /D "_DEBUG" /mktyplib203 /win32
+# ADD BASE RSC /l 0x409 /d "_DEBUG"
+# ADD RSC /l 0x409 /d "_DEBUG"
+BSC32=bscmake.exe
+# ADD BASE BSC32 /nologo
+# ADD BSC32 /nologo
+LINK32=link.exe
+# ADD BASE LINK32 kernel32.lib user32.lib gdi32.lib winspool.lib comdlg32.lib advapi32.lib shell32.lib ole32.lib oleaut32.lib uuid.lib odbc32.lib odbccp32.lib /nologo /subsystem:windows /debug /machine:I386 /pdbtype:sept
+# ADD LINK32 kernel32.lib user32.lib gdi32.lib comdlg32.lib advapi32.lib shell32.lib comctl32.lib mfc42d.lib /nologo /subsystem:windows /debug /machine:I386 /pdbtype:sept
+
+!ENDIF
+
+# Begin Target
+
+# Name "CPC_GUI - Win32 Release"
+# Name "CPC_GUI - Win32 Debug"
+# Begin Group "Source Files"
+
+# PROP Default_Filter "cpp;c;cxx;rc;def;r;odl;idl;hpj;bat"
+# Begin Source File
+
+SOURCE=.\CPC_GUI.cpp
+# End Source File
+# Begin Source File
+
+SOURCE=.\CPC_GUI.rc
+# End Source File
+# Begin Source File
+
+SOURCE=.\NdbControls.cpp
+# End Source File
+# Begin Source File
+
+SOURCE=.\StdAfx.cpp
+# ADD CPP /Yc"stdafx.h"
+# End Source File
+# End Group
+# Begin Group "Header Files"
+
+# PROP Default_Filter "h;hpp;hxx;hm;inl"
+# Begin Source File
+
+SOURCE=.\CPC_GUI.h
+# End Source File
+# Begin Source File
+
+SOURCE=.\NdbControls.h
+# End Source File
+# Begin Source File
+
+SOURCE=.\resource.h
+# End Source File
+# Begin Source File
+
+SOURCE=.\StdAfx.h
+# End Source File
+# End Group
+# Begin Group "Resource Files"
+
+# PROP Default_Filter "ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe"
+# Begin Source File
+
+SOURCE=.\bitmap1.bmp
+# End Source File
+# Begin Source File
+
+SOURCE=.\bmp00001.bmp
+# End Source File
+# Begin Source File
+
+SOURCE=.\C.bmp
+# End Source File
+# Begin Source File
+
+SOURCE=.\Closed.BMP
+# End Source File
+# Begin Source File
+
+SOURCE=.\Closed.ICO
+# End Source File
+# Begin Source File
+
+SOURCE=.\Closed24.bmp
+# End Source File
+# Begin Source File
+
+SOURCE=.\Computer24.BMP
+# End Source File
+# Begin Source File
+
+SOURCE=.\CPC_GUI.ico
+# End Source File
+# Begin Source File
+
+SOURCE=.\Db.bmp
+# End Source File
+# Begin Source File
+
+SOURCE=.\icon1.ico
+# End Source File
+# Begin Source File
+
+SOURCE=.\O.bmp
+# End Source File
+# Begin Source File
+
+SOURCE=.\Open.BMP
+# End Source File
+# Begin Source File
+
+SOURCE=.\Open.ICO
+# End Source File
+# Begin Source File
+
+SOURCE=.\Open24.bmp
+# End Source File
+# Begin Source File
+
+SOURCE=.\small.ico
+# End Source File
+# Begin Source File
+
+SOURCE=.\toolbar.bmp
+# End Source File
+# Begin Source File
+
+SOURCE=.\toolbar1.bmp
+# End Source File
+# Begin Source File
+
+SOURCE=.\Tower2.ICO
+# End Source File
+# Begin Source File
+
+SOURCE=.\TowerIC1.BMP
+# End Source File
+# End Group
+# Begin Source File
+
+SOURCE=.\ReadMe.txt
+# End Source File
+# End Target
+# End Project
diff --git a/ndb/src/cw/cpcc-win32/C++/CPC_GUI.dsw b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.dsw
new file mode 100644
index 00000000000..1f163a31662
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.dsw
@@ -0,0 +1,29 @@
+Microsoft Developer Studio Workspace File, Format Version 6.00
+# WARNING: DO NOT EDIT OR DELETE THIS WORKSPACE FILE!
+
+###############################################################################
+
+Project: "CPC_GUI"=.\CPC_GUI.dsp - Package Owner=<4>
+
+Package=<5>
+{{{
+}}}
+
+Package=<4>
+{{{
+}}}
+
+###############################################################################
+
+Global:
+
+Package=<5>
+{{{
+}}}
+
+Package=<3>
+{{{
+}}}
+
+###############################################################################
+
diff --git a/ndb/src/cw/cpcc-win32/C++/CPC_GUI.h b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.h
new file mode 100644
index 00000000000..cf7670948a7
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.h
@@ -0,0 +1,40 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#if !defined(AFX_CPC_GUI_H__EA01C861_C56D_48F1_856F_4935E20620B1__INCLUDED_)
+#define AFX_CPC_GUI_H__EA01C861_C56D_48F1_856F_4935E20620B1__INCLUDED_
+
+#if _MSC_VER > 1000
+#pragma once
+#endif // _MSC_VER > 1000
+
+
+#define MAX_LOADSTRING 100
+
+
+
+#define TV_ROOT_ITEMS 2
+
+
+// Global Variables
+
+ATOM NdbRegisterClass(HINSTANCE) ;
+BOOL InitInstance(HINSTANCE, int) ;
+LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM) ;
+LRESULT CALLBACK About(HWND, UINT, WPARAM, LPARAM);
+
+#endif // !defined(AFX_CPC_GUI_H__EA01C861_C56D_48F1_856F_4935E20620B1__INCLUDED_)
diff --git a/ndb/src/cw/cpcc-win32/C++/CPC_GUI.ico b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.ico
new file mode 100644
index 00000000000..386883523bc
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/C++/CPC_GUI.rc b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.rc
new file mode 100644
index 00000000000..41d75b2b282
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.rc
@@ -0,0 +1,193 @@
+//Microsoft Developer Studio generated resource script.
+//
+#include "resource.h"
+
+#define APSTUDIO_READONLY_SYMBOLS
+/////////////////////////////////////////////////////////////////////////////
+//
+// Generated from the TEXTINCLUDE 2 resource.
+//
+#define APSTUDIO_HIDDEN_SYMBOLS
+#include "windows.h"
+#undef APSTUDIO_HIDDEN_SYMBOLS
+#include "resource.h"
+
+/////////////////////////////////////////////////////////////////////////////
+#undef APSTUDIO_READONLY_SYMBOLS
+
+/////////////////////////////////////////////////////////////////////////////
+// English (U.S.) resources
+
+#if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU)
+#ifdef _WIN32
+LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US
+#pragma code_page(1252)
+#endif //_WIN32
+
+/////////////////////////////////////////////////////////////////////////////
+//
+// Icon
+//
+
+// Icon with lowest ID value placed first to ensure application icon
+// remains consistent on all systems.
+IDI_CPC_GUI ICON DISCARDABLE "CPC_GUI.ICO"
+IDI_SMALL ICON DISCARDABLE "SMALL.ICO"
+
+/////////////////////////////////////////////////////////////////////////////
+//
+// Menu
+//
+
+IDM_CPC_GUI MENU DISCARDABLE
+BEGIN
+ POPUP "&File"
+ BEGIN
+ MENUITEM "E&xit", IDM_EXIT
+ END
+ POPUP "&Actions"
+ BEGIN
+ MENUITEM "&Insert...", ID_ACTIONS_INSERT
+ MENUITEM "&Delete", ID_ACTIONS_DELETE
+ MENUITEM "&Properties", ID_ACTIONS_PROPERTIES
+ END
+ POPUP "&Help"
+ BEGIN
+ MENUITEM "&About ...", IDM_ABOUT
+ END
+END
+
+IDM_TREEVIEW MENU DISCARDABLE
+BEGIN
+ MENUITEM "&Insert", ID_TREEVIEW1
+ MENUITEM "&Delete", ID_DELETE
+ MENUITEM "&Properties", ID_PROPERTIES
+END
+
+
+/////////////////////////////////////////////////////////////////////////////
+//
+// Accelerator
+//
+
+IDC_CPC_GUI ACCELERATORS MOVEABLE PURE
+BEGIN
+ "?", IDM_ABOUT, ASCII, ALT
+ "/", IDM_ABOUT, ASCII, ALT
+END
+
+
+/////////////////////////////////////////////////////////////////////////////
+//
+// Dialog
+//
+
+IDD_ABOUTBOX DIALOG DISCARDABLE 22, 17, 230, 75
+STYLE DS_MODALFRAME | WS_CAPTION | WS_SYSMENU
+CAPTION "About"
+FONT 8, "MS Sans Serif"
+BEGIN
+ LTEXT "NDB Cluster Process Control Applet v1.0",IDC_STATIC,7,8,
+ 213,8,SS_NOPREFIX
+ LTEXT "Copyright (C) 2003 MySQL AB",
+ IDC_STATIC,7,20,213,20
+ DEFPUSHBUTTON "OK",IDOK,185,55,41,16,WS_GROUP
+END
+
+
+#ifdef APSTUDIO_INVOKED
+/////////////////////////////////////////////////////////////////////////////
+//
+// TEXTINCLUDE
+//
+
+2 TEXTINCLUDE MOVEABLE PURE
+BEGIN
+ "#define APSTUDIO_HIDDEN_SYMBOLS\r\n"
+ "#include ""windows.h""\r\n"
+ "#undef APSTUDIO_HIDDEN_SYMBOLS\r\n"
+ "#include ""resource.h""\r\n"
+ "\0"
+END
+
+3 TEXTINCLUDE MOVEABLE PURE
+BEGIN
+ "\r\n"
+ "\0"
+END
+
+1 TEXTINCLUDE MOVEABLE PURE
+BEGIN
+ "resource.h\0"
+END
+
+#endif // APSTUDIO_INVOKED
+
+
+/////////////////////////////////////////////////////////////////////////////
+//
+// Bitmap
+//
+
+IDR_TOOLBAR BITMAP MOVEABLE PURE "toolbar.bmp"
+IDB_TOOLBAR BITMAP MOVEABLE PURE "bitmap1.bmp"
+IDB_COMPUTER BITMAP MOVEABLE PURE "TowerIC1.BMP"
+IDB_OPEN BITMAP MOVEABLE PURE "Open.BMP"
+IDB_CLOSED BITMAP MOVEABLE PURE "Closed.BMP"
+IDB_DATABASE BITMAP MOVEABLE PURE "DB.bmp"
+
+/////////////////////////////////////////////////////////////////////////////
+//
+// Toolbar
+//
+
+IDR_TOOLBAR TOOLBAR MOVEABLE PURE 18, 18
+BEGIN
+ BUTTON ID_BUTTON32773
+ BUTTON ID_BUTTON32783
+ BUTTON ID_BUTTON32784
+END
+
+
+/////////////////////////////////////////////////////////////////////////////
+//
+// String Table
+//
+
+STRINGTABLE DISCARDABLE
+BEGIN
+ IDS_APP_TITLE "NDB Cluster Process Control Applet"
+ IDS_TV_ROOT_COMPUTERS "Computers"
+ IDS_TV_ROOT_DATABASES "Databases"
+ IDS_LV_COMPUTER_HEADER_1 "Computer"
+ IDS_LV_COMPUTER_HEADER_2 "Hostname"
+ IDS_LV_PROCESS_HEADER_1 "Process"
+ IDC_CPC_GUI "CPC_GUI"
+ IDS_TIP_NEW "Add new computer or database"
+ IDS_TIP_DELETE "Delete selected computer or database"
+END
+
+STRINGTABLE DISCARDABLE
+BEGIN
+ IDS_TIP_PROPS "Display properties for selected computer or database"
+ IDS_LV_PROCESS_HEADER_2 "Name"
+ IDS_LV_PROCESS_HEADER_3 "Owner"
+ IDS_LV_PROCESS_HEADER_4 "Status"
+ IDS_LV_COMPUTER_HEADER_3 "Status"
+END
+
+#endif // English (U.S.) resources
+/////////////////////////////////////////////////////////////////////////////
+
+
+
+#ifndef APSTUDIO_INVOKED
+/////////////////////////////////////////////////////////////////////////////
+//
+// Generated from the TEXTINCLUDE 3 resource.
+//
+
+
+/////////////////////////////////////////////////////////////////////////////
+#endif // not APSTUDIO_INVOKED
+
diff --git a/ndb/src/cw/cpcc-win32/C++/CPC_GUI.sln b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.sln
new file mode 100644
index 00000000000..86b574d851d
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.sln
@@ -0,0 +1,21 @@
+Microsoft Visual Studio Solution File, Format Version 7.00
+Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "CPC_GUI", "CPC_GUI.vcproj", "{F5FADD9D-4353-4A73-88DC-474A4D17B485}"
+EndProject
+Global
+ GlobalSection(SolutionConfiguration) = preSolution
+ ConfigName.0 = Debug
+ ConfigName.1 = Release
+ EndGlobalSection
+ GlobalSection(ProjectDependencies) = postSolution
+ EndGlobalSection
+ GlobalSection(ProjectConfiguration) = postSolution
+ {F5FADD9D-4353-4A73-88DC-474A4D17B485}.Debug.ActiveCfg = Debug|Win32
+ {F5FADD9D-4353-4A73-88DC-474A4D17B485}.Debug.Build.0 = Debug|Win32
+ {F5FADD9D-4353-4A73-88DC-474A4D17B485}.Release.ActiveCfg = Release|Win32
+ {F5FADD9D-4353-4A73-88DC-474A4D17B485}.Release.Build.0 = Release|Win32
+ EndGlobalSection
+ GlobalSection(ExtensibilityGlobals) = postSolution
+ EndGlobalSection
+ GlobalSection(ExtensibilityAddIns) = postSolution
+ EndGlobalSection
+EndGlobal
diff --git a/ndb/src/cw/cpcc-win32/C++/CPC_GUI.suo b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.suo
new file mode 100644
index 00000000000..e7d178f04c3
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.suo
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/C++/CPC_GUI.vcproj b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.vcproj
new file mode 100644
index 00000000000..56f9f3a8511
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/CPC_GUI.vcproj
@@ -0,0 +1,240 @@
+<?xml version="1.0" encoding = "Windows-1252"?>
+<VisualStudioProject
+ ProjectType="Visual C++"
+ Version="7.00"
+ Name="CPC_GUI"
+ SccProjectName=""
+ SccLocalPath="">
+ <Platforms>
+ <Platform
+ Name="Win32"/>
+ </Platforms>
+ <Configurations>
+ <Configuration
+ Name="Release|Win32"
+ OutputDirectory=".\Release"
+ IntermediateDirectory=".\Release"
+ ConfigurationType="1"
+ UseOfMFC="0"
+ ATLMinimizesCRunTimeLibraryUsage="FALSE"
+ CharacterSet="2">
+ <Tool
+ Name="VCCLCompilerTool"
+ InlineFunctionExpansion="1"
+ PreprocessorDefinitions="WIN32,NDEBUG,_WINDOWS"
+ StringPooling="TRUE"
+ RuntimeLibrary="4"
+ EnableFunctionLevelLinking="TRUE"
+ UsePrecompiledHeader="3"
+ PrecompiledHeaderThrough="stdafx.h"
+ PrecompiledHeaderFile=".\Release/CPC_GUI.pch"
+ AssemblerListingLocation=".\Release/"
+ ObjectFile=".\Release/"
+ ProgramDataBaseFileName=".\Release/"
+ WarningLevel="3"
+ SuppressStartupBanner="TRUE"/>
+ <Tool
+ Name="VCCustomBuildTool"/>
+ <Tool
+ Name="VCLinkerTool"
+ AdditionalOptions="/MACHINE:I386"
+ AdditionalDependencies="mfc42.lib"
+ OutputFile=".\Release/CPC_GUI.exe"
+ LinkIncremental="1"
+ SuppressStartupBanner="TRUE"
+ ProgramDatabaseFile=".\Release/CPC_GUI.pdb"
+ SubSystem="2"/>
+ <Tool
+ Name="VCMIDLTool"
+ PreprocessorDefinitions="NDEBUG"
+ MkTypLibCompatible="TRUE"
+ SuppressStartupBanner="TRUE"
+ TargetEnvironment="1"
+ TypeLibraryName=".\Release/CPC_GUI.tlb"/>
+ <Tool
+ Name="VCPostBuildEventTool"/>
+ <Tool
+ Name="VCPreBuildEventTool"/>
+ <Tool
+ Name="VCPreLinkEventTool"/>
+ <Tool
+ Name="VCResourceCompilerTool"
+ PreprocessorDefinitions="NDEBUG"
+ Culture="1033"/>
+ <Tool
+ Name="VCWebServiceProxyGeneratorTool"/>
+ <Tool
+ Name="VCWebDeploymentTool"/>
+ </Configuration>
+ <Configuration
+ Name="Debug|Win32"
+ OutputDirectory=".\Debug"
+ IntermediateDirectory=".\Debug"
+ ConfigurationType="1"
+ UseOfMFC="0"
+ ATLMinimizesCRunTimeLibraryUsage="FALSE"
+ CharacterSet="2">
+ <Tool
+ Name="VCCLCompilerTool"
+ Optimization="0"
+ PreprocessorDefinitions="WIN32,_DEBUG,_WINDOWS"
+ BasicRuntimeChecks="3"
+ RuntimeLibrary="5"
+ UsePrecompiledHeader="3"
+ PrecompiledHeaderThrough="stdafx.h"
+ PrecompiledHeaderFile=".\Debug/CPC_GUI.pch"
+ AssemblerListingLocation=".\Debug/"
+ ObjectFile=".\Debug/"
+ ProgramDataBaseFileName=".\Debug/"
+ BrowseInformation="1"
+ WarningLevel="3"
+ SuppressStartupBanner="TRUE"
+ DebugInformationFormat="4"/>
+ <Tool
+ Name="VCCustomBuildTool"/>
+ <Tool
+ Name="VCLinkerTool"
+ AdditionalOptions="/MACHINE:I386"
+ AdditionalDependencies="comctl32.lib mfc70d.lib"
+ OutputFile=".\Debug/CPC_GUI.exe"
+ LinkIncremental="2"
+ SuppressStartupBanner="TRUE"
+ GenerateDebugInformation="TRUE"
+ ProgramDatabaseFile=".\Debug/CPC_GUI.pdb"
+ SubSystem="2"/>
+ <Tool
+ Name="VCMIDLTool"
+ PreprocessorDefinitions="_DEBUG"
+ MkTypLibCompatible="TRUE"
+ SuppressStartupBanner="TRUE"
+ TargetEnvironment="1"
+ TypeLibraryName=".\Debug/CPC_GUI.tlb"/>
+ <Tool
+ Name="VCPostBuildEventTool"/>
+ <Tool
+ Name="VCPreBuildEventTool"/>
+ <Tool
+ Name="VCPreLinkEventTool"/>
+ <Tool
+ Name="VCResourceCompilerTool"
+ PreprocessorDefinitions="_DEBUG"
+ Culture="1033"/>
+ <Tool
+ Name="VCWebServiceProxyGeneratorTool"/>
+ <Tool
+ Name="VCWebDeploymentTool"/>
+ </Configuration>
+ </Configurations>
+ <Files>
+ <Filter
+ Name="Source Files"
+ Filter="cpp;c;cxx;rc;def;r;odl;idl;hpj;bat">
+ <File
+ RelativePath=".\CPC_GUI.cpp">
+ </File>
+ <File
+ RelativePath=".\CPC_GUI.rc">
+ </File>
+ <File
+ RelativePath=".\NdbControls.cpp">
+ </File>
+ <File
+ RelativePath=".\StdAfx.cpp">
+ <FileConfiguration
+ Name="Release|Win32">
+ <Tool
+ Name="VCCLCompilerTool"
+ UsePrecompiledHeader="1"/>
+ </FileConfiguration>
+ <FileConfiguration
+ Name="Debug|Win32">
+ <Tool
+ Name="VCCLCompilerTool"
+ UsePrecompiledHeader="1"/>
+ </FileConfiguration>
+ </File>
+ </Filter>
+ <Filter
+ Name="Header Files"
+ Filter="h;hpp;hxx;hm;inl">
+ <File
+ RelativePath=".\CPC_GUI.h">
+ </File>
+ <File
+ RelativePath=".\NdbControls.h">
+ </File>
+ <File
+ RelativePath=".\StdAfx.h">
+ </File>
+ <File
+ RelativePath=".\resource.h">
+ </File>
+ </Filter>
+ <Filter
+ Name="Resource Files"
+ Filter="ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe">
+ <File
+ RelativePath=".\C.bmp">
+ </File>
+ <File
+ RelativePath=".\CPC_GUI.ico">
+ </File>
+ <File
+ RelativePath=".\Closed.BMP">
+ </File>
+ <File
+ RelativePath=".\Closed.ICO">
+ </File>
+ <File
+ RelativePath=".\Closed24.bmp">
+ </File>
+ <File
+ RelativePath=".\Computer24.BMP">
+ </File>
+ <File
+ RelativePath=".\Db.bmp">
+ </File>
+ <File
+ RelativePath=".\O.bmp">
+ </File>
+ <File
+ RelativePath=".\Open.BMP">
+ </File>
+ <File
+ RelativePath=".\Open.ICO">
+ </File>
+ <File
+ RelativePath=".\Open24.bmp">
+ </File>
+ <File
+ RelativePath=".\Tower2.ICO">
+ </File>
+ <File
+ RelativePath=".\TowerIC1.BMP">
+ </File>
+ <File
+ RelativePath=".\bitmap1.bmp">
+ </File>
+ <File
+ RelativePath=".\bmp00001.bmp">
+ </File>
+ <File
+ RelativePath=".\icon1.ico">
+ </File>
+ <File
+ RelativePath=".\small.ico">
+ </File>
+ <File
+ RelativePath=".\toolbar.bmp">
+ </File>
+ <File
+ RelativePath=".\toolbar1.bmp">
+ </File>
+ </Filter>
+ <File
+ RelativePath=".\ReadMe.txt">
+ </File>
+ </Files>
+ <Globals>
+ </Globals>
+</VisualStudioProject>
diff --git a/ndb/src/cw/cpcc-win32/C++/Closed.ICO b/ndb/src/cw/cpcc-win32/C++/Closed.ICO
new file mode 100644
index 00000000000..044042b42fb
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/Closed.ICO
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/C++/NdbControls.cpp b/ndb/src/cw/cpcc-win32/C++/NdbControls.cpp
new file mode 100644
index 00000000000..6bbc9a9859b
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/NdbControls.cpp
@@ -0,0 +1,436 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "stdafx.h"
+#include "NdbControls.h"
+
+
+/**
+* CNdbControl implementation
+*/
+
+BOOL CNdbControl::GetRect(LPRECT lprc) const {
+
+ _ASSERT(this) ;
+
+ return GetClientRect(m_hControl, lprc) ;
+
+}
+
+BOOL CNdbControl::Resize(LONG x, LONG y, LONG w, LONG h) const {
+
+ _ASSERT(this) ;
+
+ if(!MoveWindow(m_hControl, x, y, w, h, TRUE))
+ return FALSE ;
+ if(m_bVisible){
+ ShowWindow(m_hControl, SW_SHOW) ;
+ UpdateWindow(m_hControl) ;
+ }
+ return TRUE ;
+
+}
+
+BOOL CNdbControl::Show(BOOL bShow) {
+
+ _ASSERT(this) ;
+
+ if(bShow){
+ ShowWindow(m_hControl, SW_SHOW);
+ m_bVisible = TRUE ;
+ }else{
+ ShowWindow(m_hControl, SW_HIDE);
+ m_bVisible = FALSE ;
+ }
+ EnableWindow(m_hControl, bShow) ;
+ UpdateWindow(m_hControl) ;
+
+ return TRUE ;
+}
+
+
+
+CNdbControl::~CNdbControl(){
+
+ DestroyWindow(m_hControl) ;
+ if(m_hMenu)
+ DestroyMenu(m_hMenu) ;
+
+}
+
+
+/**
+* CNdbListView implementation
+*/
+
+BOOL CNdbListView::Create(HINSTANCE hInst, HWND hParent, DWORD dwId, NDB_ITEM_TYPE enType, PNDB_LV pstH, DWORD dwWidth) {
+
+ if(!pstH)
+ return FALSE ;
+
+ LV_COLUMN lvC ;
+ m_hInstance = hInst ;
+ m_hParent = hParent ;
+ m_dwId = dwId ;
+ m_dwWidth = dwWidth ;
+ m_dwWidth = 100 ;
+ m_enType = enType;
+ char* szLabels[MAX_LV_HEADERS] ;
+ int count = 0 ;
+
+ m_hControl = CreateWindowEx(WS_EX_OVERLAPPEDWINDOW, WC_LISTVIEW, TEXT(""),
+ WS_VISIBLE | WS_CHILD | WS_BORDER | LVS_REPORT,
+ 0, 0, 0, 0, m_hParent, (HMENU)m_dwId, hInst, NULL );
+
+ if(!m_hControl)
+ return FALSE ;
+
+ lvC.mask = LVCF_FMT | LVCF_WIDTH | LVCF_TEXT | LVCF_SUBITEM;
+ lvC.fmt = LVCFMT_LEFT;
+
+ switch(enType){
+ case ITEM_COMPR_ROOT:
+ szLabels[0] = pstH->szComputer ;
+ szLabels[1] = pstH->szHostname ;
+ szLabels[2] = pstH->szStatus ;
+ count = 3 ;
+ break ;
+ case ITEM_DB_ROOT:
+ szLabels[0] = pstH->szDatabase ;
+ szLabels[1] = pstH->szStatus ;
+ count = 2 ;
+ break ;
+ case ITEM_COMPR:
+ szLabels[0] = pstH->szProcess ;
+ szLabels[1] = pstH->szDatabase;
+ szLabels[2] = pstH->szOwner ;
+ szLabels[3] = pstH->szStatus ;
+ count = 4 ;
+ case ITEM_DB:
+ szLabels[0] = pstH->szProcess ;
+ szLabels[1] = pstH->szComputer;
+ szLabels[2] = pstH->szOwner ;
+ szLabels[3] = pstH->szStatus ;
+ count = 4 ;
+ break ;
+ NDB_DEFAULT_UNREACHABLE ;
+ }
+
+ for(int j = 0 ; j < count ; ++j){
+ lvC.iSubItem = j ;
+ lvC.cx = m_dwWidth ;
+ lvC.pszText = szLabels[j] ;
+ if(0xFFFFFFFF == ListView_InsertColumn(m_hControl, j, &lvC))
+ return FALSE ;
+ }
+
+ SendMessage(m_hControl, LVM_SETEXTENDEDLISTVIEWSTYLE, LVS_EX_FULLROWSELECT,
+ LVS_EX_FULLROWSELECT );
+
+ ShowWindow(m_hControl, SW_SHOW) ;
+
+ return TRUE ;
+
+}
+
+
+
+/**
+* CNdbToolBar implementation
+*/
+
+
+
+/**
+* CNdbTreeView implementation
+*/
+
+BOOL CNdbTreeView::Create(HINSTANCE hInst, HWND hParent, DWORD dwMenuId, DWORD dwId){
+
+ if(!CreateTreeView(hInst, hParent, dwId))
+ return FALSE ;
+
+ m_hMenu = LoadMenu(m_hInstance,MAKEINTRESOURCE(dwMenuId)) ;
+ if(!m_hMenu)
+ return FALSE ;
+
+ return TRUE ;
+}
+
+
+BOOL CNdbTreeView::CreateTreeView(HINSTANCE hInst, HWND hParent, DWORD dwId){
+
+
+ m_hInstance = hInst ;
+ m_hParent = hParent ;
+ m_dwId = dwId ;
+ HIMAGELIST himl ;
+ HBITMAP hbmp ;
+ DWORD dwCount = 0 ;
+
+ m_hControl = CreateWindowEx(WS_EX_OVERLAPPEDWINDOW, WC_TREEVIEW, "Tree View",
+ WS_VISIBLE | WS_CHILD | WS_BORDER | TVS_HASLINES |
+ TVS_HASBUTTONS | TVS_LINESATROOT | TVS_SINGLEEXPAND,
+ 0, 0, 0, 0, m_hParent, (HMENU)m_dwId, m_hInstance, NULL) ;
+
+ if(!m_hControl)
+ return FALSE ;
+
+ if((himl = ImageList_Create(nX, nY, ILC_MASK | ILC_COLOR8, 4, 0)) == NULL)
+ return FALSE ;
+
+ hbmp = LoadBitmap(m_hInstance, MAKEINTRESOURCE(IDI_OPEN));
+ hbmp = (HBITMAP)LoadImage(m_hInstance, MAKEINTRESOURCE(IDB_OPEN), IMAGE_BITMAP, nX, 0, LR_DEFAULTSIZE);
+ m_nOpen = ImageList_AddMasked(himl, hbmp, clr);
+ DeleteObject(hbmp);
+ hbmp = (HBITMAP)LoadImage(m_hInstance, MAKEINTRESOURCE(IDB_CLOSED), IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE);
+ m_nClosed = ImageList_AddMasked(himl, hbmp, clr);
+ DeleteObject(hbmp);
+ hbmp = (HBITMAP)LoadImage(m_hInstance, MAKEINTRESOURCE(IDB_COMPUTER),IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE);
+ m_nComputer = ImageList_AddMasked(himl, hbmp, clr);
+ DeleteObject(hbmp);
+ hbmp = (HBITMAP)LoadImage(m_hInstance, MAKEINTRESOURCE(IDB_DATABASE), IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE);
+ m_nDatabase = ImageList_AddMasked(himl, hbmp, clr);
+ DeleteObject(hbmp);
+
+ if(ImageList_GetImageCount(himl) < 4)
+ return FALSE ;
+
+ TreeView_SetImageList(m_hControl, himl, TVSIL_NORMAL);
+
+ ShowWindow(m_hControl, SW_SHOW) ;
+
+ return TRUE ;
+
+}
+
+
+
+HTREEITEM CNdbTreeView::AddItem(LPSTR szText, NDB_ITEM_TYPE enType, DWORD dwLVId){
+
+ TVITEM tvi ;
+ TVINSERTSTRUCT tvins ;
+ HTREEITEM hti ;
+ HTREEITEM hTemp ;
+ int nImage = m_nClosed ;
+
+ tvi.mask = TVIF_TEXT | TVIF_IMAGE
+ | TVIF_SELECTEDIMAGE | TVIF_PARAM;
+
+ tvi.pszText = szText;
+ tvi.cchTextMax = lstrlen(szText);
+
+ switch(enType){
+
+ case ITEM_COMPR_ROOT:
+ nImage = m_nClosed ;
+ if(!m_hPrevRoot)
+ tvins.hParent = TVI_ROOT;
+ else
+ tvins.hInsertAfter = m_hPrevRoot ;
+ break ;
+
+ case ITEM_DB_ROOT:
+ if(!m_hPrevRoot)
+ tvins.hParent = TVI_ROOT;
+ else
+ tvins.hInsertAfter = m_hPrevRoot ;
+ break ;
+
+ case ITEM_COMPR:
+ nImage = m_nComputer ;
+ if(!m_hPrevComputersChild || !m_hComputersRoot)
+ return 0 ;
+ else
+ tvins.hInsertAfter = m_hPrevComputersChild ;
+ tvins.hParent = m_hComputersRoot ;
+ break ;
+
+ case ITEM_DB:
+ nImage = m_nDatabase ;
+ if(!m_hPrevComputersChild || !m_hComputersRoot)
+ return 0 ;
+ else
+ tvins.hInsertAfter = m_hPrevDatabasesChild ;
+ tvins.hParent = m_hDatabasesRoot ;
+ break ;
+
+ NDB_DEFAULT_UNREACHABLE ;
+
+ }
+
+ tvi.iImage = nImage ;
+ tvi.iSelectedImage = nImage ;
+ tvi.lParam = (LPARAM) dwLVId ;
+ tvins.item = tvi ;
+
+ hTemp = TreeView_InsertItem(m_hControl, &tvins);
+ if(!hTemp)
+ return NULL ;
+
+ switch(enType){
+
+ case ITEM_COMPR_ROOT:
+ m_hComputersRoot = hTemp ;
+ break ;
+
+ case ITEM_DB_ROOT:
+ m_hDatabasesRoot = hTemp ;
+ break ;
+
+ case ITEM_COMPR:
+ m_hPrevComputersChild = hTemp ;
+ break ;
+
+ case ITEM_DB:
+ m_hPrevComputersChild = hTemp ;
+ break ;
+
+ NDB_DEFAULT_UNREACHABLE ;
+
+ }
+
+ if (ITEM_COMPR_ROOT != enType && ITEM_DB_ROOT != enType) {
+
+ hti = TreeView_GetParent(m_hControl, hTemp);
+ tvi.mask = TVIF_IMAGE | TVIF_SELECTEDIMAGE;
+ tvi.hItem = hti;
+ tvi.iImage = m_nClosed;
+ tvi.iSelectedImage = m_nClosed;
+ TreeView_SetItem(m_hControl, &tvi);
+
+ }
+
+ return hTemp ;
+}
+
+
+BOOL CNdbControls::Create(HINSTANCE hInst, HWND hParent){
+
+ m_hInstance = hInst ;
+ m_hParent = hParent ;
+ m_tb.Create(m_hInstance, m_hParent, ID_TOOLBAR, IDB_TOOLBAR) ;
+ m_sb.Create(m_hInstance, m_hParent, ID_STATUSBAR) ;
+ m_tv.Create(m_hInstance, m_hParent, IDM_TREEVIEW, ID_TREEVIEW) ;
+ _assert(AddView("Computers", ITEM_COMPR_ROOT)) ;
+ _assert(AddView("Databases", ITEM_DB_ROOT)) ;
+
+ return TRUE ;
+}
+
+BOOL CNdbControls::AddListView(NDB_ITEM_TYPE enType, DWORD dwId){
+
+ int count ;
+ CNdbListView* plv ;
+ PNDB_LV pst ;
+
+ plv = new CNdbListView ;
+
+ if(!plv)
+ return FALSE ;
+
+ count = m_map_lvc.GetCount() + m_dwFirstId_lv ;
+
+ switch(enType){
+ case ITEM_COMPR_ROOT:
+ pst = &m_stlvcRoot ;
+ break ;
+ case ITEM_DB_ROOT:
+ pst = &m_stlvdRoot ;
+ break ;
+ case ITEM_COMPR:
+ pst = &m_stlvc ;
+ break ;
+ case ITEM_DB:
+ pst = &m_stlvd ;
+ break ;
+ NDB_DEFAULT_UNREACHABLE ;
+ }
+
+ plv->Create(m_hInstance, m_hParent, dwId, enType, pst, LV_HEADER_WIDTH) ;
+
+ m_map_lvc[count] = plv ;
+
+ return TRUE ;
+}
+
+BOOL CNdbControls::AddView(LPSTR szText, NDB_ITEM_TYPE enType){
+
+ DWORD dwId_lv = m_dwNextId_lv ;
+
+ if(AddListView(enType, dwId_lv) && m_tv.AddItem(szText, enType, dwId_lv))
+ m_dwNextId_lv++ ;
+ else
+ return FALSE ;
+
+ return TRUE ;
+};
+
+
+VOID CNdbControls::ToggleListViews(LPNMTREEVIEW pnmtv){
+
+ CNdbListView* plv ;
+ int count = m_map_lvc.GetCount() + m_dwFirstId_lv ;
+
+ for(int c = FIRST_ID_LV ; c < count; ++c){
+ _assert(m_map_lvc.Lookup(c, plv)) ;
+ if(pnmtv->itemNew.lParam == (c))
+ plv->Show(TRUE) ;
+ else
+ plv->Show(FALSE) ;
+ }
+}
+
+
+
+VOID CNdbControls::Resize(){
+
+ RECT rc, rcTB, rcSB ;
+ LONG tw, sw, lx, ly, lw, lh, tvw, tvh ;
+ CNdbListView* plv ;
+ int count ; //, id ;
+
+ GetClientRect(m_hParent, &rc) ;
+ m_tb.GetRect(&rcTB) ;
+ m_sb.GetRect(&rcSB) ;
+
+ sw = rcSB.bottom ;
+ tw = rcTB.bottom ;
+
+ m_tb.Resize(0, 0, rc.right, tw) ;
+
+ tvw = rc.right / 4 ;
+ tvh = rc.bottom - sw - tw - BORDER ;
+
+ m_tv.Resize(0, tw + BORDER, tvw, tvh) ;
+
+ m_sb.Resize(0, tvh, rc.left, sw) ;
+
+ lx = tvw + BORDER - 2 ;
+ ly = tw + BORDER ;
+ lw = rc.right - tvw - BORDER + 1 ;
+ lh = tvh ;
+
+ count = m_map_lvc.GetCount() + FIRST_ID_LV ;
+
+ for(int c = FIRST_ID_LV ; c < count; ++c){
+ _assert(m_map_lvc.Lookup(c, plv)) ;
+ plv->Resize(lx, ly, lw, lh) ;
+ }
+
+ return ;
+
+}
diff --git a/ndb/src/cw/cpcc-win32/C++/Open.ICO b/ndb/src/cw/cpcc-win32/C++/Open.ICO
new file mode 100644
index 00000000000..ab7b05d9df7
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/Open.ICO
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/C++/StdAfx.cpp b/ndb/src/cw/cpcc-win32/C++/StdAfx.cpp
new file mode 100644
index 00000000000..8fcdb4ce158
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/StdAfx.cpp
@@ -0,0 +1,24 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+// stdafx.cpp : source file that includes just the standard includes
+// CPC_GUI.pch will be the pre-compiled header
+// stdafx.obj will contain the pre-compiled type information
+
+#include "stdafx.h"
+
+// TODO: reference any additional headers you need in STDAFX.H
+// and not in this file
diff --git a/ndb/src/cw/cpcc-win32/C++/StdAfx.h b/ndb/src/cw/cpcc-win32/C++/StdAfx.h
new file mode 100644
index 00000000000..d84b5811f8d
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/StdAfx.h
@@ -0,0 +1,72 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+// stdafx.h : include file for standard system include files,
+// or project specific include files that are used frequently, but
+// are changed infrequently
+//
+
+#if !defined(AFX_STDAFX_H__A9DB83DB_A9FD_11D0_BFD1_444553540000__INCLUDED_)
+#define AFX_STDAFX_H__A9DB83DB_A9FD_11D0_BFD1_444553540000__INCLUDED_
+
+#if _MSC_VER > 1000
+#pragma once
+#endif // _MSC_VER > 1000
+
+#define WIN32_LEAN_AND_MEAN // Exclude rarely-used stuff from Windows headers
+
+#ifdef _DEBUG
+#define NDB_DEFAULT_UNREACHABLE default: _ASSERT(0); break
+#elif _MSC_VER >= 1200
+#define NDB_DEFAULT_UNREACHABLE default: __assume(0); break
+#else
+#define NDB_DEFAULT_UNREACHABLE default: break
+#endif;
+
+
+#ifdef _DEBUG
+#define _assert _ASSERT
+#else
+#define _assert(expr) expr
+#endif
+
+
+#include <afx.h>
+#include <afxtempl.h>
+
+// C RunTime Header Files
+#include <stdlib.h>
+#include <malloc.h>
+#include <memory.h>
+#include <tchar.h>
+#include <commctrl.h>
+#include <shlwapi.h>
+#include <crtdbg.h>
+#include <string.h>
+#include <stdio.h>
+
+// Local Header Files
+#include "resource.h"
+#include "NdbControls.h"
+#include "CPC_GUI.h"
+
+
+// TODO: reference additional headers your program requires here
+
+//{{AFX_INSERT_LOCATION}}
+// Microsoft Visual C++ will insert additional declarations immediately before the previous line.
+
+#endif // !defined(AFX_STDAFX_H__A9DB83DB_A9FD_11D0_BFD1_444553540000__INCLUDED_)
diff --git a/ndb/src/cw/cpcc-win32/C++/TreeView.cpp b/ndb/src/cw/cpcc-win32/C++/TreeView.cpp
new file mode 100644
index 00000000000..db5c62f14bb
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/TreeView.cpp
@@ -0,0 +1,19 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "StdAfx.h"
+#include "resource.h"
+#include "CPC_GUI.h"
diff --git a/ndb/src/cw/cpcc-win32/C++/TreeView.h b/ndb/src/cw/cpcc-win32/C++/TreeView.h
new file mode 100644
index 00000000000..595f9bd6cdc
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/TreeView.h
@@ -0,0 +1,19 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+
diff --git a/ndb/src/cw/cpcc-win32/C++/bmp00001.bmp b/ndb/src/cw/cpcc-win32/C++/bmp00001.bmp
new file mode 100644
index 00000000000..e50af403eda
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/bmp00001.bmp
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/C++/resource.h b/ndb/src/cw/cpcc-win32/C++/resource.h
new file mode 100644
index 00000000000..0bec552edf6
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/resource.h
@@ -0,0 +1,90 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//{{NO_DEPENDENCIES}}
+// Microsoft Developer Studio generated include file.
+// Used by CPC_GUI.rc
+//
+#define IDC_MYICON 2
+#define IDD_CPC_GUI_DIALOG 102
+#define IDD_ABOUTBOX 103
+#define IDS_APP_TITLE 103
+#define IDM_ABOUT 104
+#define IDS_LV_ROOT_COMPUTERS 104
+#define IDS_TV_ROOT_COMPUTERS 104
+#define IDM_EXIT 105
+#define IDS_LV_ROOT_DATABASES 105
+#define IDS_TV_ROOT_DATABASES 105
+#define IDS_HELLO 106
+#define IDS_LV_COMPUTER_HEADER_1 106
+#define IDI_CPC_GUI 107
+#define IDS_LV_COMPUTER_HEADER_2 107
+#define IDI_SMALL 108
+#define IDS_LV_PROCESS_HEADER_1 108
+#define IDC_CPC_GUI 109
+#define IDM_CPC_GUI 109
+#define IDS_TIP_NEW 110
+#define IDS_TIP_DELETE 111
+#define IDS_TIP_PROPS 112
+#define IDS_LV_PROCESS_HEADER_2 113
+#define IDS_LV_PROCESS_HEADER_3 114
+#define IDS_LV_PROCESS_HEADER_4 115
+#define IDS_LV_COMPUTER_HEADER_3 116
+#define IDR_MAINFRAME 128
+#define ID_TREEVIEW 130
+#define IDM_TREEVIEW 130
+#define IDB_TOOLBAR 137
+#define ID_TOOLBAR 158
+#define IDB_COMPUTER 168
+#define IDB_CLOSED 169
+#define IDB_OPEN 170
+#define IDI_DATABASE 172
+#define IDI_CLOSED 175
+#define IDI_OPEN 176
+#define IDI_COMPUTER 177
+#define IDB_MASK 178
+#define IDB_DATABASE 182
+#define IDM_TV 183
+#define ID_TREEVIEW1 32771
+#define ID_BUTTON32773 32773
+#define IDM_NEW 32774
+#define IDM_DELETE 32775
+#define IDM_PROPS 32776
+#define ID_LIST_C 32777
+#define ID_ACTIONS_INSERT 32778
+#define ID_ACTIONS_DELETE 32779
+#define ID_DELETE 32780
+#define ID_PROPERTIES 32781
+#define ID_ACTIONS_PROPERTIES 32782
+#define ID_BUTTON32783 32783
+#define ID_BUTTON32784 32784
+#define ID_LIST_P 32785
+#define ID_STATUSBAR 32786
+#define ID_LIST_C_ROOT 32787
+#define ID_LIST_D_ROOT 32788
+#define IDM_ADDNEW 32793
+#define IDC_STATIC -1
+
+// Next default values for new objects
+//
+#ifdef APSTUDIO_INVOKED
+#ifndef APSTUDIO_READONLY_SYMBOLS
+#define _APS_NEXT_RESOURCE_VALUE 184
+#define _APS_NEXT_COMMAND_VALUE 32796
+#define _APS_NEXT_CONTROL_VALUE 1000
+#define _APS_NEXT_SYMED_VALUE 110
+#endif
+#endif
diff --git a/ndb/src/cw/cpcc-win32/C++/small.ico b/ndb/src/cw/cpcc-win32/C++/small.ico
new file mode 100644
index 00000000000..8f94d9aa828
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/small.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/C++/toolbar.bmp b/ndb/src/cw/cpcc-win32/C++/toolbar.bmp
new file mode 100644
index 00000000000..a1059352c66
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/C++/toolbar.bmp
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/csharp/App.ico b/ndb/src/cw/cpcc-win32/csharp/App.ico
new file mode 100644
index 00000000000..3a5525fd794
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/App.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/csharp/AssemblyInfo.cs b/ndb/src/cw/cpcc-win32/csharp/AssemblyInfo.cs
new file mode 100644
index 00000000000..9f89a3282c5
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/AssemblyInfo.cs
@@ -0,0 +1,58 @@
+using System.Reflection;
+using System.Runtime.CompilerServices;
+
+//
+// General Information about an assembly is controlled through the following
+// set of attributes. Change these attribute values to modify the information
+// associated with an assembly.
+//
+[assembly: AssemblyTitle("")]
+[assembly: AssemblyDescription("")]
+[assembly: AssemblyConfiguration("")]
+[assembly: AssemblyCompany("")]
+[assembly: AssemblyProduct("")]
+[assembly: AssemblyCopyright("")]
+[assembly: AssemblyTrademark("")]
+[assembly: AssemblyCulture("")]
+
+//
+// Version information for an assembly consists of the following four values:
+//
+// Major Version
+// Minor Version
+// Build Number
+// Revision
+//
+// You can specify all the values or you can default the Revision and Build Numbers
+// by using the '*' as shown below:
+
+[assembly: AssemblyVersion("1.0.*")]
+
+//
+// In order to sign your assembly you must specify a key to use. Refer to the
+// Microsoft .NET Framework documentation for more information on assembly signing.
+//
+// Use the attributes below to control which key is used for signing.
+//
+// Notes:
+// (*) If no key is specified, the assembly is not signed.
+// (*) KeyName refers to a key that has been installed in the Crypto Service
+// Provider (CSP) on your machine. KeyFile refers to a file which contains
+// a key.
+// (*) If the KeyFile and the KeyName values are both specified, the
+// following processing occurs:
+// (1) If the KeyName can be found in the CSP, that key is used.
+// (2) If the KeyName does not exist and the KeyFile does exist, the key
+// in the KeyFile is installed into the CSP and used.
+// (*) In order to create a KeyFile, you can use the sn.exe (Strong Name) utility.
+// When specifying the KeyFile, the location of the KeyFile should be
+// relative to the project output directory which is
+// %Project Directory%\obj\<configuration>. For example, if your KeyFile is
+// located in the project directory, you would specify the AssemblyKeyFile
+// attribute as [assembly: AssemblyKeyFile("..\\..\\mykey.snk")]
+// (*) Delay Signing is an advanced option - see the Microsoft .NET Framework
+// documentation for more information on this.
+//
+[assembly: AssemblyDelaySign(false)]
+[assembly: AssemblyKeyFile("")]
+[assembly: AssemblyKeyName("")]
diff --git a/ndb/src/cw/cpcc-win32/csharp/CPC_Form.cs b/ndb/src/cw/cpcc-win32/csharp/CPC_Form.cs
new file mode 100644
index 00000000000..ea1798c8c67
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/CPC_Form.cs
@@ -0,0 +1,1400 @@
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+using System.Data;
+using System.Threading;
+
+namespace NDB_CPC
+{
+ /// <summary>
+ /// Summary description for Form1.
+ /// </summary>
+ public class CPC : System.Windows.Forms.Form
+ {
+ private System.Windows.Forms.TreeView tvComputerCluster;
+ private System.Windows.Forms.ContextMenu ctxTreeViewMenu;
+ private System.Windows.Forms.ColumnHeader chComputer;
+ private System.Windows.Forms.ColumnHeader chProcessName;
+ private System.Windows.Forms.ContextMenu ctxListViewMenu;
+ private System.Windows.Forms.MenuItem mainMenuItem;
+ private System.Windows.Forms.ColumnHeader chProcesses;
+ private System.Windows.Forms.MainMenu mainMenu;
+ private System.Windows.Forms.Panel panel1;
+ private System.Windows.Forms.MenuItem menuItem7;
+ private System.Windows.Forms.MenuItem menuItem10;
+ private System.Windows.Forms.MenuItem mainMenuFile;
+ private System.Windows.Forms.MenuItem mainMenuComputer;
+ private System.Windows.Forms.MenuItem subMenuComputerAdd;
+ private System.Windows.Forms.MenuItem subMenuComputerRemove;
+ private System.Windows.Forms.MenuItem subMenuComputerDisconnect;
+ private System.Windows.Forms.MenuItem subMenuComputerProperties;
+ private System.ComponentModel.IContainer components;
+
+ private System.Windows.Forms.MenuItem menuItem3;
+ private System.Windows.Forms.MenuItem computerMenuAdd;
+ private System.Windows.Forms.MenuItem computerMenuRemove;
+ private System.Windows.Forms.MenuItem menuItem5;
+ private System.Windows.Forms.MenuItem computerMenuDisconnect;
+ private System.Windows.Forms.MenuItem computerMenuConnect;
+ private System.Windows.Forms.MenuItem computerMenuProperties;
+ private System.Windows.Forms.MenuItem menuItem11;
+ private System.Windows.Forms.MenuItem tvCtxMenuComputerAdd;
+ private System.Windows.Forms.MenuItem tvCtxMenuComputerRemove;
+ private System.Windows.Forms.MenuItem tvCtxMenuComputerConnect;
+ private System.Windows.Forms.MenuItem tvCtxMenuComputerDisconnect;
+ private System.Windows.Forms.MenuItem tvCtxMenuComputerDefine;
+ private System.Windows.Forms.MenuItem tvCtxMenuDatabaseNew;
+ private System.Windows.Forms.MenuItem menuItem1;
+ private System.Windows.Forms.MenuItem menuItem2;
+ private System.Windows.Forms.MenuItem mainMenuDatabase;
+ private System.Windows.Forms.MenuItem subMenuDatabaseCreate;
+ private System.Windows.Forms.MenuItem menuItem8;
+ private System.Windows.Forms.MenuItem tvCtxMenuProperties;
+ private System.Windows.Forms.ImageList imageTV;
+
+ private ComputerMgmt computerMgmt;
+ private System.Windows.Forms.MenuItem computerMenuRefresh;
+ private System.Windows.Forms.ListView listView;
+ private System.Windows.Forms.ColumnHeader chComputerIP;
+ private System.Windows.Forms.ColumnHeader chDatabase;
+ private System.Windows.Forms.ColumnHeader chName;
+ private System.Windows.Forms.ColumnHeader chOwner;
+ private System.Windows.Forms.ColumnHeader chStatus;
+ private System.Windows.Forms.Splitter splitter2;
+ private System.Windows.Forms.Splitter splitterVertical;
+ private System.Windows.Forms.Splitter splitterHorizont;
+ private Thread guiThread;
+ private float resizeWidthRatio;
+ private System.Windows.Forms.MenuItem menuItem6;
+ private System.Windows.Forms.MenuItem menuGetStatus;
+ private System.Windows.Forms.MenuItem menuStartProcess;
+ private System.Windows.Forms.MenuItem menuRestartProcess;
+ private System.Windows.Forms.MenuItem menuStopProcess;
+ private System.Windows.Forms.MenuItem menuRemoveProcess;
+ private System.Windows.Forms.MenuItem menuRefresh;
+ private System.Windows.Forms.OpenFileDialog openHostFileDialog;
+ private System.Windows.Forms.SaveFileDialog saveHostFileDialog;
+ private float resizeHeightRatio;
+ private System.Windows.Forms.TextBox mgmConsole;
+ int i;
+ public CPC()
+ {
+ //
+ // Required for Windows Form Designer support
+ //
+ InitializeComponent();
+
+ // TODO: Add any constructor code after InitializeComponent call
+ //
+ computerMgmt = new ComputerMgmt();
+ guiThread = new Thread(new ThreadStart(updateGuiThread));
+
+ // guiThread.Start();
+ }
+
+ /// <summary>
+ /// Clean up any resources being used.
+ /// </summary>
+ protected override void Dispose( bool disposing )
+ {
+ if( disposing )
+ {
+ if (components != null)
+ {
+ components.Dispose();
+ }
+ }
+ //guiThread.Abort();
+ base.Dispose( disposing );
+ }
+
+ #region Windows Form Designer generated code
+ /// <summary>
+ /// Required method for Designer support - do not modify
+ /// the contents of this method with the code editor.
+ /// </summary>
+ private void InitializeComponent()
+ {
+ this.components = new System.ComponentModel.Container();
+ System.Resources.ResourceManager resources = new System.Resources.ResourceManager(typeof(CPC));
+ this.tvComputerCluster = new System.Windows.Forms.TreeView();
+ this.ctxTreeViewMenu = new System.Windows.Forms.ContextMenu();
+ this.tvCtxMenuComputerAdd = new System.Windows.Forms.MenuItem();
+ this.tvCtxMenuComputerRemove = new System.Windows.Forms.MenuItem();
+ this.menuGetStatus = new System.Windows.Forms.MenuItem();
+ this.menuItem6 = new System.Windows.Forms.MenuItem();
+ this.tvCtxMenuComputerConnect = new System.Windows.Forms.MenuItem();
+ this.tvCtxMenuComputerDisconnect = new System.Windows.Forms.MenuItem();
+ this.tvCtxMenuDatabaseNew = new System.Windows.Forms.MenuItem();
+ this.tvCtxMenuComputerDefine = new System.Windows.Forms.MenuItem();
+ this.menuItem8 = new System.Windows.Forms.MenuItem();
+ this.tvCtxMenuProperties = new System.Windows.Forms.MenuItem();
+ this.imageTV = new System.Windows.Forms.ImageList(this.components);
+ this.ctxListViewMenu = new System.Windows.Forms.ContextMenu();
+ this.menuStartProcess = new System.Windows.Forms.MenuItem();
+ this.menuRestartProcess = new System.Windows.Forms.MenuItem();
+ this.menuStopProcess = new System.Windows.Forms.MenuItem();
+ this.menuRemoveProcess = new System.Windows.Forms.MenuItem();
+ this.menuRefresh = new System.Windows.Forms.MenuItem();
+ this.computerMenuAdd = new System.Windows.Forms.MenuItem();
+ this.menuItem3 = new System.Windows.Forms.MenuItem();
+ this.computerMenuRemove = new System.Windows.Forms.MenuItem();
+ this.menuItem5 = new System.Windows.Forms.MenuItem();
+ this.computerMenuDisconnect = new System.Windows.Forms.MenuItem();
+ this.computerMenuConnect = new System.Windows.Forms.MenuItem();
+ this.menuItem11 = new System.Windows.Forms.MenuItem();
+ this.computerMenuProperties = new System.Windows.Forms.MenuItem();
+ this.computerMenuRefresh = new System.Windows.Forms.MenuItem();
+ this.chComputer = new System.Windows.Forms.ColumnHeader();
+ this.chProcessName = new System.Windows.Forms.ColumnHeader();
+ this.mainMenuItem = new System.Windows.Forms.MenuItem();
+ this.chProcesses = new System.Windows.Forms.ColumnHeader();
+ this.mainMenu = new System.Windows.Forms.MainMenu();
+ this.mainMenuFile = new System.Windows.Forms.MenuItem();
+ this.menuItem2 = new System.Windows.Forms.MenuItem();
+ this.menuItem1 = new System.Windows.Forms.MenuItem();
+ this.mainMenuComputer = new System.Windows.Forms.MenuItem();
+ this.subMenuComputerAdd = new System.Windows.Forms.MenuItem();
+ this.menuItem7 = new System.Windows.Forms.MenuItem();
+ this.subMenuComputerDisconnect = new System.Windows.Forms.MenuItem();
+ this.subMenuComputerRemove = new System.Windows.Forms.MenuItem();
+ this.menuItem10 = new System.Windows.Forms.MenuItem();
+ this.subMenuComputerProperties = new System.Windows.Forms.MenuItem();
+ this.mainMenuDatabase = new System.Windows.Forms.MenuItem();
+ this.subMenuDatabaseCreate = new System.Windows.Forms.MenuItem();
+ this.panel1 = new System.Windows.Forms.Panel();
+ this.mgmConsole = new System.Windows.Forms.TextBox();
+ this.splitterHorizont = new System.Windows.Forms.Splitter();
+ this.splitter2 = new System.Windows.Forms.Splitter();
+ this.listView = new System.Windows.Forms.ListView();
+ this.chComputerIP = new System.Windows.Forms.ColumnHeader();
+ this.chStatus = new System.Windows.Forms.ColumnHeader();
+ this.chDatabase = new System.Windows.Forms.ColumnHeader();
+ this.chName = new System.Windows.Forms.ColumnHeader();
+ this.chOwner = new System.Windows.Forms.ColumnHeader();
+ this.splitterVertical = new System.Windows.Forms.Splitter();
+ this.openHostFileDialog = new System.Windows.Forms.OpenFileDialog();
+ this.saveHostFileDialog = new System.Windows.Forms.SaveFileDialog();
+ this.panel1.SuspendLayout();
+ this.SuspendLayout();
+ //
+ // tvComputerCluster
+ //
+ this.tvComputerCluster.CausesValidation = false;
+ this.tvComputerCluster.ContextMenu = this.ctxTreeViewMenu;
+ this.tvComputerCluster.Dock = System.Windows.Forms.DockStyle.Left;
+ this.tvComputerCluster.ImageList = this.imageTV;
+ this.tvComputerCluster.Name = "tvComputerCluster";
+ this.tvComputerCluster.Nodes.AddRange(new System.Windows.Forms.TreeNode[] {
+ new System.Windows.Forms.TreeNode("Computer", 0, 0),
+ new System.Windows.Forms.TreeNode("Database", 5, 5)});
+ this.tvComputerCluster.Size = new System.Drawing.Size(104, 333);
+ this.tvComputerCluster.TabIndex = 5;
+ this.tvComputerCluster.MouseDown += new System.Windows.Forms.MouseEventHandler(this.tvComputerCluster_MouseDown);
+ this.tvComputerCluster.AfterSelect += new System.Windows.Forms.TreeViewEventHandler(this.tvComputerCluster_AfterSelect);
+ this.tvComputerCluster.BeforeCollapse += new System.Windows.Forms.TreeViewCancelEventHandler(this.tvComputerCluster_BeforeCollapse);
+ this.tvComputerCluster.BeforeExpand += new System.Windows.Forms.TreeViewCancelEventHandler(this.tvComputerCluster_BeforeExpand);
+ //
+ // ctxTreeViewMenu
+ //
+ this.ctxTreeViewMenu.MenuItems.AddRange(new System.Windows.Forms.MenuItem[] {
+ this.tvCtxMenuComputerAdd,
+ this.tvCtxMenuComputerRemove,
+ this.menuGetStatus,
+ this.menuItem6,
+ this.tvCtxMenuComputerConnect,
+ this.tvCtxMenuComputerDisconnect,
+ this.tvCtxMenuDatabaseNew,
+ this.tvCtxMenuComputerDefine,
+ this.menuItem8,
+ this.tvCtxMenuProperties});
+ this.ctxTreeViewMenu.Popup += new System.EventHandler(this.ctxTreeViewMenu_Popup);
+ //
+ // tvCtxMenuComputerAdd
+ //
+ this.tvCtxMenuComputerAdd.Index = 0;
+ this.tvCtxMenuComputerAdd.Text = "Add computer";
+ this.tvCtxMenuComputerAdd.Click += new System.EventHandler(this.computerMenuAdd_Click);
+ //
+ // tvCtxMenuComputerRemove
+ //
+ this.tvCtxMenuComputerRemove.Index = 1;
+ this.tvCtxMenuComputerRemove.Text = "Remove computer";
+ this.tvCtxMenuComputerRemove.Click += new System.EventHandler(this.computerMenuRemove_Click);
+ //
+ // menuGetStatus
+ //
+ this.menuGetStatus.Index = 2;
+ this.menuGetStatus.Text = "Get Status";
+ this.menuGetStatus.Click += new System.EventHandler(this.menuGetStatus_Click);
+ //
+ // menuItem6
+ //
+ this.menuItem6.Index = 3;
+ this.menuItem6.Text = "-";
+ //
+ // tvCtxMenuComputerConnect
+ //
+ this.tvCtxMenuComputerConnect.Index = 4;
+ this.tvCtxMenuComputerConnect.Text = "Connect";
+ //
+ // tvCtxMenuComputerDisconnect
+ //
+ this.tvCtxMenuComputerDisconnect.Index = 5;
+ this.tvCtxMenuComputerDisconnect.Text = "Disconnect";
+ //
+ // tvCtxMenuDatabaseNew
+ //
+ this.tvCtxMenuDatabaseNew.Index = 6;
+ this.tvCtxMenuDatabaseNew.Text = "Create database...";
+ this.tvCtxMenuDatabaseNew.Click += new System.EventHandler(this.subMenuDatabaseCreate_Click);
+ //
+ // tvCtxMenuComputerDefine
+ //
+ this.tvCtxMenuComputerDefine.Index = 7;
+ this.tvCtxMenuComputerDefine.Text = "Define process...";
+ this.tvCtxMenuComputerDefine.Click += new System.EventHandler(this.tvCtxMenuComputerDefine_Click);
+ //
+ // menuItem8
+ //
+ this.menuItem8.Index = 8;
+ this.menuItem8.Text = "-";
+ //
+ // tvCtxMenuProperties
+ //
+ this.tvCtxMenuProperties.Index = 9;
+ this.tvCtxMenuProperties.Text = "Properties";
+ //
+ // imageTV
+ //
+ this.imageTV.ColorDepth = System.Windows.Forms.ColorDepth.Depth8Bit;
+ this.imageTV.ImageSize = new System.Drawing.Size(16, 16);
+ this.imageTV.ImageStream = ((System.Windows.Forms.ImageListStreamer)(resources.GetObject("imageTV.ImageStream")));
+ this.imageTV.TransparentColor = System.Drawing.Color.Transparent;
+ //
+ // ctxListViewMenu
+ //
+ this.ctxListViewMenu.MenuItems.AddRange(new System.Windows.Forms.MenuItem[] {
+ this.menuStartProcess,
+ this.menuRestartProcess,
+ this.menuStopProcess,
+ this.menuRemoveProcess,
+ this.menuRefresh});
+ this.ctxListViewMenu.Popup += new System.EventHandler(this.ctxListViewMenu_Popup);
+ //
+ // menuStartProcess
+ //
+ this.menuStartProcess.Index = 0;
+ this.menuStartProcess.Text = "Start process";
+ this.menuStartProcess.Click += new System.EventHandler(this.startProcess);
+ //
+ // menuRestartProcess
+ //
+ this.menuRestartProcess.Index = 1;
+ this.menuRestartProcess.Text = "Restart process";
+ this.menuRestartProcess.Click += new System.EventHandler(this.restartProcess);
+ //
+ // menuStopProcess
+ //
+ this.menuStopProcess.Index = 2;
+ this.menuStopProcess.Text = "Stop process";
+ this.menuStopProcess.Click += new System.EventHandler(this.stopProcess);
+ //
+ // menuRemoveProcess
+ //
+ this.menuRemoveProcess.Index = 3;
+ this.menuRemoveProcess.Text = "Remove process";
+ this.menuRemoveProcess.Click += new System.EventHandler(this.removeProcess);
+ //
+ // menuRefresh
+ //
+ this.menuRefresh.Index = 4;
+ this.menuRefresh.Text = "Refresh";
+ this.menuRefresh.Click += new System.EventHandler(this.menuRefresh_Click);
+ //
+ // computerMenuAdd
+ //
+ this.computerMenuAdd.Index = -1;
+ this.computerMenuAdd.Text = "Add";
+ this.computerMenuAdd.Click += new System.EventHandler(this.computerMenuAdd_Click);
+ //
+ // menuItem3
+ //
+ this.menuItem3.Index = -1;
+ this.menuItem3.Text = "-";
+ //
+ // computerMenuRemove
+ //
+ this.computerMenuRemove.Index = -1;
+ this.computerMenuRemove.Text = "Remove";
+ this.computerMenuRemove.Click += new System.EventHandler(this.computerMenuRemove_Click);
+ //
+ // menuItem5
+ //
+ this.menuItem5.Index = -1;
+ this.menuItem5.Text = "-";
+ //
+ // computerMenuDisconnect
+ //
+ this.computerMenuDisconnect.Index = -1;
+ this.computerMenuDisconnect.Text = "Disconnect";
+ //
+ // computerMenuConnect
+ //
+ this.computerMenuConnect.Index = -1;
+ this.computerMenuConnect.Text = "Connect";
+ //
+ // menuItem11
+ //
+ this.menuItem11.Index = -1;
+ this.menuItem11.Text = "-";
+ //
+ // computerMenuProperties
+ //
+ this.computerMenuProperties.Index = -1;
+ this.computerMenuProperties.Text = "Properties";
+ //
+ // computerMenuRefresh
+ //
+ this.computerMenuRefresh.Index = -1;
+ this.computerMenuRefresh.Text = "Refresh";
+ this.computerMenuRefresh.Click += new System.EventHandler(this.computerMenuRefresh_Click);
+ //
+ // chComputer
+ //
+ this.chComputer.Text = "Computer";
+ //
+ // chProcessName
+ //
+ this.chProcessName.Text = "Name";
+ //
+ // mainMenuItem
+ //
+ this.mainMenuItem.Index = -1;
+ this.mainMenuItem.Text = "File";
+ //
+ // chProcesses
+ //
+ this.chProcesses.Text = "Id";
+ //
+ // mainMenu
+ //
+ this.mainMenu.MenuItems.AddRange(new System.Windows.Forms.MenuItem[] {
+ this.mainMenuFile,
+ this.mainMenuComputer,
+ this.mainMenuDatabase});
+ //
+ // mainMenuFile
+ //
+ this.mainMenuFile.Index = 0;
+ this.mainMenuFile.MenuItems.AddRange(new System.Windows.Forms.MenuItem[] {
+ this.menuItem2,
+ this.menuItem1});
+ this.mainMenuFile.Text = "&File";
+ //
+ // menuItem2
+ //
+ this.menuItem2.Index = 0;
+ this.menuItem2.Text = "&Import...";
+ this.menuItem2.Click += new System.EventHandler(this.importHostFile);
+ //
+ // menuItem1
+ //
+ this.menuItem1.Index = 1;
+ this.menuItem1.Text = "&Export...";
+ this.menuItem1.Click += new System.EventHandler(this.exportHostFile);
+ //
+ // mainMenuComputer
+ //
+ this.mainMenuComputer.Index = 1;
+ this.mainMenuComputer.MenuItems.AddRange(new System.Windows.Forms.MenuItem[] {
+ this.subMenuComputerAdd,
+ this.menuItem7,
+ this.subMenuComputerDisconnect,
+ this.subMenuComputerRemove,
+ this.menuItem10,
+ this.subMenuComputerProperties});
+ this.mainMenuComputer.Text = "&Computer";
+ //
+ // subMenuComputerAdd
+ //
+ this.subMenuComputerAdd.Index = 0;
+ this.subMenuComputerAdd.Text = "&Add Computer";
+ this.subMenuComputerAdd.Click += new System.EventHandler(this.computerMenuAdd_Click);
+ //
+ // menuItem7
+ //
+ this.menuItem7.Index = 1;
+ this.menuItem7.Text = "-";
+ //
+ // subMenuComputerDisconnect
+ //
+ this.subMenuComputerDisconnect.Index = 2;
+ this.subMenuComputerDisconnect.Text = "&Disconnect";
+ //
+ // subMenuComputerRemove
+ //
+ this.subMenuComputerRemove.Index = 3;
+ this.subMenuComputerRemove.Text = "&Remove Computer";
+ this.subMenuComputerRemove.Click += new System.EventHandler(this.computerMenuRemove_Click);
+ //
+ // menuItem10
+ //
+ this.menuItem10.Index = 4;
+ this.menuItem10.Text = "-";
+ //
+ // subMenuComputerProperties
+ //
+ this.subMenuComputerProperties.Index = 5;
+ this.subMenuComputerProperties.Text = "&Properties";
+ //
+ // mainMenuDatabase
+ //
+ this.mainMenuDatabase.Index = 2;
+ this.mainMenuDatabase.MenuItems.AddRange(new System.Windows.Forms.MenuItem[] {
+ this.subMenuDatabaseCreate});
+ this.mainMenuDatabase.Text = "&Database";
+ this.mainMenuDatabase.Click += new System.EventHandler(this.subMenuDatabaseCreate_Click);
+ //
+ // subMenuDatabaseCreate
+ //
+ this.subMenuDatabaseCreate.Index = 0;
+ this.subMenuDatabaseCreate.Text = "&Create database...";
+ this.subMenuDatabaseCreate.Click += new System.EventHandler(this.subMenuDatabaseCreate_Click);
+ //
+ // panel1
+ //
+ this.panel1.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.mgmConsole,
+ this.splitterHorizont,
+ this.splitter2,
+ this.listView});
+ this.panel1.Dock = System.Windows.Forms.DockStyle.Fill;
+ this.panel1.Location = new System.Drawing.Point(104, 0);
+ this.panel1.Name = "panel1";
+ this.panel1.Size = new System.Drawing.Size(384, 333);
+ this.panel1.TabIndex = 6;
+ //
+ // mgmConsole
+ //
+ this.mgmConsole.AccessibleRole = System.Windows.Forms.AccessibleRole.StaticText;
+ this.mgmConsole.Dock = System.Windows.Forms.DockStyle.Bottom;
+ this.mgmConsole.Location = new System.Drawing.Point(0, 231);
+ this.mgmConsole.Multiline = true;
+ this.mgmConsole.Name = "mgmConsole";
+ this.mgmConsole.Size = new System.Drawing.Size(384, 96);
+ this.mgmConsole.TabIndex = 5;
+ this.mgmConsole.Text = "textBox1";
+ this.mgmConsole.TextChanged += new System.EventHandler(this.mgmConsole_TextChanged);
+ this.mgmConsole.Enter += new System.EventHandler(this.mgmConsole_Enter);
+ //
+ // splitterHorizont
+ //
+ this.splitterHorizont.Dock = System.Windows.Forms.DockStyle.Bottom;
+ this.splitterHorizont.Location = new System.Drawing.Point(0, 327);
+ this.splitterHorizont.MinExtra = 100;
+ this.splitterHorizont.MinSize = 100;
+ this.splitterHorizont.Name = "splitterHorizont";
+ this.splitterHorizont.Size = new System.Drawing.Size(384, 3);
+ this.splitterHorizont.TabIndex = 4;
+ this.splitterHorizont.TabStop = false;
+ //
+ // splitter2
+ //
+ this.splitter2.Dock = System.Windows.Forms.DockStyle.Bottom;
+ this.splitter2.Location = new System.Drawing.Point(0, 330);
+ this.splitter2.Name = "splitter2";
+ this.splitter2.Size = new System.Drawing.Size(384, 3);
+ this.splitter2.TabIndex = 2;
+ this.splitter2.TabStop = false;
+ //
+ // listView
+ //
+ this.listView.Columns.AddRange(new System.Windows.Forms.ColumnHeader[] {
+ this.chComputerIP,
+ this.chStatus,
+ this.chDatabase,
+ this.chName,
+ this.chOwner});
+ this.listView.ContextMenu = this.ctxListViewMenu;
+ this.listView.Dock = System.Windows.Forms.DockStyle.Fill;
+ this.listView.FullRowSelect = true;
+ this.listView.Name = "listView";
+ this.listView.Size = new System.Drawing.Size(384, 333);
+ this.listView.TabIndex = 0;
+ this.listView.View = System.Windows.Forms.View.Details;
+ this.listView.ColumnClick += new System.Windows.Forms.ColumnClickEventHandler(this.listView_ColumnClick_1);
+ this.listView.SelectedIndexChanged += new System.EventHandler(this.listView_SelectedIndexChanged);
+ //
+ // chComputerIP
+ //
+ this.chComputerIP.Text = "IP Adress";
+ //
+ // chStatus
+ //
+ this.chStatus.Text = "Status";
+ //
+ // chDatabase
+ //
+ this.chDatabase.Text = "Database";
+ //
+ // chName
+ //
+ this.chName.Text = "Name";
+ //
+ // chOwner
+ //
+ this.chOwner.Text = "Owner";
+ //
+ // splitterVertical
+ //
+ this.splitterVertical.Location = new System.Drawing.Point(104, 0);
+ this.splitterVertical.MinSize = 100;
+ this.splitterVertical.Name = "splitterVertical";
+ this.splitterVertical.Size = new System.Drawing.Size(3, 333);
+ this.splitterVertical.TabIndex = 7;
+ this.splitterVertical.TabStop = false;
+ this.splitterVertical.SplitterMoved += new System.Windows.Forms.SplitterEventHandler(this.splitterVertical_SplitterMoved);
+ //
+ // openHostFileDialog
+ //
+ this.openHostFileDialog.DefaultExt = "cpc";
+ this.openHostFileDialog.Filter = "CPCd configuration files (*.cpc)|*.cpc| All Files (*.*)|*.*";
+ this.openHostFileDialog.Title = "Import a CPCd configuration file";
+ this.openHostFileDialog.FileOk += new System.ComponentModel.CancelEventHandler(this.openHostFileDialog_FileOk);
+ //
+ // saveHostFileDialog
+ //
+ this.saveHostFileDialog.Filter = "CPCd configuration files (*.cpc)|*.cpc| All Files (*.*)|*.*";
+ this.saveHostFileDialog.Title = "Export a CPCd configuration file";
+ this.saveHostFileDialog.FileOk += new System.ComponentModel.CancelEventHandler(this.saveHostFileDialog_FileOk);
+ //
+ // CPC
+ //
+ this.AutoScaleBaseSize = new System.Drawing.Size(5, 13);
+ this.ClientSize = new System.Drawing.Size(488, 333);
+ this.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.splitterVertical,
+ this.panel1,
+ this.tvComputerCluster});
+ this.Menu = this.mainMenu;
+ this.Name = "CPC";
+ this.Text = "CPC";
+ this.Resize += new System.EventHandler(this.CPC_Resize);
+ this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.CPC_MouseDown);
+ this.Closing += new System.ComponentModel.CancelEventHandler(this.CPC_Closing);
+ this.Load += new System.EventHandler(this.CPC_Load);
+ this.Activated += new System.EventHandler(this.CPC_Activated);
+ this.Paint += new System.Windows.Forms.PaintEventHandler(this.CPC_Paint);
+ this.panel1.ResumeLayout(false);
+ this.ResumeLayout(false);
+
+ }
+ #endregion
+
+ /// <summary>
+ /// The main entry point for the application.
+ /// </summary>
+ [STAThread]
+ static void Main()
+ {
+ Application.Run(new CPC());
+
+ }
+
+ private void tvComputerCluster_AfterSelect(object sender, System.Windows.Forms.TreeViewEventArgs e)
+ {
+ if(e.Node.Text.ToString().Equals("Database"))
+ {
+ updateListViews("Database");
+
+ return;
+ }
+ if(e.Node.Text.ToString().Equals("Computer"))
+ {
+ //updateListViews();
+
+ updateListViews("Computer");
+ return;
+ }
+ if(e.Node.Parent.Text.ToString().Equals("Database"))
+ {
+ //updateListViews();
+ listView.Columns.Clear();
+ listView.Columns.Add(this.chName);
+ listView.Columns.Add(this.chDatabase);
+ listView.Columns.Add(this.chStatus);
+ listView.Columns.Add(this.chOwner);
+ updateDatabaseView(e.Node.Text.ToString());
+ }
+
+ if(e.Node.Parent.Text=="Computer")
+ {
+ //updateListViews();
+
+ Computer c=computerMgmt.getComputer(e.Node.Text.ToString());
+ string [] processcols= new string[5];
+ ArrayList processes;
+ processes = c.getProcesses();
+ listView.Items.Clear();
+ listView.Columns.Clear();
+ listView.Columns.Add(this.chComputer);
+ listView.Columns.Add(this.chDatabase);
+ listView.Columns.Add(this.chName);
+ listView.Columns.Add(this.chStatus);
+ listView.Columns.Add(this.chOwner);
+ if(processes != null )
+ {
+
+ listView.BeginUpdate();
+ foreach(Process p in processes)
+ {
+ processcols[0]=p.getComputer().getName();
+ processcols[1]=p.getDatabase();
+ processcols[2]=p.getName();
+ processcols[3]=p.getStatusString();
+ processcols[4]=p.getOwner();
+ ListViewItem lvp= new ListViewItem(processcols);
+ listView.Items.Add(lvp);
+ }
+
+ listView.EndUpdate();
+ }
+
+
+ listView.Show();
+ }
+
+ }
+
+
+
+ private void ctxTreeViewMenu_Popup(object sender, System.EventArgs e)
+ {
+ tvCtxMenuComputerAdd.Enabled=true;
+ tvCtxMenuComputerRemove.Enabled=true;
+ tvCtxMenuComputerConnect.Enabled=true;
+ tvCtxMenuComputerDisconnect.Enabled=true;
+ tvCtxMenuComputerDefine.Enabled=true;
+ menuGetStatus.Enabled=true;
+ tvCtxMenuDatabaseNew.Enabled=true;
+ tvCtxMenuComputerAdd.Visible=true;
+ tvCtxMenuComputerRemove.Visible=true;
+ tvCtxMenuComputerConnect.Visible=true;
+ tvCtxMenuComputerDisconnect.Visible=true;
+ tvCtxMenuComputerDefine.Visible=true;
+ tvCtxMenuDatabaseNew.Visible=true;
+ tvCtxMenuProperties.Visible=true;
+ menuGetStatus.Visible=true;
+
+ if(tvComputerCluster.SelectedNode.Text.Equals("Computer"))
+ {
+ tvCtxMenuComputerAdd.Enabled=true;
+ tvCtxMenuComputerRemove.Enabled=false;
+ tvCtxMenuComputerConnect.Enabled=false;
+ tvCtxMenuComputerDisconnect.Enabled=false;
+ tvCtxMenuComputerDefine.Enabled=false;
+ tvCtxMenuDatabaseNew.Visible=false;
+ menuGetStatus.Visible=false;
+ return;
+ }
+
+ if(tvComputerCluster.SelectedNode.Text.Equals("Database"))
+ {
+ // ctxTreeViewMenu.MenuItems.Add(menuDatabaseItem1);
+ tvCtxMenuComputerAdd.Visible=false;
+ tvCtxMenuComputerRemove.Visible=false;
+ tvCtxMenuComputerConnect.Visible=false;
+ tvCtxMenuComputerDisconnect.Visible=false;
+ tvCtxMenuComputerDefine.Visible=false;
+ tvCtxMenuDatabaseNew.Visible=true;
+ tvCtxMenuDatabaseNew.Enabled=true;
+ menuGetStatus.Visible=false;
+ menuItem6.Visible=false;
+ return;
+ }
+ if(tvComputerCluster.SelectedNode.Parent.Text.Equals("Computer"))
+ {
+
+ Computer c= computerMgmt.getComputer(tvComputerCluster.SelectedNode.Text.ToString());
+ if(c.getStatus().Equals(Computer.Status.Disconnected))
+ {
+ tvCtxMenuComputerConnect.Enabled=true;
+ tvCtxMenuComputerDisconnect.Enabled=false;
+ }
+ else
+ {
+ tvCtxMenuComputerDisconnect.Enabled=true;
+ tvCtxMenuComputerConnect.Enabled=false;
+ }
+
+ tvCtxMenuComputerAdd.Enabled=false;
+ tvCtxMenuComputerRemove.Enabled=true;
+ menuGetStatus.Visible=false;
+
+ tvCtxMenuComputerDefine.Enabled=true;
+ tvCtxMenuDatabaseNew.Visible=false;
+ return;
+ }
+
+ if(tvComputerCluster.SelectedNode.Parent.Text.Equals("Database"))
+ {
+ tvCtxMenuComputerAdd.Enabled=true;
+ tvCtxMenuComputerRemove.Enabled=false;
+ tvCtxMenuComputerConnect.Enabled=false;
+ tvCtxMenuComputerDisconnect.Enabled=false;
+ tvCtxMenuComputerDefine.Enabled=false;
+ tvCtxMenuDatabaseNew.Visible=true;
+ menuGetStatus.Visible=true;
+ return;
+ }
+
+
+ }
+
+
+ private void listView_SelectedIndexChanged(object sender, System.EventArgs e)
+ {
+ //MessageBox.Show(listView.SelectedItems[0].Text);
+ }
+
+
+ private void tvComputerCluster_MouseDown(object sender, System.Windows.Forms.MouseEventArgs e)
+ { /*
+ TreeNode node = tvComputerCluster.GetNodeAt(e.X,e.Y);
+ if(node==null)
+ {
+ return;
+ }
+ tvComputerCluster.SelectedNode=node;
+// updateListViews();
+ tvComputerCluster.SelectedNode.Expand();
+ */
+ }
+
+
+ private void subMenuComputerRemove_Click(object sender, System.EventArgs e)
+ {
+ //ComputerRemoveDialog crd=new ComputerRemoveDialog(computerMgmt);
+ //crd.Show();
+ //updateListViews();
+/* string computer = tvComputerCluster.SelectedNode.Text.ToString();
+ if(MessageBox.Show(this,"Are you sure you want to remove: " +computer+ "?","Remove computer",MessageBoxButtons.YesNo)==DialogResult.Yes)
+ {
+ computerMgmt.RemoveComputer(computer);
+ }
+*/
+ }
+
+ private void subMenuComputerAdd_Click(object sender, System.EventArgs e)
+ {
+ ComputerAddDialog cad=new ComputerAddDialog(computerMgmt);
+ cad.ShowDialog();
+ cad.Dispose();
+/// updateListViews(tvComputerCluster.SelectedNode.Text.ToString());
+ }
+
+
+
+ private void updateListViews(string node)
+ {
+ if(node.Equals("Computer"))
+ {
+ listView.Columns.Clear();
+ listView.Items.Clear();
+ ArrayList list= computerMgmt.getComputerCollection();
+ string [] computercols= new string[2];
+
+
+ listView.BeginUpdate();
+ listView.Columns.Add(this.chComputer);
+ listView.Columns.Add(this.chStatus);
+ foreach (Computer computer in list)
+ {
+ computercols[0]=computer.getName();
+ computercols[1]=computer.getStatusString();
+
+ ListViewItem lvc= new ListViewItem(computercols);
+
+ listView.Items.Add(lvc);
+
+ }
+ listView.EndUpdate();
+ listView.Show();
+ }
+
+ if(node.Equals("Database"))
+ {
+
+ ArrayList databases= computerMgmt.getDatabaseCollection();
+ string [] dbcols= new string[3];
+
+
+ listView.BeginUpdate();
+ listView.Items.Clear();
+ listView.Columns.Clear();
+ listView.Columns.Add(this.chDatabase);
+ listView.Columns.Add(this.chStatus);
+ listView.Columns.Add(this.chOwner);
+ foreach (Database db in databases)
+ {
+ dbcols[0]=db.getName();
+ dbcols[1]=db.getStatusString();
+ dbcols[2]=db.getOwner();
+
+ ListViewItem lvc= new ListViewItem(dbcols);
+
+ listView.Items.Add(lvc);
+
+ }
+ listView.EndUpdate();
+
+ listView.Show();
+ }
+
+ }
+
+ public void updateDatabaseView(string database)
+ {
+ Database d=computerMgmt.getDatabase(database);
+ string [] processcols= new string[5];
+ ArrayList processes = d.getProcesses();
+ listView.Items.Clear();
+ if(processes != null )
+ {
+
+ listView.BeginUpdate();
+ listView.Columns.Clear();
+ listView.Columns.Add(this.chComputer);
+ listView.Columns.Add(this.chDatabase);
+ listView.Columns.Add(this.chName);
+ listView.Columns.Add(this.chStatus);
+ listView.Columns.Add(this.chOwner);
+
+ foreach(Process p in processes)
+ {
+ processcols[0]=p.getComputer().getName();
+ processcols[1]=p.getDatabase();
+ processcols[2]=p.getName();
+ processcols[3]=p.getStatusString();
+ processcols[4]=p.getOwner();
+ ListViewItem lvp= new ListViewItem(processcols);
+ listView.Items.Add(lvp);
+ }
+
+ listView.EndUpdate();
+ }
+
+ listView.Show();
+ }
+
+ private void updateTreeViews()
+ {
+ //tvComputerCluster.Nodes.Clear();
+ ArrayList computers= computerMgmt.getComputerCollection();
+
+ ArrayList databases= computerMgmt.getDatabaseCollection();
+
+ tvComputerCluster.BeginUpdate();
+ tvComputerCluster.Nodes[0].Nodes.Clear();
+ tvComputerCluster.Nodes[1].Nodes.Clear();
+ if(computers != null)
+ {
+ foreach (Computer computer in computers)
+ {
+ tvComputerCluster.Nodes[0].Nodes.Add(new TreeNode(computer.getName().ToString()));
+ }
+ }
+ if(databases != null)
+ {
+ foreach (Database db in databases)
+ {
+ tvComputerCluster.Nodes[1].Nodes.Add(new TreeNode(db.getName().ToString()));
+ }
+ }
+
+ tvComputerCluster.EndUpdate();
+ }
+
+
+ private void CPC_MouseDown(object sender, System.Windows.Forms.MouseEventArgs e)
+ {
+ //updateListViews();
+ //updateTreeViews();
+
+ }
+
+ private void CPC_Paint(object sender, System.Windows.Forms.PaintEventArgs e)
+ {
+ if(tvComputerCluster.SelectedNode!=null)
+ {
+ if(tvComputerCluster.SelectedNode.Text.ToString().Equals("Computer"))
+ updateListViews("Computer");
+ }
+
+ //updateListViews();
+ //updateTreeViews();
+ }
+
+ private void CPC_Activated(object sender, System.EventArgs e)
+ {
+ updateListViews(tvComputerCluster.SelectedNode.Text.ToString());
+ //updateListViews();
+ updateTreeViews();
+ }
+
+
+ private void computerMenuAdd_Click(object sender, System.EventArgs e)
+ {
+ ComputerAddDialog cad=new ComputerAddDialog(computerMgmt);
+ cad.ShowDialog();
+ cad.Dispose();
+
+ }
+
+ private void computerMenuRemove_Click(object sender, System.EventArgs e)
+ {
+
+ string computer = tvComputerCluster.SelectedNode.Text.ToString();
+ if(MessageBox.Show("Are you sure you want to remove: " + computer +"?\n" + "This will remove all processes on the computer!" ,"Remove selected computer",MessageBoxButtons.YesNo, MessageBoxIcon.Question)== DialogResult.Yes)
+ {
+ removeComputer(computer);
+ }
+ }
+
+ private void removeComputer(string computer)
+ {
+ ArrayList processes;
+ Computer c=computerMgmt.getComputer(computer);
+ processes = c.getProcesses();
+
+ /*foreach(Process p in processes)
+ {
+ removeProcess(computer,p.getName());
+ processes=c.getProcesses();
+ }
+*/
+ if(computerMgmt.RemoveComputer(computer))
+ {
+ tvComputerCluster.SelectedNode=tvComputerCluster.SelectedNode.PrevVisibleNode;
+ this.updateTreeViews();
+ this.updateListViews("Computer");
+
+ if(tvComputerCluster.SelectedNode!=null)
+ this.updateListViews(tvComputerCluster.SelectedNode.Text.ToString());
+ //updateListViews();
+ }
+ }
+
+ private void listView_ColumnClick(object sender, System.Windows.Forms.ColumnClickEventArgs e)
+ {
+
+ if(listView.Sorting.Equals(SortOrder.Ascending))
+ listView.Sorting=SortOrder.Descending;
+ else
+ listView.Sorting=SortOrder.Ascending;
+
+ }
+
+
+ private void subMenuDatabaseCreate_Click(object sender, System.EventArgs e)
+ {
+ PanelWizard p = new PanelWizard(this.computerMgmt);
+ p.ShowDialog();
+ }
+
+ private void tvCtxMenuComputerDefine_Click(object sender, System.EventArgs e)
+ {
+ ProcessDefineDialog pdd = new ProcessDefineDialog(this.computerMgmt,
+ tvComputerCluster.SelectedNode.Text.ToString());
+ pdd.Show();
+ }
+
+ private void listView_ItemActivate(object sender, System.EventArgs e)
+ {
+ updateDatabaseView(listView.SelectedItems[0].Text.ToString());
+ for(int i=0;i<tvComputerCluster.Nodes[1].Nodes.Count;i++)
+ {
+ if(tvComputerCluster.Nodes[1].Nodes[i].Text.ToString().Equals(listView.SelectedItems[0].Text.ToString()))
+ {
+ tvComputerCluster.SelectedNode=tvComputerCluster.Nodes[1].Nodes[i];
+ break;
+ }
+ }
+
+
+ }
+
+ private void CPC_Resize(object sender, System.EventArgs e)
+ {
+ if(this.Width < 200) this.Width=200;
+ if(this.Height <200) this.Height=200;
+ this.tvComputerCluster.Width=(int)(this.Width*this.resizeWidthRatio);
+ this.listView.Height=(int)(this.Height*this.resizeHeightRatio);
+
+ //this.Size=new System.Drawing.Size((int)(this.Size.Width*this.tvComputerCluster.Width
+
+ }
+
+
+
+ private void updateGuiThread()
+ {
+ while(true) {
+ if(tvComputerCluster.SelectedNode!=null)
+ {
+ if(tvComputerCluster.SelectedNode.Text.ToString().Equals("Computer"))
+ updateListViews("Computer");
+ }
+ Thread.Sleep(1000);
+ }
+ }
+
+ private void computerMenuRefresh_Click(object sender, System.EventArgs e)
+ {
+ updateListViews("Computer");
+ }
+
+ private void CPC_Closing(object sender, System.ComponentModel.CancelEventArgs e)
+ {
+ /*clean up*/
+ ArrayList comp = this.computerMgmt.getComputerCollection();
+ foreach(Computer c in comp)
+ {
+ c.disconnect();
+ }
+ }
+
+ private void CPC_Load(object sender, System.EventArgs e)
+ {
+ this.tvComputerCluster.Width=104;
+ resizeWidthRatio =(float) ((float)(this.tvComputerCluster.Width)/(float)(this.Width));
+ resizeHeightRatio = (float) ((float)(this.listView.Height)/(float)(this.Height));
+ listView.Columns.Clear();
+ listView.Columns.Add(this.chComputer);
+ listView.Columns.Add(this.chStatus);
+
+ }
+
+ private void splitterVertical_SplitterMoved(object sender, System.Windows.Forms.SplitterEventArgs e)
+ {
+ if(this.Width < 500)
+ this.Width=500;
+ }
+
+ private void menuGetStatus_Click(object sender, System.EventArgs e)
+ {
+
+ }
+
+ private void tvComputerCluster_BeforeExpand(object sender, System.Windows.Forms.TreeViewCancelEventArgs e)
+ {
+ if (e.Node.Parent!=null && e.Node.Nodes.Count !=0)
+ e.Cancel=true;
+ if(e.Node.IsExpanded)
+ e.Cancel=true;
+ }
+
+ private void tvComputerCluster_BeforeCollapse(object sender, System.Windows.Forms.TreeViewCancelEventArgs e)
+ {
+ e.Cancel=true;
+ if (e.Node.Parent!=null && e.Node.Nodes.Count !=0)
+ e.Cancel=true;
+ if(e.Node.IsExpanded)
+ e.Cancel=false;
+ }
+
+
+
+ private void ctxListViewMenu_Popup(object sender, System.EventArgs e)
+ {
+
+ menuStartProcess.Visible=false;
+ menuStopProcess.Visible=false;
+ menuRestartProcess.Visible=false;
+ menuRemoveProcess.Visible=false;
+ menuRefresh.Visible=false;
+
+
+ if(this.tvComputerCluster.SelectedNode.Text.Equals("Computer"))
+ {
+ return;
+ }
+
+ if(this.tvComputerCluster.SelectedNode.Text.Equals("Database"))
+ {
+ return;
+ }
+
+ if(this.tvComputerCluster.SelectedNode.Parent.Text.Equals("Computer"))
+ {
+ if(listView.SelectedItems==null)
+ return;
+ menuRefresh.Visible=true;
+ }
+ if(this.tvComputerCluster.SelectedNode.Parent.Text.Equals("Database"))
+ {
+ if(listView.SelectedItems==null)
+ return;
+ menuStartProcess.Visible=true;
+ menuStopProcess.Visible=true;
+ menuRestartProcess.Visible=true;
+ menuRemoveProcess.Visible=true;
+ menuRefresh.Visible=true;
+ menuStopProcess.Enabled=true;
+ menuStartProcess.Enabled=true;
+ menuRestartProcess.Enabled=true;
+ menuRemoveProcess.Enabled=true;
+ menuRefresh.Enabled=true;
+ }
+
+
+ computerMenuRemove.Enabled=true;
+ computerMenuConnect.Enabled=true;
+ computerMenuDisconnect.Enabled=true;
+ computerMenuRefresh.Enabled=true;
+ string selectedItem="";
+ if(listView.SelectedItems.Count>0)
+ selectedItem=listView.FocusedItem.Text.ToString();
+
+
+ if(selectedItem.Equals(""))
+ {
+ computerMenuAdd.Enabled=true;
+ computerMenuRemove.Enabled=false;
+ computerMenuConnect.Enabled=false;
+ computerMenuDisconnect.Enabled=false;
+ return;
+ }
+ else
+ {
+ computerMenuAdd.Enabled=false;
+ if(computerMgmt.getStatus(selectedItem).Equals(Computer.Status.Connected))
+ {
+ computerMenuConnect.Enabled=false;
+ computerMenuRemove.Enabled=true;
+ }
+ if(computerMgmt.getStatus(selectedItem).Equals(Computer.Status.Disconnected))
+ computerMenuDisconnect.Enabled=false;
+ }
+
+
+ }
+
+ private void startProcess(object sender, System.EventArgs e)
+ {
+ if(listView.SelectedItems.Count==0)
+ return;
+
+ string computer = listView.SelectedItems[0].SubItems[0].Text.ToString();
+ string process = listView.SelectedItems[0].SubItems[2].Text.ToString();
+
+ if(computerMgmt.getComputer(computer).getProcessByName(process).getStatus()==Process.Status.Running)
+ {
+ MessageBox.Show(this,"The process is already started!" ,"Process failed to start",MessageBoxButtons.OK);
+ return;
+ }
+
+ int status = startProcess(listView.SelectedItems[0].SubItems[0].Text.ToString(),listView.SelectedItems[0].SubItems[2].Text.ToString());
+
+
+ if(status < 0)
+ MessageBox.Show(this,"Either the link is not OK, or the process is misconfigured! Status : " + status,"Process failed to start",MessageBoxButtons.OK);
+ else
+ MessageBox.Show(this,"The process was sucessfully started!","Process started",MessageBoxButtons.OK);
+
+ }
+
+ private int startProcess(string computer, string process)
+ {
+ Computer c=computerMgmt.getComputer(computer);
+ int status = c.startProcess(c.getProcessByName(process));
+ return status;
+ }
+
+ private void listView_ColumnClick_1(object sender, System.Windows.Forms.ColumnClickEventArgs e)
+ {
+ // if(listView.Columns[e.Column].Text.Equals("Computer"))
+ // {
+ if(listView.Sorting.Equals(SortOrder.Ascending))
+ {
+ listView.Sorting=SortOrder.Descending;
+ }
+ else
+ {
+ listView.Sorting=SortOrder.Ascending;
+ }
+ // }
+ }
+
+ private void removeProcess(object sender, System.EventArgs e)
+ {
+ if(listView.SelectedItems.Count==0)
+ return;
+ string process = listView.SelectedItems[0].SubItems[2].Text.ToString();
+ string computer = listView.SelectedItems[0].SubItems[0].Text.ToString();
+
+ if(MessageBox.Show("Are you sure that you want to remove " + process + " permanently?","Remove process",MessageBoxButtons.YesNo) == DialogResult.No)
+ return;
+ removeProcess(computer,process);
+ MessageBox.Show(this,"The process was sucessfully removed!","Remove process",MessageBoxButtons.OK);
+ }
+
+ private void removeProcess(string computer, string process)
+ {
+
+ Computer c=computerMgmt.getComputer(computer);
+ stopProcess(computer,process);
+ int status = c.undefineProcess(c.getProcessByName(process));
+ //if(status < 0)
+ // MessageBox.Show(this,"The process could not be removed!","Failed to remove process",MessageBoxButtons.OK);
+ // else
+ // {
+ Database db = computerMgmt.getDatabase((c.getProcessByName(process).getDatabase()));
+ db.removeProcess(process);
+ c.removeProcess(process,db.getName());
+ updateListViews("Database");
+ // }
+ }
+
+ private void stopProcess(object sender, System.EventArgs e)
+ {
+ if(listView.SelectedItems.Count==0)
+ return;
+ string computer = listView.SelectedItems[0].SubItems[0].Text.ToString();
+ string process = listView.SelectedItems[0].SubItems[2].Text.ToString();
+ if(computerMgmt.getComputer(computer).getProcessByName(process).getStatus()==Process.Status.Stopped)
+ {
+ MessageBox.Show(this,"The process is already stopped!" ,"Process failed to stop",MessageBoxButtons.OK);
+ return;
+ }
+
+ if(DialogResult.No==MessageBox.Show(this,"Are you sure you want to stop the " + process + " process?","Stop process!", MessageBoxButtons.YesNo))
+ return;
+
+ int status = stopProcess(computer, process);
+ if(status < 0)
+ MessageBox.Show(this,"The process could not be stopped. Status: " + status ,"Process failed to stop",MessageBoxButtons.OK);
+ else
+ MessageBox.Show(this,"The process was sucessfully stopped!","Process stopped",MessageBoxButtons.OK);
+ }
+
+ private int stopProcess(string computer, string process)
+ {
+ Computer c=computerMgmt.getComputer(computer);
+ int status = c.stopProcess(c.getProcessByName(process));
+ return status;
+ }
+
+ private void restartProcess(object sender, System.EventArgs e)
+ {
+ if(listView.SelectedItems.Count==0)
+ return;
+ string computer = listView.SelectedItems[0].SubItems[0].Text.ToString();
+ string process = listView.SelectedItems[0].SubItems[2].Text.ToString();
+ if(stopProcess(computer, process)<0)
+ {
+ MessageBox.Show("Restart process failed!!!", "Restart process");
+ return;
+ }
+ if(startProcess(computer, process)<0)
+ {
+ MessageBox.Show("Restart process failed!!!", "Restart process");
+ return;
+ }
+ MessageBox.Show("Succesfully restarted the process!","Restart process");
+ }
+
+ private void menuRefresh_Click(object sender, System.EventArgs e)
+ {
+ //string computer = tvComputerCluster.SelectedNode.Text;
+
+ this.listProcesses();
+ }
+
+ private void importHostFile(object sender, System.EventArgs e)
+ {
+ openHostFileDialog.ShowDialog();
+ }
+
+ private void exportHostFile(object sender, System.EventArgs e)
+ {
+ saveHostFileDialog.ShowDialog();
+ }
+
+ private void listProcesses()
+ {
+ /* add process in computer list*/
+ ArrayList computers = computerMgmt.getComputerCollection();
+ foreach(Computer c in computers)
+ {
+ c.listProcesses();
+ ArrayList processes = c.getProcesses();
+ if(processes!=null)
+ {
+ foreach(Process p in processes)
+ {
+ Database db = computerMgmt.getDatabase(p.getDatabase());
+ if(db!=null)
+ {
+ p.setDefined(true);
+ db.addProcessCheck(p);
+ }
+ }
+ }
+ }
+ updateListViews("Computer");
+ updateListViews("Database");
+ }
+
+ private void openHostFileDialog_FileOk(object sender, System.ComponentModel.CancelEventArgs e)
+ {
+ computerMgmt.importHostFile(openHostFileDialog.FileName);
+ this.updateTreeViews();
+ openHostFileDialog.Dispose();
+ listProcesses();
+ }
+
+ private void saveHostFileDialog_FileOk(object sender, System.ComponentModel.CancelEventArgs e)
+ {
+ computerMgmt.exportHostFile(saveHostFileDialog.FileName);
+ saveHostFileDialog.Dispose();
+ }
+
+ private void mgmConsole_Enter(object sender, System.EventArgs e)
+ {/*
+ //telnetclient.telnetClient tc= new telnetclient.telnetClient("10.0.13.1",10000,mgmConsole);
+ socketcomm.SocketComm sc = new socketcomm.SocketComm("10.0.13.1",10000);
+ sc.doConnect();
+ while(!sc.isConnected())
+ {
+ Thread.Sleep(100);
+ }
+ sc.writeMessage("get status\r");
+ string line = sc.readLine();
+ while(!line.Equals(""))
+ {
+ MessageBox.Show(line);
+ line=sc.readLine();
+ }
+*/
+ }
+
+ private void mgmConsole_TextChanged(object sender, System.EventArgs e)
+ {
+
+ }
+
+
+
+
+
+
+
+
+
+ }
+
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/Computer.cs b/ndb/src/cw/cpcc-win32/csharp/Computer.cs
new file mode 100644
index 00000000000..9763fac5622
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/Computer.cs
@@ -0,0 +1,256 @@
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+using System.Data;
+using System.IO;
+using NDB_CPC.socketcomm;
+using NDB_CPC.simpleparser;
+
+
+namespace NDB_CPC
+{
+ /// <summary>
+ /// Summary description for Computer.
+ /// </summary>
+ public class Computer
+ {
+ public enum Status {Disconnected=1,Connected=2, Unknown=3}
+ private string m_ip;
+ private int m_cpcdPort;
+ private string m_name;
+ private Status m_status;
+ private ArrayList m_processes;
+ private SocketComm m_socket;
+ public Computer(string name, int port)
+ {
+ m_name = name;
+ m_status = Status.Disconnected;
+ m_processes = new ArrayList();
+ m_cpcdPort=port;
+ m_socket = new SocketComm(m_name,m_cpcdPort);
+ }
+
+ public Computer(string name, string ip)
+ {
+ m_ip = ip;
+ m_name = name;
+ m_status = Status.Disconnected;
+ m_processes = new ArrayList();
+ m_cpcdPort=1234; //default port
+ m_socket = new SocketComm(m_ip,m_cpcdPort);
+ }
+
+ public void connectToCpcd()
+ {
+ m_socket.doConnect();
+ }
+
+ private bool sendMessage(string str)
+ {
+ return m_socket.writeMessage(str);
+
+ }
+
+ public string getName() {return m_name;}
+ public string getIp() {return m_ip;}
+ public ArrayList getProcesses()
+ {
+ if(m_processes.Count>0)
+ return m_processes;
+ else
+ return null;
+ }
+ public string getStatusString()
+ {
+ try
+ {
+ if(m_socket.isConnected())
+ return "Connected";
+ else
+ return "Disconnected";
+ }
+ catch(Exception e)
+ {
+ return "Unknown";
+ }
+ }
+
+
+ public bool isConnected()
+ {
+ if(m_socket.isConnected())
+ return true;
+ return false;
+ }
+
+ public Status getStatus()
+ {
+ try
+ {
+ if(m_socket.isConnected())
+ return Status.Connected;
+ else
+ return Status.Disconnected;
+ }
+ catch(Exception e)
+ {
+ return Status.Unknown;
+ }
+ }
+
+ public void setStatus(Status status)
+ {
+ m_status=status;
+ }
+
+ public void addProcess(Process process)
+ {
+ m_processes.Add(process);
+ }
+
+ public Process getProcessByName(string name)
+ {
+ foreach(Process process in m_processes)
+ {
+ if(process.getName().Equals(name))
+ return process;
+ }
+ return null;
+ }
+
+
+ public bool removeProcess(string name, string database)
+ {
+ foreach(Process p in m_processes)
+ {
+ if(p.getName().Equals(name) && p.getDatabase().Equals(database))
+ {
+ m_processes.Remove(p);
+ return true;
+ }
+ }
+ return false;
+ }
+
+ public void disconnect()
+ {
+ m_socket.disconnect();
+ }
+ public Process getProcess(string id)
+ {
+ foreach(Process process in m_processes)
+ {
+ if(process.getId().Equals(id))
+ return process;
+ }
+ return null;
+ }
+
+ public int listProcesses()
+ {
+ string list = "list processes\n\n";
+
+ if(!sendMessage(list))
+ return -2;
+
+ SimpleCPCParser.parse(m_processes, this, m_socket);
+ return 1;
+ }
+
+ public int defineProcess(Process p)
+ {
+ string define = "define process \n";
+ define = define + "name:" + p.getName() + "\n";
+ define = define + "group:" + p.getDatabase() + "\n";
+ define = define + "env:" + "NDB_CONNECTSTRING="+p.getConnectString() ;
+ if(p.getEnv().Equals(""))
+ define = define + "\n";
+ else
+ define = define + " " + p.getEnv() + "\n";
+
+ //if(p.getPath().EndsWith("\\"))
+ // define = define + "path:" + p.getPath()+ "ndb" + "\n";
+ //else
+ define = define + "path:" + p.getPath() + "\n";
+ define = define + "args:" + p.getArgs() + "\n";
+ define = define + "type:" + "permanent" + "\n";
+ define = define + "cwd:" + p.getCwd() + "\n";
+ define = define + "owner:" + "ejohson" + "\n\n";
+
+ if(!sendMessage(define))
+ return -2;
+
+ SimpleCPCParser.parse(p, m_socket);
+ if(p.isDefined())
+ return 1;
+ else
+ return -1;
+
+ }
+
+ public int startProcess(Process p)
+ {
+ if(!p.isDefined())
+ {
+ this.defineProcess(p);
+ if(!p.isDefined())
+ return -4; //process misconfigured
+
+ }
+ string start= "start process \n";
+ start = start + "id:" + p.getId() + "\n\n";
+ if(!sendMessage(start))
+ return -2;
+ SimpleCPCParser.parse(p, m_socket);
+ if(p.getStatus().Equals(Process.Status.Running))
+ return 1;
+ else
+ return -1;
+ }
+
+ public int stopProcess(Process p)
+ {
+ if(!p.isDefined())
+ {
+ return -4; //process not defined
+ }
+ string stop= "stop process \n";
+ stop = stop + "id:" + p.getId() + "\n\n";
+ if(!sendMessage(stop))
+ return -2;
+ SimpleCPCParser.parse(p, m_socket);
+
+ if(p.getStatus().Equals(Process.Status.Stopped))
+ return 1;
+ else
+ return -1;
+ }
+
+ public int undefineProcess(Process p)
+ {
+ if(!p.isDefined())
+ {
+ return -4; //process not defined
+ }
+ string undefine= "undefine process \n";
+ undefine = undefine + "id:" + p.getId() + "\n\n";
+ if(!sendMessage(undefine))
+ return -2;
+ SimpleCPCParser.parse(p, m_socket);
+ if(!p.isDefined())
+ {
+ return 1;
+
+ }
+ return -1;
+ }
+
+ public int getCpcdPort()
+ {
+ return this.m_cpcdPort;
+ }
+
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/ComputerAddDialog.cs b/ndb/src/cw/cpcc-win32/csharp/ComputerAddDialog.cs
new file mode 100644
index 00000000000..c01e41f3e60
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/ComputerAddDialog.cs
@@ -0,0 +1,242 @@
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+
+namespace NDB_CPC
+{
+ /// <summary>
+ /// Summary description for ComputerAddDialog.
+ /// </summary>
+ public class ComputerAddDialog : System.Windows.Forms.Form
+ {
+ private System.Windows.Forms.Label label1;
+ private System.Windows.Forms.TextBox textboxComputerName;
+ private System.Windows.Forms.Button btnAdd;
+ private System.Windows.Forms.Button btnCancel;
+ private System.Windows.Forms.Label label2;
+ /// <summary>
+ /// Required designer variable.
+ /// </summary>
+ private System.ComponentModel.Container components = null;
+ private System.Windows.Forms.Label label6;
+ private System.Windows.Forms.CheckBox checkBoxDefault;
+ private System.Windows.Forms.TextBox textBoxPort;
+
+ private ComputerMgmt mgmt;
+ public ComputerAddDialog(ComputerMgmt mgmt)
+ {
+ //
+ // Required for Windows Form Designer support
+ //
+ InitializeComponent();
+
+ //
+ // TODO: Add any constructor code after InitializeComponent call
+ //
+ this.mgmt=mgmt;
+ }
+ /// <summary>
+ /// Clean up any resources being used.
+ /// </summary>
+ protected override void Dispose( bool disposing )
+ {
+ if( disposing )
+ {
+ if(components != null)
+ {
+ components.Dispose();
+ }
+ }
+ base.Dispose( disposing );
+ }
+
+ #region Windows Form Designer generated code
+ /// <summary>
+ /// Required method for Designer support - do not modify
+ /// the contents of this method with the code editor.
+ /// </summary>
+ private void InitializeComponent()
+ {
+ this.textboxComputerName = new System.Windows.Forms.TextBox();
+ this.label1 = new System.Windows.Forms.Label();
+ this.btnAdd = new System.Windows.Forms.Button();
+ this.btnCancel = new System.Windows.Forms.Button();
+ this.label2 = new System.Windows.Forms.Label();
+ this.label6 = new System.Windows.Forms.Label();
+ this.textBoxPort = new System.Windows.Forms.TextBox();
+ this.checkBoxDefault = new System.Windows.Forms.CheckBox();
+ this.SuspendLayout();
+ //
+ // textboxComputerName
+ //
+ this.textboxComputerName.Location = new System.Drawing.Point(128, 16);
+ this.textboxComputerName.Name = "textboxComputerName";
+ this.textboxComputerName.Size = new System.Drawing.Size(136, 20);
+ this.textboxComputerName.TabIndex = 0;
+ this.textboxComputerName.Text = "";
+ //
+ // label1
+ //
+ this.label1.Location = new System.Drawing.Point(40, 16);
+ this.label1.Name = "label1";
+ this.label1.Size = new System.Drawing.Size(88, 23);
+ this.label1.TabIndex = 1;
+ this.label1.Text = "Computer name:";
+ this.label1.TextAlign = System.Drawing.ContentAlignment.MiddleRight;
+ //
+ // btnAdd
+ //
+ this.btnAdd.Location = new System.Drawing.Point(112, 128);
+ this.btnAdd.Name = "btnAdd";
+ this.btnAdd.Size = new System.Drawing.Size(80, 24);
+ this.btnAdd.TabIndex = 4;
+ this.btnAdd.Text = "Add";
+ this.btnAdd.Click += new System.EventHandler(this.btnAdd_Click);
+ //
+ // btnCancel
+ //
+ this.btnCancel.DialogResult = System.Windows.Forms.DialogResult.Cancel;
+ this.btnCancel.Location = new System.Drawing.Point(200, 128);
+ this.btnCancel.Name = "btnCancel";
+ this.btnCancel.Size = new System.Drawing.Size(80, 24);
+ this.btnCancel.TabIndex = 5;
+ this.btnCancel.Text = "Cancel";
+ this.btnCancel.Click += new System.EventHandler(this.btnCancel_Click);
+ //
+ // label2
+ //
+ this.label2.Location = new System.Drawing.Point(128, 40);
+ this.label2.Name = "label2";
+ this.label2.Size = new System.Drawing.Size(136, 16);
+ this.label2.TabIndex = 4;
+ this.label2.Text = "(e.g. Ndb01 or 10.0.1.1)";
+ //
+ // label6
+ //
+ this.label6.Location = new System.Drawing.Point(48, 64);
+ this.label6.Name = "label6";
+ this.label6.Size = new System.Drawing.Size(80, 24);
+ this.label6.TabIndex = 9;
+ this.label6.Text = "CPCd port:";
+ this.label6.TextAlign = System.Drawing.ContentAlignment.MiddleRight;
+ //
+ // textBoxPort
+ //
+ this.textBoxPort.Enabled = false;
+ this.textBoxPort.Location = new System.Drawing.Point(128, 64);
+ this.textBoxPort.Name = "textBoxPort";
+ this.textBoxPort.Size = new System.Drawing.Size(136, 20);
+ this.textBoxPort.TabIndex = 2;
+ this.textBoxPort.TabStop = false;
+ this.textBoxPort.Text = "";
+ //
+ // checkBoxDefault
+ //
+ this.checkBoxDefault.Checked = true;
+ this.checkBoxDefault.CheckState = System.Windows.Forms.CheckState.Checked;
+ this.checkBoxDefault.Location = new System.Drawing.Point(96, 96);
+ this.checkBoxDefault.Name = "checkBoxDefault";
+ this.checkBoxDefault.Size = new System.Drawing.Size(168, 16);
+ this.checkBoxDefault.TabIndex = 3;
+ this.checkBoxDefault.Text = "Use default port (1234)?";
+ this.checkBoxDefault.CheckedChanged += new System.EventHandler(this.checkBoxDefault_CheckedChanged);
+ //
+ // ComputerAddDialog
+ //
+ this.AcceptButton = this.btnAdd;
+ this.AutoScaleBaseSize = new System.Drawing.Size(5, 13);
+ this.CancelButton = this.btnCancel;
+ this.ClientSize = new System.Drawing.Size(298, 159);
+ this.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.checkBoxDefault,
+ this.label6,
+ this.textBoxPort,
+ this.label2,
+ this.btnCancel,
+ this.btnAdd,
+ this.label1,
+ this.textboxComputerName});
+ this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.FixedDialog;
+ this.MaximizeBox = false;
+ this.MinimizeBox = false;
+ this.Name = "ComputerAddDialog";
+ this.StartPosition = System.Windows.Forms.FormStartPosition.CenterParent;
+ this.Text = "Add a computer";
+ this.Load += new System.EventHandler(this.ComputerAddDialog_Load);
+ this.ResumeLayout(false);
+
+ }
+ #endregion
+
+ private void btnCancel_Click(object sender, System.EventArgs e)
+ {
+ this.Close();
+ this.Dispose();
+ }
+
+ private void btnAdd_Click(object sender, System.EventArgs e)
+ {
+ int port;
+ if(this.textboxComputerName.Text.Equals(""))
+ {
+ MessageBox.Show(this,"A computer must have an IP address or a host name","Warning!",MessageBoxButtons.OK);
+ return;
+ }
+ if(this.checkBoxDefault.Checked)
+ {
+ port=1234;
+ }
+ else
+ {
+ if(this.textBoxPort.Text.Equals(""))
+ {
+ MessageBox.Show(this,"You must specify a port number!!!","Warning!",MessageBoxButtons.OK);
+ return;
+ }
+ else
+ {
+ try
+ {
+ port=Convert.ToInt32(this.textBoxPort.Text.ToString());
+
+ }
+ catch (Exception exception)
+ {
+ MessageBox.Show(this,"Port number must be numeric!!!","Warning!",MessageBoxButtons.OK);
+ return;
+ }
+ }
+ }
+
+ if(mgmt.getComputer(this.textboxComputerName.Text)==null)
+ {
+ mgmt.AddComputer(this.textboxComputerName.Text.ToString(),port);}
+ else
+ {
+ MessageBox.Show("This computer does already exist!", "Add computer");
+ return;
+ }
+
+ this.Dispose();
+ }
+
+ private void ComputerAddDialog_Load(object sender, System.EventArgs e)
+ {
+
+ }
+
+ private void checkBoxDefault_CheckedChanged(object sender, System.EventArgs e)
+ {
+ if(checkBoxDefault.Checked)
+ textBoxPort.Enabled=false;
+ else
+ textBoxPort.Enabled=true;
+ }
+
+
+
+
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/ComputerRemoveDialog.cs b/ndb/src/cw/cpcc-win32/csharp/ComputerRemoveDialog.cs
new file mode 100644
index 00000000000..5b4d1b56df7
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/ComputerRemoveDialog.cs
@@ -0,0 +1,228 @@
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+
+namespace NDB_CPC
+{
+ /// <summary>
+ /// Summary description for ComputerRemoveDialog.
+ /// </summary>
+ public class ComputerRemoveDialog : System.Windows.Forms.Form
+ {
+ private System.Windows.Forms.Label label1;
+ private System.Windows.Forms.ComboBox comboComputer;
+ private System.Windows.Forms.Button btnCancel;
+ private System.Windows.Forms.Button btnRemove;
+ /// <summary>
+ /// Required designer variable.
+ /// </summary>
+ private System.ComponentModel.Container components = null;
+ private ComputerMgmt mgmt;
+
+ public ComputerRemoveDialog(ComputerMgmt mgmt)
+ {
+ //
+ // Required for Windows Form Designer support
+ //
+ InitializeComponent();
+
+ //
+ // TODO: Add any constructor code after InitializeComponent call
+ //
+ this.mgmt=mgmt;
+ }
+
+ /// <summary>
+ /// Clean up any resources being used.
+ /// </summary>
+ protected override void Dispose( bool disposing )
+ {
+ if( disposing )
+ {
+ if(components != null)
+ {
+ components.Dispose();
+ }
+ }
+ base.Dispose( disposing );
+ }
+
+ #region Windows Form Designer generated code
+ /// <summary>
+ /// Required method for Designer support - do not modify
+ /// the contents of this method with the code editor.
+ /// </summary>
+ private void InitializeComponent()
+ {
+ System.Resources.ResourceManager resources = new System.Resources.ResourceManager(typeof(ComputerRemoveDialog));
+ this.label1 = new System.Windows.Forms.Label();
+ this.comboComputer = new System.Windows.Forms.ComboBox();
+ this.btnCancel = new System.Windows.Forms.Button();
+ this.btnRemove = new System.Windows.Forms.Button();
+ this.SuspendLayout();
+ //
+ // label1
+ //
+ this.label1.AccessibleDescription = ((string)(resources.GetObject("label1.AccessibleDescription")));
+ this.label1.AccessibleName = ((string)(resources.GetObject("label1.AccessibleName")));
+ this.label1.Anchor = ((System.Windows.Forms.AnchorStyles)(resources.GetObject("label1.Anchor")));
+ this.label1.AutoSize = ((bool)(resources.GetObject("label1.AutoSize")));
+ this.label1.Dock = ((System.Windows.Forms.DockStyle)(resources.GetObject("label1.Dock")));
+ this.label1.Enabled = ((bool)(resources.GetObject("label1.Enabled")));
+ this.label1.Font = ((System.Drawing.Font)(resources.GetObject("label1.Font")));
+ this.label1.Image = ((System.Drawing.Image)(resources.GetObject("label1.Image")));
+ this.label1.ImageAlign = ((System.Drawing.ContentAlignment)(resources.GetObject("label1.ImageAlign")));
+ this.label1.ImageIndex = ((int)(resources.GetObject("label1.ImageIndex")));
+ this.label1.ImeMode = ((System.Windows.Forms.ImeMode)(resources.GetObject("label1.ImeMode")));
+ this.label1.Location = ((System.Drawing.Point)(resources.GetObject("label1.Location")));
+ this.label1.Name = "label1";
+ this.label1.RightToLeft = ((System.Windows.Forms.RightToLeft)(resources.GetObject("label1.RightToLeft")));
+ this.label1.Size = ((System.Drawing.Size)(resources.GetObject("label1.Size")));
+ this.label1.TabIndex = ((int)(resources.GetObject("label1.TabIndex")));
+ this.label1.Text = resources.GetString("label1.Text");
+ this.label1.TextAlign = ((System.Drawing.ContentAlignment)(resources.GetObject("label1.TextAlign")));
+ this.label1.Visible = ((bool)(resources.GetObject("label1.Visible")));
+ //
+ // comboComputer
+ //
+ this.comboComputer.AccessibleDescription = ((string)(resources.GetObject("comboComputer.AccessibleDescription")));
+ this.comboComputer.AccessibleName = ((string)(resources.GetObject("comboComputer.AccessibleName")));
+ this.comboComputer.Anchor = ((System.Windows.Forms.AnchorStyles)(resources.GetObject("comboComputer.Anchor")));
+ this.comboComputer.BackgroundImage = ((System.Drawing.Image)(resources.GetObject("comboComputer.BackgroundImage")));
+ this.comboComputer.Dock = ((System.Windows.Forms.DockStyle)(resources.GetObject("comboComputer.Dock")));
+ this.comboComputer.Enabled = ((bool)(resources.GetObject("comboComputer.Enabled")));
+ this.comboComputer.Font = ((System.Drawing.Font)(resources.GetObject("comboComputer.Font")));
+ this.comboComputer.ImeMode = ((System.Windows.Forms.ImeMode)(resources.GetObject("comboComputer.ImeMode")));
+ this.comboComputer.IntegralHeight = ((bool)(resources.GetObject("comboComputer.IntegralHeight")));
+ this.comboComputer.ItemHeight = ((int)(resources.GetObject("comboComputer.ItemHeight")));
+ this.comboComputer.Location = ((System.Drawing.Point)(resources.GetObject("comboComputer.Location")));
+ this.comboComputer.MaxDropDownItems = ((int)(resources.GetObject("comboComputer.MaxDropDownItems")));
+ this.comboComputer.MaxLength = ((int)(resources.GetObject("comboComputer.MaxLength")));
+ this.comboComputer.Name = "comboComputer";
+ this.comboComputer.RightToLeft = ((System.Windows.Forms.RightToLeft)(resources.GetObject("comboComputer.RightToLeft")));
+ this.comboComputer.Size = ((System.Drawing.Size)(resources.GetObject("comboComputer.Size")));
+ this.comboComputer.Sorted = true;
+ this.comboComputer.TabIndex = ((int)(resources.GetObject("comboComputer.TabIndex")));
+ this.comboComputer.Text = resources.GetString("comboComputer.Text");
+ this.comboComputer.Visible = ((bool)(resources.GetObject("comboComputer.Visible")));
+ this.comboComputer.SelectedIndexChanged += new System.EventHandler(this.comboComputer_SelectedIndexChanged);
+ //
+ // btnCancel
+ //
+ this.btnCancel.AccessibleDescription = ((string)(resources.GetObject("btnCancel.AccessibleDescription")));
+ this.btnCancel.AccessibleName = ((string)(resources.GetObject("btnCancel.AccessibleName")));
+ this.btnCancel.Anchor = ((System.Windows.Forms.AnchorStyles)(resources.GetObject("btnCancel.Anchor")));
+ this.btnCancel.BackgroundImage = ((System.Drawing.Image)(resources.GetObject("btnCancel.BackgroundImage")));
+ this.btnCancel.DialogResult = System.Windows.Forms.DialogResult.Cancel;
+ this.btnCancel.Dock = ((System.Windows.Forms.DockStyle)(resources.GetObject("btnCancel.Dock")));
+ this.btnCancel.Enabled = ((bool)(resources.GetObject("btnCancel.Enabled")));
+ this.btnCancel.FlatStyle = ((System.Windows.Forms.FlatStyle)(resources.GetObject("btnCancel.FlatStyle")));
+ this.btnCancel.Font = ((System.Drawing.Font)(resources.GetObject("btnCancel.Font")));
+ this.btnCancel.Image = ((System.Drawing.Image)(resources.GetObject("btnCancel.Image")));
+ this.btnCancel.ImageAlign = ((System.Drawing.ContentAlignment)(resources.GetObject("btnCancel.ImageAlign")));
+ this.btnCancel.ImageIndex = ((int)(resources.GetObject("btnCancel.ImageIndex")));
+ this.btnCancel.ImeMode = ((System.Windows.Forms.ImeMode)(resources.GetObject("btnCancel.ImeMode")));
+ this.btnCancel.Location = ((System.Drawing.Point)(resources.GetObject("btnCancel.Location")));
+ this.btnCancel.Name = "btnCancel";
+ this.btnCancel.RightToLeft = ((System.Windows.Forms.RightToLeft)(resources.GetObject("btnCancel.RightToLeft")));
+ this.btnCancel.Size = ((System.Drawing.Size)(resources.GetObject("btnCancel.Size")));
+ this.btnCancel.TabIndex = ((int)(resources.GetObject("btnCancel.TabIndex")));
+ this.btnCancel.Text = resources.GetString("btnCancel.Text");
+ this.btnCancel.TextAlign = ((System.Drawing.ContentAlignment)(resources.GetObject("btnCancel.TextAlign")));
+ this.btnCancel.Visible = ((bool)(resources.GetObject("btnCancel.Visible")));
+ this.btnCancel.Click += new System.EventHandler(this.btnCancel_Click);
+ //
+ // btnRemove
+ //
+ this.btnRemove.AccessibleDescription = ((string)(resources.GetObject("btnRemove.AccessibleDescription")));
+ this.btnRemove.AccessibleName = ((string)(resources.GetObject("btnRemove.AccessibleName")));
+ this.btnRemove.Anchor = ((System.Windows.Forms.AnchorStyles)(resources.GetObject("btnRemove.Anchor")));
+ this.btnRemove.BackgroundImage = ((System.Drawing.Image)(resources.GetObject("btnRemove.BackgroundImage")));
+ this.btnRemove.Dock = ((System.Windows.Forms.DockStyle)(resources.GetObject("btnRemove.Dock")));
+ this.btnRemove.Enabled = ((bool)(resources.GetObject("btnRemove.Enabled")));
+ this.btnRemove.FlatStyle = ((System.Windows.Forms.FlatStyle)(resources.GetObject("btnRemove.FlatStyle")));
+ this.btnRemove.Font = ((System.Drawing.Font)(resources.GetObject("btnRemove.Font")));
+ this.btnRemove.Image = ((System.Drawing.Image)(resources.GetObject("btnRemove.Image")));
+ this.btnRemove.ImageAlign = ((System.Drawing.ContentAlignment)(resources.GetObject("btnRemove.ImageAlign")));
+ this.btnRemove.ImageIndex = ((int)(resources.GetObject("btnRemove.ImageIndex")));
+ this.btnRemove.ImeMode = ((System.Windows.Forms.ImeMode)(resources.GetObject("btnRemove.ImeMode")));
+ this.btnRemove.Location = ((System.Drawing.Point)(resources.GetObject("btnRemove.Location")));
+ this.btnRemove.Name = "btnRemove";
+ this.btnRemove.RightToLeft = ((System.Windows.Forms.RightToLeft)(resources.GetObject("btnRemove.RightToLeft")));
+ this.btnRemove.Size = ((System.Drawing.Size)(resources.GetObject("btnRemove.Size")));
+ this.btnRemove.TabIndex = ((int)(resources.GetObject("btnRemove.TabIndex")));
+ this.btnRemove.Text = resources.GetString("btnRemove.Text");
+ this.btnRemove.TextAlign = ((System.Drawing.ContentAlignment)(resources.GetObject("btnRemove.TextAlign")));
+ this.btnRemove.Visible = ((bool)(resources.GetObject("btnRemove.Visible")));
+ this.btnRemove.Click += new System.EventHandler(this.btnRemove_Click);
+ //
+ // ComputerRemoveDialog
+ //
+ this.AcceptButton = this.btnRemove;
+ this.AccessibleDescription = ((string)(resources.GetObject("$this.AccessibleDescription")));
+ this.AccessibleName = ((string)(resources.GetObject("$this.AccessibleName")));
+ this.Anchor = ((System.Windows.Forms.AnchorStyles)(resources.GetObject("$this.Anchor")));
+ this.AutoScaleBaseSize = ((System.Drawing.Size)(resources.GetObject("$this.AutoScaleBaseSize")));
+ this.AutoScroll = ((bool)(resources.GetObject("$this.AutoScroll")));
+ this.AutoScrollMargin = ((System.Drawing.Size)(resources.GetObject("$this.AutoScrollMargin")));
+ this.AutoScrollMinSize = ((System.Drawing.Size)(resources.GetObject("$this.AutoScrollMinSize")));
+ this.BackgroundImage = ((System.Drawing.Image)(resources.GetObject("$this.BackgroundImage")));
+ this.CancelButton = this.btnCancel;
+ this.ClientSize = ((System.Drawing.Size)(resources.GetObject("$this.ClientSize")));
+ this.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.btnRemove,
+ this.btnCancel,
+ this.comboComputer,
+ this.label1});
+ this.Dock = ((System.Windows.Forms.DockStyle)(resources.GetObject("$this.Dock")));
+ this.Enabled = ((bool)(resources.GetObject("$this.Enabled")));
+ this.Font = ((System.Drawing.Font)(resources.GetObject("$this.Font")));
+ this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.FixedDialog;
+ this.Icon = ((System.Drawing.Icon)(resources.GetObject("$this.Icon")));
+ this.ImeMode = ((System.Windows.Forms.ImeMode)(resources.GetObject("$this.ImeMode")));
+ this.Location = ((System.Drawing.Point)(resources.GetObject("$this.Location")));
+ this.MaximizeBox = false;
+ this.MaximumSize = ((System.Drawing.Size)(resources.GetObject("$this.MaximumSize")));
+ this.MinimizeBox = false;
+ this.MinimumSize = ((System.Drawing.Size)(resources.GetObject("$this.MinimumSize")));
+ this.Name = "ComputerRemoveDialog";
+ this.RightToLeft = ((System.Windows.Forms.RightToLeft)(resources.GetObject("$this.RightToLeft")));
+ this.StartPosition = ((System.Windows.Forms.FormStartPosition)(resources.GetObject("$this.StartPosition")));
+ this.Text = resources.GetString("$this.Text");
+ this.Visible = ((bool)(resources.GetObject("$this.Visible")));
+ this.Load += new System.EventHandler(this.ComputerRemoveDialog_Load);
+ this.ResumeLayout(false);
+
+ }
+ #endregion
+
+ private void btnRemove_Click(object sender, System.EventArgs e)
+ {
+ mgmt.RemoveComputer(comboComputer.SelectedItem.ToString());
+ this.Dispose();
+ }
+
+ private void ComputerRemoveDialog_Load(object sender, System.EventArgs e)
+ {
+ ArrayList list = mgmt.getComputerCollection();
+ foreach (Computer computer in list)
+ {
+ comboComputer.Items.Add(computer.getName());
+ }
+ }
+
+ private void btnCancel_Click(object sender, System.EventArgs e)
+ {
+ this.Close();
+ this.Dispose();
+ }
+
+ private void comboComputer_SelectedIndexChanged(object sender, System.EventArgs e)
+ {
+ }
+
+
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/DATABASE.ICO b/ndb/src/cw/cpcc-win32/csharp/DATABASE.ICO
new file mode 100644
index 00000000000..9689aa88361
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/DATABASE.ICO
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/csharp/Database.cs b/ndb/src/cw/cpcc-win32/csharp/Database.cs
new file mode 100644
index 00000000000..39b8c160159
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/Database.cs
@@ -0,0 +1,162 @@
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+using System.Data;
+
+namespace NDB_CPC
+{
+ /// <summary>
+ /// Summary description for Database.
+ /// </summary>
+ public class Database
+ {
+ public enum Status {Disconnected=1,Connected=2, Unknown=3}
+ private string m_name;
+ private string m_owner;
+ private int m_mgmtPort;
+ private Status m_status;
+ private ArrayList m_processes;
+ public Database(string name)
+ {
+ m_name=name;
+ m_processes = new ArrayList();
+ }
+ public Database(string name, string owner)
+ {
+ m_name=name;
+ m_owner=owner;
+ m_processes = new ArrayList();
+ }
+ public Database()
+ {
+ m_processes = new ArrayList();
+ }
+
+ public string getName()
+ {
+ return m_name;
+ }
+
+ public void setName(string name)
+ {
+ m_name=name;
+ }
+
+ public void setMgmtPort(int port)
+ {
+ m_mgmtPort=port;
+ }
+
+ public string getOwner()
+ {
+ return m_owner;
+ }
+
+ public void setOwner(string name)
+ {
+ m_owner=name;
+ }
+
+
+ public Status getStatus()
+ {
+ return m_status;
+ }
+
+ public string getStatusString()
+ {
+ if(m_status.Equals(Status.Connected))
+ return "Connected";
+ if(m_status.Equals(Status.Disconnected))
+ return "Disconnected";
+ if(m_status.Equals(Status.Unknown))
+ return "Unknown";
+ return "Unknown";
+ }
+ public void setStatus(Status status)
+ {
+ m_status=status;
+ }
+
+ public void addProcess(Process process)
+ {
+ /*if(check)
+ {
+ if(m_processes==null)
+ return;
+ if(m_processes.Count>0)
+ {
+ foreach (Process p in m_processes)
+ {
+ if(process.getId().Equals(p.getId()))
+ return;
+ }
+ }
+ }
+ */
+ m_processes.Add(process);
+ }
+ public void addProcessCheck(Process process)
+ {
+
+ if(m_processes==null)
+ return;
+ if(m_processes.Count>0)
+ {
+ foreach (Process p in m_processes)
+ {
+ if(process.getId().Equals(p.getId()))
+ return;
+ }
+ }
+ m_processes.Add(process);
+ }
+
+ public Process getProcess(string id)
+ {
+ foreach(Process process in m_processes)
+ {
+ if(process.getId().Equals(id))
+ return process;
+ }
+ return null;
+ }
+
+ public Process getProcessByName(string name)
+ {
+ foreach(Process process in m_processes)
+ {
+ if(process.getName().Equals(name))
+ return process;
+ }
+ return null;
+ }
+
+ public void removeProcess( string processName)
+ {
+ Process p = this.getProcessByName(processName);
+ m_processes.Remove(p);
+ }
+
+ public void removeAllProcesses()
+ {
+ Computer c;
+ foreach(Process p in m_processes)
+ {
+ c=p.getComputer();
+ if(c.removeProcess(p.getName(),m_name).Equals(false))
+ {
+
+ }
+ }
+ m_processes.Clear();
+ }
+
+ public ArrayList getProcesses()
+ {
+ return m_processes;
+ }
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.csproj b/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.csproj
new file mode 100644
index 00000000000..6384eff8329
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.csproj
@@ -0,0 +1,240 @@
+<VisualStudioProject>
+ <CSHARP
+ ProjectType = "Local"
+ ProductVersion = "7.0.9466"
+ SchemaVersion = "1.0"
+ ProjectGuid = "{B78F6720-D36C-43DD-B442-F583718D0286}"
+ >
+ <Build>
+ <Settings
+ ApplicationIcon = "App.ico"
+ AssemblyKeyContainerName = ""
+ AssemblyName = "NDB_CPC"
+ AssemblyOriginatorKeyFile = ""
+ DefaultClientScript = "JScript"
+ DefaultHTMLPageLayout = "Grid"
+ DefaultTargetSchema = "IE50"
+ DelaySign = "false"
+ OutputType = "WinExe"
+ RootNamespace = "NDB_CPC"
+ StartupObject = ""
+ >
+ <Config
+ Name = "Debug"
+ AllowUnsafeBlocks = "false"
+ BaseAddress = "285212672"
+ CheckForOverflowUnderflow = "false"
+ ConfigurationOverrideFile = ""
+ DefineConstants = "DEBUG;TRACE"
+ DocumentationFile = ""
+ DebugSymbols = "true"
+ FileAlignment = "4096"
+ IncrementalBuild = "true"
+ Optimize = "false"
+ OutputPath = "bin\Debug\"
+ RegisterForComInterop = "false"
+ RemoveIntegerChecks = "false"
+ TreatWarningsAsErrors = "false"
+ WarningLevel = "4"
+ />
+ <Config
+ Name = "Release"
+ AllowUnsafeBlocks = "false"
+ BaseAddress = "285212672"
+ CheckForOverflowUnderflow = "false"
+ ConfigurationOverrideFile = ""
+ DefineConstants = "TRACE"
+ DocumentationFile = ""
+ DebugSymbols = "false"
+ FileAlignment = "4096"
+ IncrementalBuild = "false"
+ Optimize = "true"
+ OutputPath = "bin\Release\"
+ RegisterForComInterop = "false"
+ RemoveIntegerChecks = "false"
+ TreatWarningsAsErrors = "false"
+ WarningLevel = "4"
+ />
+ </Settings>
+ <References>
+ <Reference
+ Name = "System.Data"
+ AssemblyName = "System.Data"
+ HintPath = "..\..\..\..\..\WINNT\Microsoft.NET\Framework\v1.0.3705\System.Data.dll"
+ />
+ <Reference
+ Name = "System.Drawing"
+ AssemblyName = "System.Drawing"
+ HintPath = "..\..\..\..\..\WINNT\Microsoft.NET\Framework\v1.0.3705\System.Drawing.dll"
+ />
+ <Reference
+ Name = "System.Windows.Forms"
+ AssemblyName = "System.Windows.Forms"
+ HintPath = "..\..\..\..\..\WINNT\Microsoft.NET\Framework\v1.0.3705\System.Windows.Forms.dll"
+ />
+ <Reference
+ Name = "System.XML"
+ AssemblyName = "System.Xml"
+ HintPath = "..\..\..\..\..\WINNT\Microsoft.NET\Framework\v1.0.3705\System.XML.dll"
+ />
+ <Reference
+ Name = "System"
+ AssemblyName = "System"
+ HintPath = "..\..\..\..\..\WINNT\Microsoft.NET\Framework\v1.0.3705\System.dll"
+ />
+ <Reference
+ Name = "VBIDE"
+ Guid = "{0002E157-0000-0000-C000-000000000046}"
+ VersionMajor = "5"
+ VersionMinor = "3"
+ Lcid = "0"
+ WrapperTool = "tlbimp"
+ />
+ <Reference
+ Name = "stdole"
+ Guid = "{00020430-0000-0000-C000-000000000046}"
+ VersionMajor = "2"
+ VersionMinor = "0"
+ Lcid = "0"
+ WrapperTool = "primary"
+ />
+ <Reference
+ Name = "Microsoft.Office.Core"
+ Guid = "{2DF8D04C-5BFA-101B-BDE5-00AA0044DE52}"
+ VersionMajor = "2"
+ VersionMinor = "2"
+ Lcid = "0"
+ WrapperTool = "tlbimp"
+ />
+ </References>
+ </Build>
+ <Files>
+ <Include>
+ <File
+ RelPath = "app.config"
+ BuildAction = "None"
+ />
+ <File
+ RelPath = "App.ico"
+ BuildAction = "Content"
+ />
+ <File
+ RelPath = "AssemblyInfo.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "Computer.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "ComputerAddDialog.cs"
+ SubType = "Form"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "ComputerAddDialog.resx"
+ DependentUpon = "ComputerAddDialog.cs"
+ BuildAction = "EmbeddedResource"
+ />
+ <File
+ RelPath = "ComputerMgmt.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "ComputerRemoveDialog.cs"
+ SubType = "Form"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "ComputerRemoveDialog.af.resx"
+ DependentUpon = "ComputerRemoveDialog.cs"
+ BuildAction = "EmbeddedResource"
+ />
+ <File
+ RelPath = "ComputerRemoveDialog.resx"
+ DependentUpon = "ComputerRemoveDialog.cs"
+ BuildAction = "EmbeddedResource"
+ />
+ <File
+ RelPath = "CPC_Form.cs"
+ SubType = "Form"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "CPC_Form.resx"
+ DependentUpon = "CPC_Form.cs"
+ BuildAction = "EmbeddedResource"
+ />
+ <File
+ RelPath = "Database.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "PanelWizard.cs"
+ SubType = "Form"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "PanelWizard.resx"
+ DependentUpon = "PanelWizard.cs"
+ BuildAction = "EmbeddedResource"
+ />
+ <File
+ RelPath = "Process.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "ProcessDefineDialog.cs"
+ SubType = "Form"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "ProcessDefineDialog.resx"
+ DependentUpon = "ProcessDefineDialog.cs"
+ BuildAction = "EmbeddedResource"
+ />
+ <File
+ RelPath = "startDatabaseDlg.cs"
+ SubType = "Form"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "startDatabaseDlg.resx"
+ DependentUpon = "startDatabaseDlg.cs"
+ BuildAction = "EmbeddedResource"
+ />
+ <File
+ RelPath = "fileaccess\FileMgmt.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "simpleparser\SimpleCPCParser.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "socketcomm\myTcpClient.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "socketcomm\SocketComm.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ <File
+ RelPath = "telnetclient\telnetClient.cs"
+ SubType = "Code"
+ BuildAction = "Compile"
+ />
+ </Include>
+ </Files>
+ </CSHARP>
+</VisualStudioProject>
+
diff --git a/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.csproj.user b/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.csproj.user
new file mode 100644
index 00000000000..68937906d93
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.csproj.user
@@ -0,0 +1,48 @@
+<VisualStudioProject>
+ <CSHARP>
+ <Build>
+ <Settings ReferencePath = "" >
+ <Config
+ Name = "Debug"
+ EnableASPDebugging = "false"
+ EnableASPXDebugging = "false"
+ EnableUnmanagedDebugging = "false"
+ EnableSQLServerDebugging = "false"
+ RemoteDebugEnabled = "false"
+ RemoteDebugMachine = ""
+ StartAction = "Project"
+ StartArguments = ""
+ StartPage = ""
+ StartProgram = ""
+ StartURL = ""
+ StartWorkingDirectory = ""
+ StartWithIE = "false"
+ />
+ <Config
+ Name = "Release"
+ EnableASPDebugging = "false"
+ EnableASPXDebugging = "false"
+ EnableUnmanagedDebugging = "false"
+ EnableSQLServerDebugging = "false"
+ RemoteDebugEnabled = "false"
+ RemoteDebugMachine = ""
+ StartAction = "Project"
+ StartArguments = ""
+ StartPage = ""
+ StartProgram = ""
+ StartURL = ""
+ StartWorkingDirectory = ""
+ StartWithIE = "true"
+ />
+ </Settings>
+ </Build>
+ <OtherProjectSettings
+ CopyProjectDestinationFolder = ""
+ CopyProjectUncPath = ""
+ CopyProjectOption = "0"
+ ProjectView = "ProjectFiles"
+ ProjectTrust = "0"
+ />
+ </CSHARP>
+</VisualStudioProject>
+
diff --git a/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.ncb b/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.ncb
new file mode 100644
index 00000000000..ed3460476b0
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.ncb
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.sln b/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.sln
new file mode 100644
index 00000000000..ef18b5e94ce
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/NDB_CPC.sln
@@ -0,0 +1,21 @@
+Microsoft Visual Studio Solution File, Format Version 7.00
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "NDB_CPC", "NDB_CPC.csproj", "{B78F6720-D36C-43DD-B442-F583718D0286}"
+EndProject
+Global
+ GlobalSection(SolutionConfiguration) = preSolution
+ ConfigName.0 = Debug
+ ConfigName.1 = Release
+ EndGlobalSection
+ GlobalSection(ProjectDependencies) = postSolution
+ EndGlobalSection
+ GlobalSection(ProjectConfiguration) = postSolution
+ {B78F6720-D36C-43DD-B442-F583718D0286}.Debug.ActiveCfg = Debug|.NET
+ {B78F6720-D36C-43DD-B442-F583718D0286}.Debug.Build.0 = Debug|.NET
+ {B78F6720-D36C-43DD-B442-F583718D0286}.Release.ActiveCfg = Release|.NET
+ {B78F6720-D36C-43DD-B442-F583718D0286}.Release.Build.0 = Release|.NET
+ EndGlobalSection
+ GlobalSection(ExtensibilityGlobals) = postSolution
+ EndGlobalSection
+ GlobalSection(ExtensibilityAddIns) = postSolution
+ EndGlobalSection
+EndGlobal
diff --git a/ndb/src/cw/cpcc-win32/csharp/PanelWizard.cs b/ndb/src/cw/cpcc-win32/csharp/PanelWizard.cs
new file mode 100644
index 00000000000..f492aa64c60
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/PanelWizard.cs
@@ -0,0 +1,1883 @@
+//author:Arun
+//date:Nov 13,2002
+//Wizard using panel
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+
+namespace NDB_CPC
+{
+ /// <summary>
+ /// Summary description for MDXQueryBuilderWizard.
+ /// </summary>
+ public class PanelWizard : System.Windows.Forms.Form
+ {
+ private System.Windows.Forms.Button btnCancel;
+ private System.Windows.Forms.Button btnback;
+ private System.Windows.Forms.Button btnNext;
+ private System.Windows.Forms.Button btnFinish;
+
+ //---enabling and disabling the buttons
+ private bool cancelEnabled;
+ private bool backEnabled;
+ private bool nextEnabled;
+ private bool finishEnabled;
+ //--------
+ //--set the next and back panel
+ private Panel nextPanel;
+ private Panel backPanel;
+ private Panel presentPanel;
+ //
+ private Panel[] arrayPanel;
+ private System.Windows.Forms.Panel panel1;
+ private System.Windows.Forms.Panel panel2;
+ private System.Windows.Forms.Panel panel3;
+ private System.Windows.Forms.RadioButton radioBtnYes;
+ private System.Windows.Forms.RadioButton radioBtnNo;
+ private System.Windows.Forms.ListBox listBoxComputers;
+ private System.Windows.Forms.Label label5;
+ private System.Windows.Forms.Label label1;
+ private System.ComponentModel.IContainer components;
+ private System.Windows.Forms.Button buttonComputerAdd;
+ private System.Windows.Forms.Label label2;
+ private System.Windows.Forms.Label label6;
+ private System.Windows.Forms.Label label7;
+ private System.Windows.Forms.ComboBox comboNDB;
+ private System.Windows.Forms.ComboBox comboAPI;
+ private System.Windows.Forms.Label label8;
+ private System.Windows.Forms.ComboBox comboMGM;
+ private System.Windows.Forms.Button btnTransferNodeToComp;
+ private System.Windows.Forms.TreeView tvComputer;
+ private System.Windows.Forms.ListView lvNode;
+ private System.Windows.Forms.Button btnTransferCompToNode;
+ private System.Windows.Forms.Label label3;
+ private System.Windows.Forms.Label label9;
+ private System.Windows.Forms.Label label10;
+ private int m_nMGM;
+ private ComputerMgmt mgmt;
+ private int m_nNDB;
+ private int m_nAPI;
+ private Database m_db;
+ private System.Windows.Forms.Label label11;
+ private System.Windows.Forms.TextBox textDbName;
+ private System.Windows.Forms.Label label31;
+ private System.Windows.Forms.Label label32;
+ private System.Windows.Forms.Label label33;
+ private System.Windows.Forms.Label label18;
+ private System.Windows.Forms.Label labelTitle;
+ private System.Windows.Forms.Label labelCwd;
+ private System.Windows.Forms.Label labelArgs;
+ private System.Windows.Forms.Label labelOther;
+ private System.Windows.Forms.Label labelPath;
+ private int m_noOfConfiguredNodes;
+ private int m_noOfConfiguredMgmt;
+ private int m_noOfConfiguredNdb;
+ private string m_mgmHost;
+ private string m_mgmPort;
+ private System.Windows.Forms.TextBox textCwd;
+ private System.Windows.Forms.TextBox textArgs;
+ private System.Windows.Forms.TextBox textOther;
+ private System.Windows.Forms.TextBox textPath;
+ private System.Windows.Forms.TextBox textComputer;
+ private System.Windows.Forms.TextBox textDatabase;
+ private System.Windows.Forms.TextBox textName;
+ private int m_noOfConfiguredApi;
+ private bool m_bMgmt;
+ private System.Windows.Forms.Button buttonSave;
+ private System.Windows.Forms.CheckBox checkBoxReuse;
+ private System.Windows.Forms.Label label4;
+ private System.Windows.Forms.Panel panel4;
+ private System.Windows.Forms.CheckBox checkBoxLater;
+ private System.Windows.Forms.RadioButton radioYes;
+ private System.Windows.Forms.RadioButton radioNo;
+ private System.Windows.Forms.Panel panel6;
+ private System.Windows.Forms.Panel panel5;
+ private System.Windows.Forms.RadioButton radioStartNo;
+ private System.Windows.Forms.RadioButton radioStartYes;
+ private System.Windows.Forms.ImageList imageListComp;
+ private System.Windows.Forms.Label label12;
+ private System.Windows.Forms.TextBox textOwner;
+ private System.Windows.Forms.Label label13;
+ private System.Windows.Forms.TextBox textEnv;
+ private bool m_bNdb;
+ public PanelWizard(ComputerMgmt comp)
+ {
+ mgmt=comp;
+ m_noOfConfiguredNodes=0;
+ m_noOfConfiguredMgmt=0;
+ m_noOfConfiguredNdb=0;
+ m_noOfConfiguredApi=0;
+ Size panelSize= new Size(350,300);
+ Size s= new Size(355,360);
+ Point cancel= new Point(8,310);
+ Point back= new Point(96,310);
+ Point next = new Point(184,310);
+ Point finish= new Point(272,310);
+ InitializeComponent();
+ this.Size=s;
+ this.btnCancel.Location=cancel;
+
+ this.btnback.Location=back;
+ this.btnNext.Location=next;
+ this.btnFinish.Location=finish;
+
+ arrayPanel=new Panel[]{panel1,panel2,panel3,panel4,panel5,panel6};//,panel5, panel6};
+ panel1.Size=panelSize;
+
+ comboNDB.SelectedIndex=0;
+ comboAPI.SelectedIndex=0;
+ comboMGM.SelectedIndex=0;
+ m_bMgmt=false;
+ m_bNdb=false;
+
+ m_db = new Database();
+ if(listBoxComputers.Items.Count.Equals(0))
+ btnNext.Enabled=false;
+ }
+
+ /// <summary>
+ /// Clean up any resources being used.
+ /// </summary>
+ protected override void Dispose( bool disposing )
+ {
+ if( disposing )
+ {
+ if(components != null)
+ {
+ components.Dispose();
+ }
+ }
+ base.Dispose( disposing );
+ }
+
+ #region Windows Form Designer generated code
+ /// <summary>
+ /// Required method for Designer support - do not modify
+ /// the contents of this method with the code editor.
+ /// </summary>
+ private void InitializeComponent()
+ {
+ this.components = new System.ComponentModel.Container();
+ System.Resources.ResourceManager resources = new System.Resources.ResourceManager(typeof(PanelWizard));
+ this.panel1 = new System.Windows.Forms.Panel();
+ this.buttonComputerAdd = new System.Windows.Forms.Button();
+ this.label1 = new System.Windows.Forms.Label();
+ this.label5 = new System.Windows.Forms.Label();
+ this.listBoxComputers = new System.Windows.Forms.ListBox();
+ this.radioBtnNo = new System.Windows.Forms.RadioButton();
+ this.radioBtnYes = new System.Windows.Forms.RadioButton();
+ this.panel2 = new System.Windows.Forms.Panel();
+ this.label12 = new System.Windows.Forms.Label();
+ this.textOwner = new System.Windows.Forms.TextBox();
+ this.label11 = new System.Windows.Forms.Label();
+ this.textDbName = new System.Windows.Forms.TextBox();
+ this.label8 = new System.Windows.Forms.Label();
+ this.label7 = new System.Windows.Forms.Label();
+ this.label6 = new System.Windows.Forms.Label();
+ this.comboMGM = new System.Windows.Forms.ComboBox();
+ this.comboAPI = new System.Windows.Forms.ComboBox();
+ this.comboNDB = new System.Windows.Forms.ComboBox();
+ this.label2 = new System.Windows.Forms.Label();
+ this.panel3 = new System.Windows.Forms.Panel();
+ this.checkBoxLater = new System.Windows.Forms.CheckBox();
+ this.label10 = new System.Windows.Forms.Label();
+ this.label9 = new System.Windows.Forms.Label();
+ this.label3 = new System.Windows.Forms.Label();
+ this.btnTransferCompToNode = new System.Windows.Forms.Button();
+ this.btnTransferNodeToComp = new System.Windows.Forms.Button();
+ this.lvNode = new System.Windows.Forms.ListView();
+ this.tvComputer = new System.Windows.Forms.TreeView();
+ this.imageListComp = new System.Windows.Forms.ImageList(this.components);
+ this.panel6 = new System.Windows.Forms.Panel();
+ this.radioStartNo = new System.Windows.Forms.RadioButton();
+ this.radioStartYes = new System.Windows.Forms.RadioButton();
+ this.label18 = new System.Windows.Forms.Label();
+ this.btnCancel = new System.Windows.Forms.Button();
+ this.btnback = new System.Windows.Forms.Button();
+ this.btnNext = new System.Windows.Forms.Button();
+ this.btnFinish = new System.Windows.Forms.Button();
+ this.panel4 = new System.Windows.Forms.Panel();
+ this.textEnv = new System.Windows.Forms.TextBox();
+ this.label13 = new System.Windows.Forms.Label();
+ this.checkBoxReuse = new System.Windows.Forms.CheckBox();
+ this.buttonSave = new System.Windows.Forms.Button();
+ this.labelTitle = new System.Windows.Forms.Label();
+ this.textComputer = new System.Windows.Forms.TextBox();
+ this.textCwd = new System.Windows.Forms.TextBox();
+ this.textArgs = new System.Windows.Forms.TextBox();
+ this.textOther = new System.Windows.Forms.TextBox();
+ this.textPath = new System.Windows.Forms.TextBox();
+ this.textDatabase = new System.Windows.Forms.TextBox();
+ this.textName = new System.Windows.Forms.TextBox();
+ this.labelCwd = new System.Windows.Forms.Label();
+ this.labelArgs = new System.Windows.Forms.Label();
+ this.labelOther = new System.Windows.Forms.Label();
+ this.labelPath = new System.Windows.Forms.Label();
+ this.label31 = new System.Windows.Forms.Label();
+ this.label32 = new System.Windows.Forms.Label();
+ this.label33 = new System.Windows.Forms.Label();
+ this.panel5 = new System.Windows.Forms.Panel();
+ this.radioNo = new System.Windows.Forms.RadioButton();
+ this.radioYes = new System.Windows.Forms.RadioButton();
+ this.label4 = new System.Windows.Forms.Label();
+ this.panel1.SuspendLayout();
+ this.panel2.SuspendLayout();
+ this.panel3.SuspendLayout();
+ this.panel6.SuspendLayout();
+ this.panel4.SuspendLayout();
+ this.panel5.SuspendLayout();
+ this.SuspendLayout();
+ //
+ // panel1
+ //
+ this.panel1.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.buttonComputerAdd,
+ this.label1,
+ this.label5,
+ this.listBoxComputers,
+ this.radioBtnNo,
+ this.radioBtnYes});
+ this.panel1.Name = "panel1";
+ this.panel1.Size = new System.Drawing.Size(344, 312);
+ this.panel1.TabIndex = 0;
+ this.panel1.Paint += new System.Windows.Forms.PaintEventHandler(this.panel1_Paint);
+ //
+ // buttonComputerAdd
+ //
+ this.buttonComputerAdd.Enabled = false;
+ this.buttonComputerAdd.Location = new System.Drawing.Point(192, 232);
+ this.buttonComputerAdd.Name = "buttonComputerAdd";
+ this.buttonComputerAdd.Size = new System.Drawing.Size(96, 24);
+ this.buttonComputerAdd.TabIndex = 3;
+ this.buttonComputerAdd.Text = "Add computer...";
+ this.buttonComputerAdd.Click += new System.EventHandler(this.buttonComputerAdd_Click);
+ //
+ // label1
+ //
+ this.label1.Font = new System.Drawing.Font("Microsoft Sans Serif", 12F, System.Drawing.FontStyle.Bold, System.Drawing.GraphicsUnit.Point, ((System.Byte)(0)));
+ this.label1.Location = new System.Drawing.Point(80, 8);
+ this.label1.Name = "label1";
+ this.label1.Size = new System.Drawing.Size(200, 23);
+ this.label1.TabIndex = 5;
+ this.label1.Text = "Configure computers";
+ this.label1.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
+ //
+ // label5
+ //
+ this.label5.Location = new System.Drawing.Point(24, 40);
+ this.label5.Name = "label5";
+ this.label5.Size = new System.Drawing.Size(128, 23);
+ this.label5.TabIndex = 4;
+ this.label5.Text = "Available computers:";
+ this.label5.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
+ //
+ // listBoxComputers
+ //
+ this.listBoxComputers.Location = new System.Drawing.Point(24, 64);
+ this.listBoxComputers.Name = "listBoxComputers";
+ this.listBoxComputers.Size = new System.Drawing.Size(128, 212);
+ this.listBoxComputers.TabIndex = 3;
+ this.listBoxComputers.SelectedIndexChanged += new System.EventHandler(this.listBoxComputers_SelectedIndexChanged);
+ //
+ // radioBtnNo
+ //
+ this.radioBtnNo.AutoCheck = false;
+ this.radioBtnNo.Location = new System.Drawing.Point(168, 168);
+ this.radioBtnNo.Name = "radioBtnNo";
+ this.radioBtnNo.Size = new System.Drawing.Size(152, 64);
+ this.radioBtnNo.TabIndex = 2;
+ this.radioBtnNo.Text = "No, I have to add more computers in order to deploy NDB Cluster. ";
+ this.radioBtnNo.Click += new System.EventHandler(this.radioBtnNo_Click);
+ //
+ // radioBtnYes
+ //
+ this.radioBtnYes.AutoCheck = false;
+ this.radioBtnYes.Location = new System.Drawing.Point(168, 72);
+ this.radioBtnYes.Name = "radioBtnYes";
+ this.radioBtnYes.Size = new System.Drawing.Size(152, 80);
+ this.radioBtnYes.TabIndex = 1;
+ this.radioBtnYes.Text = "Yes, all the computers that I need to deploy NDB Cluster exists in the list \"Avai" +
+ "lable computers\"";
+ this.radioBtnYes.Click += new System.EventHandler(this.radioBtnYes_Click);
+ //
+ // panel2
+ //
+ this.panel2.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.label12,
+ this.textOwner,
+ this.label11,
+ this.textDbName,
+ this.label8,
+ this.label7,
+ this.label6,
+ this.comboMGM,
+ this.comboAPI,
+ this.comboNDB,
+ this.label2});
+ this.panel2.Location = new System.Drawing.Point(0, 320);
+ this.panel2.Name = "panel2";
+ this.panel2.Size = new System.Drawing.Size(344, 312);
+ this.panel2.TabIndex = 1;
+ this.panel2.Validating += new System.ComponentModel.CancelEventHandler(this.panel2_Validating);
+ this.panel2.Paint += new System.Windows.Forms.PaintEventHandler(this.panel2_Paint);
+ //
+ // label12
+ //
+ this.label12.Location = new System.Drawing.Point(72, 216);
+ this.label12.Name = "label12";
+ this.label12.Size = new System.Drawing.Size(112, 24);
+ this.label12.TabIndex = 16;
+ this.label12.Text = "Database owner:";
+ this.label12.TextAlign = System.Drawing.ContentAlignment.MiddleRight;
+ //
+ // textOwner
+ //
+ this.textOwner.Location = new System.Drawing.Point(192, 216);
+ this.textOwner.Name = "textOwner";
+ this.textOwner.TabIndex = 5;
+ this.textOwner.Text = "";
+ this.textOwner.TextChanged += new System.EventHandler(this.textOwner_TextChanged);
+ //
+ // label11
+ //
+ this.label11.Location = new System.Drawing.Point(72, 184);
+ this.label11.Name = "label11";
+ this.label11.Size = new System.Drawing.Size(112, 24);
+ this.label11.TabIndex = 14;
+ this.label11.Text = "Database name:";
+ this.label11.TextAlign = System.Drawing.ContentAlignment.MiddleRight;
+ this.label11.Click += new System.EventHandler(this.label11_Click);
+ //
+ // textDbName
+ //
+ this.textDbName.Location = new System.Drawing.Point(192, 184);
+ this.textDbName.Name = "textDbName";
+ this.textDbName.TabIndex = 4;
+ this.textDbName.Text = "";
+
+ this.textDbName.TextChanged += new System.EventHandler(this.textDbName_TextChanged);
+ //
+ // label8
+ //
+ this.label8.Location = new System.Drawing.Point(16, 120);
+ this.label8.Name = "label8";
+ this.label8.Size = new System.Drawing.Size(176, 24);
+ this.label8.TabIndex = 12;
+ this.label8.Text = "Number of management servers:";
+ this.label8.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
+ //
+ // label7
+ //
+ this.label7.Location = new System.Drawing.Point(16, 88);
+ this.label7.Name = "label7";
+ this.label7.Size = new System.Drawing.Size(120, 24);
+ this.label7.TabIndex = 11;
+ this.label7.Text = "Number of API nodes:";
+ this.label7.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
+ //
+ // label6
+ //
+ this.label6.Location = new System.Drawing.Point(16, 56);
+ this.label6.Name = "label6";
+ this.label6.Size = new System.Drawing.Size(144, 24);
+ this.label6.TabIndex = 10;
+ this.label6.Text = "Number of database nodes:";
+ this.label6.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
+ //
+ // comboMGM
+ //
+ this.comboMGM.DisplayMember = "0";
+ this.comboMGM.Items.AddRange(new object[] {
+ "1"});
+ this.comboMGM.Location = new System.Drawing.Point(192, 120);
+ this.comboMGM.Name = "comboMGM";
+ this.comboMGM.Size = new System.Drawing.Size(104, 21);
+ this.comboMGM.TabIndex = 3;
+ this.comboMGM.Text = "comboBox3";
+ //
+ // comboAPI
+ //
+ this.comboAPI.DisplayMember = "0";
+ this.comboAPI.Items.AddRange(new object[] {
+ "1",
+ "2",
+ "3",
+ "4",
+ "5",
+ "6",
+ "7",
+ "8",
+ "9",
+ "10"});
+ this.comboAPI.Location = new System.Drawing.Point(192, 88);
+ this.comboAPI.Name = "comboAPI";
+ this.comboAPI.Size = new System.Drawing.Size(104, 21);
+ this.comboAPI.TabIndex = 2;
+ this.comboAPI.Text = "comboBox2";
+ //
+ // comboNDB
+ //
+ this.comboNDB.DisplayMember = "0";
+ this.comboNDB.Items.AddRange(new object[] {
+ "1",
+ "2",
+ "4",
+ "8"});
+ this.comboNDB.Location = new System.Drawing.Point(192, 56);
+ this.comboNDB.Name = "comboNDB";
+ this.comboNDB.Size = new System.Drawing.Size(104, 21);
+ this.comboNDB.TabIndex = 1;
+ this.comboNDB.Text = "comboBox1";
+ //
+ // label2
+ //
+ this.label2.Font = new System.Drawing.Font("Microsoft Sans Serif", 12F, System.Drawing.FontStyle.Bold, System.Drawing.GraphicsUnit.Point, ((System.Byte)(0)));
+ this.label2.Location = new System.Drawing.Point(80, 8);
+ this.label2.Name = "label2";
+ this.label2.Size = new System.Drawing.Size(208, 23);
+ this.label2.TabIndex = 6;
+ this.label2.Text = "Setup NDB Cluster nodes";
+ this.label2.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
+ //
+ // panel3
+ //
+ this.panel3.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.checkBoxLater,
+ this.label10,
+ this.label9,
+ this.label3,
+ this.btnTransferCompToNode,
+ this.btnTransferNodeToComp,
+ this.lvNode,
+ this.tvComputer});
+ this.panel3.Location = new System.Drawing.Point(360, 8);
+ this.panel3.Name = "panel3";
+ this.panel3.Size = new System.Drawing.Size(320, 312);
+ this.panel3.TabIndex = 2;
+ //
+ // checkBoxLater
+ //
+ this.checkBoxLater.Location = new System.Drawing.Point(40, 256);
+ this.checkBoxLater.Name = "checkBoxLater";
+ this.checkBoxLater.Size = new System.Drawing.Size(240, 16);
+ this.checkBoxLater.TabIndex = 9;
+ this.checkBoxLater.Text = "I will configure these nodes manually, later.";
+ //
+ // label10
+ //
+ this.label10.Location = new System.Drawing.Point(16, 40);
+ this.label10.Name = "label10";
+ this.label10.Size = new System.Drawing.Size(104, 16);
+ this.label10.TabIndex = 8;
+ this.label10.Text = "NDB Cluster nodes:";
+ this.label10.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
+ //
+ // label9
+ //
+ this.label9.Location = new System.Drawing.Point(192, 40);
+ this.label9.Name = "label9";
+ this.label9.Size = new System.Drawing.Size(100, 16);
+ this.label9.TabIndex = 7;
+ this.label9.Text = "Computers:";
+ this.label9.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
+ //
+ // label3
+ //
+ this.label3.Font = new System.Drawing.Font("Microsoft Sans Serif", 12F, System.Drawing.FontStyle.Bold, System.Drawing.GraphicsUnit.Point, ((System.Byte)(0)));
+ this.label3.Location = new System.Drawing.Point(40, 8);
+ this.label3.Name = "label3";
+ this.label3.Size = new System.Drawing.Size(280, 23);
+ this.label3.TabIndex = 6;
+ this.label3.Text = "Assign NDB nodes to computers";
+ this.label3.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
+ //
+ // btnTransferCompToNode
+ //
+ this.btnTransferCompToNode.Location = new System.Drawing.Point(144, 160);
+ this.btnTransferCompToNode.Name = "btnTransferCompToNode";
+ this.btnTransferCompToNode.Size = new System.Drawing.Size(40, 24);
+ this.btnTransferCompToNode.TabIndex = 4;
+ this.btnTransferCompToNode.Text = "<---";
+ //
+ // btnTransferNodeToComp
+ //
+ this.btnTransferNodeToComp.Location = new System.Drawing.Point(144, 128);
+ this.btnTransferNodeToComp.Name = "btnTransferNodeToComp";
+ this.btnTransferNodeToComp.Size = new System.Drawing.Size(40, 24);
+ this.btnTransferNodeToComp.TabIndex = 3;
+ this.btnTransferNodeToComp.Text = "--->";
+ this.btnTransferNodeToComp.Click += new System.EventHandler(this.btnTransferNodeToComp_Click);
+ //
+ // lvNode
+ //
+ this.lvNode.HideSelection = false;
+ this.lvNode.Location = new System.Drawing.Point(16, 56);
+ this.lvNode.Name = "lvNode";
+ this.lvNode.Size = new System.Drawing.Size(112, 192);
+ this.lvNode.TabIndex = 2;
+ this.lvNode.View = System.Windows.Forms.View.List;
+ this.lvNode.SelectedIndexChanged += new System.EventHandler(this.lvNode_SelectedIndexChanged);
+ //
+ // tvComputer
+ //
+ this.tvComputer.HideSelection = false;
+ this.tvComputer.ImageList = this.imageListComp;
+ this.tvComputer.Location = new System.Drawing.Point(192, 56);
+ this.tvComputer.Name = "tvComputer";
+ this.tvComputer.Size = new System.Drawing.Size(120, 192);
+ this.tvComputer.TabIndex = 1;
+ this.tvComputer.MouseDown += new System.Windows.Forms.MouseEventHandler(this.tvComputer_MouseDown);
+ this.tvComputer.AfterSelect += new System.Windows.Forms.TreeViewEventHandler(this.tvComputer_AfterSelect);
+ this.tvComputer.MouseLeave += new System.EventHandler(this.tvComputer_MouseLeave);
+ this.tvComputer.DragDrop += new System.Windows.Forms.DragEventHandler(this.tvComputer_DragDrop);
+ //
+ // imageListComp
+ //
+ this.imageListComp.ColorDepth = System.Windows.Forms.ColorDepth.Depth8Bit;
+ this.imageListComp.ImageSize = new System.Drawing.Size(16, 16);
+ this.imageListComp.ImageStream = ((System.Windows.Forms.ImageListStreamer)(resources.GetObject("imageListComp.ImageStream")));
+ this.imageListComp.TransparentColor = System.Drawing.Color.Transparent;
+ //
+ // panel6
+ //
+ this.panel6.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.radioStartNo,
+ this.radioStartYes,
+ this.label18});
+ this.panel6.Location = new System.Drawing.Point(344, 336);
+ this.panel6.Name = "panel6";
+ this.panel6.Size = new System.Drawing.Size(344, 312);
+ this.panel6.TabIndex = 3;
+ this.panel6.Paint += new System.Windows.Forms.PaintEventHandler(this.panel4_Paint);
+ //
+ // radioStartNo
+ //
+ this.radioStartNo.Location = new System.Drawing.Point(40, 144);
+ this.radioStartNo.Name = "radioStartNo";
+ this.radioStartNo.Size = new System.Drawing.Size(272, 48);
+ this.radioStartNo.TabIndex = 81;
+ this.radioStartNo.Text = "Manually start NDB Cluster. The Magician will exit and you must start NDB Cluster" +
+ " manually.";
+ this.radioStartNo.CheckedChanged += new System.EventHandler(this.radioStartNo_CheckedChanged);
+ //
+ // radioStartYes
+ //
+ this.radioStartYes.Location = new System.Drawing.Point(40, 40);
+ this.radioStartYes.Name = "radioStartYes";
+ this.radioStartYes.Size = new System.Drawing.Size(272, 88);
+ this.radioStartYes.TabIndex = 80;
+ this.radioStartYes.Text = "Start NDB Cluster now. The Magician will start NDB Cluster and exit. MAKE SURE YO" +
+ "U HAVE STARTED THE MGMTSRVR WITH THE CORRECT CONFIGURATION FILE!!!";
+ this.radioStartYes.CheckedChanged += new System.EventHandler(this.radioStartYes_CheckedChanged);
+ //
+ // label18
+ //
+ this.label18.Font = new System.Drawing.Font("Microsoft Sans Serif", 12F, System.Drawing.FontStyle.Bold, System.Drawing.GraphicsUnit.Point, ((System.Byte)(0)));
+ this.label18.Location = new System.Drawing.Point(56, 8);
+ this.label18.Name = "label18";
+ this.label18.Size = new System.Drawing.Size(224, 24);
+ this.label18.TabIndex = 79;
+ this.label18.Text = "Start NDB Cluster and finish";
+ //
+ // btnCancel
+ //
+ this.btnCancel.Location = new System.Drawing.Point(8, 656);
+ this.btnCancel.Name = "btnCancel";
+ this.btnCancel.Size = new System.Drawing.Size(70, 23);
+ this.btnCancel.TabIndex = 10;
+ this.btnCancel.Text = "Cancel";
+ this.btnCancel.Click += new System.EventHandler(this.btnCancel_Click);
+ //
+ // btnback
+ //
+ this.btnback.Location = new System.Drawing.Point(96, 656);
+ this.btnback.Name = "btnback";
+ this.btnback.Size = new System.Drawing.Size(70, 23);
+ this.btnback.TabIndex = 11;
+ this.btnback.Text = "< Back";
+ this.btnback.Click += new System.EventHandler(this.btnback_Click);
+ //
+ // btnNext
+ //
+ this.btnNext.Location = new System.Drawing.Point(184, 656);
+ this.btnNext.Name = "btnNext";
+ this.btnNext.Size = new System.Drawing.Size(70, 23);
+ this.btnNext.TabIndex = 12;
+ this.btnNext.Text = "Next >";
+ this.btnNext.Click += new System.EventHandler(this.btnNext_Click);
+ //
+ // btnFinish
+ //
+ this.btnFinish.Location = new System.Drawing.Point(272, 656);
+ this.btnFinish.Name = "btnFinish";
+ this.btnFinish.Size = new System.Drawing.Size(70, 23);
+ this.btnFinish.TabIndex = 13;
+ this.btnFinish.Text = "Finish";
+ this.btnFinish.Click += new System.EventHandler(this.btnFinish_Click);
+ //
+ // panel4
+ //
+ this.panel4.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.textEnv,
+ this.label13,
+ this.checkBoxReuse,
+ this.buttonSave,
+ this.labelTitle,
+ this.textComputer,
+ this.textCwd,
+ this.textArgs,
+ this.textOther,
+ this.textPath,
+ this.textDatabase,
+ this.textName,
+ this.labelCwd,
+ this.labelArgs,
+ this.labelOther,
+ this.labelPath,
+ this.label31,
+ this.label32,
+ this.label33});
+ this.panel4.Location = new System.Drawing.Point(672, 8);
+ this.panel4.Name = "panel4";
+ this.panel4.Size = new System.Drawing.Size(344, 312);
+ this.panel4.TabIndex = 62;
+ this.panel4.Paint += new System.Windows.Forms.PaintEventHandler(this.panel5_Paint);
+ //
+ // textEnv
+ //
+ this.textEnv.Location = new System.Drawing.Point(136, 136);
+ this.textEnv.Name = "textEnv";
+ this.textEnv.Size = new System.Drawing.Size(184, 20);
+ this.textEnv.TabIndex = 2;
+ this.textEnv.TabStop = false;
+ this.textEnv.Text = "";
+ //
+ // label13
+ //
+ this.label13.Location = new System.Drawing.Point(8, 136);
+ this.label13.Name = "label13";
+ this.label13.Size = new System.Drawing.Size(136, 24);
+ this.label13.TabIndex = 81;
+ this.label13.Text = "Environment variables:";
+ //
+ // checkBoxReuse
+ //
+ this.checkBoxReuse.Location = new System.Drawing.Point(88, 232);
+ this.checkBoxReuse.Name = "checkBoxReuse";
+ this.checkBoxReuse.Size = new System.Drawing.Size(240, 32);
+ this.checkBoxReuse.TabIndex = 5;
+ this.checkBoxReuse.TabStop = false;
+ this.checkBoxReuse.Text = "Use the same configuration for ALL NDB nodes?";
+ //
+ // buttonSave
+ //
+ this.buttonSave.Location = new System.Drawing.Point(184, 264);
+ this.buttonSave.Name = "buttonSave";
+ this.buttonSave.Size = new System.Drawing.Size(88, 24);
+ this.buttonSave.TabIndex = 6;
+ this.buttonSave.Text = "Save";
+ this.buttonSave.Click += new System.EventHandler(this.buttonSave_Click);
+ //
+ // labelTitle
+ //
+ this.labelTitle.Font = new System.Drawing.Font("Microsoft Sans Serif", 12F, System.Drawing.FontStyle.Bold, System.Drawing.GraphicsUnit.Point, ((System.Byte)(0)));
+ this.labelTitle.Location = new System.Drawing.Point(80, 16);
+ this.labelTitle.Name = "labelTitle";
+ this.labelTitle.Size = new System.Drawing.Size(192, 23);
+ this.labelTitle.TabIndex = 79;
+ this.labelTitle.Text = "Mgmtsrvr configuration";
+ //
+ // textComputer
+ //
+ this.textComputer.Location = new System.Drawing.Point(136, 40);
+ this.textComputer.Name = "textComputer";
+ this.textComputer.ReadOnly = true;
+ this.textComputer.Size = new System.Drawing.Size(184, 20);
+ this.textComputer.TabIndex = 77;
+ this.textComputer.TabStop = false;
+ this.textComputer.Text = "";
+ //
+ // textCwd
+ //
+ this.textCwd.Location = new System.Drawing.Point(136, 208);
+ this.textCwd.Name = "textCwd";
+ this.textCwd.Size = new System.Drawing.Size(184, 20);
+ this.textCwd.TabIndex = 5;
+ this.textCwd.TabStop = false;
+ this.textCwd.Text = "";
+ //
+ // textArgs
+ //
+ this.textArgs.Location = new System.Drawing.Point(136, 184);
+ this.textArgs.Name = "textArgs";
+ this.textArgs.Size = new System.Drawing.Size(184, 20);
+ this.textArgs.TabIndex = 4;
+ this.textArgs.TabStop = false;
+ this.textArgs.Text = "";
+ //
+ // textOther
+ //
+ this.textOther.Location = new System.Drawing.Point(136, 160);
+ this.textOther.Name = "textOther";
+ this.textOther.Size = new System.Drawing.Size(184, 20);
+ this.textOther.TabIndex = 3;
+ this.textOther.TabStop = false;
+ this.textOther.Text = "";
+ //
+ // textPath
+ //
+ this.textPath.Location = new System.Drawing.Point(136, 112);
+ this.textPath.Name = "textPath";
+ this.textPath.Size = new System.Drawing.Size(184, 20);
+ this.textPath.TabIndex = 1;
+ this.textPath.TabStop = false;
+ this.textPath.Text = "";
+ this.textPath.TextChanged += new System.EventHandler(this.textPath_TextChanged);
+ //
+ // textDatabase
+ //
+ this.textDatabase.Location = new System.Drawing.Point(136, 88);
+ this.textDatabase.Name = "textDatabase";
+ this.textDatabase.ReadOnly = true;
+ this.textDatabase.Size = new System.Drawing.Size(184, 20);
+ this.textDatabase.TabIndex = 62;
+ this.textDatabase.TabStop = false;
+ this.textDatabase.Text = "";
+ //
+ // textName
+ //
+ this.textName.Location = new System.Drawing.Point(136, 64);
+ this.textName.Name = "textName";
+ this.textName.ReadOnly = true;
+ this.textName.Size = new System.Drawing.Size(184, 20);
+ this.textName.TabIndex = 60;
+ this.textName.TabStop = false;
+ this.textName.Text = "";
+ //
+ // labelCwd
+ //
+ this.labelCwd.Location = new System.Drawing.Point(8, 208);
+ this.labelCwd.Name = "labelCwd";
+ this.labelCwd.Size = new System.Drawing.Size(112, 24);
+ this.labelCwd.TabIndex = 72;
+ this.labelCwd.Text = "Current working dir.:";
+ //
+ // labelArgs
+ //
+ this.labelArgs.Location = new System.Drawing.Point(8, 184);
+ this.labelArgs.Name = "labelArgs";
+ this.labelArgs.Size = new System.Drawing.Size(128, 24);
+ this.labelArgs.TabIndex = 70;
+ this.labelArgs.Text = "Arguments to mgmtsrvr:";
+ //
+ // labelOther
+ //
+ this.labelOther.Location = new System.Drawing.Point(8, 160);
+ this.labelOther.Name = "labelOther";
+ this.labelOther.Size = new System.Drawing.Size(136, 24);
+ this.labelOther.TabIndex = 69;
+ this.labelOther.Text = "Mgmtsrvr port:";
+ //
+ // labelPath
+ //
+ this.labelPath.Location = new System.Drawing.Point(8, 112);
+ this.labelPath.Name = "labelPath";
+ this.labelPath.Size = new System.Drawing.Size(128, 24);
+ this.labelPath.TabIndex = 67;
+ this.labelPath.Text = "Path to mgmtsrvr binary:";
+ //
+ // label31
+ //
+ this.label31.Location = new System.Drawing.Point(8, 88);
+ this.label31.Name = "label31";
+ this.label31.Size = new System.Drawing.Size(88, 24);
+ this.label31.TabIndex = 65;
+ this.label31.Text = "Database:";
+ //
+ // label32
+ //
+ this.label32.Location = new System.Drawing.Point(8, 64);
+ this.label32.Name = "label32";
+ this.label32.Size = new System.Drawing.Size(88, 24);
+ this.label32.TabIndex = 63;
+ this.label32.Text = "Process name:";
+ //
+ // label33
+ //
+ this.label33.Location = new System.Drawing.Point(8, 40);
+ this.label33.Name = "label33";
+ this.label33.Size = new System.Drawing.Size(64, 24);
+ this.label33.TabIndex = 61;
+ this.label33.Text = "Computer:";
+ //
+ // panel5
+ //
+ this.panel5.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.radioNo,
+ this.radioYes,
+ this.label4});
+ this.panel5.Location = new System.Drawing.Point(672, 328);
+ this.panel5.Name = "panel5";
+ this.panel5.Size = new System.Drawing.Size(344, 312);
+ this.panel5.TabIndex = 63;
+ //
+ // radioNo
+ //
+ this.radioNo.Location = new System.Drawing.Point(72, 160);
+ this.radioNo.Name = "radioNo";
+ this.radioNo.Size = new System.Drawing.Size(240, 48);
+ this.radioNo.TabIndex = 1;
+ this.radioNo.Text = "I already have a configuration file that I want to use for this configuration.";
+ this.radioNo.CheckedChanged += new System.EventHandler(this.radioNo_CheckedChanged);
+ //
+ // radioYes
+ //
+ this.radioYes.Checked = true;
+ this.radioYes.Location = new System.Drawing.Point(72, 56);
+ this.radioYes.Name = "radioYes";
+ this.radioYes.Size = new System.Drawing.Size(240, 88);
+ this.radioYes.TabIndex = 0;
+ this.radioYes.TabStop = true;
+ this.radioYes.Text = "Generate a configuration file template (initconfig.txt) for the mgmtsrvr based on" +
+ " the specified configuration? Notepad will be started with a template that you m" +
+ "ust complete and save in the cwd of the mgmtsrvr.";
+ this.radioYes.CheckedChanged += new System.EventHandler(this.radioYes_CheckedChanged);
+ //
+ // label4
+ //
+ this.label4.Font = new System.Drawing.Font("Microsoft Sans Serif", 12F, System.Drawing.FontStyle.Bold, System.Drawing.GraphicsUnit.Point, ((System.Byte)(0)));
+ this.label4.Location = new System.Drawing.Point(88, 8);
+ this.label4.Name = "label4";
+ this.label4.Size = new System.Drawing.Size(192, 40);
+ this.label4.TabIndex = 79;
+ this.label4.Text = "Tying up the configuration";
+ this.label4.TextAlign = System.Drawing.ContentAlignment.TopCenter;
+ //
+ // PanelWizard
+ //
+ this.AutoScaleBaseSize = new System.Drawing.Size(5, 13);
+ this.ClientSize = new System.Drawing.Size(1030, 755);
+ this.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.panel5,
+ this.panel4,
+ this.panel1,
+ this.btnFinish,
+ this.btnNext,
+ this.btnback,
+ this.btnCancel,
+ this.panel6,
+ this.panel3,
+ this.panel2});
+ this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.FixedDialog;
+ this.MaximizeBox = false;
+ this.MinimizeBox = false;
+ this.Name = "PanelWizard";
+ this.SizeGripStyle = System.Windows.Forms.SizeGripStyle.Hide;
+ this.Text = "Create Database Magician";
+ this.Load += new System.EventHandler(this.MDXQueryBuilderWizard_Load);
+ this.Activated += new System.EventHandler(this.PanelWizard_Activated);
+ this.panel1.ResumeLayout(false);
+ this.panel2.ResumeLayout(false);
+ this.panel3.ResumeLayout(false);
+ this.panel6.ResumeLayout(false);
+ this.panel4.ResumeLayout(false);
+ this.panel5.ResumeLayout(false);
+ this.ResumeLayout(false);
+
+ }
+ #endregion
+
+ private void MDXQueryBuilderWizard_Load(object sender, System.EventArgs e)
+ {
+
+ foreach(Control ct in this.Controls)
+ {
+ if(ct.GetType().Name=="Panel")
+ {
+ ct.Left=0;
+ ct.Top=0;
+ ct.Visible=false;
+ }
+
+ }
+ presentPanel=arrayPanel[0];
+ //--set the properties
+ setBtnPanProperty(getPosition(presentPanel));
+ //------
+ refreshLook();
+ }
+
+ //-set the buttons and panel
+ private void refreshLook()
+ {
+ if(cancelEnabled)
+ btnCancel.Enabled=true;
+ else
+ btnCancel.Enabled=false;
+
+ if(backEnabled)
+ btnback.Enabled=true;
+ else
+ btnback.Enabled=false;
+
+ if(nextEnabled)
+ btnNext.Enabled=true;
+ else
+ btnNext.Enabled=false;
+
+ if(finishEnabled)
+ btnFinish.Enabled=true;
+ else
+ btnFinish.Enabled=false;
+
+ if(presentPanel!=null)
+ {
+ presentPanel.Show();
+ presentPanel.BringToFront();
+ }
+ }
+ //--------
+ private int getPosition(Panel p)
+ {
+ int result=-1;
+ for(int i=0;i<arrayPanel.Length;i++)
+ {
+ if(arrayPanel[i]==p)
+ {
+ result=i;
+ break;
+ }
+ }
+ return result ;
+ }
+ //----
+
+ private void setBtnPanProperty(int presentPanelPosition )
+ {
+ int panelLength=arrayPanel.Length-1;
+ if(presentPanelPosition==0)
+ {
+ //first panel...no back ,no finish
+ cancelEnabled=true;
+ backEnabled=false;
+ nextEnabled=false;
+ finishEnabled=false;
+ presentPanel=arrayPanel[presentPanelPosition];
+ nextPanel=arrayPanel[presentPanelPosition+1];
+ backPanel=null;
+
+ }
+ else if(presentPanelPosition==1)
+ {
+ cancelEnabled=true;
+ backEnabled=true;
+ nextEnabled=false;
+ finishEnabled=false;
+ presentPanel=arrayPanel[presentPanelPosition];
+ nextPanel=arrayPanel[presentPanelPosition+1];
+ backPanel=arrayPanel[presentPanelPosition-1];
+ }
+
+ else if(presentPanelPosition==2)
+ {
+ cancelEnabled=true;
+ backEnabled=true;
+ nextEnabled=false;
+ finishEnabled=false;
+ presentPanel=arrayPanel[presentPanelPosition];
+ nextPanel=arrayPanel[presentPanelPosition+1];
+ backPanel=arrayPanel[presentPanelPosition-1];
+ }
+
+ else if(presentPanelPosition==3)
+ {
+ //last panel...no next,finish
+ cancelEnabled=true;
+ backEnabled=true;
+ nextEnabled=true;
+ finishEnabled=false;
+ presentPanel=arrayPanel[presentPanelPosition];
+ nextPanel=arrayPanel[presentPanelPosition+1];
+ backPanel=arrayPanel[presentPanelPosition-1];
+ }
+ else if(presentPanelPosition==4)
+ {
+ //last panel...no next,finish
+ cancelEnabled=true;
+ backEnabled=true;
+ nextEnabled=true;
+ finishEnabled=false;
+ presentPanel=arrayPanel[presentPanelPosition];
+ nextPanel=presentPanel;
+ backPanel=arrayPanel[presentPanelPosition-1];
+ }
+
+ else if(presentPanelPosition==5)
+ {
+ //last panel...no next,finish yes
+ cancelEnabled=true;
+ backEnabled=true;
+ nextEnabled=false;
+ finishEnabled=false;
+ presentPanel=arrayPanel[presentPanelPosition];
+ nextPanel=null;
+ backPanel=arrayPanel[presentPanelPosition-1];
+ }
+
+ else
+ {
+ //no finish,next and back
+ cancelEnabled=true;
+ backEnabled=true;
+ nextEnabled=false;
+ finishEnabled=true;
+ presentPanel=arrayPanel[presentPanelPosition];
+ nextPanel=null;
+ backPanel=arrayPanel[presentPanelPosition-1];
+ }
+ }
+
+ private void btnNext_Click(object sender, System.EventArgs e)
+ {
+
+ if(arrayPanel[getPosition(presentPanel)].Equals(panel1))
+ {
+ presentPanel=arrayPanel[getPosition(presentPanel)+1];
+ setBtnPanProperty(getPosition(presentPanel));
+ refreshLook();
+ return;
+ }
+ if(arrayPanel[getPosition(presentPanel)].Equals(panel2))
+ {
+ m_db.setName(textDbName.Text.ToString());
+ m_db.setOwner(textOwner.Text.ToString());
+ presentPanel=arrayPanel[getPosition(presentPanel)+1];
+ //presentPanel
+ setBtnPanProperty(getPosition(presentPanel));
+ prepareNodeAssignmentPanel();
+ refreshLook();
+
+ return;
+ }
+ if(arrayPanel[getPosition(presentPanel)].Equals(panel3))
+ {
+ prepareNodeConfigurationPanel();
+ presentPanel=arrayPanel[getPosition(presentPanel)+1];
+ setBtnPanProperty(getPosition(presentPanel));
+ refreshLook();
+ return;
+ }
+ if(arrayPanel[getPosition(presentPanel)].Equals(panel4))
+ {
+ nextEnabled=true;
+ finishEnabled=true;
+ backEnabled=true;
+ cancelEnabled=true;
+ presentPanel=arrayPanel[getPosition(presentPanel)+1];
+ setBtnPanProperty(getPosition(presentPanel));
+ refreshLook();
+ return;
+ }
+ if(arrayPanel[getPosition(presentPanel)].Equals(panel5))
+ {
+ generateInitConfig();
+ presentPanel=arrayPanel[getPosition(presentPanel)+1];
+ setBtnPanProperty(getPosition(presentPanel));
+ refreshLook();
+ return;
+ }
+
+ if(arrayPanel[getPosition(presentPanel)].Equals(panel6))
+ {
+ // presentPanel=arrayPanel[getPosition(presentPanel)+1];
+ setBtnPanProperty(getPosition(presentPanel));
+ refreshLook();
+ return;
+ }
+ /*else
+ {
+ presentPanel=arrayPanel[getPosition(presentPanel)+1];
+ setBtnPanProperty(getPosition(presentPanel));
+ updateListViews();
+ refreshLook();
+ }*/
+ }
+
+
+ private void btnback_Click(object sender, System.EventArgs e)
+ {
+ presentPanel=arrayPanel[getPosition(presentPanel)-1];
+ setBtnPanProperty(getPosition(presentPanel));
+ m_noOfConfiguredNodes=0;
+ m_noOfConfiguredNodes=0;
+ m_noOfConfiguredMgmt=0;
+ m_noOfConfiguredNdb=0;
+ m_noOfConfiguredApi=0;
+ m_bNdb=false;
+ m_bMgmt=false;
+ refreshLook();
+ }
+
+
+ private void btnCancel_Click(object sender, System.EventArgs e)
+ {
+ m_db.removeAllProcesses();
+ this.Dispose(true);
+ }
+
+
+
+
+ private void radioBtnYes_Click(object sender, System.EventArgs e)
+ {
+ if(radioBtnNo.Checked.Equals(false))
+ {
+ if(radioBtnYes.Checked.Equals(true))
+ radioBtnYes.Checked=false;
+ else
+ {
+ radioBtnYes.Checked=true;
+ this.btnNext.Enabled=true;
+ }
+
+ }
+ if(radioBtnNo.Checked.Equals(true))
+ {
+ radioBtnNo.Checked=false;
+ radioBtnYes.Checked=true;
+ buttonComputerAdd.Enabled=false;
+ this.btnNext.Enabled=true;
+ }
+
+ }
+
+ private void radioBtnNo_Click(object sender, System.EventArgs e)
+ {
+ if(radioBtnYes.Checked.Equals(false))
+ {
+ if(radioBtnNo.Checked.Equals(true))
+ {
+ radioBtnNo.Checked=false;
+ buttonComputerAdd.Enabled=false;
+ }
+ else
+ {
+ radioBtnNo.Checked=true;
+ buttonComputerAdd.Enabled=true;
+ this.btnNext.Enabled=false;
+ }
+
+ }
+ if(radioBtnYes.Checked.Equals(true))
+ {
+ radioBtnYes.Checked=false;
+ radioBtnNo.Checked=true;
+ buttonComputerAdd.Enabled=true;
+ this.btnNext.Enabled=false;
+ }
+ }
+
+ private void buttonComputerAdd_Click(object sender, System.EventArgs e)
+ {
+ if(getPosition(presentPanel)==0)
+ {
+ if(radioBtnNo.Checked.Equals(true))
+ {
+ ComputerAddDialog cad = new ComputerAddDialog(mgmt);
+ cad.ShowDialog();
+ }
+ }
+ }
+
+ private void PanelWizard_Activated(object sender, System.EventArgs e)
+ {
+ updateComputers();
+ }
+
+ private void updateComputers()
+ {
+ ArrayList list = mgmt.getComputerCollection();
+ this.listBoxComputers.BeginUpdate();
+ this.listBoxComputers.Items.Clear();
+ foreach(Computer c in list)
+ {
+ this.listBoxComputers.Items.Add(c.getName());
+ }
+ if(listBoxComputers.Items.Count > 0)
+ {
+ btnNext.Enabled=true;
+ }
+ this.listBoxComputers.EndUpdate();
+ this.listBoxComputers.Refresh();
+ }
+
+
+ private void tvComputer_AfterSelect(object sender, System.Windows.Forms.TreeViewEventArgs e)
+ {
+ tvComputer.SelectedNode.Expand();
+
+ }
+
+ private void tvComputer_DragDrop(object sender, System.Windows.Forms.DragEventArgs e)
+ {
+
+ }
+
+ private void tvComputer_MouseLeave(object sender, System.EventArgs e)
+ {
+
+ }
+
+ private void tvComputer_MouseDown(object sender, System.Windows.Forms.MouseEventArgs e)
+ {
+ TreeNode prevNode = tvComputer.SelectedNode;
+ if(prevNode!=null)
+ {
+ prevNode.BackColor=Color.White;
+ }
+ TreeNode node = tvComputer.GetNodeAt(e.X,e.Y);
+ if(node==null)
+ {
+ return;
+ }
+
+ tvComputer.SelectedNode=node;
+ tvComputer.SelectedNode.BackColor=Color.LightGray;
+
+ }
+
+ private void btnTransferNodeToComp_Click(object sender, System.EventArgs e)
+ {
+
+ if(tvComputer.SelectedNode==null)
+ return;
+ if(lvNode.SelectedItems.Equals(null))
+ return;
+ int itemCount=lvNode.SelectedItems.Count;
+ lvNode.BeginUpdate();
+ tvComputer.BeginUpdate();
+ for(int i=0;i < itemCount;i++)
+ {
+ tvComputer.SelectedNode.Nodes.Add(lvNode.SelectedItems[i].Text.ToString());
+ }
+
+ for(int i=0;i < itemCount;i++)
+ {
+ lvNode.Items.RemoveAt(lvNode.SelectedIndices[0]);
+
+ }
+ if(lvNode.Items.Count.Equals(0))
+ btnNext.Enabled=true;
+ else
+ btnNext.Enabled=false;
+ tvComputer.SelectedNode.Expand();
+ lvNode.EndUpdate();
+ tvComputer.EndUpdate();
+ }
+
+ private void lvNode_SelectedIndexChanged(object sender, System.EventArgs e)
+ {
+ }
+
+ private void prepareNodeAssignmentPanel()
+ {
+ ArrayList computers = mgmt.getComputerCollection();
+ m_nNDB=Convert.ToInt32(comboNDB.SelectedItem.ToString());
+ m_nAPI=Convert.ToInt32(comboAPI.SelectedItem.ToString());
+ m_nMGM=Convert.ToInt32(comboMGM.SelectedItem.ToString());
+
+ lvNode.Items.Clear();
+ tvComputer.Nodes.Clear();
+ for (int i=1;i<=m_nMGM;i++)
+ lvNode.Items.Add("mgm."+i);
+
+ for (int i=m_nMGM+1;i<=(m_nNDB+m_nMGM);i++)
+ lvNode.Items.Add("ndb."+i);
+
+ for (int i=m_nMGM+m_nNDB+1;i<=(m_nNDB+m_nMGM+m_nAPI);i++)
+ lvNode.Items.Add("api."+i);
+
+ foreach(Computer c in computers)
+ {
+ if(c.getStatus() == Computer.Status.Connected)
+ tvComputer.Nodes.Add(c.getName());
+ }
+
+ }
+ private void prepareNodeConfigurationPanel()
+ {
+ Computer c;
+ for(int i=0;i<tvComputer.Nodes.Count;i++)
+ {
+ c=mgmt.getComputer(tvComputer.Nodes[i].Text.ToString());
+ for(int j=0; j < tvComputer.Nodes[i].Nodes.Count;j++)
+ {
+ m_db.addProcess(new Process(tvComputer.Nodes[i].Nodes[j].Text.ToString(),m_db.getOwner(),m_db.getName(),c));
+ c.addProcess(m_db.getProcessByName(tvComputer.Nodes[i].Nodes[j].Text.ToString()));
+ }
+ }
+ }
+
+ private void updateListViews()
+ {/*
+ lvConfig.Items.Clear();
+ ArrayList processes = m_db.getProcesses();
+ string [] processcols= new string[5];
+ foreach (Process process in processes)
+ {
+ processcols[0]=process.getName();
+ processcols[1]=process.getComputer().getName();
+ processcols[2]=process.getPath();
+ processcols[3]="";
+ processcols[4]="";
+
+ ListViewItem lvc= new ListViewItem(processcols);
+
+
+ lvConfig.Items.Add(lvc);
+ }
+ lvConfig.EndUpdate();
+ */
+ }
+
+ private void btnConfigure_Click(object sender, System.EventArgs e)
+ {
+
+ }
+
+ private void textDbName_TextChanged(object sender, System.EventArgs e)
+ {
+ if(textOwner.TextLength>0 && textDbName.TextLength > 0)
+ nextEnabled=true;
+ else
+ nextEnabled=false;
+
+ refreshLook();
+
+ }
+
+ private void checkBoxLater_CheckedChanged(object sender, System.EventArgs e)
+ {
+ if(checkBoxLater.Checked.Equals(true))
+ {
+ this.finishEnabled=true;
+ this.nextEnabled=false;
+ }
+ else
+ {
+ this.finishEnabled=false;
+ this.nextEnabled=true;
+ }
+ this.refreshLook();
+ }
+
+ private void btnFinish_Click(object sender, System.EventArgs e)
+ {
+ mgmt.AddDatabase(this.m_db);
+
+ if(radioStartYes.Checked==true)
+ startDatabase();
+ this.Dispose();
+ }
+
+ private void panel4_Paint(object sender, System.Windows.Forms.PaintEventArgs e)
+ {
+
+ // Point location= new Point(8,40);
+ // Size s= new Size(panel4.Size.Width-8,panel4.Size.Height-120);
+ // lvConfig.Location=location;
+ // lvConfig.Size=s;
+
+
+ }
+
+ private void configureMgmt()
+ {
+ //clear old
+ textOther.Text="";
+ textArgs.Text="";
+ textCwd.Text="";
+ textPath.Text="";
+
+ textPath.Clear();
+ textEnv.Clear();
+ textOther.Clear();
+ textCwd.Clear();
+ textArgs.Clear();
+
+ textPath.ClearUndo();
+ textEnv.ClearUndo();
+ textOther.ClearUndo();
+ textCwd.ClearUndo();
+ textArgs.ClearUndo();
+
+
+ textOther.Enabled=true;
+ textArgs.Enabled=true;
+ textCwd.Enabled=true;
+ textPath.Enabled=true;
+
+ textPath.TabStop=true;
+ textOther.TabStop=true;
+ textArgs.TabStop=true;
+ textCwd.TabStop=true;
+ textEnv.TabStop=true;
+
+ labelTitle.Text="Mgmtsrvr configuration";
+ labelPath.Text="Path to mgmtsrvr binary:";
+ labelArgs.Text="Arguments to mgmtsrvr:";
+ labelOther.Text="Mgmtsrvr port (-p X):";
+
+ //get new
+ String process="mgm." + Convert.ToString(m_noOfConfiguredMgmt+1);
+ Process mgmt=m_db.getProcessByName(process);
+ textComputer.Text=mgmt.getComputer().getName();
+ textName.Text=mgmt.getName().ToString();
+ textDatabase.Text=mgmt.getDatabase().ToString();
+ m_mgmHost=mgmt.getComputer().getName();
+ textPath.Focus();
+ }
+ private void configureApi()
+ {
+ checkBoxReuse.Text="Use the same configuration for ALL API nodes?";
+ if(m_nAPI > 1)
+ {
+ checkBoxReuse.Visible=true;
+ checkBoxReuse.Enabled=true;
+
+ }
+ else
+ {
+ checkBoxReuse.Enabled=false;
+ checkBoxReuse.Visible=true;
+ }
+
+ // clear previous and get a new api
+
+ textOther.Text="";
+ textArgs.Text="";
+ //textCwd.Text="";
+ //textPath.Text="";
+ //get new api
+ textOther.Enabled=false;
+ textArgs.Enabled=true;
+ labelTitle.Text="API node configuration";
+ labelPath.Text="Path to api binary:";
+ labelArgs.Text="Arguments to api:";
+ labelOther.Text="NDB_CONNECTSTRING";
+ String process="api." + Convert.ToString(m_noOfConfiguredApi+m_nMGM+m_nNDB+1);
+ Process api=m_db.getProcessByName(process);
+ textComputer.Text=api.getComputer().getName();
+ textName.Text=api.getName().ToString();
+ textOther.Text="nodeid=" + Convert.ToString(m_noOfConfiguredApi+m_nMGM+m_nNDB+1) + ";host="+this.m_mgmHost + ":" + this.m_mgmPort;
+ textDatabase.Text=api.getDatabase().ToString();
+ textPath.Focus();
+ }
+
+ private void configureNdb()
+ {
+
+
+ checkBoxReuse.Text="Use the same configuration for ALL NDB nodes?";
+
+
+ if(this.m_nNDB > 1)
+ {
+ checkBoxReuse.Visible=true;
+ checkBoxReuse.Enabled=true;
+
+ }
+ else
+ {
+ checkBoxReuse.Enabled=false;
+ checkBoxReuse.Visible=true;
+ }
+
+
+
+ labelPath.Text="Path to ndb binary:";
+ labelArgs.Text="Arguments to ndb:";
+
+ // clear previous and get a new ndb
+
+ labelOther.Text="NDB_CONNECTSTRING";
+ textArgs.Text="-i";
+ textOther.Enabled=false;
+ textArgs.Enabled=false;
+
+ textPath.TabStop=true;
+ textEnv.TabStop=true;
+ textOther.TabStop=false;
+ textArgs.TabStop=false;
+ textCwd.TabStop=true;
+
+ //textCwd.Text="";
+ //textPath.Text="";
+ //get new
+
+ String process="ndb." + Convert.ToString(m_noOfConfiguredNdb+m_nMGM+1);
+ textOther.Text="nodeid=" + Convert.ToString(m_noOfConfiguredNdb+m_nMGM+1) + ";host="+this.m_mgmHost + ":" + this.m_mgmPort;
+ Process ndb=m_db.getProcessByName(process);
+ textComputer.Text=ndb.getComputer().getName();
+ textName.Text=ndb.getName().ToString();
+ textDatabase.Text=ndb.getDatabase().ToString();
+ textPath.Focus();
+ }
+
+
+ public void saveMgm()
+ {
+ String process="mgm." + Convert.ToString(m_noOfConfiguredMgmt+1);
+ Process mgmt=m_db.getProcessByName(process);
+ mgmt.setOther(textOther.Text.ToString());
+ mgmt.setEnv(textEnv.Text.ToString());
+ m_mgmPort = textOther.Text.ToString();
+ try
+ {
+ m_db.setMgmtPort(Convert.ToInt32(m_mgmPort));
+ }
+ catch(Exception e)
+ {
+ MessageBox.Show("Port number must be numeric!!!", "Error",MessageBoxButtons.OK);
+ this.configureMgmt();
+ return;
+ }
+ mgmt.setPath(textPath.Text.ToString());
+ mgmt.setCwd(textCwd.Text.ToString());
+ mgmt.setProcessType("permanent");
+ mgmt.setArgs("-i initconfig.txt");
+ mgmt.setConnectString("nodeid=" + Convert.ToString(m_noOfConfiguredMgmt+1)+";host="+m_mgmHost+":" + m_mgmPort);
+ this.m_noOfConfiguredMgmt++;
+ }
+
+ public void saveApi()
+ {
+ if(checkBoxReuse.Checked)
+ {
+ for(;m_noOfConfiguredApi<m_nAPI;m_noOfConfiguredApi++)
+ {
+ String process="api." + Convert.ToString(m_noOfConfiguredApi+m_nMGM+m_nNDB+1);
+ Process api=m_db.getProcessByName(process);
+ textName.Text=process;
+ api.setPath(textPath.Text.ToString());
+ api.setArgs(textArgs.Text.ToString());
+ api.setCwd(textCwd.Text.ToString());
+ api.setEnv(textEnv.Text.ToString());
+ api.setConnectString("nodeid=" + Convert.ToString(m_noOfConfiguredApi+m_nNDB+m_nMGM+1)+";host="+m_mgmHost+":" + m_mgmPort);
+ api.setProcessType("permanent");
+ }
+
+ }
+ else
+ {
+ String process="api." + Convert.ToString(m_noOfConfiguredApi+m_nMGM+m_nNDB+1);
+ Process api=m_db.getProcessByName(process);
+ api.setPath(textPath.Text.ToString());
+ api.setCwd(textCwd.Text.ToString());
+ api.setEnv(textEnv.Text.ToString());
+ api.setConnectString("nodeid=" + Convert.ToString(m_noOfConfiguredApi+m_nNDB+m_nMGM+1)+";host="+m_mgmHost+":" + m_mgmPort);
+ api.setArgs(textArgs.Text.ToString());
+ api.setProcessType("permanent");
+ this.m_noOfConfiguredApi++;
+ }
+ }
+
+ public void saveNdb()
+ {
+
+ if(checkBoxReuse.Checked)
+ {
+ for(;m_noOfConfiguredNdb<m_nNDB;m_noOfConfiguredNdb++)
+ {
+ String process="ndb." + Convert.ToString(m_noOfConfiguredNdb+m_nMGM+1);
+ Process ndb=m_db.getProcessByName(process);
+ ndb.setConnectString("nodeid=" + Convert.ToString(m_noOfConfiguredNdb+m_nMGM+1)+";host="+m_mgmHost+":" + m_mgmPort);
+ ndb.setPath(textPath.Text.ToString());
+ ndb.setArgs(textArgs.Text.ToString());
+ ndb.setEnv(textEnv.Text.ToString());
+ ndb.setCwd(textCwd.Text.ToString());
+ ndb.setProcessType("permanent");
+ }
+ checkBoxReuse.Checked=false;
+ return;
+ }
+ else
+ {
+ String process="ndb." + Convert.ToString(m_noOfConfiguredNdb+m_nMGM+1);
+ Process ndb=m_db.getProcessByName(process);
+ ndb.setConnectString("nodeid=" + Convert.ToString(m_noOfConfiguredNdb+m_nMGM+1)+";host="+m_mgmHost+":" + m_mgmPort);
+ ndb.setPath(textPath.Text.ToString());
+ ndb.setCwd(textCwd.Text.ToString());
+ ndb.setArgs(textArgs.Text.ToString());
+ ndb.setEnv(textEnv.Text.ToString());
+ ndb.setProcessType("permanent");
+ m_noOfConfiguredNdb++;
+ }
+
+ }
+
+
+ private void panel5_Paint(object sender, System.Windows.Forms.PaintEventArgs e)
+ {
+ nextEnabled=false;
+ buttonSave.Enabled=true;
+ checkBoxReuse.Visible=false;
+ refreshLook();
+ configureMgmt();
+ }
+
+ private void buttonSave_Click(object sender, System.EventArgs e)
+ {
+ Process p = m_db.getProcessByName(textName.Text.ToString());
+
+ if(textOther.Text.ToString().Equals(""))
+ {
+ if(textName.Text.StartsWith("mgm"))
+ {
+ MessageBox.Show("You have to specify a port.","Warning",MessageBoxButtons.OK);
+ return;
+ }
+ if(textName.Text.StartsWith("ndb"))
+ {
+ MessageBox.Show("You have to specify a filesystem path.","Warning",MessageBoxButtons.OK);
+ return;
+ }
+
+ }
+
+ if(textPath.Text.ToString().Equals(""))
+ {
+ if(textName.Text.StartsWith("mgm"))
+ {
+ MessageBox.Show("You have to specify the path to the mgmtsrvr.","Warning",MessageBoxButtons.OK);
+ return;
+ }
+ if(textName.Text.StartsWith("ndb"))
+ {
+ MessageBox.Show("You have to specify the path to ndb.","Warning",MessageBoxButtons.OK);
+ return;
+ }
+ }
+
+ if(textArgs.Text.ToString().Equals(""))
+ {
+ if(textName.Text.StartsWith("mgm"))
+ {
+ MessageBox.Show("You have to specify the arguments to the mgmtsrvr.","Warning",MessageBoxButtons.OK);
+ return;
+ }
+ }
+
+ if(textCwd.Text.ToString().Equals(""))
+ {
+ if(textCwd.Text.StartsWith("mgm"))
+ {
+ MessageBox.Show("You have to specify the current working directory for the mgmtsrvr.","Warning",MessageBoxButtons.OK);
+ return;
+ }
+ }
+
+
+ /*
+ * INPUT IS FINE AT THIS POINT
+ * Everything needed for respective process is ok
+ * */
+
+ if(textName.Text.StartsWith("mgm"))
+ {
+ //MessageBox.Show(textOther.Text.ToString());
+ saveMgm();
+
+ }
+
+ if(textName.Text.StartsWith("ndb"))
+ {
+ saveNdb();
+
+ }
+
+ if(textName.Text.StartsWith("api"))
+ {
+ saveApi();
+
+ }
+
+ if(m_noOfConfiguredMgmt < m_nMGM)
+ {
+ //load another Mgmt
+ labelTitle.Text="Mgmtsrvr configuration";
+ configureMgmt();
+ }
+ else
+ {
+ m_bMgmt=true;
+ }
+
+ if(m_bMgmt)
+ {
+ labelTitle.Text="NDB node configuration";
+ if(m_noOfConfiguredNdb < m_nNDB)
+ {
+ configureNdb();
+ }
+ else
+ m_bNdb=true;
+ }
+
+ if(m_bNdb && m_bMgmt)
+ {
+ labelTitle.Text="API node configuration";
+ if(m_noOfConfiguredApi < m_nAPI)
+ configureApi();
+ else
+ {
+ nextEnabled=true;
+ buttonSave.Enabled=false;
+ refreshLook();
+ }
+ }
+
+ }
+
+ private void listBoxComputers_SelectedIndexChanged(object sender, System.EventArgs e)
+ {
+
+ }
+
+ private void panel1_Paint(object sender, System.Windows.Forms.PaintEventArgs e)
+ {
+ updateComputers();
+ }
+
+ private void radioYes_CheckedChanged(object sender, System.EventArgs e)
+ {
+ if(radioYes.Checked==true)
+ {
+ radioNo.Checked=false;
+ }
+ if(radioYes.Checked==false)
+ {
+ radioNo.Checked=true;
+ }
+
+ }
+
+ private void radioNo_CheckedChanged(object sender, System.EventArgs e)
+ {
+ if(radioNo.Checked==true)
+ {
+ radioYes.Checked=false;
+ }
+ if(radioNo.Checked==false)
+ {
+ radioYes.Checked=true;
+ }
+ }
+
+ private void radioStartYes_CheckedChanged(object sender, System.EventArgs e)
+ {
+
+ if(radioStartYes.Checked==true)
+ {
+ radioStartNo.Checked=false;
+ }
+ if(radioStartYes.Checked==false)
+ {
+ radioStartNo.Checked=true;
+ }
+ finishEnabled=true;
+ refreshLook();
+ }
+
+ private void radioStartNo_CheckedChanged(object sender, System.EventArgs e)
+ {
+ if(radioStartNo.Checked==true)
+ {
+ radioStartYes.Checked=false;
+ }
+ if(radioStartNo.Checked==false)
+ {
+ radioStartYes.Checked=true;
+ }
+ finishEnabled=true;
+ refreshLook();
+ }
+
+
+
+
+ public void startDatabase()
+ {
+ startDatabaseDlg x = new startDatabaseDlg(this.m_db);
+
+
+ x.ShowDialog();
+
+
+ }
+
+
+ public void generateInitConfig()
+ {
+ MessageBox.Show("Generate initconfig.txt");
+ }
+
+ private void label11_Click(object sender, System.EventArgs e)
+ {
+
+ }
+
+ private void textOwner_TextChanged(object sender, System.EventArgs e)
+ {
+ if(textDbName.TextLength > 0 && textOwner.TextLength > 0)
+ nextEnabled=true;
+ else
+ nextEnabled=false;
+
+ refreshLook();
+ }
+
+ private void panel2_Paint(object sender, System.Windows.Forms.PaintEventArgs e)
+ {
+ textOwner.Text=System.Environment.UserName;
+ this.Validate();
+ if(textDbName.TextLength > 0 && textOwner.TextLength>0)
+ {
+ nextEnabled=true;
+ }
+ else
+ {
+ nextEnabled=false;
+ }
+ refreshLook();
+ }
+
+ private void textPath_TextChanged(object sender, System.EventArgs e)
+ {
+ try
+ {
+
+ }
+ catch (Exception exc)
+ {
+ MessageBox.Show(exc.ToString());
+ }
+ }
+
+ private void panel2_Validating(object sender, System.ComponentModel.CancelEventArgs e)
+ {
+ if(textOwner.TextLength>0 && textDbName.TextLength > 0)
+ nextEnabled=true;
+ else
+ nextEnabled=false;
+ }
+
+
+
+
+
+
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/Process.cs b/ndb/src/cw/cpcc-win32/csharp/Process.cs
new file mode 100644
index 00000000000..c1ee1b2fe9e
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/Process.cs
@@ -0,0 +1,144 @@
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+using System.Data;
+
+namespace NDB_CPC
+{
+ /// <summary>
+ /// Summary description for Process.
+ /// </summary>
+ public class Process
+ {
+ public enum Status {Running, Stopped, Unknown}
+ private string m_id;
+ protected string m_name;
+ private Status m_status;
+ private Computer m_computer;
+ private string m_owner;
+ private string m_cwd;
+ private string m_type;
+ private string m_path;
+ private string m_other;
+ private string m_args;
+ private string m_env;
+ private string m_database;
+ private string m_connectString;
+ private bool m_defined;
+ public Process( string name,
+ string owner, string database,
+ Computer computer)
+ {
+ m_name=name;
+ m_owner=owner;
+ m_computer=computer;
+ m_status=Status.Unknown;
+ m_database=database;
+ m_defined=false;
+ m_path="";
+ m_cwd="";
+ m_args="";
+ m_other="";
+ }
+ public Process()
+ {
+
+ }
+ public Process(string id)
+ {
+ m_id=id;
+ }
+
+ public Process( string name,
+ string database,
+ Computer computer)
+ {
+ m_name=name;
+ m_computer=computer;
+ m_status=Status.Unknown;
+ m_database=database;
+ m_defined=false;
+ }
+
+ public Process( string name,
+ Computer computer)
+ {
+ m_name=name;
+ m_computer=computer;
+ m_status=Status.Unknown;
+ m_defined=false;
+ }
+
+
+ public string getStatusString()
+ {
+ if(m_status.Equals(Status.Running))
+ return "Running";
+ if(m_status.Equals(Status.Stopped))
+ return "Stopped";
+ return "Unknown";
+ }
+
+ public Computer getComputer() {return m_computer;}
+ public string getName() {return m_name;}
+ public string getDatabase() {return m_database;}
+ public string getOwner() {return m_owner;}
+ public string getId() {return m_id;}
+ public void setId(string id) {m_id=id;}
+
+ public void setCwd(string cwd) {m_cwd=cwd;}
+ public void setPath(string path) {m_path=path;}
+ public void setArgs(string args) {m_args=args;}
+ public void setOther(string other) {m_other=other;}
+ public void setEnv(string env) {m_env=env;}
+ public void setName(string name) {m_name=name;}
+ public void setOwner(string owner) {m_owner=owner;}
+ public void setDatabase(string db) {m_database=db;}
+ public void setComputer(Computer c) {m_computer=c;}
+
+
+ public string getCwd() {return m_cwd;}
+ public string getPath() {return m_path;}
+ public string getArgs() {return m_args;}
+ public string getOther() {return m_other;}
+ public string getEnv() {return m_env;}
+
+ public bool isDefined() {return m_defined;}
+ public void setDefined(bool defined)
+ {
+ m_defined=defined;
+ }
+
+ public Status getStatus()
+ {
+ return m_status;
+ }
+
+ public void setConnectString(string cs)
+ {
+ m_connectString=cs;
+ }
+
+ public string getConnectString()
+ {
+ return m_connectString;
+ }
+ public void setStatus(Status status)
+ {
+ m_status=status;
+ }
+
+
+ public void setProcessType(string type)
+ {
+ m_type=type;
+ }
+ public string getProcessType()
+ {
+ return m_type;
+ }
+
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/ProcessDefineDialog.cs b/ndb/src/cw/cpcc-win32/csharp/ProcessDefineDialog.cs
new file mode 100644
index 00000000000..581b8383e7c
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/ProcessDefineDialog.cs
@@ -0,0 +1,435 @@
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+
+namespace NDB_CPC
+{
+ /// <summary>
+ /// Summary description for ProcessDefineDialog.
+ /// </summary>
+ public class ProcessDefineDialog : System.Windows.Forms.Form
+ {
+ private System.Windows.Forms.ComboBox comboComputer;
+ private System.Windows.Forms.Label label1;
+ private System.Windows.Forms.Label label2;
+ private System.Windows.Forms.Label label3;
+ private System.Windows.Forms.Label label4;
+ private System.Windows.Forms.Label label5;
+ private System.Windows.Forms.Label label6;
+ private System.Windows.Forms.Label label7;
+ private System.Windows.Forms.Label label8;
+ private System.Windows.Forms.Label label9;
+ private System.Windows.Forms.TextBox textProcessName;
+ private System.Windows.Forms.TextBox textProcessGroup;
+ private System.Windows.Forms.TextBox textProcessEnv;
+ private System.Windows.Forms.TextBox textProcessPath;
+ private System.Windows.Forms.TextBox textProcessArgs;
+ private System.Windows.Forms.TextBox textProcessCWD;
+ private System.Windows.Forms.TextBox textProcessOwner;
+ private System.Windows.Forms.ComboBox comboType;
+ private System.Windows.Forms.Label label10;
+ private System.Windows.Forms.Label label11;
+ private System.Windows.Forms.Label label12;
+ private System.Windows.Forms.Label label13;
+ private System.Windows.Forms.Label label15;
+ private System.Windows.Forms.Label label16;
+ private System.Windows.Forms.Label label14;
+ private System.Windows.Forms.Label label17;
+ private System.Windows.Forms.Label label18;
+ private System.Windows.Forms.Button btnAdd;
+ private System.Windows.Forms.Button btnCancel;
+ /// <summary>
+ /// Required designer variable.
+ /// </summary>
+ private System.ComponentModel.Container components = null;
+ private ComputerMgmt c_mgmt;
+ private string m_selComputer;
+ public ProcessDefineDialog(ComputerMgmt mgmt, string computer)
+ {
+
+ // Required for Windows Form Designer support
+ //
+ InitializeComponent();
+
+ //
+ // TODO: Add any constructor code after InitializeComponent call
+ //
+ m_selComputer =computer; //the selected computer in the TreeView
+ c_mgmt=mgmt;
+ }
+
+ /// <summary>
+ /// Clean up any resources being used.
+ /// </summary>
+ protected override void Dispose( bool disposing )
+ {
+ if( disposing )
+ {
+ if(components != null)
+ {
+ components.Dispose();
+ }
+ }
+ base.Dispose( disposing );
+ }
+
+ #region Windows Form Designer generated code
+ /// <summary>
+ /// Required method for Designer support - do not modify
+ /// the contents of this method with the code editor.
+ /// </summary>
+ private void InitializeComponent()
+ {
+ this.comboComputer = new System.Windows.Forms.ComboBox();
+ this.label1 = new System.Windows.Forms.Label();
+ this.label2 = new System.Windows.Forms.Label();
+ this.label3 = new System.Windows.Forms.Label();
+ this.label4 = new System.Windows.Forms.Label();
+ this.label5 = new System.Windows.Forms.Label();
+ this.label6 = new System.Windows.Forms.Label();
+ this.label7 = new System.Windows.Forms.Label();
+ this.label8 = new System.Windows.Forms.Label();
+ this.label9 = new System.Windows.Forms.Label();
+ this.textProcessName = new System.Windows.Forms.TextBox();
+ this.textProcessGroup = new System.Windows.Forms.TextBox();
+ this.textProcessEnv = new System.Windows.Forms.TextBox();
+ this.textProcessPath = new System.Windows.Forms.TextBox();
+ this.textProcessArgs = new System.Windows.Forms.TextBox();
+ this.textProcessCWD = new System.Windows.Forms.TextBox();
+ this.textProcessOwner = new System.Windows.Forms.TextBox();
+ this.comboType = new System.Windows.Forms.ComboBox();
+ this.label10 = new System.Windows.Forms.Label();
+ this.label11 = new System.Windows.Forms.Label();
+ this.label12 = new System.Windows.Forms.Label();
+ this.label13 = new System.Windows.Forms.Label();
+ this.label15 = new System.Windows.Forms.Label();
+ this.label16 = new System.Windows.Forms.Label();
+ this.label14 = new System.Windows.Forms.Label();
+ this.label17 = new System.Windows.Forms.Label();
+ this.label18 = new System.Windows.Forms.Label();
+ this.btnAdd = new System.Windows.Forms.Button();
+ this.btnCancel = new System.Windows.Forms.Button();
+ this.SuspendLayout();
+ //
+ // comboComputer
+ //
+ this.comboComputer.ItemHeight = 13;
+ this.comboComputer.Location = new System.Drawing.Point(152, 24);
+ this.comboComputer.Name = "comboComputer";
+ this.comboComputer.Size = new System.Drawing.Size(112, 21);
+ this.comboComputer.TabIndex = 0;
+ //
+ // label1
+ //
+ this.label1.Location = new System.Drawing.Point(24, 24);
+ this.label1.Name = "label1";
+ this.label1.Size = new System.Drawing.Size(64, 24);
+ this.label1.TabIndex = 1;
+ this.label1.Text = "Computer:";
+ //
+ // label2
+ //
+ this.label2.Location = new System.Drawing.Point(24, 48);
+ this.label2.Name = "label2";
+ this.label2.Size = new System.Drawing.Size(88, 24);
+ this.label2.TabIndex = 2;
+ this.label2.Text = "Process name:";
+ //
+ // label3
+ //
+ this.label3.Location = new System.Drawing.Point(24, 72);
+ this.label3.Name = "label3";
+ this.label3.Size = new System.Drawing.Size(88, 24);
+ this.label3.TabIndex = 3;
+ this.label3.Text = "Group:";
+ //
+ // label4
+ //
+ this.label4.Location = new System.Drawing.Point(24, 96);
+ this.label4.Name = "label4";
+ this.label4.Size = new System.Drawing.Size(88, 24);
+ this.label4.TabIndex = 4;
+ this.label4.Text = "Env. variables:";
+ //
+ // label5
+ //
+ this.label5.Location = new System.Drawing.Point(24, 120);
+ this.label5.Name = "label5";
+ this.label5.Size = new System.Drawing.Size(88, 24);
+ this.label5.TabIndex = 5;
+ this.label5.Text = "Path to binary:";
+ //
+ // label6
+ //
+ this.label6.Location = new System.Drawing.Point(24, 144);
+ this.label6.Name = "label6";
+ this.label6.Size = new System.Drawing.Size(112, 24);
+ this.label6.TabIndex = 6;
+ this.label6.Text = "Arguments to binary:";
+ //
+ // label7
+ //
+ this.label7.Location = new System.Drawing.Point(24, 168);
+ this.label7.Name = "label7";
+ this.label7.Size = new System.Drawing.Size(112, 24);
+ this.label7.TabIndex = 7;
+ this.label7.Text = "Type of process:";
+ //
+ // label8
+ //
+ this.label8.Location = new System.Drawing.Point(24, 192);
+ this.label8.Name = "label8";
+ this.label8.Size = new System.Drawing.Size(112, 24);
+ this.label8.TabIndex = 8;
+ this.label8.Text = "Current working dir.:";
+ //
+ // label9
+ //
+ this.label9.Location = new System.Drawing.Point(24, 216);
+ this.label9.Name = "label9";
+ this.label9.Size = new System.Drawing.Size(112, 24);
+ this.label9.TabIndex = 9;
+ this.label9.Text = "Owner:";
+ //
+ // textProcessName
+ //
+ this.textProcessName.Location = new System.Drawing.Point(152, 48);
+ this.textProcessName.Name = "textProcessName";
+ this.textProcessName.Size = new System.Drawing.Size(112, 20);
+ this.textProcessName.TabIndex = 1;
+ this.textProcessName.Text = "";
+ //
+ // textProcessGroup
+ //
+ this.textProcessGroup.Location = new System.Drawing.Point(152, 72);
+ this.textProcessGroup.Name = "textProcessGroup";
+ this.textProcessGroup.Size = new System.Drawing.Size(112, 20);
+ this.textProcessGroup.TabIndex = 2;
+ this.textProcessGroup.Text = "";
+ //
+ // textProcessEnv
+ //
+ this.textProcessEnv.Location = new System.Drawing.Point(152, 96);
+ this.textProcessEnv.Name = "textProcessEnv";
+ this.textProcessEnv.Size = new System.Drawing.Size(112, 20);
+ this.textProcessEnv.TabIndex = 3;
+ this.textProcessEnv.Text = "";
+ //
+ // textProcessPath
+ //
+ this.textProcessPath.Location = new System.Drawing.Point(152, 120);
+ this.textProcessPath.Name = "textProcessPath";
+ this.textProcessPath.Size = new System.Drawing.Size(112, 20);
+ this.textProcessPath.TabIndex = 4;
+ this.textProcessPath.Text = "";
+ //
+ // textProcessArgs
+ //
+ this.textProcessArgs.Location = new System.Drawing.Point(152, 144);
+ this.textProcessArgs.Name = "textProcessArgs";
+ this.textProcessArgs.Size = new System.Drawing.Size(112, 20);
+ this.textProcessArgs.TabIndex = 5;
+ this.textProcessArgs.Text = "";
+ //
+ // textProcessCWD
+ //
+ this.textProcessCWD.Location = new System.Drawing.Point(152, 192);
+ this.textProcessCWD.Name = "textProcessCWD";
+ this.textProcessCWD.Size = new System.Drawing.Size(112, 20);
+ this.textProcessCWD.TabIndex = 7;
+ this.textProcessCWD.Text = "";
+ //
+ // textProcessOwner
+ //
+ this.textProcessOwner.Location = new System.Drawing.Point(152, 216);
+ this.textProcessOwner.Name = "textProcessOwner";
+ this.textProcessOwner.Size = new System.Drawing.Size(112, 20);
+ this.textProcessOwner.TabIndex = 8;
+ this.textProcessOwner.Text = "";
+ //
+ // comboType
+ //
+ this.comboType.ItemHeight = 13;
+ this.comboType.Items.AddRange(new object[] {
+ "Permanent",
+ "Interactive"});
+ this.comboType.Location = new System.Drawing.Point(152, 168);
+ this.comboType.Name = "comboType";
+ this.comboType.Size = new System.Drawing.Size(112, 21);
+ this.comboType.TabIndex = 6;
+ //
+ // label10
+ //
+ this.label10.Location = new System.Drawing.Point(272, 32);
+ this.label10.Name = "label10";
+ this.label10.Size = new System.Drawing.Size(88, 16);
+ this.label10.TabIndex = 19;
+ this.label10.Text = "(Mandatory)";
+ //
+ // label11
+ //
+ this.label11.Location = new System.Drawing.Point(272, 56);
+ this.label11.Name = "label11";
+ this.label11.Size = new System.Drawing.Size(88, 16);
+ this.label11.TabIndex = 20;
+ this.label11.Text = "(Mandatory)";
+ //
+ // label12
+ //
+ this.label12.Location = new System.Drawing.Point(272, 80);
+ this.label12.Name = "label12";
+ this.label12.Size = new System.Drawing.Size(88, 16);
+ this.label12.TabIndex = 21;
+ this.label12.Text = "(Mandatory)";
+ //
+ // label13
+ //
+ this.label13.Location = new System.Drawing.Point(272, 127);
+ this.label13.Name = "label13";
+ this.label13.Size = new System.Drawing.Size(88, 16);
+ this.label13.TabIndex = 22;
+ this.label13.Text = "(Mandatory)";
+ //
+ // label15
+ //
+ this.label15.Location = new System.Drawing.Point(272, 176);
+ this.label15.Name = "label15";
+ this.label15.Size = new System.Drawing.Size(88, 16);
+ this.label15.TabIndex = 24;
+ this.label15.Text = "(Mandatory)";
+ //
+ // label16
+ //
+ this.label16.Location = new System.Drawing.Point(272, 200);
+ this.label16.Name = "label16";
+ this.label16.Size = new System.Drawing.Size(88, 16);
+ this.label16.TabIndex = 25;
+ this.label16.Text = "(Mandatory)";
+ //
+ // label14
+ //
+ this.label14.Location = new System.Drawing.Point(272, 224);
+ this.label14.Name = "label14";
+ this.label14.Size = new System.Drawing.Size(88, 16);
+ this.label14.TabIndex = 26;
+ this.label14.Text = "(Mandatory)";
+ //
+ // label17
+ //
+ this.label17.Location = new System.Drawing.Point(272, 104);
+ this.label17.Name = "label17";
+ this.label17.Size = new System.Drawing.Size(88, 16);
+ this.label17.TabIndex = 27;
+ this.label17.Text = "(Optional)";
+ //
+ // label18
+ //
+ this.label18.Location = new System.Drawing.Point(272, 152);
+ this.label18.Name = "label18";
+ this.label18.Size = new System.Drawing.Size(88, 16);
+ this.label18.TabIndex = 28;
+ this.label18.Text = "(Optional)";
+ //
+ // btnAdd
+ //
+ this.btnAdd.Location = new System.Drawing.Point(288, 248);
+ this.btnAdd.Name = "btnAdd";
+ this.btnAdd.TabIndex = 9;
+ this.btnAdd.Text = "Define...";
+ this.btnAdd.Click += new System.EventHandler(this.btnAdd_Click);
+ //
+ // btnCancel
+ //
+ this.btnCancel.Location = new System.Drawing.Point(152, 248);
+ this.btnCancel.Name = "btnCancel";
+ this.btnCancel.TabIndex = 10;
+ this.btnCancel.Text = "Cancel";
+ this.btnCancel.Click += new System.EventHandler(this.btnCancel_Click);
+ //
+ // ProcessDefineDialog
+ //
+ this.AutoScaleBaseSize = new System.Drawing.Size(5, 13);
+ this.ClientSize = new System.Drawing.Size(370, 279);
+ this.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.btnCancel,
+ this.btnAdd,
+ this.label18,
+ this.label17,
+ this.label14,
+ this.label16,
+ this.label15,
+ this.label13,
+ this.label12,
+ this.label11,
+ this.label10,
+ this.comboType,
+ this.textProcessOwner,
+ this.textProcessCWD,
+ this.textProcessArgs,
+ this.textProcessPath,
+ this.textProcessEnv,
+ this.textProcessGroup,
+ this.textProcessName,
+ this.label9,
+ this.label8,
+ this.label7,
+ this.label6,
+ this.label5,
+ this.label4,
+ this.label3,
+ this.label2,
+ this.label1,
+ this.comboComputer});
+ this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.FixedDialog;
+ this.MaximizeBox = false;
+ this.MinimizeBox = false;
+ this.Name = "ProcessDefineDialog";
+ this.StartPosition = System.Windows.Forms.FormStartPosition.CenterParent;
+ this.Text = "Define Process";
+ this.Load += new System.EventHandler(this.ProcessDefineDialog_Load);
+ this.ResumeLayout(false);
+
+ }
+ #endregion
+
+ private void btnCancel_Click(object sender, System.EventArgs e)
+ {
+ this.Dispose();
+ this.Close();
+ }
+
+ private void btnAdd_Click(object sender, System.EventArgs e)
+ {
+ //TODO: ERROR CHECK
+
+ Computer c;
+ c=c_mgmt.getComputer(this.m_selComputer);
+
+ c.addProcess(new Process(this.textProcessName.Text.ToString(),
+ this.textProcessOwner.Text.ToString(),
+ this.textProcessGroup.Text.ToString(),
+ c));
+ this.Close();
+ this.Dispose();
+ }
+
+ private void ProcessDefineDialog_Load(object sender, System.EventArgs e)
+ {
+ comboType.SelectedIndex=0;
+ ArrayList list = c_mgmt.getComputerCollection();
+ int i=0, selIndex=0;
+ foreach(Computer computer in list)
+ {
+ this.comboComputer.Items.Add(computer.getName());
+ if(computer.getName().Equals(m_selComputer))
+ selIndex=i;
+ i++;
+ }
+ comboComputer.SelectedIndex=selIndex;
+ }
+
+
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/fileaccess/FileMgmt.cs b/ndb/src/cw/cpcc-win32/csharp/fileaccess/FileMgmt.cs
new file mode 100644
index 00000000000..b3a2361bcb0
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/fileaccess/FileMgmt.cs
@@ -0,0 +1,41 @@
+using System;
+using System.Text;
+using System.Collections.Specialized;
+using System.IO;
+using System.Windows.Forms;
+namespace NDB_CPC.fileaccess
+{
+ /// <summary>
+ /// Summary description for FileMgmt.
+ /// </summary>
+ public class FileMgmt
+ {
+ public FileMgmt()
+ {
+ }
+
+ public StringCollection importHostFile(string filename)
+ {
+ StringCollection sc = new StringCollection();
+ StreamReader SR = new StreamReader(filename);
+ string line ="";
+ line = SR.ReadLine();
+ while(!line.Equals(""))
+ {
+ sc.Add(line);
+ line = SR.ReadLine();
+ }
+ return sc;
+ }
+
+ public void exportHostFile(string filename, string content)
+ {
+ StreamWriter SW = new StreamWriter(filename,false);
+ SW.Write(content);
+ SW.WriteLine("");
+ SW.WriteLine("");
+ SW.Close();
+ }
+
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/simpleparser/SimpleCPCParser.cs b/ndb/src/cw/cpcc-win32/csharp/simpleparser/SimpleCPCParser.cs
new file mode 100644
index 00000000000..b8ff2844af9
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/simpleparser/SimpleCPCParser.cs
@@ -0,0 +1,360 @@
+using System;
+using System.Collections;
+using System.IO;
+using System.Windows.Forms;
+using NDB_CPC;
+using NDB_CPC.socketcomm;
+
+namespace NDB_CPC.simpleparser
+{
+ /// <summary>
+ /// Summary description for SimpleCPCParser.
+ /// </summary>
+ public class SimpleCPCParser
+ {
+ public SimpleCPCParser()
+ {
+ //
+ // TODO: Add constructor logic here
+ //
+ }
+
+ public static void parse(Process p, SocketComm comm)
+ {
+
+ string line=comm.readLine();//reader.ReadLine();
+ while(line.Equals(""))
+ {
+ line=comm.readLine();
+ }
+ if(line.Equals("define process"))
+ {
+ defineProcess(p, comm);
+ line="";
+ return;
+ }
+ if(line.Equals("start process"))
+ {
+ startProcess(p,comm);
+ line="";
+ return;
+ }
+ if(line.Equals("stop process"))
+ {
+ stopProcess(p,comm);
+ line="";
+ return;
+ }
+ if(line.Equals("undefine process"))
+ {
+ undefineProcess(p,comm);
+ line="";
+ return;
+ }
+
+ }
+
+ public static void parse(ArrayList processes, Computer c, SocketComm comm)
+ {
+
+ string line=comm.readLine();//reader.ReadLine();
+ while(line.Equals(""))
+ {
+ line=comm.readLine();
+ }
+
+ if(line.Equals("start processes"))
+ {
+ listProcesses(processes, c, comm);
+ line="";
+ return;
+ }
+
+ }
+
+ private static void defineProcess(Process p, SocketComm comm)
+ {
+ string line=comm.readLine();//reader.ReadLine();
+ while(!line.Equals(""))
+ {
+ if(line.StartsWith("status:"))
+ {
+ line=line.Remove(0,7);
+ line=line.Trim();
+ if(line.Equals("1"))
+ {
+ p.setDefined(true);
+ p.setStatus(Process.Status.Stopped);
+ }
+ else
+ p.setDefined(false);
+ }
+ if(line.StartsWith("id:"))
+ {
+ line=line.Remove(0,3);
+ line=line.Trim();
+ p.setId(line);
+ }
+ line=comm.readLine();
+ }
+ }
+
+
+ private static void startProcess(Process p, SocketComm comm)
+ {
+ string line=comm.readLine();//reader.ReadLine();
+ while(!line.Equals(""))
+ {
+ if(line.StartsWith("status:"))
+ {
+ line=line.Remove(0,7);
+ line=line.Trim();
+ if(line.Equals("1"))
+ p.setStatus(NDB_CPC.Process.Status.Running);
+ else
+ p.setStatus(NDB_CPC.Process.Status.Unknown);
+
+ }
+ if(line.StartsWith("id:"))
+ {
+ line=line.Remove(0,3);
+ line=line.Trim();
+ if(p.getId().Equals(line))
+ {
+ ;
+ }
+ else
+ {
+ //damn something is wrong
+ p.setStatus(NDB_CPC.Process.Status.Unknown);
+ }
+
+ }
+ line=comm.readLine();
+ }
+ }
+ private static void undefineProcess(Process p, SocketComm comm)
+ {
+ string line=comm.readLine();//reader.ReadLine();
+ while(!line.Equals(""))
+ {
+ if(line.StartsWith("status:"))
+ {
+
+ line=line.Remove(0,7);
+ line=line.Trim();
+ if(line.Equals("1"))
+ p.setDefined(false);
+ else
+ p.setDefined(true);
+
+ }
+ if(line.StartsWith("id:"))
+ {
+ line=line.Remove(0,3);
+ line=line.Trim();
+ }
+ line=comm.readLine();
+ }
+ }
+
+ private static void stopProcess(Process p, SocketComm comm)
+ {
+ string line=comm.readLine();//reader.ReadLine();
+ while(!line.Equals(""))
+ {
+ if(line.StartsWith("status:"))
+ {
+ line=line.Remove(0,7);
+ line=line.Trim();
+ if(line.Equals("1"))
+ p.setStatus(NDB_CPC.Process.Status.Stopped);
+ else
+ p.setStatus(NDB_CPC.Process.Status.Unknown);
+
+ }
+ if(line.StartsWith("id:"))
+ {
+ line=line.Remove(0,3);
+ line=line.Trim();
+ if(p.getId().Equals(line))
+ {
+ ;
+ }
+ else
+ {
+ //damn something is wrong
+ p.setStatus(NDB_CPC.Process.Status.Unknown);
+ }
+
+ }
+ line=comm.readLine();
+ }
+ }
+ private static void listProcesses(ArrayList processes, Computer c, SocketComm comm)
+ {
+ bool processExist = false;
+
+ string line=comm.readLine();//reader.ReadLine();
+ while(!line.Equals("end processes"))
+ {
+ if(line.Equals("process"))
+ {
+ line=comm.readLine();
+ Process p = new Process();
+
+ while(!line.Equals(""))
+ {
+ if(line.StartsWith("id:"))
+ {
+ string pid;
+ line=line.Remove(0,3);
+ pid=line.Trim();
+ /*check if process already exist*/
+ processExist=findProcess(processes,pid);
+ if(!processExist)
+ {
+ p.setId(pid);
+ }
+ }
+
+ if(line.StartsWith("name:"))
+ {
+
+ line=line.Remove(0,5);
+ line=line.Trim();
+ /*check if process already exist*/
+ if(!processExist)
+ {
+ p.setName(line);
+ }
+ }
+
+ if(line.StartsWith("path:"))
+ {
+
+ line=line.Remove(0,5);
+ line=line.Trim();
+ /*check if process already exist*/
+ if(!processExist)
+ {
+ p.setPath(line);
+ }
+ }
+
+ if(line.StartsWith("args:"))
+ {
+
+ line=line.Remove(0,5);
+ line=line.Trim();
+ /*check if process already exist*/
+ if(!processExist)
+ {
+ p.setArgs(line);
+ }
+ }
+
+ if(line.StartsWith("type:"))
+ {
+
+ line=line.Remove(0,5);
+ line=line.Trim();
+ /*check if process already exist*/
+ if(!processExist)
+ {
+
+ }
+ }
+
+ if(line.StartsWith("cwd:"))
+ {
+
+ line=line.Remove(0,4);
+ line=line.Trim();
+ /*check if process already exist*/
+ if(!processExist)
+ {
+ p.setCwd(line);
+ }
+ }
+
+ if(line.StartsWith("env:"))
+ {
+
+ line=line.Remove(0,4);
+ line=line.Trim();
+ /*check if process already exist*/
+ if(!processExist)
+ {
+ p.setEnv(line);
+ }
+ }
+
+ if(line.StartsWith("owner:"))
+ {
+
+ line=line.Remove(0,6);
+ line=line.Trim();
+ /*check if process already exist*/
+ if(!processExist)
+ {
+ p.setOwner(line);
+ }
+ }
+ if(line.StartsWith("group:"))
+ {
+
+ line=line.Remove(0,6);
+ line=line.Trim();
+ /*check if process already exist*/
+ if(!processExist)
+ {
+ p.setDatabase(line);
+ }
+ }
+
+ if(line.StartsWith("status:"))
+ {
+
+ line=line.Remove(0,7);
+ line=line.Trim();
+ /*check if process already exist*/
+ //if(!processExist)
+ //{
+ if(line.Equals("0"))
+ p.setStatus(Process.Status.Stopped);
+ if(line.Equals("1"))
+ p.setStatus(Process.Status.Running);
+ if(line.Equals("2"))
+ p.setStatus(Process.Status.Unknown);
+ //}
+ }
+
+
+ line=comm.readLine();
+ }
+ if(!processExist)
+ {
+ p.setComputer(c);
+ p.setDefined(true);
+ processes.Add(p);
+ }
+ processExist=false;
+ }
+ line=comm.readLine();
+
+ }
+ }
+
+ private static bool findProcess(ArrayList processes, string pid)
+ {
+ foreach (Process p in processes)
+ {
+ if(p.getId().Equals(pid))
+ return true;
+ }
+ return false;
+
+ }
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/socketcomm/SocketComm.cs b/ndb/src/cw/cpcc-win32/csharp/socketcomm/SocketComm.cs
new file mode 100644
index 00000000000..2cef5d34f17
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/socketcomm/SocketComm.cs
@@ -0,0 +1,207 @@
+using System;
+using System.Net;
+using System.Net.Sockets;
+using System.Text;
+using System.Windows.Forms;
+using System.Threading;
+using System.IO;
+
+namespace NDB_CPC.socketcomm
+{
+ /// <summary>
+ /// Summary description for SocketComm.
+ /// </summary>
+ public class SocketComm
+ {
+ private myTcpClient sender;
+ private StreamWriter writer;
+ private StreamReader reader;
+ private string m_host;
+ private int m_port;
+ private bool m_connected;
+ private bool m_connecting;
+ private Thread connectThread;
+ public SocketComm(string host, int port)
+ {
+
+ m_host=host;
+ m_port=port;
+ m_connected=false;
+ m_connecting=false;
+ }
+
+
+
+ public bool isConnected()
+ {
+ return m_connected;
+ }
+
+ public void doConnect()
+ {
+ if(!m_connecting && !m_connected)
+ {
+ connectThread= new Thread(new ThreadStart(connect));
+ connectThread.Start();
+ }
+
+ }
+
+ private void connect()
+ {
+ m_connecting=true;
+ while(true)
+ {
+ if(!m_connected)
+ {
+ try
+ {
+ // Establish the remote endpoint for the socket.
+ // The name of the
+ // remote device is "host.contoso.com".
+
+ // Create a TCP/IP socket.
+ sender = new myTcpClient();
+ // Connect the socket to the remote endpoint. Catch any errors.
+ try
+ {
+ /*
+ IPAddress ipAddress = Dns.Resolve(host).AddressList[0];
+ IPEndPoint ipLocalEndPoint = new IPEndPoint(ipAddress, 11000);
+*/
+
+
+ sender.Connect(m_host,m_port);;
+
+ writer = new StreamWriter(sender.GetStream(), Encoding.ASCII);
+ reader = new StreamReader(sender.GetStream(), Encoding.ASCII);
+ m_connected=true;
+ m_connecting=false;
+ // break;
+ Console.WriteLine("Socket connected to {0}",
+ sender.ToString());
+
+ }
+ catch (ArgumentNullException ane)
+ {
+ Console.WriteLine("ArgumentNullException : {0}",ane.ToString());
+ m_connected=false;
+ }
+ catch (SocketException se)
+ {
+ Console.WriteLine("SocketException : {0}",se.ToString());
+ m_connected=false;
+ }
+ }
+ catch (Exception e)
+ {
+ Console.WriteLine("Unexpected exception : {0}", e.ToString());
+ m_connected=false;
+ }
+
+ }
+
+ Thread.Sleep(200);
+ }
+ }
+
+ public bool disconnect()
+ {
+ try
+ {
+ this.m_connected=false;
+ this.m_connecting=false;
+ sender.GetUnderlyingSocket().Shutdown(SocketShutdown.Both);
+ sender.GetUnderlyingSocket().Close();
+ writer.Close();
+ reader.Close();
+ sender.Close();
+
+ }
+ catch (ArgumentNullException ane)
+ {
+ Console.WriteLine("ArgumentNullException : {0}",ane.ToString());
+ connectThread.Abort();
+ return false;
+ }
+ catch (SocketException se)
+ {
+ Console.WriteLine("SocketException : {0}",se.ToString());
+ connectThread.Abort();
+ return false;
+ }
+ catch (Exception e)
+ {
+ Console.WriteLine("Unexpected exception : {0}", e.ToString());
+ connectThread.Abort();
+ return false;
+ }
+ connectThread.Abort();
+ return true;
+ }
+
+ public bool writeMessage(string message)
+ {
+ int attempts=0;
+ while (attempts < 10)
+ {
+ try
+ {
+ writer.WriteLine(message);
+ writer.Flush();
+ message="";
+ return true;
+ }
+ catch(IOException e)
+ {
+ this.disconnect();
+ this.doConnect();
+ Thread.Sleep(200);
+ attempts++;
+ }
+ catch(System.NullReferenceException)
+ {
+ this.disconnect();
+ this.doConnect();
+
+ Thread.Sleep(200);
+ attempts++;
+ }
+ }
+ return false;
+ }
+
+ public string readLine()
+ {
+ int attempts=0;
+ string line="";
+ while (attempts < 10){
+ try
+ {
+ line = reader.ReadLine();
+ if(line==null)
+ line="";
+ return line;
+ }
+ catch(IOException e)
+ {
+ this.disconnect();
+ this.doConnect();
+ Thread.Sleep(400);
+ attempts++;
+ }
+ catch(System.NullReferenceException)
+ {
+ this.disconnect();
+ this.doConnect();
+ Thread.Sleep(400);
+ attempts++;
+ }
+ }
+ return "";
+
+ }
+
+ }
+}
+
diff --git a/ndb/src/cw/cpcc-win32/csharp/socketcomm/myTcpClient.cs b/ndb/src/cw/cpcc-win32/csharp/socketcomm/myTcpClient.cs
new file mode 100644
index 00000000000..9c0d82a0b27
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/socketcomm/myTcpClient.cs
@@ -0,0 +1,26 @@
+using System;
+using System.Net;
+using System.Net.Sockets;
+using System.Text;
+using System.Threading;
+using System.IO;
+
+
+namespace NDB_CPC.socketcomm
+{
+ public class myTcpClient : TcpClient
+ {
+ private Socket s;
+ public myTcpClient(): base()
+ {
+ if(this.Active)
+ {
+ s = this.Client;
+ }
+ }
+ public Socket GetUnderlyingSocket()
+ {
+ return s;
+ }
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/startDatabaseDlg.cs b/ndb/src/cw/cpcc-win32/csharp/startDatabaseDlg.cs
new file mode 100644
index 00000000000..cecfcaeb0f3
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/startDatabaseDlg.cs
@@ -0,0 +1,251 @@
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+using NDB_CPC.simpleparser;
+
+namespace NDB_CPC
+{
+ /// <summary>
+ /// Summary description for startDatabase.
+ /// </summary>
+ public class startDatabaseDlg : System.Windows.Forms.Form
+ {
+ private System.Windows.Forms.TextBox textAction;
+ private System.Windows.Forms.Label label1;
+ /// <summary>
+ /// Required designer variable.
+ /// </summary>
+ private System.ComponentModel.Container components = null;
+ private System.Windows.Forms.ProgressBar progressBar;
+ private System.Windows.Forms.Label label2;
+ private System.Windows.Forms.Button buttonGo;
+ private Database m_db;
+ public startDatabaseDlg(Database db)
+ {
+
+ //
+ // Required for Windows Form Designer support
+ //
+ InitializeComponent();
+
+ //
+ // TODO: Add any constructor code after InitializeComponent call
+ //
+ m_db=db;
+ }
+
+ /// <summary>
+ /// Clean up any resources being used.
+ /// </summary>
+ protected override void Dispose( bool disposing )
+ {
+ if( disposing )
+ {
+ if(components != null)
+ {
+ components.Dispose();
+ }
+ }
+ base.Dispose( disposing );
+ }
+
+ #region Windows Form Designer generated code
+ /// <summary>
+ /// Required method for Designer support - do not modify
+ /// the contents of this method with the code editor.
+ /// </summary>
+ private void InitializeComponent()
+ {
+ this.textAction = new System.Windows.Forms.TextBox();
+ this.label1 = new System.Windows.Forms.Label();
+ this.progressBar = new System.Windows.Forms.ProgressBar();
+ this.label2 = new System.Windows.Forms.Label();
+ this.buttonGo = new System.Windows.Forms.Button();
+ this.SuspendLayout();
+ //
+ // textAction
+ //
+ this.textAction.Location = new System.Drawing.Point(104, 40);
+ this.textAction.Name = "textAction";
+ this.textAction.ReadOnly = true;
+ this.textAction.Size = new System.Drawing.Size(256, 20);
+ this.textAction.TabIndex = 0;
+ this.textAction.Text = "";
+ //
+ // label1
+ //
+ this.label1.Location = new System.Drawing.Point(8, 40);
+ this.label1.Name = "label1";
+ this.label1.Size = new System.Drawing.Size(96, 16);
+ this.label1.TabIndex = 1;
+ this.label1.Text = "Current activity:";
+ this.label1.TextAlign = System.Drawing.ContentAlignment.MiddleRight;
+ //
+ // progressBar
+ //
+ this.progressBar.Location = new System.Drawing.Point(104, 88);
+ this.progressBar.Name = "progressBar";
+ this.progressBar.Size = new System.Drawing.Size(152, 16);
+ this.progressBar.TabIndex = 2;
+ //
+ // label2
+ //
+ this.label2.Location = new System.Drawing.Point(8, 88);
+ this.label2.Name = "label2";
+ this.label2.Size = new System.Drawing.Size(96, 16);
+ this.label2.TabIndex = 3;
+ this.label2.Text = "Activity progress:";
+ this.label2.TextAlign = System.Drawing.ContentAlignment.MiddleCenter;
+ //
+ // buttonGo
+ //
+ this.buttonGo.Location = new System.Drawing.Point(152, 136);
+ this.buttonGo.Name = "buttonGo";
+ this.buttonGo.Size = new System.Drawing.Size(96, 24);
+ this.buttonGo.TabIndex = 4;
+ this.buttonGo.Text = "Go!";
+ this.buttonGo.Click += new System.EventHandler(this.buttonGo_Click);
+ //
+ // startDatabaseDlg
+ //
+ this.AutoScaleBaseSize = new System.Drawing.Size(5, 13);
+ this.ClientSize = new System.Drawing.Size(378, 167);
+ this.Controls.AddRange(new System.Windows.Forms.Control[] {
+ this.buttonGo,
+ this.label2,
+ this.progressBar,
+ this.label1,
+ this.textAction});
+ this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.FixedSingle;
+ this.MaximizeBox = false;
+ this.MinimizeBox = false;
+ this.Name = "startDatabaseDlg";
+ this.Text = "Starting database";
+ this.Load += new System.EventHandler(this.startDatabase_Load);
+ this.Paint += new System.Windows.Forms.PaintEventHandler(this.startDatabase_Paint);
+ this.ResumeLayout(false);
+
+ }
+ #endregion
+
+ private void startDatabase_Load(object sender, System.EventArgs e)
+ {
+
+ }
+
+ private void startDatabase_Paint(object sender, System.Windows.Forms.PaintEventArgs e)
+ {
+
+
+
+ }
+ private void defineProcesses()
+ {
+ ArrayList processes = m_db.getProcesses();
+ progressBar.Maximum = processes.Count;
+ progressBar.Minimum = 0;
+
+ int retry=0;
+ //sc.connect("130.100.232.7");
+ foreach (Process p in processes)
+ {
+ Computer comp;
+ retry=0;
+ //if(p.getName().StartsWith("ndb") || p.getName().StartsWith("mgm"))
+ //{
+ textAction.Text="Defining process " + p.getName();
+ textAction.Refresh();
+ comp=p.getComputer();
+ while(retry<10)
+ {
+ if(!comp.isConnected())
+ {
+ comp.connectToCpcd();
+
+ }
+ else
+ {
+ if(comp.defineProcess(p)<0)
+ {
+ ;
+ }
+ else
+ break;
+ }
+ if(retry==9)
+ {
+ if(MessageBox.Show(this,"Failed to define process. Try again?","Warning!!!",MessageBoxButtons.YesNo)==DialogResult.Yes)
+ retry=0;
+ }
+ retry++;
+ //comp.undefineProcess(p);
+ }
+ //}
+ progressBar.PerformStep();
+ }
+ }
+
+ private void startProcesses()
+ {
+
+ ArrayList processes = m_db.getProcesses();
+ progressBar.Maximum = processes.Count;
+ progressBar.Minimum = 0;
+ string start = "start process \n";
+
+ int retry=0;
+ //sc.connect("130.100.232.7");
+ foreach (Process p in processes)
+ {
+ Computer comp;
+ if((p.getName().StartsWith("ndb")) || (p.getName().StartsWith("mgm")))
+ {
+ textAction.Text="Starting process " + p.getName();
+ textAction.Refresh();
+ start = start + "id:" + p.getId() + "\n\n";
+ comp=p.getComputer();
+ while(retry<10)
+ {
+ if(!comp.isConnected())
+ {
+ comp.connectToCpcd();
+ }
+ else
+ {
+ if(comp.startProcess(p)<0)
+ {
+ ;
+ }
+ else
+ break;
+ }
+ if(retry==9)
+ {
+ if(MessageBox.Show(this,"Failed to start process. Retry again?","Warning!!!",MessageBoxButtons.YesNo)==DialogResult.Yes)
+ retry=0;
+ }
+
+ retry++;
+ }
+ }
+ progressBar.PerformStep();
+
+ }
+
+ }
+
+ private void buttonGo_Click(object sender, System.EventArgs e)
+ {
+ buttonGo.Enabled=false;
+ progressBar.Step=1;
+ defineProcesses();
+ progressBar.Value=0;
+ startProcesses();
+
+ }
+
+
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/csharp/telnetclient/telnetClient.cs b/ndb/src/cw/cpcc-win32/csharp/telnetclient/telnetClient.cs
new file mode 100644
index 00000000000..a7966947e1f
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/csharp/telnetclient/telnetClient.cs
@@ -0,0 +1,408 @@
+using System;
+using System.Drawing;
+using System.Collections;
+using System.ComponentModel;
+using System.Windows.Forms;
+using System.Data;
+using System.Net;
+using System.Net.Sockets;
+using System.Text;
+using System.IO;
+using System.Threading ;
+
+namespace NDB_CPC.telnetclient
+{
+ /// <summary>
+ /// Summary description for telnetClient.
+ /// </summary>
+ public class telnetClient
+ {
+ Char IAC = Convert.ToChar(255);
+ Char DO = Convert.ToChar(253);
+ Char DONT = Convert.ToChar(254);
+ Char WILL = Convert.ToChar(251);
+ Char WONT = Convert.ToChar(252);
+ Char SB = Convert.ToChar(250);
+ Char SE = Convert.ToChar(240);
+ const Char IS = '0';
+ const Char SEND = '1';
+ const Char INFO = '2';
+ const Char VAR = '0';
+ const Char VALUE = '1';
+ const Char ESC = '2';
+ const Char USERVAR = '3';
+ string m_strResp;
+
+ private ArrayList m_ListOptions = new ArrayList();
+ private IPEndPoint iep ;
+ private AsyncCallback callbackProc ;
+ private string address ;
+ private int port ;
+ private Socket s ;
+ private TextBox textBox1;
+ Byte[] m_byBuff = new Byte[32767];
+
+
+ public telnetClient(string ip, int p, TextBox tb)
+ {
+
+ address = ip;
+ port = p;
+ textBox1=tb;
+ IPHostEntry IPHost = Dns.Resolve(address);
+ string []aliases = IPHost.Aliases;
+ IPAddress[] addr = IPHost.AddressList;
+
+ try
+ {
+ // Create New Socket
+ s = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
+ // Create New EndPoint
+ iep = new IPEndPoint(addr[0],port);
+ // This is a non blocking IO
+ s.Blocking = false ;
+ // Assign Callback function to read from Asyncronous Socket
+ callbackProc = new AsyncCallback(ConnectCallback);
+ // Begin Asyncronous Connection
+ s.BeginConnect(iep , callbackProc, s ) ;
+
+ }
+ catch(Exception eeeee )
+ {
+ MessageBox.Show(eeeee.Message , "Application Error!!!" , MessageBoxButtons.OK , MessageBoxIcon.Stop );
+ Application.Exit();
+ }
+ }
+
+ public void ConnectCallback( IAsyncResult ar )
+ {
+ try
+ {
+ // Get The connection socket from the callback
+ Socket sock1 = (Socket)ar.AsyncState;
+ if ( sock1.Connected )
+ {
+ // Define a new Callback to read the data
+ AsyncCallback recieveData = new AsyncCallback( OnRecievedData );
+ // Begin reading data asyncronously
+ sock1.BeginReceive( m_byBuff, 0, m_byBuff.Length, SocketFlags.None, recieveData , sock1 );
+ }
+ }
+ catch( Exception ex )
+ {
+ MessageBox.Show(ex.Message, "Setup Recieve callbackProc failed!" );
+ }
+ }
+
+
+ public void OnRecievedData( IAsyncResult ar )
+ {
+ // Get The connection socket from the callback
+ Socket sock = (Socket)ar.AsyncState;
+ // Get The data , if any
+ int nBytesRec = sock.EndReceive( ar );
+ if( nBytesRec > 0 )
+ {
+ string sRecieved = Encoding.ASCII.GetString( m_byBuff, 0, nBytesRec );
+ string m_strLine="";
+ for ( int i=0; i < nBytesRec;i++)
+ {
+ Char ch = Convert.ToChar(m_byBuff[i]);
+ switch( ch )
+ {
+ case '\r':
+ m_strLine += Convert.ToString("\r\n");
+ break;
+ case '\n':
+ break;
+ default:
+ m_strLine += Convert.ToString(ch);
+ break;
+ }
+ }
+ try
+ {
+ int strLinelen = m_strLine.Length ;
+ if ( strLinelen == 0 )
+ {
+ m_strLine = Convert.ToString("\r\n");
+ }
+
+ Byte[] mToProcess = new Byte[strLinelen];
+ for ( int i=0; i < strLinelen ; i++)
+ mToProcess[i] = Convert.ToByte(m_strLine[i]);
+ // Process the incoming data
+ string mOutText = ProcessOptions(mToProcess);
+ if ( mOutText != "" )
+ textBox1.AppendText(mOutText);
+
+ // Respond to any incoming commands
+ RespondToOptions();
+ }
+ catch( Exception ex )
+ {
+ Object x = this ;
+ MessageBox.Show(ex.Message , "Information!" );
+ }
+ }
+ else
+ {
+ // If no data was recieved then the connection is probably dead
+ Console.WriteLine( "Disconnected", sock.RemoteEndPoint );
+ sock.Shutdown( SocketShutdown.Both );
+ sock.Close();
+ Application.Exit();
+ }
+ }
+
+ private string ProcessOptions(byte[] m_strLineToProcess)
+ {
+ string m_DISPLAYTEXT ="";
+ string m_strTemp ="" ;
+ string m_strOption ="";
+ string m_strNormalText ="";
+ bool bScanDone =false;
+ int ndx =0;
+ int ldx =0;
+ char ch ;
+ try
+ {
+ for ( int i=0; i < m_strLineToProcess.Length ; i++)
+ {
+ Char ss = Convert.ToChar(m_strLineToProcess[i]);
+ m_strTemp = m_strTemp + Convert.ToString(ss);
+ }
+
+ while(bScanDone != true )
+ {
+ int lensmk = m_strTemp.Length;
+ ndx = m_strTemp.IndexOf(Convert.ToString(IAC));
+ if ( ndx > lensmk )
+ ndx = m_strTemp.Length;
+
+ if(ndx != -1)
+ {
+ m_DISPLAYTEXT+= m_strTemp.Substring(0,ndx);
+ ch = m_strTemp[ndx + 1];
+ if ( ch == DO || ch == DONT || ch == WILL || ch == WONT )
+ {
+ m_strOption = m_strTemp.Substring(ndx, 3);
+ string txt = m_strTemp.Substring(ndx + 3);
+ m_DISPLAYTEXT+= m_strTemp.Substring(0,ndx);
+ m_ListOptions.Add(m_strOption);
+ m_strTemp = txt ;
+ }
+ else
+ if ( ch == IAC)
+ {
+ m_DISPLAYTEXT= m_strTemp.Substring(0,ndx);
+ m_strTemp = m_strTemp.Substring(ndx + 1);
+ }
+ else
+ if ( ch == SB )
+ {
+ m_DISPLAYTEXT= m_strTemp.Substring(0,ndx);
+ ldx = m_strTemp.IndexOf(Convert.ToString(SE));
+ m_strOption = m_strTemp.Substring(ndx, ldx);
+ m_ListOptions.Add(m_strOption);
+ m_strTemp = m_strTemp.Substring(ldx);
+ }
+ }
+ else
+ {
+ m_DISPLAYTEXT = m_DISPLAYTEXT + m_strTemp;
+ bScanDone = true ;
+ }
+ }
+ m_strNormalText = m_DISPLAYTEXT;
+ }
+ catch(Exception eP)
+ {
+ MessageBox.Show(eP.Message , "Application Error!!!" , MessageBoxButtons.OK , MessageBoxIcon.Stop );
+ Application.Exit();
+ }
+ return m_strNormalText ;
+ }
+
+ void DispatchMessage(string strText)
+ {
+ try
+ {
+ Byte[] smk = new Byte[strText.Length];
+ for ( int i=0; i < strText.Length ; i++)
+ {
+ Byte ss = Convert.ToByte(strText[i]);
+ smk[i] = ss ;
+ }
+
+ IAsyncResult ar2 = s.BeginSend(smk , 0 , smk.Length , SocketFlags.None , callbackProc , s );
+ s.EndSend(ar2);
+ }
+ catch(Exception ers)
+ {
+ MessageBox.Show("ERROR IN RESPOND OPTIONS");
+ }
+ }
+
+ void RespondToOptions()
+ {
+ try
+ {
+ string strOption;
+ for ( int i=0; i < m_ListOptions.Count; i++)
+ {
+ strOption = (string)m_ListOptions[i];
+ ArrangeReply(strOption);
+ }
+ DispatchMessage(m_strResp);
+ m_strResp ="";
+ m_ListOptions.Clear();
+ }
+ catch(Exception ers)
+ {
+ MessageBox.Show("ERROR IN RESPOND OPTIONS");
+ }
+ }
+ void ArrangeReply(string strOption)
+ {
+ try
+ {
+
+ Char Verb;
+ Char Option;
+ Char Modifier;
+ Char ch;
+ bool bDefined = false;
+
+ if(strOption.Length < 3) return;
+
+ Verb = strOption[1];
+ Option = strOption[2];
+
+ if ( Option == 1 || Option == 3 )
+ {
+ // case 1: // Echo
+ // case 3: // Suppress Go-Ahead
+ bDefined = true;
+ // break;
+ }
+
+ m_strResp += IAC;
+
+ if(bDefined == true )
+ {
+ if ( Verb == DO )
+ {
+ // case DO:
+ ch = WILL;
+ m_strResp += ch;
+ m_strResp += Option;
+ // break;
+ }
+ if ( Verb == DONT )
+ {
+ ch = WONT;
+ m_strResp += ch;
+ m_strResp += Option;
+ // break;
+ }
+ if ( Verb == WILL )
+ {
+ ch = DO;
+ m_strResp += ch;
+ m_strResp += Option;
+ //break;
+ }
+ if ( Verb == WONT)
+ {
+ ch = DONT;
+ m_strResp += ch;
+ m_strResp += Option;
+ // break;
+ }
+ if ( Verb == SB)
+ {
+ Modifier = strOption[3];
+ if(Modifier == SEND)
+ {
+ ch = SB;
+ m_strResp += ch;
+ m_strResp += Option;
+ m_strResp += IS;
+ m_strResp += IAC;
+ m_strResp += SE;
+ }
+ // break;
+ }
+ }
+ else
+ {
+ // switch(Verb)
+ // {
+ if ( Verb == DO )
+ {
+ ch = WONT;
+ m_strResp += ch;
+ m_strResp += Option;
+ // break;
+ }
+ if ( Verb == DONT)
+ {
+ ch = WONT;
+ m_strResp += ch;
+ m_strResp += Option;
+ // break;
+ }
+ if ( Verb == WILL)
+ {
+ ch = DONT;
+ m_strResp += ch;
+ m_strResp += Option;
+ // break;
+ }
+ if ( Verb == WONT)
+ {
+ ch = DONT;
+ m_strResp += ch;
+ m_strResp += Option;
+ // break;
+ }
+ }
+ }
+ catch(Exception eeeee )
+ {
+ MessageBox.Show(eeeee.Message , "Application Error!!!" , MessageBoxButtons.OK , MessageBoxIcon.Stop );
+ Application.Exit();
+ }
+
+ }
+
+ private void textBox1_KeyPress_1(object sender, System.Windows.Forms.KeyPressEventArgs e)
+ {
+ if ( e.KeyChar == 13 )
+ {
+ DispatchMessage("\r\n");
+ }
+ else
+ if ( e.KeyChar == 8 )
+ {
+ try
+ {
+// string mtmp = textBox1.Text.Substring(0,textBox1.Text.Length-1);
+// textBox1.Text = "" ;
+ }
+ catch(Exception ebs)
+ {
+ MessageBox.Show("ERROR IN BACKSPACE");
+ }
+ }
+ else
+ {
+ string str = e.KeyChar.ToString();
+ DispatchMessage(str);
+ }
+ }
+
+
+ }
+}
diff --git a/ndb/src/cw/cpcc-win32/vb6/Computer.cls b/ndb/src/cw/cpcc-win32/vb6/Computer.cls
new file mode 100644
index 00000000000..5b42dfeadb6
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Computer.cls
@@ -0,0 +1,20 @@
+VERSION 1.0 CLASS
+BEGIN
+ MultiUse = -1 'True
+ Persistable = 0 'NotPersistable
+ DataBindingBehavior = 0 'vbNone
+ DataSourceBehavior = 0 'vbNone
+ MTSTransactionMode = 0 'NotAnMTSObject
+END
+Attribute VB_Name = "Computer"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = True
+Attribute VB_PredeclaredId = False
+Attribute VB_Exposed = False
+Attribute VB_Ext_KEY = "SavedWithClassBuilder6" ,"Yes"
+Attribute VB_Ext_KEY = "Top_Level" ,"Yes"
+Public m_ip As String
+Public m_name As String
+Public m_status As String
+Public m_processes As Collection
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/Database.cls b/ndb/src/cw/cpcc-win32/vb6/Database.cls
new file mode 100644
index 00000000000..dfb1195d910
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Database.cls
@@ -0,0 +1,18 @@
+VERSION 1.0 CLASS
+BEGIN
+ MultiUse = -1 'True
+ Persistable = 0 'NotPersistable
+ DataBindingBehavior = 0 'vbNone
+ DataSourceBehavior = 0 'vbNone
+ MTSTransactionMode = 0 'NotAnMTSObject
+END
+Attribute VB_Name = "Database_"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = True
+Attribute VB_PredeclaredId = False
+Attribute VB_Exposed = False
+Attribute VB_Ext_KEY = "SavedWithClassBuilder6" ,"Yes"
+Attribute VB_Ext_KEY = "Top_Level" ,"Yes"
+Public m_name As String
+Public m_processes As Collection
+Public m_status As String
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 110.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 110.ico
new file mode 100644
index 00000000000..34b85992394
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 110.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 231.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 231.ico
new file mode 100644
index 00000000000..fe30ff5d1e6
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 231.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 237.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 237.ico
new file mode 100644
index 00000000000..af0a1294f9e
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 237.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 241.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 241.ico
new file mode 100644
index 00000000000..e8caf6e9a73
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 241.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 242.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 242.ico
new file mode 100644
index 00000000000..2deff5472bc
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 242.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 270.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 270.ico
new file mode 100644
index 00000000000..9cab239de23
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 270.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 271.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 271.ico
new file mode 100644
index 00000000000..f05c95f74fe
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 271.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 273.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 273.ico
new file mode 100644
index 00000000000..800606eda0c
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 273.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 31.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 31.ico
new file mode 100644
index 00000000000..a2404977771
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 31.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 337.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 337.ico
new file mode 100644
index 00000000000..9dadb12cfbe
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 337.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 338.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 338.ico
new file mode 100644
index 00000000000..a13c80c81b4
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 338.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/Icon 339.ico b/ndb/src/cw/cpcc-win32/vb6/Icon 339.ico
new file mode 100644
index 00000000000..5eb4c06815d
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Icon 339.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/MSSCCPRJ.SCC b/ndb/src/cw/cpcc-win32/vb6/MSSCCPRJ.SCC
new file mode 100644
index 00000000000..3100640f8bd
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/MSSCCPRJ.SCC
@@ -0,0 +1,5 @@
+[SCC]
+SCC=This is a source code control file
+[NdbCPC.vbp]
+SCC_Project_Name=this project is not under source code control
+SCC_Aux_Path=<This is an empty string for the mssccprj.scc file>
diff --git a/ndb/src/cw/cpcc-win32/vb6/Module1.bas b/ndb/src/cw/cpcc-win32/vb6/Module1.bas
new file mode 100644
index 00000000000..ae8ed444a41
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Module1.bas
@@ -0,0 +1,233 @@
+Attribute VB_Name = "Module1"
+Option Explicit
+Public fMainForm As frmMain
+Public g_computers As New Collection
+Public g_databases As New Collection
+
+Sub Main()
+ If False Then
+ Dim fLogin As New frmLogin
+ fLogin.Show vbModal
+ If Not fLogin.OK Then
+ 'Login Failed so exit app
+ End
+ End If
+ Unload fLogin
+
+ frmSplash.Show
+ frmSplash.Refresh
+ End If
+
+ init
+
+ Set fMainForm = New frmMain
+ Load fMainForm
+ Unload frmSplash
+
+ fMainForm.Show
+End Sub
+
+Private Sub init()
+ Dim c As Computer
+ Dim p As Process
+
+ ' ---
+ ' One node configuration
+ '
+ Set c = New Computer
+ With c
+ .m_ip = "130.100.232.31"
+ .m_name = "ndb-client31"
+ .m_status = "Connected"
+ Set .m_processes = New Collection
+ End With
+ addComputer c
+
+ Set p = New Process
+ With p
+ .m_id = "1"
+ .m_name = "mgm-1"
+ .m_database = "elathal"
+ .m_status = "Running"
+ .m_owner = "elathal"
+ Set .m_computer = c
+ End With
+ addProcess c, p
+
+ Set p = New Process
+ With p
+ .m_id = "2"
+ .m_name = "ndb-2"
+ .m_database = "elathal"
+ .m_status = "Running"
+ .m_owner = "elathal"
+ Set .m_computer = c
+ End With
+ addProcess c, p
+
+ Set p = New Process
+ With p
+ .m_id = "3"
+ .m_name = "api-3"
+ .m_database = "elathal"
+ .m_status = "Running"
+ .m_owner = "elathal"
+ Set .m_computer = c
+ End With
+ addProcess c, p
+
+ ' ---
+ ' Two node configuration
+ '
+ Set p = New Process
+ With p
+ .m_id = "4"
+ .m_name = "mgm-1"
+ .m_database = "ejonore-2-node"
+ .m_status = "Running"
+ .m_owner = "ejonore"
+ Set .m_computer = c
+ End With
+ addProcess c, p
+
+ Set c = New Computer
+ With c
+ .m_ip = "10.0.1.1"
+ .m_name = "cluster-1"
+ .m_status = "Connected"
+ Set .m_processes = New Collection
+ End With
+ addComputer c
+
+ Set p = New Process
+ With p
+ .m_id = "1"
+ .m_name = "ndb-2"
+ .m_database = "ejonore-2-node"
+ .m_status = "Running"
+ .m_owner = "ejonore"
+ Set .m_computer = c
+ End With
+ addProcess c, p
+
+ Set c = New Computer
+ With c
+ .m_ip = "10.0.2.1"
+ .m_name = "cluster-2"
+ .m_status = "Connected"
+ Set .m_processes = New Collection
+ End With
+ addComputer c
+
+ Set p = New Process
+ With p
+ .m_id = "1"
+ .m_name = "ndb-3"
+ .m_database = "ejonore-2-node"
+ .m_status = "Running"
+ .m_owner = "ejonore"
+ Set .m_computer = c
+ End With
+ addProcess c, p
+
+ Set c = New Computer
+ With c
+ .m_ip = "10.0.3.1"
+ .m_name = "cluster-3"
+ .m_status = "Connected"
+ Set .m_processes = New Collection
+ End With
+ addComputer c
+
+ Set p = New Process
+ With p
+ .m_id = "1"
+ .m_name = "api-4"
+ .m_database = "ejonore-2-node"
+ .m_status = "Running"
+ .m_owner = "ejonore"
+ Set .m_computer = c
+ End With
+ addProcess c, p
+
+ Set c = New Computer
+ With c
+ .m_ip = "10.0.4.1"
+ .m_name = "cluster-4"
+ .m_status = "Connected"
+ Set .m_processes = New Collection
+ End With
+ addComputer c
+
+ Set p = New Process
+ With p
+ .m_id = "1"
+ .m_name = "api-5"
+ .m_database = "ejonore-2-node"
+ .m_status = "Running"
+ .m_owner = "ejonore"
+ Set .m_computer = c
+ End With
+ addProcess c, p
+
+ Set c = New Computer
+ With c
+ .m_ip = "130.100.232.5"
+ .m_name = "ndbs05"
+ .m_status = "Not connected"
+ Set .m_processes = New Collection
+ End With
+ addComputer c
+
+ Set c = New Computer
+ With c
+ .m_ip = "130.100.232.7"
+ .m_name = "ndb-srv7"
+ .m_status = "No contact"
+ Set .m_processes = New Collection
+ End With
+ addComputer c
+
+End Sub
+
+Public Sub addComputer(ByRef c As Computer)
+ g_computers.Add c, "_" & c.m_name
+End Sub
+
+Private Sub addProcess(ByRef c As Computer, ByRef p As Process)
+ c.m_processes.Add p, "_" & p.m_id
+
+ Dim cl As Database_
+ If Not Exists(g_databases, "_" & p.m_database) Then
+ Set cl = New Database_
+ With cl
+ .m_name = p.m_database
+ .m_status = "Unknown"
+ Set .m_processes = New Collection
+ End With
+ g_databases.Add cl, "_" & p.m_database
+ Else
+ Set cl = g_databases("_" & p.m_database)
+ End If
+ cl.m_processes.Add p, "_" & p.m_computer.m_name & "_" & p.m_id
+End Sub
+
+Public Function Exists(ByRef c As Collection, ByVal k As String) As Boolean
+ Dim r As Boolean
+ Dim o As Object
+
+ r = True
+
+ On Error GoTo NotFound
+ Set o = c.Item(k)
+ GoTo Continue
+NotFound:
+ If Err.Number <> 5 Then
+ Err.Raise Err.Number, Err.Source, Err.Description
+ End If
+
+ r = False
+Continue:
+ Exists = r
+End Function
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/NdbCPC.vbp b/ndb/src/cw/cpcc-win32/vb6/NdbCPC.vbp
new file mode 100644
index 00000000000..dc8f3780a74
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/NdbCPC.vbp
@@ -0,0 +1,49 @@
+Type=Exe
+Object={831FDD16-0C5C-11D2-A9FC-0000F8754DA1}#2.0#0; mscomctl.ocx
+Module=Module1; Module1.bas
+Form=frmMain.frm
+Form=frmSplash.frm
+Form=frmLogin.frm
+Form=frmOptions.frm
+Form=frmAbout.frm
+Class=Computer; Computer.cls
+Class=Process; Process.cls
+Class=Database_; Database.cls
+Form=frmNewComputer.frm
+Form=frmNewDatabase3.frm
+Form=frmNewDatabase1.frm
+Form=frmNewDatabase2.frm
+IconForm="frmAbout"
+Startup="Sub Main"
+HelpFile=""
+Title="NdbCPC"
+ExeName32="NdbCPC.exe"
+Command32=""
+Name="NdbCPC"
+HelpContextID="0"
+CompatibleMode="0"
+MajorVer=1
+MinorVer=0
+RevisionVer=0
+AutoIncrementVer=0
+ServerSupportFiles=0
+VersionCompanyName="ctp"
+CompilationType=0
+OptimizationType=0
+FavorPentiumPro(tm)=0
+CodeViewDebugInfo=0
+NoAliasing=0
+BoundsCheck=0
+OverflowCheck=0
+FlPointCheck=0
+FDIVCheck=0
+UnroundedFP=0
+StartMode=0
+Unattended=0
+Retained=0
+ThreadPerObject=0
+MaxNumberOfThreads=1
+DebugStartupOption=0
+
+[MS Transaction Server]
+AutoRefresh=1
diff --git a/ndb/src/cw/cpcc-win32/vb6/NdbCPC.vbw b/ndb/src/cw/cpcc-win32/vb6/NdbCPC.vbw
new file mode 100644
index 00000000000..825abbc923a
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/NdbCPC.vbw
@@ -0,0 +1,13 @@
+Module1 = 44, 44, 577, 492,
+frmMain = 44, 44, 577, 492, , 66, 66, 599, 514, C
+frmSplash = 132, 132, 670, 576, C, 88, 88, 621, 536, C
+frmLogin = 0, 0, 538, 444, C, 110, 110, 643, 558, C
+frmOptions = 176, 176, 714, 620, C, 132, 132, 665, 580, C
+frmAbout = 132, 132, 759, 511, C, 154, 154, 687, 602, C
+Computer = 110, 110, 648, 554,
+Process = 132, 132, 670, 576, C
+Database_ = 88, 88, 626, 532, C
+frmNewComputer = 44, 44, 582, 488, , 22, 22, 390, 218, C
+frmNewDatabase3 = 0, 0, 506, 444, , 0, 0, 506, 444, C
+frmNewDatabase1 = 132, 132, 638, 550, , 154, 154, 660, 572, C
+frmNewDatabase2 = 198, 198, 704, 616, , 176, 176, 682, 594, C
diff --git a/ndb/src/cw/cpcc-win32/vb6/Process.cls b/ndb/src/cw/cpcc-win32/vb6/Process.cls
new file mode 100644
index 00000000000..fcb4c2cbb2c
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/Process.cls
@@ -0,0 +1,22 @@
+VERSION 1.0 CLASS
+BEGIN
+ MultiUse = -1 'True
+ Persistable = 0 'NotPersistable
+ DataBindingBehavior = 0 'vbNone
+ DataSourceBehavior = 0 'vbNone
+ MTSTransactionMode = 0 'NotAnMTSObject
+END
+Attribute VB_Name = "Process"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = True
+Attribute VB_PredeclaredId = False
+Attribute VB_Exposed = False
+Attribute VB_Ext_KEY = "SavedWithClassBuilder6" ,"Yes"
+Attribute VB_Ext_KEY = "Top_Level" ,"Yes"
+Public m_computer As Computer
+Public m_id As String
+Public m_name As String
+Public m_database As String
+Public m_status As String
+Public m_owner As String
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/closed folder.ico b/ndb/src/cw/cpcc-win32/vb6/closed folder.ico
new file mode 100644
index 00000000000..fe82350d376
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/closed folder.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/computer.ico b/ndb/src/cw/cpcc-win32/vb6/computer.ico
new file mode 100644
index 00000000000..d73302d1cd5
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/computer.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmAbout.frm b/ndb/src/cw/cpcc-win32/vb6/frmAbout.frm
new file mode 100644
index 00000000000..b842d20de21
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmAbout.frm
@@ -0,0 +1,245 @@
+VERSION 5.00
+Begin VB.Form frmAbout
+ BorderStyle = 3 'Fixed Dialog
+ Caption = "About NdbCPC"
+ ClientHeight = 3630
+ ClientLeft = 45
+ ClientTop = 330
+ ClientWidth = 5865
+ ClipControls = 0 'False
+ LinkTopic = "Form1"
+ MaxButton = 0 'False
+ MinButton = 0 'False
+ ScaleHeight = 3630
+ ScaleWidth = 5865
+ ShowInTaskbar = 0 'False
+ StartUpPosition = 1 'CenterOwner
+ Tag = "About NdbCPC"
+ Begin VB.PictureBox picIcon
+ AutoSize = -1 'True
+ BackColor = &H00C0C0C0&
+ ClipControls = 0 'False
+ Height = 540
+ Left = 240
+ Picture = "frmAbout.frx":0000
+ ScaleHeight = 480
+ ScaleMode = 0 'User
+ ScaleWidth = 480
+ TabIndex = 2
+ TabStop = 0 'False
+ Top = 240
+ Width = 540
+ End
+ Begin VB.CommandButton cmdOK
+ Cancel = -1 'True
+ Caption = "OK"
+ Default = -1 'True
+ Height = 345
+ Left = 4245
+ TabIndex = 0
+ Tag = "OK"
+ Top = 2625
+ Width = 1467
+ End
+ Begin VB.CommandButton cmdSysInfo
+ Caption = "&System Info..."
+ Height = 345
+ Left = 4260
+ TabIndex = 1
+ Tag = "&System Info..."
+ Top = 3075
+ Width = 1452
+ End
+ Begin VB.Label lblDescription
+ Caption = "App Description"
+ ForeColor = &H00000000&
+ Height = 1170
+ Left = 1050
+ TabIndex = 6
+ Tag = "App Description"
+ Top = 1125
+ Width = 4092
+ End
+ Begin VB.Label lblTitle
+ Caption = "Application Title"
+ ForeColor = &H00000000&
+ Height = 480
+ Left = 1050
+ TabIndex = 5
+ Tag = "Application Title"
+ Top = 240
+ Width = 4092
+ End
+ Begin VB.Line Line1
+ BorderColor = &H00808080&
+ BorderStyle = 6 'Inside Solid
+ Index = 1
+ X1 = 225
+ X2 = 5657
+ Y1 = 2430
+ Y2 = 2430
+ End
+ Begin VB.Line Line1
+ BorderColor = &H00FFFFFF&
+ BorderWidth = 2
+ Index = 0
+ X1 = 240
+ X2 = 5657
+ Y1 = 2445
+ Y2 = 2445
+ End
+ Begin VB.Label lblVersion
+ Caption = "Version"
+ Height = 225
+ Left = 1050
+ TabIndex = 4
+ Tag = "Version"
+ Top = 780
+ Width = 4092
+ End
+ Begin VB.Label lblDisclaimer
+ Caption = "Warning: ..."
+ ForeColor = &H00000000&
+ Height = 825
+ Left = 255
+ TabIndex = 3
+ Tag = "Warning: ..."
+ Top = 2625
+ Width = 3870
+ End
+End
+Attribute VB_Name = "frmAbout"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = False
+Attribute VB_PredeclaredId = True
+Attribute VB_Exposed = False
+' Reg Key Security Options...
+Const KEY_ALL_ACCESS = &H2003F
+
+
+' Reg Key ROOT Types...
+Const HKEY_LOCAL_MACHINE = &H80000002
+Const ERROR_SUCCESS = 0
+Const REG_SZ = 1 ' Unicode nul terminated string
+Const REG_DWORD = 4 ' 32-bit number
+
+
+Const gREGKEYSYSINFOLOC = "SOFTWARE\Microsoft\Shared Tools Location"
+Const gREGVALSYSINFOLOC = "MSINFO"
+Const gREGKEYSYSINFO = "SOFTWARE\Microsoft\Shared Tools\MSINFO"
+Const gREGVALSYSINFO = "PATH"
+
+
+Private Declare Function RegOpenKeyEx Lib "advapi32" Alias "RegOpenKeyExA" (ByVal hKey As Long, ByVal lpSubKey As String, ByVal ulOptions As Long, ByVal samDesired As Long, ByRef phkResult As Long) As Long
+Private Declare Function RegQueryValueEx Lib "advapi32" Alias "RegQueryValueExA" (ByVal hKey As Long, ByVal lpValueName As String, ByVal lpReserved As Long, ByRef lpType As Long, ByVal lpData As String, ByRef lpcbData As Long) As Long
+Private Declare Function RegCloseKey Lib "advapi32" (ByVal hKey As Long) As Long
+
+Private Sub Form_Load()
+ lblVersion.Caption = "Version " & App.Major & "." & App.Minor & "." & App.Revision
+ lblTitle.Caption = App.Title
+End Sub
+
+
+
+Private Sub cmdSysInfo_Click()
+ Call StartSysInfo
+End Sub
+
+
+Private Sub cmdOK_Click()
+ Unload Me
+End Sub
+
+
+Public Sub StartSysInfo()
+ On Error GoTo SysInfoErr
+
+
+ Dim rc As Long
+ Dim SysInfoPath As String
+
+
+ ' Try To Get System Info Program Path\Name From Registry...
+ If GetKeyValue(HKEY_LOCAL_MACHINE, gREGKEYSYSINFO, gREGVALSYSINFO, SysInfoPath) Then
+ ' Try To Get System Info Program Path Only From Registry...
+ ElseIf GetKeyValue(HKEY_LOCAL_MACHINE, gREGKEYSYSINFOLOC, gREGVALSYSINFOLOC, SysInfoPath) Then
+ ' Validate Existance Of Known 32 Bit File Version
+ If (Dir(SysInfoPath & "\MSINFO32.EXE") <> "") Then
+ SysInfoPath = SysInfoPath & "\MSINFO32.EXE"
+
+
+ ' Error - File Can Not Be Found...
+ Else
+ GoTo SysInfoErr
+ End If
+ ' Error - Registry Entry Can Not Be Found...
+ Else
+ GoTo SysInfoErr
+ End If
+
+
+ Call Shell(SysInfoPath, vbNormalFocus)
+
+
+ Exit Sub
+SysInfoErr:
+ MsgBox "System Information Is Unavailable At This Time", vbOKOnly
+End Sub
+
+
+Public Function GetKeyValue(KeyRoot As Long, KeyName As String, SubKeyRef As String, ByRef KeyVal As String) As Boolean
+ Dim i As Long ' Loop Counter
+ Dim rc As Long ' Return Code
+ Dim hKey As Long ' Handle To An Open Registry Key
+ Dim hDepth As Long '
+ Dim KeyValType As Long ' Data Type Of A Registry Key
+ Dim tmpVal As String ' Tempory Storage For A Registry Key Value
+ Dim KeyValSize As Long ' Size Of Registry Key Variable
+ '------------------------------------------------------------
+ ' Open RegKey Under KeyRoot {HKEY_LOCAL_MACHINE...}
+ '------------------------------------------------------------
+ rc = RegOpenKeyEx(KeyRoot, KeyName, 0, KEY_ALL_ACCESS, hKey) ' Open Registry Key
+
+
+ If (rc <> ERROR_SUCCESS) Then GoTo GetKeyError ' Handle Error...
+
+
+ tmpVal = String$(1024, 0) ' Allocate Variable Space
+ KeyValSize = 1024 ' Mark Variable Size
+
+
+ '------------------------------------------------------------
+ ' Retrieve Registry Key Value...
+ '------------------------------------------------------------
+ rc = RegQueryValueEx(hKey, SubKeyRef, 0, KeyValType, tmpVal, KeyValSize) ' Get/Create Key Value
+
+
+ If (rc <> ERROR_SUCCESS) Then GoTo GetKeyError ' Handle Errors
+
+
+ tmpVal = VBA.Left(tmpVal, InStr(tmpVal, VBA.Chr(0)) - 1)
+ '------------------------------------------------------------
+ ' Determine Key Value Type For Conversion...
+ '------------------------------------------------------------
+ Select Case KeyValType ' Search Data Types...
+ Case REG_SZ ' String Registry Key Data Type
+ KeyVal = tmpVal ' Copy String Value
+ Case REG_DWORD ' Double Word Registry Key Data Type
+ For i = Len(tmpVal) To 1 Step -1 ' Convert Each Bit
+ KeyVal = KeyVal + Hex(Asc(Mid(tmpVal, i, 1))) ' Build Value Char. By Char.
+ Next
+ KeyVal = Format$("&h" + KeyVal) ' Convert Double Word To String
+ End Select
+
+
+ GetKeyValue = True ' Return Success
+ rc = RegCloseKey(hKey) ' Close Registry Key
+ Exit Function ' Exit
+
+
+GetKeyError: ' Cleanup After An Error Has Occured...
+ KeyVal = "" ' Set Return Val To Empty String
+ GetKeyValue = False ' Return Failure
+ rc = RegCloseKey(hKey) ' Close Registry Key
+End Function
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmLogin.frm b/ndb/src/cw/cpcc-win32/vb6/frmLogin.frm
new file mode 100644
index 00000000000..d4d663c93c2
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmLogin.frm
@@ -0,0 +1,119 @@
+VERSION 5.00
+Begin VB.Form frmLogin
+ BorderStyle = 3 'Fixed Dialog
+ Caption = "Login"
+ ClientHeight = 1590
+ ClientLeft = 45
+ ClientTop = 330
+ ClientWidth = 3750
+ LinkTopic = "Form1"
+ MaxButton = 0 'False
+ MinButton = 0 'False
+ ScaleHeight = 1590
+ ScaleWidth = 3750
+ ShowInTaskbar = 0 'False
+ StartUpPosition = 2 'CenterScreen
+ Tag = "Login"
+ Begin VB.CommandButton cmdCancel
+ Cancel = -1 'True
+ Caption = "Cancel"
+ Height = 360
+ Left = 2100
+ TabIndex = 5
+ Tag = "Cancel"
+ Top = 1020
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdOK
+ Caption = "OK"
+ Default = -1 'True
+ Height = 360
+ Left = 495
+ TabIndex = 4
+ Tag = "OK"
+ Top = 1020
+ Width = 1140
+ End
+ Begin VB.TextBox txtPassword
+ Height = 285
+ IMEMode = 3 'DISABLE
+ Left = 1305
+ PasswordChar = "*"
+ TabIndex = 1
+ Top = 525
+ Width = 2325
+ End
+ Begin VB.TextBox txtUserName
+ Height = 285
+ Left = 1305
+ TabIndex = 3
+ Top = 135
+ Width = 2325
+ End
+ Begin VB.Label lblLabels
+ Caption = "&Password:"
+ Height = 248
+ Index = 1
+ Left = 105
+ TabIndex = 0
+ Tag = "&Password:"
+ Top = 540
+ Width = 1080
+ End
+ Begin VB.Label lblLabels
+ Caption = "&User Name:"
+ Height = 248
+ Index = 0
+ Left = 105
+ TabIndex = 2
+ Tag = "&User Name:"
+ Top = 150
+ Width = 1080
+ End
+End
+Attribute VB_Name = "frmLogin"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = False
+Attribute VB_PredeclaredId = True
+Attribute VB_Exposed = False
+Private Declare Function GetUserName Lib "advapi32.dll" Alias "GetUserNameA" (ByVal lpbuffer As String, nSize As Long) As Long
+
+
+Public OK As Boolean
+Private Sub Form_Load()
+ Dim sBuffer As String
+ Dim lSize As Long
+
+
+ sBuffer = Space$(255)
+ lSize = Len(sBuffer)
+ Call GetUserName(sBuffer, lSize)
+ If lSize > 0 Then
+ txtUserName.Text = Left$(sBuffer, lSize)
+ Else
+ txtUserName.Text = vbNullString
+ End If
+End Sub
+
+
+
+Private Sub cmdCancel_Click()
+ OK = False
+ Me.Hide
+End Sub
+
+
+Private Sub cmdOK_Click()
+ 'ToDo: create test for correct password
+ 'check for correct password
+ If txtPassword.Text = "" Then
+ OK = True
+ Me.Hide
+ Else
+ MsgBox "Invalid Password, try again!", , "Login"
+ txtPassword.SetFocus
+ txtPassword.SelStart = 0
+ txtPassword.SelLength = Len(txtPassword.Text)
+ End If
+End Sub
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmMain.frm b/ndb/src/cw/cpcc-win32/vb6/frmMain.frm
new file mode 100644
index 00000000000..a4bf5b58941
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmMain.frm
@@ -0,0 +1,1207 @@
+VERSION 5.00
+Object = "{831FDD16-0C5C-11D2-A9FC-0000F8754DA1}#2.0#0"; "mscomctl.ocx"
+Begin VB.Form frmMain
+ Caption = "NdbCPC"
+ ClientHeight = 5955
+ ClientLeft = 2115
+ ClientTop = 2250
+ ClientWidth = 8880
+ LinkTopic = "Form1"
+ ScaleHeight = 5955
+ ScaleWidth = 8880
+ Begin MSComctlLib.ImageList ImageList1
+ Left = 6840
+ Top = 3120
+ _ExtentX = 1005
+ _ExtentY = 1005
+ BackColor = 16777215
+ ImageWidth = 16
+ ImageHeight = 16
+ MaskColor = 12632256
+ _Version = 393216
+ BeginProperty Images {2C247F25-8591-11D1-B16A-00C0F0283628}
+ NumListImages = 11
+ BeginProperty ListImage1 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":0000
+ Key = "close"
+ EndProperty
+ BeginProperty ListImage2 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":27B4
+ Key = "open"
+ EndProperty
+ BeginProperty ListImage3 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":4F68
+ Key = "computer_unknown"
+ EndProperty
+ BeginProperty ListImage4 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":5284
+ Key = "computer_stopped"
+ EndProperty
+ BeginProperty ListImage5 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":55A0
+ Key = "computer_started"
+ EndProperty
+ BeginProperty ListImage6 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":58BC
+ Key = ""
+ EndProperty
+ BeginProperty ListImage7 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":5BD8
+ Key = ""
+ EndProperty
+ BeginProperty ListImage8 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":5EF4
+ Key = ""
+ EndProperty
+ BeginProperty ListImage9 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":6210
+ Key = "db"
+ EndProperty
+ BeginProperty ListImage10 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":652A
+ Key = "computer"
+ EndProperty
+ BeginProperty ListImage11 {2C247F27-8591-11D1-B16A-00C0F0283628}
+ Picture = "frmMain.frx":6844
+ Key = "properties"
+ EndProperty
+ EndProperty
+ End
+ Begin VB.PictureBox picSplitter
+ BackColor = &H00808080&
+ BorderStyle = 0 'None
+ FillColor = &H00808080&
+ Height = 4800
+ Left = 5400
+ ScaleHeight = 2090.126
+ ScaleMode = 0 'User
+ ScaleWidth = 780
+ TabIndex = 6
+ Top = 705
+ Width = 72
+ Visible = 0 'False
+ End
+ Begin MSComctlLib.TreeView tvTreeView
+ Height = 4800
+ Left = 0
+ TabIndex = 5
+ Top = 705
+ Width = 2016
+ _ExtentX = 3545
+ _ExtentY = 8467
+ _Version = 393217
+ HideSelection = 0 'False
+ Indentation = 0
+ LineStyle = 1
+ Sorted = -1 'True
+ Style = 7
+ FullRowSelect = -1 'True
+ ImageList = "ImageList1"
+ Appearance = 1
+ End
+ Begin VB.PictureBox picTitles
+ Align = 1 'Align Top
+ Appearance = 0 'Flat
+ BorderStyle = 0 'None
+ ForeColor = &H80000008&
+ Height = 300
+ Left = 0
+ ScaleHeight = 300
+ ScaleWidth = 8880
+ TabIndex = 2
+ TabStop = 0 'False
+ Top = 420
+ Width = 8880
+ Begin VB.Label lblTitle
+ BorderStyle = 1 'Fixed Single
+ Caption = " ListView:"
+ Height = 270
+ Index = 1
+ Left = 2078
+ TabIndex = 4
+ Tag = " ListView:"
+ Top = 12
+ Width = 3216
+ End
+ Begin VB.Label lblTitle
+ BorderStyle = 1 'Fixed Single
+ Caption = " TreeView:"
+ Height = 270
+ Index = 0
+ Left = 0
+ TabIndex = 3
+ Tag = " TreeView:"
+ Top = 12
+ Width = 2016
+ End
+ End
+ Begin MSComctlLib.Toolbar tbToolBar
+ Align = 1 'Align Top
+ Height = 420
+ Left = 0
+ TabIndex = 1
+ Top = 0
+ Width = 8880
+ _ExtentX = 15663
+ _ExtentY = 741
+ ButtonWidth = 609
+ ButtonHeight = 582
+ Appearance = 1
+ ImageList = "ImageList1"
+ _Version = 393216
+ BeginProperty Buttons {66833FE8-8583-11D1-B16A-00C0F0283628}
+ NumButtons = 5
+ BeginProperty Button1 {66833FEA-8583-11D1-B16A-00C0F0283628}
+ Style = 3
+ EndProperty
+ BeginProperty Button2 {66833FEA-8583-11D1-B16A-00C0F0283628}
+ Key = "Add computer"
+ Object.ToolTipText = "Add computer"
+ ImageKey = "computer"
+ EndProperty
+ BeginProperty Button3 {66833FEA-8583-11D1-B16A-00C0F0283628}
+ Key = "New database"
+ Object.ToolTipText = "New database"
+ ImageKey = "db"
+ EndProperty
+ BeginProperty Button4 {66833FEA-8583-11D1-B16A-00C0F0283628}
+ Style = 3
+ EndProperty
+ BeginProperty Button5 {66833FEA-8583-11D1-B16A-00C0F0283628}
+ Key = "Properties"
+ Object.ToolTipText = "Properties"
+ ImageKey = "properties"
+ EndProperty
+ EndProperty
+ End
+ Begin MSComctlLib.StatusBar sbStatusBar
+ Align = 2 'Align Bottom
+ Height = 270
+ Left = 0
+ TabIndex = 0
+ Top = 5685
+ Width = 8880
+ _ExtentX = 15663
+ _ExtentY = 476
+ _Version = 393216
+ BeginProperty Panels {8E3867A5-8586-11D1-B16A-00C0F0283628}
+ NumPanels = 3
+ BeginProperty Panel1 {8E3867AB-8586-11D1-B16A-00C0F0283628}
+ AutoSize = 1
+ Object.Width = 10028
+ Text = "Status"
+ TextSave = "Status"
+ EndProperty
+ BeginProperty Panel2 {8E3867AB-8586-11D1-B16A-00C0F0283628}
+ Style = 6
+ AutoSize = 2
+ TextSave = "2002-10-15"
+ EndProperty
+ BeginProperty Panel3 {8E3867AB-8586-11D1-B16A-00C0F0283628}
+ Style = 5
+ AutoSize = 2
+ TextSave = "09:44"
+ EndProperty
+ EndProperty
+ End
+ Begin MSComctlLib.ListView lvProcesses
+ Height = 4815
+ Left = 2040
+ TabIndex = 8
+ Top = 720
+ Width = 3255
+ _ExtentX = 5741
+ _ExtentY = 8493
+ Sorted = -1 'True
+ MultiSelect = -1 'True
+ LabelWrap = -1 'True
+ HideSelection = 0 'False
+ AllowReorder = -1 'True
+ FullRowSelect = -1 'True
+ _Version = 393217
+ ForeColor = -2147483640
+ BackColor = -2147483643
+ BorderStyle = 1
+ Appearance = 1
+ NumItems = 6
+ BeginProperty ColumnHeader(1) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ Key = "Id"
+ Text = "Id"
+ Object.Width = 2540
+ EndProperty
+ BeginProperty ColumnHeader(2) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ SubItemIndex = 1
+ Key = "Computer"
+ Text = "Computer"
+ Object.Width = 2540
+ EndProperty
+ BeginProperty ColumnHeader(3) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ SubItemIndex = 2
+ Key = "Database"
+ Text = "Database"
+ Object.Width = 2540
+ EndProperty
+ BeginProperty ColumnHeader(4) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ SubItemIndex = 3
+ Key = "Name"
+ Text = "Name"
+ Object.Width = 2540
+ EndProperty
+ BeginProperty ColumnHeader(5) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ SubItemIndex = 4
+ Key = "Status"
+ Text = "Status"
+ Object.Width = 2540
+ EndProperty
+ BeginProperty ColumnHeader(6) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ SubItemIndex = 5
+ Key = "Owner"
+ Text = "Owner"
+ Object.Width = 2540
+ EndProperty
+ End
+ Begin MSComctlLib.ListView lvComputers
+ Height = 4815
+ Left = 2040
+ TabIndex = 7
+ Top = 720
+ Width = 3255
+ _ExtentX = 5741
+ _ExtentY = 8493
+ Sorted = -1 'True
+ MultiSelect = -1 'True
+ LabelWrap = -1 'True
+ HideSelection = -1 'True
+ AllowReorder = -1 'True
+ FullRowSelect = -1 'True
+ _Version = 393217
+ Icons = "ImageList1"
+ SmallIcons = "ImageList1"
+ ForeColor = -2147483640
+ BackColor = -2147483643
+ BorderStyle = 1
+ Appearance = 1
+ NumItems = 2
+ BeginProperty ColumnHeader(1) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ Text = "Computer"
+ Object.Width = 2540
+ EndProperty
+ BeginProperty ColumnHeader(2) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ SubItemIndex = 1
+ Text = "Status"
+ Object.Width = 2540
+ EndProperty
+ End
+ Begin MSComctlLib.ListView lvDatabases
+ Height = 4815
+ Left = 2040
+ TabIndex = 9
+ Top = 720
+ Width = 3255
+ _ExtentX = 5741
+ _ExtentY = 8493
+ View = 3
+ Sorted = -1 'True
+ MultiSelect = -1 'True
+ LabelWrap = -1 'True
+ HideSelection = -1 'True
+ AllowReorder = -1 'True
+ FullRowSelect = -1 'True
+ _Version = 393217
+ Icons = "ImageList1"
+ SmallIcons = "ImageList1"
+ ForeColor = -2147483640
+ BackColor = -2147483643
+ BorderStyle = 1
+ Appearance = 1
+ NumItems = 2
+ BeginProperty ColumnHeader(1) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ Key = "Database"
+ Text = "Database"
+ Object.Width = 2540
+ EndProperty
+ BeginProperty ColumnHeader(2) {BDD1F052-858B-11D1-B16A-00C0F0283628}
+ SubItemIndex = 1
+ Key = "Status"
+ Text = "Status"
+ Object.Width = 2540
+ EndProperty
+ End
+ Begin VB.Image imgSplitter
+ Height = 4788
+ Left = 1965
+ MousePointer = 9 'Size W E
+ Top = 705
+ Width = 150
+ End
+ Begin VB.Menu mnuFile
+ Caption = "&File"
+ Begin VB.Menu mnuFileOpen
+ Caption = "&Open..."
+ End
+ Begin VB.Menu mnuFileFind
+ Caption = "&Find"
+ End
+ Begin VB.Menu mnuFileBar0
+ Caption = "-"
+ End
+ Begin VB.Menu mnuFileSendTo
+ Caption = "Sen&d to"
+ End
+ Begin VB.Menu mnuFileBar1
+ Caption = "-"
+ End
+ Begin VB.Menu mnuFileNew
+ Caption = "&New"
+ Shortcut = ^N
+ End
+ Begin VB.Menu mnuFileBar2
+ Caption = "-"
+ End
+ Begin VB.Menu mnuFileDelete
+ Caption = "&Delete"
+ End
+ Begin VB.Menu mnuFileRename
+ Caption = "Rena&me"
+ End
+ Begin VB.Menu mnuFileProperties
+ Caption = "Propert&ies"
+ End
+ Begin VB.Menu mnuFileBar3
+ Caption = "-"
+ End
+ Begin VB.Menu mnuFileMRU
+ Caption = ""
+ Index = 1
+ Visible = 0 'False
+ End
+ Begin VB.Menu mnuFileMRU
+ Caption = ""
+ Index = 2
+ Visible = 0 'False
+ End
+ Begin VB.Menu mnuFileMRU
+ Caption = ""
+ Index = 3
+ Visible = 0 'False
+ End
+ Begin VB.Menu mnuFileBar4
+ Caption = "-"
+ Visible = 0 'False
+ End
+ Begin VB.Menu mnuFileBar5
+ Caption = "-"
+ End
+ Begin VB.Menu mnuFileClose
+ Caption = "&Close"
+ End
+ End
+ Begin VB.Menu mnuEdit
+ Caption = "&Edit"
+ Begin VB.Menu mnuEditUndo
+ Caption = "&Undo"
+ End
+ Begin VB.Menu mnuEditBar0
+ Caption = "-"
+ End
+ Begin VB.Menu mnuEditCut
+ Caption = "Cu&t"
+ Shortcut = ^X
+ End
+ Begin VB.Menu mnuEditCopy
+ Caption = "&Copy"
+ Shortcut = ^C
+ End
+ Begin VB.Menu mnuEditPaste
+ Caption = "&Paste"
+ Shortcut = ^V
+ End
+ Begin VB.Menu mnuEditPasteSpecial
+ Caption = "Paste &Special..."
+ End
+ Begin VB.Menu mnuEditBar1
+ Caption = "-"
+ End
+ Begin VB.Menu mnuEditSelectAll
+ Caption = "Select &All"
+ Shortcut = ^A
+ End
+ Begin VB.Menu mnuEditInvertSelection
+ Caption = "&Invert Selection"
+ End
+ End
+ Begin VB.Menu mnuView
+ Caption = "&View"
+ Begin VB.Menu mnuViewToolbar
+ Caption = "&Toolbar"
+ Checked = -1 'True
+ End
+ Begin VB.Menu mnuViewStatusBar
+ Caption = "Status &Bar"
+ Checked = -1 'True
+ End
+ Begin VB.Menu mnuViewBar0
+ Caption = "-"
+ End
+ Begin VB.Menu mnuListViewMode
+ Caption = "Lar&ge Icons"
+ Index = 0
+ End
+ Begin VB.Menu mnuListViewMode
+ Caption = "S&mall Icons"
+ Index = 1
+ End
+ Begin VB.Menu mnuListViewMode
+ Caption = "&List"
+ Index = 2
+ End
+ Begin VB.Menu mnuListViewMode
+ Caption = "&Details"
+ Index = 3
+ End
+ Begin VB.Menu mnuViewBar1
+ Caption = "-"
+ End
+ Begin VB.Menu mnuViewArrangeIcons
+ Caption = "Arrange &Icons"
+ End
+ Begin VB.Menu mnuViewBar2
+ Caption = "-"
+ End
+ Begin VB.Menu mnuViewRefresh
+ Caption = "&Refresh"
+ End
+ Begin VB.Menu mnuViewOptions
+ Caption = "&Options..."
+ End
+ Begin VB.Menu mnuViewWebBrowser
+ Caption = "&Web Browser"
+ End
+ End
+ Begin VB.Menu mnuHelp
+ Caption = "&Help"
+ Begin VB.Menu mnuHelpContents
+ Caption = "&Contents"
+ End
+ Begin VB.Menu mnuHelpSearchForHelpOn
+ Caption = "&Search For Help On..."
+ End
+ Begin VB.Menu mnuHelpBar0
+ Caption = "-"
+ End
+ Begin VB.Menu mnuHelpAbout
+ Caption = "&About "
+ End
+ End
+ Begin VB.Menu mnuPopComputers
+ Caption = ""
+ Visible = 0 'False
+ Begin VB.Menu mnuPopAddComputer
+ Caption = "Add computer"
+ End
+ Begin VB.Menu mnuPop__
+ Caption = "-"
+ End
+ Begin VB.Menu mnuPopSortComputers
+ Caption = "Sorted"
+ End
+ End
+ Begin VB.Menu mnuPopDatabases
+ Caption = ""
+ Visible = 0 'False
+ Begin VB.Menu mnuPopNewDatabase
+ Caption = "New database"
+ End
+ Begin VB.Menu mnuPopSortDatabases0
+ Caption = "-"
+ End
+ Begin VB.Menu mnuPopSortDatabases
+ Caption = "Sorted"
+ End
+ End
+ Begin VB.Menu mnuPopComputer
+ Caption = ""
+ Visible = 0 'False
+ Begin VB.Menu mnuPopComputerName
+ Caption = "ComputerName"
+ Enabled = 0 'False
+ End
+ Begin VB.Menu mnuPopComputer0
+ Caption = "-"
+ End
+ Begin VB.Menu mnuPopConnectComputer
+ Caption = "Connect"
+ End
+ Begin VB.Menu mnuPopDisconnectComputer
+ Caption = "Disconnect"
+ End
+ Begin VB.Menu mnuPopRemoveComputer
+ Caption = "Remove"
+ End
+ Begin VB.Menu mnuComputer1
+ Caption = "-"
+ End
+ Begin VB.Menu mnuPopComputerProperties
+ Caption = "Properties"
+ End
+ End
+End
+Attribute VB_Name = "frmMain"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = False
+Attribute VB_PredeclaredId = True
+Attribute VB_Exposed = False
+Option Explicit
+Private Declare Function OSWinHelp% Lib "user32" Alias "WinHelpA" (ByVal hwnd&, ByVal HelpFile$, ByVal wCommand%, dwData As Any)
+
+Dim mbMoving As Boolean
+Const sglSplitLimit = 500
+Dim m_currentNode As MSComctlLib.Node
+Dim m_currentList As ListView
+
+Dim m_currentView As Integer
+Dim m_computerWidth As Integer
+Dim m_databaseWidth As Integer
+
+Dim m_currentComputer As Computer
+Dim m_currentDatabase As Database_
+
+Private Sub Form_Load()
+ tvTreeView.Nodes.Clear
+ lvComputers.ListItems.Clear
+ lvProcesses.ListItems.Clear
+ lvDatabases.ListItems.Clear
+
+ Me.Left = GetSetting(App.Title, "Settings", "MainLeft", 1000)
+ Me.Top = GetSetting(App.Title, "Settings", "MainTop", 1000)
+ Me.Width = GetSetting(App.Title, "Settings", "MainWidth", 6500)
+ Me.Height = GetSetting(App.Title, "Settings", "MainHeight", 6500)
+
+ tvTreeView.Nodes.Add , tvwChild, "Computers", "Computers", 1, 2
+ Dim c As Computer
+ For Each c In g_computers
+ addComputer c
+ Next
+
+ Set m_currentNode = tvTreeView.Nodes("Computers")
+ Set m_currentList = lvComputers
+
+ tvTreeView.Nodes.Add , tvwChild, "Databases", "Databases", 1, 2
+ Dim d As Database_
+ For Each d In g_databases
+ AddDatabase d
+ Next
+
+ lvComputers.Visible = True
+ lvProcesses.Visible = False
+ lvDatabases.Visible = False
+ lvComputers.View = lvwReport
+ lvProcesses.View = lvwReport
+ lvDatabases.View = lvwReport
+ m_computerWidth = lvProcesses.ColumnHeaders("Computer").Width
+ m_databaseWidth = lvProcesses.ColumnHeaders("Database").Width
+ lvProcesses.ColumnHeaders("Id").Width = 0
+End Sub
+
+Private Sub setComputer(ByVal f_ip As String)
+ Dim c As Computer
+ Set c = g_computers(f_ip)
+ If c Is Nothing Then
+ MsgBox "Unknown computer: " & f_ip
+ Exit Sub
+ End If
+
+ Set m_currentComputer = c
+
+ lblTitle(1).Caption = "Processes defined on computer: " & c.m_name
+ setProcesses c.m_processes
+
+ If lvProcesses.ColumnHeaders("Computer").Width <> 0 Then
+ m_computerWidth = lvProcesses.ColumnHeaders("Computer").Width
+ lvProcesses.ColumnHeaders("Computer").Width = 0
+ End If
+
+ If lvProcesses.ColumnHeaders("Database").Width = 0 Then
+ lvProcesses.ColumnHeaders("Database").Width = m_databaseWidth
+ End If
+End Sub
+
+Private Sub setDatabase(ByVal f_name As String)
+ Dim c As Database_
+ Set c = g_databases(f_name)
+ If c Is Nothing Then
+ MsgBox "Unknown database: " & f_name
+ Exit Sub
+ End If
+
+ Set m_currentDatabase = c
+
+ lblTitle(1).Caption = "Processes defined for database: " & c.m_name
+ setProcesses c.m_processes
+
+ If lvProcesses.ColumnHeaders("Database").Width <> 0 Then
+ m_databaseWidth = lvProcesses.ColumnHeaders("Database").Width
+ lvProcesses.ColumnHeaders("Database").Width = 0
+ End If
+
+ If lvProcesses.ColumnHeaders("Computer").Width = 0 Then
+ lvProcesses.ColumnHeaders("Computer").Width = m_computerWidth
+ End If
+
+End Sub
+
+Private Sub setProcesses(ByRef c As Collection)
+ lvProcesses.ListItems.Clear
+ Dim p As Process
+ For Each p In c
+ Dim li As ListItem
+ Set li = lvProcesses.ListItems.Add(, "_" & p.m_computer.m_name & "_" & p.m_id, p.m_id)
+ li.SubItems(1) = p.m_computer.m_name
+ li.SubItems(2) = p.m_database
+ li.SubItems(3) = p.m_name
+ li.SubItems(4) = p.m_status
+ li.SubItems(5) = p.m_owner
+ Next
+End Sub
+
+Public Sub addComputer(ByRef c As Computer)
+ Dim icon As Integer
+ Select Case c.m_status
+ Case "No contact"
+ icon = 4
+ Case "Connected"
+ icon = 5
+ Case Else
+ icon = 3
+ End Select
+
+ Dim li As ListItem
+ Set li = lvComputers.ListItems.Add(, "_" & c.m_name, c.m_name, icon, icon)
+ li.SubItems(1) = c.m_status
+
+ tvTreeView.Nodes.Add "Computers", tvwChild, "_" & c.m_name, c.m_name, icon, icon
+End Sub
+
+Public Sub removeComputer(ByRef name As String)
+ lvComputers.ListItems.Remove "_" & name
+ tvTreeView.Nodes.Remove "_" & name
+
+ '
+ ' Check if should remove database
+ Dim c As Computer
+ Set c = g_computers("_" & name)
+ Dim db As Database_
+ Dim dbs As New Collection
+ Dim p As Process
+ For Each p In c.m_processes
+ Set db = g_databases("_" & p.m_database)
+ db.m_processes.Remove "_" & p.m_computer.m_name & "_" & p.m_id
+ If Not Exists(dbs, p.m_database) Then dbs.Add db, p.m_database
+ Next
+
+ For Each db In dbs
+ If db.m_processes.Count = 0 Then
+ g_databases.Remove "_" & db.m_name
+ tvTreeView.Nodes.Remove "_" & db.m_name
+ End If
+ Next
+
+ g_computers.Remove "_" & name
+
+ '
+ ' Check if should remove database
+
+ Dim n As MSComctlLib.Node
+ Set n = tvTreeView.SelectedItem
+ selectNode n
+End Sub
+
+Private Sub AddDatabase(ByRef c As Database_)
+ Dim li As ListItem
+ Set li = lvDatabases.ListItems.Add(, "_" & c.m_name, c.m_name, 9, 9)
+ li.SubItems(1) = c.m_status
+ tvTreeView.Nodes.Add "Databases", tvwChild, "_" & c.m_name, c.m_name, 9, 9
+End Sub
+
+Private Sub Form_Unload(Cancel As Integer)
+ Dim i As Integer
+
+
+ 'close all sub forms
+ For i = Forms.Count - 1 To 1 Step -1
+ Unload Forms(i)
+ Next
+ If Me.WindowState <> vbMinimized Then
+ SaveSetting App.Title, "Settings", "MainLeft", Me.Left
+ SaveSetting App.Title, "Settings", "MainTop", Me.Top
+ SaveSetting App.Title, "Settings", "MainWidth", Me.Width
+ SaveSetting App.Title, "Settings", "MainHeight", Me.Height
+ End If
+End Sub
+
+Private Sub Form_Resize()
+ On Error Resume Next
+ If Me.Width < 3000 Then Me.Width = 3000
+ SizeControls imgSplitter.Left
+End Sub
+
+Private Sub imgSplitter_MouseDown(Button As Integer, Shift As Integer, X As Single, Y As Single)
+ With imgSplitter
+ picSplitter.Move .Left, .Top, .Width \ 2, .Height - 20
+ End With
+ picSplitter.Visible = True
+ mbMoving = True
+End Sub
+
+Private Sub imgSplitter_MouseMove(Button As Integer, Shift As Integer, X As Single, Y As Single)
+ Dim sglPos As Single
+
+
+ If mbMoving Then
+ sglPos = X + imgSplitter.Left
+ If sglPos < sglSplitLimit Then
+ picSplitter.Left = sglSplitLimit
+ ElseIf sglPos > Me.Width - sglSplitLimit Then
+ picSplitter.Left = Me.Width - sglSplitLimit
+ Else
+ picSplitter.Left = sglPos
+ End If
+ End If
+End Sub
+
+
+Private Sub imgSplitter_MouseUp(Button As Integer, Shift As Integer, X As Single, Y As Single)
+ SizeControls picSplitter.Left
+ picSplitter.Visible = False
+ mbMoving = False
+End Sub
+
+
+Private Sub TreeView1_DragDrop(Source As Control, X As Single, Y As Single)
+ If Source = imgSplitter Then
+ SizeControls X
+ End If
+End Sub
+
+
+Sub SizeControls(X As Single)
+ On Error Resume Next
+
+ 'set the width
+ If X < 1500 Then X = 1500
+ If X > (Me.Width - 1500) Then X = Me.Width - 1500
+ tvTreeView.Width = X
+ imgSplitter.Left = X
+
+ Dim t_left, t_width As Integer
+ t_left = X + 40
+ t_width = Me.Width - (tvTreeView.Width + 140)
+
+ lblTitle(0).Width = tvTreeView.Width
+ lblTitle(1).Left = t_left + 20
+ lblTitle(1).Width = t_width - 40
+
+
+ 'set the top
+ If tbToolBar.Visible Then
+ tvTreeView.Top = tbToolBar.Height + picTitles.Height
+ Else
+ tvTreeView.Top = picTitles.Height
+ End If
+
+
+ 'set the height
+ If sbStatusBar.Visible Then
+ tvTreeView.Height = Me.ScaleHeight - (picTitles.Top + picTitles.Height + sbStatusBar.Height)
+ Else
+ tvTreeView.Height = Me.ScaleHeight - (picTitles.Top + picTitles.Height)
+ End If
+
+
+ imgSplitter.Top = tvTreeView.Top
+ imgSplitter.Height = tvTreeView.Height
+
+ setListDimensions t_left, t_width, tvTreeView.Top, tvTreeView.Height
+End Sub
+
+Private Sub setListView(ByVal f_View As Integer)
+ lvComputers.View = f_View
+ lvProcesses.View = f_View
+End Sub
+
+Private Sub setListDimensions(ByVal f_Left As Integer, ByVal f_Width As Integer, ByVal f_Top As Integer, ByVal f_Height As Integer)
+ With lvComputers
+ .Left = f_Left
+ .Width = f_Width
+ .Top = f_Top
+ .Height = f_Height
+ End With
+ With lvProcesses
+ .Left = f_Left
+ .Width = f_Width
+ .Top = f_Top
+ .Height = f_Height
+ End With
+ With lvDatabases
+ .Left = f_Left
+ .Width = f_Width
+ .Top = f_Top
+ .Height = f_Height
+ End With
+End Sub
+
+Private Sub tbToolBar_ButtonClick(ByVal Button As MSComctlLib.Button)
+ On Error Resume Next
+ Select Case Button.Key
+ Case "New database"
+ 'ToDo: Add 'Back' button code.
+ mnuPopNewDatabase_Click
+ Case "Add computer"
+ 'ToDo: Add 'Forward' button code.
+ frmNewComputer.Show vbModal, Me
+ Dim c As Computer
+ For Each c In frmNewComputer.m_hosts
+ addComputer c
+ g_computers.Add c, "_" & c.m_name
+ Next
+ Case "Properties"
+ mnuFileProperties_Click
+ End Select
+End Sub
+
+Private Sub mnuHelpAbout_Click()
+ frmAbout.Show vbModal, Me
+End Sub
+
+Private Sub mnuHelpSearchForHelpOn_Click()
+ Dim nRet As Integer
+
+
+ 'if there is no helpfile for this project display a message to the user
+ 'you can set the HelpFile for your application in the
+ 'Project Properties dialog
+ If Len(App.HelpFile) = 0 Then
+ MsgBox "Unable to display Help Contents. There is no Help associated with this project.", vbInformation, Me.Caption
+ Else
+ On Error Resume Next
+ nRet = OSWinHelp(Me.hwnd, App.HelpFile, 261, 0)
+ If Err Then
+ MsgBox Err.Description
+ End If
+ End If
+
+End Sub
+
+Private Sub mnuHelpContents_Click()
+ Dim nRet As Integer
+
+
+ 'if there is no helpfile for this project display a message to the user
+ 'you can set the HelpFile for your application in the
+ 'Project Properties dialog
+ If Len(App.HelpFile) = 0 Then
+ MsgBox "Unable to display Help Contents. There is no Help associated with this project.", vbInformation, Me.Caption
+ Else
+ On Error Resume Next
+ nRet = OSWinHelp(Me.hwnd, App.HelpFile, 3, 0)
+ If Err Then
+ MsgBox Err.Description
+ End If
+ End If
+
+End Sub
+
+
+Private Sub mnuViewWebBrowser_Click()
+ 'ToDo: Add 'mnuViewWebBrowser_Click' code.
+ MsgBox "Add 'mnuViewWebBrowser_Click' code."
+End Sub
+
+Private Sub mnuViewOptions_Click()
+ frmOptions.Show vbModal, Me
+End Sub
+
+Private Sub mnuViewRefresh_Click()
+ 'ToDo: Add 'mnuViewRefresh_Click' code.
+ MsgBox "Add 'mnuViewRefresh_Click' code."
+End Sub
+
+
+Private Sub mnuViewStatusBar_Click()
+ mnuViewStatusBar.Checked = Not mnuViewStatusBar.Checked
+ sbStatusBar.Visible = mnuViewStatusBar.Checked
+ SizeControls imgSplitter.Left
+End Sub
+
+Private Sub mnuViewToolbar_Click()
+ mnuViewToolbar.Checked = Not mnuViewToolbar.Checked
+ tbToolBar.Visible = mnuViewToolbar.Checked
+ SizeControls imgSplitter.Left
+End Sub
+
+Private Sub mnuEditInvertSelection_Click()
+ 'ToDo: Add 'mnuEditInvertSelection_Click' code.
+ MsgBox "Add 'mnuEditInvertSelection_Click' code."
+End Sub
+
+Private Sub mnuEditSelectAll_Click()
+ 'ToDo: Add 'mnuEditSelectAll_Click' code.
+ MsgBox "Add 'mnuEditSelectAll_Click' code."
+End Sub
+
+Private Sub mnuEditPasteSpecial_Click()
+ 'ToDo: Add 'mnuEditPasteSpecial_Click' code.
+ MsgBox "Add 'mnuEditPasteSpecial_Click' code."
+End Sub
+
+Private Sub mnuEditPaste_Click()
+ 'ToDo: Add 'mnuEditPaste_Click' code.
+ MsgBox "Add 'mnuEditPaste_Click' code."
+End Sub
+
+Private Sub mnuEditCopy_Click()
+ 'ToDo: Add 'mnuEditCopy_Click' code.
+ MsgBox "Add 'mnuEditCopy_Click' code."
+End Sub
+
+Private Sub mnuEditCut_Click()
+ 'ToDo: Add 'mnuEditCut_Click' code.
+ MsgBox "Add 'mnuEditCut_Click' code."
+End Sub
+
+Private Sub mnuEditUndo_Click()
+ 'ToDo: Add 'mnuEditUndo_Click' code.
+ MsgBox "Add 'mnuEditUndo_Click' code."
+End Sub
+
+Private Sub mnuFileClose_Click()
+ 'unload the form
+ Unload Me
+
+End Sub
+
+Private Sub mnuFileProperties_Click()
+ 'ToDo: Add 'mnuFileProperties_Click' code.
+ MsgBox "Add 'mnuFileProperties_Click' code."
+End Sub
+
+Private Sub mnuFileRename_Click()
+ 'ToDo: Add 'mnuFileRename_Click' code.
+ MsgBox "Add 'mnuFileRename_Click' code."
+End Sub
+
+Private Sub mnuFileDelete_Click()
+ 'ToDo: Add 'mnuFileDelete_Click' code.
+ MsgBox "Add 'mnuFileDelete_Click' code."
+End Sub
+
+Private Sub mnuFileNew_Click()
+ 'ToDo: Add 'mnuFileNew_Click' code.
+ MsgBox "Add 'mnuFileNew_Click' code."
+End Sub
+
+Private Sub mnuFileSendTo_Click()
+ 'ToDo: Add 'mnuFileSendTo_Click' code.
+ MsgBox "Add 'mnuFileSendTo_Click' code."
+End Sub
+
+Private Sub mnuFileFind_Click()
+ 'ToDo: Add 'mnuFileFind_Click' code.
+ MsgBox "Add 'mnuFileFind_Click' code."
+End Sub
+
+Private Sub mnuFileOpen_Click()
+ Dim sFile As String
+End Sub
+
+Private Sub mnuPopComputerProperties_Click()
+ mnuFileProperties_Click
+End Sub
+
+Private Sub mnuPopNewDatabase_Click()
+ frmNewDatabase1.Show vbModal, Me
+ frmNewDatabase2.Show vbModal, Me
+ frmNewDatabase3.Show vbModal, Me
+End Sub
+
+Private Sub mnuPopAddComputer_Click()
+ frmNewComputer.Show vbModal, Me
+ Dim c As Computer
+ For Each c In frmNewComputer.m_hosts
+ addComputer c
+ g_computers.Add c, "_" & c.m_name
+ Next
+End Sub
+
+Private Sub mnuPopSortComputers_Click()
+ If m_currentNode.Sorted = True Then
+ mnuPopSortComputers.Checked = False
+ m_currentNode.Sorted = False
+ Else
+ mnuPopSortComputers.Checked = True
+ m_currentNode.Sorted = True
+ End If
+End Sub
+
+Private Sub mnuPopRemoveComputer_Click()
+ Dim res As VbMsgBoxResult
+ Dim str As String
+ str = "Remove computer " & m_currentComputer.m_name
+ res = MsgBox(str, vbOKCancel, str)
+ If res = vbOK Then
+ removeComputer (m_currentComputer.m_name)
+ End If
+End Sub
+
+Private Sub mnuPopSortDatabases_Click()
+ If m_currentNode.Sorted = True Then
+ mnuPopSortDatabases.Checked = False
+ m_currentNode.Sorted = False
+ Else
+ mnuPopSortDatabases.Checked = True
+ m_currentNode.Sorted = True
+ End If
+End Sub
+
+Private Sub tvTreeView_BeforeLabelEdit(Cancel As Integer)
+ Cancel = True
+End Sub
+
+Private Sub tvTreeView_Collapse(ByVal Node As MSComctlLib.Node)
+ 'MsgBox "tvTreeView_Collapse"
+End Sub
+
+Private Sub tvTreeView_Expand(ByVal Node As MSComctlLib.Node)
+ 'MsgBox "tvTreeView_Expand"
+End Sub
+
+Private Sub tvTreeView_NodeClick(ByVal Node As MSComctlLib.Node)
+ selectNode Node
+End Sub
+
+Private Sub tvTreeView_MouseUp(Button As Integer, Shift As Integer, X As Single, Y As Single)
+ 'MsgBox "tvTreeView_MouseUp Button: " & Button & " Shift: " & Shift
+ Dim Node As MSComctlLib.Node
+ Dim place As Integer
+
+ Set Node = tvTreeView.HitTest(X, Y)
+ place = selectNode(Node)
+ If Button = vbRightButton Then
+ ShowPopup place
+ End If
+End Sub
+
+Private Function selectNode(ByRef n As MSComctlLib.Node) As Integer
+ Dim list As ListView
+ Dim place As Integer
+
+ If n Is Nothing Then
+ If Not m_currentNode Is Nothing Then
+ place = 1
+ m_currentNode.Selected = False
+ Else
+ place = 2
+ End If
+ Else
+ n.Selected = True
+ If n.Text = "Computers" Then
+ place = 3
+ Set list = lvComputers
+ lblTitle(1).Caption = "Computers"
+ ElseIf n.Text = "Databases" Then
+ place = 4
+ Set list = lvDatabases
+ lblTitle(1).Caption = "Databases"
+ ElseIf n.Parent.Text = "Computers" Then
+ place = 5
+ Set list = lvProcesses
+ setComputer (n.Key)
+ ElseIf n.Parent.Text = "Databases" Then
+ place = 6
+ Set list = lvProcesses
+ setDatabase (n.Key)
+ End If
+
+ If m_currentList.hwnd <> list.hwnd Then
+ m_currentList.Visible = False
+ list.Visible = True
+ Set m_currentList = list
+ End If
+ End If
+ Set m_currentNode = n
+ selectNode = place
+End Function
+
+Private Sub lvComputers_MouseUp(Button As Integer, Shift As Integer, X As Single, Y As Single)
+ Dim li As ListItem
+ Set li = lvComputers.HitTest(X, Y)
+ If Button = vbRightButton And Not li Is Nothing Then
+ Dim c As Computer
+ Set m_currentComputer = g_computers(li.Key)
+ ShowPopup 5
+ End If
+End Sub
+
+Private Sub ShowPopup(ByVal place As Integer)
+ Select Case place
+ Case 3
+ PopupMenu mnuPopComputers
+ Case 4
+ PopupMenu mnuPopDatabases
+ Case 5
+ mnuPopComputerName.Caption = m_currentComputer.m_name & ": " & m_currentComputer.m_status
+ Select Case m_currentComputer.m_status
+ Case "Connected"
+ mnuPopConnectComputer.Enabled = False
+ mnuPopDisconnectComputer.Enabled = True
+ Case "Connecting"
+ mnuPopConnectComputer.Enabled = False
+ mnuPopDisconnectComputer.Enabled = True
+ Case "Not connected"
+ mnuPopConnectComputer.Enabled = True
+ mnuPopDisconnectComputer.Enabled = False
+ Case "No contact"
+ mnuPopConnectComputer.Enabled = True
+ mnuPopDisconnectComputer.Enabled = False
+ Case Else
+ mnuPopConnectComputer.Enabled = False
+ mnuPopDisconnectComputer.Enabled = False
+ End Select
+
+ PopupMenu mnuPopComputer, , , , mnuPopComputerName
+ End Select
+End Sub
+
+Private Sub lvComputers_BeforeLabelEdit(Cancel As Integer)
+ Cancel = True
+End Sub
+
+Private Sub lvProcesses_BeforeLabelEdit(Cancel As Integer)
+ Cancel = True
+End Sub
+
+Private Sub lvDatabases_BeforeLabelEdit(Cancel As Integer)
+ Cancel = True
+End Sub
+
+Private Sub ColumnClick(ByRef list As ListView, i As Integer)
+ i = i - 1
+ If list.SortKey = i Then
+ list.SortOrder = 1 - list.SortOrder
+ Else
+ list.SortKey = i
+ End If
+End Sub
+
+Private Sub lvComputers_ColumnClick(ByVal ColumnHeader As MSComctlLib.ColumnHeader)
+ ColumnClick lvComputers, ColumnHeader.Index
+End Sub
+
+Private Sub lvProcesses_ColumnClick(ByVal ColumnHeader As MSComctlLib.ColumnHeader)
+ ColumnClick lvProcesses, ColumnHeader.Index
+End Sub
+
+Private Sub lvDatabases_ColumnClick(ByVal ColumnHeader As MSComctlLib.ColumnHeader)
+ ColumnClick lvDatabases, ColumnHeader.Index
+End Sub
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmNewComputer.frm b/ndb/src/cw/cpcc-win32/vb6/frmNewComputer.frm
new file mode 100644
index 00000000000..eae5802493c
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmNewComputer.frm
@@ -0,0 +1,124 @@
+VERSION 5.00
+Begin VB.Form frmNewComputer
+ Caption = "Add computer"
+ ClientHeight = 1545
+ ClientLeft = 60
+ ClientTop = 345
+ ClientWidth = 4605
+ LinkTopic = "Form1"
+ ScaleHeight = 1545
+ ScaleWidth = 4605
+ StartUpPosition = 3 'Windows Default
+ Begin VB.CommandButton Command3
+ Caption = "Apply"
+ Default = -1 'True
+ Height = 360
+ Left = 3240
+ TabIndex = 4
+ Tag = "OK"
+ Top = 840
+ Width = 1140
+ End
+ Begin VB.CommandButton Command2
+ Caption = "Cancel"
+ Height = 360
+ Left = 1920
+ TabIndex = 3
+ Tag = "OK"
+ Top = 840
+ Width = 1140
+ End
+ Begin VB.CommandButton Command1
+ Caption = "OK"
+ Height = 360
+ Left = 600
+ TabIndex = 2
+ Tag = "OK"
+ Top = 840
+ Width = 1140
+ End
+ Begin VB.TextBox Text1
+ Height = 285
+ Left = 1440
+ TabIndex = 1
+ Top = 240
+ Width = 2925
+ End
+ Begin VB.Label lblLabels
+ Caption = "Computer name:"
+ Height = 255
+ Index = 1
+ Left = 120
+ TabIndex = 0
+ Tag = "&User Name:"
+ Top = 240
+ Width = 1440
+ End
+End
+Attribute VB_Name = "frmNewComputer"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = False
+Attribute VB_PredeclaredId = True
+Attribute VB_Exposed = False
+Public m_hosts As New Collection
+
+Private Sub Form_Load()
+ If m_hosts.Count > 0 Then
+ For i = m_hosts.Count To 1 Step -1
+ m_hosts.Remove i
+ Next
+ End If
+End Sub
+
+Private Sub Command1_Click()
+ If Text1.Text = "" Then
+ MsgBox "Invalid hostname"
+ Exit Sub
+ End If
+
+ If Exists(g_computers, "_" & Text1.Text) Then
+ MsgBox Text1.Text & " already exists"
+ Exit Sub
+ End If
+
+ Dim c As New Computer
+ With c
+ .m_ip = ""
+ .m_name = Text1.Text
+ .m_status = "Not connected"
+ Set .m_processes = New Collection
+ End With
+
+ m_hosts.Add c
+
+ Unload Me
+End Sub
+
+Private Sub Command2_Click()
+ Unload Me
+End Sub
+
+Private Sub Command3_Click()
+ If Text1.Text = "" Then
+ MsgBox "Invalid hostname"
+ Exit Sub
+ End If
+
+ If Exists(g_computers, "_" & Text1.Text) Then
+ MsgBox Text1.Text & " already exists"
+ Exit Sub
+ End If
+
+ Dim c As New Computer
+ With c
+ .m_ip = ""
+ .m_name = Text1.Text
+ .m_status = "Not connected"
+ Set .m_processes = New Collection
+ End With
+
+ m_hosts.Add c
+
+ Text1.Text = ""
+End Sub
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmNewComputer.frx b/ndb/src/cw/cpcc-win32/vb6/frmNewComputer.frx
new file mode 100644
index 00000000000..593f4708db8
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmNewComputer.frx
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase.frx b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase.frx
new file mode 100644
index 00000000000..b20c2b651ae
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase.frx
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase1.frm b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase1.frm
new file mode 100644
index 00000000000..3fa1fd4c4e8
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase1.frm
@@ -0,0 +1,187 @@
+VERSION 5.00
+Begin VB.Form frmNewDatabase1
+ BorderStyle = 5 'Sizable ToolWindow
+ Caption = "Nodes"
+ ClientHeight = 3000
+ ClientLeft = 2850
+ ClientTop = 3450
+ ClientWidth = 6240
+ LinkTopic = "Form1"
+ MaxButton = 0 'False
+ MinButton = 0 'False
+ ScaleHeight = 3281.25
+ ScaleMode = 0 'User
+ ScaleWidth = 6359.712
+ ShowInTaskbar = 0 'False
+ StartUpPosition = 2 'CenterScreen
+ Begin VB.TextBox textApiNodes
+ Height = 285
+ Left = 2760
+ TabIndex = 12
+ Text = "4"
+ Top = 1665
+ Width = 375
+ End
+ Begin VB.VScrollBar VScroll1
+ Height = 255
+ Left = 3240
+ TabIndex = 11
+ Top = 1680
+ Width = 135
+ End
+ Begin VB.OptionButton Option4
+ Alignment = 1 'Right Justify
+ Caption = "1"
+ Height = 375
+ Left = 2760
+ TabIndex = 10
+ Top = 1020
+ Width = 375
+ End
+ Begin VB.OptionButton Option3
+ Alignment = 1 'Right Justify
+ Caption = "4"
+ Height = 375
+ Left = 3960
+ TabIndex = 9
+ Top = 360
+ Width = 375
+ End
+ Begin VB.OptionButton Option2
+ Alignment = 1 'Right Justify
+ Caption = "2"
+ Height = 375
+ Left = 3360
+ TabIndex = 8
+ Top = 360
+ Width = 375
+ End
+ Begin VB.OptionButton Option1
+ Alignment = 1 'Right Justify
+ Caption = "1"
+ Height = 375
+ Left = 2760
+ TabIndex = 7
+ Top = 360
+ Value = -1 'True
+ Width = 375
+ End
+ Begin VB.CommandButton cmdCancel
+ Cancel = -1 'True
+ Caption = "Cancel"
+ Height = 305
+ Left = 1320
+ TabIndex = 3
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdFinish
+ Caption = "Finish"
+ Enabled = 0 'False
+ Height = 305
+ Left = 5040
+ TabIndex = 2
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdBack
+ Caption = "Back"
+ Default = -1 'True
+ Enabled = 0 'False
+ Height = 305
+ Left = 2640
+ TabIndex = 0
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdNext
+ Caption = "Next"
+ Height = 305
+ Left = 3720
+ TabIndex = 1
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.Label Label3
+ Caption = "No of api nodes"
+ Height = 255
+ Left = 240
+ TabIndex = 6
+ Top = 1680
+ Width = 2415
+ End
+ Begin VB.Label Label2
+ Caption = "No of management nodes"
+ Height = 255
+ Left = 240
+ TabIndex = 5
+ Top = 1080
+ Width = 2415
+ End
+ Begin VB.Label Label1
+ Caption = "No of database nodes"
+ Height = 255
+ Left = 240
+ TabIndex = 4
+ Top = 420
+ Width = 2415
+ End
+ Begin VB.Line Line1
+ BorderColor = &H80000003&
+ X1 = 122.302
+ X2 = 6237.41
+ Y1 = 2493.75
+ Y2 = 2493.75
+ End
+End
+Attribute VB_Name = "frmNewDatabase1"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = False
+Attribute VB_PredeclaredId = True
+Attribute VB_Exposed = False
+Option Explicit
+
+Private Sub Form_Resize()
+ If Me.Width < 6375 Then Me.Width = 6375
+ cmdCancel.Left = Me.ScaleWidth - 5136 + 400
+ cmdBack.Left = Me.ScaleWidth - 3897 + 400
+ cmdNext.Left = Me.ScaleWidth - 2883 + 400
+ cmdFinish.Left = Me.ScaleWidth - 1643 + 400
+ Line1.X2 = Me.ScaleWidth - 480 + 400
+
+ cmdCancel.Top = Me.ScaleHeight - 375
+ cmdBack.Top = Me.ScaleHeight - 375
+ cmdNext.Top = Me.ScaleHeight - 375
+ cmdFinish.Top = Me.ScaleHeight - 375
+ Line1.Y1 = Me.ScaleHeight - 475
+ Line1.Y2 = Me.ScaleHeight - 475
+End Sub
+
+Private Sub cmdCancel_Click()
+ 'set the global var to false
+ 'to denote a failed login
+ Unload Me
+End Sub
+
+Private Sub Option1_Click()
+ Option2.Value = False
+ Option3.Value = False
+End Sub
+
+Private Sub Option2_Click()
+ Option1.Value = False
+ Option3.Value = False
+End Sub
+
+Private Sub Option3_Click()
+ Option1.Value = False
+ Option2.Value = False
+End Sub
+
+Private Sub Option4_Click()
+ Option4.Value = True
+End Sub
+
+Private Sub textApiNodes_Validate(Cancel As Boolean)
+ 'If Not isnumber(textApiNodes.Text) Then Cancel = False
+End Sub
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase2.frm b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase2.frm
new file mode 100644
index 00000000000..49806a695ea
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase2.frm
@@ -0,0 +1,136 @@
+VERSION 5.00
+Begin VB.Form frmNewDatabase2
+ BorderStyle = 5 'Sizable ToolWindow
+ Caption = "Computers"
+ ClientHeight = 2895
+ ClientLeft = 2850
+ ClientTop = 3450
+ ClientWidth = 6240
+ LinkTopic = "Form1"
+ MaxButton = 0 'False
+ MinButton = 0 'False
+ ScaleHeight = 3166.406
+ ScaleMode = 0 'User
+ ScaleWidth = 6359.712
+ ShowInTaskbar = 0 'False
+ StartUpPosition = 2 'CenterScreen
+ Begin VB.ComboBox Combo1
+ Height = 315
+ Left = 2400
+ TabIndex = 7
+ Text = "Combo1"
+ Top = 360
+ Width = 1455
+ End
+ Begin VB.VScrollBar VScroll2
+ Height = 255
+ Left = 4560
+ TabIndex = 6
+ Top = 375
+ Width = 135
+ End
+ Begin VB.TextBox Text1
+ Height = 285
+ Left = 4080
+ TabIndex = 5
+ Text = "4"
+ Top = 360
+ Width = 375
+ End
+ Begin VB.CommandButton cmdCancel
+ Cancel = -1 'True
+ Caption = "Cancel"
+ Height = 305
+ Left = 1320
+ TabIndex = 3
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdFinish
+ Caption = "Finish"
+ Enabled = 0 'False
+ Height = 305
+ Left = 5040
+ TabIndex = 2
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdBack
+ Caption = "Back"
+ Default = -1 'True
+ Enabled = 0 'False
+ Height = 305
+ Left = 2640
+ TabIndex = 0
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdNext
+ Caption = "Next"
+ Height = 305
+ Left = 3720
+ TabIndex = 1
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.Label Label5
+ Caption = "Computer"
+ Height = 255
+ Left = 2400
+ TabIndex = 9
+ Top = 0
+ Width = 975
+ End
+ Begin VB.Label Label4
+ Caption = "Node id"
+ Height = 255
+ Left = 4080
+ TabIndex = 8
+ Top = 120
+ Width = 615
+ End
+ Begin VB.Label Label1
+ Caption = "Database node 1"
+ Height = 255
+ Left = 240
+ TabIndex = 4
+ Top = 420
+ Width = 2415
+ End
+ Begin VB.Line Line1
+ BorderColor = &H80000003&
+ X1 = 122.302
+ X2 = 6237.41
+ Y1 = 2493.75
+ Y2 = 2493.75
+ End
+End
+Attribute VB_Name = "frmNewDatabase2"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = False
+Attribute VB_PredeclaredId = True
+Attribute VB_Exposed = False
+Option Explicit
+
+Private Sub Form_Resize()
+ If Me.Width < 6375 Then Me.Width = 6375
+ cmdCancel.Left = Me.ScaleWidth - 5136 + 400
+ cmdBack.Left = Me.ScaleWidth - 3897 + 400
+ cmdNext.Left = Me.ScaleWidth - 2883 + 400
+ cmdFinish.Left = Me.ScaleWidth - 1643 + 400
+ Line1.X2 = Me.ScaleWidth - 480 + 400
+
+ cmdCancel.Top = Me.ScaleHeight - 375
+ cmdBack.Top = Me.ScaleHeight - 375
+ cmdNext.Top = Me.ScaleHeight - 375
+ cmdFinish.Top = Me.ScaleHeight - 375
+ Line1.Y1 = Me.ScaleHeight - 475
+ Line1.Y2 = Me.ScaleHeight - 475
+End Sub
+
+Private Sub cmdCancel_Click()
+ 'set the global var to false
+ 'to denote a failed login
+ Unload Me
+End Sub
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase2.log b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase2.log
new file mode 100644
index 00000000000..808b21866e5
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase2.log
@@ -0,0 +1 @@
+Line 2: The Form or MDIForm name frmNewDatabase1 is already in use; cannot load this form.
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase3.frm b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase3.frm
new file mode 100644
index 00000000000..ba050a58a09
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmNewDatabase3.frm
@@ -0,0 +1,88 @@
+VERSION 5.00
+Begin VB.Form frmNewDatabase3
+ BorderStyle = 5 'Sizable ToolWindow
+ Caption = "Database configuration"
+ ClientHeight = 3000
+ ClientLeft = 2850
+ ClientTop = 3450
+ ClientWidth = 6240
+ LinkTopic = "Form1"
+ MaxButton = 0 'False
+ MinButton = 0 'False
+ ScaleHeight = 3281.25
+ ScaleMode = 0 'User
+ ScaleWidth = 6359.712
+ ShowInTaskbar = 0 'False
+ StartUpPosition = 2 'CenterScreen
+ Begin VB.CommandButton cmdCancel
+ Cancel = -1 'True
+ Caption = "Cancel"
+ Height = 305
+ Left = 1320
+ TabIndex = 3
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdFinish
+ Caption = "Finish"
+ Enabled = 0 'False
+ Height = 305
+ Left = 5040
+ TabIndex = 2
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdBack
+ Caption = "Back"
+ Default = -1 'True
+ Enabled = 0 'False
+ Height = 305
+ Left = 2640
+ TabIndex = 0
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.CommandButton cmdNext
+ Caption = "Next"
+ Height = 305
+ Left = 3720
+ TabIndex = 1
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.Line Line1
+ BorderColor = &H80000003&
+ X1 = 122.302
+ X2 = 6237.41
+ Y1 = 2493.75
+ Y2 = 2493.75
+ End
+End
+Attribute VB_Name = "frmNewDatabase3"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = False
+Attribute VB_PredeclaredId = True
+Attribute VB_Exposed = False
+Option Explicit
+
+Private Sub Form_Resize()
+ If Me.Width < 6375 Then Me.Width = 6375
+ cmdCancel.Left = Me.ScaleWidth - 5136 + 400
+ cmdBack.Left = Me.ScaleWidth - 3897 + 400
+ cmdNext.Left = Me.ScaleWidth - 2883 + 400
+ cmdFinish.Left = Me.ScaleWidth - 1643 + 400
+ Line1.X2 = Me.ScaleWidth - 480 + 400
+
+ cmdCancel.Top = Me.ScaleHeight - 375
+ cmdBack.Top = Me.ScaleHeight - 375
+ cmdNext.Top = Me.ScaleHeight - 375
+ cmdFinish.Top = Me.ScaleHeight - 375
+ Line1.Y1 = Me.ScaleHeight - 475
+ Line1.Y2 = Me.ScaleHeight - 475
+End Sub
+
+Private Sub cmdCancel_Click()
+ 'set the global var to false
+ 'to denote a failed login
+ Unload Me
+End Sub
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmOptions.frm b/ndb/src/cw/cpcc-win32/vb6/frmOptions.frm
new file mode 100644
index 00000000000..e526a35b3ec
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmOptions.frm
@@ -0,0 +1,231 @@
+VERSION 5.00
+Object = "{831FDD16-0C5C-11D2-A9FC-0000F8754DA1}#2.0#0"; "mscomctl.ocx"
+Begin VB.Form frmOptions
+ BorderStyle = 3 'Fixed Dialog
+ Caption = "Options"
+ ClientHeight = 5040
+ ClientLeft = 6600
+ ClientTop = 4575
+ ClientWidth = 6150
+ KeyPreview = -1 'True
+ LinkTopic = "Form1"
+ MaxButton = 0 'False
+ MinButton = 0 'False
+ ScaleHeight = 5040
+ ScaleWidth = 6150
+ ShowInTaskbar = 0 'False
+ Tag = "Options"
+ Begin VB.CommandButton cmdOK
+ Caption = "OK"
+ Height = 375
+ Left = 2490
+ TabIndex = 1
+ Tag = "OK"
+ Top = 4455
+ Width = 1095
+ End
+ Begin VB.CommandButton cmdCancel
+ Cancel = -1 'True
+ Caption = "Cancel"
+ Height = 375
+ Left = 3720
+ TabIndex = 3
+ Tag = "Cancel"
+ Top = 4455
+ Width = 1095
+ End
+ Begin VB.CommandButton cmdApply
+ Caption = "&Apply"
+ Height = 375
+ Left = 4920
+ TabIndex = 5
+ Tag = "&Apply"
+ Top = 4455
+ Width = 1095
+ End
+ Begin VB.PictureBox picOptions
+ BorderStyle = 0 'None
+ Height = 3780
+ Index = 3
+ Left = -20000
+ ScaleHeight = 3840.968
+ ScaleMode = 0 'User
+ ScaleWidth = 5745.64
+ TabIndex = 7
+ TabStop = 0 'False
+ Top = 480
+ Width = 5685
+ Begin VB.Frame fraSample4
+ Caption = "Sample 4"
+ Height = 2022
+ Left = 505
+ TabIndex = 11
+ Tag = "Sample 4"
+ Top = 502
+ Width = 2033
+ End
+ End
+ Begin VB.PictureBox picOptions
+ BorderStyle = 0 'None
+ Height = 3780
+ Index = 2
+ Left = -20000
+ ScaleHeight = 3840.968
+ ScaleMode = 0 'User
+ ScaleWidth = 5745.64
+ TabIndex = 9
+ TabStop = 0 'False
+ Top = 480
+ Width = 5685
+ Begin VB.Frame fraSample3
+ Caption = "Sample 3"
+ Height = 2022
+ Left = 406
+ TabIndex = 10
+ Tag = "Sample 3"
+ Top = 403
+ Width = 2033
+ End
+ End
+ Begin VB.PictureBox picOptions
+ BorderStyle = 0 'None
+ Height = 3780
+ Index = 1
+ Left = -20000
+ ScaleHeight = 3840.968
+ ScaleMode = 0 'User
+ ScaleWidth = 5745.64
+ TabIndex = 6
+ TabStop = 0 'False
+ Top = 480
+ Width = 5685
+ Begin VB.Frame fraSample2
+ Caption = "Sample 2"
+ Height = 2022
+ Left = 307
+ TabIndex = 8
+ Tag = "Sample 2"
+ Top = 305
+ Width = 2033
+ End
+ End
+ Begin VB.PictureBox picOptions
+ BorderStyle = 0 'None
+ Height = 3780
+ Index = 0
+ Left = 210
+ ScaleHeight = 3840.968
+ ScaleMode = 0 'User
+ ScaleWidth = 5745.64
+ TabIndex = 2
+ TabStop = 0 'False
+ Top = 480
+ Width = 5685
+ Begin VB.Frame fraSample1
+ Caption = "Sample 1"
+ Height = 2022
+ Left = 208
+ TabIndex = 4
+ Tag = "Sample 1"
+ Top = 207
+ Width = 2033
+ End
+ End
+ Begin MSComctlLib.TabStrip tbsOptions
+ Height = 4245
+ Left = 105
+ TabIndex = 0
+ Top = 120
+ Width = 5895
+ _ExtentX = 10398
+ _ExtentY = 7488
+ _Version = 393216
+ BeginProperty Tabs {1EFB6598-857C-11D1-B16A-00C0F0283628}
+ NumTabs = 4
+ BeginProperty Tab1 {1EFB659A-857C-11D1-B16A-00C0F0283628}
+ Caption = "Group 1"
+ ImageVarType = 2
+ EndProperty
+ BeginProperty Tab2 {1EFB659A-857C-11D1-B16A-00C0F0283628}
+ Caption = "Group 2"
+ ImageVarType = 2
+ EndProperty
+ BeginProperty Tab3 {1EFB659A-857C-11D1-B16A-00C0F0283628}
+ Caption = "Group 3"
+ ImageVarType = 2
+ EndProperty
+ BeginProperty Tab4 {1EFB659A-857C-11D1-B16A-00C0F0283628}
+ Caption = "Group 4"
+ ImageVarType = 2
+ EndProperty
+ EndProperty
+ End
+End
+Attribute VB_Name = "frmOptions"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = False
+Attribute VB_PredeclaredId = True
+Attribute VB_Exposed = False
+Private Sub cmdApply_Click()
+ 'ToDo: Add 'cmdApply_Click' code.
+ MsgBox "Apply Code goes here to set options w/o closing dialog!"
+End Sub
+
+
+Private Sub cmdCancel_Click()
+ Unload Me
+End Sub
+
+
+Private Sub cmdOK_Click()
+ 'ToDo: Add 'cmdOK_Click' code.
+ MsgBox "Code goes here to set options and close dialog!"
+ Unload Me
+End Sub
+
+
+Private Sub Form_KeyDown(KeyCode As Integer, Shift As Integer)
+ Dim i As Integer
+ i = tbsOptions.SelectedItem.Index
+ 'handle ctrl+tab to move to the next tab
+ If (Shift And 3) = 2 And KeyCode = vbKeyTab Then
+ If i = tbsOptions.Tabs.Count Then
+ 'last tab so we need to wrap to tab 1
+ Set tbsOptions.SelectedItem = tbsOptions.Tabs(1)
+ Else
+ 'increment the tab
+ Set tbsOptions.SelectedItem = tbsOptions.Tabs(i + 1)
+ End If
+ ElseIf (Shift And 3) = 3 And KeyCode = vbKeyTab Then
+ If i = 1 Then
+ 'last tab so we need to wrap to tab 1
+ Set tbsOptions.SelectedItem = tbsOptions.Tabs(tbsOptions.Tabs.Count)
+ Else
+ 'increment the tab
+ Set tbsOptions.SelectedItem = tbsOptions.Tabs(i - 1)
+ End If
+ End If
+End Sub
+
+
+
+
+Private Sub tbsOptions_Click()
+
+
+ Dim i As Integer
+ 'show and enable the selected tab's controls
+ 'and hide and disable all others
+ For i = 0 To tbsOptions.Tabs.Count - 1
+ If i = tbsOptions.SelectedItem.Index - 1 Then
+ picOptions(i).Left = 210
+ picOptions(i).Enabled = True
+ Else
+ picOptions(i).Left = -20000
+ picOptions(i).Enabled = False
+ End If
+ Next
+
+
+End Sub
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmSplash.frm b/ndb/src/cw/cpcc-win32/vb6/frmSplash.frm
new file mode 100644
index 00000000000..56ccbd79876
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmSplash.frm
@@ -0,0 +1,159 @@
+VERSION 5.00
+Begin VB.Form frmSplash
+ BorderStyle = 3 'Fixed Dialog
+ ClientHeight = 4710
+ ClientLeft = 45
+ ClientTop = 45
+ ClientWidth = 7455
+ ControlBox = 0 'False
+ LinkTopic = "Form1"
+ MaxButton = 0 'False
+ MinButton = 0 'False
+ ScaleHeight = 4710
+ ScaleWidth = 7455
+ ShowInTaskbar = 0 'False
+ StartUpPosition = 2 'CenterScreen
+ Visible = 0 'False
+ Begin VB.Frame fraMainFrame
+ Height = 4590
+ Left = 45
+ TabIndex = 0
+ Top = -15
+ Width = 7380
+ Begin VB.PictureBox picLogo
+ Height = 2385
+ Left = 510
+ Picture = "frmSplash.frx":0000
+ ScaleHeight = 2325
+ ScaleWidth = 1755
+ TabIndex = 2
+ Top = 855
+ Width = 1815
+ End
+ Begin VB.Label lblLicenseTo
+ Alignment = 1 'Right Justify
+ Caption = "LicenseTo"
+ Height = 255
+ Left = 270
+ TabIndex = 1
+ Tag = "LicenseTo"
+ Top = 300
+ Width = 6855
+ End
+ Begin VB.Label lblProductName
+ AutoSize = -1 'True
+ Caption = "Product"
+ BeginProperty Font
+ Name = "MS Sans Serif"
+ Size = 29.25
+ Charset = 0
+ Weight = 700
+ Underline = 0 'False
+ Italic = 0 'False
+ Strikethrough = 0 'False
+ EndProperty
+ Height = 720
+ Left = 2670
+ TabIndex = 9
+ Tag = "Product"
+ Top = 1200
+ Width = 2190
+ End
+ Begin VB.Label lblCompanyProduct
+ AutoSize = -1 'True
+ Caption = "CompanyProduct"
+ BeginProperty Font
+ Name = "MS Sans Serif"
+ Size = 18
+ Charset = 0
+ Weight = 700
+ Underline = 0 'False
+ Italic = 0 'False
+ Strikethrough = 0 'False
+ EndProperty
+ Height = 435
+ Left = 2505
+ TabIndex = 8
+ Tag = "CompanyProduct"
+ Top = 765
+ Width = 3000
+ End
+ Begin VB.Label lblPlatform
+ Alignment = 1 'Right Justify
+ AutoSize = -1 'True
+ Caption = "Platform"
+ BeginProperty Font
+ Name = "MS Sans Serif"
+ Size = 13.5
+ Charset = 0
+ Weight = 700
+ Underline = 0 'False
+ Italic = 0 'False
+ Strikethrough = 0 'False
+ EndProperty
+ Height = 360
+ Left = 5865
+ TabIndex = 7
+ Tag = "Platform"
+ Top = 2400
+ Width = 1140
+ End
+ Begin VB.Label lblVersion
+ Alignment = 1 'Right Justify
+ AutoSize = -1 'True
+ Caption = "Version"
+ BeginProperty Font
+ Name = "MS Sans Serif"
+ Size = 12
+ Charset = 0
+ Weight = 700
+ Underline = 0 'False
+ Italic = 0 'False
+ Strikethrough = 0 'False
+ EndProperty
+ Height = 300
+ Left = 6075
+ TabIndex = 6
+ Tag = "Version"
+ Top = 2760
+ Width = 930
+ End
+ Begin VB.Label lblWarning
+ Caption = "Warning"
+ Height = 195
+ Left = 300
+ TabIndex = 3
+ Tag = "Warning"
+ Top = 3720
+ Width = 6855
+ End
+ Begin VB.Label lblCompany
+ Caption = "Company"
+ Height = 255
+ Left = 4710
+ TabIndex = 5
+ Tag = "Company"
+ Top = 3330
+ Width = 2415
+ End
+ Begin VB.Label lblCopyright
+ Caption = "Copyright"
+ Height = 255
+ Left = 4710
+ TabIndex = 4
+ Tag = "Copyright"
+ Top = 3120
+ Width = 2415
+ End
+ End
+End
+Attribute VB_Name = "frmSplash"
+Attribute VB_GlobalNameSpace = False
+Attribute VB_Creatable = False
+Attribute VB_PredeclaredId = True
+Attribute VB_Exposed = False
+Private Sub Form_Load()
+ lblVersion.Caption = "Version " & App.Major & "." & App.Minor & "." & App.Revision
+ lblProductName.Caption = App.Title
+End Sub
+
diff --git a/ndb/src/cw/cpcc-win32/vb6/frmSplash.frx b/ndb/src/cw/cpcc-win32/vb6/frmSplash.frx
new file mode 100644
index 00000000000..fee0c5c59de
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/frmSplash.frx
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/networking.ico b/ndb/src/cw/cpcc-win32/vb6/networking.ico
new file mode 100644
index 00000000000..6bbf8022fc6
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/networking.ico
Binary files differ
diff --git a/ndb/src/cw/cpcc-win32/vb6/open folder.ico b/ndb/src/cw/cpcc-win32/vb6/open folder.ico
new file mode 100644
index 00000000000..7bb32cc83d3
--- /dev/null
+++ b/ndb/src/cw/cpcc-win32/vb6/open folder.ico
Binary files differ
diff --git a/ndb/src/cw/cpcd/APIService.cpp b/ndb/src/cw/cpcd/APIService.cpp
new file mode 100644
index 00000000000..9cf17addcc2
--- /dev/null
+++ b/ndb/src/cw/cpcd/APIService.cpp
@@ -0,0 +1,385 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <Parser.hpp>
+#include <NdbOut.hpp>
+#include <Properties.hpp>
+#include <socket_io.h>
+
+#include "APIService.hpp"
+#include "CPCD.hpp"
+#include <NdbMutex.h>
+#include <NdbStdio.h>
+#include <OutputStream.hpp>
+
+/**
+ const char * name;
+ const char * realName;
+ const Type type;
+ const ArgType argType;
+ const ArgRequired argRequired;
+ const ArgMinMax argMinMax;
+ const int minVal;
+ const int maxVal;
+ void (T::* function)(const class Properties & args);
+ const char * description;
+*/
+
+#define CPCD_CMD(name, fun, desc) \
+ { name, \
+ 0, \
+ ParserRow<CPCDAPISession>::Cmd, \
+ ParserRow<CPCDAPISession>::String, \
+ ParserRow<CPCDAPISession>::Optional, \
+ ParserRow<CPCDAPISession>::IgnoreMinMax, \
+ 0, 0, \
+ fun, \
+ desc }
+
+#define CPCD_ARG(name, type, opt, desc) \
+ { name, \
+ 0, \
+ ParserRow<CPCDAPISession>::Arg, \
+ ParserRow<CPCDAPISession>::type, \
+ ParserRow<CPCDAPISession>::opt, \
+ ParserRow<CPCDAPISession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ desc }
+
+#define CPCD_ARG2(name, type, opt, min, max, desc) \
+ { name, \
+ 0, \
+ ParserRow<CPCDAPISession>::Arg, \
+ ParserRow<CPCDAPISession>::type, \
+ ParserRow<CPCDAPISession>::opt, \
+ ParserRow<CPCDAPISession>::IgnoreMinMax, \
+ min, max, \
+ 0, \
+ desc }
+
+#define CPCD_END() \
+ { 0, \
+ 0, \
+ ParserRow<CPCDAPISession>::Arg, \
+ ParserRow<CPCDAPISession>::Int, \
+ ParserRow<CPCDAPISession>::Optional, \
+ ParserRow<CPCDAPISession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0 }
+
+#define CPCD_CMD_ALIAS(name, realName, fun) \
+ { name, \
+ realName, \
+ ParserRow<CPCDAPISession>::CmdAlias, \
+ ParserRow<CPCDAPISession>::Int, \
+ ParserRow<CPCDAPISession>::Optional, \
+ ParserRow<CPCDAPISession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0 }
+
+#define CPCD_ARG_ALIAS(name, realName, fun) \
+ { name, \
+ realName, \
+ ParserRow<CPCDAPISession>::ArgAlias, \
+ ParserRow<CPCDAPISession>::Int, \
+ ParserRow<CPCDAPISession>::Optional, \
+ ParserRow<CPCDAPISession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0 }
+
+const
+ParserRow<CPCDAPISession> commands[] =
+{
+ CPCD_CMD("define process" , &CPCDAPISession::defineProcess, ""),
+ CPCD_ARG("id", Int, Optional, "Id of process."),
+ CPCD_ARG("name", String, Mandatory, "Name of process"),
+ CPCD_ARG("group", String, Mandatory, "Group of process"),
+ CPCD_ARG("env", String, Optional, "Environment variables for process"),
+ CPCD_ARG("path", String, Mandatory, "Path to binary"),
+ CPCD_ARG("args", String, Optional, "Arguments to process"),
+ CPCD_ARG("type", String, Mandatory, "Type of process"),
+ CPCD_ARG("cwd", String, Mandatory, "Working directory of process"),
+ CPCD_ARG("owner", String, Mandatory, "Owner of process"),
+ CPCD_ARG("runas", String, Optional, "Run as user"),
+ CPCD_ARG("stdout", String, Optional, "Redirection of stdout"),
+ CPCD_ARG("stderr", String, Optional, "Redirection of stderr"),
+ CPCD_ARG("stdin", String, Optional, "Redirection of stderr"),
+ CPCD_ARG("ulimit", String, Optional, "ulimit"),
+
+ CPCD_CMD("undefine process", &CPCDAPISession::undefineProcess, ""),
+ CPCD_CMD_ALIAS("undef", "undefine process", 0),
+ CPCD_ARG("id", Int, Mandatory, "Id of process"),
+ CPCD_ARG_ALIAS("i", "id", 0),
+
+ CPCD_CMD("start process", &CPCDAPISession::startProcess, ""),
+ CPCD_ARG("id", Int, Mandatory, "Id of process"),
+
+ CPCD_CMD("stop process", &CPCDAPISession::stopProcess, ""),
+ CPCD_ARG("id", Int, Mandatory, "Id of process"),
+
+ CPCD_CMD("list processes", &CPCDAPISession::listProcesses, ""),
+
+ CPCD_END()
+};
+CPCDAPISession::CPCDAPISession(NDB_SOCKET_TYPE sock,
+ CPCD & cpcd)
+ : SocketServer::Session(sock)
+ , m_cpcd(cpcd)
+{
+ m_input = new SocketInputStream(sock);
+ m_output = new SocketOutputStream(sock);
+ m_parser = new Parser<CPCDAPISession>(commands, *m_input, true, true, true);
+}
+
+CPCDAPISession::CPCDAPISession(FILE * f, CPCD & cpcd)
+ : SocketServer::Session(1)
+ , m_cpcd(cpcd)
+{
+ m_input = new FileInputStream(f);
+ m_parser = new Parser<CPCDAPISession>(commands, *m_input, true, true, true);
+}
+
+CPCDAPISession::~CPCDAPISession() {
+ delete m_input;
+ delete m_parser;
+}
+
+void
+CPCDAPISession::runSession(){
+ Parser_t::Context ctx;
+ while(!m_stop){
+ m_parser->run(ctx, * this);
+ if(ctx.m_currentToken == 0)
+ break;
+
+ switch(ctx.m_status){
+ case Parser_t::Ok:
+ for(size_t i = 0; i<ctx.m_aliasUsed.size(); i++)
+ ndbout_c("Used alias: %s -> %s",
+ ctx.m_aliasUsed[i]->name, ctx.m_aliasUsed[i]->realName);
+ break;
+ case Parser_t::NoLine:
+ case Parser_t::EmptyLine:
+ break;
+ default:
+ break;
+ }
+ }
+ NDB_CLOSE_SOCKET(m_socket);
+}
+
+void
+CPCDAPISession::stopSession(){
+ CPCD::RequestStatus rs;
+ for(size_t i = 0; i<m_temporaryProcesses.size(); i++){
+ Uint32 id = m_temporaryProcesses[i];
+ m_cpcd.undefineProcess(&rs, id);
+ }
+}
+
+void
+CPCDAPISession::loadFile(){
+ Parser_t::Context ctx;
+ while(!m_stop){
+ m_parser->run(ctx, * this);
+ if(ctx.m_currentToken == 0)
+ break;
+
+ switch(ctx.m_status){
+ case Parser_t::Ok:
+ for(size_t i = 0; i<ctx.m_aliasUsed.size(); i++)
+ ndbout_c("Used alias: %s -> %s",
+ ctx.m_aliasUsed[i]->name, ctx.m_aliasUsed[i]->realName);
+ break;
+ case Parser_t::NoLine:
+ case Parser_t::EmptyLine:
+ break;
+ default:
+ break;
+ }
+ }
+}
+
+static const int g_TimeOut = 1000;
+
+void
+CPCDAPISession::defineProcess(Parser_t::Context & /* unused */,
+ const class Properties & args){
+
+ CPCD::Process * p = new CPCD::Process(args, &m_cpcd);
+
+ CPCD::RequestStatus rs;
+
+ bool ret = m_cpcd.defineProcess(&rs, p);
+ if(!m_cpcd.loadingProcessList) {
+ m_output->println("define process");
+ m_output->println("status: %d", rs.getStatus());
+ if(ret == true){
+ m_output->println("id: %d", p->m_id);
+ if(p->m_processType == TEMPORARY){
+ m_temporaryProcesses.push_back(p->m_id);
+ }
+ } else {
+ m_output->println("errormessage: %s", rs.getErrMsg());
+ }
+ m_output->println("");
+ }
+}
+
+void
+CPCDAPISession::undefineProcess(Parser_t::Context & /* unused */,
+ const class Properties & args){
+ Uint32 id;
+ CPCD::RequestStatus rs;
+
+ args.get("id", &id);
+ bool ret = m_cpcd.undefineProcess(&rs, id);
+
+ m_output->println("undefine process");
+ m_output->println("id: %d", id);
+ m_output->println("status: %d", rs.getStatus());
+ if(!ret)
+ m_output->println("errormessage: %s", rs.getErrMsg());
+
+ m_output->println("");
+}
+
+void
+CPCDAPISession::startProcess(Parser_t::Context & /* unused */,
+ const class Properties & args){
+ Uint32 id;
+ CPCD::RequestStatus rs;
+
+ args.get("id", &id);
+ const int ret = m_cpcd.startProcess(&rs, id);
+
+ if(!m_cpcd.loadingProcessList) {
+ m_output->println("start process");
+ m_output->println("id: %d", id);
+ m_output->println("status: %d", rs.getStatus());
+ if(!ret)
+ m_output->println("errormessage: %s", rs.getErrMsg());
+ m_output->println("");
+ }
+}
+
+void
+CPCDAPISession::stopProcess(Parser_t::Context & /* unused */,
+ const class Properties & args){
+ Uint32 id;
+ CPCD::RequestStatus rs;
+
+ args.get("id", &id);
+ int ret = m_cpcd.stopProcess(&rs, id);
+
+ m_output->println("stop process");
+ m_output->println("id: %d", id);
+ m_output->println("status: %d", rs.getStatus());
+ if(!ret)
+ m_output->println("errormessage: %s", rs.getErrMsg());
+
+ m_output->println("");
+}
+
+static const char *
+propToString(Properties *prop, const char *key) {
+ static char buf[32];
+ const char *retval = NULL;
+ PropertiesType pt;
+
+ prop->getTypeOf(key, &pt);
+ switch(pt) {
+ case PropertiesType_Uint32:
+ Uint32 val;
+ prop->get(key, &val);
+ snprintf(buf, sizeof buf, "%d", val);
+ retval = buf;
+ break;
+ case PropertiesType_char:
+ const char *str;
+ prop->get(key, &str);
+ retval = str;
+ break;
+ default:
+ snprintf(buf, sizeof buf, "(unknown)");
+ retval = buf;
+ }
+ return retval;
+}
+
+void
+CPCDAPISession::printProperty(Properties *prop, const char *key) {
+ m_output->println("%s: %s", key, propToString(prop, key));
+}
+
+void
+CPCDAPISession::listProcesses(Parser_t::Context & /* unused */,
+ const class Properties & /* unused */){
+ m_cpcd.m_processes.lock();
+ MutexVector<CPCD::Process *> *proclist = m_cpcd.getProcessList();
+
+ m_output->println("start processes");
+ m_output->println("");
+
+
+ for(size_t i = 0; i < proclist->size(); i++) {
+ CPCD::Process *p = (*proclist)[i];
+
+ m_output->println("process");
+
+ m_output->println("id: %d", p->m_id);
+ m_output->println("name: %s", p->m_name.c_str());
+ m_output->println("path: %s", p->m_path.c_str());
+ m_output->println("args: %s", p->m_args.c_str());
+ m_output->println("type: %s", p->m_type.c_str());
+ m_output->println("cwd: %s", p->m_cwd.c_str());
+ m_output->println("env: %s", p->m_env.c_str());
+ m_output->println("owner: %s", p->m_owner.c_str());
+ m_output->println("group: %s", p->m_group.c_str());
+ m_output->println("runas: %s", p->m_runas.c_str());
+ m_output->println("stdin: %s", p->m_stdin.c_str());
+ m_output->println("stdout: %s", p->m_stdout.c_str());
+ m_output->println("stderr: %s", p->m_stderr.c_str());
+ m_output->println("ulimit: %s", p->m_ulimit.c_str());
+ switch(p->m_status){
+ case STOPPED:
+ m_output->println("status: stopped");
+ break;
+ case STARTING:
+ m_output->println("status: starting");
+ break;
+ case RUNNING:
+ m_output->println("status: running");
+ break;
+ case STOPPING:
+ m_output->println("status: stopping");
+ break;
+ }
+
+ m_output->println("");
+
+ }
+
+ m_output->println("end processes");
+ m_output->println("");
+
+ m_cpcd.m_processes.unlock();
+}
diff --git a/ndb/src/cw/cpcd/APIService.hpp b/ndb/src/cw/cpcd/APIService.hpp
new file mode 100644
index 00000000000..ef988785f89
--- /dev/null
+++ b/ndb/src/cw/cpcd/APIService.hpp
@@ -0,0 +1,64 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CPCD_API_HPP
+#define CPCD_API_HPP
+
+#include <Parser.hpp>
+#include <InputStream.hpp>
+#include <SocketServer.hpp>
+
+class CPCD;
+
+class CPCDAPISession : public SocketServer::Session {
+ typedef Parser<CPCDAPISession> Parser_t;
+
+ class CPCD & m_cpcd;
+ InputStream *m_input;
+ OutputStream *m_output;
+ Parser_t *m_parser;
+
+ Vector<int> m_temporaryProcesses;
+
+ void printProperty(Properties *prop, const char *key);
+public:
+ CPCDAPISession(NDB_SOCKET_TYPE, class CPCD &);
+ CPCDAPISession(FILE * f, CPCD & cpcd);
+ ~CPCDAPISession();
+
+ virtual void runSession();
+ virtual void stopSession();
+ void loadFile();
+
+ void defineProcess(Parser_t::Context & ctx, const class Properties & args);
+ void undefineProcess(Parser_t::Context & ctx, const class Properties & args);
+ void startProcess(Parser_t::Context & ctx, const class Properties & args);
+ void stopProcess(Parser_t::Context & ctx, const class Properties & args);
+ void showProcess(Parser_t::Context & ctx, const class Properties & args);
+ void listProcesses(Parser_t::Context & ctx, const class Properties & args);
+};
+
+class CPCDAPIService : public SocketServer::Service {
+ class CPCD & m_cpcd;
+public:
+ CPCDAPIService(class CPCD & cpcd) : m_cpcd(cpcd) {}
+
+ CPCDAPISession * newSession(NDB_SOCKET_TYPE theSock){
+ return new CPCDAPISession(theSock, m_cpcd);
+ }
+};
+
+#endif
diff --git a/ndb/src/cw/cpcd/CPCD.cpp b/ndb/src/cw/cpcd/CPCD.cpp
new file mode 100644
index 00000000000..8864ccf6e4e
--- /dev/null
+++ b/ndb/src/cw/cpcd/CPCD.cpp
@@ -0,0 +1,435 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <string.h>
+#include <NdbOut.hpp>
+#include <NdbUnistd.h>
+#include <NdbStdio.h>
+#include <errno.h>
+
+#include "APIService.hpp"
+#include "CPCD.hpp"
+#include <NdbMutex.h>
+
+#include "common.hpp"
+
+extern const ParserRow<CPCDAPISession> commands[];
+
+
+CPCD::CPCD() {
+ loadingProcessList = false;
+ m_processes.clear();
+ m_monitor = NULL;
+ m_monitor = new Monitor(this);
+ m_procfile = "ndb_cpcd.db";
+}
+
+CPCD::~CPCD() {
+ if(m_monitor != NULL) {
+ delete m_monitor;
+ m_monitor = NULL;
+ }
+}
+
+int
+CPCD::findUniqueId() {
+ int id;
+ bool ok = false;
+ m_processes.lock();
+
+ while(!ok) {
+ ok = true;
+ id = random() % 8192; /* Don't want so big numbers */
+
+ if(id == 0)
+ ok = false;
+
+ for(size_t i = 0; i<m_processes.size(); i++) {
+ if(m_processes[i]->m_id == id)
+ ok = false;
+ }
+ }
+ m_processes.unlock();
+ return id;
+}
+
+bool
+CPCD::defineProcess(RequestStatus * rs, Process * arg){
+ if(arg->m_id == -1)
+ arg->m_id = findUniqueId();
+
+ Guard tmp(m_processes);
+
+ for(size_t i = 0; i<m_processes.size(); i++) {
+ Process * proc = m_processes[i];
+
+ if((strcmp(arg->m_name.c_str(), proc->m_name.c_str()) == 0) &&
+ (strcmp(arg->m_group.c_str(), proc->m_group.c_str()) == 0)) {
+ /* Identical names in the same group */
+ rs->err(AlreadyExists, "Name already exists");
+ return false;
+ }
+
+ if(arg->m_id == proc->m_id) {
+ /* Identical ID numbers */
+ rs->err(AlreadyExists, "Id already exists");
+ return false;
+ }
+ }
+
+ m_processes.push_back(arg, false);
+
+ notifyChanges();
+ report(arg->m_id, CPCEvent::ET_PROC_USER_DEFINE);
+
+ return true;
+}
+
+bool
+CPCD::undefineProcess(CPCD::RequestStatus *rs, int id) {
+
+ Guard tmp(m_processes);
+
+ Process * proc = 0;
+ size_t i;
+ for(i = 0; i < m_processes.size(); i++) {
+ if(m_processes[i]->m_id == id) {
+ proc = m_processes[i];
+ break;
+ }
+ }
+
+ if(proc == 0){
+ rs->err(NotExists, "No such process");
+ return false;
+ }
+
+ switch(proc->m_status){
+ case RUNNING:
+ case STOPPED:
+ case STOPPING:
+ case STARTING:
+ proc->stop();
+ m_processes.erase(i, false /* Already locked */);
+ }
+
+
+ notifyChanges();
+
+ report(id, CPCEvent::ET_PROC_USER_UNDEFINE);
+
+ return true;
+}
+
+bool
+CPCD::startProcess(CPCD::RequestStatus *rs, int id) {
+
+ Process * proc = 0;
+ {
+
+ Guard tmp(m_processes);
+
+ for(size_t i = 0; i < m_processes.size(); i++) {
+ if(m_processes[i]->m_id == id) {
+ proc = m_processes[i];
+ break;
+ }
+ }
+
+ if(proc == 0){
+ rs->err(NotExists, "No such process");
+ return false;
+ }
+
+ switch(proc->m_status){
+ case STOPPED:
+ proc->m_status = STARTING;
+ if(proc->start() != 0){
+ rs->err(Error, "Failed to start");
+ return false;
+ }
+ break;
+ case STARTING:
+ rs->err(Error, "Already starting");
+ return false;
+ case RUNNING:
+ rs->err(Error, "Already started");
+ return false;
+ case STOPPING:
+ rs->err(Error, "Currently stopping");
+ return false;
+ }
+
+ notifyChanges();
+ }
+ report(id, CPCEvent::ET_PROC_USER_START);
+
+ return true;
+}
+
+bool
+CPCD::stopProcess(CPCD::RequestStatus *rs, int id) {
+
+ Guard tmp(m_processes);
+
+ Process * proc = 0;
+ for(size_t i = 0; i < m_processes.size(); i++) {
+ if(m_processes[i]->m_id == id) {
+ proc = m_processes[i];
+ break;
+ }
+ }
+
+ if(proc == 0){
+ rs->err(NotExists, "No such process");
+ return false;
+ }
+
+ switch(proc->m_status){
+ case STARTING:
+ case RUNNING:
+ proc->stop();
+ break;
+ case STOPPED:
+ rs->err(AlreadyStopped, "Already stopped");
+ return false;
+ break;
+ case STOPPING:
+ rs->err(Error, "Already stopping");
+ return false;
+ }
+
+ notifyChanges();
+
+ report(id, CPCEvent::ET_PROC_USER_START);
+
+ return true;
+}
+
+bool
+CPCD::notifyChanges() {
+ bool ret = true;
+ if(!loadingProcessList)
+ ret = saveProcessList();
+
+ m_monitor->signal();
+
+ return ret;
+}
+
+/* Must be called with m_processlist locked */
+bool
+CPCD::saveProcessList(){
+ char newfile[PATH_MAX+4];
+ char oldfile[PATH_MAX+4];
+ char curfile[PATH_MAX];
+ FILE *f;
+
+ /* Create the filenames that we will use later */
+ snprintf(newfile, sizeof(newfile), "%s.new", m_procfile.c_str());
+ snprintf(oldfile, sizeof(oldfile), "%s.old", m_procfile.c_str());
+ snprintf(curfile, sizeof(curfile), "%s", m_procfile.c_str());
+
+ f = fopen(newfile, "w");
+
+ if(f == NULL) {
+ /* XXX What should be done here? */
+ logger.critical("Cannot open `%s': %s\n", newfile, strerror(errno));
+ return false;
+ }
+
+ for(size_t i = 0; i<m_processes.size(); i++){
+ m_processes[i]->print(f);
+ fprintf(f, "\n");
+
+ if(m_processes[i]->m_processType == TEMPORARY){
+ /**
+ * Interactive process should never be "restarted" on cpcd restart
+ */
+ continue;
+ }
+
+ if(m_processes[i]->m_status == RUNNING ||
+ m_processes[i]->m_status == STARTING){
+ fprintf(f, "start process\nid: %d\n\n", m_processes[i]->m_id);
+ }
+ }
+
+ fclose(f);
+ f = NULL;
+
+ /* This will probably only work on reasonably Unix-like systems. You have
+ * been warned...
+ *
+ * The motivation behind all this link()ing is that the daemon might
+ * crash right in the middle of updating the configuration file, and in
+ * that case we want to be sure that the old file is around until we are
+ * guaranteed that there is always at least one copy of either the old or
+ * the new configuration file left.
+ */
+
+ /* Remove an old config file if it exists */
+ unlink(oldfile);
+
+ if(link(curfile, oldfile) != 0) /* make a backup of the running config */
+ logger.error("Cannot rename '%s' -> '%s'", curfile, oldfile);
+ else {
+ if(unlink(curfile) != 0) { /* remove the running config file */
+ logger.critical("Cannot remove file '%s'", curfile);
+ return false;
+ }
+ }
+
+ if(link(newfile, curfile) != 0) { /* put the new config file in place */
+ printf("-->%d\n", __LINE__);
+
+ logger.critical("Cannot rename '%s' -> '%s': %s",
+ curfile, newfile, strerror(errno));
+ return false;
+ }
+
+ /* XXX Ideally we would fsync() the directory here, but I'm not sure if
+ * that actually works.
+ */
+
+ unlink(newfile); /* remove the temporary file */
+ unlink(oldfile); /* remove the old file */
+
+ logger.info("Process list saved as '%s'", curfile);
+
+ return true;
+}
+
+bool
+CPCD::loadProcessList(){
+ BaseString secondfile;
+ FILE *f;
+
+ loadingProcessList = true;
+
+ secondfile.assfmt("%s.new", m_procfile.c_str());
+
+ /* Try to open the config file */
+ f = fopen(m_procfile.c_str(), "r");
+
+ /* If it did not exist, try to open the backup. See the saveProcessList()
+ * method for an explanation why it is done this way.
+ */
+ if(f == NULL) {
+ f = fopen(secondfile.c_str(), "r");
+
+ if(f == NULL) {
+ /* XXX What to do here? */
+ logger.info("Configuration file `%s' not found",
+ m_procfile.c_str());
+ logger.info("Starting with empty configuration");
+ loadingProcessList = false;
+ return false;
+ } else {
+ logger.info("Configuration file `%s' missing",
+ m_procfile.c_str());
+ logger.info("Backup configuration file `%s' is used",
+ secondfile.c_str());
+ /* XXX Maybe we should just rename the backup file to the official
+ * name, and be done with it?
+ */
+ }
+ }
+
+ CPCDAPISession sess(f, *this);
+ sess.loadFile();
+ loadingProcessList = false;
+
+ Vector<int> temporary;
+ for(size_t i = 0; i<m_processes.size(); i++){
+ Process * proc = m_processes[i];
+ proc->readPid();
+ if(proc->m_processType == TEMPORARY){
+ temporary.push_back(proc->m_id);
+ }
+ }
+
+ for(size_t i = 0; i<temporary.size(); i++){
+ RequestStatus rs;
+ undefineProcess(&rs, temporary[i]);
+ }
+
+ /* Don't call notifyChanges here, as that would save the file we just
+ loaded */
+ m_monitor->signal();
+ return true;
+}
+
+MutexVector<CPCD::Process *> *
+CPCD::getProcessList() {
+ return &m_processes;
+}
+
+void
+CPCD::RequestStatus::err(enum RequestStatusCode status, char *msg) {
+ m_status = status;
+ snprintf(m_errorstring, sizeof(m_errorstring), "%s", msg);
+}
+
+#if 0
+void
+CPCD::sigchild(int pid){
+ m_processes.lock();
+ for(size_t i = 0; i<m_processes.size(); i++){
+ if(m_processes[i].m_pid == pid){
+ }
+ }
+ wait(pid, 0, 0);
+}
+#endif
+
+ /** Register event subscriber */
+void
+CPCD::do_register(EventSubscriber * sub){
+ m_subscribers.lock();
+ m_subscribers.push_back(sub, false);
+ m_subscribers.unlock();
+}
+
+EventSubscriber*
+CPCD::do_unregister(EventSubscriber * sub){
+ m_subscribers.lock();
+
+ for(size_t i = 0; i<m_subscribers.size(); i++){
+ if(m_subscribers[i] == sub){
+ m_subscribers.erase(i);
+ m_subscribers.unlock();
+ return sub;
+ }
+ }
+
+ m_subscribers.unlock();
+ return 0;
+}
+
+void
+CPCD::report(int id, CPCEvent::EventType t){
+ CPCEvent e;
+ e.m_time = time(0);
+ e.m_proc = id;
+ e.m_type = t;
+ m_subscribers.lock();
+ for(size_t i = 0; i<m_subscribers.size(); i++){
+ (* m_subscribers[i]).report(e);
+ }
+ m_subscribers.unlock();
+}
diff --git a/ndb/src/cw/cpcd/CPCD.hpp b/ndb/src/cw/cpcd/CPCD.hpp
new file mode 100644
index 00000000000..4a7cab23bab
--- /dev/null
+++ b/ndb/src/cw/cpcd/CPCD.hpp
@@ -0,0 +1,382 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CPCD_HPP
+#define CPCD_HPP
+
+#include <Vector.hpp>
+#include <Properties.hpp>
+#include <NdbOut.hpp>
+#include <NdbThread.h>
+#include <NdbCondition.h>
+#include <BaseString.hpp>
+
+/* XXX Need to figure out how to do this for non-Unix systems */
+#define CPCD_DEFAULT_WORK_DIR "/var/run/ndb_cpcd"
+#define CPCD_DEFAULT_PROC_FILE "ndb_cpcd.conf"
+#define CPCD_DEFAULT_TCP_PORT 1234
+#define CPCD_DEFAULT_POLLING_INTERVAL 5 /* seconds */
+#define CPCD_DEFAULT_CONFIG_FILE "/etc/ndb_cpcd.conf"
+
+enum ProcessStatus {
+ STOPPED = 0,
+ STARTING = 1,
+ RUNNING = 2,
+ STOPPING = 3
+};
+
+enum ProcessType {
+ PERMANENT = 0,
+ TEMPORARY = 1
+};
+
+struct CPCEvent {
+ enum EventType {
+ ET_USER_CONNECT,
+ ET_USER_DISCONNECT,
+
+ ET_PROC_USER_DEFINE, // Defined proc
+ ET_PROC_USER_UNDEFINE, // Undefined proc
+ ET_PROC_USER_START, // Proc ordered to start
+ ET_PROC_USER_STOP, // Proc ordered to stop
+ ET_PROC_STATE_RUNNING, // exec returned(?) ok
+ ET_PROC_STATE_STOPPED // detected that proc is ! running
+ };
+
+ int m_proc;
+ time_t m_time;
+ EventType m_type;
+};
+
+struct EventSubscriber {
+ virtual void report(const CPCEvent &) = 0;
+};
+
+/**
+ * @brief Error codes for CPCD requests
+ */
+enum RequestStatusCode {
+ OK = 0, ///< Everything OK
+ Error = 1, ///< Generic error
+ AlreadyExists = 2, ///< Entry already exists in list
+ NotExists = 3, ///< Entry does not exist in list
+ AlreadyStopped = 4
+};
+
+/**
+ * @class CPCD
+ * @brief Manages processes, letting them be controlled with a TCP connection.
+ *
+ * The class implementing the Cluster Process Control Daemon
+ */
+class CPCD {
+public:
+ /** @brief Describes the status of a client request */
+ class RequestStatus {
+ public:
+ /** @brief Constructs an empty RequestStatus */
+ RequestStatus() { m_status = OK; m_errorstring[0] = '\0'; };
+
+ /** @brief Sets an errorcode and a printable message */
+ void err(enum RequestStatusCode, char *);
+
+ /** @brief Returns the error message */
+ char *getErrMsg() { return m_errorstring; };
+
+ /** @brief Returns the error code */
+ enum RequestStatusCode getStatus() { return m_status; };
+ private:
+ enum RequestStatusCode m_status;
+ char m_errorstring[256];
+ };
+ /**
+ * @brief Manages a process
+ */
+ class Process {
+ int m_pid;
+ public:
+ /**
+ * @brief Constructs and empty Process
+ */
+ Process(const Properties & props, class CPCD *cpcd);
+ /**
+ * @brief Monitors the process
+ *
+ * The process is started or stopped as needed.
+ */
+ void monitor();
+
+ /**
+ * @brief Checks if the process is running or not
+ *
+ * @return
+ * - true if the process is running,
+ * - false if the process is not running
+ */
+ bool isRunning();
+
+ /** @brief Starts the process */
+ int start();
+
+ /** @brief Stops the process */
+ void stop();
+
+ /**
+ * @brief Reads the pid from stable storage
+ *
+ * @return The pid number
+ */
+ int readPid();
+
+ /**
+ * @brief Writes the pid from stable storage
+ *
+ * @return
+ * - 0 if successful
+ - -1 and sets errno if an error occured
+ */
+ int writePid(int pid);
+
+ /**
+ * @brief Prints a textual description of the process on a file
+ */
+ void print(FILE *);
+
+ /** Id number of the Process.
+ *
+ * @note This is not the same as a pid. This number is used in the
+ * protocol, and will not be changed if a processes is restarted.
+ */
+ int m_id;
+
+ /** @brief The name shown to the user */
+ BaseString m_name;
+
+ /** @brief Used to group a number of processes */
+ BaseString m_group;
+
+ /** @brief Environment variables
+ *
+ * Environmentvariables to add for the process.
+ *
+ * @note
+ * - The environment cpcd started with is preserved
+ * - There is no way to delete variables
+ */
+ BaseString m_env;
+
+ /** @brief Path to the binary to run */
+ BaseString m_path;
+
+ /** @brief Arguments to the process.
+ *
+ * @note
+ * - This includes argv[0].
+ * - If no argv[0] is given, argv[0] will be set to m_path.
+ */
+ BaseString m_args;
+
+ /**
+ * @brief Type of process
+ *
+ * Either set to "interactive" or "permanent".
+ */
+ BaseString m_type;
+ ProcessType m_processType;
+
+ /**
+ * @brief Working directory
+ *
+ * Working directory the process will start in.
+ */
+ BaseString m_cwd;
+
+ /**
+ * @brief Owner of the process.
+ *
+ * @note This will not affect the process' uid or gid;
+ * it is only used for managemental purposes.
+ * @see m_runas
+ */
+ BaseString m_owner;
+
+ /**
+ * @bried Run as
+ * @note This affects uid
+ * @see m_owner
+ */
+ BaseString m_runas;
+
+ /**
+ * @brief redirection for stdin
+ */
+ BaseString m_stdin;
+
+ /**
+ * @brief redirection for stdout
+ */
+ BaseString m_stdout;
+
+ /**
+ * @brief redirection for stderr
+ */
+ BaseString m_stderr;
+
+ /** @brief Status of the process */
+ enum ProcessStatus m_status;
+
+ /**
+ * @brief ulimits for process
+ * @desc Format c:unlimited d:0 ...
+ */
+ BaseString m_ulimit;
+ private:
+ class CPCD *m_cpcd;
+ void do_exec();
+ };
+
+ /**
+ * @brief Starts and stops processes as needed
+ *
+ * At a specified interval (default 5 seconds) calls the monitor function
+ * of all the processes in the CPCDs list, causing the to start or
+ * stop, depending on the configuration.
+ */
+ class Monitor {
+ public:
+ /** Creates a new CPCD::Monitor object, connected to the specified
+ * CPCD.
+ * A new thread will be created, which will poll the processes of
+ * the CPCD at the specifed interval.
+ */
+ Monitor(CPCD *cpcd, int poll = CPCD_DEFAULT_POLLING_INTERVAL);
+
+ /** Stops the monitor, but does not stop the processes */
+ ~Monitor();
+
+ /** Runs the monitor thread. */
+ void run();
+
+ /** Signals configuration changes to the monitor thread, causing it to
+ * do the check without waiting for the timeout */
+ void signal();
+ private:
+ class CPCD *m_cpcd;
+ struct NdbThread *m_monitorThread;
+ bool m_monitorThreadQuitFlag;
+ struct NdbCondition *m_changeCondition;
+ NdbMutex *m_changeMutex;
+ int m_pollingInterval; /* seconds */
+ };
+
+ /** @brief Constructs a CPCD object */
+ CPCD();
+
+ /**
+ * @brief Destroys a CPCD object,
+ * but does not stop the processes it manages
+ */
+ ~CPCD();
+
+ /** Adds a Process to the CPCDs list of managed Processes.
+ *
+ * @note The process will not be started until it is explicitly
+ * marked as running with CPCD::startProcess().
+ *
+ * @return
+ * - true if the addition was successful,
+ * - false if not
+ * - RequestStatus will be filled in with a suitable error
+ * if an error occured.
+ */
+ bool defineProcess(RequestStatus *rs, Process * arg);
+
+ /** Removes a Process from the CPCD.
+ *
+ * @note A Process that is running cannot be removed.
+ *
+ * @return
+ * - true if the removal was successful,
+ * - false if not
+ * - The RequestStatus will be filled in with a suitable error
+ * if an error occured.
+ */
+ bool undefineProcess(RequestStatus *rs, int id);
+
+ /** Marks a Process for starting.
+ *
+ * @note The fact that a process has started does not mean it will actually
+ * start properly. This command only makes sure the CPCD will
+ * try to start it.
+ *
+ * @return
+ * - true if the marking was successful
+ * - false if not
+ * - RequestStatus will be filled in with a suitable error
+ * if an error occured.
+ */
+ bool startProcess(RequestStatus *rs, int id);
+
+ /** Marks a Process for stopping.
+ *
+ * @return
+ * - true if the marking was successful
+ * - false if not
+ * - The RequestStatus will be filled in with a suitable error
+ * if an error occured.
+ */
+ bool stopProcess(RequestStatus *rs, int id);
+
+ /** Generates a list of processes, and sends them to the CPCD client */
+ bool listProcesses(RequestStatus *rs, MutexVector<const char *> &);
+
+ /** Set to true while the CPCD is reading the configuration file */
+ bool loadingProcessList;
+
+ /** Saves the list of Processes and their status to the configuration file.
+ * Called whenever the configuration is changed.
+ */
+ bool saveProcessList();
+
+ /** Loads the list of Processes and their status from the configuration
+ * file.
+ * @note This function should only be called when the CPCD is starting,
+ * calling it at other times will cause unspecified behaviour.
+ */
+ bool loadProcessList();
+
+ /** Returns the list of processes */
+ MutexVector<Process *> *getProcessList();
+
+ /** The list of processes. Should not be used directly */
+ MutexVector<Process *> m_processes;
+
+ /** Register event subscriber */
+ void do_register(EventSubscriber * sub);
+ EventSubscriber* do_unregister(EventSubscriber * sub);
+
+private:
+ friend class Process;
+ bool notifyChanges();
+ int findUniqueId();
+ BaseString m_procfile;
+ Monitor *m_monitor;
+
+ void report(int id, CPCEvent::EventType);
+ MutexVector<EventSubscriber *> m_subscribers;
+};
+
+#endif
diff --git a/ndb/src/cw/cpcd/Makefile b/ndb/src/cw/cpcd/Makefile
new file mode 100644
index 00000000000..f214fb087d2
--- /dev/null
+++ b/ndb/src/cw/cpcd/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := util
+BIN_TARGET := ndb_cpcd
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = main.cpp CPCD.cpp Process.cpp APIService.cpp Monitor.cpp common.cpp
+
+BIN_TARGET_LIBS += logger
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/cw/cpcd/Monitor.cpp b/ndb/src/cw/cpcd/Monitor.cpp
new file mode 100644
index 00000000000..a96f3509ee8
--- /dev/null
+++ b/ndb/src/cw/cpcd/Monitor.cpp
@@ -0,0 +1,76 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbThread.h>
+#include <NdbOut.hpp>
+#include <NdbUnistd.h>
+#include <NdbSleep.h>
+
+#include "CPCD.hpp"
+#include "common.hpp"
+
+static void *
+monitor_thread_create_wrapper(void * arg) {
+ CPCD::Monitor *mon = (CPCD::Monitor *)arg;
+ mon->run();
+ return NULL;
+}
+
+CPCD::Monitor::Monitor(CPCD *cpcd, int poll) {
+ m_cpcd = cpcd;
+ m_pollingInterval = poll;
+ m_changeCondition = NdbCondition_Create();
+ m_changeMutex = NdbMutex_Create();
+ m_monitorThread = NdbThread_Create(monitor_thread_create_wrapper,
+ (NDB_THREAD_ARG*) this,
+ 32768,
+ "ndb_cpcd_monitor",
+ NDB_THREAD_PRIO_MEAN);
+ m_monitorThreadQuitFlag = false;
+}
+
+CPCD::Monitor::~Monitor() {
+ NdbThread_Destroy(&m_monitorThread);
+ NdbCondition_Destroy(m_changeCondition);
+ NdbMutex_Destroy(m_changeMutex);
+}
+
+void
+CPCD::Monitor::run() {
+ while(1) {
+ NdbMutex_Lock(m_changeMutex);
+ NdbCondition_WaitTimeout(m_changeCondition,
+ m_changeMutex,
+ m_pollingInterval * 1000);
+
+ MutexVector<CPCD::Process *> &proc = *m_cpcd->getProcessList();
+
+ proc.lock();
+
+ for(size_t i = 0; i < proc.size(); i++) {
+ proc[i]->monitor();
+ }
+
+ proc.unlock();
+
+ NdbMutex_Unlock(m_changeMutex);
+ }
+}
+
+void
+CPCD::Monitor::signal() {
+ NdbCondition_Signal(m_changeCondition);
+}
diff --git a/ndb/src/cw/cpcd/Process.cpp b/ndb/src/cw/cpcd/Process.cpp
new file mode 100644
index 00000000000..01a63a5c653
--- /dev/null
+++ b/ndb/src/cw/cpcd/Process.cpp
@@ -0,0 +1,482 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <sys/types.h>
+#include <signal.h>
+
+#include <assert.h>
+#include <stdlib.h>
+
+#include <NdbUnistd.h>
+#include <BaseString.hpp>
+#include <InputStream.hpp>
+
+#include "common.hpp"
+#include "CPCD.hpp"
+
+#include <pwd.h>
+#include <sys/types.h>
+#include <unistd.h>
+#include <sys/stat.h>
+#include <sys/resource.h>
+
+void
+CPCD::Process::print(FILE * f){
+ fprintf(f, "define process\n");
+ fprintf(f, "id: %d\n", m_id);
+ fprintf(f, "name: %s\n", m_name.c_str() ? m_name.c_str() : "");
+ fprintf(f, "group: %s\n", m_group.c_str() ? m_group.c_str() : "");
+ fprintf(f, "env: %s\n", m_env.c_str() ? m_env.c_str() : "");
+ fprintf(f, "path: %s\n", m_path.c_str() ? m_path.c_str() : "");
+ fprintf(f, "args: %s\n", m_args.c_str() ? m_args.c_str() : "");
+ fprintf(f, "type: %s\n", m_type.c_str() ? m_type.c_str() : "");
+ fprintf(f, "cwd: %s\n", m_cwd.c_str() ? m_cwd.c_str() : "");
+ fprintf(f, "owner: %s\n", m_owner.c_str() ? m_owner.c_str() : "");
+ fprintf(f, "runas: %s\n", m_runas.c_str() ? m_runas.c_str() : "");
+ fprintf(f, "stdin: %s\n", m_stdin.c_str() ? m_stdin.c_str() : "");
+ fprintf(f, "stdout: %s\n", m_stdout.c_str() ? m_stdout.c_str() : "");
+ fprintf(f, "stderr: %s\n", m_stderr.c_str() ? m_stderr.c_str() : "");
+ fprintf(f, "ulimit: %s\n", m_ulimit.c_str() ? m_ulimit.c_str() : "");
+}
+
+CPCD::Process::Process(const Properties & props, class CPCD *cpcd) {
+ m_id = -1;
+ m_pid = -1;
+ props.get("id", (Uint32 *) &m_id);
+ props.get("name", m_name);
+ props.get("group", m_group);
+ props.get("env", m_env);
+ props.get("path", m_path);
+ props.get("args", m_args);
+ props.get("cwd", m_cwd);
+ props.get("owner", m_owner);
+ props.get("type", m_type);
+ props.get("runas", m_runas);
+
+ props.get("stdin", m_stdin);
+ props.get("stdout", m_stdout);
+ props.get("stderr", m_stderr);
+ props.get("ulimit", m_ulimit);
+ m_status = STOPPED;
+
+ if(strcasecmp(m_type.c_str(), "temporary") == 0){
+ m_processType = TEMPORARY;
+ } else {
+ m_processType = PERMANENT;
+ }
+
+ m_cpcd = cpcd;
+}
+
+void
+CPCD::Process::monitor() {
+ switch(m_status) {
+ case STARTING:
+ break;
+ case RUNNING:
+ if(!isRunning()){
+ m_cpcd->report(m_id, CPCEvent::ET_PROC_STATE_STOPPED);
+ if(m_processType == TEMPORARY){
+ m_status = STOPPED;
+ } else {
+ start();
+ }
+ }
+ break;
+ case STOPPED:
+ assert(!isRunning());
+ break;
+ case STOPPING:
+ break;
+ }
+}
+
+bool
+CPCD::Process::isRunning() {
+
+ if(m_pid <= 1){
+ logger.critical("isRunning(%d) invalid pid: %d", m_id, m_pid);
+ return false;
+ }
+ /* Check if there actually exists a process with such a pid */
+ errno = 0;
+ int s = kill((pid_t) m_pid, 0); /* Sending "signal" 0 to a process only
+ * checkes if the process actually exists */
+ if(s != 0) {
+ switch(errno) {
+ case EPERM:
+ logger.critical("Not enough privileges to control pid %d\n", m_pid);
+ break;
+ case ESRCH:
+ /* The pid in the file does not exist, which probably means that it
+ has died, or the file contains garbage for some other reason */
+ break;
+ default:
+ logger.critical("Cannot not control pid %d: %s\n", m_pid, strerror(errno));
+ break;
+ }
+ return false;
+ }
+
+ return true;
+}
+
+int
+CPCD::Process::readPid() {
+ if(m_pid != -1){
+ logger.critical("Reading pid while != -1(%d)", m_pid);
+ return m_pid;
+ }
+
+ char filename[PATH_MAX*2+1];
+ char buf[1024];
+ FILE *f;
+
+ memset(buf, 0, sizeof(buf));
+
+ snprintf(filename, sizeof(filename), "%d", m_id);
+
+ f = fopen(filename, "r");
+
+ if(f == NULL){
+ logger.debug("readPid - %s not found", filename);
+ return -1; /* File didn't exist */
+ }
+
+ errno = 0;
+ size_t r = fread(buf, 1, sizeof(buf), f);
+ fclose(f);
+ if(r > 0)
+ m_pid = strtol(buf, (char **)NULL, 0);
+
+ if(errno == 0){
+ return m_pid;
+ }
+
+ return -1;
+}
+
+int
+CPCD::Process::writePid(int pid) {
+ char tmpfilename[PATH_MAX+1+4+8];
+ char filename[PATH_MAX*2+1];
+ FILE *f;
+
+ snprintf(tmpfilename, sizeof(tmpfilename), "tmp.XXXXXX");
+ snprintf(filename, sizeof(filename), "%d", m_id);
+
+ int fd = mkstemp(tmpfilename);
+ if(fd < 0) {
+ logger.error("Cannot open `%s': %s\n", tmpfilename, strerror(errno));
+ return -1; /* Couldn't open file */
+ }
+
+ f = fdopen(fd, "w");
+
+ if(f == NULL) {
+ logger.error("Cannot open `%s': %s\n", tmpfilename, strerror(errno));
+ return -1; /* Couldn't open file */
+ }
+
+ fprintf(f, "%d", pid);
+ fclose(f);
+
+ if(rename(tmpfilename, filename) == -1){
+ logger.error("Unable to rename from %s to %s", tmpfilename, filename);
+ return -1;
+ }
+ return 0;
+}
+
+static void
+setup_environment(const char *env) {
+ char **p;
+ p = BaseString::argify("", env);
+ for(int i = 0; p[i] != NULL; i++){
+ /*int res = */ putenv(p[i]);
+ }
+}
+
+static
+int
+set_ulimit(const BaseString & pair){
+ errno = 0;
+ do {
+ Vector<BaseString> list;
+ pair.split(list, ":");
+ if(list.size() != 2){
+ break;
+ }
+
+ int resource = 0;
+ rlim_t value = RLIM_INFINITY;
+ if(!(list[1].trim() == "unlimited")){
+ value = atoi(list[1].c_str());
+ }
+ if(list[0].trim() == "c"){
+ resource = RLIMIT_CORE;
+ } else if(list[0] == "d"){
+ resource = RLIMIT_DATA;
+ } else if(list[0] == "f"){
+ resource = RLIMIT_FSIZE;
+ } else if(list[0] == "n"){
+ resource = RLIMIT_NOFILE;
+ } else if(list[0] == "s"){
+ resource = RLIMIT_STACK;
+ } else if(list[0] == "t"){
+ resource = RLIMIT_CPU;
+ } else {
+ errno = EINVAL;
+ break;
+ }
+ struct rlimit rlp;
+ if(getrlimit(resource, &rlp) != 0){
+ break;
+ }
+
+ rlp.rlim_cur = value;
+ if(setrlimit(resource, &rlp) != 0){
+ break;
+ }
+ return 0;
+ } while(false);
+ logger.error("Unable to process ulimit: %s(%s)",
+ pair.c_str(), strerror(errno));
+ return -1;
+}
+
+void
+CPCD::Process::do_exec() {
+
+ setup_environment(m_env.c_str());
+
+ char **argv = BaseString::argify(m_path.c_str(), m_args.c_str());
+
+ if(strlen(m_cwd.c_str()) > 0) {
+ int err = chdir(m_cwd.c_str());
+ if(err == -1) {
+ BaseString err;
+ logger.error("%s: %s\n", m_cwd.c_str(), strerror(errno));
+ _exit(1);
+ }
+ }
+
+ Vector<BaseString> ulimit;
+ m_ulimit.split(ulimit);
+ for(size_t i = 0; i<ulimit.size(); i++){
+ if(ulimit[i].trim().length() > 0 && set_ulimit(ulimit[i]) != 0){
+ _exit(1);
+ }
+ }
+
+ int fd = open("/dev/null", O_RDWR, 0);
+ if(fd == -1) {
+ logger.error("Cannot open `/dev/null': %s\n", strerror(errno));
+ _exit(1);
+ }
+
+ BaseString * redirects[] = { &m_stdin, &m_stdout, &m_stderr };
+ int fds[3];
+ for(int i = 0; i<3; i++){
+ if(redirects[i]->empty()){
+#ifndef DEBUG
+ dup2(fd, i);
+#endif
+ continue;
+ }
+
+ if((* redirects[i]) == "2>&1" && i == 2){
+ dup2(fds[1], 2);
+ continue;
+ }
+
+ /**
+ * Make file
+ */
+ int flags = 0;
+ int mode = S_IRUSR | S_IWUSR ;
+ if(i == 0){
+ flags |= O_RDONLY;
+ } else {
+ flags |= O_WRONLY | O_CREAT | O_APPEND;
+ }
+ int f = fds[i]= open(redirects[i]->c_str(), flags, mode);
+ if(f == -1){
+ logger.error("Cannot redirect %d to/from '%s' : %s\n", i,
+ redirects[i]->c_str(), strerror(errno));
+ _exit(1);
+ }
+ dup2(f, i);
+ }
+
+ /* Close all filedescriptors */
+ for(int i = STDERR_FILENO+1; i < getdtablesize(); i++)
+ close(i);
+
+ execv(m_path.c_str(), argv);
+ /* XXX If we reach this point, an error has occurred, but it's kind of hard
+ * to report it, because we've closed all files... So we should probably
+ * create a new logger here */
+ logger.error("Exec failed: %s\n", strerror(errno));
+ /* NOTREACHED */
+}
+
+int
+CPCD::Process::start() {
+ /* We need to fork() twice, so that the second child (grandchild?) can
+ * become a daemon. The original child then writes the pid file,
+ * so that the monitor knows the pid of the new process, and then
+ * exit()s. That way, the monitor process can pickup the pid, and
+ * the running process is a daemon.
+ *
+ * This is a bit tricky but has the following advantages:
+ * - the cpcd can die, and "reconnect" to the monitored clients
+ * without restarting them.
+ * - the cpcd does not have to wait() for the childs. init(1) will
+ * take care of that.
+ */
+ logger.info("Starting %d: %s", m_id, m_name.c_str());
+ m_status = STARTING;
+
+ int pid = -1;
+ switch(m_processType){
+ case TEMPORARY:{
+ /**
+ * Simple fork
+ * don't ignore child
+ */
+ switch(pid = fork()) {
+ case 0: /* Child */
+
+ if(runas(m_runas.c_str()) == 0){
+ writePid(getpid());
+ do_exec();
+ }
+ _exit(1);
+ break;
+ case -1: /* Error */
+ logger.error("Cannot fork: %s\n", strerror(errno));
+ m_status = STOPPED;
+ return -1;
+ break;
+ default: /* Parent */
+ logger.debug("Started temporary %d : pid=%d", m_id, pid);
+ m_cpcd->report(m_id, CPCEvent::ET_PROC_STATE_RUNNING);
+ break;
+ }
+ break;
+ }
+ case PERMANENT:{
+ /**
+ * PERMANENT
+ */
+ switch(fork()) {
+ case 0: /* Child */
+ if(runas(m_runas.c_str()) != 0){
+ writePid(-1);
+ _exit(1);
+ }
+ signal(SIGCHLD, SIG_IGN);
+ pid_t pid;
+ switch(pid = fork()) {
+ case 0: /* Child */
+ writePid(getpid());
+ setsid();
+ do_exec();
+ _exit(1);
+ /* NOTREACHED */
+ break;
+ case -1: /* Error */
+ logger.error("Cannot fork: %s\n", strerror(errno));
+ writePid(-1);
+ _exit(1);
+ break;
+ default: /* Parent */
+ logger.debug("Started permanent %d : pid=%d", m_id, pid);
+ _exit(0);
+ break;
+ }
+ break;
+ case -1: /* Error */
+ logger.error("Cannot fork: %s\n", strerror(errno));
+ m_status = STOPPED;
+ return -1;
+ break;
+ default: /* Parent */
+ m_cpcd->report(m_id, CPCEvent::ET_PROC_STATE_RUNNING);
+ break;
+ }
+ break;
+ }
+ default:
+ logger.critical("Unknown process type");
+ return -1;
+ }
+
+ while(readPid() < 0){
+ sched_yield();
+ }
+
+ if(pid != -1 && pid != m_pid){
+ logger.error("pid and m_pid don't match: %d %d", pid, m_pid);
+ }
+
+ if(isRunning()){
+ m_status = RUNNING;
+ return 0;
+ }
+ m_status = STOPPED;
+ return -1;
+}
+
+void
+CPCD::Process::stop() {
+
+ char filename[PATH_MAX*2+1];
+ snprintf(filename, sizeof(filename), "%d", m_id);
+ unlink(filename);
+
+ if(m_pid <= 1){
+ logger.critical("Stopping process with bogus pid: %d", m_pid);
+ return;
+ }
+ m_status = STOPPING;
+
+ int ret = kill((pid_t)m_pid, SIGTERM);
+ switch(ret) {
+ case 0:
+ logger.debug("Sent SIGTERM to pid %d", (int)m_pid);
+ break;
+ default:
+ logger.debug("kill pid: %d : %s", (int)m_pid, strerror(errno));
+ break;
+ }
+
+ if(isRunning()){
+ ret = kill((pid_t)m_pid, SIGKILL);
+ switch(ret) {
+ case 0:
+ logger.debug("Sent SIGKILL to pid %d", (int)m_pid);
+ break;
+ default:
+ logger.debug("kill pid: %d : %s\n", (int)m_pid, strerror(errno));
+ break;
+ }
+ }
+
+ m_pid = -1;
+ m_status = STOPPED;
+}
diff --git a/ndb/src/cw/cpcd/common.cpp b/ndb/src/cw/cpcd/common.cpp
new file mode 100644
index 00000000000..731866b22fd
--- /dev/null
+++ b/ndb/src/cw/cpcd/common.cpp
@@ -0,0 +1,158 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "common.hpp"
+#include <logger/Logger.hpp>
+#include <pwd.h>
+#include <sys/types.h>
+#include <unistd.h>
+
+#include <Properties.hpp>
+#include <BaseString.hpp>
+
+int debug = 0;
+
+Logger logger;
+
+int
+runas(const char * user){
+ if(user == 0 || strlen(user) == 0){
+ return 0;
+ }
+ struct passwd * pw = getpwnam(user);
+ if(pw == 0){
+ logger.error("Can't find user to %s", user);
+ return -1;
+ }
+ uid_t uid = pw->pw_uid;
+ gid_t gid = pw->pw_gid;
+ int res = setgid(gid);
+ if(res != 0){
+ logger.error("Can't change group to %s(%d)", user, gid);
+ return res;
+ }
+
+ res = setuid(uid);
+ if(res != 0){
+ logger.error("Can't change user to %s(%d)", user, uid);
+ }
+ return res;
+}
+
+int
+insert(const char * pair, Properties & p){
+ BaseString tmp(pair);
+
+ tmp.trim(" \t\n\r");
+
+ Vector<BaseString> split;
+ tmp.split(split, ":=", 2);
+
+ if(split.size() != 2)
+ return -1;
+
+ p.put(split[0].trim().c_str(), split[1].trim().c_str());
+
+ return 0;
+}
+
+int
+insert_file(FILE * f, class Properties& p, bool break_on_empty){
+ if(f == 0)
+ return -1;
+
+ while(!feof(f)){
+ char buf[1024];
+ fgets(buf, 1024, f);
+ BaseString tmp = buf;
+
+ if(tmp.length() > 0 && tmp.c_str()[0] == '#')
+ continue;
+
+ if(insert(tmp.c_str(), p) != 0 && break_on_empty)
+ break;
+ }
+
+ return 0;
+}
+
+int
+insert_file(const char * filename, class Properties& p){
+ FILE * f = fopen(filename, "r");
+ int res = insert_file(f, p);
+ if(f) fclose(f);
+ return res;
+}
+
+int
+parse_config_file(struct getargs args[], int num_arg, const Properties& p){
+ Properties::Iterator it(&p);
+ for(const char * name = it.first(); name != 0; name = it.next()){
+ bool found = false;
+ for(int i = 0; i<num_arg; i++){
+ if(strcmp(name, args[i].long_name) != 0)
+ continue;
+
+ found = true;
+
+ const char * tmp;
+ p.get(name, &tmp);
+
+ int t = 1;
+
+ switch(args[i].type){
+ case arg_integer:{
+ int val = atoi(tmp);
+ if(args[i].value){
+ *((int*)args[i].value) = val;
+ }
+ }
+ break;
+ case arg_string:
+ if(args[i].value){
+ *((const char**)args[i].value) = tmp;
+ }
+ break;
+ case arg_negative_flag:
+ t = 0;
+ case arg_flag:
+ if(args[i].value){
+ if(!strcasecmp(tmp, "y") ||
+ !strcasecmp(tmp, "on") ||
+ !strcasecmp(tmp, "true") ||
+ !strcasecmp(tmp, "1")){
+ *((int*)args[i].value) = t;
+ }
+ if(!strcasecmp(tmp, "n") ||
+ !strcasecmp(tmp, "off") ||
+ !strcasecmp(tmp, "false") ||
+ !strcasecmp(tmp, "0")){
+ *((int*)args[i].value) = t;
+ }
+ }
+ t = 1;
+ break;
+ case arg_strings:
+ case arg_double:
+ case arg_collect:
+ case arg_counter:
+ break;
+ }
+ }
+ if(!found)
+ printf("Unknown parameter: %s\n", name);
+ }
+}
diff --git a/ndb/src/cw/cpcd/common.hpp b/ndb/src/cw/cpcd/common.hpp
new file mode 100644
index 00000000000..65fcce05f66
--- /dev/null
+++ b/ndb/src/cw/cpcd/common.hpp
@@ -0,0 +1,35 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __CPCD_COMMON_HPP_INCLUDED__
+#define __CPCD_COMMON_HPP_INCLUDED__
+
+#include <stdio.h>
+#include <logger/Logger.hpp>
+#include <getarg.h>
+
+extern int debug;
+
+extern Logger logger;
+
+int runas(const char * user);
+int insert(const char * pair, class Properties & p);
+
+int insert_file(const char * filename, class Properties&);
+int insert_file(FILE *, class Properties&, bool break_on_empty = false);
+int parse_config_file(struct getargs args[], int num_arg, const Properties& p);
+
+#endif /* ! __CPCD_COMMON_HPP_INCLUDED__ */
diff --git a/ndb/src/cw/cpcd/main.cpp b/ndb/src/cw/cpcd/main.cpp
new file mode 100644
index 00000000000..8dd4f2b4608
--- /dev/null
+++ b/ndb/src/cw/cpcd/main.cpp
@@ -0,0 +1,178 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <sys/types.h> /* Needed for mkdir(2) */
+#include <sys/stat.h> /* Needed for mkdir(2) */
+#include <errno.h>
+#include <signal.h>
+
+#include "CPCD.hpp"
+#include "APIService.hpp"
+#include <NdbMain.h>
+#include <NdbSleep.h>
+#include <BaseString.hpp>
+#include <getarg.h>
+#include <logger/Logger.hpp>
+#include <logger/FileLogHandler.hpp>
+#include <logger/SysLogHandler.hpp>
+
+#include "common.hpp"
+
+static char *work_dir = CPCD_DEFAULT_WORK_DIR;
+static int port = CPCD_DEFAULT_TCP_PORT;
+static int use_syslog = 0;
+static char *logfile = NULL;
+static char *config_file = CPCD_DEFAULT_CONFIG_FILE;
+static char *user = 0;
+
+static struct getargs args[] = {
+ { "work-dir", 'w', arg_string, &work_dir,
+ "Work directory", "directory" },
+ { "port", 'p', arg_integer, &port,
+ "TCP port to listen on", "port" },
+ { "syslog", 'S', arg_flag, &use_syslog,
+ "Log events to syslog", NULL},
+ { "logfile", 'L', arg_string, &logfile,
+ "File to log events to", "file"},
+ { "debug", 'D', arg_flag, &debug,
+ "Enable debug mode", NULL},
+ { "config", 'c', arg_string, &config_file, "Config file", NULL },
+ { "user", 'u', arg_string, &user, "Run as user", NULL }
+};
+
+static const int num_args = sizeof(args) / sizeof(args[0]);
+
+static CPCD * g_cpcd = 0;
+#if 0
+extern "C" static void sig_child(int signo, siginfo_t*, void*);
+#endif
+
+const char *progname = "ndb_cpcd";
+
+NDB_MAIN(ndb_cpcd){
+ int optind = 0;
+
+ if(getarg(args, num_args, argc, argv, &optind)) {
+ arg_printusage(args, num_args, progname, "");
+ exit(1);
+ }
+
+ Properties p;
+ insert_file(config_file, p);
+ if(parse_config_file(args, num_args, p)){
+ ndbout_c("Invalid config file: %s", config_file);
+ exit(1);
+ }
+
+ if(getarg(args, num_args, argc, argv, &optind)) {
+ arg_printusage(args, num_args, progname, "");
+ exit(1);
+ }
+
+ logger.setCategory(progname);
+ logger.enable(Logger::LL_ALL);
+
+ if(debug)
+ logger.createConsoleHandler();
+
+ if(user && runas(user) != 0){
+ logger.critical("Unable to change user: %s", user);
+ _exit(1);
+ }
+
+ if(logfile != NULL){
+ BaseString tmp;
+ if(logfile[0] != '/')
+ tmp.append(work_dir);
+ tmp.append(logfile);
+ logger.addHandler(new FileLogHandler(tmp.c_str()));
+ }
+
+ if(use_syslog)
+ logger.addHandler(new SysLogHandler());
+
+ logger.info("Starting");
+
+ CPCD cpcd;
+ g_cpcd = &cpcd;
+
+ /* XXX This will probably not work on !unix */
+ int err = mkdir(work_dir, S_IRWXU | S_IRGRP | S_IROTH);
+ if(err != 0) {
+ switch(errno) {
+ case EEXIST:
+ break;
+ default:
+ fprintf(stderr, "Cannot mkdir %s: %s\n", work_dir, strerror(errno));
+ exit(1);
+ }
+ }
+
+ if(strlen(work_dir) > 0){
+ logger.debug("Changing dir to '%s'", work_dir);
+ if((err = chdir(work_dir)) != 0){
+ fprintf(stderr, "Cannot chdir %s: %s\n", work_dir, strerror(errno));
+ exit(1);
+ }
+ }
+
+ cpcd.loadProcessList();
+
+ SocketServer * ss = new SocketServer();
+ CPCDAPIService * serv = new CPCDAPIService(cpcd);
+ if(!ss->setup(serv, port)){
+ logger.critical("Cannot setup server: %s", strerror(errno));
+ sleep(1);
+ delete ss;
+ delete serv;
+ return 1;
+ }
+
+ ss->startServer();
+
+ {
+ signal(SIGPIPE, SIG_IGN);
+ signal(SIGCHLD, SIG_IGN);
+#if 0
+ struct sigaction act;
+ act.sa_handler = 0;
+ act.sa_sigaction = sig_child;
+ sigemptyset(&act.sa_mask);
+ act.sa_flags = SA_SIGINFO;
+ sigaction(SIGCHLD, &act, 0);
+#endif
+ }
+
+ logger.debug("Start completed");
+ while(true) NdbSleep_MilliSleep(1000);
+
+ delete ss;
+ return 0;
+}
+
+#if 0
+extern "C"
+void
+sig_child(int signo, siginfo_t* info, void*){
+ printf("signo: %d si_signo: %d si_errno: %d si_code: %d si_pid: %d\n",
+ signo,
+ info->si_signo,
+ info->si_errno,
+ info->si_code,
+ info->si_pid);
+
+}
+#endif
diff --git a/ndb/src/cw/test/socketclient/Makefile b/ndb/src/cw/test/socketclient/Makefile
new file mode 100644
index 00000000000..04f11f031e5
--- /dev/null
+++ b/ndb/src/cw/test/socketclient/Makefile
@@ -0,0 +1,24 @@
+include .defs.mk
+
+TYPE :=
+
+BIN_TARGET := socketclient
+
+
+
+CCFLAGS_LOC += -I../../util/ -I../../cpcd/
+
+LIBS_LOC += -L$(NDB_TOP)/lib/ -L$(EXTERNAL_LIB_DIR)/sci
+
+LIBS_SPEC += -lsocketclient
+
+
+SOURCES = socketClientTest.cpp
+
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
diff --git a/ndb/src/cw/test/socketclient/socketClientTest.cpp b/ndb/src/cw/test/socketclient/socketClientTest.cpp
new file mode 100644
index 00000000000..a4a0ed1e933
--- /dev/null
+++ b/ndb/src/cw/test/socketclient/socketClientTest.cpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbOut.hpp>
+#include <Properties.hpp>
+#include <socket_io.h>
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbSleep.h>
+#include "SocketService.hpp"
+#include "SocketRegistry.hpp"
+#include "SocketClient.hpp"
+#include "ClientInterface.hpp"
+
+#include <InputStream.hpp>
+
+#include <Parser.hpp>
+
+NDB_MAIN(socketclient) {
+
+
+ if(argc<3) {
+ printf("wrong args: socketclient <hostname> <port>\n");
+ return 0;
+ }
+ const char * remotehost = argv[1];
+ const int port = atoi(argv[2]);
+
+
+ ClientInterface * ci = new ClientInterface(2);
+ ci->connectCPCDdaemon(remotehost,port);
+
+ /*ci->listProcesses(remotehost);
+
+ ci->startProcess(remotehost, "1247");
+
+ ci->stopProcess(remotehost, "1247");*/
+
+ ci->defineProcess(remotehost, "ndb", "ndb-cluster1", "envirnm", "/ndb/bin",
+ "-i", "permanent", "/ndb/ndb.2", "team");
+
+ ci->startProcess(remotehost, "1247");
+
+ ci->listProcesses(remotehost);
+
+ //ci->undefineProcess(remotehost, "1247");
+
+ ci->disconnectCPCDdaemon(remotehost);
+}
diff --git a/ndb/src/cw/util/ClientInterface.cpp b/ndb/src/cw/util/ClientInterface.cpp
new file mode 100644
index 00000000000..627b622f1dd
--- /dev/null
+++ b/ndb/src/cw/util/ClientInterface.cpp
@@ -0,0 +1,185 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "ClientInterface.hpp"
+
+
+
+ClientInterface::ClientInterface(Uint32 maxNoOfCPCD) {
+ sr = new SocketRegistry<SocketService>(maxNoOfCPCD);
+ ss = new SocketService();
+}
+
+
+ClientInterface::~ClientInterface() {
+ delete sr;
+ delete ss;
+
+}
+
+
+void ClientInterface::connectCPCDdaemon(const char * remotehost, Uint16 port)
+{
+ sr->createSocketClient(remotehost, port);
+}
+
+void ClientInterface::disconnectCPCDdaemon(const char * remotehost)
+{
+ sr->removeSocketClient(remotehost);
+}
+
+void ClientInterface::removeCPCDdaemon(const char * remotehost)
+{
+ sr->removeSocketClient(remotehost);
+}
+
+void ClientInterface::startProcess(const char * remotehost, char * id) {
+ char buf[255] = "start process ";
+ char str[80];
+ char line[10];
+
+ strcpy(line, id);
+ strcpy(str, "id:");
+ strcat(str, line);
+ strcat(str, "\n\n");
+ strcat(buf, str);
+ printf("Request: %s\n", buf);
+
+ sr->performSend(buf,255,remotehost);
+ sr->syncPerformReceive(remotehost, *ss, 0);
+ ss->getPropertyObject();
+}
+
+void ClientInterface::stopProcess(const char * remotehost, char * id) {
+ char buf[255] = "stop process ";
+ char str[80];
+ char line[10];
+
+ strcpy(line, id);
+ strcpy(str, "id:");
+ strcat(str, line);
+ strcat(str, "\n\n");
+ strcat(buf, str);
+ printf("Request: %s\n", buf);
+
+ sr->performSend(buf,255,remotehost);
+ sr->syncPerformReceive(remotehost, *ss, 0);
+ ss->getPropertyObject();
+}
+
+void ClientInterface::defineProcess(const char * remotehost, char * name,
+ char * group, char * env, char * path,
+ char * args, char * type, char * cwd, char * owner){
+ char buf[255] = "define process ";
+ char str[80];
+ char line[10];
+
+ strcpy(line, name);
+ strcpy(str, "name:");
+ strcat(str, line);
+ strcat(buf, str);
+ strcat(buf, " \n");
+
+ strcpy(line, group);
+ strcpy(str, "group:");
+ strcat(str, line);
+ strcat(buf, str);
+ strcat(buf, " \n");
+
+ strcpy(line, env);
+ strcpy(str, "env:");
+ strcat(str, line);
+ strcat(buf, str);
+ strcat(buf, " \n");
+
+ strcpy(line, path);
+ strcpy(str, "path:");
+ strcat(str, line);
+ strcat(buf, str);
+ strcat(buf, " \n");
+
+ strcpy(line, args);
+ strcpy(str, "args:");
+ strcat(str, line);
+ strcat(buf, str);
+ strcat(buf, " \n");
+
+ strcpy(line, type);
+ strcpy(str, "type:");
+ strcat(str, line);
+ strcat(buf, str);
+ strcat(buf, " \n");
+
+ strcpy(line, cwd);
+ strcpy(str, "cwd:");
+ strcat(str, line);
+ strcat(buf, str);
+ strcat(buf, " \n");
+
+ strcpy(line, owner);
+ strcpy(str, "owner:");
+ strcat(str, line);
+ strcat(buf, str);
+ strcat(buf, "\n\n");
+
+ printf("Request: %s\n", buf);
+
+ sr->performSend(buf,255,remotehost);
+ sr->syncPerformReceive(remotehost, *ss, 0);
+ ss->getPropertyObject();
+}
+
+void ClientInterface::undefineProcess(const char * remotehost, char * id){
+ char buf[255] = "undefine process ";
+ char str[80];
+ char line[10];
+
+ strcpy(line, id);
+ strcpy(str, "id:");
+ strcat(str, line);
+ strcat(str, "\n\n");
+ strcat(buf, str);
+ printf("Request: %s\n", buf);
+
+ sr->performSend(buf,255,remotehost);
+ sr->syncPerformReceive(remotehost, *ss, 0);
+ ss->getPropertyObject();
+}
+
+void ClientInterface::listProcesses(const char * remotehost) {
+ char buf[255]="list processes\n\n";
+ printf("Request: %s\n", buf);
+ sr->performSend(buf,255,remotehost);
+ sr->syncPerformReceive(remotehost, *ss, 0);
+ ss->getPropertyObject();
+}
+
+void ClientInterface::showProcess(const char * remotehost, char * id) {
+ char buf[255] = "show process ";
+ char str[80];
+ char line[10];
+
+ strcpy(line, id);
+ strcpy(str, "id:");
+ strcat(str, line);
+ strcat(str, "\n\n");
+ strcat(buf, str);
+ printf("Request: %s\n", buf);
+
+ sr->performSend(buf,255,remotehost);
+ sr->syncPerformReceive(remotehost, *ss, 0);
+ ss->getPropertyObject();
+}
diff --git a/ndb/src/cw/util/ClientInterface.hpp b/ndb/src/cw/util/ClientInterface.hpp
new file mode 100644
index 00000000000..764705afacd
--- /dev/null
+++ b/ndb/src/cw/util/ClientInterface.hpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CLIENT_IF_HPP
+#define CLIENT_IF_HPP
+#include <Parser.hpp>
+#include <InputStream.hpp>
+#include <Parser.hpp>
+#include <NdbOut.hpp>
+#include <Properties.hpp>
+#include "SocketRegistry.hpp"
+#include "SocketService.hpp"
+#include "string.h"
+#include <stdio.h>
+#include <stdlib.h>
+
+class ClientInterface {
+private:
+ SocketService * ss;
+ SocketRegistry<SocketService> * sr;
+
+public:
+ ClientInterface(Uint32 maxNoOfCPC);
+ ~ClientInterface();
+ void startProcess(const char * remotehost, char * id);
+ void stopProcess(const char * remotehost, char * id);
+ void defineProcess(const char * remotehost, char * name, char * group,
+ char * env, char * path, char * args, char * type,
+ char * cwd, char * owner);
+ void undefineProcess(const char * remotehost, char * id);
+ void listProcesses(const char * remotehost);
+ void showProcess(const char * remotehost, char * id);
+ void connectCPCDdaemon(const char * remotehost, Uint16 port);
+ void disconnectCPCDdaemon(const char * remotehost);
+ void removeCPCDdaemon(const char * remotehost);
+
+};
+#endif
diff --git a/ndb/src/cw/util/Makefile b/ndb/src/cw/util/Makefile
new file mode 100644
index 00000000000..f5ab16721be
--- /dev/null
+++ b/ndb/src/cw/util/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+TYPE := ndbapi
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := socketclient
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = ClientInterface.cpp SocketService.cpp SocketClient.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/cw/util/SocketRegistry.cpp b/ndb/src/cw/util/SocketRegistry.cpp
new file mode 100644
index 00000000000..1dbb402f7c9
--- /dev/null
+++ b/ndb/src/cw/util/SocketRegistry.cpp
@@ -0,0 +1,213 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "SocketRegistry.hpp"
+#include <Parser.hpp>
+
+template<class T>
+SocketRegistry<T>::SocketRegistry(Uint32 maxSocketClients) {
+
+}
+
+
+template<class T>
+SocketRegistry<T>::~SocketRegistry() {
+ delete [] m_socketClients;
+}
+
+template<class T>
+bool
+SocketRegistry<T>::createSocketClient(const char * host, Uint16 port) {
+
+ if(port == 0)
+ return false;
+ if(host==NULL)
+ return false;
+
+ SocketClient * socketClient = new SocketClient(host, port);
+
+ if(socketClient->openSocket() < 0 || socketClient == NULL) {
+ ndbout << "could not connect" << endl;
+ delete socketClient;
+ return false;
+ }
+ else {
+ m_socketClients[m_nSocketClients] = socketClient;
+ m_nSocketClients++;
+ }
+ return true;
+}
+
+template<class T>
+int
+SocketRegistry<T>::pollSocketClients(Uint32 timeOutMillis) {
+
+
+
+ // Return directly if there are no TCP transporters configured
+ if (m_nSocketClients == 0){
+ tcpReadSelectReply = 0;
+ return 0;
+ }
+ struct timeval timeout;
+ timeout.tv_sec = timeOutMillis / 1000;
+ timeout.tv_usec = (timeOutMillis % 1000) * 1000;
+
+
+ NDB_SOCKET_TYPE maxSocketValue = 0;
+
+ // Needed for TCP/IP connections
+ // The read- and writeset are used by select
+
+ FD_ZERO(&tcpReadset);
+
+ // Prepare for sending and receiving
+ for (Uint32 i = 0; i < m_nSocketClients; i++) {
+ SocketClient * t = m_socketClients[i];
+
+ // If the socketclient is connected
+ if (t->isConnected()) {
+
+ const NDB_SOCKET_TYPE socket = t->getSocket();
+ // Find the highest socket value. It will be used by select
+ if (socket > maxSocketValue)
+ maxSocketValue = socket;
+
+ // Put the connected transporters in the socket read-set
+ FD_SET(socket, &tcpReadset);
+ }
+ }
+
+ // The highest socket value plus one
+ maxSocketValue++;
+
+ tcpReadSelectReply = select(maxSocketValue, &tcpReadset, 0, 0, &timeout);
+#ifdef NDB_WIN32
+ if(tcpReadSelectReply == SOCKET_ERROR)
+ {
+ NdbSleep_MilliSleep(timeOutMillis);
+ }
+#endif
+
+ return tcpReadSelectReply;
+
+}
+
+template<class T>
+bool
+SocketRegistry<T>::performSend(const char * buf,
+ Uint32 len,
+ const char * remotehost)
+{
+ SocketClient * socketClient;
+ for(Uint32 i=0; i < m_nSocketClients; i++) {
+ socketClient = m_socketClients[i];
+ if(strcmp(socketClient->gethostname(), remotehost)==0) {
+ if(socketClient->isConnected()) {
+ if(socketClient->writeSocket(buf, len)>0)
+ return true;
+ else
+ return false;
+ }
+ }
+ }
+ return false;
+}
+
+template<class T>
+int
+SocketRegistry<T>::performReceive(T & t) {
+ char buf[255] ; //temp. just for testing. must fix better
+
+ if(tcpReadSelectReply > 0){
+ for (Uint32 i=0; i<m_nSocketClients; i++) {
+ SocketClient *sc = m_socketClients[i];
+ const NDB_SOCKET_TYPE socket = sc->getSocket();
+ if(sc->isConnected() && FD_ISSET(socket, &tcpReadset)) {
+ t->runSession(socket,t);
+ }
+ }
+ return 1;
+ }
+ return 0;
+
+}
+
+
+
+template<class T>
+inline
+int
+SocketRegistry<T>::syncPerformReceive(char * host,
+ T & t,
+ Uint32 timeOutMillis) {
+ char buf[255] ; //temp. just for testing. must fix better
+ struct timeval timeout;
+ timeout.tv_sec = timeOutMillis / 1000;
+ timeout.tv_usec = (timeOutMillis % 1000) * 1000;
+ int reply;
+ SocketClient * sc;
+ for(Uint32 i=0; i < m_nSocketClients; i++) {
+ sc = m_socketClients[i];
+ if(strcmp(sc->gethostname(), remotehost)==0) {
+ if(sc->isConnected()) {
+ FD_ZERO(&tcpReadset);
+ reply = select(sc->getSocket(), &tcpReadset, 0, 0, &timeout);
+ if(reply > 0) {
+ return t->runSession(sc->getSocket(), t);
+ }
+ }
+
+ }
+ }
+ return 0;
+}
+
+
+
+template<class T>
+bool
+SocketRegistry<T>::reconnect(const char * host){
+ for(Uint32 i=0; i < m_nSocketClients; i++) {
+ SocketClient * socketClient = m_socketClients[i];
+ if(strcmp(socketClient->gethostname(), host)==0) {
+ if(!socketClient->isConnected()) {
+ if(socketClient->openSocket() > 0)
+ return true;
+ else return false;
+ }
+ }
+ }
+ return false;
+}
+
+template<class T>
+bool
+SocketRegistry<T>::removeSocketClient(const char * host){
+ for(Uint32 i=0; i < m_nSocketClients; i++) {
+ SocketClient * socketClient = m_socketClients[i];
+ if(strcmp(socketClient->gethostname(), host)==0) {
+ if(!socketClient->isConnected()) {
+ if(socketClient->closeSocket() > 0) {
+ delete socketClient;
+ return true;
+ }
+ else return false;
+ }
+ }
+ }
+ return false;
+}
diff --git a/ndb/src/cw/util/SocketRegistry.hpp b/ndb/src/cw/util/SocketRegistry.hpp
new file mode 100644
index 00000000000..2b079156967
--- /dev/null
+++ b/ndb/src/cw/util/SocketRegistry.hpp
@@ -0,0 +1,290 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SocketClientRegistry_H
+#define SocketClientRegistry_H
+
+#include <NdbTCP.h>
+#include <NdbOut.hpp>
+
+#include "SocketClient.hpp"
+
+template<class T>
+class SocketRegistry {
+
+public:
+ SocketRegistry(Uint32 maxSocketClients);
+ ~SocketRegistry();
+ /**
+ * creates and adds a SocketClient to m_socketClients[]
+ * @param host - host name
+ * @param port - port to connect to
+ */
+ bool createSocketClient(const char * host, const Uint16 port);
+
+ /**
+ * performReceive reads from sockets should do more stuff
+ */
+ int performReceive(T &);
+
+
+ /**
+ * performReceive reads from sockets should do more stuff
+ */
+ int syncPerformReceive(const char* ,T &, Uint32);
+
+
+ /**
+ * performSend sends a command to a host
+ */
+ bool performSend(const char * buf, Uint32 len, const char * remotehost);
+
+ /**
+ * pollSocketClients performs a select (for a max. of timeoutmillis) or
+ * until there is data to be read from any SocketClient
+ * @param timeOutMillis - select timeout
+ */
+ int pollSocketClients(Uint32 timeOutMillis);
+
+ /**
+ * reconnect tries to reconnect to a cpcd given its hostname
+ * @param host - name of host to reconnect to.
+ */
+ bool reconnect(const char * host);
+
+
+ /**
+ * removeSocketClient
+ * @param host - name of host for which to remove the SocketConnection
+ */
+ bool removeSocketClient(const char * host);
+
+private:
+ SocketClient** m_socketClients;
+ Uint32 m_maxSocketClients;
+ Uint32 m_nSocketClients;
+ int tcpReadSelectReply;
+ fd_set tcpReadset;
+
+
+};
+
+
+template<class T>
+inline
+SocketRegistry<T>::SocketRegistry(Uint32 maxSocketClients) {
+ m_maxSocketClients = maxSocketClients;
+ m_socketClients = new SocketClient * [m_maxSocketClients];
+ m_nSocketClients = 0;
+}
+
+
+template<class T>
+inline
+SocketRegistry<T>::~SocketRegistry() {
+ delete [] m_socketClients;
+}
+
+template<class T>
+inline
+bool
+SocketRegistry<T>::createSocketClient(const char * host, Uint16 port) {
+
+ if(port == 0)
+ return false;
+ if(host==NULL)
+ return false;
+
+ SocketClient * socketClient = new SocketClient(host, port);
+
+ if(socketClient->openSocket() < 0 || socketClient == NULL) {
+ ndbout << "could not connect" << endl;
+ delete socketClient;
+ return false;
+ }
+ else {
+ m_socketClients[m_nSocketClients] = socketClient;
+ m_nSocketClients++;
+ }
+ return true;
+}
+
+template<class T>
+inline
+int
+SocketRegistry<T>::pollSocketClients(Uint32 timeOutMillis) {
+
+
+
+ // Return directly if there are no TCP transporters configured
+ if (m_nSocketClients == 0){
+ tcpReadSelectReply = 0;
+ return 0;
+ }
+ struct timeval timeout;
+ timeout.tv_sec = timeOutMillis / 1000;
+ timeout.tv_usec = (timeOutMillis % 1000) * 1000;
+
+
+ NDB_SOCKET_TYPE maxSocketValue = 0;
+
+ // Needed for TCP/IP connections
+ // The read- and writeset are used by select
+
+ FD_ZERO(&tcpReadset);
+
+ // Prepare for sending and receiving
+ for (Uint32 i = 0; i < m_nSocketClients; i++) {
+ SocketClient * t = m_socketClients[i];
+
+ // If the socketclient is connected
+ if (t->isConnected()) {
+
+ const NDB_SOCKET_TYPE socket = t->getSocket();
+ // Find the highest socket value. It will be used by select
+ if (socket > maxSocketValue)
+ maxSocketValue = socket;
+
+ // Put the connected transporters in the socket read-set
+ FD_SET(socket, &tcpReadset);
+ }
+ }
+
+ // The highest socket value plus one
+ maxSocketValue++;
+
+ tcpReadSelectReply = select(maxSocketValue, &tcpReadset, 0, 0, &timeout);
+#ifdef NDB_WIN32
+ if(tcpReadSelectReply == SOCKET_ERROR)
+ {
+ NdbSleep_MilliSleep(timeOutMillis);
+ }
+#endif
+
+ return tcpReadSelectReply;
+
+}
+
+template<class T>
+inline
+bool
+SocketRegistry<T>::performSend(const char * buf, Uint32 len, const char * remotehost)
+{
+ SocketClient * socketClient;
+ for(Uint32 i=0; i < m_nSocketClients; i++) {
+ socketClient = m_socketClients[i];
+ if(strcmp(socketClient->gethostname(), remotehost)==0) {
+ if(socketClient->isConnected()) {
+ if(socketClient->writeSocket(buf, len)>0)
+ return true;
+ else
+ return false;
+ }
+ }
+ }
+ return false;
+}
+
+template<class T>
+inline
+int
+SocketRegistry<T>::performReceive(T & t) {
+ char buf[255] ; //temp. just for testing. must fix better
+
+ if(tcpReadSelectReply > 0){
+ for (Uint32 i=0; i<m_nSocketClients; i++) {
+ SocketClient *sc = m_socketClients[i];
+ const NDB_SOCKET_TYPE socket = sc->getSocket();
+ if(sc->isConnected() && FD_ISSET(socket, &tcpReadset)) {
+ t->runSession(socket,t);
+ }
+ }
+ return 1;
+ }
+ return 0;
+
+}
+
+
+
+template<class T>
+inline
+int
+SocketRegistry<T>::syncPerformReceive(const char * remotehost,
+ T & t,
+ Uint32 timeOutMillis) {
+ char buf[255] ; //temp. just for testing. must fix better
+ struct timeval timeout;
+ timeout.tv_sec = timeOutMillis / 1000;
+ timeout.tv_usec = (timeOutMillis % 1000) * 1000;
+ int reply;
+ SocketClient * sc;
+ for(Uint32 i=0; i < m_nSocketClients; i++) {
+ sc = m_socketClients[i];
+ if(strcmp(sc->gethostname(), remotehost)==0) {
+ if(sc->isConnected()) {
+ /*FD_ZERO(&tcpReadset);
+ reply = select(sc->getSocket()+1, 0, 0, 0, &timeout);
+ reply=1;
+ if(reply > 0) {*/
+ t.runSession(sc->getSocket(), t);
+ //}
+ }
+
+ }
+ }
+}
+
+
+
+template<class T>
+inline
+bool
+SocketRegistry<T>::reconnect(const char * host){
+ for(Uint32 i=0; i < m_nSocketClients; i++) {
+ SocketClient * socketClient = m_socketClients[i];
+ if(strcmp(socketClient->gethostname(), host)==0) {
+ if(!socketClient->isConnected()) {
+ if(socketClient->openSocket() > 0)
+ return true;
+ else return false;
+ }
+ }
+ }
+ return false;
+}
+
+template<class T>
+inline
+bool
+SocketRegistry<T>::removeSocketClient(const char * host){
+ for(Uint32 i=0; i < m_nSocketClients; i++) {
+ SocketClient * socketClient = m_socketClients[i];
+ if(strcmp(socketClient->gethostname(), host)==0) {
+ if(!socketClient->isConnected()) {
+ if(socketClient->closeSocket() > 0) {
+ delete socketClient;
+ return true;
+ }
+ else return false;
+ }
+ }
+ }
+ return false;
+}
+
+
+#endif // Define of SocketRegistry
diff --git a/ndb/src/cw/util/SocketService.cpp b/ndb/src/cw/util/SocketService.cpp
new file mode 100644
index 00000000000..b993ec8c2c1
--- /dev/null
+++ b/ndb/src/cw/util/SocketService.cpp
@@ -0,0 +1,60 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <Parser.hpp>
+#include <NdbOut.hpp>
+#include <Properties.hpp>
+#include "SocketService.hpp"
+
+SocketService::SocketService() {
+
+}
+
+SocketService::~SocketService() {
+
+}
+
+int
+SocketService::runSession(NDB_SOCKET_TYPE socket, SocketService & ss){
+ InputStream *m_input = new SocketInputStream(socket);
+ char buf[255];
+
+ m_input->gets(buf,255);
+ ndbout_c("SocketService:received: %s\n", buf);
+ ndbout_c("This should now be parsed\n");
+ ndbout_c("and put in a property object.\n");
+ ndbout_c("The propery is then accessible from the ClientInterface.\n");
+ ndbout_c("by getPropertyObject.\n");
+ ndbout_c("At least this is the idea.");
+ /*Parser_t *m_parser =
+ new Parser<SocketService>(commands, *m_input, true, true, true);
+ */
+ /** to do
+ * add a proprty object to which the parser will put its result.
+ */
+
+ return 1 ; //succesful
+ //return 0; //unsuccesful
+
+}
+
+void
+SocketService::getPropertyObject() {
+ ndbout << "get property object. return to front end or something" << endl;
+}
+
+
diff --git a/ndb/src/cw/util/SocketService.hpp b/ndb/src/cw/util/SocketService.hpp
new file mode 100644
index 00000000000..7a0c3a2fd91
--- /dev/null
+++ b/ndb/src/cw/util/SocketService.hpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SOCKET_SERVICE_HPP
+#define SOCKET_SERVICE_HPP
+#include <Parser.hpp>
+#include <InputStream.hpp>
+#include <Parser.hpp>
+#include <NdbOut.hpp>
+#include <Properties.hpp>
+#include "SocketRegistry.hpp"
+
+
+
+
+class SocketService {
+ friend class SocketRegistry<SocketService>;
+private:
+ typedef Parser<SocketService> Parser_t;
+ int runSession(NDB_SOCKET_TYPE socket, SocketService &);
+public:
+ void getPropertyObject();
+ SocketService();
+ ~SocketService();
+
+};
+
+
+
+
+
+
+#endif
diff --git a/ndb/src/external/LINUX.x86/sci/include/list.h b/ndb/src/external/LINUX.x86/sci/include/list.h
new file mode 100644
index 00000000000..81c467a461b
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/list.h
@@ -0,0 +1,56 @@
+/* $Id: list.h,v 1.1 2002/12/13 12:17:20 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+
+#ifndef _LIST_H
+#define _LIST_H
+#include "sci_types.h"
+
+
+typedef struct ListElement *ListElement_t;
+typedef struct List *List_t;
+
+struct ListElement {
+ void *element;
+ u_vkaddr_t key;
+ ListElement_t prev,next;
+};
+
+void *Get_Element(ListElement_t el);
+void Set_Element(ListElement_t el,void *elptr,u_vkaddr_t key);
+void Create_Element(ListElement_t *el);
+void Destroy_Element(ListElement_t *el);
+void Create_List(List_t *list);
+void Destroy_List(List_t *list);
+void Add_Element(List_t list,ListElement_t el);
+void Remove_Element(List_t list,ListElement_t el);
+ListElement_t Find_Element(List_t list,u_vkaddr_t key);
+scibool List_Empty(List_t);
+scibool List_Elements(List_t);
+ListElement_t First_Element(List_t list);
+ListElement_t Last_Element(List_t list);
+ListElement_t Next_Element(ListElement_t el);
+
+#endif /* _LIST_H */
diff --git a/ndb/src/external/LINUX.x86/sci/include/os/inttypes.h b/ndb/src/external/LINUX.x86/sci/include/os/inttypes.h
new file mode 100644
index 00000000000..b9e5a6cb19f
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/os/inttypes.h
@@ -0,0 +1,53 @@
+/* $Id: inttypes.h,v 1.1 2002/12/13 12:17:21 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+#ifndef _SCI_OS_INTTYPES_H_
+#define _SCI_OS_INTTYPES_H_
+
+/*
+ * --------------------------------------------------------------------------------------
+ * Basic types of various sizes.
+ * --------------------------------------------------------------------------------------
+ */
+typedef unsigned char unsigned8;
+typedef unsigned short unsigned16;
+typedef unsigned int unsigned32;
+typedef unsigned long long unsigned64;
+
+typedef signed char signed8;
+typedef signed short signed16;
+typedef signed int signed32;
+typedef signed long long signed64;
+
+
+#ifdef CPU_WORD_IS_64_BIT
+typedef unsigned64 uptr_t;
+typedef signed64 iptr_t;
+#else
+typedef unsigned32 uptr_t;
+typedef signed32 iptr_t;
+#endif
+
+#endif /* _SCI_OS_INTTYPES_H_ */
diff --git a/ndb/src/external/LINUX.x86/sci/include/rmlib.h b/ndb/src/external/LINUX.x86/sci/include/rmlib.h
new file mode 100644
index 00000000000..9d2722e9798
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/rmlib.h
@@ -0,0 +1,212 @@
+/* $Id: rmlib.h,v 1.1 2002/12/13 12:17:20 hin Exp $ */
+
+/*********************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU Lesser General Public License as published by *
+ * the Free Software Foundation; either version 2.1 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 Lesser 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. *
+ * *
+ * *
+ *********************************************************************************/
+
+/********************************************************************************/
+/* This header file contains the declarations of the SCI Reflective Memory */
+/* library rmlib. The implementation of the library functions is in rmlib.c. */
+/* The library contains all the functions that operate on the reflective */
+/* memory. */
+/* */
+/* NB! */
+/* */
+/* DOLPHIN'S SCI REFLECTIVE MEMORY FILES ARE UNDER DEVELOPMENT AND MAY CHANGE. */
+/* PLEASE CONTACT DOLPHIN FOR FURTHER INFORMATION. */
+/* */
+/* */
+/********************************************************************************/
+
+#include "sisci_error.h"
+#include "sisci_api.h"
+#include "sisci_demolib.h"
+#include "sisci_types.h"
+
+unsigned int seqerr, syncseqerr;
+
+#ifndef _RMLIB_H
+#define _RMLIB_H
+
+
+#if defined(_REENTRANT)
+
+#define _RMLIB_EXPAND_NAME(name) _RMLIB_MT_ ## name
+
+#else
+
+#define _RMLIB_EXPAND_NAME(name) _RMLIB_ST_ ## name
+
+#endif
+
+#ifdef __sparc
+#define CACHE_SIZE 2097152
+#else
+#define CACHE_SIZE 8192
+#endif
+
+/*********************************************************************************/
+/* FLAG VALUES */
+/*********************************************************************************/
+
+#define REFLECT_ERRCHECK 0x2
+
+struct ReflectiveMemorySpace {
+ unsigned int localAdapterNo;
+ unsigned int localNodeId;
+ unsigned int remoteNodeId;
+ sci_desc_t sd;
+ sci_desc_t syncsd;
+ sci_map_t localMap;
+ sci_map_t remoteMap;
+ unsigned int localSegmentId;
+ unsigned int remoteSegmentId;
+ unsigned int syncSegmentId;
+ unsigned int sync_rSegmentId;
+ unsigned int segmentSize;
+ unsigned int *localMapAddr;
+ volatile unsigned int *remoteMapAddr;
+ sci_local_segment_t localSegment;
+ sci_remote_segment_t remoteSegment;
+ sci_local_segment_t syncSegment;
+ sci_remote_segment_t sync_rSegment;
+ sci_map_t syncMap;
+ sci_map_t sync_rMap;
+ sci_sequence_t syncsequence;
+ sci_sequence_t sequence;
+ unsigned int protection;
+ unsigned int retry_value;
+ sci_sequence_status_t sequenceStatus, syncsequenceStatus;
+ volatile unsigned int *syncMapAddr;
+ volatile unsigned int *sync_rMapAddr;
+};
+
+/*********************************************************************************/
+/* P R I N T R E F L E C T I V E M E M O R Y S P A C E */
+/* */
+/*********************************************************************************/
+#define ReflectPrintParameters _RMLIB_EXPAND_NAME(ReflectPrintParameters)
+void ReflectPrintParameters(FILE *stream, struct ReflectiveMemorySpace RM_space);
+
+/*********************************************************************************/
+/* R E F L E C T D M A S E T U P */
+/* */
+/*********************************************************************************/
+#define ReflectDmaSetup _RMLIB_EXPAND_NAME(ReflectDmaSetup)
+sci_error_t ReflectDmaSetup(struct ReflectiveMemorySpace RM_space, sci_dma_queue_t *dmaQueue);
+
+/*********************************************************************************/
+/* R E F L E C T D M A R E M O V E */
+/* */
+/*********************************************************************************/
+#define ReflectDmaRemove _RMLIB_EXPAND_NAME(ReflectDmaRemove)
+sci_error_t ReflectDmaRemove(sci_dma_queue_t dmaQueue);
+
+/*********************************************************************************/
+/* R E F L E C T D M A R U N */
+/* */
+/*********************************************************************************/
+#define ReflectDmaRun _RMLIB_EXPAND_NAME(ReflectDmaRun)
+sci_error_t ReflectDmaRun(struct ReflectiveMemorySpace RM_space,
+ unsigned int* privateSrc,
+ unsigned int size,
+ unsigned int offset,
+ sci_dma_queue_t dmaQueue);
+/*********************************************************************************/
+/* C L O S E R E F L E C T I V E M E M O R Y S P A C E */
+/* */
+/*********************************************************************************/
+#define ReflectClose _RMLIB_EXPAND_NAME(ReflectClose)
+sci_error_t ReflectClose(struct ReflectiveMemorySpace RM_space, unsigned int segment_no);
+
+/*********************************************************************************/
+/* O P E N R E F L E C T I V E M E M O R Y S P A C E */
+/* */
+/*********************************************************************************/
+#define ReflectOpen _RMLIB_EXPAND_NAME(ReflectOpen)
+sci_error_t ReflectOpen(struct ReflectiveMemorySpace *RM_space,
+ unsigned int size,
+ unsigned int segment_no,
+ unsigned int localAdapterNo,
+ unsigned int remoteNodeId,
+ unsigned int protection,
+ unsigned int retry_value);
+
+/*********************************************************************************/
+/* R E F L E C T G E T A C C E S S */
+/* */
+/*********************************************************************************/
+#define ReflectGetAccess _RMLIB_EXPAND_NAME(ReflectGetAccess)
+sci_error_t ReflectGetAccess(struct ReflectiveMemorySpace *RM_space);
+
+/*********************************************************************************/
+/* R E F L E C T R E L E A S E A C C E S S */
+/* */
+/*********************************************************************************/
+#define ReflectReleaseAccess _RMLIB_EXPAND_NAME(ReflectReleaseAccess)
+sci_error_t ReflectReleaseAccess(struct ReflectiveMemorySpace *RM_space);
+
+/*********************************************************************************/
+/* R E F L E C T D M A */
+/* */
+/*********************************************************************************/
+#define ReflectDma _RMLIB_EXPAND_NAME(ReflectDma)
+sci_error_t ReflectDma(struct ReflectiveMemorySpace RM_space,
+ unsigned int* privateSrc,
+ unsigned int size,
+ unsigned int offset);
+
+/*********************************************************************************/
+/* R E F L E C T M E M C O P Y */
+/* */
+/*********************************************************************************/
+#define ReflectMemCopy _RMLIB_EXPAND_NAME(ReflectMemCopy)
+sci_error_t ReflectMemCopy(struct ReflectiveMemorySpace RM_space,
+ unsigned int* privateSrc,
+ unsigned int size,
+ unsigned int offset,
+ unsigned int flags);
+
+/*********************************************************************************/
+/* R E F L E C T S E T */
+/* */
+/*********************************************************************************/
+#define ReflectSet _RMLIB_EXPAND_NAME(ReflectSet)
+sci_error_t ReflectSet(struct ReflectiveMemorySpace RM_space,
+ unsigned int value,
+ unsigned int size,
+ unsigned int offset,
+ unsigned int flags
+ );
+
+/*********************************************************************************/
+/* R E F L E C T P R I N T */
+/* */
+/*********************************************************************************/
+#define ReflectPrint _RMLIB_EXPAND_NAME(ReflectPrint)
+sci_error_t ReflectPrint(FILE *stream,
+ struct ReflectiveMemorySpace RM_space,
+ unsigned int size,
+ unsigned int offset
+ );
+
+
+#endif
diff --git a/ndb/src/external/LINUX.x86/sci/include/sci_errno.h b/ndb/src/external/LINUX.x86/sci/include/sci_errno.h
new file mode 100644
index 00000000000..03f3256a86f
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/sci_errno.h
@@ -0,0 +1,216 @@
+/* $Id: sci_errno.h,v 1.1 2002/12/13 12:17:20 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+
+#ifndef _SCI_ERRNO_H_
+#define _SCI_ERRNO_H_
+
+
+/*
+ * SCI Error return values always have 30 bit set
+ * Remote errors should have bit 0 set
+ */
+#define SCI_ERR_MASK 0x40000000
+#define ESCI_REMOTE_MASK 0x01000000
+
+#define SCI_ERR(u) ((unsigned32)(u)&0x7FFFFFFF )
+#define _SCI_ERROR(x) ((x) | SCI_ERR_MASK)
+#define _SCI_REMOTE_ERROR(x) ( _SCI_ERROR(x) | ESCI_REMOTE_MASK )
+
+/*
+ * Error codes
+ */
+typedef enum {
+ ESCI_OK = 0x000,
+ ESCI_STILL_EXPORTED = _SCI_ERROR(0x800),
+
+ ESCI_BUS_ERR = _SCI_ERROR(0x900),
+ ESCI_PEND_SCIERR = _SCI_ERROR(0x901),
+ ESCI_SCI_ERR = _SCI_ERROR(0x902),
+
+ /*
+ * Specific SCI error responses:
+ */
+ ESCI_SCI_ERR_DATA = _SCI_ERROR(0x9021),
+ ESCI_SCI_ERR_TYPE = _SCI_ERROR(0x9022),
+ ESCI_SCI_ERR_ADDR = _SCI_ERROR(0x9023),
+
+ ESCI_LINK_TIMEOUT = _SCI_ERROR(0x903),
+ ESCI_EXDEV_TIMEOUT = _SCI_ERROR(0x904),
+ ESCI_REMOTE_ERR = _SCI_ERROR(0x905),
+ ESCI_MBX_BUSY = _SCI_ERROR(0x906),
+ ESCI_DMAERR = _SCI_ERROR(0x907),
+ ESCI_DMA_DISABLED = _SCI_ERROR(0x908),
+ ESCI_SW_MBX_SEND_FAILED = _SCI_ERROR(0x909),
+ ESCI_HW_MBX_SEND_FAILED = _SCI_ERROR(0x90A),
+ ESCI_HAS_NO_SESSION = _SCI_ERROR(0xA00),
+ ESCI_CONNREFUSED_SESSION = _SCI_ERROR(0xA01),
+ ESCI_SESSION_NOT_ESTABLISHED = _SCI_ERROR(0xA11),
+ ESCI_REMOTE_NO_VALID_SESSION = _SCI_ERROR(0xA02),
+ ESCI_SESSION_DISABLED = _SCI_ERROR(0xA03),
+ ESCI_NODE_CLOSED = _SCI_ERROR(0xA04),
+ ESCI_NODE_DISABLED = _SCI_ERROR(0xA05),
+
+ ESCI_LOCAL_MASTER_ERR = _SCI_ERROR(0xA06),
+ ESCI_REMOTE_MASTER_ERR = _SCI_REMOTE_ERROR(0xA06),
+
+ ESCI_ILLEGAL_CMD_RECEIVED = _SCI_ERROR(0xA08),
+ ESCI_ILLEGAL_CMD_SENT = _SCI_ERROR(0xA09),
+
+ /* used above: ESCI_SESSION_NOT_ESTABLISHED = _SCI_ERROR(0xA11), */
+
+ /*
+ * Remote error codes
+ */
+ ESCI_CONNREFUSED = _SCI_ERROR(0xB00),
+ ESCI_NODE_NOT_RESPONDING = _SCI_ERROR(0xB01),
+ ESCI_ISCONN = _SCI_ERROR(0xB02),
+ ESCI_HOSTUNREACH = _SCI_ERROR(0xB03),
+ ESCI_NO_SUCH_USER_ID = _SCI_ERROR(0xB04),
+ ESCI_REMOTE_NO_SUCH_USER_ID = _SCI_REMOTE_ERROR(0xB04), /* ESCI_NO_SUCH_USER_ID */
+ ESCI_NO_SUCH_KEY = _SCI_ERROR(0xB04), /* ESCI_NO_SUCH_USER_ID */
+ ESCI_REMOTE_NO_SUCH_KEY = _SCI_REMOTE_ERROR(0xB04), /* ESCI_REMOTE_NO_SUCH_USER_ID */
+ ESCI_NODE_ERR = _SCI_ERROR(0xB06),
+ ESCI_REMOTE_NODE_ERR = _SCI_REMOTE_ERROR(0xB06), /* ESCI_NODE_ERR */
+ ESCI_NOSPC = _SCI_ERROR(0xB08),
+ ESCI_REMOTE_NOSPC = _SCI_REMOTE_ERROR(0xB08), /* ESCI_NOSPC */
+ ESCI_NODMASPC = _SCI_ERROR(0xB0A),
+ ESCI_REMOTE_NODMASPC = _SCI_REMOTE_ERROR(0xB0A), /* ESCI_NODMASPC */
+ ESCI_NOTMAP = _SCI_ERROR(0xC00),
+ ESCI_ISMAP = _SCI_ERROR(0xC01),
+ ESCI_NOT_INITIALIZED = _SCI_ERROR(0xD00),
+ ESCI_REMOTE_NOT_INITIALIZED = _SCI_REMOTE_ERROR(ESCI_NOT_INITIALIZED),
+ /*
+ * ???
+ */
+ ESCI_PARAM_ERR = _SCI_ERROR(0xD01),
+ ESCI_NO_FREE_VC = _SCI_ERROR(0xD02),
+ ESCI_REMOTE_NO_FREE_VC = _SCI_REMOTE_ERROR(0xD02), /* ESCI_NO_FREE_VC */
+
+ /*
+ * Adapter state related error codes:
+ */
+ ESCI_SUSPENDED = _SCI_ERROR(0xD03),
+ ESCI_NOT_SUSPENDED = _SCI_ERROR(0xD04),
+ ESCI_NOT_READY = _SCI_ERROR(0xD05),
+ ESCI_NOT_CONFIGURED = _SCI_ERROR(0xD06),
+ ESCI_INVALID_ADAPTERID = _SCI_ERROR(0xD07), /* if an adapter-id is out of range */
+ ESCI_NONEXIST_ADAPTERID = _SCI_ERROR(0xD08), /* if adapter-id is valid but no adapter matches */
+ ESCI_ADAPTERID_INUSE = _SCI_ERROR(0xD09),
+
+ ESCI_INVALID_INSTANCE = _SCI_ERROR(0xD0A),
+ ESCI_NONEXIST_INSTANCE = _SCI_ERROR(0xD0B),
+
+ ESCI_ADAPTER_INIT_FAILURE = _SCI_ERROR(0xD0C),
+
+ ESCI_PAUSED = _SCI_ERROR(0xD0D),
+ ESCI_NOT_PAUSED = _SCI_ERROR(0xD0E),
+ ESCI_ADAPTER_NEED_RESET = _SCI_ERROR(0xD0F),
+
+ ESCI_NONEXIST_SERIAL_NUMBER = _SCI_ERROR(0xD10),
+ ESCI_NOT_AVAILABLE = _SCI_ERROR(0xD11),
+
+ ESCI_EACCESS = _SCI_ERROR(0xD12),
+
+ /*
+ * Local error codes
+ */
+ ESCI_NO_LOCAL_ACCESS = _SCI_ERROR(0xE00),
+ ESCI_LRESOURCE_BUSY = _SCI_ERROR(0xE01),
+ ESCI_LRESOURCE_EXIST = _SCI_ERROR(0xE02),
+ ESCI_NO_LRESOURCE = _SCI_ERROR(0xE03),
+ ESCI_NOTCONN = _SCI_ERROR(0xE04),
+ ESCI_LOCAL_ERR = _SCI_ERROR(0xE05),
+ ESCI_NOVAL_NODEID = _SCI_ERROR(0xE06),
+ ESCI_NOT_SUPPORTED = _SCI_ERROR(0xE07),
+ ESCI_TIMEOUT = _SCI_ERROR(0xE08),
+ ESCI_NO_LOCAL_LC_ACCESS = _SCI_ERROR(0xE0A),
+ ESCI_INVALID_ATT = _SCI_ERROR(0xE0B),
+ ESCI_BAD_CHECKSUM = _SCI_ERROR(0xE0C),
+ ESCI_INTERRUPT_FLAG_DISABLED = _SCI_ERROR(0xE0D),
+ ESCI_COND_INT_RACE_PROBLEM = _SCI_ERROR(0xE0E),
+ ESCI_OVERFLOW = _SCI_ERROR(0xE0F),
+ ESCI_BLINK_PARITY_ERROR = _SCI_ERROR(0xE10),
+ ESCI_FIRMWARE_VERSION_MISMATCH = _SCI_ERROR(0xE11),
+
+ /*
+ * Link error codes
+ */
+ ESCI_NO_LINK_ACCESS = _SCI_ERROR(0xF00),
+ ESCI_NO_REMOTE_LINK_ACCESS = _SCI_REMOTE_ERROR(0xF00), /* ESCI_NO_LINK_ACCESS */
+
+ ESCI_NO_SUCH_NODE = _SCI_ERROR(0xF02),
+ ESCI_USR_ACCESS_DISABLED = _SCI_ERROR(0xF03),
+ ESCI_HW_AVOID_DEADLOCK = _SCI_ERROR(0xF04),
+ ESCI_POTENTIAL_ERROR = _SCI_ERROR(0xF05),
+
+ ESCI_FENCED = _SCI_ERROR(0xF06),
+ ESCI_SWITCH_HW_FAILURE = _SCI_ERROR(0xF07),
+ ESCI_SWITCH_WRONG_BLINK_ID = _SCI_ERROR(0xF08),
+ ESCI_SWITCH_WRONG_PORT_NUMB = _SCI_ERROR(0xF09),
+ ESCI_SWITCH_WRONG_INIT_TYPE = _SCI_ERROR(0xF0A), /* It is determined that the swith initialization
+ * do not match the local adapter initialization
+ */
+ ESCI_SWITCH_WRONG_SWITCH_NUMB = _SCI_ERROR(0xF0B), /* It is determined that we are operationg on the
+ * wrong switch port
+ */
+ ESCI_SWITCH_NOT_CONNECTED = _SCI_ERROR(0xF0C),
+ ESCI_SWITCH_NOT_RECOGNIZED = _SCI_ERROR(0xF0D),
+ ESCI_SWITCH_INIT_IN_PROGRESS = _SCI_ERROR(0xF0E), /* Switch TINI initialization in progress */
+
+
+ ESCI_NO_BACKBONE_LINK_ACCESS = _SCI_ERROR(0xF20),
+ ESCI_BACKBONE_FENCED = _SCI_ERROR(0xF21),
+ ESCI_NO_BACKBONE_ACCESS = _SCI_ERROR(0xF22),
+ ESCI_BACKBONE_CABLE_PROBLEM = _SCI_ERROR(0xF23),
+ ESCI_BACKBONE_BLINK_PROBLEM = _SCI_ERROR(0xF24),
+ ESCI_BACKBONE_HWINIT_PROBLEM = _SCI_ERROR(0xF25),
+ ESCI_BACKBONE_ID_PROBLEM = _SCI_ERROR(0xF26),
+ ESCI_BACKBONE_STATE_PROBLEM = _SCI_ERROR(0xF27),
+ ESCI_BACKBONE_REQ_LINK_PROBLEM = _SCI_ERROR(0xF28),
+ ESCI_BACKBONE_UNFENCING = _SCI_ERROR(0xF29), /* Unfencing in progress */
+
+ /*
+ * added for pci port
+ */
+ ESCI_AGAIN = _SCI_ERROR(0xF15),
+ ESCI_ORANGE = _SCI_ERROR(0xF16), /* Out of range */
+ ESCI_NOSYS = _SCI_ERROR(0xF17), /* Used instead of ENOSYS. Means function not implemented */
+ ESCI_REMOTE_NOSYS = _SCI_REMOTE_ERROR(ESCI_NOSYS),
+ ESCI_INTR = _SCI_ERROR(0xF18), /* Used instead of EINTR from sys/errno.h */
+ ESCI_IO = _SCI_ERROR(0xF19), /* Used instead of EIO from sys/errno.h */
+ ESCI_FAULT = _SCI_ERROR(0xF1A), /* Used instead of EFAULT from sys/errno.h */
+ ESCI_BUSY = _SCI_ERROR(0xF1B), /* Used instead of EBUST from sys/errno.h */
+ ESCI_INVAL = _SCI_ERROR(0xF1C), /* Used instead of EINVAL from sys/errno.h */
+ ESCI_NXIO = _SCI_ERROR(0xF1D), /* Used instead of ENXIO from sys/errno.h */
+ ESCI_EXIST = _SCI_ERROR(0xF1E) /* Used instead of EEXIST from sys/errno.h */
+
+} scierror_t;
+
+#endif /* _SCI_ERRNO_H_ */
+
+
+
+
diff --git a/ndb/src/external/LINUX.x86/sci/include/sci_types.h b/ndb/src/external/LINUX.x86/sci/include/sci_types.h
new file mode 100644
index 00000000000..740b3a45cfd
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/sci_types.h
@@ -0,0 +1,300 @@
+/* $Id: sci_types.h,v 1.1 2002/12/13 12:17:21 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+#ifndef _SCI_TYPES_H_
+#define _SCI_TYPES_H_
+
+/*
+ * Remains for the time being for backward compatibility ....
+ */
+
+/* #define UNIQUE(type) struct { type x; } * */
+#ifndef UNIQUE
+#define UNIQUE(type) type
+#endif
+#include "os/inttypes.h"
+
+#if defined(WIN32)
+#if defined(_KERNEL)
+#include <ntddk.h>
+#else
+#include <WTYPES.H>
+#endif /* _KERNEL */
+#else
+#if defined(Linux)
+#if defined(__KERNEL__)
+#include <linux/types.h>
+#else
+#include <sys/types.h>
+#endif
+#else
+#include <sys/types.h>
+#endif
+#ifdef SUNOS5
+#include <sys/ddi.h>
+#include <sys/sunddi.h>
+#endif
+#ifdef OS_IS_TRU64
+#include <io/common/devdriver.h>
+#endif
+#ifdef OS_IS_HP_UX11
+#if defined(_KERNEL)
+#include <../wsio/wsio.h>
+#else
+#include <sys/wsio.h>
+#endif
+#endif
+#endif
+
+/* See comments about "UNCONFIGURED_ADAPTERS" in config.h */
+#define UNCONFIGURED_ADAPTERS 100
+
+#ifndef FALSE
+#define FALSE 0
+#endif
+
+#ifndef TRUE
+#define TRUE 1
+#endif
+
+#ifndef NULL
+#define NULL 0
+#endif
+
+#ifndef IN
+#define IN
+#endif
+
+#ifndef NOT
+#define NOT !
+#endif
+
+/*
+ * --------------------------------------------------------------------------------------
+ * Basic types of various sizes.
+ * --------------------------------------------------------------------------------------
+ */
+
+typedef signed32 scibool;
+#ifndef OS_IS_VXWORKS
+typedef signed32 BOOL;
+#else
+/* VXWORKS has already defined BOOL */
+#endif
+typedef unsigned32 node_t; /* This is the logical nodeid */
+typedef unsigned32 sciNodeId_t; /* This is the physical 16 bit SCI nodeid */
+
+/*
+ * --------------------------------------------------------------------------------------
+ * Various register types.
+ * --------------------------------------------------------------------------------------
+ */
+typedef volatile unsigned32 register32;
+
+
+/*
+Temporary for Windows NT, until we use only the above types.
+*/
+
+#ifdef WIN32
+
+typedef unsigned char u_char;
+typedef unsigned short u_short;
+typedef unsigned long u_long;
+typedef unsigned int u_int;
+typedef char * caddr_t;
+
+typedef long off_t;
+typedef unsigned int size_t;
+
+#endif
+#ifdef OS_IS_VXWORKS
+#include <vxWorks.h>
+#endif
+
+/*
+ * --------------------------------------------------------------------------------------
+ * Various address types.
+ *
+ * We are using a struct * instead of unsigned long (int) inorder to enforce strong
+ * type checking
+ *
+ * --------------------------------------------------------------------------------------
+ */
+typedef UNIQUE(void *) vkaddr_t; /* Virtual kernel address */
+typedef UNIQUE(uptr_t) vuaddr_t; /* Virtual user address */
+
+typedef UNIQUE(unsigned32) remaddr_t; /* Remote IO address (physical address on PCs) */
+typedef UNIQUE(unsigned32) sciofs_lo_t; /* Lower 32 bits of an SCI offset. */
+typedef UNIQUE(unsigned32) sciofs_hi_t; /* The upper 16 bits of an SCI offset. */
+
+typedef UNIQUE(unsigned32) ioaddr_t; /* Local IO address (physical address on PCs) */
+typedef unsigned32 u_ioaddr_t;
+typedef unsigned32 iooffset_t;
+typedef unsigned32 iosize_t;
+
+typedef uptr_t vkoffset_t;
+typedef uptr_t u_vkaddr_t;
+typedef uptr_t u_vuaddr_t;
+typedef unsigned32 u_sciofs_lo_t;
+typedef unsigned32 u_sciofs_hi_t;
+typedef unsigned32 u_remaddr_t;
+typedef unsigned32 attOffset_t; /* Address displacement from start of ATT entry */
+
+typedef unsigned32 adapterNo_t;
+
+typedef enum {
+ NO_NODE = 0,
+ AD_MEM_NODE = 1,
+ AD_ALT_NODE = 2,
+ AD_MBX_NODE = 3,
+ AD_LC_NODE = 4,
+ AD_LC_PORT_0 = 5,
+ AD_LC_PORT_1 = 6,
+ AD_LC_PORT_2 = 7,
+ PHYS_NODE = 8
+} node_type_t;
+
+
+/*
+ * Currently we don't support more than 32 bit sizes.
+ */
+#define SIZEOF(x) ((unsigned32)sizeof(x))
+
+#if defined(_KERNEL)
+
+/*
+ * --------------------------------------------------------------------------------------
+ * Some small macros intended to ease the transition to more strongly typed address
+ * types. The intention is that they in the long run shall be removed ...
+ * --------------------------------------------------------------------------------------
+ */
+#define P2SIZE_T(x) ((size_t)((uptr_t)(x))) /* Pointer to size_t */
+#define P2U32(x) ((unsigned32)((uptr_t)(x))) /* Pointer to Unsigned 32-bit int */
+#ifdef WIN32
+#define PHADDR(x) ((ioaddr_t)(x))
+#define HASV(x) (x)
+#endif
+#if 0
+static vkaddr_t VKPTR (void * ptr) { return (vkaddr_t)ptr; }
+static vkaddr_t VKADDR(volatile void * ptr) { return (vkaddr_t)ptr; }
+#else
+#define VKPTR(ptr) (vkaddr_t)ptr
+#define VKADDR(ptr) (vkaddr_t)ptr
+#endif
+
+#ifdef KLOG
+#define KLOG_LOG(n,m,v) ts_log((n),(m),(v))
+#else
+#define KLOG_LOG(n,m,v)
+#endif /* KLOG */
+
+
+/*
+ * --------------------------------------------------------------------------------------
+ *
+ * M E M A R E A T
+ *
+ * Memory area descriptor.
+ *
+ * paddr -- Physical address (aligned) of memory area
+ * ual_vaddr -- (Kernel )Virtual address of the unaligned memory area
+ * vaddr -- (Kernel) Virtual address of memory area
+ * rsize -- Real (Physical) Size of memory area
+ * msize -- Mapped (Virtual) Size of memory area (Size of area mapped
+ * into virtual) memory
+ *
+ * --------------------------------------
+ * | | <----- msize ----->| |
+ * |<------|------- rsize ------|------>|
+ * --------------------------------------
+ * /|\ /|\
+ * | |
+ * ual_vaddr vaddr/paddr
+ *
+ * --------------------------------------------------------------------------------------
+ */
+struct _memarea_ {
+ ioaddr_t ioaddr;
+ vkaddr_t vaddr;
+ vkaddr_t ual_vaddr;
+ size_t rsize;
+ size_t msize;
+ char *id;
+ unsigned32 cookie;
+
+
+#ifdef SUNOS5
+#ifdef _USE_NEW_SOLARIS_DDI_INTERFACE
+ ddi_acc_handle_t mem_handle;
+ ddi_dma_handle_t dma_handle;
+#else
+ ddi_dma_handle_t handle;
+#endif
+#endif
+#ifdef OS_IS_TRU64
+ dma_handle_t dma_handle;
+#endif
+
+#if OS_IS_LINUX
+ unsigned long ph_base_addr;
+#endif
+
+#ifdef OS_IS_HP_UX11
+ struct isc_table_type * isc;
+ wsio_shmem_attr_t type;
+#endif
+};
+
+typedef struct _memarea_ memarea_t;
+
+#ifdef SCI_MALLOC_DEBUG
+struct _maddr_ {
+ char *id;
+ size_t size;
+ struct _maddr_ *next;
+ struct _maddr_ **prev;
+ unsigned32 cookie;
+};
+
+typedef struct _maddr_ maddr_t;
+
+#define MALLOC_COOKIE 0xc3c3c3c3
+
+#else
+
+typedef struct { void *p; } *maddr_t;
+
+#endif /* SCI_MALLOC_DEBUG */
+
+
+typedef struct {
+ scibool disabled;
+ unsigned32 disable_cnt;
+} disable_info_t;
+
+#endif /* _KERNEL */
+
+#endif /* _SCI_TYPES_H_ */
diff --git a/ndb/src/external/LINUX.x86/sci/include/sisci_api.h b/ndb/src/external/LINUX.x86/sci/include/sisci_api.h
new file mode 100644
index 00000000000..38fdf54125f
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/sisci_api.h
@@ -0,0 +1,2170 @@
+/* $Id: sisci_api.h,v 1.1 2002/12/13 12:17:21 hin Exp $ */
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2001 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU Lesser General Public License as published by *
+ * the Free Software Foundation; either version 2.1 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 Lesser 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+#ifndef _SISCI_API_H
+#define _SISCI_API_H
+
+#include "sisci_types.h"
+#include "sisci_error.h"
+
+
+#ifdef WIN32
+#ifdef API_DLL
+#define DLL __declspec(dllexport)
+#elif CLIENT_DLL
+#define DLL __declspec(dllimport)
+#endif
+#endif /* WIN32 */
+
+
+#ifndef DLL
+#define DLL
+#endif
+
+#if defined(_REENTRANT)
+#define _SISCI_EXPANDE_FUNCTION_NAME(name) _SISCI_PUBLIC_FUNC_MT_ ## name
+#define _SISCI_EXPANDE_VARIABLE_NAME(name) _SISCI_PUBLIC_VAR_MT_ ## name
+#else
+#define _SISCI_EXPANDE_FUNCTION_NAME(name) _SISCI_PUBLIC_FUNC_ST_ ## name
+#define _SISCI_EXPANDE_VARIABLE_NAME(name) _SISCI_PUBLIC_VAR_ST_ ## name
+#endif
+#define _SISCI_EXPANDE_CONSTANT_NAME(name) _SISCI_PUBLIC_CONST_ ## name
+
+#if defined(CPLUSPLUS) || defined(__cplusplus)
+extern "C" {
+#endif
+
+
+/*********************************************************************************/
+/* FLAG VALUES */
+/*********************************************************************************/
+
+#define SCI_FLAG_FIXED_INTNO _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FIXED_INTNO)
+extern const unsigned int SCI_FLAG_FIXED_INTNO;
+
+#define SCI_FLAG_SHARED_INT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_SHARED_INT)
+extern const unsigned int SCI_FLAG_SHARED_INT;
+
+#define SCI_FLAG_FIXED_MAP_ADDR _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FIXED_MAP_ADDR)
+extern const unsigned int SCI_FLAG_FIXED_MAP_ADDR;
+
+#define SCI_FLAG_READONLY_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_READONLY_MAP)
+extern const unsigned int SCI_FLAG_READONLY_MAP;
+
+#define SCI_FLAG_USE_CALLBACK _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_USE_CALLBACK)
+extern const unsigned int SCI_FLAG_USE_CALLBACK;
+
+#define SCI_FLAG_BLOCK_READ _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_BLOCK_READ)
+extern const unsigned int SCI_FLAG_BLOCK_READ;
+
+#define SCI_FLAG_THREAD_SAFE _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_THREAD_SAFE)
+extern const unsigned int SCI_FLAG_THREAD_SAFE;
+
+#define SCI_FLAG_ASYNCHRONOUS_CONNECT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_ASYNCHRONOUS_CONNECT)
+extern const unsigned int SCI_FLAG_ASYNCHRONOUS_CONNECT;
+
+#define SCI_FLAG_EMPTY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_EMPTY)
+extern const unsigned int SCI_FLAG_EMPTY;
+
+#define SCI_FLAG_PRIVATE _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_PRIVATE)
+extern const unsigned int SCI_FLAG_PRIVATE;
+
+#define SCI_FLAG_FORCE_DISCONNECT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FORCE_DISCONNECT)
+extern const unsigned int SCI_FLAG_FORCE_DISCONNECT;
+
+#define SCI_FLAG_NOTIFY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NOTIFY)
+extern const unsigned int SCI_FLAG_NOTIFY;
+
+#define SCI_FLAG_DMA_READ _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_READ)
+extern const unsigned int SCI_FLAG_DMA_READ;
+
+#define SCI_FLAG_DMA_POST _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_POST)
+extern const unsigned int SCI_FLAG_DMA_POST;
+
+#define SCI_FLAG_DMA_WAIT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_WAIT)
+extern const unsigned int SCI_FLAG_DMA_WAIT;
+
+#define SCI_FLAG_DMA_RESET _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_RESET)
+extern const unsigned int SCI_FLAG_DMA_RESET;
+
+#define SCI_FLAG_NO_FLUSH _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NO_FLUSH)
+extern const unsigned int SCI_FLAG_NO_FLUSH;
+
+#define SCI_FLAG_NO_STORE_BARRIER _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NO_STORE_BARRIER)
+extern const unsigned int SCI_FLAG_NO_STORE_BARRIER;
+
+#define SCI_FLAG_FAST_BARRIER _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FAST_BARRIER)
+extern const unsigned int SCI_FLAG_FAST_BARRIER;
+
+#define SCI_FLAG_ERROR_CHECK _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_ERROR_CHECK)
+extern const unsigned int SCI_FLAG_ERROR_CHECK;
+
+#define SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY)
+extern const unsigned int SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY;
+
+/* the FLUSH_CPU_BUFFERS_ONLY flag is for backwards compabillity only and should never be used */
+#define FLUSH_CPU_BUFFERS_ONLY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY)
+
+#define SCI_FLAG_LOCK_OPERATION _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_LOCK_OPERATION)
+extern const unsigned int SCI_FLAG_LOCK_OPERATION;
+
+#define SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP)
+extern const unsigned int SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP;
+
+#define SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP)
+extern const unsigned int SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP;
+
+#define SCI_FLAG_IO_MAP_IOSPACE _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_IO_MAP_IOSPACE)
+extern const unsigned int SCI_FLAG_IO_MAP_IOSPACE;
+
+#define SCI_FLAG_DMOVE_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMOVE_MAP)
+extern const unsigned int SCI_FLAG_DMOVE_MAP;
+
+#define SCI_FLAG_WRITES_DISABLE_GATHER_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_WRITES_DISABLE_GATHER_MAP)
+extern const unsigned int SCI_FLAG_WRITES_DISABLE_GATHER_MAP;
+
+#define SCI_FLAG_DISABLE_128_BYTES_PACKETS _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DISABLE_128_BYTES_PACKETS)
+extern const unsigned int SCI_FLAG_DISABLE_128_BYTES_PACKETS;
+
+#define SCI_FLAG_SHARED_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_SHARED_MAP)
+extern const unsigned int SCI_FLAG_SHARED_MAP;
+
+#define SCI_FLAG_DMA_SOURCE_ONLY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_SOURCE_ONLY)
+extern const unsigned int SCI_FLAG_DMA_SOURCE_ONLY;
+
+#define SCI_FLAG_CONDITIONAL_INTERRUPT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_CONDITIONAL_INTERRUPT)
+extern const unsigned int SCI_FLAG_CONDITIONAL_INTERRUPT;
+
+#define SCI_FLAG_CONDITIONAL_INTERRUPT_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_CONDITIONAL_INTERRUPT_MAP)
+extern const unsigned int SCI_FLAG_CONDITIONAL_INTERRUPT_MAP;
+
+#define SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP)
+extern const unsigned int SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP;
+
+#define SCI_FLAG_NO_MEMORY_LOOPBACK_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NO_MEMORY_LOOPBACK_MAP)
+extern const unsigned int SCI_FLAG_NO_MEMORY_LOOPBACK_MAP;
+
+#if defined(OS_IS_LYNXOS) || defined(OS_IS_VXWORKS)
+#define SCI_FLAG_WRITE_BACK_CACHE_MAP _SISCI_EXPANDE_CONSTANT_NAME(WRITE_BACK_CACHE_MAP)
+extern const unsigned int SCI_FLAG_WRITE_BACK_CACHE_MAP;
+#endif
+
+#define SCI_FLAG_DMA_PHDMA _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_PHDMA)
+extern const unsigned int SCI_FLAG_DMA_PHDMA;
+
+/*********************************************************************************/
+/* GENERAL VALUES */
+/*********************************************************************************/
+#define SCI_LOCAL_HOST _SISCI_EXPANDE_CONSTANT_NAME(SCI_LOCAL_HOST)
+extern const unsigned int SCI_LOCAL_HOST;
+
+#define SCI_INFINITE_TIMEOUT _SISCI_EXPANDE_CONSTANT_NAME(SCI_INFINITE_TIMEOUT)
+extern const unsigned int SCI_INFINITE_TIMEOUT;
+
+/*********************************************************************************/
+/* GENERAL ERROR CODES */
+/* */
+/* SCI_ERR_ILLEGAL_FLAG - Illegal flag value. */
+/* SCI_ERR_FLAG_NOT_IMPLEMENTED - Flag legal but flag feature not implemented. */
+/* SCI_ERR_NOT_IMPLEMENTED - Function not implemented. */
+/* SCI_ERR_SYSTEM - A system error. Check errno. */
+/* SCI_ERR_NOSPC - Unable to allocate OS resources. */
+/* SCI_ERR_API_NOSPC - Unable to allocate API resources. */
+/* SCI_ERR_HW_NOSPC - Unable to allocate HW resources (Hardware) */
+/* */
+/*********************************************************************************/
+
+
+/*********************************************************************************/
+/* GENERAL "ADAPTER" ERROR CODES */
+/* */
+/* SCI_ERR_NO_SUCH_ADAPTERNO - Adapter number is legal but does not exist. */
+/* SCI_ERR_ILLEGAL_ADAPTERNO - Illegal local adapter number (i.e. outside */
+/* legal range). */
+/* */
+/*********************************************************************************/
+
+
+/*********************************************************************************/
+/* GENERAL "NODEID" ERROR CODES */
+/* */
+/* SCI_ERR_NO_SUCH_NODEID - The remote adapter identified by nodeId does */
+/* not respond, but the intermediate link(s) */
+/* seem(s) to be operational. */
+/* SCI_ERR_ILLEGAL_NODEID - Illegal NodeId. */
+/* */
+/*********************************************************************************/
+
+
+
+/*********************************************************************************
+ * *
+ * S C I I N I T I A L I Z E *
+ * *
+ * This function initializes the SISCI library. *
+ * The function must be called before SCIOpen(). *
+ * *
+ * Flags: *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * None *
+ * *
+ *********************************************************************************/
+#define SCIInitialize _SISCI_EXPANDE_FUNCTION_NAME(SCIInitialize)
+DLL void SCIInitialize(unsigned int flags,
+ sci_error_t *error);
+#if 0
+unsigned int __Internal_SISCI_version_var;
+#endif
+
+/*********************************************************************************
+ * *
+ * S C I T E R M I N A T E *
+ * *
+ * This function terminates the SISCI library. *
+ * The function must be called after SCIClose(). *
+ * *
+ * *
+ *********************************************************************************/
+#define SCITerminate _SISCI_EXPANDE_FUNCTION_NAME(SCITerminate)
+DLL void SCITerminate(void);
+
+/*********************************************************************************
+ * *
+ * S C I O P E N *
+ * *
+ * *
+ * Opens a SCI virtual device. *
+ * Caller must supply a pointer to a variable of type sci_desc_t to be *
+ * initialized. *
+ * *
+ * Flags *
+ * SCI_FLAG_THREAD_SAFE - Operations on resources associated with this *
+ * descriptor will be performed in a multithread-safe *
+ * manner. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_INCONSISTENT_VERSIONS - Inconsistency between the SISCI library *
+ * and the SISCI driver versions. *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIOpen _SISCI_EXPANDE_FUNCTION_NAME(SCIOpen)
+DLL void SCIOpen(sci_desc_t *sd,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C L O S E *
+ * *
+ * This function closes an open SCI virtual device. *
+ * *
+ * Flags: *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - All resources are not deallocated. *
+ * *
+ *********************************************************************************/
+#define SCIClose _SISCI_EXPANDE_FUNCTION_NAME(SCIClose)
+DLL void SCIClose(sci_desc_t sd,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C O N N E C T S E G M E N T *
+ * *
+ * Connects to a remote shared memory segment located at <nodeId> with the *
+ * identifier <segmentId>. *
+ * The user may then call SCIMapRemoteSegment() to map shared memory *
+ * into user space. *
+ * *
+ * Flags *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_ASYNCHRONOUS_CONNECT *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_SUCH_SEGMENT - Could not find the remote segment with the *
+ * given segmentId. *
+ * SCI_ERR_CONNECTION_REFUSED - Connection attempt refused by remote node. *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCIConnectSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIConnectSegment)
+DLL void SCIConnectSegment(sci_desc_t sd,
+ sci_remote_segment_t *segment,
+ unsigned int nodeId,
+ unsigned int segmentId,
+ unsigned int localAdapterNo,
+ sci_cb_remote_segment_t callback,
+ void *callbackArg,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I D I S C O N N E C T S E G M E N T *
+ * *
+ * Disconnects from the give mapped shared memory segment *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - The segment is currently mapped or in use. *
+ * *
+ *********************************************************************************/
+#define SCIDisconnectSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIDisconnectSegment)
+DLL void SCIDisconnectSegment(sci_remote_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T R E M O T E S E G M E N T S I Z E *
+ * *
+ *********************************************************************************/
+#define SCIGetRemoteSegmentSize _SISCI_EXPANDE_FUNCTION_NAME(SCIGetRemoteSegmentSize)
+DLL unsigned int SCIGetRemoteSegmentSize(sci_remote_segment_t segment);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R R E M O T E S E G M E N T E V E N T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation. *
+ * SCI_ERR_CANCELLED - The wait operation has been cancelled du *
+ * to a SCIDisconnectSegment() on the same *
+ * handle. The handle is invalid when this *
+ * error is returned. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForRemoteSegmentEvent _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForRemoteSegmentEvent)
+DLL sci_segment_cb_reason_t SCIWaitForRemoteSegmentEvent(
+ sci_remote_segment_t segment,
+ sci_error_t *status,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M A P R E M O T E S E G M E N T *
+ * *
+ * This function is used to include a shared memory segment in the virtual *
+ * address space of the application. *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_SHARED_MAP - The low level physical map may be shared by *
+ * other applications. *
+ * *
+ * SCI_FLAG_FIXED_MAP_ADDR - Map at the suggested virtual address *
+ * SCI_FLAG_READONLY_MAP - The segment is mapped in read-only mode *
+ * SCI_FLAG_LOCK_OPERATION - Enable Lock operations (fetch and add) *
+ * SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP *
+ * - Enable aggressive prefetch with speculative *
+ * hold. *
+ * *
+ * SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP *
+ * - The PSB66 will prefetch 64 bytes. As soon *
+ * as the PCI read retry has been accepted, *
+ * the stream will change state to FREE, even *
+ * if less than 64 bytes were actually read. *
+ * *
+ * SCI_FLAG_IO_MAP_IOSPACE - Enable No Prefetch, no speculative hold. *
+ * *
+ * SCI_FLAG_DMOVE_MAP - Enable DMOVE packet type. The stream will be *
+ * set into FREE state immediately. *
+ * *
+ * SCI_FLAG_WRITES_DISABLE_GATHER_MAP *
+ * - Disable use of gather. *
+ * *
+ * SCI_FLAG_DISABLE_128_BYTES_PACKETS *
+ * - Disable use of 128-Byte packets *
+ * *
+ * SCI_FLAG_CONDITIONAL_INTERRUPT_MAP *
+ * - Write operations through this map will cause *
+ * an atomic "fetch-and-add-one" operation on *
+ * remote memory, but in addition an interrupt *
+ * will be generated if the target memory *
+ * location contained a "null value" before the *
+ * add operation was carried out. *
+ * The conditional interrupt flag must also be *
+ * specified in the SCIRegisterInterruptFlag() *
+ * function. *
+ * *
+ * SCI_FLAG_UNCONDITIONAL_INTERRUPT_MAP *
+ * - Write operations through this map will cause *
+ * an interrupt for the remote adapter *
+ * "in addition to" updating the corresponding *
+ * remote memory location with the data being *
+ * written. *
+ * The unconditional interrupt flag must also *
+ * be specified in the *
+ * SCIRegisterInterruptFlag() function. *
+ * *
+ * SCI_FLAG_WRITE_BACK_CACHE_MAP *
+ * - Enable cacheing of the mapped region. *
+ * Writes through this map will be written to a *
+ * write back cache, hence no remote SCI updates*
+ * until the cache line is flushed. The *
+ * application is responsible for the cache *
+ * flush operation. *
+ * The SCImemCopy() function will handle this *
+ * correctly by doing cache flushes internally. *
+ * This feature is architechture dependent and *
+ * not be available on all plattforms. *
+ * *
+ * SCI_FLAG_NO_MEMORY_LOOPBACK_MAP *
+ * - Forces a map to a remote segment located *
+ * in the local machine to be mapped using *
+ * SCI loopback. This is useful i.e. if you *
+ * want to use a regular map access to be *
+ * serialized with lock operations. *
+ * The default behaviour is to access a remte *
+ * segment located in the local machine as a *
+ * local MMU operation. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the offset and size is *
+ * larger than the segment size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as *
+ * required by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as *
+ * required by the implementation. *
+ * *
+ *********************************************************************************/
+#define SCIMapRemoteSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIMapRemoteSegment)
+DLL volatile void *SCIMapRemoteSegment(
+ sci_remote_segment_t segment,
+ sci_map_t *map,
+ unsigned int offset,
+ unsigned int size,
+ void *addr,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M A P L O C A L S E G M E N T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_FIXED_MAP_ADDR *
+ * SCI_FLAG_READONLY_MAP *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the offset and size is *
+ * larger than the segment size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as *
+ * required by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as *
+ * required by the implementation. *
+ * *
+ *********************************************************************************/
+#define SCIMapLocalSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIMapLocalSegment)
+DLL void *SCIMapLocalSegment(sci_local_segment_t segment,
+ sci_map_t *map,
+ unsigned int offset,
+ unsigned int size,
+ void *addr,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I U N M A P S E G M E N T *
+ * *
+ * This function unmaps pages of shared memory from the callers virtual *
+ * address space. *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - The map is currently in use. *
+ * *
+ *********************************************************************************/
+#define SCIUnmapSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIUnmapSegment)
+DLL void SCIUnmapSegment(sci_map_t map,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E S E G M E N T *
+ * *
+ * Make the specified segment available for connections via the specified *
+ * adapter. If successful, the segment can be accessed from remote nodes *
+ * via the specified adapter. *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_USE_CALLBACK - The callback function will be invoked for events *
+ * on this segment. *
+ * SCI_FLAG_EMPTY - No memory will be allocated for the segment. *
+ * SCI_FLAG_PRIVATE - The segment will be private meaning it will never *
+ * be any connections to it. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_SEGMENTID_USED - The segment with this segmentId is already used *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * *
+ *********************************************************************************/
+#define SCICreateSegment _SISCI_EXPANDE_FUNCTION_NAME(SCICreateSegment)
+DLL void SCICreateSegment(sci_desc_t sd,
+ sci_local_segment_t *segment,
+ unsigned int segmentId,
+ unsigned int size,
+ sci_cb_local_segment_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R L O C A L S E G M E N T E V E N T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_TIMEOUT - The function timed out after specified timeout value. *
+ * SCI_ERR_CANCELLED - The wait operation has been cancelled du to a *
+ * SCIRemoveSegment() on the same handle. *
+ * The handle is invalid when this error is returned. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForLocalSegmentEvent _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForLocalSegmentEvent)
+DLL sci_segment_cb_reason_t SCIWaitForLocalSegmentEvent(
+ sci_local_segment_t segment,
+ unsigned int *sourcenodeId,
+ unsigned int *localAdapterNo,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P R E P A R E S E G M E N T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_SOURCE_ONLY - The segment will be used as a source segment *
+ * for DMA operations. On some system types this *
+ * will enable the SISCI driver to use performance *
+ * improving features. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIPrepareSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIPrepareSegment)
+DLL void SCIPrepareSegment(sci_local_segment_t segment,
+ unsigned int localAdapterNo,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E S E G M E N T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - Unable to remove the segment. The segment is currently *
+ * in use. *
+ * *
+ *********************************************************************************/
+#define SCIRemoveSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveSegment)
+DLL void SCIRemoveSegment(sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T S E G M E N T A V A I L A B L E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * SCI_ERR_SEGMENT_NOT_PREPARED - The segment has not been prepared for access *
+ * from this adapter. *
+ * SCI_ERR_ILLEGAL_OPERATION - The segment is created with the *
+ * SCI_FLAG_PRIVATE flag specified and *
+ * therefore has no segmentId. *
+ * *
+ *********************************************************************************/
+#define SCISetSegmentAvailable _SISCI_EXPANDE_FUNCTION_NAME(SCISetSegmentAvailable)
+DLL void SCISetSegmentAvailable(sci_local_segment_t segment,
+ unsigned int localAdapterNo,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T S E G M E N T U N A V A I L A B L E *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_FORCE_DISCONNECT *
+ * SCI_FLAG_NOTIFY *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation. *
+ * *
+ *********************************************************************************/
+#define SCISetSegmentUnavailable _SISCI_EXPANDE_FUNCTION_NAME(SCISetSegmentUnavailable)
+DLL void SCISetSegmentUnavailable(sci_local_segment_t segment,
+ unsigned int localAdapterNo,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E M A P S E Q U E N C E *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_FAST_BARRIER *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCICreateMapSequence _SISCI_EXPANDE_FUNCTION_NAME(SCICreateMapSequence)
+DLL void SCICreateMapSequence(sci_map_t map,
+ sci_sequence_t *sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E S E Q U E N C E *
+ * *
+ * Flags: *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIRemoveSequence _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveSequence)
+DLL void SCIRemoveSequence(sci_sequence_t sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S T A R T S E Q U E N C E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIStartSequence _SISCI_EXPANDE_FUNCTION_NAME(SCIStartSequence)
+DLL sci_sequence_status_t SCIStartSequence(sci_sequence_t sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C H E C K S E Q U E N CE *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_NO_FLUSH *
+ * SCI_FLAG_NO_STORE_BARRIER *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCICheckSequence _SISCI_EXPANDE_FUNCTION_NAME(SCICheckSequence)
+DLL sci_sequence_status_t SCICheckSequence(sci_sequence_t sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S T O R E B A R R I E R *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIStoreBarrier _SISCI_EXPANDE_FUNCTION_NAME(SCIStoreBarrier)
+DLL void SCIStoreBarrier(sci_sequence_t sequence,
+ unsigned int flags);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I F L U S H R E A D B U F F E R S *
+ * *
+ *********************************************************************************/
+#define SCIFlushReadBuffers _SISCI_EXPANDE_FUNCTION_NAME(SCIFlushReadBuffers)
+DLL void SCIFlushReadBuffers(sci_sequence_t sequence);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P R O B E N O D E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCIProbeNode _SISCI_EXPANDE_FUNCTION_NAME(SCIProbeNode)
+DLL int SCIProbeNode(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int nodeId,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T C S R R E G I S T E R *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCIGetCSRRegister _SISCI_EXPANDE_FUNCTION_NAME(SCIGetCSRRegister)
+DLL unsigned int SCIGetCSRRegister(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int SCINodeId,
+ unsigned int CSROffset,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T C S R R E G I S T E R *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCISetCSRRegister _SISCI_EXPANDE_FUNCTION_NAME(SCISetCSRRegister)
+DLL void SCISetCSRRegister(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int SCINodeId,
+ unsigned int CSROffset,
+ unsigned int CSRValue,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T L O C A L C S R *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIGetLocalCSR _SISCI_EXPANDE_FUNCTION_NAME(SCIGetLocalCSR)
+DLL unsigned int SCIGetLocalCSR(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int CSROffset,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T L O C A L C S R *
+ * *
+ * SISCI Priveleged function
+ *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCISetLocalCSR _SISCI_EXPANDE_FUNCTION_NAME(SCISetLocalCSR)
+DLL void SCISetLocalCSR(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int CSROffset,
+ unsigned int CSRValue,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I A T T A C H P H Y S I C A L M E M O R Y *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * Description: *
+ * *
+ * This function enables usage of physical devices and memory regions where the *
+ * Physical PCI bus address ( and mapped CPU address ) are already known. *
+ * The function will register the physical memory as a SISCI segment which can *
+ * be connected and mapped as a regular SISCI segment. *
+ * *
+ * Requirements: *
+ * *
+ * SCICreateSegment() with flag SCI_FLAG_EMPTY must have been called in advance *
+ * *
+ * Parameter description: *
+ * sci_ioaddr_t ioaddress : This is the address on the PCI bus that a PCI bus *
+ * master has to use to write to the specified memory *
+ * void * address : This is the (mapped) virtual address that the *
+ * application has to use to access the device. *
+ * This means that the device has to be mapped in *
+ * advance bye the devices own driver. *
+ * If the device is not to be accessed by the local *
+ * CPU, the address pointer shold be set to NULL *
+ * Flags *
+ * *
+ * None *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIAttachPhysicalMemory _SISCI_EXPANDE_FUNCTION_NAME(SCIAttachPhysicalMemory)
+DLL void SCIAttachPhysicalMemory(sci_ioaddr_t ioaddress,
+ void *address,
+ unsigned int busNo,
+ unsigned int size,
+ sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I Q U E R Y *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_QUERY - Unrecognized command. *
+ * *
+ *********************************************************************************/
+#define SCIQuery _SISCI_EXPANDE_FUNCTION_NAME(SCIQuery)
+DLL void SCIQuery(unsigned int command,
+ void *data,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/* MAJOR QUERY COMMANDS */
+
+/* This command requires a pointer to a structure of type */
+/* "sci_query_string". The string will be filled in by the query. */
+#define SCI_Q_VENDORID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_VENDORID)
+extern const unsigned int SCI_Q_VENDORID;
+
+
+/* Same as for SCI_VENDOR_ID */
+#define SCI_Q_API _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_API)
+extern const unsigned int SCI_Q_API;
+
+
+/* User passes a pointer to an allocated object of the */
+/* "sci_query_adapter" struct. */
+#define SCI_Q_ADAPTER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER)
+extern const unsigned int SCI_Q_ADAPTER;
+
+
+/* User passes a pointer to an allocated object of the */
+/* "sci_query_system" struct. */
+#define SCI_Q_SYSTEM _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM)
+extern const unsigned int SCI_Q_SYSTEM;
+
+#define SCI_Q_LOCAL_SEGMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_LOCAL_SEGMENT)
+extern const unsigned int SCI_Q_LOCAL_SEGMENT;
+
+#define SCI_Q_REMOTE_SEGMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_REMOTE_SEGMENT)
+extern const unsigned int SCI_Q_REMOTE_SEGMENT;
+
+#define SCI_Q_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_MAP)
+extern const unsigned int SCI_Q_MAP;
+
+typedef struct {
+ char *str; /* Pointer to a string of minimum "length" characters */
+ unsigned int length;
+} sci_query_string_t;
+
+
+typedef struct {
+ unsigned int localAdapterNo; /* The adapter no. that the query concern. */
+ unsigned int portNo; /* The SCI Link port number that the query concern. */
+ unsigned int subcommand; /* A subcommand as specified below. */
+ void *data; /* A pointer to an unsigned int that will return */
+ /* the response to the query. */
+} sci_query_adapter_t;
+
+
+typedef struct {
+ unsigned int subcommand; /* A subcommand as specified below. */
+ void *data; /* A pointer to an unsigned int that will return */
+ /* the response to the query. */
+} sci_query_system_t;
+
+typedef struct {
+ sci_local_segment_t segment;
+ unsigned int subcommand;
+ union {
+ sci_ioaddr_t ioaddr;
+ } data;
+} sci_query_local_segment_t;
+
+typedef struct {
+ sci_remote_segment_t segment;
+ unsigned int subcommand;
+ union {
+ sci_ioaddr_t ioaddr;
+ } data;
+} sci_query_remote_segment_t;
+
+typedef struct {
+ sci_map_t map;
+ unsigned int subcommand;
+ unsigned int data;
+} sci_query_map_t;
+
+/* Minor query commands (sub-commands) for adapter specific information SCI_ADAPTER */
+#define SCI_Q_ADAPTER_DMA_SIZE_ALIGNMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_DMA_SIZE_ALIGNMENT)
+extern const unsigned int SCI_Q_ADAPTER_DMA_SIZE_ALIGNMENT;
+
+#define SCI_Q_ADAPTER_DMA_OFFSET_ALIGNMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_DMA_OFFSET_ALIGNMENT)
+extern const unsigned int SCI_Q_ADAPTER_DMA_OFFSET_ALIGNMENT;
+
+#define SCI_Q_ADAPTER_DMA_MTU _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_DMA_MTU)
+extern const unsigned int SCI_Q_ADAPTER_DMA_MTU;
+
+#define SCI_Q_ADAPTER_SUGGESTED_MIN_DMA_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SUGGESTED_MIN_DMA_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_SUGGESTED_MIN_DMA_SIZE;
+
+#define SCI_Q_ADAPTER_SUGGESTED_MIN_BLOCK_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SUGGESTED_MIN_BLOCK_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_SUGGESTED_MIN_BLOCK_SIZE;
+
+#define SCI_Q_ADAPTER_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_NODEID;
+
+#define SCI_Q_ADAPTER_SERIAL_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SERIAL_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_SERIAL_NUMBER;
+
+#define SCI_Q_ADAPTER_CARD_TYPE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CARD_TYPE)
+extern const unsigned int SCI_Q_ADAPTER_CARD_TYPE;
+
+#define SCI_Q_ADAPTER_NUMBER_OF_STREAMS _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_NUMBER_OF_STREAMS)
+extern const unsigned int SCI_Q_ADAPTER_NUMBER_OF_STREAMS;
+
+#define SCI_Q_ADAPTER_STREAM_BUFFER_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_STREAM_BUFFER_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_STREAM_BUFFER_SIZE;
+
+#define SCI_Q_ADAPTER_CONFIGURED _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CONFIGURED)
+extern const unsigned int SCI_Q_ADAPTER_CONFIGURED;
+
+#define SCI_Q_ADAPTER_LINK_OPERATIONAL _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_LINK_OPERATIONAL)
+extern const unsigned int SCI_Q_ADAPTER_LINK_OPERATIONAL;
+
+#define SCI_Q_ADAPTER_HW_LINK_STATUS_IS_OK _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_HW_LINK_STATUS_IS_OK)
+extern const unsigned int SCI_Q_ADAPTER_HW_LINK_STATUS_IS_OK;
+
+#define SCI_Q_ADAPTER_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_NUMBER;
+
+#define SCI_Q_ADAPTER_INSTANCE_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_INSTANCE_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_INSTANCE_NUMBER;
+
+#define SCI_Q_ADAPTER_FIRMWARE_OK _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_FIRMWARE_OK)
+extern const unsigned int SCI_Q_ADAPTER_FIRMWARE_OK;
+
+#define SCI_Q_ADAPTER_CONNECTED_TO_SWITCH _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CONNECTED_TO_SWITCH)
+extern const unsigned int SCI_Q_ADAPTER_CONNECTED_TO_SWITCH;
+
+#define SCI_Q_ADAPTER_LOCAL_SWITCH_TYPE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_LOCAL_SWITCH_TYPE)
+extern const unsigned int SCI_Q_ADAPTER_LOCAL_SWITCH_TYPE;
+
+#define SCI_Q_ADAPTER_LOCAL_SWITCH_PORT_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_LOCAL_SWITCH_PORT_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_LOCAL_SWITCH_PORT_NUMBER;
+
+#define SCI_Q_ADAPTER_CONNECTED_TO_EXPECTED_SWITCH_PORT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CONNECTED_TO_EXPECTED_SWITCH_PORT)
+extern const unsigned int SCI_Q_ADAPTER_CONNECTED_TO_EXPECTED_SWITCH_PORT;
+
+#define SCI_Q_ADAPTER_ATT_PAGE_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_ATT_PAGE_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_ATT_PAGE_SIZE;
+
+#define SCI_Q_ADAPTER_ATT_NUMBER_OF_ENTRIES _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_ATT_NUMBER_OF_ENTRIES)
+extern const unsigned int SCI_Q_ADAPTER_ATT_NUMBER_OF_ENTRIES;
+
+#define SCI_Q_ADAPTER_ATT_AVAILABLE_ENTRIES _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_ATT_AVAILABLE_ENTRIES)
+extern const unsigned int SCI_Q_ADAPTER_ATT_AVAILABLE_ENTRIES;
+
+#define SCI_Q_ADAPTER_PHYS_MEM_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SCI_Q_ADAPTER_PHYS_MEM_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_PHYS_MEM_NODEID;
+
+#define SCI_Q_ADAPTER_PHYS_MBX_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SCI_Q_ADAPTER_PHYS_MBX_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_PHYS_MBX_NODEID;
+
+#define SCI_Q_ADAPTER_PHYS_LINK_PORT_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_PHYS_LINK_PORT_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_PHYS_LINK_PORT_NODEID;
+
+#define SCI_Q_ADAPTER_SCI_LINK_FREQUENCY _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SCI_LINK_FREQUENCY)
+extern const unsigned int SCI_Q_ADAPTER_SCI_LINK_FREQUENCY;
+
+#define SCI_Q_ADAPTER_B_LINK_FREQUENCY _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_B_LINK_FREQUENCY)
+extern const unsigned int SCI_Q_ADAPTER_B_LINK_FREQUENCY;
+
+#define SCI_Q_ADAPTER_IO_BUS_FREQUENCY _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_IO_BUS_FREQUENCY)
+extern const unsigned int SCI_Q_ADAPTER_IO_BUS_FREQUENCY;
+
+/* Minor query commands (sub-commands) for adapter specific information SCI_SYSTEM */
+#define SCI_Q_SYSTEM_HOSTBRIDGE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM_HOSTBRIDGE)
+extern const unsigned int SCI_Q_SYSTEM_HOSTBRIDGE;
+
+#define SCI_Q_SYSTEM_WRITE_POSTING_ENABLED _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM_WRITE_POSTING_ENABLED)
+extern const unsigned int SCI_Q_SYSTEM_WRITE_POSTING_ENABLED;
+
+#define SCI_Q_SYSTEM_WRITE_COMBINING_ENABLED _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM_WRITE_COMBINING_ENABLED)
+extern const unsigned int SCI_Q_SYSTEM_WRITE_COMBINING_ENABLED;
+
+#define SCI_Q_LOCAL_SEGMENT_IOADDR _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_LOCAL_SEGMENT_IOADDR)
+extern const unsigned int SCI_Q_LOCAL_SEGMENT_IOADDR;
+
+#define SCI_Q_REMOTE_SEGMENT_IOADDR _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_REMOTE_SEGMENT_IOADDR)
+extern const unsigned int SCI_Q_REMOTE_SEGMENT_IOADDR;
+
+#define SCI_Q_MAP_MAPPED_TO_LOCAL_TARGET _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_MAP_MAPPED_TO_LOCAL_TARGET)
+extern const unsigned int SCI_Q_MAP_MAPPED_TO_LOCAL_TARGET;
+
+#define SCI_Q_MAP_QUERY_REMOTE_MAPPED_TO_LOCAL_TARGET _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_MAP_QUERY_REMOTE_MAPPED_TO_LOCAL_TARGET)
+extern const unsigned int SCI_Q_MAP_QUERY_REMOTE_MAPPED_TO_LOCAL_TARGET;
+
+#define HOSTBRIDGE_NOT_AVAILABLE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_NOT_AVAILABLE)
+extern const unsigned int HOSTBRIDGE_NOT_AVAILABLE;
+
+#define HOSTBRIDGE_UNKNOWN _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_UNKNOWN)
+extern const unsigned int HOSTBRIDGE_UNKNOWN;
+
+#define HOSTBRIDGE_440FX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440FX)
+extern const unsigned int HOSTBRIDGE_440FX;
+
+#define HOSTBRIDGE_440LX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440LX)
+extern const unsigned int HOSTBRIDGE_440LX;
+
+#define HOSTBRIDGE_440BX_A _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440BX_A)
+extern const unsigned int HOSTBRIDGE_440BX_A;
+
+#define HOSTBRIDGE_440BX_B _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440BX_B)
+extern const unsigned int HOSTBRIDGE_440BX_B;
+
+#define HOSTBRIDGE_440GX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440GX)
+extern const unsigned int HOSTBRIDGE_440GX;
+
+#define HOSTBRIDGE_450KX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450KX)
+extern const unsigned int HOSTBRIDGE_450KX;
+
+#define HOSTBRIDGE_430NX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_430NX)
+extern const unsigned int HOSTBRIDGE_430NX;
+
+#define HOSTBRIDGE_450NX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450NX)
+extern const unsigned int HOSTBRIDGE_450NX;
+
+#define HOSTBRIDGE_450NX_MICO _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450NX_MICO)
+extern const unsigned int HOSTBRIDGE_450NX_MICO;
+
+#define HOSTBRIDGE_450NX_PXB _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450NX_PXB)
+extern const unsigned int HOSTBRIDGE_450NX_PXB;
+
+#define HOSTBRIDGE_I810 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I810)
+extern const unsigned int HOSTBRIDGE_I810;
+
+#define HOSTBRIDGE_I810_DC100 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I810_DC100)
+extern const unsigned int HOSTBRIDGE_I810_DC100;
+
+#define HOSTBRIDGE_I810E _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I810E)
+extern const unsigned int HOSTBRIDGE_I810E;
+
+#define HOSTBRIDGE_I815 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I815)
+extern const unsigned int HOSTBRIDGE_I815;
+
+#define HOSTBRIDGE_I840 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I840)
+extern const unsigned int HOSTBRIDGE_I840;
+
+#define HOSTBRIDGE_I850 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I850)
+extern const unsigned int HOSTBRIDGE_I850;
+
+#define HOSTBRIDGE_I860 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I860)
+extern const unsigned int HOSTBRIDGE_I860;
+
+#define HOSTBRIDGE_INTEL_E7500 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_INTEL_E7500)
+extern const unsigned int HOSTBRIDGE_INTEL_E7500;
+
+#define HOSTBRIDGE_VIA_KT133 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_KT133)
+extern const unsigned int HOSTBRIDGE_VIA_KT133;
+
+#define HOSTBRIDGE_VIA_KX133 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_KX133)
+extern const unsigned int HOSTBRIDGE_VIA_KX133;
+
+#define HOSTBRIDGE_VIA_APOLLO_PRO_133A _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_APOLLO_PRO_133A)
+extern const unsigned int HOSTBRIDGE_VIA_APOLLO_PRO_133A;
+
+#define HOSTBRIDGE_VIA_APOLLO_PRO_266 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_APOLLO_PRO_266)
+extern const unsigned int HOSTBRIDGE_VIA_APOLLO_PRO_266;
+
+#define HOSTBRIDGE_AMD_760_MP _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_AMD_760_MP)
+extern const unsigned int HOSTBRIDGE_AMD_760_MP;
+
+#define HOSTBRIDGE_AMD_HAMMER _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_AMD_HAMMER)
+extern const unsigned int HOSTBRIDGE_AMD_HAMMER;
+
+#define HOSTBRIDGE_SERVERWORKS_HE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_HE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_HE;
+
+#define HOSTBRIDGE_SERVERWORKS_HE_B _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_HE_B)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_HE_B;
+
+#define HOSTBRIDGE_SERVERWORKS_LE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_LE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_LE;
+
+#define HOSTBRIDGE_SERVERWORKS_GC_HE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_GC_HE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_GC_HE;
+
+#define HOSTBRIDGE_SERVERWORKS_GC_LE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_GC_LE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_GC_LE;
+
+#define HOSTBRIDGE_SERVERWORKS_GC_WS _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_GC_WS)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_GC_WS;
+
+#define HOSTBRIDGE_SERVERWORKS_GC_SL _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_GC_SL)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_GC_SL;
+
+
+#define HOSTBRIDGE_WRITE_POSTING_DISABLED _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_WRITE_POSTING_DISABLED)
+extern const unsigned int HOSTBRIDGE_WRITE_POSTING_DISABLED;
+
+#define HOSTBRIDGE_WRITE_POSTING_ENABLED _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_WRITE_POSTING_ENABLED)
+extern const unsigned int HOSTBRIDGE_WRITE_POSTING_ENABLED;
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E D M A Q U E U E *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_PHDMA : Create physical DMA queue. Please note that this is an *
+ * priveleged operation. *
+ * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCICreateDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCICreateDMAQueue)
+DLL void SCICreateDMAQueue(sci_desc_t sd,
+ sci_dma_queue_t *dq,
+ unsigned int localAdapterNo,
+ unsigned int maxEntries,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E D M A Q U E U E *
+ * *
+ * Flags *
+ * None. * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Not allowed in this queue state. *
+ * *
+ *********************************************************************************/
+#define SCIRemoveDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveDMAQueue)
+DLL void SCIRemoveDMAQueue(sci_dma_queue_t dq,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I E N Q U E U E D M A T R A N S F E R *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_DMA_READ - The DMA will be remote --> local *
+ * (default is local --> remote) *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the offset and size is larger *
+ * than the segment size or larger than max *
+ * DMA size. *
+ * SCI_ERR_MAX_ENTRIES - The DMA queue is full *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_SEGMENT_NOT_PREPARED - The local segment has not been prepared for *
+ * access from the adapter associated with the *
+ * queue. *
+ * SCI_ERR_SEGMENT_NOT_CONNECTED - The remote segment is not connected through *
+ * the adapter associated with the queue. *
+ *********************************************************************************/
+#define SCIEnqueueDMATransfer _SISCI_EXPANDE_FUNCTION_NAME(SCIEnqueueDMATransfer)
+DLL sci_dma_queue_state_t SCIEnqueueDMATransfer(sci_dma_queue_t dq,
+ sci_local_segment_t localSegment,
+ sci_remote_segment_t remoteSegment,
+ unsigned int localOffset,
+ unsigned int remoteOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P O S T D M A Q U E U E *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_USE_CALLBACK - The end of the transfer will cause the callback *
+ * function to be invoked. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * *
+ *********************************************************************************/
+#define SCIPostDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIPostDMAQueue)
+DLL void SCIPostDMAQueue(sci_dma_queue_t dq,
+ sci_cb_dma_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I A B O R T D M A Q U E U E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * *
+ *********************************************************************************/
+#define SCIAbortDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIAbortDMAQueue)
+DLL void SCIAbortDMAQueue(sci_dma_queue_t dq,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I R E S E T D M A Q U E U E *
+ * *
+ * Flags *
+ * None. * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIResetDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIResetDMAQueue)
+DLL void SCIResetDMAQueue(sci_dma_queue_t dq,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I D M A Q U E U E S T A T E *
+ * *
+ *********************************************************************************/
+#define SCIDMAQueueState _SISCI_EXPANDE_FUNCTION_NAME(SCIDMAQueueState)
+DLL sci_dma_queue_state_t SCIDMAQueueState(sci_dma_queue_t dq);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R D M A Q U E U E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForDMAQueue)
+DLL sci_dma_queue_state_t SCIWaitForDMAQueue(sci_dma_queue_t dq,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P H D M A E N Q U E U E *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_READ *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIphDmaEnqueue _SISCI_EXPANDE_FUNCTION_NAME(SCIphDmaEnqueue)
+DLL void SCIphDmaEnqueue(sci_dma_queue_t dmaqueue,
+ unsigned int size,
+ sci_ioaddr_t localBusAddr,
+ unsigned int remote_nodeid,
+ unsigned int remote_highaddr,
+ unsigned int remote_lowaddr,
+ unsigned int flags,
+ sci_error_t *error);
+
+/*********************************************************************************
+ * *
+ * S C I P H D M A S T A R T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_WAIT *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_DMA_RESET *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIphDmaStart _SISCI_EXPANDE_FUNCTION_NAME(SCIphDmaStart)
+DLL sci_dma_queue_state_t SCIphDmaStart(sci_dma_queue_t dmaqueue,
+ sci_cb_dma_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E I N T E R R U P T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_FIXED_INTNO *
+ * SCI_FLAG_SHARED_INT *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_INTNO_USED - This interrupt number is already used. *
+ * *
+ *********************************************************************************/
+#define SCICreateInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCICreateInterrupt)
+DLL void SCICreateInterrupt(sci_desc_t sd,
+ sci_local_interrupt_t *interrupt,
+ unsigned int localAdapterNo,
+ unsigned int *interruptNo,
+ sci_cb_interrupt_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIRemoveInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveInterrupt)
+DLL void SCIRemoveInterrupt(sci_local_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_TIMEOUT - The function timed out after specified timeout value. *
+ * SCI_ERR_CANCELLED - The wait was interrupted by a call to *
+ * SCIRemoveInterrupt. *
+ * The handle is invalid when this error code is returned.*
+ * *
+ *********************************************************************************/
+#define SCIWaitForInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForInterrupt)
+DLL void SCIWaitForInterrupt(sci_local_interrupt_t interrupt,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C O N N E C T I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_SUCH_INTNO - No such interrupt number. *
+ * SCI_ERR_CONNECTION_REFUSED - Connection attempt refused by remote node. *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * *
+ *********************************************************************************/
+#define SCIConnectInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIConnectInterrupt)
+DLL void SCIConnectInterrupt(sci_desc_t sd,
+ sci_remote_interrupt_t *interrupt,
+ unsigned int nodeId,
+ unsigned int localAdapterNo,
+ unsigned int interruptNo,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I D I S C O N N E C T I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIDisconnectInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIDisconnectInterrupt)
+DLL void SCIDisconnectInterrupt(sci_remote_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I T R I G G E R I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCITriggerInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCITriggerInterrupt)
+DLL void SCITriggerInterrupt(sci_remote_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E G I S T E R I N T E R R U P T F L A G *
+ * *
+ * *
+ * This function register an "interrupt flag" that is identified as an unique *
+ * location within a local segment. If successful, the resulting interrupt *
+ * handle will have been associated with the specified local segment. *
+ * *
+ * It is up to the (remote) client(s) to set up an "interrupt mapping" for the *
+ * corresponding segment offset using either the *
+ * *
+ * - SCI_FLAG_CONDITIONAL_INTERRUPT_MAP *
+ * *
+ * or the *
+ * *
+ * - SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP *
+ * *
+ * option to "SCIMapRemoteSegment()". - I.e. after having established a *
+ * connection to the corresponding segment. A trigger operation can then *
+ * be implemented using a store operation via the relevant "interrupt map". *
+ * *
+ * *
+ * *
+ * *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_CONDITIONAL_INTERRUPT - Triggering is to take place using *
+ * "conditional interrupts". *
+ * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * None. *
+ * *
+ *********************************************************************************/
+#define SCIRegisterInterruptFlag _SISCI_EXPANDE_FUNCTION_NAME(SCIRegisterInterruptFlag)
+DLL void SCIRegisterInterruptFlag(
+ unsigned int localAdapterNo,
+ sci_local_interrupt_t *interrupt,
+ sci_local_segment_t segment,
+ unsigned int offset,
+ sci_cb_interrupt_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I E N A B L E C O N D I T I O N A L I N T E R R U P T *
+ * *
+ * *
+ * This function make sure that another HW interrupt will take place the next *
+ * time the corresponding interrupt flag is triggered by a *
+ * "conditional interrupt" operation. *
+ * *
+ * Default semantics: *
+ * *
+ * When successful, the client can rely on that the first subsequent trigger *
+ * operation will cause a HW interrupt and subsequently cause the client *
+ * handler function to be invoked. *
+ * *
+ * If an interrupt was triggered in parallell with the enable operation, then *
+ * the operation will fail (SCI_ERR_COND_INT_RACE_PROBLEM), and the client can *
+ * not rely on another trigger operation will lead to handler invocation. *
+ * Hence, any state checking normally associated with handling the *
+ * corresponding interrupt should take place before attempting to enable *
+ * again. *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_COND_INT_RACE_PROBLEM - The enable operation failed because an *
+ * incomming trigger operation happened *
+ * concurrently. *
+ * *
+ *********************************************************************************/
+#define SCIEnableConditionalInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIEnableConditionalInterrupt)
+DLL void SCIEnableConditionalInterrupt(
+ sci_local_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I D I S A B L E C O N D I T I O N A L I N T E R R U P T *
+ * *
+ * *
+ * Prevent subsequent "conditional interrupt"trigger operations for *
+ * the specified interupt flag from causing HW interrupt and handler *
+ * invocations. *
+ * *
+ * *
+ * Default semantics: *
+ * *
+ * If successful, no subsequent HW interrupts will take place, but handler *
+ * invocations that have already been scheduled may still take place. *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIDisableConditionalInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIDisableConditionalInterrupt)
+DLL void SCIDisableConditionalInterrupt(
+ sci_local_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T C O N D I T I O N A L I N T E R R U P T C O U N T E R *
+ * *
+ * *
+ * Returns a value that indicates the number of times this flag has *
+ * been trigged since the last time it was enabled or disabled. *
+ * Calling the SCIEnableConditionalInterrupt / SCIDisableConditionalInterrupt *
+ * functions will reset the counter value. *
+ * *
+ * Default semantics: *
+ * *
+ * If successful, the current trig count is returned in the *
+ * interruptTrigCounter parameter. *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OVERFLOW - The number of trig operations have exceeded the range *
+ * that can be counted. *
+ *********************************************************************************/
+#define SCIGetConditionalInterruptTrigCounter _SISCI_EXPANDE_FUNCTION_NAME(SCIGetConditionalInterruptTrigCounter)
+DLL void SCIGetConditionalInterruptTrigCounter(
+ sci_local_interrupt_t interrupt,
+ unsigned int *interruptTrigCounter,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I T R A N S F E R B L O C K *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger *
+ * than the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+#define SCITransferBlock _SISCI_EXPANDE_FUNCTION_NAME(SCITransferBlock)
+DLL void SCITransferBlock(sci_map_t sourceMap,
+ unsigned int sourceOffset,
+ sci_map_t destinationMap,
+ unsigned int destinationOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I T R A N S F E R B L O C K A S Y N C *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_BLOCK_READ *
+ * SCI_FLAG_USE_CALLBACK *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger than *
+ * the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required by *
+ * the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+#define SCITransferBlockAsync _SISCI_EXPANDE_FUNCTION_NAME(SCITransferBlockAsync)
+DLL void SCITransferBlockAsync(sci_map_t sourceMap,
+ unsigned int sourceOffset,
+ sci_map_t destinationMap,
+ unsigned int destinationOffset,
+ unsigned int size,
+ sci_block_transfer_t *block,
+ sci_cb_block_transfer_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R B L O C K T R A N S F E R *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForBlockTransfer _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForBlockTransfer)
+DLL void SCIWaitForBlockTransfer(sci_block_transfer_t block,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I A B O R T B L O C K T R A N S F E R *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * *
+ *********************************************************************************/
+#define SCIAbortBlockTransfer _SISCI_EXPANDE_FUNCTION_NAME(SCIAbortBlockTransfer)
+DLL void SCIAbortBlockTransfer(sci_block_transfer_t block,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M E M C P Y *
+ * *
+ * Flags: *
+ * SCI_FLAG_BLOCK_READ *
+ * SCI_FLAG_ERROR_CHECK *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger *
+ * than the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+
+#define SCIMemCpy _SISCI_EXPANDE_FUNCTION_NAME(SCIMemCpy)
+DLL void SCIMemCpy(sci_sequence_t sequence,
+ void *memAddr,
+ sci_map_t remoteMap,
+ unsigned int remoteOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M E M C O P Y *
+ * *
+ * Flags: *
+ * SCI_FLAG_BLOCK_READ *
+ * SCI_FLAG_ERROR_CHECK *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger *
+ * than the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+
+
+#define SCIMemCopy _SISCI_EXPANDE_FUNCTION_NAME(SCIMemCopy)
+DLL void SCIMemCopy(void *memAddr,
+ sci_map_t remoteMap,
+ unsigned int remoteOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E G I S T E R S E G M E N T M E M O R Y *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required by *
+ * the implementation. *
+ * SCI_ERR_ILLEGAL_ADDRESS - Illegal address. *
+ * SCI_ERR_OUT_OF_RANGE - Size is larger than the maximum size for the *
+ * local segment. *
+ * *
+ *********************************************************************************/
+#define SCIRegisterSegmentMemory _SISCI_EXPANDE_FUNCTION_NAME(SCIRegisterSegmentMemory)
+DLL void SCIRegisterSegmentMemory(void *address,
+ unsigned int size,
+ sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C O N N E C T S C I S P A C E *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_CONNECTION_REFUSED - Connection attempt refused by remote node. *
+ * *
+ *********************************************************************************/
+#define SCIConnectSCISpace _SISCI_EXPANDE_FUNCTION_NAME(SCIConnectSCISpace)
+DLL void SCIConnectSCISpace(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ sci_remote_segment_t *segment,
+ sci_address_t address,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*
+ * =====================================================================================
+ *
+ * S C I A T T A C H L O C A L S E G M E N T
+ * Description:
+ *
+ * SCIAttachLocalSegment() permits an application to "attach" to an already existing
+ * local segment, implying that two or more application want
+ * share the same local segment. The prerequest, is that the
+ * application which originally created the segment ("owner") has
+ * preformed a SCIShareSegment() in order to mark the segment
+ * "shareable".
+ *
+ *
+ * Flags:
+ *
+ * SCI_FLAG_USE_CALLBACK - The callback function will be invoked for events
+ * on this segment.
+ *
+ *
+ * Specific error codes for this function:
+ *
+ * SCI_ERR_ACCESS - No such shared segment
+ * SCI_ERR_NO_SUCH_SEGMENT - No such segment
+ * Note: Current implenentation will return SCI_ERR_ACCESS for both cases. This will
+ * change from next release. Application should handle both cases.
+ *
+ * =====================================================================================
+ */
+#define SCIAttachLocalSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIAttachLocalSegment)
+
+DLL void
+SCIAttachLocalSegment(sci_desc_t sd,
+ sci_local_segment_t *segment,
+ unsigned int segmentId,
+ unsigned int *size,
+ sci_cb_local_segment_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+/*
+ * =====================================================================================
+ *
+ * S C I S H A R E S E G M E N T
+ *
+ * Description:
+ *
+ * SCIShareSegment() permits other application to "attach" to an already existing
+ * local segment, implying that two or more application want
+ * share the same local segment. The prerequest, is that the
+ * application which originally created the segment ("owner") has
+ * preformed a SCIShareSegment() in order to mark the segment
+ * "shareable".
+ *
+ *
+ * Flags:
+ * none
+ *
+ * Specific error codes for this function:
+ *
+ *
+ *
+ * =====================================================================================
+ */
+#define SCIShareSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIShareSegment)
+
+DLL void
+SCIShareSegment(sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I F L U S H *
+ * *
+ * This function will flush the CPU buffers and the PSB buffers. *
+ * *
+ * Flags *
+ * SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY : *
+ * Only flush CPU buffers ( Write combining *
+ * etc buffers). *
+ * *
+ *********************************************************************************/
+
+#define SCIFlush _SISCI_EXPANDE_FUNCTION_NAME(SCIFlush)
+DLL void SCIFlush(sci_sequence_t sequence,
+ unsigned int flags);
+
+#if defined(CPLUSPLUS) || defined(__cplusplus)
+}
+#endif
+
+
+#endif
+
+
+
+
+
+
+
+
+
+
+
diff --git a/ndb/src/external/LINUX.x86/sci/include/sisci_demolib.h b/ndb/src/external/LINUX.x86/sci/include/sisci_demolib.h
new file mode 100644
index 00000000000..4284fc5585c
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/sisci_demolib.h
@@ -0,0 +1,226 @@
+/* $Id: sisci_demolib.h,v 1.1 2002/12/13 12:17:21 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+#ifndef _SISCI_DEMOLIB_H
+#define _SISCI_DEMOLIB_H
+
+
+#if defined(_REENTRANT)
+
+#define _SISCI_DEMOLIB_EXPAND_NAME(name) _SISCI_DEMOLIB_MT_ ## name
+
+#else
+
+#define _SISCI_DEMOLIB_EXPAND_NAME(name) _SISCI_DEMOLIB_ST_ ## name
+
+#endif
+
+/*********************************************************************************/
+/* Q U E R Y A D A P T E R */
+/* */
+/*********************************************************************************/
+
+#define QueryAdapter _SISCI_DEMOLIB_EXPAND_NAME(QueryAdapter)
+
+sci_error_t QueryAdapter(
+ unsigned int subcommand,
+ unsigned int localAdapterNo,
+ unsigned int portNo,
+ unsigned int *data);
+
+
+/*********************************************************************************/
+/* Q U E R Y S Y S T E M */
+/* */
+/*********************************************************************************/
+
+#define QuerySystem _SISCI_DEMOLIB_EXPAND_NAME(QuerySystem)
+
+sci_error_t QuerySystem(
+ unsigned int subcommand,
+ unsigned int *data);
+
+
+/*********************************************************************************/
+/* D E T E C T F I R S T A D A P T E R C A R D */
+/* */
+/*********************************************************************************/
+
+#define DetectFirstAdapterCard _SISCI_DEMOLIB_EXPAND_NAME(DetectFirstAdapterCard)
+
+sci_error_t DetectFirstAdapterCard(
+ unsigned int *localAdapterNo,
+ unsigned int *localNodeId);
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R T Y P E */
+/* */
+/*********************************************************************************/
+
+#define GetAdapterType _SISCI_DEMOLIB_EXPAND_NAME(GetAdapterType)
+
+sci_error_t GetAdapterType(unsigned int localAdapterNo,
+ unsigned int *adapterType);
+
+
+/*********************************************************************************/
+/* G E T L O C A L N O D E I D */
+/* */
+/*********************************************************************************/
+
+#define GetLocalNodeId _SISCI_DEMOLIB_EXPAND_NAME(GetLocalNodeId)
+
+sci_error_t GetLocalNodeId(
+ unsigned int localAdapterNo,
+ unsigned int *localNodeId);
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R S E R I A L N U M B E R */
+/* */
+/*********************************************************************************/
+
+#define GetAdapterSerialNumber _SISCI_DEMOLIB_EXPAND_NAME(GetAdapterSerialNumber)
+
+sci_error_t GetAdapterSerialNumber(
+ unsigned int localAdapterNo,
+ unsigned int *serialNo);
+
+
+
+/*********************************************************************************/
+/* G E T H O S T B R I D G E T Y P E */
+/* */
+/*********************************************************************************/
+
+#define GetHostbridgeType _SISCI_DEMOLIB_EXPAND_NAME(GetHostbridgeType)
+
+sci_error_t GetHostbridgeType(unsigned int *hostbridgeType);
+
+
+
+/*********************************************************************************/
+/* P R I N T H O S T B R I D G E T Y P E */
+/* */
+/*********************************************************************************/
+
+#define PrintHostbridgeType _SISCI_DEMOLIB_EXPAND_NAME(PrintHostbridgeType)
+
+void PrintHostbridgeType(unsigned int hostbridge);
+
+
+
+/*********************************************************************************/
+/* G E T A P I V E R S I O N S T R I N G */
+/* */
+/*********************************************************************************/
+
+#define GetAPIVersionString _SISCI_DEMOLIB_EXPAND_NAME(GetAPIVersionString)
+
+sci_error_t GetAPIVersionString(char str[], unsigned int strLength);
+
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R I O B U S F R E Q U E N C Y */
+/* */
+/*********************************************************************************/
+
+sci_error_t GetAdapterIoBusFrequency(unsigned int localAdapterNo,
+ unsigned int *ioBusFrequency);
+
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R S C I L I N K F R E Q U E N C Y */
+/* */
+/*********************************************************************************/
+
+sci_error_t GetAdapterSciLinkFrequency(unsigned int localAdapterNo,
+ unsigned int *sciLinkFrequency);
+
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R B L I N K F R E Q U E N C Y */
+/* */
+/*********************************************************************************/
+
+sci_error_t GetAdapterBlinkFrequency(unsigned int localAdapterNo,
+ unsigned int *bLinkFrequency);
+
+
+/*********************************************************************************/
+/* S E N D I N T E R R U P T */
+/* */
+/*********************************************************************************/
+
+#define SendInterrupt _SISCI_DEMOLIB_EXPAND_NAME(SendInterrupt)
+
+sci_error_t SendInterrupt(
+ sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int localNodeId,
+ unsigned int remoteNodeId,
+ unsigned int interruptNo);
+
+
+/*********************************************************************************/
+/* R E C E I V E I N T E R R U P T */
+/* */
+/*********************************************************************************/
+
+#define ReceiveInterrupt _SISCI_DEMOLIB_EXPAND_NAME(ReceiveInterrupt)
+
+sci_error_t ReceiveInterrupt(
+ sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int localNodeId,
+ unsigned int interruptNo);
+
+
+/*********************************************************************************/
+/* E N D I A N S W A P */
+/* */
+/*********************************************************************************/
+
+#define EndianSwap _SISCI_DEMOLIB_EXPAND_NAME(EndianSwap)
+
+unsigned int EndianSwap (unsigned int value);
+
+
+/*********************************************************************************/
+/* S L E E P M I L L I S E C O N D S */
+/* */
+/*********************************************************************************/
+
+#define SleepMilliseconds _SISCI_DEMOLIB_EXPAND_NAME(SleepMilliseconds)
+
+void SleepMilliseconds(int milliseconds);
+
+
+
+
+#endif
diff --git a/ndb/src/external/LINUX.x86/sci/include/sisci_error.h b/ndb/src/external/LINUX.x86/sci/include/sisci_error.h
new file mode 100644
index 00000000000..c53fe8ad5d9
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/sisci_error.h
@@ -0,0 +1,89 @@
+/* $Id: sisci_error.h,v 1.1 2002/12/13 12:17:21 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+
+
+#ifndef _SISCI_ERROR_H_
+#define _SISCI_ERROR_H_
+
+
+/* SCI Error return values always have 30 bit set */
+#define SCI_ERR_MASK 0x40000000
+#define SCI_ERR_REMOTE_MASK 0x01 /* Remote errors should have bit 0 set */
+
+#define SCI_ERR(u) ((unsigned32)(u)&0x7FFFFFFF )
+
+/* Error codes */
+typedef enum {
+ SCI_ERR_OK = 0x000,
+
+
+ SCI_ERR_BUSY = (0x900 | SCI_ERR_MASK),
+ SCI_ERR_FLAG_NOT_IMPLEMENTED = (0x901 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_FLAG = (0x902 | SCI_ERR_MASK),
+ SCI_ERR_NOSPC = (0x904 | SCI_ERR_MASK),
+ SCI_ERR_API_NOSPC = (0x905 | SCI_ERR_MASK),
+ SCI_ERR_HW_NOSPC = (0x906 | SCI_ERR_MASK),
+ SCI_ERR_NOT_IMPLEMENTED = (0x907 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_ADAPTERNO = (0x908 | SCI_ERR_MASK),
+ SCI_ERR_NO_SUCH_ADAPTERNO = (0x909 | SCI_ERR_MASK),
+ SCI_ERR_TIMEOUT = (0x90A | SCI_ERR_MASK),
+ SCI_ERR_OUT_OF_RANGE = (0x90B | SCI_ERR_MASK),
+ SCI_ERR_NO_SUCH_SEGMENT = (0x90C | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_NODEID = (0x90D | SCI_ERR_MASK),
+ SCI_ERR_CONNECTION_REFUSED = (0x90E | SCI_ERR_MASK),
+ SCI_ERR_SEGMENT_NOT_CONNECTED = (0x90F | SCI_ERR_MASK),
+ SCI_ERR_SIZE_ALIGNMENT = (0x910 | SCI_ERR_MASK),
+ SCI_ERR_OFFSET_ALIGNMENT = (0x911 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_PARAMETER = (0x912 | SCI_ERR_MASK),
+ SCI_ERR_MAX_ENTRIES = (0x913 | SCI_ERR_MASK),
+ SCI_ERR_SEGMENT_NOT_PREPARED = (0x914 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_ADDRESS = (0x915 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_OPERATION = (0x916 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_QUERY = (0x917 | SCI_ERR_MASK),
+ SCI_ERR_SEGMENTID_USED = (0x918 | SCI_ERR_MASK),
+ SCI_ERR_SYSTEM = (0x919 | SCI_ERR_MASK),
+ SCI_ERR_CANCELLED = (0x91A | SCI_ERR_MASK),
+ SCI_ERR_NOT_CONNECTED = (0x91B | SCI_ERR_MASK),
+ SCI_ERR_NOT_AVAILABLE = (0x91C | SCI_ERR_MASK),
+ SCI_ERR_INCONSISTENT_VERSIONS = (0x91D | SCI_ERR_MASK),
+ SCI_ERR_COND_INT_RACE_PROBLEM = (0x91E | SCI_ERR_MASK),
+ SCI_ERR_OVERFLOW = (0x91F | SCI_ERR_MASK),
+ SCI_ERR_NOT_INITIALIZED = (0x920 | SCI_ERR_MASK),
+
+ SCI_ERR_ACCESS = (0x921 | SCI_ERR_MASK),
+
+ SCI_ERR_NO_SUCH_NODEID = (0xA00 | SCI_ERR_MASK),
+ SCI_ERR_NODE_NOT_RESPONDING = (0xA02 | SCI_ERR_MASK),
+ SCI_ERR_NO_REMOTE_LINK_ACCESS = (0xA04 | SCI_ERR_MASK),
+ SCI_ERR_NO_LINK_ACCESS = (0xA05 | SCI_ERR_MASK),
+ SCI_ERR_TRANSFER_FAILED = (0xA06 | SCI_ERR_MASK)
+} sci_error_t;
+
+
+#endif /* _SCI_ERROR_H_ */
+
+
+
diff --git a/ndb/src/external/LINUX.x86/sci/include/sisci_types.h b/ndb/src/external/LINUX.x86/sci/include/sisci_types.h
new file mode 100644
index 00000000000..f4125fdec69
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/sisci_types.h
@@ -0,0 +1,133 @@
+/* $Id: sisci_types.h,v 1.1 2002/12/13 12:17:21 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+#ifndef _SISCI_TYPES_H
+#define _SISCI_TYPES_H
+
+#include "sisci_error.h"
+
+#ifndef IN
+#define IN
+#endif
+
+#ifndef OUT
+#define OUT
+#endif
+
+#ifndef IN_OUT
+#define IN_OUT
+#endif
+
+/* Opaque data types for descriptors/handles */
+typedef struct sci_desc *sci_desc_t;
+typedef struct sci_local_segment *sci_local_segment_t;
+typedef struct sci_remote_segment *sci_remote_segment_t;
+
+typedef struct sci_map *sci_map_t;
+typedef struct sci_sequence *sci_sequence_t;
+#ifndef KERNEL
+typedef struct sci_dma_queue *sci_dma_queue_t;
+#endif
+typedef struct sci_remote_interrupt *sci_remote_interrupt_t;
+typedef struct sci_local_interrupt *sci_local_interrupt_t;
+typedef struct sci_block_transfer *sci_block_transfer_t;
+
+/*
+ * Constants defining reasons for segment callbacks:
+ */
+
+typedef enum {
+ SCI_CB_CONNECT = 1,
+ SCI_CB_DISCONNECT,
+ SCI_CB_NOT_OPERATIONAL,
+ SCI_CB_OPERATIONAL,
+ SCI_CB_LOST
+} sci_segment_cb_reason_t;
+
+#define MAX_CB_REASON SCI_CB_LOST
+
+/* dma_queue_states is identical to the dma_queue_state_t in genif.h, they must be consistent.*/
+typedef enum {
+ SCI_DMAQUEUE_IDLE,
+ SCI_DMAQUEUE_GATHER,
+ SCI_DMAQUEUE_POSTED,
+ SCI_DMAQUEUE_DONE,
+ SCI_DMAQUEUE_ABORTED,
+ SCI_DMAQUEUE_ERROR
+} sci_dma_queue_state_t;
+
+
+typedef enum {
+ SCI_SEQ_OK,
+ SCI_SEQ_RETRIABLE,
+ SCI_SEQ_NOT_RETRIABLE,
+ SCI_SEQ_PENDING
+} sci_sequence_status_t;
+
+
+typedef struct {
+ unsigned short nodeId; /* SCI Address bit 63 - 48 */
+ unsigned short offsHi; /* SCI Address bit 47 - 32 */
+ unsigned int offsLo; /* SCI Address bit 31 - 0 */
+} sci_address_t;
+
+
+typedef unsigned int sci_ioaddr_t;
+
+typedef enum {
+ SCI_CALLBACK_CANCEL = 1,
+ SCI_CALLBACK_CONTINUE
+} sci_callback_action_t;
+
+#ifndef KERNEL
+typedef sci_callback_action_t (*sci_cb_local_segment_t)(void *arg,
+ sci_local_segment_t segment,
+ sci_segment_cb_reason_t reason,
+ unsigned int nodeId,
+ unsigned int localAdapterNo,
+ sci_error_t error);
+
+typedef sci_callback_action_t (*sci_cb_remote_segment_t)(void *arg,
+ sci_remote_segment_t segment,
+ sci_segment_cb_reason_t reason,
+ sci_error_t status);
+
+
+typedef sci_callback_action_t (*sci_cb_dma_t)(void IN *arg,
+ sci_dma_queue_t queue,
+ sci_error_t status);
+
+
+typedef int (*sci_cb_block_transfer_t)(void *arg,
+ sci_block_transfer_t block,
+ sci_error_t status);
+
+
+typedef sci_callback_action_t (*sci_cb_interrupt_t)(void *arg,
+ sci_local_interrupt_t interrupt,
+ sci_error_t status);
+
+#endif /* KERNEL */
+#endif
diff --git a/ndb/src/external/LINUX.x86/sci/include/sisci_version.h b/ndb/src/external/LINUX.x86/sci/include/sisci_version.h
new file mode 100644
index 00000000000..f1807c33aa5
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/sisci_version.h
@@ -0,0 +1,91 @@
+/* $Id: sisci_version.h,v 1.1 2002/12/13 12:17:21 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU Lesser General Public License as published by *
+ * the Free Software Foundation; either version 2.1 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 Lesser 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+#ifndef SISCI_VERSION_H
+#define SISCI_VERSION_H
+
+
+#define SISCI_API_VER_MAJOR 0x01
+#define SISCI_API_VER_MAJORC "1"
+
+#define SISCI_API_VER_MINOR 0x010
+#define SISCI_API_VER_MINORC "10"
+#define SISCI_API_VER_MICRO 0x005
+#define SISCI_API_VER_MICROC "5"
+
+#define SISCI_SIGN_VERSION_MASK 0xfffff000 /* used to mask off API_VER_MICRO */
+
+#define SISCI_API_VERSION (SISCI_API_VER_MAJOR << 24 | SISCI_API_VER_MINOR << 12 | SISCI_API_VER_MICRO)
+
+/* the rules are:
+ *
+ * Changes in API_VER_MICRO should be binary compatible, New flags, functions added. No changes to user code
+ * required if new features is not needed.
+ *
+ * Changes in API_VER_MINOR requires recompilation of user code.
+ *
+ * Changes in the API_VER_MAJOR will most likely require changes to user code. This should not happen very
+ * often...
+ *
+ */
+
+#ifndef BUILD_DATE
+#define BUILD_DATE __DATE__
+#endif
+
+#ifndef BUILD_NAME
+#define BUILD_NAME ""
+#endif
+
+#define API_VERSION "SISCI API version " SISCI_API_VER_MAJORC "." SISCI_API_VER_MINORC "."SISCI_API_VER_MICROC " ( "BUILD_NAME" "BUILD_DATE" )"
+
+#endif
+
+
+/* Version info: */
+/* */
+/* 1.5.2 First SISCI version */
+/* 1.5.3 Some bug fixes */
+/* 1.5.4 Some bug fixes */
+/* 1.5.5 No release */
+/* 1.5.6 Lock flag implemented in function SCIConnectSegment */
+/* 1.5.7 Expanded query functionality */
+/* 1.5.8 Updated error checking (sequence) functionality for D320 */
+/* 1.6.0 Updated error checking (sequence) D320 and IRM 1.9 support */
+/* 1.9.0 Ported to Solaris_sparc, Solaris_x86 and Linux. IRM 1.9. */
+/* 1.9.1 Some bug fixes */
+/* 1.9.2 Added more adapter queries */
+/* 1.9.3 Bug fix in SCIMapLocalSegment and SCIMapRemoteSegment */
+/* 1.9.4 NT Release Developers Kit 2.40 */
+/* 1.9.5 Added flush after data transfer in SCIMemCopy() */
+/* 1.9.5 NT Release Developers Kit 2.44 */
+/* 1.10.0:
+ * New SCIInitialize(), SCITerminate() functions.
+ * Support for D330
+ *
+ *
+ */
+
+
diff --git a/ndb/src/external/LINUX.x86/sci/include/version.h b/ndb/src/external/LINUX.x86/sci/include/version.h
new file mode 100644
index 00000000000..a0e1fa6c5cd
--- /dev/null
+++ b/ndb/src/external/LINUX.x86/sci/include/version.h
@@ -0,0 +1,25 @@
+#ifndef _VERSION_H
+#define _VERSION_H
+
+
+/*
+#define DEMO_VER_MAJOR "1"
+#define DEMO_VER_MINOR "5"
+#define DEMO_VER_MICRO "0"
+*/
+
+#ifndef BUILD_DATE
+#define BUILD_DATE __DATE__
+#endif
+
+#ifndef BUILD_NAME
+#define BUILD_NAME ""
+#endif
+
+/*
+#define DEMO_VERSION "version " DEMO_VER_MAJOR "." DEMO_VER_MINOR "."DEMO_VER_MICRO " ("BUILD_NAME" "BUILD_DATE" )"
+*/
+#define DEMO_VERSION " ("BUILD_NAME" "BUILD_DATE" )"
+
+
+#endif
diff --git a/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_api.h b/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_api.h
new file mode 100644
index 00000000000..d02d3aafe7f
--- /dev/null
+++ b/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_api.h
@@ -0,0 +1,2148 @@
+/* $Id: sisci_api.h,v 1.1 2002/12/13 12:17:22 hin Exp $ */
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2001 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU Lesser General Public License as published by *
+ * the Free Software Foundation; either version 2.1 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 Lesser 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+#ifndef _SISCI_API_H
+#define _SISCI_API_H
+
+#include "sisci_types.h"
+#include "sisci_error.h"
+
+
+#ifdef WIN32
+#ifdef API_DLL
+#define DLL __declspec(dllexport)
+#elif CLIENT_DLL
+#define DLL __declspec(dllimport)
+#endif
+#endif /* WIN32 */
+
+
+#ifndef DLL
+#define DLL
+#endif
+
+#if defined(_REENTRANT)
+#define _SISCI_EXPANDE_FUNCTION_NAME(name) _SISCI_PUBLIC_FUNC_MT_ ## name
+#define _SISCI_EXPANDE_VARIABLE_NAME(name) _SISCI_PUBLIC_VAR_MT_ ## name
+#else
+#define _SISCI_EXPANDE_FUNCTION_NAME(name) _SISCI_PUBLIC_FUNC_ST_ ## name
+#define _SISCI_EXPANDE_VARIABLE_NAME(name) _SISCI_PUBLIC_VAR_ST_ ## name
+#endif
+#define _SISCI_EXPANDE_CONSTANT_NAME(name) _SISCI_PUBLIC_CONST_ ## name
+
+#if defined(CPLUSPLUS) || defined(__cplusplus)
+extern "C" {
+#endif
+
+
+/*********************************************************************************/
+/* FLAG VALUES */
+/*********************************************************************************/
+
+#define SCI_FLAG_FIXED_INTNO _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FIXED_INTNO)
+extern const unsigned int SCI_FLAG_FIXED_INTNO;
+
+#define SCI_FLAG_SHARED_INT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_SHARED_INT)
+extern const unsigned int SCI_FLAG_SHARED_INT;
+
+#define SCI_FLAG_FIXED_MAP_ADDR _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FIXED_MAP_ADDR)
+extern const unsigned int SCI_FLAG_FIXED_MAP_ADDR;
+
+#define SCI_FLAG_READONLY_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_READONLY_MAP)
+extern const unsigned int SCI_FLAG_READONLY_MAP;
+
+#define SCI_FLAG_USE_CALLBACK _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_USE_CALLBACK)
+extern const unsigned int SCI_FLAG_USE_CALLBACK;
+
+#define SCI_FLAG_BLOCK_READ _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_BLOCK_READ)
+extern const unsigned int SCI_FLAG_BLOCK_READ;
+
+#define SCI_FLAG_THREAD_SAFE _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_THREAD_SAFE)
+extern const unsigned int SCI_FLAG_THREAD_SAFE;
+
+#define SCI_FLAG_ASYNCHRONOUS_CONNECT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_ASYNCHRONOUS_CONNECT)
+extern const unsigned int SCI_FLAG_ASYNCHRONOUS_CONNECT;
+
+#define SCI_FLAG_EMPTY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_EMPTY)
+extern const unsigned int SCI_FLAG_EMPTY;
+
+#define SCI_FLAG_PRIVATE _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_PRIVATE)
+extern const unsigned int SCI_FLAG_PRIVATE;
+
+#define SCI_FLAG_FORCE_DISCONNECT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FORCE_DISCONNECT)
+extern const unsigned int SCI_FLAG_FORCE_DISCONNECT;
+
+#define SCI_FLAG_NOTIFY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NOTIFY)
+extern const unsigned int SCI_FLAG_NOTIFY;
+
+#define SCI_FLAG_DMA_READ _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_READ)
+extern const unsigned int SCI_FLAG_DMA_READ;
+
+#define SCI_FLAG_DMA_POST _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_POST)
+extern const unsigned int SCI_FLAG_DMA_POST;
+
+#define SCI_FLAG_DMA_WAIT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_WAIT)
+extern const unsigned int SCI_FLAG_DMA_WAIT;
+
+#define SCI_FLAG_DMA_RESET _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_RESET)
+extern const unsigned int SCI_FLAG_DMA_RESET;
+
+#define SCI_FLAG_NO_FLUSH _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NO_FLUSH)
+extern const unsigned int SCI_FLAG_NO_FLUSH;
+
+#define SCI_FLAG_NO_STORE_BARRIER _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NO_STORE_BARRIER)
+extern const unsigned int SCI_FLAG_NO_STORE_BARRIER;
+
+#define SCI_FLAG_FAST_BARRIER _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FAST_BARRIER)
+extern const unsigned int SCI_FLAG_FAST_BARRIER;
+
+#define SCI_FLAG_ERROR_CHECK _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_ERROR_CHECK)
+extern const unsigned int SCI_FLAG_ERROR_CHECK;
+
+#define SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY)
+extern const unsigned int SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY;
+
+/* the FLUSH_CPU_BUFFERS_ONLY flag is for backwards compabillity only and should never be used */
+#define FLUSH_CPU_BUFFERS_ONLY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY)
+
+#define SCI_FLAG_LOCK_OPERATION _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_LOCK_OPERATION)
+extern const unsigned int SCI_FLAG_LOCK_OPERATION;
+
+#define SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP)
+extern const unsigned int SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP;
+
+#define SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP)
+extern const unsigned int SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP;
+
+#define SCI_FLAG_IO_MAP_IOSPACE _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_IO_MAP_IOSPACE)
+extern const unsigned int SCI_FLAG_IO_MAP_IOSPACE;
+
+#define SCI_FLAG_DMOVE_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMOVE_MAP)
+extern const unsigned int SCI_FLAG_DMOVE_MAP;
+
+#define SCI_FLAG_WRITES_DISABLE_GATHER_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_WRITES_DISABLE_GATHER_MAP)
+extern const unsigned int SCI_FLAG_WRITES_DISABLE_GATHER_MAP;
+
+#define SCI_FLAG_DISABLE_128_BYTES_PACKETS _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DISABLE_128_BYTES_PACKETS)
+extern const unsigned int SCI_FLAG_DISABLE_128_BYTES_PACKETS;
+
+#define SCI_FLAG_DMA_SOURCE_ONLY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_SOURCE_ONLY)
+extern const unsigned int SCI_FLAG_DMA_SOURCE_ONLY;
+
+#define SCI_FLAG_CONDITIONAL_INTERRUPT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_CONDITIONAL_INTERRUPT)
+extern const unsigned int SCI_FLAG_CONDITIONAL_INTERRUPT;
+
+#define SCI_FLAG_CONDITIONAL_INTERRUPT_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_CONDITIONAL_INTERRUPT_MAP)
+extern const unsigned int SCI_FLAG_CONDITIONAL_INTERRUPT_MAP;
+
+#define SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP)
+extern const unsigned int SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP;
+
+#define SCI_FLAG_NO_MEMORY_LOOPBACK_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NO_MEMORY_LOOPBACK_MAP)
+extern const unsigned int SCI_FLAG_NO_MEMORY_LOOPBACK_MAP;
+
+#if defined(OS_IS_LYNXOS) || defined(OS_IS_VXWORKS)
+#define SCI_FLAG_WRITE_BACK_CACHE_MAP _SISCI_EXPANDE_CONSTANT_NAME(WRITE_BACK_CACHE_MAP)
+extern const unsigned int SCI_FLAG_WRITE_BACK_CACHE_MAP;
+#endif
+
+#define SCI_FLAG_DMA_PHDMA _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_PHDMA)
+extern const unsigned int SCI_FLAG_DMA_PHDMA;
+
+/*********************************************************************************/
+/* GENERAL VALUES */
+/*********************************************************************************/
+#define SCI_LOCAL_HOST _SISCI_EXPANDE_CONSTANT_NAME(SCI_LOCAL_HOST)
+extern const unsigned int SCI_LOCAL_HOST;
+
+#define SCI_INFINITE_TIMEOUT _SISCI_EXPANDE_CONSTANT_NAME(SCI_INFINITE_TIMEOUT)
+extern const unsigned int SCI_INFINITE_TIMEOUT;
+
+/*********************************************************************************/
+/* GENERAL ERROR CODES */
+/* */
+/* SCI_ERR_ILLEGAL_FLAG - Illegal flag value. */
+/* SCI_ERR_FLAG_NOT_IMPLEMENTED - Flag legal but flag feature not implemented. */
+/* SCI_ERR_NOT_IMPLEMENTED - Function not implemented. */
+/* SCI_ERR_SYSTEM - A system error. Check errno. */
+/* SCI_ERR_NOSPC - Unable to allocate OS resources. */
+/* SCI_ERR_API_NOSPC - Unable to allocate API resources. */
+/* SCI_ERR_HW_NOSPC - Unable to allocate HW resources (Hardware) */
+/* */
+/*********************************************************************************/
+
+
+/*********************************************************************************/
+/* GENERAL "ADAPTER" ERROR CODES */
+/* */
+/* SCI_ERR_NO_SUCH_ADAPTERNO - Adapter number is legal but does not exist. */
+/* SCI_ERR_ILLEGAL_ADAPTERNO - Illegal local adapter number (i.e. outside */
+/* legal range). */
+/* */
+/*********************************************************************************/
+
+
+/*********************************************************************************/
+/* GENERAL "NODEID" ERROR CODES */
+/* */
+/* SCI_ERR_NO_SUCH_NODEID - The remote adapter identified by nodeId does */
+/* not respond, but the intermediate link(s) */
+/* seem(s) to be operational. */
+/* SCI_ERR_ILLEGAL_NODEID - Illegal NodeId. */
+/* */
+/*********************************************************************************/
+
+
+
+/*********************************************************************************
+ * *
+ * S C I I N I T I A L I Z E *
+ * *
+ * This function initializes the SISCI library. *
+ * The function must be called before SCIOpen(). *
+ * *
+ * Flags: *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * None *
+ * *
+ *********************************************************************************/
+#define SCIInitialize _SISCI_EXPANDE_FUNCTION_NAME(SCIInitialize)
+DLL void SCIInitialize(unsigned int flags,
+ sci_error_t *error);
+#if 0
+unsigned int __Internal_SISCI_version_var;
+#endif
+
+/*********************************************************************************
+ * *
+ * S C I T E R M I N A T E *
+ * *
+ * This function terminates the SISCI library. *
+ * The function must be called after SCIClose(). *
+ * *
+ * *
+ *********************************************************************************/
+#define SCITerminate _SISCI_EXPANDE_FUNCTION_NAME(SCITerminate)
+DLL void SCITerminate(void);
+
+/*********************************************************************************
+ * *
+ * S C I O P E N *
+ * *
+ * *
+ * Opens a SCI virtual device. *
+ * Caller must supply a pointer to a variable of type sci_desc_t to be *
+ * initialized. *
+ * *
+ * Flags *
+ * SCI_FLAG_THREAD_SAFE - Operations on resources associated with this *
+ * descriptor will be performed in a multithread-safe *
+ * manner. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_INCONSISTENT_VERSIONS - Inconsistency between the SISCI library *
+ * and the SISCI driver versions. *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIOpen _SISCI_EXPANDE_FUNCTION_NAME(SCIOpen)
+DLL void SCIOpen(sci_desc_t *sd,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C L O S E *
+ * *
+ * This function closes an open SCI virtual device. *
+ * *
+ * Flags: *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - All resources are not deallocated. *
+ * *
+ *********************************************************************************/
+#define SCIClose _SISCI_EXPANDE_FUNCTION_NAME(SCIClose)
+DLL void SCIClose(sci_desc_t sd,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C O N N E C T S E G M E N T *
+ * *
+ * Connects to a remote shared memory segment located at <nodeId> with the *
+ * identifier <segmentId>. *
+ * The user may then call SCIMapRemoteSegment() to map shared memory *
+ * into user space. *
+ * *
+ * Flags *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_ASYNCHRONOUS_CONNECT *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_SUCH_SEGMENT - Could not find the remote segment with the *
+ * given segmentId. *
+ * SCI_ERR_CONNECTION_REFUSED - Connection attempt refused by remote node. *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCIConnectSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIConnectSegment)
+DLL void SCIConnectSegment(sci_desc_t sd,
+ sci_remote_segment_t *segment,
+ unsigned int nodeId,
+ unsigned int segmentId,
+ unsigned int localAdapterNo,
+ sci_cb_remote_segment_t callback,
+ void *callbackArg,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I D I S C O N N E C T S E G M E N T *
+ * *
+ * Disconnects from the give mapped shared memory segment *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - The segment is currently mapped or in use. *
+ * *
+ *********************************************************************************/
+#define SCIDisconnectSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIDisconnectSegment)
+DLL void SCIDisconnectSegment(sci_remote_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T R E M O T E S E G M E N T S I Z E *
+ * *
+ *********************************************************************************/
+#define SCIGetRemoteSegmentSize _SISCI_EXPANDE_FUNCTION_NAME(SCIGetRemoteSegmentSize)
+DLL unsigned int SCIGetRemoteSegmentSize(sci_remote_segment_t segment);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R R E M O T E S E G M E N T E V E N T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation. *
+ * SCI_ERR_CANCELLED - The wait operation has been cancelled du *
+ * to a SCIDisconnectSegment() on the same *
+ * handle. The handle is invalid when this *
+ * error is returned. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForRemoteSegmentEvent _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForRemoteSegmentEvent)
+DLL sci_segment_cb_reason_t SCIWaitForRemoteSegmentEvent(
+ sci_remote_segment_t segment,
+ sci_error_t *status,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M A P R E M O T E S E G M E N T *
+ * *
+ * This function is used to include a shared memory segment in the virtual *
+ * address space of the application. *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_FIXED_MAP_ADDR - Map at the suggested virtual address *
+ * SCI_FLAG_READONLY_MAP - The segment is mapped in read-only mode *
+ * SCI_FLAG_LOCK_OPERATION - Enable Lock operations (fetch and add) *
+ * SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP *
+ * - Enable aggressive prefetch with speculative *
+ * hold. *
+ * *
+ * SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP *
+ * - The PSB66 will prefetch 64 bytes. As soon *
+ * as the PCI read retry has been accepted, *
+ * the stream will change state to FREE, even *
+ * if less than 64 bytes were actually read. *
+ * *
+ * SCI_FLAG_IO_MAP_IOSPACE - Enable No Prefetch, no speculative hold. *
+ * *
+ * SCI_FLAG_DMOVE_MAP - Enable DMOVE packet type. The stream will be *
+ * set into FREE state immediately. *
+ * *
+ * SCI_FLAG_WRITES_DISABLE_GATHER_MAP *
+ * - Disable use of gather. *
+ * *
+ * SCI_FLAG_DISABLE_128_BYTES_PACKETS *
+ * - Disable use of 128-Byte packets *
+ * *
+ * SCI_FLAG_CONDITIONAL_INTERRUPT_MAP *
+ * - Write operations through this map will cause *
+ * an atomic "fetch-and-add-one" operation on *
+ * remote memory, but in addition an interrupt *
+ * will be generated if the target memory *
+ * location contained a "null value" before the *
+ * add operation was carried out. *
+ * The conditional interrupt flag must also be *
+ * specified in the SCIRegisterInterruptFlag() *
+ * function. *
+ * *
+ * SCI_FLAG_UNCONDITIONAL_INTERRUPT_MAP *
+ * - Write operations through this map will cause *
+ * an interrupt for the remote adapter *
+ * "in addition to" updating the corresponding *
+ * remote memory location with the data being *
+ * written. *
+ * The unconditional interrupt flag must also *
+ * be specified in the *
+ * SCIRegisterInterruptFlag() function. *
+ * *
+ * SCI_FLAG_WRITE_BACK_CACHE_MAP *
+ * - Enable cacheing of the mapped region. *
+ * Writes through this map will be written to a *
+ * write back cache, hence no remote SCI updates*
+ * until the cache line is flushed. The *
+ * application is responsible for the cache *
+ * flush operation. *
+ * The SCImemCopy() function will handle this *
+ * correctly by doing cache flushes internally. *
+ * This feature is architechture dependent and *
+ * not be available on all plattforms. *
+ * *
+ * SCI_FLAG_NO_MEMORY_LOOPBACK_MAP *
+ * - Forces a map to a remote segment located *
+ * in the local machine to be mapped using *
+ * SCI loopback. This is useful i.e. if you *
+ * want to use a regular map access to be *
+ * serialized with lock operations. *
+ * The default behaviour is to access a remte *
+ * segment located in the local machine as a *
+ * local MMU operation. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the offset and size is *
+ * larger than the segment size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as *
+ * required by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as *
+ * required by the implementation. *
+ * *
+ *********************************************************************************/
+#define SCIMapRemoteSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIMapRemoteSegment)
+DLL volatile void *SCIMapRemoteSegment(
+ sci_remote_segment_t segment,
+ sci_map_t *map,
+ unsigned int offset,
+ unsigned int size,
+ void *addr,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M A P L O C A L S E G M E N T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_FIXED_MAP_ADDR *
+ * SCI_FLAG_READONLY_MAP *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the offset and size is *
+ * larger than the segment size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as *
+ * required by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as *
+ * required by the implementation. *
+ * *
+ *********************************************************************************/
+#define SCIMapLocalSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIMapLocalSegment)
+DLL void *SCIMapLocalSegment(sci_local_segment_t segment,
+ sci_map_t *map,
+ unsigned int offset,
+ unsigned int size,
+ void *addr,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I U N M A P S E G M E N T *
+ * *
+ * This function unmaps pages of shared memory from the callers virtual *
+ * address space. *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - The map is currently in use. *
+ * *
+ *********************************************************************************/
+#define SCIUnmapSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIUnmapSegment)
+DLL void SCIUnmapSegment(sci_map_t map,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E S E G M E N T *
+ * *
+ * Make the specified segment available for connections via the specified *
+ * adapter. If successful, the segment can be accessed from remote nodes *
+ * via the specified adapter. *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_USE_CALLBACK - The callback function will be invoked for events *
+ * on this segment. *
+ * SCI_FLAG_EMPTY - No memory will be allocated for the segment. *
+ * SCI_FLAG_PRIVATE - The segment will be private meaning it will never *
+ * be any connections to it. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_SEGMENTID_USED - The segment with this segmentId is already used *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * *
+ *********************************************************************************/
+#define SCICreateSegment _SISCI_EXPANDE_FUNCTION_NAME(SCICreateSegment)
+DLL void SCICreateSegment(sci_desc_t sd,
+ sci_local_segment_t *segment,
+ unsigned int segmentId,
+ unsigned int size,
+ sci_cb_local_segment_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R L O C A L S E G M E N T E V E N T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_TIMEOUT - The function timed out after specified timeout value. *
+ * SCI_ERR_CANCELLED - The wait operation has been cancelled du to a *
+ * SCIRemoveSegment() on the same handle. *
+ * The handle is invalid when this error is returned. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForLocalSegmentEvent _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForLocalSegmentEvent)
+DLL sci_segment_cb_reason_t SCIWaitForLocalSegmentEvent(
+ sci_local_segment_t segment,
+ unsigned int *sourcenodeId,
+ unsigned int *localAdapterNo,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P R E P A R E S E G M E N T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_SOURCE_ONLY - The segment will be used as a source segment *
+ * for DMA operations. On some system types this *
+ * will enable the SISCI driver to use performance *
+ * improving features. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIPrepareSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIPrepareSegment)
+DLL void SCIPrepareSegment(sci_local_segment_t segment,
+ unsigned int localAdapterNo,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E S E G M E N T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - Unable to remove the segment. The segment is currently *
+ * in use. *
+ * *
+ *********************************************************************************/
+#define SCIRemoveSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveSegment)
+DLL void SCIRemoveSegment(sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T S E G M E N T A V A I L A B L E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * SCI_ERR_SEGMENT_NOT_PREPARED - The segment has not been prepared for access *
+ * from this adapter. *
+ * SCI_ERR_ILLEGAL_OPERATION - The segment is created with the *
+ * SCI_FLAG_PRIVATE flag specified and *
+ * therefore has no segmentId. *
+ * *
+ *********************************************************************************/
+#define SCISetSegmentAvailable _SISCI_EXPANDE_FUNCTION_NAME(SCISetSegmentAvailable)
+DLL void SCISetSegmentAvailable(sci_local_segment_t segment,
+ unsigned int localAdapterNo,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T S E G M E N T U N A V A I L A B L E *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_FORCE_DISCONNECT *
+ * SCI_FLAG_NOTIFY *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation. *
+ * *
+ *********************************************************************************/
+#define SCISetSegmentUnavailable _SISCI_EXPANDE_FUNCTION_NAME(SCISetSegmentUnavailable)
+DLL void SCISetSegmentUnavailable(sci_local_segment_t segment,
+ unsigned int localAdapterNo,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E M A P S E Q U E N C E *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_FAST_BARRIER *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCICreateMapSequence _SISCI_EXPANDE_FUNCTION_NAME(SCICreateMapSequence)
+DLL void SCICreateMapSequence(sci_map_t map,
+ sci_sequence_t *sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E S E Q U E N C E *
+ * *
+ * Flags: *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIRemoveSequence _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveSequence)
+DLL void SCIRemoveSequence(sci_sequence_t sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S T A R T S E Q U E N C E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIStartSequence _SISCI_EXPANDE_FUNCTION_NAME(SCIStartSequence)
+DLL sci_sequence_status_t SCIStartSequence(sci_sequence_t sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C H E C K S E Q U E N CE *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_NO_FLUSH *
+ * SCI_FLAG_NO_STORE_BARRIER *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCICheckSequence _SISCI_EXPANDE_FUNCTION_NAME(SCICheckSequence)
+DLL sci_sequence_status_t SCICheckSequence(sci_sequence_t sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S T O R E B A R R I E R *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIStoreBarrier _SISCI_EXPANDE_FUNCTION_NAME(SCIStoreBarrier)
+DLL void SCIStoreBarrier(sci_sequence_t sequence,
+ unsigned int flags);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I F L U S H R E A D B U F F E R S *
+ * *
+ *********************************************************************************/
+#define SCIFlushReadBuffers _SISCI_EXPANDE_FUNCTION_NAME(SCIFlushReadBuffers)
+DLL void SCIFlushReadBuffers(sci_sequence_t sequence);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P R O B E N O D E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCIProbeNode _SISCI_EXPANDE_FUNCTION_NAME(SCIProbeNode)
+DLL int SCIProbeNode(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int nodeId,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T C S R R E G I S T E R *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCIGetCSRRegister _SISCI_EXPANDE_FUNCTION_NAME(SCIGetCSRRegister)
+DLL unsigned int SCIGetCSRRegister(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int SCINodeId,
+ unsigned int CSROffset,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T C S R R E G I S T E R *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCISetCSRRegister _SISCI_EXPANDE_FUNCTION_NAME(SCISetCSRRegister)
+DLL void SCISetCSRRegister(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int SCINodeId,
+ unsigned int CSROffset,
+ unsigned int CSRValue,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T L O C A L C S R *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIGetLocalCSR _SISCI_EXPANDE_FUNCTION_NAME(SCIGetLocalCSR)
+DLL unsigned int SCIGetLocalCSR(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int CSROffset,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T L O C A L C S R *
+ * *
+ * SISCI Priveleged function
+ *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCISetLocalCSR _SISCI_EXPANDE_FUNCTION_NAME(SCISetLocalCSR)
+DLL void SCISetLocalCSR(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int CSROffset,
+ unsigned int CSRValue,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I A T T A C H P H Y S I C A L M E M O R Y *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * Description: *
+ * *
+ * This function enables usage of physical devices and memory regions where the *
+ * Physical PCI bus address ( and mapped CPU address ) are already known. *
+ * The function will register the physical memory as a SISCI segment which can *
+ * be connected and mapped as a regular SISCI segment. *
+ * *
+ * Requirements: *
+ * *
+ * SCICreateSegment() with flag SCI_FLAG_EMPTY must have been called in advance *
+ * *
+ * Parameter description: *
+ * sci_ioaddr_t ioaddress : This is the address on the PCI bus that a PCI bus *
+ * master has to use to write to the specified memory *
+ * void * address : This is the (mapped) virtual address that the *
+ * application has to use to access the device. *
+ * This means that the device has to be mapped in *
+ * advance bye the devices own driver. *
+ * If the device is not to be accessed by the local *
+ * CPU, the address pointer shold be set to NULL *
+ * Flags *
+ * *
+ * None *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIAttachPhysicalMemory _SISCI_EXPANDE_FUNCTION_NAME(SCIAttachPhysicalMemory)
+DLL void SCIAttachPhysicalMemory(sci_ioaddr_t ioaddress,
+ void *address,
+ unsigned int busNo,
+ unsigned int size,
+ sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I Q U E R Y *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_QUERY - Unrecognized command. *
+ * *
+ *********************************************************************************/
+#define SCIQuery _SISCI_EXPANDE_FUNCTION_NAME(SCIQuery)
+DLL void SCIQuery(unsigned int command,
+ void *data,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/* MAJOR QUERY COMMANDS */
+
+/* This command requires a pointer to a structure of type */
+/* "sci_query_string". The string will be filled in by the query. */
+#define SCI_Q_VENDORID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_VENDORID)
+extern const unsigned int SCI_Q_VENDORID;
+
+
+/* Same as for SCI_VENDOR_ID */
+#define SCI_Q_API _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_API)
+extern const unsigned int SCI_Q_API;
+
+
+/* User passes a pointer to an allocated object of the */
+/* "sci_query_adapter" struct. */
+#define SCI_Q_ADAPTER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER)
+extern const unsigned int SCI_Q_ADAPTER;
+
+
+/* User passes a pointer to an allocated object of the */
+/* "sci_query_system" struct. */
+#define SCI_Q_SYSTEM _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM)
+extern const unsigned int SCI_Q_SYSTEM;
+
+#define SCI_Q_LOCAL_SEGMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_LOCAL_SEGMENT)
+extern const unsigned int SCI_Q_LOCAL_SEGMENT;
+
+#define SCI_Q_REMOTE_SEGMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_REMOTE_SEGMENT)
+extern const unsigned int SCI_Q_REMOTE_SEGMENT;
+
+#define SCI_Q_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_MAP)
+extern const unsigned int SCI_Q_MAP;
+
+typedef struct {
+ char *str; /* Pointer to a string of minimum "length" characters */
+ unsigned int length;
+} sci_query_string_t;
+
+
+typedef struct {
+ unsigned int localAdapterNo; /* The adapter no. that the query concern. */
+ unsigned int portNo; /* The SCI Link port number that the query concern. */
+ unsigned int subcommand; /* A subcommand as specified below. */
+ void *data; /* A pointer to an unsigned int that will return */
+ /* the response to the query. */
+} sci_query_adapter_t;
+
+
+typedef struct {
+ unsigned int subcommand; /* A subcommand as specified below. */
+ void *data; /* A pointer to an unsigned int that will return */
+ /* the response to the query. */
+} sci_query_system_t;
+
+typedef struct {
+ sci_local_segment_t segment;
+ unsigned int subcommand;
+ union {
+ sci_ioaddr_t ioaddr;
+ } data;
+} sci_query_local_segment_t;
+
+typedef struct {
+ sci_remote_segment_t segment;
+ unsigned int subcommand;
+ union {
+ sci_ioaddr_t ioaddr;
+ } data;
+} sci_query_remote_segment_t;
+
+typedef struct {
+ sci_map_t map;
+ unsigned int subcommand;
+ unsigned int data;
+} sci_query_map_t;
+
+/* Minor query commands (sub-commands) for adapter specific information SCI_ADAPTER */
+#define SCI_Q_ADAPTER_DMA_SIZE_ALIGNMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_DMA_SIZE_ALIGNMENT)
+extern const unsigned int SCI_Q_ADAPTER_DMA_SIZE_ALIGNMENT;
+
+#define SCI_Q_ADAPTER_DMA_OFFSET_ALIGNMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_DMA_OFFSET_ALIGNMENT)
+extern const unsigned int SCI_Q_ADAPTER_DMA_OFFSET_ALIGNMENT;
+
+#define SCI_Q_ADAPTER_DMA_MTU _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_DMA_MTU)
+extern const unsigned int SCI_Q_ADAPTER_DMA_MTU;
+
+#define SCI_Q_ADAPTER_SUGGESTED_MIN_DMA_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SUGGESTED_MIN_DMA_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_SUGGESTED_MIN_DMA_SIZE;
+
+#define SCI_Q_ADAPTER_SUGGESTED_MIN_BLOCK_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SUGGESTED_MIN_BLOCK_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_SUGGESTED_MIN_BLOCK_SIZE;
+
+#define SCI_Q_ADAPTER_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_NODEID;
+
+#define SCI_Q_ADAPTER_SERIAL_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SERIAL_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_SERIAL_NUMBER;
+
+#define SCI_Q_ADAPTER_CARD_TYPE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CARD_TYPE)
+extern const unsigned int SCI_Q_ADAPTER_CARD_TYPE;
+
+#define SCI_Q_ADAPTER_NUMBER_OF_STREAMS _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_NUMBER_OF_STREAMS)
+extern const unsigned int SCI_Q_ADAPTER_NUMBER_OF_STREAMS;
+
+#define SCI_Q_ADAPTER_STREAM_BUFFER_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_STREAM_BUFFER_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_STREAM_BUFFER_SIZE;
+
+#define SCI_Q_ADAPTER_CONFIGURED _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CONFIGURED)
+extern const unsigned int SCI_Q_ADAPTER_CONFIGURED;
+
+#define SCI_Q_ADAPTER_LINK_OPERATIONAL _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_LINK_OPERATIONAL)
+extern const unsigned int SCI_Q_ADAPTER_LINK_OPERATIONAL;
+
+#define SCI_Q_ADAPTER_HW_LINK_STATUS_IS_OK _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_HW_LINK_STATUS_IS_OK)
+extern const unsigned int SCI_Q_ADAPTER_HW_LINK_STATUS_IS_OK;
+
+#define SCI_Q_ADAPTER_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_NUMBER;
+
+#define SCI_Q_ADAPTER_INSTANCE_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_INSTANCE_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_INSTANCE_NUMBER;
+
+#define SCI_Q_ADAPTER_FIRMWARE_OK _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_FIRMWARE_OK)
+extern const unsigned int SCI_Q_ADAPTER_FIRMWARE_OK;
+
+#define SCI_Q_ADAPTER_CONNECTED_TO_SWITCH _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CONNECTED_TO_SWITCH)
+extern const unsigned int SCI_Q_ADAPTER_CONNECTED_TO_SWITCH;
+
+#define SCI_Q_ADAPTER_LOCAL_SWITCH_TYPE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_LOCAL_SWITCH_TYPE)
+extern const unsigned int SCI_Q_ADAPTER_LOCAL_SWITCH_TYPE;
+
+#define SCI_Q_ADAPTER_LOCAL_SWITCH_PORT_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_LOCAL_SWITCH_PORT_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_LOCAL_SWITCH_PORT_NUMBER;
+
+#define SCI_Q_ADAPTER_CONNECTED_TO_EXPECTED_SWITCH_PORT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CONNECTED_TO_EXPECTED_SWITCH_PORT)
+extern const unsigned int SCI_Q_ADAPTER_CONNECTED_TO_EXPECTED_SWITCH_PORT;
+
+#define SCI_Q_ADAPTER_ATT_PAGE_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_ATT_PAGE_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_ATT_PAGE_SIZE;
+
+#define SCI_Q_ADAPTER_ATT_NUMBER_OF_ENTRIES _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_ATT_NUMBER_OF_ENTRIES)
+extern const unsigned int SCI_Q_ADAPTER_ATT_NUMBER_OF_ENTRIES;
+
+#define SCI_Q_ADAPTER_ATT_AVAILABLE_ENTRIES _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_ATT_AVAILABLE_ENTRIES)
+extern const unsigned int SCI_Q_ADAPTER_ATT_AVAILABLE_ENTRIES;
+
+#define SCI_Q_ADAPTER_PHYS_MEM_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SCI_Q_ADAPTER_PHYS_MEM_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_PHYS_MEM_NODEID;
+
+#define SCI_Q_ADAPTER_PHYS_MBX_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SCI_Q_ADAPTER_PHYS_MBX_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_PHYS_MBX_NODEID;
+
+#define SCI_Q_ADAPTER_PHYS_LINK_PORT_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_PHYS_LINK_PORT_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_PHYS_LINK_PORT_NODEID;
+
+#define SCI_Q_ADAPTER_SCI_LINK_FREQUENCY _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SCI_LINK_FREQUENCY)
+extern const unsigned int SCI_Q_ADAPTER_SCI_LINK_FREQUENCY;
+
+#define SCI_Q_ADAPTER_B_LINK_FREQUENCY _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_B_LINK_FREQUENCY)
+extern const unsigned int SCI_Q_ADAPTER_B_LINK_FREQUENCY;
+
+#define SCI_Q_ADAPTER_IO_BUS_FREQUENCY _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_IO_BUS_FREQUENCY)
+extern const unsigned int SCI_Q_ADAPTER_IO_BUS_FREQUENCY;
+
+/* Minor query commands (sub-commands) for adapter specific information SCI_SYSTEM */
+#define SCI_Q_SYSTEM_HOSTBRIDGE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM_HOSTBRIDGE)
+extern const unsigned int SCI_Q_SYSTEM_HOSTBRIDGE;
+
+#define SCI_Q_SYSTEM_WRITE_POSTING_ENABLED _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM_WRITE_POSTING_ENABLED)
+extern const unsigned int SCI_Q_SYSTEM_WRITE_POSTING_ENABLED;
+
+#define SCI_Q_SYSTEM_WRITE_COMBINING_ENABLED _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM_WRITE_COMBINING_ENABLED)
+extern const unsigned int SCI_Q_SYSTEM_WRITE_COMBINING_ENABLED;
+
+#define SCI_Q_LOCAL_SEGMENT_IOADDR _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_LOCAL_SEGMENT_IOADDR)
+extern const unsigned int SCI_Q_LOCAL_SEGMENT_IOADDR;
+
+#define SCI_Q_REMOTE_SEGMENT_IOADDR _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_REMOTE_SEGMENT_IOADDR)
+extern const unsigned int SCI_Q_REMOTE_SEGMENT_IOADDR;
+
+#define SCI_Q_MAP_MAPPED_TO_LOCAL_TARGET _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_MAP_MAPPED_TO_LOCAL_TARGET)
+extern const unsigned int SCI_Q_MAP_MAPPED_TO_LOCAL_TARGET;
+
+#define SCI_Q_MAP_QUERY_REMOTE_MAPPED_TO_LOCAL_TARGET _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_MAP_QUERY_REMOTE_MAPPED_TO_LOCAL_TARGET)
+extern const unsigned int SCI_Q_MAP_QUERY_REMOTE_MAPPED_TO_LOCAL_TARGET;
+
+#define HOSTBRIDGE_NOT_AVAILABLE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_NOT_AVAILABLE)
+extern const unsigned int HOSTBRIDGE_NOT_AVAILABLE;
+
+#define HOSTBRIDGE_UNKNOWN _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_UNKNOWN)
+extern const unsigned int HOSTBRIDGE_UNKNOWN;
+
+#define HOSTBRIDGE_440FX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440FX)
+extern const unsigned int HOSTBRIDGE_440FX;
+
+#define HOSTBRIDGE_440LX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440LX)
+extern const unsigned int HOSTBRIDGE_440LX;
+
+#define HOSTBRIDGE_440BX_A _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440BX_A)
+extern const unsigned int HOSTBRIDGE_440BX_A;
+
+#define HOSTBRIDGE_440BX_B _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440BX_B)
+extern const unsigned int HOSTBRIDGE_440BX_B;
+
+#define HOSTBRIDGE_440GX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440GX)
+extern const unsigned int HOSTBRIDGE_440GX;
+
+#define HOSTBRIDGE_450KX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450KX)
+extern const unsigned int HOSTBRIDGE_450KX;
+
+#define HOSTBRIDGE_430NX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_430NX)
+extern const unsigned int HOSTBRIDGE_430NX;
+
+#define HOSTBRIDGE_450NX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450NX)
+extern const unsigned int HOSTBRIDGE_450NX;
+
+#define HOSTBRIDGE_450NX_MICO _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450NX_MICO)
+extern const unsigned int HOSTBRIDGE_450NX_MICO;
+
+#define HOSTBRIDGE_450NX_PXB _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450NX_PXB)
+extern const unsigned int HOSTBRIDGE_450NX_PXB;
+
+#define HOSTBRIDGE_I810 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I810)
+extern const unsigned int HOSTBRIDGE_I810;
+
+#define HOSTBRIDGE_I810_DC100 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I810_DC100)
+extern const unsigned int HOSTBRIDGE_I810_DC100;
+
+#define HOSTBRIDGE_I810E _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I810E)
+extern const unsigned int HOSTBRIDGE_I810E;
+
+#define HOSTBRIDGE_I815 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I815)
+extern const unsigned int HOSTBRIDGE_I815;
+
+#define HOSTBRIDGE_I840 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I840)
+extern const unsigned int HOSTBRIDGE_I840;
+
+#define HOSTBRIDGE_I850 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I850)
+extern const unsigned int HOSTBRIDGE_I850;
+
+#define HOSTBRIDGE_I860 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I860)
+extern const unsigned int HOSTBRIDGE_I860;
+
+#define HOSTBRIDGE_VIA_KT133 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_KT133)
+extern const unsigned int HOSTBRIDGE_VIA_KT133;
+
+#define HOSTBRIDGE_VIA_KX133 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_KX133)
+extern const unsigned int HOSTBRIDGE_VIA_KX133;
+
+#define HOSTBRIDGE_VIA_APOLLO_PRO_133A _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_APOLLO_PRO_133A)
+extern const unsigned int HOSTBRIDGE_VIA_APOLLO_PRO_133A;
+
+#define HOSTBRIDGE_VIA_APOLLO_PRO_266 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_APOLLO_PRO_266)
+extern const unsigned int HOSTBRIDGE_VIA_APOLLO_PRO_266;
+
+#define HOSTBRIDGE_AMD_760_MP _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_AMD_760_MP)
+extern const unsigned int HOSTBRIDGE_AMD_760_MP;
+
+#define HOSTBRIDGE_SERVERWORKS_HE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_HE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_HE;
+
+#define HOSTBRIDGE_SERVERWORKS_HE_B _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_HE_B)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_HE_B;
+
+#define HOSTBRIDGE_SERVERWORKS_LE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_LE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_LE;
+
+
+
+#define HOSTBRIDGE_WRITE_POSTING_DISABLED _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_WRITE_POSTING_DISABLED)
+extern const unsigned int HOSTBRIDGE_WRITE_POSTING_DISABLED;
+
+#define HOSTBRIDGE_WRITE_POSTING_ENABLED _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_WRITE_POSTING_ENABLED)
+extern const unsigned int HOSTBRIDGE_WRITE_POSTING_ENABLED;
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E D M A Q U E U E *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_PHDMA : Create physical DMA queue. Please note that this is an *
+ * priveleged operation. *
+ * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCICreateDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCICreateDMAQueue)
+DLL void SCICreateDMAQueue(sci_desc_t sd,
+ sci_dma_queue_t *dq,
+ unsigned int localAdapterNo,
+ unsigned int maxEntries,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E D M A Q U E U E *
+ * *
+ * Flags *
+ * None. * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Not allowed in this queue state. *
+ * *
+ *********************************************************************************/
+#define SCIRemoveDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveDMAQueue)
+DLL void SCIRemoveDMAQueue(sci_dma_queue_t dq,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I E N Q U E U E D M A T R A N S F E R *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_DMA_READ - The DMA will be remote --> local *
+ * (default is local --> remote) *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the offset and size is larger *
+ * than the segment size or larger than max *
+ * DMA size. *
+ * SCI_ERR_MAX_ENTRIES - The DMA queue is full *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_SEGMENT_NOT_PREPARED - The local segment has not been prepared for *
+ * access from the adapter associated with the *
+ * queue. *
+ * SCI_ERR_SEGMENT_NOT_CONNECTED - The remote segment is not connected through *
+ * the adapter associated with the queue. *
+ *********************************************************************************/
+#define SCIEnqueueDMATransfer _SISCI_EXPANDE_FUNCTION_NAME(SCIEnqueueDMATransfer)
+DLL sci_dma_queue_state_t SCIEnqueueDMATransfer(sci_dma_queue_t dq,
+ sci_local_segment_t localSegment,
+ sci_remote_segment_t remoteSegment,
+ unsigned int localOffset,
+ unsigned int remoteOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P O S T D M A Q U E U E *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_USE_CALLBACK - The end of the transfer will cause the callback *
+ * function to be invoked. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * *
+ *********************************************************************************/
+#define SCIPostDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIPostDMAQueue)
+DLL void SCIPostDMAQueue(sci_dma_queue_t dq,
+ sci_cb_dma_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I A B O R T D M A Q U E U E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * *
+ *********************************************************************************/
+#define SCIAbortDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIAbortDMAQueue)
+DLL void SCIAbortDMAQueue(sci_dma_queue_t dq,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I R E S E T D M A Q U E U E *
+ * *
+ * Flags *
+ * None. * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIResetDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIResetDMAQueue)
+DLL void SCIResetDMAQueue(sci_dma_queue_t dq,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I D M A Q U E U E S T A T E *
+ * *
+ *********************************************************************************/
+#define SCIDMAQueueState _SISCI_EXPANDE_FUNCTION_NAME(SCIDMAQueueState)
+DLL sci_dma_queue_state_t SCIDMAQueueState(sci_dma_queue_t dq);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R D M A Q U E U E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForDMAQueue)
+DLL sci_dma_queue_state_t SCIWaitForDMAQueue(sci_dma_queue_t dq,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P H D M A E N Q U E U E *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_READ *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIphDmaEnqueue _SISCI_EXPANDE_FUNCTION_NAME(SCIphDmaEnqueue)
+DLL void SCIphDmaEnqueue(sci_dma_queue_t dmaqueue,
+ unsigned int size,
+ sci_ioaddr_t localBusAddr,
+ unsigned int remote_nodeid,
+ unsigned int remote_highaddr,
+ unsigned int remote_lowaddr,
+ unsigned int flags,
+ sci_error_t *error);
+
+/*********************************************************************************
+ * *
+ * S C I P H D M A S T A R T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_WAIT *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_DMA_RESET *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIphDmaStart _SISCI_EXPANDE_FUNCTION_NAME(SCIphDmaStart)
+DLL sci_dma_queue_state_t SCIphDmaStart(sci_dma_queue_t dmaqueue,
+ sci_cb_dma_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E I N T E R R U P T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_FIXED_INTNO *
+ * SCI_FLAG_SHARED_INT *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_INTNO_USED - This interrupt number is already used. *
+ * *
+ *********************************************************************************/
+#define SCICreateInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCICreateInterrupt)
+DLL void SCICreateInterrupt(sci_desc_t sd,
+ sci_local_interrupt_t *interrupt,
+ unsigned int localAdapterNo,
+ unsigned int *interruptNo,
+ sci_cb_interrupt_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIRemoveInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveInterrupt)
+DLL void SCIRemoveInterrupt(sci_local_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_TIMEOUT - The function timed out after specified timeout value. *
+ * SCI_ERR_CANCELLED - The wait was interrupted by a call to *
+ * SCIRemoveInterrupt. *
+ * The handle is invalid when this error code is returned.*
+ * *
+ *********************************************************************************/
+#define SCIWaitForInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForInterrupt)
+DLL void SCIWaitForInterrupt(sci_local_interrupt_t interrupt,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C O N N E C T I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_SUCH_INTNO - No such interrupt number. *
+ * SCI_ERR_CONNECTION_REFUSED - Connection attempt refused by remote node. *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * *
+ *********************************************************************************/
+#define SCIConnectInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIConnectInterrupt)
+DLL void SCIConnectInterrupt(sci_desc_t sd,
+ sci_remote_interrupt_t *interrupt,
+ unsigned int nodeId,
+ unsigned int localAdapterNo,
+ unsigned int interruptNo,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I D I S C O N N E C T I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIDisconnectInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIDisconnectInterrupt)
+DLL void SCIDisconnectInterrupt(sci_remote_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I T R I G G E R I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCITriggerInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCITriggerInterrupt)
+DLL void SCITriggerInterrupt(sci_remote_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E G I S T E R I N T E R R U P T F L A G *
+ * *
+ * *
+ * This function register an "interrupt flag" that is identified as an unique *
+ * location within a local segment. If successful, the resulting interrupt *
+ * handle will have been associated with the specified local segment. *
+ * *
+ * It is up to the (remote) client(s) to set up an "interrupt mapping" for the *
+ * corresponding segment offset using either the *
+ * *
+ * - SCI_FLAG_CONDITIONAL_INTERRUPT_MAP *
+ * *
+ * or the *
+ * *
+ * - SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP *
+ * *
+ * option to "SCIMapRemoteSegment()". - I.e. after having established a *
+ * connection to the corresponding segment. A trigger operation can then *
+ * be implemented using a store operation via the relevant "interrupt map". *
+ * *
+ * *
+ * *
+ * *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_CONDITIONAL_INTERRUPT - Triggering is to take place using *
+ * "conditional interrupts". *
+ * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * None. *
+ * *
+ *********************************************************************************/
+#define SCIRegisterInterruptFlag _SISCI_EXPANDE_FUNCTION_NAME(SCIRegisterInterruptFlag)
+DLL void SCIRegisterInterruptFlag(
+ unsigned int localAdapterNo,
+ sci_local_interrupt_t *interrupt,
+ sci_local_segment_t segment,
+ unsigned int offset,
+ sci_cb_interrupt_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I E N A B L E C O N D I T I O N A L I N T E R R U P T *
+ * *
+ * *
+ * This function make sure that another HW interrupt will take place the next *
+ * time the corresponding interrupt flag is triggered by a *
+ * "conditional interrupt" operation. *
+ * *
+ * Default semantics: *
+ * *
+ * When successful, the client can rely on that the first subsequent trigger *
+ * operation will cause a HW interrupt and subsequently cause the client *
+ * handler function to be invoked. *
+ * *
+ * If an interrupt was triggered in parallell with the enable operation, then *
+ * the operation will fail (SCI_ERR_COND_INT_RACE_PROBLEM), and the client can *
+ * not rely on another trigger operation will lead to handler invocation. *
+ * Hence, any state checking normally associated with handling the *
+ * corresponding interrupt should take place before attempting to enable *
+ * again. *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_COND_INT_RACE_PROBLEM - The enable operation failed because an *
+ * incomming trigger operation happened *
+ * concurrently. *
+ * *
+ *********************************************************************************/
+#define SCIEnableConditionalInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIEnableConditionalInterrupt)
+DLL void SCIEnableConditionalInterrupt(
+ sci_local_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I D I S A B L E C O N D I T I O N A L I N T E R R U P T *
+ * *
+ * *
+ * Prevent subsequent "conditional interrupt"trigger operations for *
+ * the specified interupt flag from causing HW interrupt and handler *
+ * invocations. *
+ * *
+ * *
+ * Default semantics: *
+ * *
+ * If successful, no subsequent HW interrupts will take place, but handler *
+ * invocations that have already been scheduled may still take place. *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIDisableConditionalInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIDisableConditionalInterrupt)
+DLL void SCIDisableConditionalInterrupt(
+ sci_local_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T C O N D I T I O N A L I N T E R R U P T C O U N T E R *
+ * *
+ * *
+ * Returns a value that indicates the number of times this flag has *
+ * been trigged since the last time it was enabled or disabled. *
+ * Calling the SCIEnableConditionalInterrupt / SCIDisableConditionalInterrupt *
+ * functions will reset the counter value. *
+ * *
+ * Default semantics: *
+ * *
+ * If successful, the current trig count is returned in the *
+ * interruptTrigCounter parameter. *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OVERFLOW - The number of trig operations have exceeded the range *
+ * that can be counted. *
+ *********************************************************************************/
+#define SCIGetConditionalInterruptTrigCounter _SISCI_EXPANDE_FUNCTION_NAME(SCIGetConditionalInterruptTrigCounter)
+DLL void SCIGetConditionalInterruptTrigCounter(
+ sci_local_interrupt_t interrupt,
+ unsigned int *interruptTrigCounter,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I T R A N S F E R B L O C K *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger *
+ * than the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+#define SCITransferBlock _SISCI_EXPANDE_FUNCTION_NAME(SCITransferBlock)
+DLL void SCITransferBlock(sci_map_t sourceMap,
+ unsigned int sourceOffset,
+ sci_map_t destinationMap,
+ unsigned int destinationOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I T R A N S F E R B L O C K A S Y N C *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_BLOCK_READ *
+ * SCI_FLAG_USE_CALLBACK *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger than *
+ * the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required by *
+ * the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+#define SCITransferBlockAsync _SISCI_EXPANDE_FUNCTION_NAME(SCITransferBlockAsync)
+DLL void SCITransferBlockAsync(sci_map_t sourceMap,
+ unsigned int sourceOffset,
+ sci_map_t destinationMap,
+ unsigned int destinationOffset,
+ unsigned int size,
+ sci_block_transfer_t *block,
+ sci_cb_block_transfer_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R B L O C K T R A N S F E R *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForBlockTransfer _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForBlockTransfer)
+DLL void SCIWaitForBlockTransfer(sci_block_transfer_t block,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I A B O R T B L O C K T R A N S F E R *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * *
+ *********************************************************************************/
+#define SCIAbortBlockTransfer _SISCI_EXPANDE_FUNCTION_NAME(SCIAbortBlockTransfer)
+DLL void SCIAbortBlockTransfer(sci_block_transfer_t block,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M E M C P Y *
+ * *
+ * Flags: *
+ * SCI_FLAG_BLOCK_READ *
+ * SCI_FLAG_ERROR_CHECK *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger *
+ * than the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+
+#define SCIMemCpy _SISCI_EXPANDE_FUNCTION_NAME(SCIMemCpy)
+DLL void SCIMemCpy(sci_sequence_t sequence,
+ void *memAddr,
+ sci_map_t remoteMap,
+ unsigned int remoteOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M E M C O P Y *
+ * *
+ * Flags: *
+ * SCI_FLAG_BLOCK_READ *
+ * SCI_FLAG_ERROR_CHECK *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger *
+ * than the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+
+
+#define SCIMemCopy _SISCI_EXPANDE_FUNCTION_NAME(SCIMemCopy)
+DLL void SCIMemCopy(void *memAddr,
+ sci_map_t remoteMap,
+ unsigned int remoteOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E G I S T E R S E G M E N T M E M O R Y *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required by *
+ * the implementation. *
+ * SCI_ERR_ILLEGAL_ADDRESS - Illegal address. *
+ * SCI_ERR_OUT_OF_RANGE - Size is larger than the maximum size for the *
+ * local segment. *
+ * *
+ *********************************************************************************/
+#define SCIRegisterSegmentMemory _SISCI_EXPANDE_FUNCTION_NAME(SCIRegisterSegmentMemory)
+DLL void SCIRegisterSegmentMemory(void *address,
+ unsigned int size,
+ sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C O N N E C T S C I S P A C E *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_CONNECTION_REFUSED - Connection attempt refused by remote node. *
+ * *
+ *********************************************************************************/
+#define SCIConnectSCISpace _SISCI_EXPANDE_FUNCTION_NAME(SCIConnectSCISpace)
+DLL void SCIConnectSCISpace(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ sci_remote_segment_t *segment,
+ sci_address_t address,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*
+ * =====================================================================================
+ *
+ * S C I A T T A C H L O C A L S E G M E N T
+ * Description:
+ *
+ * SCIAttachLocalSegment() permits an application to "attach" to an already existing
+ * local segment, implying that two or more application want
+ * share the same local segment. The prerequest, is that the
+ * application which originally created the segment ("owner") has
+ * preformed a SCIShareSegment() in order to mark the segment
+ * "shareable".
+ *
+ *
+ * Flags:
+ *
+ * SCI_FLAG_USE_CALLBACK - The callback function will be invoked for events
+ * on this segment.
+ *
+ *
+ * Specific error codes for this function:
+ *
+ * SCI_ERR_ACCESS - No such shared segment
+ * SCI_ERR_NO_SUCH_SEGMENT - No such segment
+ * Note: Current implenentation will return SCI_ERR_ACCESS for both cases. This will
+ * change from next release. Application should handle both cases.
+ *
+ * =====================================================================================
+ */
+#define SCIAttachLocalSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIAttachLocalSegment)
+
+DLL void
+SCIAttachLocalSegment(sci_desc_t sd,
+ sci_local_segment_t *segment,
+ unsigned int segmentId,
+ unsigned int *size,
+ sci_cb_local_segment_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+/*
+ * =====================================================================================
+ *
+ * S C I S H A R E S E G M E N T
+ *
+ * Description:
+ *
+ * SCIShareSegment() permits other application to "attach" to an already existing
+ * local segment, implying that two or more application want
+ * share the same local segment. The prerequest, is that the
+ * application which originally created the segment ("owner") has
+ * preformed a SCIShareSegment() in order to mark the segment
+ * "shareable".
+ *
+ *
+ * Flags:
+ * none
+ *
+ * Specific error codes for this function:
+ *
+ *
+ *
+ * =====================================================================================
+ */
+#define SCIShareSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIShareSegment)
+
+DLL void
+SCIShareSegment(sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I F L U S H *
+ * *
+ * This function will flush the CPU buffers and the PSB buffers. *
+ * *
+ * Flags *
+ * SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY : *
+ * Only flush CPU buffers ( Write combining *
+ * etc buffers). *
+ * *
+ *********************************************************************************/
+
+#define SCIFlush _SISCI_EXPANDE_FUNCTION_NAME(SCIFlush)
+DLL void SCIFlush(sci_sequence_t sequence,
+ unsigned int flags);
+
+#if defined(CPLUSPLUS) || defined(__cplusplus)
+}
+#endif
+
+
+#endif
+
+
+
+
+
+
+
+
+
+
+
diff --git a/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_error.h b/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_error.h
new file mode 100644
index 00000000000..aab7c136d3a
--- /dev/null
+++ b/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_error.h
@@ -0,0 +1,89 @@
+/* $Id: sisci_error.h,v 1.1 2002/12/13 12:17:22 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+
+
+#ifndef _SISCI_ERROR_H_
+#define _SISCI_ERROR_H_
+
+
+/* SCI Error return values always have 30 bit set */
+#define SCI_ERR_MASK 0x40000000
+#define SCI_ERR_REMOTE_MASK 0x01 /* Remote errors should have bit 0 set */
+
+#define SCI_ERR(u) ((unsigned32)(u)&0x7FFFFFFF )
+
+/* Error codes */
+typedef enum {
+ SCI_ERR_OK = 0x000,
+
+
+ SCI_ERR_BUSY = (0x900 | SCI_ERR_MASK),
+ SCI_ERR_FLAG_NOT_IMPLEMENTED = (0x901 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_FLAG = (0x902 | SCI_ERR_MASK),
+ SCI_ERR_NOSPC = (0x904 | SCI_ERR_MASK),
+ SCI_ERR_API_NOSPC = (0x905 | SCI_ERR_MASK),
+ SCI_ERR_HW_NOSPC = (0x906 | SCI_ERR_MASK),
+ SCI_ERR_NOT_IMPLEMENTED = (0x907 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_ADAPTERNO = (0x908 | SCI_ERR_MASK),
+ SCI_ERR_NO_SUCH_ADAPTERNO = (0x909 | SCI_ERR_MASK),
+ SCI_ERR_TIMEOUT = (0x90A | SCI_ERR_MASK),
+ SCI_ERR_OUT_OF_RANGE = (0x90B | SCI_ERR_MASK),
+ SCI_ERR_NO_SUCH_SEGMENT = (0x90C | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_NODEID = (0x90D | SCI_ERR_MASK),
+ SCI_ERR_CONNECTION_REFUSED = (0x90E | SCI_ERR_MASK),
+ SCI_ERR_SEGMENT_NOT_CONNECTED = (0x90F | SCI_ERR_MASK),
+ SCI_ERR_SIZE_ALIGNMENT = (0x910 | SCI_ERR_MASK),
+ SCI_ERR_OFFSET_ALIGNMENT = (0x911 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_PARAMETER = (0x912 | SCI_ERR_MASK),
+ SCI_ERR_MAX_ENTRIES = (0x913 | SCI_ERR_MASK),
+ SCI_ERR_SEGMENT_NOT_PREPARED = (0x914 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_ADDRESS = (0x915 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_OPERATION = (0x916 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_QUERY = (0x917 | SCI_ERR_MASK),
+ SCI_ERR_SEGMENTID_USED = (0x918 | SCI_ERR_MASK),
+ SCI_ERR_SYSTEM = (0x919 | SCI_ERR_MASK),
+ SCI_ERR_CANCELLED = (0x91A | SCI_ERR_MASK),
+ SCI_ERR_NOT_CONNECTED = (0x91B | SCI_ERR_MASK),
+ SCI_ERR_NOT_AVAILABLE = (0x91C | SCI_ERR_MASK),
+ SCI_ERR_INCONSISTENT_VERSIONS = (0x91D | SCI_ERR_MASK),
+ SCI_ERR_COND_INT_RACE_PROBLEM = (0x91E | SCI_ERR_MASK),
+ SCI_ERR_OVERFLOW = (0x91F | SCI_ERR_MASK),
+ SCI_ERR_NOT_INITIALIZED = (0x920 | SCI_ERR_MASK),
+
+ SCI_ERR_ACCESS = (0x921 | SCI_ERR_MASK),
+
+ SCI_ERR_NO_SUCH_NODEID = (0xA00 | SCI_ERR_MASK),
+ SCI_ERR_NODE_NOT_RESPONDING = (0xA02 | SCI_ERR_MASK),
+ SCI_ERR_NO_REMOTE_LINK_ACCESS = (0xA04 | SCI_ERR_MASK),
+ SCI_ERR_NO_LINK_ACCESS = (0xA05 | SCI_ERR_MASK),
+ SCI_ERR_TRANSFER_FAILED = (0xA06 | SCI_ERR_MASK)
+} sci_error_t;
+
+
+#endif /* _SCI_ERROR_H_ */
+
+
+
diff --git a/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_types.h b/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_types.h
new file mode 100644
index 00000000000..77989ffca59
--- /dev/null
+++ b/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_types.h
@@ -0,0 +1,133 @@
+/* $Id: sisci_types.h,v 1.1 2002/12/13 12:17:22 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+#ifndef _SISCI_TYPES_H
+#define _SISCI_TYPES_H
+
+#include "sisci_error.h"
+
+#ifndef IN
+#define IN
+#endif
+
+#ifndef OUT
+#define OUT
+#endif
+
+#ifndef IN_OUT
+#define IN_OUT
+#endif
+
+/* Opaque data types for descriptors/handles */
+typedef struct sci_desc *sci_desc_t;
+typedef struct sci_local_segment *sci_local_segment_t;
+typedef struct sci_remote_segment *sci_remote_segment_t;
+
+typedef struct sci_map *sci_map_t;
+typedef struct sci_sequence *sci_sequence_t;
+#ifndef KERNEL
+typedef struct sci_dma_queue *sci_dma_queue_t;
+#endif
+typedef struct sci_remote_interrupt *sci_remote_interrupt_t;
+typedef struct sci_local_interrupt *sci_local_interrupt_t;
+typedef struct sci_block_transfer *sci_block_transfer_t;
+
+/*
+ * Constants defining reasons for segment callbacks:
+ */
+
+typedef enum {
+ SCI_CB_CONNECT = 1,
+ SCI_CB_DISCONNECT,
+ SCI_CB_NOT_OPERATIONAL,
+ SCI_CB_OPERATIONAL,
+ SCI_CB_LOST
+} sci_segment_cb_reason_t;
+
+#define MAX_CB_REASON SCI_CB_LOST
+
+/* dma_queue_states is identical to the dma_queue_state_t in genif.h, they must be consistent.*/
+typedef enum {
+ SCI_DMAQUEUE_IDLE,
+ SCI_DMAQUEUE_GATHER,
+ SCI_DMAQUEUE_POSTED,
+ SCI_DMAQUEUE_DONE,
+ SCI_DMAQUEUE_ABORTED,
+ SCI_DMAQUEUE_ERROR
+} sci_dma_queue_state_t;
+
+
+typedef enum {
+ SCI_SEQ_OK,
+ SCI_SEQ_RETRIABLE,
+ SCI_SEQ_NOT_RETRIABLE,
+ SCI_SEQ_PENDING
+} sci_sequence_status_t;
+
+
+typedef struct {
+ unsigned short nodeId; /* SCI Address bit 63 - 48 */
+ unsigned short offsHi; /* SCI Address bit 47 - 32 */
+ unsigned int offsLo; /* SCI Address bit 31 - 0 */
+} sci_address_t;
+
+
+typedef unsigned int sci_ioaddr_t;
+
+typedef enum {
+ SCI_CALLBACK_CANCEL = 1,
+ SCI_CALLBACK_CONTINUE
+} sci_callback_action_t;
+
+#ifndef KERNEL
+typedef sci_callback_action_t (*sci_cb_local_segment_t)(void *arg,
+ sci_local_segment_t segment,
+ sci_segment_cb_reason_t reason,
+ unsigned int nodeId,
+ unsigned int localAdapterNo,
+ sci_error_t error);
+
+typedef sci_callback_action_t (*sci_cb_remote_segment_t)(void *arg,
+ sci_remote_segment_t segment,
+ sci_segment_cb_reason_t reason,
+ sci_error_t status);
+
+
+typedef sci_callback_action_t (*sci_cb_dma_t)(void IN *arg,
+ sci_dma_queue_t queue,
+ sci_error_t status);
+
+
+typedef int (*sci_cb_block_transfer_t)(void *arg,
+ sci_block_transfer_t block,
+ sci_error_t status);
+
+
+typedef sci_callback_action_t (*sci_cb_interrupt_t)(void *arg,
+ sci_local_interrupt_t interrupt,
+ sci_error_t status);
+
+#endif /* KERNEL */
+#endif
diff --git a/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_version.h b/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_version.h
new file mode 100644
index 00000000000..c2fccb9ec33
--- /dev/null
+++ b/ndb/src/external/SOLARIS.SPARC/sci/include/sisci_version.h
@@ -0,0 +1,91 @@
+/* $Id: sisci_version.h,v 1.1 2002/12/13 12:17:22 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU Lesser General Public License as published by *
+ * the Free Software Foundation; either version 2.1 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 Lesser 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+#ifndef SISCI_VERSION_H
+#define SISCI_VERSION_H
+
+
+#define SISCI_API_VER_MAJOR 1
+#define SISCI_API_VER_MAJORC "1"
+
+#define SISCI_API_VER_MINOR 10
+#define SISCI_API_VER_MINORC "10"
+#define SISCI_API_VER_MICRO 4
+#define SISCI_API_VER_MICROC "4"
+
+#define SISCI_SIGN_VERSION_MASK 0xfffff000 /* used to mask off API_VER_MICRO */
+
+#define SISCI_API_VERSION (SISCI_API_VER_MAJOR << 24 | SISCI_API_VER_MINOR << 12 | SISCI_API_VER_MICRO)
+
+/* the rules are:
+ *
+ * Changes in API_VER_MICRO should be binary compatible, New flags, functions added. No changes to user code
+ * required if new features is not needed.
+ *
+ * Changes in API_VER_MINOR requires recompilation of user code.
+ *
+ * Changes in the API_VER_MAJOR will most likely require changes to user code. This should not happen very
+ * often...
+ *
+ */
+
+#ifndef BUILD_DATE
+#define BUILD_DATE __DATE__
+#endif
+
+#ifndef BUILD_NAME
+#define BUILD_NAME ""
+#endif
+
+#define API_VERSION "SISCI API version " SISCI_API_VER_MAJORC "." SISCI_API_VER_MINORC "."SISCI_API_VER_MICROC " ( "BUILD_NAME" "BUILD_DATE" )"
+
+#endif
+
+
+/* Version info: */
+/* */
+/* 1.5.2 First SISCI version */
+/* 1.5.3 Some bug fixes */
+/* 1.5.4 Some bug fixes */
+/* 1.5.5 No release */
+/* 1.5.6 Lock flag implemented in function SCIConnectSegment */
+/* 1.5.7 Expanded query functionality */
+/* 1.5.8 Updated error checking (sequence) functionality for D320 */
+/* 1.6.0 Updated error checking (sequence) D320 and IRM 1.9 support */
+/* 1.9.0 Ported to Solaris_sparc, Solaris_x86 and Linux. IRM 1.9. */
+/* 1.9.1 Some bug fixes */
+/* 1.9.2 Added more adapter queries */
+/* 1.9.3 Bug fix in SCIMapLocalSegment and SCIMapRemoteSegment */
+/* 1.9.4 NT Release Developers Kit 2.40 */
+/* 1.9.5 Added flush after data transfer in SCIMemCopy() */
+/* 1.9.5 NT Release Developers Kit 2.44 */
+/* 1.10.0:
+ * New SCIInitialize(), SCITerminate() functions.
+ * Support for D330
+ *
+ *
+ */
+
+
diff --git a/ndb/src/external/WIN32.x86/sci/include/rmlib.h b/ndb/src/external/WIN32.x86/sci/include/rmlib.h
new file mode 100644
index 00000000000..87ba20db99f
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/include/rmlib.h
@@ -0,0 +1,212 @@
+/* $Id: rmlib.h,v 1.1 2002/12/13 12:17:23 hin Exp $ */
+
+/*********************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU Lesser General Public License as published by *
+ * the Free Software Foundation; either version 2.1 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 Lesser 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. *
+ * *
+ * *
+ *********************************************************************************/
+
+/********************************************************************************/
+/* This header file contains the declarations of the SCI Reflective Memory */
+/* library rmlib. The implementation of the library functions is in rmlib.c. */
+/* The library contains all the functions that operate on the reflective */
+/* memory. */
+/* */
+/* NB! */
+/* */
+/* DOLPHIN'S SCI REFLECTIVE MEMORY FILES ARE UNDER DEVELOPMENT AND MAY CHANGE. */
+/* PLEASE CONTACT DOLPHIN FOR FURTHER INFORMATION. */
+/* */
+/* */
+/********************************************************************************/
+
+#include "sisci_error.h"
+#include "sisci_api.h"
+#include "sisci_demolib.h"
+#include "sisci_types.h"
+
+unsigned int seqerr, syncseqerr;
+
+#ifndef _RMLIB_H
+#define _RMLIB_H
+
+
+#if defined(_REENTRANT)
+
+#define _RMLIB_EXPAND_NAME(name) _RMLIB_MT_ ## name
+
+#else
+
+#define _RMLIB_EXPAND_NAME(name) _RMLIB_ST_ ## name
+
+#endif
+
+#ifdef __sparc
+#define CACHE_SIZE 2097152
+#else
+#define CACHE_SIZE 8192
+#endif
+
+/*********************************************************************************/
+/* FLAG VALUES */
+/*********************************************************************************/
+
+#define REFLECT_ERRCHECK 0x2
+
+struct ReflectiveMemorySpace {
+ unsigned int localAdapterNo;
+ unsigned int localNodeId;
+ unsigned int remoteNodeId;
+ sci_desc_t sd;
+ sci_desc_t syncsd;
+ sci_map_t localMap;
+ sci_map_t remoteMap;
+ unsigned int localSegmentId;
+ unsigned int remoteSegmentId;
+ unsigned int syncSegmentId;
+ unsigned int sync_rSegmentId;
+ unsigned int segmentSize;
+ unsigned int *localMapAddr;
+ volatile unsigned int *remoteMapAddr;
+ sci_local_segment_t localSegment;
+ sci_remote_segment_t remoteSegment;
+ sci_local_segment_t syncSegment;
+ sci_remote_segment_t sync_rSegment;
+ sci_map_t syncMap;
+ sci_map_t sync_rMap;
+ sci_sequence_t syncsequence;
+ sci_sequence_t sequence;
+ unsigned int protection;
+ unsigned int retry_value;
+ sci_sequence_status_t sequenceStatus, syncsequenceStatus;
+ volatile unsigned int *syncMapAddr;
+ volatile unsigned int *sync_rMapAddr;
+};
+
+/*********************************************************************************/
+/* P R I N T R E F L E C T I V E M E M O R Y S P A C E */
+/* */
+/*********************************************************************************/
+#define ReflectPrintParameters _RMLIB_EXPAND_NAME(ReflectPrintParameters)
+void ReflectPrintParameters(FILE *stream, struct ReflectiveMemorySpace RM_space);
+
+/*********************************************************************************/
+/* R E F L E C T D M A S E T U P */
+/* */
+/*********************************************************************************/
+#define ReflectDmaSetup _RMLIB_EXPAND_NAME(ReflectDmaSetup)
+sci_error_t ReflectDmaSetup(struct ReflectiveMemorySpace RM_space, sci_dma_queue_t *dmaQueue);
+
+/*********************************************************************************/
+/* R E F L E C T D M A R E M O V E */
+/* */
+/*********************************************************************************/
+#define ReflectDmaRemove _RMLIB_EXPAND_NAME(ReflectDmaRemove)
+sci_error_t ReflectDmaRemove(sci_dma_queue_t dmaQueue);
+
+/*********************************************************************************/
+/* R E F L E C T D M A R U N */
+/* */
+/*********************************************************************************/
+#define ReflectDmaRun _RMLIB_EXPAND_NAME(ReflectDmaRun)
+sci_error_t ReflectDmaRun(struct ReflectiveMemorySpace RM_space,
+ unsigned int* privateSrc,
+ unsigned int size,
+ unsigned int offset,
+ sci_dma_queue_t dmaQueue);
+/*********************************************************************************/
+/* C L O S E R E F L E C T I V E M E M O R Y S P A C E */
+/* */
+/*********************************************************************************/
+#define ReflectClose _RMLIB_EXPAND_NAME(ReflectClose)
+sci_error_t ReflectClose(struct ReflectiveMemorySpace RM_space, unsigned int segment_no);
+
+/*********************************************************************************/
+/* O P E N R E F L E C T I V E M E M O R Y S P A C E */
+/* */
+/*********************************************************************************/
+#define ReflectOpen _RMLIB_EXPAND_NAME(ReflectOpen)
+sci_error_t ReflectOpen(struct ReflectiveMemorySpace *RM_space,
+ unsigned int size,
+ unsigned int segment_no,
+ unsigned int localAdapterNo,
+ unsigned int remoteNodeId,
+ unsigned int protection,
+ unsigned int retry_value);
+
+/*********************************************************************************/
+/* R E F L E C T G E T A C C E S S */
+/* */
+/*********************************************************************************/
+#define ReflectGetAccess _RMLIB_EXPAND_NAME(ReflectGetAccess)
+sci_error_t ReflectGetAccess(struct ReflectiveMemorySpace *RM_space);
+
+/*********************************************************************************/
+/* R E F L E C T R E L E A S E A C C E S S */
+/* */
+/*********************************************************************************/
+#define ReflectReleaseAccess _RMLIB_EXPAND_NAME(ReflectReleaseAccess)
+sci_error_t ReflectReleaseAccess(struct ReflectiveMemorySpace *RM_space);
+
+/*********************************************************************************/
+/* R E F L E C T D M A */
+/* */
+/*********************************************************************************/
+#define ReflectDma _RMLIB_EXPAND_NAME(ReflectDma)
+sci_error_t ReflectDma(struct ReflectiveMemorySpace RM_space,
+ unsigned int* privateSrc,
+ unsigned int size,
+ unsigned int offset);
+
+/*********************************************************************************/
+/* R E F L E C T M E M C O P Y */
+/* */
+/*********************************************************************************/
+#define ReflectMemCopy _RMLIB_EXPAND_NAME(ReflectMemCopy)
+sci_error_t ReflectMemCopy(struct ReflectiveMemorySpace RM_space,
+ unsigned int* privateSrc,
+ unsigned int size,
+ unsigned int offset,
+ unsigned int flags);
+
+/*********************************************************************************/
+/* R E F L E C T S E T */
+/* */
+/*********************************************************************************/
+#define ReflectSet _RMLIB_EXPAND_NAME(ReflectSet)
+sci_error_t ReflectSet(struct ReflectiveMemorySpace RM_space,
+ unsigned int value,
+ unsigned int size,
+ unsigned int offset,
+ unsigned int flags
+ );
+
+/*********************************************************************************/
+/* R E F L E C T P R I N T */
+/* */
+/*********************************************************************************/
+#define ReflectPrint _RMLIB_EXPAND_NAME(ReflectPrint)
+sci_error_t ReflectPrint(FILE *stream,
+ struct ReflectiveMemorySpace RM_space,
+ unsigned int size,
+ unsigned int offset
+ );
+
+
+#endif
diff --git a/ndb/src/external/WIN32.x86/sci/include/scilib.h b/ndb/src/external/WIN32.x86/sci/include/scilib.h
new file mode 100644
index 00000000000..d1501082bab
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/include/scilib.h
@@ -0,0 +1,330 @@
+/* $Id: scilib.h,v 1.1 2002/12/13 12:17:23 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 2002 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ *******************************************************************************/
+
+
+#if defined(_REENTRANT)
+#define _SCIL_EXPANDE_FUNCTION_NAME(name) _SCIL_PUBLIC_FUNC_MT_ ## name
+#define _SCIL_EXPANDE_VARIABLE_NAME(name) _SCIL_PUBLIC_VAR_MT_ ## name
+#else
+#define _SCIL_EXPANDE_FUNCTION_NAME(name) _SCIL_PUBLIC_FUNC_ST_ ## name
+#define _SCIL_EXPANDE_VARIABLE_NAME(name) _SCIL_PUBLIC_VAR_ST_ ## name
+#endif
+#define _SCIL_EXPANDE_CONSTANT_NAME(name) _SCIL_PUBLIC_CONST_ ## name
+
+#include "sisci_api.h"
+
+#if defined(CPLUSPLUS) || defined(__cplusplus)
+extern "C" {
+#endif
+
+
+/*
+ * SISCI segment id pollution:
+ * ===========================
+ * The SISCI library uses regular SISCI segmens internally.
+ * The MSG_QUEUE_LIB_IDENTIFIER_MASK is a mask which is used by the SISCI
+ * library to identify internal SISCI segments ids, from segments used directly
+ * by the user.
+ *
+ * Future versions of the library may have its own namespace.
+ *
+ */
+
+#define MSG_QUEUE_LIB_IDENTIFIER_MASK 0x10000000
+
+
+/*********************************************************************************/
+/* FLAG VALUES */
+/*********************************************************************************/
+
+#define SCIL_FLAG_ERROR_CHECK_DATA _SCIL_EXPANDE_CONSTANT_NAME(SCIL_FLAG_ERROR_CHECK_DATA)
+extern const unsigned int SCIL_FLAG_ERROR_CHECK_DATA;
+
+#define SCIL_FLAG_ERROR_CHECK_PROT _SCIL_EXPANDE_CONSTANT_NAME(SCIL_FLAG_ERROR_CHECK_PROT)
+extern const unsigned int SCIL_FLAG_ERROR_CHECK_PROT;
+
+#define SCIL_FLAG_FULL_ERROR_CHECK _SCIL_EXPANDE_CONSTANT_NAME(SCIL_FLAG_FULL_ERROR_CHECK)
+extern const unsigned int SCIL_FLAG_FULL_ERROR_CHECK;
+
+
+
+
+
+typedef struct sci_msq_queue *sci_msq_queue_t;
+
+
+/*********************************************************************************
+ * *
+ * S C I L C r e a t e M s g Q u e u e *
+ * *
+ * Parameters: *
+ * *
+ * Creates a message queue. The message queue identifier object will be allocated*
+ * if the sci_msq_queue_t * msg pointer is NULL. The function will create a *
+ * remote connection. If this connection times out, the function shoud be *
+ * repeated until connection is established. SCILRemoveMsgQueue() must be called *
+ * to remove the connection and deallocate the message queue identifier. *
+ * *
+ * sci_msq_queue_t *msq : Message queue identifier *
+ * The function must be called with a null pointer *
+ * the first time. *
+ * unsigned int localAdapterNo: Local Adapter Number *
+ * unsigned int remoteNodeId : Remote nodeId *
+ * unsigned int msqId : Message queue number *
+ * unsigned int maxMsgCount : The maximum count of messages in queue *
+ * unsigned int maxMsgSize, : The maximum size of each messages in queue *
+ * unsigned int timeout : Time to wait for successful connection *
+ * unsigned int flags : Flags. *
+ * *
+ * Flags *
+ * *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * None. Normal SISIC error codes. *
+ * *
+ *********************************************************************************/
+#define SCILCreateMsgQueue _SCIL_EXPANDE_FUNCTION_NAME(SCILCreateMsgQueue)
+DLL sci_error_t SCILCreateMsgQueue(sci_msq_queue_t *msq,
+ unsigned int localAdapterNo,
+ unsigned int remoteNodeId,
+ unsigned int msqId,
+ unsigned int maxMsgCount,
+ unsigned int maxMsgSize,
+ unsigned int timeout,
+ unsigned int flags);
+
+
+/*********************************************************************************
+ * *
+ * S C I L R e c e i v e M s g *
+ * *
+ * *
+ * Receives a message from the queue. *
+ * *
+ * Paremeters *
+ * *
+ * sci_msq_queue_t msq : message queue identifier *
+ * void *msg : Location to store received data *
+ * unsigned int size : Size of message to read *
+ * unsigned int *sizeLeft: Bytes left in buffer, after current receive. This is *
+ * just a hint. There may be more. *
+ * *
+ * Flags *
+ * *
+ * SCIL_FLAG_ERROR_CHECK_PROT: The internal buffer management is done using full*
+ * error checking.
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_EWOULD_BLOCK : There is not enough data in the message buffer *
+ * to read the specified number of bytes. *
+ * . *
+ * SCI_ERR_NOT_CONNECTED : The connection is not established. *
+ * *
+ *********************************************************************************/
+#define SCILReceiveMsg _SCIL_EXPANDE_FUNCTION_NAME(SCILReceiveMsg)
+DLL sci_error_t SCILReceiveMsg(
+ sci_msq_queue_t msq,
+ void *msg,
+ unsigned int size,
+ unsigned int *sizeLeft,
+ unsigned int flags);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I L S e n d M s g *
+ * *
+ * *
+ * Sends a message to the queue. *
+ * *
+ * Paremeters *
+ * *
+ * sci_msq_queue_t msq : Message queue identifier *
+ * void *msg : Send data *
+ * unsigned int size : Size of message to send *
+ * unsigned int *sizeFree: Bytes free in buffer, after current send. This is *
+ * just a hint. There may be more. *
+ * *
+ * Flags *
+ * *
+ * SCIL_FLAG_ERROR_CHECK_DATA: The data is transmitted using full error checking*
+ * SCIL_FLAG_ERROR_CHECK_PROT: The internal buffer management is done using full*
+ * error checking. *
+ * SCIL_FLSG_FULL_ERROR_CHECK: This flag is an combination of both above. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_EWOULD_BLOCK : There is not enough data in the message buffer *
+ * to send the specified number of bytes. *
+ * . *
+ * SCI_ERR_NOT_CONNECTED : The connection is not established. *
+ * *
+ *********************************************************************************/
+#define SCILSendMsg _SCIL_EXPANDE_FUNCTION_NAME(SCILSendMsg)
+DLL sci_error_t SCILSendMsg(
+ sci_msq_queue_t msq,
+ void *msg,
+ unsigned int size,
+ unsigned int *sizeFree,
+ unsigned int flags);
+
+
+/*********************************************************************************
+ * *
+ * S C I L R e m o v e M s g Q u e ue *
+ * *
+ * *
+ * Removes a message queue. *
+ * *
+ * sci_msq_queue_t msq : Message queue identifier *
+ * *
+ * Flags *
+ * *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * None *
+ * *
+ *********************************************************************************/
+#define SCILRemoveMsgQueue _SCIL_EXPANDE_FUNCTION_NAME(SCILRemoveMsgQueue)
+DLL sci_error_t SCILRemoveMsgQueue(
+ sci_msq_queue_t *msq,
+ unsigned int flags);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I L I n i t *
+ * *
+ * *
+ * Initializes the SCI library. This function must be called before any other *
+ * function in the library. *
+ * *
+ * Flags *
+ * *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * None *
+ * . *
+ *********************************************************************************/
+#define SCILInit _SCIL_EXPANDE_FUNCTION_NAME(SCILInit)
+DLL sci_error_t SCILInit(unsigned int flags);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I L D e s t r o y *
+ * *
+ * *
+ * Removes internal resources allocated by the SCI Library. No other library *
+ * function should be called after this function. *
+ * *
+ * Flags *
+ * *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * None *
+ * *
+ *********************************************************************************/
+#define SCILDestroy _SCIL_EXPANDE_FUNCTION_NAME(SCILDestroy)
+DLL sci_error_t SCILDestroy(unsigned int flags);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I L C o n n e c t M s g Q u e u e *
+ * *
+ * *
+ * Makes a connection to a remote message queue. This must be done before *
+ * SCILSendMsg() is called. *
+ * *
+ * Parameters: *
+ * *
+ * sci_msq_queue_t *msq : Message queue identifier *
+ * unsigned int localAdapterNo: Local Adapter Number *
+ * unsigned int remoteNodeId : Remote nodeId *
+ * unsigned int msqId : Message queue number *
+ * unsigned int maxMsgCount : The maximum count of messages in queue *
+ * unsigned int maxMsgSize, : The maximum size of each messages in queue *
+ * unsigned int timeout : Time to wait for successful connection *
+ * unsigned int flags : Flags. *
+ * *
+ * Flags *
+ * *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * None. Normal SISIC error codes. *
+ * *
+ *********************************************************************************/
+#define SCILConnectMsgQueue _SCIL_EXPANDE_FUNCTION_NAME(SCILConnectMsgQueue)
+DLL sci_error_t SCILConnectMsgQueue(sci_msq_queue_t *msq,
+ unsigned int localAdapterNo,
+ unsigned int remoteNodeId,
+ unsigned int rmsgId,
+ unsigned int maxMsgCount,
+ unsigned int maxMsgSize,
+ unsigned int timeout,
+ unsigned int flags);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I L D i s c o n n e c t M s g Q u e u e *
+ * *
+ * *
+ * Disconnects from a remote message queue. *
+ * *
+ * Parameters: *
+ * *
+ * sci_msq_queue_t *msq : Message queue identifier *
+ * *
+ * Flags *
+ * *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * None *
+ * *
+ *********************************************************************************/
+#define SCILDisconnectMsgQueue _SCIL_EXPANDE_FUNCTION_NAME(SCILDisconnectMsgQueue)
+DLL sci_error_t SCILDisconnectMsgQueue(sci_msq_queue_t *msq,
+ unsigned int flags);
+
+
+
+#if defined(CPLUSPLUS) || defined(__cplusplus)
+}
+#endif
+
+
+
+
+
+
+
+
+
+
diff --git a/ndb/src/external/WIN32.x86/sci/include/sisci_api.h b/ndb/src/external/WIN32.x86/sci/include/sisci_api.h
new file mode 100644
index 00000000000..9f4a1ddffb3
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/include/sisci_api.h
@@ -0,0 +1,2217 @@
+/* $Id: sisci_api.h,v 1.1 2002/12/13 12:17:23 hin Exp $ */
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2001 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * This program is free software; you can redistribute it and/or modify *
+ * it under the terms of the GNU Lesser General Public License as published by *
+ * the Free Software Foundation; either version 2.1 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 Lesser 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+#ifndef _SISCI_API_H
+#define _SISCI_API_H
+
+#include "sisci_types.h"
+#include "sisci_error.h"
+
+
+#ifdef WIN32
+#ifdef API_DLL
+#define DLL __declspec(dllexport)
+#elif CLIENT_DLL
+#define DLL __declspec(dllimport)
+#endif
+#endif /* WIN32 */
+
+
+#ifndef DLL
+#define DLL
+#endif
+
+#if defined(_REENTRANT)
+#define _SISCI_EXPANDE_FUNCTION_NAME(name) _SISCI_PUBLIC_FUNC_MT_ ## name
+#define _SISCI_EXPANDE_VARIABLE_NAME(name) _SISCI_PUBLIC_VAR_MT_ ## name
+#else
+#define _SISCI_EXPANDE_FUNCTION_NAME(name) _SISCI_PUBLIC_FUNC_ST_ ## name
+#define _SISCI_EXPANDE_VARIABLE_NAME(name) _SISCI_PUBLIC_VAR_ST_ ## name
+#endif
+#define _SISCI_EXPANDE_CONSTANT_NAME(name) _SISCI_PUBLIC_CONST_ ## name
+
+#if defined(CPLUSPLUS) || defined(__cplusplus)
+extern "C" {
+#endif
+
+
+/*********************************************************************************/
+/* FLAG VALUES */
+/*********************************************************************************/
+
+#define SCI_FLAG_FIXED_INTNO _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FIXED_INTNO)
+extern const unsigned int SCI_FLAG_FIXED_INTNO;
+
+#define SCI_FLAG_SHARED_INT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_SHARED_INT)
+extern const unsigned int SCI_FLAG_SHARED_INT;
+
+#define SCI_FLAG_COUNTING_INT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_COUNTING_INT)
+extern const unsigned int SCI_FLAG_COUNTING_INT;
+
+#define SCI_FLAG_FIXED_MAP_ADDR _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FIXED_MAP_ADDR)
+extern const unsigned int SCI_FLAG_FIXED_MAP_ADDR;
+
+#define SCI_FLAG_READONLY_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_READONLY_MAP)
+extern const unsigned int SCI_FLAG_READONLY_MAP;
+
+#define SCI_FLAG_USE_CALLBACK _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_USE_CALLBACK)
+extern const unsigned int SCI_FLAG_USE_CALLBACK;
+
+#define SCI_FLAG_BLOCK_READ _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_BLOCK_READ)
+extern const unsigned int SCI_FLAG_BLOCK_READ;
+
+#define SCI_FLAG_THREAD_SAFE _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_THREAD_SAFE)
+extern const unsigned int SCI_FLAG_THREAD_SAFE;
+
+#define SCI_FLAG_ASYNCHRONOUS_CONNECT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_ASYNCHRONOUS_CONNECT)
+extern const unsigned int SCI_FLAG_ASYNCHRONOUS_CONNECT;
+
+#define SCI_FLAG_EMPTY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_EMPTY)
+extern const unsigned int SCI_FLAG_EMPTY;
+
+#define SCI_FLAG_PRIVATE _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_PRIVATE)
+extern const unsigned int SCI_FLAG_PRIVATE;
+
+#define SCI_FLAG_FORCE_DISCONNECT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FORCE_DISCONNECT)
+extern const unsigned int SCI_FLAG_FORCE_DISCONNECT;
+
+#define SCI_FLAG_NOTIFY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NOTIFY)
+extern const unsigned int SCI_FLAG_NOTIFY;
+
+#define SCI_FLAG_DMA_READ _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_READ)
+extern const unsigned int SCI_FLAG_DMA_READ;
+
+#define SCI_FLAG_DMA_POST _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_POST)
+extern const unsigned int SCI_FLAG_DMA_POST;
+
+#define SCI_FLAG_DMA_WAIT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_WAIT)
+extern const unsigned int SCI_FLAG_DMA_WAIT;
+
+#define SCI_FLAG_DMA_RESET _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_RESET)
+extern const unsigned int SCI_FLAG_DMA_RESET;
+
+#define SCI_FLAG_NO_FLUSH _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NO_FLUSH)
+extern const unsigned int SCI_FLAG_NO_FLUSH;
+
+#define SCI_FLAG_NO_STORE_BARRIER _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NO_STORE_BARRIER)
+extern const unsigned int SCI_FLAG_NO_STORE_BARRIER;
+
+#define SCI_FLAG_FAST_BARRIER _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FAST_BARRIER)
+extern const unsigned int SCI_FLAG_FAST_BARRIER;
+
+#define SCI_FLAG_ERROR_CHECK _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_ERROR_CHECK)
+extern const unsigned int SCI_FLAG_ERROR_CHECK;
+
+#define SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY)
+extern const unsigned int SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY;
+
+/* the FLUSH_CPU_BUFFERS_ONLY flag is for backwards compabillity only and should never be used */
+#define FLUSH_CPU_BUFFERS_ONLY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY)
+
+#define SCI_FLAG_LOCK_OPERATION _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_LOCK_OPERATION)
+extern const unsigned int SCI_FLAG_LOCK_OPERATION;
+
+#define SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP)
+extern const unsigned int SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP;
+
+#define SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP)
+extern const unsigned int SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP;
+
+#define SCI_FLAG_IO_MAP_IOSPACE _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_IO_MAP_IOSPACE)
+extern const unsigned int SCI_FLAG_IO_MAP_IOSPACE;
+
+#define SCI_FLAG_DMOVE_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMOVE_MAP)
+extern const unsigned int SCI_FLAG_DMOVE_MAP;
+
+#define SCI_FLAG_WRITES_DISABLE_GATHER_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_WRITES_DISABLE_GATHER_MAP)
+extern const unsigned int SCI_FLAG_WRITES_DISABLE_GATHER_MAP;
+
+#define SCI_FLAG_DISABLE_128_BYTES_PACKETS _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DISABLE_128_BYTES_PACKETS)
+extern const unsigned int SCI_FLAG_DISABLE_128_BYTES_PACKETS;
+
+#define SCI_FLAG_SHARED_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_SHARED_MAP)
+extern const unsigned int SCI_FLAG_SHARED_MAP;
+
+#define SCI_FLAG_DMA_SOURCE_ONLY _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_SOURCE_ONLY)
+extern const unsigned int SCI_FLAG_DMA_SOURCE_ONLY;
+
+#define SCI_FLAG_CONDITIONAL_INTERRUPT _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_CONDITIONAL_INTERRUPT)
+extern const unsigned int SCI_FLAG_CONDITIONAL_INTERRUPT;
+
+#define SCI_FLAG_CONDITIONAL_INTERRUPT_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_CONDITIONAL_INTERRUPT_MAP)
+extern const unsigned int SCI_FLAG_CONDITIONAL_INTERRUPT_MAP;
+
+#define SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP)
+extern const unsigned int SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP;
+
+#define SCI_FLAG_NO_MEMORY_LOOPBACK_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_NO_MEMORY_LOOPBACK_MAP)
+extern const unsigned int SCI_FLAG_NO_MEMORY_LOOPBACK_MAP;
+
+#if defined(OS_IS_LYNXOS) || defined(OS_IS_VXWORKS)
+#define SCI_FLAG_WRITE_BACK_CACHE_MAP _SISCI_EXPANDE_CONSTANT_NAME(WRITE_BACK_CACHE_MAP)
+extern const unsigned int SCI_FLAG_WRITE_BACK_CACHE_MAP;
+#endif
+
+#define SCI_FLAG_DMA_PHDMA _SISCI_EXPANDE_CONSTANT_NAME(SCI_FLAG_DMA_PHDMA)
+extern const unsigned int SCI_FLAG_DMA_PHDMA;
+
+/*********************************************************************************/
+/* GENERAL VALUES */
+/*********************************************************************************/
+#define SCI_LOCAL_HOST _SISCI_EXPANDE_CONSTANT_NAME(SCI_LOCAL_HOST)
+extern const unsigned int SCI_LOCAL_HOST;
+
+#define SCI_INFINITE_TIMEOUT _SISCI_EXPANDE_CONSTANT_NAME(SCI_INFINITE_TIMEOUT)
+extern const unsigned int SCI_INFINITE_TIMEOUT;
+
+/*********************************************************************************/
+/* GENERAL ERROR CODES */
+/* */
+/* SCI_ERR_ILLEGAL_FLAG - Illegal flag value. */
+/* SCI_ERR_FLAG_NOT_IMPLEMENTED - Flag legal but flag feature not implemented. */
+/* SCI_ERR_NOT_IMPLEMENTED - Function not implemented. */
+/* SCI_ERR_SYSTEM - A system error. Check errno. */
+/* SCI_ERR_NOSPC - Unable to allocate OS resources. */
+/* SCI_ERR_API_NOSPC - Unable to allocate API resources. */
+/* SCI_ERR_HW_NOSPC - Unable to allocate HW resources (Hardware) */
+/* */
+/*********************************************************************************/
+
+
+/*********************************************************************************/
+/* GENERAL "ADAPTER" ERROR CODES */
+/* */
+/* SCI_ERR_NO_SUCH_ADAPTERNO - Adapter number is legal but does not exist. */
+/* SCI_ERR_ILLEGAL_ADAPTERNO - Illegal local adapter number (i.e. outside */
+/* legal range). */
+/* */
+/*********************************************************************************/
+
+
+/*********************************************************************************/
+/* GENERAL "NODEID" ERROR CODES */
+/* */
+/* SCI_ERR_NO_SUCH_NODEID - The remote adapter identified by nodeId does */
+/* not respond, but the intermediate link(s) */
+/* seem(s) to be operational. */
+/* SCI_ERR_ILLEGAL_NODEID - Illegal NodeId. */
+/* */
+/*********************************************************************************/
+
+
+
+/*********************************************************************************
+ * *
+ * S C I I N I T I A L I Z E *
+ * *
+ * This function initializes the SISCI library. *
+ * The function must be called before SCIOpen(). *
+ * *
+ * Flags: *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * None *
+ * *
+ *********************************************************************************/
+#define SCIInitialize _SISCI_EXPANDE_FUNCTION_NAME(SCIInitialize)
+DLL void SCIInitialize(unsigned int flags,
+ sci_error_t *error);
+#if 0
+unsigned int __Internal_SISCI_version_var;
+#endif
+
+/*********************************************************************************
+ * *
+ * S C I T E R M I N A T E *
+ * *
+ * This function terminates the SISCI library. *
+ * The function must be called after SCIClose(). *
+ * *
+ * *
+ *********************************************************************************/
+#define SCITerminate _SISCI_EXPANDE_FUNCTION_NAME(SCITerminate)
+DLL void SCITerminate(void);
+
+/*********************************************************************************
+ * *
+ * S C I O P E N *
+ * *
+ * *
+ * Opens a SCI virtual device. *
+ * Caller must supply a pointer to a variable of type sci_desc_t to be *
+ * initialized. *
+ * *
+ * Flags *
+ * SCI_FLAG_THREAD_SAFE - Operations on resources associated with this *
+ * descriptor will be performed in a multithread-safe *
+ * manner. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_INCONSISTENT_VERSIONS - Inconsistency between the SISCI library *
+ * and the SISCI driver versions. *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIOpen _SISCI_EXPANDE_FUNCTION_NAME(SCIOpen)
+DLL void SCIOpen(sci_desc_t *sd,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C L O S E *
+ * *
+ * This function closes an open SCI virtual device. *
+ * *
+ * Flags: *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - All resources are not deallocated. *
+ * *
+ *********************************************************************************/
+#define SCIClose _SISCI_EXPANDE_FUNCTION_NAME(SCIClose)
+DLL void SCIClose(sci_desc_t sd,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C O N N E C T S E G M E N T *
+ * *
+ * Connects to a remote shared memory segment located at <nodeId> with the *
+ * identifier <segmentId>. *
+ * The user may then call SCIMapRemoteSegment() to map shared memory *
+ * into user space. *
+ * *
+ * Flags *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_ASYNCHRONOUS_CONNECT *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_SUCH_SEGMENT - Could not find the remote segment with the *
+ * given segmentId. *
+ * SCI_ERR_CONNECTION_REFUSED - Connection attempt refused by remote node. *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCIConnectSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIConnectSegment)
+DLL void SCIConnectSegment(sci_desc_t sd,
+ sci_remote_segment_t *segment,
+ unsigned int nodeId,
+ unsigned int segmentId,
+ unsigned int localAdapterNo,
+ sci_cb_remote_segment_t callback,
+ void *callbackArg,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I D I S C O N N E C T S E G M E N T *
+ * *
+ * Disconnects from the give mapped shared memory segment *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - The segment is currently mapped or in use. *
+ * *
+ *********************************************************************************/
+#define SCIDisconnectSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIDisconnectSegment)
+DLL void SCIDisconnectSegment(sci_remote_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T R E M O T E S E G M E N T S I Z E *
+ * *
+ *********************************************************************************/
+#define SCIGetRemoteSegmentSize _SISCI_EXPANDE_FUNCTION_NAME(SCIGetRemoteSegmentSize)
+DLL unsigned int SCIGetRemoteSegmentSize(sci_remote_segment_t segment);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R R E M O T E S E G M E N T E V E N T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation. *
+ * SCI_ERR_CANCELLED - The wait operation has been cancelled du *
+ * to a SCIDisconnectSegment() on the same *
+ * handle. The handle is invalid when this *
+ * error is returned. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForRemoteSegmentEvent _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForRemoteSegmentEvent)
+DLL sci_segment_cb_reason_t SCIWaitForRemoteSegmentEvent(
+ sci_remote_segment_t segment,
+ sci_error_t *status,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M A P R E M O T E S E G M E N T *
+ * *
+ * This function is used to include a shared memory segment in the virtual *
+ * address space of the application. *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_SHARED_MAP - The low level physical map may be shared by *
+ * other applications. *
+ * *
+ * SCI_FLAG_FIXED_MAP_ADDR - Map at the suggested virtual address *
+ * SCI_FLAG_READONLY_MAP - The segment is mapped in read-only mode *
+ * SCI_FLAG_LOCK_OPERATION - Enable Lock operations (fetch and add) *
+ * SCI_FLAG_READ_PREFETCH_AGGR_HOLD_MAP *
+ * - Enable aggressive prefetch with speculative *
+ * hold. *
+ * *
+ * SCI_FLAG_READ_PREFETCH_NO_HOLD_MAP *
+ * - The PSB66 will prefetch 64 bytes. As soon *
+ * as the PCI read retry has been accepted, *
+ * the stream will change state to FREE, even *
+ * if less than 64 bytes were actually read. *
+ * *
+ * SCI_FLAG_IO_MAP_IOSPACE - Enable No Prefetch, no speculative hold. *
+ * *
+ * SCI_FLAG_DMOVE_MAP - Enable DMOVE packet type. The stream will be *
+ * set into FREE state immediately. *
+ * *
+ * SCI_FLAG_WRITES_DISABLE_GATHER_MAP *
+ * - Disable use of gather. *
+ * *
+ * SCI_FLAG_DISABLE_128_BYTES_PACKETS *
+ * - Disable use of 128-Byte packets *
+ * *
+ * SCI_FLAG_CONDITIONAL_INTERRUPT_MAP *
+ * - Write operations through this map will cause *
+ * an atomic "fetch-and-add-one" operation on *
+ * remote memory, but in addition an interrupt *
+ * will be generated if the target memory *
+ * location contained a "null value" before the *
+ * add operation was carried out. *
+ * The conditional interrupt flag must also be *
+ * specified in the SCIRegisterInterruptFlag() *
+ * function. *
+ * *
+ * SCI_FLAG_UNCONDITIONAL_INTERRUPT_MAP *
+ * - Write operations through this map will cause *
+ * an interrupt for the remote adapter *
+ * "in addition to" updating the corresponding *
+ * remote memory location with the data being *
+ * written. *
+ * The unconditional interrupt flag must also *
+ * be specified in the *
+ * SCIRegisterInterruptFlag() function. *
+ * *
+ * SCI_FLAG_WRITE_BACK_CACHE_MAP *
+ * - Enable cacheing of the mapped region. *
+ * Writes through this map will be written to a *
+ * write back cache, hence no remote SCI updates*
+ * until the cache line is flushed. The *
+ * application is responsible for the cache *
+ * flush operation. *
+ * The SCImemCopy() function will handle this *
+ * correctly by doing cache flushes internally. *
+ * This feature is architechture dependent and *
+ * not be available on all plattforms. *
+ * *
+ * SCI_FLAG_NO_MEMORY_LOOPBACK_MAP *
+ * - Forces a map to a remote segment located *
+ * in the local machine to be mapped using *
+ * SCI loopback. This is useful i.e. if you *
+ * want to use a regular map access to be *
+ * serialized with lock operations. *
+ * The default behaviour is to access a remte *
+ * segment located in the local machine as a *
+ * local MMU operation. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the offset and size is *
+ * larger than the segment size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as *
+ * required by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as *
+ * required by the implementation. *
+ * *
+ *********************************************************************************/
+#define SCIMapRemoteSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIMapRemoteSegment)
+DLL volatile void *SCIMapRemoteSegment(
+ sci_remote_segment_t segment,
+ sci_map_t *map,
+ unsigned int offset,
+ unsigned int size,
+ void *addr,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M A P L O C A L S E G M E N T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_FIXED_MAP_ADDR *
+ * SCI_FLAG_READONLY_MAP *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the offset and size is *
+ * larger than the segment size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as *
+ * required by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as *
+ * required by the implementation. *
+ * *
+ *********************************************************************************/
+#define SCIMapLocalSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIMapLocalSegment)
+DLL void *SCIMapLocalSegment(sci_local_segment_t segment,
+ sci_map_t *map,
+ unsigned int offset,
+ unsigned int size,
+ void *addr,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I U N M A P S E G M E N T *
+ * *
+ * This function unmaps pages of shared memory from the callers virtual *
+ * address space. *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - The map is currently in use. *
+ * *
+ *********************************************************************************/
+#define SCIUnmapSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIUnmapSegment)
+DLL void SCIUnmapSegment(sci_map_t map,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E S E G M E N T *
+ * *
+ * Make the specified segment available for connections via the specified *
+ * adapter. If successful, the segment can be accessed from remote nodes *
+ * via the specified adapter. *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_USE_CALLBACK - The callback function will be invoked for events *
+ * on this segment. *
+ * SCI_FLAG_EMPTY - No memory will be allocated for the segment. *
+ * SCI_FLAG_PRIVATE - The segment will be private meaning it will never *
+ * be any connections to it. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_SEGMENTID_USED - The segment with this segmentId is already used *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * *
+ *********************************************************************************/
+#define SCICreateSegment _SISCI_EXPANDE_FUNCTION_NAME(SCICreateSegment)
+DLL void SCICreateSegment(sci_desc_t sd,
+ sci_local_segment_t *segment,
+ unsigned int segmentId,
+ unsigned int size,
+ sci_cb_local_segment_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R L O C A L S E G M E N T E V E N T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_TIMEOUT - The function timed out after specified timeout value. *
+ * SCI_ERR_CANCELLED - The wait operation has been cancelled du to a *
+ * SCIRemoveSegment() on the same handle. *
+ * The handle is invalid when this error is returned. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForLocalSegmentEvent _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForLocalSegmentEvent)
+DLL sci_segment_cb_reason_t SCIWaitForLocalSegmentEvent(
+ sci_local_segment_t segment,
+ unsigned int *sourcenodeId,
+ unsigned int *localAdapterNo,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P R E P A R E S E G M E N T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_SOURCE_ONLY - The segment will be used as a source segment *
+ * for DMA operations. On some system types this *
+ * will enable the SISCI driver to use performance *
+ * improving features. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIPrepareSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIPrepareSegment)
+DLL void SCIPrepareSegment(sci_local_segment_t segment,
+ unsigned int localAdapterNo,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E S E G M E N T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_BUSY - Unable to remove the segment. The segment is currently *
+ * in use. *
+ * *
+ *********************************************************************************/
+#define SCIRemoveSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveSegment)
+DLL void SCIRemoveSegment(sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T S E G M E N T A V A I L A B L E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * SCI_ERR_SEGMENT_NOT_PREPARED - The segment has not been prepared for access *
+ * from this adapter. *
+ * SCI_ERR_ILLEGAL_OPERATION - The segment is created with the *
+ * SCI_FLAG_PRIVATE flag specified and *
+ * therefore has no segmentId. *
+ * *
+ *********************************************************************************/
+#define SCISetSegmentAvailable _SISCI_EXPANDE_FUNCTION_NAME(SCISetSegmentAvailable)
+DLL void SCISetSegmentAvailable(sci_local_segment_t segment,
+ unsigned int localAdapterNo,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T S E G M E N T U N A V A I L A B L E *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_FORCE_DISCONNECT *
+ * SCI_FLAG_NOTIFY *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation. *
+ * *
+ *********************************************************************************/
+#define SCISetSegmentUnavailable _SISCI_EXPANDE_FUNCTION_NAME(SCISetSegmentUnavailable)
+DLL void SCISetSegmentUnavailable(sci_local_segment_t segment,
+ unsigned int localAdapterNo,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E M A P S E Q U E N C E *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_FAST_BARRIER *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCICreateMapSequence _SISCI_EXPANDE_FUNCTION_NAME(SCICreateMapSequence)
+DLL void SCICreateMapSequence(sci_map_t map,
+ sci_sequence_t *sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E S E Q U E N C E *
+ * *
+ * Flags: *
+ * None *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIRemoveSequence _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveSequence)
+DLL void SCIRemoveSequence(sci_sequence_t sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S T A R T S E Q U E N C E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIStartSequence _SISCI_EXPANDE_FUNCTION_NAME(SCIStartSequence)
+DLL sci_sequence_status_t SCIStartSequence(sci_sequence_t sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C H E C K S E Q U E N CE *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_NO_FLUSH *
+ * SCI_FLAG_NO_STORE_BARRIER *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCICheckSequence _SISCI_EXPANDE_FUNCTION_NAME(SCICheckSequence)
+DLL sci_sequence_status_t SCICheckSequence(sci_sequence_t sequence,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S T O R E B A R R I E R *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIStoreBarrier _SISCI_EXPANDE_FUNCTION_NAME(SCIStoreBarrier)
+DLL void SCIStoreBarrier(sci_sequence_t sequence,
+ unsigned int flags);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I F L U S H R E A D B U F F E R S *
+ * *
+ *********************************************************************************/
+#define SCIFlushReadBuffers _SISCI_EXPANDE_FUNCTION_NAME(SCIFlushReadBuffers)
+DLL void SCIFlushReadBuffers(sci_sequence_t sequence);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P R O B E N O D E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCIProbeNode _SISCI_EXPANDE_FUNCTION_NAME(SCIProbeNode)
+DLL int SCIProbeNode(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int nodeId,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T C S R R E G I S T E R *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCIGetCSRRegister _SISCI_EXPANDE_FUNCTION_NAME(SCIGetCSRRegister)
+DLL unsigned int SCIGetCSRRegister(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int SCINodeId,
+ unsigned int CSROffset,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T C S R R E G I S T E R *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_LINK_ACCESS - It was not possible to communicate via the *
+ * local adapter. *
+ * SCI_ERR_NO_REMOTE_LINK_ACCESS - It was not possible to communicate via a *
+ * remote switch port. *
+ * *
+ *********************************************************************************/
+#define SCISetCSRRegister _SISCI_EXPANDE_FUNCTION_NAME(SCISetCSRRegister)
+DLL void SCISetCSRRegister(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int SCINodeId,
+ unsigned int CSROffset,
+ unsigned int CSRValue,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T L O C A L C S R *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIGetLocalCSR _SISCI_EXPANDE_FUNCTION_NAME(SCIGetLocalCSR)
+DLL unsigned int SCIGetLocalCSR(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int CSROffset,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I S E T L O C A L C S R *
+ * *
+ * SISCI Priveleged function
+ *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCISetLocalCSR _SISCI_EXPANDE_FUNCTION_NAME(SCISetLocalCSR)
+DLL void SCISetLocalCSR(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int CSROffset,
+ unsigned int CSRValue,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I A T T A C H P H Y S I C A L M E M O R Y *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * Description: *
+ * *
+ * This function enables usage of physical devices and memory regions where the *
+ * Physical PCI bus address ( and mapped CPU address ) are already known. *
+ * The function will register the physical memory as a SISCI segment which can *
+ * be connected and mapped as a regular SISCI segment. *
+ * *
+ * Requirements: *
+ * *
+ * SCICreateSegment() with flag SCI_FLAG_EMPTY must have been called in advance *
+ * *
+ * Parameter description: *
+ * sci_ioaddr_t ioaddress : This is the address on the PCI bus that a PCI bus *
+ * master has to use to write to the specified memory *
+ * void * address : This is the (mapped) virtual address that the *
+ * application has to use to access the device. *
+ * This means that the device has to be mapped in *
+ * advance bye the devices own driver. *
+ * If the device is not to be accessed by the local *
+ * CPU, the address pointer shold be set to NULL *
+ * Flags *
+ * *
+ * None *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIAttachPhysicalMemory _SISCI_EXPANDE_FUNCTION_NAME(SCIAttachPhysicalMemory)
+DLL void SCIAttachPhysicalMemory(sci_ioaddr_t ioaddress,
+ void *address,
+ unsigned int busNo,
+ unsigned int size,
+ sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I Q U E R Y *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_QUERY - Unrecognized command. *
+ * *
+ *********************************************************************************/
+#define SCIQuery _SISCI_EXPANDE_FUNCTION_NAME(SCIQuery)
+DLL void SCIQuery(unsigned int command,
+ void *data,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/* MAJOR QUERY COMMANDS */
+
+/* This command requires a pointer to a structure of type */
+/* "sci_query_string". The string will be filled in by the query. */
+#define SCI_Q_VENDORID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_VENDORID)
+extern const unsigned int SCI_Q_VENDORID;
+
+
+/* Same as for SCI_VENDOR_ID */
+#define SCI_Q_API _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_API)
+extern const unsigned int SCI_Q_API;
+
+
+/* User passes a pointer to an allocated object of the */
+/* "sci_query_adapter" struct. */
+#define SCI_Q_ADAPTER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER)
+extern const unsigned int SCI_Q_ADAPTER;
+
+
+/* User passes a pointer to an allocated object of the */
+/* "sci_query_system" struct. */
+#define SCI_Q_SYSTEM _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM)
+extern const unsigned int SCI_Q_SYSTEM;
+
+#define SCI_Q_LOCAL_SEGMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_LOCAL_SEGMENT)
+extern const unsigned int SCI_Q_LOCAL_SEGMENT;
+
+#define SCI_Q_REMOTE_SEGMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_REMOTE_SEGMENT)
+extern const unsigned int SCI_Q_REMOTE_SEGMENT;
+
+#define SCI_Q_MAP _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_MAP)
+extern const unsigned int SCI_Q_MAP;
+
+typedef char* sci_semaphoreId_t;
+
+#ifdef WIN32
+struct _semaphoreid {
+ char *semaphoreName;
+};
+#endif /*WIN32*/
+
+typedef struct {
+ char *str; /* Pointer to a string of minimum "length" characters */
+ unsigned int length;
+} sci_query_string_t;
+
+
+typedef struct {
+ unsigned int localAdapterNo; /* The adapter no. that the query concern. */
+ unsigned int portNo; /* The SCI Link port number that the query concern. */
+ unsigned int subcommand; /* A subcommand as specified below. */
+ void *data; /* A pointer to an unsigned int that will return */
+ /* the response to the query. */
+} sci_query_adapter_t;
+
+
+typedef struct {
+ unsigned int subcommand; /* A subcommand as specified below. */
+ void *data; /* A pointer to an unsigned int that will return */
+ /* the response to the query. */
+} sci_query_system_t;
+
+typedef struct {
+ sci_local_segment_t segment;
+ unsigned int subcommand;
+ union {
+ sci_ioaddr_t ioaddr;
+ } data;
+} sci_query_local_segment_t;
+
+typedef struct {
+ sci_remote_segment_t segment;
+ unsigned int subcommand;
+ union {
+ sci_ioaddr_t ioaddr;
+ } data;
+} sci_query_remote_segment_t;
+
+typedef struct {
+ sci_map_t map;
+ unsigned int subcommand;
+ unsigned int data;
+} sci_query_map_t;
+
+/* Minor query commands (sub-commands) for adapter specific information SCI_ADAPTER */
+#define SCI_Q_ADAPTER_DMA_SIZE_ALIGNMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_DMA_SIZE_ALIGNMENT)
+extern const unsigned int SCI_Q_ADAPTER_DMA_SIZE_ALIGNMENT;
+
+#define SCI_Q_ADAPTER_DMA_OFFSET_ALIGNMENT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_DMA_OFFSET_ALIGNMENT)
+extern const unsigned int SCI_Q_ADAPTER_DMA_OFFSET_ALIGNMENT;
+
+#define SCI_Q_ADAPTER_DMA_MTU _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_DMA_MTU)
+extern const unsigned int SCI_Q_ADAPTER_DMA_MTU;
+
+#define SCI_Q_ADAPTER_SUGGESTED_MIN_DMA_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SUGGESTED_MIN_DMA_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_SUGGESTED_MIN_DMA_SIZE;
+
+#define SCI_Q_ADAPTER_SUGGESTED_MIN_BLOCK_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SUGGESTED_MIN_BLOCK_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_SUGGESTED_MIN_BLOCK_SIZE;
+
+#define SCI_Q_ADAPTER_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_NODEID;
+
+#define SCI_Q_ADAPTER_SERIAL_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SERIAL_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_SERIAL_NUMBER;
+
+#define SCI_Q_ADAPTER_CARD_TYPE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CARD_TYPE)
+extern const unsigned int SCI_Q_ADAPTER_CARD_TYPE;
+
+#define SCI_Q_ADAPTER_NUMBER_OF_STREAMS _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_NUMBER_OF_STREAMS)
+extern const unsigned int SCI_Q_ADAPTER_NUMBER_OF_STREAMS;
+
+#define SCI_Q_ADAPTER_STREAM_BUFFER_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_STREAM_BUFFER_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_STREAM_BUFFER_SIZE;
+
+#define SCI_Q_ADAPTER_CONFIGURED _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CONFIGURED)
+extern const unsigned int SCI_Q_ADAPTER_CONFIGURED;
+
+#define SCI_Q_ADAPTER_LINK_OPERATIONAL _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_LINK_OPERATIONAL)
+extern const unsigned int SCI_Q_ADAPTER_LINK_OPERATIONAL;
+
+#define SCI_Q_ADAPTER_HW_LINK_STATUS_IS_OK _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_HW_LINK_STATUS_IS_OK)
+extern const unsigned int SCI_Q_ADAPTER_HW_LINK_STATUS_IS_OK;
+
+#define SCI_Q_ADAPTER_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_NUMBER;
+
+#define SCI_Q_ADAPTER_INSTANCE_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_INSTANCE_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_INSTANCE_NUMBER;
+
+#define SCI_Q_ADAPTER_FIRMWARE_OK _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_FIRMWARE_OK)
+extern const unsigned int SCI_Q_ADAPTER_FIRMWARE_OK;
+
+#define SCI_Q_ADAPTER_CONNECTED_TO_SWITCH _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CONNECTED_TO_SWITCH)
+extern const unsigned int SCI_Q_ADAPTER_CONNECTED_TO_SWITCH;
+
+#define SCI_Q_ADAPTER_LOCAL_SWITCH_TYPE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_LOCAL_SWITCH_TYPE)
+extern const unsigned int SCI_Q_ADAPTER_LOCAL_SWITCH_TYPE;
+
+#define SCI_Q_ADAPTER_LOCAL_SWITCH_PORT_NUMBER _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_LOCAL_SWITCH_PORT_NUMBER)
+extern const unsigned int SCI_Q_ADAPTER_LOCAL_SWITCH_PORT_NUMBER;
+
+#define SCI_Q_ADAPTER_CONNECTED_TO_EXPECTED_SWITCH_PORT _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_CONNECTED_TO_EXPECTED_SWITCH_PORT)
+extern const unsigned int SCI_Q_ADAPTER_CONNECTED_TO_EXPECTED_SWITCH_PORT;
+
+#define SCI_Q_ADAPTER_ATT_PAGE_SIZE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_ATT_PAGE_SIZE)
+extern const unsigned int SCI_Q_ADAPTER_ATT_PAGE_SIZE;
+
+#define SCI_Q_ADAPTER_ATT_NUMBER_OF_ENTRIES _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_ATT_NUMBER_OF_ENTRIES)
+extern const unsigned int SCI_Q_ADAPTER_ATT_NUMBER_OF_ENTRIES;
+
+#define SCI_Q_ADAPTER_ATT_AVAILABLE_ENTRIES _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_ATT_AVAILABLE_ENTRIES)
+extern const unsigned int SCI_Q_ADAPTER_ATT_AVAILABLE_ENTRIES;
+
+#define SCI_Q_ADAPTER_PHYS_MEM_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SCI_Q_ADAPTER_PHYS_MEM_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_PHYS_MEM_NODEID;
+
+#define SCI_Q_ADAPTER_PHYS_MBX_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SCI_Q_ADAPTER_PHYS_MBX_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_PHYS_MBX_NODEID;
+
+#define SCI_Q_ADAPTER_PHYS_LINK_PORT_NODEID _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_PHYS_LINK_PORT_NODEID)
+extern const unsigned int SCI_Q_ADAPTER_PHYS_LINK_PORT_NODEID;
+
+#define SCI_Q_ADAPTER_SCI_LINK_FREQUENCY _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_SCI_LINK_FREQUENCY)
+extern const unsigned int SCI_Q_ADAPTER_SCI_LINK_FREQUENCY;
+
+#define SCI_Q_ADAPTER_B_LINK_FREQUENCY _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_B_LINK_FREQUENCY)
+extern const unsigned int SCI_Q_ADAPTER_B_LINK_FREQUENCY;
+
+#define SCI_Q_ADAPTER_IO_BUS_FREQUENCY _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_ADAPTER_IO_BUS_FREQUENCY)
+extern const unsigned int SCI_Q_ADAPTER_IO_BUS_FREQUENCY;
+
+/* Minor query commands (sub-commands) for adapter specific information SCI_SYSTEM */
+#define SCI_Q_SYSTEM_HOSTBRIDGE _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM_HOSTBRIDGE)
+extern const unsigned int SCI_Q_SYSTEM_HOSTBRIDGE;
+
+#define SCI_Q_SYSTEM_WRITE_POSTING_ENABLED _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM_WRITE_POSTING_ENABLED)
+extern const unsigned int SCI_Q_SYSTEM_WRITE_POSTING_ENABLED;
+
+#define SCI_Q_SYSTEM_WRITE_COMBINING_ENABLED _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_SYSTEM_WRITE_COMBINING_ENABLED)
+extern const unsigned int SCI_Q_SYSTEM_WRITE_COMBINING_ENABLED;
+
+#define SCI_Q_LOCAL_SEGMENT_IOADDR _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_LOCAL_SEGMENT_IOADDR)
+extern const unsigned int SCI_Q_LOCAL_SEGMENT_IOADDR;
+
+#define SCI_Q_REMOTE_SEGMENT_IOADDR _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_REMOTE_SEGMENT_IOADDR)
+extern const unsigned int SCI_Q_REMOTE_SEGMENT_IOADDR;
+
+#define SCI_Q_MAP_MAPPED_TO_LOCAL_TARGET _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_MAP_MAPPED_TO_LOCAL_TARGET)
+extern const unsigned int SCI_Q_MAP_MAPPED_TO_LOCAL_TARGET;
+
+#define SCI_Q_MAP_QUERY_REMOTE_MAPPED_TO_LOCAL_TARGET _SISCI_EXPANDE_CONSTANT_NAME(SCI_Q_MAP_QUERY_REMOTE_MAPPED_TO_LOCAL_TARGET)
+extern const unsigned int SCI_Q_MAP_QUERY_REMOTE_MAPPED_TO_LOCAL_TARGET;
+
+#define HOSTBRIDGE_NOT_AVAILABLE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_NOT_AVAILABLE)
+extern const unsigned int HOSTBRIDGE_NOT_AVAILABLE;
+
+#define HOSTBRIDGE_UNKNOWN _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_UNKNOWN)
+extern const unsigned int HOSTBRIDGE_UNKNOWN;
+
+#define HOSTBRIDGE_440FX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440FX)
+extern const unsigned int HOSTBRIDGE_440FX;
+
+#define HOSTBRIDGE_440LX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440LX)
+extern const unsigned int HOSTBRIDGE_440LX;
+
+#define HOSTBRIDGE_440BX_A _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440BX_A)
+extern const unsigned int HOSTBRIDGE_440BX_A;
+
+#define HOSTBRIDGE_440BX_B _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440BX_B)
+extern const unsigned int HOSTBRIDGE_440BX_B;
+
+#define HOSTBRIDGE_440GX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_440GX)
+extern const unsigned int HOSTBRIDGE_440GX;
+
+#define HOSTBRIDGE_450KX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450KX)
+extern const unsigned int HOSTBRIDGE_450KX;
+
+#define HOSTBRIDGE_430NX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_430NX)
+extern const unsigned int HOSTBRIDGE_430NX;
+
+#define HOSTBRIDGE_450NX _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450NX)
+extern const unsigned int HOSTBRIDGE_450NX;
+
+#define HOSTBRIDGE_450NX_MICO _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450NX_MICO)
+extern const unsigned int HOSTBRIDGE_450NX_MICO;
+
+#define HOSTBRIDGE_450NX_PXB _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_450NX_PXB)
+extern const unsigned int HOSTBRIDGE_450NX_PXB;
+
+#define HOSTBRIDGE_I810 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I810)
+extern const unsigned int HOSTBRIDGE_I810;
+
+#define HOSTBRIDGE_I810_DC100 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I810_DC100)
+extern const unsigned int HOSTBRIDGE_I810_DC100;
+
+#define HOSTBRIDGE_I810E _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I810E)
+extern const unsigned int HOSTBRIDGE_I810E;
+
+#define HOSTBRIDGE_I815 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I815)
+extern const unsigned int HOSTBRIDGE_I815;
+
+#define HOSTBRIDGE_I840 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I840)
+extern const unsigned int HOSTBRIDGE_I840;
+
+#define HOSTBRIDGE_I850 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I850)
+extern const unsigned int HOSTBRIDGE_I850;
+
+#define HOSTBRIDGE_I860 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_I860)
+extern const unsigned int HOSTBRIDGE_I860;
+
+#define HOSTBRIDGE_INTEL_E7500 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_INTEL_E7500)
+extern const unsigned int HOSTBRIDGE_INTEL_E7500;
+
+#define HOSTBRIDGE_VIA_KT133 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_KT133)
+extern const unsigned int HOSTBRIDGE_VIA_KT133;
+
+#define HOSTBRIDGE_VIA_KX133 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_KX133)
+extern const unsigned int HOSTBRIDGE_VIA_KX133;
+
+#define HOSTBRIDGE_VIA_APOLLO_PRO_133A _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_APOLLO_PRO_133A)
+extern const unsigned int HOSTBRIDGE_VIA_APOLLO_PRO_133A;
+
+#define HOSTBRIDGE_VIA_APOLLO_PRO_266 _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_VIA_APOLLO_PRO_266)
+extern const unsigned int HOSTBRIDGE_VIA_APOLLO_PRO_266;
+
+#define HOSTBRIDGE_AMD_760_MP _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_AMD_760_MP)
+extern const unsigned int HOSTBRIDGE_AMD_760_MP;
+
+#define HOSTBRIDGE_AMD_HAMMER _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_AMD_HAMMER)
+extern const unsigned int HOSTBRIDGE_AMD_HAMMER;
+
+#define HOSTBRIDGE_SERVERWORKS_HE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_HE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_HE;
+
+#define HOSTBRIDGE_SERVERWORKS_HE_B _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_HE_B)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_HE_B;
+
+#define HOSTBRIDGE_SERVERWORKS_LE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_LE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_LE;
+
+#define HOSTBRIDGE_SERVERWORKS_GC_HE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_GC_HE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_GC_HE;
+
+#define HOSTBRIDGE_SERVERWORKS_GC_LE _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_GC_LE)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_GC_LE;
+
+#define HOSTBRIDGE_SERVERWORKS_GC_WS _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_GC_WS)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_GC_WS;
+
+#define HOSTBRIDGE_SERVERWORKS_GC_SL _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_SERVERWORKS_GC_SL)
+extern const unsigned int HOSTBRIDGE_SERVERWORKS_GC_SL;
+
+
+#define HOSTBRIDGE_WRITE_POSTING_DISABLED _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_WRITE_POSTING_DISABLED)
+extern const unsigned int HOSTBRIDGE_WRITE_POSTING_DISABLED;
+
+#define HOSTBRIDGE_WRITE_POSTING_ENABLED _SISCI_EXPANDE_CONSTANT_NAME(HOSTBRIDGE_WRITE_POSTING_ENABLED)
+extern const unsigned int HOSTBRIDGE_WRITE_POSTING_ENABLED;
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E D M A Q U E U E *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_PHDMA : Create physical DMA queue. Please note that this is an *
+ * priveleged operation. *
+ * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCICreateDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCICreateDMAQueue)
+DLL void SCICreateDMAQueue(sci_desc_t sd,
+ sci_dma_queue_t *dq,
+ unsigned int localAdapterNo,
+ unsigned int maxEntries,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E D M A Q U E U E *
+ * *
+ * Flags *
+ * None. * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Not allowed in this queue state. *
+ * *
+ *********************************************************************************/
+#define SCIRemoveDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveDMAQueue)
+DLL void SCIRemoveDMAQueue(sci_dma_queue_t dq,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I E N Q U E U E D M A T R A N S F E R *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_DMA_READ - The DMA will be remote --> local *
+ * (default is local --> remote) *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the offset and size is larger *
+ * than the segment size or larger than max *
+ * DMA size. *
+ * SCI_ERR_MAX_ENTRIES - The DMA queue is full *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_SEGMENT_NOT_PREPARED - The local segment has not been prepared for *
+ * access from the adapter associated with the *
+ * queue. *
+ * SCI_ERR_SEGMENT_NOT_CONNECTED - The remote segment is not connected through *
+ * the adapter associated with the queue. *
+ *********************************************************************************/
+#define SCIEnqueueDMATransfer _SISCI_EXPANDE_FUNCTION_NAME(SCIEnqueueDMATransfer)
+DLL sci_dma_queue_state_t SCIEnqueueDMATransfer(sci_dma_queue_t dq,
+ sci_local_segment_t localSegment,
+ sci_remote_segment_t remoteSegment,
+ unsigned int localOffset,
+ unsigned int remoteOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P O S T D M A Q U E U E *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_USE_CALLBACK - The end of the transfer will cause the callback *
+ * function to be invoked. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * *
+ *********************************************************************************/
+#define SCIPostDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIPostDMAQueue)
+DLL void SCIPostDMAQueue(sci_dma_queue_t dq,
+ sci_cb_dma_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I A B O R T D M A Q U E U E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * *
+ *********************************************************************************/
+#define SCIAbortDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIAbortDMAQueue)
+DLL void SCIAbortDMAQueue(sci_dma_queue_t dq,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I R E S E T D M A Q U E U E *
+ * *
+ * Flags *
+ * None. * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIResetDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIResetDMAQueue)
+DLL void SCIResetDMAQueue(sci_dma_queue_t dq,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I D M A Q U E U E S T A T E *
+ * *
+ *********************************************************************************/
+#define SCIDMAQueueState _SISCI_EXPANDE_FUNCTION_NAME(SCIDMAQueueState)
+DLL sci_dma_queue_state_t SCIDMAQueueState(sci_dma_queue_t dq);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R D M A Q U E U E *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForDMAQueue _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForDMAQueue)
+DLL sci_dma_queue_state_t SCIWaitForDMAQueue(sci_dma_queue_t dq,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I P H D M A E N Q U E U E *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_READ *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIphDmaEnqueue _SISCI_EXPANDE_FUNCTION_NAME(SCIphDmaEnqueue)
+DLL void SCIphDmaEnqueue(sci_dma_queue_t dmaqueue,
+ unsigned int size,
+ sci_ioaddr_t localBusAddr,
+ unsigned int remote_nodeid,
+ unsigned int remote_highaddr,
+ unsigned int remote_lowaddr,
+ unsigned int flags,
+ sci_error_t *error);
+
+/*********************************************************************************
+ * *
+ * S C I P H D M A S T A R T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_DMA_WAIT *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_DMA_RESET *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIphDmaStart _SISCI_EXPANDE_FUNCTION_NAME(SCIphDmaStart)
+DLL sci_dma_queue_state_t SCIphDmaStart(sci_dma_queue_t dmaqueue,
+ sci_cb_dma_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+#ifdef WIN32
+/*********************************************************************************
+ * *
+ * S C I C R E A T E N A M E D I N T E R R U P T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_FIXED_INTNO *
+ * SCI_FLAG_SHARED_INT *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_INTNO_USED - This interrupt number is already used. *
+ * *
+ *********************************************************************************/
+#define SCICreateNamedInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCICreateNamedInterrupt)
+DLL void SCICreateNamedInterrupt(sci_desc_t sd,
+ sci_local_interrupt_t *interrupt,
+ unsigned int localAdapterNo,
+ unsigned int *interruptNo,
+ sci_cb_interrupt_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_semaphoreId_t semId,
+ sci_error_t *error);
+#endif /*WIN32*/
+
+/*********************************************************************************
+ * *
+ * S C I C R E A T E I N T E R R U P T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_USE_CALLBACK *
+ * SCI_FLAG_FIXED_INTNO *
+ * SCI_FLAG_SHARED_INT *
+ * SCI_FLAG_COUNTING_INT: This flag will enable counting interrupts. This means *
+ * that the number of trigged interrupts is equal to the *
+ * number of received interrupts. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_INTNO_USED - This interrupt number is already used. *
+ * *
+ *********************************************************************************/
+#define SCICreateInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCICreateInterrupt)
+DLL void SCICreateInterrupt(sci_desc_t sd,
+ sci_local_interrupt_t *interrupt,
+ unsigned int localAdapterNo,
+ unsigned int *interruptNo,
+ sci_cb_interrupt_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E M O V E I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ *********************************************************************************/
+#define SCIRemoveInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIRemoveInterrupt)
+DLL void SCIRemoveInterrupt(sci_local_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_TIMEOUT - The function timed out after specified timeout value. *
+ * SCI_ERR_CANCELLED - The wait was interrupted by a call to *
+ * SCIRemoveInterrupt. *
+ * The handle is invalid when this error code is returned.*
+ * *
+ *********************************************************************************/
+#define SCIWaitForInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForInterrupt)
+DLL void SCIWaitForInterrupt(sci_local_interrupt_t interrupt,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C O N N E C T I N T E R R U P T *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_COUNTING_INT: This flag will enable counting interrupts. This means *
+ * that the number of trigged interrupts is equal to the *
+ * number of received interrupts. *
+ * if SCI_FLAG_COUNTING_INT is not used, the interface *
+ * guarentees that there always will be an remote *
+ * interrupt generated after the first and after the last*
+ * trigger. If interupts is triggered faster than the *
+ * remote interrupt handler can handle, interrupts may be*
+ * lost. *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_NO_SUCH_INTNO - No such interrupt number. *
+ * SCI_ERR_CONNECTION_REFUSED - Connection attempt refused by remote node. *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * *
+ *********************************************************************************/
+#define SCIConnectInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIConnectInterrupt)
+DLL void SCIConnectInterrupt(sci_desc_t sd,
+ sci_remote_interrupt_t *interrupt,
+ unsigned int nodeId,
+ unsigned int localAdapterNo,
+ unsigned int interruptNo,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I D I S C O N N E C T I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIDisconnectInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIDisconnectInterrupt)
+DLL void SCIDisconnectInterrupt(sci_remote_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I T R I G G E R I N T E R R U P T *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCITriggerInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCITriggerInterrupt)
+DLL void SCITriggerInterrupt(sci_remote_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E G I S T E R I N T E R R U P T F L A G *
+ * *
+ * *
+ * This function register an "interrupt flag" that is identified as an unique *
+ * location within a local segment. If successful, the resulting interrupt *
+ * handle will have been associated with the specified local segment. *
+ * *
+ * It is up to the (remote) client(s) to set up an "interrupt mapping" for the *
+ * corresponding segment offset using either the *
+ * *
+ * - SCI_FLAG_CONDITIONAL_INTERRUPT_MAP *
+ * *
+ * or the *
+ * *
+ * - SCI_FLAG_UNCONDITIONAL_DATA_INTERRUPT_MAP *
+ * *
+ * option to "SCIMapRemoteSegment()". - I.e. after having established a *
+ * connection to the corresponding segment. A trigger operation can then *
+ * be implemented using a store operation via the relevant "interrupt map". *
+ * *
+ * *
+ * *
+ * *
+ * *
+ * Flags: *
+ * *
+ * SCI_FLAG_CONDITIONAL_INTERRUPT - Triggering is to take place using *
+ * "conditional interrupts". *
+ * *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * None. *
+ * *
+ *********************************************************************************/
+#define SCIRegisterInterruptFlag _SISCI_EXPANDE_FUNCTION_NAME(SCIRegisterInterruptFlag)
+DLL void SCIRegisterInterruptFlag(
+ unsigned int localAdapterNo,
+ sci_local_interrupt_t *interrupt,
+ sci_local_segment_t segment,
+ unsigned int offset,
+ sci_cb_interrupt_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I E N A B L E C O N D I T I O N A L I N T E R R U P T *
+ * *
+ * *
+ * This function make sure that another HW interrupt will take place the next *
+ * time the corresponding interrupt flag is triggered by a *
+ * "conditional interrupt" operation. *
+ * *
+ * Default semantics: *
+ * *
+ * When successful, the client can rely on that the first subsequent trigger *
+ * operation will cause a HW interrupt and subsequently cause the client *
+ * handler function to be invoked. *
+ * *
+ * If an interrupt was triggered in parallell with the enable operation, then *
+ * the operation will fail (SCI_ERR_COND_INT_RACE_PROBLEM), and the client can *
+ * not rely on another trigger operation will lead to handler invocation. *
+ * Hence, any state checking normally associated with handling the *
+ * corresponding interrupt should take place before attempting to enable *
+ * again. *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_COND_INT_RACE_PROBLEM - The enable operation failed because an *
+ * incomming trigger operation happened *
+ * concurrently. *
+ * *
+ *********************************************************************************/
+#define SCIEnableConditionalInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIEnableConditionalInterrupt)
+DLL void SCIEnableConditionalInterrupt(
+ sci_local_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I D I S A B L E C O N D I T I O N A L I N T E R R U P T *
+ * *
+ * *
+ * Prevent subsequent "conditional interrupt"trigger operations for *
+ * the specified interupt flag from causing HW interrupt and handler *
+ * invocations. *
+ * *
+ * *
+ * Default semantics: *
+ * *
+ * If successful, no subsequent HW interrupts will take place, but handler *
+ * invocations that have already been scheduled may still take place. *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * *
+ *********************************************************************************/
+#define SCIDisableConditionalInterrupt _SISCI_EXPANDE_FUNCTION_NAME(SCIDisableConditionalInterrupt)
+DLL void SCIDisableConditionalInterrupt(
+ sci_local_interrupt_t interrupt,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I G E T C O N D I T I O N A L I N T E R R U P T C O U N T E R *
+ * *
+ * *
+ * Returns a value that indicates the number of times this flag has *
+ * been trigged since the last time it was enabled or disabled. *
+ * Calling the SCIEnableConditionalInterrupt / SCIDisableConditionalInterrupt *
+ * functions will reset the counter value. *
+ * *
+ * Default semantics: *
+ * *
+ * If successful, the current trig count is returned in the *
+ * interruptTrigCounter parameter. *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OVERFLOW - The number of trig operations have exceeded the range *
+ * that can be counted. *
+ *********************************************************************************/
+#define SCIGetConditionalInterruptTrigCounter _SISCI_EXPANDE_FUNCTION_NAME(SCIGetConditionalInterruptTrigCounter)
+DLL void SCIGetConditionalInterruptTrigCounter(
+ sci_local_interrupt_t interrupt,
+ unsigned int *interruptTrigCounter,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I T R A N S F E R B L O C K *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger *
+ * than the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+#define SCITransferBlock _SISCI_EXPANDE_FUNCTION_NAME(SCITransferBlock)
+DLL void SCITransferBlock(sci_map_t sourceMap,
+ unsigned int sourceOffset,
+ sci_map_t destinationMap,
+ unsigned int destinationOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I T R A N S F E R B L O C K A S Y N C *
+ * *
+ * Flags *
+ * *
+ * SCI_FLAG_BLOCK_READ *
+ * SCI_FLAG_USE_CALLBACK *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger than *
+ * the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required by *
+ * the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+#define SCITransferBlockAsync _SISCI_EXPANDE_FUNCTION_NAME(SCITransferBlockAsync)
+DLL void SCITransferBlockAsync(sci_map_t sourceMap,
+ unsigned int sourceOffset,
+ sci_map_t destinationMap,
+ unsigned int destinationOffset,
+ unsigned int size,
+ sci_block_transfer_t *block,
+ sci_cb_block_transfer_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I W A I T F O R B L O C K T R A N S F E R *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * SCI_ERR_TIMEOUT - The function timed out after specified *
+ * timeout value. *
+ * *
+ *********************************************************************************/
+#define SCIWaitForBlockTransfer _SISCI_EXPANDE_FUNCTION_NAME(SCIWaitForBlockTransfer)
+DLL void SCIWaitForBlockTransfer(sci_block_transfer_t block,
+ unsigned int timeout,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I A B O R T B L O C K T R A N S F E R *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_ILLEGAL_OPERATION - Illegal operation *
+ * *
+ *********************************************************************************/
+#define SCIAbortBlockTransfer _SISCI_EXPANDE_FUNCTION_NAME(SCIAbortBlockTransfer)
+DLL void SCIAbortBlockTransfer(sci_block_transfer_t block,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M E M C P Y *
+ * *
+ * Flags: *
+ * SCI_FLAG_BLOCK_READ *
+ * SCI_FLAG_ERROR_CHECK *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger *
+ * than the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+
+#define SCIMemCpy _SISCI_EXPANDE_FUNCTION_NAME(SCIMemCpy)
+DLL void SCIMemCpy(sci_sequence_t sequence,
+ void *memAddr,
+ sci_map_t remoteMap,
+ unsigned int remoteOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I M E M C O P Y *
+ * *
+ * Flags: *
+ * SCI_FLAG_BLOCK_READ *
+ * SCI_FLAG_ERROR_CHECK *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_OUT_OF_RANGE - The sum of the size and offset is larger *
+ * than the corresponding map size. *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_OFFSET_ALIGNMENT - Offset is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_TRANSFER_FAILED - The data transfer failed. *
+ * *
+ *********************************************************************************/
+
+
+#define SCIMemCopy _SISCI_EXPANDE_FUNCTION_NAME(SCIMemCopy)
+DLL void SCIMemCopy(void *memAddr,
+ sci_map_t remoteMap,
+ unsigned int remoteOffset,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*********************************************************************************
+ * *
+ * S C I R E G I S T E R S E G M E N T M E M O R Y *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required by *
+ * the implementation. *
+ * SCI_ERR_ILLEGAL_ADDRESS - Illegal address. *
+ * SCI_ERR_OUT_OF_RANGE - Size is larger than the maximum size for the *
+ * local segment. *
+ * *
+ *********************************************************************************/
+#define SCIRegisterSegmentMemory _SISCI_EXPANDE_FUNCTION_NAME(SCIRegisterSegmentMemory)
+DLL void SCIRegisterSegmentMemory(void *address,
+ unsigned int size,
+ sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+
+
+/*********************************************************************************
+ * *
+ * S C I C O N N E C T S C I S P A C E *
+ * *
+ * SISCI Priveleged function *
+ * *
+ * *
+ * Flags *
+ * None. *
+ * *
+ * *
+ * Specific error codes for this function: *
+ * *
+ * SCI_ERR_SIZE_ALIGNMENT - Size is not correctly aligned as required *
+ * by the implementation. *
+ * SCI_ERR_CONNECTION_REFUSED - Connection attempt refused by remote node. *
+ * *
+ *********************************************************************************/
+#define SCIConnectSCISpace _SISCI_EXPANDE_FUNCTION_NAME(SCIConnectSCISpace)
+DLL void SCIConnectSCISpace(sci_desc_t sd,
+ unsigned int localAdapterNo,
+ sci_remote_segment_t *segment,
+ sci_address_t address,
+ unsigned int size,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+
+/*
+ * =====================================================================================
+ *
+ * S C I A T T A C H L O C A L S E G M E N T
+ * Description:
+ *
+ * SCIAttachLocalSegment() permits an application to "attach" to an already existing
+ * local segment, implying that two or more application want
+ * share the same local segment. The prerequest, is that the
+ * application which originally created the segment ("owner") has
+ * preformed a SCIShareSegment() in order to mark the segment
+ * "shareable".
+ *
+ *
+ * Flags:
+ *
+ * SCI_FLAG_USE_CALLBACK - The callback function will be invoked for events
+ * on this segment.
+ *
+ *
+ * Specific error codes for this function:
+ *
+ * SCI_ERR_ACCESS - No such shared segment
+ * SCI_ERR_NO_SUCH_SEGMENT - No such segment
+ * Note: Current implenentation will return SCI_ERR_ACCESS for both cases. This will
+ * change from next release. Application should handle both cases.
+ *
+ * =====================================================================================
+ */
+#define SCIAttachLocalSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIAttachLocalSegment)
+
+DLL void
+SCIAttachLocalSegment(sci_desc_t sd,
+ sci_local_segment_t *segment,
+ unsigned int segmentId,
+ unsigned int *size,
+ sci_cb_local_segment_t callback,
+ void *callbackArg,
+ unsigned int flags,
+ sci_error_t *error);
+/*
+ * =====================================================================================
+ *
+ * S C I S H A R E S E G M E N T
+ *
+ * Description:
+ *
+ * SCIShareSegment() permits other application to "attach" to an already existing
+ * local segment, implying that two or more application want
+ * share the same local segment. The prerequest, is that the
+ * application which originally created the segment ("owner") has
+ * preformed a SCIShareSegment() in order to mark the segment
+ * "shareable".
+ *
+ *
+ * Flags:
+ * none
+ *
+ * Specific error codes for this function:
+ *
+ *
+ *
+ * =====================================================================================
+ */
+#define SCIShareSegment _SISCI_EXPANDE_FUNCTION_NAME(SCIShareSegment)
+
+DLL void
+SCIShareSegment(sci_local_segment_t segment,
+ unsigned int flags,
+ sci_error_t *error);
+
+
+/*********************************************************************************
+ * *
+ * S C I F L U S H *
+ * *
+ * This function will flush the CPU buffers and the PSB buffers. *
+ * *
+ * Flags *
+ * SCI_FLAG_FLUSH_CPU_BUFFERS_ONLY : *
+ * Only flush CPU buffers ( Write combining *
+ * etc buffers). *
+ * *
+ *********************************************************************************/
+
+#define SCIFlush _SISCI_EXPANDE_FUNCTION_NAME(SCIFlush)
+DLL void SCIFlush(sci_sequence_t sequence,
+ unsigned int flags);
+
+#if defined(CPLUSPLUS) || defined(__cplusplus)
+}
+#endif
+
+
+#endif
+
+
+
+
+
+
+
+
+
+
+
diff --git a/ndb/src/external/WIN32.x86/sci/include/sisci_demolib.h b/ndb/src/external/WIN32.x86/sci/include/sisci_demolib.h
new file mode 100644
index 00000000000..ce5bb2aec8e
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/include/sisci_demolib.h
@@ -0,0 +1,226 @@
+/* $Id: sisci_demolib.h,v 1.1 2002/12/13 12:17:23 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+#ifndef _SISCI_DEMOLIB_H
+#define _SISCI_DEMOLIB_H
+
+
+#if defined(_REENTRANT)
+
+#define _SISCI_DEMOLIB_EXPAND_NAME(name) _SISCI_DEMOLIB_MT_ ## name
+
+#else
+
+#define _SISCI_DEMOLIB_EXPAND_NAME(name) _SISCI_DEMOLIB_ST_ ## name
+
+#endif
+
+/*********************************************************************************/
+/* Q U E R Y A D A P T E R */
+/* */
+/*********************************************************************************/
+
+#define QueryAdapter _SISCI_DEMOLIB_EXPAND_NAME(QueryAdapter)
+
+sci_error_t QueryAdapter(
+ unsigned int subcommand,
+ unsigned int localAdapterNo,
+ unsigned int portNo,
+ unsigned int *data);
+
+
+/*********************************************************************************/
+/* Q U E R Y S Y S T E M */
+/* */
+/*********************************************************************************/
+
+#define QuerySystem _SISCI_DEMOLIB_EXPAND_NAME(QuerySystem)
+
+sci_error_t QuerySystem(
+ unsigned int subcommand,
+ unsigned int *data);
+
+
+/*********************************************************************************/
+/* D E T E C T F I R S T A D A P T E R C A R D */
+/* */
+/*********************************************************************************/
+
+#define DetectFirstAdapterCard _SISCI_DEMOLIB_EXPAND_NAME(DetectFirstAdapterCard)
+
+sci_error_t DetectFirstAdapterCard(
+ unsigned int *localAdapterNo,
+ unsigned int *localNodeId);
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R T Y P E */
+/* */
+/*********************************************************************************/
+
+#define GetAdapterType _SISCI_DEMOLIB_EXPAND_NAME(GetAdapterType)
+
+sci_error_t GetAdapterType(unsigned int localAdapterNo,
+ unsigned int *adapterType);
+
+
+/*********************************************************************************/
+/* G E T L O C A L N O D E I D */
+/* */
+/*********************************************************************************/
+
+#define GetLocalNodeId _SISCI_DEMOLIB_EXPAND_NAME(GetLocalNodeId)
+
+sci_error_t GetLocalNodeId(
+ unsigned int localAdapterNo,
+ unsigned int *localNodeId);
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R S E R I A L N U M B E R */
+/* */
+/*********************************************************************************/
+
+#define GetAdapterSerialNumber _SISCI_DEMOLIB_EXPAND_NAME(GetAdapterSerialNumber)
+
+sci_error_t GetAdapterSerialNumber(
+ unsigned int localAdapterNo,
+ unsigned int *serialNo);
+
+
+
+/*********************************************************************************/
+/* G E T H O S T B R I D G E T Y P E */
+/* */
+/*********************************************************************************/
+
+#define GetHostbridgeType _SISCI_DEMOLIB_EXPAND_NAME(GetHostbridgeType)
+
+sci_error_t GetHostbridgeType(unsigned int *hostbridgeType);
+
+
+
+/*********************************************************************************/
+/* P R I N T H O S T B R I D G E T Y P E */
+/* */
+/*********************************************************************************/
+
+#define PrintHostbridgeType _SISCI_DEMOLIB_EXPAND_NAME(PrintHostbridgeType)
+
+void PrintHostbridgeType(unsigned int hostbridge);
+
+
+
+/*********************************************************************************/
+/* G E T A P I V E R S I O N S T R I N G */
+/* */
+/*********************************************************************************/
+
+#define GetAPIVersionString _SISCI_DEMOLIB_EXPAND_NAME(GetAPIVersionString)
+
+sci_error_t GetAPIVersionString(char str[], unsigned int strLength);
+
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R I O B U S F R E Q U E N C Y */
+/* */
+/*********************************************************************************/
+
+sci_error_t GetAdapterIoBusFrequency(unsigned int localAdapterNo,
+ unsigned int *ioBusFrequency);
+
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R S C I L I N K F R E Q U E N C Y */
+/* */
+/*********************************************************************************/
+
+sci_error_t GetAdapterSciLinkFrequency(unsigned int localAdapterNo,
+ unsigned int *sciLinkFrequency);
+
+
+
+/*********************************************************************************/
+/* G E T A D A P T E R B L I N K F R E Q U E N C Y */
+/* */
+/*********************************************************************************/
+
+sci_error_t GetAdapterBlinkFrequency(unsigned int localAdapterNo,
+ unsigned int *bLinkFrequency);
+
+
+/*********************************************************************************/
+/* S E N D I N T E R R U P T */
+/* */
+/*********************************************************************************/
+
+#define SendInterrupt _SISCI_DEMOLIB_EXPAND_NAME(SendInterrupt)
+
+sci_error_t SendInterrupt(
+ sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int localNodeId,
+ unsigned int remoteNodeId,
+ unsigned int interruptNo);
+
+
+/*********************************************************************************/
+/* R E C E I V E I N T E R R U P T */
+/* */
+/*********************************************************************************/
+
+#define ReceiveInterrupt _SISCI_DEMOLIB_EXPAND_NAME(ReceiveInterrupt)
+
+sci_error_t ReceiveInterrupt(
+ sci_desc_t sd,
+ unsigned int localAdapterNo,
+ unsigned int localNodeId,
+ unsigned int interruptNo);
+
+
+/*********************************************************************************/
+/* E N D I A N S W A P */
+/* */
+/*********************************************************************************/
+
+#define EndianSwap _SISCI_DEMOLIB_EXPAND_NAME(EndianSwap)
+
+unsigned int EndianSwap (unsigned int value);
+
+
+/*********************************************************************************/
+/* S L E E P M I L L I S E C O N D S */
+/* */
+/*********************************************************************************/
+
+#define SleepMilliseconds _SISCI_DEMOLIB_EXPAND_NAME(SleepMilliseconds)
+
+void SleepMilliseconds(int milliseconds);
+
+
+
+
+#endif
diff --git a/ndb/src/external/WIN32.x86/sci/include/sisci_error.h b/ndb/src/external/WIN32.x86/sci/include/sisci_error.h
new file mode 100644
index 00000000000..56fa0d18b3a
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/include/sisci_error.h
@@ -0,0 +1,94 @@
+/* $Id: sisci_error.h,v 1.1 2002/12/13 12:17:23 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+
+
+#ifndef _SISCI_ERROR_H_
+#define _SISCI_ERROR_H_
+
+
+/* SCI Error return values always have 30 bit set */
+#define SCI_ERR_MASK 0x40000000
+#define SCI_ERR_REMOTE_MASK 0x01 /* Remote errors should have bit 0 set */
+
+#define SCI_ERR(u) ((unsigned32)(u)&0x7FFFFFFF )
+
+/* Error codes */
+typedef enum {
+ SCI_ERR_OK = 0x000,
+
+
+ SCI_ERR_BUSY = (0x900 | SCI_ERR_MASK),
+ SCI_ERR_FLAG_NOT_IMPLEMENTED = (0x901 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_FLAG = (0x902 | SCI_ERR_MASK),
+ SCI_ERR_NOSPC = (0x904 | SCI_ERR_MASK),
+ SCI_ERR_API_NOSPC = (0x905 | SCI_ERR_MASK),
+ SCI_ERR_HW_NOSPC = (0x906 | SCI_ERR_MASK),
+ SCI_ERR_NOT_IMPLEMENTED = (0x907 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_ADAPTERNO = (0x908 | SCI_ERR_MASK),
+ SCI_ERR_NO_SUCH_ADAPTERNO = (0x909 | SCI_ERR_MASK),
+ SCI_ERR_TIMEOUT = (0x90A | SCI_ERR_MASK),
+ SCI_ERR_OUT_OF_RANGE = (0x90B | SCI_ERR_MASK),
+ SCI_ERR_NO_SUCH_SEGMENT = (0x90C | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_NODEID = (0x90D | SCI_ERR_MASK),
+ SCI_ERR_CONNECTION_REFUSED = (0x90E | SCI_ERR_MASK),
+ SCI_ERR_SEGMENT_NOT_CONNECTED = (0x90F | SCI_ERR_MASK),
+ SCI_ERR_SIZE_ALIGNMENT = (0x910 | SCI_ERR_MASK),
+ SCI_ERR_OFFSET_ALIGNMENT = (0x911 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_PARAMETER = (0x912 | SCI_ERR_MASK),
+ SCI_ERR_MAX_ENTRIES = (0x913 | SCI_ERR_MASK),
+ SCI_ERR_SEGMENT_NOT_PREPARED = (0x914 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_ADDRESS = (0x915 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_OPERATION = (0x916 | SCI_ERR_MASK),
+ SCI_ERR_ILLEGAL_QUERY = (0x917 | SCI_ERR_MASK),
+ SCI_ERR_SEGMENTID_USED = (0x918 | SCI_ERR_MASK),
+ SCI_ERR_SYSTEM = (0x919 | SCI_ERR_MASK),
+ SCI_ERR_CANCELLED = (0x91A | SCI_ERR_MASK),
+ SCI_ERR_NOT_CONNECTED = (0x91B | SCI_ERR_MASK),
+ SCI_ERR_NOT_AVAILABLE = (0x91C | SCI_ERR_MASK),
+ SCI_ERR_INCONSISTENT_VERSIONS = (0x91D | SCI_ERR_MASK),
+ SCI_ERR_COND_INT_RACE_PROBLEM = (0x91E | SCI_ERR_MASK),
+ SCI_ERR_OVERFLOW = (0x91F | SCI_ERR_MASK),
+ SCI_ERR_NOT_INITIALIZED = (0x920 | SCI_ERR_MASK),
+
+ SCI_ERR_ACCESS = (0x921 | SCI_ERR_MASK),
+
+ SCI_ERR_NO_SUCH_NODEID = (0xA00 | SCI_ERR_MASK),
+ SCI_ERR_NODE_NOT_RESPONDING = (0xA02 | SCI_ERR_MASK),
+ SCI_ERR_NO_REMOTE_LINK_ACCESS = (0xA04 | SCI_ERR_MASK),
+ SCI_ERR_NO_LINK_ACCESS = (0xA05 | SCI_ERR_MASK),
+ SCI_ERR_TRANSFER_FAILED = (0xA06 | SCI_ERR_MASK),
+
+ SCI_ERR_EWOULD_BLOCK = ( 0xB00 | SCI_ERR_MASK),
+ SCI_ERR_SEMAPHORE_COUNT_EXCEEDED = ( 0xB01 | SCI_ERR_MASK),
+ SCI_ERR_IRQL_ILLEGAL = ( 0xB02 | SCI_ERR_MASK)
+
+} sci_error_t;
+
+
+#endif /* _SCI_ERROR_H_ */
+
+
+
diff --git a/ndb/src/external/WIN32.x86/sci/include/sisci_types.h b/ndb/src/external/WIN32.x86/sci/include/sisci_types.h
new file mode 100644
index 00000000000..03e7957c3f2
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/include/sisci_types.h
@@ -0,0 +1,133 @@
+/* $Id: sisci_types.h,v 1.1 2002/12/13 12:17:23 hin Exp $ */
+
+/*******************************************************************************
+ * *
+ * Copyright (C) 1993 - 2000 *
+ * Dolphin Interconnect Solutions AS *
+ * *
+ * 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. *
+ * *
+ * *
+ *******************************************************************************/
+
+
+#ifndef _SISCI_TYPES_H
+#define _SISCI_TYPES_H
+
+#include "sisci_error.h"
+
+#ifndef IN
+#define IN
+#endif
+
+#ifndef OUT
+#define OUT
+#endif
+
+#ifndef IN_OUT
+#define IN_OUT
+#endif
+
+/* Opaque data types for descriptors/handles */
+typedef struct sci_desc *sci_desc_t;
+typedef struct sci_local_segment *sci_local_segment_t;
+typedef struct sci_remote_segment *sci_remote_segment_t;
+
+typedef struct sci_map *sci_map_t;
+typedef struct sci_sequence *sci_sequence_t;
+#ifndef KERNEL
+typedef struct sci_dma_queue *sci_dma_queue_t;
+#endif
+typedef struct sci_remote_interrupt *sci_remote_interrupt_t;
+typedef struct sci_local_interrupt *sci_local_interrupt_t;
+typedef struct sci_block_transfer *sci_block_transfer_t;
+
+/*
+ * Constants defining reasons for segment callbacks:
+ */
+
+typedef enum {
+ SCI_CB_CONNECT = 1,
+ SCI_CB_DISCONNECT,
+ SCI_CB_NOT_OPERATIONAL,
+ SCI_CB_OPERATIONAL,
+ SCI_CB_LOST
+} sci_segment_cb_reason_t;
+
+#define MAX_CB_REASON SCI_CB_LOST
+
+/* dma_queue_states is identical to the dma_queue_state_t in genif.h, they must be consistent.*/
+typedef enum {
+ SCI_DMAQUEUE_IDLE,
+ SCI_DMAQUEUE_GATHER,
+ SCI_DMAQUEUE_POSTED,
+ SCI_DMAQUEUE_DONE,
+ SCI_DMAQUEUE_ABORTED,
+ SCI_DMAQUEUE_ERROR
+} sci_dma_queue_state_t;
+
+
+typedef enum {
+ SCI_SEQ_OK,
+ SCI_SEQ_RETRIABLE,
+ SCI_SEQ_NOT_RETRIABLE,
+ SCI_SEQ_PENDING
+} sci_sequence_status_t;
+
+
+typedef struct {
+ unsigned short nodeId; /* SCI Address bit 63 - 48 */
+ unsigned short offsHi; /* SCI Address bit 47 - 32 */
+ unsigned int offsLo; /* SCI Address bit 31 - 0 */
+} sci_address_t;
+
+
+typedef unsigned int sci_ioaddr_t;
+
+typedef enum {
+ SCI_CALLBACK_CANCEL = 1,
+ SCI_CALLBACK_CONTINUE
+} sci_callback_action_t;
+
+#ifndef KERNEL
+typedef sci_callback_action_t (*sci_cb_local_segment_t)(void *arg,
+ sci_local_segment_t segment,
+ sci_segment_cb_reason_t reason,
+ unsigned int nodeId,
+ unsigned int localAdapterNo,
+ sci_error_t error);
+
+typedef sci_callback_action_t (*sci_cb_remote_segment_t)(void *arg,
+ sci_remote_segment_t segment,
+ sci_segment_cb_reason_t reason,
+ sci_error_t status);
+
+
+typedef sci_callback_action_t (*sci_cb_dma_t)(void IN *arg,
+ sci_dma_queue_t queue,
+ sci_error_t status);
+
+
+typedef int (*sci_cb_block_transfer_t)(void *arg,
+ sci_block_transfer_t block,
+ sci_error_t status);
+
+
+typedef sci_callback_action_t (*sci_cb_interrupt_t)(void *arg,
+ sci_local_interrupt_t interrupt,
+ sci_error_t status);
+
+#endif /* KERNEL */
+#endif
diff --git a/ndb/src/external/WIN32.x86/sci/lib/SISCI_LIBRARY_WIN32.TXT b/ndb/src/external/WIN32.x86/sci/lib/SISCI_LIBRARY_WIN32.TXT
new file mode 100644
index 00000000000..97fe959bb2c
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/lib/SISCI_LIBRARY_WIN32.TXT
@@ -0,0 +1,77 @@
+SISCI_API LIBRARIES AND LINKING
+===============================
+
+
+/MD, /ML, /MT (Use Run-Time Library)
+
+
+
+sisci_api.lib - Single threaded
+
+sisci_api_md.lib - Multithreaded DLL
+
+sisci_api_mt.lib - Multithreaded
+
+
+
+
+With these libraries, you can select either single-threaded or multithreaded run-time routines,
+indicate that a multithreaded module is a dynamic-link library (DLL), and select the retail
+or debug version of the library.
+
+Note Having more than one copy of the run-time libraries in a process can cause problems,
+because static data in one copy is not shared with the other copy. To ensure that your process
+contains only one copy, avoid mixing static and dynamic versions of the run-time libraries.
+The linker will prevent you from linking with both static and dynamic versions within one .EXE file,
+but you can still end up with two (or more) copies of the run-time libraries.
+For example, a dynamic-link library linked with the static (non-DLL) versions of the run-time
+libraries can cause problems when used with an .EXE file that was linked with the dynamic (DLL)
+version of the run-time libraries. (You should also avoid mixing the debug and non-debug versions
+of the libraries in one process.)
+
+
+MD Multithreaded
+----------------
+
+/MD Multithreaded DLL Defines _MT and _DLL so that both multithread- and DLL-specific versions
+of the run-time routines are selected from the standard .H files. This option also causes the
+compiler to place the library name MSVCRT.LIB into the .OBJ file.
+Applications compiled with this option are statically linked to MSVCRT.LIB. This library provides
+a layer of code that allows the linker to resolve external references. The actual working code is
+contained in MSVCRT.DLL, which must be available at run time to applications linked with MSVCRT.LIB.
+
+
+/MDd Debug Multithreaded DLL Defines _DEBUG, _MT, and _DLL so that debug multithread- and DLL-specific
+versions of the run-time routines are selected from the standard .H files. It also causes the compiler
+to place the library name MSVCRTD.LIB into the .OBJ file.
+
+
+ML Single-Threaded
+------------------
+
+
+/ML Single-Threaded Causes the compiler to place the library name LIBC.LIB into the .OBJ file so
+that the linker will use LIBC.LIB to resolve external symbols. This is the compiler’s default action.
+LIBC.LIB does not provide multithread support.
+
+
+/MLd Debug Single-Threaded Defines _DEBUG and causes the compiler to place the library name LIBCD.LIB
+into the .OBJ file so that the linker will use LIBCD.LIB to resolve external symbols. LIBCD.LIB does
+not provide multithread support.
+
+
+MT Multithreaded
+----------------
+
+
+/MT Multithreaded Defines _MT so that multithread-specific versions of the run-time routines are
+selected from the standard header (.H) files. This option also causes the compiler to place the library
+name LIBCMT.LIB into the .OBJ file so that the linker will use LIBCMT.LIB to resolve external symbols.
+
+Either /MT or /MD (or their debug equivalents /MTd or /MDd) is required to create multithreaded programs.
+/MTd Debug Multithreaded Defines _DEBUG and _MT. Defining _MT causes multithread-specific versions of
+the run-time routines to be selected from the standard .H files. This option also causes the compiler
+to place the library name LIBCMTD.LIB into the .OBJ file so that the linker will use LIBCMTD.LIB to
+resolve external symbols. Either /MTd or /MDd (or their non-debug equivalents /MT or MD) is required to
+create multithreaded programs.
+
diff --git a/ndb/src/external/WIN32.x86/sci/lib/scilib.lib b/ndb/src/external/WIN32.x86/sci/lib/scilib.lib
new file mode 100644
index 00000000000..572169a2016
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/lib/scilib.lib
Binary files differ
diff --git a/ndb/src/external/WIN32.x86/sci/lib/scilib_md.lib b/ndb/src/external/WIN32.x86/sci/lib/scilib_md.lib
new file mode 100644
index 00000000000..f18cba61336
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/lib/scilib_md.lib
Binary files differ
diff --git a/ndb/src/external/WIN32.x86/sci/lib/scilib_mt.lib b/ndb/src/external/WIN32.x86/sci/lib/scilib_mt.lib
new file mode 100644
index 00000000000..3e9982468ea
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/lib/scilib_mt.lib
Binary files differ
diff --git a/ndb/src/external/WIN32.x86/sci/lib/sisci_api.lib b/ndb/src/external/WIN32.x86/sci/lib/sisci_api.lib
new file mode 100644
index 00000000000..3fbff6ec809
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/lib/sisci_api.lib
Binary files differ
diff --git a/ndb/src/external/WIN32.x86/sci/lib/sisci_api_md.lib b/ndb/src/external/WIN32.x86/sci/lib/sisci_api_md.lib
new file mode 100644
index 00000000000..1d8d42d1d35
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/lib/sisci_api_md.lib
Binary files differ
diff --git a/ndb/src/external/WIN32.x86/sci/lib/sisci_api_mt.lib b/ndb/src/external/WIN32.x86/sci/lib/sisci_api_mt.lib
new file mode 100644
index 00000000000..017fad7ba31
--- /dev/null
+++ b/ndb/src/external/WIN32.x86/sci/lib/sisci_api_mt.lib
Binary files differ
diff --git a/ndb/src/kernel/Makefile b/ndb/src/kernel/Makefile
new file mode 100644
index 00000000000..11261c047a6
--- /dev/null
+++ b/ndb/src/kernel/Makefile
@@ -0,0 +1,5 @@
+include .defs.mk
+
+DIRS := error blocks vm ndb-main
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/ERROR_codes.txt b/ndb/src/kernel/blocks/ERROR_codes.txt
new file mode 100644
index 00000000000..595afe9650e
--- /dev/null
+++ b/ndb/src/kernel/blocks/ERROR_codes.txt
@@ -0,0 +1,425 @@
+Next QMGR 1
+Next NDBCNTR 1000
+Next NDBFS 2000
+Next DBACC 3001
+Next DBTUP 4007
+Next DBLQH 5031
+Next DBDICT 6003
+Next DBDIH 7173
+Next DBTC 8035
+Next CMVMI 9000
+Next BACKUP 10022
+Next DBUTIL 11002
+Next DBTUX 12001
+Next SUMA 13001
+Next DBDICT 14003
+
+TESTING NODE FAILURE, ARBITRATION
+---------------------------------
+
+911 - 919:
+Crash president when he starts to run in ArbitState 1-9.
+
+910: Crash new president after node crash
+
+ERROR CODES FOR TESTING NODE FAILURE, GLOBAL CHECKPOINT HANDLING:
+-----------------------------------------------------------------
+
+7000:
+Insert system error in master when global checkpoint is idle.
+
+7001:
+Insert system error in master after receiving GCP_PREPARE from
+all nodes in the cluster.
+
+7002:
+Insert system error in master after receiving GCP_NODEFINISH from
+all nodes in the cluster.
+
+7003:
+Insert system error in master after receiving GCP_SAVECONF from
+all nodes in the cluster.
+
+7004:
+Insert system error in master after completing global checkpoint with
+all nodes in the cluster.
+
+7005:
+Insert system error in GCP participant when receiving GCP_PREPARE.
+
+7006:
+Insert system error in GCP participant when receiving GCP_COMMIT.
+
+7007:
+Insert system error in GCP participant when receiving GCP_TCFINISHED.
+
+7008:
+Insert system error in GCP participant when receiving COPY_GCICONF.
+
+5000:
+Insert system error in GCP participant when receiving GCP_SAVEREQ.
+
+5007:
+Delay GCP_SAVEREQ by 10 secs
+
+ERROR CODES FOR TESTING NODE FAILURE, LOCAL CHECKPOINT HANDLING:
+-----------------------------------------------------------------
+
+7009:
+Insert system error in master when local checkpoint is idle.
+
+7010:
+Insert system error in master when local checkpoint is in the
+state clcpStatus = CALCULATE_KEEP_GCI.
+
+7011:
+Stop local checkpoint in the state CALCULATE_KEEP_GCI.
+
+7012:
+Restart local checkpoint after stopping in CALCULATE_KEEP_GCI.
+
+Method:
+1) Error 7011 in master, wait until report of stopped.
+2) Error xxxx in participant to crash it.
+3) Error 7012 in master to start again.
+
+7013:
+Insert system error in master when local checkpoint is in the
+state clcpStatus = COPY_GCI before sending COPY_GCIREQ.
+
+7014:
+Insert system error in master when local checkpoint is in the
+state clcpStatus = TC_CLOPSIZE before sending TC_CLOPSIZEREQ.
+
+7015:
+Insert system error in master when local checkpoint is in the
+state clcpStatus = START_LCP_ROUND before sending START_LCP_ROUND.
+
+7016:
+Insert system error in master when local checkpoint is in the
+state clcpStatus = START_LCP_ROUND after receiving LCP_REPORT.
+
+7017:
+Insert system error in master when local checkpoint is in the
+state clcpStatus = TAB_COMPLETED.
+
+7018:
+Insert system error in master when local checkpoint is in the
+state clcpStatus = TAB_SAVED before sending DIH_LCPCOMPLETE.
+
+7019:
+Insert system error in master when local checkpoint is in the
+state clcpStatus = IDLE before sending CONTINUEB(ZCHECK_TC_COUNTER).
+
+7020:
+Insert system error in local checkpoint participant at reception of
+COPY_GCIREQ.
+
+7075: Master
+Don't send any LCP_FRAG_ORD(last=true)
+And crash when all have "not" been sent
+
+8000: Crash particpant when receiving TCGETOPSIZEREQ
+8001: Crash particpant when receiving TC_CLOPSIZEREQ
+5010: Crash any when receiving LCP_FRAGORD
+
+7021: Crash in master when receiving START_LCP_REQ
+7022: Crash in !master when receiving START_LCP_REQ
+
+7023: Crash in master when sending START_LCP_CONF
+7024: Crash in !master when sending START_LCP_CONF
+
+7025: Crash in master when receiving LCP_FRAG_REP
+7016: Crash in !master when receiving LCP_FRAG_REP
+
+7026: Crash in master when changing state to LCP_TAB_COMPLETED
+7017: Crash in !master when changing state to LCP_TAB_COMPLETED
+
+7027: Crash in master when changing state to LCP_TAB_SAVED
+7018: Crash in master when changing state to LCP_TAB_SAVED
+
+ERROR CODES FOR TESTING NODE FAILURE, FAILURE IN COPY FRAGMENT PROCESS:
+-----------------------------------------------------------------------
+
+5002:
+Insert node failure in starting node when receiving a tuple copied from the copy node
+as part of copy fragment process.
+5003:
+Insert node failure when receiving ABORT signal.
+
+5004:
+Insert node failure handling when receiving COMMITREQ.
+
+5005:
+Insert node failure handling when receiving COMPLETEREQ.
+
+5006:
+Insert node failure handling when receiving ABORTREQ.
+
+These error code can be combined with error codes for testing time-out
+handling in DBTC to ensure that node failures are also well handled in
+time-out handling. They can also be used to test multiple node failure
+handling.
+
+ERROR CODES FOR TESTING TIME-OUT HANDLING IN DBLQH
+-------------------------------------------------
+5011:
+Delay execution of COMMIT signal 2 seconds to generate time-out.
+
+5012 (use 5017):
+First delay execution of COMMIT signal 2 seconds to generate COMMITREQ.
+Delay execution of COMMITREQ signal 2 seconds to generate time-out.
+
+5013:
+Delay execution of COMPLETE signal 2 seconds to generate time-out.
+
+5014 (use 5018):
+First delay execution of COMPLETE signal 2 seconds to generate COMPLETEREQ.
+Delay execution of COMPLETEREQ signal 2 seconds to generate time-out.
+
+5015:
+Delay execution of ABORT signal 2 seconds to generate time-out.
+
+5016: (ABORTREQ only as part of take-over)
+Delay execution of ABORTREQ signal 2 seconds to generate time-out.
+
+5031: lqhKeyRef, ZNO_TC_CONNECT_ERROR
+5032: lqhKeyRef, ZTEMPORARY_REDO_LOG_FAILURE
+5033: lqhKeyRef, ZTAIL_PROBLEM_IN_LOG_ERROR
+
+ERROR CODES FOR TESTING TIME-OUT HANDLING IN DBTC
+-------------------------------------------------
+8040:
+Delay execution of ABORTED signal 2 seconds to generate time-out.
+
+8041:
+Delay execution of COMMITTED signal 2 seconds to generate time-out.
+8042 (use 8046):
+Delay execution of COMMITTED signal 2 seconds to generate COMMITCONF.
+Delay execution of COMMITCONF signal 2 seconds to generate time-out.
+
+8043:
+Delay execution of COMPLETED signal 2 seconds to generate time-out.
+
+8044 (use 8047):
+Delay execution of COMPLETED signal 2 seconds to generate COMPLETECONF.
+Delay execution of COMPLETECONF signal 2 seconds to generate time-out.
+
+8045: (ABORTCONF only as part of take-over)
+Delay execution of ABORTCONF signal 2 seconds to generate time-out.
+
+ERROR CODES FOR TESTING TIME-OUT HANDLING IN DBTC
+-------------------------------------------------
+
+8003: Throw away a LQHKEYCONF in state STARTED
+8004: Throw away a LQHKEYCONF in state RECEIVING
+8005: Throw away a LQHKEYCONF in state REC_COMMITTING
+8006: Throw away a LQHKEYCONF in state START_COMMITTING
+
+8007: Ignore send of LQHKEYREQ in state STARTED
+8008: Ignore send of LQHKEYREQ in state START_COMMITTING
+
+8009: Ignore send of LQHKEYREQ+ATTRINFO in state STARTED
+8010: Ignore send of LQHKEYREQ+ATTRINFO in state START_COMMITTING
+
+8011: Abort at send of CONTINUEB(ZSEND_ATTRINFO) in state STARTED
+8012: Abort at send of CONTINUEB(ZSEND_ATTRINFO) in state START_COMMITTING
+
+8013: Ignore send of CONTINUEB(ZSEND_COMPLETE_LOOP) (should crash eventually)
+8014: Ignore send of CONTINUEB(ZSEND_COMMIT_LOOP) (should crash eventually)
+
+8015: Ignore ATTRINFO signal in DBTC in state REC_COMMITTING
+8016: Ignore ATTRINFO signal in DBTC in state RECEIVING
+
+8017: Return immediately from DIVERIFYCONF (should crash eventually)
+8018: Throw away a COMMITTED signal
+8019: Throw away a COMPLETED signal
+
+TESTING TAKE-OVER FUNCTIONALITY IN DBTC
+---------------------------------------
+
+8002: Crash when sending LQHKEYREQ
+8029: Crash when receiving LQHKEYCONF
+8030: Crash when receiving COMMITTED
+8031: Crash when receiving COMPLETED
+8020: Crash when all COMMITTED has arrived
+8021: Crash when all COMPLETED has arrived
+8022: Crash when all LQHKEYCONF has arrived
+
+COMBINATION OF TIME-OUT + CRASH
+-------------------------------
+
+8023 (use 8024): Ignore LQHKEYCONF and crash when ABORTED signal arrives by setting 8024
+8025 (use 8026): Ignore COMMITTED and crash when COMMITCONF signal arrives by setting 8026
+8027 (use 8028): Ignore COMPLETED and crash when COMPLETECONF signal arrives by setting 8028
+
+ABORT OF TCKEYREQ
+-----------------
+
+8032: No free TC records any more
+
+
+CMVMI
+-----
+9000 Set RestartOnErrorInsert to restart -n
+9999 Crash system immediatly
+
+Test Crashes in handling node restarts
+--------------------------------------
+
+7121: Crash after receiving permission to start (START_PERMCONF) in starting
+ node.
+7122: Crash master when receiving request for permission to start (START_PERMREQ).
+7123: Crash any non-starting node when receiving information about a starting node
+ (START_INFOREQ)
+7124: Respond negatively on an info request (START_INFOREQ)
+7125: Stop an invalidate Node LCP process in the middle to test if START_INFOREQ
+ stopped by long-running processes are handled in a correct manner.
+7126: Allow node restarts for all nodes (used in conjunction with 7025)
+7127: Crash when receiving a INCL_NODEREQ message.
+7128: Crash master after receiving all INCL_NODECONF from all nodes
+7129: Crash master after receiving all INCL_NODECONF from all nodes and releasing
+ the lock on the dictionary
+7130: Crash starting node after receiving START_MECONF
+7131: Crash when receiving START_COPYREQ in master node
+7132: Crash when receiving START_COPYCONF in starting node
+
+DICT:
+6000 Crash during NR when receiving DICTSTARTREQ
+6001 Crash during NR when receiving SCHEMA_INFO
+6002 Crash during NR soon after sending GET_TABINFO_REQ
+
+LQH:
+5026 Crash when receiving COPY_ACTIVEREQ
+5027 Crash when receiving STAT_RECREQ
+
+Test Crashes in handling take over
+----------------------------------
+
+7133: Crash when receiving START_TOREQ
+7134: Crash master after receiving all START_TOCONF
+7135: Crash master after copying table 0 to starting node
+7136: Crash master after completing copy of tables
+7137: Crash master after adding a fragment before copying it
+7138: Crash when receiving CREATE_FRAGREQ in prepare phase
+7139: Crash when receiving CREATE_FRAGREQ in commit phase
+7140: Crash master when receiving all CREATE_FRAGCONF in prepare phase
+7141: Crash master when receiving all CREATE_FRAGCONF in commit phase
+7142: Crash master when receiving COPY_FRAGCONF
+7143: Crash master when receiving COPY_ACTIVECONF
+7144: Crash when receiving END_TOREQ
+7145: Crash master after receiving first END_TOCONF
+7146: Crash master after receiving all END_TOCONF
+7147: Crash master after receiving first START_TOCONF
+7148: Crash master after receiving first CREATE_FRAGCONF
+7152: Crash master after receiving first UPDATE_TOCONF
+7153: Crash master after receiving all UPDATE_TOCONF
+7154: Crash when receiving UPDATE_TOREQ
+7155: Crash master when completing writing start take over info
+7156: Crash master when completing writing end take over info
+
+Test failures in various states in take over functionality
+----------------------------------------------------------
+7157: Block take over at start take over
+7158: Block take over at sending of START_TOREQ
+7159: Block take over at selecting next fragment
+7160: Block take over at creating new fragment
+7161: Block take over at sending of CREATE_FRAGREQ in prepare phase
+7162: Block take over at sending of CREATE_FRAGREQ in commit phase
+7163: Block take over at sending of UPDATE_TOREQ at end of copy frag
+7164: Block take over at sending of END_TOREQ
+7169: Block take over at sending of UPDATE_TOREQ at end of copy
+
+5008: Crash at reception of EMPTY_LCPREQ (at master take over after NF)
+5009: Crash at sending of EMPTY_LCPCONF (at master take over after NF)
+
+Test Crashes in Handling Graceful Shutdown
+------------------------------------------
+7065: Crash when receiving STOP_PERMREQ in master
+7066: Crash when receiving STOP_PERMREQ in slave
+7067: Crash when receiving DIH_SWITCH_REPLICA_REQ
+7068: Crash when receiving DIH_SWITCH_REPLICA_CONF
+
+
+Backup Stuff:
+------------------------------------------
+10001: Crash on NODE_FAILREP in Backup coordinator
+10002: Crash on NODE_FAILREP when coordinatorTakeOver
+10003: Crash on PREP_CREATE_TRIG_{CONF/REF} (only coordinator)
+10004: Crash on START_BACKUP_{CONF/REF} (only coordinator)
+10005: Crash on CREATE_TRIG_{CONF/REF} (only coordinator)
+10006: Crash on WAIT_GCP_REF (only coordinator)
+10007: Crash on WAIT_GCP_CONF (only coordinator)
+10008: Crash on WAIT_GCP_CONF during start of backup (only coordinator)
+10009: Crash on WAIT_GCP_CONF during stop of backup (only coordinator)
+10010: Crash on BACKUP_FRAGMENT_CONF (only coordinator)
+10011: Crash on BACKUP_FRAGMENT_REF (only coordinator)
+10012: Crash on DROP_TRIG_{CONF/REF} (only coordinator)
+10013: Crash on STOP_BACKUP_{CONF/REF} (only coordinator)
+10014: Crash on DEFINE_BACKUP_REQ (participant)
+10015: Crash on START_BACKUP_REQ (participant)
+10016: Crash on BACKUP_FRAGMENT_REQ (participant)
+10017: Crash on SCAN_FRAGCONF (participant)
+10018: Crash on FSAPPENDCONF (participant)
+10019: Crash on TRIG_ATTRINFO (participant)
+10020: Crash on STOP_BACKUP_REQ (participant)
+10021: Crash on NODE_FAILREP in participant not becoming coordinator
+
+10022: Fake no backup records at DEFINE_BACKUP_REQ (participant)
+10023: Abort backup by error at reception of UTIL_SEQUENCE_CONF (code 300)
+10024: Abort backup by error at reception of DEFINE_BACKUP_CONF (code 301)
+10025: Abort backup by error at reception of CREATE_TRIG_CONF last (code 302)
+10026: Abort backup by error at reception of START_BACKUP_CONF (code 303)
+10027: Abort backup by error at reception of DEFINE_BACKUP_REQ at master (code 304)
+10028: Abort backup by error at reception of BACKUP_FRAGMENT_CONF at master (code 305)
+10029: Abort backup by error at reception of FSAPPENDCONF in slave (FileOrScanError = 5)
+10030: Simulate buffer full from trigger execution => abort backup
+
+11001: Send UTIL_SEQUENCE_REF (in master)
+
+5028: Crash when receiving LQHKEYREQ (in non-master)
+
+Drop Table/Index:
+-----------------
+4001: Crash on REL_TABMEMREQ in TUP
+4002: Crash on DROP_TABFILEREQ in TUP
+4003: Fail next trigger create in TUP
+8033: Fail next trigger create in TC
+8034: Fail next index create in TC
+
+System Restart:
+---------------
+
+5020: Force system to read pages form file when executing prepare operation record
+3000: Delay writing of datapages in ACC when LCP is started
+4000: Delay writing of datapages in TUP when LCP is started
+7070: Set TimeBetweenLcp to min value
+7071: Set TimeBetweenLcp to max value
+7072: Split START_FRAGREQ into several log nodes
+7073: Don't include own node in START_FRAGREQ
+7074: 7072 + 7073
+
+Scan:
+------
+
+5021: Crash when receiving SCAN_NEXTREQ if sender is own node
+5022: Crash when receiving SCAN_NEXTREQ if sender is NOT own node
+5023: Drop SCAN_NEXTREQ if sender is own node
+5024: Drop SCAN_NEXTREQ if sender is NOT own node
+5025: Delay SCAN_NEXTREQ 1 second if sender is NOT own node
+5030: Drop all SCAN_NEXTREQ until node is shutdown with SYSTEM_ERROR
+ because of scan fragment timeout
+
+Test routing of signals:
+-----------------------
+4006: Turn on routing of TRANSID_AI signals from TUP
+5029: Turn on routing of KEYINFO20 signals from LQH
+
+Ordered index:
+--------------
+
+Dbdict:
+-------
+14000 Crash in participant @ CreateTabReq::Prepare
+14001 Crash in participant @ CreateTabReq::Commit
+14002 Crash in participant @ CreateTabReq::CreateDrop
diff --git a/ndb/src/kernel/blocks/Makefile b/ndb/src/kernel/blocks/Makefile
new file mode 100644
index 00000000000..ce554dfc3b8
--- /dev/null
+++ b/ndb/src/kernel/blocks/Makefile
@@ -0,0 +1,28 @@
+#--------------------------------------------------------------------------
+#
+# Name Makefile
+#
+#
+#
+# List subdirectories to be travered
+include .defs.mk
+
+DIRS := \
+ cmvmi \
+ dbacc \
+ dbdict \
+ dbdih \
+ dblqh \
+ dbtc \
+ dbtup \
+ ndbfs \
+ ndbcntr \
+ qmgr \
+ trix \
+ backup \
+ dbutil \
+ suma \
+ grep \
+ dbtux
+
+include ${NDB_TOP}/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/NodeRestart.new.txt b/ndb/src/kernel/blocks/NodeRestart.new.txt
new file mode 100644
index 00000000000..00ab8f0c208
--- /dev/null
+++ b/ndb/src/kernel/blocks/NodeRestart.new.txt
@@ -0,0 +1,82 @@
+
+Master DIH Starting DIH Starting DICT
+---------------------- ---------------------- ---------------------
+
+ Check for sysfile
+ DIH_RESTARTCONF ->
+
+******************************************************************************
+* NDB_STTOR interal startphase = 1
+******************************************************************************
+
+ Read schema file
+
+******************************************************************************
+* NDB_STTOR interal startphase = 2
+******************************************************************************
+
+ <- START_PERMREQ
+
+XXX
+
+START_PERMCONF ->
+
+******************************************************************************
+* NDB_STTOR interal startphase = 3
+******************************************************************************
+
+ <- START_MEREQ
+
+START_RECREQ -> starting LQH
+ <- START_RECCONF
+
+For each table
+ COPY_TABREQ -> starting DIH
+
+DICTSTARTREQ -> starting DICT
+ GET_SCHEMA_INFOREQ
+ (to master DICT)
+
+ ->SCHEMA_INFO
+ (schema file)
+
+ 1) For each table
+ If TableStatus OK
+ ReadTableFile
+ else
+ GET_TABINFOREQ
+ 2) DIADDTABREQ->DIH
+
+ For each local frag
+ ADD_FRAG_REQ -> local DICT
+ DI_ADD_TAB_CONF
+ <- DICTSTARTCONF
+
+INCL_NODEREQ -> all DIH
+
+START_MECONF -> starting DIH
+ (including sysfile)
+
+******************************************************************************
+* NDB_STTOR interal startphase = 5
+******************************************************************************
+
+ <- START_COPYREQ
+
+START_TOREQ -> all DIH
+
+For each fragment
+ CREATE_FRAGREQ -> all DIH
+
+ COPY-DATA (LQHKEYREQ++)
+
+ UPDATE_TOREQ -> all DIH
+
+ COPY_ACTIVEREQ -> starting LQH
+
+ CREATE_FRAGREQ -> all DIH
+
+START_COPYCONF ->
+
+LOCAL CHECKPOINT
+
diff --git a/ndb/src/kernel/blocks/NodeRestart.txt b/ndb/src/kernel/blocks/NodeRestart.txt
new file mode 100644
index 00000000000..e9f277bb39e
--- /dev/null
+++ b/ndb/src/kernel/blocks/NodeRestart.txt
@@ -0,0 +1,80 @@
+
+Master DIH Starting DIH Starting DICT
+---------------------- ---------------------- ---------------------
+
+ Check for sysfile
+ DIH_RESTARTCONF ->
+
+******************************************************************************
+* NDB_STTOR interal startphase = 1
+******************************************************************************
+
+ Read schema file
+
+******************************************************************************
+* NDB_STTOR interal startphase = 2
+******************************************************************************
+
+ <- START_PERMREQ
+
+XXX
+
+START_PERMCONF ->
+
+******************************************************************************
+* NDB_STTOR interal startphase = 3
+******************************************************************************
+
+ <- START_MEREQ
+
+START_RECREQ -> starting LQH
+ <- START_RECCONF
+
+DICTSTARTREQ -> starting DICT
+ GET_SCHEMA_INFOREQ
+ (to master DICT)
+
+ ->SCHEMA_INFO
+ (schema file)
+
+ 1) For each table
+ 1) If TableStatus match
+ ReadTableFile
+ else
+ GET_TABINFOREQ
+
+ <- DICTSTARTCONF
+
+For each table
+ COPY_TABREQ -> starting DIH
+
+INCL_NODEREQ -> all DIH
+
+START_MECONF -> starting DIH
+ (including sysfile)
+
+******************************************************************************
+* NDB_STTOR interal startphase = 5
+******************************************************************************
+
+ <- START_COPYREQ
+
+START_TOREQ -> all DIH
+
+For each fragment
+ ADD_FRAG_REQ -> local DICT -> LQHFRAGREQ -> starting LQH
+
+ CREATE_FRAGREQ -> all DIH
+
+ COPY-DATA (LQHKEYREQ++)
+
+ UPDATE_TOREQ -> all DIH
+
+ COPY_ACTIVEREQ -> starting LQH
+
+ CREATE_FRAGREQ -> all DIH
+
+START_COPYCONF ->
+
+LOCAL CHECKPOINT
+
diff --git a/ndb/src/kernel/blocks/Start.txt b/ndb/src/kernel/blocks/Start.txt
new file mode 100644
index 00000000000..545296d44f1
--- /dev/null
+++ b/ndb/src/kernel/blocks/Start.txt
@@ -0,0 +1,97 @@
+
+--- Start phase 1 - Qmgr -------------------------------------------
+
+1) Set timer 1 - TimeToWaitAlive
+
+2) Send CM_REGREQ to all connected(and connecting) nodes
+
+3) Wait until -
+a) The precident answers CM_REGCONF
+b) All nodes has answered and I'm the candidate -> election won
+c) 30s has passed and I'm the candidate -> election won
+d) TimeToWaitAlive has passed -> Failure to start
+
+When receiving CM_REGCONF
+4) Send CM_NODEINFOREQ to all connected(and connecting) nodes
+ reported in CM_REGCONF
+
+5) Wait until -
+a) All CM_NODEINFO_CONF has arrived
+b) TimeToWaitAlive has passed -> Failure to start
+
+6) Send CM_ACKADD to president
+
+7) Wait until -
+a) Receive CM_ADD(CommitNew) from president -> I'm in the qmgr cluster
+b) TimeToWaitAlive has passed -> Failure to start
+
+NOTE:
+30s is hardcoded in 3c.
+TimeToWaitAlive should be atleast X sec greater than 30s. i.e. 30+X sec
+to support "partial starts"
+
+NOTE:
+In 3b, a more correct number (instead of all) would be
+N-NG+1 where N is #nodes and NG is #node groups = (N/R where R is # replicas)
+But Qmgr has no notion about node groups or replicas
+
+--- Start phase X - Qmgr -------------------------------------------
+
+President - When accepting a CM_REGREQ
+1) Send CM_REGCONF to starting node
+2) Send CM_ADD(Prepare) to all started nodes + starting node
+3) Send CM_ADD(AddCommit) to all started nodes
+4) Send CM_ADD(CommitNew) to starting node
+
+Cluster participant -
+1) Wait for both CM_NODEINFOREQ from starting and CM_ADD(Prepare) from pres.
+2) Send CM_ACKADD(Prepare)
+3) Wait for CM_ADD(AddCommit) from president
+4) Send CM_ACKADD(AddCommit)
+
+--- Start phase 2 - NdbCntr ----------------------------------------
+
+- Use same TimeToWaitAliveTimer
+
+1) Check sysfile (DIH_RESTART_REQ)
+2) Read nodes (from Qmgr) P = qmgr president
+
+3) Send CNTR_MASTER_REQ to cntr(P)
+ including info in DIH_RESTART_REF/CONF
+
+4) Wait until -
+a) Receiving CNTR_MASTER_CONF -> continue
+b) Receiving CNTR_MASTER_REF -> P = node specified in REF, goto 3
+c) TimeToWaitAlive has passed -> Failure to start
+
+4) Run ndb-startphase 1
+
+--
+Initial start/System restart NdbCntr (on qmgr president node)
+
+1) Wait until -
+a) Receiving all CNTR_MASTER_REQ (all = those in READ_NODES_CONF)
+b) TimeToWaitAlive has passed -> Failure to start
+
+2) Wait until -
+a) Enough nodes (at least 1 in each node group and 1 full node group)
+ has sent me CNTR_MASTER_REQ
+b) TimeToWaitAlive has passed -> Failure to start
+
+3) Decide what kind of start to perform (initial / system restart)
+ Decide who should be the master (the one with greatest GCI)
+ Send CNTR_MASTER_CONF(initial/system restart) to all nodes included in start
+
+--
+Running NdbCntr
+
+When receiving CNTR_MASTER_REQ
+1) If I'm not master send CNTR_MASTER_REF (including master node id)
+2) If I'm master
+ Coordinate parallell node restarts
+ send CNTR_MASTER_CONF (node restart)
+
+NOTE:
+2a Specified with a command line/config parameter the system could
+ start using only one node in each node group (if possible w.r.t LCP/GCP)
+
diff --git a/ndb/src/kernel/blocks/SystemRestart.new.txt b/ndb/src/kernel/blocks/SystemRestart.new.txt
new file mode 100644
index 00000000000..3738de28df8
--- /dev/null
+++ b/ndb/src/kernel/blocks/SystemRestart.new.txt
@@ -0,0 +1,61 @@
+
+DIH DICT CNTR
+---------------------- ---------------------- ---------------------
+ <- DIHRESTARTREQ
+Check for sysfile
+DIH_RESTARTCONF ->
+
+NDB_STTORY -> DICT (sp=1)
+ Read schema file
+
+******************************************************************************
+* Elect master
+******************************************************************************
+
+-- Master DIH --
+
+Read sysfile
+
+COPY_GCIREQ -> all DIHs
+
+DICTSTARTREQ -> local DICT (master)
+
+ master
+ ======
+ For each table (that should be started)
+ 1) ReadTableFile
+ 2) DI_ADD_TAB_REQ -> local DIH
+
+1) ReadTableFile (DIH)
+2) COPY_TABREQ -> all DIH (but self)
+3) For each local frag
+ ADD_FRAG_REQ -> local DICT
+4) DI_ADD_TAB_CONF
+
+ SCHEMA_INFO -> all DICTs
+ Info = schema file
+
+ Participant
+ ===========
+ 1) For each table
+ 1) If TableStatus match
+ ReadTableFile
+ else
+ GET_TABINFOREQ
+ 2) WriteTableFile
+ 3) Parse Table Data
+ 4) DI_ADD_TAB_REQ -> local DIH
+
+ <- SCHEMA_INFOCONF
+
+
+ <- DICTSTARTCONF
+
+For each fragment
+ IF Fragment is logged
+ START_FRAGREQ -> LQH x
+
+ START_RECREQ -> all LQH
+ Note does not wait for START_FRAGCONF
+
+NDB_STARTCONF ->
diff --git a/ndb/src/kernel/blocks/SystemRestart.txt b/ndb/src/kernel/blocks/SystemRestart.txt
new file mode 100644
index 00000000000..235dfb968fa
--- /dev/null
+++ b/ndb/src/kernel/blocks/SystemRestart.txt
@@ -0,0 +1,61 @@
+
+NDBCNTR DIH DICT
+---------------------- ---------------------- ---------------
+DIH_RESTARTREQ -> DIH
+ Check for sysfile
+ <- DIH_RESTARTCONF
+
+NDB_STTORY -> DICT
+sp = 1
+ Read schema file
+
+---- Master
+
+NDB_STARTREQ -> DIH
+ Read sysfile
+
+ COPY_GCIREQ -> all DIHs
+
+ DICTSTARTREQ -> local DICT
+ local
+ ======
+ SCHEMA_INFO -> all DICTs
+ Info = schema file
+
+ Participant
+ ===========
+ 1) For each table
+ If TableStatus match
+ ReadTableFile
+ else
+ GET_TABINFOREQ
+
+ <- SCHEMA_INFOCONF
+
+ local
+ ======
+ For each table
+ DIHSTARTTABREQ -> DIH
+
+ <- DICTSTARTCONF
+
+ For each table (STARTED)
+ Read table description
+ from disk
+
+ For each fragment
+ IF Fragment dont have LCP
+ ADD_FRAGREQ -> local DICT
+ 1) LQHFRAGREQ -> LQH x
+ 2) For each attribute
+ LQHADDATTREQ
+ IF Fragment is logged
+ START_FRAGREQ -> LQH x
+
+ START_RECREQ -> all LQH
+ Note does not wait for START_FRAGCONF
+
+ For each table
+ COPY_TABREQ -> all DIH (but self)
+
+ <- NDB_STARTCONF
diff --git a/ndb/src/kernel/blocks/backup/Backup.cpp b/ndb/src/kernel/blocks/backup/Backup.cpp
new file mode 100644
index 00000000000..4342a9d6d94
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/Backup.cpp
@@ -0,0 +1,4691 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Backup.hpp"
+
+#include <ndb_version.h>
+
+#include <NdbTCP.h>
+#include <Bitmask.hpp>
+
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/ReadNodesConf.hpp>
+
+#include <signaldata/ScanFrag.hpp>
+
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/ListTables.hpp>
+
+#include <signaldata/FsOpenReq.hpp>
+#include <signaldata/FsAppendReq.hpp>
+#include <signaldata/FsCloseReq.hpp>
+#include <signaldata/FsConf.hpp>
+#include <signaldata/FsRef.hpp>
+#include <signaldata/FsRemoveReq.hpp>
+
+#include <signaldata/BackupImpl.hpp>
+#include <signaldata/BackupSignalData.hpp>
+#include <signaldata/BackupContinueB.hpp>
+
+#include <signaldata/UtilSequence.hpp>
+
+#include <signaldata/CreateTrig.hpp>
+#include <signaldata/AlterTrig.hpp>
+#include <signaldata/DropTrig.hpp>
+#include <signaldata/FireTrigOrd.hpp>
+#include <signaldata/TrigAttrInfo.hpp>
+#include <AttributeHeader.hpp>
+
+#include <signaldata/WaitGCP.hpp>
+
+#include <NdbTick.h>
+
+static NDB_TICKS startTime;
+
+static const Uint32 BACKUP_SEQUENCE = 0x1F000000;
+
+#ifdef VM_TRACE
+#define DEBUG_OUT(x) ndbout << x << endl
+#else
+#define DEBUG_OUT(x)
+#endif
+
+//#define DEBUG_ABORT
+
+//---------------------------------------------------------
+// Ignore this since a completed abort could have preceded
+// this message.
+//---------------------------------------------------------
+#define slaveAbortCheck() \
+if ((ptr.p->backupId != backupId) || \
+ (ptr.p->slaveState.getState() == ABORTING)) { \
+ jam(); \
+ return; \
+}
+
+#define masterAbortCheck() \
+if ((ptr.p->backupId != backupId) || \
+ (ptr.p->masterData.state.getState() == ABORTING)) { \
+ jam(); \
+ return; \
+}
+
+#define defineSlaveAbortCheck() \
+ if (ptr.p->slaveState.getState() == ABORTING) { \
+ jam(); \
+ closeFiles(signal, ptr); \
+ return; \
+ }
+
+static Uint32 g_TypeOfStart = NodeState::ST_ILLEGAL_TYPE;
+
+void
+Backup::execSTTOR(Signal* signal)
+{
+ jamEntry();
+
+ const Uint32 startphase = signal->theData[1];
+ const Uint32 typeOfStart = signal->theData[7];
+
+ if (startphase == 3) {
+ jam();
+ g_TypeOfStart = typeOfStart;
+ signal->theData[0] = reference();
+ sendSignal(NDBCNTR_REF, GSN_READ_NODESREQ, signal, 1, JBB);
+ return;
+ }//if
+
+ if(startphase == 7 && g_TypeOfStart == NodeState::ST_INITIAL_START &&
+ c_masterNodeId == getOwnNodeId()){
+ jam();
+ createSequence(signal);
+ return;
+ }//if
+
+ sendSTTORRY(signal);
+ return;
+}//Dbdict::execSTTOR()
+
+void
+Backup::execREAD_NODESCONF(Signal* signal)
+{
+ jamEntry();
+ ReadNodesConf * conf = (ReadNodesConf *)signal->getDataPtr();
+
+ c_aliveNodes.clear();
+
+ Uint32 count = 0;
+ for (Uint32 i = 0; i<MAX_NDB_NODES; i++) {
+ jam();
+ if(NodeBitmask::get(conf->allNodes, i)){
+ jam();
+ count++;
+
+ NodePtr node;
+ ndbrequire(c_nodes.seize(node));
+
+ node.p->nodeId = i;
+ if(NodeBitmask::get(conf->inactiveNodes, i)) {
+ jam();
+ node.p->alive = 0;
+ } else {
+ jam();
+ node.p->alive = 1;
+ c_aliveNodes.set(i);
+ }//if
+ }//if
+ }//for
+ c_masterNodeId = conf->masterNodeId;
+ ndbrequire(count == conf->noOfNodes);
+ sendSTTORRY(signal);
+}
+
+void
+Backup::sendSTTORRY(Signal* signal)
+{
+ signal->theData[0] = 0;
+ signal->theData[3] = 1;
+ signal->theData[4] = 3;
+ signal->theData[5] = 7;
+ signal->theData[6] = 255; // No more start phases from missra
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 7, JBB);
+}
+
+void
+Backup::createSequence(Signal* signal)
+{
+ UtilSequenceReq * req = (UtilSequenceReq*)signal->getDataPtrSend();
+
+ req->senderData = RNIL;
+ req->sequenceId = BACKUP_SEQUENCE;
+ req->requestType = UtilSequenceReq::Create;
+
+ sendSignal(DBUTIL_REF, GSN_UTIL_SEQUENCE_REQ,
+ signal, UtilSequenceReq::SignalLength, JBB);
+}
+
+void
+Backup::execCONTINUEB(Signal* signal)
+{
+ jamEntry();
+ const Uint32 Tdata0 = signal->theData[0];
+ const Uint32 Tdata1 = signal->theData[1];
+ const Uint32 Tdata2 = signal->theData[2];
+
+ switch(Tdata0) {
+ case BackupContinueB::START_FILE_THREAD:
+ case BackupContinueB::BUFFER_UNDERFLOW:
+ {
+ jam();
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, Tdata1);
+ checkFile(signal, filePtr);
+ return;
+ }
+ break;
+ case BackupContinueB::BUFFER_FULL_SCAN:
+ {
+ jam();
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, Tdata1);
+ checkScan(signal, filePtr);
+ return;
+ }
+ break;
+ case BackupContinueB::BUFFER_FULL_FRAG_COMPLETE:
+ {
+ jam();
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, Tdata1);
+ fragmentCompleted(signal, filePtr);
+ return;
+ }
+ break;
+ case BackupContinueB::BUFFER_FULL_META:
+ {
+ jam();
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, Tdata1);
+
+ if (ptr.p->slaveState.getState() == ABORTING) {
+ jam();
+ closeFiles(signal, ptr);
+ return;
+ }//if
+ BackupFilePtr filePtr;
+ ptr.p->files.getPtr(filePtr, ptr.p->ctlFilePtr);
+ FsBuffer & buf = filePtr.p->operation.dataBuffer;
+
+ if(buf.getFreeSize() + buf.getMinRead() < buf.getUsableSize()) {
+ jam();
+ TablePtr tabPtr;
+ c_tablePool.getPtr(tabPtr, Tdata2);
+
+ DEBUG_OUT("Backup - Buffer full - " << buf.getFreeSize()
+ << " + " << buf.getMinRead()
+ << " < " << buf.getUsableSize()
+ << " - tableId = " << tabPtr.p->tableId);
+
+ signal->theData[0] = BackupContinueB::BUFFER_FULL_META;
+ signal->theData[1] = Tdata1;
+ signal->theData[2] = Tdata2;
+ sendSignalWithDelay(BACKUP_REF, GSN_CONTINUEB, signal, 100, 3);
+ return;
+ }//if
+
+ TablePtr tabPtr;
+ c_tablePool.getPtr(tabPtr, Tdata2);
+ GetTabInfoReq * req = (GetTabInfoReq *)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = ptr.i;
+ req->requestType = GetTabInfoReq::RequestById |
+ GetTabInfoReq::LongSignalConf;
+ req->tableId = tabPtr.p->tableId;
+ sendSignal(DBDICT_REF, GSN_GET_TABINFOREQ, signal,
+ GetTabInfoReq::SignalLength, JBB);
+ return;
+ }
+ default:
+ ndbrequire(0);
+ }//switch
+}
+
+void
+Backup::execDUMP_STATE_ORD(Signal* signal)
+{
+ jamEntry();
+
+ if(signal->theData[0] == 20){
+ if(signal->length() > 1){
+ c_defaults.m_dataBufferSize = (signal->theData[1] * 1024 * 1024);
+ }
+ if(signal->length() > 2){
+ c_defaults.m_logBufferSize = (signal->theData[2] * 1024 * 1024);
+ }
+ if(signal->length() > 3){
+ c_defaults.m_minWriteSize = signal->theData[3] * 1024;
+ }
+ if(signal->length() > 4){
+ c_defaults.m_maxWriteSize = signal->theData[4] * 1024;
+ }
+
+ infoEvent("Backup: data: %d log: %d min: %d max: %d",
+ c_defaults.m_dataBufferSize,
+ c_defaults.m_logBufferSize,
+ c_defaults.m_minWriteSize,
+ c_defaults.m_maxWriteSize);
+ return;
+ }
+ if(signal->theData[0] == 21){
+ BackupReq * req = (BackupReq*)signal->getDataPtrSend();
+ req->senderData = 23;
+ req->backupDataLen = 0;
+ sendSignal(BACKUP_REF, GSN_BACKUP_REQ,signal,BackupReq::SignalLength, JBB);
+ startTime = NdbTick_CurrentMillisecond();
+ return;
+ }
+
+ if(signal->theData[0] == 22){
+ const Uint32 seq = signal->theData[1];
+ FsRemoveReq * req = (FsRemoveReq *)signal->getDataPtrSend();
+ req->userReference = reference();
+ req->userPointer = 23;
+ req->directory = 1;
+ req->ownDirectory = 1;
+ FsOpenReq::setVersion(req->fileNumber, 2);
+ FsOpenReq::setSuffix(req->fileNumber, FsOpenReq::S_CTL);
+ FsOpenReq::v2_setSequence(req->fileNumber, seq);
+ FsOpenReq::v2_setNodeId(req->fileNumber, getOwnNodeId());
+ sendSignal(NDBFS_REF, GSN_FSREMOVEREQ, signal,
+ FsRemoveReq::SignalLength, JBA);
+ return;
+ }
+
+ if(signal->theData[0] == 23){
+ /**
+ * Print records
+ */
+ BackupRecordPtr ptr;
+ for(c_backups.first(ptr); ptr.i != RNIL; c_backups.next(ptr)){
+ infoEvent("BackupRecord %d: BackupId: %d MasterRef: %x ClientRef: %x",
+ ptr.i, ptr.p->backupId, ptr.p->masterRef, ptr.p->clientRef);
+ if(ptr.p->masterRef == reference()){
+ infoEvent(" MasterState: %d State: %d",
+ ptr.p->masterData.state.getState(),
+ ptr.p->slaveState.getState());
+ } else {
+ infoEvent(" State: %d", ptr.p->slaveState.getState());
+ }
+ BackupFilePtr filePtr;
+ for(ptr.p->files.first(filePtr); filePtr.i != RNIL;
+ ptr.p->files.next(filePtr)){
+ jam();
+ infoEvent(" file %d: type: %d open: %d running: %d done: %d scan: %d",
+ filePtr.i, filePtr.p->fileType, filePtr.p->fileOpened,
+ filePtr.p->fileRunning,
+ filePtr.p->fileDone, filePtr.p->scanRunning);
+ }
+ }
+ }
+ if(signal->theData[0] == 24){
+ /**
+ * Print size of records etc.
+ */
+ infoEvent("Backup - dump pool sizes");
+ infoEvent("BackupPool: %d BackupFilePool: %d TablePool: %d",
+ c_backupPool.getSize(), c_backupFilePool.getSize(),
+ c_tablePool.getSize());
+ infoEvent("AttrPool: %d TriggerPool: %d FragmentPool: %d",
+ c_backupPool.getSize(), c_backupFilePool.getSize(),
+ c_tablePool.getSize());
+ infoEvent("PagePool: %d",
+ c_pagePool.getSize());
+
+ }
+}
+
+bool
+Backup::findTable(const BackupRecordPtr & ptr,
+ TablePtr & tabPtr, Uint32 tableId) const
+{
+ for(ptr.p->tables.first(tabPtr);
+ tabPtr.i != RNIL;
+ ptr.p->tables.next(tabPtr)) {
+ jam();
+ if(tabPtr.p->tableId == tableId){
+ jam();
+ return true;
+ }//if
+ }//for
+ tabPtr.i = RNIL;
+ tabPtr.p = 0;
+ return false;
+}
+
+static Uint32 xps(Uint32 x, Uint64 ms)
+{
+ float fx = x;
+ float fs = ms;
+
+ if(ms == 0 || x == 0) {
+ jam();
+ return 0;
+ }//if
+ jam();
+ return ((Uint32)(1000.0f * (fx + fs/2.1f))) / ((Uint32)fs);
+}
+
+struct Number {
+ Number(Uint32 r) { val = r;}
+ Number & operator=(Uint32 r) { val = r; return * this; }
+ Uint32 val;
+};
+
+NdbOut &
+operator<< (NdbOut & out, const Number & val){
+ char p = 0;
+ Uint32 loop = 1;
+ while(val.val > loop){
+ loop *= 1000;
+ p += 3;
+ }
+ if(loop != 1){
+ p -= 3;
+ loop /= 1000;
+ }
+
+ switch(p){
+ case 0:
+ break;
+ case 3:
+ p = 'k';
+ break;
+ case 6:
+ p = 'M';
+ break;
+ case 9:
+ p = 'G';
+ break;
+ default:
+ p = 0;
+ }
+ char str[2];
+ str[0] = p;
+ str[1] = 0;
+ Uint32 tmp = (val.val + (loop >> 1)) / loop;
+#if 1
+ if(p > 0)
+ out << tmp << str;
+ else
+ out << tmp;
+#else
+ out << val.val;
+#endif
+
+ return out;
+}
+
+void
+Backup::execBACKUP_CONF(Signal* signal)
+{
+ jamEntry();
+ BackupConf * conf = (BackupConf*)signal->getDataPtr();
+
+ ndbout_c("Backup %d has started", conf->backupId);
+}
+
+void
+Backup::execBACKUP_REF(Signal* signal)
+{
+ jamEntry();
+ BackupRef * ref = (BackupRef*)signal->getDataPtr();
+
+ ndbout_c("Backup (%d) has NOT started %d", ref->senderData, ref->errorCode);
+}
+
+void
+Backup::execBACKUP_COMPLETE_REP(Signal* signal)
+{
+ jamEntry();
+ BackupCompleteRep* rep = (BackupCompleteRep*)signal->getDataPtr();
+
+ startTime = NdbTick_CurrentMillisecond() - startTime;
+
+ ndbout_c("Backup %d has completed", rep->backupId);
+ const Uint32 bytes = rep->noOfBytes;
+ const Uint32 records = rep->noOfRecords;
+
+ Number rps = xps(records, startTime);
+ Number bps = xps(bytes, startTime);
+
+ ndbout << " Data [ "
+ << Number(records) << " rows "
+ << Number(bytes) << " bytes " << startTime << " ms ] "
+ << " => "
+ << rps << " row/s & " << bps << "b/s" << endl;
+
+ bps = xps(rep->noOfLogBytes, startTime);
+ rps = xps(rep->noOfLogRecords, startTime);
+
+ ndbout << " Log [ "
+ << Number(rep->noOfLogRecords) << " log records "
+ << Number(rep->noOfLogBytes) << " bytes " << startTime << " ms ] "
+ << " => "
+ << rps << " records/s & " << bps << "b/s" << endl;
+
+}
+
+void
+Backup::execBACKUP_ABORT_REP(Signal* signal)
+{
+ jamEntry();
+ BackupAbortRep* rep = (BackupAbortRep*)signal->getDataPtr();
+
+ ndbout_c("Backup %d has been aborted %d", rep->backupId, rep->reason);
+}
+
+const TriggerEvent::Value triggerEventValues[] = {
+ TriggerEvent::TE_INSERT,
+ TriggerEvent::TE_UPDATE,
+ TriggerEvent::TE_DELETE
+};
+
+const char* triggerNameFormat[] = {
+ "NDB$BACKUP_%d_%d_INSERT",
+ "NDB$BACKUP_%d_%d_UPDATE",
+ "NDB$BACKUP_%d_%d_DELETE"
+};
+
+const Backup::State
+Backup::validMasterTransitions[] = {
+ INITIAL, DEFINING,
+ DEFINING, DEFINED,
+ DEFINED, STARTED,
+ STARTED, SCANNING,
+ SCANNING, STOPPING,
+ STOPPING, INITIAL,
+
+ DEFINING, ABORTING,
+ DEFINED, ABORTING,
+ STARTED, ABORTING,
+ SCANNING, ABORTING,
+ STOPPING, ABORTING,
+ ABORTING, ABORTING,
+
+ DEFINING, INITIAL,
+ ABORTING, INITIAL,
+ INITIAL, INITIAL
+};
+
+const Backup::State
+Backup::validSlaveTransitions[] = {
+ INITIAL, DEFINING,
+ DEFINING, DEFINED,
+ DEFINED, STARTED,
+ STARTED, STARTED, // Several START_BACKUP_REQ is sent
+ STARTED, SCANNING,
+ SCANNING, STARTED,
+ STARTED, STOPPING,
+ STOPPING, CLEANING,
+ CLEANING, INITIAL,
+
+ INITIAL, ABORTING, // Node fail
+ DEFINING, ABORTING,
+ DEFINED, ABORTING,
+ STARTED, ABORTING,
+ SCANNING, ABORTING,
+ STOPPING, ABORTING,
+ CLEANING, ABORTING, // Node fail w/ master takeover
+ ABORTING, ABORTING, // Slave who initiates ABORT should have this transition
+
+ ABORTING, INITIAL,
+ INITIAL, INITIAL
+};
+
+const Uint32
+Backup::validSlaveTransitionsCount =
+sizeof(Backup::validSlaveTransitions) / sizeof(Backup::State);
+
+const Uint32
+Backup::validMasterTransitionsCount =
+sizeof(Backup::validMasterTransitions) / sizeof(Backup::State);
+
+void
+Backup::CompoundState::setState(State newState){
+ bool found = false;
+ const State currState = state;
+ for(unsigned i = 0; i<noOfValidTransitions; i+= 2) {
+ jam();
+ if(validTransitions[i] == currState &&
+ validTransitions[i+1] == newState){
+ jam();
+ found = true;
+ break;
+ }
+ }
+ ndbrequire(found);
+
+ if (newState == INITIAL)
+ abortState = INITIAL;
+ if(newState == ABORTING && currState != ABORTING) {
+ jam();
+ abortState = currState;
+ }
+ state = newState;
+#ifdef DEBUG_ABORT
+ if (newState != currState) {
+ ndbout_c("%u: Old state = %u, new state = %u, abort state = %u",
+ id, currState, newState, abortState);
+ }
+#endif
+}
+
+void
+Backup::CompoundState::forceState(State newState)
+{
+ const State currState = state;
+ if (newState == INITIAL)
+ abortState = INITIAL;
+ if(newState == ABORTING && currState != ABORTING) {
+ jam();
+ abortState = currState;
+ }
+ state = newState;
+#ifdef DEBUG_ABORT
+ if (newState != currState) {
+ ndbout_c("%u: FORCE: Old state = %u, new state = %u, abort state = %u",
+ id, currState, newState, abortState);
+ }
+#endif
+}
+
+Backup::Table::Table(ArrayPool<Attribute> & ah,
+ ArrayPool<Fragment> & fh)
+ : attributes(ah), fragments(fh)
+{
+ triggerIds[0] = ILLEGAL_TRIGGER_ID;
+ triggerIds[1] = ILLEGAL_TRIGGER_ID;
+ triggerIds[2] = ILLEGAL_TRIGGER_ID;
+ triggerAllocated[0] = false;
+ triggerAllocated[1] = false;
+ triggerAllocated[2] = false;
+}
+
+/*****************************************************************************
+ *
+ * Node state handling
+ *
+ *****************************************************************************/
+void
+Backup::execNODE_FAILREP(Signal* signal)
+{
+ jamEntry();
+
+ NodeFailRep * rep = (NodeFailRep*)signal->getDataPtr();
+
+ bool doStuff = false;
+ /*
+ Start by saving important signal data which will be destroyed before the
+ process is completed.
+ */
+ NodeId new_master_node_id = rep->masterNodeId;
+ Uint32 theFailedNodes[NodeBitmask::Size];
+ for (Uint32 i = 0; i < NodeBitmask::Size; i++)
+ theFailedNodes[i] = rep->theNodes[i];
+
+// NodeId old_master_node_id = getMasterNodeId();
+ c_masterNodeId = new_master_node_id;
+
+ NodePtr nodePtr;
+ for(c_nodes.first(nodePtr); nodePtr.i != RNIL; c_nodes.next(nodePtr)) {
+ jam();
+ if(NodeBitmask::get(theFailedNodes, nodePtr.p->nodeId)){
+ if(nodePtr.p->alive){
+ jam();
+ ndbrequire(c_aliveNodes.get(nodePtr.p->nodeId));
+ doStuff = true;
+ } else {
+ jam();
+ ndbrequire(!c_aliveNodes.get(nodePtr.p->nodeId));
+ }//if
+ nodePtr.p->alive = 0;
+ c_aliveNodes.clear(nodePtr.p->nodeId);
+ }//if
+ }//for
+
+ if(!doStuff){
+ jam();
+ return;
+ }//if
+
+#ifdef DEBUG_ABORT
+ ndbout_c("****************** Node fail rep ******************");
+#endif
+
+ NodeId newCoordinator = c_masterNodeId;
+ BackupRecordPtr ptr;
+ for(c_backups.first(ptr); ptr.i != RNIL; c_backups.next(ptr)) {
+ jam();
+ checkNodeFail(signal, ptr, newCoordinator, theFailedNodes);
+ }
+}
+
+bool
+Backup::verifyNodesAlive(const NdbNodeBitmask& aNodeBitMask)
+{
+ for (Uint32 i = 0; i < MAX_NDB_NODES; i++) {
+ jam();
+ if(aNodeBitMask.get(i)) {
+ if(!c_aliveNodes.get(i)){
+ jam();
+ return false;
+ }//if
+ }//if
+ }//for
+ return true;
+}
+
+void
+Backup::checkNodeFail(Signal* signal,
+ BackupRecordPtr ptr,
+ NodeId newCoord,
+ Uint32 theFailedNodes[NodeBitmask::Size])
+{
+ ndbrequire( ptr.p->nodes.get(newCoord)); /* just to make sure newCoord
+ * is part of the backup
+ */
+ /* Update ptr.p->nodes to be up to date with current alive nodes
+ */
+ NodePtr nodePtr;
+ bool found = false;
+ for(c_nodes.first(nodePtr); nodePtr.i != RNIL; c_nodes.next(nodePtr)) {
+ jam();
+ if(NodeBitmask::get(theFailedNodes, nodePtr.p->nodeId)) {
+ jam();
+ if (ptr.p->nodes.get(nodePtr.p->nodeId)) {
+ jam();
+ ptr.p->nodes.clear(nodePtr.p->nodeId);
+ found = true;
+ }
+ }//if
+ }//for
+
+ if(!found) {
+ jam();
+ return; // failed node is not part of backup process, safe to continue
+ }
+
+ bool doMasterTakeover = false;
+ if(NodeBitmask::get(theFailedNodes, refToNode(ptr.p->masterRef))){
+ jam();
+ doMasterTakeover = true;
+ };
+
+ if (newCoord == getOwnNodeId()){
+ jam();
+ if (doMasterTakeover) {
+ /**
+ * I'm new master
+ */
+ CRASH_INSERTION((10002));
+#ifdef DEBUG_ABORT
+ ndbout_c("**** Master Takeover: Node failed: Master id = %u",
+ refToNode(ptr.p->masterRef));
+#endif
+ masterTakeOver(signal, ptr);
+ return;
+ }//if
+ /**
+ * I'm master for this backup
+ */
+ jam();
+ CRASH_INSERTION((10001));
+#ifdef DEBUG_ABORT
+ ndbout_c("**** Master: Node failed: Master id = %u",
+ refToNode(ptr.p->masterRef));
+#endif
+ masterAbort(signal, ptr, false);
+ return;
+ }//if
+
+ /**
+ * If there's a new master, (it's not me)
+ * but remember who it is
+ */
+ ptr.p->masterRef = calcBackupBlockRef(newCoord);
+#ifdef DEBUG_ABORT
+ ndbout_c("**** Slave: Node failed: Master id = %u",
+ refToNode(ptr.p->masterRef));
+#endif
+ /**
+ * I abort myself as slave if not master
+ */
+ CRASH_INSERTION((10021));
+ // slaveAbort(signal, ptr);
+}
+
+void
+Backup::masterTakeOver(Signal* signal, BackupRecordPtr ptr)
+{
+ ptr.p->masterRef = reference();
+ ptr.p->masterData.gsn = MAX_GSN + 1;
+
+ switch(ptr.p->slaveState.getState()){
+ case INITIAL:
+ jam();
+ ptr.p->masterData.state.forceState(INITIAL);
+ break;
+ case ABORTING:
+ jam();
+ case DEFINING:
+ jam();
+ case DEFINED:
+ jam();
+ case STARTED:
+ jam();
+ case SCANNING:
+ jam();
+ ptr.p->masterData.state.forceState(STARTED);
+ break;
+ case STOPPING:
+ jam();
+ case CLEANING:
+ jam();
+ ptr.p->masterData.state.forceState(STOPPING);
+ break;
+ default:
+ ndbrequire(false);
+ }
+ masterAbort(signal, ptr, false);
+}
+
+void
+Backup::execINCL_NODEREQ(Signal* signal)
+{
+ jamEntry();
+
+ const Uint32 senderRef = signal->theData[0];
+ const Uint32 inclNode = signal->theData[1];
+
+ NodePtr node;
+ for(c_nodes.first(node); node.i != RNIL; c_nodes.next(node)) {
+ jam();
+ const Uint32 nodeId = node.p->nodeId;
+ if(inclNode == nodeId){
+ jam();
+
+ ndbrequire(node.p->alive == 0);
+ ndbrequire(!c_aliveNodes.get(nodeId));
+
+ node.p->alive = 1;
+ c_aliveNodes.set(nodeId);
+
+ break;
+ }//if
+ }//for
+ signal->theData[0] = reference();
+ sendSignal(senderRef, GSN_INCL_NODECONF, signal, 1, JBB);
+}
+
+/*****************************************************************************
+ *
+ * Master functionallity - Define backup
+ *
+ *****************************************************************************/
+
+void
+Backup::execBACKUP_REQ(Signal* signal)
+{
+ jamEntry();
+ BackupReq * req = (BackupReq*)signal->getDataPtr();
+
+ const Uint32 senderData = req->senderData;
+ const BlockReference senderRef = signal->senderBlockRef();
+ const Uint32 dataLen32 = req->backupDataLen; // In 32 bit words
+
+ if(getOwnNodeId() != getMasterNodeId()) {
+ jam();
+ sendBackupRef(senderRef, signal, senderData, BackupRef::IAmNotMaster);
+ return;
+ }//if
+
+ if(dataLen32 != 0) {
+ jam();
+ sendBackupRef(senderRef, signal, senderData,
+ BackupRef::BackupDefinitionNotImplemented);
+ return;
+ }//if
+
+#ifdef DEBUG_ABORT
+ dumpUsedResources();
+#endif
+ /**
+ * Seize a backup record
+ */
+ BackupRecordPtr ptr;
+ c_backups.seize(ptr);
+ if(ptr.i == RNIL) {
+ jam();
+ sendBackupRef(senderRef, signal, senderData, BackupRef::OutOfBackupRecord);
+ return;
+ }//if
+
+ ndbrequire(ptr.p->pages.empty());
+ ndbrequire(ptr.p->tables.empty());
+
+ ptr.p->masterData.state.forceState(INITIAL);
+ ptr.p->masterData.state.setState(DEFINING);
+ ptr.p->clientRef = senderRef;
+ ptr.p->clientData = senderData;
+ ptr.p->masterRef = reference();
+ ptr.p->nodes = c_aliveNodes;
+ ptr.p->backupId = 0;
+ ptr.p->backupKey[0] = 0;
+ ptr.p->backupKey[1] = 0;
+ ptr.p->backupDataLen = 0;
+ ptr.p->masterData.dropTrig.tableId = RNIL;
+ ptr.p->masterData.alterTrig.tableId = RNIL;
+
+ UtilSequenceReq * utilReq = (UtilSequenceReq*)signal->getDataPtrSend();
+
+ ptr.p->masterData.gsn = GSN_UTIL_SEQUENCE_REQ;
+ utilReq->senderData = ptr.i;
+ utilReq->sequenceId = BACKUP_SEQUENCE;
+ utilReq->requestType = UtilSequenceReq::NextVal;
+ sendSignal(DBUTIL_REF, GSN_UTIL_SEQUENCE_REQ,
+ signal, UtilSequenceReq::SignalLength, JBB);
+}
+
+void
+Backup::execUTIL_SEQUENCE_REF(Signal* signal)
+{
+ BackupRecordPtr ptr;
+ jamEntry();
+ UtilSequenceRef * utilRef = (UtilSequenceRef*)signal->getDataPtr();
+ ptr.i = utilRef->senderData;
+ ndbrequire(ptr.i == RNIL);
+ c_backupPool.getPtr(ptr);
+ ndbrequire(ptr.p->masterData.gsn == GSN_UTIL_SEQUENCE_REQ);
+ ptr.p->masterData.gsn = 0;
+ sendBackupRef(signal, ptr, BackupRef::SequenceFailure);
+}//execUTIL_SEQUENCE_REF()
+
+
+void
+Backup::sendBackupRef(Signal* signal, BackupRecordPtr ptr, Uint32 errorCode)
+{
+ jam();
+ sendBackupRef(ptr.p->clientRef, signal, ptr.p->clientData, errorCode);
+ // ptr.p->masterData.state.setState(INITIAL);
+ cleanupSlaveResources(ptr);
+}
+
+void
+Backup::sendBackupRef(BlockReference senderRef, Signal *signal,
+ Uint32 senderData, Uint32 errorCode)
+{
+ jam();
+ BackupRef* ref = (BackupRef*)signal->getDataPtrSend();
+ ref->senderData = senderData;
+ ref->errorCode = errorCode;
+ ref->masterRef = numberToRef(BACKUP, getMasterNodeId());
+ sendSignal(senderRef, GSN_BACKUP_REF, signal, BackupRef::SignalLength, JBB);
+}
+
+void
+Backup::execUTIL_SEQUENCE_CONF(Signal* signal)
+{
+ jamEntry();
+
+ UtilSequenceConf * conf = (UtilSequenceConf*)signal->getDataPtr();
+
+ if(conf->requestType == UtilSequenceReq::Create) {
+ jam();
+ sendSTTORRY(signal); // At startup in NDB
+ return;
+ }
+
+ BackupRecordPtr ptr;
+ ptr.i = conf->senderData;
+ c_backupPool.getPtr(ptr);
+
+ ndbrequire(ptr.p->masterData.gsn == GSN_UTIL_SEQUENCE_REQ);
+ ptr.p->masterData.gsn = 0;
+ if (ptr.p->masterData.state.getState() == ABORTING) {
+ jam();
+ sendBackupRef(signal, ptr, ptr.p->errorCode);
+ return;
+ }//if
+ if (ERROR_INSERTED(10023)) {
+ ptr.p->masterData.state.setState(ABORTING);
+ sendBackupRef(signal, ptr, 323);
+ return;
+ }//if
+ ndbrequire(ptr.p->masterData.state.getState() == DEFINING);
+
+ ptr.p->backupId = conf->sequenceValue[0];
+ ptr.p->backupKey[0] = (getOwnNodeId() << 16) | (ptr.p->backupId & 0xFFFF);
+ ptr.p->backupKey[1] = NdbTick_CurrentMillisecond();
+
+ ptr.p->masterData.gsn = GSN_UTIL_LOCK_REQ;
+ Mutex mutex(signal, c_mutexMgr, ptr.p->masterData.m_defineBackupMutex);
+ Callback c = { safe_cast(&Backup::defineBackupMutex_locked), ptr.i };
+ ndbrequire(mutex.lock(c));
+
+ return;
+}
+
+void
+Backup::defineBackupMutex_locked(Signal* signal, Uint32 ptrI, Uint32 retVal){
+ jamEntry();
+ ndbrequire(retVal == 0);
+
+ BackupRecordPtr ptr;
+ ptr.i = ptrI;
+ c_backupPool.getPtr(ptr);
+
+ ndbrequire(ptr.p->masterData.gsn == GSN_UTIL_LOCK_REQ);
+ ptr.p->masterData.gsn = 0;
+
+ ptr.p->masterData.gsn = GSN_UTIL_LOCK_REQ;
+ Mutex mutex(signal, c_mutexMgr, ptr.p->masterData.m_dictCommitTableMutex);
+ Callback c = { safe_cast(&Backup::dictCommitTableMutex_locked), ptr.i };
+ ndbrequire(mutex.lock(c));
+}
+
+void
+Backup::dictCommitTableMutex_locked(Signal* signal, Uint32 ptrI,Uint32 retVal)
+{
+ jamEntry();
+ ndbrequire(retVal == 0);
+
+ /**
+ * We now have both the mutexes
+ */
+ BackupRecordPtr ptr;
+ ptr.i = ptrI;
+ c_backupPool.getPtr(ptr);
+
+ ndbrequire(ptr.p->masterData.gsn == GSN_UTIL_LOCK_REQ);
+ ptr.p->masterData.gsn = 0;
+
+ if (ERROR_INSERTED(10031)) {
+ ptr.p->masterData.state.setState(ABORTING);
+ ptr.p->setErrorCode(331);
+ }//if
+
+ if (ptr.p->masterData.state.getState() == ABORTING) {
+ jam();
+
+ /**
+ * Unlock mutexes
+ */
+ jam();
+ Mutex mutex1(signal, c_mutexMgr, ptr.p->masterData.m_dictCommitTableMutex);
+ jam();
+ mutex1.unlock(); // ignore response
+
+ jam();
+ Mutex mutex2(signal, c_mutexMgr, ptr.p->masterData.m_defineBackupMutex);
+ jam();
+ mutex2.unlock(); // ignore response
+
+ sendBackupRef(signal, ptr, ptr.p->errorCode);
+ return;
+ }//if
+
+ ndbrequire(ptr.p->masterData.state.getState() == DEFINING);
+
+ sendDefineBackupReq(signal, ptr);
+}
+
+/*****************************************************************************
+ *
+ * Master functionallity - Define backup cont'd (from now on all slaves are in)
+ *
+ *****************************************************************************/
+
+void
+Backup::sendSignalAllWait(BackupRecordPtr ptr, Uint32 gsn, Signal *signal,
+ Uint32 signalLength, bool executeDirect)
+{
+ jam();
+ ptr.p->masterData.gsn = gsn;
+ ptr.p->masterData.sendCounter.clearWaitingFor();
+ NodePtr node;
+ for(c_nodes.first(node); node.i != RNIL; c_nodes.next(node)){
+ jam();
+ const Uint32 nodeId = node.p->nodeId;
+ if(node.p->alive && ptr.p->nodes.get(nodeId)){
+ jam();
+
+ ptr.p->masterData.sendCounter.setWaitingFor(nodeId);
+
+ const BlockReference ref = numberToRef(BACKUP, nodeId);
+ if (!executeDirect || ref != reference()) {
+ sendSignal(ref, gsn, signal, signalLength, JBB);
+ }//if
+ }//if
+ }//for
+ if (executeDirect) {
+ EXECUTE_DIRECT(BACKUP, gsn, signal, signalLength);
+ }
+}
+
+bool
+Backup::haveAllSignals(BackupRecordPtr ptr, Uint32 gsn, Uint32 nodeId)
+{
+ ndbrequire(ptr.p->masterRef == reference());
+ ndbrequire(ptr.p->masterData.gsn == gsn);
+ ndbrequire(!ptr.p->masterData.sendCounter.done());
+ ndbrequire(ptr.p->masterData.sendCounter.isWaitingFor(nodeId));
+
+ ptr.p->masterData.sendCounter.clearWaitingFor(nodeId);
+
+ if (ptr.p->masterData.sendCounter.done())
+ ptr.p->masterData.gsn = 0;
+
+ return ptr.p->masterData.sendCounter.done();
+}
+
+void
+Backup::sendDefineBackupReq(Signal *signal, BackupRecordPtr ptr)
+{
+ /**
+ * Sending define backup to all participants
+ */
+ DefineBackupReq * req = (DefineBackupReq*)signal->getDataPtrSend();
+ req->backupId = ptr.p->backupId;
+ req->clientRef = ptr.p->clientRef;
+ req->clientData = ptr.p->clientData;
+ req->senderRef = reference();
+ req->backupPtr = ptr.i;
+ req->backupKey[0] = ptr.p->backupKey[0];
+ req->backupKey[1] = ptr.p->backupKey[1];
+ req->nodes = ptr.p->nodes;
+ req->backupDataLen = ptr.p->backupDataLen;
+
+ ptr.p->masterData.errorCode = 0;
+ ptr.p->okToCleanMaster = false; // master must wait with cleaning to last
+ sendSignalAllWait(ptr, GSN_DEFINE_BACKUP_REQ, signal,
+ DefineBackupReq::SignalLength,
+ true /* do execute direct on oneself */);
+ /**
+ * Now send backup data
+ */
+ const Uint32 len = ptr.p->backupDataLen;
+ if(len == 0){
+ /**
+ * No data to send
+ */
+ jam();
+ return;
+ }//if
+
+ /**
+ * Not implemented
+ */
+ ndbrequire(0);
+}
+
+void
+Backup::execDEFINE_BACKUP_REF(Signal* signal)
+{
+ jamEntry();
+
+ DefineBackupRef* ref = (DefineBackupRef*)signal->getDataPtr();
+
+ const Uint32 ptrI = ref->backupPtr;
+ const Uint32 backupId = ref->backupId;
+ const Uint32 nodeId = refToNode(signal->senderBlockRef());
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ masterAbortCheck(); // macro will do return if ABORTING
+
+ ptr.p->masterData.errorCode = ref->errorCode;
+ defineBackupReply(signal, ptr, nodeId);
+}
+
+void
+Backup::execDEFINE_BACKUP_CONF(Signal* signal)
+{
+ jamEntry();
+
+ DefineBackupConf* conf = (DefineBackupConf*)signal->getDataPtr();
+ const Uint32 ptrI = conf->backupPtr;
+ const Uint32 backupId = conf->backupId;
+ const Uint32 nodeId = refToNode(signal->senderBlockRef());
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ masterAbortCheck(); // macro will do return if ABORTING
+
+ if (ERROR_INSERTED(10024)) {
+ ptr.p->masterData.errorCode = 324;
+ }//if
+
+ defineBackupReply(signal, ptr, nodeId);
+}
+
+void
+Backup::defineBackupReply(Signal* signal, BackupRecordPtr ptr, Uint32 nodeId)
+{
+ if (!haveAllSignals(ptr, GSN_DEFINE_BACKUP_REQ, nodeId)) {
+ jam();
+ return;
+ }
+ /**
+ * Unlock mutexes
+ */
+ jam();
+ Mutex mutex1(signal, c_mutexMgr, ptr.p->masterData.m_dictCommitTableMutex);
+ jam();
+ mutex1.unlock(); // ignore response
+
+ jam();
+ Mutex mutex2(signal, c_mutexMgr, ptr.p->masterData.m_defineBackupMutex);
+ jam();
+ mutex2.unlock(); // ignore response
+
+ if(ptr.p->errorCode) {
+ jam();
+ ptr.p->masterData.errorCode = ptr.p->errorCode;
+ }
+
+ if(ptr.p->masterData.errorCode){
+ jam();
+ ptr.p->setErrorCode(ptr.p->masterData.errorCode);
+ sendAbortBackupOrd(signal, ptr, AbortBackupOrd::OkToClean);
+ masterSendAbortBackup(signal, ptr);
+ return;
+ }
+
+ /**
+ * Reply to client
+ */
+ BackupConf * conf = (BackupConf*)signal->getDataPtrSend();
+ conf->backupId = ptr.p->backupId;
+ conf->senderData = ptr.p->clientData;
+ conf->nodes = ptr.p->nodes;
+ sendSignal(ptr.p->clientRef, GSN_BACKUP_CONF, signal,
+ BackupConf::SignalLength, JBB);
+
+ ptr.p->masterData.state.setState(DEFINED);
+ /**
+ * Prepare Trig
+ */
+ TablePtr tabPtr;
+ ndbrequire(ptr.p->tables.first(tabPtr));
+ sendCreateTrig(signal, ptr, tabPtr);
+}
+
+/*****************************************************************************
+ *
+ * Master functionallity - Prepare triggers
+ *
+ *****************************************************************************/
+void
+Backup::createAttributeMask(TablePtr tabPtr,
+ Bitmask<MAXNROFATTRIBUTESINWORDS> & mask)
+{
+ mask.clear();
+ Table & table = * tabPtr.p;
+ for(Uint32 i = 0; i<table.noOfAttributes; i++) {
+ jam();
+ AttributePtr attr;
+ table.attributes.getPtr(attr, i);
+ if(attr.p->data.key != 0){
+ jam();
+ continue;
+ }
+ mask.set(i);
+ }
+}
+
+void
+Backup::sendCreateTrig(Signal* signal,
+ BackupRecordPtr ptr, TablePtr tabPtr)
+{
+ CreateTrigReq * req =(CreateTrigReq *)signal->getDataPtrSend();
+
+ ptr.p->errorCode = 0;
+ ptr.p->masterData.gsn = GSN_CREATE_TRIG_REQ;
+ ptr.p->masterData.sendCounter = 3;
+ ptr.p->masterData.createTrig.tableId = tabPtr.p->tableId;
+
+ req->setUserRef(reference());
+ req->setConnectionPtr(ptr.i);
+ req->setRequestType(CreateTrigReq::RT_USER);
+
+ Bitmask<MAXNROFATTRIBUTESINWORDS> attrMask;
+ createAttributeMask(tabPtr, attrMask);
+ req->setAttributeMask(attrMask);
+ req->setTableId(tabPtr.p->tableId);
+ req->setIndexId(RNIL); // not used
+ req->setTriggerId(RNIL); // to be created
+ req->setTriggerType(TriggerType::SUBSCRIPTION);
+ req->setTriggerActionTime(TriggerActionTime::TA_DETACHED);
+ req->setMonitorReplicas(true);
+ req->setMonitorAllAttributes(false);
+ req->setOnline(false); // leave trigger offline
+
+ char triggerName[MAX_TAB_NAME_SIZE];
+ Uint32 nameBuffer[2 + ((MAX_TAB_NAME_SIZE + 3) >> 2)]; // SP string
+ LinearWriter w(nameBuffer, sizeof(nameBuffer) >> 2);
+ LinearSectionPtr lsPtr[3];
+
+ for (int i=0; i < 3; i++) {
+ req->setTriggerEvent(triggerEventValues[i]);
+ snprintf(triggerName, sizeof(triggerName), triggerNameFormat[i],
+ ptr.p->backupId, tabPtr.p->tableId);
+ w.reset();
+ w.add(CreateTrigReq::TriggerNameKey, triggerName);
+ lsPtr[0].p = nameBuffer;
+ lsPtr[0].sz = w.getWordsUsed();
+ sendSignal(DBDICT_REF, GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength, JBB, lsPtr, 1);
+ }
+}
+
+void
+Backup::execCREATE_TRIG_CONF(Signal* signal)
+{
+ jamEntry();
+ CreateTrigConf * conf = (CreateTrigConf*)signal->getDataPtr();
+
+ const Uint32 ptrI = conf->getConnectionPtr();
+ const Uint32 tableId = conf->getTableId();
+ const TriggerEvent::Value type = conf->getTriggerEvent();
+ const Uint32 triggerId = conf->getTriggerId();
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ /**
+ * Verify that I'm waiting for this conf
+ */
+ ndbrequire(ptr.p->masterRef == reference());
+ ndbrequire(ptr.p->masterData.gsn == GSN_CREATE_TRIG_REQ);
+ ndbrequire(ptr.p->masterData.sendCounter.done() == false);
+ ndbrequire(ptr.p->masterData.createTrig.tableId == tableId);
+
+ TablePtr tabPtr;
+ ndbrequire(findTable(ptr, tabPtr, tableId));
+ ndbrequire(type < 3); // if some decides to change the enums
+
+ ndbrequire(tabPtr.p->triggerIds[type] == ILLEGAL_TRIGGER_ID);
+ tabPtr.p->triggerIds[type] = triggerId;
+
+ createTrigReply(signal, ptr);
+}
+
+void
+Backup::execCREATE_TRIG_REF(Signal* signal)
+{
+ CreateTrigRef* ref = (CreateTrigRef*)signal->getDataPtr();
+
+ const Uint32 ptrI = ref->getConnectionPtr();
+ const Uint32 tableId = ref->getTableId();
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ /**
+ * Verify that I'm waiting for this ref
+ */
+ ndbrequire(ptr.p->masterRef == reference());
+ ndbrequire(ptr.p->masterData.gsn == GSN_CREATE_TRIG_REQ);
+ ndbrequire(ptr.p->masterData.sendCounter.done() == false);
+ ndbrequire(ptr.p->masterData.createTrig.tableId == tableId);
+
+ ptr.p->setErrorCode(ref->getErrorCode());
+
+ createTrigReply(signal, ptr);
+}
+
+void
+Backup::createTrigReply(Signal* signal, BackupRecordPtr ptr)
+{
+ CRASH_INSERTION(10003);
+
+ /**
+ * Check finished with table
+ */
+ ptr.p->masterData.sendCounter--;
+ if(ptr.p->masterData.sendCounter.done() == false){
+ jam();
+ return;
+ }//if
+
+ ptr.p->masterData.gsn = 0;
+
+ if(ptr.p->checkError()) {
+ jam();
+ masterAbort(signal, ptr, true);
+ return;
+ }//if
+
+ if (ERROR_INSERTED(10025)) {
+ ptr.p->errorCode = 325;
+ masterAbort(signal, ptr, true);
+ return;
+ }//if
+
+ TablePtr tabPtr;
+ ndbrequire(findTable(ptr, tabPtr, ptr.p->masterData.createTrig.tableId));
+
+ /**
+ * Next table
+ */
+ ptr.p->tables.next(tabPtr);
+ if(tabPtr.i != RNIL){
+ jam();
+ sendCreateTrig(signal, ptr, tabPtr);
+ return;
+ }//if
+
+ /**
+ * Finished with all tables, send StartBackupReq
+ */
+ ptr.p->masterData.state.setState(STARTED);
+
+ ptr.p->tables.first(tabPtr);
+ ptr.p->errorCode = 0;
+ ptr.p->masterData.startBackup.signalNo = 0;
+ ptr.p->masterData.startBackup.noOfSignals =
+ (ptr.p->tables.noOfElements() + StartBackupReq::MaxTableTriggers - 1) /
+ StartBackupReq::MaxTableTriggers;
+ sendStartBackup(signal, ptr, tabPtr);
+}
+
+/*****************************************************************************
+ *
+ * Master functionallity - Start backup
+ *
+ *****************************************************************************/
+void
+Backup::sendStartBackup(Signal* signal, BackupRecordPtr ptr, TablePtr tabPtr)
+{
+
+ ptr.p->masterData.startBackup.tablePtr = tabPtr.i;
+
+ StartBackupReq* req = (StartBackupReq*)signal->getDataPtrSend();
+ req->backupId = ptr.p->backupId;
+ req->backupPtr = ptr.i;
+ req->signalNo = ptr.p->masterData.startBackup.signalNo;
+ req->noOfSignals = ptr.p->masterData.startBackup.noOfSignals;
+ Uint32 i;
+ for(i = 0; i<StartBackupReq::MaxTableTriggers; i++) {
+ jam();
+ req->tableTriggers[i].tableId = tabPtr.p->tableId;
+ req->tableTriggers[i].triggerIds[0] = tabPtr.p->triggerIds[0];
+ req->tableTriggers[i].triggerIds[1] = tabPtr.p->triggerIds[1];
+ req->tableTriggers[i].triggerIds[2] = tabPtr.p->triggerIds[2];
+ if(!ptr.p->tables.next(tabPtr)){
+ jam();
+ i++;
+ break;
+ }//if
+ }//for
+ req->noOfTableTriggers = i;
+
+ sendSignalAllWait(ptr, GSN_START_BACKUP_REQ, signal,
+ StartBackupReq::HeaderLength +
+ (i * StartBackupReq::TableTriggerLength));
+}
+
+void
+Backup::execSTART_BACKUP_REF(Signal* signal)
+{
+ jamEntry();
+
+ StartBackupRef* ref = (StartBackupRef*)signal->getDataPtr();
+ const Uint32 ptrI = ref->backupPtr;
+ const Uint32 backupId = ref->backupId;
+ const Uint32 signalNo = ref->signalNo;
+ const Uint32 nodeId = refToNode(signal->senderBlockRef());
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ masterAbortCheck(); // macro will do return if ABORTING
+
+ ptr.p->setErrorCode(ref->errorCode);
+ startBackupReply(signal, ptr, nodeId, signalNo);
+}
+
+void
+Backup::execSTART_BACKUP_CONF(Signal* signal)
+{
+ jamEntry();
+
+ StartBackupConf* conf = (StartBackupConf*)signal->getDataPtr();
+ const Uint32 ptrI = conf->backupPtr;
+ const Uint32 backupId = conf->backupId;
+ const Uint32 signalNo = conf->signalNo;
+ const Uint32 nodeId = refToNode(signal->senderBlockRef());
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ masterAbortCheck(); // macro will do return if ABORTING
+
+ startBackupReply(signal, ptr, nodeId, signalNo);
+}
+
+void
+Backup::startBackupReply(Signal* signal, BackupRecordPtr ptr,
+ Uint32 nodeId, Uint32 signalNo)
+{
+
+ CRASH_INSERTION((10004));
+
+ ndbrequire(ptr.p->masterData.startBackup.signalNo == signalNo);
+ if (!haveAllSignals(ptr, GSN_START_BACKUP_REQ, nodeId)) {
+ jam();
+ return;
+ }
+
+ if(ptr.p->checkError()){
+ jam();
+ masterAbort(signal, ptr, true);
+ return;
+ }
+
+ if (ERROR_INSERTED(10026)) {
+ ptr.p->errorCode = 326;
+ masterAbort(signal, ptr, true);
+ return;
+ }//if
+
+ TablePtr tabPtr;
+ c_tablePool.getPtr(tabPtr, ptr.p->masterData.startBackup.tablePtr);
+ for(Uint32 i = 0; i<StartBackupReq::MaxTableTriggers; i++) {
+ jam();
+ if(!ptr.p->tables.next(tabPtr)) {
+ jam();
+ break;
+ }//if
+ }//for
+
+ if(tabPtr.i != RNIL) {
+ jam();
+ ptr.p->masterData.startBackup.signalNo++;
+ sendStartBackup(signal, ptr, tabPtr);
+ return;
+ }
+
+ sendAlterTrig(signal, ptr);
+}
+
+/*****************************************************************************
+ *
+ * Master functionallity - Activate triggers
+ *
+ *****************************************************************************/
+void
+Backup::sendAlterTrig(Signal* signal, BackupRecordPtr ptr)
+{
+ AlterTrigReq * req =(AlterTrigReq *)signal->getDataPtrSend();
+
+ ptr.p->errorCode = 0;
+ ptr.p->masterData.gsn = GSN_ALTER_TRIG_REQ;
+ ptr.p->masterData.sendCounter = 0;
+
+ req->setUserRef(reference());
+ req->setConnectionPtr(ptr.i);
+ req->setRequestType(AlterTrigReq::RT_USER);
+ req->setTriggerInfo(0); // not used on ALTER via DICT
+ req->setOnline(true);
+ req->setReceiverRef(reference());
+
+ TablePtr tabPtr;
+
+ if (ptr.p->masterData.alterTrig.tableId == RNIL) {
+ jam();
+ ptr.p->tables.first(tabPtr);
+ } else {
+ jam();
+ ndbrequire(findTable(ptr, tabPtr, ptr.p->masterData.alterTrig.tableId));
+ ptr.p->tables.next(tabPtr);
+ }//if
+ if (tabPtr.i != RNIL) {
+ jam();
+ ptr.p->masterData.alterTrig.tableId = tabPtr.p->tableId;
+ req->setTableId(tabPtr.p->tableId);
+
+ req->setTriggerId(tabPtr.p->triggerIds[0]);
+ sendSignal(DBDICT_REF, GSN_ALTER_TRIG_REQ,
+ signal, AlterTrigReq::SignalLength, JBB);
+
+ req->setTriggerId(tabPtr.p->triggerIds[1]);
+ sendSignal(DBDICT_REF, GSN_ALTER_TRIG_REQ,
+ signal, AlterTrigReq::SignalLength, JBB);
+
+ req->setTriggerId(tabPtr.p->triggerIds[2]);
+ sendSignal(DBDICT_REF, GSN_ALTER_TRIG_REQ,
+ signal, AlterTrigReq::SignalLength, JBB);
+
+ ptr.p->masterData.sendCounter += 3;
+ return;
+ }//if
+ ptr.p->masterData.alterTrig.tableId = RNIL;
+ /**
+ * Finished with all tables
+ */
+ ptr.p->masterData.gsn = GSN_WAIT_GCP_REQ;
+ ptr.p->masterData.waitGCP.startBackup = true;
+
+ WaitGCPReq * waitGCPReq = (WaitGCPReq*)signal->getDataPtrSend();
+ waitGCPReq->senderRef = reference();
+ waitGCPReq->senderData = ptr.i;
+ waitGCPReq->requestType = WaitGCPReq::CompleteForceStart;
+ sendSignal(DBDIH_REF, GSN_WAIT_GCP_REQ, signal,
+ WaitGCPReq::SignalLength,JBB);
+}
+
+void
+Backup::execALTER_TRIG_CONF(Signal* signal)
+{
+ jamEntry();
+
+ AlterTrigConf* conf = (AlterTrigConf*)signal->getDataPtr();
+ const Uint32 ptrI = conf->getConnectionPtr();
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ alterTrigReply(signal, ptr);
+}
+
+void
+Backup::execALTER_TRIG_REF(Signal* signal)
+{
+ jamEntry();
+
+ AlterTrigRef* ref = (AlterTrigRef*)signal->getDataPtr();
+ const Uint32 ptrI = ref->getConnectionPtr();
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ ptr.p->setErrorCode(ref->getErrorCode());
+
+ alterTrigReply(signal, ptr);
+}
+
+void
+Backup::alterTrigReply(Signal* signal, BackupRecordPtr ptr)
+{
+
+ CRASH_INSERTION((10005));
+
+ ndbrequire(ptr.p->masterRef == reference());
+ ndbrequire(ptr.p->masterData.gsn == GSN_ALTER_TRIG_REQ);
+ ndbrequire(ptr.p->masterData.sendCounter.done() == false);
+
+ ptr.p->masterData.sendCounter--;
+
+ if(ptr.p->masterData.sendCounter.done() == false){
+ jam();
+ return;
+ }//if
+
+ ptr.p->masterData.gsn = 0;
+
+ if(ptr.p->checkError()){
+ jam();
+ masterAbort(signal, ptr, true);
+ return;
+ }//if
+
+ sendAlterTrig(signal, ptr);
+}
+
+void
+Backup::execWAIT_GCP_REF(Signal* signal)
+{
+ jamEntry();
+
+ CRASH_INSERTION((10006));
+
+ WaitGCPRef * ref = (WaitGCPRef*)signal->getDataPtr();
+ const Uint32 ptrI = ref->senderData;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ ndbrequire(ptr.p->masterRef == reference());
+ ndbrequire(ptr.p->masterData.gsn == GSN_WAIT_GCP_REQ);
+
+ WaitGCPReq * req = (WaitGCPReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = ptr.i;
+ req->requestType = WaitGCPReq::CompleteForceStart;
+ sendSignal(DBDIH_REF, GSN_WAIT_GCP_REQ, signal,
+ WaitGCPReq::SignalLength,JBB);
+}
+
+void
+Backup::execWAIT_GCP_CONF(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION((10007));
+
+ WaitGCPConf * conf = (WaitGCPConf*)signal->getDataPtr();
+ const Uint32 ptrI = conf->senderData;
+ const Uint32 gcp = conf->gcp;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ ndbrequire(ptr.p->masterRef == reference());
+ ndbrequire(ptr.p->masterData.gsn == GSN_WAIT_GCP_REQ);
+ ptr.p->masterData.gsn = 0;
+
+ if(ptr.p->checkError()) {
+ jam();
+ masterAbort(signal, ptr, true);
+ return;
+ }//if
+
+ if(ptr.p->masterData.waitGCP.startBackup) {
+ jam();
+ CRASH_INSERTION((10008));
+ ptr.p->startGCP = gcp;
+ ptr.p->masterData.state.setState(SCANNING);
+ nextFragment(signal, ptr);
+ } else {
+ jam();
+ CRASH_INSERTION((10009));
+ ptr.p->stopGCP = gcp;
+ ptr.p->masterData.state.setState(STOPPING);
+ sendDropTrig(signal, ptr); // regular dropping of triggers
+ }//if
+}
+/*****************************************************************************
+ *
+ * Master functionallity - Backup fragment
+ *
+ *****************************************************************************/
+void
+Backup::nextFragment(Signal* signal, BackupRecordPtr ptr)
+{
+ jam();
+
+ BackupFragmentReq* req = (BackupFragmentReq*)signal->getDataPtrSend();
+ req->backupPtr = ptr.i;
+ req->backupId = ptr.p->backupId;
+
+ NodeBitmask nodes = ptr.p->nodes;
+ Uint32 idleNodes = nodes.count();
+ Uint32 saveIdleNodes = idleNodes;
+ ndbrequire(idleNodes > 0);
+
+ TablePtr tabPtr;
+ ptr.p->tables.first(tabPtr);
+ for(; tabPtr.i != RNIL && idleNodes > 0; ptr.p->tables.next(tabPtr)) {
+ jam();
+ FragmentPtr fragPtr;
+ Array<Fragment> & frags = tabPtr.p->fragments;
+ const Uint32 fragCount = frags.getSize();
+
+ for(Uint32 i = 0; i<fragCount && idleNodes > 0; i++) {
+ jam();
+ tabPtr.p->fragments.getPtr(fragPtr, i);
+ const Uint32 nodeId = fragPtr.p->node;
+ if(fragPtr.p->scanning != 0) {
+ jam();
+ ndbrequire(nodes.get(nodeId));
+ nodes.clear(nodeId);
+ idleNodes--;
+ } else if(fragPtr.p->scanned == 0 && nodes.get(nodeId)){
+ jam();
+ fragPtr.p->scanning = 1;
+ nodes.clear(nodeId);
+ idleNodes--;
+
+ req->tableId = tabPtr.p->tableId;
+ req->fragmentNo = i;
+ req->count = 0;
+
+ const BlockReference ref = numberToRef(BACKUP, nodeId);
+ sendSignal(ref, GSN_BACKUP_FRAGMENT_REQ, signal,
+ BackupFragmentReq::SignalLength, JBB);
+ }//if
+ }//for
+ }//for
+
+ if(idleNodes != saveIdleNodes){
+ jam();
+ return;
+ }//if
+
+ /**
+ * Finished with all tables
+ */
+ {
+ ptr.p->masterData.gsn = GSN_WAIT_GCP_REQ;
+ ptr.p->masterData.waitGCP.startBackup = false;
+
+ WaitGCPReq * req = (WaitGCPReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = ptr.i;
+ req->requestType = WaitGCPReq::CompleteForceStart;
+ sendSignal(DBDIH_REF, GSN_WAIT_GCP_REQ, signal,
+ WaitGCPReq::SignalLength, JBB);
+ }
+}
+
+void
+Backup::execBACKUP_FRAGMENT_CONF(Signal* signal)
+{
+ jamEntry();
+
+ CRASH_INSERTION((10010));
+
+ BackupFragmentConf * conf = (BackupFragmentConf*)signal->getDataPtr();
+ const Uint32 ptrI = conf->backupPtr;
+ const Uint32 backupId = conf->backupId;
+ const Uint32 tableId = conf->tableId;
+ const Uint32 fragmentNo = conf->fragmentNo;
+ const Uint32 nodeId = refToNode(signal->senderBlockRef());
+ const Uint32 noOfBytes = conf->noOfBytes;
+ const Uint32 noOfRecords = conf->noOfRecords;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ masterAbortCheck(); // macro will do return if ABORTING
+
+ ptr.p->noOfBytes += noOfBytes;
+ ptr.p->noOfRecords += noOfRecords;
+
+ TablePtr tabPtr;
+ ndbrequire(findTable(ptr, tabPtr, tableId));
+
+ FragmentPtr fragPtr;
+ tabPtr.p->fragments.getPtr(fragPtr, fragmentNo);
+
+ ndbrequire(fragPtr.p->scanned == 0);
+ ndbrequire(fragPtr.p->scanning == 1);
+ ndbrequire(fragPtr.p->node == nodeId);
+
+ fragPtr.p->scanned = 1;
+ fragPtr.p->scanning = 0;
+
+ if(ptr.p->checkError()) {
+ jam();
+ masterAbort(signal, ptr, true);
+ return;
+ }//if
+ if (ERROR_INSERTED(10028)) {
+ ptr.p->errorCode = 328;
+ masterAbort(signal, ptr, true);
+ return;
+ }//if
+ nextFragment(signal, ptr);
+}
+
+void
+Backup::execBACKUP_FRAGMENT_REF(Signal* signal)
+{
+ jamEntry();
+
+ CRASH_INSERTION((10011));
+
+ BackupFragmentRef * ref = (BackupFragmentRef*)signal->getDataPtr();
+ const Uint32 ptrI = ref->backupPtr;
+ const Uint32 backupId = ref->backupId;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ masterAbortCheck(); // macro will do return if ABORTING
+
+ ptr.p->setErrorCode(ref->errorCode);
+ masterAbort(signal, ptr, true);
+}
+
+/*****************************************************************************
+ *
+ * Master functionallity - Drop triggers
+ *
+ *****************************************************************************/
+
+void
+Backup::sendDropTrig(Signal* signal, BackupRecordPtr ptr)
+{
+ TablePtr tabPtr;
+ if (ptr.p->masterData.dropTrig.tableId == RNIL) {
+ jam();
+ ptr.p->tables.first(tabPtr);
+ } else {
+ jam();
+ ndbrequire(findTable(ptr, tabPtr, ptr.p->masterData.dropTrig.tableId));
+ ptr.p->tables.next(tabPtr);
+ }//if
+ if (tabPtr.i != RNIL) {
+ jam();
+ sendDropTrig(signal, ptr, tabPtr);
+ } else {
+ jam();
+ ptr.p->masterData.dropTrig.tableId = RNIL;
+
+ sendAbortBackupOrd(signal, ptr, AbortBackupOrd::OkToClean);
+
+ if(ptr.p->masterData.state.getState() == STOPPING) {
+ jam();
+ sendStopBackup(signal, ptr);
+ return;
+ }//if
+ ndbrequire(ptr.p->masterData.state.getState() == ABORTING);
+ masterSendAbortBackup(signal, ptr);
+ }//if
+}
+
+void
+Backup::sendDropTrig(Signal* signal, BackupRecordPtr ptr, TablePtr tabPtr)
+{
+ jam();
+ DropTrigReq * req = (DropTrigReq *)signal->getDataPtrSend();
+
+ ptr.p->masterData.gsn = GSN_DROP_TRIG_REQ;
+ ptr.p->masterData.sendCounter = 0;
+
+ req->setConnectionPtr(ptr.i);
+ req->setUserRef(reference()); // Sending to myself
+ req->setRequestType(DropTrigReq::RT_USER);
+ req->setIndexId(RNIL);
+ req->setTriggerInfo(0); // not used on DROP via DICT
+
+ char triggerName[MAX_TAB_NAME_SIZE];
+ Uint32 nameBuffer[2 + ((MAX_TAB_NAME_SIZE + 3) >> 2)]; // SP string
+ LinearWriter w(nameBuffer, sizeof(nameBuffer) >> 2);
+ LinearSectionPtr lsPtr[3];
+
+ ptr.p->masterData.dropTrig.tableId = tabPtr.p->tableId;
+ req->setTableId(tabPtr.p->tableId);
+
+ for (int i = 0; i < 3; i++) {
+ Uint32 id = tabPtr.p->triggerIds[i];
+ req->setTriggerId(id);
+ if (id != ILLEGAL_TRIGGER_ID) {
+ sendSignal(DBDICT_REF, GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+ } else {
+ snprintf(triggerName, sizeof(triggerName), triggerNameFormat[i],
+ ptr.p->backupId, tabPtr.p->tableId);
+ w.reset();
+ w.add(CreateTrigReq::TriggerNameKey, triggerName);
+ lsPtr[0].p = nameBuffer;
+ lsPtr[0].sz = w.getWordsUsed();
+ sendSignal(DBDICT_REF, GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB, lsPtr, 1);
+ }
+ ptr.p->masterData.sendCounter ++;
+ }
+}
+
+void
+Backup::execDROP_TRIG_REF(Signal* signal)
+{
+ jamEntry();
+
+ DropTrigRef* ref = (DropTrigRef*)signal->getDataPtr();
+ const Uint32 ptrI = ref->getConnectionPtr();
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ //ndbrequire(ref->getErrorCode() == DropTrigRef::NoSuchTrigger);
+ dropTrigReply(signal, ptr);
+}
+
+void
+Backup::execDROP_TRIG_CONF(Signal* signal)
+{
+ jamEntry();
+
+ DropTrigConf* conf = (DropTrigConf*)signal->getDataPtr();
+ const Uint32 ptrI = conf->getConnectionPtr();
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ dropTrigReply(signal, ptr);
+}
+
+void
+Backup::dropTrigReply(Signal* signal, BackupRecordPtr ptr)
+{
+
+ CRASH_INSERTION((10012));
+
+ ndbrequire(ptr.p->masterRef == reference());
+ ndbrequire(ptr.p->masterData.gsn == GSN_DROP_TRIG_REQ);
+ ndbrequire(ptr.p->masterData.sendCounter.done() == false);
+
+ ptr.p->masterData.sendCounter--;
+ if(ptr.p->masterData.sendCounter.done() == false){
+ jam();
+ return;
+ }//if
+
+ ptr.p->masterData.gsn = 0;
+ sendDropTrig(signal, ptr); // recursive next
+}
+
+/*****************************************************************************
+ *
+ * Master functionallity - Stop backup
+ *
+ *****************************************************************************/
+void
+Backup::execSTOP_BACKUP_REF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(0);
+}
+
+void
+Backup::sendStopBackup(Signal* signal, BackupRecordPtr ptr)
+{
+ jam();
+ ptr.p->masterData.gsn = GSN_STOP_BACKUP_REQ;
+
+ StopBackupReq* stop = (StopBackupReq*)signal->getDataPtrSend();
+ stop->backupPtr = ptr.i;
+ stop->backupId = ptr.p->backupId;
+ stop->startGCP = ptr.p->startGCP;
+ stop->stopGCP = ptr.p->stopGCP;
+
+ sendSignalAllWait(ptr, GSN_STOP_BACKUP_REQ, signal,
+ StopBackupReq::SignalLength);
+}
+
+void
+Backup::execSTOP_BACKUP_CONF(Signal* signal)
+{
+ jamEntry();
+
+ StopBackupConf* conf = (StopBackupConf*)signal->getDataPtr();
+ const Uint32 ptrI = conf->backupPtr;
+ const Uint32 backupId = conf->backupId;
+ const Uint32 nodeId = refToNode(signal->senderBlockRef());
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ masterAbortCheck(); // macro will do return if ABORTING
+
+ ptr.p->noOfLogBytes += conf->noOfLogBytes;
+ ptr.p->noOfLogRecords += conf->noOfLogRecords;
+
+ stopBackupReply(signal, ptr, nodeId);
+}
+
+void
+Backup::stopBackupReply(Signal* signal, BackupRecordPtr ptr, Uint32 nodeId)
+{
+ CRASH_INSERTION((10013));
+
+ if (!haveAllSignals(ptr, GSN_STOP_BACKUP_REQ, nodeId)) {
+ jam();
+ return;
+ }
+
+ // ptr.p->masterData.state.setState(INITIAL);
+
+ // send backup complete first to slaves so that they know
+ sendAbortBackupOrd(signal, ptr, AbortBackupOrd::BackupComplete);
+
+ BackupCompleteRep * rep = (BackupCompleteRep*)signal->getDataPtrSend();
+ rep->backupId = ptr.p->backupId;
+ rep->senderData = ptr.p->clientData;
+ rep->startGCP = ptr.p->startGCP;
+ rep->stopGCP = ptr.p->stopGCP;
+ rep->noOfBytes = ptr.p->noOfBytes;
+ rep->noOfRecords = ptr.p->noOfRecords;
+ rep->noOfLogBytes = ptr.p->noOfLogBytes;
+ rep->noOfLogRecords = ptr.p->noOfLogRecords;
+ rep->nodes = ptr.p->nodes;
+ sendSignal(ptr.p->clientRef, GSN_BACKUP_COMPLETE_REP, signal,
+ BackupCompleteRep::SignalLength, JBB);
+}
+
+/*****************************************************************************
+ *
+ * Master functionallity - Abort backup
+ *
+ *****************************************************************************/
+void
+Backup::masterAbort(Signal* signal, BackupRecordPtr ptr, bool controlledAbort)
+{
+ if(ptr.p->masterData.state.getState() == ABORTING) {
+#ifdef DEBUG_ABORT
+ ndbout_c("---- Master already aborting");
+#endif
+ jam();
+ return;
+ }
+ jam();
+#ifdef DEBUG_ABORT
+ ndbout_c("************ masterAbort");
+#endif
+
+ sendAbortBackupOrd(signal, ptr, AbortBackupOrd::BackupFailure);
+ if (!ptr.p->checkError())
+ ptr.p->errorCode = AbortBackupOrd::BackupFailureDueToNodeFail;
+
+ const State s = ptr.p->masterData.state.getState();
+
+ ptr.p->masterData.state.setState(ABORTING);
+
+ ndbrequire(s == INITIAL ||
+ s == STARTED ||
+ s == DEFINING ||
+ s == DEFINED ||
+ s == SCANNING ||
+ s == STOPPING ||
+ s == ABORTING);
+ if(ptr.p->masterData.gsn == GSN_UTIL_SEQUENCE_REQ) {
+ jam();
+ DEBUG_OUT("masterAbort: gsn = GSN_UTIL_SEQUENCE_REQ");
+ //-------------------------------------------------------
+ // We are waiting for UTIL_SEQUENCE response. We rely on
+ // this to arrive and check for ABORTING in response.
+ // No slaves are involved at this point and ABORT simply
+ // results in BACKUP_REF to client
+ //-------------------------------------------------------
+ /**
+ * Waiting for Sequence Id
+ * @see execUTIL_SEQUENCE_CONF
+ */
+ return;
+ }//if
+
+ if(ptr.p->masterData.gsn == GSN_UTIL_LOCK_REQ) {
+ jam();
+ DEBUG_OUT("masterAbort: gsn = GSN_UTIL_LOCK_REQ");
+ //-------------------------------------------------------
+ // We are waiting for UTIL_LOCK response (mutex). We rely on
+ // this to arrive and check for ABORTING in response.
+ // No slaves are involved at this point and ABORT simply
+ // results in BACKUP_REF to client
+ //-------------------------------------------------------
+ /**
+ * Waiting for lock
+ * @see execUTIL_LOCK_CONF
+ */
+ return;
+ }//if
+
+ /**
+ * Unlock mutexes only at master
+ */
+ jam();
+ Mutex mutex1(signal, c_mutexMgr, ptr.p->masterData.m_dictCommitTableMutex);
+ jam();
+ mutex1.unlock(); // ignore response
+
+ jam();
+ Mutex mutex2(signal, c_mutexMgr, ptr.p->masterData.m_defineBackupMutex);
+ jam();
+ mutex2.unlock(); // ignore response
+
+ if (!controlledAbort) {
+ jam();
+ if (s == DEFINING) {
+ jam();
+//-------------------------------------------------------
+// If we are in the defining phase all work is done by
+// slaves. No triggers have been allocated thus slaves
+// may free all "Master" resources, let them know...
+//-------------------------------------------------------
+ sendAbortBackupOrd(signal, ptr, AbortBackupOrd::OkToClean);
+ return;
+ }//if
+ if (s == DEFINED) {
+ jam();
+//-------------------------------------------------------
+// DEFINED is the state when triggers are created. We rely
+// on that DICT will report create trigger failure in case
+// of node failure. Thus no special action is needed here.
+// We will check for errorCode != 0 when receiving
+// replies on create trigger.
+//-------------------------------------------------------
+ return;
+ }//if
+ if(ptr.p->masterData.gsn == GSN_WAIT_GCP_REQ) {
+ jam();
+ DEBUG_OUT("masterAbort: gsn = GSN_WAIT_GCP_REQ");
+//-------------------------------------------------------
+// We are waiting for WAIT_GCP response. We rely on
+// this to arrive and check for ABORTING in response.
+//-------------------------------------------------------
+
+ /**
+ * Waiting for GCP
+ * @see execWAIT_GCP_CONF
+ */
+ return;
+ }//if
+
+ if(ptr.p->masterData.gsn == GSN_ALTER_TRIG_REQ) {
+ jam();
+ DEBUG_OUT("masterAbort: gsn = GSN_ALTER_TRIG_REQ");
+//-------------------------------------------------------
+// We are waiting for ALTER_TRIG response. We rely on
+// this to arrive and check for ABORTING in response.
+//-------------------------------------------------------
+
+ /**
+ * All triggers haven't been created yet
+ */
+ return;
+ }//if
+
+ if(ptr.p->masterData.gsn == GSN_DROP_TRIG_REQ) {
+ jam();
+ DEBUG_OUT("masterAbort: gsn = GSN_DROP_TRIG_REQ");
+//-------------------------------------------------------
+// We are waiting for DROP_TRIG response. We rely on
+// this to arrive and will continue dropping triggers
+// until completed.
+//-------------------------------------------------------
+
+ /**
+ * I'm currently dropping the trigger
+ */
+ return;
+ }//if
+ }//if
+
+//-------------------------------------------------------
+// If we are waiting for START_BACKUP responses we can
+// safely start dropping triggers (state == STARTED).
+// We will ignore any START_BACKUP responses after this.
+//-------------------------------------------------------
+ DEBUG_OUT("masterAbort: sendDropTrig");
+ sendDropTrig(signal, ptr); // dropping due to error
+}
+
+void
+Backup::masterSendAbortBackup(Signal* signal, BackupRecordPtr ptr)
+{
+ if (ptr.p->masterData.state.getState() != ABORTING) {
+ sendAbortBackupOrd(signal, ptr, AbortBackupOrd::BackupFailure);
+ ptr.p->masterData.state.setState(ABORTING);
+ }
+ const State s = ptr.p->masterData.state.getAbortState();
+
+ /**
+ * First inform to client
+ */
+ if(s == DEFINING) {
+ jam();
+#ifdef DEBUG_ABORT
+ ndbout_c("** Abort: sending BACKUP_REF to mgmtsrvr");
+#endif
+ sendBackupRef(ptr.p->clientRef, signal, ptr.p->clientData,
+ ptr.p->errorCode);
+
+ } else {
+ jam();
+#ifdef DEBUG_ABORT
+ ndbout_c("** Abort: sending BACKUP_ABORT_REP to mgmtsrvr");
+#endif
+ BackupAbortRep* rep = (BackupAbortRep*)signal->getDataPtrSend();
+ rep->backupId = ptr.p->backupId;
+ rep->senderData = ptr.p->clientData;
+ rep->reason = ptr.p->errorCode;
+ sendSignal(ptr.p->clientRef, GSN_BACKUP_ABORT_REP, signal,
+ BackupAbortRep::SignalLength, JBB);
+ }//if
+
+ // ptr.p->masterData.state.setState(INITIAL);
+
+ sendAbortBackupOrd(signal, ptr, AbortBackupOrd::BackupFailure);
+}
+
+/*****************************************************************************
+ *
+ * Slave functionallity: Define Backup
+ *
+ *****************************************************************************/
+void
+Backup::defineBackupRef(Signal* signal, BackupRecordPtr ptr, Uint32 errCode)
+{
+ if (ptr.p->slaveState.getState() == ABORTING) {
+ jam();
+ return;
+ }
+ ptr.p->slaveState.setState(ABORTING);
+
+ if (errCode != 0) {
+ jam();
+ ptr.p->setErrorCode(errCode);
+ }//if
+ ndbrequire(ptr.p->errorCode != 0);
+
+ DefineBackupRef* ref = (DefineBackupRef*)signal->getDataPtrSend();
+ ref->backupId = ptr.p->backupId;
+ ref->backupPtr = ptr.i;
+ ref->errorCode = ptr.p->errorCode;
+ sendSignal(ptr.p->masterRef, GSN_DEFINE_BACKUP_REF, signal,
+ DefineBackupRef::SignalLength, JBB);
+
+ closeFiles(signal, ptr);
+}
+
+void
+Backup::execDEFINE_BACKUP_REQ(Signal* signal)
+{
+ jamEntry();
+
+ DefineBackupReq* req = (DefineBackupReq*)signal->getDataPtr();
+
+ BackupRecordPtr ptr;
+ const Uint32 ptrI = req->backupPtr;
+ const Uint32 backupId = req->backupId;
+ const BlockReference senderRef = req->senderRef;
+
+ if(senderRef == reference()){
+ /**
+ * Signal sent from myself -> record already seized
+ */
+ jam();
+ c_backupPool.getPtr(ptr, ptrI);
+ } else { // from other node
+ jam();
+#ifdef DEBUG_ABORT
+ dumpUsedResources();
+#endif
+ if(!c_backups.seizeId(ptr, ptrI)) {
+ jam();
+ ndbrequire(false); // If master has succeeded slave should succed
+ }//if
+ }//if
+
+ CRASH_INSERTION((10014));
+
+ ptr.p->slaveState.forceState(INITIAL);
+ ptr.p->slaveState.setState(DEFINING);
+ ptr.p->errorCode = 0;
+ ptr.p->clientRef = req->clientRef;
+ ptr.p->clientData = req->clientData;
+ ptr.p->masterRef = senderRef;
+ ptr.p->nodes = req->nodes;
+ ptr.p->backupId = backupId;
+ ptr.p->backupKey[0] = req->backupKey[0];
+ ptr.p->backupKey[1] = req->backupKey[1];
+ ptr.p->backupDataLen = req->backupDataLen;
+ ptr.p->masterData.dropTrig.tableId = RNIL;
+ ptr.p->masterData.alterTrig.tableId = RNIL;
+ ptr.p->noOfBytes = 0;
+ ptr.p->noOfRecords = 0;
+ ptr.p->noOfLogBytes = 0;
+ ptr.p->noOfLogRecords = 0;
+ ptr.p->currGCP = 0;
+
+ /**
+ * Allocate files
+ */
+ BackupFilePtr files[3];
+ Uint32 noOfPages[] = {
+ NO_OF_PAGES_META_FILE,
+ 2, // 32k
+ 0 // 3M
+ };
+ const Uint32 maxInsert[] = {
+ 2048, // Temporarily to solve TR515
+ //25, // 100 bytes
+ 2048, // 4k
+ 16*3000, // Max 16 tuples
+ };
+ Uint32 minWrite[] = {
+ 8192,
+ 8192,
+ 32768
+ };
+ Uint32 maxWrite[] = {
+ 8192,
+ 8192,
+ 32768
+ };
+
+ minWrite[1] = c_defaults.m_minWriteSize;
+ maxWrite[1] = c_defaults.m_maxWriteSize;
+ noOfPages[1] = (c_defaults.m_logBufferSize + sizeof(Page32) - 1) /
+ sizeof(Page32);
+ minWrite[2] = c_defaults.m_minWriteSize;
+ maxWrite[2] = c_defaults.m_maxWriteSize;
+ noOfPages[2] = (c_defaults.m_dataBufferSize + sizeof(Page32) - 1) /
+ sizeof(Page32);
+
+ for(Uint32 i = 0; i<3; i++) {
+ jam();
+ if(!ptr.p->files.seize(files[i])) {
+ jam();
+ defineBackupRef(signal, ptr,
+ DefineBackupRef::FailedToAllocateFileRecord);
+ return;
+ }//if
+
+ files[i].p->tableId = RNIL;
+ files[i].p->backupPtr = ptr.i;
+ files[i].p->filePointer = RNIL;
+ files[i].p->fileDone = 0;
+ files[i].p->fileOpened = 0;
+ files[i].p->fileRunning = 0;
+ files[i].p->scanRunning = 0;
+ files[i].p->errorCode = 0;
+
+ if(files[i].p->pages.seize(noOfPages[i]) == false) {
+ jam();
+ DEBUG_OUT("Failed to seize " << noOfPages[i] << " pages");
+ defineBackupRef(signal, ptr, DefineBackupRef::FailedToAllocateBuffers);
+ return;
+ }//if
+ Page32Ptr pagePtr;
+ files[i].p->pages.getPtr(pagePtr, 0);
+
+ const char * msg = files[i].p->
+ operation.dataBuffer.setup((Uint32*)pagePtr.p,
+ noOfPages[i] * (sizeof(Page32) >> 2),
+ 128,
+ minWrite[i] >> 2,
+ maxWrite[i] >> 2,
+ maxInsert[i]);
+ if(msg != 0) {
+ jam();
+ defineBackupRef(signal, ptr, DefineBackupRef::FailedToSetupFsBuffers);
+ return;
+ }//if
+ }//for
+ files[0].p->fileType = BackupFormat::CTL_FILE;
+ files[1].p->fileType = BackupFormat::LOG_FILE;
+ files[2].p->fileType = BackupFormat::DATA_FILE;
+
+ ptr.p->ctlFilePtr = files[0].i;
+ ptr.p->logFilePtr = files[1].i;
+ ptr.p->dataFilePtr = files[2].i;
+
+ if (!verifyNodesAlive(ptr.p->nodes)) {
+ jam();
+ defineBackupRef(signal, ptr, DefineBackupRef::Undefined);
+ // sendBackupRef(signal, ptr,
+ // ptr.p->errorCode?ptr.p->errorCode:BackupRef::Undefined);
+ return;
+ }//if
+ if (ERROR_INSERTED(10027)) {
+ jam();
+ defineBackupRef(signal, ptr, 327);
+ // sendBackupRef(signal, ptr, 327);
+ return;
+ }//if
+
+ if(ptr.p->backupDataLen == 0) {
+ jam();
+ backupAllData(signal, ptr);
+ return;
+ }//if
+
+ /**
+ * Not implemented
+ */
+ ndbrequire(0);
+}
+
+void
+Backup::backupAllData(Signal* signal, BackupRecordPtr ptr)
+{
+ /**
+ * Get all tables from dict
+ */
+ ListTablesReq * req = (ListTablesReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = ptr.i;
+ req->requestData = 0;
+ sendSignal(DBDICT_REF, GSN_LIST_TABLES_REQ, signal,
+ ListTablesReq::SignalLength, JBB);
+}
+
+void
+Backup::execLIST_TABLES_CONF(Signal* signal)
+{
+ jamEntry();
+
+ ListTablesConf* conf = (ListTablesConf*)signal->getDataPtr();
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, conf->senderData);
+
+ const Uint32 len = signal->length() - ListTablesConf::HeaderLength;
+ for(unsigned int i = 0; i<len; i++) {
+ jam();
+ Uint32 tableId = ListTablesConf::getTableId(conf->tableData[i]);
+ Uint32 tableType = ListTablesConf::getTableType(conf->tableData[i]);
+ if (tableType != DictTabInfo::SystemTable &&
+ tableType != DictTabInfo::UserTable) {
+ jam();
+ continue;
+ }//if
+ TablePtr tabPtr;
+ ptr.p->tables.seize(tabPtr);
+ if(tabPtr.i == RNIL) {
+ jam();
+ defineBackupRef(signal, ptr, DefineBackupRef::FailedToAllocateTables);
+ return;
+ }//if
+ tabPtr.p->tableId = tableId;
+ tabPtr.p->tableType = tableType;
+ }//for
+
+ if(len == ListTablesConf::DataLength) {
+ jam();
+ /**
+ * Not finished...
+ */
+ return;
+ }//if
+
+ defineSlaveAbortCheck();
+
+ /**
+ * All tables fetched
+ */
+ openFiles(signal, ptr);
+}
+
+void
+Backup::openFiles(Signal* signal, BackupRecordPtr ptr)
+{
+ jam();
+
+ BackupFilePtr filePtr;
+
+ FsOpenReq * req = (FsOpenReq *)signal->getDataPtrSend();
+ req->userReference = reference();
+ req->fileFlags =
+ FsOpenReq::OM_WRITEONLY |
+ FsOpenReq::OM_TRUNCATE |
+ FsOpenReq::OM_CREATE |
+ FsOpenReq::OM_APPEND |
+ FsOpenReq::OM_SYNC;
+ FsOpenReq::v2_setCount(req->fileNumber, 0xFFFFFFFF);
+
+ /**
+ * Ctl file
+ */
+ c_backupFilePool.getPtr(filePtr, ptr.p->ctlFilePtr);
+ ndbrequire(filePtr.p->fileRunning == 0);
+ filePtr.p->fileRunning = 1;
+
+ req->userPointer = filePtr.i;
+ FsOpenReq::setVersion(req->fileNumber, 2);
+ FsOpenReq::setSuffix(req->fileNumber, FsOpenReq::S_CTL);
+ FsOpenReq::v2_setSequence(req->fileNumber, ptr.p->backupId);
+ FsOpenReq::v2_setNodeId(req->fileNumber, getOwnNodeId());
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, FsOpenReq::SignalLength, JBA);
+
+ /**
+ * Log file
+ */
+ c_backupFilePool.getPtr(filePtr, ptr.p->logFilePtr);
+ ndbrequire(filePtr.p->fileRunning == 0);
+ filePtr.p->fileRunning = 1;
+
+ req->userPointer = filePtr.i;
+ FsOpenReq::setVersion(req->fileNumber, 2);
+ FsOpenReq::setSuffix(req->fileNumber, FsOpenReq::S_LOG);
+ FsOpenReq::v2_setSequence(req->fileNumber, ptr.p->backupId);
+ FsOpenReq::v2_setNodeId(req->fileNumber, getOwnNodeId());
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, FsOpenReq::SignalLength, JBA);
+
+ /**
+ * Data file
+ */
+ c_backupFilePool.getPtr(filePtr, ptr.p->dataFilePtr);
+ ndbrequire(filePtr.p->fileRunning == 0);
+ filePtr.p->fileRunning = 1;
+
+ req->userPointer = filePtr.i;
+ FsOpenReq::setVersion(req->fileNumber, 2);
+ FsOpenReq::setSuffix(req->fileNumber, FsOpenReq::S_DATA);
+ FsOpenReq::v2_setSequence(req->fileNumber, ptr.p->backupId);
+ FsOpenReq::v2_setNodeId(req->fileNumber, getOwnNodeId());
+ FsOpenReq::v2_setCount(req->fileNumber, 0);
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, FsOpenReq::SignalLength, JBA);
+}
+
+void
+Backup::execFSOPENREF(Signal* signal)
+{
+ jamEntry();
+
+ FsRef * ref = (FsRef *)signal->getDataPtr();
+
+ const Uint32 userPtr = ref->userPointer;
+
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, userPtr);
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, filePtr.p->backupPtr);
+ ptr.p->setErrorCode(ref->errorCode);
+ openFilesReply(signal, ptr, filePtr);
+}
+
+void
+Backup::execFSOPENCONF(Signal* signal)
+{
+ jamEntry();
+
+ FsConf * conf = (FsConf *)signal->getDataPtr();
+
+ const Uint32 userPtr = conf->userPointer;
+ const Uint32 filePointer = conf->filePointer;
+
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, userPtr);
+ filePtr.p->filePointer = filePointer;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, filePtr.p->backupPtr);
+
+ ndbrequire(filePtr.p->fileOpened == 0);
+ filePtr.p->fileOpened = 1;
+ openFilesReply(signal, ptr, filePtr);
+}
+
+void
+Backup::openFilesReply(Signal* signal,
+ BackupRecordPtr ptr, BackupFilePtr filePtr)
+{
+ jam();
+
+ /**
+ * Mark files as "opened"
+ */
+ ndbrequire(filePtr.p->fileRunning == 1);
+ filePtr.p->fileRunning = 0;
+
+ /**
+ * Check if all files have recived open_reply
+ */
+ for(ptr.p->files.first(filePtr); filePtr.i!=RNIL;ptr.p->files.next(filePtr))
+ {
+ jam();
+ if(filePtr.p->fileRunning == 1) {
+ jam();
+ return;
+ }//if
+ }//for
+
+ defineSlaveAbortCheck();
+
+ /**
+ * Did open succeed for all files
+ */
+ if(ptr.p->checkError()) {
+ jam();
+ defineBackupRef(signal, ptr);
+ return;
+ }//if
+
+ /**
+ * Insert file headers
+ */
+ ptr.p->files.getPtr(filePtr, ptr.p->ctlFilePtr);
+ if(!insertFileHeader(BackupFormat::CTL_FILE, ptr.p, filePtr.p)) {
+ jam();
+ defineBackupRef(signal, ptr, DefineBackupRef::FailedInsertFileHeader);
+ return;
+ }//if
+
+ ptr.p->files.getPtr(filePtr, ptr.p->logFilePtr);
+ if(!insertFileHeader(BackupFormat::LOG_FILE, ptr.p, filePtr.p)) {
+ jam();
+ defineBackupRef(signal, ptr, DefineBackupRef::FailedInsertFileHeader);
+ return;
+ }//if
+
+ ptr.p->files.getPtr(filePtr, ptr.p->dataFilePtr);
+ if(!insertFileHeader(BackupFormat::DATA_FILE, ptr.p, filePtr.p)) {
+ jam();
+ defineBackupRef(signal, ptr, DefineBackupRef::FailedInsertFileHeader);
+ return;
+ }//if
+
+ /**
+ * Start CTL file thread
+ */
+ ptr.p->files.getPtr(filePtr, ptr.p->ctlFilePtr);
+ filePtr.p->fileRunning = 1;
+
+ signal->theData[0] = BackupContinueB::START_FILE_THREAD;
+ signal->theData[1] = ptr.p->ctlFilePtr;
+ sendSignalWithDelay(BACKUP_REF, GSN_CONTINUEB, signal, 100, 2);
+
+ /**
+ * Insert table list in ctl file
+ */
+ FsBuffer & buf = filePtr.p->operation.dataBuffer;
+
+ const Uint32 sz =
+ (sizeof(BackupFormat::CtlFile::TableList) >> 2) +
+ ptr.p->tables.noOfElements() - 1;
+
+ Uint32 * dst;
+ ndbrequire(sz < buf.getMaxWrite());
+ if(!buf.getWritePtr(&dst, sz)) {
+ jam();
+ defineBackupRef(signal, ptr, DefineBackupRef::FailedInsertTableList);
+ return;
+ }//if
+
+ BackupFormat::CtlFile::TableList* tl =
+ (BackupFormat::CtlFile::TableList*)dst;
+ tl->SectionType = htonl(BackupFormat::TABLE_LIST);
+ tl->SectionLength = htonl(sz);
+
+ TablePtr tabPtr;
+ Uint32 count = 0;
+ for(ptr.p->tables.first(tabPtr);
+ tabPtr.i != RNIL;
+ ptr.p->tables.next(tabPtr)){
+ jam();
+ tl->TableIds[count] = htonl(tabPtr.p->tableId);
+ count++;
+ }//for
+
+ buf.updateWritePtr(sz);
+
+ /**
+ * Start getting table definition data
+ */
+ ndbrequire(ptr.p->tables.first(tabPtr));
+
+ signal->theData[0] = BackupContinueB::BUFFER_FULL_META;
+ signal->theData[1] = ptr.i;
+ signal->theData[2] = tabPtr.i;
+ sendSignalWithDelay(BACKUP_REF, GSN_CONTINUEB, signal, 100, 3);
+ return;
+}
+
+bool
+Backup::insertFileHeader(BackupFormat::FileType ft,
+ BackupRecord * ptrP,
+ BackupFile * filePtrP){
+ FsBuffer & buf = filePtrP->operation.dataBuffer;
+
+ const Uint32 sz = sizeof(BackupFormat::FileHeader) >> 2;
+
+ Uint32 * dst;
+ ndbrequire(sz < buf.getMaxWrite());
+ if(!buf.getWritePtr(&dst, sz)) {
+ jam();
+ return false;
+ }//if
+
+ BackupFormat::FileHeader* header = (BackupFormat::FileHeader*)dst;
+ ndbrequire(sizeof(header->Magic) == sizeof(BACKUP_MAGIC));
+ memcpy(header->Magic, BACKUP_MAGIC, sizeof(BACKUP_MAGIC));
+ header->NdbVersion = htonl(NDB_VERSION);
+ header->SectionType = htonl(BackupFormat::FILE_HEADER);
+ header->SectionLength = htonl(sz - 3);
+ header->FileType = htonl(ft);
+ header->BackupId = htonl(ptrP->backupId);
+ header->BackupKey_0 = htonl(ptrP->backupKey[0]);
+ header->BackupKey_1 = htonl(ptrP->backupKey[1]);
+ header->ByteOrder = 0x12345678;
+
+ buf.updateWritePtr(sz);
+ return true;
+}
+
+void
+Backup::execGET_TABINFOREF(Signal* signal)
+{
+ GetTabInfoRef * ref = (GetTabInfoRef*)signal->getDataPtr();
+
+ const Uint32 senderData = ref->senderData;
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, senderData);
+
+ defineSlaveAbortCheck();
+
+ defineBackupRef(signal, ptr, ref->errorCode);
+}
+
+void
+Backup::execGET_TABINFO_CONF(Signal* signal)
+{
+ jamEntry();
+
+ if(!assembleFragments(signal)) {
+ jam();
+ return;
+ }//if
+
+ GetTabInfoConf * const conf = (GetTabInfoConf*)signal->getDataPtr();
+ //const Uint32 senderRef = info->senderRef;
+ const Uint32 len = conf->totalLen;
+ const Uint32 senderData = conf->senderData;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, senderData);
+
+ defineSlaveAbortCheck();
+
+ SegmentedSectionPtr dictTabInfoPtr;
+ signal->getSection(dictTabInfoPtr, GetTabInfoConf::DICT_TAB_INFO);
+ ndbrequire(dictTabInfoPtr.sz == len);
+
+ /**
+ * No of pages needed
+ */
+ const Uint32 noPages = (len + sizeof(Page32) - 1) / sizeof(Page32);
+ if(ptr.p->pages.getSize() < noPages) {
+ jam();
+ ptr.p->pages.release();
+ if(ptr.p->pages.seize(noPages) == false) {
+ jam();
+ ptr.p->setErrorCode(DefineBackupRef::FailedAllocateTableMem);
+ ndbrequire(false);
+ releaseSections(signal);
+ defineBackupRef(signal, ptr);
+ return;
+ }//if
+ }//if
+
+ BackupFilePtr filePtr;
+ ptr.p->files.getPtr(filePtr, ptr.p->ctlFilePtr);
+ FsBuffer & buf = filePtr.p->operation.dataBuffer;
+ { // Write into ctl file
+ Uint32* dst, dstLen = len + 2;
+ if(!buf.getWritePtr(&dst, dstLen)) {
+ jam();
+ ndbrequire(false);
+ ptr.p->setErrorCode(DefineBackupRef::FailedAllocateTableMem);
+ releaseSections(signal);
+ defineBackupRef(signal, ptr);
+ return;
+ }//if
+ if(dst != 0) {
+ jam();
+
+ BackupFormat::CtlFile::TableDescription * desc =
+ (BackupFormat::CtlFile::TableDescription*)dst;
+ desc->SectionType = htonl(BackupFormat::TABLE_DESCRIPTION);
+ desc->SectionLength = htonl(len + 2);
+ dst += 2;
+
+ copy(dst, dictTabInfoPtr);
+ buf.updateWritePtr(dstLen);
+ }//if
+ }
+
+ ndbrequire(ptr.p->pages.getSize() >= noPages);
+ Page32Ptr pagePtr;
+ ptr.p->pages.getPtr(pagePtr, 0);
+ copy(&pagePtr.p->data[0], dictTabInfoPtr);
+ releaseSections(signal);
+
+ if(ptr.p->checkError()) {
+ jam();
+ defineBackupRef(signal, ptr);
+ return;
+ }//if
+
+ TablePtr tabPtr = parseTableDescription(signal, ptr, len);
+ if(tabPtr.i == RNIL) {
+ jam();
+ defineBackupRef(signal, ptr);
+ return;
+ }//if
+
+ ptr.p->tables.next(tabPtr);
+ if(tabPtr.i == RNIL) {
+ jam();
+
+ ptr.p->pages.release();
+
+ ndbrequire(ptr.p->tables.first(tabPtr));
+ signal->theData[0] = RNIL;
+ signal->theData[1] = tabPtr.p->tableId;
+ signal->theData[2] = ptr.i;
+ sendSignal(DBDIH_REF, GSN_DI_FCOUNTREQ, signal, 3, JBB);
+ return;
+ }//if
+
+ signal->theData[0] = BackupContinueB::BUFFER_FULL_META;
+ signal->theData[1] = ptr.i;
+ signal->theData[2] = tabPtr.i;
+ sendSignalWithDelay(BACKUP_REF, GSN_CONTINUEB, signal, 100, 3);
+ return;
+}
+
+Backup::TablePtr
+Backup::parseTableDescription(Signal* signal, BackupRecordPtr ptr, Uint32 len)
+{
+
+ Page32Ptr pagePtr;
+ ptr.p->pages.getPtr(pagePtr, 0);
+
+ SimplePropertiesLinearReader it(&pagePtr.p->data[0], len);
+
+ it.first();
+
+ DictTabInfo::Table tmpTab; tmpTab.init();
+ SimpleProperties::UnpackStatus stat;
+ stat = SimpleProperties::unpack(it, &tmpTab,
+ DictTabInfo::TableMapping,
+ DictTabInfo::TableMappingSize,
+ true, true);
+ ndbrequire(stat == SimpleProperties::Break);
+
+ TablePtr tabPtr;
+ ndbrequire(findTable(ptr, tabPtr, tmpTab.TableId));
+
+ /**
+ * Initialize table object
+ */
+ tabPtr.p->frag_mask = RNIL;
+
+ tabPtr.p->schemaVersion = tmpTab.TableVersion;
+ tabPtr.p->noOfAttributes = tmpTab.NoOfAttributes;
+ tabPtr.p->noOfKeys = tmpTab.NoOfKeyAttr;
+ tabPtr.p->noOfNull = 0;
+ tabPtr.p->noOfVariable = 0; // Computed while iterating over attribs
+ tabPtr.p->sz_FixedKeys = 0; // Computed while iterating over attribs
+ tabPtr.p->sz_FixedAttributes = 0; // Computed while iterating over attribs
+ tabPtr.p->variableKeyId = RNIL; // Computed while iterating over attribs
+ tabPtr.p->triggerIds[0] = ILLEGAL_TRIGGER_ID;
+ tabPtr.p->triggerIds[1] = ILLEGAL_TRIGGER_ID;
+ tabPtr.p->triggerIds[2] = ILLEGAL_TRIGGER_ID;
+ tabPtr.p->triggerAllocated[0] = false;
+ tabPtr.p->triggerAllocated[1] = false;
+ tabPtr.p->triggerAllocated[2] = false;
+
+ if(tabPtr.p->attributes.seize(tabPtr.p->noOfAttributes) == false) {
+ jam();
+ ptr.p->setErrorCode(DefineBackupRef::FailedToAllocateAttributeRecord);
+ tabPtr.i = RNIL;
+ return tabPtr;
+ }//if
+
+ const Uint32 count = tabPtr.p->noOfAttributes;
+ for(Uint32 i = 0; i<count; i++) {
+ jam();
+ DictTabInfo::Attribute tmp; tmp.init();
+ stat = SimpleProperties::unpack(it, &tmp,
+ DictTabInfo::AttributeMapping,
+ DictTabInfo::AttributeMappingSize,
+ true, true);
+
+ ndbrequire(stat == SimpleProperties::Break);
+
+ const Uint32 arr = tmp.AttributeArraySize;
+ const Uint32 sz = 1 << tmp.AttributeSize;
+ const Uint32 sz32 = (sz * arr + 31) >> 5;
+
+ AttributePtr attrPtr;
+ tabPtr.p->attributes.getPtr(attrPtr, tmp.AttributeId);
+
+ attrPtr.p->data.nullable = tmp.AttributeNullableFlag;
+ attrPtr.p->data.fixed = (tmp.AttributeArraySize != 0);
+ attrPtr.p->data.key = tmp.AttributeKeyFlag;
+ attrPtr.p->data.sz32 = sz32;
+
+ /**
+ * Either
+ * 1) Fixed
+ * 2) Nullable
+ * 3) Variable
+ * 4) Fixed key
+ * 5) Variable key
+ */
+ if(attrPtr.p->data.key == false) {
+ jam();
+
+ if(attrPtr.p->data.fixed == true && attrPtr.p->data.nullable == false) {
+ jam();
+ attrPtr.p->data.offset = tabPtr.p->sz_FixedAttributes;
+ tabPtr.p->sz_FixedAttributes += sz32;
+ }//if
+
+ if(attrPtr.p->data.fixed == true && attrPtr.p->data.nullable == true) {
+ jam();
+ attrPtr.p->data.offset = 0;
+
+ attrPtr.p->data.offsetNull = tabPtr.p->noOfNull;
+ tabPtr.p->noOfNull++;
+ tabPtr.p->noOfVariable++;
+ }//if
+
+ if(attrPtr.p->data.fixed == false) {
+ jam();
+ tabPtr.p->noOfVariable++;
+ ndbrequire(0);
+ }//if
+
+ } else if(attrPtr.p->data.key == true) {
+ jam();
+ ndbrequire(attrPtr.p->data.nullable == false);
+
+ if(attrPtr.p->data.fixed == true) { // Fixed key
+ jam();
+ tabPtr.p->sz_FixedKeys += sz32;
+ }//if
+
+ if(attrPtr.p->data.fixed == false) { // Variable key
+ jam();
+ attrPtr.p->data.offset = 0;
+ tabPtr.p->noOfVariable++;
+ ndbrequire(tabPtr.p->variableKeyId == RNIL); // Only one variable key
+ tabPtr.p->variableKeyId = attrPtr.i;
+ ndbrequire(0);
+ }//if
+ }//if
+
+ it.next(); // Move Past EndOfAttribute
+ }//for
+ return tabPtr;
+}
+
+void
+Backup::execDI_FCOUNTCONF(Signal* signal)
+{
+ jamEntry();
+
+ const Uint32 userPtr = signal->theData[0];
+ const Uint32 fragCount = signal->theData[1];
+ const Uint32 tableId = signal->theData[2];
+ const Uint32 senderData = signal->theData[3];
+
+ ndbrequire(userPtr == RNIL && signal->length() == 5);
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, senderData);
+
+ defineSlaveAbortCheck();
+
+ TablePtr tabPtr;
+ ndbrequire(findTable(ptr, tabPtr, tableId));
+
+ ndbrequire(tabPtr.p->fragments.seize(fragCount) != false);
+ tabPtr.p->frag_mask = calculate_frag_mask(fragCount);
+ for(Uint32 i = 0; i<fragCount; i++) {
+ jam();
+ FragmentPtr fragPtr;
+ tabPtr.p->fragments.getPtr(fragPtr, i);
+ fragPtr.p->scanned = 0;
+ fragPtr.p->scanning = 0;
+ fragPtr.p->tableId = tableId;
+ fragPtr.p->node = RNIL;
+ }//for
+
+ /**
+ * Next table
+ */
+ if(ptr.p->tables.next(tabPtr)) {
+ jam();
+ signal->theData[0] = RNIL;
+ signal->theData[1] = tabPtr.p->tableId;
+ signal->theData[2] = ptr.i;
+ sendSignal(DBDIH_REF, GSN_DI_FCOUNTREQ, signal, 3, JBB);
+ return;
+ }//if
+
+ ptr.p->tables.first(tabPtr);
+ getFragmentInfo(signal, ptr, tabPtr, 0);
+}
+
+void
+Backup::getFragmentInfo(Signal* signal,
+ BackupRecordPtr ptr, TablePtr tabPtr, Uint32 fragNo)
+{
+ jam();
+
+ for(; tabPtr.i != RNIL; ptr.p->tables.next(tabPtr)) {
+ jam();
+ const Uint32 fragCount = tabPtr.p->fragments.getSize();
+ for(; fragNo < fragCount; fragNo ++) {
+ jam();
+ FragmentPtr fragPtr;
+ tabPtr.p->fragments.getPtr(fragPtr, fragNo);
+
+ if(fragPtr.p->scanned == 0 && fragPtr.p->scanning == 0) {
+ jam();
+ signal->theData[0] = RNIL;
+ signal->theData[1] = ptr.i;
+ signal->theData[2] = tabPtr.p->tableId;
+ signal->theData[3] = fragNo;
+ sendSignal(DBDIH_REF, GSN_DIGETPRIMREQ, signal, 4, JBB);
+ return;
+ }//if
+ }//for
+ fragNo = 0;
+ }//for
+
+ getFragmentInfoDone(signal, ptr);
+}
+
+void
+Backup::execDIGETPRIMCONF(Signal* signal)
+{
+ jamEntry();
+
+ const Uint32 userPtr = signal->theData[0];
+ const Uint32 senderData = signal->theData[1];
+ const Uint32 nodeCount = signal->theData[6];
+ const Uint32 tableId = signal->theData[7];
+ const Uint32 fragNo = signal->theData[8];
+
+ ndbrequire(userPtr == RNIL && signal->length() == 9);
+ ndbrequire(nodeCount > 0 && nodeCount <= MAX_REPLICAS);
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, senderData);
+
+ defineSlaveAbortCheck();
+
+ TablePtr tabPtr;
+ ndbrequire(findTable(ptr, tabPtr, tableId));
+
+ FragmentPtr fragPtr;
+ tabPtr.p->fragments.getPtr(fragPtr, fragNo);
+
+ fragPtr.p->node = signal->theData[2];
+
+ getFragmentInfo(signal, ptr, tabPtr, fragNo + 1);
+}
+
+void
+Backup::getFragmentInfoDone(Signal* signal, BackupRecordPtr ptr)
+{
+ // Slave must now hold on to master data until
+ // AbortBackupOrd::OkToClean signal
+ ptr.p->okToCleanMaster = false;
+ ptr.p->slaveState.setState(DEFINED);
+ DefineBackupConf * conf = (DefineBackupConf*)signal->getDataPtr();
+ conf->backupPtr = ptr.i;
+ conf->backupId = ptr.p->backupId;
+ sendSignal(ptr.p->masterRef, GSN_DEFINE_BACKUP_CONF, signal,
+ DefineBackupConf::SignalLength, JBB);
+}
+
+
+/*****************************************************************************
+ *
+ * Slave functionallity: Start backup
+ *
+ *****************************************************************************/
+void
+Backup::execSTART_BACKUP_REQ(Signal* signal)
+{
+ jamEntry();
+
+ CRASH_INSERTION((10015));
+
+ StartBackupReq* req = (StartBackupReq*)signal->getDataPtr();
+ const Uint32 ptrI = req->backupPtr;
+ const Uint32 backupId = req->backupId;
+ const Uint32 signalNo = req->signalNo;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ slaveAbortCheck(); // macro will do return if ABORTING
+
+ ptr.p->slaveState.setState(STARTED);
+
+ for(Uint32 i = 0; i<req->noOfTableTriggers; i++) {
+ jam();
+ TablePtr tabPtr;
+ ndbrequire(findTable(ptr, tabPtr, req->tableTriggers[i].tableId));
+ for(Uint32 j = 0; j<3; j++) {
+ jam();
+ const Uint32 triggerId = req->tableTriggers[i].triggerIds[j];
+ tabPtr.p->triggerIds[j] = triggerId;
+
+ TriggerPtr trigPtr;
+ if(!ptr.p->triggers.seizeId(trigPtr, triggerId)) {
+ jam();
+ StartBackupRef* ref = (StartBackupRef*)signal->getDataPtrSend();
+ ref->backupPtr = ptr.i;
+ ref->backupId = ptr.p->backupId;
+ ref->signalNo = signalNo;
+ ref->errorCode = StartBackupRef::FailedToAllocateTriggerRecord;
+ sendSignal(ptr.p->masterRef, GSN_START_BACKUP_REF, signal,
+ StartBackupRef::SignalLength, JBB);
+ return;
+ }//if
+
+ tabPtr.p->triggerAllocated[i] = true;
+ trigPtr.p->backupPtr = ptr.i;
+ trigPtr.p->tableId = tabPtr.p->tableId;
+ trigPtr.p->tab_ptr_i = tabPtr.i;
+ trigPtr.p->logEntry = 0;
+ trigPtr.p->event = j;
+ trigPtr.p->maxRecordSize = 2048;
+ trigPtr.p->operation =
+ &ptr.p->files.getPtr(ptr.p->logFilePtr)->operation;
+ trigPtr.p->operation->noOfBytes = 0;
+ trigPtr.p->operation->noOfRecords = 0;
+ trigPtr.p->errorCode = 0;
+ }//for
+ }//for
+
+ /**
+ * Start file threads...
+ */
+ BackupFilePtr filePtr;
+ for(ptr.p->files.first(filePtr);
+ filePtr.i!=RNIL;
+ ptr.p->files.next(filePtr)){
+ jam();
+ if(filePtr.p->fileRunning == 0) {
+ jam();
+ filePtr.p->fileRunning = 1;
+ signal->theData[0] = BackupContinueB::START_FILE_THREAD;
+ signal->theData[1] = filePtr.i;
+ sendSignalWithDelay(BACKUP_REF, GSN_CONTINUEB, signal, 100, 2);
+ }//if
+ }//for
+
+ StartBackupConf* conf = (StartBackupConf*)signal->getDataPtrSend();
+ conf->backupPtr = ptr.i;
+ conf->backupId = ptr.p->backupId;
+ conf->signalNo = signalNo;
+ sendSignal(ptr.p->masterRef, GSN_START_BACKUP_CONF, signal,
+ StartBackupConf::SignalLength, JBB);
+}
+
+/*****************************************************************************
+ *
+ * Slave functionallity: Backup fragment
+ *
+ *****************************************************************************/
+void
+Backup::execBACKUP_FRAGMENT_REQ(Signal* signal)
+{
+ jamEntry();
+ BackupFragmentReq* req = (BackupFragmentReq*)signal->getDataPtr();
+
+ CRASH_INSERTION((10016));
+
+ const Uint32 ptrI = req->backupPtr;
+ const Uint32 backupId = req->backupId;
+ const Uint32 tableId = req->tableId;
+ const Uint32 fragNo = req->fragmentNo;
+ const Uint32 count = req->count;
+
+ /**
+ * Get backup record
+ */
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ slaveAbortCheck(); // macro will do return if ABORTING
+
+ ptr.p->slaveState.setState(SCANNING);
+
+ /**
+ * Get file
+ */
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, ptr.p->dataFilePtr);
+
+ ndbrequire(filePtr.p->backupPtr == ptrI);
+ ndbrequire(filePtr.p->fileOpened == 1);
+ ndbrequire(filePtr.p->fileRunning == 1);
+ ndbrequire(filePtr.p->scanRunning == 0);
+ ndbrequire(filePtr.p->fileDone == 0);
+
+ /**
+ * Get table
+ */
+ TablePtr tabPtr;
+ ndbrequire(findTable(ptr, tabPtr, tableId));
+
+ /**
+ * Get fragment
+ */
+ FragmentPtr fragPtr;
+ tabPtr.p->fragments.getPtr(fragPtr, fragNo);
+
+ ndbrequire(fragPtr.p->scanned == 0);
+ ndbrequire(fragPtr.p->scanning == 0 ||
+ refToNode(ptr.p->masterRef) == getOwnNodeId());
+
+ /**
+ * Init operation
+ */
+ if(filePtr.p->tableId != tableId) {
+ jam();
+ filePtr.p->operation.init(tabPtr);
+ filePtr.p->tableId = tableId;
+ }//if
+
+ /**
+ * Check for space in buffer
+ */
+ if(!filePtr.p->operation.newFragment(tableId, fragNo)) {
+ jam();
+ req->count = count + 1;
+ sendSignalWithDelay(BACKUP_REF, GSN_BACKUP_FRAGMENT_REQ, signal, 50,
+ signal->length());
+ ptr.p->slaveState.setState(STARTED);
+ return;
+ }//if
+
+ /**
+ * Mark things as "in use"
+ */
+ fragPtr.p->scanning = 1;
+ filePtr.p->fragmentNo = fragNo;
+
+ /**
+ * Start scan
+ */
+ {
+ filePtr.p->scanRunning = 1;
+
+ Table & table = * tabPtr.p;
+ ScanFragReq * req = (ScanFragReq *)signal->getDataPtrSend();
+ const Uint32 parallelism = 16;
+ const Uint32 attrLen = 5 + table.noOfAttributes - table.noOfKeys;
+
+ req->senderData = filePtr.i;
+ req->resultRef = reference();
+ req->schemaVersion = table.schemaVersion;
+ req->fragmentNo = fragNo;
+ req->requestInfo = 0;
+ req->savePointId = 0;
+ req->tableId = table.tableId;
+ ScanFragReq::setConcurrency(req->requestInfo, parallelism);
+ ScanFragReq::setLockMode(req->requestInfo, 0);
+ ScanFragReq::setHoldLockFlag(req->requestInfo, 0);
+ ScanFragReq::setKeyinfoFlag(req->requestInfo, 1);
+ ScanFragReq::setAttrLen(req->requestInfo,attrLen);
+ req->transId1 = 0;
+ req->transId2 = (BACKUP << 20) + (getOwnNodeId() << 8);
+
+ for(unsigned int i = 0; i<parallelism; i++) {
+ jam();
+ req->clientOpPtr[i] = filePtr.i;
+ }//for
+ sendSignal(DBLQH_REF, GSN_SCAN_FRAGREQ, signal, 25, JBB);
+
+ signal->theData[0] = filePtr.i;
+ signal->theData[1] = 0;
+ signal->theData[2] = (BACKUP << 20) + (getOwnNodeId() << 8);
+
+ // Return all
+ signal->theData[3] = table.noOfAttributes - table.noOfKeys;
+ signal->theData[4] = 0;
+ signal->theData[5] = 0;
+ signal->theData[6] = 0;
+ signal->theData[7] = 0;
+
+ Uint32 dataPos = 8;
+ for(Uint32 i = 0; i<table.noOfAttributes; i++) {
+ jam();
+ AttributePtr attr;
+ table.attributes.getPtr(attr, i);
+ if(attr.p->data.key != 0) {
+ jam();
+ continue;
+ }//if
+
+ AttributeHeader::init(&signal->theData[dataPos], i, 0);
+ dataPos++;
+ if(dataPos == 25) {
+ jam();
+ sendSignal(DBLQH_REF, GSN_ATTRINFO, signal, 25, JBB);
+ dataPos = 3;
+ }//if
+ }//for
+ if(dataPos != 3) {
+ jam();
+ sendSignal(DBLQH_REF, GSN_ATTRINFO, signal, dataPos, JBB);
+ }//if
+ }
+}
+
+void
+Backup::execSCAN_HBREP(Signal* signal)
+{
+ jamEntry();
+}
+
+void
+Backup::execTRANSID_AI(Signal* signal)
+{
+ jamEntry();
+
+ const Uint32 filePtrI = signal->theData[0];
+ //const Uint32 transId1 = signal->theData[1];
+ //const Uint32 transId2 = signal->theData[2];
+ const Uint32 dataLen = signal->length() - 3;
+
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, filePtrI);
+
+ OperationRecord & op = filePtr.p->operation;
+
+ TablePtr tabPtr;
+ c_tablePool.getPtr(tabPtr, op.tablePtr);
+
+ Table & table = * tabPtr.p;
+
+ /**
+ * Unpack data
+ */
+ op.attrSzTotal += dataLen;
+
+ Uint32 srcSz = dataLen;
+ const Uint32 * src = &signal->theData[3];
+
+ Uint32 * dst = op.dst;
+ Uint32 dstSz = op.attrSzLeft;
+
+ while(srcSz > 0) {
+ jam();
+
+ if(dstSz == 0) {
+ jam();
+
+ /**
+ * Finished with one attribute now find next
+ */
+ const AttributeHeader attrHead(* src);
+ const Uint32 attrId = attrHead.getAttributeId();
+ const bool null = attrHead.isNULL();
+ const Attribute::Data attr = table.attributes.getPtr(attrId)->data;
+
+ srcSz -= attrHead.getHeaderSize();
+ src += attrHead.getHeaderSize();
+
+ if(null) {
+ jam();
+ ndbrequire(attr.nullable);
+ op.nullAttribute(attr.offsetNull);
+ dstSz = 0;
+ continue;
+ }//if
+
+ dstSz = attrHead.getDataSize();
+ ndbrequire(dstSz == attr.sz32);
+ if(attr.fixed && ! attr.nullable) {
+ jam();
+ dst = op.newAttrib(attr.offset, dstSz);
+ } else if (attr.fixed && attr.nullable) {
+ jam();
+ dst = op.newNullable(attrId, dstSz);
+ } else {
+ ndbrequire(false);
+ //dst = op.newVariable(attrId, attrSize);
+ }//if
+ }//if
+
+ const Uint32 szCopy = (dstSz > srcSz) ? srcSz : dstSz;
+ memcpy(dst, src, (szCopy << 2));
+
+ srcSz -= szCopy;
+ dstSz -= szCopy;
+ src += szCopy;
+ dst += szCopy;
+ }//while
+ op.dst = dst;
+ op.attrSzLeft = dstSz;
+
+ if(op.finished()){
+ jam();
+ op.newRecord(op.dst);
+ }
+}
+
+void
+Backup::execKEYINFO20(Signal* signal)
+{
+ jamEntry();
+
+ const Uint32 filePtrI = signal->theData[0];
+ const Uint32 keyLen = signal->theData[1];
+ //const Uint32 scanInfo = signal->theData[2];
+ //const Uint32 transId1 = signal->theData[3];
+ //const Uint32 transId2 = signal->theData[4];
+ const Uint32 dataLen = signal->length() - 5;
+
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, filePtrI);
+
+ OperationRecord & op = filePtr.p->operation;
+
+ /**
+ * Unpack data
+ */
+ ndbrequire(keyLen == dataLen);
+ const Uint32 * src = &signal->theData[5];
+ const Uint32 klFixed = op.getFixedKeySize();
+ ndbrequire(keyLen >= klFixed);
+
+ Uint32 * dst = op.newKey();
+ memcpy(dst, src, klFixed << 2);
+
+ const Uint32 szLeft = (keyLen - klFixed);
+ if(szLeft > 0) {
+ jam();
+ src += klFixed;
+ dst = op.newVariableKey(szLeft);
+ memcpy(dst, src, (szLeft << 2));
+ ndbrequire(0);
+ }//if
+
+ if(op.finished()){
+ jam();
+ op.newRecord(op.dst);
+ }
+}
+
+void
+Backup::OperationRecord::init(const TablePtr & ptr)
+{
+
+ tablePtr = ptr.i;
+ noOfAttributes = (ptr.p->noOfAttributes - ptr.p->noOfKeys) + 1;
+ variableKeyId = ptr.p->variableKeyId;
+
+ sz_Bitmask = (ptr.p->noOfNull + 31) >> 5;
+ sz_FixedKeys = ptr.p->sz_FixedKeys;
+ sz_FixedAttribs = ptr.p->sz_FixedAttributes;
+
+ if(ptr.p->noOfVariable == 0) {
+ jam();
+ maxRecordSize = 1 + sz_Bitmask + sz_FixedKeys + sz_FixedAttribs;
+ } else {
+ jam();
+ maxRecordSize =
+ 1 + sz_Bitmask + 2048 /* Max tuple size */ + 2 * ptr.p->noOfVariable;
+ }//if
+}
+
+bool
+Backup::OperationRecord::newFragment(Uint32 tableId, Uint32 fragNo)
+{
+ Uint32 * tmp;
+ const Uint32 headSz = (sizeof(BackupFormat::DataFile::FragmentHeader) >> 2);
+ const Uint32 sz = headSz + 16 * maxRecordSize;
+
+ ndbrequire(sz < dataBuffer.getMaxWrite());
+ if(dataBuffer.getWritePtr(&tmp, sz)) {
+ jam();
+ BackupFormat::DataFile::FragmentHeader * head =
+ (BackupFormat::DataFile::FragmentHeader*)tmp;
+
+ head->SectionType = htonl(BackupFormat::FRAGMENT_HEADER);
+ head->SectionLength = htonl(headSz);
+ head->TableId = htonl(tableId);
+ head->FragmentNo = htonl(fragNo);
+ head->ChecksumType = htonl(0);
+
+ opNoDone = opNoConf = 0;
+ memset(attrLen, 0, sizeof(attrLen));
+ newRecord(tmp + headSz);
+ scanStart = tmp;
+ scanStop = (tmp + headSz);
+
+ noOfRecords = 0;
+ noOfBytes = 0;
+ return true;
+ }//if
+ return false;
+}
+
+bool
+Backup::OperationRecord::fragComplete(Uint32 tableId, Uint32 fragNo)
+{
+ Uint32 * tmp;
+ const Uint32 footSz = sizeof(BackupFormat::DataFile::FragmentFooter) >> 2;
+
+ if(dataBuffer.getWritePtr(&tmp, footSz + 1)) {
+ jam();
+ * tmp = 0; // Finish record stream
+ tmp++;
+ BackupFormat::DataFile::FragmentFooter * foot =
+ (BackupFormat::DataFile::FragmentFooter*)tmp;
+ foot->SectionType = htonl(BackupFormat::FRAGMENT_FOOTER);
+ foot->SectionLength = htonl(footSz);
+ foot->TableId = htonl(tableId);
+ foot->FragmentNo = htonl(fragNo);
+ foot->NoOfRecords = htonl(noOfRecords);
+ foot->Checksum = htonl(0);
+ dataBuffer.updateWritePtr(footSz + 1);
+ return true;
+ }//if
+ return false;
+}
+
+bool
+Backup::OperationRecord::newScan()
+{
+ Uint32 * tmp;
+ ndbrequire(16 * maxRecordSize < dataBuffer.getMaxWrite());
+ if(dataBuffer.getWritePtr(&tmp, 16 * maxRecordSize)) {
+ jam();
+ opNoDone = opNoConf = 0;
+ memset(attrLen, 0, sizeof(attrLen));
+ newRecord(tmp);
+ scanStart = tmp;
+ scanStop = tmp;
+ return true;
+ }//if
+ return false;
+}
+
+bool
+Backup::OperationRecord::scanConf(Uint32 noOfOps, Uint32 opLen[])
+{
+ const Uint32 done = opNoDone-opNoConf;
+
+ ndbrequire(noOfOps == done);
+ ndbrequire(memcmp(&attrLen[opNoConf], opLen, done << 2) == 0);
+ opNoConf = opNoDone;
+
+ const Uint32 len = (scanStop - scanStart);
+ ndbrequire(len < dataBuffer.getMaxWrite());
+ dataBuffer.updateWritePtr(len);
+ noOfBytes += (len << 2);
+ return true;
+}
+
+void
+Backup::execSCAN_FRAGREF(Signal* signal)
+{
+ jamEntry();
+
+ ScanFragRef * ref = (ScanFragRef*)signal->getDataPtr();
+
+ const Uint32 filePtrI = ref->senderData;
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, filePtrI);
+
+ filePtr.p->errorCode = ref->errorCode;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, filePtr.p->backupPtr);
+
+ abortFile(signal, ptr, filePtr);
+}
+
+void
+Backup::execSCAN_FRAGCONF(Signal* signal)
+{
+ jamEntry();
+
+ CRASH_INSERTION((10017));
+
+ ScanFragConf * conf = (ScanFragConf*)signal->getDataPtr();
+
+ const Uint32 filePtrI = conf->senderData;
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, filePtrI);
+
+ OperationRecord & op = filePtr.p->operation;
+ op.scanConf(conf->completedOps, conf->opReturnDataLen);
+
+ const Uint32 completed = conf->fragmentCompleted;
+ if(completed != 2) {
+ jam();
+
+ checkScan(signal, filePtr);
+ return;
+ }//if
+
+ fragmentCompleted(signal, filePtr);
+}
+
+void
+Backup::fragmentCompleted(Signal* signal, BackupFilePtr filePtr)
+{
+ jam();
+
+ if(filePtr.p->errorCode != 0){
+ jam();
+ abortFileHook(signal, filePtr, true); // Scan completed
+ return;
+ }//if
+
+ OperationRecord & op = filePtr.p->operation;
+ if(!op.fragComplete(filePtr.p->tableId, filePtr.p->fragmentNo)) {
+ jam();
+ signal->theData[0] = BackupContinueB::BUFFER_FULL_FRAG_COMPLETE;
+ signal->theData[1] = filePtr.i;
+ sendSignalWithDelay(BACKUP_REF, GSN_CONTINUEB, signal, 50, 2);
+ return;
+ }//if
+
+ filePtr.p->scanRunning = 0;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, filePtr.p->backupPtr);
+
+ BackupFragmentConf * conf = (BackupFragmentConf*)signal->getDataPtrSend();
+ conf->backupId = ptr.p->backupId;
+ conf->backupPtr = ptr.i;
+ conf->tableId = filePtr.p->tableId;
+ conf->fragmentNo = filePtr.p->fragmentNo;
+ conf->noOfRecords = op.noOfRecords;
+ conf->noOfBytes = op.noOfBytes;
+ sendSignal(ptr.p->masterRef, GSN_BACKUP_FRAGMENT_CONF, signal,
+ BackupFragmentConf::SignalLength, JBB);
+
+ ptr.p->slaveState.setState(STARTED);
+ return;
+}
+
+void
+Backup::checkScan(Signal* signal, BackupFilePtr filePtr)
+{
+ if(filePtr.p->errorCode != 0){
+ jam();
+ abortFileHook(signal, filePtr, false); // Scan not completed
+ return;
+ }//if
+
+ OperationRecord & op = filePtr.p->operation;
+ if(op.newScan()) {
+ jam();
+
+ ScanFragNextReq * req = (ScanFragNextReq *)signal->getDataPtrSend();
+ req->senderData = filePtr.i;
+ req->closeFlag = 0;
+ req->transId1 = 0;
+ req->transId2 = (BACKUP << 20) + (getOwnNodeId() << 8);
+ sendSignal(DBLQH_REF, GSN_SCAN_NEXTREQ, signal,
+ ScanFragNextReq::SignalLength, JBB);
+ return;
+ }//if
+
+ signal->theData[0] = BackupContinueB::BUFFER_FULL_SCAN;
+ signal->theData[1] = filePtr.i;
+ sendSignalWithDelay(BACKUP_REF, GSN_CONTINUEB, signal, 50, 2);
+}
+
+void
+Backup::execFSAPPENDREF(Signal* signal)
+{
+ jamEntry();
+
+ FsRef * ref = (FsRef *)signal->getDataPtr();
+
+ const Uint32 filePtrI = ref->userPointer;
+ const Uint32 errCode = ref->errorCode;
+
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, filePtrI);
+
+ filePtr.p->fileRunning = 0;
+ filePtr.p->errorCode = errCode;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, filePtr.p->backupPtr);
+
+ abortFile(signal, ptr, filePtr);
+}
+
+void
+Backup::execFSAPPENDCONF(Signal* signal)
+{
+ jamEntry();
+
+ CRASH_INSERTION((10018));
+
+ //FsConf * conf = (FsConf*)signal->getDataPtr();
+ const Uint32 filePtrI = signal->theData[0]; //conf->userPointer;
+ const Uint32 bytes = signal->theData[1]; //conf->bytes;
+
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, filePtrI);
+
+ if (ERROR_INSERTED(10029)) {
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, filePtr.p->backupPtr);
+ abortFile(signal, ptr, filePtr);
+ }//if
+
+ OperationRecord & op = filePtr.p->operation;
+
+ op.dataBuffer.updateReadPtr(bytes >> 2);
+
+ checkFile(signal, filePtr);
+}
+
+void
+Backup::checkFile(Signal* signal, BackupFilePtr filePtr)
+{
+
+#ifdef DEBUG_ABORT
+ // ndbout_c("---- check file filePtr.i = %u", filePtr.i);
+#endif
+
+ OperationRecord & op = filePtr.p->operation;
+
+ Uint32 * tmp, sz; bool eof;
+ if(op.dataBuffer.getReadPtr(&tmp, &sz, &eof)) {
+ jam();
+
+ if(filePtr.p->errorCode == 0) {
+ jam();
+ FsAppendReq * req = (FsAppendReq *)signal->getDataPtrSend();
+ req->filePointer = filePtr.p->filePointer;
+ req->userPointer = filePtr.i;
+ req->userReference = reference();
+ req->varIndex = 0;
+ req->offset = tmp - c_startOfPages;
+ req->size = sz;
+
+ sendSignal(NDBFS_REF, GSN_FSAPPENDREQ, signal,
+ FsAppendReq::SignalLength, JBA);
+ return;
+ } else {
+ jam();
+ if (filePtr.p->scanRunning == 1)
+ eof = false;
+ }//if
+ }//if
+
+ if(!eof) {
+ jam();
+ signal->theData[0] = BackupContinueB::BUFFER_UNDERFLOW;
+ signal->theData[1] = filePtr.i;
+ sendSignalWithDelay(BACKUP_REF, GSN_CONTINUEB, signal, 50, 2);
+ return;
+ }//if
+
+ ndbrequire(filePtr.p->fileDone == 1);
+
+ if(sz > 0 && filePtr.p->errorCode == 0) {
+ jam();
+ FsAppendReq * req = (FsAppendReq *)signal->getDataPtrSend();
+ req->filePointer = filePtr.p->filePointer;
+ req->userPointer = filePtr.i;
+ req->userReference = reference();
+ req->varIndex = 0;
+ req->offset = tmp - c_startOfPages;
+ req->size = sz; // Avrunda uppot
+
+ sendSignal(NDBFS_REF, GSN_FSAPPENDREQ, signal,
+ FsAppendReq::SignalLength, JBA);
+ return;
+ }//if
+
+ filePtr.p->fileRunning = 0;
+
+ FsCloseReq * req = (FsCloseReq *)signal->getDataPtrSend();
+ req->filePointer = filePtr.p->filePointer;
+ req->userPointer = filePtr.i;
+ req->userReference = reference();
+ req->fileFlag = 0;
+#ifdef DEBUG_ABORT
+ ndbout_c("***** FSCLOSEREQ filePtr.i = %u", filePtr.i);
+#endif
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, FsCloseReq::SignalLength, JBA);
+}
+
+void
+Backup::abortFile(Signal* signal, BackupRecordPtr ptr, BackupFilePtr filePtr)
+{
+ jam();
+
+ if(ptr.p->slaveState.getState() != ABORTING) {
+ /**
+ * Inform master of failure
+ */
+ jam();
+ ptr.p->slaveState.setState(ABORTING);
+ ptr.p->setErrorCode(AbortBackupOrd::FileOrScanError);
+ sendAbortBackupOrdSlave(signal, ptr, AbortBackupOrd::FileOrScanError);
+ return;
+ }//if
+
+
+ for(ptr.p->files.first(filePtr);
+ filePtr.i!=RNIL;
+ ptr.p->files.next(filePtr)){
+ jam();
+ filePtr.p->errorCode = 1;
+ }//for
+
+ closeFiles(signal, ptr);
+}
+
+void
+Backup::abortFileHook(Signal* signal, BackupFilePtr filePtr, bool scanComplete)
+{
+ jam();
+
+ if(!scanComplete) {
+ jam();
+
+ ScanFragNextReq * req = (ScanFragNextReq *)signal->getDataPtrSend();
+ req->senderData = filePtr.i;
+ req->closeFlag = 1;
+ req->transId1 = 0;
+ req->transId2 = (BACKUP << 20) + (getOwnNodeId() << 8);
+ sendSignal(DBLQH_REF, GSN_SCAN_NEXTREQ, signal,
+ ScanFragNextReq::SignalLength, JBB);
+ return;
+ }//if
+
+ filePtr.p->scanRunning = 0;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, filePtr.p->backupPtr);
+
+ filePtr.i = RNIL;
+ abortFile(signal, ptr, filePtr);
+}
+
+/****************************************************************************
+ *
+ * Slave functionallity: Perform logging
+ *
+ ****************************************************************************/
+Uint32
+Backup::calculate_frag_mask(Uint32 count)
+{
+ Uint32 mask = 1;
+ while (mask < count) mask <<= 1;
+ mask -= 1;
+ return mask;
+}
+
+void
+Backup::execBACKUP_TRIG_REQ(Signal* signal)
+{
+ /*
+ TUP asks if this trigger is to be fired on this node.
+ */
+ TriggerPtr trigPtr;
+ TablePtr tabPtr;
+ FragmentPtr fragPtr;
+ Uint32 trigger_id = signal->theData[0];
+ Uint32 frag_id = signal->theData[1];
+ Uint32 result;
+
+ jamEntry();
+ c_triggerPool.getPtr(trigPtr, trigger_id);
+ c_tablePool.getPtr(tabPtr, trigPtr.p->tab_ptr_i);
+ frag_id = frag_id & tabPtr.p->frag_mask;
+ /*
+ At the moment the fragment identity known by TUP is the
+ actual fragment id but with possibly an extra bit set.
+ This is due to that ACC splits the fragment. Thus fragment id 5 can
+ here be either 5 or 13. Thus masking with 2 ** n - 1 where number of
+ fragments <= 2 ** n will always provide a correct fragment id.
+ */
+ tabPtr.p->fragments.getPtr(fragPtr, frag_id);
+ if (fragPtr.p->node != getOwnNodeId()) {
+ jam();
+ result = ZFALSE;
+ } else {
+ jam();
+ result = ZTRUE;
+ }//if
+ signal->theData[0] = result;
+}
+
+void
+Backup::execTRIG_ATTRINFO(Signal* signal) {
+ jamEntry();
+
+ CRASH_INSERTION((10019));
+
+ TrigAttrInfo * trg = (TrigAttrInfo*)signal->getDataPtr();
+
+ TriggerPtr trigPtr;
+ c_triggerPool.getPtr(trigPtr, trg->getTriggerId());
+ ndbrequire(trigPtr.p->event != ILLEGAL_TRIGGER_ID); // Online...
+
+ if(trigPtr.p->errorCode != 0) {
+ jam();
+ return;
+ }//if
+
+ if(trg->getAttrInfoType() == TrigAttrInfo::BEFORE_VALUES) {
+ jam();
+ /**
+ * Backup is doing REDO logging and don't need before values
+ */
+ return;
+ }//if
+
+ BackupFormat::LogFile::LogEntry * logEntry = trigPtr.p->logEntry;
+ if(logEntry == 0) {
+ jam();
+ Uint32 * dst;
+ FsBuffer & buf = trigPtr.p->operation->dataBuffer;
+ ndbrequire(trigPtr.p->maxRecordSize <= buf.getMaxWrite());
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, trigPtr.p->backupPtr);
+ if(!buf.getWritePtr(&dst, trigPtr.p->maxRecordSize)) {
+ jam();
+ trigPtr.p->errorCode = AbortBackupOrd::LogBufferFull;
+ sendAbortBackupOrdSlave(signal, ptr, AbortBackupOrd::LogBufferFull);
+ return;
+ }//if
+ if(trigPtr.p->operation->noOfBytes > 123 && ERROR_INSERTED(10030)) {
+ jam();
+ trigPtr.p->errorCode = AbortBackupOrd::LogBufferFull;
+ sendAbortBackupOrdSlave(signal, ptr, AbortBackupOrd::LogBufferFull);
+ return;
+ }//if
+
+ logEntry = (BackupFormat::LogFile::LogEntry *)dst;
+ trigPtr.p->logEntry = logEntry;
+ logEntry->Length = 0;
+ logEntry->TableId = htonl(trigPtr.p->tableId);
+ logEntry->TriggerEvent = htonl(trigPtr.p->event);
+ } else {
+ ndbrequire(logEntry->TableId == htonl(trigPtr.p->tableId));
+ ndbrequire(logEntry->TriggerEvent == htonl(trigPtr.p->event));
+ }//if
+
+ const Uint32 pos = logEntry->Length;
+ const Uint32 dataLen = signal->length() - TrigAttrInfo::StaticLength;
+ memcpy(&logEntry->Data[pos], trg->getData(), dataLen << 2);
+
+ logEntry->Length = pos + dataLen;
+}
+
+void
+Backup::execFIRE_TRIG_ORD(Signal* signal)
+{
+ jamEntry();
+ FireTrigOrd* trg = (FireTrigOrd*)signal->getDataPtr();
+
+ const Uint32 gci = trg->getGCI();
+ const Uint32 trI = trg->getTriggerId();
+
+ TriggerPtr trigPtr;
+ c_triggerPool.getPtr(trigPtr, trI);
+
+ ndbrequire(trigPtr.p->event != ILLEGAL_TRIGGER_ID);
+
+ if(trigPtr.p->errorCode != 0) {
+ jam();
+ return;
+ }//if
+
+ ndbrequire(trigPtr.p->logEntry != 0);
+ Uint32 len = trigPtr.p->logEntry->Length;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, trigPtr.p->backupPtr);
+ if(gci != ptr.p->currGCP) {
+ jam();
+
+ trigPtr.p->logEntry->TriggerEvent = htonl(trigPtr.p->event | 0x10000);
+ trigPtr.p->logEntry->Data[len] = htonl(gci);
+ len ++;
+ ptr.p->currGCP = gci;
+ }//if
+
+ len += (sizeof(BackupFormat::LogFile::LogEntry) >> 2) - 2;
+ trigPtr.p->logEntry->Length = htonl(len);
+
+ ndbrequire(len + 1 <= trigPtr.p->operation->dataBuffer.getMaxWrite());
+ trigPtr.p->operation->dataBuffer.updateWritePtr(len + 1);
+ trigPtr.p->logEntry = 0;
+
+ trigPtr.p->operation->noOfBytes += (len + 1) << 2;
+ trigPtr.p->operation->noOfRecords += 1;
+}
+
+void
+Backup::sendAbortBackupOrdSlave(Signal* signal, BackupRecordPtr ptr,
+ Uint32 requestType)
+{
+ jam();
+ AbortBackupOrd *ord = (AbortBackupOrd*)signal->getDataPtrSend();
+ ord->backupId = ptr.p->backupId;
+ ord->backupPtr = ptr.i;
+ ord->requestType = requestType;
+ ord->senderData= ptr.i;
+ sendSignal(ptr.p->masterRef, GSN_ABORT_BACKUP_ORD, signal,
+ AbortBackupOrd::SignalLength, JBB);
+}
+
+void
+Backup::sendAbortBackupOrd(Signal* signal, BackupRecordPtr ptr,
+ Uint32 requestType)
+{
+ jam();
+ AbortBackupOrd *ord = (AbortBackupOrd*)signal->getDataPtrSend();
+ ord->backupId = ptr.p->backupId;
+ ord->backupPtr = ptr.i;
+ ord->requestType = requestType;
+ ord->senderData= ptr.i;
+ NodePtr node;
+ for(c_nodes.first(node); node.i != RNIL; c_nodes.next(node)) {
+ jam();
+ const Uint32 nodeId = node.p->nodeId;
+ if(node.p->alive && ptr.p->nodes.get(nodeId)) {
+ jam();
+ sendSignal(numberToRef(BACKUP, nodeId), GSN_ABORT_BACKUP_ORD, signal,
+ AbortBackupOrd::SignalLength, JBB);
+ }//if
+ }//for
+}
+
+/*****************************************************************************
+ *
+ * Slave functionallity: Stop backup
+ *
+ *****************************************************************************/
+void
+Backup::execSTOP_BACKUP_REQ(Signal* signal)
+{
+ jamEntry();
+ StopBackupReq * req = (StopBackupReq*)signal->getDataPtr();
+
+ CRASH_INSERTION((10020));
+
+ const Uint32 ptrI = req->backupPtr;
+ const Uint32 backupId = req->backupId;
+ const Uint32 startGCP = req->startGCP;
+ const Uint32 stopGCP = req->stopGCP;
+
+ /**
+ * At least one GCP must have passed
+ */
+ ndbrequire(stopGCP > startGCP);
+
+ /**
+ * Get backup record
+ */
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ ptr.p->slaveState.setState(STOPPING);
+ slaveAbortCheck(); // macro will do return if ABORTING
+
+ /**
+ * Insert footers
+ */
+ {
+ BackupFilePtr filePtr;
+ ptr.p->files.getPtr(filePtr, ptr.p->logFilePtr);
+ Uint32 * dst;
+ ndbrequire(filePtr.p->operation.dataBuffer.getWritePtr(&dst, 1));
+ * dst = 0;
+ filePtr.p->operation.dataBuffer.updateWritePtr(1);
+ }
+
+ {
+ BackupFilePtr filePtr;
+ ptr.p->files.getPtr(filePtr, ptr.p->ctlFilePtr);
+
+ const Uint32 gcpSz = sizeof(BackupFormat::CtlFile::GCPEntry) >> 2;
+
+ Uint32 * dst;
+ ndbrequire(filePtr.p->operation.dataBuffer.getWritePtr(&dst, gcpSz));
+
+ BackupFormat::CtlFile::GCPEntry * gcp =
+ (BackupFormat::CtlFile::GCPEntry*)dst;
+
+ gcp->SectionType = htonl(BackupFormat::GCP_ENTRY);
+ gcp->SectionLength = htonl(gcpSz);
+ gcp->StartGCP = htonl(startGCP);
+ gcp->StopGCP = htonl(stopGCP - 1);
+ filePtr.p->operation.dataBuffer.updateWritePtr(gcpSz);
+ }
+
+ closeFiles(signal, ptr);
+}
+
+void
+Backup::closeFiles(Signal* sig, BackupRecordPtr ptr)
+{
+ if (ptr.p->closingFiles) {
+ jam();
+ return;
+ }
+ ptr.p->closingFiles = true;
+
+ /**
+ * Close all files
+ */
+ BackupFilePtr filePtr;
+ int openCount = 0;
+ for(ptr.p->files.first(filePtr); filePtr.i!=RNIL; ptr.p->files.next(filePtr))
+ {
+ if(filePtr.p->fileOpened == 0) {
+ jam();
+ continue;
+ }
+
+ jam();
+ openCount++;
+
+ if(filePtr.p->fileDone == 1){
+ jam();
+ continue;
+ }//if
+
+ filePtr.p->fileDone = 1;
+
+ if(filePtr.p->fileRunning == 1){
+ jam();
+#ifdef DEBUG_ABORT
+ ndbout_c("Close files fileRunning == 1, filePtr.i=%u", filePtr.i);
+#endif
+ filePtr.p->operation.dataBuffer.eof();
+ } else {
+ jam();
+
+ FsCloseReq * req = (FsCloseReq *)sig->getDataPtrSend();
+ req->filePointer = filePtr.p->filePointer;
+ req->userPointer = filePtr.i;
+ req->userReference = reference();
+ req->fileFlag = 0;
+#ifdef DEBUG_ABORT
+ ndbout_c("***** FSCLOSEREQ filePtr.i = %u", filePtr.i);
+#endif
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, sig,
+ FsCloseReq::SignalLength, JBA);
+ }//if
+ }//for
+
+ if(openCount == 0){
+ jam();
+ closeFilesDone(sig, ptr);
+ }//if
+}
+
+void
+Backup::execFSCLOSEREF(Signal* signal)
+{
+ jamEntry();
+
+ FsRef * ref = (FsRef*)signal->getDataPtr();
+ const Uint32 filePtrI = ref->userPointer;
+
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, filePtrI);
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, filePtr.p->backupPtr);
+
+ /**
+ * This should only happen during abort of backup
+ */
+ ndbrequire(ptr.p->slaveState.getState() == ABORTING);
+
+ filePtr.p->fileOpened = 1;
+ FsConf * conf = (FsConf*)signal->getDataPtr();
+ conf->userPointer = filePtrI;
+
+ execFSCLOSECONF(signal);
+}
+
+void
+Backup::execFSCLOSECONF(Signal* signal)
+{
+ jamEntry();
+
+ FsConf * conf = (FsConf*)signal->getDataPtr();
+ const Uint32 filePtrI = conf->userPointer;
+
+ BackupFilePtr filePtr;
+ c_backupFilePool.getPtr(filePtr, filePtrI);
+
+#ifdef DEBUG_ABORT
+ ndbout_c("***** FSCLOSECONF filePtrI = %u", filePtrI);
+#endif
+
+ ndbrequire(filePtr.p->fileDone == 1);
+ ndbrequire(filePtr.p->fileOpened == 1);
+ ndbrequire(filePtr.p->fileRunning == 0);
+ ndbrequire(filePtr.p->scanRunning == 0);
+
+ filePtr.p->fileOpened = 0;
+
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, filePtr.p->backupPtr);
+ for(ptr.p->files.first(filePtr); filePtr.i!=RNIL;ptr.p->files.next(filePtr))
+ {
+ jam();
+ if(filePtr.p->fileOpened == 1) {
+ jam();
+#ifdef DEBUG_ABORT
+ ndbout_c("waiting for more FSCLOSECONF's filePtr.i = %u", filePtr.i);
+#endif
+ return; // we will be getting more FSCLOSECONF's
+ }//if
+ }//for
+ closeFilesDone(signal, ptr);
+}
+
+void
+Backup::closeFilesDone(Signal* signal, BackupRecordPtr ptr)
+{
+ jam();
+
+ if(ptr.p->slaveState.getState() == STOPPING) {
+ jam();
+ BackupFilePtr filePtr;
+ ptr.p->files.getPtr(filePtr, ptr.p->logFilePtr);
+
+ StopBackupConf* conf = (StopBackupConf*)signal->getDataPtrSend();
+ conf->backupId = ptr.p->backupId;
+ conf->backupPtr = ptr.i;
+ conf->noOfLogBytes = filePtr.p->operation.noOfBytes;
+ conf->noOfLogRecords = filePtr.p->operation.noOfRecords;
+ sendSignal(ptr.p->masterRef, GSN_STOP_BACKUP_CONF, signal,
+ StopBackupConf::SignalLength, JBB);
+
+ ptr.p->slaveState.setState(CLEANING);
+ return;
+ }//if
+
+ ndbrequire(ptr.p->slaveState.getState() == ABORTING);
+ removeBackup(signal, ptr);
+}
+
+/*****************************************************************************
+ *
+ * Slave functionallity: Abort backup
+ *
+ *****************************************************************************/
+void
+Backup::removeBackup(Signal* signal, BackupRecordPtr ptr)
+{
+ jam();
+
+ FsRemoveReq * req = (FsRemoveReq *)signal->getDataPtrSend();
+ req->userReference = reference();
+ req->userPointer = ptr.i;
+ req->directory = 1;
+ req->ownDirectory = 1;
+ FsOpenReq::setVersion(req->fileNumber, 2);
+ FsOpenReq::setSuffix(req->fileNumber, FsOpenReq::S_CTL);
+ FsOpenReq::v2_setSequence(req->fileNumber, ptr.p->backupId);
+ FsOpenReq::v2_setNodeId(req->fileNumber, getOwnNodeId());
+ sendSignal(NDBFS_REF, GSN_FSREMOVEREQ, signal,
+ FsRemoveReq::SignalLength, JBA);
+}
+
+void
+Backup::execFSREMOVEREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(0);
+}
+
+void
+Backup::execFSREMOVECONF(Signal* signal){
+ jamEntry();
+
+ FsConf * conf = (FsConf*)signal->getDataPtr();
+ const Uint32 ptrI = conf->userPointer;
+
+ /**
+ * Get backup record
+ */
+ BackupRecordPtr ptr;
+ c_backupPool.getPtr(ptr, ptrI);
+
+ ndbrequire(ptr.p->slaveState.getState() == ABORTING);
+ if (ptr.p->masterRef == reference()) {
+ if (ptr.p->masterData.state.getAbortState() == DEFINING) {
+ jam();
+ sendBackupRef(signal, ptr, ptr.p->errorCode);
+ return;
+ } else {
+ jam();
+ }//if
+ }//if
+ cleanupSlaveResources(ptr);
+}
+
+/*****************************************************************************
+ *
+ * Slave functionallity: Abort backup
+ *
+ *****************************************************************************/
+void
+Backup::execABORT_BACKUP_ORD(Signal* signal)
+{
+ jamEntry();
+ AbortBackupOrd* ord = (AbortBackupOrd*)signal->getDataPtr();
+
+ const Uint32 backupId = ord->backupId;
+ const AbortBackupOrd::RequestType requestType =
+ (AbortBackupOrd::RequestType)ord->requestType;
+ const Uint32 senderData = ord->senderData;
+
+#ifdef DEBUG_ABORT
+ ndbout_c("******** ABORT_BACKUP_ORD ********* nodeId = %u",
+ refToNode(signal->getSendersBlockRef()));
+ ndbout_c("backupId = %u, requestType = %u, senderData = %u, ",
+ backupId, requestType, senderData);
+ dumpUsedResources();
+#endif
+
+ BackupRecordPtr ptr;
+ if(requestType == AbortBackupOrd::ClientAbort) {
+ if (getOwnNodeId() != getMasterNodeId()) {
+ jam();
+ // forward to master
+#ifdef DEBUG_ABORT
+ ndbout_c("---- Forward to master nodeId = %u", getMasterNodeId());
+#endif
+ sendSignal(calcBackupBlockRef(getMasterNodeId()), GSN_ABORT_BACKUP_ORD,
+ signal, AbortBackupOrd::SignalLength, JBB);
+ return;
+ }
+ jam();
+ for(c_backups.first(ptr); ptr.i != RNIL; c_backups.next(ptr)) {
+ jam();
+ if(ptr.p->backupId == backupId && ptr.p->clientData == senderData) {
+ jam();
+ break;
+ }//if
+ }//for
+ if(ptr.i == RNIL) {
+ jam();
+ return;
+ }//if
+ } else {
+ if (c_backupPool.findId(senderData)) {
+ jam();
+ c_backupPool.getPtr(ptr, senderData);
+ } else { // TODO might be abort sent to not master,
+ // or master aborting too early
+ jam();
+#ifdef DEBUG_ABORT
+ ndbout_c("Backup: abort request type=%u on id=%u,%u not found",
+ requestType, backupId, senderData);
+#endif
+ return;
+ }
+ }//if
+
+ const bool isCoordinator = (ptr.p->masterRef == reference());
+
+ bool ok = false;
+ switch(requestType){
+
+ /**
+ * Requests sent to master
+ */
+
+ case AbortBackupOrd::ClientAbort:
+ jam();
+ // fall through
+ case AbortBackupOrd::LogBufferFull:
+ jam();
+ // fall through
+ case AbortBackupOrd::FileOrScanError:
+ jam();
+ if(ptr.p->masterData.state.getState() == ABORTING) {
+#ifdef DEBUG_ABORT
+ ndbout_c("---- Already aborting");
+#endif
+ jam();
+ return;
+ }
+ ptr.p->setErrorCode(requestType);
+ ndbrequire(isCoordinator); // Sent from slave to coordinator
+ masterAbort(signal, ptr, false);
+ return;
+
+ /**
+ * Info sent to slave
+ */
+
+ case AbortBackupOrd::OkToClean:
+ jam();
+ cleanupMasterResources(ptr);
+ return;
+
+ /**
+ * Requests sent to slave
+ */
+
+ case AbortBackupOrd::BackupComplete:
+ jam();
+ if (ptr.p->slaveState.getState() == CLEANING) { // TODO what if state is
+ // not CLEANING?
+ jam();
+ cleanupSlaveResources(ptr);
+ }//if
+ return;
+ break;
+ case AbortBackupOrd::BackupFailureDueToNodeFail:
+ jam();
+ ok = true;
+ if (ptr.p->errorCode != 0)
+ ptr.p->setErrorCode(requestType);
+ break;
+ case AbortBackupOrd::BackupFailure:
+ jam();
+ ok = true;
+ break;
+ }
+ ndbrequire(ok);
+
+ /**
+ * Slave abort
+ */
+ slaveAbort(signal, ptr);
+}
+
+void
+Backup::slaveAbort(Signal* signal, BackupRecordPtr ptr)
+{
+ if(ptr.p->slaveState.getState() == ABORTING) {
+#ifdef DEBUG_ABORT
+ ndbout_c("---- Slave already aborting");
+#endif
+ jam();
+ return;
+ }
+#ifdef DEBUG_ABORT
+ ndbout_c("************* slaveAbort");
+#endif
+
+ State slaveState = ptr.p->slaveState.getState();
+ ptr.p->slaveState.setState(ABORTING);
+ switch(slaveState) {
+ case DEFINING:
+ jam();
+ return;
+//------------------------------------------
+// Will watch for the abort at various places
+// in the defining phase.
+//------------------------------------------
+ case ABORTING:
+ jam();
+ //Fall through
+ case DEFINED:
+ jam();
+ //Fall through
+ case STOPPING:
+ jam();
+ closeFiles(signal, ptr);
+ return;
+ case STARTED:
+ jam();
+ //Fall through
+ case SCANNING:
+ jam();
+ BackupFilePtr filePtr;
+ filePtr.i = RNIL;
+ abortFile(signal, ptr, filePtr);
+ return;
+ case CLEANING:
+ jam();
+ cleanupSlaveResources(ptr);
+ return;
+ case INITIAL:
+ jam();
+ ndbrequire(false);
+ return;
+ }
+}
+
+void
+Backup::dumpUsedResources()
+{
+ jam();
+ BackupRecordPtr ptr;
+
+ for(c_backups.first(ptr); ptr.i != RNIL; c_backups.next(ptr)) {
+ ndbout_c("Backup id=%u, slaveState.getState = %u, errorCode=%u",
+ ptr.p->backupId,
+ ptr.p->slaveState.getState(),
+ ptr.p->errorCode);
+
+ TablePtr tabPtr;
+ for(ptr.p->tables.first(tabPtr);
+ tabPtr.i != RNIL;
+ ptr.p->tables.next(tabPtr)) {
+ jam();
+ for(Uint32 j = 0; j<3; j++) {
+ jam();
+ TriggerPtr trigPtr;
+ if(tabPtr.p->triggerAllocated[j]) {
+ jam();
+ c_triggerPool.getPtr(trigPtr, tabPtr.p->triggerIds[j]);
+ ndbout_c("Allocated[%u] Triggerid = %u, event = %u",
+ j,
+ tabPtr.p->triggerIds[j],
+ trigPtr.p->event);
+ }//if
+ }//for
+ }//for
+
+ BackupFilePtr filePtr;
+ for(ptr.p->files.first(filePtr);
+ filePtr.i != RNIL;
+ ptr.p->files.next(filePtr)) {
+ jam();
+ ndbout_c("filePtr.i = %u, filePtr.p->fileOpened=%u fileRunning=%u "
+ "scanRunning=%u",
+ filePtr.i,
+ filePtr.p->fileOpened,
+ filePtr.p->fileRunning,
+ filePtr.p->scanRunning);
+ }//for
+ }
+}
+
+void
+Backup::cleanupMasterResources(BackupRecordPtr ptr)
+{
+#ifdef DEBUG_ABORT
+ ndbout_c("******** Cleanup Master Resources *********");
+ ndbout_c("backupId = %u, errorCode = %u", ptr.p->backupId, ptr.p->errorCode);
+#endif
+
+ TablePtr tabPtr;
+ for(ptr.p->tables.first(tabPtr); tabPtr.i != RNIL;ptr.p->tables.next(tabPtr))
+ {
+ jam();
+ tabPtr.p->attributes.release();
+ tabPtr.p->fragments.release();
+ for(Uint32 j = 0; j<3; j++) {
+ jam();
+ TriggerPtr trigPtr;
+ if(tabPtr.p->triggerAllocated[j]) {
+ jam();
+ c_triggerPool.getPtr(trigPtr, tabPtr.p->triggerIds[j]);
+ trigPtr.p->event = ILLEGAL_TRIGGER_ID;
+ tabPtr.p->triggerAllocated[j] = false;
+ }//if
+ tabPtr.p->triggerIds[j] = ILLEGAL_TRIGGER_ID;
+ }//for
+ }//for
+ ptr.p->tables.release();
+ ptr.p->triggers.release();
+ ptr.p->okToCleanMaster = true;
+
+ cleanupFinalResources(ptr);
+}
+
+void
+Backup::cleanupSlaveResources(BackupRecordPtr ptr)
+{
+#ifdef DEBUG_ABORT
+ ndbout_c("******** Clean Up Slave Resources*********");
+ ndbout_c("backupId = %u, errorCode = %u", ptr.p->backupId, ptr.p->errorCode);
+#endif
+
+ BackupFilePtr filePtr;
+ for(ptr.p->files.first(filePtr);
+ filePtr.i != RNIL;
+ ptr.p->files.next(filePtr)) {
+ jam();
+ ndbrequire(filePtr.p->fileOpened == 0);
+ ndbrequire(filePtr.p->fileRunning == 0);
+ ndbrequire(filePtr.p->scanRunning == 0);
+ filePtr.p->pages.release();
+ }//for
+ ptr.p->files.release();
+
+ cleanupFinalResources(ptr);
+}
+
+void
+Backup::cleanupFinalResources(BackupRecordPtr ptr)
+{
+#ifdef DEBUG_ABORT
+ ndbout_c("******** Clean Up Final Resources*********");
+ ndbout_c("backupId = %u, errorCode = %u", ptr.p->backupId, ptr.p->errorCode);
+#endif
+
+ // if (!ptr.p->tables.empty() || !ptr.p->files.empty()) {
+ if (!ptr.p->okToCleanMaster || !ptr.p->files.empty()) {
+ jam();
+#ifdef DEBUG_ABORT
+ ndbout_c("******** Waiting to do final cleanup");
+#endif
+ return;
+ }
+ ptr.p->pages.release();
+ ptr.p->masterData.state.setState(INITIAL);
+ ptr.p->slaveState.setState(INITIAL);
+ ptr.p->backupId = 0;
+
+ ptr.p->closingFiles = false;
+ ptr.p->okToCleanMaster = true;
+
+ c_backups.release(ptr);
+ // ndbrequire(false);
+}
diff --git a/ndb/src/kernel/blocks/backup/Backup.hpp b/ndb/src/kernel/blocks/backup/Backup.hpp
new file mode 100644
index 00000000000..77669e759d3
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/Backup.hpp
@@ -0,0 +1,728 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BACKUP_H
+#define BACKUP_H
+
+#include <ndb_limits.h>
+#include <SimulatedBlock.hpp>
+
+#include "FsBuffer.hpp"
+#include "BackupFormat.hpp"
+
+#include <NodeBitmask.hpp>
+#include <SimpleProperties.hpp>
+
+#include <SLList.hpp>
+#include <ArrayList.hpp>
+#include <SignalCounter.hpp>
+#include <blocks/mutexes.hpp>
+
+#include <NdbTCP.h>
+
+/**
+ * Backup - This block manages database backup and restore
+ */
+class Backup : public SimulatedBlock
+{
+public:
+ Backup(const Configuration & conf);
+ virtual ~Backup();
+ BLOCK_DEFINES(Backup);
+
+protected:
+
+ void execSTTOR(Signal* signal);
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execREAD_NODESCONF(Signal* signal);
+ void execNODE_FAILREP(Signal* signal);
+ void execINCL_NODEREQ(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+
+ /**
+ * Testing
+ */
+ void execBACKUP_REF(Signal* signal);
+ void execBACKUP_CONF(Signal* signal);
+ void execBACKUP_ABORT_REP(Signal* signal);
+ void execBACKUP_COMPLETE_REP(Signal* signal);
+
+ /**
+ * Signals sent from master
+ */
+ void execDEFINE_BACKUP_REQ(Signal* signal);
+ void execBACKUP_DATA(Signal* signal);
+ void execSTART_BACKUP_REQ(Signal* signal);
+ void execBACKUP_FRAGMENT_REQ(Signal* signal);
+ void execSTOP_BACKUP_REQ(Signal* signal);
+ void execBACKUP_STATUS_REQ(Signal* signal);
+ void execABORT_BACKUP_ORD(Signal* signal);
+
+ /**
+ * The actual scan
+ */
+ void execSCAN_HBREP(Signal* signal);
+ void execTRANSID_AI(Signal* signal);
+ void execKEYINFO20(Signal* signal);
+ void execSCAN_FRAGREF(Signal* signal);
+ void execSCAN_FRAGCONF(Signal* signal);
+
+ /**
+ * Trigger logging
+ */
+ void execBACKUP_TRIG_REQ(Signal* signal);
+ void execTRIG_ATTRINFO(Signal* signal);
+ void execFIRE_TRIG_ORD(Signal* signal);
+
+ /**
+ * DICT signals
+ */
+ void execLIST_TABLES_CONF(Signal* signal);
+ void execGET_TABINFOREF(Signal* signal);
+ void execGET_TABINFO_CONF(Signal* signal);
+ void execCREATE_TRIG_REF(Signal* signal);
+ void execCREATE_TRIG_CONF(Signal* signal);
+ void execALTER_TRIG_REF(Signal* signal);
+ void execALTER_TRIG_CONF(Signal* signal);
+ void execDROP_TRIG_REF(Signal* signal);
+ void execDROP_TRIG_CONF(Signal* signal);
+
+ /**
+ * DIH signals
+ */
+ void execDI_FCOUNTCONF(Signal* signal);
+ void execDIGETPRIMCONF(Signal* signal);
+
+ /**
+ * FS signals
+ */
+ void execFSOPENREF(Signal* signal);
+ void execFSOPENCONF(Signal* signal);
+
+ void execFSCLOSEREF(Signal* signal);
+ void execFSCLOSECONF(Signal* signal);
+
+ void execFSAPPENDREF(Signal* signal);
+ void execFSAPPENDCONF(Signal* signal);
+
+ void execFSREMOVEREF(Signal* signal);
+ void execFSREMOVECONF(Signal* signal);
+
+ /**
+ * Master functinallity
+ */
+ void execBACKUP_REQ(Signal* signal);
+ void execABORT_BACKUP_REQ(Signal* signal);
+
+ void execDEFINE_BACKUP_REF(Signal* signal);
+ void execDEFINE_BACKUP_CONF(Signal* signal);
+
+ void execSTART_BACKUP_REF(Signal* signal);
+ void execSTART_BACKUP_CONF(Signal* signal);
+
+ void execBACKUP_FRAGMENT_REF(Signal* signal);
+ void execBACKUP_FRAGMENT_CONF(Signal* signal);
+
+ void execSTOP_BACKUP_REF(Signal* signal);
+ void execSTOP_BACKUP_CONF(Signal* signal);
+
+ void execBACKUP_STATUS_CONF(Signal* signal);
+
+ void execUTIL_SEQUENCE_REF(Signal* signal);
+ void execUTIL_SEQUENCE_CONF(Signal* signal);
+
+ void execWAIT_GCP_REF(Signal* signal);
+ void execWAIT_GCP_CONF(Signal* signal);
+
+
+private:
+ void defineBackupMutex_locked(Signal* signal, Uint32 ptrI,Uint32 retVal);
+ void dictCommitTableMutex_locked(Signal* signal, Uint32 ptrI,Uint32 retVal);
+
+public:
+ struct Node {
+ Uint32 nodeId;
+ Uint32 alive;
+ Uint32 nextList;
+ union { Uint32 prevList; Uint32 nextPool; };
+ };
+ typedef Ptr<Node> NodePtr;
+
+#define BACKUP_WORDS_PER_PAGE 8191
+ struct Page32 {
+ Uint32 data[BACKUP_WORDS_PER_PAGE];
+ Uint32 nextPool;
+ };
+ typedef Ptr<Page32> Page32Ptr;
+
+ struct Attribute {
+ struct Data {
+ Uint8 nullable;
+ Uint8 fixed;
+ Uint8 key;
+ Uint8 unused;
+ Uint32 sz32; // No of 32 bit words
+ Uint32 offset; // Relative DataFixedAttributes/DataFixedKeys
+ Uint32 offsetNull; // In NullBitmask
+ } data;
+ Uint32 nextPool;
+ };
+ typedef Ptr<Attribute> AttributePtr;
+
+ struct Fragment {
+ Uint32 tableId;
+ Uint32 node;
+ Uint16 scanned; // 0 = not scanned x = scanned by node x
+ Uint16 scanning; // 0 = not scanning x = scanning on node x
+ Uint32 nextPool;
+ };
+ typedef Ptr<Fragment> FragmentPtr;
+
+ struct Table {
+ Table(ArrayPool<Attribute> &, ArrayPool<Fragment> &);
+
+ Uint32 tableId;
+ Uint32 schemaVersion;
+ Uint32 frag_mask;
+ Uint32 tableType;
+ Uint32 noOfNull;
+ Uint32 noOfKeys;
+ Uint32 noOfAttributes;
+ Uint32 noOfVariable;
+ Uint32 sz_FixedKeys;
+ Uint32 sz_FixedAttributes;
+ Uint32 variableKeyId;
+ Uint32 triggerIds[3];
+ bool triggerAllocated[3];
+
+ Array<Attribute> attributes;
+ Array<Fragment> fragments;
+
+ Uint32 nextList;
+ union { Uint32 nextPool; Uint32 prevList; };
+ };
+ typedef Ptr<Table> TablePtr;
+
+ struct OperationRecord {
+ public:
+ OperationRecord(Backup & b) : backup(b) {}
+
+ /**
+ * Once per table
+ */
+ void init(const TablePtr & ptr);
+ inline Uint32 getFixedKeySize() const { return sz_FixedKeys; }
+
+ /**
+ * Once per fragment
+ */
+ bool newFragment(Uint32 tableId, Uint32 fragNo);
+ bool fragComplete(Uint32 tableId, Uint32 fragNo);
+
+ /**
+ * Once per scan frag (next) req/conf
+ */
+ bool newScan();
+ bool scanConf(Uint32 noOfOps, Uint32 opLen[]);
+
+ /**
+ * Per record
+ */
+ void newRecord(Uint32 * base);
+ bool finished();
+
+ /**
+ * Per attribute
+ */
+ Uint32 * newKey();
+ void nullAttribute(Uint32 nullOffset);
+ Uint32 * newNullable(Uint32 attrId, Uint32 sz);
+ Uint32 * newAttrib(Uint32 offset, Uint32 sz);
+ Uint32 * newVariable(Uint32 id, Uint32 sz);
+ Uint32 * newVariableKey(Uint32 sz);
+
+ private:
+ Uint32* base;
+ Uint32* dst_Length;
+ Uint32* dst_Bitmask;
+ Uint32* dst_FixedKeys;
+ Uint32* dst_FixedAttribs;
+ BackupFormat::DataFile::VariableData* dst_VariableData;
+
+ Uint32 noOfAttributes; // No of Attributes
+ Uint32 variableKeyId; // Id of variable key
+ Uint32 attrLeft; // No of attributes left
+
+ Uint32 opNoDone;
+ Uint32 opNoConf;
+ Uint32 attrLen[16];
+
+ public:
+ Uint32* dst;
+ Uint32 attrSzLeft; // No of words missing for current attribute
+ Uint32 attrSzTotal; // No of AI words received
+ Uint32 tablePtr; // Ptr.i to current table
+
+ FsBuffer dataBuffer;
+ Uint32 noOfRecords;
+ Uint32 noOfBytes;
+ Uint32 maxRecordSize;
+
+ private:
+ Uint32* scanStart;
+ Uint32* scanStop;
+
+ /**
+ * sizes of part
+ */
+ Uint32 sz_Bitmask;
+ Uint32 sz_FixedKeys;
+ Uint32 sz_FixedAttribs;
+
+ public:
+ union { Uint32 nextPool; Uint32 nextList; };
+ Uint32 prevList;
+ private:
+
+ Backup & backup;
+ BlockNumber number() const { return backup.number(); }
+ void progError(int line, int cause, const char * extra) {
+ backup.progError(line, cause, extra);
+ }
+ };
+ friend struct OperationRecord;
+
+ struct TriggerRecord {
+ TriggerRecord() { event = ~0;}
+ OperationRecord * operation;
+ BackupFormat::LogFile::LogEntry * logEntry;
+ Uint32 maxRecordSize;
+ Uint32 tableId;
+ Uint32 tab_ptr_i;
+ Uint32 event;
+ Uint32 backupPtr;
+ Uint32 errorCode;
+ union { Uint32 nextPool; Uint32 nextList; };
+ };
+ typedef Ptr<TriggerRecord> TriggerPtr;
+
+ /**
+ * BackupFile - At least 3 per backup
+ */
+ struct BackupFile {
+ BackupFile(Backup & backup, ArrayPool<Page32> & pp)
+ : operation(backup), pages(pp) {}
+
+ Uint32 backupPtr; // Pointer to backup record
+ Uint32 tableId;
+ Uint32 fragmentNo;
+ Uint32 filePointer;
+ Uint32 errorCode;
+ BackupFormat::FileType fileType;
+ OperationRecord operation;
+
+ Array<Page32> pages;
+ Uint32 nextList;
+ union { Uint32 prevList; Uint32 nextPool; };
+
+ Uint8 fileOpened;
+ Uint8 fileRunning;
+ Uint8 fileDone;
+ Uint8 scanRunning;
+ };
+ typedef Ptr<BackupFile> BackupFilePtr;
+
+
+ /**
+ * State for BackupRecord
+ */
+ enum State {
+ INITIAL,
+ DEFINING, // Defining backup content and parameters
+ DEFINED, // DEFINE_BACKUP_CONF sent in slave, received all in master
+ STARTED, // Creating triggers
+ SCANNING, // Scanning fragments
+ STOPPING, // Closing files
+ CLEANING, // Cleaning resources
+ ABORTING // Aborting backup
+ };
+
+ static const Uint32 validSlaveTransitionsCount;
+ static const Uint32 validMasterTransitionsCount;
+ static const State validSlaveTransitions[];
+ static const State validMasterTransitions[];
+
+ class CompoundState {
+ public:
+ CompoundState(Backup & b,
+ const State valid[],
+ Uint32 count, Uint32 _id)
+ : backup(b)
+ , validTransitions(valid),
+ noOfValidTransitions(count), id(_id)
+ {
+ state = INITIAL;
+ abortState = state;
+ }
+
+ void setState(State s);
+ State getState() const { return state;}
+ State getAbortState() const { return abortState;}
+
+ void forceState(State s);
+
+ BlockNumber number() const { return backup.number(); }
+ void progError(int line, int cause, const char * extra) {
+ backup.progError(line, cause, extra);
+ }
+ private:
+ Backup & backup;
+ State state;
+ State abortState; /**
+ When state == ABORTING, this contains the state
+ when the abort started
+ */
+ const State * validTransitions;
+ const Uint32 noOfValidTransitions;
+ const Uint32 id;
+ };
+ friend class CompoundState;
+
+ /**
+ * Backup record
+ *
+ * One record per backup
+ */
+ struct BackupRecord {
+ BackupRecord(Backup& b, ArrayPool<Page32> & pp,
+ ArrayPool<Table> & tp,
+ ArrayPool<BackupFile> & bp,
+ ArrayPool<TriggerRecord> & trp)
+ : slaveState(b, validSlaveTransitions, validSlaveTransitionsCount,1)
+ , tables(tp), triggers(trp), files(bp), pages(pp)
+ , masterData(b, validMasterTransitions, validMasterTransitionsCount)
+ , backup(b)
+ {
+ closingFiles = false;
+ okToCleanMaster = true;
+ }
+
+ CompoundState slaveState;
+
+ Uint32 clientRef;
+ Uint32 clientData;
+ Uint32 backupId;
+ Uint32 backupKey[2];
+ Uint32 masterRef;
+ Uint32 errorCode;
+ NdbNodeBitmask nodes;
+
+ bool okToCleanMaster;
+ bool closingFiles;
+
+ Uint64 noOfBytes;
+ Uint64 noOfRecords;
+ Uint64 noOfLogBytes;
+ Uint64 noOfLogRecords;
+
+ Uint32 startGCP;
+ Uint32 currGCP;
+ Uint32 stopGCP;
+ SLList<Table> tables;
+ SLList<TriggerRecord> triggers;
+
+ SLList<BackupFile> files;
+ Uint32 ctlFilePtr; // Ptr.i to ctl-file
+ Uint32 logFilePtr; // Ptr.i to log-file
+ Uint32 dataFilePtr; // Ptr.i to first data-file
+
+ Uint32 backupDataLen; // Used for (un)packing backup request
+ Array<Page32> pages; // Used for (un)packing backup request
+ SimpleProperties props;// Used for (un)packing backup request
+
+ struct MasterData {
+ MasterData(Backup & b, const State valid[], Uint32 count)
+ : state(b, valid, count, 0)
+ {
+ }
+ MutexHandle2<BACKUP_DEFINE_MUTEX> m_defineBackupMutex;
+ MutexHandle2<DICT_COMMIT_TABLE_MUTEX> m_dictCommitTableMutex;
+
+ Uint32 gsn;
+ CompoundState state;
+ SignalCounter sendCounter;
+ Uint32 errorCode;
+ struct {
+ Uint32 tableId;
+ } createTrig;
+ struct {
+ Uint32 tableId;
+ } dropTrig;
+ struct {
+ Uint32 tableId;
+ } alterTrig;
+ union {
+ struct {
+ Uint32 startBackup;
+ } waitGCP;
+ struct {
+ Uint32 signalNo;
+ Uint32 noOfSignals;
+ Uint32 tablePtr;
+ } startBackup;
+ struct {
+ Uint32 dummy;
+ } stopBackup;
+ };
+ } masterData;
+
+ Uint32 nextList;
+ union { Uint32 prevList; Uint32 nextPool; };
+
+ void setErrorCode(Uint32 errCode){
+ if(errorCode == 0)
+ errorCode = errCode;
+ }
+
+ bool checkError() const {
+ return errorCode != 0;
+ }
+
+ Backup & backup;
+ BlockNumber number() const { return backup.number(); }
+ void progError(int line, int cause, const char * extra) {
+ backup.progError(line, cause, extra);
+ }
+ };
+ friend struct BackupRecord;
+ typedef Ptr<BackupRecord> BackupRecordPtr;
+
+ struct Config {
+ Uint32 m_dataBufferSize;
+ Uint32 m_logBufferSize;
+ Uint32 m_minWriteSize;
+ Uint32 m_maxWriteSize;
+ };
+
+ /**
+ * Variables
+ */
+ Uint32 * c_startOfPages;
+ NodeId c_masterNodeId;
+ SLList<Node> c_nodes;
+ NdbNodeBitmask c_aliveNodes;
+ DLList<BackupRecord> c_backups;
+ Config c_defaults;
+
+ STATIC_CONST(NO_OF_PAGES_META_FILE = 2);
+
+ /**
+ * Pools
+ */
+ ArrayPool<Table> c_tablePool;
+ ArrayPool<Attribute> c_attributePool;
+ ArrayPool<BackupRecord> c_backupPool;
+ ArrayPool<BackupFile> c_backupFilePool;
+ ArrayPool<Page32> c_pagePool;
+ ArrayPool<Fragment> c_fragmentPool;
+ ArrayPool<Node> c_nodePool;
+ ArrayPool<TriggerRecord> c_triggerPool;
+
+ Uint32 calculate_frag_mask(Uint32);
+
+ void checkFile(Signal*, BackupFilePtr);
+ void checkScan(Signal*, BackupFilePtr);
+ void fragmentCompleted(Signal*, BackupFilePtr);
+
+ void backupAllData(Signal* signal, BackupRecordPtr);
+
+ void getFragmentInfo(Signal*, BackupRecordPtr, TablePtr, Uint32 fragNo);
+ void getFragmentInfoDone(Signal*, BackupRecordPtr);
+
+ void openFiles(Signal* signal, BackupRecordPtr ptr);
+ void openFilesReply(Signal*, BackupRecordPtr ptr, BackupFilePtr);
+ void closeFiles(Signal*, BackupRecordPtr ptr);
+ void closeFilesDone(Signal*, BackupRecordPtr ptr);
+
+ void sendDefineBackupReq(Signal *signal, BackupRecordPtr ptr);
+
+ void defineBackupReply(Signal* signal, BackupRecordPtr ptr, Uint32 nodeId);
+ void createTrigReply(Signal* signal, BackupRecordPtr ptr);
+ void alterTrigReply(Signal* signal, BackupRecordPtr ptr);
+ void startBackupReply(Signal* signal, BackupRecordPtr ptr, Uint32, Uint32);
+ void stopBackupReply(Signal* signal, BackupRecordPtr ptr, Uint32 nodeId);
+
+ void defineBackupRef(Signal*, BackupRecordPtr, Uint32 errCode = 0);
+
+ void nextFragment(Signal*, BackupRecordPtr);
+
+ void sendCreateTrig(Signal*, BackupRecordPtr ptr, TablePtr tabPtr);
+ void createAttributeMask(TablePtr tab, Bitmask<MAXNROFATTRIBUTESINWORDS>&);
+ void sendStartBackup(Signal*, BackupRecordPtr, TablePtr);
+ void sendAlterTrig(Signal*, BackupRecordPtr ptr);
+
+ void sendDropTrig(Signal*, BackupRecordPtr ptr);
+ void sendDropTrig(Signal* signal, BackupRecordPtr ptr, TablePtr tabPtr);
+ void dropTrigReply(Signal*, BackupRecordPtr ptr);
+
+ void sendSignalAllWait(BackupRecordPtr ptr, Uint32 gsn, Signal *signal,
+ Uint32 signalLength,
+ bool executeDirect = false);
+ bool haveAllSignals(BackupRecordPtr ptr, Uint32 gsn, Uint32 nodeId);
+
+ void sendStopBackup(Signal*, BackupRecordPtr ptr);
+ void sendAbortBackupOrd(Signal* signal, BackupRecordPtr ptr, Uint32 errCode);
+ void sendAbortBackupOrdSlave(Signal* signal, BackupRecordPtr ptr,
+ Uint32 errCode);
+ void masterAbort(Signal*, BackupRecordPtr ptr, bool controlledAbort);
+ void masterSendAbortBackup(Signal*, BackupRecordPtr ptr);
+ void slaveAbort(Signal*, BackupRecordPtr ptr);
+
+ void abortFile(Signal* signal, BackupRecordPtr ptr, BackupFilePtr filePtr);
+ void abortFileHook(Signal* signal, BackupFilePtr filePtr, bool scanDone);
+
+ bool verifyNodesAlive(const NdbNodeBitmask& aNodeBitMask);
+ bool checkAbort(BackupRecordPtr ptr);
+ void checkNodeFail(Signal* signal,
+ BackupRecordPtr ptr,
+ NodeId newCoord,
+ Uint32 theFailedNodes[NodeBitmask::Size]);
+ void masterTakeOver(Signal* signal, BackupRecordPtr ptr);
+
+
+ NodeId getMasterNodeId() const { return c_masterNodeId; }
+ bool findTable(const BackupRecordPtr &, TablePtr &, Uint32 tableId) const;
+ TablePtr parseTableDescription(Signal*, BackupRecordPtr ptr, Uint32 len);
+
+ bool insertFileHeader(BackupFormat::FileType, BackupRecord*, BackupFile*);
+ void sendBackupRef(Signal* signal, BackupRecordPtr ptr, Uint32 errorCode);
+ void sendBackupRef(BlockReference ref, Signal *signal,
+ Uint32 senderData, Uint32 errorCode);
+ void dumpUsedResources();
+ void cleanupMasterResources(BackupRecordPtr ptr);
+ void cleanupSlaveResources(BackupRecordPtr ptr);
+ void cleanupFinalResources(BackupRecordPtr ptr);
+ void removeBackup(Signal*, BackupRecordPtr ptr);
+
+ void sendSTTORRY(Signal*);
+ void createSequence(Signal* signal);
+ void createSequenceReply(Signal*, class UtilSequenceConf *);
+};
+
+inline
+void
+Backup::OperationRecord::newRecord(Uint32 * p){
+ base = p;
+ dst_Length = p; p += 1;
+ dst_Bitmask = p; p += sz_Bitmask;
+ dst_FixedKeys = p; p += sz_FixedKeys;
+ dst_FixedAttribs = p; p += sz_FixedAttribs;
+ dst_VariableData = (BackupFormat::DataFile::VariableData*)p;
+ BitmaskImpl::clear(sz_Bitmask, dst_Bitmask);
+ attrLeft = noOfAttributes;
+ attrSzLeft = attrSzTotal = 0;
+}
+
+inline
+Uint32 *
+Backup::OperationRecord::newAttrib(Uint32 offset, Uint32 sz){
+ attrLeft--;
+ attrSzLeft = sz;
+ dst = dst_FixedAttribs + offset;
+ return dst;
+}
+
+inline
+Uint32 *
+Backup::OperationRecord::newKey(){
+ attrLeft --;
+ attrSzLeft = 0;
+ return dst_FixedKeys;
+}
+
+inline
+void
+Backup::OperationRecord::nullAttribute(Uint32 offsetNull){
+ attrLeft --;
+ BitmaskImpl::set(sz_Bitmask, dst_Bitmask, offsetNull);
+}
+
+inline
+Uint32 *
+Backup::OperationRecord::newNullable(Uint32 id, Uint32 sz){
+ attrLeft--;
+ attrSzLeft = sz;
+
+ dst = &dst_VariableData->Data[0];
+ dst_VariableData->Sz = htonl(sz);
+ dst_VariableData->Id = htonl(id);
+
+ dst_VariableData = (BackupFormat::DataFile::VariableData *)(dst + sz);
+
+ // Clear all bits on newRecord -> dont need to clear this
+ // BitmaskImpl::clear(sz_Bitmask, dst_Bitmask, offsetNull);
+ return dst;
+}
+
+inline
+Uint32 *
+Backup::OperationRecord::newVariable(Uint32 id, Uint32 sz){
+ attrLeft--;
+ attrSzLeft = sz;
+
+ dst = &dst_VariableData->Data[0];
+ dst_VariableData->Sz = htonl(sz);
+ dst_VariableData->Id = htonl(id);
+
+ dst_VariableData = (BackupFormat::DataFile::VariableData *)(dst + sz);
+ return dst;
+}
+
+inline
+Uint32 *
+Backup::OperationRecord::newVariableKey(Uint32 sz){
+ attrLeft--;
+ attrSzLeft = 0;
+
+ dst = &dst_VariableData->Data[0];
+ dst_VariableData->Sz = htonl(sz);
+ dst_VariableData->Id = htonl(variableKeyId);
+
+ dst_VariableData = (BackupFormat::DataFile::VariableData *)(dst + sz);
+ return dst;
+}
+
+inline
+bool
+Backup::OperationRecord::finished(){
+ if(attrLeft != 0 || attrSzLeft != 0){
+ return false;
+ }
+
+ attrLen[opNoDone] = attrSzTotal;
+ opNoDone++;
+
+ scanStop = dst = (Uint32 *)dst_VariableData;
+
+ const Uint32 len = (dst - base - 1);
+ * dst_Length = htonl(len);
+
+ noOfRecords++;
+
+ return true;
+}
+
+#endif
diff --git a/ndb/src/kernel/blocks/backup/Backup.txt b/ndb/src/kernel/blocks/backup/Backup.txt
new file mode 100644
index 00000000000..ee5e02bb549
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/Backup.txt
@@ -0,0 +1,343 @@
+-- BACKUP SIGNAL DIAGRAM COMPLEMENT TO BACKUP AMENDMENTS 2003-07-11 --
+
+USER MASTER MASTER SLAVE SLAVE
+---------------------------------------------------------------------
+BACKUP_REQ
+---------------->
+ UTIL_SEQUENCE
+ --------------->
+ <---------------
+ DEFINE_BACKUP
+ ------------------------------> (Local signals)
+ LIST_TABLES
+ --------------->
+ <---------------
+ FSOPEN
+ --------------->
+ GET_TABINFO
+ <---------------
+ DI_FCOUNT
+ --------------->
+ <---------------
+ DI_GETPRIM
+ --------------->
+ <---------------
+ <-------------------------------
+BACKUP_CONF
+<----------------
+ CREATE_TRIG
+ --------------> (If master crashes here -> rouge triggers/memory leak)
+ <--------------
+ START_BACKUP
+ ------------------------------>
+ <------------------------------
+ ALTER_TRIG
+ -------------->
+ <--------------
+ WAIT_GCP
+ -------------->
+ <--------------
+ BACKUP_FRAGMENT
+ ------------------------------>
+ SCAN_FRAG
+ --------------->
+ <---------------
+ <------------------------------
+ WAIT_GCP
+ -------------->
+ <--------------
+ DROP_TRIG
+ -------------->
+ <--------------
+ STOP_BACKUP
+ ------------------------------>
+ <------------------------------
+BACKUP_COMPLETE_REP
+<----------------
+ ABORT_BACKUP
+ ------------------------------>
+
+----------------------------------------------------------------------------
+
+USER BACKUP-MASTER
+
+1) BACKUP_REQ -->
+
+2) To all slaves DEFINE_BACKUP_REQ
+ This signals contains info so that all
+ slaves can take over as master
+ Tomas: Except triggerId info...
+
+3) Wait for conf
+
+4) <-- BACKUP_CONF
+
+5) For Each Table
+ PREP_CREATE_TRIG_REQ
+ Wait for Conf
+
+6) To all slaves START_BACKUP_REQ
+ Include trigger ids
+ Wait for conf
+
+7) For Each Table
+ CREATE_TRIG_REQ
+ Wait for conf
+
+8) Wait for GCP
+
+9) For each table
+ For each fragment
+ BACKUP_FRAGMENT_REQ -->
+ <-- BACKUP_FRAGMENT_CONF
+
+10) Wait for GCP
+
+11) To all slaves STOP_BACKUP_REQ
+ This signal turns off logging
+
+12) Wait for conf
+
+13) <-- BACKUP_COMPLETE_REP
+
+----
+
+Slave: Master Died
+Wait for master take-over, max 30 sec then abort everything
+
+Slave: Master TakeOver
+
+BACKUP_STATUS_REQ --> To all nodes
+<-- BACKUP_STATUS_CONF
+
+BACKUP_STATUS_CONF
+ BACKUP_DEFINED
+ BACKUP_STARTED
+ BACKUP_FRAGMENT
+
+Master: Slave died
+
+-- Define Backup Req --
+
+1) Get backup definition
+ Which tables (all)
+
+2) Open files
+ Write table list to CTL - file
+
+3) Get definitions for all tables in backup
+
+4) Get Fragment info
+
+5) Define Backup Conf
+
+-- Define Backup Req --
+
+-- Abort Backup Req --
+
+1) Report to others
+
+2) Stop logging
+3) Stop file(s)
+4) Stop scan
+
+5) If failure/abort
+ Remove files
+
+6) If XXX
+ Report to user
+7) Clean up records/stuff
+
+-- Abort Backup --
+
+Reasons for aborting:
+
+1a) client abort
+
+1b) slave failure
+
+1c) node failure
+
+Resources to be cleaned up:
+
+Slave responsability:
+
+2a) Close and remove files
+
+2b) Free allocated resources
+
+Master responsability:
+
+2c) Drop triggers
+
+USER MASTER MASTER SLAVE SLAVE
+---------------------------------------------------------------------
+ BACKUP_ABORT_ORD:
+ -------------------------(ALL)-->
+ Set Master State ABORTING Set Slave State ABORTING
+ Drop Triggers Close and Remove files
+ CleanupSlaveResources()
+
+ BACKUP_ABORT_ORD:OkToClean
+ -------------------------(ALL)-->
+
+
+ CleanupMasterResources()
+
+BACKUP_ABORT_REP
+<---------------
+
+
+
+State descriptions:
+
+Master - INITIAL
+BACKUP_REQ ->
+Master - DEFINING
+DEFINE_BACKUP_CONF ->
+Master - DEFINED
+CREATE_TRIG_CONF ->
+Master - STARTED
+<--->
+Master - SCANNING
+WAIT_GCP_CONF ->
+Master - STOPPING
+(Master - CLEANING)
+--------
+Master - ABORTING
+
+
+Slave - INITIAL
+DEFINE_BACKUP_REQ ->
+Slave - DEFINING
+ - backupId
+ - tables
+DIGETPRIMCONF ->
+Slave - DEFINED
+START_BACKUP_REQ ->
+Slave - STARTED
+Slave - SCANNING
+STOP_BACKUP_REQ ->
+Slave - STOPPING
+FSCLOSECONF ->
+Slave - CLEANING
+-----
+Slave - ABORTING
+
+
+
+Testcases:
+
+2. Master failure at first START_BACKUP_CONF
+
+<masterId> error 10004
+start backup
+
+- Ok
+
+2. Master failure at first CREATE_TRIG_CONF
+
+<masterId> error 10003
+start backup
+
+- Ok
+
+2. Master failure at first ALTER_TRIG_CONF
+
+<masterId> error 10005
+start backup
+
+- Ok
+
+2. Master failure at WAIT_GCP_CONF
+
+<masterId> error 10007
+start backup
+
+- Ok
+
+2. Master failure at WAIT_GCP_CONF, nextFragment
+
+<masterId> error 10008
+start backup
+
+- Ok
+
+2. Master failure at WAIT_GCP_CONF, stopping
+
+<masterId> error 10009
+start backup
+
+- Ok
+
+2. Master failure at BACKUP_FRAGMENT_CONF
+
+<masterId> error 10010
+start backup
+
+- Ok
+
+2. Master failure at first DROP_TRIG_CONF
+
+<masterId> error 10012
+start backup
+
+- Ok
+
+1. Master failure at first STOP_BACKUP_CONF
+
+<masterId> error 10013
+start backup
+
+- Ok
+
+3. Multiple node failiure:
+
+<masterId> error 10001
+<otheId> error 10014
+start backup
+
+- Ok (note, mgmtsrvr does gets BACKUP_ABORT_REP but expects BACKUP_REF, hangs...)
+
+4. Multiple node failiure:
+
+<masterId> error 10007
+<takeover id> error 10002
+start backup
+
+- Ok
+
+
+
+ ndbrequire(!ERROR_INSERTED(10001));
+ ndbrequire(!ERROR_INSERTED(10002));
+ ndbrequire(!ERROR_INSERTED(10021));
+ ndbrequire(!ERROR_INSERTED(10003));
+ ndbrequire(!ERROR_INSERTED(10004));
+ ndbrequire(!ERROR_INSERTED(10005));
+ ndbrequire(!ERROR_INSERTED(10006));
+ ndbrequire(!ERROR_INSERTED(10007));
+ ndbrequire(!ERROR_INSERTED(10008));
+ ndbrequire(!ERROR_INSERTED(10009));
+ ndbrequire(!ERROR_INSERTED(10010));
+ ndbrequire(!ERROR_INSERTED(10011));
+ ndbrequire(!ERROR_INSERTED(10012));
+ ndbrequire(!ERROR_INSERTED(10013));
+ ndbrequire(!ERROR_INSERTED(10014));
+ ndbrequire(!ERROR_INSERTED(10015));
+ ndbrequire(!ERROR_INSERTED(10016));
+ ndbrequire(!ERROR_INSERTED(10017));
+ ndbrequire(!ERROR_INSERTED(10018));
+ ndbrequire(!ERROR_INSERTED(10019));
+ ndbrequire(!ERROR_INSERTED(10020));
+
+ if (ERROR_INSERTED(10023)) {
+ if (ERROR_INSERTED(10023)) {
+ if (ERROR_INSERTED(10024)) {
+ if (ERROR_INSERTED(10025)) {
+ if (ERROR_INSERTED(10026)) {
+ if (ERROR_INSERTED(10028)) {
+ if (ERROR_INSERTED(10027)) {
+ (ERROR_INSERTED(10022))) {
+ if (ERROR_INSERTED(10029)) {
+ if(trigPtr.p->operation->noOfBytes > 123 && ERROR_INSERTED(10030)) {
diff --git a/ndb/src/kernel/blocks/backup/BackupFormat.hpp b/ndb/src/kernel/blocks/backup/BackupFormat.hpp
new file mode 100644
index 00000000000..65dd2ad9053
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/BackupFormat.hpp
@@ -0,0 +1,149 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BACKUP_FORMAT_HPP
+#define BACKUP_FORMAT_HPP
+
+#include <ndb_types.h>
+
+static const char BACKUP_MAGIC[] = { 'N', 'D', 'B', 'B', 'C', 'K', 'U', 'P' };
+
+struct BackupFormat {
+
+ /**
+ * Section types in file
+ */
+ enum SectionType {
+ FILE_HEADER = 1,
+ FRAGMENT_HEADER = 2,
+ FRAGMENT_FOOTER = 3,
+ TABLE_LIST = 4,
+ TABLE_DESCRIPTION = 5,
+ GCP_ENTRY = 6
+ };
+
+ struct FileHeader {
+ char Magic[8];
+ Uint32 NdbVersion;
+
+ Uint32 SectionType;
+ Uint32 SectionLength;
+ Uint32 FileType;
+ Uint32 BackupId;
+ Uint32 BackupKey_0;
+ Uint32 BackupKey_1;
+ Uint32 ByteOrder;
+ };
+
+ /**
+ * File types
+ */
+ enum FileType {
+ CTL_FILE = 1,
+ LOG_FILE = 2,
+ DATA_FILE = 3
+ };
+
+ /**
+ * Data file formats
+ */
+ struct DataFile {
+
+ struct FragmentHeader {
+ Uint32 SectionType;
+ Uint32 SectionLength;
+ Uint32 TableId;
+ Uint32 FragmentNo;
+ Uint32 ChecksumType;
+ };
+
+ struct VariableData {
+ Uint32 Sz;
+ Uint32 Id;
+ Uint32 Data[1];
+ };
+
+ struct Record {
+ Uint32 Length;
+ Uint32 NullBitmask[1];
+ Uint32 DataFixedKeys[1];
+ Uint32 DataFixedAttributes[1];
+ VariableData DataVariableAttributes[1];
+ };
+
+ struct FragmentFooter {
+ Uint32 SectionType;
+ Uint32 SectionLength;
+ Uint32 TableId;
+ Uint32 FragmentNo;
+ Uint32 NoOfRecords;
+ Uint32 Checksum;
+ };
+ };
+
+ /**
+ * CTL file formats
+ */
+ struct CtlFile {
+
+ /**
+ * Table list
+ */
+ struct TableList {
+ Uint32 SectionType;
+ Uint32 SectionLength;
+ Uint32 TableIds[1]; // Length = SectionLength - 2
+ };
+
+ /**
+ * Table description(s)
+ */
+ struct TableDescription {
+ Uint32 SectionType;
+ Uint32 SectionLength;
+ Uint32 DictTabInfo[1]; // Length = SectionLength - 2
+ };
+
+ /**
+ * GCP Entry
+ */
+ struct GCPEntry {
+ Uint32 SectionType;
+ Uint32 SectionLength;
+ Uint32 StartGCP;
+ Uint32 StopGCP;
+ };
+ };
+
+ /**
+ * LOG file format
+ */
+ struct LogFile {
+
+ /**
+ * Log Entry
+ */
+ struct LogEntry {
+ Uint32 Length;
+ Uint32 TableId;
+ // If TriggerEvent & 0x10000 == true then GCI is right after data
+ Uint32 TriggerEvent;
+ Uint32 Data[1]; // Len = Length - 2
+ };
+ };
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/backup/BackupInit.cpp b/ndb/src/kernel/blocks/backup/BackupInit.cpp
new file mode 100644
index 00000000000..1997e560bb9
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/BackupInit.cpp
@@ -0,0 +1,215 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//****************************************************************************
+//
+// NAME
+// Backup - Database backup / restore
+//
+//===========================================================================
+#include "Backup.hpp"
+
+#include <new>
+#include <Properties.hpp>
+#include <Configuration.hpp>
+
+//extern const unsigned Ndbcntr::g_sysTableCount;
+
+Backup::Backup(const Configuration & conf) :
+ SimulatedBlock(BACKUP, conf),
+ c_nodes(c_nodePool),
+ c_backups(c_backupPool)
+{
+ BLOCK_CONSTRUCTOR(Backup);
+
+ c_nodePool.setSize(MAX_NDB_NODES);
+ c_masterNodeId = getOwnNodeId();
+
+ const Properties * p = conf.getOwnProperties();
+ ndbrequire(p != 0);
+
+ Uint32 noBackups = 0, noTables = 0, noAttribs = 0;
+ p->get("ParallelBackups", &noBackups);
+ ndbrequire(p->get("MaxNoOfTables", &noTables));
+ ndbrequire(p->get("MaxNoOfAttributes", &noAttribs));
+
+ // To allow for user tables AND SYSTAB
+ // See ClusterConfig
+ //TODO get this infor from NdbCntr
+ noTables += 2;
+
+ // Considering also TR527, this is a KISS work-around to be able to
+ // continue testing the real thing
+ noAttribs += 2 + 1;
+
+ c_backupPool.setSize(noBackups);
+ c_backupFilePool.setSize(3 * noBackups);
+ c_tablePool.setSize(noBackups * noTables);
+ c_attributePool.setSize(noBackups * noAttribs);
+ c_triggerPool.setSize(noBackups * 3 * noTables);
+
+ // 2 = no of replicas
+ c_fragmentPool.setSize(noBackups * 2 * NO_OF_FRAG_PER_NODE * noTables);
+
+ Uint32 szMem = 0;
+ p->get("BackupMemory", &szMem);
+ Uint32 noPages = (szMem + sizeof(Page32) - 1) / sizeof(Page32);
+ // We need to allocate an additional of 2 pages. 1 page because of a bug in
+ // ArrayPool and another one for DICTTAINFO.
+ c_pagePool.setSize(noPages + NO_OF_PAGES_META_FILE + 2);
+
+ Uint32 szDataBuf = (2 * 1024 * 1024);
+ Uint32 szLogBuf = (2 * 1024 * 1024);
+ Uint32 szWrite = 32768;
+ p->get("BackupDataBufferSize", &szDataBuf);
+ p->get("BackupLogBufferSize", &szLogBuf);
+ p->get("BackupWriteSize", &szWrite);
+
+ c_defaults.m_logBufferSize = szLogBuf;
+ c_defaults.m_dataBufferSize = szDataBuf;
+ c_defaults.m_minWriteSize = szWrite;
+ c_defaults.m_maxWriteSize = szWrite;
+
+ { // Init all tables
+ ArrayList<Table> tables(c_tablePool);
+ TablePtr ptr;
+ while(tables.seize(ptr)){
+ new (ptr.p) Table(c_attributePool, c_fragmentPool);
+ }
+ tables.release();
+ }
+
+ {
+ ArrayList<BackupFile> ops(c_backupFilePool);
+ BackupFilePtr ptr;
+ while(ops.seize(ptr)){
+ new (ptr.p) BackupFile(* this, c_pagePool);
+ }
+ ops.release();
+ }
+
+ {
+ ArrayList<BackupRecord> recs(c_backupPool);
+ BackupRecordPtr ptr;
+ while(recs.seize(ptr)){
+ new (ptr.p) BackupRecord(* this, c_pagePool, c_tablePool,
+ c_backupFilePool, c_triggerPool);
+ }
+ recs.release();
+ }
+
+ // Initialize BAT for interface to file system
+ {
+ Page32Ptr p;
+ ndbrequire(c_pagePool.seizeId(p, 0));
+ c_startOfPages = (Uint32 *)p.p;
+ c_pagePool.release(p);
+
+ NewVARIABLE* bat = allocateBat(1);
+ bat[0].WA = c_startOfPages;
+ bat[0].nrr = c_pagePool.getSize()*sizeof(Page32)/sizeof(Uint32);
+ }
+
+ // Add received signals
+ addRecSignal(GSN_STTOR, &Backup::execSTTOR);
+ addRecSignal(GSN_DUMP_STATE_ORD, &Backup::execDUMP_STATE_ORD);
+ addRecSignal(GSN_READ_NODESCONF, &Backup::execREAD_NODESCONF);
+ addRecSignal(GSN_NODE_FAILREP, &Backup::execNODE_FAILREP);
+ addRecSignal(GSN_INCL_NODEREQ, &Backup::execINCL_NODEREQ);
+ addRecSignal(GSN_CONTINUEB, &Backup::execCONTINUEB);
+
+ addRecSignal(GSN_SCAN_HBREP, &Backup::execSCAN_HBREP);
+ addRecSignal(GSN_TRANSID_AI, &Backup::execTRANSID_AI);
+ addRecSignal(GSN_KEYINFO20, &Backup::execKEYINFO20);
+ addRecSignal(GSN_SCAN_FRAGREF, &Backup::execSCAN_FRAGREF);
+ addRecSignal(GSN_SCAN_FRAGCONF, &Backup::execSCAN_FRAGCONF);
+
+ addRecSignal(GSN_BACKUP_TRIG_REQ, &Backup::execBACKUP_TRIG_REQ);
+ addRecSignal(GSN_TRIG_ATTRINFO, &Backup::execTRIG_ATTRINFO);
+ addRecSignal(GSN_FIRE_TRIG_ORD, &Backup::execFIRE_TRIG_ORD);
+
+ addRecSignal(GSN_LIST_TABLES_CONF, &Backup::execLIST_TABLES_CONF);
+ addRecSignal(GSN_GET_TABINFOREF, &Backup::execGET_TABINFOREF);
+ addRecSignal(GSN_GET_TABINFO_CONF, &Backup::execGET_TABINFO_CONF);
+
+ addRecSignal(GSN_CREATE_TRIG_REF, &Backup::execCREATE_TRIG_REF);
+ addRecSignal(GSN_CREATE_TRIG_CONF, &Backup::execCREATE_TRIG_CONF);
+
+ addRecSignal(GSN_ALTER_TRIG_REF, &Backup::execALTER_TRIG_REF);
+ addRecSignal(GSN_ALTER_TRIG_CONF, &Backup::execALTER_TRIG_CONF);
+
+ addRecSignal(GSN_DROP_TRIG_REF, &Backup::execDROP_TRIG_REF);
+ addRecSignal(GSN_DROP_TRIG_CONF, &Backup::execDROP_TRIG_CONF);
+
+ addRecSignal(GSN_DI_FCOUNTCONF, &Backup::execDI_FCOUNTCONF);
+ addRecSignal(GSN_DIGETPRIMCONF, &Backup::execDIGETPRIMCONF);
+
+ addRecSignal(GSN_FSOPENREF, &Backup::execFSOPENREF);
+ addRecSignal(GSN_FSOPENCONF, &Backup::execFSOPENCONF);
+
+ addRecSignal(GSN_FSCLOSEREF, &Backup::execFSCLOSEREF);
+ addRecSignal(GSN_FSCLOSECONF, &Backup::execFSCLOSECONF);
+
+ addRecSignal(GSN_FSAPPENDREF, &Backup::execFSAPPENDREF);
+ addRecSignal(GSN_FSAPPENDCONF, &Backup::execFSAPPENDCONF);
+
+ addRecSignal(GSN_FSREMOVEREF, &Backup::execFSREMOVEREF);
+ addRecSignal(GSN_FSREMOVECONF, &Backup::execFSREMOVECONF);
+
+ /*****/
+ addRecSignal(GSN_BACKUP_REQ, &Backup::execBACKUP_REQ);
+ addRecSignal(GSN_ABORT_BACKUP_ORD, &Backup::execABORT_BACKUP_ORD);
+
+ addRecSignal(GSN_DEFINE_BACKUP_REQ, &Backup::execDEFINE_BACKUP_REQ);
+ addRecSignal(GSN_DEFINE_BACKUP_REF, &Backup::execDEFINE_BACKUP_REF);
+ addRecSignal(GSN_DEFINE_BACKUP_CONF, &Backup::execDEFINE_BACKUP_CONF);
+
+ addRecSignal(GSN_START_BACKUP_REQ, &Backup::execSTART_BACKUP_REQ);
+ addRecSignal(GSN_START_BACKUP_REF, &Backup::execSTART_BACKUP_REF);
+ addRecSignal(GSN_START_BACKUP_CONF, &Backup::execSTART_BACKUP_CONF);
+
+ addRecSignal(GSN_BACKUP_FRAGMENT_REQ, &Backup::execBACKUP_FRAGMENT_REQ);
+ //addRecSignal(GSN_BACKUP_FRAGMENT_REF, &Backup::execBACKUP_FRAGMENT_REF);
+ addRecSignal(GSN_BACKUP_FRAGMENT_CONF, &Backup::execBACKUP_FRAGMENT_CONF);
+
+ addRecSignal(GSN_STOP_BACKUP_REQ, &Backup::execSTOP_BACKUP_REQ);
+ addRecSignal(GSN_STOP_BACKUP_REF, &Backup::execSTOP_BACKUP_REF);
+ addRecSignal(GSN_STOP_BACKUP_CONF, &Backup::execSTOP_BACKUP_CONF);
+
+ //addRecSignal(GSN_BACKUP_STATUS_REQ, &Backup::execBACKUP_STATUS_REQ);
+ //addRecSignal(GSN_BACKUP_STATUS_CONF, &Backup::execBACKUP_STATUS_CONF);
+
+ addRecSignal(GSN_UTIL_SEQUENCE_REF, &Backup::execUTIL_SEQUENCE_REF);
+ addRecSignal(GSN_UTIL_SEQUENCE_CONF, &Backup::execUTIL_SEQUENCE_CONF);
+
+ addRecSignal(GSN_WAIT_GCP_REF, &Backup::execWAIT_GCP_REF);
+ addRecSignal(GSN_WAIT_GCP_CONF, &Backup::execWAIT_GCP_CONF);
+
+ /**
+ * Testing
+ */
+ addRecSignal(GSN_BACKUP_REF, &Backup::execBACKUP_REF);
+ addRecSignal(GSN_BACKUP_CONF, &Backup::execBACKUP_CONF);
+ addRecSignal(GSN_BACKUP_ABORT_REP, &Backup::execBACKUP_ABORT_REP);
+ addRecSignal(GSN_BACKUP_COMPLETE_REP, &Backup::execBACKUP_COMPLETE_REP);
+}
+
+Backup::~Backup()
+{
+}
+
+BLOCK_FUNCTIONS(Backup);
+
diff --git a/ndb/src/kernel/blocks/backup/FsBuffer.hpp b/ndb/src/kernel/blocks/backup/FsBuffer.hpp
new file mode 100644
index 00000000000..4b5d95a19a5
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/FsBuffer.hpp
@@ -0,0 +1,346 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FS_BUFFER_HPP
+#define FS_BUFFER_HPP
+
+#include <assert.h>
+#include <stdlib.h>
+#include <string.h>
+#include <ndb_types.h>
+
+#define DEBUG(x)
+
+/**
+ * A circular data buffer to be used together with the FS
+ *
+ * One writer - Typically your block
+ * getWritePtr()
+ * updateWritePtr()
+ *
+ * One reader - Typically "thread" in your block sending stuff to NDBFS
+ * getReadPtr()
+ * updateReadPtr()
+ */
+class FsBuffer {
+public:
+ /**
+ * Default constructor
+ */
+ FsBuffer();
+
+ /**
+ * setup FsBuffer
+ *
+ * @param Buffer - Ptr to continuous memory
+ * @param Size - Buffer size in 32-bit words
+ * @param BlockSize - Size of block in 32-bit words
+ * @param MinRead - Min read size in 32-bit words
+ * Get rounded(down) to nearest multiple of block size.
+ * @param MaxRead - Max read size in 32-bit words
+ * Get rounded(down) to nearest multiple of block size.
+ * @param MaxWrite - Maximum write (into buffer) in 32-bit words
+ *
+ * @return NULL if everything is OK
+ * else A string describing problem
+ */
+ const char * setup(Uint32 * Buffer,
+ Uint32 Size,
+ Uint32 BlockSize = 128, // 512 bytes
+ Uint32 MinRead = 1024, // 4k
+ Uint32 MaxRead = 1024, // 4k
+ Uint32 MaxWrite = 1024); // 4k
+ /*
+ * @return NULL if everything is OK
+ * else A string describing problem
+ */
+ const char * valid() const;
+
+ Uint32 getBufferSize() const;
+ Uint32 getUsableSize() const;
+ Uint32 * getStart() const;
+
+ /**
+ * getReadPtr - Get pointer and size of data to send to FS
+ *
+ * @param ptr - Where to fetch data
+ * @param sz - How much data in 32-bit words
+ * @param eof - Is this the last fetch (only if return false)
+ *
+ * @return true - If there is data of size >= minread
+ * false - If there is can be data be if it is is < minread
+ * - else eof = true
+ */
+ bool getReadPtr(Uint32 ** ptr, Uint32 * sz, bool * eof);
+
+ /**
+ * @note: sz must be equal to sz returned by getReadPtr
+ */
+ void updateReadPtr(Uint32 sz);
+
+ /**
+ *
+ * @note Must be followed by a updateWritePtr(no of words used)
+ */
+ bool getWritePtr(Uint32 ** ptr, Uint32 sz);
+
+ void updateWritePtr(Uint32 sz);
+
+ /**
+ * There will be no more writing to this buffer
+ */
+ void eof();
+
+ /**
+ * Getters for varibles
+ */
+ Uint32 getMaxWrite() const { return m_maxWrite;}
+ Uint32 getMinRead() const { return m_minRead;}
+
+ Uint32 getFreeSize() const { return m_free; }
+
+
+private:
+
+ Uint32 m_free;
+ Uint32 m_readIndex;
+ Uint32 m_writeIndex;
+ Uint32 m_eof;
+ Uint32 * m_start;
+ Uint32 m_minRead;
+ Uint32 m_maxRead;
+ Uint32 m_maxWrite;
+ Uint32 m_size;
+
+ Uint32 * m_buffer;
+ Uint32 m_bufSize;
+ Uint32 m_blockSize;
+
+ void clear();
+};
+
+inline
+FsBuffer::FsBuffer()
+{
+ clear();
+}
+
+inline
+void
+FsBuffer::clear(){
+ m_minRead = m_maxRead = m_maxWrite = m_size = m_bufSize = m_free = 0;
+ m_buffer = m_start = 0;
+}
+
+static
+Uint32 *
+align(Uint32 * ptr, Uint32 alignment, bool downwards){
+
+ const UintPtr a = (UintPtr)ptr;
+ const UintPtr b = a % alignment;
+
+ if(downwards){
+ return (Uint32 *)(a - b);
+ } else {
+ return (Uint32 *)(a + (b == 0 ? 0 : (alignment - b)));
+ }
+}
+
+inline
+const char *
+FsBuffer::setup(Uint32 * Buffer,
+ Uint32 Size,
+ Uint32 Block,
+ Uint32 MinRead,
+ Uint32 MaxRead,
+ Uint32 MaxWrite)
+{
+ clear();
+ m_buffer = Buffer;
+ m_bufSize = Size;
+ m_blockSize = Block;
+ if(Block == 0){
+ return valid();
+ }
+
+ m_minRead = (MinRead / Block) * Block;
+ m_maxRead = (MaxRead / Block) * Block;
+ m_maxWrite = MaxWrite;
+
+ m_start = align(Buffer, Block*4, false);
+ Uint32 * stop = align(Buffer + Size - MaxWrite, Block*4, true);
+ if(stop > m_start){
+ m_size = stop - m_start;
+ } else {
+ m_size = 0;
+ }
+
+ if(m_minRead == 0)
+ m_size = 0;
+ else
+ m_size = (m_size / m_minRead) * m_minRead;
+
+#if 0
+ ndbout_c("Block = %d MinRead = %d -> %d", Block*4, MinRead*4, m_minRead*4);
+ ndbout_c("Block = %d MaxRead = %d -> %d", Block*4, MaxRead*4, m_maxRead*4);
+
+ ndbout_c("Buffer = %d -> %d", Buffer, m_start);
+ ndbout_c("Buffer = %d Size = %d MaxWrite = %d -> %d",
+ Buffer, Size*4, MaxWrite*4, m_size*4);
+#endif
+
+ m_readIndex = m_writeIndex = m_eof = 0;
+ m_free = m_size;
+ return valid();
+}
+
+inline
+const char *
+FsBuffer::valid() const {
+ if(m_buffer == 0) return "Null pointer buffer";
+ if(m_bufSize == 0) return "Zero size buffer";
+ if(m_blockSize == 0) return "Zero block size";
+ if(m_minRead < m_blockSize) return "Min read less than block size";
+ if(m_maxRead < m_blockSize) return "Max read less than block size";
+ if(m_maxRead < m_minRead) return "Max read less than min read";
+ if(m_size == 0) return "Zero usable space";
+ return 0;
+}
+
+inline
+Uint32
+FsBuffer::getBufferSize() const {
+ return m_bufSize;
+}
+
+inline
+Uint32
+FsBuffer::getUsableSize() const {
+ return m_size;
+}
+
+inline
+Uint32 *
+FsBuffer::getStart() const {
+ return m_start;
+}
+
+inline
+bool
+FsBuffer::getReadPtr(Uint32 ** ptr, Uint32 * sz, bool * _eof){
+
+ Uint32 * Tp = m_start;
+ const Uint32 Tr = m_readIndex;
+ const Uint32 Tm = m_minRead;
+ const Uint32 Ts = m_size;
+ const Uint32 Tmw = m_maxRead;
+
+ Uint32 sz1 = m_size - m_free; // Used
+
+ if(sz1 >= Tm){
+ if(Tr + sz1 > Ts)
+ sz1 = (Ts - Tr);
+
+ if(sz1 > Tmw)
+ * sz = Tmw;
+ else
+ * sz = sz1 - (sz1 % Tm);
+
+ * ptr = &Tp[Tr];
+
+ DEBUG(ndbout_c("getReadPtr() Tr: %d Tw: %d Ts: %d Tm: %d sz1: %d -> %d",
+ Tr, Tw, Ts, Tm, sz1, * sz));
+
+ return true;
+ }
+
+ if(!m_eof){
+ * _eof = false;
+
+ DEBUG(ndbout_c("getReadPtr() Tr: %d Tw: %d Ts: %d Tm: %d sz1: %d -> false",
+ Tr, Tw, Ts, Tm, sz1));
+
+ return false;
+ }
+
+ * sz = sz1;
+ * _eof = true;
+ * ptr = &Tp[Tr];
+
+ DEBUG(ndbout_c("getReadPtr() Tr: %d Tw: %d Ts: %d Tm: %d sz1: %d -> %d eof",
+ Tr, Tw, Ts, Tm, sz1, * sz));
+
+ return false;
+}
+
+inline
+void
+FsBuffer::updateReadPtr(Uint32 sz){
+ const Uint32 Tr = m_readIndex;
+ const Uint32 Ts = m_size;
+
+ m_free += sz;
+ m_readIndex = (Tr + sz) % Ts;
+}
+
+inline
+bool
+FsBuffer::getWritePtr(Uint32 ** ptr, Uint32 sz){
+ assert(sz <= m_maxWrite);
+ Uint32 * Tp = m_start;
+ const Uint32 Tw = m_writeIndex;
+ const Uint32 sz1 = m_free;
+
+ if(sz1 > sz){ // Note at least 1 word of slack
+ * ptr = &Tp[Tw];
+
+ DEBUG(ndbout_c("getWritePtr(%d) Tr: %d Tw: %d Ts: %d sz1: %d -> true",
+ sz, Tr, Tw, Ts, sz1));
+ return true;
+ }
+
+ DEBUG(ndbout_c("getWritePtr(%d) Tr: %d Tw: %d Ts: %d sz1: %d -> false",
+ sz, Tr, Tw, Ts, sz1));
+
+ return false;
+}
+
+inline
+void
+FsBuffer::updateWritePtr(Uint32 sz){
+ assert(sz <= m_maxWrite);
+ Uint32 * Tp = m_start;
+ const Uint32 Tw = m_writeIndex;
+ const Uint32 Ts = m_size;
+
+ const Uint32 Tnew = (Tw + sz);
+ m_free -= sz;
+ if(Tnew < Ts){
+ m_writeIndex = Tnew;
+ return;
+ }
+
+ memcpy(Tp, &Tp[Ts], (Tnew - Ts) << 2);
+ m_writeIndex = Tnew - Ts;
+}
+
+inline
+void
+FsBuffer::eof(){
+ m_eof = 1;
+}
+
+#endif
diff --git a/ndb/src/kernel/blocks/backup/Makefile b/ndb/src/kernel/blocks/backup/Makefile
new file mode 100644
index 00000000000..989199cbe02
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/Makefile
@@ -0,0 +1,18 @@
+include .defs.mk
+
+TYPE := kernel
+
+#ifneq ($(MYSQLCLUSTER_TOP),)
+DIRS := restore
+#endif
+
+ARCHIVE_TARGET := backup
+
+SOURCES = Backup.cpp BackupInit.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+$(NDB_TOP)/bin/readBackupFile: read.o
+ $(C++) -o $@ read.o \
+ $(NDB_TOP)/lib/libportlib.a $(NDB_TOP)/lib/libgeneral.a
+
diff --git a/ndb/src/kernel/blocks/backup/read.cpp b/ndb/src/kernel/blocks/backup/read.cpp
new file mode 100644
index 00000000000..8300c74ab43
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/read.cpp
@@ -0,0 +1,479 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include <NdbTCP.h>
+#include <NdbOut.hpp>
+#include "BackupFormat.hpp"
+#include <AttributeHeader.hpp>
+#include <SimpleProperties.hpp>
+
+bool readHeader(FILE*, BackupFormat::FileHeader *);
+bool readFragHeader(FILE*, BackupFormat::DataFile::FragmentHeader *);
+bool readFragFooter(FILE*, BackupFormat::DataFile::FragmentFooter *);
+Int32 readRecord(FILE*, Uint32 **);
+
+NdbOut & operator<<(NdbOut&, const BackupFormat::FileHeader &);
+NdbOut & operator<<(NdbOut&, const BackupFormat::DataFile::FragmentHeader &);
+NdbOut & operator<<(NdbOut&, const BackupFormat::DataFile::FragmentFooter &);
+
+bool readTableList(FILE*, BackupFormat::CtlFile::TableList **);
+bool readTableDesc(FILE*, BackupFormat::CtlFile::TableDescription **);
+bool readGCPEntry(FILE*, BackupFormat::CtlFile::GCPEntry **);
+
+NdbOut & operator<<(NdbOut&, const BackupFormat::CtlFile::TableList &);
+NdbOut & operator<<(NdbOut&, const BackupFormat::CtlFile::TableDescription &);
+NdbOut & operator<<(NdbOut&, const BackupFormat::CtlFile::GCPEntry &);
+
+Int32 readLogEntry(FILE*, Uint32**);
+
+static Uint32 recNo;
+static Uint32 logEntryNo;
+
+int
+main(int argc, const char * argv[]){
+
+ if(argc <= 1){
+ printf("Usage: %s <filename>", argv[0]);
+ exit(1);
+ }
+ FILE * f = fopen(argv[1], "rb");
+ if(!f){
+ ndbout << "No such file!" << endl;
+ exit(1);
+ }
+
+ BackupFormat::FileHeader fileHeader;
+ if(!readHeader(f, &fileHeader)){
+ ndbout << "Invalid file!" << endl;
+ exit(1);
+ }
+ ndbout << fileHeader << endl;
+
+ switch(fileHeader.FileType){
+ case BackupFormat::DATA_FILE:
+ while(!feof(f)){
+ BackupFormat::DataFile::FragmentHeader fragHeader;
+ if(!readFragHeader(f, &fragHeader))
+ break;
+ ndbout << fragHeader << endl;
+
+ Uint32 len, * data;
+ while((len = readRecord(f, &data)) > 0){
+#if 0
+ ndbout << "-> " << hex;
+ for(Uint32 i = 0; i<len; i++){
+ ndbout << data[i] << " ";
+ }
+ ndbout << endl;
+#endif
+ }
+
+ BackupFormat::DataFile::FragmentFooter fragFooter;
+ if(!readFragFooter(f, &fragFooter))
+ break;
+ ndbout << fragFooter << endl;
+ }
+ break;
+ case BackupFormat::CTL_FILE:{
+ BackupFormat::CtlFile::TableList * tabList;
+ if(!readTableList(f, &tabList)){
+ ndbout << "Invalid file! No table list" << endl;
+ break;
+ }
+ ndbout << (* tabList) << endl;
+
+ const Uint32 noOfTables = tabList->SectionLength - 2;
+ for(Uint32 i = 0; i<noOfTables; i++){
+ BackupFormat::CtlFile::TableDescription * tabDesc;
+ if(!readTableDesc(f, &tabDesc)){
+ ndbout << "Invalid file missing table description" << endl;
+ break;
+ }
+ ndbout << (* tabDesc) << endl;
+ }
+
+ BackupFormat::CtlFile::GCPEntry * gcpE;
+ if(!readGCPEntry(f, &gcpE)){
+ ndbout << "Invalid file! GCP ENtry" << endl;
+ break;
+ }
+ ndbout << (* gcpE) << endl;
+
+ break;
+ }
+ case BackupFormat::LOG_FILE:{
+ logEntryNo = 0;
+
+ typedef BackupFormat::LogFile::LogEntry LogEntry;
+
+ Uint32 len, * data;
+ while((len = readLogEntry(f, &data)) > 0){
+ LogEntry * logEntry = (LogEntry *) data;
+ /**
+ * Log Entry
+ */
+ Uint32 event = ntohl(logEntry->TriggerEvent);
+ bool gcp = (event & 0x10000) != 0;
+ event &= 0xFFFF;
+ if(gcp)
+ len --;
+
+ ndbout << "LogEntry Table: " << (Uint32)ntohl(logEntry->TableId)
+ << " Event: " << event
+ << " Length: " << (len - 2);
+
+ const Uint32 dataLen = len - 2;
+#if 0
+ Uint32 pos = 0;
+ while(pos < dataLen){
+ AttributeHeader * ah = (AttributeHeader*)&logEntry->Data[pos];
+ ndbout_c(" Attribut: %d Size: %d",
+ ah->getAttributeId(),
+ ah->getDataSize());
+ pos += ah->getDataSize() + 1;
+ }
+#endif
+ if(gcp)
+ ndbout << " GCP: " << (Uint32)ntohl(logEntry->Data[dataLen]);
+ ndbout << endl;
+ }
+ break;
+ }
+ default:
+ ndbout << "Unsupported file type for printer: "
+ << fileHeader.FileType << endl;
+ break;
+ }
+ fclose(f);
+ return 0;
+}
+
+#define RETURN_FALSE() { ndbout_c("false: %d", __LINE__); abort(); return false; }
+
+static bool endian = false;
+
+bool
+readHeader(FILE* f, BackupFormat::FileHeader * dst){
+ if(fread(dst, 4, 3, f) != 3)
+ RETURN_FALSE();
+
+ if(memcmp(dst->Magic, BACKUP_MAGIC, sizeof(BACKUP_MAGIC)) != 0)
+ RETURN_FALSE();
+
+ dst->NdbVersion = ntohl(dst->NdbVersion);
+ if(dst->NdbVersion != 210)
+ RETURN_FALSE();
+
+ if(fread(&dst->SectionType, 4, 2, f) != 2)
+ RETURN_FALSE();
+ dst->SectionType = ntohl(dst->SectionType);
+ dst->SectionLength = ntohl(dst->SectionLength);
+
+ if(dst->SectionType != BackupFormat::FILE_HEADER)
+ RETURN_FALSE();
+
+ if(dst->SectionLength != ((sizeof(BackupFormat::FileHeader) - 12) >> 2))
+ RETURN_FALSE();
+
+ if(fread(&dst->FileType, 4, dst->SectionLength - 2, f) !=
+ (dst->SectionLength - 2))
+ RETURN_FALSE();
+
+ dst->FileType = ntohl(dst->FileType);
+ dst->BackupId = ntohl(dst->BackupId);
+ dst->BackupKey_0 = ntohl(dst->BackupKey_0);
+ dst->BackupKey_1 = ntohl(dst->BackupKey_1);
+
+ if(dst->FileType < BackupFormat::CTL_FILE ||
+ dst->FileType > BackupFormat::DATA_FILE)
+ RETURN_FALSE();
+
+ if(dst->ByteOrder != 0x12345678)
+ endian = true;
+
+ return true;
+}
+
+bool
+readFragHeader(FILE* f, BackupFormat::DataFile::FragmentHeader * dst){
+ if(fread(dst, 1, sizeof(* dst), f) != sizeof(* dst))
+ return false;
+
+ dst->SectionType = ntohl(dst->SectionType);
+ dst->SectionLength = ntohl(dst->SectionLength);
+ dst->TableId = ntohl(dst->TableId);
+ dst->FragmentNo = ntohl(dst->FragmentNo);
+ dst->ChecksumType = ntohl(dst->ChecksumType);
+
+ if(dst->SectionLength != (sizeof(* dst) >> 2))
+ RETURN_FALSE();
+
+ if(dst->SectionType != BackupFormat::FRAGMENT_HEADER)
+ RETURN_FALSE();
+
+ recNo = 0;
+
+ return true;
+}
+
+bool
+readFragFooter(FILE* f, BackupFormat::DataFile::FragmentFooter * dst){
+ if(fread(dst, 1, sizeof(* dst), f) != sizeof(* dst))
+ RETURN_FALSE();
+
+ dst->SectionType = ntohl(dst->SectionType);
+ dst->SectionLength = ntohl(dst->SectionLength);
+ dst->TableId = ntohl(dst->TableId);
+ dst->FragmentNo = ntohl(dst->FragmentNo);
+ dst->NoOfRecords = ntohl(dst->NoOfRecords);
+ dst->Checksum = ntohl(dst->Checksum);
+
+ if(dst->SectionLength != (sizeof(* dst) >> 2))
+ RETURN_FALSE();
+
+ if(dst->SectionType != BackupFormat::FRAGMENT_FOOTER)
+ RETURN_FALSE();
+ return true;
+}
+
+static Uint32 buf[8192];
+
+Int32
+readRecord(FILE* f, Uint32 **dst){
+ Uint32 len;
+ if(fread(&len, 1, 4, f) != 4)
+ RETURN_FALSE();
+
+ len = ntohl(len);
+
+ if(fread(buf, 4, len, f) != len)
+ return -1;
+
+ if(len > 0)
+ recNo++;
+
+ * dst = &buf[0];
+
+ return len;
+}
+
+Int32
+readLogEntry(FILE* f, Uint32 **dst){
+ Uint32 len;
+ if(fread(&len, 1, 4, f) != 4)
+ RETURN_FALSE();
+
+ len = ntohl(len);
+
+ if(fread(&buf[1], 4, len, f) != len)
+ return -1;
+
+ buf[0] = len;
+
+ if(len > 0)
+ logEntryNo++;
+
+ * dst = &buf[0];
+
+ return len;
+}
+
+
+NdbOut &
+operator<<(NdbOut& ndbout, const BackupFormat::FileHeader & hf){
+
+ char buf[9];
+ memcpy(buf, hf.Magic, sizeof(hf.Magic));
+ buf[8] = 0;
+
+ ndbout << "-- FileHeader:" << endl;
+ ndbout << "Magic: " << buf << endl;
+ ndbout << "NdbVersion: " << hf.NdbVersion << endl;
+ ndbout << "SectionType: " << hf.SectionType << endl;
+ ndbout << "SectionLength: " << hf.SectionLength << endl;
+ ndbout << "FileType: " << hf.FileType << endl;
+ ndbout << "BackupId: " << hf.BackupId << endl;
+ ndbout << "BackupKey: [ " << hex << hf.BackupKey_0
+ << " "<< hf.BackupKey_1 << " ]" << endl;
+ ndbout << "ByteOrder: " << hex << hf.ByteOrder << endl;
+ return ndbout;
+}
+
+NdbOut & operator<<(NdbOut& ndbout,
+ const BackupFormat::DataFile::FragmentHeader & hf){
+
+ ndbout << "-- Fragment header:" << endl;
+ ndbout << "SectionType: " << hf.SectionType << endl;
+ ndbout << "SectionLength: " << hf.SectionLength << endl;
+ ndbout << "TableId: " << hf.TableId << endl;
+ ndbout << "FragmentNo: " << hf.FragmentNo << endl;
+ ndbout << "ChecksumType: " << hf.ChecksumType << endl;
+
+ return ndbout;
+}
+NdbOut & operator<<(NdbOut& ndbout,
+ const BackupFormat::DataFile::FragmentFooter & hf){
+
+ ndbout << "-- Fragment footer:" << endl;
+ ndbout << "SectionType: " << hf.SectionType << endl;
+ ndbout << "SectionLength: " << hf.SectionLength << endl;
+ ndbout << "TableId: " << hf.TableId << endl;
+ ndbout << "FragmentNo: " << hf.FragmentNo << endl;
+ ndbout << "NoOfRecords: " << hf.NoOfRecords << endl;
+ ndbout << "Checksum: " << hf.Checksum << endl;
+
+ return ndbout;
+}
+
+bool
+readTableList(FILE* f, BackupFormat::CtlFile::TableList **ret){
+ BackupFormat::CtlFile::TableList * dst =
+ (BackupFormat::CtlFile::TableList *)&buf[0];
+
+ if(fread(dst, 4, 2, f) != 2)
+ RETURN_FALSE();
+
+ dst->SectionType = ntohl(dst->SectionType);
+ dst->SectionLength = ntohl(dst->SectionLength);
+
+ if(dst->SectionType != BackupFormat::TABLE_LIST)
+ RETURN_FALSE();
+
+ const Uint32 len = dst->SectionLength - 2;
+ if(fread(&dst->TableIds[0], 4, len, f) != len)
+ RETURN_FALSE();
+
+ for(Uint32 i = 0; i<len; i++){
+ dst->TableIds[i] = ntohl(dst->TableIds[i]);
+ }
+
+ * ret = dst;
+
+ return true;
+}
+
+bool
+readTableDesc(FILE* f, BackupFormat::CtlFile::TableDescription **ret){
+ BackupFormat::CtlFile::TableDescription * dst =
+ (BackupFormat::CtlFile::TableDescription *)&buf[0];
+
+ if(fread(dst, 4, 2, f) != 2)
+ RETURN_FALSE();
+
+ dst->SectionType = ntohl(dst->SectionType);
+ dst->SectionLength = ntohl(dst->SectionLength);
+
+ if(dst->SectionType != BackupFormat::TABLE_DESCRIPTION)
+ RETURN_FALSE();
+
+ const Uint32 len = dst->SectionLength - 2;
+ if(fread(&dst->DictTabInfo[0], 4, len, f) != len)
+ RETURN_FALSE();
+
+ * ret = dst;
+
+ return true;
+}
+
+bool
+readGCPEntry(FILE* f, BackupFormat::CtlFile::GCPEntry **ret){
+ BackupFormat::CtlFile::GCPEntry * dst =
+ (BackupFormat::CtlFile::GCPEntry *)&buf[0];
+
+ if(fread(dst, 4, 4, f) != 4)
+ RETURN_FALSE();
+
+ dst->SectionType = ntohl(dst->SectionType);
+ dst->SectionLength = ntohl(dst->SectionLength);
+
+ if(dst->SectionType != BackupFormat::GCP_ENTRY)
+ RETURN_FALSE();
+
+ dst->StartGCP = ntohl(dst->StartGCP);
+ dst->StopGCP = ntohl(dst->StopGCP);
+
+ * ret = dst;
+
+ return true;
+}
+
+
+NdbOut &
+operator<<(NdbOut& ndbout, const BackupFormat::CtlFile::TableList & hf) {
+ ndbout << "-- Table List:" << endl;
+ ndbout << "SectionType: " << hf.SectionType << endl;
+ ndbout << "SectionLength: " << hf.SectionLength << endl;
+ for(Uint32 i = 0; i < hf.SectionLength - 2; i++){
+ ndbout << hf.TableIds[i] << " ";
+ if((i + 1) % 16 == 0)
+ ndbout << endl;
+ }
+ return ndbout;
+}
+
+NdbOut &
+operator<<(NdbOut& ndbout, const BackupFormat::CtlFile::TableDescription & hf){
+ ndbout << "-- Table Description:" << endl;
+ ndbout << "SectionType: " << hf.SectionType << endl;
+ ndbout << "SectionLength: " << hf.SectionLength << endl;
+
+ SimplePropertiesLinearReader it(&hf.DictTabInfo[0], hf.SectionLength - 2);
+ char buf[1024];
+ for(it.first(); it.valid(); it.next()){
+ switch(it.getValueType()){
+ case SimpleProperties::Uint32Value:
+ ndbout << "Key: " << it.getKey()
+ << " value(" << it.getValueLen() << ") : "
+ << it.getUint32() << endl;
+ break;
+ case SimpleProperties::StringValue:
+ if(it.getValueLen() < sizeof(buf)){
+ it.getString(buf);
+ ndbout << "Key: " << it.getKey()
+ << " value(" << it.getValueLen() << ") : "
+ << "\"" << buf << "\"" << endl;
+ } else {
+ ndbout << "Key: " << it.getKey()
+ << " value(" << it.getValueLen() << ") : "
+ << "\"" << "<TOO LONG>" << "\"" << endl;
+
+ }
+ break;
+ default:
+ ndbout << "Unknown type for key: " << it.getKey()
+ << " type: " << it.getValueType() << endl;
+ }
+ }
+
+ return ndbout;
+}
+
+NdbOut &
+operator<<(NdbOut& ndbout, const BackupFormat::CtlFile::GCPEntry & hf) {
+ ndbout << "-- GCP Entry:" << endl;
+ ndbout << "SectionType: " << hf.SectionType << endl;
+ ndbout << "SectionLength: " << hf.SectionLength << endl;
+ ndbout << "Start GCP: " << hf.StartGCP << endl;
+ ndbout << "Stop GCP: " << hf.StopGCP << endl;
+
+ return ndbout;
+}
+
diff --git a/ndb/src/kernel/blocks/backup/restore/Makefile b/ndb/src/kernel/blocks/backup/restore/Makefile
new file mode 100644
index 00000000000..f99e3e3da0d
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/restore/Makefile
@@ -0,0 +1,20 @@
+include .defs.mk
+
+TYPE := ndbapi ndbapiclient
+
+BIN_TARGET := restore
+BIN_TARGET_LIBS :=
+BIN_TARGET_ARCHIVES := NDB_API general
+
+CCFLAGS_LOC = -I.. -I$(NDB_TOP)/src/ndbapi
+
+#ifneq ($(MYSQLCLUSTER_TOP),)
+#CCFLAGS_LOC +=-I$(MYSQLCLUSTER_TOP)/include -D USE_MYSQL
+#LDFLAGS_LOC += -L$(MYSQLCLUSTER_TOP)/libmysql_r/ -lmysqlclient_r
+#endif
+
+SOURCES = main.cpp Restore.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/src/kernel/blocks/backup/restore/Restore.cpp b/ndb/src/kernel/blocks/backup/restore/Restore.cpp
new file mode 100644
index 00000000000..f91651d9720
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/restore/Restore.cpp
@@ -0,0 +1,1112 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <assert.h>
+#include "Restore.hpp"
+#include "BackupFormat.hpp"
+#include <NdbTCP.h>
+#include <NdbStdio.h>
+#include <OutputStream.hpp>
+#include <Bitmask.hpp>
+
+#include <AttributeHeader.hpp>
+#include <trigger_definitions.h>
+#include <SimpleProperties.hpp>
+#include <signaldata/DictTabInfo.hpp>
+
+// from src/ndbapi
+#include <NdbDictionaryImpl.hpp>
+
+Uint16 Twiddle16(Uint16 in); // Byte shift 16-bit data
+Uint32 Twiddle32(Uint32 in); // Byte shift 32-bit data
+Uint64 Twiddle64(Uint64 in); // Byte shift 64-bit data
+
+bool
+BackupFile::Twiddle(AttributeS* attr, Uint32 arraySize){
+
+ if(m_hostByteOrder)
+ return true;
+
+ if(arraySize == 0){
+ arraySize = attr->Desc->arraySize;
+ }
+
+ switch(attr->Desc->size){
+ case 8:
+
+ return true;
+ case 16:
+ for(unsigned i = 0; i<arraySize; i++){
+ attr->Data.u_int16_value[i] = Twiddle16(attr->Data.u_int16_value[i]);
+ }
+ return true;
+ case 32:
+ for(unsigned i = 0; i<arraySize; i++){
+ attr->Data.u_int32_value[i] = Twiddle32(attr->Data.u_int32_value[i]);
+ }
+ return true;
+ case 64:
+ for(unsigned i = 0; i<arraySize; i++){
+ attr->Data.u_int64_value[i] = Twiddle64(attr->Data.u_int64_value[i]);
+ }
+ return true;
+ default:
+ return false;
+ } // switch
+
+} // Twiddle
+
+FilteredNdbOut err(* new FileOutputStream(stderr), 0, 0);
+FilteredNdbOut info(* new FileOutputStream(stdout), 1, 1);
+FilteredNdbOut debug(* new FileOutputStream(stdout), 2, 0);
+
+// To decide in what byte order data is
+const Uint32 magicByteOrder = 0x12345678;
+const Uint32 swappedMagicByteOrder = 0x78563412;
+
+RestoreMetaData::RestoreMetaData(const char* path, Uint32 nodeId, Uint32 bNo) {
+
+ debug << "RestoreMetaData constructor" << endl;
+ setCtlFile(nodeId, bNo, path);
+}
+
+RestoreMetaData::~RestoreMetaData(){
+ for(int i = 0; i<allTables.size(); i++)
+ delete allTables[i];
+ allTables.clear();
+}
+
+const TableS *
+RestoreMetaData::getTable(Uint32 tableId) const {
+ for(int i = 0; i<allTables.size(); i++)
+ if(allTables[i]->getTableId() == tableId)
+ return allTables[i];
+ return NULL;
+}
+
+Uint32
+RestoreMetaData::getStopGCP() const {
+ return m_stopGCP;
+}
+
+int
+RestoreMetaData::loadContent(const char * catalog,
+ const char * schema)
+{
+
+#if NDB_VERSION_MAJOR >= VERSION_3X
+ if(getMajor(m_fileHeader.NdbVersion) < VERSION_3X) {
+ if(catalog == NULL)
+ return -1;
+ if(schema == NULL)
+ return -1;
+ }
+
+
+ /**
+ * if backup is of version 3 or higher, then
+ * return -2 to indicate for the user that he
+ * cannot restore tables to a certain catalog/schema
+ */
+ if(getMajor(m_fileHeader.NdbVersion) >= VERSION_3X &&
+ (catalog != NULL ||
+ schema != NULL)) {
+ return -2;
+ }
+#endif
+#if NDB_VERSION_MAJOR < VERSION_3X
+ if(getMajor(m_fileHeader.NdbVersion) >= VERSION_3X)
+ {
+ return -2;
+ }
+#endif
+
+ Uint32 noOfTables = readMetaTableList();
+ if(noOfTables == 0)
+ return -3;
+ for(Uint32 i = 0; i<noOfTables; i++){
+ if(!readMetaTableDesc(catalog, schema)){
+ return 0;
+ }
+ }
+ if(!readGCPEntry())
+ return 0;
+ return 1;
+}
+
+Uint32
+RestoreMetaData::readMetaTableList() {
+
+ Uint32 sectionInfo[2];
+
+ if (fread(&sectionInfo, sizeof(sectionInfo), 1, m_file) != 1){
+ return 0;
+ }
+ sectionInfo[0] = ntohl(sectionInfo[0]);
+ sectionInfo[1] = ntohl(sectionInfo[1]);
+
+ const Uint32 tabCount = sectionInfo[1] - 2;
+
+ const Uint32 len = 4 * tabCount;
+ if(createBuffer(len) == 0)
+ abort();
+
+ if (fread(m_buffer, 1, len, m_file) != len){
+ return 0;
+ }
+
+ return tabCount;
+}
+
+bool
+RestoreMetaData::readMetaTableDesc(const char * catalog,
+ const char * schema) {
+
+ Uint32 sectionInfo[2];
+
+ // Read section header
+ if (fread(&sectionInfo, sizeof(sectionInfo), 1, m_file) != 1){
+ err << "readMetaTableDesc read header error" << endl;
+ return false;
+ } // if
+ sectionInfo[0] = ntohl(sectionInfo[0]);
+ sectionInfo[1] = ntohl(sectionInfo[1]);
+
+ assert(sectionInfo[0] == BackupFormat::TABLE_DESCRIPTION);
+
+ // Allocate temporary storage for dictTabInfo buffer
+ const Uint32 len = (sectionInfo[1] - 2);
+ if (createBuffer(4 * (len+1)) == NULL) {
+ err << "readMetaTableDesc allocation error" << endl;
+ return false;
+ } // if
+
+ // Read dictTabInfo buffer
+ if (fread(m_buffer, 4, len, m_file) != len){
+ err << "readMetaTableDesc read error" << endl;
+ return false;
+ } // if
+
+ return parseTableDescriptor(m_buffer,
+ len,
+ catalog,
+ schema);
+}
+
+bool
+RestoreMetaData::readGCPEntry() {
+
+ Uint32 data[4];
+
+
+ BackupFormat::CtlFile::GCPEntry * dst =
+ (BackupFormat::CtlFile::GCPEntry *)&data[0];
+
+ if(fread(dst, 4, 4, m_file) != 4){
+ err << "readGCPEntry read error" << endl;
+ return false;
+ }
+
+ dst->SectionType = ntohl(dst->SectionType);
+ dst->SectionLength = ntohl(dst->SectionLength);
+
+ if(dst->SectionType != BackupFormat::GCP_ENTRY){
+ err << "readGCPEntry invalid format" << endl;
+ return false;
+ }
+
+ dst->StartGCP = ntohl(dst->StartGCP);
+ dst->StopGCP = ntohl(dst->StopGCP);
+
+ m_startGCP = dst->StartGCP;
+ m_stopGCP = dst->StopGCP;
+ return true;
+}
+
+
+struct tmpTableS {
+ Uint32 tableId;
+ Uint32 schemaVersion;
+ Uint32 noOfAttributes;
+}; // tmpTableS
+
+static const
+SimpleProperties::SP2StructMapping
+RestoreTabMap[] = {
+ // Map the basic stuff to begin with
+ DTIMAP(tmpTableS, TableId, tableId),
+ DTIMAP(tmpTableS, TableVersion, schemaVersion),
+ DTIMAP(tmpTableS, NoOfAttributes, noOfAttributes),
+
+ DTIBREAK(AttributeName)
+}; // RestoreTabMap
+
+static const Uint32
+TabMapSize = sizeof(RestoreTabMap)
+ / sizeof(SimpleProperties::SP2StructMapping);
+
+/**
+ * Use a temporary struct to keep variables in AttributeDesc private
+ * and DTIMAP requires all Uint32
+ */
+struct tmpAttrS {
+ // Just the basic needed stuff is yet implemented
+ char name[AttrNameLenC];
+ Uint32 attrId;
+ Uint32 type;
+ Uint32 nullable;
+ Uint32 key;
+ Uint32 size;
+ Uint32 arraySize;
+};
+
+static const
+SimpleProperties::SP2StructMapping
+RestoreAttrMap[] = {
+ // Map the most basic properties
+ DTIMAP(tmpAttrS, AttributeId, attrId),
+ DTIMAP(tmpAttrS, AttributeType, type),
+ DTIMAP(tmpAttrS, AttributeNullableFlag, nullable),
+ DTIMAP(tmpAttrS, AttributeKeyFlag, key),
+ DTIMAP(tmpAttrS, AttributeSize, size),
+ DTIMAP(tmpAttrS, AttributeArraySize, arraySize),
+ DTIBREAK(AttributeEnd)
+}; // RestoreAttrMap
+static const Uint32
+AttrMapSize = sizeof(RestoreAttrMap)
+ / sizeof(SimpleProperties::SP2StructMapping);
+
+struct v2xKernel_to_v3xAPIMapping {
+ Int32 kernelConstant;
+ Int32 apiConstant;
+};
+
+enum v2xKernelTypes {
+ ExtUndefined=0,// Undefined
+ ExtInt, // 32 bit
+ ExtUnsigned, // 32 bit
+ ExtBigint, // 64 bit
+ ExtBigunsigned,// 64 Bit
+ ExtFloat, // 32-bit float
+ ExtDouble, // 64-bit float
+ ExtDecimal, // Precision, Scale
+ ExtChar, // Len
+ ExtVarchar, // Max len
+ ExtBinary, // Len
+ ExtVarbinary, // Max len
+ ExtDatetime, // Precision down to 1 sec (sizeof(Datetime) == 8 bytes )
+ ExtTimespec // Precision down to 1 nsec (sizeof(Datetime) == 12 bytes )
+};
+
+const
+v2xKernel_to_v3xAPIMapping
+columnTypeMapping[] = {
+ { ExtInt, NdbDictionary::Column::Int },
+ { ExtUnsigned, NdbDictionary::Column::Unsigned },
+ { ExtBigint, NdbDictionary::Column::Bigint },
+ { ExtBigunsigned, NdbDictionary::Column::Bigunsigned },
+ { ExtFloat, NdbDictionary::Column::Float },
+ { ExtDouble, NdbDictionary::Column::Double },
+ { ExtDecimal, NdbDictionary::Column::Decimal },
+ { ExtChar, NdbDictionary::Column::Char },
+ { ExtVarchar, NdbDictionary::Column::Varchar },
+ { ExtBinary, NdbDictionary::Column::Binary },
+ { ExtVarbinary, NdbDictionary::Column::Varbinary },
+ { ExtDatetime, NdbDictionary::Column::Datetime },
+ { ExtTimespec, NdbDictionary::Column::Timespec },
+ { -1, -1 }
+};
+
+static
+NdbDictionary::Column::Type
+convertToV3x(Int32 kernelConstant, const v2xKernel_to_v3xAPIMapping map[],
+ Int32 def)
+{
+ int i = 0;
+ while(map[i].kernelConstant != kernelConstant){
+ if(map[i].kernelConstant == -1 &&
+ map[i].apiConstant == -1){
+ return (NdbDictionary::Column::Type)def;
+ }
+ i++;
+ }
+ return (NdbDictionary::Column::Type)map[i].apiConstant;
+}
+
+
+
+// Parse dictTabInfo buffer and pushback to to vector storage
+// Using SimpleProperties (here we don't need ntohl, ref:ejonore)
+bool
+RestoreMetaData::parseTableDescriptor(const Uint32 * data,
+ Uint32 len,
+ const char * catalog,
+ const char * schema) {
+ SimplePropertiesLinearReader it(data, len);
+ SimpleProperties::UnpackStatus spStatus;
+
+ // Parse table name
+ if (it.getKey() != DictTabInfo::TableName) {
+ err << "readMetaTableDesc getKey table name error" << endl;
+ return false;
+ } // if
+
+ /**
+ * if backup was taken in v21x then there is no info about catalog,
+ * and schema. This infomration is concatenated to the tableName.
+ *
+ */
+ char tableName[MAX_TAB_NAME_SIZE*2]; // * 2 for db and schema.-.
+
+
+ char tmpTableName[MAX_TAB_NAME_SIZE];
+ it.getString(tmpTableName);
+#if NDB_VERSION_MAJOR >= VERSION_3X
+ /**
+ * only mess with name in version 3.
+ */
+ /* switch(getMajor(m_fileHeader.NdbVersion)) {
+ */
+ if(getMajor(m_fileHeader.NdbVersion) < VERSION_3X)
+ {
+
+ if(strcmp(tmpTableName, "SYSTAB_0") == 0 ||
+ strcmp(tmpTableName, "NDB$EVENTS_0") == 0)
+ {
+ sprintf(tableName,"sys/def/%s",tmpTableName);
+ }
+ else {
+ if(catalog == NULL && schema == NULL)
+ {
+ sprintf(tableName,"%s",tmpTableName);
+ }
+ else
+ {
+ sprintf(tableName,"%s/%s/%s",catalog,schema,tmpTableName);
+ }
+ }
+ }
+ else
+ sprintf(tableName,"%s",tmpTableName);
+#elif NDB_VERSION_MAJOR < VERSION_3X
+ /**
+ * this is version two!
+ */
+ sprintf(tableName,"%s",tmpTableName);
+#endif
+ if (strlen(tableName) == 0) {
+ err << "readMetaTableDesc getString table name error" << endl;
+ return false;
+ } // if
+
+ TableS * table = new TableS(tableName);
+ if(table == NULL) {
+ return false;
+ }
+
+ table->setBackupVersion(m_fileHeader.NdbVersion);
+ tmpTableS tmpTable;
+ spStatus = SimpleProperties::unpack(it, &tmpTable,
+ RestoreTabMap, TabMapSize, true, true);
+ if ((spStatus != SimpleProperties::Break) ||
+ it.getKey() != DictTabInfo::AttributeName) {
+ err << "readMetaTableDesc sp.unpack error" << endl;
+ delete table;
+ return false;
+ } // if
+
+ debug << "Parsed table id " << tmpTable.tableId << endl;
+ table->setTableId(tmpTable.tableId);
+ debug << "Parsed table #attr " << tmpTable.noOfAttributes << endl;
+ debug << "Parsed table schema version not used " << endl;
+
+ for (Uint32 i = 0; i < tmpTable.noOfAttributes; i++) {
+ if (it.getKey() != DictTabInfo::AttributeName) {
+ err << "readMetaTableDesc error " << endl;
+ delete table;
+ return false;
+ } // if
+
+ tmpAttrS tmpAttr;
+ if(it.getValueLen() > AttrNameLenC){
+ err << "readMetaTableDesc attribute name too long??" << endl;
+ delete table;
+ return false;
+ }
+ it.getString(tmpAttr.name);
+
+ spStatus = SimpleProperties::unpack(it, &tmpAttr, RestoreAttrMap,
+ AttrMapSize, true, true);
+ if ((spStatus != SimpleProperties::Break) ||
+ (it.getKey() != DictTabInfo::AttributeEnd)) {
+ err << "readMetaTableDesc sp unpack attribute " << i << " error"
+ << endl;
+ delete table;
+ return false;
+ } // if
+
+ debug << "Creating attribute " << i << " " << tmpAttr.name << endl;
+
+ bool thisNullable = (bool)(tmpAttr.nullable); // Really not needed (now)
+ KeyType thisKey = (KeyType)(tmpAttr.key); // These are identical (right now)
+ // Convert attribute size from enum to Uint32
+ // The static consts are really enum taking the value in DictTabInfo
+ // e.g. 3 is not ...0011 but rather ...0100
+ //TODO: rather do a switch if the constants should change
+ Uint32 thisSize = 1 << tmpAttr.size;
+ // Convert attribute type to AttrType
+ AttrType thisType;
+ switch (tmpAttr.type) {
+ case 0: // SignedType
+ thisType = Signed;
+ break;
+ case 1: // UnSignedType
+ thisType = UnSigned;
+ break;
+ case 2: // FloatingPointType
+ thisType = Float;
+ break;
+ case 3: // StringType:
+ debug << "String type detected " << endl;
+ thisType = String;
+ break;
+ default:
+ // What, default to unsigned?
+ thisType = UnSigned;
+ break;
+ } // switch
+ /* ndbout_c << " type: " << thisType << " size: " << thisSize <<" arraySize: "
+ << tmpAttr.arraySize << " nullable: " << thisNullable << " key: "
+ << thisKey << endl;
+ */
+ table->createAttr(tmpAttr.name, thisType,
+ thisSize, tmpAttr.arraySize,
+ thisNullable, thisKey);
+ if (!it.next()) {
+ break;
+ // Check number of created attributes and compare with expected
+ //ndbout << "readMetaTableDesc expecting more attributes" << endl;
+ //return false;
+ } // if
+ } // for
+
+ debug << "Pushing table " << tableName << endl;
+ debug << " with " << table->getNoOfAttributes() << " attributes" << endl;
+ allTables.push_back(table);
+
+#ifndef restore_old_types
+ NdbTableImpl* tableImpl = 0;
+ int ret = NdbDictInterface::parseTableInfo(&tableImpl, data, len);
+#if NDB_VERSION_MAJOR >= VERSION_3X
+ NdbDictionary::Column::Type type;
+ if(getMajor(m_fileHeader.NdbVersion) < VERSION_3X) {
+ tableImpl->setName(tableName);
+ for(Uint32 i = 0 ; i < tableImpl->getNoOfColumns(); i++) {
+ type = convertToV3x(tableImpl->getColumn(i)->m_extType,
+ columnTypeMapping,
+ -1);
+ if(type == -1)
+ {
+ ndbout_c("Restore: Was not able to map external type %d (in v2x) "
+ " to a proper type in v3x", tableImpl->getColumn(i)->m_extType);
+ return false;
+ }
+ else
+ {
+ tableImpl->getColumn(i)->m_type = type;
+ }
+
+
+
+
+ }
+ }
+#endif
+ if (ret != 0) {
+ err << "parseTableInfo " << tableName << " failed" << endl;
+ return false;
+ }
+ if(tableImpl == 0)
+ return false;
+ debug << "parseTableInfo " << tableName << " done" << endl;
+ table->m_dictTable = tableImpl;
+#endif
+ return true;
+}
+
+// Constructor
+RestoreDataIterator::RestoreDataIterator(const RestoreMetaData & md)
+ : m_metaData(md)
+{
+ debug << "RestoreDataIterator constructor" << endl;
+ setDataFile(md, 0);
+}
+
+RestoreDataIterator::~RestoreDataIterator(){
+}
+
+bool
+TupleS::prepareRecord(const TableS & tab){
+ m_currentTable = &tab;
+ for(int i = 0; i<allAttributes.size(); i++) {
+ if(!allAttributes[i] == NULL)
+ delete allAttributes[i];
+ }
+ allAttributes.clear();
+ AttributeS * a;
+ for(int i = 0; i<tab.getNoOfAttributes(); i++){
+ a = new AttributeS;
+ if(a == NULL) {
+ ndbout_c("Restore: Failed to allocate memory");
+ return false;
+ }
+ a->Desc = tab[i];
+ allAttributes.push_back(a);
+ }
+ return true;
+}
+
+const TupleS *
+RestoreDataIterator::getNextTuple(int & res) {
+ TupleS * tup = new TupleS();
+ if(tup == NULL) {
+ ndbout_c("Restore: Failed to allocate memory");
+ res = -1;
+ return NULL;
+ }
+ if(!tup->prepareRecord(* m_currentTable)) {
+ res =-1;
+ return NULL;
+ }
+
+
+ Uint32 dataLength = 0;
+ // Read record length
+ if (fread(&dataLength, sizeof(dataLength), 1, m_file) != 1){
+ err << "getNextTuple:Error reading length of data part" << endl;
+ delete tup;
+ res = -1;
+ return NULL;
+ } // if
+
+ // Convert length from network byte order
+ dataLength = ntohl(dataLength);
+ const Uint32 dataLenBytes = 4 * dataLength;
+
+ if (dataLength == 0) {
+ // Zero length for last tuple
+ // End of this data fragment
+ debug << "End of fragment" << endl;
+ res = 0;
+ delete tup;
+ return NULL;
+ } // if
+
+ tup->createDataRecord(dataLenBytes);
+ // Read tuple data
+ if (fread(tup->getDataRecord(), 1, dataLenBytes, m_file) != dataLenBytes) {
+ err << "getNextTuple:Read error: " << endl;
+ delete tup;
+ res = -1;
+ return NULL;
+ }
+
+ Uint32 * ptr = tup->getDataRecord();
+ ptr += m_currentTable->m_nullBitmaskSize;
+
+ for(int i = 0; i < m_currentTable->m_fixedKeys.size(); i++){
+ assert(ptr < tup->getDataRecord() + dataLength);
+
+ const Uint32 attrId = m_currentTable->m_fixedKeys[i]->attrId;
+ AttributeS * attr = tup->allAttributes[attrId];
+
+ const Uint32 sz = attr->Desc->getSizeInWords();
+
+ attr->Data.null = false;
+ attr->Data.void_value = ptr;
+
+ if(!Twiddle(attr))
+ {
+ res = -1;
+ return NULL;
+ }
+ ptr += sz;
+ }
+
+ for(int i = 0; i<m_currentTable->m_fixedAttribs.size(); i++){
+ assert(ptr < tup->getDataRecord() + dataLength);
+
+ const Uint32 attrId = m_currentTable->m_fixedAttribs[i]->attrId;
+ AttributeS * attr = tup->allAttributes[attrId];
+
+ const Uint32 sz = attr->Desc->getSizeInWords();
+
+ attr->Data.null = false;
+ attr->Data.void_value = ptr;
+
+ if(!Twiddle(attr))
+ {
+ res = -1;
+ return NULL;
+ }
+
+ ptr += sz;
+ }
+
+ for(int i = 0; i<m_currentTable->m_variableAttribs.size(); i++){
+ const Uint32 attrId = m_currentTable->m_variableAttribs[i]->attrId;
+ AttributeS * attr = tup->allAttributes[attrId];
+
+ if(attr->Desc->nullable){
+ const Uint32 ind = attr->Desc->m_nullBitIndex;
+ if(BitmaskImpl::get(m_currentTable->m_nullBitmaskSize,
+ tup->getDataRecord(),ind)){
+ attr->Data.null = true;
+ attr->Data.void_value = NULL;
+ continue;
+ }
+ }
+
+ assert(ptr < tup->getDataRecord() + dataLength);
+
+ typedef BackupFormat::DataFile::VariableData VarData;
+ VarData * data = (VarData *)ptr;
+ Uint32 sz = ntohl(data->Sz);
+ Uint32 id = ntohl(data->Id);
+ assert(id == attrId);
+
+ attr->Data.null = false;
+ attr->Data.void_value = &data->Data[0];
+
+ /**
+ * Compute array size
+ */
+ const Uint32 arraySize = (4 * sz) / (attr->Desc->size / 8);
+ assert(arraySize >= attr->Desc->arraySize);
+ if(!Twiddle(attr, attr->Desc->arraySize))
+ {
+ res = -1;
+ return NULL;
+ }
+
+ ptr += (sz + 2);
+ }
+
+ m_count ++;
+ res = 0;
+ return tup;
+} // RestoreDataIterator::getNextTuple
+
+BackupFile::BackupFile(){
+ m_file = 0;
+ m_path[0] = 0;
+ m_fileName[0] = 0;
+ m_buffer = 0;
+ m_bufferSize = 0;
+}
+
+BackupFile::~BackupFile(){
+ if(m_file != 0)
+ fclose(m_file);
+ if(m_buffer != 0)
+ free(m_buffer);
+}
+
+bool
+BackupFile::openFile(){
+ if(m_file != NULL){
+ fclose(m_file);
+ m_file = 0;
+ }
+
+ m_file = fopen(m_fileName, "r");
+ return m_file != 0;
+}
+
+Uint32 *
+BackupFile::createBuffer(Uint32 bytes){
+ if(bytes > m_bufferSize){
+ if(m_buffer != 0)
+ free(m_buffer);
+ m_bufferSize = m_bufferSize + 2 * bytes;
+ m_buffer = (Uint32*)malloc(m_bufferSize);
+ }
+ return m_buffer;
+}
+
+void
+BackupFile::setCtlFile(Uint32 nodeId, Uint32 backupId, const char * path){
+ m_nodeId = nodeId;
+ m_expectedFileHeader.BackupId = backupId;
+ m_expectedFileHeader.FileType = BackupFormat::CTL_FILE;
+
+ char name[PATH_MAX]; const Uint32 sz = sizeof(name);
+ snprintf(name, sz, "BACKUP-%d.%d.ctl", backupId, nodeId);
+ setName(path, name);
+}
+
+void
+BackupFile::setDataFile(const BackupFile & bf, Uint32 no){
+ m_nodeId = bf.m_nodeId;
+ m_expectedFileHeader = bf.m_fileHeader;
+ m_expectedFileHeader.FileType = BackupFormat::DATA_FILE;
+
+ char name[PATH_MAX]; const Uint32 sz = sizeof(name);
+ snprintf(name, sz, "BACKUP-%d-%d.%d.Data",
+ m_expectedFileHeader.BackupId, no, m_nodeId);
+ setName(bf.m_path, name);
+}
+
+void
+BackupFile::setLogFile(const BackupFile & bf, Uint32 no){
+ m_nodeId = bf.m_nodeId;
+ m_expectedFileHeader = bf.m_fileHeader;
+ m_expectedFileHeader.FileType = BackupFormat::LOG_FILE;
+
+ char name[PATH_MAX]; const Uint32 sz = sizeof(name);
+ snprintf(name, sz, "BACKUP-%d.%d.log",
+ m_expectedFileHeader.BackupId, m_nodeId);
+ setName(bf.m_path, name);
+}
+
+void
+BackupFile::setName(const char * p, const char * n){
+ const Uint32 sz = sizeof(m_path);
+ if(p != 0 && strlen(p) > 0){
+ if(p[strlen(p)-1] == '/'){
+ snprintf(m_path, sz, "%s", p);
+ } else {
+ snprintf(m_path, sz, "%s%s", p, "/");
+ }
+ } else {
+ m_path[0] = 0;
+ }
+
+ snprintf(m_fileName, sizeof(m_fileName), "%s%s", m_path, n);
+ debug << "Filename = " << m_fileName << endl;
+}
+
+bool
+BackupFile::readHeader(){
+ if(!openFile()){
+ return false;
+ }
+
+ if(fread(&m_fileHeader, sizeof(m_fileHeader), 1, m_file) != 1){
+ err << "readDataFileHeader: Error reading header" << endl;
+ return false;
+ }
+
+ // Convert from network to host byte order for platform compatibility
+ m_fileHeader.NdbVersion = ntohl(m_fileHeader.NdbVersion);
+ m_fileHeader.SectionType = ntohl(m_fileHeader.SectionType);
+ m_fileHeader.SectionLength = ntohl(m_fileHeader.SectionLength);
+ m_fileHeader.FileType = ntohl(m_fileHeader.FileType);
+ m_fileHeader.BackupId = ntohl(m_fileHeader.BackupId);
+ m_fileHeader.BackupKey_0 = ntohl(m_fileHeader.BackupKey_0);
+ m_fileHeader.BackupKey_1 = ntohl(m_fileHeader.BackupKey_1);
+
+ debug << "FileHeader: " << m_fileHeader.Magic << " " <<
+ m_fileHeader.NdbVersion << " " <<
+ m_fileHeader.SectionType << " " <<
+ m_fileHeader.SectionLength << " " <<
+ m_fileHeader.FileType << " " <<
+ m_fileHeader.BackupId << " " <<
+ m_fileHeader.BackupKey_0 << " " <<
+ m_fileHeader.BackupKey_1 << " " <<
+ m_fileHeader.ByteOrder << endl;
+
+ debug << "ByteOrder is " << m_fileHeader.ByteOrder << endl;
+ debug << "magicByteOrder is " << magicByteOrder << endl;
+
+ if (m_fileHeader.FileType != m_expectedFileHeader.FileType){
+ abort();
+ }
+
+ // Check for BackupFormat::FileHeader::ByteOrder if swapping is needed
+ if (m_fileHeader.ByteOrder == magicByteOrder) {
+ m_hostByteOrder = true;
+ } else if (m_fileHeader.ByteOrder == swappedMagicByteOrder){
+ m_hostByteOrder = false;
+ } else {
+ abort();
+ }
+
+ return true;
+} // BackupFile::readHeader
+
+bool
+BackupFile::validateFooter(){
+ return true;
+}
+
+bool
+RestoreDataIterator::readFragmentHeader(int & ret)
+{
+ BackupFormat::DataFile::FragmentHeader Header;
+
+ debug << "RestoreDataIterator::getNextFragment" << endl;
+
+ if (fread(&Header, sizeof(Header), 1, m_file) != 1){
+ ret = 0;
+ return false;
+ } // if
+
+ Header.SectionType = ntohl(Header.SectionType);
+ Header.SectionLength = ntohl(Header.SectionLength);
+ Header.TableId = ntohl(Header.TableId);
+ Header.FragmentNo = ntohl(Header.FragmentNo);
+ Header.ChecksumType = ntohl(Header.ChecksumType);
+
+ debug << "FragmentHeader: " << Header.SectionType
+ << " " << Header.SectionLength
+ << " " << Header.TableId
+ << " " << Header.FragmentNo
+ << " " << Header.ChecksumType << endl;
+
+ m_currentTable = m_metaData.getTable(Header.TableId);
+ if(m_currentTable == 0){
+ ret = -1;
+ return false;
+ }
+
+ info << "_____________________________________________________" << endl
+ << "Restoring data in table: " << m_currentTable->getTableName()
+ << "(" << Header.TableId << ") fragment "
+ << Header.FragmentNo << endl;
+
+ m_count = 0;
+ ret = 0;
+ return true;
+} // RestoreDataIterator::getNextFragment
+
+
+bool
+RestoreDataIterator::validateFragmentFooter() {
+ BackupFormat::DataFile::FragmentFooter footer;
+
+ if (fread(&footer, sizeof(footer), 1, m_file) != 1){
+ err << "getFragmentFooter:Error reading fragment footer" << endl;
+ return false;
+ }
+
+ // TODO: Handle footer, nothing yet
+ footer.SectionType = ntohl(footer.SectionType);
+ footer.SectionLength = ntohl(footer.SectionLength);
+ footer.TableId = ntohl(footer.TableId);
+ footer.FragmentNo = ntohl(footer.FragmentNo);
+ footer.NoOfRecords = ntohl(footer.NoOfRecords);
+ footer.Checksum = ntohl(footer.Checksum);
+
+ assert(m_count == footer.NoOfRecords);
+
+ return true;
+} // RestoreDataIterator::getFragmentFooter
+
+void TableS::createAttr(const char* name,
+ const AttrType type,
+ const unsigned int size, // in bytes
+ const unsigned int arraySize,
+ const bool nullable,
+ const KeyType key)
+{
+ AttributeDesc desc;
+
+ strncpy(desc.name, name, AttrNameLenC);
+ desc.type = type;
+ desc.size = size;
+ desc.arraySize = arraySize;
+ desc.nullable = nullable;
+ desc.key = key;
+ desc.attrId = allAttributesDesc.size();
+
+ AttributeDesc * d = new AttributeDesc(desc);
+ if(d == NULL) {
+ ndbout_c("Restore: Failed to allocate memory");
+ abort();
+ }
+ allAttributesDesc.push_back(d);
+
+ if(desc.key != NoKey /* && not variable */){
+ m_fixedKeys.push_back(d);
+ return;
+ }
+ if(!nullable){
+ m_fixedAttribs.push_back(d);
+ return;
+ }
+ if(nullable){
+ d->m_nullBitIndex = m_noOfNullable;
+ m_noOfNullable++;
+ m_nullBitmaskSize = (m_noOfNullable + 31) / 32;
+ }
+ m_variableAttribs.push_back(d);
+} // TableS::createAttr
+
+Uint16 Twiddle16(Uint16 in)
+{
+ Uint16 retVal = 0;
+
+ retVal = ((in & 0xFF00) >> 8) |
+ ((in & 0x00FF) << 8);
+
+ return(retVal);
+} // Twiddle16
+
+Uint32 Twiddle32(Uint32 in)
+{
+ Uint32 retVal = 0;
+
+ retVal = ((in & 0x000000FF) << 24) |
+ ((in & 0x0000FF00) << 8) |
+ ((in & 0x00FF0000) >> 8) |
+ ((in & 0xFF000000) >> 24);
+
+ return(retVal);
+} // Twiddle32
+
+Uint64 Twiddle64(Uint64 in)
+{
+ Uint64 retVal = 0;
+
+ retVal =
+ ((in & (Uint64)0x00000000000000FFLL) << 56) |
+ ((in & (Uint64)0x000000000000FF00LL) << 40) |
+ ((in & (Uint64)0x0000000000FF0000LL) << 24) |
+ ((in & (Uint64)0x00000000FF000000LL) << 8) |
+ ((in & (Uint64)0x000000FF00000000LL) >> 8) |
+ ((in & (Uint64)0x0000FF0000000000LL) >> 24) |
+ ((in & (Uint64)0x00FF000000000000LL) >> 40) |
+ ((in & (Uint64)0xFF00000000000000LL) >> 56);
+
+ return(retVal);
+} // Twiddle64
+
+
+RestoreLogIterator::RestoreLogIterator(const RestoreMetaData & md)
+ : m_metaData(md)
+{
+ debug << "RestoreLog constructor" << endl;
+ setLogFile(md, 0);
+
+ m_count = 0;
+}
+
+const LogEntry *
+RestoreLogIterator::getNextLogEntry(int & res) {
+ // Read record length
+ typedef BackupFormat::LogFile::LogEntry LogE;
+
+ Uint32 gcp = 0;
+ LogE * logE = 0;
+ Uint32 len = ~0;
+ const Uint32 stopGCP = m_metaData.getStopGCP();
+ do {
+
+ if(createBuffer(4) == 0) {
+ res = -1;
+ return NULL;
+ }
+
+
+ if (fread(m_buffer, sizeof(Uint32), 1, m_file) != 1){
+ res = -1;
+ return NULL;
+ }
+
+ m_buffer[0] = ntohl(m_buffer[0]);
+ len = m_buffer[0];
+ if(len == 0){
+ res = 0;
+ return 0;
+ }
+
+ if(createBuffer(4 * (len + 1)) == 0){
+ res = -1;
+ return NULL;
+ }
+
+ if (fread(&m_buffer[1], 4, len, m_file) != len) {
+ res = -1;
+ return NULL;
+ }
+
+ logE = (LogE *)&m_buffer[0];
+ logE->TableId = ntohl(logE->TableId);
+ logE->TriggerEvent = ntohl(logE->TriggerEvent);
+
+ const bool hasGcp = (logE->TriggerEvent & 0x10000) != 0;
+ logE->TriggerEvent &= 0xFFFF;
+
+ if(hasGcp){
+ len--;
+ gcp = ntohl(logE->Data[len-2]);
+ }
+ } while(gcp > stopGCP + 1);
+
+ for(int i=0; i<m_logEntry.m_values.size();i++)
+ delete m_logEntry.m_values[i];
+ m_logEntry.m_values.clear();
+ m_logEntry.m_table = m_metaData.getTable(logE->TableId);
+ switch(logE->TriggerEvent){
+ case TriggerEvent::TE_INSERT:
+ m_logEntry.m_type = LogEntry::LE_INSERT;
+ break;
+ case TriggerEvent::TE_UPDATE:
+ m_logEntry.m_type = LogEntry::LE_UPDATE;
+ break;
+ case TriggerEvent::TE_DELETE:
+ m_logEntry.m_type = LogEntry::LE_DELETE;
+ break;
+ default:
+ res = -1;
+ return NULL;
+ }
+
+ const TableS * tab = m_logEntry.m_table;
+
+ AttributeHeader * ah = (AttributeHeader *)&logE->Data[0];
+ AttributeHeader *end = (AttributeHeader *)&logE->Data[len - 2];
+ AttributeS * attr;
+ while(ah < end){
+ attr = new AttributeS;
+ if(attr == NULL) {
+ ndbout_c("Restore: Failed to allocate memory");
+ res = -1;
+ return NULL;
+ }
+ attr->Desc = (* tab)[ah->getAttributeId()];
+ assert(attr->Desc != 0);
+
+ const Uint32 sz = ah->getDataSize();
+ if(sz == 0){
+ attr->Data.null = true;
+ attr->Data.void_value = NULL;
+ } else {
+ attr->Data.null = false;
+ attr->Data.void_value = ah->getDataPtr();
+ }
+
+ Twiddle(attr);
+ m_logEntry.m_values.push_back(attr);
+
+ ah = ah->getNext();
+ }
+
+ m_count ++;
+ res = 0;
+ return &m_logEntry;
+}
diff --git a/ndb/src/kernel/blocks/backup/restore/Restore.hpp b/ndb/src/kernel/blocks/backup/restore/Restore.hpp
new file mode 100644
index 00000000000..f214bcb1380
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/restore/Restore.hpp
@@ -0,0 +1,328 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef RESTORE_H
+#define RESTORE_H
+
+#include <stddef.h>
+#include <BackupFormat.hpp>
+#include <NdbApi.hpp>
+#include <AttrType.hpp>
+
+#include <NdbOut.hpp>
+#include "myVector.hpp"
+#include <NdbStdio.h>
+#include <NdbUnistd.h>
+#include <NdbString.h>
+
+#include <ndb_version.h>
+#include <version.h>
+
+#define VERSION_3X 3
+
+
+const int FileNameLenC = 256;
+const int TableNameLenC = 256;
+const int AttrNameLenC = 256;
+const Uint32 timeToWaitForNdbC = 10000;
+const Uint32 opsDefaultC = 1000;
+
+// Forward declarations
+//class AttributeDesc;
+struct AttributeDesc;
+struct AttributeData;
+struct AttributeS;
+
+struct AttributeData {
+ bool null;
+ Uint32 size;
+ union {
+ Int8 * int8_value;
+ Uint8 * u_int8_value;
+
+ Int16 * int16_value;
+ Uint16 * u_int16_value;
+
+ Int32 * int32_value;
+ Uint32 * u_int32_value;
+
+ Int64 * int64_value;
+ Uint64 * u_int64_value;
+
+ char * string_value;
+
+ void* void_value;
+ };
+};
+
+struct AttributeDesc {
+ //private:
+ // TODO (sometimes): use a temporary variable in DTIMAP so we can
+ // hide AttributeDesc private variables
+ friend class TupleS;
+ friend class TableS;
+ friend class RestoreDataIterator;
+ friend class RestoreMetaData;
+ friend struct AttributeS;
+ char name[AttrNameLenC];
+ Uint32 attrId;
+ AttrType type;
+ bool nullable;
+ KeyType key;
+ Uint32 size; // bits
+ Uint32 arraySize;
+
+ Uint32 m_nullBitIndex;
+public:
+
+ AttributeDesc() {
+ name[0] = 0;
+ }
+
+ Uint32 getSizeInWords() const { return (size * arraySize + 31)/ 32;}
+}; // AttributeDesc
+
+struct AttributeS {
+ const AttributeDesc * Desc;
+ AttributeData Data;
+};
+
+class TupleS {
+private:
+ friend class RestoreDataIterator;
+
+ const TableS * m_currentTable;
+ myVector<AttributeS*> allAttributes;
+ Uint32 * dataRecord;
+ bool prepareRecord(const TableS &);
+
+public:
+ TupleS() {dataRecord = NULL;};
+ ~TupleS() {if(dataRecord != NULL) delete [] dataRecord;};
+ int getNoOfAttributes() const { return allAttributes.size(); };
+ const TableS * getTable() const { return m_currentTable;};
+ const AttributeS * operator[](int i) const { return allAttributes[i];};
+ Uint32 * getDataRecord() { return dataRecord;};
+ void createDataRecord(Uint32 bytes) { dataRecord = new Uint32[bytes];};
+}; // class TupleS
+
+class TableS {
+
+ friend class TupleS;
+ friend class RestoreMetaData;
+ friend class RestoreDataIterator;
+
+ Uint32 tableId;
+ char tableName[TableNameLenC];
+ Uint32 schemaVersion;
+ Uint32 backupVersion;
+ myVector<AttributeDesc *> allAttributesDesc;
+ myVector<AttributeDesc *> m_fixedKeys;
+ //myVector<AttributeDesc *> m_variableKey;
+ myVector<AttributeDesc *> m_fixedAttribs;
+ myVector<AttributeDesc *> m_variableAttribs;
+
+ Uint32 m_noOfNullable;
+ Uint32 m_nullBitmaskSize;
+
+ int pos;
+ char create_string[2048];
+ /*
+ char mysqlTableName[1024];
+ char mysqlDatabaseName[1024];
+ */
+
+ void createAttr(const char* name,
+ const AttrType type,
+ const unsigned int size, // in bits
+ const unsigned int arraySize,
+ const bool nullable,
+ const KeyType key);
+
+#ifndef restore_old_types
+public:
+ class NdbDictionary::Table* m_dictTable;
+#endif
+public:
+ TableS (const char * name){
+ snprintf(tableName, sizeof(tableName), name);
+ m_noOfNullable = m_nullBitmaskSize = 0;
+ }
+
+ void setTableId (Uint32 id) {
+ tableId = id;
+ }
+
+ Uint32 getTableId() const {
+ return tableId;
+ }
+ /*
+ void setMysqlTableName(char * tableName) {
+ strpcpy(mysqlTableName, tableName);
+ }
+
+ char *
+ void setMysqlDatabaseName(char * databaseName) {
+ strpcpy(mysqlDatabaseName, databaseName);
+ }
+
+ table.setMysqlDatabaseName(database);
+ */
+ void setBackupVersion(Uint32 version) {
+ backupVersion = version;
+ }
+
+
+ Uint32 getBackupVersion() const {
+ return backupVersion;
+ }
+
+ const char * getTableName() const {
+ return m_dictTable->getName();
+ }
+
+ int getNoOfAttributes() const {
+ return allAttributesDesc.size();
+ };
+
+ /**
+ * Get attribute descriptor
+ */
+ const AttributeDesc * operator[](int attributeId) const {
+ return allAttributesDesc[attributeId];
+ }
+
+ TableS& operator=(TableS& org) ;
+}; // TableS;
+
+class BackupFile {
+protected:
+ FILE * m_file;
+ char m_path[PATH_MAX];
+ char m_fileName[PATH_MAX];
+ bool m_hostByteOrder;
+ BackupFormat::FileHeader m_fileHeader;
+ BackupFormat::FileHeader m_expectedFileHeader;
+
+ Uint32 m_nodeId;
+ Uint32 * m_buffer;
+ Uint32 m_bufferSize;
+ Uint32 * createBuffer(Uint32 bytes);
+
+ bool openFile();
+ void setCtlFile(Uint32 nodeId, Uint32 backupId, const char * path);
+ void setDataFile(const BackupFile & bf, Uint32 no);
+ void setLogFile(const BackupFile & bf, Uint32 no);
+
+ void setName(const char * path, const char * name);
+
+ BackupFile();
+ ~BackupFile();
+public:
+ bool readHeader();
+ bool validateFooter();
+
+ const char * getPath() const { return m_path;}
+ const char * getFilename() const { return m_fileName;}
+ Uint32 getNodeId() const { return m_nodeId;}
+ const BackupFormat::FileHeader & getFileHeader() const { return m_fileHeader;}
+ bool Twiddle(AttributeS * attr, Uint32 arraySize = 0);
+};
+
+class RestoreMetaData : public BackupFile {
+
+ myVector<TableS *> allTables;
+ bool readMetaFileHeader();
+ bool readMetaTableDesc(const char * catalog,
+ const char * schema);
+
+ bool readGCPEntry();
+ Uint32 readMetaTableList();
+
+ Uint32 m_startGCP;
+ Uint32 m_stopGCP;
+
+ bool parseTableDescriptor(const Uint32 * data, Uint32 len,
+ const char * catalog,
+ const char * schema);
+
+public:
+
+ RestoreMetaData(const char * path, Uint32 nodeId, Uint32 bNo);
+ ~RestoreMetaData();
+
+ int loadContent(const char * catalog,
+ const char * schema);
+
+
+
+ Uint32 getNoOfTables() const { return allTables.size();}
+
+ const TableS * operator[](int i) const { return allTables[i];}
+ const TableS * getTable(Uint32 tableId) const;
+
+ Uint32 getStopGCP() const;
+}; // RestoreMetaData
+
+
+class RestoreDataIterator : public BackupFile {
+ const RestoreMetaData & m_metaData;
+
+ Uint32 m_count;
+ TupleS m_tuple;
+ const TableS* m_currentTable;
+public:
+
+ // Constructor
+ RestoreDataIterator(const RestoreMetaData &);
+ ~RestoreDataIterator();
+
+ // Read data file fragment header
+ bool readFragmentHeader(int & res);
+ bool validateFragmentFooter();
+
+ const TupleS *getNextTuple(int & res);
+};
+
+class LogEntry {
+public:
+ enum EntryType {
+ LE_INSERT,
+ LE_DELETE,
+ LE_UPDATE
+ };
+ EntryType m_type;
+ const TableS * m_table;
+ myVector<AttributeS*> m_values;
+
+
+};
+
+class RestoreLogIterator : public BackupFile {
+private:
+ const RestoreMetaData & m_metaData;
+
+ Uint32 m_count;
+ LogEntry m_logEntry;
+public:
+ RestoreLogIterator(const RestoreMetaData &);
+
+ const LogEntry * getNextLogEntry(int & res);
+};
+
+#endif
+
+
diff --git a/ndb/src/kernel/blocks/backup/restore/main.cpp b/ndb/src/kernel/blocks/backup/restore/main.cpp
new file mode 100644
index 00000000000..52857aa2c42
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/restore/main.cpp
@@ -0,0 +1,1689 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <assert.h>
+#include "Restore.hpp"
+#include <getarg.h>
+#include <NdbSleep.h>
+#include <Vector.hpp>
+#include <ndb_limits.h>
+#ifdef USE_MYSQL
+#include <mysql.h>
+#endif
+NdbOut& operator<<(NdbOut& ndbout, const TupleS& tuple);
+NdbOut& operator<<(NdbOut& ndbout, const LogEntry& logEntry);
+NdbOut& operator<<(NdbOut& ndbout, const RestoreMetaData &);
+
+extern FilteredNdbOut err;
+extern FilteredNdbOut info;
+extern FilteredNdbOut debug;
+
+static const char * delimiter = ";"; // Delimiter in file dump
+
+static int ga_nodeId = 0;
+static int ga_nParallelism = 1;
+static int ga_backupId = 0;
+static bool ga_dont_ignore_systab_0 = false;
+static myVector<class BackupConsumer *> g_consumers;
+
+#ifdef USE_MYSQL
+/**
+ * mysql specific stuff:
+ */
+static const char* ga_user = "root";
+static const char* ga_host = "localhost";
+static const char* ga_socket = "/tmp/mysql.sock";
+static const char* ga_password = "";
+static const char* ga_database = "";
+static int ga_port = 3306;
+static bool use_mysql = false;
+static MYSQL mysql;
+#endif
+
+
+#ifdef NDB_WIN32
+static const char* ga_backupPath = ".\\";
+#else
+static const char* ga_backupPath = "./";
+#endif
+
+typedef struct {
+ void * ndb;
+ void * restore;
+ TupleS * tup;
+ int transaction;
+ int retries;
+} restore_callback_t;
+
+static const char* ga_connect_NDB = NULL;
+static const char* ga_schema = NULL;
+static const char* ga_catalog = NULL;
+
+
+
+/**
+ * print and restore flags
+ */
+static bool ga_restore = false;
+static bool ga_print = false;
+
+
+
+class BackupConsumer {
+public:
+ virtual bool init() { return true;}
+ virtual bool table(const TableS &){return true;}
+#ifdef USE_MYSQL
+ virtual bool table(const TableS &, MYSQL* mysqlp) {return true;};
+#endif
+ virtual void tuple(const TupleS &){}
+ virtual void tupleAsynch(const TupleS &, restore_callback_t * callback) {};
+ // virtual bool asynchErrorHandler(NdbConnection * trans){return true;};
+ virtual void asynchExitHandler(){};
+ virtual void endOfTuples(){}
+ virtual void logEntry(const LogEntry &){}
+ virtual void endOfLogEntrys(){}
+protected:
+ int create_table_string(const TableS & table, char * ,char *);
+};
+
+class BackupPrinter : public BackupConsumer
+{
+ NdbOut & m_ndbout;
+public:
+ BackupPrinter(NdbOut & out = ndbout) : m_ndbout(out)
+ {
+ m_print = false;
+ m_print_log = false;
+ m_print_data = false;
+ m_print_meta = false;
+ }
+
+ virtual bool table(const TableS &);
+#ifdef USE_MYSQL
+ virtual bool table(const TableS &, MYSQL* mysqlp);
+#endif
+ virtual void tuple(const TupleS &);
+ virtual void logEntry(const LogEntry &);
+ virtual void endOfTuples() {};
+ virtual void endOfLogEntrys();
+ virtual void tupleAsynch(const TupleS &, restore_callback_t * callback);
+ bool m_print;
+ bool m_print_log;
+ bool m_print_data;
+ bool m_print_meta;
+ Uint32 m_logCount;
+ Uint32 m_dataCount;
+
+};
+
+class BackupRestore : public BackupConsumer
+{
+public:
+ BackupRestore()
+ {
+ m_ndb = 0;
+ m_logCount = m_dataCount = 0;
+ m_restore = false;
+ m_restore_meta = false;
+ }
+
+ virtual ~BackupRestore();
+
+ virtual bool init();
+ virtual bool table(const TableS &);
+#ifdef USE_MYSQL
+ virtual bool table(const TableS &, MYSQL* mysqlp);
+#endif
+ virtual void tuple(const TupleS &);
+ virtual void tupleAsynch(const TupleS &, restore_callback_t * callback);
+ virtual void asynchExitHandler();
+ virtual void endOfTuples();
+ virtual void logEntry(const LogEntry &);
+ virtual void endOfLogEntrys();
+ void connectToMysql();
+ Ndb * m_ndb;
+ bool m_restore;
+ bool m_restore_meta;
+ Uint32 m_logCount;
+ Uint32 m_dataCount;
+};
+bool
+readArguments(const int argc, const char** argv)
+{
+ BackupPrinter* printer = new BackupPrinter();
+ if (printer == NULL)
+ return false;
+ BackupRestore* restore = new BackupRestore();
+ if (restore == NULL)
+ {
+ delete printer;
+ return false;
+ }
+
+ int _print = 0;
+ int _print_meta = 0;
+ int _print_data = 0;
+ int _print_log = 0;
+ int _restore_data = 0;
+ int _restore_meta = 0;
+
+
+ struct getargs args[] =
+ {
+ { "connect", 'c', arg_string, &ga_connect_NDB,
+ "NDB Cluster connection", "\"nodeid=<api id>;host=<hostname:port>\""},
+ { "nodeid", 'n', arg_integer, &ga_nodeId,
+ "Backup files from node", "db node id"},
+ { "backupid", 'b',arg_integer, &ga_backupId, "Backup id", "backup id"},
+ { "print", '\0', arg_flag, &_print,
+ "Print data and log to stdout", "print data and log"},
+ { "print_data", '\0', arg_flag, &_print_data,
+ "Print data to stdout", "print data"},
+ { "print_meta", '\0', arg_flag, &_print_meta,
+ "Print meta data to stdout", "print meta data"},
+ { "print_log", '\0', arg_flag, &_print_log,
+ "Print log to stdout", "print log"},
+ { "restore_data", 'r', arg_flag, &_restore_data,
+ "Restore table data/logs into NDB Cluster using NDBAPI",
+ "Restore table data/log"},
+ { "restore_meta", 'm', arg_flag, &_restore_meta,
+ "Restore meta data into NDB Cluster using NDBAPI", "Restore meta data"},
+ { "parallelism", 'p', arg_integer, &ga_nParallelism,
+ "No of parallel transactions during restore of data."
+ "(parallelism can be 1 to 1024)",
+ "Parallelism"},
+#if NDB_VERSION_MAJOR >= VERSION_3X
+ { "catalog", 'd', arg_string, &ga_catalog,
+ "Specifies the catalog/database where the data should be restored to. "
+ "Restores only to backups taken with v.2.x and restored on >v.3.x "
+ "systems. Note: system tables (if restored) defaults to sys/def/ ",
+ "catalog"},
+ { "schema", 's', arg_string, &ga_schema,
+ "Specifies the schema where the data should be restored to."
+ "Restores only to backups taken with v.2.x and restored on >v.3.x "
+ "systems. Note: system tables (if restored) defaults to sys/def/ ",
+ "schema"},
+#endif
+#ifdef USE_MYSQL
+ { "use_mysql", '\0', arg_flag, &use_mysql,
+ "Restore meta data via mysql. Systab will be ignored. Data is restored "
+ "using NDBAPI.", "use mysql"},
+ { "user", '\0', arg_string, &ga_user, "MySQL user", "Default: root"},
+ { "password", '\0', arg_string, &ga_password, "MySQL user's password",
+ "Default: \"\" "},
+ { "host", '\0', arg_string, &ga_host, "Hostname of MySQL server",
+ "Default: localhost"},
+ { "socket", '\0', arg_string, &ga_socket, "Path to MySQL server socket file",
+ "Default: /tmp/mysql.sock"},
+ { "port", '\0', arg_integer, &ga_port, "Port number of MySQL server",
+ "Default: 3306"},
+#endif
+ { "dont_ignore_systab_0", 'f', arg_flag, &ga_dont_ignore_systab_0,
+ "Experimental. Do not ignore system table during restore.",
+ "dont_ignore_systab_0"}
+
+ };
+
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+
+ if (getarg(args, num_args, argc, argv, &optind) ||
+ ga_nodeId == 0 ||
+ ga_backupId == 0 ||
+ ga_nParallelism < 1 ||
+ ga_nParallelism >1024)
+ {
+
+ arg_printusage(args, num_args, argv[0], "<path to backup files>\n");
+ delete printer;
+ delete restore;
+ return false;
+ }
+
+ /**
+ * Got segmentation fault when using the printer's attributes directly
+ * in getargs... Do not have the time to found out why... this is faster...
+ */
+ if (_print)
+ {
+ ga_print = true;
+ ga_restore = true;
+ printer->m_print = true;
+ }
+ if (_print_meta)
+ {
+ ga_print = true;
+ printer->m_print_meta = true;
+ }
+ if (_print_data)
+ {
+ ga_print = true;
+ printer->m_print_data = true;
+ }
+ if (_print_log)
+ {
+ ga_print = true;
+ printer->m_print_log = true;
+ }
+
+ if (_restore_data)
+ {
+ ga_restore = true;
+ restore->m_restore = true;
+ }
+
+ if (_restore_meta)
+ {
+ // ga_restore = true;
+ restore->m_restore_meta = true;
+ }
+
+ {
+ BackupConsumer * c = printer;
+ g_consumers.push_back(c);
+ }
+ {
+ BackupConsumer * c = restore;
+ g_consumers.push_back(c);
+ }
+ // Set backup file path
+ if (argv[optind] != NULL)
+ {
+ ga_backupPath = argv[optind];
+ }
+#ifdef USE_MYSQL
+ if(use_mysql) {
+ ga_dont_ignore_systab_0 = false;
+ ga_database = ""; //not used yet. pethaps later if we want to
+ // restore meta data in an existing mysql database,
+ // and not just restore it to the same database
+ // as when the backup was taken.
+ // If implementing this, then the
+ // tupleAsynch must also be changed so that the
+ // table data is restored to the correct table.
+ // also, mysql_select_db must be set properly (ie.,
+ // ignored in codw below)
+ }
+#endif
+
+ return true;
+}
+
+
+void
+clearConsumers()
+{
+ for(int i = 0; i<g_consumers.size(); i++)
+ delete g_consumers[i];
+ g_consumers.clear();
+}
+
+static bool asynchErrorHandler(NdbConnection * trans, Ndb * ndb);
+static NdbConnection * asynchTrans[1024];
+
+bool
+checkSysTable(const char *tableName)
+{
+ return ga_dont_ignore_systab_0 ||
+ (strcmp(tableName, "SYSTAB_0") != 0 &&
+ strcmp(tableName, "NDB$EVENTS_0") != 0 &&
+ strcmp(tableName, "sys/def/SYSTAB_0") != 0 &&
+ strcmp(tableName, "sys/def/NDB$EVENTS_0") != 0);
+}
+
+
+int
+main(int argc, const char** argv)
+{
+ if (!readArguments(argc, argv))
+ {
+ return -1;
+ }
+ // Turn off table name completion
+#if NDB_VERSION_MAJOR >= VERSION_3X
+ Ndb::useFullyQualifiedNames(false);
+#endif
+
+ /**
+ * we must always load meta data, even if we will only print it to stdout
+ */
+ RestoreMetaData metaData(ga_backupPath, ga_nodeId, ga_backupId);
+ if (!metaData.readHeader())
+ {
+ ndbout << "Failed to read " << metaData.getFilename() << endl << endl;
+ return -1;
+ }
+ /**
+ * check wheater we can restore the backup (right version, and if that
+ * version needs catalog and schema specified.
+ */
+ int res = metaData.loadContent(ga_catalog, ga_schema);
+
+ if (res == 0)
+ {
+ ndbout_c("Restore: Failed to load content");
+ return -1;
+ }
+ if (res == -1)
+ {
+ ndbout_c("Restore: The backup is from a NDB Cluster v.2.x version. "
+ "To restore this backup on a > 3.x version you must specify "
+ "catalog and schema.");
+ return -1;
+ }
+ if (res == -2)
+ {
+#ifdef NDB_VERSION
+ ndbout_c("Restore: The backup is from a NDB Cluster v.3.x version "
+ "Catalog and schema are invalid parameters since they "
+ "already exist implicitly.");
+#endif
+#ifdef NDB_KERNEL_VERSION
+ ndbout_c("Restore: The backup is from a NDB Cluster v.3.x version "
+ "It is not possible to restore a 3.x backup on v.2.x. ");
+#endif
+ return -1;
+ }
+
+ if (res == -3)
+ {
+ ndbout_c("Restore: The backup contains no tables "
+ "Catalog and schema are invalid parameters. ");
+ return -1;
+ }
+
+
+ if (!metaData.validateFooter())
+ {
+ ndbout_c("Restore: Failed to validate footer.");
+ return -1;
+ }
+
+
+ for(int i = 0; i<g_consumers.size(); i++)
+ {
+ if (!g_consumers[i]->init())
+ {
+ clearConsumers();
+ return -11;
+ }
+
+ }
+
+ for(Uint32 i = 0; i<metaData.getNoOfTables(); i++)
+ {
+ if (checkSysTable(metaData[i]->getTableName()))
+ {
+ for(int j = 0; j<g_consumers.size(); j++)
+#ifdef USE_MYSQL
+ if(use_mysql) {
+ if (!g_consumers[j]->table(* metaData[i], &mysql))
+ {
+ ndbout_c("Restore: Failed to restore table: %s. "
+ "Exiting...",
+ metaData[i]->getTableName());
+ return -11;
+ }
+ } else
+#endif
+ if (!g_consumers[j]->table(* metaData[i]))
+ {
+ ndbout_c("Restore: Failed to restore table: %s. "
+ "Exiting...",
+ metaData[i]->getTableName());
+ return -11;
+ }
+
+ }
+ }
+
+
+
+ if (ga_restore || ga_print)
+ {
+ if (ga_restore)
+ {
+ RestoreDataIterator dataIter(metaData);
+
+ // Read data file header
+ if (!dataIter.readHeader())
+ {
+ ndbout << "Failed to read header of data file. Exiting..." ;
+ return -11;
+ }
+
+
+ while (dataIter.readFragmentHeader(res))
+ {
+ const TupleS* tuple = 0;
+ while ((tuple = dataIter.getNextTuple(res)) != NULL)
+ {
+ if (checkSysTable(tuple->getTable()->getTableName()))
+ {
+ for(int i = 0; i<g_consumers.size(); i++)
+ {
+ g_consumers[i]->tupleAsynch(* tuple, 0);
+ }
+ }
+ } while (tuple != NULL);
+
+ if (res < 0)
+ {
+ ndbout_c("Restore: An error occured while restoring data. "
+ "Exiting...");
+ return -1;
+ }
+ if (!dataIter.validateFragmentFooter()) {
+ ndbout_c("Restore: Error validating fragment footer. "
+ "Exiting...");
+ return -1;
+ }
+ } // while (dataIter.readFragmentHeader(res))
+
+ if (res < 0)
+ {
+ ndbout_c("Restore: An error occured while restoring data. "
+ "Exiting...");
+ return -1;
+ }
+
+
+ dataIter.validateFooter(); //not implemented
+ for (int i = 0; i<g_consumers.size(); i++)
+ g_consumers[i]->endOfTuples();
+
+ RestoreLogIterator logIter(metaData);
+ if (!logIter.readHeader())
+ {
+ ndbout << "Failed to read header of data file. Exiting...";
+ return -1;
+ }
+
+ /**
+ * I have not touched the part below : -johan 040218
+ * except fixing return values.
+ */
+ const LogEntry * logEntry = 0;
+ while ((logEntry = logIter.getNextLogEntry(res)))
+ {
+ if (checkSysTable(logEntry->m_table->getTableName()))
+ {
+ for(int i = 0; i<g_consumers.size(); i++)
+ g_consumers[i]->logEntry(* logEntry);
+ }
+ }
+ if (res < 0)
+ {
+ ndbout_c("Restore: An restoring the data log"
+ "Exiting...");
+ return -1;
+ }
+ logIter.validateFooter(); //not implemented
+ for (int i = 0; i<g_consumers.size(); i++)
+ g_consumers[i]->endOfLogEntrys();
+ }
+ }
+ clearConsumers();
+ return 1;
+} // main
+
+NdbOut &
+operator<<(NdbOut& ndbout, const AttributeS& attr){
+ const AttributeData & data = attr.Data;
+ const AttributeDesc & desc = * attr.Desc;
+
+ if (data.null)
+ {
+ ndbout << "<NULL>";
+ return ndbout;
+ }
+
+ if (desc.arraySize > 1)
+ ndbout << "[ ";
+ for (Uint32 j = 0; j < desc.arraySize; j++)
+ {
+ // Print strings without spaces,
+ // (but ndbout char does not work as expected, see below)
+ switch (desc.type)
+ {
+ case Signed:
+ switch (desc.size)
+ {
+ case 8:
+ ndbout << (short)data.int8_value[j];
+ break;
+ case 16:
+ ndbout << data.int16_value[j];
+ break;
+ case 32:
+ ndbout << data.int32_value[j];
+ break;
+ case 64:
+ ndbout << data.int64_value[j];
+ break;
+ case 128:
+ ndbout << "Signed sz = 128 - this is something wrong??" << endl;
+ break;
+ default:
+ // Unknown, error
+ break;
+ } // switch size
+ break;
+ case UnSigned:
+ switch (desc.size)
+ {
+ case 8:
+ ndbout << (short)data.u_int8_value[j];
+ break;
+ case 16:
+ ndbout << data.u_int16_value[j];
+ break;
+ case 32:
+ ndbout << data.u_int32_value[j];
+ break;
+ case 64:
+ ndbout << data.u_int64_value[j];
+ break;
+ case 128:
+ ndbout << "UnSigned sz = 128 - this is something wrong??" << endl;
+ break;
+ default:
+ // Unknown, error
+ break;
+ } // switch size
+ break;
+ case (String):
+ if (desc.size == 8)
+ {
+ ndbout << data.string_value;
+ j = desc.arraySize;
+ } // if
+ else
+ {
+ ndbout << "String sz != 8 - this is something wrong??" << endl;
+ }
+ break;
+ case (Float):
+ // Not yet supported to print float
+ ndbout << "float";
+ break;
+ default:
+ ndbout << "Not defined Attr Type";
+ } // switch AttrType
+ ndbout << " ";
+ } // for ArraySize
+ if (desc.arraySize > 1)
+ {
+ ndbout << "]";
+ }
+ return ndbout;
+}
+
+// Print tuple data
+NdbOut&
+operator<<(NdbOut& ndbout, const TupleS& tuple)
+{
+ ndbout << tuple.getTable()->getTableName() << "; ";
+ for (int i = 0; i < tuple.getNoOfAttributes(); i++)
+ {
+ const AttributeS * attr = tuple[i];
+ debug << i << " " << attr->Desc->name;
+
+ ndbout << (* attr);
+
+ if (i != (tuple.getNoOfAttributes() - 1))
+ ndbout << delimiter << " ";
+ } // for
+ return ndbout;
+}
+
+// Print tuple data
+NdbOut&
+operator<<(NdbOut& ndbout, const LogEntry& logE)
+{
+ switch(logE.m_type)
+ {
+ case LogEntry::LE_INSERT:
+ ndbout << "INSERT " << logE.m_table->getTableName() << " ";
+ break;
+ case LogEntry::LE_DELETE:
+ ndbout << "DELETE " << logE.m_table->getTableName() << " ";
+ break;
+ case LogEntry::LE_UPDATE:
+ ndbout << "UPDATE " << logE.m_table->getTableName() << " ";
+ break;
+ default:
+ ndbout << "Unknown log entry type (not insert, delete or update)" ;
+ }
+
+ for (int i = 0; i < logE.m_values.size();i++)
+ {
+ const AttributeS * attr = logE.m_values[i];
+ ndbout << attr->Desc->name << "=";
+ ndbout << (* attr);
+ if (i < (logE.m_values.size() - 1))
+ ndbout << ", ";
+ }
+ return ndbout;
+}
+
+
+NdbOut &
+operator<<(NdbOut& ndbout, const TableS & table){
+ ndbout << endl << "Table: " << table.getTableName() << endl;
+ for (int j = 0; j < table.getNoOfAttributes(); j++)
+ {
+ const AttributeDesc * desc = table[j];
+ ndbout << desc->name << ": ";
+ NdbDictionary::Column::Type type = table.m_dictTable->getColumn(desc->attrId)->getType();
+ switch(type){
+ case NdbDictionary::Column::Int:
+ ndbout << "Int ";
+ break;
+ case NdbDictionary::Column::Unsigned:
+ ndbout << "Unsigned ";
+ break;
+ case NdbDictionary::Column::Float:
+ ndbout << "Float ";
+ break;
+ case NdbDictionary::Column::Decimal:
+ ndbout << "Decimal ";
+ break;
+ case NdbDictionary::Column::Char:
+ ndbout << "Char ";
+ break;
+ case NdbDictionary::Column::Varchar:
+ ndbout << "Varchar ";
+ break;
+ case NdbDictionary::Column::Binary:
+ ndbout << "Binary ";
+ break;
+ case NdbDictionary::Column::Varbinary:
+ ndbout << "Varbinary ";
+ break;
+ case NdbDictionary::Column::Bigint:
+ ndbout << "Bigint ";
+ break;
+ case NdbDictionary::Column::Bigunsigned:
+ ndbout << "Bigunsigned ";
+ break;
+ case NdbDictionary::Column::Double:
+ ndbout << "Double ";
+ break;
+ case NdbDictionary::Column::Datetime:
+ ndbout << "Datetime ";
+ break;
+ case NdbDictionary::Column::Timespec:
+ ndbout << "Timespec ";
+ break;
+ case NdbDictionary::Column::Undefined:
+ ndbout << "Undefined ";
+ break;
+ default:
+ ndbout << "Unknown(" << type << ")";
+ }
+ ndbout << " key: " << desc->key;
+ ndbout << " array: " << desc->arraySize;
+ ndbout << " size: " << desc->size << endl;
+ } // for
+ return ndbout;
+}
+
+
+#if 0
+/*****************************************
+ *
+ * Callback function for asynchronous transactions
+ *
+ * Idea for error handling: Transaction objects have to be stored globally when
+ * they are prepared.
+ * In the callback function if the transaction:
+ * succeeded: delete the object from global storage
+ * failed but can be retried: execute the object that is in global storage
+ * failed but fatal: delete the object from global storage
+ *
+ ******************************************/
+static void restoreCallback(int result, // Result for transaction
+ NdbConnection *object, // Transaction object
+ void *anything) // Not used
+{
+ static Uint32 counter = 0;
+
+
+ debug << "restoreCallback function called " << counter << " time(s)" << endl;
+
+ ++counter;
+
+ if (result == -1)
+ {
+ ndbout << " restoreCallback (" << counter;
+ if ((counter % 10) == 1)
+ {
+ ndbout << "st";
+ } // if
+ else if ((counter % 10) == 2)
+ {
+ ndbout << "nd";
+ } // else if
+ else if ((counter % 10 ) ==3)
+ {
+ ndbout << "rd";
+ } // else if
+ else
+ {
+ ndbout << "th";
+ } // else
+ err << " time: error detected " << object->getNdbError() << endl;
+ } // if
+
+} // restoreCallback
+#endif
+
+
+
+
+bool
+BackupPrinter::table(const TableS & tab)
+{
+ if (m_print || m_print_meta)
+ {
+ m_ndbout << tab;
+ ndbout_c("Successfully printed table: %s", tab.m_dictTable->getName());
+ }
+ return true;
+}
+
+#ifdef USE_MYSQL
+bool
+BackupPrinter::table(const TableS & tab, MYSQL * mysql)
+{
+ if (m_print || m_print_meta)
+ {
+
+ char tmpTabName[MAX_TAB_NAME_SIZE*2];
+ sprintf(tmpTabName, "%s", tab.getTableName());
+ char * database = strtok(tmpTabName, "/");
+ char * schema = strtok( NULL , "/");
+ char * tableName = strtok( NULL , "/");
+
+ /**
+ * this means that the user did not specify schema
+ * and it is a v2x backup
+ */
+ if(database == NULL)
+ return false;
+ if(schema == NULL)
+ return false;
+ if(tableName==NULL)
+ tableName = schema;
+
+ char stmtCreateDB[255];
+
+ sprintf(stmtCreateDB,"CREATE DATABASE %s", database);
+ ndbout_c("%s", stmtCreateDB);
+
+
+ char buf [2048];
+ create_table_string(tab, tableName, buf);
+ ndbout_c("%s", buf);
+
+ ndbout_c("Successfully printed table: %s", tab.m_dictTable->getName());
+ }
+ return true;
+}
+
+#endif
+
+void
+BackupPrinter::tuple(const TupleS & tup)
+{
+ if (m_print || m_print_data)
+ m_ndbout << tup << endl;
+}
+
+void
+BackupPrinter::logEntry(const LogEntry & logE)
+{
+ if (m_print || m_print_log)
+ m_ndbout << logE << endl;
+ m_logCount++;
+}
+
+bool
+BackupRestore::init()
+{
+
+ if (!m_restore && !m_restore_meta)
+ return true;
+
+ if (ga_connect_NDB != NULL)
+ {
+ // Use connection string
+ Ndb::setConnectString(ga_connect_NDB);
+ }
+
+ m_ndb = new Ndb("TEST_DB");
+ if (m_ndb == NULL)
+ return false;
+
+ m_ndb->init(1024);
+ if (m_ndb->waitUntilReady(30) != 0)
+ {
+ ndbout << "Failed to connect to ndb!!" << endl;
+ delete m_ndb;
+ return false;
+ }
+ ndbout << "Connected to ndb!!" << endl;
+
+#if USE_MYSQL
+ if(use_mysql)
+ {
+ if ( mysql_thread_safe() == 0 )
+ {
+ ndbout << "Not thread safe mysql library..." << endl;
+ exit(-1);
+ }
+
+ ndbout << "Connecting to MySQL..." <<endl;
+
+ /**
+ * nwe param:
+ * port
+ * host
+ * user
+ */
+ bool returnValue = true;
+ mysql_init(&mysql);
+ {
+ int portNo = 3306;
+ if ( mysql_real_connect(&mysql,
+ ga_host,
+ ga_user,
+ ga_password,
+ ga_database,
+ ga_port,
+ ga_socket,
+ 0) == NULL )
+ {
+ ndbout_c("Connect failed: %s", mysql_error(&mysql));
+ returnValue = false;
+ }
+ ndbout << "Connected to MySQL!!!" <<endl;
+ }
+
+ /* if(returnValue){
+ mysql_set_server_option(&mysql, MYSQL_OPTION_MULTI_STATEMENTS_ON);
+ }
+ */
+ return returnValue;
+ }
+#endif
+ return true;
+
+}
+
+BackupRestore::~BackupRestore()
+{
+ if (m_ndb != 0)
+ delete m_ndb;
+}
+#ifdef USE_MYSQL
+bool
+BackupRestore::table(const TableS & table, MYSQL * mysqlp){
+ if (!m_restore_meta)
+ {
+ return true;
+ }
+
+ char tmpTabName[MAX_TAB_NAME_SIZE*2];
+ sprintf(tmpTabName, "%s", table.getTableName());
+ char * database = strtok(tmpTabName, "/");
+ char * schema = strtok( NULL , "/");
+ char * tableName = strtok( NULL , "/");
+
+ /**
+ * this means that the user did not specify schema
+ * and it is a v2x backup
+ */
+ if(database == NULL)
+ return false;
+ if(schema == NULL)
+ return false;
+ if(tableName==NULL)
+ tableName = schema;
+
+ char stmtCreateDB[255];
+ sprintf(stmtCreateDB,"CREATE DATABASE %s", database);
+
+ /*ignore return value. mysql_select_db will trap errors anyways*/
+ if (mysql_query(mysqlp,stmtCreateDB) == 0)
+ {
+ //ndbout_c("%s", stmtCreateDB);
+ }
+
+ if (mysql_select_db(&mysql, database) != 0)
+ {
+ ndbout_c("Error: %s", mysql_error(&mysql));
+ return false;
+ }
+
+ char buf [2048];
+ /**
+ * create table ddl
+ */
+ if (create_table_string(table, tableName, buf))
+ {
+ ndbout_c("Unable to create a table definition since the "
+ "backup contains undefined types");
+ return false;
+ }
+
+ //ndbout_c("%s", buf);
+
+ if (mysql_query(mysqlp,buf) != 0)
+ {
+ ndbout_c("Error: %s", mysql_error(&mysql));
+ return false;
+ } else
+ {
+ ndbout_c("Successfully restored table %s into database %s", tableName, database);
+ }
+
+ return true;
+}
+#endif
+
+int
+BackupConsumer::create_table_string(const TableS & table,
+ char * tableName,
+ char *buf){
+ int pos = 0;
+ int pos2 = 0;
+ char buf2[2048];
+
+ pos += sprintf(buf+pos, "%s%s", "CREATE TABLE ", tableName);
+ pos += sprintf(buf+pos, "%s", "(");
+ pos2 += sprintf(buf2+pos2, "%s", " primary key(");
+
+ for (int j = 0; j < table.getNoOfAttributes(); j++)
+ {
+ const AttributeDesc * desc = table[j];
+ // ndbout << desc->name << ": ";
+ pos += sprintf(buf+pos, "%s%s", desc->name," ");
+ NdbDictionary::Column::Type type = table.m_dictTable->getColumn(desc->attrId)->getType();
+ switch(type){
+ case NdbDictionary::Column::Int:
+ pos += sprintf(buf+pos, "%s", "int");
+ break;
+ case NdbDictionary::Column::Unsigned:
+ pos += sprintf(buf+pos, "%s", "int unsigned");
+ break;
+ case NdbDictionary::Column::Float:
+ pos += sprintf(buf+pos, "%s", "float");
+ break;
+ case NdbDictionary::Column::Decimal:
+ pos += sprintf(buf+pos, "%s", "decimal");
+ break;
+ case NdbDictionary::Column::Char:
+ pos += sprintf(buf+pos, "%s", "char");
+ break;
+ case NdbDictionary::Column::Varchar:
+ pos += sprintf(buf+pos, "%s", "varchar");
+ break;
+ case NdbDictionary::Column::Binary:
+ pos += sprintf(buf+pos, "%s", "binary");
+ break;
+ case NdbDictionary::Column::Varbinary:
+ pos += sprintf(buf+pos, "%s", "varchar binary");
+ break;
+ case NdbDictionary::Column::Bigint:
+ pos += sprintf(buf+pos, "%s", "bigint");
+ break;
+ case NdbDictionary::Column::Bigunsigned:
+ pos += sprintf(buf+pos, "%s", "bigint unsigned");
+ break;
+ case NdbDictionary::Column::Double:
+ pos += sprintf(buf+pos, "%s", "double");
+ break;
+ case NdbDictionary::Column::Datetime:
+ pos += sprintf(buf+pos, "%s", "datetime");
+ break;
+ case NdbDictionary::Column::Timespec:
+ pos += sprintf(buf+pos, "%s", "time");
+ break;
+ case NdbDictionary::Column::Undefined:
+ // pos += sprintf(buf+pos, "%s", "varchar binary");
+ return -1;
+ break;
+ default:
+ //pos += sprintf(buf+pos, "%s", "varchar binary");
+ return -1;
+ }
+ if (desc->arraySize > 1) {
+ int attrSize = desc->arraySize;
+ pos += sprintf(buf+pos, "%s%u%s",
+ "(",
+ attrSize,
+ ")");
+ }
+ if (table.m_dictTable->getColumn(desc->attrId)->getPrimaryKey()) {
+ pos += sprintf(buf+pos, "%s", " not null");
+ pos2 += sprintf(buf2+pos2, "%s%s", desc->name, ",");
+ }
+ pos += sprintf(buf+pos, "%s", ",");
+ } // for
+ pos2--; // remove trailing comma
+ pos2 += sprintf(buf2+pos2, "%s", ")");
+ // pos--; // remove trailing comma
+
+ pos += sprintf(buf+pos, "%s", buf2);
+ pos += sprintf(buf+pos, "%s", ") type=ndbcluster");
+ return 0;
+}
+
+
+
+bool
+BackupRestore::table(const TableS & table){
+ if (!m_restore_meta)
+ {
+ return true;
+ }
+#ifndef restore_old_types
+ NdbDictionary::Dictionary* dict = m_ndb->getDictionary();
+ if (dict->createTable(*table.m_dictTable) == -1)
+ {
+ err << "Create table " << table.getTableName() << " failed: "
+ << dict->getNdbError() << endl;
+ return false;
+ }
+ info << "Successfully restored table " << table.getTableName()<< endl ;
+ return true;
+#else
+ NdbSchemaCon * tableTransaction = 0;
+ NdbSchemaOp * tableOp = 0;
+
+ tableTransaction = m_ndb->startSchemaTransaction();
+ if (tableTransaction == NULL)
+ {
+ err << table.getTableName()
+ << " - BackupRestore::table cannot startSchemaTransaction: "
+ << tableTransaction->getNdbError() << endl;
+ return false;
+ } // if
+
+ tableOp = tableTransaction->getNdbSchemaOp();
+ if (tableOp == NULL)
+ {
+ err << table.getTableName()
+ << " - BackupRestore::table cannot getNdbSchemaOp: "
+ << tableTransaction->getNdbError() << endl;
+ m_ndb->closeSchemaTransaction(tableTransaction);
+ return false;
+ } // if
+
+ // TODO: check for errors in table attributes. set aTupleKey
+ int check = 0;
+ check = tableOp->createTable(table.getTableName());
+ // aTableSize = 8, Not used?
+ // aTupleKey = TupleKey, go through attributes and check if there is a PK
+ // and so on....
+ if (check == -1)
+ {
+ err << table.getTableName()
+ << " - BackupRestore::table cannot createTable: "
+ << tableTransaction->getNdbError() << endl;
+ m_ndb->closeSchemaTransaction(tableTransaction);
+ return false;
+ } // if
+
+ // Create attributes from meta data
+ for (int i = 0; i < table.getNoOfAttributes(); i++)
+ {
+ const AttributeDesc* desc = table[i];
+ check = tableOp->createAttribute(desc->name, // Attr name
+ desc->key, // Key type
+ desc->size, // bits
+ desc->arraySize,
+ desc->type,
+ MMBased, // only supported
+ desc->nullable
+ // Rest is don't care for the moment
+ );
+
+ if (check == -1)
+ {
+ err << table.getTableName()
+ << " - RestoreDataIterator::createTable cannot createAttribute: "
+ << tableTransaction->getNdbError() << endl;
+ m_ndb->closeSchemaTransaction(tableTransaction);
+ return false;
+ } // if
+ } // for
+
+ if (tableTransaction->execute() == -1)
+ {
+ err << table.getTableName()
+ << " - RestoreDataIterator::createTable cannot execute transaction: "
+ << tableTransaction->getNdbError() << endl;
+ m_ndb->closeSchemaTransaction(tableTransaction);
+ return false;
+ } // if
+
+ m_ndb->closeSchemaTransaction(tableTransaction);
+ info << "Successfully created table " << table.getTableName() << endl;
+ return true ;
+#endif
+}
+
+
+
+/*
+ * callback : This is called when the transaction is polled
+ *
+ * (This function must have three arguments:
+ * - The result of the transaction,
+ * - The NdbConnection object, and
+ * - A pointer to an arbitrary object.)
+ */
+
+static void
+callback(int result, NdbConnection* trans, void* aObject)
+{
+ restore_callback_t * cbData = (restore_callback_t *)aObject;
+ if (result<0)
+ {
+ /**
+ * Error. temporary or permanent?
+ */
+ if (asynchErrorHandler(trans, (Ndb*)cbData->ndb))
+ {
+ ((Ndb*)cbData->ndb)->closeTransaction(asynchTrans[cbData->transaction]);
+ cbData->retries++;
+ ((BackupRestore*)cbData)->tupleAsynch( * (TupleS*)(cbData->tup), cbData);
+ }
+ else
+ {
+ ndbout_c("Restore: Failed to restore data "
+ "due to a unrecoverable error. Exiting...");
+ delete (Ndb*)cbData->ndb;
+ delete cbData->tup;
+ delete cbData;
+ exit(-1);
+ }
+ }
+ else
+ {
+ /**
+ * OK! close transaction
+ */
+ ((Ndb*)cbData->ndb)->closeTransaction(asynchTrans[cbData->transaction]);
+ delete cbData->tup;
+ delete cbData;
+ }
+}
+
+static int nPreparedTransactions = 0;
+void
+BackupPrinter::tupleAsynch(const TupleS & tup, restore_callback_t * callback)
+{
+ m_dataCount++;
+ if (m_print || m_print_data)
+ m_ndbout << tup << endl;
+}
+
+void BackupRestore::tupleAsynch(const TupleS & tup, restore_callback_t * cbData)
+{
+
+ if (!m_restore)
+ {
+ delete &tup;
+ return;
+ }
+ Uint32 retries;
+ if (cbData!=0)
+ retries = cbData->retries;
+ else
+ retries = 0;
+
+ while (retries < 10)
+ {
+ /**
+ * start transactions
+ */
+ asynchTrans[nPreparedTransactions] = m_ndb->startTransaction();
+ if (asynchTrans[nPreparedTransactions] == NULL)
+ {
+ if (asynchErrorHandler(asynchTrans[nPreparedTransactions], m_ndb))
+ {
+ retries++;
+ continue;
+ }
+ asynchExitHandler();
+ } // if
+
+ const TableS * table = tup.getTable();
+ NdbOperation * op =
+ asynchTrans[nPreparedTransactions]->getNdbOperation(table->getTableName());
+
+ if (op == NULL)
+ {
+ if (asynchErrorHandler(asynchTrans[nPreparedTransactions], m_ndb))
+ {
+ retries++;
+ continue;
+ }
+ asynchExitHandler();
+ } // if
+
+ if (op->writeTuple() == -1)
+ {
+ if (asynchErrorHandler(asynchTrans[nPreparedTransactions], m_ndb))
+ {
+ retries++;
+ continue;
+ }
+ asynchExitHandler();
+ } // if
+
+ Uint32 ret = 0;
+ for (int i = 0; i < tup.getNoOfAttributes(); i++)
+ {
+ const AttributeS * attr = tup[i];
+ int size = attr->Desc->size;
+ int arraySize = attr->Desc->arraySize;
+ const KeyType key = attr->Desc->key;
+ char * dataPtr = attr->Data.string_value;
+ Uint32 length = (size * arraySize) / 8;
+ if (key == TupleKey)
+ {
+#if NDB_VERSION_MAJOR >= VERSION3X
+ /**
+ * Convert VARCHAR from v.2x to v3x representation
+ */
+ if (getMajor(tup.getTable()->getBackupVersion()) < VERSION_3X &&
+ ((tup.getTable()->m_dictTable->getColumn(i)->getType() ==
+ NdbDictionary::Column::Varbinary ) ||
+ (tup.getTable()->m_dictTable->getColumn(i)->getType() ==
+ NdbDictionary::Column::Varchar)) && !attr->Data.null)
+ {
+ char * src = dataPtr;
+ char var_len[2];
+ var_len[0]= *(dataPtr+length - 2);
+ var_len[1]= *(dataPtr+length - 1);
+ memmove((char*)dataPtr+2, dataPtr, length);
+ src[0] = var_len[0];
+ src[1] = var_len[1];
+ dataPtr = src;
+ }
+#endif
+ ret = op->equal(i, dataPtr, length);
+ if (ret<0)
+ {
+ ndbout_c("Column: %d type %d",i,
+ tup.getTable()->m_dictTable->getColumn(i)->getType());
+
+ if (asynchErrorHandler(asynchTrans[nPreparedTransactions],m_ndb))
+ {
+ retries++;
+ continue;
+ }
+ asynchExitHandler();
+ }
+ }
+ }
+
+ for (int i = 0; i < tup.getNoOfAttributes(); i++)
+ {
+ const AttributeS * attr = tup[i];
+ int size = attr->Desc->size;
+ int arraySize = attr->Desc->arraySize;
+ KeyType key = attr->Desc->key;
+ char * dataPtr = attr->Data.string_value;
+ Uint32 length = (size * arraySize) / 8;
+#if NDB_VERSION_MAJOR >= VERSION3X
+ /**
+ * Convert VARCHAR from v.2x to v3x representation
+ */
+ if (getMajor(tup.getTable()->getBackupVersion()) < VERSION_3X &&
+ ((tup.getTable()->m_dictTable->getColumn(i)->getType() ==
+ NdbDictionary::Column::Varbinary ) ||
+ (tup.getTable()->m_dictTable->getColumn(i)->getType() ==
+ NdbDictionary::Column::Varchar)) && !attr->Data.null)
+ {
+ char * src = dataPtr;
+ char var_len[2];
+ var_len[0]= *(dataPtr+length - 2);//length is last 2 bytes
+ var_len[1]= *(dataPtr+length - 1);
+ memmove((char*)dataPtr+2, dataPtr, length);
+ src[0] = var_len[0];
+ src[1] = var_len[1];
+ dataPtr = src;
+ }
+#endif
+
+ if (key == NoKey && !attr->Data.null)
+ {
+ ret = op->setValue(i, dataPtr, length);
+ }
+ else if (key == NoKey && attr->Data.null)
+ {
+ ret = op->setValue(i, NULL, 0);
+ }
+
+ if (ret<0)
+ {
+ ndbout_c("Column: %d type %d",i,
+ tup.getTable()->m_dictTable->getColumn(i)->getType());
+
+ if (asynchErrorHandler(asynchTrans[nPreparedTransactions], m_ndb))
+ {
+ retries++;
+ continue;
+ }
+
+
+ asynchExitHandler();
+ }
+ }
+ restore_callback_t * cb;
+ if (cbData ==0)
+ {
+ cb = new restore_callback_t;
+ cb->retries = 0;
+ }
+ else
+ cb =cbData;
+ cb->ndb = m_ndb;
+ cb->restore = this;
+ cb->tup = (TupleS*)&tup;
+ cb->transaction = nPreparedTransactions;
+
+ // Prepare transaction (the transaction is NOT yet sent to NDB)
+ asynchTrans[nPreparedTransactions]->executeAsynchPrepare(Commit,
+ &callback,
+ cb);
+ if (nPreparedTransactions == ga_nParallelism-1)
+ {
+ // send-poll all transactions
+ // close transaction is done in callback
+ m_ndb->sendPollNdb(3000, ga_nParallelism);
+ nPreparedTransactions=0;
+ }
+ else
+ nPreparedTransactions++;
+ m_dataCount++;
+ return;
+ }
+ ndbout_c("Unable to recover from errors. Exiting...");
+ asynchExitHandler();
+}
+
+void BackupRestore::asynchExitHandler()
+{
+ if (m_ndb != NULL)
+ delete m_ndb;
+ exit(-1);
+}
+/**
+ * returns true if is recoverable,
+ * Error handling based on hugo
+ * false if it is an error that generates an abort.
+ */
+static
+bool asynchErrorHandler(NdbConnection * trans, Ndb* ndb)
+{
+
+ NdbError error = trans->getNdbError();
+ ndb->closeTransaction(trans);
+ switch(error.status)
+ {
+ case NdbError::Success:
+ return false;
+ // ERROR!
+ break;
+
+ case NdbError::TemporaryError:
+ NdbSleep_MilliSleep(10);
+ return true;
+ // RETRY
+ break;
+
+ case NdbError::UnknownResult:
+ ndbout << error << endl;
+ return false;
+ // ERROR!
+ break;
+
+ default:
+ case NdbError::PermanentError:
+ switch (error.code)
+ {
+ case 499:
+ case 250:
+ NdbSleep_MilliSleep(10);
+ return true; //temp errors?
+ default:
+ break;
+ }
+ //ERROR
+ ndbout << error << endl;
+ return false;
+ break;
+ }
+ return false;
+}
+
+
+
+void
+BackupRestore::tuple(const TupleS & tup)
+{
+ if (!m_restore)
+ return;
+ while (1)
+ {
+ NdbConnection * trans = m_ndb->startTransaction();
+ if (trans == NULL)
+ {
+ // Deep shit, TODO: handle the error
+ ndbout << "Cannot start transaction" << endl;
+ exit(-1);
+ } // if
+
+ const TableS * table = tup.getTable();
+ NdbOperation * op = trans->getNdbOperation(table->getTableName());
+ if (op == NULL)
+ {
+ ndbout << "Cannot get operation: ";
+ ndbout << trans->getNdbError() << endl;
+ exit(-1);
+ } // if
+
+ // TODO: check return value and handle error
+ if (op->writeTuple() == -1)
+ {
+ ndbout << "writeTuple call failed: ";
+ ndbout << trans->getNdbError() << endl;
+ exit(-1);
+ } // if
+
+ for (int i = 0; i < tup.getNoOfAttributes(); i++)
+ {
+ const AttributeS * attr = tup[i];
+ int size = attr->Desc->size;
+ int arraySize = attr->Desc->arraySize;
+ KeyType key = attr->Desc->key;
+ const char * dataPtr = attr->Data.string_value;
+
+ const Uint32 length = (size * arraySize) / 8;
+ if (key == TupleKey)
+ {
+ op->equal(i, dataPtr, length);
+ }
+ }
+
+ for (int i = 0; i < tup.getNoOfAttributes(); i++)
+ {
+ const AttributeS * attr = tup[i];
+ int size = attr->Desc->size;
+ int arraySize = attr->Desc->arraySize;
+ KeyType key = attr->Desc->key;
+ const char * dataPtr = attr->Data.string_value;
+
+ const Uint32 length = (size * arraySize) / 8;
+ if (key == NoKey && !attr->Data.null)
+ {
+ op->setValue(i, dataPtr, length);
+ }
+ else if (key == NoKey && attr->Data.null)
+ {
+ op->setValue(i, NULL, 0);
+ }
+ }
+ int ret = trans->execute(Commit);
+ if (ret != 0)
+ {
+ ndbout << "execute failed: ";
+ ndbout << trans->getNdbError() << endl;
+ exit(-1);
+ }
+ m_ndb->closeTransaction(trans);
+ if (ret == 0)
+ break;
+ }
+ m_dataCount++;
+}
+
+void
+BackupRestore::endOfTuples()
+{
+ if (!m_restore)
+ return;
+ // Send all transactions to NDB
+ m_ndb->sendPreparedTransactions(0);
+ // Poll all transactions
+ m_ndb->pollNdb(3000, nPreparedTransactions);
+ // Close all transactions
+ // for (int i = 0; i < nPreparedTransactions; i++)
+ // m_ndb->closeTransaction(asynchTrans[i]);
+ nPreparedTransactions=0;
+}
+
+void
+BackupRestore::logEntry(const LogEntry & tup)
+{
+ if (!m_restore)
+ return;
+
+ NdbConnection * trans = m_ndb->startTransaction();
+ if (trans == NULL)
+ {
+ // Deep shit, TODO: handle the error
+ ndbout << "Cannot start transaction" << endl;
+ exit(-1);
+ } // if
+
+ const TableS * table = tup.m_table;
+ NdbOperation * op = trans->getNdbOperation(table->getTableName());
+ if (op == NULL)
+ {
+ ndbout << "Cannot get operation: ";
+ ndbout << trans->getNdbError() << endl;
+ exit(-1);
+ } // if
+
+ int check = 0;
+ switch(tup.m_type)
+ {
+ case LogEntry::LE_INSERT:
+ check = op->insertTuple();
+ break;
+ case LogEntry::LE_UPDATE:
+ check = op->updateTuple();
+ break;
+ case LogEntry::LE_DELETE:
+ check = op->deleteTuple();
+ break;
+ default:
+ ndbout << "Log entry has wrong operation type."
+ << " Exiting...";
+ exit(-1);
+ }
+
+ for (int i = 0; i < tup.m_values.size(); i++)
+ {
+ const AttributeS * attr = tup.m_values[i];
+ int size = attr->Desc->size;
+ int arraySize = attr->Desc->arraySize;
+ KeyType key = attr->Desc->key;
+ const char * dataPtr = attr->Data.string_value;
+
+ const Uint32 length = (size / 8) * arraySize;
+ if (key == TupleKey)
+ {
+ op->equal(attr->Desc->attrId, dataPtr, length);
+ }
+ else if (key == NoKey)
+ {
+ op->setValue(attr->Desc->attrId, dataPtr, length);
+ }
+ }
+
+#if 1
+ trans->execute(Commit);
+#else
+ const int ret = trans->execute(Commit);
+ // Both insert update and delete can fail during log running
+ // and it's ok
+
+ if (ret != 0)
+ {
+ ndbout << "execute failed: ";
+ ndbout << trans->getNdbError() << endl;
+ exit(-1);
+ }
+#endif
+
+ m_ndb->closeTransaction(trans);
+ m_logCount++;
+}
+
+void
+BackupRestore::endOfLogEntrys()
+{
+ if (ga_restore)
+ {
+ ndbout << "Restored " << m_dataCount << " tuples and "
+ << m_logCount << " log entries" << endl;
+ }
+}
+
+void
+BackupPrinter::endOfLogEntrys()
+{
+ if (m_print || m_print_log)
+ {
+ ndbout << "Printed " << m_dataCount << " tuples and "
+ << m_logCount << " log entries"
+ << " to stdout." << endl;
+ }
+}
+
+
+
+
diff --git a/ndb/src/kernel/blocks/backup/restore/myVector.hpp b/ndb/src/kernel/blocks/backup/restore/myVector.hpp
new file mode 100644
index 00000000000..c858999d2be
--- /dev/null
+++ b/ndb/src/kernel/blocks/backup/restore/myVector.hpp
@@ -0,0 +1,128 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MY_VECTOR_HPP
+#define MY_VECTOR_HPP
+
+// Template class for std::vector-like class (hopefully works in OSE)
+template <class T>
+class myVector
+{
+
+ // Note that last element in array is used for end() and is always empty
+ int sizeIncrement;
+ int thisSize;
+ int used;
+ T *storage;
+
+public:
+
+ // Assignment of whole vector
+ myVector<T> & operator=(myVector<T> & org) {
+
+ // Don't copy if they point to the same address
+ if (!(this == &org)) {
+ // Check memory space
+ if (thisSize < org.thisSize) {
+ // We have to increase memory for destination
+ T* tmpStorage = new T[org.thisSize];
+ delete[] storage;
+ storage = tmpStorage;
+ } // if
+ thisSize = org.thisSize;
+ sizeIncrement = org.sizeIncrement;
+ used = org.used;
+ for (int i = 0; i < thisSize; i++) {
+ storage[i] = org.storage[i];
+ } // for
+ } // if
+ return *this;
+ } // operator=
+
+ // Construct with size s+1
+ myVector(int s = 1) : sizeIncrement(5), // sizeIncrement(s),
+ thisSize(s + 1),
+ used(0),
+ storage(new T[s + 1]) { }
+
+ ~myVector() { delete[] storage; } // Destructor: deallocate memory
+
+ T& operator[](int i) { // Return by index
+ if ((i < 0) || (i >= used)) {
+ // Index error
+ ndbout << "vector index out of range" << endl;
+ abort();
+ return storage[used - 1];
+ } // if
+ else {
+ return storage[i];
+ } // else
+ } // operator[]
+
+ const T& operator[](int i) const { // Return by index
+ if ((i < 0) || (i >= used)) {
+ // Index error
+ ndbout << "vector index out of range" << endl;
+ abort();
+ return storage[used - 1];
+ } // if
+ else {
+ return storage[i];
+ } // else
+ } // operator[]
+
+ int getSize() const { return used; }
+
+ void push_back (T& item) {
+ if (used >= thisSize - 1) {
+ // We have to allocate new storage
+ int newSize = thisSize + sizeIncrement;
+ T* tmpStorage = new T[newSize];
+ if (tmpStorage == NULL) {
+ // Memory allocation error! break
+ ndbout << "PANIC: Memory allocation error in vector" << endl;
+ return;
+ } // if
+ thisSize = newSize;
+ for (int i = 0; i < used; i++) {
+ tmpStorage[i] = storage[i];
+ } // for
+ delete[] storage;
+ storage = tmpStorage;
+ } // if
+
+ // Now push
+ storage[used] = item;
+ used++;
+ }; // myVector<> push_back()
+
+ // Remove item at back
+ void pop_back() {
+ if (used > 0) {
+ used--;
+ } // if
+ } // pop_back()
+
+ int size() const { return used; };
+
+ bool empty() const { return(used == 0); }
+
+ void clear() {
+ used = 0;
+ }
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/cmvmi/Cmvmi.cpp b/ndb/src/kernel/blocks/cmvmi/Cmvmi.cpp
new file mode 100644
index 00000000000..2735cac0c8e
--- /dev/null
+++ b/ndb/src/kernel/blocks/cmvmi/Cmvmi.cpp
@@ -0,0 +1,1531 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Cmvmi.hpp"
+
+#include <ClusterConfiguration.hpp>
+#include <Configuration.hpp>
+#include <kernel_types.h>
+#include <TransporterRegistry.hpp>
+#include <NdbOut.hpp>
+#include <NdbMem.h>
+
+#include <SignalLoggerManager.hpp>
+#include <FastScheduler.hpp>
+
+#define DEBUG(x) { ndbout << "CMVMI::" << x << endl; }
+
+#include <signaldata/TestOrd.hpp>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/TamperOrd.hpp>
+#include <signaldata/SetVarReq.hpp>
+#include <signaldata/StartOrd.hpp>
+#include <signaldata/CmvmiCfgConf.hpp>
+#include <signaldata/CmInit.hpp>
+#include <signaldata/CloseComReqConf.hpp>
+#include <signaldata/SetLogLevelOrd.hpp>
+#include <signaldata/EventSubscribeReq.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+#include <signaldata/ArbitSignalData.hpp>
+#include <signaldata/DisconnectRep.hpp>
+
+#include <EventLogger.hpp>
+#include <TimeQueue.hpp>
+#include <new>
+
+#include <SafeCounter.hpp>
+
+// Used here only to print event reports on stdout/console.
+static EventLogger eventLogger;
+
+Cmvmi::Cmvmi(const Configuration & conf) :
+ SimulatedBlock(CMVMI, conf)
+ ,theConfig((Configuration&)conf)
+ ,theCConfig(conf.clusterConfiguration()),
+ subscribers(subscriberPool)
+{
+ BLOCK_CONSTRUCTOR(Cmvmi);
+
+ // Add received signals
+ addRecSignal(GSN_CONNECT_REP, &Cmvmi::execCONNECT_REP);
+ addRecSignal(GSN_DISCONNECT_REP, &Cmvmi::execDISCONNECT_REP);
+
+ addRecSignal(GSN_NDB_TAMPER, &Cmvmi::execNDB_TAMPER, true);
+ addRecSignal(GSN_SET_LOGLEVELORD, &Cmvmi::execSET_LOGLEVELORD);
+ addRecSignal(GSN_EVENT_REP, &Cmvmi::execEVENT_REP);
+ addRecSignal(GSN_STTOR, &Cmvmi::execSTTOR_Local);
+ addRecSignal(GSN_CM_RUN, &Cmvmi::execCM_RUN);
+ addRecSignal(GSN_CM_INFOREQ, &Cmvmi::execCM_INFOREQ);
+ addRecSignal(GSN_CMVMI_CFGREQ, &Cmvmi::execCMVMI_CFGREQ);
+ addRecSignal(GSN_CLOSE_COMREQ, &Cmvmi::execCLOSE_COMREQ);
+ addRecSignal(GSN_ENABLE_COMORD, &Cmvmi::execENABLE_COMORD);
+ addRecSignal(GSN_OPEN_COMREQ, &Cmvmi::execOPEN_COMREQ);
+ addRecSignal(GSN_SIZEALT_ACK, &Cmvmi::execSIZEALT_ACK);
+ addRecSignal(GSN_TEST_ORD, &Cmvmi::execTEST_ORD);
+
+ addRecSignal(GSN_STATISTICS_REQ, &Cmvmi::execSTATISTICS_REQ);
+ addRecSignal(GSN_TAMPER_ORD, &Cmvmi::execTAMPER_ORD);
+ addRecSignal(GSN_SET_VAR_REQ, &Cmvmi::execSET_VAR_REQ);
+ addRecSignal(GSN_SET_VAR_CONF, &Cmvmi::execSET_VAR_CONF);
+ addRecSignal(GSN_SET_VAR_REF, &Cmvmi::execSET_VAR_REF);
+ addRecSignal(GSN_STOP_ORD, &Cmvmi::execSTOP_ORD);
+ addRecSignal(GSN_START_ORD, &Cmvmi::execSTART_ORD);
+ addRecSignal(GSN_EVENT_SUBSCRIBE_REQ,
+ &Cmvmi::execEVENT_SUBSCRIBE_REQ);
+
+ addRecSignal(GSN_DUMP_STATE_ORD, &Cmvmi::execDUMP_STATE_ORD);
+
+ addRecSignal(GSN_TESTSIG, &Cmvmi::execTESTSIG);
+
+ subscriberPool.setSize(5);
+
+ // Print to stdout/console
+ eventLogger.createConsoleHandler();
+ eventLogger.setCategory("NDB");
+ eventLogger.enable(Logger::LL_INFO, Logger::LL_ALERT); // Log INFO to ALERT
+
+ const ClusterConfiguration::ClusterData & clData =
+ theConfig.clusterConfigurationData() ;
+
+ clogLevel = clData.SizeAltData.logLevel;
+
+ for(Uint32 i= 0; i< clData.SizeAltData.noOfNodes; i++ ){
+ jam();
+ const Uint32 nodeId = clData.nodeData[i].nodeId;
+ switch(clData.nodeData[i].nodeType){
+ case NodeInfo::DB:
+ case NodeInfo::API:
+ case NodeInfo::MGM:
+ case NodeInfo::REP:
+ break;
+ default:
+ ndbrequire(false);
+ }
+ setNodeInfo(nodeId).m_type = clData.nodeData[i].nodeType;
+ }
+}
+
+Cmvmi::~Cmvmi()
+{
+}
+
+
+void Cmvmi::execNDB_TAMPER(Signal* signal)
+{
+ jamEntry();
+ SET_ERROR_INSERT_VALUE(signal->theData[0]);
+ if(ERROR_INSERTED(9999)){
+ CRASH_INSERTION(9999);
+ }
+}//execNDB_TAMPER()
+
+void Cmvmi::execSET_LOGLEVELORD(Signal* signal)
+{
+ SetLogLevelOrd * const llOrd = (SetLogLevelOrd *)&signal->theData[0];
+ LogLevel::EventCategory category;
+ Uint32 level;
+ jamEntry();
+
+ for(unsigned int i = 0; i<llOrd->noOfEntries; i++){
+ category = (LogLevel::EventCategory)llOrd->theCategories[i];
+ level = llOrd->theLevels[i];
+
+ clogLevel.setLogLevel(category, level);
+ }
+}//execSET_LOGLEVELORD()
+
+void Cmvmi::execEVENT_REP(Signal* signal)
+{
+ //-----------------------------------------------------------------------
+ // This message is sent to report any types of events in NDB.
+ // Based on the log level they will be either ignored or
+ // reported. Currently they are printed, but they will be
+ // transferred to the management server for further distribution
+ // to the graphical management interface.
+ //-----------------------------------------------------------------------
+ EventReport * const eventReport = (EventReport *)&signal->theData[0];
+ EventReport::EventType eventType = eventReport->getEventType();
+
+ jamEntry();
+
+ /**
+ * If entry is not found
+ */
+ Uint32 threshold = 16;
+ LogLevel::EventCategory eventCategory = (LogLevel::EventCategory)0;
+
+ for(unsigned int i = 0; i< EventLogger::matrixSize; i++){
+ if(EventLogger::matrix[i].eventType == eventType){
+ eventCategory = EventLogger::matrix[i].eventCategory;
+ threshold = EventLogger::matrix[i].threshold;
+ break;
+ }
+ }
+
+ if(threshold > 15){
+ // No entry found in matrix (or event that should never be printed)
+ return;
+ }
+
+ SubscriberPtr ptr;
+ for(subscribers.first(ptr); ptr.i != RNIL; subscribers.next(ptr)){
+ if(ptr.p->logLevel.getLogLevel(eventCategory) < threshold){
+ continue;
+ }
+
+ sendSignal(ptr.p->blockRef, GSN_EVENT_REP, signal, signal->length(), JBB);
+ }
+
+ if(clogLevel.getLogLevel(eventCategory) < threshold){
+ return;
+ }
+
+ // Print the event info
+ eventLogger.log(eventReport->getEventType(), signal->theData);
+
+}//execEVENT_REP()
+
+void
+Cmvmi::execEVENT_SUBSCRIBE_REQ(Signal * signal){
+ EventSubscribeReq * subReq = (EventSubscribeReq *)&signal->theData[0];
+ SubscriberPtr ptr;
+
+ jamEntry();
+
+ /**
+ * Search for subcription
+ */
+ for(subscribers.first(ptr); ptr.i != RNIL; subscribers.next(ptr)){
+ if(ptr.p->blockRef == subReq->blockRef)
+ break;
+ }
+
+ if(ptr.i == RNIL){
+ /**
+ * Create a new one
+ */
+ if(subscribers.seize(ptr) == false){
+ sendSignal(subReq->blockRef, GSN_EVENT_SUBSCRIBE_REF, signal, 1, JBB);
+ return;
+ }
+ /**
+ * If it's a new subscription, clear the loglevel
+ *
+ * Clear only if noOfEntries is 0, this is needed beacuse we set
+ * the default loglevels for the MGMT nodes during the inital connect phase.
+ * See reportConnected().
+ */
+ if (subReq->noOfEntries == 0){
+ ptr.p->logLevel.clear();
+ }
+
+ ptr.p->blockRef = subReq->blockRef;
+ }
+
+ if(subReq->noOfEntries == 0){
+ /**
+ * Cancel subscription
+ */
+ subscribers.release(ptr.i);
+ } else {
+ /**
+ * Update subscription
+ */
+ LogLevel::EventCategory category;
+ Uint32 level = 0;
+ for(Uint32 i = 0; i<subReq->noOfEntries; i++){
+ category = (LogLevel::EventCategory)subReq->theCategories[i];
+ level = subReq->theLevels[i];
+ ptr.p->logLevel.setLogLevel(category,
+ level);
+ }
+ }
+
+ signal->theData[0] = ptr.i;
+ sendSignal(ptr.p->blockRef, GSN_EVENT_SUBSCRIBE_CONF, signal, 1, JBB);
+}
+
+void
+Cmvmi::cancelSubscription(NodeId nodeId){
+
+ SubscriberPtr ptr;
+ subscribers.first(ptr);
+
+ while(ptr.i != RNIL){
+ Uint32 i = ptr.i;
+ BlockReference blockRef = ptr.p->blockRef;
+
+ subscribers.next(ptr);
+
+ if(refToNode(blockRef) == nodeId){
+ subscribers.release(i);
+ }
+ }
+}
+
+void Cmvmi::sendSTTORRY(Signal* signal)
+{
+ if( theStartPhase == 1 ) {
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData() ;
+ const int myNodeId = globalData.ownId;
+ int MyNodeFound = 0;
+
+ jam();
+
+ CmInit * const cmInit = (CmInit *)&signal->theData[0];
+
+ cmInit->heartbeatDbDb = clusterConf.ispValues[0][2];
+ cmInit->heartbeatDbApi = clusterConf.ispValues[0][3];
+ cmInit->arbitTimeout = clusterConf.ispValues[0][5];
+
+ NodeBitmask::clear(cmInit->allNdbNodes);
+ for(unsigned int i = 0; i < clusterConf.SizeAltData.noOfNodes; i++ ) {
+ jam();
+ if (clusterConf.nodeData[i].nodeType == NodeInfo::DB){
+ jam();
+ const NodeId nodeId = clusterConf.nodeData[i].nodeId;
+ if (nodeId == myNodeId) {
+ jam();
+ MyNodeFound = 1;
+ }//if
+ NodeBitmask::set(cmInit->allNdbNodes, nodeId);
+ }//if
+ }//for
+
+ if (MyNodeFound == 0) {
+ ERROR_SET(fatal, ERR_NODE_NOT_IN_CONFIG, "", "");
+ }//if
+
+ sendSignal(QMGR_REF, GSN_CM_INIT, signal, CmInit::SignalLength, JBB);
+
+ // these do not fit into CM_INIT
+ ArbitSignalData* const sd = (ArbitSignalData*)&signal->theData[0];
+ for (unsigned rank = 1; rank <= 2; rank++) {
+ sd->sender = myNodeId;
+ sd->code = rank;
+ sd->node = 0;
+ sd->ticket.clear();
+ sd->mask.clear();
+ for (int i = 0; i < MAX_NODES; i++) {
+ if (clusterConf.nodeData[i].arbitRank == rank)
+ sd->mask.set(clusterConf.nodeData[i].nodeId);
+ }
+ sendSignal(QMGR_REF, GSN_ARBIT_CFG, signal,
+ ArbitSignalData::SignalLength, JBB);
+ }
+ } else {
+ jam();
+ signal->theData[0] = theSignalKey;
+ signal->theData[3] = 1;
+ signal->theData[4] = 3;
+ signal->theData[5] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal,6, JBB);
+ }
+}//Cmvmi::sendSTTORRY
+
+
+// Received a restart signal.
+// Answer it like any other block
+// PR0 : StartCase
+// DR0 : StartPhase
+// DR1 : ?
+// DR2 : ?
+// DR3 : ?
+// DR4 : ?
+// DR5 : SignalKey
+
+void Cmvmi::execSTTOR_Local(Signal* signal)
+{
+ theStartPhase = signal->theData[1];
+ theSignalKey = signal->theData[6];
+
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData();
+ jamEntry();
+ if (theStartPhase == 1 && clusterConf.SizeAltData.exist == true){
+ jam();
+ signalCount = 0;
+ execSIZEALT_ACK(signal);
+ return;
+ } else if (theStartPhase == 3) {
+ jam();
+ globalData.activateSendPacked = 1;
+ sendSTTORRY(signal);
+ } else {
+ jam();
+ sendSTTORRY(signal);
+ }
+}
+
+void Cmvmi::execSIZEALT_ACK(Signal* signal)
+{
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData();
+ jamEntry();
+
+ if (signalCount < NDB_SIZEALT_OFF){
+ jam();
+ BlockNumber blockNo = clusterConf.SizeAltData.blockNo[signalCount];
+ signal->theData[0] = CMVMI_REF;
+
+ /**
+ * This send SizeAlt(s) to blocks
+ * Definition of data content can be found in SignalData/XXXSizeAltReq.H
+ */
+ const unsigned int noOfWords = 20;
+ for(unsigned int i = 1; i<noOfWords; i++){
+ signal->theData[i] = clusterConf.SizeAltData.varSize[signalCount][i].nrr;
+ }
+
+ signalCount++;
+ sendSignal(numberToRef(blockNo, 0), GSN_SIZEALT_REP, signal,21, JBB);
+ } else {
+ jam();
+ sendSTTORRY(signal);
+ }
+}
+
+void Cmvmi::execCM_INFOREQ(Signal* signal)
+{
+ int id = signal->theData[1];
+ const BlockReference userRef = signal->theData[0];
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData();
+ const int myNodeId = globalData.ownId;
+
+ jamEntry();
+ signal->theData[0] = id;
+
+ for(unsigned int i= 0; i< clusterConf.SizeAltData.noOfNodes; i++ ) {
+ jam();
+ if (clusterConf.nodeData[i].nodeType == NodeInfo::DB){
+ NodeId nodeId = clusterConf.nodeData[i].nodeId;
+ if (nodeId != myNodeId) {
+ jam();
+ globalTransporterRegistry.setPerformState(nodeId, PerformConnect);
+ }
+ }
+ }
+
+ sendSignal(userRef, GSN_CM_INFOCONF, signal, 1, JBB);
+}
+
+void Cmvmi::execCM_RUN(Signal* signal)
+{
+ jamEntry();
+ if (signal->theData[0] == 0) {
+ jam();
+ signal->theData[0] = theSignalKey;
+ signal->theData[3] = 1;
+ signal->theData[4] = 3;
+ signal->theData[5] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 6, JBB);
+ } else {
+ globalData.theStartLevel = NodeState::SL_STARTED;
+
+ // Connect to all application nodes.
+ // Enable communication with all NDB blocks.
+
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData();
+ jam();
+ for(unsigned int i= 0; i< clusterConf.SizeAltData.noOfNodes; i++ ) {
+ NodeId nodeId = clusterConf.nodeData[i].nodeId;
+ jam();
+ if (clusterConf.nodeData[i].nodeType != NodeInfo::DB &&
+ clusterConf.nodeData[i].nodeType != NodeInfo::MGM){
+
+ jam();
+ globalTransporterRegistry.setPerformState(nodeId, PerformConnect);
+ globalTransporterRegistry.setIOState(nodeId, HaltIO);
+ //-----------------------------------------------------
+ // Report that the connection to the node is opened
+ //-----------------------------------------------------
+ signal->theData[0] = EventReport::CommunicationOpened;
+ signal->theData[1] = clusterConf.nodeData[i].nodeId;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+ //-----------------------------------------------------
+ }
+ }
+ }
+}
+
+void Cmvmi::execCMVMI_CFGREQ(Signal* signal)
+{
+ const BlockReference userRef = signal->theData[0];
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData();
+
+ int theStart_phase = signal->theData[1];
+
+ jamEntry();
+
+ CmvmiCfgConf * const cfgConf = (CmvmiCfgConf *)&signal->theData[0];
+
+ cfgConf->startPhase = theStart_phase;
+ for(unsigned int i = 0; i<CmvmiCfgConf::NO_OF_WORDS; i++)
+ cfgConf->theData[i] = clusterConf.ispValues[theStart_phase][i];
+
+ sendSignal(userRef, GSN_CMVMI_CFGCONF, signal, CmvmiCfgConf::LENGTH,JBB );
+}
+
+void Cmvmi::execCLOSE_COMREQ(Signal* signal)
+{
+ // Close communication with the node and halt input/output from
+ // other blocks than QMGR
+
+ CloseComReqConf * const closeCom = (CloseComReqConf *)&signal->theData[0];
+
+ const BlockReference userRef = closeCom->xxxBlockRef;
+ Uint32 failNo = closeCom->failNo;
+// Uint32 noOfNodes = closeCom->noOfNodes;
+
+ jamEntry();
+ for (unsigned i = 0; i < MAX_NODES; i++){
+ if(NodeBitmask::get(closeCom->theNodes, i)){
+
+ jam();
+
+ //-----------------------------------------------------
+ // Report that the connection to the node is closed
+ //-----------------------------------------------------
+ signal->theData[0] = EventReport::CommunicationClosed;
+ signal->theData[1] = i;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ globalTransporterRegistry.setIOState(i, HaltIO);
+ globalTransporterRegistry.setPerformState(i, PerformDisconnect);
+
+ /**
+ * Cancel possible event subscription
+ */
+ cancelSubscription(i);
+ }
+ }
+ if (failNo != 0) {
+ jam();
+ signal->theData[0] = userRef;
+ signal->theData[1] = failNo;
+ sendSignal(QMGR_REF, GSN_CLOSE_COMCONF, signal, 19, JBA);
+ }
+}
+
+void Cmvmi::execOPEN_COMREQ(Signal* signal)
+{
+ // Connect to the specifed NDB node, only QMGR allowed communication
+ // so far with the node
+
+ const BlockReference userRef = signal->theData[0];
+ Uint32 tStartingNode = signal->theData[1];
+
+ jamEntry();
+ if (userRef != 0) {
+ jam();
+ signal->theData[0] = signal->theData[1];
+ sendSignal(userRef, GSN_OPEN_COMCONF, signal, 2,JBA);
+ }
+ globalTransporterRegistry.setPerformState(tStartingNode, PerformConnect);
+ //-----------------------------------------------------
+ // Report that the connection to the node is opened
+ //-----------------------------------------------------
+ signal->theData[0] = EventReport::CommunicationOpened;
+ signal->theData[1] = tStartingNode;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+ //-----------------------------------------------------
+}
+
+void Cmvmi::execENABLE_COMORD(Signal* signal)
+{
+ // Enable communication with all our NDB blocks to this node
+
+ Uint32 tStartingNode = signal->theData[0];
+ globalTransporterRegistry.setIOState(tStartingNode, NoHalt);
+ setNodeInfo(tStartingNode).m_connected = true;
+ //-----------------------------------------------------
+ // Report that the version of the node
+ //-----------------------------------------------------
+ signal->theData[0] = EventReport::ConnectedApiVersion;
+ signal->theData[1] = tStartingNode;
+ signal->theData[2] = getNodeInfo(tStartingNode).m_version;
+
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+ //-----------------------------------------------------
+
+ jamEntry();
+}
+
+void Cmvmi::execDISCONNECT_REP(Signal *signal)
+{
+ const DisconnectRep * const rep = (DisconnectRep *)&signal->theData[0];
+ const Uint32 hostId = rep->nodeId;
+ const Uint32 errNo = rep->err;
+
+ jamEntry();
+
+ setNodeInfo(hostId).m_connected = false;
+ setNodeInfo(hostId).m_connectCount++;
+ const NodeInfo::NodeType type = getNodeInfo(hostId).getType();
+ ndbrequire(type != NodeInfo::INVALID);
+
+ if (globalTransporterRegistry.performState(hostId) != PerformDisconnect) {
+ jam();
+
+ // -------------------------------------------------------------------
+ // We do not report the disconnection when disconnection is already ongoing.
+ // This reporting should be looked into but this secures that we avoid
+ // crashes due to too quick re-reporting of disconnection.
+ // -------------------------------------------------------------------
+ if(type == NodeInfo::DB || globalData.theStartLevel == NodeState::SL_STARTED){
+ jam();
+ DisconnectRep * const rep = (DisconnectRep *)&signal->theData[0];
+ rep->nodeId = hostId;
+ rep->err = errNo;
+ sendSignal(QMGR_REF, GSN_DISCONNECT_REP, signal,
+ DisconnectRep::SignalLength, JBA);
+ globalTransporterRegistry.setPerformState(hostId, PerformDisconnect);
+ } else if(globalData.theStartLevel == NodeState::SL_CMVMI ||
+ globalData.theStartLevel == NodeState::SL_STARTING) {
+ /**
+ * Someone disconnected during cmvmi period
+ */
+ if(type == NodeInfo::MGM){
+ jam();
+ globalTransporterRegistry.setPerformState(hostId, PerformConnect);
+ } else {
+ globalTransporterRegistry.setPerformState(hostId, PerformDisconnect);
+ }
+ }
+ }
+
+ signal->theData[0] = EventReport::Disconnected;
+ signal->theData[1] = hostId;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+}
+
+void Cmvmi::execCONNECT_REP(Signal *signal){
+ const Uint32 hostId = signal->theData[0];
+
+ jamEntry();
+
+ const NodeInfo::NodeType type = (NodeInfo::NodeType)getNodeInfo(hostId).m_type;
+ ndbrequire(type != NodeInfo::INVALID);
+ globalData.m_nodeInfo[hostId].m_version = 0;
+ globalData.m_nodeInfo[hostId].m_signalVersion = 0;
+
+ if(type == NodeInfo::DB || globalData.theStartLevel >= NodeState::SL_STARTED){
+ jam();
+
+ /**
+ * Inform QMGR that client has connected
+ */
+
+ signal->theData[0] = hostId;
+ sendSignal(QMGR_REF, GSN_CONNECT_REP, signal, 1, JBA);
+ } else if(globalData.theStartLevel == NodeState::SL_CMVMI ||
+ globalData.theStartLevel == NodeState::SL_STARTING) {
+ jam();
+ /**
+ * Someone connected before start was finished
+ */
+ if(type == NodeInfo::MGM){
+ jam();
+ } else {
+ /**
+ * Dont allow api nodes to connect
+ */
+ globalTransporterRegistry.setPerformState(hostId, PerformDisconnect);
+ }
+ }
+
+ /* Automatically subscribe events for MGM nodes.
+ */
+ if(type == NodeInfo::MGM){
+ jam();
+ globalTransporterRegistry.setIOState(hostId, NoHalt);
+
+ EventSubscribeReq* dst = (EventSubscribeReq *)&signal->theData[0];
+
+ for (Uint32 i = 0; i < EventLogger::defEventLogMatrixSize; i++) {
+ dst->theCategories[i] = EventLogger::defEventLogMatrix[i].eventCategory;
+ dst->theLevels[i] = EventLogger::defEventLogMatrix[i].threshold;
+ }
+
+ dst->noOfEntries = EventLogger::defEventLogMatrixSize;
+ /* The BlockNumber is hardcoded as 1 in MgmtSrvr */
+ dst->blockRef = numberToRef(MIN_API_BLOCK_NO, hostId);
+
+ execEVENT_SUBSCRIBE_REQ(signal);
+
+ }
+
+ //------------------------------------------
+ // Also report this event to the Event handler
+ //------------------------------------------
+ signal->theData[0] = EventReport::Connected;
+ signal->theData[1] = hostId;
+ signal->header.theLength = 2;
+
+ execEVENT_REP(signal);
+}
+
+#ifdef VM_TRACE
+void
+modifySignalLogger(bool allBlocks, BlockNumber bno,
+ TestOrd::Command cmd,
+ TestOrd::SignalLoggerSpecification spec){
+ SignalLoggerManager::LogMode logMode;
+
+ /**
+ * Mapping between SignalLoggerManager::LogMode and
+ * TestOrd::SignalLoggerSpecification
+ */
+ switch(spec){
+ case TestOrd::InputSignals:
+ logMode = SignalLoggerManager::LogIn;
+ break;
+ case TestOrd::OutputSignals:
+ logMode = SignalLoggerManager::LogOut;
+ break;
+ case TestOrd::InputOutputSignals:
+ logMode = SignalLoggerManager::LogInOut;
+ break;
+ default:
+ return;
+ break;
+ }
+
+ switch(cmd){
+ case TestOrd::On:
+ globalSignalLoggers.logOn(allBlocks, bno, logMode);
+ break;
+ case TestOrd::Off:
+ globalSignalLoggers.logOff(allBlocks, bno, logMode);
+ break;
+ case TestOrd::Toggle:
+ globalSignalLoggers.logToggle(allBlocks, bno, logMode);
+ break;
+ case TestOrd::KeepUnchanged:
+ // Do nothing
+ break;
+ }
+ globalSignalLoggers.flushSignalLog();
+}
+#endif
+
+void
+Cmvmi::execTEST_ORD(Signal * signal){
+ jamEntry();
+
+#ifdef VM_TRACE
+ TestOrd * const testOrd = (TestOrd *)&signal->theData[0];
+
+ TestOrd::Command cmd;
+
+ {
+ /**
+ * Process Trace command
+ */
+ TestOrd::TraceSpecification traceSpec;
+
+ testOrd->getTraceCommand(cmd, traceSpec);
+ unsigned long traceVal = traceSpec;
+ unsigned long currentTraceVal = globalSignalLoggers.getTrace();
+ switch(cmd){
+ case TestOrd::On:
+ currentTraceVal |= traceVal;
+ break;
+ case TestOrd::Off:
+ currentTraceVal &= (~traceVal);
+ break;
+ case TestOrd::Toggle:
+ currentTraceVal ^= traceVal;
+ break;
+ case TestOrd::KeepUnchanged:
+ // Do nothing
+ break;
+ }
+ globalSignalLoggers.setTrace(currentTraceVal);
+ }
+
+ {
+ /**
+ * Process Log command
+ */
+ TestOrd::SignalLoggerSpecification logSpec;
+ BlockNumber bno;
+ unsigned int loggers = testOrd->getNoOfSignalLoggerCommands();
+
+ if(loggers == (unsigned)~0){ // Apply command to all blocks
+ testOrd->getSignalLoggerCommand(0, bno, cmd, logSpec);
+ modifySignalLogger(true, bno, cmd, logSpec);
+ } else {
+ for(unsigned int i = 0; i<loggers; i++){
+ testOrd->getSignalLoggerCommand(i, bno, cmd, logSpec);
+ modifySignalLogger(false, bno, cmd, logSpec);
+ }
+ }
+ }
+
+ {
+ /**
+ * Process test command
+ */
+ testOrd->getTestCommand(cmd);
+ switch(cmd){
+ case TestOrd::On:{
+ SET_GLOBAL_TEST_ON;
+ }
+ break;
+ case TestOrd::Off:{
+ SET_GLOBAL_TEST_OFF;
+ }
+ break;
+ case TestOrd::Toggle:{
+ TOGGLE_GLOBAL_TEST_FLAG;
+ }
+ break;
+ case TestOrd::KeepUnchanged:
+ // Do nothing
+ break;
+ }
+ }
+
+#endif
+}
+
+void Cmvmi::execSTATISTICS_REQ(Signal* signal)
+{
+ // TODO Note ! This is only a test implementation...
+
+ static int stat1 = 0;
+ jamEntry();
+
+ //ndbout << "data 1: " << signal->theData[1];
+
+ int x = signal->theData[0];
+ stat1++;
+ signal->theData[0] = stat1;
+ sendSignal(x, GSN_STATISTICS_CONF, signal, 7, JBB);
+
+}//execSTATISTICS_REQ()
+
+
+
+void Cmvmi::execSTOP_ORD(Signal* signal)
+{
+ jamEntry();
+ globalData.theRestartFlag = perform_stop;
+}//execSTOP_ORD()
+
+void
+Cmvmi::execSTART_ORD(Signal* signal) {
+
+ StartOrd * const startOrd = (StartOrd *)&signal->theData[0];
+ jamEntry();
+
+ Uint32 tmp = startOrd->restartInfo;
+ if(StopReq::getPerformRestart(tmp)){
+ jam();
+ /**
+ *
+ */
+ NdbRestartType type = NRT_Default;
+ if(StopReq::getNoStart(tmp) && StopReq::getInitialStart(tmp))
+ type = NRT_NoStart_InitialStart;
+ if(StopReq::getNoStart(tmp) && !StopReq::getInitialStart(tmp))
+ type = NRT_NoStart_Restart;
+ if(!StopReq::getNoStart(tmp) && StopReq::getInitialStart(tmp))
+ type = NRT_DoStart_InitialStart;
+ if(!StopReq::getNoStart(tmp)&&!StopReq::getInitialStart(tmp))
+ type = NRT_DoStart_Restart;
+ NdbShutdown(NST_Restart, type);
+ }
+
+ if(globalData.theRestartFlag == system_started){
+ jam()
+ /**
+ * START_ORD received when already started(ignored)
+ */
+ //ndbout << "START_ORD received when already started(ignored)" << endl;
+ return;
+ }
+
+ if(globalData.theRestartFlag == perform_stop){
+ jam()
+ /**
+ * START_ORD received when stopping(ignored)
+ */
+ //ndbout << "START_ORD received when stopping(ignored)" << endl;
+ return;
+ }
+
+ if(globalData.theStartLevel == NodeState::SL_NOTHING){
+ jam();
+ globalData.theStartLevel = NodeState::SL_CMVMI;
+ /**
+ * Open connections to management servers
+ */
+
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData() ;
+
+ for(unsigned int i= 0; i < clusterConf.SizeAltData.noOfNodes; i++ ){
+ NodeId nodeId = clusterConf.nodeData[i].nodeId;
+
+ if (clusterConf.nodeData[i].nodeType == NodeInfo::MGM){
+
+ if(globalTransporterRegistry.performState(nodeId) != PerformIO){
+ globalTransporterRegistry.setPerformState(nodeId, PerformConnect);
+ globalTransporterRegistry.setIOState(nodeId, NoHalt);
+ }
+ }
+ }
+ return ;
+ }
+
+ if(globalData.theStartLevel == NodeState::SL_CMVMI){
+ jam();
+ globalData.theStartLevel = NodeState::SL_STARTING;
+ globalData.theRestartFlag = system_started;
+ /**
+ * StartLevel 1
+ *
+ * Do Restart
+ */
+
+ globalScheduler.clear();
+ globalTimeQueue.clear();
+
+ // Disconnect all nodes as part of the system restart.
+ // We need to ensure that we are starting up
+ // without any connected nodes.
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData() ;
+ const int myNodeId = globalData.ownId;
+
+ for(unsigned int i= 0; i < clusterConf.SizeAltData.noOfNodes; i++ ){
+ NodeId nodeId = clusterConf.nodeData[i].nodeId;
+ if (myNodeId != nodeId &&
+ clusterConf.nodeData[i].nodeType != NodeInfo::MGM){
+
+ globalTransporterRegistry.setPerformState(nodeId, PerformDisconnect);
+ globalTransporterRegistry.setIOState(nodeId, HaltIO);
+ }
+ }
+
+ /**
+ * Start running startphases
+ */
+ sendSignal(NDBCNTR_REF, GSN_START_ORD, signal, 1, JBA);
+ return;
+ }
+}//execSTART_ORD()
+
+void Cmvmi::execTAMPER_ORD(Signal* signal)
+{
+ jamEntry();
+ // TODO We should maybe introduce a CONF and REF signal
+ // to be able to indicate if we really introduced an error.
+#ifdef ERROR_INSERT
+ TamperOrd* const tamperOrd = (TamperOrd*)&signal->theData[0];
+
+ signal->theData[1] = tamperOrd->errorNo;
+ signal->theData[0] = 5;
+ sendSignal(DBDIH_REF, GSN_DIHNDBTAMPER, signal, 3,JBB);
+#endif
+
+}//execTAMPER_ORD()
+
+
+
+void Cmvmi::execSET_VAR_REQ(Signal* signal)
+{
+
+ SetVarReq* const setVarReq = (SetVarReq*)&signal->theData[0];
+ ConfigParamId var = setVarReq->variable();
+ jamEntry();
+ switch (var) {
+
+ // NDBCNTR_REF
+
+ // DBTC
+ case TransactionDeadlockDetectionTimeout:
+ case TransactionInactiveTime:
+ case NoOfConcurrentProcessesHandleTakeover:
+ sendSignal(DBTC_REF, GSN_SET_VAR_REQ, signal, 3, JBB);
+ break;
+
+ // DBDIH
+ case TimeBetweenLocalCheckpoints:
+ case TimeBetweenGlobalCheckpoints:
+ sendSignal(DBDIH_REF, GSN_SET_VAR_REQ, signal, 3, JBB);
+ break;
+
+ // DBLQH
+ case NoOfConcurrentCheckpointsDuringRestart:
+ case NoOfConcurrentCheckpointsAfterRestart:
+ sendSignal(DBLQH_REF, GSN_SET_VAR_REQ, signal, 3, JBB);
+ break;
+
+ // DBACC
+ case NoOfDiskPagesToDiskDuringRestartACC:
+ case NoOfDiskPagesToDiskAfterRestartACC:
+ sendSignal(DBACC_REF, GSN_SET_VAR_REQ, signal, 3, JBB);
+ break;
+
+ // DBTUP
+ case NoOfDiskPagesToDiskDuringRestartTUP:
+ case NoOfDiskPagesToDiskAfterRestartTUP:
+ sendSignal(DBTUP_REF, GSN_SET_VAR_REQ, signal, 3, JBB);
+ break;
+
+ // DBDICT
+
+ // NDBCNTR
+ case TimeToWaitAlive:
+
+ // QMGR
+ case HeartbeatIntervalDbDb: // TODO ev till Ndbcnt också
+ case HeartbeatIntervalDbApi:
+ case ArbitTimeout:
+ sendSignal(QMGR_REF, GSN_SET_VAR_REQ, signal, 3, JBB);
+ break;
+
+ // NDBFS
+
+ // CMVMI
+ case MaxNoOfSavedMessages:
+ case LockPagesInMainMemory:
+ case TimeBetweenWatchDogCheck:
+ case StopOnError:
+ handleSET_VAR_REQ(signal);
+ break;
+
+
+ // Not possible to update (this could of course be handled by each block
+ // instead but I havn't investigated where they belong)
+ case Id:
+ case ExecuteOnComputer:
+ case ShmKey:
+ case MaxNoOfConcurrentOperations:
+ case MaxNoOfConcurrentTransactions:
+ case MemorySpaceIndexes:
+ case MemorySpaceTuples:
+ case MemoryDiskPages:
+ case NoOfFreeDiskClusters:
+ case NoOfDiskClusters:
+ case NoOfFragmentLogFiles:
+ case NoOfDiskClustersPerDiskFile:
+ case NoOfDiskFiles:
+ case MaxNoOfSavedEvents:
+ default:
+
+ int mgmtSrvr = setVarReq->mgmtSrvrBlockRef();
+ sendSignal(mgmtSrvr, GSN_SET_VAR_REF, signal, 0, JBB);
+ } // switch
+
+
+}//execSET_VAR_REQ()
+
+
+void Cmvmi::execSET_VAR_CONF(Signal* signal)
+{
+ int mgmtSrvr = signal->theData[0];
+ sendSignal(mgmtSrvr, GSN_SET_VAR_CONF, signal, 0, JBB);
+
+}//execSET_VAR_CONF()
+
+
+void Cmvmi::execSET_VAR_REF(Signal* signal)
+{
+ int mgmtSrvr = signal->theData[0];
+ sendSignal(mgmtSrvr, GSN_SET_VAR_REF, signal, 0, JBB);
+
+}//execSET_VAR_REF()
+
+
+void Cmvmi::handleSET_VAR_REQ(Signal* signal) {
+
+ SetVarReq* const setVarReq = (SetVarReq*)&signal->theData[0];
+ ConfigParamId var = setVarReq->variable();
+ int val = setVarReq->value();
+
+ switch (var) {
+ case MaxNoOfSavedMessages:
+ theConfig.maxNoOfErrorLogs(val);
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ case LockPagesInMainMemory:
+ int result;
+ if (val == 0) {
+ result = NdbMem_MemUnlockAll();
+ }
+ else {
+ result = NdbMem_MemLockAll();
+ }
+ if (result == 0) {
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ }
+ else {
+ sendSignal(CMVMI_REF, GSN_SET_VAR_REF, signal, 1, JBB);
+ }
+ break;
+
+ case TimeBetweenWatchDogCheck:
+ theConfig.timeBetweenWatchDogCheck(val);
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ case StopOnError:
+ theConfig.stopOnError(val);
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ default:
+ sendSignal(CMVMI_REF, GSN_SET_VAR_REF, signal, 1, JBB);
+ return;
+ } // switch
+
+}
+
+#ifdef VM_TRACE
+class RefSignalTest {
+public:
+ enum ErrorCode {
+ OK = 0,
+ NF_FakeErrorREF = 7
+ };
+ Uint32 senderRef;
+ Uint32 senderData;
+ Uint32 errorCode;
+};
+#endif
+
+void
+Cmvmi::execDUMP_STATE_ORD(Signal* signal)
+{
+
+ sendSignal(QMGR_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(NDBCNTR_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(DBTC_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(DBDIH_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(DBDICT_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(DBLQH_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(DBTUP_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(DBACC_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(NDBFS_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(BACKUP_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(DBUTIL_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(SUMA_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(GREP_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(TRIX_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+ sendSignal(DBTUX_REF, GSN_DUMP_STATE_ORD, signal, signal->length(), JBB);
+
+ /**
+ *
+ * Here I can dump CMVMI state if needed
+ */
+ if(signal->theData[0] == 13){
+ infoEvent("Cmvmi: signalCount = %d", signalCount);
+ }
+
+ DumpStateOrd * const & dumpState = (DumpStateOrd *)&signal->theData[0];
+ if (dumpState->args[0] == DumpStateOrd::CmvmiDumpConnections){
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData() ;
+
+ for(unsigned int i= 0; i < clusterConf.SizeAltData.noOfNodes; i++ ){
+ NodeId nodeId = clusterConf.nodeData[i].nodeId;
+
+ const char* nodeTypeStr = "";
+ switch(clusterConf.nodeData[i].nodeType){
+ case NodeInfo::DB:
+ nodeTypeStr = "DB";
+ break;
+ case NodeInfo::API:
+ nodeTypeStr = "API";
+ break;
+ case NodeInfo::MGM:
+ nodeTypeStr = "MGM";
+ break;
+ case NodeInfo::REP:
+ nodeTypeStr = "REP";
+ break;
+ default:
+ nodeTypeStr = "<UNKNOWN>";
+ }
+
+ const char* actionStr = "";
+ switch (globalTransporterRegistry.performState(nodeId)){
+ case PerformNothing:
+ actionStr = "does nothing";
+ break;
+ case PerformIO:
+ actionStr = "is connected";
+ break;
+ case PerformConnect:
+ actionStr = "is trying to connect";
+ break;
+ case PerformDisconnect:
+ actionStr = "is trying to disconnect";
+ break;
+ case RemoveTransporter:
+ actionStr = "will be removed";
+ break;
+ }
+
+ infoEvent("Connection to %d (%s) %s",
+ nodeId,
+ nodeTypeStr,
+ actionStr);
+ }
+ }
+
+ if (dumpState->args[0] == DumpStateOrd::CmvmiDumpLongSignalMemory){
+ infoEvent("Cmvmi: g_sectionSegmentPool size: %d free: %d",
+ g_sectionSegmentPool.getSize(),
+ g_sectionSegmentPool.getNoOfFree());
+ }
+
+ if (dumpState->args[0] == DumpStateOrd::CmvmiSetRestartOnErrorInsert){
+ if(signal->getLength() == 1)
+ theConfig.setRestartOnErrorInsert((int)NRT_NoStart_Restart);
+ else
+ theConfig.setRestartOnErrorInsert(signal->theData[1]);
+ }
+
+ if (dumpState->args[0] == DumpStateOrd::CmvmiTestLongSigWithDelay) {
+ Uint32 loopCount = dumpState->args[1];
+ const unsigned len0 = 11;
+ const unsigned len1 = 123;
+ Uint32 sec0[len0];
+ Uint32 sec1[len1];
+ for (unsigned i = 0; i < len0; i++)
+ sec0[i] = i;
+ for (unsigned i = 0; i < len1; i++)
+ sec1[i] = 16 * i;
+ Uint32* sig = signal->getDataPtrSend();
+ sig[0] = reference();
+ sig[1] = 20; // test type
+ sig[2] = 0;
+ sig[3] = 0;
+ sig[4] = loopCount;
+ sig[5] = len0;
+ sig[6] = len1;
+ sig[7] = 0;
+ LinearSectionPtr ptr[3];
+ ptr[0].p = sec0;
+ ptr[0].sz = len0;
+ ptr[1].p = sec1;
+ ptr[1].sz = len1;
+ sendSignal(reference(), GSN_TESTSIG, signal, 8, JBB, ptr, 2);
+ }
+
+#ifdef VM_TRACE
+#if 0
+ {
+ SafeCounterManager mgr(* this); mgr.setSize(1);
+ SafeCounterHandle handle;
+
+ {
+ SafeCounter tmp(mgr, handle);
+ tmp.init<RefSignalTest>(CMVMI, GSN_TESTSIG, /* senderData */ 13);
+ tmp.setWaitingFor(3);
+ ndbrequire(!tmp.done());
+ ndbout_c("Allocted");
+ }
+ ndbrequire(!handle.done());
+ {
+ SafeCounter tmp(mgr, handle);
+ tmp.clearWaitingFor(3);
+ ndbrequire(tmp.done());
+ ndbout_c("Deallocted");
+ }
+ ndbrequire(handle.done());
+ }
+#endif
+#endif
+}//Cmvmi::execDUMP_STATE_ORD()
+
+
+BLOCK_FUNCTIONS(Cmvmi);
+
+static Uint32 g_print;
+static LinearSectionPtr g_test[3];
+
+void
+Cmvmi::execTESTSIG(Signal* signal){
+ /**
+ * Test of SafeCounter
+ */
+ jamEntry();
+
+ if(!assembleFragments(signal)){
+ jam();
+ return;
+ }
+
+ Uint32 ref = signal->theData[0];
+ Uint32 testType = signal->theData[1];
+ Uint32 fragmentLength = signal->theData[2];
+ g_print = signal->theData[3];
+// Uint32 returnCount = signal->theData[4];
+ Uint32 * secSizes = &signal->theData[5];
+
+ if(g_print){
+ SignalLoggerManager::printSignalHeader(stdout,
+ signal->header,
+ 0,
+ getOwnNodeId(),
+ true);
+ ndbout_c("-- Fixed section --");
+ for(Uint32 i = 0; i<signal->length(); i++){
+ fprintf(stdout, "H'0x%.8x ", signal->theData[i]);
+ if(((i + 1) % 6) == 0)
+ fprintf(stdout, "\n");
+ }
+ fprintf(stdout, "\n");
+
+ for(Uint32 i = 0; i<signal->header.m_noOfSections; i++){
+ SegmentedSectionPtr ptr;
+ ndbout_c("-- Section %d --", i);
+ signal->getSection(ptr, i);
+ ndbrequire(ptr.p != 0);
+ print(ptr, stdout);
+ ndbrequire(ptr.sz == secSizes[i]);
+ }
+ }
+
+ /**
+ * Validate length:s
+ */
+ for(Uint32 i = 0; i<signal->header.m_noOfSections; i++){
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr, i);
+ ndbrequire(ptr.p != 0);
+ ndbrequire(ptr.sz == secSizes[i]);
+ }
+
+ /**
+ * Testing send with delay.
+ */
+ if (testType == 20) {
+ if (signal->theData[4] == 0) {
+ releaseSections(signal);
+ return;
+ }
+ signal->theData[4]--;
+ sendSignalWithDelay(reference(), GSN_TESTSIG, signal, 100, 8);
+ return;
+ }
+
+ NodeReceiverGroup rg; rg.m_block = CMVMI;
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfig.clusterConfigurationData() ;
+ for(unsigned int i = 0; i < clusterConf.SizeAltData.noOfNodes; i++ ){
+ NodeId nodeId = clusterConf.nodeData[i].nodeId;
+ if (clusterConf.nodeData[i].nodeType == NodeInfo::DB){
+ rg.m_nodes.set(nodeId);
+ }
+ }
+
+ if(signal->getSendersBlockRef() == ref){
+ /**
+ * Signal from API (not via NodeReceiverGroup)
+ */
+ if((testType % 2) == 1){
+ signal->theData[4] = 1;
+ } else {
+ signal->theData[1] --;
+ signal->theData[4] = rg.m_nodes.count();
+ }
+ }
+
+ switch(testType){
+ case 1:
+ sendSignal(ref, GSN_TESTSIG, signal, signal->length(), JBB);
+ break;
+ case 2:
+ sendSignal(rg, GSN_TESTSIG, signal, signal->length(), JBB);
+ break;
+ case 3:
+ case 4:{
+ LinearSectionPtr ptr[3];
+ const Uint32 secs = signal->getNoOfSections();
+ for(Uint32 i = 0; i<secs; i++){
+ SegmentedSectionPtr sptr;
+ signal->getSection(sptr, i);
+ ptr[i].sz = sptr.sz;
+ ptr[i].p = new Uint32[sptr.sz];
+ copy(ptr[i].p, sptr);
+ }
+
+ if(testType == 3){
+ sendSignal(ref, GSN_TESTSIG, signal, signal->length(), JBB, ptr, secs);
+ } else {
+ sendSignal(rg, GSN_TESTSIG, signal, signal->length(), JBB, ptr, secs);
+ }
+ for(Uint32 i = 0; i<secs; i++){
+ delete[] ptr[i].p;
+ }
+ break;
+ }
+ case 5:
+ case 6:{
+
+ NodeReceiverGroup tmp;
+ if(testType == 5){
+ tmp = ref;
+ } else {
+ tmp = rg;
+ }
+
+ FragmentSendInfo fragSend;
+ sendFirstFragment(fragSend,
+ tmp,
+ GSN_TESTSIG,
+ signal,
+ signal->length(),
+ JBB,
+ fragmentLength);
+ int count = 1;
+ while(fragSend.m_status != FragmentSendInfo::SendComplete){
+ count++;
+ if(g_print)
+ ndbout_c("Sending fragment %d", count);
+ sendNextSegmentedFragment(signal, fragSend);
+ }
+ break;
+ }
+ case 7:
+ case 8:{
+ LinearSectionPtr ptr[3];
+ const Uint32 secs = signal->getNoOfSections();
+ for(Uint32 i = 0; i<secs; i++){
+ SegmentedSectionPtr sptr;
+ signal->getSection(sptr, i);
+ ptr[i].sz = sptr.sz;
+ ptr[i].p = new Uint32[sptr.sz];
+ copy(ptr[i].p, sptr);
+ }
+
+ NodeReceiverGroup tmp;
+ if(testType == 7){
+ tmp = ref;
+ } else {
+ tmp = rg;
+ }
+
+ FragmentSendInfo fragSend;
+ sendFirstFragment(fragSend,
+ tmp,
+ GSN_TESTSIG,
+ signal,
+ signal->length(),
+ JBB,
+ ptr,
+ secs,
+ fragmentLength);
+
+ int count = 1;
+ while(fragSend.m_status != FragmentSendInfo::SendComplete){
+ count++;
+ if(g_print)
+ ndbout_c("Sending fragment %d", count);
+ sendNextLinearFragment(signal, fragSend);
+ }
+
+ for(Uint32 i = 0; i<secs; i++){
+ delete[] ptr[i].p;
+ }
+ break;
+ }
+ case 9:
+ case 10:{
+
+ Callback m_callBack;
+ m_callBack.m_callbackFunction =
+ safe_cast(&Cmvmi::sendFragmentedComplete);
+
+ if(testType == 9){
+ m_callBack.m_callbackData = 9;
+ sendFragmentedSignal(ref,
+ GSN_TESTSIG, signal, signal->length(), JBB,
+ m_callBack,
+ fragmentLength);
+ } else {
+ m_callBack.m_callbackData = 10;
+ sendFragmentedSignal(rg,
+ GSN_TESTSIG, signal, signal->length(), JBB,
+ m_callBack,
+ fragmentLength);
+ }
+ break;
+ }
+ case 11:
+ case 12:{
+
+ const Uint32 secs = signal->getNoOfSections();
+ memset(g_test, 0, sizeof(g_test));
+ for(Uint32 i = 0; i<secs; i++){
+ SegmentedSectionPtr sptr;
+ signal->getSection(sptr, i);
+ g_test[i].sz = sptr.sz;
+ g_test[i].p = new Uint32[sptr.sz];
+ copy(g_test[i].p, sptr);
+ }
+
+
+ Callback m_callBack;
+ m_callBack.m_callbackFunction =
+ safe_cast(&Cmvmi::sendFragmentedComplete);
+
+ if(testType == 11){
+ m_callBack.m_callbackData = 11;
+ sendFragmentedSignal(ref,
+ GSN_TESTSIG, signal, signal->length(), JBB,
+ g_test, secs,
+ m_callBack,
+ fragmentLength);
+ } else {
+ m_callBack.m_callbackData = 12;
+ sendFragmentedSignal(rg,
+ GSN_TESTSIG, signal, signal->length(), JBB,
+ g_test, secs,
+ m_callBack,
+ fragmentLength);
+ }
+ break;
+ }
+ default:
+ ndbrequire(false);
+ }
+ return;
+}
+
+void
+Cmvmi::sendFragmentedComplete(Signal* signal, Uint32 data, Uint32 returnCode){
+ if(g_print)
+ ndbout_c("sendFragmentedComplete: %d", data);
+ if(data == 11 || data == 12){
+ for(Uint32 i = 0; i<3; i++){
+ if(g_test[i].p != 0)
+ delete[] g_test[i].p;
+ }
+ }
+}
diff --git a/ndb/src/kernel/blocks/cmvmi/Cmvmi.hpp b/ndb/src/kernel/blocks/cmvmi/Cmvmi.hpp
new file mode 100644
index 00000000000..4f42c2efc93
--- /dev/null
+++ b/ndb/src/kernel/blocks/cmvmi/Cmvmi.hpp
@@ -0,0 +1,131 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef Cmvmi_H_
+#define Cmvmi_H_
+
+#include <pc.hpp>
+#include <SimulatedBlock.hpp>
+#include <LogLevel.hpp>
+
+#include <ArrayList.hpp>
+
+/**
+ * Cmvmi class
+ */
+class Cmvmi : public SimulatedBlock {
+public:
+ Cmvmi(const Configuration & conf);
+ virtual ~Cmvmi();
+
+private:
+ /**
+ * These methods used to be reportXXX
+ *
+ * But they in a nasty way intefere with the execution model
+ * they been turned in to exec-Method used via prio A signals
+ */
+ void execDISCONNECT_REP(Signal*);
+ void execCONNECT_REP(Signal*);
+
+private:
+ BLOCK_DEFINES(Cmvmi);
+
+ // The signal processing functions
+ void execNDB_TAMPER(Signal* signal);
+ void execSET_LOGLEVELORD(Signal* signal);
+ void execEVENT_REP(Signal* signal);
+ void execSTTOR_Local(Signal* signal);
+ void execCM_RUN(Signal* signal);
+ void execCM_INFOREQ(Signal* signal);
+ void execCMVMI_CFGREQ(Signal* signal);
+ void execCLOSE_COMREQ(Signal* signal);
+ void execENABLE_COMORD(Signal* signal);
+ void execOPEN_COMREQ(Signal* signal);
+ void execSIZEALT_ACK(Signal* signal);
+ void execTEST_ORD(Signal* signal);
+
+ void execSTATISTICS_REQ(Signal* signal);
+ void execSTOP_ORD(Signal* signal);
+ void execSTART_ORD(Signal* signal);
+ void execTAMPER_ORD(Signal* signal);
+ void execSET_VAR_REQ(Signal* signal);
+ void execSET_VAR_CONF(Signal* signal);
+ void execSET_VAR_REF(Signal* signal);
+
+ void execDUMP_STATE_ORD(Signal* signal);
+
+ void execEVENT_SUBSCRIBE_REQ(Signal *);
+ void cancelSubscription(NodeId nodeId);
+
+ void handleSET_VAR_REQ(Signal* signal);
+
+ void execTESTSIG(Signal* signal);
+
+ int signalCount;
+ int theSignalKey;
+ int theStartPhase;
+ int theNumberOfNodes;
+
+ char theErrorMessage[256];
+ void sendSTTORRY(Signal* signal);
+
+ LogLevel clogLevel;
+ class Configuration & theConfig;
+ const class ClusterConfiguration & theCConfig;
+
+ /**
+ * This struct defines the data needed for a EVENT_REP subscriber
+ */
+ struct EventRepSubscriber {
+ /**
+ * What log level is the subscriber using
+ */
+ LogLevel logLevel;
+
+ /**
+ * What block reference does he use
+ * (Where should the EVENT_REP's be forwarded)
+ */
+ BlockReference blockRef;
+
+ /**
+ * Next ptr (used in pool/list)
+ */
+ union { Uint32 nextPool; Uint32 nextList; };
+ Uint32 prevList;
+ };
+ typedef Ptr<EventRepSubscriber> SubscriberPtr;
+
+ /**
+ * Pool of EventRepSubscriber record
+ */
+ ArrayPool<EventRepSubscriber> subscriberPool;
+
+ /**
+ * List of current subscribers
+ */
+ ArrayList<EventRepSubscriber> subscribers;
+
+private:
+ // Declared but not defined
+ Cmvmi(const Cmvmi &obj);
+ void operator = (const Cmvmi &);
+
+ void sendFragmentedComplete(Signal* signal, Uint32 data, Uint32 returnCode);
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/cmvmi/Makefile b/ndb/src/kernel/blocks/cmvmi/Makefile
new file mode 100644
index 00000000000..d75e5dbf08b
--- /dev/null
+++ b/ndb/src/kernel/blocks/cmvmi/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := cmvmi
+
+SOURCES = Cmvmi.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dbacc/Dbacc.hpp b/ndb/src/kernel/blocks/dbacc/Dbacc.hpp
new file mode 100644
index 00000000000..fef41be88c4
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbacc/Dbacc.hpp
@@ -0,0 +1,1568 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBACC_H
+#define DBACC_H
+
+
+
+#include <pc.hpp>
+#include <SimulatedBlock.hpp>
+
+#ifdef DBACC_C
+// Debug Macros
+#define dbgWord32(ptr, ind, val)
+
+/*
+#define dbgWord32(ptr, ind, val) \
+if(debug_jan){ \
+tmp_val = val; \
+switch(ind){ \
+case 1: strcpy(tmp_string, "ZPOS_PAGE_TYPE "); \
+break; \
+case 2: strcpy(tmp_string, "ZPOS_NO_ELEM_IN_PAGE"); \
+break; \
+case 3: strcpy(tmp_string, "ZPOS_CHECKSUM "); \
+break; \
+case 4: strcpy(tmp_string, "ZPOS_OVERFLOWREC "); \
+break; \
+case 5: strcpy(tmp_string, "ZPOS_FREE_AREA_IN_PAGE"); \
+break; \
+case 6: strcpy(tmp_string, "ZPOS_LAST_INDEX "); \
+break; \
+case 7: strcpy(tmp_string, "ZPOS_INSERT_INDEX "); \
+break; \
+case 8: strcpy(tmp_string, "ZPOS_ARRAY_POS "); \
+break; \
+case 9: strcpy(tmp_string, "ZPOS_NEXT_FREE_INDEX"); \
+break; \
+case 10: strcpy(tmp_string, "ZPOS_NEXT_PAGE "); \
+break; \
+case 11: strcpy(tmp_string, "ZPOS_PREV_PAGE "); \
+break; \
+default: sprintf(tmp_string, "%-20d", ind);\
+} \
+ndbout << "Ptr: " << ptr.p->word32 << " \tIndex: " << tmp_string << " \tValue: " << tmp_val << " \tLINE: " << __LINE__ << endl; \
+}\
+*/
+
+#define dbgUndoword(ptr, ind, val)
+
+// Constants
+/** ------------------------------------------------------------------------
+ * THESE ARE CONSTANTS THAT ARE USED FOR DEFINING THE SIZE OF BUFFERS, THE
+ * SIZE OF PAGE HEADERS, THE NUMBER OF BUFFERS IN A PAGE AND A NUMBER OF
+ * OTHER CONSTANTS WHICH ARE CHANGED WHEN THE BUFFER SIZE IS CHANGED.
+ * ----------------------------------------------------------------------- */
+#define ZHEAD_SIZE 32
+#define ZCON_HEAD_SIZE 2
+#define ZBUF_SIZE 28
+#define ZEMPTYLIST 72
+#define ZUP_LIMIT 14
+#define ZDOWN_LIMIT 12
+#define ZSHIFT_PLUS 5
+#define ZSHIFT_MINUS 2
+#define ZFREE_LIMIT 65
+#define ZNO_CONTAINERS 64
+#define ZELEM_HEAD_SIZE 1
+/* ------------------------------------------------------------------------- */
+/* THESE CONSTANTS DEFINE THE USE OF THE PAGE HEADER IN THE INDEX PAGES. */
+/* ------------------------------------------------------------------------- */
+#define ZPOS_PAGE_ID 0
+#define ZPOS_PAGE_TYPE 1
+#define ZPOS_PAGE_TYPE_BIT 14
+#define ZPOS_EMPTY_LIST 1
+#define ZPOS_ALLOC_CONTAINERS 2
+#define ZPOS_CHECKSUM 3
+#define ZPOS_OVERFLOWREC 4
+#define ZPOS_NO_ELEM_IN_PAGE 2
+#define ZPOS_FREE_AREA_IN_PAGE 5
+#define ZPOS_LAST_INDEX 6
+#define ZPOS_INSERT_INDEX 7
+#define ZPOS_ARRAY_POS 8
+#define ZPOS_NEXT_FREE_INDEX 9
+#define ZPOS_NEXT_PAGE 10
+#define ZPOS_PREV_PAGE 11
+#define ZNORMAL_PAGE_TYPE 0
+#define ZOVERFLOW_PAGE_TYPE 1
+#define ZLONG_PAGE_TYPE 2
+#define ZDEFAULT_LIST 3
+#define ZWORDS_IN_PAGE 2048
+/* --------------------------------------------------------------------------------- */
+/* CONSTANTS FOR THE ZERO PAGES */
+/* --------------------------------------------------------------------------------- */
+#define ZPAGEZERO_PREV_UNDOP 8
+#define ZPAGEZERO_NO_OVER_PAGE 9
+#define ZPAGEZERO_TABID 10
+#define ZPAGEZERO_FRAGID0 11
+#define ZPAGEZERO_FRAGID1 12
+#define ZPAGEZERO_HASH_CHECK 13
+#define ZPAGEZERO_DIRSIZE 14
+#define ZPAGEZERO_EXPCOUNTER 15
+#define ZPAGEZERO_NEXT_UNDO_FILE 16
+#define ZPAGEZERO_SLACK 17
+#define ZPAGEZERO_NO_PAGES 18
+#define ZPAGEZERO_HASHCHECKBIT 19
+#define ZPAGEZERO_K 20
+#define ZPAGEZERO_LHFRAGBITS 21
+#define ZPAGEZERO_LHDIRBITS 22
+#define ZPAGEZERO_LOCALKEYLEN 23
+#define ZPAGEZERO_MAXP 24
+#define ZPAGEZERO_MAXLOADFACTOR 25
+#define ZPAGEZERO_MINLOADFACTOR 26
+#define ZPAGEZERO_MYFID 27
+#define ZPAGEZERO_LAST_OVER_INDEX 28
+#define ZPAGEZERO_P 29
+#define ZPAGEZERO_NO_OF_ELEMENTS 30
+#define ZPAGEZERO_ELEMENT_LENGTH 31
+#define ZPAGEZERO_KEY_LENGTH 32
+#define ZPAGEZERO_NODETYPE 33
+#define ZPAGEZERO_SLACK_CHECK 34
+/* --------------------------------------------------------------------------------- */
+/* CONSTANTS FOR THE LONG KEY PAGES */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+// Maximum number of elements in long key page = (ZWORDS_IN_PAGE - ZHEAD_SIZE) /
+// (MinKeySize + IndexSize) = (2048 - 32) / (8 + 1) = 224. MinKeySize is actually 9
+// because 8 is the largest normal key size.
+#define ZMAX_NO_OF_LONGKEYS_IN_PAGE 225
+#define ZMAX_LONG_KEY_ARRAY_INDEX 3
+#define ZACTIVE_LONG_KEY_LEN 1
+/* --------------------------------------------------------------------------------- */
+/* CONSTANTS IN ALPHABETICAL ORDER */
+/* --------------------------------------------------------------------------------- */
+#define ZADDFRAG 0
+#define ZCOPY_NEXT 1
+#define ZCOPY_NEXT_COMMIT 2
+#define ZCOPY_COMMIT 3
+#define ZCOPY_REPEAT 4
+#define ZCOPY_ABORT 5
+#define ZCOPY_CLOSE 6
+#define ZDIRARRAY 68
+#define ZDIRRANGESIZE 65
+//#define ZEMPTY_FRAGMENT 0
+#define ZFRAGMENTSIZE 64
+#define ZFIRSTTIME 1
+#define ZFS_CONNECTSIZE 300
+#define ZFS_OPSIZE 100
+#define ZKEYINKEYREQ 4
+#define ZLCP_CONNECTSIZE 30
+#define ZLEFT 1
+#define ZLOCALLOGFILE 2
+#define ZLOCKED 0
+#define ZMAXSCANSIGNALLEN 20
+#define ZMAINKEYLEN 8
+#define ZMAX_UNDO_VERSION 4
+#define ZNO_OF_DISK_VERSION 3
+#define ZNO_OF_OP_PER_SIGNAL 20
+//#define ZNOT_EMPTY_FRAGMENT 1
+#define ZNR_OF_UNDO_PAGE_GROUP 16
+#define ZOP_HEAD_INFO_LN 3
+#define ZOPRECSIZE 740
+#define ZOVERFLOWRECSIZE 5
+#define ZPAGE8_BASE_ADD 1
+#define ZPAGESIZE 128
+#define ZPARALLEL_QUEUE 1
+#define ZPDIRECTORY 1
+#define ZSCAN_MAX_LOCK 4
+#define ZSERIAL_QUEUE 2
+#define ZSPH1 1
+#define ZSPH2 2
+#define ZSPH3 3
+#define ZSPH6 6
+#define ZREADLOCK 0
+#define ZRIGHT 2
+#define ZROOTFRAGMENTSIZE 32
+#define ZSCAN_LOCK_ALL 3
+#define ZSCAN_OP 5
+#define ZSCAN_REC_SIZE 256
+#define ZSR_VERSION_REC_SIZE 16
+#define ZSTAND_BY 2
+#define ZTABLESIZE 16
+#define ZTABMAXINDEX 3
+#define ZUNDEFINED_OP 6
+#define ZUNDOPAGESIZE 64
+#define ZUNDOHEADSIZE 7
+#define ZUNLOCKED 1
+#define ZUNDOPAGE_BASE_ADD 2
+#define ZUNDOPAGEINDEXBITS 13
+#define ZUNDOPAGEINDEX_MASK 0x1fff
+#define ZWRITEPAGESIZE 8
+#define ZWRITE_UNDOPAGESIZE 2
+#define ZMIN_UNDO_PAGES_AT_COMMIT 4
+#define ZMIN_UNDO_PAGES_AT_OPERATION 10
+#define ZMIN_UNDO_PAGES_AT_EXPAND 16
+
+/* --------------------------------------------------------------------------------- */
+/* CONTINUEB CODES */
+/* --------------------------------------------------------------------------------- */
+#define ZLOAD_BAL_LCP_TIMER 0
+#define ZINITIALISE_RECORDS 1
+#define ZSR_READ_PAGES_ALLOC 2
+#define ZSTART_UNDO 3
+#define ZSEND_SCAN_HBREP 4
+#define ZREL_ROOT_FRAG 5
+#define ZREL_FRAG 6
+#define ZREL_DIR 7
+#define ZREPORT_MEMORY_USAGE 8
+
+/* ------------------------------------------------------------------------- */
+/* ERROR CODES */
+/* ------------------------------------------------------------------------- */
+#define ZLIMIT_OF_ERROR 600 // Limit check for error codes
+#define ZCHECKROOT_ERROR 601 // Delete fragment error code
+#define ZCONNECT_SIZE_ERROR 602 // ACC_SEIZEREF
+#define ZDIR_RANGE_ERROR 603 // Add fragment error code
+#define ZFULL_FRAGRECORD_ERROR 604 // Add fragment error code
+#define ZFULL_ROOTFRAGRECORD_ERROR 605 // Add fragment error code
+#define ZROOTFRAG_STATE_ERROR 606 // Add fragment
+#define ZOVERTAB_REC_ERROR 607 // Add fragment
+
+#define ZSCAN_REFACC_CONNECT_ERROR 608 // ACC_SCANREF
+#define ZFOUR_ACTIVE_SCAN_ERROR 609 // ACC_SCANREF
+#define ZNULL_SCAN_REC_ERROR 610 // ACC_SCANREF
+
+#define ZDIRSIZE_ERROR 623
+#define ZOVER_REC_ERROR 624 // Insufficient Space
+#define ZPAGESIZE_ERROR 625
+#define ZTUPLE_DELETED_ERROR 626
+#define ZREAD_ERROR 626
+#define ZWRITE_ERROR 630
+#define ZTO_OP_STATE_ERROR 631
+#define ZTOO_EARLY_ACCESS_ERROR 632
+#define ZTEMPORARY_ACC_UNDO_FAILURE 677
+#endif
+
+class ElementHeader {
+ /**
+ *
+ * l = Locked -- If true contains operation else scan bits + hash value
+ * s = Scan bits
+ * h = Hash value
+ * o = Operation ptr I
+ *
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * lssssssssssss hhhhhhhhhhhhhhhh
+ * ooooooooooooooooooooooooooooooo
+ */
+public:
+ STATIC_CONST( HASH_VALUE_PART_MASK = 0xFFFF );
+
+ static bool getLocked(Uint32 data);
+ static bool getUnlocked(Uint32 data);
+ static Uint32 getScanBits(Uint32 data);
+ static Uint32 getHashValuePart(Uint32 data);
+ static Uint32 getOpPtrI(Uint32 data);
+
+ static Uint32 setLocked(Uint32 opPtrI);
+ static Uint32 setUnlocked(Uint32 hashValuePart, Uint32 scanBits);
+ static Uint32 setScanBit(Uint32 header, Uint32 scanBit);
+ static Uint32 clearScanBit(Uint32 header, Uint32 scanBit);
+};
+
+inline
+bool
+ElementHeader::getLocked(Uint32 data){
+ return (data & 1) == 0;
+}
+
+inline
+bool
+ElementHeader::getUnlocked(Uint32 data){
+ return (data & 1) == 1;
+}
+
+inline
+Uint32
+ElementHeader::getScanBits(Uint32 data){
+ assert(getUnlocked(data));
+ return (data >> 1) & ((1 << MAX_PARALLEL_SCANS_PER_FRAG) - 1);
+}
+
+inline
+Uint32
+ElementHeader::getHashValuePart(Uint32 data){
+ assert(getUnlocked(data));
+ return data >> 16;
+}
+
+inline
+Uint32
+ElementHeader::getOpPtrI(Uint32 data){
+ assert(getLocked(data));
+ return data >> 1;
+}
+
+inline
+Uint32
+ElementHeader::setLocked(Uint32 opPtrI){
+ return (opPtrI << 1) + 0;
+}
+inline
+Uint32
+ElementHeader::setUnlocked(Uint32 hashValue, Uint32 scanBits){
+ return (hashValue << 16) + (scanBits << 1) + 1;
+}
+
+inline
+Uint32
+ElementHeader::setScanBit(Uint32 header, Uint32 scanBit){
+ assert(getUnlocked(header));
+ return header | (scanBit << 1);
+}
+
+inline
+Uint32
+ElementHeader::clearScanBit(Uint32 header, Uint32 scanBit){
+ assert(getUnlocked(header));
+ return header & (~(scanBit << 1));
+}
+
+
+class Dbacc: public SimulatedBlock {
+public:
+// State values
+enum State {
+ FREEFRAG = 0,
+ ACTIVEFRAG = 1,
+ SEND_QUE_OP = 2,
+ WAIT_ACC_LCPREQ = 3,
+ LCP_SEND_PAGES = 4,
+ LCP_SEND_OVER_PAGES = 5,
+ LCP_SEND_ZERO_PAGE = 6,
+ SR_READ_PAGES = 7,
+ SR_READ_OVER_PAGES = 8,
+ WAIT_ZERO_PAGE_STORED = 9,
+ WAIT_NOTHING = 10,
+ WAIT_OPEN_UNDO_LCP = 11,
+ WAIT_OPEN_UNDO_LCP_NEXT = 12,
+ WAIT_OPEN_DATA_FILE_FOR_READ = 13,
+ WAIT_OPEN_DATA_FILE_FOR_WRITE = 14,
+ OPEN_UNDO_FILE_SR = 15,
+ READ_UNDO_PAGE = 16,
+ READ_UNDO_PAGE_AND_CLOSE = 17,
+ WAIT_READ_DATA = 18,
+ WAIT_READ_PAGE_ZERO = 19,
+ WAIT_WRITE_DATA = 20,
+ WAIT_WRITE_UNDO = 21,
+ WAIT_WRITE_UNDO_EXIT = 22,
+ WAIT_CLOSE_UNDO = 23,
+ LCP_CLOSE_DATA = 24,
+ SR_CLOSE_DATA = 25,
+ WAIT_ONE_CONF = 26,
+ WAIT_TWO_CONF = 27,
+ LCP_FREE = 28,
+ LCP_ACTIVE = 29,
+ FREE_OP = 30,
+ WAIT_EXE_OP = 32,
+ WAIT_IN_QUEUE = 34,
+ EXE_OP = 35,
+ SCAN_ACTIVE = 36,
+ SCAN_WAIT_IN_QUEUE = 37,
+ IDLE = 39,
+ ACTIVE = 40,
+ WAIT_COMMIT_ABORT = 41,
+ ABORT = 42,
+ ABORTADDFRAG = 43,
+ REFUSEADDFRAG = 44,
+ DELETEFRAG = 45,
+ DELETETABLE = 46,
+ UNDEFINEDROOT = 47,
+ ADDFIRSTFRAG = 48,
+ ADDSECONDFRAG = 49,
+ DELETEFIRSTFRAG = 50,
+ DELETESECONDFRAG = 51,
+ ACTIVEROOT = 52,
+ LCP_CREATION = 53
+};
+
+// Records
+
+
+//----------------------------------------------------------------------------------
+// LONGKEY PAGE RECORD
+//
+// A long key page consist of a header part, a key data part and an index part. The
+// page starts with a header of size HEAD_SIZE. As you can see below, not every word
+// in the header is used. After the header comes the data part, where the actual
+// keys are stored. A key is always inserted after the existing keys in the data
+// part. If we have a fragmented data part and a new key doesn't fit after the
+// existing keys we reorganize the keys. The index part starts at the end of the
+// page and grows towards the end of the data part. This means that the limit
+// between the data part and the index part is floating. Each inserted key have a
+// word in the index part that describes size and position of the key in the data
+// part. The free indexes in the index part are single linked.
+//----------------------------------------------------------------------------------
+ union LongKeyPage {
+ struct {
+ Uint32 pageId; // ZPOS_PAGE_ID 0
+ Uint32 b;
+ // The number of keys in page.
+ Uint32 noOfElements; // ZPOS_NO_ELEM_IN_PAGE 2
+ Uint32 d;
+ Uint32 e;
+ // The free area in the data part of page.
+ Uint32 freeArea; // ZPOS_FREE_AREA_IN_PAGE 5
+ // The index position, which defines the limit between the data and the index part.
+ Uint32 highestIndex; // ZPOS_LAST_INDEX 6
+ // The position where to insert the actual key in the data part.
+ Uint32 insertPos; // ZPOS_INSERT_INDEX 7
+ // Position in a page array where the pages are stored in a double linked list.
+ // Based on the free area in the page. Values 0 to 3.
+ Uint32 pageArrayPos; // ZPOS_ARRAY_POS 8
+ // Next free position in the index part.
+ Uint32 nextFreeIndex; // ZPOS_NEXT_FREE_INDEX 9
+ // Next page in the double linked list.
+ Uint32 nextPage; // ZPOS_NEXT_PAGE 10
+ // Previous page in the double linked list.
+ Uint32 prevPage; // ZPOS_PREV_PAGE 11
+ } header;
+ // This is kept to keep the logic and to make changes to a minimum.
+ Uint32 word32[2048];
+ };
+
+/* --------------------------------------------------------------------------------- */
+/* UNDO HEADER RECORD */
+/* --------------------------------------------------------------------------------- */
+
+ struct UndoHeader {
+ enum UndoHeaderType{
+ ZPAGE_INFO = 0,
+ ZOVER_PAGE_INFO = 1,
+ ZOP_INFO = 2,
+ ZUNDO_INSERT_LONG_KEY = 3,
+ ZUNDO_DELETE_LONG_KEY = 4,
+ ZNO_UNDORECORD_TYPES = 5
+ };
+ UintR tableId;
+ UintR rootFragId;
+ UintR localFragId;
+ UintR variousInfo;
+ UintR logicalPageId;
+ UintR prevUndoAddressForThisFrag;
+ UintR prevUndoAddress;
+ };
+
+/* --------------------------------------------------------------------------------- */
+/* DIRECTORY RANGE */
+/* --------------------------------------------------------------------------------- */
+ struct DirRange {
+ Uint32 dirArray[256];
+ }; /* p2c: size = 1024 bytes */
+
+ typedef Ptr<DirRange> DirRangePtr;
+
+/* --------------------------------------------------------------------------------- */
+/* DIRECTORYARRAY */
+/* --------------------------------------------------------------------------------- */
+struct Directoryarray {
+ Uint32 pagep[256];
+}; /* p2c: size = 1024 bytes */
+
+ typedef Ptr<Directoryarray> DirectoryarrayPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* FRAGMENTREC. ALL INFORMATION ABOUT FRAMENT AND HASH TABLE IS SAVED IN FRAGMENT */
+/* REC A POINTER TO FRAGMENT RECORD IS SAVED IN ROOTFRAGMENTREC FRAGMENT */
+/* --------------------------------------------------------------------------------- */
+struct Fragmentrec {
+//-----------------------------------------------------------------------------
+// References to long key pages with free area. Some type of buddy structure
+// where references in higher index have more free space.
+//-----------------------------------------------------------------------------
+ Uint32 longKeyPageArray[4];
+
+//-----------------------------------------------------------------------------
+// These variables keep track of allocated pages, the number of them and the
+// start file page of them. Used during local checkpoints.
+//-----------------------------------------------------------------------------
+ Uint32 datapages[8];
+ Uint32 activeDataPage;
+ Uint32 activeDataFilePage;
+
+//-----------------------------------------------------------------------------
+// Temporary variables used during shrink and expand process.
+//-----------------------------------------------------------------------------
+ Uint32 expReceivePageptr;
+ Uint32 expReceiveIndex;
+ Uint32 expReceiveForward;
+ Uint32 expSenderDirIndex;
+ Uint32 expSenderDirptr;
+ Uint32 expSenderIndex;
+ Uint32 expSenderPageptr;
+
+//-----------------------------------------------------------------------------
+// List of lock owners and list of lock waiters to support LCP handling
+//-----------------------------------------------------------------------------
+ Uint32 lockOwnersList;
+ Uint32 firstWaitInQueOp;
+ Uint32 lastWaitInQueOp;
+ Uint32 sentWaitInQueOp;
+
+//-----------------------------------------------------------------------------
+// References to Directory Ranges (which in turn references directories, which
+// in its turn references the pages) for the bucket pages and the overflow
+// bucket pages.
+//-----------------------------------------------------------------------------
+ Uint32 directory;
+ Uint32 dirsize;
+ Uint32 overflowdir;
+ Uint32 lastOverIndex;
+
+//-----------------------------------------------------------------------------
+// These variables are used to support LCP and Restore from disk.
+// lcpDirIndex: used during LCP as the frag page id currently stored.
+// lcpMaxDirIndex: The dirsize at start of LCP.
+// lcpMaxOverDirIndex: The xx at start of LCP
+// During a LCP one writes the minimum of the number of pages in the directory
+// and the number of pages at the start of the LCP.
+// noStoredPages: Number of bucket pages written in LCP used at restore
+// noOfOverStoredPages: Number of overflow pages written in LCP used at restore
+// This variable is also used during LCP to calculate this number.
+//-----------------------------------------------------------------------------
+ Uint32 lcpDirIndex;
+ Uint32 lcpMaxDirIndex;
+ Uint32 lcpMaxOverDirIndex;
+ Uint32 noStoredPages;
+ Uint32 noOfStoredOverPages;
+
+//-----------------------------------------------------------------------------
+// We have a list of overflow pages with free areas. We have a special record,
+// the overflow record representing these pages. The reason is that the
+// same record is also used to represent pages in the directory array that have
+// been released since they were empty (there were however higher indexes with
+// data in them). These are put in the firstFreeDirIndexRec-list.
+// An overflow record representing a page can only be in one of these lists.
+//-----------------------------------------------------------------------------
+ Uint32 firstOverflowRec;
+ Uint32 lastOverflowRec;
+ Uint32 firstFreeDirindexRec;
+
+//-----------------------------------------------------------------------------
+// localCheckpId is used during execution of UNDO log to ensure that we only
+// apply UNDO log records from the restored LCP of the fragment.
+// lcpLqhPtr keeps track of LQH record for this fragment to checkpoint
+//-----------------------------------------------------------------------------
+ Uint32 localCheckpId;
+ Uint32 lcpLqhPtr;
+
+//-----------------------------------------------------------------------------
+// Counter keeping track of how many times we have expanded. We need to ensure
+// that we do not shrink so many times that this variable becomes negative.
+//-----------------------------------------------------------------------------
+ Uint32 expandCounter;
+//-----------------------------------------------------------------------------
+// Reference to record for open file at LCP and restore
+//-----------------------------------------------------------------------------
+ Uint32 fsConnPtr;
+
+//-----------------------------------------------------------------------------
+// These variables are important for the linear hashing algorithm.
+// localkeylen is the size of the local key (1 and 2 is currently supported)
+// maxloadfactor is the factor specifying when to expand
+// minloadfactor is the factor specifying when to shrink (hysteresis model)
+// maxp and p
+// maxp and p is the variables most central to linear hashing. p + maxp + 1 is the
+// current number of buckets. maxp is the largest value of the type 2**n - 1
+// which is smaller than the number of buckets. These values are used to find
+// correct bucket with the aid of the hash value.
+//
+// slack is the variable keeping track of whether we have inserted more than
+// the current size is suitable for or less. Slack together with the boundaries
+// set by maxloadfactor and minloadfactor decides when to expand/shrink
+// slackCheck When slack goes over this value it is time to expand.
+// slackCheck = (maxp + p + 1)*(maxloadfactor - minloadfactor) or
+// bucketSize * hysteresis
+//-----------------------------------------------------------------------------
+ Uint32 localkeylen;
+ Uint32 maxp;
+ Uint32 maxloadfactor;
+ Uint32 minloadfactor;
+ Uint32 p;
+ Uint32 slack;
+ Uint32 slackCheck;
+
+//-----------------------------------------------------------------------------
+// myfid is the fragment id of the fragment
+// myroot is the reference to the root fragment record
+// nextfreefrag is the next free fragment if linked into a free list
+//-----------------------------------------------------------------------------
+ Uint32 myfid;
+ Uint32 myroot;
+ Uint32 myTableId;
+ Uint32 nextfreefrag;
+
+//-----------------------------------------------------------------------------
+// This variable is used during restore to keep track of page id of read pages.
+// During read of bucket pages this is used to calculate the page id and also
+// to verify that the page id of the read page is correct. During read of over-
+// flow pages it is only used to keep track of the number of pages read.
+//-----------------------------------------------------------------------------
+ Uint32 nextAllocPage;
+
+//-----------------------------------------------------------------------------
+// Keeps track of undo position for fragment during LCP and restore.
+//-----------------------------------------------------------------------------
+ Uint32 prevUndoposition;
+
+//-----------------------------------------------------------------------------
+// Page reference during LCP and restore of page zero where fragment data is
+// saved
+//-----------------------------------------------------------------------------
+ Uint32 zeroPagePtr;
+
+//-----------------------------------------------------------------------------
+// Number of pages read from file during restore
+//-----------------------------------------------------------------------------
+ Uint32 noOfExpectedPages;
+
+//-----------------------------------------------------------------------------
+// Fragment State, mostly applicable during LCP and restore
+//-----------------------------------------------------------------------------
+ State fragState;
+
+//-----------------------------------------------------------------------------
+// Keep track of number of outstanding writes of UNDO log records to ensure that
+// we have saved all UNDO info before concluding local checkpoint.
+//-----------------------------------------------------------------------------
+ Uint32 nrWaitWriteUndoExit;
+
+//-----------------------------------------------------------------------------
+// lastUndoIsStored is used to handle parallel writes of UNDO log and pages to
+// know when LCP is completed
+//-----------------------------------------------------------------------------
+ Uint8 lastUndoIsStored;
+
+//-----------------------------------------------------------------------------
+// Set to ZTRUE when local checkpoint freeze occurs and set to ZFALSE when
+// local checkpoint concludes.
+//-----------------------------------------------------------------------------
+ Uint8 createLcp;
+
+//-----------------------------------------------------------------------------
+// Flag indicating whether we are in the load phase of restore still.
+//-----------------------------------------------------------------------------
+ Uint8 loadingFlag;
+
+//-----------------------------------------------------------------------------
+// elementLength: Length of element in bucket and overflow pages
+// keyLength: Length of key (== 0 if long key or variable key length)
+//-----------------------------------------------------------------------------
+ Uint8 elementLength;
+ Uint8 keyLength;
+
+//-----------------------------------------------------------------------------
+// This flag is used to avoid sending a big number of expand or shrink signals
+// when simultaneously committing many inserts or deletes.
+//-----------------------------------------------------------------------------
+ Uint8 expandFlag;
+
+//-----------------------------------------------------------------------------
+// hashcheckbit is the bit to check whether to send element to split bucket or not
+// k (== 6) is the number of buckets per page
+// lhfragbits is the number of bits used to calculate the fragment id
+// lhdirbits is the number of bits used to calculate the page id
+//-----------------------------------------------------------------------------
+ Uint8 hashcheckbit;
+ Uint8 k;
+ Uint8 lhfragbits;
+ Uint8 lhdirbits;
+
+//-----------------------------------------------------------------------------
+// nodetype can only be STORED in this release. Is currently only set, never read
+// stopQueOp is indicator that locked operations will not start until LCP have
+// released the lock on the fragment
+//-----------------------------------------------------------------------------
+ Uint8 nodetype;
+ Uint8 stopQueOp;
+};
+
+ typedef Ptr<Fragmentrec> FragmentrecPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* FS_CONNECTREC */
+/* --------------------------------------------------------------------------------- */
+struct FsConnectrec {
+ Uint32 fsNext;
+ Uint32 fsPrev;
+ Uint32 fragrecPtr;
+ Uint32 fsPtr;
+ State fsState;
+ Uint8 activeFragId;
+ Uint8 fsPart;
+}; /* p2c: size = 24 bytes */
+
+ typedef Ptr<FsConnectrec> FsConnectrecPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* FS_OPREC */
+/* --------------------------------------------------------------------------------- */
+struct FsOprec {
+ Uint32 fsOpnext;
+ Uint32 fsOpfragrecPtr;
+ Uint32 fsConptr;
+ State fsOpstate;
+ Uint16 fsOpMemPage;
+}; /* p2c: size = 20 bytes */
+
+ typedef Ptr<FsOprec> FsOprecPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* LCP_CONNECTREC */
+/* --------------------------------------------------------------------------------- */
+struct LcpConnectrec {
+ Uint32 nextLcpConn;
+ Uint32 lcpUserptr;
+ Uint32 rootrecptr;
+ State syncUndopageState;
+ State lcpstate;
+ Uint32 lcpUserblockref;
+ Uint16 localCheckPid;
+ Uint8 noOfLcpConf;
+};
+ typedef Ptr<LcpConnectrec> LcpConnectrecPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* OPERATIONREC */
+/* --------------------------------------------------------------------------------- */
+struct Operationrec {
+ Uint32 keydata[8];
+ Uint32 localdata[2];
+ Uint32 elementIsforward;
+ Uint32 elementPage;
+ Uint32 elementPointer;
+ Uint32 fid;
+ Uint32 fragptr;
+ Uint32 hashvaluePart;
+ Uint32 hashValue;
+ Uint32 insertDeleteLen;
+ Uint32 keyinfoPage;
+ Uint32 nextLockOwnerOp;
+ Uint32 nextOp;
+ Uint32 nextParallelQue;
+ Uint32 nextQueOp;
+ Uint32 nextSerialQue;
+ Uint32 prevOp;
+ Uint32 prevLockOwnerOp;
+ Uint32 prevParallelQue;
+ Uint32 prevQueOp;
+ Uint32 prevSerialQue;
+ Uint32 scanRecPtr;
+ Uint32 transId1;
+ Uint32 transId2;
+ Uint32 longPagePtr;
+ Uint32 longKeyPageIndex;
+ State opState;
+ Uint32 userptr;
+ State transactionstate;
+ Uint16 elementContainer;
+ Uint16 tupkeylen;
+ Uint32 userblockref;
+ Uint32 scanBits;
+ Uint8 elementIsDisappeared;
+ Uint8 insertIsDone;
+ Uint8 lockMode;
+ Uint8 lockOwner;
+ Uint8 nodeType;
+ Uint8 operation;
+ Uint8 opSimple;
+ Uint8 dirtyRead;
+ Uint8 commitDeleteCheckFlag;
+ Uint8 isAccLockReq;
+ Uint32 nextOpList;
+}; /* p2c: size = 168 bytes */
+
+ typedef Ptr<Operationrec> OperationrecPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* OVERFLOW_RECORD */
+/* --------------------------------------------------------------------------------- */
+struct OverflowRecord {
+ Uint32 dirindex;
+ Uint32 nextOverRec;
+ Uint32 nextOverList;
+ Uint32 prevOverRec;
+ Uint32 prevOverList;
+ Uint32 overpage;
+ Uint32 nextfreeoverrec;
+};
+
+ typedef Ptr<OverflowRecord> OverflowRecordPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* PAGE8 */
+/* --------------------------------------------------------------------------------- */
+struct Page8 {
+ Uint32 word32[2048];
+}; /* p2c: size = 8192 bytes */
+
+ typedef Ptr<Page8> Page8Ptr;
+
+/* --------------------------------------------------------------------------------- */
+/* ROOTFRAGMENTREC */
+/* DURING EXPAND FRAGMENT PROCESS, EACH FRAGMEND WILL BE EXPAND INTO TWO */
+/* NEW FRAGMENTS.TO MAKE THIS PROCESS EASIER, DURING ADD FRAGMENT PROCESS */
+/* NEXT FRAGMENT IDENTIIES WILL BE CALCULATED, AND TWO FRAGMENTS WILL BE */
+/* ADDED IN (NDBACC). THEREBY EXPAND OF FRAGMENT CAN BE PERFORMED QUICK AND */
+/* EASY.THE NEW FRAGMENT ID SENDS TO TUP MANAGER FOR ALL OPERATION PROCESS. */
+/* --------------------------------------------------------------------------------- */
+struct Rootfragmentrec {
+ Uint32 scan[MAX_PARALLEL_SCANS_PER_FRAG];
+ Uint32 fragmentptr[2];
+ Uint32 fragmentid[2];
+ Uint32 lcpPtr;
+ Uint32 mytabptr;
+ Uint32 nextroot;
+ Uint32 roothashcheck;
+ Uint32 noOfElements;
+ State rootState;
+}; /* p2c: size = 72 bytes */
+
+ typedef Ptr<Rootfragmentrec> RootfragmentrecPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* SCAN_REC */
+/* --------------------------------------------------------------------------------- */
+struct ScanRec {
+ enum ScanState {
+ WAIT_NEXT,
+ SCAN_DISCONNECT
+ };
+ enum ScanBucketState {
+ FIRST_LAP,
+ SECOND_LAP,
+ SCAN_COMPLETED
+ };
+ Uint32 activeLocalFrag;
+ Uint32 rootPtr;
+ Uint32 nextBucketIndex;
+ Uint32 scanNextfreerec;
+ Uint32 scanFirstActiveOp;
+ Uint32 scanFirstLockedOp;
+ Uint32 scanLastLockedOp;
+ Uint32 scanFirstQueuedOp;
+ Uint32 scanLastQueuedOp;
+ Uint32 scanUserptr;
+ Uint32 scanTrid1;
+ Uint32 scanTrid2;
+ Uint32 startNoOfBuckets;
+ Uint32 minBucketIndexToRescan;
+ Uint32 maxBucketIndexToRescan;
+ Uint32 scanOpsAllocated;
+ ScanBucketState scanBucketState;
+ ScanState scanState;
+ Uint16 scanLockHeld;
+ Uint32 scanUserblockref;
+ Uint32 scanMask;
+ Uint8 scanLockMode;
+ Uint8 scanKeyinfoFlag;
+ Uint8 scanTimer;
+ Uint8 scanContinuebCounter;
+ Uint8 scanReadCommittedFlag;
+};
+
+ typedef Ptr<ScanRec> ScanRecPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* SR_VERSION_REC */
+/* --------------------------------------------------------------------------------- */
+struct SrVersionRec {
+ Uint32 nextFreeSr;
+ Uint32 checkPointId;
+ Uint32 prevAddress;
+ Uint32 srUnused; /* p2c: Not used */
+}; /* p2c: size = 16 bytes */
+
+ typedef Ptr<SrVersionRec> SrVersionRecPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* TABREC */
+/* --------------------------------------------------------------------------------- */
+struct Tabrec {
+ Uint32 fragholder[NO_OF_FRAG_PER_NODE];
+ Uint32 fragptrholder[NO_OF_FRAG_PER_NODE];
+ Uint32 tabUserPtr;
+ BlockReference tabUserRef;
+};
+ typedef Ptr<Tabrec> TabrecPtr;
+
+/* --------------------------------------------------------------------------------- */
+/* UNDOPAGE */
+/* --------------------------------------------------------------------------------- */
+struct Undopage {
+ Uint32 undoword[8192];
+}; /* p2c: size = 32768 bytes */
+
+ typedef Ptr<Undopage> UndopagePtr;
+
+public:
+ Dbacc(const class Configuration &);
+ virtual ~Dbacc();
+
+private:
+ BLOCK_DEFINES(Dbacc);
+
+ // Transit signals
+ void execDEBUG_SIG(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+ void execACC_CHECK_SCAN(Signal* signal);
+ void execEXPANDCHECK2(Signal* signal);
+ void execSHRINKCHECK2(Signal* signal);
+ void execACC_OVER_REC(Signal* signal);
+ void execACC_SAVE_PAGES(Signal* signal);
+ void execNEXTOPERATION(Signal* signal);
+
+ // Received signals
+ void execSTTOR(Signal* signal);
+ void execSR_FRAGIDREQ(Signal* signal);
+ void execLCP_FRAGIDREQ(Signal* signal);
+ void execLCP_HOLDOPREQ(Signal* signal);
+ void execEND_LCPREQ(Signal* signal);
+ void execACC_LCPREQ(Signal* signal);
+ void execSTART_RECREQ(Signal* signal);
+ void execACC_CONTOPREQ(Signal* signal);
+ void execACCKEYREQ(Signal* signal);
+ void execACCSEIZEREQ(Signal* signal);
+ void execACCFRAGREQ(Signal* signal);
+ void execACC_SRREQ(Signal* signal);
+ void execNEXT_SCANREQ(Signal* signal);
+ void execACC_ABORTREQ(Signal* signal);
+ void execACC_SCANREQ(Signal* signal);
+ void execACCMINUPDATE(Signal* signal);
+ void execACC_COMMITREQ(Signal* signal);
+ void execACC_TO_REQ(Signal* signal);
+ void execACC_LOCKREQ(Signal* signal);
+ void execFSOPENCONF(Signal* signal);
+ void execFSOPENREF(Signal* signal);
+ void execFSCLOSECONF(Signal* signal);
+ void execFSCLOSEREF(Signal* signal);
+ void execFSWRITECONF(Signal* signal);
+ void execFSWRITEREF(Signal* signal);
+ void execFSREADCONF(Signal* signal);
+ void execFSREADREF(Signal* signal);
+ void execNDB_STTOR(Signal* signal);
+ void execDROP_TAB_REQ(Signal* signal);
+ void execFSREMOVECONF(Signal* signal);
+ void execFSREMOVEREF(Signal* signal);
+ void execSIZEALT_REP(Signal* signal);
+ void execSET_VAR_REQ(Signal* signal);
+ void execDUMP_STATE_ORD(Signal* signal);
+
+ // Statement blocks
+ void ACCKEY_error(Uint32 fromWhere);
+
+ void commitDeleteCheck();
+
+ void initRootFragPageZero(RootfragmentrecPtr, Page8Ptr);
+ void initRootFragSr(RootfragmentrecPtr, Page8Ptr);
+ void initFragAdd(Signal*, Uint32 rootFragIndex, Uint32 rootIndex, FragmentrecPtr);
+ void initFragPageZero(FragmentrecPtr, Page8Ptr);
+ void initFragSr(FragmentrecPtr, Page8Ptr);
+ void initFragGeneral(FragmentrecPtr);
+ void verifyFragCorrect(FragmentrecPtr regFragPtr);
+ void sendFSREMOVEREQ(Signal* signal, Uint32 tableId);
+ void sendDROP_TABFILECONF(Signal* signal, TabrecPtr tabPtr);
+ void releaseFragResources(Signal* signal, Uint32 fragIndex);
+ void releaseRootFragRecord(Signal* signal, RootfragmentrecPtr rootPtr);
+ void sendREL_TABMEMCONF(Signal* signal, TabrecPtr tabPtr);
+ void releaseRootFragResources(Signal* signal, Uint32 tableId);
+ void releaseDirResources(Signal* signal,
+ Uint32 fragIndex,
+ Uint32 dirIndex,
+ Uint32 startIndex);
+ void releaseDirectoryResources(Signal* signal,
+ Uint32 fragIndex,
+ Uint32 dirIndex,
+ Uint32 startIndex,
+ Uint32 directoryIndex);
+ void releaseOverflowResources(Signal* signal, FragmentrecPtr regFragPtr);
+ void releaseDirIndexResources(Signal* signal, FragmentrecPtr regFragPtr);
+ void releaseFragRecord(Signal* signal, FragmentrecPtr regFragPtr);
+ Uint32 remainingUndoPages();
+ void updateLastUndoPageIdWritten(Signal* signal, Uint32 aNewValue);
+ void updateUndoPositionPage(Signal* signal, Uint32 aNewValue);
+ void srCheckPage(Signal* signal);
+ void srCheckContainer(Signal* signal);
+ void initScanFragmentPart(Signal* signal);
+ Uint32 checkScanExpand(Signal* signal);
+ Uint32 checkScanShrink(Signal* signal);
+ void sendInitialiseRecords(Signal* signal);
+ void initialiseDirRec(Signal* signal);
+ void initialiseDirRangeRec(Signal* signal);
+ void initialiseFragRec(Signal* signal);
+ void initialiseFsConnectionRec(Signal* signal);
+ void initialiseFsOpRec(Signal* signal);
+ void initialiseLcpConnectionRec(Signal* signal);
+ void initialiseOperationRec(Signal* signal);
+ void initialiseOverflowRec(Signal* signal);
+ void initialisePageRec(Signal* signal);
+ void initialiseLcpPages(Signal* signal);
+ void initialiseRootfragRec(Signal* signal);
+ void initialiseScanRec(Signal* signal);
+ void initialiseSrVerRec(Signal* signal);
+ void initialiseTableRec(Signal* signal);
+ bool addfragtotab(Signal* signal, Uint32 rootIndex, Uint32 fragId);
+ void initOpRec(Signal* signal);
+ void sendAcckeyconf(Signal* signal);
+ Uint32 placeReadInLockQueue(Signal* signal);
+ void placeSerialQueueRead(Signal* signal);
+ void checkOnlyReadEntry(Signal* signal);
+ void getNoParallelTransaction(Signal* signal);
+ void moveLastParallelQueue(Signal* signal);
+ void moveLastParallelQueueWrite(Signal* signal);
+ Uint32 placeWriteInLockQueue(Signal* signal);
+ void placeSerialQueueWrite(Signal* signal);
+ void expandcontainer(Signal* signal);
+ void shrinkcontainer(Signal* signal);
+ void nextcontainerinfoExp(Signal* signal);
+ void lcpCopyPage(Signal* signal);
+ void lcpUpdatePage(Signal* signal);
+ void checkUndoPages(Signal* signal);
+ void undoWritingProcess(Signal* signal);
+ void writeUndoDataInfo(Signal* signal);
+ void writeUndoHeader(Signal* signal,
+ Uint32 logicalPageId,
+ UndoHeader::UndoHeaderType pageType);
+ void writeUndoOpInfo(Signal* signal);
+ void checksumControl(Signal* signal, Uint32 checkPage);
+ void startActiveUndo(Signal* signal);
+ void releaseAndCommitActiveOps(Signal* signal);
+ void releaseAndCommitQueuedOps(Signal* signal);
+ void releaseAndAbortLockedOps(Signal* signal);
+ void containerinfo(Signal* signal);
+ bool getScanElement(Signal* signal);
+ void initScanOpRec(Signal* signal);
+ void nextcontainerinfo(Signal* signal);
+ void putActiveScanOp(Signal* signal);
+ void putOpScanLockQue();
+ void putReadyScanQueue(Signal* signal, Uint32 scanRecIndex);
+ void releaseScanBucket(Signal* signal);
+ void releaseScanContainer(Signal* signal);
+ void releaseScanRec(Signal* signal);
+ bool searchScanContainer(Signal* signal);
+ void sendNextScanConf(Signal* signal);
+ void sendScaninfo(Signal* signal);
+ void setlock(Signal* signal);
+ void takeOutActiveScanOp(Signal* signal);
+ void takeOutScanLockQueue(Uint32 scanRecIndex);
+ void takeOutReadyScanQueue(Signal* signal);
+ void insertElement(Signal* signal);
+ void insertContainer(Signal* signal);
+ void addnewcontainer(Signal* signal);
+ void getfreelist(Signal* signal);
+ void increaselistcont(Signal* signal);
+ void seizeLeftlist(Signal* signal);
+ void seizeRightlist(Signal* signal);
+ void allocLongOverflowPage(Signal* signal);
+ void allocSpecificLongOverflowPage(Signal* signal);
+ void getLongKeyPage(Signal* signal);
+ void initLongOverpage(Signal* signal);
+ void storeLongKeys(Signal* signal);
+ void storeLongKeysAtPos(Signal* signal);
+ void reorgLongPage(Signal* signal);
+ void getElement(Signal* signal);
+ void searchLongKey(Signal* signal);
+ void getdirindex(Signal* signal);
+ void commitdelete(Signal* signal, bool systemRestart);
+ void deleteElement(Signal* signal);
+ void getLastAndRemove(Signal* signal);
+ void releaseLeftlist(Signal* signal);
+ void releaseRightlist(Signal* signal);
+ void checkoverfreelist(Signal* signal);
+ void deleteLongKey(Signal* signal);
+ void removeFromPageArrayList(Signal* signal);
+ void insertPageArrayList(Signal* signal);
+ void checkPageArrayList(Signal* signal, char *);
+ void checkPageB4Insert(Uint32, char *);
+ void checkPageB4Remove(Uint32, char *);
+ void checkIndexInLongKeyPage(Uint32, char *);
+ void printoutInfoAndShutdown(LongKeyPage *);
+ void releaseLongPage(Signal* signal);
+ void abortOperation(Signal* signal);
+ void accAbortReqLab(Signal* signal, bool sendConf);
+ void commitOperation(Signal* signal);
+ void copyOpInfo(Signal* signal);
+ Uint32 executeNextOperation(Signal* signal);
+ void releaselock(Signal* signal);
+ void takeOutFragWaitQue(Signal* signal);
+ void allocOverflowPage(Signal* signal);
+ bool getrootfragmentrec(Signal* signal, RootfragmentrecPtr&, Uint32 fragId);
+ void insertLockOwnersList(Signal* signal, const OperationrecPtr&);
+ void takeOutLockOwnersList(Signal* signal, const OperationrecPtr&);
+ void initFsOpRec(Signal* signal);
+ void initLcpConnRec(Signal* signal);
+ void initOverpage(Signal* signal);
+ void initPage(Signal* signal);
+ void initPageZero(Signal* signal);
+ void initRootfragrec(Signal* signal);
+ void putOpInFragWaitQue(Signal* signal);
+ void putOverflowRecInFrag(Signal* signal);
+ void putRecInFreeOverdir(Signal* signal);
+ void releaseDirectory(Signal* signal);
+ void releaseDirrange(Signal* signal);
+ void releaseFsConnRec(Signal* signal);
+ void releaseFsOpRec(Signal* signal);
+ void releaseLcpConnectRec(Signal* signal);
+ void releaseOpRec(Signal* signal);
+ void releaseOverflowRec(Signal* signal);
+ void releaseOverpage(Signal* signal);
+ void releasePage(Signal* signal);
+ void releaseLcpPage(Signal* signal);
+ void releaseSrRec(Signal* signal);
+ void releaseLogicalPage(Fragmentrec * fragP, Uint32 logicalPageId);
+ void seizeDirectory(Signal* signal);
+ void seizeDirrange(Signal* signal);
+ void seizeFragrec(Signal* signal);
+ void seizeFsConnectRec(Signal* signal);
+ void seizeFsOpRec(Signal* signal);
+ void seizeLcpConnectRec(Signal* signal);
+ void seizeOpRec(Signal* signal);
+ void seizeOverRec(Signal* signal);
+ void seizePage(Signal* signal);
+ void seizeLcpPage(Page8Ptr&);
+ void seizeRootfragrec(Signal* signal);
+ void seizeScanRec(Signal* signal);
+ void seizeSrVerRec(Signal* signal);
+ void sendSystemerror(Signal* signal);
+ void takeRecOutOfFreeOverdir(Signal* signal);
+ void takeRecOutOfFreeOverpage(Signal* signal);
+ void sendScanHbRep(Signal* signal, Uint32);
+
+ void addFragRefuse(Signal* signal, Uint32 errorCode);
+ void ndbsttorryLab(Signal* signal);
+ void srCloseDataFileLab(Signal* signal);
+ void acckeyref1Lab(Signal* signal, Uint32 result_code);
+ void insertelementLab(Signal* signal);
+ void startUndoLab(Signal* signal);
+ void checkNextFragmentLab(Signal* signal);
+ void endofexpLab(Signal* signal);
+ void endofshrinkbucketLab(Signal* signal);
+ void srStartUndoLab(Signal* signal);
+ void senddatapagesLab(Signal* signal);
+ void undoNext2Lab(Signal* signal);
+ void sttorrysignalLab(Signal* signal);
+ void sendholdconfsignalLab(Signal* signal);
+ void accIsLockedLab(Signal* signal);
+ void insertExistElemLab(Signal* signal);
+ void refaccConnectLab(Signal* signal);
+ void srReadOverPagesLab(Signal* signal);
+ void releaseScanLab(Signal* signal);
+ void exeoperationLab(Signal* signal);
+ void saveKeyDataLab(Signal* signal);
+ void lcpOpenUndofileConfLab(Signal* signal);
+ void srFsOpenConfLab(Signal* signal);
+ void checkSyncUndoPagesLab(Signal* signal);
+ void sendaccSrconfLab(Signal* signal);
+ void checkSendLcpConfLab(Signal* signal);
+ void endsaveoverpageLab(Signal* signal);
+ void lcpCloseDataFileLab(Signal* signal);
+ void srOpenDataFileLoopLab(Signal* signal);
+ void srReadPagesLab(Signal* signal);
+ void srDoUndoLab(Signal* signal);
+ void ndbrestart1Lab(Signal* signal);
+ void initialiseRecordsLab(Signal* signal);
+ void srReadPagesAllocLab(Signal* signal);
+ void checkNextBucketLab(Signal* signal);
+ void endsavepageLab(Signal* signal);
+ void saveZeroPageLab(Signal* signal);
+ void srAllocPage0011Lab(Signal* signal);
+ void allocscanrecLab(Signal* signal);
+ void sendLcpFragidconfLab(Signal* signal);
+ void savepagesLab(Signal* signal);
+ void saveOverPagesLab(Signal* signal);
+ void srReadPageZeroLab(Signal* signal);
+ void storeDataPageInDirectoryLab(Signal* signal);
+ void lcpFsOpenConfLab(Signal* signal);
+
+ void zpagesize_error(const char* where);
+
+ void reportMemoryUsage(Signal* signal, int gth);
+
+
+ // Initialisation
+ void initData();
+ void initRecords();
+
+ // Variables
+/* --------------------------------------------------------------------------------- */
+/* DIRECTORY RANGE */
+/* --------------------------------------------------------------------------------- */
+ DirRange *dirRange;
+ DirRangePtr expDirRangePtr;
+ DirRangePtr gnsDirRangePtr;
+ DirRangePtr newDirRangePtr;
+ DirRangePtr rdDirRangePtr;
+ DirRangePtr nciOverflowrangeptr;
+ Uint32 cdirrangesize;
+ Uint32 cfirstfreeDirrange;
+/* --------------------------------------------------------------------------------- */
+/* DIRECTORYARRAY */
+/* --------------------------------------------------------------------------------- */
+ Directoryarray *directoryarray;
+ DirectoryarrayPtr expDirptr;
+ DirectoryarrayPtr rdDirptr;
+ DirectoryarrayPtr sdDirptr;
+ DirectoryarrayPtr nciOverflowDirptr;
+ Uint32 cdirarraysize;
+ Uint32 cdirmemory;
+ Uint32 cfirstfreedir;
+/* --------------------------------------------------------------------------------- */
+/* FRAGMENTREC. ALL INFORMATION ABOUT FRAMENT AND HASH TABLE IS SAVED IN FRAGMENT */
+/* REC A POINTER TO FRAGMENT RECORD IS SAVED IN ROOTFRAGMENTREC FRAGMENT */
+/* --------------------------------------------------------------------------------- */
+ Fragmentrec *fragmentrec;
+ FragmentrecPtr fragrecptr;
+ Uint32 cfirstfreefrag;
+ Uint32 cfragmentsize;
+/* --------------------------------------------------------------------------------- */
+/* FS_CONNECTREC */
+/* --------------------------------------------------------------------------------- */
+ FsConnectrec *fsConnectrec;
+ FsConnectrecPtr fsConnectptr;
+ Uint32 cfsConnectsize;
+ Uint32 cfsFirstfreeconnect;
+/* --------------------------------------------------------------------------------- */
+/* FS_OPREC */
+/* --------------------------------------------------------------------------------- */
+ FsOprec *fsOprec;
+ FsOprecPtr fsOpptr;
+ Uint32 cfsOpsize;
+ Uint32 cfsFirstfreeop;
+/* --------------------------------------------------------------------------------- */
+/* LCP_CONNECTREC */
+/* --------------------------------------------------------------------------------- */
+ LcpConnectrec *lcpConnectrec;
+ LcpConnectrecPtr lcpConnectptr;
+ Uint32 clcpConnectsize;
+ Uint32 cfirstfreelcpConnect;
+/* --------------------------------------------------------------------------------- */
+/* OPERATIONREC */
+/* --------------------------------------------------------------------------------- */
+ Operationrec *operationrec;
+ OperationrecPtr operationRecPtr;
+ OperationrecPtr idrOperationRecPtr;
+ OperationrecPtr copyInOperPtr;
+ OperationrecPtr copyOperPtr;
+ OperationrecPtr mlpqOperPtr;
+ OperationrecPtr queOperPtr;
+ OperationrecPtr readWriteOpPtr;
+ OperationrecPtr tgnptMainOpPtr;
+ Uint32 cfreeopRec;
+ Uint32 coprecsize;
+/* --------------------------------------------------------------------------------- */
+/* OVERFLOW_RECORD */
+/* --------------------------------------------------------------------------------- */
+ OverflowRecord *overflowRecord;
+ OverflowRecordPtr iopOverflowRecPtr;
+ OverflowRecordPtr tfoOverflowRecPtr;
+ OverflowRecordPtr porOverflowRecPtr;
+ OverflowRecordPtr priOverflowRecPtr;
+ OverflowRecordPtr rorOverflowRecPtr;
+ OverflowRecordPtr sorOverflowRecPtr;
+ OverflowRecordPtr troOverflowRecPtr;
+ Uint32 cfirstfreeoverrec;
+ Uint32 coverflowrecsize;
+
+/* --------------------------------------------------------------------------------- */
+/* PAGE8 */
+/* --------------------------------------------------------------------------------- */
+ Page8 *page8;
+ /* 8 KB PAGE */
+ Page8Ptr aslpPageptr;
+ Page8Ptr alpPageptr;
+ Page8Ptr ancPageptr;
+ Page8Ptr colPageptr;
+ Page8Ptr ccoPageptr;
+ Page8Ptr datapageptr;
+ Page8Ptr delPageptr;
+ Page8Ptr excPageptr;
+ Page8Ptr expPageptr;
+ Page8Ptr gdiPageptr;
+ Page8Ptr gePageptr;
+ Page8Ptr gflPageptr;
+ Page8Ptr glkPageptr;
+ Page8Ptr idrPageptr;
+ Page8Ptr ilcPageptr;
+ Page8Ptr iloPageptr;
+ Page8Ptr inpPageptr;
+ Page8Ptr iopPageptr;
+ Page8Ptr ipzPageptr;
+ Page8Ptr lastPageptr;
+ Page8Ptr lastPrevpageptr;
+ Page8Ptr lcnPageptr;
+ Page8Ptr lcnCopyPageptr;
+ Page8Ptr lupPageptr;
+ Page8Ptr dlkPageptr;
+ Page8Ptr ipaPagePtr;
+ Page8Ptr priPageptr;
+ Page8Ptr pwiPageptr;
+ Page8Ptr rfpPageptr;
+ Page8Ptr relpPageptr;
+ Page8Ptr rlopPageptr;
+ Page8Ptr slkPageptr;
+ Page8Ptr slkCopyPageptr;
+ Page8Ptr slkapPageptr;
+ Page8Ptr slkapCopyPageptr;
+ Page8Ptr ciPageidptr;
+ Page8Ptr gsePageidptr;
+ Page8Ptr isoPageptr;
+ Page8Ptr nciPageidptr;
+ Page8Ptr rsbPageidptr;
+ Page8Ptr rscPageidptr;
+ Page8Ptr slPageidptr;
+ Page8Ptr sscPageidptr;
+ Page8Ptr rlPageptr;
+ Page8Ptr rlpPageptr;
+ Page8Ptr ropPageptr;
+ Page8Ptr rpPageptr;
+ Page8Ptr slPageptr;
+ Page8Ptr slpPageptr;
+ Page8Ptr spPageptr;
+ Uint32 cfirstfreepage;
+ Uint32 cfreepage;
+ Uint32 cpagesize;
+ Uint32 cfirstfreeLcpPage;
+ Uint32 cnoOfAllocatedPages;
+ Uint32 cnoLcpPages;
+/* --------------------------------------------------------------------------------- */
+/* ROOTFRAGMENTREC */
+/* DURING EXPAND FRAGMENT PROCESS, EACH FRAGMEND WILL BE EXPAND INTO TWO */
+/* NEW FRAGMENTS.TO MAKE THIS PROCESS EASIER, DURING ADD FRAGMENT PROCESS */
+/* NEXT FRAGMENT IDENTIIES WILL BE CALCULATED, AND TWO FRAGMENTS WILL BE */
+/* ADDED IN (NDBACC). THEREBY EXPAND OF FRAGMENT CAN BE PERFORMED QUICK AND */
+/* EASY.THE NEW FRAGMENT ID SENDS TO TUP MANAGER FOR ALL OPERATION PROCESS. */
+/* --------------------------------------------------------------------------------- */
+ Rootfragmentrec *rootfragmentrec;
+ RootfragmentrecPtr rootfragrecptr;
+ RootfragmentrecPtr tmprootfrgptr;
+ Uint32 crootfragmentsize;
+ Uint32 cfirstfreerootfrag;
+/* --------------------------------------------------------------------------------- */
+/* SCAN_REC */
+/* --------------------------------------------------------------------------------- */
+ ScanRec *scanRec;
+ ScanRecPtr scanPtr;
+ Uint32 cscanRecSize;
+ Uint32 cfirstFreeScanRec;
+/* --------------------------------------------------------------------------------- */
+/* SR_VERSION_REC */
+/* --------------------------------------------------------------------------------- */
+ SrVersionRec *srVersionRec;
+ SrVersionRecPtr srVersionPtr;
+ Uint32 csrVersionRecSize;
+ Uint32 cfirstFreeSrVersionRec;
+/* --------------------------------------------------------------------------------- */
+/* TABREC */
+/* --------------------------------------------------------------------------------- */
+ Tabrec *tabrec;
+ TabrecPtr tabptr;
+ Uint32 ctablesize;
+/* --------------------------------------------------------------------------------- */
+/* UNDOPAGE */
+/* --------------------------------------------------------------------------------- */
+ Undopage *undopage;
+ /* 32 KB PAGE */
+ UndopagePtr undopageptr;
+ Uint32 tpwiElementptr;
+ Uint32 tpriElementptr;
+ Uint32 tgseElementptr;
+ Uint32 tgseContainerptr;
+ Uint32 tiloIndex;
+ Uint32 trlHead;
+ Uint32 trlRelCon;
+ Uint32 trlNextused;
+ Uint32 trlPrevused;
+ Uint32 tlcnChecksum;
+ Uint32 tlupElemIndex;
+ Uint32 tlupIndex;
+ Uint32 tlupForward;
+ Uint32 tslkPageIndex;
+ Uint32 tslkKeyLen;
+ Uint32 tslkapKeyLen;
+ Uint32 tslkapPageIndex;
+ Uint32 tipaArrayPos;
+ Uint32 trfpArrayPos;
+ Uint32 tdlkLogicalPageIndex;
+ Uint32 tancNext;
+ Uint32 tancBufType;
+ Uint32 tancContainerptr;
+ Uint32 tancPageindex;
+ Uint32 tancPageid;
+ Uint32 tidrResult;
+ Uint32 tidrKeyLen;
+ Uint32 tidrElemhead;
+ Uint32 tidrForward;
+ Uint32 tidrPageindex;
+ Uint32 tidrContainerptr;
+ Uint32 tidrContainerhead;
+ Uint32 tlastForward;
+ Uint32 tlastPageindex;
+ Uint32 tlastContainerlen;
+ Uint32 tlastElementptr;
+ Uint32 tlastContainerptr;
+ Uint32 tlastContainerhead;
+ Uint32 trlPageindex;
+ Uint32 tdelContainerptr;
+ Uint32 tdelElementptr;
+ Uint32 tdelForward;
+ Uint32 tiopPageId;
+ Uint32 tipPageId;
+ Uint32 ttupKeyLength;
+ Uint32 tgeLocked;
+ Uint32 tgeResult;
+ Uint32 tgeContainerptr;
+ Uint32 tgeElementptr;
+ Uint32 tgeForward;
+ Uint32 tslcResult;
+ Uint32 tslcPagedir;
+ Uint32 tslcPageIndex;
+ Uint32 tundoElemIndex;
+ Uint32 texpReceivedBucket;
+ Uint32 texpDirInd;
+ Uint32 texpDirRangeIndex;
+ Uint32 texpDirPageIndex;
+ Uint32 tdata0;
+ Uint32 tcheckpointid;
+ Uint32 tciContainerptr;
+ Uint32 tnciContainerptr;
+ Uint32 tisoContainerptr;
+ Uint32 trscContainerptr;
+ Uint32 tsscContainerptr;
+ Uint32 tciContainerlen;
+ Uint32 trscContainerlen;
+ Uint32 tsscContainerlen;
+ Uint32 tciContainerhead;
+ Uint32 tnciContainerhead;
+ Uint32 tslElementptr;
+ Uint32 tisoElementptr;
+ Uint32 tsscElementptr;
+ Uint32 tfid;
+ Uint32 tscanFlag;
+ Uint32 theadundoindex;
+ Uint32 tgflBufType;
+ Uint32 thashvalue;
+ Uint32 tgseIsforward;
+ Uint32 tsscIsforward;
+ Uint32 trscIsforward;
+ Uint32 tciIsforward;
+ Uint32 tnciIsforward;
+ Uint32 tisoIsforward;
+ Uint32 tgseIsLocked;
+ Uint32 tsscIsLocked;
+ Uint32 tkey1;
+ Uint32 tkey2;
+ Uint32 tkey3;
+ Uint32 tkey4;
+ Uint32 tkeylen;
+ Uint32 tkSize;
+ Uint32 tlhfragbits;
+ Uint32 tlhdirbits;
+ Uint32 tlocalkeylen;
+ Uint32 tmaxloadfactor;
+ Uint32 tminloadfactor;
+ Uint32 tmp;
+ Uint32 tmpP;
+ Uint32 tmpP2;
+ Uint32 taslpDirIndex;
+ Uint32 tmp1;
+ Uint32 tmp2;
+ Uint32 tgflPageindex;
+ Uint32 tmpindex;
+ Uint32 tslNextfree;
+ Uint32 tslPageindex;
+ Uint32 tgsePageindex;
+ Uint32 tnciNextSamePage;
+ Uint32 tslPrevfree;
+ Uint32 tciPageindex;
+ Uint32 trsbPageindex;
+ Uint32 tnciPageindex;
+ Uint32 tlastPrevconptr;
+ Uint32 treqinfo;
+ Uint32 transactionid1;
+ Uint32 transactionid2;
+ Uint32 tresult;
+ Uint32 tslUpdateHeader;
+ Uint32 tuserptr;
+ BlockReference tuserblockref;
+ Uint32 tundoindex;
+ Uint32 tlqhPointer;
+ Uint32 tholdSentOp;
+ Uint32 tholdMore;
+ Uint32 tlcpLqhCheckV;
+ Uint32 tgdiPageindex;
+ Uint32 tiopIndex;
+ Uint32 tnciTmp;
+ Uint32 tlenKeyinfo;
+ Uint32 tullIndex;
+ Uint32 turlIndex;
+ Uint32 tlfrTmp1;
+ Uint32 tlfrTmp2;
+ Uint32 tgnptNrTransaction;
+ Uint32 tudqeIndex;
+ Uint32 tscanTrid1;
+ Uint32 tscanTrid2;
+ Uint32 taccscanTmp;
+
+ Uint16 clastUndoPageIdWritten;
+ Uint32 cactiveCheckpId;
+ Uint32 cactiveRootfrag;
+ Uint32 cactiveSrFsPtr;
+ Uint32 cactiveUndoFilePage;
+ Uint32 cactiveOpenUndoFsPtr;
+ Uint32 cactiveSrUndoPage;
+ Uint32 cprevUndoaddress;
+ Uint32 creadyUndoaddress;
+ Uint32 ctest;
+ Uint32 cundoLogActive;
+ Uint32 clqhPtr;
+ BlockReference clqhBlockRef;
+ Uint32 cminusOne;
+ NodeId cmynodeid;
+ Uint32 cactiveUndoFileVersion;
+ BlockReference cownBlockref;
+ BlockReference cndbcntrRef;
+ Uint16 csignalkey;
+ Uint32 cundopagesize;
+ Uint32 cundoposition;
+ Uint32 cundoElemIndex;
+ Uint32 cundoinfolength;
+ Uint32 czero;
+ Uint32 csrVersList[16];
+ Uint32 clblPageCounter;
+ Uint32 clblPageOver;
+ Uint32 clblPagesPerTick;
+ Uint32 clblPagesPerTickAfterSr;
+ Uint32 csystemRestart;
+ Uint32 cexcForward;
+ Uint32 cexcPageindex;
+ Uint32 cexcContainerptr;
+ Uint32 cexcContainerhead;
+ Uint32 cexcContainerlen;
+ Uint32 cexcElementptr;
+ Uint32 cexcPrevconptr;
+ Uint32 cexcMovedLen;
+ Uint32 cexcPrevpageptr;
+ Uint32 cexcPrevpageindex;
+ Uint32 cexcPrevforward;
+ Uint32 clocalkey[32];
+ Uint32 ckeys[2048];
+
+ Uint32 c_errorInsert3000_TableId;
+ Uint32 cSrUndoRecords[5];
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/dbacc/DbaccInit.cpp b/ndb/src/kernel/blocks/dbacc/DbaccInit.cpp
new file mode 100644
index 00000000000..107420c7148
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbacc/DbaccInit.cpp
@@ -0,0 +1,248 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#define DBACC_C
+#include "Dbacc.hpp"
+
+#define DEBUG(x) { ndbout << "ACC::" << x << endl; }
+
+void Dbacc::initData()
+{
+ cdirarraysize = ZDIRARRAY;
+ coprecsize = ZOPRECSIZE;
+ cpagesize = ZPAGESIZE;
+ clcpConnectsize = ZLCP_CONNECTSIZE;
+ ctablesize = ZTABLESIZE;
+ cfragmentsize = ZFRAGMENTSIZE;
+ crootfragmentsize = ZROOTFRAGMENTSIZE;
+ cdirrangesize = ZDIRRANGESIZE;
+ coverflowrecsize = ZOVERFLOWRECSIZE;
+ cundopagesize = ZUNDOPAGESIZE;
+ cfsConnectsize = ZFS_CONNECTSIZE;
+ cfsOpsize = ZFS_OPSIZE;
+ cscanRecSize = ZSCAN_REC_SIZE;
+ csrVersionRecSize = ZSR_VERSION_REC_SIZE;
+
+
+ dirRange = 0;
+ directoryarray = 0;
+ fragmentrec = 0;
+ fsConnectrec = 0;
+ fsOprec = 0;
+ lcpConnectrec = 0;
+ operationrec = 0;
+ overflowRecord = 0;
+ page8 = 0;
+ rootfragmentrec = 0;
+ scanRec = 0;
+ srVersionRec = 0;
+ tabrec = 0;
+ undopage = 0;
+
+ // Records with constant sizes
+}//Dbacc::initData()
+
+void Dbacc::initRecords()
+{
+ // Records with dynamic sizes
+ dirRange = (DirRange*)allocRecord("DirRange",
+ sizeof(DirRange),
+ cdirrangesize);
+
+ directoryarray = (Directoryarray*)allocRecord("Directoryarray",
+ sizeof(Directoryarray),
+ cdirarraysize);
+
+ fragmentrec = (Fragmentrec*)allocRecord("Fragmentrec",
+ sizeof(Fragmentrec),
+ cfragmentsize);
+
+ fsConnectrec = (FsConnectrec*)allocRecord("FsConnectrec",
+ sizeof(FsConnectrec),
+ cfsConnectsize);
+
+ fsOprec = (FsOprec*)allocRecord("FsOprec",
+ sizeof(FsOprec),
+ cfsOpsize);
+
+ lcpConnectrec = (LcpConnectrec*)allocRecord("LcpConnectrec",
+ sizeof(LcpConnectrec),
+ clcpConnectsize);
+
+ operationrec = (Operationrec*)allocRecord("Operationrec",
+ sizeof(Operationrec),
+ coprecsize);
+
+ overflowRecord = (OverflowRecord*)allocRecord("OverflowRecord",
+ sizeof(OverflowRecord),
+ coverflowrecsize);
+
+ page8 = (Page8*)allocRecord("Page8",
+ sizeof(Page8),
+ cpagesize);
+
+ rootfragmentrec = (Rootfragmentrec*)allocRecord("Rootfragmentrec",
+ sizeof(Rootfragmentrec),
+ crootfragmentsize);
+
+ scanRec = (ScanRec*)allocRecord("ScanRec",
+ sizeof(ScanRec),
+ cscanRecSize);
+
+ srVersionRec = (SrVersionRec*)allocRecord("SrVersionRec",
+ sizeof(SrVersionRec),
+ csrVersionRecSize);
+
+ tabrec = (Tabrec*)allocRecord("Tabrec",
+ sizeof(Tabrec),
+ ctablesize);
+
+ undopage = (Undopage*)allocRecord("Undopage",
+ sizeof(Undopage),
+ cundopagesize);
+
+ // Initialize BAT for interface to file system
+
+ NewVARIABLE* bat = allocateBat(3);
+ bat[1].WA = &page8->word32[0];
+ bat[1].nrr = cpagesize;
+ bat[1].ClusterSize = sizeof(Page8);
+ bat[1].bits.q = 11;
+ bat[1].bits.v = 5;
+ bat[2].WA = &undopage->undoword[0];
+ bat[2].nrr = cundopagesize;
+ bat[2].ClusterSize = sizeof(Undopage);
+ bat[2].bits.q = 13;
+ bat[2].bits.v = 5;
+}//Dbacc::initRecords()
+
+Dbacc::Dbacc(const class Configuration & conf):
+ SimulatedBlock(DBACC, conf)
+{
+ BLOCK_CONSTRUCTOR(Dbacc);
+
+ // Transit signals
+ addRecSignal(GSN_DUMP_STATE_ORD, &Dbacc::execDUMP_STATE_ORD);
+ addRecSignal(GSN_DEBUG_SIG, &Dbacc::execDEBUG_SIG);
+ addRecSignal(GSN_CONTINUEB, &Dbacc::execCONTINUEB);
+ addRecSignal(GSN_ACC_CHECK_SCAN, &Dbacc::execACC_CHECK_SCAN);
+ addRecSignal(GSN_EXPANDCHECK2, &Dbacc::execEXPANDCHECK2);
+ addRecSignal(GSN_SHRINKCHECK2, &Dbacc::execSHRINKCHECK2);
+ addRecSignal(GSN_ACC_OVER_REC, &Dbacc::execACC_OVER_REC);
+ addRecSignal(GSN_ACC_SAVE_PAGES, &Dbacc::execACC_SAVE_PAGES);
+ addRecSignal(GSN_NEXTOPERATION, &Dbacc::execNEXTOPERATION);
+
+ // Received signals
+ addRecSignal(GSN_STTOR, &Dbacc::execSTTOR);
+ addRecSignal(GSN_SR_FRAGIDREQ, &Dbacc::execSR_FRAGIDREQ);
+ addRecSignal(GSN_LCP_FRAGIDREQ, &Dbacc::execLCP_FRAGIDREQ);
+ addRecSignal(GSN_LCP_HOLDOPREQ, &Dbacc::execLCP_HOLDOPREQ);
+ addRecSignal(GSN_END_LCPREQ, &Dbacc::execEND_LCPREQ);
+ addRecSignal(GSN_ACC_LCPREQ, &Dbacc::execACC_LCPREQ);
+ addRecSignal(GSN_START_RECREQ, &Dbacc::execSTART_RECREQ);
+ addRecSignal(GSN_ACC_CONTOPREQ, &Dbacc::execACC_CONTOPREQ);
+ addRecSignal(GSN_ACCKEYREQ, &Dbacc::execACCKEYREQ);
+ addRecSignal(GSN_ACCSEIZEREQ, &Dbacc::execACCSEIZEREQ);
+ addRecSignal(GSN_ACCFRAGREQ, &Dbacc::execACCFRAGREQ);
+ addRecSignal(GSN_ACC_SRREQ, &Dbacc::execACC_SRREQ);
+ addRecSignal(GSN_NEXT_SCANREQ, &Dbacc::execNEXT_SCANREQ);
+ addRecSignal(GSN_ACC_ABORTREQ, &Dbacc::execACC_ABORTREQ);
+ addRecSignal(GSN_ACC_SCANREQ, &Dbacc::execACC_SCANREQ);
+ addRecSignal(GSN_ACCMINUPDATE, &Dbacc::execACCMINUPDATE);
+ addRecSignal(GSN_ACC_COMMITREQ, &Dbacc::execACC_COMMITREQ);
+ addRecSignal(GSN_ACC_TO_REQ, &Dbacc::execACC_TO_REQ);
+ addRecSignal(GSN_ACC_LOCKREQ, &Dbacc::execACC_LOCKREQ);
+ addRecSignal(GSN_FSOPENCONF, &Dbacc::execFSOPENCONF);
+ addRecSignal(GSN_FSOPENREF, &Dbacc::execFSOPENREF);
+ addRecSignal(GSN_FSCLOSECONF, &Dbacc::execFSCLOSECONF);
+ addRecSignal(GSN_FSCLOSEREF, &Dbacc::execFSCLOSEREF);
+ addRecSignal(GSN_FSWRITECONF, &Dbacc::execFSWRITECONF);
+ addRecSignal(GSN_FSWRITEREF, &Dbacc::execFSWRITEREF);
+ addRecSignal(GSN_FSREADCONF, &Dbacc::execFSREADCONF);
+ addRecSignal(GSN_FSREADREF, &Dbacc::execFSREADREF);
+ addRecSignal(GSN_NDB_STTOR, &Dbacc::execNDB_STTOR);
+ addRecSignal(GSN_DROP_TAB_REQ, &Dbacc::execDROP_TAB_REQ);
+ addRecSignal(GSN_FSREMOVECONF, &Dbacc::execFSREMOVECONF);
+ addRecSignal(GSN_FSREMOVEREF, &Dbacc::execFSREMOVEREF);
+ addRecSignal(GSN_SIZEALT_REP, &Dbacc::execSIZEALT_REP);
+ addRecSignal(GSN_SET_VAR_REQ, &Dbacc::execSET_VAR_REQ);
+
+ initData();
+}//Dbacc::Dbacc()
+
+Dbacc::~Dbacc()
+{
+ deallocRecord((void **)&dirRange, "DirRange",
+ sizeof(DirRange),
+ cdirrangesize);
+
+ deallocRecord((void **)&directoryarray, "Directoryarray",
+ sizeof(Directoryarray),
+ cdirarraysize);
+
+ deallocRecord((void **)&fragmentrec, "Fragmentrec",
+ sizeof(Fragmentrec),
+ cfragmentsize);
+
+ deallocRecord((void **)&fsConnectrec, "FsConnectrec",
+ sizeof(FsConnectrec),
+ cfsConnectsize);
+
+ deallocRecord((void **)&fsOprec, "FsOprec",
+ sizeof(FsOprec),
+ cfsOpsize);
+
+ deallocRecord((void **)&lcpConnectrec, "LcpConnectrec",
+ sizeof(LcpConnectrec),
+ clcpConnectsize);
+
+ deallocRecord((void **)&operationrec, "Operationrec",
+ sizeof(Operationrec),
+ coprecsize);
+
+ deallocRecord((void **)&overflowRecord, "OverflowRecord",
+ sizeof(OverflowRecord),
+ coverflowrecsize);
+
+ deallocRecord((void **)&page8, "Page8",
+ sizeof(Page8),
+ cpagesize);
+
+ deallocRecord((void **)&rootfragmentrec, "Rootfragmentrec",
+ sizeof(Rootfragmentrec),
+ crootfragmentsize);
+
+ deallocRecord((void **)&scanRec, "ScanRec",
+ sizeof(ScanRec),
+ cscanRecSize);
+
+ deallocRecord((void **)&srVersionRec, "SrVersionRec",
+ sizeof(SrVersionRec),
+ csrVersionRecSize);
+
+ deallocRecord((void **)&tabrec, "Tabrec",
+ sizeof(Tabrec),
+ ctablesize);
+
+ deallocRecord((void **)&undopage, "Undopage",
+ sizeof(Undopage),
+ cundopagesize);
+
+}//Dbacc::~Dbacc()
+
+BLOCK_FUNCTIONS(Dbacc);
diff --git a/ndb/src/kernel/blocks/dbacc/DbaccMain.cpp b/ndb/src/kernel/blocks/dbacc/DbaccMain.cpp
new file mode 100644
index 00000000000..ea8d808458b
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbacc/DbaccMain.cpp
@@ -0,0 +1,13285 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBACC_C
+#include "Dbacc.hpp"
+
+#include <signaldata/AccFrag.hpp>
+#include <signaldata/AccScan.hpp>
+#include <signaldata/AccLock.hpp>
+#include <signaldata/AccSizeAltReq.hpp>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/FsConf.hpp>
+#include <signaldata/FsRef.hpp>
+#include <signaldata/FsRemoveReq.hpp>
+#include <signaldata/DropTab.hpp>
+#include <signaldata/SetVarReq.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+
+// TO_DO_RONM is a label for comments on what needs to be improved in future versions
+// when more time is given.
+
+#ifdef VM_TRACE
+#define DEBUG(x) ndbout << "DBACC: "<< x << endl;
+#else
+#define DEBUG(x)
+#endif
+
+
+Uint32
+Dbacc::remainingUndoPages(){
+ Uint32 HeadPage = cundoposition >> ZUNDOPAGEINDEXBITS;
+ Uint32 TailPage = clastUndoPageIdWritten;
+
+ // Head must be larger or same as tail
+ ndbrequire(HeadPage>=TailPage);
+
+ Uint32 UsedPages = HeadPage - TailPage;
+ Uint32 Remaining = cundopagesize - UsedPages;
+
+ // There can not be more than cundopagesize remaining
+ ndbrequire(Remaining<=cundopagesize);
+
+ return Remaining;
+}//Dbacc::remainingUndoPages()
+
+void
+Dbacc::updateLastUndoPageIdWritten(Signal* signal, Uint32 aNewValue){
+ if (remainingUndoPages() < ZMIN_UNDO_PAGES_AT_COMMIT) {
+ clastUndoPageIdWritten = aNewValue;
+ if (remainingUndoPages() >= ZMIN_UNDO_PAGES_AT_COMMIT) {
+ jam();
+ EXECUTE_DIRECT(DBLQH, GSN_ACC_COM_UNBLOCK, signal, 1);
+ jamEntry();
+ }//if
+ } else {
+ clastUndoPageIdWritten = aNewValue;
+ }//if
+}//Dbacc::updateLastUndoPageIdWritten()
+
+void
+Dbacc::updateUndoPositionPage(Signal* signal, Uint32 aNewValue){
+ if (remainingUndoPages() >= ZMIN_UNDO_PAGES_AT_COMMIT) {
+ cundoposition = aNewValue;
+ if (remainingUndoPages() < ZMIN_UNDO_PAGES_AT_COMMIT) {
+ jam();
+ EXECUTE_DIRECT(DBLQH, GSN_ACC_COM_BLOCK, signal, 1);
+ jamEntry();
+ }//if
+ } else {
+ cundoposition = aNewValue;
+ }//if
+}//Dbacc::updateUndoPositionPage()
+
+// Signal entries and statement blocks
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* COMMON SIGNAL RECEPTION MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+
+/* --------------------------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* CONTINUEB CONTINUE SIGNAL */
+/* ******************------------------------------+ */
+/* SENDER: ACC, LEVEL B */
+void Dbacc::execCONTINUEB(Signal* signal)
+{
+ Uint32 tcase;
+
+ jamEntry();
+ tcase = signal->theData[0];
+ tdata0 = signal->theData[1];
+ tresult = 0;
+ switch (tcase) {
+ case ZLOAD_BAL_LCP_TIMER:
+ if (clblPageOver == 0) {
+ jam();
+ clblPageCounter = clblPagesPerTick;
+ } else {
+ if (clblPageOver > clblPagesPerTick) {
+ jam();
+ clblPageOver = clblPageOver - clblPagesPerTick;
+ } else {
+ jam();
+ clblPageOver = 0;
+ clblPageCounter = clblPagesPerTick - clblPageOver;
+ }//if
+ }//if
+ signal->theData[0] = ZLOAD_BAL_LCP_TIMER;
+ sendSignalWithDelay(cownBlockref, GSN_CONTINUEB, signal, 100, 1);
+ return;
+ break;
+ case ZINITIALISE_RECORDS:
+ jam();
+ initialiseRecordsLab(signal);
+ return;
+ break;
+ case ZSR_READ_PAGES_ALLOC:
+ jam();
+ fragrecptr.i = tdata0;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ srReadPagesAllocLab(signal);
+ return;
+ break;
+ case ZSTART_UNDO:
+ jam();
+ startUndoLab(signal);
+ return;
+ break;
+ case ZSEND_SCAN_HBREP:
+ jam();
+ sendScanHbRep(signal, tdata0);
+ break;
+ case ZREL_ROOT_FRAG:
+ {
+ jam();
+ Uint32 tableId = signal->theData[1];
+ releaseRootFragResources(signal, tableId);
+ break;
+ }
+ case ZREL_FRAG:
+ {
+ jam();
+ Uint32 fragIndex = signal->theData[1];
+ releaseFragResources(signal, fragIndex);
+ break;
+ }
+ case ZREL_DIR:
+ {
+ jam();
+ Uint32 fragIndex = signal->theData[1];
+ Uint32 dirIndex = signal->theData[2];
+ Uint32 startIndex = signal->theData[3];
+ releaseDirResources(signal, fragIndex, dirIndex, startIndex);
+ break;
+ }
+ case ZREPORT_MEMORY_USAGE:{
+ jam();
+ static int c_currentMemUsed = 0;
+ int now = (cnoOfAllocatedPages * 100)/cpagesize;
+ const int thresholds[] = { 99, 90, 80, 0};
+
+ Uint32 i = 0;
+ const Uint32 sz = sizeof(thresholds)/sizeof(thresholds[0]);
+ for(i = 0; i<sz; i++){
+ if(now >= thresholds[i]){
+ now = thresholds[i];
+ break;
+ }
+ }
+
+ if(now != c_currentMemUsed){
+ reportMemoryUsage(signal, now > c_currentMemUsed ? 1 : -1);
+ }
+
+ c_currentMemUsed = now;
+
+ signal->theData[0] = ZREPORT_MEMORY_USAGE;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 2000, 1);
+ return;
+ }
+
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbacc::execCONTINUEB()
+
+/* ******************--------------------------------------------------------------- */
+/* FSCLOSECONF CLOSE FILE CONF */
+/* ******************------------------------------+ */
+/* SENDER: FS, LEVEL B */
+void Dbacc::execFSCLOSECONF(Signal* signal)
+{
+ jamEntry();
+ fsConnectptr.i = signal->theData[0];
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ tresult = 0;
+ switch (fsConnectptr.p->fsState) {
+ case WAIT_CLOSE_UNDO:
+ jam();
+ releaseFsConnRec(signal);
+ break;
+ case LCP_CLOSE_DATA:
+ jam();
+ checkSyncUndoPagesLab(signal);
+ return;
+ break;
+ case SR_CLOSE_DATA:
+ jam();
+ sendaccSrconfLab(signal);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbacc::execFSCLOSECONF()
+
+/* ******************--------------------------------------------------------------- */
+/* FSCLOSEREF OPENFILE CONF */
+/* ******************------------------------------+ */
+/* SENDER: FS, LEVEL B */
+void Dbacc::execFSCLOSEREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dbacc::execFSCLOSEREF()
+
+/* ******************--------------------------------------------------------------- */
+/* FSOPENCONF OPENFILE CONF */
+/* ******************------------------------------+ */
+/* SENDER: FS, LEVEL B */
+void Dbacc::execFSOPENCONF(Signal* signal)
+{
+ jamEntry();
+ fsConnectptr.i = signal->theData[0];
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ tuserptr = signal->theData[1];
+ tresult = 0; /* RESULT CHECK VALUE */
+ switch (fsConnectptr.p->fsState) {
+ case WAIT_OPEN_UNDO_LCP:
+ jam();
+ lcpOpenUndofileConfLab(signal);
+ return;
+ break;
+ case WAIT_OPEN_UNDO_LCP_NEXT:
+ jam();
+ fsConnectptr.p->fsPtr = tuserptr;
+ return;
+ break;
+ case OPEN_UNDO_FILE_SR:
+ jam();
+ fsConnectptr.p->fsPtr = tuserptr;
+ srStartUndoLab(signal);
+ return;
+ break;
+ case WAIT_OPEN_DATA_FILE_FOR_WRITE:
+ jam();
+ lcpFsOpenConfLab(signal);
+ return;
+ break;
+ case WAIT_OPEN_DATA_FILE_FOR_READ:
+ jam();
+ fsConnectptr.p->fsPtr = tuserptr;
+ srFsOpenConfLab(signal);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbacc::execFSOPENCONF()
+
+/* ******************--------------------------------------------------------------- */
+/* FSOPENREF OPENFILE REF */
+/* ******************------------------------------+ */
+/* SENDER: FS, LEVEL B */
+void Dbacc::execFSOPENREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dbacc::execFSOPENREF()
+
+/* ******************--------------------------------------------------------------- */
+/* FSREADCONF OPENFILE CONF */
+/* ******************------------------------------+ */
+/* SENDER: FS, LEVEL B */
+void Dbacc::execFSREADCONF(Signal* signal)
+{
+ jamEntry();
+ fsConnectptr.i = signal->theData[0];
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ tresult = 0; /* RESULT CHECK VALUE */
+ switch (fsConnectptr.p->fsState) {
+ case WAIT_READ_PAGE_ZERO:
+ jam();
+ fragrecptr.i = fsConnectptr.p->fragrecPtr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ srReadPageZeroLab(signal);
+ return;
+ break;
+ case WAIT_READ_DATA:
+ jam();
+ fragrecptr.i = fsConnectptr.p->fragrecPtr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ storeDataPageInDirectoryLab(signal);
+ return;
+ break;
+ case READ_UNDO_PAGE:
+ jam();
+ srDoUndoLab(signal);
+ return;
+ break;
+ case READ_UNDO_PAGE_AND_CLOSE:
+ jam();
+ fsConnectptr.p->fsState = WAIT_CLOSE_UNDO;
+ /* ************************ */
+ /* FSCLOSEREQ */
+ /* ************************ */
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsConnectptr.i;
+ signal->theData[3] = 0;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+ /* FLAG = DO NOT DELETE FILE */
+ srDoUndoLab(signal);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbacc::execFSREADCONF()
+
+/* ******************--------------------------------------------------------------- */
+/* FSREADRREF OPENFILE CONF */
+/* ******************------------------------------+ */
+/* SENDER: FS, LEVEL B */
+void Dbacc::execFSREADREF(Signal* signal)
+{
+ jamEntry();
+ progError(0, __LINE__, "Read of file refused");
+ return;
+}//Dbacc::execFSREADREF()
+
+/* ******************--------------------------------------------------------------- */
+/* FSWRITECONF OPENFILE CONF */
+/* ******************------------------------------+ */
+/* SENDER: FS, LEVEL B */
+void Dbacc::execFSWRITECONF(Signal* signal)
+{
+ jamEntry();
+ fsOpptr.i = signal->theData[0];
+ ptrCheckGuard(fsOpptr, cfsOpsize, fsOprec);
+ /* FS_OPERATION PTR */
+ tresult = 0; /* RESULT CHECK VALUE */
+ fsConnectptr.i = fsOpptr.p->fsConptr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ fragrecptr.i = fsOpptr.p->fsOpfragrecPtr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ switch (fsOpptr.p->fsOpstate) {
+ case WAIT_WRITE_UNDO:
+ jam();
+ updateLastUndoPageIdWritten(signal, fsOpptr.p->fsOpMemPage);
+ releaseFsOpRec(signal);
+ if (fragrecptr.p->nrWaitWriteUndoExit == 0) {
+ jam();
+ checkSendLcpConfLab(signal);
+ return;
+ } else {
+ jam();
+ fragrecptr.p->lastUndoIsStored = ZTRUE;
+ }//if
+ return;
+ break;
+ case WAIT_WRITE_UNDO_EXIT:
+ jam();
+ updateLastUndoPageIdWritten(signal, fsOpptr.p->fsOpMemPage);
+ releaseFsOpRec(signal);
+ if (fragrecptr.p->nrWaitWriteUndoExit > 0) {
+ jam();
+ fragrecptr.p->nrWaitWriteUndoExit--;
+ }//if
+ if (fsConnectptr.p->fsState == WAIT_CLOSE_UNDO) {
+ jam();
+ /* ************************ */
+ /* FSCLOSEREQ */
+ /* ************************ */
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsConnectptr.i;
+ signal->theData[3] = ZFALSE;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+ }//if
+ if (fragrecptr.p->nrWaitWriteUndoExit == 0) {
+ if (fragrecptr.p->lastUndoIsStored == ZTRUE) {
+ jam();
+ fragrecptr.p->lastUndoIsStored = ZFALSE;
+ checkSendLcpConfLab(signal);
+ return;
+ }//if
+ }//if
+ return;
+ break;
+ case WAIT_WRITE_DATA:
+ jam();
+ releaseFsOpRec(signal);
+ fragrecptr.p->activeDataFilePage += ZWRITEPAGESIZE;
+ fragrecptr.p->activeDataPage = 0;
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ lcpConnectptr.i = rootfragrecptr.p->lcpPtr;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ switch (fragrecptr.p->fragState) {
+ case LCP_SEND_PAGES:
+ jam();
+ savepagesLab(signal);
+ return;
+ break;
+ case LCP_SEND_OVER_PAGES:
+ jam();
+ saveOverPagesLab(signal);
+ return;
+ break;
+ case LCP_SEND_ZERO_PAGE:
+ jam();
+ saveZeroPageLab(signal);
+ return;
+ break;
+ case WAIT_ZERO_PAGE_STORED:
+ jam();
+ lcpCloseDataFileLab(signal);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ break;
+ }//switch
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbacc::execFSWRITECONF()
+
+/* ******************--------------------------------------------------------------- */
+/* FSWRITEREF OPENFILE CONF */
+/* ******************------------------------------+ */
+/* SENDER: FS, LEVEL B */
+void Dbacc::execFSWRITEREF(Signal* signal)
+{
+ jamEntry();
+ progError(0, __LINE__, "Write to file refused");
+ return;
+}//Dbacc::execFSWRITEREF()
+
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* */
+/* END OF COMMON SIGNAL RECEPTION MODULE */
+/* */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* */
+/* SYSTEM RESTART MODULE */
+/* */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+void Dbacc::execNDB_STTOR(Signal* signal)
+{
+ Uint32 tstartphase;
+ Uint32 tconfig1;
+ Uint32 tconfig2;
+ Uint32 tlqhConfig1;
+ Uint32 tStartType;
+
+ jamEntry();
+ cndbcntrRef = signal->theData[0];
+ cmynodeid = signal->theData[1];
+ tstartphase = signal->theData[2];
+ tStartType = signal->theData[3];
+ tlqhConfig1 = signal->theData[10]; /* DBLQH */
+ tconfig1 = signal->theData[16]; /* DBACC */
+ tconfig2 = signal->theData[17]; /* DBACC */
+ switch (tstartphase) {
+ case ZSPH1:
+ jam();
+ ndbsttorryLab(signal);
+ return;
+ break;
+ case ZSPH2:
+ cnoLcpPages = 2 * (ZWRITEPAGESIZE + 1);
+ initialiseLcpPages(signal);
+ ndbsttorryLab(signal);
+ return;
+ break;
+ case ZSPH3:
+ if ((tStartType == NodeState::ST_NODE_RESTART) ||
+ (tStartType == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+ //---------------------------------------------
+ // csystemRestart is used to check what is needed
+ // during log execution. When starting a node it
+ // is not a log execution and rather a normal
+ // execution. Thus we reset the variable here to
+ // avoid unnecessary system crashes.
+ //---------------------------------------------
+ csystemRestart = ZFALSE;
+ }//if
+ if (tconfig1 > 0) {
+ jam();
+ clblPagesPerTick = tconfig1;
+ } else {
+ jam();
+ clblPagesPerTick = 1;
+ }//if
+ clblPageCounter = clblPagesPerTick;
+ if (tconfig2 > 0) {
+ jam();
+ clblPagesPerTickAfterSr = tconfig2;
+ } else {
+ jam();
+ clblPagesPerTickAfterSr = 1;
+ }//if
+ signal->theData[0] = ZLOAD_BAL_LCP_TIMER;
+ sendSignalWithDelay(cownBlockref, GSN_CONTINUEB, signal, 100, 1);
+ break;
+ case ZSPH6:
+ jam();
+ clblPagesPerTick = clblPagesPerTickAfterSr;
+ csystemRestart = ZFALSE;
+
+ signal->theData[0] = ZREPORT_MEMORY_USAGE;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 2000, 1);
+ break;
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }//switch
+ ndbsttorryLab(signal);
+ return;
+}//Dbacc::execNDB_STTOR()
+
+/* ******************--------------------------------------------------------------- */
+/* STTOR START / RESTART */
+/* ******************------------------------------+ */
+/* SENDER: ANY, LEVEL B */
+void Dbacc::execSTTOR(Signal* signal)
+{
+ jamEntry();
+ // tstartphase = signal->theData[1];
+ tuserblockref = signal->theData[3];
+ csignalkey = signal->theData[6];
+ sttorrysignalLab(signal);
+ return;
+}//Dbacc::execSTTOR()
+
+/* --------------------------------------------------------------------------------- */
+/* ZSPH1 */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::ndbrestart1Lab(Signal* signal)
+{
+ cmynodeid = globalData.ownId;
+ cownBlockref = numberToRef(DBACC, cmynodeid);
+ czero = 0;
+ cminusOne = czero - 1;
+ ctest = 0;
+ cundoLogActive = ZFALSE;
+ csystemRestart = ZTRUE;
+ clblPageOver = 0;
+ clblPageCounter = 0;
+ cactiveUndoFilePage = 0;
+ cprevUndoaddress = cminusOne;
+ cundoposition = 0;
+ clastUndoPageIdWritten = 0;
+ cactiveUndoFileVersion = RNIL;
+ cactiveOpenUndoFsPtr = RNIL;
+ for (Uint32 tmp = 0; tmp < ZMAX_UNDO_VERSION; tmp++) {
+ csrVersList[tmp] = RNIL;
+ }//for
+ tdata0 = 0;
+ initialiseRecordsLab(signal);
+ return;
+}//Dbacc::ndbrestart1Lab()
+
+void Dbacc::initialiseRecordsLab(Signal* signal)
+{
+ switch (tdata0) {
+ case 0:
+ jam();
+ initialiseTableRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 1:
+ jam();
+ initialiseFsConnectionRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 2:
+ jam();
+ initialiseFsOpRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 3:
+ jam();
+ initialiseLcpConnectionRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 4:
+ jam();
+ initialiseDirRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 5:
+ jam();
+ initialiseDirRangeRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 6:
+ jam();
+ initialiseFragRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 7:
+ jam();
+ initialiseOverflowRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 8:
+ jam();
+ initialiseOperationRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 9:
+ jam();
+ initialisePageRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 10:
+ jam();
+ initialiseRootfragRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 11:
+ jam();
+ initialiseScanRec(signal);
+ sendInitialiseRecords(signal);
+ break;
+ case 12:
+ jam();
+ initialiseSrVerRec(signal);
+ signal->theData[0] = cownBlockref;
+ sendSignal(CMVMI_REF, GSN_SIZEALT_ACK, signal, 1, JBB);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbacc::initialiseRecordsLab()
+
+/* --------------------------------------------------------------------------------- */
+/* SEND REAL-TIME BREAK DURING INITIALISATION OF VARIABLES DURING SYSTEM RESTART.*/
+/* --------------------------------------------------------------------------------- */
+void Dbacc::sendInitialiseRecords(Signal* signal)
+{
+ signal->theData[0] = ZINITIALISE_RECORDS;
+ signal->theData[1] = tdata0 + 1;
+ signal->theData[2] = 0;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 3, JBB);
+}//Dbacc::sendInitialiseRecords()
+
+/* *********************************<< */
+/* NDB_STTORRY */
+/* *********************************<< */
+void Dbacc::ndbsttorryLab(Signal* signal)
+{
+ signal->theData[0] = cownBlockref;
+ sendSignal(cndbcntrRef, GSN_NDB_STTORRY, signal, 1, JBB);
+ return;
+}//Dbacc::ndbsttorryLab()
+
+/* *********************************<< */
+/* SIZEALT_REP SIZE ALTERATION */
+/* *********************************<< */
+void Dbacc::execSIZEALT_REP(Signal* signal)
+{
+ Uint32 tsizealtBlockRef;
+
+ jamEntry();
+ tsizealtBlockRef = signal->theData[AccSizeAltReq::IND_BLOCK_REF];
+ cdirrangesize = signal->theData[AccSizeAltReq::IND_DIR_RANGE];
+ cdirarraysize = signal->theData[AccSizeAltReq::IND_DIR_ARRAY];
+ cfragmentsize = signal->theData[AccSizeAltReq::IND_FRAGMENT];
+ coprecsize = signal->theData[AccSizeAltReq::IND_OP_RECS];
+ coverflowrecsize = signal->theData[AccSizeAltReq::IND_OVERFLOW_RECS];
+ cpagesize = signal->theData[AccSizeAltReq::IND_PAGE8];
+ crootfragmentsize = signal->theData[AccSizeAltReq::IND_ROOT_FRAG];
+ ctablesize = signal->theData[AccSizeAltReq::IND_TABLE];
+ cscanRecSize = signal->theData[AccSizeAltReq::IND_SCAN];
+ initRecords();
+ ndbrestart1Lab(signal);
+ return;
+}//Dbacc::execSIZEALT_REP()
+
+/* *********************************<< */
+/* STTORRY */
+/* *********************************<< */
+void Dbacc::sttorrysignalLab(Signal* signal)
+{
+ signal->theData[0] = csignalkey;
+ signal->theData[1] = 3;
+ /* BLOCK CATEGORY */
+ signal->theData[2] = 2;
+ /* SIGNAL VERSION NUMBER */
+ signal->theData[3] = ZSPH1;
+ signal->theData[4] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 5, JBB);
+ /* END OF START PHASES */
+ return;
+}//Dbacc::sttorrysignalLab()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_DIR_REC */
+/* INITIALATES THE DIRECTORY RECORDS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseDirRec(Signal* signal)
+{
+ DirectoryarrayPtr idrDirptr;
+ ndbrequire(cdirarraysize > 0);
+ for (idrDirptr.i = 0; idrDirptr.i < cdirarraysize; idrDirptr.i++) {
+ ptrAss(idrDirptr, directoryarray);
+ for (Uint32 i = 0; i <= 255; i++) {
+ idrDirptr.p->pagep[i] = RNIL;
+ }//for
+ }//for
+ cdirmemory = 0;
+ cfirstfreedir = RNIL;
+}//Dbacc::initialiseDirRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_DIR_RANGE_REC */
+/* INITIALATES THE DIR_RANGE RECORDS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseDirRangeRec(Signal* signal)
+{
+ DirRangePtr idrDirRangePtr;
+
+ ndbrequire(cdirrangesize > 0);
+ for (idrDirRangePtr.i = 0; idrDirRangePtr.i < cdirrangesize; idrDirRangePtr.i++) {
+ ptrAss(idrDirRangePtr, dirRange);
+ idrDirRangePtr.p->dirArray[0] = idrDirRangePtr.i + 1;
+ for (Uint32 i = 1; i < 256; i++) {
+ idrDirRangePtr.p->dirArray[i] = RNIL;
+ }//for
+ }//for
+ idrDirRangePtr.i = cdirrangesize - 1;
+ ptrAss(idrDirRangePtr, dirRange);
+ idrDirRangePtr.p->dirArray[0] = RNIL;
+ cfirstfreeDirrange = 0;
+}//Dbacc::initialiseDirRangeRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_FRAG_REC */
+/* INITIALATES THE FRAGMENT RECORDS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseFragRec(Signal* signal)
+{
+ FragmentrecPtr regFragPtr;
+ ndbrequire(cfragmentsize > 0);
+ for (regFragPtr.i = 0; regFragPtr.i < cfragmentsize; regFragPtr.i++) {
+ jam();
+ ptrAss(regFragPtr, fragmentrec);
+ initFragGeneral(regFragPtr);
+ regFragPtr.p->nextfreefrag = regFragPtr.i + 1;
+ }//for
+ regFragPtr.i = cfragmentsize - 1;
+ ptrAss(regFragPtr, fragmentrec);
+ regFragPtr.p->nextfreefrag = RNIL;
+ cfirstfreefrag = 0;
+}//Dbacc::initialiseFragRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_FS_CONNECTION_REC */
+/* INITIALATES THE FS_CONNECTION RECORDS */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseFsConnectionRec(Signal* signal)
+{
+ ndbrequire(cfsConnectsize > 0);
+ for (fsConnectptr.i = 0; fsConnectptr.i < cfsConnectsize; fsConnectptr.i++) {
+ ptrAss(fsConnectptr, fsConnectrec);
+ fsConnectptr.p->fsNext = fsConnectptr.i + 1;
+ fsConnectptr.p->fsPrev = RNIL;
+ fsConnectptr.p->fragrecPtr = RNIL;
+ fsConnectptr.p->fsState = WAIT_NOTHING;
+ }//for
+ fsConnectptr.i = cfsConnectsize - 1;
+ ptrAss(fsConnectptr, fsConnectrec);
+ fsConnectptr.p->fsNext = RNIL; /* INITIALITES THE LAST CONNECTRECORD */
+ cfsFirstfreeconnect = 0; /* INITIATES THE FIRST FREE CONNECT RECORD */
+}//Dbacc::initialiseFsConnectionRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_FS_OP_REC */
+/* INITIALATES THE FS_OP RECORDS */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseFsOpRec(Signal* signal)
+{
+ ndbrequire(cfsOpsize > 0);
+ for (fsOpptr.i = 0; fsOpptr.i < cfsOpsize; fsOpptr.i++) {
+ ptrAss(fsOpptr, fsOprec);
+ fsOpptr.p->fsOpnext = fsOpptr.i + 1;
+ fsOpptr.p->fsOpfragrecPtr = RNIL;
+ fsOpptr.p->fsConptr = RNIL;
+ fsOpptr.p->fsOpstate = WAIT_NOTHING;
+ }//for
+ fsOpptr.i = cfsOpsize - 1;
+ ptrAss(fsOpptr, fsOprec);
+ fsOpptr.p->fsOpnext = RNIL;
+ cfsFirstfreeop = 0;
+}//Dbacc::initialiseFsOpRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_LCP_CONNECTION_REC */
+/* INITIALATES THE LCP_CONNECTION RECORDS */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseLcpConnectionRec(Signal* signal)
+{
+ ndbrequire(clcpConnectsize > 0);
+ for (lcpConnectptr.i = 0; lcpConnectptr.i < clcpConnectsize; lcpConnectptr.i++) {
+ ptrAss(lcpConnectptr, lcpConnectrec);
+ lcpConnectptr.p->nextLcpConn = lcpConnectptr.i + 1;
+ lcpConnectptr.p->lcpUserptr = RNIL;
+ lcpConnectptr.p->rootrecptr = RNIL;
+ lcpConnectptr.p->lcpstate = LCP_FREE;
+ }//for
+ lcpConnectptr.i = clcpConnectsize - 1;
+ ptrAss(lcpConnectptr, lcpConnectrec);
+ lcpConnectptr.p->nextLcpConn = RNIL;
+ cfirstfreelcpConnect = 0;
+}//Dbacc::initialiseLcpConnectionRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_OPERATION_REC */
+/* INITIALATES THE OPERATION RECORDS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseOperationRec(Signal* signal)
+{
+ ndbrequire(coprecsize > 0);
+ for (operationRecPtr.i = 0; operationRecPtr.i < coprecsize; operationRecPtr.i++) {
+ ptrAss(operationRecPtr, operationrec);
+ operationRecPtr.p->transactionstate = IDLE;
+ operationRecPtr.p->operation = ZUNDEFINED_OP;
+ operationRecPtr.p->opState = FREE_OP;
+ operationRecPtr.p->nextOp = operationRecPtr.i + 1;
+ }//for
+ operationRecPtr.i = coprecsize - 1;
+ ptrAss(operationRecPtr, operationrec);
+ operationRecPtr.p->nextOp = RNIL;
+ cfreeopRec = 0;
+}//Dbacc::initialiseOperationRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_OVERFLOW_REC */
+/* INITIALATES THE OVERFLOW RECORDS */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseOverflowRec(Signal* signal)
+{
+ OverflowRecordPtr iorOverflowRecPtr;
+
+ ndbrequire(coverflowrecsize > 0);
+ for (iorOverflowRecPtr.i = 0; iorOverflowRecPtr.i < coverflowrecsize; iorOverflowRecPtr.i++) {
+ ptrAss(iorOverflowRecPtr, overflowRecord);
+ iorOverflowRecPtr.p->nextfreeoverrec = iorOverflowRecPtr.i + 1;
+ }//for
+ iorOverflowRecPtr.i = coverflowrecsize - 1;
+ ptrAss(iorOverflowRecPtr, overflowRecord);
+ iorOverflowRecPtr.p->nextfreeoverrec = RNIL;
+ cfirstfreeoverrec = 0;
+}//Dbacc::initialiseOverflowRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_PAGE_REC */
+/* INITIALATES THE PAGE RECORDS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialisePageRec(Signal* signal)
+{
+ ndbrequire(cpagesize > 0);
+ cfreepage = 0;
+ cfirstfreepage = RNIL;
+ cnoOfAllocatedPages = 0;
+}//Dbacc::initialisePageRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_LCP_PAGES */
+/* INITIALATES THE LCP PAGE RECORDS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseLcpPages(Signal* signal)
+{
+ Uint32 tilpIndex;
+
+ ndbrequire(cnoLcpPages >= (2 * (ZWRITEPAGESIZE + 1)));
+ /* --------------------------------------------------------------------------------- */
+ /* AN ABSOLUTE MINIMUM IS THAT WE HAVE 16 LCP PAGES TO HANDLE TWO CONCURRENT */
+ /* LCP'S ON LOCAL FRAGMENTS. */
+ /* --------------------------------------------------------------------------------- */
+ ndbrequire(cpagesize >= (cnoLcpPages + 8));
+ /* --------------------------------------------------------------------------------- */
+ /* THE NUMBER OF PAGES MUST BE AT LEAST 8 PLUS THE NUMBER OF PAGES REQUIRED BY */
+ /* THE LOCAL CHECKPOINT PROCESS. THIS NUMBER IS 8 TIMES THE PARALLELISM OF */
+ /* LOCAL CHECKPOINTS. */
+ /* --------------------------------------------------------------------------------- */
+ /* --------------------------------------------------------------------------------- */
+ /* WE SET UP A LINKED LIST OF PAGES FOR EXCLUSIVE USE BY LOCAL CHECKPOINTS. */
+ /* --------------------------------------------------------------------------------- */
+ cfirstfreeLcpPage = RNIL;
+ for (tilpIndex = 0; tilpIndex < cnoLcpPages; tilpIndex++) {
+ jam();
+ seizePage(signal);
+ rlpPageptr = spPageptr;
+ releaseLcpPage(signal);
+ }//for
+}//Dbacc::initialiseLcpPages()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_ROOTFRAG_REC */
+/* INITIALATES THE ROOTFRAG RECORDS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseRootfragRec(Signal* signal)
+{
+ ndbrequire(crootfragmentsize > 0);
+ for (rootfragrecptr.i = 0; rootfragrecptr.i < crootfragmentsize; rootfragrecptr.i++) {
+ ptrAss(rootfragrecptr, rootfragmentrec);
+ rootfragrecptr.p->nextroot = rootfragrecptr.i + 1;
+ rootfragrecptr.p->fragmentptr[0] = RNIL;
+ rootfragrecptr.p->fragmentptr[1] = RNIL;
+ }//for
+ rootfragrecptr.i = crootfragmentsize - 1;
+ ptrAss(rootfragrecptr, rootfragmentrec);
+ rootfragrecptr.p->nextroot = RNIL;
+ cfirstfreerootfrag = 0;
+}//Dbacc::initialiseRootfragRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_SCAN_REC */
+/* INITIALATES THE QUE_SCAN RECORDS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseScanRec(Signal* signal)
+{
+ ndbrequire(cscanRecSize > 0);
+ for (scanPtr.i = 0; scanPtr.i < cscanRecSize; scanPtr.i++) {
+ ptrAss(scanPtr, scanRec);
+ scanPtr.p->scanNextfreerec = scanPtr.i + 1;
+ scanPtr.p->scanState = ScanRec::SCAN_DISCONNECT;
+ scanPtr.p->scanTimer = 0;
+ scanPtr.p->scanContinuebCounter = 0;
+ }//for
+ scanPtr.i = cscanRecSize - 1;
+ ptrAss(scanPtr, scanRec);
+ scanPtr.p->scanNextfreerec = RNIL;
+ cfirstFreeScanRec = 0;
+}//Dbacc::initialiseScanRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_SR_VER_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseSrVerRec(Signal* signal)
+{
+ ndbrequire(csrVersionRecSize > 0);
+ for (srVersionPtr.i = 0; srVersionPtr.i < csrVersionRecSize; srVersionPtr.i++) {
+ ptrAss(srVersionPtr, srVersionRec);
+ srVersionPtr.p->nextFreeSr = srVersionPtr.i + 1;
+ }//for
+ srVersionPtr.i = csrVersionRecSize - 1;
+ ptrAss(srVersionPtr, srVersionRec);
+ srVersionPtr.p->nextFreeSr = RNIL;
+ cfirstFreeSrVersionRec = 0;
+}//Dbacc::initialiseSrVerRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INITIALISE_TABLE_REC */
+/* INITIALATES THE TABLE RECORDS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initialiseTableRec(Signal* signal)
+{
+ ndbrequire(ctablesize > 0);
+ for (tabptr.i = 0; tabptr.i < ctablesize; tabptr.i++) {
+ ptrAss(tabptr, tabrec);
+ for (Uint32 i = 0; i < NO_OF_FRAG_PER_NODE; i++) {
+ tabptr.p->fragholder[i] = RNIL;
+ tabptr.p->fragptrholder[i] = RNIL;
+ }//for
+ }//for
+}//Dbacc::initialiseTableRec()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF SYSTEM RESTART MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* ADD/DELETE FRAGMENT MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+
+void Dbacc::initRootfragrec(Signal* signal)
+{
+ const AccFragReq * const req = (AccFragReq*)&signal->theData[0];
+ rootfragrecptr.p->mytabptr = req->tableId;
+ rootfragrecptr.p->roothashcheck = req->kValue + req->lhFragBits;
+ rootfragrecptr.p->noOfElements = 0;
+ for (Uint32 i = 0; i < MAX_PARALLEL_SCANS_PER_FRAG; i++) {
+ rootfragrecptr.p->scan[i] = RNIL;
+ }//for
+}//Dbacc::initRootfragrec()
+
+void Dbacc::execACCFRAGREQ(Signal* signal)
+{
+ const AccFragReq * const req = (AccFragReq*)&signal->theData[0];
+ jamEntry();
+ tabptr.i = req->tableId;
+ ptrCheckGuard(tabptr, ctablesize, tabrec);
+ ndbrequire((req->reqInfo & 0xF) == ZADDFRAG);
+ ndbrequire(!getrootfragmentrec(signal, rootfragrecptr, req->fragId));
+ if (cfirstfreerootfrag == RNIL) {
+ jam();
+ addFragRefuse(signal, ZFULL_ROOTFRAGRECORD_ERROR);
+ return;
+ }//if
+ seizeRootfragrec(signal);
+ if (!addfragtotab(signal, rootfragrecptr.i, req->fragId)) {
+ jam();
+ releaseRootFragRecord(signal, rootfragrecptr);
+ addFragRefuse(signal, ZFULL_ROOTFRAGRECORD_ERROR);
+ return;
+ }//if
+ initRootfragrec(signal);
+ for (Uint32 i = 0; i < 2; i++) {
+ jam();
+ if (cfirstfreefrag == RNIL) {
+ jam();
+ addFragRefuse(signal, ZFULL_FRAGRECORD_ERROR);
+ return;
+ }//if
+ seizeFragrec(signal);
+ initFragGeneral(fragrecptr);
+ initFragAdd(signal, i, rootfragrecptr.i, fragrecptr);
+ rootfragrecptr.p->fragmentptr[i] = fragrecptr.i;
+ rootfragrecptr.p->fragmentid[i] = fragrecptr.p->myfid;
+ if (cfirstfreeDirrange == RNIL) {
+ jam();
+ addFragRefuse(signal, ZDIR_RANGE_ERROR);
+ return;
+ } else {
+ jam();
+ seizeDirrange(signal);
+ }//if
+ fragrecptr.p->directory = newDirRangePtr.i;
+ seizeDirectory(signal);
+ if (tresult < ZLIMIT_OF_ERROR) {
+ jam();
+ newDirRangePtr.p->dirArray[0] = sdDirptr.i;
+ } else {
+ jam();
+ addFragRefuse(signal, tresult);
+ return;
+ }//if
+ seizePage(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ addFragRefuse(signal, tresult);
+ return;
+ }//if
+ sdDirptr.p->pagep[0] = spPageptr.i;
+ tipPageId = 0;
+ inpPageptr = spPageptr;
+ initPage(signal);
+ if (cfirstfreeDirrange == RNIL) {
+ jam();
+ addFragRefuse(signal, ZDIR_RANGE_ERROR);
+ return;
+ } else {
+ jam();
+ seizeDirrange(signal);
+ }//if
+ fragrecptr.p->overflowdir = newDirRangePtr.i;
+ seizeDirectory(signal);
+ if (tresult < ZLIMIT_OF_ERROR) {
+ jam();
+ newDirRangePtr.p->dirArray[0] = sdDirptr.i;
+ } else {
+ jam();
+ addFragRefuse(signal, tresult);
+ return;
+ }//if
+ }//for
+ Uint32 userPtr = req->userPtr;
+ BlockReference retRef = req->userRef;
+ rootfragrecptr.p->rootState = ACTIVEROOT;
+ AccFragConf * const conf = (AccFragConf*)&signal->theData[0];
+
+ conf->userPtr = userPtr;
+ conf->rootFragPtr = rootfragrecptr.i;
+ conf->fragId[0] = rootfragrecptr.p->fragmentid[0];
+ conf->fragId[1] = rootfragrecptr.p->fragmentid[1];
+ conf->fragPtr[0] = rootfragrecptr.p->fragmentptr[0];
+ conf->fragPtr[1] = rootfragrecptr.p->fragmentptr[1];
+ conf->rootHashCheck = rootfragrecptr.p->roothashcheck;
+ sendSignal(retRef, GSN_ACCFRAGCONF, signal, AccFragConf::SignalLength, JBB);
+}//Dbacc::execACCFRAGREQ()
+
+void Dbacc::addFragRefuse(Signal* signal, Uint32 errorCode)
+{
+ const AccFragReq * const req = (AccFragReq*)&signal->theData[0];
+ AccFragRef * const ref = (AccFragRef*)&signal->theData[0];
+ Uint32 userPtr = req->userPtr;
+ BlockReference retRef = req->userRef;
+
+ ref->userPtr = userPtr;
+ ref->errorCode = errorCode;
+ sendSignal(retRef, GSN_ACCFRAGREF, signal, AccFragRef::SignalLength, JBB);
+ return;
+}//Dbacc::addFragRefuseEarly()
+
+void
+Dbacc::execDROP_TAB_REQ(Signal* signal){
+ jamEntry();
+ DropTabReq* req = (DropTabReq*)signal->getDataPtr();
+
+ TabrecPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctablesize, tabrec);
+
+ tabPtr.p->tabUserRef = req->senderRef;
+ tabPtr.p->tabUserPtr = req->senderData;
+
+ signal->theData[0] = ZREL_ROOT_FRAG;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 2, JBB);
+}
+
+void Dbacc::releaseRootFragResources(Signal* signal, Uint32 tableId)
+{
+ RootfragmentrecPtr rootPtr;
+ TabrecPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctablesize, tabrec);
+ for (Uint32 i = 0; i < NO_OF_FRAG_PER_NODE; i++) {
+ jam();
+ if (tabPtr.p->fragholder[i] != RNIL) {
+ jam();
+ Uint32 fragIndex;
+ rootPtr.i = tabPtr.p->fragptrholder[i];
+ ptrCheckGuard(rootPtr, crootfragmentsize, rootfragmentrec);
+ if (rootPtr.p->fragmentptr[0] != RNIL) {
+ jam();
+ fragIndex = rootPtr.p->fragmentptr[0];
+ rootPtr.p->fragmentptr[0] = RNIL;
+ } else if (rootPtr.p->fragmentptr[1] != RNIL) {
+ jam();
+ fragIndex = rootPtr.p->fragmentptr[1];
+ rootPtr.p->fragmentptr[1] = RNIL;
+ } else {
+ jam();
+ releaseRootFragRecord(signal, rootPtr);
+ tabPtr.p->fragholder[i] = RNIL;
+ tabPtr.p->fragptrholder[i] = RNIL;
+ continue;
+ }//if
+ releaseFragResources(signal, fragIndex);
+ return;
+ }//if
+ }//for
+
+ /**
+ * Finished...
+ */
+ sendFSREMOVEREQ(signal, tableId);
+}//Dbacc::releaseRootFragResources()
+
+void Dbacc::releaseRootFragRecord(Signal* signal, RootfragmentrecPtr rootPtr)
+{
+ rootPtr.p->nextroot = cfirstfreerootfrag;
+ cfirstfreerootfrag = rootPtr.i;
+}//Dbacc::releaseRootFragRecord()
+
+void Dbacc::releaseFragResources(Signal* signal, Uint32 fragIndex)
+{
+ FragmentrecPtr regFragPtr;
+ regFragPtr.i = fragIndex;
+ ptrCheckGuard(regFragPtr, cfragmentsize, fragmentrec);
+ verifyFragCorrect(regFragPtr);
+ if (regFragPtr.p->directory != RNIL) {
+ jam();
+ releaseDirResources(signal, regFragPtr.i, regFragPtr.p->directory, 0);
+ regFragPtr.p->directory = RNIL;
+ } else if (regFragPtr.p->overflowdir != RNIL) {
+ jam();
+ releaseDirResources(signal, regFragPtr.i, regFragPtr.p->overflowdir, 0);
+ regFragPtr.p->overflowdir = RNIL;
+ } else if (regFragPtr.p->firstOverflowRec != RNIL) {
+ jam();
+ releaseOverflowResources(signal, regFragPtr);
+ } else if (regFragPtr.p->firstFreeDirindexRec != RNIL) {
+ jam();
+ releaseDirIndexResources(signal, regFragPtr);
+ } else {
+ RootfragmentrecPtr rootPtr;
+ jam();
+ rootPtr.i = regFragPtr.p->myroot;
+ ptrCheckGuard(rootPtr, crootfragmentsize, rootfragmentrec);
+ releaseFragRecord(signal, regFragPtr);
+ signal->theData[0] = ZREL_ROOT_FRAG;
+ signal->theData[1] = rootPtr.p->mytabptr;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+}//Dbacc::releaseFragResources()
+
+void Dbacc::verifyFragCorrect(FragmentrecPtr regFragPtr)
+{
+ for (Uint32 i = 0; i < ZWRITEPAGESIZE; i++) {
+ jam();
+ ndbrequire(regFragPtr.p->datapages[i] == RNIL);
+ }//for
+ ndbrequire(regFragPtr.p->lockOwnersList == RNIL);
+ ndbrequire(regFragPtr.p->firstWaitInQueOp == RNIL);
+ ndbrequire(regFragPtr.p->lastWaitInQueOp == RNIL);
+ ndbrequire(regFragPtr.p->sentWaitInQueOp == RNIL);
+ //ndbrequire(regFragPtr.p->fsConnPtr == RNIL);
+ ndbrequire(regFragPtr.p->zeroPagePtr == RNIL);
+ ndbrequire(regFragPtr.p->nrWaitWriteUndoExit == 0);
+ ndbrequire(regFragPtr.p->sentWaitInQueOp == RNIL);
+}//Dbacc::verifyFragCorrect()
+
+void Dbacc::releaseDirResources(Signal* signal,
+ Uint32 fragIndex,
+ Uint32 dirIndex,
+ Uint32 startIndex)
+{
+ DirRangePtr regDirRangePtr;
+ regDirRangePtr.i = dirIndex;
+ ptrCheckGuard(regDirRangePtr, cdirrangesize, dirRange);
+ for (Uint32 i = startIndex; i < 256; i++) {
+ jam();
+ if (regDirRangePtr.p->dirArray[i] != RNIL) {
+ jam();
+ Uint32 directoryIndex = regDirRangePtr.p->dirArray[i];
+ regDirRangePtr.p->dirArray[i] = RNIL;
+ releaseDirectoryResources(signal, fragIndex, dirIndex, (i + 1), directoryIndex);
+ return;
+ }//if
+ }//for
+ rdDirRangePtr = regDirRangePtr;
+ releaseDirrange(signal);
+ signal->theData[0] = ZREL_FRAG;
+ signal->theData[1] = fragIndex;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 2, JBB);
+}//Dbacc::releaseDirResources()
+
+void Dbacc::releaseDirectoryResources(Signal* signal,
+ Uint32 fragIndex,
+ Uint32 dirIndex,
+ Uint32 startIndex,
+ Uint32 directoryIndex)
+{
+ DirectoryarrayPtr regDirPtr;
+ regDirPtr.i = directoryIndex;
+ ptrCheckGuard(regDirPtr, cdirarraysize, directoryarray);
+ for (Uint32 i = 0; i < 256; i++) {
+ jam();
+ if (regDirPtr.p->pagep[i] != RNIL) {
+ jam();
+ rpPageptr.i = regDirPtr.p->pagep[i];
+ ptrCheckGuard(rpPageptr, cpagesize, page8);
+ releasePage(signal);
+ regDirPtr.p->pagep[i] = RNIL;
+ }//if
+ }//for
+ rdDirptr = regDirPtr;
+ releaseDirectory(signal);
+ signal->theData[0] = ZREL_DIR;
+ signal->theData[1] = fragIndex;
+ signal->theData[2] = dirIndex;
+ signal->theData[3] = startIndex;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 4, JBB);
+}//Dbacc::releaseDirectoryResources()
+
+void Dbacc::releaseOverflowResources(Signal* signal, FragmentrecPtr regFragPtr)
+{
+ Uint32 loopCount = 0;
+ OverflowRecordPtr regOverflowRecPtr;
+ while ((regFragPtr.p->firstOverflowRec != RNIL) &&
+ (loopCount < 1)) {
+ jam();
+ regOverflowRecPtr.i = regFragPtr.p->firstOverflowRec;
+ ptrCheckGuard(regOverflowRecPtr, coverflowrecsize, overflowRecord);
+ regFragPtr.p->firstOverflowRec = regOverflowRecPtr.p->nextOverRec;
+ rorOverflowRecPtr = regOverflowRecPtr;
+ releaseOverflowRec(signal);
+ loopCount++;
+ }//while
+ signal->theData[0] = ZREL_FRAG;
+ signal->theData[1] = regFragPtr.i;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 2, JBB);
+}//Dbacc::releaseOverflowResources()
+
+void Dbacc::releaseDirIndexResources(Signal* signal, FragmentrecPtr regFragPtr)
+{
+ Uint32 loopCount = 0;
+ OverflowRecordPtr regOverflowRecPtr;
+ while ((regFragPtr.p->firstFreeDirindexRec != RNIL) &&
+ (loopCount < 1)) {
+ jam();
+ regOverflowRecPtr.i = regFragPtr.p->firstFreeDirindexRec;
+ ptrCheckGuard(regOverflowRecPtr, coverflowrecsize, overflowRecord);
+ regFragPtr.p->firstFreeDirindexRec = regOverflowRecPtr.p->nextOverList;
+ rorOverflowRecPtr = regOverflowRecPtr;
+ releaseOverflowRec(signal);
+ loopCount++;
+ }//while
+ signal->theData[0] = ZREL_FRAG;
+ signal->theData[1] = regFragPtr.i;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 2, JBB);
+}//Dbacc::releaseDirIndexResources()
+
+void Dbacc::releaseFragRecord(Signal* signal, FragmentrecPtr regFragPtr)
+{
+ regFragPtr.p->nextfreefrag = cfirstfreefrag;
+ cfirstfreefrag = regFragPtr.i;
+ initFragGeneral(regFragPtr);
+}//Dbacc::releaseFragRecord()
+
+void Dbacc::sendFSREMOVEREQ(Signal* signal, Uint32 tableId)
+{
+ FsRemoveReq * const fsReq = (FsRemoveReq *)signal->getDataPtrSend();
+ fsReq->userReference = cownBlockref;
+ fsReq->userPointer = tableId;
+ fsReq->fileNumber[0] = tableId;
+ fsReq->fileNumber[1] = (Uint32)-1; // Remove all fragments
+ fsReq->fileNumber[2] = (Uint32)-1; // Remove all data files within fragment
+ fsReq->fileNumber[3] = 255 | // No P-value used here
+ (3 << 8) | // Data-files in D3
+ (0 << 16) | // Data-files
+ (1 << 24); // Version 1 of fileNumber
+ fsReq->directory = 1;
+ fsReq->ownDirectory = 1;
+ sendSignal(NDBFS_REF, GSN_FSREMOVEREQ, signal, FsRemoveReq::SignalLength, JBA);
+}//Dbacc::sendFSREMOVEREQ()
+
+void Dbacc::execFSREMOVECONF(Signal* signal)
+{
+ FsConf * const fsConf = (FsConf *)signal->getDataPtrSend();
+ TabrecPtr tabPtr;
+ tabPtr.i = fsConf->userPointer;
+ ptrCheckGuard(tabPtr, ctablesize, tabrec);
+
+ DropTabConf * const dropConf = (DropTabConf *)signal->getDataPtrSend();
+ dropConf->senderRef = reference();
+ dropConf->senderData = tabPtr.p->tabUserPtr;
+ dropConf->tableId = tabPtr.i;
+ sendSignal(tabPtr.p->tabUserRef, GSN_DROP_TAB_CONF,
+ signal, DropTabConf::SignalLength, JBB);
+
+ tabPtr.p->tabUserPtr = RNIL;
+ tabPtr.p->tabUserRef = 0;
+}//Dbacc::execFSREMOVECONF()
+
+void Dbacc::execFSREMOVEREF(Signal* signal)
+{
+ ndbrequire(false);
+}//Dbacc::execFSREMOVEREF()
+
+/* -------------------------------------------------------------------------- */
+/* ADDFRAGTOTAB */
+/* DESCRIPTION: PUTS A FRAGMENT ID AND A POINTER TO ITS RECORD INTO */
+/* TABLE ARRRAY OF THE TABLE RECORD. */
+/* -------------------------------------------------------------------------- */
+bool Dbacc::addfragtotab(Signal* signal, Uint32 rootIndex, Uint32 fid)
+{
+ for (Uint32 i = 0; i < NO_OF_FRAG_PER_NODE; i++) {
+ jam();
+ if (tabptr.p->fragholder[i] == RNIL) {
+ jam();
+ tabptr.p->fragholder[i] = fid;
+ tabptr.p->fragptrholder[i] = rootIndex;
+ return true;
+ }//if
+ }//for
+ return false;
+}//Dbacc::addfragtotab()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF ADD/DELETE FRAGMENT MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* CONNECTION MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* ACCSEIZEREQ SEIZE REQ */
+/* SENDER: LQH, LEVEL B */
+/* ENTER ACCSEIZEREQ WITH */
+/* TUSERPTR , CONECTION PTR OF LQH */
+/* TUSERBLOCKREF BLOCK REFERENCE OF LQH */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* ACCSEIZEREQ SEIZE REQ */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execACCSEIZEREQ(Signal* signal)
+{
+ jamEntry();
+ tuserptr = signal->theData[0];
+ /* CONECTION PTR OF LQH */
+ tuserblockref = signal->theData[1];
+ /* BLOCK REFERENCE OF LQH */
+ tresult = 0;
+ if (cfreeopRec == RNIL) {
+ jam();
+ refaccConnectLab(signal);
+ return;
+ }//if
+ seizeOpRec(signal);
+ ptrGuard(operationRecPtr);
+ operationRecPtr.p->userptr = tuserptr;
+ operationRecPtr.p->userblockref = tuserblockref;
+ operationRecPtr.p->operation = ZUNDEFINED_OP;
+ operationRecPtr.p->transactionstate = IDLE;
+ /* ******************************< */
+ /* ACCSEIZECONF */
+ /* ******************************< */
+ signal->theData[0] = tuserptr;
+ signal->theData[1] = operationRecPtr.i;
+ sendSignal(tuserblockref, GSN_ACCSEIZECONF, signal, 2, JBB);
+ return;
+}//Dbacc::execACCSEIZEREQ()
+
+void Dbacc::refaccConnectLab(Signal* signal)
+{
+ tresult = ZCONNECT_SIZE_ERROR;
+ /* ******************************< */
+ /* ACCSEIZEREF */
+ /* ******************************< */
+ signal->theData[0] = tuserptr;
+ signal->theData[1] = tresult;
+ sendSignal(tuserblockref, GSN_ACCSEIZEREF, signal, 2, JBB);
+ return;
+}//Dbacc::refaccConnectLab()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF CONNECTION MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* EXECUTE OPERATION MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* INIT_OP_REC */
+/* INFORMATION WHICH IS RECIEVED BY ACCKEYREQ WILL BE SAVED */
+/* IN THE OPERATION RECORD. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initOpRec(Signal* signal)
+{
+ register Uint32 Treqinfo;
+
+ Treqinfo = signal->theData[2];
+
+ operationRecPtr.p->hashValue = signal->theData[3];
+ operationRecPtr.p->tupkeylen = signal->theData[4];
+ operationRecPtr.p->transId1 = signal->theData[5];
+ operationRecPtr.p->transId2 = signal->theData[6];
+ operationRecPtr.p->transactionstate = ACTIVE;
+ operationRecPtr.p->commitDeleteCheckFlag = ZFALSE;
+ operationRecPtr.p->operation = Treqinfo & 0x7;
+ /* --------------------------------------------------------------------------------- */
+ // opSimple is not used in this version. Is needed for deadlock handling later on.
+ /* --------------------------------------------------------------------------------- */
+ // operationRecPtr.p->opSimple = (Treqinfo >> 3) & 0x1;
+
+ operationRecPtr.p->lockMode = (Treqinfo >> 4) & 0x3;
+
+ Uint32 readFlag = (((Treqinfo >> 4) & 0x3) == 0); // Only 1 if Read
+ Uint32 dirtyFlag = (((Treqinfo >> 6) & 0x1) == 1); // Only 1 if Dirty
+ Uint32 dirtyReadFlag = readFlag & dirtyFlag;
+ operationRecPtr.p->dirtyRead = dirtyReadFlag;
+
+ operationRecPtr.p->nodeType = (Treqinfo >> 7) & 0x3;
+ operationRecPtr.p->fid = fragrecptr.p->myfid;
+ operationRecPtr.p->fragptr = fragrecptr.i;
+ operationRecPtr.p->nextParallelQue = RNIL;
+ operationRecPtr.p->prevParallelQue = RNIL;
+ operationRecPtr.p->prevQueOp = RNIL;
+ operationRecPtr.p->nextQueOp = RNIL;
+ operationRecPtr.p->nextSerialQue = RNIL;
+ operationRecPtr.p->prevSerialQue = RNIL;
+ operationRecPtr.p->elementPage = RNIL;
+ operationRecPtr.p->keyinfoPage = RNIL;
+ operationRecPtr.p->lockOwner = ZFALSE;
+ operationRecPtr.p->insertIsDone = ZFALSE;
+ operationRecPtr.p->elementIsDisappeared = ZFALSE;
+ operationRecPtr.p->insertDeleteLen = fragrecptr.p->elementLength;
+ operationRecPtr.p->longPagePtr = RNIL;
+ operationRecPtr.p->longKeyPageIndex = RNIL;
+ operationRecPtr.p->scanRecPtr = RNIL;
+
+ // bit to mark lock operation
+ operationRecPtr.p->isAccLockReq = (Treqinfo >> 31) & 0x1;
+}//Dbacc::initOpRec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEND_ACCKEYCONF */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::sendAcckeyconf(Signal* signal)
+{
+ signal->theData[0] = operationRecPtr.p->userptr;
+ signal->theData[1] = operationRecPtr.p->insertIsDone;
+ signal->theData[2] = operationRecPtr.p->fid;
+ signal->theData[3] = operationRecPtr.p->localdata[0];
+ signal->theData[4] = operationRecPtr.p->localdata[1];
+ signal->theData[5] = fragrecptr.p->localkeylen;
+}//Dbacc::sendAcckeyconf()
+
+
+void Dbacc::ACCKEY_error(Uint32 fromWhere)
+{
+ switch(fromWhere) {
+ case 0:
+ ndbrequire(false);
+ case 1:
+ ndbrequire(false);
+ case 2:
+ ndbrequire(false);
+ case 3:
+ ndbrequire(false);
+ case 4:
+ ndbrequire(false);
+ case 5:
+ ndbrequire(false);
+ case 6:
+ ndbrequire(false);
+ case 7:
+ ndbrequire(false);
+ case 8:
+ ndbrequire(false);
+ case 9:
+ ndbrequire(false);
+ default:
+ ndbrequire(false);
+ }//switch
+}//Dbacc::ACCKEY_error()
+
+/* ******************--------------------------------------------------------------- */
+/* ACCKEYREQ REQUEST FOR INSERT, DELETE, */
+/* RERAD AND UPDATE, A TUPLE. */
+/* SENDER: LQH, LEVEL B */
+/* SIGNAL DATA: OPERATION_REC_PTR, CONNECTION PTR */
+/* TABPTR, TABLE ID = TABLE RECORD POINTER */
+/* TREQINFO, */
+/* THASHVALUE, HASH VALUE OF THE TUP */
+/* TKEYLEN, LENGTH OF THE PRIMARY KEYS */
+/* TKEY1, PRIMARY KEY 1 */
+/* TKEY2, PRIMARY KEY 2 */
+/* TKEY3, PRIMARY KEY 3 */
+/* TKEY4, PRIMARY KEY 4 */
+/* ******************--------------------------------------------------------------- */
+void Dbacc::execACCKEYREQ(Signal* signal)
+{
+ jamEntry();
+ operationRecPtr.i = signal->theData[0]; /* CONNECTION PTR */
+ fragrecptr.i = signal->theData[1]; /* FRAGMENT RECORD POINTER */
+ if (!((operationRecPtr.i < coprecsize) ||
+ (fragrecptr.i < cfragmentsize))) {
+ ACCKEY_error(0);
+ return;
+ }//if
+ ptrAss(operationRecPtr, operationrec);
+ ptrAss(fragrecptr, fragmentrec);
+ ndbrequire(operationRecPtr.p->transactionstate == IDLE);
+
+ initOpRec(signal);
+ /*---------------------------------------------------------------*/
+ /* */
+ /* WE WILL USE THE HASH VALUE TO LOOK UP THE PROPER MEMORY */
+ /* PAGE AND MEMORY PAGE INDEX TO START THE SEARCH WITHIN. */
+ /* WE REMEMBER THESE ADDRESS IF WE LATER NEED TO INSERT */
+ /* THE ITEM AFTER NOT FINDING THE ITEM. */
+ /*---------------------------------------------------------------*/
+ getElement(signal);
+
+ if (tgeResult == ZTRUE) {
+ switch (operationRecPtr.p->operation) {
+ case ZREAD:
+ case ZUPDATE:
+ case ZDELETE:
+ case ZWRITE:
+ case ZSCAN_OP:
+ if (!tgeLocked){
+ sendAcckeyconf(signal);
+ if (operationRecPtr.p->dirtyRead == ZFALSE) {
+ /*---------------------------------------------------------------*/
+ // It is not a dirty read. We proceed by locking and continue with
+ // the operation.
+ /*---------------------------------------------------------------*/
+ Uint32 eh = gePageptr.p->word32[tgeElementptr];
+ operationRecPtr.p->scanBits = ElementHeader::getScanBits(eh);
+ operationRecPtr.p->hashvaluePart = ElementHeader::getHashValuePart(eh);
+ operationRecPtr.p->elementPage = gePageptr.i;
+ operationRecPtr.p->elementContainer = tgeContainerptr;
+ operationRecPtr.p->elementPointer = tgeElementptr;
+ operationRecPtr.p->elementIsforward = tgeForward;
+
+ eh = ElementHeader::setLocked(operationRecPtr.i);
+ dbgWord32(gePageptr, tgeElementptr, eh);
+ gePageptr.p->word32[tgeElementptr] = eh;
+
+ insertLockOwnersList(signal , operationRecPtr);
+ return;
+ } else {
+ jam();
+ /*---------------------------------------------------------------*/
+ // It is a dirty read. We do not lock anything. Set state to
+ // IDLE since no COMMIT call will come.
+ /*---------------------------------------------------------------*/
+ operationRecPtr.p->transactionstate = IDLE;
+ operationRecPtr.p->operation = ZUNDEFINED_OP;
+ return;
+ }//if
+ } else {
+ jam();
+ accIsLockedLab(signal);
+ return;
+ }//if
+ break;
+ case ZINSERT:
+ jam();
+ insertExistElemLab(signal);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ } else if (tgeResult == ZFALSE) {
+ switch (operationRecPtr.p->operation) {
+ case ZINSERT:
+ case ZWRITE:
+ jam();
+ // If a write operation makes an insert we switch operation to ZINSERT so
+ // that the commit-method knows an insert has been made and updates noOfElements.
+ operationRecPtr.p->operation = ZINSERT;
+ operationRecPtr.p->insertIsDone = ZTRUE;
+ insertelementLab(signal);
+ return;
+ break;
+ case ZREAD:
+ case ZUPDATE:
+ case ZDELETE:
+ case ZSCAN_OP:
+ jam();
+ acckeyref1Lab(signal, ZREAD_ERROR);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ } else {
+ jam();
+ acckeyref1Lab(signal, tgeResult);
+ return;
+ }//if
+ return;
+}//Dbacc::execACCKEYREQ()
+
+void Dbacc::accIsLockedLab(Signal* signal)
+{
+ ndbrequire(csystemRestart == ZFALSE);
+ queOperPtr.i = ElementHeader::getOpPtrI(gePageptr.p->word32[tgeElementptr]);
+ ptrCheckGuard(queOperPtr, coprecsize, operationrec);
+ if (operationRecPtr.p->dirtyRead == ZFALSE) {
+ Uint32 return_result;
+ if (operationRecPtr.p->lockMode == ZREADLOCK) {
+ jam();
+ priPageptr = gePageptr;
+ tpriElementptr = tgeElementptr;
+ return_result = placeReadInLockQueue(signal);
+ } else {
+ jam();
+ pwiPageptr = gePageptr;
+ tpwiElementptr = tgeElementptr;
+ return_result = placeWriteInLockQueue(signal);
+ }//if
+ if (return_result == ZPARALLEL_QUEUE) {
+ jam();
+ sendAcckeyconf(signal);
+ return;
+ } else if (return_result == ZSERIAL_QUEUE) {
+ jam();
+ signal->theData[0] = RNIL;
+ return;
+ } else if (return_result == ZWRITE_ERROR) {
+ jam();
+ acckeyref1Lab(signal, return_result);
+ return;
+ }//if
+ ndbrequire(false);
+ } else {
+ if (queOperPtr.p->elementIsDisappeared == ZFALSE) {
+ jam();
+ /*---------------------------------------------------------------*/
+ // It is a dirty read. We do not lock anything. Set state to
+ // IDLE since no COMMIT call will arrive.
+ /*---------------------------------------------------------------*/
+ sendAcckeyconf(signal);
+ operationRecPtr.p->transactionstate = IDLE;
+ operationRecPtr.p->operation = ZUNDEFINED_OP;
+ return;
+ } else {
+ jam();
+ /*---------------------------------------------------------------*/
+ // The tuple does not exist in the committed world currently.
+ // Report read error.
+ /*---------------------------------------------------------------*/
+ acckeyref1Lab(signal, ZREAD_ERROR);
+ return;
+ }//if
+ }//if
+}//Dbacc::accIsLockedLab()
+
+/* --------------------------------------------------------------------------------- */
+/* I N S E R T E X I S T E L E M E N T */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::insertExistElemLab(Signal* signal)
+{
+ if (!tgeLocked){
+ jam();
+ acckeyref1Lab(signal, ZWRITE_ERROR);/* THE ELEMENT ALREADY EXIST */
+ return;
+ }//if
+ accIsLockedLab(signal);
+}//Dbacc::insertExistElemLab()
+
+/* --------------------------------------------------------------------------------- */
+/* INSERTELEMENT */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::insertelementLab(Signal* signal)
+{
+ Uint32 tinsKeyLen;
+
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ if (remainingUndoPages() < ZMIN_UNDO_PAGES_AT_OPERATION) {
+ jam();
+ acckeyref1Lab(signal, ZTEMPORARY_ACC_UNDO_FAILURE);
+ return;
+ }//if
+ }//if
+ if (fragrecptr.p->firstOverflowRec == RNIL) {
+ jam();
+ allocOverflowPage(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ acckeyref1Lab(signal, tresult);
+ return;
+ }//if
+ }//if
+ if (fragrecptr.p->keyLength != operationRecPtr.p->tupkeylen) {
+ ndbrequire(fragrecptr.p->keyLength == 0);
+ }//if
+ if (fragrecptr.p->keyLength != 0) {
+ ndbrequire(operationRecPtr.p->tupkeylen <= 8);
+ for (Uint32 i = 0; i < operationRecPtr.p->tupkeylen; i++) {
+ jam();
+ ckeys[i] = signal->theData[i + 7];
+ }//for
+ tinsKeyLen = operationRecPtr.p->tupkeylen;
+ } else {
+ jam();
+ seizePage(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ acckeyref1Lab(signal, tresult);
+ return;
+ }//if
+ operationRecPtr.p->keyinfoPage = spPageptr.i;
+ for (Uint32 i = 0; i < signal->theData[4]; i++) {
+ spPageptr.p->word32[i] = signal->theData[i + 7];
+ }//for
+
+ getLongKeyPage(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ acckeyref1Lab(signal, tresult);
+ return;
+ }//if
+ slkPageptr = glkPageptr;
+ slkCopyPageptr.i = operationRecPtr.p->keyinfoPage;
+ ptrCheckGuard(slkCopyPageptr, cpagesize, page8);
+ tslkKeyLen = operationRecPtr.p->tupkeylen;
+ storeLongKeys(signal);
+ ckeys[0] = (slkPageptr.p->word32[ZPOS_PAGE_ID] << 10) + tslkPageIndex;
+ tinsKeyLen = ZACTIVE_LONG_KEY_LEN;
+ rpPageptr.i = operationRecPtr.p->keyinfoPage;
+ ptrCheckGuard(rpPageptr, cpagesize, page8);
+ releasePage(signal);
+ operationRecPtr.p->keyinfoPage = RNIL;
+ }//if
+
+ signal->theData[0] = operationRecPtr.p->userptr;
+ Uint32 blockNo = refToBlock(operationRecPtr.p->userblockref);
+ EXECUTE_DIRECT(blockNo, GSN_LQH_ALLOCREQ, signal, 1);
+ jamEntry();
+ if (signal->theData[0] != 0) {
+ jam();
+ Uint32 result_code = signal->theData[0];
+ acckeyref1Lab(signal, result_code);
+ return;
+ }//if
+ Uint32 localKey = (signal->theData[1] << MAX_TUPLES_BITS) + signal->theData[2];
+
+ insertLockOwnersList(signal, operationRecPtr);
+
+ const Uint32 tmp = fragrecptr.p->k + fragrecptr.p->lhfragbits;
+ operationRecPtr.p->hashvaluePart =
+ (operationRecPtr.p->hashValue >> tmp) & 0xFFFF;
+ operationRecPtr.p->scanBits = 0; /* NOT ANY ACTIVE SCAN */
+ tidrElemhead = ElementHeader::setLocked(operationRecPtr.i);
+ idrPageptr = gdiPageptr;
+ tidrPageindex = tgdiPageindex;
+ tidrForward = ZTRUE;
+ tidrKeyLen = tinsKeyLen;
+ idrOperationRecPtr = operationRecPtr;
+ clocalkey[0] = localKey;
+ operationRecPtr.p->localdata[0] = localKey;
+ /* --------------------------------------------------------------------------------- */
+ /* WE SET THE LOCAL KEY TO MINUS ONE TO INDICATE IT IS NOT YET VALID. */
+ /* --------------------------------------------------------------------------------- */
+ insertElement(signal);
+ sendAcckeyconf(signal);
+ return;
+}//Dbacc::insertelementLab()
+
+/* --------------------------------------------------------------------------------- */
+/* PLACE_READ_IN_LOCK_QUEUE */
+/* INPUT: OPERATION_REC_PTR OUR OPERATION POINTER */
+/* QUE_OPER_PTR LOCK QUEUE OWNER OPERATION POINTER */
+/* PRI_PAGEPTR PAGE POINTER OF ELEMENT */
+/* TPRI_ELEMENTPTR ELEMENT POINTER OF ELEMENT */
+/* OUTPUT TRESULT = */
+/* ZPARALLEL_QUEUE OPERATION PLACED IN PARALLEL QUEUE */
+/* OPERATION CAN PROCEED NOW. */
+/* ZSERIAL_QUEUE OPERATION PLACED IN SERIAL QUEUE */
+/* ERROR CODE OPERATION NEEDS ABORTING */
+/* THE ELEMENT WAS LOCKED AND WE WANT TO READ THE TUPLE. WE WILL CHECK THE LOCK */
+/* QUEUES TO PERFORM THE PROPER ACTION. */
+/* */
+/* IN SOME PLACES IN THE CODE BELOW THAT HANDLES WHAT TO DO WHEN THE TUPLE IS LOCKED */
+/* WE DO ASSUME THAT NEXT_PARALLEL_QUEUE AND NEXT_SERIAL_QUEUE ON OPERATION_REC_PTR */
+/* HAVE BEEN INITIALISED TO RNIL. THUS WE DO NOT PERFORM THIS ONCE MORE EVEN IF IT */
+/* COULD BE NICE FOR READABILITY. */
+/* --------------------------------------------------------------------------------- */
+Uint32 Dbacc::placeReadInLockQueue(Signal* signal)
+{
+ tgnptMainOpPtr = queOperPtr;
+ getNoParallelTransaction(signal);
+ if (tgnptNrTransaction == 1) {
+ if ((queOperPtr.p->transId1 == operationRecPtr.p->transId1) &&
+ (queOperPtr.p->transId2 == operationRecPtr.p->transId2)) {
+ /* --------------------------------------------------------------------------------- */
+ /* WE ARE PERFORMING A READ OPERATION AND THIS TRANSACTION ALREADY OWNS THE LOCK */
+ /* ALONE. PUT THE OPERATION LAST IN THE PARALLEL QUEUE. */
+ /* --------------------------------------------------------------------------------- */
+ jam();
+ mlpqOperPtr = queOperPtr;
+ moveLastParallelQueue(signal);
+ operationRecPtr.p->localdata[0] = queOperPtr.p->localdata[0];
+ operationRecPtr.p->localdata[1] = queOperPtr.p->localdata[1];
+ operationRecPtr.p->prevParallelQue = mlpqOperPtr.i;
+ mlpqOperPtr.p->nextParallelQue = operationRecPtr.i;
+ switch (queOperPtr.p->lockMode) {
+ case ZREADLOCK:
+ jam();
+ /*empty*/;
+ break;
+ default:
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* IF THE TRANSACTION PREVIOUSLY SET A WRITE LOCK WE MUST ENSURE THAT ALL */
+ /* OPERATIONS IN THE PARALLEL QUEUE HAVE WRITE LOCK MODE TO AVOID STRANGE BUGS.*/
+ /* --------------------------------------------------------------------------------- */
+ operationRecPtr.p->lockMode = queOperPtr.p->lockMode;
+ break;
+ }//switch
+ return ZPARALLEL_QUEUE;
+ }//if
+ }//if
+ if (queOperPtr.p->nextSerialQue == RNIL) {
+ /* --------------------------------------------------------------------------------- */
+ /* WE ARE PERFORMING A READ OPERATION AND THERE IS NO SERIAL QUEUE. IF THERE IS NO */
+ /* WRITE OPERATION THAT OWNS THE LOCK OR ANY WRITE OPERATION IN THE PARALLEL QUEUE */
+ /* IT IS ENOUGH TO CHECK THE LOCK MODE OF THE LEADER IN THE PARALLEL QUEUE. IF IT IS */
+ /* A READ LOCK THEN WE PLACE OURSELVES IN THE PARALLEL QUEUE OTHERWISE WE GO ON TO */
+ /* PLACE OURSELVES IN THE SERIAL QUEUE. */
+ /* --------------------------------------------------------------------------------- */
+ switch (queOperPtr.p->lockMode) {
+ case ZREADLOCK:
+ jam();
+ mlpqOperPtr = queOperPtr;
+ moveLastParallelQueue(signal);
+ operationRecPtr.p->prevParallelQue = mlpqOperPtr.i;
+ mlpqOperPtr.p->nextParallelQue = operationRecPtr.i;
+ operationRecPtr.p->localdata[0] = queOperPtr.p->localdata[0];
+ operationRecPtr.p->localdata[1] = queOperPtr.p->localdata[1];
+ return ZPARALLEL_QUEUE;
+ default:
+ jam();
+ queOperPtr.p->nextSerialQue = operationRecPtr.i;
+ operationRecPtr.p->prevSerialQue = queOperPtr.i;
+ putOpInFragWaitQue(signal);
+ break;
+ }//switch
+ } else {
+ jam();
+ placeSerialQueueRead(signal);
+ }//if
+ return ZSERIAL_QUEUE;
+}//Dbacc::placeReadInLockQueue()
+
+/* --------------------------------------------------------------------------------- */
+/* WE WILL CHECK IF THIS TRANSACTION IS ALREADY PLACED AT SOME SPOT IN THE PARALLEL */
+/* SERIAL QUEUE WITHOUT ANY NEIGHBORS FROM OTHER TRANSACTION. IF SO WE WILL INSERT */
+/* IT IN THAT PARALLEL QUEUE. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::placeSerialQueueRead(Signal* signal)
+{
+ readWriteOpPtr.i = queOperPtr.p->nextSerialQue;
+ ptrCheckGuard(readWriteOpPtr, coprecsize, operationrec);
+ PSQR_LOOP:
+ jam();
+ if (readWriteOpPtr.p->nextSerialQue == RNIL) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THERE WAS NO PREVIOUS OPERATION IN THIS TRANSACTION WHICH WE COULD PUT IT */
+ /* IN THE PARALLEL QUEUE TOGETHER WITH. */
+ /* --------------------------------------------------------------------------------- */
+ checkOnlyReadEntry(signal);
+ return;
+ }//if
+ tgnptMainOpPtr = readWriteOpPtr;
+ getNoParallelTransaction(signal);
+ if (tgnptNrTransaction == 1) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THERE WAS ONLY ONE TRANSACTION INVOLVED IN THE PARALLEL QUEUE. IF THIS IS OUR */
+ /* TRANSACTION WE CAN STILL GET HOLD OF THE LOCK. */
+ /* --------------------------------------------------------------------------------- */
+ if ((readWriteOpPtr.p->transId1 == operationRecPtr.p->transId1) &&
+ (readWriteOpPtr.p->transId2 == operationRecPtr.p->transId2)) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE ARE PERFORMING A READ IN THE SAME TRANSACTION WHERE WE ALREADY */
+ /* PREVIOUSLY HAVE EXECUTED AN OPERATION. INSERT-DELETE, READ-UPDATE, READ-READ, */
+ /* UPDATE-UPDATE, UPDATE-DELETE, READ-DELETE, INSERT-READ, INSERT-UPDATE ARE ALLOWED */
+ /* COMBINATIONS. A NEW INSERT AFTER A DELETE IS NOT ALLOWED AND SUCH AN INSERT WILL */
+ /* GO TO THE SERIAL LOCK QUEUE WHICH IT WILL NOT LEAVE UNTIL A TIME-OUT AND THE */
+ /* TRANSACTION IS ABORTED. READS AND UPDATES AFTER DELETES IS ALSO NOT ALLOWED. */
+ /* --------------------------------------------------------------------------------- */
+ mlpqOperPtr = readWriteOpPtr;
+ moveLastParallelQueue(signal);
+ readWriteOpPtr = mlpqOperPtr;
+ operationRecPtr.p->prevParallelQue = readWriteOpPtr.i;
+ readWriteOpPtr.p->nextParallelQue = operationRecPtr.i;
+ operationRecPtr.p->localdata[0] = readWriteOpPtr.p->localdata[0];
+ operationRecPtr.p->localdata[1] = readWriteOpPtr.p->localdata[1];
+ switch (readWriteOpPtr.p->lockMode) {
+ case ZREADLOCK:
+ jam();
+ /*empty*/;
+ break;
+ default:
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* IF THE TRANSACTION PREVIOUSLY SET A WRITE LOCK WE MUST ENSURE THAT ALL */
+ /* OPERATIONS IN THE PARALLEL QUEUE HAVE WRITE LOCK MODE TO AVOID STRANGE BUGS.*/
+ /* --------------------------------------------------------------------------------- */
+ operationRecPtr.p->lockMode = readWriteOpPtr.p->lockMode;
+ break;
+ }//switch
+ putOpInFragWaitQue(signal);
+ return;
+ }//if
+ }//if
+ readWriteOpPtr.i = readWriteOpPtr.p->nextSerialQue;
+ ptrCheckGuard(readWriteOpPtr, coprecsize, operationrec);
+ goto PSQR_LOOP;
+}//Dbacc::placeSerialQueueRead()
+
+/* --------------------------------------------------------------------------------- */
+/* WE WILL CHECK IF THE LAST ENTRY IN THE SERIAL QUEUE CONTAINS ONLY READ */
+/* OPERATIONS. IF SO WE WILL INSERT IT IN THAT PARALLEL QUEUE. OTHERWISE WE */
+/* WILL PLACE IT AT THE END OF THE SERIAL QUEUE. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::checkOnlyReadEntry(Signal* signal)
+{
+ switch (readWriteOpPtr.p->lockMode) {
+ case ZREADLOCK:
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* SINCE THIS LAST QUEUE ONLY CONTAINS READ LOCKS WE CAN JOIN THE PARALLEL QUEUE AT */
+ /* THE END. */
+ /* --------------------------------------------------------------------------------- */
+ mlpqOperPtr = readWriteOpPtr;
+ moveLastParallelQueue(signal);
+ readWriteOpPtr = mlpqOperPtr;
+ operationRecPtr.p->prevParallelQue = readWriteOpPtr.i;
+ readWriteOpPtr.p->nextParallelQue = operationRecPtr.i;
+ operationRecPtr.p->localdata[0] = readWriteOpPtr.p->localdata[0];
+ operationRecPtr.p->localdata[1] = readWriteOpPtr.p->localdata[1];
+ break;
+ default:
+ jam(); /* PUT THE OPERATION RECORD IN THE SERIAL QUEUE */
+ readWriteOpPtr.p->nextSerialQue = operationRecPtr.i;
+ operationRecPtr.p->prevSerialQue = readWriteOpPtr.i;
+ break;
+ }//switch
+ putOpInFragWaitQue(signal);
+}//Dbacc::checkOnlyReadEntry()
+
+/* --------------------------------------------------------------------------------- */
+/* GET_NO_PARALLEL_TRANSACTION */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::getNoParallelTransaction(Signal* signal)
+{
+ OperationrecPtr tnptOpPtr;
+
+ tgnptNrTransaction = 1;
+ tnptOpPtr.i = tgnptMainOpPtr.p->nextParallelQue;
+ while ((tnptOpPtr.i != RNIL) &&
+ (tgnptNrTransaction == 1)) {
+ jam();
+ ptrCheckGuard(tnptOpPtr, coprecsize, operationrec);
+ if ((tnptOpPtr.p->transId1 == tgnptMainOpPtr.p->transId1) &&
+ (tnptOpPtr.p->transId2 == tgnptMainOpPtr.p->transId2)) {
+ tnptOpPtr.i = tnptOpPtr.p->nextParallelQue;
+ } else {
+ jam();
+ tgnptNrTransaction++;
+ }//if
+ }//while
+}//Dbacc::getNoParallelTransaction()
+
+void Dbacc::moveLastParallelQueue(Signal* signal)
+{
+ while (mlpqOperPtr.p->nextParallelQue != RNIL) {
+ jam();
+ mlpqOperPtr.i = mlpqOperPtr.p->nextParallelQue;
+ ptrCheckGuard(mlpqOperPtr, coprecsize, operationrec);
+ }//if
+}//Dbacc::moveLastParallelQueue()
+
+void Dbacc::moveLastParallelQueueWrite(Signal* signal)
+{
+ /* --------------------------------------------------------------------------------- */
+ /* ENSURE THAT ALL OPERATIONS HAVE LOCK MODE SET TO WRITE SINCE WE INSERT A */
+ /* WRITE LOCK INTO THE PARALLEL QUEUE. */
+ /* --------------------------------------------------------------------------------- */
+ while (mlpqOperPtr.p->nextParallelQue != RNIL) {
+ jam();
+ mlpqOperPtr.p->lockMode = operationRecPtr.p->lockMode;
+ mlpqOperPtr.i = mlpqOperPtr.p->nextParallelQue;
+ ptrCheckGuard(mlpqOperPtr, coprecsize, operationrec);
+ }//if
+ mlpqOperPtr.p->lockMode = operationRecPtr.p->lockMode;
+}//Dbacc::moveLastParallelQueueWrite()
+
+/* --------------------------------------------------------------------------------- */
+/* PLACE_WRITE_IN_LOCK_QUEUE */
+/* INPUT: OPERATION_REC_PTR OUR OPERATION POINTER */
+/* QUE_OPER_PTR LOCK QUEUE OWNER OPERATION POINTER */
+/* PWI_PAGEPTR PAGE POINTER OF ELEMENT */
+/* TPWI_ELEMENTPTR ELEMENT POINTER OF ELEMENT */
+/* OUTPUT TRESULT = */
+/* ZPARALLEL_QUEUE OPERATION PLACED IN PARALLEL QUEUE */
+/* OPERATION CAN PROCEED NOW. */
+/* ZSERIAL_QUEUE OPERATION PLACED IN SERIAL QUEUE */
+/* ERROR CODE OPERATION NEEDS ABORTING */
+/* --------------------------------------------------------------------------------- */
+Uint32 Dbacc::placeWriteInLockQueue(Signal* signal)
+{
+ tgnptMainOpPtr = queOperPtr;
+ getNoParallelTransaction(signal);
+ if (!((tgnptNrTransaction == 1) &&
+ (queOperPtr.p->transId1 == operationRecPtr.p->transId1) &&
+ (queOperPtr.p->transId2 == operationRecPtr.p->transId2))) {
+ jam();
+ placeSerialQueueWrite(signal);
+ return ZSERIAL_QUEUE;
+ }//if
+
+ /*
+ WE ARE PERFORMING AN READ EXCLUSIVE, INSERT, UPDATE OR DELETE IN THE SAME
+ TRANSACTION WHERE WE PREVIOUSLY HAVE EXECUTED AN OPERATION.
+ Read-All, Update-All, Insert-All and Delete-Insert are allowed
+ combinations.
+ Delete-Read, Delete-Update and Delete-Delete are not an allowed
+ combination and will result in tuple not found error.
+ */
+ mlpqOperPtr = queOperPtr;
+ moveLastParallelQueueWrite(signal);
+
+ if (operationRecPtr.p->operation == ZINSERT &&
+ mlpqOperPtr.p->operation != ZDELETE){
+ jam();
+ return ZWRITE_ERROR;
+ }//if
+
+ operationRecPtr.p->localdata[0] = queOperPtr.p->localdata[0];
+ operationRecPtr.p->localdata[1] = queOperPtr.p->localdata[1];
+ operationRecPtr.p->prevParallelQue = mlpqOperPtr.i;
+ mlpqOperPtr.p->nextParallelQue = operationRecPtr.i;
+ return ZPARALLEL_QUEUE;
+}//Dbacc::placeWriteInLockQueue()
+
+/* --------------------------------------------------------------------------------- */
+/* WE HAVE TO PLACE IT SOMEWHERE IN THE SERIAL QUEUE INSTEAD. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::placeSerialQueueWrite(Signal* signal)
+{
+ readWriteOpPtr = queOperPtr;
+ PSQW_LOOP:
+ if (readWriteOpPtr.p->nextSerialQue == RNIL) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE COULD NOT PUT IN ANY PARALLEL QUEUE. WE MUST PUT IT LAST IN THE SERIAL QUEUE. */
+ /* --------------------------------------------------------------------------------- */
+ readWriteOpPtr.p->nextSerialQue = operationRecPtr.i;
+ operationRecPtr.p->prevSerialQue = readWriteOpPtr.i;
+ putOpInFragWaitQue(signal);
+ return;
+ }//if
+ readWriteOpPtr.i = readWriteOpPtr.p->nextSerialQue;
+ ptrCheckGuard(readWriteOpPtr, coprecsize, operationrec);
+ tgnptMainOpPtr = readWriteOpPtr;
+ getNoParallelTransaction(signal);
+ if (tgnptNrTransaction == 1) {
+ /* --------------------------------------------------------------------------------- */
+ /* THERE WAS ONLY ONE TRANSACTION INVOLVED IN THE PARALLEL QUEUE. IF THIS IS OUR */
+ /* TRANSACTION WE CAN STILL GET HOLD OF THE LOCK. */
+ /* --------------------------------------------------------------------------------- */
+ if ((readWriteOpPtr.p->transId1 == operationRecPtr.p->transId1) &&
+ (readWriteOpPtr.p->transId2 == operationRecPtr.p->transId2)) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE ARE PERFORMING AN UPDATE OR DELETE IN THE SAME TRANSACTION WHERE WE ALREADY */
+ /* PREVIOUSLY HAVE EXECUTED AN OPERATION. INSERT-DELETE, READ-UPDATE, READ-READ, */
+ /* UPDATE-UPDATE, UPDATE-DELETE, READ-DELETE, INSERT-READ, INSERT-UPDATE ARE ALLOWED */
+ /* COMBINATIONS. A NEW INSERT AFTER A DELETE IS NOT ALLOWED AND SUCH AN INSERT WILL */
+ /* GO TO THE SERIAL LOCK QUEUE WHICH IT WILL NOT LEAVE UNTIL A TIME-OUT AND THE */
+ /* TRANSACTION IS ABORTED. READS AND UPDATES AFTER DELETES IS ALSO NOT ALLOWED. */
+ /* --------------------------------------------------------------------------------- */
+ mlpqOperPtr = readWriteOpPtr;
+ moveLastParallelQueueWrite(signal);
+ readWriteOpPtr = mlpqOperPtr;
+ operationRecPtr.p->prevParallelQue = readWriteOpPtr.i;
+ readWriteOpPtr.p->nextParallelQue = operationRecPtr.i;
+ operationRecPtr.p->localdata[0] = readWriteOpPtr.p->localdata[0];
+ operationRecPtr.p->localdata[1] = readWriteOpPtr.p->localdata[1];
+ putOpInFragWaitQue(signal);
+ return;
+ }//if
+ }//if
+ goto PSQW_LOOP;
+}//Dbacc::placeSerialQueueWrite()
+
+/* ------------------------------------------------------------------------- */
+/* ACC KEYREQ END */
+/* ------------------------------------------------------------------------- */
+void Dbacc::acckeyref1Lab(Signal* signal, Uint32 result_code)
+{
+ if (operationRecPtr.p->keyinfoPage != RNIL) {
+ jam();
+ rpPageptr.i = operationRecPtr.p->keyinfoPage;
+ ptrCheckGuard(rpPageptr, cpagesize, page8);
+ releasePage(signal);
+ operationRecPtr.p->keyinfoPage = RNIL;
+ }//if
+ operationRecPtr.p->transactionstate = WAIT_COMMIT_ABORT;
+ /* ************************<< */
+ /* ACCKEYREF */
+ /* ************************<< */
+ signal->theData[0] = cminusOne;
+ signal->theData[1] = result_code;
+ return;
+}//Dbacc::acckeyref1Lab()
+
+/* ******************--------------------------------------------------------------- */
+/* ACCMINUPDATE UPDATE LOCAL KEY REQ */
+/* DESCRIPTION: UPDATES LOCAL KEY OF AN ELEMENTS IN THE HASH TABLE */
+/* THIS SIGNAL IS WAITED AFTER ANY INSERT REQ */
+/* ENTER ACCMINUPDATE WITH SENDER: LQH, LEVEL B */
+/* OPERATION_REC_PTR, OPERATION RECORD PTR */
+/* CLOCALKEY(0), LOCAL KEY 1 */
+/* CLOCALKEY(1) LOCAL KEY 2 */
+/* ******************--------------------------------------------------------------- */
+void Dbacc::execACCMINUPDATE(Signal* signal)
+{
+ Page8Ptr ulkPageidptr;
+ Uint32 tulkLocalPtr;
+ Uint32 tlocalkey1, tlocalkey2;
+ Uint32 TlogStart;
+
+ jamEntry();
+ operationRecPtr.i = signal->theData[0];
+ tlocalkey1 = signal->theData[1];
+ tlocalkey2 = signal->theData[2];
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ if (operationRecPtr.p->transactionstate == ACTIVE) {
+ fragrecptr.i = operationRecPtr.p->fragptr;
+ ulkPageidptr.i = operationRecPtr.p->elementPage;
+ tulkLocalPtr = operationRecPtr.p->elementPointer + operationRecPtr.p->elementIsforward;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ ptrCheckGuard(ulkPageidptr, cpagesize, page8);
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ //----------------------------------------------------------
+ // To avoid undo log the element header we take care to only
+ // undo log the local key part.
+ //----------------------------------------------------------
+ if (operationRecPtr.p->elementIsforward == 1) {
+ jam();
+ TlogStart = tulkLocalPtr;
+ } else {
+ jam();
+ TlogStart = tulkLocalPtr - fragrecptr.p->localkeylen + 1;
+ }//if
+ datapageptr.p = ulkPageidptr.p;
+ cundoinfolength = fragrecptr.p->localkeylen;
+ cundoElemIndex = TlogStart;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(ulkPageidptr, tulkLocalPtr, tlocalkey1);
+ arrGuard(tulkLocalPtr, 2048);
+ ulkPageidptr.p->word32[tulkLocalPtr] = tlocalkey1;
+ operationRecPtr.p->localdata[0] = tlocalkey1;
+ if (fragrecptr.p->localkeylen == 1) {
+ return;
+ } else if (fragrecptr.p->localkeylen == 2) {
+ jam();
+ tulkLocalPtr = tulkLocalPtr + operationRecPtr.p->elementIsforward;
+ operationRecPtr.p->localdata[1] = tlocalkey2;
+ dbgWord32(ulkPageidptr, tulkLocalPtr, tlocalkey2);
+ arrGuard(tulkLocalPtr, 2048);
+ ulkPageidptr.p->word32[tulkLocalPtr] = tlocalkey2;
+ return;
+ } else {
+ jam();
+ }//if
+ }//if
+ ndbrequire(false);
+}//Dbacc::execACCMINUPDATE()
+
+/* ******************--------------------------------------------------------------- */
+/* ACC_COMMITREQ COMMIT TRANSACTION */
+/* SENDER: LQH, LEVEL B */
+/* INPUT: OPERATION_REC_PTR , */
+/* ******************--------------------------------------------------------------- */
+void Dbacc::execACC_COMMITREQ(Signal* signal)
+{
+ Uint8 Toperation;
+ jamEntry();
+ operationRecPtr.i = signal->theData[0];
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ ndbrequire(operationRecPtr.p->transactionstate == ACTIVE);
+ fragrecptr.i = operationRecPtr.p->fragptr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ commitOperation(signal);
+ Toperation = operationRecPtr.p->operation;
+ operationRecPtr.p->transactionstate = IDLE;
+ operationRecPtr.p->operation = ZUNDEFINED_OP;
+ if (Toperation != ZINSERT) {
+ if (Toperation != ZDELETE) {
+ return;
+ } else {
+ jam();
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ rootfragrecptr.p->noOfElements--;
+ fragrecptr.p->slack += operationRecPtr.p->insertDeleteLen;
+ if (fragrecptr.p->slack > fragrecptr.p->slackCheck) { /* TIME FOR JOIN BUCKETS PROCESS */
+ if (fragrecptr.p->expandCounter > 0) {
+ if (fragrecptr.p->expandFlag == 0) {
+ jam();
+ fragrecptr.p->expandFlag = 1;
+ signal->theData[0] = fragrecptr.i;
+ signal->theData[1] = fragrecptr.p->p;
+ signal->theData[2] = fragrecptr.p->maxp;
+ sendSignal(cownBlockref, GSN_SHRINKCHECK2, signal, 3, JBB);
+ }//if
+ }//if
+ }//if
+ }//if
+ } else {
+ jam(); /* EXPAND PROCESS HANDLING */
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ rootfragrecptr.p->noOfElements++;
+ fragrecptr.p->slack -= operationRecPtr.p->insertDeleteLen;
+ if (fragrecptr.p->slack >= (Uint32)(1 << 31)) { /* IT MEANS THAT IF SLACK < ZERO */
+ if (fragrecptr.p->expandFlag == 0) {
+ jam();
+ fragrecptr.p->expandFlag = 1;
+ signal->theData[0] = fragrecptr.i;
+ signal->theData[1] = fragrecptr.p->p;
+ signal->theData[2] = fragrecptr.p->maxp;
+ sendSignal(cownBlockref, GSN_EXPANDCHECK2, signal, 3, JBB);
+ }//if
+ }//if
+ }//if
+ return;
+}//Dbacc::execACC_COMMITREQ()
+
+/* ******************--------------------------------------------------------------- */
+/* ACC ABORT REQ ABORT ALL OPERATION OF THE TRANSACTION */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+/* ******************--------------------------------------------------------------- */
+/* ACC ABORT REQ ABORT TRANSACTION */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execACC_ABORTREQ(Signal* signal)
+{
+ jamEntry();
+ accAbortReqLab(signal, true);
+}//Dbacc::execACC_ABORTREQ()
+
+void Dbacc::accAbortReqLab(Signal* signal, bool sendConf)
+{
+ operationRecPtr.i = signal->theData[0];
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ tresult = 0; /* ZFALSE */
+ if ((operationRecPtr.p->transactionstate == ACTIVE) ||
+ (operationRecPtr.p->transactionstate == WAIT_COMMIT_ABORT)) {
+ jam();
+ fragrecptr.i = operationRecPtr.p->fragptr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ operationRecPtr.p->transactionstate = ABORT;
+ abortOperation(signal);
+ } else {
+ ndbrequire(operationRecPtr.p->transactionstate == IDLE);
+ jam();
+ }//if
+ operationRecPtr.p->transactionstate = IDLE;
+ operationRecPtr.p->operation = ZUNDEFINED_OP;
+ if (! sendConf)
+ return;
+ signal->theData[0] = operationRecPtr.p->userptr;
+ sendSignal(operationRecPtr.p->userblockref, GSN_ACC_ABORTCONF, signal, 1, JBB);
+ return;
+}//Dbacc::accAbortReqLab()
+
+/*
+ * Lock or unlock tuple.
+ */
+void Dbacc::execACC_LOCKREQ(Signal* signal)
+{
+ jamEntry();
+ AccLockReq* sig = (AccLockReq*)signal->getDataPtrSend();
+ AccLockReq reqCopy = *sig;
+ AccLockReq* const req = &reqCopy;
+ Uint32 lockOp = (req->requestInfo & 0xFF);
+ if (lockOp == AccLockReq::LockShared ||
+ lockOp == AccLockReq::LockExclusive) {
+ jam();
+ // find table
+ tabptr.i = req->tableId;
+ ptrCheckGuard(tabptr, ctablesize, tabrec);
+ // find fragment (TUX will know it)
+ if (req->fragPtrI == RNIL) {
+ for (Uint32 i = 0; i < NO_OF_FRAG_PER_NODE; i++) {
+ jam();
+ if (tabptr.p->fragptrholder[i] != RNIL) {
+ rootfragrecptr.i = tabptr.p->fragptrholder[i];
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ if (rootfragrecptr.p->fragmentid[0] == req->fragId) {
+ jam();
+ req->fragPtrI = rootfragrecptr.p->fragmentptr[0];
+ break;
+ }
+ if (rootfragrecptr.p->fragmentid[1] == req->fragId) {
+ jam();
+ req->fragPtrI = rootfragrecptr.p->fragmentptr[1];
+ break;
+ }
+ }
+ }
+ }
+ fragrecptr.i = req->fragPtrI;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ // caller must be explicit here
+ ndbrequire(req->accOpPtr == RNIL);
+ // seize operation to hold the lock
+ if (cfreeopRec != RNIL) {
+ jam();
+ seizeOpRec(signal);
+ // init as in ACCSEIZEREQ
+ operationRecPtr.p->userptr = req->userPtr;
+ operationRecPtr.p->userblockref = req->userRef;
+ operationRecPtr.p->operation = ZUNDEFINED_OP;
+ operationRecPtr.p->transactionstate = IDLE;
+ // do read with lock via ACCKEYREQ
+ Uint32 lockMode = (lockOp == AccLockReq::LockShared) ? 0 : 1;
+ Uint32 opCode = ZSCAN_OP;
+ signal->theData[0] = operationRecPtr.i;
+ signal->theData[1] = fragrecptr.i;
+ signal->theData[2] = opCode | (lockMode << 4) | (1 << 31);
+ signal->theData[3] = req->hashValue;
+ signal->theData[4] = 1; // fake primKeyLen
+ signal->theData[5] = req->transId1;
+ signal->theData[6] = req->transId2;
+ signal->theData[7] = req->tupAddr;
+ EXECUTE_DIRECT(DBACC, GSN_ACCKEYREQ, signal, 8);
+ // translate the result
+ if (signal->theData[0] < RNIL) {
+ jam();
+ req->returnCode = AccLockReq::Success;
+ req->accOpPtr = operationRecPtr.i;
+ } else if (signal->theData[0] == RNIL) {
+ jam();
+ req->returnCode = AccLockReq::IsBlocked;
+ req->accOpPtr = operationRecPtr.i;
+ } else {
+ ndbrequire(signal->theData[0] == (UintR)-1);
+ releaseOpRec(signal);
+ req->returnCode = AccLockReq::Refused;
+ req->accOpPtr = RNIL;
+ }
+ } else {
+ jam();
+ req->returnCode = AccLockReq::NoFreeOp;
+ }
+ *sig = *req;
+ return;
+ }
+ if (lockOp == AccLockReq::Unlock) {
+ jam();
+ // do unlock via ACC_COMMITREQ (immediate)
+ signal->theData[0] = req->accOpPtr;
+ EXECUTE_DIRECT(DBACC, GSN_ACC_COMMITREQ, signal, 1);
+ releaseOpRec(signal);
+ req->returnCode = AccLockReq::Success;
+ *sig = *req;
+ return;
+ }
+ if (lockOp == AccLockReq::Abort) {
+ jam();
+ // do abort via ACC_ABORTREQ (immediate)
+ signal->theData[0] = req->accOpPtr;
+ accAbortReqLab(signal, false);
+ releaseOpRec(signal);
+ req->returnCode = AccLockReq::Success;
+ *sig = *req;
+ return;
+ }
+ if (lockOp == AccLockReq::AbortWithConf) {
+ jam();
+ // do abort via ACC_ABORTREQ (with conf signal)
+ signal->theData[0] = req->accOpPtr;
+ accAbortReqLab(signal, true);
+ releaseOpRec(signal);
+ req->returnCode = AccLockReq::Success;
+ *sig = *req;
+ return;
+ }
+ ndbrequire(false);
+}
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF EXECUTE OPERATION MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* MODULE: INSERT */
+/* THE FOLLOWING SUBROUTINES ARE ONLY USED BY INSERT_ELEMENT. THIS */
+/* ROUTINE IS THE SOLE INTERFACE TO INSERT ELEMENTS INTO THE INDEX. */
+/* CURRENT USERS ARE INSERT REQUESTS, EXPAND CONTAINER AND SHRINK */
+/* CONTAINER. */
+/* */
+/* THE FOLLOWING SUBROUTINES ARE INCLUDED IN THIS MODULE: */
+/* INSERT_ELEMENT */
+/* INSERT_CONTAINER */
+/* ADDNEWCONTAINER */
+/* GETFREELIST */
+/* INCREASELISTCONT */
+/* SEIZE_LEFTLIST */
+/* SEIZE_RIGHTLIST */
+/* */
+/* THESE ROUTINES ARE ONLY USED BY THIS MODULE AND BY NO ONE ELSE. */
+/* ALSO THE ROUTINES MAKE NO USE OF ROUTINES IN OTHER MODULES. */
+/* TAKE_REC_OUT_OF_FREE_OVERPAGE AND RELEASE_OVERFLOW_REC ARE */
+/* EXCEPTIONS TO THIS RULE. */
+/* */
+/* THE ONLY SHORT-LIVED VARIABLES USED IN OTHER PARTS OF THE BLOCK ARE */
+/* THOSE DEFINED AS INPUT AND OUTPUT IN INSERT_ELEMENT */
+/* SHORT-LIVED VARIABLES INCLUDE TEMPORARY VARIABLES, COMMON VARIABLES */
+/* AND POINTER VARIABLES. */
+/* THE ONLY EXCEPTION TO THIS RULE IS FRAGRECPTR WHICH POINTS TO THE */
+/* FRAGMENT RECORD. THIS IS MORE LESS STATIC ALWAYS DURING A SIGNAL */
+/* EXECUTION. */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* INSERT_ELEMENT */
+/* INPUT: */
+/* IDR_PAGEPTR (POINTER TO THE ACTIVE PAGE REC) */
+/* TIDR_PAGEINDEX (INDEX OF THE CONTAINER) */
+/* TIDR_FORWARD (DIRECTION FORWARD OR BACKWARD) */
+/* TIDR_ELEMHEAD (HEADER OF ELEMENT TO BE INSERTED */
+/* CIDR_KEYS(ARRAY OF TUPLE KEYS) */
+/* CLOCALKEY(ARRAY OF LOCAL KEYS). */
+/* FRAGRECPTR */
+/* IDR_OPERATION_REC_PTR */
+/* TIDR_KEY_LEN */
+/* */
+/* OUTPUT: */
+/* TIDR_PAGEINDEX (PAGE INDEX OF INSERTED ELEMENT) */
+/* IDR_PAGEPTR (PAGE POINTER OF INSERTED ELEMENT) */
+/* TIDR_FORWARD (CONTAINER DIRECTION OF INSERTED ELEMENT) */
+/* NONE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::insertElement(Signal* signal)
+{
+ DirRangePtr inrOverflowrangeptr;
+ DirectoryarrayPtr inrOverflowDirptr;
+ OverflowRecordPtr inrOverflowRecPtr;
+ Page8Ptr inrNewPageptr;
+ Uint32 tinrNextSamePage;
+ Uint32 tinrTmp;
+
+ do {
+ insertContainer(signal);
+ if (tidrResult != ZFALSE) {
+ jam();
+ return;
+ /* INSERTION IS DONE, OR */
+ /* AN ERROR IS DETECTED */
+ }//if
+ if (((tidrContainerhead >> 7) & 0x3) != 0) {
+ tinrNextSamePage = (tidrContainerhead >> 9) & 0x1; /* CHECK BIT FOR CHECKING WHERE */
+ /* THE NEXT CONTAINER IS IN THE SAME PAGE */
+ tidrPageindex = tidrContainerhead & 0x7f; /* NEXT CONTAINER PAGE INDEX 7 BITS */
+ if (((tidrContainerhead >> 7) & 3) == ZLEFT) {
+ jam();
+ tidrForward = ZTRUE;
+ } else if (((tidrContainerhead >> 7) & 3) == ZRIGHT) {
+ jam();
+ tidrForward = cminusOne;
+ } else {
+ ndbrequire(false);
+ return;
+ }//if
+ if (tinrNextSamePage == ZFALSE) {
+ jam(); /* NEXT CONTAINER IS IN AN OVERFLOW PAGE */
+ tinrTmp = idrPageptr.p->word32[tidrContainerptr + 1];
+ inrOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ ptrCheckGuard(inrOverflowrangeptr, cdirrangesize, dirRange);
+ arrGuard((tinrTmp >> 8), 256);
+ inrOverflowDirptr.i = inrOverflowrangeptr.p->dirArray[tinrTmp >> 8];
+ ptrCheckGuard(inrOverflowDirptr, cdirarraysize, directoryarray);
+ idrPageptr.i = inrOverflowDirptr.p->pagep[tinrTmp & 0xff];
+ ptrCheckGuard(idrPageptr, cpagesize, page8);
+ }//if
+ ndbrequire(tidrPageindex < ZEMPTYLIST);
+ } else {
+ break;
+ }//if
+ } while (1);
+ gflPageptr.p = idrPageptr.p;
+ getfreelist(signal);
+ if (tgflPageindex == ZEMPTYLIST) {
+ jam();
+ /* NO FREE BUFFER IS FOUND */
+ if (fragrecptr.p->firstOverflowRec == RNIL) {
+ jam();
+ allocOverflowPage(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ }//if
+ inrOverflowRecPtr.i = fragrecptr.p->firstOverflowRec;
+ ptrCheckGuard(inrOverflowRecPtr, coverflowrecsize, overflowRecord);
+ inrNewPageptr.i = inrOverflowRecPtr.p->overpage;
+ ptrCheckGuard(inrNewPageptr, cpagesize, page8);
+ gflPageptr.p = inrNewPageptr.p;
+ getfreelist(signal);
+ ndbrequire(tgflPageindex != ZEMPTYLIST);
+ tancNext = 0;
+ } else {
+ jam();
+ inrNewPageptr = idrPageptr;
+ tancNext = 1;
+ }//if
+ tslUpdateHeader = ZTRUE;
+ tslPageindex = tgflPageindex;
+ slPageptr.p = inrNewPageptr.p;
+ if (tgflBufType == ZLEFT) {
+ seizeLeftlist(signal);
+ tidrForward = ZTRUE;
+ } else {
+ seizeRightlist(signal);
+ tidrForward = cminusOne;
+ }//if
+ tancPageindex = tgflPageindex;
+ tancPageid = inrNewPageptr.p->word32[ZPOS_PAGE_ID];
+ tancBufType = tgflBufType;
+ tancContainerptr = tidrContainerptr;
+ ancPageptr.p = idrPageptr.p;
+ addnewcontainer(signal);
+
+ idrPageptr = inrNewPageptr;
+ tidrPageindex = tgflPageindex;
+ insertContainer(signal);
+ ndbrequire(tidrResult == ZTRUE);
+}//Dbacc::insertElement()
+
+/* --------------------------------------------------------------------------------- */
+/* INSERT_CONTAINER */
+/* INPUT: */
+/* IDR_PAGEPTR (POINTER TO THE ACTIVE PAGE REC) */
+/* TIDR_PAGEINDEX (INDEX OF THE CONTAINER) */
+/* TIDR_FORWARD (DIRECTION FORWARD OR BACKWARD) */
+/* TIDR_ELEMHEAD (HEADER OF ELEMENT TO BE INSERTED */
+/* CKEYS(ARRAY OF TUPLE KEYS) */
+/* CLOCALKEY(ARRAY 0F LOCAL KEYS). */
+/* TIDR_KEY_LEN */
+/* FRAGRECPTR */
+/* IDR_OPERATION_REC_PTR */
+/* OUTPUT: */
+/* TIDR_RESULT (ZTRUE FOR SUCCESS AND ZFALSE OTHERWISE) */
+/* TIDR_CONTAINERHEAD (HEADER OF CONTAINER) */
+/* TIDR_CONTAINERPTR (POINTER TO CONTAINER HEADER) */
+/* */
+/* DESCRIPTION: */
+/* THE FREE AREA OF THE CONTAINER WILL BE CALCULATED. IF IT IS */
+/* LARGER THAN OR EQUAL THE ELEMENT LENGTH. THE ELEMENT WILL BE */
+/* INSERT IN THE CONTAINER AND CONTAINER HEAD WILL BE UPDATED. */
+/* THIS ROUTINE ALWAYS DEALS WITH ONLY ONE CONTAINER AND DO NEVER */
+/* START ANYTHING OUTSIDE OF THIS CONTAINER. */
+/* */
+/* SHORT FORM: IDR */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::insertContainer(Signal* signal)
+{
+ Uint32 tidrContainerlen;
+ Uint32 tidrConfreelen;
+ Uint32 tidrNextSide;
+ Uint32 tidrNextConLen;
+ Uint32 tidrIndex;
+ Uint32 tidrInputIndex;
+ Uint32 tidrContLen;
+ Uint32 guard26;
+
+ tidrResult = ZFALSE;
+ tidrContainerptr = (tidrPageindex << ZSHIFT_PLUS) - (tidrPageindex << ZSHIFT_MINUS);
+ tidrContainerptr = tidrContainerptr + ZHEAD_SIZE;
+ /* --------------------------------------------------------------------------------- */
+ /* CALCULATE THE POINTER TO THE ELEMENT TO BE INSERTED AND THE POINTER TO THE */
+ /* CONTAINER HEADER OF THE OTHER SIDE OF THE BUFFER. */
+ /* --------------------------------------------------------------------------------- */
+ if (tidrForward == ZTRUE) {
+ jam();
+ tidrNextSide = tidrContainerptr + (ZBUF_SIZE - ZCON_HEAD_SIZE);
+ arrGuard(tidrNextSide + 1, 2048);
+ tidrContainerhead = idrPageptr.p->word32[tidrContainerptr];
+ tidrContainerlen = tidrContainerhead >> 26;
+ tidrIndex = tidrContainerptr + tidrContainerlen;
+ } else {
+ jam();
+ tidrNextSide = tidrContainerptr;
+ tidrContainerptr = tidrContainerptr + (ZBUF_SIZE - ZCON_HEAD_SIZE);
+ arrGuard(tidrContainerptr + 1, 2048);
+ tidrContainerhead = idrPageptr.p->word32[tidrContainerptr];
+ tidrContainerlen = tidrContainerhead >> 26;
+ tidrIndex = (tidrContainerptr - tidrContainerlen) + (ZCON_HEAD_SIZE - 1);
+ }//if
+ if (tidrContainerlen > (ZBUF_SIZE - 3)) {
+ return;
+ }//if
+ tidrConfreelen = ZBUF_SIZE - tidrContainerlen;
+ /* --------------------------------------------------------------------------------- */
+ /* WE CALCULATE THE TOTAL LENGTH THE CONTAINER CAN EXPAND TO */
+ /* THIS INCLUDES THE OTHER SIDE OF THE BUFFER IF POSSIBLE TO EXPAND THERE. */
+ /* --------------------------------------------------------------------------------- */
+ if (((tidrContainerhead >> 10) & 1) == 0) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE HAVE NOT EXPANDED TO THE ENTIRE BUFFER YET. WE CAN THUS READ THE OTHER */
+ /* SIDE'S CONTAINER HEADER TO READ HIS LENGTH. */
+ /* --------------------------------------------------------------------------------- */
+ tidrNextConLen = idrPageptr.p->word32[tidrNextSide] >> 26;
+ tidrConfreelen = tidrConfreelen - tidrNextConLen;
+ if (tidrConfreelen > ZBUF_SIZE) {
+ ndbrequire(false);
+ /* --------------------------------------------------------------------------------- */
+ /* THE BUFFERS ARE PLACED ON TOP OF EACH OTHER. THIS SHOULD NEVER OCCUR. */
+ /* --------------------------------------------------------------------------------- */
+ return;
+ }//if
+ } else {
+ jam();
+ tidrNextConLen = 1; /* INDICATE OTHER SIDE IS NOT PART OF FREE LIST */
+ }//if
+ if (tidrConfreelen < fragrecptr.p->elementLength) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE CONTAINER COULD NOT BE EXPANDED TO FIT THE NEW ELEMENT. WE HAVE TO */
+ /* RETURN AND FIND A NEW CONTAINER TO INSERT IT INTO. */
+ /* --------------------------------------------------------------------------------- */
+ return;
+ }//if
+ tidrContainerlen = tidrContainerlen + fragrecptr.p->elementLength;
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ datapageptr.p = idrPageptr.p;
+ cundoElemIndex = tidrContainerptr;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ if (tidrNextConLen == 0) {
+ /* EACH SIDE OF THE BUFFER WHICH BELONG TO A FREE */
+ /* LIST, HAS ZERO AS LENGTH. */
+ if (tidrContainerlen > ZUP_LIMIT) {
+ dbgWord32(idrPageptr, tidrContainerptr, idrPageptr.p->word32[tidrContainerptr] | (1 << 10));
+ idrPageptr.p->word32[tidrContainerptr] = idrPageptr.p->word32[tidrContainerptr] | (1 << 10);
+ tslUpdateHeader = ZFALSE;
+ tslPageindex = tidrPageindex;
+ slPageptr.p = idrPageptr.p;
+ if (tidrForward == ZTRUE) {
+ jam();
+ seizeRightlist(signal); /* REMOVE THE RIGHT SIDE OF THE BUFFER FROM THE LIST */
+ } else {
+ jam();
+ /* OF THE FREE CONTAINERS */
+ seizeLeftlist(signal); /* REMOVE THE LEFT SIDE OF THE BUFFER FROM THE LIST */
+ }//if
+ }//if
+ }//if
+ /* OF THE FREE CONTAINERS */
+ /* --------------------------------------------------------------------------------- */
+ /* WE HAVE NOW FOUND A FREE SPOT IN THE CURRENT CONTAINER. WE INSERT THE */
+ /* ELEMENT HERE. THE ELEMENT CONTAINS A HEADER, A LOCAL KEY AND A TUPLE KEY. */
+ /* BEFORE INSERTING THE ELEMENT WE WILL UPDATE THE OPERATION RECORD WITH THE */
+ /* DATA CONCERNING WHERE WE INSERTED THE ELEMENT. THIS MAKES IT EASY TO FIND */
+ /* THIS INFORMATION WHEN WE RETURN TO UPDATE THE LOCAL KEY OR RETURN TO COMMIT */
+ /* OR ABORT THE INSERT. IF NO OPERATION RECORD EXIST IT MEANS THAT WE ARE */
+ /* PERFORMING THIS AS A PART OF THE EXPAND OR SHRINK PROCESS. */
+ /* --------------------------------------------------------------------------------- */
+ if (idrOperationRecPtr.i != RNIL) {
+ jam();
+ idrOperationRecPtr.p->elementIsforward = tidrForward;
+ idrOperationRecPtr.p->elementPage = idrPageptr.i;
+ idrOperationRecPtr.p->elementContainer = tidrContainerptr;
+ idrOperationRecPtr.p->elementPointer = tidrIndex;
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* WE CHOOSE TO UNDO LOG INSERTS BY WRITING THE BEFORE VALUE TO THE UNDO LOG. */
+ /* WE COULD ALSO HAVE DONE THIS BY WRITING THIS BEFORE VALUE WHEN DELETING */
+ /* ELEMENTS. WE CHOOSE TO PUT IT HERE SINCE WE THEREBY ENSURE THAT WE ALWAYS */
+ /* UNDO LOG ALL WRITES TO PAGE MEMORY. IT SHOULD BE EASIER TO MAINTAIN SUCH A */
+ /* STRUCTURE. IT IS RATHER DIFFICULT TO MAINTAIN A LOGICAL STRUCTURE WHERE */
+ /* DELETES ARE INSERTS AND INSERTS ARE PURELY DELETES. */
+ /* --------------------------------------------------------------------------------- */
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ if (tidrForward == ZTRUE) {
+ cundoElemIndex = tidrIndex;
+ } else {
+ cundoElemIndex = (tidrIndex + 1) - fragrecptr.p->elementLength;
+ }//if
+ cundoinfolength = fragrecptr.p->elementLength;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(idrPageptr, tidrIndex, tidrElemhead);
+ idrPageptr.p->word32[tidrIndex] = tidrElemhead; /* INSERTS THE HEAD OF THE ELEMENT */
+ tidrIndex += tidrForward;
+ guard26 = fragrecptr.p->localkeylen - 1;
+ arrGuard(guard26, 2);
+ for (tidrInputIndex = 0; tidrInputIndex <= guard26; tidrInputIndex++) {
+ dbgWord32(idrPageptr, tidrIndex, clocalkey[tidrInputIndex]);
+ arrGuard(tidrIndex, 2048);
+ idrPageptr.p->word32[tidrIndex] = clocalkey[tidrInputIndex]; /* INSERTS LOCALKEY */
+ tidrIndex += tidrForward;
+ }//for
+ guard26 = tidrKeyLen - 1;
+ arrGuard(guard26, 8);
+ for (tidrInputIndex = 0; tidrInputIndex <= guard26; tidrInputIndex++) {
+ dbgWord32(idrPageptr, tidrIndex, ckeys[tidrInputIndex]);
+ arrGuard(tidrIndex, 2048);
+ idrPageptr.p->word32[tidrIndex] = ckeys[tidrInputIndex]; /* INSERTS TUPLE KEY */
+ tidrIndex += tidrForward;
+ }//for
+ tidrContLen = idrPageptr.p->word32[tidrContainerptr] << 6;
+ tidrContLen = tidrContLen >> 6;
+ dbgWord32(idrPageptr, tidrContainerptr, (tidrContainerlen << 26) | tidrContLen);
+ idrPageptr.p->word32[tidrContainerptr] = (tidrContainerlen << 26) | tidrContLen;
+ tidrResult = ZTRUE;
+}//Dbacc::insertContainer()
+
+/* --------------------------------------------------------------------------------- */
+/* ADDNEWCONTAINER */
+/* INPUT: */
+/* TANC_CONTAINERPTR */
+/* ANC_PAGEPTR */
+/* TANC_NEXT */
+/* TANC_PAGEINDEX */
+/* TANC_BUF_TYPE */
+/* TANC_PAGEID */
+/* OUTPUT: */
+/* NONE */
+/* */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::addnewcontainer(Signal* signal)
+{
+ Uint32 tancTmp1;
+
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ cundoElemIndex = tancContainerptr;
+ datapageptr.p = ancPageptr.p;
+ cundoinfolength = 2;
+ undoWritingProcess(signal); /* WHEN UNDO PROCESS HAS STARTED, */
+ }//if
+ /* THE OLD DATA IS STORED ON AN UNDO PAGE */
+ /* --------------------------------------------------------------------------------- */
+ /* KEEP LENGTH INFORMATION IN BIT 26-31. */
+ /* SET BIT 9 INDICATING IF NEXT BUFFER IN THE SAME PAGE USING TANC_NEXT. */
+ /* SET TYPE OF NEXT CONTAINER IN BIT 7-8. */
+ /* SET PAGE INDEX OF NEXT CONTAINER IN BIT 0-6. */
+ /* KEEP INDICATOR OF OWNING OTHER SIDE OF BUFFER IN BIT 10. */
+ /* --------------------------------------------------------------------------------- */
+ tancTmp1 = ancPageptr.p->word32[tancContainerptr] >> 10;
+ tancTmp1 = tancTmp1 << 1;
+ tancTmp1 = tancTmp1 | tancNext;
+ tancTmp1 = tancTmp1 << 2;
+ tancTmp1 = tancTmp1 | tancBufType; /* TYPE OF THE NEXT CONTAINER */
+ tancTmp1 = tancTmp1 << 7;
+ tancTmp1 = tancTmp1 | tancPageindex;
+ dbgWord32(ancPageptr, tancContainerptr, tancTmp1);
+ ancPageptr.p->word32[tancContainerptr] = tancTmp1; /* HEAD OF THE CONTAINER IS UPDATED */
+ dbgWord32(ancPageptr, tancContainerptr + 1, tancPageid);
+ ancPageptr.p->word32[tancContainerptr + 1] = tancPageid;
+}//Dbacc::addnewcontainer()
+
+/* --------------------------------------------------------------------------------- */
+/* GETFREELIST */
+/* INPUT: */
+/* GFL_PAGEPTR (POINTER TO A PAGE RECORD). */
+/* OUTPUT: */
+/* TGFL_PAGEINDEX(POINTER TO A FREE BUFFER IN THE FREEPAGE), AND */
+/* TGFL_BUF_TYPE( TYPE OF THE FREE BUFFER). */
+/* DESCRIPTION: SEARCHS IN THE FREE LIST OF THE FREE BUFFER IN THE PAGE HEAD */
+/* (WORD32(1)),AND RETURN ADDRESS OF A FREE BUFFER OR NIL. */
+/* THE FREE BUFFER CAN BE A RIGHT CONTAINER OR A LEFT ONE */
+/* THE KIND OF THE CONTAINER IS NOTED BY TGFL_BUF_TYPE. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::getfreelist(Signal* signal)
+{
+ Uint32 tgflTmp;
+
+ tgflTmp = gflPageptr.p->word32[ZPOS_EMPTY_LIST];
+ tgflPageindex = (tgflTmp >> 7) & 0x7f; /* LEFT FREE LIST */
+ tgflBufType = ZLEFT;
+ if (tgflPageindex == ZEMPTYLIST) {
+ jam();
+ tgflPageindex = tgflTmp & 0x7f; /* RIGHT FREE LIST */
+ tgflBufType = ZRIGHT;
+ }//if
+ ndbrequire(tgflPageindex <= ZEMPTYLIST);
+}//Dbacc::getfreelist()
+
+/* --------------------------------------------------------------------------------- */
+/* INCREASELISTCONT */
+/* INPUT: */
+/* ILC_PAGEPTR PAGE POINTER TO INCREASE NUMBER OF CONTAINERS IN */
+/* A CONTAINER OF AN OVERFLOW PAGE (FREEPAGEPTR) IS ALLOCATED, NR OF */
+/* ALLOCATED CONTAINER HAVE TO BE INCRESE BY ONE . */
+/* IF THE NUMBER OF ALLOCATED CONTAINERS IS ABOVE THE FREE LIMIT WE WILL */
+/* REMOVE THE PAGE FROM THE FREE LIST. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::increaselistcont(Signal* signal)
+{
+ OverflowRecordPtr ilcOverflowRecPtr;
+
+ dbgWord32(ilcPageptr, ZPOS_ALLOC_CONTAINERS, ilcPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] + 1);
+ ilcPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] = ilcPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] + 1;
+ if (ilcPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] > ZFREE_LIMIT) {
+ if (ilcPageptr.p->word32[ZPOS_OVERFLOWREC] != RNIL) {
+ jam();
+ ilcOverflowRecPtr.i = ilcPageptr.p->word32[ZPOS_OVERFLOWREC];
+ dbgWord32(ilcPageptr, ZPOS_OVERFLOWREC, RNIL);
+ ilcPageptr.p->word32[ZPOS_OVERFLOWREC] = RNIL;
+ ptrCheckGuard(ilcOverflowRecPtr, coverflowrecsize, overflowRecord);
+ tfoOverflowRecPtr = ilcOverflowRecPtr;
+ takeRecOutOfFreeOverpage(signal);
+ rorOverflowRecPtr = ilcOverflowRecPtr;
+ releaseOverflowRec(signal);
+ }//if
+ }//if
+}//Dbacc::increaselistcont()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_LEFTLIST */
+/* INPUT: */
+/* TSL_PAGEINDEX PAGE INDEX OF CONTAINER TO SEIZE */
+/* SL_PAGEPTR PAGE POINTER OF CONTAINER TO SEIZE */
+/* TSL_UPDATE_HEADER SHOULD WE UPDATE THE CONTAINER HEADER */
+/* */
+/* OUTPUT: */
+/* NONE */
+/* DESCRIPTION: THE BUFFER NOTED BY TSL_PAGEINDEX WILL BE REMOVED FROM THE */
+/* LIST OF LEFT FREE CONTAINER, IN THE HEADER OF THE PAGE */
+/* (FREEPAGEPTR). PREVIOUS AND NEXT BUFFER OF REMOVED BUFFER */
+/* WILL BE UPDATED. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeLeftlist(Signal* signal)
+{
+ Uint32 tsllTmp1;
+ Uint32 tsllNewHead;
+ Uint32 tsllHeadIndex;
+ Uint32 tsllTmp;
+
+ tsllHeadIndex = ((tslPageindex << ZSHIFT_PLUS) - (tslPageindex << ZSHIFT_MINUS)) + ZHEAD_SIZE;
+ arrGuard(tsllHeadIndex + 1, 2048);
+ tslNextfree = slPageptr.p->word32[tsllHeadIndex];
+ tslPrevfree = slPageptr.p->word32[tsllHeadIndex + 1];
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ datapageptr.p = slPageptr.p;
+ cundoElemIndex = tsllHeadIndex;
+ cundoinfolength = 2;
+ undoWritingProcess(signal);
+ }//if
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ cundoElemIndex = ZPOS_EMPTY_LIST;
+ cundoinfolength = 2;
+ undoWritingProcess(signal);
+ }//if
+ if (tslPrevfree == ZEMPTYLIST) {
+ jam();
+ /* UPDATE FREE LIST OF LEFT CONTAINER IN PAGE HEAD */
+ tsllTmp1 = slPageptr.p->word32[ZPOS_EMPTY_LIST];
+ tsllTmp = tsllTmp1 & 0x7f;
+ tsllTmp1 = (tsllTmp1 >> 14) << 14;
+ tsllTmp1 = (tsllTmp1 | (tslNextfree << 7)) | tsllTmp;
+ dbgWord32(slPageptr, ZPOS_EMPTY_LIST, tsllTmp1);
+ slPageptr.p->word32[ZPOS_EMPTY_LIST] = tsllTmp1;
+ } else {
+ ndbrequire(tslPrevfree < ZEMPTYLIST);
+ jam();
+ tsllTmp = ((tslPrevfree << ZSHIFT_PLUS) - (tslPrevfree << ZSHIFT_MINUS)) + ZHEAD_SIZE;
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ cundoElemIndex = tsllTmp;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(slPageptr, tsllTmp, tslNextfree);
+ slPageptr.p->word32[tsllTmp] = tslNextfree;
+ }//if
+ if (tslNextfree < ZEMPTYLIST) {
+ jam();
+ tsllTmp = (((tslNextfree << ZSHIFT_PLUS) - (tslNextfree << ZSHIFT_MINUS)) + ZHEAD_SIZE) + 1;
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ cundoElemIndex = tsllTmp;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(slPageptr, tsllTmp, tslPrevfree);
+ slPageptr.p->word32[tsllTmp] = tslPrevfree;
+ } else {
+ ndbrequire(tslNextfree == ZEMPTYLIST);
+ jam();
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* IF WE ARE UPDATING THE HEADER WE ARE CREATING A NEW CONTAINER IN THE PAGE. */
+ /* TO BE ABLE TO FIND ALL LOCKED ELEMENTS WE KEEP ALL CONTAINERS IN LINKED */
+ /* LISTS IN THE PAGE. */
+ /* */
+ /* ZPOS_EMPTY_LIST CONTAINS A NEXT POINTER IN BIT 16-22 THAT REFERS TO THE */
+ /* FIRST CONTAINER IN A LIST OF USED RIGHT CONTAINERS IN THE PAGE. */
+ /* ZPOS_EMPTY_LIST CONTAINS A NEXT POINTER IN BIT 23-29 THAT REFERS TO THE */
+ /* FIRST CONTAINER IN A LIST OF USED LEFT CONTAINERS IN THE PAGE. */
+ /* EACH CONTAINER IN THE LIST CONTAINS A NEXT POINTER IN BIT 11-17 AND IT */
+ /* CONTAINS A PREVIOUS POINTER IN BIT 18-24. */
+ /* WE ALSO SET BIT 25 TO INDICATE THAT IT IS A CONTAINER HEADER. */
+ /* --------------------------------------------------------------------------------- */
+ if (tslUpdateHeader == ZTRUE) {
+ jam();
+ tslNextfree = (slPageptr.p->word32[ZPOS_EMPTY_LIST] >> 23) & 0x7f;
+ tsllNewHead = ZCON_HEAD_SIZE;
+ tsllNewHead = ((tsllNewHead << 8) + ZEMPTYLIST) + (1 << 7);
+ tsllNewHead = (tsllNewHead << 7) + tslNextfree;
+ tsllNewHead = tsllNewHead << 11;
+ dbgWord32(slPageptr, tsllHeadIndex, tsllNewHead);
+ slPageptr.p->word32[tsllHeadIndex] = tsllNewHead;
+ tsllTmp = slPageptr.p->word32[ZPOS_EMPTY_LIST] & 0xc07fffff;
+ tsllTmp = tsllTmp | (tslPageindex << 23);
+ dbgWord32(slPageptr, ZPOS_EMPTY_LIST, tsllTmp);
+ slPageptr.p->word32[ZPOS_EMPTY_LIST] = tsllTmp;
+ if (tslNextfree < ZEMPTYLIST) {
+ jam();
+ tsllTmp = ((tslNextfree << ZSHIFT_PLUS) - (tslNextfree << ZSHIFT_MINUS)) + ZHEAD_SIZE;
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ cundoElemIndex = tsllTmp;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ tsllTmp1 = slPageptr.p->word32[tsllTmp] & 0xfe03ffff;
+ tsllTmp1 = tsllTmp1 | (tslPageindex << 18);
+ dbgWord32(slPageptr, tsllTmp, tsllTmp1);
+ slPageptr.p->word32[tsllTmp] = tsllTmp1;
+ } else {
+ ndbrequire(tslNextfree == ZEMPTYLIST);
+ jam();
+ }//if
+ }//if
+ ilcPageptr.p = slPageptr.p;
+ increaselistcont(signal);
+}//Dbacc::seizeLeftlist()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_RIGHTLIST */
+/* DESCRIPTION: THE BUFFER NOTED BY TSL_PAGEINDEX WILL BE REMOVED FROM THE */
+/* LIST OF RIGHT FREE CONTAINER, IN THE HEADER OF THE PAGE */
+/* (SL_PAGEPTR). PREVIOUS AND NEXT BUFFER OF REMOVED BUFFER */
+/* WILL BE UPDATED. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeRightlist(Signal* signal)
+{
+ Uint32 tsrlTmp1;
+ Uint32 tsrlNewHead;
+ Uint32 tsrlHeadIndex;
+ Uint32 tsrlTmp;
+
+ tsrlHeadIndex = ((tslPageindex << ZSHIFT_PLUS) - (tslPageindex << ZSHIFT_MINUS)) + ((ZHEAD_SIZE + ZBUF_SIZE) - ZCON_HEAD_SIZE);
+ arrGuard(tsrlHeadIndex + 1, 2048);
+ tslNextfree = slPageptr.p->word32[tsrlHeadIndex];
+ tslPrevfree = slPageptr.p->word32[tsrlHeadIndex + 1];
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ datapageptr.p = slPageptr.p;
+ cundoElemIndex = tsrlHeadIndex;
+ cundoinfolength = 2;
+ undoWritingProcess(signal);
+ }//if
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ cundoElemIndex = ZPOS_EMPTY_LIST;
+ cundoinfolength = 2;
+ undoWritingProcess(signal);
+ }//if
+ if (tslPrevfree == ZEMPTYLIST) {
+ jam();
+ tsrlTmp = slPageptr.p->word32[ZPOS_EMPTY_LIST];
+ dbgWord32(slPageptr, ZPOS_EMPTY_LIST, ((tsrlTmp >> 7) << 7) | tslNextfree);
+ slPageptr.p->word32[ZPOS_EMPTY_LIST] = ((tsrlTmp >> 7) << 7) | tslNextfree;
+ } else {
+ ndbrequire(tslPrevfree < ZEMPTYLIST);
+ jam();
+ tsrlTmp = ((tslPrevfree << ZSHIFT_PLUS) - (tslPrevfree << ZSHIFT_MINUS)) + ((ZHEAD_SIZE + ZBUF_SIZE) - ZCON_HEAD_SIZE);
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ cundoElemIndex = tsrlTmp;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(slPageptr, tsrlTmp, tslNextfree);
+ slPageptr.p->word32[tsrlTmp] = tslNextfree;
+ }//if
+ if (tslNextfree < ZEMPTYLIST) {
+ jam();
+ tsrlTmp = ((tslNextfree << ZSHIFT_PLUS) - (tslNextfree << ZSHIFT_MINUS)) + ((ZHEAD_SIZE + ZBUF_SIZE) - (ZCON_HEAD_SIZE - 1));
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ cundoElemIndex = tsrlTmp;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(slPageptr, tsrlTmp, tslPrevfree);
+ slPageptr.p->word32[tsrlTmp] = tslPrevfree;
+ } else {
+ ndbrequire(tslNextfree == ZEMPTYLIST);
+ jam();
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* IF WE ARE UPDATING THE HEADER WE ARE CREATING A NEW CONTAINER IN THE PAGE. */
+ /* TO BE ABLE TO FIND ALL LOCKED ELEMENTS WE KEEP ALL CONTAINERS IN LINKED */
+ /* LISTS IN THE PAGE. */
+ /* */
+ /* ZPOS_EMPTY_LIST CONTAINS A NEXT POINTER IN BIT 16-22 THAT REFERS TO THE */
+ /* FIRST CONTAINER IN A LIST OF USED RIGHT CONTAINERS IN THE PAGE. */
+ /* ZPOS_EMPTY_LIST CONTAINS A NEXT POINTER IN BIT 23-29 THAT REFERS TO THE */
+ /* FIRST CONTAINER IN A LIST OF USED LEFT CONTAINERS IN THE PAGE. */
+ /* EACH CONTAINER IN THE LIST CONTAINS A NEXT POINTER IN BIT 11-17 AND IT */
+ /* CONTAINS A PREVIOUS POINTER IN BIT 18-24. */
+ /* --------------------------------------------------------------------------------- */
+ if (tslUpdateHeader == ZTRUE) {
+ jam();
+ tslNextfree = (slPageptr.p->word32[ZPOS_EMPTY_LIST] >> 16) & 0x7f;
+ tsrlNewHead = ZCON_HEAD_SIZE;
+ tsrlNewHead = ((tsrlNewHead << 8) + ZEMPTYLIST) + (1 << 7);
+ tsrlNewHead = (tsrlNewHead << 7) + tslNextfree;
+ tsrlNewHead = tsrlNewHead << 11;
+ dbgWord32(slPageptr, tsrlHeadIndex, tsrlNewHead);
+ slPageptr.p->word32[tsrlHeadIndex] = tsrlNewHead;
+ tsrlTmp = slPageptr.p->word32[ZPOS_EMPTY_LIST] & 0xff80ffff;
+ dbgWord32(slPageptr, ZPOS_EMPTY_LIST, tsrlTmp | (tslPageindex << 16));
+ slPageptr.p->word32[ZPOS_EMPTY_LIST] = tsrlTmp | (tslPageindex << 16);
+ if (tslNextfree < ZEMPTYLIST) {
+ jam();
+ tsrlTmp = ((tslNextfree << ZSHIFT_PLUS) - (tslNextfree << ZSHIFT_MINUS)) + ((ZHEAD_SIZE + ZBUF_SIZE) - ZCON_HEAD_SIZE);
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ cundoElemIndex = tsrlTmp;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ tsrlTmp1 = slPageptr.p->word32[tsrlTmp] & 0xfe03ffff;
+ dbgWord32(slPageptr, tsrlTmp, tsrlTmp1 | (tslPageindex << 18));
+ slPageptr.p->word32[tsrlTmp] = tsrlTmp1 | (tslPageindex << 18);
+ } else {
+ ndbrequire(tslNextfree == ZEMPTYLIST);
+ jam();
+ }//if
+ }//if
+ ilcPageptr.p = slPageptr.p;
+ increaselistcont(signal);
+}//Dbacc::seizeRightlist()
+
+
+//---------------------------------------------------------------------------------
+// ALLOC_SPECIFIC_LONG_OVERFLOW_PAGE
+//
+// DESCRIPTION: ALLOCATES A LONG OVER FLOW PAGE AND PUTS IT IN A SPECIFIED
+// DIRINDEX. THIS IS TO SUPPORT AN UNDO_DELETE AFTER AN
+// UNDO_INSERT ON THE SAME LONG KEY IN A LCP.
+// UNDO_INSERT ONLY HAVE A REFERENCE TO THE KEY AND TO MAKE
+// IT POSSIBLE TO DELETE THE KEY, THE REFERENCE MUST BE
+// ACCURATE, WHICH MEANS THE KEY MUST BE SAVED ON THE SAME
+// PLACE IT WAS DELETED FROM.
+//---------------------------------------------------------------------------------
+void Dbacc::allocSpecificLongOverflowPage(Signal* signal)
+{
+ DirRangePtr aloDirRangePtr;
+ DirectoryarrayPtr aloOverflowDirptr;
+
+ if ((cfirstfreepage == RNIL) &&
+ (cfreepage >= cpagesize)) {
+ jam();
+ zpagesize_error("Dbacc::allocSpecificLongOverflowPage");
+ tresult = ZPAGESIZE_ERROR;
+ return;
+ }
+
+ if ((cfirstfreedir == RNIL) &&
+ (cdirarraysize <= cdirmemory)) {
+ jam();
+ tresult = ZDIRSIZE_ERROR;
+ return;
+ }
+
+ tmpP = taslpDirIndex;
+ aloDirRangePtr.i = fragrecptr.p->overflowdir;
+ tmpP2 = tmpP >> 8;
+ tmpP = tmpP & 0xff;
+ ptrCheckGuard(aloDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(tmpP2, 256);
+
+ if (aloDirRangePtr.p->dirArray[tmpP2] == RNIL) {
+ jam();
+ seizeDirectory(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ sendSystemerror(signal);
+ return;
+ }
+ aloDirRangePtr.p->dirArray[tmpP2] = sdDirptr.i;
+ } else {
+ jam();
+ sdDirptr.i = RNIL;
+ ptrNull(sdDirptr);
+ }
+
+ aloOverflowDirptr.i = aloDirRangePtr.p->dirArray[tmpP2];
+ ptrCheckGuard(aloOverflowDirptr, cdirarraysize, directoryarray);
+ seizePage(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ sendSystemerror(signal);
+ return;
+ }//if
+
+ if (aloOverflowDirptr.p->pagep[tmpP] != RNIL) {
+ jam();
+ sendSystemerror(signal);
+ return;
+ }
+
+ aloOverflowDirptr.p->pagep[tmpP] = spPageptr.i;
+ iloPageptr.p = spPageptr.p;
+ iloPageptr.i = spPageptr.i;
+ tiloIndex = taslpDirIndex;
+ initLongOverpage(signal);
+ aslpPageptr.i = spPageptr.i;
+ aslpPageptr.p = spPageptr.p;
+}//Dbacc::allocSpecificLongOverflowPage
+
+/* --------------------------------------------------------------------------------- */
+/* ALLOC_LONG_OVERFLOW_PAGE */
+/* DESCRIPTION: */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::allocLongOverflowPage(Signal* signal)
+{
+ DirRangePtr aloDirRangePtr;
+ DirectoryarrayPtr aloOverflowDirptr;
+ OverflowRecordPtr aloOverflowRecPtr;
+ Uint32 taloIndex;
+
+ if ((cfirstfreepage == RNIL) &&
+ (cfreepage >= cpagesize)) {
+ jam();
+ zpagesize_error("Dbacc::allocLongOverflowPage");
+ tresult = ZPAGESIZE_ERROR;
+ return;
+ }//if
+ if ((cfirstfreedir == RNIL) &&
+ (cdirarraysize <= cdirmemory)) {
+ jam();
+ tresult = ZDIRSIZE_ERROR;
+ return;
+ }//if
+ if (fragrecptr.p->firstFreeDirindexRec != RNIL) {
+ jam();
+ aloOverflowRecPtr.i = fragrecptr.p->firstFreeDirindexRec;
+ ptrCheckGuard(aloOverflowRecPtr, coverflowrecsize, overflowRecord);
+ troOverflowRecPtr.p = aloOverflowRecPtr.p;
+ takeRecOutOfFreeOverdir(signal);
+ taloIndex = aloOverflowRecPtr.p->dirindex;
+ rorOverflowRecPtr = aloOverflowRecPtr;
+ releaseOverflowRec(signal);
+ } else {
+ jam();
+ taloIndex = fragrecptr.p->lastOverIndex;
+ fragrecptr.p->lastOverIndex++;
+ }//if
+ tmpP = taloIndex;
+ aloDirRangePtr.i = fragrecptr.p->overflowdir;
+ tmpP2 = tmpP >> 8;
+ tmpP = tmpP & 0xff;
+ ptrCheckGuard(aloDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(tmpP2, 256);
+ if (aloDirRangePtr.p->dirArray[tmpP2] == RNIL) {
+ jam();
+ seizeDirectory(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ aloDirRangePtr.p->dirArray[tmpP2] = sdDirptr.i;
+ } else {
+ jam();
+ sdDirptr.i = RNIL;
+ ptrNull(sdDirptr);
+ }//if
+ aloOverflowDirptr.i = aloDirRangePtr.p->dirArray[tmpP2];
+ ptrCheckGuard(aloOverflowDirptr, cdirarraysize, directoryarray);
+ seizePage(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ aloOverflowDirptr.p->pagep[tmpP] = spPageptr.i;
+ iloPageptr = spPageptr;
+ tiloIndex = taloIndex;
+ initLongOverpage(signal);
+ alpPageptr = spPageptr;
+ ipaPagePtr = spPageptr;
+ tipaArrayPos = 3;
+ insertPageArrayList(signal);
+}//Dbacc::allocLongOverflowPage()
+
+/* --------------------------------------------------------------------------------- */
+/* GET_LONG_KEY_PAGE */
+/* DESCRIPTION: SEARCH FOR A FREE OVERFLOW PAGE TO STORE A LONG KEY. */
+/* LONG_KEY_PAGE_PTR IS RETURNED. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::getLongKeyPage(Signal* signal)
+{
+ LongKeyPage *glkPage;
+
+ jam();
+
+ Uint32 tglkLongIndex = 0;
+
+ ndbrequire(operationRecPtr.p->tupkeylen <= ZWORDS_IN_PAGE - ZHEAD_SIZE);
+
+ // Do not look in longKeyPageArray[tglkLongIndex] where the pages are to small.
+ if(operationRecPtr.p->tupkeylen < 128) {
+ jam();
+ tglkLongIndex = 0;
+ } else {
+ jam();
+ tglkLongIndex = (operationRecPtr.p->tupkeylen - 128) / 512;
+ }//if
+
+ // Go through the longKeyPageArray and search for a page.
+ for (; tglkLongIndex <= ZMAX_LONG_KEY_ARRAY_INDEX; tglkLongIndex++) {
+ jam();
+ glkPageptr.i = fragrecptr.p->longKeyPageArray[tglkLongIndex];
+
+ if (glkPageptr.i != RNIL) {
+ // A page is found.
+ jam();
+ do {
+ ptrCheckGuard(glkPageptr, cpagesize, page8);
+ glkPage = (LongKeyPage *) &glkPageptr.p->word32[0];
+
+ // Check page if there is enough memory available. Accept only page
+ // with free_area > tupkeylen, this leaves at least one word for eventually
+ // an increase in the index area.
+ if (glkPage->header.freeArea > operationRecPtr.p->tupkeylen){
+ // The page found is OK
+ jam();
+ return;
+ } else {
+ // Not enough space in page, look in the next page if not RNIL,
+ // otherwise continue with for-loop.
+ jam();
+ glkPageptr.i = glkPage->header.nextPage;
+ }
+ }//do
+ while (glkPageptr.i != RNIL);
+ }//if
+ }//for
+
+ // No page with enough space was available, allocate a new page!
+ jam();
+ allocLongOverflowPage(signal);
+ glkPageptr = alpPageptr;
+}//Dbacc::getLongKeyPage()
+
+/* --------------------------------------------------------------------------------- */
+/* INIT_LONG_OVERPAGE */
+/* INPUT. ILO_PAGEPTR, POINTER TO AN OVERFLOW PAGE RECORD */
+/* DESCRIPTION: CONTAINERS AND FREE LISTS OF THE PAGE, GET INITIALE VALUE */
+/* ACCORDING TO LH3 AND PAGE STRUCTOR DISACRIPTION OF NDBACC BLOCK */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initLongOverpage(Signal* signal)
+{
+ iloPageptr.p->word32[ZPOS_PAGE_ID] = tiloIndex;
+ iloPageptr.p->word32[ZPOS_PAGE_TYPE] = ZLONG_PAGE_TYPE << ZPOS_PAGE_TYPE_BIT;
+ iloPageptr.p->word32[ZPOS_NO_ELEM_IN_PAGE] = 0;
+ iloPageptr.p->word32[ZPOS_OVERFLOWREC] = RNIL;
+ iloPageptr.p->word32[ZPOS_FREE_AREA_IN_PAGE] = ZWORDS_IN_PAGE - ZHEAD_SIZE;
+ iloPageptr.p->word32[ZPOS_LAST_INDEX] = 0;
+ iloPageptr.p->word32[ZPOS_INSERT_INDEX] = ZHEAD_SIZE;
+ iloPageptr.p->word32[ZPOS_ARRAY_POS] = ZDEFAULT_LIST;
+ iloPageptr.p->word32[ZPOS_NEXT_FREE_INDEX] = 0;
+ iloPageptr.p->word32[ZPOS_NEXT_PAGE] = RNIL;
+ iloPageptr.p->word32[ZPOS_PREV_PAGE] = RNIL;
+ iloPageptr.p->word32[12] = 0;
+ iloPageptr.p->word32[13] = 0;
+ iloPageptr.p->word32[14] = 0;
+ iloPageptr.p->word32[15] = 0;
+ // Initialize free indexes
+ for (int i = 1; i < (ZWORDS_IN_PAGE - ZHEAD_SIZE); i++)
+ iloPageptr.p->word32[ZWORDS_IN_PAGE - i] = i + 1;
+}//Dbacc::initLongOverpage()
+
+//---------------------------------------------------------------------------------
+// STORE_LONG_KEYS_AT_POS
+//
+// INPUT: SLKAP_PAGEPTR
+// SLKAP_COPY_PAGEPTR
+// TSLKAP_KEY_LEN
+// TSLKAP_PAGE_INDEX
+//
+// DESCRIPTION: A LONG ELEMENT IS STORED ON A LONG_KEY_PAGE AT A
+// SPECIFIC POSITION. THIS FUNCTION IS USED BY UNDO_DELETE.
+//---------------------------------------------------------------------------------
+void Dbacc::storeLongKeysAtPos(Signal* signal)
+{
+ Uint32 tslkapHighestIndex;
+ Uint32 tslkapLastSize;
+ Uint32 tslkapInsertIndex;
+ Uint32 tslkapIndexIncreaseSize;
+ Uint32 tslkapTmp;
+
+ LongKeyPage *slkapPage;
+
+ jam();
+ slkapPage = (LongKeyPage *) &slkapPageptr.p->word32[0];
+
+#ifdef VM_TRACE
+ checkIndexInLongKeyPage(slkapPageptr.i, "storeLongKeysAtPos");
+#endif
+
+ // if (csystemRestart != ZTRUE) {
+ if (cundoLogActive != ZTRUE) {
+ //-------------------------------------------------------------
+ // This function is only allowed to be called during
+ // undolog execution.
+ //-------------------------------------------------------------
+ jam();
+ sendSystemerror(signal);
+ return;
+ }
+
+ if (slkapPage->word32[ZWORDS_IN_PAGE - tslkapPageIndex] >> 16 != 0 ) {
+ //-------------------------------------------------------------
+ // The index should be empty, we have a serious problem.
+ //-------------------------------------------------------------
+ jam();
+ sendSystemerror(signal);
+ return;
+ }
+
+ //-------------------------------------------------------------
+ // Calculate some variables to use later.
+ //-------------------------------------------------------------
+ tslkapHighestIndex = slkapPage->header.highestIndex;
+ tslkapPageIndex > tslkapHighestIndex ?
+ tslkapIndexIncreaseSize = tslkapPageIndex - tslkapHighestIndex :
+ tslkapIndexIncreaseSize = 0;
+
+ slkapPage->header.highestIndex += tslkapIndexIncreaseSize;
+
+ if ((slkapPage->header.freeArea - tslkapIndexIncreaseSize)
+ < tslkapKeyLen) {
+ //-------------------------------------------------------------
+ // Not enough area in the page, a serious problem.
+ //-------------------------------------------------------------
+ jam();
+ sendSystemerror(signal);
+ return;
+ }
+
+ //-------------------------------------------------------------
+ // Fix the free index list. We might put in a key in the
+ // middle of the list, so we must fix the free list and the
+ // free index pointers.
+ //-------------------------------------------------------------
+ slkapPage->header.nextFreeIndex = 0;
+
+ for (Uint32 i = tslkapHighestIndex + tslkapIndexIncreaseSize; i > 0; i--) {
+ if (i == tslkapPageIndex) {
+ // The key index shall not be in the free list.
+ continue;
+ }
+
+ if (slkapPage->word32[ZWORDS_IN_PAGE - i] >> 16 == 0 ) {
+ // Go through all empty indexes.
+ slkapPage->word32[ZWORDS_IN_PAGE - i] = slkapPage->header.nextFreeIndex;
+ arrGuard(i, 2048);
+ slkapPage->header.nextFreeIndex = i;
+ }
+ }
+
+ //-------------------------------------------------------------
+ // Decrement the free area in page according to the above
+ // increase in index size.
+ //-------------------------------------------------------------
+ slkapPage->header.freeArea -= tslkapIndexIncreaseSize;
+
+ tslkapLastSize = ZWORDS_IN_PAGE - slkapPage->header.highestIndex
+ - slkapPage->header.insertPos;
+
+ //-------------------------------------------------------------
+ // Check if we have to reorganize the page.
+ //-------------------------------------------------------------
+ if (tslkapLastSize >= tslkapKeyLen) {
+ jam();
+ } else {
+ jam();
+ relpPageptr.p = slkapPageptr.p;
+ reorgLongPage(signal);
+ }
+
+ //-------------------------------------------------------------
+ // Insert the key and update page attributes.
+ //-------------------------------------------------------------
+ jam();
+ // Increase the number of element in the page.
+ slkapPage->header.noOfElements++;
+ jam();
+ // Put in the key reference into the index. The reference
+ // consists of key length and insert position.
+ arrGuard(ZWORDS_IN_PAGE - tslkapPageIndex, 2048);
+ slkapPage->word32[ZWORDS_IN_PAGE - tslkapPageIndex] =
+ slkapPage->header.insertPos | (tslkapKeyLen << 16);
+ jam();
+ // Increase the key insert position.
+ tslkapInsertIndex = slkapPage->header.insertPos;
+ slkapPage->header.insertPos += tslkapKeyLen;
+ jam();
+ // Decrease the free area.
+ slkapPage->header.freeArea -= tslkapKeyLen;
+ jam();
+
+ // Update pageArrayPos. insertPageArrayList() called from execACC_OVER_REC
+ // needs this value.
+ if (slkapPage->header.freeArea < 128) {
+ jam();
+ slkapPage->header.pageArrayPos = 4;
+ } else {
+ jam();
+ slkapPage->header.pageArrayPos = (slkapPage->header.freeArea - 128) / 512;
+ }//if
+
+ // Store the actual key at the insert position.
+ Uint32 guard27 = tslkapKeyLen - 1;
+ arrGuard(guard27 + tslkapInsertIndex, 2048);
+ for (tslkapTmp = 0; tslkapTmp <= guard27; tslkapTmp++) {
+ jam();
+ slkapPage->word32[tslkapTmp + tslkapInsertIndex] = slkapCopyPageptr.p->word32[tslkapTmp];
+ }//for
+}//Dbacc::storeLongKeysAtPos
+
+/* --------------------------------------------------------------------------------- */
+/* STORE_LONG_KEYS */
+/* INPUT: SLK_PAGEPTR */
+/* SLK_COPY_PAGEPTR */
+/* TSLK_KEY_LEN */
+/* OUTPUT: TSLK_PAGE_INDEX */
+/* */
+/* DESCRIPTION: A LONG ELEMENT IS STORED ON A LONG_KEY_PAGE. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::storeLongKeys(Signal* signal)
+{
+ Uint32 tslkLastSize;
+ Uint32 tslkInsertIndex;
+ Uint32 tslkArrayPos;
+ Uint32 tslkTmp;
+ Uint32 guard27;
+ LongKeyPage *slkPage;
+
+ jam();
+ slkPage = (LongKeyPage *) &slkPageptr.p->word32[0];
+
+#ifdef VM_TRACE
+ checkIndexInLongKeyPage(slkPageptr.i, "storeLongKeys1");
+#endif
+
+ // Accept only page with free_area > tupkeylen, this leaves at least
+ // one word for eventually an increase in the index area.
+ ndbrequire(slkPage->header.freeArea > tslkKeyLen);
+
+ dbgWord32(slkPageptr, ZPOS_LAST_INDEX, slkPage->header.highestIndex);
+ dbgWord32(slkPageptr, ZPOS_INSERT_INDEX, slkPage->header.insertPos);
+
+ tslkLastSize = ZWORDS_IN_PAGE - slkPage->header.highestIndex - slkPage->header.insertPos;
+
+ if (tslkLastSize > operationRecPtr.p->tupkeylen) {
+ // WE DO NOT NEED TO REORGANIZE THE PAGE TO INSERT THE NEW KEY. IT FITS INTO THE
+ // SIZE REMAINING AT THE END.
+ jam();
+ } else {
+ // THE KEY FITS INTO THE PAGE BUT ONLY AFTER REORGANISING THE PAGE.
+ jam();
+ relpPageptr.p = slkPageptr.p;
+ reorgLongPage(signal);
+ }//if
+
+ if (slkPage->header.nextFreeIndex == 0) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE PAGE INDEX HAS NO EMPTY SLOTS. WE MUST EXTEND THE PAGE INDEX BY ONE NEW SLOT.*/
+ /* --------------------------------------------------------------------------------- */
+ tslkPageIndex = slkPage->header.highestIndex + 1;
+ } else {
+ jam();
+ tslkPageIndex = slkPage->header.nextFreeIndex;
+ }//if
+
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* ON LONG PAGES WE USE A PHYSIOLOGICAL LOGGING SCHEME. THIS MEANS THAT WE ONLY NEED*/
+ /* TO SPECIFY WHICH INDEX TO DELETE IN ORDER TO UNDO THE CHANGES WE DO. THE */
+ /* POSSIBLE REORGANISATION DO NOT CHANGE THE LOGICAL LAYOUT OF THE PAGE. */
+ /* --------------------------------------------------------------------------------- */
+ datapageptr.p = slkPageptr.p;
+ cundoElemIndex = tslkPageIndex;
+ cundoinfolength = 0;
+ undoWritingProcess(signal);
+ }//if
+
+ if (slkPage->header.nextFreeIndex == 0) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE PAGE INDEX HAS NO EMPTY SLOTS. WE MUST EXTEND THE PAGE INDEX BY ONE NEW SLOT.*/
+ /* --------------------------------------------------------------------------------- */
+ dbgWord32(slkPageptr, ZPOS_LAST_INDEX, slkPage->header.highestIndex + 1);
+ slkPage->header.highestIndex++;
+ ndbrequire(slkPage->header.insertPos < (ZWORDS_IN_PAGE - slkPage->header.highestIndex));
+ // Reset index. We have already checked that we can increase "highestIndex" value
+ // without overwriting the data part.
+ slkPage->word32[ZWORDS_IN_PAGE - slkPage->header.highestIndex] = 0;
+ dbgWord32(slkPageptr, ZPOS_FREE_AREA_IN_PAGE, slkPage->header.freeArea - 1);
+ slkPage->header.freeArea--;
+ } else {
+ jam();
+ dbgWord32(slkPageptr, ZPOS_NEXT_FREE_INDEX, slkPage->word32[ZWORDS_IN_PAGE - tslkPageIndex]);
+ arrGuard(ZWORDS_IN_PAGE - tslkPageIndex, 2048);
+ arrGuard(slkPage->word32[ZWORDS_IN_PAGE - tslkPageIndex], 2048);
+
+ slkPage->header.nextFreeIndex = slkPage->word32[ZWORDS_IN_PAGE - tslkPageIndex];
+ if(slkPage->header.nextFreeIndex > slkPage->header.highestIndex){
+ slkPage->header.nextFreeIndex = 0;
+ dbgWord32(slkPageptr, ZPOS_NEXT_FREE_INDEX, slkPage->header.nextFreeIndex);
+ }
+ }//if
+
+ dbgWord32(slkPageptr, ZWORDS_IN_PAGE - tslkPageIndex, tslkKeyLen);
+ dbgWord32(slkPageptr, ZWORDS_IN_PAGE - tslkPageIndex, slkPage->header.insertPos);
+ arrGuard(ZWORDS_IN_PAGE - tslkPageIndex, 2048);
+ slkPage->word32[ZWORDS_IN_PAGE - tslkPageIndex] =
+ slkPage->header.insertPos | (tslkKeyLen << 16);
+
+ dbgWord32(slkPageptr, ZPOS_INSERT_INDEX, slkPage->header.insertPos);
+ tslkInsertIndex = slkPage->header.insertPos;
+ slkPage->header.insertPos += tslkKeyLen;
+
+ dbgWord32(slkPageptr, ZPOS_FREE_AREA_IN_PAGE, slkPage->header.freeArea - tslkKeyLen);
+ slkPage->header.freeArea = slkPage->header.freeArea - tslkKeyLen;
+ if (slkPage->header.freeArea < 128) {
+ jam();
+ tslkArrayPos = 4;
+ } else {
+ jam();
+ tslkArrayPos = (slkPage->header.freeArea - 128) / 512;
+ }//if
+
+ if (tslkArrayPos != slkPage->header.pageArrayPos) {
+ jam();
+ if (cundoLogActive != ZTRUE) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE ONLY HANDLE THE LISTS WHEN WE ARE NOT IN A SYSTEM RESTART. */
+ /* --------------------------------------------------------------------------------- */
+ rfpPageptr = slkPageptr;
+ trfpArrayPos = slkPage->header.pageArrayPos;
+ removeFromPageArrayList(signal);
+ ipaPagePtr = slkPageptr;
+ tipaArrayPos = tslkArrayPos;
+ slkPage->header.pageArrayPos = tipaArrayPos;
+ if (tslkArrayPos != 4) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE PAGE WILL STILL BE ON ONE OF THE FREE LISTS SINCE AT LEAST 128 * 4 */
+ /* BYTES OF FREE SPACE REMAINS ON THE PAGE. */
+ /* --------------------------------------------------------------------------------- */
+ insertPageArrayList(signal);
+ }//if
+ } else {
+ // This should never happen. Should use storeLongKeysAtPos() instead when executing
+ // undolog.
+ ndbrequire(false);
+ }
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* INCREASE THE NUMBER OF ELEMENTS IN THE PAGE. */
+ /* --------------------------------------------------------------------------------- */
+ dbgWord32(slkPageptr, ZPOS_NO_ELEM_IN_PAGE, slkPage->header.noOfElements + 1);
+ slkPage->header.noOfElements++;
+
+ guard27 = tslkKeyLen - 1;
+ arrGuard(guard27 + tslkInsertIndex, 2048);
+ for (tslkTmp = 0; tslkTmp <= guard27; tslkTmp++) {
+ dbgWord32(slkPageptr, tslkTmp + tslkInsertIndex, slkCopyPageptr.p->word32[tslkTmp]);
+ slkPage->word32[tslkTmp + tslkInsertIndex] = slkCopyPageptr.p->word32[tslkTmp];
+ }//for
+
+ // Used by abortoperation() in case of an abort.
+ operationRecPtr.p->longPagePtr = slkPageptr.i;
+
+ // This is for an eventual LCP start in the middle of this locked operation.
+ operationRecPtr.p->longKeyPageIndex = tslkPageIndex;
+
+#ifdef VM_TRACE
+ if (cundoLogActive != ZTRUE) checkPageArrayList(signal, "storeLongKeys");
+ checkIndexInLongKeyPage(slkPageptr.i, "storeLongKeys2");
+#endif
+
+}//Dbacc::storeLongKeys()
+
+/* --------------------------------------------------------------------------------- */
+/* REORGANIZE THE PAGE BY COPYING IT TEMPORARILY TO A NEW AREA AND THEN SIMPLY */
+/* PUTTING THE OBJECTS BACK ON THE PAGE IN THE SAME ORDER AS THEY ARE PLACED IN THE */
+/* INDEX. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::reorgLongPage(Signal* signal)
+{
+ Uint32 indexStartPos;
+ Uint32 pagePos;
+ Uint32 pagePos2;
+ Uint32 indexNo;
+ Uint32 insertPos;
+ Uint32 indexValue;
+ Uint32 keyLength;
+ Uint32 keyPos;
+ Uint32 keyEndPos;
+ LongKeyPage *reOrgPage;
+
+ ptrGuard(relpPageptr);
+ reOrgPage = (LongKeyPage *) &relpPageptr.p->word32[0];
+
+ dbgWord32(relpPageptr, ZPOS_LAST_INDEX, reOrgPage->header.highestIndex);
+ indexStartPos = ZWORDS_IN_PAGE - reOrgPage->header.highestIndex;
+
+ // Copy key data part of page to a temporary page.
+ for (pagePos = ZHEAD_SIZE; pagePos < indexStartPos; pagePos++) {
+ jam();
+ arrGuard(pagePos, 2048);
+ ckeys[pagePos] = reOrgPage->word32[pagePos];
+ }//for
+
+ insertPos = ZHEAD_SIZE;
+
+ // Walk through all the indexes.
+ for (indexNo = 1; indexNo <= reOrgPage->header.highestIndex; indexNo++) {
+ jam();
+ arrGuard(ZWORDS_IN_PAGE - indexNo, 2048);
+ dbgWord32(relpPageptr, ZWORDS_IN_PAGE - indexNo, reOrgPage->word32[ZWORDS_IN_PAGE - indexNo]);
+ indexValue = reOrgPage->word32[ZWORDS_IN_PAGE - indexNo];
+
+ if ((indexValue >> 16) != 0) {
+ // The index contains a reference to a key.
+ jam();
+ keyPos = indexValue & 0xffff;
+ keyLength = indexValue >> 16;
+ dbgWord32(relpPageptr, ZWORDS_IN_PAGE - indexNo, insertPos + (keyLength << 16));
+ arrGuard(ZWORDS_IN_PAGE - indexNo, 2048);
+
+ // Refresh the index data with the new key start position in the data part.
+ reOrgPage->word32[ZWORDS_IN_PAGE - indexNo] = insertPos + (keyLength << 16);
+ keyEndPos = keyPos + keyLength;
+ arrGuard(keyEndPos, 2048);
+
+ // Copy the key from the temporary page
+ // to the insert position at original page.
+ for (pagePos2 = keyPos; pagePos2 < keyEndPos; pagePos2++, insertPos++) {
+ jam();
+ dbgWord32(relpPageptr, insertPos, ckeys[pagePos2]);
+ arrGuard(insertPos, 2048);
+ arrGuard(pagePos2, 2048);
+ reOrgPage->word32[insertPos] = ckeys[pagePos2];
+ }//for
+ }//if
+ }//for
+ dbgWord32(relpPageptr, ZPOS_INSERT_INDEX, insertPos);
+ reOrgPage->header.insertPos = insertPos;
+}//Dbacc::reorgLongPage()
+
+
+/* --------------------------------------------------------------------------------- */
+/* DELETE_LONG_KEY */
+/* INPUT: DLK_PAGEPTR PAGE POINTER OF DELETED KEY OBJECT */
+/* TDLK_LOGICAL_PAGE_INDEX LOGICAL PAGE INDEX OF DELETED KEY OBJECT */
+/* */
+/* DESCRIPTION: DELETE AN ELEMENT OF A LONG_KEY_PAGE. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::deleteLongKey(Signal* signal)
+{
+ Uint32 tdlkLastIndex;
+ Uint32 tdlkNextPosition;
+ Uint32 tdlkFreeArea;
+ Uint32 tdlkArrayPos;
+ Uint32 tdlkOldArrayPos;
+ LongKeyPage *dlkPage;
+
+ jam();
+ dlkPage = (LongKeyPage *) &dlkPageptr.p->word32[0];
+
+#ifdef VM_TRACE
+ checkIndexInLongKeyPage(dlkPageptr.i, "deleteLongKey1");
+#endif
+
+ dbgWord32(dlkPageptr, ZWORDS_IN_PAGE - tdlkLogicalPageIndex, dlkPage->word32[ZWORDS_IN_PAGE - tdlkLogicalPageIndex] >> 16);
+ dbgWord32(dlkPageptr, ZWORDS_IN_PAGE - tdlkLogicalPageIndex, dlkPage->word32[ZWORDS_IN_PAGE - tdlkLogicalPageIndex] & 0xffff);
+ arrGuard(ZWORDS_IN_PAGE - tdlkLogicalPageIndex, 2048);
+
+ const Uint32 tdlkIndexValue = dlkPage->word32[ZWORDS_IN_PAGE - tdlkLogicalPageIndex];
+ const Uint32 tdlkKeyLen = tdlkIndexValue >> 16;
+ const Uint32 tdlkPhysPageIndex = tdlkIndexValue & 0xffff;
+
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE LOG THE DELETE LONG KEY BY LOGGING THE DELETED KEY AND ITS LOGICAL INDEX.*/
+ /* --------------------------------------------------------------------------------- */
+ datapageptr.p = dlkPageptr.p;
+ cundoElemIndex = tdlkLogicalPageIndex;
+ cundoinfolength = tdlkKeyLen;
+ undoWritingProcess(signal);
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* DECREASE THE NUMBER OF ELEMENTS IN THE PAGE. */
+ /* --------------------------------------------------------------------------------- */
+ dbgWord32(dlkPageptr, ZPOS_NO_ELEM_IN_PAGE, dlkPage->header.noOfElements - 1);
+ dlkPage->header.noOfElements--;
+
+ arrGuard(dlkPage->header.noOfElements, ZMAX_NO_OF_LONGKEYS_IN_PAGE);
+
+ /* --------------------------------------------------------------------------------- */
+ /* INCREASE THE FREE AREA IN THE PAGE. */
+ /* --------------------------------------------------------------------------------- */
+ dbgWord32(dlkPageptr, ZPOS_FREE_AREA_IN_PAGE, dlkPage->header.freeArea + tdlkKeyLen);
+ dbgWord32(dlkPageptr, ZPOS_LAST_INDEX, dlkPage->header.highestIndex);
+
+ dlkPage->header.freeArea += tdlkKeyLen;
+
+ if (dlkPage->header.noOfElements == 0) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE PAGE IS NOW EMPTY, WE CAN RELEASE IT. */
+ /* --------------------------------------------------------------------------------- */
+ if (dlkPage->header.freeArea !=
+ (ZWORDS_IN_PAGE - ZHEAD_SIZE - dlkPage->header.highestIndex )) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* SOME AREA IN THE PAGE IS STILL LEFT BUT NO ELEMENTS, INCONSISTENT */
+ /* --------------------------------------------------------------------------------- */
+ sendSystemerror(signal);
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* WE REMOVE THE PAGE FROM THE LIST OF FREE LONG PAGES. THERE IS NO RISK THAT IT */
+ /* DID NOT BELONG TO ANY SINCE IT IS NOT ALLOWED TO HAVE THAT LARGE KEYS. */
+ /* --------------------------------------------------------------------------------- */
+
+ if (cundoLogActive != ZTRUE) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WHEN DELETING KEYS DURING SYSTEM RESTART WE NEED NOT UPDATE THE LISTS. */
+ /* --------------------------------------------------------------------------------- */
+ // REMOVEFROMLIST is done by releaseLongPage(). EDTJAMO.
+ // rfpPageptr = dlkPageptr;
+ // trfpArrayPos = dlkPage->header.pageArrayPos;
+ // removeFromPageArrayList(signal, "deleteLongKey");
+ rlopPageptr = dlkPageptr;
+ releaseLongPage(signal);
+ return;
+ } else {
+ // Must remove reference to the removed key, otherwise left in index. EDTJAMO.
+ arrGuard(ZWORDS_IN_PAGE - tdlkLogicalPageIndex, 2048);
+ arrGuard(tdlkLogicalPageIndex, 2048);
+
+ tdlkNextPosition = dlkPage->header.nextFreeIndex;
+ dlkPage->header.nextFreeIndex = tdlkLogicalPageIndex;
+ dbgWord32(dlkPageptr, ZWORDS_IN_PAGE - tdlkLogicalPageIndex, tdlkNextPosition);
+ dlkPage->word32[ZWORDS_IN_PAGE - tdlkLogicalPageIndex] = tdlkNextPosition;
+ }
+ } else {
+ /* --------------------------------------------------------------------------------- */
+ /* THE PAGE IS NOT EMPTY SO WE WILL REMOVE THE KEY OBJECT AND UPDATE THE */
+ /* HEADER INFORMATION AND PLACE THE PAGE IN THE PROPER PAGE LIST. */
+ /* --------------------------------------------------------------------------------- */
+ tdlkLastIndex = dlkPage->header.highestIndex;
+ arrGuard(ZWORDS_IN_PAGE - tdlkLastIndex, 2048);
+ if (tdlkLastIndex == tdlkLogicalPageIndex) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE DELETE THE LAST PAGE INDEX SO WE NEED TO UPDATE THE VALUE. WE MOVE */
+ /* BACKWARDS UNTIL WE EITHER FIND A USED INDEX OR THAT WE COME TO INDEX ZERO. */
+ /* --------------------------------------------------------------------------------- */
+ tdlkLastIndex--;
+ while( (tdlkLastIndex > 1) &&
+ (dlkPage->word32[ZWORDS_IN_PAGE - tdlkLastIndex] >> 16) == 0 ) {
+ jam();
+ tdlkLastIndex--;
+ }
+ //-----------------------------------------------------
+ // Reorganize the rest of the index. Set up the free
+ // list and the free index.
+ //-----------------------------------------------------
+ UintR dlkTmp = tdlkLastIndex;
+ dlkPage->header.nextFreeIndex = 0;
+ while( dlkTmp > 0) {
+ if ( (dlkPage->word32[ZWORDS_IN_PAGE - dlkTmp] >> 16) == 0 ) {
+ jam();
+ dlkPage->word32[ZWORDS_IN_PAGE - dlkTmp] = dlkPage->header.nextFreeIndex;
+ arrGuard(dlkTmp, 2048);
+ dlkPage->header.nextFreeIndex = dlkTmp;
+ }
+ dlkTmp--;
+ }
+ //-----------------------------------------------------
+ // Update free area in page and last index.
+ //-----------------------------------------------------
+ dbgWord32(dlkPageptr, ZPOS_LAST_INDEX, tdlkLastIndex);
+ dlkPage->header.highestIndex = tdlkLastIndex;
+ dlkPage->header.freeArea = tdlkLogicalPageIndex +
+ dlkPage->header.freeArea - tdlkLastIndex;
+ tdlkNextPosition = 0;
+ } else {
+ if (dlkPage->header.highestIndex > tdlkLogicalPageIndex) {
+ jam();
+ tdlkNextPosition = dlkPage->header.nextFreeIndex;
+ dbgWord32(dlkPageptr, ZPOS_NEXT_FREE_INDEX, tdlkLogicalPageIndex);
+ arrGuard(tdlkLogicalPageIndex, 2048);
+ dlkPage->header.nextFreeIndex = tdlkLogicalPageIndex;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* LOGICAL PAGE INDEX LARGER THAN LARGEST INDEX, INCONSISTENT. */
+ /* --------------------------------------------------------------------------------- */
+ sendSystemerror(signal);
+ return; // Just to keep compiler happy
+ }//if
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* WE INSERT ZERO INTO THE LENGTH PART TO INDICATE A FREE INDEX POSITION. */
+ /* WE INSERT A POINTER TO THE NEXT FREE INDEX TO AS TO PUT IT INTO A FREE */
+ /* LIST OF INDEX POSITIONS. WE ONLY DO SO IF IT WAS NOT THE LAST INDEX. */
+ /* --------------------------------------------------------------------------------- */
+ dbgWord32(dlkPageptr, ZWORDS_IN_PAGE - tdlkLogicalPageIndex, tdlkNextPosition);
+ arrGuard(ZWORDS_IN_PAGE - tdlkLogicalPageIndex, 2048);
+ dlkPage->word32[ZWORDS_IN_PAGE - tdlkLogicalPageIndex] = tdlkNextPosition;
+ if (dlkPage->header.insertPos == (tdlkPhysPageIndex + tdlkKeyLen)) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THIS ENTRY IS THE LAST ON THE PAGE SO WE WILL UPDATE THE INSERT INDEX */
+ /* --------------------------------------------------------------------------------- */
+ dbgWord32(dlkPageptr, ZPOS_INSERT_INDEX, tdlkPhysPageIndex);
+ dlkPage->header.insertPos = tdlkPhysPageIndex;
+ }//if
+ }//if
+ dbgWord32(dlkPageptr, ZPOS_FREE_AREA_IN_PAGE, dlkPage->header.freeArea);
+ tdlkFreeArea = dlkPage->header.freeArea;
+ ndbrequire(tdlkFreeArea <= (ZWORDS_IN_PAGE - ZHEAD_SIZE));
+ if (tdlkFreeArea < 128) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* FREE AREA IS STILL LESS THAN 128 WORDS SO IT SHOULD NOT BE PLACED IN ANY OF THE */
+ /* FREE LISTS. */
+ /* --------------------------------------------------------------------------------- */
+ dbgWord32(dlkPageptr, ZPOS_ARRAY_POS, dlkPage->header.pageArrayPos);
+ ndbrequire(dlkPage->header.pageArrayPos == 4);
+ } else {
+ jam();
+ // Calculate an eventually new arraypos.
+ dbgWord32(dlkPageptr, 0, (tdlkFreeArea - 128) / 512);
+ tdlkArrayPos = (tdlkFreeArea - 128) / 512;
+
+ if (cundoLogActive != ZTRUE) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WHEN DELETING KEYS DURING SYSTEM RESTART WE NEED NOT UPDATE THE LISTS. */
+ /* --------------------------------------------------------------------------------- */
+ dbgWord32(dlkPageptr, ZPOS_ARRAY_POS, dlkPage->header.pageArrayPos);
+ tdlkOldArrayPos = dlkPage->header.pageArrayPos;
+ if (tdlkArrayPos != tdlkOldArrayPos) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE NEW MEMORY AREA HAS ENABLED THE PAGE TO MOVE TO A NEW FREE PAGE LIST */
+ /* --------------------------------------------------------------------------------- */
+ rfpPageptr = dlkPageptr;
+ trfpArrayPos = tdlkOldArrayPos;
+ if (tdlkOldArrayPos != 4) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THERE WAS A FREE PAGE LIST TO REMOVE THE PAGE FROM. IF FREE SPACE IS LESS THAN */
+ /* 128 BYTES THEN IT IS NOT ON ANY FREE LIST. */
+ /* --------------------------------------------------------------------------------- */
+ removeFromPageArrayList(signal);
+ }//if
+ dlkPage->header.pageArrayPos = tdlkArrayPos;
+ ipaPagePtr = dlkPageptr;
+ tipaArrayPos = tdlkArrayPos;
+ insertPageArrayList(signal);
+ }//if
+ } else {
+ // Update pageArrayPos. We are in a SR, executing undolog, insertPageArrayList() called
+ // from execACC_OVER_REC needs this value later.
+ dlkPage->header.pageArrayPos = tdlkArrayPos;
+ }
+ }//if
+#ifdef VM_TRACE
+ if (cundoLogActive != ZTRUE) checkPageArrayList(signal, "deleteLongKey");
+ checkIndexInLongKeyPage(dlkPageptr.i, "deleteLongKey2");
+#endif
+}//Dbacc::deleteLongKey()
+
+
+void Dbacc::checkIndexInLongKeyPage(Uint32 pageId, char *calledFrom) {
+ Page8Ptr pagePtr;
+ LongKeyPage *page;
+ Uint32 indexNo;
+ Uint32 indexValue;
+ Uint32 keyLength;
+ Uint32 keyPos;
+
+ pagePtr.i = pageId;
+ ptrCheckGuard(pagePtr, cpagesize, page8);
+ page = (LongKeyPage *) &pagePtr.p->word32[0];
+
+ // Check the header variables.
+ if (page->header.nextFreeIndex > 2048 ||
+ page->header.highestIndex > 2048 ||
+ page->header.insertPos > 2048 ||
+ page->header.freeArea > 2048 ||
+ page->header.noOfElements > 225) {
+ ndbout << " ERROR in checkIndexInLongKeyPage, called from " << calledFrom << endl
+ << " pagePtr.i = " << pageId << endl;
+ printoutInfoAndShutdown(page);
+ }
+
+ // Walk through all the indexes.
+ for (indexNo = 1; indexNo <= page->header.highestIndex; indexNo++) {
+ jam();
+ indexValue = page->word32[ZWORDS_IN_PAGE - indexNo];
+
+ if ((indexValue >> 16) == 0) {
+ ; // key length is 0, means no key reference at this position in index.
+ } else {
+ // The index contains a reference to a key.
+ jam();
+ keyPos = indexValue & 0xffff;
+ keyLength = indexValue >> 16;
+ if (keyPos >= ZWORDS_IN_PAGE || keyLength >= ZWORDS_IN_PAGE) {
+ jam();
+ ndbout << " ERROR in checkIndexInLongKeyPage, called from " << calledFrom << endl
+ << " keyPos = " << keyPos << endl
+ << " keyLength = " << keyLength << endl
+ << " page->header.noOfElements = " << page->header.noOfElements << endl
+ << " page->header.freeArea = " << page->header.freeArea << endl
+ << " indexNo = " << indexNo << endl
+ << " page->header.highestIndex = " << page->header.highestIndex << endl;
+ ndbrequire(false);
+ }
+ }
+ }
+}//Dbacc::checkIndexInLongKeyPage
+
+
+/* --------------------------------------------------------------------------------- */
+/* REMOVE A PAGE FROM THE PAGE ARRAY LIST. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::removeFromPageArrayList(Signal* signal)
+{
+ Page8Ptr rfpPrevPageptr;
+ Page8Ptr rfpNextPageptr;
+ LongKeyPage *page;
+ LongKeyPage *prevPage;
+ LongKeyPage *nextPage;
+
+ jam();
+
+#ifdef VM_TRACE
+ checkPageB4Remove(rfpPageptr.i, "removeFromPageArrayList");
+#endif
+
+ page = (LongKeyPage *) &rfpPageptr.p->word32[0];
+
+ if (page->header.prevPage == RNIL) {
+ jam();
+ arrGuard(trfpArrayPos, 4);
+ // This page was first in list, remove reference
+ // to this page from the start of the list.
+ ndbrequire(fragrecptr.p->longKeyPageArray[trfpArrayPos] == rfpPageptr.i);
+ fragrecptr.p->longKeyPageArray[trfpArrayPos] = page->header.nextPage;
+ } else {
+ jam();
+ rfpPrevPageptr.i = page->header.prevPage;
+ ptrCheckGuard(rfpPrevPageptr, cpagesize, page8);
+ prevPage = (LongKeyPage *) &rfpPrevPageptr.p->word32[0];
+ // This page wasn't first in list, remove reference
+ // to this page from the previous page.
+ ndbrequire(prevPage->header.nextPage == rfpPageptr.i);
+ prevPage->header.nextPage = page->header.nextPage;
+ }//if
+
+ if (page->header.nextPage != RNIL) {
+ jam();
+ rfpNextPageptr.i = page->header.nextPage;
+ ptrCheckGuard(rfpNextPageptr, cpagesize, page8);
+ nextPage = (LongKeyPage *) &rfpNextPageptr.p->word32[0];
+ // This page wasn't last in list, remove reference
+ // to this page from the next page.
+ ndbrequire(nextPage->header.prevPage == rfpPageptr.i);
+ nextPage->header.prevPage = page->header.prevPage;
+ // Remove reference to next page in list.
+ page->header.nextPage = RNIL;
+ }//if
+
+ // This couldn't be set until now.
+ // Remove reference to previous page in list.
+ page->header.prevPage = RNIL;
+
+#ifdef VM_TRACE
+ checkPageArrayList(signal, "removeFromPageArrayList");
+#endif
+}//Dbacc::removeFromPageArrayList()
+
+/* --------------------------------------------------------------------------------- */
+/* INSERT A PAGE INTO THE PAGE ARRAY LIST. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::insertPageArrayList(Signal* signal)
+{
+ Page8Ptr ipaNextPagePtr;
+ LongKeyPage *page;
+ LongKeyPage *nextPage;
+
+ jam();
+
+#ifdef VM_TRACE
+ checkPageArrayList(signal, "insertPageArrayList1");
+ checkPageB4Insert(ipaPagePtr.i, "insertPageArrayList1");
+#endif
+
+ page = (LongKeyPage *) &ipaPagePtr.p->word32[0];
+
+ arrGuard(tipaArrayPos, 4);
+
+ if (fragrecptr.p->longKeyPageArray[tipaArrayPos] != RNIL) {
+ jam();
+ ipaNextPagePtr.i = fragrecptr.p->longKeyPageArray[tipaArrayPos];
+ ptrCheckGuard(ipaNextPagePtr, cpagesize, page8);
+ nextPage = (LongKeyPage *) &ipaNextPagePtr.p->word32[0];
+
+ // A page already existed in the list, add reference
+ // to this page in the next page.
+ nextPage->header.prevPage = ipaPagePtr.i;
+ }//if
+
+ page->header.prevPage = RNIL;
+ page->header.nextPage = fragrecptr.p->longKeyPageArray[tipaArrayPos];
+ page->header.pageArrayPos = tipaArrayPos;
+
+ fragrecptr.p->longKeyPageArray[tipaArrayPos] = ipaPagePtr.i;
+
+#ifdef VM_TRACE
+ checkPageArrayList(signal, "insertPageArrayList2");
+#endif
+}//Dbacc::insertPageArrayList()
+
+// --------------------------------------------------------------------------------- */
+// Check the page array list.
+// --------------------------------------------------------------------------------- */
+void Dbacc::checkPageArrayList(Signal* signal, char *calledFrom)
+{
+ Page8Ptr pagePtr;
+ Uint32 pageArrayIndex;
+ LongKeyPage *page;
+ Uint32 prevPage;
+
+ // Go through the longKeyPageArray and search for a page.
+ for (pageArrayIndex = 0; pageArrayIndex <= ZMAX_LONG_KEY_ARRAY_INDEX; pageArrayIndex++) {
+ jam();
+ pagePtr.i = fragrecptr.p->longKeyPageArray[pageArrayIndex];
+ prevPage = RNIL;
+
+ if (pagePtr.i != RNIL) {
+ // A page is found.
+ jam();
+ do {
+ ptrCheckGuard(pagePtr, cpagesize, page8);
+ page = (LongKeyPage *) &pagePtr.p->word32[0];
+
+ if ((page->header.freeArea >= 128) &&
+ (((page->header.freeArea - 128) / 512) == page->header.pageArrayPos) &&
+ (pageArrayIndex == page->header.pageArrayPos) &&
+ (page->header.prevPage == prevPage)) {
+ // The page found is OK, test next page.
+ prevPage = pagePtr.i;
+ pagePtr.i = page->header.nextPage;
+ jam();
+ } else {
+ jam();
+ ndbout << " ERROR in checkPageArrayList, called from " << calledFrom << endl
+ << " pagePtr.i = " << pagePtr.i << endl
+ << " prevPage = " << prevPage << endl
+ << " pageArrayIndex = " << pageArrayIndex << endl;
+ printoutInfoAndShutdown(page);
+ }
+ }//do
+ while (pagePtr.i != RNIL);
+ }//if
+ }//for
+}//Dbacc::checkPageArrayList()
+
+// --------------------------------------------------------------------------------- */
+// Check the page to put into the pageArrayList.
+// --------------------------------------------------------------------------------- */
+void Dbacc::checkPageB4Insert(Uint32 pageId, char *calledFrom) {
+ Page8Ptr pagePtr;
+ Uint32 pageArrayIndex;
+ LongKeyPage *page;
+
+ pagePtr.i = pageId;
+ ptrCheckGuard(pagePtr, cpagesize, page8);
+ page = (LongKeyPage *) &pagePtr.p->word32[0];
+
+ if ((page->header.nextPage != RNIL) ||
+ (page->header.prevPage != RNIL)) {
+ jam();
+ ndbout << " ERROR in checkPageB4Insert, called from " << calledFrom << endl
+ << " pagePtr.i = " << pagePtr.i << endl
+ << " page->header.nextPage = " << page->header.nextPage << endl
+ << " page->header.prevPage = " << page->header.prevPage << endl;
+ ndbrequire(false);
+ }
+
+ // Page should not be inserted in list if free area is less than 512 byte.
+ if (page->header.freeArea < 128) {
+ jam();
+ ndbout << " ERROR in checkPageB4Insert, called from " << calledFrom << endl
+ << " Page has to little free area to be in list." << endl
+ << " pagePtr.i = " << pagePtr.i << endl
+ << " tipaArrayPos = " << tipaArrayPos << endl;
+ printoutInfoAndShutdown(page);
+ }
+
+ // Check if position in list is correct
+ if ((((page->header.freeArea - 128) / 512) != page->header.pageArrayPos) ||
+ (page->header.pageArrayPos != tipaArrayPos)) {
+ ndbout << " ERROR in checkPageB4Insert, called from " << calledFrom << endl
+ << " Incorrect position in list." << endl
+ << " pagePtr.i = " << pagePtr.i << endl
+ << " tipaArrayPos = " << tipaArrayPos << endl;
+ printoutInfoAndShutdown(page);
+ }
+
+ // Check if page is already in list.
+ for (pageArrayIndex = 0; pageArrayIndex <= ZMAX_LONG_KEY_ARRAY_INDEX; pageArrayIndex++) {
+ jam();
+ pagePtr.i = fragrecptr.p->longKeyPageArray[pageArrayIndex];
+
+ if (pagePtr.i != RNIL) {
+ // A page is found.
+ jam();
+ do {
+ ptrCheckGuard(pagePtr, cpagesize, page8);
+ page = (LongKeyPage *) &pagePtr.p->word32[0];
+ if (pagePtr.i == pageId) {
+ jam();
+ ndbout << "ERROR in checkPageB4Insert, called from " << calledFrom << endl
+ << "Page exists already in list." << endl
+ << " pagePtr.i = " << pagePtr.i << endl;
+ printoutInfoAndShutdown(page);
+ }
+ pagePtr.i = page->header.nextPage;
+ }//do
+ while (pagePtr.i != RNIL);
+ }//if
+ }//for
+}//Dbacc::checkPageB4Insert()
+
+// --------------------------------------------------------------------------------- */
+// Check the page to remove from the pageArrayList.
+// --------------------------------------------------------------------------------- */
+void Dbacc::checkPageB4Remove(Uint32 pageId, char *calledFrom) {
+ Page8Ptr pagePtr;
+ Uint32 pageArrayIndex;
+ Uint32 noOfOccurrence = 0;
+ Uint32 noOfPagesInList = 0;
+ LongKeyPage *page;
+
+ LongKeyPage *prevPage;
+ LongKeyPage *nextPage;
+ Page8Ptr rfpPrevPageptr;
+ Page8Ptr rfpNextPageptr;
+
+
+ pagePtr.i = pageId;
+ ptrCheckGuard(pagePtr, cpagesize, page8);
+ page = (LongKeyPage *) &pagePtr.p->word32[0];
+
+ // Check that page is in list.
+ for (pageArrayIndex = 0; pageArrayIndex <= ZMAX_LONG_KEY_ARRAY_INDEX; pageArrayIndex++) {
+ jam();
+ pagePtr.i = fragrecptr.p->longKeyPageArray[pageArrayIndex];
+
+ if (pagePtr.i != RNIL) {
+ // A page is found.
+ jam();
+ do {
+ noOfPagesInList++;
+ ptrCheckGuard(pagePtr, cpagesize, page8);
+ page = (LongKeyPage *) &pagePtr.p->word32[0];
+ if (pagePtr.i == pageId) {
+ // Check the consistent in list.
+ if (page->header.prevPage != RNIL) {
+ rfpPrevPageptr.i = page->header.prevPage;
+ ptrCheckGuard(rfpPrevPageptr, cpagesize, page8);
+ prevPage = (LongKeyPage *) &rfpPrevPageptr.p->word32[0];
+ if (prevPage->header.nextPage != pageId) {
+ ndbout << "ERROR: inconsistent in checkPageB4Remove, called from " << calledFrom << endl
+ << "prevPage->header.nextPage = " << prevPage->header.nextPage << endl
+ << "pageId = " << pageId << endl;
+ printoutInfoAndShutdown(page);
+ }
+ }
+ // Check the consistent in list.
+ if (page->header.nextPage != RNIL) {
+ rfpNextPageptr.i = page->header.nextPage;
+ ptrCheckGuard(rfpNextPageptr, cpagesize, page8);
+ nextPage = (LongKeyPage *) &rfpNextPageptr.p->word32[0];
+ if (nextPage->header.prevPage != pageId) {
+ ndbout << "ERROR: inconsistent in checkPageB4Remove, called from " << calledFrom << endl
+ << "nextPage->header.prevPage = " << nextPage->header.prevPage << endl
+ << "pageId = " << pageId << endl;
+ printoutInfoAndShutdown(page);
+ }
+ }
+ jam();
+ noOfOccurrence++;
+ }
+ pagePtr.i = page->header.nextPage;
+ }//do
+ while (pagePtr.i != RNIL);
+ }//if
+ }//for
+
+ if (noOfOccurrence != 1) {
+ pagePtr.i = pageId;
+ ptrCheckGuard(pagePtr, cpagesize, page8);
+ page = (LongKeyPage *) &pagePtr.p->word32[0];
+ ndbout << "ERROR in checkPageB4Remove, called from " << calledFrom << endl
+ << "Page occur " << noOfOccurrence << " times in list" << endl
+ << "pageId = " << pageId << endl;
+ printoutInfoAndShutdown(page);
+ }
+}//Dbacc::checkPageB4Remove()
+
+
+// --------------------------------------------------------------------------------- */
+// Printout an error message and shutdown node.
+// --------------------------------------------------------------------------------- */
+void Dbacc::printoutInfoAndShutdown(LongKeyPage *page) {
+ ndbout << " page->header.pageArrayPos = " << page->header.pageArrayPos << endl
+ << " ((page->header.freeArea - 128) / 512) = "
+ << ((page->header.freeArea - 128) / 512) << endl
+ << " page->header.freeArea = " << page->header.freeArea << endl
+ << " page->header.noOfElements = " << page->header.noOfElements << endl
+ << " page->header.nextPage = " << page->header.nextPage << endl
+ << " page->header.prevPage = " << page->header.prevPage << endl
+ << " page->header.nextFreeIndex = " << page->header.nextFreeIndex << endl
+ << " page->header.insertPos = " << page->header.insertPos << endl
+ << " page->header.highestIndex = " << page->header.highestIndex << endl
+ << " page->header.pageId = " << page->header.pageId << endl;
+ ndbrequire(false);
+}//Dbacc::printoutInfoAndShutdown()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF INSERT_ELEMENT MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* MODULE: READ */
+/* THE FOLLOWING SUBROUTINES ARE ONLY USED BY GET_ELEMENT AND */
+/* GETDIRINDEX. THIS ROUTINE IS THE SOLE INTERFACE TO GET ELEMENTS */
+/* FROM THE INDEX. CURRENT USERS ARE ALL REQUESTS AND EXECUTE UNDO LOG */
+/* */
+/* THE FOLLOWING SUBROUTINES ARE INCLUDED IN THIS MODULE: */
+/* GET_ELEMENT */
+/* GET_DIRINDEX */
+/* SEARCH_LONG_KEY */
+/* */
+/* THESE ROUTINES ARE ONLY USED BY THIS MODULE AND BY NO ONE ELSE. */
+/* ALSO THE ROUTINES MAKE NO USE OF ROUTINES IN OTHER MODULES. */
+/* THE ONLY SHORT-LIVED VARIABLES USED IN OTHER PARTS OF THE BLOCK ARE */
+/* THOSE DEFINED AS INPUT AND OUTPUT IN GET_ELEMENT AND GETDIRINDEX */
+/* SHORT-LIVED VARIABLES INCLUDE TEMPORARY VARIABLES, COMMON VARIABLES */
+/* AND POINTER VARIABLES. */
+/* THE ONLY EXCEPTION TO THIS RULE IS FRAGRECPTR WHICH POINTS TO THE */
+/* FRAGMENT RECORD. THIS IS MORE LESS STATIC ALWAYS DURING A SIGNAL */
+/* EXECUTION. */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* GETDIRINDEX */
+/* SUPPORT ROUTINE FOR INSERT ELEMENT, GET ELEMENT AND COMMITDELETE */
+/* INPUT:FRAGRECPTR ( POINTER TO THE ACTIVE FRAGMENT REC) */
+/* OPERATION_REC_PTR (POINTER TO THE OPERATION REC). */
+/* */
+/* OUTPUT:GDI_PAGEPTR ( POINTER TO THE PAGE OF THE ELEMENT) */
+/* TGDI_PAGEINDEX ( INDEX OF THE ELEMENT IN THE PAGE). */
+/* */
+/* DESCRIPTION: CHECK THE HASH VALUE OF THE OPERATION REC AND CALCULATE THE */
+/* THE ADDRESS OF THE ELEMENT IN THE HASH TABLE,(GDI_PAGEPTR, */
+/* TGDI_PAGEINDEX) ACCORDING TO LH3. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::getdirindex(Signal* signal)
+{
+ DirRangePtr gdiDirRangePtr;
+ DirectoryarrayPtr gdiDirptr;
+ Uint32 tgdiTmp;
+ Uint32 tgdiAddress;
+
+ tgdiTmp = fragrecptr.p->k + fragrecptr.p->lhfragbits; /* OBS K = 6 */
+ tgdiPageindex = operationRecPtr.p->hashValue & ((1 << fragrecptr.p->k) - 1);
+ tgdiTmp = operationRecPtr.p->hashValue >> tgdiTmp;
+ tgdiTmp = (tgdiTmp << fragrecptr.p->k) | tgdiPageindex;
+ tgdiAddress = tgdiTmp & fragrecptr.p->maxp;
+ gdiDirRangePtr.i = fragrecptr.p->directory;
+ ptrCheckGuard(gdiDirRangePtr, cdirrangesize, dirRange);
+ if (tgdiAddress < fragrecptr.p->p) {
+ jam();
+ tgdiAddress = tgdiTmp & ((fragrecptr.p->maxp << 1) | 1);
+ }//if
+ tgdiTmp = tgdiAddress >> fragrecptr.p->k;
+ arrGuard((tgdiTmp >> 8), 256);
+ gdiDirptr.i = gdiDirRangePtr.p->dirArray[tgdiTmp >> 8];
+ ptrCheckGuard(gdiDirptr, cdirarraysize, directoryarray);
+ gdiPageptr.i = gdiDirptr.p->pagep[tgdiTmp & 0xff]; /* DIRECTORY INDEX OF SEND BUCKET PAGE */
+ ptrCheckGuard(gdiPageptr, cpagesize, page8);
+}//Dbacc::getdirindex()
+
+/* --------------------------------------------------------------------------------- */
+/* GET_ELEMENT */
+/* INPUT: */
+/* OPERATION_REC_PTR */
+/* FRAGRECPTR */
+/* OUTPUT: */
+/* TGE_RESULT RESULT SUCCESS = ZTRUE OTHERWISE ZFALSE */
+/* TGE_LOCKED LOCK INFORMATION IF SUCCESSFUL RESULT */
+/* GE_PAGEPTR PAGE POINTER OF FOUND ELEMENT */
+/* TGE_CONTAINERPTR CONTAINER INDEX OF FOUND ELEMENT */
+/* TGE_ELEMENTPTR ELEMENT INDEX OF FOUND ELEMENT */
+/* TGE_FORWARD DIRECTION OF CONTAINER WHERE ELEMENT FOUND */
+/* */
+/* DESCRIPTION: THE SUBROUTIN GOES THROUGH ALL CONTAINERS OF THE ACTIVE */
+/* BUCKET, AND SERCH FOR ELEMENT.THE PRIMARY KEYS WHICH IS SAVED */
+/* IN THE OPERATION REC ARE THE CHECK ITEMS IN THE SEARCHING. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::getElement(Signal* signal)
+{
+ DirRangePtr geOverflowrangeptr;
+ DirectoryarrayPtr geOverflowDirptr;
+ OperationrecPtr geTmpOperationRecPtr;
+ Uint32 tgeElementHeader;
+ Uint32 tgeElemStep;
+ Uint32 tgeContainerhead;
+ Uint32 tgePageindex;
+ Uint32 tgeActivePageDir;
+ Uint32 tgeNextptrtype;
+ register Uint32 tgeKeyptr;
+ register Uint32 tgeRemLen;
+ register Uint32 tgeCompareLen;
+ register Uint32 TelemLen = fragrecptr.p->elementLength;
+ register Uint32* Tkeydata = (Uint32*)&signal->theData[7];
+
+ getdirindex(signal);
+ tgePageindex = tgdiPageindex;
+ gePageptr = gdiPageptr;
+ tgeResult = ZFALSE;
+ tgeCompareLen = fragrecptr.p->keyLength;
+ const Uint32 isAccLockReq = operationRecPtr.p->isAccLockReq;
+ if (isAccLockReq) {
+ jam();
+ tgeCompareLen = 0;
+ }
+
+ // We can handle keylength up to 8, but not more (0 means dynamic)
+ if (tgeCompareLen >= 9) {
+ ACCKEY_error(2); return;
+ }//if
+ if (TelemLen < 3) {
+ ACCKEY_error(3); return;
+ }//if
+ tgeNextptrtype = ZLEFT;
+ tgeLocked = 0;
+
+ const Uint32 tmp = fragrecptr.p->k + fragrecptr.p->lhfragbits;
+ const Uint32 opHashValuePart = (operationRecPtr.p->hashValue >> tmp) &0xFFFF;
+ do {
+ tgeContainerptr = (tgePageindex << ZSHIFT_PLUS) - (tgePageindex << ZSHIFT_MINUS);
+ if (tgeNextptrtype == ZLEFT) {
+ jam();
+ tgeContainerptr = tgeContainerptr + ZHEAD_SIZE;
+ tgeElementptr = tgeContainerptr + ZCON_HEAD_SIZE;
+ tgeKeyptr = (tgeElementptr + ZELEM_HEAD_SIZE) + fragrecptr.p->localkeylen;
+ tgeElemStep = TelemLen;
+ tgeForward = 1;
+ if (tgeContainerptr >= 2048) { ACCKEY_error(4); return;}
+ tgeRemLen = gePageptr.p->word32[tgeContainerptr] >> 26;
+ if ((tgeContainerptr + tgeRemLen - 1) >= 2048) { ACCKEY_error(5); return;}
+ } else if (tgeNextptrtype == ZRIGHT) {
+ jam();
+ tgeContainerptr = tgeContainerptr + ((ZHEAD_SIZE + ZBUF_SIZE) - ZCON_HEAD_SIZE);
+ tgeElementptr = tgeContainerptr - 1;
+ tgeKeyptr = (tgeElementptr - ZELEM_HEAD_SIZE) - fragrecptr.p->localkeylen;
+ tgeElemStep = 0 - TelemLen;
+ tgeForward = (Uint32)-1;
+ if (tgeContainerptr >= 2048) { ACCKEY_error(4); return;}
+ tgeRemLen = gePageptr.p->word32[tgeContainerptr] >> 26;
+ if ((tgeContainerptr - tgeRemLen) >= 2048) { ACCKEY_error(5); return;}
+ } else {
+ ACCKEY_error(6); return;
+ }//if
+ if (tgeRemLen >= TelemLen) {
+ if (tgeRemLen > ZBUF_SIZE) {
+ ACCKEY_error(7); return;
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ // There is at least one element in this container. Check if it is the element
+ // searched for.
+ /* --------------------------------------------------------------------------------- */
+ if (tgeCompareLen != 0) {
+ /* --------------------------------------------------------------------------------- */
+ /* THIS PART IS USED TO SEARCH FOR KEYS WITH FIXED SIZE. THE LOOP TAKES CARE */
+ /* OF SEARCHING THROUGH ALL ELEMENTS IN ONE CONTAINER. */
+ /* --------------------------------------------------------------------------------- */
+ do {
+ register Uint32 TdataIndex = 0;
+ register Uint32 TgeIndex = 0;
+ jam();
+ tgeRemLen = tgeRemLen - TelemLen;
+ do {
+ if (gePageptr.p->word32[tgeKeyptr + TgeIndex] != Tkeydata[TdataIndex]) {
+ goto compare_next;
+ }//if
+ TdataIndex++;
+ TgeIndex += tgeForward;
+ } while (TdataIndex < tgeCompareLen);
+ /* --------------------------------------------------------------------------------- */
+ /* WE HAVE FOUND THE ELEMENT. GET THE LOCK INDICATOR AND RETURN FOUND. */
+ /* --------------------------------------------------------------------------------- */
+ jam();
+ tgeLocked = ElementHeader::getLocked(gePageptr.p->word32[tgeElementptr]);
+ tgeResult = ZTRUE;
+ TdataIndex = tgeElementptr + tgeForward;
+ TgeIndex = TdataIndex + tgeForward;
+ operationRecPtr.p->localdata[0] = gePageptr.p->word32[TdataIndex];
+ operationRecPtr.p->localdata[1] = gePageptr.p->word32[TgeIndex];
+ return;
+ /* --------------------------------------------------------------------------------- */
+ /* COMPARE NEXT ELEMENT */
+ /* --------------------------------------------------------------------------------- */
+ compare_next:
+ if (tgeRemLen <= ZCON_HEAD_SIZE) {
+ break;
+ }//if
+ tgeKeyptr = tgeKeyptr + tgeElemStep;
+ tgeElementptr = tgeElementptr + tgeElemStep;
+ } while (1);
+ } else if (! isAccLockReq) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THIS PART IS USED TO SEARCH FOR KEYS WITH VARIABLE LENGTH OR FIXED LENGTH */
+ /* GREATER THAN 32 BYTES. IN THIS CASE THE KEY PART IS STORED IN A SPECIAL */
+ /* LONG PAGE PART AND THE HASH INDEX CONTAINS A REFERENCE TO THERE PLUS A */
+ /* PART OF THE HASH VALUE. */
+ /* --------------------------------------------------------------------------------- */
+ do {
+ tgeElementHeader = gePageptr.p->word32[tgeElementptr];
+ tgeRemLen = tgeRemLen - TelemLen;
+ Uint32 hashValuePart;
+ if (ElementHeader::getLocked(tgeElementHeader)) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* IN THIS CASE THE HASH VALUE PART OF THE ELEMENT HEADER IS STORED IN THE */
+ /* OPERATION THAT OWNS THE LOCK. IN THIS CASE WE MIGHT AS WELL GO AHEAD AND */
+ /* CHECK THE KEY IN THE LONG PAGE. */
+ /* --------------------------------------------------------------------------------- */
+ geTmpOperationRecPtr.i =
+ ElementHeader::getOpPtrI(tgeElementHeader);
+ ptrCheckGuard(geTmpOperationRecPtr, coprecsize, operationrec);
+ hashValuePart = geTmpOperationRecPtr.p->hashvaluePart;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* IN THIS CASE THE HASH VALUE PART CAN BE CHECKED TO SEE IF THE HASH VALUE */
+ /* GIVES US A REASON TO CONTINUE CHECKING THE FULL KEY. */
+ /* --------------------------------------------------------------------------------- */
+ hashValuePart = ElementHeader::getHashValuePart(tgeElementHeader);
+ }//if
+
+ if (hashValuePart == opHashValuePart) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* IF THE HASH VALUES ARE EQUAL THEN XOR-ING THEM WILL GIVE THE RESULT 0. */
+ /* --------------------------------------------------------------------------------- */
+ /* WE HAVE FOUND A KEY WITH IDENTICAL HASH VALUE. MOST LIKELY WE HAVE FOUND THE*/
+ /* ELEMENT BUT FIRST WE NEED TO PERFORM A KEY COMPARISON. */
+ /* --------------------------------------------------------------------------------- */
+ tslcPageIndex = gePageptr.p->word32[tgeKeyptr] & 0x3ff;
+ tslcPagedir = gePageptr.p->word32[tgeKeyptr] >> 10;
+ searchLongKey(signal);
+ if (tslcResult == ZTRUE) {
+ register Uint32 TlocData1, TlocData2;
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE HAVE FOUND THE ELEMENT. GET THE LOCK INDICATOR AND RETURN FOUND. */
+ /* --------------------------------------------------------------------------------- */
+ tgeLocked = ElementHeader::getLocked(tgeElementHeader);
+ tgeResult = ZTRUE;
+ TlocData1 = tgeElementptr + tgeForward;
+ TlocData2 = TlocData1 + tgeForward;
+ operationRecPtr.p->localdata[0] = gePageptr.p->word32[TlocData1];
+ operationRecPtr.p->localdata[1] = gePageptr.p->word32[TlocData2];
+ return;
+ }//if
+ }
+ /* --------------------------------------------------------------------------------- */
+ /* COMPARE NEXT ELEMENT */
+ /* --------------------------------------------------------------------------------- */
+ if (tgeRemLen <= ZCON_HEAD_SIZE) {
+ break;
+ }//if
+ tgeKeyptr = tgeKeyptr + tgeElemStep;
+ tgeElementptr = tgeElementptr + tgeElemStep;
+ } while (1);
+ } else {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* Search for local key in a lock request */
+ /* --------------------------------------------------------------------------------- */
+ do {
+ tgeRemLen = tgeRemLen - TelemLen;
+ // position of local key word 1
+ Uint32 TdataIndex = tgeElementptr + tgeForward;
+ // XXX assume localkeylen is 1
+ if (gePageptr.p->word32[TdataIndex] == Tkeydata[0]) {
+ jam();
+ tgeLocked = ElementHeader::getLocked(gePageptr.p->word32[tgeElementptr]);
+ tgeResult = ZTRUE;
+ // position of local key word 2
+ Uint32 TgeIndex = TdataIndex + tgeForward;
+ operationRecPtr.p->localdata[0] = gePageptr.p->word32[TdataIndex];
+ operationRecPtr.p->localdata[1] = gePageptr.p->word32[TgeIndex];
+ return;
+ }//if
+ if (tgeRemLen <= ZCON_HEAD_SIZE) {
+ break;
+ }//if
+ tgeElementptr = tgeElementptr + tgeElemStep;
+ } while (1);
+ }//if
+ }//if
+ if (tgeRemLen != ZCON_HEAD_SIZE) {
+ ACCKEY_error(8); return;
+ }//if
+ tgeContainerhead = gePageptr.p->word32[tgeContainerptr];
+ tgeNextptrtype = (tgeContainerhead >> 7) & 0x3;
+ if (tgeNextptrtype == 0) {
+ jam();
+ return; /* NO MORE CONTAINER */
+ }//if
+ tgePageindex = tgeContainerhead & 0x7f; /* NEXT CONTAINER PAGE INDEX 7 BITS */
+ if (tgePageindex > ZEMPTYLIST) {
+ ACCKEY_error(9); return;
+ }//if
+ if (((tgeContainerhead >> 9) & 1) == ZFALSE) {
+ jam();
+ tgeActivePageDir = gePageptr.p->word32[tgeContainerptr + 1]; /* NEXT PAGE ID */
+ geOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ ptrCheckGuard(geOverflowrangeptr, cdirrangesize, dirRange);
+ arrGuard((tgeActivePageDir >> 8), 256);
+ geOverflowDirptr.i = geOverflowrangeptr.p->dirArray[tgeActivePageDir >> 8];
+ ptrCheckGuard(geOverflowDirptr, cdirarraysize, directoryarray);
+ gePageptr.i = geOverflowDirptr.p->pagep[tgeActivePageDir & 0xff];
+ ptrCheckGuard(gePageptr, cpagesize, page8);
+ }//if
+ } while (1);
+ return;
+}//Dbacc::getElement()
+
+/* --------------------------------------------------------------------------------- */
+/* SEARCH_LONG_KEY */
+/* INPUT: */
+/* TSLC_PAGEDIR PAGE DIRECTORY OF LONG PAGE */
+/* TSLC_PAGE_INDEX PAGE INDEX IN LONG PAGE */
+/* GE_OPERATION_REC_PTR */
+/* OUTPUT: */
+/* TSLC_RESULT */
+/* DESCRIPTION: SEARCH FOR AN ELEMENT IN A LONG_KEY_PAGE. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::searchLongKey(Signal* signal)
+{
+ DirRangePtr slcOverflowrangeptr;
+ DirectoryarrayPtr slcOverflowDirptr;
+ Page8Ptr slcPageptr;
+ Uint32 tslcIndexValue;
+ Uint32 tslcStartIndex;
+ Uint32 tslcIndex;
+ Uint32 guard30;
+ Uint32* Tkeydata = (Uint32*)&signal->theData[7];
+
+
+ slcOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ ptrCheckGuard(slcOverflowrangeptr, cdirrangesize, dirRange);
+ arrGuard((tslcPagedir >> 8), 256);
+ slcOverflowDirptr.i = slcOverflowrangeptr.p->dirArray[tslcPagedir >> 8];
+ ptrCheckGuard(slcOverflowDirptr, cdirarraysize, directoryarray);
+
+ // dbgWord32(slcOverflowDirptr, (int) (tslcPagedir & 0xff), slcOverflowDirptr.p->pagep[tslcPagedir & 0xff]);
+
+ slcPageptr.i = slcOverflowDirptr.p->pagep[tslcPagedir & 0xff];
+ ptrCheckGuard(slcPageptr, cpagesize, page8);
+ arrGuard(ZWORDS_IN_PAGE - tslcPageIndex, 2048);
+ dbgWord32(slcPageptr, ZWORDS_IN_PAGE - tslcPageIndex, (int)slcPageptr.p->word32[ZWORDS_IN_PAGE - tslcPageIndex] & 0xffff);
+ dbgWord32(slcPageptr, ZWORDS_IN_PAGE - tslcPageIndex, slcPageptr.p->word32[ZWORDS_IN_PAGE - tslcPageIndex] >> 16);
+ tslcIndexValue = slcPageptr.p->word32[ZWORDS_IN_PAGE - tslcPageIndex];
+ if ((tslcIndexValue >> 16) != operationRecPtr.p->tupkeylen) {
+ jam();
+ tslcResult = ZFALSE;
+ return;
+ }//if
+ tslcStartIndex = tslcIndexValue & 0xffff;
+ guard30 = operationRecPtr.p->tupkeylen - 1;
+ arrGuard(guard30, 2048);
+ arrGuard(guard30 + tslcStartIndex, 2048);
+ for (tslcIndex = 0; tslcIndex <= guard30; tslcIndex++) {
+ dbgWord32(slcPageptr, tslcIndex + tslcStartIndex, slcPageptr.p->word32[tslcIndex + tslcStartIndex]);
+ if (slcPageptr.p->word32[tslcIndex + tslcStartIndex] != Tkeydata[tslcIndex]) {
+ jam();
+ tslcResult = ZFALSE;
+ return;
+ }//if
+ }//for
+ jam();
+ tslcResult = ZTRUE;
+ operationRecPtr.p->longPagePtr = slcPageptr.i;
+ operationRecPtr.p->longKeyPageIndex = tslcPageIndex;
+ arrGuard(tslcPageIndex, ZMAX_NO_OF_LONGKEYS_IN_PAGE);
+ arrGuard(slcPageptr.i, cpagesize);
+}//Dbacc::searchLongKey()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF GET_ELEMENT MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* MODULE: DELETE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* COMMITDELETE */
+/* INPUT: OPERATION_REC_PTR, PTR TO AN OPERATION RECORD. */
+/* FRAGRECPTR, PTR TO A FRAGMENT RECORD */
+/* */
+/* OUTPUT: */
+/* NONE */
+/* DESCRIPTION: DELETE OPERATIONS WILL BE COMPLETED AT THE COMMIT OF TRANSA- */
+/* CTION. THIS SUBROUTINE SEARCHS FOR ELEMENT AND DELETES IT. IT DOES SO BY */
+/* REPLACING IT WITH THE LAST ELEMENT IN THE BUCKET. IF THE DELETED ELEMENT */
+/* IS ALSO THE LAST ELEMENT THEN IT IS ONLY NECESSARY TO REMOVE THE ELEMENT. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::commitdelete(Signal* signal, bool systemRestart)
+{
+ if (!systemRestart) {
+ jam();
+ signal->theData[0] = fragrecptr.p->myfid;
+ signal->theData[1] = fragrecptr.p->myTableId;
+ signal->theData[2] = operationRecPtr.p->localdata[0];
+ Uint32 localKey = operationRecPtr.p->localdata[0];
+ Uint32 pageId = localKey >> MAX_TUPLES_BITS;
+ Uint32 pageIndex = localKey & ((1 << MAX_TUPLES_BITS) - 1);
+ signal->theData[2] = pageId;
+ signal->theData[3] = pageIndex;
+ EXECUTE_DIRECT(DBTUP, GSN_TUP_DEALLOCREQ, signal, 4);
+ jamEntry();
+ }//if
+ if (fragrecptr.p->keyLength == 0) {
+ jam();
+ tdlkLogicalPageIndex = operationRecPtr.p->longKeyPageIndex;
+ dlkPageptr.i = operationRecPtr.p->longPagePtr;
+ ptrCheckGuard(dlkPageptr, cpagesize, page8);
+ deleteLongKey(signal);
+ }//if
+ getdirindex(signal);
+ tlastPageindex = tgdiPageindex;
+ lastPageptr.i = gdiPageptr.i;
+ lastPageptr.p = gdiPageptr.p;
+ tlastForward = ZTRUE;
+ tlastContainerptr = (tlastPageindex << ZSHIFT_PLUS) - (tlastPageindex << ZSHIFT_MINUS);
+ tlastContainerptr = tlastContainerptr + ZHEAD_SIZE;
+ arrGuard(tlastContainerptr, 2048);
+ tlastContainerhead = lastPageptr.p->word32[tlastContainerptr];
+ tlastContainerlen = tlastContainerhead >> 26;
+ lastPrevpageptr.i = RNIL;
+ ptrNull(lastPrevpageptr);
+ tlastPrevconptr = 0;
+ getLastAndRemove(signal);
+
+ delPageptr.i = operationRecPtr.p->elementPage;
+ ptrCheckGuard(delPageptr, cpagesize, page8);
+ tdelElementptr = operationRecPtr.p->elementPointer;
+ /* --------------------------------------------------------------------------------- */
+ // Here we have to take extreme care since we do not want locks to end up after the
+ // log execution. Thus it is necessary to put back the element in unlocked shape.
+ // We thus update the element header to ensure we log an unlocked element. We do not
+ // need to restore it later since it is deleted immediately anyway.
+ /* --------------------------------------------------------------------------------- */
+ const Uint32 hv = operationRecPtr.p->hashvaluePart;
+ const Uint32 eh = ElementHeader::setUnlocked(hv, 0);
+ delPageptr.p->word32[tdelElementptr] = eh;
+ if (operationRecPtr.p->elementPage == lastPageptr.i) {
+ if (operationRecPtr.p->elementPointer == tlastElementptr) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE LAST ELEMENT WAS THE ELEMENT TO BE DELETED. WE NEED NOT COPY IT. */
+ /* --------------------------------------------------------------------------------- */
+ return;
+ }//if
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* THE DELETED ELEMENT IS NOT THE LAST. WE READ THE LAST ELEMENT AND OVERWRITE THE */
+ /* DELETED ELEMENT. */
+ /* --------------------------------------------------------------------------------- */
+ tdelContainerptr = operationRecPtr.p->elementContainer;
+ tdelForward = operationRecPtr.p->elementIsforward;
+ deleteElement(signal);
+}//Dbacc::commitdelete()
+
+/* --------------------------------------------------------------------------------- */
+/* DELETE_ELEMENT */
+/* INPUT: FRAGRECPTR, POINTER TO A FRAGMENT RECORD */
+/* LAST_PAGEPTR, POINTER TO THE PAGE OF THE LAST ELEMENT */
+/* DEL_PAGEPTR, POINTER TO THE PAGE OF THE DELETED ELEMENT */
+/* TLAST_ELEMENTPTR, ELEMENT POINTER OF THE LAST ELEMENT */
+/* TDEL_ELEMENTPTR, ELEMENT POINTER OF THE DELETED ELEMENT */
+/* TLAST_FORWARD, DIRECTION OF LAST ELEMENT */
+/* TDEL_FORWARD, DIRECTION OF DELETED ELEMENT */
+/* TDEL_CONTAINERPTR, CONTAINER POINTER OF DELETED ELEMENT */
+/* DESCRIPTION: COPY LAST ELEMENT TO DELETED ELEMENT AND UPDATE UNDO LOG AND */
+/* UPDATE ANY ACTIVE OPERATION ON THE MOVED ELEMENT. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::deleteElement(Signal* signal)
+{
+ OperationrecPtr deOperationRecPtr;
+ Uint32 tdeIndex;
+ Uint32 tlastMoveElemptr;
+ Uint32 tdelMoveElemptr;
+ Uint32 guard31;
+
+ if (tlastElementptr >= 2048)
+ goto deleteElement_index_error1;
+ {
+ const Uint32 tdeElemhead = lastPageptr.p->word32[tlastElementptr];
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ datapageptr.p = delPageptr.p;
+ cundoinfolength = fragrecptr.p->elementLength;
+ if (tdelForward == ZTRUE) {
+ jam();
+ cundoElemIndex = tdelElementptr;
+ } else {
+ jam();
+ cundoElemIndex = (tdelElementptr + 1) - fragrecptr.p->elementLength;
+ }//if
+ undoWritingProcess(signal);
+ }//if
+ tlastMoveElemptr = tlastElementptr;
+ tdelMoveElemptr = tdelElementptr;
+ guard31 = fragrecptr.p->elementLength - 1;
+ for (tdeIndex = 0; tdeIndex <= guard31; tdeIndex++) {
+ dbgWord32(delPageptr, tdelMoveElemptr, lastPageptr.p->word32[tlastMoveElemptr]);
+ if ((tlastMoveElemptr >= 2048) ||
+ (tdelMoveElemptr >= 2048))
+ goto deleteElement_index_error2;
+ delPageptr.p->word32[tdelMoveElemptr] = lastPageptr.p->word32[tlastMoveElemptr];
+ tdelMoveElemptr = tdelMoveElemptr + tdelForward;
+ tlastMoveElemptr = tlastMoveElemptr + tlastForward;
+ }//for
+ if (ElementHeader::getLocked(tdeElemhead)) {
+ /* --------------------------------------------------------------------------------- */
+ /* THE LAST ELEMENT IS LOCKED AND IS THUS REFERENCED BY AN OPERATION RECORD. WE NEED */
+ /* TO UPDATE THE OPERATION RECORD WITH THE NEW REFERENCE TO THE ELEMENT. */
+ /* --------------------------------------------------------------------------------- */
+ deOperationRecPtr.i = ElementHeader::getOpPtrI(tdeElemhead);
+ ptrCheckGuard(deOperationRecPtr, coprecsize, operationrec);
+ if (cundoLogActive == ZFALSE) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE DO NOT BOTHER WITH THIS INFORMATION DURING EXECUTION OF THE UNDO LOG. */
+ /* --------------------------------------------------------------------------------- */
+ deOperationRecPtr.p->elementPage = delPageptr.i;
+ deOperationRecPtr.p->elementContainer = tdelContainerptr;
+ deOperationRecPtr.p->elementPointer = tdelElementptr;
+ deOperationRecPtr.p->elementIsforward = tdelForward;
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ // We need to take extreme care to not install locked records after system restart.
+ // An undo of the delete will reinstall the moved record. We have to ensure that the
+ // lock is removed to ensure that no such thing happen.
+ /* --------------------------------------------------------------------------------- */
+ Uint32 eh = ElementHeader::setUnlocked(deOperationRecPtr.p->hashvaluePart,
+ 0);
+ lastPageptr.p->word32[tlastElementptr] = eh;
+ }//if
+ return;
+ }
+
+ deleteElement_index_error1:
+ arrGuard(tlastElementptr, 2048);
+ return;
+
+ deleteElement_index_error2:
+ arrGuard(tdelMoveElemptr + guard31, 2048);
+ arrGuard(tlastMoveElemptr, 2048);
+ return;
+
+}//Dbacc::deleteElement()
+
+/* --------------------------------------------------------------------------------- */
+/* GET_LAST_AND_REMOVE */
+/* INPUT: */
+/* LAST_PAGEPTR PAGE POINTER OF FIRST CONTAINER IN SEARCH OF LAST*/
+/* TLAST_CONTAINERPTR CONTAINER INDEX OF THE SAME */
+/* TLAST_CONTAINERHEAD CONTAINER HEADER OF THE SAME */
+/* TLAST_PAGEINDEX PAGE INDEX OF THE SAME */
+/* TLAST_FORWARD CONTAINER DIRECTION OF THE SAME */
+/* TLAST_CONTAINERLEN CONTAINER LENGTH OF THE SAME */
+/* LAST_PREVPAGEPTR PAGE POINTER OF PREVIOUS CONTAINER OF THE SAME */
+/* TLAST_PREVCONPTR CONTAINER INDEX OF PREVIOUS CONTAINER OF THE SAME*/
+/* */
+/* OUTPUT: */
+/* ALL VARIABLES FROM INPUT BUT NOW CONTAINING INFO ABOUT LAST */
+/* CONTAINER. */
+/* TLAST_ELEMENTPTR LAST ELEMENT POINTER IN LAST CONTAINER */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::getLastAndRemove(Signal* signal)
+{
+ DirRangePtr glrOverflowrangeptr;
+ DirectoryarrayPtr glrOverflowDirptr;
+ Uint32 tglrHead;
+ Uint32 tglrTmp;
+
+ GLR_LOOP_10:
+ if (((tlastContainerhead >> 7) & 0x3) != 0) {
+ jam();
+ lastPrevpageptr.i = lastPageptr.i;
+ lastPrevpageptr.p = lastPageptr.p;
+ tlastPrevconptr = tlastContainerptr;
+ tlastPageindex = tlastContainerhead & 0x7f;
+ if (((tlastContainerhead >> 9) & 0x1) == ZFALSE) {
+ jam();
+ arrGuard(tlastContainerptr + 1, 2048);
+ tglrTmp = lastPageptr.p->word32[tlastContainerptr + 1];
+ glrOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ ptrCheckGuard(glrOverflowrangeptr, cdirrangesize, dirRange);
+ arrGuard((tglrTmp >> 8), 256);
+ glrOverflowDirptr.i = glrOverflowrangeptr.p->dirArray[tglrTmp >> 8];
+ ptrCheckGuard(glrOverflowDirptr, cdirarraysize, directoryarray);
+ lastPageptr.i = glrOverflowDirptr.p->pagep[tglrTmp & 0xff];
+ ptrCheckGuard(lastPageptr, cpagesize, page8);
+ }//if
+ tlastContainerptr = (tlastPageindex << ZSHIFT_PLUS) - (tlastPageindex << ZSHIFT_MINUS);
+ if (((tlastContainerhead >> 7) & 3) == ZLEFT) {
+ jam();
+ tlastForward = ZTRUE;
+ tlastContainerptr = tlastContainerptr + ZHEAD_SIZE;
+ } else if (((tlastContainerhead >> 7) & 3) == ZRIGHT) {
+ jam();
+ tlastForward = cminusOne;
+ tlastContainerptr = ((tlastContainerptr + ZHEAD_SIZE) + ZBUF_SIZE) - ZCON_HEAD_SIZE;
+ } else {
+ ndbrequire(false);
+ return;
+ }//if
+ arrGuard(tlastContainerptr, 2048);
+ tlastContainerhead = lastPageptr.p->word32[tlastContainerptr];
+ tlastContainerlen = tlastContainerhead >> 26;
+ ndbrequire(tlastContainerlen >= ((Uint32)ZCON_HEAD_SIZE + fragrecptr.p->elementLength));
+ goto GLR_LOOP_10;
+ }//if
+ tlastContainerlen = tlastContainerlen - fragrecptr.p->elementLength;
+ if (tlastForward == ZTRUE) {
+ jam();
+ tlastElementptr = tlastContainerptr + tlastContainerlen;
+ } else {
+ jam();
+ tlastElementptr = (tlastContainerptr + (ZCON_HEAD_SIZE - 1)) - tlastContainerlen;
+ }//if
+ rlPageptr.i = lastPageptr.i;
+ rlPageptr.p = lastPageptr.p;
+ trlPageindex = tlastPageindex;
+ if (((tlastContainerhead >> 10) & 1) == 1) {
+ /* --------------------------------------------------------------------------------- */
+ /* WE HAVE OWNERSHIP OF BOTH PARTS OF THE CONTAINER ENDS. */
+ /* --------------------------------------------------------------------------------- */
+ if (tlastContainerlen < ZDOWN_LIMIT) {
+ /* --------------------------------------------------------------------------------- */
+ /* WE HAVE DECREASED THE SIZE BELOW THE DOWN LIMIT, WE MUST GIVE UP THE OTHER */
+ /* SIDE OF THE BUFFER. */
+ /* --------------------------------------------------------------------------------- */
+ tlastContainerhead = tlastContainerhead ^ (1 << 10);
+ trlRelCon = ZFALSE;
+ if (tlastForward == ZTRUE) {
+ jam();
+ turlIndex = tlastContainerptr + (ZBUF_SIZE - ZCON_HEAD_SIZE);
+ releaseRightlist(signal);
+ } else {
+ jam();
+ tullIndex = tlastContainerptr - (ZBUF_SIZE - ZCON_HEAD_SIZE);
+ releaseLeftlist(signal);
+ }//if
+ }//if
+ }//if
+ if (tlastContainerlen <= 2) {
+ ndbrequire(tlastContainerlen == 2);
+ if (lastPrevpageptr.i != RNIL) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE LAST CONTAINER IS EMPTY AND IS NOT THE FIRST CONTAINER WHICH IS NOT REMOVED. */
+ /* DELETE THE LAST CONTAINER AND UPDATE THE PREVIOUS CONTAINER. ALSO PUT THIS */
+ /* CONTAINER IN FREE CONTAINER LIST OF THE PAGE. */
+ /* --------------------------------------------------------------------------------- */
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ datapageptr.p = lastPrevpageptr.p;
+ cundoElemIndex = tlastPrevconptr;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ ndbrequire(tlastPrevconptr < 2048);
+ tglrTmp = lastPrevpageptr.p->word32[tlastPrevconptr] >> 9;
+ dbgWord32(lastPrevpageptr, tlastPrevconptr, tglrTmp << 9);
+ lastPrevpageptr.p->word32[tlastPrevconptr] = tglrTmp << 9;
+ trlRelCon = ZTRUE;
+ if (tlastForward == ZTRUE) {
+ jam();
+ tullIndex = tlastContainerptr;
+ releaseLeftlist(signal);
+ } else {
+ jam();
+ turlIndex = tlastContainerptr;
+ releaseRightlist(signal);
+ }//if
+ return;
+ }//if
+ }//if
+ tglrHead = tlastContainerhead << 6;
+ tglrHead = tglrHead >> 6;
+ tglrHead = tglrHead | (tlastContainerlen << 26);
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ datapageptr.p = lastPageptr.p;
+ cundoElemIndex = tlastContainerptr;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(lastPageptr, tlastContainerptr, tglrHead);
+ arrGuard(tlastContainerptr, 2048);
+ lastPageptr.p->word32[tlastContainerptr] = tglrHead;
+}//Dbacc::getLastAndRemove()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_LEFTLIST */
+/* INPUT: */
+/* RL_PAGEPTR PAGE POINTER OF CONTAINER TO BE RELEASED */
+/* TRL_PAGEINDEX PAGE INDEX OF CONTAINER TO BE RELEASED */
+/* TURL_INDEX INDEX OF CONTAINER TO BE RELEASED */
+/* TRL_REL_CON TRUE IF CONTAINER RELEASED OTHERWISE ONLY */
+/* A PART IS RELEASED. */
+/* */
+/* OUTPUT: */
+/* NONE */
+/* */
+/* THE FREE LIST OF LEFT FREE BUFFER IN THE PAGE WILL BE UPDATE */
+/* TULL_INDEX IS INDEX TO THE FIRST WORD IN THE LEFT SIDE OF THE BUFFER */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseLeftlist(Signal* signal)
+{
+ Uint32 tullTmp;
+ Uint32 tullTmp1;
+
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ datapageptr.p = rlPageptr.p;
+ cundoElemIndex = tullIndex;
+ cundoinfolength = 2;
+ undoWritingProcess(signal);
+ }//if
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ cundoElemIndex = ZPOS_EMPTY_LIST;
+ cundoinfolength = 2;
+ undoWritingProcess(signal);
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* IF A CONTAINER IS RELEASED AND NOT ONLY A PART THEN WE HAVE TO REMOVE IT */
+ /* FROM THE LIST OF USED CONTAINERS IN THE PAGE. THIS IN ORDER TO ENSURE THAT */
+ /* WE CAN FIND ALL LOCKED ELEMENTS DURING LOCAL CHECKPOINT. */
+ /* --------------------------------------------------------------------------------- */
+ if (trlRelCon == ZTRUE) {
+ arrGuard(tullIndex, 2048);
+ trlHead = rlPageptr.p->word32[tullIndex];
+ trlNextused = (trlHead >> 11) & 0x7f;
+ trlPrevused = (trlHead >> 18) & 0x7f;
+ if (trlNextused < ZEMPTYLIST) {
+ jam();
+ tullTmp1 = (trlNextused << ZSHIFT_PLUS) - (trlNextused << ZSHIFT_MINUS);
+ tullTmp1 = tullTmp1 + ZHEAD_SIZE;
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ cundoElemIndex = tullTmp1;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ tullTmp = rlPageptr.p->word32[tullTmp1] & 0xfe03ffff;
+ dbgWord32(rlPageptr, tullTmp1, tullTmp | (trlPrevused << 18));
+ rlPageptr.p->word32[tullTmp1] = tullTmp | (trlPrevused << 18);
+ } else {
+ ndbrequire(trlNextused == ZEMPTYLIST);
+ jam();
+ }//if
+ if (trlPrevused < ZEMPTYLIST) {
+ jam();
+ tullTmp1 = (trlPrevused << ZSHIFT_PLUS) - (trlPrevused << ZSHIFT_MINUS);
+ tullTmp1 = tullTmp1 + ZHEAD_SIZE;
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ cundoElemIndex = tullTmp1;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ tullTmp = rlPageptr.p->word32[tullTmp1] & 0xfffc07ff;
+ dbgWord32(rlPageptr, tullTmp1, tullTmp | (trlNextused << 11));
+ rlPageptr.p->word32[tullTmp1] = tullTmp | (trlNextused << 11);
+ } else {
+ ndbrequire(trlPrevused == ZEMPTYLIST);
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE ARE FIRST IN THE LIST AND THUS WE NEED TO UPDATE THE FIRST POINTER. */
+ /* --------------------------------------------------------------------------------- */
+ tullTmp = rlPageptr.p->word32[ZPOS_EMPTY_LIST] & 0xc07fffff;
+ dbgWord32(rlPageptr, ZPOS_EMPTY_LIST, tullTmp | (trlNextused << 23));
+ rlPageptr.p->word32[ZPOS_EMPTY_LIST] = tullTmp | (trlNextused << 23);
+ }//if
+ }//if
+ dbgWord32(rlPageptr, tullIndex + 1, ZEMPTYLIST);
+ arrGuard(tullIndex + 1, 2048);
+ rlPageptr.p->word32[tullIndex + 1] = ZEMPTYLIST;
+ tullTmp1 = (rlPageptr.p->word32[ZPOS_EMPTY_LIST] >> 7) & 0x7f;
+ dbgWord32(rlPageptr, tullIndex, tullTmp1);
+ arrGuard(tullIndex, 2048);
+ rlPageptr.p->word32[tullIndex] = tullTmp1;
+ if (tullTmp1 < ZEMPTYLIST) {
+ jam();
+ tullTmp1 = (tullTmp1 << ZSHIFT_PLUS) - (tullTmp1 << ZSHIFT_MINUS);
+ tullTmp1 = (tullTmp1 + ZHEAD_SIZE) + 1;
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ cundoElemIndex = tullTmp1;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(rlPageptr, tullTmp1, trlPageindex);
+ rlPageptr.p->word32[tullTmp1] = trlPageindex; /* UPDATES PREV POINTER IN THE NEXT FREE */
+ } else {
+ ndbrequire(tullTmp1 == ZEMPTYLIST);
+ }//if
+ tullTmp = rlPageptr.p->word32[ZPOS_EMPTY_LIST];
+ tullTmp = (((tullTmp >> 14) << 14) | (trlPageindex << 7)) | (tullTmp & 0x7f);
+ dbgWord32(rlPageptr, ZPOS_EMPTY_LIST, tullTmp);
+ rlPageptr.p->word32[ZPOS_EMPTY_LIST] = tullTmp;
+ dbgWord32(rlPageptr, ZPOS_ALLOC_CONTAINERS, rlPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] - 1);
+ rlPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] = rlPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] - 1;
+ ndbrequire(rlPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] <= ZNIL);
+ if (((rlPageptr.p->word32[ZPOS_EMPTY_LIST] >> ZPOS_PAGE_TYPE_BIT) & 3) == 1) {
+ jam();
+ colPageptr.i = rlPageptr.i;
+ colPageptr.p = rlPageptr.p;
+ ptrCheck(colPageptr, cpagesize, page8);
+ checkoverfreelist(signal);
+ }//if
+}//Dbacc::releaseLeftlist()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_RIGHTLIST */
+/* INPUT: */
+/* RL_PAGEPTR PAGE POINTER OF CONTAINER TO BE RELEASED */
+/* TRL_PAGEINDEX PAGE INDEX OF CONTAINER TO BE RELEASED */
+/* TURL_INDEX INDEX OF CONTAINER TO BE RELEASED */
+/* TRL_REL_CON TRUE IF CONTAINER RELEASED OTHERWISE ONLY */
+/* A PART IS RELEASED. */
+/* */
+/* OUTPUT: */
+/* NONE */
+/* */
+/* THE FREE LIST OF RIGHT FREE BUFFER IN THE PAGE WILL BE UPDATE. */
+/* TURL_INDEX IS INDEX TO THE FIRST WORD IN THE RIGHT SIDE OF */
+/* THE BUFFER, WHICH IS THE LAST WORD IN THE BUFFER. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseRightlist(Signal* signal)
+{
+ Uint32 turlTmp1;
+ Uint32 turlTmp;
+
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ datapageptr.p = rlPageptr.p;
+ cundoElemIndex = turlIndex;
+ cundoinfolength = 2;
+ undoWritingProcess(signal);
+ }//if
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ cundoElemIndex = ZPOS_EMPTY_LIST;
+ cundoinfolength = 2;
+ undoWritingProcess(signal);
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* IF A CONTAINER IS RELEASED AND NOT ONLY A PART THEN WE HAVE TO REMOVE IT */
+ /* FROM THE LIST OF USED CONTAINERS IN THE PAGE. THIS IN ORDER TO ENSURE THAT */
+ /* WE CAN FIND ALL LOCKED ELEMENTS DURING LOCAL CHECKPOINT. */
+ /* --------------------------------------------------------------------------------- */
+ if (trlRelCon == ZTRUE) {
+ jam();
+ arrGuard(turlIndex, 2048);
+ trlHead = rlPageptr.p->word32[turlIndex];
+ trlNextused = (trlHead >> 11) & 0x7f;
+ trlPrevused = (trlHead >> 18) & 0x7f;
+ if (trlNextused < ZEMPTYLIST) {
+ jam();
+ turlTmp1 = (trlNextused << ZSHIFT_PLUS) - (trlNextused << ZSHIFT_MINUS);
+ turlTmp1 = turlTmp1 + ((ZHEAD_SIZE + ZBUF_SIZE) - ZCON_HEAD_SIZE);
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ cundoElemIndex = turlTmp1;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ turlTmp = rlPageptr.p->word32[turlTmp1] & 0xfe03ffff;
+ dbgWord32(rlPageptr, turlTmp1, turlTmp | (trlPrevused << 18));
+ rlPageptr.p->word32[turlTmp1] = turlTmp | (trlPrevused << 18);
+ } else {
+ ndbrequire(trlNextused == ZEMPTYLIST);
+ jam();
+ }//if
+ if (trlPrevused < ZEMPTYLIST) {
+ jam();
+ turlTmp1 = (trlPrevused << ZSHIFT_PLUS) - (trlPrevused << ZSHIFT_MINUS);
+ turlTmp1 = turlTmp1 + ((ZHEAD_SIZE + ZBUF_SIZE) - ZCON_HEAD_SIZE);
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ cundoElemIndex = turlTmp1;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ turlTmp = rlPageptr.p->word32[turlTmp1] & 0xfffc07ff;
+ dbgWord32(rlPageptr, turlTmp1, turlTmp | (trlNextused << 11));
+ rlPageptr.p->word32[turlTmp1] = turlTmp | (trlNextused << 11);
+ } else {
+ ndbrequire(trlPrevused == ZEMPTYLIST);
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE ARE FIRST IN THE LIST AND THUS WE NEED TO UPDATE THE FIRST POINTER */
+ /* OF THE RIGHT CONTAINER LIST. */
+ /* --------------------------------------------------------------------------------- */
+ turlTmp = rlPageptr.p->word32[ZPOS_EMPTY_LIST] & 0xff80ffff;
+ dbgWord32(rlPageptr, ZPOS_EMPTY_LIST, turlTmp | (trlNextused << 16));
+ rlPageptr.p->word32[ZPOS_EMPTY_LIST] = turlTmp | (trlNextused << 16);
+ }//if
+ }//if
+ dbgWord32(rlPageptr, turlIndex + 1, ZEMPTYLIST);
+ arrGuard(turlIndex + 1, 2048);
+ rlPageptr.p->word32[turlIndex + 1] = ZEMPTYLIST;
+ turlTmp1 = rlPageptr.p->word32[ZPOS_EMPTY_LIST] & 0x7f;
+ dbgWord32(rlPageptr, turlIndex, turlTmp1);
+ arrGuard(turlIndex, 2048);
+ rlPageptr.p->word32[turlIndex] = turlTmp1;
+ if (turlTmp1 < ZEMPTYLIST) {
+ jam();
+ turlTmp = (turlTmp1 << ZSHIFT_PLUS) - (turlTmp1 << ZSHIFT_MINUS);
+ turlTmp = turlTmp + ((ZHEAD_SIZE + ZBUF_SIZE) - (ZCON_HEAD_SIZE - 1));
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ cundoElemIndex = turlTmp;
+ cundoinfolength = 1;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(rlPageptr, turlTmp, trlPageindex);
+ rlPageptr.p->word32[turlTmp] = trlPageindex; /* UPDATES PREV POINTER IN THE NEXT FREE */
+ } else {
+ ndbrequire(turlTmp1 == ZEMPTYLIST);
+ }//if
+ turlTmp = rlPageptr.p->word32[ZPOS_EMPTY_LIST];
+ dbgWord32(rlPageptr, ZPOS_EMPTY_LIST, ((turlTmp >> 7) << 7) | trlPageindex);
+ rlPageptr.p->word32[ZPOS_EMPTY_LIST] = ((turlTmp >> 7) << 7) | trlPageindex;
+ dbgWord32(rlPageptr, ZPOS_ALLOC_CONTAINERS, rlPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] - 1);
+ rlPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] = rlPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] - 1;
+ ndbrequire(rlPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] <= ZNIL);
+ if (((rlPageptr.p->word32[ZPOS_EMPTY_LIST] >> ZPOS_PAGE_TYPE_BIT) & 3) == 1) {
+ jam();
+ colPageptr.i = rlPageptr.i;
+ colPageptr.p = rlPageptr.p;
+ checkoverfreelist(signal);
+ }//if
+}//Dbacc::releaseRightlist()
+
+/* --------------------------------------------------------------------------------- */
+/* CHECKOVERFREELIST */
+/* INPUT: COL_PAGEPTR, POINTER OF AN OVERFLOW PAGE RECORD. */
+/* DESCRIPTION: CHECKS IF THE PAGE HAVE TO PUT IN FREE LIST OF OVER FLOW */
+/* PAGES. WHEN IT HAVE TO, AN OVERFLOW REC PTR WILL BE ALLOCATED */
+/* TO KEEP NFORMATION ABOUT THE PAGE. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::checkoverfreelist(Signal* signal)
+{
+ Uint32 tcolTmp;
+
+ if (fragrecptr.p->loadingFlag == ZFALSE) {
+ tcolTmp = colPageptr.p->word32[ZPOS_ALLOC_CONTAINERS];
+ if (tcolTmp <= ZFREE_LIMIT) {
+ if (tcolTmp == 0) {
+ jam();
+ ropPageptr = colPageptr;
+ releaseOverpage(signal);
+ } else {
+ jam();
+ if (colPageptr.p->word32[ZPOS_OVERFLOWREC] == RNIL) {
+ ndbrequire(cfirstfreeoverrec != RNIL);
+ jam();
+ seizeOverRec(signal);
+ sorOverflowRecPtr.p->dirindex = colPageptr.p->word32[ZPOS_PAGE_ID];
+ sorOverflowRecPtr.p->overpage = colPageptr.i;
+ dbgWord32(colPageptr, ZPOS_OVERFLOWREC, sorOverflowRecPtr.i);
+ colPageptr.p->word32[ZPOS_OVERFLOWREC] = sorOverflowRecPtr.i;
+ porOverflowRecPtr = sorOverflowRecPtr;
+ putOverflowRecInFrag(signal);
+ }//if
+ }//if
+ }//if
+ }//if
+}//Dbacc::checkoverfreelist()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_LONG_PAGE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseLongPage(Signal* signal)
+{
+ DirRangePtr rlpOverflowrangeptr;
+ DirectoryarrayPtr rlpOverflowDirptr;
+ Uint32 trlpTmp1;
+ Uint32 trlpTmp2;
+ Uint32 trlpTmp3;
+
+ jam();
+ seizeOverRec(signal);
+ sorOverflowRecPtr.p->dirindex = rlopPageptr.p->word32[ZPOS_PAGE_ID];
+ sorOverflowRecPtr.p->overpage = RNIL;
+ priOverflowRecPtr = sorOverflowRecPtr;
+ putRecInFreeOverdir(signal);
+ trlpTmp1 = sorOverflowRecPtr.p->dirindex;
+ rlpOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ trlpTmp2 = trlpTmp1 >> 8;
+ trlpTmp3 = trlpTmp1 & 0xff;
+ ptrCheckGuard(rlpOverflowrangeptr, cdirrangesize, dirRange);
+ arrGuard(trlpTmp2, 256);
+ rlpOverflowDirptr.i = rlpOverflowrangeptr.p->dirArray[trlpTmp2];
+ ptrCheckGuard(rlpOverflowDirptr, cdirarraysize, directoryarray);
+ rlpOverflowDirptr.p->pagep[trlpTmp3] = RNIL;
+
+ if (cundoLogActive != ZTRUE) {
+ // Remove from page array.
+ trfpArrayPos = rlopPageptr.p->word32[ZPOS_ARRAY_POS];
+ rfpPageptr = rlopPageptr;
+ removeFromPageArrayList(signal);
+ }
+
+ // Reset page header
+ iloPageptr = rlopPageptr;
+ tiloIndex = rlopPageptr.p->word32[ZPOS_PAGE_ID];
+ initLongOverpage(signal);
+
+ rpPageptr = rlopPageptr;
+ releasePage(signal);
+}//Dbacc::releaseLongPage()
+
+
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* */
+/* END OF DELETE MODULE */
+/* */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* */
+/* COMMIT AND ABORT MODULE */
+/* */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ABORT_OPERATION */
+/*DESCRIPTION: AN OPERATION RECORD CAN BE IN A LOCK QUEUE OF AN ELEMENT OR */
+/*OWNS THE LOCK. BY THIS SUBROUTINE THE LOCK STATE OF THE OPERATION WILL */
+/*BE CHECKED. THE OPERATION RECORD WILL BE REMOVED FROM THE QUEUE IF IT */
+/*BELONGED TO ANY ONE, OTHERWISE THE ELEMENT HEAD WILL BE UPDATED. */
+/* ------------------------------------------------------------------------- */
+void Dbacc::abortOperation(Signal* signal)
+{
+ OperationrecPtr aboOperRecPtr;
+ OperationrecPtr TaboOperRecPtr;
+ Page8Ptr aboPageidptr;
+ Uint32 taboElementptr;
+ Uint32 tmp2Olq;
+
+ if (operationRecPtr.p->lockOwner == ZTRUE) {
+ takeOutLockOwnersList(signal, operationRecPtr);
+ if (operationRecPtr.p->insertIsDone == ZTRUE) {
+ jam();
+ operationRecPtr.p->elementIsDisappeared = ZTRUE;
+ }//if
+ if ((operationRecPtr.p->nextParallelQue != RNIL) ||
+ (operationRecPtr.p->nextSerialQue != RNIL)) {
+ jam();
+ releaselock(signal);
+ } else {
+ /* --------------------------------------------------------------------------------- */
+ /* WE ARE OWNER OF THE LOCK AND NO OTHER OPERATIONS ARE QUEUED. IF INSERT OR STANDBY */
+ /* WE DELETE THE ELEMENT OTHERWISE WE REMOVE THE LOCK FROM THE ELEMENT. */
+ /* --------------------------------------------------------------------------------- */
+ if (operationRecPtr.p->elementIsDisappeared == ZFALSE) {
+ jam();
+ taboElementptr = operationRecPtr.p->elementPointer;
+ aboPageidptr.i = operationRecPtr.p->elementPage;
+ tmp2Olq = ElementHeader::setUnlocked(operationRecPtr.p->hashvaluePart,
+ operationRecPtr.p->scanBits);
+ ptrCheckGuard(aboPageidptr, cpagesize, page8);
+ dbgWord32(aboPageidptr, taboElementptr, tmp2Olq);
+ arrGuard(taboElementptr, 2048);
+ aboPageidptr.p->word32[taboElementptr] = tmp2Olq;
+ return;
+ } else {
+ jam();
+ commitdelete(signal, false);
+ }//if
+ }//if
+ } else {
+ /* --------------------------------------------------------------- */
+ // We are not the lock owner.
+ /* --------------------------------------------------------------- */
+ jam();
+ takeOutFragWaitQue(signal);
+ if (operationRecPtr.p->prevParallelQue != RNIL) {
+ jam();
+ /* ---------------------------------------------------------------------------------- */
+ /* SINCE WE ARE NOT QUEUE LEADER WE NEED NOT CONSIDER IF THE ELEMENT IS TO BE DELETED.*/
+ /* We will simply remove it from the parallel list without any other rearrangements. */
+ /* ---------------------------------------------------------------------------------- */
+ aboOperRecPtr.i = operationRecPtr.p->prevParallelQue;
+ ptrCheckGuard(aboOperRecPtr, coprecsize, operationrec);
+ aboOperRecPtr.p->nextParallelQue = operationRecPtr.p->nextParallelQue;
+ if (operationRecPtr.p->nextParallelQue != RNIL) {
+ jam();
+ aboOperRecPtr.i = operationRecPtr.p->nextParallelQue;
+ ptrCheckGuard(aboOperRecPtr, coprecsize, operationrec);
+ aboOperRecPtr.p->prevParallelQue = operationRecPtr.p->prevParallelQue;
+ }//if
+ } else if (operationRecPtr.p->prevSerialQue != RNIL) {
+ /* ------------------------------------------------------------------------- */
+ // We are not in the parallel queue owning the lock. Thus we are in another parallel
+ // queue longer down in the serial queue. We are however first since prevParallelQue
+ // == RNIL.
+ /* ------------------------------------------------------------------------- */
+ if (operationRecPtr.p->nextParallelQue != RNIL) {
+ jam();
+ /* ------------------------------------------------------------------------- */
+ // We have an operation in the queue after us. We simply rearrange this parallel queue.
+ // The new leader of this parallel queue will be operation in the serial queue.
+ /* ------------------------------------------------------------------------- */
+ aboOperRecPtr.i = operationRecPtr.p->nextParallelQue;
+ ptrCheckGuard(aboOperRecPtr, coprecsize, operationrec);
+ aboOperRecPtr.p->nextSerialQue = operationRecPtr.p->nextSerialQue;
+ aboOperRecPtr.p->prevSerialQue = operationRecPtr.p->prevSerialQue;
+ aboOperRecPtr.p->prevParallelQue = RNIL; // Queue Leader
+ if (operationRecPtr.p->nextSerialQue != RNIL) {
+ jam();
+ TaboOperRecPtr.i = operationRecPtr.p->nextSerialQue;
+ ptrCheckGuard(TaboOperRecPtr, coprecsize, operationrec);
+ TaboOperRecPtr.p->prevSerialQue = aboOperRecPtr.i;
+ }//if
+ TaboOperRecPtr.i = operationRecPtr.p->prevSerialQue;
+ ptrCheckGuard(TaboOperRecPtr, coprecsize, operationrec);
+ TaboOperRecPtr.p->nextSerialQue = aboOperRecPtr.i;
+ } else {
+ jam();
+ /* ------------------------------------------------------------------------- */
+ // We are the only operation in this parallel queue. We will thus shrink the serial
+ // queue.
+ /* ------------------------------------------------------------------------- */
+ aboOperRecPtr.i = operationRecPtr.p->prevSerialQue;
+ ptrCheckGuard(aboOperRecPtr, coprecsize, operationrec);
+ aboOperRecPtr.p->nextSerialQue = operationRecPtr.p->nextSerialQue;
+ if (operationRecPtr.p->nextSerialQue != RNIL) {
+ jam();
+ aboOperRecPtr.i = operationRecPtr.p->nextSerialQue;
+ ptrCheckGuard(aboOperRecPtr, coprecsize, operationrec);
+ aboOperRecPtr.p->prevSerialQue = operationRecPtr.p->prevSerialQue;
+ }//if
+ }//if
+ }//if
+ }//if
+ /* ------------------------------------------------------------------------- */
+ // If prevParallelQue = RNIL and prevSerialQue = RNIL and we are not owner of the
+ // lock then we cannot be in any lock queue at all.
+ /* ------------------------------------------------------------------------- */
+}//Dbacc::abortOperation()
+
+void Dbacc::commitDeleteCheck()
+{
+ OperationrecPtr opPtr;
+ OperationrecPtr lastOpPtr;
+ OperationrecPtr deleteOpPtr;
+ bool elementDeleted = false;
+ bool deleteCheckOngoing = true;
+ Uint32 hashValue = 0;
+ lastOpPtr = operationRecPtr;
+ opPtr.i = operationRecPtr.p->nextParallelQue;
+ while (opPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(opPtr, coprecsize, operationrec);
+ lastOpPtr = opPtr;
+ opPtr.i = opPtr.p->nextParallelQue;
+ }//while
+ deleteOpPtr = lastOpPtr;
+ do {
+ if (deleteOpPtr.p->operation == ZDELETE) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* IF THE CURRENT OPERATION TO BE COMMITTED IS A DELETE OPERATION DUE TO A */
+ /* SCAN-TAKEOVER THE ACTUAL DELETE WILL BE PERFORMED BY THE PREVIOUS OPERATION (SCAN)*/
+ /* IN THE PARALLEL QUEUE WHICH OWNS THE LOCK.THE PROBLEM IS THAT THE SCAN OPERATION */
+ /* DOES NOT HAVE A HASH VALUE ASSIGNED TO IT SO WE COPY IT FROM THIS OPERATION. */
+ /* */
+ /* WE ASSUME THAT THIS SOLUTION WILL WORK BECAUSE THE ONLY WAY A SCAN CAN PERFORM */
+ /* A DELETE IS BY BEING FOLLOWED BY A NORMAL DELETE-OPERATION THAT HAS A HASH VALUE. */
+ /* --------------------------------------------------------------------------------- */
+ hashValue = deleteOpPtr.p->hashValue;
+ elementDeleted = true;
+ deleteCheckOngoing = false;
+ } else if ((deleteOpPtr.p->operation == ZREAD) ||
+ (deleteOpPtr.p->operation == ZSCAN_OP)) {
+ /* --------------------------------------------------------------------------------- */
+ /* We are trying to find out whether the commit will in the end delete the tuple. */
+ /* Normally the delete will be the last operation in the list of operations on this */
+ /* It is however possible to issue reads and scans in the same savepoint as the */
+ /* delete operation was issued and these can end up after the delete in the list of */
+ /* operations in the parallel queue. Thus if we discover a read or a scan we have to */
+ /* continue scanning the list looking for a delete operation. */
+ /* --------------------------------------------------------------------------------- */
+ deleteOpPtr.i = deleteOpPtr.p->prevParallelQue;
+ if (deleteOpPtr.i == RNIL) {
+ jam();
+ deleteCheckOngoing = false;
+ } else {
+ jam();
+ ptrCheckGuard(deleteOpPtr, coprecsize, operationrec);
+ }//if
+ } else {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* Finding an UPDATE or INSERT before finding a DELETE means we cannot be deleting */
+ /* as the end result of this transaction. */
+ /* --------------------------------------------------------------------------------- */
+ deleteCheckOngoing = false;
+ }//if
+ } while (deleteCheckOngoing);
+ opPtr = lastOpPtr;
+ do {
+ jam();
+ opPtr.p->commitDeleteCheckFlag = ZTRUE;
+ if (elementDeleted) {
+ jam();
+ opPtr.p->elementIsDisappeared = ZTRUE;
+ opPtr.p->hashValue = hashValue;
+ }//if
+ opPtr.i = opPtr.p->prevParallelQue;
+ if (opPtr.i == RNIL) {
+ jam();
+ break;
+ }//if
+ ptrCheckGuard(opPtr, coprecsize, operationrec);
+ } while (true);
+}//Dbacc::commitDeleteCheck()
+
+/* ------------------------------------------------------------------------- */
+/* COMMIT_OPERATION */
+/* INPUT: OPERATION_REC_PTR, POINTER TO AN OPERATION RECORD */
+/* DESCRIPTION: THE OPERATION RECORD WILL BE TAKE OUT OF ANY LOCK QUEUE. */
+/* IF IT OWNS THE ELEMENT LOCK. HEAD OF THE ELEMENT WILL BE UPDATED. */
+/* ------------------------------------------------------------------------- */
+void Dbacc::commitOperation(Signal* signal)
+{
+ OperationrecPtr tolqTmpPtr;
+ Page8Ptr coPageidptr;
+ Uint32 tcoElementptr;
+ Uint32 tmp2Olq;
+
+ if ((operationRecPtr.p->commitDeleteCheckFlag == ZFALSE) &&
+ (operationRecPtr.p->operation != ZSCAN_OP)) {
+ jam();
+ /* This method is used to check whether the end result of the transaction
+ will be to delete the tuple. In this case all operation will be marked
+ with elementIsDisappeared = true to ensure that the last operation
+ committed will remove the tuple. We only run this once per transaction
+ (commitDeleteCheckFlag = true if performed earlier) and we don't
+ execute this code when committing a scan operation since committing
+ a scan operation only means that the scan is continuing and the scan
+ lock is released.
+ */
+ commitDeleteCheck();
+ }//if
+ if (operationRecPtr.p->lockOwner == ZTRUE) {
+ takeOutLockOwnersList(signal, operationRecPtr);
+ if ((operationRecPtr.p->nextParallelQue == RNIL) &&
+ (operationRecPtr.p->nextSerialQue == RNIL) &&
+ (operationRecPtr.p->elementIsDisappeared == ZFALSE)) {
+ /*
+ This is the normal path through the commit for operations owning the
+ lock without any queues and not a delete operation.
+ */
+ coPageidptr.i = operationRecPtr.p->elementPage;
+ tcoElementptr = operationRecPtr.p->elementPointer;
+ tmp2Olq = ElementHeader::setUnlocked(operationRecPtr.p->hashvaluePart,
+ operationRecPtr.p->scanBits);
+ ptrCheckGuard(coPageidptr, cpagesize, page8);
+ dbgWord32(coPageidptr, tcoElementptr, tmp2Olq);
+ arrGuard(tcoElementptr, 2048);
+ coPageidptr.p->word32[tcoElementptr] = tmp2Olq;
+ return;
+ } else if ((operationRecPtr.p->nextParallelQue != RNIL) ||
+ (operationRecPtr.p->nextSerialQue != RNIL)) {
+ jam();
+ /*
+ The case when there is a queue lined up.
+ Release the lock and pass it to the next operation lined up.
+ */
+ releaselock(signal);
+ return;
+ } else {
+ jam();
+ /*
+ No queue and elementIsDisappeared is true. We perform the actual delete
+ operation.
+ */
+ commitdelete(signal, false);
+ return;
+ }//if
+ } else {
+ /*
+ THE OPERATION DOES NOT OWN THE LOCK. IT MUST BE IN A LOCK QUEUE OF THE
+ ELEMENT.
+ */
+ ndbrequire(operationRecPtr.p->prevParallelQue != RNIL);
+ jam();
+ tolqTmpPtr.i = operationRecPtr.p->prevParallelQue;
+ ptrCheckGuard(tolqTmpPtr, coprecsize, operationrec);
+ tolqTmpPtr.p->nextParallelQue = operationRecPtr.p->nextParallelQue;
+ if (operationRecPtr.p->nextParallelQue != RNIL) {
+ jam();
+ tolqTmpPtr.i = operationRecPtr.p->nextParallelQue;
+ ptrCheckGuard(tolqTmpPtr, coprecsize, operationrec);
+ tolqTmpPtr.p->prevParallelQue = operationRecPtr.p->prevParallelQue;
+ }//if
+ }//if
+}//Dbacc::commitOperation()
+
+/* ------------------------------------------------------------------------- */
+/* RELEASELOCK */
+/* RESETS LOCK OF AN ELEMENT. */
+/* INFORMATION ABOUT THE ELEMENT IS SAVED IN THE OPERATION RECORD */
+/* THESE INFORMATION IS USED TO UPDATE HEADER OF THE ELEMENT */
+/* ------------------------------------------------------------------------- */
+void Dbacc::releaselock(Signal* signal)
+{
+ OperationrecPtr rloOperPtr;
+ OperationrecPtr trlOperPtr;
+ OperationrecPtr trlTmpOperPtr;
+ Uint32 TelementIsDisappeared;
+
+ trlOperPtr.i = RNIL;
+ if (operationRecPtr.p->nextParallelQue != RNIL) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* NEXT OPERATION TAKES OVER THE LOCK. We will simply move the info from the leader */
+ // to the new queue leader.
+ /* --------------------------------------------------------------------------------- */
+ trlOperPtr.i = operationRecPtr.p->nextParallelQue;
+ ptrCheckGuard(trlOperPtr, coprecsize, operationrec);
+ copyInOperPtr = trlOperPtr;
+ copyOperPtr = operationRecPtr;
+ copyOpInfo(signal);
+ trlOperPtr.p->prevParallelQue = RNIL;
+ if (operationRecPtr.p->nextSerialQue != RNIL) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THERE IS A SERIAL QUEUE. MOVE IT FROM RELEASED OP REC TO THE NEW LOCK OWNER. */
+ /* --------------------------------------------------------------------------------- */
+ trlOperPtr.p->nextSerialQue = operationRecPtr.p->nextSerialQue;
+ trlTmpOperPtr.i = trlOperPtr.p->nextSerialQue;
+ ptrCheckGuard(trlTmpOperPtr, coprecsize, operationrec);
+ trlTmpOperPtr.p->prevSerialQue = trlOperPtr.i;
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* SINCE THERE ARE STILL ITEMS IN THE PARALLEL QUEUE WE NEED NOT WORRY ABOUT */
+ /* STARTING QUEUED OPERATIONS. THUS WE CAN END HERE. */
+ /* --------------------------------------------------------------------------------- */
+ } else {
+ ndbrequire(operationRecPtr.p->nextSerialQue != RNIL);
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE PARALLEL QUEUE IS EMPTY AND THE SERIAL QUEUE IS NOT EMPTY. WE NEED TO */
+ /* REARRANGE LISTS AND START A NUMBER OF OPERATIONS. */
+ /* --------------------------------------------------------------------------------- */
+ trlOperPtr.i = operationRecPtr.p->nextSerialQue;
+ ptrCheckGuard(trlOperPtr, coprecsize, operationrec);
+ copyOperPtr = operationRecPtr;
+ copyInOperPtr = trlOperPtr;
+ copyOpInfo(signal);
+ trlOperPtr.p->prevSerialQue = RNIL;
+ ndbrequire(trlOperPtr.p->prevParallelQue == RNIL);
+ /* --------------------------------------------------------------------------------- */
+ /* WE HAVE MOVED TO THE NEXT PARALLEL QUEUE. WE MUST START ALL OF THOSE */
+ /* OPERATIONS WHICH UP TILL NOW HAVE BEEN QUEUED WAITING FOR THE LOCK. */
+ /* --------------------------------------------------------------------------------- */
+ rloOperPtr = operationRecPtr;
+ trlTmpOperPtr = trlOperPtr;
+ TelementIsDisappeared = trlOperPtr.p->elementIsDisappeared;
+ Uint32 ThashValue = trlOperPtr.p->hashValue;
+ do {
+ /* --------------------------------------------------------------------------------- */
+ // Ensure that all operations in the queue are assigned with the elementIsDisappeared
+ // to ensure that the element is removed after a previous delete. An insert does
+ // however revert this decision since the element is put back again. Local checkpoints
+ // complicate life here since they do not execute the next operation but simply change
+ // the state on the operation. We need to set-up the variable elementIsDisappeared
+ // properly even when local checkpoints and inserts/writes after deletes occur.
+ /* --------------------------------------------------------------------------------- */
+ trlTmpOperPtr.p->elementIsDisappeared = TelementIsDisappeared;
+ if (TelementIsDisappeared == ZTRUE) {
+ /* --------------------------------------------------------------------------------- */
+ // If the elementIsDisappeared is set then we know that the hashValue is also set
+ // since it always originates from a committing abort or a aborting insert. Scans
+ // do not initialise the hashValue and must have this value initialised if they are
+ // to successfully commit the delete.
+ /* --------------------------------------------------------------------------------- */
+ jam();
+ trlTmpOperPtr.p->hashValue = ThashValue;
+ }//if
+ trlTmpOperPtr.p->localdata[0] = trlOperPtr.p->localdata[0];
+ trlTmpOperPtr.p->localdata[1] = trlOperPtr.p->localdata[1];
+ /* --------------------------------------------------------------------------------- */
+ // Restart the queued operation.
+ /* --------------------------------------------------------------------------------- */
+ operationRecPtr = trlTmpOperPtr;
+ TelementIsDisappeared = executeNextOperation(signal);
+ ThashValue = operationRecPtr.p->hashValue;
+ if (trlTmpOperPtr.p->nextParallelQue != RNIL) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // We will continue with the next operation in the parallel queue and start this as
+ // well.
+ /* --------------------------------------------------------------------------------- */
+ trlTmpOperPtr.i = trlTmpOperPtr.p->nextParallelQue;
+ ptrCheckGuard(trlTmpOperPtr, coprecsize, operationrec);
+ } else {
+ jam();
+ break;
+ }//if
+ } while (1);
+ operationRecPtr = rloOperPtr;
+ }//if
+
+ // Insert the next op into the lock owner list
+ insertLockOwnersList(signal, trlOperPtr);
+ return;
+}//Dbacc::releaselock()
+
+/* --------------------------------------------------------------------------------- */
+/* COPY_OP_INFO */
+/* INPUT: COPY_IN_OPER_PTR AND COPY_OPER_PTR. */
+/* DESCRIPTION:INFORMATION ABOUT THE ELEMENT WILL BE MOVED FROM OPERATION */
+/* REC TO QUEUE OP REC. QUE OP REC TAKES OVER THE LOCK. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::copyOpInfo(Signal* signal)
+{
+ Page8Ptr coiPageidptr;
+
+ copyInOperPtr.p->elementPage = copyOperPtr.p->elementPage;
+ copyInOperPtr.p->elementIsforward = copyOperPtr.p->elementIsforward;
+ copyInOperPtr.p->elementContainer = copyOperPtr.p->elementContainer;
+ copyInOperPtr.p->elementPointer = copyOperPtr.p->elementPointer;
+ copyInOperPtr.p->scanBits = copyOperPtr.p->scanBits;
+ copyInOperPtr.p->hashvaluePart = copyOperPtr.p->hashvaluePart;
+ copyInOperPtr.p->elementIsDisappeared = copyOperPtr.p->elementIsDisappeared;
+ if (copyInOperPtr.p->elementIsDisappeared == ZTRUE) {
+ /* --------------------------------------------------------------------------------- */
+ // If the elementIsDisappeared is set then we know that the hashValue is also set
+ // since it always originates from a committing abort or a aborting insert. Scans
+ // do not initialise the hashValue and must have this value initialised if they are
+ // to successfully commit the delete.
+ /* --------------------------------------------------------------------------------- */
+ jam();
+ copyInOperPtr.p->hashValue = copyOperPtr.p->hashValue;
+ }//if
+ coiPageidptr.i = copyOperPtr.p->elementPage;
+ ptrCheckGuard(coiPageidptr, cpagesize, page8);
+ const Uint32 tmp = ElementHeader::setLocked(copyInOperPtr.i);
+ dbgWord32(coiPageidptr, copyOperPtr.p->elementPointer, tmp);
+ arrGuard(copyOperPtr.p->elementPointer, 2048);
+ coiPageidptr.p->word32[copyOperPtr.p->elementPointer] = tmp;
+ copyInOperPtr.p->localdata[0] = copyOperPtr.p->localdata[0];
+ copyInOperPtr.p->localdata[1] = copyOperPtr.p->localdata[1];
+}//Dbacc::copyOpInfo()
+
+/* ******************--------------------------------------------------------------- */
+/* EXECUTE NEXT OPERATION */
+/* NEXT OPERATION IN A LOCK QUEUE WILL BE EXECUTED. */
+/* --------------------------------------------------------------------------------- */
+Uint32 Dbacc::executeNextOperation(Signal* signal)
+{
+ ndbrequire(operationRecPtr.p->transactionstate == ACTIVE);
+ if (fragrecptr.p->stopQueOp == ZTRUE) {
+ Uint32 TelemDisappeared;
+ jam();
+ TelemDisappeared = operationRecPtr.p->elementIsDisappeared;
+ if ((operationRecPtr.p->elementIsDisappeared == ZTRUE) &&
+ (operationRecPtr.p->prevParallelQue == RNIL) &&
+ ((operationRecPtr.p->operation == ZINSERT) ||
+ (operationRecPtr.p->operation == ZWRITE))) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // In this case we do not wish to change the elementIsDisappeared since that would
+ // create an error the next time this method is called for this operation after local
+ // checkpoint starts up operations again. We must however ensure that operations
+ // that follow in the queue do not get the value ZTRUE when actually an INSERT/WRITE
+ // precedes them (only if the INSERT/WRITE is the first operation).
+ /* --------------------------------------------------------------------------------- */
+ TelemDisappeared = ZFALSE;
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* A LOCAL CHECKPOINT HAS STOPPED OPERATIONS. WE MUST NOT START THE OPERATION */
+ /* AT THIS TIME. WE SET THE STATE TO INDICATE THAT WE ARE READY TO START AS */
+ /* SOON AS WE ARE ALLOWED. */
+ /* --------------------------------------------------------------------------------- */
+ operationRecPtr.p->opState = WAIT_EXE_OP;
+ return TelemDisappeared;
+ }//if
+ takeOutFragWaitQue(signal);
+ if (operationRecPtr.p->elementIsDisappeared == ZTRUE) {
+ /* --------------------------------------------------------------------------------- */
+ /* PREVIOUS OPERATION WAS DELETE OPERATION AND THE ELEMENT IS ALREADY DELETED. */
+ /* --------------------------------------------------------------------------------- */
+ if (((operationRecPtr.p->operation != ZINSERT) &&
+ (operationRecPtr.p->operation != ZWRITE)) ||
+ (operationRecPtr.p->prevParallelQue != RNIL)) {
+ if (operationRecPtr.p->operation != ZSCAN_OP ||
+ operationRecPtr.p->isAccLockReq) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // Updates and reads with a previous delete simply aborts with read error indicating
+ // that tuple did not exist. Also inserts and writes not being the first operation.
+ /* --------------------------------------------------------------------------------- */
+ operationRecPtr.p->transactionstate = WAIT_COMMIT_ABORT;
+ signal->theData[0] = operationRecPtr.p->userptr;
+ signal->theData[1] = ZREAD_ERROR;
+ sendSignal(operationRecPtr.p->userblockref, GSN_ACCKEYREF, signal, 2, JBB);
+ return operationRecPtr.p->elementIsDisappeared;
+ } else {
+ /* --------------------------------------------------------------------------------- */
+ /* ABORT OF OPERATION NEEDED BUT THE OPERATION IS A SCAN => SPECIAL TREATMENT. */
+ /* IF THE SCAN WAITS IN QUEUE THEN WE MUST REMOVE THE OPERATION FROM THE SCAN */
+ /* LOCK QUEUE AND IF NO MORE OPERATIONS ARE QUEUED THEN WE SHOULD RESTART THE */
+ /* SCAN PROCESS. OTHERWISE WE SIMPLY RELEASE THE OPERATION AND DECREASE THE */
+ /* NUMBER OF LOCKS HELD. */
+ /* --------------------------------------------------------------------------------- */
+ takeOutScanLockQueue(operationRecPtr.p->scanRecPtr);
+ putReadyScanQueue(signal, operationRecPtr.p->scanRecPtr);
+ return operationRecPtr.p->elementIsDisappeared;
+ }//if
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ // Insert and writes can continue but need to be converted to inserts.
+ /* --------------------------------------------------------------------------------- */
+ jam();
+ operationRecPtr.p->elementIsDisappeared = ZFALSE;
+ operationRecPtr.p->operation = ZINSERT;
+ operationRecPtr.p->insertIsDone = ZTRUE;
+ } else if (operationRecPtr.p->operation == ZINSERT) {
+ bool abortFlag = true;
+ if (operationRecPtr.p->prevParallelQue != RNIL) {
+ OperationrecPtr prevOpPtr;
+ jam();
+ prevOpPtr.i = operationRecPtr.p->prevParallelQue;
+ ptrCheckGuard(prevOpPtr, coprecsize, operationrec);
+ if (prevOpPtr.p->operation == ZDELETE) {
+ jam();
+ abortFlag = false;
+ }//if
+ }//if
+ if (abortFlag) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* ELEMENT STILL REMAINS AND WE ARE TRYING TO INSERT IT AGAIN. THIS IS CLEARLY */
+ /* NOT A GOOD IDEA. */
+ /* --------------------------------------------------------------------------------- */
+ operationRecPtr.p->transactionstate = WAIT_COMMIT_ABORT;
+ signal->theData[0] = operationRecPtr.p->userptr;
+ signal->theData[1] = ZWRITE_ERROR;
+ sendSignal(operationRecPtr.p->userblockref, GSN_ACCKEYREF, signal, 2, JBB);
+ return operationRecPtr.p->elementIsDisappeared;
+ }//if
+ }//if
+ if (operationRecPtr.p->operation == ZSCAN_OP &&
+ ! operationRecPtr.p->isAccLockReq) {
+ jam();
+ takeOutScanLockQueue(operationRecPtr.p->scanRecPtr);
+ putReadyScanQueue(signal, operationRecPtr.p->scanRecPtr);
+ } else {
+ jam();
+ sendAcckeyconf(signal);
+ sendSignal(operationRecPtr.p->userblockref, GSN_ACCKEYCONF, signal, 6, JBB);
+ }//if
+ return operationRecPtr.p->elementIsDisappeared;
+}//Dbacc::executeNextOperation()
+
+/* --------------------------------------------------------------------------------- */
+/* TAKE_OUT_FRAG_WAIT_QUE */
+/* DESCRIPTION: AN OPERATION WHICH OWNS A LOCK OF AN ELEMENT, IS IN A LIST */
+/* OF THE FRAGMENT. THIS LIST IS USED TO STOP THE QUEUE OPERATION */
+/* DURING CREATE CHECK POINT PROSESS FOR STOP AND RESTART OF THE */
+/* OPERATIONS. THIS SUBRUTIN TAKES A OPERATION RECORD OUT OF THE LIST */
+/* -------------------------------------------------------------------------------- */
+void Dbacc::takeOutFragWaitQue(Signal* signal)
+{
+ OperationrecPtr tofwqOperRecPtr;
+
+ if (operationRecPtr.p->opState == WAIT_IN_QUEUE) {
+ if (fragrecptr.p->sentWaitInQueOp == operationRecPtr.i) {
+ jam();
+ fragrecptr.p->sentWaitInQueOp = operationRecPtr.p->nextQueOp;
+ }//if
+ if (operationRecPtr.p->prevQueOp != RNIL) {
+ jam();
+ tofwqOperRecPtr.i = operationRecPtr.p->prevQueOp;
+ ptrCheckGuard(tofwqOperRecPtr, coprecsize, operationrec);
+ tofwqOperRecPtr.p->nextQueOp = operationRecPtr.p->nextQueOp;
+ } else {
+ jam();
+ fragrecptr.p->firstWaitInQueOp = operationRecPtr.p->nextQueOp;
+ }//if
+ if (operationRecPtr.p->nextQueOp != RNIL) {
+ jam();
+ tofwqOperRecPtr.i = operationRecPtr.p->nextQueOp;
+ ptrCheckGuard(tofwqOperRecPtr, coprecsize, operationrec);
+ tofwqOperRecPtr.p->prevQueOp = operationRecPtr.p->prevQueOp;
+ } else {
+ jam();
+ fragrecptr.p->lastWaitInQueOp = operationRecPtr.p->prevQueOp;
+ }//if
+ operationRecPtr.p->opState = FREE_OP;
+ return;
+ } else {
+ ndbrequire(operationRecPtr.p->opState == FREE_OP);
+ }//if
+}//Dbacc::takeOutFragWaitQue()
+
+/**
+ * takeOutLockOwnersList
+ *
+ * Description: Take out an operation from the doubly linked
+ * lock owners list on the fragment.
+ *
+ */
+void Dbacc::takeOutLockOwnersList(Signal* signal,
+ const OperationrecPtr& outOperPtr)
+{
+ const Uint32 Tprev = outOperPtr.p->prevLockOwnerOp;
+ const Uint32 Tnext = outOperPtr.p->nextLockOwnerOp;
+
+#ifdef VM_TRACE
+ // Check that operation is already in the list
+ OperationrecPtr tmpOperPtr;
+ bool inList = false;
+ tmpOperPtr.i = fragrecptr.p->lockOwnersList;
+ while (tmpOperPtr.i != RNIL){
+ ptrCheckGuard(tmpOperPtr, coprecsize, operationrec);
+ if (tmpOperPtr.i == outOperPtr.i)
+ inList = true;
+ tmpOperPtr.i = tmpOperPtr.p->nextLockOwnerOp;
+ }
+ ndbrequire(inList == true);
+#endif
+
+ ndbrequire(outOperPtr.p->lockOwner == ZTRUE);
+ outOperPtr.p->lockOwner = ZFALSE;
+
+ // Fast path through the code for the common case.
+ if ((Tprev == RNIL) && (Tnext == RNIL)) {
+ ndbrequire(fragrecptr.p->lockOwnersList == outOperPtr.i);
+ fragrecptr.p->lockOwnersList = RNIL;
+ return;
+ }
+
+ // Check previous operation
+ if (Tprev != RNIL) {
+ jam();
+ arrGuard(Tprev, coprecsize);
+ operationrec[Tprev].nextLockOwnerOp = Tnext;
+ } else {
+ fragrecptr.p->lockOwnersList = Tnext;
+ }//if
+
+ // Check next operation
+ if (Tnext == RNIL) {
+ return;
+ } else {
+ jam();
+ arrGuard(Tnext, coprecsize);
+ operationrec[Tnext].prevLockOwnerOp = Tprev;
+ }//if
+
+ return;
+}//Dbacc::takeOutLockOwnersList()
+
+/**
+ * insertLockOwnersList
+ *
+ * Description: Insert an operation first in the dubly linked lock owners
+ * list on the fragment.
+ *
+ */
+void Dbacc::insertLockOwnersList(Signal* signal,
+ const OperationrecPtr& insOperPtr)
+{
+ OperationrecPtr tmpOperPtr;
+
+#ifdef VM_TRACE
+ // Check that operation is not already in list
+ tmpOperPtr.i = fragrecptr.p->lockOwnersList;
+ while(tmpOperPtr.i != RNIL){
+ ptrCheckGuard(tmpOperPtr, coprecsize, operationrec);
+ ndbrequire(tmpOperPtr.i != insOperPtr.i);
+ tmpOperPtr.i = tmpOperPtr.p->nextLockOwnerOp;
+ }
+#endif
+
+ ndbrequire(insOperPtr.p->lockOwner == ZFALSE);
+
+ insOperPtr.p->lockOwner = ZTRUE;
+ insOperPtr.p->prevLockOwnerOp = RNIL;
+ tmpOperPtr.i = fragrecptr.p->lockOwnersList;
+ fragrecptr.p->lockOwnersList = insOperPtr.i;
+ insOperPtr.p->nextLockOwnerOp = tmpOperPtr.i;
+ if (tmpOperPtr.i == RNIL) {
+ return;
+ } else {
+ jam();
+ ptrCheckGuard(tmpOperPtr, coprecsize, operationrec);
+ tmpOperPtr.p->prevLockOwnerOp = insOperPtr.i;
+ }//if
+}//Dbacc::insertLockOwnersList()
+
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF COMMIT AND ABORT MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* ALLOC_OVERFLOW_PAGE */
+/* DESCRIPTION: */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::allocOverflowPage(Signal* signal)
+{
+ DirRangePtr aopDirRangePtr;
+ DirectoryarrayPtr aopOverflowDirptr;
+ OverflowRecordPtr aopOverflowRecPtr;
+ Uint32 taopTmp1;
+ Uint32 taopTmp2;
+ Uint32 taopTmp3;
+
+ tresult = 0;
+ if ((cfirstfreepage == RNIL) &&
+ (cfreepage >= cpagesize)) {
+ jam();
+ zpagesize_error("Dbacc::allocOverflowPage");
+ tresult = ZPAGESIZE_ERROR;
+ return;
+ }//if
+ if (fragrecptr.p->firstFreeDirindexRec != RNIL) {
+ jam();
+ /* FRAGRECPTR:FIRST_FREE_DIRINDEX_REC POINTS */
+ /* TO THE FIRST ELEMENT IN A FREE LIST OF THE */
+ /* DIRECTORY INDEX WICH HAVE NULL AS PAGE */
+ aopOverflowRecPtr.i = fragrecptr.p->firstFreeDirindexRec;
+ ptrCheckGuard(aopOverflowRecPtr, coverflowrecsize, overflowRecord);
+ troOverflowRecPtr.p = aopOverflowRecPtr.p;
+ takeRecOutOfFreeOverdir(signal);
+ } else if (cfirstfreeoverrec == RNIL) {
+ jam();
+ tresult = ZOVER_REC_ERROR;
+ return;
+ } else if ((cfirstfreedir == RNIL) &&
+ (cdirarraysize <= cdirmemory)) {
+ jam();
+ tresult = ZDIRSIZE_ERROR;
+ return;
+ } else {
+ jam();
+ seizeOverRec(signal);
+ aopOverflowRecPtr = sorOverflowRecPtr;
+ aopOverflowRecPtr.p->dirindex = fragrecptr.p->lastOverIndex;
+ }//if
+ aopOverflowRecPtr.p->nextOverRec = RNIL;
+ aopOverflowRecPtr.p->prevOverRec = RNIL;
+ fragrecptr.p->firstOverflowRec = aopOverflowRecPtr.i;
+ fragrecptr.p->lastOverflowRec = aopOverflowRecPtr.i;
+ taopTmp1 = aopOverflowRecPtr.p->dirindex;
+ aopDirRangePtr.i = fragrecptr.p->overflowdir;
+ taopTmp2 = taopTmp1 >> 8;
+ taopTmp3 = taopTmp1 & 0xff;
+ ptrCheckGuard(aopDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(taopTmp2, 256);
+ if (aopDirRangePtr.p->dirArray[taopTmp2] == RNIL) {
+ jam();
+ seizeDirectory(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ aopDirRangePtr.p->dirArray[taopTmp2] = sdDirptr.i;
+ }//if
+ aopOverflowDirptr.i = aopDirRangePtr.p->dirArray[taopTmp2];
+ seizePage(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ ptrCheckGuard(aopOverflowDirptr, cdirarraysize, directoryarray);
+ aopOverflowDirptr.p->pagep[taopTmp3] = spPageptr.i;
+ tiopPageId = aopOverflowRecPtr.p->dirindex;
+ iopOverflowRecPtr = aopOverflowRecPtr;
+ iopPageptr = spPageptr;
+ initOverpage(signal);
+ aopOverflowRecPtr.p->overpage = spPageptr.i;
+ if (fragrecptr.p->lastOverIndex <= aopOverflowRecPtr.p->dirindex) {
+ jam();
+ ndbrequire(fragrecptr.p->lastOverIndex == aopOverflowRecPtr.p->dirindex);
+ fragrecptr.p->lastOverIndex++;
+ }//if
+}//Dbacc::allocOverflowPage()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* EXPAND/SHRINK MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/*EXPANDCHECK EXPAND BUCKET ORD */
+/* SENDER: ACC, LEVEL B */
+/* INPUT: FRAGRECPTR, POINTS TO A FRAGMENT RECORD. */
+/* DESCRIPTION: A BUCKET OF A FRAGMENT PAGE WILL BE EXPAND INTO TWO BUCKETS */
+/* ACCORDING TO LH3. */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* EXPANDCHECK EXPAND BUCKET ORD */
+/* ******************------------------------------+ */
+/* SENDER: ACC, LEVEL B */
+/* A BUCKET OF THE FRAGMENT WILL */
+/* BE EXPANDED ACORDING TO LH3, */
+/* AND COMMIT TRANSACTION PROCESS */
+/* WILL BE CONTINUED */
+Uint32 Dbacc::checkScanExpand(Signal* signal)
+{
+ Uint32 Ti;
+ Uint32 TreturnCode = 0;
+ Uint32 TPageIndex;
+ Uint32 TDirInd;
+ Uint32 TSplit;
+ Uint32 TreleaseInd = 0;
+ Uint32 TreleaseScanBucket;
+ Uint32 TreleaseScanIndicator[4];
+ DirectoryarrayPtr TDirptr;
+ DirRangePtr TDirRangePtr;
+ Page8Ptr TPageptr;
+ ScanRecPtr TscanPtr;
+ RootfragmentrecPtr Trootfragrecptr;
+
+ Trootfragrecptr.i = fragrecptr.p->myroot;
+ TSplit = fragrecptr.p->p;
+ ptrCheckGuard(Trootfragrecptr, crootfragmentsize, rootfragmentrec);
+ for (Ti = 0; Ti < 4; Ti++) {
+ TreleaseScanIndicator[Ti] = 0;
+ if (Trootfragrecptr.p->scan[Ti] != RNIL) {
+ //-------------------------------------------------------------
+ // A scan is ongoing on this particular local fragment. We have
+ // to check its current state.
+ //-------------------------------------------------------------
+ TscanPtr.i = Trootfragrecptr.p->scan[Ti];
+ ptrCheckGuard(TscanPtr, cscanRecSize, scanRec);
+ if (TscanPtr.p->activeLocalFrag == fragrecptr.i) {
+ if (TscanPtr.p->scanBucketState == ScanRec::FIRST_LAP) {
+ if (TSplit == TscanPtr.p->nextBucketIndex) {
+ jam();
+ //-------------------------------------------------------------
+ // We are currently scanning this bucket. We cannot split it
+ // simultaneously with the scan. We have to pass this offer for
+ // splitting the bucket.
+ //-------------------------------------------------------------
+ TreturnCode = 1;
+ return TreturnCode;
+ } else if (TSplit > TscanPtr.p->nextBucketIndex) {
+ jam();
+ //-------------------------------------------------------------
+ // This bucket has not yet been scanned. We must reset the scanned
+ // bit indicator for this scan on this bucket.
+ //-------------------------------------------------------------
+ TreleaseScanIndicator[Ti] = 1;
+ TreleaseInd = 1;
+ } else {
+ jam();
+ }//if
+ } else if (TscanPtr.p->scanBucketState == ScanRec::SECOND_LAP) {
+ jam();
+ //-------------------------------------------------------------
+ // We are performing a second lap to handle buckets that was
+ // merged during the first lap of scanning. During this second
+ // lap we do not allow any splits or merges.
+ //-------------------------------------------------------------
+ TreturnCode = 1;
+ return TreturnCode;
+ } else {
+ ndbrequire(TscanPtr.p->scanBucketState == ScanRec::SCAN_COMPLETED);
+ jam();
+ //-------------------------------------------------------------
+ // The scan is completed and we can thus go ahead and perform
+ // the split.
+ //-------------------------------------------------------------
+ }//if
+ }//if
+ }//if
+ }//for
+ if (TreleaseInd == 1) {
+ TreleaseScanBucket = TSplit;
+ TDirRangePtr.i = fragrecptr.p->directory;
+ TPageIndex = TreleaseScanBucket & ((1 << fragrecptr.p->k) - 1); /* PAGE INDEX OBS K = 6 */
+ TDirInd = TreleaseScanBucket >> fragrecptr.p->k; /* DIRECTORY INDEX OBS K = 6 */
+ ptrCheckGuard(TDirRangePtr, cdirrangesize, dirRange);
+ arrGuard((TDirInd >> 8), 256);
+ TDirptr.i = TDirRangePtr.p->dirArray[TDirInd >> 8];
+ ptrCheckGuard(TDirptr, cdirarraysize, directoryarray);
+ TPageptr.i = TDirptr.p->pagep[TDirInd & 0xff];
+ ptrCheckGuard(TPageptr, cpagesize, page8);
+ for (Ti = 0; Ti < 4; Ti++) {
+ if (TreleaseScanIndicator[Ti] == 1) {
+ jam();
+ scanPtr.i = Trootfragrecptr.p->scan[Ti];
+ ptrCheckGuard(scanPtr, cscanRecSize, scanRec);
+ rsbPageidptr = TPageptr;
+ trsbPageindex = TPageIndex;
+ releaseScanBucket(signal);
+ }//if
+ }//for
+ }//if
+ return TreturnCode;
+}//Dbacc::checkScanExpand()
+
+void Dbacc::execEXPANDCHECK2(Signal* signal)
+{
+ DirectoryarrayPtr newDirptr;
+
+ jamEntry();
+ fragrecptr.i = signal->theData[0];
+ tresult = 0; /* 0= FALSE,1= TRUE,> ZLIMIT_OF_ERROR =ERRORCODE */
+ Uint32 tmp = 1;
+ tmp = tmp << 31;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ fragrecptr.p->expandFlag = 0;
+ if (fragrecptr.p->slack < tmp) {
+ jam();
+ /* IT MEANS THAT IF SLACK > ZERO */
+ /*--------------------------------------------------------------*/
+ /* THE SLACK HAS IMPROVED AND IS NOW ACCEPTABLE AND WE */
+ /* CAN FORGET ABOUT THE EXPAND PROCESS. */
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ if (fragrecptr.p->firstOverflowRec == RNIL) {
+ jam();
+ allocOverflowPage(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ /*--------------------------------------------------------------*/
+ /* WE COULD NOT ALLOCATE ANY OVERFLOW PAGE. THUS WE HAVE TO STOP*/
+ /* THE EXPAND SINCE WE CANNOT GUARANTEE ITS COMPLETION. */
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ }//if
+ if (cfirstfreepage == RNIL) {
+ if (cfreepage >= cpagesize) {
+ jam();
+ /*--------------------------------------------------------------*/
+ /* WE HAVE TO STOP THE EXPAND PROCESS SINCE THERE ARE NO FREE */
+ /* PAGES. THIS MEANS THAT WE COULD BE FORCED TO CRASH SINCE WE */
+ /* CANNOT COMPLETE THE EXPAND. TO AVOID THE CRASH WE EXIT HERE. */
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ }//if
+ if (checkScanExpand(signal) == 1) {
+ jam();
+ /*--------------------------------------------------------------*/
+ // A scan state was inconsistent with performing an expand
+ // operation.
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ if (remainingUndoPages() < ZMIN_UNDO_PAGES_AT_EXPAND) {
+ jam();
+ /*--------------------------------------------------------------*/
+ // We did not have enough undo log buffers to start up an
+ // expand operation
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ }//if
+ /*--------------------------------------------------------------------------*/
+ /* WE START BY FINDING THE PAGE, THE PAGE INDEX AND THE PAGE DIRECTORY*/
+ /* OF THE NEW BUCKET WHICH SHALL RECEIVE THE ELEMENT WHICH HAVE A 1 IN*/
+ /* THE NEXT HASH BIT. THIS BIT IS USED IN THE SPLIT MECHANISM TO */
+ /* DECIDE WHICH ELEMENT GOES WHERE. */
+ /*--------------------------------------------------------------------------*/
+ expDirRangePtr.i = fragrecptr.p->directory;
+ texpReceivedBucket = (fragrecptr.p->maxp + fragrecptr.p->p) + 1; /* RECEIVED BUCKET */
+ texpDirInd = texpReceivedBucket >> fragrecptr.p->k;
+ newDirptr.i = RNIL;
+ ptrNull(newDirptr);
+ texpDirRangeIndex = texpDirInd >> 8;
+ ptrCheckGuard(expDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(texpDirRangeIndex, 256);
+ expDirptr.i = expDirRangePtr.p->dirArray[texpDirRangeIndex];
+ if (expDirptr.i == RNIL) {
+ jam();
+ seizeDirectory(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ return;
+ } else {
+ jam();
+ newDirptr = sdDirptr;
+ expDirptr = sdDirptr;
+ expDirRangePtr.p->dirArray[texpDirRangeIndex] = sdDirptr.i;
+ }//if
+ } else {
+ ptrCheckGuard(expDirptr, cdirarraysize, directoryarray);
+ }//if
+ texpDirPageIndex = texpDirInd & 0xff;
+ expPageptr.i = expDirptr.p->pagep[texpDirPageIndex];
+ if (expPageptr.i == RNIL) {
+ jam();
+ seizePage(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ if (newDirptr.i != RNIL) {
+ jam();
+ rdDirptr.i = newDirptr.i;
+ releaseDirectory(signal);
+ }//if
+ return;
+ }//if
+ expDirptr.p->pagep[texpDirPageIndex] = spPageptr.i;
+ tipPageId = texpDirInd;
+ inpPageptr = spPageptr;
+ initPage(signal);
+ fragrecptr.p->dirsize++;
+ expPageptr = spPageptr;
+ } else {
+ ptrCheckGuard(expPageptr, cpagesize, page8);
+ }//if
+ fragrecptr.p->expReceivePageptr = expPageptr.i;
+ fragrecptr.p->expReceiveIndex = texpReceivedBucket & ((1 << fragrecptr.p->k) - 1);
+ /*--------------------------------------------------------------------------*/
+ /* THE NEXT ACTION IS TO FIND THE PAGE, THE PAGE INDEX AND THE PAGE */
+ /* DIRECTORY OF THE BUCKET TO BE SPLIT. */
+ /*--------------------------------------------------------------------------*/
+ expDirRangePtr.i = fragrecptr.p->directory;
+ cexcPageindex = fragrecptr.p->p & ((1 << fragrecptr.p->k) - 1); /* PAGE INDEX OBS K = 6 */
+ texpDirInd = fragrecptr.p->p >> fragrecptr.p->k; /* DIRECTORY INDEX OBS K = 6 */
+ ptrCheckGuard(expDirRangePtr, cdirrangesize, dirRange);
+ arrGuard((texpDirInd >> 8), 256);
+ expDirptr.i = expDirRangePtr.p->dirArray[texpDirInd >> 8];
+ ptrCheckGuard(expDirptr, cdirarraysize, directoryarray);
+ excPageptr.i = expDirptr.p->pagep[texpDirInd & 0xff];
+ fragrecptr.p->expSenderIndex = cexcPageindex;
+ fragrecptr.p->expSenderPageptr = excPageptr.i;
+ if (excPageptr.i == RNIL) {
+ jam();
+ endofexpLab(signal); /* EMPTY BUCKET */
+ return;
+ }//if
+ fragrecptr.p->expReceiveForward = ZTRUE;
+ ptrCheckGuard(excPageptr, cpagesize, page8);
+ expandcontainer(signal);
+ endofexpLab(signal);
+ return;
+}//Dbacc::execEXPANDCHECK2()
+
+void Dbacc::endofexpLab(Signal* signal)
+{
+ fragrecptr.p->p++;
+ fragrecptr.p->slack += fragrecptr.p->maxloadfactor;
+ fragrecptr.p->expandCounter++;
+ if (fragrecptr.p->p > fragrecptr.p->maxp) {
+ jam();
+ fragrecptr.p->maxp = (fragrecptr.p->maxp << 1) | 1;
+ fragrecptr.p->lhdirbits++;
+ fragrecptr.p->hashcheckbit++;
+ fragrecptr.p->p = 0;
+ }//if
+ Uint32 noOfBuckets = (fragrecptr.p->maxp + 1) + fragrecptr.p->p;
+ Uint32 Thysteres = fragrecptr.p->maxloadfactor - fragrecptr.p->minloadfactor;
+ fragrecptr.p->slackCheck = noOfBuckets * Thysteres;
+ if (fragrecptr.p->slack > (Uint32)(1 << 31)) {
+ jam();
+ /* IT MEANS THAT IF SLACK < ZERO */
+ /* --------------------------------------------------------------------------------- */
+ /* IT IS STILL NECESSARY TO EXPAND THE FRAGMENT EVEN MORE. START IT FROM HERE */
+ /* WITHOUT WAITING FOR NEXT COMMIT ON THE FRAGMENT. */
+ /* --------------------------------------------------------------------------------- */
+ fragrecptr.p->expandFlag = 1;
+ signal->theData[0] = fragrecptr.i;
+ signal->theData[1] = fragrecptr.p->p;
+ signal->theData[2] = fragrecptr.p->maxp;
+ sendSignal(cownBlockref, GSN_EXPANDCHECK2, signal, 3, JBB);
+ }//if
+ return;
+}//Dbacc::endofexpLab()
+
+void Dbacc::execDEBUG_SIG(Signal* signal)
+{
+ jamEntry();
+ expPageptr.i = signal->theData[0];
+
+ progError(__LINE__,
+ ERR_SR_UNDOLOG);
+ return;
+}//Dbacc::execDEBUG_SIG()
+
+/* --------------------------------------------------------------------------------- */
+/* EXPANDCONTAINER */
+/* INPUT: EXC_PAGEPTR (POINTER TO THE ACTIVE PAGE RECORD) */
+/* CEXC_PAGEINDEX (INDEX OF THE BUCKET). */
+/* */
+/* DESCRIPTION: THE HASH VALUE OF ALL ELEMENTS IN THE CONTAINER WILL BE */
+/* CHECKED. SOME OF THIS ELEMENTS HAVE TO MOVE TO THE NEW CONTAINER */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::expandcontainer(Signal* signal)
+{
+ Uint32 texcHashvalue;
+ Uint32 texcTmp;
+ Uint32 texcIndex;
+ Uint32 texpKeyLen;
+ Uint32 guard20;
+
+ texpKeyLen = fragrecptr.p->keyLength;
+ if (texpKeyLen == 0) {
+ jam();
+ texpKeyLen = ZACTIVE_LONG_KEY_LEN;
+ }//if
+ cexcPrevpageptr = RNIL;
+ cexcPrevconptr = 0;
+ cexcForward = ZTRUE;
+ EXP_CONTAINER_LOOP:
+ cexcContainerptr = (cexcPageindex << ZSHIFT_PLUS) - (cexcPageindex << ZSHIFT_MINUS);
+ if (cexcForward == ZTRUE) {
+ jam();
+ cexcContainerptr = cexcContainerptr + ZHEAD_SIZE;
+ cexcElementptr = cexcContainerptr + ZCON_HEAD_SIZE;
+ } else {
+ jam();
+ cexcContainerptr = ((cexcContainerptr + ZHEAD_SIZE) + ZBUF_SIZE) - ZCON_HEAD_SIZE;
+ cexcElementptr = cexcContainerptr - 1;
+ }//if
+ arrGuard(cexcContainerptr, 2048);
+ cexcContainerhead = excPageptr.p->word32[cexcContainerptr];
+ cexcContainerlen = cexcContainerhead >> 26;
+ cexcMovedLen = ZCON_HEAD_SIZE;
+ if (cexcContainerlen <= ZCON_HEAD_SIZE) {
+ ndbrequire(cexcContainerlen >= ZCON_HEAD_SIZE);
+ jam();
+ goto NEXT_ELEMENT;
+ }//if
+ NEXT_ELEMENT_LOOP:
+ idrOperationRecPtr.i = RNIL;
+ ptrNull(idrOperationRecPtr);
+ /* --------------------------------------------------------------------------------- */
+ /* CEXC_PAGEINDEX PAGE INDEX OF CURRENT CONTAINER BEING EXAMINED. */
+ /* CEXC_CONTAINERPTR INDEX OF CURRENT CONTAINER BEING EXAMINED. */
+ /* CEXC_ELEMENTPTR INDEX OF CURRENT ELEMENT BEING EXAMINED. */
+ /* EXC_PAGEPTR PAGE WHERE CURRENT ELEMENT RESIDES. */
+ /* CEXC_PREVPAGEPTR PAGE OF PREVIOUS CONTAINER. */
+ /* CEXC_PREVCONPTR INDEX OF PREVIOUS CONTAINER */
+ /* CEXC_FORWARD DIRECTION OF CURRENT CONTAINER */
+ /* --------------------------------------------------------------------------------- */
+ arrGuard(cexcElementptr, 2048);
+ tidrElemhead = excPageptr.p->word32[cexcElementptr];
+ if (ElementHeader::getUnlocked(tidrElemhead)){
+ jam();
+ texcHashvalue = ElementHeader::getHashValuePart(tidrElemhead);
+ } else {
+ jam();
+ idrOperationRecPtr.i = ElementHeader::getOpPtrI(tidrElemhead);
+ ptrCheckGuard(idrOperationRecPtr, coprecsize, operationrec);
+ texcHashvalue = idrOperationRecPtr.p->hashvaluePart;
+ if ((fragrecptr.p->createLcp == ZTRUE) &&
+ (((texcHashvalue >> fragrecptr.p->hashcheckbit) & 1) != 0)) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // During local checkpoints we must ensure that we restore the element header in
+ // unlocked state and with the hash value part there with tuple status zeroed.
+ // Otherwise a later insert over the same element will write an UNDO log that will
+ // ensure that the now removed element is restored together with its locked element
+ // header and without the hash value part.
+ /* --------------------------------------------------------------------------------- */
+ const Uint32 hv = idrOperationRecPtr.p->hashvaluePart;
+ const Uint32 eh = ElementHeader::setUnlocked(hv, 0);
+ excPageptr.p->word32[cexcElementptr] = eh;
+ }//if
+ }//if
+ if (((texcHashvalue >> fragrecptr.p->hashcheckbit) & 1) == 0) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THIS ELEMENT IS NOT TO BE MOVED. WE CALCULATE THE WHEREABOUTS OF THE NEXT */
+ /* ELEMENT AND PROCEED WITH THAT OR END THE SEARCH IF THERE ARE NO MORE */
+ /* ELEMENTS IN THIS CONTAINER. */
+ /* --------------------------------------------------------------------------------- */
+ goto NEXT_ELEMENT;
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* THE HASH BIT WAS SET AND WE SHALL MOVE THIS ELEMENT TO THE NEW BUCKET. */
+ /* WE START BY READING THE ELEMENT TO BE ABLE TO INSERT IT INTO THE NEW BUCKET.*/
+ /* THEN WE INSERT THE ELEMENT INTO THE NEW BUCKET. THE NEXT STEP IS TO DELETE */
+ /* THE ELEMENT FROM THIS BUCKET. THIS IS PERFORMED BY REPLACING IT WITH THE */
+ /* LAST ELEMENT IN THE BUCKET. IF THIS ELEMENT IS TO BE MOVED WE MOVE IT AND */
+ /* GET THE LAST ELEMENT AGAIN UNTIL WE EITHER FIND ONE THAT STAYS OR THIS */
+ /* ELEMENT IS THE LAST ELEMENT. */
+ /* --------------------------------------------------------------------------------- */
+ texcTmp = cexcElementptr + cexcForward;
+ guard20 = fragrecptr.p->localkeylen - 1;
+ for (texcIndex = 0; texcIndex <= guard20; texcIndex++) {
+ arrGuard(texcIndex, 2);
+ arrGuard(texcTmp, 2048);
+ clocalkey[texcIndex] = excPageptr.p->word32[texcTmp];
+ texcTmp = texcTmp + cexcForward;
+ }//for
+ guard20 = texpKeyLen - 1;
+ for (texcIndex = 0; texcIndex <= guard20; texcIndex++) {
+ arrGuard(texcIndex, 2048);
+ arrGuard(texcTmp, 2048);
+ ckeys[texcIndex] = excPageptr.p->word32[texcTmp];
+ texcTmp = texcTmp + cexcForward;
+ }//for
+ tidrPageindex = fragrecptr.p->expReceiveIndex;
+ idrPageptr.i = fragrecptr.p->expReceivePageptr;
+ ptrCheckGuard(idrPageptr, cpagesize, page8);
+ tidrForward = fragrecptr.p->expReceiveForward;
+ tidrKeyLen = texpKeyLen;
+ insertElement(signal);
+ fragrecptr.p->expReceiveIndex = tidrPageindex;
+ fragrecptr.p->expReceivePageptr = idrPageptr.i;
+ fragrecptr.p->expReceiveForward = tidrForward;
+ REMOVE_LAST_LOOP:
+ jam();
+ lastPageptr.i = excPageptr.i;
+ lastPageptr.p = excPageptr.p;
+ tlastContainerptr = cexcContainerptr;
+ lastPrevpageptr.i = cexcPrevpageptr;
+ ptrCheck(lastPrevpageptr, cpagesize, page8);
+ tlastPrevconptr = cexcPrevconptr;
+ arrGuard(tlastContainerptr, 2048);
+ tlastContainerhead = lastPageptr.p->word32[tlastContainerptr];
+ tlastContainerlen = tlastContainerhead >> 26;
+ tlastForward = cexcForward;
+ tlastPageindex = cexcPageindex;
+ getLastAndRemove(signal);
+ if (excPageptr.i == lastPageptr.i) {
+ if (cexcElementptr == tlastElementptr) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE CURRENT ELEMENT WAS ALSO THE LAST ELEMENT. */
+ /* --------------------------------------------------------------------------------- */
+ return;
+ }//if
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* THE CURRENT ELEMENT WAS NOT THE LAST ELEMENT. IF THE LAST ELEMENT SHOULD */
+ /* STAY WE COPY IT TO THE POSITION OF THE CURRENT ELEMENT, OTHERWISE WE INSERT */
+ /* INTO THE NEW BUCKET, REMOVE IT AND TRY WITH THE NEW LAST ELEMENT. */
+ /* --------------------------------------------------------------------------------- */
+ idrOperationRecPtr.i = RNIL;
+ ptrNull(idrOperationRecPtr);
+ arrGuard(tlastElementptr, 2048);
+ tidrElemhead = lastPageptr.p->word32[tlastElementptr];
+ if (ElementHeader::getUnlocked(tidrElemhead)) {
+ jam();
+ texcHashvalue = ElementHeader::getHashValuePart(tidrElemhead);
+ } else {
+ jam();
+ idrOperationRecPtr.i = ElementHeader::getOpPtrI(tidrElemhead);
+ ptrCheckGuard(idrOperationRecPtr, coprecsize, operationrec);
+ texcHashvalue = idrOperationRecPtr.p->hashvaluePart;
+ if ((fragrecptr.p->createLcp == ZTRUE) &&
+ (((texcHashvalue >> fragrecptr.p->hashcheckbit) & 1) != 0)) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // During local checkpoints we must ensure that we restore the element header in
+ // unlocked state and with the hash value part there with tuple status zeroed.
+ // Otherwise a later insert over the same element will write an UNDO log that will
+ // ensure that the now removed element is restored together with its locked element
+ // header and without the hash value part.
+ /* --------------------------------------------------------------------------------- */
+ const Uint32 hv = idrOperationRecPtr.p->hashvaluePart;
+ const Uint32 eh = ElementHeader::setUnlocked(hv, 0);
+ lastPageptr.p->word32[tlastElementptr] = eh;
+ }//if
+ }//if
+ if (((texcHashvalue >> fragrecptr.p->hashcheckbit) & 1) == 0) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE LAST ELEMENT IS NOT TO BE MOVED. WE COPY IT TO THE CURRENT ELEMENT. */
+ /* --------------------------------------------------------------------------------- */
+ delPageptr = excPageptr;
+ tdelContainerptr = cexcContainerptr;
+ tdelForward = cexcForward;
+ tdelElementptr = cexcElementptr;
+ deleteElement(signal);
+ } else {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE LAST ELEMENT IS ALSO TO BE MOVED. */
+ /* --------------------------------------------------------------------------------- */
+ texcTmp = tlastElementptr + tlastForward;
+ for (texcIndex = 0; texcIndex < fragrecptr.p->localkeylen; texcIndex++) {
+ arrGuard(texcIndex, 2);
+ arrGuard(texcTmp, 2048);
+ clocalkey[texcIndex] = lastPageptr.p->word32[texcTmp];
+ texcTmp = texcTmp + tlastForward;
+ }//for
+ for (texcIndex = 0; texcIndex < texpKeyLen; texcIndex++) {
+ arrGuard(texcIndex, 2048);
+ arrGuard(texcTmp, 2048);
+ ckeys[texcIndex] = lastPageptr.p->word32[texcTmp];
+ texcTmp = texcTmp + tlastForward;
+ }//for
+ tidrPageindex = fragrecptr.p->expReceiveIndex;
+ idrPageptr.i = fragrecptr.p->expReceivePageptr;
+ ptrCheckGuard(idrPageptr, cpagesize, page8);
+ tidrForward = fragrecptr.p->expReceiveForward;
+ tidrKeyLen = texpKeyLen;
+ insertElement(signal);
+ fragrecptr.p->expReceiveIndex = tidrPageindex;
+ fragrecptr.p->expReceivePageptr = idrPageptr.i;
+ fragrecptr.p->expReceiveForward = tidrForward;
+ goto REMOVE_LAST_LOOP;
+ }//if
+ NEXT_ELEMENT:
+ arrGuard(cexcContainerptr, 2048);
+ cexcContainerhead = excPageptr.p->word32[cexcContainerptr];
+ cexcMovedLen = cexcMovedLen + fragrecptr.p->elementLength;
+ if ((cexcContainerhead >> 26) > cexcMovedLen) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE HAVE NOT YET MOVED THE COMPLETE CONTAINER. WE PROCEED WITH THE NEXT */
+ /* ELEMENT IN THE CONTAINER. IT IS IMPORTANT TO READ THE CONTAINER LENGTH */
+ /* FROM THE CONTAINER HEADER SINCE IT MIGHT CHANGE BY REMOVING THE LAST */
+ /* ELEMENT IN THE BUCKET. */
+ /* --------------------------------------------------------------------------------- */
+ cexcElementptr = cexcElementptr + (cexcForward * fragrecptr.p->elementLength);
+ goto NEXT_ELEMENT_LOOP;
+ }//if
+ if (((cexcContainerhead >> 7) & 3) != 0) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WE PROCEED TO THE NEXT CONTAINER IN THE BUCKET. */
+ /* --------------------------------------------------------------------------------- */
+ cexcPrevpageptr = excPageptr.i;
+ cexcPrevconptr = cexcContainerptr;
+ nextcontainerinfoExp(signal);
+ goto EXP_CONTAINER_LOOP;
+ }//if
+}//Dbacc::expandcontainer()
+
+/* ******************--------------------------------------------------------------- */
+/* SHRINKCHECK JOIN BUCKET ORD */
+/* SENDER: ACC, LEVEL B */
+/* INPUT: FRAGRECPTR, POINTS TO A FRAGMENT RECORD. */
+/* DESCRIPTION: TWO BUCKET OF A FRAGMENT PAGE WILL BE JOINED TOGETHER */
+/* ACCORDING TO LH3. */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* SHRINKCHECK JOIN BUCKET ORD */
+/* ******************------------------------------+ */
+/* SENDER: ACC, LEVEL B */
+/* TWO BUCKETS OF THE FRAGMENT */
+/* WILL BE JOINED ACORDING TO LH3 */
+/* AND COMMIT TRANSACTION PROCESS */
+/* WILL BE CONTINUED */
+Uint32 Dbacc::checkScanShrink(Signal* signal)
+{
+ Uint32 Ti;
+ Uint32 TreturnCode = 0;
+ Uint32 TPageIndex;
+ Uint32 TDirInd;
+ Uint32 TmergeDest;
+ Uint32 TmergeSource;
+ Uint32 TreleaseScanBucket;
+ Uint32 TreleaseInd = 0;
+ Uint32 TreleaseScanIndicator[4];
+ DirectoryarrayPtr TDirptr;
+ DirRangePtr TDirRangePtr;
+ Page8Ptr TPageptr;
+ ScanRecPtr TscanPtr;
+ RootfragmentrecPtr Trootfragrecptr;
+
+ Trootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(Trootfragrecptr, crootfragmentsize, rootfragmentrec);
+ if (fragrecptr.p->p == 0) {
+ jam();
+ TmergeDest = fragrecptr.p->maxp >> 1;
+ } else {
+ jam();
+ TmergeDest = fragrecptr.p->p - 1;
+ }//if
+ TmergeSource = fragrecptr.p->maxp + fragrecptr.p->p;
+ for (Ti = 0; Ti < 4; Ti++) {
+ TreleaseScanIndicator[Ti] = 0;
+ if (Trootfragrecptr.p->scan[Ti] != RNIL) {
+ TscanPtr.i = Trootfragrecptr.p->scan[Ti];
+ ptrCheckGuard(TscanPtr, cscanRecSize, scanRec);
+ if (TscanPtr.p->activeLocalFrag == fragrecptr.i) {
+ //-------------------------------------------------------------
+ // A scan is ongoing on this particular local fragment. We have
+ // to check its current state.
+ //-------------------------------------------------------------
+ if (TscanPtr.p->scanBucketState == ScanRec::FIRST_LAP) {
+ jam();
+ if ((TmergeDest == TscanPtr.p->nextBucketIndex) ||
+ (TmergeSource == TscanPtr.p->nextBucketIndex)) {
+ jam();
+ //-------------------------------------------------------------
+ // We are currently scanning one of the buckets involved in the
+ // merge. We cannot merge while simultaneously performing a scan.
+ // We have to pass this offer for merging the buckets.
+ //-------------------------------------------------------------
+ TreturnCode = 1;
+ return TreturnCode;
+ } else if (TmergeDest < TscanPtr.p->nextBucketIndex) {
+ jam();
+ TreleaseScanIndicator[Ti] = 1;
+ TreleaseInd = 1;
+ }//if
+ } else if (TscanPtr.p->scanBucketState == ScanRec::SECOND_LAP) {
+ jam();
+ //-------------------------------------------------------------
+ // We are performing a second lap to handle buckets that was
+ // merged during the first lap of scanning. During this second
+ // lap we do not allow any splits or merges.
+ //-------------------------------------------------------------
+ TreturnCode = 1;
+ return TreturnCode;
+ } else if (TscanPtr.p->scanBucketState == ScanRec::SCAN_COMPLETED) {
+ jam();
+ //-------------------------------------------------------------
+ // The scan is completed and we can thus go ahead and perform
+ // the split.
+ //-------------------------------------------------------------
+ } else {
+ jam();
+ sendSystemerror(signal);
+ return TreturnCode;
+ }//if
+ }//if
+ }//if
+ }//for
+ if (TreleaseInd == 1) {
+ jam();
+ TreleaseScanBucket = TmergeSource;
+ TDirRangePtr.i = fragrecptr.p->directory;
+ TPageIndex = TreleaseScanBucket & ((1 << fragrecptr.p->k) - 1); /* PAGE INDEX OBS K = 6 */
+ TDirInd = TreleaseScanBucket >> fragrecptr.p->k; /* DIRECTORY INDEX OBS K = 6 */
+ ptrCheckGuard(TDirRangePtr, cdirrangesize, dirRange);
+ arrGuard((TDirInd >> 8), 256);
+ TDirptr.i = TDirRangePtr.p->dirArray[TDirInd >> 8];
+ ptrCheckGuard(TDirptr, cdirarraysize, directoryarray);
+ TPageptr.i = TDirptr.p->pagep[TDirInd & 0xff];
+ ptrCheckGuard(TPageptr, cpagesize, page8);
+ for (Ti = 0; Ti < 4; Ti++) {
+ if (TreleaseScanIndicator[Ti] == 1) {
+ jam();
+ scanPtr.i = Trootfragrecptr.p->scan[Ti];
+ ptrCheckGuard(scanPtr, cscanRecSize, scanRec);
+ rsbPageidptr.i = TPageptr.i;
+ rsbPageidptr.p = TPageptr.p;
+ trsbPageindex = TPageIndex;
+ releaseScanBucket(signal);
+ if (TmergeDest < scanPtr.p->minBucketIndexToRescan) {
+ jam();
+ //-------------------------------------------------------------
+ // We have to keep track of the starting bucket to Rescan in the
+ // second lap.
+ //-------------------------------------------------------------
+ scanPtr.p->minBucketIndexToRescan = TmergeDest;
+ }//if
+ if (TmergeDest > scanPtr.p->maxBucketIndexToRescan) {
+ jam();
+ //-------------------------------------------------------------
+ // We have to keep track of the ending bucket to Rescan in the
+ // second lap.
+ //-------------------------------------------------------------
+ scanPtr.p->maxBucketIndexToRescan = TmergeDest;
+ }//if
+ }//if
+ }//for
+ }//if
+ return TreturnCode;
+}//Dbacc::checkScanShrink()
+
+void Dbacc::execSHRINKCHECK2(Signal* signal)
+{
+ Uint32 tshrTmp1;
+
+ jamEntry();
+ fragrecptr.i = signal->theData[0];
+ tresult = 0; /* 0= FALSE,1= TRUE,> ZLIMIT_OF_ERROR =ERRORCODE */
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ fragrecptr.p->expandFlag = 0;
+ if (fragrecptr.p->slack <= fragrecptr.p->slackCheck) {
+ jam();
+ /* TIME FOR JOIN BUCKETS PROCESS */
+ /*--------------------------------------------------------------*/
+ /* NO LONGER NECESSARY TO SHRINK THE FRAGMENT. */
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ if (fragrecptr.p->slack > (Uint32)(1 << 31)) {
+ jam();
+ /*--------------------------------------------------------------*/
+ /* THE SLACK IS NEGATIVE, IN THIS CASE WE WILL NOT NEED ANY */
+ /* SHRINK. */
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ texpDirInd = (fragrecptr.p->maxp + fragrecptr.p->p) >> fragrecptr.p->k;
+ if (((fragrecptr.p->maxp + fragrecptr.p->p) & ((1 << fragrecptr.p->k) - 1)) == 0) {
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ if (fragrecptr.p->fragState == LCP_SEND_PAGES) {
+ if (fragrecptr.p->lcpMaxDirIndex > texpDirInd) {
+ if (fragrecptr.p->lcpDirIndex <= texpDirInd) {
+ jam();
+ /*--------------------------------------------------------------*/
+ /* WE DO NOT ALLOW ANY SHRINKS THAT REMOVE PAGES THAT ARE */
+ /* NEEDED AS PART OF THE LOCAL CHECKPOINT. */
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ }//if
+ }//if
+ }//if
+ }//if
+ if (fragrecptr.p->firstOverflowRec == RNIL) {
+ jam();
+ allocOverflowPage(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ return;
+ }//if
+ }//if
+ if (cfirstfreepage == RNIL) {
+ if (cfreepage >= cpagesize) {
+ jam();
+ /*--------------------------------------------------------------*/
+ /* WE HAVE TO STOP THE SHRINK PROCESS SINCE THERE ARE NO FREE */
+ /* PAGES. THIS MEANS THAT WE COULD BE FORCED TO CRASH SINCE WE */
+ /* CANNOT COMPLETE THE SHRINK. TO AVOID THE CRASH WE EXIT HERE. */
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ }//if
+ if (checkScanShrink(signal) == 1) {
+ jam();
+ /*--------------------------------------------------------------*/
+ // A scan state was inconsistent with performing a shrink
+ // operation.
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ if (remainingUndoPages() < ZMIN_UNDO_PAGES_AT_EXPAND) {
+ jam();
+ /*--------------------------------------------------------------*/
+ // We did not have enough undo log buffers to start up an
+ // shrink operation
+ /*--------------------------------------------------------------*/
+ return;
+ }//if
+ }//if
+ if (fragrecptr.p->p == 0) {
+ jam();
+ fragrecptr.p->maxp = fragrecptr.p->maxp >> 1;
+ fragrecptr.p->p = fragrecptr.p->maxp;
+ fragrecptr.p->lhdirbits--;
+ fragrecptr.p->hashcheckbit--;
+ } else {
+ jam();
+ fragrecptr.p->p--;
+ }//if
+ /*--------------------------------------------------------------------------*/
+ /* WE START BY FINDING THE NECESSARY INFORMATION OF THE BUCKET TO BE */
+ /* REMOVED WHICH WILL SEND ITS ELEMENTS TO THE RECEIVING BUCKET. */
+ /*--------------------------------------------------------------------------*/
+ expDirRangePtr.i = fragrecptr.p->directory;
+ cexcPageindex = ((fragrecptr.p->maxp + fragrecptr.p->p) + 1) & ((1 << fragrecptr.p->k) - 1);
+ texpDirInd = ((fragrecptr.p->maxp + fragrecptr.p->p) + 1) >> fragrecptr.p->k;
+ texpDirRangeIndex = texpDirInd >> 8;
+ texpDirPageIndex = texpDirInd & 0xff;
+ ptrCheckGuard(expDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(texpDirRangeIndex, 256);
+ expDirptr.i = expDirRangePtr.p->dirArray[texpDirRangeIndex];
+ ptrCheckGuard(expDirptr, cdirarraysize, directoryarray);
+ excPageptr.i = expDirptr.p->pagep[texpDirPageIndex];
+ fragrecptr.p->expSenderDirptr = expDirptr.i;
+ fragrecptr.p->expSenderIndex = cexcPageindex;
+ fragrecptr.p->expSenderPageptr = excPageptr.i;
+ fragrecptr.p->expSenderDirIndex = texpDirInd;
+ /*--------------------------------------------------------------------------*/
+ /* WE NOW PROCEED BY FINDING THE NECESSARY INFORMATION ABOUT THE */
+ /* RECEIVING BUCKET. */
+ /*--------------------------------------------------------------------------*/
+ expDirRangePtr.i = fragrecptr.p->directory;
+ texpReceivedBucket = fragrecptr.p->p >> fragrecptr.p->k;
+ ptrCheckGuard(expDirRangePtr, cdirrangesize, dirRange);
+ arrGuard((texpReceivedBucket >> 8), 256);
+ expDirptr.i = expDirRangePtr.p->dirArray[texpReceivedBucket >> 8];
+ ptrCheckGuard(expDirptr, cdirarraysize, directoryarray);
+ fragrecptr.p->expReceivePageptr = expDirptr.p->pagep[texpReceivedBucket & 0xff];
+ fragrecptr.p->expReceiveIndex = fragrecptr.p->p & ((1 << fragrecptr.p->k) - 1);
+ fragrecptr.p->expReceiveForward = ZTRUE;
+ if (excPageptr.i == RNIL) {
+ jam();
+ endofshrinkbucketLab(signal); /* EMPTY BUCKET */
+ return;
+ }//if
+ /*--------------------------------------------------------------------------*/
+ /* INITIALISE THE VARIABLES FOR THE SHRINK PROCESS. */
+ /*--------------------------------------------------------------------------*/
+ ptrCheckGuard(excPageptr, cpagesize, page8);
+ cexcForward = ZTRUE;
+ cexcContainerptr = (cexcPageindex << ZSHIFT_PLUS) - (cexcPageindex << ZSHIFT_MINUS);
+ cexcContainerptr = cexcContainerptr + ZHEAD_SIZE;
+ arrGuard(cexcContainerptr, 2048);
+ cexcContainerhead = excPageptr.p->word32[cexcContainerptr];
+ cexcContainerlen = cexcContainerhead >> 26;
+ if (cexcContainerlen <= ZCON_HEAD_SIZE) {
+ ndbrequire(cexcContainerlen == ZCON_HEAD_SIZE);
+ } else {
+ jam();
+ shrinkcontainer(signal);
+ }//if
+ /*--------------------------------------------------------------------------*/
+ /* THIS CONTAINER IS NOT YET EMPTY AND WE REMOVE ALL THE ELEMENTS. */
+ /*--------------------------------------------------------------------------*/
+ if (((cexcContainerhead >> 10) & 1) == 1) {
+ jam();
+ rlPageptr = excPageptr;
+ trlPageindex = cexcPageindex;
+ trlRelCon = ZFALSE;
+ turlIndex = cexcContainerptr + (ZBUF_SIZE - ZCON_HEAD_SIZE);
+ releaseRightlist(signal);
+ }//if
+ tshrTmp1 = ZCON_HEAD_SIZE;
+ tshrTmp1 = tshrTmp1 << 26;
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ datapageptr.p = excPageptr.p;
+ cundoinfolength = 1;
+ cundoElemIndex = cexcContainerptr;
+ undoWritingProcess(signal);
+ }//if
+ dbgWord32(excPageptr, cexcContainerptr, tshrTmp1);
+ arrGuard(cexcContainerptr, 2048);
+ excPageptr.p->word32[cexcContainerptr] = tshrTmp1;
+ if (((cexcContainerhead >> 7) & 0x3) == 0) {
+ jam();
+ endofshrinkbucketLab(signal);
+ return;
+ }//if
+ nextcontainerinfoExp(signal);
+ do {
+ cexcContainerptr = (cexcPageindex << ZSHIFT_PLUS) - (cexcPageindex << ZSHIFT_MINUS);
+ if (cexcForward == ZTRUE) {
+ jam();
+ cexcContainerptr = cexcContainerptr + ZHEAD_SIZE;
+ } else {
+ jam();
+ cexcContainerptr = ((cexcContainerptr + ZHEAD_SIZE) + ZBUF_SIZE) - ZCON_HEAD_SIZE;
+ }//if
+ arrGuard(cexcContainerptr, 2048);
+ cexcContainerhead = excPageptr.p->word32[cexcContainerptr];
+ cexcContainerlen = cexcContainerhead >> 26;
+ ndbrequire(cexcContainerlen > ZCON_HEAD_SIZE);
+ /*--------------------------------------------------------------------------*/
+ /* THIS CONTAINER IS NOT YET EMPTY AND WE REMOVE ALL THE ELEMENTS. */
+ /*--------------------------------------------------------------------------*/
+ shrinkcontainer(signal);
+ cexcPrevpageptr = excPageptr.i;
+ cexcPrevpageindex = cexcPageindex;
+ cexcPrevforward = cexcForward;
+ if (((cexcContainerhead >> 7) & 0x3) != 0) {
+ jam();
+ /*--------------------------------------------------------------------------*/
+ /* WE MUST CALL THE NEXT CONTAINER INFO ROUTINE BEFORE WE RELEASE THE */
+ /* CONTAINER SINCE THE RELEASE WILL OVERWRITE THE NEXT POINTER. */
+ /*--------------------------------------------------------------------------*/
+ nextcontainerinfoExp(signal);
+ }//if
+ rlPageptr.i = cexcPrevpageptr;
+ ptrCheckGuard(rlPageptr, cpagesize, page8);
+ trlPageindex = cexcPrevpageindex;
+ if (cexcPrevforward == ZTRUE) {
+ jam();
+ if (((cexcContainerhead >> 10) & 1) == 1) {
+ jam();
+ trlRelCon = ZFALSE;
+ turlIndex = cexcContainerptr + (ZBUF_SIZE - ZCON_HEAD_SIZE);
+ releaseRightlist(signal);
+ }//if
+ trlRelCon = ZTRUE;
+ tullIndex = cexcContainerptr;
+ releaseLeftlist(signal);
+ } else {
+ jam();
+ if (((cexcContainerhead >> 10) & 1) == 1) {
+ jam();
+ trlRelCon = ZFALSE;
+ tullIndex = cexcContainerptr - (ZBUF_SIZE - ZCON_HEAD_SIZE);
+ releaseLeftlist(signal);
+ }//if
+ trlRelCon = ZTRUE;
+ turlIndex = cexcContainerptr;
+ releaseRightlist(signal);
+ }//if
+ } while (((cexcContainerhead >> 7) & 0x3) != 0);
+ endofshrinkbucketLab(signal);
+ return;
+}//Dbacc::execSHRINKCHECK2()
+
+void Dbacc::endofshrinkbucketLab(Signal* signal)
+{
+ fragrecptr.p->expandCounter--;
+ fragrecptr.p->slack -= fragrecptr.p->maxloadfactor;
+ if (fragrecptr.p->expSenderIndex == 0) {
+ jam();
+ fragrecptr.p->dirsize--;
+ if (fragrecptr.p->expSenderPageptr != RNIL) {
+ jam();
+ rpPageptr.i = fragrecptr.p->expSenderPageptr;
+ ptrCheckGuard(rpPageptr, cpagesize, page8);
+ releasePage(signal);
+ expDirptr.i = fragrecptr.p->expSenderDirptr;
+ ptrCheckGuard(expDirptr, cdirarraysize, directoryarray);
+ expDirptr.p->pagep[fragrecptr.p->expSenderDirIndex & 0xff] = RNIL;
+ }//if
+ if (((((fragrecptr.p->p + fragrecptr.p->maxp) + 1) >> fragrecptr.p->k) & 0xff) == 0) {
+ jam();
+ rdDirptr.i = fragrecptr.p->expSenderDirptr;
+ releaseDirectory(signal);
+ expDirRangePtr.i = fragrecptr.p->directory;
+ ptrCheckGuard(expDirRangePtr, cdirrangesize, dirRange);
+ arrGuard((fragrecptr.p->expSenderDirIndex >> 8), 256);
+ expDirRangePtr.p->dirArray[fragrecptr.p->expSenderDirIndex >> 8] = RNIL;
+ }//if
+ }//if
+ if (fragrecptr.p->slack < (Uint32)(1 << 31)) {
+ jam();
+ /*--------------------------------------------------------------*/
+ /* THE SLACK IS POSITIVE, IN THIS CASE WE WILL CHECK WHETHER */
+ /* WE WILL CONTINUE PERFORM ANOTHER SHRINK. */
+ /*--------------------------------------------------------------*/
+ Uint32 noOfBuckets = (fragrecptr.p->maxp + 1) + fragrecptr.p->p;
+ Uint32 Thysteresis = fragrecptr.p->maxloadfactor - fragrecptr.p->minloadfactor;
+ fragrecptr.p->slackCheck = noOfBuckets * Thysteresis;
+ if (fragrecptr.p->slack > Thysteresis) {
+ /*--------------------------------------------------------------*/
+ /* IT IS STILL NECESSARY TO SHRINK THE FRAGMENT MORE. THIS*/
+ /* CAN HAPPEN WHEN A NUMBER OF SHRINKS GET REJECTED */
+ /* DURING A LOCAL CHECKPOINT. WE START A NEW SHRINK */
+ /* IMMEDIATELY FROM HERE WITHOUT WAITING FOR A COMMIT TO */
+ /* START IT. */
+ /*--------------------------------------------------------------*/
+ if (fragrecptr.p->expandCounter > 0) {
+ jam();
+ /*--------------------------------------------------------------*/
+ /* IT IS VERY IMPORTANT TO NOT TRY TO SHRINK MORE THAN */
+ /* WAS EXPANDED. IF MAXP IS SET TO A VALUE BELOW 63 THEN */
+ /* WE WILL LOSE RECORDS SINCE GETDIRINDEX CANNOT HANDLE */
+ /* SHRINKING BELOW 2^K - 1 (NOW 63). THIS WAS A BUG THAT */
+ /* WAS REMOVED 2000-05-12. */
+ /*--------------------------------------------------------------*/
+ fragrecptr.p->expandFlag = 1;
+ signal->theData[0] = fragrecptr.i;
+ signal->theData[1] = fragrecptr.p->p;
+ signal->theData[2] = fragrecptr.p->maxp;
+ sendSignal(cownBlockref, GSN_SHRINKCHECK2, signal, 3, JBB);
+ }//if
+ }//if
+ }//if
+ ndbrequire(fragrecptr.p->maxp >= (Uint32)((1 << fragrecptr.p->k) - 1));
+ return;
+}//Dbacc::endofshrinkbucketLab()
+
+/* --------------------------------------------------------------------------------- */
+/* SHRINKCONTAINER */
+/* INPUT: EXC_PAGEPTR (POINTER TO THE ACTIVE PAGE RECORD) */
+/* CEXC_CONTAINERLEN (LENGTH OF THE CONTAINER). */
+/* CEXC_CONTAINERPTR (ARRAY INDEX OF THE CONTAINER). */
+/* CEXC_FORWARD (CONTAINER FORWARD (+1) OR BACKWARD (-1)) */
+/* */
+/* DESCRIPTION: ALL ELEMENTS OF THE ACTIVE CONTAINER HAVE TO MOVE TO THE NEW */
+/* CONTAINER. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::shrinkcontainer(Signal* signal)
+{
+ Uint32 tshrElementptr;
+ Uint32 tshrRemLen;
+ Uint32 tshrInc;
+ Uint32 tshrKeyLen;
+ Uint32 tshrTmp;
+ Uint32 tshrIndex;
+ Uint32 guard21;
+
+ tshrRemLen = cexcContainerlen - ZCON_HEAD_SIZE;
+ tshrKeyLen = fragrecptr.p->keyLength;
+ if (tshrKeyLen == 0) {
+ jam();
+ tshrKeyLen = ZACTIVE_LONG_KEY_LEN;
+ }//if
+ tshrInc = (ZELEM_HEAD_SIZE + tshrKeyLen) + fragrecptr.p->localkeylen;
+ if (cexcForward == ZTRUE) {
+ jam();
+ tshrElementptr = cexcContainerptr + ZCON_HEAD_SIZE;
+ } else {
+ jam();
+ tshrElementptr = cexcContainerptr - 1;
+ }//if
+ SHR_LOOP:
+ idrOperationRecPtr.i = RNIL;
+ ptrNull(idrOperationRecPtr);
+ /* --------------------------------------------------------------------------------- */
+ /* THE CODE BELOW IS ALL USED TO PREPARE FOR THE CALL TO INSERT_ELEMENT AND */
+ /* HANDLE THE RESULT FROM INSERT_ELEMENT. INSERT_ELEMENT INSERTS THE ELEMENT */
+ /* INTO ANOTHER BUCKET. */
+ /* --------------------------------------------------------------------------------- */
+ arrGuard(tshrElementptr, 2048);
+ tidrElemhead = excPageptr.p->word32[tshrElementptr];
+ if (ElementHeader::getLocked(tidrElemhead)) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* IF THE ELEMENT IS LOCKED WE MUST UPDATE THE ELEMENT INFO IN THE OPERATION */
+ /* RECORD OWNING THE LOCK. WE DO THIS BY READING THE OPERATION RECORD POINTER */
+ /* FROM THE ELEMENT HEADER. */
+ /* --------------------------------------------------------------------------------- */
+ idrOperationRecPtr.i = ElementHeader::getOpPtrI(tidrElemhead);
+ ptrCheckGuard(idrOperationRecPtr, coprecsize, operationrec);
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // During local checkpoints we must ensure that we restore the element header in
+ // unlocked state and with the hash value part there with tuple status zeroed.
+ // Otherwise a later insert over the same element will write an UNDO log that will
+ // ensure that the now removed element is restored together with its locked element
+ // header and without the hash value part.
+ /* --------------------------------------------------------------------------------- */
+ const Uint32 hv = idrOperationRecPtr.p->hashvaluePart;
+ const Uint32 eh = ElementHeader::setUnlocked(hv, 0);
+ excPageptr.p->word32[tshrElementptr] = eh;
+ }//if
+ }//if
+ tshrTmp = tshrElementptr + cexcForward;
+ guard21 = fragrecptr.p->localkeylen - 1;
+ for (tshrIndex = 0; tshrIndex <= guard21; tshrIndex++) {
+ arrGuard(tshrIndex, 2);
+ arrGuard(tshrTmp, 2048);
+ clocalkey[tshrIndex] = excPageptr.p->word32[tshrTmp];
+ tshrTmp = tshrTmp + cexcForward;
+ }//for
+ guard21 = tshrKeyLen - 1;
+ for (tshrIndex = 0; tshrIndex <= guard21; tshrIndex++) {
+ arrGuard(tshrIndex, 2048);
+ arrGuard(tshrTmp, 2048);
+ ckeys[tshrIndex] = excPageptr.p->word32[tshrTmp];
+ tshrTmp = tshrTmp + cexcForward;
+ }//for
+ tidrPageindex = fragrecptr.p->expReceiveIndex;
+ idrPageptr.i = fragrecptr.p->expReceivePageptr;
+ ptrCheckGuard(idrPageptr, cpagesize, page8);
+ tidrForward = fragrecptr.p->expReceiveForward;
+ tidrKeyLen = tshrKeyLen;
+ insertElement(signal);
+ /* --------------------------------------------------------------------------------- */
+ /* TAKE CARE OF RESULT FROM INSERT_ELEMENT. */
+ /* --------------------------------------------------------------------------------- */
+ fragrecptr.p->expReceiveIndex = tidrPageindex;
+ fragrecptr.p->expReceivePageptr = idrPageptr.i;
+ fragrecptr.p->expReceiveForward = tidrForward;
+ if (tshrRemLen < tshrInc) {
+ jam();
+ sendSystemerror(signal);
+ }//if
+ tshrRemLen = tshrRemLen - tshrInc;
+ if (tshrRemLen != 0) {
+ jam();
+ tshrElementptr = tshrTmp;
+ goto SHR_LOOP;
+ }//if
+}//Dbacc::shrinkcontainer()
+
+/* --------------------------------------------------------------------------------- */
+/* NEXTCONTAINERINFO_EXP */
+/* DESCRIPTION:THE CONTAINER HEAD WILL BE CHECKED TO CALCULATE INFORMATION */
+/* ABOUT NEXT CONTAINER IN THE BUCKET. */
+/* INPUT: CEXC_CONTAINERHEAD */
+/* CEXC_CONTAINERPTR */
+/* EXC_PAGEPTR */
+/* OUTPUT: */
+/* CEXC_PAGEINDEX (INDEX FROM WHICH PAGE INDEX CAN BE CALCULATED. */
+/* EXC_PAGEPTR (PAGE REFERENCE OF NEXT CONTAINER) */
+/* CEXC_FORWARD */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::nextcontainerinfoExp(Signal* signal)
+{
+ tnciNextSamePage = (cexcContainerhead >> 9) & 0x1; /* CHECK BIT FOR CHECKING WHERE */
+ /* THE NEXT CONTAINER IS IN THE SAME PAGE */
+ cexcPageindex = cexcContainerhead & 0x7f; /* NEXT CONTAINER PAGE INDEX 7 BITS */
+ if (((cexcContainerhead >> 7) & 3) == ZLEFT) {
+ jam();
+ cexcForward = ZTRUE;
+ } else if (((cexcContainerhead >> 7) & 3) == ZRIGHT) {
+ jam();
+ cexcForward = cminusOne;
+ } else {
+ jam();
+ sendSystemerror(signal);
+ cexcForward = 0; /* DUMMY FOR COMPILER */
+ }//if
+ if (tnciNextSamePage == ZFALSE) {
+ jam();
+ /* NEXT CONTAINER IS IN AN OVERFLOW PAGE */
+ arrGuard(cexcContainerptr + 1, 2048);
+ tnciTmp = excPageptr.p->word32[cexcContainerptr + 1];
+ nciOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ ptrCheckGuard(nciOverflowrangeptr, cdirrangesize, dirRange);
+ arrGuard((tnciTmp >> 8), 256);
+ nciOverflowDirptr.i = nciOverflowrangeptr.p->dirArray[tnciTmp >> 8];
+ ptrCheckGuard(nciOverflowDirptr, cdirarraysize, directoryarray);
+ excPageptr.i = nciOverflowDirptr.p->pagep[tnciTmp & 0xff];
+ ptrCheckGuard(excPageptr, cpagesize, page8);
+ }//if
+}//Dbacc::nextcontainerinfoExp()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF EXPAND/SHRINK MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* LOCAL CHECKPOINT MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* LCP_FRAGIDREQ */
+/* SENDER: LQH, LEVEL B */
+/* ENTER LCP_FRAGIDREQ WITH */
+/* TUSERPTR LQH CONNECTION PTR */
+/* TUSERBLOCKREF, LQH BLOCK REFERENCE */
+/* TCHECKPOINTID, THE CHECKPOINT NUMBER TO USE */
+/* (E.G. 1,2 OR 3) */
+/* TABPTR, TABLE ID = TABLE RECORD POINTER */
+/* TFID ROOT FRAGMENT ID */
+/* CACTIVE_UNDO_FILE_VERSION UNDO FILE VERSION 0,1,2 OR 3. */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* LCP_FRAGIDREQ REQUEST FOR LIST OF STOPED OPERATION */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execLCP_FRAGIDREQ(Signal* signal)
+{
+ jamEntry();
+ tuserptr = signal->theData[0]; /* LQH CONNECTION PTR */
+ tuserblockref = signal->theData[1]; /* LQH BLOCK REFERENCE */
+ tcheckpointid = signal->theData[2]; /* THE CHECKPOINT NUMBER TO USE */
+ /* (E.G. 1,2 OR 3) */
+ tabptr.i = signal->theData[3]; /* TABLE ID = TABLE RECORD POINTER */
+ ptrCheck(tabptr, ctablesize, tabrec);
+ tfid = signal->theData[4]; /* ROOT FRAGMENT ID */
+ cactiveUndoFileVersion = signal->theData[5]; /* UNDO FILE VERSION 0,1,2 OR 3. */
+ tresult = 0;
+ ndbrequire(getrootfragmentrec(signal, rootfragrecptr, tfid));
+ ndbrequire(rootfragrecptr.p->rootState == ACTIVEROOT);
+ seizeLcpConnectRec(signal);
+ initLcpConnRec(signal);
+ lcpConnectptr.p->rootrecptr = rootfragrecptr.i;
+ rootfragrecptr.p->lcpPtr = lcpConnectptr.i;
+ lcpConnectptr.p->localCheckPid = tcheckpointid;
+ lcpConnectptr.p->lcpstate = LCP_ACTIVE;
+ rootfragrecptr.p->rootState = LCP_CREATION;
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[0];
+ /* D6 AT FSOPENREQ =#010003FF. */
+ tlfrTmp1 = 0x010003ff; /* FILE TYPE = .DATA ,VERSION OF FILENAME = 1 */
+ tlfrTmp2 = 0x301; /* D7 CREATE, WRITE ONLY, TRUNCATE TO ZERO */
+ ndbrequire(cfsFirstfreeconnect != RNIL);
+ seizeFsConnectRec(signal);
+ fsConnectptr.p->fragrecPtr = fragrecptr.i;
+ fsConnectptr.p->fsState = WAIT_OPEN_DATA_FILE_FOR_WRITE;
+ /* ----------- FILENAME (FILESYSTEM)/D3/DBACC/"T"TABID/"F"FRAGID/"S"VERSIONID.DATA ------------ */
+ /* ************************ */
+ /* FSOPENREQ */
+ /* ************************ */
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = fsConnectptr.i;
+ signal->theData[2] = tabptr.i; /* TABLE IDENTITY */
+ signal->theData[3] = rootfragrecptr.p->fragmentid[0]; /* FRAGMENT IDENTITY */
+ signal->theData[4] = lcpConnectptr.p->localCheckPid; /* CHECKPOINT ID */
+ signal->theData[5] = tlfrTmp1;
+ signal->theData[6] = tlfrTmp2;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+ return;
+}//Dbacc::execLCP_FRAGIDREQ()
+
+/* ******************--------------------------------------------------------------- */
+/* FSOPENCONF OPENFILE CONF */
+/* SENDER: FS, LEVEL B */
+/* ENTER FSOPENCONF WITH */
+/* FS_CONNECTPTR, FS_CONNECTION PTR */
+/* TUSERPOINTER, FILE POINTER */
+/* ******************--------------------------------------------------------------- */
+void Dbacc::lcpFsOpenConfLab(Signal* signal)
+{
+ fsConnectptr.p->fsPtr = tuserptr;
+ fragrecptr.i = fsConnectptr.p->fragrecPtr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ fragrecptr.p->activeDataFilePage = 1; /* ZERO IS KEPT FOR PAGE_ZERO */
+ fragrecptr.p->fsConnPtr = fsConnectptr.i;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ lcpConnectptr.i = rootfragrecptr.p->lcpPtr;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ if (rootfragrecptr.p->fragmentptr[0] == fragrecptr.i) {
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[1];
+ ptrCheck(fragrecptr, cfragmentsize, fragmentrec);
+ /* ----------- FILENAME (FILESYSTEM)/D3/DBACC/"T"TABID/"F"FRAGID/"S"VERSIONID.DATA ------------ */
+ /* D6 AT FSOPENREQ =#010003FF. */
+ tlfrTmp1 = 0x010003ff; /* FILE TYPE = .DATA ,VERSION OF FILENAME = 1 */
+ tlfrTmp2 = 0x301; /* D7 CREATE, WRITE ONLY, TRUNCATE TO ZERO */
+ ndbrequire(cfsFirstfreeconnect != RNIL);
+ seizeFsConnectRec(signal);
+ fsConnectptr.p->fragrecPtr = fragrecptr.i;
+ fsConnectptr.p->fsState = WAIT_OPEN_DATA_FILE_FOR_WRITE;
+ /* ************************ */
+ /* FSOPENREQ */
+ /* ************************ */
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = fsConnectptr.i;
+ signal->theData[2] = rootfragrecptr.p->mytabptr; /* TABLE IDENTITY */
+ signal->theData[3] = rootfragrecptr.p->fragmentid[1]; /* FRAGMENT IDENTITY */
+ signal->theData[4] = lcpConnectptr.p->localCheckPid; /* CHECKPOINT ID */
+ signal->theData[5] = tlfrTmp1;
+ signal->theData[6] = tlfrTmp2;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+ return;
+ } else {
+ ndbrequire(rootfragrecptr.p->fragmentptr[1] == fragrecptr.i);
+ }//if
+ /*---- BOTH DATA FILES ARE OPEN------*/
+ /* ----IF THE UNDO FILE IS CLOSED , OPEN IT.----- */
+ if (cactiveOpenUndoFsPtr != RNIL) {
+ jam();
+ sendLcpFragidconfLab(signal);
+ return;
+ }//if
+ cactiveUndoFilePage = 0;
+ cprevUndoaddress = cminusOne;
+ cundoposition = 0;
+ clastUndoPageIdWritten = 0;
+ ndbrequire(cfsFirstfreeconnect != RNIL);
+ seizeFsConnectRec(signal);
+ fsConnectptr.p->fsState = WAIT_OPEN_UNDO_LCP;
+ fsConnectptr.p->fsPart = 0; /* FILE INDEX, SECOND FILE IN THE DIRECTORY */
+ cactiveOpenUndoFsPtr = fsConnectptr.i;
+ cactiveRootfrag = rootfragrecptr.i;
+ tlfrTmp1 = 1; /* FILE VERSION */
+ tlfrTmp1 = (tlfrTmp1 << 8) + ZLOCALLOGFILE; /* .LOCLOG = 2 */
+ tlfrTmp1 = (tlfrTmp1 << 8) + 4; /* ROOT DIRECTORY = D4 */
+ tlfrTmp1 = (tlfrTmp1 << 8) + fsConnectptr.p->fsPart; /* P2 */
+ tlfrTmp2 = 0x302; /* D7 CREATE , READ / WRITE , TRUNCATE TO ZERO */
+ /* ---FILE NAME "D4"/"DBACC"/LCP_CONNECTPTR:LOCAL_CHECK_PID/FS_CONNECTPTR:FS_PART".LOCLOG-- */
+ /* ************************ */
+ /* FSOPENREQ */
+ /* ************************ */
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = fsConnectptr.i;
+ signal->theData[2] = cminusOne; /* #FFFFFFFF */
+ signal->theData[3] = cminusOne; /* #FFFFFFFF */
+ signal->theData[4] = cactiveUndoFileVersion;
+ /* A GROUP OF UNDO FILES WHICH ARE UPDATED */
+ signal->theData[5] = tlfrTmp1;
+ signal->theData[6] = tlfrTmp2;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+ return;
+}//Dbacc::lcpFsOpenConfLab()
+
+void Dbacc::lcpOpenUndofileConfLab(Signal* signal)
+{
+ ptrGuard(fsConnectptr);
+ fsConnectptr.p->fsState = WAIT_NOTHING;
+ rootfragrecptr.i = cactiveRootfrag;
+ ptrCheck(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ fsConnectptr.p->fsPtr = tuserptr;
+ sendLcpFragidconfLab(signal);
+ return;
+}//Dbacc::lcpOpenUndofileConfLab()
+
+void Dbacc::sendLcpFragidconfLab(Signal* signal)
+{
+ ptrGuard(rootfragrecptr);
+ lcpConnectptr.i = rootfragrecptr.p->lcpPtr;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ /* ************************ */
+ /* LCP_FRAGIDCONF */
+ /* ************************ */
+ signal->theData[0] = lcpConnectptr.p->lcpUserptr;
+ signal->theData[1] = lcpConnectptr.i;
+ signal->theData[2] = 2;
+ /* NO OF LOCAL FRAGMENTS */
+ signal->theData[3] = rootfragrecptr.p->fragmentid[0];
+ signal->theData[4] = rootfragrecptr.p->fragmentid[1];
+ signal->theData[5] = RNIL;
+ signal->theData[6] = RNIL;
+ sendSignal(lcpConnectptr.p->lcpUserblockref, GSN_LCP_FRAGIDCONF, signal, 7, JBB);
+ return;
+}//Dbacc::sendLcpFragidconfLab()
+
+/* ******************--------------------------------------------------------------- */
+/* LCP_HOLDOPERATION REQUEST FOR LIST OF STOPED OPERATION */
+/* SENDER: LQH, LEVEL B */
+/* ENTER LCP_HOLDOPREQ WITH */
+/* LCP_CONNECTPTR CONNECTION POINTER */
+/* TFID, LOCAL FRAGMENT ID */
+/* THOLD_PREV_SENT_OP NR OF SENT OPERATIONS AT */
+/* PREVIOUS SIGNALS */
+/* TLQH_POINTER LQH USER POINTER */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* LCP_HOLDOPERATION REQUEST FOR LIST OF STOPED OPERATION */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execLCP_HOLDOPREQ(Signal* signal)
+{
+ Uint32 tholdPrevSentOp;
+
+ jamEntry();
+ lcpConnectptr.i = signal->theData[0]; /* CONNECTION POINTER */
+ tfid = signal->theData[1]; /* LOCAL FRAGMENT ID */
+ tholdPrevSentOp = signal->theData[2]; /* NR OF SENT OPERATIONS AT */
+ /* PREVIOUS SIGNALS */
+ tlqhPointer = signal->theData[3]; /* LQH USER POINTER */
+
+ tresult = 0;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ ndbrequire(lcpConnectptr.p->lcpstate == LCP_ACTIVE);
+ rootfragrecptr.i = lcpConnectptr.p->rootrecptr;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ if (rootfragrecptr.p->fragmentid[0] == tfid) {
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[0];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ } else {
+ ndbrequire(rootfragrecptr.p->fragmentid[1] == tfid);
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[1];
+ }//if
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ fragrecptr.p->lcpLqhPtr = tlqhPointer;
+ if (tholdPrevSentOp != 0) {
+ ndbrequire(fragrecptr.p->fragState == SEND_QUE_OP);
+ } else if (tholdPrevSentOp == 0) {
+ jam();
+ fragrecptr.p->fragState = SEND_QUE_OP;
+ fragrecptr.p->stopQueOp = ZTRUE;
+ fragrecptr.p->sentWaitInQueOp = fragrecptr.p->firstWaitInQueOp;
+ }//if
+ tholdSentOp = 0; /* NR OF OPERATION WHICH ARE SENT THIS TIME */
+ operationRecPtr.i = fragrecptr.p->sentWaitInQueOp;
+
+ /* --------------------------------------------- */
+ /* GO THROUGH ALL OPERATION IN THE WAIT */
+ /* LIST AND SEND THE LQH CONNECTION PTR OF THE */
+ /* OPERATIONS TO THE LQH BLOCK. MAX 23 0PERATION */
+ /* PER SIGNAL */
+ /* --------------------------------------------- */
+ while (operationRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ ckeys[tholdSentOp] = operationRecPtr.p->userptr;
+ operationRecPtr.i = operationRecPtr.p->nextQueOp;
+ tholdSentOp++;
+ if ((tholdSentOp >= 23) &&
+ (operationRecPtr.i != RNIL)) {
+ jam();
+ /* ----------------------------------------------- */
+ /* THERE IS MORE THAN 23 WAIT OPERATION. WE */
+ /* HAVE TO SEND THESE 23 AND WAITE FOR NEXT SIGNAL */
+ /* ----------------------------------------------- */
+ tholdMore = ZTRUE; /* SECOUND DATA AT THE CONF SIGNAL , = MORE */
+ fragrecptr.p->sentWaitInQueOp = operationRecPtr.i;
+ sendholdconfsignalLab(signal);
+ return;
+ }//if
+ }//while
+ /* ----------------------------------------------- */
+ /* OPERATION_REC_PTR = RNIL */
+ /* THERE IS NO MORE WAITING OPERATION, STATE OF */
+ /* THE FRAGMENT RRECORD IS CHANGED AND RETURN */
+ /* SIGNAL IS SENT */
+ /* ----------------------------------------------- */
+ fragrecptr.p->sentWaitInQueOp = RNIL;
+ tholdMore = ZFALSE; /* SECOND DATA AT THE CONF SIGNAL , = NOT MORE */
+ fragrecptr.p->fragState = WAIT_ACC_LCPREQ;
+ sendholdconfsignalLab(signal);
+ return;
+}//Dbacc::execLCP_HOLDOPREQ()
+
+void Dbacc::sendholdconfsignalLab(Signal* signal)
+{
+ tholdMore = (tholdMore << 16) + tholdSentOp;
+ /* SECOND SIGNAL DATA, LENGTH + MORE */
+ /* ************************ */
+ /* LCP_HOLDOPCONF */
+ /* ************************ */
+ signal->theData[0] = fragrecptr.p->lcpLqhPtr;
+ signal->theData[1] = tholdMore;
+ signal->theData[2] = ckeys[0];
+ signal->theData[3] = ckeys[1];
+ signal->theData[4] = ckeys[2];
+ signal->theData[5] = ckeys[3];
+ signal->theData[6] = ckeys[4];
+ signal->theData[7] = ckeys[5];
+ signal->theData[8] = ckeys[6];
+ signal->theData[9] = ckeys[7];
+ signal->theData[10] = ckeys[8];
+ signal->theData[11] = ckeys[9];
+ signal->theData[12] = ckeys[10];
+ signal->theData[13] = ckeys[11];
+ signal->theData[14] = ckeys[12];
+ signal->theData[15] = ckeys[13];
+ signal->theData[16] = ckeys[14];
+ signal->theData[17] = ckeys[15];
+ signal->theData[18] = ckeys[16];
+ signal->theData[19] = ckeys[17];
+ signal->theData[20] = ckeys[18];
+ signal->theData[21] = ckeys[19];
+ signal->theData[22] = ckeys[20];
+ signal->theData[23] = ckeys[21];
+ signal->theData[24] = ckeys[22];
+ sendSignal(lcpConnectptr.p->lcpUserblockref, GSN_LCP_HOLDOPCONF, signal, 25, JBA);
+ return;
+}//Dbacc::sendholdconfsignalLab()
+
+/**
+ * execACC_LCPREQ
+ * Perform local checkpoint of a fragment
+ *
+ * SENDER: LQH, LEVEL B
+ * ENTER ACC_LCPREQ WITH
+ * LCP_CONNECTPTR, OPERATION RECORD PTR
+ * TLCP_LQH_CHECK_V, LQH'S LOCAL FRAG CHECK VALUE
+ * TLCP_LOCAL_FRAG_ID, LOCAL FRAG ID
+ *
+ */
+void Dbacc::execACC_LCPREQ(Signal* signal)
+{
+ Uint32 tlcpLocalFragId;
+ Uint32 tlcpLqhCheckV;
+
+ jamEntry();
+ lcpConnectptr.i = signal->theData[0]; // CONNECTION PTR
+ tlcpLqhCheckV = signal->theData[1]; // LQH'S LOCAL FRAG CHECK VALUE
+ tlcpLocalFragId = signal->theData[2]; // LOCAL FRAG ID
+ tresult = 0;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ ndbrequire(lcpConnectptr.p->lcpstate == LCP_ACTIVE);
+
+ rootfragrecptr.i = lcpConnectptr.p->rootrecptr;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ if (rootfragrecptr.p->fragmentid[0] == tlcpLocalFragId) {
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[0];
+ } else {
+ ndbrequire(rootfragrecptr.p->fragmentid[1] == tlcpLocalFragId);
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[1];
+ }//if
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ ndbrequire(fragrecptr.p->fragState == WAIT_ACC_LCPREQ);
+ fragrecptr.p->lcpLqhPtr = tlcpLqhCheckV;
+
+ Page8Ptr zeroPagePtr;
+ seizeLcpPage(zeroPagePtr);
+ fragrecptr.p->zeroPagePtr = zeroPagePtr.i;
+ fragrecptr.p->prevUndoposition = cminusOne;
+ initRootFragPageZero(rootfragrecptr, zeroPagePtr);
+ initFragPageZero(fragrecptr, zeroPagePtr);
+ /*-----------------------------------------------------------------*/
+ /* SEIZE ZERO PAGE FIRST AND THEN SEIZE DATA PAGES IN */
+ /* BACKWARDS ORDER. THIS IS TO ENSURE THAT WE GET THE PAGES */
+ /* IN ORDER. ON WINDOWS NT THIS WILL BE A BENEFIT SINCE WE */
+ /* CAN THEN DO 1 WRITE_FILE INSTEAD OF 8. */
+ /* WHEN WE RELEASE THE PAGES WE RELEASE THEM IN THE OPPOSITE */
+ /* ORDER. */
+ /*-----------------------------------------------------------------*/
+ for (Uint32 taspTmp = ZWRITEPAGESIZE - 1; (Uint32)~taspTmp; taspTmp--) {
+ Page8Ptr dataPagePtr;
+ jam();
+ ndbrequire(fragrecptr.p->datapages[taspTmp] == RNIL);
+ seizeLcpPage(dataPagePtr);
+ fragrecptr.p->datapages[taspTmp] = dataPagePtr.i;
+ }//for
+ fragrecptr.p->lcpMaxDirIndex = fragrecptr.p->dirsize;
+ fragrecptr.p->lcpMaxOverDirIndex = fragrecptr.p->lastOverIndex;
+ fragrecptr.p->createLcp = ZTRUE;
+ operationRecPtr.i = fragrecptr.p->lockOwnersList;
+ while (operationRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+
+ if ((operationRecPtr.p->operation == ZINSERT) ||
+ (operationRecPtr.p->elementIsDisappeared == ZTRUE)){
+ /*******************************************************************
+ * Only log inserts and elements that are marked as dissapeared.
+ * All other operations update the element header and that is handled
+ * when pages are written to disk
+ ********************************************************************/
+ undopageptr.i = (cundoposition>>ZUNDOPAGEINDEXBITS) & (cundopagesize-1);
+ ptrAss(undopageptr, undopage);
+ theadundoindex = cundoposition & ZUNDOPAGEINDEX_MASK;
+ tundoindex = theadundoindex + ZUNDOHEADSIZE;
+
+ writeUndoOpInfo(signal);/* THE INFORMATION ABOUT ELEMENT HEADER, STORED*/
+ /* IN OP REC, IS WRITTEN AT UNDO PAGES */
+ cundoElemIndex = 0;/* DEFAULT VALUE USED BY WRITE_UNDO_HEADER SUBROTINE */
+ writeUndoHeader(signal, RNIL, UndoHeader::ZOP_INFO); /* WRITE THE HEAD OF THE UNDO ELEMENT */
+ checkUndoPages(signal); /* SEND UNDO PAGE TO DISK WHEN A GROUP OF */
+ /* UNDO PAGES,CURRENTLY 8, IS FILLED */
+ }//if
+
+ operationRecPtr.i = operationRecPtr.p->nextLockOwnerOp;
+ }//while
+
+ signal->theData[0] = fragrecptr.p->lcpLqhPtr;
+ sendSignal(lcpConnectptr.p->lcpUserblockref, GSN_ACC_LCPSTARTED,
+ signal, 1, JBA);
+
+ fragrecptr.p->activeDataPage = 0;
+ fragrecptr.p->lcpDirIndex = 0;
+ fragrecptr.p->fragState = LCP_SEND_PAGES;
+
+ signal->theData[0] = lcpConnectptr.i;
+ signal->theData[1] = fragrecptr.i;
+ sendSignal(cownBlockref, GSN_ACC_SAVE_PAGES, signal, 2, JBB);
+ return;
+}//Dbacc::execACC_LCPREQ()
+
+/* ******************--------------------------------------------------------------- */
+/* ACC_SAVE_PAGES A GROUP OF PAGES IS ALLOCATED. THE PAGES AND OVERFLOW */
+/* PAGES OF THE FRAGMENT ARE COPIED IN THEM AND IS SEND TO */
+/* THE DATA FILE OF THE CHECK POINT. */
+/* SENDER: ACC, LEVEL B */
+/* ENTER ACC_SAVE_PAGES WITH */
+/* LCP_CONNECTPTR, CONNECTION RECORD PTR */
+/* FRAGRECPTR FRAGMENT RECORD PTR */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* ACC_SAVE_PAGES REQUEST TO SEND THE PAGE TO DISK */
+/* ******************------------------------------+ UNDO PAGES */
+/* SENDER: ACC, LEVEL B */
+void Dbacc::execACC_SAVE_PAGES(Signal* signal)
+{
+ jamEntry();
+ lcpConnectptr.i = signal->theData[0];
+ /* CONNECTION RECORD PTR */
+ fragrecptr.i = signal->theData[1];
+ /* FRAGMENT RECORD PTR */
+ tresult = 0;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ if (lcpConnectptr.p->lcpstate != LCP_ACTIVE) {
+ jam();
+ sendSystemerror(signal);
+ return;
+ }//if
+ if (ERROR_INSERTED(3000)) {
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ if (rootfragrecptr.p->mytabptr == c_errorInsert3000_TableId){
+ ndbout << "Delay writing of datapages" << endl;
+ // Delay writing of pages
+ jam();
+ sendSignalWithDelay(cownBlockref, GSN_ACC_SAVE_PAGES, signal, 1000, 2);
+ return;
+ }
+ }
+ if (clblPageCounter == 0) {
+ jam();
+ signal->theData[0] = lcpConnectptr.i;
+ signal->theData[1] = fragrecptr.i;
+ sendSignalWithDelay(cownBlockref, GSN_ACC_SAVE_PAGES, signal, 100, 2);
+ return;
+ } else {
+ jam();
+ clblPageCounter = clblPageCounter - 1;
+ }//if
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ if (fragrecptr.p->fragState == LCP_SEND_PAGES) {
+ jam();
+ savepagesLab(signal);
+ return;
+ } else {
+ if (fragrecptr.p->fragState == LCP_SEND_OVER_PAGES) {
+ jam();
+ saveOverPagesLab(signal);
+ return;
+ } else {
+ ndbrequire(fragrecptr.p->fragState == LCP_SEND_ZERO_PAGE);
+ jam();
+ saveZeroPageLab(signal);
+ return;
+ }//if
+ }//if
+}//Dbacc::execACC_SAVE_PAGES()
+
+void Dbacc::savepagesLab(Signal* signal)
+{
+ DirRangePtr spDirRangePtr;
+ DirectoryarrayPtr spDirptr;
+ Page8Ptr aspPageptr;
+ Page8Ptr aspCopyPageptr;
+ Uint32 taspDirindex;
+ Uint32 taspDirIndex;
+ Uint32 taspIndex;
+
+ if ((fragrecptr.p->lcpDirIndex >= fragrecptr.p->dirsize) ||
+ (fragrecptr.p->lcpDirIndex >= fragrecptr.p->lcpMaxDirIndex)) {
+ jam();
+ endsavepageLab(signal);
+ return;
+ }//if
+ /* SOME EXPAND PROCESSES HAVE BEEN PERFORMED. */
+ /* THE ADDED PAGE ARE NOT SENT TO DISK */
+ arrGuard(fragrecptr.p->activeDataPage, 8);
+ aspCopyPageptr.i = fragrecptr.p->datapages[fragrecptr.p->activeDataPage];
+ ptrCheckGuard(aspCopyPageptr, cpagesize, page8);
+ taspDirindex = fragrecptr.p->lcpDirIndex; /* DIRECTORY OF ACTIVE PAGE */
+ spDirRangePtr.i = fragrecptr.p->directory;
+ taspDirIndex = taspDirindex >> 8;
+ taspIndex = taspDirindex & 0xff;
+ ptrCheckGuard(spDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(taspDirIndex, 256);
+ spDirptr.i = spDirRangePtr.p->dirArray[taspDirIndex];
+ ptrCheckGuard(spDirptr, cdirarraysize, directoryarray);
+ aspPageptr.i = spDirptr.p->pagep[taspIndex];
+ ptrCheckGuard(aspPageptr, cpagesize, page8);
+ ndbrequire(aspPageptr.p->word32[ZPOS_PAGE_ID] == fragrecptr.p->lcpDirIndex);
+ lcnPageptr = aspPageptr;
+ lcnCopyPageptr = aspCopyPageptr;
+ lcpCopyPage(signal);
+ fragrecptr.p->lcpDirIndex++;
+ fragrecptr.p->activeDataPage++;
+ if (fragrecptr.p->activeDataPage < ZWRITEPAGESIZE) {
+ jam();
+ signal->theData[0] = lcpConnectptr.i;
+ signal->theData[1] = fragrecptr.i;
+ sendSignal(cownBlockref, GSN_ACC_SAVE_PAGES, signal, 2, JBB);
+ return;
+ }//if
+ senddatapagesLab(signal);
+ return;
+}//Dbacc::savepagesLab()
+
+/* FRAGRECPTR:ACTIVE_DATA_PAGE = ZWRITEPAGESIZE */
+/* SEND A GROUP OF PAGES TO DISK */
+void Dbacc::senddatapagesLab(Signal* signal)
+{
+ fsConnectptr.i = fragrecptr.p->fsConnPtr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ seizeFsOpRec(signal);
+ initFsOpRec(signal);
+ fsOpptr.p->fsOpstate = WAIT_WRITE_DATA;
+ ndbrequire(fragrecptr.p->activeDataPage <= 8);
+ for (Uint32 i = 0; i < fragrecptr.p->activeDataPage; i++) {
+ signal->theData[i + 6] = fragrecptr.p->datapages[i];
+ }//for
+ signal->theData[fragrecptr.p->activeDataPage + 6] = fragrecptr.p->activeDataFilePage;
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsOpptr.i;
+ signal->theData[3] = 0x2;
+ signal->theData[4] = ZPAGE8_BASE_ADD;
+ signal->theData[5] = fragrecptr.p->activeDataPage;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 15, JBA);
+ return;
+}//Dbacc::senddatapagesLab()
+
+void Dbacc::endsavepageLab(Signal* signal)
+{
+ Page8Ptr espPageidptr;
+
+ espPageidptr.i = fragrecptr.p->zeroPagePtr;
+ ptrCheckGuard(espPageidptr, cpagesize, page8);
+ dbgWord32(espPageidptr, ZPAGEZERO_NO_PAGES, fragrecptr.p->lcpDirIndex);
+ espPageidptr.p->word32[ZPAGEZERO_NO_PAGES] = fragrecptr.p->lcpDirIndex;
+ fragrecptr.p->fragState = LCP_SEND_OVER_PAGES;
+ fragrecptr.p->noOfStoredOverPages = 0;
+ fragrecptr.p->lcpDirIndex = 0;
+ saveOverPagesLab(signal);
+ return;
+}//Dbacc::endsavepageLab()
+
+/* ******************--------------------------------------------------------------- */
+/* ACC_SAVE_OVER_PAGES CONTINUE SAVING THE LEFT OVERPAGES. */
+/* ******************--------------------------------------------------------------- */
+void Dbacc::saveOverPagesLab(Signal* signal)
+{
+ DirRangePtr sopDirRangePtr;
+ DirectoryarrayPtr sopOverflowDirptr;
+ Page8Ptr sopPageptr;
+ Page8Ptr sopCopyPageptr;
+ Uint32 tsopDirindex;
+ Uint32 tsopDirInd;
+ Uint32 tsopIndex;
+
+ if ((fragrecptr.p->lcpDirIndex >= fragrecptr.p->lastOverIndex) ||
+ (fragrecptr.p->lcpDirIndex >= fragrecptr.p->lcpMaxOverDirIndex)) {
+ jam();
+ endsaveoverpageLab(signal);
+ return;
+ }//if
+ arrGuard(fragrecptr.p->activeDataPage, 8);
+ sopCopyPageptr.i = fragrecptr.p->datapages[fragrecptr.p->activeDataPage];
+ ptrCheckGuard(sopCopyPageptr, cpagesize, page8);
+ tsopDirindex = fragrecptr.p->lcpDirIndex;
+ sopDirRangePtr.i = fragrecptr.p->overflowdir;
+ tsopDirInd = tsopDirindex >> 8;
+ tsopIndex = tsopDirindex & 0xff;
+ ptrCheckGuard(sopDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(tsopDirInd, 256);
+ sopOverflowDirptr.i = sopDirRangePtr.p->dirArray[tsopDirInd];
+ ptrCheckGuard(sopOverflowDirptr, cdirarraysize, directoryarray);
+ sopPageptr.i = sopOverflowDirptr.p->pagep[tsopIndex];
+ fragrecptr.p->lcpDirIndex++;
+ if (sopPageptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(sopPageptr, cpagesize, page8);
+ ndbrequire(sopPageptr.p->word32[ZPOS_PAGE_ID] == tsopDirindex);
+ ndbrequire(((sopPageptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) != ZNORMAL_PAGE_TYPE);
+ lcnPageptr = sopPageptr;
+ lcnCopyPageptr = sopCopyPageptr;
+ lcpCopyPage(signal);
+ fragrecptr.p->noOfStoredOverPages++;
+ fragrecptr.p->activeDataPage++;
+ if ((sopPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] == 0)) {
+ //ndbrequire(((sopPageptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) == ZOVERFLOW_PAGE_TYPE);
+ if (((sopPageptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) ==
+ ZOVERFLOW_PAGE_TYPE) {
+ /*--------------------------------------------------------------------------------*/
+ /* THE PAGE IS EMPTY AND WAITING TO BE RELEASED. IT COULD NOT BE RELEASED */
+ /* EARLIER SINCE IT WAS PART OF A LOCAL CHECKPOINT. */
+ /*--------------------------------------------------------------------------------*/
+ jam();
+ ropPageptr = sopPageptr;
+ releaseOverpage(signal);
+ } else if (((sopPageptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) ==
+ ZLONG_PAGE_TYPE) {
+ //----------------------------------------------------------------------
+ // The long key page is empty, release it.
+ //----------------------------------------------------------------------
+ jam();
+ rlopPageptr = sopPageptr;
+ releaseLongPage(signal);
+ } else {
+ jam();
+ sendSystemerror(signal);
+ }
+ }//if
+ }
+ if (fragrecptr.p->activeDataPage == ZWRITEPAGESIZE) {
+ jam();
+ senddatapagesLab(signal);
+ return;
+ }//if
+ signal->theData[0] = lcpConnectptr.i;
+ signal->theData[1] = fragrecptr.i;
+ sendSignal(cownBlockref, GSN_ACC_SAVE_PAGES, signal, 2, JBB);
+ return;
+}//Dbacc::saveOverPagesLab()
+
+void Dbacc::endsaveoverpageLab(Signal* signal)
+{
+ Page8Ptr esoPageidptr;
+
+ esoPageidptr.i = fragrecptr.p->zeroPagePtr;
+ ptrCheckGuard(esoPageidptr, cpagesize, page8);
+ dbgWord32(esoPageidptr, ZPAGEZERO_NO_OVER_PAGE, fragrecptr.p->noOfStoredOverPages);
+ esoPageidptr.p->word32[ZPAGEZERO_NO_OVER_PAGE] = fragrecptr.p->noOfStoredOverPages;
+ fragrecptr.p->fragState = LCP_SEND_ZERO_PAGE;
+ if (fragrecptr.p->activeDataPage != 0) {
+ jam();
+ senddatapagesLab(signal); /* SEND LEFT PAGES TO DISK */
+ return;
+ }//if
+ saveZeroPageLab(signal);
+ return;
+}//Dbacc::endsaveoverpageLab()
+
+/* ******************--------------------------------------------------------------- */
+/* ACC_SAVE_ZERO_PAGE PAGE ZERO IS SENT TO DISK.IT IS THE LAST STAGE AT THE */
+/* CREATION LCP. ACC_LCPCONF IS RETURND. */
+/* ******************--------------------------------------------------------------- */
+void Dbacc::saveZeroPageLab(Signal* signal)
+{
+ Page8Ptr szpPageidptr;
+ Uint32 Tchs;
+ Uint32 Ti;
+
+ fragrecptr.p->createLcp = ZFALSE;
+ fsConnectptr.i = fragrecptr.p->fsConnPtr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ szpPageidptr.i = fragrecptr.p->zeroPagePtr;
+ ptrCheckGuard(szpPageidptr, cpagesize, page8);
+ dbgWord32(szpPageidptr, ZPAGEZERO_PREV_UNDOP, fragrecptr.p->prevUndoposition);
+ szpPageidptr.p->word32[ZPAGEZERO_PREV_UNDOP] = fragrecptr.p->prevUndoposition;
+ dbgWord32(szpPageidptr, ZPAGEZERO_NEXT_UNDO_FILE, cactiveUndoFileVersion);
+ szpPageidptr.p->word32[ZPAGEZERO_NEXT_UNDO_FILE] = cactiveUndoFileVersion;
+ fragrecptr.p->fragState = WAIT_ZERO_PAGE_STORED;
+
+ /* --------------------------------------------------------------------------------- */
+ // Calculate the checksum and store it for the zero page of the fragment.
+ /* --------------------------------------------------------------------------------- */
+ szpPageidptr.p->word32[ZPOS_CHECKSUM] = 0;
+ Tchs = 0;
+ for (Ti = 0; Ti < 2048; Ti++) {
+ Tchs = Tchs ^ szpPageidptr.p->word32[Ti];
+ }//for
+ szpPageidptr.p->word32[ZPOS_CHECKSUM] = Tchs;
+ dbgWord32(szpPageidptr, ZPOS_CHECKSUM, Tchs);
+
+ seizeFsOpRec(signal);
+ initFsOpRec(signal);
+ fsOpptr.p->fsOpstate = WAIT_WRITE_DATA;
+ if (clblPageCounter > 0) {
+ jam();
+ clblPageCounter = clblPageCounter - 1;
+ } else {
+ jam();
+ clblPageOver = clblPageOver + 1;
+ }//if
+ /* ************************ */
+ /* FSWRITEREQ */
+ /* ************************ */
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsOpptr.i;
+ signal->theData[3] = 0x10;
+ /* FLAG = LIST MEM PAGES, LIST FILE PAGES */
+ /* SYNC FILE AFTER WRITING */
+ signal->theData[4] = ZPAGE8_BASE_ADD;
+ signal->theData[5] = 1;
+ /* NO OF PAGES */
+ signal->theData[6] = fragrecptr.p->zeroPagePtr;
+ /* ZERO PAGE */
+ signal->theData[7] = 0;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+ /* ZERO PAGE AT DATA FILE */
+ return;
+}//Dbacc::saveZeroPageLab()
+
+/* ******************--------------------------------------------------------------- */
+/* FSWRITECONF OPENFILE CONF */
+/* ENTER FSWRITECONF WITH SENDER: FS, LEVEL B */
+/* FS_OPPTR FS_CONNECTION PTR */
+/* ******************--------------------------------------------------------------- */
+void Dbacc::lcpCloseDataFileLab(Signal* signal)
+{
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ lcpConnectptr.i = rootfragrecptr.p->lcpPtr;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ fsConnectptr.p->fsState = LCP_CLOSE_DATA;
+ /* ************************ */
+ /* FSCLOSEREQ */
+ /* ************************ */
+ /* CLOSE DATA FILE */
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsConnectptr.i;
+ signal->theData[3] = ZFALSE;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+ /* FLAG = 0, DO NOT DELETE FILE */
+ return;
+}//Dbacc::lcpCloseDataFileLab()
+
+void Dbacc::checkSyncUndoPagesLab(Signal* signal)
+{
+ fragrecptr.i = fsConnectptr.p->fragrecPtr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ releaseFsConnRec(signal);
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ lcpConnectptr.i = rootfragrecptr.p->lcpPtr;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ switch (lcpConnectptr.p->syncUndopageState) {
+ case WAIT_NOTHING:
+ jam();
+ lcpConnectptr.p->syncUndopageState = WAIT_ONE_CONF;
+ break;
+ case WAIT_ONE_CONF:
+ jam();
+ lcpConnectptr.p->syncUndopageState = WAIT_TWO_CONF;
+ break;
+ default:
+ jam();
+ sendSystemerror(signal);
+ return;
+ break;
+ }//switch
+
+ /* ACTIVE UNDO PAGE ID */
+ Uint32 tundoPageId = cundoposition >> ZUNDOPAGEINDEXBITS;
+ tmp1 = tundoPageId - (tundoPageId & (ZWRITE_UNDOPAGESIZE - 1));
+ /* START PAGE OF THE LAST UNDO PAGES GROUP */
+ tmp2 = (tundoPageId - tmp1) + 1; /* NO OF LEFT UNDO PAGES */
+ tmp1 = tmp1 & (cundopagesize - 1); /* 1 MBYTE PAGE WINDOW IN MEMORY */
+ fsConnectptr.i = cactiveOpenUndoFsPtr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ seizeFsOpRec(signal);
+ initFsOpRec(signal);
+ fsOpptr.p->fsOpstate = WAIT_WRITE_UNDO;
+ fsOpptr.p->fsOpMemPage = tundoPageId; /* RECORD MEMORY PAGE WRITTEN */
+ if (clblPageCounter >= (4 * tmp2)) {
+ jam();
+ clblPageCounter = clblPageCounter - (4 * tmp2);
+ } else {
+ jam();
+ clblPageOver = clblPageOver + ((4 * tmp2) - clblPageCounter);
+ clblPageCounter = 0;
+ }//if
+ /* ************************ */
+ /* FSWRITEREQ */
+ /* ************************ */
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsOpptr.i;
+ /* FLAG = START MEM PAGES, START FILE PAGES */
+ /* SYNC FILE AFTER WRITING */
+ signal->theData[3] = 0x11;
+ signal->theData[4] = ZUNDOPAGE_BASE_ADD;
+ /* NO OF UNDO PAGES */
+ signal->theData[5] = tmp2;
+ /* FIRST MEMORY PAGE */
+ signal->theData[6] = tmp1;
+ /* ACTIVE PAGE AT UNDO FILE */
+ signal->theData[7] = cactiveUndoFilePage;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+
+ return;
+}//Dbacc::checkSyncUndoPagesLab()
+
+void Dbacc::checkSendLcpConfLab(Signal* signal)
+{
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ lcpConnectptr.i = rootfragrecptr.p->lcpPtr;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ ndbrequire(lcpConnectptr.p->lcpstate == LCP_ACTIVE);
+ switch (lcpConnectptr.p->syncUndopageState) {
+ case WAIT_ONE_CONF:
+ jam();
+ lcpConnectptr.p->syncUndopageState = WAIT_NOTHING;
+ break;
+ case WAIT_TWO_CONF:
+ jam();
+ lcpConnectptr.p->syncUndopageState = WAIT_ONE_CONF;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ lcpConnectptr.p->noOfLcpConf++;
+ ndbrequire(lcpConnectptr.p->noOfLcpConf <= 2);
+ fragrecptr.p->fragState = ACTIVEFRAG;
+ rlpPageptr.i = fragrecptr.p->zeroPagePtr;
+ ptrCheckGuard(rlpPageptr, cpagesize, page8);
+ releaseLcpPage(signal);
+ fragrecptr.p->zeroPagePtr = RNIL;
+ for (Uint32 i = 0; i < ZWRITEPAGESIZE; i++) {
+ jam();
+ if (fragrecptr.p->datapages[i] != RNIL) {
+ jam();
+ rlpPageptr.i = fragrecptr.p->datapages[i];
+ ptrCheckGuard(rlpPageptr, cpagesize, page8);
+ releaseLcpPage(signal);
+ fragrecptr.p->datapages[i] = RNIL;
+ }//if
+ }//for
+ signal->theData[0] = fragrecptr.p->lcpLqhPtr;
+ sendSignal(lcpConnectptr.p->lcpUserblockref, GSN_ACC_LCPCONF, signal, 1, JBB);
+ if (lcpConnectptr.p->noOfLcpConf == 2) {
+ jam();
+ releaseLcpConnectRec(signal);
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ rootfragrecptr.p->rootState = ACTIVEROOT;
+ }//if
+}//Dbacc::checkSendLcpConfLab()
+
+/* ******************--------------------------------------------------------------- */
+/* ACC_CONTOPREQ */
+/* SENDER: LQH, LEVEL B */
+/* ENTER ACC_CONTOPREQ WITH */
+/* LCP_CONNECTPTR */
+/* TMP1 LOCAL FRAG ID */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* ACC_CONTOPREQ COMMIT TRANSACTION */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execACC_CONTOPREQ(Signal* signal)
+{
+ Uint32 tcorLocalFrag;
+
+ jamEntry();
+ lcpConnectptr.i = signal->theData[0];
+ /* CONNECTION PTR */
+ tcorLocalFrag = signal->theData[1];
+ /* LOCAL FRAG ID */
+ tresult = 0;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ ndbrequire(lcpConnectptr.p->lcpstate == LCP_ACTIVE);
+ rootfragrecptr.i = lcpConnectptr.p->rootrecptr;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ if (rootfragrecptr.p->fragmentid[0] == tcorLocalFrag) {
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[0];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ } else {
+ ndbrequire(rootfragrecptr.p->fragmentid[1] == tcorLocalFrag);
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[1];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ }//if
+ operationRecPtr.i = fragrecptr.p->firstWaitInQueOp;
+ fragrecptr.p->sentWaitInQueOp = RNIL;
+ fragrecptr.p->stopQueOp = ZFALSE;
+ while (operationRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ if (operationRecPtr.p->opState == WAIT_EXE_OP) {
+ jam();
+ //------------------------------------------------------------
+ // Indicate that we are now a normal waiter in the queue. We
+ // will remove the operation from the queue as part of starting
+ // operation again.
+ //------------------------------------------------------------
+ operationRecPtr.p->opState = WAIT_IN_QUEUE;
+ executeNextOperation(signal);
+ }//if
+ operationRecPtr.i = operationRecPtr.p->nextQueOp;
+ }//while
+ signal->theData[0] = fragrecptr.p->lcpLqhPtr;
+ sendSignal(lcpConnectptr.p->lcpUserblockref, GSN_ACC_CONTOPCONF, signal, 1, JBA);
+ return; /* ALL QUEUED OPERATION ARE RESTARTED IF NEEDED. */
+}//Dbacc::execACC_CONTOPREQ()
+
+/* ******************--------------------------------------------------------------- */
+/* END_LCPREQ END OF LOCAL CHECK POINT */
+/* ENTER END_LCPREQ WITH SENDER: LQH, LEVEL B */
+/* CLQH_PTR, LQH PTR */
+/* CLQH_BLOCK_REF LQH BLOCK REF */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* END_LCPREQ PERFORM A LOCAL CHECK POINT */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execEND_LCPREQ(Signal* signal)
+{
+ jamEntry();
+ clqhPtr = signal->theData[0];
+ /* LQH PTR */
+ clqhBlockRef = signal->theData[1];
+ /* LQH BLOCK REF */
+ tresult = 0;
+ fsConnectptr.i = cactiveOpenUndoFsPtr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ fsConnectptr.p->fsState = WAIT_CLOSE_UNDO; /* CLOSE FILE AFTER WRITTING */
+ /* ************************ */
+ /* FSCLOSEREQ */
+ /* ************************ */
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsConnectptr.i;
+ signal->theData[3] = ZFALSE;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+ /* FLAG = 0, DO NOT DELETE FILE */
+ cactiveUndoFileVersion = RNIL;
+ cactiveOpenUndoFsPtr = RNIL;
+ /* ************************ */
+ /* END_LCPCONF */
+ /* ************************ */
+ signal->theData[0] = clqhPtr;
+ sendSignal(clqhBlockRef, GSN_END_LCPCONF, signal, 1, JBB);
+ return;
+}//Dbacc::execEND_LCPREQ()
+
+/*-----------------------------------------------------------------*/
+/* WHEN WE COPY THE PAGE WE ALSO WRITE THE ELEMENT HEADER AS */
+/* UNLOCKED IF THEY ARE CURRENTLY LOCKED. */
+/*-----------------------------------------------------------------*/
+void Dbacc::lcpCopyPage(Signal* signal)
+{
+ Uint32 tlcnNextContainer;
+ Uint32 tlcnTmp;
+ Uint32 tlcnConIndex;
+ Uint32 tlcnIndex;
+ Uint32 Tmp1;
+ Uint32 Tmp2;
+ Uint32 Tmp3;
+ Uint32 Tmp4;
+ Uint32 Ti;
+ Uint32 Tchs;
+ Uint32 Tlimit;
+
+ Tchs = 0;
+ lupPageptr.p = lcnCopyPageptr.p;
+ lcnPageptr.p->word32[ZPOS_CHECKSUM] = Tchs;
+ for (Ti = 0; Ti < 32 ; Ti++) {
+ Tlimit = 16 + (Ti << 6);
+ for (tlcnTmp = (Ti << 6); tlcnTmp < Tlimit; tlcnTmp ++) {
+ Tmp1 = lcnPageptr.p->word32[tlcnTmp];
+ Tmp2 = lcnPageptr.p->word32[tlcnTmp + 16];
+ Tmp3 = lcnPageptr.p->word32[tlcnTmp + 32];
+ Tmp4 = lcnPageptr.p->word32[tlcnTmp + 48];
+
+ lcnCopyPageptr.p->word32[tlcnTmp] = Tmp1;
+ lcnCopyPageptr.p->word32[tlcnTmp + 16] = Tmp2;
+ lcnCopyPageptr.p->word32[tlcnTmp + 32] = Tmp3;
+ lcnCopyPageptr.p->word32[tlcnTmp + 48] = Tmp4;
+
+ Tchs = Tchs ^ Tmp1;
+ Tchs = Tchs ^ Tmp2;
+ Tchs = Tchs ^ Tmp3;
+ Tchs = Tchs ^ Tmp4;
+ }//for
+ }//for
+ tlcnChecksum = Tchs;
+ if (((lcnCopyPageptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) != ZLONG_PAGE_TYPE) {
+ jam();
+ if (((lcnCopyPageptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) == ZNORMAL_PAGE_TYPE) {
+ jam();
+ /*-----------------------------------------------------------------*/
+ /* TAKE CARE OF ALL 64 BUFFERS ADDRESSED BY ALGORITHM IN */
+ /* FIRST PAGE. IF THEY ARE EMPTY THEY STILL HAVE A CONTAINER */
+ /* HEADER OF 2 WORDS. */
+ /*-----------------------------------------------------------------*/
+ tlcnConIndex = ZHEAD_SIZE;
+ tlupForward = 1;
+ for (tlcnIndex = 0; tlcnIndex <= ZNO_CONTAINERS - 1; tlcnIndex++) {
+ tlupIndex = tlcnConIndex;
+ tlupElemIndex = tlcnConIndex + ZCON_HEAD_SIZE;
+ lcpUpdatePage(signal);
+ tlcnConIndex = tlcnConIndex + ZBUF_SIZE;
+ }//for
+ }//if
+ /*-----------------------------------------------------------------*/
+ /* TAKE CARE OF ALL USED BUFFERS ON THE LEFT SIDE. */
+ /*-----------------------------------------------------------------*/
+ tlcnNextContainer = (lcnCopyPageptr.p->word32[ZPOS_EMPTY_LIST] >> 23) & 0x7f;
+ while (tlcnNextContainer < ZEMPTYLIST) {
+ tlcnConIndex = (tlcnNextContainer << ZSHIFT_PLUS) - (tlcnNextContainer << ZSHIFT_MINUS);
+ tlcnConIndex = tlcnConIndex + ZHEAD_SIZE;
+ tlupIndex = tlcnConIndex;
+ tlupElemIndex = tlcnConIndex + ZCON_HEAD_SIZE;
+ tlupForward = 1;
+ lcpUpdatePage(signal);
+ tlcnNextContainer = (lcnCopyPageptr.p->word32[tlcnConIndex] >> 11) & 0x7f;
+ }//while
+ if (tlcnNextContainer == ZEMPTYLIST) {
+ jam();
+ /*empty*/;
+ } else {
+ jam();
+ sendSystemerror(signal);
+ return;
+ }//if
+ /*-----------------------------------------------------------------*/
+ /* TAKE CARE OF ALL USED BUFFERS ON THE RIGHT SIDE. */
+ /*-----------------------------------------------------------------*/
+ tlupForward = cminusOne;
+ tlcnNextContainer = (lcnCopyPageptr.p->word32[ZPOS_EMPTY_LIST] >> 16) & 0x7f;
+ while (tlcnNextContainer < ZEMPTYLIST) {
+ tlcnConIndex = (tlcnNextContainer << ZSHIFT_PLUS) - (tlcnNextContainer << ZSHIFT_MINUS);
+ tlcnConIndex = tlcnConIndex + ((ZHEAD_SIZE + ZBUF_SIZE) - ZCON_HEAD_SIZE);
+ tlupIndex = tlcnConIndex;
+ tlupElemIndex = tlcnConIndex - 1;
+ lcpUpdatePage(signal);
+ tlcnNextContainer = (lcnCopyPageptr.p->word32[tlcnConIndex] >> 11) & 0x7f;
+ }//while
+ if (tlcnNextContainer == ZEMPTYLIST) {
+ jam();
+ /*empty*/;
+ } else {
+ jam();
+ sendSystemerror(signal);
+ return;
+ }//if
+ }//if
+ lcnCopyPageptr.p->word32[ZPOS_CHECKSUM] = tlcnChecksum;
+}//Dbacc::lcpCopyPage()
+
+/* --------------------------------------------------------------------------------- */
+/* THIS SUBROUTINE GOES THROUGH ONE CONTAINER TO CHECK FOR LOCKED ELEMENTS AND */
+/* UPDATING THEM TO ENSURE ALL ELEMENTS ARE UNLOCKED ON DISK. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::lcpUpdatePage(Signal* signal)
+{
+ OperationrecPtr lupOperationRecPtr;
+ Uint32 tlupElemHead;
+ Uint32 tlupElemLen;
+ Uint32 tlupElemStep;
+ Uint32 tlupConLen;
+
+ tlupConLen = lupPageptr.p->word32[tlupIndex] >> 26;
+ tlupElemLen = fragrecptr.p->elementLength;
+ tlupElemStep = tlupForward * tlupElemLen;
+ while (tlupConLen > ZCON_HEAD_SIZE) {
+ jam();
+ tlupElemHead = lupPageptr.p->word32[tlupElemIndex];
+ if (ElementHeader::getLocked(tlupElemHead)) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* WHEN CHANGING THE ELEMENT HEADER WE ALSO HAVE TO UPDATE THE CHECKSUM. IN */
+ /* DOING THIS WE USE THE FORMULA (A XOR B) XOR B = A WHICH MEANS THAT IF WE */
+ /* XOR SOMETHING TWICE WITH THE SAME OPERAND THEN WE RETURN TO THE ORIGINAL */
+ /* VALUE. THEN WE ALSO HAVE TO USE THE NEW ELEMENT HEADER IN THE CHECKSUM */
+ /* CALCULATION. */
+ /* --------------------------------------------------------------------------------- */
+ tlcnChecksum = tlcnChecksum ^ tlupElemHead;
+ lupOperationRecPtr.i = ElementHeader::getOpPtrI(tlupElemHead);
+ ptrCheckGuard(lupOperationRecPtr, coprecsize, operationrec);
+ const Uint32 hv = lupOperationRecPtr.p->hashvaluePart;
+ tlupElemHead = ElementHeader::setUnlocked(hv , 0);
+ arrGuard(tlupElemIndex, 2048);
+ lupPageptr.p->word32[tlupElemIndex] = tlupElemHead;
+ tlcnChecksum = tlcnChecksum ^ tlupElemHead;
+ }//if
+ tlupConLen = tlupConLen - tlupElemLen;
+ tlupElemIndex = tlupElemIndex + tlupElemStep;
+ }//while
+ if (tlupConLen < ZCON_HEAD_SIZE) {
+ jam();
+ sendSystemerror(signal);
+ }//if
+}//Dbacc::lcpUpdatePage()
+
+/*-----------------------------------------------------------------*/
+// At a system restart we check that the page do not contain any
+// locks that hinder the system restart procedure.
+/*-----------------------------------------------------------------*/
+void Dbacc::srCheckPage(Signal* signal)
+{
+ Uint32 tlcnNextContainer;
+ Uint32 tlcnConIndex;
+ Uint32 tlcnIndex;
+
+ lupPageptr.p = lcnCopyPageptr.p;
+ if (((lcnCopyPageptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) != ZLONG_PAGE_TYPE) {
+ jam();
+ if (((lcnCopyPageptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) == ZNORMAL_PAGE_TYPE) {
+ jam();
+ /*-----------------------------------------------------------------*/
+ /* TAKE CARE OF ALL 64 BUFFERS ADDRESSED BY ALGORITHM IN */
+ /* FIRST PAGE. IF THEY ARE EMPTY THEY STILL HAVE A CONTAINER */
+ /* HEADER OF 2 WORDS. */
+ /*-----------------------------------------------------------------*/
+ tlcnConIndex = ZHEAD_SIZE;
+ tlupForward = 1;
+ for (tlcnIndex = 0; tlcnIndex <= ZNO_CONTAINERS - 1; tlcnIndex++) {
+ tlupIndex = tlcnConIndex;
+ tlupElemIndex = tlcnConIndex + ZCON_HEAD_SIZE;
+ srCheckContainer(signal);
+ if (tresult != 0) {
+ jam();
+ return;
+ }//if
+ tlcnConIndex = tlcnConIndex + ZBUF_SIZE;
+ }//for
+ }//if
+ /*-----------------------------------------------------------------*/
+ /* TAKE CARE OF ALL USED BUFFERS ON THE LEFT SIDE. */
+ /*-----------------------------------------------------------------*/
+ tlcnNextContainer = (lcnCopyPageptr.p->word32[ZPOS_EMPTY_LIST] >> 23) & 0x7f;
+ while (tlcnNextContainer < ZEMPTYLIST) {
+ tlcnConIndex = (tlcnNextContainer << ZSHIFT_PLUS) - (tlcnNextContainer << ZSHIFT_MINUS);
+ tlcnConIndex = tlcnConIndex + ZHEAD_SIZE;
+ tlupIndex = tlcnConIndex;
+ tlupElemIndex = tlcnConIndex + ZCON_HEAD_SIZE;
+ tlupForward = 1;
+ srCheckContainer(signal);
+ if (tresult != 0) {
+ jam();
+ return;
+ }//if
+ tlcnNextContainer = (lcnCopyPageptr.p->word32[tlcnConIndex] >> 11) & 0x7f;
+ }//while
+ if (tlcnNextContainer == ZEMPTYLIST) {
+ jam();
+ /*empty*/;
+ } else {
+ jam();
+ tresult = 4;
+ return;
+ }//if
+ /*-----------------------------------------------------------------*/
+ /* TAKE CARE OF ALL USED BUFFERS ON THE RIGHT SIDE. */
+ /*-----------------------------------------------------------------*/
+ tlupForward = cminusOne;
+ tlcnNextContainer = (lcnCopyPageptr.p->word32[ZPOS_EMPTY_LIST] >> 16) & 0x7f;
+ while (tlcnNextContainer < ZEMPTYLIST) {
+ tlcnConIndex = (tlcnNextContainer << ZSHIFT_PLUS) - (tlcnNextContainer << ZSHIFT_MINUS);
+ tlcnConIndex = tlcnConIndex + ((ZHEAD_SIZE + ZBUF_SIZE) - ZCON_HEAD_SIZE);
+ tlupIndex = tlcnConIndex;
+ tlupElemIndex = tlcnConIndex - 1;
+ srCheckContainer(signal);
+ if (tresult != 0) {
+ jam();
+ return;
+ }//if
+ tlcnNextContainer = (lcnCopyPageptr.p->word32[tlcnConIndex] >> 11) & 0x7f;
+ }//while
+ if (tlcnNextContainer == ZEMPTYLIST) {
+ jam();
+ /*empty*/;
+ } else {
+ jam();
+ tresult = 4;
+ return;
+ }//if
+ }//if
+}//Dbacc::srCheckPage()
+
+/* --------------------------------------------------------------------------------- */
+/* THIS SUBROUTINE GOES THROUGH ONE CONTAINER TO CHECK FOR LOCKED ELEMENTS AND */
+/* UPDATING THEM TO ENSURE ALL ELEMENTS ARE UNLOCKED ON DISK. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::srCheckContainer(Signal* signal)
+{
+ Uint32 tlupElemLen;
+ Uint32 tlupElemStep;
+ Uint32 tlupConLen;
+
+ tlupConLen = lupPageptr.p->word32[tlupIndex] >> 26;
+ tlupElemLen = fragrecptr.p->elementLength;
+ tlupElemStep = tlupForward * tlupElemLen;
+ while (tlupConLen > ZCON_HEAD_SIZE) {
+ jam();
+ const Uint32 tlupElemHead = lupPageptr.p->word32[tlupElemIndex];
+ if (ElementHeader::getLocked(tlupElemHead)){
+ jam();
+ //-------------------------------------------------------
+ // This is absolutely undesirable. We have a lock remaining
+ // after the system restart. We send a crash signal that will
+ // enter the trace file.
+ //-------------------------------------------------------
+ tresult = 2;
+ return;
+ }//if
+ tlupConLen = tlupConLen - tlupElemLen;
+ tlupElemIndex = tlupElemIndex + tlupElemStep;
+ }//while
+ if (tlupConLen < ZCON_HEAD_SIZE) {
+ jam();
+ tresult = 3;
+ }//if
+ return;
+}//Dbacc::srCheckContainer()
+
+/* ------------------------------------------------------------------------- */
+/* CHECK_UNDO_PAGES */
+/* DESCRIPTION: CHECKS WHEN A PAGE OR A GROUP OF UNDO PAGES IS FILLED.WHEN */
+/* A PAGE IS FILLED, CUNDOPOSITION WILL BE UPDATE, THE NEW */
+/* POSITION IS THE BEGNING OF THE NEXT UNDO PAGE. */
+/* IN CASE THAT A GROUP IS FILLED THE PAGES ARE SENT TO DISK, */
+/* AND A NEW GROUP IS CHOSEN. */
+/* ------------------------------------------------------------------------- */
+void Dbacc::checkUndoPages(Signal* signal)
+{
+
+ fragrecptr.p->prevUndoposition = cundoposition;
+ cprevUndoaddress = cundoposition;
+
+ // Calculate active undo page id
+ Uint32 tundoPageId = cundoposition >> ZUNDOPAGEINDEXBITS;
+
+ /**
+ * WE WILL WRITE UNTIL WE HAVE ABOUT 8 KBYTE REMAINING ON THE 32 KBYTE
+ * PAGE. THIS IS TO ENSURE THAT WE DO NOT HAVE ANY UNDO LOG RECORDS THAT PASS
+ * A PAGE BOUNDARIE. THIS SIMPLIFIES CODING TRADING SOME INEFFICIENCY.
+ */
+ static const Uint32 ZMAXUNDOPAGEINDEX = 7100;
+ if (tundoindex < ZMAXUNDOPAGEINDEX) {
+ jam();
+ cundoposition = (tundoPageId << ZUNDOPAGEINDEXBITS) + tundoindex;
+ return;
+ }//if
+
+ /**
+ * WE CHECK IF MORE THAN 1 MBYTE OF WRITES ARE OUTSTANDING TO THE UNDO FILE.
+ * IF SO WE HAVE TO CRASH SINCE WE HAVE NO MORE SPACE TO WRITE UNDO LOG
+ * RECORDS IN
+ */
+ Uint16 nextUndoPageId = tundoPageId + 1;
+ if (nextUndoPageId > (clastUndoPageIdWritten + cundopagesize)){
+ // No more undolog, crash node
+ progError(__LINE__,
+ ERR_NO_MORE_UNDOLOG,
+ "There are more than 1Mbyte undolog writes outstanding");
+ }
+ updateUndoPositionPage(signal, nextUndoPageId << ZUNDOPAGEINDEXBITS);
+
+ if ((tundoPageId & (ZWRITE_UNDOPAGESIZE - 1)) == (ZWRITE_UNDOPAGESIZE - 1)) {
+ jam();
+ /* ---------- SEND A GROUP OF UNDO PAGES TO DISK --------- */
+ fsConnectptr.i = cactiveOpenUndoFsPtr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ Uint32 tcupTmp1 = (tundoPageId - ZWRITE_UNDOPAGESIZE) + 1;
+ tcupTmp1 = tcupTmp1 & (cundopagesize - 1); /* 1 MBYTE PAGE WINDOW */
+ seizeFsOpRec(signal);
+ initFsOpRec(signal);
+ fsOpptr.p->fsOpstate = WAIT_WRITE_UNDO_EXIT;
+ fsOpptr.p->fsOpMemPage = tundoPageId;
+ fragrecptr.p->nrWaitWriteUndoExit++;
+ if (clblPageCounter >= 8) {
+ jam();
+ clblPageCounter = clblPageCounter - 8;
+ } else {
+ jam();
+ clblPageOver = clblPageOver + (8 - clblPageCounter);
+ clblPageCounter = 0;
+ }//if
+ /* ************************ */
+ /* FSWRITEREQ */
+ /* ************************ */
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsOpptr.i;
+ signal->theData[3] = 0x1;
+ /* FLAG = START MEM PAGES, START FILE PAGES */
+ signal->theData[4] = ZUNDOPAGE_BASE_ADD;
+ signal->theData[5] = ZWRITE_UNDOPAGESIZE;
+ signal->theData[6] = tcupTmp1;
+ signal->theData[7] = cactiveUndoFilePage;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+ cactiveUndoFilePage = cactiveUndoFilePage + ZWRITE_UNDOPAGESIZE;
+ }//if
+}//Dbacc::checkUndoPages()
+
+/* --------------------------------------------------------------------------------- */
+/* UNDO_WRITING_PROCESS */
+/* INPUT: FRAGRECPTR, CUNDO_ELEM_INDEX, DATAPAGEPTR, CUNDOINFOLENGTH */
+/* DESCRIPTION: WHEN THE PROCESS OF CREATION LOCAL CHECK POINT HAS */
+/* STARTED. IF THE ACTIVE PAGE IS NOT ALREADY SENT TO DISK, THE */
+/* OLD VALUE OF THE ITEM WHICH IS GOING TO BE CHECKED IS STORED ON */
+/* THE ACTIVE UNDO PAGE. INFORMATION ABOUT UNDO PROCESS IN THE */
+/* BLOCK AND IN THE FRAGMENT WILL BE UPDATED. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::undoWritingProcess(Signal* signal)
+{
+ const Uint32 tactivePageDir = datapageptr.p->word32[ZPOS_PAGE_ID];
+ const Uint32 tpageType = (datapageptr.p->word32[ZPOS_EMPTY_LIST] >> ZPOS_PAGE_TYPE_BIT) & 3;
+ if (fragrecptr.p->fragState == LCP_SEND_PAGES) {
+ if (tpageType == ZNORMAL_PAGE_TYPE) {
+ /* --------------------------------------------------------------------------- */
+ /* HANDLING OF LOG OF NORMAL PAGES DURING WRITE OF NORMAL PAGES. */
+ /* --------------------------------------------------------------------------- */
+ if (tactivePageDir < fragrecptr.p->lcpDirIndex) {
+ jam();
+ /* ------------------------------------------------------------------- */
+ /* THIS PAGE HAS ALREADY BEEN WRITTEN IN THE LOCAL CHECKPOINT. */
+ /* ------------------------------------------------------------------- */
+ /*empty*/;
+ } else {
+ if (tactivePageDir >= fragrecptr.p->lcpMaxDirIndex) {
+ jam();
+ /* --------------------------------------------------------------------------- */
+ /* OBVIOUSLY THE FRAGMENT HAS EXPANDED SINCE THE START OF THE LOCAL CHECKPOINT.*/
+ /* WE NEED NOT LOG ANY UPDATES OF PAGES THAT DID NOT EXIST AT START OF LCP. */
+ /* --------------------------------------------------------------------------- */
+ /*empty*/;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------------- */
+ /* IN ALL OTHER CASES WE HAVE TO WRITE TO THE UNDO LOG. */
+ /* --------------------------------------------------------------------------- */
+ undopageptr.i = (cundoposition >> ZUNDOPAGEINDEXBITS) & (cundopagesize - 1);
+ ptrAss(undopageptr, undopage);
+ theadundoindex = cundoposition & ZUNDOPAGEINDEX_MASK;
+ tundoindex = theadundoindex + ZUNDOHEADSIZE;
+ writeUndoHeader(signal, tactivePageDir, UndoHeader::ZPAGE_INFO);
+ tundoElemIndex = cundoElemIndex;
+ writeUndoDataInfo(signal);
+ checkUndoPages(signal);
+ }//if
+ }//if
+ } else if (tpageType == ZOVERFLOW_PAGE_TYPE) {
+ /* --------------------------------------------------------------------------------- */
+ /* OVERFLOW PAGE HANDLING DURING WRITE OF NORMAL PAGES. */
+ /* --------------------------------------------------------------------------------- */
+ if (tactivePageDir >= fragrecptr.p->lcpMaxOverDirIndex) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* OBVIOUSLY THE FRAGMENT HAS EXPANDED THE NUMBER OF OVERFLOW PAGES SINCE THE */
+ /* START OF THE LOCAL CHECKPOINT. WE NEED NOT LOG ANY UPDATES OF PAGES THAT DID*/
+ /* NOT EXIST AT START OF LCP. */
+ /* --------------------------------------------------------------------------------- */
+ /*empty*/;
+ } else {
+ jam();
+ undopageptr.i = (cundoposition >> ZUNDOPAGEINDEXBITS) & (cundopagesize - 1);
+ ptrAss(undopageptr, undopage);
+ theadundoindex = cundoposition & ZUNDOPAGEINDEX_MASK;
+ tundoindex = theadundoindex + ZUNDOHEADSIZE;
+ writeUndoHeader(signal, tactivePageDir, UndoHeader::ZOVER_PAGE_INFO);
+ tundoElemIndex = cundoElemIndex;
+ writeUndoDataInfo(signal);
+ checkUndoPages(signal);
+ }//if
+ } else if (tpageType != ZLONG_PAGE_TYPE) {
+ jam();
+ /* --------------------------------------------------------------------------- */
+ /* ONLY PAGE INFO AND OVERFLOW PAGE INFO CAN BE LOGGED BY THIS ROUTINE. A */
+ /* SERIOUS ERROR. */
+ /* --------------------------------------------------------------------------- */
+ sendSystemerror(signal);
+ } else {
+ /* --------------------------------------------------------------------------------- */
+ /* THIS LOG RECORD IS GENERATED ON A LONG KEY PAGE. THESE PAGES USE LOGICAL */
+ /* LOGGING. */
+ /* --------------------------------------------------------------------------------- */
+ if (tactivePageDir >= fragrecptr.p->lcpMaxOverDirIndex) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* OBVIOUSLY THE FRAGMENT HAS EXPANDED THE NUMBER OF OVERFLOW PAGES SINCE THE */
+ /* START OF THE LOCAL CHECKPOINT. WE NEED NOT LOG ANY UPDATES OF PAGES THAT DID*/
+ /* NOT EXIST AT START OF LCP. */
+ /* --------------------------------------------------------------------------------- */
+ /*empty*/;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* LOGICAL LOGGING OF LONG KEY PAGES CAN EITHER BE UNDO OF AN INSERT OR UNDO */
+ /* OF A DELETE KEY. UNDO OF DELETE NEEDS TO LOG THE KEY TO BE REINSERTED WHILE */
+ /* UNDO OF INSERT ONLY NEEDS TO LOG THE INDEX TO BE DELETED. */
+ /* --------------------------------------------------------------------------------- */
+ undopageptr.i = (cundoposition >> ZUNDOPAGEINDEXBITS) & (cundopagesize - 1);
+ ptrAss(undopageptr, undopage);
+ theadundoindex = cundoposition & ZUNDOPAGEINDEX_MASK;
+ tundoindex = theadundoindex + ZUNDOHEADSIZE;
+ if (cundoinfolength == 0) {
+ jam();
+ writeUndoHeader(signal, tactivePageDir, UndoHeader::ZUNDO_INSERT_LONG_KEY);
+ } else {
+ jam();
+ writeUndoHeader(signal, tactivePageDir, UndoHeader::ZUNDO_DELETE_LONG_KEY);
+ arrGuard(ZWORDS_IN_PAGE - cundoElemIndex, 2048);
+ tundoElemIndex = datapageptr.p->word32[ZWORDS_IN_PAGE - cundoElemIndex] & 0xffff;
+ writeUndoDataInfo(signal);
+ }//if
+ checkUndoPages(signal);
+ }//if
+ }//if
+ } else {
+ if (fragrecptr.p->fragState == LCP_SEND_OVER_PAGES) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* DURING WRITE OF OVERFLOW PAGES WE NEED NOT WORRY ANYMORE ABOUT NORMAL PAGES.*/
+ /* --------------------------------------------------------------------------------- */
+ if (tpageType == ZOVERFLOW_PAGE_TYPE) {
+ if (tactivePageDir < fragrecptr.p->lcpDirIndex) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THIS PAGE HAS ALREADY BEEN WRITTEN IN THE LOCAL CHECKPOINT. */
+ /* --------------------------------------------------------------------------------- */
+ /*empty*/;
+ } else {
+ if (tactivePageDir >= fragrecptr.p->lcpMaxOverDirIndex) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* OBVIOUSLY THE FRAGMENT HAS EXPANDED THE NUMBER OF OVERFLOW PAGES SINCE THE */
+ /* START OF THE LOCAL CHECKPOINT. WE NEED NOT LOG ANY UPDATES OF PAGES THAT DID*/
+ /* NOT EXIST AT START OF LCP. */
+ /* --------------------------------------------------------------------------------- */
+ /*empty*/;
+ } else {
+ jam();
+ undopageptr.i = (cundoposition >> ZUNDOPAGEINDEXBITS) & (cundopagesize - 1);
+ ptrAss(undopageptr, undopage);
+ theadundoindex = cundoposition & ZUNDOPAGEINDEX_MASK;
+ tundoindex = theadundoindex + ZUNDOHEADSIZE;
+ writeUndoHeader(signal, tactivePageDir, UndoHeader::ZOVER_PAGE_INFO);
+ tundoElemIndex = cundoElemIndex;
+ writeUndoDataInfo(signal);
+ checkUndoPages(signal);
+ }//if
+ }//if
+ } else if (tpageType == ZLONG_PAGE_TYPE) {
+ if (tactivePageDir < fragrecptr.p->lcpDirIndex) {
+ jam();
+ // -------------------------------------------------------------
+ // THIS PAGE HAS ALREADY BEEN WRITTEN IN THE LOCAL CHECKPOINT.
+ // -------------------------------------------------------------
+ } else {
+ if (tactivePageDir >= fragrecptr.p->lcpMaxOverDirIndex) {
+ jam();
+ // -------------------------------------------------------------
+ // OBVIOUSLY THE FRAGMENT HAS EXPANDED THE NUMBER OF OVERFLOW
+ // PAGES SINCE THE START OF THE LOCAL CHECKPOINT. WE NEED NOT
+ // LOG ANY UPDATES OF PAGES THAT DID NOT EXIST AT START OF LCP.
+ // -------------------------------------------------------------
+ } else {
+ jam();
+ // -------------------------------------------------------------
+ // LOGICAL LOGGING OF LONG KEY PAGES CAN EITHER BE UNDO OF AN
+ // INSERT OR UNDO OF A DELETE KEY. UNDO OF DELETE NEEDS TO LOG
+ // THE KEY TO BE REINSERTED WHILE UNDO OF INSERT ONLY NEEDS TO
+ // LOG THE INDEX TO BE DELETED.
+ // -------------------------------------------------------------
+ undopageptr.i = (cundoposition >> ZUNDOPAGEINDEXBITS) & (cundopagesize - 1);
+ ptrAss(undopageptr, undopage);
+ theadundoindex = cundoposition & ZUNDOPAGEINDEX_MASK;
+ tundoindex = theadundoindex + ZUNDOHEADSIZE;
+ if (cundoinfolength == 0) {
+ jam();
+ writeUndoHeader(signal, tactivePageDir, UndoHeader::ZUNDO_INSERT_LONG_KEY);
+ } else {
+ jam();
+ writeUndoHeader(signal, tactivePageDir, UndoHeader::ZUNDO_DELETE_LONG_KEY);
+ arrGuard(ZWORDS_IN_PAGE - cundoElemIndex, 2048);
+ tundoElemIndex = datapageptr.p->word32[ZWORDS_IN_PAGE - cundoElemIndex] & 0xffff;
+ writeUndoDataInfo(signal);
+ }//if
+ checkUndoPages(signal);
+ }//if
+ }//if
+ }//if
+ }//if
+ }//if
+}//Dbacc::undoWritingProcess()
+
+/* --------------------------------------------------------------------------------- */
+/* OTHER STATES MEANS THAT WE HAVE ALREADY WRITTEN ALL PAGES BUT NOT YET RESET */
+/* THE CREATE_LCP FLAG. */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* WRITE_UNDO_DATA_INFO */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::writeUndoDataInfo(Signal* signal)
+{
+ Uint32 twudiIndex;
+ Uint32 guard22;
+
+ guard22 = cundoinfolength;
+ arrGuard((tundoindex + guard22 - 1), 8192);
+ arrGuard((tundoElemIndex + guard22 - 1), 2048);
+ for (twudiIndex = 1; twudiIndex <= guard22; twudiIndex++) {
+ undopageptr.p->undoword[tundoindex] = datapageptr.p->word32[tundoElemIndex];
+ tundoindex++;
+ tundoElemIndex++;
+ }//for
+}//Dbacc::writeUndoDataInfo()
+
+/* --------------------------------------------------------------------------------- */
+/* WRITE_UNDO_HEADER */
+/* THE HEAD OF UNDO ELEMENT IS 24 BYTES AND CONTAINS THE FOLLOWING INFORMATION: */
+/* TABLE IDENTITY 32 BITS */
+/* ROOT FRAGMENT IDENTITY 32 BITS */
+/* LOCAL FRAGMENT IDENTITY 32 BITS */
+/* LENGTH OF ELEMENT INF0 (BIT 31 - 18) 14 BITS */
+/* INFO TYPE (BIT 17 - 14) 4 BITS */
+/* PAGE INDEX OF THE FIRST FIELD IN THE FRAGMENT (BIT 13 - 0) 14 BITS */
+/* DIRECTORY INDEX OF THE PAGE IN THE FRAGMENT 32 BITS */
+/* ADDRESS OF THE PREVIOUS ELEMENT OF THE FRAGMENT 64 BITS */
+/* ADDRESS OF THE PREVIOUS ELEMENT IN THE UNDO PAGES 64 BITS */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::writeUndoHeader(Signal* signal,
+ Uint32 logicalPageId,
+ UndoHeader::UndoHeaderType pageType)
+{
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ arrGuard(theadundoindex + 6, 8192);
+
+ // Set the structpointer to point at the undo page at the right address.
+ UndoHeader * const & undoHeaderPtr =
+ (UndoHeader *) &undopageptr.p->undoword[theadundoindex];
+
+ undoHeaderPtr->tableId = rootfragrecptr.p->mytabptr;
+ undoHeaderPtr->rootFragId = rootfragrecptr.p->fragmentid[0];
+ undoHeaderPtr->localFragId = fragrecptr.p->myfid;
+ Uint32 Ttmp = cundoinfolength;
+ Ttmp = (Ttmp << 4) + pageType;
+ Ttmp = Ttmp << 14;
+ undoHeaderPtr->variousInfo = Ttmp + cundoElemIndex;
+ undoHeaderPtr->logicalPageId = logicalPageId;
+ undoHeaderPtr->prevUndoAddressForThisFrag = fragrecptr.p->prevUndoposition;
+ undoHeaderPtr->prevUndoAddress = cprevUndoaddress;
+}//Dbacc::writeUndoHeader()
+
+/* --------------------------------------------------------------------------------- */
+/* WRITE_UNDO_OP_INFO */
+/* FOR A LOCKED ELEMENT, OPERATION TYPE, UNDO OF ELEMENT HEADER AND THE LENGTH OF*/
+/* THE TUPLE KEY HAVE TO BE SAVED IN UNDO PAGES. IN THIS CASE AN UNDO ELEMENT */
+/* INCLUDES THE FLLOWING ITEMS. */
+/* OPERATION TYPE 32 BITS */
+/* HASH VALUE 32 BITS */
+/* LENGTH OF THE TUPLE = N 32 BITS */
+/* TUPLE KEYS N * 32 BITS */
+/* */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::writeUndoOpInfo(Signal* signal)
+{
+ Page8Ptr locPageptr;
+
+ arrGuard((tundoindex + 3), 8192);
+ undopageptr.p->undoword[tundoindex] = operationRecPtr.p->operation;
+ undopageptr.p->undoword[tundoindex + 1] = operationRecPtr.p->hashValue;
+ undopageptr.p->undoword[tundoindex + 2] = operationRecPtr.p->tupkeylen;
+ tundoindex = tundoindex + 3;
+ if (fragrecptr.p->keyLength != 0) {
+ // Fixed size keys
+ jam();
+ locPageptr.i = operationRecPtr.p->elementPage;
+ ptrCheckGuard(locPageptr, cpagesize, page8);
+ Uint32 Tforward = operationRecPtr.p->elementIsforward;
+ Uint32 TelemPtr = operationRecPtr.p->elementPointer;
+ TelemPtr += Tforward;
+ TelemPtr += Tforward;
+ //---------------------------------------------------------------------------------
+ // Now the pointer is at the start of the key part of the element. Now copy from there
+ // to the UNDO log.
+ //---------------------------------------------------------------------------------
+ Uint32 keyLen = operationRecPtr.p->tupkeylen;
+ ndbrequire(keyLen <= 8);
+ arrGuard(tundoindex+keyLen, 8192);
+ for (Uint32 twuoiIndex = 0; twuoiIndex < keyLen; twuoiIndex++) {
+ jam();
+ arrGuard(TelemPtr, 2048);
+ undopageptr.p->undoword[tundoindex] = locPageptr.p->word32[TelemPtr];
+ tundoindex++;
+ TelemPtr += Tforward;
+ }//for
+ cundoinfolength = ZOP_HEAD_INFO_LN + operationRecPtr.p->tupkeylen;
+ } else {
+ // Long keys
+ jam();
+
+ arrGuard(operationRecPtr.p->longKeyPageIndex, ZMAX_NO_OF_LONGKEYS_IN_PAGE);
+ locPageptr.i = operationRecPtr.p->longPagePtr;
+ ptrCheckGuard(locPageptr, cpagesize, page8);
+
+ Uint32 indexValue =
+ locPageptr.p->word32[ZWORDS_IN_PAGE - operationRecPtr.p->longKeyPageIndex];
+ Uint32 keyLen = indexValue >> 16;
+ Uint32 physPageIndex = indexValue & 0xffff;
+ ndbrequire(keyLen == operationRecPtr.p->tupkeylen);
+
+ arrGuard(tundoindex+keyLen, 8192);
+ arrGuard(physPageIndex+keyLen, 2048);
+ for (Uint32 i = 0; i < keyLen; i++){
+ undopageptr.p->undoword[tundoindex + i] = locPageptr.p->word32[physPageIndex+i];
+ }
+ tundoindex = tundoindex + keyLen;
+ cundoinfolength = ZOP_HEAD_INFO_LN + keyLen;
+ }//if
+}//Dbacc::writeUndoOpInfo()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF LOCAL CHECKPOINT MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* SYSTEM RESTART MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* SR_FRAGIDREQ REQUEST FOR RESTART OF A FRAGMENT */
+/* SENDER: LQH, LEVEL B */
+/* ENTER SR_FRAGIDREQ WITH */
+/* TUSERPTR, LQH CONNECTION PTR */
+/* TUSERBLOCKREF, LQH BLOCK REFERENCE */
+/* TCHECKPOINTID, THE CHECKPOINT NUMBER TO USE */
+/* (E.G. 1,2 OR 3) */
+/* TABPTR, TABLE ID = TABLE RECORD POINTER */
+/* TFID, ROOT FRAGMENT ID */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* SR_FRAGIDREQ REQUEST FOR LIST OF STOPED OPERATION */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execSR_FRAGIDREQ(Signal* signal)
+{
+ jamEntry();
+ tuserptr = signal->theData[0]; /* LQH CONNECTION PTR */
+ tuserblockref = signal->theData[1]; /* LQH BLOCK REFERENCE */
+ tcheckpointid = signal->theData[2]; /* THE CHECKPOINT NUMBER TO USE */
+ /* (E.G. 1,2 OR 3) */
+ tabptr.i = signal->theData[3];
+ ptrCheckGuard(tabptr, ctablesize, tabrec);
+ /* TABLE ID = TABLE RECORD POINTER */
+ tfid = signal->theData[4]; /* ROOT FRAGMENT ID */
+ tresult = 0; /* 0= FALSE,1= TRUE,> ZLIMIT_OF_ERROR =ERRORCODE */
+ seizeLcpConnectRec(signal);
+ initLcpConnRec(signal);
+
+ ndbrequire(getrootfragmentrec(signal, rootfragrecptr, tfid));
+ rootfragrecptr.p->lcpPtr = lcpConnectptr.i;
+ lcpConnectptr.p->rootrecptr = rootfragrecptr.i;
+ lcpConnectptr.p->localCheckPid = tcheckpointid;
+ for (Uint32 i = 0; i < 2; i++) {
+ Page8Ptr zeroPagePtr;
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[i];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ seizeLcpPage(zeroPagePtr);
+ fragrecptr.p->zeroPagePtr = zeroPagePtr.i;
+ }//for
+
+ /* ---------------------------OPEN THE DATA FILE WHICH BELONGS TO TFID AND TCHECK POINT ---- */
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[0];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ tfid = rootfragrecptr.p->fragmentid[0];
+ tmp = 0;
+ srOpenDataFileLoopLab(signal);
+
+ return;
+}//Dbacc::execSR_FRAGIDREQ()
+
+void Dbacc::srOpenDataFileLoopLab(Signal* signal)
+{
+ /* D6 AT FSOPENREQ. FILE TYPE = .DATA */
+ tmp1 = 0x010003ff; /* VERSION OF FILENAME = 1 */
+ tmp2 = 0x0; /* D7 DON'T CREATE, READ ONLY */
+ ndbrequire(cfsFirstfreeconnect != RNIL);
+ seizeFsConnectRec(signal);
+
+ fragrecptr.p->fsConnPtr = fsConnectptr.i;
+ fsConnectptr.p->fragrecPtr = fragrecptr.i;
+ fsConnectptr.p->fsState = WAIT_OPEN_DATA_FILE_FOR_READ;
+ fsConnectptr.p->activeFragId = tmp; /* LOCAL FRAG INDEX */
+ /* ************************ */
+ /* FSOPENREQ */
+ /* ************************ */
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = fsConnectptr.i;
+ signal->theData[2] = rootfragrecptr.p->mytabptr; /* TABLE IDENTITY */
+ signal->theData[3] = tfid; /* FRAGMENT IDENTITY */
+ signal->theData[4] = lcpConnectptr.p->localCheckPid; /* CHECKPOINT ID */
+ signal->theData[5] = tmp1;
+ signal->theData[6] = tmp2;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+ return;
+}//Dbacc::srOpenDataFileLoopLab()
+
+void Dbacc::srFsOpenConfLab(Signal* signal)
+{
+ fsConnectptr.p->fsState = WAIT_READ_PAGE_ZERO;
+ /* ------------------------ READ ZERO PAGE ---------- */
+ fragrecptr.i = fsConnectptr.p->fragrecPtr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsConnectptr.i;
+ signal->theData[3] = 0x0;
+ /* FLAG = LIST MEM PAGES, LIST FILE PAGES */
+ signal->theData[4] = ZPAGE8_BASE_ADD;
+ signal->theData[5] = 1; /* NO OF PAGES */
+ signal->theData[6] = fragrecptr.p->zeroPagePtr; /* ZERO PAGE */
+ signal->theData[7] = 0; /* PAGE ZERO OF THE DATA FILE */
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 8, JBA);
+ return;
+}//Dbacc::srFsOpenConfLab()
+
+void Dbacc::srReadPageZeroLab(Signal* signal)
+{
+ Page8Ptr srzPageptr;
+
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ fragrecptr.p->activeDataFilePage = 1;
+ srzPageptr.i = fragrecptr.p->zeroPagePtr;
+ ptrCheckGuard(srzPageptr, cpagesize, page8);
+ /* --------------------------------------------------------------------------------- */
+ // Check that the checksum of the zero page is ok.
+ /* --------------------------------------------------------------------------------- */
+ ccoPageptr.p = srzPageptr.p;
+ checksumControl(signal, (Uint32)0);
+ if (tresult > 0) {
+ jam();
+ return; // We will crash through a DEBUG_SIG
+ }//if
+
+ ndbrequire(srzPageptr.p->word32[ZPAGEZERO_FRAGID0] == rootfragrecptr.p->fragmentid[0]);
+ lcpConnectptr.i = rootfragrecptr.p->lcpPtr;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ if (fsConnectptr.p->activeFragId == 0) {
+ jam();
+ rootfragrecptr.p->fragmentid[1] = srzPageptr.p->word32[ZPAGEZERO_FRAGID1];
+ /* ---------------------------OPEN THE DATA FILE FOR NEXT LOCAL FRAGMENT ----------- ---- */
+ tfid = rootfragrecptr.p->fragmentid[1];
+ tmp = 1; /* LOCAL FRAG INDEX */
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[1];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ srOpenDataFileLoopLab(signal);
+ return;
+ } else {
+ jam();
+ lcpConnectptr.p->lcpstate = LCP_ACTIVE;
+ signal->theData[0] = lcpConnectptr.p->lcpUserptr;
+ signal->theData[1] = lcpConnectptr.i;
+ signal->theData[2] = 2; /* NO OF LOCAL FRAGMENTS */
+ signal->theData[3] = srzPageptr.p->word32[ZPAGEZERO_FRAGID0];
+ /* ROOTFRAGRECPTR:FRAGMENTID(0) */
+ signal->theData[4] = srzPageptr.p->word32[ZPAGEZERO_FRAGID1];
+ /* ROOTFRAGRECPTR:FRAGMENTID(1) */
+ signal->theData[5] = RNIL;
+ signal->theData[6] = RNIL;
+ signal->theData[7] = rootfragrecptr.p->fragmentptr[0];
+ signal->theData[8] = rootfragrecptr.p->fragmentptr[1];
+ signal->theData[9] = srzPageptr.p->word32[ZPAGEZERO_HASH_CHECK];
+ sendSignal(lcpConnectptr.p->lcpUserblockref, GSN_SR_FRAGIDCONF, signal, 10, JBB);
+ }//if
+ return;
+}//Dbacc::srReadPageZeroLab()
+
+void Dbacc::initFragAdd(Signal* signal,
+ Uint32 rootFragIndex,
+ Uint32 rootIndex,
+ FragmentrecPtr regFragPtr)
+{
+ const AccFragReq * const req = (AccFragReq*)&signal->theData[0];
+ Uint32 lhFragBits = req->lhFragBits + 1;
+ Uint32 minLoadFactor = (req->minLoadFactor * ZBUF_SIZE) / 100;
+ Uint32 maxLoadFactor = (req->maxLoadFactor * ZBUF_SIZE) / 100;
+ if (minLoadFactor >= maxLoadFactor) {
+ jam();
+ minLoadFactor = maxLoadFactor - 1;
+ }//if
+ regFragPtr.p->fragState = ACTIVEFRAG;
+ // NOTE: next line must match calculation in Dblqh::execLQHFRAGREQ
+ regFragPtr.p->myfid = (rootFragIndex << (lhFragBits - 1)) | req->fragId;
+ regFragPtr.p->myroot = rootIndex;
+ regFragPtr.p->myTableId = req->tableId;
+ ndbrequire(req->kValue == 6);
+ regFragPtr.p->k = req->kValue; /* TK_SIZE = 6 IN THIS VERSION */
+ regFragPtr.p->expandCounter = 0;
+ regFragPtr.p->expandFlag = 0;
+ regFragPtr.p->p = 0;
+ regFragPtr.p->maxp = (1 << req->kValue) - 1;
+ regFragPtr.p->minloadfactor = minLoadFactor;
+ regFragPtr.p->maxloadfactor = maxLoadFactor;
+ regFragPtr.p->slack = (regFragPtr.p->maxp + 1) * maxLoadFactor;
+ regFragPtr.p->lhfragbits = lhFragBits;
+ regFragPtr.p->lhdirbits = 0;
+ regFragPtr.p->hashcheckbit = 0; //lhFragBits;
+ regFragPtr.p->localkeylen = req->localKeyLen;
+ regFragPtr.p->nodetype = (req->reqInfo >> 4) & 0x3;
+ regFragPtr.p->lastOverIndex = 0;
+ regFragPtr.p->dirsize = 1;
+ regFragPtr.p->loadingFlag = ZFALSE;
+ regFragPtr.p->keyLength = req->keyLength;
+ if (req->keyLength == 0) {
+ jam();
+ regFragPtr.p->elementLength = (1 + ZELEM_HEAD_SIZE) + regFragPtr.p->localkeylen;
+ } else {
+ jam();
+ regFragPtr.p->elementLength = (ZELEM_HEAD_SIZE + regFragPtr.p->localkeylen) + regFragPtr.p->keyLength;
+ }//if
+ Uint32 Tmp1 = (regFragPtr.p->maxp + 1) + regFragPtr.p->p;
+ Uint32 Tmp2 = regFragPtr.p->maxloadfactor - regFragPtr.p->minloadfactor;
+ Tmp2 = Tmp1 * Tmp2;
+ regFragPtr.p->slackCheck = Tmp2;
+}//Dbacc::initFragAdd()
+
+void Dbacc::initFragGeneral(FragmentrecPtr regFragPtr)
+{
+ regFragPtr.p->directory = RNIL;
+ regFragPtr.p->overflowdir = RNIL;
+ regFragPtr.p->fsConnPtr = RNIL;
+ regFragPtr.p->firstOverflowRec = RNIL;
+ regFragPtr.p->lastOverflowRec = RNIL;
+ regFragPtr.p->firstWaitInQueOp = RNIL;
+ regFragPtr.p->lastWaitInQueOp = RNIL;
+ regFragPtr.p->sentWaitInQueOp = RNIL;
+ regFragPtr.p->lockOwnersList = RNIL;
+ regFragPtr.p->firstFreeDirindexRec = RNIL;
+ regFragPtr.p->zeroPagePtr = RNIL;
+
+ regFragPtr.p->activeDataPage = 0;
+ regFragPtr.p->createLcp = ZFALSE;
+ regFragPtr.p->stopQueOp = ZFALSE;
+ regFragPtr.p->nextAllocPage = 0;
+ regFragPtr.p->nrWaitWriteUndoExit = 0;
+ regFragPtr.p->lastUndoIsStored = ZFALSE;
+ regFragPtr.p->loadingFlag = ZFALSE;
+ regFragPtr.p->fragState = FREEFRAG;
+ for (Uint32 i = 0; i < ZWRITEPAGESIZE; i++) {
+ regFragPtr.p->datapages[i] = RNIL;
+ }//for
+ for (Uint32 i = 0; i < 4; i++) {
+ regFragPtr.p->longKeyPageArray[i] = RNIL;
+ }//for
+}//Dbacc::initFragGeneral()
+
+void Dbacc::initFragSr(FragmentrecPtr regFragPtr, Page8Ptr regPagePtr)
+{
+ regFragPtr.p->prevUndoposition = regPagePtr.p->word32[ZPAGEZERO_PREV_UNDOP];
+ regFragPtr.p->noOfStoredOverPages = regPagePtr.p->word32[ZPAGEZERO_NO_OVER_PAGE];
+ regFragPtr.p->noStoredPages = regPagePtr.p->word32[ZPAGEZERO_NO_PAGES];
+ regFragPtr.p->dirsize = regPagePtr.p->word32[ZPAGEZERO_DIRSIZE];
+ regFragPtr.p->expandCounter = regPagePtr.p->word32[ZPAGEZERO_EXPCOUNTER];
+ regFragPtr.p->slack = regPagePtr.p->word32[ZPAGEZERO_SLACK];
+ regFragPtr.p->hashcheckbit = regPagePtr.p->word32[ZPAGEZERO_HASHCHECKBIT];
+ regFragPtr.p->k = regPagePtr.p->word32[ZPAGEZERO_K];
+ regFragPtr.p->lhfragbits = regPagePtr.p->word32[ZPAGEZERO_LHFRAGBITS];
+ regFragPtr.p->lhdirbits = regPagePtr.p->word32[ZPAGEZERO_LHDIRBITS];
+ regFragPtr.p->localkeylen = regPagePtr.p->word32[ZPAGEZERO_LOCALKEYLEN];
+ regFragPtr.p->maxp = regPagePtr.p->word32[ZPAGEZERO_MAXP];
+ regFragPtr.p->maxloadfactor = regPagePtr.p->word32[ZPAGEZERO_MAXLOADFACTOR];
+ regFragPtr.p->minloadfactor = regPagePtr.p->word32[ZPAGEZERO_MINLOADFACTOR];
+ regFragPtr.p->myfid = regPagePtr.p->word32[ZPAGEZERO_MYFID];
+ regFragPtr.p->lastOverIndex = regPagePtr.p->word32[ZPAGEZERO_LAST_OVER_INDEX];
+ regFragPtr.p->nodetype = regPagePtr.p->word32[ZPAGEZERO_NODETYPE];
+ regFragPtr.p->p = regPagePtr.p->word32[ZPAGEZERO_P];
+ regFragPtr.p->elementLength = regPagePtr.p->word32[ZPAGEZERO_ELEMENT_LENGTH];
+ regFragPtr.p->keyLength = regPagePtr.p->word32[ZPAGEZERO_KEY_LENGTH];
+ regFragPtr.p->slackCheck = regPagePtr.p->word32[ZPAGEZERO_SLACK_CHECK];
+
+ regFragPtr.p->loadingFlag = ZTRUE;
+
+}//Dbacc::initFragSr()
+
+void Dbacc::initFragPageZero(FragmentrecPtr regFragPtr, Page8Ptr regPagePtr)
+{
+ //------------------------------------------------------------------
+ // PREV_UNDOP, NEXT_UNDO_FILE, NO_OVER_PAGE, NO_PAGES
+ // is set at end of copy phase
+ //------------------------------------------------------------------
+ regPagePtr.p->word32[ZPAGEZERO_DIRSIZE] = regFragPtr.p->dirsize;
+ regPagePtr.p->word32[ZPAGEZERO_EXPCOUNTER] = regFragPtr.p->expandCounter;
+ regPagePtr.p->word32[ZPAGEZERO_SLACK] = regFragPtr.p->slack;
+ regPagePtr.p->word32[ZPAGEZERO_HASHCHECKBIT] = regFragPtr.p->hashcheckbit;
+ regPagePtr.p->word32[ZPAGEZERO_K] = regFragPtr.p->k;
+ regPagePtr.p->word32[ZPAGEZERO_LHFRAGBITS] = regFragPtr.p->lhfragbits;
+ regPagePtr.p->word32[ZPAGEZERO_LHDIRBITS] = regFragPtr.p->lhdirbits;
+ regPagePtr.p->word32[ZPAGEZERO_LOCALKEYLEN] = regFragPtr.p->localkeylen;
+ regPagePtr.p->word32[ZPAGEZERO_MAXP] = regFragPtr.p->maxp;
+ regPagePtr.p->word32[ZPAGEZERO_MAXLOADFACTOR] = regFragPtr.p->maxloadfactor;
+ regPagePtr.p->word32[ZPAGEZERO_MINLOADFACTOR] = regFragPtr.p->minloadfactor;
+ regPagePtr.p->word32[ZPAGEZERO_MYFID] = regFragPtr.p->myfid;
+ regPagePtr.p->word32[ZPAGEZERO_LAST_OVER_INDEX] = regFragPtr.p->lastOverIndex;
+ regPagePtr.p->word32[ZPAGEZERO_NODETYPE] = regFragPtr.p->nodetype;
+ regPagePtr.p->word32[ZPAGEZERO_P] = regFragPtr.p->p;
+ regPagePtr.p->word32[ZPAGEZERO_ELEMENT_LENGTH] = regFragPtr.p->elementLength;
+ regPagePtr.p->word32[ZPAGEZERO_KEY_LENGTH] = regFragPtr.p->keyLength;
+ regPagePtr.p->word32[ZPAGEZERO_SLACK_CHECK] = regFragPtr.p->slackCheck;
+}//Dbacc::initFragPageZero()
+
+void Dbacc::initRootFragPageZero(RootfragmentrecPtr rootPtr, Page8Ptr regPagePtr)
+{
+ regPagePtr.p->word32[ZPAGEZERO_TABID] = rootPtr.p->mytabptr;
+ regPagePtr.p->word32[ZPAGEZERO_FRAGID0] = rootPtr.p->fragmentid[0];
+ regPagePtr.p->word32[ZPAGEZERO_FRAGID1] = rootPtr.p->fragmentid[1];
+ regPagePtr.p->word32[ZPAGEZERO_HASH_CHECK] = rootPtr.p->roothashcheck;
+ regPagePtr.p->word32[ZPAGEZERO_NO_OF_ELEMENTS] = rootPtr.p->noOfElements;
+}//Dbacc::initRootFragPageZero()
+
+void Dbacc::initRootFragSr(RootfragmentrecPtr rootPtr, Page8Ptr regPagePtr)
+{
+ rootPtr.p->roothashcheck = regPagePtr.p->word32[ZPAGEZERO_HASH_CHECK];
+ rootPtr.p->noOfElements = regPagePtr.p->word32[ZPAGEZERO_NO_OF_ELEMENTS];
+}//Dbacc::initRootFragSr()
+
+/* ******************--------------------------------------------------------------- */
+/* ACC_SRREQ SYSTEM RESTART OF A LOCAL CHECK POINT */
+/* SENDER: LQH, LEVEL B */
+/* ENTER ACC_SRREQ WITH */
+/* LCP_CONNECTPTR, OPERATION RECORD PTR */
+/* TMP2, LQH'S LOCAL FRAG CHECK VALUE */
+/* TFID, LOCAL FRAG ID */
+/* TMP1, LOCAL CHECKPOINT ID */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* ACC_SRREQ PERFORM A LOCAL CHECK POINT */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execACC_SRREQ(Signal* signal)
+{
+ Page8Ptr asrPageidptr;
+ jamEntry();
+ lcpConnectptr.i = signal->theData[0];
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ Uint32 lqhPtr = signal->theData[1];
+ Uint32 fragId = signal->theData[2];
+ Uint32 lcpId = signal->theData[3];
+ tresult = 0;
+ ndbrequire(lcpConnectptr.p->lcpstate == LCP_ACTIVE);
+ rootfragrecptr.i = lcpConnectptr.p->rootrecptr;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ if (rootfragrecptr.p->fragmentid[0] == fragId) {
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[0];
+ } else {
+ ndbrequire(rootfragrecptr.p->fragmentid[1] == fragId);
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[1];
+ }//if
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ fragrecptr.p->lcpLqhPtr = lqhPtr;
+ fragrecptr.p->localCheckpId = lcpId;
+ asrPageidptr.i = fragrecptr.p->zeroPagePtr;
+ ptrCheckGuard(asrPageidptr, cpagesize, page8);
+ ndbrequire(asrPageidptr.p->word32[ZPAGEZERO_TABID] == rootfragrecptr.p->mytabptr);
+ ndbrequire(asrPageidptr.p->word32[ZPAGEZERO_FRAGID0] == rootfragrecptr.p->fragmentid[0]);
+ ndbrequire(asrPageidptr.p->word32[ZPAGEZERO_FRAGID1] == rootfragrecptr.p->fragmentid[1]);
+ initRootFragSr(rootfragrecptr, asrPageidptr);
+ initFragSr(fragrecptr, asrPageidptr);
+ for (Uint32 i = 0; i < ZMAX_UNDO_VERSION; i++) {
+ jam();
+ if (csrVersList[i] != RNIL) {
+ jam();
+ srVersionPtr.i = csrVersList[i];
+ ptrCheckGuard(srVersionPtr, csrVersionRecSize, srVersionRec);
+ if (fragrecptr.p->localCheckpId == srVersionPtr.p->checkPointId) {
+ jam();
+ ndbrequire(srVersionPtr.p->checkPointId == asrPageidptr.p->word32[ZPAGEZERO_NEXT_UNDO_FILE]);
+ /*--------------------------------------------------------------------------------*/
+ /* SINCE -1 IS THE END OF LOG CODE WE MUST TREAT THIS CODE WITH CARE. WHEN */
+ /* COMPARING IT IS LARGER THAN EVERYTHING ELSE BUT SHOULD BE TREATED AS THE */
+ /* SMALLEST POSSIBLE VALUE, MEANING EMPTY. */
+ /*--------------------------------------------------------------------------------*/
+ if (fragrecptr.p->prevUndoposition != cminusOne) {
+ if (srVersionPtr.p->prevAddress < fragrecptr.p->prevUndoposition) {
+ jam();
+ srVersionPtr.p->prevAddress = fragrecptr.p->prevUndoposition;
+ } else if (srVersionPtr.p->prevAddress == cminusOne) {
+ jam();
+ srVersionPtr.p->prevAddress = fragrecptr.p->prevUndoposition;
+ }//if
+ }//if
+ srAllocPage0011Lab(signal);
+ return;
+ }//if
+ } else {
+ jam();
+ seizeSrVerRec(signal);
+ srVersionPtr.p->checkPointId = fragrecptr.p->localCheckpId;
+ srVersionPtr.p->prevAddress = fragrecptr.p->prevUndoposition;
+ csrVersList[i] = srVersionPtr.i;
+ srAllocPage0011Lab(signal);
+ return;
+ }//if
+ }//for
+ ndbrequire(false);
+}//Dbacc::execACC_SRREQ()
+
+void
+Dbacc::releaseLogicalPage(Fragmentrec * fragP, Uint32 logicalPageId){
+ Ptr<struct DirRange> dirRangePtr;
+ dirRangePtr.i = fragP->directory;
+ ptrCheckGuard(dirRangePtr, cdirrangesize, dirRange);
+
+ const Uint32 lp1 = logicalPageId >> 8;
+ const Uint32 lp2 = logicalPageId & 0xFF;
+ ndbrequire(lp1 < 256);
+
+ Ptr<struct Directoryarray> dirArrPtr;
+ dirArrPtr.i = dirRangePtr.p->dirArray[lp1];
+ ptrCheckGuard(dirArrPtr, cdirarraysize, directoryarray);
+
+ const Uint32 physicalPageId = dirArrPtr.p->pagep[lp2];
+
+ rpPageptr.i = physicalPageId;
+ ptrCheckGuard(rpPageptr, cpagesize, page8);
+ releasePage(0);
+
+ dirArrPtr.p->pagep[lp2] = RNIL;
+}
+
+void Dbacc::srAllocPage0011Lab(Signal* signal)
+{
+ releaseLogicalPage(fragrecptr.p, 0);
+
+#if JONAS
+ ndbrequire(cfirstfreeDirrange != RNIL);
+ seizeDirrange(signal);
+ fragrecptr.p->directory = newDirRangePtr.i;
+ ndbrequire(cfirstfreeDirrange != RNIL);
+ seizeDirrange(signal);
+ fragrecptr.p->overflowdir = newDirRangePtr.i;
+ seizeDirectory(signal);
+ ndbrequire(tresult < ZLIMIT_OF_ERROR);
+ newDirRangePtr.p->dirArray[0] = sdDirptr.i;
+#endif
+
+ fragrecptr.p->nextAllocPage = 0;
+ fragrecptr.p->fragState = SR_READ_PAGES;
+ srReadPagesLab(signal);
+ return;
+}//Dbacc::srAllocPage0011Lab()
+
+void Dbacc::srReadPagesLab(Signal* signal)
+{
+ if (fragrecptr.p->nextAllocPage >= fragrecptr.p->noStoredPages) {
+ /*--------------------------------------------------------------------------------*/
+ /* WE HAVE NOW READ ALL NORMAL PAGES FROM THE FILE. */
+ /*--------------------------------------------------------------------------------*/
+ if (fragrecptr.p->nextAllocPage == fragrecptr.p->dirsize) {
+ jam();
+ /*--------------------------------------------------------------------------------*/
+ /* WE HAVE NOW READ ALL NORMAL PAGES AND ALLOCATED ALL THE NEEDED PAGES. */
+ /*--------------------------------------------------------------------------------*/
+ fragrecptr.p->nextAllocPage = 0; /* THE NEXT OVER FLOW PAGE WHICH WILL BE READ */
+ fragrecptr.p->fragState = SR_READ_OVER_PAGES;
+ srReadOverPagesLab(signal);
+ } else {
+ ndbrequire(fragrecptr.p->nextAllocPage < fragrecptr.p->dirsize);
+ jam();
+ /*--------------------------------------------------------------------------------*/
+ /* WE NEEDED TO ALLOCATE PAGES THAT WERE DEALLOCATED DURING THE LOCAL */
+ /* CHECKPOINT. */
+ /* ALLOCATE THE PAGE AND INITIALISE IT. THEN WE INSERT A REAL-TIME BREAK. */
+ /*--------------------------------------------------------------------------------*/
+ seizePage(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ tipPageId = fragrecptr.p->nextAllocPage;
+ inpPageptr.i = spPageptr.i;
+ ptrCheckGuard(inpPageptr, cpagesize, page8);
+ initPage(signal);
+ fragrecptr.p->noOfExpectedPages = 1;
+ fragrecptr.p->datapages[0] = spPageptr.i;
+ signal->theData[0] = ZSR_READ_PAGES_ALLOC;
+ signal->theData[1] = fragrecptr.i;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+ return;
+ }//if
+ Uint32 limitLoop;
+ if ((fragrecptr.p->noStoredPages - fragrecptr.p->nextAllocPage) < ZWRITEPAGESIZE) {
+ jam();
+ limitLoop = fragrecptr.p->noStoredPages - fragrecptr.p->nextAllocPage;
+ } else {
+ jam();
+ limitLoop = ZWRITEPAGESIZE;
+ }//if
+ ndbrequire(limitLoop <= 8);
+ for (Uint32 i = 0; i < limitLoop; i++) {
+ jam();
+ seizePage(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ fragrecptr.p->datapages[i] = spPageptr.i;
+ signal->theData[i + 6] = spPageptr.i;
+ }//for
+ signal->theData[limitLoop + 6] = fragrecptr.p->activeDataFilePage;
+ fragrecptr.p->noOfExpectedPages = limitLoop;
+ /* -----------------SEND READ PAGES SIGNAL TO THE FILE MANAGER --------- */
+ fsConnectptr.i = fragrecptr.p->fsConnPtr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ fsConnectptr.p->fsState = WAIT_READ_DATA;
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsConnectptr.i;
+ signal->theData[3] = 2;
+ /* FLAG = LIST MEM PAGES, RANGE OF FILE PAGES */
+ signal->theData[4] = ZPAGE8_BASE_ADD;
+ signal->theData[5] = fragrecptr.p->noOfExpectedPages;
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 15, JBA);
+ return;
+}//Dbacc::srReadPagesLab()
+
+void Dbacc::storeDataPageInDirectoryLab(Signal* signal)
+{
+ fragrecptr.p->activeDataFilePage += fragrecptr.p->noOfExpectedPages;
+ srReadPagesAllocLab(signal);
+ return;
+}//Dbacc::storeDataPageInDirectoryLab()
+
+void Dbacc::srReadPagesAllocLab(Signal* signal)
+{
+ DirRangePtr srpDirRangePtr;
+ DirectoryarrayPtr srpDirptr;
+ DirectoryarrayPtr srpOverflowDirptr;
+ Page8Ptr srpPageidptr;
+
+ if (fragrecptr.p->fragState == SR_READ_PAGES) {
+ jam();
+ for (Uint32 i = 0; i < fragrecptr.p->noOfExpectedPages; i++) {
+ jam();
+ tmpP = fragrecptr.p->nextAllocPage;
+ srpDirRangePtr.i = fragrecptr.p->directory;
+ tmpP2 = tmpP >> 8;
+ tmp = tmpP & 0xff;
+ ptrCheckGuard(srpDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(tmpP2, 256);
+ if (srpDirRangePtr.p->dirArray[tmpP2] == RNIL) {
+ seizeDirectory(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ srpDirptr.i = sdDirptr.i;
+ srpDirRangePtr.p->dirArray[tmpP2] = srpDirptr.i;
+ } else {
+ jam();
+ srpDirptr.i = srpDirRangePtr.p->dirArray[tmpP2];
+ }//if
+ ptrCheckGuard(srpDirptr, cdirarraysize, directoryarray);
+ arrGuard(i, 8);
+ srpDirptr.p->pagep[tmp] = fragrecptr.p->datapages[i];
+ srpPageidptr.i = fragrecptr.p->datapages[i];
+ ptrCheckGuard(srpPageidptr, cpagesize, page8);
+ ndbrequire(srpPageidptr.p->word32[ZPOS_PAGE_ID] == fragrecptr.p->nextAllocPage);
+ ndbrequire(((srpPageidptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) == 0);
+ ccoPageptr.p = srpPageidptr.p;
+ checksumControl(signal, (Uint32)1);
+ if (tresult > 0) {
+ jam();
+ return; // We will crash through a DEBUG_SIG
+ }//if
+ dbgWord32(srpPageidptr, ZPOS_OVERFLOWREC, RNIL);
+ srpPageidptr.p->word32[ZPOS_OVERFLOWREC] = RNIL;
+ fragrecptr.p->datapages[i] = RNIL;
+ fragrecptr.p->nextAllocPage++;
+ }//for
+ srReadPagesLab(signal);
+ return;
+ } else {
+ ndbrequire(fragrecptr.p->fragState == SR_READ_OVER_PAGES);
+ for (Uint32 i = 0; i < fragrecptr.p->noOfExpectedPages; i++) {
+ jam();
+ arrGuard(i, 8);
+ srpPageidptr.i = fragrecptr.p->datapages[i];
+ ptrCheckGuard(srpPageidptr, cpagesize, page8);
+ tmpP = srpPageidptr.p->word32[ZPOS_PAGE_ID]; /* DIR INDEX OF THE OVERFLOW PAGE */
+ /*--------------------------------------------------------------------------------*/
+ /* IT IS POSSIBLE THAT WE HAVE LOGICAL PAGES WHICH ARE NOT PART OF THE LOCAL*/
+ /* CHECKPOINT. THUS WE USE THE LOGICAL PAGE ID FROM THE PAGE HERE. */
+ /*--------------------------------------------------------------------------------*/
+ srpDirRangePtr.i = fragrecptr.p->overflowdir;
+ tmpP2 = tmpP >> 8;
+ tmpP = tmpP & 0xff;
+ ptrCheckGuard(srpDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(tmpP2, 256);
+ if (srpDirRangePtr.p->dirArray[tmpP2] == RNIL) {
+ jam();
+ seizeDirectory(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ srpDirRangePtr.p->dirArray[tmpP2] = sdDirptr.i;
+ }//if
+ srpOverflowDirptr.i = srpDirRangePtr.p->dirArray[tmpP2];
+ ndbrequire(((srpPageidptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) != 0);
+ ndbrequire(((srpPageidptr.p->word32[ZPOS_PAGE_TYPE] >> ZPOS_PAGE_TYPE_BIT) & 3) != 3);
+ ptrCheckGuard(srpOverflowDirptr, cdirarraysize, directoryarray);
+ ndbrequire(srpOverflowDirptr.p->pagep[tmpP] == RNIL);
+ srpOverflowDirptr.p->pagep[tmpP] = srpPageidptr.i;
+ ccoPageptr.p = srpPageidptr.p;
+ checksumControl(signal, (Uint32)1);
+ ndbrequire(tresult == 0);
+ dbgWord32(srpPageidptr, ZPOS_OVERFLOWREC, RNIL);
+ srpPageidptr.p->word32[ZPOS_OVERFLOWREC] = RNIL;
+ fragrecptr.p->nextAllocPage++;
+ }//for
+ srReadOverPagesLab(signal);
+ return;
+ }//if
+}//Dbacc::srReadPagesAllocLab()
+
+void Dbacc::srReadOverPagesLab(Signal* signal)
+{
+ if (fragrecptr.p->nextAllocPage >= fragrecptr.p->noOfStoredOverPages) {
+ fragrecptr.p->nextAllocPage = 0;
+ if (fragrecptr.p->prevUndoposition == cminusOne) {
+ jam();
+ /* ************************ */
+ /* ACC_OVER_REC */
+ /* ************************ */
+ /*--------------------------------------------------------------------------------*/
+ /* UPDATE FREE LIST OF OVERFLOW PAGES AS PART OF SYSTEM RESTART AFTER */
+ /* READING PAGES AND EXECUTING THE UNDO LOG. */
+ /*--------------------------------------------------------------------------------*/
+ signal->theData[0] = fragrecptr.i;
+ sendSignal(cownBlockref, GSN_ACC_OVER_REC, signal, 1, JBB);
+ } else {
+ jam();
+ srCloseDataFileLab(signal);
+ }//if
+ return;
+ }//if
+ Uint32 limitLoop;
+ if ((fragrecptr.p->noOfStoredOverPages - fragrecptr.p->nextAllocPage) < ZWRITEPAGESIZE) {
+ jam();
+ limitLoop = fragrecptr.p->noOfStoredOverPages - fragrecptr.p->nextAllocPage;
+ } else {
+ jam();
+ limitLoop = ZWRITEPAGESIZE;
+ }//if
+ ndbrequire(limitLoop <= 8);
+ for (Uint32 i = 0; i < limitLoop; i++) {
+ jam();
+ seizePage(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+ fragrecptr.p->datapages[i] = spPageptr.i;
+ signal->theData[i + 6] = spPageptr.i;
+ }//for
+ fragrecptr.p->noOfExpectedPages = limitLoop;
+ signal->theData[limitLoop + 6] = fragrecptr.p->activeDataFilePage;
+ /* -----------------SEND READ PAGES SIGNAL TO THE FILE MANAGER --------- */
+ fsConnectptr.i = fragrecptr.p->fsConnPtr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ fsConnectptr.p->fsState = WAIT_READ_DATA;
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsConnectptr.i;
+ signal->theData[3] = 2;
+ signal->theData[4] = ZPAGE8_BASE_ADD;
+ signal->theData[5] = fragrecptr.p->noOfExpectedPages;
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 15, JBA);
+ return;
+}//Dbacc::srReadOverPagesLab()
+
+void Dbacc::srCloseDataFileLab(Signal* signal)
+{
+ fsConnectptr.i = fragrecptr.p->fsConnPtr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ fsConnectptr.p->fsState = SR_CLOSE_DATA;
+ /* ************************ */
+ /* FSCLOSEREQ */
+ /* ************************ */
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsConnectptr.i;
+ signal->theData[3] = 0;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+ return;
+}//Dbacc::srCloseDataFileLab()
+
+/* ************************ */
+/* ACC_SRCONF */
+/* ************************ */
+void Dbacc::sendaccSrconfLab(Signal* signal)
+{
+ fragrecptr.i = fsConnectptr.p->fragrecPtr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ releaseFsConnRec(signal);
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ lcpConnectptr.i = rootfragrecptr.p->lcpPtr;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ fragrecptr.p->fragState = ACTIVEFRAG;
+ fragrecptr.p->fsConnPtr = RNIL;
+ for (Uint32 i = 0; i < ZWRITEPAGESIZE; i++) {
+ fragrecptr.p->datapages[i] = RNIL;
+ }//for
+ rlpPageptr.i = fragrecptr.p->zeroPagePtr;
+ ptrCheckGuard(rlpPageptr, cpagesize, page8);
+ releaseLcpPage(signal);
+ fragrecptr.p->zeroPagePtr = RNIL;
+ signal->theData[0] = fragrecptr.p->lcpLqhPtr;
+ sendSignal(lcpConnectptr.p->lcpUserblockref, GSN_ACC_SRCONF, signal, 1, JBB);
+ lcpConnectptr.p->noOfLcpConf++;
+ if (lcpConnectptr.p->noOfLcpConf == 2) {
+ jam();
+ releaseLcpConnectRec(signal);
+ rootfragrecptr.p->lcpPtr = RNIL;
+ rootfragrecptr.p->rootState = ACTIVEROOT;
+ }//if
+ return;
+}//Dbacc::sendaccSrconfLab()
+
+/* --------------------------------------------------------------------------------- */
+/* CHECKSUM_CONTROL */
+/* INPUT: CCO_PAGEPTR */
+/* OUTPUT: TRESULT */
+/* */
+/* CHECK THAT CHECKSUM IN PAGE IS CORRECT TO ENSURE THAT NO ONE HAS CORRUPTED */
+/* THE PAGE INFORMATION. WHEN CALCULATING THE CHECKSUM WE REMOVE THE CHECKSUM */
+/* ITSELF FROM THE CHECKSUM BY XOR'ING THE CHECKSUM TWICE. WHEN CALCULATING */
+/* THE CHECKSUM THE CHECKSUM WORD IS ZERO WHICH MEANS NO CHANGE FROM XOR'ING. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::checksumControl(Signal* signal, Uint32 checkPage)
+{
+ Uint32 Tchs;
+ Uint32 tccoIndex;
+ Uint32 Ti;
+ Uint32 Tmp1;
+ Uint32 Tmp2;
+ Uint32 Tmp3;
+ Uint32 Tmp4;
+ Uint32 Tlimit;
+
+ Tchs = 0;
+ for (Ti = 0; Ti < 32 ; Ti++) {
+ Tlimit = 16 + (Ti << 6);
+ for (tccoIndex = (Ti << 6); tccoIndex < Tlimit; tccoIndex ++) {
+ Tmp1 = ccoPageptr.p->word32[tccoIndex];
+ Tmp2 = ccoPageptr.p->word32[tccoIndex + 16];
+ Tmp3 = ccoPageptr.p->word32[tccoIndex + 32];
+ Tmp4 = ccoPageptr.p->word32[tccoIndex + 48];
+
+ Tchs = Tchs ^ Tmp1;
+ Tchs = Tchs ^ Tmp2;
+ Tchs = Tchs ^ Tmp3;
+ Tchs = Tchs ^ Tmp4;
+ }//for
+ }//for
+ if (Tchs == 0) {
+ tresult = 0;
+ if (checkPage != 0) {
+ jam();
+ lcnCopyPageptr.p = ccoPageptr.p;
+ srCheckPage(signal);
+ }//if
+ } else {
+ tresult = 1;
+ }//if
+ if (tresult != 0) {
+ jam();
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ signal->theData[0] = RNIL;
+ signal->theData[1] = rootfragrecptr.p->mytabptr;
+ signal->theData[2] = fragrecptr.p->myfid;
+ signal->theData[3] = ccoPageptr.p->word32[ZPOS_PAGE_ID];
+ signal->theData[4] = tlupElemIndex;
+ signal->theData[5] = ccoPageptr.p->word32[ZPOS_PAGE_TYPE];
+ signal->theData[6] = tresult;
+ sendSignal(cownBlockref, GSN_DEBUG_SIG, signal, 7, JBA);
+ }//if
+}//Dbacc::checksumControl()
+
+/* ******************--------------------------------------------------------------- */
+/* START_RECREQ REQUEST TO START UNDO PROCESS */
+/* SENDER: LQH, LEVEL B */
+/* ENTER START_RECREQ WITH */
+/* CLQH_PTR, LQH CONNECTION PTR */
+/* CLQH_BLOCK_REF, LQH BLOCK REFERENCE */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* START_RECREQ REQUEST TO START UNDO PROCESS */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execSTART_RECREQ(Signal* signal)
+{
+ jamEntry();
+ clqhPtr = signal->theData[0]; /* LQH CONNECTION PTR */
+ clqhBlockRef = signal->theData[1]; /* LQH BLOCK REFERENCE */
+ tresult = 0; /* 0= FALSE,1= TRUE,> ZLIMIT_OF_ERROR =ERRORCODE */
+ for (int i = 0; i < UndoHeader::ZNO_UNDORECORD_TYPES; i++)
+ cSrUndoRecords[i] = 0;
+ startUndoLab(signal);
+ return;
+}//Dbacc::execSTART_RECREQ()
+
+void Dbacc::startUndoLab(Signal* signal)
+{
+ cundoLogActive = ZTRUE;
+ /* ----- OPEN UNDO FILES --------- */
+ for (tmp = 0; tmp <= ZMAX_UNDO_VERSION - 1; tmp++) {
+ jam();
+ if (csrVersList[tmp] != RNIL) {
+ jam();
+ /*---------------------------------------------------------------------------*/
+ /* SELECT THE NEXT SYSTEM RESTART RECORD WHICH CONTAINS AN UNDO LOG */
+ /* THAT NEEDS TO BE EXECUTED AND SET UP THE DATA TO EXECUTE IT. */
+ /*---------------------------------------------------------------------------*/
+ srVersionPtr.i = csrVersList[tmp];
+ csrVersList[tmp] = RNIL;
+ ptrCheckGuard(srVersionPtr, csrVersionRecSize, srVersionRec);
+ cactiveUndoFilePage = srVersionPtr.p->prevAddress >> 13;
+ cprevUndoaddress = srVersionPtr.p->prevAddress;
+ cactiveCheckpId = srVersionPtr.p->checkPointId;
+
+ releaseSrRec(signal);
+ startActiveUndo(signal);
+ return;
+ }//if
+ }//for
+
+ // Send report of how many undo log records where executed
+ signal->theData[0] = EventReport::UNDORecordsExecuted;
+ signal->theData[1] = DBACC; // From block
+ signal->theData[2] = 0; // Total records executed
+ for (int i = 0; i < 10; i++){
+ if (i < UndoHeader::ZNO_UNDORECORD_TYPES){
+ signal->theData[i+3] = cSrUndoRecords[i];
+ signal->theData[2] += cSrUndoRecords[i];
+ }else{
+ signal->theData[i+3] = 0;
+ }
+ }
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 12, JBB);
+
+ /* ******************************< */
+ /* START_RECCONF */
+ /* ******************************< */
+ /*---------------------------------------------------------------------------*/
+ /* REPORT COMPLETION OF UNDO LOG EXECUTION. */
+ /*---------------------------------------------------------------------------*/
+ cundoLogActive = ZFALSE;
+ signal->theData[0] = clqhPtr;
+ sendSignal(clqhBlockRef, GSN_START_RECCONF, signal, 1, JBB);
+ /* LQH CONNECTION PTR */
+ return;
+}//Dbacc::startUndoLab()
+
+/*---------------------------------------------------------------------------*/
+/* START THE UNDO OF AN UNDO LOG FILE BY OPENING THE UNDO LOG FILE. */
+/*---------------------------------------------------------------------------*/
+void Dbacc::startActiveUndo(Signal* signal)
+{
+ if (cprevUndoaddress == cminusOne) {
+ jam();
+ /*---------------------------------------------------------------------------*/
+ /* THERE WAS NO UNDO LOG INFORMATION IN THIS LOG FILE. WE GET THE NEXT */
+ /* OR REPORT COMPLETION. */
+ /*---------------------------------------------------------------------------*/
+ signal->theData[0] = ZSTART_UNDO;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 1, JBB);
+ } else {
+ jam();
+ /*---------------------------------------------------------------------------*/
+ /* OPEN THE LOG FILE PERTAINING TO THIS UNDO LOG. */
+ /*---------------------------------------------------------------------------*/
+ if (cfsFirstfreeconnect == RNIL) {
+ jam();
+ sendSystemerror(signal);
+ }//if
+ seizeFsConnectRec(signal);
+ cactiveSrFsPtr = fsConnectptr.i;
+ fsConnectptr.p->fsState = OPEN_UNDO_FILE_SR;
+ fsConnectptr.p->fsPart = 0;
+ tmp1 = 1; /* FILE VERSION ? */
+ tmp1 = (tmp1 << 8) + ZLOCALLOGFILE; /* .LOCLOG = 2 */
+ tmp1 = (tmp1 << 8) + 4; /* ROOT DIRECTORY = D4 */
+ tmp1 = (tmp1 << 8) + fsConnectptr.p->fsPart; /* P2 */
+ tmp2 = 0x0; /* D7 DON'T CREATE , READ ONLY */
+ /* DON'T TRUNCATE TO ZERO */
+ /* ---FILE NAME "D4"/"DBACC"/LCP_CONNECTPTR:LOCAL_CHECK_PID/FS_CONNECTPTR:FS_PART".LOCLOG-- */
+ /* ************************ */
+ /* FSOPENREQ */
+ /* ************************ */
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = fsConnectptr.i;
+ signal->theData[2] = cminusOne; /* #FFFFFFFF */
+ signal->theData[3] = cminusOne; /* #FFFFFFFF */
+ signal->theData[4] = cactiveCheckpId; /* CHECKPOINT VERSION */
+ signal->theData[5] = tmp1;
+ signal->theData[6] = tmp2;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+ }//if
+}//Dbacc::startActiveUndo()
+
+/* ------- READ A GROUP OF UNDO PAGES --------------- */
+void Dbacc::srStartUndoLab(Signal* signal)
+{
+ /*---------------------------------------------------------------------------*/
+ /* ALL LOG FILES HAVE BEEN OPENED. WE CAN NOW READ DATA FROM THE LAST */
+ /* PAGE IN THE LAST LOG FILE AND BACKWARDS UNTIL WE REACH THE VERY */
+ /* FIRST UNDO LOG RECORD. */
+ /*---------------------------------------------------------------------------*/
+ if (cactiveUndoFilePage >= ZWRITE_UNDOPAGESIZE) {
+ jam();
+ tmp1 = ZWRITE_UNDOPAGESIZE; /* NO OF READ UNDO PAGES */
+ cactiveSrUndoPage = ZWRITE_UNDOPAGESIZE - 1; /* LAST PAGE */
+ } else {
+ jam();
+ tmp1 = cactiveUndoFilePage + 1; /* NO OF READ UNDO PAGES */
+ cactiveSrUndoPage = cactiveUndoFilePage;
+ }//if
+ fsConnectptr.i = cactiveSrFsPtr;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ signal->theData[0] = fsConnectptr.p->fsPtr;
+ signal->theData[1] = cownBlockref;
+ signal->theData[2] = fsConnectptr.i;
+ signal->theData[3] = 0;
+ /* FLAG = LIST MEM PAGES, LIST FILE PAGES */
+ signal->theData[4] = ZUNDOPAGE_BASE_ADD;
+ signal->theData[5] = tmp1;
+ signal->theData[6] = 0;
+ signal->theData[7] = (cactiveUndoFilePage - tmp1) + 1;
+ signal->theData[8] = 1;
+ signal->theData[9] = cactiveUndoFilePage;
+
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 10, JBA);
+ if (tmp1 > cactiveUndoFilePage) {
+ jam();
+ /*---------------------------------------------------------------------------*/
+ /* THIS IS THE LAST READ IN THIS LOG FILE. WE SET THE ACTIVE FILE */
+ /* POINTER. IF IT IS THE FIRST WE SHOULD NEVER ATTEMPT ANY MORE READS */
+ /* SINCE WE SHOULD ENCOUNTER A FIRST LOG RECORD WITH PREVIOUS PAGE ID */
+ /* EQUAL TO RNIL. */
+ /*---------------------------------------------------------------------------*/
+ cactiveSrFsPtr = RNIL;
+ fsConnectptr.p->fsState = READ_UNDO_PAGE_AND_CLOSE;
+ } else {
+ jam();
+ /*---------------------------------------------------------------------------*/
+ /* WE STILL HAVE MORE INFORMATION IN THIS LOG FILE. WE ONLY MOVE BACK */
+ /* THE FILE PAGE. */
+ /*---------------------------------------------------------------------------*/
+ cactiveUndoFilePage = cactiveUndoFilePage - tmp1;
+ fsConnectptr.p->fsState = READ_UNDO_PAGE;
+ }//if
+ return;
+}//Dbacc::srStartUndoLab()
+
+/* ------- DO UNDO ---------------------------*/
+/* ******************--------------------------------------------------------------- */
+/* NEXTOPERATION ORD FOR EXECUTION OF NEXT OP */
+/* ******************------------------------------+ */
+/* SENDER: ACC, LEVEL B */
+void Dbacc::execNEXTOPERATION(Signal* signal)
+{
+ jamEntry();
+ tresult = 0;
+ srDoUndoLab(signal);
+ return;
+}//Dbacc::execNEXTOPERATION()
+
+void Dbacc::srDoUndoLab(Signal* signal)
+{
+ DirRangePtr souDirRangePtr;
+ DirectoryarrayPtr souDirptr;
+ Page8Ptr souPageidptr;
+ Uint32 tundoPageindex;
+ UndoHeader *undoHeaderPtr;
+ Uint32 tmpindex;
+
+ jam();
+ undopageptr.i = cactiveSrUndoPage;
+ ptrCheckGuard(undopageptr, cundopagesize, undopage);
+ /*---------------------------------------------------------------------------*/
+ /* LAYOUT OF AN UNDO LOG RECORD: */
+ /* ***************************** */
+ /* */
+ /* |----------------------------------------------------| */
+ /* | TABLE ID | */
+ /* |----------------------------------------------------| */
+ /* | ROOT FRAGMENT ID | */
+ /* |----------------------------------------------------| */
+ /* | LOCAL FRAGMENT ID | */
+ /* |----------------------------------------------------| */
+ /* | UNDO INFO LEN 14 b | TYPE 4 b | PAGE INDEX 14 b | */
+ /* |----------------------------------------------------| */
+ /* | INDEX INTO PAGE DIRECTORY (LOGICAL PAGE ID) | */
+ /* |----------------------------------------------------| */
+ /* | PREVIOUS UNDO LOG RECORD FOR THE FRAGMENT | */
+ /* |----------------------------------------------------| */
+ /* | PREVIOUS UNDO LOG RECORD FOR ALL FRAGMENTS | */
+ /* |----------------------------------------------------| */
+ /* | TYPE SPECIFIC PART | */
+ /* |----------------------------------------------------| */
+ /*---------------------------------------------------------------------------*/
+ /*---------------------------------------------------------------------------*/
+ /* SET THE PAGE POINTER. WE ONLY WORK WITH TWO PAGES IN THIS RESTART */
+ /* ACTIVITY. GET THE PAGE POINTER AND THE PAGE INDEX TO READ FROM. */
+ /*---------------------------------------------------------------------------*/
+ tundoindex = cprevUndoaddress & ZUNDOPAGEINDEX_MASK; //0x1fff, 13 bits.
+ undoHeaderPtr = (UndoHeader *) &undopageptr.p->undoword[tundoindex];
+ tundoindex = tundoindex + ZUNDOHEADSIZE;
+
+ /*------------------------------------------------------------------------*/
+ /* READ TABLE ID AND ROOT FRAGMENT ID AND USE THIS TO GET ROOT RECORD. */
+ /*------------------------------------------------------------------------*/
+ arrGuard((tundoindex + 6), 8192);
+
+ // TABLE ID
+ tabptr.i = undoHeaderPtr->tableId;
+ ptrCheckGuard(tabptr, ctablesize, tabrec);
+
+ // ROOT FRAGMENT ID
+ tfid = undoHeaderPtr->rootFragId;
+ if (!getrootfragmentrec(signal, rootfragrecptr, tfid)) {
+ jam();
+ /*---------------------------------------------------------------------*/
+ /* THE ROOT RECORD WAS NOT FOUND. OBVIOUSLY WE ARE NOT RESTARTING THIS */
+ /* FRAGMENT. WE THUS IGNORE THIS LOG RECORD AND PROCEED WITH THE NEXT. */
+ /*---------------------------------------------------------------------*/
+ creadyUndoaddress = cprevUndoaddress;
+ // PREVIOUS UNDO LOG RECORD FOR ALL FRAGMENTS
+ cprevUndoaddress = undoHeaderPtr->prevUndoAddress;
+ undoNext2Lab(signal);
+ return;
+ }//if
+ /*-----------------------------------------------------------------------*/
+ /* READ THE LOCAL FRAGMENT ID AND VERIFY THAT IT IS CORRECT. */
+ /*-----------------------------------------------------------------------*/
+ if (rootfragrecptr.p->fragmentid[0] == undoHeaderPtr->localFragId) {
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[0];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ } else {
+ if (rootfragrecptr.p->fragmentid[1] == undoHeaderPtr->localFragId) {
+ jam();
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[1];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ } else {
+ jam();
+ progError(__LINE__, 0, "Invalid local fragment id in undo log");
+ return;
+ }//if
+ }//if
+ /*------------------------------------------------------------------------*/
+ /* READ UNDO INFO LENGTH, TYPE OF LOG RECORD AND PAGE INDEX WHERE TO */
+ /* APPLY THIS LOG RECORD. ALSO STEP INDEX TO PREPARE READ OF LOGICAL */
+ /* PAGE ID. SET TMPINDEX TO INDEX THE FIRST WORD IN THE TYPE SPECIFIC */
+ /* PART. */
+ /*------------------------------------------------------------------------*/
+ // UNDO INFO LENGTH 14 b | TYPE 4 b | PAGE INDEX 14 b
+ const Uint32 tmp1 = undoHeaderPtr->variousInfo;
+ cundoinfolength = tmp1 >> 18;
+ const Uint32 tpageType = (tmp1 >> 14) & 0xf;
+ tundoPageindex = tmp1 & 0x3fff;
+
+ // INDEX INTO PAGE DIRECTORY (LOGICAL PAGE ID)
+ tmpP = undoHeaderPtr->logicalPageId ;
+ tmpindex = tundoindex;
+ arrGuard((tmpindex + cundoinfolength - 1), 8192);
+ if (fragrecptr.p->localCheckpId != cactiveCheckpId) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ /* THE FRAGMENT DID EXIST BUT IS NOT AFFECTED BY THIS UNDO LOG */
+ /* EXECUTION. EITHER IT BELONGS TO ANOTHER OR IT IS CREATED AND ONLY IN */
+ /* NEED OF EXECUTION OF REDO LOG RECORDS FROM LQH. */
+ /*-----------------------------------------------------------------------*/
+ creadyUndoaddress = cprevUndoaddress;
+ // PREVIOUS UNDO LOG RECORD FOR ALL FRAGMENTS
+ cprevUndoaddress = undoHeaderPtr->prevUndoAddress;
+
+ undoNext2Lab(signal);
+ return;
+ }//if
+ /*-----------------------------------------------------------------------*/
+ /* VERIFY CONSISTENCY OF UNDO LOG RECORDS. */
+ /*-----------------------------------------------------------------------*/
+ ndbrequire(fragrecptr.p->prevUndoposition == cprevUndoaddress);
+ cSrUndoRecords[tpageType]++;
+ switch(tpageType){
+
+ case UndoHeader::ZPAGE_INFO:{
+ jam();
+ /*----------------------------------------------------------------------*/
+ /* WE HAVE TO UNDO UPDATES IN A NORMAL PAGE. GET THE PAGE POINTER BY */
+ /* USING THE LOGICAL PAGE ID. THEN RESET THE OLD VALUE IN THE PAGE BY */
+ /* USING THE OLD DATA WHICH IS STORED IN THIS UNDO LOG RECORD. */
+ /*----------------------------------------------------------------------*/
+ souDirRangePtr.i = fragrecptr.p->directory;
+ tmpP2 = tmpP >> 8;
+ tmpP = tmpP & 0xff;
+ ptrCheckGuard(souDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(tmpP2, 256);
+ souDirptr.i = souDirRangePtr.p->dirArray[tmpP2];
+ ptrCheckGuard(souDirptr, cdirarraysize, directoryarray);
+ souPageidptr.i = souDirptr.p->pagep[tmpP];
+ ptrCheckGuard(souPageidptr, cpagesize, page8);
+ Uint32 loopLimit = tundoPageindex + cundoinfolength;
+ ndbrequire(loopLimit <= 2048);
+ for (Uint32 tmp = tundoPageindex; tmp < loopLimit; tmp++) {
+ dbgWord32(souPageidptr, tmp, undopageptr.p->undoword[tmpindex]);
+ souPageidptr.p->word32[tmp] = undopageptr.p->undoword[tmpindex];
+ tmpindex = tmpindex + 1;
+ }//for
+ break;
+ }
+
+ case UndoHeader::ZOVER_PAGE_INFO:{
+ jam();
+ /*----------------------------------------------------------------------*/
+ /* WE HAVE TO UNDO UPDATES IN AN OVERFLOW PAGE. GET THE PAGE POINTER BY*/
+ /* USING THE LOGICAL PAGE ID. THEN RESET THE OLD VALUE IN THE PAGE BY */
+ /* USING THE OLD DATA WHICH IS STORED IN THIS UNDO LOG RECORD. */
+ /*----------------------------------------------------------------------*/
+ souDirRangePtr.i = fragrecptr.p->overflowdir;
+ tmpP2 = tmpP >> 8;
+ tmpP = tmpP & 0xff;
+ ptrCheckGuard(souDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(tmpP2, 256);
+ souDirptr.i = souDirRangePtr.p->dirArray[tmpP2];
+ ptrCheckGuard(souDirptr, cdirarraysize, directoryarray);
+ souPageidptr.i = souDirptr.p->pagep[tmpP];
+ ptrCheckGuard(souPageidptr, cpagesize, page8);
+ Uint32 loopLimit = tundoPageindex + cundoinfolength;
+ ndbrequire(loopLimit <= 2048);
+ for (Uint32 tmp = tundoPageindex; tmp < loopLimit; tmp++) {
+ dbgWord32(souPageidptr, tmp, undopageptr.p->undoword[tmpindex]);
+ souPageidptr.p->word32[tmp] = undopageptr.p->undoword[tmpindex];
+ tmpindex = tmpindex + 1;
+ }//for
+ break;
+ }
+
+ case UndoHeader::ZUNDO_INSERT_LONG_KEY:{
+ jam();
+ /*---------------------------------------------------------------------*/
+ /* WE WILL UNDO AN INSERT OF A LONG KEY. THIS IS PERFORMED BY DELETING */
+ /* THE LONG KEY. */
+ /*---------------------------------------------------------------------*/
+ souDirRangePtr.i = fragrecptr.p->overflowdir;
+ tmpP2 = tmpP >> 8;
+ tmpP = tmpP & 0xff;
+ arrGuard(tmpP2, 256);
+ ptrCheckGuard(souDirRangePtr, cdirrangesize, dirRange);
+ souDirptr.i = souDirRangePtr.p->dirArray[tmpP2];
+ ptrCheckGuard(souDirptr, cdirarraysize, directoryarray);
+ dlkPageptr.i = souDirptr.p->pagep[tmpP];
+ ptrCheckGuard(dlkPageptr, cpagesize, page8);
+ tdlkLogicalPageIndex = tundoPageindex;
+ deleteLongKey(signal);
+ break;
+ }
+
+ case UndoHeader::ZUNDO_DELETE_LONG_KEY: {
+ jam();
+ /*----------------------------------------------------------------------*/
+ /* WE WILL UNDO DELETE OF A LONG KEY. THIS IS PERFORMED BY INSERTING */
+ /* IT AGAIN. */
+ /*----------------------------------------------------------------------*/
+ souDirRangePtr.i = fragrecptr.p->overflowdir;
+ taslpDirIndex = tmpP;
+ tmpP2 = tmpP >> 8;
+ tmpP = tmpP & 0xff;
+ ptrCheckGuard(souDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(tmpP2, 256);
+ souDirptr.i = souDirRangePtr.p->dirArray[tmpP2];
+
+ if(souDirptr.i == RNIL) {
+ //----------------------------------------------------------------
+ // Allocate a directory.
+ //----------------------------------------------------------------
+ jam();
+ seizeDirectory(signal);
+ if (tresult > ZLIMIT_OF_ERROR) {
+ jam();
+ sendSystemerror(signal);
+ return;
+ }
+ souDirRangePtr.p->dirArray[tmpP2] = sdDirptr.i;
+ souDirptr.i = souDirRangePtr.p->dirArray[tmpP2];
+ }
+
+ ptrCheckGuard(souDirptr, cdirarraysize, directoryarray);
+ slkapPageptr.i = souDirptr.p->pagep[tmpP];
+
+ if(slkapPageptr.i == RNIL) {
+ //----------------------------------------------------------------
+ // The delete operation was probably the last on the page and the
+ // page was released and not written down to disk. We need to
+ // allocate a page and put it in the same dirindex as it was in
+ // before it was released.
+ // This is because an eventual UNDO_INSERT on the same key in the
+ // same LCP must be able to find the key and it has only the
+ // dirindex to go on, the key itself is not saved on disk in a
+ // UNDO_INSERT.
+ //----------------------------------------------------------------
+ jam();
+ allocSpecificLongOverflowPage(signal);
+ slkapPageptr.i = aslpPageptr.i;
+ }
+
+ ptrCheckGuard(slkapPageptr, cpagesize, page8);
+ seizePage(signal);
+ ndbrequire(tresult <= ZLIMIT_OF_ERROR);
+
+ slkapCopyPageptr = spPageptr;
+ ndbrequire(cundoinfolength <= 2048);
+
+ for (Uint32 tmp = 0; tmp < cundoinfolength; tmp++) {
+ dbgWord32(slkapCopyPageptr, tmp, undopageptr.p->undoword[tmpindex]);
+ slkapCopyPageptr.p->word32[tmp] = undopageptr.p->undoword[tmpindex];
+ tmpindex = tmpindex + 1;
+ }//for
+ jam();
+ //----------------------------------------------------------------
+ // We must store the key at the same place it was deleted from.
+ // This is because an eventual UNDO_INSERT on the same key in the
+ // same LCP must be able to find the key and it has only the index
+ // information stored on disk to go on, the key itself is not
+ // saved on disk in an UNDO_INSERT.
+ //----------------------------------------------------------------
+ tslkapKeyLen = cundoinfolength;
+ tslkapPageIndex = tundoPageindex;
+ storeLongKeysAtPos(signal);
+
+ rpPageptr = slkapCopyPageptr;
+ releasePage(signal);
+ break;
+ }
+
+ case UndoHeader::ZOP_INFO: {
+ jam();
+ /*---------------------------------------------------------------------*/
+ /* AN OPERATION WAS ACTIVE WHEN LOCAL CHECKPOINT WAS EXECUTED. WE NEED */
+ /* TO RESET THE LOCKS IT HAS SET. IF THE OPERATION WAS AN INSERT OR */
+ /* THE ELEMENT WAS MARKED AS DISSAPEARED IT WILL ALSO BE REMOVED */
+ /* FROM THE PAGE */
+ /* */
+ /* BEGIN BY SEARCHING AFTER THE ELEMENT, WHEN FOUND UNDO THE */
+ /* CHANGES ON THE ELEMENT HEADER. IF IT WAS AN INSERT OPERATION OR */
+ /* MARKED AS DISSAPEARED PROCEED BY REMOVING THE ELEMENT. */
+ /*---------------------------------------------------------------------*/
+ seizeOpRec(signal);
+ // Initialise the opRec
+ operationRecPtr.p->transId1 = 0;
+ operationRecPtr.p->transId2 = RNIL;
+ operationRecPtr.p->transactionstate = ACTIVE;
+ operationRecPtr.p->commitDeleteCheckFlag = ZFALSE;
+ operationRecPtr.p->lockMode = 0;
+ operationRecPtr.p->dirtyRead = 0;
+ operationRecPtr.p->nodeType = 0;
+ operationRecPtr.p->fid = fragrecptr.p->myfid;
+ operationRecPtr.p->nextParallelQue = RNIL;
+ operationRecPtr.p->prevParallelQue = RNIL;
+ operationRecPtr.p->nextQueOp = RNIL;
+ operationRecPtr.p->prevQueOp = RNIL;
+ operationRecPtr.p->nextSerialQue = RNIL;
+ operationRecPtr.p->prevSerialQue = RNIL;
+ operationRecPtr.p->elementPage = RNIL;
+ operationRecPtr.p->keyinfoPage = RNIL;
+ operationRecPtr.p->insertIsDone = ZFALSE;
+ operationRecPtr.p->lockOwner = ZFALSE;
+ operationRecPtr.p->elementIsDisappeared = ZFALSE;
+ operationRecPtr.p->insertDeleteLen = fragrecptr.p->elementLength;
+ operationRecPtr.p->longPagePtr = RNIL;
+ operationRecPtr.p->longKeyPageIndex = RNIL;
+ operationRecPtr.p->scanRecPtr = RNIL;
+ operationRecPtr.p->isAccLockReq = ZFALSE;
+
+ // Read operation values from undo page
+ operationRecPtr.p->operation = undopageptr.p->undoword[tmpindex];
+ tmpindex++;
+ operationRecPtr.p->hashValue = undopageptr.p->undoword[tmpindex];
+ tmpindex++;
+ const Uint32 tkeylen = undopageptr.p->undoword[tmpindex];
+ tmpindex++;
+ operationRecPtr.p->tupkeylen = tkeylen;
+ operationRecPtr.p->fragptr = fragrecptr.i;
+
+ ndbrequire((fragrecptr.p->keyLength == 0) ||
+ ((fragrecptr.p->keyLength != 0) &&
+ (fragrecptr.p->keyLength == tkeylen)));
+
+ // Read keydata from undo page
+ for (Uint32 tmp = 0; tmp < tkeylen; tmp++) {
+ signal->theData[7+tmp] = undopageptr.p->undoword[tmpindex];
+ tmpindex = tmpindex + 1;
+ }//for
+ arrGuard((tmpindex - 1), 8192);
+ getElement(signal);
+ if (tgeResult != ZTRUE) {
+ jam();
+ signal->theData[0] = RNIL;
+ signal->theData[1] = tabptr.i;
+ signal->theData[2] = cactiveCheckpId;
+ signal->theData[3] = cprevUndoaddress;
+ signal->theData[4] = operationRecPtr.p->operation;
+ signal->theData[5] = operationRecPtr.p->hashValue;
+ signal->theData[6] = operationRecPtr.p->tupkeylen;
+ sendSignal(cownBlockref, GSN_DEBUG_SIG, signal, 11, JBA);
+ return;
+ }//if
+
+ operationRecPtr.p->elementPage = gePageptr.i;
+ operationRecPtr.p->elementContainer = tgeContainerptr;
+ operationRecPtr.p->elementPointer = tgeElementptr;
+ operationRecPtr.p->elementIsforward = tgeForward;
+
+ commitdelete(signal, true);
+ releaseOpRec(signal);
+ break;
+ }
+
+ default:
+ jam();
+ progError(__LINE__, 0, "Invalid pagetype in undo log");
+ break;
+
+ }//switch(tpageType)
+
+ /*----------------------------------------------------------------------*/
+ /* READ THE PAGE ID AND THE PAGE INDEX OF THE PREVIOUS UNDO LOG RECORD */
+ /* FOR THIS FRAGMENT. */
+ /*----------------------------------------------------------------------*/
+ fragrecptr.p->prevUndoposition = undoHeaderPtr->prevUndoAddressForThisFrag;
+ /*----------------------------------------------------------------------*/
+ /* READ THE PAGE ID AND THE PAGE INDEX OF THE PREVIOUS UNDO LOG RECORD */
+ /* FOR THIS UNDO LOG. */
+ /*----------------------------------------------------------------------*/
+ creadyUndoaddress = cprevUndoaddress;
+ cprevUndoaddress = undoHeaderPtr->prevUndoAddress;
+
+ if (fragrecptr.p->prevUndoposition == cminusOne) {
+ jam();
+ /*---------------------------------------------------------------------*/
+ /* WE HAVE NOW EXECUTED ALL UNDO LOG RECORDS FOR THIS FRAGMENT. WE */
+ /* NOW NEED TO UPDATE THE FREE LIST OF OVERFLOW PAGES. */
+ /*---------------------------------------------------------------------*/
+ ndbrequire(fragrecptr.p->nextAllocPage == 0);
+
+ signal->theData[0] = fragrecptr.i;
+ sendSignal(cownBlockref, GSN_ACC_OVER_REC, signal, 1, JBB);
+ return;
+ }//if
+ undoNext2Lab(signal);
+ return;
+}//Dbacc::srDoUndoLab()
+
+void Dbacc::undoNext2Lab(Signal* signal)
+{
+ /*---------------------------------------------------------------------------*/
+ /* EXECUTE NEXT UNDO LOG RECORD. */
+ /*---------------------------------------------------------------------------*/
+ if (cprevUndoaddress == cminusOne) {
+ jam();
+ /*---------------------------------------------------------------------------*/
+ /* WE HAVE EXECUTED THIS UNDO LOG TO COMPLETION. IT IS NOW TIME TO TAKE*/
+ /* OF THE NEXT UNDO LOG OR REPORT COMPLETION OF UNDO LOG EXECUTION. */
+ /*---------------------------------------------------------------------------*/
+ signal->theData[0] = ZSTART_UNDO;
+ sendSignal(cownBlockref, GSN_CONTINUEB, signal, 1, JBB);
+ return;
+ }//if
+ if ((creadyUndoaddress >> 13) != (cprevUndoaddress >> 13)) {
+ /*---------------------------------------------------------------------------*/
+ /* WE ARE CHANGING PAGE. */
+ /*---------------------------------------------------------------------------*/
+ if (cactiveSrUndoPage == 0) {
+ jam();
+ /*---------------------------------------------------------------------------*/
+ /* WE HAVE READ AND EXECUTED ALL UNDO LOG INFORMATION IN THE CURRENTLY */
+ /* READ PAGES. WE STILL HAVE MORE INFORMATION TO READ FROM FILE SINCE */
+ /* WE HAVEN'T FOUND THE FIRST LOG RECORD IN THE LOG FILE YET. */
+ /*---------------------------------------------------------------------------*/
+ srStartUndoLab(signal);
+ return;
+ } else {
+ jam();
+ /*---------------------------------------------------------------------------*/
+ /* WE HAVE ANOTHER PAGE READ THAT WE NEED TO EXECUTE. */
+ /*---------------------------------------------------------------------------*/
+ cactiveSrUndoPage = cactiveSrUndoPage - 1;
+ }//if
+ }//if
+ /*---------------------------------------------------------------------------*/
+ /* REAL-TIME BREAK */
+ /*---------------------------------------------------------------------------*/
+ /* ******************************< */
+ /* NEXTOPERATION */
+ /* ******************************< */
+ sendSignal(cownBlockref, GSN_NEXTOPERATION, signal, 1, JBB);
+ return;
+}//Dbacc::undoNext2Lab()
+
+/*-----------------------------------------------------------------------------------*/
+/* AFTER COMPLETING THE READING OF DATA PAGES FROM DISK AND EXECUTING THE UNDO */
+/* LOG WE ARE READY TO UPDATE THE FREE LIST OF OVERFLOW PAGES. THIS LIST MUST */
+/* BE BUILT AGAIN SINCE IT IS NOT CHECKPOINTED. WHEN THE PAGES ARE ALLOCATED */
+/* THEY ARE NOT PART OF ANY LIST. PAGES CAN EITHER BE PUT IN FREE LIST, NOT */
+/* IN FREE LIST OR BE PUT INTO LIST OF LONG KEY PAGES. */
+/*-----------------------------------------------------------------------------------*/
+void Dbacc::execACC_OVER_REC(Signal* signal)
+{
+ DirRangePtr pnoDirRangePtr;
+ DirectoryarrayPtr pnoOverflowDirptr;
+ Page8Ptr pnoPageidptr;
+ Uint32 tpnoPageType;
+ Uint32 toverPageCheck;
+
+ jamEntry();
+ fragrecptr.i = signal->theData[0];
+ toverPageCheck = 0;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ ndbrequire((fragrecptr.p->nextAllocPage != 0) ||
+ (fragrecptr.p->firstOverflowRec == RNIL));
+ /*-----------------------------------------------------------------------------------*/
+ /* WHO HAS PUT SOMETHING INTO THE LIST BEFORE WE EVEN STARTED PUTTING THINGS */
+ /* THERE. */
+ /*-----------------------------------------------------------------------------------*/
+ ndbrequire(fragrecptr.p->loadingFlag == ZTRUE);
+ /*---------------------------------------------------------------------------*/
+ /* LOADING HAS STOPPED BEFORE WE HAVE LOADED, SYSTEM ERROR. */
+ /*---------------------------------------------------------------------------*/
+ while (toverPageCheck < ZNO_OF_OP_PER_SIGNAL) {
+ jam();
+ if (fragrecptr.p->nextAllocPage >= fragrecptr.p->lastOverIndex) {
+ jam();
+ fragrecptr.p->loadingFlag = ZFALSE;
+ rootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ if (rootfragrecptr.p->lcpPtr != RNIL) {
+ jam();
+ srCloseDataFileLab(signal);
+ } else {
+ jam();
+ undoNext2Lab(signal);
+ }//if
+ return;
+ }//if
+ tmpP = fragrecptr.p->nextAllocPage;
+ pnoDirRangePtr.i = fragrecptr.p->overflowdir;
+ tmpP2 = tmpP >> 8;
+ tmpP = tmpP & 0xff;
+ arrGuard(tmpP2, 256);
+ ptrCheckGuard(pnoDirRangePtr, cdirrangesize, dirRange);
+ if (pnoDirRangePtr.p->dirArray[tmpP2] == RNIL) {
+ jam();
+ pnoPageidptr.i = RNIL;
+ } else {
+ pnoOverflowDirptr.i = pnoDirRangePtr.p->dirArray[tmpP2];
+ if (pnoOverflowDirptr.i == RNIL) {
+ jam();
+ pnoPageidptr.i = RNIL;
+ } else {
+ jam();
+ ptrCheckGuard(pnoOverflowDirptr, cdirarraysize, directoryarray);
+ pnoPageidptr.i = pnoOverflowDirptr.p->pagep[tmpP];
+ }//if
+ }//if
+ if (pnoPageidptr.i == RNIL) {
+ jam();
+ seizeOverRec(signal);
+ sorOverflowRecPtr.p->dirindex = fragrecptr.p->nextAllocPage;
+ sorOverflowRecPtr.p->overpage = RNIL;
+ priOverflowRecPtr = sorOverflowRecPtr;
+ putRecInFreeOverdir(signal);
+ } else {
+ ptrCheckGuard(pnoPageidptr, cpagesize, page8);
+ tpnoPageType = pnoPageidptr.p->word32[ZPOS_PAGE_TYPE];
+ tpnoPageType = (tpnoPageType >> ZPOS_PAGE_TYPE_BIT) & 3;
+ if (tpnoPageType == ZLONG_PAGE_TYPE) {
+ jam();
+ // This is to clean the list parameters.
+ pnoPageidptr.p->word32[ZPOS_PREV_PAGE] = RNIL;
+ pnoPageidptr.p->word32[ZPOS_NEXT_PAGE] = RNIL;
+ if (pnoPageidptr.p->word32[ZPOS_ARRAY_POS] != 4) {
+ jam();
+ /*---------------------------------------------------------------------------*/
+ /* THE PAGE WAS A LONG PAGE AND IT BELONGED TO A FREE LIST. PUT IT INTO ONE */
+ /* OF THE FREE LIST THEN. */
+ /*---------------------------------------------------------------------------*/
+ // Insert page!
+ ipaPagePtr = pnoPageidptr;
+ tipaArrayPos = pnoPageidptr.p->word32[ZPOS_ARRAY_POS];
+ insertPageArrayList(signal);
+ }//if
+ } else {
+ if (pnoPageidptr.p->word32[ZPOS_ALLOC_CONTAINERS] > ZFREE_LIMIT) {
+ jam();
+ dbgWord32(pnoPageidptr, ZPOS_OVERFLOWREC, RNIL);
+ pnoPageidptr.p->word32[ZPOS_OVERFLOWREC] = RNIL;
+ ndbrequire(pnoPageidptr.p->word32[ZPOS_PAGE_ID] == fragrecptr.p->nextAllocPage);
+ } else {
+ jam();
+ seizeOverRec(signal);
+ sorOverflowRecPtr.p->dirindex = pnoPageidptr.p->word32[ZPOS_PAGE_ID];
+ ndbrequire(sorOverflowRecPtr.p->dirindex == fragrecptr.p->nextAllocPage);
+ dbgWord32(pnoPageidptr, ZPOS_OVERFLOWREC, sorOverflowRecPtr.i);
+ pnoPageidptr.p->word32[ZPOS_OVERFLOWREC] = sorOverflowRecPtr.i;
+ sorOverflowRecPtr.p->overpage = pnoPageidptr.i;
+ porOverflowRecPtr = sorOverflowRecPtr;
+ putOverflowRecInFrag(signal);
+ if (pnoPageidptr.p->word32[ZPOS_ALLOC_CONTAINERS] == 0) {
+ jam();
+ ropPageptr = pnoPageidptr;
+ releaseOverpage(signal);
+ }//if
+ }//if
+ }//if
+ }//if
+ fragrecptr.p->nextAllocPage++;
+ toverPageCheck++;
+ }//while
+ signal->theData[0] = fragrecptr.i;
+ sendSignal(cownBlockref, GSN_ACC_OVER_REC, signal, 1, JBB);
+}//Dbacc::execACC_OVER_REC()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF SYSTEM RESTART MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* SCAN MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* ACC_SCANREQ START OF A SCAN PROCESS */
+/* SENDER: LQH, LEVEL B */
+/* ENTER ACC_SCANREQ WITH */
+/* TUSERPTR, LQH SCAN_CONNECT POINTER */
+/* TUSERBLOCKREF, LQH BLOCK REFERENCE */
+/* TABPTR, TABLE IDENTITY AND PTR */
+/* TFID ROOT FRAGMENT IDENTITY */
+/* TSCAN_FLAG , = ZCOPY, ZSCAN, ZSCAN_LOCK_ALL */
+/* ZREADLOCK, ZWRITELOCK */
+/* TSCAN_TRID1 , TRANSACTION ID PART 1 */
+/* TSCAN_TRID2 TRANSACTION ID PART 2 */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* ACC_SCANREQ START OF A SCAN PROCESS */
+/* ******************------------------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execACC_SCANREQ(Signal* signal)
+{
+ jamEntry();
+ AccScanReq * req = (AccScanReq*)&signal->theData[0];
+ tuserptr = req->senderData;
+ tuserblockref = req->senderRef;
+ tabptr.i = req->tableId;
+ tfid = req->fragmentNo;
+ tscanFlag = req->requestInfo;
+ tscanTrid1 = req->transId1;
+ tscanTrid2 = req->transId2;
+
+ tresult = 0;
+ ptrCheckGuard(tabptr, ctablesize, tabrec);
+ ndbrequire(getrootfragmentrec(signal,rootfragrecptr, tfid));
+
+ Uint32 i;
+ for (i = 0; i < MAX_PARALLEL_SCANS_PER_FRAG; i++) {
+ jam();
+ if (rootfragrecptr.p->scan[i] == RNIL) {
+ jam();
+ break;
+ }
+ }
+ ndbrequire(i != MAX_PARALLEL_SCANS_PER_FRAG);
+ ndbrequire(cfirstFreeScanRec != RNIL);
+ seizeScanRec(signal);
+
+ rootfragrecptr.p->scan[i] = scanPtr.i;
+ scanPtr.p->scanBucketState = ScanRec::FIRST_LAP;
+ scanPtr.p->scanLockMode = AccScanReq::getLockMode(tscanFlag);
+ scanPtr.p->scanKeyinfoFlag = AccScanReq::getKeyinfoFlag(tscanFlag);
+ scanPtr.p->scanReadCommittedFlag = AccScanReq::getReadCommittedFlag(tscanFlag);
+
+ /* TWELVE BITS OF THE ELEMENT HEAD ARE SCAN */
+ /* CHECK BITS. THE MASK NOTES WHICH BIT IS */
+ /* ALLOCATED FOR THE ACTIVE SCAN */
+ scanPtr.p->scanMask = 1 << i;
+ scanPtr.p->scanUserptr = tuserptr;
+ scanPtr.p->scanUserblockref = tuserblockref;
+ scanPtr.p->scanTrid1 = tscanTrid1;
+ scanPtr.p->scanTrid2 = tscanTrid2;
+ scanPtr.p->rootPtr = rootfragrecptr.i;
+ scanPtr.p->scanLockHeld = 0;
+ scanPtr.p->scanOpsAllocated = 0;
+ scanPtr.p->scanFirstActiveOp = RNIL;
+ scanPtr.p->scanFirstQueuedOp = RNIL;
+ scanPtr.p->scanLastQueuedOp = RNIL;
+ scanPtr.p->scanFirstLockedOp = RNIL;
+ scanPtr.p->scanLastLockedOp = RNIL;
+ scanPtr.p->scanState = ScanRec::WAIT_NEXT;
+ fragrecptr.i = rootfragrecptr.p->fragmentptr[0];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ initScanFragmentPart(signal);
+
+ /*------------------------------------------------------*/
+ /* We start the timeout loop for the scan process here. */
+ /*------------------------------------------------------*/
+ ndbrequire(scanPtr.p->scanTimer == 0);
+ if (scanPtr.p->scanContinuebCounter == 0) {
+ jam();
+ scanPtr.p->scanContinuebCounter = 1;
+ signal->theData[0] = ZSEND_SCAN_HBREP;
+ signal->theData[1] = scanPtr.i;
+ sendSignalWithDelay(cownBlockref, GSN_CONTINUEB, signal, 100, 2);
+ }//if
+ scanPtr.p->scanTimer = scanPtr.p->scanContinuebCounter;
+ /* ************************ */
+ /* ACC_SCANCONF */
+ /* ************************ */
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = scanPtr.i;
+ signal->theData[2] = 2;
+ /* NR OF LOCAL FRAGMENT */
+ signal->theData[3] = rootfragrecptr.p->fragmentid[0];
+ signal->theData[4] = rootfragrecptr.p->fragmentid[1];
+ signal->theData[7] = AccScanConf::ZNOT_EMPTY_FRAGMENT;
+ sendSignal(scanPtr.p->scanUserblockref, GSN_ACC_SCANCONF, signal, 8, JBB);
+ /* NOT EMPTY FRAGMENT */
+ return;
+}//Dbacc::execACC_SCANREQ()
+
+/* ******************--------------------------------------------------------------- */
+/* NEXT_SCANREQ REQUEST FOR NEXT ELEMENT OF */
+/* ******************------------------------------+ A FRAGMENT. */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execNEXT_SCANREQ(Signal* signal)
+{
+ Uint32 tscanNextFlag;
+ jamEntry();
+ scanPtr.i = signal->theData[0];
+ operationRecPtr.i = signal->theData[1];
+ tscanNextFlag = signal->theData[2];
+ /* ------------------------------------------ */
+ /* 1 = ZCOPY_NEXT GET NEXT ELEMENT */
+ /* 2 = ZCOPY_NEXT_COMMIT COMMIT THE */
+ /* ACTIVE ELEMENT AND GET THE NEXT ONE */
+ /* 3 = ZCOPY_COMMIT COMMIT THE ACTIVE ELEMENT */
+ /* 4 = ZCOPY_REPEAT GET THE ACTIVE ELEMENT */
+ /* 5 = ZCOPY_ABORT RELOCK THE ACTIVE ELEMENT */
+ /* 6 = ZCOPY_CLOSE THE SCAN PROCESS IS READY */
+ /* ------------------------------------------ */
+ tresult = 0;
+ ptrCheckGuard(scanPtr, cscanRecSize, scanRec);
+ ndbrequire(scanPtr.p->scanState == ScanRec::WAIT_NEXT);
+
+ scanPtr.p->scanTimer = scanPtr.p->scanContinuebCounter;
+ switch (tscanNextFlag) {
+ case ZCOPY_NEXT:
+ jam();
+ /*empty*/;
+ break;
+ case ZCOPY_NEXT_COMMIT:
+ case ZCOPY_COMMIT:
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* COMMIT ACTIVE OPERATION. SEND NEXT SCAN ELEMENT IF IT IS ZCOPY_NEXT_COMMIT. */
+ /* --------------------------------------------------------------------------------- */
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ fragrecptr.i = operationRecPtr.p->fragptr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ if (!scanPtr.p->scanReadCommittedFlag) {
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ if (remainingUndoPages() < ZMIN_UNDO_PAGES_AT_COMMIT) {
+ jam();
+ /*--------------------------------------------------------------*/
+ // We did not have enough undo log buffers to safely commit an
+ // operation. Try again in 10 milliseconds.
+ /*--------------------------------------------------------------*/
+ sendSignalWithDelay(cownBlockref, GSN_NEXT_SCANREQ, signal, 10, 3);
+ return;
+ }//if
+ }//if
+ commitOperation(signal);
+ }//if
+ takeOutActiveScanOp(signal);
+ releaseOpRec(signal);
+ scanPtr.p->scanOpsAllocated--;
+ if (tscanNextFlag == ZCOPY_COMMIT) {
+ jam();
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ Uint32 blockNo = refToBlock(scanPtr.p->scanUserblockref);
+ EXECUTE_DIRECT(blockNo, GSN_NEXT_SCANCONF, signal, 1);
+ return;
+ }//if
+ break;
+ case ZCOPY_CLOSE:
+ jam();
+ fragrecptr.i = scanPtr.p->activeLocalFrag;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ if (!scanPtr.p->scanReadCommittedFlag) {
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ if (remainingUndoPages() < ZMIN_UNDO_PAGES_AT_OPERATION) {
+ jam();
+ /*--------------------------------------------------------------*/
+ // We did not have enough undo log buffers to commit a set of
+ // operations. Try again in 10 milliseconds.
+ /*--------------------------------------------------------------*/
+ sendSignalWithDelay(cownBlockref, GSN_NEXT_SCANREQ, signal, 10, 3);
+ return;
+ }//if
+ }//if
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* THE SCAN PROCESS IS FINISHED. RELOCK ALL LOCKED EL. RELESE ALL INVOLVED REC. */
+ /* --------------------------------------------------------------------------------- */
+ releaseScanLab(signal);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ signal->theData[0] = scanPtr.i;
+ signal->theData[1] = AccCheckScan::ZNOT_CHECK_LCP_STOP;
+ execACC_CHECK_SCAN(signal);
+ return;
+}//Dbacc::execNEXT_SCANREQ()
+
+void Dbacc::checkNextBucketLab(Signal* signal)
+{
+ DirRangePtr cscDirRangePtr;
+ DirectoryarrayPtr cscDirptr;
+ DirectoryarrayPtr tnsDirptr;
+ Page8Ptr nsPageptr;
+ Page8Ptr cscPageidptr;
+ Page8Ptr gnsPageidptr;
+ Page8Ptr tnsPageidptr;
+ Uint32 tnsElementptr;
+ Uint32 tnsContainerptr;
+ Uint32 tnsIsLocked;
+ Uint32 tnsTmp1;
+ Uint32 tnsTmp2;
+ Uint32 tnsCopyIndex1;
+ Uint32 tnsCopyIndex2;
+ Uint32 tnsCopyDir;
+
+ tnsCopyDir = scanPtr.p->nextBucketIndex >> fragrecptr.p->k;
+ tnsCopyIndex1 = tnsCopyDir >> 8;
+ tnsCopyIndex2 = tnsCopyDir & 0xff;
+ arrGuard(tnsCopyIndex1, 256);
+ tnsDirptr.i = gnsDirRangePtr.p->dirArray[tnsCopyIndex1];
+ ptrCheckGuard(tnsDirptr, cdirarraysize, directoryarray);
+ tnsPageidptr.i = tnsDirptr.p->pagep[tnsCopyIndex2];
+ ptrCheckGuard(tnsPageidptr, cpagesize, page8);
+ gnsPageidptr.i = tnsPageidptr.i;
+ gnsPageidptr.p = tnsPageidptr.p;
+ tnsTmp1 = (1 << fragrecptr.p->k) - 1;
+ tgsePageindex = scanPtr.p->nextBucketIndex & tnsTmp1;
+ gsePageidptr.i = gnsPageidptr.i;
+ gsePageidptr.p = gnsPageidptr.p;
+ if (!getScanElement(signal)) {
+ scanPtr.p->nextBucketIndex++;
+ if (scanPtr.p->scanBucketState == ScanRec::SECOND_LAP) {
+ if (scanPtr.p->nextBucketIndex > scanPtr.p->maxBucketIndexToRescan) {
+ /* --------------------------------------------------------------------------------- */
+ // We have finished the rescan phase. We are ready to proceed with the next fragment part.
+ /* --------------------------------------------------------------------------------- */
+ jam();
+ checkNextFragmentLab(signal);
+ return;
+ }//if
+ } else if (scanPtr.p->scanBucketState == ScanRec::FIRST_LAP) {
+ if ((fragrecptr.p->p + fragrecptr.p->maxp) < scanPtr.p->nextBucketIndex) {
+ /* --------------------------------------------------------------------------------- */
+ // All buckets have been scanned a first time.
+ /* --------------------------------------------------------------------------------- */
+ if (scanPtr.p->minBucketIndexToRescan == 0xFFFFFFFF) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // We have not had any merges behind the scan. Thus it is not necessary to perform
+ // any rescan any buckets and we can proceed immediately with the next fragment part.
+ /* --------------------------------------------------------------------------------- */
+ checkNextFragmentLab(signal);
+ return;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // Some buckets are in the need of rescanning due to merges that have moved records
+ // from in front of the scan to behind the scan. During the merges we kept track of
+ // which buckets that need a rescan. We start with the minimum and end with maximum.
+ /* --------------------------------------------------------------------------------- */
+ scanPtr.p->nextBucketIndex = scanPtr.p->minBucketIndexToRescan;
+ scanPtr.p->scanBucketState = ScanRec::SECOND_LAP;
+ if (scanPtr.p->maxBucketIndexToRescan > (fragrecptr.p->p + fragrecptr.p->maxp)) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // If we have had so many merges that the maximum is bigger than the number of buckets
+ // then we will simply satisfy ourselves with scanning to the end. This can only happen
+ // after bringing down the total of buckets to less than half and the minimum should
+ // be 0 otherwise there is some problem.
+ /* --------------------------------------------------------------------------------- */
+ if (scanPtr.p->minBucketIndexToRescan != 0) {
+ jam();
+ sendSystemerror(signal);
+ return;
+ }//if
+ scanPtr.p->maxBucketIndexToRescan = fragrecptr.p->p + fragrecptr.p->maxp;
+ }//if
+ }//if
+ }//if
+ }//if
+ if ((scanPtr.p->scanBucketState == ScanRec::FIRST_LAP) &&
+ (scanPtr.p->nextBucketIndex <= scanPtr.p->startNoOfBuckets)) {
+ /* --------------------------------------------------------------------------------- */
+ // We will only reset the scan indicator on the buckets that existed at the start of the
+ // scan. The others will be handled by the split and merge code.
+ /* --------------------------------------------------------------------------------- */
+ tnsTmp2 = (1 << fragrecptr.p->k) - 1;
+ trsbPageindex = scanPtr.p->nextBucketIndex & tnsTmp2;
+ if (trsbPageindex != 0) {
+ jam();
+ rsbPageidptr.i = gnsPageidptr.i;
+ rsbPageidptr.p = gnsPageidptr.p;
+ } else {
+ jam();
+ cscDirRangePtr.i = fragrecptr.p->directory;
+ tmpP = scanPtr.p->nextBucketIndex >> fragrecptr.p->k;
+ tmpP2 = tmpP >> 8;
+ tmpP = tmpP & 0xff;
+ ptrCheckGuard(cscDirRangePtr, cdirrangesize, dirRange);
+ arrGuard(tmpP2, 256);
+ cscDirptr.i = cscDirRangePtr.p->dirArray[tmpP2];
+ ptrCheckGuard(cscDirptr, cdirarraysize, directoryarray);
+ cscPageidptr.i = cscDirptr.p->pagep[tmpP];
+ ptrCheckGuard(cscPageidptr, cpagesize, page8);
+ tmp1 = (1 << fragrecptr.p->k) - 1;
+ trsbPageindex = scanPtr.p->nextBucketIndex & tmp1;
+ rsbPageidptr.i = cscPageidptr.i;
+ rsbPageidptr.p = cscPageidptr.p;
+ }//if
+ releaseScanBucket(signal);
+ }//if
+ signal->theData[0] = scanPtr.i;
+ signal->theData[1] = AccCheckScan::ZCHECK_LCP_STOP;
+ sendSignal(cownBlockref, GSN_ACC_CHECK_SCAN, signal, 2, JBB);
+ return;
+ }//if
+ /* ----------------------------------------------------------------------- */
+ /* AN ELEMENT WHICH HAVE NOT BEEN SCANNED WAS FOUND. WE WILL PREPARE IT */
+ /* TO BE SENT TO THE LQH BLOCK FOR FURTHER PROCESSING. */
+ /* WE ASSUME THERE ARE OPERATION RECORDS AVAILABLE SINCE LQH SHOULD HAVE*/
+ /* GUARANTEED THAT THROUGH EARLY BOOKING. */
+ /* ----------------------------------------------------------------------- */
+ tnsIsLocked = tgseIsLocked;
+ tnsElementptr = tgseElementptr;
+ tnsContainerptr = tgseContainerptr;
+ nsPageptr.i = gsePageidptr.i;
+ nsPageptr.p = gsePageidptr.p;
+ seizeOpRec(signal);
+ tisoIsforward = tgseIsforward;
+ tisoContainerptr = tnsContainerptr;
+ tisoElementptr = tnsElementptr;
+ isoPageptr.i = nsPageptr.i;
+ isoPageptr.p = nsPageptr.p;
+ initScanOpRec(signal);
+
+ if (!tnsIsLocked){
+ if (!scanPtr.p->scanReadCommittedFlag) {
+ jam();
+ slPageidptr = nsPageptr;
+ tslElementptr = tnsElementptr;
+ setlock(signal);
+ insertLockOwnersList(signal, operationRecPtr);
+ }//if
+ } else {
+ arrGuard(tnsElementptr, 2048);
+ queOperPtr.i =
+ ElementHeader::getOpPtrI(nsPageptr.p->word32[tnsElementptr]);
+ ptrCheckGuard(queOperPtr, coprecsize, operationrec);
+ if (queOperPtr.p->elementIsDisappeared == ZTRUE) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // If the lock owner indicates the element is disappeared then we will not report this
+ // tuple. We will continue with the next tuple.
+ /* --------------------------------------------------------------------------------- */
+ releaseOpRec(signal);
+ scanPtr.p->scanOpsAllocated--;
+ signal->theData[0] = scanPtr.i;
+ signal->theData[1] = AccCheckScan::ZCHECK_LCP_STOP;
+ sendSignal(cownBlockref, GSN_ACC_CHECK_SCAN, signal, 2, JBB);
+ return;
+ }//if
+ if (!scanPtr.p->scanReadCommittedFlag) {
+ Uint32 return_result;
+ if (scanPtr.p->scanLockMode == ZREADLOCK) {
+ jam();
+ priPageptr = nsPageptr;
+ tpriElementptr = tnsElementptr;
+ return_result = placeReadInLockQueue(signal);
+ } else {
+ jam();
+ pwiPageptr = nsPageptr;
+ tpwiElementptr = tnsElementptr;
+ return_result = placeWriteInLockQueue(signal);
+ }//if
+ if (return_result == ZSERIAL_QUEUE) {
+ /* --------------------------------------------------------------------------------- */
+ /* WE PLACED THE OPERATION INTO A SERIAL QUEUE AND THUS WE HAVE TO WAIT FOR */
+ /* THE LOCK TO BE RELEASED. WE CONTINUE WITH THE NEXT ELEMENT. */
+ /* --------------------------------------------------------------------------------- */
+ putOpScanLockQue(); /* PUT THE OP IN A QUE IN THE SCAN REC */
+ signal->theData[0] = scanPtr.i;
+ signal->theData[1] = AccCheckScan::ZCHECK_LCP_STOP;
+ sendSignal(cownBlockref, GSN_ACC_CHECK_SCAN, signal, 2, JBB);
+ return;
+ } else if (return_result == ZWRITE_ERROR) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // The tuple is either not committed yet or a delete in the same transaction (not
+ // possible here since we are a scan). Thus we simply continue with the next tuple.
+ /* --------------------------------------------------------------------------------- */
+ releaseOpRec(signal);
+ scanPtr.p->scanOpsAllocated--;
+ signal->theData[0] = scanPtr.i;
+ signal->theData[1] = AccCheckScan::ZCHECK_LCP_STOP;
+ sendSignal(cownBlockref, GSN_ACC_CHECK_SCAN, signal, 2, JBB);
+ return;
+ }//if
+ ndbassert(return_result == ZPARALLEL_QUEUE);
+ }//if
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ // Committed read proceed without caring for locks immediately down here except when
+ // the tuple was deleted permanently and no new operation has inserted it again.
+ /* --------------------------------------------------------------------------------- */
+ putActiveScanOp(signal);
+ sendNextScanConf(signal);
+ return;
+}//Dbacc::checkNextBucketLab()
+
+
+void Dbacc::checkNextFragmentLab(Signal* signal)
+{
+ RootfragmentrecPtr cnfRootfragrecptr;
+
+ cnfRootfragrecptr.i = fragrecptr.p->myroot;
+ ptrCheckGuard(cnfRootfragrecptr, crootfragmentsize, rootfragmentrec);
+ if (scanPtr.p->activeLocalFrag == cnfRootfragrecptr.p->fragmentptr[0]) {
+ jam();
+ fragrecptr.i = cnfRootfragrecptr.p->fragmentptr[1];
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ initScanFragmentPart(signal);
+ signal->theData[0] = scanPtr.i;
+ signal->theData[1] = AccCheckScan::ZCHECK_LCP_STOP;
+ sendSignal(cownBlockref, GSN_ACC_CHECK_SCAN, signal, 2, JBB);
+ return;
+ } else {
+ if (scanPtr.p->activeLocalFrag == cnfRootfragrecptr.p->fragmentptr[1]) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ // Both fragments have completed their scan part and we can indicate that the scan is
+ // now completed.
+ /* --------------------------------------------------------------------------------- */
+ scanPtr.p->scanBucketState = ScanRec::SCAN_COMPLETED;
+ /*empty*/;
+ } else {
+ jam();
+ /* ALL ELEMENTS ARE SENT */
+ sendSystemerror(signal);
+ }//if
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ // The scan is completed. ACC_CHECK_SCAN will perform all the necessary checks to see
+ // what the next step is.
+ /* --------------------------------------------------------------------------------- */
+ signal->theData[0] = scanPtr.i;
+ signal->theData[1] = AccCheckScan::ZCHECK_LCP_STOP;
+ execACC_CHECK_SCAN(signal);
+ return;
+}//Dbacc::checkNextFragmentLab()
+
+void Dbacc::initScanFragmentPart(Signal* signal)
+{
+ DirRangePtr cnfDirRangePtr;
+ DirectoryarrayPtr cnfDirptr;
+ Page8Ptr cnfPageidptr;
+ /* --------------------------------------------------------------------------------- */
+ // Set the active fragment part.
+ // Set the current bucket scanned to the first.
+ // Start with the first lap.
+ // Remember the number of buckets at start of the scan.
+ // Set the minimum and maximum to values that will always be smaller and larger than.
+ // Reset the scan indicator on the first bucket.
+ /* --------------------------------------------------------------------------------- */
+ scanPtr.p->activeLocalFrag = fragrecptr.i;
+ scanPtr.p->nextBucketIndex = 0; /* INDEX OF SCAN BUCKET */
+ scanPtr.p->scanBucketState = ScanRec::FIRST_LAP;
+ scanPtr.p->startNoOfBuckets = fragrecptr.p->p + fragrecptr.p->maxp;
+ scanPtr.p->minBucketIndexToRescan = 0xFFFFFFFF;
+ scanPtr.p->maxBucketIndexToRescan = 0;
+ cnfDirRangePtr.i = fragrecptr.p->directory;
+ ptrCheckGuard(cnfDirRangePtr, cdirrangesize, dirRange);
+ cnfDirptr.i = cnfDirRangePtr.p->dirArray[0];
+ ptrCheckGuard(cnfDirptr, cdirarraysize, directoryarray);
+ cnfPageidptr.i = cnfDirptr.p->pagep[0];
+ ptrCheckGuard(cnfPageidptr, cpagesize, page8);
+ trsbPageindex = scanPtr.p->nextBucketIndex & ((1 << fragrecptr.p->k) - 1);
+ rsbPageidptr.i = cnfPageidptr.i;
+ rsbPageidptr.p = cnfPageidptr.p;
+ releaseScanBucket(signal);
+}//Dbacc::initScanFragmentPart()
+
+/* --------------------------------------------------------------------------------- */
+/* FLAG = 6 = ZCOPY_CLOSE THE SCAN PROCESS IS READY OR ABORTED. ALL OPERATION IN THE */
+/* ACTIVE OR WAIT QUEUE ARE RELEASED, SCAN FLAG OF ROOT FRAG IS RESET AND THE SCAN */
+/* RECORD IS RELEASED. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseScanLab(Signal* signal)
+{
+ releaseAndCommitActiveOps(signal);
+ releaseAndCommitQueuedOps(signal);
+ releaseAndAbortLockedOps(signal);
+
+ rootfragrecptr.i = scanPtr.p->rootPtr;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ for (tmp = 0; tmp < MAX_PARALLEL_SCANS_PER_FRAG; tmp++) {
+ jam();
+ if (rootfragrecptr.p->scan[tmp] == scanPtr.i) {
+ jam();
+ rootfragrecptr.p->scan[tmp] = RNIL;
+ }//if
+ }//for
+ // Stops the heartbeat.
+ scanPtr.p->scanTimer = 0;
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = RNIL;
+ signal->theData[2] = RNIL;
+ sendSignal(scanPtr.p->scanUserblockref, GSN_NEXT_SCANCONF, signal, 3, JBB);
+ releaseScanRec(signal);
+ return;
+}//Dbacc::releaseScanLab()
+
+
+void Dbacc::releaseAndCommitActiveOps(Signal* signal)
+{
+ OperationrecPtr trsoOperPtr;
+ operationRecPtr.i = scanPtr.p->scanFirstActiveOp;
+ while (operationRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ trsoOperPtr.i = operationRecPtr.p->nextOp;
+ fragrecptr.i = operationRecPtr.p->fragptr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ if (!scanPtr.p->scanReadCommittedFlag) {
+ jam();
+ commitOperation(signal);
+ }//if
+ takeOutActiveScanOp(signal);
+ releaseOpRec(signal);
+ scanPtr.p->scanOpsAllocated--;
+ operationRecPtr.i = trsoOperPtr.i;
+ }//if
+}//Dbacc::releaseAndCommitActiveOps()
+
+
+void Dbacc::releaseAndCommitQueuedOps(Signal* signal)
+{
+ OperationrecPtr trsoOperPtr;
+ operationRecPtr.i = scanPtr.p->scanFirstQueuedOp;
+ while (operationRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ trsoOperPtr.i = operationRecPtr.p->nextOp;
+ fragrecptr.i = operationRecPtr.p->fragptr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ if (!scanPtr.p->scanReadCommittedFlag) {
+ jam();
+ commitOperation(signal);
+ }//if
+ takeOutReadyScanQueue(signal);
+ releaseOpRec(signal);
+ scanPtr.p->scanOpsAllocated--;
+ operationRecPtr.i = trsoOperPtr.i;
+ }//if
+}//Dbacc::releaseAndCommitQueuedOps()
+
+void Dbacc::releaseAndAbortLockedOps(Signal* signal) {
+
+ OperationrecPtr trsoOperPtr;
+ operationRecPtr.i = scanPtr.p->scanFirstLockedOp;
+ while (operationRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ trsoOperPtr.i = operationRecPtr.p->nextOp;
+ fragrecptr.i = operationRecPtr.p->fragptr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ if (!scanPtr.p->scanReadCommittedFlag) {
+ jam();
+ abortOperation(signal);
+ }//if
+ takeOutScanLockQueue(scanPtr.i);
+ releaseOpRec(signal);
+ scanPtr.p->scanOpsAllocated--;
+ operationRecPtr.i = trsoOperPtr.i;
+ }//if
+}//Dbacc::releaseAndAbortLockedOps()
+
+/* 3.18.3 ACC_CHECK_SCAN */
+/* ******************--------------------------------------------------------------- */
+/* ACC_CHECK_SCAN */
+/* ENTER ACC_CHECK_SCAN WITH */
+/* SCAN_PTR */
+/* ******************--------------------------------------------------------------- */
+/* ******************--------------------------------------------------------------- */
+/* ACC_CHECK_SCAN */
+/* ******************------------------------------+ */
+void Dbacc::execACC_CHECK_SCAN(Signal* signal)
+{
+ Uint32 TcheckLcpStop;
+ jamEntry();
+ scanPtr.i = signal->theData[0];
+ TcheckLcpStop = signal->theData[1];
+ ptrCheckGuard(scanPtr, cscanRecSize, scanRec);
+ while (scanPtr.p->scanFirstQueuedOp != RNIL) {
+ jam();
+ //----------------------------------------------------------------------------
+ // An operation has been released from the lock queue. We are in the parallel
+ // queue of this tuple. We are ready to report the tuple now.
+ //----------------------------------------------------------------------------
+ operationRecPtr.i = scanPtr.p->scanFirstQueuedOp;
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ takeOutReadyScanQueue(signal);
+ if (operationRecPtr.p->elementIsDisappeared == ZTRUE) {
+ jam();
+ fragrecptr.i = operationRecPtr.p->fragptr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ if (fragrecptr.p->createLcp == ZTRUE) {
+ if (remainingUndoPages() < ZMIN_UNDO_PAGES_AT_COMMIT) {
+ jam();
+ /*--------------------------------------------------------------*/
+ // We did not have enough undo log buffers to safely abort an
+ // operation. Try again in 10 milliseconds.
+ /*--------------------------------------------------------------*/
+ sendSignalWithDelay(cownBlockref, GSN_ACC_CHECK_SCAN, signal, 10, 2);
+ return;
+ }//if
+ }//if
+ abortOperation(signal);
+ releaseOpRec(signal);
+ scanPtr.p->scanOpsAllocated--;
+ continue;
+ }//if
+ putActiveScanOp(signal);
+ sendNextScanConf(signal);
+ return;
+ }//while
+
+
+ if ((scanPtr.p->scanBucketState == ScanRec::SCAN_COMPLETED) &&
+ (scanPtr.p->scanLockHeld == 0)) {
+ jam();
+ //----------------------------------------------------------------------------
+ // The scan is now completed and there are no more locks outstanding. Thus we
+ // we will report the scan as completed to LQH.
+ //----------------------------------------------------------------------------
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = RNIL;
+ signal->theData[2] = RNIL;
+ sendSignal(scanPtr.p->scanUserblockref, GSN_NEXT_SCANCONF, signal, 3, JBB);
+ return;
+ }//if
+ if (TcheckLcpStop == AccCheckScan::ZCHECK_LCP_STOP) {
+ //---------------------------------------------------------------------------
+ // To ensure that the block of the fragment occurring at the start of a local
+ // checkpoint is not held for too long we insert a release and reacquiring of
+ // that lock here. This is performed in LQH. If we are blocked or if we have
+ // requested a sleep then we will receive RNIL in the returning signal word.
+ //---------------------------------------------------------------------------
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] =
+ ((scanPtr.p->scanLockHeld >= ZSCAN_MAX_LOCK) ||
+ (scanPtr.p->scanBucketState == ScanRec::SCAN_COMPLETED));
+ EXECUTE_DIRECT(DBLQH, GSN_CHECK_LCP_STOP, signal, 2);
+ jamEntry();
+ if (signal->theData[0] == RNIL) {
+ jam();
+ return;
+ }//if
+ }//if
+ /**
+ * If we have more than max locks held OR
+ * scan is completed AND at least one lock held
+ * - Inform LQH about this condition
+ */
+ if ((scanPtr.p->scanLockHeld >= ZSCAN_MAX_LOCK) ||
+ (cfreeopRec == RNIL) ||
+ ((scanPtr.p->scanBucketState == ScanRec::SCAN_COMPLETED) &&
+ (scanPtr.p->scanLockHeld > 0))) {
+ jam();
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = RNIL; // No operation is returned
+ signal->theData[2] = 512; // MASV
+ sendSignal(scanPtr.p->scanUserblockref, GSN_NEXT_SCANCONF, signal, 3, JBB);
+ return;
+ }
+ if (scanPtr.p->scanBucketState == ScanRec::SCAN_COMPLETED) {
+ jam();
+ signal->theData[0] = scanPtr.i;
+ signal->theData[1] = AccCheckScan::ZCHECK_LCP_STOP;
+ execACC_CHECK_SCAN(signal);
+ return;
+ }//if
+
+ scanPtr.p->scanTimer = scanPtr.p->scanContinuebCounter;
+
+ fragrecptr.i = scanPtr.p->activeLocalFrag;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ gnsDirRangePtr.i = fragrecptr.p->directory;
+ ptrCheckGuard(gnsDirRangePtr, cdirrangesize, dirRange);
+ checkNextBucketLab(signal);
+ return;
+}//Dbacc::execACC_CHECK_SCAN()
+
+/* ******************---------------------------------------------------- */
+/* ACC_TO_REQ PERFORM A TAKE OVER */
+/* ******************-------------------+ */
+/* SENDER: LQH, LEVEL B */
+void Dbacc::execACC_TO_REQ(Signal* signal)
+{
+ OperationrecPtr tatrOpPtr;
+
+ jamEntry();
+ tatrOpPtr.i = signal->theData[1]; /* OPER PTR OF ACC */
+ ptrCheckGuard(tatrOpPtr, coprecsize, operationrec);
+ if (tatrOpPtr.p->operation == ZSCAN_OP) {
+ tatrOpPtr.p->transId1 = signal->theData[2];
+ tatrOpPtr.p->transId2 = signal->theData[3];
+ } else {
+ jam();
+ signal->theData[0] = cminusOne;
+ signal->theData[1] = ZTO_OP_STATE_ERROR;
+ }//if
+ return;
+}//Dbacc::execACC_TO_REQ()
+
+/* --------------------------------------------------------------------------------- */
+/* CONTAINERINFO */
+/* INPUT: */
+/* CI_PAGEIDPTR (PAGE POINTER WHERE CONTAINER RESIDES) */
+/* TCI_PAGEINDEX (INDEX OF CONTAINER, USED TO CALCULATE PAGE INDEX) */
+/* TCI_ISFORWARD (DIRECTION OF CONTAINER FORWARD OR BACKWARD) */
+/* */
+/* OUTPUT: */
+/* TCI_CONTAINERPTR (A POINTER TO THE HEAD OF THE CONTAINER) */
+/* TCI_CONTAINERLEN (LENGTH OF THE CONTAINER */
+/* TCI_CONTAINERHEAD (THE HEADER OF THE CONTAINER) */
+/* */
+/* DESCRIPTION: THE ADDRESS OF THE CONTAINER WILL BE CALCULATED AND */
+/* ALL INFORMATION ABOUT THE CONTAINER WILL BE READ */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::containerinfo(Signal* signal)
+{
+ tciContainerptr = (tciPageindex << ZSHIFT_PLUS) - (tciPageindex << ZSHIFT_MINUS);
+ if (tciIsforward == ZTRUE) {
+ jam();
+ tciContainerptr = tciContainerptr + ZHEAD_SIZE;
+ } else {
+ jam();
+ tciContainerptr = ((tciContainerptr + ZHEAD_SIZE) + ZBUF_SIZE) - ZCON_HEAD_SIZE;
+ }//if
+ arrGuard(tciContainerptr, 2048);
+ tciContainerhead = ciPageidptr.p->word32[tciContainerptr];
+ tciContainerlen = tciContainerhead >> 26;
+}//Dbacc::containerinfo()
+
+/* --------------------------------------------------------------------------------- */
+/* GET_SCAN_ELEMENT */
+/* INPUT: GSE_PAGEIDPTR */
+/* TGSE_PAGEINDEX */
+/* OUTPUT: TGSE_IS_LOCKED (IF TRESULT /= ZFALSE) */
+/* GSE_PAGEIDPTR */
+/* TGSE_PAGEINDEX */
+/* --------------------------------------------------------------------------------- */
+bool Dbacc::getScanElement(Signal* signal)
+{
+ tgseIsforward = ZTRUE;
+ NEXTSEARCH_SCAN_LOOP:
+ ciPageidptr.i = gsePageidptr.i;
+ ciPageidptr.p = gsePageidptr.p;
+ tciPageindex = tgsePageindex;
+ tciIsforward = tgseIsforward;
+ containerinfo(signal);
+ sscPageidptr.i = gsePageidptr.i;
+ sscPageidptr.p = gsePageidptr.p;
+ tsscContainerlen = tciContainerlen;
+ tsscContainerptr = tciContainerptr;
+ tsscIsforward = tciIsforward;
+ if (searchScanContainer(signal)) {
+ jam();
+ tgseIsLocked = tsscIsLocked;
+ tgseElementptr = tsscElementptr;
+ tgseContainerptr = tsscContainerptr;
+ return true;
+ }//if
+ if (((tciContainerhead >> 7) & 0x3) != 0) {
+ jam();
+ nciPageidptr.i = gsePageidptr.i;
+ nciPageidptr.p = gsePageidptr.p;
+ tnciContainerhead = tciContainerhead;
+ tnciContainerptr = tciContainerptr;
+ nextcontainerinfo(signal);
+ tgsePageindex = tnciPageindex;
+ gsePageidptr.i = nciPageidptr.i;
+ gsePageidptr.p = nciPageidptr.p;
+ tgseIsforward = tnciIsforward;
+ goto NEXTSEARCH_SCAN_LOOP;
+ }//if
+ return false;
+}//Dbacc::getScanElement()
+
+/* --------------------------------------------------------------------------------- */
+/* INIT_SCAN_OP_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initScanOpRec(Signal* signal)
+{
+ Uint32 tisoTmp;
+ Uint32 tisoLocalPtr;
+ Uint32 guard24;
+ Uint32 tisoPageIndex;
+ Uint32 tisoPagedir;
+ DirRangePtr tisoOverflowrangeptr;
+ DirectoryarrayPtr tisoOverflowDirptr;
+ Page8Ptr tisoPageptr;
+
+ scanPtr.p->scanOpsAllocated++;
+
+ operationRecPtr.p->scanRecPtr = scanPtr.i;
+ operationRecPtr.p->operation = ZSCAN_OP;
+ operationRecPtr.p->transactionstate = ACTIVE;
+ operationRecPtr.p->commitDeleteCheckFlag = ZFALSE;
+ operationRecPtr.p->lockMode = scanPtr.p->scanLockMode;
+ operationRecPtr.p->fid = fragrecptr.p->myfid;
+ operationRecPtr.p->fragptr = fragrecptr.i;
+ operationRecPtr.p->elementIsDisappeared = ZFALSE;
+ operationRecPtr.p->nextParallelQue = RNIL;
+ operationRecPtr.p->prevParallelQue = RNIL;
+ operationRecPtr.p->nextSerialQue = RNIL;
+ operationRecPtr.p->prevSerialQue = RNIL;
+ operationRecPtr.p->prevQueOp = RNIL;
+ operationRecPtr.p->nextQueOp = RNIL;
+ operationRecPtr.p->keyinfoPage = RNIL; // Safety precaution
+ operationRecPtr.p->transId1 = scanPtr.p->scanTrid1;
+ operationRecPtr.p->transId2 = scanPtr.p->scanTrid2;
+ operationRecPtr.p->lockOwner = ZFALSE;
+ operationRecPtr.p->dirtyRead = 0;
+ operationRecPtr.p->nodeType = 0; // Not a stand-by node
+ operationRecPtr.p->elementIsforward = tisoIsforward;
+ operationRecPtr.p->elementContainer = tisoContainerptr;
+ operationRecPtr.p->elementPointer = tisoElementptr;
+ operationRecPtr.p->elementPage = isoPageptr.i;
+ operationRecPtr.p->isAccLockReq = ZFALSE;
+ tisoLocalPtr = tisoElementptr + tisoIsforward;
+ guard24 = fragrecptr.p->localkeylen - 1;
+ for (tisoTmp = 0; tisoTmp <= guard24; tisoTmp++) {
+ arrGuard(tisoTmp, 2);
+ arrGuard(tisoLocalPtr, 2048);
+ operationRecPtr.p->localdata[tisoTmp] = isoPageptr.p->word32[tisoLocalPtr];
+ tisoLocalPtr = tisoLocalPtr + tisoIsforward;
+ }//for
+ arrGuard(tisoLocalPtr, 2048);
+ operationRecPtr.p->keydata[0] = isoPageptr.p->word32[tisoLocalPtr];
+ if (fragrecptr.p->keyLength != 0) {
+ jam();
+ operationRecPtr.p->tupkeylen = fragrecptr.p->keyLength;
+ guard24 = fragrecptr.p->keyLength - 1;
+ for (tisoTmp = 0; tisoTmp <= guard24; tisoTmp++) {
+ arrGuard(tisoTmp, 8);
+ arrGuard(tisoLocalPtr, 2048);
+ operationRecPtr.p->keydata[tisoTmp] = isoPageptr.p->word32[tisoLocalPtr];
+ tisoLocalPtr = tisoLocalPtr + tisoIsforward;
+ }//for
+ } else {
+ // Long key handling. Put the long key reference in the operation records.
+ tisoPageIndex = operationRecPtr.p->keydata[0] & 0x3ff;
+ arrGuard(ZWORDS_IN_PAGE - tisoPageIndex, 2048);
+
+ tisoPagedir = operationRecPtr.p->keydata[0] >> 10;
+ arrGuard((tisoPagedir >> 8), 256);
+
+ tisoOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ ptrCheckGuard(tisoOverflowrangeptr, cdirrangesize, dirRange);
+
+ tisoOverflowDirptr.i = tisoOverflowrangeptr.p->dirArray[tisoPagedir >> 8];
+ ptrCheckGuard(tisoOverflowDirptr, cdirarraysize, directoryarray);
+
+ tisoPageptr.i = tisoOverflowDirptr.p->pagep[tisoPagedir & 0xff];
+ ptrCheckGuard(tisoPageptr, cpagesize, page8);
+
+ operationRecPtr.p->longPagePtr = tisoPageptr.i;
+ operationRecPtr.p->longKeyPageIndex = tisoPageIndex;
+
+ // Read length of key from page
+ Uint32 tmp = tisoPageptr.p->word32[ZWORDS_IN_PAGE - tisoPageIndex];
+ operationRecPtr.p->tupkeylen = tmp >> 16;
+ }
+}//Dbacc::initScanOpRec()
+
+/* --------------------------------------------------------------------------------- */
+/* NEXTCONTAINERINFO */
+/* DESCRIPTION:THE CONTAINER HEAD WILL BE CHECKED TO CALCULATE INFORMATION */
+/* ABOUT NEXT CONTAINER IN THE BUCKET. */
+/* INPUT: TNCI_CONTAINERHEAD */
+/* NCI_PAGEIDPTR */
+/* TNCI_CONTAINERPTR */
+/* OUTPUT: */
+/* TNCI_PAGEINDEX (INDEX FROM WHICH PAGE INDEX CAN BE CALCULATED). */
+/* TNCI_ISFORWARD (IS THE NEXT CONTAINER FORWARD (+1) OR BACKWARD (-1) */
+/* NCI_PAGEIDPTR (PAGE REFERENCE OF NEXT CONTAINER) */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::nextcontainerinfo(Signal* signal)
+{
+ tnciNextSamePage = (tnciContainerhead >> 9) & 0x1; /* CHECK BIT FOR CHECKING WHERE */
+ /* THE NEXT CONTAINER IS IN THE SAME PAGE */
+ tnciPageindex = tnciContainerhead & 0x7f; /* NEXT CONTAINER PAGE INDEX 7 BITS */
+ if (((tnciContainerhead >> 7) & 3) == ZLEFT) {
+ jam();
+ tnciIsforward = ZTRUE;
+ } else {
+ jam();
+ tnciIsforward = cminusOne;
+ }//if
+ if (tnciNextSamePage == ZFALSE) {
+ jam();
+ /* NEXT CONTAINER IS IN AN OVERFLOW PAGE */
+ arrGuard(tnciContainerptr + 1, 2048);
+ tnciTmp = nciPageidptr.p->word32[tnciContainerptr + 1];
+ nciOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ ptrCheckGuard(nciOverflowrangeptr, cdirrangesize, dirRange);
+ arrGuard((tnciTmp >> 8), 256);
+ nciOverflowDirptr.i = nciOverflowrangeptr.p->dirArray[tnciTmp >> 8];
+ ptrCheckGuard(nciOverflowDirptr, cdirarraysize, directoryarray);
+ nciPageidptr.i = nciOverflowDirptr.p->pagep[tnciTmp & 0xff];
+ ptrCheckGuard(nciPageidptr, cpagesize, page8);
+ }//if
+}//Dbacc::nextcontainerinfo()
+
+/* --------------------------------------------------------------------------------- */
+/* PUT_ACTIVE_SCAN_OP */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::putActiveScanOp(Signal* signal)
+{
+ OperationrecPtr pasOperationRecPtr;
+ pasOperationRecPtr.i = scanPtr.p->scanFirstActiveOp;
+ if (pasOperationRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(pasOperationRecPtr, coprecsize, operationrec);
+ pasOperationRecPtr.p->prevOp = operationRecPtr.i;
+ }//if
+ operationRecPtr.p->nextOp = pasOperationRecPtr.i;
+ operationRecPtr.p->prevOp = RNIL;
+ scanPtr.p->scanFirstActiveOp = operationRecPtr.i;
+}//Dbacc::putActiveScanOp()
+
+/**
+ * putOpScanLockQueue
+ *
+ * Description: Put an operation in the doubly linked
+ * lock list on a scan record. The list is used to
+ * keep track of which operations belonging
+ * to the scan are put in serial lock list of another
+ * operation
+ *
+ * @note Use takeOutScanLockQueue to remove an operation
+ * from the list
+ *
+ */
+void Dbacc::putOpScanLockQue()
+{
+
+#ifdef VM_TRACE
+ // DEBUG CODE
+ // Check that there are as many operations in the lockqueue as
+ // scanLockHeld indicates
+ OperationrecPtr tmpOp;
+ int numLockedOpsBefore = 0;
+ tmpOp.i = scanPtr.p->scanFirstLockedOp;
+ while(tmpOp.i != RNIL){
+ numLockedOpsBefore++;
+ ptrCheckGuard(tmpOp, coprecsize, operationrec);
+ if (tmpOp.p->nextOp == RNIL)
+ ndbrequire(tmpOp.i == scanPtr.p->scanLastLockedOp);
+ tmpOp.i = tmpOp.p->nextOp;
+ }
+ ndbrequire(numLockedOpsBefore==scanPtr.p->scanLockHeld);
+#endif
+
+ OperationrecPtr pslOperationRecPtr;
+ ScanRec theScanRec;
+ theScanRec = *scanPtr.p;
+
+ pslOperationRecPtr.i = scanPtr.p->scanLastLockedOp;
+ operationRecPtr.p->prevOp = pslOperationRecPtr.i;
+ operationRecPtr.p->nextOp = RNIL;
+ if (pslOperationRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(pslOperationRecPtr, coprecsize, operationrec);
+ pslOperationRecPtr.p->nextOp = operationRecPtr.i;
+ } else {
+ jam();
+ scanPtr.p->scanFirstLockedOp = operationRecPtr.i;
+ }//if
+ scanPtr.p->scanLastLockedOp = operationRecPtr.i;
+ scanPtr.p->scanLockHeld++;
+
+}//Dbacc::putOpScanLockQue()
+
+/* --------------------------------------------------------------------------------- */
+/* PUT_READY_SCAN_QUEUE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::putReadyScanQueue(Signal* signal, Uint32 scanRecIndex)
+{
+ OperationrecPtr prsOperationRecPtr;
+ ScanRecPtr TscanPtr;
+
+ TscanPtr.i = scanRecIndex;
+ ptrCheckGuard(TscanPtr, cscanRecSize, scanRec);
+
+ prsOperationRecPtr.i = TscanPtr.p->scanLastQueuedOp;
+ operationRecPtr.p->prevOp = prsOperationRecPtr.i;
+ operationRecPtr.p->nextOp = RNIL;
+ TscanPtr.p->scanLastQueuedOp = operationRecPtr.i;
+ if (prsOperationRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(prsOperationRecPtr, coprecsize, operationrec);
+ prsOperationRecPtr.p->nextOp = operationRecPtr.i;
+ } else {
+ jam();
+ TscanPtr.p->scanFirstQueuedOp = operationRecPtr.i;
+ }//if
+}//Dbacc::putReadyScanQueue()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_SCAN_BUCKET */
+// Input:
+// rsbPageidptr.i Index to page where buckets starts
+// rsbPageidptr.p Pointer to page where bucket starts
+// trsbPageindex Page index of starting container in bucket
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseScanBucket(Signal* signal)
+{
+ Uint32 trsbIsforward;
+
+ trsbIsforward = ZTRUE;
+ NEXTRELEASESCANLOOP:
+ ciPageidptr.i = rsbPageidptr.i;
+ ciPageidptr.p = rsbPageidptr.p;
+ tciPageindex = trsbPageindex;
+ tciIsforward = trsbIsforward;
+ containerinfo(signal);
+ rscPageidptr.i = rsbPageidptr.i;
+ rscPageidptr.p = rsbPageidptr.p;
+ trscContainerlen = tciContainerlen;
+ trscContainerptr = tciContainerptr;
+ trscIsforward = trsbIsforward;
+ releaseScanContainer(signal);
+ if (((tciContainerhead >> 7) & 0x3) != 0) {
+ jam();
+ nciPageidptr.i = rsbPageidptr.i;
+ nciPageidptr.p = rsbPageidptr.p;
+ tnciContainerhead = tciContainerhead;
+ tnciContainerptr = tciContainerptr;
+ nextcontainerinfo(signal);
+ rsbPageidptr.i = nciPageidptr.i;
+ rsbPageidptr.p = nciPageidptr.p;
+ trsbPageindex = tnciPageindex;
+ trsbIsforward = tnciIsforward;
+ goto NEXTRELEASESCANLOOP;
+ }//if
+}//Dbacc::releaseScanBucket()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_SCAN_CONTAINER */
+/* INPUT: TRSC_CONTAINERLEN */
+/* RSC_PAGEIDPTR */
+/* TRSC_CONTAINERPTR */
+/* TRSC_ISFORWARD */
+/* SCAN_PTR */
+/* */
+/* DESCRIPTION: SEARCHS IN A CONTAINER, AND THE SCAN BIT OF THE ELEMENTS */
+/* OF THE CONTAINER IS RESET */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseScanContainer(Signal* signal)
+{
+ OperationrecPtr rscOperPtr;
+ Uint32 trscElemStep;
+ Uint32 trscElementptr;
+ Uint32 trscElemlens;
+ Uint32 trscElemlen;
+
+ if (trscContainerlen < 5) {
+ if (trscContainerlen != ZCON_HEAD_SIZE) {
+ jam();
+ sendSystemerror(signal);
+ }//if
+ return; /* 3 IS THE MINIMUM SIZE OF THE ELEMENT */
+ }//if
+ trscElemlens = trscContainerlen - 2;
+ if (fragrecptr.p->keyLength != 0) {
+ jam();
+ trscElemlen = (1 + fragrecptr.p->keyLength) + fragrecptr.p->localkeylen; /* LENGTH OF THE ELEMENT */
+ } else {
+ jam();
+ trscElemlen = (1 + ZACTIVE_LONG_KEY_LEN) + fragrecptr.p->localkeylen; /* LENGTH OF THE ELEMENT */
+ }//if
+ if (trscIsforward == 1) {
+ jam();
+ trscElementptr = trscContainerptr + ZCON_HEAD_SIZE;
+ trscElemStep = trscElemlen;
+ } else {
+ jam();
+ trscElementptr = trscContainerptr - 1;
+ trscElemStep = 0 - trscElemlen;
+ }//if
+ do {
+ arrGuard(trscElementptr, 2048);
+ const Uint32 eh = rscPageidptr.p->word32[trscElementptr];
+ const Uint32 scanMask = scanPtr.p->scanMask;
+ if (ElementHeader::getUnlocked(eh)) {
+ jam();
+ const Uint32 tmp = ElementHeader::clearScanBit(eh, scanMask);
+ dbgWord32(rscPageidptr, trscElementptr, tmp);
+ rscPageidptr.p->word32[trscElementptr] = tmp;
+ } else {
+ jam();
+ rscOperPtr.i = ElementHeader::getOpPtrI(eh);
+ ptrCheckGuard(rscOperPtr, coprecsize, operationrec);
+ rscOperPtr.p->scanBits &= ~scanMask;
+ }//if
+ trscElemlens = trscElemlens - trscElemlen;
+ trscElementptr = trscElementptr + trscElemStep;
+ } while (trscElemlens > 2);
+ if (trscElemlens != 0) {
+ jam();
+ sendSystemerror(signal);
+ }//if
+}//Dbacc::releaseScanContainer()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_SCAN_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseScanRec(Signal* signal)
+{
+ // Check that all ops this scan has allocated have been
+ // released
+ ndbrequire(scanPtr.p->scanOpsAllocated==0);
+
+ // Check that all locks this scan might have aquired
+ // have been properly released
+ ndbrequire(scanPtr.p->scanLockHeld == 0);
+ ndbrequire(scanPtr.p->scanFirstLockedOp == RNIL);
+ ndbrequire(scanPtr.p->scanLastLockedOp == RNIL);
+
+ // Check that all active operations have been
+ // properly released
+ ndbrequire(scanPtr.p->scanFirstActiveOp == RNIL);
+
+ // Check that all queued operations have been
+ // properly released
+ ndbrequire(scanPtr.p->scanFirstQueuedOp == RNIL);
+ ndbrequire(scanPtr.p->scanLastQueuedOp == RNIL);
+
+ // Put scan record in free list
+ scanPtr.p->scanNextfreerec = cfirstFreeScanRec;
+ scanPtr.p->scanState = ScanRec::SCAN_DISCONNECT;
+ cfirstFreeScanRec = scanPtr.i;
+
+}//Dbacc::releaseScanRec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEARCH_SCAN_CONTAINER */
+/* INPUT: TSSC_CONTAINERLEN */
+/* TSSC_CONTAINERPTR */
+/* TSSC_ISFORWARD */
+/* SSC_PAGEIDPTR */
+/* SCAN_PTR */
+/* OUTPUT: TSSC_IS_LOCKED */
+/* */
+/* DESCRIPTION: SEARCH IN A CONTAINER TO FIND THE NEXT SCAN ELEMENT. */
+/* TO DO THIS THE SCAN BIT OF THE ELEMENT HEADER IS CHECKED. IF */
+/* THIS BIT IS ZERO, IT IS SET TO ONE AND THE ELEMENT IS RETURNED.*/
+/* --------------------------------------------------------------------------------- */
+bool Dbacc::searchScanContainer(Signal* signal)
+{
+ OperationrecPtr sscOperPtr;
+ Uint32 tsscScanBits;
+ Uint32 tsscElemlens;
+ Uint32 tsscElemlen;
+ Uint32 tsscElemStep;
+
+ if (tsscContainerlen < 5) {
+ jam();
+ return false; /* 3 IS THE MINIMUM SIZE OF THE ELEMENT */
+ }//if
+ tsscElemlens = tsscContainerlen - ZCON_HEAD_SIZE;
+ if (fragrecptr.p->keyLength == 0) {
+ jam();
+ tsscElemlen = (ZELEM_HEAD_SIZE + ZACTIVE_LONG_KEY_LEN) + fragrecptr.p->localkeylen;
+ } else {
+ jam();
+ /* LENGTH OF THE ELEMENT */
+ tsscElemlen = (ZELEM_HEAD_SIZE + fragrecptr.p->keyLength) + fragrecptr.p->localkeylen;
+ }//if
+ /* LENGTH OF THE ELEMENT */
+ if (tsscIsforward == 1) {
+ jam();
+ tsscElementptr = tsscContainerptr + ZCON_HEAD_SIZE;
+ tsscElemStep = tsscElemlen;
+ } else {
+ jam();
+ tsscElementptr = tsscContainerptr - 1;
+ tsscElemStep = 0 - tsscElemlen;
+ }//if
+ SCANELEMENTLOOP001:
+ arrGuard(tsscElementptr, 2048);
+ const Uint32 eh = sscPageidptr.p->word32[tsscElementptr];
+ tsscIsLocked = ElementHeader::getLocked(eh);
+ if (!tsscIsLocked){
+ jam();
+ tsscScanBits = ElementHeader::getScanBits(eh);
+ if ((scanPtr.p->scanMask & tsscScanBits) == 0) {
+ jam();
+ const Uint32 tmp = ElementHeader::setScanBit(eh, scanPtr.p->scanMask);
+ dbgWord32(sscPageidptr, tsscElementptr, tmp);
+ sscPageidptr.p->word32[tsscElementptr] = tmp;
+ return true;
+ }//if
+ } else {
+ jam();
+ sscOperPtr.i = ElementHeader::getOpPtrI(eh);
+ ptrCheckGuard(sscOperPtr, coprecsize, operationrec);
+ if ((sscOperPtr.p->scanBits & scanPtr.p->scanMask) == 0) {
+ jam();
+ sscOperPtr.p->scanBits |= scanPtr.p->scanMask;
+ return true;
+ }//if
+ }//if
+ /* THE ELEMENT IS ALREADY SENT. */
+ /* SEARCH FOR NEXT ONE */
+ tsscElemlens = tsscElemlens - tsscElemlen;
+ if (tsscElemlens > 2) {
+ jam();
+ tsscElementptr = tsscElementptr + tsscElemStep;
+ goto SCANELEMENTLOOP001;
+ }//if
+ return false;
+}//Dbacc::searchScanContainer()
+
+/* --------------------------------------------------------------------------------- */
+/* SEND THE RESPONSE NEXT_SCANCONF AND POSSIBLE KEYINFO SIGNALS AS WELL. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::sendNextScanConf(Signal* signal)
+{
+ scanPtr.p->scanTimer = scanPtr.p->scanContinuebCounter;
+ Uint32 blockNo = refToBlock(scanPtr.p->scanUserblockref);
+ if (!scanPtr.p->scanKeyinfoFlag){
+ jam();
+ /** ---------------------------------------------------------------------
+ * LQH WILL NOT HAVE ANY USE OF THE TUPLE KEY LENGTH IN THIS CASE AND
+ * SO WE DO NOT PROVIDE IT. IN THIS CASE THESE VALUES ARE UNDEFINED.
+ * ---------------------------------------------------------------------- */
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = operationRecPtr.i;
+ signal->theData[2] = operationRecPtr.p->fid;
+ signal->theData[3] = operationRecPtr.p->localdata[0];
+ signal->theData[4] = operationRecPtr.p->localdata[1];
+ signal->theData[5] = fragrecptr.p->localkeylen;
+ EXECUTE_DIRECT(blockNo, GSN_NEXT_SCANCONF, signal, 6);
+ return;
+ }//if
+
+ fragrecptr.i = operationRecPtr.p->fragptr;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ if (fragrecptr.p->keyLength != 0) {
+ jam();
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = operationRecPtr.i;
+ signal->theData[2] = operationRecPtr.p->fid;
+ signal->theData[3] = operationRecPtr.p->localdata[0];
+ signal->theData[4] = operationRecPtr.p->localdata[1];
+ signal->theData[5] = fragrecptr.p->localkeylen;
+ signal->theData[6] = fragrecptr.p->keyLength;
+ signal->theData[7] = operationRecPtr.p->keydata[0];
+ signal->theData[8] = operationRecPtr.p->keydata[1];
+ signal->theData[9] = operationRecPtr.p->keydata[2];
+ signal->theData[10] = operationRecPtr.p->keydata[3];
+ EXECUTE_DIRECT(blockNo, GSN_NEXT_SCANCONF, signal, 11);
+ if (fragrecptr.p->keyLength > ZKEYINKEYREQ) {
+ jam();
+ /* = 4 */
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = operationRecPtr.i;
+ signal->theData[2] = operationRecPtr.p->fid;
+ signal->theData[3] = fragrecptr.p->keyLength - ZKEYINKEYREQ;
+ signal->theData[4] = operationRecPtr.p->keydata[4];
+ signal->theData[5] = operationRecPtr.p->keydata[5];
+ signal->theData[6] = operationRecPtr.p->keydata[6];
+ signal->theData[7] = operationRecPtr.p->keydata[7];
+ EXECUTE_DIRECT(blockNo, GSN_ACC_SCAN_INFO, signal, 8);
+ return;
+ }//if
+ } else {
+ jam();
+ sendScaninfo(signal);
+ return;
+ }//if
+}//Dbacc::sendNextScanConf()
+
+/* --------------------------------------------------------------------------------- */
+/* SEND_SCANINFO */
+/* DESCRIPTION: SCAN AN ELEMENT OF A LONG_KEY_PAGE. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::sendScaninfo(Signal* signal)
+{
+ DirRangePtr ssiOverflowrangeptr;
+ DirectoryarrayPtr ssiOverflowDirptr;
+ Page8Ptr ssiPageptr;
+ Uint32 tssiPageIndex;
+ Uint32 tssiPagedir;
+ Uint32 tssiKeyLen;
+ Uint32 tssiStartIndex;
+ Uint32 tssiIndexValue;
+ Uint32 tssiTmp;
+
+ Uint32 blockNo = refToBlock(scanPtr.p->scanUserblockref);
+
+ tssiPageIndex = operationRecPtr.p->keydata[0] & 0x3ff;
+ tssiPagedir = operationRecPtr.p->keydata[0] >> 10;
+ ssiOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ ptrCheckGuard(ssiOverflowrangeptr, cdirrangesize, dirRange);
+ arrGuard((tssiPagedir >> 8), 256);
+ ssiOverflowDirptr.i = ssiOverflowrangeptr.p->dirArray[tssiPagedir >> 8];
+ ptrCheckGuard(ssiOverflowDirptr, cdirarraysize, directoryarray);
+ ssiPageptr.i = ssiOverflowDirptr.p->pagep[tssiPagedir & 0xff];
+ ptrCheckGuard(ssiPageptr, cpagesize, page8);
+ arrGuard(ZWORDS_IN_PAGE - tssiPageIndex, 2048);
+ tssiIndexValue = ssiPageptr.p->word32[ZWORDS_IN_PAGE - tssiPageIndex];
+ tssiStartIndex = tssiIndexValue & 0xffff;
+ tssiKeyLen = tssiIndexValue >> 16;
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = operationRecPtr.i;
+ signal->theData[2] = operationRecPtr.p->fid;
+ signal->theData[3] = operationRecPtr.p->localdata[0];
+ signal->theData[4] = operationRecPtr.p->localdata[1];
+ signal->theData[5] = fragrecptr.p->localkeylen;
+ signal->theData[6] = tssiKeyLen;
+ arrGuard(tssiStartIndex + 3, 2048);
+ signal->theData[7] = ssiPageptr.p->word32[tssiStartIndex];
+ signal->theData[8] = ssiPageptr.p->word32[tssiStartIndex + 1];
+ signal->theData[9] = ssiPageptr.p->word32[tssiStartIndex + 2];
+ signal->theData[10] = ssiPageptr.p->word32[tssiStartIndex + 3];
+ EXECUTE_DIRECT(blockNo, GSN_NEXT_SCANCONF, signal, 11);
+ if (tssiKeyLen > 4) {
+ tssiKeyLen = tssiKeyLen - 4;
+ tssiStartIndex = tssiStartIndex + 4;
+ SSI_LOOP_10:
+ jamEntry();
+ if (tssiKeyLen > ZMAXSCANSIGNALLEN) {
+ jam();
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = operationRecPtr.i;
+ signal->theData[2] = operationRecPtr.p->fid;
+ signal->theData[3] = ZMAXSCANSIGNALLEN;
+ arrGuard(tssiStartIndex + 19, 2048);
+ signal->theData[4] = ssiPageptr.p->word32[tssiStartIndex];
+ signal->theData[5] = ssiPageptr.p->word32[tssiStartIndex + 1];
+ signal->theData[6] = ssiPageptr.p->word32[tssiStartIndex + 2];
+ signal->theData[7] = ssiPageptr.p->word32[tssiStartIndex + 3];
+ signal->theData[8] = ssiPageptr.p->word32[tssiStartIndex + 4];
+ signal->theData[9] = ssiPageptr.p->word32[tssiStartIndex + 5];
+ signal->theData[10] = ssiPageptr.p->word32[tssiStartIndex + 6];
+ signal->theData[11] = ssiPageptr.p->word32[tssiStartIndex + 7];
+ signal->theData[12] = ssiPageptr.p->word32[tssiStartIndex + 8];
+ signal->theData[13] = ssiPageptr.p->word32[tssiStartIndex + 9];
+ signal->theData[14] = ssiPageptr.p->word32[tssiStartIndex + 10];
+ signal->theData[15] = ssiPageptr.p->word32[tssiStartIndex + 11];
+ signal->theData[16] = ssiPageptr.p->word32[tssiStartIndex + 12];
+ signal->theData[17] = ssiPageptr.p->word32[tssiStartIndex + 13];
+ signal->theData[18] = ssiPageptr.p->word32[tssiStartIndex + 14];
+ signal->theData[19] = ssiPageptr.p->word32[tssiStartIndex + 15];
+ signal->theData[20] = ssiPageptr.p->word32[tssiStartIndex + 16];
+ signal->theData[21] = ssiPageptr.p->word32[tssiStartIndex + 17];
+ signal->theData[22] = ssiPageptr.p->word32[tssiStartIndex + 18];
+ signal->theData[23] = ssiPageptr.p->word32[tssiStartIndex + 19];
+ EXECUTE_DIRECT(blockNo, GSN_ACC_SCAN_INFO24, signal, 24);
+ tssiStartIndex = tssiStartIndex + ZMAXSCANSIGNALLEN;
+ tssiKeyLen = tssiKeyLen - ZMAXSCANSIGNALLEN;
+ goto SSI_LOOP_10;
+ } else {
+ jam();
+ ndbrequire((tssiStartIndex + tssiKeyLen) <= 2048);
+ for (tssiTmp = 0; tssiTmp < tssiKeyLen; tssiTmp++) {
+ ckeys[tssiTmp] = ssiPageptr.p->word32[tssiStartIndex + tssiTmp];
+ }//for
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = operationRecPtr.i;
+ signal->theData[2] = operationRecPtr.p->fid;
+ /* LOCAL FRAGMENT IDENTITY */
+ signal->theData[3] = tssiKeyLen;
+ signal->theData[4] = ckeys[0];
+ signal->theData[5] = ckeys[1];
+ signal->theData[6] = ckeys[2];
+ signal->theData[7] = ckeys[3];
+ signal->theData[8] = ckeys[4];
+ signal->theData[9] = ckeys[5];
+ signal->theData[10] = ckeys[6];
+ signal->theData[11] = ckeys[7];
+ signal->theData[12] = ckeys[8];
+ signal->theData[13] = ckeys[9];
+ signal->theData[14] = ckeys[10];
+ signal->theData[15] = ckeys[11];
+ signal->theData[16] = ckeys[12];
+ signal->theData[17] = ckeys[13];
+ signal->theData[18] = ckeys[14];
+ signal->theData[19] = ckeys[15];
+ signal->theData[20] = ckeys[16];
+ signal->theData[21] = ckeys[17];
+ signal->theData[22] = ckeys[18];
+ signal->theData[23] = ckeys[19];
+ EXECUTE_DIRECT(blockNo, GSN_ACC_SCAN_INFO24, signal, 24);
+ }//if
+ }//if
+}//Dbacc::sendScaninfo()
+
+/*---------------------------------------------------------------------------
+ * sendScanHbRep
+ * Description: Using Dispatcher::execute() to send a heartbeat to DBTC
+ * from DBLQH telling the scan is alive. We use the sendScanHbRep()
+ * in DBLQH, this needs to be done here in DBACC since it can take
+ * a while before LQH receives an answer the normal way from ACC.
+ *--------------------------------------------------------------------------*/
+void Dbacc::sendScanHbRep(Signal* signal, Uint32 scanPtrIndex)
+{
+ scanPtr.i = scanPtrIndex;
+ ptrCheckGuard(scanPtr, cscanRecSize, scanRec);
+
+ // If the timer status is on we continue with a new heartbeat in one second,
+ // else the loop stops and we will not send a new CONTINUEB
+ if (scanPtr.p->scanTimer != 0){
+ if (scanPtr.p->scanTimer == scanPtr.p->scanContinuebCounter){
+ jam();
+ ndbrequire(scanPtr.p->scanState != ScanRec::SCAN_DISCONNECT);
+
+ signal->theData[0] = scanPtr.p->scanUserptr;
+ signal->theData[1] = scanPtr.p->scanTrid1;
+ signal->theData[2] = scanPtr.p->scanTrid2;
+ EXECUTE_DIRECT(DBLQH, GSN_SCAN_HBREP, signal, 3);
+ jamEntry();
+ }//if
+ scanPtr.p->scanContinuebCounter++;
+ signal->theData[0] = ZSEND_SCAN_HBREP;
+ signal->theData[1] = scanPtr.i;
+ sendSignalWithDelay(cownBlockref, GSN_CONTINUEB, signal, 100, 2);
+ } else {
+ jam();
+ scanPtr.p->scanContinuebCounter = 0;
+ }//if
+}//Dbacc::sendScanHbRep()
+
+/* --------------------------------------------------------------------------------- */
+/* SETLOCK */
+/* DESCRIPTION:SETS LOCK ON AN ELEMENT. INFORMATION ABOUT THE ELEMENT IS */
+/* SAVED IN THE ELEMENT HEAD.A COPY OF THIS INFORMATION WILL */
+/* BE PUT IN THE OPERATION RECORD. A FIELD IN THE HEADER OF */
+/* THE ELEMENT POINTS TO THE OPERATION RECORD. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::setlock(Signal* signal)
+{
+ Uint32 tselTmp1;
+
+ arrGuard(tslElementptr, 2048);
+ tselTmp1 = slPageidptr.p->word32[tslElementptr];
+ operationRecPtr.p->scanBits = ElementHeader::getScanBits(tselTmp1);
+ operationRecPtr.p->hashvaluePart = ElementHeader::getHashValuePart(tselTmp1);
+
+ tselTmp1 = ElementHeader::setLocked(operationRecPtr.i);
+ dbgWord32(slPageidptr, tslElementptr, tselTmp1);
+ slPageidptr.p->word32[tslElementptr] = tselTmp1;
+}//Dbacc::setlock()
+
+/* --------------------------------------------------------------------------------- */
+/* TAKE_OUT_ACTIVE_SCAN_OP */
+/* DESCRIPTION: AN ACTIVE SCAN OPERATION IS BELOGED TO AN ACTIVE LIST OF THE */
+/* SCAN RECORD. BY THIS SUBRUTIN THE LIST IS UPDATED. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::takeOutActiveScanOp(Signal* signal)
+{
+ OperationrecPtr tasOperationRecPtr;
+
+ if (operationRecPtr.p->prevOp != RNIL) {
+ jam();
+ tasOperationRecPtr.i = operationRecPtr.p->prevOp;
+ ptrCheckGuard(tasOperationRecPtr, coprecsize, operationrec);
+ tasOperationRecPtr.p->nextOp = operationRecPtr.p->nextOp;
+ } else {
+ jam();
+ scanPtr.p->scanFirstActiveOp = operationRecPtr.p->nextOp;
+ }//if
+ if (operationRecPtr.p->nextOp != RNIL) {
+ jam();
+ tasOperationRecPtr.i = operationRecPtr.p->nextOp;
+ ptrCheckGuard(tasOperationRecPtr, coprecsize, operationrec);
+ tasOperationRecPtr.p->prevOp = operationRecPtr.p->prevOp;
+ }//if
+}//Dbacc::takeOutActiveScanOp()
+
+/**
+ * takeOutScanLockQueue
+ *
+ * Description: Take out an operation from the doubly linked
+ * lock list on a scan record.
+ *
+ * @note Use putOpScanLockQue to insert a operation in
+ * the list
+ *
+ */
+void Dbacc::takeOutScanLockQueue(Uint32 scanRecIndex)
+{
+ OperationrecPtr tslOperationRecPtr;
+ ScanRecPtr TscanPtr;
+
+ TscanPtr.i = scanRecIndex;
+ ptrCheckGuard(TscanPtr, cscanRecSize, scanRec);
+
+ if (operationRecPtr.p->prevOp != RNIL) {
+ jam();
+ tslOperationRecPtr.i = operationRecPtr.p->prevOp;
+ ptrCheckGuard(tslOperationRecPtr, coprecsize, operationrec);
+ tslOperationRecPtr.p->nextOp = operationRecPtr.p->nextOp;
+ } else {
+ jam();
+ // Check that first are pointing at operation to take out
+ ndbrequire(TscanPtr.p->scanFirstLockedOp==operationRecPtr.i);
+ TscanPtr.p->scanFirstLockedOp = operationRecPtr.p->nextOp;
+ }//if
+ if (operationRecPtr.p->nextOp != RNIL) {
+ jam();
+ tslOperationRecPtr.i = operationRecPtr.p->nextOp;
+ ptrCheckGuard(tslOperationRecPtr, coprecsize, operationrec);
+ tslOperationRecPtr.p->prevOp = operationRecPtr.p->prevOp;
+ } else {
+ jam();
+ // Check that last are pointing at operation to take out
+ ndbrequire(TscanPtr.p->scanLastLockedOp==operationRecPtr.i);
+ TscanPtr.p->scanLastLockedOp = operationRecPtr.p->prevOp;
+ }//if
+ TscanPtr.p->scanLockHeld--;
+
+#ifdef VM_TRACE
+ // DEBUG CODE
+ // Check that there are as many operations in the lockqueue as
+ // scanLockHeld indicates
+ OperationrecPtr tmpOp;
+ int numLockedOps = 0;
+ tmpOp.i = TscanPtr.p->scanFirstLockedOp;
+ while(tmpOp.i != RNIL){
+ numLockedOps++;
+ ptrCheckGuard(tmpOp, coprecsize, operationrec);
+ if (tmpOp.p->nextOp == RNIL)
+ ndbrequire(tmpOp.i == TscanPtr.p->scanLastLockedOp);
+ tmpOp.i = tmpOp.p->nextOp;
+ }
+ ndbrequire(numLockedOps==TscanPtr.p->scanLockHeld);
+#endif
+}//Dbacc::takeOutScanLockQueue()
+
+/* --------------------------------------------------------------------------------- */
+/* TAKE_OUT_READY_SCAN_QUEUE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::takeOutReadyScanQueue(Signal* signal)
+{
+ OperationrecPtr trsOperationRecPtr;
+
+ if (operationRecPtr.p->prevOp != RNIL) {
+ jam();
+ trsOperationRecPtr.i = operationRecPtr.p->prevOp;
+ ptrCheckGuard(trsOperationRecPtr, coprecsize, operationrec);
+ trsOperationRecPtr.p->nextOp = operationRecPtr.p->nextOp;
+ } else {
+ jam();
+ scanPtr.p->scanFirstQueuedOp = operationRecPtr.p->nextOp;
+ }//if
+ if (operationRecPtr.p->nextOp != RNIL) {
+ jam();
+ trsOperationRecPtr.i = operationRecPtr.p->nextOp;
+ ptrCheckGuard(trsOperationRecPtr, coprecsize, operationrec);
+ trsOperationRecPtr.p->prevOp = operationRecPtr.p->prevOp;
+ } else {
+ jam();
+ scanPtr.p->scanLastQueuedOp = operationRecPtr.p->nextOp;
+ }//if
+}//Dbacc::takeOutReadyScanQueue()
+
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+/* */
+/* END OF SCAN MODULE */
+/* */
+/* --------------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------------- */
+
+bool Dbacc::getrootfragmentrec(Signal* signal, RootfragmentrecPtr& rootPtr, Uint32 fid)
+{
+ for (Uint32 i = 0; i < NO_OF_FRAG_PER_NODE; i++) {
+ jam();
+ if (tabptr.p->fragholder[i] == fid) {
+ jam();
+ rootPtr.i = tabptr.p->fragptrholder[i];
+ ptrCheckGuard(rootPtr, crootfragmentsize, rootfragmentrec);
+ return true;
+ }//if
+ }//for
+ return false;
+}//Dbacc::getrootfragmentrec()
+
+/* --------------------------------------------------------------------------------- */
+/* INIT_FS_OP_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initFsOpRec(Signal* signal)
+{
+ fsOpptr.p->fsOpfragrecPtr = fragrecptr.i;
+ fsOpptr.p->fsConptr = fsConnectptr.i;
+}//Dbacc::initFsOpRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INIT_LCP_CONN_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initLcpConnRec(Signal* signal)
+{
+ lcpConnectptr.p->lcpUserblockref = tuserblockref;
+ lcpConnectptr.p->lcpUserptr = tuserptr;
+ lcpConnectptr.p->noOfLcpConf = 0; /* NO OF RETUREND CONF SIGNALS */
+ lcpConnectptr.p->syncUndopageState = WAIT_NOTHING;
+}//Dbacc::initLcpConnRec()
+
+/* --------------------------------------------------------------------------------- */
+/* INIT_OVERPAGE */
+/* INPUT. IOP_PAGEPTR, POINTER TO AN OVERFLOW PAGE RECORD */
+/* DESCRIPTION: CONTAINERS AND FREE LISTS OF THE PAGE, GET INITIALE VALUE */
+/* ACCORDING TO LH3 AND PAGE STRUCTOR DESCRIPTION OF NDBACC BLOCK */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initOverpage(Signal* signal)
+{
+ Uint32 tiopTmp;
+ Uint32 tiopPrevFree;
+ Uint32 tiopNextFree;
+
+ for (tiopIndex = 0; tiopIndex <= 2047; tiopIndex++) {
+ iopPageptr.p->word32[tiopIndex] = 0;
+ }//for
+ iopPageptr.p->word32[ZPOS_OVERFLOWREC] = iopOverflowRecPtr.i;
+ iopPageptr.p->word32[ZPOS_CHECKSUM] = 0;
+ iopPageptr.p->word32[ZPOS_PAGE_ID] = tiopPageId;
+ iopPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] = 0;
+ tiopTmp = ZEMPTYLIST;
+ tiopTmp = (tiopTmp << 16) + (tiopTmp << 23);
+ iopPageptr.p->word32[ZPOS_EMPTY_LIST] = tiopTmp + (1 << ZPOS_PAGE_TYPE_BIT);
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE PREVIOUS PART OF DOUBLY LINKED LIST FOR LEFT CONTAINERS. */
+ /* --------------------------------------------------------------------------------- */
+ tiopIndex = ZHEAD_SIZE + 1;
+ iopPageptr.p->word32[tiopIndex] = ZEMPTYLIST;
+ for (tiopPrevFree = 0; tiopPrevFree <= ZEMPTYLIST - 2; tiopPrevFree++) {
+ tiopIndex = tiopIndex + ZBUF_SIZE;
+ iopPageptr.p->word32[tiopIndex] = tiopPrevFree;
+ }//for
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE NEXT PART OF DOUBLY LINKED LIST FOR LEFT CONTAINERS. */
+ /* --------------------------------------------------------------------------------- */
+ tiopIndex = ZHEAD_SIZE;
+ for (tiopNextFree = 1; tiopNextFree <= ZEMPTYLIST - 1; tiopNextFree++) {
+ iopPageptr.p->word32[tiopIndex] = tiopNextFree;
+ tiopIndex = tiopIndex + ZBUF_SIZE;
+ }//for
+ iopPageptr.p->word32[tiopIndex] = ZEMPTYLIST; /* LEFT_LIST IS UPDATED */
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE PREVIOUS PART OF DOUBLY LINKED LIST FOR RIGHT CONTAINERS. */
+ /* --------------------------------------------------------------------------------- */
+ tiopIndex = (ZBUF_SIZE + ZHEAD_SIZE) - 1;
+ iopPageptr.p->word32[tiopIndex] = ZEMPTYLIST;
+ for (tiopPrevFree = 0; tiopPrevFree <= ZEMPTYLIST - 2; tiopPrevFree++) {
+ tiopIndex = tiopIndex + ZBUF_SIZE;
+ iopPageptr.p->word32[tiopIndex] = tiopPrevFree;
+ }//for
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE NEXT PART OF DOUBLY LINKED LIST FOR RIGHT CONTAINERS. */
+ /* --------------------------------------------------------------------------------- */
+ tiopIndex = (ZBUF_SIZE + ZHEAD_SIZE) - 2;
+ for (tiopNextFree = 1; tiopNextFree <= ZEMPTYLIST - 1; tiopNextFree++) {
+ iopPageptr.p->word32[tiopIndex] = tiopNextFree;
+ tiopIndex = tiopIndex + ZBUF_SIZE;
+ }//for
+ iopPageptr.p->word32[tiopIndex] = ZEMPTYLIST; /* RIGHT_LIST IS UPDATED */
+}//Dbacc::initOverpage()
+
+/* --------------------------------------------------------------------------------- */
+/* INIT_PAGE */
+/* INPUT. INP_PAGEPTR, POINTER TO A PAGE RECORD */
+/* DESCRIPTION: CONTAINERS AND FREE LISTS OF THE PAGE, GET INITIALE VALUE */
+/* ACCORDING TO LH3 AND PAGE STRUCTOR DISACRIPTION OF NDBACC BLOCK */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::initPage(Signal* signal)
+{
+ Uint32 tinpTmp1;
+ Uint32 tinpIndex;
+ Uint32 tinpTmp;
+ Uint32 tinpPrevFree;
+ Uint32 tinpNextFree;
+
+ for (tiopIndex = 0; tiopIndex <= 2047; tiopIndex++) {
+ inpPageptr.p->word32[tiopIndex] = 0;
+ }//for
+ /* --------------------------------------------------------------------------------- */
+ /* SET PAGE ID FOR USE OF CHECKPOINTER. */
+ /* PREPARE CONTAINER HEADERS INDICATING EMPTY CONTAINERS WITHOUT NEXT. */
+ /* --------------------------------------------------------------------------------- */
+ inpPageptr.p->word32[ZPOS_PAGE_ID] = tipPageId;
+ tinpTmp1 = ZCON_HEAD_SIZE;
+ tinpTmp1 = tinpTmp1 << 26;
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE ZNO_CONTAINERS PREDEFINED HEADERS ON LEFT SIZE. */
+ /* --------------------------------------------------------------------------------- */
+ tinpIndex = ZHEAD_SIZE;
+ for (tinpTmp = 0; tinpTmp <= ZNO_CONTAINERS - 1; tinpTmp++) {
+ inpPageptr.p->word32[tinpIndex] = tinpTmp1;
+ tinpIndex = tinpIndex + ZBUF_SIZE;
+ }//for
+ /* WORD32(ZPOS_EMPTY_LIST) DATA STRUCTURE:*/
+ /*--------------------------------------- */
+ /*| PAGE TYPE|LEFT FREE|RIGHT FREE */
+ /*| 1 | LIST | LIST */
+ /*| BIT | 7 BITS | 7 BITS */
+ /*--------------------------------------- */
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE FIRST POINTER TO DOUBLY LINKED LIST OF FREE CONTAINERS. */
+ /* INITIALISE EMPTY LISTS OF USED CONTAINERS. */
+ /* INITIALISE LEFT FREE LIST TO 64 AND RIGHT FREE LIST TO ZERO. */
+ /* ALSO INITIALISE PAGE TYPE TO NOT OVERFLOW PAGE. */
+ /* --------------------------------------------------------------------------------- */
+ tinpTmp = ZEMPTYLIST;
+ tinpTmp = (tinpTmp << 16) + (tinpTmp << 23);
+ tinpTmp = tinpTmp + (ZNO_CONTAINERS << 7);
+ inpPageptr.p->word32[ZPOS_EMPTY_LIST] = tinpTmp;
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE PREVIOUS PART OF DOUBLY LINKED LIST FOR RIGHT CONTAINERS. */
+ /* --------------------------------------------------------------------------------- */
+ tinpIndex = (ZHEAD_SIZE + ZBUF_SIZE) - 1;
+ inpPageptr.p->word32[tinpIndex] = ZEMPTYLIST;
+ for (tinpPrevFree = 0; tinpPrevFree <= ZEMPTYLIST - 2; tinpPrevFree++) {
+ tinpIndex = tinpIndex + ZBUF_SIZE;
+ inpPageptr.p->word32[tinpIndex] = tinpPrevFree;
+ }//for
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE NEXT PART OF DOUBLY LINKED LIST FOR RIGHT CONTAINERS. */
+ /* --------------------------------------------------------------------------------- */
+ tinpIndex = (ZHEAD_SIZE + ZBUF_SIZE) - 2;
+ for (tinpNextFree = 1; tinpNextFree <= ZEMPTYLIST - 1; tinpNextFree++) {
+ inpPageptr.p->word32[tinpIndex] = tinpNextFree;
+ tinpIndex = tinpIndex + ZBUF_SIZE;
+ }//for
+ inpPageptr.p->word32[tinpIndex] = ZEMPTYLIST;
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE PREVIOUS PART OF DOUBLY LINKED LIST FOR LEFT CONTAINERS. */
+ /* THE FIRST ZNO_CONTAINERS ARE NOT PUT INTO FREE LIST SINCE THEY ARE */
+ /* PREDEFINED AS OCCUPIED. */
+ /* --------------------------------------------------------------------------------- */
+ tinpIndex = (ZNO_CONTAINERS * ZBUF_SIZE) + ZHEAD_SIZE;
+ for (tinpNextFree = ZNO_CONTAINERS + 1; tinpNextFree <= ZEMPTYLIST - 1; tinpNextFree++) {
+ inpPageptr.p->word32[tinpIndex] = tinpNextFree;
+ tinpIndex = tinpIndex + ZBUF_SIZE;
+ }//for
+ inpPageptr.p->word32[tinpIndex] = ZEMPTYLIST;
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE NEXT PART OF DOUBLY LINKED LIST FOR LEFT CONTAINERS. */
+ /* THE FIRST ZNO_CONTAINERS ARE NOT PUT INTO FREE LIST SINCE THEY ARE */
+ /* PREDEFINED AS OCCUPIED. */
+ /* --------------------------------------------------------------------------------- */
+ tinpIndex = ((ZNO_CONTAINERS * ZBUF_SIZE) + ZHEAD_SIZE) + 1;
+ inpPageptr.p->word32[tinpIndex] = ZEMPTYLIST;
+ for (tinpPrevFree = ZNO_CONTAINERS; tinpPrevFree <= ZEMPTYLIST - 2; tinpPrevFree++) {
+ tinpIndex = tinpIndex + ZBUF_SIZE;
+ inpPageptr.p->word32[tinpIndex] = tinpPrevFree;
+ }//for
+ /* --------------------------------------------------------------------------------- */
+ /* INITIALISE HEADER POSITIONS NOT CURRENTLY USED AND ENSURE USE OF OVERFLOW */
+ /* RECORD POINTER ON THIS PAGE LEADS TO ERROR. */
+ /* --------------------------------------------------------------------------------- */
+ inpPageptr.p->word32[ZPOS_CHECKSUM] = 0;
+ inpPageptr.p->word32[ZPOS_ALLOC_CONTAINERS] = 0;
+ inpPageptr.p->word32[ZPOS_OVERFLOWREC] = RNIL;
+}//Dbacc::initPage()
+
+/* --------------------------------------------------------------------------------- */
+/* PUT_OP_IN_FRAG_WAIT_QUE */
+/* DESCRIPTION: AN OPERATION WHICH OWNS A LOCK OF AN ELEMENT, IS PUT IN A */
+/* LIST OF THE FRAGMENT. THIS LIST IS USED TO STOP THE QUEUE */
+/* OPERATION DURING CREATE CHECK POINT PROSESS FOR STOP AND */
+/* RESTART OF THE OPERATIONS. */
+/* */
+/* IF CONTINUEB SIGNALS ARE INTRODUCED AFTER STARTING TO EXECUTE ACCKEYREQ WE */
+/* MUST PUT IT IN THIS LIST BEFORE EXITING TO ENSURE THAT WE ARE NOT BEING */
+/* LOCKED AFTER THAT LQH HAS RECEIVED ALL LCP_HOLDOP'S. THEN THE LCP WILL NEVER*/
+/* PROCEED. WE ALSO PUT IT INTO THIS LIST WHEN WAITING FOR LONG KEYS. THIS IS */
+/* ONLY NEEDED IF SIGNALS CAN ENTER BETWEEN THE KEYDATA CARRYING SIGNALS. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::putOpInFragWaitQue(Signal* signal)
+{
+ OperationrecPtr tpiwOperRecPtr;
+
+ if (operationRecPtr.p->operation != ZSCAN_OP) {
+ if (fragrecptr.p->firstWaitInQueOp == RNIL) {
+ jam();
+ fragrecptr.p->firstWaitInQueOp = operationRecPtr.i;
+ } else {
+ jam();
+ tpiwOperRecPtr.i = fragrecptr.p->lastWaitInQueOp;
+ ptrCheckGuard(tpiwOperRecPtr, coprecsize, operationrec);
+ tpiwOperRecPtr.p->nextQueOp = operationRecPtr.i;
+ }//if
+ operationRecPtr.p->opState = WAIT_IN_QUEUE;
+ operationRecPtr.p->nextQueOp = RNIL;
+ operationRecPtr.p->prevQueOp = fragrecptr.p->lastWaitInQueOp;
+ fragrecptr.p->lastWaitInQueOp = operationRecPtr.i;
+ }//if
+}//Dbacc::putOpInFragWaitQue()
+
+/* --------------------------------------------------------------------------------- */
+/* PUT_OVERFLOW_REC_IN_FRAG */
+/* DESCRIPTION: AN OVERFLOW RECORD WITCH IS USED TO KEEP INFORMATION ABOUT */
+/* OVERFLOW PAGE WILL BE PUT IN A LIST OF OVERFLOW RECORDS IN */
+/* THE FRAGMENT RECORD. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::putOverflowRecInFrag(Signal* signal)
+{
+ OverflowRecordPtr tpifNextOverrecPtr;
+ OverflowRecordPtr tpifPrevOverrecPtr;
+
+ tpifNextOverrecPtr.i = fragrecptr.p->firstOverflowRec;
+ tpifPrevOverrecPtr.i = RNIL;
+ while (tpifNextOverrecPtr.i != RNIL) {
+ ptrCheckGuard(tpifNextOverrecPtr, coverflowrecsize, overflowRecord);
+ if (tpifNextOverrecPtr.p->dirindex < porOverflowRecPtr.p->dirindex) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* PROCEED IN LIST TO THE NEXT IN THE LIST SINCE THE ENTRY HAD A LOWER PAGE ID.*/
+ /* WE WANT TO ENSURE THAT LOWER PAGE ID'S ARE KEPT FULL RATHER THAN THE */
+ /* OPPOSITE TO ENSURE THAT HIGH PAGE ID'S CAN BE REMOVED WHEN SHRINKS ARE */
+ /* PERFORMED. */
+ /* --------------------------------------------------------------------------------- */
+ tpifPrevOverrecPtr = tpifNextOverrecPtr;
+ tpifNextOverrecPtr.i = tpifNextOverrecPtr.p->nextOverRec;
+ } else {
+ jam();
+ ndbrequire(tpifNextOverrecPtr.p->dirindex != porOverflowRecPtr.p->dirindex);
+ /* --------------------------------------------------------------------------------- */
+ /* TRYING TO INSERT THE SAME PAGE TWICE. SYSTEM ERROR. */
+ /* --------------------------------------------------------------------------------- */
+ break;
+ }//if
+ }//while
+ if (tpifNextOverrecPtr.i == RNIL) {
+ jam();
+ fragrecptr.p->lastOverflowRec = porOverflowRecPtr.i;
+ } else {
+ jam();
+ tpifNextOverrecPtr.p->prevOverRec = porOverflowRecPtr.i;
+ }//if
+ if (tpifPrevOverrecPtr.i == RNIL) {
+ jam();
+ fragrecptr.p->firstOverflowRec = porOverflowRecPtr.i;
+ } else {
+ jam();
+ tpifPrevOverrecPtr.p->nextOverRec = porOverflowRecPtr.i;
+ }//if
+ porOverflowRecPtr.p->prevOverRec = tpifPrevOverrecPtr.i;
+ porOverflowRecPtr.p->nextOverRec = tpifNextOverrecPtr.i;
+}//Dbacc::putOverflowRecInFrag()
+
+/* --------------------------------------------------------------------------------- */
+/* PUT_REC_IN_FREE_OVERDIR */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::putRecInFreeOverdir(Signal* signal)
+{
+ OverflowRecordPtr tpfoNextOverrecPtr;
+ OverflowRecordPtr tpfoPrevOverrecPtr;
+
+ tpfoNextOverrecPtr.i = fragrecptr.p->firstFreeDirindexRec;
+ tpfoPrevOverrecPtr.i = RNIL;
+ while (tpfoNextOverrecPtr.i != RNIL) {
+ ptrCheckGuard(tpfoNextOverrecPtr, coverflowrecsize, overflowRecord);
+ if (tpfoNextOverrecPtr.p->dirindex < priOverflowRecPtr.p->dirindex) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* PROCEED IN LIST TO THE NEXT IN THE LIST SINCE THE ENTRY HAD A LOWER PAGE ID.*/
+ /* WE WANT TO ENSURE THAT LOWER PAGE ID'S ARE KEPT FULL RATHER THAN THE */
+ /* OPPOSITE TO ENSURE THAT HIGH PAGE ID'S CAN BE REMOVED WHEN SHRINKS ARE */
+ /* PERFORMED. */
+ /* --------------------------------------------------------------------------------- */
+ tpfoPrevOverrecPtr = tpfoNextOverrecPtr;
+ tpfoNextOverrecPtr.i = tpfoNextOverrecPtr.p->nextOverList;
+ } else {
+ jam();
+ ndbrequire(tpfoNextOverrecPtr.p->dirindex != priOverflowRecPtr.p->dirindex);
+ /* --------------------------------------------------------------------------------- */
+ /* ENSURE WE ARE NOT TRYING TO INSERT THE SAME PAGE TWICE. */
+ /* --------------------------------------------------------------------------------- */
+ break;
+ }//if
+ }//while
+ if (tpfoNextOverrecPtr.i != RNIL) {
+ jam();
+ tpfoNextOverrecPtr.p->prevOverList = priOverflowRecPtr.i;
+ }//if
+ if (tpfoPrevOverrecPtr.i == RNIL) {
+ jam();
+ fragrecptr.p->firstFreeDirindexRec = priOverflowRecPtr.i;
+ } else {
+ jam();
+ tpfoPrevOverrecPtr.p->nextOverList = priOverflowRecPtr.i;
+ }//if
+ priOverflowRecPtr.p->prevOverList = tpfoPrevOverrecPtr.i;
+ priOverflowRecPtr.p->nextOverList = tpfoNextOverrecPtr.i;
+}//Dbacc::putRecInFreeOverdir()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_DIRECTORY */
+/* --------------------------------------- ----------------------------------------- */
+void Dbacc::releaseDirectory(Signal* signal)
+{
+ ptrCheckGuard(rdDirptr, cdirarraysize, directoryarray);
+ rdDirptr.p->pagep[0] = cfirstfreedir;
+ cfirstfreedir = rdDirptr.i;
+}//Dbacc::releaseDirectory()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_DIRRANGE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseDirrange(Signal* signal)
+{
+ ptrCheckGuard(rdDirRangePtr, cdirrangesize, dirRange);
+ rdDirRangePtr.p->dirArray[0] = cfirstfreeDirrange;
+ cfirstfreeDirrange = rdDirRangePtr.i;
+}//Dbacc::releaseDirrange()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_FS_CONN_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseFsConnRec(Signal* signal)
+{
+ fsConnectptr.p->fsNext = cfsFirstfreeconnect;
+ cfsFirstfreeconnect = fsConnectptr.i;
+ fsConnectptr.p->fsState = WAIT_NOTHING;
+}//Dbacc::releaseFsConnRec()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_FS_OP_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseFsOpRec(Signal* signal)
+{
+ fsOpptr.p->fsOpnext = cfsFirstfreeop;
+ cfsFirstfreeop = fsOpptr.i;
+ fsOpptr.p->fsOpstate = WAIT_NOTHING;
+}//Dbacc::releaseFsOpRec()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_LCP_CONNECT_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseLcpConnectRec(Signal* signal)
+{
+ lcpConnectptr.p->lcpstate = LCP_FREE;
+ lcpConnectptr.p->nextLcpConn = cfirstfreelcpConnect;
+ lcpConnectptr.p->lcpstate = LCP_FREE;
+ cfirstfreelcpConnect = lcpConnectptr.i;
+}//Dbacc::releaseLcpConnectRec()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE OP RECORD */
+/* PUT A FREE OPERATION IN A FREE LIST OF THE OPERATIONS */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseOpRec(Signal* signal)
+{
+#ifdef VM_TRACE
+ // DEBUG CODE
+ // Check that the operation to be released isn't
+ // already in the list of free operations
+ // Since this code loops through the entire list of free operations
+ // it's only enabled in VM_TRACE mode
+ OperationrecPtr opRecPtr;
+ bool opInList = false;
+ opRecPtr.i = cfreeopRec;
+ while (opRecPtr.i != RNIL){
+ if (opRecPtr.i == operationRecPtr.i){
+ opInList = true;
+ break;
+ }
+ ptrCheckGuard(opRecPtr, coprecsize, operationrec);
+ opRecPtr.i = opRecPtr.p->nextOp;
+ }
+ ndbrequire(opInList == false);
+#endif
+ ndbrequire(operationRecPtr.p->lockOwner == ZFALSE);
+
+ operationRecPtr.p->nextOp = cfreeopRec;
+ cfreeopRec = operationRecPtr.i; /* UPDATE FREE LIST OF OP RECORDS */
+ operationRecPtr.p->prevOp = RNIL;
+ operationRecPtr.p->opState = FREE_OP;
+ operationRecPtr.p->transactionstate = IDLE;
+ operationRecPtr.p->operation = ZUNDEFINED_OP;
+}//Dbacc::releaseOpRec()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_OVERFLOW_REC */
+/* PUT A FREE OVERFLOW REC IN A FREE LIST OF THE OVERFLOW RECORDS */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseOverflowRec(Signal* signal)
+{
+ rorOverflowRecPtr.p->nextfreeoverrec = cfirstfreeoverrec;
+ cfirstfreeoverrec = rorOverflowRecPtr.i;
+}//Dbacc::releaseOverflowRec()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_OVERPAGE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseOverpage(Signal* signal)
+{
+ DirRangePtr ropOverflowrangeptr;
+ DirectoryarrayPtr ropOverflowDirptr;
+ OverflowRecordPtr ropOverflowRecPtr;
+ OverflowRecordPtr tuodOverflowRecPtr;
+ Uint32 tropTmp;
+ Uint32 tropTmp1;
+ Uint32 tropTmp2;
+
+ ropOverflowRecPtr.i = ropPageptr.p->word32[ZPOS_OVERFLOWREC];
+ ndbrequire(ropOverflowRecPtr.i != RNIL);
+ /* THE OVERFLOW REC WILL BE TAKEN OUT OF THE */
+ /* FREELIST OF OVERFLOW PAGE WITH FREE */
+ /* CONTAINER AND WILL BE PUT IN THE FREE LIST */
+ /* OF THE FREE DIRECTORY INDEXES. */
+ if ((fragrecptr.p->lastOverflowRec == ropOverflowRecPtr.i) &&
+ (fragrecptr.p->firstOverflowRec == ropOverflowRecPtr.i)) {
+ jam();
+ return; /* THERE IS ONLY ONE OVERFLOW PAGE */
+ }//if
+ if ((fragrecptr.p->createLcp == ZTRUE) &&
+ (fragrecptr.p->lcpMaxOverDirIndex > ropPageptr.p->word32[ZPOS_PAGE_ID])) {
+ /* --------------------------------------------------------------------------------- */
+ /* THE PAGE PARTICIPATES IN THE LOCAL CHECKPOINT. */
+ /* --------------------------------------------------------------------------------- */
+ if (fragrecptr.p->fragState == LCP_SEND_PAGES) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* THE PAGE PARTICIPATES IN THE LOCAL CHECKPOINT AND THE WRITE TO DISK HAS NOT */
+ /* YET BEEN COMPLETED. WE MUST KEEP IT A WHILE LONGER SINCE AN EMPTY PAGE IS */
+ /* NOT EQUIVALENT TO AN INITIALISED PAGE SINCE THE FREE LISTS CAN DIFFER. */
+ /* --------------------------------------------------------------------------------- */
+ return;
+ } else {
+ if ((fragrecptr.p->fragState == LCP_SEND_OVER_PAGES) &&
+ (fragrecptr.p->lcpDirIndex <= ropPageptr.p->word32[ZPOS_PAGE_ID])) {
+ jam();
+ /* --------------------------------------------------------------------------------- */
+ /* SEE COMMENT ABOVE */
+ /* --------------------------------------------------------------------------------- */
+ return;
+ }//if
+ }//if
+ }//if
+#if kalle
+ logicalPage = 0;
+
+ i = fragrecptr.p->directory;
+ p = dirRange.getPtr(i);
+
+ i1 = logicalPage >> 8;
+ i2 = logicalPage & 0xFF;
+
+ ndbrequire(i1 < 256);
+
+ i = p->dirArray[i1];
+ p = directoryarray.getPtr(i);
+
+ physicPageId = p->pagep[i2];
+ physicPageP = page8.getPtr(physicPageId);
+
+ p->pagep[i2] = RNIL;
+ rpPageptr = { physicPageId, physicPageP };
+ releasePage(signal);
+
+#endif
+
+ /* --------------------------------------------------------------------------------- */
+ /* IT WAS OK TO RELEASE THE PAGE. */
+ /* --------------------------------------------------------------------------------- */
+ ptrCheckGuard(ropOverflowRecPtr, coverflowrecsize, overflowRecord);
+ tfoOverflowRecPtr = ropOverflowRecPtr;
+ takeRecOutOfFreeOverpage(signal);
+ ropOverflowRecPtr.p->overpage = RNIL;
+ priOverflowRecPtr = ropOverflowRecPtr;
+ putRecInFreeOverdir(signal);
+ tropTmp = ropPageptr.p->word32[ZPOS_PAGE_ID];
+ ropOverflowrangeptr.i = fragrecptr.p->overflowdir;
+ tropTmp1 = tropTmp >> 8;
+ tropTmp2 = tropTmp & 0xff;
+ ptrCheckGuard(ropOverflowrangeptr, cdirrangesize, dirRange);
+ arrGuard(tropTmp1, 256);
+ ropOverflowDirptr.i = ropOverflowrangeptr.p->dirArray[tropTmp1];
+ ptrCheckGuard(ropOverflowDirptr, cdirarraysize, directoryarray);
+ ropOverflowDirptr.p->pagep[tropTmp2] = RNIL;
+ rpPageptr = ropPageptr;
+ releasePage(signal);
+ if (ropOverflowRecPtr.p->dirindex != (fragrecptr.p->lastOverIndex - 1)) {
+ jam();
+ return;
+ }//if
+ /* --------------------------------------------------------------------------------- */
+ /* THE LAST PAGE IN THE DIRECTORY WAS RELEASED IT IS NOW NECESSARY TO REMOVE */
+ /* ALL RELEASED OVERFLOW DIRECTORIES AT THE END OF THE LIST. */
+ /* --------------------------------------------------------------------------------- */
+ do {
+ fragrecptr.p->lastOverIndex--;
+ if (tropTmp2 == 0) {
+ jam();
+ ndbrequire(tropTmp1 != 0);
+ ropOverflowrangeptr.p->dirArray[tropTmp1] = RNIL;
+ rdDirptr.i = ropOverflowDirptr.i;
+ releaseDirectory(signal);
+ tropTmp1--;
+ tropTmp2 = 255;
+ } else {
+ jam();
+ tropTmp2--;
+ }//if
+ ropOverflowDirptr.i = ropOverflowrangeptr.p->dirArray[tropTmp1];
+ ptrCheckGuard(ropOverflowDirptr, cdirarraysize, directoryarray);
+ } while (ropOverflowDirptr.p->pagep[tropTmp2] == RNIL);
+ /* --------------------------------------------------------------------------------- */
+ /* RELEASE ANY OVERFLOW RECORDS THAT ARE PART OF THE FREE INDEX LIST WHICH */
+ /* DIRECTORY INDEX NOW HAS BEEN RELEASED. */
+ /* --------------------------------------------------------------------------------- */
+ tuodOverflowRecPtr.i = fragrecptr.p->firstFreeDirindexRec;
+ jam();
+ while (tuodOverflowRecPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(tuodOverflowRecPtr, coverflowrecsize, overflowRecord);
+ if (tuodOverflowRecPtr.p->dirindex >= fragrecptr.p->lastOverIndex) {
+ jam();
+ rorOverflowRecPtr = tuodOverflowRecPtr;
+ troOverflowRecPtr.p = tuodOverflowRecPtr.p;
+ tuodOverflowRecPtr.i = troOverflowRecPtr.p->nextOverList;
+ takeRecOutOfFreeOverdir(signal);
+ releaseOverflowRec(signal);
+ } else {
+ jam();
+ tuodOverflowRecPtr.i = tuodOverflowRecPtr.p->nextOverList;
+ }//if
+ }//while
+}//Dbacc::releaseOverpage()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_PAGE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releasePage(Signal* signal)
+{
+#ifdef VM_TRACE
+ bool inList = false;
+ Uint32 numInList = 0;
+ Page8Ptr tmpPagePtr;
+ tmpPagePtr.i = cfirstfreepage;
+ while (tmpPagePtr.i != RNIL){
+ ptrCheckGuard(tmpPagePtr, cpagesize, page8);
+ if (tmpPagePtr.i == rpPageptr.i){
+ jam(); inList = true;
+ break;
+ }
+ numInList++;
+ tmpPagePtr.i = tmpPagePtr.p->word32[0];
+ }
+ ndbrequire(inList == false);
+ // ndbrequire(numInList == cnoOfAllocatedPages);
+#endif
+ rpPageptr.p->word32[0] = cfirstfreepage;
+ cfirstfreepage = rpPageptr.i;
+ cnoOfAllocatedPages--;
+}//Dbacc::releasePage()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_LCP_PAGE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseLcpPage(Signal* signal)
+{
+ rlpPageptr.p->word32[0] = cfirstfreeLcpPage;
+ cfirstfreeLcpPage = rlpPageptr.i;
+}//Dbacc::releaseLcpPage()
+
+/* --------------------------------------------------------------------------------- */
+/* RELEASE_SR_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::releaseSrRec(Signal* signal)
+{
+ srVersionPtr.p->nextFreeSr = cfirstFreeSrVersionRec;
+ cfirstFreeSrVersionRec = srVersionPtr.i;
+}//Dbacc::releaseSrRec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_DIRECTORY */
+/* DESCRIPTION: A DIRECTORY BLOCK (ZDIRBLOCKSIZE NUMBERS OF DIRECTORY */
+/* RECORDS WILL BE ALLOCATED AND RETURNED. */
+/* SIZE OF DIRECTORY ERROR_CODE, WILL BE RETURNED IF THERE IS NO ANY */
+/* FREE BLOCK */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeDirectory(Signal* signal)
+{
+ Uint32 tsdyIndex;
+
+ if (cfirstfreedir == RNIL) {
+ jam();
+ if (cdirarraysize <= cdirmemory) {
+ jam();
+ tresult = ZDIRSIZE_ERROR;
+ return;
+ } else {
+ jam();
+ sdDirptr.i = cdirmemory;
+ ptrCheckGuard(sdDirptr, cdirarraysize, directoryarray);
+ cdirmemory = cdirmemory + 1;
+ }//if
+ } else {
+ jam();
+ sdDirptr.i = cfirstfreedir;
+ ptrCheckGuard(sdDirptr, cdirarraysize, directoryarray);
+ cfirstfreedir = sdDirptr.p->pagep[0];
+ sdDirptr.p->pagep[0] = RNIL;
+ }//if
+ for (tsdyIndex = 0; tsdyIndex <= 255; tsdyIndex++) {
+ sdDirptr.p->pagep[tsdyIndex] = RNIL;
+ }//for
+}//Dbacc::seizeDirectory()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_DIRRANGE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeDirrange(Signal* signal)
+{
+ Uint32 tsdeIndex;
+
+ newDirRangePtr.i = cfirstfreeDirrange;
+ ptrCheckGuard(newDirRangePtr, cdirrangesize, dirRange);
+ cfirstfreeDirrange = newDirRangePtr.p->dirArray[0];
+ for (tsdeIndex = 0; tsdeIndex <= 255; tsdeIndex++) {
+ newDirRangePtr.p->dirArray[tsdeIndex] = RNIL;
+ }//for
+}//Dbacc::seizeDirrange()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE FRAGREC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeFragrec(Signal* signal)
+{
+ fragrecptr.i = cfirstfreefrag;
+ ptrCheckGuard(fragrecptr, cfragmentsize, fragmentrec);
+ cfirstfreefrag = fragrecptr.p->nextfreefrag;
+ fragrecptr.p->nextfreefrag = RNIL;
+}//Dbacc::seizeFragrec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_FS_CONNECT_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeFsConnectRec(Signal* signal)
+{
+ fsConnectptr.i = cfsFirstfreeconnect;
+ ptrCheckGuard(fsConnectptr, cfsConnectsize, fsConnectrec);
+ cfsFirstfreeconnect = fsConnectptr.p->fsNext;
+ fsConnectptr.p->fsNext = RNIL;
+ fsConnectptr.p->fsState = WAIT_NOTHING;
+}//Dbacc::seizeFsConnectRec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_FS_OP_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeFsOpRec(Signal* signal)
+{
+ fsOpptr.i = cfsFirstfreeop;
+ ptrCheckGuard(fsOpptr, cfsOpsize, fsOprec);
+ cfsFirstfreeop = fsOpptr.p->fsOpnext;
+ fsOpptr.p->fsOpnext = RNIL;
+}//Dbacc::seizeFsOpRec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_LCP_CONNECT_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeLcpConnectRec(Signal* signal)
+{
+ lcpConnectptr.i = cfirstfreelcpConnect;
+ ptrCheckGuard(lcpConnectptr, clcpConnectsize, lcpConnectrec);
+ cfirstfreelcpConnect = lcpConnectptr.p->nextLcpConn;
+ lcpConnectptr.p->nextLcpConn = RNIL;
+}//Dbacc::seizeLcpConnectRec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_OP_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeOpRec(Signal* signal)
+{
+ operationRecPtr.i = cfreeopRec;
+ ptrCheckGuard(operationRecPtr, coprecsize, operationrec);
+ cfreeopRec = operationRecPtr.p->nextOp; /* UPDATE FREE LIST OF OP RECORDS */
+ /* PUTS OPERTION RECORD PTR IN THE LIST */
+ /* OF OPERATION IN CONNECTION RECORD */
+ operationRecPtr.p->nextOp = RNIL;
+}//Dbacc::seizeOpRec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE OVERFLOW RECORD */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeOverRec(Signal* signal) {
+ sorOverflowRecPtr.i = cfirstfreeoverrec;
+ ptrCheckGuard(sorOverflowRecPtr, coverflowrecsize, overflowRecord);
+ cfirstfreeoverrec = sorOverflowRecPtr.p->nextfreeoverrec;
+ sorOverflowRecPtr.p->nextfreeoverrec = RNIL;
+ sorOverflowRecPtr.p->prevOverRec = RNIL;
+ sorOverflowRecPtr.p->nextOverRec = RNIL;
+}//Dbacc::seizeOverRec()
+
+
+/**
+ * A ZPAGESIZE_ERROR has occured, out of index pages
+ * Print some debug info if debug compiled
+ */
+void Dbacc::zpagesize_error(const char* where){
+ DEBUG(where << endl
+ << " ZPAGESIZE_ERROR" << endl
+ << " cfirstfreepage=" << cfirstfreepage << endl
+ << " cfreepage=" <<cfreepage<<endl
+ << " cpagesize=" <<cpagesize<<endl
+ << " cnoOfAllocatedPages="<<cnoOfAllocatedPages);
+}
+
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_PAGE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizePage(Signal* signal)
+{
+ tresult = 0;
+ if (cfirstfreepage == RNIL) {
+ if (cfreepage < cpagesize) {
+ jam();
+ spPageptr.i = cfreepage;
+ ptrCheckGuard(spPageptr, cpagesize, page8);
+ cfreepage++;
+ cnoOfAllocatedPages++;
+ } else {
+ jam();
+ zpagesize_error("Dbacc::seizePage");
+ tresult = ZPAGESIZE_ERROR;
+ }//if
+ } else {
+ jam();
+ spPageptr.i = cfirstfreepage;
+ ptrCheckGuard(spPageptr, cpagesize, page8);
+ cfirstfreepage = spPageptr.p->word32[0];
+ cnoOfAllocatedPages++;
+ }//if
+}//Dbacc::seizePage()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_PAGE */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeLcpPage(Page8Ptr& regPagePtr)
+{
+ regPagePtr.i = cfirstfreeLcpPage;
+ ptrCheckGuard(regPagePtr, cpagesize, page8);
+ cfirstfreeLcpPage = regPagePtr.p->word32[0];
+}//Dbacc::seizeLcpPage()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_ROOTFRAGREC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeRootfragrec(Signal* signal)
+{
+ rootfragrecptr.i = cfirstfreerootfrag;
+ ptrCheckGuard(rootfragrecptr, crootfragmentsize, rootfragmentrec);
+ cfirstfreerootfrag = rootfragrecptr.p->nextroot;
+ rootfragrecptr.p->nextroot = RNIL;
+}//Dbacc::seizeRootfragrec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_SCAN_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeScanRec(Signal* signal)
+{
+ scanPtr.i = cfirstFreeScanRec;
+ ptrCheckGuard(scanPtr, cscanRecSize, scanRec);
+ ndbrequire(scanPtr.p->scanState == ScanRec::SCAN_DISCONNECT);
+ cfirstFreeScanRec = scanPtr.p->scanNextfreerec;
+}//Dbacc::seizeScanRec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEIZE_SR_VERSION_REC */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::seizeSrVerRec(Signal* signal)
+{
+ srVersionPtr.i = cfirstFreeSrVersionRec;
+ ptrCheckGuard(srVersionPtr, csrVersionRecSize, srVersionRec);
+ cfirstFreeSrVersionRec = srVersionPtr.p->nextFreeSr;
+}//Dbacc::seizeSrVerRec()
+
+/* --------------------------------------------------------------------------------- */
+/* SEND_SYSTEMERROR */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::sendSystemerror(Signal* signal)
+{
+ progError(0, 0);
+}//Dbacc::sendSystemerror()
+
+/* --------------------------------------------------------------------------------- */
+/* TAKE_REC_OUT_OF_FREE_OVERDIR */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::takeRecOutOfFreeOverdir(Signal* signal)
+{
+ OverflowRecordPtr tofoOverrecPtr;
+ if (troOverflowRecPtr.p->nextOverList != RNIL) {
+ jam();
+ tofoOverrecPtr.i = troOverflowRecPtr.p->nextOverList;
+ ptrCheckGuard(tofoOverrecPtr, coverflowrecsize, overflowRecord);
+ tofoOverrecPtr.p->prevOverList = troOverflowRecPtr.p->prevOverList;
+ }//if
+ if (troOverflowRecPtr.p->prevOverList != RNIL) {
+ jam();
+ tofoOverrecPtr.i = troOverflowRecPtr.p->prevOverList;
+ ptrCheckGuard(tofoOverrecPtr, coverflowrecsize, overflowRecord);
+ tofoOverrecPtr.p->nextOverList = troOverflowRecPtr.p->nextOverList;
+ } else {
+ jam();
+ fragrecptr.p->firstFreeDirindexRec = troOverflowRecPtr.p->nextOverList;
+ }//if
+}//Dbacc::takeRecOutOfFreeOverdir()
+
+/* --------------------------------------------------------------------------------- */
+/* TAKE_REC_OUT_OF_FREE_OVERPAGE */
+/* DESCRIPTION: AN OVERFLOW PAGE WHICH IS EMPTY HAVE TO BE TAKE OUT OF THE */
+/* FREE LIST OF OVERFLOW PAGE. BY THIS SUBROUTINE THIS LIST */
+/* WILL BE UPDATED. */
+/* --------------------------------------------------------------------------------- */
+void Dbacc::takeRecOutOfFreeOverpage(Signal* signal)
+{
+ OverflowRecordPtr tfoNextOverflowRecPtr;
+ OverflowRecordPtr tfoPrevOverflowRecPtr;
+
+ if (tfoOverflowRecPtr.p->nextOverRec != RNIL) {
+ jam();
+ tfoNextOverflowRecPtr.i = tfoOverflowRecPtr.p->nextOverRec;
+ ptrCheckGuard(tfoNextOverflowRecPtr, coverflowrecsize, overflowRecord);
+ tfoNextOverflowRecPtr.p->prevOverRec = tfoOverflowRecPtr.p->prevOverRec;
+ } else {
+ ndbrequire(fragrecptr.p->lastOverflowRec == tfoOverflowRecPtr.i);
+ jam();
+ fragrecptr.p->lastOverflowRec = tfoOverflowRecPtr.p->prevOverRec;
+ }//if
+ if (tfoOverflowRecPtr.p->prevOverRec != RNIL) {
+ jam();
+ tfoPrevOverflowRecPtr.i = tfoOverflowRecPtr.p->prevOverRec;
+ ptrCheckGuard(tfoPrevOverflowRecPtr, coverflowrecsize, overflowRecord);
+ tfoPrevOverflowRecPtr.p->nextOverRec = tfoOverflowRecPtr.p->nextOverRec;
+ } else {
+ ndbrequire(fragrecptr.p->firstOverflowRec == tfoOverflowRecPtr.i);
+ jam();
+ fragrecptr.p->firstOverflowRec = tfoOverflowRecPtr.p->nextOverRec;
+ }//if
+}//Dbacc::takeRecOutOfFreeOverpage()
+
+void
+Dbacc::reportMemoryUsage(Signal* signal, int gth){
+ signal->theData[0] = EventReport::MemoryUsage;
+ signal->theData[1] = gth;
+ signal->theData[2] = sizeof(* rpPageptr.p);
+ signal->theData[3] = cnoOfAllocatedPages;
+ signal->theData[4] = cpagesize;
+ signal->theData[5] = DBACC;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 6, JBB);
+}
+
+void
+Dbacc::execDUMP_STATE_ORD(Signal* signal)
+{
+ DumpStateOrd * const dumpState = (DumpStateOrd *)&signal->theData[0];
+ if (dumpState->args[0] == DumpStateOrd::AccDumpOneScanRec){
+ Uint32 recordNo = RNIL;
+ if (signal->length() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ if (recordNo >= cscanRecSize)
+ return;
+
+ scanPtr.i = recordNo;
+ ptrAss(scanPtr, scanRec);
+ infoEvent("Dbacc::ScanRec[%d]: state=%d, transid(0x%x, 0x%x)",
+ scanPtr.i, scanPtr.p->scanState,scanPtr.p->scanTrid1,
+ scanPtr.p->scanTrid2);
+ infoEvent(" timer=%d, continueBCount=%d, "
+ "activeLocalFrag=%d, root=%d, nextBucketIndex=%d",
+ scanPtr.p->scanTimer,
+ scanPtr.p->scanContinuebCounter,
+ scanPtr.p->activeLocalFrag,
+ scanPtr.p->rootPtr,
+ scanPtr.p->nextBucketIndex);
+ infoEvent(" scanNextfreerec=%d firstActOp=%d firstLockedOp=%d, "
+ "scanLastLockedOp=%d firstQOp=%d lastQOp=%d",
+ scanPtr.p->scanNextfreerec,
+ scanPtr.p->scanFirstActiveOp,
+ scanPtr.p->scanFirstLockedOp,
+ scanPtr.p->scanLastLockedOp,
+ scanPtr.p->scanFirstQueuedOp,
+ scanPtr.p->scanLastQueuedOp);
+ infoEvent(" scanUserP=%d, startNoBuck=%d, minBucketIndexToRescan=%d, "
+ "maxBucketIndexToRescan=%d",
+ scanPtr.p->scanUserptr,
+ scanPtr.p->startNoOfBuckets,
+ scanPtr.p->minBucketIndexToRescan,
+ scanPtr.p->maxBucketIndexToRescan);
+ infoEvent(" scanBucketState=%d, scanLockHeld=%d, userBlockRef=%d, "
+ "scanMask=%d scanLockMode=%d, keyInfoFlag=%d",
+ scanPtr.p->scanBucketState,
+ scanPtr.p->scanLockHeld,
+ scanPtr.p->scanUserblockref,
+ scanPtr.p->scanMask,
+ scanPtr.p->scanLockMode,
+ scanPtr.p->scanKeyinfoFlag);
+ return;
+ }
+
+ // Dump all ScanRec(ords)
+ if (dumpState->args[0] == DumpStateOrd::AccDumpAllScanRec){
+ Uint32 recordNo = 0;
+ if (signal->length() == 1)
+ infoEvent("ACC: Dump all ScanRec - size: %d",
+ cscanRecSize);
+ else if (signal->length() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ dumpState->args[0] = DumpStateOrd::AccDumpOneScanRec;
+ dumpState->args[1] = recordNo;
+ execDUMP_STATE_ORD(signal);
+
+ if (recordNo < cscanRecSize-1){
+ dumpState->args[0] = DumpStateOrd::AccDumpAllScanRec;
+ dumpState->args[1] = recordNo+1;
+ sendSignal(reference(), GSN_DUMP_STATE_ORD, signal, 2, JBB);
+ }
+ return;
+ }
+
+ // Dump all active ScanRec(ords)
+ if (dumpState->args[0] == DumpStateOrd::AccDumpAllActiveScanRec){
+ Uint32 recordNo = 0;
+ if (signal->length() == 1)
+ infoEvent("ACC: Dump active ScanRec - size: %d",
+ cscanRecSize);
+ else if (signal->length() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ ScanRecPtr sp;
+ sp.i = recordNo;
+ ptrAss(sp, scanRec);
+ if (sp.p->scanState != ScanRec::SCAN_DISCONNECT){
+ dumpState->args[0] = DumpStateOrd::AccDumpOneScanRec;
+ dumpState->args[1] = recordNo;
+ execDUMP_STATE_ORD(signal);
+ }
+
+ if (recordNo < cscanRecSize-1){
+ dumpState->args[0] = DumpStateOrd::AccDumpAllActiveScanRec;
+ dumpState->args[1] = recordNo+1;
+ sendSignal(reference(), GSN_DUMP_STATE_ORD, signal, 2, JBB);
+ }
+ return;
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::DumpPageMemory){
+ reportMemoryUsage(signal, 0);
+ return;
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::EnableUndoDelayDataWrite){
+ ndbout << "Dbacc:: delay write of datapages for table = "
+ << dumpState->args[1]<< endl;
+ c_errorInsert3000_TableId = dumpState->args[1];
+ SET_ERROR_INSERT_VALUE(3000);
+ return;
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::AccDumpOneOperationRec){
+ Uint32 recordNo = RNIL;
+ if (signal->length() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ if (recordNo >= coprecsize)
+ return;
+
+ OperationrecPtr tmpOpPtr;
+ tmpOpPtr.i = recordNo;
+ ptrAss(tmpOpPtr, operationrec);
+ infoEvent("Dbacc::operationrec[%d]: opState=%d, transid(0x%x, 0x%x)",
+ tmpOpPtr.i, tmpOpPtr.p->opState, tmpOpPtr.p->transId1,
+ tmpOpPtr.p->transId2);
+ infoEvent("elementIsforward=%d, elementPage=%d, elementPointer=%d ",
+ tmpOpPtr.p->elementIsforward, tmpOpPtr.p->elementPage,
+ tmpOpPtr.p->elementPointer);
+ infoEvent("fid=%d, fragptr=%d, hashvaluePart=%d ",
+ tmpOpPtr.p->fid, tmpOpPtr.p->fragptr,
+ tmpOpPtr.p->hashvaluePart);
+ infoEvent("hashValue=%d, insertDeleteLen=%d, keyinfoPage=%d ",
+ tmpOpPtr.p->hashValue, tmpOpPtr.p->insertDeleteLen,
+ tmpOpPtr.p->keyinfoPage);
+ infoEvent("nextLockOwnerOp=%d, nextOp=%d, nextParallelQue=%d ",
+ tmpOpPtr.p->nextLockOwnerOp, tmpOpPtr.p->nextOp,
+ tmpOpPtr.p->nextParallelQue);
+ infoEvent("nextQueOp=%d, nextSerialQue=%d, prevOp=%d ",
+ tmpOpPtr.p->nextQueOp, tmpOpPtr.p->nextSerialQue,
+ tmpOpPtr.p->prevOp);
+ infoEvent("prevLockOwnerOp=%d, prevParallelQue=%d, prevQueOp=%d ",
+ tmpOpPtr.p->prevLockOwnerOp, tmpOpPtr.p->nextParallelQue,
+ tmpOpPtr.p->prevQueOp);
+ infoEvent("prevSerialQue=%d, scanRecPtr=%d, longPagePtr=%d ",
+ tmpOpPtr.p->prevSerialQue, tmpOpPtr.p->scanRecPtr,
+ tmpOpPtr.p->longPagePtr);
+ infoEvent("transactionstate=%d, elementIsDisappeared=%d, insertIsDone=%d ",
+ tmpOpPtr.p->transactionstate, tmpOpPtr.p->elementIsDisappeared,
+ tmpOpPtr.p->insertIsDone);
+ infoEvent("lockMode=%d, lockOwner=%d, nodeType=%d ",
+ tmpOpPtr.p->lockMode, tmpOpPtr.p->lockOwner,
+ tmpOpPtr.p->nodeType);
+ infoEvent("operation=%d, opSimple=%d, dirtyRead=%d,scanBits=%d ",
+ tmpOpPtr.p->operation, tmpOpPtr.p->opSimple,
+ tmpOpPtr.p->dirtyRead, tmpOpPtr.p->scanBits);
+ return;
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::AccDumpNumOpRecs){
+
+ Uint32 freeOpRecs = 0;
+ OperationrecPtr opRecPtr;
+ opRecPtr.i = cfreeopRec;
+ while (opRecPtr.i != RNIL){
+ freeOpRecs++;
+ ptrCheckGuard(opRecPtr, coprecsize, operationrec);
+ opRecPtr.i = opRecPtr.p->nextOp;
+ }
+
+ infoEvent("Dbacc::OperationRecords: num=%d, free=%d",
+ coprecsize, freeOpRecs);
+
+ return;
+ }
+ if(dumpState->args[0] == DumpStateOrd::AccDumpFreeOpRecs){
+
+ OperationrecPtr opRecPtr;
+ opRecPtr.i = cfreeopRec;
+ while (opRecPtr.i != RNIL){
+
+ dumpState->args[0] = DumpStateOrd::AccDumpOneOperationRec;
+ dumpState->args[1] = opRecPtr.i;
+ execDUMP_STATE_ORD(signal);
+
+ ptrCheckGuard(opRecPtr, coprecsize, operationrec);
+ opRecPtr.i = opRecPtr.p->nextOp;
+ }
+
+
+ return;
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::AccDumpNotFreeOpRecs){
+ Uint32 recordStart = RNIL;
+ if (signal->length() == 2)
+ recordStart = dumpState->args[1];
+ else
+ return;
+
+ if (recordStart >= coprecsize)
+ return;
+
+ for (Uint32 i = recordStart; i < coprecsize; i++){
+
+ bool inFreeList = false;
+ OperationrecPtr opRecPtr;
+ opRecPtr.i = cfreeopRec;
+ while (opRecPtr.i != RNIL){
+ if (opRecPtr.i == i){
+ inFreeList = true;
+ break;
+ }
+ ptrCheckGuard(opRecPtr, coprecsize, operationrec);
+ opRecPtr.i = opRecPtr.p->nextOp;
+ }
+ if (inFreeList == false){
+ dumpState->args[0] = DumpStateOrd::AccDumpOneOperationRec;
+ dumpState->args[1] = i;
+ execDUMP_STATE_ORD(signal);
+ }
+ }
+ return;
+ }
+
+#if 0
+ if (type == 100) {
+ RelTabMemReq * const req = (RelTabMemReq *)signal->getDataPtrSend();
+ req->primaryTableId = 2;
+ req->secondaryTableId = RNIL;
+ req->userPtr = 2;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownBlockref, GSN_REL_TABMEMREQ, signal,
+ RelTabMemReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 101) {
+ RelTabMemReq * const req = (RelTabMemReq *)signal->getDataPtrSend();
+ req->primaryTableId = 4;
+ req->secondaryTableId = 5;
+ req->userPtr = 4;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownBlockref, GSN_REL_TABMEMREQ, signal,
+ RelTabMemReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 102) {
+ RelTabMemReq * const req = (RelTabMemReq *)signal->getDataPtrSend();
+ req->primaryTableId = 6;
+ req->secondaryTableId = 8;
+ req->userPtr = 6;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownBlockref, GSN_REL_TABMEMREQ, signal,
+ RelTabMemReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 103) {
+ DropTabFileReq * const req = (DropTabFileReq *)signal->getDataPtrSend();
+ req->primaryTableId = 2;
+ req->secondaryTableId = RNIL;
+ req->userPtr = 2;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownBlockref, GSN_DROP_TABFILEREQ, signal,
+ DropTabFileReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 104) {
+ DropTabFileReq * const req = (DropTabFileReq *)signal->getDataPtrSend();
+ req->primaryTableId = 4;
+ req->secondaryTableId = 5;
+ req->userPtr = 4;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownBlockref, GSN_DROP_TABFILEREQ, signal,
+ DropTabFileReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 105) {
+ DropTabFileReq * const req = (DropTabFileReq *)signal->getDataPtrSend();
+ req->primaryTableId = 6;
+ req->secondaryTableId = 8;
+ req->userPtr = 6;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownBlockref, GSN_DROP_TABFILEREQ, signal,
+ DropTabFileReq::SignalLength, JBB);
+ return;
+ }//if
+#endif
+}//Dbacc::execDUMP_STATE_ORD()
+
+void Dbacc::execSET_VAR_REQ(Signal* signal)
+{
+ SetVarReq* const setVarReq = (SetVarReq*)&signal->theData[0];
+ ConfigParamId var = setVarReq->variable();
+ int val = setVarReq->value();
+
+
+ switch (var) {
+
+ case NoOfDiskPagesToDiskAfterRestartACC:
+ clblPagesPerTick = val;
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ case NoOfDiskPagesToDiskDuringRestartACC:
+ // Valid only during start so value not set.
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ default:
+ sendSignal(CMVMI_REF, GSN_SET_VAR_REF, signal, 1, JBB);
+ } // switch
+
+
+}//execSET_VAR_REQ()
diff --git a/ndb/src/kernel/blocks/dbacc/Makefile b/ndb/src/kernel/blocks/dbacc/Makefile
new file mode 100644
index 00000000000..93a830cec95
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbacc/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := dbacc
+
+SOURCES = \
+ DbaccInit.cpp \
+ DbaccMain.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dbdict/CreateIndex.txt b/ndb/src/kernel/blocks/dbdict/CreateIndex.txt
new file mode 100644
index 00000000000..3d11e501c07
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/CreateIndex.txt
@@ -0,0 +1,152 @@
+Unique Hash Index
+=================
+
+unique hash index X on T(A1,...,An) becomes:
+table X with primary key A1,...,An and extra attribute NDB$PK
+
+NDB$PK is primary key of T concatenated at 4-byte boundaries
+
+Protocols:
+
+U - user, initiator of protocol
+C - coordinator
+P - participants, including coordinator node
+
+RT_ - request type, current state
+
+P always replies to C with current RT_ (initially RT_DICT_PREPARE)
+C replies to U at the end
+
+CREATE INDEX
+------------
+
+U: RT_USER
+
+C: forward request to P's
+P: check and reply
+
+C: invoke CREATE TABLE for index table
+
+C: invoke ALTER INDEX online
+
+C: send RT_DICT_COMMIT to P's
+P: reply
+
+C: reply to U
+
+DROP INDEX
+----------
+
+[ todo ]
+
+ALTER INDEX online
+------------------
+
+U: RT_USER, RT_CREATE_INDEX, RT_NODERESTART, RT_SYSTEMRESTART
+
+C: forward request to P's
+P: check and reply
+
+C: send RT_DICT_TC to P's
+P: create index in local TC, and reply
+
+C: invoke CREATE TRIGGER for insert/update/delete triggers
+
+C: invoke BUILD INDEX
+
+C: send RT_DICT_COMMIT to P's
+P: reply
+
+C: reply to U
+
+ALTER INDEX offline
+-------------------
+
+[ todo ]
+
+BUILD INDEX
+-----------
+
+U: RT_USER, RT_ALTER_INDEX
+
+C: forward request to P's
+P: check and reply
+
+C: invoke CREATE TRIGGER for read-only constraint on NDB$PK
+
+C: send RT_DICT_TRIX to P's
+P: build index via local TRIX, and reply
+
+C: invoke DROP TRIGGER for read-only constraint on NDB$PK
+
+C: send RT_DICT_TC to P's
+P: online index in local TC, and reply
+
+CREATE TRIGGER
+--------------
+
+U: RT_USER, RT_ALTER_INDEX, RT_BUILD_INDEX
+
+C: forward request to P's
+P: check and reply
+
+C: seize trigger id and send RT_DICT_CREATE to P's
+P: create trigger in DICT (also connect to index record), and reply
+
+C: invoke ALTER TRIGGER online [ not if subscription trigger ]
+
+C: send RT_DICT_COMMIT to P's
+P: reply
+
+C: reply to U
+
+DROP TRIGGER
+------------
+
+[ todo ]
+
+ALTER TRIGGER online
+--------------------
+
+U: RT_USER, RT_CREATE_TRIGGER
+
+C: forward request to P's
+P: check and reply
+
+C: send RT_DICT_TC to P's
+P: create trigger in local TC, and reply
+
+C: send RT_DICT_LQH to P's
+P: create trigger in local LQH (which just forwards to TUP), and reply
+
+C: send RT_DICT_COMMIT to P's
+P: reply
+
+C: reply to U
+
+ALTER TRIGGER offline
+---------------------
+
+[ todo ]
+
+Ordered Index << under work >>
+=============
+
+created as DICT table, as before, to reuse the code
+
+keep NDB$PK as last attribute (not used but logically correct)
+
+create fragments and attributes must be modified
+
+global metadata? implemented but will use signals anyway
+
+create (after-) insert/update/delete triggers as DICT objects, as before
+
+skip following:
+- create index in TC
+- create triggers in TC
+- read-only constraint on NDB$PK
+
+create (before-) commit trigger in TUP
+
+alter online (in TUX, instead of TC) is needed
diff --git a/ndb/src/kernel/blocks/dbdict/CreateTable.new.txt b/ndb/src/kernel/blocks/dbdict/CreateTable.new.txt
new file mode 100644
index 00000000000..d37732dcda1
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/CreateTable.new.txt
@@ -0,0 +1,29 @@
+
+1) Receive from client (sequence of DICTTABINFO)
+
+2) CREATE_FRAGMENTATION_REQ -> local DIH
+ Returns all fragments for table + some other stuff
+ NOTE without side effects in DIH
+
+3) Pack table description
+
+4) CREATE_TAB -> all DICTs (including table data)
+ 1) Write schema file (ADD_STARTED)
+ 2) Write table descriptor to file
+ 3) CREATE_TAB (DIADDTABREQ) -> local DIH (including fragment info)
+ 4) DIH
+ 1) write table descriptor
+ 2) For each local fragment
+ ADD_FRAG -> local DICT
+ LQHFRAGREQ -> local LQH
+ LQHADDATTREQ -> local LQH
+ 5) TAB_COMMITREQ -> local LQH
+
+5) WAIT_GCP
+
+6) ALTER_TAB (activate) -> all DICTs
+ 1) Write schema file (CREATED)
+ 2) TAB_COMMITREQ -> local DIH
+ 3) TC_SCHVERREQ -> local TC
+
+
diff --git a/ndb/src/kernel/blocks/dbdict/CreateTable.txt b/ndb/src/kernel/blocks/dbdict/CreateTable.txt
new file mode 100644
index 00000000000..0b37e5d767f
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/CreateTable.txt
@@ -0,0 +1,35 @@
+
+1) Receive from client (sequence of DICTTABINFO)
+
+2) DICT_SCHEMAREQ -> all DICTs
+ Write ADD_STARTED in schema file
+
+3) Pack table description
+
+4) DICTTABINFO -> all DICTs (but self) (containing packed table info)
+ self -> Write 2 file
+ 1) Write 2 file
+
+5) DICT_SCHEMAREQ -> all DICTs
+ Write UPDATE_PAGE_COUNT in schema file
+
+6) DIADDTABREQ -> local DIH
+ 1) Create fragments
+ 2) For each fragment
+ DIHADDFRAGREQ -> all DIH
+ 3) For each fragment
+ DICTFRAGSREQ -> local DICT
+ 1) LQHFRAGREQ -> concerned LQH
+ 2) For each attribute
+ LQHADDATTREQ -> concerned LQH
+
+7) WAIT_GCP -> local DIH
+
+8) DICT_SCHEMAREQ -> all DICTs
+ Write TABLE_ADD_COMMITTED in schema file
+
+9) TAB_COMMITREQ -> all LQH & DIH
+
+10) TC_SCHVERREQ -> all TC
+
+11) UNBLO_DICTREQ -> all DICT
diff --git a/ndb/src/kernel/blocks/dbdict/Dbdict.cpp b/ndb/src/kernel/blocks/dbdict/Dbdict.cpp
new file mode 100644
index 00000000000..9a72d9deb50
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/Dbdict.cpp
@@ -0,0 +1,11628 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBDICT_C
+#include "Dbdict.hpp"
+
+#include <ndb_limits.h>
+#include <NdbOut.hpp>
+#include <Properties.hpp>
+#include <Configuration.hpp>
+#include <SectionReader.hpp>
+#include <SimpleProperties.hpp>
+#include <AttributeHeader.hpp>
+#include <signaldata/DictSchemaInfo.hpp>
+#include <signaldata/DictSizeAltReq.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/DropTabFile.hpp>
+
+#include <signaldata/EventReport.hpp>
+#include <signaldata/FsCloseReq.hpp>
+#include <signaldata/FsConf.hpp>
+#include <signaldata/FsOpenReq.hpp>
+#include <signaldata/FsReadWriteReq.hpp>
+#include <signaldata/FsRef.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/GetTableId.hpp>
+#include <signaldata/HotSpareRep.hpp>
+#include <signaldata/NFCompleteRep.hpp>
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/ReadNodesConf.hpp>
+#include <signaldata/RelTabMem.hpp>
+#include <signaldata/WaitGCP.hpp>
+#include <signaldata/ListTables.hpp>
+
+#include <signaldata/CreateTrig.hpp>
+#include <signaldata/AlterTrig.hpp>
+#include <signaldata/DropTrig.hpp>
+#include <signaldata/CreateIndx.hpp>
+#include <signaldata/DropIndx.hpp>
+#include <signaldata/BuildIndx.hpp>
+
+#include <signaldata/CreateEvnt.hpp>
+#include <signaldata/UtilPrepare.hpp>
+#include <signaldata/UtilExecute.hpp>
+#include <signaldata/UtilRelease.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <GrepError.hpp>
+//#include <signaldata/DropEvnt.hpp>
+
+#include <signaldata/LqhFrag.hpp>
+
+#include <signaldata/DiAddTab.hpp>
+#include <signaldata/DihStartTab.hpp>
+
+#include <signaldata/DropTable.hpp>
+#include <signaldata/DropTab.hpp>
+#include <signaldata/PrepDropTab.hpp>
+
+#include <signaldata/CreateTable.hpp>
+#include <signaldata/AlterTable.hpp>
+#include <signaldata/AlterTab.hpp>
+#include <signaldata/CreateFragmentation.hpp>
+#include <signaldata/CreateTab.hpp>
+#include "../dbtc/Dbtc.hpp"
+#include <NdbSleep.h>
+
+#define ZNOT_FOUND 626
+#define ZALREADYEXIST 630
+
+//#define EVENT_PH2_DEBUG
+//#define EVENT_PH3_DEBUG
+//#define EVENT_DEBUG
+
+#define EVENT_TRACE \
+// ndbout_c("Event debug trace: File: %s Line: %u", __FILE__, __LINE__)
+
+#define DIV(x,y) (((x)+(y)-1)/(y))
+#include <ndb_version.h>
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: GENERAL MODULE -------------------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* This module contains general stuff. Mostly debug signals and */
+/* general signals that go into a specific module after checking a */
+/* state variable. Also general subroutines used by many. */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+/* ---------------------------------------------------------------- */
+// This signal is used to dump states of various variables in the
+// block by command.
+/* ---------------------------------------------------------------- */
+void
+Dbdict::execDUMP_STATE_ORD(Signal* signal)
+{
+ jamEntry();
+
+#ifdef NDB_DEBUG
+ if(signal->theData[0] == 1222){
+ const Uint32 tab = signal->theData[1];
+ PrepDropTabReq* req = (PrepDropTabReq*)signal->getDataPtr();
+ req->senderRef = reference();
+ req->senderData = 1222;
+ req->tableId = tab;
+ sendSignal(DBLQH_REF, GSN_PREP_DROP_TAB_REQ, signal,
+ PrepDropTabReq::SignalLength, JBB);
+ }
+
+ if(signal->theData[0] == 1223){
+ const Uint32 tab = signal->theData[1];
+ PrepDropTabReq* req = (PrepDropTabReq*)signal->getDataPtr();
+ req->senderRef = reference();
+ req->senderData = 1222;
+ req->tableId = tab;
+ sendSignal(DBTC_REF, GSN_PREP_DROP_TAB_REQ, signal,
+ PrepDropTabReq::SignalLength, JBB);
+ }
+
+ if(signal->theData[0] == 1224){
+ const Uint32 tab = signal->theData[1];
+ PrepDropTabReq* req = (PrepDropTabReq*)signal->getDataPtr();
+ req->senderRef = reference();
+ req->senderData = 1222;
+ req->tableId = tab;
+ sendSignal(DBDIH_REF, GSN_PREP_DROP_TAB_REQ, signal,
+ PrepDropTabReq::SignalLength, JBB);
+ }
+
+ if(signal->theData[0] == 1225){
+ const Uint32 tab = signal->theData[1];
+ const Uint32 ver = signal->theData[2];
+ TableRecordPtr tabRecPtr;
+ c_tableRecordPool.getPtr(tabRecPtr, tab);
+ DropTableReq * req = (DropTableReq*)signal->getDataPtr();
+ req->senderData = 1225;
+ req->senderRef = numberToRef(1,1);
+ req->tableId = tab;
+ req->tableVersion = tabRecPtr.p->tableVersion + ver;
+ sendSignal(DBDICT_REF, GSN_DROP_TABLE_REQ, signal,
+ DropTableReq::SignalLength, JBB);
+ }
+#endif
+
+ return;
+}//Dbdict::execDUMP_STATE_ORD()
+
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+// CONTINUEB is used when a real-time break is needed for long
+// processes.
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+void Dbdict::execCONTINUEB(Signal* signal)
+{
+ jamEntry();
+ switch (signal->theData[0]) {
+ case ZPACK_TABLE_INTO_PAGES :
+ jam();
+ packTableIntoPages(signal, signal->theData[1], signal->theData[2]);
+ break;
+
+ case ZSEND_GET_TAB_RESPONSE :
+ jam();
+ sendGetTabResponse(signal);
+ break;
+
+ default :
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//execCONTINUEB()
+
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+// Routine to handle pack table into pages.
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+
+void Dbdict::packTableIntoPages(Signal* signal, Uint32 tableId, Uint32 pageId)
+{
+
+ PageRecordPtr pagePtr;
+ TableRecordPtr tablePtr;
+ c_pageRecordArray.getPtr(pagePtr, pageId);
+
+ memset(&pagePtr.p->word[0], 0, 4 * ZPAGE_HEADER_SIZE);
+ c_tableRecordPool.getPtr(tablePtr, tableId);
+ LinearWriter w(&pagePtr.p->word[ZPAGE_HEADER_SIZE],
+ 8 * ZSIZE_OF_PAGES_IN_WORDS);
+
+ w.first();
+ packTableIntoPagesImpl(w, tablePtr);
+
+ Uint32 wordsOfTable = w.getWordsUsed();
+ Uint32 pagesUsed =
+ DIV(wordsOfTable + ZPAGE_HEADER_SIZE, ZSIZE_OF_PAGES_IN_WORDS);
+ pagePtr.p->word[ZPOS_CHECKSUM] =
+ computeChecksum(&pagePtr.p->word[0], pagesUsed * ZSIZE_OF_PAGES_IN_WORDS);
+
+ switch (c_packTable.m_state) {
+ case PackTable::PTS_IDLE:
+ case PackTable::PTS_ADD_TABLE_MASTER:
+ case PackTable::PTS_ADD_TABLE_SLAVE:
+ case PackTable::PTS_RESTART:
+ ndbrequire(false);
+ break;
+ case PackTable::PTS_GET_TAB:
+ jam();
+ c_retrieveRecord.retrievedNoOfPages = pagesUsed;
+ c_retrieveRecord.retrievedNoOfWords = wordsOfTable;
+ sendGetTabResponse(signal);
+ return;
+ break;
+ }//switch
+ ndbrequire(false);
+ return;
+}//packTableIntoPages()
+
+void
+Dbdict::packTableIntoPagesImpl(SimpleProperties::Writer & w,
+ TableRecordPtr tablePtr){
+
+ w.add(DictTabInfo::TableName, tablePtr.p->tableName);
+ w.add(DictTabInfo::TableId, tablePtr.i);
+ w.add(DictTabInfo::SecondTableId, tablePtr.p->secondTable);
+ w.add(DictTabInfo::TableVersion, tablePtr.p->tableVersion);
+ w.add(DictTabInfo::NoOfKeyAttr, tablePtr.p->noOfPrimkey);
+ w.add(DictTabInfo::NoOfAttributes, tablePtr.p->noOfAttributes);
+ w.add(DictTabInfo::NoOfNullable, tablePtr.p->noOfNullAttr);
+ w.add(DictTabInfo::NoOfVariable, (Uint32)0);
+ w.add(DictTabInfo::KeyLength, tablePtr.p->tupKeyLength);
+
+ w.add(DictTabInfo::TableLoggedFlag, tablePtr.p->storedTable);
+ w.add(DictTabInfo::MinLoadFactor, tablePtr.p->minLoadFactor);
+ w.add(DictTabInfo::MaxLoadFactor, tablePtr.p->maxLoadFactor);
+ w.add(DictTabInfo::TableKValue, tablePtr.p->kValue);
+ w.add(DictTabInfo::FragmentTypeVal, tablePtr.p->fragmentType);
+ w.add(DictTabInfo::FragmentKeyTypeVal, tablePtr.p->fragmentKeyType);
+ w.add(DictTabInfo::TableTypeVal, tablePtr.p->tableType);
+
+ if (tablePtr.p->primaryTableId != RNIL){
+ TableRecordPtr primTab;
+ c_tableRecordPool.getPtr(primTab, tablePtr.p->primaryTableId);
+ w.add(DictTabInfo::PrimaryTable, primTab.p->tableName);
+ w.add(DictTabInfo::PrimaryTableId, tablePtr.p->primaryTableId);
+ w.add(DictTabInfo::IndexState, tablePtr.p->indexState);
+ w.add(DictTabInfo::InsertTriggerId, tablePtr.p->insertTriggerId);
+ w.add(DictTabInfo::UpdateTriggerId, tablePtr.p->updateTriggerId);
+ w.add(DictTabInfo::DeleteTriggerId, tablePtr.p->deleteTriggerId);
+ w.add(DictTabInfo::CustomTriggerId, tablePtr.p->customTriggerId);
+ }
+ w.add(DictTabInfo::FrmLen, tablePtr.p->frmLen);
+ w.add(DictTabInfo::FrmData, tablePtr.p->frmData, tablePtr.p->frmLen);
+
+ Uint32 nextAttribute = tablePtr.p->firstAttribute;
+ AttributeRecordPtr attrPtr;
+ do {
+ jam();
+ c_attributeRecordPool.getPtr(attrPtr, nextAttribute);
+
+ w.add(DictTabInfo::AttributeName, attrPtr.p->attributeName);
+ w.add(DictTabInfo::AttributeId, attrPtr.p->attributeId);
+ w.add(DictTabInfo::AttributeKeyFlag, attrPtr.p->tupleKey > 0);
+
+ const Uint32 desc = attrPtr.p->attributeDescriptor;
+ const Uint32 attrType = AttributeDescriptor::getType(desc);
+ const Uint32 attrSize = AttributeDescriptor::getSize(desc);
+ const Uint32 arraySize = AttributeDescriptor::getArraySize(desc);
+ const Uint32 nullable = AttributeDescriptor::getNullable(desc);
+ const Uint32 DGroup = AttributeDescriptor::getDGroup(desc);
+ const Uint32 DKey = AttributeDescriptor::getDKey(desc);
+ const Uint32 attrStoredInd = AttributeDescriptor::getStoredInTup(desc);
+
+ w.add(DictTabInfo::AttributeType, attrType);
+ w.add(DictTabInfo::AttributeSize, attrSize);
+ w.add(DictTabInfo::AttributeArraySize, arraySize);
+ w.add(DictTabInfo::AttributeNullableFlag, nullable);
+ w.add(DictTabInfo::AttributeDGroup, DGroup);
+ w.add(DictTabInfo::AttributeDKey, DKey);
+ w.add(DictTabInfo::AttributeStoredInd, attrStoredInd);
+ w.add(DictTabInfo::AttributeExtType, attrPtr.p->extType);
+ w.add(DictTabInfo::AttributeExtPrecision, attrPtr.p->extPrecision);
+ w.add(DictTabInfo::AttributeExtScale, attrPtr.p->extScale);
+ w.add(DictTabInfo::AttributeExtLength, attrPtr.p->extLength);
+ w.add(DictTabInfo::AttributeAutoIncrement,
+ (Uint32)attrPtr.p->autoIncrement);
+ w.add(DictTabInfo::AttributeDefaultValue, attrPtr.p->defaultValue);
+
+ w.add(DictTabInfo::AttributeEnd, 1);
+ nextAttribute = attrPtr.p->nextAttrInTable;
+ } while (nextAttribute != RNIL);
+
+ w.add(DictTabInfo::TableEnd, 1);
+}
+
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+// The routines to handle responses from file system.
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+
+/* ---------------------------------------------------------------- */
+// A file was successfully closed.
+/* ---------------------------------------------------------------- */
+void Dbdict::execFSCLOSECONF(Signal* signal)
+{
+ FsConnectRecordPtr fsPtr;
+ FsConf * const fsConf = (FsConf *)&signal->theData[0];
+ jamEntry();
+ c_fsConnectRecordPool.getPtr(fsPtr, fsConf->userPointer);
+ switch (fsPtr.p->fsState) {
+ case FsConnectRecord::CLOSE_WRITE_SCHEMA:
+ jam();
+ closeWriteSchemaConf(signal, fsPtr);
+ break;
+ case FsConnectRecord::CLOSE_READ_SCHEMA:
+ jam();
+ closeReadSchemaConf(signal, fsPtr);
+ break;
+ case FsConnectRecord::CLOSE_READ_TAB_FILE:
+ jam();
+ closeReadTableConf(signal, fsPtr);
+ break;
+ case FsConnectRecord::CLOSE_WRITE_TAB_FILE:
+ jam();
+ closeWriteTableConf(signal, fsPtr);
+ break;
+ default:
+ jamLine((fsPtr.p->fsState & 0xFFF));
+ ndbrequire(false);
+ break;
+ }//switch
+}//execFSCLOSECONF()
+
+/* ---------------------------------------------------------------- */
+// A close file was refused.
+/* ---------------------------------------------------------------- */
+void Dbdict::execFSCLOSEREF(Signal* signal)
+{
+ jamEntry();
+ progError(0, 0);
+}//execFSCLOSEREF()
+
+/* ---------------------------------------------------------------- */
+// A file was successfully opened.
+/* ---------------------------------------------------------------- */
+void Dbdict::execFSOPENCONF(Signal* signal)
+{
+ FsConnectRecordPtr fsPtr;
+ jamEntry();
+ FsConf * const fsConf = (FsConf *)&signal->theData[0];
+ c_fsConnectRecordPool.getPtr(fsPtr, fsConf->userPointer);
+
+ Uint32 filePointer = fsConf->filePointer;
+ fsPtr.p->filePtr = filePointer;
+ switch (fsPtr.p->fsState) {
+ case FsConnectRecord::OPEN_WRITE_SCHEMA:
+ jam();
+ fsPtr.p->fsState = FsConnectRecord::WRITE_SCHEMA;
+ writeSchemaFile(signal, filePointer, fsPtr.i);
+ break;
+ case FsConnectRecord::OPEN_READ_SCHEMA1:
+ jam();
+ fsPtr.p->fsState = FsConnectRecord::READ_SCHEMA1;
+ readSchemaFile(signal, filePointer, fsPtr.i);
+ break;
+ case FsConnectRecord::OPEN_READ_SCHEMA2:
+ jam();
+ fsPtr.p->fsState = FsConnectRecord::READ_SCHEMA2;
+ readSchemaFile(signal, filePointer, fsPtr.i);
+ break;
+ case FsConnectRecord::OPEN_READ_TAB_FILE1:
+ jam();
+ fsPtr.p->fsState = FsConnectRecord::READ_TAB_FILE1;
+ readTableFile(signal, filePointer, fsPtr.i);
+ break;
+ case FsConnectRecord::OPEN_READ_TAB_FILE2:
+ jam();
+ fsPtr.p->fsState = FsConnectRecord::READ_TAB_FILE2;
+ readTableFile(signal, filePointer, fsPtr.i);
+ break;
+ case FsConnectRecord::OPEN_WRITE_TAB_FILE:
+ jam();
+ fsPtr.p->fsState = FsConnectRecord::WRITE_TAB_FILE;
+ writeTableFile(signal, filePointer, fsPtr.i);
+ break;
+ default:
+ jamLine((fsPtr.p->fsState & 0xFFF));
+ ndbrequire(false);
+ break;
+ }//switch
+}//execFSOPENCONF()
+
+/* ---------------------------------------------------------------- */
+// An open file was refused.
+/* ---------------------------------------------------------------- */
+void Dbdict::execFSOPENREF(Signal* signal)
+{
+ jamEntry();
+ FsRef * const fsRef = (FsRef *)&signal->theData[0];
+ FsConnectRecordPtr fsPtr;
+ c_fsConnectRecordPool.getPtr(fsPtr, fsRef->userPointer);
+ switch (fsPtr.p->fsState) {
+ case FsConnectRecord::OPEN_READ_SCHEMA1:
+ openReadSchemaRef(signal, fsPtr);
+ break;
+ case FsConnectRecord::OPEN_READ_TAB_FILE1:
+ jam();
+ openReadTableRef(signal, fsPtr);
+ break;
+ default:
+ jamLine((fsPtr.p->fsState & 0xFFF));
+ ndbrequire(false);
+ break;
+ }//switch
+}//execFSOPENREF()
+
+/* ---------------------------------------------------------------- */
+// A file was successfully read.
+/* ---------------------------------------------------------------- */
+void Dbdict::execFSREADCONF(Signal* signal)
+{
+ jamEntry();
+ FsConf * const fsConf = (FsConf *)&signal->theData[0];
+ FsConnectRecordPtr fsPtr;
+ c_fsConnectRecordPool.getPtr(fsPtr, fsConf->userPointer);
+ switch (fsPtr.p->fsState) {
+ case FsConnectRecord::READ_SCHEMA1:
+ case FsConnectRecord::READ_SCHEMA2:
+ readSchemaConf(signal ,fsPtr);
+ break;
+ case FsConnectRecord::READ_TAB_FILE1:
+ case FsConnectRecord::READ_TAB_FILE2:
+ jam();
+ readTableConf(signal ,fsPtr);
+ break;
+ default:
+ jamLine((fsPtr.p->fsState & 0xFFF));
+ ndbrequire(false);
+ break;
+ }//switch
+}//execFSREADCONF()
+
+/* ---------------------------------------------------------------- */
+// A read file was refused.
+/* ---------------------------------------------------------------- */
+void Dbdict::execFSREADREF(Signal* signal)
+{
+ jamEntry();
+ FsRef * const fsRef = (FsRef *)&signal->theData[0];
+ FsConnectRecordPtr fsPtr;
+ c_fsConnectRecordPool.getPtr(fsPtr, fsRef->userPointer);
+ switch (fsPtr.p->fsState) {
+ case FsConnectRecord::READ_SCHEMA1:
+ readSchemaRef(signal, fsPtr);
+ break;
+ case FsConnectRecord::READ_TAB_FILE1:
+ jam();
+ readTableRef(signal, fsPtr);
+ break;
+ default:
+ jamLine((fsPtr.p->fsState & 0xFFF));
+ ndbrequire(false);
+ break;
+ }//switch
+}//execFSREADREF()
+
+/* ---------------------------------------------------------------- */
+// A file was successfully written.
+/* ---------------------------------------------------------------- */
+void Dbdict::execFSWRITECONF(Signal* signal)
+{
+ FsConf * const fsConf = (FsConf *)&signal->theData[0];
+ FsConnectRecordPtr fsPtr;
+ jamEntry();
+ c_fsConnectRecordPool.getPtr(fsPtr, fsConf->userPointer);
+ switch (fsPtr.p->fsState) {
+ case FsConnectRecord::WRITE_TAB_FILE:
+ writeTableConf(signal, fsPtr);
+ break;
+ case FsConnectRecord::WRITE_SCHEMA:
+ jam();
+ writeSchemaConf(signal, fsPtr);
+ break;
+ default:
+ jamLine((fsPtr.p->fsState & 0xFFF));
+ ndbrequire(false);
+ break;
+ }//switch
+}//execFSWRITECONF()
+
+/* ---------------------------------------------------------------- */
+// A write file was refused.
+/* ---------------------------------------------------------------- */
+void Dbdict::execFSWRITEREF(Signal* signal)
+{
+ jamEntry();
+ progError(0, 0);
+}//execFSWRITEREF()
+
+/* ---------------------------------------------------------------- */
+// Routines to handle Read/Write of Table Files
+/* ---------------------------------------------------------------- */
+void
+Dbdict::writeTableFile(Signal* signal, Uint32 tableId,
+ SegmentedSectionPtr tabInfoPtr, Callback* callback){
+
+ ndbrequire(c_writeTableRecord.tableWriteState == WriteTableRecord::IDLE);
+
+ Uint32 sz = tabInfoPtr.sz + ZPAGE_HEADER_SIZE;
+
+ c_writeTableRecord.noOfPages = DIV(sz, ZSIZE_OF_PAGES_IN_WORDS);
+ c_writeTableRecord.tableWriteState = WriteTableRecord::CALLBACK;
+ c_writeTableRecord.m_callback = * callback;
+
+ c_writeTableRecord.pageId = 0;
+ ndbrequire(c_writeTableRecord.noOfPages < 8);
+
+ PageRecordPtr pageRecPtr;
+ c_pageRecordArray.getPtr(pageRecPtr, c_writeTableRecord.pageId);
+ copy(&pageRecPtr.p->word[ZPAGE_HEADER_SIZE], tabInfoPtr);
+
+ memset(&pageRecPtr.p->word[0], 0, 4 * ZPAGE_HEADER_SIZE);
+ pageRecPtr.p->word[ZPOS_CHECKSUM] =
+ computeChecksum(&pageRecPtr.p->word[0],
+ c_writeTableRecord.noOfPages * ZSIZE_OF_PAGES_IN_WORDS);
+
+ startWriteTableFile(signal, tableId);
+
+}
+
+void Dbdict::startWriteTableFile(Signal* signal, Uint32 tableId)
+{
+ FsConnectRecordPtr fsPtr;
+ c_writeTableRecord.tableId = tableId;
+ c_fsConnectRecordPool.getPtr(fsPtr, getFsConnRecord());
+ fsPtr.p->fsState = FsConnectRecord::OPEN_WRITE_TAB_FILE;
+ openTableFile(signal, 0, fsPtr.i, tableId, true);
+ c_writeTableRecord.noOfTableFilesHandled = 0;
+}//Dbdict::startWriteTableFile()
+
+void Dbdict::openTableFile(Signal* signal,
+ Uint32 fileNo,
+ Uint32 fsConPtr,
+ Uint32 tableId,
+ bool writeFlag)
+{
+ TableRecordPtr tablePtr;
+ FsOpenReq * const fsOpenReq = (FsOpenReq *)&signal->theData[0];
+ c_tableRecordPool.getPtr(tablePtr, tableId);
+
+ fsOpenReq->userReference = reference();
+ fsOpenReq->userPointer = fsConPtr;
+ if (writeFlag) {
+ jam();
+ fsOpenReq->fileFlags =
+ FsOpenReq::OM_WRITEONLY |
+ FsOpenReq::OM_TRUNCATE |
+ FsOpenReq::OM_CREATE |
+ FsOpenReq::OM_SYNC;
+ } else {
+ jam();
+ fsOpenReq->fileFlags = FsOpenReq::OM_READONLY;
+ }//if
+ ndbrequire(tablePtr.p->tableVersion < ZNIL);
+ fsOpenReq->fileNumber[3] = 0; // Initialise before byte changes
+ FsOpenReq::setVersion(fsOpenReq->fileNumber, 1);
+ FsOpenReq::setSuffix(fsOpenReq->fileNumber, FsOpenReq::S_TABLELIST);
+ FsOpenReq::v1_setDisk(fsOpenReq->fileNumber, (fileNo + 1));
+ FsOpenReq::v1_setTable(fsOpenReq->fileNumber, tableId);
+ FsOpenReq::v1_setFragment(fsOpenReq->fileNumber, (Uint32)-1);
+ FsOpenReq::v1_setS(fsOpenReq->fileNumber, tablePtr.p->tableVersion);
+ FsOpenReq::v1_setP(fsOpenReq->fileNumber, 255);
+/* ---------------------------------------------------------------- */
+// File name : D1/DBDICT/T0/S1.TableList
+// D1 means Disk 1 (set by fileNo + 1)
+// T0 means table id = 0
+// S1 means tableVersion 1
+// TableList indicates that this is a file for a table description.
+/* ---------------------------------------------------------------- */
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, FsOpenReq::SignalLength, JBA);
+}//openTableFile()
+
+void Dbdict::writeTableFile(Signal* signal, Uint32 filePtr, Uint32 fsConPtr)
+{
+ FsReadWriteReq * const fsRWReq = (FsReadWriteReq *)&signal->theData[0];
+
+ fsRWReq->filePointer = filePtr;
+ fsRWReq->userReference = reference();
+ fsRWReq->userPointer = fsConPtr;
+ fsRWReq->operationFlag = 0; // Initialise before bit changes
+ FsReadWriteReq::setSyncFlag(fsRWReq->operationFlag, 1);
+ FsReadWriteReq::setFormatFlag(fsRWReq->operationFlag,
+ FsReadWriteReq::fsFormatArrayOfPages);
+ fsRWReq->varIndex = ZALLOCATE;
+ fsRWReq->numberOfPages = c_writeTableRecord.noOfPages;
+ fsRWReq->data.arrayOfPages.varIndex = c_writeTableRecord.pageId;
+ fsRWReq->data.arrayOfPages.fileOffset = 0; // Write to file page 0
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+}//writeTableFile()
+
+void Dbdict::writeTableConf(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ fsPtr.p->fsState = FsConnectRecord::CLOSE_WRITE_TAB_FILE;
+ closeFile(signal, fsPtr.p->filePtr, fsPtr.i);
+ return;
+}//Dbdict::writeTableConf()
+
+void Dbdict::closeWriteTableConf(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ c_writeTableRecord.noOfTableFilesHandled++;
+ if (c_writeTableRecord.noOfTableFilesHandled < 2) {
+ jam();
+ fsPtr.p->fsState = FsConnectRecord::OPEN_WRITE_TAB_FILE;
+ openTableFile(signal, 1, fsPtr.i, c_writeTableRecord.tableId, true);
+ return;
+ }
+ ndbrequire(c_writeTableRecord.noOfTableFilesHandled == 2);
+ c_fsConnectRecordPool.release(fsPtr);
+ WriteTableRecord::TableWriteState state = c_writeTableRecord.tableWriteState;
+ c_writeTableRecord.tableWriteState = WriteTableRecord::IDLE;
+ switch (state) {
+ case WriteTableRecord::IDLE:
+ case WriteTableRecord::WRITE_ADD_TABLE_MASTER :
+ case WriteTableRecord::WRITE_ADD_TABLE_SLAVE :
+ case WriteTableRecord::WRITE_RESTART_FROM_MASTER :
+ case WriteTableRecord::WRITE_RESTART_FROM_OWN :
+ ndbrequire(false);
+ break;
+ case WriteTableRecord::CALLBACK:
+ jam();
+ execute(signal, c_writeTableRecord.m_callback, 0);
+ return;
+ }
+ ndbrequire(false);
+}//Dbdict::closeWriteTableConf()
+
+void Dbdict::startReadTableFile(Signal* signal, Uint32 tableId)
+{
+ //globalSignalLoggers.log(number(), "startReadTableFile");
+ ndbrequire(!c_readTableRecord.inUse);
+
+ FsConnectRecordPtr fsPtr;
+ c_fsConnectRecordPool.getPtr(fsPtr, getFsConnRecord());
+ c_readTableRecord.inUse = true;
+ c_readTableRecord.tableId = tableId;
+ fsPtr.p->fsState = FsConnectRecord::OPEN_READ_TAB_FILE1;
+ openTableFile(signal, 0, fsPtr.i, tableId, false);
+}//Dbdict::startReadTableFile()
+
+void Dbdict::openReadTableRef(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ fsPtr.p->fsState = FsConnectRecord::OPEN_READ_TAB_FILE2;
+ openTableFile(signal, 1, fsPtr.i, c_readTableRecord.tableId, false);
+ return;
+}//Dbdict::openReadTableConf()
+
+void Dbdict::readTableFile(Signal* signal, Uint32 filePtr, Uint32 fsConPtr)
+{
+ FsReadWriteReq * const fsRWReq = (FsReadWriteReq *)&signal->theData[0];
+
+ fsRWReq->filePointer = filePtr;
+ fsRWReq->userReference = reference();
+ fsRWReq->userPointer = fsConPtr;
+ fsRWReq->operationFlag = 0; // Initialise before bit changes
+ FsReadWriteReq::setSyncFlag(fsRWReq->operationFlag, 0);
+ FsReadWriteReq::setFormatFlag(fsRWReq->operationFlag,
+ FsReadWriteReq::fsFormatArrayOfPages);
+ fsRWReq->varIndex = ZALLOCATE;
+ fsRWReq->numberOfPages = c_readTableRecord.noOfPages;
+ fsRWReq->data.arrayOfPages.varIndex = c_readTableRecord.pageId;
+ fsRWReq->data.arrayOfPages.fileOffset = 0; // Write to file page 0
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 8, JBA);
+}//readTableFile()
+
+void Dbdict::readTableConf(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ /* ---------------------------------------------------------------- */
+ // Verify the data read from disk
+ /* ---------------------------------------------------------------- */
+ bool crashInd;
+ if (fsPtr.p->fsState == FsConnectRecord::READ_TAB_FILE1) {
+ jam();
+ crashInd = false;
+ } else {
+ jam();
+ crashInd = true;
+ }//if
+
+ PageRecordPtr tmpPagePtr;
+ c_pageRecordArray.getPtr(tmpPagePtr, c_readTableRecord.pageId);
+ Uint32 sz = c_readTableRecord.noOfPages * ZSIZE_OF_PAGES_IN_WORDS;
+ Uint32 chk = computeChecksum((const Uint32*)tmpPagePtr.p, sz);
+
+ ndbrequire((chk == 0) || !crashInd);
+ if(chk != 0){
+ jam();
+ ndbrequire(fsPtr.p->fsState == FsConnectRecord::READ_TAB_FILE1);
+ readTableRef(signal, fsPtr);
+ return;
+ }//if
+
+ fsPtr.p->fsState = FsConnectRecord::CLOSE_READ_TAB_FILE;
+ closeFile(signal, fsPtr.p->filePtr, fsPtr.i);
+ return;
+}//Dbdict::readTableConf()
+
+void Dbdict::readTableRef(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ fsPtr.p->fsState = FsConnectRecord::OPEN_READ_TAB_FILE2;
+ openTableFile(signal, 1, fsPtr.i, c_readTableRecord.tableId, false);
+ return;
+}//Dbdict::readTableRef()
+
+void Dbdict::closeReadTableConf(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ c_fsConnectRecordPool.release(fsPtr);
+ c_readTableRecord.inUse = false;
+
+ execute(signal, c_readTableRecord.m_callback, 0);
+ return;
+}//Dbdict::closeReadTableConf()
+
+/* ---------------------------------------------------------------- */
+// Routines to handle Read/Write of Schema Files
+/* ---------------------------------------------------------------- */
+void
+Dbdict::updateSchemaState(Signal* signal, Uint32 tableId,
+ SchemaFile::TableEntry* te, Callback* callback){
+
+ jam();
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.schemaPage);
+
+ ndbrequire(tableId < c_tableRecordPool.getSize());
+ SchemaFile::TableEntry * tableEntry = getTableEntry(pagePtr.p, tableId);
+
+ SchemaFile::TableState newState =
+ (SchemaFile::TableState)te->m_tableState;
+ SchemaFile::TableState oldState =
+ (SchemaFile::TableState)tableEntry->m_tableState;
+
+ Uint32 newVersion = te->m_tableVersion;
+ Uint32 oldVersion = tableEntry->m_tableVersion;
+
+ bool ok = false;
+ switch(newState){
+ case SchemaFile::ADD_STARTED:
+ jam();
+ ok = true;
+ ndbrequire((oldVersion + 1) == newVersion);
+ ndbrequire(oldState == SchemaFile::INIT ||
+ oldState == SchemaFile::DROP_TABLE_COMMITTED);
+ break;
+ case SchemaFile::TABLE_ADD_COMMITTED:
+ jam();
+ ok = true;
+ ndbrequire(newVersion == oldVersion);
+ ndbrequire(oldState == SchemaFile::ADD_STARTED);
+ break;
+ case SchemaFile::ALTER_TABLE_COMMITTED:
+ jam();
+ ok = true;
+ ndbrequire((oldVersion + 1) == newVersion);
+ ndbrequire(oldState == SchemaFile::TABLE_ADD_COMMITTED ||
+ oldState == SchemaFile::ALTER_TABLE_COMMITTED);
+ break;
+ case SchemaFile::DROP_TABLE_STARTED:
+ jam();
+ case SchemaFile::DROP_TABLE_COMMITTED:
+ jam();
+ ok = true;
+ ndbrequire(false);
+ break;
+ case SchemaFile::INIT:
+ jam();
+ ok = true;
+ ndbrequire((oldState == SchemaFile::ADD_STARTED));
+ }//if
+ ndbrequire(ok);
+
+ * tableEntry = * te;
+ computeChecksum((SchemaFile*)pagePtr.p);
+
+ ndbrequire(c_writeSchemaRecord.inUse == false);
+ c_writeSchemaRecord.inUse = true;
+
+ c_writeSchemaRecord.pageId = c_schemaRecord.schemaPage;
+ c_writeSchemaRecord.m_callback = * callback;
+
+ startWriteSchemaFile(signal);
+}
+
+void Dbdict::startWriteSchemaFile(Signal* signal)
+{
+ FsConnectRecordPtr fsPtr;
+ c_fsConnectRecordPool.getPtr(fsPtr, getFsConnRecord());
+ fsPtr.p->fsState = FsConnectRecord::OPEN_WRITE_SCHEMA;
+ openSchemaFile(signal, 0, fsPtr.i, true);
+ c_writeSchemaRecord.noOfSchemaFilesHandled = 0;
+}//Dbdict::startWriteSchemaFile()
+
+void Dbdict::openSchemaFile(Signal* signal,
+ Uint32 fileNo,
+ Uint32 fsConPtr,
+ bool writeFlag)
+{
+ FsOpenReq * const fsOpenReq = (FsOpenReq *)&signal->theData[0];
+ fsOpenReq->userReference = reference();
+ fsOpenReq->userPointer = fsConPtr;
+ if (writeFlag) {
+ jam();
+ fsOpenReq->fileFlags =
+ FsOpenReq::OM_WRITEONLY |
+ FsOpenReq::OM_TRUNCATE |
+ FsOpenReq::OM_CREATE |
+ FsOpenReq::OM_SYNC;
+ } else {
+ jam();
+ fsOpenReq->fileFlags = FsOpenReq::OM_READONLY;
+ }//if
+ fsOpenReq->fileNumber[3] = 0; // Initialise before byte changes
+ FsOpenReq::setVersion(fsOpenReq->fileNumber, 1);
+ FsOpenReq::setSuffix(fsOpenReq->fileNumber, FsOpenReq::S_SCHEMALOG);
+ FsOpenReq::v1_setDisk(fsOpenReq->fileNumber, (fileNo + 1));
+ FsOpenReq::v1_setTable(fsOpenReq->fileNumber, (Uint32)-1);
+ FsOpenReq::v1_setFragment(fsOpenReq->fileNumber, (Uint32)-1);
+ FsOpenReq::v1_setS(fsOpenReq->fileNumber, (Uint32)-1);
+ FsOpenReq::v1_setP(fsOpenReq->fileNumber, 0);
+/* ---------------------------------------------------------------- */
+// File name : D1/DBDICT/P0.SchemaLog
+// D1 means Disk 1 (set by fileNo + 1). Writes to both D1 and D2
+// SchemaLog indicates that this is a file giving a list of current tables.
+/* ---------------------------------------------------------------- */
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, FsOpenReq::SignalLength, JBA);
+}//openSchemaFile()
+
+void Dbdict::writeSchemaFile(Signal* signal, Uint32 filePtr, Uint32 fsConPtr)
+{
+ FsReadWriteReq * const fsRWReq = (FsReadWriteReq *)&signal->theData[0];
+
+ fsRWReq->filePointer = filePtr;
+ fsRWReq->userReference = reference();
+ fsRWReq->userPointer = fsConPtr;
+ fsRWReq->operationFlag = 0; // Initialise before bit changes
+ FsReadWriteReq::setSyncFlag(fsRWReq->operationFlag, 1);
+ FsReadWriteReq::setFormatFlag(fsRWReq->operationFlag,
+ FsReadWriteReq::fsFormatArrayOfPages);
+ fsRWReq->varIndex = ZALLOCATE;
+ fsRWReq->numberOfPages = 1;
+// Write from memory page
+ fsRWReq->data.arrayOfPages.varIndex = c_writeSchemaRecord.pageId;
+ fsRWReq->data.arrayOfPages.fileOffset = 0; // Write to file page 0
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+}//writeSchemaFile()
+
+void Dbdict::writeSchemaConf(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ fsPtr.p->fsState = FsConnectRecord::CLOSE_WRITE_SCHEMA;
+ closeFile(signal, fsPtr.p->filePtr, fsPtr.i);
+ return;
+}//Dbdict::writeSchemaConf()
+
+void Dbdict::closeFile(Signal* signal, Uint32 filePtr, Uint32 fsConPtr)
+{
+ FsCloseReq * const fsCloseReq = (FsCloseReq *)&signal->theData[0];
+ fsCloseReq->filePointer = filePtr;
+ fsCloseReq->userReference = reference();
+ fsCloseReq->userPointer = fsConPtr;
+ FsCloseReq::setRemoveFileFlag(fsCloseReq->fileFlag, false);
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, FsCloseReq::SignalLength, JBA);
+ return;
+}//closeFile()
+
+void Dbdict::closeWriteSchemaConf(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ c_writeSchemaRecord.noOfSchemaFilesHandled++;
+ if (c_writeSchemaRecord.noOfSchemaFilesHandled < 2) {
+ jam();
+ fsPtr.p->fsState = FsConnectRecord::OPEN_WRITE_SCHEMA;
+ openSchemaFile(signal, 1, fsPtr.i, true);
+ return;
+ }
+ ndbrequire(c_writeSchemaRecord.noOfSchemaFilesHandled == 2);
+
+ c_fsConnectRecordPool.release(fsPtr);
+
+ c_writeSchemaRecord.inUse = false;
+ execute(signal, c_writeSchemaRecord.m_callback, 0);
+ return;
+}//Dbdict::closeWriteSchemaConf()
+
+void Dbdict::startReadSchemaFile(Signal* signal)
+{
+ //globalSignalLoggers.log(number(), "startReadSchemaFile");
+ FsConnectRecordPtr fsPtr;
+ c_fsConnectRecordPool.getPtr(fsPtr, getFsConnRecord());
+ fsPtr.p->fsState = FsConnectRecord::OPEN_READ_SCHEMA1;
+ openSchemaFile(signal, 0, fsPtr.i, false);
+}//Dbdict::startReadSchemaFile()
+
+void Dbdict::openReadSchemaRef(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ fsPtr.p->fsState = FsConnectRecord::OPEN_READ_SCHEMA2;
+ openSchemaFile(signal, 1, fsPtr.i, false);
+}//Dbdict::openReadSchemaRef()
+
+void Dbdict::readSchemaFile(Signal* signal, Uint32 filePtr, Uint32 fsConPtr)
+{
+ FsReadWriteReq * const fsRWReq = (FsReadWriteReq *)&signal->theData[0];
+
+ fsRWReq->filePointer = filePtr;
+ fsRWReq->userReference = reference();
+ fsRWReq->userPointer = fsConPtr;
+ fsRWReq->operationFlag = 0; // Initialise before bit changes
+ FsReadWriteReq::setSyncFlag(fsRWReq->operationFlag, 0);
+ FsReadWriteReq::setFormatFlag(fsRWReq->operationFlag,
+ FsReadWriteReq::fsFormatArrayOfPages);
+ fsRWReq->varIndex = ZALLOCATE;
+ fsRWReq->numberOfPages = 1;
+ fsRWReq->data.arrayOfPages.varIndex = c_readSchemaRecord.pageId;
+ fsRWReq->data.arrayOfPages.fileOffset = 0;
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 8, JBA);
+}//readSchemaFile()
+
+void Dbdict::readSchemaConf(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+/* ---------------------------------------------------------------- */
+// Verify the data read from disk
+/* ---------------------------------------------------------------- */
+ bool crashInd;
+ if (fsPtr.p->fsState == FsConnectRecord::READ_SCHEMA1) {
+ jam();
+ crashInd = false;
+ } else {
+ jam();
+ crashInd = true;
+ }//if
+ PageRecordPtr tmpPagePtr;
+ c_pageRecordArray.getPtr(tmpPagePtr, c_readSchemaRecord.pageId);
+
+ Uint32 sz = ZSIZE_OF_PAGES_IN_WORDS;
+ Uint32 chk = computeChecksum((const Uint32*)tmpPagePtr.p, sz);
+
+ ndbrequire((chk == 0) || !crashInd);
+
+ if (chk != 0){
+ jam();
+ ndbrequire(fsPtr.p->fsState == FsConnectRecord::READ_SCHEMA1);
+ readSchemaRef(signal, fsPtr);
+ return;
+ }//if
+ fsPtr.p->fsState = FsConnectRecord::CLOSE_READ_SCHEMA;
+ closeFile(signal, fsPtr.p->filePtr, fsPtr.i);
+ return;
+}//Dbdict::readSchemaConf()
+
+void Dbdict::readSchemaRef(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ fsPtr.p->fsState = FsConnectRecord::OPEN_READ_SCHEMA2;
+ openSchemaFile(signal, 1, fsPtr.i, false);
+ return;
+}//Dbdict::readSchemaRef()
+
+void Dbdict::closeReadSchemaConf(Signal* signal,
+ FsConnectRecordPtr fsPtr)
+{
+ c_fsConnectRecordPool.release(fsPtr);
+ ReadSchemaRecord::SchemaReadState state = c_readSchemaRecord.schemaReadState;
+ c_readSchemaRecord.schemaReadState = ReadSchemaRecord::IDLE;
+
+ switch(state) {
+ case ReadSchemaRecord::INITIAL_READ :
+ jam();
+ sendNDB_STTORRY(signal);
+ break;
+
+ default :
+ ndbrequire(false);
+ break;
+
+ }//switch
+}//Dbdict::closeReadSchemaConf()
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: INITIALISATION MODULE ------------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* This module contains initialisation of data at start/restart. */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+Dbdict::Dbdict(const class Configuration & conf):
+ SimulatedBlock(DBDICT, conf),
+ c_tableRecordHash(c_tableRecordPool),
+ c_attributeRecordHash(c_attributeRecordPool),
+ c_triggerRecordHash(c_triggerRecordPool),
+ c_opCreateTable(c_opRecordPool),
+ c_opDropTable(c_opRecordPool),
+ c_opCreateIndex(c_opRecordPool),
+ c_opDropIndex(c_opRecordPool),
+ c_opAlterIndex(c_opRecordPool),
+ c_opBuildIndex(c_opRecordPool),
+ c_opCreateEvent(c_opRecordPool),
+ c_opSubEvent(c_opRecordPool),
+ c_opDropEvent(c_opRecordPool),
+ c_opSignalUtil(c_opRecordPool),
+ c_opCreateTrigger(c_opRecordPool),
+ c_opDropTrigger(c_opRecordPool),
+ c_opAlterTrigger(c_opRecordPool),
+ c_opRecordSequence(0)
+{
+ BLOCK_CONSTRUCTOR(Dbdict);
+
+ const Properties * p = conf.getOwnProperties();
+ ndbrequire(p != 0);
+
+ p->get("MaxNoOfTriggers", &c_maxNoOfTriggers);
+ // Transit signals
+ addRecSignal(GSN_DUMP_STATE_ORD, &Dbdict::execDUMP_STATE_ORD);
+ addRecSignal(GSN_GET_TABINFOREQ, &Dbdict::execGET_TABINFOREQ);
+ addRecSignal(GSN_GET_TABLEID_REQ, &Dbdict::execGET_TABLEDID_REQ);
+ addRecSignal(GSN_GET_TABINFO_CONF, &Dbdict::execGET_TABINFO_CONF);
+ addRecSignal(GSN_CONTINUEB, &Dbdict::execCONTINUEB);
+
+ addRecSignal(GSN_CREATE_TABLE_REQ, &Dbdict::execCREATE_TABLE_REQ);
+ addRecSignal(GSN_CREATE_TAB_REQ, &Dbdict::execCREATE_TAB_REQ);
+ addRecSignal(GSN_CREATE_TAB_REF, &Dbdict::execCREATE_TAB_REF);
+ addRecSignal(GSN_CREATE_TAB_CONF, &Dbdict::execCREATE_TAB_CONF);
+ addRecSignal(GSN_CREATE_FRAGMENTATION_REF, &Dbdict::execCREATE_FRAGMENTATION_REF);
+ addRecSignal(GSN_CREATE_FRAGMENTATION_CONF, &Dbdict::execCREATE_FRAGMENTATION_CONF);
+ addRecSignal(GSN_DIADDTABCONF, &Dbdict::execDIADDTABCONF);
+ addRecSignal(GSN_DIADDTABREF, &Dbdict::execDIADDTABREF);
+ addRecSignal(GSN_ADD_FRAGREQ, &Dbdict::execADD_FRAGREQ);
+ addRecSignal(GSN_TAB_COMMITCONF, &Dbdict::execTAB_COMMITCONF);
+ addRecSignal(GSN_TAB_COMMITREF, &Dbdict::execTAB_COMMITREF);
+ addRecSignal(GSN_ALTER_TABLE_REQ, &Dbdict::execALTER_TABLE_REQ);
+ addRecSignal(GSN_ALTER_TAB_REQ, &Dbdict::execALTER_TAB_REQ);
+ addRecSignal(GSN_ALTER_TAB_REF, &Dbdict::execALTER_TAB_REF);
+ addRecSignal(GSN_ALTER_TAB_CONF, &Dbdict::execALTER_TAB_CONF);
+
+ // Index signals
+ addRecSignal(GSN_CREATE_INDX_REQ, &Dbdict::execCREATE_INDX_REQ);
+ addRecSignal(GSN_CREATE_INDX_CONF, &Dbdict::execCREATE_INDX_CONF);
+ addRecSignal(GSN_CREATE_INDX_REF, &Dbdict::execCREATE_INDX_REF);
+
+ addRecSignal(GSN_ALTER_INDX_REQ, &Dbdict::execALTER_INDX_REQ);
+ addRecSignal(GSN_ALTER_INDX_CONF, &Dbdict::execALTER_INDX_CONF);
+ addRecSignal(GSN_ALTER_INDX_REF, &Dbdict::execALTER_INDX_REF);
+
+ addRecSignal(GSN_CREATE_TABLE_CONF, &Dbdict::execCREATE_TABLE_CONF);
+ addRecSignal(GSN_CREATE_TABLE_REF, &Dbdict::execCREATE_TABLE_REF);
+
+ addRecSignal(GSN_DROP_INDX_REQ, &Dbdict::execDROP_INDX_REQ);
+ addRecSignal(GSN_DROP_INDX_CONF, &Dbdict::execDROP_INDX_CONF);
+ addRecSignal(GSN_DROP_INDX_REF, &Dbdict::execDROP_INDX_REF);
+
+ addRecSignal(GSN_DROP_TABLE_CONF, &Dbdict::execDROP_TABLE_CONF);
+ addRecSignal(GSN_DROP_TABLE_REF, &Dbdict::execDROP_TABLE_REF);
+
+ addRecSignal(GSN_BUILDINDXREQ, &Dbdict::execBUILDINDXREQ);
+ addRecSignal(GSN_BUILDINDXCONF, &Dbdict::execBUILDINDXCONF);
+ addRecSignal(GSN_BUILDINDXREF, &Dbdict::execBUILDINDXREF);
+
+ // Util signals
+ addRecSignal(GSN_UTIL_PREPARE_CONF, &Dbdict::execUTIL_PREPARE_CONF);
+ addRecSignal(GSN_UTIL_PREPARE_REF, &Dbdict::execUTIL_PREPARE_REF);
+
+ addRecSignal(GSN_UTIL_EXECUTE_CONF, &Dbdict::execUTIL_EXECUTE_CONF);
+ addRecSignal(GSN_UTIL_EXECUTE_REF, &Dbdict::execUTIL_EXECUTE_REF);
+
+ addRecSignal(GSN_UTIL_RELEASE_CONF, &Dbdict::execUTIL_RELEASE_CONF);
+ addRecSignal(GSN_UTIL_RELEASE_REF, &Dbdict::execUTIL_RELEASE_REF);
+
+ // Event signals
+ addRecSignal(GSN_CREATE_EVNT_REQ, &Dbdict::execCREATE_EVNT_REQ);
+ addRecSignal(GSN_CREATE_EVNT_CONF, &Dbdict::execCREATE_EVNT_CONF);
+ addRecSignal(GSN_CREATE_EVNT_REF, &Dbdict::execCREATE_EVNT_REF);
+
+ addRecSignal(GSN_CREATE_SUBID_CONF, &Dbdict::execCREATE_SUBID_CONF);
+ addRecSignal(GSN_CREATE_SUBID_REF, &Dbdict::execCREATE_SUBID_REF);
+
+ addRecSignal(GSN_SUB_CREATE_CONF, &Dbdict::execSUB_CREATE_CONF);
+ addRecSignal(GSN_SUB_CREATE_REF, &Dbdict::execSUB_CREATE_REF);
+
+ addRecSignal(GSN_SUB_START_REQ, &Dbdict::execSUB_START_REQ);
+ addRecSignal(GSN_SUB_START_CONF, &Dbdict::execSUB_START_CONF);
+ addRecSignal(GSN_SUB_START_REF, &Dbdict::execSUB_START_REF);
+
+ addRecSignal(GSN_SUB_STOP_REQ, &Dbdict::execSUB_STOP_REQ);
+ addRecSignal(GSN_SUB_STOP_CONF, &Dbdict::execSUB_STOP_CONF);
+ addRecSignal(GSN_SUB_STOP_REF, &Dbdict::execSUB_STOP_REF);
+
+ addRecSignal(GSN_SUB_SYNC_CONF, &Dbdict::execSUB_SYNC_CONF);
+ addRecSignal(GSN_SUB_SYNC_REF, &Dbdict::execSUB_SYNC_REF);
+
+ addRecSignal(GSN_DROP_EVNT_REQ, &Dbdict::execDROP_EVNT_REQ);
+
+ addRecSignal(GSN_SUB_REMOVE_REQ, &Dbdict::execSUB_REMOVE_REQ);
+ addRecSignal(GSN_SUB_REMOVE_CONF, &Dbdict::execSUB_REMOVE_CONF);
+ addRecSignal(GSN_SUB_REMOVE_REF, &Dbdict::execSUB_REMOVE_REF);
+
+ // Trigger signals
+ addRecSignal(GSN_CREATE_TRIG_REQ, &Dbdict::execCREATE_TRIG_REQ);
+ addRecSignal(GSN_CREATE_TRIG_CONF, &Dbdict::execCREATE_TRIG_CONF);
+ addRecSignal(GSN_CREATE_TRIG_REF, &Dbdict::execCREATE_TRIG_REF);
+ addRecSignal(GSN_ALTER_TRIG_REQ, &Dbdict::execALTER_TRIG_REQ);
+ addRecSignal(GSN_ALTER_TRIG_CONF, &Dbdict::execALTER_TRIG_CONF);
+ addRecSignal(GSN_ALTER_TRIG_REF, &Dbdict::execALTER_TRIG_REF);
+ addRecSignal(GSN_DROP_TRIG_REQ, &Dbdict::execDROP_TRIG_REQ);
+ addRecSignal(GSN_DROP_TRIG_CONF, &Dbdict::execDROP_TRIG_CONF);
+ addRecSignal(GSN_DROP_TRIG_REF, &Dbdict::execDROP_TRIG_REF);
+
+ // Received signals
+ addRecSignal(GSN_HOT_SPAREREP, &Dbdict::execHOT_SPAREREP);
+ addRecSignal(GSN_GET_SCHEMA_INFOREQ, &Dbdict::execGET_SCHEMA_INFOREQ);
+ addRecSignal(GSN_SCHEMA_INFO, &Dbdict::execSCHEMA_INFO);
+ addRecSignal(GSN_SCHEMA_INFOCONF, &Dbdict::execSCHEMA_INFOCONF);
+ addRecSignal(GSN_DICTSTARTREQ, &Dbdict::execDICTSTARTREQ);
+ addRecSignal(GSN_READ_NODESCONF, &Dbdict::execREAD_NODESCONF);
+ addRecSignal(GSN_FSOPENCONF, &Dbdict::execFSOPENCONF);
+ addRecSignal(GSN_FSOPENREF, &Dbdict::execFSOPENREF);
+ addRecSignal(GSN_FSCLOSECONF, &Dbdict::execFSCLOSECONF);
+ addRecSignal(GSN_FSCLOSEREF, &Dbdict::execFSCLOSEREF);
+ addRecSignal(GSN_FSWRITECONF, &Dbdict::execFSWRITECONF);
+ addRecSignal(GSN_FSWRITEREF, &Dbdict::execFSWRITEREF);
+ addRecSignal(GSN_FSREADCONF, &Dbdict::execFSREADCONF);
+ addRecSignal(GSN_FSREADREF, &Dbdict::execFSREADREF);
+ addRecSignal(GSN_LQHFRAGCONF, &Dbdict::execLQHFRAGCONF);
+ addRecSignal(GSN_LQHADDATTCONF, &Dbdict::execLQHADDATTCONF);
+ addRecSignal(GSN_LQHADDATTREF, &Dbdict::execLQHADDATTREF);
+ addRecSignal(GSN_LQHFRAGREF, &Dbdict::execLQHFRAGREF);
+ addRecSignal(GSN_NDB_STTOR, &Dbdict::execNDB_STTOR);
+ addRecSignal(GSN_SIZEALT_REP, &Dbdict::execSIZEALT_REP);
+ addRecSignal(GSN_STTOR, &Dbdict::execSTTOR);
+ addRecSignal(GSN_TC_SCHVERCONF, &Dbdict::execTC_SCHVERCONF);
+ addRecSignal(GSN_NODE_FAILREP, &Dbdict::execNODE_FAILREP);
+ addRecSignal(GSN_INCL_NODEREQ, &Dbdict::execINCL_NODEREQ);
+ addRecSignal(GSN_API_FAILREQ, &Dbdict::execAPI_FAILREQ);
+
+ addRecSignal(GSN_WAIT_GCP_REF, &Dbdict::execWAIT_GCP_REF);
+ addRecSignal(GSN_WAIT_GCP_CONF, &Dbdict::execWAIT_GCP_CONF);
+
+ addRecSignal(GSN_LIST_TABLES_REQ, &Dbdict::execLIST_TABLES_REQ);
+
+ addRecSignal(GSN_DROP_TABLE_REQ, &Dbdict::execDROP_TABLE_REQ);
+
+ addRecSignal(GSN_PREP_DROP_TAB_REQ, &Dbdict::execPREP_DROP_TAB_REQ);
+ addRecSignal(GSN_PREP_DROP_TAB_REF, &Dbdict::execPREP_DROP_TAB_REF);
+ addRecSignal(GSN_PREP_DROP_TAB_CONF, &Dbdict::execPREP_DROP_TAB_CONF);
+
+ addRecSignal(GSN_DROP_TAB_REQ, &Dbdict::execDROP_TAB_REQ);
+ addRecSignal(GSN_DROP_TAB_REF, &Dbdict::execDROP_TAB_REF);
+ addRecSignal(GSN_DROP_TAB_CONF, &Dbdict::execDROP_TAB_CONF);
+}//Dbdict::Dbdict()
+
+Dbdict::~Dbdict()
+{
+}//Dbdict::~Dbdict()
+
+BLOCK_FUNCTIONS(Dbdict);
+
+void Dbdict::initCommonData()
+{
+/* ---------------------------------------------------------------- */
+// Initialise all common variables.
+/* ---------------------------------------------------------------- */
+ initRetrieveRecord(0, 0, 0);
+ initSchemaRecord();
+ initRestartRecord();
+ initSendSchemaRecord();
+ initReadTableRecord();
+ initWriteTableRecord();
+ initReadSchemaRecord();
+ initWriteSchemaRecord();
+
+ c_masterNodeId = ZNIL;
+ c_numberNode = 0;
+ c_noNodesFailed = 0;
+ c_failureNr = 0;
+ c_blockState = BS_IDLE;
+ c_packTable.m_state = PackTable::PTS_IDLE;
+ c_startPhase = 0;
+ c_restartType = 255; //Ensure not used restartType
+ c_tabinfoReceived = 0;
+ c_initialStart = false;
+ c_systemRestart = false;
+ c_initialNodeRestart = false;
+ c_nodeRestart = false;
+}//Dbdict::initCommonData()
+
+void Dbdict::initRecords()
+{
+ initNodeRecords();
+ initPageRecords();
+ initTableRecords();
+ initTriggerRecords();
+}//Dbdict::initRecords()
+
+void Dbdict::initSendSchemaRecord()
+{
+ c_sendSchemaRecord.noOfWords = (Uint32)-1;
+ c_sendSchemaRecord.pageId = RNIL;
+ c_sendSchemaRecord.noOfWordsCurrentlySent = 0;
+ c_sendSchemaRecord.noOfSignalsSentSinceDelay = 0;
+ c_sendSchemaRecord.inUse = false;
+ //c_sendSchemaRecord.sendSchemaState = SendSchemaRecord::IDLE;
+}//initSendSchemaRecord()
+
+void Dbdict::initReadTableRecord()
+{
+ c_readTableRecord.noOfPages = (Uint32)-1;
+ c_readTableRecord.pageId = RNIL;
+ c_readTableRecord.tableId = ZNIL;
+ c_readTableRecord.inUse = false;
+}//initReadTableRecord()
+
+void Dbdict::initWriteTableRecord()
+{
+ c_writeTableRecord.noOfPages = (Uint32)-1;
+ c_writeTableRecord.pageId = RNIL;
+ c_writeTableRecord.noOfTableFilesHandled = 3;
+ c_writeTableRecord.tableId = ZNIL;
+ c_writeTableRecord.tableWriteState = WriteTableRecord::IDLE;
+}//initWriteTableRecord()
+
+void Dbdict::initReadSchemaRecord()
+{
+ c_readSchemaRecord.pageId = RNIL;
+ c_readSchemaRecord.schemaReadState = ReadSchemaRecord::IDLE;
+}//initReadSchemaRecord()
+
+void Dbdict::initWriteSchemaRecord()
+{
+ c_writeSchemaRecord.inUse = false;
+ c_writeSchemaRecord.pageId = RNIL;
+ c_writeSchemaRecord.noOfSchemaFilesHandled = 3;
+}//initWriteSchemaRecord()
+
+void Dbdict::initRetrieveRecord(Signal* signal, Uint32 i, Uint32 returnCode)
+{
+ c_retrieveRecord.busyState = false;
+ c_retrieveRecord.blockRef = 0;
+ c_retrieveRecord.m_senderData = RNIL;
+ c_retrieveRecord.tableId = RNIL;
+ c_retrieveRecord.currentSent = 0;
+ c_retrieveRecord.retrievedNoOfPages = 0;
+ c_retrieveRecord.retrievedNoOfWords = 0;
+ c_retrieveRecord.m_useLongSig = false;
+}//initRetrieveRecord()
+
+void Dbdict::initSchemaRecord()
+{
+ c_schemaRecord.schemaPage = RNIL;
+}//Dbdict::initSchemaRecord()
+
+void Dbdict::initRestartRecord()
+{
+ c_restartRecord.gciToRestart = 0;
+ c_restartRecord.activeTable = ZNIL;
+}//Dbdict::initRestartRecord()
+
+void Dbdict::initNodeRecords()
+{
+ jam();
+ for (unsigned i = 1; i < MAX_NODES; i++) {
+ NodeRecordPtr nodePtr;
+ c_nodes.getPtr(nodePtr, i);
+ nodePtr.p->hotSpare = false;
+ nodePtr.p->nodeState = NodeRecord::API_NODE;
+ }//for
+}//Dbdict::initNodeRecords()
+
+void Dbdict::initPageRecords()
+{
+ c_schemaRecord.schemaPage = ZMAX_PAGES_OF_TABLE_DEFINITION;
+ c_schemaRecord.oldSchemaPage = ZMAX_PAGES_OF_TABLE_DEFINITION + 1;
+ c_retrieveRecord.retrievePage = ZMAX_PAGES_OF_TABLE_DEFINITION + 2;
+ ndbrequire(ZNUMBER_OF_PAGES >= (2 * ZMAX_PAGES_OF_TABLE_DEFINITION + 2));
+}//Dbdict::initPageRecords()
+
+void Dbdict::initTableRecords()
+{
+ TableRecordPtr tablePtr;
+ while (1) {
+ jam();
+ c_tableRecordPool.seize(tablePtr);
+ if (tablePtr.i == RNIL) {
+ jam();
+ break;
+ }//if
+ initialiseTableRecord(tablePtr);
+ }//while
+}//Dbdict::initTableRecords()
+
+void Dbdict::initialiseTableRecord(TableRecordPtr tablePtr)
+{
+ tablePtr.p->activePage = RNIL;
+ tablePtr.p->filePtr[0] = RNIL;
+ tablePtr.p->filePtr[1] = RNIL;
+ tablePtr.p->firstAttribute = RNIL;
+ tablePtr.p->firstPage = RNIL;
+ tablePtr.p->lastAttribute = RNIL;
+ tablePtr.p->tableId = tablePtr.i;
+ tablePtr.p->tableVersion = (Uint32)-1;
+ tablePtr.p->tabState = TableRecord::NOT_DEFINED;
+ tablePtr.p->tabReturnState = TableRecord::TRS_IDLE;
+ tablePtr.p->storageType = DictTabInfo::MainMemory;
+ tablePtr.p->myConnect = RNIL;
+ tablePtr.p->fragmentType = DictTabInfo::AllNodesSmallTable;
+ tablePtr.p->fragmentKeyType = DictTabInfo::PrimaryKey;
+ memset(tablePtr.p->tableName, 0, sizeof(tablePtr.p->tableName));
+ tablePtr.p->gciTableCreated = 0;
+ tablePtr.p->noOfAttributes = ZNIL;
+ tablePtr.p->noOfNullAttr = 0;
+ tablePtr.p->frmLen = 0;
+ memset(tablePtr.p->frmData, 0, sizeof(tablePtr.p->frmData));
+ /*
+ tablePtr.p->lh3PageIndexBits = 0;
+ tablePtr.p->lh3DistrBits = 0;
+ tablePtr.p->lh3PageBits = 6;
+ */
+ tablePtr.p->kValue = 6;
+ tablePtr.p->localKeyLen = 1;
+ tablePtr.p->maxLoadFactor = 80;
+ tablePtr.p->minLoadFactor = 70;
+ tablePtr.p->noOfPrimkey = 1;
+ tablePtr.p->tupKeyLength = 1;
+ tablePtr.p->storedTable = true;
+ tablePtr.p->tableType = DictTabInfo::UserTable;
+ tablePtr.p->primaryTableId = RNIL;
+ // volatile elements
+ tablePtr.p->indexState = TableRecord::IS_UNDEFINED;
+ tablePtr.p->insertTriggerId = RNIL;
+ tablePtr.p->updateTriggerId = RNIL;
+ tablePtr.p->deleteTriggerId = RNIL;
+ tablePtr.p->customTriggerId = RNIL;
+ tablePtr.p->buildTriggerId = RNIL;
+ tablePtr.p->indexLocal = 0;
+}//Dbdict::initialiseTableRecord()
+
+void Dbdict::initTriggerRecords()
+{
+ TriggerRecordPtr triggerPtr;
+ while (1) {
+ jam();
+ c_triggerRecordPool.seize(triggerPtr);
+ if (triggerPtr.i == RNIL) {
+ jam();
+ break;
+ }//if
+ initialiseTriggerRecord(triggerPtr);
+ }//while
+}
+
+void Dbdict::initialiseTriggerRecord(TriggerRecordPtr triggerPtr)
+{
+ triggerPtr.p->triggerState = TriggerRecord::TS_NOT_DEFINED;
+ triggerPtr.p->triggerLocal = 0;
+ memset(triggerPtr.p->triggerName, 0, sizeof(triggerPtr.p->triggerName));
+ triggerPtr.p->triggerId = RNIL;
+ triggerPtr.p->tableId = RNIL;
+ triggerPtr.p->triggerType = (TriggerType::Value)~0;
+ triggerPtr.p->triggerActionTime = (TriggerActionTime::Value)~0;
+ triggerPtr.p->triggerEvent = (TriggerEvent::Value)~0;
+ triggerPtr.p->monitorReplicas = false;
+ triggerPtr.p->monitorAllAttributes = false;
+ triggerPtr.p->attributeMask.clear();
+ triggerPtr.p->indexId = RNIL;
+}
+
+Uint32 Dbdict::getFsConnRecord()
+{
+ FsConnectRecordPtr fsPtr;
+ c_fsConnectRecordPool.seize(fsPtr);
+ ndbrequire(fsPtr.i != RNIL);
+ fsPtr.p->filePtr = (Uint32)-1;
+ fsPtr.p->ownerPtr = RNIL;
+ fsPtr.p->fsState = FsConnectRecord::IDLE;
+ return fsPtr.i;
+}//Dbdict::getFsConnRecord()
+
+Uint32 Dbdict::getFreeTableRecord(Uint32 primaryTableId)
+{
+ Uint32 minId = (primaryTableId == RNIL ? 0 : primaryTableId + 1);
+ TableRecordPtr tablePtr;
+ TableRecordPtr firstTablePtr;
+ bool firstFound = false;
+ Uint32 tabSize = c_tableRecordPool.getSize();
+ for (tablePtr.i = minId; tablePtr.i < tabSize ; tablePtr.i++) {
+ jam();
+ c_tableRecordPool.getPtr(tablePtr);
+ if (tablePtr.p->tabState == TableRecord::NOT_DEFINED) {
+ jam();
+ initialiseTableRecord(tablePtr);
+ tablePtr.p->tabState = TableRecord::DEFINING;
+ firstFound = true;
+ firstTablePtr.i = tablePtr.i;
+ firstTablePtr.p = tablePtr.p;
+ break;
+ }//if
+ }//for
+ if (!firstFound) {
+ jam();
+ return RNIL;
+ }//if
+ bool secondFound = false;
+ for (tablePtr.i = firstTablePtr.i + 1; tablePtr.i < tabSize ; tablePtr.i++) {
+ jam();
+ c_tableRecordPool.getPtr(tablePtr);
+ if (tablePtr.p->tabState == TableRecord::NOT_DEFINED) {
+ jam();
+ initialiseTableRecord(tablePtr);
+ tablePtr.p->tabState = TableRecord::REORG_TABLE_PREPARED;
+ tablePtr.p->secondTable = firstTablePtr.i;
+ firstTablePtr.p->secondTable = tablePtr.i;
+ secondFound = true;
+ break;
+ }//if
+ }//for
+ if (!secondFound) {
+ jam();
+ firstTablePtr.p->tabState = TableRecord::NOT_DEFINED;
+ return RNIL;
+ }//if
+ return firstTablePtr.i;
+}//Dbdict::getFreeTableRecord()
+
+Uint32 Dbdict::getFreeTriggerRecord()
+{
+ const Uint32 size = c_triggerRecordPool.getSize();
+ TriggerRecordPtr triggerPtr;
+ for (triggerPtr.i = 0; triggerPtr.i < size; triggerPtr.i++) {
+ jam();
+ c_triggerRecordPool.getPtr(triggerPtr);
+ if (triggerPtr.p->triggerState == TriggerRecord::TS_NOT_DEFINED) {
+ jam();
+ initialiseTriggerRecord(triggerPtr);
+ return triggerPtr.i;
+ }
+ }
+ return RNIL;
+}
+
+bool
+Dbdict::getNewAttributeRecord(TableRecordPtr tablePtr,
+ AttributeRecordPtr & attrPtr)
+{
+ c_attributeRecordPool.seize(attrPtr);
+ if(attrPtr.i == RNIL){
+ return false;
+ }
+
+ memset(attrPtr.p->attributeName, 0, sizeof(attrPtr.p->attributeName));
+ attrPtr.p->attributeDescriptor = 0x00012255; //Default value
+ attrPtr.p->attributeId = ZNIL;
+ attrPtr.p->nextAttrInTable = RNIL;
+ attrPtr.p->tupleKey = 0;
+ memset(attrPtr.p->defaultValue, 0, sizeof(attrPtr.p->defaultValue));
+
+ /* ---------------------------------------------------------------- */
+ // A free attribute record has been acquired. We will now link it
+ // to the table record.
+ /* ---------------------------------------------------------------- */
+ if (tablePtr.p->lastAttribute == RNIL) {
+ jam();
+ tablePtr.p->firstAttribute = attrPtr.i;
+ } else {
+ jam();
+ AttributeRecordPtr lastAttrPtr;
+ c_attributeRecordPool.getPtr(lastAttrPtr, tablePtr.p->lastAttribute);
+ lastAttrPtr.p->nextAttrInTable = attrPtr.i;
+ }//if
+ tablePtr.p->lastAttribute = attrPtr.i;
+ return true;
+}//Dbdict::getNewAttributeRecord()
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: START/RESTART HANDLING ------------------------ */
+/* ---------------------------------------------------------------- */
+/* */
+/* This module contains the code that is common for all */
+/* start/restart types. */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+/* ---------------------------------------------------------------- */
+// This is sent as the first signal during start/restart.
+/* ---------------------------------------------------------------- */
+void Dbdict::execSTTOR(Signal* signal)
+{
+ jamEntry();
+ c_startPhase = signal->theData[1];
+ switch (c_startPhase) {
+ case 1:
+ initCommonData();
+ break;
+ case 3:
+ c_restartType = signal->theData[7]; /* valid if 3 */
+ ndbrequire(c_restartType == NodeState::ST_INITIAL_START ||
+ c_restartType == NodeState::ST_SYSTEM_RESTART ||
+ c_restartType == NodeState::ST_INITIAL_NODE_RESTART ||
+ c_restartType == NodeState::ST_NODE_RESTART);
+ break;
+ }
+ sendSTTORRY(signal);
+}//execSTTOR()
+
+void Dbdict::sendSTTORRY(Signal* signal)
+{
+ signal->theData[0] = 0; /* garbage SIGNAL KEY */
+ signal->theData[1] = 0; /* garbage SIGNAL VERSION NUMBER */
+ signal->theData[2] = 0; /* garbage */
+ signal->theData[3] = 1; /* first wanted start phase */
+ signal->theData[4] = 3; /* get type of start */
+ signal->theData[5] = ZNOMOREPHASES;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 6, JBB);
+}
+
+/* ---------------------------------------------------------------- */
+// We receive information about sizes of records.
+/* ---------------------------------------------------------------- */
+void Dbdict::execSIZEALT_REP(Signal* signal)
+{
+ jamEntry();
+ BlockReference tblockref;
+ tblockref = signal->theData[DictSizeAltReq::IND_BLOCK_REF];
+ Uint32 attributesize = signal->theData[DictSizeAltReq::IND_ATTRIBUTE];
+// Uint32 connectsize = signal->theData[DictSizeAltReq::IND_CONNECT];
+ Uint32 tablerecSize = signal->theData[DictSizeAltReq::IND_TABLE];
+
+ c_attributeRecordPool.setSize(attributesize);
+ c_attributeRecordHash.setSize(64);
+ c_fsConnectRecordPool.setSize(ZFS_CONNECT_SIZE);
+ c_nodes.setSize(MAX_NODES);
+ c_pageRecordArray.setSize(ZNUMBER_OF_PAGES);
+ c_tableRecordPool.setSize(tablerecSize);
+ c_tableRecordHash.setSize(tablerecSize);
+ c_triggerRecordPool.setSize(c_maxNoOfTriggers);
+ c_triggerRecordHash.setSize(c_maxNoOfTriggers);
+ c_opRecordPool.setSize(256); // XXX need config params
+ c_opCreateTable.setSize(8);
+ c_opDropTable.setSize(8);
+ c_opCreateIndex.setSize(8);
+ c_opCreateEvent.setSize(8);
+ c_opSubEvent.setSize(8);
+ c_opDropEvent.setSize(8);
+ c_opSignalUtil.setSize(8);
+ c_opDropIndex.setSize(8);
+ c_opAlterIndex.setSize(8);
+ c_opBuildIndex.setSize(8);
+ c_opCreateTrigger.setSize(8);
+ c_opDropTrigger.setSize(8);
+ c_opAlterTrigger.setSize(8);
+
+ // Initialize BAT for interface to file system
+ PageRecordPtr pageRecPtr;
+ c_pageRecordArray.getPtr(pageRecPtr, 0);
+ NewVARIABLE* bat = allocateBat(2);
+ bat[1].WA = &pageRecPtr.p->word[0];
+ bat[1].nrr = ZNUMBER_OF_PAGES;
+ bat[1].ClusterSize = ZSIZE_OF_PAGES_IN_WORDS * 4;
+ bat[1].bits.q = ZLOG_SIZE_OF_PAGES_IN_WORDS; // 2**13 = 8192 elements
+ bat[1].bits.v = 5; // 32 bits per element
+
+ initRecords();
+ signal->theData[0] = DBDICT_REF;
+ sendSignal(tblockref, GSN_SIZEALT_ACK, signal, 2, JBB);
+}//execSIZEALT_REP()
+
+/* ---------------------------------------------------------------- */
+// Start phase signals sent by CNTR. We reply with NDB_STTORRY when
+// we completed this phase.
+/* ---------------------------------------------------------------- */
+void Dbdict::execNDB_STTOR(Signal* signal)
+{
+ jamEntry();
+ c_startPhase = signal->theData[2];
+ const Uint32 restartType = signal->theData[3];
+ if (restartType == NodeState::ST_INITIAL_START) {
+ jam();
+ c_initialStart = true;
+ } else if (restartType == NodeState::ST_SYSTEM_RESTART) {
+ jam();
+ c_systemRestart = true;
+ } else if (restartType == NodeState::ST_INITIAL_NODE_RESTART) {
+ jam();
+ c_initialNodeRestart = true;
+ } else if (restartType == NodeState::ST_NODE_RESTART) {
+ jam();
+ c_nodeRestart = true;
+ } else {
+ ndbrequire(false);
+ }//if
+ switch (c_startPhase) {
+ case 1:
+ jam();
+ initSchemaFile(signal);
+ break;
+ case 3:
+ jam();
+ signal->theData[0] = reference();
+ sendSignal(NDBCNTR_REF, GSN_READ_NODESREQ, signal, 1, JBB);
+ break;
+ case 6:
+ jam();
+ c_initialStart = false;
+ c_systemRestart = false;
+ c_initialNodeRestart = false;
+ c_nodeRestart = false;
+ sendNDB_STTORRY(signal);
+ break;
+ case 7:
+ // uses c_restartType
+ if(restartType == NodeState::ST_SYSTEM_RESTART &&
+ c_masterNodeId == getOwnNodeId()){
+ rebuildIndexes(signal, 0);
+ return;
+ }
+ sendNDB_STTORRY(signal);
+ break;
+ default:
+ jam();
+ sendNDB_STTORRY(signal);
+ break;
+ }//switch
+}//execNDB_STTOR()
+
+void Dbdict::sendNDB_STTORRY(Signal* signal)
+{
+ signal->theData[0] = reference();
+ sendSignal(NDBCNTR_REF, GSN_NDB_STTORRY, signal, 1, JBB);
+ return;
+}//sendNDB_STTORRY()
+
+/* ---------------------------------------------------------------- */
+// We receive the information about which nodes that are up and down.
+/* ---------------------------------------------------------------- */
+void Dbdict::execREAD_NODESCONF(Signal* signal)
+{
+ jamEntry();
+
+ ReadNodesConf * const readNodes = (ReadNodesConf *)&signal->theData[0];
+ c_numberNode = readNodes->noOfNodes;
+ c_masterNodeId = readNodes->masterNodeId;
+
+ c_noNodesFailed = 0;
+ c_aliveNodes.clear();
+ for (unsigned i = 1; i < MAX_NDB_NODES; i++) {
+ jam();
+ NodeRecordPtr nodePtr;
+ c_nodes.getPtr(nodePtr, i);
+
+ if (NodeBitmask::get(readNodes->allNodes, i)) {
+ jam();
+ nodePtr.p->nodeState = NodeRecord::NDB_NODE_ALIVE;
+ if (NodeBitmask::get(readNodes->inactiveNodes, i)) {
+ jam();
+ /**-------------------------------------------------------------------
+ *
+ * THIS NODE IS DEFINED IN THE CLUSTER BUT IS NOT ALIVE CURRENTLY.
+ * WE ADD THE NODE TO THE SET OF FAILED NODES AND ALSO SET THE
+ * BLOCKSTATE TO BUSY TO AVOID ADDING TABLES WHILE NOT ALL NODES ARE
+ * ALIVE.
+ *------------------------------------------------------------------*/
+ nodePtr.p->nodeState = NodeRecord::NDB_NODE_DEAD;
+ c_noNodesFailed++;
+ } else {
+ c_aliveNodes.set(i);
+ }
+ }//if
+ }//for
+ sendNDB_STTORRY(signal);
+}//execREAD_NODESCONF()
+
+/* ---------------------------------------------------------------- */
+// HOT_SPAREREP informs DBDICT about which nodes that have become
+// hot spare nodes.
+/* ---------------------------------------------------------------- */
+void Dbdict::execHOT_SPAREREP(Signal* signal)
+{
+ Uint32 hotSpareNodes = 0;
+ jamEntry();
+ HotSpareRep * const hotSpare = (HotSpareRep*)&signal->theData[0];
+ for (unsigned i = 1; i < MAX_NDB_NODES; i++) {
+ if (NodeBitmask::get(hotSpare->theHotSpareNodes, i)) {
+ NodeRecordPtr nodePtr;
+ c_nodes.getPtr(nodePtr, i);
+ nodePtr.p->hotSpare = true;
+ hotSpareNodes++;
+ }//if
+ }//for
+ ndbrequire(hotSpareNodes == hotSpare->noHotSpareNodes);
+ c_noHotSpareNodes = hotSpareNodes;
+ return;
+}//execHOT_SPAREREP()
+
+void Dbdict::initSchemaFile(Signal* signal)
+{
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.schemaPage);
+ SchemaFile * schemaFile = (SchemaFile *)pagePtr.p;
+ initSchemaFile(schemaFile, 4 * ZSIZE_OF_PAGES_IN_WORDS);
+
+ if (c_initialStart || c_initialNodeRestart) {
+ jam();
+ ndbrequire(c_writeSchemaRecord.inUse == false);
+ c_writeSchemaRecord.inUse = true;
+ c_writeSchemaRecord.pageId = c_schemaRecord.schemaPage;
+
+ c_writeSchemaRecord.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::initSchemaFile_conf);
+
+ startWriteSchemaFile(signal);
+ } else if (c_systemRestart || c_nodeRestart) {
+ jam();
+ ndbrequire(c_readSchemaRecord.schemaReadState == ReadSchemaRecord::IDLE);
+ c_readSchemaRecord.pageId = c_schemaRecord.oldSchemaPage;
+ c_readSchemaRecord.schemaReadState = ReadSchemaRecord::INITIAL_READ;
+ startReadSchemaFile(signal);
+ } else {
+ ndbrequire(false);
+ }//if
+}//Dbdict::initSchemaFile()
+
+void
+Dbdict::initSchemaFile_conf(Signal* signal, Uint32 callbackData, Uint32 rv){
+ jam();
+ sendNDB_STTORRY(signal);
+}
+
+void
+Dbdict::activateIndexes(Signal* signal, Uint32 i)
+{
+ AlterIndxReq* req = (AlterIndxReq*)signal->getDataPtrSend();
+ TableRecordPtr tablePtr;
+ for (; i < c_tableRecordPool.getSize(); i++) {
+ tablePtr.i = i;
+ c_tableRecordPool.getPtr(tablePtr);
+ if (tablePtr.p->tabState != TableRecord::DEFINED)
+ continue;
+ if (! tablePtr.p->isIndex())
+ continue;
+ jam();
+ req->setUserRef(reference());
+ req->setConnectionPtr(i);
+ req->setTableId(tablePtr.p->primaryTableId);
+ req->setIndexId(tablePtr.i);
+ req->setIndexVersion(tablePtr.p->tableVersion);
+ req->setOnline(true);
+ if (c_restartType == NodeState::ST_SYSTEM_RESTART) {
+ if (c_masterNodeId != getOwnNodeId())
+ continue;
+ // from file index state is not defined currently
+ req->setRequestType(AlterIndxReq::RT_SYSTEMRESTART);
+ req->addRequestFlag((Uint32)RequestFlag::RF_NOBUILD);
+ }
+ else if (
+ c_restartType == NodeState::ST_NODE_RESTART ||
+ c_restartType == NodeState::ST_INITIAL_NODE_RESTART) {
+ // from master index must be online
+ if (tablePtr.p->indexState != TableRecord::IS_ONLINE)
+ continue;
+ req->setRequestType(AlterIndxReq::RT_NODERESTART);
+ // activate locally, rebuild not needed
+ req->addRequestFlag((Uint32)RequestFlag::RF_LOCAL);
+ req->addRequestFlag((Uint32)RequestFlag::RF_NOBUILD);
+ } else {
+ ndbrequire(false);
+ }
+ sendSignal(reference(), GSN_ALTER_INDX_REQ,
+ signal, AlterIndxReq::SignalLength, JBB);
+ return;
+ }
+ signal->theData[0] = reference();
+ sendSignal(c_restartRecord.returnBlockRef, GSN_DICTSTARTCONF,
+ signal, 1, JBB);
+}
+
+void
+Dbdict::rebuildIndexes(Signal* signal, Uint32 i){
+ BuildIndxReq* const req = (BuildIndxReq*)signal->getDataPtrSend();
+
+ TableRecordPtr indexPtr;
+ for (; i < c_tableRecordPool.getSize(); i++) {
+ indexPtr.i = i;
+ c_tableRecordPool.getPtr(indexPtr);
+ if (indexPtr.p->tabState != TableRecord::DEFINED)
+ continue;
+ if (! indexPtr.p->isIndex())
+ continue;
+
+ jam();
+
+ req->setUserRef(reference());
+ req->setConnectionPtr(i);
+ req->setRequestType(BuildIndxReq::RT_SYSTEMRESTART);
+ req->setBuildId(0); // not used
+ req->setBuildKey(0); // not used
+ req->setIndexType(indexPtr.p->tableType);
+ req->setIndexId(indexPtr.i);
+ req->setTableId(indexPtr.p->primaryTableId);
+ req->setParallelism(16);
+
+ // from file index state is not defined currently
+ if (indexPtr.p->storedTable) {
+ // rebuild not needed
+ req->addRequestFlag((Uint32)RequestFlag::RF_NOBUILD);
+ }
+
+ // send
+ sendSignal(reference(), GSN_BUILDINDXREQ,
+ signal, BuildIndxReq::SignalLength, JBB);
+ return;
+ }
+ sendNDB_STTORRY(signal);
+}
+
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: SYSTEM RESTART MODULE ------------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* This module contains code specific for system restart */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+/* ---------------------------------------------------------------- */
+// DIH asks DICT to read in table data from disk during system
+// restart. DIH also asks DICT to send information about which
+// tables that should be started as part of this system restart.
+// DICT will also activate the tables in TC as part of this process.
+/* ---------------------------------------------------------------- */
+void Dbdict::execDICTSTARTREQ(Signal* signal)
+{
+ jamEntry();
+ c_restartRecord.gciToRestart = signal->theData[0];
+ c_restartRecord.returnBlockRef = signal->theData[1];
+ if (c_nodeRestart || c_initialNodeRestart) {
+ jam();
+
+ CRASH_INSERTION(6000);
+
+ BlockReference dictRef = calcDictBlockRef(c_masterNodeId);
+ signal->theData[0] = getOwnNodeId();
+ sendSignal(dictRef, GSN_GET_SCHEMA_INFOREQ, signal, 1, JBB);
+ return;
+ }
+ ndbrequire(c_systemRestart);
+ ndbrequire(c_masterNodeId == getOwnNodeId());
+
+ c_schemaRecord.m_callback.m_callbackData = 0;
+ c_schemaRecord.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::masterRestart_checkSchemaStatusComplete);
+
+ c_restartRecord.activeTable = 0;
+ c_schemaRecord.schemaPage = c_schemaRecord.oldSchemaPage;
+ checkSchemaStatus(signal);
+}//execDICTSTARTREQ()
+
+void
+Dbdict::masterRestart_checkSchemaStatusComplete(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+
+ c_schemaRecord.schemaPage = ZMAX_PAGES_OF_TABLE_DEFINITION;
+
+ LinearSectionPtr ptr[3];
+
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.oldSchemaPage);
+
+ ptr[0].p = &pagePtr.p->word[0];
+ ptr[0].sz = ZSIZE_OF_PAGES_IN_WORDS;
+
+ c_sendSchemaRecord.m_SCHEMAINFO_Counter = c_aliveNodes;
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+
+ rg.m_nodes.clear(getOwnNodeId());
+ Callback c = { 0, 0 };
+ sendFragmentedSignal(rg,
+ GSN_SCHEMA_INFO,
+ signal,
+ 1, //SchemaInfo::SignalLength,
+ JBB,
+ ptr,
+ 1,
+ c);
+
+ PageRecordPtr newPagePtr;
+ c_pageRecordArray.getPtr(newPagePtr, c_schemaRecord.schemaPage);
+ memcpy(&newPagePtr.p->word[0], &pagePtr.p->word[0],
+ 4 * ZSIZE_OF_PAGES_IN_WORDS);
+
+ signal->theData[0] = getOwnNodeId();
+ sendSignal(reference(), GSN_SCHEMA_INFOCONF, signal, 1, JBB);
+}
+
+void
+Dbdict::execGET_SCHEMA_INFOREQ(Signal* signal){
+
+ const Uint32 ref = signal->getSendersBlockRef();
+ //const Uint32 senderData = signal->theData[0];
+
+ ndbrequire(c_sendSchemaRecord.inUse == false);
+ c_sendSchemaRecord.inUse = true;
+
+ LinearSectionPtr ptr[3];
+
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.schemaPage);
+
+ ptr[0].p = &pagePtr.p->word[0];
+ ptr[0].sz = ZSIZE_OF_PAGES_IN_WORDS;
+
+ Callback c = { safe_cast(&Dbdict::sendSchemaComplete), 0 };
+ sendFragmentedSignal(ref,
+ GSN_SCHEMA_INFO,
+ signal,
+ 1, //GetSchemaInfoConf::SignalLength,
+ JBB,
+ ptr,
+ 1,
+ c);
+}//Dbdict::execGET_SCHEMA_INFOREQ()
+
+void
+Dbdict::sendSchemaComplete(Signal * signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ ndbrequire(c_sendSchemaRecord.inUse == true);
+ c_sendSchemaRecord.inUse = false;
+
+}
+
+
+/* ---------------------------------------------------------------- */
+// We receive the schema info from master as part of all restarts
+// except the initial start where no tables exists.
+/* ---------------------------------------------------------------- */
+void Dbdict::execSCHEMA_INFO(Signal* signal)
+{
+ jamEntry();
+ if(!assembleFragments(signal)){
+ jam();
+ return;
+ }
+
+ if(getNodeState().getNodeRestartInProgress()){
+ CRASH_INSERTION(6001);
+ }
+
+ SegmentedSectionPtr schemaDataPtr;
+ signal->getSection(schemaDataPtr, 0);
+
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.schemaPage);
+ copy(&pagePtr.p->word[0], schemaDataPtr);
+ releaseSections(signal);
+
+ validateChecksum((SchemaFile*)pagePtr.p);
+
+ ndbrequire(signal->getSendersBlockRef() != reference());
+
+ /* ---------------------------------------------------------------- */
+ // Synchronise our view on data with other nodes in the cluster.
+ // This is an important part of restart handling where we will handle
+ // cases where the table have been added but only partially, where
+ // tables have been deleted but not completed the deletion yet and
+ // other scenarios needing synchronisation.
+ /* ---------------------------------------------------------------- */
+ c_schemaRecord.m_callback.m_callbackData = 0;
+ c_schemaRecord.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::restart_checkSchemaStatusComplete);
+ c_restartRecord.activeTable = 0;
+ checkSchemaStatus(signal);
+}//execSCHEMA_INFO()
+
+void
+Dbdict::restart_checkSchemaStatusComplete(Signal * signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+
+ ndbrequire(c_writeSchemaRecord.inUse == false);
+ c_writeSchemaRecord.inUse = true;
+ c_writeSchemaRecord.pageId = c_schemaRecord.schemaPage;
+ c_writeSchemaRecord.m_callback.m_callbackData = 0;
+ c_writeSchemaRecord.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::restart_writeSchemaConf);
+
+ startWriteSchemaFile(signal);
+}
+
+void
+Dbdict::restart_writeSchemaConf(Signal * signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+
+ if(c_systemRestart){
+ jam();
+ signal->theData[0] = getOwnNodeId();
+ sendSignal(calcDictBlockRef(c_masterNodeId), GSN_SCHEMA_INFOCONF,
+ signal, 1, JBB);
+ return;
+ }
+
+ ndbrequire(c_nodeRestart || c_initialNodeRestart);
+ c_blockState = BS_IDLE;
+ activateIndexes(signal, 0);
+ return;
+}
+
+void Dbdict::execSCHEMA_INFOCONF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(signal->getNoOfSections() == 0);
+
+/* ---------------------------------------------------------------- */
+// This signal is received in the master as part of system restart
+// from all nodes (including the master) after they have synchronised
+// their data with the master node's schema information.
+/* ---------------------------------------------------------------- */
+ const Uint32 nodeId = signal->theData[0];
+ c_sendSchemaRecord.m_SCHEMAINFO_Counter.clearWaitingFor(nodeId);
+
+ if (!c_sendSchemaRecord.m_SCHEMAINFO_Counter.done()){
+ jam();
+ return;
+ }//if
+ activateIndexes(signal, 0);
+}//execSCHEMA_INFOCONF()
+
+void Dbdict::checkSchemaStatus(Signal* signal)
+{
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.schemaPage);
+
+ PageRecordPtr oldPagePtr;
+ c_pageRecordArray.getPtr(oldPagePtr, c_schemaRecord.oldSchemaPage);
+
+ for (; c_restartRecord.activeTable < MAX_TABLES;
+ c_restartRecord.activeTable++) {
+ jam();
+
+ Uint32 tableId = c_restartRecord.activeTable;
+ SchemaFile::TableEntry *newEntry = getTableEntry(pagePtr.p, tableId);
+ SchemaFile::TableEntry *oldEntry = getTableEntry(oldPagePtr.p, tableId,
+ true);
+ SchemaFile::TableState schemaState =
+ (SchemaFile::TableState)newEntry->m_tableState;
+ SchemaFile::TableState oldSchemaState =
+ (SchemaFile::TableState)oldEntry->m_tableState;
+
+ if (c_restartRecord.activeTable >= c_tableRecordPool.getSize()) {
+ jam();
+ ndbrequire(schemaState == SchemaFile::INIT);
+ ndbrequire(oldSchemaState == SchemaFile::INIT);
+ continue;
+ }//if
+
+ switch(schemaState){
+ case SchemaFile::INIT:{
+ jam();
+ bool ok = false;
+ switch(oldSchemaState) {
+ case SchemaFile::INIT:
+ jam();
+ case SchemaFile::DROP_TABLE_COMMITTED:
+ jam();
+ ok = true;
+ jam();
+ break;
+
+ case SchemaFile::ADD_STARTED:
+ jam();
+ case SchemaFile::TABLE_ADD_COMMITTED:
+ jam();
+ case SchemaFile::DROP_TABLE_STARTED:
+ jam();
+ case SchemaFile::ALTER_TABLE_COMMITTED:
+ jam();
+ ok = true;
+ jam();
+ newEntry->m_tableState = SchemaFile::INIT;
+ restartDropTab(signal, tableId);
+ return;
+ }//switch
+ ndbrequire(ok);
+ break;
+ }
+ case SchemaFile::ADD_STARTED:{
+ jam();
+ bool ok = false;
+ switch(oldSchemaState) {
+ case SchemaFile::INIT:
+ jam();
+ case SchemaFile::DROP_TABLE_COMMITTED:
+ jam();
+ ok = true;
+ break;
+ case SchemaFile::ADD_STARTED:
+ jam();
+ case SchemaFile::DROP_TABLE_STARTED:
+ jam();
+ case SchemaFile::TABLE_ADD_COMMITTED:
+ jam();
+ case SchemaFile::ALTER_TABLE_COMMITTED:
+ jam();
+ ok = true;
+ //------------------------------------------------------------------
+ // Add Table was started but not completed. Will be dropped in all
+ // nodes. Update schema information (restore table version).
+ //------------------------------------------------------------------
+ newEntry->m_tableState = SchemaFile::INIT;
+ restartDropTab(signal, tableId);
+ return;
+ }
+ ndbrequire(ok);
+ break;
+ }
+ case SchemaFile::TABLE_ADD_COMMITTED:{
+ jam();
+ bool ok = false;
+ switch(oldSchemaState) {
+ case SchemaFile::INIT:
+ jam();
+ case SchemaFile::ADD_STARTED:
+ jam();
+ case SchemaFile::DROP_TABLE_STARTED:
+ jam();
+ case SchemaFile::DROP_TABLE_COMMITTED:
+ jam();
+ ok = true;
+ //------------------------------------------------------------------
+ // Table was added in the master node but not in our node. We can
+ // retrieve the table definition from the master.
+ //------------------------------------------------------------------
+ restartCreateTab(signal, tableId, oldEntry, false);
+ return;
+ break;
+ case SchemaFile::TABLE_ADD_COMMITTED:
+ jam();
+ case SchemaFile::ALTER_TABLE_COMMITTED:
+ jam();
+ ok = true;
+ //------------------------------------------------------------------
+ // Table was added in both our node and the master node. We can
+ // retrieve the table definition from our own disk.
+ //------------------------------------------------------------------
+ if(* newEntry == * oldEntry){
+ jam();
+
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, tableId);
+ tablePtr.p->tableVersion = oldEntry->m_tableVersion;
+ tablePtr.p->tableType = (DictTabInfo::TableType)oldEntry->m_tableType;
+
+ // On NR get index from master because index state is not on file
+ const bool file = c_systemRestart || tablePtr.p->isTable();
+ restartCreateTab(signal, tableId, oldEntry, file);
+
+ return;
+ } else {
+ //------------------------------------------------------------------
+ // Must be a new version of the table if anything differs. Both table
+ // version and global checkpoint must be different.
+ // This should not happen for the master node. This can happen after
+ // drop table followed by add table or after change table.
+ // Not supported in this version.
+ //------------------------------------------------------------------
+ ndbrequire(c_masterNodeId != getOwnNodeId());
+ ndbrequire(newEntry->m_tableVersion != oldEntry->m_tableVersion);
+ jam();
+
+ restartCreateTab(signal, tableId, oldEntry, false);
+ return;
+ }//if
+ ndbrequire(ok);
+ break;
+ }
+ }
+ case SchemaFile::DROP_TABLE_STARTED:
+ jam();
+ case SchemaFile::DROP_TABLE_COMMITTED:{
+ jam();
+ bool ok = false;
+ switch(oldSchemaState){
+ case SchemaFile::INIT:
+ jam();
+ case SchemaFile::DROP_TABLE_COMMITTED:
+ jam();
+ ok = true;
+ break;
+ case SchemaFile::ADD_STARTED:
+ jam();
+ case SchemaFile::TABLE_ADD_COMMITTED:
+ jam();
+ case SchemaFile::DROP_TABLE_STARTED:
+ jam();
+ case SchemaFile::ALTER_TABLE_COMMITTED:
+ jam();
+ newEntry->m_tableState = SchemaFile::INIT;
+ restartDropTab(signal, tableId);
+ return;
+ }
+ ndbrequire(ok);
+ break;
+ }
+ case SchemaFile::ALTER_TABLE_COMMITTED: {
+ jam();
+ bool ok = false;
+ switch(oldSchemaState) {
+ case SchemaFile::INIT:
+ jam();
+ case SchemaFile::ADD_STARTED:
+ jam();
+ case SchemaFile::DROP_TABLE_STARTED:
+ jam();
+ case SchemaFile::DROP_TABLE_COMMITTED:
+ jam();
+ case SchemaFile::TABLE_ADD_COMMITTED:
+ jam();
+ ok = true;
+ //------------------------------------------------------------------
+ // Table was altered in the master node but not in our node. We can
+ // retrieve the altered table definition from the master.
+ //------------------------------------------------------------------
+ restartCreateTab(signal, tableId, oldEntry, false);
+ return;
+ break;
+ case SchemaFile::ALTER_TABLE_COMMITTED:
+ jam();
+ ok = true;
+
+ //------------------------------------------------------------------
+ // Table was altered in both our node and the master node. We can
+ // retrieve the table definition from our own disk.
+ //------------------------------------------------------------------
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, tableId);
+ tablePtr.p->tableVersion = oldEntry->m_tableVersion;
+ tablePtr.p->tableType = (DictTabInfo::TableType)oldEntry->m_tableType;
+
+ // On NR get index from master because index state is not on file
+ const bool file = c_systemRestart || tablePtr.p->isTable();
+ restartCreateTab(signal, tableId, oldEntry, file);
+
+ return;
+ }
+ ndbrequire(ok);
+ break;
+ }
+ }
+ }
+
+ execute(signal, c_schemaRecord.m_callback, 0);
+}//checkSchemaStatus()
+
+void
+Dbdict::restartCreateTab(Signal* signal, Uint32 tableId,
+ const SchemaFile::TableEntry * te, bool file){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ c_opCreateTable.seize(createTabPtr);
+ ndbrequire(!createTabPtr.isNull());
+
+ createTabPtr.p->key = ++c_opRecordSequence;
+ c_opCreateTable.add(createTabPtr);
+
+ createTabPtr.p->m_errorCode = 0;
+ createTabPtr.p->m_tablePtrI = tableId;
+ createTabPtr.p->m_coordinatorRef = reference();
+ createTabPtr.p->m_senderRef = 0;
+ createTabPtr.p->m_senderData = RNIL;
+ createTabPtr.p->m_tabInfoPtrI = RNIL;
+ createTabPtr.p->m_dihAddFragPtr = RNIL;
+
+ if(file && !ERROR_INSERTED(6002)){
+ jam();
+
+ c_readTableRecord.noOfPages = te->m_noOfPages;
+ c_readTableRecord.pageId = 0;
+ c_readTableRecord.m_callback.m_callbackData = createTabPtr.p->key;
+ c_readTableRecord.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::restartCreateTab_readTableConf);
+
+ startReadTableFile(signal, tableId);
+ return;
+ } else {
+
+ ndbrequire(c_masterNodeId != getOwnNodeId());
+
+ /**
+ * Get from master
+ */
+ GetTabInfoReq * const req = (GetTabInfoReq *)&signal->theData[0];
+ req->senderRef = reference();
+ req->senderData = createTabPtr.p->key;
+ req->requestType = GetTabInfoReq::RequestById |
+ GetTabInfoReq::LongSignalConf;
+ req->tableId = tableId;
+ sendSignal(calcDictBlockRef(c_masterNodeId), GSN_GET_TABINFOREQ, signal,
+ GetTabInfoReq::SignalLength, JBB);
+
+ if(ERROR_INSERTED(6002)){
+ NdbSleep_MilliSleep(10);
+ CRASH_INSERTION(6002);
+ }
+ }
+}
+
+void
+Dbdict::restartCreateTab_readTableConf(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ PageRecordPtr pageRecPtr;
+ c_pageRecordArray.getPtr(pageRecPtr, c_readTableRecord.pageId);
+
+ ParseDictTabInfoRecord parseRecord;
+ parseRecord.requestType = DictTabInfo::GetTabInfoConf;
+ parseRecord.errorCode = 0;
+
+ Uint32 sz = c_readTableRecord.noOfPages * ZSIZE_OF_PAGES_IN_WORDS;
+ SimplePropertiesLinearReader r(&pageRecPtr.p->word[0], sz);
+ handleTabInfoInit(r, &parseRecord);
+ ndbrequire(parseRecord.errorCode == 0);
+
+ /* ---------------------------------------------------------------- */
+ // We have read the table description from disk as part of system restart.
+ // We will also write it back again to ensure that both copies are ok.
+ /* ---------------------------------------------------------------- */
+ ndbrequire(c_writeTableRecord.tableWriteState == WriteTableRecord::IDLE);
+ c_writeTableRecord.noOfPages = c_readTableRecord.noOfPages;
+ c_writeTableRecord.pageId = c_readTableRecord.pageId;
+ c_writeTableRecord.tableWriteState = WriteTableRecord::CALLBACK;
+ c_writeTableRecord.m_callback.m_callbackData = callbackData;
+ c_writeTableRecord.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::restartCreateTab_writeTableConf);
+ startWriteTableFile(signal, c_readTableRecord.tableId);
+}
+
+void
+Dbdict::execGET_TABINFO_CONF(Signal* signal){
+ jamEntry();
+
+ if(!assembleFragments(signal)){
+ jam();
+ return;
+ }
+
+ GetTabInfoConf * const conf = (GetTabInfoConf*)signal->getDataPtr();
+
+ const Uint32 tableId = conf->tableId;
+ const Uint32 senderData = conf->senderData;
+
+ SegmentedSectionPtr tabInfoPtr;
+ signal->getSection(tabInfoPtr, GetTabInfoConf::DICT_TAB_INFO);
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, senderData));
+ ndbrequire(!createTabPtr.isNull());
+ ndbrequire(createTabPtr.p->m_tablePtrI == tableId);
+
+ /**
+ * Put data into table record
+ */
+ ParseDictTabInfoRecord parseRecord;
+ parseRecord.requestType = DictTabInfo::GetTabInfoConf;
+ parseRecord.errorCode = 0;
+
+ SimplePropertiesSectionReader r(tabInfoPtr, getSectionSegmentPool());
+ handleTabInfoInit(r, &parseRecord);
+ ndbrequire(parseRecord.errorCode == 0);
+
+ Callback callback;
+ callback.m_callbackData = createTabPtr.p->key;
+ callback.m_callbackFunction =
+ safe_cast(&Dbdict::restartCreateTab_writeTableConf);
+
+ writeTableFile(signal, createTabPtr.p->m_tablePtrI, tabInfoPtr, &callback);
+}
+
+void
+Dbdict::restartCreateTab_writeTableConf(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ Callback callback;
+ callback.m_callbackData = callbackData;
+ callback.m_callbackFunction =
+ safe_cast(&Dbdict::restartCreateTab_dihComplete);
+
+ SegmentedSectionPtr fragDataPtr; fragDataPtr.setNull();
+ createTab_dih(signal, createTabPtr, fragDataPtr, &callback);
+}
+
+void
+Dbdict::restartCreateTab_dihComplete(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ //@todo check error
+ ndbrequire(createTabPtr.p->m_errorCode == 0);
+
+ Callback callback;
+ callback.m_callbackData = callbackData;
+ callback.m_callbackFunction =
+ safe_cast(&Dbdict::restartCreateTab_activateComplete);
+
+ alterTab_activate(signal, createTabPtr, &callback);
+}
+
+void
+Dbdict::restartCreateTab_activateComplete(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+ tabPtr.p->tabState = TableRecord::DEFINED;
+
+ c_opCreateTable.release(createTabPtr);
+
+ c_restartRecord.activeTable++;
+ checkSchemaStatus(signal);
+}
+
+void
+Dbdict::restartDropTab(Signal* signal, Uint32 tableId){
+
+ const Uint32 key = ++c_opRecordSequence;
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.seize(dropTabPtr));
+
+ dropTabPtr.p->key = key;
+ c_opDropTable.add(dropTabPtr);
+
+ dropTabPtr.p->m_errorCode = 0;
+ dropTabPtr.p->m_request.tableId = tableId;
+ dropTabPtr.p->m_coordinatorRef = 0;
+ dropTabPtr.p->m_requestType = DropTabReq::RestartDropTab;
+ dropTabPtr.p->m_participantData.m_gsn = GSN_DROP_TAB_REQ;
+
+
+ dropTabPtr.p->m_participantData.m_block = 0;
+ dropTabPtr.p->m_participantData.m_callback.m_callbackData = key;
+ dropTabPtr.p->m_participantData.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::restartDropTab_complete);
+ dropTab_nextStep(signal, dropTabPtr);
+}
+
+void
+Dbdict::restartDropTab_complete(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, callbackData));
+
+ //@todo check error
+
+ c_opDropTable.release(dropTabPtr);
+
+ c_restartRecord.activeTable++;
+ checkSchemaStatus(signal);
+}
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: NODE FAILURE HANDLING ------------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* This module contains the code that is used when nodes */
+/* (kernel/api) fails. */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+/* ---------------------------------------------------------------- */
+// We receive a report of an API that failed.
+/* ---------------------------------------------------------------- */
+void Dbdict::execAPI_FAILREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 failedApiNode = signal->theData[0];
+ BlockReference retRef = signal->theData[1];
+
+#if 0
+ Uint32 userNode = refToNode(c_connRecord.userBlockRef);
+ if (userNode == failedApiNode) {
+ jam();
+ c_connRecord.userBlockRef = (Uint32)-1;
+ }//if
+#endif
+
+ signal->theData[0] = failedApiNode;
+ signal->theData[1] = reference();
+ sendSignal(retRef, GSN_API_FAILCONF, signal, 2, JBB);
+}//execAPI_FAILREQ()
+
+/* ---------------------------------------------------------------- */
+// We receive a report of one or more node failures of kernel nodes.
+/* ---------------------------------------------------------------- */
+void Dbdict::execNODE_FAILREP(Signal* signal)
+{
+ jamEntry();
+ NodeFailRep * const nodeFail = (NodeFailRep *)&signal->theData[0];
+
+ c_failureNr = nodeFail->failNo;
+ const Uint32 numberOfFailedNodes = nodeFail->noOfNodes;
+ const bool masterFailed = (c_masterNodeId != nodeFail->masterNodeId);
+ c_masterNodeId = nodeFail->masterNodeId;
+
+ c_noNodesFailed += numberOfFailedNodes;
+ Uint32 theFailedNodes[NodeBitmask::Size];
+ memcpy(theFailedNodes, nodeFail->theNodes, sizeof(theFailedNodes));
+
+ c_counterMgr.execNODE_FAILREP(signal);
+
+ bool ok = false;
+ switch(c_blockState){
+ case BS_IDLE:
+ jam();
+ ok = true;
+ if(c_opRecordPool.getSize() != c_opRecordPool.getNoOfFree()){
+ jam();
+ c_blockState = BS_NODE_FAILURE;
+ }
+ break;
+ case BS_CREATE_TAB:
+ jam();
+ ok = true;
+ if(!masterFailed)
+ break;
+ // fall through
+ case BS_BUSY:
+ case BS_NODE_FAILURE:
+ jam();
+ c_blockState = BS_NODE_FAILURE;
+ ok = true;
+ break;
+ }
+ ndbrequire(ok);
+
+ for(unsigned i = 1; i < MAX_NDB_NODES; i++) {
+ jam();
+ if(NodeBitmask::get(theFailedNodes, i)) {
+ jam();
+ NodeRecordPtr nodePtr;
+ c_nodes.getPtr(nodePtr, i);
+
+ nodePtr.p->nodeState = NodeRecord::NDB_NODE_DEAD;
+ NFCompleteRep * const nfCompRep = (NFCompleteRep *)&signal->theData[0];
+ nfCompRep->blockNo = DBDICT;
+ nfCompRep->nodeId = getOwnNodeId();
+ nfCompRep->failedNodeId = nodePtr.i;
+ sendSignal(DBDIH_REF, GSN_NF_COMPLETEREP, signal,
+ NFCompleteRep::SignalLength, JBB);
+
+ c_aliveNodes.clear(i);
+ }//if
+ }//for
+
+}//execNODE_FAILREP()
+
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: NODE START HANDLING --------------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* This module contains the code that is used when kernel nodes */
+/* starts. */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+/* ---------------------------------------------------------------- */
+// Include a starting node in list of nodes to be part of adding
+// and dropping tables.
+/* ---------------------------------------------------------------- */
+void Dbdict::execINCL_NODEREQ(Signal* signal)
+{
+ jamEntry();
+ NodeRecordPtr nodePtr;
+ BlockReference retRef = signal->theData[0];
+ nodePtr.i = signal->theData[1];
+
+ ndbrequire(c_noNodesFailed > 0);
+ c_noNodesFailed--;
+
+ c_nodes.getPtr(nodePtr);
+ ndbrequire(nodePtr.p->nodeState = NodeRecord::NDB_NODE_DEAD);
+ nodePtr.p->nodeState = NodeRecord::NDB_NODE_ALIVE;
+ signal->theData[0] = reference();
+ sendSignal(retRef, GSN_INCL_NODECONF, signal, 1, JBB);
+
+ c_aliveNodes.set(nodePtr.i);
+}//execINCL_NODEREQ()
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: ADD TABLE HANDLING ---------------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* This module contains the code that is used when adding a table. */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+/* ---------------------------------------------------------------- */
+// This signal receives information about a table from either:
+// API, Ndbcntr or from other DICT.
+/* ---------------------------------------------------------------- */
+void
+Dbdict::execCREATE_TABLE_REQ(Signal* signal){
+ jamEntry();
+ if(!assembleFragments(signal)){
+ return;
+ }
+
+ CreateTableReq* const req = (CreateTableReq*)signal->getDataPtr();
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 senderData = req->senderData;
+
+ ParseDictTabInfoRecord parseRecord;
+ do {
+ if(getOwnNodeId() != c_masterNodeId){
+ jam();
+ parseRecord.errorCode = CreateTableRef::NotMaster;
+ break;
+ }
+
+ if (c_blockState != BS_IDLE){
+ jam();
+ parseRecord.errorCode = CreateTableRef::Busy;
+ break;
+ }
+
+ CreateTableRecordPtr createTabPtr;
+ c_opCreateTable.seize(createTabPtr);
+
+ if(createTabPtr.isNull()){
+ jam();
+ parseRecord.errorCode = CreateTableRef::Busy;
+ break;
+ }
+
+ parseRecord.requestType = DictTabInfo::CreateTableFromAPI;
+ parseRecord.errorCode = 0;
+
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr, CreateTableReq::DICT_TAB_INFO);
+ SimplePropertiesSectionReader r(ptr, getSectionSegmentPool());
+
+ handleTabInfoInit(r, &parseRecord);
+ releaseSections(signal);
+
+ if(parseRecord.errorCode != 0){
+ jam();
+ c_opCreateTable.release(createTabPtr);
+ break;
+ }
+
+ createTabPtr.p->key = ++c_opRecordSequence;
+ c_opCreateTable.add(createTabPtr);
+ createTabPtr.p->m_errorCode = 0;
+ createTabPtr.p->m_senderRef = senderRef;
+ createTabPtr.p->m_senderData = senderData;
+ createTabPtr.p->m_tablePtrI = parseRecord.tablePtr.i;
+ createTabPtr.p->m_coordinatorRef = reference();
+ createTabPtr.p->m_fragmentsPtrI = RNIL;
+ createTabPtr.p->m_dihAddFragPtr = RNIL;
+
+ Uint32 * theData = signal->getDataPtrSend();
+ CreateFragmentationReq * const req = (CreateFragmentationReq*)theData;
+ req->senderRef = reference();
+ req->senderData = createTabPtr.p->key;
+ req->fragmentationType = parseRecord.tablePtr.p->fragmentType;
+ req->noOfFragments = 0;
+ req->fragmentNode = 0;
+ req->primaryTableId = RNIL;
+ if (parseRecord.tablePtr.p->isOrderedIndex()) {
+ // ordered index has same fragmentation as the table
+ const Uint32 primaryTableId = parseRecord.tablePtr.p->primaryTableId;
+ TableRecordPtr primaryTablePtr;
+ c_tableRecordPool.getPtr(primaryTablePtr, primaryTableId);
+ // fragmentationType must be consistent
+ req->fragmentationType = primaryTablePtr.p->fragmentType;
+ req->primaryTableId = primaryTableId;
+ }
+ sendSignal(DBDIH_REF, GSN_CREATE_FRAGMENTATION_REQ, signal,
+ CreateFragmentationReq::SignalLength, JBB);
+
+ c_blockState = BS_CREATE_TAB;
+ return;
+ } while(0);
+
+ /**
+ * Something went wrong
+ */
+ releaseSections(signal);
+
+ CreateTableRef * ref = (CreateTableRef*)signal->getDataPtrSend();
+ ref->senderData = senderData;
+ ref->senderRef = reference();
+ ref->masterNodeId = c_masterNodeId;
+ ref->errorCode = parseRecord.errorCode;
+ ref->errorLine = parseRecord.errorLine;
+ ref->errorKey = parseRecord.errorKey;
+ ref->status = parseRecord.status;
+ sendSignal(senderRef, GSN_CREATE_TABLE_REF, signal,
+ CreateTableRef::SignalLength, JBB);
+}
+
+void
+Dbdict::execALTER_TABLE_REQ(Signal* signal)
+{
+ // Received by master
+ jamEntry();
+ if(!assembleFragments(signal)){
+ return;
+ }
+ AlterTableReq* const req = (AlterTableReq*)signal->getDataPtr();
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 senderData = req->senderData;
+ const Uint32 changeMask = req->changeMask;
+ const Uint32 tableId = req->tableId;
+ const Uint32 tableVersion = req->tableVersion;
+ ParseDictTabInfoRecord* aParseRecord;
+
+ // Get table definition
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, tableId, false);
+ if(tablePtr.isNull()){
+ jam();
+ alterTableRef(signal, req, AlterTableRef::NoSuchTable);
+ return;
+ }
+
+ if(getOwnNodeId() != c_masterNodeId){
+ jam();
+ alterTableRef(signal, req, AlterTableRef::NotMaster);
+ return;
+ }
+
+ if(c_blockState != BS_IDLE){
+ jam();
+ alterTableRef(signal, req, AlterTableRef::Busy);
+ return;
+ }
+
+ const TableRecord::TabState tabState = tablePtr.p->tabState;
+ bool ok = false;
+ switch(tabState){
+ case TableRecord::NOT_DEFINED:
+ case TableRecord::REORG_TABLE_PREPARED:
+ case TableRecord::DEFINING:
+ case TableRecord::CHECKED:
+ jam();
+ alterTableRef(signal, req, AlterTableRef::NoSuchTable);
+ return;
+ case TableRecord::DEFINED:
+ ok = true;
+ jam();
+ break;
+ case TableRecord::PREPARE_DROPPING:
+ case TableRecord::DROPPING:
+ jam();
+ alterTableRef(signal, req, AlterTableRef::DropInProgress);
+ return;
+ }
+ ndbrequire(ok);
+
+ if(tablePtr.p->tableVersion != tableVersion){
+ jam();
+ alterTableRef(signal, req, AlterTableRef::InvalidTableVersion);
+ return;
+ }
+ // Parse new table defintion
+ ParseDictTabInfoRecord parseRecord;
+ aParseRecord = &parseRecord;
+
+ CreateTableRecordPtr alterTabPtr; // Reuse create table records
+ c_opCreateTable.seize(alterTabPtr);
+ CreateTableRecord * regAlterTabPtr = alterTabPtr.p;
+
+ if(alterTabPtr.isNull()){
+ jam();
+ alterTableRef(signal, req, AlterTableRef::Busy);
+ return;
+ }
+
+ regAlterTabPtr->m_changeMask = changeMask;
+ parseRecord.requestType = DictTabInfo::AlterTableFromAPI;
+ parseRecord.errorCode = 0;
+
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr, AlterTableReq::DICT_TAB_INFO);
+ SimplePropertiesSectionReader r(ptr, getSectionSegmentPool());
+
+ handleTabInfoInit(r, &parseRecord, false); // Will not save info
+
+ if(parseRecord.errorCode != 0){
+ jam();
+ c_opCreateTable.release(alterTabPtr);
+ parseRecord.tablePtr.p->tabState = TableRecord::NOT_DEFINED;
+ releaseTableObject(parseRecord.tablePtr.i, false);
+ alterTableRef(signal, req,
+ (AlterTableRef::ErrorCode) parseRecord.errorCode,
+ aParseRecord);
+ return;
+ }
+
+ releaseSections(signal);
+ regAlterTabPtr->key = ++c_opRecordSequence;
+ c_opCreateTable.add(alterTabPtr);
+ ndbrequire(c_opCreateTable.find(alterTabPtr, regAlterTabPtr->key));
+ regAlterTabPtr->m_errorCode = 0;
+ regAlterTabPtr->m_senderRef = senderRef;
+ regAlterTabPtr->m_senderData = senderData;
+ regAlterTabPtr->m_tablePtrI = parseRecord.tablePtr.i;
+ regAlterTabPtr->m_alterTableFailed = false;
+ regAlterTabPtr->m_coordinatorRef = reference();
+ regAlterTabPtr->m_fragmentsPtrI = RNIL;
+ regAlterTabPtr->m_dihAddFragPtr = RNIL;
+
+ // Alter table on all nodes
+ c_blockState = BS_BUSY;
+
+ // Send prepare request to all alive nodes
+ SimplePropertiesSectionWriter w(getSectionSegmentPool());
+ packTableIntoPagesImpl(w, parseRecord.tablePtr);
+
+ SegmentedSectionPtr tabInfoPtr;
+ w.getPtr(tabInfoPtr);
+ signal->setSection(tabInfoPtr, AlterTabReq::DICT_TAB_INFO);
+
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ regAlterTabPtr->m_coordinatorData.m_gsn = GSN_ALTER_TAB_REQ;
+ SafeCounter safeCounter(c_counterMgr, regAlterTabPtr->m_coordinatorData.m_counter);
+ safeCounter.init<AlterTabRef>(rg, regAlterTabPtr->key);
+
+ AlterTabReq * const lreq = (AlterTabReq*)signal->getDataPtrSend();
+ lreq->senderRef = reference();
+ lreq->senderData = regAlterTabPtr->key;
+ lreq->clientRef = regAlterTabPtr->m_senderRef;
+ lreq->clientData = regAlterTabPtr->m_senderData;
+ lreq->changeMask = changeMask;
+ lreq->tableId = tableId;
+ lreq->tableVersion = tableVersion + 1;
+ lreq->gci = tablePtr.p->gciTableCreated;
+ lreq->requestType = AlterTabReq::AlterTablePrepare;
+
+ sendSignal(rg, GSN_ALTER_TAB_REQ, signal,
+ AlterTabReq::SignalLength, JBB);
+
+}
+
+void Dbdict::alterTableRef(Signal * signal,
+ AlterTableReq * req,
+ AlterTableRef::ErrorCode errCode,
+ ParseDictTabInfoRecord* parseRecord)
+{
+ jam();
+ releaseSections(signal);
+ AlterTableRef * ref = (AlterTableRef*)signal->getDataPtrSend();
+ Uint32 senderRef = req->senderRef;
+ ref->senderData = req->senderData;
+ ref->senderRef = reference();
+ ref->masterNodeId = c_masterNodeId;
+ if (parseRecord) {
+ ref->errorCode = parseRecord->errorCode;
+ ref->errorLine = parseRecord->errorLine;
+ ref->errorKey = parseRecord->errorKey;
+ ref->status = parseRecord->status;
+ }
+ else {
+ ref->errorCode = errCode;
+ ref->errorLine = 0;
+ ref->errorKey = 0;
+ ref->status = 0;
+ }
+ sendSignal(senderRef, GSN_ALTER_TABLE_REF, signal,
+ AlterTableRef::SignalLength, JBB);
+}
+
+void
+Dbdict::execALTER_TAB_REQ(Signal * signal)
+{
+ // Received in all nodes to handle change locally
+ jamEntry();
+
+ if(!assembleFragments(signal)){
+ return;
+ }
+ AlterTabReq* const req = (AlterTabReq*)signal->getDataPtr();
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 senderData = req->senderData;
+ const Uint32 changeMask = req->changeMask;
+ const Uint32 tableId = req->tableId;
+ const Uint32 tableVersion = req->tableVersion;
+ const Uint32 gci = req->gci;
+ AlterTabReq::RequestType requestType =
+ (AlterTabReq::RequestType) req->requestType;
+
+ SegmentedSectionPtr tabInfoPtr;
+ signal->getSection(tabInfoPtr, AlterTabReq::DICT_TAB_INFO);
+
+ CreateTableRecordPtr alterTabPtr; // Reuse create table records
+
+ if (senderRef != reference()) {
+ jam();
+ c_blockState = BS_BUSY;
+ }
+ if ((requestType == AlterTabReq::AlterTablePrepare)
+ && (senderRef != reference())) {
+ jam();
+ c_opCreateTable.seize(alterTabPtr);
+ if(!alterTabPtr.isNull())
+ alterTabPtr.p->m_changeMask = changeMask;
+ }
+ else {
+ jam();
+ ndbrequire(c_opCreateTable.find(alterTabPtr, senderData));
+ }
+ if(alterTabPtr.isNull()){
+ jam();
+ alterTabRef(signal, req, AlterTableRef::Busy);
+ return;
+ }
+ CreateTableRecord * regAlterTabPtr = alterTabPtr.p;
+ regAlterTabPtr->m_alterTableId = tableId;
+ regAlterTabPtr->m_coordinatorRef = senderRef;
+
+ // Get table definition
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, tableId, false);
+ if(tablePtr.isNull()){
+ jam();
+ alterTabRef(signal, req, AlterTableRef::NoSuchTable);
+ return;
+ }
+
+ switch(requestType) {
+ case(AlterTabReq::AlterTablePrepare): {
+ ParseDictTabInfoRecord* aParseRecord;
+
+ const TableRecord::TabState tabState = tablePtr.p->tabState;
+ bool ok = false;
+ switch(tabState){
+ case TableRecord::NOT_DEFINED:
+ case TableRecord::REORG_TABLE_PREPARED:
+ case TableRecord::DEFINING:
+ case TableRecord::CHECKED:
+ jam();
+ alterTabRef(signal, req, AlterTableRef::NoSuchTable);
+ return;
+ case TableRecord::DEFINED:
+ ok = true;
+ jam();
+ break;
+ case TableRecord::PREPARE_DROPPING:
+ case TableRecord::DROPPING:
+ jam();
+ alterTabRef(signal, req, AlterTableRef::DropInProgress);
+ return;
+ }
+ ndbrequire(ok);
+
+ if(tablePtr.p->tableVersion + 1 != tableVersion){
+ jam();
+ alterTabRef(signal, req, AlterTableRef::InvalidTableVersion);
+ return;
+ }
+ TableRecordPtr newTablePtr;
+ if (senderRef != reference()) {
+ jam();
+ // Parse altered table defintion
+ ParseDictTabInfoRecord parseRecord;
+ aParseRecord = &parseRecord;
+
+ parseRecord.requestType = DictTabInfo::AlterTableFromAPI;
+ parseRecord.errorCode = 0;
+
+ SimplePropertiesSectionReader r(tabInfoPtr, getSectionSegmentPool());
+
+ handleTabInfoInit(r, &parseRecord, false); // Will not save info
+
+ if(parseRecord.errorCode != 0){
+ jam();
+ c_opCreateTable.release(alterTabPtr);
+ parseRecord.tablePtr.p->tabState = TableRecord::NOT_DEFINED;
+ releaseTableObject(parseRecord.tablePtr.i, false);
+ alterTabRef(signal, req,
+ (AlterTableRef::ErrorCode) parseRecord.errorCode,
+ aParseRecord);
+ return;
+ }
+ regAlterTabPtr->key = senderData;
+ c_opCreateTable.add(alterTabPtr);
+ regAlterTabPtr->m_errorCode = 0;
+ regAlterTabPtr->m_senderRef = senderRef;
+ regAlterTabPtr->m_senderData = senderData;
+ regAlterTabPtr->m_tablePtrI = parseRecord.tablePtr.i;
+ regAlterTabPtr->m_fragmentsPtrI = RNIL;
+ regAlterTabPtr->m_dihAddFragPtr = RNIL;
+ newTablePtr = parseRecord.tablePtr;
+ newTablePtr.p->tableVersion = tableVersion;
+ }
+ else { // (req->senderRef == reference())
+ jam();
+ c_tableRecordPool.getPtr(newTablePtr, regAlterTabPtr->m_tablePtrI);
+ newTablePtr.p->tableVersion = tableVersion;
+ }
+ if (handleAlterTab(req, regAlterTabPtr, tablePtr, newTablePtr) == -1) {
+ jam();
+ c_opCreateTable.release(alterTabPtr);
+ alterTabRef(signal, req, AlterTableRef::UnsupportedChange);
+ return;
+ }
+ releaseSections(signal);
+ // Propagate alter table to other local blocks
+ AlterTabReq * req = (AlterTabReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = senderData;
+ req->changeMask = changeMask;
+ req->tableId = tableId;
+ req->tableVersion = tableVersion;
+ req->gci = gci;
+ req->requestType = requestType;
+ sendSignal(DBLQH_REF, GSN_ALTER_TAB_REQ, signal,
+ AlterTabReq::SignalLength, JBB);
+ return;
+ }
+ case(AlterTabReq::AlterTableCommit): {
+ jam();
+ // Write schema for altered table to disk
+ SegmentedSectionPtr tabInfoPtr;
+ signal->getSection(tabInfoPtr, AlterTabReq::DICT_TAB_INFO);
+ regAlterTabPtr->m_tabInfoPtrI = tabInfoPtr.i;
+
+ signal->header.m_noOfSections = 0;
+
+ // Update table record
+ tablePtr.p->packedSize = tabInfoPtr.sz;
+ tablePtr.p->tableVersion = tableVersion;
+ tablePtr.p->gciTableCreated = gci;
+
+ SchemaFile::TableEntry tabEntry;
+ tabEntry.m_tableVersion = tableVersion;
+ tabEntry.m_tableType = tablePtr.p->tableType;
+ tabEntry.m_tableState = SchemaFile::ALTER_TABLE_COMMITTED;
+ tabEntry.m_gcp = gci;
+ tabEntry.m_noOfPages =
+ DIV(tabInfoPtr.sz + ZPAGE_HEADER_SIZE, ZSIZE_OF_PAGES_IN_WORDS);
+
+ Callback callback;
+ callback.m_callbackData = senderData;
+ callback.m_callbackFunction =
+ safe_cast(&Dbdict::alterTab_writeSchemaConf);
+
+ updateSchemaState(signal, tableId, &tabEntry, &callback);
+ break;
+ }
+ case(AlterTabReq::AlterTableRevert): {
+ jam();
+ // Revert failed alter table
+ revertAlterTable(signal, changeMask, tableId, regAlterTabPtr);
+ // Acknowledge the reverted alter table
+ AlterTabConf * conf = (AlterTabConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ conf->changeMask = changeMask;
+ conf->tableId = tableId;
+ conf->tableVersion = tableVersion;
+ conf->gci = gci;
+ conf->requestType = requestType;
+ sendSignal(senderRef, GSN_ALTER_TAB_CONF, signal,
+ AlterTabConf::SignalLength, JBB);
+ break;
+ }
+ default: ndbrequire(false);
+ }
+}
+
+void Dbdict::alterTabRef(Signal * signal,
+ AlterTabReq * req,
+ AlterTableRef::ErrorCode errCode,
+ ParseDictTabInfoRecord* parseRecord)
+{
+ jam();
+ releaseSections(signal);
+ AlterTabRef * ref = (AlterTabRef*)signal->getDataPtrSend();
+ Uint32 senderRef = req->senderRef;
+ ref->senderData = req->senderData;
+ ref->senderRef = reference();
+ if (parseRecord) {
+ jam();
+ ref->errorCode = parseRecord->errorCode;
+ ref->errorLine = parseRecord->errorLine;
+ ref->errorKey = parseRecord->errorKey;
+ ref->errorStatus = parseRecord->status;
+ }
+ else {
+ jam();
+ ref->errorCode = errCode;
+ ref->errorLine = 0;
+ ref->errorKey = 0;
+ ref->errorStatus = 0;
+ }
+ sendSignal(senderRef, GSN_ALTER_TAB_REF, signal,
+ AlterTabRef::SignalLength, JBB);
+
+ c_blockState = BS_IDLE;
+}
+
+void Dbdict::execALTER_TAB_REF(Signal * signal){
+ jamEntry();
+
+ AlterTabRef * ref = (AlterTabRef*)signal->getDataPtr();
+
+ Uint32 senderRef = ref->senderRef;
+ Uint32 senderData = ref->senderData;
+ Uint32 errorCode = ref->errorCode;
+ Uint32 errorLine = ref->errorLine;
+ Uint32 errorKey = ref->errorKey;
+ Uint32 errorStatus = ref->errorStatus;
+ AlterTabReq::RequestType requestType =
+ (AlterTabReq::RequestType) ref->requestType;
+ CreateTableRecordPtr alterTabPtr;
+ ndbrequire(c_opCreateTable.find(alterTabPtr, senderData));
+ CreateTableRecord * regAlterTabPtr = alterTabPtr.p;
+ Uint32 changeMask = regAlterTabPtr->m_changeMask;
+ SafeCounter safeCounter(c_counterMgr, regAlterTabPtr->m_coordinatorData.m_counter);
+ safeCounter.clearWaitingFor(refToNode(senderRef));
+ switch (requestType) {
+ case(AlterTabReq::AlterTablePrepare): {
+ if (safeCounter.done()) {
+ jam();
+ // Send revert request to all alive nodes
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, regAlterTabPtr->m_alterTableId);
+ Uint32 tableId = tablePtr.p->tableId;
+ Uint32 tableVersion = tablePtr.p->tableVersion;
+ Uint32 gci = tablePtr.p->gciTableCreated;
+ SimplePropertiesSectionWriter w(getSectionSegmentPool());
+ packTableIntoPagesImpl(w, tablePtr);
+ SegmentedSectionPtr spDataPtr;
+ w.getPtr(spDataPtr);
+ signal->setSection(spDataPtr, AlterTabReq::DICT_TAB_INFO);
+
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ regAlterTabPtr->m_coordinatorData.m_gsn = GSN_ALTER_TAB_REQ;
+ safeCounter.init<AlterTabRef>(rg, regAlterTabPtr->key);
+
+ AlterTabReq * const lreq = (AlterTabReq*)signal->getDataPtrSend();
+ lreq->senderRef = reference();
+ lreq->senderData = regAlterTabPtr->key;
+ lreq->clientRef = regAlterTabPtr->m_senderRef;
+ lreq->clientData = regAlterTabPtr->m_senderData;
+ lreq->changeMask = changeMask;
+ lreq->tableId = tableId;
+ lreq->tableVersion = tableVersion;
+ lreq->gci = gci;
+ lreq->requestType = AlterTabReq::AlterTableRevert;
+
+ sendSignal(rg, GSN_ALTER_TAB_REQ, signal,
+ AlterTabReq::SignalLength, JBB);
+ }
+ else {
+ jam();
+ regAlterTabPtr->m_alterTableFailed = true;
+ }
+ break;
+ }
+ case(AlterTabReq::AlterTableCommit):
+ jam();
+ case(AlterTabReq::AlterTableRevert): {
+ AlterTableRef * apiRef = (AlterTableRef*)signal->getDataPtrSend();
+
+ apiRef->senderData = senderData;
+ apiRef->senderRef = reference();
+ apiRef->masterNodeId = c_masterNodeId;
+ apiRef->errorCode = errorCode;
+ apiRef->errorLine = errorLine;
+ apiRef->errorKey = errorKey;
+ apiRef->status = errorStatus;
+ if (safeCounter.done()) {
+ jam();
+ sendSignal(senderRef, GSN_ALTER_TABLE_REF, signal,
+ AlterTableRef::SignalLength, JBB);
+ c_blockState = BS_IDLE;
+ }
+ else {
+ jam();
+ regAlterTabPtr->m_alterTableFailed = true;
+ regAlterTabPtr->m_alterTableRef = *apiRef;
+ }
+ break;
+ }
+ default: ndbrequire(false);
+ }
+}
+
+void
+Dbdict::execALTER_TAB_CONF(Signal * signal){
+ jamEntry();
+ AlterTabConf * const conf = (AlterTabConf*)signal->getDataPtr();
+ Uint32 senderRef = conf->senderRef;
+ Uint32 senderData = conf->senderData;
+ Uint32 changeMask = conf->changeMask;
+ Uint32 tableId = conf->tableId;
+ Uint32 tableVersion = conf->tableVersion;
+ Uint32 gci = conf->gci;
+ AlterTabReq::RequestType requestType =
+ (AlterTabReq::RequestType) conf->requestType;
+ CreateTableRecordPtr alterTabPtr;
+ ndbrequire(c_opCreateTable.find(alterTabPtr, senderData));
+ CreateTableRecord * regAlterTabPtr = alterTabPtr.p;
+
+ switch (requestType) {
+ case(AlterTabReq::AlterTablePrepare): {
+ switch(refToBlock(signal->getSendersBlockRef())) {
+ case DBLQH: {
+ jam();
+ AlterTabReq * req = (AlterTabReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = senderData;
+ req->changeMask = changeMask;
+ req->tableId = tableId;
+ req->tableVersion = tableVersion;
+ req->gci = gci;
+ req->requestType = requestType;
+ sendSignal(DBDIH_REF, GSN_ALTER_TAB_REQ, signal,
+ AlterTabReq::SignalLength, JBB);
+ return;
+ }
+ case DBDIH: {
+ jam();
+ AlterTabReq * req = (AlterTabReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = senderData;
+ req->changeMask = changeMask;
+ req->tableId = tableId;
+ req->tableVersion = tableVersion;
+ req->gci = gci;
+ req->requestType = requestType;
+ sendSignal(DBTC_REF, GSN_ALTER_TAB_REQ, signal,
+ AlterTabReq::SignalLength, JBB);
+ return;
+ }
+ case DBTC: {
+ jam();
+ // Participant is done with prepare phase, send conf to coordinator
+ AlterTabConf * conf = (AlterTabConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ conf->changeMask = changeMask;
+ conf->tableId = tableId;
+ conf->tableVersion = tableVersion;
+ conf->gci = gci;
+ conf->requestType = requestType;
+ sendSignal(regAlterTabPtr->m_coordinatorRef, GSN_ALTER_TAB_CONF, signal,
+ AlterTabConf::SignalLength, JBB);
+ return;
+ }
+ default :break;
+ }
+ // Coordinator only
+ SafeCounter safeCounter(c_counterMgr, regAlterTabPtr->m_coordinatorData.m_counter);
+ safeCounter.clearWaitingFor(refToNode(senderRef));
+ if (safeCounter.done()) {
+ jam();
+ // We have received all local confirmations
+ if (regAlterTabPtr->m_alterTableFailed) {
+ jam();
+ // Send revert request to all alive nodes
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, regAlterTabPtr->m_alterTableId);
+ Uint32 tableId = tablePtr.p->tableId;
+ Uint32 tableVersion = tablePtr.p->tableVersion;
+ Uint32 gci = tablePtr.p->gciTableCreated;
+ SimplePropertiesSectionWriter w(getSectionSegmentPool());
+ packTableIntoPagesImpl(w, tablePtr);
+ SegmentedSectionPtr spDataPtr;
+ w.getPtr(spDataPtr);
+ signal->setSection(spDataPtr, AlterTabReq::DICT_TAB_INFO);
+
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ regAlterTabPtr->m_coordinatorData.m_gsn = GSN_ALTER_TAB_REQ;
+ safeCounter.init<AlterTabRef>(rg, regAlterTabPtr->key);
+
+ AlterTabReq * const lreq = (AlterTabReq*)signal->getDataPtrSend();
+ lreq->senderRef = reference();
+ lreq->senderData = regAlterTabPtr->key;
+ lreq->clientRef = regAlterTabPtr->m_senderRef;
+ lreq->clientData = regAlterTabPtr->m_senderData;
+ lreq->changeMask = changeMask;
+ lreq->tableId = tableId;
+ lreq->tableVersion = tableVersion;
+ lreq->gci = gci;
+ lreq->requestType = AlterTabReq::AlterTableRevert;
+
+ sendSignal(rg, GSN_ALTER_TAB_REQ, signal,
+ AlterTabReq::SignalLength, JBB);
+ }
+ else {
+ jam();
+ // Send commit request to all alive nodes
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, tableId);
+ SimplePropertiesSectionWriter w(getSectionSegmentPool());
+ packTableIntoPagesImpl(w, tablePtr);
+ SegmentedSectionPtr spDataPtr;
+ w.getPtr(spDataPtr);
+ signal->setSection(spDataPtr, AlterTabReq::DICT_TAB_INFO);
+
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ regAlterTabPtr->m_coordinatorData.m_gsn = GSN_ALTER_TAB_REQ;
+ safeCounter.init<AlterTabRef>(rg, regAlterTabPtr->key);
+
+ AlterTabReq * const lreq = (AlterTabReq*)signal->getDataPtrSend();
+ lreq->senderRef = reference();
+ lreq->senderData = regAlterTabPtr->key;
+ lreq->clientRef = regAlterTabPtr->m_senderRef;
+ lreq->clientData = regAlterTabPtr->m_senderData;
+ lreq->changeMask = changeMask;
+ lreq->tableId = tableId;
+ lreq->tableVersion = tableVersion;
+ lreq->gci = gci;
+ lreq->requestType = AlterTabReq::AlterTableCommit;
+
+ sendSignal(rg, GSN_ALTER_TAB_REQ, signal,
+ AlterTabReq::SignalLength, JBB);
+ }
+ }
+ else {
+ // (!safeCounter.done())
+ jam();
+ }
+ break;
+ }
+ case(AlterTabReq::AlterTableRevert):
+ jam();
+ case(AlterTabReq::AlterTableCommit): {
+ SafeCounter safeCounter(c_counterMgr, regAlterTabPtr->m_coordinatorData.m_counter);
+ safeCounter.clearWaitingFor(refToNode(senderRef));
+ if (safeCounter.done()) {
+ jam();
+ // We have received all local confirmations
+ releaseSections(signal);
+ if (regAlterTabPtr->m_alterTableFailed) {
+ jam();
+ AlterTableRef * apiRef =
+ (AlterTableRef*)signal->getDataPtrSend();
+ *apiRef = regAlterTabPtr->m_alterTableRef;
+ sendSignal(regAlterTabPtr->m_senderRef, GSN_ALTER_TABLE_REF, signal,
+ AlterTableRef::SignalLength, JBB);
+ }
+ else {
+ jam();
+ // Alter table completed, inform API
+ AlterTableConf * const apiConf =
+ (AlterTableConf*)signal->getDataPtrSend();
+ apiConf->senderRef = reference();
+ apiConf->senderData = regAlterTabPtr->m_senderData;
+ apiConf->tableId = tableId;
+ apiConf->tableVersion = tableVersion;
+
+ //@todo check api failed
+ sendSignal(regAlterTabPtr->m_senderRef, GSN_ALTER_TABLE_CONF, signal,
+ AlterTableConf::SignalLength, JBB);
+ }
+
+ // Release resources
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, regAlterTabPtr->m_tablePtrI);
+ tabPtr.p->tabState = TableRecord::NOT_DEFINED;
+ releaseTableObject(tabPtr.i, false);
+ c_opCreateTable.release(alterTabPtr);
+ c_blockState = BS_IDLE;
+ }
+ else {
+ // (!safeCounter.done())
+ jam();
+ }
+ break;
+ }
+ default: ndbrequire(false);
+ }
+}
+
+// For debugging
+inline
+void Dbdict::printTables()
+{
+ DLHashTable<TableRecord>::Iterator iter;
+ bool moreTables = c_tableRecordHash.first(iter);
+ printf("TABLES IN DICT:\n");
+ while (moreTables) {
+ TableRecordPtr tablePtr = iter.curr;
+ printf("%s ", tablePtr.p->tableName);
+ moreTables = c_tableRecordHash.next(iter);
+ }
+ printf("\n");
+}
+
+int Dbdict::handleAlterTab(AlterTabReq * req,
+ CreateTableRecord * regAlterTabPtr,
+ TableRecordPtr origTablePtr,
+ TableRecordPtr newTablePtr)
+{
+ Uint32 changeMask = req->changeMask;
+
+ if (AlterTableReq::getNameFlag(changeMask)) {
+ jam();
+ // Table rename
+ // Remove from hashtable
+ c_tableRecordHash.remove(origTablePtr);
+ strcpy(regAlterTabPtr->previousTableName, origTablePtr.p->tableName);
+ strcpy(origTablePtr.p->tableName, newTablePtr.p->tableName);
+ // Set new schema version
+ origTablePtr.p->tableVersion = newTablePtr.p->tableVersion;
+ // Put it back
+ c_tableRecordHash.add(origTablePtr);
+
+ return 0;
+ }
+ jam();
+ return -1;
+}
+
+void Dbdict::revertAlterTable(Signal * signal,
+ Uint32 changeMask,
+ Uint32 tableId,
+ CreateTableRecord * regAlterTabPtr)
+{
+ if (AlterTableReq::getNameFlag(changeMask)) {
+ jam();
+ // Table rename
+ // Restore previous name
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, tableId);
+ // Remove from hashtable
+ c_tableRecordHash.remove(tablePtr);
+ // Restore name
+ strcpy(tablePtr.p->tableName, regAlterTabPtr->previousTableName);
+ // Revert schema version
+ tablePtr.p->tableVersion = tablePtr.p->tableVersion - 1;
+ // Put it back
+ c_tableRecordHash.add(tablePtr);
+
+ return;
+ }
+
+ ndbrequire(false);
+}
+
+void
+Dbdict::alterTab_writeSchemaConf(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ Uint32 key = callbackData;
+ CreateTableRecordPtr alterTabPtr;
+ ndbrequire(c_opCreateTable.find(alterTabPtr, key));
+ CreateTableRecord * regAlterTabPtr = alterTabPtr.p;
+ Uint32 tableId = regAlterTabPtr->m_alterTableId;
+
+ Callback callback;
+ callback.m_callbackData = regAlterTabPtr->key;
+ callback.m_callbackFunction =
+ safe_cast(&Dbdict::alterTab_writeTableConf);
+
+ SegmentedSectionPtr tabInfoPtr;
+ getSection(tabInfoPtr, regAlterTabPtr->m_tabInfoPtrI);
+
+ writeTableFile(signal, tableId, tabInfoPtr, &callback);
+
+ signal->setSection(tabInfoPtr, 0);
+ releaseSections(signal);
+}
+
+void
+Dbdict::alterTab_writeTableConf(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ CreateTableRecordPtr alterTabPtr;
+ ndbrequire(c_opCreateTable.find(alterTabPtr, callbackData));
+ CreateTableRecord * regAlterTabPtr = alterTabPtr.p;
+ Uint32 coordinatorRef = regAlterTabPtr->m_coordinatorRef;
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, regAlterTabPtr->m_alterTableId);
+
+ // Alter table commit request handled successfully
+ AlterTabConf * conf = (AlterTabConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = callbackData;
+ conf->tableId = tabPtr.p->tableId;
+ conf->tableVersion = tabPtr.p->tableVersion;
+ conf->gci = tabPtr.p->gciTableCreated;
+ conf->requestType = AlterTabReq::AlterTableCommit;
+ sendSignal(coordinatorRef, GSN_ALTER_TAB_CONF, signal,
+ AlterTabConf::SignalLength, JBB);
+ if(coordinatorRef != reference()) {
+ jam();
+ // Release resources
+ c_tableRecordPool.getPtr(tabPtr, regAlterTabPtr->m_tablePtrI);
+ tabPtr.p->tabState = TableRecord::NOT_DEFINED;
+ releaseTableObject(tabPtr.i, false);
+ c_opCreateTable.release(alterTabPtr);
+ c_blockState = BS_IDLE;
+ }
+}
+
+void
+Dbdict::execCREATE_FRAGMENTATION_REF(Signal * signal){
+ jamEntry();
+ const Uint32 * theData = signal->getDataPtr();
+ CreateFragmentationRef * const ref = (CreateFragmentationRef*)theData;
+ (void)ref;
+ ndbrequire(false);
+}
+
+void
+Dbdict::execCREATE_FRAGMENTATION_CONF(Signal* signal){
+ jamEntry();
+ const Uint32 * theData = signal->getDataPtr();
+ CreateFragmentationConf * const conf = (CreateFragmentationConf*)theData;
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, conf->senderData));
+
+ ndbrequire(signal->getNoOfSections() == 1);
+
+ SegmentedSectionPtr fragDataPtr;
+ signal->getSection(fragDataPtr, CreateFragmentationConf::FRAGMENTS);
+
+ signal->header.m_noOfSections = 0;
+
+ /**
+ * Correct table
+ */
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.schemaPage);
+ SchemaFile::TableEntry * tabEntry = getTableEntry(pagePtr.p, tabPtr.i);
+
+ /**
+ * Update table version
+ */
+ tabPtr.p->tableVersion = tabEntry->m_tableVersion + 1;
+
+ SimplePropertiesSectionWriter w(getSectionSegmentPool());
+ packTableIntoPagesImpl(w, tabPtr);
+
+ SegmentedSectionPtr spDataPtr;
+ w.getPtr(spDataPtr);
+
+ signal->setSection(spDataPtr, CreateTabReq::DICT_TAB_INFO);
+ signal->setSection(fragDataPtr, CreateTabReq::FRAGMENTATION);
+
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ SafeCounter tmp(c_counterMgr, createTabPtr.p->m_coordinatorData.m_counter);
+ createTabPtr.p->m_coordinatorData.m_gsn = GSN_CREATE_TAB_REQ;
+ createTabPtr.p->m_coordinatorData.m_requestType = CreateTabReq::CreateTablePrepare;
+ tmp.init<CreateTabRef>(rg, GSN_CREATE_TAB_REF, createTabPtr.p->key);
+
+ CreateTabReq * const req = (CreateTabReq*)theData;
+ req->senderRef = reference();
+ req->senderData = createTabPtr.p->key;
+ req->clientRef = createTabPtr.p->m_senderRef;
+ req->clientData = createTabPtr.p->m_senderData;
+ req->requestType = CreateTabReq::CreateTablePrepare;
+
+ req->gci = 0;
+ req->tableId = tabPtr.i;
+ req->tableVersion = tabEntry->m_tableVersion + 1;
+
+ sendSignal(rg, GSN_CREATE_TAB_REQ, signal,
+ CreateTabReq::SignalLength, JBB);
+
+
+ return;
+}
+
+void
+Dbdict::execCREATE_TAB_REF(Signal* signal){
+ jamEntry();
+
+ CreateTabRef * const ref = (CreateTabRef*)signal->getDataPtr();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, ref->senderData));
+
+ ndbrequire(createTabPtr.p->m_coordinatorRef == reference());
+ ndbrequire(createTabPtr.p->m_coordinatorData.m_gsn == GSN_CREATE_TAB_REQ);
+
+ if(ref->errorCode != CreateTabRef::NF_FakeErrorREF){
+ createTabPtr.p->setErrorCode(ref->errorCode);
+ }
+ createTab_reply(signal, createTabPtr, refToNode(ref->senderRef));
+}
+
+void
+Dbdict::execCREATE_TAB_CONF(Signal* signal){
+ jamEntry();
+
+ ndbrequire(signal->getNoOfSections() == 0);
+
+ CreateTabConf * const conf = (CreateTabConf*)signal->getDataPtr();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, conf->senderData));
+
+ ndbrequire(createTabPtr.p->m_coordinatorRef == reference());
+ ndbrequire(createTabPtr.p->m_coordinatorData.m_gsn == GSN_CREATE_TAB_REQ);
+
+ createTab_reply(signal, createTabPtr, refToNode(conf->senderRef));
+}
+
+void
+Dbdict::createTab_reply(Signal* signal,
+ CreateTableRecordPtr createTabPtr,
+ Uint32 nodeId)
+{
+
+ SafeCounter tmp(c_counterMgr, createTabPtr.p->m_coordinatorData.m_counter);
+ if(!tmp.clearWaitingFor(nodeId)){
+ jam();
+ return;
+ }
+
+ switch(createTabPtr.p->m_coordinatorData.m_requestType){
+ case CreateTabReq::CreateTablePrepare:{
+
+ if(createTabPtr.p->m_errorCode != 0){
+ jam();
+ /**
+ * Failed to prepare on atleast one node -> abort on all
+ */
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ createTabPtr.p->m_coordinatorData.m_gsn = GSN_CREATE_TAB_REQ;
+ createTabPtr.p->m_coordinatorData.m_requestType = CreateTabReq::CreateTableDrop;
+ ndbrequire(tmp.init<CreateTabRef>(rg, createTabPtr.p->key));
+
+ CreateTabReq * const req = (CreateTabReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = createTabPtr.p->key;
+ req->requestType = CreateTabReq::CreateTableDrop;
+
+ sendSignal(rg, GSN_CREATE_TAB_REQ, signal,
+ CreateTabReq::SignalLength, JBB);
+ return;
+ }
+
+ /**
+ * Lock mutex before commiting table
+ */
+ Mutex mutex(signal, c_mutexMgr, createTabPtr.p->m_startLcpMutex);
+ Callback c = { safe_cast(&Dbdict::createTab_startLcpMutex_locked),
+ createTabPtr.p->key};
+
+ ndbrequire(mutex.lock(c));
+ return;
+ }
+ case CreateTabReq::CreateTableCommit:{
+ jam();
+ ndbrequire(createTabPtr.p->m_errorCode == 0);
+
+ /**
+ * Unlock mutex before commiting table
+ */
+ Mutex mutex(signal, c_mutexMgr, createTabPtr.p->m_startLcpMutex);
+ Callback c = { safe_cast(&Dbdict::createTab_startLcpMutex_unlocked),
+ createTabPtr.p->key};
+ mutex.unlock(c);
+ return;
+ }
+ case CreateTabReq::CreateTableDrop:{
+ jam();
+ CreateTableRef * const ref = (CreateTableRef*)signal->getDataPtr();
+ ref->senderRef = reference();
+ ref->senderData = createTabPtr.p->m_senderData;
+ ref->errorCode = createTabPtr.p->m_errorCode;
+
+ //@todo check api failed
+ sendSignal(createTabPtr.p->m_senderRef, GSN_CREATE_TABLE_REF, signal,
+ CreateTableRef::SignalLength, JBB);
+ c_opCreateTable.release(createTabPtr);
+ c_blockState = BS_IDLE;
+ return;
+ }
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::createTab_startLcpMutex_locked(Signal* signal,
+ Uint32 callbackData,
+ Uint32 retValue){
+ jamEntry();
+
+ ndbrequire(retValue == 0);
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ createTabPtr.p->m_coordinatorData.m_gsn = GSN_CREATE_TAB_REQ;
+ createTabPtr.p->m_coordinatorData.m_requestType = CreateTabReq::CreateTableCommit;
+ SafeCounter tmp(c_counterMgr, createTabPtr.p->m_coordinatorData.m_counter);
+ tmp.init<CreateTabRef>(rg, GSN_CREATE_TAB_REF, createTabPtr.p->key);
+
+ CreateTabReq * const req = (CreateTabReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = createTabPtr.p->key;
+ req->requestType = CreateTabReq::CreateTableCommit;
+
+ sendSignal(rg, GSN_CREATE_TAB_REQ, signal,
+ CreateTabReq::SignalLength, JBB);
+}
+
+void
+Dbdict::createTab_startLcpMutex_unlocked(Signal* signal,
+ Uint32 callbackData,
+ Uint32 retValue){
+ jamEntry();
+
+ ndbrequire(retValue == 0);
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ createTabPtr.p->m_startLcpMutex.release(c_mutexMgr);
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+
+ CreateTableConf * const conf = (CreateTableConf*)signal->getDataPtr();
+ conf->senderRef = reference();
+ conf->senderData = createTabPtr.p->m_senderData;
+ conf->tableId = createTabPtr.p->m_tablePtrI;
+ conf->tableVersion = tabPtr.p->tableVersion;
+
+ //@todo check api failed
+ sendSignal(createTabPtr.p->m_senderRef, GSN_CREATE_TABLE_CONF, signal,
+ CreateTableConf::SignalLength, JBB);
+ c_opCreateTable.release(createTabPtr);
+ c_blockState = BS_IDLE;
+ return;
+}
+
+/***********************************************************
+ * CreateTable participant code
+ **********************************************************/
+void
+Dbdict::execCREATE_TAB_REQ(Signal* signal){
+ jamEntry();
+
+ if(!assembleFragments(signal)){
+ jam();
+ return;
+ }
+
+ CreateTabReq * const req = (CreateTabReq*)signal->getDataPtr();
+
+ CreateTabReq::RequestType rt = (CreateTabReq::RequestType)req->requestType;
+ switch(rt){
+ case CreateTabReq::CreateTablePrepare:
+ CRASH_INSERTION2(14000, getOwnNodeId() != c_masterNodeId);
+ createTab_prepare(signal, req);
+ return;
+ case CreateTabReq::CreateTableCommit:
+ CRASH_INSERTION2(14001, getOwnNodeId() != c_masterNodeId);
+ createTab_commit(signal, req);
+ return;
+ case CreateTabReq::CreateTableDrop:
+ CRASH_INSERTION2(14002, getOwnNodeId() != c_masterNodeId);
+ createTab_drop(signal, req);
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::createTab_prepare(Signal* signal, CreateTabReq * req){
+
+ const Uint32 gci = req->gci;
+ const Uint32 tableId = req->tableId;
+ const Uint32 tableVersion = req->tableVersion;
+
+ SegmentedSectionPtr tabInfoPtr;
+ signal->getSection(tabInfoPtr, CreateTabReq::DICT_TAB_INFO);
+
+ CreateTableRecordPtr createTabPtr;
+ if(req->senderRef == reference()){
+ jam();
+ ndbrequire(c_opCreateTable.find(createTabPtr, req->senderData));
+ } else {
+ jam();
+ c_opCreateTable.seize(createTabPtr);
+
+ ndbrequire(!createTabPtr.isNull());
+
+ createTabPtr.p->key = req->senderData;
+ c_opCreateTable.add(createTabPtr);
+ createTabPtr.p->m_errorCode = 0;
+ createTabPtr.p->m_tablePtrI = tableId;
+ createTabPtr.p->m_coordinatorRef = req->senderRef;
+ createTabPtr.p->m_senderRef = req->clientRef;
+ createTabPtr.p->m_senderData = req->clientData;
+ createTabPtr.p->m_dihAddFragPtr = RNIL;
+
+ /**
+ * Put data into table record
+ */
+ ParseDictTabInfoRecord parseRecord;
+ parseRecord.requestType = DictTabInfo::AddTableFromDict;
+ parseRecord.errorCode = 0;
+
+ SimplePropertiesSectionReader r(tabInfoPtr, getSectionSegmentPool());
+
+ handleTabInfoInit(r, &parseRecord);
+
+ ndbrequire(parseRecord.errorCode == 0);
+ }
+
+ ndbrequire(!createTabPtr.isNull());
+
+ SegmentedSectionPtr fragPtr;
+ signal->getSection(fragPtr, CreateTabReq::FRAGMENTATION);
+
+ createTabPtr.p->m_tabInfoPtrI = tabInfoPtr.i;
+ createTabPtr.p->m_fragmentsPtrI = fragPtr.i;
+
+ signal->header.m_noOfSections = 0;
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, tableId);
+ tabPtr.p->packedSize = tabInfoPtr.sz;
+ tabPtr.p->tableVersion = tableVersion;
+ tabPtr.p->gciTableCreated = gci;
+
+ SchemaFile::TableEntry tabEntry;
+ tabEntry.m_tableVersion = tableVersion;
+ tabEntry.m_tableType = tabPtr.p->tableType;
+ tabEntry.m_tableState = SchemaFile::ADD_STARTED;
+ tabEntry.m_gcp = gci;
+ tabEntry.m_noOfPages =
+ DIV(tabInfoPtr.sz + ZPAGE_HEADER_SIZE, ZSIZE_OF_PAGES_IN_WORDS);
+
+ Callback callback;
+ callback.m_callbackData = createTabPtr.p->key;
+ callback.m_callbackFunction =
+ safe_cast(&Dbdict::createTab_writeSchemaConf1);
+
+ updateSchemaState(signal, tableId, &tabEntry, &callback);
+}
+
+void getSection(SegmentedSectionPtr & ptr, Uint32 i);
+
+void
+Dbdict::createTab_writeSchemaConf1(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ Callback callback;
+ callback.m_callbackData = createTabPtr.p->key;
+ callback.m_callbackFunction =
+ safe_cast(&Dbdict::createTab_writeTableConf);
+
+ SegmentedSectionPtr tabInfoPtr;
+ getSection(tabInfoPtr, createTabPtr.p->m_tabInfoPtrI);
+
+ writeTableFile(signal, createTabPtr.p->m_tablePtrI, tabInfoPtr, &callback);
+
+ signal->setSection(tabInfoPtr, 0);
+ releaseSections(signal);
+}
+
+void
+Dbdict::createTab_writeTableConf(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ SegmentedSectionPtr fragDataPtr;
+ getSection(fragDataPtr, createTabPtr.p->m_fragmentsPtrI);
+
+ Callback callback;
+ callback.m_callbackData = callbackData;
+ callback.m_callbackFunction =
+ safe_cast(&Dbdict::createTab_dihComplete);
+
+ createTab_dih(signal, createTabPtr, fragDataPtr, &callback);
+}
+
+void
+Dbdict::createTab_dih(Signal* signal,
+ CreateTableRecordPtr createTabPtr,
+ SegmentedSectionPtr fragDataPtr,
+ Callback * c){
+ jam();
+
+ createTabPtr.p->m_callback = * c;
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+
+ DiAddTabReq * req = (DiAddTabReq*)signal->getDataPtrSend();
+ req->connectPtr = createTabPtr.p->key;
+ req->tableId = tabPtr.i;
+ req->fragType = tabPtr.p->fragmentType;
+ req->kValue = tabPtr.p->kValue;
+ req->noOfReplicas = 0;
+ req->storedTable = tabPtr.p->storedTable;
+ req->tableType = tabPtr.p->tableType;
+ req->schemaVersion = tabPtr.p->tableVersion;
+ req->primaryTableId = tabPtr.p->primaryTableId;
+
+ if(!fragDataPtr.isNull()){
+ signal->setSection(fragDataPtr, DiAddTabReq::FRAGMENTATION);
+ }
+
+ sendSignal(DBDIH_REF, GSN_DIADDTABREQ, signal,
+ DiAddTabReq::SignalLength, JBB);
+}
+
+static
+void
+calcLHbits(Uint32 * lhPageBits, Uint32 * lhDistrBits,
+ Uint32 fid, Uint32 totalFragments)
+{
+ Uint32 distrBits = 0;
+ Uint32 pageBits = 0;
+
+ Uint32 tmp = 1;
+ while (tmp < totalFragments) {
+ jam();
+ tmp <<= 1;
+ distrBits++;
+ }//while
+ if (tmp != totalFragments) {
+ tmp >>= 1;
+ if ((fid >= (totalFragments - tmp)) && (fid < (tmp - 1))) {
+ distrBits--;
+ }//if
+ }//if
+ * lhPageBits = pageBits;
+ * lhDistrBits = distrBits;
+
+}//calcLHbits()
+
+
+void
+Dbdict::execADD_FRAGREQ(Signal* signal) {
+ jamEntry();
+
+ AddFragReq * const req = (AddFragReq*)signal->getDataPtr();
+
+ Uint32 dihPtr = req->dihPtr;
+ Uint32 senderData = req->senderData;
+ Uint32 tableId = req->tableId;
+ Uint32 fragId = req->fragmentId;
+ Uint32 node = req->nodeId;
+ Uint32 lcpNo = req->nextLCP;
+ Uint32 fragCount = req->totalFragments;
+ Uint32 requestInfo = req->requestInfo;
+ Uint32 startGci = req->startGci;
+
+ ndbrequire(node == getOwnNodeId());
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, senderData));
+
+ createTabPtr.p->m_dihAddFragPtr = dihPtr;
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, tableId);
+
+#if 0
+ tabPtr.p->gciTableCreated = (startGci > tabPtr.p->gciTableCreated ? startGci:
+ startGci > tabPtr.p->gciTableCreated);
+#endif
+
+ /**
+ * Calc lh3PageBits
+ */
+ Uint32 lhDistrBits = 0;
+ Uint32 lhPageBits = 0;
+ ::calcLHbits(&lhPageBits, &lhDistrBits, fragId, fragCount);
+
+ {
+ LqhFragReq* req = (LqhFragReq*)signal->getDataPtrSend();
+ req->senderData = senderData;
+ req->senderRef = reference();
+ req->fragmentId = fragId;
+ req->requestInfo = requestInfo;
+ req->tableId = tableId;
+ req->localKeyLength = tabPtr.p->localKeyLen;
+ req->maxLoadFactor = tabPtr.p->maxLoadFactor;
+ req->minLoadFactor = tabPtr.p->minLoadFactor;
+ req->kValue = tabPtr.p->kValue;
+ req->lh3DistrBits = lhDistrBits;
+ req->lh3PageBits = lhPageBits;
+ req->noOfAttributes = tabPtr.p->noOfAttributes;
+ req->noOfNullAttributes = tabPtr.p->noOfNullAttr;
+ req->noOfPagesToPreAllocate = 0;
+ req->schemaVersion = tabPtr.p->tableVersion;
+ Uint32 keyLen = tabPtr.p->tupKeyLength;
+ req->keyLength = keyLen > 8 ? 0 : keyLen; // Put this into ACC instead
+ req->nextLCP = lcpNo;
+
+ req->noOfKeyAttr = tabPtr.p->noOfPrimkey;
+ req->noOfNewAttr = 0;
+ req->checksumIndicator = 1;
+ req->noOfAttributeGroups = 1;
+ req->GCPIndicator = 0;
+ req->startGci = startGci;
+ req->tableType = tabPtr.p->tableType;
+ req->primaryTableId = tabPtr.p->primaryTableId;
+ sendSignal(DBLQH_REF, GSN_LQHFRAGREQ, signal,
+ LqhFragReq::SignalLength, JBB);
+ }
+}
+
+void
+Dbdict::execLQHFRAGREF(Signal * signal){
+ jamEntry();
+ LqhFragRef * const ref = (LqhFragRef*)signal->getDataPtr();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, ref->senderData));
+
+ createTabPtr.p->setErrorCode(ref->errorCode);
+
+ {
+ AddFragRef * const ref = (AddFragRef*)signal->getDataPtr();
+ ref->dihPtr = createTabPtr.p->m_dihAddFragPtr;
+ sendSignal(DBDIH_REF, GSN_ADD_FRAGREF, signal,
+ AddFragRef::SignalLength, JBB);
+ }
+}
+
+void
+Dbdict::execLQHFRAGCONF(Signal * signal){
+ jamEntry();
+ LqhFragConf * const conf = (LqhFragConf*)signal->getDataPtr();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, conf->senderData));
+
+ createTabPtr.p->m_lqhFragPtr = conf->lqhFragPtr;
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+ sendLQHADDATTRREQ(signal, createTabPtr, tabPtr.p->firstAttribute);
+}
+
+void
+Dbdict::sendLQHADDATTRREQ(Signal* signal,
+ CreateTableRecordPtr createTabPtr,
+ Uint32 attributePtrI){
+ jam();
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+ LqhAddAttrReq * const req = (LqhAddAttrReq*)signal->getDataPtrSend();
+ Uint32 i = 0;
+ for(i = 0; i<LqhAddAttrReq::MAX_ATTRIBUTES && attributePtrI != RNIL; i++){
+ jam();
+ AttributeRecordPtr attrPtr;
+ c_attributeRecordPool.getPtr(attrPtr, attributePtrI);
+ LqhAddAttrReq::Entry& entry = req->attributes[i];
+ entry.attrId = attrPtr.p->attributeId;
+ entry.attrDescriptor = attrPtr.p->attributeDescriptor;
+ entry.extTypeInfo = attrPtr.p->extType;
+ if (tabPtr.p->isIndex()) {
+ Uint32 primaryAttrId;
+ if (attrPtr.p->nextAttrInTable != RNIL) {
+ getIndexAttr(tabPtr, attributePtrI, &primaryAttrId);
+ } else {
+ primaryAttrId = ZNIL;
+ if (tabPtr.p->isOrderedIndex())
+ entry.attrId = 0; // attribute goes to TUP
+ }
+ entry.attrId |= (primaryAttrId << 16);
+ }
+ attributePtrI = attrPtr.p->nextAttrInTable;
+ }
+ req->lqhFragPtr = createTabPtr.p->m_lqhFragPtr;
+ req->senderData = createTabPtr.p->key;
+ req->senderAttrPtr = attributePtrI;
+ req->noOfAttributes = i;
+
+ sendSignal(DBLQH_REF, GSN_LQHADDATTREQ, signal,
+ LqhAddAttrReq::HeaderLength + LqhAddAttrReq::EntryLength * i, JBB);
+}
+
+void
+Dbdict::execLQHADDATTREF(Signal * signal){
+ jamEntry();
+ LqhAddAttrRef * const ref = (LqhAddAttrRef*)signal->getDataPtr();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, ref->senderData));
+
+ createTabPtr.p->setErrorCode(ref->errorCode);
+
+ {
+ AddFragRef * const ref = (AddFragRef*)signal->getDataPtr();
+ ref->dihPtr = createTabPtr.p->m_dihAddFragPtr;
+ sendSignal(DBDIH_REF, GSN_ADD_FRAGREF, signal,
+ AddFragRef::SignalLength, JBB);
+ }
+
+}
+
+void
+Dbdict::execLQHADDATTCONF(Signal * signal){
+ jamEntry();
+ LqhAddAttrConf * const conf = (LqhAddAttrConf*)signal->getDataPtr();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, conf->senderData));
+
+ const Uint32 fragId = conf->fragId;
+ const Uint32 nextAttrPtr = conf->senderAttrPtr;
+ if(nextAttrPtr != RNIL){
+ jam();
+ sendLQHADDATTRREQ(signal, createTabPtr, nextAttrPtr);
+ return;
+ }
+
+ {
+ AddFragConf * const conf = (AddFragConf*)signal->getDataPtr();
+ conf->dihPtr = createTabPtr.p->m_dihAddFragPtr;
+ conf->fragId = fragId;
+ sendSignal(DBDIH_REF, GSN_ADD_FRAGCONF, signal,
+ AddFragConf::SignalLength, JBB);
+ }
+}
+
+void
+Dbdict::execDIADDTABREF(Signal* signal){
+ jam();
+
+ DiAddTabRef * const ref = (DiAddTabRef*)signal->getDataPtr();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, ref->senderData));
+
+ createTabPtr.p->setErrorCode(ref->errorCode);
+ execute(signal, createTabPtr.p->m_callback, 0);
+}
+
+void
+Dbdict::execDIADDTABCONF(Signal* signal){
+ jam();
+
+ DiAddTabConf * const conf = (DiAddTabConf*)signal->getDataPtr();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, conf->senderData));
+
+ signal->theData[0] = createTabPtr.p->key;
+ signal->theData[1] = reference();
+ signal->theData[2] = createTabPtr.p->m_tablePtrI;
+
+ if(createTabPtr.p->m_dihAddFragPtr != RNIL){
+ jam();
+
+ /**
+ * We did perform at least one LQHFRAGREQ
+ */
+ sendSignal(DBLQH_REF, GSN_TAB_COMMITREQ, signal, 3, JBB);
+ return;
+ } else {
+ /**
+ * No local fragment (i.e. no LQHFRAGREQ)
+ */
+ sendSignal(DBDIH_REF, GSN_TAB_COMMITREQ, signal, 3, JBB);
+ }
+}
+
+void
+Dbdict::execTAB_COMMITREF(Signal* signal) {
+ jamEntry();
+ ndbrequire(false);
+}//execTAB_COMMITREF()
+
+void
+Dbdict::execTAB_COMMITCONF(Signal* signal){
+ jamEntry();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, signal->theData[0]));
+
+ if(refToBlock(signal->getSendersBlockRef()) == DBLQH){
+
+ execute(signal, createTabPtr.p->m_callback, 0);
+ return;
+ }
+
+ if(refToBlock(signal->getSendersBlockRef()) == DBDIH){
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+
+ signal->theData[0] = tabPtr.i;
+ signal->theData[1] = tabPtr.p->tableVersion;
+ signal->theData[2] = (Uint32)tabPtr.p->storedTable;
+ signal->theData[3] = reference();
+ signal->theData[4] = (Uint32)tabPtr.p->tableType;
+ signal->theData[5] = createTabPtr.p->key;
+ sendSignal(DBTC_REF, GSN_TC_SCHVERREQ, signal, 6, JBB);
+ return;
+ }
+
+ ndbrequire(false);
+}
+
+void
+Dbdict::createTab_dihComplete(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ //@todo check for master failed
+
+ if(createTabPtr.p->m_errorCode == 0){
+ jam();
+
+ CreateTabConf * const conf = (CreateTabConf*)signal->getDataPtr();
+ conf->senderRef = reference();
+ conf->senderData = createTabPtr.p->key;
+ sendSignal(createTabPtr.p->m_coordinatorRef, GSN_CREATE_TAB_CONF,
+ signal, CreateTabConf::SignalLength, JBB);
+ return;
+ }
+
+ CreateTabRef * const ref = (CreateTabRef*)signal->getDataPtr();
+ ref->senderRef = reference();
+ ref->senderData = createTabPtr.p->key;
+ ref->errorCode = createTabPtr.p->m_errorCode;
+ ref->errorLine = 0;
+ ref->errorKey = 0;
+ ref->errorStatus = 0;
+
+ sendSignal(createTabPtr.p->m_coordinatorRef, GSN_CREATE_TAB_REF,
+ signal, CreateTabRef::SignalLength, JBB);
+}
+
+void
+Dbdict::createTab_commit(Signal * signal, CreateTabReq * req){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, req->senderData));
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+
+ SchemaFile::TableEntry tabEntry;
+ tabEntry.m_tableVersion = tabPtr.p->tableVersion;
+ tabEntry.m_tableType = tabPtr.p->tableType;
+ tabEntry.m_tableState = SchemaFile::TABLE_ADD_COMMITTED;
+ tabEntry.m_gcp = tabPtr.p->gciTableCreated;
+ tabEntry.m_noOfPages =
+ DIV(tabPtr.p->packedSize + ZPAGE_HEADER_SIZE, ZSIZE_OF_PAGES_IN_WORDS);
+
+ Callback callback;
+ callback.m_callbackData = createTabPtr.p->key;
+ callback.m_callbackFunction =
+ safe_cast(&Dbdict::createTab_writeSchemaConf2);
+
+ updateSchemaState(signal, tabPtr.i, &tabEntry, &callback);
+}
+
+void
+Dbdict::createTab_writeSchemaConf2(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ Callback c;
+ c.m_callbackData = callbackData;
+ c.m_callbackFunction = safe_cast(&Dbdict::createTab_alterComplete);
+ alterTab_activate(signal, createTabPtr, &c);
+}
+
+void
+Dbdict::createTab_alterComplete(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+ tabPtr.p->tabState = TableRecord::DEFINED;
+
+ //@todo check error
+ //@todo check master failed
+
+ CreateTabConf * const conf = (CreateTabConf*)signal->getDataPtr();
+ conf->senderRef = reference();
+ conf->senderData = createTabPtr.p->key;
+ sendSignal(createTabPtr.p->m_coordinatorRef, GSN_CREATE_TAB_CONF,
+ signal, CreateTabConf::SignalLength, JBB);
+
+ if(createTabPtr.p->m_coordinatorRef != reference()){
+ jam();
+ c_opCreateTable.release(createTabPtr);
+ }
+}
+
+void
+Dbdict::createTab_drop(Signal* signal, CreateTabReq * req){
+ jam();
+
+ const Uint32 key = req->senderData;
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, key));
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+ tabPtr.p->tabState = TableRecord::DROPPING;
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.seize(dropTabPtr));
+
+ dropTabPtr.p->key = key;
+ c_opDropTable.add(dropTabPtr);
+
+ dropTabPtr.p->m_errorCode = 0;
+ dropTabPtr.p->m_request.tableId = createTabPtr.p->m_tablePtrI;
+ dropTabPtr.p->m_requestType = DropTabReq::CreateTabDrop;
+ dropTabPtr.p->m_coordinatorRef = createTabPtr.p->m_coordinatorRef;
+ dropTabPtr.p->m_participantData.m_gsn = GSN_DROP_TAB_REQ;
+
+ dropTabPtr.p->m_participantData.m_block = 0;
+ dropTabPtr.p->m_participantData.m_callback.m_callbackData = req->senderData;
+ dropTabPtr.p->m_participantData.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::createTab_dropComplete);
+ dropTab_nextStep(signal, dropTabPtr);
+}
+
+void
+Dbdict::createTab_dropComplete(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode){
+ jam();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, callbackData));
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, callbackData));
+
+ TableRecordPtr tabPtr;
+ c_tableRecordPool.getPtr(tabPtr, createTabPtr.p->m_tablePtrI);
+ tabPtr.p->tabState = TableRecord::NOT_DEFINED;
+
+ releaseTableObject(tabPtr.i);
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.schemaPage);
+
+ SchemaFile::TableEntry * tableEntry = getTableEntry(pagePtr.p, tabPtr.i);
+ tableEntry->m_tableState = SchemaFile::DROP_TABLE_COMMITTED;
+
+ //@todo check error
+ //@todo check master failed
+
+ CreateTabConf * const conf = (CreateTabConf*)signal->getDataPtr();
+ conf->senderRef = reference();
+ conf->senderData = createTabPtr.p->key;
+ sendSignal(createTabPtr.p->m_coordinatorRef, GSN_CREATE_TAB_CONF,
+ signal, CreateTabConf::SignalLength, JBB);
+
+ if(createTabPtr.p->m_coordinatorRef != reference()){
+ jam();
+ c_opCreateTable.release(createTabPtr);
+ }
+
+ c_opDropTable.release(dropTabPtr);
+}
+
+void
+Dbdict::alterTab_activate(Signal* signal, CreateTableRecordPtr createTabPtr,
+ Callback * c){
+
+ createTabPtr.p->m_callback = * c;
+
+ signal->theData[0] = createTabPtr.p->key;
+ signal->theData[1] = reference();
+ signal->theData[2] = createTabPtr.p->m_tablePtrI;
+ sendSignal(DBDIH_REF, GSN_TAB_COMMITREQ, signal, 3, JBB);
+}
+
+void
+Dbdict::execTC_SCHVERCONF(Signal* signal){
+ jamEntry();
+
+ CreateTableRecordPtr createTabPtr;
+ ndbrequire(c_opCreateTable.find(createTabPtr, signal->theData[1]));
+
+ execute(signal, createTabPtr.p->m_callback, 0);
+}
+
+#define tabRequire(cond, error) \
+ if (!(cond)) { \
+ jam(); \
+ parseP->errorCode = error; parseP->errorLine = __LINE__; \
+ parseP->errorKey = it.getKey(); \
+ return; \
+ }//if
+
+// handleAddTableFailure(signal, __LINE__, allocatedTable);
+
+void Dbdict::handleTabInfoInit(SimpleProperties::Reader & it,
+ ParseDictTabInfoRecord * parseP,
+ bool checkExist)
+{
+/* ---------------------------------------------------------------- */
+// We always start by handling table name since this must be the first
+// item in the list. Through the table name we can derive if it is a
+// correct name, a new name or an already existing table.
+/* ---------------------------------------------------------------- */
+
+ it.first();
+
+ SimpleProperties::UnpackStatus status;
+ DictTabInfo::Table tableDesc; tableDesc.init();
+ status = SimpleProperties::unpack(it, &tableDesc,
+ DictTabInfo::TableMapping,
+ DictTabInfo::TableMappingSize,
+ true, true);
+
+ if(status != SimpleProperties::Break){
+ parseP->errorCode = CreateTableRef::InvalidFormat;
+ parseP->status = status;
+ parseP->errorKey = it.getKey();
+ parseP->errorLine = __LINE__;
+ return;
+ }
+
+ /* ---------------------------------------------------------------- */
+ // Verify that table name is an allowed table name.
+ // TODO
+ /* ---------------------------------------------------------------- */
+ const Uint32 tableNameLength = strlen(tableDesc.TableName) + 1;
+
+ TableRecord keyRecord;
+ tabRequire(tableNameLength <= sizeof(keyRecord.tableName),
+ CreateTableRef::TableNameTooLong);
+ strcpy(keyRecord.tableName, tableDesc.TableName);
+
+ TableRecordPtr tablePtr;
+ c_tableRecordHash.find(tablePtr, keyRecord);
+
+ if (checkExist)
+ jam();
+ /* ---------------------------------------------------------------- */
+ // Check if table already existed.
+ /* ---------------------------------------------------------------- */
+ tabRequire(tablePtr.i == RNIL, CreateTableRef::TableAlreadyExist);
+
+ switch (parseP->requestType) {
+ case DictTabInfo::CreateTableFromAPI: {
+ jam();
+ }
+ case DictTabInfo::AlterTableFromAPI:{
+ jam();
+ tablePtr.i = getFreeTableRecord(tableDesc.PrimaryTableId);
+ /* ---------------------------------------------------------------- */
+ // Check if no free tables existed.
+ /* ---------------------------------------------------------------- */
+ tabRequire(tablePtr.i != RNIL, CreateTableRef::NoMoreTableRecords);
+
+ c_tableRecordPool.getPtr(tablePtr);
+ break;
+ }
+ case DictTabInfo::AddTableFromDict:
+ case DictTabInfo::ReadTableFromDiskSR:
+ case DictTabInfo::GetTabInfoConf:
+ {
+/* ---------------------------------------------------------------- */
+// Get table id and check that table doesn't already exist
+/* ---------------------------------------------------------------- */
+ tablePtr.i = tableDesc.TableId;
+
+ if (parseP->requestType == DictTabInfo::ReadTableFromDiskSR) {
+ ndbrequire(tablePtr.i == c_restartRecord.activeTable);
+ }//if
+ if (parseP->requestType == DictTabInfo::GetTabInfoConf) {
+ ndbrequire(tablePtr.i == c_restartRecord.activeTable);
+ }//if
+
+ c_tableRecordPool.getPtr(tablePtr);
+ ndbrequire(tablePtr.p->tabState == TableRecord::NOT_DEFINED);
+
+ //Uint32 oldTableVersion = tablePtr.p->tableVersion;
+ initialiseTableRecord(tablePtr);
+ if (parseP->requestType == DictTabInfo::AddTableFromDict) {
+ jam();
+ tablePtr.p->tabState = TableRecord::DEFINING;
+ }//if
+
+/* ---------------------------------------------------------------- */
+// Get id of second table id and check that table doesn't already exist
+// and set up links between first and second table.
+/* ---------------------------------------------------------------- */
+ TableRecordPtr secondTablePtr;
+ secondTablePtr.i = tableDesc.SecondTableId;
+ c_tableRecordPool.getPtr(secondTablePtr);
+ ndbrequire(secondTablePtr.p->tabState == TableRecord::NOT_DEFINED);
+
+ initialiseTableRecord(secondTablePtr);
+ secondTablePtr.p->tabState = TableRecord::REORG_TABLE_PREPARED;
+ secondTablePtr.p->secondTable = tablePtr.i;
+ tablePtr.p->secondTable = secondTablePtr.i;
+
+/* ---------------------------------------------------------------- */
+// Set table version
+/* ---------------------------------------------------------------- */
+ Uint32 tableVersion = tableDesc.TableVersion;
+ tablePtr.p->tableVersion = tableVersion;
+
+ break;
+ }
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ parseP->tablePtr = tablePtr;
+
+ strcpy(tablePtr.p->tableName, keyRecord.tableName);
+ if (parseP->requestType != DictTabInfo::AlterTableFromAPI) {
+ jam();
+ c_tableRecordHash.add(tablePtr);
+ }
+
+#ifdef VM_TRACE
+ ndbout_c("Dbdict: name=%s,id=%u", tablePtr.p->tableName, tablePtr.i);
+#endif
+
+ //tablePtr.p->noOfPrimkey = tableDesc.NoOfKeyAttr;
+ //tablePtr.p->noOfNullAttr = tableDesc.NoOfNullable;
+ //tablePtr.p->tupKeyLength = tableDesc.KeyLength;
+ tablePtr.p->noOfAttributes = tableDesc.NoOfAttributes;
+ tablePtr.p->storedTable = tableDesc.TableLoggedFlag;
+ tablePtr.p->minLoadFactor = tableDesc.MinLoadFactor;
+ tablePtr.p->maxLoadFactor = tableDesc.MaxLoadFactor;
+ tablePtr.p->fragmentType = (DictTabInfo::FragmentType)tableDesc.FragmentType;
+ tablePtr.p->fragmentKeyType = (DictTabInfo::FragmentKeyType)tableDesc.FragmentKeyType;
+ tablePtr.p->tableType = (DictTabInfo::TableType)tableDesc.TableType;
+ tablePtr.p->kValue = tableDesc.TableKValue;
+
+ tablePtr.p->frmLen = tableDesc.FrmLen;
+ memcpy(tablePtr.p->frmData, tableDesc.FrmData, tableDesc.FrmLen);
+
+ if(tableDesc.PrimaryTableId != RNIL) {
+
+ tablePtr.p->primaryTableId = tableDesc.PrimaryTableId;
+ tablePtr.p->indexState = (TableRecord::IndexState)tableDesc.IndexState;
+ tablePtr.p->insertTriggerId = tableDesc.InsertTriggerId;
+ tablePtr.p->updateTriggerId = tableDesc.UpdateTriggerId;
+ tablePtr.p->deleteTriggerId = tableDesc.DeleteTriggerId;
+ tablePtr.p->customTriggerId = tableDesc.CustomTriggerId;
+ } else {
+ tablePtr.p->primaryTableId = RNIL;
+ tablePtr.p->indexState = TableRecord::IS_UNDEFINED;
+ tablePtr.p->insertTriggerId = RNIL;
+ tablePtr.p->updateTriggerId = RNIL;
+ tablePtr.p->deleteTriggerId = RNIL;
+ tablePtr.p->customTriggerId = RNIL;
+ }
+ tablePtr.p->buildTriggerId = RNIL;
+ tablePtr.p->indexLocal = 0;
+
+ handleTabInfo(it, parseP);
+
+ if(parseP->errorCode != 0){
+ /**
+ * Release table
+ */
+ releaseTableObject(tablePtr.i);
+ }
+}//handleTabInfoInit()
+
+void Dbdict::handleTabInfo(SimpleProperties::Reader & it,
+ ParseDictTabInfoRecord * parseP)
+{
+ TableRecordPtr tablePtr = parseP->tablePtr;
+
+ SimpleProperties::UnpackStatus status;
+
+ Uint32 keyCount = 0;
+ Uint32 keyLength = 0;
+ Uint32 attrCount = tablePtr.p->noOfAttributes;
+ Uint32 nullCount = 0;
+ Uint32 recordLength = 0;
+ AttributeRecordPtr attrPtr;
+ c_attributeRecordHash.removeAll();
+
+ for(Uint32 i = 0; i<attrCount; i++){
+ /**
+ * Attribute Name
+ */
+ DictTabInfo::Attribute attrDesc; attrDesc.init();
+ status = SimpleProperties::unpack(it, &attrDesc,
+ DictTabInfo::AttributeMapping,
+ DictTabInfo::AttributeMappingSize,
+ true, true);
+ if(status != SimpleProperties::Break){
+ parseP->errorCode = CreateTableRef::InvalidFormat;
+ parseP->status = status;
+ parseP->errorKey = it.getKey();
+ parseP->errorLine = __LINE__;
+ return;
+ }
+
+ /**
+ * Check that attribute is not defined twice
+ */
+ AttributeRecord tmpAttr;
+ {
+ strcpy(tmpAttr.attributeName, attrDesc.AttributeName);
+
+ AttributeRecordPtr attrPtr;
+ c_attributeRecordHash.find(attrPtr, tmpAttr);
+
+ if(attrPtr.i != RNIL){
+ parseP->errorCode = CreateTableRef::AttributeNameTwice;
+ return;
+ }
+ }
+
+ if(!getNewAttributeRecord(tablePtr, attrPtr)){
+ jam();
+ parseP->errorCode = CreateTableRef::NoMoreAttributeRecords;
+ return;
+ }
+
+ /**
+ * TmpAttrib to Attribute mapping
+ */
+ strcpy(attrPtr.p->attributeName, attrDesc.AttributeName);
+ attrPtr.p->attributeId = attrDesc.AttributeId;
+ attrPtr.p->tupleKey = (keyCount + 1) * attrDesc.AttributeKeyFlag;
+
+ attrPtr.p->extType = attrDesc.AttributeExtType;
+ attrPtr.p->extPrecision = attrDesc.AttributeExtPrecision;
+ attrPtr.p->extScale = attrDesc.AttributeExtScale;
+ attrPtr.p->extLength = attrDesc.AttributeExtLength;
+
+ /**
+ * Ignore incoming old-style type and recompute it.
+ */
+ bool translateOk = attrDesc.translateExtType();
+ tabRequire(translateOk, CreateTableRef::Inconsistency);
+
+ if(attrDesc.AttributeArraySize > 65535){
+ parseP->errorCode = CreateTableRef::ArraySizeTooBig;
+ parseP->status = status;
+ parseP->errorKey = it.getKey();
+ parseP->errorLine = __LINE__;
+ return;
+ }
+
+ Uint32 desc = 0;
+ AttributeDescriptor::setType(desc, attrDesc.AttributeType);
+ AttributeDescriptor::setSize(desc, attrDesc.AttributeSize);
+ AttributeDescriptor::setArray(desc, attrDesc.AttributeArraySize);
+ AttributeDescriptor::setNullable(desc, attrDesc.AttributeNullableFlag);
+ AttributeDescriptor::setDGroup(desc, attrDesc.AttributeDGroup);
+ AttributeDescriptor::setDKey(desc, attrDesc.AttributeDKey);
+ AttributeDescriptor::setPrimaryKey(desc, attrDesc.AttributeKeyFlag);
+
+ AttributeDescriptor::setStoredInTup(desc, attrDesc.AttributeStoredInd);
+ attrPtr.p->attributeDescriptor = desc;
+ attrPtr.p->autoIncrement = attrDesc.AttributeAutoIncrement;
+ strcpy(attrPtr.p->defaultValue, attrDesc.AttributeDefaultValue);
+
+ tabRequire(attrDesc.AttributeId == i, CreateTableRef::InvalidFormat);
+
+ attrCount ++;
+ keyCount += attrDesc.AttributeKeyFlag;
+ nullCount += attrDesc.AttributeNullableFlag;
+
+ const Uint32 aSz = (1 << attrDesc.AttributeSize);
+ const Uint32 sz = ((aSz * attrDesc.AttributeArraySize) + 31) >> 5;
+
+ recordLength += sz;
+ if(attrDesc.AttributeKeyFlag){
+ keyLength += sz;
+
+ if(attrDesc.AttributeNullableFlag){
+ parseP->errorCode = CreateTableRef::NullablePrimaryKey;
+ parseP->status = status;
+ parseP->errorKey = it.getKey();
+ parseP->errorLine = __LINE__;
+ return;
+ }
+ }
+
+ if (parseP->requestType != DictTabInfo::AlterTableFromAPI)
+ c_attributeRecordHash.add(attrPtr);
+
+ if(!it.next())
+ break;
+
+ if(it.getKey() != DictTabInfo::AttributeName)
+ break;
+ }//while
+
+ tablePtr.p->noOfPrimkey = keyCount;
+ tablePtr.p->noOfNullAttr = nullCount;
+ tablePtr.p->tupKeyLength = keyLength;
+
+ tabRequire(recordLength<= MAX_TUPLE_SIZE_IN_WORDS,
+ CreateTableRef::RecordTooBig);
+ tabRequire(keyLength <= MAX_KEY_SIZE_IN_WORDS,
+ CreateTableRef::InvalidPrimaryKeySize);
+ tabRequire(keyLength > 0,
+ CreateTableRef::InvalidPrimaryKeySize);
+
+}//handleTabInfo()
+
+
+/* ---------------------------------------------------------------- */
+// DICTTABCONF is sent when participants have received all DICTTABINFO
+// and successfully handled it.
+// Also sent to self (DICT master) when index table creation ready.
+/* ---------------------------------------------------------------- */
+void Dbdict::execCREATE_TABLE_CONF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(signal->getNoOfSections() == 0);
+
+ CreateTableConf * const conf = (CreateTableConf *)signal->getDataPtr();
+ // assume part of create index operation
+ OpCreateIndexPtr opPtr;
+ c_opCreateIndex.find(opPtr, conf->senderData);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->m_request.setIndexId(conf->tableId);
+ opPtr.p->m_request.setIndexVersion(conf->tableVersion);
+ createIndex_fromCreateTable(signal, opPtr);
+}//execCREATE_TABLE_CONF()
+
+void Dbdict::execCREATE_TABLE_REF(Signal* signal)
+{
+ jamEntry();
+
+ CreateTableRef * const ref = (CreateTableRef *)signal->getDataPtr();
+ // assume part of create index operation
+ OpCreateIndexPtr opPtr;
+ c_opCreateIndex.find(opPtr, ref->senderData);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ createIndex_fromCreateTable(signal, opPtr);
+}//execCREATE_TABLE_REF()
+
+/* ---------------------------------------------------------------- */
+// New global checkpoint created.
+/* ---------------------------------------------------------------- */
+void Dbdict::execWAIT_GCP_CONF(Signal* signal)
+{
+#if 0
+ TableRecordPtr tablePtr;
+ jamEntry();
+ WaitGCPConf* const conf = (WaitGCPConf*)&signal->theData[0];
+ c_tableRecordPool.getPtr(tablePtr, c_connRecord.connTableId);
+ tablePtr.p->gciTableCreated = conf->gcp;
+ sendUpdateSchemaState(signal,
+ tablePtr.i,
+ SchemaFile::TABLE_ADD_COMMITTED,
+ c_connRecord.noOfPagesForTable,
+ conf->gcp);
+#endif
+}//execWAIT_GCP_CONF()
+
+/* ---------------------------------------------------------------- */
+// Refused new global checkpoint.
+/* ---------------------------------------------------------------- */
+void Dbdict::execWAIT_GCP_REF(Signal* signal)
+{
+ jamEntry();
+ WaitGCPRef* const ref = (WaitGCPRef*)&signal->theData[0];
+/* ---------------------------------------------------------------- */
+// Error Handling code needed
+/* ---------------------------------------------------------------- */
+ progError(ref->errorCode, 0);
+}//execWAIT_GCP_REF()
+
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: DROP TABLE -------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* This module contains the code used to drop a table. */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+void
+Dbdict::execDROP_TABLE_REQ(Signal* signal){
+ jamEntry();
+ DropTableReq* req = (DropTableReq*)signal->getDataPtr();
+
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, req->tableId, false);
+ if(tablePtr.isNull()){
+ jam();
+ dropTableRef(signal, req, DropTableRef::NoSuchTable);
+ return;
+ }
+
+ if(getOwnNodeId() != c_masterNodeId){
+ jam();
+ dropTableRef(signal, req, DropTableRef::NotMaster);
+ return;
+ }
+
+ if(c_blockState != BS_IDLE){
+ jam();
+ dropTableRef(signal, req, DropTableRef::Busy);
+ return;
+ }
+
+ const TableRecord::TabState tabState = tablePtr.p->tabState;
+ bool ok = false;
+ switch(tabState){
+ case TableRecord::NOT_DEFINED:
+ case TableRecord::REORG_TABLE_PREPARED:
+ case TableRecord::DEFINING:
+ case TableRecord::CHECKED:
+ jam();
+ dropTableRef(signal, req, DropTableRef::NoSuchTable);
+ return;
+ case TableRecord::DEFINED:
+ ok = true;
+ jam();
+ break;
+ case TableRecord::PREPARE_DROPPING:
+ case TableRecord::DROPPING:
+ jam();
+ dropTableRef(signal, req, DropTableRef::DropInProgress);
+ return;
+ }
+ ndbrequire(ok);
+
+ if(tablePtr.p->tableVersion != req->tableVersion){
+ jam();
+ dropTableRef(signal, req, DropTableRef::InvalidTableVersion);
+ return;
+ }
+
+ /**
+ * Seems ok
+ */
+ DropTableRecordPtr dropTabPtr;
+ c_opDropTable.seize(dropTabPtr);
+
+ if(dropTabPtr.isNull()){
+ jam();
+ dropTableRef(signal, req, DropTableRef::NoDropTableRecordAvailable);
+ return;
+ }
+
+ c_blockState = BS_BUSY;
+
+ dropTabPtr.p->key = ++c_opRecordSequence;
+ c_opDropTable.add(dropTabPtr);
+
+ tablePtr.p->tabState = TableRecord::PREPARE_DROPPING;
+
+ dropTabPtr.p->m_request = * req;
+ dropTabPtr.p->m_errorCode = 0;
+ dropTabPtr.p->m_requestType = DropTabReq::OnlineDropTab;
+ dropTabPtr.p->m_coordinatorRef = reference();
+ dropTabPtr.p->m_coordinatorData.m_gsn = GSN_PREP_DROP_TAB_REQ;
+ dropTabPtr.p->m_coordinatorData.m_block = 0;
+ prepDropTab_nextStep(signal, dropTabPtr);
+}
+
+void
+Dbdict::dropTableRef(Signal * signal,
+ DropTableReq * req, DropTableRef::ErrorCode errCode){
+
+ Uint32 tableId = req->tableId;
+ Uint32 tabVersion = req->tableVersion;
+ Uint32 senderData = req->senderData;
+ Uint32 senderRef = req->senderRef;
+
+ DropTableRef * ref = (DropTableRef*)signal->getDataPtrSend();
+ ref->tableId = tableId;
+ ref->tableVersion = tabVersion;
+ ref->senderData = senderData;
+ ref->senderRef = reference();
+ ref->errorCode = errCode;
+ ref->masterNodeId = c_masterNodeId;
+ sendSignal(senderRef, GSN_DROP_TABLE_REF, signal,
+ DropTableRef::SignalLength, JBB);
+}
+
+void
+Dbdict::prepDropTab_nextStep(Signal* signal, DropTableRecordPtr dropTabPtr){
+
+ /**
+ * No errors currently allowed
+ */
+ ndbrequire(dropTabPtr.p->m_errorCode == 0);
+
+ Uint32 block = 0;
+ switch(dropTabPtr.p->m_coordinatorData.m_block){
+ case 0:
+ jam();
+ block = dropTabPtr.p->m_coordinatorData.m_block = DBDICT;
+ break;
+ case DBDICT:
+ jam();
+ block = dropTabPtr.p->m_coordinatorData.m_block = DBLQH;
+ break;
+ case DBLQH:
+ jam();
+ block = dropTabPtr.p->m_coordinatorData.m_block = DBTC;
+ break;
+ case DBTC:
+ jam();
+ block = dropTabPtr.p->m_coordinatorData.m_block = DBDIH;
+ break;
+ case DBDIH:
+ jam();
+ prepDropTab_complete(signal, dropTabPtr);
+ return;
+ default:
+ ndbrequire(false);
+ }
+
+ PrepDropTabReq * prep = (PrepDropTabReq*)signal->getDataPtrSend();
+ prep->senderRef = reference();
+ prep->senderData = dropTabPtr.p->key;
+ prep->tableId = dropTabPtr.p->m_request.tableId;
+ prep->requestType = dropTabPtr.p->m_requestType;
+
+ dropTabPtr.p->m_coordinatorData.m_signalCounter = c_aliveNodes;
+ NodeReceiverGroup rg(block, c_aliveNodes);
+ sendSignal(rg, GSN_PREP_DROP_TAB_REQ, signal,
+ PrepDropTabReq::SignalLength, JBB);
+
+#if 0
+ for (Uint32 i = 1; i < MAX_NDB_NODES; i++){
+ if(c_aliveNodes.get(i)){
+ jam();
+ BlockReference ref = numberToRef(block, i);
+
+ dropTabPtr.p->m_coordinatorData.m_signalCounter.setWaitingFor(i);
+ }
+ }
+#endif
+}
+
+void
+Dbdict::execPREP_DROP_TAB_CONF(Signal * signal){
+ jamEntry();
+
+ PrepDropTabConf * prep = (PrepDropTabConf*)signal->getDataPtr();
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, prep->senderData));
+
+ ndbrequire(dropTabPtr.p->m_coordinatorRef == reference());
+ ndbrequire(dropTabPtr.p->m_request.tableId == prep->tableId);
+ ndbrequire(dropTabPtr.p->m_coordinatorData.m_gsn == GSN_PREP_DROP_TAB_REQ);
+
+ Uint32 nodeId = refToNode(prep->senderRef);
+ dropTabPtr.p->m_coordinatorData.m_signalCounter.clearWaitingFor(nodeId);
+
+ if(!dropTabPtr.p->m_coordinatorData.m_signalCounter.done()){
+ jam();
+ return;
+ }
+ prepDropTab_nextStep(signal, dropTabPtr);
+}
+
+void
+Dbdict::execPREP_DROP_TAB_REF(Signal* signal){
+ jamEntry();
+
+ PrepDropTabRef * prep = (PrepDropTabRef*)signal->getDataPtr();
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, prep->senderData));
+
+ ndbrequire(dropTabPtr.p->m_coordinatorRef == reference());
+ ndbrequire(dropTabPtr.p->m_request.tableId == prep->tableId);
+ ndbrequire(dropTabPtr.p->m_coordinatorData.m_gsn == GSN_PREP_DROP_TAB_REQ);
+
+ Uint32 nodeId = refToNode(prep->senderRef);
+ dropTabPtr.p->m_coordinatorData.m_signalCounter.clearWaitingFor(nodeId);
+
+ dropTabPtr.p->setErrorCode((Uint32)prep->errorCode);
+ if(!dropTabPtr.p->m_coordinatorData.m_signalCounter.done()){
+ jam();
+ return;
+ }
+ prepDropTab_nextStep(signal, dropTabPtr);
+}
+
+void
+Dbdict::prepDropTab_complete(Signal* signal, DropTableRecordPtr dropTabPtr){
+ jam();
+
+ dropTabPtr.p->m_coordinatorData.m_gsn = GSN_DROP_TAB_REQ;
+ dropTabPtr.p->m_coordinatorData.m_block = DBDICT;
+
+ DropTabReq * req = (DropTabReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = dropTabPtr.p->key;
+ req->tableId = dropTabPtr.p->m_request.tableId;
+ req->requestType = dropTabPtr.p->m_requestType;
+
+ dropTabPtr.p->m_coordinatorData.m_signalCounter = c_aliveNodes;
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ sendSignal(rg, GSN_DROP_TAB_REQ, signal,
+ DropTabReq::SignalLength, JBB);
+}
+
+void
+Dbdict::execDROP_TAB_REF(Signal* signal){
+ jamEntry();
+
+ ndbrequire(false);
+}
+
+void
+Dbdict::execDROP_TAB_CONF(Signal* signal){
+ jamEntry();
+
+ DropTabConf * const req = (DropTabConf*)signal->getDataPtr();
+
+ if(refToBlock(req->senderRef) != DBDICT){
+ jam();
+ ndbrequire(refToNode(req->senderRef) == getOwnNodeId());
+ dropTab_localDROP_TAB_CONF(signal);
+ return;
+ }
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, req->senderData));
+
+ ndbrequire(dropTabPtr.p->m_coordinatorRef == reference());
+ ndbrequire(dropTabPtr.p->m_request.tableId == req->tableId);
+ ndbrequire(dropTabPtr.p->m_coordinatorData.m_gsn == GSN_DROP_TAB_REQ);
+
+ Uint32 nodeId = refToNode(req->senderRef);
+ dropTabPtr.p->m_coordinatorData.m_signalCounter.clearWaitingFor(nodeId);
+
+ if(!dropTabPtr.p->m_coordinatorData.m_signalCounter.done()){
+ jam();
+ return;
+ }
+
+ DropTableConf* conf = (DropTableConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = dropTabPtr.p->m_request.senderData;
+ conf->tableId = dropTabPtr.p->m_request.tableId;
+ conf->tableVersion = dropTabPtr.p->m_request.tableVersion;
+
+ Uint32 ref = dropTabPtr.p->m_request.senderRef;
+ sendSignal(ref, GSN_DROP_TABLE_CONF, signal,
+ DropTableConf::SignalLength, JBB);
+
+ c_opDropTable.release(dropTabPtr);
+ c_blockState = BS_IDLE;
+}
+
+/**
+ * DROP TABLE PARTICIPANT CODE
+ */
+void
+Dbdict::execPREP_DROP_TAB_REQ(Signal* signal){
+ jamEntry();
+ PrepDropTabReq * prep = (PrepDropTabReq*)signal->getDataPtrSend();
+
+ DropTableRecordPtr dropTabPtr;
+ if(prep->senderRef == reference()){
+ jam();
+ ndbrequire(c_opDropTable.find(dropTabPtr, prep->senderData));
+ ndbrequire(dropTabPtr.p->m_requestType == prep->requestType);
+ } else {
+ jam();
+ c_opDropTable.seize(dropTabPtr);
+ if(!dropTabPtr.isNull()){
+ dropTabPtr.p->key = prep->senderData;
+ c_opDropTable.add(dropTabPtr);
+ }
+ }
+
+ ndbrequire(!dropTabPtr.isNull());
+
+ dropTabPtr.p->m_errorCode = 0;
+ dropTabPtr.p->m_request.tableId = prep->tableId;
+ dropTabPtr.p->m_requestType = prep->requestType;
+ dropTabPtr.p->m_coordinatorRef = prep->senderRef;
+ dropTabPtr.p->m_participantData.m_gsn = GSN_PREP_DROP_TAB_REQ;
+
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, prep->tableId);
+ tablePtr.p->tabState = TableRecord::PREPARE_DROPPING;
+
+ /**
+ * Modify schema
+ */
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.schemaPage);
+
+ SchemaFile::TableEntry * tableEntry = getTableEntry(pagePtr.p, tablePtr.i);
+ SchemaFile::TableState tabState =
+ (SchemaFile::TableState)tableEntry->m_tableState;
+ ndbrequire(tabState == SchemaFile::TABLE_ADD_COMMITTED ||
+ tabState == SchemaFile::ALTER_TABLE_COMMITTED);
+ tableEntry->m_tableState = SchemaFile::DROP_TABLE_STARTED;
+ computeChecksum((SchemaFile*)pagePtr.p);
+
+ ndbrequire(c_writeSchemaRecord.inUse == false);
+ c_writeSchemaRecord.inUse = true;
+
+ c_writeSchemaRecord.pageId = c_schemaRecord.schemaPage;
+ c_writeSchemaRecord.m_callback.m_callbackData = dropTabPtr.p->key;
+ c_writeSchemaRecord.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::prepDropTab_writeSchemaConf);
+ startWriteSchemaFile(signal);
+}
+
+void
+Dbdict::prepDropTab_writeSchemaConf(Signal* signal,
+ Uint32 dropTabPtrI,
+ Uint32 returnCode){
+ jam();
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, dropTabPtrI));
+
+ ndbrequire(dropTabPtr.p->m_participantData.m_gsn == GSN_PREP_DROP_TAB_REQ);
+
+ /**
+ * There probably should be node fail handlign here
+ *
+ * To check that coordinator hasn't died
+ */
+
+ PrepDropTabConf * prep = (PrepDropTabConf*)signal->getDataPtr();
+ prep->senderRef = reference();
+ prep->senderData = dropTabPtrI;
+ prep->tableId = dropTabPtr.p->m_request.tableId;
+
+ dropTabPtr.p->m_participantData.m_gsn = GSN_PREP_DROP_TAB_CONF;
+ sendSignal(dropTabPtr.p->m_coordinatorRef, GSN_PREP_DROP_TAB_CONF, signal,
+ PrepDropTabConf::SignalLength, JBB);
+}
+
+void
+Dbdict::execDROP_TAB_REQ(Signal* signal){
+ jamEntry();
+ DropTabReq * req = (DropTabReq*)signal->getDataPtrSend();
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, req->senderData));
+
+ ndbrequire(dropTabPtr.p->m_participantData.m_gsn == GSN_PREP_DROP_TAB_CONF);
+ dropTabPtr.p->m_participantData.m_gsn = GSN_DROP_TAB_REQ;
+
+ ndbrequire(dropTabPtr.p->m_requestType == req->requestType);
+
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, dropTabPtr.p->m_request.tableId);
+ tablePtr.p->tabState = TableRecord::DROPPING;
+
+ dropTabPtr.p->m_participantData.m_block = 0;
+ dropTabPtr.p->m_participantData.m_callback.m_callbackData = dropTabPtr.p->key;
+ dropTabPtr.p->m_participantData.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::dropTab_complete);
+ dropTab_nextStep(signal, dropTabPtr);
+}
+
+#include <DebuggerNames.hpp>
+
+void
+Dbdict::dropTab_nextStep(Signal* signal, DropTableRecordPtr dropTabPtr){
+
+ /**
+ * No errors currently allowed
+ */
+ ndbrequire(dropTabPtr.p->m_errorCode == 0);
+
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, dropTabPtr.p->m_request.tableId);
+
+ Uint32 block = 0;
+ switch(dropTabPtr.p->m_participantData.m_block){
+ case 0:
+ jam();
+ block = DBTC;
+ break;
+ case DBTC:
+ jam();
+ if (tablePtr.p->isTable() || tablePtr.p->isHashIndex())
+ block = DBACC;
+ if (tablePtr.p->isOrderedIndex())
+ block = DBTUP;
+ break;
+ case DBACC:
+ jam();
+ block = DBTUP;
+ break;
+ case DBTUP:
+ jam();
+ if (tablePtr.p->isTable() || tablePtr.p->isHashIndex())
+ block = DBLQH;
+ if (tablePtr.p->isOrderedIndex())
+ block = DBTUX;
+ break;
+ case DBTUX:
+ jam();
+ block = DBLQH;
+ break;
+ case DBLQH:
+ jam();
+ block = DBDIH;
+ break;
+ case DBDIH:
+ jam();
+ execute(signal, dropTabPtr.p->m_participantData.m_callback, 0);
+ return;
+ }
+ ndbrequire(block != 0);
+ dropTabPtr.p->m_participantData.m_block = block;
+
+ DropTabReq * req = (DropTabReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = dropTabPtr.p->key;
+ req->tableId = dropTabPtr.p->m_request.tableId;
+ req->requestType = dropTabPtr.p->m_requestType;
+
+ const Uint32 nodeId = getOwnNodeId();
+ dropTabPtr.p->m_participantData.m_signalCounter.clearWaitingFor();
+ dropTabPtr.p->m_participantData.m_signalCounter.setWaitingFor(nodeId);
+ BlockReference ref = numberToRef(block, 0);
+ sendSignal(ref, GSN_DROP_TAB_REQ, signal, DropTabReq::SignalLength, JBB);
+}
+
+void
+Dbdict::dropTab_localDROP_TAB_CONF(Signal* signal){
+ jamEntry();
+
+ DropTabConf * conf = (DropTabConf*)signal->getDataPtr();
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, conf->senderData));
+
+ ndbrequire(dropTabPtr.p->m_request.tableId == conf->tableId);
+ ndbrequire(dropTabPtr.p->m_participantData.m_gsn == GSN_DROP_TAB_REQ);
+
+ Uint32 nodeId = refToNode(conf->senderRef);
+ dropTabPtr.p->m_participantData.m_signalCounter.clearWaitingFor(nodeId);
+
+ if(!dropTabPtr.p->m_participantData.m_signalCounter.done()){
+ jam();
+ ndbrequire(false);
+ return;
+ }
+ dropTab_nextStep(signal, dropTabPtr);
+}
+
+void
+Dbdict::dropTab_complete(Signal* signal,
+ Uint32 dropTabPtrI,
+ Uint32 returnCode){
+ jam();
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, dropTabPtrI));
+
+ Uint32 tableId = dropTabPtr.p->m_request.tableId;
+
+ /**
+ * Write to schema file
+ */
+ PageRecordPtr pagePtr;
+ c_pageRecordArray.getPtr(pagePtr, c_schemaRecord.schemaPage);
+
+ SchemaFile::TableEntry * tableEntry = getTableEntry(pagePtr.p, tableId);
+ SchemaFile::TableState tabState =
+ (SchemaFile::TableState)tableEntry->m_tableState;
+ ndbrequire(tabState == SchemaFile::DROP_TABLE_STARTED);
+ tableEntry->m_tableState = SchemaFile::DROP_TABLE_COMMITTED;
+ computeChecksum((SchemaFile*)pagePtr.p);
+
+ ndbrequire(c_writeSchemaRecord.inUse == false);
+ c_writeSchemaRecord.inUse = true;
+
+ c_writeSchemaRecord.pageId = c_schemaRecord.schemaPage;
+ c_writeSchemaRecord.m_callback.m_callbackData = dropTabPtr.p->key;
+ c_writeSchemaRecord.m_callback.m_callbackFunction =
+ safe_cast(&Dbdict::dropTab_writeSchemaConf);
+ startWriteSchemaFile(signal);
+}
+
+void
+Dbdict::dropTab_writeSchemaConf(Signal* signal,
+ Uint32 dropTabPtrI,
+ Uint32 returnCode){
+ jam();
+
+ DropTableRecordPtr dropTabPtr;
+ ndbrequire(c_opDropTable.find(dropTabPtr, dropTabPtrI));
+
+ ndbrequire(dropTabPtr.p->m_participantData.m_gsn == GSN_DROP_TAB_REQ);
+
+ dropTabPtr.p->m_participantData.m_gsn = GSN_DROP_TAB_CONF;
+
+ releaseTableObject(dropTabPtr.p->m_request.tableId);
+
+ DropTabConf * conf = (DropTabConf*)signal->getDataPtr();
+ conf->senderRef = reference();
+ conf->senderData = dropTabPtrI;
+ conf->tableId = dropTabPtr.p->m_request.tableId;
+
+ dropTabPtr.p->m_participantData.m_gsn = GSN_DROP_TAB_CONF;
+ sendSignal(dropTabPtr.p->m_coordinatorRef, GSN_DROP_TAB_CONF, signal,
+ DropTabConf::SignalLength, JBB);
+
+ if(dropTabPtr.p->m_coordinatorRef != reference()){
+ c_opDropTable.release(dropTabPtr);
+ }
+}
+
+void Dbdict::releaseTableObject(Uint32 tableId, bool removeFromHash)
+{
+ TableRecordPtr tablePtr;
+ AttributeRecordPtr attrPtr;
+ c_tableRecordPool.getPtr(tablePtr, tableId);
+ if (removeFromHash)
+ c_tableRecordHash.remove(tablePtr);
+
+ Uint32 nextAttrRecord = tablePtr.p->firstAttribute;
+ while (nextAttrRecord != RNIL) {
+ jam();
+/* ---------------------------------------------------------------- */
+// Release all attribute records
+/* ---------------------------------------------------------------- */
+ c_attributeRecordPool.getPtr(attrPtr, nextAttrRecord);
+ nextAttrRecord = attrPtr.p->nextAttrInTable;
+ c_attributeRecordPool.release(attrPtr);
+ }//if
+ Uint32 secondTableId = tablePtr.p->secondTable;
+ initialiseTableRecord(tablePtr);
+ c_tableRecordPool.getPtr(tablePtr, secondTableId);
+ initialiseTableRecord(tablePtr);
+ return;
+}//releaseTableObject()
+
+/**
+ * DICT receives these on index create and drop.
+ */
+void Dbdict::execDROP_TABLE_CONF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(signal->getNoOfSections() == 0);
+
+ DropTableConf * const conf = (DropTableConf *)signal->getDataPtr();
+ // assume part of drop index operation
+ OpDropIndexPtr opPtr;
+ c_opDropIndex.find(opPtr, conf->senderData);
+ ndbrequire(! opPtr.isNull());
+ ndbrequire(opPtr.p->m_request.getIndexId() == conf->tableId);
+ ndbrequire(opPtr.p->m_request.getIndexVersion() == conf->tableVersion);
+ dropIndex_fromDropTable(signal, opPtr);
+}
+
+void Dbdict::execDROP_TABLE_REF(Signal* signal)
+{
+ jamEntry();
+
+ DropTableRef * const ref = (DropTableRef *)signal->getDataPtr();
+ // assume part of drop index operation
+ OpDropIndexPtr opPtr;
+ c_opDropIndex.find(opPtr, ref->senderData);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ opPtr.p->m_errorLine = __LINE__;
+ dropIndex_fromDropTable(signal, opPtr);
+}
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: EXTERNAL INTERFACE TO DATA -------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* This module contains the code that is used by other modules to. */
+/* access the data within DBDICT. */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+void Dbdict::execGET_TABLEDID_REQ(Signal * signal)
+{
+ jamEntry();
+ ndbrequire(signal->getNoOfSections() == 1);
+ GetTableIdReq const * req = (GetTableIdReq *)signal->getDataPtr();
+ Uint32 senderData = req->senderData;
+ Uint32 senderRef = req->senderRef;
+ Uint32 len = req->len;
+
+ if(len>MAX_TAB_NAME_SIZE)
+ {
+ jam();
+ sendGET_TABLEID_REF((Signal*)signal,
+ (GetTableIdReq *)req,
+ GetTableIdRef::TableNameTooLong);
+ return;
+ }
+
+ char tableName[MAX_TAB_NAME_SIZE];
+ TableRecord keyRecord;
+ SegmentedSectionPtr ssPtr;
+ signal->getSection(ssPtr,GetTableIdReq::TABLE_NAME);
+ copy((Uint32*)tableName, ssPtr);
+ strcpy(keyRecord.tableName, tableName);
+ releaseSections(signal);
+
+ if(len > sizeof(keyRecord.tableName)){
+ jam();
+ sendGET_TABLEID_REF((Signal*)signal,
+ (GetTableIdReq *)req,
+ GetTableIdRef::TableNameTooLong);
+ return;
+ }
+
+ TableRecordPtr tablePtr;
+ if(!c_tableRecordHash.find(tablePtr, keyRecord)) {
+ jam();
+ sendGET_TABLEID_REF((Signal*)signal,
+ (GetTableIdReq *)req,
+ GetTableIdRef::TableNotDefined);
+ return;
+ }
+ GetTableIdConf * conf = (GetTableIdConf *)req;
+ conf->tableId = tablePtr.p->tableId;
+ conf->schemaVersion = tablePtr.p->tableVersion;
+ conf->senderData = senderData;
+ sendSignal(senderRef, GSN_GET_TABLEID_CONF, signal,
+ GetTableIdConf::SignalLength, JBB);
+
+}
+
+
+void Dbdict::sendGET_TABLEID_REF(Signal* signal,
+ GetTableIdReq * req,
+ GetTableIdRef::ErrorCode errorCode)
+{
+ GetTableIdRef * const ref = (GetTableIdRef *)req;
+ /**
+ * The format of GetTabInfo Req/Ref is the same
+ */
+ BlockReference retRef = req->senderRef;
+ ref->err = errorCode;
+ sendSignal(retRef, GSN_GET_TABLEID_REF, signal,
+ GetTableIdRef::SignalLength, JBB);
+}//sendGET_TABINFOREF()
+
+/* ---------------------------------------------------------------- */
+// Get a full table description.
+/* ---------------------------------------------------------------- */
+void Dbdict::execGET_TABINFOREQ(Signal* signal)
+{
+ jamEntry();
+ if(!assembleFragments(signal)) { return; }
+
+ GetTabInfoReq * const req = (GetTabInfoReq *)&signal->theData[0];
+
+ /**
+ * If I get a GET_TABINFO_REQ from myself
+ * it's is a one from the time queue
+ */
+ bool fromTimeQueue = (signal->senderBlockRef() == reference());
+
+ if (c_retrieveRecord.busyState && fromTimeQueue == true) {
+ jam();
+
+ sendSignalWithDelay(reference(), GSN_GET_TABINFOREQ, signal, 30,
+ signal->length());
+ return;
+ }//if
+
+ const Uint32 MAX_WAITERS = 5;
+
+ if(c_retrieveRecord.busyState && fromTimeQueue == false){
+ jam();
+ if(c_retrieveRecord.noOfWaiters < MAX_WAITERS){
+ jam();
+ c_retrieveRecord.noOfWaiters++;
+
+ sendSignalWithDelay(reference(), GSN_GET_TABINFOREQ, signal, 30,
+ signal->length());
+ return;
+ }
+
+ sendGET_TABINFOREF(signal, req, GetTabInfoRef::Busy);
+ return;
+ }
+
+ if(fromTimeQueue){
+ jam();
+ c_retrieveRecord.noOfWaiters--;
+ }
+
+ const bool useLongSig = (req->requestType & GetTabInfoReq::LongSignalConf);
+ const Uint32 reqType = req->requestType & (~GetTabInfoReq::LongSignalConf);
+
+ TableRecordPtr tablePtr;
+ if(reqType == GetTabInfoReq::RequestByName){
+ jam();
+ ndbrequire(signal->getNoOfSections() == 1);
+ const Uint32 len = req->tableNameLen;
+
+ TableRecord keyRecord;
+ if(len > sizeof(keyRecord.tableName)){
+ jam();
+ releaseSections(signal);
+ sendGET_TABINFOREF(signal, req, GetTabInfoRef::TableNameTooLong);
+ return;
+ }
+
+ char tableName[MAX_TAB_NAME_SIZE];
+ SegmentedSectionPtr ssPtr;
+ signal->getSection(ssPtr,GetTabInfoReq::TABLE_NAME);
+ SimplePropertiesSectionReader r0(ssPtr, getSectionSegmentPool());
+ r0.reset(); // undo implicit first()
+ if(r0.getWords((Uint32*)tableName, len))
+ memcpy(keyRecord.tableName, tableName, len);
+ else {
+ jam();
+ releaseSections(signal);
+ sendGET_TABINFOREF(signal, req, GetTabInfoRef::TableNotDefined);
+ return;
+ }
+ releaseSections(signal);
+ // memcpy(keyRecord.tableName, req->tableName, len);
+ //ntohS(&keyRecord.tableName[0], len);
+
+ c_tableRecordHash.find(tablePtr, keyRecord);
+ } else {
+ jam();
+ c_tableRecordPool.getPtr(tablePtr, req->tableId, false);
+ }
+
+ // The table seached for was not found
+ if(tablePtr.i == RNIL){
+ jam();
+ sendGET_TABINFOREF(signal, req, GetTabInfoRef::InvalidTableId);
+ return;
+ }//if
+
+ if (tablePtr.p->tabState != TableRecord::DEFINED) {
+ jam();
+ sendGET_TABINFOREF(signal, req, GetTabInfoRef::TableNotDefined);
+ return;
+ }//if
+
+ c_retrieveRecord.busyState = true;
+ c_retrieveRecord.blockRef = req->senderRef;
+ c_retrieveRecord.m_senderData = req->senderData;
+ c_retrieveRecord.tableId = tablePtr.i;
+ c_retrieveRecord.currentSent = 0;
+ c_retrieveRecord.m_useLongSig = useLongSig;
+
+ c_packTable.m_state = PackTable::PTS_GET_TAB;
+
+ signal->theData[0] = ZPACK_TABLE_INTO_PAGES;
+ signal->theData[1] = tablePtr.i;
+ signal->theData[2] = c_retrieveRecord.retrievePage;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+}//execGET_TABINFOREQ()
+
+void Dbdict::sendGetTabResponse(Signal* signal)
+{
+ PageRecordPtr pagePtr;
+ DictTabInfo * const conf = (DictTabInfo *)&signal->theData[0];
+ conf->senderRef = reference();
+ conf->senderData = c_retrieveRecord.m_senderData;
+ conf->requestType = DictTabInfo::GetTabInfoConf;
+ conf->totalLen = c_retrieveRecord.retrievedNoOfWords;
+
+ c_pageRecordArray.getPtr(pagePtr, c_retrieveRecord.retrievePage);
+ Uint32* pagePointer = (Uint32*)&pagePtr.p->word[0] + ZPAGE_HEADER_SIZE;
+
+ if(c_retrieveRecord.m_useLongSig){
+ jam();
+ GetTabInfoConf* conf = (GetTabInfoConf*)signal->getDataPtr();
+ conf->gci = 0;
+ conf->tableId = c_retrieveRecord.tableId;
+ conf->senderData = c_retrieveRecord.m_senderData;
+ conf->totalLen = c_retrieveRecord.retrievedNoOfWords;
+
+ Callback c = { safe_cast(&Dbdict::initRetrieveRecord), 0 };
+ LinearSectionPtr ptr[3];
+ ptr[0].p = pagePointer;
+ ptr[0].sz = c_retrieveRecord.retrievedNoOfWords;
+ sendFragmentedSignal(c_retrieveRecord.blockRef,
+ GSN_GET_TABINFO_CONF,
+ signal,
+ GetTabInfoConf::SignalLength,
+ JBB,
+ ptr,
+ 1,
+ c);
+ return;
+ }
+
+ ndbrequire(false);
+}//sendGetTabResponse()
+
+void Dbdict::sendGET_TABINFOREF(Signal* signal,
+ GetTabInfoReq * req,
+ GetTabInfoRef::ErrorCode errorCode)
+{
+ jamEntry();
+ GetTabInfoRef * const ref = (GetTabInfoRef *)&signal->theData[0];
+ /**
+ * The format of GetTabInfo Req/Ref is the same
+ */
+ BlockReference retRef = req->senderRef;
+ ref->errorCode = errorCode;
+
+ sendSignal(retRef, GSN_GET_TABINFOREF, signal, signal->length(), JBB);
+}//sendGET_TABINFOREF()
+
+Uint32 convertEndian(Uint32 in) {
+#ifdef _BIG_ENDIAN
+ Uint32 ut = 0;
+ ut += ((in >> 24) & 255);
+ ut += (((in >> 16) & 255) << 8);
+ ut += (((in >> 8) & 255) << 16);
+ ut += ((in & 255) << 24);
+ return ut;
+#else
+ return in;
+#endif
+}
+void
+Dbdict::execLIST_TABLES_REQ(Signal* signal)
+{
+ jamEntry();
+ ListTablesReq * req = (ListTablesReq*)signal->getDataPtr();
+ Uint32 senderRef = req->senderRef;
+ Uint32 senderData = req->senderData;
+ // save req flags
+ const Uint32 reqTableId = req->getTableId();
+ const Uint32 reqTableType = req->getTableType();
+ const bool reqListNames = req->getListNames();
+ const bool reqListIndexes = req->getListIndexes();
+ // init the confs
+ ListTablesConf * conf = (ListTablesConf *)signal->getDataPtrSend();
+ conf->senderData = senderData;
+ conf->counter = 0;
+ Uint32 pos = 0;
+ for (Uint32 i = 0; i < c_tableRecordPool.getSize(); i++) {
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, i);
+ // filter
+ if (tablePtr.p->tabState == TableRecord::NOT_DEFINED ||
+ tablePtr.p->tabState == TableRecord::REORG_TABLE_PREPARED)
+ continue;
+
+
+ if ((reqTableType != (Uint32)0) && (reqTableType != (unsigned)tablePtr.p->tableType))
+ continue;
+ if (reqListIndexes && reqTableId != tablePtr.p->primaryTableId)
+ continue;
+ conf->tableData[pos] = 0;
+ // id
+ conf->setTableId(pos, tablePtr.i);
+ // type
+ conf->setTableType(pos, tablePtr.p->tableType);
+ // state
+ if (tablePtr.p->isTable()) {
+ switch (tablePtr.p->tabState) {
+ case TableRecord::DEFINING:
+ case TableRecord::CHECKED:
+ conf->setTableState(pos, DictTabInfo::StateBuilding);
+ break;
+ case TableRecord::PREPARE_DROPPING:
+ case TableRecord::DROPPING:
+ conf->setTableState(pos, DictTabInfo::StateDropping);
+ break;
+ case TableRecord::DEFINED:
+ conf->setTableState(pos, DictTabInfo::StateOnline);
+ break;
+ default:
+ conf->setTableState(pos, DictTabInfo::StateBroken);
+ break;
+ }
+ }
+ if (tablePtr.p->isIndex()) {
+ switch (tablePtr.p->indexState) {
+ case TableRecord::IS_OFFLINE:
+ conf->setTableState(pos, DictTabInfo::StateOffline);
+ break;
+ case TableRecord::IS_BUILDING:
+ conf->setTableState(pos, DictTabInfo::StateBuilding);
+ break;
+ case TableRecord::IS_DROPPING:
+ conf->setTableState(pos, DictTabInfo::StateDropping);
+ break;
+ case TableRecord::IS_ONLINE:
+ conf->setTableState(pos, DictTabInfo::StateOnline);
+ break;
+ default:
+ conf->setTableState(pos, DictTabInfo::StateBroken);
+ break;
+ }
+ }
+ // store
+ if (! tablePtr.p->storedTable) {
+ conf->setTableStore(pos, DictTabInfo::StoreTemporary);
+ } else {
+ conf->setTableStore(pos, DictTabInfo::StorePermanent);
+ }
+ pos++;
+ if (pos >= ListTablesConf::DataLength) {
+ sendSignal(senderRef, GSN_LIST_TABLES_CONF, signal,
+ ListTablesConf::SignalLength, JBB);
+ conf->counter++;
+ pos = 0;
+ }
+ if (! reqListNames)
+ continue;
+ const Uint32 size = strlen(tablePtr.p->tableName) + 1;
+ conf->tableData[pos] = size;
+ pos++;
+ if (pos >= ListTablesConf::DataLength) {
+ sendSignal(senderRef, GSN_LIST_TABLES_CONF, signal,
+ ListTablesConf::SignalLength, JBB);
+ conf->counter++;
+ pos = 0;
+ }
+ Uint32 i = 0;
+ while (i < size) {
+ char* p = (char*)&conf->tableData[pos];
+ for (Uint32 j = 0; j < 4; j++) {
+ if (i < size)
+ *p++ = tablePtr.p->tableName[i++];
+ else
+ *p++ = 0;
+ }
+ pos++;
+ if (pos >= ListTablesConf::DataLength) {
+ sendSignal(senderRef, GSN_LIST_TABLES_CONF, signal,
+ ListTablesConf::SignalLength, JBB);
+ conf->counter++;
+ pos = 0;
+ }
+ }
+ }
+ // XXX merge with above somehow
+ for (Uint32 i = 0; i < c_triggerRecordPool.getSize(); i++) {
+ if (reqListIndexes)
+ break;
+ TriggerRecordPtr triggerPtr;
+ c_triggerRecordPool.getPtr(triggerPtr, i);
+ if (triggerPtr.p->triggerState == TriggerRecord::TS_NOT_DEFINED)
+ continue;
+ // constant 10 hardcoded
+ Uint32 type = 10 + triggerPtr.p->triggerType;
+ if (reqTableType != 0 && reqTableType != type)
+ continue;
+ conf->tableData[pos] = 0;
+ conf->setTableId(pos, triggerPtr.i);
+ conf->setTableType(pos, type);
+ switch (triggerPtr.p->triggerState) {
+ case TriggerRecord::TS_OFFLINE:
+ conf->setTableState(pos, DictTabInfo::StateOffline);
+ break;
+ case TriggerRecord::TS_ONLINE:
+ conf->setTableState(pos, DictTabInfo::StateOnline);
+ break;
+ default:
+ conf->setTableState(pos, DictTabInfo::StateBroken);
+ break;
+ }
+ conf->setTableStore(pos, DictTabInfo::StoreTemporary);
+ pos++;
+ if (pos >= ListTablesConf::DataLength) {
+ sendSignal(senderRef, GSN_LIST_TABLES_CONF, signal,
+ ListTablesConf::SignalLength, JBB);
+ conf->counter++;
+ pos = 0;
+ }
+ if (! reqListNames)
+ continue;
+ const Uint32 size = strlen(triggerPtr.p->triggerName) + 1;
+ conf->tableData[pos] = size;
+ pos++;
+ if (pos >= ListTablesConf::DataLength) {
+ sendSignal(senderRef, GSN_LIST_TABLES_CONF, signal,
+ ListTablesConf::SignalLength, JBB);
+ conf->counter++;
+ pos = 0;
+ }
+ Uint32 i = 0;
+ while (i < size) {
+ char* p = (char*)&conf->tableData[pos];
+ for (Uint32 j = 0; j < 4; j++) {
+ if (i < size)
+ *p++ = triggerPtr.p->triggerName[i++];
+ else
+ *p++ = 0;
+ }
+ pos++;
+ if (pos >= ListTablesConf::DataLength) {
+ sendSignal(senderRef, GSN_LIST_TABLES_CONF, signal,
+ ListTablesConf::SignalLength, JBB);
+ conf->counter++;
+ pos = 0;
+ }
+ }
+ }
+ // last signal must have less than max length
+ sendSignal(senderRef, GSN_LIST_TABLES_CONF, signal,
+ ListTablesConf::HeaderLength + pos, JBB);
+}
+
+/**
+ * MODULE: Create index
+ *
+ * Create index in DICT via create table operation. Then invoke alter
+ * index opearation to online the index.
+ *
+ * Request type in CREATE_INDX signals:
+ *
+ * RT_USER - from API to DICT master
+ * RT_DICT_PREPARE - prepare participants
+ * RT_DICT_COMMIT - commit participants
+ * RT_TC - create index in TC (part of alter index operation)
+ */
+
+void
+Dbdict::execCREATE_INDX_REQ(Signal* signal)
+{
+ jamEntry();
+ CreateIndxReq* const req = (CreateIndxReq*)signal->getDataPtrSend();
+ OpCreateIndexPtr opPtr;
+ const Uint32 senderRef = signal->senderBlockRef();
+ const CreateIndxReq::RequestType requestType = req->getRequestType();
+ if (requestType == CreateIndxReq::RT_USER) {
+ jam();
+ if (! assembleFragments(signal)) {
+ jam();
+ return;
+ }
+ if (signal->getLength() == CreateIndxReq::SignalLength) {
+ jam();
+ if (getOwnNodeId() != c_masterNodeId) {
+ jam();
+ // forward to DICT master
+ sendSignal(calcDictBlockRef(c_masterNodeId), GSN_CREATE_INDX_REQ,
+ signal, signal->getLength(), JBB);
+ return;
+ }
+ // forward initial request plus operation key to all
+ req->setOpKey(++c_opRecordSequence);
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ sendSignal(rg, GSN_CREATE_INDX_REQ,
+ signal, CreateIndxReq::SignalLength + 1, JBB);
+ return;
+ }
+ // seize operation record
+ ndbrequire(signal->getLength() == CreateIndxReq::SignalLength + 1);
+ const Uint32 opKey = req->getOpKey();
+ OpCreateIndex opBusy;
+ if (! c_opCreateIndex.seize(opPtr))
+ opPtr.p = &opBusy;
+ opPtr.p->save(req);
+ opPtr.p->m_coordinatorRef = senderRef;
+ opPtr.p->m_isMaster = (senderRef == reference());
+ opPtr.p->key = opKey;
+ opPtr.p->m_requestType = CreateIndxReq::RT_DICT_PREPARE;
+ if (opPtr.p == &opBusy) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::Busy;
+ opPtr.p->m_errorLine = __LINE__;
+ releaseSections(signal);
+ createIndex_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ c_opCreateIndex.add(opPtr);
+ // save attribute list
+ SegmentedSectionPtr ssPtr;
+ signal->getSection(ssPtr, CreateIndxReq::ATTRIBUTE_LIST_SECTION);
+ SimplePropertiesSectionReader r0(ssPtr, getSectionSegmentPool());
+ r0.reset(); // undo implicit first()
+ if (! r0.getWord(&opPtr.p->m_attrList.sz) ||
+ ! r0.getWords(opPtr.p->m_attrList.id, opPtr.p->m_attrList.sz)) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::InvalidName;
+ opPtr.p->m_errorLine = __LINE__;
+ releaseSections(signal);
+ createIndex_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ // save name and index table properties
+ signal->getSection(ssPtr, CreateIndxReq::INDEX_NAME_SECTION);
+ SimplePropertiesSectionReader r1(ssPtr, getSectionSegmentPool());
+ DictTabInfo::Table tableDesc;
+ tableDesc.init();
+ SimpleProperties::UnpackStatus status = SimpleProperties::unpack(
+ r1, &tableDesc,
+ DictTabInfo::TableMapping, DictTabInfo::TableMappingSize,
+ true, true);
+ if (status != SimpleProperties::Eof) {
+ opPtr.p->m_errorCode = CreateIndxRef::InvalidName;
+ opPtr.p->m_errorLine = __LINE__;
+ releaseSections(signal);
+ createIndex_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ memcpy(opPtr.p->m_indexName, tableDesc.TableName, MAX_TAB_NAME_SIZE);
+ opPtr.p->m_storedIndex = tableDesc.TableLoggedFlag;
+ releaseSections(signal);
+ // master expects to hear from all
+ if (opPtr.p->m_isMaster)
+ opPtr.p->m_signalCounter = c_aliveNodes;
+ createIndex_slavePrepare(signal, opPtr);
+ createIndex_sendReply(signal, opPtr, false);
+ return;
+ }
+ c_opCreateIndex.find(opPtr, req->getConnectionPtr());
+ if (! opPtr.isNull()) {
+ opPtr.p->m_requestType = requestType;
+ if (requestType == CreateIndxReq::RT_DICT_COMMIT ||
+ requestType == CreateIndxReq::RT_DICT_ABORT) {
+ jam();
+ if (requestType == CreateIndxReq::RT_DICT_COMMIT) {
+ opPtr.p->m_request.setIndexId(req->getIndexId());
+ opPtr.p->m_request.setIndexVersion(req->getIndexVersion());
+ createIndex_slaveCommit(signal, opPtr);
+ } else {
+ createIndex_slaveAbort(signal, opPtr);
+ }
+ createIndex_sendReply(signal, opPtr, false);
+ // done in slave
+ if (! opPtr.p->m_isMaster)
+ c_opCreateIndex.release(opPtr);
+ return;
+ }
+ }
+ jam();
+ // return to sender
+ releaseSections(signal);
+ OpCreateIndex opBad;
+ opPtr.p = &opBad;
+ opPtr.p->save(req);
+ opPtr.p->m_errorCode = CreateIndxRef::BadRequestType;
+ opPtr.p->m_errorLine = __LINE__;
+ createIndex_sendReply(signal, opPtr, true);
+}
+
+void
+Dbdict::execCREATE_INDX_CONF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(signal->getNoOfSections() == 0);
+ CreateIndxConf* conf = (CreateIndxConf*)signal->getDataPtrSend();
+ createIndex_recvReply(signal, conf, 0);
+}
+
+void
+Dbdict::execCREATE_INDX_REF(Signal* signal)
+{
+ jamEntry();
+ CreateIndxRef* ref = (CreateIndxRef*)signal->getDataPtrSend();
+ createIndex_recvReply(signal, ref->getConf(), ref);
+}
+
+void
+Dbdict::createIndex_recvReply(Signal* signal, const CreateIndxConf* conf,
+ const CreateIndxRef* ref)
+{
+ jam();
+ const Uint32 senderRef = signal->senderBlockRef();
+ const CreateIndxReq::RequestType requestType = conf->getRequestType();
+ const Uint32 key = conf->getConnectionPtr();
+ if (requestType == CreateIndxReq::RT_TC) {
+ jam();
+ // part of alter index operation
+ OpAlterIndexPtr opPtr;
+ c_opAlterIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ alterIndex_fromCreateTc(signal, opPtr);
+ return;
+ }
+ OpCreateIndexPtr opPtr;
+ c_opCreateIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ ndbrequire(opPtr.p->m_isMaster);
+ ndbrequire(opPtr.p->m_requestType == requestType);
+ opPtr.p->setError(ref);
+ opPtr.p->m_signalCounter.clearWaitingFor(refToNode(senderRef));
+ if (! opPtr.p->m_signalCounter.done()) {
+ jam();
+ return;
+ }
+ if (requestType == CreateIndxReq::RT_DICT_COMMIT ||
+ requestType == CreateIndxReq::RT_DICT_ABORT) {
+ jam();
+ // send reply to user
+ createIndex_sendReply(signal, opPtr, true);
+ c_opCreateIndex.release(opPtr);
+ return;
+ }
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = CreateIndxReq::RT_DICT_ABORT;
+ createIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (requestType == CreateIndxReq::RT_DICT_PREPARE) {
+ jam();
+ // start index table create
+ createIndex_toCreateTable(signal, opPtr);
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = CreateIndxReq::RT_DICT_ABORT;
+ createIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::createIndex_slavePrepare(Signal* signal, OpCreateIndexPtr opPtr)
+{
+ jam();
+}
+
+void
+Dbdict::createIndex_toCreateTable(Signal* signal, OpCreateIndexPtr opPtr)
+{
+ jam();
+ const CreateIndxReq* const req = &opPtr.p->m_request;
+ // signal data writer
+ Uint32* wbuffer = &c_indexPage.word[0];
+ LinearWriter w(wbuffer, sizeof(c_indexPage) >> 2);
+ w.first();
+ // get table being indexed
+ if (! (req->getTableId() < c_tableRecordPool.getSize())) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::InvalidPrimaryTable;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, req->getTableId());
+ if (tablePtr.p->tabState != TableRecord::DEFINED) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::InvalidPrimaryTable;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ if (! tablePtr.p->isTable()) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::InvalidPrimaryTable;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ // compute index table record
+ TableRecord indexRec;
+ TableRecordPtr indexPtr;
+ indexPtr.i = RNIL; // invalid
+ indexPtr.p = &indexRec;
+ initialiseTableRecord(indexPtr);
+ if (req->getIndexType() == DictTabInfo::UniqueHashIndex) {
+ indexPtr.p->storedTable = opPtr.p->m_storedIndex;
+ indexPtr.p->fragmentType = tablePtr.p->fragmentType;
+ } else if (req->getIndexType() == DictTabInfo::OrderedIndex) {
+ // first version will not supported logging
+ if (opPtr.p->m_storedIndex) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::InvalidIndexType;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ indexPtr.p->storedTable = false;
+ // follows table fragmentation
+ indexPtr.p->fragmentType = tablePtr.p->fragmentType;
+ } else {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::InvalidIndexType;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ indexPtr.p->tableType = (DictTabInfo::TableType)req->getIndexType();
+ indexPtr.p->primaryTableId = req->getTableId();
+ indexPtr.p->noOfAttributes = opPtr.p->m_attrList.sz;
+ indexPtr.p->tupKeyLength = 0;
+ if (indexPtr.p->noOfAttributes == 0) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::InvalidIndexType;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ if (indexPtr.p->isOrderedIndex()) {
+ // tree node size in words (make configurable later)
+ indexPtr.p->tupKeyLength = MAX_TTREE_NODE_SIZE;
+ }
+ // hash index attributes must currently be in table order
+ Uint32 prevAttrId = RNIL;
+ for (Uint32 k = 0; k < opPtr.p->m_attrList.sz; k++) {
+ jam();
+ bool found = false;
+ for (Uint32 tAttr = tablePtr.p->firstAttribute; tAttr != RNIL; ) {
+ AttributeRecord* aRec = c_attributeRecordPool.getPtr(tAttr);
+ tAttr = aRec->nextAttrInTable;
+ if (aRec->attributeId != opPtr.p->m_attrList.id[k])
+ continue;
+ jam();
+ found = true;
+ const Uint32 a = aRec->attributeDescriptor;
+ bool isNullable = AttributeDescriptor::getNullable(a);
+ // We do not allow more than one NULLable attribute for hash index
+ if (isNullable &&
+ indexPtr.p->isHashIndex() &&
+ (opPtr.p->m_attrList.sz > 1)) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::AttributeNullable;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ if (indexPtr.p->isHashIndex()) {
+ const Uint32 s1 = AttributeDescriptor::getSize(a);
+ const Uint32 s2 = AttributeDescriptor::getArraySize(a);
+ indexPtr.p->tupKeyLength += ((1 << s1) * s2 + 31) >> 5;
+ }
+ }
+ if (! found) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::BadRequestType;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ if (indexPtr.p->isHashIndex() &&
+ k > 0 && prevAttrId >= opPtr.p->m_attrList.id[k]) {
+ jam();
+ opPtr.p->m_errorCode = CreateIndxRef::InvalidAttributeOrder;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ prevAttrId = opPtr.p->m_attrList.id[k];
+ }
+ indexPtr.p->noOfPrimkey = indexPtr.p->noOfAttributes;
+ // plus concatenated primary table key attribute
+ indexPtr.p->noOfAttributes += 1;
+ indexPtr.p->noOfNullAttr = 0;
+ // write index table
+ w.add(DictTabInfo::TableName, opPtr.p->m_indexName);
+ w.add(DictTabInfo::TableLoggedFlag, indexPtr.p->storedTable);
+ w.add(DictTabInfo::FragmentTypeVal, indexPtr.p->fragmentType);
+ w.add(DictTabInfo::TableTypeVal, indexPtr.p->tableType);
+ w.add(DictTabInfo::PrimaryTable, tablePtr.p->tableName);
+ w.add(DictTabInfo::PrimaryTableId, tablePtr.i);
+ w.add(DictTabInfo::NoOfAttributes, indexPtr.p->noOfAttributes);
+ w.add(DictTabInfo::NoOfKeyAttr, indexPtr.p->noOfPrimkey);
+ w.add(DictTabInfo::NoOfNullable, indexPtr.p->noOfNullAttr);
+ w.add(DictTabInfo::KeyLength, indexPtr.p->tupKeyLength);
+ // write index key attributes
+ AttributeRecordPtr aRecPtr;
+ c_attributeRecordPool.getPtr(aRecPtr, tablePtr.p->firstAttribute);
+ for (Uint32 k = 0; k < opPtr.p->m_attrList.sz; k++) {
+ jam();
+ for (Uint32 tAttr = tablePtr.p->firstAttribute; tAttr != RNIL; ) {
+ AttributeRecord* aRec = c_attributeRecordPool.getPtr(tAttr);
+ tAttr = aRec->nextAttrInTable;
+ if (aRec->attributeId != opPtr.p->m_attrList.id[k])
+ continue;
+ jam();
+ const Uint32 a = aRec->attributeDescriptor;
+ bool isNullable = AttributeDescriptor::getNullable(a);
+ w.add(DictTabInfo::AttributeName, aRec->attributeName);
+ w.add(DictTabInfo::AttributeId, k);
+ if (indexPtr.p->isHashIndex()) {
+ w.add(DictTabInfo::AttributeKeyFlag, (Uint32)true);
+ w.add(DictTabInfo::AttributeNullableFlag, (Uint32)false);
+ }
+ if (indexPtr.p->isOrderedIndex()) {
+ w.add(DictTabInfo::AttributeKeyFlag, (Uint32)false);
+ w.add(DictTabInfo::AttributeNullableFlag, (Uint32)isNullable);
+ }
+ w.add(DictTabInfo::AttributeStoredInd, (Uint32)DictTabInfo::Stored);
+ // ext type overrides
+ w.add(DictTabInfo::AttributeExtType, aRec->extType);
+ w.add(DictTabInfo::AttributeExtLength, aRec->extLength);
+ w.add(DictTabInfo::AttributeEnd, (Uint32)true);
+ }
+ }
+ if (indexPtr.p->isHashIndex()) {
+ jam();
+ // write concatenated primary table key attribute
+ w.add(DictTabInfo::AttributeName, "NDB$PK");
+ w.add(DictTabInfo::AttributeId, opPtr.p->m_attrList.sz);
+ w.add(DictTabInfo::AttributeKeyFlag, (Uint32)false);
+ w.add(DictTabInfo::AttributeStoredInd, (Uint32)DictTabInfo::Stored);
+ w.add(DictTabInfo::AttributeNullableFlag, (Uint32)false);
+ // ext type overrides
+ w.add(DictTabInfo::AttributeExtType, (Uint32)DictTabInfo::ExtUnsigned);
+ w.add(DictTabInfo::AttributeExtLength, tablePtr.p->tupKeyLength);
+ w.add(DictTabInfo::AttributeEnd, (Uint32)true);
+ }
+ if (indexPtr.p->isOrderedIndex()) {
+ jam();
+ // write index tree node as Uint32 array attribute
+ w.add(DictTabInfo::AttributeName, "NDB$TNODE");
+ w.add(DictTabInfo::AttributeId, opPtr.p->m_attrList.sz);
+ w.add(DictTabInfo::AttributeKeyFlag, (Uint32)true);
+ w.add(DictTabInfo::AttributeStoredInd, (Uint32)DictTabInfo::Stored);
+ w.add(DictTabInfo::AttributeNullableFlag, (Uint32)false);
+ // ext type overrides
+ w.add(DictTabInfo::AttributeExtType, (Uint32)DictTabInfo::ExtUnsigned);
+ w.add(DictTabInfo::AttributeExtLength, indexPtr.p->tupKeyLength);
+ w.add(DictTabInfo::AttributeEnd, (Uint32)true);
+ }
+ // finish
+ w.add(DictTabInfo::TableEnd, (Uint32)true);
+ // remember to...
+ releaseSections(signal);
+ // send create index table request
+ CreateTableReq * const cre = (CreateTableReq*)signal->getDataPtrSend();
+ cre->senderRef = reference();
+ cre->senderData = opPtr.p->key;
+ LinearSectionPtr lsPtr[3];
+ lsPtr[0].p = wbuffer;
+ lsPtr[0].sz = w.getWordsUsed();
+ sendSignal(DBDICT_REF, GSN_CREATE_TABLE_REQ,
+ signal, CreateTableReq::SignalLength, JBB, lsPtr, 1);
+}
+
+void
+Dbdict::createIndex_fromCreateTable(Signal* signal, OpCreateIndexPtr opPtr)
+{
+ jam();
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = CreateIndxReq::RT_DICT_ABORT;
+ createIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (! opPtr.p->m_request.getOnline()) {
+ jam();
+ opPtr.p->m_requestType = CreateIndxReq::RT_DICT_COMMIT;
+ createIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ createIndex_toAlterIndex(signal, opPtr);
+}
+
+void
+Dbdict::createIndex_toAlterIndex(Signal* signal, OpCreateIndexPtr opPtr)
+{
+ jam();
+ AlterIndxReq* const req = (AlterIndxReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(AlterIndxReq::RT_CREATE_INDEX);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ req->setTableId(opPtr.p->m_request.getTableId());
+ req->setIndexId(opPtr.p->m_request.getIndexId());
+ req->setIndexVersion(opPtr.p->m_request.getIndexVersion());
+ req->setOnline(true);
+ sendSignal(reference(), GSN_ALTER_INDX_REQ,
+ signal, AlterIndxReq::SignalLength, JBB);
+}
+
+void
+Dbdict::createIndex_fromAlterIndex(Signal* signal, OpCreateIndexPtr opPtr)
+{
+ jam();
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = CreateIndxReq::RT_DICT_ABORT;
+ createIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ opPtr.p->m_requestType = CreateIndxReq::RT_DICT_COMMIT;
+ createIndex_sendSlaveReq(signal, opPtr);
+}
+
+void
+Dbdict::createIndex_slaveCommit(Signal* signal, OpCreateIndexPtr opPtr)
+{
+ jam();
+ const Uint32 indexId = opPtr.p->m_request.getIndexId();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, indexId);
+ if (! opPtr.p->m_request.getOnline()) {
+ ndbrequire(indexPtr.p->indexState == TableRecord::IS_UNDEFINED);
+ indexPtr.p->indexState = TableRecord::IS_OFFLINE;
+ } else {
+ ndbrequire(indexPtr.p->indexState == TableRecord::IS_ONLINE);
+ }
+}
+
+void
+Dbdict::createIndex_slaveAbort(Signal* signal, OpCreateIndexPtr opPtr)
+{
+ jam();
+ CreateIndxReq* const req = &opPtr.p->m_request;
+ const Uint32 indexId = req->getIndexId();
+ if (indexId >= c_tableRecordPool.getSize()) {
+ jam();
+ return;
+ }
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, indexId);
+ if (! indexPtr.p->isIndex()) {
+ jam();
+ return;
+ }
+ indexPtr.p->indexState = TableRecord::IS_BROKEN;
+}
+
+void
+Dbdict::createIndex_sendSlaveReq(Signal* signal, OpCreateIndexPtr opPtr)
+{
+ jam();
+ CreateIndxReq* const req = (CreateIndxReq*)signal->getDataPtrSend();
+ *req = opPtr.p->m_request;
+ req->setUserRef(opPtr.p->m_coordinatorRef);
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(opPtr.p->m_requestType);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ opPtr.p->m_signalCounter = c_aliveNodes;
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ sendSignal(rg, GSN_CREATE_INDX_REQ,
+ signal, CreateIndxReq::SignalLength, JBB);
+}
+
+void
+Dbdict::createIndex_sendReply(Signal* signal, OpCreateIndexPtr opPtr,
+ bool toUser)
+{
+ CreateIndxRef* rep = (CreateIndxRef*)signal->getDataPtrSend();
+ Uint32 gsn = GSN_CREATE_INDX_CONF;
+ Uint32 length = CreateIndxConf::InternalLength;
+ bool sendRef = opPtr.p->hasError();
+ if (! toUser) {
+ rep->setUserRef(opPtr.p->m_coordinatorRef);
+ rep->setConnectionPtr(opPtr.p->key);
+ rep->setRequestType(opPtr.p->m_requestType);
+ if (opPtr.p->m_requestType == CreateIndxReq::RT_DICT_ABORT)
+ sendRef = false;
+ } else {
+ rep->setUserRef(opPtr.p->m_request.getUserRef());
+ rep->setConnectionPtr(opPtr.p->m_request.getConnectionPtr());
+ rep->setRequestType(opPtr.p->m_request.getRequestType());
+ length = CreateIndxConf::SignalLength;
+ }
+ rep->setTableId(opPtr.p->m_request.getTableId());
+ rep->setIndexId(opPtr.p->m_request.getIndexId());
+ rep->setIndexVersion(opPtr.p->m_request.getIndexVersion());
+ if (sendRef) {
+ if (opPtr.p->m_errorNode == 0)
+ opPtr.p->m_errorNode = getOwnNodeId();
+ rep->setErrorCode(opPtr.p->m_errorCode);
+ rep->setErrorLine(opPtr.p->m_errorLine);
+ rep->setErrorNode(opPtr.p->m_errorNode);
+ gsn = GSN_CREATE_INDX_REF;
+ length = CreateIndxRef::SignalLength;
+ }
+ sendSignal(rep->getUserRef(), gsn, signal, length, JBB);
+}
+
+/**
+ * MODULE: Drop index.
+ *
+ * Drop index. First alters the index offline (i.e. drops metadata in
+ * other blocks) and then drops the index table.
+ */
+
+void
+Dbdict::execDROP_INDX_REQ(Signal* signal)
+{
+ jamEntry();
+ DropIndxReq* const req = (DropIndxReq*)signal->getDataPtrSend();
+ OpDropIndexPtr opPtr;
+ const Uint32 senderRef = signal->senderBlockRef();
+ const DropIndxReq::RequestType requestType = req->getRequestType();
+ if (requestType == DropIndxReq::RT_USER) {
+ jam();
+ if (signal->getLength() == DropIndxReq::SignalLength) {
+ jam();
+ if (getOwnNodeId() != c_masterNodeId) {
+ jam();
+ // forward to DICT master
+ sendSignal(calcDictBlockRef(c_masterNodeId), GSN_DROP_INDX_REQ,
+ signal, signal->getLength(), JBB);
+ return;
+ }
+ // forward initial request plus operation key to all
+ req->setOpKey(++c_opRecordSequence);
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ sendSignal(rg, GSN_DROP_INDX_REQ,
+ signal, DropIndxReq::SignalLength + 1, JBB);
+ return;
+ }
+ // seize operation record
+ ndbrequire(signal->getLength() == DropIndxReq::SignalLength + 1);
+ const Uint32 opKey = req->getOpKey();
+ OpDropIndex opBusy;
+ if (! c_opDropIndex.seize(opPtr))
+ opPtr.p = &opBusy;
+ opPtr.p->save(req);
+ opPtr.p->m_coordinatorRef = senderRef;
+ opPtr.p->m_isMaster = (senderRef == reference());
+ opPtr.p->key = opKey;
+ opPtr.p->m_requestType = DropIndxReq::RT_DICT_PREPARE;
+ if (opPtr.p == &opBusy) {
+ jam();
+ opPtr.p->m_errorCode = DropIndxRef::Busy;
+ opPtr.p->m_errorLine = __LINE__;
+ dropIndex_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ c_opDropIndex.add(opPtr);
+ // master expects to hear from all
+ if (opPtr.p->m_isMaster)
+ opPtr.p->m_signalCounter = c_aliveNodes;
+ dropIndex_slavePrepare(signal, opPtr);
+ dropIndex_sendReply(signal, opPtr, false);
+ return;
+ }
+ c_opDropIndex.find(opPtr, req->getConnectionPtr());
+ if (! opPtr.isNull()) {
+ opPtr.p->m_requestType = requestType;
+ if (requestType == DropIndxReq::RT_DICT_COMMIT ||
+ requestType == DropIndxReq::RT_DICT_ABORT) {
+ jam();
+ if (requestType == DropIndxReq::RT_DICT_COMMIT)
+ dropIndex_slaveCommit(signal, opPtr);
+ else
+ dropIndex_slaveAbort(signal, opPtr);
+ dropIndex_sendReply(signal, opPtr, false);
+ // done in slave
+ if (! opPtr.p->m_isMaster)
+ c_opDropIndex.release(opPtr);
+ return;
+ }
+ }
+ jam();
+ // return to sender
+ OpDropIndex opBad;
+ opPtr.p = &opBad;
+ opPtr.p->save(req);
+ opPtr.p->m_errorCode = DropIndxRef::BadRequestType;
+ opPtr.p->m_errorLine = __LINE__;
+ dropIndex_sendReply(signal, opPtr, true);
+}
+
+void
+Dbdict::execDROP_INDX_CONF(Signal* signal)
+{
+ jamEntry();
+ DropIndxConf* conf = (DropIndxConf*)signal->getDataPtrSend();
+ dropIndex_recvReply(signal, conf, 0);
+}
+
+void
+Dbdict::execDROP_INDX_REF(Signal* signal)
+{
+ jamEntry();
+ DropIndxRef* ref = (DropIndxRef*)signal->getDataPtrSend();
+ dropIndex_recvReply(signal, ref->getConf(), ref);
+}
+
+void
+Dbdict::dropIndex_recvReply(Signal* signal, const DropIndxConf* conf,
+ const DropIndxRef* ref)
+{
+ jam();
+ const Uint32 senderRef = signal->senderBlockRef();
+ const DropIndxReq::RequestType requestType = conf->getRequestType();
+ const Uint32 key = conf->getConnectionPtr();
+ if (requestType == DropIndxReq::RT_TC) {
+ jam();
+ // part of alter index operation
+ OpAlterIndexPtr opPtr;
+ c_opAlterIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ alterIndex_fromDropTc(signal, opPtr);
+ return;
+ }
+ OpDropIndexPtr opPtr;
+ c_opDropIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ ndbrequire(opPtr.p->m_isMaster);
+ ndbrequire(opPtr.p->m_requestType == requestType);
+ opPtr.p->setError(ref);
+ opPtr.p->m_signalCounter.clearWaitingFor(refToNode(senderRef));
+ if (! opPtr.p->m_signalCounter.done()) {
+ jam();
+ return;
+ }
+ if (requestType == DropIndxReq::RT_DICT_COMMIT ||
+ requestType == DropIndxReq::RT_DICT_ABORT) {
+ jam();
+ // send reply to user
+ dropIndex_sendReply(signal, opPtr, true);
+ c_opDropIndex.release(opPtr);
+ return;
+ }
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = DropIndxReq::RT_DICT_ABORT;
+ dropIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (requestType == DropIndxReq::RT_DICT_PREPARE) {
+ jam();
+ // start alter offline
+ dropIndex_toAlterIndex(signal, opPtr);
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::dropIndex_slavePrepare(Signal* signal, OpDropIndexPtr opPtr)
+{
+ jam();
+ DropIndxReq* const req = &opPtr.p->m_request;
+ // check index exists
+ TableRecordPtr indexPtr;
+ if (! (req->getIndexId() < c_tableRecordPool.getSize())) {
+ jam();
+ opPtr.p->m_errorCode = DropIndxRef::IndexNotFound;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ c_tableRecordPool.getPtr(indexPtr, req->getIndexId());
+ if (indexPtr.p->tabState != TableRecord::DEFINED) {
+ jam();
+ opPtr.p->m_errorCode = DropIndxRef::IndexNotFound;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ if (! indexPtr.p->isIndex()) {
+ jam();
+ opPtr.p->m_errorCode = DropIndxRef::NotAnIndex;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ // ignore incoming primary table id
+ req->setTableId(indexPtr.p->primaryTableId);
+}
+
+void
+Dbdict::dropIndex_toAlterIndex(Signal* signal, OpDropIndexPtr opPtr)
+{
+ jam();
+ AlterIndxReq* const req = (AlterIndxReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(AlterIndxReq::RT_DROP_INDEX);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ req->setTableId(opPtr.p->m_request.getTableId());
+ req->setIndexId(opPtr.p->m_request.getIndexId());
+ req->setIndexVersion(opPtr.p->m_request.getIndexVersion());
+ req->setOnline(false);
+ sendSignal(reference(), GSN_ALTER_INDX_REQ,
+ signal, AlterIndxReq::SignalLength, JBB);
+}
+
+void
+Dbdict::dropIndex_fromAlterIndex(Signal* signal, OpDropIndexPtr opPtr)
+{
+ jam();
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = DropIndxReq::RT_DICT_ABORT;
+ dropIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ dropIndex_toDropTable(signal, opPtr);
+}
+
+void
+Dbdict::dropIndex_toDropTable(Signal* signal, OpDropIndexPtr opPtr)
+{
+ jam();
+ DropTableReq* const req = (DropTableReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = opPtr.p->key;
+ req->tableId = opPtr.p->m_request.getIndexId();
+ req->tableVersion = opPtr.p->m_request.getIndexVersion();
+ sendSignal(reference(), GSN_DROP_TABLE_REQ,
+ signal,DropTableReq::SignalLength, JBB);
+}
+
+void
+Dbdict::dropIndex_fromDropTable(Signal* signal, OpDropIndexPtr opPtr)
+{
+ jam();
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = DropIndxReq::RT_DICT_ABORT;
+ dropIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ opPtr.p->m_requestType = DropIndxReq::RT_DICT_COMMIT;
+ dropIndex_sendSlaveReq(signal, opPtr);
+}
+
+void
+Dbdict::dropIndex_slaveCommit(Signal* signal, OpDropIndexPtr opPtr)
+{
+ jam();
+}
+
+void
+Dbdict::dropIndex_slaveAbort(Signal* signal, OpDropIndexPtr opPtr)
+{
+ jam();
+ DropIndxReq* const req = &opPtr.p->m_request;
+ const Uint32 indexId = req->getIndexId();
+ if (indexId >= c_tableRecordPool.getSize()) {
+ jam();
+ return;
+ }
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, indexId);
+ indexPtr.p->indexState = TableRecord::IS_BROKEN;
+}
+
+void
+Dbdict::dropIndex_sendSlaveReq(Signal* signal, OpDropIndexPtr opPtr)
+{
+ DropIndxReq* const req = (DropIndxReq*)signal->getDataPtrSend();
+ *req = opPtr.p->m_request;
+ req->setUserRef(opPtr.p->m_coordinatorRef);
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(opPtr.p->m_requestType);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ opPtr.p->m_signalCounter = c_aliveNodes;
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ sendSignal(rg, GSN_DROP_INDX_REQ,
+ signal, DropIndxReq::SignalLength, JBB);
+}
+
+void
+Dbdict::dropIndex_sendReply(Signal* signal, OpDropIndexPtr opPtr,
+ bool toUser)
+{
+ DropIndxRef* rep = (DropIndxRef*)signal->getDataPtrSend();
+ Uint32 gsn = GSN_DROP_INDX_CONF;
+ Uint32 length = DropIndxConf::InternalLength;
+ bool sendRef = opPtr.p->hasError();
+ if (! toUser) {
+ rep->setUserRef(opPtr.p->m_coordinatorRef);
+ rep->setConnectionPtr(opPtr.p->key);
+ rep->setRequestType(opPtr.p->m_requestType);
+ if (opPtr.p->m_requestType == DropIndxReq::RT_DICT_ABORT)
+ sendRef = false;
+ } else {
+ rep->setUserRef(opPtr.p->m_request.getUserRef());
+ rep->setConnectionPtr(opPtr.p->m_request.getConnectionPtr());
+ rep->setRequestType(opPtr.p->m_request.getRequestType());
+ length = DropIndxConf::SignalLength;
+ }
+ rep->setTableId(opPtr.p->m_request.getTableId());
+ rep->setIndexId(opPtr.p->m_request.getIndexId());
+ rep->setIndexVersion(opPtr.p->m_request.getIndexVersion());
+ if (sendRef) {
+ if (opPtr.p->m_errorNode == 0)
+ opPtr.p->m_errorNode = getOwnNodeId();
+ rep->setErrorCode(opPtr.p->m_errorCode);
+ rep->setErrorLine(opPtr.p->m_errorLine);
+ rep->setErrorNode(opPtr.p->m_errorNode);
+ gsn = GSN_DROP_INDX_REF;
+ length = DropIndxRef::SignalLength;
+ }
+ sendSignal(rep->getUserRef(), gsn, signal, length, JBB);
+}
+
+/*****************************************************
+ *
+ * Util signalling
+ *
+ *****************************************************/
+
+int
+Dbdict::sendSignalUtilReq(Callback *pcallback,
+ BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections)
+{
+ jam();
+ EVENT_TRACE;
+ OpSignalUtilPtr utilRecPtr;
+
+ // Seize a Util Send record
+ if (!c_opSignalUtil.seize(utilRecPtr)) {
+ // Failed to allocate util record
+ return -1;
+ }
+ utilRecPtr.p->m_callback = *pcallback;
+
+ // should work for all util signal classes
+ UtilPrepareReq *req = (UtilPrepareReq*)signal->getDataPtrSend();
+ utilRecPtr.p->m_userData = req->getSenderData();
+ req->setSenderData(utilRecPtr.i);
+
+ if (ptr) {
+ jam();
+ sendSignal(ref, gsn, signal, length, jbuf, ptr, noOfSections);
+ } else {
+ jam();
+ sendSignal(ref, gsn, signal, length, jbuf);
+ }
+
+ return 0;
+}
+
+int
+Dbdict::recvSignalUtilReq(Signal* signal, Uint32 returnCode)
+{
+ jam();
+ EVENT_TRACE;
+ UtilPrepareConf * const req = (UtilPrepareConf*)signal->getDataPtr();
+ OpSignalUtilPtr utilRecPtr;
+ utilRecPtr.i = req->getSenderData();
+ if ((utilRecPtr.p = c_opSignalUtil.getPtr(utilRecPtr.i)) == NULL) {
+ jam();
+ return -1;
+ }
+
+ req->setSenderData(utilRecPtr.p->m_userData);
+ Callback c = utilRecPtr.p->m_callback;
+ c_opSignalUtil.release(utilRecPtr);
+
+ execute(signal, c, returnCode);
+ return 0;
+}
+
+void Dbdict::execUTIL_PREPARE_CONF(Signal *signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ ndbrequire(recvSignalUtilReq(signal, 0) == 0);
+}
+
+void
+Dbdict::execUTIL_PREPARE_REF(Signal *signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ ndbrequire(recvSignalUtilReq(signal, 1) == 0);
+}
+
+void Dbdict::execUTIL_EXECUTE_CONF(Signal *signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ ndbrequire(recvSignalUtilReq(signal, 0) == 0);
+}
+
+void Dbdict::execUTIL_EXECUTE_REF(Signal *signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+
+#ifdef EVENT_DEBUG
+ UtilExecuteRef * ref = (UtilExecuteRef *)signal->getDataPtrSend();
+
+ ndbout_c("execUTIL_EXECUTE_REF");
+ ndbout_c("senderData %u",ref->getSenderData());
+ ndbout_c("errorCode %u",ref->getErrorCode());
+ ndbout_c("TCErrorCode %u",ref->getTCErrorCode());
+#endif
+
+ ndbrequire(recvSignalUtilReq(signal, 1) == 0);
+}
+void Dbdict::execUTIL_RELEASE_CONF(Signal *signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ ndbrequire(false);
+ ndbrequire(recvSignalUtilReq(signal, 0) == 0);
+}
+void Dbdict::execUTIL_RELEASE_REF(Signal *signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ ndbrequire(false);
+ ndbrequire(recvSignalUtilReq(signal, 1) == 0);
+}
+
+/**
+ * MODULE: Create event
+ *
+ * Create event in DICT.
+ *
+ *
+ * Request type in CREATE_EVNT signals:
+ *
+ * Signalflow see Dbdict.txt
+ *
+ */
+
+/*****************************************************************
+ *
+ * Systable stuff
+ *
+ */
+
+const Uint32 Dbdict::sysTab_NDBEVENTS_0_szs[EVENT_SYSTEM_TABLE_LENGTH] = {
+ sizeof(((sysTab_NDBEVENTS_0*)0)->NAME),
+ sizeof(((sysTab_NDBEVENTS_0*)0)->EVENT_TYPE),
+ sizeof(((sysTab_NDBEVENTS_0*)0)->TABLE_NAME),
+ sizeof(((sysTab_NDBEVENTS_0*)0)->ATTRIBUTE_MASK),
+ sizeof(((sysTab_NDBEVENTS_0*)0)->SUBID),
+ sizeof(((sysTab_NDBEVENTS_0*)0)->SUBKEY)
+};
+
+void
+Dbdict::prepareTransactionEventSysTable (Callback *pcallback,
+ Signal* signal,
+ Uint32 senderData,
+ UtilPrepareReq::OperationTypeValue prepReq)
+{
+ // find table id for event system table
+ TableRecord keyRecord;
+ strcpy(keyRecord.tableName, EVENT_SYSTEM_TABLE_NAME);
+
+ TableRecordPtr tablePtr;
+ c_tableRecordHash.find(tablePtr, keyRecord);
+
+ ndbrequire(tablePtr.i != RNIL); // system table must exist
+
+ Uint32 tableId = tablePtr.p->tableId; /* System table */
+ Uint32 noAttr = tablePtr.p->noOfAttributes;
+ ndbrequire(noAttr == EVENT_SYSTEM_TABLE_LENGTH);
+
+ switch (prepReq) {
+ case UtilPrepareReq::Update:
+ case UtilPrepareReq::Insert:
+ case UtilPrepareReq::Write:
+ case UtilPrepareReq::Read:
+ jam();
+ break;
+ case UtilPrepareReq::Delete:
+ jam();
+ noAttr = 1; // only involves Primary key which should be the first
+ break;
+ }
+ prepareUtilTransaction(pcallback, signal, senderData, tableId, NULL,
+ prepReq, noAttr, NULL, NULL);
+}
+
+void
+Dbdict::prepareUtilTransaction(Callback *pcallback,
+ Signal* signal,
+ Uint32 senderData,
+ Uint32 tableId,
+ const char* tableName,
+ UtilPrepareReq::OperationTypeValue prepReq,
+ Uint32 noAttr,
+ Uint32 attrIds[],
+ const char *attrNames[])
+{
+ jam();
+ EVENT_TRACE;
+
+ UtilPrepareReq * utilPrepareReq =
+ (UtilPrepareReq *)signal->getDataPtrSend();
+
+ utilPrepareReq->setSenderRef(reference());
+ utilPrepareReq->setSenderData(senderData);
+
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0],128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, prepReq);
+ if (tableName) {
+ jam();
+ w.add(UtilPrepareReq::TableName, tableName);
+ } else {
+ jam();
+ w.add(UtilPrepareReq::TableId, tableId);
+ }
+ for(Uint32 i = 0; i < noAttr; i++)
+ if (tableName) {
+ jam();
+ w.add(UtilPrepareReq::AttributeName, attrNames[i]);
+ } else {
+ if (attrIds) {
+ jam();
+ w.add(UtilPrepareReq::AttributeId, attrIds[i]);
+ } else {
+ jam();
+ w.add(UtilPrepareReq::AttributeId, i);
+ }
+ }
+#ifdef EVENT_DEBUG
+ // Debugging
+ SimplePropertiesLinearReader reader(propPage, w.getWordsUsed());
+ printf("Dict::prepareInsertTransactions: Sent SimpleProperties:\n");
+ reader.printAll(ndbout);
+#endif
+
+ struct LinearSectionPtr sectionsPtr[UtilPrepareReq::NoOfSections];
+ sectionsPtr[UtilPrepareReq::PROPERTIES_SECTION].p = propPage;
+ sectionsPtr[UtilPrepareReq::PROPERTIES_SECTION].sz = w.getWordsUsed();
+
+ sendSignalUtilReq(pcallback, DBUTIL_REF, GSN_UTIL_PREPARE_REQ, signal,
+ UtilPrepareReq::SignalLength, JBB,
+ sectionsPtr, UtilPrepareReq::NoOfSections);
+}
+
+/*****************************************************************
+ *
+ * CREATE_EVNT_REQ has three types RT_CREATE, RT_GET (from user)
+ * and RT_DICT_AFTER_GET send from master DICT to slaves
+ *
+ * This function just dscpaches these to
+ *
+ * createEvent_RT_USER_CREATE
+ * createEvent_RT_USER_GET
+ * createEvent_RT_DICT_AFTER_GET
+ *
+ * repectively
+ *
+ */
+
+void
+Dbdict::execCREATE_EVNT_REQ(Signal* signal)
+{
+ jamEntry();
+
+#if 0
+ {
+ SafeCounterHandle handle;
+ {
+ SafeCounter tmp(c_counterMgr, handle);
+ tmp.init<CreateEvntRef>(CMVMI, GSN_DUMP_STATE_ORD, /* senderData */ 13);
+ tmp.clearWaitingFor();
+ tmp.setWaitingFor(3);
+ ndbrequire(!tmp.done());
+ ndbout_c("Allocted");
+ }
+ ndbrequire(!handle.done());
+ {
+ SafeCounter tmp(c_counterMgr, handle);
+ tmp.clearWaitingFor(3);
+ ndbrequire(tmp.done());
+ ndbout_c("Deallocted");
+ }
+ ndbrequire(handle.done());
+ }
+ {
+ NodeBitmask nodes;
+ nodes.clear();
+
+ nodes.set(2);
+ nodes.set(3);
+ nodes.set(4);
+ nodes.set(5);
+
+ {
+ Uint32 i = 0;
+ while((i = nodes.find(i)) != NodeBitmask::NotFound){
+ ndbout_c("1 Node id = %u", i);
+ i++;
+ }
+ }
+
+ NodeReceiverGroup rg(DBDICT, nodes);
+ RequestTracker rt2;
+ ndbrequire(rt2.done());
+ ndbrequire(!rt2.hasRef());
+ ndbrequire(!rt2.hasConf());
+ rt2.init<CreateEvntRef>(c_counterMgr, rg, GSN_CREATE_EVNT_REF, 13);
+
+ RequestTracker rt3;
+ rt3.init<CreateEvntRef>(c_counterMgr, rg, GSN_CREATE_EVNT_REF, 13);
+
+ ndbrequire(!rt2.done());
+ ndbrequire(!rt3.done());
+
+ rt2.reportRef(c_counterMgr, 2);
+ rt3.reportConf(c_counterMgr, 2);
+
+ ndbrequire(!rt2.done());
+ ndbrequire(!rt3.done());
+
+ rt2.reportConf(c_counterMgr, 3);
+ rt3.reportConf(c_counterMgr, 3);
+
+ ndbrequire(!rt2.done());
+ ndbrequire(!rt3.done());
+
+ rt2.reportConf(c_counterMgr, 4);
+ rt3.reportConf(c_counterMgr, 4);
+
+ ndbrequire(!rt2.done());
+ ndbrequire(!rt3.done());
+
+ rt2.reportConf(c_counterMgr, 5);
+ rt3.reportConf(c_counterMgr, 5);
+
+ ndbrequire(rt2.done());
+ ndbrequire(rt3.done());
+ }
+#endif
+
+ if (! assembleFragments(signal)) {
+ jam();
+ return;
+ }
+
+ CreateEvntReq *req = (CreateEvntReq*)signal->getDataPtr();
+ const CreateEvntReq::RequestType requestType = req->getRequestType();
+ const Uint32 requestFlag = req->getRequestFlag();
+
+ OpCreateEventPtr evntRecPtr;
+ // Seize a Create Event record
+ if (!c_opCreateEvent.seize(evntRecPtr)) {
+ // Failed to allocate event record
+ jam();
+ releaseSections(signal);
+
+ CreateEvntRef * ret = (CreateEvntRef *)signal->getDataPtrSend();
+ ret->senderRef = reference();
+ ret->setErrorCode(CreateEvntRef::SeizeError);
+ ret->setErrorLine(__LINE__);
+ ret->setErrorNode(reference());
+ sendSignal(signal->senderBlockRef(), GSN_CREATE_EVNT_REF, signal,
+ CreateEvntRef::SignalLength, JBB);
+ return;
+ }
+
+#ifdef EVENT_DEBUG
+ ndbout_c("DBDICT::execCREATE_EVNT_REQ from %u evntRecId = (%d)", refToNode(signal->getSendersBlockRef()), evntRecPtr.i);
+#endif
+
+ ndbrequire(req->getUserRef() == signal->getSendersBlockRef());
+
+ evntRecPtr.p->init(req,this);
+
+ if (requestFlag & (Uint32)CreateEvntReq::RT_DICT_AFTER_GET) {
+ jam();
+ EVENT_TRACE;
+ createEvent_RT_DICT_AFTER_GET(signal, evntRecPtr);
+ return;
+ }
+ if (requestType == CreateEvntReq::RT_USER_GET) {
+ jam();
+ EVENT_TRACE;
+ createEvent_RT_USER_GET(signal, evntRecPtr);
+ return;
+ }
+ if (requestType == CreateEvntReq::RT_USER_CREATE) {
+ jam();
+ EVENT_TRACE;
+ createEvent_RT_USER_CREATE(signal, evntRecPtr);
+ return;
+ }
+
+#ifdef EVENT_DEBUG
+ ndbout << "Dbdict.cpp: Dbdict::execCREATE_EVNT_REQ other" << endl;
+#endif
+ jam();
+ releaseSections(signal);
+
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+
+ createEvent_sendReply(signal, evntRecPtr);
+}
+
+/********************************************************************
+ *
+ * Event creation
+ *
+ *****************************************************************/
+
+void
+Dbdict::createEvent_RT_USER_CREATE(Signal* signal, OpCreateEventPtr evntRecPtr){
+ jam();
+ evntRecPtr.p->m_request.setUserRef(signal->senderBlockRef());
+
+#ifdef EVENT_DEBUG
+ ndbout << "Dbdict.cpp: Dbdict::execCREATE_EVNT_REQ RT_USER" << endl;
+ char buf[128] = {0};
+ AttributeMask mask = evntRecPtr.p->m_request.getAttrListBitmask();
+ mask.getText(buf);
+ ndbout_c("mask = %s", buf);
+#endif
+
+ // Interpret the long signal
+
+ SegmentedSectionPtr ssPtr;
+ // save name and event properties
+ signal->getSection(ssPtr, CreateEvntReq::EVENT_NAME_SECTION);
+
+ SimplePropertiesSectionReader r0(ssPtr, getSectionSegmentPool());
+#ifdef EVENT_DEBUG
+ r0.printAll(ndbout);
+#endif
+ // event name
+ if ((!r0.first()) ||
+ (r0.getValueType() != SimpleProperties::StringValue) ||
+ (r0.getValueLen() <= 0)) {
+ jam();
+ releaseSections(signal);
+
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+
+ createEvent_sendReply(signal, evntRecPtr);
+ return;
+ }
+ r0.getString(evntRecPtr.p->m_eventRec.NAME);
+ {
+ int len = strlen(evntRecPtr.p->m_eventRec.NAME);
+ memset(evntRecPtr.p->m_eventRec.NAME+len, 0, MAX_TAB_NAME_SIZE-len);
+#ifdef EVENT_DEBUG
+ printf("CreateEvntReq::RT_USER_CREATE; EventName %s, len %u\n",
+ evntRecPtr.p->m_eventRec.NAME, len);
+ for(int i = 0; i < MAX_TAB_NAME_SIZE/4; i++)
+ printf("H'%.8x ", ((Uint32*)evntRecPtr.p->m_eventRec.NAME)[i]);
+ printf("\n");
+#endif
+ }
+ // table name
+ if ((!r0.next()) ||
+ (r0.getValueType() != SimpleProperties::StringValue) ||
+ (r0.getValueLen() <= 0)) {
+ jam();
+ releaseSections(signal);
+
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+
+ createEvent_sendReply(signal, evntRecPtr);
+ return;
+ }
+ r0.getString(evntRecPtr.p->m_eventRec.TABLE_NAME);
+ {
+ int len = strlen(evntRecPtr.p->m_eventRec.TABLE_NAME);
+ memset(evntRecPtr.p->m_eventRec.TABLE_NAME+len, 0, MAX_TAB_NAME_SIZE-len);
+ }
+
+#ifdef EVENT_DEBUG
+ ndbout_c("event name: %s",evntRecPtr.p->m_eventRec.NAME);
+ ndbout_c("table name: %s",evntRecPtr.p->m_eventRec.TABLE_NAME);
+#endif
+
+ releaseSections(signal);
+
+ // Send request to SUMA
+
+ CreateSubscriptionIdReq * sumaIdReq =
+ (CreateSubscriptionIdReq *)signal->getDataPtrSend();
+
+ // make sure we save the original sender for later
+ sumaIdReq->senderData = evntRecPtr.i;
+#ifdef EVENT_DEBUG
+ ndbout << "sumaIdReq->senderData = " << sumaIdReq->senderData << endl;
+#endif
+ sendSignal(SUMA_REF, GSN_CREATE_SUBID_REQ, signal,
+ CreateSubscriptionIdReq::SignalLength, JBB);
+ // we should now return in either execCREATE_SUBID_CONF
+ // or execCREATE_SUBID_REF
+}
+
+void Dbdict::execCREATE_SUBID_REF(Signal* signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ CreateSubscriptionIdRef * const ref =
+ (CreateSubscriptionIdRef *)signal->getDataPtr();
+ OpCreateEventPtr evntRecPtr;
+
+ evntRecPtr.i = ref->senderData;
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+
+ createEvent_sendReply(signal, evntRecPtr);
+}
+
+void Dbdict::execCREATE_SUBID_CONF(Signal* signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+
+ CreateSubscriptionIdConf const * sumaIdConf =
+ (CreateSubscriptionIdConf *)signal->getDataPtr();
+
+ Uint32 evntRecId = sumaIdConf->senderData;
+ OpCreateEvent *evntRec;
+
+ ndbrequire((evntRec = c_opCreateEvent.getPtr(evntRecId)) != NULL);
+
+ evntRec->m_request.setEventId(sumaIdConf->subscriptionId);
+ evntRec->m_request.setEventKey(sumaIdConf->subscriptionKey);
+
+ releaseSections(signal);
+
+ Callback c = { safe_cast(&Dbdict::createEventUTIL_PREPARE), 0 };
+
+ prepareTransactionEventSysTable(&c, signal, evntRecId,
+ UtilPrepareReq::Insert);
+}
+
+void
+Dbdict::createEventComplete_RT_USER_CREATE(Signal* signal,
+ OpCreateEventPtr evntRecPtr){
+ jam();
+ createEvent_sendReply(signal, evntRecPtr);
+}
+
+/*********************************************************************
+ *
+ * UTIL_PREPARE, UTIL_EXECUTE
+ *
+ * insert or read systable NDB$EVENTS_0
+ */
+
+void interpretUtilPrepareErrorCode(UtilPrepareRef::ErrorCode errorCode,
+ bool& temporary, Uint32& line)
+{
+ switch (errorCode) {
+ case UtilPrepareRef::NO_ERROR:
+ jam();
+ line = __LINE__;
+ EVENT_TRACE;
+ break;
+ case UtilPrepareRef::PREPARE_SEIZE_ERROR:
+ jam();
+ temporary = true;
+ line = __LINE__;
+ EVENT_TRACE;
+ break;
+ case UtilPrepareRef::PREPARE_PAGES_SEIZE_ERROR:
+ jam();
+ line = __LINE__;
+ EVENT_TRACE;
+ break;
+ case UtilPrepareRef::PREPARED_OPERATION_SEIZE_ERROR:
+ jam();
+ line = __LINE__;
+ EVENT_TRACE;
+ break;
+ case UtilPrepareRef::DICT_TAB_INFO_ERROR:
+ jam();
+ line = __LINE__;
+ EVENT_TRACE;
+ break;
+ case UtilPrepareRef::MISSING_PROPERTIES_SECTION:
+ jam();
+ line = __LINE__;
+ EVENT_TRACE;
+ break;
+ default:
+ jam();
+ line = __LINE__;
+ EVENT_TRACE;
+ break;
+ }
+}
+
+void
+Dbdict::createEventUTIL_PREPARE(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ EVENT_TRACE;
+ if (returnCode == 0) {
+ UtilPrepareConf* const req = (UtilPrepareConf*)signal->getDataPtr();
+ OpCreateEventPtr evntRecPtr;
+ jam();
+ evntRecPtr.i = req->getSenderData();
+ const Uint32 prepareId = req->getPrepareId();
+
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ Callback c = { safe_cast(&Dbdict::createEventUTIL_EXECUTE), 0 };
+
+ switch (evntRecPtr.p->m_requestType) {
+ case CreateEvntReq::RT_USER_GET:
+#ifdef EVENT_DEBUG
+ printf("get type = %d\n", CreateEvntReq::RT_USER_GET);
+#endif
+ jam();
+ executeTransEventSysTable(&c, signal,
+ evntRecPtr.i, evntRecPtr.p->m_eventRec,
+ prepareId, UtilPrepareReq::Read);
+ break;
+ case CreateEvntReq::RT_USER_CREATE:
+#ifdef EVENT_DEBUG
+ printf("create type = %d\n", CreateEvntReq::RT_USER_CREATE);
+#endif
+ {
+ evntRecPtr.p->m_eventRec.EVENT_TYPE = evntRecPtr.p->m_request.getEventType();
+ AttributeMask m = evntRecPtr.p->m_request.getAttrListBitmask();
+ memcpy(evntRecPtr.p->m_eventRec.ATTRIBUTE_MASK, &m,
+ sizeof(evntRecPtr.p->m_eventRec.ATTRIBUTE_MASK));
+ evntRecPtr.p->m_eventRec.SUBID = evntRecPtr.p->m_request.getEventId();
+ evntRecPtr.p->m_eventRec.SUBKEY = evntRecPtr.p->m_request.getEventKey();
+ }
+ jam();
+ executeTransEventSysTable(&c, signal,
+ evntRecPtr.i, evntRecPtr.p->m_eventRec,
+ prepareId, UtilPrepareReq::Insert);
+ break;
+ default:
+#ifdef EVENT_DEBUG
+ printf("type = %d\n", evntRecPtr.p->m_requestType);
+ printf("bet type = %d\n", CreateEvntReq::RT_USER_GET);
+ printf("create type = %d\n", CreateEvntReq::RT_USER_CREATE);
+#endif
+ ndbrequire(false);
+ }
+ } else { // returnCode != 0
+ UtilPrepareRef* const ref = (UtilPrepareRef*)signal->getDataPtr();
+
+ const UtilPrepareRef::ErrorCode errorCode =
+ (UtilPrepareRef::ErrorCode)ref->getErrorCode();
+
+ OpCreateEventPtr evntRecPtr;
+ evntRecPtr.i = ref->getSenderData();
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ bool temporary = false;
+ interpretUtilPrepareErrorCode(errorCode,
+ temporary, evntRecPtr.p->m_errorLine);
+ if (temporary) {
+ evntRecPtr.p->m_errorCode =
+ CreateEvntRef::makeTemporary(CreateEvntRef::Undefined);
+ }
+
+ if (evntRecPtr.p->m_errorCode == 0) {
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ }
+ evntRecPtr.p->m_errorNode = reference();
+
+ createEvent_sendReply(signal, evntRecPtr);
+ }
+}
+
+void Dbdict::executeTransEventSysTable(Callback *pcallback, Signal *signal,
+ const Uint32 ptrI,
+ sysTab_NDBEVENTS_0& m_eventRec,
+ const Uint32 prepareId,
+ UtilPrepareReq::OperationTypeValue prepReq)
+{
+ jam();
+ const Uint32 noAttr = EVENT_SYSTEM_TABLE_LENGTH;
+ Uint32 total_len = 0;
+
+ Uint32* attrHdr = signal->theData + 25;
+ Uint32* attrPtr = attrHdr;
+
+ Uint32 id=0;
+ // attribute 0 event name: Primary Key
+ {
+ AttributeHeader::init(attrPtr, id, sysTab_NDBEVENTS_0_szs[id]/4);
+ total_len += sysTab_NDBEVENTS_0_szs[id];
+ attrPtr++; id++;
+ }
+
+ switch (prepReq) {
+ case UtilPrepareReq::Read:
+ jam();
+ EVENT_TRACE;
+ // no more
+ while ( id < noAttr )
+ AttributeHeader::init(attrPtr++, id++, 0);
+ ndbrequire(id == (Uint32) noAttr);
+ break;
+ case UtilPrepareReq::Insert:
+ jam();
+ EVENT_TRACE;
+ while ( id < noAttr ) {
+ AttributeHeader::init(attrPtr, id, sysTab_NDBEVENTS_0_szs[id]/4);
+ total_len += sysTab_NDBEVENTS_0_szs[id];
+ attrPtr++; id++;
+ }
+ ndbrequire(id == (Uint32) noAttr);
+ break;
+ case UtilPrepareReq::Delete:
+ ndbrequire(id == 1);
+ break;
+ default:
+ ndbrequire(false);
+ }
+
+ LinearSectionPtr headerPtr;
+ LinearSectionPtr dataPtr;
+
+ headerPtr.p = attrHdr;
+ headerPtr.sz = noAttr;
+
+ dataPtr.p = (Uint32*)&m_eventRec;
+ dataPtr.sz = total_len/4;
+
+ ndbrequire((total_len == sysTab_NDBEVENTS_0_szs[0]) ||
+ (total_len == sizeof(sysTab_NDBEVENTS_0)));
+
+#if 0
+ printf("Header size %u\n", headerPtr.sz);
+ for(int i = 0; i < (int)headerPtr.sz; i++)
+ printf("H'%.8x ", attrHdr[i]);
+ printf("\n");
+
+ printf("Data size %u\n", dataPtr.sz);
+ for(int i = 0; i < (int)dataPtr.sz; i++)
+ printf("H'%.8x ", dataPage[i]);
+ printf("\n");
+#endif
+
+ executeTransaction(pcallback, signal,
+ ptrI,
+ prepareId,
+ id,
+ headerPtr,
+ dataPtr);
+}
+
+void Dbdict::executeTransaction(Callback *pcallback,
+ Signal* signal,
+ Uint32 senderData,
+ Uint32 prepareId,
+ Uint32 noAttr,
+ LinearSectionPtr headerPtr,
+ LinearSectionPtr dataPtr)
+{
+ jam();
+ EVENT_TRACE;
+
+ UtilExecuteReq * utilExecuteReq =
+ (UtilExecuteReq *)signal->getDataPtrSend();
+
+ utilExecuteReq->setSenderRef(reference());
+ utilExecuteReq->setSenderData(senderData);
+ utilExecuteReq->setPrepareId(prepareId);
+ utilExecuteReq->setReleaseFlag(); // must be done after setting prepareId
+
+#if 0
+ printf("Header size %u\n", headerPtr.sz);
+ for(int i = 0; i < (int)headerPtr.sz; i++)
+ printf("H'%.8x ", headerBuffer[i]);
+ printf("\n");
+
+ printf("Data size %u\n", dataPtr.sz);
+ for(int i = 0; i < (int)dataPtr.sz; i++)
+ printf("H'%.8x ", dataBuffer[i]);
+ printf("\n");
+#endif
+
+ struct LinearSectionPtr sectionsPtr[UtilExecuteReq::NoOfSections];
+ sectionsPtr[UtilExecuteReq::HEADER_SECTION].p = headerPtr.p;
+ sectionsPtr[UtilExecuteReq::HEADER_SECTION].sz = noAttr;
+ sectionsPtr[UtilExecuteReq::DATA_SECTION].p = dataPtr.p;
+ sectionsPtr[UtilExecuteReq::DATA_SECTION].sz = dataPtr.sz;
+
+ sendSignalUtilReq(pcallback, DBUTIL_REF, GSN_UTIL_EXECUTE_REQ, signal,
+ UtilExecuteReq::SignalLength, JBB,
+ sectionsPtr, UtilExecuteReq::NoOfSections);
+}
+
+void Dbdict::parseReadEventSys(Signal* signal, sysTab_NDBEVENTS_0& m_eventRec)
+{
+ SegmentedSectionPtr headerPtr, dataPtr;
+ jam();
+ signal->getSection(headerPtr, UtilExecuteReq::HEADER_SECTION);
+ SectionReader headerReader(headerPtr, getSectionSegmentPool());
+
+ signal->getSection(dataPtr, UtilExecuteReq::DATA_SECTION);
+ SectionReader dataReader(dataPtr, getSectionSegmentPool());
+
+ AttributeHeader header;
+ Uint32 *dst = (Uint32*)&m_eventRec;
+
+ for (int i = 0; i < EVENT_SYSTEM_TABLE_LENGTH; i++) {
+ headerReader.getWord((Uint32 *)&header);
+ int sz = header.getDataSize();
+ for (int i=0; i < sz; i++)
+ dataReader.getWord(dst++);
+ }
+
+ ndbrequire( ((char*)dst-(char*)&m_eventRec) == sizeof(m_eventRec) );
+
+ releaseSections(signal);
+}
+
+void Dbdict::createEventUTIL_EXECUTE(Signal *signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ EVENT_TRACE;
+ if (returnCode == 0) {
+ // Entry into system table all set
+ UtilExecuteConf* const conf = (UtilExecuteConf*)signal->getDataPtr();
+ jam();
+ OpCreateEventPtr evntRecPtr;
+ evntRecPtr.i = conf->getSenderData();
+
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+ OpCreateEvent *evntRec = evntRecPtr.p;
+
+ switch (evntRec->m_requestType) {
+ case CreateEvntReq::RT_USER_GET: {
+#ifdef EVENT_DEBUG
+ printf("get type = %d\n", CreateEvntReq::RT_USER_GET);
+#endif
+ parseReadEventSys(signal, evntRecPtr.p->m_eventRec);
+
+ evntRec->m_request.setEventType(evntRecPtr.p->m_eventRec.EVENT_TYPE);
+ evntRec->m_request.setAttrListBitmask(*(AttributeMask*)evntRecPtr.p->m_eventRec.ATTRIBUTE_MASK);
+ evntRec->m_request.setEventId(evntRecPtr.p->m_eventRec.SUBID);
+ evntRec->m_request.setEventKey(evntRecPtr.p->m_eventRec.SUBKEY);
+
+#ifdef EVENT_DEBUG
+ printf("EventName: %s\n", evntRec->m_eventRec.NAME);
+ printf("TableName: %s\n", evntRec->m_eventRec.TABLE_NAME);
+#endif
+
+ // find table id for event table
+ TableRecord keyRecord;
+ strcpy(keyRecord.tableName, evntRecPtr.p->m_eventRec.TABLE_NAME);
+
+ TableRecordPtr tablePtr;
+ c_tableRecordHash.find(tablePtr, keyRecord);
+
+ if (tablePtr.i == RNIL) {
+ jam();
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+
+ createEvent_sendReply(signal, evntRecPtr);
+ return;
+ }
+
+ evntRec->m_request.setTableId(tablePtr.p->tableId);
+
+ createEventComplete_RT_USER_GET(signal, evntRecPtr);
+ return;
+ }
+ case CreateEvntReq::RT_USER_CREATE: {
+#ifdef EVENT_DEBUG
+ printf("create type = %d\n", CreateEvntReq::RT_USER_CREATE);
+#endif
+ jam();
+ createEventComplete_RT_USER_CREATE(signal, evntRecPtr);
+ return;
+ }
+ break;
+ default:
+ ndbrequire(false);
+ }
+ } else { // returnCode != 0
+ UtilExecuteRef * const ref = (UtilExecuteRef *)signal->getDataPtr();
+ OpCreateEventPtr evntRecPtr;
+ evntRecPtr.i = ref->getSenderData();
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+ jam();
+ evntRecPtr.p->m_errorNode = reference();
+ evntRecPtr.p->m_errorLine = __LINE__;
+
+ switch (ref->getErrorCode()) {
+ case UtilExecuteRef::TCError:
+ switch (ref->getTCErrorCode()) {
+ case ZNOT_FOUND:
+ jam();
+ evntRecPtr.p->m_errorCode = CreateEvntRef::EventNotFound;
+ break;
+ case ZALREADYEXIST:
+ jam();
+ evntRecPtr.p->m_errorCode = CreateEvntRef::EventExists;
+ break;
+ default:
+ jam();
+ evntRecPtr.p->m_errorCode = CreateEvntRef::UndefinedTCError;
+ break;
+ }
+ break;
+ default:
+ jam();
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ break;
+ }
+
+ createEvent_sendReply(signal, evntRecPtr);
+ }
+}
+
+/***********************************************************************
+ *
+ * NdbEventOperation, reading systable, creating event in suma
+ *
+ */
+
+void
+Dbdict::createEvent_RT_USER_GET(Signal* signal, OpCreateEventPtr evntRecPtr){
+ jam();
+ EVENT_TRACE;
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("DBDICT(Coordinator) got GSN_CREATE_EVNT_REQ::RT_USER_GET evntRecPtr.i = (%d), ref = %u", evntRecPtr.i, evntRecPtr.p->m_request.getUserRef());
+#endif
+
+ SegmentedSectionPtr ssPtr;
+
+ signal->getSection(ssPtr, 0);
+
+ SimplePropertiesSectionReader r0(ssPtr, getSectionSegmentPool());
+#ifdef EVENT_DEBUG
+ r0.printAll(ndbout);
+#endif
+ if ((!r0.first()) ||
+ (r0.getValueType() != SimpleProperties::StringValue) ||
+ (r0.getValueLen() <= 0)) {
+ jam();
+ releaseSections(signal);
+
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+
+ createEvent_sendReply(signal, evntRecPtr);
+ return;
+ }
+
+ r0.getString(evntRecPtr.p->m_eventRec.NAME);
+ int len = strlen(evntRecPtr.p->m_eventRec.NAME);
+ memset(evntRecPtr.p->m_eventRec.NAME+len, 0, MAX_TAB_NAME_SIZE-len);
+
+ releaseSections(signal);
+
+ Callback c = { safe_cast(&Dbdict::createEventUTIL_PREPARE), 0 };
+
+ prepareTransactionEventSysTable(&c, signal, evntRecPtr.i,
+ UtilPrepareReq::Read);
+ /*
+ * Will read systable and fill an OpCreateEventPtr
+ * and return below
+ */
+}
+
+void
+Dbdict::createEventComplete_RT_USER_GET(Signal* signal,
+ OpCreateEventPtr evntRecPtr){
+ jam();
+
+ // Send to oneself and the other DICT's
+ CreateEvntReq * req = (CreateEvntReq *)signal->getDataPtrSend();
+
+ *req = evntRecPtr.p->m_request;
+ req->senderRef = reference();
+ req->senderData = evntRecPtr.i;
+
+ req->addRequestFlag(CreateEvntReq::RT_DICT_AFTER_GET);
+
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("DBDICT(Coordinator) sending GSN_CREATE_EVNT_REQ::RT_DICT_AFTER_GET to DBDICT participants evntRecPtr.i = (%d)", evntRecPtr.i);
+#endif
+
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ evntRecPtr.p->m_reqTracker.init<CreateEvntRef>
+ (c_counterMgr, rg, GSN_CREATE_EVNT_REF, evntRecPtr.i);
+ sendSignal(rg, GSN_CREATE_EVNT_REQ, signal, CreateEvntReq::SignalLength, JBB);
+}
+
+void
+Dbdict::createEvent_nodeFailCallback(Signal* signal, Uint32 eventRecPtrI,
+ Uint32 returnCode){
+ OpCreateEventPtr evntRecPtr;
+ c_opCreateEvent.getPtr(evntRecPtr, eventRecPtrI);
+ createEvent_sendReply(signal, evntRecPtr);
+}
+
+void Dbdict::execCREATE_EVNT_REF(Signal* signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ CreateEvntRef * const ref = (CreateEvntRef *)signal->getDataPtr();
+ OpCreateEventPtr evntRecPtr;
+
+ evntRecPtr.i = ref->getUserData();
+
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("DBDICT(Coordinator) got GSN_CREATE_EVNT_REF evntRecPtr.i = (%d)", evntRecPtr.i);
+#endif
+
+ if (ref->errorCode == CreateEvntRef::NF_FakeErrorREF){
+ jam();
+ evntRecPtr.p->m_reqTracker.ignoreRef(c_counterMgr, refToNode(ref->senderRef));
+ } else {
+ jam();
+ evntRecPtr.p->m_reqTracker.reportRef(c_counterMgr, refToNode(ref->senderRef));
+ }
+ createEvent_sendReply(signal, evntRecPtr);
+
+ return;
+}
+
+void Dbdict::execCREATE_EVNT_CONF(Signal* signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ CreateEvntConf * const conf = (CreateEvntConf *)signal->getDataPtr();
+ OpCreateEventPtr evntRecPtr;
+
+ evntRecPtr.i = conf->getUserData();
+
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("DBDICT(Coordinator) got GSN_CREATE_EVNT_CONF evntRecPtr.i = (%d)", evntRecPtr.i);
+#endif
+
+ evntRecPtr.p->m_reqTracker.reportConf(c_counterMgr, refToNode(conf->senderRef));
+
+ // we will only have a valid tablename if it the master DICT sending this
+ // but that's ok
+ LinearSectionPtr ptr[1];
+ ptr[0].p = (Uint32 *)evntRecPtr.p->m_eventRec.TABLE_NAME;
+ ptr[0].sz =
+ (strlen(evntRecPtr.p->m_eventRec.TABLE_NAME)+4)/4; // to make sure we have a null
+
+ createEvent_sendReply(signal, evntRecPtr, ptr, 1);
+
+ return;
+}
+
+/************************************************
+ *
+ * Participant stuff
+ *
+ */
+
+void
+Dbdict::createEvent_RT_DICT_AFTER_GET(Signal* signal, OpCreateEventPtr evntRecPtr){
+ jam();
+ evntRecPtr.p->m_request.setUserRef(signal->senderBlockRef());
+
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("DBDICT(Participant) got CREATE_EVNT_REQ::RT_DICT_AFTER_GET evntRecPtr.i = (%d)", evntRecPtr.i);
+#endif
+
+ // the signal comes from the DICT block that got the first user request!
+ // This code runs on all DICT nodes, including oneself
+
+ // Seize a Create Event record, the Coordinator will now have two seized
+ // but that's ok, it's like a recursion
+
+ SubCreateReq * sumaReq = (SubCreateReq *)signal->getDataPtrSend();
+
+ sumaReq->subscriberRef = reference(); // reference to DICT
+ sumaReq->subscriberData = evntRecPtr.i;
+ sumaReq->subscriptionId = evntRecPtr.p->m_request.getEventId();
+ sumaReq->subscriptionKey = evntRecPtr.p->m_request.getEventKey();
+ sumaReq->subscriptionType = SubCreateReq::TableEvent;
+ sumaReq->tableId = evntRecPtr.p->m_request.getTableId();
+
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("sending GSN_SUB_CREATE_REQ");
+#endif
+
+ sendSignal(SUMA_REF, GSN_SUB_CREATE_REQ, signal,
+ SubCreateReq::SignalLength+1 /*to get table Id*/, JBB);
+}
+
+void Dbdict::execSUB_CREATE_REF(Signal* signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ SubCreateRef * const ref = (SubCreateRef *)signal->getDataPtr();
+ OpCreateEventPtr evntRecPtr;
+
+ evntRecPtr.i = ref->subscriberData;
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("DBDICT(Participant) got SUB_CREATE_REF evntRecPtr.i = (%d)", evntRecPtr.i);
+#endif
+
+ if (ref->err == GrepError::SUBSCRIPTION_ID_NOT_UNIQUE) {
+ jam();
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("SUBSCRIPTION_ID_NOT_UNIQUE");
+#endif
+ createEvent_sendReply(signal, evntRecPtr);
+ return;
+ }
+
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("Other error");
+#endif
+
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+
+ createEvent_sendReply(signal, evntRecPtr);
+}
+
+void Dbdict::execSUB_CREATE_CONF(Signal* signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+
+ SubCreateConf * const sumaConf = (SubCreateConf *)signal->getDataPtr();
+
+ const Uint32 subscriptionId = sumaConf->subscriptionId;
+ const Uint32 subscriptionKey = sumaConf->subscriptionKey;
+ const Uint32 evntRecId = sumaConf->subscriberData;
+
+ OpCreateEvent *evntRec;
+ ndbrequire((evntRec = c_opCreateEvent.getPtr(evntRecId)) != NULL);
+
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("DBDICT(Participant) got SUB_CREATE_CONF evntRecPtr.i = (%d)", evntRecId);
+#endif
+
+ SubSyncReq *sumaSync = (SubSyncReq *)signal->getDataPtrSend();
+
+ sumaSync->subscriptionId = subscriptionId;
+ sumaSync->subscriptionKey = subscriptionKey;
+ sumaSync->part = (Uint32) SubscriptionData::MetaData;
+ sumaSync->subscriberData = evntRecId;
+
+ sendSignal(SUMA_REF, GSN_SUB_SYNC_REQ, signal,
+ SubSyncReq::SignalLength, JBB);
+}
+
+void Dbdict::execSUB_SYNC_REF(Signal* signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+ SubSyncRef * const ref = (SubSyncRef *)signal->getDataPtr();
+ OpCreateEventPtr evntRecPtr;
+
+ evntRecPtr.i = ref->subscriberData;
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+
+ createEvent_sendReply(signal, evntRecPtr);
+}
+
+void Dbdict::execSUB_SYNC_CONF(Signal* signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+
+ SubSyncConf * const sumaSyncConf = (SubSyncConf *)signal->getDataPtr();
+
+ // Uint32 subscriptionId = sumaSyncConf->subscriptionId;
+ // Uint32 subscriptionKey = sumaSyncConf->subscriptionKey;
+ OpCreateEventPtr evntRecPtr;
+
+ evntRecPtr.i = sumaSyncConf->subscriberData;
+ ndbrequire((evntRecPtr.p = c_opCreateEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ ndbrequire(sumaSyncConf->part == (Uint32)SubscriptionData::MetaData);
+
+ createEvent_sendReply(signal, evntRecPtr);
+}
+
+/****************************************************
+ *
+ * common create reply method
+ *
+ *******************************************************/
+
+void Dbdict::createEvent_sendReply(Signal* signal,
+ OpCreateEventPtr evntRecPtr,
+ LinearSectionPtr *ptr, int noLSP)
+{
+ jam();
+ EVENT_TRACE;
+
+ // check if we're ready to sent reply
+ // if we are the master dict we might be waiting for conf/ref
+
+ if (!evntRecPtr.p->m_reqTracker.done()) {
+ jam();
+ return; // there's more to come
+ }
+
+ if (evntRecPtr.p->m_reqTracker.hasRef()) {
+ ptr = NULL; // we don't want to return anything if there's an error
+ if (!evntRecPtr.p->hasError()) {
+ evntRecPtr.p->m_errorCode = CreateEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+ jam();
+ } else
+ jam();
+ }
+
+ // reference to API if master DICT
+ // else reference to master DICT
+ Uint32 senderRef = evntRecPtr.p->m_request.getUserRef();
+ Uint32 signalLength;
+ Uint32 gsn;
+
+ if (evntRecPtr.p->hasError()) {
+ jam();
+ EVENT_TRACE;
+ CreateEvntRef * ret = (CreateEvntRef *)signal->getDataPtrSend();
+
+ ret->setEventId(evntRecPtr.p->m_request.getEventId());
+ ret->setEventKey(evntRecPtr.p->m_request.getEventKey());
+ ret->setUserData(evntRecPtr.p->m_request.getUserData());
+ ret->senderRef = reference();
+ ret->setTableId(evntRecPtr.p->m_request.getTableId());
+ ret->setEventType(evntRecPtr.p->m_request.getEventType());
+ ret->setRequestType(evntRecPtr.p->m_request.getRequestType());
+
+ ret->setErrorCode(evntRecPtr.p->m_errorCode);
+ ret->setErrorLine(evntRecPtr.p->m_errorLine);
+ ret->setErrorNode(evntRecPtr.p->m_errorNode);
+
+ signalLength = CreateEvntRef::SignalLength;
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("DBDICT sending GSN_CREATE_EVNT_REF to evntRecPtr.i = (%d) node = %u ref = %u", evntRecPtr.i, refToNode(senderRef), senderRef);
+ ndbout_c("errorCode = %u", evntRecPtr.p->m_errorCode);
+ ndbout_c("errorLine = %u", evntRecPtr.p->m_errorLine);
+#endif
+ gsn = GSN_CREATE_EVNT_REF;
+
+ } else {
+ jam();
+ EVENT_TRACE;
+ CreateEvntConf * evntConf = (CreateEvntConf *)signal->getDataPtrSend();
+
+ evntConf->setEventId(evntRecPtr.p->m_request.getEventId());
+ evntConf->setEventKey(evntRecPtr.p->m_request.getEventKey());
+ evntConf->setUserData(evntRecPtr.p->m_request.getUserData());
+ evntConf->senderRef = reference();
+ evntConf->setTableId(evntRecPtr.p->m_request.getTableId());
+ evntConf->setAttrListBitmask(evntRecPtr.p->m_request.getAttrListBitmask());
+ evntConf->setEventType(evntRecPtr.p->m_request.getEventType());
+ evntConf->setRequestType(evntRecPtr.p->m_request.getRequestType());
+
+ signalLength = CreateEvntConf::SignalLength;
+#ifdef EVENT_PH2_DEBUG
+ ndbout_c("DBDICT sending GSN_CREATE_EVNT_CONF to evntRecPtr.i = (%d) node = %u ref = %u", evntRecPtr.i, refToNode(senderRef), senderRef);
+#endif
+ gsn = GSN_CREATE_EVNT_CONF;
+ }
+
+ if (ptr) {
+ jam();
+ sendSignal(senderRef, gsn, signal, signalLength, JBB, ptr, noLSP);
+ } else {
+ jam();
+ sendSignal(senderRef, gsn, signal, signalLength, JBB);
+ }
+
+ c_opCreateEvent.release(evntRecPtr);
+}
+
+/*************************************************************/
+
+/********************************************************************
+ *
+ * Start event
+ *
+ *******************************************************************/
+
+void Dbdict::execSUB_START_REQ(Signal* signal)
+{
+ jamEntry();
+
+ Uint32 origSenderRef = signal->senderBlockRef();
+
+ OpSubEventPtr subbPtr;
+ if (!c_opSubEvent.seize(subbPtr)) {
+ SubStartRef * ref = (SubStartRef *)signal->getDataPtrSend();
+ { // fix
+ Uint32 subcriberRef = ((SubStartReq*)signal->getDataPtr())->subscriberRef;
+ ref->subscriberRef = subcriberRef;
+ }
+ jam();
+ // ret->setErrorCode(SubStartRef::SeizeError);
+ // ret->setErrorLine(__LINE__);
+ // ret->setErrorNode(reference());
+ ref->senderRef = reference();
+ ref->setTemporary(SubStartRef::Busy);
+
+ sendSignal(origSenderRef, GSN_SUB_START_REF, signal,
+ SubStartRef::SignalLength2, JBB);
+ return;
+ }
+
+ {
+ const SubStartReq* req = (SubStartReq*) signal->getDataPtr();
+ subbPtr.p->m_senderRef = req->senderRef;
+ subbPtr.p->m_senderData = req->senderData;
+ subbPtr.p->m_errorCode = 0;
+ }
+
+ if (refToBlock(origSenderRef) != DBDICT) {
+ /*
+ * Coordinator
+ */
+ jam();
+
+ subbPtr.p->m_senderRef = origSenderRef; // not sure if API sets correctly
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ subbPtr.p->m_reqTracker.init<SubStartRef>(c_counterMgr, rg, GSN_SUB_START_REF, subbPtr.i);
+
+ SubStartReq* req = (SubStartReq*) signal->getDataPtrSend();
+
+ req->senderRef = reference();
+ req->senderData = subbPtr.i;
+
+#ifdef EVENT_PH3_DEBUG
+ ndbout_c("DBDICT(Coordinator) sending GSN_SUB_START_REQ to DBDICT participants subbPtr.i = (%d)", subbPtr.i);
+#endif
+
+ sendSignal(rg, GSN_SUB_START_REQ, signal, SubStartReq::SignalLength2, JBB);
+ return;
+ }
+ /*
+ * Participant
+ */
+ ndbrequire(refToBlock(origSenderRef) == DBDICT);
+
+ {
+ SubStartReq* req = (SubStartReq*) signal->getDataPtrSend();
+
+ req->senderRef = reference();
+ req->senderData = subbPtr.i;
+
+#ifdef EVENT_PH3_DEBUG
+ ndbout_c("DBDICT(Participant) sending GSN_SUB_START_REQ to SUMA subbPtr.i = (%d)", subbPtr.i);
+#endif
+ sendSignal(SUMA_REF, GSN_SUB_START_REQ, signal, SubStartReq::SignalLength2, JBB);
+ }
+}
+
+void Dbdict::execSUB_START_REF(Signal* signal)
+{
+ jamEntry();
+
+ const SubStartRef* ref = (SubStartRef*) signal->getDataPtr();
+ Uint32 senderRef = ref->senderRef;
+
+ OpSubEventPtr subbPtr;
+ c_opSubEvent.getPtr(subbPtr, ref->senderData);
+
+ if (refToBlock(senderRef) == SUMA) {
+ /*
+ * Participant
+ */
+ jam();
+
+#ifdef EVENT_PH3_DEBUG
+ ndbout_c("DBDICT(Participant) got GSN_SUB_START_REF = (%d)", subbPtr.i);
+#endif
+
+ if (ref->isTemporary()){
+ jam();
+ SubStartReq* req = (SubStartReq*)signal->getDataPtrSend();
+ { // fix
+ Uint32 subscriberRef = ref->subscriberRef;
+ req->subscriberRef = subscriberRef;
+ }
+ req->senderRef = reference();
+ req->senderData = subbPtr.i;
+ sendSignal(SUMA_REF, GSN_SUB_START_REQ,
+ signal, SubStartReq::SignalLength2, JBB);
+ } else {
+ jam();
+
+ SubStartRef* ref = (SubStartRef*) signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = subbPtr.p->m_senderData;
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_START_REF,
+ signal, SubStartRef::SignalLength2, JBB);
+ c_opSubEvent.release(subbPtr);
+ }
+ return;
+ }
+ /*
+ * Coordinator
+ */
+ ndbrequire(refToBlock(senderRef) == DBDICT);
+#ifdef EVENT_PH3_DEBUG
+ ndbout_c("DBDICT(Coordinator) got GSN_SUB_START_REF = (%d)", subbPtr.i);
+#endif
+ if (ref->errorCode == SubStartRef::NF_FakeErrorREF){
+ jam();
+ subbPtr.p->m_reqTracker.ignoreRef(c_counterMgr, refToNode(senderRef));
+ } else {
+ jam();
+ subbPtr.p->m_reqTracker.reportRef(c_counterMgr, refToNode(senderRef));
+ }
+ completeSubStartReq(signal,subbPtr.i,0);
+}
+
+void Dbdict::execSUB_START_CONF(Signal* signal)
+{
+ jamEntry();
+
+ const SubStartConf* conf = (SubStartConf*) signal->getDataPtr();
+ Uint32 senderRef = conf->senderRef;
+
+ OpSubEventPtr subbPtr;
+ c_opSubEvent.getPtr(subbPtr, conf->senderData);
+
+ if (refToBlock(senderRef) == SUMA) {
+ /*
+ * Participant
+ */
+ jam();
+ SubStartConf* conf = (SubStartConf*) signal->getDataPtrSend();
+
+#ifdef EVENT_PH3_DEBUG
+ ndbout_c("DBDICT(Participant) got GSN_SUB_START_CONF = (%d)", subbPtr.i);
+#endif
+
+ conf->senderRef = reference();
+ conf->senderData = subbPtr.p->m_senderData;
+
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_START_CONF,
+ signal, SubStartConf::SignalLength2, JBB);
+ c_opSubEvent.release(subbPtr);
+ return;
+ }
+ /*
+ * Coordinator
+ */
+ ndbrequire(refToBlock(senderRef) == DBDICT);
+#ifdef EVENT_PH3_DEBUG
+ ndbout_c("DBDICT(Coordinator) got GSN_SUB_START_CONF = (%d)", subbPtr.i);
+#endif
+ subbPtr.p->m_reqTracker.reportConf(c_counterMgr, refToNode(senderRef));
+ completeSubStartReq(signal,subbPtr.i,0);
+}
+
+/*
+ * Coordinator
+ */
+void Dbdict::completeSubStartReq(Signal* signal,
+ Uint32 ptrI,
+ Uint32 returnCode){
+ jam();
+
+ OpSubEventPtr subbPtr;
+ c_opSubEvent.getPtr(subbPtr, ptrI);
+
+ if (!subbPtr.p->m_reqTracker.done()){
+ jam();
+ return;
+ }
+
+ if (subbPtr.p->m_reqTracker.hasRef()) {
+ jam();
+#ifdef EVENT_DEBUG
+ ndbout_c("SUB_START_REF");
+#endif
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_START_REF,
+ signal, SubStartRef::SignalLength, JBB);
+ if (subbPtr.p->m_reqTracker.hasConf()) {
+ // stopStartedNodes(signal);
+ }
+ c_opSubEvent.release(subbPtr);
+ return;
+ }
+#ifdef EVENT_DEBUG
+ ndbout_c("SUB_START_CONF");
+#endif
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_START_CONF,
+ signal, SubStartConf::SignalLength, JBB);
+ c_opSubEvent.release(subbPtr);
+}
+
+/********************************************************************
+ *
+ * Stop event
+ *
+ *******************************************************************/
+
+void Dbdict::execSUB_STOP_REQ(Signal* signal)
+{
+ jamEntry();
+
+ Uint32 origSenderRef = signal->senderBlockRef();
+
+ OpSubEventPtr subbPtr;
+ if (!c_opSubEvent.seize(subbPtr)) {
+ SubStopRef * ref = (SubStopRef *)signal->getDataPtrSend();
+ jam();
+ // ret->setErrorCode(SubStartRef::SeizeError);
+ // ret->setErrorLine(__LINE__);
+ // ret->setErrorNode(reference());
+ ref->senderRef = reference();
+ ref->setTemporary(SubStopRef::Busy);
+
+ sendSignal(origSenderRef, GSN_SUB_STOP_REF, signal,
+ SubStopRef::SignalLength, JBB);
+ return;
+ }
+
+ {
+ const SubStopReq* req = (SubStopReq*) signal->getDataPtr();
+ subbPtr.p->m_senderRef = req->senderRef;
+ subbPtr.p->m_senderData = req->senderData;
+ subbPtr.p->m_errorCode = 0;
+ }
+
+ if (refToBlock(origSenderRef) != DBDICT) {
+ /*
+ * Coordinator
+ */
+ jam();
+#ifdef EVENT_DEBUG
+ ndbout_c("SUB_STOP_REQ 1");
+#endif
+ subbPtr.p->m_senderRef = origSenderRef; // not sure if API sets correctly
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ subbPtr.p->m_reqTracker.init<SubStopRef>(c_counterMgr, rg, GSN_SUB_STOP_REF, subbPtr.i);
+
+ SubStopReq* req = (SubStopReq*) signal->getDataPtrSend();
+
+ req->senderRef = reference();
+ req->senderData = subbPtr.i;
+
+ sendSignal(rg, GSN_SUB_STOP_REQ, signal, SubStopReq::SignalLength, JBB);
+ return;
+ }
+ /*
+ * Participant
+ */
+#ifdef EVENT_DEBUG
+ ndbout_c("SUB_STOP_REQ 2");
+#endif
+ ndbrequire(refToBlock(origSenderRef) == DBDICT);
+ {
+ SubStopReq* req = (SubStopReq*) signal->getDataPtrSend();
+
+ req->senderRef = reference();
+ req->senderData = subbPtr.i;
+
+ sendSignal(SUMA_REF, GSN_SUB_STOP_REQ, signal, SubStopReq::SignalLength, JBB);
+ }
+}
+
+void Dbdict::execSUB_STOP_REF(Signal* signal)
+{
+ jamEntry();
+ const SubStopRef* ref = (SubStopRef*) signal->getDataPtr();
+ Uint32 senderRef = ref->senderRef;
+
+ OpSubEventPtr subbPtr;
+ c_opSubEvent.getPtr(subbPtr, ref->senderData);
+
+ if (refToBlock(senderRef) == SUMA) {
+ /*
+ * Participant
+ */
+ jam();
+ if (ref->isTemporary()){
+ jam();
+ SubStopReq* req = (SubStopReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = subbPtr.i;
+ sendSignal(SUMA_REF, GSN_SUB_STOP_REQ,
+ signal, SubStopReq::SignalLength, JBB);
+ } else {
+ jam();
+ SubStopRef* ref = (SubStopRef*) signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = subbPtr.p->m_senderData;
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_STOP_REF,
+ signal, SubStopRef::SignalLength, JBB);
+ c_opSubEvent.release(subbPtr);
+ }
+ return;
+ }
+ /*
+ * Coordinator
+ */
+ ndbrequire(refToBlock(senderRef) == DBDICT);
+ if (ref->errorCode == SubStopRef::NF_FakeErrorREF){
+ jam();
+ subbPtr.p->m_reqTracker.ignoreRef(c_counterMgr, refToNode(senderRef));
+ } else {
+ jam();
+ subbPtr.p->m_reqTracker.reportRef(c_counterMgr, refToNode(senderRef));
+ }
+ completeSubStopReq(signal,subbPtr.i,0);
+}
+
+void Dbdict::execSUB_STOP_CONF(Signal* signal)
+{
+ jamEntry();
+
+ const SubStopConf* conf = (SubStopConf*) signal->getDataPtr();
+ Uint32 senderRef = conf->senderRef;
+
+ OpSubEventPtr subbPtr;
+ c_opSubEvent.getPtr(subbPtr, conf->senderData);
+
+ if (refToBlock(senderRef) == SUMA) {
+ /*
+ * Participant
+ */
+ jam();
+ SubStopConf* conf = (SubStopConf*) signal->getDataPtrSend();
+
+ conf->senderRef = reference();
+ conf->senderData = subbPtr.p->m_senderData;
+
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_STOP_CONF,
+ signal, SubStopConf::SignalLength, JBB);
+ c_opSubEvent.release(subbPtr);
+ return;
+ }
+ /*
+ * Coordinator
+ */
+ ndbrequire(refToBlock(senderRef) == DBDICT);
+ subbPtr.p->m_reqTracker.reportConf(c_counterMgr, refToNode(senderRef));
+ completeSubStopReq(signal,subbPtr.i,0);
+}
+
+/*
+ * Coordinator
+ */
+void Dbdict::completeSubStopReq(Signal* signal,
+ Uint32 ptrI,
+ Uint32 returnCode){
+ OpSubEventPtr subbPtr;
+ c_opSubEvent.getPtr(subbPtr, ptrI);
+
+ if (!subbPtr.p->m_reqTracker.done()){
+ jam();
+ return;
+ }
+
+ if (subbPtr.p->m_reqTracker.hasRef()) {
+ jam();
+#ifdef EVENT_DEBUG
+ ndbout_c("SUB_STOP_REF");
+#endif
+ SubStopRef* ref = (SubStopRef*)signal->getDataPtrSend();
+
+ ref->senderRef = reference();
+ ref->senderData = subbPtr.p->m_senderData;
+ /*
+ ref->subscriptionId = subbPtr.p->m_senderData;
+ ref->subscriptionKey = subbPtr.p->m_senderData;
+ ref->part = subbPtr.p->m_part; // SubscriptionData::Part
+ ref->subscriberData = subbPtr.p->m_subscriberData;
+ ref->subscriberRef = subbPtr.p->m_subscriberRef;
+ */
+ ref->errorCode = subbPtr.p->m_errorCode;
+
+
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_STOP_REF,
+ signal, SubStopRef::SignalLength, JBB);
+ if (subbPtr.p->m_reqTracker.hasConf()) {
+ // stopStartedNodes(signal);
+ }
+ c_opSubEvent.release(subbPtr);
+ return;
+ }
+#ifdef EVENT_DEBUG
+ ndbout_c("SUB_STOP_CONF");
+#endif
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_STOP_CONF,
+ signal, SubStopConf::SignalLength, JBB);
+ c_opSubEvent.release(subbPtr);
+}
+
+/***************************************************************
+ * MODULE: Drop event.
+ *
+ * Drop event.
+ *
+ * TODO
+ */
+
+void
+Dbdict::execDROP_EVNT_REQ(Signal* signal)
+{
+ jamEntry();
+ EVENT_TRACE;
+
+ DropEvntReq *req = (DropEvntReq*)signal->getDataPtr();
+ const Uint32 senderRef = signal->senderBlockRef();
+ OpDropEventPtr evntRecPtr;
+
+ // Seize a Create Event record
+ if (!c_opDropEvent.seize(evntRecPtr)) {
+ // Failed to allocate event record
+ jam();
+ releaseSections(signal);
+
+ DropEvntRef * ret = (DropEvntRef *)signal->getDataPtrSend();
+ ret->setErrorCode(DropEvntRef::SeizeError);
+ ret->setErrorLine(__LINE__);
+ ret->setErrorNode(reference());
+ sendSignal(senderRef, GSN_DROP_EVNT_REF, signal,
+ DropEvntRef::SignalLength, JBB);
+ return;
+ }
+
+#ifdef EVENT_DEBUG
+ ndbout_c("DBDICT::execDROP_EVNT_REQ evntRecId = (%d)", evntRecPtr.i);
+#endif
+
+ OpDropEvent* evntRec = evntRecPtr.p;
+ evntRec->init(req);
+
+ SegmentedSectionPtr ssPtr;
+
+ signal->getSection(ssPtr, 0);
+
+ SimplePropertiesSectionReader r0(ssPtr, getSectionSegmentPool());
+#ifdef EVENT_DEBUG
+ r0.printAll(ndbout);
+#endif
+ // event name
+ if ((!r0.first()) ||
+ (r0.getValueType() != SimpleProperties::StringValue) ||
+ (r0.getValueLen() <= 0)) {
+ jam();
+ releaseSections(signal);
+
+ evntRecPtr.p->m_errorCode = DropEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorNode = reference();
+
+ dropEvent_sendReply(signal, evntRecPtr);
+ return;
+ }
+ r0.getString(evntRecPtr.p->m_eventRec.NAME);
+ {
+ int len = strlen(evntRecPtr.p->m_eventRec.NAME);
+ memset(evntRecPtr.p->m_eventRec.NAME+len, 0, MAX_TAB_NAME_SIZE-len);
+#ifdef EVENT_DEBUG
+ printf("DropEvntReq; EventName %s, len %u\n",
+ evntRecPtr.p->m_eventRec.NAME, len);
+ for(int i = 0; i < MAX_TAB_NAME_SIZE/4; i++)
+ printf("H'%.8x ", ((Uint32*)evntRecPtr.p->m_eventRec.NAME)[i]);
+ printf("\n");
+#endif
+ }
+
+ releaseSections(signal);
+
+ Callback c = { safe_cast(&Dbdict::dropEventUTIL_PREPARE_READ), 0 };
+
+ prepareTransactionEventSysTable(&c, signal, evntRecPtr.i,
+ UtilPrepareReq::Read);
+}
+
+void
+Dbdict::dropEventUTIL_PREPARE_READ(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ EVENT_TRACE;
+ if (returnCode != 0) {
+ EVENT_TRACE;
+ dropEventUtilPrepareRef(signal, callbackData, returnCode);
+ return;
+ }
+
+ UtilPrepareConf* const req = (UtilPrepareConf*)signal->getDataPtr();
+ OpDropEventPtr evntRecPtr;
+ evntRecPtr.i = req->getSenderData();
+ const Uint32 prepareId = req->getPrepareId();
+
+ ndbrequire((evntRecPtr.p = c_opDropEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ Callback c = { safe_cast(&Dbdict::dropEventUTIL_EXECUTE_READ), 0 };
+
+ executeTransEventSysTable(&c, signal,
+ evntRecPtr.i, evntRecPtr.p->m_eventRec,
+ prepareId, UtilPrepareReq::Read);
+}
+
+void
+Dbdict::dropEventUTIL_EXECUTE_READ(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ EVENT_TRACE;
+ if (returnCode != 0) {
+ EVENT_TRACE;
+ dropEventUtilExecuteRef(signal, callbackData, returnCode);
+ return;
+ }
+
+ OpDropEventPtr evntRecPtr;
+ UtilExecuteConf * const ref = (UtilExecuteConf *)signal->getDataPtr();
+ jam();
+ evntRecPtr.i = ref->getSenderData();
+ ndbrequire((evntRecPtr.p = c_opDropEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ parseReadEventSys(signal, evntRecPtr.p->m_eventRec);
+
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ evntRecPtr.p->m_reqTracker.init<SubRemoveRef>(c_counterMgr, rg, GSN_SUB_REMOVE_REF,
+ evntRecPtr.i);
+
+ SubRemoveReq* req = (SubRemoveReq*) signal->getDataPtrSend();
+
+ req->senderRef = reference();
+ req->senderData = evntRecPtr.i;
+ req->subscriptionId = evntRecPtr.p->m_eventRec.SUBID;
+ req->subscriptionKey = evntRecPtr.p->m_eventRec.SUBKEY;
+
+ sendSignal(rg, GSN_SUB_REMOVE_REQ, signal, SubRemoveReq::SignalLength, JBB);
+}
+
+/*
+ * Participant
+ */
+
+void
+Dbdict::execSUB_REMOVE_REQ(Signal* signal)
+{
+ jamEntry();
+
+ Uint32 origSenderRef = signal->senderBlockRef();
+
+ OpSubEventPtr subbPtr;
+ if (!c_opSubEvent.seize(subbPtr)) {
+ SubRemoveRef * ref = (SubRemoveRef *)signal->getDataPtrSend();
+ jam();
+ ref->senderRef = reference();
+ ref->setTemporary(SubRemoveRef::Busy);
+
+ sendSignal(origSenderRef, GSN_SUB_REMOVE_REF, signal,
+ SubRemoveRef::SignalLength, JBB);
+ return;
+ }
+
+ {
+ const SubRemoveReq* req = (SubRemoveReq*) signal->getDataPtr();
+ subbPtr.p->m_senderRef = req->senderRef;
+ subbPtr.p->m_senderData = req->senderData;
+ subbPtr.p->m_errorCode = 0;
+ }
+
+ SubRemoveReq* req = (SubRemoveReq*) signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = subbPtr.i;
+
+ sendSignal(SUMA_REF, GSN_SUB_REMOVE_REQ, signal, SubRemoveReq::SignalLength, JBB);
+}
+
+/*
+ * Coordintor/Participant
+ */
+
+void
+Dbdict::execSUB_REMOVE_REF(Signal* signal)
+{
+ jamEntry();
+ const SubRemoveRef* ref = (SubRemoveRef*) signal->getDataPtr();
+ Uint32 senderRef = ref->senderRef;
+
+ if (refToBlock(senderRef) == SUMA) {
+ /*
+ * Participant
+ */
+ jam();
+ OpSubEventPtr subbPtr;
+ c_opSubEvent.getPtr(subbPtr, ref->senderData);
+ if (ref->errorCode == (Uint32) GrepError::SUBSCRIPTION_ID_NOT_FOUND) {
+ // conf this since this may occur if a nodefailiure has occured
+ // earlier so that the systable was not cleared
+ SubRemoveConf* conf = (SubRemoveConf*) signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = subbPtr.p->m_senderData;
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_REMOVE_CONF,
+ signal, SubRemoveConf::SignalLength, JBB);
+ } else {
+ SubRemoveRef* ref = (SubRemoveRef*) signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = subbPtr.p->m_senderData;
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_REMOVE_REF,
+ signal, SubRemoveRef::SignalLength, JBB);
+ }
+ c_opSubEvent.release(subbPtr);
+ return;
+ }
+ /*
+ * Coordinator
+ */
+ ndbrequire(refToBlock(senderRef) == DBDICT);
+ OpDropEventPtr eventRecPtr;
+ c_opDropEvent.getPtr(eventRecPtr, ref->senderData);
+ if (ref->errorCode == SubRemoveRef::NF_FakeErrorREF){
+ jam();
+ eventRecPtr.p->m_reqTracker.ignoreRef(c_counterMgr, refToNode(senderRef));
+ } else {
+ jam();
+ eventRecPtr.p->m_reqTracker.reportRef(c_counterMgr, refToNode(senderRef));
+ }
+ completeSubRemoveReq(signal,eventRecPtr.i,0);
+}
+
+void
+Dbdict::execSUB_REMOVE_CONF(Signal* signal)
+{
+ jamEntry();
+ const SubRemoveConf* conf = (SubRemoveConf*) signal->getDataPtr();
+ Uint32 senderRef = conf->senderRef;
+
+ if (refToBlock(senderRef) == SUMA) {
+ /*
+ * Participant
+ */
+ jam();
+ OpSubEventPtr subbPtr;
+ c_opSubEvent.getPtr(subbPtr, conf->senderData);
+ SubRemoveConf* conf = (SubRemoveConf*) signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = subbPtr.p->m_senderData;
+ sendSignal(subbPtr.p->m_senderRef, GSN_SUB_REMOVE_CONF,
+ signal, SubRemoveConf::SignalLength, JBB);
+ c_opSubEvent.release(subbPtr);
+ return;
+ }
+ /*
+ * Coordinator
+ */
+ ndbrequire(refToBlock(senderRef) == DBDICT);
+ OpDropEventPtr eventRecPtr;
+ c_opDropEvent.getPtr(eventRecPtr, conf->senderData);
+ eventRecPtr.p->m_reqTracker.reportConf(c_counterMgr, refToNode(senderRef));
+ completeSubRemoveReq(signal,eventRecPtr.i,0);
+}
+
+void
+Dbdict::completeSubRemoveReq(Signal* signal, Uint32 ptrI, Uint32 xxx)
+{
+ OpDropEventPtr evntRecPtr;
+ c_opDropEvent.getPtr(evntRecPtr, ptrI);
+
+ if (!evntRecPtr.p->m_reqTracker.done()){
+ jam();
+ return;
+ }
+
+ if (evntRecPtr.p->m_reqTracker.hasRef()) {
+ jam();
+ evntRecPtr.p->m_errorNode = reference();
+ evntRecPtr.p->m_errorLine = __LINE__;
+ evntRecPtr.p->m_errorCode = DropEvntRef::Undefined;
+ dropEvent_sendReply(signal, evntRecPtr);
+ return;
+ }
+
+ Callback c = { safe_cast(&Dbdict::dropEventUTIL_PREPARE_DELETE), 0 };
+
+ prepareTransactionEventSysTable(&c, signal, evntRecPtr.i,
+ UtilPrepareReq::Delete);
+}
+
+void
+Dbdict::dropEventUTIL_PREPARE_DELETE(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ EVENT_TRACE;
+ if (returnCode != 0) {
+ EVENT_TRACE;
+ dropEventUtilPrepareRef(signal, callbackData, returnCode);
+ return;
+ }
+
+ UtilPrepareConf* const req = (UtilPrepareConf*)signal->getDataPtr();
+ OpDropEventPtr evntRecPtr;
+ jam();
+ evntRecPtr.i = req->getSenderData();
+ const Uint32 prepareId = req->getPrepareId();
+
+ ndbrequire((evntRecPtr.p = c_opDropEvent.getPtr(evntRecPtr.i)) != NULL);
+#ifdef EVENT_DEBUG
+ printf("DropEvntUTIL_PREPARE; evntRecPtr.i len %u\n",evntRecPtr.i);
+#endif
+
+ Callback c = { safe_cast(&Dbdict::dropEventUTIL_EXECUTE_DELETE), 0 };
+
+ executeTransEventSysTable(&c, signal,
+ evntRecPtr.i, evntRecPtr.p->m_eventRec,
+ prepareId, UtilPrepareReq::Delete);
+}
+
+void
+Dbdict::dropEventUTIL_EXECUTE_DELETE(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ EVENT_TRACE;
+ if (returnCode != 0) {
+ EVENT_TRACE;
+ dropEventUtilExecuteRef(signal, callbackData, returnCode);
+ return;
+ }
+
+ OpDropEventPtr evntRecPtr;
+ UtilExecuteConf * const ref = (UtilExecuteConf *)signal->getDataPtr();
+ jam();
+ evntRecPtr.i = ref->getSenderData();
+ ndbrequire((evntRecPtr.p = c_opDropEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ dropEvent_sendReply(signal, evntRecPtr);
+}
+
+void
+Dbdict::dropEventUtilPrepareRef(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ EVENT_TRACE;
+ UtilPrepareRef * const ref = (UtilPrepareRef *)signal->getDataPtr();
+ OpDropEventPtr evntRecPtr;
+ evntRecPtr.i = ref->getSenderData();
+ ndbrequire((evntRecPtr.p = c_opDropEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ bool temporary = false;
+ interpretUtilPrepareErrorCode((UtilPrepareRef::ErrorCode)ref->getErrorCode(),
+ temporary, evntRecPtr.p->m_errorLine);
+ if (temporary) {
+ evntRecPtr.p->m_errorCode = (DropEvntRef::ErrorCode)
+ ((Uint32) DropEvntRef::Undefined | (Uint32) DropEvntRef::Temporary);
+ }
+
+ if (evntRecPtr.p->m_errorCode == 0) {
+ evntRecPtr.p->m_errorCode = DropEvntRef::Undefined;
+ evntRecPtr.p->m_errorLine = __LINE__;
+ }
+ evntRecPtr.p->m_errorNode = reference();
+
+ dropEvent_sendReply(signal, evntRecPtr);
+}
+
+void
+Dbdict::dropEventUtilExecuteRef(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode)
+{
+ jam();
+ EVENT_TRACE;
+ OpDropEventPtr evntRecPtr;
+ UtilExecuteRef * const ref = (UtilExecuteRef *)signal->getDataPtr();
+ jam();
+ evntRecPtr.i = ref->getSenderData();
+ ndbrequire((evntRecPtr.p = c_opDropEvent.getPtr(evntRecPtr.i)) != NULL);
+
+ evntRecPtr.p->m_errorNode = reference();
+ evntRecPtr.p->m_errorLine = __LINE__;
+
+ switch (ref->getErrorCode()) {
+ case UtilExecuteRef::TCError:
+ switch (ref->getTCErrorCode()) {
+ case ZNOT_FOUND:
+ jam();
+ evntRecPtr.p->m_errorCode = DropEvntRef::EventNotFound;
+ break;
+ default:
+ jam();
+ evntRecPtr.p->m_errorCode = DropEvntRef::UndefinedTCError;
+ break;
+ }
+ break;
+ default:
+ jam();
+ evntRecPtr.p->m_errorCode = DropEvntRef::Undefined;
+ break;
+ }
+ dropEvent_sendReply(signal, evntRecPtr);
+}
+
+void Dbdict::dropEvent_sendReply(Signal* signal,
+ OpDropEventPtr evntRecPtr)
+{
+ jam();
+ EVENT_TRACE;
+ Uint32 senderRef = evntRecPtr.p->m_request.getUserRef();
+
+ if (evntRecPtr.p->hasError()) {
+ jam();
+ DropEvntRef * ret = (DropEvntRef *)signal->getDataPtrSend();
+
+ ret->setUserData(evntRecPtr.p->m_request.getUserData());
+ ret->setUserRef(evntRecPtr.p->m_request.getUserRef());
+
+ ret->setErrorCode(evntRecPtr.p->m_errorCode);
+ ret->setErrorLine(evntRecPtr.p->m_errorLine);
+ ret->setErrorNode(evntRecPtr.p->m_errorNode);
+
+ sendSignal(senderRef, GSN_DROP_EVNT_REF, signal,
+ DropEvntRef::SignalLength, JBB);
+ } else {
+ jam();
+ DropEvntConf * evntConf = (DropEvntConf *)signal->getDataPtrSend();
+
+ evntConf->setUserData(evntRecPtr.p->m_request.getUserData());
+ evntConf->setUserRef(evntRecPtr.p->m_request.getUserRef());
+
+ sendSignal(senderRef, GSN_DROP_EVNT_CONF, signal,
+ DropEvntConf::SignalLength, JBB);
+ }
+
+ c_opDropEvent.release(evntRecPtr);
+}
+
+/**
+ * MODULE: Alter index
+ *
+ * Alter index state. Alter online creates the index in each TC and
+ * then invokes create trigger and alter trigger protocols to activate
+ * the 3 triggers. Alter offline does the opposite.
+ *
+ * Request type received in REQ and returned in CONF/REF:
+ *
+ * RT_USER - from API to DICT master
+ * RT_CREATE_INDEX - part of create index operation
+ * RT_DROP_INDEX - part of drop index operation
+ * RT_NODERESTART - node restart, activate locally only
+ * RT_SYSTEMRESTART - system restart, activate and build if not logged
+ * RT_DICT_PREPARE - prepare participants
+ * RT_DICT_TC - to local TC via each participant
+ * RT_DICT_COMMIT - commit in each participant
+ */
+
+void
+Dbdict::execALTER_INDX_REQ(Signal* signal)
+{
+ jamEntry();
+ AlterIndxReq* const req = (AlterIndxReq*)signal->getDataPtrSend();
+ OpAlterIndexPtr opPtr;
+ const Uint32 senderRef = signal->senderBlockRef();
+ const AlterIndxReq::RequestType requestType = req->getRequestType();
+ if (requestType == AlterIndxReq::RT_USER ||
+ requestType == AlterIndxReq::RT_CREATE_INDEX ||
+ requestType == AlterIndxReq::RT_DROP_INDEX ||
+ requestType == AlterIndxReq::RT_NODERESTART ||
+ requestType == AlterIndxReq::RT_SYSTEMRESTART) {
+ jam();
+ const bool isLocal = req->getRequestFlag() & RequestFlag::RF_LOCAL;
+ NdbNodeBitmask receiverNodes = c_aliveNodes;
+ if (isLocal) {
+ receiverNodes.clear();
+ receiverNodes.set(getOwnNodeId());
+ }
+ if (signal->getLength() == AlterIndxReq::SignalLength) {
+ jam();
+ if (! isLocal && getOwnNodeId() != c_masterNodeId) {
+ jam();
+ // forward to DICT master
+ sendSignal(calcDictBlockRef(c_masterNodeId), GSN_ALTER_INDX_REQ,
+ signal, signal->getLength(), JBB);
+ return;
+ }
+ // forward initial request plus operation key to all
+ req->setOpKey(++c_opRecordSequence);
+ NodeReceiverGroup rg(DBDICT, receiverNodes);
+ sendSignal(rg, GSN_ALTER_INDX_REQ,
+ signal, AlterIndxReq::SignalLength + 1, JBB);
+ return;
+ }
+ // seize operation record
+ ndbrequire(signal->getLength() == AlterIndxReq::SignalLength + 1);
+ const Uint32 opKey = req->getOpKey();
+ OpAlterIndex opBusy;
+ if (! c_opAlterIndex.seize(opPtr))
+ opPtr.p = &opBusy;
+ opPtr.p->save(req);
+ opPtr.p->m_coordinatorRef = senderRef;
+ opPtr.p->m_isMaster = (senderRef == reference());
+ opPtr.p->key = opKey;
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_PREPARE;
+ if (opPtr.p == &opBusy) {
+ jam();
+ opPtr.p->m_errorCode = AlterIndxRef::Busy;
+ opPtr.p->m_errorLine = __LINE__;
+ alterIndex_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ c_opAlterIndex.add(opPtr);
+ // master expects to hear from all
+ if (opPtr.p->m_isMaster)
+ opPtr.p->m_signalCounter = receiverNodes;
+ // check request in all participants
+ alterIndex_slavePrepare(signal, opPtr);
+ alterIndex_sendReply(signal, opPtr, false);
+ return;
+ }
+ c_opAlterIndex.find(opPtr, req->getConnectionPtr());
+ if (! opPtr.isNull()) {
+ opPtr.p->m_requestType = requestType;
+ if (requestType == AlterIndxReq::RT_DICT_TC) {
+ jam();
+ if (opPtr.p->m_request.getOnline())
+ alterIndex_toCreateTc(signal, opPtr);
+ else
+ alterIndex_toDropTc(signal, opPtr);
+ return;
+ }
+ if (requestType == AlterIndxReq::RT_DICT_COMMIT ||
+ requestType == AlterIndxReq::RT_DICT_ABORT) {
+ jam();
+ if (requestType == AlterIndxReq::RT_DICT_COMMIT)
+ alterIndex_slaveCommit(signal, opPtr);
+ else
+ alterIndex_slaveAbort(signal, opPtr);
+ alterIndex_sendReply(signal, opPtr, false);
+ // done in slave
+ if (! opPtr.p->m_isMaster)
+ c_opAlterIndex.release(opPtr);
+ return;
+ }
+ }
+ jam();
+ // return to sender
+ OpAlterIndex opBad;
+ opPtr.p = &opBad;
+ opPtr.p->save(req);
+ opPtr.p->m_errorCode = AlterIndxRef::BadRequestType;
+ opPtr.p->m_errorLine = __LINE__;
+ alterIndex_sendReply(signal, opPtr, true);
+}
+
+void
+Dbdict::execALTER_INDX_CONF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(signal->getNoOfSections() == 0);
+ AlterIndxConf* conf = (AlterIndxConf*)signal->getDataPtrSend();
+ alterIndex_recvReply(signal, conf, 0);
+}
+
+void
+Dbdict::execALTER_INDX_REF(Signal* signal)
+{
+ jamEntry();
+ AlterIndxRef* ref = (AlterIndxRef*)signal->getDataPtrSend();
+ alterIndex_recvReply(signal, ref->getConf(), ref);
+}
+
+void
+Dbdict::alterIndex_recvReply(Signal* signal, const AlterIndxConf* conf,
+ const AlterIndxRef* ref)
+{
+ jam();
+ const Uint32 senderRef = signal->senderBlockRef();
+ const AlterIndxReq::RequestType requestType = conf->getRequestType();
+ const Uint32 key = conf->getConnectionPtr();
+ if (requestType == AlterIndxReq::RT_CREATE_INDEX) {
+ jam();
+ // part of create index operation
+ OpCreateIndexPtr opPtr;
+ c_opCreateIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ createIndex_fromAlterIndex(signal, opPtr);
+ return;
+ }
+ if (requestType == AlterIndxReq::RT_DROP_INDEX) {
+ jam();
+ // part of drop index operation
+ OpDropIndexPtr opPtr;
+ c_opDropIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ dropIndex_fromAlterIndex(signal, opPtr);
+ return;
+ }
+ if (requestType == AlterIndxReq::RT_TC ||
+ requestType == AlterIndxReq::RT_TUX) {
+ jam();
+ // part of build index operation
+ OpBuildIndexPtr opPtr;
+ c_opBuildIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ buildIndex_fromOnline(signal, opPtr);
+ return;
+ }
+ if (requestType == AlterIndxReq::RT_NODERESTART) {
+ jam();
+ if (ref == 0) {
+ infoEvent("DICT: index %u activated", (unsigned)key);
+ } else {
+ warningEvent("DICT: index %u activation failed: code=%d line=%d",
+ (unsigned)key,
+ ref->getErrorCode(), ref->getErrorLine());
+ }
+ activateIndexes(signal, key + 1);
+ return;
+ }
+ if (requestType == AlterIndxReq::RT_SYSTEMRESTART) {
+ jam();
+ if (ref == 0) {
+ infoEvent("DICT: index %u activated done", (unsigned)key);
+ } else {
+ warningEvent("DICT: index %u activated failed: code=%d line=%d node=%d",
+ (unsigned)key,
+ ref->getErrorCode(), ref->getErrorLine(), ref->getErrorNode());
+ }
+ activateIndexes(signal, key + 1);
+ return;
+ }
+ OpAlterIndexPtr opPtr;
+ c_opAlterIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ ndbrequire(opPtr.p->m_isMaster);
+ ndbrequire(opPtr.p->m_requestType == requestType);
+ opPtr.p->setError(ref);
+ opPtr.p->m_signalCounter.clearWaitingFor(refToNode(senderRef));
+ if (! opPtr.p->m_signalCounter.done()) {
+ jam();
+ return;
+ }
+ if (requestType == AlterIndxReq::RT_DICT_COMMIT ||
+ requestType == AlterIndxReq::RT_DICT_ABORT) {
+ jam();
+ // send reply to user
+ alterIndex_sendReply(signal, opPtr, true);
+ c_opAlterIndex.release(opPtr);
+ return;
+ }
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_ABORT;
+ alterIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ if (indexPtr.p->isHashIndex()) {
+ if (requestType == AlterIndxReq::RT_DICT_PREPARE) {
+ jam();
+ if (opPtr.p->m_request.getOnline()) {
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_TC;
+ alterIndex_sendSlaveReq(signal, opPtr);
+ } else {
+ // start drop triggers
+ alterIndex_toDropTrigger(signal, opPtr);
+ }
+ return;
+ }
+ if (requestType == AlterIndxReq::RT_DICT_TC) {
+ jam();
+ if (opPtr.p->m_request.getOnline()) {
+ // start create triggers
+ alterIndex_toCreateTrigger(signal, opPtr);
+ } else {
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_COMMIT;
+ alterIndex_sendSlaveReq(signal, opPtr);
+ }
+ return;
+ }
+ }
+ if (indexPtr.p->isOrderedIndex()) {
+ if (requestType == AlterIndxReq::RT_DICT_PREPARE) {
+ jam();
+ if (opPtr.p->m_request.getOnline()) {
+ // start create triggers
+ alterIndex_toCreateTrigger(signal, opPtr);
+ } else {
+ // start drop triggers
+ alterIndex_toDropTrigger(signal, opPtr);
+ }
+ return;
+ }
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::alterIndex_slavePrepare(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ const AlterIndxReq* const req = &opPtr.p->m_request;
+ if (! (req->getIndexId() < c_tableRecordPool.getSize())) {
+ jam();
+ opPtr.p->m_errorCode = AlterIndxRef::Inconsistency;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, req->getIndexId());
+ if (indexPtr.p->tabState != TableRecord::DEFINED) {
+ jam();
+ opPtr.p->m_errorCode = AlterIndxRef::IndexNotFound;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ if (! indexPtr.p->isIndex()) {
+ jam();
+ opPtr.p->m_errorCode = AlterIndxRef::NotAnIndex;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ if (req->getOnline())
+ indexPtr.p->indexState = TableRecord::IS_BUILDING;
+ else
+ indexPtr.p->indexState = TableRecord::IS_DROPPING;
+}
+
+void
+Dbdict::alterIndex_toCreateTc(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ // request to create index in local TC
+ CreateIndxReq* const req = (CreateIndxReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(CreateIndxReq::RT_TC);
+ req->setIndexType(indexPtr.p->tableType);
+ req->setTableId(indexPtr.p->primaryTableId);
+ req->setIndexId(indexPtr.i);
+ req->setOnline(true);
+ getIndexAttrList(indexPtr, opPtr.p->m_attrList);
+ // send
+ LinearSectionPtr lsPtr[3];
+ lsPtr[0].p = (Uint32*)&opPtr.p->m_attrList;
+ lsPtr[0].sz = 1 + opPtr.p->m_attrList.sz;
+ sendSignal(calcTcBlockRef(getOwnNodeId()), GSN_CREATE_INDX_REQ,
+ signal, CreateIndxReq::SignalLength, JBB, lsPtr, 1);
+}
+
+void
+Dbdict::alterIndex_fromCreateTc(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ // mark created in local TC
+ if (! opPtr.p->hasError()) {
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ indexPtr.p->indexLocal |= TableRecord::IL_CREATED_TC;
+ }
+ // forward CONF or REF to master
+ ndbrequire(opPtr.p->m_requestType == AlterIndxReq::RT_DICT_TC);
+ alterIndex_sendReply(signal, opPtr, false);
+}
+
+void
+Dbdict::alterIndex_toDropTc(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ // broken index
+ if (! (indexPtr.p->indexLocal & TableRecord::IL_CREATED_TC)) {
+ jam();
+ alterIndex_sendReply(signal, opPtr, false);
+ return;
+ }
+ // request to drop in local TC
+ DropIndxReq* const req = (DropIndxReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(DropIndxReq::RT_TC);
+ req->setTableId(indexPtr.p->primaryTableId);
+ req->setIndexId(indexPtr.i);
+ req->setIndexVersion(indexPtr.p->tableVersion);
+ // send
+ sendSignal(calcTcBlockRef(getOwnNodeId()), GSN_DROP_INDX_REQ,
+ signal, DropIndxReq::SignalLength, JBB);
+}
+
+void
+Dbdict::alterIndex_fromDropTc(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ ndbrequire(opPtr.p->m_requestType == AlterIndxReq::RT_DICT_TC);
+ if (! opPtr.p->hasError()) {
+ // mark dropped in local TC
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ indexPtr.p->indexLocal &= ~TableRecord::IL_CREATED_TC;
+ }
+ // forward CONF or REF to master
+ alterIndex_sendReply(signal, opPtr, false);
+}
+
+void
+Dbdict::alterIndex_toCreateTrigger(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ // start creation of index triggers
+ CreateTrigReq* const req = (CreateTrigReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(CreateTrigReq::RT_ALTER_INDEX);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ req->setTableId(opPtr.p->m_request.getTableId());
+ req->setIndexId(opPtr.p->m_request.getIndexId());
+ req->setTriggerId(RNIL);
+ req->setTriggerActionTime(TriggerActionTime::TA_AFTER);
+ req->setMonitorAllAttributes(false);
+ req->setOnline(true); // alter online after create
+ req->setReceiverRef(0); // implicit for index triggers
+ getIndexAttrMask(indexPtr, req->getAttributeMask());
+ // name section
+ char triggerName[MAX_TAB_NAME_SIZE];
+ Uint32 buffer[2 + ((MAX_TAB_NAME_SIZE + 3) >> 2)]; // SP string
+ LinearWriter w(buffer, sizeof(buffer) >> 2);
+ LinearSectionPtr lsPtr[3];
+ if (indexPtr.p->isHashIndex()) {
+ req->setTriggerType(TriggerType::SECONDARY_INDEX);
+ req->setMonitorReplicas(false);
+ // insert
+ if (opPtr.p->m_requestFlag & RequestFlag::RF_LOCAL)
+ req->setTriggerId(indexPtr.p->insertTriggerId);
+ req->setTriggerEvent(TriggerEvent::TE_INSERT);
+ sprintf(triggerName, "NDB$INDEX_%u_INSERT", opPtr.p->m_request.getIndexId());
+ w.reset();
+ w.add(CreateTrigReq::TriggerNameKey, triggerName);
+ lsPtr[0].p = buffer;
+ lsPtr[0].sz = w.getWordsUsed();
+ sendSignal(reference(), GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength, JBB, lsPtr, 1);
+ // update
+ if (opPtr.p->m_requestFlag & RequestFlag::RF_LOCAL)
+ req->setTriggerId(indexPtr.p->updateTriggerId);
+ req->setTriggerEvent(TriggerEvent::TE_UPDATE);
+ sprintf(triggerName, "NDB$INDEX_%u_UPDATE", opPtr.p->m_request.getIndexId());
+ w.reset();
+ w.add(CreateTrigReq::TriggerNameKey, triggerName);
+ lsPtr[0].p = buffer;
+ lsPtr[0].sz = w.getWordsUsed();
+ sendSignal(reference(), GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength, JBB, lsPtr, 1);
+ // delete
+ if (opPtr.p->m_requestFlag & RequestFlag::RF_LOCAL)
+ req->setTriggerId(indexPtr.p->deleteTriggerId);
+ req->setTriggerEvent(TriggerEvent::TE_DELETE);
+ sprintf(triggerName, "NDB$INDEX_%u_DELETE", opPtr.p->m_request.getIndexId());
+ w.reset();
+ w.add(CreateTrigReq::TriggerNameKey, triggerName);
+ lsPtr[0].p = buffer;
+ lsPtr[0].sz = w.getWordsUsed();
+ sendSignal(reference(), GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength, JBB, lsPtr, 1);
+ // triggers left to create
+ opPtr.p->m_triggerCounter = 3;
+ return;
+ }
+ if (indexPtr.p->isOrderedIndex()) {
+ req->addRequestFlag(RequestFlag::RF_NOTCTRIGGER);
+ req->setTriggerType(TriggerType::ORDERED_INDEX);
+ req->setTriggerActionTime(TriggerActionTime::TA_CUSTOM);
+ req->setMonitorReplicas(true);
+ // one trigger for 5 events (insert, update, delete, commit, abort)
+ if (opPtr.p->m_requestFlag & RequestFlag::RF_LOCAL)
+ req->setTriggerId(indexPtr.p->customTriggerId);
+ req->setTriggerEvent(TriggerEvent::TE_CUSTOM);
+ sprintf(triggerName, "NDB$INDEX_%u_CUSTOM", opPtr.p->m_request.getIndexId());
+ w.reset();
+ w.add(CreateTrigReq::TriggerNameKey, triggerName);
+ lsPtr[0].p = buffer;
+ lsPtr[0].sz = w.getWordsUsed();
+ sendSignal(reference(), GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength, JBB, lsPtr, 1);
+ // triggers left to create
+ opPtr.p->m_triggerCounter = 1;
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::alterIndex_fromCreateTrigger(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ ndbrequire(opPtr.p->m_triggerCounter != 0);
+ if (--opPtr.p->m_triggerCounter != 0) {
+ jam();
+ return;
+ }
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_ABORT;
+ alterIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if(opPtr.p->m_requestType != AlterIndxReq::RT_SYSTEMRESTART){
+ // send build request
+ alterIndex_toBuildIndex(signal, opPtr);
+ return;
+ }
+
+ /**
+ * During system restart,
+ * leave index in activated but not build state.
+ *
+ * Build a bit later when REDO has been run
+ */
+ alterIndex_sendReply(signal, opPtr, true);
+}
+
+void
+Dbdict::alterIndex_toDropTrigger(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ // start drop of index triggers
+ DropTrigReq* const req = (DropTrigReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(DropTrigReq::RT_ALTER_INDEX);
+ req->setTableId(opPtr.p->m_request.getTableId());
+ req->setIndexId(opPtr.p->m_request.getIndexId());
+ req->setTriggerInfo(0); // not used
+ opPtr.p->m_triggerCounter = 0;
+ // insert
+ if (indexPtr.p->insertTriggerId != RNIL) {
+ req->setTriggerId(indexPtr.p->insertTriggerId);
+ sendSignal(reference(), GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+ opPtr.p->m_triggerCounter++;
+ }
+ // update
+ if (indexPtr.p->updateTriggerId != RNIL) {
+ req->setTriggerId(indexPtr.p->updateTriggerId);
+ sendSignal(reference(), GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+ opPtr.p->m_triggerCounter++;
+ }
+ // delete
+ if (indexPtr.p->deleteTriggerId != RNIL) {
+ req->setTriggerId(indexPtr.p->deleteTriggerId);
+ sendSignal(reference(), GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+ opPtr.p->m_triggerCounter++;
+ }
+ // custom
+ if (indexPtr.p->customTriggerId != RNIL) {
+ req->setTriggerId(indexPtr.p->customTriggerId);
+ sendSignal(reference(), GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+ opPtr.p->m_triggerCounter++;
+ }
+ // build
+ if (indexPtr.p->buildTriggerId != RNIL) {
+ req->setTriggerId(indexPtr.p->buildTriggerId);
+ sendSignal(reference(), GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+ opPtr.p->m_triggerCounter++;
+ }
+ if (opPtr.p->m_triggerCounter == 0) {
+ // drop in each TC
+ jam();
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_TC;
+ alterIndex_sendSlaveReq(signal, opPtr);
+ }
+}
+
+void
+Dbdict::alterIndex_fromDropTrigger(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ ndbrequire(opPtr.p->m_triggerCounter != 0);
+ if (--opPtr.p->m_triggerCounter != 0) {
+ jam();
+ return;
+ }
+ // finally drop index in each TC
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ const bool isHashIndex = indexPtr.p->isHashIndex();
+ const bool isOrderedIndex = indexPtr.p->isOrderedIndex();
+ ndbrequire(isHashIndex != isOrderedIndex); // xor
+ if (isHashIndex)
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_TC;
+ if (isOrderedIndex)
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_COMMIT;
+ alterIndex_sendSlaveReq(signal, opPtr);
+}
+
+void
+Dbdict::alterIndex_toBuildIndex(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ // get index and table records
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, indexPtr.p->primaryTableId);
+ // build request to self (short signal)
+ BuildIndxReq* const req = (BuildIndxReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(BuildIndxReq::RT_ALTER_INDEX);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ req->setBuildId(0); // not used
+ req->setBuildKey(0); // not used
+ req->setIndexType(indexPtr.p->tableType);
+ req->setIndexId(indexPtr.i);
+ req->setTableId(indexPtr.p->primaryTableId);
+ req->setParallelism(16);
+ // send
+ sendSignal(reference(), GSN_BUILDINDXREQ,
+ signal, BuildIndxReq::SignalLength, JBB);
+}
+
+void
+Dbdict::alterIndex_fromBuildIndex(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_ABORT;
+ alterIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ opPtr.p->m_requestType = AlterIndxReq::RT_DICT_COMMIT;
+ alterIndex_sendSlaveReq(signal, opPtr);
+}
+
+void
+Dbdict::alterIndex_slaveCommit(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ // get index record
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ indexPtr.p->indexState = TableRecord::IS_ONLINE;
+}
+
+void
+Dbdict::alterIndex_slaveAbort(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ jam();
+ // find index record
+ const Uint32 indexId = opPtr.p->m_request.getIndexId();
+ if (indexId >= c_tableRecordPool.getSize())
+ return;
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, indexId);
+ if (! indexPtr.p->isIndex())
+ return;
+ // mark broken
+ indexPtr.p->indexState = TableRecord::IS_BROKEN;
+}
+
+void
+Dbdict::alterIndex_sendSlaveReq(Signal* signal, OpAlterIndexPtr opPtr)
+{
+ AlterIndxReq* const req = (AlterIndxReq*)signal->getDataPtrSend();
+ *req = opPtr.p->m_request;
+ req->setUserRef(opPtr.p->m_coordinatorRef);
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(opPtr.p->m_requestType);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ NdbNodeBitmask receiverNodes = c_aliveNodes;
+ if (opPtr.p->m_requestFlag & RequestFlag::RF_LOCAL) {
+ receiverNodes.clear();
+ receiverNodes.set(getOwnNodeId());
+ }
+ opPtr.p->m_signalCounter = receiverNodes;
+ NodeReceiverGroup rg(DBDICT, receiverNodes);
+ sendSignal(rg, GSN_ALTER_INDX_REQ,
+ signal, AlterIndxReq::SignalLength, JBB);
+}
+
+void
+Dbdict::alterIndex_sendReply(Signal* signal, OpAlterIndexPtr opPtr,
+ bool toUser)
+{
+ AlterIndxRef* rep = (AlterIndxRef*)signal->getDataPtrSend();
+ Uint32 gsn = GSN_ALTER_INDX_CONF;
+ Uint32 length = AlterIndxConf::InternalLength;
+ bool sendRef = opPtr.p->hasError();
+ if (! toUser) {
+ rep->setUserRef(opPtr.p->m_coordinatorRef);
+ rep->setConnectionPtr(opPtr.p->key);
+ rep->setRequestType(opPtr.p->m_requestType);
+ if (opPtr.p->m_requestType == AlterIndxReq::RT_DICT_ABORT)
+ sendRef = false;
+ } else {
+ rep->setUserRef(opPtr.p->m_request.getUserRef());
+ rep->setConnectionPtr(opPtr.p->m_request.getConnectionPtr());
+ rep->setRequestType(opPtr.p->m_request.getRequestType());
+ length = AlterIndxConf::SignalLength;
+ }
+ rep->setTableId(opPtr.p->m_request.getTableId());
+ rep->setIndexId(opPtr.p->m_request.getIndexId());
+ if (sendRef) {
+ if (opPtr.p->m_errorNode == 0)
+ opPtr.p->m_errorNode = getOwnNodeId();
+ rep->setErrorCode(opPtr.p->m_errorCode);
+ rep->setErrorLine(opPtr.p->m_errorLine);
+ rep->setErrorNode(opPtr.p->m_errorNode);
+ gsn = GSN_ALTER_INDX_REF;
+ length = AlterIndxRef::SignalLength;
+ }
+ sendSignal(rep->getUserRef(), gsn, signal, length, JBB);
+}
+
+/**
+ * MODULE: Build index
+ *
+ * Build index or all indexes on a table. Request type:
+ *
+ * RT_USER - normal user request, not yet used
+ * RT_ALTER_INDEX - from alter index
+ * RT_SYSTEM_RESTART -
+ * RT_DICT_PREPARE - prepare participants
+ * RT_DICT_TRIX - to participant on way to local TRIX
+ * RT_DICT_COMMIT - commit in each participant
+ * RT_DICT_ABORT - abort
+ * RT_TRIX - to local TRIX
+ */
+
+void
+Dbdict::execBUILDINDXREQ(Signal* signal)
+{
+ jamEntry();
+ BuildIndxReq* const req = (BuildIndxReq*)signal->getDataPtrSend();
+ OpBuildIndexPtr opPtr;
+ const Uint32 senderRef = signal->senderBlockRef();
+ const BuildIndxReq::RequestType requestType = req->getRequestType();
+ if (requestType == BuildIndxReq::RT_USER ||
+ requestType == BuildIndxReq::RT_ALTER_INDEX ||
+ requestType == BuildIndxReq::RT_SYSTEMRESTART) {
+ jam();
+ if (signal->getLength() == BuildIndxReq::SignalLength) {
+ jam();
+ if (getOwnNodeId() != c_masterNodeId) {
+ jam();
+ // forward to DICT master
+ sendSignal(calcDictBlockRef(c_masterNodeId), GSN_BUILDINDXREQ,
+ signal, signal->getLength(), JBB);
+ return;
+ }
+ // forward initial request plus operation key to all
+ req->setOpKey(++c_opRecordSequence);
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ sendSignal(rg, GSN_BUILDINDXREQ,
+ signal, BuildIndxReq::SignalLength + 1, JBB);
+ return;
+ }
+ // seize operation record
+ ndbrequire(signal->getLength() == BuildIndxReq::SignalLength + 1);
+ const Uint32 opKey = req->getOpKey();
+ OpBuildIndex opBusy;
+ if (! c_opBuildIndex.seize(opPtr))
+ opPtr.p = &opBusy;
+ opPtr.p->save(req);
+ opPtr.p->m_coordinatorRef = senderRef;
+ opPtr.p->m_isMaster = (senderRef == reference());
+ opPtr.p->key = opKey;
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_PREPARE;
+ if (opPtr.p == &opBusy) {
+ jam();
+ opPtr.p->m_errorCode = BuildIndxRef::Busy;
+ opPtr.p->m_errorLine = __LINE__;
+ buildIndex_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ c_opBuildIndex.add(opPtr);
+ // master expects to hear from all
+ opPtr.p->m_signalCounter = c_aliveNodes;
+ buildIndex_sendReply(signal, opPtr, false);
+ return;
+ }
+ c_opBuildIndex.find(opPtr, req->getConnectionPtr());
+ if (! opPtr.isNull()) {
+ opPtr.p->m_requestType = requestType;
+ if (requestType == BuildIndxReq::RT_DICT_TRIX) {
+ jam();
+ buildIndex_buildTrix(signal, opPtr);
+ return;
+ }
+ if (requestType == BuildIndxReq::RT_DICT_TC ||
+ requestType == BuildIndxReq::RT_DICT_TUX) {
+ jam();
+ buildIndex_toOnline(signal, opPtr);
+ return;
+ }
+ if (requestType == BuildIndxReq::RT_DICT_COMMIT ||
+ requestType == BuildIndxReq::RT_DICT_ABORT) {
+ jam();
+ buildIndex_sendReply(signal, opPtr, false);
+ // done in slave
+ if (! opPtr.p->m_isMaster)
+ c_opBuildIndex.release(opPtr);
+ return;
+ }
+ }
+ jam();
+ // return to sender
+ OpBuildIndex opBad;
+ opPtr.p = &opBad;
+ opPtr.p->save(req);
+ opPtr.p->m_errorCode = BuildIndxRef::BadRequestType;
+ opPtr.p->m_errorLine = __LINE__;
+ buildIndex_sendReply(signal, opPtr, true);
+}
+
+void
+Dbdict::execBUILDINDXCONF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(signal->getNoOfSections() == 0);
+ BuildIndxConf* conf = (BuildIndxConf*)signal->getDataPtrSend();
+ buildIndex_recvReply(signal, conf, 0);
+}
+
+void
+Dbdict::execBUILDINDXREF(Signal* signal)
+{
+ jamEntry();
+ BuildIndxRef* ref = (BuildIndxRef*)signal->getDataPtrSend();
+ buildIndex_recvReply(signal, ref->getConf(), ref);
+}
+
+void
+Dbdict::buildIndex_recvReply(Signal* signal, const BuildIndxConf* conf,
+ const BuildIndxRef* ref)
+{
+ jam();
+ const Uint32 senderRef = signal->senderBlockRef();
+ const BuildIndxReq::RequestType requestType = conf->getRequestType();
+ const Uint32 key = conf->getConnectionPtr();
+ if (requestType == BuildIndxReq::RT_ALTER_INDEX) {
+ jam();
+ // part of alter index operation
+ OpAlterIndexPtr opPtr;
+ c_opAlterIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ alterIndex_fromBuildIndex(signal, opPtr);
+ return;
+ }
+
+ if (requestType == BuildIndxReq::RT_SYSTEMRESTART) {
+ jam();
+ if (ref == 0) {
+ infoEvent("DICT: index %u rebuild done", (unsigned)key);
+ } else {
+ warningEvent("DICT: index %u rebuild failed: code=%d line=%d node=%d",
+ (unsigned)key, ref->getErrorCode());
+ }
+ rebuildIndexes(signal, key + 1);
+ return;
+ }
+
+ OpBuildIndexPtr opPtr;
+ c_opBuildIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ if (requestType == BuildIndxReq::RT_TRIX) {
+ jam();
+ // forward to master
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_TRIX;
+ buildIndex_sendReply(signal, opPtr, false);
+ return;
+ }
+ ndbrequire(opPtr.p->m_isMaster);
+ ndbrequire(opPtr.p->m_requestType == requestType);
+ opPtr.p->m_signalCounter.clearWaitingFor(refToNode(senderRef));
+ if (! opPtr.p->m_signalCounter.done()) {
+ jam();
+ return;
+ }
+ if (requestType == BuildIndxReq::RT_DICT_COMMIT ||
+ requestType == BuildIndxReq::RT_DICT_ABORT) {
+ jam();
+ // send reply to user
+ buildIndex_sendReply(signal, opPtr, true);
+ c_opBuildIndex.release(opPtr);
+ return;
+ }
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_ABORT;
+ buildIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ if (indexPtr.p->isHashIndex()) {
+ if (requestType == BuildIndxReq::RT_DICT_PREPARE) {
+ jam();
+ if (! (opPtr.p->m_requestFlag & RequestFlag::RF_NOBUILD)) {
+ buildIndex_toCreateConstr(signal, opPtr);
+ } else {
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_TC;
+ buildIndex_sendSlaveReq(signal, opPtr);
+ }
+ return;
+ }
+ if (requestType == BuildIndxReq::RT_DICT_TRIX) {
+ jam();
+ ndbrequire(! (opPtr.p->m_requestFlag & RequestFlag::RF_NOBUILD));
+ buildIndex_toDropConstr(signal, opPtr);
+ return;
+ }
+ if (requestType == BuildIndxReq::RT_DICT_TC) {
+ jam();
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_COMMIT;
+ buildIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ }
+ if (indexPtr.p->isOrderedIndex()) {
+ if (requestType == BuildIndxReq::RT_DICT_PREPARE) {
+ jam();
+ if (! (opPtr.p->m_requestFlag & RequestFlag::RF_NOBUILD)) {
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_TRIX;
+ buildIndex_sendSlaveReq(signal, opPtr);
+ } else {
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_TUX;
+ buildIndex_sendSlaveReq(signal, opPtr);
+ }
+ return;
+ }
+ if (requestType == BuildIndxReq::RT_DICT_TRIX) {
+ jam();
+ ndbrequire(! (opPtr.p->m_requestFlag & RequestFlag::RF_NOBUILD));
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_TUX;
+ buildIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (requestType == BuildIndxReq::RT_DICT_TUX) {
+ jam();
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_COMMIT;
+ buildIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::buildIndex_toCreateConstr(Signal* signal, OpBuildIndexPtr opPtr)
+{
+ jam();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ // request to create constraint trigger
+ CreateTrigReq* req = (CreateTrigReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(CreateTrigReq::RT_BUILD_INDEX);
+ req->addRequestFlag(0); // none
+ req->setTableId(indexPtr.i);
+ req->setIndexId(RNIL);
+ req->setTriggerId(RNIL);
+ req->setTriggerType(TriggerType::READ_ONLY_CONSTRAINT);
+ req->setTriggerActionTime(TriggerActionTime::TA_AFTER);
+ req->setTriggerEvent(TriggerEvent::TE_UPDATE);
+ req->setMonitorReplicas(false);
+ req->setMonitorAllAttributes(false);
+ req->setOnline(true); // alter online after create
+ req->setReceiverRef(0); // no receiver, REF-ed by TUP
+ req->getAttributeMask().clear();
+ // NDB$PK is last attribute
+ req->getAttributeMask().set(indexPtr.p->noOfAttributes - 1);
+ // name section
+ char triggerName[MAX_TAB_NAME_SIZE];
+ Uint32 buffer[2 + ((MAX_TAB_NAME_SIZE + 3) >> 2)]; // SP string
+ LinearWriter w(buffer, sizeof(buffer) >> 2);
+ LinearSectionPtr lsPtr[3];
+ sprintf(triggerName, "NDB$INDEX_%u_BUILD", indexPtr.i);
+ w.reset();
+ w.add(CreateTrigReq::TriggerNameKey, triggerName);
+ lsPtr[0].p = buffer;
+ lsPtr[0].sz = w.getWordsUsed();
+ sendSignal(reference(), GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength, JBB, lsPtr, 1);
+}
+
+void
+Dbdict::buildIndex_fromCreateConstr(Signal* signal, OpBuildIndexPtr opPtr)
+{
+ jam();
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_ABORT;
+ buildIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_TRIX;
+ buildIndex_sendSlaveReq(signal, opPtr);
+}
+
+void
+Dbdict::buildIndex_buildTrix(Signal* signal, OpBuildIndexPtr opPtr)
+{
+ jam();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, indexPtr.p->primaryTableId);
+ // build request
+ BuildIndxReq* const req = (BuildIndxReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(BuildIndxReq::RT_TRIX);
+ req->setBuildId(0); // not yet..
+ req->setBuildKey(0); // ..in use
+ req->setIndexType(indexPtr.p->tableType);
+ req->setIndexId(indexPtr.i);
+ req->setTableId(indexPtr.p->primaryTableId);
+ req->setParallelism(16);
+ if (indexPtr.p->isHashIndex()) {
+ jam();
+ getIndexAttrList(indexPtr, opPtr.p->m_attrList);
+ getTableKeyList(tablePtr, opPtr.p->m_tableKeyList);
+ // send
+ LinearSectionPtr lsPtr[3];
+ lsPtr[0].sz = opPtr.p->m_attrList.sz;
+ lsPtr[0].p = opPtr.p->m_attrList.id;
+ lsPtr[1].sz = opPtr.p->m_tableKeyList.sz;
+ lsPtr[1].p = opPtr.p->m_tableKeyList.id;
+ sendSignal(calcTrixBlockRef(getOwnNodeId()), GSN_BUILDINDXREQ,
+ signal, BuildIndxReq::SignalLength, JBB, lsPtr, 2);
+ return;
+ }
+ if (indexPtr.p->isOrderedIndex()) {
+ jam();
+ sendSignal(calcTupBlockRef(getOwnNodeId()), GSN_BUILDINDXREQ,
+ signal, BuildIndxReq::SignalLength, JBB);
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::buildIndex_toDropConstr(Signal* signal, OpBuildIndexPtr opPtr)
+{
+ jam();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ // request to drop constraint trigger
+ DropTrigReq* req = (DropTrigReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(DropTrigReq::RT_BUILD_INDEX);
+ req->addRequestFlag(0); // none
+ req->setTableId(indexPtr.i);
+ req->setIndexId(RNIL);
+ req->setTriggerId(opPtr.p->m_constrTriggerId);
+ req->setTriggerInfo(0); // not used
+ sendSignal(reference(), GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+}
+
+void
+Dbdict::buildIndex_fromDropConstr(Signal* signal, OpBuildIndexPtr opPtr)
+{
+ jam();
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_ABORT;
+ buildIndex_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ opPtr.p->m_requestType = BuildIndxReq::RT_DICT_TC;
+ buildIndex_sendSlaveReq(signal, opPtr);
+}
+
+void
+Dbdict::buildIndex_toOnline(Signal* signal, OpBuildIndexPtr opPtr)
+{
+ jam();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, opPtr.p->m_request.getIndexId());
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, indexPtr.p->primaryTableId);
+ // request to set index online in TC or TUX
+ AlterIndxReq* const req = (AlterIndxReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ if (opPtr.p->m_requestType == BuildIndxReq::RT_DICT_TC) {
+ req->setRequestType(AlterIndxReq::RT_TC);
+ } else if (opPtr.p->m_requestType == BuildIndxReq::RT_DICT_TUX) {
+ req->setRequestType(AlterIndxReq::RT_TUX);
+ } else {
+ ndbrequire(false);
+ }
+ req->setTableId(tablePtr.i);
+ req->setIndexId(indexPtr.i);
+ req->setIndexVersion(indexPtr.p->tableVersion);
+ req->setOnline(true);
+ BlockReference blockRef = 0;
+ if (opPtr.p->m_requestType == BuildIndxReq::RT_DICT_TC) {
+ blockRef = calcTcBlockRef(getOwnNodeId());
+ } else if (opPtr.p->m_requestType == BuildIndxReq::RT_DICT_TUX) {
+ blockRef = calcTuxBlockRef(getOwnNodeId());
+ } else {
+ ndbrequire(false);
+ }
+ // send
+ sendSignal(blockRef, GSN_ALTER_INDX_REQ,
+ signal, BuildIndxReq::SignalLength, JBB);
+}
+
+void
+Dbdict::buildIndex_fromOnline(Signal* signal, OpBuildIndexPtr opPtr)
+{
+ jam();
+ // forward to master
+ buildIndex_sendReply(signal, opPtr, false);
+}
+
+void
+Dbdict::buildIndex_sendSlaveReq(Signal* signal, OpBuildIndexPtr opPtr)
+{
+ BuildIndxReq* const req = (BuildIndxReq*)signal->getDataPtrSend();
+ *req = opPtr.p->m_request;
+ req->setUserRef(opPtr.p->m_coordinatorRef);
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(opPtr.p->m_requestType);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ opPtr.p->m_signalCounter = c_aliveNodes;
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ sendSignal(rg, GSN_BUILDINDXREQ,
+ signal, BuildIndxReq::SignalLength, JBB);
+}
+
+void
+Dbdict::buildIndex_sendReply(Signal* signal, OpBuildIndexPtr opPtr,
+ bool toUser)
+{
+ BuildIndxRef* rep = (BuildIndxRef*)signal->getDataPtrSend();
+ Uint32 gsn = GSN_BUILDINDXCONF;
+ Uint32 length = BuildIndxConf::InternalLength;
+ bool sendRef = opPtr.p->hasError();
+ if (! toUser) {
+ rep->setUserRef(opPtr.p->m_coordinatorRef);
+ rep->setConnectionPtr(opPtr.p->key);
+ rep->setRequestType(opPtr.p->m_requestType);
+ if (opPtr.p->m_requestType == BuildIndxReq::RT_DICT_ABORT)
+ sendRef = false;
+ } else {
+ rep->setUserRef(opPtr.p->m_request.getUserRef());
+ rep->setConnectionPtr(opPtr.p->m_request.getConnectionPtr());
+ rep->setRequestType(opPtr.p->m_request.getRequestType());
+ length = BuildIndxConf::SignalLength;
+ }
+ rep->setIndexType(opPtr.p->m_request.getIndexType());
+ rep->setTableId(opPtr.p->m_request.getTableId());
+ rep->setIndexId(opPtr.p->m_request.getIndexId());
+ if (sendRef) {
+ rep->setErrorCode(opPtr.p->m_errorCode);
+ gsn = GSN_BUILDINDXREF;
+ length = BuildIndxRef::SignalLength;
+ }
+ sendSignal(rep->getUserRef(), gsn, signal, length, JBB);
+}
+
+/**
+ * MODULE: Create trigger
+ *
+ * Create trigger in all DICT blocks. Optionally start alter trigger
+ * operation to set the trigger online.
+ *
+ * Request type received in REQ and returned in CONF/REF:
+ *
+ * RT_USER - normal user e.g. BACKUP
+ * RT_ALTER_INDEX - from alter index online
+ * RT_DICT_PREPARE - seize operation in each DICT
+ * RT_DICT_COMMIT - commit create in each DICT
+ * RT_TC - sending to TC (operation alter trigger)
+ * RT_LQH - sending to LQH (operation alter trigger)
+ */
+
+void
+Dbdict::execCREATE_TRIG_REQ(Signal* signal)
+{
+ jamEntry();
+ CreateTrigReq* const req = (CreateTrigReq*)signal->getDataPtrSend();
+ OpCreateTriggerPtr opPtr;
+ const Uint32 senderRef = signal->senderBlockRef();
+ const CreateTrigReq::RequestType requestType = req->getRequestType();
+ if (requestType == CreateTrigReq::RT_USER ||
+ requestType == CreateTrigReq::RT_ALTER_INDEX ||
+ requestType == CreateTrigReq::RT_BUILD_INDEX) {
+ jam();
+ if (! assembleFragments(signal)) {
+ jam();
+ return;
+ }
+ const bool isLocal = req->getRequestFlag() & RequestFlag::RF_LOCAL;
+ NdbNodeBitmask receiverNodes = c_aliveNodes;
+ if (isLocal) {
+ receiverNodes.clear();
+ receiverNodes.set(getOwnNodeId());
+ }
+ if (signal->getLength() == CreateTrigReq::SignalLength) {
+ jam();
+ if (! isLocal && getOwnNodeId() != c_masterNodeId) {
+ jam();
+ // forward to DICT master
+ sendSignal(calcDictBlockRef(c_masterNodeId), GSN_CREATE_TRIG_REQ,
+ signal, signal->getLength(), JBB);
+ return;
+ }
+ // forward initial request plus operation key to all
+ req->setOpKey(++c_opRecordSequence);
+ NodeReceiverGroup rg(DBDICT, receiverNodes);
+ sendSignal(rg, GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength + 1, JBB);
+ return;
+ }
+ // seize operation record
+ ndbrequire(signal->getLength() == CreateTrigReq::SignalLength + 1);
+ const Uint32 opKey = req->getOpKey();
+ OpCreateTrigger opBusy;
+ if (! c_opCreateTrigger.seize(opPtr))
+ opPtr.p = &opBusy;
+ opPtr.p->save(req);
+ opPtr.p->m_coordinatorRef = senderRef;
+ opPtr.p->m_isMaster = (senderRef == reference());
+ opPtr.p->key = opKey;
+ opPtr.p->m_requestType = CreateTrigReq::RT_DICT_PREPARE;
+ if (opPtr.p == &opBusy) {
+ jam();
+ opPtr.p->m_errorCode = CreateTrigRef::Busy;
+ opPtr.p->m_errorLine = __LINE__;
+ releaseSections(signal);
+ createTrigger_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ c_opCreateTrigger.add(opPtr);
+ {
+ // save name
+ SegmentedSectionPtr ssPtr;
+ signal->getSection(ssPtr, CreateTrigReq::TRIGGER_NAME_SECTION);
+ SimplePropertiesSectionReader ssReader(ssPtr, getSectionSegmentPool());
+ if (ssReader.getKey() != CreateTrigReq::TriggerNameKey ||
+ ! ssReader.getString(opPtr.p->m_triggerName)) {
+ jam();
+ opPtr.p->m_errorCode = CreateTrigRef::InvalidName;
+ opPtr.p->m_errorLine = __LINE__;
+ releaseSections(signal);
+ createTrigger_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ }
+ releaseSections(signal);
+ {
+ // check that trigger name is unique
+ TriggerRecordPtr triggerPtr;
+ TriggerRecord keyRecord;
+ strcpy(keyRecord.triggerName, opPtr.p->m_triggerName);
+ c_triggerRecordHash.find(triggerPtr, keyRecord);
+ if (triggerPtr.i != RNIL) {
+ jam();
+ opPtr.p->m_errorCode = CreateTrigRef::TriggerExists;
+ opPtr.p->m_errorLine = __LINE__;
+ createTrigger_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ }
+
+ // master expects to hear from all
+ if (opPtr.p->m_isMaster)
+ opPtr.p->m_signalCounter = receiverNodes;
+ // check request in all participants
+ createTrigger_slavePrepare(signal, opPtr);
+ createTrigger_sendReply(signal, opPtr, false);
+ return;
+ }
+ c_opCreateTrigger.find(opPtr, req->getConnectionPtr());
+ if (! opPtr.isNull()) {
+ opPtr.p->m_requestType = requestType;
+ if (requestType == CreateTrigReq::RT_DICT_CREATE) {
+ jam();
+ // master has set trigger id
+ opPtr.p->m_request.setTriggerId(req->getTriggerId());
+ createTrigger_slaveCreate(signal, opPtr);
+ createTrigger_sendReply(signal, opPtr, false);
+ return;
+ }
+ if (requestType == CreateTrigReq::RT_DICT_COMMIT ||
+ requestType == CreateTrigReq::RT_DICT_ABORT) {
+ jam();
+ if (requestType == CreateTrigReq::RT_DICT_COMMIT)
+ createTrigger_slaveCommit(signal, opPtr);
+ else
+ createTrigger_slaveAbort(signal, opPtr);
+ createTrigger_sendReply(signal, opPtr, false);
+ // done in slave
+ if (! opPtr.p->m_isMaster)
+ c_opCreateTrigger.release(opPtr);
+ return;
+ }
+ }
+ jam();
+ // return to sender
+ releaseSections(signal);
+ OpCreateTrigger opBad;
+ opPtr.p = &opBad;
+ opPtr.p->save(req);
+ opPtr.p->m_errorCode = CreateTrigRef::BadRequestType;
+ opPtr.p->m_errorLine = __LINE__;
+ createTrigger_sendReply(signal, opPtr, true);
+}
+
+void
+Dbdict::execCREATE_TRIG_CONF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(signal->getNoOfSections() == 0);
+ CreateTrigConf* conf = (CreateTrigConf*)signal->getDataPtrSend();
+ createTrigger_recvReply(signal, conf, 0);
+}
+
+void
+Dbdict::execCREATE_TRIG_REF(Signal* signal)
+{
+ jamEntry();
+ CreateTrigRef* ref = (CreateTrigRef*)signal->getDataPtrSend();
+ createTrigger_recvReply(signal, ref->getConf(), ref);
+}
+
+void
+Dbdict::createTrigger_recvReply(Signal* signal, const CreateTrigConf* conf,
+ const CreateTrigRef* ref)
+{
+ jam();
+ const Uint32 senderRef = signal->senderBlockRef();
+ const CreateTrigReq::RequestType requestType = conf->getRequestType();
+ const Uint32 key = conf->getConnectionPtr();
+ if (requestType == CreateTrigReq::RT_ALTER_INDEX) {
+ jam();
+ // part of alter index operation
+ OpAlterIndexPtr opPtr;
+ c_opAlterIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ alterIndex_fromCreateTrigger(signal, opPtr);
+ return;
+ }
+ if (requestType == CreateTrigReq::RT_BUILD_INDEX) {
+ jam();
+ // part of build index operation
+ OpBuildIndexPtr opPtr;
+ c_opBuildIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ // fill in trigger id
+ opPtr.p->m_constrTriggerId = conf->getTriggerId();
+ buildIndex_fromCreateConstr(signal, opPtr);
+ return;
+ }
+ if (requestType == CreateTrigReq::RT_TC ||
+ requestType == CreateTrigReq::RT_LQH) {
+ jam();
+ // part of alter trigger operation
+ OpAlterTriggerPtr opPtr;
+ c_opAlterTrigger.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ alterTrigger_fromCreateLocal(signal, opPtr);
+ return;
+ }
+ OpCreateTriggerPtr opPtr;
+ c_opCreateTrigger.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ ndbrequire(opPtr.p->m_isMaster);
+ ndbrequire(opPtr.p->m_requestType == requestType);
+ opPtr.p->setError(ref);
+ opPtr.p->m_signalCounter.clearWaitingFor(refToNode(senderRef));
+ if (! opPtr.p->m_signalCounter.done()) {
+ jam();
+ return;
+ }
+ if (requestType == CreateTrigReq::RT_DICT_COMMIT ||
+ requestType == CreateTrigReq::RT_DICT_ABORT) {
+ jam();
+ // send reply to user
+ createTrigger_sendReply(signal, opPtr, true);
+ c_opCreateTrigger.release(opPtr);
+ return;
+ }
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = CreateTrigReq::RT_DICT_ABORT;
+ createTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (requestType == CreateTrigReq::RT_DICT_PREPARE) {
+ jam();
+ // seize trigger id in master
+ createTrigger_masterSeize(signal, opPtr);
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = CreateTrigReq::RT_DICT_ABORT;
+ createTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ opPtr.p->m_requestType = CreateTrigReq::RT_DICT_CREATE;
+ createTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (requestType == CreateTrigReq::RT_DICT_CREATE) {
+ jam();
+ if (opPtr.p->m_request.getOnline()) {
+ jam();
+ // start alter online
+ createTrigger_toAlterTrigger(signal, opPtr);
+ return;
+ }
+ opPtr.p->m_requestType = CreateTrigReq::RT_DICT_COMMIT;
+ createTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::createTrigger_slavePrepare(Signal* signal, OpCreateTriggerPtr opPtr)
+{
+ jam();
+ const CreateTrigReq* const req = &opPtr.p->m_request;
+ // check trigger type
+ if (req->getRequestType() == CreateTrigReq::RT_USER &&
+ req->getTriggerType() == TriggerType::SUBSCRIPTION ||
+ req->getRequestType() == CreateTrigReq::RT_ALTER_INDEX &&
+ req->getTriggerType() == TriggerType::SECONDARY_INDEX ||
+ req->getRequestType() == CreateTrigReq::RT_ALTER_INDEX &&
+ req->getTriggerType() == TriggerType::ORDERED_INDEX ||
+ req->getRequestType() == CreateTrigReq::RT_BUILD_INDEX &&
+ req->getTriggerType() == TriggerType::READ_ONLY_CONSTRAINT) {
+ ;
+ } else {
+ jam();
+ opPtr.p->m_errorCode = CreateTrigRef::UnsupportedTriggerType;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ // check the table
+ const Uint32 tableId = req->getTableId();
+ if (! (tableId < c_tableRecordPool.getSize())) {
+ jam();
+ opPtr.p->m_errorCode = CreateTrigRef::InvalidTable;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, tableId);
+ if (tablePtr.p->tabState != TableRecord::DEFINED) {
+ jam();
+ opPtr.p->m_errorCode = CreateTrigRef::InvalidTable;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+}
+
+void
+Dbdict::createTrigger_masterSeize(Signal* signal, OpCreateTriggerPtr opPtr)
+{
+ TriggerRecordPtr triggerPtr;
+ if (opPtr.p->m_requestFlag & RequestFlag::RF_LOCAL) {
+ triggerPtr.i = opPtr.p->m_request.getTriggerId();
+ } else {
+ triggerPtr.i = getFreeTriggerRecord();
+ if (triggerPtr.i == RNIL) {
+ jam();
+ opPtr.p->m_errorCode = CreateTrigRef::TooManyTriggers;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ }
+ c_triggerRecordPool.getPtr(triggerPtr);
+ initialiseTriggerRecord(triggerPtr);
+ triggerPtr.p->triggerState = TriggerRecord::TS_DEFINING;
+ opPtr.p->m_request.setTriggerId(triggerPtr.i);
+}
+
+void
+Dbdict::createTrigger_slaveCreate(Signal* signal, OpCreateTriggerPtr opPtr)
+{
+ jam();
+ const CreateTrigReq* const req = &opPtr.p->m_request;
+ // get the trigger record
+ const Uint32 triggerId = req->getTriggerId();
+ TriggerRecordPtr triggerPtr;
+ c_triggerRecordPool.getPtr(triggerPtr, triggerId);
+ initialiseTriggerRecord(triggerPtr);
+ // fill in trigger data
+ strcpy(triggerPtr.p->triggerName, opPtr.p->m_triggerName);
+ triggerPtr.p->triggerId = triggerId;
+ triggerPtr.p->tableId = req->getTableId();
+ triggerPtr.p->indexId = RNIL;
+ triggerPtr.p->triggerType = req->getTriggerType();
+ triggerPtr.p->triggerActionTime = req->getTriggerActionTime();
+ triggerPtr.p->triggerEvent = req->getTriggerEvent();
+ triggerPtr.p->monitorReplicas = req->getMonitorReplicas();
+ triggerPtr.p->monitorAllAttributes = req->getMonitorAllAttributes();
+ triggerPtr.p->attributeMask = req->getAttributeMask();
+ triggerPtr.p->triggerState = TriggerRecord::TS_OFFLINE;
+ // add to hash table
+ // ndbout_c("++++++++++++ Adding trigger id %u, %s", triggerPtr.p->triggerId, triggerPtr.p->triggerName);
+ c_triggerRecordHash.add(triggerPtr);
+ if (triggerPtr.p->triggerType == TriggerType::SECONDARY_INDEX ||
+ triggerPtr.p->triggerType == TriggerType::ORDERED_INDEX) {
+ jam();
+ // connect to index record XXX should be done in caller instead
+ triggerPtr.p->indexId = req->getIndexId();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, triggerPtr.p->indexId);
+ switch (triggerPtr.p->triggerEvent) {
+ case TriggerEvent::TE_INSERT:
+ indexPtr.p->insertTriggerId = triggerPtr.p->triggerId;
+ break;
+ case TriggerEvent::TE_UPDATE:
+ indexPtr.p->updateTriggerId = triggerPtr.p->triggerId;
+ break;
+ case TriggerEvent::TE_DELETE:
+ indexPtr.p->deleteTriggerId = triggerPtr.p->triggerId;
+ break;
+ case TriggerEvent::TE_CUSTOM:
+ indexPtr.p->customTriggerId = triggerPtr.p->triggerId;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+ }
+ if (triggerPtr.p->triggerType == TriggerType::READ_ONLY_CONSTRAINT) {
+ jam();
+ // connect to index record XXX should be done in caller instead
+ triggerPtr.p->indexId = req->getTableId();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, triggerPtr.p->indexId);
+ indexPtr.p->buildTriggerId = triggerPtr.p->triggerId;
+ }
+}
+
+void
+Dbdict::createTrigger_toAlterTrigger(Signal* signal, OpCreateTriggerPtr opPtr)
+{
+ jam();
+ AlterTrigReq* req = (AlterTrigReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(AlterTrigReq::RT_CREATE_TRIGGER);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ req->setTableId(opPtr.p->m_request.getTableId());
+ req->setTriggerId(opPtr.p->m_request.getTriggerId());
+ req->setTriggerInfo(0); // not used
+ req->setOnline(true);
+ req->setReceiverRef(opPtr.p->m_request.getReceiverRef());
+ sendSignal(reference(), GSN_ALTER_TRIG_REQ,
+ signal, AlterTrigReq::SignalLength, JBB);
+}
+
+void
+Dbdict::createTrigger_fromAlterTrigger(Signal* signal, OpCreateTriggerPtr opPtr)
+{
+ jam();
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = CreateTrigReq::RT_DICT_ABORT;
+ createTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ opPtr.p->m_requestType = CreateTrigReq::RT_DICT_COMMIT;
+ createTrigger_sendSlaveReq(signal, opPtr);
+}
+
+void
+Dbdict::createTrigger_slaveCommit(Signal* signal, OpCreateTriggerPtr opPtr)
+{
+ jam();
+ const CreateTrigReq* const req = &opPtr.p->m_request;
+ // get the trigger record
+ const Uint32 triggerId = req->getTriggerId();
+ TriggerRecordPtr triggerPtr;
+ c_triggerRecordPool.getPtr(triggerPtr, triggerId);
+ if (! req->getOnline()) {
+ triggerPtr.p->triggerState = TriggerRecord::TS_OFFLINE;
+ } else {
+ ndbrequire(triggerPtr.p->triggerState == TriggerRecord::TS_ONLINE);
+ }
+}
+
+void
+Dbdict::createTrigger_slaveAbort(Signal* signal, OpCreateTriggerPtr opPtr)
+{
+ jam();
+}
+
+void
+Dbdict::createTrigger_sendSlaveReq(Signal* signal, OpCreateTriggerPtr opPtr)
+{
+ CreateTrigReq* const req = (CreateTrigReq*)signal->getDataPtrSend();
+ *req = opPtr.p->m_request;
+ req->setUserRef(opPtr.p->m_coordinatorRef);
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(opPtr.p->m_requestType);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ NdbNodeBitmask receiverNodes = c_aliveNodes;
+ if (opPtr.p->m_requestFlag & RequestFlag::RF_LOCAL) {
+ receiverNodes.clear();
+ receiverNodes.set(getOwnNodeId());
+ }
+ opPtr.p->m_signalCounter = receiverNodes;
+ NodeReceiverGroup rg(DBDICT, receiverNodes);
+ sendSignal(rg, GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength, JBB);
+}
+
+void
+Dbdict::createTrigger_sendReply(Signal* signal, OpCreateTriggerPtr opPtr,
+ bool toUser)
+{
+ CreateTrigRef* rep = (CreateTrigRef*)signal->getDataPtrSend();
+ Uint32 gsn = GSN_CREATE_TRIG_CONF;
+ Uint32 length = CreateTrigConf::InternalLength;
+ bool sendRef = opPtr.p->hasError();
+ if (! toUser) {
+ rep->setUserRef(opPtr.p->m_coordinatorRef);
+ rep->setConnectionPtr(opPtr.p->key);
+ rep->setRequestType(opPtr.p->m_requestType);
+ if (opPtr.p->m_requestType == CreateTrigReq::RT_DICT_ABORT)
+ sendRef = false;
+ } else {
+ rep->setUserRef(opPtr.p->m_request.getUserRef());
+ rep->setConnectionPtr(opPtr.p->m_request.getConnectionPtr());
+ rep->setRequestType(opPtr.p->m_request.getRequestType());
+ length = CreateTrigConf::SignalLength;
+ }
+ rep->setTableId(opPtr.p->m_request.getTableId());
+ rep->setIndexId(opPtr.p->m_request.getIndexId());
+ rep->setTriggerId(opPtr.p->m_request.getTriggerId());
+ rep->setTriggerInfo(opPtr.p->m_request.getTriggerInfo());
+ if (sendRef) {
+ if (opPtr.p->m_errorNode == 0)
+ opPtr.p->m_errorNode = getOwnNodeId();
+ rep->setErrorCode(opPtr.p->m_errorCode);
+ rep->setErrorLine(opPtr.p->m_errorLine);
+ rep->setErrorNode(opPtr.p->m_errorNode);
+ gsn = GSN_CREATE_TRIG_REF;
+ length = CreateTrigRef::SignalLength;
+ }
+ sendSignal(rep->getUserRef(), gsn, signal, length, JBB);
+}
+
+/**
+ * MODULE: Drop trigger.
+ */
+
+void
+Dbdict::execDROP_TRIG_REQ(Signal* signal)
+{
+ jamEntry();
+ DropTrigReq* const req = (DropTrigReq*)signal->getDataPtrSend();
+ OpDropTriggerPtr opPtr;
+ const Uint32 senderRef = signal->senderBlockRef();
+ const DropTrigReq::RequestType requestType = req->getRequestType();
+
+ if (signal->getNoOfSections() > 0) {
+ ndbrequire(signal->getNoOfSections() == 1);
+ jam();
+ TriggerRecord keyRecord;
+ OpDropTrigger opTmp;
+ opPtr.p=&opTmp;
+
+ SegmentedSectionPtr ssPtr;
+ signal->getSection(ssPtr, DropTrigReq::TRIGGER_NAME_SECTION);
+ SimplePropertiesSectionReader ssReader(ssPtr, getSectionSegmentPool());
+ if (ssReader.getKey() != DropTrigReq::TriggerNameKey ||
+ ! ssReader.getString(keyRecord.triggerName)) {
+ jam();
+ opPtr.p->m_errorCode = DropTrigRef::InvalidName;
+ opPtr.p->m_errorLine = __LINE__;
+ releaseSections(signal);
+ dropTrigger_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ releaseSections(signal);
+
+ TriggerRecordPtr triggerPtr;
+
+ // ndbout_c("++++++++++++++ Looking for trigger %s", keyRecord.triggerName);
+ c_triggerRecordHash.find(triggerPtr, keyRecord);
+ if (triggerPtr.i == RNIL) {
+ jam();
+ req->setTriggerId(RNIL);
+ } else {
+ jam();
+ // ndbout_c("++++++++++ Found trigger %s", triggerPtr.p->triggerName);
+ req->setTriggerId(triggerPtr.p->triggerId);
+ req->setTableId(triggerPtr.p->tableId);
+ }
+ }
+ if (requestType == DropTrigReq::RT_USER ||
+ requestType == DropTrigReq::RT_ALTER_INDEX ||
+ requestType == DropTrigReq::RT_BUILD_INDEX) {
+ jam();
+ if (signal->getLength() == DropTrigReq::SignalLength) {
+ if (getOwnNodeId() != c_masterNodeId) {
+ jam();
+ // forward to DICT master
+ sendSignal(calcDictBlockRef(c_masterNodeId), GSN_DROP_TRIG_REQ,
+ signal, signal->getLength(), JBB);
+ return;
+ }
+ if (!c_triggerRecordPool.findId(req->getTriggerId())) {
+ jam();
+ // return to sender
+ OpDropTrigger opBad;
+ opPtr.p = &opBad;
+ opPtr.p->save(req);
+ opPtr.p->m_errorCode = DropTrigRef::TriggerNotFound;
+ opPtr.p->m_errorLine = __LINE__;
+ dropTrigger_sendReply(signal, opPtr, true);
+ return;
+ }
+ // forward initial request plus operation key to all
+ req->setOpKey(++c_opRecordSequence);
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ sendSignal(rg, GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength + 1, JBB);
+ return;
+ }
+ // seize operation record
+ ndbrequire(signal->getLength() == DropTrigReq::SignalLength + 1);
+ const Uint32 opKey = req->getOpKey();
+ OpDropTrigger opBusy;
+ if (! c_opDropTrigger.seize(opPtr))
+ opPtr.p = &opBusy;
+ opPtr.p->save(req);
+ opPtr.p->m_coordinatorRef = senderRef;
+ opPtr.p->m_isMaster = (senderRef == reference());
+ opPtr.p->key = opKey;
+ opPtr.p->m_requestType = DropTrigReq::RT_DICT_PREPARE;
+ if (opPtr.p == &opBusy) {
+ jam();
+ opPtr.p->m_errorCode = DropTrigRef::Busy;
+ opPtr.p->m_errorLine = __LINE__;
+ dropTrigger_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ c_opDropTrigger.add(opPtr);
+ // master expects to hear from all
+ if (opPtr.p->m_isMaster)
+ opPtr.p->m_signalCounter = c_aliveNodes;
+ dropTrigger_slavePrepare(signal, opPtr);
+ dropTrigger_sendReply(signal, opPtr, false);
+ return;
+ }
+ c_opDropTrigger.find(opPtr, req->getConnectionPtr());
+ if (! opPtr.isNull()) {
+ opPtr.p->m_requestType = requestType;
+ if (requestType == DropTrigReq::RT_DICT_COMMIT ||
+ requestType == DropTrigReq::RT_DICT_ABORT) {
+ jam();
+ if (requestType == DropTrigReq::RT_DICT_COMMIT)
+ dropTrigger_slaveCommit(signal, opPtr);
+ else
+ dropTrigger_slaveAbort(signal, opPtr);
+ dropTrigger_sendReply(signal, opPtr, false);
+ // done in slave
+ if (! opPtr.p->m_isMaster)
+ c_opDropTrigger.release(opPtr);
+ return;
+ }
+ }
+ jam();
+ // return to sender
+ OpDropTrigger opBad;
+ opPtr.p = &opBad;
+ opPtr.p->save(req);
+ opPtr.p->m_errorCode = DropTrigRef::BadRequestType;
+ opPtr.p->m_errorLine = __LINE__;
+ dropTrigger_sendReply(signal, opPtr, true);
+}
+
+void
+Dbdict::execDROP_TRIG_CONF(Signal* signal)
+{
+ jamEntry();
+ DropTrigConf* conf = (DropTrigConf*)signal->getDataPtrSend();
+ dropTrigger_recvReply(signal, conf, 0);
+}
+
+void
+Dbdict::execDROP_TRIG_REF(Signal* signal)
+{
+ jamEntry();
+ DropTrigRef* ref = (DropTrigRef*)signal->getDataPtrSend();
+ dropTrigger_recvReply(signal, ref->getConf(), ref);
+}
+
+void
+Dbdict::dropTrigger_recvReply(Signal* signal, const DropTrigConf* conf,
+ const DropTrigRef* ref)
+{
+ jam();
+ const Uint32 senderRef = signal->senderBlockRef();
+ const DropTrigReq::RequestType requestType = conf->getRequestType();
+ const Uint32 key = conf->getConnectionPtr();
+ if (requestType == DropTrigReq::RT_ALTER_INDEX) {
+ jam();
+ // part of alter index operation
+ OpAlterIndexPtr opPtr;
+ c_opAlterIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ alterIndex_fromDropTrigger(signal, opPtr);
+ return;
+ }
+ if (requestType == DropTrigReq::RT_BUILD_INDEX) {
+ jam();
+ // part of build index operation
+ OpBuildIndexPtr opPtr;
+ c_opBuildIndex.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ buildIndex_fromDropConstr(signal, opPtr);
+ return;
+ }
+ if (requestType == DropTrigReq::RT_TC ||
+ requestType == DropTrigReq::RT_LQH) {
+ jam();
+ // part of alter trigger operation
+ OpAlterTriggerPtr opPtr;
+ c_opAlterTrigger.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ alterTrigger_fromDropLocal(signal, opPtr);
+ return;
+ }
+ OpDropTriggerPtr opPtr;
+ c_opDropTrigger.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ ndbrequire(opPtr.p->m_isMaster);
+ ndbrequire(opPtr.p->m_requestType == requestType);
+ opPtr.p->setError(ref);
+ opPtr.p->m_signalCounter.clearWaitingFor(refToNode(senderRef));
+ if (! opPtr.p->m_signalCounter.done()) {
+ jam();
+ return;
+ }
+ if (requestType == DropTrigReq::RT_DICT_COMMIT ||
+ requestType == DropTrigReq::RT_DICT_ABORT) {
+ jam();
+ // send reply to user
+ dropTrigger_sendReply(signal, opPtr, true);
+ c_opDropTrigger.release(opPtr);
+ return;
+ }
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = DropTrigReq::RT_DICT_ABORT;
+ dropTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (requestType == DropTrigReq::RT_DICT_PREPARE) {
+ jam();
+ // start alter offline
+ dropTrigger_toAlterTrigger(signal, opPtr);
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::dropTrigger_slavePrepare(Signal* signal, OpDropTriggerPtr opPtr)
+{
+ jam();
+}
+
+void
+Dbdict::dropTrigger_toAlterTrigger(Signal* signal, OpDropTriggerPtr opPtr)
+{
+ jam();
+ AlterTrigReq* req = (AlterTrigReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(AlterTrigReq::RT_DROP_TRIGGER);
+ req->setTableId(opPtr.p->m_request.getTableId());
+ req->setTriggerId(opPtr.p->m_request.getTriggerId());
+ req->setTriggerInfo(0); // not used
+ req->setOnline(false);
+ req->setReceiverRef(0);
+ sendSignal(reference(), GSN_ALTER_TRIG_REQ,
+ signal, AlterTrigReq::SignalLength, JBB);
+}
+
+void
+Dbdict::dropTrigger_fromAlterTrigger(Signal* signal, OpDropTriggerPtr opPtr)
+{
+ jam();
+ // remove in all
+ opPtr.p->m_requestType = DropTrigReq::RT_DICT_COMMIT;
+ dropTrigger_sendSlaveReq(signal, opPtr);
+}
+
+void
+Dbdict::dropTrigger_sendSlaveReq(Signal* signal, OpDropTriggerPtr opPtr)
+{
+ DropTrigReq* const req = (DropTrigReq*)signal->getDataPtrSend();
+ *req = opPtr.p->m_request;
+ req->setUserRef(opPtr.p->m_coordinatorRef);
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(opPtr.p->m_requestType);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ opPtr.p->m_signalCounter = c_aliveNodes;
+ NodeReceiverGroup rg(DBDICT, c_aliveNodes);
+ sendSignal(rg, GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+}
+
+void
+Dbdict::dropTrigger_slaveCommit(Signal* signal, OpDropTriggerPtr opPtr)
+{
+ jam();
+ const DropTrigReq* const req = &opPtr.p->m_request;
+ // get trigger record
+ const Uint32 triggerId = req->getTriggerId();
+ TriggerRecordPtr triggerPtr;
+ c_triggerRecordPool.getPtr(triggerPtr, triggerId);
+ if (triggerPtr.p->triggerType == TriggerType::SECONDARY_INDEX ||
+ triggerPtr.p->triggerType == TriggerType::ORDERED_INDEX) {
+ jam();
+ // disconnect from index if index trigger XXX move to drop index
+ triggerPtr.p->indexId = req->getIndexId();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, triggerPtr.p->indexId);
+ ndbrequire(! indexPtr.isNull());
+ switch (triggerPtr.p->triggerEvent) {
+ case TriggerEvent::TE_INSERT:
+ indexPtr.p->insertTriggerId = RNIL;
+ break;
+ case TriggerEvent::TE_UPDATE:
+ indexPtr.p->updateTriggerId = RNIL;
+ break;
+ case TriggerEvent::TE_DELETE:
+ indexPtr.p->deleteTriggerId = RNIL;
+ break;
+ case TriggerEvent::TE_CUSTOM:
+ indexPtr.p->customTriggerId = RNIL;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+ }
+ if (triggerPtr.p->triggerType == TriggerType::READ_ONLY_CONSTRAINT) {
+ jam();
+ // disconnect from index record XXX should be done in caller instead
+ triggerPtr.p->indexId = req->getTableId();
+ TableRecordPtr indexPtr;
+ c_tableRecordPool.getPtr(indexPtr, triggerPtr.p->indexId);
+ indexPtr.p->buildTriggerId = RNIL;
+ }
+ // remove trigger
+ // ndbout_c("++++++++++++ Removing trigger id %u, %s", triggerPtr.p->triggerId, triggerPtr.p->triggerName);
+ c_triggerRecordHash.remove(triggerPtr);
+ triggerPtr.p->triggerState = TriggerRecord::TS_NOT_DEFINED;
+}
+
+void
+Dbdict::dropTrigger_slaveAbort(Signal* signal, OpDropTriggerPtr opPtr)
+{
+ jam();
+}
+
+void
+Dbdict::dropTrigger_sendReply(Signal* signal, OpDropTriggerPtr opPtr,
+ bool toUser)
+{
+ DropTrigRef* rep = (DropTrigRef*)signal->getDataPtrSend();
+ Uint32 gsn = GSN_DROP_TRIG_CONF;
+ Uint32 length = DropTrigConf::InternalLength;
+ bool sendRef = opPtr.p->hasError();
+ if (! toUser) {
+ rep->setUserRef(opPtr.p->m_coordinatorRef);
+ rep->setConnectionPtr(opPtr.p->key);
+ rep->setRequestType(opPtr.p->m_requestType);
+ if (opPtr.p->m_requestType == DropTrigReq::RT_DICT_ABORT)
+ sendRef = false;
+ } else {
+ rep->setUserRef(opPtr.p->m_request.getUserRef());
+ rep->setConnectionPtr(opPtr.p->m_request.getConnectionPtr());
+ rep->setRequestType(opPtr.p->m_request.getRequestType());
+ length = DropTrigConf::SignalLength;
+ }
+ rep->setTableId(opPtr.p->m_request.getTableId());
+ rep->setIndexId(opPtr.p->m_request.getIndexId());
+ rep->setTriggerId(opPtr.p->m_request.getTriggerId());
+ if (sendRef) {
+ if (opPtr.p->m_errorNode == 0)
+ opPtr.p->m_errorNode = getOwnNodeId();
+ rep->setErrorCode(opPtr.p->m_errorCode);
+ rep->setErrorLine(opPtr.p->m_errorLine);
+ rep->setErrorNode(opPtr.p->m_errorNode);
+ gsn = GSN_DROP_TRIG_REF;
+ length = CreateTrigRef::SignalLength;
+ }
+ sendSignal(rep->getUserRef(), gsn, signal, length, JBB);
+}
+
+/**
+ * MODULE: Alter trigger.
+ *
+ * Alter trigger state. Alter online creates the trigger first in all
+ * TC (if index trigger) and then in all LQH-TUP.
+ *
+ * Request type received in REQ and returned in CONF/REF:
+ *
+ * RT_USER - normal user e.g. BACKUP
+ * RT_CREATE_TRIGGER - from create trigger
+ * RT_DROP_TRIGGER - from drop trigger
+ * RT_DICT_PREPARE - seize operations and check request
+ * RT_DICT_TC - master to each DICT on way to TC
+ * RT_DICT_LQH - master to each DICT on way to LQH-TUP
+ * RT_DICT_COMMIT - commit state change in each DICT (no reply)
+ */
+
+void
+Dbdict::execALTER_TRIG_REQ(Signal* signal)
+{
+ jamEntry();
+ AlterTrigReq* const req = (AlterTrigReq*)signal->getDataPtrSend();
+ OpAlterTriggerPtr opPtr;
+ const Uint32 senderRef = signal->senderBlockRef();
+ const AlterTrigReq::RequestType requestType = req->getRequestType();
+ if (requestType == AlterTrigReq::RT_USER ||
+ requestType == AlterTrigReq::RT_CREATE_TRIGGER ||
+ requestType == AlterTrigReq::RT_DROP_TRIGGER) {
+ jam();
+ const bool isLocal = req->getRequestFlag() & RequestFlag::RF_LOCAL;
+ NdbNodeBitmask receiverNodes = c_aliveNodes;
+ if (isLocal) {
+ receiverNodes.clear();
+ receiverNodes.set(getOwnNodeId());
+ }
+ if (signal->getLength() == AlterTrigReq::SignalLength) {
+ jam();
+ if (! isLocal && getOwnNodeId() != c_masterNodeId) {
+ jam();
+ // forward to DICT master
+ sendSignal(calcDictBlockRef(c_masterNodeId), GSN_ALTER_TRIG_REQ,
+ signal, AlterTrigReq::SignalLength, JBB);
+ return;
+ }
+ // forward initial request plus operation key to all
+ req->setOpKey(++c_opRecordSequence);
+ NodeReceiverGroup rg(DBDICT, receiverNodes);
+ sendSignal(rg, GSN_ALTER_TRIG_REQ,
+ signal, AlterTrigReq::SignalLength + 1, JBB);
+ return;
+ }
+ // seize operation record
+ ndbrequire(signal->getLength() == AlterTrigReq::SignalLength + 1);
+ const Uint32 opKey = req->getOpKey();
+ OpAlterTrigger opBusy;
+ if (! c_opAlterTrigger.seize(opPtr))
+ opPtr.p = &opBusy;
+ opPtr.p->save(req);
+ opPtr.p->m_coordinatorRef = senderRef;
+ opPtr.p->m_isMaster = (senderRef == reference());
+ opPtr.p->key = opKey;
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_PREPARE;
+ if (opPtr.p == &opBusy) {
+ jam();
+ opPtr.p->m_errorCode = AlterTrigRef::Busy;
+ opPtr.p->m_errorLine = __LINE__;
+ alterTrigger_sendReply(signal, opPtr, opPtr.p->m_isMaster);
+ return;
+ }
+ c_opAlterTrigger.add(opPtr);
+ // master expects to hear from all
+ if (opPtr.p->m_isMaster) {
+ opPtr.p->m_nodes = receiverNodes;
+ opPtr.p->m_signalCounter = receiverNodes;
+ }
+ alterTrigger_slavePrepare(signal, opPtr);
+ alterTrigger_sendReply(signal, opPtr, false);
+ return;
+ }
+ c_opAlterTrigger.find(opPtr, req->getConnectionPtr());
+ if (! opPtr.isNull()) {
+ opPtr.p->m_requestType = requestType;
+ if (requestType == AlterTrigReq::RT_DICT_TC ||
+ requestType == AlterTrigReq::RT_DICT_LQH) {
+ jam();
+ if (req->getOnline())
+ alterTrigger_toCreateLocal(signal, opPtr);
+ else
+ alterTrigger_toDropLocal(signal, opPtr);
+ return;
+ }
+ if (requestType == AlterTrigReq::RT_DICT_COMMIT ||
+ requestType == AlterTrigReq::RT_DICT_ABORT) {
+ jam();
+ if (requestType == AlterTrigReq::RT_DICT_COMMIT)
+ alterTrigger_slaveCommit(signal, opPtr);
+ else
+ alterTrigger_slaveAbort(signal, opPtr);
+ alterTrigger_sendReply(signal, opPtr, false);
+ // done in slave
+ if (! opPtr.p->m_isMaster)
+ c_opAlterTrigger.release(opPtr);
+ return;
+ }
+ }
+ jam();
+ // return to sender
+ OpAlterTrigger opBad;
+ opPtr.p = &opBad;
+ opPtr.p->save(req);
+ opPtr.p->m_errorCode = AlterTrigRef::BadRequestType;
+ opPtr.p->m_errorLine = __LINE__;
+ alterTrigger_sendReply(signal, opPtr, true);
+ return;
+}
+
+void
+Dbdict::execALTER_TRIG_CONF(Signal* signal)
+{
+ jamEntry();
+ AlterTrigConf* conf = (AlterTrigConf*)signal->getDataPtrSend();
+ alterTrigger_recvReply(signal, conf, 0);
+}
+
+void
+Dbdict::execALTER_TRIG_REF(Signal* signal)
+{
+ jamEntry();
+ AlterTrigRef* ref = (AlterTrigRef*)signal->getDataPtrSend();
+ alterTrigger_recvReply(signal, ref->getConf(), ref);
+}
+
+void
+Dbdict::alterTrigger_recvReply(Signal* signal, const AlterTrigConf* conf,
+ const AlterTrigRef* ref)
+{
+ jam();
+ const Uint32 senderRef = signal->senderBlockRef();
+ const AlterTrigReq::RequestType requestType = conf->getRequestType();
+ const Uint32 key = conf->getConnectionPtr();
+ if (requestType == AlterTrigReq::RT_CREATE_TRIGGER) {
+ jam();
+ // part of create trigger operation
+ OpCreateTriggerPtr opPtr;
+ c_opCreateTrigger.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ createTrigger_fromAlterTrigger(signal, opPtr);
+ return;
+ }
+ if (requestType == AlterTrigReq::RT_DROP_TRIGGER) {
+ jam();
+ // part of drop trigger operation
+ OpDropTriggerPtr opPtr;
+ c_opDropTrigger.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ opPtr.p->setError(ref);
+ dropTrigger_fromAlterTrigger(signal, opPtr);
+ return;
+ }
+ OpAlterTriggerPtr opPtr;
+ c_opAlterTrigger.find(opPtr, key);
+ ndbrequire(! opPtr.isNull());
+ ndbrequire(opPtr.p->m_isMaster);
+ ndbrequire(opPtr.p->m_requestType == requestType);
+ /*
+ * If refuse on drop trig, because of non-existent trigger,
+ * comes from anyone but the master node - ignore it and
+ * remove the node from forter ALTER_TRIG communication
+ * This will happen if a new node has started since the
+ * trigger whas created.
+ */
+ if (ref &&
+ refToNode(senderRef) != refToNode(reference()) &&
+ opPtr.p->m_request.getRequestType() == AlterTrigReq::RT_DROP_TRIGGER &&
+ ref->getErrorCode() == AlterTrigRef::TriggerNotFound) {
+ jam();
+ ref = 0; // ignore this error
+ opPtr.p->m_nodes.clear(refToNode(senderRef)); // remove this from group
+ }
+ opPtr.p->setError(ref);
+ opPtr.p->m_signalCounter.clearWaitingFor(refToNode(senderRef));
+ if (! opPtr.p->m_signalCounter.done()) {
+ jam();
+ return;
+ }
+ if (requestType == AlterTrigReq::RT_DICT_COMMIT ||
+ requestType == AlterTrigReq::RT_DICT_ABORT) {
+ jam();
+ // send reply to user
+ alterTrigger_sendReply(signal, opPtr, true);
+ c_opAlterTrigger.release(opPtr);
+ return;
+ }
+ if (opPtr.p->hasError()) {
+ jam();
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_ABORT;
+ alterTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (! (opPtr.p->m_request.getRequestFlag() & RequestFlag::RF_NOTCTRIGGER)) {
+ if (requestType == AlterTrigReq::RT_DICT_PREPARE) {
+ jam();
+ if (opPtr.p->m_request.getOnline())
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_TC;
+ else
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_LQH;
+ alterTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (requestType == AlterTrigReq::RT_DICT_TC) {
+ jam();
+ if (opPtr.p->m_request.getOnline())
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_LQH;
+ else
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_COMMIT;
+ alterTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (requestType == AlterTrigReq::RT_DICT_LQH) {
+ jam();
+ if (opPtr.p->m_request.getOnline())
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_COMMIT;
+ else
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_TC;
+ alterTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ } else {
+ if (requestType == AlterTrigReq::RT_DICT_PREPARE) {
+ jam();
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_LQH;
+ alterTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ if (requestType == AlterTrigReq::RT_DICT_LQH) {
+ jam();
+ opPtr.p->m_requestType = AlterTrigReq::RT_DICT_COMMIT;
+ alterTrigger_sendSlaveReq(signal, opPtr);
+ return;
+ }
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::alterTrigger_slavePrepare(Signal* signal, OpAlterTriggerPtr opPtr)
+{
+ jam();
+ const AlterTrigReq* const req = &opPtr.p->m_request;
+ const Uint32 triggerId = req->getTriggerId();
+ TriggerRecordPtr triggerPtr;
+ if (! (triggerId < c_triggerRecordPool.getSize())) {
+ jam();
+ opPtr.p->m_errorCode = AlterTrigRef::TriggerNotFound;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+ c_triggerRecordPool.getPtr(triggerPtr, triggerId);
+ if (triggerPtr.p->triggerState == TriggerRecord::TS_NOT_DEFINED) {
+ jam();
+ opPtr.p->m_errorCode = AlterTrigRef::TriggerNotFound;
+ opPtr.p->m_errorLine = __LINE__;
+ return;
+ }
+}
+
+void
+Dbdict::alterTrigger_toCreateLocal(Signal* signal, OpAlterTriggerPtr opPtr)
+{
+ jam();
+ // find trigger record
+ const Uint32 triggerId = opPtr.p->m_request.getTriggerId();
+ TriggerRecordPtr triggerPtr;
+ c_triggerRecordPool.getPtr(triggerPtr, triggerId);
+ CreateTrigReq* const req = (CreateTrigReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_TC) {
+ req->setRequestType(CreateTrigReq::RT_TC);
+ } else if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_LQH) {
+ req->setRequestType(CreateTrigReq::RT_LQH);
+ } else {
+ ndbassert(false);
+ }
+ req->setTableId(triggerPtr.p->tableId);
+ req->setIndexId(triggerPtr.p->indexId);
+ req->setTriggerId(triggerPtr.i);
+ req->setTriggerType(triggerPtr.p->triggerType);
+ req->setTriggerActionTime(triggerPtr.p->triggerActionTime);
+ req->setTriggerEvent(triggerPtr.p->triggerEvent);
+ req->setMonitorReplicas(triggerPtr.p->monitorReplicas);
+ req->setMonitorAllAttributes(triggerPtr.p->monitorAllAttributes);
+ req->setOnline(true);
+ req->setReceiverRef(opPtr.p->m_request.getReceiverRef());
+ BlockReference blockRef = 0;
+ if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_TC) {
+ blockRef = calcTcBlockRef(getOwnNodeId());
+ } else if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_LQH) {
+ blockRef = calcLqhBlockRef(getOwnNodeId());
+ } else {
+ ndbassert(false);
+ }
+ req->setAttributeMask(triggerPtr.p->attributeMask);
+ sendSignal(blockRef, GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength, JBB);
+}
+
+void
+Dbdict::alterTrigger_fromCreateLocal(Signal* signal, OpAlterTriggerPtr opPtr)
+{
+ jam();
+ if (! opPtr.p->hasError()) {
+ // mark created locally
+ TriggerRecordPtr triggerPtr;
+ c_triggerRecordPool.getPtr(triggerPtr, opPtr.p->m_request.getTriggerId());
+ if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_TC) {
+ triggerPtr.p->triggerLocal |= TriggerRecord::TL_CREATED_TC;
+ } else if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_LQH) {
+ triggerPtr.p->triggerLocal |= TriggerRecord::TL_CREATED_LQH;
+ } else {
+ ndbrequire(false);
+ }
+ }
+ // forward CONF or REF to master
+ alterTrigger_sendReply(signal, opPtr, false);
+}
+
+void
+Dbdict::alterTrigger_toDropLocal(Signal* signal, OpAlterTriggerPtr opPtr)
+{
+ jam();
+ TriggerRecordPtr triggerPtr;
+ c_triggerRecordPool.getPtr(triggerPtr, opPtr.p->m_request.getTriggerId());
+ DropTrigReq* const req = (DropTrigReq*)signal->getDataPtrSend();
+ req->setUserRef(reference());
+ req->setConnectionPtr(opPtr.p->key);
+ if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_TC) {
+ // broken trigger
+ if (! (triggerPtr.p->triggerLocal & TriggerRecord::TL_CREATED_TC)) {
+ jam();
+ alterTrigger_sendReply(signal, opPtr, false);
+ return;
+ }
+ req->setRequestType(DropTrigReq::RT_TC);
+ } else if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_LQH) {
+ // broken trigger
+ if (! (triggerPtr.p->triggerLocal & TriggerRecord::TL_CREATED_LQH)) {
+ jam();
+ alterTrigger_sendReply(signal, opPtr, false);
+ return;
+ }
+ req->setRequestType(DropTrigReq::RT_LQH);
+ } else {
+ ndbassert(false);
+ }
+ req->setTableId(triggerPtr.p->tableId);
+ req->setIndexId(triggerPtr.p->indexId);
+ req->setTriggerId(triggerPtr.i);
+ req->setTriggerType(triggerPtr.p->triggerType);
+ req->setTriggerActionTime(triggerPtr.p->triggerActionTime);
+ req->setTriggerEvent(triggerPtr.p->triggerEvent);
+ req->setMonitorReplicas(triggerPtr.p->monitorReplicas);
+ req->setMonitorAllAttributes(triggerPtr.p->monitorAllAttributes);
+ BlockReference blockRef = 0;
+ if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_TC) {
+ blockRef = calcTcBlockRef(getOwnNodeId());
+ } else if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_LQH) {
+ blockRef = calcLqhBlockRef(getOwnNodeId());
+ } else {
+ ndbassert(false);
+ }
+ sendSignal(blockRef, GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+}
+
+void
+Dbdict::alterTrigger_fromDropLocal(Signal* signal, OpAlterTriggerPtr opPtr)
+{
+ jam();
+ if (! opPtr.p->hasError()) {
+ // mark dropped locally
+ TriggerRecordPtr triggerPtr;
+ c_triggerRecordPool.getPtr(triggerPtr, opPtr.p->m_request.getTriggerId());
+ if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_TC) {
+ triggerPtr.p->triggerLocal &= ~TriggerRecord::TL_CREATED_TC;
+ } else if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_LQH) {
+ triggerPtr.p->triggerLocal &= ~TriggerRecord::TL_CREATED_LQH;
+ } else {
+ ndbrequire(false);
+ }
+ }
+ // forward CONF or REF to master
+ alterTrigger_sendReply(signal, opPtr, false);
+}
+
+void
+Dbdict::alterTrigger_slaveCommit(Signal* signal, OpAlterTriggerPtr opPtr)
+{
+ jam();
+ TriggerRecordPtr triggerPtr;
+ c_triggerRecordPool.getPtr(triggerPtr, opPtr.p->m_request.getTriggerId());
+ // set state
+ triggerPtr.p->triggerState = TriggerRecord::TS_ONLINE;
+}
+
+void
+Dbdict::alterTrigger_slaveAbort(Signal* signal, OpAlterTriggerPtr opPtr)
+{
+ jam();
+}
+
+void
+Dbdict::alterTrigger_sendSlaveReq(Signal* signal, OpAlterTriggerPtr opPtr)
+{
+ AlterTrigReq* const req = (AlterTrigReq*)signal->getDataPtrSend();
+ *req = opPtr.p->m_request;
+ req->setUserRef(opPtr.p->m_coordinatorRef);
+ req->setConnectionPtr(opPtr.p->key);
+ req->setRequestType(opPtr.p->m_requestType);
+ req->addRequestFlag(opPtr.p->m_requestFlag);
+ NdbNodeBitmask receiverNodes = c_aliveNodes;
+ if (opPtr.p->m_requestFlag & RequestFlag::RF_LOCAL) {
+ receiverNodes.clear();
+ receiverNodes.set(getOwnNodeId());
+ } else {
+ opPtr.p->m_nodes.bitAND(receiverNodes);
+ receiverNodes = opPtr.p->m_nodes;
+ }
+ opPtr.p->m_signalCounter = receiverNodes;
+ NodeReceiverGroup rg(DBDICT, receiverNodes);
+ sendSignal(rg, GSN_ALTER_TRIG_REQ,
+ signal, AlterTrigReq::SignalLength, JBB);
+}
+
+void
+Dbdict::alterTrigger_sendReply(Signal* signal, OpAlterTriggerPtr opPtr,
+ bool toUser)
+{
+ jam();
+ AlterTrigRef* rep = (AlterTrigRef*)signal->getDataPtrSend();
+ Uint32 gsn = GSN_ALTER_TRIG_CONF;
+ Uint32 length = AlterTrigConf::InternalLength;
+ bool sendRef = opPtr.p->hasError();
+ if (! toUser) {
+ rep->setUserRef(opPtr.p->m_coordinatorRef);
+ rep->setConnectionPtr(opPtr.p->key);
+ rep->setRequestType(opPtr.p->m_requestType);
+ if (opPtr.p->m_requestType == AlterTrigReq::RT_DICT_ABORT) {
+ jam();
+ sendRef = false;
+ } else {
+ jam();
+ }
+ } else {
+ jam();
+ rep->setUserRef(opPtr.p->m_request.getUserRef());
+ rep->setConnectionPtr(opPtr.p->m_request.getConnectionPtr());
+ rep->setRequestType(opPtr.p->m_request.getRequestType());
+ length = AlterTrigConf::SignalLength;
+ }
+ rep->setTableId(opPtr.p->m_request.getTableId());
+ rep->setTriggerId(opPtr.p->m_request.getTriggerId());
+ if (sendRef) {
+ if (opPtr.p->m_errorNode == 0) {
+ jam();
+ opPtr.p->m_errorNode = getOwnNodeId();
+ } else {
+ jam();
+ }
+ rep->setErrorCode(opPtr.p->m_errorCode);
+ rep->setErrorLine(opPtr.p->m_errorLine);
+ rep->setErrorNode(opPtr.p->m_errorNode);
+ gsn = GSN_ALTER_TRIG_REF;
+ length = AlterTrigRef::SignalLength;
+ }
+ sendSignal(rep->getUserRef(), gsn, signal, length, JBB);
+}
+
+/**
+ * MODULE: Support routines for index and trigger.
+ */
+
+void
+Dbdict::getTableKeyList(TableRecordPtr tablePtr, AttributeList& list)
+{
+ jam();
+ list.sz = 0;
+ for (Uint32 tAttr = tablePtr.p->firstAttribute; tAttr != RNIL; ) {
+ AttributeRecord* aRec = c_attributeRecordPool.getPtr(tAttr);
+ if (aRec->tupleKey)
+ list.id[list.sz++] = aRec->attributeId;
+ tAttr = aRec->nextAttrInTable;
+ }
+}
+
+// XXX should store the primary attribute id
+void
+Dbdict::getIndexAttr(TableRecordPtr indexPtr, Uint32 itAttr, Uint32* id)
+{
+ jam();
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, indexPtr.p->primaryTableId);
+ AttributeRecord* iaRec = c_attributeRecordPool.getPtr(itAttr);
+ for (Uint32 tAttr = tablePtr.p->firstAttribute; tAttr != RNIL; ) {
+ AttributeRecord* aRec = c_attributeRecordPool.getPtr(tAttr);
+ if (iaRec->equal(*aRec)) {
+ id[0] = aRec->attributeId;
+ return;
+ }
+ tAttr = aRec->nextAttrInTable;
+ }
+ ndbrequire(false);
+}
+
+void
+Dbdict::getIndexAttrList(TableRecordPtr indexPtr, AttributeList& list)
+{
+ jam();
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, indexPtr.p->primaryTableId);
+ list.sz = 0;
+ memset(list.id, 0, sizeof(list.id));
+ ndbrequire(indexPtr.p->noOfAttributes >= 2);
+ Uint32 itAttr = indexPtr.p->firstAttribute;
+ for (Uint32 i = 0; i < (Uint32)indexPtr.p->noOfAttributes - 1; i++) {
+ getIndexAttr(indexPtr, itAttr, &list.id[list.sz++]);
+ AttributeRecord* iaRec = c_attributeRecordPool.getPtr(itAttr);
+ itAttr = iaRec->nextAttrInTable;
+ }
+}
+
+void
+Dbdict::getIndexAttrMask(TableRecordPtr indexPtr, AttributeMask& mask)
+{
+ jam();
+ TableRecordPtr tablePtr;
+ c_tableRecordPool.getPtr(tablePtr, indexPtr.p->primaryTableId);
+ mask.clear();
+ ndbrequire(indexPtr.p->noOfAttributes >= 2);
+ Uint32 itAttr = indexPtr.p->firstAttribute;
+ for (Uint32 i = 0; i < (Uint32)indexPtr.p->noOfAttributes - 1; i++) {
+ Uint32 id;
+ getIndexAttr(indexPtr, itAttr, &id);
+ mask.set(id);
+ AttributeRecord* iaRec = c_attributeRecordPool.getPtr(itAttr);
+ itAttr = iaRec->nextAttrInTable;
+ }
+}
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* MODULE: STORE/RESTORE SCHEMA FILE---------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* General module used to store the schema file on disk and */
+/* similar function to restore it from disk. */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+void
+Dbdict::initSchemaFile(SchemaFile * sf, Uint32 fileSz){
+ memcpy(sf->Magic, "NDBSCHMA", sizeof(sf->Magic));
+ sf->ByteOrder = 0x12345678;
+ sf->NdbVersion = NDB_VERSION;
+ sf->FileSize = fileSz;
+ sf->CheckSum = 0;
+
+ Uint32 headSz = (sizeof(SchemaFile)-sizeof(SchemaFile::TableEntry));
+ Uint32 noEntries = (fileSz - headSz) / sizeof(SchemaFile::TableEntry);
+ Uint32 slack = (fileSz - headSz) - noEntries * sizeof(SchemaFile::TableEntry);
+
+ ndbrequire(noEntries > MAX_TABLES);
+
+ sf->NoOfTableEntries = noEntries;
+ memset(sf->TableEntries, 0, sizeof(noEntries*sizeof(SchemaFile::TableEntry)));
+ memset(&(sf->TableEntries[noEntries]), 0, slack);
+ computeChecksum(sf);
+}
+
+void
+Dbdict::computeChecksum(SchemaFile * sf){
+ sf->CheckSum = 0;
+ sf->CheckSum = computeChecksum((const Uint32*)sf, sf->FileSize/4);
+}
+
+bool
+Dbdict::validateChecksum(const SchemaFile * sf){
+
+ Uint32 c = computeChecksum((const Uint32*)sf, sf->FileSize/4);
+ return c == 0;
+}
+
+Uint32
+Dbdict::computeChecksum(const Uint32 * src, Uint32 len){
+ Uint32 ret = 0;
+ for(Uint32 i = 0; i<len; i++)
+ ret ^= src[i];
+ return ret;
+}
+
+SchemaFile::TableEntry *
+Dbdict::getTableEntry(void * p, Uint32 tableId, bool allowTooBig){
+ SchemaFile * sf = (SchemaFile*)p;
+
+ ndbrequire(allowTooBig || tableId < sf->NoOfTableEntries);
+ return &sf->TableEntries[tableId];
+}
+
+// global metadata support
+
+int
+Dbdict::getMetaTablePtr(TableRecordPtr& tablePtr, Uint32 tableId, Uint32 tableVersion)
+{
+ if (tableId >= c_tableRecordPool.getSize()) {
+ return MetaData::InvalidArgument;
+ }
+ c_tableRecordPool.getPtr(tablePtr, tableId);
+ if (tablePtr.p->tabState == TableRecord::NOT_DEFINED) {
+ return MetaData::TableNotFound;
+ }
+ if (tablePtr.p->tableVersion != tableVersion) {
+ return MetaData::InvalidTableVersion;
+ }
+ // online flag is not maintained by DICT
+ tablePtr.p->online =
+ tablePtr.p->isTable() && tablePtr.p->tabState == TableRecord::DEFINED ||
+ tablePtr.p->isIndex() && tablePtr.p->indexState == TableRecord::IS_ONLINE;
+ return 0;
+}
+
+int
+Dbdict::getMetaTable(MetaData::Table& table, Uint32 tableId, Uint32 tableVersion)
+{
+ int ret;
+ TableRecordPtr tablePtr;
+ if ((ret = getMetaTablePtr(tablePtr, tableId, tableVersion)) < 0) {
+ return ret;
+ }
+ new (&table) MetaData::Table(*tablePtr.p);
+ return 0;
+}
+
+int
+Dbdict::getMetaTable(MetaData::Table& table, const char* tableName)
+{
+ int ret;
+ TableRecordPtr tablePtr;
+ if (strlen(tableName) + 1 > MAX_TAB_NAME_SIZE) {
+ return MetaData::InvalidArgument;
+ }
+ TableRecord keyRecord;
+ strcpy(keyRecord.tableName, tableName);
+ c_tableRecordHash.find(tablePtr, keyRecord);
+ if (tablePtr.i == RNIL) {
+ return MetaData::TableNotFound;
+ }
+ if ((ret = getMetaTablePtr(tablePtr, tablePtr.i, tablePtr.p->tableVersion)) < 0) {
+ return ret;
+ }
+ new (&table) MetaData::Table(*tablePtr.p);
+ return 0;
+}
+
+int
+Dbdict::getMetaAttribute(MetaData::Attribute& attr, const MetaData::Table& table, Uint32 attributeId)
+{
+ int ret;
+ TableRecordPtr tablePtr;
+ if ((ret = getMetaTablePtr(tablePtr, table.tableId, table.tableVersion)) < 0) {
+ return ret;
+ }
+ AttributeRecordPtr attrPtr;
+ attrPtr.i = tablePtr.p->firstAttribute;
+ while (attrPtr.i != RNIL) {
+ c_attributeRecordPool.getPtr(attrPtr);
+ if (attrPtr.p->attributeId == attributeId)
+ break;
+ attrPtr.i = attrPtr.p->nextAttrInTable;
+ }
+ if (attrPtr.i == RNIL) {
+ return MetaData::AttributeNotFound;
+ }
+ new (&attr) MetaData::Attribute(*attrPtr.p);
+ return 0;
+}
+
+int
+Dbdict::getMetaAttribute(MetaData::Attribute& attr, const MetaData::Table& table, const char* attributeName)
+{
+ int ret;
+ TableRecordPtr tablePtr;
+ if ((ret = getMetaTablePtr(tablePtr, table.tableId, table.tableVersion)) < 0) {
+ return ret;
+ }
+ AttributeRecordPtr attrPtr;
+ attrPtr.i = tablePtr.p->firstAttribute;
+ while (attrPtr.i != RNIL) {
+ c_attributeRecordPool.getPtr(attrPtr);
+ if (strcmp(attrPtr.p->attributeName, attributeName) == 0)
+ break;
+ attrPtr.i = attrPtr.p->nextAttrInTable;
+ }
+ if (attrPtr.i == RNIL) {
+ return MetaData::AttributeNotFound;
+ }
+ new (&attr) MetaData::Attribute(*attrPtr.p);
+ return 0;
+}
diff --git a/ndb/src/kernel/blocks/dbdict/Dbdict.hpp b/ndb/src/kernel/blocks/dbdict/Dbdict.hpp
new file mode 100644
index 00000000000..02e2d4496bf
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/Dbdict.hpp
@@ -0,0 +1,1987 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBDICT_H
+#define DBDICT_H
+
+/**
+ * Dict : Dictionary Block
+ */
+
+#include <ndb_limits.h>
+#include <trigger_definitions.h>
+#include <pc.hpp>
+#include <ArrayList.hpp>
+#include <DLHashTable.hpp>
+#include <CArray.hpp>
+#include <KeyTable2.hpp>
+#include <SimulatedBlock.hpp>
+#include <SimpleProperties.hpp>
+#include <SignalCounter.hpp>
+#include <Bitmask.hpp>
+#include <AttributeList.hpp>
+#include <signaldata/GetTableId.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/CreateTable.hpp>
+#include <signaldata/CreateTab.hpp>
+#include <signaldata/DropTable.hpp>
+#include <signaldata/AlterTable.hpp>
+#include <signaldata/AlterTab.hpp>
+#include <signaldata/CreateIndx.hpp>
+#include <signaldata/DropIndx.hpp>
+#include <signaldata/AlterIndx.hpp>
+#include <signaldata/BuildIndx.hpp>
+#include <signaldata/UtilPrepare.hpp>
+#include <signaldata/CreateEvnt.hpp>
+#include <signaldata/CreateTrig.hpp>
+#include <signaldata/DropTrig.hpp>
+#include <signaldata/AlterTrig.hpp>
+#include "SchemaFile.hpp"
+#include <blocks/mutexes.hpp>
+#include <SafeCounter.hpp>
+#include <RequestTracker.hpp>
+
+#ifdef DBDICT_C
+// Debug Macros
+
+/*--------------------------------------------------------------*/
+// Constants for CONTINUEB
+/*--------------------------------------------------------------*/
+#define ZPACK_TABLE_INTO_PAGES 0
+#define ZSEND_GET_TAB_RESPONSE 3
+
+
+/*--------------------------------------------------------------*/
+// Other constants in alphabetical order
+/*--------------------------------------------------------------*/
+#define ZNOMOREPHASES 255
+
+/*--------------------------------------------------------------*/
+// Schema file defines
+/*--------------------------------------------------------------*/
+#define ZSCHEMA_WORDS 4
+
+/*--------------------------------------------------------------*/
+// Page constants
+/*--------------------------------------------------------------*/
+#define ZALLOCATE 1 //Variable number of page for NDBFS
+#define ZPAGE_HEADER_SIZE 32
+#define ZPOS_PAGE_SIZE 16
+#define ZPOS_CHECKSUM 17
+#define ZPOS_VERSION 18
+#define ZPOS_PAGE_HEADER_SIZE 19
+
+/*--------------------------------------------------------------*/
+// Size constants
+/*--------------------------------------------------------------*/
+#define ZFS_CONNECT_SIZE 4
+#define ZSIZE_OF_PAGES_IN_WORDS 8192
+#define ZLOG_SIZE_OF_PAGES_IN_WORDS 13
+#define ZMAX_PAGES_OF_TABLE_DEFINITION 8
+#define ZNUMBER_OF_PAGES (2 * ZMAX_PAGES_OF_TABLE_DEFINITION + 2)
+#define ZNO_OF_FRAGRECORD 5
+
+/*--------------------------------------------------------------*/
+// Error codes
+/*--------------------------------------------------------------*/
+#define ZNODE_FAILURE_ERROR 704
+#endif
+
+/**
+ * DICT - This blocks handles all metadata
+ */
+class Dbdict: public SimulatedBlock {
+public:
+ /*
+ * 2.3 RECORD AND FILESIZES
+ */
+ /**
+ * Shared table / index record. Most of this is permanent data stored
+ * on disk. Index trigger ids are volatile.
+ */
+ struct TableRecord : public MetaData::Table {
+ /****************************************************
+ * Support variables for table handling
+ ****************************************************/
+
+ /* Active page which is sent to disk */
+ Uint32 activePage;
+
+ /** File pointer received from disk */
+ Uint32 filePtr[2];
+
+ /** Pointer to first attribute in table */
+ Uint32 firstAttribute;
+
+ /* Pointer to first page of table description */
+ Uint32 firstPage;
+
+ /** Pointer to last attribute in table */
+ Uint32 lastAttribute;
+
+ /* Temporary record used during add/drop table */
+ Uint32 myConnect;
+
+ /* Second table used by this table (for table reorg) */
+ Uint32 secondTable;
+
+ /* Next record in Pool */
+ Uint32 nextPool;
+
+ /* Next record in hash table */
+ Uint32 nextHash;
+
+ /* Previous record in Pool */
+ Uint32 prevPool;
+
+ /* Previous record in hash table */
+ Uint32 prevHash;
+
+ enum TabState {
+ NOT_DEFINED = 0,
+ REORG_TABLE_PREPARED = 1,
+ DEFINING = 2,
+ CHECKED = 3,
+ DEFINED = 4,
+ PREPARE_DROPPING = 5,
+ DROPPING = 6
+ };
+ TabState tabState;
+
+ /* State when returning from TC_SCHVERREQ */
+ enum TabReturnState {
+ TRS_IDLE = 0,
+ ADD_TABLE = 1,
+ SLAVE_SYSTEM_RESTART = 2,
+ MASTER_SYSTEM_RESTART = 3
+ };
+ TabReturnState tabReturnState;
+
+ /** Number of words */
+ Uint32 packedSize;
+
+ /** Index state (volatile data) */
+ enum IndexState {
+ IS_UNDEFINED = 0, // initial
+ IS_OFFLINE = 1, // index table created
+ IS_BUILDING = 2, // building (local state)
+ IS_DROPPING = 3, // dropping (local state)
+ IS_ONLINE = 4, // online
+ IS_BROKEN = 9 // build or drop aborted
+ };
+ IndexState indexState;
+
+ /** Trigger ids of index (volatile data) */
+ Uint32 insertTriggerId;
+ Uint32 updateTriggerId;
+ Uint32 deleteTriggerId;
+ Uint32 customTriggerId; // ordered index
+ Uint32 buildTriggerId; // temp during build
+
+ /** Index state in other blocks on this node */
+ enum IndexLocal {
+ IL_CREATED_TC = 1 << 0 // created in TC
+ };
+ Uint32 indexLocal;
+
+ inline bool equal(TableRecord & rec) const {
+ return strcmp(tableName, rec.tableName) == 0;
+ }
+
+ inline Uint32 hashValue() const {
+ Uint32 h = 0;
+ for (const char* p = tableName; *p != 0; p++)
+ h = (h << 5) + h + (*p);
+ return h;
+ }
+
+ /** frm data for this table */
+ /** TODO Could preferrably be made dynamic size */
+ Uint32 frmLen;
+ char frmData[MAX_FRM_DATA_SIZE];
+
+
+ };
+
+ typedef Ptr<TableRecord> TableRecordPtr;
+ ArrayPool<TableRecord> c_tableRecordPool;
+ DLHashTable<TableRecord> c_tableRecordHash;
+
+ /**
+ * Table attributes. Permanent data.
+ *
+ * Indexes have an attribute list which duplicates primary table
+ * attributes. This is wrong but convenient.
+ */
+ struct AttributeRecord : public MetaData::Attribute {
+ union {
+ /** Pointer to the next attribute used by ArrayPool */
+ Uint32 nextPool;
+
+ /** Pointer to the next attribute used by DLHash */
+ Uint32 nextHash;
+ };
+
+ /** Pointer to the previous attribute used by DLHash */
+ Uint32 prevHash;
+
+ /** Pointer to the next attribute in table */
+ Uint32 nextAttrInTable;
+
+ inline bool equal(AttributeRecord & rec) const {
+ return strcmp(attributeName, rec.attributeName) == 0;
+ }
+
+ inline Uint32 hashValue() const {
+ Uint32 h = 0;
+ for (const char* p = attributeName; *p != 0; p++)
+ h = (h << 5) + h + (*p);
+ return h;
+ }
+ };
+
+ typedef Ptr<AttributeRecord> AttributeRecordPtr;
+ ArrayPool<AttributeRecord> c_attributeRecordPool;
+ DLHashTable<AttributeRecord> c_attributeRecordHash;
+
+ /**
+ * Triggers. This is volatile data not saved on disk. Setting a
+ * trigger online creates the trigger in TC (if index) and LQH-TUP.
+ */
+ struct TriggerRecord {
+
+ /** Trigger state */
+ enum TriggerState {
+ TS_NOT_DEFINED = 0,
+ TS_DEFINING = 1,
+ TS_OFFLINE = 2, // created globally in DICT
+ TS_BUILDING = 3,
+ TS_DROPPING = 4,
+ TS_ONLINE = 5 // activated globally
+ };
+ TriggerState triggerState;
+
+ /** Trigger state in other blocks on this node */
+ enum IndexLocal {
+ TL_CREATED_TC = 1 << 0, // created in TC
+ TL_CREATED_LQH = 1 << 1 // created in LQH-TUP
+ };
+ Uint32 triggerLocal;
+
+ /** Trigger name, used by DICT to identify the trigger */
+ char triggerName[MAX_TAB_NAME_SIZE];
+
+ /** Trigger id, used by TRIX, TC, LQH, and TUP to identify the trigger */
+ Uint32 triggerId;
+
+ /** Table id, the table the trigger is defined on */
+ Uint32 tableId;
+
+ /** Trigger type, defines what the trigger is used for */
+ TriggerType::Value triggerType;
+
+ /** Trigger action time, defines when the trigger should fire */
+ TriggerActionTime::Value triggerActionTime;
+
+ /** Trigger event, defines what events the trigger should monitor */
+ TriggerEvent::Value triggerEvent;
+
+ /** Monitor all replicas */
+ bool monitorReplicas;
+
+ /** Monitor all, the trigger monitors changes of all attributes in table */
+ bool monitorAllAttributes;
+
+ /**
+ * Attribute mask, defines what attributes are to be monitored.
+ * Can be seen as a compact representation of SQL column name list.
+ */
+ AttributeMask attributeMask;
+
+ /** Index id, only used by secondary_index triggers */
+ Uint32 indexId;
+
+ union {
+ /** Pointer to the next attribute used by ArrayPool */
+ Uint32 nextPool;
+
+ /** Next record in hash table */
+ Uint32 nextHash;
+ };
+
+ /** Previous record in hash table */
+ Uint32 prevHash;
+
+ /** Equal function, used by DLHashTable */
+ inline bool equal(TriggerRecord & rec) const {
+ return strcmp(triggerName, rec.triggerName) == 0;
+ }
+
+ /** Hash value function, used by DLHashTable */
+ inline Uint32 hashValue() const {
+ Uint32 h = 0;
+ for (const char* p = triggerName; *p != 0; p++)
+ h = (h << 5) + h + (*p);
+ return h;
+ }
+ };
+
+ Uint32 c_maxNoOfTriggers;
+ typedef Ptr<TriggerRecord> TriggerRecordPtr;
+ ArrayPool<TriggerRecord> c_triggerRecordPool;
+ DLHashTable<TriggerRecord> c_triggerRecordHash;
+
+ /**
+ * Information for each FS connection.
+ ****************************************************************************/
+ struct FsConnectRecord {
+ enum FsState {
+ IDLE = 0,
+ OPEN_WRITE_SCHEMA = 1,
+ WRITE_SCHEMA = 2,
+ CLOSE_WRITE_SCHEMA = 3,
+ OPEN_READ_SCHEMA1 = 4,
+ OPEN_READ_SCHEMA2 = 5,
+ READ_SCHEMA1 = 6,
+ READ_SCHEMA2 = 7,
+ CLOSE_READ_SCHEMA = 8,
+ OPEN_READ_TAB_FILE1 = 9,
+ OPEN_READ_TAB_FILE2 = 10,
+ READ_TAB_FILE1 = 11,
+ READ_TAB_FILE2 = 12,
+ CLOSE_READ_TAB_FILE = 13,
+ OPEN_WRITE_TAB_FILE = 14,
+ WRITE_TAB_FILE = 15,
+ CLOSE_WRITE_TAB_FILE = 16
+ };
+ /** File Pointer for this file system connection */
+ Uint32 filePtr;
+
+ /** Reference of owner record */
+ Uint32 ownerPtr;
+
+ /** State of file system connection */
+ FsState fsState;
+
+ /** Used by Array Pool for free list handling */
+ Uint32 nextPool;
+ };
+
+ typedef Ptr<FsConnectRecord> FsConnectRecordPtr;
+ ArrayPool<FsConnectRecord> c_fsConnectRecordPool;
+
+ /**
+ * This record stores all the information about a node and all its attributes
+ ****************************************************************************/
+ struct NodeRecord {
+ enum NodeState {
+ API_NODE = 0,
+ NDB_NODE_ALIVE = 1,
+ NDB_NODE_DEAD = 2
+ };
+ bool hotSpare;
+ NodeState nodeState;
+ };
+
+ typedef Ptr<NodeRecord> NodeRecordPtr;
+ CArray<NodeRecord> c_nodes;
+ NdbNodeBitmask c_aliveNodes;
+
+ /**
+ * This record stores all the information about a table and all its attributes
+ ****************************************************************************/
+ struct PageRecord {
+ Uint32 word[8192];
+ };
+
+ typedef Ptr<PageRecord> PageRecordPtr;
+ CArray<PageRecord> c_pageRecordArray;
+
+ /**
+ * A page for create index table signal.
+ */
+ PageRecord c_indexPage;
+
+public:
+ Dbdict(const class Configuration &);
+ virtual ~Dbdict();
+
+private:
+ BLOCK_DEFINES(Dbdict);
+
+ // Signal receivers
+ void execDICTSTARTREQ(Signal* signal);
+
+ void execGET_TABINFOREQ(Signal* signal);
+ void execGET_TABLEDID_REQ(Signal* signal);
+ void execGET_TABINFO_REF(Signal* signal);
+ void execGET_TABINFO_CONF(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execHOT_SPAREREP(Signal* signal);
+ void execDIADDTABCONF(Signal* signal);
+ void execDIADDTABREF(Signal* signal);
+ void execTAB_COMMITCONF(Signal* signal);
+ void execTAB_COMMITREF(Signal* signal);
+ void execGET_SCHEMA_INFOREQ(Signal* signal);
+ void execSCHEMA_INFO(Signal* signal);
+ void execSCHEMA_INFOCONF(Signal* signal);
+ void execREAD_NODESCONF(Signal* signal);
+ void execFSCLOSECONF(Signal* signal);
+ void execFSCLOSEREF(Signal* signal);
+ void execFSOPENCONF(Signal* signal);
+ void execFSOPENREF(Signal* signal);
+ void execFSREADCONF(Signal* signal);
+ void execFSREADREF(Signal* signal);
+ void execFSWRITECONF(Signal* signal);
+ void execFSWRITEREF(Signal* signal);
+ void execNDB_STTOR(Signal* signal);
+ void execSIZEALT_REP(Signal* signal);
+ void execSTTOR(Signal* signal);
+ void execTC_SCHVERCONF(Signal* signal);
+ void execNODE_FAILREP(Signal* signal);
+ void execINCL_NODEREQ(Signal* signal);
+ void execAPI_FAILREQ(Signal* signal);
+
+ void execWAIT_GCP_REF(Signal* signal);
+ void execWAIT_GCP_CONF(Signal* signal);
+
+ void execLIST_TABLES_REQ(Signal* signal);
+
+ // Index signals
+ void execCREATE_INDX_REQ(Signal* signal);
+ void execCREATE_INDX_CONF(Signal* signal);
+ void execCREATE_INDX_REF(Signal* signal);
+
+ void execALTER_INDX_REQ(Signal* signal);
+ void execALTER_INDX_CONF(Signal* signal);
+ void execALTER_INDX_REF(Signal* signal);
+
+ void execCREATE_TABLE_CONF(Signal* signal);
+ void execCREATE_TABLE_REF(Signal* signal);
+
+ void execDROP_INDX_REQ(Signal* signal);
+ void execDROP_INDX_CONF(Signal* signal);
+ void execDROP_INDX_REF(Signal* signal);
+
+ void execDROP_TABLE_CONF(Signal* signal);
+ void execDROP_TABLE_REF(Signal* signal);
+
+ void execBUILDINDXREQ(Signal* signal);
+ void execBUILDINDXCONF(Signal* signal);
+ void execBUILDINDXREF(Signal* signal);
+
+ // Util signals used by Event code
+ void execUTIL_PREPARE_CONF(Signal* signal);
+ void execUTIL_PREPARE_REF (Signal* signal);
+ void execUTIL_EXECUTE_CONF(Signal* signal);
+ void execUTIL_EXECUTE_REF (Signal* signal);
+ void execUTIL_RELEASE_CONF(Signal* signal);
+ void execUTIL_RELEASE_REF (Signal* signal);
+
+
+ // Event signals from API
+ void execCREATE_EVNT_REQ (Signal* signal);
+ void execCREATE_EVNT_CONF(Signal* signal);
+ void execCREATE_EVNT_REF (Signal* signal);
+
+ void execDROP_EVNT_REQ (Signal* signal);
+
+ void execSUB_START_REQ (Signal* signal);
+ void execSUB_START_CONF (Signal* signal);
+ void execSUB_START_REF (Signal* signal);
+
+ void execSUB_STOP_REQ (Signal* signal);
+ void execSUB_STOP_CONF (Signal* signal);
+ void execSUB_STOP_REF (Signal* signal);
+
+ // Event signals from SUMA
+
+ void execCREATE_SUBID_CONF(Signal* signal);
+ void execCREATE_SUBID_REF (Signal* signal);
+
+ void execSUB_CREATE_CONF(Signal* signal);
+ void execSUB_CREATE_REF (Signal* signal);
+
+ void execSUB_SYNC_CONF(Signal* signal);
+ void execSUB_SYNC_REF (Signal* signal);
+
+ void execSUB_REMOVE_REQ(Signal* signal);
+ void execSUB_REMOVE_CONF(Signal* signal);
+ void execSUB_REMOVE_REF(Signal* signal);
+
+ // Trigger signals
+ void execCREATE_TRIG_REQ(Signal* signal);
+ void execCREATE_TRIG_CONF(Signal* signal);
+ void execCREATE_TRIG_REF(Signal* signal);
+ void execALTER_TRIG_REQ(Signal* signal);
+ void execALTER_TRIG_CONF(Signal* signal);
+ void execALTER_TRIG_REF(Signal* signal);
+ void execDROP_TRIG_REQ(Signal* signal);
+ void execDROP_TRIG_CONF(Signal* signal);
+ void execDROP_TRIG_REF(Signal* signal);
+
+ void execDROP_TABLE_REQ(Signal* signal);
+
+ void execPREP_DROP_TAB_REQ(Signal* signal);
+ void execPREP_DROP_TAB_REF(Signal* signal);
+ void execPREP_DROP_TAB_CONF(Signal* signal);
+
+ void execDROP_TAB_REQ(Signal* signal);
+ void execDROP_TAB_REF(Signal* signal);
+ void execDROP_TAB_CONF(Signal* signal);
+
+ void execCREATE_TABLE_REQ(Signal* signal);
+ void execALTER_TABLE_REQ(Signal* signal);
+ void execCREATE_FRAGMENTATION_REF(Signal*);
+ void execCREATE_FRAGMENTATION_CONF(Signal*);
+ void execCREATE_TAB_REQ(Signal* signal);
+ void execADD_FRAGREQ(Signal* signal);
+ void execLQHFRAGREF(Signal* signal);
+ void execLQHFRAGCONF(Signal* signal);
+ void execLQHADDATTREF(Signal* signal);
+ void execLQHADDATTCONF(Signal* signal);
+ void execCREATE_TAB_REF(Signal* signal);
+ void execCREATE_TAB_CONF(Signal* signal);
+ void execALTER_TAB_REQ(Signal* signal);
+ void execALTER_TAB_REF(Signal* signal);
+ void execALTER_TAB_CONF(Signal* signal);
+
+ /*
+ * 2.4 COMMON STORED VARIABLES
+ */
+
+ /**
+ * This record stores all the state needed
+ * when the schema page is being sent to other nodes
+ ***************************************************************************/
+ struct SendSchemaRecord {
+ /** Number of words of schema data */
+ Uint32 noOfWords;
+ /** Page Id of schema data */
+ Uint32 pageId;
+
+ Uint32 nodeId;
+ SignalCounter m_SCHEMAINFO_Counter;
+
+ Uint32 noOfWordsCurrentlySent;
+ Uint32 noOfSignalsSentSinceDelay;
+
+ bool inUse;
+ };
+ SendSchemaRecord c_sendSchemaRecord;
+
+ /**
+ * This record stores all the state needed
+ * when a table file is being read from disk
+ ****************************************************************************/
+ struct ReadTableRecord {
+ /** Number of Pages */
+ Uint32 noOfPages;
+ /** Page Id*/
+ Uint32 pageId;
+ /** Table Id of read table */
+ Uint32 tableId;
+
+ bool inUse;
+ Callback m_callback;
+ };
+ ReadTableRecord c_readTableRecord;
+
+ /**
+ * This record stores all the state needed
+ * when a table file is being written to disk
+ ****************************************************************************/
+ struct WriteTableRecord {
+ /** Number of Pages */
+ Uint32 noOfPages;
+ /** Page Id*/
+ Uint32 pageId;
+ /** Table Files Handled, local state variable */
+ Uint32 noOfTableFilesHandled;
+ /** Table Id of written table */
+ Uint32 tableId;
+ /** State, indicates from where it was called */
+ enum TableWriteState {
+ IDLE = 0,
+ WRITE_ADD_TABLE_MASTER = 1,
+ WRITE_ADD_TABLE_SLAVE = 2,
+ WRITE_RESTART_FROM_MASTER = 3,
+ WRITE_RESTART_FROM_OWN = 4,
+ CALLBACK = 5
+ };
+ TableWriteState tableWriteState;
+ Callback m_callback;
+ };
+ WriteTableRecord c_writeTableRecord;
+
+ /**
+ * This record stores all the state needed
+ * when a schema file is being read from disk
+ ****************************************************************************/
+ struct ReadSchemaRecord {
+ /** Page Id of schema page */
+ Uint32 pageId;
+ /** State, indicates from where it was called */
+ enum SchemaReadState {
+ IDLE = 0,
+ INITIAL_READ = 1
+ };
+ SchemaReadState schemaReadState;
+ };
+ ReadSchemaRecord c_readSchemaRecord;
+
+private:
+ /**
+ * This record stores all the state needed
+ * when a schema file is being written to disk
+ ****************************************************************************/
+ struct WriteSchemaRecord {
+ /** Page Id of schema page */
+ Uint32 pageId;
+ /** Schema Files Handled, local state variable */
+ Uint32 noOfSchemaFilesHandled;
+
+ bool inUse;
+ Callback m_callback;
+ };
+ WriteSchemaRecord c_writeSchemaRecord;
+
+ /**
+ * This record stores all the information needed
+ * when a file is being read from disk
+ ****************************************************************************/
+ struct RestartRecord {
+ /** Global check point identity */
+ Uint32 gciToRestart;
+
+ /** The active table at restart process */
+ Uint32 activeTable;
+
+ /** The active table at restart process */
+ BlockReference returnBlockRef;
+ };
+ RestartRecord c_restartRecord;
+
+ /**
+ * This record stores all the information needed
+ * when a file is being read from disk
+ ****************************************************************************/
+ struct RetrieveRecord {
+ RetrieveRecord(){ noOfWaiters = 0;}
+
+ /** Only one retrieve table definition at a time */
+ bool busyState;
+
+ /**
+ * No of waiting in time queue
+ */
+ Uint32 noOfWaiters;
+
+ /** Block Reference of retriever */
+ BlockReference blockRef;
+
+ /** Id of retriever */
+ Uint32 m_senderData;
+
+ /** Table id of retrieved table */
+ Uint32 tableId;
+
+ /** Starting page to retrieve data from */
+ Uint32 retrievePage;
+
+ /** Number of pages retrieved */
+ Uint32 retrievedNoOfPages;
+
+ /** Number of words retrieved */
+ Uint32 retrievedNoOfWords;
+
+ /** Number of words sent currently */
+ Uint32 currentSent;
+
+ /**
+ * Long signal stuff
+ */
+ bool m_useLongSig;
+ };
+ RetrieveRecord c_retrieveRecord;
+
+ /**
+ * This record stores all the information needed
+ * when a file is being read from disk
+ *
+ * This is the info stored in one entry of the schema
+ * page. Each table has 4 words of info.
+ * Word 1: Schema version (upper 16 bits)
+ * Table State (lower 16 bits)
+ * Word 2: Number of pages of table description
+ * Word 3: Global checkpoint id table was created
+ * Word 4: Currently zero
+ ****************************************************************************/
+ struct SchemaRecord {
+ /** Schema page */
+ Uint32 schemaPage;
+
+ /** Old Schema page (used at node restart) */
+ Uint32 oldSchemaPage;
+
+ Callback m_callback;
+ };
+ SchemaRecord c_schemaRecord;
+
+ void initSchemaFile(SchemaFile *, Uint32 sz);
+ void computeChecksum(SchemaFile *);
+ bool validateChecksum(const SchemaFile *);
+ SchemaFile::TableEntry * getTableEntry(void * buf, Uint32 tableId,
+ bool allowTooBig = false);
+
+ Uint32 computeChecksum(const Uint32 * src, Uint32 len);
+
+
+ /* ----------------------------------------------------------------------- */
+ // Node References
+ /* ----------------------------------------------------------------------- */
+ Uint16 c_masterNodeId;
+
+ /* ----------------------------------------------------------------------- */
+ // Various current system properties
+ /* ----------------------------------------------------------------------- */
+ Uint16 c_numberNode;
+ Uint16 c_noHotSpareNodes;
+ Uint16 c_noNodesFailed;
+ Uint32 c_failureNr;
+
+ /* ----------------------------------------------------------------------- */
+ // State variables
+ /* ----------------------------------------------------------------------- */
+
+ enum BlockState {
+ BS_IDLE = 0,
+ BS_CREATE_TAB = 1,
+ BS_BUSY = 2,
+ BS_NODE_FAILURE = 3
+ };
+ BlockState c_blockState;
+
+ struct PackTable {
+
+ enum PackTableState {
+ PTS_IDLE = 0,
+ PTS_ADD_TABLE_MASTER = 1,
+ PTS_ADD_TABLE_SLAVE = 2,
+ PTS_GET_TAB = 3,
+ PTS_RESTART = 4
+ } m_state;
+
+ } c_packTable;
+
+ Uint32 c_startPhase;
+ Uint32 c_restartType;
+ bool c_initialStart;
+ bool c_systemRestart;
+ bool c_nodeRestart;
+ bool c_initialNodeRestart;
+ Uint32 c_tabinfoReceived;
+
+ /**
+ * Temporary structure used when parsing table info
+ */
+ struct ParseDictTabInfoRecord {
+ DictTabInfo::RequestType requestType;
+ Uint32 errorCode;
+ Uint32 errorLine;
+
+ SimpleProperties::UnpackStatus status;
+ Uint32 errorKey;
+ TableRecordPtr tablePtr;
+ };
+
+ // Operation records
+
+ /**
+ * Common part of operation records. Uses KeyTable2. Note that each
+ * seize/release invokes ctor/dtor automatically.
+ */
+ struct OpRecordCommon {
+ Uint32 key; // key shared between master and slaves
+ Uint32 nextHash;
+ Uint32 prevHash;
+ Uint32 hashValue() const {
+ return key;
+ }
+ bool equal(const OpRecordCommon& rec) const {
+ return key == rec.key;
+ }
+ };
+
+ /**
+ * Create table record
+ */
+ struct CreateTableRecord : OpRecordCommon {
+ Uint32 m_senderRef;
+ Uint32 m_senderData;
+ Uint32 m_coordinatorRef;
+
+ Uint32 m_errorCode;
+ void setErrorCode(Uint32 c){ if(m_errorCode == 0) m_errorCode = c;}
+
+ // For alter table
+ Uint32 m_changeMask;
+ bool m_alterTableFailed;
+ AlterTableRef m_alterTableRef;
+ Uint32 m_alterTableId;
+
+ /* Previous table name (used for reverting failed table rename) */
+ char previousTableName[MAX_TAB_NAME_SIZE];
+
+ Uint32 m_tablePtrI;
+ Uint32 m_tabInfoPtrI;
+ Uint32 m_fragmentsPtrI;
+
+ Uint32 m_dihAddFragPtr; // Connect ptr towards DIH
+ Uint32 m_lqhFragPtr; // Connect ptr towards LQH
+
+ Callback m_callback; // Who's using local create tab
+ MutexHandle2<DIH_START_LCP_MUTEX> m_startLcpMutex;
+
+ struct CoordinatorData {
+ Uint32 m_gsn;
+ SafeCounterHandle m_counter;
+ CreateTabReq::RequestType m_requestType;
+ } m_coordinatorData;
+ };
+ typedef Ptr<CreateTableRecord> CreateTableRecordPtr;
+
+ /**
+ * Drop table record
+ */
+ struct DropTableRecord : OpRecordCommon {
+ DropTableReq m_request;
+
+ Uint32 m_requestType;
+ Uint32 m_coordinatorRef;
+
+ Uint32 m_errorCode;
+ void setErrorCode(Uint32 c){ if(m_errorCode == 0) m_errorCode = c;}
+
+ /**
+ * When sending stuff around
+ */
+ struct CoordinatorData {
+ Uint32 m_gsn;
+ Uint32 m_block;
+ SignalCounter m_signalCounter;
+ } m_coordinatorData;
+
+ struct ParticipantData {
+ Uint32 m_gsn;
+ Uint32 m_block;
+ SignalCounter m_signalCounter;
+
+ Callback m_callback;
+ } m_participantData;
+ };
+ typedef Ptr<DropTableRecord> DropTableRecordPtr;
+
+ /**
+ * Request flags passed in signals along with request type and
+ * propagated across operations.
+ */
+ struct RequestFlag {
+ enum {
+ RF_LOCAL = 1 << 0, // create on local node only
+ RF_NOBUILD = 1 << 1, // no need to build index
+ RF_NOTCTRIGGER = 1 << 2 // alter trigger: no trigger in TC
+ };
+ };
+
+ /**
+ * Operation record for create index.
+ */
+ struct OpCreateIndex : OpRecordCommon {
+ // original request (index id will be added)
+ CreateIndxReq m_request;
+ AttributeList m_attrList;
+ char m_indexName[MAX_TAB_NAME_SIZE];
+ bool m_storedIndex;
+ // coordinator DICT
+ Uint32 m_coordinatorRef;
+ bool m_isMaster;
+ // state info
+ CreateIndxReq::RequestType m_requestType;
+ Uint32 m_requestFlag;
+ // error info
+ CreateIndxRef::ErrorCode m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+ // counters
+ SignalCounter m_signalCounter;
+ // ctor
+ OpCreateIndex() {
+ memset(&m_request, 0, sizeof(m_request));
+ m_coordinatorRef = 0;
+ m_requestType = CreateIndxReq::RT_UNDEFINED;
+ m_requestFlag = 0;
+ m_errorCode = CreateIndxRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ }
+ void save(const CreateIndxReq* req) {
+ m_request = *req;
+ m_requestType = req->getRequestType();
+ m_requestFlag = req->getRequestFlag();
+ }
+ bool hasError() {
+ return m_errorCode != CreateIndxRef::NoError;
+ }
+ void setError(const CreateIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const CreateTableRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ switch (ref->getErrorCode()) {
+ case CreateTableRef::TableAlreadyExist:
+ m_errorCode = CreateIndxRef::IndexExists;
+ break;
+ default:
+ m_errorCode = (CreateIndxRef::ErrorCode)ref->getErrorCode();
+ break;
+ }
+ m_errorLine = ref->getErrorLine();
+ }
+ }
+ void setError(const AlterIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (CreateIndxRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ };
+ typedef Ptr<OpCreateIndex> OpCreateIndexPtr;
+
+ /**
+ * Operation record for drop index.
+ */
+ struct OpDropIndex : OpRecordCommon {
+ // original request
+ DropIndxReq m_request;
+ // coordinator DICT
+ Uint32 m_coordinatorRef;
+ bool m_isMaster;
+ // state info
+ DropIndxReq::RequestType m_requestType;
+ Uint32 m_requestFlag;
+ // error info
+ DropIndxRef::ErrorCode m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+ // counters
+ SignalCounter m_signalCounter;
+ // ctor
+ OpDropIndex() {
+ memset(&m_request, 0, sizeof(m_request));
+ m_coordinatorRef = 0;
+ m_requestType = DropIndxReq::RT_UNDEFINED;
+ m_requestFlag = 0;
+ m_errorCode = DropIndxRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ }
+ void save(const DropIndxReq* req) {
+ m_request = *req;
+ m_requestType = req->getRequestType();
+ m_requestFlag = req->getRequestFlag();
+ }
+ bool hasError() {
+ return m_errorCode != DropIndxRef::NoError;
+ }
+ void setError(const DropIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const AlterIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (DropIndxRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const DropTableRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ switch(ref->errorCode) {
+ case(DropTableRef::Busy):
+ m_errorCode = DropIndxRef::Busy;
+ break;
+ case(DropTableRef::NoSuchTable):
+ m_errorCode = DropIndxRef::IndexNotFound;
+ break;
+ case(DropTableRef::DropInProgress):
+ m_errorCode = DropIndxRef::Busy;
+ break;
+ case(DropTableRef::NoDropTableRecordAvailable):
+ m_errorCode = DropIndxRef::Busy;
+ break;
+ default:
+ m_errorCode = (DropIndxRef::ErrorCode)ref->errorCode;
+ break;
+ }
+ //m_errorLine = ref->getErrorLine();
+ //m_errorNode = ref->getErrorNode();
+ }
+ }
+ };
+ typedef Ptr<OpDropIndex> OpDropIndexPtr;
+
+ /**
+ * Operation record for alter index.
+ */
+ struct OpAlterIndex : OpRecordCommon {
+ // original request plus buffer for attribute lists
+ AlterIndxReq m_request;
+ AttributeList m_attrList;
+ AttributeList m_tableKeyList;
+ // coordinator DICT
+ Uint32 m_coordinatorRef;
+ bool m_isMaster;
+ // state info
+ AlterIndxReq::RequestType m_requestType;
+ Uint32 m_requestFlag;
+ // error info
+ AlterIndxRef::ErrorCode m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+ // counters
+ SignalCounter m_signalCounter;
+ Uint32 m_triggerCounter;
+ // ctor
+ OpAlterIndex() {
+ memset(&m_request, 0, sizeof(m_request));
+ m_coordinatorRef = 0;
+ m_requestType = AlterIndxReq::RT_UNDEFINED;
+ m_requestFlag = 0;
+ m_errorCode = AlterIndxRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ m_triggerCounter = 0;
+ }
+ void save(const AlterIndxReq* req) {
+ m_request = *req;
+ m_requestType = req->getRequestType();
+ m_requestFlag = req->getRequestFlag();
+ }
+ bool hasError() {
+ return m_errorCode != AlterIndxRef::NoError;
+ }
+ void setError(const AlterIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const CreateIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (AlterIndxRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const DropIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (AlterIndxRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const BuildIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (AlterIndxRef::ErrorCode)ref->getErrorCode();
+ }
+ }
+ void setError(const CreateTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (AlterIndxRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const DropTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (AlterIndxRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ };
+ typedef Ptr<OpAlterIndex> OpAlterIndexPtr;
+
+ /**
+ * Operation record for build index.
+ */
+ struct OpBuildIndex : OpRecordCommon {
+ // original request plus buffer for attribute lists
+ BuildIndxReq m_request;
+ AttributeList m_attrList;
+ AttributeList m_tableKeyList;
+ // coordinator DICT
+ Uint32 m_coordinatorRef;
+ bool m_isMaster;
+ // state info
+ BuildIndxReq::RequestType m_requestType;
+ Uint32 m_requestFlag;
+ Uint32 m_constrTriggerId;
+ // error info
+ BuildIndxRef::ErrorCode m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+ // counters
+ SignalCounter m_signalCounter;
+ // ctor
+ OpBuildIndex() {
+ memset(&m_request, 0, sizeof(m_request));
+ m_coordinatorRef = 0;
+ m_requestType = BuildIndxReq::RT_UNDEFINED;
+ m_requestFlag = 0;
+// Uint32 m_constrTriggerId = RNIL;
+ m_errorCode = BuildIndxRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ }
+ void save(const BuildIndxReq* req) {
+ m_request = *req;
+ m_requestType = req->getRequestType();
+ m_requestFlag = req->getRequestFlag();
+ }
+ bool hasError() {
+ return m_errorCode != BuildIndxRef::NoError;
+ }
+ void setError(const BuildIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = ref->getErrorCode();
+ }
+ }
+ void setError(const AlterIndxRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (BuildIndxRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const CreateTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (BuildIndxRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const DropTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (BuildIndxRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ };
+ typedef Ptr<OpBuildIndex> OpBuildIndexPtr;
+
+ /**
+ * Operation record for Util Signals.
+ */
+ struct OpSignalUtil : OpRecordCommon{
+ Callback m_callback;
+ Uint32 m_userData;
+ };
+ typedef Ptr<OpSignalUtil> OpSignalUtilPtr;
+
+ /**
+ * Operation record for subscribe-start-stop
+ */
+ struct OpSubEvent : OpRecordCommon {
+ Uint32 m_senderRef;
+ Uint32 m_senderData;
+ Uint32 m_errorCode;
+ RequestTracker m_reqTracker;
+ };
+ typedef Ptr<OpSubEvent> OpSubEventPtr;
+
+ /**
+ * Systable NDB$EVENTS_0
+ */
+
+#define EVENT_SYSTEM_TABLE_NAME "sys/def/NDB$EVENTS_0"
+#define EVENT_SYSTEM_TABLE_LENGTH 6
+
+ struct sysTab_NDBEVENTS_0 {
+ char NAME[MAX_TAB_NAME_SIZE];
+ Uint32 EVENT_TYPE;
+ char TABLE_NAME[MAX_TAB_NAME_SIZE];
+ Uint32 ATTRIBUTE_MASK[MAXNROFATTRIBUTESINWORDS];
+ Uint32 SUBID;
+ Uint32 SUBKEY;
+ };
+
+ static const Uint32 sysTab_NDBEVENTS_0_szs[];
+
+ /**
+ * Operation record for create event.
+ */
+ struct OpCreateEvent : OpRecordCommon {
+ // original request (event id will be added)
+ CreateEvntReq m_request;
+ //AttributeMask m_attrListBitmask;
+ // AttributeList m_attrList;
+ sysTab_NDBEVENTS_0 m_eventRec;
+ // char m_eventName[MAX_TAB_NAME_SIZE];
+ // char m_tableName[MAX_TAB_NAME_SIZE];
+
+ // coordinator DICT
+ RequestTracker m_reqTracker;
+ // state info
+ CreateEvntReq::RequestType m_requestType;
+ Uint32 m_requestFlag;
+ // error info
+ CreateEvntRef::ErrorCode m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+ // ctor
+ OpCreateEvent() {
+ memset(&m_request, 0, sizeof(m_request));
+ m_requestType = CreateEvntReq::RT_UNDEFINED;
+ m_requestFlag = 0;
+ m_errorCode = CreateEvntRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ }
+ void init(const CreateEvntReq* req, Dbdict* dp) {
+ m_request = *req;
+ m_errorCode = CreateEvntRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ m_requestType = req->getRequestType();
+ m_requestFlag = req->getRequestFlag();
+ }
+ bool hasError() {
+ return m_errorCode != CreateEvntRef::NoError;
+ }
+ void setError(const CreateEvntRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+
+ };
+ typedef Ptr<OpCreateEvent> OpCreateEventPtr;
+
+ /**
+ * Operation record for drop event.
+ */
+ struct OpDropEvent : OpRecordCommon {
+ // original request
+ DropEvntReq m_request;
+ // char m_eventName[MAX_TAB_NAME_SIZE];
+ sysTab_NDBEVENTS_0 m_eventRec;
+ RequestTracker m_reqTracker;
+ // error info
+ DropEvntRef::ErrorCode m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+ // ctor
+ OpDropEvent() {
+ memset(&m_request, 0, sizeof(m_request));
+ m_errorCode = DropEvntRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ }
+ void init(const DropEvntReq* req) {
+ m_request = *req;
+ m_errorCode = DropEvntRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ }
+ bool hasError() {
+ return m_errorCode != DropEvntRef::NoError;
+ }
+ void setError(const DropEvntRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ };
+ typedef Ptr<OpDropEvent> OpDropEventPtr;
+
+ /**
+ * Operation record for create trigger.
+ */
+ struct OpCreateTrigger : OpRecordCommon {
+ // original request (trigger id will be added)
+ CreateTrigReq m_request;
+ char m_triggerName[MAX_TAB_NAME_SIZE];
+ // coordinator DICT
+ Uint32 m_coordinatorRef;
+ bool m_isMaster;
+ // state info
+ CreateTrigReq::RequestType m_requestType;
+ Uint32 m_requestFlag;
+ // error info
+ CreateTrigRef::ErrorCode m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+ // counters
+ SignalCounter m_signalCounter;
+ // ctor
+ OpCreateTrigger() {
+ memset(&m_request, 0, sizeof(m_request));
+ m_coordinatorRef = 0;
+ m_requestType = CreateTrigReq::RT_UNDEFINED;
+ m_requestFlag = 0;
+ m_errorCode = CreateTrigRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ }
+ void save(const CreateTrigReq* req) {
+ m_request = *req;
+ m_requestType = req->getRequestType();
+ m_requestFlag = req->getRequestFlag();
+ }
+ bool hasError() {
+ return m_errorCode != CreateTrigRef::NoError;
+ }
+ void setError(const CreateTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const AlterTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (CreateTrigRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ };
+ typedef Ptr<OpCreateTrigger> OpCreateTriggerPtr;
+
+ /**
+ * Operation record for drop trigger.
+ */
+ struct OpDropTrigger : OpRecordCommon {
+ // original request
+ DropTrigReq m_request;
+ // coordinator DICT
+ Uint32 m_coordinatorRef;
+ bool m_isMaster;
+ // state info
+ DropTrigReq::RequestType m_requestType;
+ Uint32 m_requestFlag;
+ // error info
+ DropTrigRef::ErrorCode m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+ // counters
+ SignalCounter m_signalCounter;
+ // ctor
+ OpDropTrigger() {
+ memset(&m_request, 0, sizeof(m_request));
+ m_coordinatorRef = 0;
+ m_requestType = DropTrigReq::RT_UNDEFINED;
+ m_requestFlag = 0;
+ m_errorCode = DropTrigRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ }
+ void save(const DropTrigReq* req) {
+ m_request = *req;
+ m_requestType = req->getRequestType();
+ m_requestFlag = req->getRequestFlag();
+ }
+ bool hasError() {
+ return m_errorCode != DropTrigRef::NoError;
+ }
+ void setError(const DropTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const AlterTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (DropTrigRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ };
+ typedef Ptr<OpDropTrigger> OpDropTriggerPtr;
+
+ /**
+ * Operation record for alter trigger.
+ */
+ struct OpAlterTrigger : OpRecordCommon {
+ // original request
+ AlterTrigReq m_request;
+ // nodes participating in operation
+ NdbNodeBitmask m_nodes;
+ // coordinator DICT
+ Uint32 m_coordinatorRef;
+ bool m_isMaster;
+ // state info
+ AlterTrigReq::RequestType m_requestType;
+ Uint32 m_requestFlag;
+ // error info
+ AlterTrigRef::ErrorCode m_errorCode;
+ Uint32 m_errorLine;
+ Uint32 m_errorNode;
+ // counters
+ SignalCounter m_signalCounter;
+ // ctor
+ OpAlterTrigger() {
+ memset(&m_request, 0, sizeof(m_request));
+ m_coordinatorRef = 0;
+ m_requestType = AlterTrigReq::RT_UNDEFINED;
+ m_requestFlag = 0;
+ m_errorCode = AlterTrigRef::NoError;
+ m_errorLine = 0;
+ m_errorNode = 0;
+ }
+ void save(const AlterTrigReq* req) {
+ m_request = *req;
+ m_requestType = req->getRequestType();
+ m_requestFlag = req->getRequestFlag();
+ }
+ bool hasError() {
+ return m_errorCode != AlterTrigRef::NoError;
+ }
+ void setError(const AlterTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (AlterTrigRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const CreateTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (AlterTrigRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ void setError(const DropTrigRef* ref) {
+ if (ref != 0 && ! hasError()) {
+ m_errorCode = (AlterTrigRef::ErrorCode)ref->getErrorCode();
+ m_errorLine = ref->getErrorLine();
+ m_errorNode = ref->getErrorNode();
+ }
+ }
+ };
+ typedef Ptr<OpAlterTrigger> OpAlterTriggerPtr;
+
+ // Common operation record pool
+public:
+ static const size_t opCreateTableSize = sizeof(CreateTableRecord);
+ static const size_t opDropTableSize = sizeof(DropTableRecord);
+ static const size_t opCreateIndexSize = sizeof(OpCreateIndex);
+ static const size_t opDropIndexSize = sizeof(OpDropIndex);
+ static const size_t opAlterIndexSize = sizeof(OpAlterIndex);
+ static const size_t opBuildIndexSize = sizeof(OpBuildIndex);
+ static const size_t opCreateEventSize = sizeof(OpCreateEvent);
+ static const size_t opSubEventSize = sizeof(OpSubEvent);
+ static const size_t opDropEventSize = sizeof(OpDropEvent);
+ static const size_t opSignalUtilSize = sizeof(OpSignalUtil);
+ static const size_t opCreateTriggerSize = sizeof(OpCreateTrigger);
+ static const size_t opDropTriggerSize = sizeof(OpDropTrigger);
+ static const size_t opAlterTriggerSize = sizeof(OpAlterTrigger);
+private:
+#define PTR_ALIGN(n) ((((n)+sizeof(void*)-1)>>2)&~((sizeof(void*)-1)>>2))
+ union OpRecordUnion {
+ Uint32 u_opCreateTable [PTR_ALIGN(opCreateTableSize)];
+ Uint32 u_opDropTable [PTR_ALIGN(opDropTableSize)];
+ Uint32 u_opCreateIndex [PTR_ALIGN(opCreateIndexSize)];
+ Uint32 u_opDropIndex [PTR_ALIGN(opDropIndexSize)];
+ Uint32 u_opCreateEvent [PTR_ALIGN(opCreateEventSize)];
+ Uint32 u_opSubEvent [PTR_ALIGN(opSubEventSize)];
+ Uint32 u_opDropEvent [PTR_ALIGN(opDropEventSize)];
+ Uint32 u_opSignalUtil [PTR_ALIGN(opSignalUtilSize)];
+ Uint32 u_opAlterIndex [PTR_ALIGN(opAlterIndexSize)];
+ Uint32 u_opBuildIndex [PTR_ALIGN(opBuildIndexSize)];
+ Uint32 u_opCreateTrigger[PTR_ALIGN(opCreateTriggerSize)];
+ Uint32 u_opDropTrigger [PTR_ALIGN(opDropTriggerSize)];
+ Uint32 u_opAlterTrigger [PTR_ALIGN(opAlterTriggerSize)];
+ Uint32 nextPool;
+ };
+ ArrayPool<OpRecordUnion> c_opRecordPool;
+
+ // Operation records
+ KeyTable2<CreateTableRecord, OpRecordUnion> c_opCreateTable;
+ KeyTable2<DropTableRecord, OpRecordUnion> c_opDropTable;
+ KeyTable2<OpCreateIndex, OpRecordUnion> c_opCreateIndex;
+ KeyTable2<OpDropIndex, OpRecordUnion> c_opDropIndex;
+ KeyTable2<OpAlterIndex, OpRecordUnion> c_opAlterIndex;
+ KeyTable2<OpBuildIndex, OpRecordUnion> c_opBuildIndex;
+ KeyTable2<OpCreateEvent, OpRecordUnion> c_opCreateEvent;
+ KeyTable2<OpSubEvent, OpRecordUnion> c_opSubEvent;
+ KeyTable2<OpDropEvent, OpRecordUnion> c_opDropEvent;
+ KeyTable2<OpSignalUtil, OpRecordUnion> c_opSignalUtil;
+ KeyTable2<OpCreateTrigger, OpRecordUnion> c_opCreateTrigger;
+ KeyTable2<OpDropTrigger, OpRecordUnion> c_opDropTrigger;
+ KeyTable2<OpAlterTrigger, OpRecordUnion> c_opAlterTrigger;
+
+ // Unique key for operation XXX move to some system table
+ Uint32 c_opRecordSequence;
+
+ // Statement blocks
+
+ /* ------------------------------------------------------------ */
+ // Start/Restart Handling
+ /* ------------------------------------------------------------ */
+ void sendSTTORRY(Signal* signal);
+ void sendNDB_STTORRY(Signal* signal);
+ void initSchemaFile(Signal* signal);
+
+ /* ------------------------------------------------------------ */
+ // Drop Table Handling
+ /* ------------------------------------------------------------ */
+ void releaseTableObject(Uint32 tableId, bool removeFromHash = true);
+
+ /* ------------------------------------------------------------ */
+ // General Stuff
+ /* ------------------------------------------------------------ */
+ Uint32 getFreeTableRecord(Uint32 primaryTableId);
+ Uint32 getFreeTriggerRecord();
+ bool getNewAttributeRecord(TableRecordPtr tablePtr,
+ AttributeRecordPtr & attrPtr);
+ void packTableIntoPages(Signal* signal, Uint32 tableId, Uint32 pageId);
+ void packTableIntoPagesImpl(SimpleProperties::Writer &, TableRecordPtr);
+
+ void sendGET_TABINFOREQ(Signal* signal,
+ Uint32 tableId);
+ void sendTC_SCHVERREQ(Signal* signal,
+ Uint32 tableId,
+ BlockReference tcRef);
+
+ /* ------------------------------------------------------------ */
+ // System Restart Handling
+ /* ------------------------------------------------------------ */
+ void initSendSchemaData(Signal* signal);
+ void sendSchemaData(Signal* signal);
+ Uint32 sendSCHEMA_INFO(Signal* signal, Uint32 nodeId, Uint32* pagePointer);
+ void checkSchemaStatus(Signal* signal);
+ void sendDIHSTARTTAB_REQ(Signal* signal);
+
+ /* ------------------------------------------------------------ */
+ // Receive Table Handling
+ /* ------------------------------------------------------------ */
+ void handleTabInfoInit(SimpleProperties::Reader &,
+ ParseDictTabInfoRecord *,
+ bool checkExist = true);
+ void handleTabInfo(SimpleProperties::Reader & it, ParseDictTabInfoRecord *);
+
+ void handleAddTableFailure(Signal* signal,
+ Uint32 failureLine,
+ Uint32 tableId);
+ bool verifyTableCorrect(Signal* signal, Uint32 tableId);
+
+ /* ------------------------------------------------------------ */
+ // Add Table Handling
+ /* ------------------------------------------------------------ */
+
+ /* ------------------------------------------------------------ */
+ // Add Fragment Handling
+ /* ------------------------------------------------------------ */
+ void sendLQHADDATTRREQ(Signal*, CreateTableRecordPtr, Uint32 attributePtrI);
+
+ /* ------------------------------------------------------------ */
+ // Read/Write Schema and Table files
+ /* ------------------------------------------------------------ */
+ void updateSchemaState(Signal* signal, Uint32 tableId,
+ SchemaFile::TableEntry*, Callback*);
+ void startWriteSchemaFile(Signal* signal);
+ void openSchemaFile(Signal* signal,
+ Uint32 fileNo,
+ Uint32 fsPtr,
+ bool writeFlag);
+ void writeSchemaFile(Signal* signal, Uint32 filePtr, Uint32 fsPtr);
+ void writeSchemaConf(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+ void closeFile(Signal* signal, Uint32 filePtr, Uint32 fsPtr);
+ void closeWriteSchemaConf(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+ void initSchemaFile_conf(Signal* signal, Uint32 i, Uint32 returnCode);
+
+ void writeTableFile(Signal* signal, Uint32 tableId,
+ SegmentedSectionPtr tabInfo, Callback*);
+ void startWriteTableFile(Signal* signal, Uint32 tableId);
+ void openTableFile(Signal* signal,
+ Uint32 fileNo,
+ Uint32 fsPtr,
+ Uint32 tableId,
+ bool writeFlag);
+ void writeTableFile(Signal* signal, Uint32 filePtr, Uint32 fsPtr);
+ void writeTableConf(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+ void closeWriteTableConf(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+
+ void startReadTableFile(Signal* signal, Uint32 tableId);
+ void openReadTableRef(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+ void readTableFile(Signal* signal, Uint32 filePtr, Uint32 fsPtr);
+ void readTableConf(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+ void readTableRef(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+ void closeReadTableConf(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+
+ void startReadSchemaFile(Signal* signal);
+ void openReadSchemaRef(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+ void readSchemaFile(Signal* signal, Uint32 filePtr, Uint32 fsPtr);
+ void readSchemaConf(Signal* signal, FsConnectRecordPtr fsPtr);
+ void readSchemaRef(Signal* signal, FsConnectRecordPtr fsPtr);
+ void closeReadSchemaConf(Signal* signal,
+ FsConnectRecordPtr fsPtr);
+
+ /* ------------------------------------------------------------ */
+ // Get table definitions
+ /* ------------------------------------------------------------ */
+ void sendGET_TABINFOREF(Signal* signal,
+ GetTabInfoReq*,
+ GetTabInfoRef::ErrorCode errorCode);
+
+ void sendGET_TABLEID_REF(Signal* signal,
+ GetTableIdReq * req,
+ GetTableIdRef::ErrorCode errorCode);
+
+ void sendGetTabResponse(Signal* signal);
+
+ /* ------------------------------------------------------------ */
+ // Indexes and triggers
+ /* ------------------------------------------------------------ */
+
+ // reactivate and rebuild indexes on start up
+ void activateIndexes(Signal* signal, Uint32 i);
+ void rebuildIndexes(Signal* signal, Uint32 i);
+
+ // create index
+ void createIndex_recvReply(Signal* signal, const CreateIndxConf* conf,
+ const CreateIndxRef* ref);
+ void createIndex_slavePrepare(Signal* signal, OpCreateIndexPtr opPtr);
+ void createIndex_toCreateTable(Signal* signal, OpCreateIndexPtr opPtr);
+ void createIndex_fromCreateTable(Signal* signal, OpCreateIndexPtr opPtr);
+ void createIndex_toAlterIndex(Signal* signal, OpCreateIndexPtr opPtr);
+ void createIndex_fromAlterIndex(Signal* signal, OpCreateIndexPtr opPtr);
+ void createIndex_slaveCommit(Signal* signal, OpCreateIndexPtr opPtr);
+ void createIndex_slaveAbort(Signal* signal, OpCreateIndexPtr opPtr);
+ void createIndex_sendSlaveReq(Signal* signal, OpCreateIndexPtr opPtr);
+ void createIndex_sendReply(Signal* signal, OpCreateIndexPtr opPtr, bool);
+ // drop index
+ void dropIndex_recvReply(Signal* signal, const DropIndxConf* conf,
+ const DropIndxRef* ref);
+ void dropIndex_slavePrepare(Signal* signal, OpDropIndexPtr opPtr);
+ void dropIndex_toAlterIndex(Signal* signal, OpDropIndexPtr opPtr);
+ void dropIndex_fromAlterIndex(Signal* signal, OpDropIndexPtr opPtr);
+ void dropIndex_toDropTable(Signal* signal, OpDropIndexPtr opPtr);
+ void dropIndex_fromDropTable(Signal* signal, OpDropIndexPtr opPtr);
+ void dropIndex_slaveCommit(Signal* signal, OpDropIndexPtr opPtr);
+ void dropIndex_slaveAbort(Signal* signal, OpDropIndexPtr opPtr);
+ void dropIndex_sendSlaveReq(Signal* signal, OpDropIndexPtr opPtr);
+ void dropIndex_sendReply(Signal* signal, OpDropIndexPtr opPtr, bool);
+ // alter index
+ void alterIndex_recvReply(Signal* signal, const AlterIndxConf* conf,
+ const AlterIndxRef* ref);
+ void alterIndex_slavePrepare(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_toCreateTc(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_fromCreateTc(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_toDropTc(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_fromDropTc(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_toCreateTrigger(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_fromCreateTrigger(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_toDropTrigger(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_fromDropTrigger(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_toBuildIndex(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_fromBuildIndex(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_slaveCommit(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_slaveAbort(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_sendSlaveReq(Signal* signal, OpAlterIndexPtr opPtr);
+ void alterIndex_sendReply(Signal* signal, OpAlterIndexPtr opPtr, bool);
+ // build index
+ void buildIndex_recvReply(Signal* signal, const BuildIndxConf* conf,
+ const BuildIndxRef* ref);
+ void buildIndex_toCreateConstr(Signal* signal, OpBuildIndexPtr opPtr);
+ void buildIndex_fromCreateConstr(Signal* signal, OpBuildIndexPtr opPtr);
+ void buildIndex_buildTrix(Signal* signal, OpBuildIndexPtr opPtr);
+ void buildIndex_toDropConstr(Signal* signal, OpBuildIndexPtr opPtr);
+ void buildIndex_fromDropConstr(Signal* signal, OpBuildIndexPtr opPtr);
+ void buildIndex_toOnline(Signal* signal, OpBuildIndexPtr opPtr);
+ void buildIndex_fromOnline(Signal* signal, OpBuildIndexPtr opPtr);
+ void buildIndex_sendSlaveReq(Signal* signal, OpBuildIndexPtr opPtr);
+ void buildIndex_sendReply(Signal* signal, OpBuildIndexPtr opPtr, bool);
+
+ // Events
+ void
+ createEventUTIL_PREPARE(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+ void
+ createEventUTIL_EXECUTE(Signal *signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+ void
+ dropEventUTIL_PREPARE_READ(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+ void
+ dropEventUTIL_EXECUTE_READ(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+ void
+ dropEventUTIL_PREPARE_DELETE(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+ void
+ dropEventUTIL_EXECUTE_DELETE(Signal *signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+ void
+ dropEventUtilPrepareRef(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+ void
+ dropEventUtilExecuteRef(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+ int
+ sendSignalUtilReq(Callback *c,
+ BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections);
+ int
+ recvSignalUtilReq(Signal* signal, Uint32 returnCode);
+
+ void completeSubStartReq(Signal* signal, Uint32 ptrI, Uint32 returnCode);
+ void completeSubStopReq(Signal* signal, Uint32 ptrI, Uint32 returnCode);
+ void completeSubRemoveReq(Signal* signal, Uint32 ptrI, Uint32 returnCode);
+
+ void dropEvent_sendReply(Signal* signal,
+ OpDropEventPtr evntRecPtr);
+
+ void createEvent_RT_USER_CREATE(Signal* signal, OpCreateEventPtr evntRecPtr);
+ void createEventComplete_RT_USER_CREATE(Signal* signal,
+ OpCreateEventPtr evntRecPtr);
+ void createEvent_RT_USER_GET(Signal* signal, OpCreateEventPtr evntRecPtr);
+ void createEventComplete_RT_USER_GET(Signal* signal, OpCreateEventPtr evntRecPtr);
+
+ void createEvent_RT_DICT_AFTER_GET(Signal* signal, OpCreateEventPtr evntRecPtr);
+
+ void createEvent_nodeFailCallback(Signal* signal, Uint32 eventRecPtrI,
+ Uint32 returnCode);
+ void createEvent_sendReply(Signal* signal, OpCreateEventPtr evntRecPtr,
+ LinearSectionPtr *ptr = NULL, int noLSP = 0);
+
+ void prepareTransactionEventSysTable (Callback *c,
+ Signal* signal,
+ Uint32 senderData,
+ UtilPrepareReq::OperationTypeValue prepReq);
+ void prepareUtilTransaction(Callback *c,
+ Signal* signal,
+ Uint32 senderData,
+ Uint32 tableId,
+ const char *tableName,
+ UtilPrepareReq::OperationTypeValue prepReq,
+ Uint32 noAttr,
+ Uint32 attrIds[],
+ const char *attrNames[]);
+
+ void executeTransEventSysTable(Callback *c,
+ Signal *signal,
+ const Uint32 ptrI,
+ sysTab_NDBEVENTS_0& m_eventRec,
+ const Uint32 prepareId,
+ UtilPrepareReq::OperationTypeValue prepReq);
+ void executeTransaction(Callback *c,
+ Signal* signal,
+ Uint32 senderData,
+ Uint32 prepareId,
+ Uint32 noAttr,
+ LinearSectionPtr headerPtr,
+ LinearSectionPtr dataPtr);
+
+ void parseReadEventSys(Signal *signal, sysTab_NDBEVENTS_0& m_eventRec);
+
+ // create trigger
+ void createTrigger_recvReply(Signal* signal, const CreateTrigConf* conf,
+ const CreateTrigRef* ref);
+ void createTrigger_slavePrepare(Signal* signal, OpCreateTriggerPtr opPtr);
+ void createTrigger_masterSeize(Signal* signal, OpCreateTriggerPtr opPtr);
+ void createTrigger_slaveCreate(Signal* signal, OpCreateTriggerPtr opPtr);
+ void createTrigger_toAlterTrigger(Signal* signal, OpCreateTriggerPtr opPtr);
+ void createTrigger_fromAlterTrigger(Signal* signal, OpCreateTriggerPtr opPtr);
+ void createTrigger_slaveCommit(Signal* signal, OpCreateTriggerPtr opPtr);
+ void createTrigger_slaveAbort(Signal* signal, OpCreateTriggerPtr opPtr);
+ void createTrigger_sendSlaveReq(Signal* signal, OpCreateTriggerPtr opPtr);
+ void createTrigger_sendReply(Signal* signal, OpCreateTriggerPtr opPtr, bool);
+ // drop trigger
+ void dropTrigger_recvReply(Signal* signal, const DropTrigConf* conf,
+ const DropTrigRef* ref);
+ void dropTrigger_slavePrepare(Signal* signal, OpDropTriggerPtr opPtr);
+ void dropTrigger_toAlterTrigger(Signal* signal, OpDropTriggerPtr opPtr);
+ void dropTrigger_fromAlterTrigger(Signal* signal, OpDropTriggerPtr opPtr);
+ void dropTrigger_slaveCommit(Signal* signal, OpDropTriggerPtr opPtr);
+ void dropTrigger_slaveAbort(Signal* signal, OpDropTriggerPtr opPtr);
+ void dropTrigger_sendSlaveReq(Signal* signal, OpDropTriggerPtr opPtr);
+ void dropTrigger_sendReply(Signal* signal, OpDropTriggerPtr opPtr, bool);
+ // alter trigger
+ void alterTrigger_recvReply(Signal* signal, const AlterTrigConf* conf,
+ const AlterTrigRef* ref);
+ void alterTrigger_slavePrepare(Signal* signal, OpAlterTriggerPtr opPtr);
+ void alterTrigger_toCreateLocal(Signal* signal, OpAlterTriggerPtr opPtr);
+ void alterTrigger_fromCreateLocal(Signal* signal, OpAlterTriggerPtr opPtr);
+ void alterTrigger_toDropLocal(Signal* signal, OpAlterTriggerPtr opPtr);
+ void alterTrigger_fromDropLocal(Signal* signal, OpAlterTriggerPtr opPtr);
+ void alterTrigger_slaveCommit(Signal* signal, OpAlterTriggerPtr opPtr);
+ void alterTrigger_slaveAbort(Signal* signal, OpAlterTriggerPtr opPtr);
+ void alterTrigger_sendSlaveReq(Signal* signal, OpAlterTriggerPtr opPtr);
+ void alterTrigger_sendReply(Signal* signal, OpAlterTriggerPtr opPtr, bool);
+ // support
+ void getTableKeyList(TableRecordPtr tablePtr, AttributeList& list);
+ void getIndexAttr(TableRecordPtr indexPtr, Uint32 itAttr, Uint32* id);
+ void getIndexAttrList(TableRecordPtr indexPtr, AttributeList& list);
+ void getIndexAttrMask(TableRecordPtr indexPtr, AttributeMask& mask);
+
+ /* ------------------------------------------------------------ */
+ // Initialisation
+ /* ------------------------------------------------------------ */
+ void initCommonData();
+ void initRecords();
+ void initConnectRecord();
+ void initRetrieveRecord(Signal*, Uint32, Uint32 returnCode);
+ void initSchemaRecord();
+ void initRestartRecord();
+ void initSendSchemaRecord();
+ void initReadTableRecord();
+ void initWriteTableRecord();
+ void initReadSchemaRecord();
+ void initWriteSchemaRecord();
+
+ void initNodeRecords();
+ void initTableRecords();
+ void initialiseTableRecord(TableRecordPtr tablePtr);
+ void initTriggerRecords();
+ void initialiseTriggerRecord(TriggerRecordPtr triggerPtr);
+ void initPageRecords();
+
+ Uint32 getFsConnRecord();
+
+ bool getIsFailed(Uint32 nodeId) const;
+
+ void dropTableRef(Signal * signal, DropTableReq *, DropTableRef::ErrorCode);
+ void printTables(); // For debugging only
+ int handleAlterTab(AlterTabReq * req,
+ CreateTableRecord * regAlterTabPtr,
+ TableRecordPtr origTablePtr,
+ TableRecordPtr newTablePtr);
+ void revertAlterTable(Signal * signal,
+ Uint32 changeMask,
+ Uint32 tableId,
+ CreateTableRecord * regAlterTabPtr);
+ void alterTableRef(Signal * signal,
+ AlterTableReq *, AlterTableRef::ErrorCode,
+ ParseDictTabInfoRecord* parseRecord = NULL);
+ void alterTabRef(Signal * signal,
+ AlterTabReq *, AlterTableRef::ErrorCode,
+ ParseDictTabInfoRecord* parseRecord = NULL);
+ void alterTab_writeSchemaConf(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+ void alterTab_writeTableConf(Signal* signal,
+ Uint32 callbackData,
+ Uint32 returnCode);
+
+ void prepDropTab_nextStep(Signal* signal, DropTableRecordPtr);
+ void prepDropTab_complete(Signal* signal, DropTableRecordPtr);
+ void prepDropTab_writeSchemaConf(Signal* signal, Uint32 dropTabPtrI, Uint32);
+
+ void dropTab_localDROP_TAB_CONF(Signal* signal);
+ void dropTab_nextStep(Signal* signal, DropTableRecordPtr);
+ void dropTab_complete(Signal* signal, Uint32 dropTabPtrI, Uint32);
+ void dropTab_writeSchemaConf(Signal* signal, Uint32 dropTabPtrI, Uint32);
+
+ void createTab_prepare(Signal* signal, CreateTabReq * req);
+ void createTab_writeSchemaConf1(Signal* signal, Uint32 callback, Uint32);
+ void createTab_writeTableConf(Signal* signal, Uint32 callbackData, Uint32);
+ void createTab_dih(Signal*, CreateTableRecordPtr,
+ SegmentedSectionPtr, Callback*);
+ void createTab_dihComplete(Signal* signal, Uint32 callbackData, Uint32);
+
+ void createTab_startLcpMutex_locked(Signal* signal, Uint32, Uint32);
+ void createTab_startLcpMutex_unlocked(Signal* signal, Uint32, Uint32);
+
+ void createTab_commit(Signal* signal, CreateTabReq * req);
+ void createTab_writeSchemaConf2(Signal* signal, Uint32 callbackData, Uint32);
+ void createTab_alterComplete(Signal*, Uint32 callbackData, Uint32);
+
+ void createTab_drop(Signal* signal, CreateTabReq * req);
+ void createTab_dropComplete(Signal* signal, Uint32 callbackData, Uint32);
+
+ void createTab_reply(Signal* signal, CreateTableRecordPtr, Uint32 nodeId);
+ void alterTab_activate(Signal*, CreateTableRecordPtr, Callback*);
+
+ void restartCreateTab(Signal*, Uint32, const SchemaFile::TableEntry *, bool);
+ void restartCreateTab_readTableConf(Signal* signal, Uint32 callback, Uint32);
+ void restartCreateTab_writeTableConf(Signal* signal, Uint32 callback, Uint32);
+ void restartCreateTab_dihComplete(Signal* signal, Uint32 callback, Uint32);
+ void restartCreateTab_activateComplete(Signal*, Uint32 callback, Uint32);
+
+ void restartDropTab(Signal* signal, Uint32 tableId);
+ void restartDropTab_complete(Signal*, Uint32 callback, Uint32);
+
+ void restart_checkSchemaStatusComplete(Signal*, Uint32 callback, Uint32);
+ void restart_writeSchemaConf(Signal*, Uint32 callbackData, Uint32);
+ void masterRestart_checkSchemaStatusComplete(Signal*, Uint32, Uint32);
+
+ void sendSchemaComplete(Signal*, Uint32 callbackData, Uint32);
+
+ // global metadata support
+ friend class MetaData;
+ int getMetaTablePtr(TableRecordPtr& tablePtr, Uint32 tableId, Uint32 tableVersion);
+ int getMetaTable(MetaData::Table& table, Uint32 tableId, Uint32 tableVersion);
+ int getMetaTable(MetaData::Table& table, const char* tableName);
+ int getMetaAttribute(MetaData::Attribute& attribute, const MetaData::Table& table, Uint32 attributeId);
+ int getMetaAttribute(MetaData::Attribute& attribute, const MetaData::Table& table, const char* attributeName);
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/dbdict/Dbdict.txt b/ndb/src/kernel/blocks/dbdict/Dbdict.txt
new file mode 100644
index 00000000000..8d4267a1c42
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/Dbdict.txt
@@ -0,0 +1,88 @@
+
+Event creation
+
+USER DICT(Master) UTIL SUMA
+================================================================================
+CREATE_EVENT_REQ::create
+-------------------------->
+ - Get ID
+ CREATE_SUBID
+ ----------------------------------------------->
+ <-----------------------------------------------
+ - insert into system table
+ UTIL_PREPARE::insert
+ ------------------------>
+ <------------------------
+ UTIL_EXECUTE
+ ------------------------>
+ <------------------------
+CREATE_EVENT_CONF
+<--------------------------
+
+
+Event dropping
+
+USER DICT(Master) UTIL SUMA
+================================================================================
+DROP_EVENT_REQ
+-------------------------->
+ - remove from system table
+ UTIL_PREPARE::delete
+ ------------------------>
+ <------------------------
+ UTIL_EXECUTE
+ ------------------------>
+ <------------------------
+DROP_EVENT_CONF
+<--------------------------
+
+
+
+create NdbEventOperation
+
+USER DICT(Master) (Slaves) UTIL
+=======================================================================
+CREATE_EVENT_REQ::get
+-------------------------->
+ - read from system table
+ UTIL_PREPARE::read
+ ---------------------------------------->
+ <----------------------------------------
+ UTIL_EXECUTE
+ ---------------------------------------->
+ <----------------------------------------
+ SUMA
+ CREATE_EVENT_REQ::after_get ======
+ ---------------------->
+ SUB_CREATE
+ ------------------>
+ <------------------
+ SUB_SYNC
+ ------------------>
+ <------------------
+ CREATE_EVENT_CONF
+ <----------------------
+CREATE_EVENT_CONF
+<-------------------------
+
+
+
+USER DICT(Master) (Slaves) SUMA
+=======================================================================
+SUB_START_REQ
+-------------------------->
+ SUB_START_REQ
+ ---------------------->
+ SUB_START
+ ------------------>
+ <------------------
+ SUB_START_CONF
+ <----------------------
+SUB_START_CONF
+<-------------------------
+
+
+SUB_STOP analogous to SUB_STOP
+
+
+
diff --git a/ndb/src/kernel/blocks/dbdict/DropTable.txt b/ndb/src/kernel/blocks/dbdict/DropTable.txt
new file mode 100644
index 00000000000..8d364d15c57
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/DropTable.txt
@@ -0,0 +1,140 @@
+DROP TABLE DESCRIPTION
+----------------------
+
+Drop table is controlled by DICT.
+
+Drop table is used in the following cases in some sort.
+ - Drop Table
+ - Abort Add Table
+ - Drop table in node restart
+ - Drop table in system restart
+
+Sequence of Drop Table:
+-----------------------
+
+1) PREP_DROP_TAB_REQ -> all DICT
+ Update schema files on disk
+ Table status = DROPPING
+
+2) Controlling DICT only
+ Report Table Dropped secured but not yet completed.
+
+------ PREP DROP
+
+4) PREP_DROP_TAB_REQ -> all LQHs
+
+5) PREP_DROP_TAB_REQ -> all TCs
+
+6) PREP_DROP_TAB_REQ -> all DIHs
+
+
+--- LQH::PREP_DROP_TAB_REQ
+
+*) Mark the table so that no new operations will start
+*) Mark all fragments so that new LCP_FRAG_ORD gets replied directly
+ w.o actually checkpointing the fragment
+2) Start waiting for completion
+3) Reply PREP_DROP_TAB_CONF
+
+- After this LQH accepts WAIT_DROP_TAB_REQ
+
+--- TC::PREP_DROP_TAB_REQ
+
+1) Mark the table so that no new transactions will start on the table
+2) Send WAIT_DROP_TAB_REQ -> all connected LQH's
+3) Wait for CONF (including NF-handling) from LQH:s
+4) Reply PREP_DROP_TAB_CONF
+
+--- DIH::PREP_DROP_TAB_REQ
+
+1) Mark the table so that no new LCP will start on the table
+2) If master (unlink any queued LCP_FRAG_ORD)
+3) Send WAIT_DROP_TAB_REQ -> all connected LQH's
+4) Wait for CONF (including NF-handling) from LQH:s
+5) Reply PREP_DROP_TAB_CONF
+
+--- LQH::WAIT_DROP_TAB_REQ
+
+1) Wait for running operations
+ Wait for running LCP
+
+2) Reply
+
+------ PREP_DROP
+
+7) DROP_TAB_REQ -> all DICT's
+ *) DROP_TAB_REQ -> TC
+ *) DROP_TAB_REQ -> ACC
+ *) DROP_TAB_REQ -> TUP
+ *) DROP_TAB_REQ -> DIH
+ *) DROP_TAB_REQ -> LQH
+ *) Update schema files on disk DROPPED
+
+8) DICT_SCHEMAREQ -> all DICT
+ Table status = DROPPED
+
+---------------------------------
+
+Sequence of Drop table in node/system restart
+---------------------------------------------
+
+In both node and system restart the node receives the schema information from
+the master. If the table is in a state where it needs to complete the drop
+table activity then DBACC, DBTUP, DBDIH, DBDICT is contacted to drop all files
+related to the table. After this the schema information is updated with the new
+state. Since all nodes receive the same schema information there is no risk of
+different behaviour in the various NDB nodes.
+
+API Requirements for Drop Table
+-------------------------------
+Definition:
+
+ Two tables are NOT the same if they were created with two create
+ tables at different points in time, even if the two create tables
+ had exactly the same definition.
+
+Requirements:
+
+1. Each operation in a transaction refering to a table (by name or by id)
+ should operate on the same table. (This is probably necessary.)
+
+2. Each operation in a transaction refering to a table (by name or by
+ id) should operate on the same table as were defined at the
+ startTransaction timepoint. (This is not strictly necessary for
+ API consistency.)
+
+ Example 1:
+
+ startTransaction()
+
+ drop("TableName1")
+ create("TableName1")
+
+ getNdbOperation("TableName1")
+
+ execute(commit)
+
+ - If both requirements 1 and 2 are fulfilled, then this should lead
+ to "Error: Invalid Schema Version" or similar error
+
+ - If only requirement 1 is fulfilled, then this may be executed
+ without any errors.
+
+
+ Example 2:
+
+ startTransaction()
+
+ getNdbOperation("TableName1")
+ execute(NoCommit)
+
+ drop("TableName1")
+ create("TableName1")
+
+ getNdbOperation("TableName1")
+
+ execute(commit)
+
+ - This should always lead to "Error: Invalid Schema Version" or
+ similar error.
+
diff --git a/ndb/src/kernel/blocks/dbdict/Event.txt b/ndb/src/kernel/blocks/dbdict/Event.txt
new file mode 100644
index 00000000000..553c915d9c5
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/Event.txt
@@ -0,0 +1,102 @@
+
+Event creation
+
+USER DICT(Master) UTIL SUMA
+================================================================================
+CREATE_EVENT_REQ::create
+-------------------------->
+ - Get ID
+ CREATE_SUBID
+ ----------------------------------------------->
+ <-----------------------------------------------
+ - insert into system table
+ UTIL_PREPARE::insert
+ ------------------------>
+ <------------------------
+ UTIL_EXECUTE
+ ------------------------>
+ <------------------------
+CREATE_EVENT_CONF
+<--------------------------
+
+
+Event dropping
+
+USER DICT(Master) (Slaves) UTIL SUMA
+================================================================================
+DROP_EVENT_REQ
+-------------------------->
+ - read from system table
+ UTIL_PREPARE::read
+ ------------------------------------>
+ <------------------------------------
+ UTIL_EXECUTE
+ ------------------------------------>
+ <------------------------------------
+ SUB_REMOVE_REQ
+ -------------------->
+ SUB_REMOVE
+ ------------------------------>
+ <------------------------------
+ SUB_REMOVE_CONF
+ <--------------------
+ - remove from system table
+ UTIL_PREPARE::delete
+ ------------------------------------>
+ <------------------------------------
+ UTIL_EXECUTE
+ ------------------------------------>
+ <------------------------------------
+DROP_EVENT_CONF
+<--------------------------
+
+
+
+create NdbEventOperation
+
+USER DICT(Master) (Slaves) UTIL
+=======================================================================
+CREATE_EVENT_REQ::get
+-------------------------->
+ - read from system table
+ UTIL_PREPARE::read
+ ---------------------------------------->
+ <----------------------------------------
+ UTIL_EXECUTE
+ ---------------------------------------->
+ <----------------------------------------
+ SUMA
+ CREATE_EVENT_REQ::after_get ======
+ ---------------------->
+ SUB_CREATE
+ ------------------>
+ <------------------
+ SUB_SYNC
+ ------------------>
+ <------------------
+ CREATE_EVENT_CONF
+ <----------------------
+CREATE_EVENT_CONF
+<-------------------------
+
+
+
+USER DICT(Master) (Slaves) SUMA
+=======================================================================
+SUB_START_REQ
+-------------------------->
+ SUB_START_REQ
+ ---------------------->
+ SUB_START
+ ------------------>
+ <------------------
+ SUB_START_CONF
+ <----------------------
+SUB_START_CONF
+<-------------------------
+
+
+SUB_STOP analogous to SUB_STOP
+
+
+
diff --git a/ndb/src/kernel/blocks/dbdict/Makefile b/ndb/src/kernel/blocks/dbdict/Makefile
new file mode 100644
index 00000000000..46d938114fb
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := dbdict
+
+SOURCES = \
+ Dbdict.cpp
+
+DIRS := printSchemafile
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dbdict/Master_AddTable.sfl b/ndb/src/kernel/blocks/dbdict/Master_AddTable.sfl
new file mode 100644
index 00000000000..1bcec156ef7
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/Master_AddTable.sfl
@@ -0,0 +1,751 @@
+// ---------------------------------------------------------------------------
+// This file contains a signal log trace for DBDICT at the master for a
+// create table. Another file contains the signal log for the participant
+// node. Master node is 2, participant node 4 and api node is 3.
+//
+
+// ---------------------------------------------------------------------------
+// First arrives the table description in a number of DICTTABINFO signals.
+// These have a header of 5 words (see DictTabInfo.hpp for details) and
+// upto 20 words of property data per signal. The property data is packed
+// by the SimpleProperties class.
+// ---------------------------------------------------------------------------
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57069 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 0 "API", s.proc: 3, s.sigId: 940284 length: 25 trace: 0
+ H'00010003 H'00047700 H'00000001 H'00000042 H'00000000 H'4e444250 H'524f5053
+ H'00010000 H'00000000 H'1c0a1203 H'524f4c46 H'00020001 H'0000000a H'56504e5f
+ H'55534552 H'53000000 H'0001000a H'0000004b H'000203e8 H'00000007 H'56504e5f
+ H'49440000 H'000103ee H'00000001 H'000203e8
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57069 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 0 "API", s.proc: 3, s.sigId: 940284 length: 25 trace: 0
+ H'00010003 H'00047700 H'00000001 H'00000042 H'00000014 H'00000007 H'56504e5f
+ H'4e420000 H'000103ee H'00000001 H'000203e8 H'0000000d H'44495245 H'43544f52
+ H'595f4e42 H'00000000 H'000103eb H'00000003 H'000103ed H'0000000a H'000103ec
+ H'00000002 H'000203e8 H'00000010 H'4c415354
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57069 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 0 "API", s.proc: 3, s.sigId: 940284 length: 25 trace: 0
+ H'00010003 H'00047700 H'00000001 H'00000042 H'00000028 H'5f43414c H'4c5f5041
+ H'52545900 H'000103eb H'00000003 H'000103ed H'0000000a H'000103ec H'00000002
+ H'000203e8 H'00000006 H'44455343 H'52000000 H'000103eb H'00000003 H'000103ed
+ H'00000064 H'000103ec H'00000002 H'00010005
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57069 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 0 "API", s.proc: 3, s.sigId: 940284 length: 11 trace: 0
+ H'00010003 H'00047700 H'00000001 H'00000042 H'0000003c H'00000002 H'00010006
+ H'00000005 H'0001000c H'00000002 H'0000ffff
+
+// ---------------------------------------------------------------------------
+// Send DICT_SCHEMAREQ to all nodes including ourselves to write the state
+// ADD_STARTED in the schema file for the new table.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57069 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000000 H'00000000 H'00000001
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57069 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000000 H'00000000 H'00000001
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57069 gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57077 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000000 H'00000000 H'00000001
+
+// ---------------------------------------------------------------------------
+// Write both schema files with new state of table added.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57069 length: 7 trace: 0
+ UserReference: H'00fa0002, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050100
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57081 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57082 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 99
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57081 length: 8 trace: 0
+ FilePointer: 99
+ UserReference: H'00fa0002, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57090 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57091 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57090 length: 4 trace: 0
+ FilePointer: 99
+ UserReference: H'00fa0002, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57099 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57100 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57099 length: 7 trace: 0
+ UserReference: H'00fa0002, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050200
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57111 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57112 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 100
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57111 length: 8 trace: 0
+ FilePointer: 100
+ UserReference: H'00fa0002, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57123 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57124 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57123 length: 4 trace: 0
+ FilePointer: 100
+ UserReference: H'00fa0002, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57133 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 1 trace: 0
+ H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57135 length: 1 trace: 0
+ H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, s.sigId: 46718 length: 1 trace: 0
+ H'00000004
+
+// ---------------------------------------------------------------------------
+// Pack Table description into pages in DICT using SimpleProperties class.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 3 trace: 0
+ H'00000001 H'00000002 H'00000000
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57140 length: 3 trace: 0
+ H'00000001 H'00000002 H'00000000
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57141 length: 2 trace: 0
+ H'00000002 H'00000002
+
+// ---------------------------------------------------------------------------
+// Send the table description over to the other NDB nodes.
+// A CONTINUEB is sent for each signal sent to avoid overloading the
+// transporters.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000000 H'4e444250 H'524f5053
+ H'00002000 H'0000001c H'1c0a1203 H'524f4c46 H'00020001 H'0000000a H'56504e5f
+ H'55534552 H'53000000 H'0001000a H'0000004b H'000203e8 H'00000007 H'56504e5f
+ H'49440000 H'1cc03924 H'00000001 H'000203e8
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57142 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000014 H'00000007 H'56504e5f
+ H'4e420000 H'000103ee H'00000001 H'000203e8 H'0000000d H'44495245 H'43544f52
+ H'595f4e42 H'00000000 H'000103eb H'00000003 H'524f4c46 H'00020001 H'0000000a
+ H'56504e5f H'55534552 H'53000010 H'00010002
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57143 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000028 H'00000002 H'00010011
+ H'00000003 H'00010003 H'00000001 H'00010005 H'00000002 H'00010006 H'00000005
+ H'0001000a H'0000004b H'0001000c H'00000002 H'000203e8 H'00000007 H'56504e5f
+ H'49440064 H'000103e9 H'00000000 H'000103ee
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57144 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'0000003c H'00000001 H'000203e8
+ H'00000007 H'56504e5f H'4e420002 H'000103e9 H'00000001 H'000103ee H'00000001
+ H'000203e8 H'0000000d H'44495245 H'43544f52 H'595f4e42 H'00000000 H'000103e9
+ H'00000002 H'000103eb H'00000003 H'000103ec
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57145 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000050 H'00000002 H'000103ed
+ H'0000000a H'000203e8 H'00000010 H'4c415354 H'5f43414c H'4c5f5041 H'52545900
+ H'000103e9 H'00000003 H'000103eb H'00000003 H'000103ec H'00000002 H'000103ed
+ H'0000000a H'000203e8 H'00000006 H'44455343
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57132 gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57146 length: 2 trace: 0
+ H'00000002 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 15 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000064 H'52000000 H'000103e9
+ H'00000004 H'000103eb H'00000003 H'000103ec H'00000002 H'000103ed H'00000064
+ H'0000ffff
+
+// ---------------------------------------------------------------------------
+// In parallel with sending the table description to other nodes we will also
+// write the table description to our local file system.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57132 length: 7 trace: 0
+ UserReference: H'00fa0002, userPointer: H'00000000
+ FileNumber[1-4]: H'00000002 H'ffffffff H'00000001 H'010401ff
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57165 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57166 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 101
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57165 length: 8 trace: 0
+ FilePointer: 101
+ UserReference: H'00fa0002, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000000, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57177 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57178 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57177 length: 4 trace: 0
+ FilePointer: 101
+ UserReference: H'00fa0002, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57186 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57187 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57186 length: 7 trace: 0
+ UserReference: H'00fa0002, userPointer: H'00000000
+ FileNumber[1-4]: H'00000002 H'ffffffff H'00000001 H'010402ff
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57195 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57196 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 102
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57195 length: 8 trace: 0
+ FilePointer: 102
+ UserReference: H'00fa0002, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000000, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57204 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57205 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57204 length: 4 trace: 0
+ FilePointer: 102
+ UserReference: H'00fa0002, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57218 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57219 length: 1 trace: 0
+ UserPointer: H'00000000
+
+// ---------------------------------------------------------------------------
+// Completed writing to our file system the table description.
+// ---------------------------------------------------------------------------
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57229 gsn: 24 "DICTTABCONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, s.sigId: 46803 length: 2 trace: 0
+ H'00000002 H'00000004
+
+// ---------------------------------------------------------------------------
+// Also the participant have completed writing the table description to file.
+// ---------------------------------------------------------------------------
+
+// ---------------------------------------------------------------------------
+// Write the state UPDATE_PAGE_COUNT to schema file for the new table.
+// This also contains the number of pages used for the table description.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57229 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000001 H'00000000 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57229 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000001 H'00000000 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57229 gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57234 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000001 H'00000000 H'00000002
+
+// ---------------------------------------------------------------------------
+// Write schema file to disk
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57229 length: 7 trace: 0
+ UserReference: H'00fa0002, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050100
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57238 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57239 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 103
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57238 length: 8 trace: 0
+ FilePointer: 103
+ UserReference: H'00fa0002, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57247 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57248 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57247 length: 4 trace: 0
+ FilePointer: 103
+ UserReference: H'00fa0002, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57257 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57258 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57257 length: 7 trace: 0
+ UserReference: H'00fa0002, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050200
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57267 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57268 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 104
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57267 length: 8 trace: 0
+ FilePointer: 104
+ UserReference: H'00fa0002, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57279 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57283 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57279 length: 4 trace: 0
+ FilePointer: 104
+ UserReference: H'00fa0002, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57290 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 57291 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57290 length: 1 trace: 0
+ H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57290 gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57293 length: 1 trace: 0
+ H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57299 gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, s.sigId: 46860 length: 1 trace: 0
+ H'00000004
+
+// ---------------------------------------------------------------------------
+// All schema files in the system have been updated.
+// ---------------------------------------------------------------------------
+
+// ---------------------------------------------------------------------------
+// Now control is given to DIH for adding the fragments needed by this table.
+// We first seize a record in DIH and then we send the add table request with
+// the needed table parameters.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 246 "DBDIH", r.proc: 2, gsn: 238 "DISEIZEREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57299 length: 2 trace: 0
+ H'00000000 H'00fa0002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57299 gsn: 236 "DISEIZECONF" prio: 1
+s.bn: 246 "DBDIH", s.proc: 2, s.sigId: 57304 length: 2 trace: 0
+ H'00000000 H'00000210
+---- Send ----- Signal ----------------
+r.bn: 246 "DBDIH", r.proc: 2, gsn: 187 "DIADDTABREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57299 length: 6 trace: 0
+ H'00000210 H'00000002 H'00000000 H'00000006 H'00000000 H'00000001
+
+// ---------------------------------------------------------------------------
+// DIH requests us to add a certain fragment replica.
+// ---------------------------------------------------------------------------
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57400 gsn: 195 "DICTFRAGSREQ" prio: 1
+s.bn: 246 "DBDIH", s.proc: 2, s.sigId: 57418 length: 7 trace: 0
+ H'00000000 H'00000000 H'00000000 H'00000002 H'00150040 H'00000001 H'00000002
+
+// ---------------------------------------------------------------------------
+// We add the fragment by contacting LQH through sending a LQHFRAGREQ and
+// a number of LQHADDATTREQ (in this case only one since not more than 8
+// attributes).
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 247 "DBLQH", r.proc: 2, gsn: 313 "LQHFRAGREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57400 length: 17 trace: 0
+ H'00000000 H'00fa0002 H'00000000 H'00000000 H'00000002 H'00000001 H'00000050
+ H'0000004b H'00000006 H'00000001 H'00000000 H'00000005 H'00000000 H'00000000
+ H'00000001 H'00000002 H'00000000
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57400 gsn: 311 "LQHFRAGCONF" prio: 1
+s.bn: 247 "DBLQH", s.proc: 2, s.sigId: 57428 length: 2 trace: 0
+ H'00000000 H'00000000
+---- Send ----- Signal ----------------
+r.bn: 247 "DBLQH", r.proc: 2, gsn: 310 "LQHADDATTREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57400 length: 12 trace: 0
+ H'00000000 H'00000005 H'00000000 H'00012255 H'00000001 H'00012255 H'00000002
+ H'000a2236 H'00000003 H'000a2236 H'00000004 H'00642236
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57400 gsn: 308 "LQHADDATTCONF" prio: 1
+s.bn: 247 "DBLQH", s.proc: 2, s.sigId: 57450 length: 1 trace: 0
+ H'00000000
+
+// ---------------------------------------------------------------------------
+// When we have completed adding the fragment we send DINEXTNODEREQ (should
+// change name to DICTFRAGSCONF) to DIH indicate we have completed the task.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 246 "DBDIH", r.proc: 2, gsn: 231 "DINEXTNODEREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57400 length: 4 trace: 0
+ H'00000210 H'00000000 H'00000001 H'00000000
+
+// ---------------------------------------------------------------------------
+// We continue by performing the same task again for the next fragment replica.
+// We skip this from this log since they contain no more interesting stuff.
+// ---------------------------------------------------------------------------
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 57618 gsn: 185 "DIADDTABCONF" prio: 1
+s.bn: 246 "DBDIH", s.proc: 2, s.sigId: 57655 length: 2 trace: 0
+ H'00000000 H'00000002
+
+// ---------------------------------------------------------------------------
+// Now that we have added all fragments DIH gives back control to DICT by
+// sending DIADDTABCONF.
+// ---------------------------------------------------------------------------
+
+// ---------------------------------------------------------------------------
+// It is now time to decide which global checkpoint this table will be born.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 246 "DBDIH", r.proc: 2, gsn: 499 "WAIT_GCP_REQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 57618 length: 3 trace: 0
+ H'00fa0002 H'00000000 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58288 gsn: 501 "WAIT_GCP_CONF" prio: 1
+s.bn: 246 "DBDIH", s.proc: 2, s.sigId: 58296 length: 2 trace: 0
+ H'00000000 H'0000000c
+
+// ---------------------------------------------------------------------------
+// We can update all schema files in the system with this global checkpoint
+// number. We are certain that no transaction will be performed on the table
+// before this global checkpoint.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58288 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000001 H'0000000c H'00000003
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58288 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000001 H'0000000c H'00000003
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58288 gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 58298 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000001 H'0000000c H'00000003
+
+// ---------------------------------------------------------------------------
+// Write schema files as usual when updating schema file state.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58288 length: 7 trace: 0
+ UserReference: H'00fa0002, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050100
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58304 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 58305 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 117
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58304 length: 8 trace: 0
+ FilePointer: 117
+ UserReference: H'00fa0002, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58315 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 58316 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58315 length: 4 trace: 0
+ FilePointer: 117
+ UserReference: H'00fa0002, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58326 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 58327 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58326 length: 7 trace: 0
+ UserReference: H'00fa0002, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050200
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58339 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 58340 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 118
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58339 length: 8 trace: 0
+ FilePointer: 118
+ UserReference: H'00fa0002, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58348 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 58349 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 2, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58348 length: 4 trace: 0
+ FilePointer: 118
+ UserReference: H'00fa0002, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 2, s.sigId: 58360 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 1 trace: 0
+ H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 58364 length: 1 trace: 0
+ H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, s.sigId: 47846 length: 1 trace: 0
+ H'00000004
+
+// ---------------------------------------------------------------------------
+// Commit the table for usage in DIH and LQH in all nodes.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 247 "DBLQH", r.proc: 2, gsn: 398 "TAB_COMMITREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 3 trace: 0
+ H'00000000 H'00fa0002 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 246 "DBDIH", r.proc: 2, gsn: 398 "TAB_COMMITREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 3 trace: 0
+ H'00000001 H'00fa0002 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 247 "DBLQH", r.proc: 4, gsn: 398 "TAB_COMMITREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 3 trace: 0
+ H'00000000 H'00fa0002 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 246 "DBDIH", r.proc: 4, gsn: 398 "TAB_COMMITREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 3 trace: 0
+ H'00000001 H'00fa0002 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 396 "TAB_COMMITCONF" prio: 1
+s.bn: 247 "DBLQH", s.proc: 2, s.sigId: 58370 length: 3 trace: 0
+ H'00000000 H'00000002 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 396 "TAB_COMMITCONF" prio: 1
+s.bn: 246 "DBDIH", s.proc: 2, s.sigId: 58371 length: 3 trace: 0
+ H'00000001 H'00000002 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 396 "TAB_COMMITCONF" prio: 1
+s.bn: 247 "DBLQH", s.proc: 4, s.sigId: 47846 length: 3 trace: 0
+ H'00000000 H'00000004 H'00000002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 396 "TAB_COMMITCONF" prio: 1
+s.bn: 246 "DBDIH", s.proc: 4, s.sigId: 47846 length: 3 trace: 0
+ H'00000001 H'00000004 H'00000002
+
+// ---------------------------------------------------------------------------
+// Finally also open the table for usage from TC in all nodes.
+// After this signal is received in TC it is ok to execute transactions on
+// this new empty table.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 245 "DBTC", r.proc: 2, gsn: 404 "TC_SCHVERREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 5 trace: 0
+ H'00000002 H'00000001 H'00000001 H'00fa0002 H'00000000
+---- Send ----- Signal ----------------
+r.bn: 245 "DBTC", r.proc: 4, gsn: 404 "TC_SCHVERREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 5 trace: 0
+ H'00000002 H'00000001 H'00000001 H'00fa0002 H'00000000
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 403 "TC_SCHVERCONF" prio: 1
+s.bn: 245 "DBTC", s.proc: 2, s.sigId: 58376 length: 2 trace: 0
+ H'00000002 H'00000000
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 403 "TC_SCHVERCONF" prio: 1
+s.bn: 245 "DBTC", s.proc: 4, s.sigId: 47846 length: 2 trace: 0
+ H'00000002 H'00000001
+
+// ---------------------------------------------------------------------------
+// Unblock dictionary to allow for another add table.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 444 "UNBLO_DICTREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 1 trace: 0
+ H'00fa0002
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 444 "UNBLO_DICTREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 1 trace: 0
+ H'00fa0002
+
+// ---------------------------------------------------------------------------
+// Send the confirmation to the requesting application process.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 1 "API", r.proc: 3, gsn: 24 "DICTTABCONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 3 trace: 0
+ H'00047700 H'00000002 H'00000001
+
+// ---------------------------------------------------------------------------
+// Also release the connection in DIH that was previously established.
+// ---------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 246 "DBDIH", r.proc: 2, gsn: 234 "DIRELEASEREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, sigId: 58359 length: 3 trace: 0
+ H'00000210 H'00000000 H'00fa0002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 444 "UNBLO_DICTREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 58378 length: 1 trace: 0
+ H'00fa0002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, sigId: 58359 gsn: 232 "DIRELEASECONF" prio: 1
+s.bn: 246 "DBDIH", s.proc: 2, s.sigId: 58380 length: 1 trace: 0
+ H'00000000
+
+// ---------------------------------------------------------------------------
+// Now all actions regarding this add table have completed.
+// ---------------------------------------------------------------------------
diff --git a/ndb/src/kernel/blocks/dbdict/SchemaFile.hpp b/ndb/src/kernel/blocks/dbdict/SchemaFile.hpp
new file mode 100644
index 00000000000..7c3223d3d14
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/SchemaFile.hpp
@@ -0,0 +1,57 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBDICT_SCHEMA_FILE_HPP
+#define DBDICT_SCHEMA_FILE_HPP
+
+#include <ndb_types.h>
+#include <string.h>
+
+struct SchemaFile {
+ char Magic[8];
+ Uint32 ByteOrder;
+ Uint32 NdbVersion;
+ Uint32 FileSize; // In bytes
+ Uint32 Unused;
+
+ Uint32 CheckSum;
+
+ enum TableState {
+ INIT = 0,
+ ADD_STARTED = 1,
+ TABLE_ADD_COMMITTED = 2,
+ DROP_TABLE_STARTED = 3,
+ DROP_TABLE_COMMITTED = 4,
+ ALTER_TABLE_COMMITTED = 5
+ };
+
+ struct TableEntry {
+ Uint32 m_tableState;
+ Uint32 m_tableVersion;
+ Uint32 m_tableType;
+ Uint32 m_noOfPages;
+ Uint32 m_gcp;
+
+ bool operator==(const TableEntry& o) const {
+ return memcmp(this, &o, sizeof(* this))== 0;
+ }
+ };
+
+ Uint32 NoOfTableEntries;
+ TableEntry TableEntries[1];
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/dbdict/Slave_AddTable.sfl b/ndb/src/kernel/blocks/dbdict/Slave_AddTable.sfl
new file mode 100644
index 00000000000..8740be9595d
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/Slave_AddTable.sfl
@@ -0,0 +1,416 @@
+// ---------------------------------------------------------------------------
+// This file contains a signal log trace for DBDICT at the participant for a
+// add table. Another file contains the signal log for the master
+// node. Master node is 2, participant node 4 and api node is 3.
+//
+
+// ---------------------------------------------------------------------------
+//--------------------------------------------------------------------------
+// Master requests us to save a new state of the table in the schema file
+// == ADD_STARTED
+//--------------------------------------------------------------------------
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46661 gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57069 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000000 H'00000000 H'00000001
+
+//--------------------------------------------------------------------------
+// Write the new state to the schema files.
+//--------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46661 length: 7 trace: 0
+ UserReference: H'00fa0004, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050100
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46669 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46670 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 99
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46669 length: 8 trace: 0
+ FilePointer: 99
+ UserReference: H'00fa0004, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46679 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46680 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46679 length: 4 trace: 0
+ FilePointer: 99
+ UserReference: H'00fa0004, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46690 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46691 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46690 length: 7 trace: 0
+ UserReference: H'00fa0004, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050200
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46700 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46701 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 100
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46700 length: 8 trace: 0
+ FilePointer: 100
+ UserReference: H'00fa0004, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46709 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46710 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46709 length: 4 trace: 0
+ FilePointer: 100
+ UserReference: H'00fa0004, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46718 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46719 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46718 length: 1 trace: 0
+ H'00000004
+
+//--------------------------------------------------------------------------
+// We receive the table description from the master node.
+// We set the data in the DICT block. (table and attribute records).
+//--------------------------------------------------------------------------
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46718 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000000 H'4e444250 H'524f5053
+ H'00002000 H'0000001c H'1c0a1203 H'524f4c46 H'00020001 H'0000000a H'56504e5f
+ H'55534552 H'53000000 H'0001000a H'0000004b H'000203e8 H'00000007 H'56504e5f
+ H'49440000 H'1cc03924 H'00000001 H'000203e8
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46718 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000014 H'00000007 H'56504e5f
+ H'4e420000 H'000103ee H'00000001 H'000203e8 H'0000000d H'44495245 H'43544f52
+ H'595f4e42 H'00000000 H'000103eb H'00000003 H'524f4c46 H'00020001 H'0000000a
+ H'56504e5f H'55534552 H'53000010 H'00010002
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46718 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000028 H'00000002 H'00010011
+ H'00000003 H'00010003 H'00000001 H'00010005 H'00000002 H'00010006 H'00000005
+ H'0001000a H'0000004b H'0001000c H'00000002 H'000203e8 H'00000007 H'56504e5f
+ H'49440064 H'000103e9 H'00000000 H'000103ee
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46718 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'0000003c H'00000001 H'000203e8
+ H'00000007 H'56504e5f H'4e420002 H'000103e9 H'00000001 H'000103ee H'00000001
+ H'000203e8 H'0000000d H'44495245 H'43544f52 H'595f4e42 H'00000000 H'000103e9
+ H'00000002 H'000103eb H'00000003 H'000103ec
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46718 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57132 length: 25 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000050 H'00000002 H'000103ed
+ H'0000000a H'000203e8 H'00000010 H'4c415354 H'5f43414c H'4c5f5041 H'52545900
+ H'000103e9 H'00000003 H'000103eb H'00000003 H'000103ec H'00000002 H'000103ed
+ H'0000000a H'000203e8 H'00000006 H'44455343
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46718 gsn: 204 "DICTTABINFO" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57132 length: 15 trace: 0
+ H'00fa0002 H'00000000 H'00000002 H'0000006e H'00000064 H'52000000 H'000103e9
+ H'00000004 H'000103eb H'00000003 H'000103ec H'00000002 H'000103ed H'00000064
+ H'0000ffff
+
+//--------------------------------------------------------------------------
+// Pack the table description into pages.
+//--------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46718 length: 3 trace: 0
+ H'00000001 H'00000002 H'00000000
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46718 gsn: 164 "CONTINUEB" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, s.sigId: 46730 length: 3 trace: 0
+ H'00000001 H'00000002 H'00000000
+
+//--------------------------------------------------------------------------
+// Write the pages of the table description to disk.
+//--------------------------------------------------------------------------
+
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46718 length: 7 trace: 0
+ UserReference: H'00fa0004, userPointer: H'00000000
+ FileNumber[1-4]: H'00000002 H'ffffffff H'00000001 H'010401ff
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46748 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46749 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 101
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46748 length: 8 trace: 0
+ FilePointer: 101
+ UserReference: H'00fa0004, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000000, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46757 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46758 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46757 length: 4 trace: 0
+ FilePointer: 101
+ UserReference: H'00fa0004, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46766 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46767 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46766 length: 7 trace: 0
+ UserReference: H'00fa0004, userPointer: H'00000000
+ FileNumber[1-4]: H'00000002 H'ffffffff H'00000001 H'010402ff
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46783 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46784 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 102
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46783 length: 8 trace: 0
+ FilePointer: 102
+ UserReference: H'00fa0004, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000000, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46794 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46795 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46794 length: 4 trace: 0
+ FilePointer: 102
+ UserReference: H'00fa0004, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46803 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46804 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 24 "DICTTABCONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46803 length: 2 trace: 0
+ H'00000002 H'00000004
+
+//--------------------------------------------------------------------------
+// Update schema file ín memory and on disk to UPDATE_PAGE_COUNT.
+//--------------------------------------------------------------------------
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46803 gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 57229 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000001 H'00000000 H'00000002
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46803 length: 7 trace: 0
+ UserReference: H'00fa0004, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050100
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46813 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46814 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 103
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46813 length: 8 trace: 0
+ FilePointer: 103
+ UserReference: H'00fa0004, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46823 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46824 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46823 length: 4 trace: 0
+ FilePointer: 103
+ UserReference: H'00fa0004, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46833 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46834 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46833 length: 7 trace: 0
+ UserReference: H'00fa0004, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050200
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46842 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46843 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 104
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46842 length: 8 trace: 0
+ FilePointer: 104
+ UserReference: H'00fa0004, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46851 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46852 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46851 length: 4 trace: 0
+ FilePointer: 104
+ UserReference: H'00fa0004, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 46860 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 46861 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 46860 length: 1 trace: 0
+ H'00000004
+
+//--------------------------------------------------------------------------
+// Update schema file with information about the starting global checkpoint
+// identity.
+//--------------------------------------------------------------------------
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 47782 gsn: 132 "DICT_SCHEMAREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 58288 length: 7 trace: 0
+ H'00010003 H'00047700 H'00000002 H'00000001 H'00000001 H'0000000c H'00000003
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 47782 length: 7 trace: 0
+ UserReference: H'00fa0004, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050100
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 47793 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 47794 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 117
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 47793 length: 8 trace: 0
+ FilePointer: 117
+ UserReference: H'00fa0004, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 47804 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 47805 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 47804 length: 4 trace: 0
+ FilePointer: 117
+ UserReference: H'00fa0004, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 47817 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 47818 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 261 "FSOPENREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 47817 length: 7 trace: 0
+ UserReference: H'00fa0004, userPointer: H'00000000
+ FileNumber[1-4]: H'ffffffff H'ffffffff H'ffffffff H'01050200
+ FileFlags: H'00000311 Open write only, Create new file, Truncate existing file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 47826 gsn: 259 "FSOPENCONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 47827 length: 3 trace: 0
+ UserPointer: H'00000000
+ FilePointer: 118
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 272 "FSWRITEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 47826 length: 8 trace: 0
+ FilePointer: 118
+ UserReference: H'00fa0004, UserPointer: H'00000000
+ Operation flag: H'00000011, Sync, Format=Array of pages
+ varIndex: 1
+ numberOfPages: 1
+ pageData: H'00000008, H'00000000
+
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 47836 gsn: 270 "FSWRITECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 47837 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 253 "NDBFS", r.proc: 4, gsn: 257 "FSCLOSEREQ" prio: 0
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 47836 length: 4 trace: 0
+ FilePointer: 118
+ UserReference: H'00fa0004, userPointer: H'00000000
+ Flags: H'00000000, Don't remove file
+---- Received - Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 47846 gsn: 255 "FSCLOSECONF" prio: 1
+s.bn: 253 "NDBFS", s.proc: 4, s.sigId: 47847 length: 1 trace: 0
+ UserPointer: H'00000000
+---- Send ----- Signal ----------------
+r.bn: 250 "DBDICT", r.proc: 2, gsn: 133 "DICT_SCHEMACONF" prio: 1
+s.bn: 250 "DBDICT", s.proc: 4, sigId: 47846 length: 1 trace: 0
+ H'00000004
+---- Received - Signal ----------------
+
+//--------------------------------------------------------------------------
+// Finally unblock the DICT block so that it can handle add table as master
+// if it becomes master in the future.
+//--------------------------------------------------------------------------
+
+r.bn: 250 "DBDICT", r.proc: 4, sigId: 47846 gsn: 444 "UNBLO_DICTREQ" prio: 1
+s.bn: 250 "DBDICT", s.proc: 2, s.sigId: 58359 length: 1 trace: 0
+ H'00fa0002
+
+//--------------------------------------------------------------------------
+// We completed the add table operation.
+//--------------------------------------------------------------------------
+
diff --git a/ndb/src/kernel/blocks/dbdict/printSchemafile/Makefile b/ndb/src/kernel/blocks/dbdict/printSchemafile/Makefile
new file mode 100644
index 00000000000..1b097e2ce37
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/printSchemafile/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := printSchemafile
+BIN_TARGET_ARCHIVES := portlib general
+
+CCFLAGS_LOC += -I..
+
+SOURCES := printSchemafile.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dbdict/printSchemafile/printSchemafile.cpp b/ndb/src/kernel/blocks/dbdict/printSchemafile/printSchemafile.cpp
new file mode 100644
index 00000000000..b16990bda6c
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdict/printSchemafile/printSchemafile.cpp
@@ -0,0 +1,99 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <SchemaFile.hpp>
+#include <stdlib.h>
+#include <string.h>
+#include <sys/stat.h>
+
+void
+usage(const char * prg){
+ ndbout << "Usage " << prg
+ << " P0.SchemaLog" << endl;
+}
+
+void
+fill(const char * buf, int mod){
+ int len = strlen(buf)+1;
+ ndbout << buf << " ";
+ while((len % mod) != 0){
+ ndbout << " ";
+ len++;
+ }
+}
+
+void
+print(const char * filename, const SchemaFile * file){
+ ndbout << "----- Schemafile: " << filename << " -----" << endl;
+ ndbout_c("Magic: %.*s ByteOrder: %.8x NdbVersion: %d FileSize: %d",
+ sizeof(file->Magic), file->Magic,
+ file->ByteOrder,
+ file->NdbVersion,
+ file->FileSize);
+
+ for(Uint32 i = 0; i<file->NoOfTableEntries; i++){
+ SchemaFile::TableEntry te = file->TableEntries[i];
+ if(te.m_tableState != SchemaFile::INIT){
+ ndbout << "Table " << i << ": State = " << te.m_tableState
+ << " version = " << te.m_tableVersion
+ << " type = " << te.m_tableType
+ << " noOfPages = " << te.m_noOfPages
+ << " gcp: " << te.m_gcp << endl;
+ }
+ }
+}
+
+NDB_COMMAND(printSchemafile,
+ "printSchemafile", "printSchemafile", "Prints a schemafile", 16384){
+ if(argc < 2){
+ usage(argv[0]);
+ return 0;
+ }
+
+ const char * filename = argv[1];
+
+ struct stat sbuf;
+ const int res = stat(filename, &sbuf);
+ if(res != 0){
+ ndbout << "Could not find file: \"" << filename << "\"" << endl;
+ return 0;
+ }
+ const Uint32 bytes = sbuf.st_size;
+
+ Uint32 * buf = new Uint32[bytes/4+1];
+
+ FILE * f = fopen(filename, "rb");
+ if(f == 0){
+ ndbout << "Failed to open file" << endl;
+ delete [] buf;
+ return 0;
+ }
+ Uint32 sz = fread(buf, 1, bytes, f);
+ fclose(f);
+ if(sz != bytes){
+ ndbout << "Failure while reading file" << endl;
+ delete [] buf;
+ return 0;
+ }
+
+ print(filename, (SchemaFile *)&buf[0]);
+ delete [] buf;
+ return 0;
+}
diff --git a/ndb/src/kernel/blocks/dbdih/Dbdih.hpp b/ndb/src/kernel/blocks/dbdih/Dbdih.hpp
new file mode 100644
index 00000000000..4ec699cebec
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdih/Dbdih.hpp
@@ -0,0 +1,1606 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBDIH_H
+#define DBDIH_H
+
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <SimulatedBlock.hpp>
+#include "Sysfile.hpp"
+#include <ArrayList.hpp>
+#include <SignalCounter.hpp>
+
+#include <signaldata/MasterLCP.hpp>
+#include <signaldata/CopyGCIReq.hpp>
+#include <blocks/mutexes.hpp>
+
+#ifdef DBDIH_C
+
+/*###################*/
+/* FILE SYSTEM FLAGS */
+/*###################*/
+#define ZLIST_OF_PAIRS 0
+#define ZLIST_OF_PAIRS_SYNCH 16
+#define ZOPEN_READ_WRITE 2
+#define ZCREATE_READ_WRITE 0x302
+#define ZCLOSE_NO_DELETE 0
+#define ZCLOSE_DELETE 1
+
+/*###############*/
+/* NODE STATES */
+/*###############*/
+#define ZIDLE 0
+#define ZACTIVE 1
+
+/*#########*/
+/* GENERAL */
+/*#########*/
+#define ZVAR_NO_WORD 1
+#define ZVAR_NO_CRESTART_INFO 20
+#define ZVAR_NO_CRESTART_INFO_TO_FILE 21
+#define ZVALID 1
+#define ZINVALID 2
+
+/*###############*/
+/* ERROR CODES */
+/*###############*/
+// ------------------------------------------
+// Error Codes for Transactions (None sofar)
+// ------------------------------------------
+
+// --------------------------------------
+// Error Codes for Add Table
+// --------------------------------------
+#define ZREPLERROR1 306
+#define ZNOTIMPLEMENTED 307
+#define ZTABLEINSTALLED 310
+// --------------------------------------
+// Error Codes for Scan Table
+// --------------------------------------
+#define ZERRONOUSSTATE 308
+
+// --------------------------------------
+// Crash Codes
+// --------------------------------------
+#define ZCOULD_NOT_OCCUR_ERROR 300
+#define ZNOT_MASTER_ERROR 301
+#define ZWRONG_FAILURE_NUMBER_ERROR 302
+#define ZWRONG_START_NODE_ERROR 303
+#define ZNO_REPLICA_FOUND_ERROR 304
+#define ZNODE_ALREADY_STARTING_ERROR 305
+#define ZNODE_START_DISALLOWED_ERROR 309
+
+// --------------------------------------
+// Codes from LQH
+// --------------------------------------
+#define ZNODE_FAILURE_ERROR 400
+
+
+/*#########*/
+/* PHASES */
+/*#########*/
+#define ZNDB_SPH1 1
+#define ZNDB_SPH2 2
+#define ZNDB_SPH3 3
+#define ZNDB_SPH4 4
+#define ZNDB_SPH5 5
+#define ZNDB_SPH6 6
+#define ZNDB_SPH7 7
+#define ZNDB_SPH8 8
+/*#########*/
+/* SIZES */
+/*#########*/
+#define ZPAGEREC 100
+#define ZCREATE_REPLICA_FILE_SIZE 4
+#define ZPROXY_MASTER_FILE_SIZE 10
+#define ZPROXY_FILE_SIZE 10
+#endif
+
+class Dbdih: public SimulatedBlock {
+public:
+
+ // Records
+
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤
+ * THE API CONNECT RECORD IS THE SAME RECORD POINTER AS USED IN THE TC BLOCK
+ *
+ * IT KEEPS TRACK OF ALL THE OPERATIONS CONNECTED TO THIS TRANSACTION.
+ * IT IS LINKED INTO A QUEUE IN CASE THE GLOBAL CHECKPOINT IS CURRENTLY
+ * ONGOING */
+ struct ApiConnectRecord {
+ Uint32 apiGci;
+ Uint32 nextApi;
+ };
+ typedef Ptr<ApiConnectRecord> ApiConnectRecordPtr;
+
+ /*############## CONNECT_RECORD ##############*/
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ /* THE CONNECT RECORD IS CREATED WHEN A TRANSACTION HAS TO START. IT KEEPS
+ ALL INTERMEDIATE INFORMATION NECESSARY FOR THE TRANSACTION FROM THE
+ DISTRIBUTED MANAGER. THE RECORD KEEPS INFORMATION ABOUT THE
+ OPERATIONS THAT HAVE TO BE CARRIED OUT BY THE TRANSACTION AND
+ ALSO THE TRAIL OF NODES FOR EACH OPERATION IN THE THE
+ TRANSACTION.
+ */
+ struct ConnectRecord {
+ enum ConnectState {
+ INUSE = 0,
+ FREE = 1,
+ STARTED = 2
+ };
+ Uint32 nodes[MAX_REPLICAS];
+ ConnectState connectState;
+ Uint32 nfConnect;
+ Uint32 table;
+ Uint32 userpointer;
+ Uint32 nodeCount;
+ BlockReference userblockref;
+ };
+ typedef Ptr<ConnectRecord> ConnectRecordPtr;
+
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ /* THESE RECORDS ARE USED WHEN CREATING REPLICAS DURING SYSTEM */
+ /* RESTART. I NEED A COMPLEX DATA STRUCTURE DESCRIBING THE REPLICAS */
+ /* I WILL TRY TO CREATE FOR EACH FRAGMENT. */
+ /* */
+ /* I STORE A REFERENCE TO THE FOUR POSSIBLE CREATE REPLICA RECORDS */
+ /* IN A COMMON STORED VARIABLE. I ALLOW A MAXIMUM OF 4 REPLICAS TO */
+ /* BE RESTARTED PER FRAGMENT. */
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ struct CreateReplicaRecord {
+ Uint32 logStartGci[MAX_LOG_EXEC];
+ Uint32 logStopGci[MAX_LOG_EXEC];
+ Uint16 logNodeId[MAX_LOG_EXEC];
+ Uint32 createLcpId;
+
+ bool hotSpareUse;
+ Uint32 replicaRec;
+ Uint16 dataNodeId;
+ Uint16 lcpNo;
+ Uint16 noLogNodes;
+ };
+ typedef Ptr<CreateReplicaRecord> CreateReplicaRecordPtr;
+
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ /* THIS RECORD CONTAINS A FILE DESCRIPTION. THERE ARE TWO */
+ /* FILES PER TABLE TO RAISE SECURITY LEVEL AGAINST DISK CRASHES. */
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ struct FileRecord {
+ enum FileStatus {
+ CLOSED = 0,
+ CRASHED = 1,
+ OPEN = 2
+ };
+ enum FileType {
+ TABLE_FILE = 0,
+ GCP_FILE = 1
+ };
+ enum ReqStatus {
+ IDLE = 0,
+ CREATING_GCP = 1,
+ OPENING_GCP = 2,
+ OPENING_COPY_GCI = 3,
+ WRITING_COPY_GCI = 4,
+ CREATING_COPY_GCI = 5,
+ OPENING_TABLE = 6,
+ READING_GCP = 7,
+ READING_TABLE = 8,
+ WRITE_INIT_GCP = 9,
+ TABLE_CREATE = 10,
+ TABLE_WRITE = 11,
+ TABLE_CLOSE = 12,
+ CLOSING_GCP = 13,
+ CLOSING_TABLE_CRASH = 14,
+ CLOSING_TABLE_SR = 15,
+ CLOSING_GCP_CRASH = 16,
+ TABLE_OPEN_FOR_DELETE = 17,
+ TABLE_CLOSE_DELETE = 18
+ };
+ Uint32 fileName[4];
+ Uint32 fileRef;
+ FileStatus fileStatus;
+ FileType fileType;
+ Uint32 nextFile;
+ ReqStatus reqStatus;
+ Uint32 tabRef;
+ };
+ typedef Ptr<FileRecord> FileRecordPtr;
+
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ /* THIS RECORD KEEPS THE STORAGE AND DECISIONS INFORMATION OF A FRAGMENT */
+ /* AND ITS REPLICAS. IF FRAGMENT HAS MORE THAN ONE BACK UP */
+ /* REPLICA THEN A LIST OF MORE NODES IS ATTACHED TO THIS RECORD. */
+ /* EACH RECORD IN MORE LIST HAS INFORMATION ABOUT ONE BACKUP. THIS RECORD */
+ /* ALSO HAVE THE STATUS OF THE FRAGMENT. */
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ /* */
+ /* FRAGMENTSTORE RECORD ALIGNED TO BE 64 BYTES */
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ struct Fragmentstore {
+ Uint16 activeNodes[MAX_REPLICAS];
+ Uint32 preferredPrimary;
+
+ Uint32 oldStoredReplicas; /* "DEAD" STORED REPLICAS */
+ Uint32 storedReplicas; /* "ALIVE" STORED REPLICAS */
+ Uint32 nextFragmentChunk;
+
+ Uint8 distributionKey;
+ Uint8 fragReplicas;
+ Uint8 noOldStoredReplicas; /* NUMBER OF "DEAD" STORED REPLICAS */
+ Uint8 noStoredReplicas; /* NUMBER OF "ALIVE" STORED REPLICAS*/
+ Uint8 noLcpReplicas; ///< No of replicas remaining to be LCP:ed
+ };
+ typedef Ptr<Fragmentstore> FragmentstorePtr;
+
+ /*########### PAGE RECORD ############*/
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ /* THIS RECORD KEEPS INFORMATION ABOUT NODE GROUPS. */
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ struct NodeGroupRecord {
+ Uint32 nodesInGroup[MAX_REPLICAS + 1];
+ Uint32 nextReplicaNode;
+ Uint32 nodeCount;
+ bool activeTakeOver;
+ };
+ typedef Ptr<NodeGroupRecord> NodeGroupRecordPtr;
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ /* THIS RECORD KEEPS INFORMATION ABOUT NODES. */
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ /* RECORD ALIGNED TO BE 64 BYTES. */
+ /*¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤*/
+ enum NodefailHandlingStep {
+ NF_REMOVE_NODE_FROM_TABLE = 1,
+ NF_GCP_TAKE_OVER = 2,
+ NF_LCP_TAKE_OVER = 4
+ };
+
+ struct NodeRecord {
+ NodeRecord() { m_nodefailSteps.clear();}
+
+ enum NodeStatus {
+ NOT_IN_CLUSTER = 0,
+ ALIVE = 1,
+ STARTING = 2,
+ DIED_NOW = 3,
+ DYING = 4,
+ DEAD = 5
+ };
+
+ struct FragmentCheckpointInfo {
+ Uint32 tableId;
+ Uint32 fragId;
+ Uint32 replicaPtr;
+ };
+
+ enum GcpState {
+ READY = 0,
+ PREPARE_SENT = 1,
+ PREPARE_RECEIVED = 2,
+ COMMIT_SENT = 3,
+ NODE_FINISHED = 4,
+ SAVE_REQ_SENT = 5,
+ SAVE_RECEIVED = 6,
+ COPY_GCI_SENT = 7
+ };
+
+ GcpState gcpstate;
+ Sysfile::ActiveStatus activeStatus;
+
+ NodeStatus nodeStatus;
+ bool useInTransactions;
+ bool allowNodeStart;
+ bool copyCompleted;
+ bool m_inclDihLcp;
+
+ FragmentCheckpointInfo startedChkpt[2];
+ FragmentCheckpointInfo queuedChkpt[2];
+
+ Bitmask<1> m_nodefailSteps;
+ Uint32 activeTabptr;
+ Uint32 nextNode;
+ Uint32 ndbversion;
+ Uint32 nodeGroup;
+
+ SignalCounter m_NF_COMPLETE_REP;
+
+ Uint8 dbtcFailCompleted;
+ Uint8 dblqhFailCompleted;
+ Uint8 dbdihFailCompleted;
+ Uint8 dbdictFailCompleted;
+ Uint8 recNODE_FAILREP;
+
+ Uint8 noOfQueuedChkpt;
+ Uint8 noOfStartedChkpt;
+
+ MasterLCPConf::State lcpStateAtTakeOver;
+ };
+ typedef Ptr<NodeRecord> NodeRecordPtr;
+ /**********************************************************************/
+ /* THIS RECORD KEEPS THE INFORMATION ABOUT A TABLE AND ITS FRAGMENTS */
+ /**********************************************************************/
+ struct PageRecord {
+ Uint32 word[2048];
+ /* 8 KBYTE PAGE*/
+ Uint32 nextfreepage;
+ };
+ typedef Ptr<PageRecord> PageRecordPtr;
+
+ /************ REPLICA RECORD *************/
+ /**********************************************************************/
+ /* THIS RECORD KEEPS THE INFORMATION ABOUT A REPLICA OF A FRAGMENT */
+ /**********************************************************************/
+ struct ReplicaRecord {
+ /* -------------------------------------------------------------------- */
+ /* THE GLOBAL CHECKPOINT IDENTITY WHEN THIS REPLICA WAS CREATED. */
+ /* THERE IS ONE INDEX PER REPLICA. A REPLICA INDEX IS CREATED WHEN ANODE*/
+ /* CRASH OCCURS. */
+ /* -------------------------------------------------------------------- */
+ Uint32 createGci[8];
+ /* -------------------------------------------------------------------- */
+ /* THE LAST GLOBAL CHECKPOINT IDENTITY WHICH HAS BEEN SAVED ON DISK. */
+ /* THIS VARIABLE IS ONLY VALID FOR REPLICAS WHICH HAVE "DIED". A REPLICA*/
+ /* "DIES" EITHER WHEN THE NODE CRASHES THAT KEPT THE REPLICA OR BY BEING*/
+ /* STOPPED IN A CONTROLLED MANNER. */
+ /* THERE IS ONE INDEX PER REPLICA. A REPLICA INDEX IS CREATED WHEN ANODE*/
+ /* CRASH OCCURS. */
+ /* -------------------------------------------------------------------- */
+ Uint32 replicaLastGci[8];
+ /* -------------------------------------------------------------------- */
+ /* THE LOCAL CHECKPOINT IDENTITY OF A LOCAL CHECKPOINT. */
+ /* -------------------------------------------------------------------- */
+ Uint32 lcpId[MAX_LCP_STORED];
+ /* -------------------------------------------------------------------- */
+ /* THIS VARIABLE KEEPS TRACK OF THE MAXIMUM GLOBAL CHECKPOINT COMPLETED */
+ /* FOR EACH OF THE LOCAL CHECKPOINTS IN THIS FRAGMENT REPLICA. */
+ /* -------------------------------------------------------------------- */
+ Uint32 maxGciCompleted[MAX_LCP_STORED];
+ /* -------------------------------------------------------------------- */
+ /* THIS VARIABLE KEEPS TRACK OF THE MINIMUM GLOBAL CHECKPOINT STARTEDFOR*/
+ /* EACH OF THE LOCAL CHECKPOINTS IN THIS FRAGMENT REPLICA. */
+ /* -------------------------------------------------------------------- */
+ Uint32 maxGciStarted[MAX_LCP_STORED];
+ /* -------------------------------------------------------------------- */
+ /* THE GLOBAL CHECKPOINT IDENTITY WHEN THE TABLE WAS CREATED. */
+ /* -------------------------------------------------------------------- */
+ Uint32 initialGci;
+
+ /* -------------------------------------------------------------------- */
+ /* THE REFERENCE TO THE NEXT REPLICA. EITHER IT REFERS TO THE NEXT IN */
+ /* THE FREE LIST OR IT REFERS TO THE NEXT IN A LIST OF REPLICAS ON A */
+ /* FRAGMENT. */
+ /* -------------------------------------------------------------------- */
+ Uint32 nextReplica;
+
+ /* -------------------------------------------------------------------- */
+ /* THE NODE ID WHERE THIS REPLICA IS STORED. */
+ /* -------------------------------------------------------------------- */
+ Uint16 procNode;
+
+ /* -------------------------------------------------------------------- */
+ /* The last local checkpoint id started or queued on this replica. */
+ /* -------------------------------------------------------------------- */
+ Uint32 lcpIdStarted; // Started or queued
+
+ /* -------------------------------------------------------------------- */
+ /* THIS VARIABLE SPECIFIES WHAT THE STATUS OF THE LOCAL CHECKPOINT IS.IT*/
+ /* CAN EITHER BE VALID OR INVALID. AT CREATION OF A FRAGMENT REPLICA ALL*/
+ /* LCP'S ARE INVALID. ALSO IF IF INDEX >= NO_LCP THEN THELOCALCHECKPOINT*/
+ /* IS ALWAYS INVALID. IF THE LCP BEFORE THE NEXT_LCP HAS LCP_ID THAT */
+ /* DIFFERS FROM THE LATEST LCP_ID STARTED THEN THE NEXT_LCP IS ALSO */
+ /* INVALID */
+ /* -------------------------------------------------------------------- */
+ Uint8 lcpStatus[MAX_LCP_STORED];
+
+ /* -------------------------------------------------------------------- */
+ /* THE NEXT LOCAL CHECKPOINT TO EXECUTE IN THIS FRAGMENT REPLICA. */
+ /* -------------------------------------------------------------------- */
+ Uint8 nextLcp;
+
+ /* -------------------------------------------------------------------- */
+ /* THE NUMBER OF CRASHED REPLICAS IN THIS REPLICAS SO FAR. */
+ /* -------------------------------------------------------------------- */
+ Uint8 noCrashedReplicas;
+
+ /**
+ * Is a LCP currently ongoing on fragment
+ */
+ Uint8 lcpOngoingFlag;
+ };
+ typedef Ptr<ReplicaRecord> ReplicaRecordPtr;
+
+ /*************************************************************************
+ * TAB_DESCRIPTOR IS A DESCRIPTOR OF THE LOCATION OF THE FRAGMENTS BELONGING
+ * TO THE TABLE.THE INFORMATION ABOUT FRAGMENTS OF A TABLE ARE STORED IN
+ * CHUNKS OF FRAGMENTSTORE RECORDS.
+ * THIS RECORD ALSO HAS THE NECESSARY INFORMATION TO LOCATE A FRAGMENT AND
+ * TO LOCATE A FRAGMENT AND TO TRANSLATE A KEY OF A TUPLE TO THE FRAGMENT IT
+ * BELONGS
+ */
+ struct TabRecord {
+ /**
+ * State for copying table description into pages
+ */
+ enum CopyStatus {
+ CS_IDLE,
+ CS_SR_PHASE1_READ_PAGES,
+ CS_SR_PHASE2_READ_TABLE,
+ CS_SR_PHASE3_COPY_TABLE,
+ CS_REMOVE_NODE,
+ CS_LCP_READ_TABLE,
+ CS_COPY_TAB_REQ,
+ CS_COPY_NODE_STATE,
+ CS_ADD_TABLE_MASTER,
+ CS_ADD_TABLE_SLAVE,
+ CS_INVALIDATE_NODE_LCP
+ };
+ /**
+ * State for copying pages to disk
+ */
+ enum UpdateState {
+ US_IDLE,
+ US_LOCAL_CHECKPOINT,
+ US_REMOVE_NODE,
+ US_COPY_TAB_REQ,
+ US_ADD_TABLE_MASTER,
+ US_ADD_TABLE_SLAVE,
+ US_INVALIDATE_NODE_LCP
+ };
+ enum TabLcpStatus {
+ TLS_ACTIVE = 1,
+ TLS_WRITING_TO_FILE = 2,
+ TLS_COMPLETED = 3
+ };
+ enum TabStatus {
+ TS_IDLE = 0,
+ TS_ACTIVE = 1,
+ TS_CREATING = 2,
+ TS_DROPPING = 3
+ };
+ enum Method {
+ HASH = 0,
+ NOTDEFINED = 1
+ };
+ CopyStatus tabCopyStatus;
+ UpdateState tabUpdateState;
+ TabLcpStatus tabLcpStatus;
+ TabStatus tabStatus;
+ Method method;
+
+ Uint32 pageRef[8];
+//-----------------------------------------------------------------------------
+// Each entry in this array contains a reference to 16 fragment records in a
+// row. Thus finding the correct record is very quick provided the fragment id.
+//-----------------------------------------------------------------------------
+ Uint32 startFid[MAX_NDB_NODES];
+
+ Uint32 tabFile[2];
+ Uint32 connectrec;
+ Uint32 hashpointer;
+ Uint32 mask;
+ Uint32 noOfWords;
+ Uint32 schemaVersion;
+ Uint32 tabRemoveNode;
+ Uint32 totalfragments;
+ Uint32 noOfFragChunks;
+ Uint32 tabErrorCode;
+ struct {
+ Uint32 tabUserRef;
+ Uint32 tabUserPtr;
+ } m_dropTab;
+
+ struct DropTable {
+ Uint32 senderRef;
+ Uint32 senderData;
+ SignalCounter waitDropTabCount;
+ } m_prepDropTab;
+
+ Uint8 kvalue;
+ Uint8 noOfBackups;
+ Uint8 noPages;
+ Uint8 storedTable; /* 0 IF THE TABLE IS A TEMPORARY TABLE */
+ Uint16 tableType;
+ Uint16 primaryTableId;
+ };
+ typedef Ptr<TabRecord> TabRecordPtr;
+
+ /***************************************************************************/
+ /* THIS RECORD IS USED TO KEEP TRACK OF TAKE OVER AND STARTING A NODE. */
+ /* WE KEEP IT IN A RECORD TO ENABLE IT TO BE PARALLELISED IN THE FUTURE. */
+ /**************************************************************************/
+ struct TakeOverRecord {
+ enum ToMasterStatus {
+ IDLE = 0,
+ TO_WAIT_START_TAKE_OVER = 1,
+ TO_START_COPY = 2,
+ TO_START_COPY_ONGOING = 3,
+ TO_WAIT_START = 4,
+ STARTING = 5,
+ SELECTING_NEXT = 6,
+ TO_WAIT_PREPARE_CREATE = 9,
+ PREPARE_CREATE = 10,
+ COPY_FRAG = 11,
+ TO_WAIT_UPDATE_TO = 12,
+ TO_UPDATE_TO = 13,
+ COPY_ACTIVE = 14,
+ TO_WAIT_COMMIT_CREATE = 15,
+ LOCK_MUTEX = 23,
+ COMMIT_CREATE = 16,
+ TO_COPY_COMPLETED = 17,
+ WAIT_LCP = 18,
+ TO_END_COPY = 19,
+ TO_END_COPY_ONGOING = 20,
+ TO_WAIT_ENDING = 21,
+ ENDING = 22
+ };
+ enum ToSlaveStatus {
+ TO_SLAVE_IDLE = 0,
+ TO_SLAVE_STARTED = 1,
+ TO_SLAVE_CREATE_PREPARE = 2,
+ TO_SLAVE_COPY_FRAG_COMPLETED = 3,
+ TO_SLAVE_CREATE_COMMIT = 4,
+ TO_SLAVE_COPY_COMPLETED = 5
+ };
+ Uint32 startGci;
+ Uint32 toCopyNode;
+ Uint32 toCurrentFragid;
+ Uint32 toCurrentReplica;
+ Uint32 toCurrentTabref;
+ Uint32 toFailedNode;
+ Uint32 toStartingNode;
+ Uint32 nextTakeOver;
+ Uint32 prevTakeOver;
+ bool toNodeRestart;
+ ToMasterStatus toMasterStatus;
+ ToSlaveStatus toSlaveStatus;
+ MutexHandle2<DIH_SWITCH_PRIMARY_MUTEX> m_switchPrimaryMutexHandle;
+ };
+ typedef Ptr<TakeOverRecord> TakeOverRecordPtr;
+
+public:
+ Dbdih(const class Configuration &);
+ virtual ~Dbdih();
+
+ struct RWFragment {
+ Uint32 pageIndex;
+ Uint32 wordIndex;
+ Uint32 fragId;
+ TabRecordPtr rwfTabPtr;
+ PageRecordPtr rwfPageptr;
+ };
+ struct CopyTableNode {
+ Uint32 pageIndex;
+ Uint32 wordIndex;
+ Uint32 noOfWords;
+ TabRecordPtr ctnTabPtr;
+ PageRecordPtr ctnPageptr;
+ };
+
+private:
+ BLOCK_DEFINES(Dbdih);
+
+ void execDUMP_STATE_ORD(Signal *);
+ void execNDB_TAMPER(Signal *);
+ void execDEBUG_SIG(Signal *);
+ void execEMPTY_LCP_CONF(Signal *);
+ void execMASTER_GCPREF(Signal *);
+ void execMASTER_GCPREQ(Signal *);
+ void execMASTER_GCPCONF(Signal *);
+ void execMASTER_LCPREF(Signal *);
+ void execMASTER_LCPREQ(Signal *);
+ void execMASTER_LCPCONF(Signal *);
+ void execNF_COMPLETEREP(Signal *);
+ void execSTART_PERMREQ(Signal *);
+ void execSTART_PERMCONF(Signal *);
+ void execSTART_PERMREF(Signal *);
+ void execINCL_NODEREQ(Signal *);
+ void execINCL_NODECONF(Signal *);
+ void execEND_TOREQ(Signal *);
+ void execEND_TOCONF(Signal *);
+ void execSTART_TOREQ(Signal *);
+ void execSTART_TOCONF(Signal *);
+ void execSTART_MEREQ(Signal *);
+ void execSTART_MECONF(Signal *);
+ void execSTART_MEREF(Signal *);
+ void execSTART_COPYREQ(Signal *);
+ void execSTART_COPYCONF(Signal *);
+ void execSTART_COPYREF(Signal *);
+ void execCREATE_FRAGREQ(Signal *);
+ void execCREATE_FRAGCONF(Signal *);
+ void execDIVERIFYREQ(Signal *);
+ void execGCP_SAVECONF(Signal *);
+ void execGCP_PREPARECONF(Signal *);
+ void execGCP_PREPARE(Signal *);
+ void execGCP_NODEFINISH(Signal *);
+ void execGCP_COMMIT(Signal *);
+ void execDIHNDBTAMPER(Signal *);
+ void execCONTINUEB(Signal *);
+ void execCOPY_GCIREQ(Signal *);
+ void execCOPY_GCICONF(Signal *);
+ void execCOPY_TABREQ(Signal *);
+ void execCOPY_TABCONF(Signal *);
+ void execTCGETOPSIZECONF(Signal *);
+ void execTC_CLOPSIZECONF(Signal *);
+
+ void execLCP_FRAG_REP(Signal *);
+ void execLCP_COMPLETE_REP(Signal *);
+ void execSTART_LCP_REQ(Signal *);
+ void execSTART_LCP_CONF(Signal *);
+ MutexHandle2<DIH_START_LCP_MUTEX> c_startLcpMutexHandle;
+ void startLcpMutex_locked(Signal* signal, Uint32, Uint32);
+ void startLcpMutex_unlocked(Signal* signal, Uint32, Uint32);
+
+ MutexHandle2<DIH_SWITCH_PRIMARY_MUTEX> c_switchPrimaryMutexHandle;
+ void switchPrimaryMutex_locked(Signal* signal, Uint32, Uint32);
+ void switchPrimaryMutex_unlocked(Signal* signal, Uint32, Uint32);
+ void switch_primary_stop_node(Signal* signal, Uint32, Uint32);
+
+ void execBLOCK_COMMIT_ORD(Signal *);
+ void execUNBLOCK_COMMIT_ORD(Signal *);
+
+ void execDIH_SWITCH_REPLICA_REQ(Signal *);
+ void execDIH_SWITCH_REPLICA_REF(Signal *);
+ void execDIH_SWITCH_REPLICA_CONF(Signal *);
+
+ void execSTOP_PERM_REQ(Signal *);
+ void execSTOP_PERM_REF(Signal *);
+ void execSTOP_PERM_CONF(Signal *);
+
+ void execSTOP_ME_REQ(Signal *);
+ void execSTOP_ME_REF(Signal *);
+ void execSTOP_ME_CONF(Signal *);
+
+ void execSIZEALT_REP(Signal *);
+ void execUNBLO_DICTCONF(Signal *);
+ void execCOPY_ACTIVECONF(Signal *);
+ void execTAB_COMMITREQ(Signal *);
+ void execNODE_FAILREP(Signal *);
+ void execCOPY_FRAGCONF(Signal *);
+ void execCOPY_FRAGREF(Signal *);
+ void execDIADDTABREQ(Signal *);
+ void execDIGETNODESREQ(Signal *);
+ void execDIRELEASEREQ(Signal *);
+ void execDISEIZEREQ(Signal *);
+ void execSTTOR(Signal *);
+ void execDI_FCOUNTREQ(Signal *);
+ void execDIGETPRIMREQ(Signal *);
+ void execGCP_SAVEREF(Signal *);
+ void execGCP_TCFINISHED(Signal *);
+ void execREAD_NODESCONF(Signal *);
+ void execNDB_STTOR(Signal *);
+ void execDICTSTARTCONF(Signal *);
+ void execNDB_STARTREQ(Signal *);
+ void execGETGCIREQ(Signal *);
+ void execDIH_RESTARTREQ(Signal *);
+ void execCNTR_CHANGEREP(Signal *);
+ void execSTART_RECCONF(Signal *);
+ void execSTART_FRAGCONF(Signal *);
+ void execADD_FRAGCONF(Signal *);
+ void execADD_FRAGREF(Signal *);
+ void execFSOPENCONF(Signal *);
+ void execFSOPENREF(Signal *);
+ void execFSCLOSECONF(Signal *);
+ void execFSCLOSEREF(Signal *);
+ void execFSREADCONF(Signal *);
+ void execFSREADREF(Signal *);
+ void execFSWRITECONF(Signal *);
+ void execFSWRITEREF(Signal *);
+ void execSET_VAR_REQ(Signal *);
+ void execCHECKNODEGROUPSREQ(Signal *);
+ void execSTART_INFOREQ(Signal*);
+ void execSTART_INFOREF(Signal*);
+ void execSTART_INFOCONF(Signal*);
+ void execWAIT_GCP_REQ(Signal* signal);
+ void execWAIT_GCP_REF(Signal* signal);
+ void execWAIT_GCP_CONF(Signal* signal);
+ void execUPDATE_TOREQ(Signal* signal);
+ void execUPDATE_TOCONF(Signal* signal);
+
+ void execPREP_DROP_TAB_REQ(Signal* signal);
+ void execWAIT_DROP_TAB_REF(Signal* signal);
+ void execWAIT_DROP_TAB_CONF(Signal* signal);
+ void execDROP_TAB_REQ(Signal* signal);
+
+ void execALTER_TAB_REQ(Signal* signal);
+
+ void execCREATE_FRAGMENTATION_REQ(Signal*);
+
+ void waitDropTabWritingToFile(Signal *, TabRecordPtr tabPtr);
+ void checkPrepDropTabComplete(Signal *, TabRecordPtr tabPtr);
+ void checkWaitDropTabFailedLqh(Signal *, Uint32 nodeId, Uint32 tableId);
+
+ // Statement blocks
+//------------------------------------
+// Methods that send signals
+//------------------------------------
+ void nullRoutine(Signal *, Uint32 nodeId);
+ void sendCOPY_GCIREQ(Signal *, Uint32 nodeId);
+ void sendDIH_SWITCH_REPLICA_REQ(Signal *, Uint32 nodeId);
+ void sendEMPTY_LCP_REQ(Signal *, Uint32 nodeId);
+ void sendEND_TOREQ(Signal *, Uint32 nodeId);
+ void sendGCP_COMMIT(Signal *, Uint32 nodeId);
+ void sendGCP_PREPARE(Signal *, Uint32 nodeId);
+ void sendGCP_SAVEREQ(Signal *, Uint32 nodeId);
+ void sendINCL_NODEREQ(Signal *, Uint32 nodeId);
+ void sendMASTER_GCPREQ(Signal *, Uint32 nodeId);
+ void sendMASTER_LCPREQ(Signal *, Uint32 nodeId);
+ void sendMASTER_LCPCONF(Signal * signal);
+ void sendSTART_RECREQ(Signal *, Uint32 nodeId);
+ void sendSTART_INFOREQ(Signal *, Uint32 nodeId);
+ void sendSTART_TOREQ(Signal *, Uint32 nodeId);
+ void sendSTOP_ME_REQ(Signal *, Uint32 nodeId);
+ void sendTC_CLOPSIZEREQ(Signal *, Uint32 nodeId);
+ void sendTCGETOPSIZEREQ(Signal *, Uint32 nodeId);
+ void sendUPDATE_TOREQ(Signal *, Uint32 nodeId);
+ void sendSTART_LCP_REQ(Signal *, Uint32 nodeId);
+
+ void sendLCP_FRAG_ORD(Signal*, NodeRecord::FragmentCheckpointInfo info);
+ void sendLastLCP_FRAG_ORD(Signal *);
+
+ void sendCopyTable(Signal *, CopyTableNode* ctn,
+ BlockReference ref, Uint32 reqinfo);
+ void sendCreateFragReq(Signal *,
+ Uint32 startGci,
+ Uint32 storedType,
+ Uint32 takeOverPtr);
+ void sendDihfragreq(Signal *,
+ TabRecordPtr regTabPtr,
+ Uint32 fragId);
+ void sendStartFragreq(Signal *,
+ TabRecordPtr regTabPtr,
+ Uint32 fragId);
+ void sendHOT_SPAREREP(Signal *);
+ void sendAddFragreq(Signal *,
+ TabRecordPtr regTabPtr,
+ Uint32 fragId,
+ Uint32 lcpNo,
+ Uint32 param);
+
+ void sendAddFragreq(Signal*, ConnectRecordPtr, TabRecordPtr, Uint32 fragId);
+ void addTable_closeConf(Signal* signal, Uint32 tabPtrI);
+ void resetReplicaSr(TabRecordPtr tabPtr);
+ void resetReplicaLcp(ReplicaRecord * replicaP, Uint32 stopGci);
+
+//------------------------------------
+// Methods for LCP functionality
+//------------------------------------
+ void checkKeepGci(Uint32 replicaStartIndex);
+ void checkLcpStart(Signal *, Uint32 lineNo);
+ void checkStartMoreLcp(Signal *, Uint32 nodeId);
+ bool reportLcpCompletion(const class LcpFragRep *);
+ void sendLCP_COMPLETE_REP(Signal *);
+
+//------------------------------------
+// Methods for Delete Table Files
+//------------------------------------
+ void startDeleteFile(Signal* signal, TabRecordPtr tabPtr);
+ void openTableFileForDelete(Signal* signal, Uint32 fileIndex);
+ void tableOpenLab(Signal* signal, FileRecordPtr regFilePtr);
+ void tableDeleteLab(Signal* signal, FileRecordPtr regFilePtr);
+
+//------------------------------------
+// File Record specific methods
+//------------------------------------
+ void closeFile(Signal *, FileRecordPtr regFilePtr);
+ void closeFileDelete(Signal *, FileRecordPtr regFilePtr);
+ void createFileRw(Signal *, FileRecordPtr regFilePtr);
+ void openFileRw(Signal *, FileRecordPtr regFilePtr);
+ void seizeFile(FileRecordPtr& regFilePtr);
+ void releaseFile(Uint32 fileIndex);
+
+//------------------------------------
+// Methods called when completing file
+// operation.
+//------------------------------------
+ void creatingGcpLab(Signal *, FileRecordPtr regFilePtr);
+ void openingGcpLab(Signal *, FileRecordPtr regFilePtr);
+ void openingTableLab(Signal *, FileRecordPtr regFilePtr);
+ void tableCreateLab(Signal *, FileRecordPtr regFilePtr);
+ void creatingGcpErrorLab(Signal *, FileRecordPtr regFilePtr);
+ void openingCopyGciErrorLab(Signal *, FileRecordPtr regFilePtr);
+ void creatingCopyGciErrorLab(Signal *, FileRecordPtr regFilePtr);
+ void openingGcpErrorLab(Signal *, FileRecordPtr regFilePtr);
+ void openingTableErrorLab(Signal *, FileRecordPtr regFilePtr);
+ void tableCreateErrorLab(Signal *, FileRecordPtr regFilePtr);
+ void closingGcpLab(Signal *, FileRecordPtr regFilePtr);
+ void closingGcpCrashLab(Signal *, FileRecordPtr regFilePtr);
+ void closingTableCrashLab(Signal *, FileRecordPtr regFilePtr);
+ void closingTableSrLab(Signal *, FileRecordPtr regFilePtr);
+ void tableCloseLab(Signal *, FileRecordPtr regFilePtr);
+ void tableCloseErrorLab(FileRecordPtr regFilePtr);
+ void readingGcpLab(Signal *, FileRecordPtr regFilePtr);
+ void readingTableLab(Signal *, FileRecordPtr regFilePtr);
+ void readingGcpErrorLab(Signal *, FileRecordPtr regFilePtr);
+ void readingTableErrorLab(Signal *, FileRecordPtr regFilePtr);
+ void writingCopyGciLab(Signal *, FileRecordPtr regFilePtr);
+ void writeInitGcpLab(Signal *, FileRecordPtr regFilePtr);
+ void tableWriteLab(Signal *, FileRecordPtr regFilePtr);
+ void writeInitGcpErrorLab(Signal *, FileRecordPtr regFilePtr);
+
+
+ void calculateHotSpare();
+ void checkEscalation();
+ void clearRestartInfoBits(Signal *);
+ void invalidateLcpInfoAfterSr();
+
+ bool isMaster();
+ bool isActiveMaster();
+
+ void emptyverificbuffer(Signal *, bool aContintueB);
+ Uint32 findHotSpare();
+ void handleGcpStateInMaster(Signal *, NodeRecordPtr failedNodeptr);
+ void initRestartInfo();
+ void initRestorableGciFiles();
+ void makeNodeGroups(Uint32 nodeArray[]);
+ void makePrnList(class ReadNodesConf * readNodes, Uint32 nodeArray[]);
+ void nodeResetStart();
+ void releaseTabPages(Uint32 tableId);
+ void replication(Uint32 noOfReplicas,
+ NodeGroupRecordPtr NGPtr,
+ FragmentstorePtr regFragptr);
+ void selectMasterCandidateAndSend(Signal *);
+ void setInitialActiveStatus();
+ void setLcpActiveStatusEnd();
+ void setLcpActiveStatusStart(Signal *);
+ void setNodeActiveStatus();
+ void setNodeGroups();
+ void setNodeInfo(Signal *);
+ void setNodeLcpActiveStatus();
+ void setNodeRestartInfoBits();
+ void startGcp(Signal *);
+
+ void readFragment(RWFragment* rf, FragmentstorePtr regFragptr);
+ Uint32 readPageWord(RWFragment* rf);
+ void readReplica(RWFragment* rf, ReplicaRecordPtr readReplicaPtr);
+ void readReplicas(RWFragment* rf, FragmentstorePtr regFragptr);
+ void readRestorableGci(Signal *, FileRecordPtr regFilePtr);
+ void readTabfile(Signal *, TabRecord* tab, FileRecordPtr regFilePtr);
+ void writeFragment(RWFragment* wf, FragmentstorePtr regFragptr);
+ void writePageWord(RWFragment* wf, Uint32 dataWord);
+ void writeReplicas(RWFragment* wf, Uint32 replicaStartIndex);
+ void writeRestorableGci(Signal *, FileRecordPtr regFilePtr);
+ void writeTabfile(Signal *, TabRecord* tab, FileRecordPtr regFilePtr);
+ void copyTabReq_complete(Signal* signal, TabRecordPtr tabPtr);
+
+ void gcpcommitreqLab(Signal *);
+ void gcpsavereqLab(Signal *);
+ void copyGciLab(Signal *, CopyGCIReq::CopyReason reason);
+ void storeNewLcpIdLab(Signal *);
+ void startLcpRoundLoopLab(Signal *, Uint32 startTableId, Uint32 startFragId);
+
+ void nodeFailCompletedCheckLab(Signal*, NodeRecordPtr failedNodePtr);
+
+ /**
+ *
+ */
+ void setLocalNodefailHandling(Signal*, Uint32 failedNodeId,
+ NodefailHandlingStep step);
+ void checkLocalNodefailComplete(Signal*, Uint32 failedNodeId,
+ NodefailHandlingStep step);
+
+ void ndbsttorry10Lab(Signal *, Uint32 _line);
+ void createMutexes(Signal* signal, Uint32 no);
+ void createMutex_done(Signal* signal, Uint32 no, Uint32 retVal);
+ void crashSystemAtGcpStop(Signal *);
+ void sendFirstDictfragsreq(Signal *, TabRecordPtr regTabPtr);
+ void addtabrefuseLab(Signal *, ConnectRecordPtr regConnectPtr, Uint32 errorCode);
+ void GCP_SAVEhandling(Signal *, Uint32 nodeId);
+ void packTableIntoPagesLab(Signal *, Uint32 tableId);
+ void readPagesIntoTableLab(Signal *, Uint32 tableId);
+ void readPagesIntoFragLab(Signal *, RWFragment* rf);
+ void readTabDescriptionLab(Signal *, Uint32 tableId);
+ void copyTableLab(Signal *, Uint32 tableId);
+ void breakCopyTableLab(Signal *,
+ TabRecordPtr regTabPtr,
+ Uint32 nodeId);
+ void checkAddfragCompletedLab(Signal *,
+ TabRecordPtr regTabPtr,
+ Uint32 fragId);
+ void completeRestartLab(Signal *);
+ void readTableFromPagesLab(Signal *, TabRecordPtr regTabPtr);
+ void srPhase2ReadTableLab(Signal *, TabRecordPtr regTabPtr);
+ void checkTcCounterLab(Signal *);
+ void calculateKeepGciLab(Signal *, Uint32 tableId, Uint32 fragId);
+ void tableUpdateLab(Signal *, TabRecordPtr regTabPtr);
+ void checkLcpCompletedLab(Signal *);
+ void initLcpLab(Signal *, Uint32 masterRef, Uint32 tableId);
+ void startGcpLab(Signal *, Uint32 aWaitTime);
+ void checkGcpStopLab(Signal *);
+ void MASTER_GCPhandling(Signal *, Uint32 failedNodeId);
+ void MASTER_LCPhandling(Signal *, Uint32 failedNodeId);
+ void rnfTableNotReadyLab(Signal *, TabRecordPtr regTabPtr, Uint32 removeNodeId);
+ void startLcpTakeOverLab(Signal *, Uint32 failedNodeId);
+
+ void startLcpMasterTakeOver(Signal *, Uint32 failedNodeId);
+ void startGcpMasterTakeOver(Signal *, Uint32 failedNodeId);
+ void checkGcpOutstanding(Signal*, Uint32 failedNodeId);
+
+ void checkEmptyLcpComplete(Signal *);
+ void lcpBlockedLab(Signal *);
+ void breakCheckTabCompletedLab(Signal *, TabRecordPtr regTabptr);
+ void readGciFileLab(Signal *);
+ void openingCopyGciSkipInitLab(Signal *, FileRecordPtr regFilePtr);
+ void startLcpRoundLab(Signal *);
+ void gcpBlockedLab(Signal *);
+ void initialStartCompletedLab(Signal *);
+ void allNodesLcpCompletedLab(Signal *);
+ void nodeRestartPh2Lab(Signal *);
+ void initGciFilesLab(Signal *);
+ void dictStartConfLab(Signal *);
+ void nodeDictStartConfLab(Signal *);
+ void ndbStartReqLab(Signal *, BlockReference ref);
+ void nodeRestartStartRecConfLab(Signal *);
+ void dihCopyCompletedLab(Signal *);
+ void copyTableNode(Signal *,
+ CopyTableNode* ctn,
+ NodeRecordPtr regNodePtr);
+ void startFragment(Signal *, Uint32 tableId, Uint32 fragId);
+ bool checkLcpAllTablesDoneInLqh();
+
+ void lcpStateAtNodeFailureLab(Signal *, Uint32 nodeId);
+ void copyNodeLab(Signal *, Uint32 tableId);
+ void copyGciReqLab(Signal *);
+ void allLab(Signal *,
+ ConnectRecordPtr regConnectPtr,
+ TabRecordPtr regTabPtr);
+ void tableCopyNodeLab(Signal *, TabRecordPtr regTabPtr);
+
+ void removeNodeFromTables(Signal *, Uint32 tableId, Uint32 nodeId);
+ void removeNodeFromTable(Signal *, Uint32 tableId, TabRecordPtr tabPtr);
+ void removeNodeFromTablesComplete(Signal* signal, Uint32 nodeId);
+
+ void packFragIntoPagesLab(Signal *, RWFragment* wf);
+ void startNextChkpt(Signal *);
+ void failedNodeLcpHandling(Signal*, NodeRecordPtr failedNodePtr);
+ void failedNodeSynchHandling(Signal *, NodeRecordPtr failedNodePtr);
+ void checkCopyTab(NodeRecordPtr failedNodePtr);
+
+ void initCommonData();
+ void initialiseRecordsLab(Signal *, Uint32 stepNo);
+
+ void findReplica(ReplicaRecordPtr& regReplicaPtr,
+ Fragmentstore* fragPtrP, Uint32 nodeId);
+//------------------------------------
+// Node failure handling methods
+//------------------------------------
+ void startRemoveFailedNode(Signal *, NodeRecordPtr failedNodePtr);
+ void handleGcpTakeOver(Signal *, NodeRecordPtr failedNodePtr);
+ void handleLcpTakeOver(Signal *, NodeRecordPtr failedNodePtr);
+ void handleNewMaster(Signal *, NodeRecordPtr failedNodePtr);
+ void checkTakeOverInMasterAllNodeFailure(Signal*, NodeRecordPtr failedNode);
+ void checkTakeOverInMasterCopyNodeFailure(Signal*, Uint32 failedNodeId);
+ void checkTakeOverInMasterStartNodeFailure(Signal*, Uint32 takeOverPtr);
+ void checkTakeOverInNonMasterStartNodeFailure(Signal*, Uint32 takeOverPtr);
+ void handleLcpMasterTakeOver(Signal *, Uint32 nodeId);
+
+//------------------------------------
+// Replica record specific methods
+//------------------------------------
+ Uint32 findLogInterval(ConstPtr<ReplicaRecord> regReplicaPtr,
+ Uint32 startGci);
+ void findMinGci(ReplicaRecordPtr fmgReplicaPtr,
+ Uint32& keeGci,
+ Uint32& oldestRestorableGci);
+ bool findStartGci(ConstPtr<ReplicaRecord> fstReplicaPtr,
+ Uint32 tfstStopGci,
+ Uint32& tfstStartGci,
+ Uint32& tfstLcp);
+ void newCrashedReplica(Uint32 nodeId, ReplicaRecordPtr ncrReplicaPtr);
+ void packCrashedReplicas(ReplicaRecordPtr pcrReplicaPtr);
+ void releaseReplicas(Uint32 replicaPtr);
+ void removeOldCrashedReplicas(ReplicaRecordPtr rocReplicaPtr);
+ void removeTooNewCrashedReplicas(ReplicaRecordPtr rtnReplicaPtr);
+ void seizeReplicaRec(ReplicaRecordPtr& replicaPtr);
+
+//------------------------------------
+// Methods operating on a fragment and
+// its connected replicas and nodes.
+//------------------------------------
+ void allocStoredReplica(FragmentstorePtr regFragptr,
+ ReplicaRecordPtr& newReplicaPtr,
+ Uint32 nodeId);
+ Uint32 extractNodeInfo(const Fragmentstore * fragPtr, Uint32 nodes[]);
+ bool findBestLogNode(CreateReplicaRecord* createReplica,
+ FragmentstorePtr regFragptr,
+ Uint32 startGci,
+ Uint32 stopGci,
+ Uint32 logNode,
+ Uint32& fblStopGci);
+ bool findLogNodes(CreateReplicaRecord* createReplica,
+ FragmentstorePtr regFragptr,
+ Uint32 startGci,
+ Uint32 stopGci);
+ void findToReplica(TakeOverRecord* regTakeOver,
+ Uint32 replicaType,
+ FragmentstorePtr regFragptr,
+ ReplicaRecordPtr& ftrReplicaPtr);
+ void initFragstore(FragmentstorePtr regFragptr);
+ void insertBackup(FragmentstorePtr regFragptr, Uint32 nodeId);
+ void insertfraginfo(FragmentstorePtr regFragptr,
+ Uint32 noOfBackups,
+ Uint32* nodeArray);
+ void linkOldStoredReplica(FragmentstorePtr regFragptr,
+ ReplicaRecordPtr replicaPtr);
+ void linkStoredReplica(FragmentstorePtr regFragptr,
+ ReplicaRecordPtr replicaPtr);
+ void prepareReplicas(FragmentstorePtr regFragptr);
+ void removeNodeFromStored(Uint32 nodeId,
+ FragmentstorePtr regFragptr,
+ ReplicaRecordPtr replicaPtr);
+ void removeOldStoredReplica(FragmentstorePtr regFragptr,
+ ReplicaRecordPtr replicaPtr);
+ void removeStoredReplica(FragmentstorePtr regFragptr,
+ ReplicaRecordPtr replicaPtr);
+ void searchStoredReplicas(FragmentstorePtr regFragptr);
+ void updateNodeInfo(FragmentstorePtr regFragptr);
+
+//------------------------------------
+// Fragment allocation, deallocation and
+// find methods
+//------------------------------------
+ void allocFragments(Uint32 noOfFragments, TabRecordPtr regTabPtr);
+ void releaseFragments(TabRecordPtr regTabPtr);
+ void getFragstore(TabRecord *, Uint32 fragNo, FragmentstorePtr & ptr);
+ void initialiseFragstore();
+
+//------------------------------------
+// Page Record specific methods
+//------------------------------------
+ void allocpage(PageRecordPtr& regPagePtr);
+ void releasePage(Uint32 pageIndex);
+
+//------------------------------------
+// Table Record specific methods
+//------------------------------------
+ void initTable(TabRecordPtr regTabPtr);
+ void initTableFile(TabRecordPtr regTabPtr);
+ void releaseTable(TabRecordPtr tabPtr);
+ Uint32 findTakeOver(Uint32 failedNodeId);
+ void handleTakeOverMaster(Signal *, Uint32 takeOverPtr);
+ void handleTakeOverNewMaster(Signal *, Uint32 takeOverPtr);
+
+//------------------------------------
+// TakeOver Record specific methods
+//------------------------------------
+ void initTakeOver(TakeOverRecordPtr regTakeOverptr);
+ void seizeTakeOver(TakeOverRecordPtr& regTakeOverptr);
+ void allocateTakeOver(TakeOverRecordPtr& regTakeOverptr);
+ void releaseTakeOver(Uint32 takeOverPtr);
+ bool anyActiveTakeOver();
+ void checkToCopy();
+ void checkToCopyCompleted(Signal *);
+ bool checkToInterrupted(TakeOverRecordPtr& regTakeOverptr);
+ Uint32 getStartNode(Uint32 takeOverPtr);
+
+//------------------------------------
+// Methods for take over functionality
+//------------------------------------
+ void changeNodeGroups(Uint32 startNode, Uint32 nodeTakenOver);
+ void endTakeOver(Uint32 takeOverPtr);
+ void initStartTakeOver(const class StartToReq *,
+ TakeOverRecordPtr regTakeOverPtr);
+
+ void nodeRestartTakeOver(Signal *, Uint32 startNodeId);
+ void systemRestartTakeOverLab(Signal *);
+ void startTakeOver(Signal *,
+ Uint32 takeOverPtr,
+ Uint32 startNode,
+ Uint32 toNode);
+ void sendStartTo(Signal *, Uint32 takeOverPtr);
+ void startNextCopyFragment(Signal *, Uint32 takeOverPtr);
+ void toCopyFragLab(Signal *, Uint32 takeOverPtr);
+ void startHsAddFragConfLab(Signal *);
+ void prepareSendCreateFragReq(Signal *, Uint32 takeOverPtr);
+ void sendUpdateTo(Signal *, Uint32 takeOverPtr, Uint32 updateState);
+ void toCopyCompletedLab(Signal *, TakeOverRecordPtr regTakeOverptr);
+ void takeOverCompleted(Uint32 aNodeId);
+ void sendEndTo(Signal *, Uint32 takeOverPtr);
+
+//------------------------------------
+// Node Record specific methods
+//------------------------------------
+ void checkStartTakeOver(Signal *);
+ void insertAlive(NodeRecordPtr newNodePtr);
+ void insertDeadNode(NodeRecordPtr removeNodePtr);
+ void removeAlive(NodeRecordPtr removeNodePtr);
+ void removeDeadNode(NodeRecordPtr removeNodePtr);
+
+ NodeRecord::NodeStatus getNodeStatus(Uint32 nodeId);
+ void setNodeStatus(Uint32 nodeId, NodeRecord::NodeStatus);
+ Sysfile::ActiveStatus getNodeActiveStatus(Uint32 nodeId);
+ void setNodeActiveStatus(Uint32 nodeId, Sysfile::ActiveStatus newStatus);
+ void setNodeLcpActiveStatus(Uint32 nodeId, bool newState);
+ bool getNodeLcpActiveStatus(Uint32 nodeId);
+ bool getAllowNodeStart(Uint32 nodeId);
+ void setAllowNodeStart(Uint32 nodeId, bool newState);
+ bool getNodeCopyCompleted(Uint32 nodeId);
+ void setNodeCopyCompleted(Uint32 nodeId, bool newState);
+ void initNodeState(NodeRecordPtr regNodePtr);
+ bool checkNodeAlive(Uint32 nodeId);
+
+ // Initialisation
+ void initData();
+ void initRecords();
+
+ // Variables to support record structures and their free lists
+
+ ApiConnectRecord *apiConnectRecord;
+ Uint32 capiConnectFileSize;
+
+ ConnectRecord *connectRecord;
+ Uint32 cfirstconnect;
+ Uint32 cconnectFileSize;
+
+ CreateReplicaRecord *createReplicaRecord;
+ Uint32 cnoOfCreateReplicas;
+
+ FileRecord *fileRecord;
+ Uint32 cfirstfreeFile;
+ Uint32 cfileFileSize;
+
+ Fragmentstore *fragmentstore;
+ Uint32 cfirstfragstore;
+ Uint32 cfragstoreFileSize;
+
+ Uint32 c_nextNodeGroup;
+ NodeGroupRecord *nodeGroupRecord;
+
+ NodeRecord *nodeRecord;
+
+ PageRecord *pageRecord;
+ Uint32 cfirstfreepage;
+ Uint32 cpageFileSize;
+
+ ReplicaRecord *replicaRecord;
+ Uint32 cfirstfreeReplica;
+ Uint32 cnoFreeReplicaRec;
+ Uint32 creplicaFileSize;
+
+ TabRecord *tabRecord;
+ Uint32 ctabFileSize;
+
+ TakeOverRecord *takeOverRecord;
+ Uint32 cfirstfreeTakeOver;
+
+ /*
+ 2.4 C O M M O N S T O R E D V A R I A B L E S
+ ----------------------------------------------------
+ */
+ Uint32 cfirstVerifyQueue;
+ Uint32 clastVerifyQueue;
+ Uint32 cverifyQueueCounter;
+
+ /*------------------------------------------------------------------------*/
+ /* THIS VARIABLE KEEPS THE REFERENCES TO FILE RECORDS THAT DESCRIBE */
+ /* THE TWO FILES THAT ARE USED TO STORE THE VARIABLE CRESTART_INFO */
+ /* ON DISK. */
+ /*------------------------------------------------------------------------*/
+ Uint32 crestartInfoFile[2];
+ /*------------------------------------------------------------------------*/
+ /* THIS VARIABLE KEEPS TRACK OF THE STATUS OF A GLOBAL CHECKPOINT */
+ /* PARTICIPANT. THIS IS NEEDED TO HANDLE A NODE FAILURE. WHEN A NODE*/
+ /* FAILURE OCCURS IT IS EASY THAT THE PROTOCOL STOPS IF NO ACTION IS*/
+ /* TAKEN TO PREVENT THIS. THIS VARIABLE ENSURES SUCH ACTION CAN BE */
+ /* TAKEN. */
+ /*------------------------------------------------------------------------*/
+ enum GcpParticipantState {
+ GCP_PARTICIPANT_READY = 0,
+ GCP_PARTICIPANT_PREPARE_RECEIVED = 1,
+ GCP_PARTICIPANT_COMMIT_RECEIVED = 2,
+ GCP_PARTICIPANT_TC_FINISHED = 3,
+ GCP_PARTICIPANT_COPY_GCI_RECEIVED = 4
+ };
+ GcpParticipantState cgcpParticipantState;
+ /*------------------------------------------------------------------------*/
+ /* THESE VARIABLES ARE USED TO CONTROL THAT GCP PROCESSING DO NOT */
+ /*STOP FOR SOME REASON. */
+ /*------------------------------------------------------------------------*/
+ enum GcpStatus {
+ GCP_READY = 0,
+ GCP_PREPARE_SENT = 1,
+ GCP_COMMIT_SENT = 2,
+ GCP_NODE_FINISHED = 3,
+ GCP_SAVE_LQH_FINISHED = 4
+ };
+ GcpStatus cgcpStatus;
+ Uint32 cgcpStartCounter;
+ Uint32 coldGcpStatus;
+ Uint32 coldGcpId;
+ /*------------------------------------------------------------------------*/
+ /* THIS VARIABLE KEEPS TRACK OF THE STATE OF THIS NODE AS MASTER. */
+ /*------------------------------------------------------------------------*/
+ enum MasterState {
+ MASTER_IDLE = 0,
+ MASTER_ACTIVE = 1,
+ MASTER_TAKE_OVER_GCP = 2
+ };
+ MasterState cmasterState;
+ Uint16 cmasterTakeOverNode;
+ /* NODE IS NOT MASTER */
+ /* NODE IS ACTIVE AS MASTER */
+ /* NODE IS TAKING OVER AS MASTER */
+
+ struct CopyGCIMaster {
+ CopyGCIMaster(){ m_copyReason = m_waiting = CopyGCIReq::IDLE;}
+ /*------------------------------------------------------------------------*/
+ /* THIS STATE VARIABLE IS USED TO INDICATE IF COPYING OF RESTART */
+ /* INFO WAS STARTED BY A LOCAL CHECKPOINT OR AS PART OF A SYSTEM */
+ /* RESTART. */
+ /*------------------------------------------------------------------------*/
+ CopyGCIReq::CopyReason m_copyReason;
+
+ /*------------------------------------------------------------------------*/
+ /* COPYING RESTART INFO CAN BE STARTED BY LOCAL CHECKPOINTS AND BY */
+ /* GLOBAL CHECKPOINTS. WE CAN HOWEVER ONLY HANDLE ONE SUCH COPY AT */
+ /* THE TIME. THUS WE HAVE TO KEEP WAIT INFORMATION IN THIS VARIABLE.*/
+ /*------------------------------------------------------------------------*/
+ CopyGCIReq::CopyReason m_waiting;
+ } c_copyGCIMaster;
+
+ struct CopyGCISlave {
+ CopyGCISlave(){ m_copyReason = CopyGCIReq::IDLE; m_expectedNextWord = 0;}
+ /*------------------------------------------------------------------------*/
+ /* THIS STATE VARIABLE IS USED TO INDICATE IF COPYING OF RESTART */
+ /* INFO WAS STARTED BY A LOCAL CHECKPOINT OR AS PART OF A SYSTEM */
+ /* RESTART. THIS VARIABLE IS USED BY THE NODE THAT RECEIVES */
+ /* COPY_GCI_REQ. */
+ /*------------------------------------------------------------------------*/
+ Uint32 m_senderData;
+ BlockReference m_senderRef;
+ CopyGCIReq::CopyReason m_copyReason;
+
+ Uint32 m_expectedNextWord;
+ } c_copyGCISlave;
+
+ /*------------------------------------------------------------------------*/
+ /* THIS VARIABLE IS USED TO KEEP TRACK OF THE STATE OF LOCAL */
+ /* CHECKPOINTS. */
+ /*------------------------------------------------------------------------*/
+public:
+ enum LcpStatus {
+ LCP_STATUS_IDLE = 0,
+ LCP_TCGET = 1, // Only master
+ LCP_STATUS_ACTIVE = 2,
+ LCP_CALCULATE_KEEP_GCI = 4, // Only master
+ LCP_COPY_GCI = 5,
+ LCP_INIT_TABLES = 6,
+ LCP_TC_CLOPSIZE = 7, // Only master
+ LCP_START_LCP_ROUND = 8,
+ LCP_TAB_COMPLETED = 9,
+ LCP_TAB_SAVED = 10
+ };
+private:
+
+ struct LcpState {
+ LcpStatus lcpStatus;
+ Uint32 lcpStatusUpdatedPlace;
+
+ void setLcpStatus(LcpStatus status, Uint32 line){
+ lcpStatus = status;
+ lcpStatusUpdatedPlace = line;
+ }
+
+ Uint32 lcpStart;
+ Uint32 lcpStartGcp;
+ Uint32 keepGci; /* USED TO CALCULATE THE GCI TO KEEP AFTER A LCP */
+ Uint32 oldestRestorableGci;
+
+ struct CurrentFragment {
+ Uint32 tableId;
+ Uint32 fragmentId;
+ } currentFragment;
+
+ Uint32 noOfLcpFragRepOutstanding;
+
+ /*------------------------------------------------------------------------*/
+ /* USED TO ENSURE THAT LCP'S ARE EXECUTED WITH CERTAIN TIMEINTERVALS*/
+ /* EVEN WHEN SYSTEM IS NOT DOING ANYTHING. */
+ /*------------------------------------------------------------------------*/
+ Uint32 ctimer;
+ Uint32 ctcCounter;
+ Uint32 clcpDelay; /* MAX. 2^(CLCP_DELAY - 2) SEC BETWEEN LCP'S */
+
+ /*------------------------------------------------------------------------*/
+ /* THIS STATE IS USED TO TELL IF THE FIRST LCP AFTER START/RESTART */
+ /* HAS BEEN RUN. AFTER A NODE RESTART THE NODE DOES NOT ENTER */
+ /* STARTED STATE BEFORE THIS IS DONE. */
+ /*------------------------------------------------------------------------*/
+ bool immediateLcpStart;
+ bool m_LCP_COMPLETE_REP_From_Master_Received;
+ SignalCounter m_LCP_COMPLETE_REP_Counter_DIH;
+ SignalCounter m_LCP_COMPLETE_REP_Counter_LQH;
+ SignalCounter m_LAST_LCP_FRAG_ORD;
+ NdbNodeBitmask m_participatingLQH;
+ NdbNodeBitmask m_participatingDIH;
+
+ Uint32 m_masterLcpDihRef;
+ bool m_MASTER_LCPREQ_Received;
+ Uint32 m_MASTER_LCPREQ_FailedNodeId;
+ } c_lcpState;
+
+ /*------------------------------------------------------------------------*/
+ /* THIS VARIABLE KEEPS TRACK OF HOW MANY TABLES ARE ACTIVATED WHEN */
+ /* STARTING A LOCAL CHECKPOINT WE SHOULD AVOID STARTING A CHECKPOINT*/
+ /* WHEN NO TABLES ARE ACTIVATED. */
+ /*------------------------------------------------------------------------*/
+ Uint32 cnoOfActiveTables;
+ Uint32 cgcpDelay; /* Delay between global checkpoints */
+
+ BlockReference cdictblockref; /* DICTIONARY BLOCK REFERENCE */
+ Uint32 cfailurenr; /* EVERY TIME WHEN A NODE FAILURE IS REPORTED
+ THIS NUMBER IS INCREMENTED. AT THE START OF
+ THE SYSTEM THIS NUMBER MUST BE INITIATED TO
+ ZERO */
+ bool cgckptflag; /* A FLAG WHICH IS SET WHILE A NEW GLOBAL CHECK
+ POINT IS BEING CREATED. NO VERIFICATION IS ALLOWED
+ IF THE FLAG IS SET*/
+ Uint32 cgcpOrderBlocked;
+ BlockReference clocallqhblockref;
+ BlockReference clocaltcblockref;
+ BlockReference cmasterdihref;
+ Uint16 cownNodeId;
+ Uint32 cnewgcp;
+ BlockReference cndbStartReqBlockref;
+ BlockReference cntrlblockref;
+ Uint32 cgcpSameCounter;
+ Uint32 coldgcp;
+ Uint32 con_lineNodes;
+ Uint32 creceivedfrag;
+ Uint32 cremainingfrags;
+ Uint32 cstarttype;
+ Uint32 csystemnodes;
+ Uint32 currentgcp;
+
+ enum GcpMasterTakeOverState {
+ GMTOS_IDLE = 0,
+ GMTOS_INITIAL = 1,
+ ALL_READY = 2,
+ ALL_PREPARED = 3,
+ COMMIT_STARTED_NOT_COMPLETED = 4,
+ COMMIT_COMPLETED = 5,
+ PREPARE_STARTED_NOT_COMMITTED = 6,
+ SAVE_STARTED_NOT_COMPLETED = 7
+ };
+ GcpMasterTakeOverState cgcpMasterTakeOverState;
+
+public:
+ enum LcpMasterTakeOverState {
+ LMTOS_IDLE = 0,
+ LMTOS_WAIT_EMPTY_LCP = 1, // Currently doing empty LCP
+ LMTOS_WAIT_LCP_FRAG_REP = 2,// Currently waiting for outst. LCP_FRAG_REP
+ LMTOS_INITIAL = 3,
+ LMTOS_ALL_IDLE = 4,
+ LMTOS_ALL_ACTIVE = 5,
+ LMTOS_LCP_CONCLUDING = 6,
+ LMTOS_COPY_ONGOING = 7
+ };
+private:
+ class MasterTakeOverState {
+ public:
+ void set(LcpMasterTakeOverState s, Uint32 line) {
+ state = s; updatePlace = line;
+ }
+
+ LcpMasterTakeOverState state;
+ Uint32 updatePlace;
+
+ Uint32 minTableId;
+ Uint32 minFragId;
+ Uint32 failedNodeId;
+ } c_lcpMasterTakeOverState;
+
+ Uint16 cmasterNodeId;
+ Uint8 cnoHotSpare;
+
+ struct NodeStartMasterRecord {
+ Uint32 startNode;
+ Uint32 wait;
+ Uint32 failNr;
+ Uint32 ndbVersion;
+ bool activeState;
+ bool blockLcp;
+ bool blockGcp;
+ Uint32 startInfoErrorCode;
+ Uint32 m_outstandingGsn;
+ };
+ NodeStartMasterRecord c_nodeStartMaster;
+
+ struct NodeStartSlaveRecord {
+ NodeStartSlaveRecord() { nodeId = 0;}
+
+ Uint32 nodeId;
+ };
+ NodeStartSlaveRecord c_nodeStartSlave;
+
+ Uint32 cfirstAliveNode;
+ Uint32 cfirstDeadNode;
+ Uint32 cstartPhase;
+ Uint32 cnoReplicas;
+
+ Uint32 c_startToLock;
+ Uint32 c_endToLock;
+ Uint32 c_createFragmentLock;
+ Uint32 c_updateToLock;
+
+ bool cwaitLcpSr;
+ Uint32 cnoOfNodeGroups;
+ bool cstartGcpNow;
+
+ Uint32 crestartGci; /* VALUE OF GCI WHEN SYSTEM RESTARTED OR STARTED */
+ Uint32 cminHotSpareNodes;
+
+ /**
+ * Counter variables keeping track of the number of outstanding signals
+ * for particular signals in various protocols.
+ */
+ SignalCounter c_COPY_GCIREQ_Counter;
+ SignalCounter c_COPY_TABREQ_Counter;
+ SignalCounter c_CREATE_FRAGREQ_Counter;
+ SignalCounter c_DIH_SWITCH_REPLICA_REQ_Counter;
+ SignalCounter c_EMPTY_LCP_REQ_Counter;
+ SignalCounter c_END_TOREQ_Counter;
+ SignalCounter c_GCP_COMMIT_Counter;
+ SignalCounter c_GCP_PREPARE_Counter;
+ SignalCounter c_GCP_SAVEREQ_Counter;
+ SignalCounter c_INCL_NODEREQ_Counter;
+ SignalCounter c_MASTER_GCPREQ_Counter;
+ SignalCounter c_MASTER_LCPREQ_Counter;
+ SignalCounter c_START_INFOREQ_Counter;
+ SignalCounter c_START_RECREQ_Counter;
+ SignalCounter c_START_TOREQ_Counter;
+ SignalCounter c_STOP_ME_REQ_Counter;
+ SignalCounter c_TC_CLOPSIZEREQ_Counter;
+ SignalCounter c_TCGETOPSIZEREQ_Counter;
+ SignalCounter c_UPDATE_TOREQ_Counter;
+ SignalCounter c_START_LCP_REQ_Counter;
+
+ bool c_blockCommit;
+ Uint32 c_blockCommitNo;
+
+ bool getBlockCommit() const {
+ return c_blockCommit == true || cgckptflag == true;
+ }
+
+ /**
+ * SwitchReplicaRecord - Should only be used by master
+ */
+ struct SwitchReplicaRecord {
+ void clear(){}
+
+ Uint32 nodeId;
+ Uint32 tableId;
+ Uint32 fragNo;
+ };
+ SwitchReplicaRecord c_switchReplicas;
+
+ struct StopPermProxyRecord {
+ StopPermProxyRecord() { clientRef = 0; }
+
+ Uint32 clientData;
+ BlockReference clientRef;
+ BlockReference masterRef;
+ };
+
+ struct StopPermMasterRecord {
+ StopPermMasterRecord() { clientRef = 0;}
+
+ Uint32 returnValue;
+
+ Uint32 clientData;
+ BlockReference clientRef;
+ };
+
+ StopPermProxyRecord c_stopPermProxy;
+ StopPermMasterRecord c_stopPermMaster;
+
+ void checkStopPermProxy(Signal*, NodeId failedNodeId);
+ void checkStopPermMaster(Signal*, NodeRecordPtr failedNodePtr);
+
+ void switchReplica(Signal*,
+ Uint32 nodeId,
+ Uint32 tableId,
+ Uint32 fragNo);
+
+ void switchReplicaReply(Signal*, NodeId nodeId);
+
+ /**
+ * Wait GCP (proxy)
+ */
+ struct WaitGCPProxyRecord {
+ WaitGCPProxyRecord() { clientRef = 0;}
+
+ Uint32 clientData;
+ BlockReference clientRef;
+ BlockReference masterRef;
+
+ union { Uint32 nextPool; Uint32 nextList; };
+ Uint32 prevList;
+ };
+ typedef Ptr<WaitGCPProxyRecord> WaitGCPProxyPtr;
+
+ /**
+ * Wait GCP (master)
+ */
+ struct WaitGCPMasterRecord {
+ WaitGCPMasterRecord() { clientRef = 0;}
+ Uint32 clientData;
+ BlockReference clientRef;
+
+ union { Uint32 nextPool; Uint32 nextList; };
+ Uint32 prevList;
+ };
+ typedef Ptr<WaitGCPMasterRecord> WaitGCPMasterPtr;
+
+ /**
+ * Pool/list of WaitGCPProxyRecord record
+ */
+ ArrayPool<WaitGCPProxyRecord> waitGCPProxyPool;
+ ArrayList<WaitGCPProxyRecord> c_waitGCPProxyList;
+
+ /**
+ * Pool/list of WaitGCPMasterRecord record
+ */
+ ArrayPool<WaitGCPMasterRecord> waitGCPMasterPool;
+ ArrayList<WaitGCPMasterRecord> c_waitGCPMasterList;
+
+ void checkWaitGCPProxy(Signal*, NodeId failedNodeId);
+ void checkWaitGCPMaster(Signal*, NodeId failedNodeId);
+ void emptyWaitGCPMasterQueue(Signal*);
+
+ /**
+ * Stop me
+ */
+ struct StopMeRecord {
+ StopMeRecord() { clientRef = 0;}
+
+ BlockReference clientRef;
+ Uint32 clientData;
+ };
+ StopMeRecord c_stopMe;
+
+ void checkStopMe(Signal *, NodeRecordPtr failedNodePtr);
+
+#define DIH_CDATA_SIZE 128
+ /**
+ * This variable must be atleast the size of Sysfile::SYSFILE_SIZE32
+ */
+ Uint32 cdata[DIH_CDATA_SIZE]; /* TEMPORARY ARRAY VARIABLE */
+
+ /**
+ * Sys file data
+ */
+ Uint32 sysfileData[DIH_CDATA_SIZE];
+ Uint32 sysfileDataToFile[DIH_CDATA_SIZE];
+
+ /**
+ * When a node comes up without filesystem
+ * we have to clear all LCP for that node
+ */
+ void invalidateNodeLCP(Signal *, Uint32 nodeId, Uint32 tableId);
+ void invalidateNodeLCP(Signal *, Uint32 nodeId, TabRecordPtr);
+
+ /**
+ * Reply from nodeId
+ */
+ void startInfoReply(Signal *, Uint32 nodeId);
+};
+
+#if (DIH_CDATA_SIZE < _SYSFILE_SIZE32)
+#error "cdata is to small compared to Sysfile size"
+#endif
+
+#endif
+
diff --git a/ndb/src/kernel/blocks/dbdih/DbdihInit.cpp b/ndb/src/kernel/blocks/dbdih/DbdihInit.cpp
new file mode 100644
index 00000000000..b115cc0297a
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdih/DbdihInit.cpp
@@ -0,0 +1,319 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBDIH_C
+#include "Dbdih.hpp"
+#include <ndb_limits.h>
+#include <new>
+
+#define DEBUG(x) { ndbout << "DIH::" << x << endl; }
+
+void Dbdih::initData()
+{
+ cpageFileSize = ZPAGEREC;
+
+ apiConnectRecord = 0;
+ connectRecord = 0;
+ fileRecord = 0;
+ fragmentstore = 0;
+ pageRecord = 0;
+ replicaRecord = 0;
+ tabRecord = 0;
+ takeOverRecord = 0;
+ createReplicaRecord = 0;
+ nodeGroupRecord = 0;
+ nodeRecord = 0;
+ c_nextNodeGroup = 0;
+
+ // Records with constant sizes
+ createReplicaRecord = (CreateReplicaRecord*)
+ allocRecord("CreateReplicaRecord", sizeof(CreateReplicaRecord),
+ ZCREATE_REPLICA_FILE_SIZE);
+
+ nodeGroupRecord = (NodeGroupRecord*)
+ allocRecord("NodeGroupRecord", sizeof(NodeGroupRecord), MAX_NDB_NODES);
+
+ nodeRecord = (NodeRecord*)
+ allocRecord("NodeRecord", sizeof(NodeRecord), MAX_NDB_NODES);
+
+ for(Uint32 i = 0; i<MAX_NDB_NODES; i++){
+ new (&nodeRecord[i]) NodeRecord();
+ }
+
+ takeOverRecord = (TakeOverRecord*)allocRecord("TakeOverRecord",
+ sizeof(TakeOverRecord),
+ MAX_NDB_NODES);
+ for(Uint32 i = 0; i<MAX_NDB_NODES; i++)
+ new (&takeOverRecord[i]) TakeOverRecord();
+
+ for(Uint32 i = 0; i<MAX_NDB_NODES; i++)
+ new (&takeOverRecord[i]) TakeOverRecord();
+
+ waitGCPProxyPool.setSize(ZPROXY_FILE_SIZE);
+ waitGCPMasterPool.setSize(ZPROXY_MASTER_FILE_SIZE);
+
+ cgcpOrderBlocked = 0;
+ c_lcpState.ctcCounter = 0;
+ cwaitLcpSr = false;
+ c_blockCommit = false;
+ c_blockCommitNo = 1;
+}//Dbdih::initData()
+
+void Dbdih::initRecords()
+{
+ // Records with dynamic sizes
+ apiConnectRecord = (ApiConnectRecord*)
+ allocRecord("ApiConnectRecord",
+ sizeof(ApiConnectRecord),
+ capiConnectFileSize);
+
+ connectRecord = (ConnectRecord*)allocRecord("ConnectRecord",
+ sizeof(ConnectRecord),
+ cconnectFileSize);
+
+ fileRecord = (FileRecord*)allocRecord("FileRecord",
+ sizeof(FileRecord),
+ cfileFileSize);
+
+ fragmentstore = (Fragmentstore*)allocRecord("Fragmentstore",
+ sizeof(Fragmentstore),
+ cfragstoreFileSize);
+
+ pageRecord = (PageRecord*)allocRecord("PageRecord",
+ sizeof(PageRecord),
+ cpageFileSize);
+
+ replicaRecord = (ReplicaRecord*)allocRecord("ReplicaRecord",
+ sizeof(ReplicaRecord),
+ creplicaFileSize);
+
+ tabRecord = (TabRecord*)allocRecord("TabRecord",
+ sizeof(TabRecord),
+ ctabFileSize);
+
+ // Initialize BAT for interface to file system
+ NewVARIABLE* bat = allocateBat(22);
+ bat[1].WA = &pageRecord->word[0];
+ bat[1].nrr = cpageFileSize;
+ bat[1].ClusterSize = sizeof(PageRecord);
+ bat[1].bits.q = 11;
+ bat[1].bits.v = 5;
+ bat[20].WA = &sysfileData[0];
+ bat[20].nrr = 1;
+ bat[20].ClusterSize = sizeof(sysfileData);
+ bat[20].bits.q = 7;
+ bat[20].bits.v = 5;
+ bat[21].WA = &sysfileDataToFile[0];
+ bat[21].nrr = 1;
+ bat[21].ClusterSize = sizeof(sysfileDataToFile);
+ bat[21].bits.q = 7;
+ bat[21].bits.v = 5;
+}//Dbdih::initRecords()
+
+Dbdih::Dbdih(const class Configuration & config):
+ SimulatedBlock(DBDIH, config),
+ c_waitGCPProxyList(waitGCPProxyPool),
+ c_waitGCPMasterList(waitGCPMasterPool)
+{
+ BLOCK_CONSTRUCTOR(Dbdih);
+
+ addRecSignal(GSN_DUMP_STATE_ORD, &Dbdih::execDUMP_STATE_ORD);
+ addRecSignal(GSN_NDB_TAMPER, &Dbdih::execNDB_TAMPER, true);
+ addRecSignal(GSN_DEBUG_SIG, &Dbdih::execDEBUG_SIG);
+ addRecSignal(GSN_MASTER_GCPREQ, &Dbdih::execMASTER_GCPREQ);
+ addRecSignal(GSN_MASTER_GCPREF, &Dbdih::execMASTER_GCPREF);
+ addRecSignal(GSN_MASTER_GCPCONF, &Dbdih::execMASTER_GCPCONF);
+ addRecSignal(GSN_EMPTY_LCP_CONF, &Dbdih::execEMPTY_LCP_CONF);
+ addRecSignal(GSN_MASTER_LCPREQ, &Dbdih::execMASTER_LCPREQ);
+ addRecSignal(GSN_MASTER_LCPREF, &Dbdih::execMASTER_LCPREF);
+ addRecSignal(GSN_MASTER_LCPCONF, &Dbdih::execMASTER_LCPCONF);
+ addRecSignal(GSN_NF_COMPLETEREP, &Dbdih::execNF_COMPLETEREP);
+ addRecSignal(GSN_START_PERMREQ, &Dbdih::execSTART_PERMREQ);
+ addRecSignal(GSN_START_PERMCONF, &Dbdih::execSTART_PERMCONF);
+ addRecSignal(GSN_START_PERMREF, &Dbdih::execSTART_PERMREF);
+ addRecSignal(GSN_INCL_NODEREQ, &Dbdih::execINCL_NODEREQ);
+ addRecSignal(GSN_INCL_NODECONF, &Dbdih::execINCL_NODECONF);
+ addRecSignal(GSN_END_TOREQ, &Dbdih::execEND_TOREQ);
+ addRecSignal(GSN_END_TOCONF, &Dbdih::execEND_TOCONF);
+ addRecSignal(GSN_START_TOREQ, &Dbdih::execSTART_TOREQ);
+ addRecSignal(GSN_START_TOCONF, &Dbdih::execSTART_TOCONF);
+ addRecSignal(GSN_START_MEREQ, &Dbdih::execSTART_MEREQ);
+ addRecSignal(GSN_START_MECONF, &Dbdih::execSTART_MECONF);
+ addRecSignal(GSN_START_MEREF, &Dbdih::execSTART_MEREF);
+ addRecSignal(GSN_START_COPYREQ, &Dbdih::execSTART_COPYREQ);
+ addRecSignal(GSN_START_COPYCONF, &Dbdih::execSTART_COPYCONF);
+ addRecSignal(GSN_START_COPYREF, &Dbdih::execSTART_COPYREF);
+ addRecSignal(GSN_CREATE_FRAGREQ, &Dbdih::execCREATE_FRAGREQ);
+ addRecSignal(GSN_CREATE_FRAGCONF, &Dbdih::execCREATE_FRAGCONF);
+ addRecSignal(GSN_DIVERIFYREQ, &Dbdih::execDIVERIFYREQ);
+ addRecSignal(GSN_GCP_SAVECONF, &Dbdih::execGCP_SAVECONF);
+ addRecSignal(GSN_GCP_PREPARECONF, &Dbdih::execGCP_PREPARECONF);
+ addRecSignal(GSN_GCP_PREPARE, &Dbdih::execGCP_PREPARE);
+ addRecSignal(GSN_GCP_NODEFINISH, &Dbdih::execGCP_NODEFINISH);
+ addRecSignal(GSN_GCP_COMMIT, &Dbdih::execGCP_COMMIT);
+ addRecSignal(GSN_DIHNDBTAMPER, &Dbdih::execDIHNDBTAMPER);
+ addRecSignal(GSN_CONTINUEB, &Dbdih::execCONTINUEB);
+ addRecSignal(GSN_COPY_GCIREQ, &Dbdih::execCOPY_GCIREQ);
+ addRecSignal(GSN_COPY_GCICONF, &Dbdih::execCOPY_GCICONF);
+ addRecSignal(GSN_COPY_TABREQ, &Dbdih::execCOPY_TABREQ);
+ addRecSignal(GSN_COPY_TABCONF, &Dbdih::execCOPY_TABCONF);
+ addRecSignal(GSN_TCGETOPSIZECONF, &Dbdih::execTCGETOPSIZECONF);
+ addRecSignal(GSN_TC_CLOPSIZECONF, &Dbdih::execTC_CLOPSIZECONF);
+
+ addRecSignal(GSN_LCP_COMPLETE_REP, &Dbdih::execLCP_COMPLETE_REP);
+ addRecSignal(GSN_LCP_FRAG_REP, &Dbdih::execLCP_FRAG_REP);
+ addRecSignal(GSN_START_LCP_REQ, &Dbdih::execSTART_LCP_REQ);
+ addRecSignal(GSN_START_LCP_CONF, &Dbdih::execSTART_LCP_CONF);
+
+ addRecSignal(GSN_SIZEALT_REP, &Dbdih::execSIZEALT_REP);
+ addRecSignal(GSN_UNBLO_DICTCONF, &Dbdih::execUNBLO_DICTCONF);
+ addRecSignal(GSN_COPY_ACTIVECONF, &Dbdih::execCOPY_ACTIVECONF);
+ addRecSignal(GSN_TAB_COMMITREQ, &Dbdih::execTAB_COMMITREQ);
+ addRecSignal(GSN_NODE_FAILREP, &Dbdih::execNODE_FAILREP);
+ addRecSignal(GSN_COPY_FRAGCONF, &Dbdih::execCOPY_FRAGCONF);
+ addRecSignal(GSN_COPY_FRAGREF, &Dbdih::execCOPY_FRAGREF);
+ addRecSignal(GSN_DIADDTABREQ, &Dbdih::execDIADDTABREQ);
+ addRecSignal(GSN_DIGETNODESREQ, &Dbdih::execDIGETNODESREQ);
+ addRecSignal(GSN_DIRELEASEREQ, &Dbdih::execDIRELEASEREQ);
+ addRecSignal(GSN_DISEIZEREQ, &Dbdih::execDISEIZEREQ);
+ addRecSignal(GSN_STTOR, &Dbdih::execSTTOR);
+ addRecSignal(GSN_DI_FCOUNTREQ, &Dbdih::execDI_FCOUNTREQ);
+ addRecSignal(GSN_DIGETPRIMREQ, &Dbdih::execDIGETPRIMREQ);
+ addRecSignal(GSN_GCP_SAVEREF, &Dbdih::execGCP_SAVEREF);
+ addRecSignal(GSN_GCP_TCFINISHED, &Dbdih::execGCP_TCFINISHED);
+ addRecSignal(GSN_READ_NODESCONF, &Dbdih::execREAD_NODESCONF);
+ addRecSignal(GSN_NDB_STTOR, &Dbdih::execNDB_STTOR);
+ addRecSignal(GSN_DICTSTARTCONF, &Dbdih::execDICTSTARTCONF);
+ addRecSignal(GSN_NDB_STARTREQ, &Dbdih::execNDB_STARTREQ);
+ addRecSignal(GSN_GETGCIREQ, &Dbdih::execGETGCIREQ);
+ addRecSignal(GSN_DIH_RESTARTREQ, &Dbdih::execDIH_RESTARTREQ);
+ addRecSignal(GSN_CNTR_CHANGEREP, &Dbdih::execCNTR_CHANGEREP);
+ addRecSignal(GSN_START_RECCONF, &Dbdih::execSTART_RECCONF);
+ addRecSignal(GSN_START_FRAGCONF, &Dbdih::execSTART_FRAGCONF);
+ addRecSignal(GSN_ADD_FRAGCONF, &Dbdih::execADD_FRAGCONF);
+ addRecSignal(GSN_ADD_FRAGREF, &Dbdih::execADD_FRAGREF);
+ addRecSignal(GSN_FSOPENCONF, &Dbdih::execFSOPENCONF);
+ addRecSignal(GSN_FSOPENREF, &Dbdih::execFSOPENREF);
+ addRecSignal(GSN_FSCLOSECONF, &Dbdih::execFSCLOSECONF);
+ addRecSignal(GSN_FSCLOSEREF, &Dbdih::execFSCLOSEREF);
+ addRecSignal(GSN_FSREADCONF, &Dbdih::execFSREADCONF);
+ addRecSignal(GSN_FSREADREF, &Dbdih::execFSREADREF);
+ addRecSignal(GSN_FSWRITECONF, &Dbdih::execFSWRITECONF);
+ addRecSignal(GSN_FSWRITEREF, &Dbdih::execFSWRITEREF);
+ addRecSignal(GSN_SET_VAR_REQ, &Dbdih::execSET_VAR_REQ);
+
+ addRecSignal(GSN_START_INFOREQ,
+ &Dbdih::execSTART_INFOREQ);
+ addRecSignal(GSN_START_INFOREF,
+ &Dbdih::execSTART_INFOREF);
+ addRecSignal(GSN_START_INFOCONF,
+ &Dbdih::execSTART_INFOCONF);
+
+ addRecSignal(GSN_CHECKNODEGROUPSREQ, &Dbdih::execCHECKNODEGROUPSREQ);
+
+ addRecSignal(GSN_BLOCK_COMMIT_ORD,
+ &Dbdih::execBLOCK_COMMIT_ORD);
+ addRecSignal(GSN_UNBLOCK_COMMIT_ORD,
+ &Dbdih::execUNBLOCK_COMMIT_ORD);
+
+ addRecSignal(GSN_DIH_SWITCH_REPLICA_REQ,
+ &Dbdih::execDIH_SWITCH_REPLICA_REQ);
+
+ addRecSignal(GSN_DIH_SWITCH_REPLICA_REF,
+ &Dbdih::execDIH_SWITCH_REPLICA_REF);
+
+ addRecSignal(GSN_DIH_SWITCH_REPLICA_CONF,
+ &Dbdih::execDIH_SWITCH_REPLICA_CONF);
+
+ addRecSignal(GSN_STOP_PERM_REQ, &Dbdih::execSTOP_PERM_REQ);
+ addRecSignal(GSN_STOP_PERM_REF, &Dbdih::execSTOP_PERM_REF);
+ addRecSignal(GSN_STOP_PERM_CONF, &Dbdih::execSTOP_PERM_CONF);
+
+ addRecSignal(GSN_STOP_ME_REQ, &Dbdih::execSTOP_ME_REQ);
+ addRecSignal(GSN_STOP_ME_REF, &Dbdih::execSTOP_ME_REF);
+ addRecSignal(GSN_STOP_ME_CONF, &Dbdih::execSTOP_ME_CONF);
+
+ addRecSignal(GSN_WAIT_GCP_REQ, &Dbdih::execWAIT_GCP_REQ);
+ addRecSignal(GSN_WAIT_GCP_REF, &Dbdih::execWAIT_GCP_REF);
+ addRecSignal(GSN_WAIT_GCP_CONF, &Dbdih::execWAIT_GCP_CONF);
+
+ addRecSignal(GSN_UPDATE_TOREQ, &Dbdih::execUPDATE_TOREQ);
+ addRecSignal(GSN_UPDATE_TOCONF, &Dbdih::execUPDATE_TOCONF);
+
+ addRecSignal(GSN_PREP_DROP_TAB_REQ, &Dbdih::execPREP_DROP_TAB_REQ);
+ addRecSignal(GSN_WAIT_DROP_TAB_CONF, &Dbdih::execWAIT_DROP_TAB_CONF);
+ addRecSignal(GSN_DROP_TAB_REQ, &Dbdih::execDROP_TAB_REQ);
+
+ addRecSignal(GSN_ALTER_TAB_REQ, &Dbdih::execALTER_TAB_REQ);
+
+ addRecSignal(GSN_CREATE_FRAGMENTATION_REQ,
+ &Dbdih::execCREATE_FRAGMENTATION_REQ);
+
+ initData();
+}//Dbdih::Dbdih()
+
+Dbdih::~Dbdih()
+{
+ deallocRecord((void **)&apiConnectRecord, "ApiConnectRecord",
+ sizeof(ApiConnectRecord),
+ capiConnectFileSize);
+
+ deallocRecord((void **)&connectRecord, "ConnectRecord",
+ sizeof(ConnectRecord),
+ cconnectFileSize);
+
+ deallocRecord((void **)&fileRecord, "FileRecord",
+ sizeof(FileRecord),
+ cfileFileSize);
+
+ deallocRecord((void **)&fragmentstore, "Fragmentstore",
+ sizeof(Fragmentstore),
+ cfragstoreFileSize);
+
+ deallocRecord((void **)&pageRecord, "PageRecord",
+ sizeof(PageRecord),
+ cpageFileSize);
+
+ deallocRecord((void **)&replicaRecord, "ReplicaRecord",
+ sizeof(ReplicaRecord),
+ creplicaFileSize);
+
+ deallocRecord((void **)&tabRecord, "TabRecord",
+ sizeof(TabRecord),
+ ctabFileSize);
+
+ // Records with constant sizes
+ deallocRecord((void **)&createReplicaRecord,
+ "CreateReplicaRecord", sizeof(CreateReplicaRecord),
+ ZCREATE_REPLICA_FILE_SIZE);
+
+ deallocRecord((void **)&nodeGroupRecord, "NodeGroupRecord",
+ sizeof(NodeGroupRecord), MAX_NDB_NODES);
+
+ deallocRecord((void **)&nodeRecord, "NodeRecord",
+ sizeof(NodeRecord), MAX_NDB_NODES);
+
+ deallocRecord((void **)&takeOverRecord, "TakeOverRecord",
+ sizeof(TakeOverRecord),
+ MAX_NDB_NODES);
+
+}//Dbdih::~Dbdih()
+
+BLOCK_FUNCTIONS(Dbdih);
+
+
+
diff --git a/ndb/src/kernel/blocks/dbdih/DbdihMain.cpp b/ndb/src/kernel/blocks/dbdih/DbdihMain.cpp
new file mode 100644
index 00000000000..cefbb3e66a3
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdih/DbdihMain.cpp
@@ -0,0 +1,14104 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBDIH_C
+#include <ndb_limits.h>
+#include <ndb_version.h>
+#include <NdbOut.hpp>
+
+#include "Dbdih.hpp"
+#include "Configuration.hpp"
+
+#include <signaldata/BlockCommitOrd.hpp>
+#include <signaldata/CheckNodeGroups.hpp>
+#include <signaldata/CreateFrag.hpp>
+#include <signaldata/CopyActive.hpp>
+#include <signaldata/CopyFrag.hpp>
+#include <signaldata/CopyGCIReq.hpp>
+#include <signaldata/DiAddTab.hpp>
+#include <signaldata/DictStart.hpp>
+#include <signaldata/DiGetNodes.hpp>
+#include <signaldata/DihContinueB.hpp>
+#include <signaldata/DihSizeAltReq.hpp>
+#include <signaldata/DihSwitchReplica.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+#include <signaldata/EmptyLcp.hpp>
+#include <signaldata/EndTo.hpp>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/GCPSave.hpp>
+#include <signaldata/HotSpareRep.hpp>
+#include <signaldata/MasterGCP.hpp>
+#include <signaldata/MasterLCP.hpp>
+#include <signaldata/NFCompleteRep.hpp>
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/ReadNodesConf.hpp>
+#include <signaldata/SetVarReq.hpp>
+#include <signaldata/StartFragReq.hpp>
+#include <signaldata/StartInfo.hpp>
+#include <signaldata/StartMe.hpp>
+#include <signaldata/StartPerm.hpp>
+#include <signaldata/StartRec.hpp>
+#include <signaldata/StartTo.hpp>
+#include <signaldata/StopPerm.hpp>
+#include <signaldata/StopMe.hpp>
+#include <signaldata/TestOrd.hpp>
+#include <signaldata/UpdateTo.hpp>
+#include <signaldata/WaitGCP.hpp>
+#include <signaldata/DihStartTab.hpp>
+#include <signaldata/LCP.hpp>
+#include <signaldata/SystemError.hpp>
+
+#include <signaldata/DropTab.hpp>
+#include <signaldata/AlterTab.hpp>
+#include <signaldata/PrepDropTab.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/CreateFragmentation.hpp>
+#include <signaldata/LqhFrag.hpp>
+
+#include <DebuggerNames.hpp>
+
+#define SYSFILE ((Sysfile *)&sysfileData[0])
+
+#define RETURN_IF_NODE_NOT_ALIVE(node) \
+ if (!checkNodeAlive((node))) { \
+ jam(); \
+ return; \
+ } \
+
+#define RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverIndex, regTOPtr) \
+ regTOPtr.i = takeOverIndex; \
+ ptrCheckGuard(regTOPtr, MAX_NDB_NODES, takeOverRecord); \
+ if (checkToInterrupted(regTOPtr)) { \
+ jam(); \
+ return; \
+ } \
+
+#define receiveLoopMacro(sigName, receiveNodeId)\
+{ \
+ c_##sigName##_Counter.clearWaitingFor(receiveNodeId); \
+ if(c_##sigName##_Counter.done() == false){ \
+ jam(); \
+ return; \
+ } \
+}
+
+#define sendLoopMacro(sigName, signalRoutine) \
+{ \
+ c_##sigName##_Counter.clearWaitingFor(); \
+ NodeRecordPtr specNodePtr; \
+ specNodePtr.i = cfirstAliveNode; \
+ do { \
+ jam(); \
+ ptrCheckGuard(specNodePtr, MAX_NDB_NODES, nodeRecord); \
+ c_##sigName##_Counter.setWaitingFor(specNodePtr.i); \
+ signalRoutine(signal, specNodePtr.i); \
+ specNodePtr.i = specNodePtr.p->nextNode; \
+ } while (specNodePtr.i != RNIL); \
+}
+
+static
+Uint32
+prevLcpNo(Uint32 lcpNo){
+ if(lcpNo == 0)
+ return MAX_LCP_STORED - 1;
+ return lcpNo - 1;
+}
+
+static
+Uint32
+nextLcpNo(Uint32 lcpNo){
+ lcpNo++;
+ if(lcpNo == MAX_LCP_STORED)
+ return 0;
+ return lcpNo;
+}
+
+#define gth(x, y) ndbrequire(((int)x)>((int)y))
+
+void Dbdih::nullRoutine(Signal* signal, Uint32 nodeId)
+{
+}//Dbdih::nullRoutine()
+
+void Dbdih::sendCOPY_GCIREQ(Signal* signal, Uint32 nodeId)
+{
+ ndbrequire(c_copyGCIMaster.m_copyReason != CopyGCIReq::IDLE);
+
+ const BlockReference ref = calcDihBlockRef(nodeId);
+ const Uint32 wordPerSignal = CopyGCIReq::DATA_SIZE;
+ const Uint32 noOfSignals = ((Sysfile::SYSFILE_SIZE32 + (wordPerSignal - 1)) /
+ wordPerSignal);
+
+ CopyGCIReq * const copyGCI = (CopyGCIReq *)&signal->theData[0];
+ copyGCI->anyData = nodeId;
+ copyGCI->copyReason = c_copyGCIMaster.m_copyReason;
+ copyGCI->startWord = 0;
+
+ for(Uint32 i = 0; i < noOfSignals; i++) {
+ jam();
+ { // Do copy
+ const int startWord = copyGCI->startWord;
+ for(Uint32 j = 0; j < wordPerSignal; j++) {
+ copyGCI->data[j] = sysfileData[j+startWord];
+ }//for
+ }
+ sendSignal(ref, GSN_COPY_GCIREQ, signal, 25, JBB);
+ copyGCI->startWord += wordPerSignal;
+ }//for
+}//Dbdih::sendCOPY_GCIREQ()
+
+
+void Dbdih::sendDIH_SWITCH_REPLICA_REQ(Signal* signal, Uint32 nodeId)
+{
+ const BlockReference ref = calcDihBlockRef(nodeId);
+ sendSignal(ref, GSN_DIH_SWITCH_REPLICA_REQ, signal,
+ DihSwitchReplicaReq::SignalLength, JBB);
+}//Dbdih::sendDIH_SWITCH_REPLICA_REQ()
+
+void Dbdih::sendEMPTY_LCP_REQ(Signal* signal, Uint32 nodeId)
+{
+ BlockReference ref = calcLqhBlockRef(nodeId);
+ sendSignal(ref, GSN_EMPTY_LCP_REQ, signal, EmptyLcpReq::SignalLength, JBB);
+}//Dbdih::sendEMPTY_LCPREQ()
+
+void Dbdih::sendEND_TOREQ(Signal* signal, Uint32 nodeId)
+{
+ BlockReference ref = calcDihBlockRef(nodeId);
+ sendSignal(ref, GSN_END_TOREQ, signal, EndToReq::SignalLength, JBB);
+}//Dbdih::sendEND_TOREQ()
+
+void Dbdih::sendGCP_COMMIT(Signal* signal, Uint32 nodeId)
+{
+ BlockReference ref = calcDihBlockRef(nodeId);
+ signal->theData[0] = cownNodeId;
+ signal->theData[1] = cnewgcp;
+ sendSignal(ref, GSN_GCP_COMMIT, signal, 2, JBA);
+}//Dbdih::sendGCP_COMMIT()
+
+void Dbdih::sendGCP_PREPARE(Signal* signal, Uint32 nodeId)
+{
+ BlockReference ref = calcDihBlockRef(nodeId);
+ signal->theData[0] = cownNodeId;
+ signal->theData[1] = cnewgcp;
+ sendSignal(ref, GSN_GCP_PREPARE, signal, 2, JBA);
+}//Dbdih::sendGCP_PREPARE()
+
+void Dbdih::sendGCP_SAVEREQ(Signal* signal, Uint32 nodeId)
+{
+ GCPSaveReq * const saveReq = (GCPSaveReq*)&signal->theData[0];
+ BlockReference ref = calcLqhBlockRef(nodeId);
+ saveReq->dihBlockRef = reference();
+ saveReq->dihPtr = nodeId;
+ saveReq->gci = coldgcp;
+ sendSignal(ref, GSN_GCP_SAVEREQ, signal, GCPSaveReq::SignalLength, JBB);
+}//Dbdih::sendGCP_SAVEREQ()
+
+void Dbdih::sendINCL_NODEREQ(Signal* signal, Uint32 nodeId)
+{
+ BlockReference nodeDihRef = calcDihBlockRef(nodeId);
+ signal->theData[0] = reference();
+ signal->theData[1] = c_nodeStartMaster.startNode;
+ signal->theData[2] = c_nodeStartMaster.failNr;
+ signal->theData[3] = c_nodeStartMaster.ndbVersion;
+ signal->theData[4] = currentgcp;
+ sendSignal(nodeDihRef, GSN_INCL_NODEREQ, signal, 5, JBB);
+}//Dbdih::sendINCL_NODEREQ()
+
+void Dbdih::sendMASTER_GCPREQ(Signal* signal, Uint32 nodeId)
+{
+ BlockReference ref = calcDihBlockRef(nodeId);
+ sendSignal(ref, GSN_MASTER_GCPREQ, signal, MasterGCPReq::SignalLength, JBB);
+}//Dbdih::sendMASTER_GCPREQ()
+
+void Dbdih::sendMASTER_LCPREQ(Signal* signal, Uint32 nodeId)
+{
+ BlockReference ref = calcDihBlockRef(nodeId);
+ sendSignal(ref, GSN_MASTER_LCPREQ, signal, MasterLCPReq::SignalLength, JBB);
+}//Dbdih::sendMASTER_LCPREQ()
+
+void Dbdih::sendSTART_INFOREQ(Signal* signal, Uint32 nodeId)
+{
+ const BlockReference ref = calcDihBlockRef(nodeId);
+ sendSignal(ref, GSN_START_INFOREQ, signal, StartInfoReq::SignalLength, JBB);
+}//sendSTART_INFOREQ()
+
+void Dbdih::sendSTART_RECREQ(Signal* signal, Uint32 nodeId)
+{
+ StartRecReq * const req = (StartRecReq*)&signal->theData[0];
+ BlockReference ref = calcLqhBlockRef(nodeId);
+ req->receivingNodeId = nodeId;
+ req->senderRef = reference();
+ req->keepGci = SYSFILE->keepGCI;
+ req->lastCompletedGci = SYSFILE->lastCompletedGCI[nodeId];
+ req->newestGci = SYSFILE->newestRestorableGCI;
+ sendSignal(ref, GSN_START_RECREQ, signal, StartRecReq::SignalLength, JBB);
+
+ signal->theData[0] = EventReport::StartREDOLog;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = SYSFILE->keepGCI;
+ signal->theData[3] = SYSFILE->lastCompletedGCI[nodeId];
+ signal->theData[4] = SYSFILE->newestRestorableGCI;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 5, JBB);
+}//Dbdih::sendSTART_RECREQ()
+
+void Dbdih::sendSTART_TOREQ(Signal* signal, Uint32 nodeId)
+{
+ BlockReference ref = calcDihBlockRef(nodeId);
+ sendSignal(ref, GSN_START_TOREQ, signal, StartToReq::SignalLength, JBB);
+}//Dbdih::sendSTART_TOREQ()
+
+void Dbdih::sendSTOP_ME_REQ(Signal* signal, Uint32 nodeId)
+{
+ if (nodeId != getOwnNodeId()) {
+ jam();
+ const BlockReference ref = calcDihBlockRef(nodeId);
+ sendSignal(ref, GSN_STOP_ME_REQ, signal, StopMeReq::SignalLength, JBB);
+ }//if
+}//Dbdih::sendSTOP_ME_REQ()
+
+void Dbdih::sendTC_CLOPSIZEREQ(Signal* signal, Uint32 nodeId)
+{
+ BlockReference ref = calcTcBlockRef(nodeId);
+ signal->theData[0] = nodeId;
+ signal->theData[1] = reference();
+ sendSignal(ref, GSN_TC_CLOPSIZEREQ, signal, 2, JBB);
+}//Dbdih::sendTC_CLOPSIZEREQ()
+
+void Dbdih::sendTCGETOPSIZEREQ(Signal* signal, Uint32 nodeId)
+{
+ BlockReference ref = calcTcBlockRef(nodeId);
+ signal->theData[0] = nodeId;
+ signal->theData[1] = reference();
+ sendSignal(ref, GSN_TCGETOPSIZEREQ, signal, 2, JBB);
+}//Dbdih::sendTCGETOPSIZEREQ()
+
+void Dbdih::sendUPDATE_TOREQ(Signal* signal, Uint32 nodeId)
+{
+ const BlockReference ref = calcDihBlockRef(nodeId);
+ sendSignal(ref, GSN_UPDATE_TOREQ, signal, UpdateToReq::SignalLength, JBB);
+}//sendUPDATE_TOREQ()
+
+void Dbdih::execCNTR_CHANGEREP(Signal* signal)
+{
+ (void)signal; // Don't want compiler warning
+ jamEntry();
+ return;
+}//Dbdih::execCNTR_CHANGEREP()
+
+void Dbdih::execCONTINUEB(Signal* signal)
+{
+ jamEntry();
+ switch ((DihContinueB::Type)signal->theData[0]) {
+ case DihContinueB::ZPACK_TABLE_INTO_PAGES:
+ {
+ jam();
+ Uint32 tableId = signal->theData[1];
+ packTableIntoPagesLab(signal, tableId);
+ return;
+ break;
+ }
+ case DihContinueB::ZPACK_FRAG_INTO_PAGES:
+ {
+ RWFragment wf;
+ jam();
+ wf.rwfTabPtr.i = signal->theData[1];
+ ptrCheckGuard(wf.rwfTabPtr, ctabFileSize, tabRecord);
+ wf.fragId = signal->theData[2];
+ wf.pageIndex = signal->theData[3];
+ wf.wordIndex = signal->theData[4];
+ packFragIntoPagesLab(signal, &wf);
+ return;
+ break;
+ }
+ case DihContinueB::ZREAD_PAGES_INTO_TABLE:
+ {
+ jam();
+ Uint32 tableId = signal->theData[1];
+ readPagesIntoTableLab(signal, tableId);
+ return;
+ break;
+ }
+ case DihContinueB::ZREAD_PAGES_INTO_FRAG:
+ {
+ RWFragment rf;
+ jam();
+ rf.rwfTabPtr.i = signal->theData[1];
+ ptrCheckGuard(rf.rwfTabPtr, ctabFileSize, tabRecord);
+ rf.fragId = signal->theData[2];
+ rf.pageIndex = signal->theData[3];
+ rf.wordIndex = signal->theData[4];
+ readPagesIntoFragLab(signal, &rf);
+ return;
+ break;
+ }
+ case DihContinueB::ZCOPY_TABLE:
+ {
+ jam();
+ Uint32 tableId = signal->theData[1];
+ copyTableLab(signal, tableId);
+ return;
+ }
+ case DihContinueB::ZCOPY_TABLE_NODE:
+ {
+ NodeRecordPtr nodePtr;
+ CopyTableNode ctn;
+ jam();
+ ctn.ctnTabPtr.i = signal->theData[1];
+ ptrCheckGuard(ctn.ctnTabPtr, ctabFileSize, tabRecord);
+ nodePtr.i = signal->theData[2];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ ctn.pageIndex = signal->theData[3];
+ ctn.wordIndex = signal->theData[4];
+ ctn.noOfWords = signal->theData[5];
+ copyTableNode(signal, &ctn, nodePtr);
+ return;
+ }
+ case DihContinueB::ZSTART_FRAGMENT:
+ {
+ jam();
+ Uint32 tableId = signal->theData[1];
+ Uint32 fragId = signal->theData[2];
+ startFragment(signal, tableId, fragId);
+ return;
+ }
+ case DihContinueB::ZCOMPLETE_RESTART:
+ jam();
+ completeRestartLab(signal);
+ return;
+ case DihContinueB::ZREAD_TABLE_FROM_PAGES:
+ {
+ TabRecordPtr tabPtr;
+ jam();
+ tabPtr.i = signal->theData[1];
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ readTableFromPagesLab(signal, tabPtr);
+ return;
+ }
+ case DihContinueB::ZSR_PHASE2_READ_TABLE:
+ {
+ TabRecordPtr tabPtr;
+ jam();
+ tabPtr.i = signal->theData[1];
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ srPhase2ReadTableLab(signal, tabPtr);
+ return;
+ }
+ case DihContinueB::ZCHECK_TC_COUNTER:
+ jam();
+#ifndef NO_LCP
+ checkTcCounterLab(signal);
+#endif
+ return;
+ case DihContinueB::ZCALCULATE_KEEP_GCI:
+ {
+ jam();
+ Uint32 tableId = signal->theData[1];
+ Uint32 fragId = signal->theData[2];
+ calculateKeepGciLab(signal, tableId, fragId);
+ return;
+ }
+ case DihContinueB::ZSTORE_NEW_LCP_ID:
+ jam();
+ storeNewLcpIdLab(signal);
+ return;
+ case DihContinueB::ZTABLE_UPDATE:
+ {
+ TabRecordPtr tabPtr;
+ jam();
+ tabPtr.i = signal->theData[1];
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ tableUpdateLab(signal, tabPtr);
+ return;
+ }
+ case DihContinueB::ZCHECK_LCP_COMPLETED:
+ {
+ jam();
+ checkLcpCompletedLab(signal);
+ return;
+ }
+ case DihContinueB::ZINIT_LCP:
+ {
+ jam();
+ Uint32 senderRef = signal->theData[1];
+ Uint32 tableId = signal->theData[2];
+ initLcpLab(signal, senderRef, tableId);
+ return;
+ }
+ case DihContinueB::ZADD_TABLE_MASTER_PAGES:
+ {
+ TabRecordPtr tabPtr;
+ jam();
+ tabPtr.i = signal->theData[1];
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ tabPtr.p->tabUpdateState = TabRecord::US_ADD_TABLE_MASTER;
+ tableUpdateLab(signal, tabPtr);
+ return;
+ break;
+ }
+ case DihContinueB::ZDIH_ADD_TABLE_MASTER:
+ {
+ jam();
+ addTable_closeConf(signal, signal->theData[1]);
+ return;
+ }
+ case DihContinueB::ZADD_TABLE_SLAVE_PAGES:
+ {
+ TabRecordPtr tabPtr;
+ jam();
+ tabPtr.i = signal->theData[1];
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ tabPtr.p->tabUpdateState = TabRecord::US_ADD_TABLE_SLAVE;
+ tableUpdateLab(signal, tabPtr);
+ return;
+ }
+ case DihContinueB::ZDIH_ADD_TABLE_SLAVE:
+ {
+ ndbrequire(false);
+ return;
+ }
+ case DihContinueB::ZSTART_GCP:
+ jam();
+#ifndef NO_GCP
+ startGcpLab(signal, signal->theData[1]);
+#endif
+ return;
+ break;
+ case DihContinueB::ZCOPY_GCI:{
+ jam();
+ CopyGCIReq::CopyReason reason = (CopyGCIReq::CopyReason)signal->theData[1];
+ ndbrequire(c_copyGCIMaster.m_copyReason == reason);
+ sendLoopMacro(COPY_GCIREQ, sendCOPY_GCIREQ);
+ return;
+ }
+ break;
+ case DihContinueB::ZEMPTY_VERIFY_QUEUE:
+ jam();
+ emptyverificbuffer(signal, true);
+ return;
+ break;
+ case DihContinueB::ZCHECK_GCP_STOP:
+ jam();
+#ifndef NO_GCP
+ checkGcpStopLab(signal);
+#endif
+ return;
+ break;
+ case DihContinueB::ZREMOVE_NODE_FROM_TABLE:
+ {
+ jam();
+ Uint32 nodeId = signal->theData[1];
+ Uint32 tableId = signal->theData[2];
+ removeNodeFromTables(signal, nodeId, tableId);
+ return;
+ }
+ case DihContinueB::ZCOPY_NODE:
+ {
+ jam();
+ Uint32 tableId = signal->theData[1];
+ copyNodeLab(signal, tableId);
+ return;
+ }
+ case DihContinueB::ZSTART_TAKE_OVER:
+ {
+ jam();
+ Uint32 takeOverPtrI = signal->theData[1];
+ Uint32 startNode = signal->theData[2];
+ Uint32 toNode = signal->theData[3];
+ startTakeOver(signal, takeOverPtrI, startNode, toNode);
+ return;
+ break;
+ }
+ case DihContinueB::ZCHECK_START_TAKE_OVER:
+ jam();
+ checkStartTakeOver(signal);
+ break;
+ case DihContinueB::ZTO_START_COPY_FRAG:
+ {
+ jam();
+ Uint32 takeOverPtrI = signal->theData[1];
+ startNextCopyFragment(signal, takeOverPtrI);
+ return;
+ }
+ case DihContinueB::ZINVALIDATE_NODE_LCP:
+ {
+ jam();
+ const Uint32 nodeId = signal->theData[1];
+ const Uint32 tableId = signal->theData[2];
+ invalidateNodeLCP(signal, nodeId, tableId);
+ return;
+ }
+ case DihContinueB::ZINITIALISE_RECORDS:
+ jam();
+ initialiseRecordsLab(signal, signal->theData[1]);
+ return;
+ break;
+ case DihContinueB::ZSTART_PERMREQ_AGAIN:
+ jam();
+ nodeRestartPh2Lab(signal);
+ return;
+ break;
+ case DihContinueB::SwitchReplica:
+ {
+ jam();
+ const Uint32 nodeId = signal->theData[1];
+ const Uint32 tableId = signal->theData[2];
+ const Uint32 fragNo = signal->theData[3];
+ switchReplica(signal, nodeId, tableId, fragNo);
+ return;
+ }
+ case DihContinueB::ZSEND_START_TO:
+ {
+ jam();
+ Uint32 takeOverPtrI = signal->theData[1];
+ sendStartTo(signal, takeOverPtrI);
+ return;
+ }
+ case DihContinueB::ZSEND_ADD_FRAG:
+ {
+ jam();
+ Uint32 takeOverPtrI = signal->theData[1];
+ toCopyFragLab(signal, takeOverPtrI);
+ return;
+ }
+ case DihContinueB::ZSEND_UPDATE_TO:
+ {
+ jam();
+ Uint32 takeOverPtrI = signal->theData[1];
+ Uint32 updateState = signal->theData[4];
+ sendUpdateTo(signal, takeOverPtrI, updateState);
+ return;
+ }
+ case DihContinueB::ZSEND_END_TO:
+ {
+ jam();
+ Uint32 takeOverPtrI = signal->theData[1];
+ sendEndTo(signal, takeOverPtrI);
+ return;
+ }
+ case DihContinueB::ZSEND_CREATE_FRAG:
+ {
+ jam();
+ Uint32 takeOverPtrI = signal->theData[1];
+ Uint32 storedType = signal->theData[2];
+ Uint32 startGci = signal->theData[3];
+ sendCreateFragReq(signal, startGci, storedType, takeOverPtrI);
+ return;
+ }
+ case DihContinueB::WAIT_DROP_TAB_WRITING_TO_FILE:{
+ jam();
+ TabRecordPtr tabPtr;
+ tabPtr.i = signal->theData[1];
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ waitDropTabWritingToFile(signal, tabPtr);
+ return;
+ }
+ case DihContinueB::CHECK_WAIT_DROP_TAB_FAILED_LQH:{
+ jam();
+ Uint32 nodeId = signal->theData[1];
+ Uint32 tableId = signal->theData[2];
+ checkWaitDropTabFailedLqh(signal, nodeId, tableId);
+ return;
+ }
+ }//switch
+
+ ndbrequire(false);
+ return;
+}//Dbdih::execCONTINUEB()
+
+void Dbdih::execCOPY_GCIREQ(Signal* signal)
+{
+ CopyGCIReq * const copyGCI = (CopyGCIReq *)&signal->theData[0];
+ jamEntry();
+ CopyGCIReq::CopyReason reason = (CopyGCIReq::CopyReason)copyGCI->copyReason;
+ const Uint32 tstart = copyGCI->startWord;
+
+ ndbrequire(cmasterdihref == signal->senderBlockRef()) ;
+ ndbrequire(c_copyGCISlave.m_copyReason == CopyGCIReq::IDLE);
+ ndbrequire(c_copyGCISlave.m_expectedNextWord == tstart);
+ ndbrequire(reason != CopyGCIReq::IDLE);
+
+ arrGuard(tstart + CopyGCIReq::DATA_SIZE, sizeof(sysfileData)/4);
+ for(Uint32 i = 0; i<CopyGCIReq::DATA_SIZE; i++)
+ cdata[tstart+i] = copyGCI->data[i];
+
+ if ((tstart + CopyGCIReq::DATA_SIZE) >= Sysfile::SYSFILE_SIZE32) {
+ jam();
+ c_copyGCISlave.m_expectedNextWord = 0;
+ } else {
+ jam();
+ c_copyGCISlave.m_expectedNextWord += CopyGCIReq::DATA_SIZE;
+ return;
+ }//if
+
+ memcpy(sysfileData, cdata, sizeof(sysfileData));
+
+ c_copyGCISlave.m_copyReason = reason;
+ c_copyGCISlave.m_senderRef = signal->senderBlockRef();
+ c_copyGCISlave.m_senderData = copyGCI->anyData;
+
+ CRASH_INSERTION2(7020, reason==CopyGCIReq::LOCAL_CHECKPOINT);
+ CRASH_INSERTION2(7008, reason==CopyGCIReq::GLOBAL_CHECKPOINT);
+
+ /* -------------------------------------------------------------------------*/
+ /* WE SET THE REQUESTER OF THE COPY GCI TO THE CURRENT MASTER. IF THE */
+ /* CURRENT MASTER WE DO NOT WANT THE NEW MASTER TO RECEIVE CONFIRM OF */
+ /* SOMETHING HE HAS NOT SENT. THE TAKE OVER MUST BE CAREFUL. */
+ /* -------------------------------------------------------------------------*/
+ bool ok = false;
+ switch(reason){
+ case CopyGCIReq::IDLE:
+ ok = true;
+ jam();
+ ndbrequire(false);
+ break;
+ case CopyGCIReq::LOCAL_CHECKPOINT: {
+ ok = true;
+ jam();
+ c_lcpState.setLcpStatus(LCP_COPY_GCI, __LINE__);
+ c_lcpState.m_masterLcpDihRef = cmasterdihref;
+ setNodeInfo(signal);
+ break;
+ }
+ case CopyGCIReq::RESTART: {
+ ok = true;
+ jam();
+ coldgcp = SYSFILE->newestRestorableGCI;
+ crestartGci = SYSFILE->newestRestorableGCI;
+ Sysfile::setRestartOngoing(SYSFILE->systemRestartBits);
+ currentgcp = coldgcp + 1;
+ cnewgcp = coldgcp + 1;
+ setNodeInfo(signal);
+ if ((Sysfile::getLCPOngoing(SYSFILE->systemRestartBits))) {
+ jam();
+ /* -------------------------------------------------------------------- */
+ // IF THERE WAS A LOCAL CHECKPOINT ONGOING AT THE CRASH MOMENT WE WILL
+ // INVALIDATE THAT LOCAL CHECKPOINT.
+ /* -------------------------------------------------------------------- */
+ invalidateLcpInfoAfterSr();
+ }//if
+ break;
+ }
+ case CopyGCIReq::GLOBAL_CHECKPOINT: {
+ ok = true;
+ jam();
+ cgcpParticipantState = GCP_PARTICIPANT_COPY_GCI_RECEIVED;
+ setNodeInfo(signal);
+ break;
+ }//if
+ case CopyGCIReq::INITIAL_START_COMPLETED:
+ ok = true;
+ jam();
+ break;
+ }
+ ndbrequire(ok);
+
+ /* ----------------------------------------------------------------------- */
+ /* WE START BY TRYING TO OPEN THE FIRST RESTORABLE GCI FILE. */
+ /* ----------------------------------------------------------------------- */
+ FileRecordPtr filePtr;
+ filePtr.i = crestartInfoFile[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ if (filePtr.p->fileStatus == FileRecord::OPEN) {
+ jam();
+ openingCopyGciSkipInitLab(signal, filePtr);
+ return;
+ }//if
+ openFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::OPENING_COPY_GCI;
+ return;
+}//Dbdih::execCOPY_GCIREQ()
+
+void Dbdih::execDICTSTARTCONF(Signal* signal)
+{
+ jamEntry();
+ Uint32 nodeId = refToNode(signal->getSendersBlockRef());
+ if (nodeId != getOwnNodeId()) {
+ jam();
+ nodeDictStartConfLab(signal);
+ } else {
+ jam();
+ dictStartConfLab(signal);
+ }//if
+}//Dbdih::execDICTSTARTCONF()
+
+void Dbdih::execFSCLOSECONF(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ jamEntry();
+ filePtr.i = signal->theData[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ filePtr.p->fileStatus = FileRecord::CLOSED;
+ FileRecord::ReqStatus status = filePtr.p->reqStatus;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ switch (status) {
+ case FileRecord::CLOSING_GCP:
+ jam();
+ closingGcpLab(signal, filePtr);
+ break;
+ case FileRecord::CLOSING_GCP_CRASH:
+ jam();
+ closingGcpCrashLab(signal, filePtr);
+ break;
+ case FileRecord::CLOSING_TABLE_CRASH:
+ jam();
+ closingTableCrashLab(signal, filePtr);
+ break;
+ case FileRecord::CLOSING_TABLE_SR:
+ jam();
+ closingTableSrLab(signal, filePtr);
+ break;
+ case FileRecord::TABLE_CLOSE:
+ jam();
+ tableCloseLab(signal, filePtr);
+ break;
+ case FileRecord::TABLE_CLOSE_DELETE:
+ jam();
+ tableDeleteLab(signal, filePtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbdih::execFSCLOSECONF()
+
+void Dbdih::execFSCLOSEREF(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ jamEntry();
+ filePtr.i = signal->theData[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ FileRecord::ReqStatus status = filePtr.p->reqStatus;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ switch (status) {
+ case FileRecord::CLOSING_GCP:
+ ndbrequire(false);
+ break;
+ case FileRecord::CLOSING_GCP_CRASH:
+ jam();
+ closingGcpCrashLab(signal, filePtr);
+ break;
+ case FileRecord::CLOSING_TABLE_CRASH:
+ jam();
+ closingTableCrashLab(signal, filePtr);
+ break;
+ case FileRecord::CLOSING_TABLE_SR:
+ ndbrequire(false);
+ break;
+ case FileRecord::TABLE_CLOSE:
+ ndbrequire(false);
+ break;
+ case FileRecord::TABLE_CLOSE_DELETE:
+ ndbrequire(false);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbdih::execFSCLOSEREF()
+
+void Dbdih::execFSOPENCONF(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ jamEntry();
+ filePtr.i = signal->theData[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ filePtr.p->fileRef = signal->theData[1];
+ filePtr.p->fileStatus = FileRecord::OPEN;
+ FileRecord::ReqStatus status = filePtr.p->reqStatus;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ switch (status) {
+ case FileRecord::CREATING_GCP:
+ jam();
+ creatingGcpLab(signal, filePtr);
+ break;
+ case FileRecord::OPENING_COPY_GCI:
+ jam();
+ openingCopyGciSkipInitLab(signal, filePtr);
+ break;
+ case FileRecord::CREATING_COPY_GCI:
+ jam();
+ openingCopyGciSkipInitLab(signal, filePtr);
+ break;
+ case FileRecord::OPENING_GCP:
+ jam();
+ openingGcpLab(signal, filePtr);
+ break;
+ case FileRecord::OPENING_TABLE:
+ jam();
+ openingTableLab(signal, filePtr);
+ break;
+ case FileRecord::TABLE_CREATE:
+ jam();
+ tableCreateLab(signal, filePtr);
+ break;
+ case FileRecord::TABLE_OPEN_FOR_DELETE:
+ jam();
+ tableOpenLab(signal, filePtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbdih::execFSOPENCONF()
+
+void Dbdih::execFSOPENREF(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ jamEntry();
+ filePtr.i = signal->theData[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ FileRecord::ReqStatus status = filePtr.p->reqStatus;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ switch (status) {
+ case FileRecord::CREATING_GCP:
+ /* --------------------------------------------------------------------- */
+ /* WE DID NOT MANAGE TO CREATE A GLOBAL CHECKPOINT FILE. SERIOUS ERROR */
+ /* WHICH CAUSES A SYSTEM RESTART. */
+ /* --------------------------------------------------------------------- */
+ ndbrequire(false);
+ break;
+ case FileRecord::OPENING_COPY_GCI:
+ jam();
+ openingCopyGciErrorLab(signal, filePtr);
+ break;
+ case FileRecord::CREATING_COPY_GCI:
+ ndbrequire(false);
+ break;
+ case FileRecord::OPENING_GCP:
+ jam();
+ openingGcpErrorLab(signal, filePtr);
+ break;
+ case FileRecord::OPENING_TABLE:
+ jam();
+ openingTableErrorLab(signal, filePtr);
+ break;
+ case FileRecord::TABLE_CREATE:
+ ndbrequire(false);
+ break;
+ case FileRecord::TABLE_OPEN_FOR_DELETE:
+ jam();
+ tableDeleteLab(signal, filePtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbdih::execFSOPENREF()
+
+void Dbdih::execFSREADCONF(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ jamEntry();
+ filePtr.i = signal->theData[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ FileRecord::ReqStatus status = filePtr.p->reqStatus;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ switch (status) {
+ case FileRecord::READING_GCP:
+ jam();
+ readingGcpLab(signal, filePtr);
+ break;
+ case FileRecord::READING_TABLE:
+ jam();
+ readingTableLab(signal, filePtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbdih::execFSREADCONF()
+
+void Dbdih::execFSREADREF(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ jamEntry();
+ filePtr.i = signal->theData[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ FileRecord::ReqStatus status = filePtr.p->reqStatus;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ switch (status) {
+ case FileRecord::READING_GCP:
+ jam();
+ readingGcpErrorLab(signal, filePtr);
+ break;
+ case FileRecord::READING_TABLE:
+ jam();
+ readingTableErrorLab(signal, filePtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbdih::execFSREADREF()
+
+void Dbdih::execFSWRITECONF(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ jamEntry();
+ filePtr.i = signal->theData[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ FileRecord::ReqStatus status = filePtr.p->reqStatus;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ switch (status) {
+ case FileRecord::WRITING_COPY_GCI:
+ jam();
+ writingCopyGciLab(signal, filePtr);
+ break;
+ case FileRecord::WRITE_INIT_GCP:
+ jam();
+ writeInitGcpLab(signal, filePtr);
+ break;
+ case FileRecord::TABLE_WRITE:
+ jam();
+ tableWriteLab(signal, filePtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbdih::execFSWRITECONF()
+
+void Dbdih::execFSWRITEREF(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ jamEntry();
+ filePtr.i = signal->theData[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ FileRecord::ReqStatus status = filePtr.p->reqStatus;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ switch (status) {
+ case FileRecord::WRITING_COPY_GCI:
+ /* --------------------------------------------------------------------- */
+ /* EVEN CREATING THE FILE DID NOT WORK. WE WILL THEN CRASH. */
+ /* ERROR IN WRITING FILE. WE WILL NOT CONTINUE FROM HERE. */
+ /* --------------------------------------------------------------------- */
+ ndbrequire(false);
+ break;
+ case FileRecord::WRITE_INIT_GCP:
+ /* --------------------------------------------------------------------- */
+ /* AN ERROR OCCURRED IN WRITING A GCI FILE WHICH IS A SERIOUS ERROR */
+ /* THAT CAUSE A SYSTEM RESTART. */
+ /* --------------------------------------------------------------------- */
+ ndbrequire(false);
+ break;
+ case FileRecord::TABLE_WRITE:
+ ndbrequire(false);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbdih::execFSWRITEREF()
+
+void Dbdih::execGETGCIREQ(Signal* signal)
+{
+
+ jamEntry();
+ Uint32 userPtr = signal->theData[0];
+ BlockReference userRef = signal->theData[1];
+
+ signal->theData[0] = userPtr;
+ signal->theData[1] = SYSFILE->newestRestorableGCI;
+ sendSignal(userRef, GSN_GETGCICONF, signal, 2, JBB);
+}//Dbdih::execGETGCIREQ()
+
+void Dbdih::execSIZEALT_REP(Signal* signal)
+{
+ jamEntry();
+ capiConnectFileSize = signal->theData[DihSizeAltReq::IND_API_CONNECT];
+ cconnectFileSize = signal->theData[DihSizeAltReq::IND_CONNECT];
+ cfragstoreFileSize = signal->theData[DihSizeAltReq::IND_FRAG_CONNECT];
+ creplicaFileSize = signal->theData[DihSizeAltReq::IND_REPLICAS];
+ ctabFileSize = signal->theData[DihSizeAltReq::IND_TABLE];
+ cfileFileSize = (2 * ctabFileSize) + 2;
+ initRecords();
+ initialiseRecordsLab(signal, 0);
+ return;
+}//Dbdih::execSIZEALT_REP()
+
+void Dbdih::execSTART_COPYREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dbdih::execSTART_COPYREF()
+
+void Dbdih::execSTART_FRAGCONF(Signal* signal)
+{
+ (void)signal; // Don't want compiler warning
+ /* ********************************************************************* */
+ /* If anyone wants to add functionality in this method, be aware that */
+ /* for temporary tables no START_FRAGREQ is sent and therefore no */
+ /* START_FRAGCONF signal will be received for those tables!! */
+ /* ********************************************************************* */
+ jamEntry();
+ return;
+}//Dbdih::execSTART_FRAGCONF()
+
+void Dbdih::execSTART_MEREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dbdih::execSTART_MEREF()
+
+void Dbdih::execTAB_COMMITREQ(Signal* signal)
+{
+ TabRecordPtr tabPtr;
+ jamEntry();
+ Uint32 tdictPtr = signal->theData[0];
+ BlockReference tdictBlockref = signal->theData[1];
+ tabPtr.i = signal->theData[2];
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ ndbrequire(tabPtr.p->tabStatus == TabRecord::TS_CREATING);
+ tabPtr.p->tabStatus = TabRecord::TS_ACTIVE;
+ signal->theData[0] = tdictPtr;
+ signal->theData[1] = cownNodeId;
+ signal->theData[2] = tabPtr.i;
+ sendSignal(tdictBlockref, GSN_TAB_COMMITCONF, signal, 3, JBB);
+ return;
+}//Dbdih::execTAB_COMMITREQ()
+
+/*
+ 3.2 S T A N D A R D S U B P R O G R A M S I N P L E X
+ *************************************************************
+ */
+/*
+ 3.2.1 S T A R T / R E S T A R T
+ **********************************
+ */
+/*****************************************************************************/
+/* ********** START / RESTART MODULE *************/
+/*****************************************************************************/
+/*
+ 3.2.1.1 LOADING O W N B L O C K R E F E R E N C E (ABSOLUTE PHASE 1)
+ *****************************************************************************
+ */
+void Dbdih::execDIH_RESTARTREQ(Signal* signal)
+{
+ jamEntry();
+ cntrlblockref = signal->theData[0];
+ if(theConfiguration.getInitialStart()){
+ sendSignal(cntrlblockref, GSN_DIH_RESTARTREF, signal, 1, JBB);
+ } else {
+ readGciFileLab(signal);
+ }
+ return;
+}//Dbdih::execDIH_RESTARTREQ()
+
+void Dbdih::execSTTOR(Signal* signal)
+{
+ jamEntry();
+
+ signal->theData[0] = 0;
+ signal->theData[1] = 0;
+ signal->theData[2] = 0;
+ signal->theData[3] = 1; // Next start phase
+ signal->theData[4] = 255; // Next start phase
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 5, JBB);
+ return;
+}//Dbdih::execSTTOR()
+
+void Dbdih::initialStartCompletedLab(Signal* signal)
+{
+ /*-------------------------------------------------------------------------*/
+ /* NOW THAT (RE)START IS COMPLETED WE CAN START THE LCP.*/
+ /*-------------------------------------------------------------------------*/
+ return;
+}//Dbdih::initialStartCompletedLab()
+
+/*
+ * ***************************************************************************
+ * S E N D I N G R E P L Y T O S T A R T / R E S T A R T R E Q U E S T S
+ * ****************************************************************************
+ */
+void Dbdih::ndbsttorry10Lab(Signal* signal, Uint32 _line)
+{
+ /*-------------------------------------------------------------------------*/
+ // AN NDB START PHASE HAS BEEN COMPLETED. WHEN START PHASE 6 IS COMPLETED WE
+ // RECORD THAT THE SYSTEM IS RUNNING.
+ /*-------------------------------------------------------------------------*/
+ signal->theData[0] = reference();
+ sendSignal(cntrlblockref, GSN_NDB_STTORRY, signal, 1, JBB);
+ return;
+}//Dbdih::ndbsttorry10Lab()
+
+/*
+****************************************
+I N T E R N A L P H A S E S
+****************************************
+*/
+/*---------------------------------------------------------------------------*/
+/*NDB_STTOR START SIGNAL AT START/RESTART */
+/*---------------------------------------------------------------------------*/
+void Dbdih::execNDB_STTOR(Signal* signal)
+{
+ jamEntry();
+ BlockReference cntrRef = signal->theData[0]; /* SENDERS BLOCK REFERENCE */
+ Uint32 ownNodeId = signal->theData[1]; /* OWN PROCESSOR ID*/
+ Uint32 phase = signal->theData[2]; /* INTERNAL START PHASE*/
+ Uint32 typestart = signal->theData[3];
+ const Uint32 tconfig1 = signal->theData[8];
+ const Uint32 tconfig2 = signal->theData[9];
+
+ cstarttype = typestart;
+
+ cstartPhase = phase;
+
+ switch (phase){
+ case ZNDB_SPH1:
+ jam();
+ /*----------------------------------------------------------------------*/
+ /* Set the delay between local checkpoints in ndb startphase 1. */
+ /*----------------------------------------------------------------------*/
+ c_lcpState.clcpDelay = tconfig1 > 31 ? 31 : tconfig1;
+
+ cminHotSpareNodes = tconfig2 > 2 ? 2 : tconfig2;
+
+ cownNodeId = ownNodeId;
+ /*-----------------------------------------------------------------------*/
+ // Compute all static block references in this node as part of
+ // ndb start phase 1.
+ /*-----------------------------------------------------------------------*/
+ cntrlblockref = cntrRef;
+ clocaltcblockref = calcTcBlockRef(ownNodeId);
+ clocallqhblockref = calcLqhBlockRef(ownNodeId);
+ cdictblockref = calcDictBlockRef(ownNodeId);
+ ndbsttorry10Lab(signal, __LINE__);
+ break;
+
+ case ZNDB_SPH2:
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // Set the number of replicas, maximum is 4 replicas.
+ // Read the ndb nodes from the configuration.
+ /*-----------------------------------------------------------------------*/
+ cnoReplicas = tconfig1 > 4 ? 4 : tconfig1;
+ cgcpDelay = tconfig2 > 60000 ? 60000 : (tconfig2 < 10 ? 10 : tconfig2);
+
+ /*-----------------------------------------------------------------------*/
+ // For node restarts we will also add a request for permission
+ // to continue the system restart.
+ // The permission is given by the master node in the alive set.
+ /*-----------------------------------------------------------------------*/
+ createMutexes(signal, 0);
+ break;
+
+ case ZNDB_SPH3:
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // Non-master nodes performing an initial start will execute
+ // the start request here since the
+ // initial start do not synchronise so much from the master.
+ // In the master nodes the start
+ // request will be sent directly to dih (in ndb_startreq) when all
+ // nodes have completed phase 3 of the start.
+ /*-----------------------------------------------------------------------*/
+ cmasterState = MASTER_IDLE;
+ if(cstarttype == NodeState::ST_INITIAL_START ||
+ cstarttype == NodeState::ST_SYSTEM_RESTART){
+ jam();
+ cmasterState = isMaster() ? MASTER_ACTIVE : MASTER_IDLE;
+ }
+ if (!isMaster() && cstarttype == NodeState::ST_INITIAL_START) {
+ jam();
+ ndbStartReqLab(signal, cntrRef);
+ return;
+ }//if
+ ndbsttorry10Lab(signal, __LINE__);
+ break;
+
+ case ZNDB_SPH4:
+ jam();
+ c_lcpState.setLcpStatus(LCP_STATUS_IDLE, __LINE__);
+ cmasterTakeOverNode = ZNIL;
+ switch(typestart){
+ case NodeState::ST_INITIAL_START:
+ jam();
+ ndbsttorry10Lab(signal, __LINE__);
+ return;
+ case NodeState::ST_SYSTEM_RESTART:
+ jam();
+ if (isMaster()) {
+ jam();
+ systemRestartTakeOverLab(signal);
+ if (anyActiveTakeOver() && false) {
+ jam();
+ ndbout_c("1 - anyActiveTakeOver == true");
+ return;
+ }
+ }
+ ndbsttorry10Lab(signal, __LINE__);
+ return;
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ case NodeState::ST_NODE_RESTART:
+ jam();
+ /***********************************************************************
+ * When starting nodes while system is operational we must be controlled
+ * by the master since only one node restart is allowed at a time.
+ * When this signal is confirmed the master has also copied the
+ * dictionary and the distribution information.
+ */
+ StartMeReq * req = (StartMeReq*)&signal->theData[0];
+ req->startingRef = reference();
+ req->startingVersion = NDB_VERSION;
+ sendSignal(cmasterdihref, GSN_START_MEREQ, signal,
+ StartMeReq::SignalLength, JBB);
+ return;
+ }
+ ndbrequire(false);
+ break;
+ case ZNDB_SPH5:
+ jam();
+ switch(typestart){
+ case NodeState::ST_INITIAL_START:
+ case NodeState::ST_SYSTEM_RESTART:
+ jam();
+ jam();
+ /*---------------------------------------------------------------------*/
+ // WE EXECUTE A LOCAL CHECKPOINT AS A PART OF A SYSTEM RESTART.
+ // THE IDEA IS THAT WE NEED TO
+ // ENSURE THAT WE CAN RECOVER FROM PROBLEMS CAUSED BY MANY NODE
+ // CRASHES THAT CAUSES THE LOG
+ // TO GROW AND THE NUMBER OF LOG ROUNDS TO EXECUTE TO GROW.
+ // THIS CAN OTHERWISE GET US INTO
+ // A SITUATION WHICH IS UNREPAIRABLE. THUS WE EXECUTE A CHECKPOINT
+ // BEFORE ALLOWING ANY TRANSACTIONS TO START.
+ /*---------------------------------------------------------------------*/
+ if (!isMaster()) {
+ jam();
+ ndbsttorry10Lab(signal, __LINE__);
+ return;
+ }//if
+
+ c_lcpState.immediateLcpStart = true;
+ cwaitLcpSr = true;
+ checkLcpStart(signal, __LINE__);
+ return;
+ case NodeState::ST_NODE_RESTART:
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ jam();
+ signal->theData[0] = cownNodeId;
+ signal->theData[1] = reference();
+ sendSignal(cmasterdihref, GSN_START_COPYREQ, signal, 2, JBB);
+ return;
+ }
+ ndbrequire(false);
+ case ZNDB_SPH6:
+ jam();
+ switch(typestart){
+ case NodeState::ST_INITIAL_START:
+ case NodeState::ST_SYSTEM_RESTART:
+ jam();
+ if(isMaster()){
+ jam();
+ startGcp(signal);
+ }
+ ndbsttorry10Lab(signal, __LINE__);
+ return;
+ case NodeState::ST_NODE_RESTART:
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ ndbsttorry10Lab(signal, __LINE__);
+ return;
+ }
+ ndbrequire(false);
+ break;
+ default:
+ jam();
+ ndbsttorry10Lab(signal, __LINE__);
+ break;
+ }//switch
+}//Dbdih::execNDB_STTOR()
+
+void
+Dbdih::createMutexes(Signal * signal, Uint32 count){
+ Callback c = { safe_cast(&Dbdih::createMutex_done), count };
+
+ switch(count){
+ case 0:{
+ Mutex mutex(signal, c_mutexMgr, c_startLcpMutexHandle);
+ mutex.create(c);
+ return;
+ }
+ case 1:{
+ Mutex mutex(signal, c_mutexMgr, c_switchPrimaryMutexHandle);
+ mutex.create(c);
+ return;
+ }
+ }
+
+ signal->theData[0] = reference();
+ sendSignal(cntrlblockref, GSN_READ_NODESREQ, signal, 1, JBB);
+}
+
+void
+Dbdih::createMutex_done(Signal* signal, Uint32 senderData, Uint32 retVal){
+ jamEntry();
+ ndbrequire(retVal == 0);
+
+ switch(senderData){
+ case 0:{
+ Mutex mutex(signal, c_mutexMgr, c_startLcpMutexHandle);
+ mutex.release();
+ }
+ case 1:{
+ Mutex mutex(signal, c_mutexMgr, c_switchPrimaryMutexHandle);
+ mutex.release();
+ }
+ }
+
+ createMutexes(signal, senderData + 1);
+}
+
+/*****************************************************************************/
+/* ------------------------------------------------------------------------- */
+/* WE HAVE BEEN REQUESTED BY NDBCNTR TO PERFORM A RESTART OF THE */
+/* DATABASE TABLES. */
+/* THIS SIGNAL IS SENT AFTER COMPLETING PHASE 3 IN ALL BLOCKS IN A */
+/* SYSTEM RESTART. WE WILL ALSO JUMP TO THIS LABEL FROM PHASE 3 IN AN */
+/* INITIAL START. */
+/* ------------------------------------------------------------------------- */
+/*****************************************************************************/
+void Dbdih::execNDB_STARTREQ(Signal* signal)
+{
+ jamEntry();
+ BlockReference ref = signal->theData[0];
+ cstarttype = signal->theData[1];
+ ndbStartReqLab(signal, ref);
+}//Dbdih::execNDB_STARTREQ()
+
+void Dbdih::ndbStartReqLab(Signal* signal, BlockReference ref)
+{
+ cndbStartReqBlockref = ref;
+ if (cstarttype == NodeState::ST_INITIAL_START) {
+ jam();
+ initRestartInfo();
+ initGciFilesLab(signal);
+ return;
+ }
+
+ ndbrequire(isMaster());
+ copyGciLab(signal, CopyGCIReq::RESTART); // We have already read the file!
+}//Dbdih::ndbStartReqLab()
+
+void Dbdih::execREAD_NODESCONF(Signal* signal)
+{
+ ReadNodesConf * const readNodes = (ReadNodesConf *)&signal->theData[0];
+ jamEntry();
+ Uint32 nodeArray[MAX_NDB_NODES];
+
+ csystemnodes = readNodes->noOfNodes;
+ cmasterNodeId = readNodes->masterNodeId;
+ int index = 0;
+ for (unsigned i = 1; i < MAX_NDB_NODES; i++){
+ jam();
+ if(NodeBitmask::get(readNodes->allNodes, i)){
+ jam();
+ nodeArray[index] = i;
+ if(NodeBitmask::get(readNodes->inactiveNodes, i) == false){
+ jam();
+ con_lineNodes++;
+ }//if
+ index++;
+ }//if
+ }//for
+ ndbrequire(csystemnodes >= 1 && csystemnodes < MAX_NDB_NODES);
+ if (cstarttype == NodeState::ST_INITIAL_START) {
+ jam();
+ ndbrequire(cnoReplicas <= csystemnodes);
+ calculateHotSpare();
+ ndbrequire(cnoReplicas <= (csystemnodes - cnoHotSpare));
+ }//if
+
+ cmasterdihref = calcDihBlockRef(cmasterNodeId);
+ /*-------------------------------------------------------------------------*/
+ /* MAKE THE LIST OF PRN-RECORD WHICH IS ONE OF THE NODES-LIST IN THIS BLOCK*/
+ /*-------------------------------------------------------------------------*/
+ makePrnList(readNodes, nodeArray);
+ if (cstarttype == NodeState::ST_INITIAL_START) {
+ jam();
+ /**----------------------------------------------------------------------
+ * WHEN WE INITIALLY START A DATABASE WE WILL CREATE NODE GROUPS.
+ * ALL NODES ARE PUT INTO NODE GROUPS ALTHOUGH HOT SPARE NODES ARE PUT
+ * INTO A SPECIAL NODE GROUP. IN EACH NODE GROUP WE HAVE THE SAME AMOUNT
+ * OF NODES AS THERE ARE NUMBER OF REPLICAS.
+ * ONE POSSIBLE USAGE OF NODE GROUPS ARE TO MAKE A NODE GROUP A COMPLETE
+ * FRAGMENT OF THE DATABASE. THIS MEANS THAT ALL REPLICAS WILL BE STORED
+ * IN THE NODE GROUP.
+ *-----------------------------------------------------------------------*/
+ makeNodeGroups(nodeArray);
+ }//if
+ ndbrequire(checkNodeAlive(cmasterNodeId));
+ if (cstarttype == NodeState::ST_INITIAL_START) {
+ jam();
+ /**-----------------------------------------------------------------------
+ * INITIALISE THE SECOND NODE-LIST AND SET NODE BITS AND SOME NODE STATUS.
+ * VERY CONNECTED WITH MAKE_NODE_GROUPS. CHANGING ONE WILL AFFECT THE
+ * OTHER AS WELL.
+ *-----------------------------------------------------------------------*/
+ setInitialActiveStatus();
+ } else if (cstarttype == NodeState::ST_SYSTEM_RESTART) {
+ jam();
+ /*empty*/;
+ } else if ((cstarttype == NodeState::ST_NODE_RESTART) ||
+ (cstarttype == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+ nodeRestartPh2Lab(signal);
+ return;
+ } else {
+ ndbrequire(false);
+ }//if
+ /**------------------------------------------------------------------------
+ * ESTABLISH CONNECTIONS WITH THE OTHER DIH BLOCKS AND INITIALISE THIS
+ * NODE-LIST THAT HANDLES CONNECTION WITH OTHER DIH BLOCKS.
+ *-------------------------------------------------------------------------*/
+ ndbsttorry10Lab(signal, __LINE__);
+}//Dbdih::execREAD_NODESCONF()
+
+/*---------------------------------------------------------------------------*/
+/* START NODE LOGIC FOR NODE RESTART */
+/*---------------------------------------------------------------------------*/
+void Dbdih::nodeRestartPh2Lab(Signal* signal)
+{
+ /*------------------------------------------------------------------------*/
+ // REQUEST FOR PERMISSION FROM MASTER TO START A NODE IN AN ALREADY
+ // RUNNING SYSTEM.
+ /*------------------------------------------------------------------------*/
+ StartPermReq * const req = (StartPermReq *)&signal->theData[0];
+
+ req->blockRef = reference();
+ req->nodeId = cownNodeId;
+ req->startType = cstarttype;
+ sendSignal(cmasterdihref, GSN_START_PERMREQ, signal, 3, JBB);
+}//Dbdih::nodeRestartPh2Lab()
+
+void Dbdih::execSTART_PERMCONF(Signal* signal)
+{
+ jamEntry();
+ CRASH_INSERTION(7121);
+ Uint32 nodeId = signal->theData[0];
+ cfailurenr = signal->theData[1];
+ ndbrequire(nodeId == cownNodeId);
+ ndbsttorry10Lab(signal, __LINE__);
+}//Dbdih::execSTART_PERMCONF()
+
+void Dbdih::execSTART_PERMREF(Signal* signal)
+{
+ jamEntry();
+ Uint32 errorCode = signal->theData[1];
+ if (errorCode == ZNODE_ALREADY_STARTING_ERROR) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // The master was busy adding another node. We will wait for a second and
+ // try again.
+ /*-----------------------------------------------------------------------*/
+ signal->theData[0] = DihContinueB::ZSTART_PERMREQ_AGAIN;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 3000, 1);
+ return;
+ }//if
+ /*------------------------------------------------------------------------*/
+ // Some node process in another node involving our node was still active. We
+ // will recover from this by crashing here.
+ // This is controlled restart using the
+ // already existing features of node crashes. It is not a bug getting here.
+ /*-------------------------------------------------------------------------*/
+ ndbrequire(false);
+ return;
+}//Dbdih::execSTART_PERMREF()
+
+/*---------------------------------------------------------------------------*/
+/* THIS SIGNAL IS RECEIVED IN THE STARTING NODE WHEN THE START_MEREQ */
+/* HAS BEEN EXECUTED IN THE MASTER NODE. */
+/*---------------------------------------------------------------------------*/
+void Dbdih::execSTART_MECONF(Signal* signal)
+{
+ jamEntry();
+ StartMeConf * const startMe = (StartMeConf *)&signal->theData[0];
+ Uint32 nodeId = startMe->startingNodeId;
+ const Uint32 startWord = startMe->startWord;
+
+ CRASH_INSERTION(7130);
+ ndbrequire(nodeId == cownNodeId);
+ arrGuard(startWord + StartMeConf::DATA_SIZE, sizeof(cdata)/4);
+ for(Uint32 i = 0; i < StartMeConf::DATA_SIZE; i++)
+ cdata[startWord+i] = startMe->data[i];
+
+ if(startWord + StartMeConf::DATA_SIZE < Sysfile::SYSFILE_SIZE32){
+ jam();
+ /**
+ * We are still waiting for data
+ */
+ return;
+ }
+ jam();
+
+ /**
+ * Copy into sysfile
+ *
+ * But dont copy lastCompletedGCI:s
+ */
+ Uint32 tempGCP[MAX_NDB_NODES];
+ for(Uint32 i = 0; i < MAX_NDB_NODES; i++)
+ tempGCP[i] = SYSFILE->lastCompletedGCI[i];
+
+ for(Uint32 i = 0; i < Sysfile::SYSFILE_SIZE32; i++)
+ sysfileData[i] = cdata[i];
+ for(Uint32 i = 0; i < MAX_NDB_NODES; i++)
+ SYSFILE->lastCompletedGCI[i] = tempGCP[i];
+
+ setNodeActiveStatus();
+ setNodeGroups();
+ ndbsttorry10Lab(signal, __LINE__);
+}//Dbdih::execSTART_MECONF()
+
+void Dbdih::execSTART_COPYCONF(Signal* signal)
+{
+ jamEntry();
+ Uint32 nodeId = signal->theData[0];
+ ndbrequire(nodeId == cownNodeId);
+ CRASH_INSERTION(7132);
+ ndbsttorry10Lab(signal, __LINE__);
+ return;
+}//Dbdih::execSTART_COPYCONF()
+
+/*---------------------------------------------------------------------------*/
+/* MASTER LOGIC FOR NODE RESTART */
+/*---------------------------------------------------------------------------*/
+/* NODE RESTART PERMISSION REQUEST */
+/*---------------------------------------------------------------------------*/
+// A REQUEST FROM A STARTING NODE TO PERFORM A NODE RESTART. IF NO OTHER NODE
+// IS ACTIVE IN PERFORMING A NODE RESTART AND THERE ARE NO ACTIVE PROCESSES IN
+// THIS NODE INVOLVING THE STARTING NODE THIS REQUEST WILL BE GRANTED.
+/*---------------------------------------------------------------------------*/
+void Dbdih::execSTART_PERMREQ(Signal* signal)
+{
+ StartPermReq * const req = (StartPermReq*)&signal->theData[0];
+ jamEntry();
+ const BlockReference retRef = req->blockRef;
+ const Uint32 nodeId = req->nodeId;
+ const Uint32 typeStart = req->startType;
+
+ CRASH_INSERTION(7122);
+ ndbrequire(isMaster());
+ ndbrequire(refToNode(retRef) == nodeId);
+ if ((c_nodeStartMaster.activeState) ||
+ (c_nodeStartMaster.wait != ZFALSE)) {
+ jam();
+ signal->theData[0] = nodeId;
+ signal->theData[1] = ZNODE_ALREADY_STARTING_ERROR;
+ sendSignal(retRef, GSN_START_PERMREF, signal, 2, JBB);
+ return;
+ }//if
+ if (getNodeStatus(nodeId) != NodeRecord::DEAD){
+ ndbout << "nodeStatus in START_PERMREQ = "
+ << getNodeStatus(nodeId) << endl;
+ ndbrequire(false);
+ }//if
+
+ /*----------------------------------------------------------------------
+ * WE START THE INCLUSION PROCEDURE
+ * ---------------------------------------------------------------------*/
+ c_nodeStartMaster.failNr = cfailurenr;
+ c_nodeStartMaster.wait = ZFALSE;
+ c_nodeStartMaster.startInfoErrorCode = 0;
+ c_nodeStartMaster.startNode = nodeId;
+ c_nodeStartMaster.activeState = true;
+
+ setNodeStatus(nodeId, NodeRecord::STARTING);
+ /**
+ * But if it's a NodeState::ST_INITIAL_NODE_RESTART
+ *
+ * We first have to clear LCP's
+ * For normal node restart we simply ensure that all nodes
+ * are informed of the node restart
+ */
+ StartInfoReq *const r =(StartInfoReq*)&signal->theData[0];
+ r->startingNodeId = nodeId;
+ r->typeStart = typeStart;
+ r->systemFailureNo = cfailurenr;
+ sendLoopMacro(START_INFOREQ, sendSTART_INFOREQ);
+}//Dbdih::execSTART_PERMREQ()
+
+void Dbdih::execSTART_INFOREF(Signal* signal)
+{
+ StartInfoRef * ref = (StartInfoRef*)&signal->theData[0];
+ if (getNodeStatus(ref->startingNodeId) != NodeRecord::STARTING) {
+ jam();
+ return;
+ }//if
+ ndbrequire(c_nodeStartMaster.startNode == ref->startingNodeId);
+ c_nodeStartMaster.startInfoErrorCode = ref->errorCode;
+ startInfoReply(signal, ref->sendingNodeId);
+}//Dbdih::execSTART_INFOREF()
+
+void Dbdih::execSTART_INFOCONF(Signal* signal)
+{
+ jamEntry();
+ StartInfoConf * conf = (StartInfoConf*)&signal->theData[0];
+ if (getNodeStatus(conf->startingNodeId) != NodeRecord::STARTING) {
+ jam();
+ return;
+ }//if
+ ndbrequire(c_nodeStartMaster.startNode == conf->startingNodeId);
+ startInfoReply(signal, conf->sendingNodeId);
+}//Dbdih::execSTART_INFOCONF()
+
+void Dbdih::startInfoReply(Signal* signal, Uint32 nodeId)
+{
+ receiveLoopMacro(START_INFOREQ, nodeId);
+ /**
+ * We're finished with the START_INFOREQ's
+ */
+ if (c_nodeStartMaster.startInfoErrorCode == 0) {
+ jam();
+ /**
+ * Everything has been a success so far
+ */
+ StartPermConf * conf = (StartPermConf*)&signal->theData[0];
+ conf->startingNodeId = c_nodeStartMaster.startNode;
+ conf->systemFailureNo = cfailurenr;
+ sendSignal(calcDihBlockRef(c_nodeStartMaster.startNode),
+ GSN_START_PERMCONF, signal, StartPermConf::SignalLength, JBB);
+ c_nodeStartMaster.m_outstandingGsn = GSN_START_PERMCONF;
+ } else {
+ jam();
+ StartPermRef * ref = (StartPermRef*)&signal->theData[0];
+ ref->startingNodeId = c_nodeStartMaster.startNode;
+ ref->errorCode = c_nodeStartMaster.startInfoErrorCode;
+ sendSignal(calcDihBlockRef(c_nodeStartMaster.startNode),
+ GSN_START_PERMREF, signal, StartPermRef::SignalLength, JBB);
+ nodeResetStart();
+ }//if
+}//Dbdih::startInfoReply()
+
+/*---------------------------------------------------------------------------*/
+/* NODE RESTART CONTINUE REQUEST */
+/*---------------------------------------------------------------------------*/
+// THIS SIGNAL AND THE CODE BELOW IS EXECUTED BY THE MASTER WHEN IT HAS BEEN
+// REQUESTED TO START UP A NEW NODE. The master instructs the starting node
+// how to set up its log for continued execution.
+/*---------------------------------------------------------------------------*/
+void Dbdih::execSTART_MEREQ(Signal* signal)
+{
+ StartMeReq * req = (StartMeReq*)&signal->theData[0];
+ jamEntry();
+ const BlockReference Tblockref = req->startingRef;
+ const Uint32 Tnodeid = refToNode(Tblockref);
+ const Uint32 TndbVersion = req->startingVersion;
+
+ ndbrequire(isMaster());
+ ndbrequire(c_nodeStartMaster.startNode == Tnodeid);
+ ndbrequire(getNodeStatus(Tnodeid) == NodeRecord::STARTING);
+
+ c_nodeStartMaster.ndbVersion = TndbVersion;
+ sendSTART_RECREQ(signal, Tnodeid);
+}//Dbdih::execSTART_MEREQ()
+
+void Dbdih::nodeRestartStartRecConfLab(Signal* signal)
+{
+ c_nodeStartMaster.blockLcp = true;
+ if ((c_lcpState.lcpStatus != LCP_STATUS_IDLE) &&
+ (c_lcpState.lcpStatus != LCP_TCGET)) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // WE WILL NOT ALLOW A NODE RESTART TO COME IN WHEN A LOCAL CHECKPOINT IS
+ // ONGOING. IT WOULD COMPLICATE THE LCP PROTOCOL TOO MUCH. WE WILL ADD THIS
+ // LATER.
+ /*-----------------------------------------------------------------------*/
+ return;
+ }//if
+ lcpBlockedLab(signal);
+}//Dbdih::nodeRestartStartRecConfLab()
+
+void Dbdih::lcpBlockedLab(Signal* signal)
+{
+ ndbrequire(getNodeStatus(c_nodeStartMaster.startNode)==NodeRecord::STARTING);
+ /*------------------------------------------------------------------------*/
+ // NOW WE HAVE COPIED ALL INFORMATION IN DICT WE ARE NOW READY TO COPY ALL
+ // INFORMATION IN DIH TO THE NEW NODE.
+ /*------------------------------------------------------------------------*/
+ c_nodeStartMaster.wait = 10;
+ signal->theData[0] = DihContinueB::ZCOPY_NODE;
+ signal->theData[1] = 0;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ c_nodeStartMaster.m_outstandingGsn = GSN_COPY_TABREQ;
+}//Dbdih::lcpBlockedLab()
+
+void Dbdih::nodeDictStartConfLab(Signal* signal)
+{
+ /*-------------------------------------------------------------------------*/
+ // NOW WE HAVE COPIED BOTH DIH AND DICT INFORMATION. WE ARE NOW READY TO
+ // INTEGRATE THE NODE INTO THE LCP AND GCP PROTOCOLS AND TO ALLOW UPDATES OF
+ // THE DICTIONARY AGAIN.
+ /*-------------------------------------------------------------------------*/
+ c_nodeStartMaster.wait = ZFALSE;
+ c_nodeStartMaster.blockGcp = true;
+ if (cgcpStatus != GCP_READY) {
+ /*-----------------------------------------------------------------------*/
+ // The global checkpoint is executing. Wait until it is completed before we
+ // continue processing the node recovery.
+ /*-----------------------------------------------------------------------*/
+ jam();
+ return;
+ }//if
+ gcpBlockedLab(signal);
+
+ /*-----------------------------------------------------------------*/
+ // Report that node restart has completed copy of dictionary.
+ /*-----------------------------------------------------------------*/
+ signal->theData[0] = EventReport::NR_CopyDict;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 1, JBB);
+}//Dbdih::nodeDictStartConfLab()
+
+void Dbdih::dihCopyCompletedLab(Signal* signal)
+{
+ BlockReference ref = calcDictBlockRef(c_nodeStartMaster.startNode);
+ DictStartReq * req = (DictStartReq*)&signal->theData[0];
+ req->restartGci = cnewgcp;
+ req->senderRef = reference();
+ sendSignal(ref, GSN_DICTSTARTREQ,
+ signal, DictStartReq::SignalLength, JBB);
+ c_nodeStartMaster.m_outstandingGsn = GSN_DICTSTARTREQ;
+ c_nodeStartMaster.wait = 0;
+}//Dbdih::dihCopyCompletedLab()
+
+void Dbdih::gcpBlockedLab(Signal* signal)
+{
+ /*-----------------------------------------------------------------*/
+ // Report that node restart has completed copy of distribution info.
+ /*-----------------------------------------------------------------*/
+ signal->theData[0] = EventReport::NR_CopyDistr;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 1, JBB);
+
+ /**
+ * The node DIH will be part of LCP
+ */
+ NodeRecordPtr nodePtr;
+ nodePtr.i = c_nodeStartMaster.startNode;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ nodePtr.p->m_inclDihLcp = true;
+
+ /*-------------------------------------------------------------------------*/
+ // NOW IT IS TIME TO INFORM ALL OTHER NODES IN THE CLUSTER OF THE STARTED
+ // NODE SUCH THAT THEY ALSO INCLUDE THE NODE IN THE NODE LISTS AND SO FORTH.
+ /*------------------------------------------------------------------------*/
+ sendLoopMacro(INCL_NODEREQ, sendINCL_NODEREQ);
+ /*-------------------------------------------------------------------------*/
+ // We also need to send to the starting node to ensure he is aware of the
+ // global checkpoint id and the correct state. We do not wait for any reply
+ // since the starting node will not send any.
+ /*-------------------------------------------------------------------------*/
+ sendINCL_NODEREQ(signal, c_nodeStartMaster.startNode);
+}//Dbdih::gcpBlockedLab()
+
+/*---------------------------------------------------------------------------*/
+// THIS SIGNAL IS EXECUTED IN BOTH SLAVES AND IN THE MASTER
+/*---------------------------------------------------------------------------*/
+void Dbdih::execINCL_NODECONF(Signal* signal)
+{
+ Uint32 TsendNodeId;
+ Uint32 TstartNode_or_blockref;
+
+ jamEntry();
+ TstartNode_or_blockref = signal->theData[0];
+ TsendNodeId = signal->theData[1];
+
+ if (TstartNode_or_blockref == clocallqhblockref) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // THIS SIGNAL CAME FROM THE LOCAL LQH BLOCK.
+ // WE WILL NOW SEND INCLUDE TO THE TC BLOCK.
+ /*-----------------------------------------------------------------------*/
+ signal->theData[0] = reference();
+ signal->theData[1] = c_nodeStartSlave.nodeId;
+ sendSignal(clocaltcblockref, GSN_INCL_NODEREQ, signal, 2, JBB);
+ return;
+ }//if
+ if (TstartNode_or_blockref == clocaltcblockref) {
+ jam();
+ /*----------------------------------------------------------------------*/
+ // THIS SIGNAL CAME FROM THE LOCAL LQH BLOCK.
+ // WE WILL NOW SEND INCLUDE TO THE DICT BLOCK.
+ /*----------------------------------------------------------------------*/
+ signal->theData[0] = reference();
+ signal->theData[1] = c_nodeStartSlave.nodeId;
+ sendSignal(cdictblockref, GSN_INCL_NODEREQ, signal, 2, JBB);
+ return;
+ }//if
+ if (TstartNode_or_blockref == cdictblockref) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // THIS SIGNAL CAME FROM THE LOCAL DICT BLOCK. WE WILL NOW SEND CONF TO THE
+ // BACKUP.
+ /*-----------------------------------------------------------------------*/
+ signal->theData[0] = reference();
+ signal->theData[1] = c_nodeStartSlave.nodeId;
+ sendSignal(BACKUP_REF, GSN_INCL_NODEREQ, signal, 2, JBB);
+
+ // Suma will not send response to this for now, later...
+ sendSignal(SUMA_REF, GSN_INCL_NODEREQ, signal, 2, JBB);
+ // Grep will not send response to this for now, later...
+ sendSignal(GREP_REF, GSN_INCL_NODEREQ, signal, 2, JBB);
+ return;
+ }//if
+ if (TstartNode_or_blockref == numberToRef(BACKUP, getOwnNodeId())){
+ jam();
+ signal->theData[0] = c_nodeStartSlave.nodeId;
+ signal->theData[1] = cownNodeId;
+ sendSignal(cmasterdihref, GSN_INCL_NODECONF, signal, 2, JBB);
+ c_nodeStartSlave.nodeId = 0;
+ return;
+ }
+
+ ndbrequire(cmasterdihref = reference());
+ receiveLoopMacro(INCL_NODEREQ, TsendNodeId);
+
+ CRASH_INSERTION(7128);
+ /*-------------------------------------------------------------------------*/
+ // Now that we have included the starting node in the node lists in the
+ // various blocks we are ready to start the global checkpoint protocol
+ /*------------------------------------------------------------------------*/
+ c_nodeStartMaster.wait = 11;
+ c_nodeStartMaster.blockGcp = false;
+
+ signal->theData[0] = reference();
+ sendSignal(reference(), GSN_UNBLO_DICTCONF, signal, 1, JBB);
+}//Dbdih::execINCL_NODECONF()
+
+void Dbdih::execUNBLO_DICTCONF(Signal* signal)
+{
+ jamEntry();
+ c_nodeStartMaster.wait = ZFALSE;
+ if (!c_nodeStartMaster.activeState) {
+ jam();
+ return;
+ }//if
+
+ CRASH_INSERTION(7129);
+ /**-----------------------------------------------------------------------
+ * WE HAVE NOW PREPARED IT FOR INCLUSION IN THE LCP PROTOCOL.
+ * WE CAN NOW START THE LCP PROTOCOL AGAIN.
+ * WE HAVE ALSO MADE THIS FOR THE GCP PROTOCOL.
+ * WE ARE READY TO START THE PROTOCOLS AND RESPOND TO THE START REQUEST
+ * FROM THE STARTING NODE.
+ *------------------------------------------------------------------------*/
+
+ StartMeConf * const startMe = (StartMeConf *)&signal->theData[0];
+
+ const Uint32 wordPerSignal = StartMeConf::DATA_SIZE;
+ const int noOfSignals = ((Sysfile::SYSFILE_SIZE32 + (wordPerSignal - 1)) /
+ wordPerSignal);
+
+ startMe->startingNodeId = c_nodeStartMaster.startNode;
+ startMe->startWord = 0;
+
+ const Uint32 ref = calcDihBlockRef(c_nodeStartMaster.startNode);
+ for(int i = 0; i < noOfSignals; i++){
+ jam();
+ { // Do copy
+ const int startWord = startMe->startWord;
+ for(Uint32 j = 0; j < wordPerSignal; j++){
+ startMe->data[j] = sysfileData[j+startWord];
+ }
+ }
+ sendSignal(ref, GSN_START_MECONF, signal, StartMeConf::SignalLength, JBB);
+ startMe->startWord += wordPerSignal;
+ }//for
+ c_nodeStartMaster.m_outstandingGsn = GSN_START_MECONF;
+}//Dbdih::execUNBLO_DICTCONF()
+
+/*---------------------------------------------------------------------------*/
+/* NODE RESTART COPY REQUEST */
+/*---------------------------------------------------------------------------*/
+// A NODE RESTART HAS REACHED ITS FINAL PHASE WHEN THE DATA IS TO BE COPIED
+// TO THE NODE. START_COPYREQ IS EXECUTED BY THE MASTER NODE.
+/*---------------------------------------------------------------------------*/
+void Dbdih::execSTART_COPYREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 startNodeId = signal->theData[0];
+ //BlockReference startingRef = signal->theData[1];
+ ndbrequire(c_nodeStartMaster.startNode == startNodeId);
+ /*-------------------------------------------------------------------------*/
+ // REPORT Copy process of node restart is now about to start up.
+ /*-------------------------------------------------------------------------*/
+ signal->theData[0] = EventReport::NR_CopyFragsStarted;
+ signal->theData[1] = startNodeId;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ CRASH_INSERTION(7131);
+ nodeRestartTakeOver(signal, startNodeId);
+ // BlockReference ref = calcQmgrBlockRef(startNodeId);
+ // signal->theData[0] = cownNodeId;
+ // Remove comments as soon as I open up the Qmgr block
+ // TODO_RONM
+ // sendSignal(ref, GSN_ALLOW_NODE_CRASHORD, signal, 1, JBB);
+}//Dbdih::execSTART_COPYREQ()
+
+/*---------------------------------------------------------------------------*/
+/* SLAVE LOGIC FOR NODE RESTART */
+/*---------------------------------------------------------------------------*/
+void Dbdih::execSTART_INFOREQ(Signal* signal)
+{
+ jamEntry();
+ StartInfoReq *const req =(StartInfoReq*)&signal->theData[0];
+ Uint32 startNode = req->startingNodeId;
+ if (cfailurenr != req->systemFailureNo) {
+ jam();
+ //---------------------------------------------------------------
+ // A failure occurred since master sent this request. We will ignore
+ // this request since the node is already dead that is starting.
+ //---------------------------------------------------------------
+ return;
+ }//if
+ CRASH_INSERTION(7123);
+ if (isMaster()) {
+ jam();
+ ndbrequire(getNodeStatus(startNode) == NodeRecord::STARTING);
+ } else {
+ jam();
+ ndbrequire(getNodeStatus(startNode) == NodeRecord::DEAD);
+ }//if
+ if ((!getAllowNodeStart(startNode)) ||
+ (c_nodeStartSlave.nodeId != 0) ||
+ (ERROR_INSERTED(7124))) {
+ jam();
+ StartInfoRef *const ref =(StartInfoRef*)&signal->theData[0];
+ ref->sendingNodeId = cownNodeId;
+ ref->errorCode = ZNODE_START_DISALLOWED_ERROR;
+ sendSignal(cmasterdihref, GSN_START_INFOREF, signal, 2, JBB);
+ return;
+ }//if
+ setNodeStatus(startNode, NodeRecord::STARTING);
+ if (req->typeStart == NodeState::ST_INITIAL_NODE_RESTART) {
+ jam();
+ setAllowNodeStart(startNode, false);
+ invalidateNodeLCP(signal, startNode, 0);
+ } else {
+ jam();
+ StartInfoConf * c = (StartInfoConf*)&signal->theData[0];
+ c->sendingNodeId = cownNodeId;
+ c->startingNodeId = startNode;
+ sendSignal(cmasterdihref, GSN_START_INFOCONF, signal,
+ StartInfoConf::SignalLength, JBB);
+ return;
+ }//if
+}//Dbdih::execSTART_INFOREQ()
+
+void Dbdih::execINCL_NODEREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 retRef = signal->theData[0];
+ Uint32 nodeId = signal->theData[1];
+ Uint32 tnodeStartFailNr = signal->theData[2];
+ Uint32 TndbVersion = signal->theData[3];
+ currentgcp = signal->theData[4];
+ CRASH_INSERTION(7127);
+ cnewgcp = currentgcp;
+ coldgcp = currentgcp - 1;
+ if (!isMaster()) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // We don't want to change the state of the master since he can be in the
+ // state LCP_TCGET at this time.
+ /*-----------------------------------------------------------------------*/
+ c_lcpState.setLcpStatus(LCP_STATUS_IDLE, __LINE__);
+ }//if
+
+ /*-------------------------------------------------------------------------*/
+ // When a node is restarted we must ensure that a lcp will be run
+ // as soon as possible and the reset the delay according to the original
+ // configuration.
+ // Without an initial local checkpoint the new node will not be available.
+ /*-------------------------------------------------------------------------*/
+ if (getOwnNodeId() == nodeId) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // We are the starting node. We came here only to set the global checkpoint
+ // id's and the lcp status.
+ /*-----------------------------------------------------------------------*/
+ CRASH_INSERTION(7171);
+ return;
+ }//if
+ if (getNodeStatus(nodeId) != NodeRecord::STARTING) {
+ jam();
+ return;
+ }//if
+ ndbrequire(cfailurenr == tnodeStartFailNr);
+ ndbrequire (c_nodeStartSlave.nodeId == 0);
+ c_nodeStartSlave.nodeId = nodeId;
+
+ ndbrequire (retRef == cmasterdihref);
+
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+
+ Sysfile::ActiveStatus TsaveState = nodePtr.p->activeStatus;
+ Uint32 TnodeGroup = nodePtr.p->nodeGroup;
+
+ initNodeState(nodePtr);
+ nodePtr.p->nodeGroup = TnodeGroup;
+ nodePtr.p->activeStatus = TsaveState;
+ nodePtr.p->nodeStatus = NodeRecord::ALIVE;
+ nodePtr.p->useInTransactions = true;
+ nodePtr.p->m_inclDihLcp = true;
+ nodePtr.p->ndbversion = TndbVersion;
+
+ removeDeadNode(nodePtr);
+ insertAlive(nodePtr);
+ con_lineNodes++;
+
+ /*-------------------------------------------------------------------------*/
+ // WE WILL ALSO SEND THE INCLUDE NODE REQUEST TO THE LOCAL LQH BLOCK.
+ /*-------------------------------------------------------------------------*/
+ signal->theData[0] = reference();
+ signal->theData[1] = nodeId;
+ signal->theData[2] = currentgcp;
+ sendSignal(clocallqhblockref, GSN_INCL_NODEREQ, signal, 3, JBB);
+}//Dbdih::execINCL_NODEREQ()
+
+/* ------------------------------------------------------------------------- */
+// execINCL_NODECONF() is found in the master logic part since it is used by
+// both the master and the slaves.
+/* ------------------------------------------------------------------------- */
+
+/*****************************************************************************/
+/*********** TAKE OVER DECISION MODULE *************/
+/*****************************************************************************/
+// This module contains the subroutines that take the decision whether to take
+// over a node now or not.
+/* ------------------------------------------------------------------------- */
+/* MASTER LOGIC FOR SYSTEM RESTART */
+/* ------------------------------------------------------------------------- */
+// WE ONLY COME HERE IF WE ARE THE MASTER AND WE ARE PERFORMING A SYSTEM
+// RESTART. WE ALSO COME HERE DURING THIS SYSTEM RESTART ONE TIME PER NODE
+// THAT NEEDS TAKE OVER.
+/*---------------------------------------------------------------------------*/
+// WE CHECK IF ANY NODE NEEDS TO BE TAKEN OVER AND THE TAKE OVER HAS NOT YET
+// BEEN STARTED OR COMPLETED.
+/*---------------------------------------------------------------------------*/
+void
+Dbdih::systemRestartTakeOverLab(Signal* signal)
+{
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ switch (nodePtr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ case Sysfile::NS_ActiveMissed_1:
+ jam();
+ break;
+ /*---------------------------------------------------------------------*/
+ // WE HAVE NOT REACHED A STATE YET WHERE THIS NODE NEEDS TO BE TAKEN OVER
+ /*---------------------------------------------------------------------*/
+ case Sysfile::NS_ActiveMissed_2:
+ case Sysfile::NS_NotActive_NotTakenOver:
+ jam();
+ /*---------------------------------------------------------------------*/
+ // THIS NODE IS IN TROUBLE.
+ // WE MUST SUCCEED WITH A LOCAL CHECKPOINT WITH THIS NODE TO REMOVE THE
+ // DANGER. IF THE NODE IS NOT ALIVE THEN THIS WILL NOT BE
+ // POSSIBLE AND WE CAN START THE TAKE OVER IMMEDIATELY IF WE HAVE ANY
+ // NODES THAT CAN PERFORM A TAKE OVER.
+ /*---------------------------------------------------------------------*/
+ if (nodePtr.p->nodeStatus != NodeRecord::ALIVE) {
+ jam();
+ Uint32 ThotSpareNode = findHotSpare();
+ if (ThotSpareNode != RNIL) {
+ jam();
+ startTakeOver(signal, RNIL, ThotSpareNode, nodePtr.i);
+ }//if
+ } else if(nodePtr.p->activeStatus == Sysfile::NS_NotActive_NotTakenOver){
+ jam();
+ /*-------------------------------------------------------------------*/
+ // NOT ACTIVE NODES THAT HAVE NOT YET BEEN TAKEN OVER NEEDS TAKE OVER
+ // IMMEDIATELY. IF WE ARE ALIVE WE TAKE OVER OUR OWN NODE.
+ /*-------------------------------------------------------------------*/
+ startTakeOver(signal, RNIL, nodePtr.i, nodePtr.i);
+ }//if
+ break;
+ case Sysfile::NS_TakeOver:
+ /**-------------------------------------------------------------------
+ * WE MUST HAVE FAILED IN THE MIDDLE OF THE TAKE OVER PROCESS.
+ * WE WILL CONCLUDE THE TAKE OVER PROCESS NOW.
+ *-------------------------------------------------------------------*/
+ if (nodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ jam();
+ Uint32 takeOverNode = Sysfile::getTakeOverNode(nodePtr.i,
+ SYSFILE->takeOver);
+ if(takeOverNode == 0){
+ jam();
+ warningEvent("Bug in take-over code restarting");
+ takeOverNode = nodePtr.i;
+ }
+ startTakeOver(signal, RNIL, nodePtr.i, takeOverNode);
+ } else {
+ jam();
+ /**-------------------------------------------------------------------
+ * We are not currently taking over, change our active status.
+ *-------------------------------------------------------------------*/
+ nodePtr.p->activeStatus = Sysfile::NS_NotActive_NotTakenOver;
+ setNodeRestartInfoBits();
+ }//if
+ break;
+ case Sysfile::NS_HotSpare:
+ jam();
+ break;
+ /*---------------------------------------------------------------------*/
+ // WE NEED NOT TAKE OVER NODES THAT ARE HOT SPARE.
+ /*---------------------------------------------------------------------*/
+ case Sysfile::NS_NotDefined:
+ jam();
+ break;
+ /*---------------------------------------------------------------------*/
+ // WE NEED NOT TAKE OVER NODES THAT DO NOT EVEN EXIST IN THE CLUSTER.
+ /*---------------------------------------------------------------------*/
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ }//for
+ /*-------------------------------------------------------------------------*/
+ /* NO TAKE OVER HAS BEEN INITIATED. */
+ /*-------------------------------------------------------------------------*/
+}//Dbdih::systemRestartTakeOverLab()
+
+/*---------------------------------------------------------------------------*/
+// This subroutine is called as part of node restart in the master node.
+/*---------------------------------------------------------------------------*/
+void Dbdih::nodeRestartTakeOver(Signal* signal, Uint32 startNodeId)
+{
+ switch (getNodeActiveStatus(startNodeId)) {
+ case Sysfile::NS_Active:
+ case Sysfile::NS_ActiveMissed_1:
+ case Sysfile::NS_ActiveMissed_2:
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // AN ACTIVE NODE HAS BEEN STARTED. THE ACTIVE NODE MUST THEN GET ALL DATA
+ // IT HAD BEFORE ITS CRASH. WE START THE TAKE OVER IMMEDIATELY.
+ // SINCE WE ARE AN ACTIVE NODE WE WILL TAKE OVER OUR OWN NODE THAT
+ // PREVIOUSLY CRASHED.
+ /*-----------------------------------------------------------------------*/
+ startTakeOver(signal, RNIL, startNodeId, startNodeId);
+ break;
+ case Sysfile::NS_HotSpare:{
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // WHEN STARTING UP A HOT SPARE WE WILL CHECK IF ANY NODE NEEDS TO TAKEN
+ // OVER. IF SO THEN WE WILL START THE TAKE OVER.
+ /*-----------------------------------------------------------------------*/
+ bool takeOverStarted = false;
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if (nodePtr.p->activeStatus == Sysfile::NS_NotActive_NotTakenOver) {
+ jam();
+ takeOverStarted = true;
+ startTakeOver(signal, RNIL, startNodeId, nodePtr.i);
+ }//if
+ }//for
+ if (!takeOverStarted) {
+ jam();
+ /*-------------------------------------------------------------------*/
+ // NO TAKE OVER WAS NEEDED AT THE MOMENT WE START-UP AND WAIT UNTIL A
+ // TAKE OVER IS NEEDED.
+ /*-------------------------------------------------------------------*/
+ BlockReference ref = calcDihBlockRef(startNodeId);
+ signal->theData[0] = startNodeId;
+ sendSignal(ref, GSN_START_COPYCONF, signal, 1, JBB);
+ }//if
+ break;
+ }
+ case Sysfile::NS_NotActive_NotTakenOver:
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // ALL DATA IN THE NODE IS LOST BUT WE HAVE NOT TAKEN OVER YET. WE WILL
+ // TAKE OVER OUR OWN NODE
+ /*-----------------------------------------------------------------------*/
+ startTakeOver(signal, RNIL, startNodeId, startNodeId);
+ break;
+ case Sysfile::NS_TakeOver:{
+ jam();
+ /*--------------------------------------------------------------------
+ * We were in the process of taking over but it was not completed.
+ * We will complete it now instead.
+ *--------------------------------------------------------------------*/
+ Uint32 takeOverNode = Sysfile::getTakeOverNode(startNodeId,
+ SYSFILE->takeOver);
+ startTakeOver(signal, RNIL, startNodeId, takeOverNode);
+ break;
+ }
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ nodeResetStart();
+}//Dbdih::nodeRestartTakeOver()
+
+/*************************************************************************/
+// Ths routine is called when starting a local checkpoint.
+/*************************************************************************/
+void Dbdih::checkStartTakeOver(Signal* signal)
+{
+ NodeRecordPtr csoNodeptr;
+ Uint32 tcsoHotSpareNode;
+ Uint32 tcsoTakeOverNode;
+ if (isMaster()) {
+ /*-----------------------------------------------------------------*/
+ /* WE WILL ONLY START TAKE OVER IF WE ARE MASTER. */
+ /*-----------------------------------------------------------------*/
+ /* WE WILL ONLY START THE TAKE OVER IF THERE WERE A NEED OF */
+ /* A TAKE OVER. */
+ /*-----------------------------------------------------------------*/
+ /* WE CAN ONLY PERFORM THE TAKE OVER IF WE HAVE A HOT SPARE */
+ /* AVAILABLE. */
+ /*-----------------------------------------------------------------*/
+ tcsoTakeOverNode = 0;
+ tcsoHotSpareNode = 0;
+ for (csoNodeptr.i = 1; csoNodeptr.i < MAX_NDB_NODES; csoNodeptr.i++) {
+ ptrAss(csoNodeptr, nodeRecord);
+ if (csoNodeptr.p->activeStatus == Sysfile::NS_NotActive_NotTakenOver) {
+ jam();
+ tcsoTakeOverNode = csoNodeptr.i;
+ } else {
+ jam();
+ if (csoNodeptr.p->activeStatus == Sysfile::NS_HotSpare) {
+ jam();
+ tcsoHotSpareNode = csoNodeptr.i;
+ }//if
+ }//if
+ }//for
+ if ((tcsoTakeOverNode != 0) &&
+ (tcsoHotSpareNode != 0)) {
+ jam();
+ startTakeOver(signal, RNIL, tcsoHotSpareNode, tcsoTakeOverNode);
+ }//if
+ }//if
+}//Dbdih::checkStartTakeOver()
+
+/*****************************************************************************/
+/*********** NODE ADDING MODULE *************/
+/*********** CODE TO HANDLE TAKE OVER *************/
+/*****************************************************************************/
+// A take over can be initiated by a number of things:
+// 1) A node restart, usually the node takes over itself but can also take
+// over somebody else if its own data was already taken over
+// 2) At system restart it is necessary to use the take over code to recover
+// nodes which had too old checkpoints to be restorable by the usual
+// restoration from disk.
+// 3) When a node has missed too many local checkpoints and is decided by the
+// master to be taken over by a hot spare node that sits around waiting
+// for this to happen.
+//
+// To support multiple node failures efficiently the code is written such that
+// only one take over can handle transitions in state but during a copy
+// fragment other take over's can perform state transitions.
+/*****************************************************************************/
+void Dbdih::startTakeOver(Signal* signal,
+ Uint32 takeOverPtrI,
+ Uint32 startNode,
+ Uint32 nodeTakenOver)
+{
+ NodeRecordPtr toNodePtr;
+ NodeGroupRecordPtr NGPtr;
+ toNodePtr.i = nodeTakenOver;
+ ptrCheckGuard(toNodePtr, MAX_NDB_NODES, nodeRecord);
+ NGPtr.i = toNodePtr.p->nodeGroup;
+ ptrCheckGuard(NGPtr, MAX_NDB_NODES, nodeGroupRecord);
+ TakeOverRecordPtr takeOverPtr;
+ if (takeOverPtrI == RNIL) {
+ jam();
+ setAllowNodeStart(startNode, false);
+ seizeTakeOver(takeOverPtr);
+ if (startNode == c_nodeStartMaster.startNode) {
+ jam();
+ takeOverPtr.p->toNodeRestart = true;
+ }//if
+ takeOverPtr.p->toStartingNode = startNode;
+ takeOverPtr.p->toFailedNode = nodeTakenOver;
+ } else {
+ jam();
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+ ndbrequire(takeOverPtr.p->toStartingNode == startNode);
+ ndbrequire(takeOverPtr.p->toFailedNode == nodeTakenOver);
+ ndbrequire(takeOverPtr.p->toMasterStatus == TakeOverRecord::TO_WAIT_START_TAKE_OVER);
+ }//if
+ if ((NGPtr.p->activeTakeOver) || (ERROR_INSERTED(7157))) {
+ jam();
+ /**------------------------------------------------------------------------
+ * A take over is already active in this node group. We only allow one
+ * take over per node group. Otherwise we will overload the node group and
+ * also we will require much more checks when starting up copying of
+ * fragments. The parallelism for take over is mainly to ensure that we
+ * can handle take over efficiently in large systems with 4 nodes and above
+ * A typical case is a 8 node system executing on two 8-cpu boxes.
+ * A box crash in one of the boxes will mean 4 nodes crashes.
+ * We want to be able to restart those four nodes to some
+ * extent in parallel.
+ *
+ * We will wait for a few seconds and then try again.
+ */
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_WAIT_START_TAKE_OVER;
+ signal->theData[0] = DihContinueB::ZSTART_TAKE_OVER;
+ signal->theData[1] = takeOverPtr.i;
+ signal->theData[2] = startNode;
+ signal->theData[3] = nodeTakenOver;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 5000, 4);
+ return;
+ }//if
+ NGPtr.p->activeTakeOver = true;
+ if (startNode == nodeTakenOver) {
+ jam();
+ switch (getNodeActiveStatus(nodeTakenOver)) {
+ case Sysfile::NS_Active:
+ case Sysfile::NS_ActiveMissed_1:
+ case Sysfile::NS_ActiveMissed_2:
+ jam();
+ break;
+ case Sysfile::NS_NotActive_NotTakenOver:
+ case Sysfile::NS_TakeOver:
+ jam();
+ setNodeActiveStatus(nodeTakenOver, Sysfile::NS_TakeOver);
+ break;
+ default:
+ ndbrequire(false);
+ }//switch
+ } else {
+ jam();
+ setNodeActiveStatus(nodeTakenOver, Sysfile::NS_HotSpare);
+ setNodeActiveStatus(startNode, Sysfile::NS_TakeOver);
+ changeNodeGroups(startNode, nodeTakenOver);
+ }//if
+ setNodeRestartInfoBits();
+ /* ---------------------------------------------------------------------- */
+ /* WE SET THE RESTART INFORMATION TO INDICATE THAT WE ARE ABOUT TO TAKE */
+ /* OVER THE FAILED NODE. WE SET THIS INFORMATION AND WAIT UNTIL THE */
+ /* GLOBAL CHECKPOINT HAS WRITTEN THE RESTART INFORMATION. */
+ /* ---------------------------------------------------------------------- */
+ Sysfile::setTakeOverNode(takeOverPtr.p->toFailedNode, SYSFILE->takeOver,
+ startNode);
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_START_COPY;
+
+ cstartGcpNow = true;
+}//Dbdih::startTakeOver()
+
+void Dbdih::changeNodeGroups(Uint32 startNode, Uint32 nodeTakenOver)
+{
+ NodeRecordPtr startNodePtr;
+ NodeRecordPtr toNodePtr;
+ startNodePtr.i = startNode;
+ ptrCheckGuard(startNodePtr, MAX_NDB_NODES, nodeRecord);
+ toNodePtr.i = nodeTakenOver;
+ ptrCheckGuard(toNodePtr, MAX_NDB_NODES, nodeRecord);
+ ndbrequire(startNodePtr.p->nodeGroup == ZNIL);
+ NodeGroupRecordPtr NGPtr;
+
+ NGPtr.i = toNodePtr.p->nodeGroup;
+ ptrCheckGuard(NGPtr, MAX_NDB_NODES, nodeGroupRecord);
+ bool nodeFound = false;
+ for (Uint32 i = 0; i < NGPtr.p->nodeCount; i++) {
+ jam();
+ if (NGPtr.p->nodesInGroup[i] == nodeTakenOver) {
+ jam();
+ NGPtr.p->nodesInGroup[i] = startNode;
+ nodeFound = true;
+ }//if
+ }//for
+ ndbrequire(nodeFound);
+ Sysfile::setNodeGroup(startNodePtr.i, SYSFILE->nodeGroups, toNodePtr.p->nodeGroup);
+ startNodePtr.p->nodeGroup = toNodePtr.p->nodeGroup;
+ Sysfile::setNodeGroup(toNodePtr.i, SYSFILE->nodeGroups, NO_NODE_GROUP_ID);
+ toNodePtr.p->nodeGroup = ZNIL;
+}//Dbdih::changeNodeGroups()
+
+void Dbdih::checkToCopy()
+{
+ TakeOverRecordPtr takeOverPtr;
+ for (takeOverPtr.i = 0;takeOverPtr.i < MAX_NDB_NODES; takeOverPtr.i++) {
+ ptrAss(takeOverPtr, takeOverRecord);
+ /*----------------------------------------------------------------------*/
+ // TAKE OVER HANDLING WRITES RESTART INFORMATION THROUGH
+ // THE GLOBAL CHECKPOINT
+ // PROTOCOL. WE CHECK HERE BEFORE STARTING A WRITE OF THE RESTART
+ // INFORMATION.
+ /*-----------------------------------------------------------------------*/
+ if (takeOverPtr.p->toMasterStatus == TakeOverRecord::TO_START_COPY) {
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_START_COPY_ONGOING;
+ } else if (takeOverPtr.p->toMasterStatus == TakeOverRecord::TO_END_COPY) {
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_END_COPY_ONGOING;
+ }//if
+ }//for
+}//Dbdih::checkToCopy()
+
+void Dbdih::checkToCopyCompleted(Signal* signal)
+{
+ /* ------------------------------------------------------------------------*/
+ /* WE CHECK HERE IF THE WRITING OF TAKE OVER INFORMATION ALSO HAS BEEN */
+ /* COMPLETED. */
+ /* ------------------------------------------------------------------------*/
+ TakeOverRecordPtr toPtr;
+ for (toPtr.i = 0; toPtr.i < MAX_NDB_NODES; toPtr.i++) {
+ ptrAss(toPtr, takeOverRecord);
+ if (toPtr.p->toMasterStatus == TakeOverRecord::TO_START_COPY_ONGOING){
+ jam();
+ sendStartTo(signal, toPtr.i);
+ } else if (toPtr.p->toMasterStatus == TakeOverRecord::TO_END_COPY_ONGOING){
+ jam();
+ sendEndTo(signal, toPtr.i);
+ } else {
+ jam();
+ }//if
+ }//for
+}//Dbdih::checkToCopyCompleted()
+
+bool Dbdih::checkToInterrupted(TakeOverRecordPtr& takeOverPtr)
+{
+ if (checkNodeAlive(takeOverPtr.p->toStartingNode)) {
+ jam();
+ return false;
+ } else {
+ jam();
+ endTakeOver(takeOverPtr.i);
+ return true;
+ }//if
+}//Dbdih::checkToInterrupted()
+
+void Dbdih::sendStartTo(Signal* signal, Uint32 takeOverPtrI)
+{
+ TakeOverRecordPtr takeOverPtr;
+ CRASH_INSERTION(7155);
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+ if ((c_startToLock != RNIL) || (ERROR_INSERTED(7158))) {
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_WAIT_START;
+ signal->theData[0] = DihContinueB::ZSEND_START_TO;
+ signal->theData[1] = takeOverPtrI;
+ signal->theData[2] = takeOverPtr.p->toStartingNode;
+ signal->theData[3] = takeOverPtr.p->toFailedNode;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 30, 4);
+ return;
+ }//if
+ c_startToLock = takeOverPtrI;
+ StartToReq * const req = (StartToReq *)&signal->theData[0];
+ req->userPtr = takeOverPtr.i;
+ req->userRef = reference();
+ req->startingNodeId = takeOverPtr.p->toStartingNode;
+ req->nodeTakenOver = takeOverPtr.p->toFailedNode;
+ req->nodeRestart = takeOverPtr.p->toNodeRestart;
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::STARTING;
+ sendLoopMacro(START_TOREQ, sendSTART_TOREQ);
+}//Dbdih::sendStartTo()
+
+void Dbdih::execSTART_TOREQ(Signal* signal)
+{
+ TakeOverRecordPtr takeOverPtr;
+ jamEntry();
+ const StartToReq * const req = (StartToReq *)&signal->theData[0];
+ takeOverPtr.i = req->userPtr;
+ BlockReference ref = req->userRef;
+ Uint32 startingNode = req->startingNodeId;
+
+ CRASH_INSERTION(7133);
+ RETURN_IF_NODE_NOT_ALIVE(req->startingNodeId);
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ allocateTakeOver(takeOverPtr);
+ initStartTakeOver(req, takeOverPtr);
+
+ StartToConf * const conf = (StartToConf *)&signal->theData[0];
+ conf->userPtr = takeOverPtr.i;
+ conf->sendingNodeId = cownNodeId;
+ conf->startingNodeId = startingNode;
+ sendSignal(ref, GSN_START_TOCONF, signal, StartToConf::SignalLength, JBB);
+}//Dbdih::execSTART_TOREQ()
+
+void Dbdih::execSTART_TOCONF(Signal* signal)
+{
+ TakeOverRecordPtr takeOverPtr;
+ jamEntry();
+ const StartToConf * const conf = (StartToConf *)&signal->theData[0];
+
+ CRASH_INSERTION(7147);
+
+ RETURN_IF_NODE_NOT_ALIVE(conf->startingNodeId);
+
+ takeOverPtr.i = conf->userPtr;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ ndbrequire(takeOverPtr.p->toMasterStatus == TakeOverRecord::STARTING);
+ ndbrequire(takeOverPtr.p->toStartingNode == conf->startingNodeId);
+ receiveLoopMacro(START_TOREQ, conf->sendingNodeId);
+ CRASH_INSERTION(7134);
+ c_startToLock = RNIL;
+
+ startNextCopyFragment(signal, takeOverPtr.i);
+}//Dbdih::execSTART_TOCONF()
+
+void Dbdih::initStartTakeOver(const StartToReq * req,
+ TakeOverRecordPtr takeOverPtr)
+{
+ takeOverPtr.p->toCurrentTabref = 0;
+ takeOverPtr.p->toCurrentFragid = 0;
+ takeOverPtr.p->toStartingNode = req->startingNodeId;
+ takeOverPtr.p->toFailedNode = req->nodeTakenOver;
+ takeOverPtr.p->toSlaveStatus = TakeOverRecord::TO_SLAVE_STARTED;
+ takeOverPtr.p->toCopyNode = RNIL;
+ takeOverPtr.p->toCurrentReplica = RNIL;
+ takeOverPtr.p->toNodeRestart = req->nodeRestart;
+}//Dbdih::initStartTakeOver()
+
+void Dbdih::startNextCopyFragment(Signal* signal, Uint32 takeOverPtrI)
+{
+ TabRecordPtr tabPtr;
+ TakeOverRecordPtr takeOverPtr;
+ Uint32 loopCount;
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::SELECTING_NEXT;
+ loopCount = 0;
+ if (ERROR_INSERTED(7159)) {
+ loopCount = 100;
+ }//if
+ while (loopCount++ < 100) {
+ tabPtr.i = takeOverPtr.p->toCurrentTabref;
+ if (tabPtr.i >= ctabFileSize) {
+ jam();
+ CRASH_INSERTION(7136);
+ sendUpdateTo(signal, takeOverPtr.i, UpdateToReq::TO_COPY_COMPLETED);
+ return;
+ }//if
+ ptrAss(tabPtr, tabRecord);
+ if (tabPtr.p->tabStatus != TabRecord::TS_ACTIVE){
+ jam();
+ takeOverPtr.p->toCurrentFragid = 0;
+ takeOverPtr.p->toCurrentTabref++;
+ continue;
+ }//if
+ Uint32 fragId = takeOverPtr.p->toCurrentFragid;
+ if (fragId >= tabPtr.p->totalfragments) {
+ jam();
+ takeOverPtr.p->toCurrentFragid = 0;
+ takeOverPtr.p->toCurrentTabref++;
+ if (ERROR_INSERTED(7135)) {
+ if (takeOverPtr.p->toCurrentTabref == 1) {
+ ndbrequire(false);
+ }//if
+ }//if
+ continue;
+ }//if
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ ReplicaRecordPtr loopReplicaPtr;
+ loopReplicaPtr.i = fragPtr.p->oldStoredReplicas;
+ while (loopReplicaPtr.i != RNIL) {
+ ptrCheckGuard(loopReplicaPtr, creplicaFileSize, replicaRecord);
+ if (loopReplicaPtr.p->procNode == takeOverPtr.p->toFailedNode) {
+ jam();
+ /* ----------------------------------------------------------------- */
+ /* WE HAVE FOUND A REPLICA THAT BELONGED THE FAILED NODE THAT NEEDS */
+ /* TAKE OVER. WE TAKE OVER THIS REPLICA TO THE NEW NODE. */
+ /* ----------------------------------------------------------------- */
+ takeOverPtr.p->toCurrentReplica = loopReplicaPtr.i;
+ toCopyFragLab(signal, takeOverPtr.i);
+ return;
+ } else if (loopReplicaPtr.p->procNode == takeOverPtr.p->toStartingNode) {
+ jam();
+ /* ----------------------------------------------------------------- */
+ /* WE HAVE OBVIOUSLY STARTED TAKING OVER THIS WITHOUT COMPLETING IT. */
+ /* WE */
+ /* NEED TO COMPLETE THE TAKE OVER OF THIS REPLICA. */
+ /* ----------------------------------------------------------------- */
+ takeOverPtr.p->toCurrentReplica = loopReplicaPtr.i;
+ toCopyFragLab(signal, takeOverPtr.i);
+ return;
+ } else {
+ jam();
+ loopReplicaPtr.i = loopReplicaPtr.p->nextReplica;
+ }//if
+ }//while
+ takeOverPtr.p->toCurrentFragid++;
+ }//while
+ signal->theData[0] = DihContinueB::ZTO_START_COPY_FRAG;
+ signal->theData[1] = takeOverPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+}//Dbdih::startNextCopyFragment()
+
+void Dbdih::toCopyFragLab(Signal* signal,
+ Uint32 takeOverPtrI)
+{
+ TakeOverRecordPtr takeOverPtr;
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+
+ CreateReplicaRecordPtr createReplicaPtr;
+ createReplicaPtr.i = 0;
+ ptrAss(createReplicaPtr, createReplicaRecord);
+
+ ReplicaRecordPtr replicaPtr;
+ replicaPtr.i = takeOverPtr.p->toCurrentReplica;
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = takeOverPtr.p->toCurrentTabref;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ /* ----------------------------------------------------------------------- */
+ /* WE HAVE FOUND A REPLICA THAT NEEDS TAKE OVER. WE WILL START THIS TAKE */
+ /* OVER BY ADDING THE FRAGMENT WHEREAFTER WE WILL ORDER THE PRIMARY */
+ /* REPLICA TO COPY ITS CONTENT TO THE NEW STARTING REPLICA. */
+ /* THIS OPERATION IS A SINGLE USER OPERATION UNTIL WE HAVE SENT */
+ /* COPY_FRAGREQ. AFTER SENDING COPY_FRAGREQ WE ARE READY TO START A NEW */
+ /* FRAGMENT REPLICA. WE WILL NOT IMPLEMENT THIS IN THE FIRST PHASE. */
+ /* ----------------------------------------------------------------------- */
+ cnoOfCreateReplicas = 1;
+ createReplicaPtr.p->hotSpareUse = true;
+ createReplicaPtr.p->dataNodeId = takeOverPtr.p->toStartingNode;
+
+ prepareSendCreateFragReq(signal, takeOverPtrI);
+}//Dbdih::toCopyFragLab()
+
+void Dbdih::prepareSendCreateFragReq(Signal* signal, Uint32 takeOverPtrI)
+{
+ TakeOverRecordPtr takeOverPtr;
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = takeOverPtr.p->toCurrentTabref;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ FragmentstorePtr fragPtr;
+
+ getFragstore(tabPtr.p, takeOverPtr.p->toCurrentFragid, fragPtr);
+ Uint32 nodes[MAX_REPLICAS];
+ extractNodeInfo(fragPtr.p, nodes);
+ takeOverPtr.p->toCopyNode = nodes[0];
+ sendCreateFragReq(signal, 0, CreateFragReq::STORED, takeOverPtr.i);
+}//Dbdih::prepareSendCreateFragReq()
+
+void Dbdih::sendCreateFragReq(Signal* signal,
+ Uint32 startGci,
+ Uint32 replicaType,
+ Uint32 takeOverPtrI)
+{
+ TakeOverRecordPtr takeOverPtr;
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+ if ((c_createFragmentLock != RNIL) ||
+ ((ERROR_INSERTED(7161))&&(replicaType == CreateFragReq::STORED)) ||
+ ((ERROR_INSERTED(7162))&&(replicaType == CreateFragReq::COMMIT_STORED))){
+ if (replicaType == CreateFragReq::STORED) {
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_WAIT_PREPARE_CREATE;
+ } else {
+ ndbrequire(replicaType == CreateFragReq::COMMIT_STORED);
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_WAIT_COMMIT_CREATE;
+ }//if
+ signal->theData[0] = DihContinueB::ZSEND_CREATE_FRAG;
+ signal->theData[1] = takeOverPtr.i;
+ signal->theData[2] = replicaType;
+ signal->theData[3] = startGci;
+ signal->theData[4] = takeOverPtr.p->toStartingNode;
+ signal->theData[5] = takeOverPtr.p->toFailedNode;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 50, 6);
+ return;
+ }//if
+ c_createFragmentLock = takeOverPtr.i;
+ sendLoopMacro(CREATE_FRAGREQ, nullRoutine);
+
+ CreateFragReq * const req = (CreateFragReq *)&signal->theData[0];
+ req->userPtr = takeOverPtr.i;
+ req->userRef = reference();
+ req->tableId = takeOverPtr.p->toCurrentTabref;
+ req->fragId = takeOverPtr.p->toCurrentFragid;
+ req->startingNodeId = takeOverPtr.p->toStartingNode;
+ req->copyNodeId = takeOverPtr.p->toCopyNode;
+ req->startGci = startGci;
+ req->replicaType = replicaType;
+
+ NodeRecordPtr nodePtr;
+ nodePtr.i = cfirstAliveNode;
+ do {
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ BlockReference ref = calcDihBlockRef(nodePtr.i);
+ sendSignal(ref, GSN_CREATE_FRAGREQ, signal,
+ CreateFragReq::SignalLength, JBB);
+ nodePtr.i = nodePtr.p->nextNode;
+ } while (nodePtr.i != RNIL);
+
+ if (replicaType == CreateFragReq::STORED) {
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::PREPARE_CREATE;
+ } else {
+ ndbrequire(replicaType == CreateFragReq::COMMIT_STORED);
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::COMMIT_CREATE;
+ }
+}//Dbdih::sendCreateFragReq()
+
+/* --------------------------------------------------------------------------*/
+/* AN ORDER TO START OR COMMIT THE REPLICA CREATION ARRIVED FROM THE */
+/* MASTER. */
+/* --------------------------------------------------------------------------*/
+void Dbdih::execCREATE_FRAGREQ(Signal* signal)
+{
+ jamEntry();
+ CreateFragReq * const req = (CreateFragReq *)&signal->theData[0];
+
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = req->userPtr;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ BlockReference retRef = req->userRef;
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ Uint32 fragId = req->fragId;
+ Uint32 tdestNodeid = req->startingNodeId;
+ Uint32 tsourceNodeid = req->copyNodeId;
+ Uint32 startGci = req->startGci;
+ Uint32 replicaType = req->replicaType;
+
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ RETURN_IF_NODE_NOT_ALIVE(tdestNodeid);
+ ReplicaRecordPtr frReplicaPtr;
+ findToReplica(takeOverPtr.p, replicaType, fragPtr, frReplicaPtr);
+ ndbrequire(frReplicaPtr.i != RNIL);
+
+ switch (replicaType) {
+ case CreateFragReq::STORED:
+ jam();
+ CRASH_INSERTION(7138);
+ /* ----------------------------------------------------------------------*/
+ /* HERE WE ARE INSERTING THE NEW BACKUP NODE IN THE EXECUTION OF ALL */
+ /* OPERATIONS. FROM HERE ON ALL OPERATIONS ON THIS FRAGMENT WILL INCLUDE*/
+ /* USE OF THE NEW REPLICA. */
+ /* --------------------------------------------------------------------- */
+ insertBackup(fragPtr, tdestNodeid);
+ takeOverPtr.p->toCopyNode = tsourceNodeid;
+ takeOverPtr.p->toSlaveStatus = TakeOverRecord::TO_SLAVE_CREATE_PREPARE;
+
+ fragPtr.p->distributionKey++;
+ fragPtr.p->distributionKey &= 255;
+ break;
+ case CreateFragReq::COMMIT_STORED:
+ jam();
+ CRASH_INSERTION(7139);
+ /* ----------------------------------------------------------------------*/
+ /* HERE WE ARE MOVING THE REPLICA TO THE STORED SECTION SINCE IT IS NOW */
+ /* FULLY LOADED WITH ALL DATA NEEDED. */
+ // We also update the order of the replicas here so that if the new
+ // replica is the desired primary we insert it as primary.
+ /* ----------------------------------------------------------------------*/
+ takeOverPtr.p->toSlaveStatus = TakeOverRecord::TO_SLAVE_CREATE_COMMIT;
+ removeOldStoredReplica(fragPtr, frReplicaPtr);
+ linkStoredReplica(fragPtr, frReplicaPtr);
+ updateNodeInfo(fragPtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+
+ /* ------------------------------------------------------------------------*/
+ /* THE NEW NODE OF THIS REPLICA IS THE STARTING NODE. */
+ /* ------------------------------------------------------------------------*/
+ if (frReplicaPtr.p->procNode != takeOverPtr.p->toStartingNode) {
+ jam();
+ /* ---------------------------------------------------------------------*/
+ /* IF WE ARE STARTING A TAKE OVER NODE WE MUST INVALIDATE ALL LCP'S. */
+ /* OTHERWISE WE WILL TRY TO START LCP'S THAT DO NOT EXIST. */
+ /* ---------------------------------------------------------------------*/
+ frReplicaPtr.p->procNode = takeOverPtr.p->toStartingNode;
+ frReplicaPtr.p->noCrashedReplicas = 0;
+ frReplicaPtr.p->createGci[0] = startGci;
+ ndbrequire(startGci != 0xF1F1F1F1);
+ frReplicaPtr.p->replicaLastGci[0] = (Uint32)-1;
+ for (Uint32 i = 0; i < MAX_LCP_STORED; i++) {
+ frReplicaPtr.p->lcpStatus[i] = ZINVALID;
+ }//for
+ } else {
+ jam();
+ const Uint32 noCrashed = frReplicaPtr.p->noCrashedReplicas;
+ arrGuard(noCrashed, 8);
+ frReplicaPtr.p->createGci[noCrashed] = startGci;
+ ndbrequire(startGci != 0xF1F1F1F1);
+ frReplicaPtr.p->replicaLastGci[noCrashed] = (Uint32)-1;
+ }//if
+ takeOverPtr.p->toCurrentTabref = tabPtr.i;
+ takeOverPtr.p->toCurrentFragid = fragId;
+ CreateFragConf * const conf = (CreateFragConf *)&signal->theData[0];
+ conf->userPtr = takeOverPtr.i;
+ conf->tableId = tabPtr.i;
+ conf->fragId = fragId;
+ conf->sendingNodeId = cownNodeId;
+ conf->startingNodeId = tdestNodeid;
+ sendSignal(retRef, GSN_CREATE_FRAGCONF, signal,
+ CreateFragConf::SignalLength, JBB);
+}//Dbdih::execCREATE_FRAGREQ()
+
+void Dbdih::execCREATE_FRAGCONF(Signal* signal)
+{
+ jamEntry();
+ CRASH_INSERTION(7148);
+ const CreateFragConf * const conf = (CreateFragConf *)&signal->theData[0];
+ Uint32 fragId = conf->fragId;
+
+ RETURN_IF_NODE_NOT_ALIVE(conf->startingNodeId);
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = conf->tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = conf->userPtr;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ ndbrequire(tabPtr.i == takeOverPtr.p->toCurrentTabref);
+ ndbrequire(fragId == takeOverPtr.p->toCurrentFragid);
+ receiveLoopMacro(CREATE_FRAGREQ, conf->sendingNodeId);
+ c_createFragmentLock = RNIL;
+
+ if (takeOverPtr.p->toMasterStatus == TakeOverRecord::PREPARE_CREATE) {
+ jam();
+ CRASH_INSERTION(7140);
+ /* --------------------------------------------------------------------- */
+ /* ALL NODES HAVE PREPARED THE INTRODUCTION OF THIS NEW NODE AND IT IS */
+ /* ALREADY IN USE. WE CAN NOW START COPYING THE FRAGMENT. */
+ /*---------------------------------------------------------------------- */
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::COPY_FRAG;
+ BlockReference ref = calcLqhBlockRef(takeOverPtr.p->toCopyNode);
+ CopyFragReq * const copyFragReq = (CopyFragReq *)&signal->theData[0];
+ copyFragReq->userPtr = takeOverPtr.i;
+ copyFragReq->userRef = reference();
+ copyFragReq->tableId = tabPtr.i;
+ copyFragReq->fragId = fragId;
+ copyFragReq->nodeId = takeOverPtr.p->toStartingNode;
+ copyFragReq->schemaVersion = tabPtr.p->schemaVersion;
+ copyFragReq->distributionKey = fragPtr.p->distributionKey;
+ sendSignal(ref, GSN_COPY_FRAGREQ, signal, CopyFragReq::SignalLength, JBB);
+ } else {
+ ndbrequire(takeOverPtr.p->toMasterStatus == TakeOverRecord::COMMIT_CREATE);
+ jam();
+ CRASH_INSERTION(7141);
+ /* --------------------------------------------------------------------- */
+ // REPORT that copy of fragment has been completed.
+ /* --------------------------------------------------------------------- */
+ signal->theData[0] = EventReport::NR_CopyFragDone;
+ signal->theData[1] = takeOverPtr.p->toStartingNode;
+ signal->theData[2] = tabPtr.i;
+ signal->theData[3] = takeOverPtr.p->toCurrentFragid;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+ /* --------------------------------------------------------------------- */
+ /* WE HAVE NOW CREATED THIS NEW REPLICA AND WE ARE READY TO TAKE THE */
+ /* THE NEXT REPLICA. */
+ /* --------------------------------------------------------------------- */
+
+ Mutex mutex(signal, c_mutexMgr, takeOverPtr.p->m_switchPrimaryMutexHandle);
+ mutex.unlock(); // ignore result
+
+ takeOverPtr.p->toCurrentFragid++;
+ startNextCopyFragment(signal, takeOverPtr.i);
+ }//if
+}//Dbdih::execCREATE_FRAGCONF()
+
+void Dbdih::execCOPY_FRAGREF(Signal* signal)
+{
+ const CopyFragRef * const ref = (CopyFragRef *)&signal->theData[0];
+ jamEntry();
+ Uint32 takeOverPtrI = ref->userPtr;
+ Uint32 startingNodeId = ref->startingNodeId;
+ Uint32 errorCode = ref->errorCode;
+
+ TakeOverRecordPtr takeOverPtr;
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+ ndbrequire(errorCode != ZNODE_FAILURE_ERROR);
+ ndbrequire(ref->tableId == takeOverPtr.p->toCurrentTabref);
+ ndbrequire(ref->fragId == takeOverPtr.p->toCurrentFragid);
+ ndbrequire(ref->startingNodeId == takeOverPtr.p->toStartingNode);
+ ndbrequire(ref->sendingNodeId == takeOverPtr.p->toCopyNode);
+ ndbrequire(takeOverPtr.p->toMasterStatus == TakeOverRecord::COPY_FRAG);
+ endTakeOver(takeOverPtrI);
+ //--------------------------------------------------------------------------
+ // For some reason we did not succeed in copying a fragment. We treat this
+ // as a serious failure and crash the starting node.
+ //--------------------------------------------------------------------------
+ BlockReference cntrRef = calcNdbCntrBlockRef(startingNodeId);
+ SystemError * const sysErr = (SystemError*)&signal->theData[0];
+ sysErr->errorCode = SystemError::CopyFragRefError;
+ sysErr->errorRef = reference();
+ sendSignal(cntrRef, GSN_SYSTEM_ERROR, signal,
+ SystemError::SignalLength, JBB);
+ return;
+}//Dbdih::execCOPY_FRAGREF()
+
+void Dbdih::execCOPY_FRAGCONF(Signal* signal)
+{
+ const CopyFragConf * const conf = (CopyFragConf *)&signal->theData[0];
+ jamEntry();
+ CRASH_INSERTION(7142);
+
+ TakeOverRecordPtr takeOverPtr;
+ Uint32 takeOverPtrI = conf->userPtr;
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+
+ ndbrequire(conf->tableId == takeOverPtr.p->toCurrentTabref);
+ ndbrequire(conf->fragId == takeOverPtr.p->toCurrentFragid);
+ ndbrequire(conf->startingNodeId == takeOverPtr.p->toStartingNode);
+ ndbrequire(conf->sendingNodeId == takeOverPtr.p->toCopyNode);
+ ndbrequire(takeOverPtr.p->toMasterStatus == TakeOverRecord::COPY_FRAG);
+ sendUpdateTo(signal, takeOverPtr.i,
+ (Uint32)UpdateToReq::TO_COPY_FRAG_COMPLETED);
+}//Dbdih::execCOPY_FRAGCONF()
+
+void Dbdih::sendUpdateTo(Signal* signal,
+ Uint32 takeOverPtrI, Uint32 updateState)
+{
+ TakeOverRecordPtr takeOverPtr;
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+ if ((c_updateToLock != RNIL) ||
+ ((ERROR_INSERTED(7163)) &&
+ (updateState == UpdateToReq::TO_COPY_FRAG_COMPLETED)) ||
+ ((ERROR_INSERTED(7169)) &&
+ (updateState == UpdateToReq::TO_COPY_COMPLETED))) {
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_WAIT_UPDATE_TO;
+ signal->theData[0] = DihContinueB::ZSEND_UPDATE_TO;
+ signal->theData[1] = takeOverPtrI;
+ signal->theData[2] = takeOverPtr.p->toStartingNode;
+ signal->theData[3] = takeOverPtr.p->toFailedNode;
+ signal->theData[4] = updateState;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 30, 5);
+ return;
+ }//if
+ c_updateToLock = takeOverPtrI;
+ if (updateState == UpdateToReq::TO_COPY_FRAG_COMPLETED) {
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_UPDATE_TO;
+ } else {
+ jam();
+ ndbrequire(updateState == UpdateToReq::TO_COPY_COMPLETED);
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_COPY_COMPLETED;
+ }//if
+
+ UpdateToReq * const req = (UpdateToReq *)&signal->theData[0];
+ req->userPtr = takeOverPtr.i;
+ req->userRef = reference();
+ req->updateState = (UpdateToReq::UpdateState)updateState;
+ req->startingNodeId = takeOverPtr.p->toStartingNode;
+ req->tableId = takeOverPtr.p->toCurrentTabref;
+ req->fragmentNo = takeOverPtr.p->toCurrentFragid;
+ sendLoopMacro(UPDATE_TOREQ, sendUPDATE_TOREQ);
+}//Dbdih::sendUpdateTo()
+
+void Dbdih::execUPDATE_TOREQ(Signal* signal)
+{
+ jamEntry();
+ const UpdateToReq * const req = (UpdateToReq *)&signal->theData[0];
+ BlockReference ref = req->userRef;
+ ndbrequire(cmasterdihref == ref);
+
+ CRASH_INSERTION(7154);
+ RETURN_IF_NODE_NOT_ALIVE(req->startingNodeId);
+
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = req->userPtr;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ ndbrequire(req->startingNodeId == takeOverPtr.p->toStartingNode);
+ if (req->updateState == UpdateToReq::TO_COPY_FRAG_COMPLETED) {
+ jam();
+ ndbrequire(takeOverPtr.p->toSlaveStatus == TakeOverRecord::TO_SLAVE_CREATE_PREPARE);
+ takeOverPtr.p->toSlaveStatus = TakeOverRecord::TO_SLAVE_COPY_FRAG_COMPLETED;
+ takeOverPtr.p->toCurrentTabref = req->tableId;
+ takeOverPtr.p->toCurrentFragid = req->fragmentNo;
+ } else {
+ jam();
+ ndbrequire(req->updateState == UpdateToReq::TO_COPY_COMPLETED);
+ takeOverPtr.p->toSlaveStatus = TakeOverRecord::TO_SLAVE_COPY_COMPLETED;
+ setNodeCopyCompleted(takeOverPtr.p->toStartingNode, true);
+ }//if
+
+
+ UpdateToConf * const conf = (UpdateToConf *)&signal->theData[0];
+ conf->userPtr = takeOverPtr.i;
+ conf->sendingNodeId = cownNodeId;
+ conf->startingNodeId = takeOverPtr.p->toStartingNode;
+ sendSignal(ref, GSN_UPDATE_TOCONF, signal, UpdateToConf::SignalLength, JBB);
+}//Dbdih::execUPDATE_TOREQ()
+
+void Dbdih::execUPDATE_TOCONF(Signal* signal)
+{
+ const UpdateToConf * const conf = (UpdateToConf *)&signal->theData[0];
+ CRASH_INSERTION(7152);
+
+ RETURN_IF_NODE_NOT_ALIVE(conf->startingNodeId);
+
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = conf->userPtr;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ receiveLoopMacro(UPDATE_TOREQ, conf->sendingNodeId);
+ CRASH_INSERTION(7153);
+ c_updateToLock = RNIL;
+
+ if (takeOverPtr.p->toMasterStatus == TakeOverRecord::TO_COPY_COMPLETED) {
+ jam();
+ toCopyCompletedLab(signal, takeOverPtr);
+ return;
+ } else {
+ ndbrequire(takeOverPtr.p->toMasterStatus == TakeOverRecord::TO_UPDATE_TO);
+ }//if
+ TabRecordPtr tabPtr;
+ tabPtr.i = takeOverPtr.p->toCurrentTabref;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, takeOverPtr.p->toCurrentFragid, fragPtr);
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::COPY_ACTIVE;
+ BlockReference lqhRef = calcLqhBlockRef(takeOverPtr.p->toStartingNode);
+ CopyActiveReq * const req = (CopyActiveReq *)&signal->theData[0];
+ req->userPtr = takeOverPtr.i;
+ req->userRef = reference();
+ req->tableId = takeOverPtr.p->toCurrentTabref;
+ req->fragId = takeOverPtr.p->toCurrentFragid;
+ req->distributionKey = fragPtr.p->distributionKey;
+
+ sendSignal(lqhRef, GSN_COPY_ACTIVEREQ, signal,
+ CopyActiveReq::SignalLength, JBB);
+}//Dbdih::execUPDATE_TOCONF()
+
+void Dbdih::execCOPY_ACTIVECONF(Signal* signal)
+{
+ const CopyActiveConf * const conf = (CopyActiveConf *)&signal->theData[0];
+ jamEntry();
+ CRASH_INSERTION(7143);
+
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = conf->userPtr;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ ndbrequire(conf->tableId == takeOverPtr.p->toCurrentTabref);
+ ndbrequire(conf->fragId == takeOverPtr.p->toCurrentFragid);
+ ndbrequire(checkNodeAlive(conf->startingNodeId));
+ ndbrequire(takeOverPtr.p->toMasterStatus == TakeOverRecord::COPY_ACTIVE);
+
+ takeOverPtr.p->startGci = conf->startGci;
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::LOCK_MUTEX;
+
+ Mutex mutex(signal, c_mutexMgr, takeOverPtr.p->m_switchPrimaryMutexHandle);
+ Callback c = { safe_cast(&Dbdih::switchPrimaryMutex_locked), takeOverPtr.i };
+ ndbrequire(mutex.lock(c));
+}//Dbdih::execCOPY_ACTIVECONF()
+
+void
+Dbdih::switchPrimaryMutex_locked(Signal* signal, Uint32 toPtrI, Uint32 retVal){
+ jamEntry();
+ ndbrequire(retVal == 0);
+
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = toPtrI;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ ndbrequire(takeOverPtr.p->toMasterStatus == TakeOverRecord::LOCK_MUTEX);
+
+ if (!checkNodeAlive((takeOverPtr.p->toStartingNode))) {
+ // We have mutex
+ Mutex mutex(signal, c_mutexMgr, takeOverPtr.p->m_switchPrimaryMutexHandle);
+ mutex.unlock(); // Ignore result
+
+ c_createFragmentLock = RNIL;
+ c_CREATE_FRAGREQ_Counter.clearWaitingFor();
+ endTakeOver(takeOverPtr.i);
+ return;
+ }
+
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::COMMIT_CREATE;
+ sendCreateFragReq(signal, takeOverPtr.p->startGci,
+ CreateFragReq::COMMIT_STORED, takeOverPtr.i);
+}
+
+void Dbdih::toCopyCompletedLab(Signal * signal, TakeOverRecordPtr takeOverPtr)
+{
+ signal->theData[0] = EventReport::NR_CopyFragsCompleted;
+ signal->theData[1] = takeOverPtr.p->toStartingNode;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ c_lcpState.immediateLcpStart = true;
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::WAIT_LCP;
+
+ /*-----------------------------------------------------------------------*/
+ /* NOW WE CAN ALLOW THE NEW NODE TO PARTICIPATE IN LOCAL CHECKPOINTS. */
+ /* WHEN THE FIRST LOCAL CHECKPOINT IS READY WE DECLARE THE TAKE OVER AS */
+ /* COMPLETED. SINCE LOCAL CHECKPOINTS HAVE BEEN BLOCKED DURING THE COPY */
+ /* PROCESS WE MUST ALSO START A NEW LOCAL CHECKPOINT PROCESS BY ENSURING */
+ /* THAT IT LOOKS LIKE IT IS TIME FOR A NEW LOCAL CHECKPOINT AND BY */
+ /* UNBLOCKING THE LOCAL CHECKPOINT AGAIN. */
+ /* --------------------------------------------------------------------- */
+}//Dbdih::toCopyCompletedLab()
+
+void Dbdih::sendEndTo(Signal* signal, Uint32 takeOverPtrI)
+{
+ TakeOverRecordPtr takeOverPtr;
+ CRASH_INSERTION(7156);
+ RETURN_IF_TAKE_OVER_INTERRUPTED(takeOverPtrI, takeOverPtr);
+ if ((c_endToLock != RNIL) || (ERROR_INSERTED(7164))) {
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_WAIT_ENDING;
+ signal->theData[0] = DihContinueB::ZSEND_END_TO;
+ signal->theData[1] = takeOverPtrI;
+ signal->theData[2] = takeOverPtr.p->toStartingNode;
+ signal->theData[3] = takeOverPtr.p->toFailedNode;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 30, 4);
+ return;
+ }//if
+ c_endToLock = takeOverPtr.i;
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::ENDING;
+ EndToReq * const req = (EndToReq *)&signal->theData[0];
+ req->userPtr = takeOverPtr.i;
+ req->userRef = reference();
+ req->startingNodeId = takeOverPtr.p->toStartingNode;
+ sendLoopMacro(END_TOREQ, sendEND_TOREQ);
+}//Dbdih::sendStartTo()
+
+void Dbdih::execEND_TOREQ(Signal* signal)
+{
+ jamEntry();
+ const EndToReq * const req = (EndToReq *)&signal->theData[0];
+ BlockReference ref = req->userRef;
+ Uint32 startingNodeId = req->startingNodeId;
+
+ CRASH_INSERTION(7144);
+ RETURN_IF_NODE_NOT_ALIVE(startingNodeId);
+
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = req->userPtr;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ ndbrequire(startingNodeId == takeOverPtr.p->toStartingNode);
+ takeOverPtr.p->toSlaveStatus = TakeOverRecord::TO_SLAVE_IDLE;
+
+ if (!isMaster()) {
+ jam();
+ endTakeOver(takeOverPtr.i);
+ }//if
+
+ EndToConf * const conf = (EndToConf *)&signal->theData[0];
+ conf->userPtr = takeOverPtr.i;
+ conf->sendingNodeId = cownNodeId;
+ conf->startingNodeId = startingNodeId;
+ sendSignal(ref, GSN_END_TOCONF, signal, EndToConf::SignalLength, JBB);
+}//Dbdih::execEND_TOREQ()
+
+void Dbdih::execEND_TOCONF(Signal* signal)
+{
+ const EndToConf * const conf = (EndToConf *)&signal->theData[0];
+ jamEntry();
+
+ const Uint32 nodeId = conf->startingNodeId;
+ CRASH_INSERTION(7145);
+
+ RETURN_IF_NODE_NOT_ALIVE(nodeId);
+
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = conf->userPtr;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ ndbrequire(takeOverPtr.p->toMasterStatus == TakeOverRecord::ENDING);
+ ndbrequire(nodeId == takeOverPtr.p->toStartingNode);
+
+ receiveLoopMacro(END_TOREQ, conf->sendingNodeId);
+ CRASH_INSERTION(7146);
+ c_endToLock = RNIL;
+
+ /* -----------------------------------------------------------------------*/
+ /* WE HAVE FINALLY COMPLETED THE TAKE OVER. WE RESET THE STATUS AND CHECK*/
+ /* IF ANY MORE TAKE OVERS ARE NEEDED AT THE MOMENT. */
+ /* FIRST WE CHECK IF A RESTART IS ONGOING. IN THAT CASE WE RESTART PHASE */
+ /* 4 AND CHECK IF ANY MORE TAKE OVERS ARE NEEDED BEFORE WE START NDB */
+ /* CLUSTER. THIS CAN ONLY HAPPEN IN A SYSTEM RESTART. */
+ /* ---------------------------------------------------------------------- */
+ if (takeOverPtr.p->toNodeRestart) {
+ jam();
+ /* ----------------------------------------------------------------------*/
+ /* THE TAKE OVER NODE WAS A STARTING NODE. WE WILL SEND START_COPYCONF */
+ /* TO THE STARTING NODE SUCH THAT THE NODE CAN COMPLETE THE START-UP. */
+ /* --------------------------------------------------------------------- */
+ BlockReference ref = calcDihBlockRef(takeOverPtr.p->toStartingNode);
+ signal->theData[0] = takeOverPtr.p->toStartingNode;
+ sendSignal(ref, GSN_START_COPYCONF, signal, 1,JBB);
+ }//if
+ endTakeOver(takeOverPtr.i);
+
+ ndbout_c("2 - endTakeOver");
+ if (cstartPhase == ZNDB_SPH4) {
+ jam();
+ ndbrequire(false);
+ if (anyActiveTakeOver()) {
+ jam();
+ ndbout_c("4 - anyActiveTakeOver == true");
+ return;
+ }//if
+ ndbout_c("5 - anyActiveTakeOver == false -> ndbsttorry10Lab");
+ ndbsttorry10Lab(signal, __LINE__);
+ return;
+ }//if
+ checkStartTakeOver(signal);
+}//Dbdih::execEND_TOCONF()
+
+void Dbdih::allocateTakeOver(TakeOverRecordPtr& takeOverPtr)
+{
+ if (isMaster()) {
+ jam();
+ //--------------------------------------------
+ // Master already seized the take over record.
+ //--------------------------------------------
+ return;
+ }//if
+ if (takeOverPtr.i == cfirstfreeTakeOver) {
+ jam();
+ seizeTakeOver(takeOverPtr);
+ } else {
+ TakeOverRecordPtr nextTakeOverptr;
+ TakeOverRecordPtr prevTakeOverptr;
+ nextTakeOverptr.i = takeOverPtr.p->nextTakeOver;
+ prevTakeOverptr.i = takeOverPtr.p->prevTakeOver;
+ if (prevTakeOverptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(prevTakeOverptr, MAX_NDB_NODES, takeOverRecord);
+ prevTakeOverptr.p->nextTakeOver = nextTakeOverptr.i;
+ }//if
+ if (nextTakeOverptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(nextTakeOverptr, MAX_NDB_NODES, takeOverRecord);
+ nextTakeOverptr.p->prevTakeOver = prevTakeOverptr.i;
+ }//if
+ }//if
+}//Dbdih::allocateTakeOver()
+
+void Dbdih::seizeTakeOver(TakeOverRecordPtr& takeOverPtr)
+{
+ TakeOverRecordPtr nextTakeOverptr;
+ ndbrequire(cfirstfreeTakeOver != RNIL);
+ takeOverPtr.i = cfirstfreeTakeOver;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ cfirstfreeTakeOver = takeOverPtr.p->nextTakeOver;
+ nextTakeOverptr.i = takeOverPtr.p->nextTakeOver;
+ if (nextTakeOverptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(nextTakeOverptr, MAX_NDB_NODES, takeOverRecord);
+ nextTakeOverptr.p->prevTakeOver = RNIL;
+ }//if
+ takeOverPtr.p->nextTakeOver = RNIL;
+ takeOverPtr.p->prevTakeOver = RNIL;
+}//Dbdih::seizeTakeOver()
+
+void Dbdih::endTakeOver(Uint32 takeOverPtrI)
+{
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = takeOverPtrI;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ releaseTakeOver(takeOverPtrI);
+ if ((takeOverPtr.p->toMasterStatus != TakeOverRecord::IDLE) &&
+ (takeOverPtr.p->toMasterStatus != TakeOverRecord::TO_WAIT_START_TAKE_OVER)) {
+ jam();
+ NodeGroupRecordPtr NGPtr;
+ NodeRecordPtr nodePtr;
+ nodePtr.i = takeOverPtr.p->toStartingNode;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ NGPtr.i = nodePtr.p->nodeGroup;
+ ptrCheckGuard(NGPtr, MAX_NDB_NODES, nodeGroupRecord);
+ NGPtr.p->activeTakeOver = false;
+ }//if
+ setAllowNodeStart(takeOverPtr.p->toStartingNode, true);
+ initTakeOver(takeOverPtr);
+}//Dbdih::endTakeOver()
+
+void Dbdih::releaseTakeOver(Uint32 takeOverPtrI)
+{
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = takeOverPtrI;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ takeOverPtr.p->nextTakeOver = cfirstfreeTakeOver;
+ cfirstfreeTakeOver = takeOverPtr.i;
+}//Dbdih::releaseTakeOver()
+
+void Dbdih::initTakeOver(TakeOverRecordPtr takeOverPtr)
+{
+ takeOverPtr.p->toCopyNode = RNIL;
+ takeOverPtr.p->toCurrentFragid = RNIL;
+ takeOverPtr.p->toCurrentReplica = RNIL;
+ takeOverPtr.p->toCurrentTabref = RNIL;
+ takeOverPtr.p->toFailedNode = RNIL;
+ takeOverPtr.p->toStartingNode = RNIL;
+ takeOverPtr.p->prevTakeOver = RNIL;
+ takeOverPtr.p->nextTakeOver = RNIL;
+ takeOverPtr.p->toNodeRestart = false;
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::IDLE;
+ takeOverPtr.p->toSlaveStatus = TakeOverRecord::TO_SLAVE_IDLE;
+}//Dbdih::initTakeOver()
+
+bool Dbdih::anyActiveTakeOver()
+{
+ TakeOverRecordPtr takeOverPtr;
+ for (takeOverPtr.i = 0; takeOverPtr.i < MAX_NDB_NODES; takeOverPtr.i++) {
+ ptrAss(takeOverPtr, takeOverRecord);
+ if (takeOverPtr.p->toMasterStatus != TakeOverRecord::IDLE) {
+ jam();
+ return true;
+ }//if
+ }//for
+ return false;
+}//Dbdih::anyActiveTakeOver()
+
+/*****************************************************************************/
+/* ------------------------------------------------------------------------- */
+/* WE HAVE BEEN REQUESTED TO PERFORM A SYSTEM RESTART. WE START BY */
+/* READING THE GCI FILES. THIS REQUEST WILL ONLY BE SENT TO THE MASTER */
+/* DIH. THAT MEANS WE HAVE TO REPLICATE THE INFORMATION WE READ FROM */
+/* OUR FILES TO ENSURE THAT ALL NODES HAVE THE SAME DISTRIBUTION */
+/* INFORMATION. */
+/* ------------------------------------------------------------------------- */
+/*****************************************************************************/
+void Dbdih::readGciFileLab(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ filePtr.i = crestartInfoFile[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ filePtr.p->reqStatus = FileRecord::OPENING_GCP;
+ openFileRw(signal, filePtr);
+}//Dbdih::readGciFileLab()
+
+void Dbdih::openingGcpLab(Signal* signal, FileRecordPtr filePtr)
+{
+ /* ----------------------------------------------------------------------- */
+ /* WE HAVE SUCCESSFULLY OPENED A FILE CONTAINING INFORMATION ABOUT */
+ /* THE GLOBAL CHECKPOINTS THAT ARE POSSIBLE TO RESTART. */
+ /* ----------------------------------------------------------------------- */
+ readRestorableGci(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::READING_GCP;
+}//Dbdih::openingGcpLab()
+
+void Dbdih::readingGcpLab(Signal* signal, FileRecordPtr filePtr)
+{
+ /* ----------------------------------------------------------------------- */
+ /* WE HAVE NOW SUCCESSFULLY MANAGED TO READ IN THE GLOBAL CHECKPOINT */
+ /* INFORMATION FROM FILE. LATER WE WILL ADD SOME FUNCTIONALITY THAT */
+ /* CHECKS THE RESTART TIMERS TO DEDUCE FROM WHERE TO RESTART. */
+ /* NOW WE WILL SIMPLY RESTART FROM THE NEWEST GLOBAL CHECKPOINT */
+ /* POSSIBLE TO RESTORE. */
+ /* */
+ /* BEFORE WE INVOKE DICT WE NEED TO COPY CRESTART_INFO TO ALL NODES. */
+ /* WE ALSO COPY TO OUR OWN NODE. TO ENABLE US TO DO THIS PROPERLY WE */
+ /* START BY CLOSING THIS FILE. */
+ /* ----------------------------------------------------------------------- */
+ closeFile(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::CLOSING_GCP;
+}//Dbdih::readingGcpLab()
+
+void Dbdih::closingGcpLab(Signal* signal, FileRecordPtr filePtr)
+{
+ if (Sysfile::getInitialStartOngoing(SYSFILE->systemRestartBits) == false){
+ jam();
+ selectMasterCandidateAndSend(signal);
+ return;
+ } else {
+ jam();
+ sendSignal(cntrlblockref, GSN_DIH_RESTARTREF, signal, 1, JBB);
+ return;
+ }//if
+}//Dbdih::closingGcpLab()
+
+/* ------------------------------------------------------------------------- */
+/* SELECT THE MASTER CANDIDATE TO BE USED IN SYSTEM RESTARTS. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::selectMasterCandidateAndSend(Signal* signal)
+{
+ Uint32 gci = 0;
+ Uint32 masterCandidateId = 0;
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if (SYSFILE->lastCompletedGCI[nodePtr.i] > gci) {
+ jam();
+ masterCandidateId = nodePtr.i;
+ gci = SYSFILE->lastCompletedGCI[nodePtr.i];
+ }//if
+ }//for
+ ndbrequire(masterCandidateId != 0);
+ signal->theData[0] = masterCandidateId;
+ signal->theData[1] = gci;
+ sendSignal(cntrlblockref, GSN_DIH_RESTARTCONF, signal, 2, JBB);
+}//Dbdih::selectMasterCandidate()
+
+/* ------------------------------------------------------------------------- */
+/* ERROR HANDLING DURING READING RESTORABLE GCI FROM FILE. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::openingGcpErrorLab(Signal* signal, FileRecordPtr filePtr)
+{
+ filePtr.p->fileStatus = FileRecord::CRASHED;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ if (crestartInfoFile[0] == filePtr.i) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* THE FIRST FILE WAS NOT ABLE TO BE OPENED. SET STATUS TO CRASHED AND */
+ /* TRY OPEN THE NEXT FILE. */
+ /* --------------------------------------------------------------------- */
+ filePtr.i = crestartInfoFile[1];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ openFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::OPENING_GCP;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* WE FAILED IN OPENING THE SECOND FILE. BOTH FILES WERE CORRUPTED. WE */
+ /* CANNOT CONTINUE THE RESTART IN THIS CASE. TELL NDBCNTR OF OUR */
+ /* FAILURE. */
+ /*---------------------------------------------------------------------- */
+ sendSignal(cntrlblockref, GSN_DIH_RESTARTREF, signal, 1, JBB);
+ return;
+ }//if
+}//Dbdih::openingGcpErrorLab()
+
+void Dbdih::readingGcpErrorLab(Signal* signal, FileRecordPtr filePtr)
+{
+ filePtr.p->fileStatus = FileRecord::CRASHED;
+ /* ----------------------------------------------------------------------- */
+ /* WE FAILED IN READING THE FILE AS WELL. WE WILL CLOSE THIS FILE. */
+ /* ----------------------------------------------------------------------- */
+ closeFile(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::CLOSING_GCP_CRASH;
+}//Dbdih::readingGcpErrorLab()
+
+void Dbdih::closingGcpCrashLab(Signal* signal, FileRecordPtr filePtr)
+{
+ if (crestartInfoFile[0] == filePtr.i) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* ERROR IN FIRST FILE, TRY THE SECOND FILE. */
+ /* --------------------------------------------------------------------- */
+ filePtr.i = crestartInfoFile[1];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ openFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::OPENING_GCP;
+ return;
+ }//if
+ /* ----------------------------------------------------------------------- */
+ /* WE DISCOVERED A FAILURE WITH THE SECOND FILE AS WELL. THIS IS A */
+ /* SERIOUS PROBLEM. REPORT FAILURE TO NDBCNTR. */
+ /* ----------------------------------------------------------------------- */
+ sendSignal(cntrlblockref, GSN_DIH_RESTARTREF, signal, 1, JBB);
+}//Dbdih::closingGcpCrashLab()
+
+/*****************************************************************************/
+/* ------------------------------------------------------------------------- */
+/* THIS IS AN INITIAL RESTART. WE WILL CREATE THE TWO FILES DESCRIBING */
+/* THE GLOBAL CHECKPOINTS THAT ARE RESTORABLE. */
+/* ------------------------------------------------------------------------- */
+/*****************************************************************************/
+void Dbdih::initGciFilesLab(Signal* signal)
+{
+ FileRecordPtr filePtr;
+ filePtr.i = crestartInfoFile[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ createFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::CREATING_GCP;
+}//Dbdih::initGciFilesLab()
+
+/* ------------------------------------------------------------------------- */
+/* GLOBAL CHECKPOINT FILE HAVE BEEN SUCCESSFULLY CREATED. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::creatingGcpLab(Signal* signal, FileRecordPtr filePtr)
+{
+ if (filePtr.i == crestartInfoFile[0]) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* IF CREATED FIRST THEN ALSO CREATE THE SECOND FILE. */
+ /* --------------------------------------------------------------------- */
+ filePtr.i = crestartInfoFile[1];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ createFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::CREATING_GCP;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* BOTH FILES HAVE BEEN CREATED. NOW WRITE THE INITIAL DATA TO BOTH */
+ /* OF THE FILES. */
+ /* --------------------------------------------------------------------- */
+ filePtr.i = crestartInfoFile[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ writeRestorableGci(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::WRITE_INIT_GCP;
+ }//if
+}//Dbdih::creatingGcpLab()
+
+/* ------------------------------------------------------------------------- */
+/* WE HAVE SUCCESSFULLY WRITTEN A GCI FILE. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::writeInitGcpLab(Signal* signal, FileRecordPtr filePtr)
+{
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ if (filePtr.i == crestartInfoFile[0]) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* WE HAVE WRITTEN THE FIRST FILE NOW ALSO WRITE THE SECOND FILE. */
+ /* --------------------------------------------------------------------- */
+ filePtr.i = crestartInfoFile[1];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ writeRestorableGci(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::WRITE_INIT_GCP;
+ } else {
+ /* --------------------------------------------------------------------- */
+ /* WE HAVE WRITTEN BOTH FILES. LEAVE BOTH FILES OPEN AND CONFIRM OUR */
+ /* PART OF THE INITIAL START. */
+ /* --------------------------------------------------------------------- */
+ if (isMaster()) {
+ jam();
+ /*---------------------------------------------------------------------*/
+ // IN MASTER NODES THE START REQUEST IS RECEIVED FROM NDBCNTR AND WE MUST
+ // RESPOND WHEN COMPLETED.
+ /*---------------------------------------------------------------------*/
+ signal->theData[0] = reference();
+ sendSignal(cndbStartReqBlockref, GSN_NDB_STARTCONF, signal, 1, JBB);
+ } else {
+ jam();
+ ndbsttorry10Lab(signal, __LINE__);
+ return;
+ }//if
+ }//if
+}//Dbdih::writeInitGcpLab()
+
+/*****************************************************************************/
+/* ********** NODES DELETION MODULE *************/
+/*****************************************************************************/
+/*---------------------------------------------------------------------------*/
+/* LOGIC FOR NODE FAILURE */
+/*---------------------------------------------------------------------------*/
+void Dbdih::execNODE_FAILREP(Signal* signal)
+{
+ Uint32 failedNodes[MAX_NDB_NODES];
+ jamEntry();
+ NodeFailRep * const nodeFail = (NodeFailRep *)&signal->theData[0];
+
+ cfailurenr = nodeFail->failNo;
+ Uint32 newMasterId = nodeFail->masterNodeId;
+ const Uint32 noOfFailedNodes = nodeFail->noOfNodes;
+
+ /*-------------------------------------------------------------------------*/
+ // The first step is to convert from a bit mask to an array of failed nodes.
+ /*-------------------------------------------------------------------------*/
+ Uint32 index = 0;
+ for (Uint32 i = 1; i < MAX_NDB_NODES; i++) {
+ jam();
+ if(NodeBitmask::get(nodeFail->theNodes, i)){
+ jam();
+ failedNodes[index] = i;
+ index++;
+ }//if
+ }//for
+ ndbrequire(noOfFailedNodes == index);
+ ndbrequire(noOfFailedNodes - 1 < MAX_NDB_NODES);
+
+ /*-------------------------------------------------------------------------*/
+ // The second step is to update the node status of the failed nodes, remove
+ // them from the alive node list and put them into the dead node list. Also
+ // update the number of nodes on-line.
+ // We also set certain state variables ensuring that the node no longer is
+ // used in transactions and also mark that we received this signal.
+ /*-------------------------------------------------------------------------*/
+ for (Uint32 i = 0; i < noOfFailedNodes; i++) {
+ jam();
+ NodeRecordPtr TNodePtr;
+ TNodePtr.i = failedNodes[i];
+ ptrCheckGuard(TNodePtr, MAX_NDB_NODES, nodeRecord);
+ TNodePtr.p->useInTransactions = false;
+ TNodePtr.p->m_inclDihLcp = false;
+ TNodePtr.p->recNODE_FAILREP = ZTRUE;
+ if (TNodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ jam();
+ con_lineNodes--;
+ TNodePtr.p->nodeStatus = NodeRecord::DIED_NOW;
+ removeAlive(TNodePtr);
+ insertDeadNode(TNodePtr);
+ }//if
+ }//for
+
+ /*-------------------------------------------------------------------------*/
+ // Verify that we can continue to operate the cluster. If we cannot we will
+ // not return from checkEscalation.
+ /*-------------------------------------------------------------------------*/
+ checkEscalation();
+
+ /*------------------------------------------------------------------------*/
+ // Verify that a starting node has also crashed. Reset the node start record.
+ /*-------------------------------------------------------------------------*/
+ if (c_nodeStartMaster.startNode != RNIL) {
+ ndbrequire(getNodeStatus(c_nodeStartMaster.startNode)!= NodeRecord::ALIVE);
+ }//if
+
+ /*--------------------------------------------------*/
+ /* */
+ /* WE CHANGE THE REFERENCE TO MASTER DIH */
+ /* BLOCK AND POINTER AT THIS PLACE IN THE CODE*/
+ /*--------------------------------------------------*/
+ Uint32 oldMasterId = cmasterNodeId;
+ BlockReference oldMasterRef = cmasterdihref;
+ cmasterdihref = calcDihBlockRef(newMasterId);
+ cmasterNodeId = newMasterId;
+
+ const bool masterTakeOver = (oldMasterId != newMasterId);
+
+ for(Uint32 i = 0; i < noOfFailedNodes; i++) {
+ NodeRecordPtr failedNodePtr;
+ failedNodePtr.i = failedNodes[i];
+ ptrCheckGuard(failedNodePtr, MAX_NDB_NODES, nodeRecord);
+ Uint32 activeTakeOverPtr = findTakeOver(failedNodes[i]);
+ if (oldMasterRef == reference()) {
+ /*-------------------------------------------------------*/
+ // Functions that need to be called only for master nodes.
+ /*-------------------------------------------------------*/
+ checkCopyTab(failedNodePtr);
+ checkStopPermMaster(signal, failedNodePtr);
+ checkWaitGCPMaster(signal, failedNodes[i]);
+ checkTakeOverInMasterAllNodeFailure(signal, failedNodePtr);
+ checkTakeOverInMasterCopyNodeFailure(signal, failedNodePtr.i);
+ checkTakeOverInMasterStartNodeFailure(signal, activeTakeOverPtr);
+ checkGcpOutstanding(signal, failedNodePtr.i);
+ } else {
+ jam();
+ /*-----------------------------------------------------------*/
+ // Functions that need to be called only for nodes that were
+ // not master before these failures.
+ /*-----------------------------------------------------------*/
+ checkStopPermProxy(signal, failedNodes[i]);
+ checkWaitGCPProxy(signal, failedNodes[i]);
+ if (isMaster()) {
+ /*-----------------------------------------------------------*/
+ // We take over as master since old master has failed
+ /*-----------------------------------------------------------*/
+ handleTakeOverNewMaster(signal, activeTakeOverPtr);
+ } else {
+ /*-----------------------------------------------------------*/
+ // We are not master and will not become master.
+ /*-----------------------------------------------------------*/
+ checkTakeOverInNonMasterStartNodeFailure(signal, activeTakeOverPtr);
+ }//if
+ }//if
+ /*--------------------------------------------------*/
+ // Functions that need to be called for all nodes.
+ /*--------------------------------------------------*/
+ checkStopMe(signal, failedNodePtr);
+ failedNodeLcpHandling(signal, failedNodePtr);
+ checkWaitDropTabFailedLqh(signal, failedNodePtr.i, 0); // 0 = start w/ tab 0
+ startRemoveFailedNode(signal, failedNodePtr);
+
+ /**
+ * This is the last function called
+ * It modifies failedNodePtr.p->nodeStatus
+ */
+ failedNodeSynchHandling(signal, failedNodePtr);
+ }//for
+
+ if(masterTakeOver){
+ jam();
+ startLcpMasterTakeOver(signal, oldMasterId);
+ startGcpMasterTakeOver(signal, oldMasterId);
+
+ if(getNodeState().getNodeRestartInProgress()){
+ jam();
+ progError(__LINE__,
+ ERR_SYSTEM_ERROR,
+ "Unhandle master failure during node restart");
+ }
+ }
+
+
+ if (isMaster()) {
+ jam();
+ setNodeRestartInfoBits();
+ }//if
+}//Dbdih::execNODE_FAILREP()
+
+void Dbdih::checkCopyTab(NodeRecordPtr failedNodePtr)
+{
+ jam();
+
+ if(c_nodeStartMaster.startNode != failedNodePtr.i){
+ jam();
+ return;
+ }
+
+ switch(c_nodeStartMaster.m_outstandingGsn){
+ case GSN_COPY_TABREQ:
+ jam();
+ ndbrequire(c_COPY_TABREQ_Counter.isWaitingFor(failedNodePtr.i));
+ releaseTabPages(failedNodePtr.p->activeTabptr);
+ c_COPY_TABREQ_Counter.clearWaitingFor(failedNodePtr.i);
+ c_nodeStartMaster.wait = ZFALSE;
+ break;
+ case GSN_START_PERMCONF:
+ case GSN_DICTSTARTREQ:
+ case GSN_START_MECONF:
+ jam();
+ break;
+ default:
+ ndbout_c("outstanding gsn: %s(%d)",
+ getSignalName(c_nodeStartMaster.m_outstandingGsn),
+ c_nodeStartMaster.m_outstandingGsn);
+ ndbrequire(false);
+ }
+
+ nodeResetStart();
+}//Dbdih::checkCopyTab()
+
+void Dbdih::checkStopMe(Signal* signal, NodeRecordPtr failedNodePtr)
+{
+ jam();
+ if (c_STOP_ME_REQ_Counter.isWaitingFor(failedNodePtr.i)){
+ jam();
+ ndbrequire(c_stopMe.clientRef != 0);
+ StopMeConf * const stopMeConf = (StopMeConf *)&signal->theData[0];
+ stopMeConf->senderRef = calcDihBlockRef(failedNodePtr.i);
+ stopMeConf->senderData = c_stopMe.clientData;
+ sendSignal(reference(), GSN_STOP_ME_CONF, signal,
+ StopMeConf::SignalLength, JBB);
+ }//if
+}//Dbdih::checkStopMe()
+
+void Dbdih::checkStopPermMaster(Signal* signal, NodeRecordPtr failedNodePtr)
+{
+ DihSwitchReplicaRef* const ref = (DihSwitchReplicaRef*)&signal->theData[0];
+ jam();
+ if (c_DIH_SWITCH_REPLICA_REQ_Counter.isWaitingFor(failedNodePtr.i)){
+ jam();
+ ndbrequire(c_stopPermMaster.clientRef != 0);
+ ref->senderNode = failedNodePtr.i;
+ ref->errorCode = StopPermRef::NF_CausedAbortOfStopProcedure;
+ sendSignal(reference(), GSN_DIH_SWITCH_REPLICA_REF, signal,
+ DihSwitchReplicaRef::SignalLength, JBB);
+ return;
+ }//if
+}//Dbdih::checkStopPermMaster()
+
+void Dbdih::checkStopPermProxy(Signal* signal, NodeId failedNodeId)
+{
+ jam();
+ if(c_stopPermProxy.clientRef != 0 &&
+ refToNode(c_stopPermProxy.masterRef) == failedNodeId){
+
+ /**
+ * The master has failed report to proxy-client
+ */
+ jam();
+ StopPermRef* const ref = (StopPermRef*)&signal->theData[0];
+
+ ref->senderData = c_stopPermProxy.clientData;
+ ref->errorCode = StopPermRef::NF_CausedAbortOfStopProcedure;
+ sendSignal(c_stopPermProxy.clientRef, GSN_STOP_PERM_REF, signal, 2, JBB);
+ c_stopPermProxy.clientRef = 0;
+ }//if
+}//Dbdih::checkStopPermProxy()
+
+void
+Dbdih::checkTakeOverInMasterAllNodeFailure(Signal* signal,
+ NodeRecordPtr failedNodePtr)
+{
+ //------------------------------------------------------------------------
+ // This code is used to handle the failure of "all" nodes during the
+ // take over when "all" nodes are informed about state changes in
+ // the take over protocol.
+ //--------------------------------------------------------------------------
+ if (c_START_TOREQ_Counter.isWaitingFor(failedNodePtr.i)){
+ jam();
+ StartToConf * const conf = (StartToConf *)&signal->theData[0];
+ conf->userPtr = c_startToLock;
+ conf->sendingNodeId = failedNodePtr.i;
+ conf->startingNodeId = getStartNode(c_startToLock);
+ sendSignal(reference(), GSN_START_TOCONF, signal,
+ StartToConf::SignalLength, JBB);
+ }//if
+ if (c_CREATE_FRAGREQ_Counter.isWaitingFor(failedNodePtr.i)){
+ jam();
+ CreateFragConf * const conf = (CreateFragConf *)&signal->theData[0];
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = c_createFragmentLock;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ conf->userPtr = takeOverPtr.i;
+ conf->tableId = takeOverPtr.p->toCurrentTabref;
+ conf->fragId = takeOverPtr.p->toCurrentFragid;
+ conf->sendingNodeId = failedNodePtr.i;
+ conf->startingNodeId = takeOverPtr.p->toStartingNode;
+ sendSignal(reference(), GSN_CREATE_FRAGCONF, signal,
+ CreateFragConf::SignalLength, JBB);
+ }//if
+ if (c_UPDATE_TOREQ_Counter.isWaitingFor(failedNodePtr.i)){
+ jam();
+ UpdateToConf * const conf = (UpdateToConf *)&signal->theData[0];
+ conf->userPtr = c_updateToLock;
+ conf->sendingNodeId = failedNodePtr.i;
+ conf->startingNodeId = getStartNode(c_updateToLock);
+ sendSignal(reference(), GSN_UPDATE_TOCONF, signal,
+ UpdateToConf::SignalLength, JBB);
+ }//if
+
+ if (c_END_TOREQ_Counter.isWaitingFor(failedNodePtr.i)){
+ jam();
+ EndToConf * const conf = (EndToConf *)&signal->theData[0];
+ conf->userPtr = c_endToLock;
+ conf->sendingNodeId = failedNodePtr.i;
+ conf->startingNodeId = getStartNode(c_endToLock);
+ sendSignal(reference(), GSN_END_TOCONF, signal,
+ EndToConf::SignalLength, JBB);
+ }//if
+}//Dbdih::checkTakeOverInMasterAllNodeFailure()
+
+void Dbdih::checkTakeOverInMasterCopyNodeFailure(Signal* signal,
+ Uint32 failedNodeId)
+{
+ //---------------------------------------------------------------------------
+ // This code is used to handle failure of the copying node during a take over
+ //---------------------------------------------------------------------------
+ TakeOverRecordPtr takeOverPtr;
+ for (Uint32 i = 0; i < MAX_NDB_NODES; i++) {
+ jam();
+ takeOverPtr.i = i;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ if ((takeOverPtr.p->toMasterStatus == TakeOverRecord::COPY_FRAG) &&
+ (takeOverPtr.p->toCopyNode == failedNodeId)) {
+ jam();
+ /**
+ * The copying node failed but the system is still operational.
+ * We restart the copy process by selecting a new copy node.
+ * We do not need to add a fragment however since it is already added.
+ * We start again from the prepare create fragment phase.
+ */
+ prepareSendCreateFragReq(signal, takeOverPtr.i);
+ }//if
+ }//for
+}//Dbdih::checkTakeOverInMasterCopyNodeFailure()
+
+void Dbdih::checkTakeOverInMasterStartNodeFailure(Signal* signal,
+ Uint32 takeOverPtrI)
+{
+ jam();
+ if (takeOverPtrI == RNIL) {
+ jam();
+ return;
+ }
+ //-----------------------------------------------------------------------
+ // We are the master and the starting node has failed during a take over.
+ // We need to handle this failure in different ways depending on the state.
+ //-----------------------------------------------------------------------
+
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = takeOverPtrI;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+
+ bool ok = false;
+ switch (takeOverPtr.p->toMasterStatus) {
+ case TakeOverRecord::IDLE:
+ //-----------------------------------------------------------------------
+ // The state cannot be idle when it has a starting node.
+ //-----------------------------------------------------------------------
+ ndbrequire(false);
+ break;
+ case TakeOverRecord::TO_WAIT_START_TAKE_OVER:
+ jam();
+ case TakeOverRecord::TO_START_COPY:
+ jam();
+ case TakeOverRecord::TO_START_COPY_ONGOING:
+ jam();
+ case TakeOverRecord::TO_WAIT_START:
+ jam();
+ case TakeOverRecord::TO_WAIT_PREPARE_CREATE:
+ jam();
+ case TakeOverRecord::TO_WAIT_UPDATE_TO:
+ jam();
+ case TakeOverRecord::TO_WAIT_COMMIT_CREATE:
+ jam();
+ case TakeOverRecord::TO_END_COPY:
+ jam();
+ case TakeOverRecord::TO_END_COPY_ONGOING:
+ jam();
+ case TakeOverRecord::TO_WAIT_ENDING:
+ jam();
+ //-----------------------------------------------------------------------
+ // We will not do anything since an internal signal process is outstanding.
+ // When the signal arrives the take over will be released.
+ //-----------------------------------------------------------------------
+ ok = true;
+ break;
+ case TakeOverRecord::STARTING:
+ jam();
+ ok = true;
+ c_startToLock = RNIL;
+ c_START_TOREQ_Counter.clearWaitingFor();
+ endTakeOver(takeOverPtr.i);
+ break;
+ case TakeOverRecord::TO_UPDATE_TO:
+ jam();
+ ok = true;
+ c_updateToLock = RNIL;
+ c_UPDATE_TOREQ_Counter.clearWaitingFor();
+ endTakeOver(takeOverPtr.i);
+ break;
+ case TakeOverRecord::ENDING:
+ jam();
+ ok = true;
+ c_endToLock = RNIL;
+ c_END_TOREQ_Counter.clearWaitingFor();
+ endTakeOver(takeOverPtr.i);
+ break;
+ case TakeOverRecord::COMMIT_CREATE:
+ ok = true;
+ jam();
+ {// We have mutex
+ Mutex m(signal, c_mutexMgr, takeOverPtr.p->m_switchPrimaryMutexHandle);
+ m.unlock(); // Ignore result
+ }
+ // Fall through
+ case TakeOverRecord::PREPARE_CREATE:
+ ok = true;
+ jam();
+ c_createFragmentLock = RNIL;
+ c_CREATE_FRAGREQ_Counter.clearWaitingFor();
+ endTakeOver(takeOverPtr.i);
+ break;
+ case TakeOverRecord::LOCK_MUTEX:
+ ok = true;
+ jam();
+ // Lock mutex will return and do endTakeOver
+ break;
+
+ //-----------------------------------------------------------------------
+ // Signals are outstanding to external nodes. These signals carry the node
+ // id of the starting node and will not use the take over record if the
+ // starting node has failed.
+ //-----------------------------------------------------------------------
+ case TakeOverRecord::COPY_FRAG:
+ ok = true;
+ jam();
+ //-----------------------------------------------------------------------
+ // The starting node will discover the problem. We will receive either
+ // COPY_FRAGREQ or COPY_FRAGCONF and then we can release the take over
+ // record and end the process. If the copying node should also die then
+ // we will try to send prepare create fragment and will then discover
+ // that the starting node has failed.
+ //-----------------------------------------------------------------------
+ break;
+ case TakeOverRecord::COPY_ACTIVE:
+ ok = true;
+ jam();
+ //-----------------------------------------------------------------------
+ // In this we are waiting for a signal from the starting node. Thus we
+ // can release the take over record and end the process.
+ //-----------------------------------------------------------------------
+ endTakeOver(takeOverPtr.i);
+ break;
+ case TakeOverRecord::WAIT_LCP:
+ ok = true;
+ jam();
+ //-----------------------------------------------------------------------
+ //-----------------------------------------------------------------------
+ endTakeOver(takeOverPtr.i);
+ break;
+ /**
+ * The following are states that it should not be possible to "be" in
+ */
+ case TakeOverRecord::SELECTING_NEXT:
+ jam();
+ case TakeOverRecord::TO_COPY_COMPLETED:
+ jam();
+ ndbrequire(false);
+ }
+ if(!ok){
+ jamLine(takeOverPtr.p->toSlaveStatus);
+ ndbrequire(ok);
+ }
+}//Dbdih::checkTakeOverInMasterStartNodeFailure()
+
+void Dbdih::checkTakeOverInNonMasterStartNodeFailure(Signal* signal,
+ Uint32 takeOverPtrI)
+{
+ jam();
+ if (takeOverPtrI == RNIL) {
+ jam();
+ return;
+ }
+ //-----------------------------------------------------------------------
+ // We are not master and not taking over as master. A take over was ongoing
+ // but the starting node has now failed. Handle it according to the state
+ // of the take over.
+ //-----------------------------------------------------------------------
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = takeOverPtrI;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ bool ok = false;
+ switch (takeOverPtr.p->toSlaveStatus) {
+ case TakeOverRecord::TO_SLAVE_IDLE:
+ ndbrequire(false);
+ break;
+ case TakeOverRecord::TO_SLAVE_STARTED:
+ jam();
+ case TakeOverRecord::TO_SLAVE_CREATE_PREPARE:
+ jam();
+ case TakeOverRecord::TO_SLAVE_COPY_FRAG_COMPLETED:
+ jam();
+ case TakeOverRecord::TO_SLAVE_CREATE_COMMIT:
+ jam();
+ case TakeOverRecord::TO_SLAVE_COPY_COMPLETED:
+ jam();
+ ok = true;
+ endTakeOver(takeOverPtr.i);
+ break;
+ }//switch
+ if(!ok){
+ jamLine(takeOverPtr.p->toSlaveStatus);
+ ndbrequire(ok);
+ }
+}//Dbdih::checkTakeOverInNonMasterStartNodeFailure()
+
+void Dbdih::failedNodeSynchHandling(Signal* signal,
+ NodeRecordPtr failedNodePtr)
+{
+ jam();
+ /*----------------------------------------------------*/
+ /* INITIALISE THE VARIABLES THAT KEEP TRACK OF */
+ /* WHEN A NODE FAILURE IS COMPLETED. */
+ /*----------------------------------------------------*/
+ failedNodePtr.p->dbdictFailCompleted = ZFALSE;
+ failedNodePtr.p->dbtcFailCompleted = ZFALSE;
+ failedNodePtr.p->dbdihFailCompleted = ZFALSE;
+ failedNodePtr.p->dblqhFailCompleted = ZFALSE;
+
+ failedNodePtr.p->m_NF_COMPLETE_REP.clearWaitingFor();
+
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRecord);
+ if (nodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ jam();
+ /**
+ * We'r waiting for nodePtr.i to complete
+ * handling of failedNodePtr.i's death
+ */
+
+ failedNodePtr.p->m_NF_COMPLETE_REP.setWaitingFor(nodePtr.i);
+ } else {
+ jam();
+ if ((nodePtr.p->nodeStatus == NodeRecord::DYING) &&
+ (nodePtr.p->m_NF_COMPLETE_REP.isWaitingFor(failedNodePtr.i))){
+ jam();
+ /*----------------------------------------------------*/
+ /* THE NODE FAILED BEFORE REPORTING THE FAILURE */
+ /* HANDLING COMPLETED ON THIS FAILED NODE. */
+ /* REPORT THAT NODE FAILURE HANDLING WAS */
+ /* COMPLETED ON THE NEW FAILED NODE FOR THIS */
+ /* PARTICULAR OLD FAILED NODE. */
+ /*----------------------------------------------------*/
+ NFCompleteRep * const nf = (NFCompleteRep *)&signal->theData[0];
+ nf->blockNo = 0;
+ nf->nodeId = failedNodePtr.i;
+ nf->failedNodeId = nodePtr.i;
+ nf->from = __LINE__;
+ sendSignal(reference(), GSN_NF_COMPLETEREP, signal,
+ NFCompleteRep::SignalLength, JBB);
+ }//if
+ }//if
+ }//for
+ if (failedNodePtr.p->nodeStatus == NodeRecord::DIED_NOW) {
+ jam();
+ failedNodePtr.p->nodeStatus = NodeRecord::DYING;
+ } else {
+ jam();
+ /*----------------------------------------------------*/
+ // No more processing needed when node not even started
+ // yet. We give the node status to DEAD since we do not
+ // care whether all nodes complete the node failure
+ // handling. The node have not been included in the
+ // node failure protocols.
+ /*----------------------------------------------------*/
+ failedNodePtr.p->nodeStatus = NodeRecord::DEAD;
+ /**-----------------------------------------------------------------------
+ * WE HAVE COMPLETED HANDLING THE NODE FAILURE IN DIH. WE CAN REPORT THIS
+ * TO DIH THAT WAIT FOR THE OTHER BLOCKS TO BE CONCLUDED AS WELL.
+ *-----------------------------------------------------------------------*/
+ NFCompleteRep * const nf = (NFCompleteRep *)&signal->theData[0];
+ nf->blockNo = DBDIH;
+ nf->nodeId = cownNodeId;
+ nf->failedNodeId = failedNodePtr.i;
+ nf->from = __LINE__;
+ sendSignal(reference(), GSN_NF_COMPLETEREP, signal,
+ NFCompleteRep::SignalLength, JBB);
+ }//if
+}//Dbdih::failedNodeSynchHandling()
+
+Uint32 Dbdih::findTakeOver(Uint32 failedNodeId)
+{
+ for (Uint32 i = 0; i < MAX_NDB_NODES; i++) {
+ jam();
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = i;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ if (takeOverPtr.p->toStartingNode == failedNodeId) {
+ jam();
+ return i;
+ }//if
+ }//for
+ return RNIL;
+}//Dbdih::findTakeOver()
+
+Uint32 Dbdih::getStartNode(Uint32 takeOverPtrI)
+{
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = takeOverPtrI;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ return takeOverPtr.p->toStartingNode;
+}//Dbdih::getStartNode()
+
+void Dbdih::failedNodeLcpHandling(Signal* signal, NodeRecordPtr failedNodePtr)
+{
+ jam();
+ const Uint32 nodeId = failedNodePtr.i;
+
+ if (c_lcpState.m_participatingLQH.get(failedNodePtr.i)){
+ /*----------------------------------------------------*/
+ /* THE NODE WAS INVOLVED IN A LOCAL CHECKPOINT. WE */
+ /* MUST UPDATE THE ACTIVE STATUS TO INDICATE THAT */
+ /* THE NODE HAVE MISSED A LOCAL CHECKPOINT. */
+ /*----------------------------------------------------*/
+ switch (failedNodePtr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ jam();
+ failedNodePtr.p->activeStatus = Sysfile::NS_ActiveMissed_1;
+ break;
+ case Sysfile::NS_ActiveMissed_1:
+ jam();
+ failedNodePtr.p->activeStatus = Sysfile::NS_ActiveMissed_2;
+ break;
+ case Sysfile::NS_ActiveMissed_2:
+ jam();
+ failedNodePtr.p->activeStatus = Sysfile::NS_NotActive_NotTakenOver;
+ break;
+ case Sysfile::NS_TakeOver:
+ jam();
+ failedNodePtr.p->activeStatus = Sysfile::NS_NotActive_NotTakenOver;
+ break;
+ default:
+ ndbout << "activeStatus = " << failedNodePtr.p->activeStatus;
+ ndbout << " at failure after NODE_FAILREP of node = ";
+ ndbout << failedNodePtr.i << endl;
+ ndbrequire(false);
+ break;
+ }//switch
+ }//if
+
+ c_lcpState.m_participatingDIH.clear(failedNodePtr.i);
+ c_lcpState.m_participatingLQH.clear(failedNodePtr.i);
+
+ if(c_lcpState.m_LCP_COMPLETE_REP_Counter_DIH.isWaitingFor(failedNodePtr.i)){
+ jam();
+ LcpCompleteRep * rep = (LcpCompleteRep*)signal->getDataPtrSend();
+ rep->nodeId = failedNodePtr.i;
+ rep->lcpId = SYSFILE->latestLCP_ID;
+ rep->blockNo = DBDIH;
+ sendSignal(reference(), GSN_LCP_COMPLETE_REP, signal,
+ LcpCompleteRep::SignalLength, JBB);
+ }
+
+ /**
+ * Check if we'r waiting for the failed node's LQH to complete
+ *
+ * Note that this is ran "before" LCP master take over
+ */
+ if(c_lcpState.m_LCP_COMPLETE_REP_Counter_LQH.isWaitingFor(nodeId)){
+ jam();
+
+ LcpCompleteRep * rep = (LcpCompleteRep*)signal->getDataPtrSend();
+ rep->nodeId = nodeId;
+ rep->lcpId = SYSFILE->latestLCP_ID;
+ rep->blockNo = DBLQH;
+ sendSignal(reference(), GSN_LCP_COMPLETE_REP, signal,
+ LcpCompleteRep::SignalLength, JBB);
+
+ if(c_lcpState.m_LAST_LCP_FRAG_ORD.isWaitingFor(nodeId)){
+ jam();
+ /**
+ * Make sure we're ready to accept it
+ */
+ c_lcpState.m_LAST_LCP_FRAG_ORD.clearWaitingFor(nodeId);
+ }
+ }
+
+ if (c_TCGETOPSIZEREQ_Counter.isWaitingFor(failedNodePtr.i)) {
+ jam();
+ signal->theData[0] = failedNodePtr.i;
+ signal->theData[1] = 0;
+ sendSignal(reference(), GSN_TCGETOPSIZECONF, signal, 2, JBB);
+ }//if
+
+ if (c_TC_CLOPSIZEREQ_Counter.isWaitingFor(failedNodePtr.i)) {
+ jam();
+ signal->theData[0] = failedNodePtr.i;
+ sendSignal(reference(), GSN_TC_CLOPSIZECONF, signal, 1, JBB);
+ }//if
+
+ if (c_START_LCP_REQ_Counter.isWaitingFor(failedNodePtr.i)) {
+ jam();
+ StartLcpConf * conf = (StartLcpConf*)signal->getDataPtrSend();
+ conf->senderRef = numberToRef(DBLQH, failedNodePtr.i);
+ conf->lcpId = SYSFILE->latestLCP_ID;
+ sendSignal(reference(), GSN_START_LCP_CONF, signal,
+ StartLcpConf::SignalLength, JBB);
+ }//if
+
+ if (c_EMPTY_LCP_REQ_Counter.isWaitingFor(failedNodePtr.i)) {
+ jam();
+ EmptyLcpConf * const rep = (EmptyLcpConf *)&signal->theData[0];
+ rep->senderNodeId = failedNodePtr.i;
+ rep->tableId = ~0;
+ rep->fragmentId = ~0;
+ rep->lcpNo = 0;
+ rep->lcpId = SYSFILE->latestLCP_ID;
+ rep->idle = true;
+ sendSignal(reference(), GSN_EMPTY_LCP_CONF, signal,
+ EmptyLcpConf::SignalLength, JBB);
+ }//if
+
+ if (c_MASTER_LCPREQ_Counter.isWaitingFor(failedNodePtr.i)) {
+ jam();
+ MasterLCPRef * const ref = (MasterLCPRef *)&signal->theData[0];
+ ref->senderNodeId = failedNodePtr.i;
+ ref->failedNodeId = cmasterTakeOverNode;
+ sendSignal(reference(), GSN_MASTER_LCPREF, signal,
+ MasterLCPRef::SignalLength, JBB);
+ }//if
+
+}//Dbdih::failedNodeLcpHandling()
+
+void Dbdih::checkGcpOutstanding(Signal* signal, Uint32 failedNodeId){
+ if (c_GCP_PREPARE_Counter.isWaitingFor(failedNodeId)){
+ jam();
+ signal->theData[0] = failedNodeId;
+ signal->theData[1] = cnewgcp;
+ sendSignal(reference(), GSN_GCP_PREPARECONF, signal, 2, JBB);
+ }//if
+
+ if (c_GCP_COMMIT_Counter.isWaitingFor(failedNodeId)) {
+ jam();
+ signal->theData[0] = failedNodeId;
+ signal->theData[1] = coldgcp;
+ signal->theData[2] = cfailurenr;
+ sendSignal(reference(), GSN_GCP_NODEFINISH, signal, 3, JBB);
+ }//if
+
+ if (c_GCP_SAVEREQ_Counter.isWaitingFor(failedNodeId)) {
+ jam();
+ GCPSaveRef * const saveRef = (GCPSaveRef*)&signal->theData[0];
+ saveRef->dihPtr = failedNodeId;
+ saveRef->nodeId = failedNodeId;
+ saveRef->gci = coldgcp;
+ saveRef->errorCode = GCPSaveRef::FakedSignalDueToNodeFailure;
+ sendSignal(reference(), GSN_GCP_SAVEREF, signal,
+ GCPSaveRef::SignalLength, JBB);
+ }//if
+
+ if (c_COPY_GCIREQ_Counter.isWaitingFor(failedNodeId)) {
+ jam();
+ signal->theData[0] = failedNodeId;
+ sendSignal(reference(), GSN_COPY_GCICONF, signal, 1, JBB);
+ }//if
+
+ if (c_MASTER_GCPREQ_Counter.isWaitingFor(failedNodeId)){
+ jam();
+ MasterGCPRef * const ref = (MasterGCPRef *)&signal->theData[0];
+ ref->senderNodeId = failedNodeId;
+ ref->failedNodeId = cmasterTakeOverNode;
+ sendSignal(reference(), GSN_MASTER_GCPREF, signal,
+ MasterGCPRef::SignalLength, JBB);
+ }//if
+}//Dbdih::handleGcpStateInMaster()
+
+
+void
+Dbdih::startLcpMasterTakeOver(Signal* signal, Uint32 nodeId){
+ jam();
+
+ c_lcpMasterTakeOverState.minTableId = ~0;
+ c_lcpMasterTakeOverState.minFragId = ~0;
+ c_lcpMasterTakeOverState.failedNodeId = nodeId;
+
+ c_lcpMasterTakeOverState.set(LMTOS_WAIT_EMPTY_LCP, __LINE__);
+
+ if(c_EMPTY_LCP_REQ_Counter.done()){
+ jam();
+ c_lcpState.m_LAST_LCP_FRAG_ORD.clearWaitingFor();
+
+ EmptyLcpReq* req = (EmptyLcpReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ sendLoopMacro(EMPTY_LCP_REQ, sendEMPTY_LCP_REQ);
+ ndbrequire(!c_EMPTY_LCP_REQ_Counter.done());
+ } else {
+ /**
+ * Node failure during master take over...
+ */
+ ndbout_c("Nodefail during master take over");
+ }
+
+ setLocalNodefailHandling(signal, nodeId, NF_LCP_TAKE_OVER);
+}
+
+void Dbdih::startGcpMasterTakeOver(Signal* signal, Uint32 oldMasterId){
+ jam();
+ /*--------------------------------------------------*/
+ /* */
+ /* THE MASTER HAVE FAILED AND WE WERE ELECTED */
+ /* TO BE THE NEW MASTER NODE. WE NEED TO QUERY*/
+ /* ALL THE OTHER NODES ABOUT THEIR STATUS IN */
+ /* ORDER TO BE ABLE TO TAKE OVER CONTROL OF */
+ /* THE GLOBAL CHECKPOINT PROTOCOL AND THE */
+ /* LOCAL CHECKPOINT PROTOCOL. */
+ /*--------------------------------------------------*/
+ if(!isMaster()){
+ jam();
+ return;
+ }
+ cmasterState = MASTER_TAKE_OVER_GCP;
+ cmasterTakeOverNode = oldMasterId;
+ MasterGCPReq * const req = (MasterGCPReq *)&signal->theData[0];
+ req->masterRef = reference();
+ req->failedNodeId = oldMasterId;
+ sendLoopMacro(MASTER_GCPREQ, sendMASTER_GCPREQ);
+ cgcpMasterTakeOverState = GMTOS_INITIAL;
+
+ signal->theData[0] = EventReport::GCP_TakeoverStarted;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 1, JBB);
+
+ setLocalNodefailHandling(signal, oldMasterId, NF_GCP_TAKE_OVER);
+}//Dbdih::handleNewMaster()
+
+void Dbdih::handleTakeOverNewMaster(Signal* signal, Uint32 takeOverPtrI)
+{
+ jam();
+ if (takeOverPtrI != RNIL) {
+ jam();
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = takeOverPtrI;
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ bool ok = false;
+ switch (takeOverPtr.p->toSlaveStatus) {
+ case TakeOverRecord::TO_SLAVE_IDLE:
+ ndbrequire(false);
+ break;
+ case TakeOverRecord::TO_SLAVE_STARTED:
+ jam();
+ case TakeOverRecord::TO_SLAVE_CREATE_PREPARE:
+ jam();
+ case TakeOverRecord::TO_SLAVE_COPY_FRAG_COMPLETED:
+ jam();
+ case TakeOverRecord::TO_SLAVE_CREATE_COMMIT:
+ jam();
+ ok = true;
+ infoEvent("Unhandled MasterTO of TO slaveStatus=%d killing node %d",
+ takeOverPtr.p->toSlaveStatus,
+ takeOverPtr.p->toStartingNode);
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::COPY_ACTIVE;
+
+ {
+ BlockReference cntrRef = calcNdbCntrBlockRef(takeOverPtr.p->toStartingNode);
+ SystemError * const sysErr = (SystemError*)&signal->theData[0];
+ sysErr->errorCode = SystemError::CopyFragRefError;
+ sysErr->errorRef = reference();
+ sendSignal(cntrRef, GSN_SYSTEM_ERROR, signal,
+ SystemError::SignalLength, JBB);
+ }
+ break;
+ case TakeOverRecord::TO_SLAVE_COPY_COMPLETED:
+ ok = true;
+ jam();
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::WAIT_LCP;
+ break;
+ }
+ ndbrequire(ok);
+ }//if
+}//Dbdih::handleTakeOverNewMaster()
+
+void Dbdih::startRemoveFailedNode(Signal* signal, NodeRecordPtr failedNodePtr)
+{
+ Uint32 nodeId = failedNodePtr.i;
+ if(failedNodePtr.p->nodeStatus != NodeRecord::DIED_NOW){
+ jam();
+ /**
+ * Is node isn't alive. It can't be part of LCP
+ */
+ ndbrequire(!c_lcpState.m_LCP_COMPLETE_REP_Counter_LQH.isWaitingFor(nodeId));
+
+ /**
+ * And there is no point in removing any replicas
+ * It's dead...
+ */
+ return;
+ }
+
+ jam();
+ signal->theData[0] = DihContinueB::ZREMOVE_NODE_FROM_TABLE;
+ signal->theData[1] = failedNodePtr.i;
+ signal->theData[2] = 0; // Tab id
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+
+ setLocalNodefailHandling(signal, failedNodePtr.i, NF_REMOVE_NODE_FROM_TABLE);
+}//Dbdih::startRemoveFailedNode()
+
+/*--------------------------------------------------*/
+/* THE MASTER HAS FAILED AND THE NEW MASTER IS*/
+/* QUERYING THIS NODE ABOUT THE STATE OF THE */
+/* GLOBAL CHECKPOINT PROTOCOL */
+/*--------------------------------------------------*/
+void Dbdih::execMASTER_GCPREQ(Signal* signal)
+{
+ NodeRecordPtr failedNodePtr;
+ MasterGCPReq * const masterGCPReq = (MasterGCPReq *)&signal->theData[0];
+ jamEntry();
+ const BlockReference newMasterBlockref = masterGCPReq->masterRef;
+ const Uint32 failedNodeId = masterGCPReq->failedNodeId;
+ if (c_copyGCISlave.m_copyReason != CopyGCIReq::IDLE) {
+ jam();
+ /*--------------------------------------------------*/
+ /* WE ARE CURRENTLY WRITING THE RESTART INFO */
+ /* IN THIS NODE. SINCE ONLY ONE PROCESS IS */
+ /* ALLOWED TO DO THIS AT A TIME WE MUST ENSURE*/
+ /* THAT THIS IS NOT ONGOING WHEN THE NEW */
+ /* MASTER TAKES OVER CONTROL. IF NOT ALL NODES*/
+ /* RECEIVE THE SAME RESTART INFO DUE TO THE */
+ /* FAILURE OF THE MASTER IT IS TAKEN CARE OF */
+ /* BY THE NEW MASTER. */
+ /*--------------------------------------------------*/
+ sendSignalWithDelay(reference(), GSN_MASTER_GCPREQ,
+ signal, 10, MasterGCPReq::SignalLength);
+ return;
+ }//if
+ failedNodePtr.i = failedNodeId;
+ ptrCheckGuard(failedNodePtr, MAX_NDB_NODES, nodeRecord);
+ if (failedNodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ jam();
+ /*--------------------------------------------------*/
+ /* ENSURE THAT WE HAVE PROCESSED THE SIGNAL */
+ /* NODE_FAILURE BEFORE WE PROCESS THIS REQUEST*/
+ /* FROM THE NEW MASTER. THIS ENSURES THAT WE */
+ /* HAVE REMOVED THE FAILED NODE FROM THE LIST */
+ /* OF ACTIVE NODES AND SO FORTH. */
+ /*--------------------------------------------------*/
+ sendSignalWithDelay(reference(), GSN_MASTER_GCPREQ,
+ signal, 10, MasterGCPReq::SignalLength);
+ return;
+ } else {
+ ndbrequire(failedNodePtr.p->nodeStatus == NodeRecord::DYING);
+ }//if
+ MasterGCPConf::State gcpState;
+ switch (cgcpParticipantState) {
+ case GCP_PARTICIPANT_READY:
+ jam();
+ /*--------------------------------------------------*/
+ /* THE GLOBAL CHECKPOINT IS NOT ACTIVE SINCE */
+ /* THE PREVIOUS GLOBAL CHECKPOINT IS COMPLETED*/
+ /* AND THE NEW HAVE NOT STARTED YET. */
+ /*--------------------------------------------------*/
+ gcpState = MasterGCPConf::GCP_READY;
+ break;
+ case GCP_PARTICIPANT_PREPARE_RECEIVED:
+ jam();
+ /*--------------------------------------------------*/
+ /* GCP_PREPARE HAVE BEEN RECEIVED AND RESPONSE*/
+ /* HAVE BEEN SENT. */
+ /*--------------------------------------------------*/
+ gcpState = MasterGCPConf::GCP_PREPARE_RECEIVED;
+ break;
+ case GCP_PARTICIPANT_COMMIT_RECEIVED:
+ jam();
+ /*------------------------------------------------*/
+ /* GCP_COMMIT HAVE BEEN RECEIVED BUT NOT YET*/
+ /* GCP_TCFINISHED FROM LOCAL TC. */
+ /*------------------------------------------------*/
+ gcpState = MasterGCPConf::GCP_COMMIT_RECEIVED;
+ break;
+ case GCP_PARTICIPANT_TC_FINISHED:
+ jam();
+ /*------------------------------------------------*/
+ /* GCP_COMMIT HAS BEEN RECEIVED AND ALSO */
+ /* GCP_TCFINISHED HAVE BEEN RECEIVED. */
+ /*------------------------------------------------*/
+ gcpState = MasterGCPConf::GCP_TC_FINISHED;
+ break;
+ case GCP_PARTICIPANT_COPY_GCI_RECEIVED:
+ /*--------------------------------------------------*/
+ /* COPY RESTART INFORMATION HAS BEEN RECEIVED */
+ /* BUT NOT YET COMPLETED. */
+ /*--------------------------------------------------*/
+ ndbrequire(false);
+ break;
+ default:
+ /*------------------------------------------------*/
+ /* */
+ /* THIS SHOULD NOT OCCUR SINCE THE ABOVE */
+ /* STATES ARE THE ONLY POSSIBLE STATES AT A */
+ /* NODE WHICH WAS NOT A MASTER NODE. */
+ /*------------------------------------------------*/
+ ndbrequire(false);
+ break;
+ }//switch
+ MasterGCPConf * const masterGCPConf = (MasterGCPConf *)&signal->theData[0];
+ masterGCPConf->gcpState = gcpState;
+ masterGCPConf->senderNodeId = cownNodeId;
+ masterGCPConf->failedNodeId = failedNodeId;
+ masterGCPConf->newGCP = cnewgcp;
+ masterGCPConf->latestLCP = SYSFILE->latestLCP_ID;
+ masterGCPConf->oldestRestorableGCI = SYSFILE->oldestRestorableGCI;
+ masterGCPConf->keepGCI = SYSFILE->keepGCI;
+ for(Uint32 i = 0; i < NdbNodeBitmask::Size; i++)
+ masterGCPConf->lcpActive[i] = SYSFILE->lcpActive[i];
+ sendSignal(newMasterBlockref, GSN_MASTER_GCPCONF, signal,
+ MasterGCPConf::SignalLength, JBB);
+}//Dbdih::execMASTER_GCPREQ()
+
+void Dbdih::execMASTER_GCPCONF(Signal* signal)
+{
+ NodeRecordPtr senderNodePtr;
+ MasterGCPConf * const masterGCPConf = (MasterGCPConf *)&signal->theData[0];
+ jamEntry();
+ senderNodePtr.i = masterGCPConf->senderNodeId;
+ ptrCheckGuard(senderNodePtr, MAX_NDB_NODES, nodeRecord);
+
+ MasterGCPConf::State gcpState = (MasterGCPConf::State)masterGCPConf->gcpState;
+ const Uint32 failedNodeId = masterGCPConf->failedNodeId;
+ const Uint32 newGcp = masterGCPConf->newGCP;
+ const Uint32 latestLcpId = masterGCPConf->latestLCP;
+ const Uint32 oldestRestorableGci = masterGCPConf->oldestRestorableGCI;
+ const Uint32 oldestKeepGci = masterGCPConf->keepGCI;
+ if (latestLcpId > SYSFILE->latestLCP_ID) {
+ jam();
+#if 0
+ ndbout_c("Dbdih: Setting SYSFILE->latestLCP_ID to %d", latestLcpId);
+ SYSFILE->latestLCP_ID = latestLcpId;
+#endif
+ SYSFILE->keepGCI = oldestKeepGci;
+ SYSFILE->oldestRestorableGCI = oldestRestorableGci;
+ for(Uint32 i = 0; i < NdbNodeBitmask::Size; i++)
+ SYSFILE->lcpActive[i] = masterGCPConf->lcpActive[i];
+ }//if
+ switch (gcpState) {
+ case MasterGCPConf::GCP_READY:
+ jam();
+ senderNodePtr.p->gcpstate = NodeRecord::READY;
+ break;
+ case MasterGCPConf::GCP_PREPARE_RECEIVED:
+ jam();
+ senderNodePtr.p->gcpstate = NodeRecord::PREPARE_RECEIVED;
+ cnewgcp = newGcp;
+ break;
+ case MasterGCPConf::GCP_COMMIT_RECEIVED:
+ jam();
+ senderNodePtr.p->gcpstate = NodeRecord::COMMIT_SENT;
+ break;
+ case MasterGCPConf::GCP_TC_FINISHED:
+ jam();
+ senderNodePtr.p->gcpstate = NodeRecord::NODE_FINISHED;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ switch (cgcpMasterTakeOverState) {
+ case GMTOS_INITIAL:
+ switch (gcpState) {
+ case MasterGCPConf::GCP_READY:
+ jam();
+ cgcpMasterTakeOverState = ALL_READY;
+ break;
+ case MasterGCPConf::GCP_PREPARE_RECEIVED:
+ jam();
+ cgcpMasterTakeOverState = ALL_PREPARED;
+ break;
+ case MasterGCPConf::GCP_COMMIT_RECEIVED:
+ jam();
+ cgcpMasterTakeOverState = COMMIT_STARTED_NOT_COMPLETED;
+ break;
+ case MasterGCPConf::GCP_TC_FINISHED:
+ jam();
+ cgcpMasterTakeOverState = COMMIT_COMPLETED;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ break;
+ case ALL_READY:
+ switch (gcpState) {
+ case MasterGCPConf::GCP_READY:
+ jam();
+ /*empty*/;
+ break;
+ case MasterGCPConf::GCP_PREPARE_RECEIVED:
+ jam();
+ cgcpMasterTakeOverState = PREPARE_STARTED_NOT_COMMITTED;
+ break;
+ case MasterGCPConf::GCP_COMMIT_RECEIVED:
+ ndbrequire(false);
+ break;
+ case MasterGCPConf::GCP_TC_FINISHED:
+ jam();
+ cgcpMasterTakeOverState = SAVE_STARTED_NOT_COMPLETED;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ break;
+ case PREPARE_STARTED_NOT_COMMITTED:
+ switch (gcpState) {
+ case MasterGCPConf::GCP_READY:
+ jam();
+ break;
+ case MasterGCPConf::GCP_PREPARE_RECEIVED:
+ jam();
+ break;
+ case MasterGCPConf::GCP_COMMIT_RECEIVED:
+ ndbrequire(false);
+ break;
+ case MasterGCPConf::GCP_TC_FINISHED:
+ ndbrequire(false);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ break;
+ case ALL_PREPARED:
+ switch (gcpState) {
+ case MasterGCPConf::GCP_READY:
+ jam();
+ cgcpMasterTakeOverState = PREPARE_STARTED_NOT_COMMITTED;
+ break;
+ case MasterGCPConf::GCP_PREPARE_RECEIVED:
+ jam();
+ break;
+ case MasterGCPConf::GCP_COMMIT_RECEIVED:
+ jam();
+ cgcpMasterTakeOverState = COMMIT_STARTED_NOT_COMPLETED;
+ break;
+ case MasterGCPConf::GCP_TC_FINISHED:
+ jam();
+ cgcpMasterTakeOverState = COMMIT_STARTED_NOT_COMPLETED;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ break;
+ case COMMIT_STARTED_NOT_COMPLETED:
+ switch (gcpState) {
+ case MasterGCPConf::GCP_READY:
+ ndbrequire(false);
+ break;
+ case MasterGCPConf::GCP_PREPARE_RECEIVED:
+ jam();
+ break;
+ case MasterGCPConf::GCP_COMMIT_RECEIVED:
+ jam();
+ break;
+ case MasterGCPConf::GCP_TC_FINISHED:
+ jam();
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ break;
+ case COMMIT_COMPLETED:
+ switch (gcpState) {
+ case MasterGCPConf::GCP_READY:
+ cgcpMasterTakeOverState = SAVE_STARTED_NOT_COMPLETED;
+ break;
+ case MasterGCPConf::GCP_PREPARE_RECEIVED:
+ jam();
+ cgcpMasterTakeOverState = COMMIT_STARTED_NOT_COMPLETED;
+ break;
+ case MasterGCPConf::GCP_COMMIT_RECEIVED:
+ jam();
+ cgcpMasterTakeOverState = COMMIT_STARTED_NOT_COMPLETED;
+ break;
+ case MasterGCPConf::GCP_TC_FINISHED:
+ jam();
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ break;
+ case SAVE_STARTED_NOT_COMPLETED:
+ switch (gcpState) {
+ case MasterGCPConf::GCP_READY:
+ jam();
+ break;
+ case MasterGCPConf::GCP_PREPARE_RECEIVED:
+ ndbrequire(false);
+ break;
+ case MasterGCPConf::GCP_COMMIT_RECEIVED:
+ ndbrequire(false);
+ break;
+ case MasterGCPConf::GCP_TC_FINISHED:
+ jam();
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ receiveLoopMacro(MASTER_GCPREQ, senderNodePtr.i);
+ /*-------------------------------------------------------------------------*/
+ // We have now received all responses and are ready to take over the GCP
+ // protocol as master.
+ /*-------------------------------------------------------------------------*/
+ MASTER_GCPhandling(signal, failedNodeId);
+ return;
+}//Dbdih::execMASTER_GCPCONF()
+
+void Dbdih::execMASTER_GCPREF(Signal* signal)
+{
+ const MasterGCPRef * const ref = (MasterGCPRef *)&signal->theData[0];
+ jamEntry();
+ receiveLoopMacro(MASTER_GCPREQ, ref->senderNodeId);
+ /*-------------------------------------------------------------------------*/
+ // We have now received all responses and are ready to take over the GCP
+ // protocol as master.
+ /*-------------------------------------------------------------------------*/
+ MASTER_GCPhandling(signal, ref->failedNodeId);
+}//Dbdih::execMASTER_GCPREF()
+
+void Dbdih::MASTER_GCPhandling(Signal* signal, Uint32 failedNodeId)
+{
+ NodeRecordPtr failedNodePtr;
+ cmasterState = MASTER_ACTIVE;
+ /*----------------------------------------------------------*/
+ /* REMOVE ALL ACTIVE STATUS ON ALREADY FAILED NODES */
+ /* THIS IS PERFORMED HERE SINCE WE GET THE LCP ACTIVE */
+ /* STATUS AS PART OF THE COPY RESTART INFO AND THIS IS*/
+ /* HANDLED BY THE MASTER GCP TAKE OVER PROTOCOL. */
+ /*----------------------------------------------------------*/
+
+ failedNodePtr.i = failedNodeId;
+ ptrCheckGuard(failedNodePtr, MAX_NDB_NODES, nodeRecord);
+ switch (cgcpMasterTakeOverState) {
+ case ALL_READY:
+ jam();
+ startGcp(signal);
+ break;
+ case PREPARE_STARTED_NOT_COMMITTED:
+ {
+ NodeRecordPtr nodePtr;
+ jam();
+ c_GCP_PREPARE_Counter.clearWaitingFor();
+ nodePtr.i = cfirstAliveNode;
+ do {
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.p->gcpstate == NodeRecord::READY) {
+ jam();
+ c_GCP_PREPARE_Counter.setWaitingFor(nodePtr.i);
+ sendGCP_PREPARE(signal, nodePtr.i);
+ }//if
+ nodePtr.i = nodePtr.p->nextNode;
+ } while(nodePtr.i != RNIL);
+ if (c_GCP_PREPARE_Counter.done()) {
+ jam();
+ gcpcommitreqLab(signal);
+ }//if
+ break;
+ }
+ case ALL_PREPARED:
+ jam();
+ gcpcommitreqLab(signal);
+ break;
+ case COMMIT_STARTED_NOT_COMPLETED:
+ {
+ NodeRecordPtr nodePtr;
+ jam();
+ c_GCP_COMMIT_Counter.clearWaitingFor();
+ nodePtr.i = cfirstAliveNode;
+ do {
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.p->gcpstate == NodeRecord::PREPARE_RECEIVED) {
+ jam();
+ sendGCP_COMMIT(signal, nodePtr.i);
+ c_GCP_COMMIT_Counter.setWaitingFor(nodePtr.i);
+ } else {
+ ndbrequire((nodePtr.p->gcpstate == NodeRecord::NODE_FINISHED) ||
+ (nodePtr.p->gcpstate == NodeRecord::COMMIT_SENT));
+ }//if
+ nodePtr.i = nodePtr.p->nextNode;
+ } while(nodePtr.i != RNIL);
+ if (c_GCP_COMMIT_Counter.done()){
+ jam();
+ gcpsavereqLab(signal);
+ }//if
+ break;
+ }
+ case COMMIT_COMPLETED:
+ jam();
+ gcpsavereqLab(signal);
+ break;
+ case SAVE_STARTED_NOT_COMPLETED:
+ {
+ NodeRecordPtr nodePtr;
+ jam();
+ SYSFILE->newestRestorableGCI = coldgcp;
+ nodePtr.i = cfirstAliveNode;
+ do {
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ SYSFILE->lastCompletedGCI[nodePtr.i] = coldgcp;
+ nodePtr.i = nodePtr.p->nextNode;
+ } while (nodePtr.i != RNIL);
+ /**-------------------------------------------------------------------
+ * THE FAILED NODE DID ALSO PARTICIPATE IN THIS GLOBAL CHECKPOINT
+ * WHICH IS RECORDED.
+ *-------------------------------------------------------------------*/
+ SYSFILE->lastCompletedGCI[failedNodeId] = coldgcp;
+ copyGciLab(signal, CopyGCIReq::GLOBAL_CHECKPOINT);
+ break;
+ }
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+
+ signal->theData[0] = EventReport::GCP_TakeoverCompleted;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 1, JBB);
+
+ /*--------------------------------------------------*/
+ /* WE SEPARATE HANDLING OF GLOBAL CHECKPOINTS */
+ /* AND LOCAL CHECKPOINTS HERE. LCP'S HAVE TO */
+ /* REMOVE ALL FAILED FRAGMENTS BEFORE WE CAN */
+ /* HANDLE THE LCP PROTOCOL. */
+ /*--------------------------------------------------*/
+ checkLocalNodefailComplete(signal, failedNodeId, NF_GCP_TAKE_OVER);
+
+ return;
+}//Dbdih::masterGcpConfFromFailedLab()
+
+void
+Dbdih::invalidateNodeLCP(Signal* signal, Uint32 nodeId, Uint32 tableId)
+{
+ jamEntry();
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ const Uint32 RT_BREAK = 64;
+ if (ERROR_INSERTED(7125)) {
+ return;
+ }//if
+ for (Uint32 i = 0; i<RT_BREAK; i++) {
+ jam();
+ if (tabPtr.i >= ctabFileSize){
+ jam();
+ /**
+ * Ready with entire loop
+ * Return to master
+ */
+ setAllowNodeStart(nodeId, true);
+ if (getNodeStatus(nodeId) == NodeRecord::STARTING) {
+ jam();
+ StartInfoConf * conf = (StartInfoConf*)&signal->theData[0];
+ conf->sendingNodeId = cownNodeId;
+ conf->startingNodeId = nodeId;
+ sendSignal(cmasterdihref, GSN_START_INFOCONF, signal,
+ StartInfoConf::SignalLength, JBB);
+ }//if
+ return;
+ }//if
+ ptrAss(tabPtr, tabRecord);
+ if (tabPtr.p->tabStatus == TabRecord::TS_ACTIVE) {
+ jam();
+ invalidateNodeLCP(signal, nodeId, tabPtr);
+ return;
+ }//if
+ tabPtr.i++;
+ }//for
+ signal->theData[0] = DihContinueB::ZINVALIDATE_NODE_LCP;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+}//Dbdih::invalidateNodeLCP()
+
+void
+Dbdih::invalidateNodeLCP(Signal* signal, Uint32 nodeId, TabRecordPtr tabPtr)
+{
+ /**
+ * Check so that no one else is using the tab descriptior
+ */
+ if (tabPtr.p->tabCopyStatus != TabRecord::CS_IDLE) {
+ jam();
+ signal->theData[0] = DihContinueB::ZINVALIDATE_NODE_LCP;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tabPtr.i;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 20, 3);
+ return;
+ }//if
+
+ /**
+ * For each fragment
+ */
+ bool modified = false;
+ FragmentstorePtr fragPtr;
+ for(Uint32 fragNo = 0; fragNo < tabPtr.p->totalfragments; fragNo++){
+ jam();
+ getFragstore(tabPtr.p, fragNo, fragPtr);
+ /**
+ * For each of replica record
+ */
+ ReplicaRecordPtr replicaPtr;
+ for(replicaPtr.i = fragPtr.p->oldStoredReplicas; replicaPtr.i != RNIL;
+ replicaPtr.i = replicaPtr.p->nextReplica) {
+ jam();
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ if(replicaPtr.p->procNode == nodeId){
+ jam();
+ /**
+ * Found one with correct node id
+ */
+ /**
+ * Invalidate all LCP's
+ */
+ modified = true;
+ for(int i = 0; i < MAX_LCP_STORED; i++) {
+ replicaPtr.p->lcpStatus[i] = ZINVALID;
+ }//if
+ /**
+ * And reset nextLcp
+ */
+ replicaPtr.p->nextLcp = 0;
+ }//if
+ }//for
+ }//for
+
+ if (modified) {
+ jam();
+ /**
+ * Save table description to disk
+ */
+ tabPtr.p->tabCopyStatus = TabRecord::CS_INVALIDATE_NODE_LCP;
+ tabPtr.p->tabUpdateState = TabRecord::US_INVALIDATE_NODE_LCP;
+ tabPtr.p->tabRemoveNode = nodeId;
+ signal->theData[0] = DihContinueB::ZPACK_TABLE_INTO_PAGES;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ }
+
+ jam();
+ /**
+ * Move to next table
+ */
+ tabPtr.i++;
+ signal->theData[0] = DihContinueB::ZINVALIDATE_NODE_LCP;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+}//Dbdih::invalidateNodeLCP()
+
+/*------------------------------------------------*/
+/* INPUT: TABPTR */
+/* TNODEID */
+/*------------------------------------------------*/
+void Dbdih::removeNodeFromTables(Signal* signal,
+ Uint32 nodeId, Uint32 tableId)
+{
+ jamEntry();
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ const Uint32 RT_BREAK = 64;
+ for (Uint32 i = 0; i<RT_BREAK; i++) {
+ jam();
+ if (tabPtr.i >= ctabFileSize){
+ jam();
+ removeNodeFromTablesComplete(signal, nodeId);
+ return;
+ }//if
+
+ ptrAss(tabPtr, tabRecord);
+ if (tabPtr.p->tabStatus == TabRecord::TS_ACTIVE) {
+ jam();
+ removeNodeFromTable(signal, nodeId, tabPtr);
+ return;
+ }//if
+ tabPtr.i++;
+ }//for
+ signal->theData[0] = DihContinueB::ZREMOVE_NODE_FROM_TABLE;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+}
+
+void Dbdih::removeNodeFromTable(Signal* signal,
+ Uint32 nodeId, TabRecordPtr tabPtr){
+
+ /**
+ * Check so that no one else is using the tab descriptior
+ */
+ if (tabPtr.p->tabCopyStatus != TabRecord::CS_IDLE) {
+ jam();
+ signal->theData[0] = DihContinueB::ZREMOVE_NODE_FROM_TABLE;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tabPtr.i;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 20, 3);
+ return;
+ }//if
+
+ /**
+ * For each fragment
+ */
+ Uint32 noOfRemovedReplicas = 0; // No of replicas removed
+ Uint32 noOfRemovedLcpReplicas = 0; // No of replicas in LCP removed
+ Uint32 noOfRemainingLcpReplicas = 0;// No of replicas in LCP remaining
+
+ //const Uint32 lcpId = SYSFILE->latestLCP_ID;
+ const bool lcpOngoingFlag = (tabPtr.p->tabLcpStatus== TabRecord::TLS_ACTIVE);
+
+ FragmentstorePtr fragPtr;
+ for(Uint32 fragNo = 0; fragNo < tabPtr.p->totalfragments; fragNo++){
+ jam();
+ getFragstore(tabPtr.p, fragNo, fragPtr);
+
+ /**
+ * For each of replica record
+ */
+ Uint32 replicaNo = 0;
+ ReplicaRecordPtr replicaPtr;
+ for(replicaPtr.i = fragPtr.p->storedReplicas; replicaPtr.i != RNIL;
+ replicaPtr.i = replicaPtr.p->nextReplica, replicaNo++) {
+ jam();
+
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ if(replicaPtr.p->procNode == nodeId){
+ jam();
+ noOfRemovedReplicas++;
+ removeNodeFromStored(nodeId, fragPtr, replicaPtr);
+ if(replicaPtr.p->lcpOngoingFlag){
+ jam();
+ /**
+ * This replica is currently LCP:ed
+ */
+ ndbrequire(fragPtr.p->noLcpReplicas > 0);
+ fragPtr.p->noLcpReplicas --;
+
+ noOfRemovedLcpReplicas ++;
+ replicaPtr.p->lcpOngoingFlag = false;
+ }
+ }
+ }
+ noOfRemainingLcpReplicas += fragPtr.p->noLcpReplicas;
+ }
+
+ if(noOfRemovedReplicas == 0){
+ jam();
+ /**
+ * The table had no replica on the failed node
+ * continue with next table
+ */
+ tabPtr.i++;
+ signal->theData[0] = DihContinueB::ZREMOVE_NODE_FROM_TABLE;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ }
+
+ /**
+ * We did remove at least one replica
+ */
+ bool ok = false;
+ switch(tabPtr.p->tabLcpStatus){
+ case TabRecord::TLS_COMPLETED:
+ ok = true;
+ jam();
+ /**
+ * WE WILL WRITE THE TABLE DESCRIPTION TO DISK AT THIS TIME
+ * INDEPENDENT OF WHAT THE LOCAL CHECKPOINT NEEDED.
+ * THIS IS TO ENSURE THAT THE FAILED NODES ARE ALSO UPDATED ON DISK
+ * IN THE DIH DATA STRUCTURES BEFORE WE COMPLETE HANDLING OF THE
+ * NODE FAILURE.
+ */
+ ndbrequire(noOfRemovedLcpReplicas == 0);
+
+ tabPtr.p->tabCopyStatus = TabRecord::CS_REMOVE_NODE;
+ tabPtr.p->tabUpdateState = TabRecord::US_REMOVE_NODE;
+ tabPtr.p->tabRemoveNode = nodeId;
+ signal->theData[0] = DihContinueB::ZPACK_TABLE_INTO_PAGES;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ case TabRecord::TLS_ACTIVE:
+ ok = true;
+ jam();
+ /**
+ * The table is participating in an LCP currently
+ */
+ // Fall through
+ break;
+ case TabRecord::TLS_WRITING_TO_FILE:
+ ok = true;
+ jam();
+ /**
+ * This should never happen since we in the beginning of this function
+ * checks the tabCopyStatus
+ */
+ ndbrequire(lcpOngoingFlag);
+ ndbrequire(false);
+ break;
+ }
+ ndbrequire(ok);
+
+ /**
+ * The table is participating in an LCP currently
+ * and we removed some replicas that should have been checkpointed
+ */
+ ndbrequire(c_lcpState.lcpStatus != LCP_STATUS_IDLE);
+ ndbrequire(tabPtr.p->tabLcpStatus == TabRecord::TLS_ACTIVE);
+
+ /**
+ * Save the table
+ */
+ tabPtr.p->tabCopyStatus = TabRecord::CS_REMOVE_NODE;
+ tabPtr.p->tabUpdateState = TabRecord::US_REMOVE_NODE;
+ tabPtr.p->tabRemoveNode = nodeId;
+ signal->theData[0] = DihContinueB::ZPACK_TABLE_INTO_PAGES;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+
+ if(noOfRemainingLcpReplicas == 0){
+ jam();
+ /**
+ * The removal on the failed node made the LCP complete
+ */
+ tabPtr.p->tabLcpStatus = TabRecord::TLS_WRITING_TO_FILE;
+ checkLcpAllTablesDoneInLqh();
+ }
+}
+
+void
+Dbdih::removeNodeFromTablesComplete(Signal* signal, Uint32 nodeId){
+ jam();
+
+ /**
+ * Check if we "accidently" completed a LCP
+ */
+ checkLcpCompletedLab(signal);
+
+ /**
+ * Check if we (DIH) are finished with node fail handling
+ */
+ checkLocalNodefailComplete(signal, nodeId, NF_REMOVE_NODE_FROM_TABLE);
+}
+
+void
+Dbdih::checkLocalNodefailComplete(Signal* signal, Uint32 failedNodeId,
+ NodefailHandlingStep step){
+ jam();
+
+ NodeRecordPtr nodePtr;
+ nodePtr.i = failedNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+
+ ndbrequire(nodePtr.p->m_nodefailSteps.get(step));
+ nodePtr.p->m_nodefailSteps.clear(step);
+
+ if(nodePtr.p->m_nodefailSteps.count() > 0){
+ jam();
+ return;
+ }
+
+ NFCompleteRep * const nf = (NFCompleteRep *)&signal->theData[0];
+ nf->blockNo = DBDIH;
+ nf->nodeId = cownNodeId;
+ nf->failedNodeId = failedNodeId;
+ nf->from = __LINE__;
+ sendSignal(reference(), GSN_NF_COMPLETEREP, signal,
+ NFCompleteRep::SignalLength, JBB);
+}
+
+
+void
+Dbdih::setLocalNodefailHandling(Signal* signal, Uint32 failedNodeId,
+ NodefailHandlingStep step){
+ jam();
+
+ NodeRecordPtr nodePtr;
+ nodePtr.i = failedNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+
+ ndbrequire(!nodePtr.p->m_nodefailSteps.get(step));
+ nodePtr.p->m_nodefailSteps.set(step);
+}
+
+void Dbdih::startLcpTakeOverLab(Signal* signal, Uint32 failedNodeId)
+{
+ /*--------------------------------------------------------------------*/
+ // Start LCP master take over process. Consists of the following steps.
+ // 1) Ensure that all LQH's have reported all fragments they have been
+ // told to checkpoint. Can be a fairly long step time-wise.
+ // 2) Query all nodes about their LCP status.
+ // During the query process we do not want our own state to change.
+ // This can change due to delayed reception of LCP_REPORT, completed
+ // save of table on disk or reception of DIH_LCPCOMPLETE from other
+ // node.
+ /*--------------------------------------------------------------------*/
+}//Dbdih::startLcpTakeOver()
+
+void Dbdih::execEMPTY_LCP_CONF(Signal* signal)
+{
+ jamEntry();
+
+ ndbrequire(c_lcpMasterTakeOverState.state == LMTOS_WAIT_EMPTY_LCP);
+
+ const EmptyLcpConf * const conf = (EmptyLcpConf *)&signal->theData[0];
+ Uint32 nodeId = conf->senderNodeId;
+
+ if(!conf->idle){
+ jam();
+ if (conf->tableId < c_lcpMasterTakeOverState.minTableId) {
+ jam();
+ c_lcpMasterTakeOverState.minTableId = conf->tableId;
+ c_lcpMasterTakeOverState.minFragId = conf->fragmentId;
+ } else if (conf->tableId == c_lcpMasterTakeOverState.minTableId &&
+ conf->fragmentId < c_lcpMasterTakeOverState.minFragId) {
+ jam();
+ c_lcpMasterTakeOverState.minFragId = conf->fragmentId;
+ }//if
+ if(isMaster()){
+ jam();
+ c_lcpState.m_LAST_LCP_FRAG_ORD.setWaitingFor(nodeId);
+ }
+ }
+
+ receiveLoopMacro(EMPTY_LCP_REQ, nodeId);
+ /*--------------------------------------------------------------------*/
+ // Received all EMPTY_LCPCONF. We can continue with next phase of the
+ // take over LCP master process.
+ /*--------------------------------------------------------------------*/
+ c_lcpMasterTakeOverState.set(LMTOS_WAIT_LCP_FRAG_REP, __LINE__);
+ checkEmptyLcpComplete(signal);
+ return;
+}//Dbdih::execEMPTY_LCPCONF()
+
+void
+Dbdih::checkEmptyLcpComplete(Signal *signal){
+
+ ndbrequire(c_lcpMasterTakeOverState.state == LMTOS_WAIT_LCP_FRAG_REP);
+
+ if(c_lcpState.noOfLcpFragRepOutstanding > 0){
+ jam();
+ return;
+ }
+
+ if(isMaster()){
+ jam();
+
+ signal->theData[0] = EventReport::LCP_TakeoverStarted;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 1, JBB);
+
+ signal->theData[0] = 7012;
+ execDUMP_STATE_ORD(signal);
+
+ c_lcpMasterTakeOverState.set(LMTOS_INITIAL, __LINE__);
+ MasterLCPReq * const req = (MasterLCPReq *)&signal->theData[0];
+ req->masterRef = reference();
+ req->failedNodeId = c_lcpMasterTakeOverState.failedNodeId;
+ sendLoopMacro(MASTER_LCPREQ, sendMASTER_LCPREQ);
+ } else {
+ sendMASTER_LCPCONF(signal);
+ }
+}
+
+/*--------------------------------------------------*/
+/* THE MASTER HAS FAILED AND THE NEW MASTER IS*/
+/* QUERYING THIS NODE ABOUT THE STATE OF THE */
+/* LOCAL CHECKPOINT PROTOCOL. */
+/*--------------------------------------------------*/
+void Dbdih::execMASTER_LCPREQ(Signal* signal)
+{
+ const MasterLCPReq * const req = (MasterLCPReq *)&signal->theData[0];
+ jamEntry();
+ const BlockReference newMasterBlockref = req->masterRef;
+
+ Uint32 failedNodeId = req->failedNodeId;
+
+ /**
+ * There can be no take over with the same master
+ */
+ ndbrequire(c_lcpState.m_masterLcpDihRef != newMasterBlockref);
+ c_lcpState.m_masterLcpDihRef = newMasterBlockref;
+ c_lcpState.m_MASTER_LCPREQ_Received = true;
+ c_lcpState.m_MASTER_LCPREQ_FailedNodeId = failedNodeId;
+
+ if(newMasterBlockref != cmasterdihref){
+ jam();
+ ndbrequire(0);
+ }
+
+ sendMASTER_LCPCONF(signal);
+}//Dbdih::execMASTER_LCPREQ()
+
+void
+Dbdih::sendMASTER_LCPCONF(Signal * signal){
+
+ if(!c_EMPTY_LCP_REQ_Counter.done()){
+ /**
+ * Have not received all EMPTY_LCP_REP
+ * dare not answer MASTER_LCP_CONF yet
+ */
+ jam();
+ return;
+ }
+
+ if(!c_lcpState.m_MASTER_LCPREQ_Received){
+ jam();
+ /**
+ * Has not received MASTER_LCPREQ yet
+ */
+ return;
+ }
+
+ if(c_lcpState.lcpStatus == LCP_INIT_TABLES){
+ jam();
+ /**
+ * Still aborting old initLcpLab
+ */
+ return;
+ }
+
+ if(c_lcpState.lcpStatus == LCP_COPY_GCI){
+ jam();
+ /**
+ * Restart it
+ */
+ //Uint32 lcpId = SYSFILE->latestLCP_ID;
+ SYSFILE->latestLCP_ID--;
+ c_lcpState.setLcpStatus(LCP_STATUS_IDLE, __LINE__);
+#if 0
+ if(c_copyGCISlave.m_copyReason == CopyGCIReq::LOCAL_CHECKPOINT){
+ ndbout_c("Dbdih: Also resetting c_copyGCISlave");
+ c_copyGCISlave.m_copyReason = CopyGCIReq::IDLE;
+ c_copyGCISlave.m_expectedNextWord = 0;
+ }
+#endif
+ }
+
+ bool ok = false;
+ MasterLCPConf::State lcpState;
+ switch (c_lcpState.lcpStatus) {
+ case LCP_STATUS_IDLE:
+ ok = true;
+ jam();
+ /*------------------------------------------------*/
+ /* LOCAL CHECKPOINT IS CURRENTLY NOT ACTIVE */
+ /* SINCE NO COPY OF RESTART INFORMATION HAVE*/
+ /* BEEN RECEIVED YET. ALSO THE PREVIOUS */
+ /* CHECKPOINT HAVE BEEN FULLY COMPLETED. */
+ /*------------------------------------------------*/
+ lcpState = MasterLCPConf::LCP_STATUS_IDLE;
+ break;
+ case LCP_STATUS_ACTIVE:
+ ok = true;
+ jam();
+ /*--------------------------------------------------*/
+ /* COPY OF RESTART INFORMATION HAS BEEN */
+ /* PERFORMED AND ALSO RESPONSE HAVE BEEN SENT.*/
+ /*--------------------------------------------------*/
+ lcpState = MasterLCPConf::LCP_STATUS_ACTIVE;
+ break;
+ case LCP_TAB_COMPLETED:
+ ok = true;
+ jam();
+ /*--------------------------------------------------------*/
+ /* ALL LCP_REPORT'S HAVE BEEN COMPLETED FOR */
+ /* ALL TABLES. SAVE OF AT LEAST ONE TABLE IS */
+ /* ONGOING YET. */
+ /*--------------------------------------------------------*/
+ lcpState = MasterLCPConf::LCP_TAB_COMPLETED;
+ break;
+ case LCP_TAB_SAVED:
+ ok = true;
+ jam();
+ /*--------------------------------------------------------*/
+ /* ALL LCP_REPORT'S HAVE BEEN COMPLETED FOR */
+ /* ALL TABLES. ALL TABLES HAVE ALSO BEEN SAVED */
+ /* ALL OTHER NODES ARE NOT YET FINISHED WITH */
+ /* THE LOCAL CHECKPOINT. */
+ /*--------------------------------------------------------*/
+ lcpState = MasterLCPConf::LCP_TAB_SAVED;
+ break;
+ case LCP_TCGET:
+ case LCP_CALCULATE_KEEP_GCI:
+ case LCP_TC_CLOPSIZE:
+ case LCP_START_LCP_ROUND:
+ /**
+ * These should only exists on the master
+ * but since this is master take over
+ * it not allowed
+ */
+ ndbrequire(false);
+ break;
+ case LCP_COPY_GCI:
+ case LCP_INIT_TABLES:
+ ok = true;
+ /**
+ * These two states are handled by if statements above
+ */
+ ndbrequire(false);
+ break;
+ }//switch
+ ndbrequire(ok);
+
+ Uint32 failedNodeId = c_lcpState.m_MASTER_LCPREQ_FailedNodeId;
+ MasterLCPConf * const conf = (MasterLCPConf *)&signal->theData[0];
+ conf->senderNodeId = cownNodeId;
+ conf->lcpState = lcpState;
+ conf->failedNodeId = failedNodeId;
+ sendSignal(c_lcpState.m_masterLcpDihRef, GSN_MASTER_LCPCONF,
+ signal, MasterLCPConf::SignalLength, JBB);
+
+ // Answer to MASTER_LCPREQ sent, reset flag so
+ // that it's not sent again before another request comes in
+ c_lcpState.m_MASTER_LCPREQ_Received = false;
+
+ if(c_lcpState.lcpStatus == LCP_TAB_SAVED){
+#ifdef VM_TRACE
+ ndbout_c("Sending extra GSN_LCP_COMPLETE_REP to new master");
+#endif
+ sendLCP_COMPLETE_REP(signal);
+ }
+
+ if(!isMaster()){
+ c_lcpMasterTakeOverState.set(LMTOS_IDLE, __LINE__);
+ checkLocalNodefailComplete(signal, failedNodeId, NF_LCP_TAKE_OVER);
+ }
+
+ return;
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbdih::LcpMasterTakeOverState state){
+ switch(state){
+ case Dbdih::LMTOS_IDLE:
+ out << "LMTOS_IDLE";
+ break;
+ case Dbdih::LMTOS_WAIT_EMPTY_LCP:
+ out << "LMTOS_WAIT_EMPTY_LCP";
+ break;
+ case Dbdih::LMTOS_WAIT_LCP_FRAG_REP:
+ out << "LMTOS_WAIT_EMPTY_LCP";
+ break;
+ case Dbdih::LMTOS_INITIAL:
+ out << "LMTOS_INITIAL";
+ break;
+ case Dbdih::LMTOS_ALL_IDLE:
+ out << "LMTOS_ALL_IDLE";
+ break;
+ case Dbdih::LMTOS_ALL_ACTIVE:
+ out << "LMTOS_ALL_ACTIVE";
+ break;
+ case Dbdih::LMTOS_LCP_CONCLUDING:
+ out << "LMTOS_LCP_CONCLUDING";
+ break;
+ case Dbdih::LMTOS_COPY_ONGOING:
+ out << "LMTOS_COPY_ONGOING";
+ break;
+ }
+ return out;
+}
+
+struct MASTERLCP_StateTransitions {
+ Dbdih::LcpMasterTakeOverState CurrentState;
+ MasterLCPConf::State ParticipantState;
+ Dbdih::LcpMasterTakeOverState NewState;
+};
+
+static const
+MASTERLCP_StateTransitions g_masterLCPTakeoverStateTransitions[] = {
+ /**
+ * Current = LMTOS_INITIAL
+ */
+ { Dbdih::LMTOS_INITIAL,
+ MasterLCPConf::LCP_STATUS_IDLE,
+ Dbdih::LMTOS_ALL_IDLE },
+
+ { Dbdih::LMTOS_INITIAL,
+ MasterLCPConf::LCP_STATUS_ACTIVE,
+ Dbdih::LMTOS_ALL_ACTIVE },
+
+ { Dbdih::LMTOS_INITIAL,
+ MasterLCPConf::LCP_TAB_COMPLETED,
+ Dbdih::LMTOS_LCP_CONCLUDING },
+
+ { Dbdih::LMTOS_INITIAL,
+ MasterLCPConf::LCP_TAB_SAVED,
+ Dbdih::LMTOS_LCP_CONCLUDING },
+
+ /**
+ * Current = LMTOS_ALL_IDLE
+ */
+ { Dbdih::LMTOS_ALL_IDLE,
+ MasterLCPConf::LCP_STATUS_IDLE,
+ Dbdih::LMTOS_ALL_IDLE },
+
+ { Dbdih::LMTOS_ALL_IDLE,
+ MasterLCPConf::LCP_STATUS_ACTIVE,
+ Dbdih::LMTOS_COPY_ONGOING },
+
+ { Dbdih::LMTOS_ALL_IDLE,
+ MasterLCPConf::LCP_TAB_COMPLETED,
+ Dbdih::LMTOS_LCP_CONCLUDING },
+
+ { Dbdih::LMTOS_ALL_IDLE,
+ MasterLCPConf::LCP_TAB_SAVED,
+ Dbdih::LMTOS_LCP_CONCLUDING },
+
+ /**
+ * Current = LMTOS_COPY_ONGOING
+ */
+ { Dbdih::LMTOS_COPY_ONGOING,
+ MasterLCPConf::LCP_STATUS_IDLE,
+ Dbdih::LMTOS_COPY_ONGOING },
+
+ { Dbdih::LMTOS_COPY_ONGOING,
+ MasterLCPConf::LCP_STATUS_ACTIVE,
+ Dbdih::LMTOS_COPY_ONGOING },
+
+ /**
+ * Current = LMTOS_ALL_ACTIVE
+ */
+ { Dbdih::LMTOS_ALL_ACTIVE,
+ MasterLCPConf::LCP_STATUS_IDLE,
+ Dbdih::LMTOS_COPY_ONGOING },
+
+ { Dbdih::LMTOS_ALL_ACTIVE,
+ MasterLCPConf::LCP_STATUS_ACTIVE,
+ Dbdih::LMTOS_ALL_ACTIVE },
+
+ { Dbdih::LMTOS_ALL_ACTIVE,
+ MasterLCPConf::LCP_TAB_COMPLETED,
+ Dbdih::LMTOS_LCP_CONCLUDING },
+
+ { Dbdih::LMTOS_ALL_ACTIVE,
+ MasterLCPConf::LCP_TAB_SAVED,
+ Dbdih::LMTOS_LCP_CONCLUDING },
+
+ /**
+ * Current = LMTOS_LCP_CONCLUDING
+ */
+ { Dbdih::LMTOS_LCP_CONCLUDING,
+ MasterLCPConf::LCP_STATUS_IDLE,
+ Dbdih::LMTOS_LCP_CONCLUDING },
+
+ { Dbdih::LMTOS_LCP_CONCLUDING,
+ MasterLCPConf::LCP_STATUS_ACTIVE,
+ Dbdih::LMTOS_LCP_CONCLUDING },
+
+ { Dbdih::LMTOS_LCP_CONCLUDING,
+ MasterLCPConf::LCP_TAB_COMPLETED,
+ Dbdih::LMTOS_LCP_CONCLUDING },
+
+ { Dbdih::LMTOS_LCP_CONCLUDING,
+ MasterLCPConf::LCP_TAB_SAVED,
+ Dbdih::LMTOS_LCP_CONCLUDING }
+};
+
+const Uint32 g_masterLCPTakeoverStateTransitionsRows =
+sizeof(g_masterLCPTakeoverStateTransitions) / sizeof(struct MASTERLCP_StateTransitions);
+
+void Dbdih::execMASTER_LCPCONF(Signal* signal)
+{
+ const MasterLCPConf * const conf = (MasterLCPConf *)&signal->theData[0];
+ jamEntry();
+ Uint32 senderNodeId = conf->senderNodeId;
+ MasterLCPConf::State lcpState = (MasterLCPConf::State)conf->lcpState;
+ const Uint32 failedNodeId = conf->failedNodeId;
+ NodeRecordPtr nodePtr;
+ nodePtr.i = senderNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ nodePtr.p->lcpStateAtTakeOver = lcpState;
+
+#ifdef VM_TRACE
+ ndbout_c("MASTER_LCPCONF");
+ printMASTER_LCP_CONF(stdout, &signal->theData[0], 0, 0);
+#endif
+
+ bool found = false;
+ for(Uint32 i = 0; i<g_masterLCPTakeoverStateTransitionsRows; i++){
+ const struct MASTERLCP_StateTransitions * valid =
+ &g_masterLCPTakeoverStateTransitions[i];
+
+ if(valid->CurrentState == c_lcpMasterTakeOverState.state &&
+ valid->ParticipantState == lcpState){
+ jam();
+ found = true;
+ c_lcpMasterTakeOverState.set(valid->NewState, __LINE__);
+ break;
+ }
+ }
+ ndbrequire(found);
+
+ bool ok = false;
+ switch(lcpState){
+ case MasterLCPConf::LCP_STATUS_IDLE:
+ ok = true;
+ break;
+ case MasterLCPConf::LCP_STATUS_ACTIVE:
+ case MasterLCPConf::LCP_TAB_COMPLETED:
+ case MasterLCPConf::LCP_TAB_SAVED:
+ ok = true;
+ c_lcpState.m_LCP_COMPLETE_REP_Counter_DIH.setWaitingFor(nodePtr.i);
+ break;
+ }
+ ndbrequire(ok);
+
+ receiveLoopMacro(MASTER_LCPREQ, senderNodeId);
+ /*-------------------------------------------------------------------------*/
+ // We have now received all responses and are ready to take over the LCP
+ // protocol as master.
+ /*-------------------------------------------------------------------------*/
+ MASTER_LCPhandling(signal, failedNodeId);
+}//Dbdih::execMASTER_LCPCONF()
+
+void Dbdih::execMASTER_LCPREF(Signal* signal)
+{
+ const MasterLCPRef * const ref = (MasterLCPRef *)&signal->theData[0];
+ jamEntry();
+ receiveLoopMacro(MASTER_LCPREQ, ref->senderNodeId);
+ /*-------------------------------------------------------------------------*/
+ // We have now received all responses and are ready to take over the LCP
+ // protocol as master.
+ /*-------------------------------------------------------------------------*/
+ MASTER_LCPhandling(signal, ref->failedNodeId);
+}//Dbdih::execMASTER_LCPREF()
+
+void Dbdih::MASTER_LCPhandling(Signal* signal, Uint32 failedNodeId)
+{
+ /*-------------------------------------------------------------------------
+ *
+ * WE ARE NOW READY TO CONCLUDE THE TAKE OVER AS MASTER.
+ * WE HAVE ENOUGH INFO TO START UP ACTIVITIES IN THE PROPER PLACE.
+ * ALSO SET THE PROPER STATE VARIABLES.
+ *------------------------------------------------------------------------*/
+ c_lcpState.currentFragment.tableId = c_lcpMasterTakeOverState.minTableId;
+ c_lcpState.currentFragment.fragmentId = c_lcpMasterTakeOverState.minFragId;
+ c_lcpState.m_LAST_LCP_FRAG_ORD = c_lcpState.m_LCP_COMPLETE_REP_Counter_LQH;
+
+ NodeRecordPtr failedNodePtr;
+ failedNodePtr.i = failedNodeId;
+ ptrCheckGuard(failedNodePtr, MAX_NDB_NODES, nodeRecord);
+
+ switch (c_lcpMasterTakeOverState.state) {
+ case LMTOS_ALL_IDLE:
+ jam();
+ /* --------------------------------------------------------------------- */
+ // All nodes were idle in the LCP protocol. Start checking for start of LCP
+ // protocol.
+ /* --------------------------------------------------------------------- */
+#ifdef VM_TRACE
+ ndbout_c("MASTER_LCPhandling:: LMTOS_ALL_IDLE -> checkLcpStart");
+#endif
+ checkLcpStart(signal, __LINE__);
+ break;
+ case LMTOS_COPY_ONGOING:
+ jam();
+ /* --------------------------------------------------------------------- */
+ // We were in the starting process of the LCP protocol. We will restart the
+ // protocol by calculating the keep gci and storing the new lcp id.
+ /* --------------------------------------------------------------------- */
+#ifdef VM_TRACE
+ ndbout_c("MASTER_LCPhandling:: LMTOS_COPY_ONGOING -> storeNewLcpId");
+#endif
+ if (c_lcpState.lcpStatus == LCP_STATUS_ACTIVE) {
+ jam();
+ /*---------------------------------------------------------------------*/
+ /* WE NEED TO DECREASE THE LATEST LCP ID SINCE WE HAVE ALREADY */
+ /* STARTED THIS */
+ /* LOCAL CHECKPOINT. */
+ /*---------------------------------------------------------------------*/
+ Uint32 lcpId = SYSFILE->latestLCP_ID;
+#ifdef VM_TRACE
+ ndbout_c("Decreasing latestLCP_ID from %d to %d", lcpId, lcpId - 1);
+#endif
+ SYSFILE->latestLCP_ID--;
+ }//if
+ storeNewLcpIdLab(signal);
+ break;
+ case LMTOS_ALL_ACTIVE:
+ {
+ jam();
+ /* -------------------------------------------------------------------
+ * Everybody was in the active phase. We will restart sending
+ * LCP_FRAGORD to the nodes from the new master.
+ * We also need to set dihLcpStatus to ZACTIVE
+ * in the master node since the master will wait for all nodes to
+ * complete before finalising the LCP process.
+ * ------------------------------------------------------------------ */
+#ifdef VM_TRACE
+ ndbout_c("MASTER_LCPhandling:: LMTOS_ALL_ACTIVE -> "
+ "startLcpRoundLoopLab(table=%u, fragment=%u)",
+ c_lcpMasterTakeOverState.minTableId,
+ c_lcpMasterTakeOverState.minFragId);
+#endif
+
+ c_lcpState.keepGci = SYSFILE->keepGCI;
+ c_lcpState.setLcpStatus(LCP_START_LCP_ROUND, __LINE__);
+ startLcpRoundLoopLab(signal, 0, 0);
+ break;
+ }
+ case LMTOS_LCP_CONCLUDING:
+ {
+ jam();
+ /* ------------------------------------------------------------------- */
+ // The LCP process is in the finalisation phase. We simply wait for it to
+ // complete with signals arriving in. We need to check also if we should
+ // change state due to table write completion during state
+ // collection phase.
+ /* ------------------------------------------------------------------- */
+ ndbrequire(c_lcpState.lcpStatus != LCP_STATUS_IDLE);
+ startLcpRoundLoopLab(signal, 0, 0);
+ break;
+ }
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ signal->theData[0] = EventReport::LCP_TakeoverCompleted;
+ signal->theData[1] = c_lcpMasterTakeOverState.state;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ signal->theData[0] = 7012;
+ execDUMP_STATE_ORD(signal);
+
+ signal->theData[0] = 7015;
+ execDUMP_STATE_ORD(signal);
+
+ c_lcpMasterTakeOverState.set(LMTOS_IDLE, __LINE__);
+
+ checkLocalNodefailComplete(signal, failedNodePtr.i, NF_LCP_TAKE_OVER);
+}
+
+/* ------------------------------------------------------------------------- */
+/* A BLOCK OR A NODE HAS COMPLETED THE HANDLING OF THE NODE FAILURE. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::execNF_COMPLETEREP(Signal* signal)
+{
+ NodeRecordPtr failedNodePtr;
+ NFCompleteRep * const nfCompleteRep = (NFCompleteRep *)&signal->theData[0];
+ jamEntry();
+ const Uint32 blockNo = nfCompleteRep->blockNo;
+ Uint32 nodeId = nfCompleteRep->nodeId;
+ failedNodePtr.i = nfCompleteRep->failedNodeId;
+
+ ptrCheckGuard(failedNodePtr, MAX_NDB_NODES, nodeRecord);
+ switch (blockNo) {
+ case DBTC:
+ jam();
+ ndbrequire(failedNodePtr.p->dbtcFailCompleted == ZFALSE);
+ /* -------------------------------------------------------------------- */
+ // Report the event that DBTC completed node failure handling.
+ /* -------------------------------------------------------------------- */
+ signal->theData[0] = EventReport::NodeFailCompleted;
+ signal->theData[1] = DBTC;
+ signal->theData[2] = failedNodePtr.i;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+
+ failedNodePtr.p->dbtcFailCompleted = ZTRUE;
+ break;
+ case DBDICT:
+ jam();
+ ndbrequire(failedNodePtr.p->dbdictFailCompleted == ZFALSE);
+ /* --------------------------------------------------------------------- */
+ // Report the event that DBDICT completed node failure handling.
+ /* --------------------------------------------------------------------- */
+ signal->theData[0] = EventReport::NodeFailCompleted;
+ signal->theData[1] = DBDICT;
+ signal->theData[2] = failedNodePtr.i;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+
+ failedNodePtr.p->dbdictFailCompleted = ZTRUE;
+ break;
+ case DBDIH:
+ jam();
+ ndbrequire(failedNodePtr.p->dbdihFailCompleted == ZFALSE);
+ /* --------------------------------------------------------------------- */
+ // Report the event that DBDIH completed node failure handling.
+ /* --------------------------------------------------------------------- */
+ signal->theData[0] = EventReport::NodeFailCompleted;
+ signal->theData[1] = DBDIH;
+ signal->theData[2] = failedNodePtr.i;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+
+ failedNodePtr.p->dbdihFailCompleted = ZTRUE;
+ break;
+ case DBLQH:
+ jam();
+ ndbrequire(failedNodePtr.p->dblqhFailCompleted == ZFALSE);
+ /* --------------------------------------------------------------------- */
+ // Report the event that DBDIH completed node failure handling.
+ /* --------------------------------------------------------------------- */
+ signal->theData[0] = EventReport::NodeFailCompleted;
+ signal->theData[1] = DBLQH;
+ signal->theData[2] = failedNodePtr.i;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+
+ failedNodePtr.p->dblqhFailCompleted = ZTRUE;
+ break;
+ case 0: /* Node has finished */
+ jam();
+ ndbrequire(nodeId < MAX_NDB_NODES);
+
+ if (failedNodePtr.p->recNODE_FAILREP == ZFALSE) {
+ jam();
+ /* ------------------------------------------------------------------- */
+ // We received a report about completion of node failure before we
+ // received the message about the NODE failure ourselves.
+ // We will send the signal to ourselves with a small delay
+ // (10 milliseconds).
+ /* ------------------------------------------------------------------- */
+ //nf->from = __LINE__;
+ sendSignalWithDelay(reference(), GSN_NF_COMPLETEREP, signal, 10,
+ signal->length());
+ return;
+ }//if
+
+ if (!failedNodePtr.p->m_NF_COMPLETE_REP.isWaitingFor(nodeId)){
+ jam();
+ return;
+ }
+
+ failedNodePtr.p->m_NF_COMPLETE_REP.clearWaitingFor(nodeId);;
+
+ /* -------------------------------------------------------------------- */
+ // Report the event that nodeId has completed node failure handling.
+ /* -------------------------------------------------------------------- */
+ signal->theData[0] = EventReport::NodeFailCompleted;
+ signal->theData[1] = 0;
+ signal->theData[2] = failedNodePtr.i;
+ signal->theData[3] = nodeId;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+
+ nodeFailCompletedCheckLab(signal, failedNodePtr);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ break;
+ }//switch
+ if (failedNodePtr.p->dbtcFailCompleted == ZFALSE) {
+ jam();
+ return;
+ }//if
+ if (failedNodePtr.p->dbdictFailCompleted == ZFALSE) {
+ jam();
+ return;
+ }//if
+ if (failedNodePtr.p->dbdihFailCompleted == ZFALSE) {
+ jam();
+ return;
+ }//if
+ if (failedNodePtr.p->dblqhFailCompleted == ZFALSE) {
+ jam();
+ return;
+ }//if
+ /* ----------------------------------------------------------------------- */
+ /* ALL BLOCKS IN THIS NODE HAVE COMPLETED THEIR PART OF HANDLING THE */
+ /* NODE FAILURE. WE CAN NOW REPORT THIS COMPLETION TO ALL OTHER NODES. */
+ /* ----------------------------------------------------------------------- */
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if (nodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ jam();
+ BlockReference ref = calcDihBlockRef(nodePtr.i);
+ NFCompleteRep * const nf = (NFCompleteRep *)&signal->theData[0];
+ nf->blockNo = 0;
+ nf->nodeId = cownNodeId;
+ nf->failedNodeId = failedNodePtr.i;
+ nf->from = __LINE__;
+ sendSignal(ref, GSN_NF_COMPLETEREP, signal,
+ NFCompleteRep::SignalLength, JBB);
+ }//if
+ }//for
+ return;
+}//Dbdih::execNF_COMPLETEREP()
+
+void Dbdih::nodeFailCompletedCheckLab(Signal* signal,
+ NodeRecordPtr failedNodePtr)
+{
+ jam();
+ if (!failedNodePtr.p->m_NF_COMPLETE_REP.done()){
+ jam();
+ return;
+ }//if
+ /* ---------------------------------------------------------------------- */
+ /* ALL BLOCKS IN ALL NODES HAVE NOW REPORTED COMPLETION OF THE NODE */
+ /* FAILURE HANDLING. WE ARE NOW READY TO ACCEPT THAT THIS NODE STARTS */
+ /* AGAIN. */
+ /* ---------------------------------------------------------------------- */
+ jam();
+ failedNodePtr.p->nodeStatus = NodeRecord::DEAD;
+ failedNodePtr.p->recNODE_FAILREP = ZFALSE;
+
+ /* ---------------------------------------------------------------------- */
+ // Report the event that all nodes completed node failure handling.
+ /* ---------------------------------------------------------------------- */
+ signal->theData[0] = EventReport::NodeFailCompleted;
+ signal->theData[1] = 0;
+ signal->theData[2] = failedNodePtr.i;
+ signal->theData[3] = 0;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+
+ /* ---------------------------------------------------------------------- */
+ // Report to QMGR that we have concluded recovery handling of this node.
+ /* ---------------------------------------------------------------------- */
+ signal->theData[0] = failedNodePtr.i;
+ sendSignal(QMGR_REF, GSN_NDB_FAILCONF, signal, 1, JBB);
+
+ if (isMaster()) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* IF WE ARE MASTER WE MUST CHECK IF COPY FRAGMENT WAS INTERRUPTED */
+ /* BY THE FAILED NODES. */
+ /* --------------------------------------------------------------------- */
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = 0;
+ ptrAss(takeOverPtr, takeOverRecord);
+ if ((takeOverPtr.p->toMasterStatus == TakeOverRecord::COPY_FRAG) &&
+ (failedNodePtr.i == takeOverPtr.p->toCopyNode)) {
+ jam();
+#ifdef VM_TRACE
+ ndbrequire("Tell jonas" == 0);
+#endif
+ /*------------------------------------------------------------------*/
+ /* WE ARE CURRENTLY IN THE PROCESS OF COPYING A FRAGMENT. WE */
+ /* WILL CHECK IF THE COPY NODE HAVE FAILED. */
+ /*------------------------------------------------------------------*/
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::SELECTING_NEXT;
+ startNextCopyFragment(signal, takeOverPtr.i);
+ return;
+ }//if
+ checkStartTakeOver(signal);
+ }//if
+ return;
+}//Dbdih::nodeFailCompletedCheckLab()
+
+/*****************************************************************************/
+/* ********** SEIZING / RELEASING MODULE *************/
+/*****************************************************************************/
+/*
+ 3.4 L O C A L N O D E S E I Z E
+ ************************************
+ */
+/*
+ 3.4.1 L O C A L N O D E S E I Z E R E Q U E S T
+ ******************************************************
+ */
+void Dbdih::execDISEIZEREQ(Signal* signal)
+{
+ ConnectRecordPtr connectPtr;
+ jamEntry();
+ Uint32 userPtr = signal->theData[0];
+ BlockReference userRef = signal->theData[1];
+ ndbrequire(cfirstconnect != RNIL);
+ connectPtr.i = cfirstconnect;
+ ptrCheckGuard(connectPtr, cconnectFileSize, connectRecord);
+ cfirstconnect = connectPtr.p->nfConnect;
+ connectPtr.p->nfConnect = RNIL;
+ connectPtr.p->userpointer = userPtr;
+ connectPtr.p->userblockref = userRef;
+ connectPtr.p->connectState = ConnectRecord::INUSE;
+ signal->theData[0] = connectPtr.p->userpointer;
+ signal->theData[1] = connectPtr.i;
+ sendSignal(userRef, GSN_DISEIZECONF, signal, 2, JBB);
+}//Dbdih::execDISEIZEREQ()
+
+/*
+ 3.5 L O C A L N O D E R E L E A S E
+ ****************************************
+ */
+/*
+ 3.5.1 L O C A L N O D E R E L E A S E R E Q U E S T
+ *******************************************************=
+ */
+void Dbdih::execDIRELEASEREQ(Signal* signal)
+{
+ ConnectRecordPtr connectPtr;
+ jamEntry();
+ connectPtr.i = signal->theData[0];
+ Uint32 userRef = signal->theData[2];
+ ptrCheckGuard(connectPtr, cconnectFileSize, connectRecord);
+ ndbrequire(connectPtr.p->connectState != ConnectRecord::FREE);
+ ndbrequire(connectPtr.p->userblockref == userRef);
+ connectPtr.p->connectState = ConnectRecord::FREE;
+ signal->theData[0] = connectPtr.p->userpointer;
+ sendSignal(connectPtr.p->userblockref, GSN_DIRELEASECONF, signal, 1, JBB);
+ connectPtr.p->nfConnect = cfirstconnect;
+ cfirstconnect = connectPtr.i;
+ connectPtr.p->userblockref = ZNIL;
+ connectPtr.p->userpointer = RNIL;
+}//Dbdih::execDIRELEASEREQ()
+
+/*
+ 3.7 A D D T A B L E
+ **********************=
+ */
+/*****************************************************************************/
+/* ********** TABLE ADDING MODULE *************/
+/*****************************************************************************/
+/*
+ 3.7.1 A D D T A B L E M A I N L Y
+ ***************************************
+ */
+void Dbdih::execCREATE_FRAGMENTATION_REQ(Signal * signal){
+ jamEntry();
+ CreateFragmentationReq * const req =
+ (CreateFragmentationReq*)signal->getDataPtr();
+
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 senderData = req->senderData;
+ const Uint32 fragmentNode = req->fragmentNode;
+ const Uint32 fragmentType = req->fragmentationType;
+ //const Uint32 fragmentCount = req->noOfFragments;
+ const Uint32 primaryTableId = req->primaryTableId;
+
+ Uint32 err = 0;
+
+ do {
+ Uint32 noOfFragments = 0;
+ Uint32 noOfReplicas = cnoReplicas;
+ switch(fragmentType){
+ case DictTabInfo::AllNodesSmallTable:
+ jam();
+ noOfFragments = cnoOfNodeGroups;
+ break;
+ case DictTabInfo::AllNodesMediumTable:
+ jam();
+ noOfFragments = 2 * cnoOfNodeGroups;
+ break;
+ case DictTabInfo::AllNodesLargeTable:
+ jam();
+ noOfFragments = 8 * cnoOfNodeGroups;
+ break;
+ case DictTabInfo::SingleFragment:
+ jam();
+ noOfFragments = 1;
+ break;
+#if 0
+ case DictTabInfo::SpecifiedFragmentCount:
+ noOfFragments = (fragmentCount == 0 ? 1 : (fragmentCount + 1)/ 2);
+ break;
+#endif
+ default:
+ jam();
+ err = CreateFragmentationRef::InvalidFragmentationType;
+ break;
+ }
+ if(err)
+ break;
+
+ NodeGroupRecordPtr NGPtr;
+ TabRecordPtr primTabPtr;
+ if (primaryTableId == RNIL) {
+ if(fragmentNode == 0){
+ jam();
+ NGPtr.i = c_nextNodeGroup;
+ c_nextNodeGroup = (NGPtr.i + 1 == cnoOfNodeGroups ? 0 : NGPtr.i + 1);
+ } else if(! (fragmentNode < MAX_NDB_NODES)) {
+ jam();
+ err = CreateFragmentationRef::InvalidNodeId;
+ } else {
+ jam();
+ const Uint32 stat = Sysfile::getNodeStatus(fragmentNode,
+ SYSFILE->nodeStatus);
+ switch (stat) {
+ case Sysfile::NS_Active:
+ case Sysfile::NS_ActiveMissed_1:
+ case Sysfile::NS_ActiveMissed_2:
+ case Sysfile::NS_TakeOver:
+ jam();
+ break;
+ case Sysfile::NS_NotActive_NotTakenOver:
+ jam();
+ break;
+ case Sysfile::NS_HotSpare:
+ jam();
+ case Sysfile::NS_NotDefined:
+ jam();
+ default:
+ jam();
+ err = CreateFragmentationRef::InvalidNodeType;
+ break;
+ }
+ if(err)
+ break;
+ NGPtr.i = Sysfile::getNodeGroup(fragmentNode,
+ SYSFILE->nodeGroups);
+ break;
+ }
+ } else {
+ if (primaryTableId >= ctabFileSize) {
+ jam();
+ err = CreateFragmentationRef::InvalidPrimaryTable;
+ break;
+ }
+ primTabPtr.i = primaryTableId;
+ ptrAss(primTabPtr, tabRecord);
+ if (primTabPtr.p->tabStatus != TabRecord::TS_ACTIVE) {
+ jam();
+ err = CreateFragmentationRef::InvalidPrimaryTable;
+ break;
+ }
+ if (noOfFragments != primTabPtr.p->totalfragments) {
+ jam();
+ err = CreateFragmentationRef::InvalidFragmentationType;
+ break;
+ }
+ }
+
+ //@todo use section writer
+ Uint32 count = 2;
+ Uint32 fragments[2 + 8*MAX_REPLICAS*MAX_NDB_NODES];
+ if (primaryTableId == RNIL) {
+ jam();
+ for(Uint32 fragNo = 0; fragNo<noOfFragments; fragNo++){
+ jam();
+ ptrCheckGuard(NGPtr, MAX_NDB_NODES, nodeGroupRecord);
+
+ Uint32 ind = NGPtr.p->nextReplicaNode;
+ const Uint32 max = NGPtr.p->nodeCount;
+
+ //-------------------------------------------------------------------
+ // We make an extra step to ensure that the primary replicas are
+ // spread among the nodes.
+ //-------------------------------------------------------------------
+ NGPtr.p->nextReplicaNode = (ind + 1 >= max ? 0 : ind + 1);
+
+ for(Uint32 replicaNo = 0; replicaNo<noOfReplicas; replicaNo++){
+ jam();
+ const Uint32 nodeId = NGPtr.p->nodesInGroup[ind++];
+ fragments[count++] = nodeId;
+ ind = (ind == max ? 0 : ind);
+ }
+
+ /**
+ * Next node group for next fragment
+ */
+ NGPtr.i++;
+ NGPtr.i = (NGPtr.i == cnoOfNodeGroups ? 0 : NGPtr.i);
+ }
+ } else {
+ for (Uint32 fragNo = 0;
+ fragNo < primTabPtr.p->totalfragments; fragNo++) {
+ jam();
+ FragmentstorePtr fragPtr;
+ ReplicaRecordPtr replicaPtr;
+ getFragstore(primTabPtr.p, fragNo, fragPtr);
+ fragments[count++] = fragPtr.p->preferredPrimary;
+ for (replicaPtr.i = fragPtr.p->storedReplicas;
+ replicaPtr.i != RNIL;
+ replicaPtr.i = replicaPtr.p->nextReplica) {
+ jam();
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ if (replicaPtr.p->procNode != fragPtr.p->preferredPrimary) {
+ jam();
+ fragments[count++] = replicaPtr.p->procNode;
+ }//if
+ }//for
+ for (replicaPtr.i = fragPtr.p->oldStoredReplicas;
+ replicaPtr.i != RNIL;
+ replicaPtr.i = replicaPtr.p->nextReplica) {
+ jam();
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ if (replicaPtr.p->procNode != fragPtr.p->preferredPrimary) {
+ jam();
+ fragments[count++] = replicaPtr.p->procNode;
+ }//if
+ }//for
+ }
+ }
+ ndbrequire(count == (2 + noOfReplicas * noOfFragments));
+
+ CreateFragmentationConf * const conf =
+ (CreateFragmentationConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ conf->noOfReplicas = noOfReplicas;
+ conf->noOfFragments = noOfFragments;
+
+ fragments[0] = noOfReplicas;
+ fragments[1] = noOfFragments;
+
+ LinearSectionPtr ptr[3];
+ ptr[0].p = &fragments[0];
+ ptr[0].sz = count;
+ sendSignal(senderRef,
+ GSN_CREATE_FRAGMENTATION_CONF,
+ signal,
+ CreateFragmentationConf::SignalLength,
+ JBB,
+ ptr,
+ 1);
+ return;
+ } while(false);
+
+ CreateFragmentationRef * const ref =
+ (CreateFragmentationRef*)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = senderData;
+ ref->errorCode = err;
+ sendSignal(senderRef, GSN_CREATE_FRAGMENTATION_REF, signal,
+ CreateFragmentationRef::SignalLength, JBB);
+}
+
+void Dbdih::execDIADDTABREQ(Signal* signal)
+{
+ jamEntry();
+
+ DiAddTabReq * const req = (DiAddTabReq*)signal->getDataPtr();
+
+ // Seize connect record
+ ndbrequire(cfirstconnect != RNIL);
+ ConnectRecordPtr connectPtr;
+ connectPtr.i = cfirstconnect;
+ ptrCheckGuard(connectPtr, cconnectFileSize, connectRecord);
+ cfirstconnect = connectPtr.p->nfConnect;
+
+ const Uint32 userPtr = req->connectPtr;
+ const BlockReference userRef = signal->getSendersBlockRef();
+ connectPtr.p->nfConnect = RNIL;
+ connectPtr.p->userpointer = userPtr;
+ connectPtr.p->userblockref = userRef;
+ connectPtr.p->connectState = ConnectRecord::INUSE;
+ connectPtr.p->table = req->tableId;
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ tabPtr.p->connectrec = connectPtr.i;
+ tabPtr.p->tableType = req->tableType;
+ tabPtr.p->schemaVersion = req->schemaVersion;
+ tabPtr.p->primaryTableId = req->primaryTableId;
+
+ if(tabPtr.p->tabStatus == TabRecord::TS_ACTIVE){
+ jam();
+ tabPtr.p->tabStatus = TabRecord::TS_CREATING;
+ sendAddFragreq(signal, connectPtr, tabPtr, 0);
+ return;
+ }
+
+ if(getNodeState().getSystemRestartInProgress() &&
+ tabPtr.p->tabStatus == TabRecord::TS_IDLE){
+ jam();
+
+ ndbrequire(cmasterNodeId == getOwnNodeId());
+ tabPtr.p->tabStatus = TabRecord::TS_CREATING;
+
+ initTableFile(tabPtr);
+ FileRecordPtr filePtr;
+ filePtr.i = tabPtr.p->tabFile[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ openFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::OPENING_TABLE;
+ return;
+ }
+
+ /*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
+ /* AT THE TIME OF INITIATING THE FILE OF TABLE */
+ /* DESCRIPTION IS CREATED FOR APPROPRIATE SIZE. EACH */
+ /* EACH RECORD IN THIS FILE HAS THE INFORMATION ABOUT */
+ /* ONE TABLE. THE POINTER TO THIS RECORD IS THE TABLE */
+ /* REFERENCE. IN THE BEGINNING ALL RECORDS ARE CREATED */
+ /* BUT THEY DO NOT HAVE ANY INFORMATION ABOUT ANY TABLE*/
+ /*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
+ tabPtr.p->tabStatus = TabRecord::TS_CREATING;
+ tabPtr.p->storedTable = req->storedTable;
+ tabPtr.p->method = TabRecord::HASH;
+ tabPtr.p->kvalue = req->kValue;
+
+ Uint32 fragments[2 + 8*MAX_REPLICAS*MAX_NDB_NODES];
+ SegmentedSectionPtr fragDataPtr;
+ signal->getSection(fragDataPtr, DiAddTabReq::FRAGMENTATION);
+ copy(fragments, fragDataPtr);
+ releaseSections(signal);
+
+ const Uint32 noReplicas = fragments[0];
+ const Uint32 noFragments = fragments[1];
+
+ tabPtr.p->noOfBackups = noReplicas - 1;
+ tabPtr.p->totalfragments = noFragments;
+ ndbrequire(noReplicas == cnoReplicas); // Only allowed
+
+ if ((noReplicas * noFragments) > cnoFreeReplicaRec) {
+ jam();
+ addtabrefuseLab(signal, connectPtr, ZREPLERROR1);
+ return;
+ }//if
+ if (noFragments > cremainingfrags) {
+ jam();
+ addtabrefuseLab(signal, connectPtr, ZREPLERROR1);
+ return;
+ }//if
+
+ Uint32 logTotalFragments = 1;
+ while (logTotalFragments <= tabPtr.p->totalfragments) {
+ jam();
+ logTotalFragments <<= 1;
+ }
+ logTotalFragments >>= 1;
+ tabPtr.p->mask = logTotalFragments - 1;
+ tabPtr.p->hashpointer = tabPtr.p->totalfragments - logTotalFragments;
+ allocFragments(tabPtr.p->totalfragments, tabPtr);
+
+ Uint32 index = 2;
+ for (Uint32 fragId = 0; fragId < noFragments; fragId++) {
+ jam();
+ FragmentstorePtr fragPtr;
+ Uint32 activeIndex = 0;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ fragPtr.p->preferredPrimary = fragments[index];
+ for (Uint32 i = 0; i<noReplicas; i++) {
+ const Uint32 nodeId = fragments[index++];
+ ReplicaRecordPtr replicaPtr;
+ allocStoredReplica(fragPtr, replicaPtr, nodeId);
+ if (getNodeStatus(nodeId) == NodeRecord::ALIVE) {
+ jam();
+ ndbrequire(activeIndex < MAX_REPLICAS);
+ fragPtr.p->activeNodes[activeIndex] = nodeId;
+ activeIndex++;
+ } else {
+ jam();
+ removeStoredReplica(fragPtr, replicaPtr);
+ linkOldStoredReplica(fragPtr, replicaPtr);
+ }//if
+ }//for
+ fragPtr.p->fragReplicas = activeIndex;
+ ndbrequire(activeIndex > 0 && fragPtr.p->storedReplicas != RNIL);
+ }
+ initTableFile(tabPtr);
+ tabPtr.p->tabCopyStatus = TabRecord::CS_ADD_TABLE_MASTER;
+ signal->theData[0] = DihContinueB::ZPACK_TABLE_INTO_PAGES;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+}
+
+void
+Dbdih::addTable_closeConf(Signal * signal, Uint32 tabPtrI){
+ TabRecordPtr tabPtr;
+ tabPtr.i = tabPtrI;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ ConnectRecordPtr connectPtr;
+ connectPtr.i = tabPtr.p->connectrec;
+ ptrCheckGuard(connectPtr, cconnectFileSize, connectRecord);
+
+ sendAddFragreq(signal, connectPtr, tabPtr, 0);
+}
+
+void
+Dbdih::sendAddFragreq(Signal* signal, ConnectRecordPtr connectPtr,
+ TabRecordPtr tabPtr, Uint32 fragId){
+ jam();
+ const Uint32 fragCount = tabPtr.p->totalfragments;
+ ReplicaRecordPtr replicaPtr; replicaPtr.i = RNIL;
+ for(; fragId<fragCount; fragId++){
+ jam();
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+
+ replicaPtr.i = fragPtr.p->storedReplicas;
+ while(replicaPtr.i != RNIL){
+ jam();
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ if(replicaPtr.p->procNode == getOwnNodeId()){
+ break;
+ }
+ replicaPtr.i = replicaPtr.p->nextReplica;
+ }
+
+ if(replicaPtr.i != RNIL){
+ jam();
+ break;
+ }
+
+ replicaPtr.i = fragPtr.p->oldStoredReplicas;
+ while(replicaPtr.i != RNIL){
+ jam();
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ if(replicaPtr.p->procNode == getOwnNodeId()){
+ break;
+ }
+ replicaPtr.i = replicaPtr.p->nextReplica;
+ }
+
+ if(replicaPtr.i != RNIL){
+ jam();
+ break;
+ }
+ }
+
+ if(replicaPtr.i != RNIL){
+ jam();
+ ndbrequire(fragId < fragCount);
+ ndbrequire(replicaPtr.p->procNode == getOwnNodeId());
+
+ Uint32 requestInfo = 0;
+ if(!tabPtr.p->storedTable){
+ requestInfo |= LqhFragReq::TemporaryTable;
+ }
+
+ if(getNodeState().getNodeRestartInProgress()){
+ requestInfo |= LqhFragReq::CreateInRunning;
+ }
+
+ AddFragReq* const req = (AddFragReq*)signal->getDataPtr();
+ req->dihPtr = connectPtr.i;
+ req->senderData = connectPtr.p->userpointer;
+ req->fragmentId = fragId;
+ req->requestInfo = requestInfo;
+ req->tableId = tabPtr.i;
+ req->nextLCP = 0;
+ req->nodeId = getOwnNodeId();
+ req->totalFragments = fragCount;
+ req->startGci = SYSFILE->newestRestorableGCI;
+ sendSignal(DBDICT_REF, GSN_ADD_FRAGREQ, signal,
+ AddFragReq::SignalLength, JBB);
+ return;
+ }
+
+ // Done
+ DiAddTabConf * const conf = (DiAddTabConf*)signal->getDataPtr();
+ conf->senderData = connectPtr.p->userpointer;
+ sendSignal(connectPtr.p->userblockref, GSN_DIADDTABCONF, signal,
+ DiAddTabConf::SignalLength, JBB);
+
+ // Release
+ connectPtr.p->userblockref = ZNIL;
+ connectPtr.p->userpointer = RNIL;
+ connectPtr.p->connectState = ConnectRecord::FREE;
+ connectPtr.p->nfConnect = cfirstconnect;
+ cfirstconnect = connectPtr.i;
+}
+
+void
+Dbdih::execADD_FRAGCONF(Signal* signal){
+ jamEntry();
+ AddFragConf * const conf = (AddFragConf*)signal->getDataPtr();
+
+ ConnectRecordPtr connectPtr;
+ connectPtr.i = conf->dihPtr;
+ ptrCheckGuard(connectPtr, cconnectFileSize, connectRecord);
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = connectPtr.p->table;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ sendAddFragreq(signal, connectPtr, tabPtr, conf->fragId + 1);
+}
+
+void
+Dbdih::execADD_FRAGREF(Signal* signal){
+ jamEntry();
+ AddFragRef * const ref = (AddFragRef*)signal->getDataPtr();
+
+ ConnectRecordPtr connectPtr;
+ connectPtr.i = ref->dihPtr;
+ ptrCheckGuard(connectPtr, cconnectFileSize, connectRecord);
+
+ {
+ DiAddTabRef * const ref = (DiAddTabRef*)signal->getDataPtr();
+ ref->senderData = connectPtr.p->userpointer;
+ ref->errorCode = ~0;
+ sendSignal(connectPtr.p->userblockref, GSN_DIADDTABREF, signal,
+ DiAddTabRef::SignalLength, JBB);
+ }
+
+ // Release
+ connectPtr.p->userblockref = ZNIL;
+ connectPtr.p->userpointer = RNIL;
+ connectPtr.p->connectState = ConnectRecord::FREE;
+ connectPtr.p->nfConnect = cfirstconnect;
+ cfirstconnect = connectPtr.i;
+}
+
+/*
+ 3.7.1.3 R E F U S E
+ *********************
+ */
+void Dbdih::addtabrefuseLab(Signal* signal, ConnectRecordPtr connectPtr, Uint32 errorCode)
+{
+ connectPtr.p->connectState = ConnectRecord::INUSE;
+ signal->theData[0] = connectPtr.p->userpointer;
+ signal->theData[1] = errorCode;
+ sendSignal(connectPtr.p->userblockref, GSN_DIADDTABREF, signal, 2, JBB);
+ return;
+}//Dbdih::addtabrefuseLab()
+
+/*
+ 3.7.2 A D D T A B L E D U P L I C A T I O N
+ *************************************************
+ */
+/*
+ 3.7.2.1 A D D T A B L E D U P L I C A T I O N R E Q U E S T
+ *******************************************************************=
+ */
+
+/*
+ D E L E T E T A B L E
+ **********************=
+ */
+/*****************************************************************************/
+/*********** DELETE TABLE MODULE *************/
+/*****************************************************************************/
+void
+Dbdih::execDROP_TAB_REQ(Signal* signal){
+ jamEntry();
+ DropTabReq* req = (DropTabReq*)signal->getDataPtr();
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ tabPtr.p->m_dropTab.tabUserRef = req->senderRef;
+ tabPtr.p->m_dropTab.tabUserPtr = req->senderData;
+
+ DropTabReq::RequestType rt = (DropTabReq::RequestType)req->requestType;
+
+ switch(rt){
+ case DropTabReq::OnlineDropTab:
+ jam();
+ ndbrequire(tabPtr.p->tabStatus == TabRecord::TS_DROPPING);
+ releaseTable(tabPtr);
+ break;
+ case DropTabReq::CreateTabDrop:
+ jam();
+ releaseTable(tabPtr);
+ break;
+ case DropTabReq::RestartDropTab:
+ break;
+ }
+
+ startDeleteFile(signal, tabPtr);
+}
+
+void Dbdih::startDeleteFile(Signal* signal, TabRecordPtr tabPtr)
+{
+ if (tabPtr.p->tabFile[0] == RNIL) {
+ jam();
+ initTableFile(tabPtr);
+ }//if
+ openTableFileForDelete(signal, tabPtr.p->tabFile[0]);
+}//Dbdih::startDeleteFile()
+
+void Dbdih::openTableFileForDelete(Signal* signal, Uint32 fileIndex)
+{
+ FileRecordPtr filePtr;
+ filePtr.i = fileIndex;
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ openFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::TABLE_OPEN_FOR_DELETE;
+}//Dbdih::openTableFileForDelete()
+
+void Dbdih::tableOpenLab(Signal* signal, FileRecordPtr filePtr)
+{
+ closeFileDelete(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::TABLE_CLOSE_DELETE;
+ return;
+}//Dbdih::tableOpenLab()
+
+void Dbdih::tableDeleteLab(Signal* signal, FileRecordPtr filePtr)
+{
+ TabRecordPtr tabPtr;
+ tabPtr.i = filePtr.p->tabRef;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ if (filePtr.i == tabPtr.p->tabFile[0]) {
+ jam();
+ openTableFileForDelete(signal, tabPtr.p->tabFile[1]);
+ return;
+ }//if
+ ndbrequire(filePtr.i == tabPtr.p->tabFile[1]);
+
+ releaseFile(tabPtr.p->tabFile[0]);
+ releaseFile(tabPtr.p->tabFile[1]);
+ tabPtr.p->tabFile[0] = tabPtr.p->tabFile[1] = RNIL;
+
+ tabPtr.p->tabStatus = TabRecord::TS_IDLE;
+
+ DropTabConf * const dropConf = (DropTabConf *)signal->getDataPtrSend();
+ dropConf->senderRef = reference();
+ dropConf->senderData = tabPtr.p->m_dropTab.tabUserPtr;
+ dropConf->tableId = tabPtr.i;
+ sendSignal(tabPtr.p->m_dropTab.tabUserRef, GSN_DROP_TAB_CONF,
+ signal, DropTabConf::SignalLength, JBB);
+
+ tabPtr.p->m_dropTab.tabUserPtr = RNIL;
+ tabPtr.p->m_dropTab.tabUserRef = 0;
+}//Dbdih::tableDeleteLab()
+
+
+void Dbdih::releaseTable(TabRecordPtr tabPtr)
+{
+ FragmentstorePtr fragPtr;
+ for (Uint32 fragId = 0; fragId < tabPtr.p->totalfragments; fragId++) {
+ jam();
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ releaseReplicas(fragPtr.p->storedReplicas);
+ releaseReplicas(fragPtr.p->oldStoredReplicas);
+ }//for
+ releaseFragments(tabPtr);
+ if (tabPtr.p->tabFile[0] != RNIL) {
+ jam();
+ releaseFile(tabPtr.p->tabFile[0]);
+ releaseFile(tabPtr.p->tabFile[1]);
+ tabPtr.p->tabFile[0] = tabPtr.p->tabFile[1] = RNIL;
+ }//if
+}//Dbdih::releaseTable()
+
+void Dbdih::releaseReplicas(Uint32 replicaPtrI)
+{
+ ReplicaRecordPtr replicaPtr;
+ replicaPtr.i = replicaPtrI;
+ jam();
+ while (replicaPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ Uint32 tmp = replicaPtr.p->nextReplica;
+ replicaPtr.p->nextReplica = cfirstfreeReplica;
+ cfirstfreeReplica = replicaPtr.i;
+ replicaPtr.i = tmp;
+ cnoFreeReplicaRec++;
+ }//while
+}//Dbdih::releaseReplicas()
+
+void Dbdih::seizeReplicaRec(ReplicaRecordPtr& replicaPtr)
+{
+ replicaPtr.i = cfirstfreeReplica;
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ cfirstfreeReplica = replicaPtr.p->nextReplica;
+ cnoFreeReplicaRec--;
+ replicaPtr.p->nextReplica = RNIL;
+}//Dbdih::seizeReplicaRec()
+
+void Dbdih::releaseFile(Uint32 fileIndex)
+{
+ FileRecordPtr filePtr;
+ filePtr.i = fileIndex;
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ filePtr.p->nextFile = cfirstfreeFile;
+ cfirstfreeFile = filePtr.i;
+}//Dbdih::releaseFile()
+
+
+void Dbdih::execALTER_TAB_REQ(Signal * signal)
+{
+ AlterTabReq* const req = (AlterTabReq*)signal->getDataPtr();
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 senderData = req->senderData;
+ const Uint32 changeMask = req->changeMask;
+ const Uint32 tableId = req->tableId;
+ const Uint32 tableVersion = req->tableVersion;
+ const Uint32 gci = req->gci;
+ AlterTabReq::RequestType requestType =
+ (AlterTabReq::RequestType) req->requestType;
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ tabPtr.p->schemaVersion = tableVersion;
+
+ // Request handled successfully
+ AlterTabConf * conf = (AlterTabConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ conf->changeMask = changeMask;
+ conf->tableId = tableId;
+ conf->tableVersion = tableVersion;
+ conf->gci = gci;
+ conf->requestType = requestType;
+ sendSignal(senderRef, GSN_ALTER_TAB_CONF, signal,
+ AlterTabConf::SignalLength, JBB);
+}
+
+/*
+ G E T N O D E S
+ **********************=
+ */
+/*****************************************************************************/
+/* ********** TRANSACTION HANDLING MODULE *************/
+/*****************************************************************************/
+/*
+ 3.8.1 G E T N O D E S R E Q U E S T
+ ******************************************
+ Asks what nodes should be part of a transaction.
+*/
+void Dbdih::execDIGETNODESREQ(Signal* signal)
+{
+ const DiGetNodesReq * const req = (DiGetNodesReq *)&signal->theData[0];
+ FragmentstorePtr fragPtr;
+ TabRecordPtr tabPtr;
+ tabPtr.i = req->tableId;
+ Uint32 hashValue = req->hashValue;
+ Uint32 ttabFileSize = ctabFileSize;
+ TabRecord* regTabDesc = tabRecord;
+ jamEntry();
+ ptrCheckGuard(tabPtr, ttabFileSize, regTabDesc);
+ hashValue = hashValue >> tabPtr.p->kvalue;
+ Uint32 fragId = tabPtr.p->mask & hashValue;
+ ndbrequire(tabPtr.p->tabStatus == TabRecord::TS_ACTIVE);
+ if (fragId < tabPtr.p->hashpointer) {
+ jam();
+ fragId = hashValue & ((tabPtr.p->mask << 1) + 1);
+ }//if
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ DiGetNodesConf * const conf = (DiGetNodesConf *)&signal->theData[0];
+ Uint32 nodeCount = extractNodeInfo(fragPtr.p, conf->nodes);
+ Uint32 sig2 = (nodeCount - 1) +
+ (fragPtr.p->distributionKey << 16);
+ conf->zero = 0;
+ conf->reqinfo = sig2;
+ conf->fragId = fragId;
+}//Dbdih::execDIGETNODESREQ()
+
+Uint32 Dbdih::extractNodeInfo(const Fragmentstore * fragPtr, Uint32 nodes[])
+{
+ Uint32 nodeCount = 0;
+ for (Uint32 i = 0; i < fragPtr->fragReplicas; i++) {
+ jam();
+ NodeRecordPtr nodePtr;
+ ndbrequire(i < MAX_REPLICAS);
+ nodePtr.i = fragPtr->activeNodes[i];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.p->useInTransactions) {
+ jam();
+ nodes[nodeCount] = nodePtr.i;
+ nodeCount++;
+ }//if
+ }//for
+ ndbrequire(nodeCount > 0);
+ return nodeCount;
+}//Dbdih::extractNodeInfo()
+
+#define NO_OF_FRAGS_PER_CHUNK 16
+#define LOG_NO_OF_FRAGS_PER_CHUNK 4
+
+void
+Dbdih::getFragstore(TabRecord * tab, //In parameter
+ Uint32 fragNo, //In parameter
+ FragmentstorePtr & fragptr) //Out parameter
+{
+ FragmentstorePtr fragPtr;
+ Uint32 chunkNo = fragNo >> LOG_NO_OF_FRAGS_PER_CHUNK;
+ Uint32 chunkIndex = fragNo & (NO_OF_FRAGS_PER_CHUNK - 1);
+ Uint32 TfragstoreFileSize = cfragstoreFileSize;
+ Fragmentstore* TfragStore = fragmentstore;
+ if (chunkNo < MAX_NDB_NODES) {
+ fragPtr.i = tab->startFid[chunkNo] + chunkIndex;
+ ptrCheckGuard(fragPtr, TfragstoreFileSize, TfragStore);
+ fragptr = fragPtr;
+ return;
+ }//if
+ ndbrequire(false);
+}//Dbdih::getFragstore()
+
+void Dbdih::allocFragments(Uint32 noOfFragments, TabRecordPtr tabPtr)
+{
+ FragmentstorePtr fragPtr;
+ Uint32 noOfChunks = (noOfFragments + (NO_OF_FRAGS_PER_CHUNK - 1)) >> LOG_NO_OF_FRAGS_PER_CHUNK;
+ ndbrequire(cremainingfrags >= noOfFragments);
+ for (Uint32 i = 0; i < noOfChunks; i++) {
+ jam();
+ Uint32 baseFrag = cfirstfragstore;
+ tabPtr.p->startFid[i] = baseFrag;
+ fragPtr.i = baseFrag;
+ ptrCheckGuard(fragPtr, cfragstoreFileSize, fragmentstore);
+ cfirstfragstore = fragPtr.p->nextFragmentChunk;
+ cremainingfrags -= NO_OF_FRAGS_PER_CHUNK;
+ for (Uint32 j = 0; j < NO_OF_FRAGS_PER_CHUNK; j++) {
+ jam();
+ fragPtr.i = baseFrag + j;
+ ptrCheckGuard(fragPtr, cfragstoreFileSize, fragmentstore);
+ initFragstore(fragPtr);
+ }//if
+ }//for
+ tabPtr.p->noOfFragChunks = noOfChunks;
+}//Dbdih::allocFragments()
+
+void Dbdih::releaseFragments(TabRecordPtr tabPtr)
+{
+ FragmentstorePtr fragPtr;
+ for (Uint32 i = 0; i < tabPtr.p->noOfFragChunks; i++) {
+ jam();
+ Uint32 baseFrag = tabPtr.p->startFid[i];
+ fragPtr.i = baseFrag;
+ ptrCheckGuard(fragPtr, cfragstoreFileSize, fragmentstore);
+ fragPtr.p->nextFragmentChunk = cfirstfragstore;
+ cfirstfragstore = baseFrag;
+ tabPtr.p->startFid[i] = RNIL;
+ cremainingfrags += NO_OF_FRAGS_PER_CHUNK;
+ }//for
+ tabPtr.p->noOfFragChunks = 0;
+}//Dbdih::releaseFragments()
+
+void Dbdih::initialiseFragstore()
+{
+ FragmentstorePtr fragPtr;
+ for (Uint32 i = 0; i < cfragstoreFileSize; i++) {
+ fragPtr.i = i;
+ ptrCheckGuard(fragPtr, cfragstoreFileSize, fragmentstore);
+ initFragstore(fragPtr);
+ }//for
+ Uint32 noOfChunks = cfragstoreFileSize >> LOG_NO_OF_FRAGS_PER_CHUNK;
+ fragPtr.i = 0;
+ cfirstfragstore = RNIL;
+ cremainingfrags = 0;
+ for (Uint32 i = 0; i < noOfChunks; i++) {
+ ptrCheckGuard(fragPtr, cfragstoreFileSize, fragmentstore);
+ fragPtr.p->nextFragmentChunk = cfirstfragstore;
+ cfirstfragstore = fragPtr.i;
+ fragPtr.i += NO_OF_FRAGS_PER_CHUNK;
+ cremainingfrags += NO_OF_FRAGS_PER_CHUNK;
+ }//for
+}//Dbdih::initialiseFragstore()
+
+/*
+ 3.9 V E R I F I C A T I O N
+ ****************************=
+ */
+/****************************************************************************/
+/* ********** VERIFICATION SUB-MODULE *************/
+/****************************************************************************/
+/*
+ 3.9.1 R E C E I V I N G O F V E R I F I C A T I O N R E Q U E S T
+ *************************************************************************
+ */
+void Dbdih::execDIVERIFYREQ(Signal* signal)
+{
+
+ jamEntry();
+ if ((getBlockCommit() == false) &&
+ (cfirstVerifyQueue == RNIL)) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // We are not blocked and the verify queue was empty currently so we can
+ // simply reply back to TC immediately. The method was called with
+ // EXECUTE_DIRECT so we reply back by setting signal data and returning.
+ // theData[0] already contains the correct information so
+ // we need not touch it.
+ /*-----------------------------------------------------------------------*/
+ signal->theData[1] = currentgcp;
+ signal->theData[2] = 0;
+ return;
+ }//if
+ /*-------------------------------------------------------------------------*/
+ // Since we are blocked we need to put this operation last in the verify
+ // queue to ensure that operation starts up in the correct order.
+ /*-------------------------------------------------------------------------*/
+ ApiConnectRecordPtr tmpApiConnectptr;
+ ApiConnectRecordPtr localApiConnectptr;
+
+ cverifyQueueCounter++;
+ localApiConnectptr.i = signal->theData[0];
+ tmpApiConnectptr.i = clastVerifyQueue;
+ ptrCheckGuard(localApiConnectptr, capiConnectFileSize, apiConnectRecord);
+ localApiConnectptr.p->apiGci = cnewgcp;
+ localApiConnectptr.p->nextApi = RNIL;
+ clastVerifyQueue = localApiConnectptr.i;
+ if (tmpApiConnectptr.i == RNIL) {
+ jam();
+ cfirstVerifyQueue = localApiConnectptr.i;
+ } else {
+ jam();
+ ptrCheckGuard(tmpApiConnectptr, capiConnectFileSize, apiConnectRecord);
+ tmpApiConnectptr.p->nextApi = localApiConnectptr.i;
+ }//if
+ emptyverificbuffer(signal, false);
+ signal->theData[2] = 1; // Indicate no immediate return
+ return;
+}//Dbdih::execDIVERIFYREQ()
+
+void Dbdih::execDI_FCOUNTREQ(Signal* signal)
+{
+ ConnectRecordPtr connectPtr;
+ TabRecordPtr tabPtr;
+ jamEntry();
+ connectPtr.i = signal->theData[0];
+ tabPtr.i = signal->theData[1];
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ ndbrequire(tabPtr.p->tabStatus == TabRecord::TS_ACTIVE);
+
+ if(connectPtr.i != RNIL){
+ ptrCheckGuard(connectPtr, cconnectFileSize, connectRecord);
+ if (connectPtr.p->connectState == ConnectRecord::INUSE) {
+ jam();
+ signal->theData[0] = connectPtr.p->userpointer;
+ signal->theData[1] = tabPtr.p->totalfragments;
+ sendSignal(connectPtr.p->userblockref, GSN_DI_FCOUNTCONF, signal,2, JBB);
+ return;
+ }//if
+ signal->theData[0] = connectPtr.p->userpointer;
+ signal->theData[1] = ZERRONOUSSTATE;
+ sendSignal(connectPtr.p->userblockref, GSN_DI_FCOUNTREF, signal, 2, JBB);
+ return;
+ }//if
+
+ //connectPtr.i == RNIL -> question without connect record
+ const Uint32 senderData = signal->theData[2];
+ const BlockReference senderRef = signal->senderBlockRef();
+ signal->theData[0] = RNIL;
+ signal->theData[1] = tabPtr.p->totalfragments;
+ signal->theData[2] = tabPtr.i;
+ signal->theData[3] = senderData;
+ signal->theData[4] = tabPtr.p->noOfBackups;
+ sendSignal(senderRef, GSN_DI_FCOUNTCONF, signal, 5, JBB);
+}//Dbdih::execDI_FCOUNTREQ()
+
+void Dbdih::execDIGETPRIMREQ(Signal* signal)
+{
+ FragmentstorePtr fragPtr;
+ ConnectRecordPtr connectPtr;
+ TabRecordPtr tabPtr;
+ jamEntry();
+ Uint32 passThrough = signal->theData[1];
+ tabPtr.i = signal->theData[2];
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ if (DictTabInfo::isOrderedIndex(tabPtr.p->tableType)) {
+ jam();
+ tabPtr.i = tabPtr.p->primaryTableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ }
+ Uint32 fragId = signal->theData[3];
+
+ ndbrequire(tabPtr.p->tabStatus == TabRecord::TS_ACTIVE);
+ connectPtr.i = signal->theData[0];
+ if(connectPtr.i != RNIL){
+ jam();
+ ptrCheckGuard(connectPtr, cconnectFileSize, connectRecord);
+ ndbrequire(connectPtr.p->connectState == ConnectRecord::INUSE);
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ connectPtr.p->nodeCount = extractNodeInfo(fragPtr.p, connectPtr.p->nodes);
+ signal->theData[0] = connectPtr.p->userpointer;
+ signal->theData[1] = passThrough;
+ signal->theData[2] = connectPtr.p->nodes[0];
+ sendSignal(connectPtr.p->userblockref, GSN_DIGETPRIMCONF, signal, 3, JBB);
+ return;
+ }//if
+ //connectPtr.i == RNIL -> question without connect record
+ Uint32 nodes[MAX_REPLICAS];
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ Uint32 count = extractNodeInfo(fragPtr.p, nodes);
+
+ signal->theData[0] = RNIL;
+ signal->theData[1] = passThrough;
+ signal->theData[2] = nodes[0];
+ signal->theData[3] = nodes[1];
+ signal->theData[4] = nodes[2];
+ signal->theData[5] = nodes[3];
+ signal->theData[6] = count;
+ signal->theData[7] = tabPtr.i;
+ signal->theData[8] = fragId;
+
+ const BlockReference senderRef = signal->senderBlockRef();
+ sendSignal(senderRef, GSN_DIGETPRIMCONF, signal, 9, JBB);
+}//Dbdih::execDIGETPRIMREQ()
+
+/****************************************************************************/
+/* ********** GLOBAL-CHECK-POINT HANDLING MODULE *************/
+/****************************************************************************/
+/*
+ 3.10 G L O B A L C H E C K P O I N T ( IN M A S T E R R O L E)
+ *******************************************************************
+ */
+void Dbdih::checkGcpStopLab(Signal* signal)
+{
+ Uint32 tgcpStatus;
+
+ tgcpStatus = cgcpStatus;
+ if (tgcpStatus == coldGcpStatus) {
+ jam();
+ if (coldGcpId == cnewgcp) {
+ jam();
+ if (cgcpStatus != GCP_READY) {
+ jam();
+ cgcpSameCounter++;
+ if (cgcpSameCounter == 1200) {
+ jam();
+#ifdef VM_TRACE
+ ndbout << "System crash due to GCP Stop in state = ";
+ ndbout << cgcpStatus << endl;
+#endif
+ crashSystemAtGcpStop(signal);
+ return;
+ }//if
+ } else {
+ jam();
+ if (cgcpOrderBlocked == 0) {
+ jam();
+ cgcpSameCounter++;
+ if (cgcpSameCounter == 1200) {
+ jam();
+#ifdef VM_TRACE
+ ndbout << "System crash due to GCP Stop in state = ";
+ ndbout << cgcpStatus << endl;
+#endif
+ crashSystemAtGcpStop(signal);
+ return;
+ }//if
+ } else {
+ jam();
+ cgcpSameCounter = 0;
+ }//if
+ }//if
+ } else {
+ jam();
+ cgcpSameCounter = 0;
+ }//if
+ } else {
+ jam();
+ cgcpSameCounter = 0;
+ }//if
+ signal->theData[0] = DihContinueB::ZCHECK_GCP_STOP;
+ signal->theData[1] = coldGcpStatus;
+ signal->theData[2] = cgcpStatus;
+ signal->theData[3] = coldGcpId;
+ signal->theData[4] = cnewgcp;
+ signal->theData[5] = cgcpSameCounter;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 6);
+ coldGcpStatus = cgcpStatus;
+ coldGcpId = cnewgcp;
+ return;
+}//Dbdih::checkGcpStopLab()
+
+void Dbdih::startGcpLab(Signal* signal, Uint32 aWaitTime)
+{
+ if ((cgcpOrderBlocked == 1) ||
+ (c_nodeStartMaster.blockGcp == true) ||
+ (cfirstVerifyQueue != RNIL)) {
+ /*************************************************************************/
+ // 1: Global Checkpoint has been stopped by management command
+ // 2: Global Checkpoint is blocked by node recovery activity
+ // 3: Previous global checkpoint is not yet completed.
+ // All this means that global checkpoint cannot start now.
+ /*************************************************************************/
+ jam();
+ cgcpStartCounter++;
+ signal->theData[0] = DihContinueB::ZSTART_GCP;
+ signal->theData[1] = aWaitTime > 100 ? (aWaitTime - 100) : 0;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 2);
+ return;
+ }//if
+ if (cstartGcpNow == false && aWaitTime > 100){
+ /*************************************************************************/
+ // We still have more than 100 milliseconds before we start the next and
+ // nobody has ordered immediate start of a global checkpoint.
+ // During initial start we will use continuos global checkpoints to
+ // speed it up since we need to complete a global checkpoint after
+ // inserting a lot of records.
+ /*************************************************************************/
+ jam();
+ cgcpStartCounter++;
+ signal->theData[0] = DihContinueB::ZSTART_GCP;
+ signal->theData[1] = (aWaitTime - 100);
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 2);
+ return;
+ }//if
+ cgcpStartCounter = 0;
+ cstartGcpNow = false;
+ /***************************************************************************/
+ // Report the event that a global checkpoint has started.
+ /***************************************************************************/
+ signal->theData[0] = EventReport::GlobalCheckpointStarted; //Event type
+ signal->theData[1] = cnewgcp;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ CRASH_INSERTION(7000);
+ cnewgcp++;
+ signal->setTrace(TestOrd::TraceGlobalCheckpoint);
+ sendLoopMacro(GCP_PREPARE, sendGCP_PREPARE);
+ cgcpStatus = GCP_PREPARE_SENT;
+}//Dbdih::startGcpLab()
+
+void Dbdih::execGCP_PREPARECONF(Signal* signal)
+{
+ jamEntry();
+ Uint32 senderNodeId = signal->theData[0];
+ Uint32 gci = signal->theData[1];
+ ndbrequire(gci == cnewgcp);
+ receiveLoopMacro(GCP_PREPARE, senderNodeId);
+ //-------------------------------------------------------------
+ // We have now received all replies. We are ready to continue
+ // with committing the global checkpoint.
+ //-------------------------------------------------------------
+ gcpcommitreqLab(signal);
+}//Dbdih::execGCP_PREPARECONF()
+
+void Dbdih::gcpcommitreqLab(Signal* signal)
+{
+ CRASH_INSERTION(7001);
+ sendLoopMacro(GCP_COMMIT, sendGCP_COMMIT);
+ cgcpStatus = GCP_COMMIT_SENT;
+ return;
+}//Dbdih::gcpcommitreqLab()
+
+void Dbdih::execGCP_NODEFINISH(Signal* signal)
+{
+ jamEntry();
+ const Uint32 senderNodeId = signal->theData[0];
+ const Uint32 gci = signal->theData[1];
+ const Uint32 failureNr = signal->theData[2];
+ if (!isMaster()) {
+ jam();
+ ndbrequire(failureNr > cfailurenr);
+ //-------------------------------------------------------------
+ // Another node thinks we are master. This could happen when he
+ // has heard of a node failure which I have not heard of. Ignore
+ // signal in this case since we will discover it by sending
+ // MASTER_GCPREQ to the node.
+ //-------------------------------------------------------------
+ return;
+ } else if (cmasterState == MASTER_TAKE_OVER_GCP) {
+ jam();
+ //-------------------------------------------------------------
+ // We are currently taking over as master. We will delay the
+ // signal until we have completed the take over gcp handling.
+ //-------------------------------------------------------------
+ sendSignalWithDelay(reference(), GSN_GCP_NODEFINISH, signal, 20, 3);
+ return;
+ } else {
+ ndbrequire(cmasterState == MASTER_ACTIVE);
+ }//if
+ ndbrequire(gci == coldgcp);
+ receiveLoopMacro(GCP_COMMIT, senderNodeId);
+ //-------------------------------------------------------------
+ // We have now received all replies. We are ready to continue
+ // with saving the global checkpoint to disk.
+ //-------------------------------------------------------------
+ CRASH_INSERTION(7002);
+ gcpsavereqLab(signal);
+ return;
+}//Dbdih::execGCP_NODEFINISH()
+
+void Dbdih::gcpsavereqLab(Signal* signal)
+{
+ sendLoopMacro(GCP_SAVEREQ, sendGCP_SAVEREQ);
+ cgcpStatus = GCP_NODE_FINISHED;
+}//Dbdih::gcpsavereqLab()
+
+void Dbdih::execGCP_SAVECONF(Signal* signal)
+{
+ jamEntry();
+ const GCPSaveConf * const saveConf = (GCPSaveConf*)&signal->theData[0];
+ ndbrequire(saveConf->gci == coldgcp);
+ ndbrequire(saveConf->nodeId == saveConf->dihPtr);
+ SYSFILE->lastCompletedGCI[saveConf->nodeId] = saveConf->gci;
+ GCP_SAVEhandling(signal, saveConf->nodeId);
+}//Dbdih::execGCP_SAVECONF()
+
+void Dbdih::execGCP_SAVEREF(Signal* signal)
+{
+ jamEntry();
+ const GCPSaveRef * const saveRef = (GCPSaveRef*)&signal->theData[0];
+ ndbrequire(saveRef->gci == coldgcp);
+ ndbrequire(saveRef->nodeId == saveRef->dihPtr);
+ /**
+ * Only allow reason not to save
+ */
+ ndbrequire(saveRef->errorCode == GCPSaveRef::NodeShutdownInProgress ||
+ saveRef->errorCode == GCPSaveRef::FakedSignalDueToNodeFailure ||
+ saveRef->errorCode == GCPSaveRef::NodeRestartInProgress);
+ GCP_SAVEhandling(signal, saveRef->nodeId);
+}//Dbdih::execGCP_SAVEREF()
+
+void Dbdih::GCP_SAVEhandling(Signal* signal, Uint32 nodeId)
+{
+ receiveLoopMacro(GCP_SAVEREQ, nodeId);
+ /*-------------------------------------------------------------------------*/
+ // All nodes have replied. We are ready to update the system file.
+ /*-------------------------------------------------------------------------*/
+ cgcpStatus = GCP_SAVE_LQH_FINISHED;
+ CRASH_INSERTION(7003);
+ checkToCopy();
+ /**------------------------------------------------------------------------
+ * SET NEW RECOVERABLE GCI. ALSO RESET RESTART COUNTER TO ZERO.
+ * THIS INDICATES THAT THE SYSTEM HAS BEEN RECOVERED AND SURVIVED AT
+ * LEAST ONE GLOBAL CHECKPOINT PERIOD. WE WILL USE THIS PARAMETER TO
+ * SET BACK THE RESTART GCI IF WE ENCOUNTER MORE THAN ONE UNSUCCESSFUL
+ * RESTART.
+ *------------------------------------------------------------------------*/
+ SYSFILE->newestRestorableGCI = coldgcp;
+ if(Sysfile::getInitialStartOngoing(SYSFILE->systemRestartBits) &&
+ getNodeState().startLevel == NodeState::SL_STARTED){
+ jam();
+#if 0
+ ndbout_c("Dbdih: Clearing initial start ongoing");
+#endif
+ Sysfile::clearInitialStartOngoing(SYSFILE->systemRestartBits);
+ }
+ copyGciLab(signal, CopyGCIReq::GLOBAL_CHECKPOINT);
+}//Dbdih::GCP_SAVEhandling()
+
+/*
+ 3.11 G L O B A L C H E C K P O I N T (N O T - M A S T E R)
+ *************************************************************
+ */
+void Dbdih::execGCP_PREPARE(Signal* signal)
+{
+ jamEntry();
+ CRASH_INSERTION(7005);
+ Uint32 masterNodeId = signal->theData[0];
+ Uint32 gci = signal->theData[1];
+ BlockReference retRef = calcDihBlockRef(masterNodeId);
+
+ ndbrequire (cmasterdihref == retRef);
+ ndbrequire (cgcpParticipantState == GCP_PARTICIPANT_READY);
+ ndbrequire (gci == (currentgcp + 1));
+
+ cgckptflag = true;
+ cgcpParticipantState = GCP_PARTICIPANT_PREPARE_RECEIVED;
+ cnewgcp = gci;
+
+ signal->theData[0] = cownNodeId;
+ signal->theData[1] = gci;
+ sendSignal(retRef, GSN_GCP_PREPARECONF, signal, 2, JBA);
+ return;
+}//Dbdih::execGCP_PREPARE()
+
+void Dbdih::execGCP_COMMIT(Signal* signal)
+{
+ jamEntry();
+ CRASH_INSERTION(7006);
+ Uint32 masterNodeId = signal->theData[0];
+ Uint32 gci = signal->theData[1];
+
+ ndbrequire(gci == (currentgcp + 1));
+ ndbrequire(masterNodeId = cmasterNodeId);
+ ndbrequire(cgcpParticipantState == GCP_PARTICIPANT_PREPARE_RECEIVED);
+
+ coldgcp = currentgcp;
+ currentgcp = cnewgcp;
+ cgckptflag = false;
+ emptyverificbuffer(signal, true);
+ cgcpParticipantState = GCP_PARTICIPANT_COMMIT_RECEIVED;
+ signal->theData[1] = coldgcp;
+ sendSignal(clocaltcblockref, GSN_GCP_NOMORETRANS, signal, 2, JBB);
+ return;
+}//Dbdih::execGCP_COMMIT()
+
+void Dbdih::execGCP_TCFINISHED(Signal* signal)
+{
+ jamEntry();
+ CRASH_INSERTION(7007);
+ Uint32 gci = signal->theData[1];
+ ndbrequire(gci == coldgcp);
+
+ cgcpParticipantState = GCP_PARTICIPANT_TC_FINISHED;
+ signal->theData[0] = cownNodeId;
+ signal->theData[1] = coldgcp;
+ signal->theData[2] = cfailurenr;
+ sendSignal(cmasterdihref, GSN_GCP_NODEFINISH, signal, 3, JBB);
+}//Dbdih::execGCP_TCFINISHED()
+
+/*****************************************************************************/
+//****** RECEIVING TAMPER REQUEST FROM NDBAPI ******
+/*****************************************************************************/
+void Dbdih::execDIHNDBTAMPER(Signal* signal)
+{
+ jamEntry();
+ Uint32 tcgcpblocked = signal->theData[0];
+ /* ACTION TO BE TAKEN BY DIH */
+ Uint32 tuserpointer = signal->theData[1];
+ BlockReference tuserblockref = signal->theData[2];
+ switch (tcgcpblocked) {
+ case 1:
+ jam();
+ if (isMaster()) {
+ jam();
+ cgcpOrderBlocked = 1;
+ } else {
+ jam();
+ /* TRANSFER THE REQUEST */
+ /* TO MASTER*/
+ signal->theData[0] = tcgcpblocked;
+ signal->theData[1] = tuserpointer;
+ signal->theData[2] = tuserblockref;
+ sendSignal(cmasterdihref, GSN_DIHNDBTAMPER, signal, 3, JBB);
+ }//if
+ break;
+ case 2:
+ jam();
+ if (isMaster()) {
+ jam();
+ cgcpOrderBlocked = 0;
+ } else {
+ jam();
+ /* TRANSFER THE REQUEST */
+ /* TO MASTER*/
+ signal->theData[0] = tcgcpblocked;
+ signal->theData[1] = tuserpointer;
+ signal->theData[2] = tuserblockref;
+ sendSignal(cmasterdihref, GSN_DIHNDBTAMPER, signal, 3, JBB);
+ }//if
+ break;
+ case 3:
+ ndbrequire(false);
+ return;
+ break;
+ case 4:
+ jam();
+ signal->theData[0] = tuserpointer;
+ signal->theData[1] = crestartGci;
+ sendSignal(tuserblockref, GSN_DIHNDBTAMPER, signal, 2, JBB);
+ break;
+#ifdef ERROR_INSERT
+ case 5:
+ jam();
+ /*----------------------------------------------------------------------*/
+ // Insert errors.
+ /*----------------------------------------------------------------------*/
+ if (tuserpointer < 1000) {
+ /*--------------------------------------------------------------------*/
+ // Insert errors into QMGR.
+ /*--------------------------------------------------------------------*/
+ jam();
+ tuserblockref = QMGR_REF;
+ } else if (tuserpointer < 2000) {
+ /*--------------------------------------------------------------------*/
+ // Insert errors into NDBCNTR.
+ /*--------------------------------------------------------------------*/
+ jam();
+ tuserblockref = NDBCNTR_REF;
+ } else if (tuserpointer < 3000) {
+ /*--------------------------------------------------------------------*/
+ // Insert errors into NDBFS.
+ /*--------------------------------------------------------------------*/
+ jam();
+ tuserblockref = NDBFS_REF;
+ } else if (tuserpointer < 4000) {
+ /*--------------------------------------------------------------------*/
+ // Insert errors into DBACC.
+ /*--------------------------------------------------------------------*/
+ jam();
+ tuserblockref = DBACC_REF;
+ } else if (tuserpointer < 5000) {
+ /*--------------------------------------------------------------------*/
+ // Insert errors into DBTUP.
+ /*--------------------------------------------------------------------*/
+ jam();
+ tuserblockref = DBTUP_REF;
+ } else if (tuserpointer < 6000) {
+ /*---------------------------------------------------------------------*/
+ // Insert errors into DBLQH.
+ /*---------------------------------------------------------------------*/
+ jam();
+ tuserblockref = DBLQH_REF;
+ } else if (tuserpointer < 7000) {
+ /*---------------------------------------------------------------------*/
+ // Insert errors into DBDICT.
+ /*---------------------------------------------------------------------*/
+ jam();
+ tuserblockref = DBDICT_REF;
+ } else if (tuserpointer < 8000) {
+ /*---------------------------------------------------------------------*/
+ // Insert errors into DBDIH.
+ /*--------------------------------------------------------------------*/
+ jam();
+ tuserblockref = DBDIH_REF;
+ } else if (tuserpointer < 9000) {
+ /*--------------------------------------------------------------------*/
+ // Insert errors into DBTC.
+ /*--------------------------------------------------------------------*/
+ jam();
+ tuserblockref = DBTC_REF;
+ } else if (tuserpointer < 10000) {
+ /*--------------------------------------------------------------------*/
+ // Insert errors into CMVMI.
+ /*--------------------------------------------------------------------*/
+ jam();
+ tuserblockref = CMVMI_REF;
+ } else if (tuserpointer < 11000) {
+ jam();
+ tuserblockref = BACKUP_REF;
+ } else if (tuserpointer < 12000) {
+ // DBUTIL_REF ?
+ jam();
+ } else if (tuserpointer < 13000) {
+ jam();
+ tuserblockref = DBTUX_REF;
+ } else if (tuserpointer < 14000) {
+ jam();
+ tuserblockref = SUMA_REF;
+ } else if (tuserpointer < 15000) {
+ jam();
+ tuserblockref = DBDICT_REF;
+ } else if (tuserpointer < 30000) {
+ /*--------------------------------------------------------------------*/
+ // Ignore errors in the 20000-range.
+ /*--------------------------------------------------------------------*/
+ jam();
+ return;
+ } else if (tuserpointer < 40000) {
+ jam();
+ /*--------------------------------------------------------------------*/
+ // Redirect errors to master DIH in the 30000-range.
+ /*--------------------------------------------------------------------*/
+ tuserblockref = cmasterdihref;
+ tuserpointer -= 30000;
+ signal->theData[0] = 5;
+ signal->theData[1] = tuserpointer;
+ signal->theData[2] = tuserblockref;
+ sendSignal(tuserblockref, GSN_DIHNDBTAMPER, signal, 3, JBB);
+ return;
+ } else if (tuserpointer < 50000) {
+ NodeRecordPtr localNodeptr;
+ Uint32 Tfound = 0;
+ jam();
+ /*--------------------------------------------------------------------*/
+ // Redirect errors to non-master DIH in the 40000-range.
+ /*--------------------------------------------------------------------*/
+ tuserpointer -= 40000;
+ for (localNodeptr.i = 1;
+ localNodeptr.i < MAX_NDB_NODES;
+ localNodeptr.i++) {
+ jam();
+ ptrAss(localNodeptr, nodeRecord);
+ if ((localNodeptr.p->nodeStatus == NodeRecord::ALIVE) &&
+ (localNodeptr.i != cmasterNodeId)) {
+ jam();
+ tuserblockref = calcDihBlockRef(localNodeptr.i);
+ Tfound = 1;
+ break;
+ }//if
+ }//for
+ if (Tfound == 0) {
+ jam();
+ /*-------------------------------------------------------------------*/
+ // Ignore since no non-master node existed.
+ /*-------------------------------------------------------------------*/
+ return;
+ }//if
+ signal->theData[0] = 5;
+ signal->theData[1] = tuserpointer;
+ signal->theData[2] = tuserblockref;
+ sendSignal(tuserblockref, GSN_DIHNDBTAMPER, signal, 3, JBB);
+ return;
+ } else {
+ jam();
+ return;
+ }//if
+ signal->theData[0] = tuserpointer;
+ if (tuserpointer != 0) {
+ sendSignal(tuserblockref, GSN_NDB_TAMPER, signal, 1, JBB);
+ } else {
+ sendSignal(QMGR_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ sendSignal(NDBCNTR_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ sendSignal(NDBFS_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ sendSignal(DBACC_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ sendSignal(DBTUP_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ sendSignal(DBLQH_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ sendSignal(DBDICT_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ sendSignal(DBDIH_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ sendSignal(DBTC_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ sendSignal(CMVMI_REF, GSN_NDB_TAMPER, signal, 1, JBB);
+ }//if
+ break;
+#endif
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbdih::execDIHNDBTAMPER()
+
+/*****************************************************************************/
+/* ********** FILE HANDLING MODULE *************/
+/*****************************************************************************/
+void Dbdih::copyGciLab(Signal* signal, CopyGCIReq::CopyReason reason)
+{
+ if(c_copyGCIMaster.m_copyReason != CopyGCIReq::IDLE){
+ /**
+ * There can currently only be one waiting
+ */
+ ndbrequire(c_copyGCIMaster.m_waiting == CopyGCIReq::IDLE);
+ c_copyGCIMaster.m_waiting = reason;
+ return;
+ }
+ c_copyGCIMaster.m_copyReason = reason;
+ sendLoopMacro(COPY_GCIREQ, sendCOPY_GCIREQ);
+
+}//Dbdih::copyGciLab()
+
+/* ------------------------------------------------------------------------- */
+/* COPY_GCICONF RESPONSE TO COPY_GCIREQ */
+/* ------------------------------------------------------------------------- */
+void Dbdih::execCOPY_GCICONF(Signal* signal)
+{
+ jamEntry();
+ NodeRecordPtr senderNodePtr;
+ senderNodePtr.i = signal->theData[0];
+ receiveLoopMacro(COPY_GCIREQ, senderNodePtr.i);
+
+ CopyGCIReq::CopyReason waiting = c_copyGCIMaster.m_waiting;
+ CopyGCIReq::CopyReason current = c_copyGCIMaster.m_copyReason;
+
+ c_copyGCIMaster.m_copyReason = CopyGCIReq::IDLE;
+ c_copyGCIMaster.m_waiting = CopyGCIReq::IDLE;
+
+ bool ok = false;
+ switch(current){
+ case CopyGCIReq::RESTART:{
+ ok = true;
+ jam();
+ DictStartReq * req = (DictStartReq*)&signal->theData[0];
+ req->restartGci = SYSFILE->newestRestorableGCI;
+ req->senderRef = reference();
+ sendSignal(cdictblockref, GSN_DICTSTARTREQ,
+ signal, DictStartReq::SignalLength, JBB);
+ break;
+ }
+ case CopyGCIReq::LOCAL_CHECKPOINT:{
+ ok = true;
+ jam();
+ startLcpRoundLab(signal);
+ break;
+ }
+ case CopyGCIReq::GLOBAL_CHECKPOINT:
+ ok = true;
+ jam();
+ checkToCopyCompleted(signal);
+
+ /************************************************************************/
+ // Report the event that a global checkpoint has completed.
+ /************************************************************************/
+ signal->setTrace(0);
+ signal->theData[0] = EventReport::GlobalCheckpointCompleted; //Event type
+ signal->theData[1] = coldgcp;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ CRASH_INSERTION(7004);
+ emptyWaitGCPMasterQueue(signal);
+ cgcpStatus = GCP_READY;
+ signal->theData[0] = DihContinueB::ZSTART_GCP;
+ signal->theData[1] = cgcpDelay;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 2);
+ if (c_nodeStartMaster.blockGcp == true) {
+ jam();
+ /* ------------------------------------------------------------------ */
+ /* A NEW NODE WANTS IN AND WE MUST ALLOW IT TO COME IN NOW SINCE THE */
+ /* GCP IS COMPLETED. */
+ /* ------------------------------------------------------------------ */
+ gcpBlockedLab(signal);
+ }//if
+ break;
+ case CopyGCIReq::INITIAL_START_COMPLETED:
+ ok = true;
+ jam();
+ initialStartCompletedLab(signal);
+ break;
+ case CopyGCIReq::IDLE:
+ ok = false;
+ jam();
+ }
+ ndbrequire(ok);
+
+ /**
+ * Pop queue
+ */
+ if(waiting != CopyGCIReq::IDLE){
+ c_copyGCIMaster.m_copyReason = waiting;
+ signal->theData[0] = DihContinueB::ZCOPY_GCI;
+ signal->theData[1] = waiting;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ }
+}//Dbdih::execCOPY_GCICONF()
+
+void Dbdih::invalidateLcpInfoAfterSr()
+{
+ NodeRecordPtr nodePtr;
+ SYSFILE->latestLCP_ID--;
+ Sysfile::clearLCPOngoing(SYSFILE->systemRestartBits);
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if (!NdbNodeBitmask::get(SYSFILE->lcpActive, nodePtr.i)){
+ jam();
+ /* ------------------------------------------------------------------- */
+ // The node was not active in the local checkpoint.
+ // To avoid that we step the active status too fast to not
+ // active we step back one step from Sysfile::NS_ActiveMissed_x.
+ /* ------------------------------------------------------------------- */
+ switch (nodePtr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ /* ----------------------------------------------------------------- */
+ // When not active in ongoing LCP and still active is a contradiction.
+ /* ----------------------------------------------------------------- */
+ ndbrequire(false);
+ case Sysfile::NS_ActiveMissed_1:
+ jam();
+ nodePtr.p->activeStatus = Sysfile::NS_Active;
+ break;
+ case Sysfile::NS_ActiveMissed_2:
+ jam();
+ nodePtr.p->activeStatus = Sysfile::NS_ActiveMissed_1;
+ break;
+ default:
+ jam();
+ break;
+ }//switch
+ }//if
+ }//for
+ setNodeRestartInfoBits();
+}//Dbdih::invalidateLcpInfoAfterSr()
+
+/* ------------------------------------------------------------------------- */
+/* THE NEXT STEP IS TO WRITE THE FILE. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::openingCopyGciSkipInitLab(Signal* signal, FileRecordPtr filePtr)
+{
+ writeRestorableGci(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::WRITING_COPY_GCI;
+ return;
+}//Dbdih::openingCopyGciSkipInitLab()
+
+void Dbdih::writingCopyGciLab(Signal* signal, FileRecordPtr filePtr)
+{
+ /* ----------------------------------------------------------------------- */
+ /* WE HAVE NOW WRITTEN THIS FILE. WRITE ALSO NEXT FILE IF THIS IS NOT */
+ /* ALREADY THE LAST. */
+ /* ----------------------------------------------------------------------- */
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ if (filePtr.i == crestartInfoFile[0]) {
+ jam();
+ filePtr.i = crestartInfoFile[1];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ if (filePtr.p->fileStatus == FileRecord::OPEN) {
+ jam();
+ openingCopyGciSkipInitLab(signal, filePtr);
+ return;
+ }//if
+ openFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::OPENING_COPY_GCI;
+ return;
+ }//if
+ /* ----------------------------------------------------------------------- */
+ /* WE HAVE COMPLETED WRITING BOTH FILES SUCCESSFULLY. NOW REPORT OUR */
+ /* SUCCESS TO THE MASTER DIH. BUT FIRST WE NEED TO RESET A NUMBER OF */
+ /* VARIABLES USED BY THE LOCAL CHECKPOINT PROCESS (ONLY IF TRIGGERED */
+ /* BY LOCAL CHECKPOINT PROCESS. */
+ /* ----------------------------------------------------------------------- */
+ CopyGCIReq::CopyReason reason = c_copyGCISlave.m_copyReason;
+
+ if (reason == CopyGCIReq::GLOBAL_CHECKPOINT) {
+ jam();
+ cgcpParticipantState = GCP_PARTICIPANT_READY;
+
+ SubGcpCompleteRep * const rep = (SubGcpCompleteRep*)signal->getDataPtr();
+ rep->gci = coldgcp;
+ rep->senderData = 0;
+ sendSignal(SUMA_REF, GSN_SUB_GCP_COMPLETE_REP, signal,
+ SubGcpCompleteRep::SignalLength, JBB);
+ }
+
+ jam();
+ c_copyGCISlave.m_copyReason = CopyGCIReq::IDLE;
+
+ if(c_copyGCISlave.m_senderRef == cmasterdihref){
+ jam();
+ /**
+ * Only if same master
+ */
+ signal->theData[0] = c_copyGCISlave.m_senderData;
+ sendSignal(c_copyGCISlave.m_senderRef, GSN_COPY_GCICONF, signal, 1, JBB);
+
+ }
+ return;
+}//Dbdih::writingCopyGciLab()
+
+void Dbdih::execSTART_LCP_REQ(Signal* signal){
+ StartLcpReq * req = (StartLcpReq*)signal->getDataPtr();
+
+ CRASH_INSERTION2(7021, isMaster());
+ CRASH_INSERTION2(7022, !isMaster());
+
+ ndbrequire(c_lcpState.m_masterLcpDihRef = req->senderRef);
+ c_lcpState.m_participatingDIH = req->participatingDIH;
+ c_lcpState.m_participatingLQH = req->participatingLQH;
+
+ c_lcpState.m_LCP_COMPLETE_REP_Counter_LQH = req->participatingLQH;
+ if(isMaster()){
+ jam();
+ ndbrequire(isActiveMaster());
+ c_lcpState.m_LCP_COMPLETE_REP_Counter_DIH = req->participatingDIH;
+
+ } else {
+ c_lcpState.m_LCP_COMPLETE_REP_Counter_DIH.clearWaitingFor();
+ }
+
+ c_lcpState.m_LCP_COMPLETE_REP_From_Master_Received = false;
+
+ c_lcpState.setLcpStatus(LCP_INIT_TABLES, __LINE__);
+
+ signal->theData[0] = DihContinueB::ZINIT_LCP;
+ signal->theData[1] = c_lcpState.m_masterLcpDihRef;
+ signal->theData[2] = 0;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+}
+
+void Dbdih::initLcpLab(Signal* signal, Uint32 senderRef, Uint32 tableId)
+{
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+
+ if(c_lcpState.m_masterLcpDihRef != senderRef){
+ jam();
+ /**
+ * This is LCP master takeover
+ */
+#ifdef VM_TRACE
+ ndbout_c("initLcpLab aborted due to LCP master takeover - 1");
+#endif
+ c_lcpState.setLcpStatus(LCP_STATUS_IDLE, __LINE__);
+ sendMASTER_LCPCONF(signal);
+ return;
+ }
+
+ if(c_lcpState.m_masterLcpDihRef != cmasterdihref){
+ jam();
+ /**
+ * Master take over but has not yet received MASTER_LCPREQ
+ */
+#ifdef VM_TRACE
+ ndbout_c("initLcpLab aborted due to LCP master takeover - 2");
+#endif
+ return;
+ }
+
+ //const Uint32 lcpId = SYSFILE->latestLCP_ID;
+
+ for(; tabPtr.i < ctabFileSize; tabPtr.i++){
+
+ ptrAss(tabPtr, tabRecord);
+
+ if (tabPtr.p->tabStatus != TabRecord::TS_ACTIVE) {
+ jam();
+ tabPtr.p->tabLcpStatus = TabRecord::TLS_COMPLETED;
+ continue;
+ }
+
+ if (tabPtr.p->storedTable == 0) {
+ /**
+ * Temporary table
+ */
+ jam();
+ tabPtr.p->tabLcpStatus = TabRecord::TLS_COMPLETED;
+ continue;
+ }
+
+ if (tabPtr.p->tabCopyStatus != TabRecord::CS_IDLE) {
+ /* ----------------------------------------------------------------- */
+ // We protect the updates of table data structures by this variable.
+ /* ----------------------------------------------------------------- */
+ jam();
+ signal->theData[0] = DihContinueB::ZINIT_LCP;
+ signal->theData[1] = senderRef;
+ signal->theData[2] = tabPtr.i;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 20, 3);
+ return;
+ }//if
+
+ /**
+ * Found a table
+ */
+ tabPtr.p->tabLcpStatus = TabRecord::TLS_ACTIVE;
+
+ /**
+ * For each fragment
+ */
+ for (Uint32 fragId = 0; fragId < tabPtr.p->totalfragments; fragId++) {
+ jam();
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+
+ /**
+ * For each of replica record
+ */
+ Uint32 replicaCount = 0;
+ ReplicaRecordPtr replicaPtr;
+ for(replicaPtr.i = fragPtr.p->storedReplicas; replicaPtr.i != RNIL;
+ replicaPtr.i = replicaPtr.p->nextReplica) {
+ jam();
+
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ Uint32 nodeId = replicaPtr.p->procNode;
+ if(c_lcpState.m_participatingLQH.get(nodeId)){
+ jam();
+ replicaCount++;
+ replicaPtr.p->lcpOngoingFlag = true;
+ }
+ }
+
+ fragPtr.p->noLcpReplicas = replicaCount;
+ }//for
+
+ signal->theData[0] = DihContinueB::ZINIT_LCP;
+ signal->theData[1] = senderRef;
+ signal->theData[2] = tabPtr.i + 1;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ }
+
+ /**
+ * No more tables
+ */
+ jam();
+
+ if (c_lcpState.m_masterLcpDihRef != reference()){
+ jam();
+ ndbrequire(!isMaster());
+ c_lcpState.setLcpStatus(LCP_STATUS_ACTIVE, __LINE__);
+ } else {
+ jam();
+ ndbrequire(isMaster());
+ }
+
+ CRASH_INSERTION2(7023, isMaster());
+ CRASH_INSERTION2(7024, !isMaster());
+
+ jam();
+ StartLcpConf * conf = (StartLcpConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ sendSignal(c_lcpState.m_masterLcpDihRef, GSN_START_LCP_CONF, signal,
+ StartLcpConf::SignalLength, JBB);
+ return;
+}//Dbdih::initLcpLab()
+
+/* ------------------------------------------------------------------------- */
+/* ERROR HANDLING FOR COPY RESTORABLE GCI FILE. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::openingCopyGciErrorLab(Signal* signal, FileRecordPtr filePtr)
+{
+ createFileRw(signal, filePtr);
+ /* ------------------------------------------------------------------------- */
+ /* ERROR IN OPENING FILE. WE WILL TRY BY CREATING FILE INSTEAD. */
+ /* ------------------------------------------------------------------------- */
+ filePtr.p->reqStatus = FileRecord::CREATING_COPY_GCI;
+ return;
+}//Dbdih::openingCopyGciErrorLab()
+
+/* ------------------------------------------------------------------------- */
+/* ENTER DICTSTARTCONF WITH */
+/* TBLOCKREF */
+/* ------------------------------------------------------------------------- */
+void Dbdih::dictStartConfLab(Signal* signal)
+{
+ /* ----------------------------------------------------------------------- */
+ /* WE HAVE NOW RECEIVED ALL THE TABLES TO RESTART. */
+ /* ----------------------------------------------------------------------- */
+ signal->theData[0] = DihContinueB::ZSTART_FRAGMENT;
+ signal->theData[1] = 0; /* START WITH TABLE 0 */
+ signal->theData[2] = 0; /* AND FRAGMENT 0 */
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+}//Dbdih::dictStartConfLab()
+
+
+void Dbdih::openingTableLab(Signal* signal, FileRecordPtr filePtr)
+{
+ /* ---------------------------------------------------------------------- */
+ /* SUCCESSFULLY OPENED A FILE. READ THE FIRST PAGE OF THIS FILE. */
+ /* ---------------------------------------------------------------------- */
+ TabRecordPtr tabPtr;
+ PageRecordPtr pagePtr;
+
+ tabPtr.i = filePtr.p->tabRef;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ tabPtr.p->noPages = 1;
+ allocpage(pagePtr);
+ tabPtr.p->pageRef[0] = pagePtr.i;
+ readTabfile(signal, tabPtr.p, filePtr);
+ filePtr.p->reqStatus = FileRecord::READING_TABLE;
+ return;
+}//Dbdih::openingTableLab()
+
+void Dbdih::openingTableErrorLab(Signal* signal, FileRecordPtr filePtr)
+{
+ TabRecordPtr tabPtr;
+ tabPtr.i = filePtr.p->tabRef;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ /* ---------------------------------------------------------------------- */
+ /* WE FAILED IN OPENING A FILE. IF THE FIRST FILE THEN TRY WITH THE */
+ /* DUPLICATE FILE, OTHERWISE WE REPORT AN ERROR IN THE SYSTEM RESTART. */
+ /* ---------------------------------------------------------------------- */
+ ndbrequire(filePtr.i == tabPtr.p->tabFile[0]);
+ filePtr.i = tabPtr.p->tabFile[1];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ openFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::OPENING_TABLE;
+}//Dbdih::openingTableErrorLab()
+
+void Dbdih::readingTableLab(Signal* signal, FileRecordPtr filePtr)
+{
+ TabRecordPtr tabPtr;
+ PageRecordPtr pagePtr;
+ /* ---------------------------------------------------------------------- */
+ /* WE HAVE SUCCESSFULLY READ A NUMBER OF PAGES IN THE TABLE FILE. IF */
+ /* MORE PAGES EXIST IN THE FILE THEN READ ALL PAGES IN THE FILE. */
+ /* ---------------------------------------------------------------------- */
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ tabPtr.i = filePtr.p->tabRef;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ pagePtr.i = tabPtr.p->pageRef[0];
+ ptrCheckGuard(pagePtr, cpageFileSize, pageRecord);
+ Uint32 noOfStoredPages = pagePtr.p->word[33];
+ if (tabPtr.p->noPages < noOfStoredPages) {
+ jam();
+ ndbrequire(noOfStoredPages <= 8);
+ for (Uint32 i = tabPtr.p->noPages; i < noOfStoredPages; i++) {
+ jam();
+ allocpage(pagePtr);
+ tabPtr.p->pageRef[i] = pagePtr.i;
+ }//for
+ tabPtr.p->noPages = noOfStoredPages;
+ readTabfile(signal, tabPtr.p, filePtr);
+ filePtr.p->reqStatus = FileRecord::READING_TABLE;
+ } else {
+ ndbrequire(tabPtr.p->noPages == pagePtr.p->word[33]);
+ ndbrequire(tabPtr.p->tabCopyStatus == TabRecord::CS_IDLE);
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* WE HAVE READ ALL PAGES. NOW READ FROM PAGES INTO TABLE AND FRAGMENT */
+ /* DATA STRUCTURES. */
+ /* --------------------------------------------------------------------- */
+ tabPtr.p->tabCopyStatus = TabRecord::CS_SR_PHASE1_READ_PAGES;
+ signal->theData[0] = DihContinueB::ZREAD_PAGES_INTO_TABLE;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ }//if
+ return;
+}//Dbdih::readingTableLab()
+
+void Dbdih::readTableFromPagesLab(Signal* signal, TabRecordPtr tabPtr)
+{
+ FileRecordPtr filePtr;
+ filePtr.i = tabPtr.p->tabFile[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ /* ---------------------------------------------------------------------- */
+ /* WE HAVE NOW COPIED TO OUR NODE. WE HAVE NOW COMPLETED RESTORING */
+ /* THIS TABLE. CONTINUE WITH THE NEXT TABLE. */
+ /* WE ALSO NEED TO CLOSE THE TABLE FILE. */
+ /* ---------------------------------------------------------------------- */
+ if (filePtr.p->fileStatus != FileRecord::OPEN) {
+ jam();
+ filePtr.i = tabPtr.p->tabFile[1];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ }//if
+ closeFile(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::CLOSING_TABLE_SR;
+ return;
+}//Dbdih::readTableFromPagesLab()
+
+void Dbdih::closingTableSrLab(Signal* signal, FileRecordPtr filePtr)
+{
+ /**
+ * Update table/fragment info
+ */
+ TabRecordPtr tabPtr;
+ tabPtr.i = filePtr.p->tabRef;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ resetReplicaSr(tabPtr);
+
+ signal->theData[0] = DihContinueB::ZCOPY_TABLE;
+ signal->theData[1] = filePtr.p->tabRef;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+
+ return;
+}//Dbdih::closingTableSrLab()
+
+void
+Dbdih::resetReplicaSr(TabRecordPtr tabPtr){
+
+ const Uint32 newestRestorableGCI = SYSFILE->newestRestorableGCI;
+
+ for(Uint32 i = 0; i<tabPtr.p->totalfragments; i++){
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, i, fragPtr);
+
+ /**
+ * 1) Start by moving all replicas into oldStoredReplicas
+ */
+ prepareReplicas(fragPtr);
+
+ /**
+ * 2) Move all "alive" replicas into storedReplicas
+ * + update noCrashedReplicas...
+ */
+ ReplicaRecordPtr replicaPtr;
+ replicaPtr.i = fragPtr.p->oldStoredReplicas;
+ while (replicaPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ const Uint32 nextReplicaPtrI = replicaPtr.p->nextReplica;
+
+ NodeRecordPtr nodePtr;
+ nodePtr.i = replicaPtr.p->procNode;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+
+ const Uint32 noCrashedReplicas = replicaPtr.p->noCrashedReplicas;
+ if (nodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ jam();
+ switch (nodePtr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ case Sysfile::NS_ActiveMissed_1:
+ case Sysfile::NS_ActiveMissed_2:{
+ jam();
+ /* --------------------------------------------------------------- */
+ /* THE NODE IS ALIVE AND KICKING AND ACTIVE, LET'S USE IT. */
+ /* --------------------------------------------------------------- */
+ arrGuard(noCrashedReplicas, 8);
+ Uint32 lastGci = replicaPtr.p->replicaLastGci[noCrashedReplicas];
+ if(lastGci >= newestRestorableGCI){
+ jam();
+ /** -------------------------------------------------------------
+ * THE REPLICA WAS ALIVE AT THE SYSTEM FAILURE. WE WILL SET THE
+ * LAST REPLICA GCI TO MINUS ONE SINCE IT HASN'T FAILED YET IN THE
+ * NEW SYSTEM.
+ *-------------------------------------------------------------- */
+ replicaPtr.p->replicaLastGci[noCrashedReplicas] = (Uint32)-1;
+ } else {
+ jam();
+ /*--------------------------------------------------------------
+ * SINCE IT WAS NOT ALIVE AT THE TIME OF THE SYSTEM CRASH THIS IS
+ * A COMPLETELY NEW REPLICA. WE WILL SET THE CREATE GCI TO BE THE
+ * NEXT GCI TO BE EXECUTED.
+ *--------_----------------------------------------------------- */
+ const Uint32 nextCrashed = noCrashedReplicas + 1;
+ replicaPtr.p->noCrashedReplicas = nextCrashed;
+ arrGuard(nextCrashed, 8);
+ replicaPtr.p->createGci[nextCrashed] = newestRestorableGCI + 1;
+ ndbrequire(newestRestorableGCI + 1 != 0xF1F1F1F1);
+ replicaPtr.p->replicaLastGci[nextCrashed] = (Uint32)-1;
+ }//if
+
+ resetReplicaLcp(replicaPtr.p, newestRestorableGCI);
+
+ /* -----------------------------------------------------------------
+ * LINK THE REPLICA INTO THE STORED REPLICA LIST. WE WILL USE THIS
+ * NODE AS A STORED REPLICA.
+ * WE MUST FIRST LINK IT OUT OF THE LIST OF OLD STORED REPLICAS.
+ * --------------------------------------------------------------- */
+ removeOldStoredReplica(fragPtr, replicaPtr);
+ linkStoredReplica(fragPtr, replicaPtr);
+
+ }
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }
+ }
+ replicaPtr.i = nextReplicaPtrI;
+ }//while
+ }
+}
+
+void
+Dbdih::resetReplicaLcp(ReplicaRecord * replicaP, Uint32 stopGci){
+
+ Uint32 lcpNo = replicaP->nextLcp;
+ const Uint32 startLcpNo = lcpNo;
+ do {
+ lcpNo = prevLcpNo(lcpNo);
+ ndbrequire(lcpNo < MAX_LCP_STORED);
+ if (replicaP->lcpStatus[lcpNo] == ZVALID) {
+ if (replicaP->maxGciStarted[lcpNo] < stopGci) {
+ jam();
+ /* ----------------------------------------------------------------- */
+ /* WE HAVE FOUND A USEFUL LOCAL CHECKPOINT THAT CAN BE USED FOR */
+ /* RESTARTING THIS FRAGMENT REPLICA. */
+ /* ----------------------------------------------------------------- */
+ return ;
+ }//if
+ }//if
+
+ /**
+ * WE COULD NOT USE THIS LOCAL CHECKPOINT. IT WAS TOO
+ * RECENT OR SIMPLY NOT A VALID CHECKPOINT.
+ * WE SHOULD THUS REMOVE THIS LOCAL CHECKPOINT SINCE IT WILL NEVER
+ * AGAIN BE USED. SET LCP_STATUS TO INVALID.
+ */
+ replicaP->nextLcp = lcpNo;
+ replicaP->lcpId[lcpNo] = 0;
+ replicaP->lcpStatus[lcpNo] = ZINVALID;
+ } while (lcpNo != startLcpNo);
+
+ replicaP->nextLcp = 0;
+}
+
+void Dbdih::readingTableErrorLab(Signal* signal, FileRecordPtr filePtr)
+{
+ TabRecordPtr tabPtr;
+ tabPtr.i = filePtr.p->tabRef;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ /* ---------------------------------------------------------------------- */
+ /* READING THIS FILE FAILED. CLOSE IT AFTER RELEASING ALL PAGES. */
+ /* ---------------------------------------------------------------------- */
+ ndbrequire(tabPtr.p->noPages <= 8);
+ for (Uint32 i = 0; i < tabPtr.p->noPages; i++) {
+ jam();
+ releasePage(tabPtr.p->pageRef[i]);
+ }//for
+ closeFile(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::CLOSING_TABLE_CRASH;
+ return;
+}//Dbdih::readingTableErrorLab()
+
+void Dbdih::closingTableCrashLab(Signal* signal, FileRecordPtr filePtr)
+{
+ TabRecordPtr tabPtr;
+ /* ---------------------------------------------------------------------- */
+ /* WE HAVE NOW CLOSED A FILE WHICH WE HAD A READ ERROR WITH. PROCEED */
+ /* WITH NEXT FILE IF NOT THE LAST OTHERWISE REPORT ERROR. */
+ /* ---------------------------------------------------------------------- */
+ tabPtr.i = filePtr.p->tabRef;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ ndbrequire(filePtr.i == tabPtr.p->tabFile[0]);
+ filePtr.i = tabPtr.p->tabFile[1];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ openFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::OPENING_TABLE;
+}//Dbdih::closingTableCrashLab()
+
+/*****************************************************************************/
+/* ********** COPY TABLE MODULE *************/
+/*****************************************************************************/
+void Dbdih::execCOPY_TABREQ(Signal* signal)
+{
+ CRASH_INSERTION(7172);
+
+ TabRecordPtr tabPtr;
+ PageRecordPtr pagePtr;
+ jamEntry();
+ BlockReference ref = signal->theData[0];
+ Uint32 reqinfo = signal->theData[1];
+ tabPtr.i = signal->theData[2];
+ Uint32 schemaVersion = signal->theData[3];
+ Uint32 noOfWords = signal->theData[4];
+ ndbrequire(ref == cmasterdihref);
+ ndbrequire(!isMaster());
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ if (reqinfo == 1) {
+ jam();
+ tabPtr.p->schemaVersion = schemaVersion;
+ initTableFile(tabPtr);
+ }//if
+ ndbrequire(tabPtr.p->noPages < 8);
+ if (tabPtr.p->noOfWords == 0) {
+ jam();
+ allocpage(pagePtr);
+ tabPtr.p->pageRef[tabPtr.p->noPages] = pagePtr.i;
+ tabPtr.p->noPages++;
+ } else {
+ jam();
+ pagePtr.i = tabPtr.p->pageRef[tabPtr.p->noPages - 1];
+ ptrCheckGuard(pagePtr, cpageFileSize, pageRecord);
+ }//if
+ ndbrequire(tabPtr.p->noOfWords + 15 < 2048);
+ ndbrequire(tabPtr.p->noOfWords < 2048);
+ MEMCOPY_NO_WORDS(&pagePtr.p->word[tabPtr.p->noOfWords], &signal->theData[5], 16);
+ tabPtr.p->noOfWords += 16;
+ if (tabPtr.p->noOfWords == 2048) {
+ jam();
+ tabPtr.p->noOfWords = 0;
+ }//if
+ if (noOfWords > 16) {
+ jam();
+ return;
+ }//if
+ tabPtr.p->noOfWords = 0;
+ ndbrequire(tabPtr.p->tabCopyStatus == TabRecord::CS_IDLE);
+ tabPtr.p->tabCopyStatus = TabRecord::CS_COPY_TAB_REQ;
+ signal->theData[0] = DihContinueB::ZREAD_PAGES_INTO_TABLE;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+}//Dbdih::execCOPY_TABREQ()
+
+void
+Dbdih::copyTabReq_complete(Signal* signal, TabRecordPtr tabPtr){
+ if (!isMaster()) {
+ jam();
+ //----------------------------------------------------------------------------
+ // In this particular case we do not release table pages if we are master. The
+ // reason is that the master could still be sending the table info to another
+ // node.
+ //----------------------------------------------------------------------------
+ releaseTabPages(tabPtr.i);
+ tabPtr.p->tabStatus = TabRecord::TS_ACTIVE;
+ for (Uint32 fragId = 0; fragId < tabPtr.p->totalfragments; fragId++) {
+ jam();
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ updateNodeInfo(fragPtr);
+ }//for
+ }//if
+ signal->theData[0] = cownNodeId;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(cmasterdihref, GSN_COPY_TABCONF, signal, 2, JBB);
+}
+
+/*****************************************************************************/
+/* ****** READ FROM A NUMBER OF PAGES INTO THE TABLE DATA STRUCTURES ********/
+/*****************************************************************************/
+void Dbdih::readPagesIntoTableLab(Signal* signal, Uint32 tableId)
+{
+ RWFragment rf;
+ rf.wordIndex = 35;
+ rf.pageIndex = 0;
+ rf.rwfTabPtr.i = tableId;
+ ptrCheckGuard(rf.rwfTabPtr, ctabFileSize, tabRecord);
+ rf.rwfPageptr.i = rf.rwfTabPtr.p->pageRef[0];
+ ptrCheckGuard(rf.rwfPageptr, cpageFileSize, pageRecord);
+ rf.rwfTabPtr.p->totalfragments = readPageWord(&rf);
+ rf.rwfTabPtr.p->noOfBackups = readPageWord(&rf);
+ rf.rwfTabPtr.p->hashpointer = readPageWord(&rf);
+ rf.rwfTabPtr.p->kvalue = readPageWord(&rf);
+ rf.rwfTabPtr.p->mask = readPageWord(&rf);
+ ndbrequire(readPageWord(&rf) == TabRecord::HASH);
+ rf.rwfTabPtr.p->method = TabRecord::HASH;
+ /* ---------------------------------- */
+ /* Type of table, 2 = temporary table */
+ /* ---------------------------------- */
+ rf.rwfTabPtr.p->storedTable = readPageWord(&rf);
+
+ Uint32 noOfFrags = rf.rwfTabPtr.p->totalfragments;
+ ndbrequire(noOfFrags > 0);
+ ndbrequire((noOfFrags * (rf.rwfTabPtr.p->noOfBackups + 1)) <= cnoFreeReplicaRec);
+ allocFragments(noOfFrags, rf.rwfTabPtr);
+
+ signal->theData[0] = DihContinueB::ZREAD_PAGES_INTO_FRAG;
+ signal->theData[1] = rf.rwfTabPtr.i;
+ signal->theData[2] = 0;
+ signal->theData[3] = rf.pageIndex;
+ signal->theData[4] = rf.wordIndex;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 5, JBB);
+ return;
+}//Dbdih::readPagesIntoTableLab()
+
+void Dbdih::readPagesIntoFragLab(Signal* signal, RWFragment* rf)
+{
+ ndbrequire(rf->pageIndex < 8);
+ rf->rwfPageptr.i = rf->rwfTabPtr.p->pageRef[rf->pageIndex];
+ ptrCheckGuard(rf->rwfPageptr, cpageFileSize, pageRecord);
+ FragmentstorePtr fragPtr;
+ getFragstore(rf->rwfTabPtr.p, rf->fragId, fragPtr);
+ readFragment(rf, fragPtr);
+ readReplicas(rf, fragPtr);
+ rf->fragId++;
+ if (rf->fragId == rf->rwfTabPtr.p->totalfragments) {
+ jam();
+ switch (rf->rwfTabPtr.p->tabCopyStatus) {
+ case TabRecord::CS_SR_PHASE1_READ_PAGES:
+ jam();
+ releaseTabPages(rf->rwfTabPtr.i);
+ rf->rwfTabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ signal->theData[0] = DihContinueB::ZREAD_TABLE_FROM_PAGES;
+ signal->theData[1] = rf->rwfTabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ case TabRecord::CS_COPY_TAB_REQ:
+ jam();
+ rf->rwfTabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ if(getNodeState().getSystemRestartInProgress()){
+ jam();
+ copyTabReq_complete(signal, rf->rwfTabPtr);
+ return;
+ }
+ rf->rwfTabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ rf->rwfTabPtr.p->tabUpdateState = TabRecord::US_COPY_TAB_REQ;
+ signal->theData[0] = DihContinueB::ZTABLE_UPDATE;
+ signal->theData[1] = rf->rwfTabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ break;
+ }//switch
+ } else {
+ jam();
+ signal->theData[0] = DihContinueB::ZREAD_PAGES_INTO_FRAG;
+ signal->theData[1] = rf->rwfTabPtr.i;
+ signal->theData[2] = rf->fragId;
+ signal->theData[3] = rf->pageIndex;
+ signal->theData[4] = rf->wordIndex;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 5, JBB);
+ }//if
+ return;
+}//Dbdih::readPagesIntoFragLab()
+
+/*****************************************************************************/
+/***** WRITING FROM TABLE DATA STRUCTURES INTO A SET OF PAGES ******/
+// execCONTINUEB(ZPACK_TABLE_INTO_PAGES)
+/*****************************************************************************/
+void Dbdih::packTableIntoPagesLab(Signal* signal, Uint32 tableId)
+{
+ RWFragment wf;
+ TabRecordPtr tabPtr;
+ allocpage(wf.rwfPageptr);
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ tabPtr.p->pageRef[0] = wf.rwfPageptr.i;
+ tabPtr.p->noPages = 1;
+ wf.wordIndex = 35;
+ wf.pageIndex = 0;
+ writePageWord(&wf, tabPtr.p->totalfragments);
+ writePageWord(&wf, tabPtr.p->noOfBackups);
+ writePageWord(&wf, tabPtr.p->hashpointer);
+ writePageWord(&wf, tabPtr.p->kvalue);
+ writePageWord(&wf, tabPtr.p->mask);
+ writePageWord(&wf, TabRecord::HASH);
+ writePageWord(&wf, tabPtr.p->storedTable);
+
+ signal->theData[0] = DihContinueB::ZPACK_FRAG_INTO_PAGES;
+ signal->theData[1] = tabPtr.i;
+ signal->theData[2] = 0;
+ signal->theData[3] = wf.pageIndex;
+ signal->theData[4] = wf.wordIndex;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 5, JBB);
+}//Dbdih::packTableIntoPagesLab()
+
+/*****************************************************************************/
+// execCONTINUEB(ZPACK_FRAG_INTO_PAGES)
+/*****************************************************************************/
+void Dbdih::packFragIntoPagesLab(Signal* signal, RWFragment* wf)
+{
+ ndbrequire(wf->pageIndex < 8);
+ wf->rwfPageptr.i = wf->rwfTabPtr.p->pageRef[wf->pageIndex];
+ ptrCheckGuard(wf->rwfPageptr, cpageFileSize, pageRecord);
+ FragmentstorePtr fragPtr;
+ getFragstore(wf->rwfTabPtr.p, wf->fragId, fragPtr);
+ writeFragment(wf, fragPtr);
+ writeReplicas(wf, fragPtr.p->storedReplicas);
+ writeReplicas(wf, fragPtr.p->oldStoredReplicas);
+ wf->fragId++;
+ if (wf->fragId == wf->rwfTabPtr.p->totalfragments) {
+ jam();
+ PageRecordPtr pagePtr;
+ pagePtr.i = wf->rwfTabPtr.p->pageRef[0];
+ ptrCheckGuard(pagePtr, cpageFileSize, pageRecord);
+ pagePtr.p->word[33] = wf->rwfTabPtr.p->noPages;
+ pagePtr.p->word[34] = ((wf->rwfTabPtr.p->noPages - 1) * 2048) + wf->wordIndex;
+ switch (wf->rwfTabPtr.p->tabCopyStatus) {
+ case TabRecord::CS_SR_PHASE2_READ_TABLE:
+ /* -------------------------------------------------------------------*/
+ // We are performing a system restart and we are now ready to copy the
+ // table from this node (the master) to all other nodes.
+ /* -------------------------------------------------------------------*/
+ jam();
+ wf->rwfTabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ signal->theData[0] = DihContinueB::ZSR_PHASE2_READ_TABLE;
+ signal->theData[1] = wf->rwfTabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ case TabRecord::CS_COPY_NODE_STATE:
+ jam();
+ tableCopyNodeLab(signal, wf->rwfTabPtr);
+ return;
+ break;
+ case TabRecord::CS_LCP_READ_TABLE:
+ jam();
+ signal->theData[0] = DihContinueB::ZTABLE_UPDATE;
+ signal->theData[1] = wf->rwfTabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ case TabRecord::CS_REMOVE_NODE:
+ case TabRecord::CS_INVALIDATE_NODE_LCP:
+ jam();
+ signal->theData[0] = DihContinueB::ZTABLE_UPDATE;
+ signal->theData[1] = wf->rwfTabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ case TabRecord::CS_ADD_TABLE_MASTER:
+ jam();
+ wf->rwfTabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ signal->theData[0] = DihContinueB::ZADD_TABLE_MASTER_PAGES;
+ signal->theData[1] = wf->rwfTabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ case TabRecord::CS_ADD_TABLE_SLAVE:
+ jam();
+ wf->rwfTabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ signal->theData[0] = DihContinueB::ZADD_TABLE_SLAVE_PAGES;
+ signal->theData[1] = wf->rwfTabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ break;
+ }//switch
+ } else {
+ jam();
+ signal->theData[0] = DihContinueB::ZPACK_FRAG_INTO_PAGES;
+ signal->theData[1] = wf->rwfTabPtr.i;
+ signal->theData[2] = wf->fragId;
+ signal->theData[3] = wf->pageIndex;
+ signal->theData[4] = wf->wordIndex;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 5, JBB);
+ }//if
+ return;
+}//Dbdih::packFragIntoPagesLab()
+
+/*****************************************************************************/
+/* ********** START FRAGMENT MODULE *************/
+/*****************************************************************************/
+void Dbdih::startFragment(Signal* signal, Uint32 tableId, Uint32 fragId)
+{
+ Uint32 TloopCount = 0;
+ TabRecordPtr tabPtr;
+ while (true) {
+ if (TloopCount > 100) {
+ jam();
+ signal->theData[0] = DihContinueB::ZSTART_FRAGMENT;
+ signal->theData[1] = tableId;
+ signal->theData[2] = 0;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ }
+
+ if (tableId >= ctabFileSize) {
+ jam();
+ signal->theData[0] = DihContinueB::ZCOMPLETE_RESTART;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 1, JBB);
+ return;
+ }//if
+
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ if (tabPtr.p->tabStatus != TabRecord::TS_ACTIVE){
+ jam();
+ TloopCount++;
+ tableId++;
+ fragId = 0;
+ continue;
+ }
+
+ if(tabPtr.p->storedTable == 0){
+ jam();
+ TloopCount++;
+ tableId++;
+ fragId = 0;
+ continue;
+ }
+
+ jam();
+ break;
+ }//while
+
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ /* ----------------------------------------------------------------------- */
+ /* WE NEED TO RESET THE REPLICA DATA STRUCTURES. THIS MEANS THAT WE */
+ /* MUST REMOVE REPLICAS THAT WAS NOT STARTED AT THE GCI TO RESTORE. WE */
+ /* NEED TO PUT ALL STORED REPLICAS ON THE LIST OF OLD STORED REPLICAS */
+ /* RESET THE NUMBER OF REPLICAS TO CREATE. */
+ /* ----------------------------------------------------------------------- */
+ cnoOfCreateReplicas = 0;
+ /* ----------------------------------------------------------------------- */
+ /* WE WILL NEVER START MORE THAN FOUR FRAGMENT REPLICAS WHATEVER THE */
+ /* DESIRED REPLICATION IS. */
+ /* ----------------------------------------------------------------------- */
+ ndbrequire(tabPtr.p->noOfBackups < 4);
+ /* ----------------------------------------------------------------------- */
+ /* SEARCH FOR STORED REPLICAS THAT CAN BE USED TO RESTART THE SYSTEM. */
+ /* ----------------------------------------------------------------------- */
+ searchStoredReplicas(fragPtr);
+ if (cnoOfCreateReplicas == 0) {
+ /* --------------------------------------------------------------------- */
+ /* THERE WERE NO STORED REPLICAS AVAILABLE THAT CAN SERVE AS REPLICA TO*/
+ /* RESTART THE SYSTEM FROM. IN A LATER RELEASE WE WILL ADD */
+ /* FUNCTIONALITY TO CHECK IF THERE ARE ANY STANDBY NODES THAT COULD DO */
+ /* THIS TASK INSTEAD IN THIS IMPLEMENTATION WE SIMPLY CRASH THE SYSTEM.*/
+ /* THIS WILL DECREASE THE GCI TO RESTORE WHICH HOPEFULLY WILL MAKE IT */
+ /* POSSIBLE TO RESTORE THE SYSTEM. */
+ /* --------------------------------------------------------------------- */
+ char buf[100];
+ snprintf(buf, sizeof(buf),
+ "Unable to find restorable replica for "
+ "table: %d fragment: %d gci: %d",
+ tableId, fragId, SYSFILE->newestRestorableGCI);
+ progError(__LINE__,
+ ERR_SYSTEM_ERROR,
+ buf);
+ ndbrequire(false);
+ return;
+ }//if
+
+ /* ----------------------------------------------------------------------- */
+ /* WE HAVE CHANGED THE NODE TO BE PRIMARY REPLICA AND THE NODES TO BE */
+ /* BACKUP NODES. WE MUST UPDATE THIS NODES DATA STRUCTURE SINCE WE */
+ /* WILL NOT COPY THE TABLE DATA TO OURSELF. */
+ /* ----------------------------------------------------------------------- */
+ updateNodeInfo(fragPtr);
+ /* ----------------------------------------------------------------------- */
+ /* NOW WE HAVE COLLECTED ALL THE REPLICAS WE COULD GET. WE WILL NOW */
+ /* RESTART THE FRAGMENT REPLICAS WE HAVE FOUND IRRESPECTIVE OF IF THERE*/
+ /* ARE ENOUGH ACCORDING TO THE DESIRED REPLICATION. */
+ /* ----------------------------------------------------------------------- */
+ /* WE START BY SENDING ADD_FRAGREQ FOR THOSE REPLICAS THAT NEED IT. */
+ /* ----------------------------------------------------------------------- */
+ CreateReplicaRecordPtr createReplicaPtr;
+ for (createReplicaPtr.i = 0;
+ createReplicaPtr.i < cnoOfCreateReplicas;
+ createReplicaPtr.i++) {
+ jam();
+ ptrCheckGuard(createReplicaPtr, 4, createReplicaRecord);
+ createReplicaPtr.p->hotSpareUse = false;
+ }//for
+
+ sendStartFragreq(signal, tabPtr, fragId);
+
+ /**
+ * Don't wait for START_FRAGCONF
+ */
+ fragId++;
+ if (fragId >= tabPtr.p->totalfragments) {
+ jam();
+ tabPtr.i++;
+ fragId = 0;
+ }//if
+ signal->theData[0] = DihContinueB::ZSTART_FRAGMENT;
+ signal->theData[1] = tabPtr.i;
+ signal->theData[2] = fragId;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+
+ return;
+}//Dbdih::startFragmentLab()
+
+
+/*****************************************************************************/
+/* ********** COMPLETE RESTART MODULE *************/
+/*****************************************************************************/
+void Dbdih::completeRestartLab(Signal* signal)
+{
+ sendLoopMacro(START_RECREQ, sendSTART_RECREQ);
+}//completeRestartLab()
+
+/* ------------------------------------------------------------------------- */
+// SYSTEM RESTART:
+/* A NODE HAS COMPLETED RESTORING ALL DATABASE FRAGMENTS. */
+// NODE RESTART:
+// THE STARTING NODE HAS PREPARED ITS LOG FILES TO ENABLE EXECUTION
+// OF TRANSACTIONS.
+// Precondition:
+// This signal must be received by the master node.
+/* ------------------------------------------------------------------------- */
+void Dbdih::execSTART_RECCONF(Signal* signal)
+{
+ jamEntry();
+ Uint32 senderNodeId = signal->theData[0];
+ ndbrequire(isMaster());
+ if (getNodeState().startLevel >= NodeState::SL_STARTED){
+ /* --------------------------------------------------------------------- */
+ // Since our node is already up and running this must be a node restart.
+ // This means that we should be the master node,
+ // otherwise we have a problem.
+ /* --------------------------------------------------------------------- */
+ jam();
+ ndbrequire(senderNodeId == c_nodeStartMaster.startNode);
+ nodeRestartStartRecConfLab(signal);
+ return;
+ } else {
+ /* --------------------------------------------------------------------- */
+ // This was the system restart case. We set the state indicating that the
+ // node has completed restoration of all fragments.
+ /* --------------------------------------------------------------------- */
+ receiveLoopMacro(START_RECREQ, senderNodeId);
+
+ signal->theData[0] = reference();
+ sendSignal(cntrlblockref, GSN_NDB_STARTCONF, signal, 1, JBB);
+ return;
+ }//if
+}//Dbdih::execSTART_RECCONF()
+
+void Dbdih::copyNodeLab(Signal* signal, Uint32 tableId)
+{
+ /* ----------------------------------------------------------------------- */
+ // This code is executed by the master to assist a node restart in receiving
+ // the data in the master.
+ /* ----------------------------------------------------------------------- */
+ Uint32 TloopCount = 0;
+
+ if (!c_nodeStartMaster.activeState) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ // Obviously the node crashed in the middle of its node restart. We will
+ // stop this process simply by returning after resetting the wait indicator.
+ /* ---------------------------------------------------------------------- */
+ c_nodeStartMaster.wait = ZFALSE;
+ return;
+ }//if
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ while (tabPtr.i < ctabFileSize) {
+ ptrAss(tabPtr, tabRecord);
+ if (tabPtr.p->tabStatus == TabRecord::TS_ACTIVE) {
+ /* -------------------------------------------------------------------- */
+ // The table is defined. We will start by packing the table into pages.
+ // The tabCopyStatus indicates to the CONTINUEB(ZPACK_TABLE_INTO_PAGES)
+ // who called it. After packing the table into page(s) it will be sent to
+ // the starting node by COPY_TABREQ signals. After returning from the
+ // starting node we will return to this subroutine and continue
+ // with the next table.
+ /* -------------------------------------------------------------------- */
+ ndbrequire(tabPtr.p->tabCopyStatus == TabRecord::CS_IDLE);
+ tabPtr.p->tabCopyStatus = TabRecord::CS_COPY_NODE_STATE;
+ signal->theData[0] = DihContinueB::ZPACK_TABLE_INTO_PAGES;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ } else {
+ jam();
+ if (TloopCount > 100) {
+ /* ------------------------------------------------------------------ */
+ // Introduce real-time break after looping through 100 not copied tables
+ /* ----------------------------------------------------------------- */
+ jam();
+ signal->theData[0] = DihContinueB::ZCOPY_NODE;
+ signal->theData[1] = tabPtr.i + 1;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ } else {
+ jam();
+ TloopCount++;
+ tabPtr.i++;
+ }//if
+ }//if
+ }//while
+ dihCopyCompletedLab(signal);
+ return;
+}//Dbdih::copyNodeLab()
+
+void Dbdih::tableCopyNodeLab(Signal* signal, TabRecordPtr tabPtr)
+{
+ /* ----------------------------------------------------------------------- */
+ /* COPY PAGES READ TO STARTING NODE. */
+ /* ----------------------------------------------------------------------- */
+ if (!c_nodeStartMaster.activeState) {
+ jam();
+ releaseTabPages(tabPtr.i);
+ c_nodeStartMaster.wait = ZFALSE;
+ return;
+ }//if
+ NodeRecordPtr copyNodePtr;
+ PageRecordPtr pagePtr;
+ copyNodePtr.i = c_nodeStartMaster.startNode;
+ ptrCheckGuard(copyNodePtr, MAX_NDB_NODES, nodeRecord);
+
+ copyNodePtr.p->activeTabptr = tabPtr.i;
+ pagePtr.i = tabPtr.p->pageRef[0];
+ ptrCheckGuard(pagePtr, cpageFileSize, pageRecord);
+
+ signal->theData[0] = DihContinueB::ZCOPY_TABLE_NODE;
+ signal->theData[1] = tabPtr.i;
+ signal->theData[2] = copyNodePtr.i;
+ signal->theData[3] = 0;
+ signal->theData[4] = 0;
+ signal->theData[5] = pagePtr.p->word[34];
+ sendSignal(reference(), GSN_CONTINUEB, signal, 6, JBB);
+}//Dbdih::tableCopyNodeLab()
+
+/* ------------------------------------------------------------------------- */
+// execCONTINUEB(ZCOPY_TABLE)
+// This routine is used to copy the table descriptions from the master to
+// other nodes. It is used in the system restart to copy from master to all
+// starting nodes.
+/* ------------------------------------------------------------------------- */
+void Dbdih::copyTableLab(Signal* signal, Uint32 tableId)
+{
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrAss(tabPtr, tabRecord);
+
+ ndbrequire(tabPtr.p->tabCopyStatus == TabRecord::CS_IDLE);
+ tabPtr.p->tabCopyStatus = TabRecord::CS_SR_PHASE2_READ_TABLE;
+ signal->theData[0] = DihContinueB::ZPACK_TABLE_INTO_PAGES;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+}//Dbdih::copyTableLab()
+
+/* ------------------------------------------------------------------------- */
+// execCONTINUEB(ZSR_PHASE2_READ_TABLE)
+/* ------------------------------------------------------------------------- */
+void Dbdih::srPhase2ReadTableLab(Signal* signal, TabRecordPtr tabPtr)
+{
+ /* ----------------------------------------------------------------------- */
+ // We set the sendCOPY_TABREQState to ZACTIVE for all nodes since it is a long
+ // process to send off all table descriptions. Thus we ensure that we do
+ // not encounter race conditions where one node is completed before the
+ // sending process is completed. This could lead to that we start off the
+ // system before we actually finished all copying of table descriptions
+ // and could lead to strange errors.
+ /* ----------------------------------------------------------------------- */
+
+ //sendLoopMacro(COPY_TABREQ, nullRoutine);
+
+ breakCopyTableLab(signal, tabPtr, cfirstAliveNode);
+ return;
+}//Dbdih::srPhase2ReadTableLab()
+
+/* ------------------------------------------------------------------------- */
+/* COPY PAGES READ TO ALL NODES. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::breakCopyTableLab(Signal* signal, TabRecordPtr tabPtr, Uint32 nodeId)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ while (nodePtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.i == getOwnNodeId()){
+ jam();
+ /* ------------------------------------------------------------------- */
+ /* NOT NECESSARY TO COPY TO MY OWN NODE. I ALREADY HAVE THE PAGES. */
+ /* I DO HOWEVER NEED TO STORE THE TABLE DESCRIPTION ONTO DISK. */
+ /* ------------------------------------------------------------------- */
+ /* IF WE ARE MASTER WE ONLY NEED TO SAVE THE TABLE ON DISK. WE ALREADY */
+ /* HAVE THE TABLE DESCRIPTION IN THE DATA STRUCTURES. */
+ // AFTER COMPLETING THE WRITE TO DISK THE MASTER WILL ALSO SEND
+ // COPY_TABCONF AS ALL THE OTHER NODES.
+ /* ------------------------------------------------------------------- */
+ c_COPY_TABREQ_Counter.setWaitingFor(nodePtr.i);
+ tabPtr.p->tabUpdateState = TabRecord::US_COPY_TAB_REQ;
+ signal->theData[0] = DihContinueB::ZTABLE_UPDATE;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ nodePtr.i = nodePtr.p->nextNode;
+ } else {
+ PageRecordPtr pagePtr;
+ /* -------------------------------------------------------------------- */
+ // RATHER THAN SENDING ALL COPY_TABREQ IN PARALLEL WE WILL SERIALISE THIS
+ // ACTIVITY AND WILL THUS CALL breakCopyTableLab AGAIN WHEN COMPLETED THE
+ // SENDING OF COPY_TABREQ'S.
+ /* -------------------------------------------------------------------- */
+ jam();
+ tabPtr.p->tabCopyStatus = TabRecord::CS_SR_PHASE3_COPY_TABLE;
+ pagePtr.i = tabPtr.p->pageRef[0];
+ ptrCheckGuard(pagePtr, cpageFileSize, pageRecord);
+ signal->theData[0] = DihContinueB::ZCOPY_TABLE_NODE;
+ signal->theData[1] = tabPtr.i;
+ signal->theData[2] = nodePtr.i;
+ signal->theData[3] = 0;
+ signal->theData[4] = 0;
+ signal->theData[5] = pagePtr.p->word[34];
+ sendSignal(reference(), GSN_CONTINUEB, signal, 6, JBB);
+ return;
+ }//if
+ }//while
+ /* ----------------------------------------------------------------------- */
+ /* WE HAVE NOW SENT THE TABLE PAGES TO ALL NODES. EXIT AND WAIT FOR ALL */
+ /* REPLIES. */
+ /* ----------------------------------------------------------------------- */
+ return;
+}//Dbdih::breakCopyTableLab()
+
+/* ------------------------------------------------------------------------- */
+// execCONTINUEB(ZCOPY_TABLE_NODE)
+/* ------------------------------------------------------------------------- */
+void Dbdih::copyTableNode(Signal* signal,
+ CopyTableNode* ctn, NodeRecordPtr nodePtr)
+{
+ if (getNodeState().startLevel >= NodeState::SL_STARTED){
+ /* --------------------------------------------------------------------- */
+ // We are in the process of performing a node restart and are copying a
+ // table description to a starting node. We will check that no nodes have
+ // crashed in this process.
+ /* --------------------------------------------------------------------- */
+ if (!c_nodeStartMaster.activeState) {
+ jam();
+ /** ------------------------------------------------------------------
+ * The starting node crashed. We will release table pages and stop this
+ * copy process and allow new node restarts to start.
+ * ------------------------------------------------------------------ */
+ releaseTabPages(ctn->ctnTabPtr.i);
+ c_nodeStartMaster.wait = ZFALSE;
+ return;
+ }//if
+ }//if
+ ndbrequire(ctn->pageIndex < 8);
+ ctn->ctnPageptr.i = ctn->ctnTabPtr.p->pageRef[ctn->pageIndex];
+ ptrCheckGuard(ctn->ctnPageptr, cpageFileSize, pageRecord);
+ /**
+ * If first page & firstWord reqinfo = 1 (first signal)
+ */
+ Uint32 reqinfo = (ctn->pageIndex == 0) && (ctn->wordIndex == 0);
+ if(reqinfo == 1){
+ c_COPY_TABREQ_Counter.setWaitingFor(nodePtr.i);
+ }
+
+ for (Uint32 i = 0; i < 16; i++) {
+ jam();
+ sendCopyTable(signal, ctn, calcDihBlockRef(nodePtr.i), reqinfo);
+ reqinfo = 0;
+ if (ctn->noOfWords <= 16) {
+ jam();
+ switch (ctn->ctnTabPtr.p->tabCopyStatus) {
+ case TabRecord::CS_SR_PHASE3_COPY_TABLE:
+ /* ------------------------------------------------------------------ */
+ // We have copied the table description to this node.
+ // We will now proceed
+ // with sending the table description to the next node in the node list.
+ /* ------------------------------------------------------------------ */
+ jam();
+ ctn->ctnTabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ breakCopyTableLab(signal, ctn->ctnTabPtr, nodePtr.p->nextNode);
+ return;
+ break;
+ case TabRecord::CS_COPY_NODE_STATE:
+ jam();
+ ctn->ctnTabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ } else {
+ jam();
+ ctn->wordIndex += 16;
+ if (ctn->wordIndex == 2048) {
+ jam();
+ ctn->wordIndex = 0;
+ ctn->pageIndex++;
+ ndbrequire(ctn->pageIndex < 8);
+ ctn->ctnPageptr.i = ctn->ctnTabPtr.p->pageRef[ctn->pageIndex];
+ ptrCheckGuard(ctn->ctnPageptr, cpageFileSize, pageRecord);
+ }//if
+ ctn->noOfWords -= 16;
+ }//if
+ }//for
+ signal->theData[0] = DihContinueB::ZCOPY_TABLE_NODE;
+ signal->theData[1] = ctn->ctnTabPtr.i;
+ signal->theData[2] = nodePtr.i;
+ signal->theData[3] = ctn->pageIndex;
+ signal->theData[4] = ctn->wordIndex;
+ signal->theData[5] = ctn->noOfWords;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 6, JBB);
+}//Dbdih::copyTableNodeLab()
+
+void Dbdih::sendCopyTable(Signal* signal, CopyTableNode* ctn,
+ BlockReference ref, Uint32 reqinfo)
+{
+ signal->theData[0] = reference();
+ signal->theData[1] = reqinfo;
+ signal->theData[2] = ctn->ctnTabPtr.i;
+ signal->theData[3] = ctn->ctnTabPtr.p->schemaVersion;
+ signal->theData[4] = ctn->noOfWords;
+ ndbrequire(ctn->wordIndex + 15 < 2048);
+ MEMCOPY_NO_WORDS(&signal->theData[5], &ctn->ctnPageptr.p->word[ctn->wordIndex], 16);
+ sendSignal(ref, GSN_COPY_TABREQ, signal, 21, JBB);
+}//Dbdih::sendCopyTable()
+
+void Dbdih::execCOPY_TABCONF(Signal* signal)
+{
+ NodeRecordPtr nodePtr;
+ jamEntry();
+ nodePtr.i = signal->theData[0];
+ Uint32 tableId = signal->theData[1];
+ if (getNodeState().startLevel >= NodeState::SL_STARTED){
+ /* --------------------------------------------------------------------- */
+ // We are in the process of performing a node restart. Continue by copying
+ // the next table to the starting node.
+ /* --------------------------------------------------------------------- */
+ jam();
+ NodeRecordPtr nodePtr;
+ nodePtr.i = signal->theData[0];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ c_COPY_TABREQ_Counter.clearWaitingFor(nodePtr.i);
+
+ releaseTabPages(tableId);
+ signal->theData[0] = DihContinueB::ZCOPY_NODE;
+ signal->theData[1] = tableId + 1;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ } else {
+ /* --------------------------------------------------------------------- */
+ // We are in the process of performing a system restart. Check if all nodes
+ // have saved the new table description to file and then continue with the
+ // next table.
+ /* --------------------------------------------------------------------- */
+ receiveLoopMacro(COPY_TABREQ, nodePtr.i);
+ /* --------------------------------------------------------------------- */
+ /* WE HAVE NOW COPIED TO ALL NODES. WE HAVE NOW COMPLETED RESTORING */
+ /* THIS TABLE. CONTINUE WITH THE NEXT TABLE. */
+ /* WE NEED TO RELEASE THE PAGES IN THE TABLE IN THIS NODE HERE. */
+ /* WE ALSO NEED TO CLOSE THE TABLE FILE. */
+ /* --------------------------------------------------------------------- */
+ releaseTabPages(tableId);
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ ConnectRecordPtr connectPtr;
+ connectPtr.i = tabPtr.p->connectrec;
+ ptrCheckGuard(connectPtr, cconnectFileSize, connectRecord);
+
+ sendAddFragreq(signal, connectPtr, tabPtr, 0);
+ return;
+ }//if
+}//Dbdih::execCOPY_TABCONF()
+
+/*
+ 3.13 L O C A L C H E C K P O I N T (M A S T E R)
+ ****************************************************
+ */
+/*****************************************************************************/
+/* ********** LOCAL-CHECK-POINT-HANDLING MODULE *************/
+/*****************************************************************************/
+/* ------------------------------------------------------------------------- */
+/* IT IS TIME TO CHECK IF IT IS TIME TO START A LOCAL CHECKPOINT. */
+/* WE WILL EITHER START AFTER 1 MILLION WORDS HAVE ARRIVED OR WE WILL */
+/* EXECUTE AFTER ABOUT 16 MINUTES HAVE PASSED BY. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::checkTcCounterLab(Signal* signal)
+{
+ CRASH_INSERTION(7009);
+ if (c_lcpState.lcpStatus != LCP_STATUS_IDLE) {
+ ndbout << "lcpStatus = " << c_lcpState.lcpStatus;
+ ndbout << "lcpStatusUpdatedPlace = " <<
+ c_lcpState.lcpStatusUpdatedPlace << endl;
+ ndbrequire(false);
+ return;
+ }//if
+ c_lcpState.ctimer += 32;
+ if ((c_nodeStartMaster.blockLcp == true) ||
+ ((c_lcpState.lcpStartGcp + 1) > currentgcp)) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ // No reason to start juggling the states and checking for start of LCP if
+ // we are blocked to start an LCP anyway.
+ // We also block LCP start if we have not completed one global checkpoints
+ // before starting another local checkpoint.
+ /* --------------------------------------------------------------------- */
+ signal->theData[0] = DihContinueB::ZCHECK_TC_COUNTER;
+ signal->theData[1] = __LINE__;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 1 * 100, 2);
+ return;
+ }//if
+ c_lcpState.setLcpStatus(LCP_TCGET, __LINE__);
+
+ c_lcpState.ctcCounter = c_lcpState.ctimer;
+ sendLoopMacro(TCGETOPSIZEREQ, sendTCGETOPSIZEREQ);
+}//Dbdih::checkTcCounterLab()
+
+void Dbdih::checkLcpStart(Signal* signal, Uint32 lineNo)
+{
+ /* ----------------------------------------------------------------------- */
+ // Verify that we are not attempting to start another instance of the LCP
+ // when it is not alright to do so.
+ /* ----------------------------------------------------------------------- */
+ ndbrequire(c_lcpState.lcpStart == ZIDLE);
+ c_lcpState.lcpStart = ZACTIVE;
+ signal->theData[0] = DihContinueB::ZCHECK_TC_COUNTER;
+ signal->theData[1] = lineNo;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 1000, 2);
+}//Dbdih::checkLcpStart()
+
+/* ------------------------------------------------------------------------- */
+/*TCGETOPSIZECONF HOW MUCH OPERATION SIZE HAVE BEEN EXECUTED BY TC */
+/* ------------------------------------------------------------------------- */
+void Dbdih::execTCGETOPSIZECONF(Signal* signal)
+{
+ jamEntry();
+ Uint32 senderNodeId = signal->theData[0];
+ c_lcpState.ctcCounter += signal->theData[1];
+
+ receiveLoopMacro(TCGETOPSIZEREQ, senderNodeId);
+
+ ndbrequire(c_lcpState.lcpStatus == LCP_TCGET);
+ ndbrequire(c_lcpState.lcpStart == ZACTIVE);
+ /* ----------------------------------------------------------------------- */
+ // We are not actively starting another LCP, still we receive this signal.
+ // This is not ok.
+ /* ---------------------------------------------------------------------- */
+ /* ALL TC'S HAVE RESPONDED NOW. NOW WE WILL CHECK IF ENOUGH OPERATIONS */
+ /* HAVE EXECUTED TO ENABLE US TO START A NEW LOCAL CHECKPOINT. */
+ /* WHILE COPYING DICTIONARY AND DISTRIBUTION INFO TO A STARTING NODE */
+ /* WE WILL ALSO NOT ALLOW THE LOCAL CHECKPOINT TO PROCEED. */
+ /*----------------------------------------------------------------------- */
+ if (c_lcpState.immediateLcpStart == false) {
+ if ((c_lcpState.ctcCounter <
+ ((Uint32)1 << c_lcpState.clcpDelay)) ||
+ (c_nodeStartMaster.blockLcp == true)) {
+ jam();
+ c_lcpState.setLcpStatus(LCP_STATUS_IDLE, __LINE__);
+
+ signal->theData[0] = DihContinueB::ZCHECK_TC_COUNTER;
+ signal->theData[1] = __LINE__;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 1 * 100, 2);
+ return;
+ }//if
+ }//if
+ c_lcpState.lcpStart = ZIDLE;
+ c_lcpState.immediateLcpStart = false;
+ /* -----------------------------------------------------------------------
+ * Now the initial lcp is started,
+ * we can reset the delay to its orginal value
+ * --------------------------------------------------------------------- */
+ CRASH_INSERTION(7010);
+ /* ----------------------------------------------------------------------- */
+ /* IF MORE THAN 1 MILLION WORDS PASSED THROUGH THE TC'S THEN WE WILL */
+ /* START A NEW LOCAL CHECKPOINT. CLEAR CTIMER. START CHECKPOINT */
+ /* ACTIVITY BY CALCULATING THE KEEP GLOBAL CHECKPOINT. */
+ // Also remember the current global checkpoint to ensure that we run at least
+ // one global checkpoints between each local checkpoint that we start up.
+ /* ----------------------------------------------------------------------- */
+ c_lcpState.ctimer = 0;
+ c_lcpState.keepGci = coldgcp;
+ c_lcpState.lcpStartGcp = currentgcp;
+ /* ----------------------------------------------------------------------- */
+ /* UPDATE THE NEW LATEST LOCAL CHECKPOINT ID. */
+ /* ----------------------------------------------------------------------- */
+ cnoOfActiveTables = 0;
+ c_lcpState.setLcpStatus(LCP_CALCULATE_KEEP_GCI, __LINE__);
+ c_lcpState.oldestRestorableGci = SYSFILE->oldestRestorableGCI;
+ ndbrequire(((int)c_lcpState.oldestRestorableGci) > 0);
+
+ if (ERROR_INSERTED(7011)) {
+ signal->theData[0] = EventReport::LCPStoppedInCalcKeepGci;
+ signal->theData[1] = 0;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+ return;
+ }//if
+ signal->theData[0] = DihContinueB::ZCALCULATE_KEEP_GCI;
+ signal->theData[1] = 0; /* TABLE ID = 0 */
+ signal->theData[2] = 0; /* FRAGMENT ID = 0 */
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+}//Dbdih::execTCGETOPSIZECONF()
+
+/* ------------------------------------------------------------------------- */
+/* WE NEED TO CALCULATE THE OLDEST GLOBAL CHECKPOINT THAT WILL BE */
+/* COMPLETELY RESTORABLE AFTER EXECUTING THIS LOCAL CHECKPOINT. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::calculateKeepGciLab(Signal* signal, Uint32 tableId, Uint32 fragId)
+{
+ TabRecordPtr tabPtr;
+ Uint32 TloopCount = 1;
+ tabPtr.i = tableId;
+ do {
+ if (tabPtr.i >= ctabFileSize) {
+ if (cnoOfActiveTables > 0) {
+ jam();
+ signal->theData[0] = DihContinueB::ZSTORE_NEW_LCP_ID;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 1, JBB);
+ return;
+ } else {
+ jam();
+ /* ------------------------------------------------------------------ */
+ /* THERE ARE NO TABLES TO CHECKPOINT. WE STOP THE CHECKPOINT ALREADY */
+ /* HERE TO AVOID STRANGE PROBLEMS LATER. */
+ /* ------------------------------------------------------------------ */
+ c_lcpState.setLcpStatus(LCP_STATUS_IDLE, __LINE__);
+ checkLcpStart(signal, __LINE__);
+ return;
+ }//if
+ }//if
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ if (tabPtr.p->tabStatus != TabRecord::TS_ACTIVE ||
+ tabPtr.p->storedTable == 0) {
+ if (TloopCount > 100) {
+ jam();
+ signal->theData[0] = DihContinueB::ZCALCULATE_KEEP_GCI;
+ signal->theData[1] = tabPtr.i + 1;
+ signal->theData[2] = 0;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ } else {
+ jam();
+ TloopCount++;
+ tabPtr.i++;
+ }//if
+ } else {
+ jam();
+ TloopCount = 0;
+ }//if
+ } while (TloopCount != 0);
+ cnoOfActiveTables++;
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ checkKeepGci(fragPtr.p->storedReplicas);
+ fragId++;
+ if (fragId >= tabPtr.p->totalfragments) {
+ jam();
+ tabPtr.i++;
+ fragId = 0;
+ }//if
+ signal->theData[0] = DihContinueB::ZCALCULATE_KEEP_GCI;
+ signal->theData[1] = tabPtr.i;
+ signal->theData[2] = fragId;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+}//Dbdih::calculateKeepGciLab()
+
+/* ------------------------------------------------------------------------- */
+/* WE NEED TO STORE ON DISK THE FACT THAT WE ARE STARTING THIS LOCAL */
+/* CHECKPOINT ROUND. THIS WILL INVALIDATE ALL THE LOCAL CHECKPOINTS */
+/* THAT WILL EVENTUALLY BE OVERWRITTEN AS PART OF THIS LOCAL CHECKPOINT*/
+/* ------------------------------------------------------------------------- */
+void Dbdih::storeNewLcpIdLab(Signal* signal)
+{
+ /***************************************************************************/
+ // Report the event that a local checkpoint has started.
+ /***************************************************************************/
+ signal->theData[0] = EventReport::LocalCheckpointStarted; //Event type
+ signal->theData[1] = SYSFILE->latestLCP_ID + 1;
+ signal->theData[2] = c_lcpState.keepGci;
+ signal->theData[3] = c_lcpState.oldestRestorableGci;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+
+ signal->setTrace(TestOrd::TraceLocalCheckpoint);
+
+ CRASH_INSERTION(7013);
+ SYSFILE->keepGCI = c_lcpState.keepGci;
+ //Uint32 lcpId = SYSFILE->latestLCP_ID;
+ SYSFILE->latestLCP_ID++;
+ SYSFILE->oldestRestorableGCI = c_lcpState.oldestRestorableGci;
+
+ const Uint32 oldestRestorableGCI = SYSFILE->oldestRestorableGCI;
+ //const Uint32 newestRestorableGCI = SYSFILE->newestRestorableGCI;
+ //ndbrequire(newestRestorableGCI >= oldestRestorableGCI);
+
+ Int32 val = oldestRestorableGCI;
+ ndbrequire(val > 0);
+
+ /* ----------------------------------------------------------------------- */
+ /* SET BIT INDICATING THAT LOCAL CHECKPOINT IS ONGOING. THIS IS CLEARED */
+ /* AT THE END OF A LOCAL CHECKPOINT. */
+ /* ----------------------------------------------------------------------- */
+ SYSFILE->setLCPOngoing(SYSFILE->systemRestartBits);
+ /* ---------------------------------------------------------------------- */
+ /* CHECK IF ANY NODE MUST BE TAKEN OUT OF SERVICE AND REFILLED WITH */
+ /* NEW FRESH DATA FROM AN ACTIVE NODE. */
+ /* ---------------------------------------------------------------------- */
+ setLcpActiveStatusStart(signal);
+ c_lcpState.setLcpStatus(LCP_COPY_GCI, __LINE__);
+ //#ifdef VM_TRACE
+ // infoEvent("LocalCheckpoint %d started", SYSFILE->latestLCP_ID);
+ // signal->theData[0] = 7012;
+ // execDUMP_STATE_ORD(signal);
+ //#endif
+
+ copyGciLab(signal, CopyGCIReq::LOCAL_CHECKPOINT);
+}//Dbdih::storeNewLcpIdLab()
+
+void Dbdih::startLcpRoundLab(Signal* signal) {
+ jam();
+
+ Mutex mutex(signal, c_mutexMgr, c_startLcpMutexHandle);
+ Callback c = { safe_cast(&Dbdih::startLcpMutex_locked), 0 };
+ ndbrequire(mutex.lock(c));
+}
+
+void
+Dbdih::startLcpMutex_locked(Signal* signal, Uint32 senderData, Uint32 retVal){
+ jamEntry();
+ ndbrequire(retVal == 0);
+
+ StartLcpReq* req = (StartLcpReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->lcpId = SYSFILE->latestLCP_ID;
+ req->participatingLQH = c_lcpState.m_participatingLQH;
+ req->participatingDIH = c_lcpState.m_participatingDIH;
+ sendLoopMacro(START_LCP_REQ, sendSTART_LCP_REQ);
+}
+void
+Dbdih::sendSTART_LCP_REQ(Signal* signal, Uint32 nodeId){
+ BlockReference ref = calcDihBlockRef(nodeId);
+ sendSignal(ref, GSN_START_LCP_REQ, signal, StartLcpReq::SignalLength, JBB);
+}
+
+void
+Dbdih::execSTART_LCP_CONF(Signal* signal){
+ StartLcpConf * conf = (StartLcpConf*)signal->getDataPtr();
+
+ Uint32 nodeId = refToNode(conf->senderRef);
+ receiveLoopMacro(START_LCP_REQ, nodeId);
+
+ Mutex mutex(signal, c_mutexMgr, c_startLcpMutexHandle);
+ Callback c = { safe_cast(&Dbdih::startLcpMutex_unlocked), 0 };
+ mutex.unlock(c);
+}
+
+void
+Dbdih::startLcpMutex_unlocked(Signal* signal, Uint32 data, Uint32 retVal){
+ jamEntry();
+ ndbrequire(retVal == 0);
+
+ Mutex mutex(signal, c_mutexMgr, c_startLcpMutexHandle);
+ mutex.release();
+
+ CRASH_INSERTION(7014);
+ c_lcpState.setLcpStatus(LCP_TC_CLOPSIZE, __LINE__);
+ sendLoopMacro(TC_CLOPSIZEREQ, sendTC_CLOPSIZEREQ);
+}
+
+void Dbdih::execTC_CLOPSIZECONF(Signal* signal) {
+ jamEntry();
+ Uint32 senderNodeId = signal->theData[0];
+ receiveLoopMacro(TC_CLOPSIZEREQ, senderNodeId);
+
+ ndbrequire(c_lcpState.lcpStatus == LCP_TC_CLOPSIZE);
+ /* ----------------------------------------------------------------------- */
+ /* ALL TC'S HAVE CLEARED THEIR OPERATION SIZE COUNTERS. NOW PROCEED BY */
+ /* STARTING THE LOCAL CHECKPOINT IN EACH LQH. */
+ /* ----------------------------------------------------------------------- */
+ c_lcpState.m_LAST_LCP_FRAG_ORD = c_lcpState.m_participatingLQH;
+
+ CRASH_INSERTION(7015);
+ c_lcpState.setLcpStatus(LCP_START_LCP_ROUND, __LINE__);
+ startLcpRoundLoopLab(signal, 0, 0);
+}//Dbdih::execTC_CLOPSIZECONF()
+
+void Dbdih::startLcpRoundLoopLab(Signal* signal,
+ Uint32 startTableId, Uint32 startFragId)
+{
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRecord);
+ if (nodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ ndbrequire(nodePtr.p->noOfStartedChkpt == 0);
+ ndbrequire(nodePtr.p->noOfQueuedChkpt == 0);
+ }//if
+ }//if
+ c_lcpState.currentFragment.tableId = startTableId;
+ c_lcpState.currentFragment.fragmentId = startFragId;
+ startNextChkpt(signal);
+}//Dbdih::startLcpRoundLoopLab()
+
+void Dbdih::startNextChkpt(Signal* signal)
+{
+ Uint32 lcpId = SYSFILE->latestLCP_ID;
+
+ NdbNodeBitmask busyNodes;
+ busyNodes.clear();
+ const Uint32 lcpNodes = c_lcpState.m_participatingLQH.count();
+
+ bool save = true;
+ LcpState::CurrentFragment curr = c_lcpState.currentFragment;
+
+ while (curr.tableId < ctabFileSize) {
+ TabRecordPtr tabPtr;
+ tabPtr.i = curr.tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ if ((tabPtr.p->tabStatus != TabRecord::TS_ACTIVE) ||
+ (tabPtr.p->tabLcpStatus != TabRecord::TLS_ACTIVE)) {
+ curr.tableId++;
+ curr.fragmentId = 0;
+ continue;
+ }//if
+
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, curr.fragmentId, fragPtr);
+
+ ReplicaRecordPtr replicaPtr;
+ for(replicaPtr.i = fragPtr.p->storedReplicas;
+ replicaPtr.i != RNIL ;
+ replicaPtr.i = replicaPtr.p->nextReplica){
+
+ jam();
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+
+ NodeRecordPtr nodePtr;
+ nodePtr.i = replicaPtr.p->procNode;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+
+ if (replicaPtr.p->lcpOngoingFlag &&
+ replicaPtr.p->lcpIdStarted < lcpId) {
+ jam();
+ //-------------------------------------------------------------------
+ // We have found a replica on a node that performs local checkpoint
+ // that is alive and that have not yet been started.
+ //-------------------------------------------------------------------
+
+ if (nodePtr.p->noOfStartedChkpt < 2) {
+ jam();
+ /**
+ * Send LCP_FRAG_ORD to LQH
+ */
+
+ /**
+ * Mark the replica so with lcpIdStarted == true
+ */
+ replicaPtr.p->lcpIdStarted = lcpId;
+
+ Uint32 i = nodePtr.p->noOfStartedChkpt;
+ nodePtr.p->startedChkpt[i].tableId = tabPtr.i;
+ nodePtr.p->startedChkpt[i].fragId = curr.fragmentId;
+ nodePtr.p->startedChkpt[i].replicaPtr = replicaPtr.i;
+ nodePtr.p->noOfStartedChkpt = i + 1;
+
+ sendLCP_FRAG_ORD(signal, nodePtr.p->startedChkpt[i]);
+ } else if (nodePtr.p->noOfQueuedChkpt < 2) {
+ jam();
+ /**
+ * Put LCP_FRAG_ORD "in queue"
+ */
+
+ /**
+ * Mark the replica so with lcpIdStarted == true
+ */
+ replicaPtr.p->lcpIdStarted = lcpId;
+
+ Uint32 i = nodePtr.p->noOfQueuedChkpt;
+ nodePtr.p->queuedChkpt[i].tableId = tabPtr.i;
+ nodePtr.p->queuedChkpt[i].fragId = curr.fragmentId;
+ nodePtr.p->queuedChkpt[i].replicaPtr = replicaPtr.i;
+ nodePtr.p->noOfQueuedChkpt = i + 1;
+ } else {
+ jam();
+
+ if(save){
+ /**
+ * Stop increasing value on first that was "full"
+ */
+ c_lcpState.currentFragment = curr;
+ save = false;
+ }
+
+ busyNodes.set(nodePtr.i);
+ if(busyNodes.count() == lcpNodes){
+ /**
+ * There were no possibility to start the local checkpoint
+ * and it was not possible to queue it up. In this case we
+ * stop the start of local checkpoints until the nodes with a
+ * backlog have performed more checkpoints. We will return and
+ * will not continue the process of starting any more checkpoints.
+ */
+ return;
+ }//if
+ }//if
+ }
+ }//while
+ curr.fragmentId++;
+ if (curr.fragmentId >= tabPtr.p->totalfragments) {
+ jam();
+ curr.fragmentId = 0;
+ curr.tableId++;
+ }//if
+ }//while
+
+ sendLastLCP_FRAG_ORD(signal);
+}//Dbdih::startNextChkpt()
+
+void Dbdih::sendLastLCP_FRAG_ORD(Signal* signal)
+{
+ LcpFragOrd * const lcpFragOrd = (LcpFragOrd *)&signal->theData[0];
+ lcpFragOrd->tableId = RNIL;
+ lcpFragOrd->fragmentId = 0;
+ lcpFragOrd->lcpId = SYSFILE->latestLCP_ID;
+ lcpFragOrd->lcpNo = 0;
+ lcpFragOrd->keepGci = c_lcpState.keepGci;
+ lcpFragOrd->lastFragmentFlag = true;
+
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+
+ if(nodePtr.p->noOfQueuedChkpt == 0 &&
+ nodePtr.p->noOfStartedChkpt == 0 &&
+ c_lcpState.m_LAST_LCP_FRAG_ORD.isWaitingFor(nodePtr.i)){
+ jam();
+
+ CRASH_INSERTION(7028);
+
+ /**
+ * Nothing queued or started <=> Complete on that node
+ *
+ */
+ c_lcpState.m_LAST_LCP_FRAG_ORD.clearWaitingFor(nodePtr.i);
+ if(ERROR_INSERTED(7075)){
+ continue;
+ }
+ BlockReference ref = calcLqhBlockRef(nodePtr.i);
+ sendSignal(ref, GSN_LCP_FRAG_ORD, signal,LcpFragOrd::SignalLength, JBB);
+ }
+ }
+ if(ERROR_INSERTED(7075)){
+ if(c_lcpState.m_LAST_LCP_FRAG_ORD.done())
+ CRASH_INSERTION(7075);
+ }
+}//Dbdih::sendLastLCP_FRAGORD()
+
+/* ------------------------------------------------------------------------- */
+/* A FRAGMENT REPLICA HAS COMPLETED EXECUTING ITS LOCAL CHECKPOINT. */
+/* CHECK IF ALL REPLICAS IN THE TABLE HAVE COMPLETED. IF SO STORE THE */
+/* THE TABLE DISTRIBUTION ON DISK. ALSO SEND LCP_REPORT TO ALL OTHER */
+/* NODES SO THAT THEY CAN STORE THE TABLE ONTO DISK AS WELL. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::execLCP_FRAG_REP(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(c_lcpState.lcpStatus != LCP_STATUS_IDLE);
+
+#if 0
+ printLCP_FRAG_REP(stdout,
+ signal->getDataPtr(),
+ signal->length(), number());
+#endif
+
+ LcpFragRep * const lcpReport = (LcpFragRep *)&signal->theData[0];
+ Uint32 nodeId = lcpReport->nodeId;
+ Uint32 tableId = lcpReport->tableId;
+ Uint32 fragId = lcpReport->fragId;
+
+ jamEntry();
+
+ CRASH_INSERTION2(7025, isMaster());
+ CRASH_INSERTION2(7016, !isMaster());
+
+ bool fromTimeQueue = (signal->senderBlockRef() == reference());
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ if(tabPtr.p->tabCopyStatus != TabRecord::CS_IDLE) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ // If the table is currently copied to disk we also
+ // stop already here to avoid strange half-way updates
+ // of the table data structures.
+ /*-----------------------------------------------------------------------*/
+ /*
+ We need to send this signal without a delay since we have discovered
+ that we have run out of space in the short time queue. This problem
+ is very erunlikely to happen but it has and it results in a node crash.
+ This should be considered a "quick fix" and not a permanent solution.
+ A cleaner/better way would be to check the time queue if it is full or
+ not before sending this signal.
+ */
+ sendSignal(reference(), GSN_LCP_FRAG_REP, signal, signal->length(), JBB);
+ /* Kept here for reference
+ sendSignalWithDelay(reference(), GSN_LCP_FRAG_REP,
+ signal, 20, signal->length());
+ */
+
+ if(!fromTimeQueue){
+ c_lcpState.noOfLcpFragRepOutstanding++;
+ }
+
+ return;
+ }//if
+
+ if(fromTimeQueue){
+ jam();
+
+ ndbrequire(c_lcpState.noOfLcpFragRepOutstanding > 0);
+ c_lcpState.noOfLcpFragRepOutstanding--;
+ }
+
+ bool tableDone = reportLcpCompletion(lcpReport);
+
+ if(tableDone){
+ jam();
+
+ if(tabPtr.p->tabStatus == TabRecord::TS_DROPPING){
+ jam();
+ ndbout_c("TS_DROPPING - Neglecting to save Table: %d Frag: %d - ",
+ tableId,
+ fragId);
+ } else {
+ jam();
+ /**
+ * Write table description to file
+ */
+ tabPtr.p->tabLcpStatus = TabRecord::TLS_WRITING_TO_FILE;
+ tabPtr.p->tabCopyStatus = TabRecord::CS_LCP_READ_TABLE;
+ tabPtr.p->tabUpdateState = TabRecord::US_LOCAL_CHECKPOINT;
+ signal->theData[0] = DihContinueB::ZPACK_TABLE_INTO_PAGES;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+
+ checkLcpAllTablesDoneInLqh();
+ }
+ }
+
+#ifdef VM_TRACE
+ /* --------------------------------------------------------------------- */
+ // REPORT that local checkpoint have completed this fragment.
+ /* --------------------------------------------------------------------- */
+ signal->theData[0] = EventReport::LCPFragmentCompleted;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tableId;
+ signal->theData[3] = fragId;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+#endif
+
+ bool ok = false;
+ switch(c_lcpMasterTakeOverState.state){
+ case LMTOS_IDLE:
+ ok = true;
+ jam();
+ /**
+ * Fall through
+ */
+ break;
+ case LMTOS_WAIT_EMPTY_LCP: // LCP Take over waiting for EMPTY_LCPCONF
+ jam();
+ return;
+ case LMTOS_WAIT_LCP_FRAG_REP:
+ jam();
+ checkEmptyLcpComplete(signal);
+ return;
+ case LMTOS_INITIAL:
+ case LMTOS_ALL_IDLE:
+ case LMTOS_ALL_ACTIVE:
+ case LMTOS_LCP_CONCLUDING:
+ case LMTOS_COPY_ONGOING:
+ ndbrequire(false);
+ }
+ ndbrequire(ok);
+
+ /* ----------------------------------------------------------------------- */
+ // Check if there are more LCP's to start up.
+ /* ----------------------------------------------------------------------- */
+ if(isMaster()){
+ jam();
+
+ /**
+ * Remove from "running" array
+ */
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+
+ const Uint32 outstanding = nodePtr.p->noOfStartedChkpt;
+ ndbrequire(outstanding > 0);
+ if(nodePtr.p->startedChkpt[0].tableId != tableId ||
+ nodePtr.p->startedChkpt[0].fragId != fragId){
+ jam();
+ ndbrequire(outstanding > 1);
+ ndbrequire(nodePtr.p->startedChkpt[1].tableId == tableId);
+ ndbrequire(nodePtr.p->startedChkpt[1].fragId == fragId);
+ } else {
+ jam();
+ nodePtr.p->startedChkpt[0] = nodePtr.p->startedChkpt[1];
+ }
+ nodePtr.p->noOfStartedChkpt--;
+ checkStartMoreLcp(signal, nodeId);
+ }
+}
+
+bool
+Dbdih::checkLcpAllTablesDoneInLqh(){
+ TabRecordPtr tabPtr;
+
+ /**
+ * Check if finished with all tables
+ */
+ for (tabPtr.i = 0; tabPtr.i < ctabFileSize; tabPtr.i++) {
+ jam();
+ ptrAss(tabPtr, tabRecord);
+ if ((tabPtr.p->tabStatus == TabRecord::TS_ACTIVE) &&
+ (tabPtr.p->tabLcpStatus == TabRecord::TLS_ACTIVE)) {
+ jam();
+ /**
+ * Nope, not finished with all tables
+ */
+ return false;
+ }//if
+ }//for
+
+ CRASH_INSERTION2(7026, isMaster());
+ CRASH_INSERTION2(7017, !isMaster());
+
+ c_lcpState.setLcpStatus(LCP_TAB_COMPLETED, __LINE__);
+ return true;
+}
+
+void Dbdih::findReplica(ReplicaRecordPtr& replicaPtr,
+ Fragmentstore* fragPtrP, Uint32 nodeId)
+{
+ replicaPtr.i = fragPtrP->storedReplicas;
+ while(replicaPtr.i != RNIL){
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ if (replicaPtr.p->procNode == nodeId) {
+ jam();
+ return;
+ } else {
+ jam();
+ replicaPtr.i = replicaPtr.p->nextReplica;
+ }//if
+ };
+
+#ifdef VM_TRACE
+ ndbout_c("Fragment Replica(node=%d) not found", nodeId);
+ replicaPtr.i = fragPtrP->oldStoredReplicas;
+ while(replicaPtr.i != RNIL){
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ if (replicaPtr.p->procNode == nodeId) {
+ jam();
+ break;
+ } else {
+ jam();
+ replicaPtr.i = replicaPtr.p->nextReplica;
+ }//if
+ };
+ if(replicaPtr.i != RNIL){
+ ndbout_c("...But was found in oldStoredReplicas");
+ } else {
+ ndbout_c("...And wasn't found in oldStoredReplicas");
+ }
+#endif
+ ndbrequire(false);
+}//Dbdih::findReplica()
+
+/**
+ * Return true if table is all fragment replicas have been checkpointed
+ * to disk (in all LQHs)
+ * false otherwise
+ */
+bool
+Dbdih::reportLcpCompletion(const LcpFragRep* lcpReport)
+{
+ Uint32 lcpNo = lcpReport->lcpNo;
+ Uint32 lcpId = lcpReport->lcpId;
+ Uint32 maxGciStarted = lcpReport->maxGciStarted;
+ Uint32 maxGciCompleted = lcpReport->maxGciCompleted;
+ Uint32 tableId = lcpReport->tableId;
+ Uint32 fragId = lcpReport->fragId;
+ Uint32 nodeId = lcpReport->nodeId;
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+
+ ReplicaRecordPtr replicaPtr;
+ findReplica(replicaPtr, fragPtr.p, nodeId);
+
+ ndbrequire(replicaPtr.p->lcpOngoingFlag == true);
+ if(lcpNo != replicaPtr.p->nextLcp){
+ ndbout_c("lcpNo = %d replicaPtr.p->nextLcp = %d",
+ lcpNo, replicaPtr.p->nextLcp);
+ ndbrequire(false);
+ }
+ ndbrequire(lcpNo == replicaPtr.p->nextLcp);
+ ndbrequire(lcpNo < MAX_LCP_STORED);
+ ndbrequire(replicaPtr.p->lcpId[lcpNo] != lcpId);
+
+ replicaPtr.p->lcpIdStarted = lcpId;
+ replicaPtr.p->lcpOngoingFlag = false;
+
+ removeOldCrashedReplicas(replicaPtr);
+ replicaPtr.p->lcpId[lcpNo] = lcpId;
+ replicaPtr.p->lcpStatus[lcpNo] = ZVALID;
+ replicaPtr.p->maxGciStarted[lcpNo] = maxGciStarted;
+ gth(maxGciStarted + 1, 0);
+ replicaPtr.p->maxGciCompleted[lcpNo] = maxGciCompleted;
+ replicaPtr.p->nextLcp = nextLcpNo(replicaPtr.p->nextLcp);
+
+ ndbrequire(fragPtr.p->noLcpReplicas > 0);
+ fragPtr.p->noLcpReplicas --;
+
+ if(fragPtr.p->noLcpReplicas > 0){
+ jam();
+ return false;
+ }
+
+ for (Uint32 fid = 0; fid < tabPtr.p->totalfragments; fid++) {
+ jam();
+ getFragstore(tabPtr.p, fid, fragPtr);
+ if (fragPtr.p->noLcpReplicas > 0){
+ jam();
+ /* ----------------------------------------------------------------- */
+ // Not all fragments in table have been checkpointed.
+ /* ----------------------------------------------------------------- */
+ if(0)
+ ndbout_c("reportLcpCompletion: fragment %d not ready", fid);
+ return false;
+ }//if
+ }//for
+ return true;
+}//Dbdih::reportLcpCompletion()
+
+void Dbdih::checkStartMoreLcp(Signal* signal, Uint32 nodeId)
+{
+ ndbrequire(isMaster());
+
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+
+ ndbrequire(nodePtr.p->noOfStartedChkpt < 2);
+
+ if (nodePtr.p->noOfQueuedChkpt > 0) {
+ jam();
+ nodePtr.p->noOfQueuedChkpt--;
+ Uint32 i = nodePtr.p->noOfStartedChkpt;
+ nodePtr.p->startedChkpt[i] = nodePtr.p->queuedChkpt[0];
+ nodePtr.p->queuedChkpt[0] = nodePtr.p->queuedChkpt[1];
+ //-------------------------------------------------------------------
+ // We can send a LCP_FRAGORD to the node ordering it to perform a
+ // local checkpoint on this fragment replica.
+ //-------------------------------------------------------------------
+ nodePtr.p->noOfStartedChkpt = i + 1;
+
+ sendLCP_FRAG_ORD(signal, nodePtr.p->startedChkpt[i]);
+ }
+
+ /* ----------------------------------------------------------------------- */
+ // When there are no more outstanding LCP reports and there are no one queued
+ // in at least one node, then we are ready to make sure all nodes have at
+ // least two outstanding LCP requests per node and at least two queued for
+ // sending.
+ /* ----------------------------------------------------------------------- */
+ startNextChkpt(signal);
+}//Dbdih::checkStartMoreLcp()
+
+void
+Dbdih::sendLCP_FRAG_ORD(Signal* signal,
+ NodeRecord::FragmentCheckpointInfo info){
+
+ ReplicaRecordPtr replicaPtr;
+ replicaPtr.i = info.replicaPtr;
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+
+ BlockReference ref = calcLqhBlockRef(replicaPtr.p->procNode);
+
+ LcpFragOrd * const lcpFragOrd = (LcpFragOrd *)&signal->theData[0];
+ lcpFragOrd->tableId = info.tableId;
+ lcpFragOrd->fragmentId = info.fragId;
+ lcpFragOrd->lcpId = SYSFILE->latestLCP_ID;
+ lcpFragOrd->lcpNo = replicaPtr.p->nextLcp;
+ lcpFragOrd->keepGci = c_lcpState.keepGci;
+ lcpFragOrd->lastFragmentFlag = false;
+ sendSignal(ref, GSN_LCP_FRAG_ORD, signal, LcpFragOrd::SignalLength, JBB);
+}
+
+void Dbdih::checkLcpCompletedLab(Signal* signal)
+{
+ if(c_lcpState.lcpStatus < LCP_TAB_COMPLETED){
+ jam();
+ return;
+ }
+
+ TabRecordPtr tabPtr;
+ for (tabPtr.i = 0; tabPtr.i < ctabFileSize; tabPtr.i++) {
+ jam();
+ ptrAss(tabPtr, tabRecord);
+ if (tabPtr.p->tabStatus == TabRecord::TS_ACTIVE) {
+ if (tabPtr.p->tabLcpStatus != TabRecord::TLS_COMPLETED) {
+ jam();
+ return;
+ }//if
+ }//if
+ }//for
+
+ CRASH_INSERTION2(7027, isMaster());
+ CRASH_INSERTION2(7018, !isMaster());
+
+ if(c_lcpState.lcpStatus == LCP_TAB_COMPLETED){
+ /**
+ * We'r done
+ */
+ c_lcpState.setLcpStatus(LCP_TAB_SAVED, __LINE__);
+ sendLCP_COMPLETE_REP(signal);
+ return;
+ }
+
+ ndbrequire(c_lcpState.lcpStatus == LCP_TAB_SAVED);
+ allNodesLcpCompletedLab(signal);
+ return;
+}//Dbdih::checkLcpCompletedLab()
+
+void
+Dbdih::sendLCP_COMPLETE_REP(Signal* signal){
+ jam();
+ LcpCompleteRep * rep = (LcpCompleteRep*)signal->getDataPtrSend();
+ rep->nodeId = getOwnNodeId();
+ rep->lcpId = SYSFILE->latestLCP_ID;
+ rep->blockNo = DBDIH;
+
+ sendSignal(c_lcpState.m_masterLcpDihRef, GSN_LCP_COMPLETE_REP, signal,
+ LcpCompleteRep::SignalLength, JBB);
+}
+
+/*-------------------------------------------------------------------------- */
+/* COMP_LCP_ROUND A LQH HAS COMPLETED A LOCAL CHECKPOINT */
+/*------------------------------------------------------------------------- */
+void Dbdih::execLCP_COMPLETE_REP(Signal* signal)
+{
+ jamEntry();
+
+#if 0
+ ndbout_c("LCP_COMPLETE_REP");
+ printLCP_COMPLETE_REP(stdout,
+ signal->getDataPtr(),
+ signal->length(), number());
+#endif
+
+ LcpCompleteRep * rep = (LcpCompleteRep*)signal->getDataPtr();
+ Uint32 lcpId = rep->lcpId;
+ Uint32 nodeId = rep->nodeId;
+ Uint32 blockNo = rep->blockNo;
+
+ if(c_lcpMasterTakeOverState.state > LMTOS_WAIT_LCP_FRAG_REP){
+ jam();
+ /**
+ * Don't allow LCP_COMPLETE_REP to arrive during
+ * LCP master take over
+ */
+ ndbrequire(isMaster());
+ ndbrequire(blockNo == DBDIH);
+ sendSignalWithDelay(reference(), GSN_LCP_COMPLETE_REP, signal, 100,
+ signal->length());
+ return;
+ }
+
+ ndbrequire(c_lcpState.lcpStatus != LCP_STATUS_IDLE);
+
+ switch(blockNo){
+ case DBLQH:
+ jam();
+ c_lcpState.m_LCP_COMPLETE_REP_Counter_LQH.clearWaitingFor(nodeId);
+ ndbrequire(!c_lcpState.m_LAST_LCP_FRAG_ORD.isWaitingFor(nodeId));
+ break;
+ case DBDIH:
+ jam();
+ ndbrequire(isMaster());
+ c_lcpState.m_LCP_COMPLETE_REP_Counter_DIH.clearWaitingFor(nodeId);
+ break;
+ case 0:
+ jam();
+ ndbrequire(!isMaster());
+ ndbrequire(c_lcpState.m_LCP_COMPLETE_REP_From_Master_Received == false);
+ c_lcpState.m_LCP_COMPLETE_REP_From_Master_Received = true;
+ break;
+ default:
+ ndbrequire(false);
+ }
+ ndbrequire(lcpId == SYSFILE->latestLCP_ID);
+
+ allNodesLcpCompletedLab(signal);
+ return;
+}
+
+void Dbdih::allNodesLcpCompletedLab(Signal* signal)
+{
+ jam();
+
+ if (c_lcpState.lcpStatus != LCP_TAB_SAVED) {
+ jam();
+ /**
+ * We have not sent LCP_COMPLETE_REP to master DIH yet
+ */
+ return;
+ }//if
+
+ if (!c_lcpState.m_LCP_COMPLETE_REP_Counter_LQH.done()){
+ jam();
+ return;
+ }
+
+ if (!c_lcpState.m_LCP_COMPLETE_REP_Counter_DIH.done()){
+ jam();
+ return;
+ }
+
+ if (!isMaster() &&
+ c_lcpState.m_LCP_COMPLETE_REP_From_Master_Received == false){
+ jam();
+ /**
+ * Wait until master DIH has signaled lcp is complete
+ */
+ return;
+ }
+
+ if(c_lcpMasterTakeOverState.state != LMTOS_IDLE){
+ jam();
+#ifdef VM_TRACE
+ ndbout_c("Exiting from allNodesLcpCompletedLab");
+#endif
+ return;
+ }
+
+
+ /*------------------------------------------------------------------------ */
+ /* WE HAVE NOW COMPLETED A LOCAL CHECKPOINT. WE ARE NOW READY TO WAIT */
+ /* FOR THE NEXT LOCAL CHECKPOINT. SEND WITHOUT TIME-OUT SINCE IT MIGHT */
+ /* BE TIME TO START THE NEXT LOCAL CHECKPOINT IMMEDIATELY. */
+ /* CLEAR BIT 3 OF SYSTEM RESTART BITS TO INDICATE THAT THERE IS NO */
+ /* LOCAL CHECKPOINT ONGOING. THIS WILL BE WRITTEN AT SOME LATER TIME */
+ /* DURING A GLOBAL CHECKPOINT. IT IS NOT NECESSARY TO WRITE IT */
+ /* IMMEDIATELY. WE WILL ALSO CLEAR BIT 2 OF SYSTEM RESTART BITS IF ALL */
+ /* CURRENTLY ACTIVE NODES COMPLETED THE LOCAL CHECKPOINT. */
+ /*------------------------------------------------------------------------ */
+ CRASH_INSERTION(7019);
+ signal->setTrace(0);
+
+ c_lcpState.setLcpStatus(LCP_STATUS_IDLE, __LINE__);
+ setLcpActiveStatusEnd();
+ Sysfile::clearLCPOngoing(SYSFILE->systemRestartBits);
+
+ if(!isMaster()){
+ jam();
+ /**
+ * We're not master, be content
+ */
+ return;
+ }
+
+ // Send LCP_COMPLETE_REP to all other nodes
+ // allowing them to set their lcpStatus to LCP_STATUS_IDLE
+ LcpCompleteRep * rep = (LcpCompleteRep*)signal->getDataPtrSend();
+ rep->nodeId = getOwnNodeId();
+ rep->lcpId = SYSFILE->latestLCP_ID;
+ rep->blockNo = 0; // 0 = Sent from master
+
+ NodeRecordPtr nodePtr;
+ nodePtr.i = cfirstAliveNode;
+ do {
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.i != cownNodeId){
+ BlockReference ref = calcDihBlockRef(nodePtr.i);
+ sendSignal(ref, GSN_LCP_COMPLETE_REP, signal,
+ LcpCompleteRep::SignalLength, JBB);
+ }
+ nodePtr.i = nodePtr.p->nextNode;
+ } while (nodePtr.i != RNIL);
+
+
+ jam();
+ /***************************************************************************/
+ // Report the event that a local checkpoint has completed.
+ /***************************************************************************/
+ signal->theData[0] = EventReport::LocalCheckpointCompleted; //Event type
+ signal->theData[1] = SYSFILE->latestLCP_ID;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ /**
+ * Start checking for next LCP
+ */
+ checkLcpStart(signal, __LINE__);
+
+ if (cwaitLcpSr == true) {
+ jam();
+ cwaitLcpSr = false;
+ ndbsttorry10Lab(signal, __LINE__);
+ return;
+ }//if
+
+ if (c_nodeStartMaster.blockLcp == true) {
+ jam();
+ lcpBlockedLab(signal);
+ return;
+ }//if
+ return;
+}//Dbdih::allNodesLcpCompletedLab()
+
+/******************************************************************************/
+/* ********** TABLE UPDATE MODULE *************/
+/* ****************************************************************************/
+/* ------------------------------------------------------------------------- */
+/* THIS MODULE IS USED TO UPDATE THE TABLE DESCRIPTION. IT STARTS BY */
+/* CREATING THE FIRST TABLE FILE, THEN UPDATES THIS FILE AND CLOSES IT.*/
+/* AFTER THAT THE SAME HAPPENS WITH THE SECOND FILE. AFTER THAT THE */
+/* TABLE DISTRIBUTION HAS BEEN UPDATED. */
+/* */
+/* THE REASON FOR CREATING THE FILE AND NOT OPENING IT IS TO ENSURE */
+/* THAT WE DO NOT GET A MIX OF OLD AND NEW INFORMATION IN THE FILE IN */
+/* ERROR SITUATIONS. */
+/* ------------------------------------------------------------------------- */
+void Dbdih::tableUpdateLab(Signal* signal, TabRecordPtr tabPtr) {
+ FileRecordPtr filePtr;
+ filePtr.i = tabPtr.p->tabFile[0];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ createFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::TABLE_CREATE;
+ return;
+}//Dbdih::tableUpdateLab()
+
+void Dbdih::tableCreateLab(Signal* signal, FileRecordPtr filePtr)
+{
+ TabRecordPtr tabPtr;
+ tabPtr.i = filePtr.p->tabRef;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ writeTabfile(signal, tabPtr.p, filePtr);
+ filePtr.p->reqStatus = FileRecord::TABLE_WRITE;
+ return;
+}//Dbdih::tableCreateLab()
+
+void Dbdih::tableWriteLab(Signal* signal, FileRecordPtr filePtr)
+{
+ closeFile(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::TABLE_CLOSE;
+ return;
+}//Dbdih::tableWriteLab()
+
+void Dbdih::tableCloseLab(Signal* signal, FileRecordPtr filePtr)
+{
+ TabRecordPtr tabPtr;
+ tabPtr.i = filePtr.p->tabRef;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ if (filePtr.i == tabPtr.p->tabFile[0]) {
+ jam();
+ filePtr.i = tabPtr.p->tabFile[1];
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ createFileRw(signal, filePtr);
+ filePtr.p->reqStatus = FileRecord::TABLE_CREATE;
+ return;
+ }//if
+ switch (tabPtr.p->tabUpdateState) {
+ case TabRecord::US_LOCAL_CHECKPOINT:
+ jam();
+ releaseTabPages(tabPtr.i);
+ signal->theData[0] = DihContinueB::ZCHECK_LCP_COMPLETED;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 1, JBB);
+
+ tabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ tabPtr.p->tabUpdateState = TabRecord::US_IDLE;
+ tabPtr.p->tabLcpStatus = TabRecord::TLS_COMPLETED;
+ return;
+ break;
+ case TabRecord::US_REMOVE_NODE:
+ jam();
+ releaseTabPages(tabPtr.i);
+ for (Uint32 fragId = 0; fragId < tabPtr.p->totalfragments; fragId++) {
+ jam();
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragId, fragPtr);
+ updateNodeInfo(fragPtr);
+ }//for
+ tabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ tabPtr.p->tabUpdateState = TabRecord::US_IDLE;
+ if (tabPtr.p->tabLcpStatus == TabRecord::TLS_WRITING_TO_FILE) {
+ jam();
+ tabPtr.p->tabLcpStatus = TabRecord::TLS_COMPLETED;
+ signal->theData[0] = DihContinueB::ZCHECK_LCP_COMPLETED;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 1, JBB);
+ }//if
+ signal->theData[0] = DihContinueB::ZREMOVE_NODE_FROM_TABLE;
+ signal->theData[1] = tabPtr.p->tabRemoveNode;
+ signal->theData[2] = tabPtr.i + 1;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ break;
+ case TabRecord::US_INVALIDATE_NODE_LCP:
+ jam();
+ releaseTabPages(tabPtr.i);
+ tabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ tabPtr.p->tabUpdateState = TabRecord::US_IDLE;
+
+ signal->theData[0] = DihContinueB::ZINVALIDATE_NODE_LCP;
+ signal->theData[1] = tabPtr.p->tabRemoveNode;
+ signal->theData[2] = tabPtr.i + 1;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ case TabRecord::US_COPY_TAB_REQ:
+ jam();
+ tabPtr.p->tabUpdateState = TabRecord::US_IDLE;
+ copyTabReq_complete(signal, tabPtr);
+ return;
+ break;
+ case TabRecord::US_ADD_TABLE_MASTER:
+ jam();
+ releaseTabPages(tabPtr.i);
+ tabPtr.p->tabUpdateState = TabRecord::US_IDLE;
+ signal->theData[0] = DihContinueB::ZDIH_ADD_TABLE_MASTER;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ case TabRecord::US_ADD_TABLE_SLAVE:
+ jam();
+ releaseTabPages(tabPtr.i);
+ tabPtr.p->tabUpdateState = TabRecord::US_IDLE;
+ signal->theData[0] = DihContinueB::ZDIH_ADD_TABLE_SLAVE;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ break;
+ }//switch
+}//Dbdih::tableCloseLab()
+
+/**
+ * GCP stop detected,
+ * send SYSTEM_ERROR to all other alive nodes
+ */
+void Dbdih::crashSystemAtGcpStop(Signal* signal){
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if (nodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ jam();
+ const BlockReference ref =
+ numberToRef(refToBlock(cntrlblockref), nodePtr.i);
+ SystemError * const sysErr = (SystemError*)&signal->theData[0];
+ sysErr->errorCode = SystemError::GCPStopDetected;
+ sysErr->errorRef = reference();
+ sysErr->data1 = cgcpStatus;
+ sysErr->data2 = cgcpOrderBlocked;
+ sendSignal(ref, GSN_SYSTEM_ERROR, signal,
+ SystemError::SignalLength, JBA);
+ }//if
+ }//for
+ return;
+}//Dbdih::crashSystemAtGcpStop()
+
+/*************************************************************************/
+/* */
+/* MODULE: ALLOCPAGE */
+/* DESCRIPTION: THE SUBROUTINE IS CALLED WITH POINTER TO PAGE */
+/* RECORD. A PAGE RECORD IS TAKEN FROM */
+/* THE FREE PAGE LIST */
+/*************************************************************************/
+void Dbdih::allocpage(PageRecordPtr& pagePtr)
+{
+ ndbrequire(cfirstfreepage != RNIL);
+ pagePtr.i = cfirstfreepage;
+ ptrCheckGuard(pagePtr, cpageFileSize, pageRecord);
+ cfirstfreepage = pagePtr.p->nextfreepage;
+ pagePtr.p->nextfreepage = RNIL;
+}//Dbdih::allocpage()
+
+/*************************************************************************/
+/* */
+/* MODULE: ALLOC_STORED_REPLICA */
+/* DESCRIPTION: THE SUBROUTINE IS CALLED TO GET A REPLICA RECORD, */
+/* TO INITIALISE IT AND TO LINK IT INTO THE FRAGMENT */
+/* STORE RECORD. USED FOR STORED REPLICAS. */
+/*************************************************************************/
+void Dbdih::allocStoredReplica(FragmentstorePtr fragPtr,
+ ReplicaRecordPtr& newReplicaPtr,
+ Uint32 nodeId)
+{
+ ReplicaRecordPtr arrReplicaPtr;
+ ReplicaRecordPtr arrPrevReplicaPtr;
+
+ seizeReplicaRec(newReplicaPtr);
+ for (Uint32 i = 0; i < MAX_LCP_STORED; i++) {
+ newReplicaPtr.p->maxGciCompleted[i] = 0;
+ newReplicaPtr.p->maxGciStarted[i] = 0;
+ newReplicaPtr.p->lcpId[i] = 0;
+ newReplicaPtr.p->lcpStatus[i] = ZINVALID;
+ }//for
+ newReplicaPtr.p->noCrashedReplicas = 0;
+ newReplicaPtr.p->initialGci = currentgcp;
+ for (Uint32 i = 0; i < 8; i++) {
+ newReplicaPtr.p->replicaLastGci[i] = (Uint32)-1;
+ newReplicaPtr.p->createGci[i] = 0;
+ }//for
+ newReplicaPtr.p->createGci[0] = currentgcp;
+ ndbrequire(currentgcp != 0xF1F1F1F1);
+ newReplicaPtr.p->nextLcp = 0;
+ newReplicaPtr.p->procNode = nodeId;
+ newReplicaPtr.p->lcpOngoingFlag = false;
+ newReplicaPtr.p->lcpIdStarted = 0;
+
+ arrPrevReplicaPtr.i = RNIL;
+ arrReplicaPtr.i = fragPtr.p->storedReplicas;
+ while (arrReplicaPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(arrReplicaPtr, creplicaFileSize, replicaRecord);
+ arrPrevReplicaPtr = arrReplicaPtr;
+ arrReplicaPtr.i = arrReplicaPtr.p->nextReplica;
+ }//while
+ if (arrPrevReplicaPtr.i == RNIL) {
+ jam();
+ fragPtr.p->storedReplicas = newReplicaPtr.i;
+ } else {
+ jam();
+ arrPrevReplicaPtr.p->nextReplica = newReplicaPtr.i;
+ }//if
+ fragPtr.p->noStoredReplicas++;
+}//Dbdih::allocStoredReplica()
+
+/*************************************************************************/
+/* CALCULATE HOW MANY HOT SPARES THAT ARE TO BE ASSIGNED IN THIS SYSTEM */
+/*************************************************************************/
+void Dbdih::calculateHotSpare()
+{
+ Uint32 tchsTmp;
+ Uint32 tchsNoNodes;
+
+ switch (cnoReplicas) {
+ case 1:
+ jam();
+ cnoHotSpare = 0;
+ break;
+ case 2:
+ case 3:
+ case 4:
+ jam();
+ if (csystemnodes > cnoReplicas) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* WITH MORE NODES THAN REPLICAS WE WILL ALWAYS USE AT LEAST ONE HOT */
+ /* SPARE IF THAT HAVE BEEN REQUESTED BY THE CONFIGURATION FILE. THE */
+ /* NUMBER OF NODES TO BE USED FOR NORMAL OPERATION IS ALWAYS */
+ /* A MULTIPLE OF THE NUMBER OF REPLICAS SINCE WE WILL ORGANISE NODES */
+ /* INTO NODE GROUPS. THE REMAINING NODES WILL BE HOT SPARE NODES. */
+ /* --------------------------------------------------------------------- */
+ if ((csystemnodes - cnoReplicas) >= cminHotSpareNodes) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ // We set the minimum number of hot spares according to users request
+ // through the configuration file.
+ /* --------------------------------------------------------------------- */
+ tchsNoNodes = csystemnodes - cminHotSpareNodes;
+ cnoHotSpare = cminHotSpareNodes;
+ } else if (cminHotSpareNodes > 0) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ // The user requested at least one hot spare node and we will support him
+ // in that.
+ /* --------------------------------------------------------------------- */
+ tchsNoNodes = csystemnodes - 1;
+ cnoHotSpare = 1;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------- */
+ // The user did not request any hot spare nodes so in this case we will
+ // only use hot spare nodes if the number of nodes is such that we cannot
+ // use all nodes as normal nodes.
+ /* --------------------------------------------------------------------- */
+ tchsNoNodes = csystemnodes;
+ cnoHotSpare = 0;
+ }//if
+ } else {
+ jam();
+ /* --------------------------------------------------------------------- */
+ // We only have enough to support the replicas. We will not have any hot
+ // spares.
+ /* --------------------------------------------------------------------- */
+ tchsNoNodes = csystemnodes;
+ cnoHotSpare = 0;
+ }//if
+ tchsTmp = tchsNoNodes - (cnoReplicas * (tchsNoNodes / cnoReplicas));
+ cnoHotSpare = cnoHotSpare + tchsTmp;
+ break;
+ default:
+ jam();
+ progError(0, 0);
+ break;
+ }//switch
+}//Dbdih::calculateHotSpare()
+
+/*************************************************************************/
+/* CHECK IF THE NODE CRASH IS TO ESCALATE INTO A SYSTEM CRASH. WE COULD */
+/* DO THIS BECAUSE ALL REPLICAS OF SOME FRAGMENT ARE LOST. WE COULD ALSO */
+/* DO IT AFTER MANY NODE FAILURES THAT MAKE IT VERY DIFFICULT TO RESTORE */
+/* DATABASE AFTER A SYSTEM CRASH. IT MIGHT EVEN BE IMPOSSIBLE AND THIS */
+/* MUST BE AVOIDED EVEN MORE THAN AVOIDING SYSTEM CRASHES. */
+/*************************************************************************/
+void Dbdih::checkEscalation()
+{
+ Uint32 TnodeGroup[MAX_NDB_NODES];
+ NodeRecordPtr nodePtr;
+ for (Uint32 i = 0; i < MAX_NDB_NODES; i++) {
+ TnodeGroup[i] = ZFALSE;
+ }//for
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if (nodePtr.p->nodeStatus == NodeRecord::ALIVE &&
+ nodePtr.p->activeStatus == Sysfile::NS_Active){
+ ndbrequire(nodePtr.p->nodeGroup < MAX_NDB_NODES);
+ TnodeGroup[nodePtr.p->nodeGroup] = ZTRUE;
+ }
+ }
+ for (Uint32 i = 0; i < cnoOfNodeGroups; i++) {
+ jam();
+ if (TnodeGroup[i] == ZFALSE) {
+ jam();
+ progError(__LINE__, ERR_SYSTEM_ERROR, "Lost node group");
+ }//if
+ }//for
+}//Dbdih::checkEscalation()
+
+/*************************************************************************/
+/* */
+/* MODULE: CHECK_KEEP_GCI */
+/* DESCRIPTION: CHECK FOR MINIMUM GCI RESTORABLE WITH NEW LOCAL */
+/* CHECKPOINT. */
+/*************************************************************************/
+void Dbdih::checkKeepGci(Uint32 replicaStartIndex)
+{
+ ReplicaRecordPtr ckgReplicaPtr;
+ ckgReplicaPtr.i = replicaStartIndex;
+ while (ckgReplicaPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(ckgReplicaPtr, creplicaFileSize, replicaRecord);
+ Uint32 keepGci;
+ Uint32 oldestRestorableGci;
+ findMinGci(ckgReplicaPtr, keepGci, oldestRestorableGci);
+ if (keepGci < c_lcpState.keepGci) {
+ jam();
+ /* ------------------------------------------------------------------- */
+ /* WE MUST KEEP LOG RECORDS SO THAT WE CAN USE ALL LOCAL CHECKPOINTS */
+ /* THAT ARE AVAILABLE. THUS WE NEED TO CALCULATE THE MINIMUM OVER ALL */
+ /* FRAGMENTS. */
+ /* ------------------------------------------------------------------- */
+ c_lcpState.keepGci = keepGci;
+ }//if
+ if (oldestRestorableGci > c_lcpState.oldestRestorableGci) {
+ jam();
+ c_lcpState.oldestRestorableGci = oldestRestorableGci;
+ ndbrequire(((int)c_lcpState.oldestRestorableGci) >= 0);
+ }//if
+ ckgReplicaPtr.i = ckgReplicaPtr.p->nextReplica;
+ }//while
+}//Dbdih::checkKeepGci()
+
+void Dbdih::closeFile(Signal* signal, FileRecordPtr filePtr)
+{
+ signal->theData[0] = filePtr.p->fileRef;
+ signal->theData[1] = reference();
+ signal->theData[2] = filePtr.i;
+ signal->theData[3] = ZCLOSE_NO_DELETE;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+}//Dbdih::closeFile()
+
+void Dbdih::closeFileDelete(Signal* signal, FileRecordPtr filePtr)
+{
+ signal->theData[0] = filePtr.p->fileRef;
+ signal->theData[1] = reference();
+ signal->theData[2] = filePtr.i;
+ signal->theData[3] = ZCLOSE_DELETE;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+}//Dbdih::closeFileDelete()
+
+void Dbdih::createFileRw(Signal* signal, FileRecordPtr filePtr)
+{
+ signal->theData[0] = reference();
+ signal->theData[1] = filePtr.i;
+ signal->theData[2] = filePtr.p->fileName[0];
+ signal->theData[3] = filePtr.p->fileName[1];
+ signal->theData[4] = filePtr.p->fileName[2];
+ signal->theData[5] = filePtr.p->fileName[3];
+ signal->theData[6] = ZCREATE_READ_WRITE;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+}//Dbdih::createFileRw()
+
+void Dbdih::emptyverificbuffer(Signal* signal, bool aContinueB)
+{
+ if(cfirstVerifyQueue == RNIL){
+ jam();
+ return;
+ }//if
+ ApiConnectRecordPtr localApiConnectptr;
+ if(getBlockCommit() == false){
+ jam();
+ ndbrequire(cverifyQueueCounter > 0);
+ cverifyQueueCounter--;
+ localApiConnectptr.i = cfirstVerifyQueue;
+ ptrCheckGuard(localApiConnectptr, capiConnectFileSize, apiConnectRecord);
+ ndbrequire(localApiConnectptr.p->apiGci <= currentgcp);
+ cfirstVerifyQueue = localApiConnectptr.p->nextApi;
+ if (cfirstVerifyQueue == RNIL) {
+ jam();
+ ndbrequire(cverifyQueueCounter == 0);
+ clastVerifyQueue = RNIL;
+ }//if
+ signal->theData[0] = localApiConnectptr.i;
+ signal->theData[1] = currentgcp;
+ sendSignal(clocaltcblockref, GSN_DIVERIFYCONF, signal, 2, JBB);
+ if (aContinueB == true) {
+ jam();
+ //-----------------------------------------------------------------------
+ // This emptying happened as part of a take-out process by continueb signals.
+ // This ensures that we will empty the queue eventually. We will also empty
+ // one item every time we insert one item to ensure that the list doesn't
+ // grow when it is not blocked.
+ //-----------------------------------------------------------------------
+ signal->theData[0] = DihContinueB::ZEMPTY_VERIFY_QUEUE;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 1, JBB);
+ }//if
+ } else {
+ jam();
+ //-----------------------------------------------------------------------
+ // We are blocked so it is no use in continuing the emptying of the
+ // verify buffer. Whenever the block is removed the emptying will
+ // restart.
+ //-----------------------------------------------------------------------
+ }
+ return;
+}//Dbdih::emptyverificbuffer()
+
+/*----------------------------------------------------------------*/
+/* FIND A FREE HOT SPARE IF AVAILABLE AND ALIVE. */
+/*----------------------------------------------------------------*/
+Uint32 Dbdih::findHotSpare()
+{
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if (nodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ if (nodePtr.p->activeStatus == Sysfile::NS_HotSpare) {
+ jam();
+ return nodePtr.i;
+ }//if
+ }//if
+ }//for
+ return RNIL;
+}//Dbdih::findHotSpare()
+
+/*************************************************************************/
+/* FIND THE NODES FROM WHICH WE CAN EXECUTE THE LOG TO RESTORE THE */
+/* DATA NODE IN A SYSTEM RESTART. */
+/*************************************************************************/
+bool Dbdih::findLogNodes(CreateReplicaRecord* createReplica,
+ FragmentstorePtr fragPtr,
+ Uint32 startGci,
+ Uint32 stopGci)
+{
+ ConstPtr<ReplicaRecord> flnReplicaPtr;
+ flnReplicaPtr.i = createReplica->replicaRec;
+ ptrCheckGuard(flnReplicaPtr, creplicaFileSize, replicaRecord);
+ /* --------------------------------------------------------------------- */
+ /* WE START BY CHECKING IF THE DATA NODE CAN HANDLE THE LOG ALL BY */
+ /* ITSELF. THIS IS THE DESIRED BEHAVIOUR. IF THIS IS NOT POSSIBLE */
+ /* THEN WE SEARCH FOR THE BEST POSSIBLE NODES AMONG THE NODES THAT */
+ /* ARE PART OF THIS SYSTEM RESTART. */
+ /* THIS CAN ONLY BE HANDLED BY THE LAST CRASHED REPLICA. */
+ /* The condition is that the replica was created before or at the */
+ /* time of the starting gci, in addition it must have been alive */
+ /* at the time of the stopping gci. This is checked by two */
+ /* conditions, the first checks replicaLastGci and the second */
+ /* checks that it is also smaller than the last gci the node was */
+ /* involved in. This is necessary to check since createGci is set */
+ /* Last + 1 and sometimes startGci = stopGci + 1 and in that case */
+ /* it could happen that replicaLastGci is set to -1 with CreateGci */
+ /* set to LastGci + 1. */
+ /* --------------------------------------------------------------------- */
+ arrGuard(flnReplicaPtr.p->noCrashedReplicas, 8);
+ const Uint32 noCrashed = flnReplicaPtr.p->noCrashedReplicas;
+
+ if (!(ERROR_INSERTED(7073) || ERROR_INSERTED(7074))&&
+ (startGci >= flnReplicaPtr.p->createGci[noCrashed]) &&
+ (stopGci <= flnReplicaPtr.p->replicaLastGci[noCrashed]) &&
+ (stopGci <= SYSFILE->lastCompletedGCI[flnReplicaPtr.p->procNode])) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* WE FOUND ALL THE LOG RECORDS NEEDED IN THE DATA NODE. WE WILL */
+ /* USE THOSE. */
+ /* --------------------------------------------------------------------- */
+ createReplica->noLogNodes = 1;
+ createReplica->logStartGci[0] = startGci;
+ createReplica->logStopGci[0] = stopGci;
+ createReplica->logNodeId[0] = flnReplicaPtr.p->procNode;
+ return true;
+ }//if
+ Uint32 logNode = 0;
+ do {
+ Uint32 fblStopGci;
+ jam();
+ if(!findBestLogNode(createReplica,
+ fragPtr,
+ startGci,
+ stopGci,
+ logNode,
+ fblStopGci)){
+ jam();
+ return false;
+ }
+
+ logNode++;
+ if (fblStopGci >= stopGci) {
+ jam();
+ createReplica->noLogNodes = logNode;
+ return true;
+ }//if
+ startGci = fblStopGci + 1;
+ if (logNode >= 4) { // Why??
+ jam();
+ break;
+ }//if
+ } while (1);
+ /* --------------------------------------------------------------------- */
+ /* IT WAS NOT POSSIBLE TO RESTORE THE REPLICA. THIS CAN EITHER BE */
+ /* BECAUSE OF LACKING NODES OR BECAUSE OF A REALLY SERIOUS PROBLEM.*/
+ /* --------------------------------------------------------------------- */
+ return false;
+}//Dbdih::findLogNodes()
+
+/*************************************************************************/
+/* FIND THE BEST POSSIBLE LOG NODE TO EXECUTE THE LOG AS SPECIFIED */
+/* BY THE INPUT PARAMETERS. WE SCAN THROUGH ALL ALIVE REPLICAS. */
+/* THIS MEANS STORED, OLD_STORED */
+/*************************************************************************/
+bool
+Dbdih::findBestLogNode(CreateReplicaRecord* createReplica,
+ FragmentstorePtr fragPtr,
+ Uint32 startGci,
+ Uint32 stopGci,
+ Uint32 logNode,
+ Uint32& fblStopGci)
+{
+ ConstPtr<ReplicaRecord> fblFoundReplicaPtr;
+ ConstPtr<ReplicaRecord> fblReplicaPtr;
+
+ /* --------------------------------------------------------------------- */
+ /* WE START WITH ZERO AS FOUND TO ENSURE THAT FIRST HIT WILL BE */
+ /* BETTER. */
+ /* --------------------------------------------------------------------- */
+ fblStopGci = 0;
+ fblReplicaPtr.i = fragPtr.p->storedReplicas;
+ while (fblReplicaPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(fblReplicaPtr, creplicaFileSize, replicaRecord);
+ if (checkNodeAlive(fblReplicaPtr.p->procNode)) {
+ jam();
+ Uint32 fliStopGci = findLogInterval(fblReplicaPtr, startGci);
+ if (fliStopGci > fblStopGci) {
+ jam();
+ fblStopGci = fliStopGci;
+ fblFoundReplicaPtr = fblReplicaPtr;
+ }//if
+ }//if
+ fblReplicaPtr.i = fblReplicaPtr.p->nextReplica;
+ }//while
+ fblReplicaPtr.i = fragPtr.p->oldStoredReplicas;
+ while (fblReplicaPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(fblReplicaPtr, creplicaFileSize, replicaRecord);
+ if (checkNodeAlive(fblReplicaPtr.p->procNode)) {
+ jam();
+ Uint32 fliStopGci = findLogInterval(fblReplicaPtr, startGci);
+ if (fliStopGci > fblStopGci) {
+ jam();
+ fblStopGci = fliStopGci;
+ fblFoundReplicaPtr = fblReplicaPtr;
+ }//if
+ }//if
+ fblReplicaPtr.i = fblReplicaPtr.p->nextReplica;
+ }//while
+ if (fblStopGci != 0) {
+ jam();
+ ndbrequire(logNode < MAX_LOG_EXEC);
+ createReplica->logNodeId[logNode] = fblFoundReplicaPtr.p->procNode;
+ createReplica->logStartGci[logNode] = startGci;
+ if (fblStopGci >= stopGci) {
+ jam();
+ createReplica->logStopGci[logNode] = stopGci;
+ } else {
+ jam();
+ createReplica->logStopGci[logNode] = fblStopGci;
+ }//if
+ }//if
+
+ return fblStopGci != 0;
+}//Dbdih::findBestLogNode()
+
+Uint32 Dbdih::findLogInterval(ConstPtr<ReplicaRecord> replicaPtr,
+ Uint32 startGci)
+{
+ ndbrequire(replicaPtr.p->noCrashedReplicas <= 8);
+ Uint32 loopLimit = replicaPtr.p->noCrashedReplicas + 1;
+ for (Uint32 i = 0; i < loopLimit; i++) {
+ jam();
+ if (replicaPtr.p->createGci[i] <= startGci) {
+ if (replicaPtr.p->replicaLastGci[i] >= startGci) {
+ jam();
+ return replicaPtr.p->replicaLastGci[i];
+ }//if
+ }//if
+ }//for
+ return 0;
+}//Dbdih::findLogInterval()
+
+/*************************************************************************/
+/* */
+/* MODULE: FIND THE MINIMUM GCI THAT THIS NODE HAS LOG RECORDS FOR.*/
+/*************************************************************************/
+void Dbdih::findMinGci(ReplicaRecordPtr fmgReplicaPtr,
+ Uint32& keepGci,
+ Uint32& oldestRestorableGci)
+{
+ Uint32 nextLcpNo;
+ Uint32 lcpNo;
+ for (Uint32 i = 0; i < MAX_LCP_STORED; i++) {
+ jam();
+ if ((fmgReplicaPtr.p->lcpStatus[i] == ZVALID) &&
+ ((fmgReplicaPtr.p->lcpId[i] + MAX_LCP_STORED) <= (SYSFILE->latestLCP_ID + 1))) {
+ jam();
+ /*--------------------------------------------------------------------*/
+ // We invalidate the checkpoint we are preparing to overwrite.
+ // The LCP id is still the old lcp id,
+ // this is the reason of comparing with lcpId + 1.
+ /*---------------------------------------------------------------------*/
+ fmgReplicaPtr.p->lcpStatus[i] = ZINVALID;
+ }//if
+ }//for
+ keepGci = (Uint32)-1;
+ oldestRestorableGci = 0;
+ nextLcpNo = fmgReplicaPtr.p->nextLcp;
+ lcpNo = fmgReplicaPtr.p->nextLcp;
+ do {
+ ndbrequire(lcpNo < MAX_LCP_STORED);
+ if (fmgReplicaPtr.p->lcpStatus[lcpNo] == ZVALID) {
+ jam();
+ keepGci = fmgReplicaPtr.p->maxGciCompleted[lcpNo];
+ oldestRestorableGci = fmgReplicaPtr.p->maxGciStarted[lcpNo];
+ ndbrequire(((int)oldestRestorableGci) >= 0);
+ return;
+ } else {
+ jam();
+ ndbrequire(fmgReplicaPtr.p->lcpStatus[lcpNo] == ZINVALID);
+ if (fmgReplicaPtr.p->createGci[0] == fmgReplicaPtr.p->initialGci) {
+ jam();
+ /*-------------------------------------------------------------------
+ * WE CAN STILL RESTORE THIS REPLICA WITHOUT ANY LOCAL CHECKPOINTS BY
+ * ONLY USING THE LOG. IF THIS IS NOT POSSIBLE THEN WE REPORT THE LAST
+ * VALID LOCAL CHECKPOINT AS THE MINIMUM GCI RECOVERABLE.
+ *-----------------------------------------------------------------*/
+ keepGci = fmgReplicaPtr.p->createGci[0];
+ }//if
+ }//if
+ lcpNo = prevLcpNo(lcpNo);
+ } while (lcpNo != nextLcpNo);
+ return;
+}//Dbdih::findMinGci()
+
+bool Dbdih::findStartGci(ConstPtr<ReplicaRecord> replicaPtr,
+ Uint32 stopGci,
+ Uint32& startGci,
+ Uint32& lcpNo)
+{
+ lcpNo = replicaPtr.p->nextLcp;
+ const Uint32 startLcpNo = lcpNo;
+ do {
+ lcpNo = prevLcpNo(lcpNo);
+ ndbrequire(lcpNo < MAX_LCP_STORED);
+ if (replicaPtr.p->lcpStatus[lcpNo] == ZVALID) {
+ if (replicaPtr.p->maxGciStarted[lcpNo] < stopGci) {
+ jam();
+ /* ----------------------------------------------------------------- */
+ /* WE HAVE FOUND A USEFUL LOCAL CHECKPOINT THAT CAN BE USED FOR */
+ /* RESTARTING THIS FRAGMENT REPLICA. */
+ /* ----------------------------------------------------------------- */
+ startGci = replicaPtr.p->maxGciCompleted[lcpNo] + 1;
+ return true;
+ }
+ }
+ } while (lcpNo != startLcpNo);
+ /* --------------------------------------------------------------------- */
+ /* NO VALID LOCAL CHECKPOINT WAS AVAILABLE. WE WILL ADD THE */
+ /* FRAGMENT. THUS THE NEXT LCP MUST BE SET TO ZERO. */
+ /* WE MUST EXECUTE THE LOG FROM THE INITIAL GLOBAL CHECKPOINT WHEN */
+ /* THE TABLE WAS CREATED. */
+ /* --------------------------------------------------------------------- */
+ startGci = replicaPtr.p->initialGci;
+ ndbrequire(replicaPtr.p->nextLcp == 0);
+ return false;
+}//Dbdih::findStartGci()
+
+/**************************************************************************/
+/* ---------------------------------------------------------------------- */
+/* FIND A TAKE OVER REPLICA WHICH IS TO BE STARTED OR COMMITTED WHEN*/
+/* TAKING OVER A FAILED NODE. */
+/* ---------------------------------------------------------------------- */
+/*************************************************************************/
+void Dbdih::findToReplica(TakeOverRecord* regTakeOver,
+ Uint32 replicaType,
+ FragmentstorePtr fragPtr,
+ ReplicaRecordPtr& ftrReplicaPtr)
+{
+ switch (replicaType) {
+ case CreateFragReq::STORED:
+ case CreateFragReq::COMMIT_STORED:
+ /* ----------------------------------------------------------------------*/
+ /* HERE WE SEARCH FOR STORED REPLICAS. THE REPLICA MUST BE STORED IN THE */
+ /* SECTION FOR OLD STORED REPLICAS SINCE WE HAVE NOT TAKEN OVER YET. */
+ /* ----------------------------------------------------------------------*/
+ ftrReplicaPtr.i = fragPtr.p->oldStoredReplicas;
+ while (ftrReplicaPtr.i != RNIL) {
+ ptrCheckGuard(ftrReplicaPtr, creplicaFileSize, replicaRecord);
+ if (ftrReplicaPtr.p->procNode == regTakeOver->toStartingNode) {
+ jam();
+ return;
+ } else {
+ if (ftrReplicaPtr.p->procNode == regTakeOver->toFailedNode) {
+ jam();
+ return;
+ } else {
+ jam();
+ ftrReplicaPtr.i = ftrReplicaPtr.p->nextReplica;
+ }//if
+ }//if
+ }//while
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+}//Dbdih::findToReplica()
+
+void Dbdih::initCommonData()
+{
+ c_blockCommit = false;
+ c_blockCommitNo = 0;
+ c_createFragmentLock = RNIL;
+ c_endToLock = RNIL;
+ cfailurenr = 1;
+ cfirstAliveNode = RNIL;
+ cfirstDeadNode = RNIL;
+ cfirstVerifyQueue = RNIL;
+ cgckptflag = false;
+ cgcpDelay = 0;
+ cgcpMasterTakeOverState = GMTOS_IDLE;
+ cgcpOrderBlocked = 0;
+ cgcpParticipantState = GCP_PARTICIPANT_READY;
+ cgcpSameCounter = 0;
+ cgcpStartCounter = 0;
+ cgcpStatus = GCP_READY;
+
+ clastVerifyQueue = RNIL;
+ c_lcpMasterTakeOverState.set(LMTOS_IDLE, __LINE__);
+
+ c_lcpState.clcpDelay = 0;
+ c_lcpState.lcpStart = ZIDLE;
+ c_lcpState.lcpStartGcp = 0;
+ c_lcpState.setLcpStatus(LCP_STATUS_IDLE, __LINE__);
+ c_lcpState.currentFragment.tableId = 0;
+ c_lcpState.currentFragment.fragmentId = 0;
+ c_lcpState.noOfLcpFragRepOutstanding = 0;
+ c_lcpState.keepGci = 0;
+ c_lcpState.oldestRestorableGci = 0;
+ c_lcpState.ctcCounter = 0;
+ c_lcpState.ctimer = 0;
+ c_lcpState.immediateLcpStart = false;
+ c_lcpState.m_MASTER_LCPREQ_Received = false;
+
+ cmasterdihref = 0;
+ cmasterNodeId = 0;
+ cmasterState = MASTER_IDLE;
+ cmasterTakeOverNode = 0;
+ cnewgcp = 0;
+ cnoHotSpare = 0;
+ cnoOfActiveTables = 0;
+ cnoOfNodeGroups = 0;
+ cnoReplicas = 0;
+ coldgcp = 0;
+ coldGcpId = 0;
+ coldGcpStatus = cgcpStatus;
+ con_lineNodes = 0;
+ creceivedfrag = 0;
+ crestartGci = 0;
+ crestartInfoFile[0] = RNIL;
+ crestartInfoFile[1] = RNIL;
+ cstartGcpNow = false;
+ cstartPhase = 0;
+ c_startToLock = RNIL;
+ cstarttype = (Uint32)-1;
+ csystemnodes = 0;
+ c_updateToLock = RNIL;
+ currentgcp = 0;
+ cverifyQueueCounter = 0;
+ cwaitLcpSr = false;
+
+ nodeResetStart();
+ c_nodeStartMaster.wait = ZFALSE;
+
+ memset(&sysfileData[0], 0, sizeof(sysfileData));
+}//Dbdih::initCommonData()
+
+void Dbdih::initFragstore(FragmentstorePtr fragPtr)
+{
+ fragPtr.p->storedReplicas = RNIL;
+ fragPtr.p->oldStoredReplicas = RNIL;
+
+ fragPtr.p->noStoredReplicas = 0;
+ fragPtr.p->noOldStoredReplicas = 0;
+ fragPtr.p->fragReplicas = 0;
+ fragPtr.p->preferredPrimary = 0;
+
+ for (Uint32 i = 0; i < MAX_REPLICAS; i++)
+ fragPtr.p->activeNodes[i] = 0;
+
+ fragPtr.p->noLcpReplicas = 0;
+ fragPtr.p->distributionKey = 0;
+}//Dbdih::initFragstore()
+
+void Dbdih::initNodeState(NodeRecordPtr nodePtr)
+{
+ nodePtr.p->gcpstate = NodeRecord::READY;
+
+ nodePtr.p->activeStatus = Sysfile::NS_NotDefined;
+ nodePtr.p->recNODE_FAILREP = ZFALSE;
+ nodePtr.p->nodeGroup = ZNIL;
+ nodePtr.p->dbtcFailCompleted = ZTRUE;
+ nodePtr.p->dbdictFailCompleted = ZTRUE;
+ nodePtr.p->dbdihFailCompleted = ZTRUE;
+ nodePtr.p->dblqhFailCompleted = ZTRUE;
+ nodePtr.p->noOfStartedChkpt = 0;
+ nodePtr.p->noOfQueuedChkpt = 0;
+ nodePtr.p->lcpStateAtTakeOver = (MasterLCPConf::State)255;
+
+ nodePtr.p->activeTabptr = RNIL;
+ nodePtr.p->nodeStatus = NodeRecord::NOT_IN_CLUSTER;
+ nodePtr.p->useInTransactions = false;
+ nodePtr.p->copyCompleted = false;
+ nodePtr.p->ndbversion = 0xffff;
+}//Dbdih::initNodeState()
+
+/*************************************************************************/
+/* */
+/* MODULE: INIT_RESTART_INFO */
+/* DESCRIPTION: INITIATE RESTART INFO VARIABLE AND VARIABLES FOR */
+/* GLOBAL CHECKPOINTS. */
+/*************************************************************************/
+void Dbdih::initRestartInfo()
+{
+ for (int i = 0; i < MAX_NDB_NODES; i++) {
+ SYSFILE->lastCompletedGCI[i] = 0;
+ }//for
+ NodeRecordPtr nodePtr;
+ nodePtr.i = cfirstAliveNode;
+ do {
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ SYSFILE->lastCompletedGCI[nodePtr.i] = 1;
+ /* FIRST GCP = 1 ALREADY SET BY LQH */
+ nodePtr.i = nodePtr.p->nextNode;
+ } while (nodePtr.i != RNIL);
+ coldgcp = 1;
+ currentgcp = 2;
+ cnewgcp = 2;
+ crestartGci = 1;
+
+ SYSFILE->keepGCI = 1;
+ SYSFILE->oldestRestorableGCI = 1;
+ SYSFILE->newestRestorableGCI = 1;
+ SYSFILE->systemRestartBits = 0;
+ for (Uint32 i = 0; i < NodeBitmask::Size; i++) {
+ SYSFILE->lcpActive[0] = 0;
+ }//for
+ for (Uint32 i = 0; i < Sysfile::TAKE_OVER_SIZE; i++) {
+ SYSFILE->takeOver[i] = 0;
+ }//for
+ Sysfile::setInitialStartOngoing(SYSFILE->systemRestartBits);
+}//Dbdih::initRestartInfo()
+
+/*--------------------------------------------------------------------*/
+/* NODE GROUP BITS ARE INITIALISED BEFORE THIS. */
+/* NODE ACTIVE BITS ARE INITIALISED BEFORE THIS. */
+/*--------------------------------------------------------------------*/
+/*************************************************************************/
+/* */
+/* MODULE: INIT_RESTORABLE_GCI_FILES */
+/* DESCRIPTION: THE SUBROUTINE SETS UP THE FILES THAT REFERS TO THE*/
+/* FILES THAT KEEP THE VARIABLE CRESTART_INFO */
+/*************************************************************************/
+void Dbdih::initRestorableGciFiles()
+{
+ Uint32 tirgTmp;
+ FileRecordPtr filePtr;
+ seizeFile(filePtr);
+ filePtr.p->tabRef = RNIL;
+ filePtr.p->fileType = FileRecord::GCP_FILE;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ filePtr.p->fileStatus = FileRecord::CLOSED;
+ crestartInfoFile[0] = filePtr.i;
+ filePtr.p->fileName[0] = (Uint32)-1; /* T DIRECTORY NOT USED */
+ filePtr.p->fileName[1] = (Uint32)-1; /* F DIRECTORY NOT USED */
+ filePtr.p->fileName[2] = (Uint32)-1; /* S PART IGNORED */
+ tirgTmp = 1; /* FILE NAME VERSION 1 */
+ tirgTmp = (tirgTmp << 8) + 6; /* .SYSFILE */
+ tirgTmp = (tirgTmp << 8) + 1; /* D1 DIRECTORY */
+ tirgTmp = (tirgTmp << 8) + 0; /* P0 FILE NAME */
+ filePtr.p->fileName[3] = tirgTmp;
+ /* --------------------------------------------------------------------- */
+ /* THE NAME BECOMES /D1/DBDICT/S0.SYSFILE */
+ /* --------------------------------------------------------------------- */
+ seizeFile(filePtr);
+ filePtr.p->tabRef = RNIL;
+ filePtr.p->fileType = FileRecord::GCP_FILE;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ filePtr.p->fileStatus = FileRecord::CLOSED;
+ crestartInfoFile[1] = filePtr.i;
+ filePtr.p->fileName[0] = (Uint32)-1; /* T DIRECTORY NOT USED */
+ filePtr.p->fileName[1] = (Uint32)-1; /* F DIRECTORY NOT USED */
+ filePtr.p->fileName[2] = (Uint32)-1; /* S PART IGNORED */
+ tirgTmp = 1; /* FILE NAME VERSION 1 */
+ tirgTmp = (tirgTmp << 8) + 6; /* .SYSFILE */
+ tirgTmp = (tirgTmp << 8) + 2; /* D1 DIRECTORY */
+ tirgTmp = (tirgTmp << 8) + 0; /* P0 FILE NAME */
+ filePtr.p->fileName[3] = tirgTmp;
+ /* --------------------------------------------------------------------- */
+ /* THE NAME BECOMES /D2/DBDICT/P0.SYSFILE */
+ /* --------------------------------------------------------------------- */
+}//Dbdih::initRestorableGciFiles()
+
+void Dbdih::initTable(TabRecordPtr tabPtr)
+{
+ tabPtr.p->method = TabRecord::NOTDEFINED;
+ tabPtr.p->tabStatus = TabRecord::TS_IDLE;
+ tabPtr.p->noOfWords = 0;
+ tabPtr.p->noPages = 0;
+ tabPtr.p->tabLcpStatus = TabRecord::TLS_COMPLETED;
+ tabPtr.p->tabCopyStatus = TabRecord::CS_IDLE;
+ tabPtr.p->tabUpdateState = TabRecord::US_IDLE;
+ tabPtr.p->noOfBackups = 0;
+ tabPtr.p->kvalue = 0;
+ tabPtr.p->hashpointer = (Uint32)-1;
+ tabPtr.p->mask = 0;
+ tabPtr.p->storedTable = 1;
+ tabPtr.p->tabErrorCode = 0;
+ tabPtr.p->schemaVersion = (Uint32)-1;
+ tabPtr.p->tabRemoveNode = RNIL;
+ tabPtr.p->totalfragments = (Uint32)-1;
+ tabPtr.p->connectrec = RNIL;
+ tabPtr.p->tabFile[0] = RNIL;
+ tabPtr.p->tabFile[1] = RNIL;
+ tabPtr.p->m_dropTab.tabUserRef = 0;
+ tabPtr.p->m_dropTab.tabUserPtr = RNIL;
+ for (Uint32 i = 0; i < MAX_NDB_NODES; i++) {
+ tabPtr.p->startFid[i] = RNIL;
+ }//for
+ for (Uint32 i = 0; i < 8; i++) {
+ tabPtr.p->pageRef[i] = RNIL;
+ }//for
+ tabPtr.p->tableType = DictTabInfo::UndefTableType;
+}//Dbdih::initTable()
+
+/*************************************************************************/
+/* */
+/* MODULE: INIT_TABLE_FILES */
+/* DESCRIPTION: THE SUBROUTINE SETS UP THE FILES THAT REFERS TO THE*/
+/* FILES THAT KEEP THE TABLE FRAGMENTATION DESCRIPTION. */
+/*************************************************************************/
+void Dbdih::initTableFile(TabRecordPtr tabPtr)
+{
+ Uint32 titfTmp;
+ FileRecordPtr filePtr;
+ seizeFile(filePtr);
+ filePtr.p->tabRef = tabPtr.i;
+ filePtr.p->fileType = FileRecord::TABLE_FILE;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ filePtr.p->fileStatus = FileRecord::CLOSED;
+ tabPtr.p->tabFile[0] = filePtr.i;
+ filePtr.p->fileName[0] = (Uint32)-1; /* T DIRECTORY NOT USED */
+ filePtr.p->fileName[1] = (Uint32)-1; /* F DIRECTORY NOT USED */
+ filePtr.p->fileName[2] = tabPtr.i; /* Stid FILE NAME */
+ titfTmp = 1; /* FILE NAME VERSION 1 */
+ titfTmp = (titfTmp << 8) + 3; /* .FRAGLIST */
+ titfTmp = (titfTmp << 8) + 1; /* D1 DIRECTORY */
+ titfTmp = (titfTmp << 8) + 255; /* P PART IGNORED */
+ filePtr.p->fileName[3] = titfTmp;
+ /* --------------------------------------------------------------------- */
+ /* THE NAME BECOMES /D1/DBDICT/Stid.FRAGLIST */
+ /* --------------------------------------------------------------------- */
+ seizeFile(filePtr);
+ filePtr.p->tabRef = tabPtr.i;
+ filePtr.p->fileType = FileRecord::TABLE_FILE;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ filePtr.p->fileStatus = FileRecord::CLOSED;
+ tabPtr.p->tabFile[1] = filePtr.i;
+ filePtr.p->fileName[0] = (Uint32)-1; /* T DIRECTORY NOT USED */
+ filePtr.p->fileName[1] = (Uint32)-1; /* F DIRECTORY NOT USED */
+ filePtr.p->fileName[2] = tabPtr.i; /* Stid FILE NAME */
+ titfTmp = 1; /* FILE NAME VERSION 1 */
+ titfTmp = (titfTmp << 8) + 3; /* .FRAGLIST */
+ titfTmp = (titfTmp << 8) + 2; /* D2 DIRECTORY */
+ titfTmp = (titfTmp << 8) + 255; /* P PART IGNORED */
+ filePtr.p->fileName[3] = titfTmp;
+ /* --------------------------------------------------------------------- */
+ /* THE NAME BECOMES /D2/DBDICT/Stid.FRAGLIST */
+ /* --------------------------------------------------------------------- */
+}//Dbdih::initTableFile()
+
+void Dbdih::initialiseRecordsLab(Signal* signal, Uint32 stepNo)
+{
+ switch (stepNo) {
+ case 0:
+ jam();
+ initCommonData();
+ break;
+ case 1:
+ {
+ ApiConnectRecordPtr apiConnectptr;
+ jam();
+ /******** INTIALIZING API CONNECT RECORDS ********/
+ for (apiConnectptr.i = 0; apiConnectptr.i < capiConnectFileSize; apiConnectptr.i++) {
+ ptrAss(apiConnectptr, apiConnectRecord);
+ apiConnectptr.p->nextApi = RNIL;
+ }//for
+ jam();
+ break;
+ }
+ case 2:
+ {
+ ConnectRecordPtr connectPtr;
+ jam();
+ /****** CONNECT ******/
+ for (connectPtr.i = 0; connectPtr.i < cconnectFileSize; connectPtr.i++) {
+ ptrAss(connectPtr, connectRecord);
+ connectPtr.p->userpointer = RNIL;
+ connectPtr.p->userblockref = ZNIL;
+ connectPtr.p->connectState = ConnectRecord::FREE;
+ connectPtr.p->table = RNIL;
+ connectPtr.p->nfConnect = connectPtr.i + 1;
+ }//for
+ connectPtr.i = cconnectFileSize - 1;
+ ptrAss(connectPtr, connectRecord);
+ connectPtr.p->nfConnect = RNIL;
+ cfirstconnect = 0;
+ break;
+ }
+ case 3:
+ {
+ FileRecordPtr filePtr;
+ jam();
+ /******** INTIALIZING FILE RECORDS ********/
+ for (filePtr.i = 0; filePtr.i < cfileFileSize; filePtr.i++) {
+ ptrAss(filePtr, fileRecord);
+ filePtr.p->nextFile = filePtr.i + 1;
+ filePtr.p->fileStatus = FileRecord::CLOSED;
+ filePtr.p->reqStatus = FileRecord::IDLE;
+ }//for
+ filePtr.i = cfileFileSize - 1;
+ ptrAss(filePtr, fileRecord);
+ filePtr.p->nextFile = RNIL;
+ cfirstfreeFile = 0;
+ initRestorableGciFiles();
+ break;
+ }
+ case 4:
+ jam();
+ initialiseFragstore();
+ break;
+ case 5:
+ {
+ jam();
+ /******* NODE GROUP RECORD ******/
+ /******* NODE RECORD ******/
+ NodeGroupRecordPtr loopNGPtr;
+ for (loopNGPtr.i = 0; loopNGPtr.i < MAX_NDB_NODES; loopNGPtr.i++) {
+ ptrAss(loopNGPtr, nodeGroupRecord);
+ loopNGPtr.p->nodesInGroup[0] = RNIL;
+ loopNGPtr.p->nodesInGroup[1] = RNIL;
+ loopNGPtr.p->nodesInGroup[2] = RNIL;
+ loopNGPtr.p->nodesInGroup[3] = RNIL;
+ loopNGPtr.p->nextReplicaNode = 0;
+ loopNGPtr.p->nodeCount = 0;
+ loopNGPtr.p->activeTakeOver = false;
+ }//for
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 0; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRecord);
+ initNodeState(nodePtr);
+ }//for
+ break;
+ }
+ case 6:
+ {
+ PageRecordPtr pagePtr;
+ jam();
+ /******* PAGE RECORD ******/
+ for (pagePtr.i = 0; pagePtr.i < cpageFileSize; pagePtr.i++) {
+ ptrAss(pagePtr, pageRecord);
+ pagePtr.p->nextfreepage = pagePtr.i + 1;
+ }//for
+ pagePtr.i = cpageFileSize - 1;
+ ptrAss(pagePtr, pageRecord);
+ pagePtr.p->nextfreepage = RNIL;
+ cfirstfreepage = 0;
+ break;
+ }
+ case 7:
+ {
+ ReplicaRecordPtr initReplicaPtr;
+ jam();
+ /******* REPLICA RECORD ******/
+ for (initReplicaPtr.i = 0; initReplicaPtr.i < creplicaFileSize;
+ initReplicaPtr.i++) {
+ ptrAss(initReplicaPtr, replicaRecord);
+ initReplicaPtr.p->lcpIdStarted = 0;
+ initReplicaPtr.p->lcpOngoingFlag = false;
+ initReplicaPtr.p->nextReplica = initReplicaPtr.i + 1;
+ }//for
+ initReplicaPtr.i = creplicaFileSize - 1;
+ ptrAss(initReplicaPtr, replicaRecord);
+ initReplicaPtr.p->nextReplica = RNIL;
+ cnoFreeReplicaRec = creplicaFileSize;
+ cfirstfreeReplica = 0;
+ break;
+ }
+ case 8:
+ {
+ TabRecordPtr loopTabptr;
+ jam();
+ /********* TAB-DESCRIPTOR ********/
+ for (loopTabptr.i = 0; loopTabptr.i < ctabFileSize; loopTabptr.i++) {
+ ptrAss(loopTabptr, tabRecord);
+ initTable(loopTabptr);
+ }//for
+ break;
+ }
+ case 9:
+ {
+ TakeOverRecordPtr takeOverPtr;
+ jam();
+ cfirstfreeTakeOver = RNIL;
+ for (takeOverPtr.i = 0; takeOverPtr.i < MAX_NDB_NODES; takeOverPtr.i++) {
+ ptrAss(takeOverPtr, takeOverRecord);
+ initTakeOver(takeOverPtr);
+ releaseTakeOver(takeOverPtr.i);
+ }//for
+ signal->theData[0] = DBDIH_REF;
+ sendSignal(CMVMI_REF, GSN_SIZEALT_ACK, signal, 2, JBB);
+ return;
+ break;
+ }
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ jam();
+ /* ------------------------------------------------------------------------- */
+ /* SEND REAL-TIME BREAK DURING INIT OF VARIABLES DURING SYSTEM RESTART. */
+ /* ------------------------------------------------------------------------- */
+ signal->theData[0] = DihContinueB::ZINITIALISE_RECORDS;
+ signal->theData[1] = stepNo + 1;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+}//Dbdih::initialiseRecordsLab()
+
+/*************************************************************************/
+/* INSERT THE NODE INTO THE LINKED LIST OF NODES INVOLVED ALL */
+/* DISTRIBUTED PROTOCOLS (EXCEPT GCP PROTOCOL THAT USES THE DIH */
+/* LINKED LIST INSTEAD). */
+/*************************************************************************/
+void Dbdih::insertAlive(NodeRecordPtr newNodePtr)
+{
+ NodeRecordPtr nodePtr;
+
+ nodePtr.i = cfirstAliveNode;
+ if (nodePtr.i == RNIL) {
+ jam();
+ cfirstAliveNode = newNodePtr.i;
+ } else {
+ do {
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.p->nextNode == RNIL) {
+ jam();
+ nodePtr.p->nextNode = newNodePtr.i;
+ break;
+ } else {
+ jam();
+ nodePtr.i = nodePtr.p->nextNode;
+ }//if
+ } while (1);
+ }//if
+ newNodePtr.p->nextNode = RNIL;
+}//Dbdih::insertAlive()
+
+void Dbdih::insertBackup(FragmentstorePtr fragPtr, Uint32 nodeId)
+{
+ for (Uint32 i = fragPtr.p->fragReplicas; i > 1; i--) {
+ jam();
+ ndbrequire(i < MAX_REPLICAS && i > 0);
+ fragPtr.p->activeNodes[i] = fragPtr.p->activeNodes[i - 1];
+ }//for
+ fragPtr.p->activeNodes[1] = nodeId;
+ fragPtr.p->fragReplicas++;
+}//Dbdih::insertBackup()
+
+void Dbdih::insertDeadNode(NodeRecordPtr newNodePtr)
+{
+ NodeRecordPtr nodePtr;
+
+ nodePtr.i = cfirstDeadNode;
+ if (nodePtr.i == RNIL) {
+ jam();
+ cfirstDeadNode = newNodePtr.i;
+ } else {
+ do {
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.p->nextNode == RNIL) {
+ jam();
+ nodePtr.p->nextNode = newNodePtr.i;
+ break;
+ } else {
+ jam();
+ nodePtr.i = nodePtr.p->nextNode;
+ }//if
+ } while (1);
+ }//if
+ newNodePtr.p->nextNode = RNIL;
+}//Dbdih::insertDeadNode()
+
+void Dbdih::linkOldStoredReplica(FragmentstorePtr fragPtr,
+ ReplicaRecordPtr replicatePtr)
+{
+ ReplicaRecordPtr losReplicaPtr;
+
+ replicatePtr.p->nextReplica = RNIL;
+ fragPtr.p->noOldStoredReplicas++;
+ losReplicaPtr.i = fragPtr.p->oldStoredReplicas;
+ if (losReplicaPtr.i == RNIL) {
+ jam();
+ fragPtr.p->oldStoredReplicas = replicatePtr.i;
+ return;
+ }//if
+ ptrCheckGuard(losReplicaPtr, creplicaFileSize, replicaRecord);
+ while (losReplicaPtr.p->nextReplica != RNIL) {
+ jam();
+ losReplicaPtr.i = losReplicaPtr.p->nextReplica;
+ ptrCheckGuard(losReplicaPtr, creplicaFileSize, replicaRecord);
+ }//if
+ losReplicaPtr.p->nextReplica = replicatePtr.i;
+}//Dbdih::linkOldStoredReplica()
+
+void Dbdih::linkStoredReplica(FragmentstorePtr fragPtr,
+ ReplicaRecordPtr replicatePtr)
+{
+ ReplicaRecordPtr lsrReplicaPtr;
+
+ fragPtr.p->noStoredReplicas++;
+ replicatePtr.p->nextReplica = RNIL;
+ lsrReplicaPtr.i = fragPtr.p->storedReplicas;
+ if (fragPtr.p->storedReplicas == RNIL) {
+ jam();
+ fragPtr.p->storedReplicas = replicatePtr.i;
+ return;
+ }//if
+ ptrCheckGuard(lsrReplicaPtr, creplicaFileSize, replicaRecord);
+ while (lsrReplicaPtr.p->nextReplica != RNIL) {
+ jam();
+ lsrReplicaPtr.i = lsrReplicaPtr.p->nextReplica;
+ ptrCheckGuard(lsrReplicaPtr, creplicaFileSize, replicaRecord);
+ }//if
+ lsrReplicaPtr.p->nextReplica = replicatePtr.i;
+}//Dbdih::linkStoredReplica()
+
+/*************************************************************************/
+/* MAKE NODE GROUPS BASED ON THE LIST OF NODES RECEIVED FROM CNTR */
+/*************************************************************************/
+void Dbdih::makeNodeGroups(Uint32 nodeArray[])
+{
+ NodeRecordPtr mngNodeptr;
+ Uint32 tmngNode;
+ Uint32 tmngNodeGroup;
+ Uint32 tmngReplica;
+ Uint32 tmngLimit;
+
+ /**-----------------------------------------------------------------------
+ * ASSIGN ALL ACTIVE NODES INTO NODE GROUPS. HOT SPARE NODES ARE ASSIGNED
+ * TO NODE GROUP ZNIL
+ *-----------------------------------------------------------------------*/
+ tmngNodeGroup = 0;
+ tmngReplica = 0;
+ tmngLimit = csystemnodes - cnoHotSpare;
+ ndbrequire(tmngLimit < MAX_NDB_NODES);
+ for (Uint32 i = 0; i < tmngLimit; i++) {
+ NodeGroupRecordPtr NGPtr;
+ jam();
+ tmngNode = nodeArray[i];
+ mngNodeptr.i = tmngNode;
+ ptrCheckGuard(mngNodeptr, MAX_NDB_NODES, nodeRecord);
+ mngNodeptr.p->nodeGroup = tmngNodeGroup;
+ NGPtr.i = tmngNodeGroup;
+ ptrCheckGuard(NGPtr, MAX_NDB_NODES, nodeGroupRecord);
+ arrGuard(tmngReplica, MAX_REPLICAS);
+ NGPtr.p->nodesInGroup[tmngReplica] = mngNodeptr.i;
+ tmngReplica++;
+ if (tmngReplica == cnoReplicas) {
+ jam();
+ tmngNodeGroup++;
+ tmngReplica = 0;
+ }//if
+ }//for
+ cnoOfNodeGroups = tmngNodeGroup;
+ ndbrequire(csystemnodes < MAX_NDB_NODES);
+ for (Uint32 i = tmngLimit + 1; i < csystemnodes; i++) {
+ jam();
+ tmngNode = nodeArray[i];
+ mngNodeptr.i = tmngNode;
+ ptrCheckGuard(mngNodeptr, MAX_NDB_NODES, nodeRecord);
+ mngNodeptr.p->nodeGroup = ZNIL;
+ }//for
+ for(int i = 0; i < MAX_NDB_NODES; i++){
+ jam();
+ Sysfile::setNodeGroup(i, SYSFILE->nodeGroups, NO_NODE_GROUP_ID);
+ }//for
+ for (mngNodeptr.i = 1; mngNodeptr.i < MAX_NDB_NODES; mngNodeptr.i++) {
+ jam();
+ ptrAss(mngNodeptr, nodeRecord);
+ if (mngNodeptr.p->nodeGroup != ZNIL) {
+ jam();
+ Sysfile::setNodeGroup(mngNodeptr.i, SYSFILE->nodeGroups, mngNodeptr.p->nodeGroup);
+ }//if
+ }//for
+}//Dbdih::makeNodeGroups()
+
+/**
+ * On node failure QMGR asks DIH about node groups. This is
+ * a direct signal (function call in same process). Input is
+ * bitmask of surviving nodes. The routine is not concerned
+ * about node count. Reply is one of:
+ * 1) win - we can survive, and nobody else can
+ * 2) lose - we cannot survive
+ * 3) partition - we can survive but there could be others
+ */
+void Dbdih::execCHECKNODEGROUPSREQ(Signal* signal)
+{
+ jamEntry();
+ CheckNodeGroups* sd = (CheckNodeGroups*)&signal->theData[0];
+
+ bool direct = (sd->requestType & CheckNodeGroups::Direct);
+ bool ok = false;
+ switch(sd->requestType & ~CheckNodeGroups::Direct){
+ case CheckNodeGroups::ArbitCheck:{
+ ok = true;
+ jam();
+ unsigned missall = 0;
+ unsigned haveall = 0;
+ for (Uint32 i = 0; i < cnoOfNodeGroups; i++) {
+ jam();
+ NodeGroupRecordPtr ngPtr;
+ ngPtr.i = i;
+ ptrAss(ngPtr, nodeGroupRecord);
+ Uint32 count = 0;
+ for (Uint32 j = 0; j < ngPtr.p->nodeCount; j++) {
+ jam();
+ Uint32 nodeId = ngPtr.p->nodesInGroup[j];
+ if (sd->mask.get(nodeId)) {
+ jam();
+ count++;
+ }//if
+ }//for
+ if (count == 0) {
+ jam();
+ missall++;
+ }//if
+ if (count == ngPtr.p->nodeCount) {
+ haveall++;
+ }//if
+ }//for
+
+ if (missall) {
+ jam();
+ sd->output = CheckNodeGroups::Lose;
+ } else if (haveall) {
+ jam();
+ sd->output = CheckNodeGroups::Win;
+ } else {
+ jam();
+ sd->output = CheckNodeGroups::Partitioning;
+ }//if
+ }
+ break;
+ case CheckNodeGroups::GetNodeGroup:
+ ok = true;
+ sd->output = Sysfile::getNodeGroup(getOwnNodeId(), SYSFILE->nodeGroups);
+ break;
+ case CheckNodeGroups::GetNodeGroupMembers: {
+ ok = true;
+ Uint32 ownNodeGoup =
+ Sysfile::getNodeGroup(sd->nodeId, SYSFILE->nodeGroups);
+
+ sd->output = ownNodeGoup;
+ sd->mask.clear();
+
+ NodeGroupRecordPtr ngPtr;
+ ngPtr.i = ownNodeGoup;
+ ptrAss(ngPtr, nodeGroupRecord);
+ for (Uint32 j = 0; j < ngPtr.p->nodeCount; j++) {
+ jam();
+ sd->mask.set(ngPtr.p->nodesInGroup[j]);
+ }
+#if 0
+ for (int i = 0; i < MAX_NDB_NODES; i++) {
+ if (ownNodeGoup ==
+ Sysfile::getNodeGroup(i, SYSFILE->nodeGroups)) {
+ sd->mask.set(i);
+ }
+ }
+#endif
+ }
+ break;
+ }
+ ndbrequire(ok);
+
+ if (!direct)
+ sendSignal(sd->blockRef, GSN_CHECKNODEGROUPSCONF, signal,
+ CheckNodeGroups::SignalLength, JBB);
+}//Dbdih::execCHECKNODEGROUPSREQ()
+
+void Dbdih::makePrnList(ReadNodesConf * readNodes, Uint32 nodeArray[])
+{
+ cfirstAliveNode = RNIL;
+ ndbrequire(con_lineNodes > 0);
+ ndbrequire(csystemnodes < MAX_NDB_NODES);
+ for (Uint32 i = 0; i < csystemnodes; i++) {
+ NodeRecordPtr nodePtr;
+ jam();
+ nodePtr.i = nodeArray[i];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ initNodeState(nodePtr);
+ if (NodeBitmask::get(readNodes->inactiveNodes, nodePtr.i) == false){
+ jam();
+ nodePtr.p->nodeStatus = NodeRecord::ALIVE;
+ nodePtr.p->useInTransactions = true;
+ nodePtr.p->copyCompleted = true;
+ nodePtr.p->m_inclDihLcp = true;
+ insertAlive(nodePtr);
+ } else {
+ jam();
+ nodePtr.p->nodeStatus = NodeRecord::DEAD;
+ insertDeadNode(nodePtr);
+ }//if
+ nodePtr.p->ndbversion = readNodes->getVersionId(nodePtr.i,
+ readNodes->theVersionIds);
+ }//for
+}//Dbdih::makePrnList()
+
+/*************************************************************************/
+/* A NEW CRASHED REPLICA IS ADDED BY A NODE FAILURE. */
+/*************************************************************************/
+void Dbdih::newCrashedReplica(Uint32 nodeId, ReplicaRecordPtr ncrReplicaPtr)
+{
+ /*----------------------------------------------------------------------*/
+ /* SET THE REPLICA_LAST_GCI OF THE CRASHED REPLICA TO LAST GCI */
+ /* EXECUTED BY THE FAILED NODE. */
+ /*----------------------------------------------------------------------*/
+ /* WE HAVE A NEW CRASHED REPLICA. INITIATE CREATE GCI TO INDICATE */
+ /* THAT THE NEW REPLICA IS NOT STARTED YET AND REPLICA_LAST_GCI IS*/
+ /* SET TO -1 TO INDICATE THAT IT IS NOT DEAD YET. */
+ /*----------------------------------------------------------------------*/
+ arrGuard(ncrReplicaPtr.p->noCrashedReplicas + 1, 8);
+ ncrReplicaPtr.p->replicaLastGci[ncrReplicaPtr.p->noCrashedReplicas] =
+ SYSFILE->lastCompletedGCI[nodeId];
+ ncrReplicaPtr.p->noCrashedReplicas = ncrReplicaPtr.p->noCrashedReplicas + 1;
+ ncrReplicaPtr.p->createGci[ncrReplicaPtr.p->noCrashedReplicas] = 0;
+ ncrReplicaPtr.p->replicaLastGci[ncrReplicaPtr.p->noCrashedReplicas] =
+ (Uint32)-1;
+}//Dbdih::newCrashedReplica()
+
+/*************************************************************************/
+/* AT NODE FAILURE DURING START OF A NEW NODE WE NEED TO RESET A */
+/* SET OF VARIABLES CONTROLLING THE START AND INDICATING ONGOING */
+/* START OF A NEW NODE. */
+/*************************************************************************/
+void Dbdih::nodeResetStart()
+{
+ jam();
+ c_nodeStartMaster.startNode = RNIL;
+ c_nodeStartMaster.failNr = cfailurenr;
+ c_nodeStartMaster.activeState = false;
+ c_nodeStartMaster.blockGcp = false;
+ c_nodeStartMaster.blockLcp = false;
+ c_nodeStartMaster.m_outstandingGsn = 0;
+}//Dbdih::nodeResetStart()
+
+void Dbdih::openFileRw(Signal* signal, FileRecordPtr filePtr)
+{
+ signal->theData[0] = reference();
+ signal->theData[1] = filePtr.i;
+ signal->theData[2] = filePtr.p->fileName[0];
+ signal->theData[3] = filePtr.p->fileName[1];
+ signal->theData[4] = filePtr.p->fileName[2];
+ signal->theData[5] = filePtr.p->fileName[3];
+ signal->theData[6] = ZOPEN_READ_WRITE;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+}//Dbdih::openFileRw()
+
+/*************************************************************************/
+/* REMOVE A CRASHED REPLICA BY PACKING THE ARRAY OF CREATED GCI AND*/
+/* THE LAST GCI OF THE CRASHED REPLICA. */
+/*************************************************************************/
+void Dbdih::packCrashedReplicas(ReplicaRecordPtr replicaPtr)
+{
+ ndbrequire(replicaPtr.p->noCrashedReplicas > 0);
+ ndbrequire(replicaPtr.p->noCrashedReplicas <= 8);
+ for (Uint32 i = 0; i < replicaPtr.p->noCrashedReplicas; i++) {
+ jam();
+ replicaPtr.p->createGci[i] = replicaPtr.p->createGci[i + 1];
+ replicaPtr.p->replicaLastGci[i] = replicaPtr.p->replicaLastGci[i + 1];
+ }//for
+ replicaPtr.p->noCrashedReplicas--;
+
+#ifdef VM_TRACE
+ for (Uint32 i = 0; i < replicaPtr.p->noCrashedReplicas; i++) {
+ jam();
+ ndbrequire(replicaPtr.p->createGci[i] != 0xF1F1F1F1);
+ ndbrequire(replicaPtr.p->replicaLastGci[i] != 0xF1F1F1F1);
+ }//for
+#endif
+}//Dbdih::packCrashedReplicas()
+
+void Dbdih::prepareReplicas(FragmentstorePtr fragPtr)
+{
+ ReplicaRecordPtr prReplicaPtr;
+ Uint32 prevReplica = RNIL;
+
+ /* --------------------------------------------------------------------- */
+ /* BEGIN BY LINKING ALL REPLICA RECORDS ONTO THE OLD STORED REPLICA*/
+ /* LIST. */
+ /* AT A SYSTEM RESTART OBVIOUSLY ALL NODES ARE OLD. */
+ /* --------------------------------------------------------------------- */
+ prReplicaPtr.i = fragPtr.p->storedReplicas;
+ while (prReplicaPtr.i != RNIL) {
+ jam();
+ prevReplica = prReplicaPtr.i;
+ ptrCheckGuard(prReplicaPtr, creplicaFileSize, replicaRecord);
+ prReplicaPtr.i = prReplicaPtr.p->nextReplica;
+ }//while
+ /* --------------------------------------------------------------------- */
+ /* LIST OF STORED REPLICAS WILL BE EMPTY NOW. */
+ /* --------------------------------------------------------------------- */
+ if (prevReplica != RNIL) {
+ prReplicaPtr.i = prevReplica;
+ ptrCheckGuard(prReplicaPtr, creplicaFileSize, replicaRecord);
+ prReplicaPtr.p->nextReplica = fragPtr.p->oldStoredReplicas;
+ fragPtr.p->oldStoredReplicas = fragPtr.p->storedReplicas;
+ fragPtr.p->storedReplicas = RNIL;
+ fragPtr.p->noOldStoredReplicas += fragPtr.p->noStoredReplicas;
+ fragPtr.p->noStoredReplicas = 0;
+ }//if
+}//Dbdih::prepareReplicas()
+
+void Dbdih::readFragment(RWFragment* rf, FragmentstorePtr fragPtr)
+{
+ Uint32 TreadFid = readPageWord(rf);
+ fragPtr.p->preferredPrimary = readPageWord(rf);
+ fragPtr.p->noStoredReplicas = readPageWord(rf);
+ fragPtr.p->noOldStoredReplicas = readPageWord(rf);
+ Uint32 TdistKey = readPageWord(rf);
+
+ ndbrequire(fragPtr.p->noStoredReplicas > 0);
+ ndbrequire(TreadFid == rf->fragId);
+ ndbrequire(TdistKey < 256);
+ if ((cstarttype == NodeState::ST_NODE_RESTART) ||
+ (cstarttype == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+ fragPtr.p->distributionKey = TdistKey;
+ }//if
+}//Dbdih::readFragment()
+
+Uint32 Dbdih::readPageWord(RWFragment* rf)
+{
+ if (rf->wordIndex >= 2048) {
+ jam();
+ ndbrequire(rf->wordIndex == 2048);
+ rf->pageIndex++;
+ ndbrequire(rf->pageIndex < 8);
+ rf->rwfPageptr.i = rf->rwfTabPtr.p->pageRef[rf->pageIndex];
+ ptrCheckGuard(rf->rwfPageptr, cpageFileSize, pageRecord);
+ rf->wordIndex = 32;
+ }//if
+ Uint32 dataWord = rf->rwfPageptr.p->word[rf->wordIndex];
+ rf->wordIndex++;
+ return dataWord;
+}//Dbdih::readPageWord()
+
+void Dbdih::readReplica(RWFragment* rf, ReplicaRecordPtr readReplicaPtr)
+{
+ readReplicaPtr.p->procNode = readPageWord(rf);
+ readReplicaPtr.p->initialGci = readPageWord(rf);
+ readReplicaPtr.p->noCrashedReplicas = readPageWord(rf);
+ readReplicaPtr.p->nextLcp = readPageWord(rf);
+
+ for (Uint32 i = 0; i < MAX_LCP_STORED; i++) {
+ readReplicaPtr.p->maxGciCompleted[i] = readPageWord(rf);
+ readReplicaPtr.p->maxGciStarted[i] = readPageWord(rf);
+ readReplicaPtr.p->lcpId[i] = readPageWord(rf);
+ readReplicaPtr.p->lcpStatus[i] = readPageWord(rf);
+ }//for
+ const Uint32 noCrashedReplicas = readReplicaPtr.p->noCrashedReplicas;
+ ndbrequire(noCrashedReplicas < 8);
+ for (Uint32 i = 0; i < noCrashedReplicas; i++) {
+ readReplicaPtr.p->createGci[i] = readPageWord(rf);
+ readReplicaPtr.p->replicaLastGci[i] = readPageWord(rf);
+ ndbrequire(readReplicaPtr.p->createGci[i] != 0xF1F1F1F1);
+ ndbrequire(readReplicaPtr.p->replicaLastGci[i] != 0xF1F1F1F1);
+ }//for
+ for(Uint32 i = noCrashedReplicas; i<8; i++){
+ readReplicaPtr.p->createGci[i] = readPageWord(rf);
+ readReplicaPtr.p->replicaLastGci[i] = readPageWord(rf);
+ // They are not initialized...
+ readReplicaPtr.p->createGci[i] = 0;
+ readReplicaPtr.p->replicaLastGci[i] = ~0;
+ }
+ /* ---------------------------------------------------------------------- */
+ /* IF THE LAST COMPLETED LOCAL CHECKPOINT IS VALID AND LARGER THAN */
+ /* THE LAST COMPLETED CHECKPOINT THEN WE WILL INVALIDATE THIS LOCAL */
+ /* CHECKPOINT FOR THIS REPLICA. */
+ /* ---------------------------------------------------------------------- */
+ Uint32 trraLcp = prevLcpNo(readReplicaPtr.p->nextLcp);
+ ndbrequire(trraLcp < MAX_LCP_STORED);
+ if ((readReplicaPtr.p->lcpStatus[trraLcp] == ZVALID) &&
+ (readReplicaPtr.p->lcpId[trraLcp] > SYSFILE->latestLCP_ID)) {
+ jam();
+ readReplicaPtr.p->lcpStatus[trraLcp] = ZINVALID;
+ }//if
+ /* ---------------------------------------------------------------------- */
+ /* WE ALSO HAVE TO INVALIDATE ANY LOCAL CHECKPOINTS THAT HAVE BEEN */
+ /* INVALIDATED BY MOVING BACK THE RESTART GCI. */
+ /* ---------------------------------------------------------------------- */
+ for (Uint32 i = 0; i < MAX_LCP_STORED; i++) {
+ jam();
+ if ((readReplicaPtr.p->lcpStatus[i] == ZVALID) &&
+ (readReplicaPtr.p->maxGciStarted[i] > SYSFILE->newestRestorableGCI)) {
+ jam();
+ readReplicaPtr.p->lcpStatus[i] = ZINVALID;
+ }//if
+ }//for
+ /* ---------------------------------------------------------------------- */
+ /* WE WILL REMOVE ANY OCCURRENCES OF REPLICAS THAT HAVE CRASHED */
+ /* THAT ARE NO LONGER VALID DUE TO MOVING RESTART GCI BACKWARDS. */
+ /* ---------------------------------------------------------------------- */
+ removeTooNewCrashedReplicas(readReplicaPtr);
+ /* ---------------------------------------------------------------------- */
+ /* WE WILL REMOVE ANY OCCURRENCES OF REPLICAS THAT HAVE CRASHED */
+ /* THAT ARE NO LONGER VALID SINCE THEY ARE NO LONGER RESTORABLE. */
+ /* ---------------------------------------------------------------------- */
+ removeOldCrashedReplicas(readReplicaPtr);
+ /* --------------------------------------------------------------------- */
+ // We set the last GCI of the replica that was alive before the node
+ // crashed last time. We set it to the last GCI which the node participated in.
+ /* --------------------------------------------------------------------- */
+ ndbrequire(readReplicaPtr.p->noCrashedReplicas < 8);
+ readReplicaPtr.p->replicaLastGci[readReplicaPtr.p->noCrashedReplicas] =
+ SYSFILE->lastCompletedGCI[readReplicaPtr.p->procNode];
+ /* ---------------------------------------------------------------------- */
+ /* FIND PROCESSOR RECORD */
+ /* ---------------------------------------------------------------------- */
+}//Dbdih::readReplica()
+
+void Dbdih::readReplicas(RWFragment* rf, FragmentstorePtr fragPtr)
+{
+ ReplicaRecordPtr newReplicaPtr;
+ Uint32 noStoredReplicas = fragPtr.p->noStoredReplicas;
+ Uint32 noOldStoredReplicas = fragPtr.p->noOldStoredReplicas;
+ /* ----------------------------------------------------------------------- */
+ /* WE CLEAR THE NUMBER OF STORED REPLICAS SINCE IT WILL BE CALCULATED */
+ /* BY THE LINKING SUBROUTINES. */
+ /* ----------------------------------------------------------------------- */
+ fragPtr.p->noStoredReplicas = 0;
+ fragPtr.p->noOldStoredReplicas = 0;
+ Uint32 replicaIndex = 0;
+ ndbrequire(noStoredReplicas + noOldStoredReplicas <= MAX_REPLICAS);
+ for (Uint32 i = 0; i < noStoredReplicas; i++) {
+ seizeReplicaRec(newReplicaPtr);
+ readReplica(rf, newReplicaPtr);
+ if (checkNodeAlive(newReplicaPtr.p->procNode)) {
+ jam();
+ ndbrequire(replicaIndex < MAX_REPLICAS);
+ fragPtr.p->activeNodes[replicaIndex] = newReplicaPtr.p->procNode;
+ replicaIndex++;
+ linkStoredReplica(fragPtr, newReplicaPtr);
+ } else {
+ jam();
+ linkOldStoredReplica(fragPtr, newReplicaPtr);
+ }//if
+ }//for
+ fragPtr.p->fragReplicas = noStoredReplicas;
+ for (Uint32 i = 0; i < noOldStoredReplicas; i++) {
+ jam();
+ seizeReplicaRec(newReplicaPtr);
+ readReplica(rf, newReplicaPtr);
+ linkOldStoredReplica(fragPtr, newReplicaPtr);
+ }//for
+}//Dbdih::readReplicas()
+
+void Dbdih::readRestorableGci(Signal* signal, FileRecordPtr filePtr)
+{
+ signal->theData[0] = filePtr.p->fileRef;
+ signal->theData[1] = reference();
+ signal->theData[2] = filePtr.i;
+ signal->theData[3] = ZLIST_OF_PAIRS;
+ signal->theData[4] = ZVAR_NO_CRESTART_INFO;
+ signal->theData[5] = 1;
+ signal->theData[6] = 0;
+ signal->theData[7] = 0;
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 8, JBA);
+}//Dbdih::readRestorableGci()
+
+void Dbdih::readTabfile(Signal* signal, TabRecord* tab, FileRecordPtr filePtr)
+{
+ signal->theData[0] = filePtr.p->fileRef;
+ signal->theData[1] = reference();
+ signal->theData[2] = filePtr.i;
+ signal->theData[3] = ZLIST_OF_PAIRS;
+ signal->theData[4] = ZVAR_NO_WORD;
+ signal->theData[5] = tab->noPages;
+ for (Uint32 i = 0; i < tab->noPages; i++) {
+ signal->theData[6 + (2 * i)] = tab->pageRef[i];
+ signal->theData[7 + (2 * i)] = i;
+ }//for
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 22, JBA);
+}//Dbdih::readTabfile()
+
+void Dbdih::releasePage(Uint32 pageIndex)
+{
+ PageRecordPtr pagePtr;
+ pagePtr.i = pageIndex;
+ ptrCheckGuard(pagePtr, cpageFileSize, pageRecord);
+ pagePtr.p->nextfreepage = cfirstfreepage;
+ cfirstfreepage = pagePtr.i;
+}//Dbdih::releasePage()
+
+void Dbdih::releaseTabPages(Uint32 tableId)
+{
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+ ndbrequire(tabPtr.p->noPages <= 8);
+ for (Uint32 i = 0; i < tabPtr.p->noPages; i++) {
+ jam();
+ releasePage(tabPtr.p->pageRef[i]);
+ }//for
+ tabPtr.p->noPages = 0;
+}//Dbdih::releaseTabPages()
+
+/*************************************************************************/
+/* REMOVE NODE FROM SET OF ALIVE NODES. */
+/*************************************************************************/
+void Dbdih::removeAlive(NodeRecordPtr removeNodePtr)
+{
+ NodeRecordPtr nodePtr;
+
+ nodePtr.i = cfirstAliveNode;
+ if (nodePtr.i == removeNodePtr.i) {
+ jam();
+ cfirstAliveNode = removeNodePtr.p->nextNode;
+ return;
+ }//if
+ do {
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.p->nextNode == removeNodePtr.i) {
+ jam();
+ nodePtr.p->nextNode = removeNodePtr.p->nextNode;
+ break;
+ } else {
+ jam();
+ nodePtr.i = nodePtr.p->nextNode;
+ }//if
+ } while (1);
+}//Dbdih::removeAlive()
+
+/*************************************************************************/
+/* REMOVE NODE FROM SET OF DEAD NODES. */
+/*************************************************************************/
+void Dbdih::removeDeadNode(NodeRecordPtr removeNodePtr)
+{
+ NodeRecordPtr nodePtr;
+
+ nodePtr.i = cfirstDeadNode;
+ if (nodePtr.i == removeNodePtr.i) {
+ jam();
+ cfirstDeadNode = removeNodePtr.p->nextNode;
+ return;
+ }//if
+ do {
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.p->nextNode == removeNodePtr.i) {
+ jam();
+ nodePtr.p->nextNode = removeNodePtr.p->nextNode;
+ break;
+ } else {
+ jam();
+ nodePtr.i = nodePtr.p->nextNode;
+ }//if
+ } while (1);
+}//Dbdih::removeDeadNode()
+
+/*---------------------------------------------------------------*/
+/* REMOVE REPLICAS OF A FAILED NODE FROM LIST OF STORED */
+/* REPLICAS AND MOVE IT TO THE LIST OF OLD STORED REPLICAS.*/
+/* ALSO UPDATE THE CRASHED REPLICA INFORMATION. */
+/*---------------------------------------------------------------*/
+void Dbdih::removeNodeFromStored(Uint32 nodeId,
+ FragmentstorePtr fragPtr,
+ ReplicaRecordPtr replicatePtr)
+{
+ newCrashedReplica(nodeId, replicatePtr);
+ removeStoredReplica(fragPtr, replicatePtr);
+ linkOldStoredReplica(fragPtr, replicatePtr);
+ ndbrequire(fragPtr.p->storedReplicas != RNIL);
+}//Dbdih::removeNodeFromStored()
+
+/*************************************************************************/
+/* REMOVE ANY OLD CRASHED REPLICAS THAT ARE NOT RESTORABLE ANY MORE*/
+/*************************************************************************/
+void Dbdih::removeOldCrashedReplicas(ReplicaRecordPtr rocReplicaPtr)
+{
+ while (rocReplicaPtr.p->noCrashedReplicas > 0) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* ONLY IF THERE IS AT LEAST ONE REPLICA THEN CAN WE REMOVE ANY. */
+ /* --------------------------------------------------------------------- */
+ if (rocReplicaPtr.p->replicaLastGci[0] < SYSFILE->oldestRestorableGCI){
+ jam();
+ /* ------------------------------------------------------------------- */
+ /* THIS CRASHED REPLICA HAS BECOME EXTINCT AND MUST BE REMOVED TO */
+ /* GIVE SPACE FOR NEW CRASHED REPLICAS. */
+ /* ------------------------------------------------------------------- */
+ packCrashedReplicas(rocReplicaPtr);
+ } else {
+ break;
+ }//if
+ }//while
+ if (rocReplicaPtr.p->createGci[0] < SYSFILE->keepGCI){
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* MOVE FORWARD THE CREATE GCI TO A GCI THAT CAN BE USED. WE HAVE */
+ /* NO CERTAINTY IN FINDING ANY LOG RECORDS FROM OLDER GCI'S. */
+ /* --------------------------------------------------------------------- */
+ rocReplicaPtr.p->createGci[0] = SYSFILE->keepGCI;
+ ndbrequire(SYSFILE->keepGCI != 0xF1F1F1F1);
+ }//if
+}//Dbdih::removeOldCrashedReplicas()
+
+void Dbdih::removeOldStoredReplica(FragmentstorePtr fragPtr,
+ ReplicaRecordPtr replicatePtr)
+{
+ ReplicaRecordPtr rosTmpReplicaPtr;
+ ReplicaRecordPtr rosPrevReplicaPtr;
+
+ fragPtr.p->noOldStoredReplicas--;
+ if (fragPtr.p->oldStoredReplicas == replicatePtr.i) {
+ jam();
+ fragPtr.p->oldStoredReplicas = replicatePtr.p->nextReplica;
+ } else {
+ rosPrevReplicaPtr.i = fragPtr.p->oldStoredReplicas;
+ ptrCheckGuard(rosPrevReplicaPtr, creplicaFileSize, replicaRecord);
+ rosTmpReplicaPtr.i = rosPrevReplicaPtr.p->nextReplica;
+ while (rosTmpReplicaPtr.i != replicatePtr.i) {
+ jam();
+ rosPrevReplicaPtr.i = rosTmpReplicaPtr.i;
+ ptrCheckGuard(rosPrevReplicaPtr, creplicaFileSize, replicaRecord);
+ ptrCheckGuard(rosTmpReplicaPtr, creplicaFileSize, replicaRecord);
+ rosTmpReplicaPtr.i = rosTmpReplicaPtr.p->nextReplica;
+ }//if
+ rosPrevReplicaPtr.p->nextReplica = replicatePtr.p->nextReplica;
+ }//if
+}//Dbdih::removeOldStoredReplica()
+
+void Dbdih::removeStoredReplica(FragmentstorePtr fragPtr,
+ ReplicaRecordPtr replicatePtr)
+{
+ ReplicaRecordPtr rsrTmpReplicaPtr;
+ ReplicaRecordPtr rsrPrevReplicaPtr;
+
+ fragPtr.p->noStoredReplicas--;
+ if (fragPtr.p->storedReplicas == replicatePtr.i) {
+ jam();
+ fragPtr.p->storedReplicas = replicatePtr.p->nextReplica;
+ } else {
+ jam();
+ rsrPrevReplicaPtr.i = fragPtr.p->storedReplicas;
+ rsrTmpReplicaPtr.i = fragPtr.p->storedReplicas;
+ ptrCheckGuard(rsrTmpReplicaPtr, creplicaFileSize, replicaRecord);
+ rsrTmpReplicaPtr.i = rsrTmpReplicaPtr.p->nextReplica;
+ while (rsrTmpReplicaPtr.i != replicatePtr.i) {
+ jam();
+ rsrPrevReplicaPtr.i = rsrTmpReplicaPtr.i;
+ ptrCheckGuard(rsrTmpReplicaPtr, creplicaFileSize, replicaRecord);
+ rsrTmpReplicaPtr.i = rsrTmpReplicaPtr.p->nextReplica;
+ }//while
+ ptrCheckGuard(rsrPrevReplicaPtr, creplicaFileSize, replicaRecord);
+ rsrPrevReplicaPtr.p->nextReplica = replicatePtr.p->nextReplica;
+ }//if
+}//Dbdih::removeStoredReplica()
+
+/*************************************************************************/
+/* REMOVE ALL TOO NEW CRASHED REPLICAS THAT IS IN THIS REPLICA. */
+/*************************************************************************/
+void Dbdih::removeTooNewCrashedReplicas(ReplicaRecordPtr rtnReplicaPtr)
+{
+ while (rtnReplicaPtr.p->noCrashedReplicas > 0) {
+ jam();
+ /* --------------------------------------------------------------------- */
+ /* REMOVE ALL REPLICAS THAT ONLY LIVED IN A PERIOD THAT HAVE BEEN */
+ /* REMOVED FROM THE RESTART INFORMATION SINCE THE RESTART FAILED */
+ /* TOO MANY TIMES. */
+ /* --------------------------------------------------------------------- */
+ arrGuard(rtnReplicaPtr.p->noCrashedReplicas - 1, 8);
+ if (rtnReplicaPtr.p->createGci[rtnReplicaPtr.p->noCrashedReplicas - 1] >
+ SYSFILE->newestRestorableGCI){
+ jam();
+ rtnReplicaPtr.p->createGci[rtnReplicaPtr.p->noCrashedReplicas - 1] =
+ (Uint32)-1;
+ rtnReplicaPtr.p->replicaLastGci[rtnReplicaPtr.p->noCrashedReplicas - 1] =
+ (Uint32)-1;
+ rtnReplicaPtr.p->noCrashedReplicas--;
+ } else {
+ break;
+ }//if
+ }//while
+}//Dbdih::removeTooNewCrashedReplicas()
+
+/*************************************************************************/
+/* */
+/* MODULE: SEARCH FOR POSSIBLE REPLICAS THAT CAN HANDLE THE GLOBAL */
+/* CHECKPOINT WITHOUT NEEDING ANY EXTRA LOGGING FACILITIES.*/
+/* A MAXIMUM OF FOUR NODES IS RETRIEVED. */
+/*************************************************************************/
+void Dbdih::searchStoredReplicas(FragmentstorePtr fragPtr)
+{
+ Uint32 nextReplicaPtrI;
+ ConstPtr<ReplicaRecord> replicaPtr;
+
+ replicaPtr.i = fragPtr.p->storedReplicas;
+ while (replicaPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ nextReplicaPtrI = replicaPtr.p->nextReplica;
+ NodeRecordPtr nodePtr;
+ nodePtr.i = replicaPtr.p->procNode;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.p->nodeStatus == NodeRecord::ALIVE) {
+ jam();
+ switch (nodePtr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ case Sysfile::NS_ActiveMissed_1:
+ case Sysfile::NS_ActiveMissed_2:{
+ /* ----------------------------------------------------------------- */
+ /* INITIALISE THE CREATE REPLICA STRUCTURE THAT IS USED FOR SENDING*/
+ /* TO LQH START_FRAGREQ. */
+ /* SET THE DATA NODE WHERE THE LOCAL CHECKPOINT IS FOUND. ALSO */
+ /* SET A REFERENCE TO THE REPLICA POINTER OF THAT. */
+ /* ----------------------------------------------------------------- */
+ CreateReplicaRecordPtr createReplicaPtr;
+ createReplicaPtr.i = cnoOfCreateReplicas;
+ ptrCheckGuard(createReplicaPtr, 4, createReplicaRecord);
+ cnoOfCreateReplicas++;
+ createReplicaPtr.p->dataNodeId = replicaPtr.p->procNode;
+ createReplicaPtr.p->replicaRec = replicaPtr.i;
+ /* ----------------------------------------------------------------- */
+ /* WE NEED TO SEARCH FOR A PROPER LOCAL CHECKPOINT TO USE FOR THE */
+ /* SYSTEM RESTART. */
+ /* ----------------------------------------------------------------- */
+ Uint32 startGci;
+ Uint32 startLcpNo;
+ Uint32 stopGci = SYSFILE->newestRestorableGCI;
+ bool result = findStartGci(replicaPtr,
+ stopGci,
+ startGci,
+ startLcpNo);
+ if (!result) {
+ jam();
+ /* --------------------------------------------------------------- */
+ /* WE COULD NOT FIND ANY LOCAL CHECKPOINT. THE FRAGMENT THUS DO NOT*/
+ /* CONTAIN ANY VALID LOCAL CHECKPOINT. IT DOES HOWEVER CONTAIN A */
+ /* VALID FRAGMENT LOG. THUS BY FIRST CREATING THE FRAGMENT AND THEN*/
+ /* EXECUTING THE FRAGMENT LOG WE CAN CREATE THE FRAGMENT AS */
+ /* DESIRED. THIS SHOULD ONLY OCCUR AFTER CREATING A FRAGMENT. */
+ /* */
+ /* TO INDICATE THAT NO LOCAL CHECKPOINT IS TO BE USED WE SET THE */
+ /* LOCAL CHECKPOINT TO ZNIL. */
+ /* --------------------------------------------------------------- */
+ createReplicaPtr.p->lcpNo = ZNIL;
+ } else {
+ jam();
+ /* --------------------------------------------------------------- */
+ /* WE FOUND A PROPER LOCAL CHECKPOINT TO RESTART FROM. */
+ /* SET LOCAL CHECKPOINT ID AND LOCAL CHECKPOINT NUMBER. */
+ /* --------------------------------------------------------------- */
+ createReplicaPtr.p->lcpNo = startLcpNo;
+ arrGuard(startLcpNo, MAX_LCP_STORED);
+ createReplicaPtr.p->createLcpId = replicaPtr.p->lcpId[startLcpNo];
+ }//if
+
+ if(ERROR_INSERTED(7073) || ERROR_INSERTED(7074)){
+ jam();
+ nodePtr.p->nodeStatus = NodeRecord::DEAD;
+ }
+
+ /* ----------------------------------------------------------------- */
+ /* WE HAVE EITHER FOUND A LOCAL CHECKPOINT OR WE ARE PLANNING TO */
+ /* EXECUTE THE LOG FROM THE INITIAL CREATION OF THE TABLE. IN BOTH */
+ /* CASES WE NEED TO FIND A SET OF LOGS THAT CAN EXECUTE SUCH THAT */
+ /* WE RECOVER TO THE SYSTEM RESTART GLOBAL CHECKPOINT. */
+ /* -_--------------------------------------------------------------- */
+ if (!findLogNodes(createReplicaPtr.p, fragPtr, startGci, stopGci)) {
+ jam();
+ /* --------------------------------------------------------------- */
+ /* WE WERE NOT ABLE TO FIND ANY WAY OF RESTORING THIS REPLICA. */
+ /* THIS IS A POTENTIAL SYSTEM ERROR. */
+ /* --------------------------------------------------------------- */
+ cnoOfCreateReplicas--;
+ return;
+ }//if
+
+ if(ERROR_INSERTED(7073) || ERROR_INSERTED(7074)){
+ jam();
+ nodePtr.p->nodeStatus = NodeRecord::ALIVE;
+ }
+
+ break;
+ }
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }//switch
+ }
+ replicaPtr.i = nextReplicaPtrI;
+ }//while
+}//Dbdih::searchStoredReplicas()
+
+/*************************************************************************/
+/* */
+/* MODULE: SEIZE_FILE */
+/* DESCRIPTION: THE SUBROUTINE SEIZES A FILE RECORD FROM THE */
+/* FREE LIST. */
+/*************************************************************************/
+void Dbdih::seizeFile(FileRecordPtr& filePtr)
+{
+ filePtr.i = cfirstfreeFile;
+ ptrCheckGuard(filePtr, cfileFileSize, fileRecord);
+ cfirstfreeFile = filePtr.p->nextFile;
+ filePtr.p->nextFile = RNIL;
+}//Dbdih::seizeFile()
+
+/*************************************************************************/
+/* SEND CREATE_FRAGREQ TO ALL NODES IN THE NDB CLUSTER. */
+/*************************************************************************/
+/*************************************************************************/
+/* */
+/* MODULE: FIND THE START GCI AND LOCAL CHECKPOINT TO USE. */
+/*************************************************************************/
+void Dbdih::sendStartFragreq(Signal* signal,
+ TabRecordPtr tabPtr, Uint32 fragId)
+{
+ CreateReplicaRecordPtr replicaPtr;
+ for (replicaPtr.i = 0; replicaPtr.i < cnoOfCreateReplicas; replicaPtr.i++) {
+ jam();
+ ptrAss(replicaPtr, createReplicaRecord);
+ BlockReference ref = calcLqhBlockRef(replicaPtr.p->dataNodeId);
+ StartFragReq * const startFragReq = (StartFragReq *)&signal->theData[0];
+ startFragReq->userPtr = replicaPtr.p->replicaRec;
+ startFragReq->userRef = reference();
+ startFragReq->lcpNo = replicaPtr.p->lcpNo;
+ startFragReq->lcpId = replicaPtr.p->createLcpId;
+ startFragReq->tableId = tabPtr.i;
+ startFragReq->fragId = fragId;
+
+ if(ERROR_INSERTED(7072) || ERROR_INSERTED(7074)){
+ jam();
+ const Uint32 noNodes = replicaPtr.p->noLogNodes;
+ Uint32 start = replicaPtr.p->logStartGci[noNodes - 1];
+ const Uint32 stop = replicaPtr.p->logStopGci[noNodes - 1];
+
+ for(Uint32 i = noNodes; i < 4 && (stop - start) > 0; i++){
+ replicaPtr.p->noLogNodes++;
+ replicaPtr.p->logStopGci[i - 1] = start;
+
+ replicaPtr.p->logNodeId[i] = replicaPtr.p->logNodeId[i-1];
+ replicaPtr.p->logStartGci[i] = start + 1;
+ replicaPtr.p->logStopGci[i] = stop;
+ start += 1;
+ }
+ }
+
+ startFragReq->noOfLogNodes = replicaPtr.p->noLogNodes;
+
+ for (Uint32 i = 0; i < 4 ; i++) {
+ startFragReq->lqhLogNode[i] = replicaPtr.p->logNodeId[i];
+ startFragReq->startGci[i] = replicaPtr.p->logStartGci[i];
+ startFragReq->lastGci[i] = replicaPtr.p->logStopGci[i];
+ }//for
+
+ sendSignal(ref, GSN_START_FRAGREQ, signal,
+ StartFragReq::SignalLength, JBB);
+ }//for
+}//Dbdih::sendStartFragreq()
+
+/*************************************************************************/
+/* SET THE INITIAL ACTIVE STATUS ON ALL NODES AND PUT INTO LISTS. */
+/*************************************************************************/
+void Dbdih::setInitialActiveStatus()
+{
+ NodeRecordPtr siaNodeptr;
+ Uint32 tsiaNodeActiveStatus;
+ Uint32 tsiaNoActiveNodes;
+
+ tsiaNoActiveNodes = csystemnodes - cnoHotSpare;
+ for(Uint32 i = 0; i<Sysfile::NODE_STATUS_SIZE; i++)
+ SYSFILE->nodeStatus[i] = 0;
+ for (siaNodeptr.i = 1; siaNodeptr.i < MAX_NDB_NODES; siaNodeptr.i++) {
+ ptrAss(siaNodeptr, nodeRecord);
+ if (siaNodeptr.p->nodeStatus == NodeRecord::ALIVE) {
+ if (tsiaNoActiveNodes == 0) {
+ jam();
+ siaNodeptr.p->activeStatus = Sysfile::NS_HotSpare;
+ } else {
+ jam();
+ tsiaNoActiveNodes = tsiaNoActiveNodes - 1;
+ siaNodeptr.p->activeStatus = Sysfile::NS_Active;
+ }//if
+ } else {
+ jam();
+ siaNodeptr.p->activeStatus = Sysfile::NS_NotDefined;
+ }//if
+ switch (siaNodeptr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ jam();
+ tsiaNodeActiveStatus = Sysfile::NS_Active;
+ break;
+ case Sysfile::NS_HotSpare:
+ jam();
+ tsiaNodeActiveStatus = Sysfile::NS_HotSpare;
+ break;
+ case Sysfile::NS_NotDefined:
+ jam();
+ tsiaNodeActiveStatus = Sysfile::NS_NotDefined;
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ break;
+ }//switch
+ Sysfile::setNodeStatus(siaNodeptr.i, SYSFILE->nodeStatus,
+ tsiaNodeActiveStatus);
+ }//for
+}//Dbdih::setInitialActiveStatus()
+
+/*************************************************************************/
+/* SET LCP ACTIVE STATUS AT THE END OF A LOCAL CHECKPOINT. */
+/*************************************************************************/
+void Dbdih::setLcpActiveStatusEnd()
+{
+ NodeRecordPtr nodePtr;
+
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if (c_lcpState.m_participatingLQH.get(nodePtr.i)){
+ switch (nodePtr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ case Sysfile::NS_ActiveMissed_1:
+ case Sysfile::NS_ActiveMissed_2:
+ jam();
+ /*-------------------------------------------------------------------*/
+ /* THE NODE PARTICIPATED IN THIS CHECKPOINT.
+ * WE CAN SET ITS STATUS TO ACTIVE */
+ /*-------------------------------------------------------------------*/
+ nodePtr.p->activeStatus = Sysfile::NS_Active;
+ takeOverCompleted(nodePtr.i);
+ break;
+ case Sysfile::NS_TakeOver:
+ jam();
+ /*-------------------------------------------------------------------*/
+ /* THE NODE HAS COMPLETED A CHECKPOINT AFTER TAKE OVER. WE CAN NOW */
+ /* SET ITS STATUS TO ACTIVE. WE CAN ALSO COMPLETE THE TAKE OVER */
+ /* AND ALSO WE CLEAR THE TAKE OVER NODE IN THE RESTART INFO. */
+ /*-------------------------------------------------------------------*/
+ nodePtr.p->activeStatus = Sysfile::NS_Active;
+ takeOverCompleted(nodePtr.i);
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ break;
+ }//switch
+ }//if
+ }//for
+
+ if(getNodeState().getNodeRestartInProgress()){
+ jam();
+ if(c_lcpState.m_participatingLQH.get(getOwnNodeId())){
+ nodePtr.i = getOwnNodeId();
+ ptrAss(nodePtr, nodeRecord);
+ ndbrequire(nodePtr.p->activeStatus == Sysfile::NS_Active);
+ ndbout_c("NR: setLcpActiveStatusEnd - m_participatingLQH");
+ } else {
+ ndbout_c("NR: setLcpActiveStatusEnd - !m_participatingLQH");
+ }
+ }
+
+ c_lcpState.m_participatingDIH.clear();
+ c_lcpState.m_participatingLQH.clear();
+ if (isMaster()) {
+ jam();
+ setNodeRestartInfoBits();
+ }//if
+}//Dbdih::setLcpActiveStatusEnd()
+
+void Dbdih::takeOverCompleted(Uint32 aNodeId)
+{
+ TakeOverRecordPtr takeOverPtr;
+ takeOverPtr.i = findTakeOver(aNodeId);
+ if (takeOverPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ if (takeOverPtr.p->toMasterStatus != TakeOverRecord::WAIT_LCP) {
+ jam();
+ ndbrequire(!isMaster());
+ return;
+ }//if
+ ndbrequire(isMaster());
+ Sysfile::setTakeOverNode(aNodeId, SYSFILE->takeOver, 0);
+ takeOverPtr.p->toMasterStatus = TakeOverRecord::TO_END_COPY;
+ cstartGcpNow = true;
+ }//if
+}//Dbdih::takeOverCompleted()
+
+/*************************************************************************/
+/* SET LCP ACTIVE STATUS BEFORE STARTING A LOCAL CHECKPOINT. */
+/*************************************************************************/
+void Dbdih::setLcpActiveStatusStart(Signal* signal)
+{
+ NodeRecordPtr nodePtr;
+
+ c_lcpState.m_participatingLQH.clear();
+ c_lcpState.m_participatingDIH.clear();
+
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRecord);
+#if 0
+ if(nodePtr.p->nodeStatus != NodeRecord::NOT_IN_CLUSTER){
+ infoEvent("Node %d nodeStatus=%d activeStatus=%d copyCompleted=%d lcp=%d",
+ nodePtr.i,
+ nodePtr.p->nodeStatus,
+ nodePtr.p->activeStatus,
+ nodePtr.p->copyCompleted,
+ nodePtr.p->m_inclDihLcp);
+ }
+#endif
+ if(nodePtr.p->nodeStatus == NodeRecord::ALIVE && nodePtr.p->m_inclDihLcp){
+ jam();
+ c_lcpState.m_participatingDIH.set(nodePtr.i);
+ }
+
+ if ((nodePtr.p->nodeStatus == NodeRecord::ALIVE) &&
+ (nodePtr.p->copyCompleted)) {
+ switch (nodePtr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ jam();
+ /*-------------------------------------------------------------------*/
+ // The normal case. Starting a LCP for a started node which hasn't
+ // missed the previous LCP.
+ /*-------------------------------------------------------------------*/
+ c_lcpState.m_participatingLQH.set(nodePtr.i);
+ break;
+ case Sysfile::NS_ActiveMissed_1:
+ jam();
+ /*-------------------------------------------------------------------*/
+ // The node is starting up and is participating in a local checkpoint
+ // as the final phase of the start-up. We can still use the checkpoints
+ // on the node after a system restart.
+ /*-------------------------------------------------------------------*/
+ c_lcpState.m_participatingLQH.set(nodePtr.i);
+ break;
+ case Sysfile::NS_ActiveMissed_2:
+ jam();
+ /*-------------------------------------------------------------------*/
+ // The node is starting up and is participating in a local checkpoint
+ // as the final phase of the start-up. We have missed so
+ // many checkpoints that we no longer can use this node to
+ // recreate fragments from disk.
+ // It must be taken over with the copy fragment process after a system
+ // crash. We indicate this by setting the active status to TAKE_OVER.
+ /*-------------------------------------------------------------------*/
+ nodePtr.p->activeStatus = Sysfile::NS_TakeOver;
+ //break; // Fall through
+ case Sysfile::NS_TakeOver:{
+ TakeOverRecordPtr takeOverPtr;
+ jam();
+ /*-------------------------------------------------------------------*/
+ /* THIS NODE IS CURRENTLY TAKING OVER A FAILED NODE. */
+ /*-------------------------------------------------------------------*/
+ takeOverPtr.i = findTakeOver(nodePtr.i);
+ if (takeOverPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(takeOverPtr, MAX_NDB_NODES, takeOverRecord);
+ if (takeOverPtr.p->toMasterStatus == TakeOverRecord::WAIT_LCP) {
+ jam();
+ /*---------------------------------------------------------------
+ * ALL THE INFORMATION HAVE BEEN REPLICATED TO THE NEW
+ * NODE AND WE ARE ONLY WAITING FOR A LOCAL CHECKPOINT TO BE
+ * PERFORMED ON THE NODE TO SET ITS STATUS TO ACTIVE.
+ */
+ infoEvent("Node %d is WAIT_LCP including in LCP", nodePtr.i);
+ c_lcpState.m_participatingLQH.set(nodePtr.i);
+ }//if
+ }//if
+ break;
+ }
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }//switch
+ } else {
+ switch (nodePtr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ jam();
+ nodePtr.p->activeStatus = Sysfile::NS_ActiveMissed_1;
+ break;
+ case Sysfile::NS_ActiveMissed_1:
+ jam();
+ nodePtr.p->activeStatus = Sysfile::NS_ActiveMissed_2;
+ break;
+ case Sysfile::NS_ActiveMissed_2:
+ jam();
+ if ((nodePtr.p->nodeStatus == NodeRecord::ALIVE) &&
+ (!nodePtr.p->copyCompleted)) {
+ jam();
+ /*-----------------------------------------------------------------*/
+ // The node is currently starting up and has not completed the
+ // copy phase.
+ // It will thus be in the TAKE_OVER state.
+ /*-----------------------------------------------------------------*/
+ ndbrequire(findTakeOver(nodePtr.i) != RNIL);
+ nodePtr.p->activeStatus = Sysfile::NS_TakeOver;
+ } else {
+ jam();
+ /*-----------------------------------------------------------------*/
+ /* THE NODE IS ACTIVE AND HAS NOT COMPLETED ANY OF THE LAST 3
+ * CHECKPOINTS */
+ /* WE MUST TAKE IT OUT OF ACTION AND START A NEW NODE TO TAKE OVER.*/
+ /*-----------------------------------------------------------------*/
+ nodePtr.p->activeStatus = Sysfile::NS_NotActive_NotTakenOver;
+ }//if
+ break;
+ case Sysfile::NS_TakeOver:
+ jam();
+ break;
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }//switch
+ }//if
+ }//for
+ if (isMaster()) {
+ jam();
+ checkStartTakeOver(signal);
+ setNodeRestartInfoBits();
+ }//if
+}//Dbdih::setLcpActiveStatusStart()
+
+/*************************************************************************/
+/* SET NODE ACTIVE STATUS AT SYSTEM RESTART AND WHEN UPDATED BY MASTER */
+/*************************************************************************/
+void Dbdih::setNodeActiveStatus()
+{
+ NodeRecordPtr snaNodeptr;
+
+ for (snaNodeptr.i = 1; snaNodeptr.i < MAX_NDB_NODES; snaNodeptr.i++) {
+ ptrAss(snaNodeptr, nodeRecord);
+ const Uint32 tsnaNodeBits = Sysfile::getNodeStatus(snaNodeptr.i,
+ SYSFILE->nodeStatus);
+ switch (tsnaNodeBits) {
+ case Sysfile::NS_Active:
+ jam();
+ snaNodeptr.p->activeStatus = Sysfile::NS_Active;
+ break;
+ case Sysfile::NS_ActiveMissed_1:
+ jam();
+ snaNodeptr.p->activeStatus = Sysfile::NS_ActiveMissed_1;
+ break;
+ case Sysfile::NS_ActiveMissed_2:
+ jam();
+ snaNodeptr.p->activeStatus = Sysfile::NS_ActiveMissed_2;
+ break;
+ case Sysfile::NS_TakeOver:
+ jam();
+ snaNodeptr.p->activeStatus = Sysfile::NS_TakeOver;
+ break;
+ case Sysfile::NS_HotSpare:
+ jam();
+ snaNodeptr.p->activeStatus = Sysfile::NS_HotSpare;
+ break;
+ case Sysfile::NS_NotActive_NotTakenOver:
+ jam();
+ snaNodeptr.p->activeStatus = Sysfile::NS_NotActive_NotTakenOver;
+ break;
+ case Sysfile::NS_NotDefined:
+ jam();
+ snaNodeptr.p->activeStatus = Sysfile::NS_NotDefined;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ }//for
+}//Dbdih::setNodeActiveStatus()
+
+/***************************************************************************/
+/* SET THE NODE GROUP BASED ON THE RESTART INFORMATION OR AS SET BY MASTER */
+/***************************************************************************/
+void Dbdih::setNodeGroups()
+{
+ NodeGroupRecordPtr NGPtr;
+ NodeRecordPtr sngNodeptr;
+ Uint32 Ti;
+
+ for (Ti = 0; Ti < MAX_NDB_NODES; Ti++) {
+ NGPtr.i = Ti;
+ ptrAss(NGPtr, nodeGroupRecord);
+ NGPtr.p->nodeCount = 0;
+ }//for
+ for (sngNodeptr.i = 1; sngNodeptr.i < MAX_NDB_NODES; sngNodeptr.i++) {
+ ptrAss(sngNodeptr, nodeRecord);
+ switch (sngNodeptr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ case Sysfile::NS_ActiveMissed_1:
+ case Sysfile::NS_ActiveMissed_2:
+ case Sysfile::NS_NotActive_NotTakenOver:
+ case Sysfile::NS_TakeOver:
+ jam();
+ sngNodeptr.p->nodeGroup = Sysfile::getNodeGroup(sngNodeptr.i,
+ SYSFILE->nodeGroups);
+ NGPtr.i = sngNodeptr.p->nodeGroup;
+ ptrCheckGuard(NGPtr, MAX_NDB_NODES, nodeGroupRecord);
+ NGPtr.p->nodesInGroup[NGPtr.p->nodeCount] = sngNodeptr.i;
+ NGPtr.p->nodeCount++;
+ break;
+ case Sysfile::NS_HotSpare:
+ case Sysfile::NS_NotDefined:
+ jam();
+ sngNodeptr.p->nodeGroup = ZNIL;
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ break;
+ }//switch
+ }//for
+ cnoOfNodeGroups = 0;
+ for (Ti = 0; Ti < MAX_NDB_NODES; Ti++) {
+ jam();
+ NGPtr.i = Ti;
+ ptrAss(NGPtr, nodeGroupRecord);
+ if (NGPtr.p->nodeCount != 0) {
+ jam();
+ cnoOfNodeGroups++;
+ }//if
+ }//for
+ cnoHotSpare = csystemnodes - (cnoOfNodeGroups * cnoReplicas);
+}//Dbdih::setNodeGroups()
+
+/*************************************************************************/
+/* SET NODE INFORMATION AFTER RECEIVING RESTART INFORMATION FROM MASTER. */
+/* WE TAKE THE OPPORTUNITY TO SYNCHRONISE OUR DATA WITH THE MASTER. IT */
+/* IS ONLY THE MASTER THAT WILL ACT ON THIS DATA. WE WILL KEEP THEM */
+/* UPDATED FOR THE CASE WHEN WE HAVE TO BECOME MASTER. */
+/*************************************************************************/
+void Dbdih::setNodeInfo(Signal* signal)
+{
+ setNodeActiveStatus();
+ setNodeGroups();
+ sendHOT_SPAREREP(signal);
+}//Dbdih::setNodeInfo()
+
+/*************************************************************************/
+// Keep also DBDICT informed about the Hot Spare situation in the cluster.
+/*************************************************************************/
+void Dbdih::sendHOT_SPAREREP(Signal* signal)
+{
+ NodeRecordPtr locNodeptr;
+ Uint32 Ti = 0;
+ HotSpareRep * const hotSpare = (HotSpareRep*)&signal->theData[0];
+ NodeBitmask::clear(hotSpare->theHotSpareNodes);
+ for (locNodeptr.i = 1; locNodeptr.i < MAX_NDB_NODES; locNodeptr.i++) {
+ ptrAss(locNodeptr, nodeRecord);
+ switch (locNodeptr.p->activeStatus) {
+ case Sysfile::NS_HotSpare:
+ jam();
+ NodeBitmask::set(hotSpare->theHotSpareNodes, locNodeptr.i);
+ Ti++;
+ break;
+ default:
+ jam();
+ break;
+ }//switch
+ }//for
+ hotSpare->noHotSpareNodes = Ti;
+ sendSignal(DBDICT_REF, GSN_HOT_SPAREREP,
+ signal, HotSpareRep::SignalLength, JBB);
+}//Dbdih::sendHOT_SPAREREP()
+
+/*************************************************************************/
+/* SET LCP ACTIVE STATUS FOR ALL NODES BASED ON THE INFORMATION IN */
+/* THE RESTART INFORMATION. */
+/*************************************************************************/
+#if 0
+void Dbdih::setNodeLcpActiveStatus()
+{
+ c_lcpState.m_lcpActiveStatus.clear();
+ for (Uint32 i = 1; i < MAX_NDB_NODES; i++) {
+ if (NodeBitmask::get(SYSFILE->lcpActive, i)) {
+ jam();
+ c_lcpState.m_lcpActiveStatus.set(i);
+ }//if
+ }//for
+}//Dbdih::setNodeLcpActiveStatus()
+#endif
+
+/*************************************************************************/
+/* SET THE RESTART INFO BITS BASED ON THE NODES ACTIVE STATUS. */
+/*************************************************************************/
+void Dbdih::setNodeRestartInfoBits()
+{
+ NodeRecordPtr nodePtr;
+ Uint32 tsnrNodeGroup;
+ Uint32 tsnrNodeActiveStatus;
+
+ for(int i = 1; i < MAX_NDB_NODES; i++){
+ Sysfile::setNodeStatus(i, SYSFILE->nodeStatus, Sysfile::NS_Active);
+ }//for
+ for(Uint32 i = 1; i < Sysfile::NODE_GROUPS_SIZE; i++){
+ SYSFILE->nodeGroups[i] = 0;
+ }//for
+ NdbNodeBitmask::clear(SYSFILE->lcpActive);
+
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRecord);
+ switch (nodePtr.p->activeStatus) {
+ case Sysfile::NS_Active:
+ jam();
+ tsnrNodeActiveStatus = Sysfile::NS_Active;
+ break;
+ case Sysfile::NS_ActiveMissed_1:
+ jam();
+ tsnrNodeActiveStatus = Sysfile::NS_ActiveMissed_1;
+ break;
+ case Sysfile::NS_ActiveMissed_2:
+ jam();
+ tsnrNodeActiveStatus = Sysfile::NS_ActiveMissed_2;
+ break;
+ case Sysfile::NS_HotSpare:
+ jam();
+ tsnrNodeActiveStatus = Sysfile::NS_HotSpare;
+ break;
+ case Sysfile::NS_TakeOver:
+ jam();
+ tsnrNodeActiveStatus = Sysfile::NS_TakeOver;
+ break;
+ case Sysfile::NS_NotActive_NotTakenOver:
+ jam();
+ tsnrNodeActiveStatus = Sysfile::NS_NotActive_NotTakenOver;
+ break;
+ case Sysfile::NS_NotDefined:
+ jam();
+ tsnrNodeActiveStatus = Sysfile::NS_NotDefined;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ Sysfile::setNodeStatus(nodePtr.i, SYSFILE->nodeStatus,
+ tsnrNodeActiveStatus);
+ if (nodePtr.p->nodeGroup == ZNIL) {
+ jam();
+ tsnrNodeGroup = NO_NODE_GROUP_ID;
+ } else {
+ jam();
+ tsnrNodeGroup = nodePtr.p->nodeGroup;
+ }//if
+ Sysfile::setNodeGroup(nodePtr.i, SYSFILE->nodeGroups, tsnrNodeGroup);
+ if (c_lcpState.m_participatingLQH.get(nodePtr.i)){
+ jam();
+ NodeBitmask::set(SYSFILE->lcpActive, nodePtr.i);
+ }//if
+ }//for
+}//Dbdih::setNodeRestartInfoBits()
+
+/*************************************************************************/
+/* START THE GLOBAL CHECKPOINT PROTOCOL IN MASTER AT START-UP */
+/*************************************************************************/
+void Dbdih::startGcp(Signal* signal)
+{
+ cgcpStatus = GCP_READY;
+ coldGcpStatus = cgcpStatus;
+ coldGcpId = cnewgcp;
+ cgcpSameCounter = 0;
+ signal->theData[0] = DihContinueB::ZSTART_GCP;
+ signal->theData[1] = 0;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+ signal->theData[0] = DihContinueB::ZCHECK_GCP_STOP;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 1);
+}//Dbdih::startGcp()
+
+void Dbdih::updateNodeInfo(FragmentstorePtr fragPtr)
+{
+ ReplicaRecordPtr replicatePtr;
+ Uint32 index = 0;
+ replicatePtr.i = fragPtr.p->storedReplicas;
+ do {
+ jam();
+ ptrCheckGuard(replicatePtr, creplicaFileSize, replicaRecord);
+ ndbrequire(index < MAX_REPLICAS);
+ fragPtr.p->activeNodes[index] = replicatePtr.p->procNode;
+ index++;
+ replicatePtr.i = replicatePtr.p->nextReplica;
+ } while (replicatePtr.i != RNIL);
+ fragPtr.p->fragReplicas = index;
+
+ /* ----------------------------------------------------------------------- */
+ // We switch primary to the preferred primary if the preferred primary is
+ // in the list.
+ /* ----------------------------------------------------------------------- */
+ const Uint32 prefPrim = fragPtr.p->preferredPrimary;
+ for (Uint32 i = 1; i < index; i++) {
+ jam();
+ ndbrequire(i < MAX_REPLICAS);
+ if (fragPtr.p->activeNodes[i] == prefPrim){
+ jam();
+ Uint32 switchNode = fragPtr.p->activeNodes[0];
+ fragPtr.p->activeNodes[0] = prefPrim;
+ fragPtr.p->activeNodes[i] = switchNode;
+ break;
+ }//if
+ }//for
+}//Dbdih::updateNodeInfo()
+
+void Dbdih::writeFragment(RWFragment* wf, FragmentstorePtr fragPtr)
+{
+ writePageWord(wf, wf->fragId);
+ writePageWord(wf, fragPtr.p->preferredPrimary);
+ writePageWord(wf, fragPtr.p->noStoredReplicas);
+ writePageWord(wf, fragPtr.p->noOldStoredReplicas);
+ writePageWord(wf, fragPtr.p->distributionKey);
+}//Dbdih::writeFragment()
+
+void Dbdih::writePageWord(RWFragment* wf, Uint32 dataWord)
+{
+ if (wf->wordIndex >= 2048) {
+ jam();
+ ndbrequire(wf->wordIndex == 2048);
+ allocpage(wf->rwfPageptr);
+ wf->wordIndex = 32;
+ wf->pageIndex++;
+ ndbrequire(wf->pageIndex < 8);
+ wf->rwfTabPtr.p->pageRef[wf->pageIndex] = wf->rwfPageptr.i;
+ wf->rwfTabPtr.p->noPages++;
+ }//if
+ wf->rwfPageptr.p->word[wf->wordIndex] = dataWord;
+ wf->wordIndex++;
+}//Dbdih::writePageWord()
+
+void Dbdih::writeReplicas(RWFragment* wf, Uint32 replicaStartIndex)
+{
+ ReplicaRecordPtr wfReplicaPtr;
+ wfReplicaPtr.i = replicaStartIndex;
+ while (wfReplicaPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(wfReplicaPtr, creplicaFileSize, replicaRecord);
+ writePageWord(wf, wfReplicaPtr.p->procNode);
+ writePageWord(wf, wfReplicaPtr.p->initialGci);
+ writePageWord(wf, wfReplicaPtr.p->noCrashedReplicas);
+ writePageWord(wf, wfReplicaPtr.p->nextLcp);
+ for (Uint32 i = 0; i < MAX_LCP_STORED; i++) {
+ writePageWord(wf, wfReplicaPtr.p->maxGciCompleted[i]);
+ writePageWord(wf, wfReplicaPtr.p->maxGciStarted[i]);
+ writePageWord(wf, wfReplicaPtr.p->lcpId[i]);
+ writePageWord(wf, wfReplicaPtr.p->lcpStatus[i]);
+ }//if
+ for (Uint32 i = 0; i < 8; i++) {
+ writePageWord(wf, wfReplicaPtr.p->createGci[i]);
+ writePageWord(wf, wfReplicaPtr.p->replicaLastGci[i]);
+ }//if
+
+ wfReplicaPtr.i = wfReplicaPtr.p->nextReplica;
+ }//while
+}//Dbdih::writeReplicas()
+
+void Dbdih::writeRestorableGci(Signal* signal, FileRecordPtr filePtr)
+{
+ for (Uint32 i = 0; i < Sysfile::SYSFILE_SIZE32; i++) {
+ sysfileDataToFile[i] = sysfileData[i];
+ }//for
+ signal->theData[0] = filePtr.p->fileRef;
+ signal->theData[1] = reference();
+ signal->theData[2] = filePtr.i;
+ signal->theData[3] = ZLIST_OF_PAIRS_SYNCH;
+ signal->theData[4] = ZVAR_NO_CRESTART_INFO_TO_FILE;
+ signal->theData[5] = 1; /* AMOUNT OF PAGES */
+ signal->theData[6] = 0; /* MEMORY PAGE = 0 SINCE COMMON STORED VARIABLE */
+ signal->theData[7] = 0;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+}//Dbdih::writeRestorableGci()
+
+void Dbdih::writeTabfile(Signal* signal, TabRecord* tab, FileRecordPtr filePtr)
+{
+ signal->theData[0] = filePtr.p->fileRef;
+ signal->theData[1] = reference();
+ signal->theData[2] = filePtr.i;
+ signal->theData[3] = ZLIST_OF_PAIRS;
+ signal->theData[4] = ZVAR_NO_WORD;
+ signal->theData[5] = tab->noPages;
+ for (Uint32 i = 0; i < tab->noPages; i++) {
+ jam();
+ signal->theData[6 + (2 * i)] = tab->pageRef[i];
+ signal->theData[7 + (2 * i)] = i;
+ }//for
+ Uint32 length = 6 + (2 * tab->noPages);
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, length, JBA);
+}//Dbdih::writeTabfile()
+
+void Dbdih::execDEBUG_SIG(Signal* signal)
+{
+ signal = signal; //Avoid compiler warnings
+}//Dbdih::execDEBUG_SIG()
+
+void
+Dbdih::execDUMP_STATE_ORD(Signal* signal)
+{
+ DumpStateOrd * const & dumpState = (DumpStateOrd *)&signal->theData[0];
+ if (dumpState->args[0] == DumpStateOrd::DihDumpNodeRestartInfo) {
+ infoEvent("c_nodeStartMaster.blockLcp = %d, c_nodeStartMaster.blockGcp = %d, c_nodeStartMaster.wait = %d",
+ c_nodeStartMaster.blockLcp, c_nodeStartMaster.blockGcp, c_nodeStartMaster.wait);
+ infoEvent("cstartGcpNow = %d, cgcpStatus = %d",
+ cstartGcpNow, cgcpStatus);
+ infoEvent("cfirstVerifyQueue = %d, cverifyQueueCounter = %d",
+ cfirstVerifyQueue, cverifyQueueCounter);
+ infoEvent("cgcpOrderBlocked = %d, cgcpStartCounter = %d",
+ cgcpOrderBlocked, cgcpStartCounter);
+ }//if
+ if (dumpState->args[0] == DumpStateOrd::DihDumpNodeStatusInfo) {
+ NodeRecordPtr localNodePtr;
+ infoEvent("Printing nodeStatus of all nodes");
+ for (localNodePtr.i = 1; localNodePtr.i < MAX_NDB_NODES; localNodePtr.i++) {
+ ptrAss(localNodePtr, nodeRecord);
+ if (localNodePtr.p->nodeStatus != NodeRecord::NOT_IN_CLUSTER) {
+ infoEvent("Node = %d has status = %d",
+ localNodePtr.i, localNodePtr.p->nodeStatus);
+ }//if
+ }//for
+ }//if
+
+ if (dumpState->args[0] == DumpStateOrd::DihPrintFragmentation){
+ infoEvent("Printing fragmentation of all tables --");
+ for(Uint32 i = 0; i<ctabFileSize; i++){
+ TabRecordPtr tabPtr;
+ tabPtr.i = i;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ if(tabPtr.p->tabStatus != TabRecord::TS_ACTIVE)
+ continue;
+
+ for(Uint32 j = 0; j < tabPtr.p->totalfragments; j++){
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, j, fragPtr);
+
+ Uint32 nodeOrder[MAX_REPLICAS];
+ const Uint32 noOfReplicas = extractNodeInfo(fragPtr.p, nodeOrder);
+ char buf[100];
+ snprintf(buf, sizeof(buf), " Table %d Fragment %d - ", tabPtr.i, j);
+ for(Uint32 k = 0; k < noOfReplicas; k++){
+ char tmp[100];
+ snprintf(tmp, sizeof(tmp), "%d ", nodeOrder[k]);
+ strcat(buf, tmp);
+ }
+ infoEvent(buf);
+ }
+ }
+ }
+
+ if (signal->theData[0] == 7000) {
+ infoEvent("ctimer = %d, cgcpParticipantState = %d, cgcpStatus = %d",
+ c_lcpState.ctimer, cgcpParticipantState, cgcpStatus);
+ infoEvent("coldGcpStatus = %d, coldGcpId = %d, cmasterState = %d",
+ coldGcpStatus, coldGcpId, cmasterState);
+ infoEvent("cmasterTakeOverNode = %d, ctcCounter = %d",
+ cmasterTakeOverNode, c_lcpState.ctcCounter);
+ }//if
+ if (signal->theData[0] == 7001) {
+ infoEvent("c_lcpState.keepGci = %d",
+ c_lcpState.keepGci);
+ infoEvent("c_lcpState.lcpStatus = %d, clcpStartGcp = %d",
+ c_lcpState.lcpStatus,
+ c_lcpState.lcpStartGcp);
+ infoEvent("cgcpStartCounter = %d, cimmediateLcpStart = %d",
+ cgcpStartCounter, c_lcpState.immediateLcpStart);
+ }//if
+ if (signal->theData[0] == 7002) {
+ infoEvent("cnoOfActiveTables = %d, cgcpDelay = %d",
+ cnoOfActiveTables, cgcpDelay);
+ infoEvent("cdictblockref = %d, cfailurenr = %d",
+ cdictblockref, cfailurenr);
+ infoEvent("con_lineNodes = %d, reference() = %d, creceivedfrag = %d",
+ con_lineNodes, reference(), creceivedfrag);
+ }//if
+ if (signal->theData[0] == 7003) {
+ infoEvent("cfirstAliveNode = %d, cgckptflag = %d",
+ cfirstAliveNode, cgckptflag);
+ infoEvent("clocallqhblockref = %d, clocaltcblockref = %d, cgcpOrderBlocked = %d",
+ clocallqhblockref, clocaltcblockref, cgcpOrderBlocked);
+ infoEvent("cstarttype = %d, csystemnodes = %d, currentgcp = %d",
+ cstarttype, csystemnodes, currentgcp);
+ }//if
+ if (signal->theData[0] == 7004) {
+ infoEvent("cmasterdihref = %d, cownNodeId = %d, cnewgcp = %d",
+ cmasterdihref, cownNodeId, cnewgcp);
+ infoEvent("cndbStartReqBlockref = %d, cremainingfrags = %d",
+ cndbStartReqBlockref, cremainingfrags);
+ infoEvent("cntrlblockref = %d, cgcpSameCounter = %d, coldgcp = %d",
+ cntrlblockref, cgcpSameCounter, coldgcp);
+ }//if
+ if (signal->theData[0] == 7005) {
+ infoEvent("crestartGci = %d",
+ crestartGci);
+ }//if
+ if (signal->theData[0] == 7006) {
+ infoEvent("clcpDelay = %d, cgcpMasterTakeOverState = %d",
+ c_lcpState.clcpDelay, cgcpMasterTakeOverState);
+ infoEvent("cmasterNodeId = %d", cmasterNodeId);
+ infoEvent("cnoHotSpare = %d, c_nodeStartMaster.startNode = %d, c_nodeStartMaster.wait = %d",
+ cnoHotSpare, c_nodeStartMaster.startNode, c_nodeStartMaster.wait);
+ }//if
+ if (signal->theData[0] == 7007) {
+ infoEvent("c_nodeStartMaster.failNr = %d, c_nodeStartMaster.ndbVersion = %d",
+ c_nodeStartMaster.failNr, c_nodeStartMaster.ndbVersion);
+ infoEvent("c_nodeStartMaster.startInfoErrorCode = %d",
+ c_nodeStartMaster.startInfoErrorCode);
+ infoEvent("c_nodeStartMaster.blockLcp = %d, c_nodeStartMaster.blockGcp = %d",
+ c_nodeStartMaster.blockLcp, c_nodeStartMaster.blockGcp);
+ }//if
+ if (signal->theData[0] == 7008) {
+ infoEvent("cfirstDeadNode = %d, cstartPhase = %d, cnoReplicas = %d",
+ cfirstDeadNode, cstartPhase, cnoReplicas);
+ infoEvent("cwaitLcpSr = %d",cwaitLcpSr);
+ }//if
+ if (signal->theData[0] == 7009) {
+ infoEvent("ccalcOldestRestorableGci = %d, cnoOfNodeGroups = %d",
+ c_lcpState.oldestRestorableGci, cnoOfNodeGroups);
+ infoEvent("cstartGcpNow = %d",
+ cstartGcpNow);
+ infoEvent("crestartGci = %d",
+ crestartGci);
+ }//if
+ if (signal->theData[0] == 7010) {
+ infoEvent("cminHotSpareNodes = %d, c_lcpState.lcpStatusUpdatedPlace = %d, cLcpStart = %d",
+ cminHotSpareNodes, c_lcpState.lcpStatusUpdatedPlace, c_lcpState.lcpStart);
+ infoEvent("c_blockCommit = %d, c_blockCommitNo = %d",
+ c_blockCommit, c_blockCommitNo);
+ }//if
+ if (signal->theData[0] == 7011){
+ infoEvent("c_COPY_GCIREQ_Counter = %s",
+ c_COPY_GCIREQ_Counter.getText());
+ infoEvent("c_COPY_TABREQ_Counter = %s",
+ c_COPY_TABREQ_Counter.getText());
+ infoEvent("c_CREATE_FRAGREQ_Counter = %s",
+ c_CREATE_FRAGREQ_Counter.getText());
+ infoEvent("c_DIH_SWITCH_REPLICA_REQ_Counter = %s",
+ c_DIH_SWITCH_REPLICA_REQ_Counter.getText());
+ infoEvent("c_EMPTY_LCP_REQ_Counter = %s",c_EMPTY_LCP_REQ_Counter.getText());
+ infoEvent("c_END_TOREQ_Counter = %s", c_END_TOREQ_Counter.getText());
+ infoEvent("c_GCP_COMMIT_Counter = %s", c_GCP_COMMIT_Counter.getText());
+ infoEvent("c_GCP_PREPARE_Counter = %s", c_GCP_PREPARE_Counter.getText());
+ infoEvent("c_GCP_SAVEREQ_Counter = %s", c_GCP_SAVEREQ_Counter.getText());
+ infoEvent("c_INCL_NODEREQ_Counter = %s", c_INCL_NODEREQ_Counter.getText());
+ infoEvent("c_MASTER_GCPREQ_Counter = %s",
+ c_MASTER_GCPREQ_Counter.getText());
+ infoEvent("c_MASTER_LCPREQ_Counter = %s",
+ c_MASTER_LCPREQ_Counter.getText());
+ infoEvent("c_START_INFOREQ_Counter = %s",
+ c_START_INFOREQ_Counter.getText());
+ infoEvent("c_START_RECREQ_Counter = %s", c_START_RECREQ_Counter.getText());
+ infoEvent("c_START_TOREQ_Counter = %s", c_START_TOREQ_Counter.getText());
+ infoEvent("c_STOP_ME_REQ_Counter = %s", c_STOP_ME_REQ_Counter.getText());
+ infoEvent("c_TC_CLOPSIZEREQ_Counter = %s",
+ c_TC_CLOPSIZEREQ_Counter.getText());
+ infoEvent("c_TCGETOPSIZEREQ_Counter = %s",
+ c_TCGETOPSIZEREQ_Counter.getText());
+ infoEvent("c_UPDATE_TOREQ_Counter = %s", c_UPDATE_TOREQ_Counter.getText());
+ }
+
+ if(signal->theData[0] == 7012){
+ char buf[c_lcpState.m_participatingDIH.TextLength+1];
+ infoEvent("ParticipatingDIH = %s", c_lcpState.m_participatingDIH.getText(buf));
+ infoEvent("ParticipatingLQH = %s", c_lcpState.m_participatingLQH.getText(buf));
+ infoEvent("m_LCP_COMPLETE_REP_Counter_DIH = %s",
+ c_lcpState.m_LCP_COMPLETE_REP_Counter_DIH.getText());
+ infoEvent("m_LCP_COMPLETE_REP_Counter_LQH = %s",
+ c_lcpState.m_LCP_COMPLETE_REP_Counter_LQH.getText());
+ infoEvent("m_LAST_LCP_FRAG_ORD = %s",
+ c_lcpState.m_LAST_LCP_FRAG_ORD.getText());
+ infoEvent("m_LCP_COMPLETE_REP_From_Master_Received = %d",
+ c_lcpState.m_LCP_COMPLETE_REP_From_Master_Received);
+
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if(nodePtr.p->nodeStatus == NodeRecord::ALIVE){
+
+ for(Uint32 i = 0; i<nodePtr.p->noOfStartedChkpt; i++){
+ infoEvent("Node %d: started: table=%d fragment=%d replica=%d",
+ nodePtr.i,
+ nodePtr.p->startedChkpt[i].tableId,
+ nodePtr.p->startedChkpt[i].fragId,
+ nodePtr.p->startedChkpt[i].replicaPtr);
+ }
+
+ for(Uint32 i = 0; i<nodePtr.p->noOfQueuedChkpt; i++){
+ infoEvent("Node %d: queued: table=%d fragment=%d replica=%d",
+ nodePtr.i,
+ nodePtr.p->queuedChkpt[i].tableId,
+ nodePtr.p->queuedChkpt[i].fragId,
+ nodePtr.p->queuedChkpt[i].replicaPtr);
+ }
+ }
+ }
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::DihDumpLCPState){
+ infoEvent("-- Node %d LCP STATE --", getOwnNodeId());
+ infoEvent("lcpStatus = %d (update place = %d) ",
+ c_lcpState.lcpStatus, c_lcpState.lcpStatusUpdatedPlace);
+ infoEvent
+ ("lcpStart = %d lcpStartGcp = %d keepGci = %d oldestRestorable = %d",
+ c_lcpState.lcpStart, c_lcpState.lcpStartGcp,
+ c_lcpState.keepGci, c_lcpState.oldestRestorableGci);
+
+ infoEvent
+ ("immediateLcpStart = %d masterLcpNodeId = %d",
+ c_lcpState.immediateLcpStart,
+ refToNode(c_lcpState.m_masterLcpDihRef));
+ infoEvent("-- Node %d LCP STATE --", getOwnNodeId());
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::DihDumpLCPMasterTakeOver){
+ infoEvent("-- Node %d LCP MASTER TAKE OVER STATE --", getOwnNodeId());
+ infoEvent
+ ("c_lcpMasterTakeOverState.state = %d updatePlace = %d failedNodeId = %d",
+ c_lcpMasterTakeOverState.state,
+ c_lcpMasterTakeOverState.updatePlace,
+ c_lcpMasterTakeOverState.failedNodeId);
+
+ infoEvent("c_lcpMasterTakeOverState.minTableId = %u minFragId = %u",
+ c_lcpMasterTakeOverState.minTableId,
+ c_lcpMasterTakeOverState.minFragId);
+
+ infoEvent("-- Node %d LCP MASTER TAKE OVER STATE --", getOwnNodeId());
+ }
+
+ if (signal->theData[0] == 7015){
+ for(Uint32 i = 0; i<ctabFileSize; i++){
+ TabRecordPtr tabPtr;
+ tabPtr.i = i;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ if(tabPtr.p->tabStatus != TabRecord::TS_ACTIVE)
+ continue;
+
+ infoEvent
+ ("Table %d: TabCopyStatus: %d TabUpdateStatus: %d TabLcpStatus: %d",
+ tabPtr.i,
+ tabPtr.p->tabCopyStatus,
+ tabPtr.p->tabUpdateState,
+ tabPtr.p->tabLcpStatus);
+
+ FragmentstorePtr fragPtr;
+ for (Uint32 fid = 0; fid < tabPtr.p->totalfragments; fid++) {
+ jam();
+ getFragstore(tabPtr.p, fid, fragPtr);
+
+ char buf[100], buf2[100];
+ snprintf(buf, sizeof(buf), " Fragment %d: noLcpReplicas==%d ",
+ fid, fragPtr.p->noLcpReplicas);
+
+ Uint32 num=0;
+ ReplicaRecordPtr replicaPtr;
+ replicaPtr.i = fragPtr.p->storedReplicas;
+ do {
+ ptrCheckGuard(replicaPtr, creplicaFileSize, replicaRecord);
+ snprintf(buf2, sizeof(buf2), "%s %d(on %d)=%d(%s)",
+ buf, num,
+ replicaPtr.p->procNode,
+ replicaPtr.p->lcpIdStarted,
+ replicaPtr.p->lcpOngoingFlag ? "Ongoing" : "Idle");
+ snprintf(buf, sizeof(buf), "%s", buf2);
+
+ num++;
+ replicaPtr.i = replicaPtr.p->nextReplica;
+ } while (replicaPtr.i != RNIL);
+ infoEvent(buf);
+ }
+ }
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::EnableUndoDelayDataWrite){
+ ndbout << "Dbdih:: delay write of datapages for table = "
+ << dumpState->args[1]<< endl;
+ // Send this dump to ACC and TUP
+ EXECUTE_DIRECT(DBACC, GSN_DUMP_STATE_ORD, signal, 2);
+ EXECUTE_DIRECT(DBTUP, GSN_DUMP_STATE_ORD, signal, 2);
+
+ // Start immediate LCP
+ c_lcpState.ctimer += (1 << c_lcpState.clcpDelay);
+ return;
+ }
+
+ if (signal->theData[0] == DumpStateOrd::DihAllAllowNodeStart) {
+ for (Uint32 i = 1; i < MAX_NDB_NODES; i++)
+ setAllowNodeStart(i, true);
+ return;
+ }//if
+ if (signal->theData[0] == DumpStateOrd::DihMinTimeBetweenLCP) {
+ // Set time between LCP to min value
+ ndbout << "Set time between LCP to min value" << endl;
+ c_lcpState.clcpDelay = 0; // TimeBetweenLocalCheckpoints.min
+ return;
+ }
+ if (signal->theData[0] == DumpStateOrd::DihMaxTimeBetweenLCP) {
+ // Set time between LCP to max value
+ ndbout << "Set time between LCP to max value" << endl;
+ c_lcpState.clcpDelay = 31; // TimeBetweenLocalCheckpoints.max
+ return;
+ }
+
+ if(dumpState->args[0] == 7098){
+ if(signal->length() == 3){
+ jam();
+ infoEvent("startLcpRoundLoopLab(tabel=%d, fragment=%d)",
+ signal->theData[1], signal->theData[2]);
+ startLcpRoundLoopLab(signal, signal->theData[1], signal->theData[2]);
+ return;
+ } else {
+ infoEvent("Invalid no of arguments to 7098 - startLcpRoundLoopLab -"
+ " expected 2 (tableId, fragmentId)");
+ }
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::DihStartLcpImmediately){
+ c_lcpState.ctimer += (1 << c_lcpState.clcpDelay);
+ return;
+ }
+}//Dbdih::execDUMP_STATE_ORD()
+
+void
+Dbdih::execPREP_DROP_TAB_REQ(Signal* signal){
+ jamEntry();
+
+ PrepDropTabReq* req = (PrepDropTabReq*)signal->getDataPtr();
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ Uint32 senderRef = req->senderRef;
+ Uint32 senderData = req->senderData;
+
+ PrepDropTabRef::ErrorCode err = PrepDropTabRef::OK;
+ { /**
+ * Check table state
+ */
+ bool ok = false;
+ switch(tabPtr.p->tabStatus){
+ case TabRecord::TS_IDLE:
+ ok = true;
+ jam();
+ err = PrepDropTabRef::NoSuchTable;
+ break;
+ case TabRecord::TS_DROPPING:
+ ok = true;
+ jam();
+ err = PrepDropTabRef::PrepDropInProgress;
+ break;
+ case TabRecord::TS_CREATING:
+ jam();
+ ok = true;
+ break;
+ case TabRecord::TS_ACTIVE:
+ ok = true;
+ jam();
+ break;
+ }
+ ndbrequire(ok);
+ }
+
+ if(err != PrepDropTabRef::OK){
+ jam();
+ PrepDropTabRef* ref = (PrepDropTabRef*)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = senderData;
+ ref->tableId = tabPtr.i;
+ ref->errorCode = err;
+ sendSignal(senderRef, GSN_PREP_DROP_TAB_REF, signal,
+ PrepDropTabRef::SignalLength, JBB);
+ return;
+ }
+
+ tabPtr.p->tabStatus = TabRecord::TS_DROPPING;
+ tabPtr.p->m_prepDropTab.senderRef = senderRef;
+ tabPtr.p->m_prepDropTab.senderData = senderData;
+
+ if(isMaster()){
+ /**
+ * Remove from queue
+ */
+ NodeRecordPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRecord);
+ if (c_lcpState.m_participatingLQH.get(nodePtr.i)){
+
+ Uint32 index = 0;
+ Uint32 count = nodePtr.p->noOfQueuedChkpt;
+ while(index < count){
+ if(nodePtr.p->queuedChkpt[index].tableId == tabPtr.i){
+ jam();
+ // ndbout_c("Unqueuing %d", index);
+
+ count--;
+ for(Uint32 i = index; i<count; i++){
+ jam();
+ nodePtr.p->queuedChkpt[i] = nodePtr.p->queuedChkpt[i + 1];
+ }
+ } else {
+ index++;
+ }
+ }
+ nodePtr.p->noOfQueuedChkpt = count;
+ }
+ }
+ }
+
+ { /**
+ * Check table lcp state
+ */
+
+ bool ok = false;
+ switch(tabPtr.p->tabLcpStatus){
+ case TabRecord::TLS_COMPLETED:
+ case TabRecord::TLS_WRITING_TO_FILE:
+ ok = true;
+ jam();
+ break;
+ return;
+ case TabRecord::TLS_ACTIVE:
+ ok = true;
+ jam();
+
+ tabPtr.p->tabLcpStatus = TabRecord::TLS_COMPLETED;
+
+ /**
+ * First check if all fragments are done
+ */
+ if(checkLcpAllTablesDoneInLqh()){
+ jam();
+
+ ndbout_c("This is the last table");
+
+ /**
+ * Then check if saving of tab info is done for all tables
+ */
+ LcpStatus a = c_lcpState.lcpStatus;
+ checkLcpCompletedLab(signal);
+
+ if(a != c_lcpState.lcpStatus){
+ ndbout_c("And all tables are written to already written disk");
+ }
+ }
+ break;
+ }
+ ndbrequire(ok);
+ }
+
+ { /**
+ * Send WaitDropTabReq to all LQH
+ */
+ WaitDropTabReq * req = (WaitDropTabReq*)signal->getDataPtrSend();
+ req->tableId = tabPtr.i;
+ req->senderRef = reference();
+
+ NodeRecordPtr nodePtr;
+ nodePtr.i = cfirstAliveNode;
+ tabPtr.p->m_prepDropTab.waitDropTabCount.clearWaitingFor();
+ while(nodePtr.i != RNIL){
+ jam();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+
+ tabPtr.p->m_prepDropTab.waitDropTabCount.setWaitingFor(nodePtr.i);
+ sendSignal(calcLqhBlockRef(nodePtr.i), GSN_WAIT_DROP_TAB_REQ,
+ signal, WaitDropTabReq::SignalLength, JBB);
+
+ nodePtr.i = nodePtr.p->nextNode;
+ }
+ }
+
+ waitDropTabWritingToFile(signal, tabPtr);
+}
+
+void
+Dbdih::waitDropTabWritingToFile(Signal* signal, TabRecordPtr tabPtr){
+
+ if(tabPtr.p->tabLcpStatus == TabRecord::TLS_WRITING_TO_FILE){
+ jam();
+ signal->theData[0] = DihContinueB::WAIT_DROP_TAB_WRITING_TO_FILE;
+ signal->theData[1] = tabPtr.i;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 2);
+ return;
+ }
+
+ ndbrequire(tabPtr.p->tabLcpStatus == TabRecord::TLS_COMPLETED);
+ checkPrepDropTabComplete(signal, tabPtr);
+}
+
+void
+Dbdih::checkPrepDropTabComplete(Signal* signal, TabRecordPtr tabPtr){
+
+ if(tabPtr.p->tabLcpStatus != TabRecord::TLS_COMPLETED){
+ jam();
+ return;
+ }
+
+ if(!tabPtr.p->m_prepDropTab.waitDropTabCount.done()){
+ jam();
+ return;
+ }
+
+ const Uint32 ref = tabPtr.p->m_prepDropTab.senderRef;
+ if(ref != 0){
+ PrepDropTabConf* conf = (PrepDropTabConf*)signal->getDataPtrSend();
+ conf->tableId = tabPtr.i;
+ conf->senderRef = reference();
+ conf->senderData = tabPtr.p->m_prepDropTab.senderData;
+ sendSignal(tabPtr.p->m_prepDropTab.senderRef, GSN_PREP_DROP_TAB_CONF,
+ signal, PrepDropTabConf::SignalLength, JBB);
+ tabPtr.p->m_prepDropTab.senderRef = 0;
+ }
+}
+
+void
+Dbdih::execWAIT_DROP_TAB_CONF(Signal* signal){
+ jamEntry();
+ WaitDropTabConf * conf = (WaitDropTabConf*)signal->getDataPtr();
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = conf->tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ ndbrequire(tabPtr.p->tabStatus == TabRecord::TS_DROPPING);
+ Uint32 nodeId = refToNode(conf->senderRef);
+ tabPtr.p->m_prepDropTab.waitDropTabCount.clearWaitingFor(nodeId);
+ checkPrepDropTabComplete(signal, tabPtr);
+}
+
+void
+Dbdih::checkWaitDropTabFailedLqh(Signal* signal, Uint32 nodeId, Uint32 tableId){
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+
+ WaitDropTabConf * conf = (WaitDropTabConf*)signal->getDataPtr();
+ conf->tableId = tableId;
+
+ const Uint32 RT_BREAK = 16;
+ for(Uint32 i = 0; i<RT_BREAK && tabPtr.i < ctabFileSize; i++, tabPtr.i++){
+ ptrAss(tabPtr, tabRecord);
+ if(tabPtr.p->tabStatus == TabRecord::TS_DROPPING){
+ if(tabPtr.p->m_prepDropTab.waitDropTabCount.isWaitingFor(nodeId)){
+ conf->senderRef = calcLqhBlockRef(nodeId);
+ execWAIT_DROP_TAB_CONF(signal);
+ tabPtr.i++;
+ break;
+ }
+ }
+ }
+
+ if(tabPtr.i == ctabFileSize){
+ /**
+ * Finished
+ */
+ jam();
+ return;
+ }
+
+ signal->theData[0] = DihContinueB::CHECK_WAIT_DROP_TAB_FAILED_LQH;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+}
+
+
+void
+Dbdih::execNDB_TAMPER(Signal* signal)
+{
+ if ((ERROR_INSERTED(7011)) &&
+ (signal->theData[0] == 7012)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ calculateKeepGciLab(signal, 0, 0);
+ return;
+ }//if
+ SET_ERROR_INSERT_VALUE(signal->theData[0]);
+ return;
+}//Dbdih::execNDB_TAMPER()
+
+void Dbdih::execSET_VAR_REQ(Signal* signal) {
+
+ SetVarReq* const setVarReq = (SetVarReq*)&signal->theData[0];
+ ConfigParamId var = setVarReq->variable();
+ int val = setVarReq->value();
+
+
+ switch (var) {
+ case TimeBetweenLocalCheckpoints:
+ c_lcpState.clcpDelay = val;
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ case TimeBetweenGlobalCheckpoints:
+ cgcpDelay = val;
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ default:
+ sendSignal(CMVMI_REF, GSN_SET_VAR_REF, signal, 1, JBB);
+ } // switch
+}
+
+void Dbdih::execBLOCK_COMMIT_ORD(Signal* signal){
+ BlockCommitOrd* const block = (BlockCommitOrd *)&signal->theData[0];
+
+ jamEntry();
+#if 0
+ ndbrequire(c_blockCommit == false ||
+ c_blockCommitNo == block->failNo);
+#else
+ if(!(c_blockCommit == false || c_blockCommitNo == block->failNo)){
+ infoEvent("Possible bug in Dbdih::execBLOCK_COMMIT_ORD c_blockCommit = %d c_blockCommitNo = %d"
+ " sig->failNo = %d", c_blockCommit, c_blockCommitNo, block->failNo);
+ }
+#endif
+ c_blockCommit = true;
+ c_blockCommitNo = block->failNo;
+}
+
+void Dbdih::execUNBLOCK_COMMIT_ORD(Signal* signal){
+ UnblockCommitOrd* const unblock = (UnblockCommitOrd *)&signal->theData[0];
+ (void)unblock;
+
+ jamEntry();
+
+ if(c_blockCommit == true){
+ jam();
+ // ndbrequire(c_blockCommitNo == unblock->failNo);
+
+ c_blockCommit = false;
+ emptyverificbuffer(signal, true);
+ }
+}
+
+void Dbdih::execSTOP_PERM_REQ(Signal* signal){
+
+ jamEntry();
+
+ StopPermReq* const req = (StopPermReq*)&signal->theData[0];
+ StopPermRef* const ref = (StopPermRef*)&signal->theData[0];
+
+ const Uint32 senderData = req->senderData;
+ const BlockReference senderRef = req->senderRef;
+ const NodeId nodeId = refToNode(senderRef);
+
+ if (isMaster()) {
+ /**
+ * Master
+ */
+ jam();
+ CRASH_INSERTION(7065);
+ if (c_stopPermMaster.clientRef != 0) {
+ jam();
+
+ ref->senderData = senderData;
+ ref->errorCode = StopPermRef::NodeShutdownInProgress;
+ sendSignal(senderRef, GSN_STOP_PERM_REF, signal,
+ StopPermRef::SignalLength, JBB);
+ return;
+ }//if
+
+ if (c_nodeStartMaster.activeState) {
+ jam();
+ ref->senderData = senderData;
+ ref->errorCode = StopPermRef::NodeStartInProgress;
+ sendSignal(senderRef, GSN_STOP_PERM_REF, signal,
+ StopPermRef::SignalLength, JBB);
+ return;
+ }//if
+
+ /**
+ * Lock
+ */
+ c_nodeStartMaster.activeState = true;
+ c_stopPermMaster.clientRef = senderRef;
+
+ c_stopPermMaster.clientData = senderData;
+ c_stopPermMaster.returnValue = 0;
+ c_switchReplicas.clear();
+
+ Mutex mutex(signal, c_mutexMgr, c_switchPrimaryMutexHandle);
+ Callback c = { safe_cast(&Dbdih::switch_primary_stop_node), nodeId };
+ ndbrequire(mutex.lock(c));
+ } else {
+ /**
+ * Proxy part
+ */
+ jam();
+ CRASH_INSERTION(7066);
+ if(c_stopPermProxy.clientRef != 0){
+ jam();
+ ref->senderData = senderData;
+ ref->errorCode = StopPermRef::NodeShutdownInProgress;
+ sendSignal(senderRef, GSN_STOP_PERM_REF, signal, 2, JBB);
+ return;
+ }//if
+
+ c_stopPermProxy.clientRef = senderRef;
+ c_stopPermProxy.masterRef = cmasterdihref;
+ c_stopPermProxy.clientData = senderData;
+
+ req->senderRef = reference();
+ req->senderData = senderData;
+ sendSignal(cmasterdihref, GSN_STOP_PERM_REQ, signal,
+ StopPermReq::SignalLength, JBB);
+ }//if
+}//Dbdih::execSTOP_PERM_REQ()
+
+void
+Dbdih::switch_primary_stop_node(Signal* signal, Uint32 node_id, Uint32 ret_val)
+{
+ ndbrequire(ret_val == 0);
+ signal->theData[0] = DihContinueB::SwitchReplica;
+ signal->theData[1] = node_id;
+ signal->theData[2] = 0; // table id
+ signal->theData[3] = 0; // fragment id
+ sendSignal(reference(), GSN_CONTINUEB, signal, 4, JBB);
+}
+
+void Dbdih::execSTOP_PERM_REF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(c_stopPermProxy.clientRef != 0);
+ ndbrequire(c_stopPermProxy.masterRef == signal->senderBlockRef());
+ sendSignal(c_stopPermProxy.clientRef, GSN_STOP_PERM_REF, signal, 2, JBB);
+ c_stopPermProxy.clientRef = 0;
+}//Dbdih::execSTOP_PERM_REF()
+
+void Dbdih::execSTOP_PERM_CONF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(c_stopPermProxy.clientRef != 0);
+ ndbrequire(c_stopPermProxy.masterRef == signal->senderBlockRef());
+ sendSignal(c_stopPermProxy.clientRef, GSN_STOP_PERM_CONF, signal, 1, JBB);
+ c_stopPermProxy.clientRef = 0;
+}//Dbdih::execSTOP_PERM_CONF()
+
+void Dbdih::execDIH_SWITCH_REPLICA_REQ(Signal* signal)
+{
+ jamEntry();
+ DihSwitchReplicaReq* const req = (DihSwitchReplicaReq*)&signal->theData[0];
+ const Uint32 tableId = req->tableId;
+ const Uint32 fragNo = req->fragNo;
+ const BlockReference senderRef = req->senderRef;
+
+ CRASH_INSERTION(7067);
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ ndbrequire(tabPtr.p->tabStatus == TabRecord::TS_ACTIVE);
+ if (tabPtr.p->tabCopyStatus != TabRecord::CS_IDLE) {
+ jam();
+ sendSignal(reference(), GSN_DIH_SWITCH_REPLICA_REQ, signal,
+ DihSwitchReplicaReq::SignalLength, JBB);
+ return;
+ }//if
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragNo, fragPtr);
+
+ /**
+ * Do funky stuff
+ */
+ Uint32 oldOrder[MAX_REPLICAS];
+ const Uint32 noOfReplicas = extractNodeInfo(fragPtr.p, oldOrder);
+
+ if (noOfReplicas < req->noOfReplicas) {
+ jam();
+ //---------------------------------------------------------------------
+ // A crash occurred in the middle of our switch handling.
+ //---------------------------------------------------------------------
+ DihSwitchReplicaRef* const ref = (DihSwitchReplicaRef*)&signal->theData[0];
+ ref->senderNode = cownNodeId;
+ ref->errorCode = StopPermRef::NF_CausedAbortOfStopProcedure;
+ sendSignal(senderRef, GSN_DIH_SWITCH_REPLICA_REF, signal,
+ DihSwitchReplicaRef::SignalLength, JBB);
+ }//if
+ for (Uint32 i = 0; i < noOfReplicas; i++) {
+ jam();
+ ndbrequire(i < MAX_REPLICAS);
+ fragPtr.p->activeNodes[i] = req->newNodeOrder[i];
+ }//for
+ /**
+ * Reply
+ */
+ DihSwitchReplicaConf* const conf = (DihSwitchReplicaConf*)&signal->theData[0];
+ conf->senderNode = cownNodeId;
+ sendSignal(senderRef, GSN_DIH_SWITCH_REPLICA_CONF, signal,
+ DihSwitchReplicaConf::SignalLength, JBB);
+}//Dbdih::execDIH_SWITCH_REPLICA_REQ()
+
+void Dbdih::execDIH_SWITCH_REPLICA_CONF(Signal* signal)
+{
+ jamEntry();
+ /**
+ * Response to master
+ */
+ CRASH_INSERTION(7068);
+ DihSwitchReplicaConf* const conf = (DihSwitchReplicaConf*)&signal->theData[0];
+ switchReplicaReply(signal, conf->senderNode);
+}//Dbdih::execDIH_SWITCH_REPLICA_CONF()
+
+void Dbdih::execDIH_SWITCH_REPLICA_REF(Signal* signal)
+{
+ jamEntry();
+ DihSwitchReplicaRef* const ref = (DihSwitchReplicaRef*)&signal->theData[0];
+ if(c_stopPermMaster.returnValue == 0){
+ jam();
+ c_stopPermMaster.returnValue = ref->errorCode;
+ }//if
+ switchReplicaReply(signal, ref->senderNode);
+}//Dbdih::execDIH_SWITCH_REPLICA_REF()
+
+void Dbdih::switchReplicaReply(Signal* signal,
+ NodeId nodeId){
+ jam();
+ receiveLoopMacro(DIH_SWITCH_REPLICA_REQ, nodeId);
+ //------------------------------------------------------
+ // We have received all responses from the nodes. Thus
+ // we have completed switching replica roles. Continue
+ // with the next fragment.
+ //------------------------------------------------------
+ if(c_stopPermMaster.returnValue != 0){
+ jam();
+ c_switchReplicas.tableId = ctabFileSize + 1;
+ }//if
+ c_switchReplicas.fragNo++;
+
+ signal->theData[0] = DihContinueB::SwitchReplica;
+ signal->theData[1] = c_switchReplicas.nodeId;
+ signal->theData[2] = c_switchReplicas.tableId;
+ signal->theData[3] = c_switchReplicas.fragNo;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 4, JBB);
+}//Dbdih::switchReplicaReply()
+
+void
+Dbdih::switchReplica(Signal* signal,
+ Uint32 nodeId,
+ Uint32 tableId,
+ Uint32 fragNo){
+ jam();
+ DihSwitchReplicaReq* const req = (DihSwitchReplicaReq*)&signal->theData[0];
+
+ const Uint32 RT_BREAK = 64;
+
+ for (Uint32 i = 0; i < RT_BREAK; i++) {
+ jam();
+ if (tableId >= ctabFileSize) {
+ jam();
+ StopPermConf* const conf = (StopPermConf*)&signal->theData[0];
+ StopPermRef* const ref = (StopPermRef*)&signal->theData[0];
+ /**
+ * Finished with all tables
+ */
+ if(c_stopPermMaster.returnValue == 0) {
+ jam();
+ conf->senderData = c_stopPermMaster.clientData;
+ sendSignal(c_stopPermMaster.clientRef, GSN_STOP_PERM_CONF,
+ signal, 1, JBB);
+ } else {
+ jam();
+ ref->senderData = c_stopPermMaster.clientData;
+ ref->errorCode = c_stopPermMaster.returnValue;
+ sendSignal(c_stopPermMaster.clientRef, GSN_STOP_PERM_REF, signal, 2,JBB);
+ }//if
+
+ /**
+ * UnLock
+ */
+ c_nodeStartMaster.activeState = false;
+ c_stopPermMaster.clientRef = 0;
+ c_stopPermMaster.clientData = 0;
+ c_stopPermMaster.returnValue = 0;
+ Mutex mutex(signal, c_mutexMgr, c_switchPrimaryMutexHandle);
+ mutex.unlock(); // ignore result
+ return;
+ }//if
+
+ TabRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabFileSize, tabRecord);
+
+ if (tabPtr.p->tabStatus != TabRecord::TS_ACTIVE) {
+ jam();
+ tableId++;
+ fragNo = 0;
+ continue;
+ }//if
+ if (fragNo >= tabPtr.p->totalfragments) {
+ jam();
+ tableId++;
+ fragNo = 0;
+ continue;
+ }//if
+ FragmentstorePtr fragPtr;
+ getFragstore(tabPtr.p, fragNo, fragPtr);
+
+ Uint32 oldOrder[MAX_REPLICAS];
+ const Uint32 noOfReplicas = extractNodeInfo(fragPtr.p, oldOrder);
+
+ if(oldOrder[0] != nodeId) {
+ jam();
+ fragNo++;
+ continue;
+ }//if
+ req->tableId = tableId;
+ req->fragNo = fragNo;
+ req->noOfReplicas = noOfReplicas;
+ for (Uint32 i = 0; i < (noOfReplicas - 1); i++) {
+ req->newNodeOrder[i] = oldOrder[i+1];
+ }//for
+ req->newNodeOrder[noOfReplicas-1] = nodeId;
+ req->senderRef = reference();
+
+ /**
+ * Initialize struct
+ */
+ c_switchReplicas.tableId = tableId;
+ c_switchReplicas.fragNo = fragNo;
+ c_switchReplicas.nodeId = nodeId;
+
+ sendLoopMacro(DIH_SWITCH_REPLICA_REQ, sendDIH_SWITCH_REPLICA_REQ);
+ return;
+ }//for
+
+ signal->theData[0] = DihContinueB::SwitchReplica;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tableId;
+ signal->theData[3] = fragNo;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 4, JBB);
+}//Dbdih::switchReplica()
+
+void Dbdih::execSTOP_ME_REQ(Signal* signal)
+{
+ jamEntry();
+ StopMeReq* const req = (StopMeReq*)&signal->theData[0];
+ const BlockReference senderRef = req->senderRef;
+ const Uint32 senderData = req->senderData;
+ const Uint32 nodeId = refToNode(senderRef);
+ {
+ /**
+ * Set node dead (remove from operations)
+ */
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ nodePtr.p->useInTransactions = false;
+ }
+ if (nodeId != getOwnNodeId()) {
+ jam();
+ StopMeConf * const stopMeConf = (StopMeConf *)&signal->theData[0];
+ stopMeConf->senderData = senderData;
+ stopMeConf->senderRef = reference();
+ sendSignal(senderRef, GSN_STOP_ME_CONF, signal,
+ StopMeConf::SignalLength, JBB);
+ return;
+ }//if
+
+ /**
+ * Local signal
+ */
+ jam();
+ ndbrequire(c_stopMe.clientRef == 0);
+
+ c_stopMe.clientData = senderData;
+ c_stopMe.clientRef = senderRef;
+
+ req->senderData = senderData;
+ req->senderRef = reference();
+
+ sendLoopMacro(STOP_ME_REQ, sendSTOP_ME_REQ);
+
+ /**
+ * Send conf to self
+ */
+ StopMeConf * const stopMeConf = (StopMeConf *)&signal->theData[0];
+ stopMeConf->senderData = senderData;
+ stopMeConf->senderRef = reference();
+ sendSignal(reference(), GSN_STOP_ME_CONF, signal,
+ StopMeConf::SignalLength, JBB);
+}//Dbdih::execSTOP_ME_REQ()
+
+void Dbdih::execSTOP_ME_REF(Signal* signal)
+{
+ ndbrequire(false);
+}
+
+void Dbdih::execSTOP_ME_CONF(Signal* signal)
+{
+ jamEntry();
+ StopMeConf * const stopMeConf = (StopMeConf *)&signal->theData[0];
+
+ const Uint32 senderRef = stopMeConf->senderRef;
+ const Uint32 senderData = stopMeConf->senderData;
+ const Uint32 nodeId = refToNode(senderRef);
+
+ ndbrequire(c_stopMe.clientRef != 0);
+ ndbrequire(c_stopMe.clientData == senderData);
+
+ receiveLoopMacro(STOP_ME_REQ, nodeId);
+ //---------------------------------------------------------
+ // All STOP_ME_REQ have been received. We will send the
+ // confirmation back to the requesting block.
+ //---------------------------------------------------------
+
+ stopMeConf->senderRef = reference();
+ stopMeConf->senderData = c_stopMe.clientData;
+ sendSignal(c_stopMe.clientRef, GSN_STOP_ME_CONF, signal,
+ StopMeConf::SignalLength, JBB);
+ c_stopMe.clientRef = 0;
+}//Dbdih::execSTOP_ME_CONF()
+
+void Dbdih::execWAIT_GCP_REQ(Signal* signal)
+{
+ jamEntry();
+ WaitGCPReq* const req = (WaitGCPReq*)&signal->theData[0];
+ WaitGCPRef* const ref = (WaitGCPRef*)&signal->theData[0];
+ WaitGCPConf* const conf = (WaitGCPConf*)&signal->theData[0];
+ const Uint32 senderData = req->senderData;
+ const BlockReference senderRef = req->senderRef;
+ const Uint32 requestType = req->requestType;
+
+ if(requestType == WaitGCPReq::CurrentGCI) {
+ jam();
+ conf->senderData = senderData;
+ conf->gcp = cnewgcp;
+ sendSignal(senderRef, GSN_WAIT_GCP_CONF, signal,
+ WaitGCPConf::SignalLength, JBB);
+ return;
+ }//if
+
+ if(isMaster()) {
+ /**
+ * Master
+ */
+ jam();
+
+ if((requestType == WaitGCPReq::CompleteIfRunning) &&
+ (cgcpStatus == GCP_READY)) {
+ jam();
+ conf->senderData = senderData;
+ conf->gcp = coldgcp;
+ sendSignal(senderRef, GSN_WAIT_GCP_CONF, signal,
+ WaitGCPConf::SignalLength, JBB);
+ return;
+ }//if
+
+ WaitGCPMasterPtr ptr;
+ if(c_waitGCPMasterList.seize(ptr) == false){
+ jam();
+ ref->senderData = senderData;
+ ref->errorCode = WaitGCPRef::NoWaitGCPRecords;
+ sendSignal(senderRef, GSN_WAIT_GCP_REF, signal,
+ WaitGCPRef::SignalLength, JBB);
+ return;
+ }//if
+ ptr.p->clientRef = senderRef;
+ ptr.p->clientData = senderData;
+
+ if((requestType == WaitGCPReq::CompleteForceStart) &&
+ (cgcpStatus == GCP_READY)) {
+ jam();
+ cstartGcpNow = true;
+ }//if
+ return;
+ } else {
+ /**
+ * Proxy part
+ */
+ jam();
+ WaitGCPProxyPtr ptr;
+ if (c_waitGCPProxyList.seize(ptr) == false) {
+ jam();
+ ref->senderData = senderData;
+ ref->errorCode = WaitGCPRef::NoWaitGCPRecords;
+ sendSignal(senderRef, GSN_WAIT_GCP_REF, signal,
+ WaitGCPRef::SignalLength, JBB);
+ return;
+ }//if
+ ptr.p->clientRef = senderRef;
+ ptr.p->clientData = senderData;
+ ptr.p->masterRef = cmasterdihref;
+
+ req->senderData = ptr.i;
+ req->senderRef = reference();
+ req->requestType = requestType;
+
+ sendSignal(cmasterdihref, GSN_WAIT_GCP_REQ, signal,
+ WaitGCPReq::SignalLength, JBB);
+ return;
+ }//if
+}//Dbdih::execWAIT_GCP_REQ()
+
+void Dbdih::execWAIT_GCP_REF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(!isMaster());
+ WaitGCPRef* const ref = (WaitGCPRef*)&signal->theData[0];
+
+ const Uint32 proxyPtr = ref->senderData;
+ const Uint32 errorCode = ref->errorCode;
+
+ WaitGCPProxyPtr ptr;
+ ptr.i = proxyPtr;
+ c_waitGCPProxyList.getPtr(ptr);
+
+ ref->senderData = ptr.p->clientData;
+ ref->errorCode = errorCode;
+ sendSignal(ptr.p->clientRef, GSN_WAIT_GCP_REF, signal,
+ WaitGCPRef::SignalLength, JBB);
+
+ c_waitGCPProxyList.release(ptr);
+}//Dbdih::execWAIT_GCP_REF()
+
+void Dbdih::execWAIT_GCP_CONF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(!isMaster());
+ WaitGCPConf* const conf = (WaitGCPConf*)&signal->theData[0];
+ const Uint32 proxyPtr = conf->senderData;
+ const Uint32 gcp = conf->gcp;
+ WaitGCPProxyPtr ptr;
+
+ ptr.i = proxyPtr;
+ c_waitGCPProxyList.getPtr(ptr);
+
+ conf->senderData = ptr.p->clientData;
+ conf->gcp = gcp;
+ sendSignal(ptr.p->clientRef, GSN_WAIT_GCP_CONF, signal,
+ WaitGCPConf::SignalLength, JBB);
+
+ c_waitGCPProxyList.release(ptr);
+}//Dbdih::execWAIT_GCP_CONF()
+
+void Dbdih::checkWaitGCPProxy(Signal* signal, NodeId failedNodeId)
+{
+ jam();
+ WaitGCPRef* const ref = (WaitGCPRef*)&signal->theData[0];
+ ref->errorCode = WaitGCPRef::NF_CausedAbortOfProcedure;
+
+ WaitGCPProxyPtr ptr;
+ c_waitGCPProxyList.first(ptr);
+ while(ptr.i != RNIL) {
+ jam();
+ const Uint32 i = ptr.i;
+ const Uint32 clientData = ptr.p->clientData;
+ const BlockReference clientRef = ptr.p->clientRef;
+ const BlockReference masterRef = ptr.p->masterRef;
+
+ c_waitGCPProxyList.next(ptr);
+ if(refToNode(masterRef) == failedNodeId) {
+ jam();
+ c_waitGCPProxyList.release(i);
+ ref->senderData = clientData;
+ sendSignal(clientRef, GSN_WAIT_GCP_REF, signal,
+ WaitGCPRef::SignalLength, JBB);
+ }//if
+ }//while
+}//Dbdih::checkWaitGCPProxy()
+
+void Dbdih::checkWaitGCPMaster(Signal* signal, NodeId failedNodeId)
+{
+ jam();
+ WaitGCPMasterPtr ptr;
+ c_waitGCPMasterList.first(ptr);
+
+ while (ptr.i != RNIL) {
+ jam();
+ const Uint32 i = ptr.i;
+ const NodeId nodeId = refToNode(ptr.p->clientRef);
+
+ c_waitGCPMasterList.next(ptr);
+ if (nodeId == failedNodeId) {
+ jam()
+ c_waitGCPMasterList.release(i);
+ }//if
+ }//while
+}//Dbdih::checkWaitGCPMaster()
+
+void Dbdih::emptyWaitGCPMasterQueue(Signal* signal)
+{
+ jam();
+ WaitGCPConf* const conf = (WaitGCPConf*)&signal->theData[0];
+ conf->gcp = coldgcp;
+
+ WaitGCPMasterPtr ptr;
+ c_waitGCPMasterList.first(ptr);
+ while(ptr.i != RNIL) {
+ jam();
+ const Uint32 i = ptr.i;
+ const Uint32 clientData = ptr.p->clientData;
+ const BlockReference clientRef = ptr.p->clientRef;
+
+ c_waitGCPMasterList.next(ptr);
+ conf->senderData = clientData;
+ sendSignal(clientRef, GSN_WAIT_GCP_CONF, signal,
+ WaitGCPConf::SignalLength, JBB);
+
+ c_waitGCPMasterList.release(i);
+ }//while
+}//Dbdih::emptyWaitGCPMasterQueue()
+
+void Dbdih::setNodeStatus(Uint32 nodeId, NodeRecord::NodeStatus newStatus)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ nodePtr.p->nodeStatus = newStatus;
+}//Dbdih::setNodeStatus()
+
+Dbdih::NodeRecord::NodeStatus Dbdih::getNodeStatus(Uint32 nodeId)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ return nodePtr.p->nodeStatus;
+}//Dbdih::getNodeStatus()
+
+Sysfile::ActiveStatus
+Dbdih::getNodeActiveStatus(Uint32 nodeId)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ return nodePtr.p->activeStatus;
+}//Dbdih::getNodeActiveStatus()
+
+
+void
+Dbdih::setNodeActiveStatus(Uint32 nodeId, Sysfile::ActiveStatus newStatus)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ nodePtr.p->activeStatus = newStatus;
+}//Dbdih::setNodeActiveStatus()
+
+void Dbdih::setAllowNodeStart(Uint32 nodeId, bool newState)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ nodePtr.p->allowNodeStart = newState;
+}//Dbdih::setAllowNodeStart()
+
+void Dbdih::setNodeCopyCompleted(Uint32 nodeId, bool newState)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ nodePtr.p->copyCompleted = newState;
+}//Dbdih::setNodeCopyCompleted()
+
+bool Dbdih::getAllowNodeStart(Uint32 nodeId)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ return nodePtr.p->allowNodeStart;
+}//Dbdih::getAllowNodeStart()
+
+bool Dbdih::getNodeCopyCompleted(Uint32 nodeId)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ return nodePtr.p->copyCompleted;
+}//Dbdih::getNodeCopyCompleted()
+
+bool Dbdih::checkNodeAlive(Uint32 nodeId)
+{
+ NodeRecordPtr nodePtr;
+ nodePtr.i = nodeId;
+ ndbrequire(nodeId > 0);
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRecord);
+ if (nodePtr.p->nodeStatus != NodeRecord::ALIVE) {
+ return false;
+ } else {
+ return true;
+ }//if
+}//Dbdih::checkNodeAlive()
+
+bool Dbdih::isMaster()
+{
+ return (reference() == cmasterdihref);
+}//Dbdih::isMaster()
+
+bool Dbdih::isActiveMaster()
+{
+ return ((reference() == cmasterdihref) && (cmasterState == MASTER_ACTIVE));
+}//Dbdih::isActiveMaster()
diff --git a/ndb/src/kernel/blocks/dbdih/LCP.txt b/ndb/src/kernel/blocks/dbdih/LCP.txt
new file mode 100644
index 00000000000..500c82f6baf
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdih/LCP.txt
@@ -0,0 +1,35 @@
+
+Master DIH LQH
+========== ==========
+
+1) TCGETOPSIZEREQ -> all TC
+
+2) If sum(operation size) < Threshold
+ Goto 1
+
+3) For each table
+ Calc Keep GCI (local using CONTINUEB)
+
+4) COPY_GCIREQ -> all DIH
+
+5) TC_CLOPSIZEREQ -> all TC
+
+6) For each fragment
+ LCP_FRAG_ORD -> LQH
+
+ Do LCP...
+ 1) LCP_FRAG_REP -> all DIH
+ 2) If last fragment
+ LCP_COMPLETE_REP -> all DIH
+
+7) When receiving LCP_COMPLETE_REP from DIH
+ 1) If all DIHs have completed
+ Goto 1
+
+All DIHs
+==========
+1) When receiving LCP_FRAG_REP
+ If all fragments & replicas done in table
+ 1) Save Table descriptor
+ 2) If all tables done + LCP_COMPLETE_REP(from lqh) has arrived
+ LCP_COMPLETE_REP -> master DIH
diff --git a/ndb/src/kernel/blocks/dbdih/Makefile b/ndb/src/kernel/blocks/dbdih/Makefile
new file mode 100644
index 00000000000..83c1b95b5c4
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdih/Makefile
@@ -0,0 +1,13 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := dbdih
+
+DIRS := printSysfile
+
+SOURCES = \
+ DbdihInit.cpp \
+ DbdihMain.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dbdih/Sysfile.hpp b/ndb/src/kernel/blocks/dbdih/Sysfile.hpp
new file mode 100644
index 00000000000..a44992d6ad0
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdih/Sysfile.hpp
@@ -0,0 +1,275 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SYSFILE_HPP
+#define SYSFILE_HPP
+
+#include <ndb_types.h>
+#include <ndb_limits.h>
+#include <NodeBitmask.hpp>
+
+/**
+ * No bits in Sysfile to represent nodeid
+ */
+#define NODEID_BITS 8
+
+/**
+ * Constant representing that node do not belong to
+ * any node group
+ */
+#define NO_NODE_GROUP_ID ((1 << NODEID_BITS) - 1)
+
+/**
+ * Dummy macro to make emacs indent better
+ */
+#define _F(x) x
+
+/**
+ * No of 32 bits word in sysfile
+ *
+ * 5 +
+ * MAX_NDB_NODES + // lastCompletedGCI
+ * NODE_ARRAY_SIZE(MAX_NDB_NODES, 4) + // nodeStatus
+ * NODE_ARRAY_SIZE(MAX_NDB_NODES, NODEID_BITS) + // nodeGroups
+ * NODE_ARRAY_SIZE(MAX_NDB_NODES, NODEID_BITS) + // takeOver
+ * NodeBitmask::NDB_NODE_BITMASK_SIZE // Lcp Active
+ */
+#define _SYSFILE_SIZE32 (5 + \
+ MAX_NDB_NODES + \
+ NODE_ARRAY_SIZE(MAX_NDB_NODES, 4) + \
+ NODE_ARRAY_SIZE(MAX_NDB_NODES, NODEID_BITS) + \
+ NODE_ARRAY_SIZE(MAX_NDB_NODES, NODEID_BITS) + \
+ _NDB_NODE_BITMASK_SIZE)
+
+/**
+ * This struct defines the format of P<X>.sysfile
+ */
+struct Sysfile {
+public:
+
+ /**
+ * No of 32 bits words in the sysfile
+ */
+ static const Uint32 SYSFILE_SIZE32 = _SYSFILE_SIZE32;
+
+ Uint32 systemRestartBits;
+
+ static bool getInitialStartOngoing(const Uint32 & systemRestartBits);
+ static void setInitialStartOngoing(Uint32 & systemRestartBits);
+ static void clearInitialStartOngoing(Uint32 & systemRestartBits);
+
+ static bool getRestartOngoing(const Uint32 & systemRestartBits);
+ static void setRestartOngoing(Uint32 & systemRestartBits);
+ static void clearRestartOngoing(Uint32 & systemRestartBits);
+
+ static bool getLCPOngoing(const Uint32 & systemRestartBits);
+ static void setLCPOngoing(Uint32 & systemRestartBits);
+ static void clearLCPOngoing(Uint32 & systemRestartBits);
+
+ Uint32 keepGCI;
+ Uint32 oldestRestorableGCI;
+ Uint32 newestRestorableGCI;
+ Uint32 latestLCP_ID;
+
+ /**
+ * Last completed GCI for each node
+ */
+ Uint32 lastCompletedGCI[MAX_NDB_NODES];
+
+ /**
+ * Active status bits
+ *
+ * It takes 4 bits to represent it
+ */
+ enum ActiveStatus {
+ NS_Active = 0
+ ,NS_ActiveMissed_1 = 1
+ ,NS_ActiveMissed_2 = 2
+ ,NS_ActiveMissed_3 = 3
+ ,NS_HotSpare = 4
+ ,NS_NotActive_NotTakenOver = 5
+ ,NS_TakeOver = 6
+ ,NS_NotActive_TakenOver = 7
+ ,NS_NotDefined = 8
+ ,NS_Standby = 9
+ };
+ static const Uint32 NODE_STATUS_SIZE = NODE_ARRAY_SIZE(MAX_NDB_NODES, 4);
+ Uint32 nodeStatus[NODE_STATUS_SIZE];
+
+ static Uint32 getNodeStatus(NodeId, const Uint32 nodeStatus[]);
+ static void setNodeStatus(NodeId, Uint32 nodeStatus[], Uint32 status);
+
+ /**
+ * The node group of each node
+ * Sizeof(NodeGroup) = 8 Bit
+ */
+ static const Uint32 NODE_GROUPS_SIZE = NODE_ARRAY_SIZE(MAX_NDB_NODES,
+ NODEID_BITS);
+ Uint32 nodeGroups[NODE_GROUPS_SIZE];
+
+ static Uint16 getNodeGroup(NodeId, const Uint32 nodeGroups[]);
+ static void setNodeGroup(NodeId, Uint32 nodeGroups[], Uint16 group);
+
+ /**
+ * Any node can take over for any node
+ */
+ static const Uint32 TAKE_OVER_SIZE = NODE_ARRAY_SIZE(MAX_NDB_NODES,
+ NODEID_BITS);
+ Uint32 takeOver[TAKE_OVER_SIZE];
+
+ static NodeId getTakeOverNode(NodeId, const Uint32 takeOver[]);
+ static void setTakeOverNode(NodeId, Uint32 takeOver[], NodeId toNode);
+
+ /**
+ * Is a node running a LCP
+ */
+ Uint32 lcpActive[NdbNodeBitmask::Size];
+};
+
+#if (MAX_NDB_NODES > (1<<NODEID_BITS))
+#error "Sysfile node id is too small"
+#endif
+
+/**
+ * Restart Info
+ *
+ * i = Initial start completed
+ * r = Crash during system restart
+ * l = Crash during local checkpoint
+
+ * 1111111111222222222233
+ * 01234567890123456789012345678901
+ * irl
+ */
+inline
+bool
+Sysfile::getInitialStartOngoing(const Uint32 & systemRestartBits){
+ return systemRestartBits & 1;
+}
+
+inline
+void
+Sysfile::setInitialStartOngoing(Uint32 & systemRestartBits){
+ systemRestartBits |= 1;
+}
+
+inline
+void
+Sysfile::clearInitialStartOngoing(Uint32 & systemRestartBits){
+ systemRestartBits &= ~1;
+}
+
+inline
+bool
+Sysfile::getRestartOngoing(const Uint32 & systemRestartBits){
+ return (systemRestartBits & 2) != 0;
+}
+
+inline
+void
+Sysfile::setRestartOngoing(Uint32 & systemRestartBits){
+ systemRestartBits |= 2;
+}
+
+inline
+void
+Sysfile::clearRestartOngoing(Uint32 & systemRestartBits){
+ systemRestartBits &= ~2;
+}
+
+inline
+bool
+Sysfile::getLCPOngoing(const Uint32 & systemRestartBits){
+ return systemRestartBits & 4;
+}
+
+inline
+void
+Sysfile::setLCPOngoing(Uint32 & systemRestartBits){
+ systemRestartBits |= 4;
+}
+
+inline
+void
+Sysfile::clearLCPOngoing(Uint32 & systemRestartBits){
+ systemRestartBits &= ~4;
+}
+
+inline
+Uint32
+Sysfile::getNodeStatus(NodeId nodeId, const Uint32 nodeStatus[]){
+ const int word = nodeId >> 3;
+ const int shift = (nodeId & 7) << 2;
+
+ return (nodeStatus[word] >> shift) & 15;
+}
+
+inline
+void
+Sysfile::setNodeStatus(NodeId nodeId, Uint32 nodeStatus[], Uint32 status){
+ const int word = nodeId >> 3;
+ const int shift = (nodeId & 7) << 2;
+
+ const Uint32 mask = ~(((Uint32)15) << shift);
+ const Uint32 tmp = nodeStatus[word];
+
+ nodeStatus[word] = (tmp & mask) | ((status & 15) << shift);
+}
+
+inline
+Uint16
+Sysfile::getNodeGroup(NodeId nodeId, const Uint32 nodeGroups[]){
+ const int word = nodeId >> 2;
+ const int shift = (nodeId & 3) << 3;
+
+ return (nodeGroups[word] >> shift) & 255;
+}
+
+inline
+void
+Sysfile::setNodeGroup(NodeId nodeId, Uint32 nodeGroups[], Uint16 group){
+ const int word = nodeId >> 2;
+ const int shift = (nodeId & 3) << 3;
+
+ const Uint32 mask = ~(((Uint32)255) << shift);
+ const Uint32 tmp = nodeGroups[word];
+
+ nodeGroups[word] = (tmp & mask) | ((group & 255) << shift);
+}
+
+inline
+NodeId
+Sysfile::getTakeOverNode(NodeId nodeId, const Uint32 takeOver[]){
+ const int word = nodeId >> 2;
+ const int shift = (nodeId & 3) << 3;
+
+ return (takeOver[word] >> shift) & 255;
+}
+
+inline
+void
+Sysfile::setTakeOverNode(NodeId nodeId, Uint32 takeOver[], NodeId toNode){
+ const int word = nodeId >> 2;
+ const int shift = (nodeId & 3) << 3;
+
+ const Uint32 mask = ~(((Uint32)255) << shift);
+ const Uint32 tmp = takeOver[word];
+
+ takeOver[word] = (tmp & mask) | ((toNode & 255) << shift);
+}
+
+
+#endif
diff --git a/ndb/src/kernel/blocks/dbdih/printSysfile/Makefile b/ndb/src/kernel/blocks/dbdih/printSysfile/Makefile
new file mode 100644
index 00000000000..4c4b1026aff
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdih/printSysfile/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := printSysfile
+BIN_TARGET_ARCHIVES := portlib general
+
+CCFLAGS_LOC += -I..
+
+SOURCES := printSysfile.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dbdih/printSysfile/printSysfile.cpp b/ndb/src/kernel/blocks/dbdih/printSysfile/printSysfile.cpp
new file mode 100644
index 00000000000..4c55425bdd7
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbdih/printSysfile/printSysfile.cpp
@@ -0,0 +1,160 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <Sysfile.hpp>
+#include <stdlib.h>
+#include <string.h>
+#include <sys/stat.h>
+
+void
+usage(const char * prg){
+ ndbout << "Usage " << prg
+ << " P[0-1].sysfile" << endl;
+}
+
+struct NSString {
+ Sysfile::ActiveStatus NodeStatus;
+ const char * desc;
+};
+
+static const
+NSString NodeStatusStrings[] = {
+ { Sysfile::NS_Active, "Active " },
+ { Sysfile::NS_ActiveMissed_1, "Active missed 1" },
+ { Sysfile::NS_ActiveMissed_2, "Active missed 2" },
+ { Sysfile::NS_ActiveMissed_3, "Active missed 3" },
+ { Sysfile::NS_HotSpare, "Hot spare " },
+ { Sysfile::NS_NotActive_NotTakenOver, "Not active " },
+ { Sysfile::NS_TakeOver, "Take over " },
+ { Sysfile::NS_NotActive_TakenOver, "Taken over " },
+ { Sysfile::NS_NotDefined, "Not defined " },
+ { Sysfile::NS_Standby, "Stand by " }
+};
+
+const
+char * getNSString(Uint32 ns){
+ for(Uint32 i = 0; i<(sizeof(NodeStatusStrings)/sizeof(NSString)); i++)
+ if((Uint32)NodeStatusStrings[i].NodeStatus == ns)
+ return NodeStatusStrings[i].desc;
+ return "<Unknown state>";
+}
+
+void
+fill(const char * buf, int mod){
+ int len = strlen(buf)+1;
+ ndbout << buf << " ";
+ while((len % mod) != 0){
+ ndbout << " ";
+ len++;
+ }
+}
+
+void
+print(const char * filename, const Sysfile * sysfile){
+ char buf[255];
+ ndbout << "----- Sysfile: " << filename << " -----" << endl;
+ ndbout << "Initial start ongoing: "
+ << Sysfile::getInitialStartOngoing(sysfile->systemRestartBits)
+ << ", ";
+
+ ndbout << "Restart Ongoing: "
+ << Sysfile::getRestartOngoing(sysfile->systemRestartBits)
+ << ", ";
+
+ ndbout << "LCP Ongoing: "
+ << Sysfile::getLCPOngoing(sysfile->systemRestartBits)
+ << endl;
+
+
+ ndbout << "-- Global Checkpoint Identities: --" << endl;
+ sprintf(buf, "keepGCI = %u", sysfile->keepGCI);
+ fill(buf, 40);
+ ndbout << " -- Tail of REDO log" << endl;
+
+ sprintf(buf, "oldestRestorableGCI = %u", sysfile->oldestRestorableGCI);
+ fill(buf, 40);
+ ndbout << " -- " << endl;
+
+ sprintf(buf, "newestRestorableGCI = %u", sysfile->newestRestorableGCI);
+ fill(buf, 40);
+ ndbout << " -- " << endl;
+
+ sprintf(buf, "latestLCP = %u", sysfile->latestLCP_ID);
+ fill(buf, 40);
+ ndbout << " -- " << endl;
+
+ ndbout << "-- Node status: --" << endl;
+ for(int i = 1; i < MAX_NDB_NODES; i++){
+ if(Sysfile::getNodeStatus(i, sysfile->nodeStatus) !=Sysfile::NS_NotDefined){
+ sprintf(buf,
+ "Node %.2d -- %s GCP: %d, NodeGroup: %d, TakeOverNode: %d, "
+ "LCP Ongoing: %s",
+ i,
+ getNSString(Sysfile::getNodeStatus(i,sysfile->nodeStatus)),
+ sysfile->lastCompletedGCI[i],
+ Sysfile::getNodeGroup(i, sysfile->nodeGroups),
+ Sysfile::getTakeOverNode(i, sysfile->takeOver),
+ BitmaskImpl::get(NdbNodeBitmask::Size,
+ sysfile->lcpActive, i) != 0 ? "yes" : "no");
+ ndbout << buf << endl;
+ }
+ }
+}
+
+NDB_COMMAND(printSysfile,
+ "printSysfile", "printSysfile", "Prints a sysfile", 16384){
+ if(argc < 2){
+ usage(argv[0]);
+ return 0;
+ }
+
+ for(int i = 1; i<argc; i++){
+ const char * filename = argv[i];
+
+ struct stat sbuf;
+ const int res = stat(filename, &sbuf);
+ if(res != 0){
+ ndbout << "Could not find file: \"" << filename << "\"" << endl;
+ continue;
+ }
+ const Uint32 bytes = sbuf.st_size;
+
+ Uint32 * buf = new Uint32[bytes/4+1];
+
+ FILE * f = fopen(filename, "rb");
+ if(f == 0){
+ ndbout << "Failed to open file" << endl;
+ delete [] buf;
+ continue;
+ }
+ Uint32 sz = fread(buf, 1, bytes, f);
+ fclose(f);
+ if(sz != bytes){
+ ndbout << "Failure while reading file" << endl;
+ delete [] buf;
+ continue;
+ }
+
+ print(filename, (Sysfile *)&buf[0]);
+ delete [] buf;
+ continue;
+ }
+ return 0;
+}
diff --git a/ndb/src/kernel/blocks/dblqh/Dblqh.hpp b/ndb/src/kernel/blocks/dblqh/Dblqh.hpp
new file mode 100644
index 00000000000..6b85ca11b27
--- /dev/null
+++ b/ndb/src/kernel/blocks/dblqh/Dblqh.hpp
@@ -0,0 +1,2899 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBLQH_H
+#define DBLQH_H
+
+#include <pc.hpp>
+#include <ndb_limits.h>
+#include <SimulatedBlock.hpp>
+#include <DLHashTable.hpp>
+
+#include <NodeBitmask.hpp>
+#include <signaldata/LCP.hpp>
+#include <signaldata/LqhTransConf.hpp>
+#include <signaldata/LqhFrag.hpp>
+
+#ifdef DBLQH_C
+// Constants
+/* ------------------------------------------------------------------------- */
+/* CONSTANTS USED WHEN MASTER REQUESTS STATE OF COPY FRAGMENTS. */
+/* ------------------------------------------------------------------------- */
+#define ZCOPY_CLOSING 0
+#define ZCOPY_ONGOING 1
+#define ZCOPY_ACTIVATION 2
+/* ------------------------------------------------------------------------- */
+/* STATES FOR THE VARIABLE GCP_LOG_PART_STATE */
+/* ------------------------------------------------------------------------- */
+#define ZIDLE 0
+#define ZWAIT_DISK 1
+#define ZON_DISK 2
+#define ZACTIVE 1
+/* ------------------------------------------------------------------------- */
+/* STATES FOR THE VARIABLE CSR_PHASES_STARTED */
+/* ------------------------------------------------------------------------- */
+#define ZSR_NO_PHASE_STARTED 0
+#define ZSR_PHASE1_COMPLETED 1
+#define ZSR_PHASE2_COMPLETED 2
+#define ZSR_BOTH_PHASES_STARTED 3
+/* ------------------------------------------------------------------------- */
+/* THE NUMBER OF PAGES IN A MBYTE, THE TWO LOGARITHM OF THIS. */
+/* THE NUMBER OF MBYTES IN A LOG FILE. */
+/* THE MAX NUMBER OF PAGES READ/WRITTEN FROM/TO DISK DURING */
+/* A WRITE OR READ. */
+/* ------------------------------------------------------------------------- */
+#define ZNOT_DIRTY 0
+#define ZDIRTY 1
+#define ZREAD_AHEAD_SIZE 8
+/* ------------------------------------------------------------------------- */
+/* CONSTANTS OF THE LOG PAGES */
+/* ------------------------------------------------------------------------- */
+#define ZPAGE_HEADER_SIZE 32
+#if defined NDB_OSE
+/**
+ * Set the fragment log file size to 2Mb in OSE
+ * This is done in order to speed up the initial start
+ */
+#define ZNO_MBYTES_IN_FILE 2
+#define ZPAGE_SIZE 2048
+#define ZPAGES_IN_MBYTE 128
+#define ZTWOLOG_NO_PAGES_IN_MBYTE 7
+#define ZTWOLOG_PAGE_SIZE 11
+#define ZMAX_MM_BUFFER_SIZE 32 // Main memory window during log execution
+#else
+#define ZNO_MBYTES_IN_FILE 16
+#define ZPAGE_SIZE 8192
+#define ZPAGES_IN_MBYTE 32
+#define ZTWOLOG_NO_PAGES_IN_MBYTE 5
+#define ZTWOLOG_PAGE_SIZE 13
+#define ZMAX_MM_BUFFER_SIZE 32 // Main memory window during log execution
+#endif
+
+#define ZMAX_PAGES_WRITTEN 8 // Max pages before writing to disk (=> config)
+#define ZMIN_READ_BUFFER_SIZE 2 // Minimum number of pages to execute log
+#define ZMIN_LOG_PAGES_OPERATION 10 // Minimum no of pages before stopping
+
+#define ZPOS_CHECKSUM 0
+#define ZPOS_LOG_LAP 1
+#define ZPOS_MAX_GCI_COMPLETED 2
+#define ZPOS_MAX_GCI_STARTED 3
+#define ZNEXT_PAGE 4
+#define ZPREV_PAGE 5
+#define ZPOS_VERSION 6
+#define ZPOS_NO_LOG_FILES 7
+#define ZCURR_PAGE_INDEX 8
+#define ZLAST_LOG_PREP_REF 10
+#define ZPOS_DIRTY 11
+/* ------------------------------------------------------------------------- */
+/* CONSTANTS FOR THE VARIOUS REPLICA AND NODE TYPES. */
+/* ------------------------------------------------------------------------- */
+#define ZPRIMARY_NODE 0
+#define ZBACKUP_NODE 1
+#define ZSTANDBY_NODE 2
+#define ZTC_NODE 3
+#define ZLOG_NODE 3
+/* ------------------------------------------------------------------------- */
+/* VARIOUS CONSTANTS USED AS FLAGS TO THE FILE MANAGER. */
+/* ------------------------------------------------------------------------- */
+#define ZOPEN_READ 0
+#define ZOPEN_WRITE 1
+#define ZOPEN_READ_WRITE 2
+#define ZVAR_NO_LOG_PAGE_WORD 1
+#define ZLIST_OF_PAIRS 0
+#define ZLIST_OF_PAIRS_SYNCH 16
+#define ZARRAY_OF_PAGES 1
+#define ZLIST_OF_MEM_PAGES 2
+#define ZLIST_OF_MEM_PAGES_SYNCH 18
+#define ZCLOSE_NO_DELETE 0
+#define ZCLOSE_DELETE 1
+#define ZPAGE_ZERO 0
+/* ------------------------------------------------------------------------- */
+/* THE FOLLOWING CONSTANTS ARE USED TO DESCRIBE THE TYPES OF */
+/* LOG RECORDS, THE SIZE OF THE VARIOUS LOG RECORD TYPES AND */
+/* THE POSITIONS WITHIN THOSE LOG RECORDS. */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* THESE CONSTANTS DESCRIBE THE SIZES OF VARIOUS TYPES OF LOG REORDS. */
+/* NEXT_LOG_SIZE IS ACTUALLY ONE. THE REASON WE SET IT TO 2 IS TO */
+/* SIMPLIFY THE CODE SINCE OTHERWISE HAVE TO USE A SPECIAL VERSION */
+/* OF READ_LOGWORD WHEN READING LOG RECORD TYPE */
+/* SINCE NEXT MBYTE TYPE COULD BE THE VERY LAST WORD IN THE MBYTE. */
+/* BY SETTING IT TO 2 WE ENSURE IT IS NEVER THE VERY LAST WORD */
+/* IN THE MBYTE. */
+/* ------------------------------------------------------------------------- */
+#define ZFD_HEADER_SIZE 3
+#define ZFD_PART_SIZE 48
+#define ZLOG_HEAD_SIZE 6
+#define ZNEXT_LOG_SIZE 2
+#define ZABORT_LOG_SIZE 3
+#define ZCOMMIT_LOG_SIZE 9
+#define ZCOMPLETED_GCI_LOG_SIZE 2
+/* ------------------------------------------------------------------------- */
+/* THESE CONSTANTS DESCRIBE THE TYPE OF A LOG RECORD. */
+/* THIS IS THE FIRST WORD OF A LOG RECORD. */
+/* ------------------------------------------------------------------------- */
+#define ZNEW_PREP_OP_TYPE 0
+#define ZPREP_OP_TYPE 1
+#define ZCOMMIT_TYPE 2
+#define ZABORT_TYPE 3
+#define ZFD_TYPE 4
+#define ZFRAG_SPLIT_TYPE 5
+#define ZNEXT_LOG_RECORD_TYPE 6
+#define ZNEXT_MBYTE_TYPE 7
+#define ZCOMPLETED_GCI_TYPE 8
+#define ZINVALID_COMMIT_TYPE 9
+/* ------------------------------------------------------------------------- */
+/* THE POSITIONS OF LOGGED DATA IN A FILE DESCRIPTOR LOG RECORD HEADER.*/
+/* ALSO THE MAXIMUM NUMBER OF FILE DESCRIPTORS IN A LOG RECORD. */
+/* ------------------------------------------------------------------------- */
+#define ZPOS_LOG_TYPE 0
+#define ZPOS_NO_FD 1
+#define ZPOS_FILE_NO 2
+#define ZMAX_LOG_FILES_IN_PAGE_ZERO 40
+/* ------------------------------------------------------------------------- */
+/* THE POSITIONS WITHIN A PREPARE LOG RECORD AND A NEW PREPARE */
+/* LOG RECORD. */
+/* ------------------------------------------------------------------------- */
+#define ZPOS_HASH_VALUE 2
+#define ZPOS_SCHEMA_VERSION 3
+#define ZPOS_TRANS_TICKET 4
+#define ZPOS_OP_TYPE 5
+#define ZPOS_NO_ATTRINFO 6
+#define ZPOS_NO_KEYINFO 7
+/* ------------------------------------------------------------------------- */
+/* THE POSITIONS WITHIN A COMMIT LOG RECORD. */
+/* ------------------------------------------------------------------------- */
+#define ZPOS_COMMIT_TRANSID1 1
+#define ZPOS_COMMIT_TRANSID2 2
+#define ZPOS_COMMIT_GCI 3
+#define ZPOS_COMMIT_TABLE_REF 4
+#define ZPOS_COMMIT_FRAGID 5
+#define ZPOS_COMMIT_FILE_NO 6
+#define ZPOS_COMMIT_START_PAGE_NO 7
+#define ZPOS_COMMIT_START_PAGE_INDEX 8
+#define ZPOS_COMMIT_STOP_PAGE_NO 9
+/* ------------------------------------------------------------------------- */
+/* THE POSITIONS WITHIN A ABORT LOG RECORD. */
+/* ------------------------------------------------------------------------- */
+#define ZPOS_ABORT_TRANSID1 1
+#define ZPOS_ABORT_TRANSID2 2
+/* ------------------------------------------------------------------------- */
+/* THE POSITION WITHIN A COMPLETED GCI LOG RECORD. */
+/* ------------------------------------------------------------------------- */
+#define ZPOS_COMPLETED_GCI 1
+/* ------------------------------------------------------------------------- */
+/* THE POSITIONS WITHIN A NEW PREPARE LOG RECORD. */
+/* ------------------------------------------------------------------------- */
+#define ZPOS_NEW_PREP_FILE_NO 8
+#define ZPOS_NEW_PREP_PAGE_REF 9
+
+#define ZLAST_WRITE_IN_FILE 1
+#define ZENFORCE_WRITE 2
+/* ------------------------------------------------------------------------- */
+/* CONSTANTS USED AS INPUT TO SUBROUTINE WRITE_LOG_PAGES AMONG OTHERS. */
+/* ------------------------------------------------------------------------- */
+#define ZNORMAL 0
+#define ZINIT 1
+/* ------------------------------------------------------------------------- */
+/* CONSTANTS USED BY CONTINUEB TO DEDUCE WHICH CONTINUE SIGNAL IS TO */
+/* BE EXECUTED AS A RESULT OF THIS CONTINUEB SIGNAL. */
+/* ------------------------------------------------------------------------- */
+#define ZLOG_LQHKEYREQ 0
+#define ZPACK_LQHKEYREQ 1
+#define ZSEND_ATTRINFO 2
+#define ZSR_GCI_LIMITS 3
+#define ZSR_LOG_LIMITS 4
+#define ZSEND_EXEC_CONF 5
+#define ZEXEC_SR 6
+#define ZSR_FOURTH_COMP 7
+#define ZINIT_FOURTH 8
+#define ZTIME_SUPERVISION 9
+#define ZSR_PHASE3_START 10
+#define ZLQH_TRANS_NEXT 11
+#define ZLQH_RELEASE_AT_NODE_FAILURE 12
+#define ZSCAN_TC_CONNECT 13
+#define ZINITIALISE_RECORDS 14
+#define ZINIT_GCP_REC 15
+#define ZRESTART_OPERATIONS_AFTER_STOP 16
+#define ZCHECK_LCP_STOP_BLOCKED 17
+#define ZSCAN_MARKERS 18
+#define ZOPERATION_EVENT_REP 19
+#define ZPREP_DROP_TABLE 20
+
+/* ------------------------------------------------------------------------- */
+/* NODE STATE DURING SYSTEM RESTART, VARIABLES CNODES_SR_STATE */
+/* AND CNODES_EXEC_SR_STATE. */
+/* ------------------------------------------------------------------------- */
+#define ZSTART_SR 1
+#define ZEXEC_SR_COMPLETED 2
+/* ------------------------------------------------------------------------- */
+/* CONSTANTS USED BY NODE STATUS TO DEDUCE THE STATUS OF A NODE. */
+/* ------------------------------------------------------------------------- */
+#define ZNODE_UP 0
+#define ZNODE_DOWN 1
+/* ------------------------------------------------------------------------- */
+/* OPERATION TYPES */
+/* ------------------------------------------------------------------------- */
+#define ZSIMPLE_READ 1
+/* ------------------------------------------------------------------------- */
+/* START PHASES */
+/* ------------------------------------------------------------------------- */
+#define ZLAST_START_PHASE 255
+#define ZSTART_PHASE1 1
+#define ZSTART_PHASE2 2
+#define ZSTART_PHASE3 3
+#define ZSTART_PHASE4 4
+#define ZSTART_PHASE6 6
+/* ------------------------------------------------------------------------- */
+/* CONSTANTS USED BY SCAN AND COPY FRAGMENT PROCEDURES */
+/* ------------------------------------------------------------------------- */
+#define ZSTORED_PROC_SCAN 0
+#define ZSTORED_PROC_COPY 2
+#define ZDELETE_STORED_PROC_ID 3
+//#define ZSCAN_NEXT 1
+//#define ZSCAN_NEXT_COMMIT 2
+//#define ZSCAN_NEXT_ABORT 12
+#define ZCOPY_COMMIT 3
+#define ZCOPY_REPEAT 4
+#define ZCOPY_ABORT 5
+#define ZCOPY_CLOSE 6
+//#define ZSCAN_CLOSE 6
+//#define ZEMPTY_FRAGMENT 0
+#define ZWRITE_LOCK 1
+#define ZSCAN_FRAG_CLOSED 2
+/* ------------------------------------------------------------------------- */
+/* ERROR CODES ADDED IN VERSION 0.1 AND 0.2 */
+/* ------------------------------------------------------------------------- */
+#define ZNOT_FOUND 1 // Not an error code, a return value
+#define ZNO_FREE_LQH_CONNECTION 414
+#define ZGET_DATAREC_ERROR 418
+#define ZGET_ATTRINBUF_ERROR 419
+#define ZNO_FREE_FRAGMENTREC 460 // Insert new fragment error code
+#define ZTAB_FILE_SIZE 464 // Insert new fragment error code + Start kernel
+#define ZNO_ADD_FRAGREC 465 // Insert new fragment error code
+/* ------------------------------------------------------------------------- */
+/* ERROR CODES ADDED IN VERSION 0.3 */
+/* ------------------------------------------------------------------------- */
+#define ZTAIL_PROBLEM_IN_LOG_ERROR 410
+#define ZGCI_TOO_LOW_ERROR 429 // GCP_SAVEREF error code
+#define ZTAB_STATE_ERROR 474 // Insert new fragment error code
+#define ZTOO_NEW_GCI_ERROR 479 // LCP Start error
+/* ------------------------------------------------------------------------- */
+/* ERROR CODES ADDED IN VERSION 0.4 */
+/* ------------------------------------------------------------------------- */
+
+#define ZNO_FREE_FRAG_SCAN_REC_ERROR 490 // SCAN_FRAGREF error code
+#define ZCOPY_NO_FRAGMENT_ERROR 491 // COPY_FRAGREF error code
+#define ZTAKE_OVER_ERROR 499
+#define ZCOPY_NODE_ERROR 1204
+#define ZTOO_MANY_COPY_ACTIVE_ERROR 1208 // COPY_FRAG and COPY_ACTIVEREF code
+#define ZCOPY_ACTIVE_ERROR 1210 // COPY_ACTIVEREF error code
+#define ZNO_TC_CONNECT_ERROR 1217 // Simple Read + SCAN
+/* ------------------------------------------------------------------------- */
+/* ERROR CODES ADDED IN VERSION 1.X */
+/* ------------------------------------------------------------------------- */
+//#define ZSCAN_BOOK_ACC_OP_ERROR 1219 // SCAN_FRAGREF error code
+#define ZFILE_CHANGE_PROBLEM_IN_LOG_ERROR 1220
+#define ZTEMPORARY_REDO_LOG_FAILURE 1221
+#define ZNO_FREE_MARKER_RECORDS_ERROR 1222
+#define ZNODE_SHUTDOWN_IN_PROGESS 1223
+#define ZTOO_MANY_FRAGMENTS 1224
+#define ZTABLE_NOT_DEFINED 1225
+#define ZDROP_TABLE_IN_PROGRESS 1226
+#define ZINVALID_SCHEMA_VERSION 1227
+
+/* ------------------------------------------------------------------------- */
+/* ERROR CODES ADDED IN VERSION 2.X */
+/* ------------------------------------------------------------------------- */
+#define ZNODE_FAILURE_ERROR 400
+/* ------------------------------------------------------------------------- */
+/* ERROR CODES FROM ACC */
+/* ------------------------------------------------------------------------- */
+#define ZNO_TUPLE_FOUND 626
+#define ZTUPLE_ALREADY_EXIST 630
+/* ------------------------------------------------------------------------- */
+/* ERROR CODES FROM TUP */
+/* ------------------------------------------------------------------------- */
+#define ZSEARCH_CONDITION_FALSE 899
+#define ZUSER_ERROR_CODE_LIMIT 6000
+#endif
+
+/**
+ * @class dblqh
+ *
+ * @section secIntro Introduction
+ *
+ * Dblqh is the coordinator of the LDM. Dblqh is responsible for
+ * performing operations on tuples. It does this job with help of
+ * Dbacc block (that manages the index structures) and Dbtup
+ * (that manages the tuples).
+ *
+ * Dblqh also keeps track of the participants and acts as a coordinator of
+ * 2-phase commits. Logical redo logging is also handled by the Dblqh
+ * block.
+ *
+ * @section secModules Modules
+ *
+ * The code is partitioned into the following modules:
+ * - START / RESTART
+ * - Start phase 1: Load our block reference and our processor id
+ * - Start phase 2: Initiate all records within the block
+ * Connect LQH with ACC and TUP.
+ * - Start phase 4: Connect LQH with LQH. Connect every LQH with
+ * every LQH in the database system.
+ * If initial start, then create the fragment log files.
+ * If system restart or node restart,
+ * then open the fragment log files and
+ * find the end of the log files.
+ * - ADD / DELETE FRAGMENT<br>
+ * Used by dictionary to create new fragments and delete old fragments.
+ * - EXECUTION<br>
+ * handles the reception of lqhkeyreq and all processing
+ * of operations on behalf of this request.
+ * This does also involve reception of various types of attrinfo
+ * and keyinfo.
+ * It also involves communication with ACC and TUP.
+ * - LOG<br>
+ * The log module handles the reading and writing of the log.
+ * It is also responsible for handling system restart.
+ * It controls the system restart in TUP and ACC as well.
+ * - TRANSACTION<br>
+ * This module handles the commit and the complete phases.
+ * - MODULE TO HANDLE TC FAILURE<br>
+ * - SCAN<br>
+ * This module contains the code that handles a scan of a particular
+ * fragment.
+ * It operates under the control of TC and orders ACC to
+ * perform a scan of all tuples in the fragment.
+ * TUP performs the necessary search conditions
+ * to ensure that only valid tuples are returned to the application.
+ * - NODE RECOVERY<br>
+ * Used when a node has failed.
+ * It performs a copy of a fragment to a new replica of the fragment.
+ * It does also shut down all connections to the failed node.
+ * - LOCAL CHECKPOINT<br>
+ * Handles execution and control of LCPs
+ * It controls the LCPs in TUP and ACC.
+ * It also interacts with DIH to control which GCPs are recoverable.
+ * - GLOBAL CHECKPOINT<br>
+ * Helps DIH in discovering when GCPs are recoverable.
+ * It handles the request gcp_savereq that requests LQH to
+ * save a particular GCP to disk and respond when completed.
+ * - FILE HANDLING<br>
+ * With submodules:
+ * - SIGNAL RECEPTION
+ * - NORMAL OPERATION
+ * - FILE CHANGE
+ * - INITIAL START
+ * - SYSTEM RESTART PHASE ONE
+ * - SYSTEM RESTART PHASE TWO,
+ * - SYSTEM RESTART PHASE THREE
+ * - SYSTEM RESTART PHASE FOUR
+ * - ERROR
+ * - TEST
+ * - LOG
+ */
+class Dblqh: public SimulatedBlock {
+public:
+
+ enum LcpCloseState {
+ LCP_IDLE = 0,
+ LCP_RUNNING = 1, // LCP is running
+ LCP_CLOSE_STARTED = 2, // Completion(closing of files) has started
+ ACC_LCP_CLOSE_COMPLETED = 3,
+ TUP_LCP_CLOSE_COMPLETED = 4
+ };
+
+ enum ExecUndoLogState {
+ EULS_IDLE = 0,
+ EULS_STARTED = 1,
+ EULS_COMPLETED = 2,
+ EULS_ACC_COMPLETED = 3,
+ EULS_TUP_COMPLETED = 4
+ };
+
+ struct AddFragRecord {
+ enum AddFragStatus {
+ FREE = 0,
+ ACC_ADDFRAG = 1,
+ WAIT_TWO_TUP = 2,
+ WAIT_ONE_TUP = 3,
+ WAIT_TWO_TUX = 4,
+ WAIT_ONE_TUX = 5,
+ WAIT_ADD_ATTR = 6,
+ TUP_ATTR_WAIT1 = 7,
+ TUP_ATTR_WAIT2 = 8,
+ TUX_ATTR_WAIT1 = 9,
+ TUX_ATTR_WAIT2 = 10
+ };
+ LqhAddAttrReq::Entry attributes[LqhAddAttrReq::MAX_ATTRIBUTES];
+ UintR accConnectptr;
+ AddFragStatus addfragStatus;
+ UintR dictConnectptr;
+ UintR fragmentPtr;
+ UintR nextAddfragrec;
+ UintR noOfAllocPages;
+ UintR schemaVer;
+ UintR tup1Connectptr;
+ UintR tup2Connectptr;
+ UintR tux1Connectptr;
+ UintR tux2Connectptr;
+ UintR checksumIndicator;
+ UintR GCPIndicator;
+ BlockReference dictBlockref;
+ Uint32 m_senderAttrPtr;
+ Uint16 addfragErrorCode;
+ Uint16 attrSentToTup;
+ Uint16 attrReceived;
+ Uint16 addFragid;
+ Uint16 fragid1;
+ Uint16 fragid2;
+ Uint16 noOfAttr;
+ Uint16 noOfNull;
+ Uint16 tabId;
+ Uint16 totalAttrReceived;
+ Uint16 fragCopyCreation;
+ Uint16 noOfKeyAttr;
+ Uint16 noOfNewAttr;
+ Uint16 noOfAttributeGroups;
+ Uint16 lh3DistrBits;
+ Uint16 tableType;
+ Uint16 primaryTableId;
+ };// Size 108 bytes
+ typedef Ptr<AddFragRecord> AddFragRecordPtr;
+
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$ ATTRIBUTE INFORMATION RECORD $$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /**
+ * Can contain one (1) attrinfo signal.
+ * One signal contains 24 attr. info words.
+ * But 32 elements are used to make plex happy.
+ * Some of the elements are used to the following things:
+ * - Data length in this record is stored in the
+ * element indexed by ZINBUF_DATA_LEN.
+ * - Next attrinbuf is pointed out by the element
+ * indexed by ZINBUF_NEXT.
+ */
+ struct Attrbuf {
+ UintR attrbuf[32];
+ }; // Size 128 bytes
+ typedef Ptr<Attrbuf> AttrbufPtr;
+
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$ DATA BUFFER $$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /**
+ * This buffer is used as a general data storage.
+ */
+ struct Databuf {
+ UintR data[4];
+ UintR nextDatabuf;
+ }; // size 20 bytes
+ typedef Ptr<Databuf> DatabufPtr;
+
+ struct Fragrecord {
+ enum ExecSrStatus {
+ IDLE = 0,
+ ACTIVE_REMOVE_AFTER = 1,
+ ACTIVE = 2
+ };
+ /**
+ * Possible state transitions are:
+ * - FREE -> DEFINED Fragment record is allocated
+ * - DEFINED -> ACTIVE Add fragment is completed and
+ * fragment is ready to
+ * receive operations.
+ * - DEFINED -> ACTIVE_CREATION Add fragment is completed and
+ * fragment is ready to
+ * receive operations in parallel
+ * with a copy fragment
+ * which is performed from the
+ * primary replica
+ * - DEFINED -> CRASH_RECOVERING A fragment is ready to be
+ * recovered from a local
+ * checkpoint on disk
+ * - ACTIVE -> BLOCKED A local checkpoint is to be
+ * started. No more operations
+ * are allowed to be started until
+ * the local checkpoint
+ * has been started.
+ * - ACTIVE -> REMOVING A fragment is removed from the node
+ * - BLOCKED -> ACTIVE Operations are allowed again in
+ * the fragment.
+ * - CRASH_RECOVERING -> ACTIVE A fragment has been recovered and
+ * are now ready for
+ * operations again.
+ * - CRASH_RECOVERING -> REMOVING Fragment recovery failed or
+ * was cancelled.
+ * - ACTIVE_CREATION -> ACTIVE A fragment is now copied and now
+ * is a normal fragment
+ * - ACTIVE_CREATION -> REMOVING Copying of the fragment failed
+ * - REMOVING -> FREE Removing of the fragment is
+ * completed and the fragment
+ * is now free again.
+ */
+ enum FragStatus {
+ FREE = 0, ///< Fragment record is currently not in use
+ FSACTIVE = 1, ///< Fragment is defined and usable for operations
+ DEFINED = 2, ///< Fragment is defined but not yet usable by
+ ///< operations
+ BLOCKED = 3, ///< LQH is waiting for all active operations to
+ ///< complete the current phase so that the
+ ///< local checkpoint can be started.
+ ACTIVE_CREATION = 4, ///< Fragment is defined and active but is under
+ ///< creation by the primary LQH.
+ CRASH_RECOVERING = 5, ///< Fragment is recovering after a crash by
+ ///< executing the fragment log and so forth.
+ ///< Will need further breakdown.
+ REMOVING = 6 ///< The fragment is currently removed.
+ ///< Operations are not allowed.
+ };
+ enum LogFlag {
+ STATE_TRUE = 0,
+ STATE_FALSE = 1
+ };
+ enum SrStatus {
+ SS_IDLE = 0,
+ SS_STARTED = 1,
+ SS_COMPLETED = 2
+ };
+ enum LcpFlag {
+ LCP_STATE_TRUE = 0,
+ LCP_STATE_FALSE = 1
+ };
+ /**
+ * Last GCI for executing the fragment log in this phase.
+ */
+ UintR execSrLastGci[4];
+ /**
+ * Start GCI for executing the fragment log in this phase.
+ */
+ UintR execSrStartGci[4];
+ /**
+ * Requesting user pointer for executing the fragment log in
+ * this phase
+ */
+ UintR execSrUserptr[4];
+ /**
+ * The LCP identifier of the LCP's.
+ * =0 means that the LCP number has not been stored.
+ * The LCP identifier is supplied by DIH when starting the LCP.
+ */
+ UintR lcpId[MAX_LCP_STORED];
+ UintR maxGciInLcp;
+ /**
+ * This variable contains the maximum global checkpoint
+ * identifier that exists in a certain local checkpoint.
+ * Maximum 4 local checkpoints is possible in this release.
+ */
+ UintR maxGciCompletedInLcp;
+ UintR srLastGci[4];
+ UintR srStartGci[4];
+ /**
+ * The fragment pointers in ACC
+ */
+ UintR accFragptr[2];
+ /**
+ * The EXEC_SR variables are used to keep track of which fragments
+ * that are interested in being executed as part of executing the
+ * fragment loop.
+ * It is initialised for every phase of executing the
+ * fragment log (the fragment log can be executed upto four times).
+ *
+ * Each execution is capable of executing the log records on four
+ * fragment replicas.
+ */
+ /**
+ * Requesting block reference for executing the fragment log
+ * in this phase.
+ */
+ BlockReference execSrBlockref[4];
+ /**
+ * This variable contains references to active scan and copy
+ * fragment operations on the fragment.
+ * A maximum of four concurrently active is allowed.
+ */
+ Uint16 fragScanRec[MAX_PARALLEL_SCANS_PER_FRAG + MAX_PARALLEL_INDEX_SCANS_PER_FRAG];
+ Uint16 srLqhLognode[4];
+ /**
+ * The fragment pointers in TUP and TUX
+ */
+ UintR tupFragptr[2];
+ UintR tuxFragptr[2];
+ /**
+ * This queue is where operations are put when blocked in ACC
+ * during start of a local chkp.
+ */
+ UintR accBlockedList;
+ /**
+ * This is the queue where all operations that are active on the
+ * fragment is put.
+ * This is used to deduct when the fragment do
+ * no longer contain any active operations.
+ * This is needed when starting a local checkpoint.
+ */
+ UintR activeList;
+ /**
+ * This variable keeps track of how many operations that are
+ * active that have skipped writing the log but not yet committed
+ * or aborted. This is used during start of fragment.
+ */
+ UintR activeTcCounter;
+ /**
+ * This status specifies whether this fragment is actively
+ * engaged in executing the fragment log.
+ */
+ ExecSrStatus execSrStatus;
+ /**
+ * The fragment id of this fragment.
+ */
+ UintR fragId;
+ /**
+ * Status of fragment
+ */
+ FragStatus fragStatus;
+ /**
+ * Indicates a local checkpoint is active and thus can generate
+ * UNDO log records.
+ */
+ UintR fragActiveStatus;
+ /**
+ * Reference to current LCP record.
+ * If no LCP is ongoing on the fragment then the value is RNIL.
+ * If LCP_REF /= RNIL then a local checkpoint is ongoing in the
+ * fragment.
+ * LCP_STATE in LCP_RECORD specifies the state of the
+ * local checkpoint.
+ */
+ UintR lcpRef;
+ /**
+ * This flag indicates whether logging is currently activated at
+ * the fragment.
+ * During a system restart it is temporarily shut off.
+ * Some fragments have it permanently shut off.
+ */
+ LogFlag logFlag;
+ UintR masterPtr;
+ /**
+ * This variable contains the maximum global checkpoint identifier
+ * which was completed when the local checkpoint was started.
+ */
+ /**
+ * Reference to the next fragment record in a free list of fragment
+ * records.
+ */
+ UintR nextFrag;
+ /**
+ * The newest GCI that has been committed on fragment
+ */
+ UintR newestGci;
+ SrStatus srStatus;
+ UintR srUserptr;
+ /**
+ * The starting global checkpoint of this fragment.
+ */
+ UintR startGci;
+ /**
+ * A reference to the table owning this fragment.
+ */
+ UintR tabRef;
+ /**
+ * This is the queue to put operations that have been blocked
+ * during start of a local chkp.
+ */
+ UintR firstWaitQueue;
+ UintR lastWaitQueue;
+ /**
+ * The block reference to ACC on the fragment makes it
+ * possible to have different ACC blocks for different
+ * fragments in the future.
+ */
+ BlockReference accBlockref;
+ /**
+ * Ordered index block.
+ */
+ BlockReference tuxBlockref;
+ /**
+ * The master block reference as sent in COPY_ACTIVEREQ.
+ */
+ BlockReference masterBlockref;
+ /**
+ * These variables are used during system restart to recall
+ * from which node to execute the fragment log and which GCI's
+ * this node should start and stop from. Also to remember who
+ * to send the response to when system restart is completed.
+ */
+ BlockReference srBlockref;
+ /**
+ * The block reference to TUP on the fragment makes it
+ * possible to have different TUP blocks for different
+ * fragments in the future.
+ */
+ BlockReference tupBlockref;
+ /**
+ * This state indicates if the fragment will participate in a
+ * checkpoint.
+ * Temporary tables with Fragrecord::logFlag permanently off
+ * will also have Fragrecord::lcpFlag off.
+ */
+ LcpFlag lcpFlag;
+ /**
+ * Used to ensure that updates started with old
+ * configuration do not arrive here after the copy fragment
+ * has started.
+ * If they are allowed to arrive after they
+ * could update a record that has already been replicated to
+ * the new node. This type of arrival should be extremely
+ * rare but we must anyway ensure that no harm is done.
+ */
+ Uint16 copyNode;
+ /**
+ * This variable ensures that only one copy fragment is
+ * active at a time on the fragment.
+ */
+ Uint8 copyFragState;
+ /**
+ * The number of fragment replicas that will execute the log
+ * records in this round of executing the fragment
+ * log. Maximum four is possible.
+ */
+ Uint8 execSrNoReplicas;
+ /**
+ * This variable contains what type of replica this fragment
+ * is. Two types are possible:
+ * - Primary/Backup replica = 0
+ * - Stand-by replica = 1
+ *
+ * It is not possible to distinguish between primary and
+ * backup on a fragment.
+ * This can only be done per transaction.
+ * DIH can change from primary to backup without informing
+ * the various replicas about this change.
+ */
+ Uint8 fragCopy;
+ /**
+ * This is the last fragment distribution key that we have
+ * heard of.
+ */
+ Uint8 fragDistributionKey;
+ /**
+ * Used to calculate which local fragment to use.
+ */
+ Uint8 hashCheckBit;
+ /**
+ * The identity of the next local checkpoint this fragment
+ * should perform.
+ */
+ Uint8 nextLcp;
+ /**
+ * The number of active scans currently in the fragment
+ * replica.
+ */
+ Uint8 noActiveScan;
+ /**
+ * How many local checkpoints does the fragment contain
+ */
+ Uint8 srChkpnr;
+ Uint8 srNoLognodes;
+ /**
+ * Table type.
+ */
+ Uint8 tableType;
+ /**
+ * For ordered index fragment, i-value of corresponding
+ * fragment in primary table.
+ */
+ UintR tableFragptr;
+ };
+ typedef Ptr<Fragrecord> FragrecordPtr;
+
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$ GLOBAL CHECKPOINT RECORD $$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /**
+ * This record describes a global checkpoint that is
+ * completed. It waits for all log records belonging to this
+ * global checkpoint to be saved on disk.
+ */
+ struct GcpRecord {
+ /**
+ * The file number within each log part where the log was
+ * located when gcp_savereq was received. The last record
+ * belonging to this global checkpoint is certainly before
+ * this place in the log. We could come even closer but it
+ * would cost performance and doesn't seem like a good
+ * idea. This is simple and it works.
+ */
+ Uint16 gcpFilePtr[4];
+ /**
+ * The page number within the file for each log part.
+ */
+ Uint16 gcpPageNo[4];
+ /**
+ * The word number within the last page that was written for
+ * each log part.
+ */
+ Uint16 gcpWordNo[4];
+ /**
+ * The identity of this global checkpoint.
+ */
+ UintR gcpId;
+ /**
+ * The state of this global checkpoint, one for each log part.
+ */
+ Uint8 gcpLogPartState[4];
+ /**
+ * The sync state of this global checkpoint, one for each
+ * log part.
+ */
+ Uint8 gcpSyncReady[4];
+ /**
+ * User pointer of the sender of gcp_savereq (= master DIH).
+ */
+ UintR gcpUserptr;
+ /**
+ * Block reference of the sender of gcp_savereq
+ * (= master DIH).
+ */
+ BlockReference gcpBlockref;
+ }; // Size 44 bytes
+ typedef Ptr<GcpRecord> GcpRecordPtr;
+
+ struct HostRecord {
+ bool inPackedList;
+ UintR noOfPackedWordsLqh;
+ UintR packedWordsLqh[30];
+ UintR noOfPackedWordsTc;
+ UintR packedWordsTc[29];
+ BlockReference hostLqhBlockRef;
+ BlockReference hostTcBlockRef;
+ };// Size 128 bytes
+ typedef Ptr<HostRecord> HostRecordPtr;
+
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$ LOCAL CHECKPOINT RECORD $$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /**
+ * This record contains the information about a local
+ * checkpoint that is ongoing. This record is also used as a
+ * system restart record.
+ */
+ struct LcpRecord {
+ LcpRecord() { m_EMPTY_LCP_REQ.clear(); }
+
+ enum LcpState {
+ LCP_IDLE = 0,
+ LCP_STARTED = 1,
+ LCP_COMPLETED = 2,
+ LCP_WAIT_FRAGID = 3,
+ LCP_WAIT_TUP_PREPLCP = 4,
+ LCP_WAIT_HOLDOPS = 5,
+ LCP_WAIT_ACTIVE_FINISH = 6,
+ LCP_START_CHKP = 7,
+ LCP_BLOCKED_COMP = 8,
+ LCP_SR_WAIT_FRAGID = 9,
+ LCP_SR_STARTED = 10,
+ LCP_SR_COMPLETED = 11
+ };
+ Uint32 firstLcpLocAcc;
+ Uint32 firstLcpLocTup;
+ Uint32 lcpAccptr;
+
+ LcpState lcpState;
+ bool lastFragmentFlag;
+
+ struct FragOrd {
+ Uint32 fragPtrI;
+ LcpFragOrd lcpFragOrd;
+ };
+ FragOrd currentFragment;
+
+ bool lcpQueued;
+ FragOrd queuedFragment;
+
+ bool reportEmpty;
+ NdbNodeBitmask m_EMPTY_LCP_REQ;
+ }; // Size 76 bytes
+ typedef Ptr<LcpRecord> LcpRecordPtr;
+
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$ LOCAL CHECKPOINT SUPPORT RECORD $$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /**
+ * This record contains the information about an outstanding
+ * request to TUP or ACC. Used for both local checkpoints and
+ * system restart.
+ */
+ struct LcpLocRecord {
+ enum LcpLocstate {
+ IDLE = 0,
+ WAIT_TUP_PREPLCP = 1,
+ WAIT_LCPHOLDOP = 2,
+ HOLDOP_READY = 3,
+ ACC_WAIT_STARTED = 4,
+ ACC_STARTED = 5,
+ ACC_COMPLETED = 6,
+ TUP_WAIT_STARTED = 7,
+ TUP_STARTED = 8,
+ TUP_COMPLETED = 9,
+ SR_ACC_STARTED = 10,
+ SR_TUP_STARTED = 11,
+ SR_ACC_COMPLETED = 12,
+ SR_TUP_COMPLETED = 13
+ };
+ enum WaitingBlock {
+ ACC = 0,
+ TUP = 1,
+ NONE = 2
+ };
+
+ LcpLocstate lcpLocstate;
+ UintR locFragid;
+ UintR masterLcpRec;
+ UintR nextLcpLoc;
+ UintR tupRef;
+ WaitingBlock waitingBlock;
+ Uint32 accContCounter;
+ }; // 28 bytes
+ typedef Ptr<LcpLocRecord> LcpLocRecordPtr;
+
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* */
+ /* THE RECORDS THAT START BY LOG_ ARE A PART OF THE LOG MANAGER. */
+ /* THESE RECORDS ARE USED TO HANDLE THE FRAGMENT LOG. */
+ /* */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$ LOG RECORD $$$$$$$ */
+ /* */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* THIS RECORD IS ALIGNED TO BE 256 BYTES. */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /**
+ * This record describes the current state of a log.
+ * A log consists of a number of log files.
+ * These log files are described by the log file record.
+ *
+ * There will be 4 sets of log files.
+ * Different tables will use different log files dependent
+ * on the table id.
+ * This ensures that more than one outstanding request can
+ * be sent to the file system.
+ * The log file to use is found by performing a very simple hash
+ * function.
+ */
+ struct LogPartRecord {
+ enum LogPartState {
+ IDLE = 0, ///< Nothing happens at the moment
+ ACTIVE = 1, ///< An operation is active logging
+ SR_FIRST_PHASE = 2, ///< Finding the end of the log and
+ ///< the information about global
+ ///< checkpoints in the log is ongoing.
+ SR_FIRST_PHASE_COMPLETED = 3, ///< First phase completed
+ SR_THIRD_PHASE_STARTED = 4, ///< Executing fragment log is in 3rd ph
+ SR_THIRD_PHASE_COMPLETED = 5,
+ SR_FOURTH_PHASE_STARTED = 6, ///< Finding the log tail and head
+ ///< is the fourth phase.
+ SR_FOURTH_PHASE_COMPLETED = 7,
+ FILE_CHANGE_PROBLEM = 8, ///< For some reason the write to
+ ///< page zero in file zero have not
+ ///< finished after 15 mbyte of
+ ///< log data have been written
+ TAIL_PROBLEM = 9 ///< Only 1 mbyte of log left.
+ ///< No operations allowed to enter the
+ ///< log. Only special log records
+ ///< are allowed
+ };
+ enum WaitWriteGciLog {
+ WWGL_TRUE = 0,
+ WWGL_FALSE = 1
+ };
+ enum LogExecState {
+ LES_IDLE = 0,
+ LES_SEARCH_STOP = 1,
+ LES_SEARCH_START = 2,
+ LES_EXEC_LOG = 3,
+ LES_EXEC_LOG_NEW_MBYTE = 4,
+ LES_EXEC_LOG_NEW_FILE = 5,
+ LES_EXEC_LOGREC_FROM_FILE = 6,
+ LES_EXEC_LOG_COMPLETED = 7,
+ LES_WAIT_READ_EXEC_SR_NEW_MBYTE = 8,
+ LES_WAIT_READ_EXEC_SR = 9,
+ LES_EXEC_LOG_INVALIDATE = 10
+ };
+
+ /**
+ * Is a CONTINUEB(ZLOG_LQHKEYREQ) signal sent and
+ * outstanding. We do not want several instances of this
+ * signal out in the air since that would create multiple
+ * writers of the list.
+ */
+ UintR LogLqhKeyReqSent;
+ /**
+ * Contains the current log file where log records are
+ * written. During system restart it is used to indicate the
+ * last log file.
+ */
+ UintR currentLogfile;
+ /**
+ * The log file used to execute log records from far behind.
+ */
+ UintR execSrExecLogFile;
+ /**
+ * The currently executing prepare record starts in this log
+ * page. This variable is used to enable that a log record is
+ * executed multiple times in execution of the log.
+ */
+ UintR execSrLogPage;
+ /**
+ * This variable keeps track of the lfo record where the
+ * pages that were read from disk when an operations log
+ * record were not found in the main memory buffer for log
+ * pages.
+ */
+ UintR execSrLfoRec;
+ /**
+ * The starting page number when reading log from far behind.
+ */
+ UintR execSrStartPageNo;
+ /**
+ * The last page number when reading log from far behind.
+ */
+ UintR execSrStopPageNo;
+ /**
+ * Contains a reference to the first log file, file number 0.
+ */
+ UintR firstLogfile;
+ /**
+ * The head of the operations queued for logging.
+ */
+ UintR firstLogQueue;
+ /**
+ * This variable contains the oldest operation in this log
+ * part which have not been committed yet.
+ */
+ UintR firstLogTcrec;
+ /**
+ * The first reference to a set of 8 pages. These are used
+ * during execution of the log to keep track of which pages
+ * are in memory and which are not.
+ */
+ UintR firstPageRef;
+ /**
+ * This variable contains the global checkpoint record
+ * waiting for disk writes to complete.
+ */
+ UintR gcprec;
+ /**
+ * The last reference to a set of 8 pages. These are used
+ * during execution of the log to keep track of which pages
+ * are in memory and which are not.
+ */
+ UintR lastPageRef;
+ /**
+ * The tail of the operations queued for logging.
+ */
+ UintR lastLogQueue;
+ /**
+ * This variable contains the newest operation in this log
+ * part which have not been committed yet.
+ */
+ UintR lastLogTcrec;
+ /**
+ * This variable indicates which was the last mbyte that was
+ * written before the system crashed. Discovered during
+ * system restart.
+ */
+ UintR lastLogfile;
+ /**
+ * This variable is used to keep track of the state during
+ * the third phase of the system restart, i.e. when
+ * LogPartRecord::logPartState ==
+ * LogPartRecord::SR_THIRD_PHASE_STARTED.
+ */
+ LogExecState logExecState;
+ /**
+ * This variable contains the lap number of this log part.
+ */
+ UintR logLap;
+ /**
+ * This variable contains the place to stop executing the log
+ * in this phase.
+ */
+ UintR logLastGci;
+ /**
+ * This variable contains the place to start executing the
+ * log in this phase.
+ */
+ UintR logStartGci;
+ /**
+ * The latest GCI completed in this log part.
+ */
+ UintR logPartNewestCompletedGCI;
+ /**
+ * The current state of this log part.
+ */
+ LogPartState logPartState;
+ /**
+ * A timer that is set every time a log page is sent to disk.
+ * Ensures that log pages are not kept in main memory for
+ * more than a certain time.
+ */
+ UintR logPartTimer;
+ /**
+ * The current timer which is set by the periodic signal
+ * received by LQH
+ */
+ UintR logTimer;
+ /**
+ * Contains the number of the log tail file and the mbyte
+ * reference within that file. This information ensures that
+ * the tail is not overwritten when writing new log records.
+ */
+ UintR logTailFileNo;
+ /**
+ * The TcConnectionrec used during execution of this log part.
+ */
+ UintR logTcConrec;
+ /**
+ * The number of pages that currently resides in the main
+ * memory buffer. It does not refer pages that are currently
+ * read from the log files. Only to pages already read
+ * from the log file.
+ */
+ UintR mmBufferSize;
+ /**
+ * Contains the current number of log files in this log part.
+ */
+ UintR noLogFiles;
+ /**
+ * This variable is used only during execution of a log
+ * record. It keeps track of in which page record a log
+ * record was started. It is used then to deduce which
+ * pages that are dirty after that the log records on the
+ * page have been executed.
+ *
+ * It is also used to find out where to write the invalidate
+ * command when that is needed.
+ */
+ UintR prevLogpage;
+ /**
+ * The number of files remaining to gather GCI information
+ * for during system restart. Only used if number of files
+ * is larger than 60.
+ */
+ UintR srRemainingFiles;
+ /**
+ * The log file where to start executing the log during
+ * system restart.
+ */
+ UintR startLogfile;
+ /**
+ * The last log file in which to execute the log during system
+ * restart.
+ */
+ UintR stopLogfile;
+ /**
+ * This variable keeps track of when we want to write a complete
+ * gci log record but have been blocked by an ongoing log operation.
+ */
+ WaitWriteGciLog waitWriteGciLog;
+ /**
+ * The currently executing prepare record starts in this index
+ * in the log page.
+ */
+ Uint16 execSrLogPageIndex;
+ /**
+ * Which of the four exec_sr's in the fragment is currently executing
+ */
+ Uint16 execSrExecuteIndex;
+ /**
+ * The number of pages executed in the current mbyte.
+ */
+ Uint16 execSrPagesExecuted;
+ /**
+ * The number of pages read from disk that have arrived and are
+ * currently awaiting execution of the log.
+ */
+ Uint16 execSrPagesRead;
+ /**
+ * The number of pages read from disk and currently not arrived
+ * to the block.
+ */
+ Uint16 execSrPagesReading;
+ /**
+ * This variable refers to the new header file where we will
+ * start writing the log after a system restart have been completed.
+ */
+ Uint16 headFileNo;
+ /**
+ * This variable refers to the page number within the header file.
+ */
+ Uint16 headPageNo;
+ /**
+ * This variable refers to the index within the new header
+ * page.
+ */
+ Uint16 headPageIndex;
+ /**
+ * This variables indicates which was the last mbyte in the last
+ * logfile before a system crash. Discovered during system restart.
+ */
+ Uint16 lastMbyte;
+ /**
+ * This variable is used only during execution of a log
+ * record. It keeps track of in which file page a log
+ * record was started. It is used if it is needed to write a
+ * dirty page to disk during log execution (this happens when
+ * commit records are invalidated).
+ */
+ Uint16 prevFilepage;
+ /**
+ * This is used to save where we were in the execution of log
+ * records when we find a commit record that needs to be
+ * executed.
+ *
+ * This variable is also used to remember the index where the
+ * log type was in the log record. It is only used in this
+ * role when finding a commit record that needs to be
+ * invalidated.
+ */
+ Uint16 savePageIndex;
+ Uint8 logTailMbyte;
+ /**
+ * The mbyte within the starting log file where to start
+ * executing the log.
+ */
+ Uint8 startMbyte;
+ /**
+ * The last mbyte in which to execute the log during system
+ * restart.
+ */
+ Uint8 stopMbyte;
+ /**
+ * This variable refers to the file where invalidation is
+ * occuring during system/node restart.
+ */
+ Uint16 invalidateFileNo;
+ /**
+ * This variable refers to the page where invalidation is
+ * occuring during system/node restart.
+ */
+ Uint16 invalidatePageNo;
+ }; // Size 164 Bytes
+ typedef Ptr<LogPartRecord> LogPartRecordPtr;
+
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$ LOG FILE RECORD $$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* THIS RECORD IS ALIGNED TO BE 288 (256 + 32) BYTES. */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /**
+ * This record contains information about a log file.
+ * A log file contains log records from several tables and
+ * fragments of a table. LQH can contain more than
+ * one log file to ensure faster log processing.
+ *
+ * The number of pages to write to disk at a time is
+ * configurable.
+ */
+ struct LogFileRecord {
+ enum FileChangeState {
+ NOT_ONGOING = 0,
+ BOTH_WRITES_ONGOING = 1,
+ LAST_WRITE_ONGOING = 2,
+ FIRST_WRITE_ONGOING = 3,
+ WRITE_PAGE_ZERO_ONGOING = 4
+ };
+ enum LogFileStatus {
+ LFS_IDLE = 0, ///< Log file record not in use
+ CLOSED = 1, ///< Log file closed
+ OPENING_INIT = 2,
+ OPEN_SR_FRONTPAGE = 3, ///< Log file opened as part of system
+ ///< restart. Open file 0 to find
+ ///< the front page of the log part.
+ OPEN_SR_LAST_FILE = 4, ///< Open last log file that was written
+ ///< before the system restart.
+ OPEN_SR_NEXT_FILE = 5, ///< Open a log file which is 16 files
+ ///< backwards to find the next
+ ///< information about GCPs.
+ OPEN_EXEC_SR_START = 6, ///< Log file opened as part of
+ ///< executing
+ ///< log during system restart.
+ OPEN_EXEC_SR_NEW_MBYTE = 7,
+ OPEN_SR_FOURTH_PHASE = 8,
+ OPEN_SR_FOURTH_NEXT = 9,
+ OPEN_SR_FOURTH_ZERO = 10,
+ OPENING_WRITE_LOG = 11, ///< Log file opened as part of writing
+ ///< log during normal operation.
+ OPEN_EXEC_LOG = 12,
+ CLOSING_INIT = 13,
+ CLOSING_SR = 14, ///< Log file closed as part of system
+ ///< restart. Currently trying to
+ ///< find where to start executing the
+ ///< log
+ CLOSING_EXEC_SR = 15, ///< Log file closed as part of
+ ///< executing log during system restart
+ CLOSING_EXEC_SR_COMPLETED = 16,
+ CLOSING_WRITE_LOG = 17, ///< Log file closed as part of writing
+ ///< log during normal operation.
+ CLOSING_EXEC_LOG = 18,
+ OPEN_INIT = 19,
+ OPEN = 20, ///< Log file open
+ OPEN_SR_INVALIDATE_PAGES = 21,
+ CLOSE_SR_INVALIDATE_PAGES = 22
+ };
+
+ /**
+ * When a new mbyte is started in the log we have to find out
+ * how far back in the log we still have prepared operations
+ * which have been neither committed or aborted. This variable
+ * keeps track of this value for each of the mbytes in this
+ * log file. This is used in writing down these values in the
+ * header of each log file. That information is used during
+ * system restart to find the tail of the log.
+ */
+ UintR logLastPrepRef[16];
+ /**
+ * The max global checkpoint completed before the mbyte in the
+ * log file was started. One variable per mbyte.
+ */
+ UintR logMaxGciCompleted[16];
+ /**
+ * The max global checkpoint started before the mbyte in the log
+ * file was started. One variable per mbyte.
+ */
+ UintR logMaxGciStarted[16];
+ /**
+ * This variable contains the file name as needed by the file
+ * system when opening the file.
+ */
+ UintR fileName[4];
+ /**
+ * This variable has a reference to the log page which is
+ * currently in use by the log.
+ */
+ UintR currentLogpage;
+ /**
+ * The number of the current mbyte in the log file.
+ */
+ UintR currentMbyte;
+ /**
+ * This variable is used when changing files. It is to find
+ * out when both the last write in the previous file and the
+ * first write in this file has been completed. After these
+ * writes have completed the variable keeps track of when the
+ * write to page zero in file zero is completed.
+ */
+ FileChangeState fileChangeState;
+ /**
+ * The number of the file within this log part.
+ */
+ UintR fileNo;
+ /**
+ * This variable shows where to read/write the next pages into
+ * the log. Used when writing the log during normal operation
+ * and when reading the log during system restart. It
+ * specifies the page position where each page is 8 kbyte.
+ */
+ UintR filePosition;
+ /**
+ * This contains the file pointer needed by the file system
+ * when reading/writing/closing and synching.
+ */
+ UintR fileRef;
+ /**
+ * The head of the pages waiting for shipment to disk.
+ * They are filled with log info.
+ */
+ UintR firstFilledPage;
+ /**
+ * A list of active read/write operations on the log file.
+ * Operations are always put in last and the first should
+ * always complete first.
+ */
+ UintR firstLfo;
+ UintR lastLfo;
+ /**
+ * The tail of the pages waiting for shipment to disk.
+ * They are filled with log info.
+ */
+ UintR lastFilledPage;
+ /**
+ * This variable keeps track of the last written page in the
+ * file while writing page zero in file zero when changing log
+ * file.
+ */
+ UintR lastPageWritten;
+ /**
+ * This variable keeps track of the last written word in the
+ * last page written in the file while writing page zero in
+ * file zero when changing log file.
+ */
+ UintR lastWordWritten;
+ /**
+ * This variable contains the last word written in the last page.
+ */
+ UintR logFilePagesToDiskWithoutSynch;
+ /**
+ * This variable keeps track of the number of pages written since
+ * last synch on this log file.
+ */
+ LogFileStatus logFileStatus;
+ /**
+ * A reference to page zero in this file.
+ * This page is written before the file is closed.
+ */
+ UintR logPageZero;
+ /**
+ * This variable contains a reference to the record describing
+ * this log part. One of four records (0,1,2 or 3).
+ */
+ UintR logPartRec;
+ /**
+ * Next free log file record or next log file in this log.
+ */
+ UintR nextLogFile;
+ /**
+ * The previous log file.
+ */
+ UintR prevLogFile;
+ /**
+ * The number of remaining words in this mbyte of the log file.
+ */
+ UintR remainingWordsInMbyte;
+ /**
+ * The current file page within the current log file. This is
+ * a reference within the file and not a reference to a log
+ * page record. It is used to deduce where log records are
+ * written. Particularly completed gcp records and prepare log
+ * records.
+ */
+ Uint16 currentFilepage;
+ /**
+ * The number of pages in the list referenced by
+ * LOG_PAGE_BUFFER.
+ */
+ Uint16 noLogpagesInBuffer;
+ }; // Size 288 bytes
+ typedef Ptr<LogFileRecord> LogFileRecordPtr;
+
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$ LOG OPERATION RECORD $$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /**
+ * This record contains a currently active file operation
+ * that has started by the log module.
+ */
+ struct LogFileOperationRecord {
+ enum LfoState {
+ IDLE = 0, ///< Operation is not used at the moment
+ INIT_WRITE_AT_END = 1, ///< Write in file so that it grows to
+ ///< 16 Mbyte
+ INIT_FIRST_PAGE = 2, ///< Initialise the first page in a file
+ WRITE_GCI_ZERO = 3,
+ WRITE_INIT_MBYTE = 4,
+ WRITE_DIRTY = 5,
+ READ_SR_FRONTPAGE = 6, ///< Read page zero in file zero during
+ ///< system restart
+ READ_SR_LAST_FILE = 7, ///< Read page zero in last file open
+ ///< before system crash
+ READ_SR_NEXT_FILE = 8, ///< Read 60 files backwards to find
+ ///< further information GCPs in page
+ ///< zero
+ READ_SR_LAST_MBYTE = 9,
+ READ_EXEC_SR = 10,
+ READ_EXEC_LOG = 11,
+ READ_SR_FOURTH_PHASE = 12,
+ READ_SR_FOURTH_ZERO = 13,
+ FIRST_PAGE_WRITE_IN_LOGFILE = 14,
+ LAST_WRITE_IN_FILE = 15,
+ WRITE_PAGE_ZERO = 16,
+ ACTIVE_WRITE_LOG = 17, ///< A write operation during
+ ///< writing of log
+ READ_SR_INVALIDATE_PAGES = 18,
+ WRITE_SR_INVALIDATE_PAGES = 19
+ };
+ /**
+ * We have to remember the log pages read.
+ * Otherwise we cannot build the linked list after the pages have
+ * arrived to main memory.
+ */
+ UintR logPageArray[16];
+ /**
+ * A list of the pages that are part of this active operation.
+ */
+ UintR firstLfoPage;
+ /**
+ * A timer to ensure that records are not lost.
+ */
+ UintR lfoTimer;
+ /**
+ * The word number of the last written word in the last during
+ * a file write.
+ */
+ UintR lfoWordWritten;
+ /**
+ * This variable contains the state of the log file operation.
+ */
+ LfoState lfoState;
+ /**
+ * The log file that the file operation affects.
+ */
+ UintR logFileRec;
+ /**
+ * The log file operations on a file are kept in a linked list.
+ */
+ UintR nextLfo;
+ /**
+ * The page number of the first read/written page during a file
+ * read/write.
+ */
+ Uint16 lfoPageNo;
+ /**
+ * The number of pages written or read during an operation to
+ * the log file.
+ */
+ Uint16 noPagesRw;
+ }; // 92 bytes
+ typedef Ptr<LogFileOperationRecord> LogFileOperationRecordPtr;
+
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /* $$$$$$$ LOG PAGE RECORD $$$$$$$ */
+ /* $$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$ */
+ /**
+ * These are the 8 k pages used to store log records before storing
+ * them in the file system.
+ * Since 64 kbyte is sent to disk at a time it is necessary to have
+ * at least 4*64 kbytes of log pages.
+ * To handle multiple outstanding requests we need some additional pages.
+ * Thus we allocate 1 mbyte to ensure that we do not get problems with
+ * insufficient number of pages.
+ */
+ struct LogPageRecord {
+ /**
+ * This variable contains the pages that are sent to disk.
+ *
+ * All pages contain a header of 12 words:
+ * - WORD 0: CHECKSUM Calculated before storing on disk and
+ * checked when read from disk.
+ * - WORD 1: LAP How many wraparounds have the log
+ * experienced since initial start of the
+ * system.
+ * - WORD 2: MAX_GCI_COMPLETED Which is the maximum gci which have
+ * completed before this page. This
+ * gci will not be found in this
+ * page and hereafter in the log.
+ * - WORD 3: MAX_GCI_STARTED The maximum gci which have started
+ * before this page.
+ * - WORD 4: NEXT_PAGE Pointer to the next page.
+ * Only used in main memory
+ * - WORD 5: PREVIOUS_PAGE Pointer to the previous page.
+ * Currently not used.
+ * - WORD 6: VERSION NDB version that wrote the page.
+ * - WORD 7: NO_LOG_FILES Number of log files in this log part.
+ * - WORD 8: CURRENT PAGE INDEX This keeps track of where we are in the
+ * page.
+ * This is only used when pages is in
+ * memory.
+ * - WORD 9: OLD PREPARE FILE NO This keeps track of the oldest prepare
+ * operation still alive (not committed
+ * or aborted) when this mbyte started.
+ * - WORD 10: OLD PREPARE PAGE REF File page reference within this file
+ * number.
+ * Page no + Page index.
+ * If no prepare was alive then these
+ * values points this mbyte.
+ * - WORD 11: DIRTY FLAG = 0 means not dirty and
+ * = 1 means the page is dirty.
+ * Is used when executing log when
+ * a need to write invalid commit
+ * records arise.
+ *
+ * The remaining 2036 words are used for log information, i.e.
+ * log records.
+ *
+ * A log record on this page has the following layout:
+ * - WORD 0: LOG RECORD TYPE
+ * The following types are supported:
+ * - PREPARE OPERATION An operation not yet committed.
+ * - NEW PREPARE OPERATION A prepared operation already
+ * logged is inserted
+ * into the log again so that the
+ * log tail can be advanced.
+ * This can happen when a transaction is
+ * committed for a long time.
+ * - ABORT TRANSACTION A previously prepared transaction
+ * was aborted.
+ * - COMMIT TRANSACTION A previously prepared transaction
+ * was committed.
+ * - INVALID COMMIT A previous commit record was
+ * invalidated by a
+ * subsequent system restart.
+ * A log record must be invalidated
+ * in a system restart if it belongs
+ * to a global checkpoint id which
+ * is not included in the system
+ * restart.
+ * Otherwise it will be included in
+ * a subsequent system restart since
+ * it will then most likely belong
+ * to a global checkpoint id which
+ * is part of that system
+ * restart.
+ * This is not a correct behaviour
+ * since this operation is lost in a
+ * system restart and should not
+ * reappear at a later system
+ * restart.
+ * - COMPLETED GCI A GCI has now been completed.
+ * - FRAGMENT SPLIT A fragment has been split
+ * (not implemented yet)
+ * - FILE DESCRIPTOR This is always the first log record
+ * in a file.
+ * It is always placed on page 0 after
+ * the header.
+ * It is written when the file is
+ * opened and when the file is closed.
+ * - NEXT LOG RECORD This log record only records where
+ * the next log record starts.
+ * - NEXT MBYTE RECORD This log record specifies that there
+ * are no more log records in this mbyte.
+ *
+ *
+ * A FILE DESCRIPTOR log record continues as follows:
+ * - WORD 1: NO_LOG_DESCRIPTORS This defines the number of
+ * descriptors of log files that
+ * will follow hereafter (max 32).
+ * the log descriptor will describe
+ * information about
+ * max_gci_completed,
+ * max_gci_started and log_lap at
+ * every 1 mbyte of the log file
+ * since a log file is 16 mbyte
+ * always, i need 16 entries in the
+ * array with max_gci_completed,
+ * max_gci_started and log_lap. thus
+ * 32 entries per log file
+ * descriptor (max 32*48 = 1536,
+ * always fits in page 0).
+ * - WORD 2: LAST LOG FILE The number of the log file currently
+ * open. This is only valid in file 0.
+ * - WORD 3 - WORD 18: MAX_GCI_COMPLETED for every 1 mbyte
+ * in this log file.
+ * - WORD 19 - WORD 34: MAX_GCI_STARTED for every 1 mbyte
+ * in this log file.
+ *
+ * Then it continues for NO_LOG_DESCRIPTORS until all subsequent
+ * log files (max 32) have been properly described.
+ *
+ *
+ * A PREPARE OPERATION log record continues as follows:
+ * - WORD 1: LOG RECORD SIZE
+ * - WORD 2: HASH VALUE
+ * - WORD 3: SCHEMA VERSION
+ * - WORD 4: OPERATION TYPE
+ * = 0 READ,
+ * = 1 UPDATE,
+ * = 2 INSERT,
+ * = 3 DELETE
+ * - WORD 5: NUMBER OF WORDS IN ATTRINFO PART
+ * - WORD 6: KEY LENGTH IN WORDS
+ * - WORD 7 - (WORD 7 + KEY_LENGTH - 1) The tuple key
+ * - (WORD 7 + KEY_LENGTH) -
+ * (WORD 7 + KEY_LENGTH + ATTRINFO_LENGTH - 1) The attrinfo
+ *
+ * A log record can be spread in several pages in some cases.
+ * The next log record always starts immediately after this log record.
+ * A log record does however never traverse a 1 mbyte boundary.
+ * This is used to ensure that we can always come back if something
+ * strange occurs in the log file.
+ * To ensure this we also have log records which only records
+ * the next log record.
+ *
+ *
+ * A COMMIT TRANSACTION log record continues as follows:
+ * - WORD 1: TRANSACTION ID PART 1
+ * - WORD 2: TRANSACTION ID PART 2
+ * - WORD 3: FRAGMENT ID OF THE OPERATION
+ * - WORD 4: TABLE ID OF THE OPERATION
+ * - WORD 5: THE FILE NUMBER OF THE PREPARE RECORD
+ * - WORD 6: THE STARTING PAGE NUMBER OF THE PREPARE RECORD
+ * - WORD 7: THE STARTING PAGE INDEX OF THE PREPARE RECORD
+ * - WORD 8: THE STOP PAGE NUMBER OF THE PREPARE RECORD
+ * - WORD 9: GLOBAL CHECKPOINT OF THE TRANSACTION
+ *
+ *
+ * An ABORT TRANSACTION log record continues as follows:
+ * - WORD 1: TRANSACTION ID PART 1
+ * - WORD 2: TRANSACTION ID PART 2
+ *
+ *
+ * A COMPLETED CGI log record continues as follows:
+ * - WORD 1: THE COMPLETED GCI
+ *
+ *
+ * A NEXT LOG RECORD log record continues as follows:
+ * - There is no more information needed.
+ * The next log record will always refer to the start of the next page.
+ *
+ * A NEXT MBYTE RECORD log record continues as follows:
+ * - There is no more information needed.
+ * The next mbyte will always refer to the start of the next mbyte.
+ */
+#ifdef NDB_OSE
+ UintR logPageWord[2048]; // Size 8 kbytes
+#else
+ UintR logPageWord[8192]; // Size 32 kbytes
+#endif
+ };
+ typedef Ptr<LogPageRecord> LogPageRecordPtr;
+
+ struct PageRefRecord {
+ UintR pageRef[8];
+ UintR prNext;
+ UintR prPrev;
+ Uint16 prFileNo;
+ Uint16 prPageNo;
+ }; // size 44 bytes
+ typedef Ptr<PageRefRecord> PageRefRecordPtr;
+
+ struct ScanRecord {
+ enum ScanState {
+ SCAN_FREE = 0,
+ WAIT_STORED_PROC_COPY = 1,
+ WAIT_STORED_PROC_SCAN = 2,
+ WAIT_NEXT_SCAN_COPY = 3,
+ WAIT_NEXT_SCAN = 4,
+ WAIT_DELETE_STORED_PROC_ID_SCAN = 5,
+ WAIT_DELETE_STORED_PROC_ID_COPY = 6,
+ WAIT_ACC_COPY = 7,
+ WAIT_ACC_SCAN = 8,
+ WAIT_SCAN_KEYINFO = 9,
+ WAIT_SCAN_NEXTREQ = 10,
+ WAIT_COPY_KEYINFO = 11,
+ WAIT_CLOSE_SCAN = 12,
+ WAIT_CLOSE_COPY = 13,
+ WAIT_RELEASE_LOCK = 14,
+ WAIT_TUPKEY_COPY = 15,
+ WAIT_LQHKEY_COPY = 16
+ };
+ enum ScanType {
+ ST_IDLE = 0,
+ SCAN = 1,
+ COPY = 2
+ };
+ UintR scanAccOpPtr[MAX_PARALLEL_OP_PER_SCAN];
+ UintR scanApiOpPtr[MAX_PARALLEL_OP_PER_SCAN];
+ UintR scanOpLength[MAX_PARALLEL_OP_PER_SCAN];
+ UintR scanLocalref[2];
+ UintR copyPtr;
+ UintR nextScanrec;
+ UintR scanAccPtr;
+ UintR scanAiLength;
+ UintR scanCompletedOperations;
+ UintR scanConcurrentOperations;
+ UintR scanErrorCounter;
+ UintR scanLocalFragid;
+ UintR scanSchemaVersion;
+ UintR scanSearchCondFalseCount;
+ UintR scanStoredProcId;
+ ScanState scanState;
+ UintR scanTcrec;
+ ScanType scanType;
+ BlockReference scanApiBlockref;
+ NodeId scanNodeId;
+ Uint8 scanCompletedStatus;
+ Uint8 scanFlag;
+ Uint8 scanLockHold;
+ Uint8 scanLockMode;
+ Uint8 readCommitted;
+ Uint8 rangeScan;
+ Uint8 scanNumber;
+ Uint8 scanReleaseCounter;
+ Uint8 scanTcWaiting;
+ Uint8 scanKeyinfoFlag;
+ }; // Size 272 bytes
+ typedef Ptr<ScanRecord> ScanRecordPtr;
+
+ struct Tablerec {
+ enum TableStatus {
+ TABLE_DEFINED = 0,
+ NOT_DEFINED = 1,
+ ADD_TABLE_ONGOING = 2,
+ PREP_DROP_TABLE_ONGOING = 3,
+ PREP_DROP_TABLE_DONE = 4
+ };
+
+ UintR fragrec[NO_OF_FRAG_PER_NODE];
+ Uint16 fragid[NO_OF_FRAG_PER_NODE];
+ /**
+ * Status of the table
+ */
+ TableStatus tableStatus;
+ /**
+ * Table type and target table of index.
+ */
+ Uint16 tableType;
+ Uint16 primaryTableId;
+ Uint32 schemaVersion;
+
+ Uint32 usageCount;
+ NdbNodeBitmask waitingTC;
+ NdbNodeBitmask waitingDIH;
+ }; // Size 100 bytes
+ typedef Ptr<Tablerec> TablerecPtr;
+
+ struct TcConnectionrec {
+ enum ListState {
+ NOT_IN_LIST = 0,
+ IN_ACTIVE_LIST = 1,
+ ACC_BLOCK_LIST = 2,
+ WAIT_QUEUE_LIST = 3
+ };
+ enum LogWriteState {
+ NOT_STARTED = 0,
+ NOT_WRITTEN = 1,
+ NOT_WRITTEN_WAIT = 2,
+ WRITTEN = 3
+ };
+ enum AbortState {
+ ABORT_IDLE = 0,
+ ABORT_ACTIVE = 1,
+ NEW_FROM_TC = 2,
+ REQ_FROM_TC = 3,
+ ABORT_FROM_TC = 4,
+ ABORT_FROM_LQH = 5
+ };
+ enum TransactionState {
+ IDLE = 0,
+
+ /* -------------------------------------------------------------------- */
+ // Transaction in progress states
+ /* -------------------------------------------------------------------- */
+ WAIT_ACC = 1,
+ WAIT_TUPKEYINFO = 2,
+ WAIT_ATTR = 3,
+ WAIT_TUP = 4,
+ STOPPED = 5,
+ LOG_QUEUED = 6,
+ PREPARED = 7,
+ LOG_COMMIT_WRITTEN_WAIT_SIGNAL = 8,
+ LOG_COMMIT_QUEUED_WAIT_SIGNAL = 9,
+
+ /* -------------------------------------------------------------------- */
+ // Commit in progress states
+ /* -------------------------------------------------------------------- */
+ COMMIT_STOPPED = 10,
+ LOG_COMMIT_QUEUED = 11,
+ COMMIT_QUEUED = 12,
+ COMMITTED = 13,
+
+ /* -------------------------------------------------------------------- */
+ // Abort in progress states
+ /* -------------------------------------------------------------------- */
+ WAIT_ACC_ABORT = 14,
+ ABORT_QUEUED = 15,
+ ABORT_STOPPED = 16,
+ WAIT_AI_AFTER_ABORT = 17,
+ LOG_ABORT_QUEUED = 18,
+ WAIT_TUP_TO_ABORT = 19,
+
+ /* -------------------------------------------------------------------- */
+ // Scan in progress states
+ /* -------------------------------------------------------------------- */
+ WAIT_SCAN_AI = 20,
+ SCAN_STATE_USED = 21,
+ SCAN_FIRST_STOPPED = 22,
+ SCAN_CHECK_STOPPED = 23,
+ SCAN_STOPPED = 24,
+ SCAN_RELEASE_STOPPED = 25,
+ SCAN_CLOSE_STOPPED = 26,
+ COPY_CLOSE_STOPPED = 27,
+ COPY_FIRST_STOPPED = 28,
+ COPY_STOPPED = 29,
+ SCAN_TUPKEY = 30,
+ COPY_TUPKEY = 31,
+
+ TC_NOT_CONNECTED = 32,
+ PREPARED_RECEIVED_COMMIT = 33, // Temporary state in write commit log
+ LOG_COMMIT_WRITTEN = 34 // Temporary state in write commit log
+ };
+ enum ConnectState {
+ DISCONNECTED = 0,
+ CONNECTED = 1,
+ COPY_CONNECTED = 2,
+ LOG_CONNECTED = 3
+ };
+ ConnectState connectState;
+ UintR copyCountWords;
+ UintR firstAttrinfo[5];
+ UintR tupkeyData[4];
+ UintR transid[2];
+ AbortState abortState;
+ UintR accConnectrec;
+ UintR applOprec;
+ UintR clientConnectrec;
+ UintR tcTimer;
+ UintR currReclenAi;
+ UintR currTupAiLen;
+ UintR firstAttrinbuf;
+ UintR firstTupkeybuf;
+ UintR fragmentid;
+ UintR fragmentptr;
+ UintR gci;
+ UintR hashValue;
+ UintR lastTupkeybuf;
+ UintR lastAttrinbuf;
+ /**
+ * Each operation (TcConnectrec) can be stored in max one out of many
+ * lists.
+ * This variable keeps track of which list it is in.
+ */
+ ListState listState;
+
+ UintR logStartFileNo;
+ LogWriteState logWriteState;
+ UintR nextHashRec;
+ UintR nextLogTcrec;
+ UintR nextTcLogQueue;
+ UintR nextTc;
+ UintR nextTcConnectrec;
+ Uint16 nodeAfterNext[2];
+ UintR prevHashRec;
+ UintR prevLogTcrec;
+ UintR prevTc;
+ UintR readlenAi;
+ UintR reqRef;
+ UintR reqinfo;
+ UintR schemaVersion;
+ UintR storedProcId;
+ UintR simpleTcConnect;
+ UintR tableref;
+ UintR tcOprec;
+ UintR tcScanInfo;
+ UintR tcScanRec;
+ UintR totReclenAi;
+ UintR totSendlenAi;
+ UintR tupConnectrec;
+ UintR savePointId;
+ TransactionState transactionState;
+ BlockReference applRef;
+ BlockReference clientBlockref;
+
+ BlockReference reqBlockref;
+ BlockReference tcBlockref;
+ BlockReference tcAccBlockref;
+ BlockReference tcTuxBlockref;
+ BlockReference tcTupBlockref;
+ Uint32 commitAckMarker;
+ UintR noFiredTriggers;
+
+ Uint16 errorCode;
+ Uint16 logStartPageIndex;
+ Uint16 logStartPageNo;
+ Uint16 logStopPageNo;
+ Uint16 nextReplica;
+ Uint16 primKeyLen;
+ Uint16 save1;
+
+ Uint8 activeCreat;
+ Uint8 apiVersionNo;
+ Uint8 dirtyOp;
+ Uint8 indTakeOver;
+ Uint8 lastReplicaNo;
+ Uint8 localFragptr;
+ Uint8 lockType;
+ Uint8 nextSeqNoReplica;
+ Uint8 opSimple;
+ Uint8 opExec;
+ Uint8 operation;
+ Uint8 reclenAiLqhkey;
+ Uint8 replicaType;
+ Uint8 simpleRead;
+ Uint8 seqNoReplica;
+ Uint8 tcNodeFailrec;
+ }; /* p2c: size = 280 bytes */
+
+ typedef Ptr<TcConnectionrec> TcConnectionrecPtr;
+
+ struct TcNodeFailRecord {
+ enum TcFailStatus {
+ TC_STATE_TRUE = 0,
+ TC_STATE_FALSE = 1,
+ TC_STATE_BREAK = 2
+ };
+ UintR lastNewTcRef;
+ UintR newTcRef;
+ TcFailStatus tcFailStatus;
+ UintR tcRecNow;
+ BlockReference lastNewTcBlockref;
+ BlockReference newTcBlockref;
+ Uint16 oldNodeId;
+ }; // Size 28 bytes
+ typedef Ptr<TcNodeFailRecord> TcNodeFailRecordPtr;
+
+ struct CommitLogRecord {
+ Uint32 startPageNo;
+ Uint32 startPageIndex;
+ Uint32 stopPageNo;
+ Uint32 fileNo;
+ };
+
+public:
+ Dblqh(const class Configuration &);
+ virtual ~Dblqh();
+
+private:
+ BLOCK_DEFINES(Dblqh);
+
+ void execPACKED_SIGNAL(Signal* signal);
+ void execDEBUG_SIG(Signal* signal);
+ void execATTRINFO(Signal* signal);
+ void execKEYINFO(Signal* signal);
+ void execLQHKEYREQ(Signal* signal);
+ void execLQHKEYREF(Signal* signal);
+ void execCOMMIT(Signal* signal);
+ void execCOMPLETE(Signal* signal);
+ void execLQHKEYCONF(Signal* signal);
+ void execTESTSIG(Signal* signal);
+ void execLQH_RESTART_OP(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+ void execSTART_RECREQ(Signal* signal);
+ void execSTART_RECCONF(Signal* signal);
+ void execEXEC_FRAGREQ(Signal* signal);
+ void execEXEC_FRAGCONF(Signal* signal);
+ void execEXEC_FRAGREF(Signal* signal);
+ void execSTART_EXEC_SR(Signal* signal);
+ void execEXEC_SRREQ(Signal* signal);
+ void execEXEC_SRCONF(Signal* signal);
+
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execACC_COM_BLOCK(Signal* signal);
+ void execACC_COM_UNBLOCK(Signal* signal);
+ void execTUP_COM_BLOCK(Signal* signal);
+ void execTUP_COM_UNBLOCK(Signal* signal);
+ void execACC_ABORTCONF(Signal* signal);
+ void execNODE_FAILREP(Signal* signal);
+ void execCHECK_LCP_STOP(Signal* signal);
+ void execSEND_PACKED(Signal* signal);
+ void execTUP_ATTRINFO(Signal* signal);
+ void execSIZEALT_REP(Signal* signal);
+ void execLQHFRAGREQ(Signal* signal);
+ void execLQHADDATTREQ(Signal* signal);
+ void execTUP_ADD_ATTCONF(Signal* signal);
+ void execTUP_ADD_ATTRREF(Signal* signal);
+ void execACCFRAGCONF(Signal* signal);
+ void execACCFRAGREF(Signal* signal);
+ void execTUPFRAGCONF(Signal* signal);
+ void execTUPFRAGREF(Signal* signal);
+ void execTAB_COMMITREQ(Signal* signal);
+ void execACCSEIZECONF(Signal* signal);
+ void execACCSEIZEREF(Signal* signal);
+ void execREAD_NODESCONF(Signal* signal);
+ void execREAD_NODESREF(Signal* signal);
+ void execSTTOR(Signal* signal);
+ void execNDB_STTOR(Signal* signal);
+ void execTUPSEIZECONF(Signal* signal);
+ void execTUPSEIZEREF(Signal* signal);
+ void execACCKEYCONF(Signal* signal);
+ void execACCKEYREF(Signal* signal);
+ void execTUPKEYCONF(Signal* signal);
+ void execTUPKEYREF(Signal* signal);
+ void execABORT(Signal* signal);
+ void execABORTREQ(Signal* signal);
+ void execCOMMITREQ(Signal* signal);
+ void execCOMPLETEREQ(Signal* signal);
+ void execMEMCHECKREQ(Signal* signal);
+ void execSCAN_FRAGREQ(Signal* signal);
+ void execSCAN_NEXTREQ(Signal* signal);
+ void execACC_SCANCONF(Signal* signal);
+ void execACC_SCANREF(Signal* signal);
+ void execNEXT_SCANCONF(Signal* signal);
+ void execNEXT_SCANREF(Signal* signal);
+ void execACC_SCAN_INFO(Signal* signal);
+ void execACC_SCAN_INFO24(Signal* signal);
+ void execACC_TO_REF(Signal* signal);
+ void execSTORED_PROCCONF(Signal* signal);
+ void execSTORED_PROCREF(Signal* signal);
+ void execCOPY_FRAGREQ(Signal* signal);
+ void execCOPY_ACTIVEREQ(Signal* signal);
+ void execCOPY_STATEREQ(Signal* signal);
+ void execLQH_TRANSREQ(Signal* signal);
+ void execTRANSID_AI(Signal* signal);
+ void execINCL_NODEREQ(Signal* signal);
+ void execACC_LCPCONF(Signal* signal);
+ void execACC_LCPREF(Signal* signal);
+ void execACC_LCPSTARTED(Signal* signal);
+ void execACC_CONTOPCONF(Signal* signal);
+ void execLCP_FRAGIDCONF(Signal* signal);
+ void execLCP_FRAGIDREF(Signal* signal);
+ void execLCP_HOLDOPCONF(Signal* signal);
+ void execLCP_HOLDOPREF(Signal* signal);
+ void execTUP_PREPLCPCONF(Signal* signal);
+ void execTUP_PREPLCPREF(Signal* signal);
+ void execTUP_LCPCONF(Signal* signal);
+ void execTUP_LCPREF(Signal* signal);
+ void execTUP_LCPSTARTED(Signal* signal);
+ void execEND_LCPCONF(Signal* signal);
+
+ void execLCP_FRAG_ORD(Signal* signal);
+ void execEMPTY_LCP_REQ(Signal* signal);
+
+ void execSTART_FRAGREQ(Signal* signal);
+ void execSTART_RECREF(Signal* signal);
+ void execSR_FRAGIDCONF(Signal* signal);
+ void execSR_FRAGIDREF(Signal* signal);
+ void execACC_SRCONF(Signal* signal);
+ void execACC_SRREF(Signal* signal);
+ void execTUP_SRCONF(Signal* signal);
+ void execTUP_SRREF(Signal* signal);
+ void execGCP_SAVEREQ(Signal* signal);
+ void execFSOPENCONF(Signal* signal);
+ void execFSOPENREF(Signal* signal);
+ void execFSCLOSECONF(Signal* signal);
+ void execFSCLOSEREF(Signal* signal);
+ void execFSWRITECONF(Signal* signal);
+ void execFSWRITEREF(Signal* signal);
+ void execFSREADCONF(Signal* signal);
+ void execFSREADREF(Signal* signal);
+ void execSCAN_HBREP(Signal* signal);
+ void execSET_VAR_REQ(Signal* signal);
+ void execTIME_SIGNAL(Signal* signal);
+ void execFSSYNCCONF(Signal* signal);
+ void execFSSYNCREF(Signal* signal);
+
+ void execALTER_TAB_REQ(Signal* signal);
+ void execALTER_TAB_CONF(Signal* signal);
+
+ void execCREATE_TRIG_CONF(Signal* signal);
+ void execCREATE_TRIG_REF(Signal* signal);
+ void execCREATE_TRIG_REQ(Signal* signal);
+
+ void execDROP_TRIG_CONF(Signal* signal);
+ void execDROP_TRIG_REF(Signal* signal);
+ void execDROP_TRIG_REQ(Signal* signal);
+
+ void execPREP_DROP_TAB_REQ(Signal* signal);
+ void execWAIT_DROP_TAB_REQ(Signal* signal);
+ void execDROP_TAB_REQ(Signal* signal);
+
+ void execLQH_ALLOCREQ(Signal* signal);
+ void execLQH_WRITELOG_REQ(Signal* signal);
+
+ void execTUXFRAGCONF(Signal* signal);
+ void execTUXFRAGREF(Signal* signal);
+ void execTUX_ADD_ATTRCONF(Signal* signal);
+ void execTUX_ADD_ATTRREF(Signal* signal);
+
+ // Statement blocks
+ void removeTable(Uint32 tableId);
+ void sendLCP_COMPLETE_REP(Signal* signal, Uint32 lcpId);
+ void sendEMPTY_LCP_CONF(Signal* signal, bool idle);
+ void sendLCP_FRAGIDREQ(Signal* signal);
+ void sendLCP_FRAG_REP(Signal * signal, const LcpRecord::FragOrd &) const;
+
+ void updatePackedList(Signal* signal, HostRecord * ahostptr, Uint16 hostId);
+ void LQHKEY_abort(Signal* signal, int errortype);
+ void LQHKEY_error(Signal* signal, int errortype);
+ void nextRecordCopy(Signal* signal);
+ void calculateHash(Signal* signal);
+ void continueAfterCheckLcpStopBlocked(Signal* signal);
+ void checkLcpStopBlockedLab(Signal* signal);
+ void sendCommittedTc(Signal* signal, BlockReference atcBlockref);
+ void sendCompletedTc(Signal* signal, BlockReference atcBlockref);
+ void sendLqhkeyconfTc(Signal* signal, BlockReference atcBlockref);
+ void sendCommitLqh(Signal* signal, BlockReference alqhBlockref);
+ void sendCompleteLqh(Signal* signal, BlockReference alqhBlockref);
+ void sendPackedSignalLqh(Signal* signal, HostRecord * ahostptr);
+ void sendPackedSignalTc(Signal* signal, HostRecord * ahostptr);
+ Uint32 handleLongTupKey(Signal* signal,
+ Uint32 lenSofar,
+ Uint32 primKeyLen,
+ Uint32* dataPtr);
+ void cleanUp(Signal* signal);
+ void sendAttrinfoLoop(Signal* signal);
+ void sendAttrinfoSignal(Signal* signal);
+ void sendLqhAttrinfoSignal(Signal* signal);
+ void sendKeyinfoAcc(Signal* signal);
+ void initScanAccOp(Signal* signal);
+ Uint32 initScanrec(const class ScanFragReq *);
+ void initScanTc(Signal* signal,
+ Uint32 transid1,
+ Uint32 transid2,
+ Uint32 fragId,
+ Uint32 nodeId);
+ void finishScanrec(Signal* signal);
+ void releaseScanrec(Signal* signal);
+ void seizeScanrec(Signal* signal);
+ void sendKeyinfo20(Signal* signal, ScanRecord *, TcConnectionrec *);
+ void sendScanFragConf(Signal* signal, Uint32 scanCompleted);
+ void initCopyrec(Signal* signal);
+ void initCopyTc(Signal* signal);
+ void sendCopyActiveConf(Signal* signal,Uint32 tableId);
+ void checkLcpCompleted(Signal* signal);
+ void checkLcpHoldop(Signal* signal);
+ void checkLcpStarted(Signal* signal);
+ void checkLcpTupprep(Signal* signal);
+ void getNextFragForLcp(Signal* signal);
+ void initLcpLocAcc(Signal* signal, Uint32 fragId);
+ void initLcpLocTup(Signal* signal, Uint32 fragId);
+ void moveAccActiveFrag(Signal* signal);
+ void moveActiveToAcc(Signal* signal);
+ void releaseLocalLcps(Signal* signal);
+ void seizeLcpLoc(Signal* signal);
+ void sendAccContOp(Signal* signal);
+ void sendStartLcp(Signal* signal);
+ void setLogTail(Signal* signal, Uint32 keepGci);
+ Uint32 remainingLogSize(const LogFileRecordPtr &sltCurrLogFilePtr,
+ const LogPartRecordPtr &sltLogPartPtr);
+ void checkGcpCompleted(Signal* signal, Uint32 pageWritten, Uint32 wordWritten);
+ void initFsopenconf(Signal* signal);
+ void initFsrwconf(Signal* signal);
+ void initLfo(Signal* signal);
+ void initLogfile(Signal* signal, Uint32 fileNo);
+ void initLogpage(Signal* signal);
+ void openFileRw(Signal* signal, LogFileRecordPtr olfLogFilePtr);
+ void openLogfileInit(Signal* signal);
+ void openNextLogfile(Signal* signal);
+ void releaseLfo(Signal* signal);
+ void releaseLfoPages(Signal* signal);
+ void releaseLogpage(Signal* signal);
+ void seizeLfo(Signal* signal);
+ void seizeLogfile(Signal* signal);
+ void seizeLogpage(Signal* signal);
+ void writeFileDescriptor(Signal* signal);
+ void writeFileHeaderOpen(Signal* signal, Uint32 type);
+ void writeInitMbyte(Signal* signal);
+ void writeSinglePage(Signal* signal, Uint32 pageNo, Uint32 wordWritten);
+ void buildLinkedLogPageList(Signal* signal);
+ void changeMbyte(Signal* signal);
+ Uint32 checkIfExecLog(Signal* signal);
+ void checkNewMbyte(Signal* signal);
+ void checkReadExecSr(Signal* signal);
+ void checkScanTcCompleted(Signal* signal);
+ void checkSrCompleted(Signal* signal);
+ void closeFile(Signal* signal, LogFileRecordPtr logFilePtr);
+ void completedLogPage(Signal* signal, Uint32 clpType);
+ void deleteFragrec(Uint32 fragId);
+ void deleteTransidHash(Signal* signal);
+ void findLogfile(Signal* signal,
+ Uint32 fileNo,
+ LogPartRecordPtr flfLogPartPtr,
+ LogFileRecordPtr* parLogFilePtr);
+ void findPageRef(Signal* signal, CommitLogRecord* commitLogRecord);
+ int findTransaction(UintR Transid1, UintR Transid2, UintR TcOprec);
+ void getFirstInLogQueue(Signal* signal);
+ bool getFragmentrec(Signal* signal, Uint32 fragId);
+ void initialiseAddfragrec(Signal* signal);
+ void initialiseAttrbuf(Signal* signal);
+ void initialiseDatabuf(Signal* signal);
+ void initialiseFragrec(Signal* signal);
+ void initialiseGcprec(Signal* signal);
+ void initialiseLcpRec(Signal* signal);
+ void initialiseLcpLocrec(Signal* signal);
+ void initialiseLfo(Signal* signal);
+ void initialiseLogFile(Signal* signal);
+ void initialiseLogPage(Signal* signal);
+ void initialiseLogPart(Signal* signal);
+ void initialisePageRef(Signal* signal);
+ void sendInitialiseRecords(Signal* signal, Uint32 data);
+ void initialiseScanrec(Signal* signal);
+ void initialiseTabrec(Signal* signal);
+ void initialiseTcrec(Signal* signal);
+ void initialiseTcNodeFailRec(Signal* signal);
+ void initFragrec(Signal* signal,
+ Uint32 tableId,
+ Uint32 fragId,
+ Uint32 copyType);
+ void initFragrecSr(Signal* signal);
+ void initGciInLogFileRec(Signal* signal, Uint32 noFdDesc);
+ void initLcpSr(Signal* signal,
+ Uint32 lcpNo,
+ Uint32 lcpId,
+ Uint32 tableId,
+ Uint32 fragId,
+ Uint32 fragPtr);
+ void initLogpart(Signal* signal);
+ void initLogPointers(Signal* signal);
+ void initReqinfoExecSr(Signal* signal);
+ bool insertFragrec(Signal* signal, Uint32 fragId);
+ void linkActiveFrag(Signal* signal);
+ void linkFragQueue(Signal* signal);
+ void linkWaitLog(Signal* signal, LogPartRecordPtr regLogPartPtr);
+ void logNextStart(Signal* signal);
+ void moveToPageRef(Signal* signal);
+ void readAttrinfo(Signal* signal);
+ void readCommitLog(Signal* signal, CommitLogRecord* commitLogRecord);
+ void readExecLog(Signal* signal);
+ void readExecSrNewMbyte(Signal* signal);
+ void readExecSr(Signal* signal);
+ void readKey(Signal* signal);
+ void readLogData(Signal* signal, Uint32 noOfWords, Uint32* dataPtr);
+ void readLogHeader(Signal* signal);
+ Uint32 readLogword(Signal* signal);
+ Uint32 readLogwordExec(Signal* signal);
+ void readSinglePage(Signal* signal, Uint32 pageNo);
+ void releaseAccList(Signal* signal);
+ void releaseActiveCopy(Signal* signal);
+ void releaseActiveFrag(Signal* signal);
+ void releaseActiveList(Signal* signal);
+ void releaseAddfragrec(Signal* signal);
+ void releaseFragrec();
+ void releaseLcpLoc(Signal* signal);
+ void releaseOprec(Signal* signal);
+ void releasePageRef(Signal* signal);
+ void releaseMmPages(Signal* signal);
+ void releasePrPages(Signal* signal);
+ void releaseTcrec(Signal* signal, TcConnectionrecPtr tcConnectptr);
+ void releaseTcrecLog(Signal* signal, TcConnectionrecPtr tcConnectptr);
+ void releaseWaitQueue(Signal* signal);
+ void removeLogTcrec(Signal* signal);
+ void removePageRef(Signal* signal);
+ Uint32 returnExecLog(Signal* signal);
+ int saveTupattrbuf(Signal* signal, Uint32* dataPtr, Uint32 length);
+ void seizeAddfragrec(Signal* signal);
+ void seizeAttrinbuf(Signal* signal);
+ void seizeFragmentrec(Signal* signal);
+ void seizePageRef(Signal* signal);
+ void seizeTcrec();
+ void seizeTupkeybuf(Signal* signal);
+ void sendAborted(Signal* signal);
+ void sendLqhTransconf(Signal* signal, LqhTransConf::OperationStatus);
+ void sendTupkey(Signal* signal);
+ void startExecSr(Signal* signal);
+ void startNextExecSr(Signal* signal);
+ void startTimeSupervision(Signal* signal);
+ void stepAhead(Signal* signal, Uint32 stepAheadWords);
+ void systemError(Signal* signal);
+ void writeAbortLog(Signal* signal);
+ void writeCommitLog(Signal* signal, LogPartRecordPtr regLogPartPtr);
+ void writeCompletedGciLog(Signal* signal);
+ void writeDirty(Signal* signal);
+ void writeKey(Signal* signal);
+ void writeLogHeader(Signal* signal);
+ void writeLogWord(Signal* signal, Uint32 data);
+ void writeNextLog(Signal* signal);
+ void errorReport(Signal* signal, int place);
+ void warningReport(Signal* signal, int place);
+ void invalidateLogAfterLastGCI(Signal *signal);
+ void readFileInInvalidate(Signal *signal);
+ void exitFromInvalidate(Signal* signal);
+ Uint32 calcPageCheckSum(LogPageRecordPtr logP);
+
+ // Generated statement blocks
+ void systemErrorLab(Signal* signal);
+ void initFourth(Signal* signal);
+ void packLqhkeyreqLab(Signal* signal);
+ void sendNdbSttorryLab(Signal* signal);
+ void execSrCompletedLab(Signal* signal);
+ void execLogRecord(Signal* signal);
+ void srPhase3Comp(Signal* signal);
+ void srLogLimits(Signal* signal);
+ void srGciLimits(Signal* signal);
+ void srPhase3Start(Signal* signal);
+ void warningHandlerLab(Signal* signal);
+ void checkStartCompletedLab(Signal* signal);
+ void continueAbortLab(Signal* signal);
+ void abortContinueAfterBlockedLab(Signal* signal, bool canBlock);
+ void abortCommonLab(Signal* signal);
+ void localCommitLab(Signal* signal);
+ void abortErrorLab(Signal* signal);
+ void continueAfterReceivingAllAiLab(Signal* signal);
+ void sendScanFragRefLateLab(Signal* signal);
+ void abortStateHandlerLab(Signal* signal);
+ void writeAttrinfoLab(Signal* signal);
+ void scanAttrinfoLab(Signal* signal, Uint32* dataPtr, Uint32 length);
+ void localAbortStateHandlerLab(Signal* signal);
+ void logLqhkeyreqLab(Signal* signal);
+ void lqhAttrinfoLab(Signal* signal, Uint32* dataPtr, Uint32 length);
+ void rwConcludedAiLab(Signal* signal);
+ void aiStateErrorCheckLab(Signal* signal, Uint32* dataPtr, Uint32 length);
+ void takeOverErrorLab(Signal* signal);
+ void endgettupkeyLab(Signal* signal);
+ void noFreeRecordLab(Signal* signal,
+ const class LqhKeyReq * lqhKeyReq,
+ Uint32 errorCode);
+ void logLqhkeyrefLab(Signal* signal);
+ void closeCopyLab(Signal* signal);
+ void commitReplyLab(Signal* signal);
+ void completeUnusualLab(Signal* signal);
+ void completeTransNotLastLab(Signal* signal);
+ void completedLab(Signal* signal);
+ void copyCompletedLab(Signal* signal);
+ void completeLcpRoundLab(Signal* signal);
+ void continueAfterLogAbortWriteLab(Signal* signal);
+ void sendAttrinfoLab(Signal* signal);
+ void sendExecConf(Signal* signal);
+ void execSr(Signal* signal);
+ void srFourthComp(Signal* signal);
+ void timeSup(Signal* signal);
+ void closeCopyRequestLab(Signal* signal);
+ void closeScanRequestLab(Signal* signal);
+ void scanTcConnectLab(Signal* signal, Uint32 startTcCon, Uint32 fragId);
+ void returnInitialiseRecordsLab(Signal* signal);
+ void initGcpRecLab(Signal* signal);
+ void prepareContinueAfterBlockedLab(Signal* signal);
+ void commitContinueAfterBlockedLab(Signal* signal);
+ void continueCopyAfterBlockedLab(Signal* signal);
+ void continueFirstCopyAfterBlockedLab(Signal* signal);
+ void continueFirstScanAfterBlockedLab(Signal* signal);
+ void continueScanAfterBlockedLab(Signal* signal);
+ void continueScanReleaseAfterBlockedLab(Signal* signal);
+ void continueCloseScanAfterBlockedLab(Signal* signal);
+ void continueCloseCopyAfterBlockedLab(Signal* signal);
+ void sendExecFragRefLab(Signal* signal);
+ void fragrefLab(Signal* signal, BlockReference retRef,
+ Uint32 retPtr, Uint32 errorCode);
+ void accFragRefLab(Signal* signal);
+ void rwConcludedLab(Signal* signal);
+ void sendsttorryLab(Signal* signal);
+ void initialiseRecordsLab(Signal* signal, Uint32 data);
+ void startphase2Lab(Signal* signal, Uint32 config);
+ void startphase3Lab(Signal* signal);
+ void startphase4Lab(Signal* signal);
+ void startphase6Lab(Signal* signal);
+ void moreconnectionsLab(Signal* signal);
+ void scanReleaseLocksLab(Signal* signal);
+ void closeScanLab(Signal* signal);
+ void nextScanConfLoopLab(Signal* signal);
+ void scanNextLoopLab(Signal* signal);
+ void commitReqLab(Signal* signal, Uint32 gci);
+ void completeTransLastLab(Signal* signal);
+ void tupScanCloseConfLab(Signal* signal);
+ void tupCopyCloseConfLab(Signal* signal);
+ void accScanCloseConfLab(Signal* signal);
+ void accCopyCloseConfLab(Signal* signal);
+ void nextScanConfScanLab(Signal* signal);
+ void nextScanConfCopyLab(Signal* signal);
+ void continueScanNextReqLab(Signal* signal);
+ bool keyinfoLab(Signal* signal, Uint32* dataPtr, Uint32 length);
+ void copySendTupkeyReqLab(Signal* signal);
+ void storedProcConfScanLab(Signal* signal);
+ void storedProcConfCopyLab(Signal* signal);
+ void copyStateFinishedLab(Signal* signal);
+ void lcpCompletedLab(Signal* signal);
+ void lcpStartedLab(Signal* signal);
+ void contChkpNextFragLab(Signal* signal);
+ void startLcpRoundLab(Signal* signal);
+ void startFragRefLab(Signal* signal);
+ void srCompletedLab(Signal* signal);
+ void openFileInitLab(Signal* signal);
+ void openSrFrontpageLab(Signal* signal);
+ void openSrLastFileLab(Signal* signal);
+ void openSrNextFileLab(Signal* signal);
+ void openExecSrStartLab(Signal* signal);
+ void openExecSrNewMbyteLab(Signal* signal);
+ void openSrFourthPhaseLab(Signal* signal);
+ void openSrFourthZeroSkipInitLab(Signal* signal);
+ void openSrFourthZeroLab(Signal* signal);
+ void openExecLogLab(Signal* signal);
+ void checkInitCompletedLab(Signal* signal);
+ void closingSrLab(Signal* signal);
+ void closeExecSrLab(Signal* signal);
+ void execLogComp(Signal* signal);
+ void closeWriteLogLab(Signal* signal);
+ void closeExecLogLab(Signal* signal);
+ void writePageZeroLab(Signal* signal);
+ void lastWriteInFileLab(Signal* signal);
+ void initWriteEndLab(Signal* signal);
+ void initFirstPageLab(Signal* signal);
+ void writeGciZeroLab(Signal* signal);
+ void writeDirtyLab(Signal* signal);
+ void writeInitMbyteLab(Signal* signal);
+ void writeLogfileLab(Signal* signal);
+ void firstPageWriteLab(Signal* signal);
+ void readSrLastMbyteLab(Signal* signal);
+ void readSrLastFileLab(Signal* signal);
+ void readSrNextFileLab(Signal* signal);
+ void readExecSrLab(Signal* signal);
+ void readExecLogLab(Signal* signal);
+ void readSrFourthPhaseLab(Signal* signal);
+ void readSrFourthZeroLab(Signal* signal);
+ void copyLqhKeyRefLab(Signal* signal);
+ void restartOperationsLab(Signal* signal);
+ void lqhTransNextLab(Signal* signal);
+ void restartOperationsAfterStopLab(Signal* signal);
+ void sttorStartphase1Lab(Signal* signal);
+ void startphase1Lab(Signal* signal, Uint32 config, Uint32 nodeId);
+ void tupkeyConfLab(Signal* signal);
+ void copyTupkeyConfLab(Signal* signal);
+ void scanTupkeyConfLab(Signal* signal);
+ void scanTupkeyRefLab(Signal* signal);
+ void accScanConfScanLab(Signal* signal);
+ void accScanConfCopyLab(Signal* signal);
+ void scanLockReleasedLab(Signal* signal);
+ void accScanInfoEnterLab(Signal* signal, Uint32* dataPtr, Uint32 length);
+ void openSrFourthNextLab(Signal* signal);
+ void closingInitLab(Signal* signal);
+ void closeExecSrCompletedLab(Signal* signal);
+ void readSrFrontpageLab(Signal* signal);
+
+ void sendAddFragReq(Signal* signal);
+ void sendAddAttrReq(Signal* signal);
+ void checkDropTab(Signal*);
+ Uint32 checkDropTabState(Tablerec::TableStatus, Uint32) const;
+
+ // Initialisation
+ void initData();
+ void initRecords();
+
+// ----------------------------------------------------------------
+// These are variables handling the records. For most records one
+// pointer to the array of structs, one pointer-struct, a file size
+// and a first free record variable. The pointer struct are temporary
+// variables that are kept on the class object since there are often a
+// great deal of those variables that exist simultaneously and
+// thus no perfect solution of handling them is currently available.
+// ----------------------------------------------------------------
+/* ------------------------------------------------------------------------- */
+/* POSITIONS WITHIN THE ATTRINBUF AND THE MAX SIZE OF DATA WITHIN AN */
+/* ATTRINBUF. */
+/* ------------------------------------------------------------------------- */
+
+
+#define ZADDFRAGREC_FILE_SIZE 1
+ AddFragRecord *addFragRecord;
+ AddFragRecordPtr addfragptr;
+ UintR cfirstfreeAddfragrec;
+ UintR caddfragrecFileSize;
+
+#define ZATTRINBUF_FILE_SIZE 10000 // 1.25 MByte
+#define ZINBUF_DATA_LEN 24 /* POSITION OF 'DATA LENGHT'-VARIABLE. */
+#define ZINBUF_NEXT 25 /* POSITION OF 'NEXT'-VARIABLE. */
+ Attrbuf *attrbuf;
+ AttrbufPtr attrinbufptr;
+ UintR cfirstfreeAttrinbuf;
+ UintR cattrinbufFileSize;
+
+#define ZDATABUF_FILE_SIZE 10000 // 200 kByte
+ Databuf *databuf;
+ DatabufPtr databufptr;
+ UintR cfirstfreeDatabuf;
+ UintR cdatabufFileSize;
+
+// Configurable
+ Fragrecord *fragrecord;
+ FragrecordPtr fragptr;
+ UintR cfirstfreeFragrec;
+ UintR cfragrecFileSize;
+
+#define ZGCPREC_FILE_SIZE 1
+ GcpRecord *gcpRecord;
+ GcpRecordPtr gcpPtr;
+ UintR cgcprecFileSize;
+
+// MAX_NDB_NODES is the size of this array
+ HostRecord *hostRecord;
+ UintR chostFileSize;
+
+#define ZNO_CONCURRENT_LCP 1
+ LcpRecord *lcpRecord;
+ LcpRecordPtr lcpPtr;
+ UintR cfirstfreeLcpLoc;
+ UintR clcpFileSize;
+
+#define ZLCP_LOCREC_FILE_SIZE 4
+ LcpLocRecord *lcpLocRecord;
+ LcpLocRecordPtr lcpLocptr;
+ UintR clcpLocrecFileSize;
+
+#define ZLOG_PART_FILE_SIZE 4
+ LogPartRecord *logPartRecord;
+ LogPartRecordPtr logPartPtr;
+ UintR clogPartFileSize;
+
+// Configurable
+ LogFileRecord *logFileRecord;
+ LogFileRecordPtr logFilePtr;
+ UintR cfirstfreeLogFile;
+ UintR clogFileFileSize;
+
+#define ZLFO_FILE_SIZE 256 /* MAX 256 OUTSTANDING FILE OPERATIONS */
+ LogFileOperationRecord *logFileOperationRecord;
+ LogFileOperationRecordPtr lfoPtr;
+ UintR cfirstfreeLfo;
+ UintR clfoFileSize;
+
+#define ZLOG_PAGE_FILE_SIZE 256 // 8 MByte
+ LogPageRecord *logPageRecord;
+ LogPageRecordPtr logPagePtr;
+ UintR cfirstfreeLogPage;
+ UintR clogPageFileSize;
+
+#define ZPAGE_REF_FILE_SIZE 20
+ PageRefRecord *pageRefRecord;
+ PageRefRecordPtr pageRefPtr;
+ UintR cfirstfreePageRef;
+ UintR cpageRefFileSize;
+
+#define ZSCANREC_FILE_SIZE 100
+ ScanRecord *scanRecord;
+ ScanRecordPtr scanptr;
+ UintR cfirstfreeScanrec;
+ UintR cscanrecFileSize;
+ UintR cscanNoFreeRec;
+
+// Configurable
+ Tablerec *tablerec;
+ TablerecPtr tabptr;
+ UintR ctabrecFileSize;
+
+// Configurable
+ TcConnectionrec *tcConnectionrec;
+ TcConnectionrecPtr tcConnectptr;
+ UintR cfirstfreeTcConrec;
+ UintR ctcConnectrecFileSize;
+
+// MAX_NDB_NODES is the size of this array
+ TcNodeFailRecord *tcNodeFailRecord;
+ TcNodeFailRecordPtr tcNodeFailptr;
+ UintR ctcNodeFailrecFileSize;
+
+ Uint16 terrorCode;
+
+ Uint32 c_firstInNodeGroup;
+
+// ------------------------------------------------------------------------
+// These variables are used to store block state which do not need arrays
+// of struct's.
+// ------------------------------------------------------------------------
+ Uint32 c_lcpId;
+ Uint32 cnoOfFragsCheckpointed;
+
+/* ------------------------------------------------------------------------- */
+// cmaxWordsAtNodeRec keeps track of how many words that currently are
+// outstanding in a node recovery situation.
+// cbookedAccOps keeps track of how many operation records that have been
+// booked in ACC for the scan processes.
+// cmaxAccOps contains the maximum number of operation records which can be
+// allocated for scan purposes in ACC.
+/* ------------------------------------------------------------------------- */
+ UintR cmaxWordsAtNodeRec;
+ UintR cbookedAccOps;
+ UintR cmaxAccOps;
+/* ------------------------------------------------------------------------- */
+/*THIS STATE VARIABLE IS ZTRUE IF AN ADD NODE IS ONGOING. ADD NODE MEANS */
+/*THAT CONNECTIONS ARE SET-UP TO THE NEW NODE. */
+/* ------------------------------------------------------------------------- */
+ Uint8 caddNodeState;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE SPECIFIES WHICH TYPE OF RESTART THAT IS ONGOING */
+/* ------------------------------------------------------------------------- */
+ Uint16 cstartType;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE INDICATES WHETHER AN INITIAL RESTART IS ONGOING OR NOT. */
+/* ------------------------------------------------------------------------- */
+ Uint8 cinitialStartOngoing;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE KEEPS TRACK OF WHEN TUP AND ACC HAVE COMPLETED EXECUTING */
+/*THEIR UNDO LOG. */
+/* ------------------------------------------------------------------------- */
+ ExecUndoLogState csrExecUndoLogState;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE KEEPS TRACK OF WHEN TUP AND ACC HAVE CONFIRMED COMPLETION */
+/*OF A LOCAL CHECKPOINT ROUND. */
+/* ------------------------------------------------------------------------- */
+ LcpCloseState clcpCompletedState;
+/* ------------------------------------------------------------------------- */
+/*DURING CONNECTION PROCESSES IN SYSTEM RESTART THESE VARIABLES KEEP TRACK */
+/*OF HOW MANY CONNECTIONS AND RELEASES THAT ARE TO BE PERFORMED. */
+/* ------------------------------------------------------------------------- */
+/***************************************************************************>*/
+/*THESE VARIABLES CONTAIN INFORMATION USED DURING SYSTEM RESTART. */
+/***************************************************************************>*/
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE IS ZTRUE IF THE SIGNAL START_REC_REQ HAVE BEEN RECEIVED. */
+/*RECEPTION OF THIS SIGNAL INDICATES THAT ALL FRAGMENTS THAT THIS NODE */
+/*SHOULD START HAVE BEEN RECEIVED. */
+/* ------------------------------------------------------------------------- */
+ Uint8 cstartRecReq;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE KEEPS TRACK OF HOW MANY FRAGMENTS THAT PARTICIPATE IN */
+/*EXECUTING THE LOG. IF ZERO WE DON'T NEED TO EXECUTE THE LOG AT ALL. */
+/* ------------------------------------------------------------------------- */
+ UintR cnoFragmentsExecSr;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE KEEPS TRACK OF WHICH OF THE FIRST TWO RESTART PHASES THAT */
+/*HAVE COMPLETED. */
+/* ------------------------------------------------------------------------- */
+ Uint8 csrPhaseStarted;
+/* ------------------------------------------------------------------------- */
+/*NUMBER OF PHASES COMPLETED OF EXECUTING THE FRAGMENT LOG. */
+/* ------------------------------------------------------------------------- */
+ Uint8 csrPhasesCompleted;
+/* ------------------------------------------------------------------------- */
+/*THE BLOCK REFERENCE OF THE MASTER DIH DURING SYSTEM RESTART. */
+/* ------------------------------------------------------------------------- */
+ BlockReference cmasterDihBlockref;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE IS THE HEAD OF A LINKED LIST OF FRAGMENTS WAITING TO BE */
+/*RESTORED FROM DISK. */
+/* ------------------------------------------------------------------------- */
+ UintR cfirstWaitFragSr;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE IS THE HEAD OF A LINKED LIST OF FRAGMENTS THAT HAVE BEEN */
+/*RESTORED FROM DISK THAT AWAITS EXECUTION OF THE FRAGMENT LOG. */
+/* ------------------------------------------------------------------------- */
+ UintR cfirstCompletedFragSr;
+
+/* ------------------------------------------------------------------------- */
+/*USED DURING SYSTEM RESTART, INDICATES THE OLDEST GCI THAT CAN BE RESTARTED */
+/*FROM AFTER THIS SYSTEM RESTART. USED TO FIND THE LOG TAIL. */
+/* ------------------------------------------------------------------------- */
+ UintR crestartOldestGci;
+/* ------------------------------------------------------------------------- */
+/*USED DURING SYSTEM RESTART, INDICATES THE NEWEST GCI THAT CAN BE RESTARTED */
+/*AFTER THIS SYSTEM RESTART. USED TO FIND THE LOG HEAD. */
+/* ------------------------------------------------------------------------- */
+ UintR crestartNewestGci;
+/* ------------------------------------------------------------------------- */
+/*THE NUMBER OF LOG FILES. SET AS A PARAMETER WHEN NDB IS STARTED. */
+/* ------------------------------------------------------------------------- */
+ UintR cnoLogFiles;
+/* ------------------------------------------------------------------------- */
+/*THESE TWO VARIABLES CONTAIN THE NEWEST GCI RECEIVED IN THE BLOCK AND THE */
+/*NEWEST COMPLETED GCI IN THE BLOCK. */
+/* ------------------------------------------------------------------------- */
+ UintR cnewestGci;
+ UintR cnewestCompletedGci;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE ONLY PASSES INFORMATION FROM STTOR TO STTORRY = TEMPORARY */
+/* ------------------------------------------------------------------------- */
+ Uint16 csignalKey;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE CONTAINS THE CURRENT START PHASE IN THE BLOCK. IS ZNIL IF */
+/*NO SYSTEM RESTART IS ONGOING. */
+/* ------------------------------------------------------------------------- */
+ Uint16 cstartPhase;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE CONTAIN THE CURRENT GLOBAL CHECKPOINT RECORD. IT'S RNIL IF */
+/*NOT A GCP SAVE IS ONGOING. */
+/* ------------------------------------------------------------------------- */
+ UintR ccurrentGcprec;
+/* ------------------------------------------------------------------------- */
+/*THESE VARIABLES ARE USED TO KEEP TRACK OF ALL ACTIVE COPY FRAGMENTS IN LQH.*/
+/* ------------------------------------------------------------------------- */
+ Uint8 cnoActiveCopy;
+ UintR cactiveCopy[4];
+
+/* ------------------------------------------------------------------------- */
+/*THESE VARIABLES CONTAIN THE BLOCK REFERENCES OF THE OTHER NDB BLOCKS. */
+/*ALSO THE BLOCK REFERENCE OF MY OWN BLOCK = LQH */
+/* ------------------------------------------------------------------------- */
+ BlockReference caccBlockref;
+ BlockReference ctupBlockref;
+ BlockReference ctuxBlockref;
+ BlockReference cownref;
+ UintR cLqhTimeOutCount;
+ UintR cLqhTimeOutCheckCount;
+ UintR cnoOfLogPages;
+ bool caccCommitBlocked;
+ bool ctupCommitBlocked;
+ bool cCommitBlocked;
+ UintR cCounterAccCommitBlocked;
+ UintR cCounterTupCommitBlocked;
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE CONTAINS MY OWN PROCESSOR ID. */
+/* ------------------------------------------------------------------------- */
+ NodeId cownNodeid;
+
+/* ------------------------------------------------------------------------- */
+/*THESE VARIABLES CONTAIN INFORMATION ABOUT THE OTHER NODES IN THE SYSTEM */
+/*THESE VARIABLES ARE MOSTLY USED AT SYSTEM RESTART AND ADD NODE TO SET-UP */
+/*AND RELEASE CONNECTIONS TO OTHER NODES IN THE CLUSTER. */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/*THIS ARRAY CONTAINS THE PROCESSOR ID'S OF THE NODES THAT ARE ALIVE. */
+/*CNO_OF_NODES SPECIFIES HOW MANY NODES THAT ARE CURRENTLY ALIVE. */
+/*CNODE_VERSION SPECIFIES THE NDB VERSION EXECUTING ON THE NODE. */
+/* ------------------------------------------------------------------------- */
+ UintR cpackedListIndex;
+ Uint16 cpackedList[MAX_NDB_NODES];
+ UintR cnodeData[MAX_NDB_NODES];
+ UintR cnodeStatus[MAX_NDB_NODES];
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE INDICATES WHETHER A CERTAIN NODE HAS SENT ALL FRAGMENTS THAT */
+/*NEED TO HAVE THE LOG EXECUTED. */
+/* ------------------------------------------------------------------------- */
+ Uint8 cnodeSrState[MAX_NDB_NODES];
+/* ------------------------------------------------------------------------- */
+/*THIS VARIABLE INDICATES WHETHER A CERTAIN NODE HAVE EXECUTED THE LOG */
+/* ------------------------------------------------------------------------- */
+ Uint8 cnodeExecSrState[MAX_NDB_NODES];
+ UintR cnoOfNodes;
+
+/* ------------------------------------------------------------------------- */
+/* THIS VARIABLE CONTAINS THE DIRECTORY OF A HASH TABLE OF ALL ACTIVE */
+/* OPERATION IN THE BLOCK. IT IS USED TO BE ABLE TO QUICKLY ABORT AN */
+/* OPERATION WHERE THE CONNECTION WAS LOST DUE TO NODE FAILURES. IT IS */
+/* ACTUALLY USED FOR ALL ABORTS COMMANDED BY TC. */
+/* ------------------------------------------------------------------------- */
+ UintR preComputedRequestInfoMask;
+ UintR ctransidHash[1024];
+
+
+public:
+ /**
+ *
+ */
+ struct CommitAckMarker {
+ Uint32 transid1;
+ Uint32 transid2;
+
+ Uint32 apiRef; // Api block ref
+ Uint32 apiOprec; // Connection Object in NDB API
+ Uint32 tcNodeId;
+ union { Uint32 nextPool; Uint32 nextHash; };
+ Uint32 prevHash;
+
+ inline bool equal(const CommitAckMarker & p) const {
+ return ((p.transid1 == transid1) && (p.transid2 == transid2));
+ }
+
+ inline Uint32 hashValue() const {
+ return transid1;
+ }
+ };
+
+ typedef Ptr<CommitAckMarker> CommitAckMarkerPtr;
+ ArrayPool<CommitAckMarker> m_commitAckMarkerPool;
+ DLHashTable<CommitAckMarker> m_commitAckMarkerHash;
+ typedef DLHashTable<CommitAckMarker>::Iterator CommitAckMarkerIterator;
+ void execREMOVE_MARKER_ORD(Signal* signal);
+ void scanMarkers(Signal* signal, Uint32 tcNodeFail, Uint32 bucket, Uint32 i);
+
+ struct Counters {
+ Uint32 operations;
+
+ inline void clear(){
+ operations = 0;
+ }
+ };
+
+ Counters c_Counters;
+
+ inline bool getAllowRead() const {
+ return getNodeState().startLevel < NodeState::SL_STOPPING_3;
+ }
+
+
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/dblqh/DblqhInit.cpp b/ndb/src/kernel/blocks/dblqh/DblqhInit.cpp
new file mode 100644
index 00000000000..615cfa4ea0b
--- /dev/null
+++ b/ndb/src/kernel/blocks/dblqh/DblqhInit.cpp
@@ -0,0 +1,416 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <pc.hpp>
+#define DBLQH_C
+#include "Dblqh.hpp"
+#include <ndb_limits.h>
+#include <new>
+
+#define DEBUG(x) { ndbout << "LQH::" << x << endl; }
+
+void Dblqh::initData()
+{
+ caddfragrecFileSize = ZADDFRAGREC_FILE_SIZE;
+ cattrinbufFileSize = ZATTRINBUF_FILE_SIZE;
+ cdatabufFileSize = ZDATABUF_FILE_SIZE;
+ cfragrecFileSize = 0;
+ cgcprecFileSize = ZGCPREC_FILE_SIZE;
+ chostFileSize = MAX_NDB_NODES;
+ clcpFileSize = ZNO_CONCURRENT_LCP;
+ clcpLocrecFileSize = ZLCP_LOCREC_FILE_SIZE;
+ clogPageFileSize = ZLOG_PAGE_FILE_SIZE;
+ clfoFileSize = ZLFO_FILE_SIZE;
+ clogFileFileSize = 0;
+ clogPartFileSize = ZLOG_PART_FILE_SIZE;
+ cpageRefFileSize = ZPAGE_REF_FILE_SIZE;
+ cscanrecFileSize = ZSCANREC_FILE_SIZE;
+ ctabrecFileSize = 0;
+ ctcConnectrecFileSize = 0;
+ ctcNodeFailrecFileSize = MAX_NDB_NODES;
+
+ addFragRecord = 0;
+ attrbuf = 0;
+ databuf = 0;
+ fragrecord = 0;
+ gcpRecord = 0;
+ hostRecord = 0;
+ lcpRecord = 0;
+ lcpLocRecord = 0;
+ logPartRecord = 0;
+ logFileRecord = 0;
+ logFileOperationRecord = 0;
+ logPageRecord = 0;
+ pageRefRecord = 0;
+ scanRecord = 0;
+ tablerec = 0;
+ tcConnectionrec = 0;
+ tcNodeFailRecord = 0;
+
+ // Records with constant sizes
+
+ cLqhTimeOutCount = 0;
+ cLqhTimeOutCheckCount = 0;
+ cbookedAccOps = 0;
+}//Dblqh::initData()
+
+void Dblqh::initRecords()
+{
+ // Records with dynamic sizes
+ addFragRecord = (AddFragRecord*)allocRecord("AddFragRecord",
+ sizeof(AddFragRecord),
+ caddfragrecFileSize);
+ attrbuf = (Attrbuf*)allocRecord("Attrbuf",
+ sizeof(Attrbuf),
+ cattrinbufFileSize);
+
+ databuf = (Databuf*)allocRecord("Databuf",
+ sizeof(Databuf),
+ cdatabufFileSize);
+
+ fragrecord = (Fragrecord*)allocRecord("Fragrecord",
+ sizeof(Fragrecord),
+ cfragrecFileSize);
+
+ gcpRecord = (GcpRecord*)allocRecord("GcpRecord",
+ sizeof(GcpRecord),
+ cgcprecFileSize);
+
+ hostRecord = (HostRecord*)allocRecord("HostRecord",
+ sizeof(HostRecord),
+ chostFileSize);
+
+ lcpRecord = (LcpRecord*)allocRecord("LcpRecord",
+ sizeof(LcpRecord),
+ clcpFileSize);
+
+ for(Uint32 i = 0; i<clcpFileSize; i++){
+ new (&lcpRecord[i])LcpRecord();
+ }
+
+ lcpLocRecord = (LcpLocRecord*)allocRecord("LcpLocRecord",
+ sizeof(LcpLocRecord),
+ clcpLocrecFileSize);
+
+ logPartRecord = (LogPartRecord*)allocRecord("LogPartRecord",
+ sizeof(LogPartRecord),
+ clogPartFileSize);
+
+ logFileRecord = (LogFileRecord*)allocRecord("LogFileRecord",
+ sizeof(LogFileRecord),
+ clogFileFileSize);
+
+ logFileOperationRecord = (LogFileOperationRecord*)
+ allocRecord("LogFileOperationRecord",
+ sizeof(LogFileOperationRecord),
+ clfoFileSize);
+
+ logPageRecord = (LogPageRecord*)allocRecord("LogPageRecord",
+ sizeof(LogPageRecord),
+ clogPageFileSize);
+
+ pageRefRecord = (PageRefRecord*)allocRecord("PageRefRecord",
+ sizeof(PageRefRecord),
+ cpageRefFileSize);
+
+ scanRecord = (ScanRecord*)allocRecord("ScanRecord",
+ sizeof(ScanRecord),
+ cscanrecFileSize);
+
+ tablerec = (Tablerec*)allocRecord("Tablerec",
+ sizeof(Tablerec),
+ ctabrecFileSize);
+
+ tcConnectionrec = (TcConnectionrec*)allocRecord("TcConnectionrec",
+ sizeof(TcConnectionrec),
+ ctcConnectrecFileSize);
+
+ m_commitAckMarkerPool.setSize(ctcConnectrecFileSize);
+ m_commitAckMarkerHash.setSize(1024);
+
+ tcNodeFailRecord = (TcNodeFailRecord*)allocRecord("TcNodeFailRecord",
+ sizeof(TcNodeFailRecord),
+ ctcNodeFailrecFileSize);
+
+ /*
+ ndbout << "FRAGREC SIZE = " << sizeof(Fragrecord) << endl;
+ ndbout << "TAB SIZE = " << sizeof(Tablerec) << endl;
+ ndbout << "GCP SIZE = " << sizeof(GcpRecord) << endl;
+ ndbout << "LCP SIZE = " << sizeof(LcpRecord) << endl;
+ ndbout << "LCPLOC SIZE = " << sizeof(LcpLocRecord) << endl;
+ ndbout << "LOGPART SIZE = " << sizeof(LogPartRecord) << endl;
+ ndbout << "LOGFILE SIZE = " << sizeof(LogFileRecord) << endl;
+ ndbout << "TC SIZE = " << sizeof(TcConnectionrec) << endl;
+ ndbout << "HOST SIZE = " << sizeof(HostRecord) << endl;
+ ndbout << "LFO SIZE = " << sizeof(LogFileOperationRecord) << endl;
+ ndbout << "PR SIZE = " << sizeof(PageRefRecord) << endl;
+ ndbout << "SCAN SIZE = " << sizeof(ScanRecord) << endl;
+*/
+
+ // Initialize BAT for interface to file system
+ NewVARIABLE* bat = allocateBat(2);
+ bat[1].WA = &logPageRecord->logPageWord[0];
+ bat[1].nrr = clogPageFileSize;
+ bat[1].ClusterSize = sizeof(LogPageRecord);
+ bat[1].bits.q = ZTWOLOG_PAGE_SIZE;
+ bat[1].bits.v = 5;
+}//Dblqh::initRecords()
+
+Dblqh::Dblqh(const class Configuration & conf):
+ SimulatedBlock(DBLQH, conf),
+ m_commitAckMarkerHash(m_commitAckMarkerPool)
+{
+ BLOCK_CONSTRUCTOR(Dblqh);
+
+ addRecSignal(GSN_PACKED_SIGNAL, &Dblqh::execPACKED_SIGNAL);
+ addRecSignal(GSN_DEBUG_SIG, &Dblqh::execDEBUG_SIG);
+ addRecSignal(GSN_ATTRINFO, &Dblqh::execATTRINFO);
+ addRecSignal(GSN_KEYINFO, &Dblqh::execKEYINFO);
+ addRecSignal(GSN_LQHKEYREQ, &Dblqh::execLQHKEYREQ);
+ addRecSignal(GSN_LQHKEYREF, &Dblqh::execLQHKEYREF);
+ addRecSignal(GSN_COMMIT, &Dblqh::execCOMMIT);
+ addRecSignal(GSN_COMPLETE, &Dblqh::execCOMPLETE);
+ addRecSignal(GSN_LQHKEYCONF, &Dblqh::execLQHKEYCONF);
+#ifdef VM_TRACE
+ addRecSignal(GSN_TESTSIG, &Dblqh::execTESTSIG);
+#endif
+ addRecSignal(GSN_LQH_RESTART_OP, &Dblqh::execLQH_RESTART_OP);
+ addRecSignal(GSN_CONTINUEB, &Dblqh::execCONTINUEB);
+ addRecSignal(GSN_START_RECREQ, &Dblqh::execSTART_RECREQ);
+ addRecSignal(GSN_START_RECCONF, &Dblqh::execSTART_RECCONF);
+ addRecSignal(GSN_EXEC_FRAGREQ, &Dblqh::execEXEC_FRAGREQ);
+ addRecSignal(GSN_EXEC_FRAGCONF, &Dblqh::execEXEC_FRAGCONF);
+ addRecSignal(GSN_EXEC_FRAGREF, &Dblqh::execEXEC_FRAGREF);
+ addRecSignal(GSN_START_EXEC_SR, &Dblqh::execSTART_EXEC_SR);
+ addRecSignal(GSN_EXEC_SRREQ, &Dblqh::execEXEC_SRREQ);
+ addRecSignal(GSN_EXEC_SRCONF, &Dblqh::execEXEC_SRCONF);
+ addRecSignal(GSN_SCAN_HBREP, &Dblqh::execSCAN_HBREP);
+
+ addRecSignal(GSN_ALTER_TAB_REQ, &Dblqh::execALTER_TAB_REQ);
+
+ // Trigger signals, transit to from TUP
+ addRecSignal(GSN_CREATE_TRIG_REQ, &Dblqh::execCREATE_TRIG_REQ);
+ addRecSignal(GSN_CREATE_TRIG_CONF, &Dblqh::execCREATE_TRIG_CONF);
+ addRecSignal(GSN_CREATE_TRIG_REF, &Dblqh::execCREATE_TRIG_REF);
+
+ addRecSignal(GSN_DROP_TRIG_REQ, &Dblqh::execDROP_TRIG_REQ);
+ addRecSignal(GSN_DROP_TRIG_CONF, &Dblqh::execDROP_TRIG_CONF);
+ addRecSignal(GSN_DROP_TRIG_REF, &Dblqh::execDROP_TRIG_REF);
+
+ addRecSignal(GSN_DUMP_STATE_ORD, &Dblqh::execDUMP_STATE_ORD);
+ addRecSignal(GSN_ACC_COM_BLOCK, &Dblqh::execACC_COM_BLOCK);
+ addRecSignal(GSN_ACC_COM_UNBLOCK, &Dblqh::execACC_COM_UNBLOCK);
+ addRecSignal(GSN_TUP_COM_BLOCK, &Dblqh::execTUP_COM_BLOCK);
+ addRecSignal(GSN_TUP_COM_UNBLOCK, &Dblqh::execTUP_COM_UNBLOCK);
+ addRecSignal(GSN_NODE_FAILREP, &Dblqh::execNODE_FAILREP);
+ addRecSignal(GSN_CHECK_LCP_STOP, &Dblqh::execCHECK_LCP_STOP);
+ addRecSignal(GSN_SEND_PACKED, &Dblqh::execSEND_PACKED);
+ addRecSignal(GSN_TUP_ATTRINFO, &Dblqh::execTUP_ATTRINFO);
+ addRecSignal(GSN_SIZEALT_REP, &Dblqh::execSIZEALT_REP);
+ addRecSignal(GSN_LQHFRAGREQ, &Dblqh::execLQHFRAGREQ);
+ addRecSignal(GSN_LQHADDATTREQ, &Dblqh::execLQHADDATTREQ);
+ addRecSignal(GSN_TUP_ADD_ATTCONF, &Dblqh::execTUP_ADD_ATTCONF);
+ addRecSignal(GSN_TUP_ADD_ATTRREF, &Dblqh::execTUP_ADD_ATTRREF);
+ addRecSignal(GSN_ACCFRAGCONF, &Dblqh::execACCFRAGCONF);
+ addRecSignal(GSN_ACCFRAGREF, &Dblqh::execACCFRAGREF);
+ addRecSignal(GSN_TUPFRAGCONF, &Dblqh::execTUPFRAGCONF);
+ addRecSignal(GSN_TUPFRAGREF, &Dblqh::execTUPFRAGREF);
+ addRecSignal(GSN_TAB_COMMITREQ, &Dblqh::execTAB_COMMITREQ);
+ addRecSignal(GSN_ACCSEIZECONF, &Dblqh::execACCSEIZECONF);
+ addRecSignal(GSN_ACCSEIZEREF, &Dblqh::execACCSEIZEREF);
+ addRecSignal(GSN_READ_NODESCONF, &Dblqh::execREAD_NODESCONF);
+ addRecSignal(GSN_READ_NODESREF, &Dblqh::execREAD_NODESREF);
+ addRecSignal(GSN_STTOR, &Dblqh::execSTTOR);
+ addRecSignal(GSN_NDB_STTOR, &Dblqh::execNDB_STTOR);
+ addRecSignal(GSN_TUPSEIZECONF, &Dblqh::execTUPSEIZECONF);
+ addRecSignal(GSN_TUPSEIZEREF, &Dblqh::execTUPSEIZEREF);
+ addRecSignal(GSN_ACCKEYCONF, &Dblqh::execACCKEYCONF);
+ addRecSignal(GSN_ACCKEYREF, &Dblqh::execACCKEYREF);
+ addRecSignal(GSN_TUPKEYCONF, &Dblqh::execTUPKEYCONF);
+ addRecSignal(GSN_TUPKEYREF, &Dblqh::execTUPKEYREF);
+ addRecSignal(GSN_ABORT, &Dblqh::execABORT);
+ addRecSignal(GSN_ABORTREQ, &Dblqh::execABORTREQ);
+ addRecSignal(GSN_COMMITREQ, &Dblqh::execCOMMITREQ);
+ addRecSignal(GSN_COMPLETEREQ, &Dblqh::execCOMPLETEREQ);
+#ifdef VM_TRACE
+ addRecSignal(GSN_MEMCHECKREQ, &Dblqh::execMEMCHECKREQ);
+#endif
+ addRecSignal(GSN_SCAN_FRAGREQ, &Dblqh::execSCAN_FRAGREQ);
+ addRecSignal(GSN_SCAN_NEXTREQ, &Dblqh::execSCAN_NEXTREQ);
+ addRecSignal(GSN_ACC_SCANCONF, &Dblqh::execACC_SCANCONF);
+ addRecSignal(GSN_ACC_SCANREF, &Dblqh::execACC_SCANREF);
+ addRecSignal(GSN_NEXT_SCANCONF, &Dblqh::execNEXT_SCANCONF);
+ addRecSignal(GSN_NEXT_SCANREF, &Dblqh::execNEXT_SCANREF);
+ addRecSignal(GSN_ACC_SCAN_INFO, &Dblqh::execACC_SCAN_INFO);
+ addRecSignal(GSN_ACC_SCAN_INFO24, &Dblqh::execACC_SCAN_INFO24);
+ addRecSignal(GSN_STORED_PROCCONF, &Dblqh::execSTORED_PROCCONF);
+ addRecSignal(GSN_STORED_PROCREF, &Dblqh::execSTORED_PROCREF);
+ addRecSignal(GSN_COPY_FRAGREQ, &Dblqh::execCOPY_FRAGREQ);
+ addRecSignal(GSN_COPY_ACTIVEREQ, &Dblqh::execCOPY_ACTIVEREQ);
+ addRecSignal(GSN_COPY_STATEREQ, &Dblqh::execCOPY_STATEREQ);
+ addRecSignal(GSN_LQH_TRANSREQ, &Dblqh::execLQH_TRANSREQ);
+ addRecSignal(GSN_TRANSID_AI, &Dblqh::execTRANSID_AI);
+ addRecSignal(GSN_INCL_NODEREQ, &Dblqh::execINCL_NODEREQ);
+ addRecSignal(GSN_ACC_LCPCONF, &Dblqh::execACC_LCPCONF);
+ addRecSignal(GSN_ACC_LCPREF, &Dblqh::execACC_LCPREF);
+ addRecSignal(GSN_ACC_LCPSTARTED, &Dblqh::execACC_LCPSTARTED);
+ addRecSignal(GSN_ACC_CONTOPCONF, &Dblqh::execACC_CONTOPCONF);
+ addRecSignal(GSN_LCP_FRAGIDCONF, &Dblqh::execLCP_FRAGIDCONF);
+ addRecSignal(GSN_LCP_FRAGIDREF, &Dblqh::execLCP_FRAGIDREF);
+ addRecSignal(GSN_LCP_HOLDOPCONF, &Dblqh::execLCP_HOLDOPCONF);
+ addRecSignal(GSN_LCP_HOLDOPREF, &Dblqh::execLCP_HOLDOPREF);
+ addRecSignal(GSN_TUP_PREPLCPCONF, &Dblqh::execTUP_PREPLCPCONF);
+ addRecSignal(GSN_TUP_PREPLCPREF, &Dblqh::execTUP_PREPLCPREF);
+ addRecSignal(GSN_TUP_LCPCONF, &Dblqh::execTUP_LCPCONF);
+ addRecSignal(GSN_TUP_LCPREF, &Dblqh::execTUP_LCPREF);
+ addRecSignal(GSN_TUP_LCPSTARTED, &Dblqh::execTUP_LCPSTARTED);
+ addRecSignal(GSN_END_LCPCONF, &Dblqh::execEND_LCPCONF);
+
+ addRecSignal(GSN_EMPTY_LCP_REQ, &Dblqh::execEMPTY_LCP_REQ);
+ addRecSignal(GSN_LCP_FRAG_ORD, &Dblqh::execLCP_FRAG_ORD);
+
+ addRecSignal(GSN_START_FRAGREQ, &Dblqh::execSTART_FRAGREQ);
+ addRecSignal(GSN_START_RECREF, &Dblqh::execSTART_RECREF);
+ addRecSignal(GSN_SR_FRAGIDCONF, &Dblqh::execSR_FRAGIDCONF);
+ addRecSignal(GSN_SR_FRAGIDREF, &Dblqh::execSR_FRAGIDREF);
+ addRecSignal(GSN_ACC_SRCONF, &Dblqh::execACC_SRCONF);
+ addRecSignal(GSN_ACC_SRREF, &Dblqh::execACC_SRREF);
+ addRecSignal(GSN_TUP_SRCONF, &Dblqh::execTUP_SRCONF);
+ addRecSignal(GSN_TUP_SRREF, &Dblqh::execTUP_SRREF);
+ addRecSignal(GSN_GCP_SAVEREQ, &Dblqh::execGCP_SAVEREQ);
+ addRecSignal(GSN_FSOPENCONF, &Dblqh::execFSOPENCONF);
+ addRecSignal(GSN_FSOPENREF, &Dblqh::execFSOPENREF);
+ addRecSignal(GSN_FSCLOSECONF, &Dblqh::execFSCLOSECONF);
+ addRecSignal(GSN_FSCLOSEREF, &Dblqh::execFSCLOSEREF);
+ addRecSignal(GSN_FSWRITECONF, &Dblqh::execFSWRITECONF);
+ addRecSignal(GSN_FSWRITEREF, &Dblqh::execFSWRITEREF);
+ addRecSignal(GSN_FSREADCONF, &Dblqh::execFSREADCONF);
+ addRecSignal(GSN_FSREADREF, &Dblqh::execFSREADREF);
+ addRecSignal(GSN_ACC_ABORTCONF, &Dblqh::execACC_ABORTCONF);
+ addRecSignal(GSN_SET_VAR_REQ, &Dblqh::execSET_VAR_REQ);
+ addRecSignal(GSN_TIME_SIGNAL, &Dblqh::execTIME_SIGNAL);
+ addRecSignal(GSN_FSSYNCCONF, &Dblqh::execFSSYNCCONF);
+ addRecSignal(GSN_FSSYNCREF, &Dblqh::execFSSYNCREF);
+ addRecSignal(GSN_REMOVE_MARKER_ORD, &Dblqh::execREMOVE_MARKER_ORD);
+
+ //addRecSignal(GSN_DROP_TAB_REQ, &Dblqh::execDROP_TAB_REQ);
+ addRecSignal(GSN_PREP_DROP_TAB_REQ, &Dblqh::execPREP_DROP_TAB_REQ);
+ addRecSignal(GSN_WAIT_DROP_TAB_REQ, &Dblqh::execWAIT_DROP_TAB_REQ);
+ addRecSignal(GSN_DROP_TAB_REQ, &Dblqh::execDROP_TAB_REQ);
+
+ addRecSignal(GSN_LQH_ALLOCREQ, &Dblqh::execLQH_ALLOCREQ);
+ addRecSignal(GSN_LQH_WRITELOG_REQ, &Dblqh::execLQH_WRITELOG_REQ);
+
+ // TUX
+ addRecSignal(GSN_TUXFRAGCONF, &Dblqh::execTUXFRAGCONF);
+ addRecSignal(GSN_TUXFRAGREF, &Dblqh::execTUXFRAGREF);
+ addRecSignal(GSN_TUX_ADD_ATTRCONF, &Dblqh::execTUX_ADD_ATTRCONF);
+ addRecSignal(GSN_TUX_ADD_ATTRREF, &Dblqh::execTUX_ADD_ATTRREF);
+
+ initData();
+}//Dblqh::Dblqh()
+
+Dblqh::~Dblqh()
+{
+ // Records with dynamic sizes
+ deallocRecord((void **)&addFragRecord, "AddFragRecord",
+ sizeof(AddFragRecord),
+ caddfragrecFileSize);
+
+ deallocRecord((void**)&attrbuf,
+ "Attrbuf",
+ sizeof(Attrbuf),
+ cattrinbufFileSize);
+
+ deallocRecord((void**)&databuf,
+ "Databuf",
+ sizeof(Databuf),
+ cdatabufFileSize);
+
+ deallocRecord((void**)&fragrecord,
+ "Fragrecord",
+ sizeof(Fragrecord),
+ cfragrecFileSize);
+
+ deallocRecord((void**)&gcpRecord,
+ "GcpRecord",
+ sizeof(GcpRecord),
+ cgcprecFileSize);
+
+ deallocRecord((void**)&hostRecord,
+ "HostRecord",
+ sizeof(HostRecord),
+ chostFileSize);
+
+ deallocRecord((void**)&lcpRecord,
+ "LcpRecord",
+ sizeof(LcpRecord),
+ clcpFileSize);
+
+ deallocRecord((void**)&lcpLocRecord,
+ "LcpLocRecord",
+ sizeof(LcpLocRecord),
+ clcpLocrecFileSize);
+
+ deallocRecord((void**)&logPartRecord,
+ "LogPartRecord",
+ sizeof(LogPartRecord),
+ clogPartFileSize);
+
+ deallocRecord((void**)&logFileRecord,
+ "LogFileRecord",
+ sizeof(LogFileRecord),
+ clogFileFileSize);
+
+ deallocRecord((void**)&logFileOperationRecord,
+ "LogFileOperationRecord",
+ sizeof(LogFileOperationRecord),
+ clfoFileSize);
+
+ deallocRecord((void**)&logPageRecord,
+ "LogPageRecord",
+ sizeof(LogPageRecord),
+ clogPageFileSize);
+
+ deallocRecord((void**)&pageRefRecord,
+ "PageRefRecord",
+ sizeof(PageRefRecord),
+ cpageRefFileSize);
+
+ deallocRecord((void**)&scanRecord,
+ "ScanRecord",
+ sizeof(ScanRecord),
+ cscanrecFileSize);
+
+ deallocRecord((void**)&tablerec,
+ "Tablerec",
+ sizeof(Tablerec),
+ ctabrecFileSize);
+
+ deallocRecord((void**)&tcConnectionrec,
+ "TcConnectionrec",
+ sizeof(TcConnectionrec),
+ ctcConnectrecFileSize);
+
+ deallocRecord((void**)&tcNodeFailRecord,
+ "TcNodeFailRecord",
+ sizeof(TcNodeFailRecord),
+ ctcNodeFailrecFileSize);
+}//Dblqh::~Dblqh()
+
+BLOCK_FUNCTIONS(Dblqh);
+
diff --git a/ndb/src/kernel/blocks/dblqh/DblqhMain.cpp b/ndb/src/kernel/blocks/dblqh/DblqhMain.cpp
new file mode 100644
index 00000000000..2596be468bc
--- /dev/null
+++ b/ndb/src/kernel/blocks/dblqh/DblqhMain.cpp
@@ -0,0 +1,18014 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBLQH_C
+#include "Dblqh.hpp"
+#include <ndb_limits.h>
+#include <md5_hash.hpp>
+
+#include <ndb_version.h>
+#include <signaldata/TuxBound.hpp>
+#include <signaldata/AccScan.hpp>
+#include <signaldata/CopyActive.hpp>
+#include <signaldata/CopyFrag.hpp>
+#include <signaldata/CreateTrig.hpp>
+#include <signaldata/DropTrig.hpp>
+#include <signaldata/EmptyLcp.hpp>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/ExecFragReq.hpp>
+#include <signaldata/GCPSave.hpp>
+#include <signaldata/TcKeyRef.hpp>
+#include <signaldata/LqhKey.hpp>
+#include <signaldata/LqhSizeAltReq.hpp>
+#include <signaldata/NextScan.hpp>
+#include <signaldata/NFCompleteRep.hpp>
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/ReadNodesConf.hpp>
+#include <signaldata/RelTabMem.hpp>
+#include <signaldata/ScanFrag.hpp>
+#include <signaldata/SetVarReq.hpp>
+#include <signaldata/SrFragidConf.hpp>
+#include <signaldata/StartFragReq.hpp>
+#include <signaldata/StartRec.hpp>
+#include <signaldata/TupKey.hpp>
+#include <signaldata/TupCommit.hpp>
+#include <signaldata/LqhFrag.hpp>
+#include <signaldata/AccFrag.hpp>
+#include <signaldata/TupFrag.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+#include <signaldata/PackedSignal.hpp>
+
+#include <signaldata/PrepDropTab.hpp>
+#include <signaldata/DropTab.hpp>
+
+#include <signaldata/AlterTab.hpp>
+
+#include <signaldata/LCP.hpp>
+
+// Use DEBUG to print messages that should be
+// seen only when we debug the product
+#ifdef VM_TRACE
+#define DEBUG(x) ndbout << "DBLQH: "<< x << endl;
+#else
+#define DEBUG(x)
+#endif
+
+const Uint32 NR_ScanNo = MAX_PARALLEL_SCANS_PER_FRAG - 1;
+const Uint32 NR_MinRangeScanNo = MAX_PARALLEL_SCANS_PER_FRAG;
+const Uint32 NR_MaxRangeScanNo = NR_MinRangeScanNo + MAX_PARALLEL_INDEX_SCANS_PER_FRAG;
+
+void Dblqh::execACC_COM_BLOCK(Signal* signal)
+{
+ jamEntry();
+/* ------------------------------------------------------------------------- */
+// Undo log buffer in ACC is in critical sector of being full.
+/* ------------------------------------------------------------------------- */
+ cCounterAccCommitBlocked++;
+ caccCommitBlocked = true;
+ cCommitBlocked = true;
+ return;
+}//Dblqh::execACC_COM_BLOCK()
+
+void Dblqh::execACC_COM_UNBLOCK(Signal* signal)
+{
+ jamEntry();
+/* ------------------------------------------------------------------------- */
+// Undo log buffer in ACC ok again.
+/* ------------------------------------------------------------------------- */
+ caccCommitBlocked = false;
+ if (ctupCommitBlocked == false) {
+ jam();
+ cCommitBlocked = false;
+ }//if
+ return;
+}//Dblqh::execACC_COM_UNBLOCK()
+
+void Dblqh::execTUP_COM_BLOCK(Signal* signal)
+{
+ jamEntry();
+/* ------------------------------------------------------------------------- */
+// Undo log buffer in TUP is in critical sector of being full.
+/* ------------------------------------------------------------------------- */
+ cCounterTupCommitBlocked++;
+ ctupCommitBlocked = true;
+ cCommitBlocked = true;
+ return;
+}//Dblqh::execTUP_COM_BLOCK()
+
+void Dblqh::execTUP_COM_UNBLOCK(Signal* signal)
+{
+ jamEntry();
+/* ------------------------------------------------------------------------- */
+// Undo log buffer in TUP ok again.
+/* ------------------------------------------------------------------------- */
+ ctupCommitBlocked = false;
+ if (caccCommitBlocked == false) {
+ jam();
+ cCommitBlocked = false;
+ }//if
+ return;
+}//Dblqh::execTUP_COM_UNBLOCK()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEND SYSTEM ERROR ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::systemError(Signal* signal)
+{
+ progError(0, 0);
+}//Dblqh::systemError()
+
+/* *************** */
+/* ACCSEIZEREF > */
+/* *************** */
+void Dblqh::execACCSEIZEREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execACCSEIZEREF()
+
+/* ******************************************************>> */
+/* THIS SIGNAL IS USED TO HANDLE REAL-TIME */
+/* BREAKS THAT ARE NECESSARY TO ENSURE REAL-TIME */
+/* OPERATION OF LQH. */
+/* This signal is also used for signal loops, for example */
+/* the timeout handling for writing logs every second. */
+/* ******************************************************>> */
+void Dblqh::execCONTINUEB(Signal* signal)
+{
+ jamEntry();
+ Uint32 tcase = signal->theData[0];
+ Uint32 data0 = signal->theData[1];
+ Uint32 data1 = signal->theData[2];
+ Uint32 data2 = signal->theData[3];
+#if 0
+ if (tcase == RNIL) {
+ tcConnectptr.i = data0;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ ndbout << "State = " << tcConnectptr.p->transactionState;
+ ndbout << " seqNoReplica = " << tcConnectptr.p->seqNoReplica;
+ ndbout << " tcNodeFailrec = " << tcConnectptr.p->tcNodeFailrec;
+ ndbout << " activeCreat = " << tcConnectptr.p->activeCreat;
+ ndbout << endl;
+ ndbout << "tupkeyData0 = " << tcConnectptr.p->tupkeyData[0];
+ ndbout << "tupkeyData1 = " << tcConnectptr.p->tupkeyData[1];
+ ndbout << "tupkeyData2 = " << tcConnectptr.p->tupkeyData[2];
+ ndbout << "tupkeyData3 = " << tcConnectptr.p->tupkeyData[3];
+ ndbout << endl;
+ ndbout << "abortState = " << tcConnectptr.p->abortState;
+ ndbout << "listState = " << tcConnectptr.p->listState;
+ ndbout << endl;
+ return;
+ }//if
+#endif
+ switch (tcase) {
+ case ZLOG_LQHKEYREQ:
+ if (cnoOfLogPages == 0) {
+ jam();
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 10, 2);
+ return;
+ }//if
+ logPartPtr.i = data0;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ logFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPagePtr.i = logFilePtr.p->currentLogpage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+
+ tcConnectptr.i = logPartPtr.p->firstLogQueue;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if ((cCommitBlocked == true) &&
+ (fragptr.p->fragActiveStatus == ZTRUE)) {
+ jam();
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 10, 2);
+ return;
+ }//if
+ logPartPtr.p->LogLqhKeyReqSent = ZFALSE;
+ getFirstInLogQueue(signal);
+
+ switch (tcConnectptr.p->transactionState) {
+ case TcConnectionrec::LOG_QUEUED:
+ if (tcConnectptr.p->abortState != TcConnectionrec::ABORT_IDLE) {
+ jam();
+ logNextStart(signal);
+ abortCommonLab(signal);
+ return;
+ } else {
+ jam();
+/*------------------------------------------------------------*/
+/* WE MUST SET THE STATE OF THE LOG PART TO IDLE TO */
+/* ENSURE THAT WE ARE NOT QUEUED AGAIN ON THE LOG PART */
+/* WE WILL SET THE LOG PART STATE TO ACTIVE IMMEDIATELY */
+/* SO NO OTHER PROCESS WILL SEE THIS STATE. IT IS MERELY*/
+/* USED TO ENABLE REUSE OF CODE. */
+/*------------------------------------------------------------*/
+ if (logPartPtr.p->logPartState == LogPartRecord::ACTIVE) {
+ jam();
+ logPartPtr.p->logPartState = LogPartRecord::IDLE;
+ }//if
+ logLqhkeyreqLab(signal);
+ return;
+ }//if
+ break;
+ case TcConnectionrec::LOG_ABORT_QUEUED:
+ jam();
+ writeAbortLog(signal);
+ removeLogTcrec(signal);
+ logNextStart(signal);
+ continueAfterLogAbortWriteLab(signal);
+ return;
+ break;
+ case TcConnectionrec::LOG_COMMIT_QUEUED:
+ case TcConnectionrec::LOG_COMMIT_QUEUED_WAIT_SIGNAL:
+ jam();
+ writeCommitLog(signal, logPartPtr);
+ logNextStart(signal);
+ if (tcConnectptr.p->transactionState == TcConnectionrec::LOG_COMMIT_QUEUED) {
+ if (tcConnectptr.p->seqNoReplica != 0) {
+ jam();
+ commitReplyLab(signal);
+ } else {
+ jam();
+ localCommitLab(signal);
+ }//if
+ return;
+ } else {
+ jam();
+ tcConnectptr.p->transactionState = TcConnectionrec::LOG_COMMIT_WRITTEN_WAIT_SIGNAL;
+ return;
+ }//if
+ break;
+ case TcConnectionrec::COMMIT_QUEUED:
+ jam();
+ logNextStart(signal);
+ localCommitLab(signal);
+ break;
+ case TcConnectionrec::ABORT_QUEUED:
+ jam();
+ logNextStart(signal);
+ abortCommonLab(signal);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+ break;
+ case ZSR_GCI_LIMITS:
+ jam();
+ signal->theData[0] = data0;
+ srGciLimits(signal);
+ return;
+ break;
+ case ZSR_LOG_LIMITS:
+ jam();
+ signal->theData[0] = data0;
+ signal->theData[1] = data1;
+ signal->theData[2] = data2;
+ srLogLimits(signal);
+ return;
+ break;
+ case ZSEND_EXEC_CONF:
+ jam();
+ signal->theData[0] = data0;
+ sendExecConf(signal);
+ return;
+ break;
+ case ZEXEC_SR:
+ jam();
+ signal->theData[0] = data0;
+ execSr(signal);
+ return;
+ break;
+ case ZSR_FOURTH_COMP:
+ jam();
+ signal->theData[0] = data0;
+ srFourthComp(signal);
+ return;
+ break;
+ case ZINIT_FOURTH:
+ jam();
+ signal->theData[0] = data0;
+ initFourth(signal);
+ return;
+ break;
+ case ZTIME_SUPERVISION:
+ jam();
+ signal->theData[0] = data0;
+ timeSup(signal);
+ return;
+ break;
+ case ZSR_PHASE3_START:
+ jam();
+ signal->theData[0] = data0;
+ srPhase3Start(signal);
+ return;
+ break;
+ case ZLQH_TRANS_NEXT:
+ jam();
+ tcNodeFailptr.i = data0;
+ ptrCheckGuard(tcNodeFailptr, ctcNodeFailrecFileSize, tcNodeFailRecord);
+ lqhTransNextLab(signal);
+ return;
+ break;
+ case ZSCAN_TC_CONNECT:
+ jam();
+ tabptr.i = data1;
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ scanTcConnectLab(signal, data0, data2);
+ return;
+ break;
+ case ZINITIALISE_RECORDS:
+ jam();
+ initialiseRecordsLab(signal, data0);
+ return;
+ break;
+ case ZINIT_GCP_REC:
+ jam();
+ gcpPtr.i = 0;
+ ptrAss(gcpPtr, gcpRecord);
+ initGcpRecLab(signal);
+ return;
+ break;
+ case ZRESTART_OPERATIONS_AFTER_STOP:
+ jam();
+ tcConnectptr.i = data0;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ if (tcConnectptr.p->listState != TcConnectionrec::WAIT_QUEUE_LIST) {
+ jam();
+ return;
+ }//if
+ releaseWaitQueue(signal);
+ linkActiveFrag(signal);
+ restartOperationsAfterStopLab(signal);
+ return;
+ break;
+ case ZCHECK_LCP_STOP_BLOCKED:
+ jam();
+ scanptr.i = data0;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ checkLcpStopBlockedLab(signal);
+ return;
+ case ZSCAN_MARKERS:
+ jam();
+ scanMarkers(signal, data0, data1, data2);
+ return;
+ break;
+
+ case ZOPERATION_EVENT_REP:
+ jam();
+ /* --------------------------------------------------------------------- */
+ // Report information about transaction activity once per second.
+ /* --------------------------------------------------------------------- */
+ if (signal->theData[1] == 0) {
+ signal->theData[0] = EventReport::OperationReportCounters;
+ signal->theData[1] = c_Counters.operations;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+ }//if
+ c_Counters.clear();
+ signal->theData[0] = ZOPERATION_EVENT_REP;
+ signal->theData[1] = 0;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 5000, 2);
+ break;
+ case ZPREP_DROP_TABLE:
+ jam();
+ checkDropTab(signal);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+}//Dblqh::execCONTINUEB()
+
+/* *********************************************************> */
+/* Request from DBDIH to include a new node in the node list */
+/* and so forth. */
+/* *********************************************************> */
+void Dblqh::execINCL_NODEREQ(Signal* signal)
+{
+ jamEntry();
+ BlockReference retRef = signal->theData[0];
+ Uint32 nodeId = signal->theData[1];
+ cnewestGci = signal->theData[2];
+ cnewestCompletedGci = signal->theData[2] - 1;
+ ndbrequire(cnoOfNodes < MAX_NDB_NODES);
+ for (Uint32 i = 0; i < cnoOfNodes; i++) {
+ jam();
+ if (cnodeData[i] == nodeId) {
+ jam();
+ cnodeStatus[i] = ZNODE_UP;
+ }//if
+ }//for
+ signal->theData[0] = cownref;
+ sendSignal(retRef, GSN_INCL_NODECONF, signal, 1, JBB);
+ return;
+}//Dblqh::execINCL_NODEREQ()
+
+void Dblqh::execTUPSEIZEREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execTUPSEIZEREF()
+
+/* ########################################################################## */
+/* ####### START / RESTART MODULE ####### */
+/* ########################################################################## */
+/* ************************************************************************>> */
+/* This is first signal that arrives in a start / restart. Sender is NDBCNTR_REF. */
+/* ************************************************************************>> */
+void Dblqh::execSTTOR(Signal* signal)
+{
+ UintR tstartPhase;
+
+ jamEntry();
+ /* START CASE */
+ tstartPhase = signal->theData[1];
+ /* SYSTEM RESTART RANK */
+ csignalKey = signal->theData[6];
+ switch (tstartPhase) {
+ case ZSTART_PHASE1:
+ jam();
+ cstartPhase = tstartPhase;
+ sttorStartphase1Lab(signal);
+ return;
+ break;
+ default:
+ jam();
+ /*empty*/;
+ sendsttorryLab(signal);
+ return;
+ break;
+ }//switch
+}//Dblqh::execSTTOR()
+
+/* ***************************************> */
+/* Restart phases 1 - 6, sender is Ndbcntr */
+/* ***************************************> */
+void Dblqh::execNDB_STTOR(Signal* signal)
+{
+ jamEntry();
+ Uint32 ownNodeId = signal->theData[1]; /* START PHASE*/
+ cstartPhase = signal->theData[2]; /* MY NODE ID */
+ cstartType = signal->theData[3]; /* START TYPE */
+ Uint32 config1 = signal->theData[10]; /* CONFIG INFO LQH */
+
+ switch (cstartPhase) {
+ case ZSTART_PHASE1:
+ jam();
+ preComputedRequestInfoMask = 0;
+ LqhKeyReq::setKeyLen(preComputedRequestInfoMask, RI_KEYLEN_MASK);
+ LqhKeyReq::setLastReplicaNo(preComputedRequestInfoMask, RI_LAST_REPL_MASK);
+ LqhKeyReq::setLockType(preComputedRequestInfoMask, RI_LOCK_TYPE_MASK);
+ // Dont LqhKeyReq::setApplicationAddressFlag
+ LqhKeyReq::setDirtyFlag(preComputedRequestInfoMask, 1);
+ // Dont LqhKeyReq::setInterpretedFlag
+ LqhKeyReq::setSimpleFlag(preComputedRequestInfoMask, 1);
+ LqhKeyReq::setOperation(preComputedRequestInfoMask, RI_OPERATION_MASK);
+ // Dont setAIInLqhKeyReq
+ // Dont setSeqNoReplica
+ // Dont setSameClientAndTcFlag
+ // Dont setReturnedReadLenAIFlag
+ // Dont setAPIVersion
+ LqhKeyReq::setMarkerFlag(preComputedRequestInfoMask, 1);
+ //preComputedRequestInfoMask = 0x003d7fff;
+ startphase1Lab(signal, config1, ownNodeId);
+
+ signal->theData[0] = ZOPERATION_EVENT_REP;
+ signal->theData[1] = 1;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 10, 2);
+ return;
+ break;
+ case ZSTART_PHASE2:
+ jam();
+ startphase2Lab(signal, config1);
+ return;
+ break;
+ case ZSTART_PHASE3:
+ jam();
+ startphase3Lab(signal);
+ return;
+ break;
+ case ZSTART_PHASE4:
+ jam();
+ startphase4Lab(signal);
+ return;
+ break;
+ case ZSTART_PHASE6:
+ jam();
+ startphase6Lab(signal);
+ return;
+ break;
+ default:
+ jam();
+ /*empty*/;
+ sendNdbSttorryLab(signal);
+ return;
+ break;
+ }//switch
+}//Dblqh::execNDB_STTOR()
+
+/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+/* +++++++ START PHASE 1 +++++++ */
+/* LOAD OUR BLOCK REFERENCE AND OUR PROCESSOR ID */
+/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+void Dblqh::sttorStartphase1Lab(Signal* signal)
+{
+ sendsttorryLab(signal);
+ return;
+}//Dblqh::sttorStartphase1Lab()
+
+/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+/* +++++++ START PHASE 2 +++++++ */
+/* */
+/* INITIATE ALL RECORDS WITHIN THE BLOCK */
+/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+void Dblqh::startphase1Lab(Signal* signal, Uint32 config, Uint32 ownNodeId)
+{
+ UintR Ti;
+ HostRecordPtr ThostPtr;
+
+/* ------- INITIATE ALL RECORDS ------- */
+ if (config == 0) {
+ jam();
+ config = 1;
+ }//if
+ cnoLogFiles = config;
+ cownNodeid = ownNodeId;
+ caccBlockref = calcAccBlockRef (cownNodeid);
+ ctupBlockref = calcTupBlockRef (cownNodeid);
+ ctuxBlockref = calcTuxBlockRef (cownNodeid);
+ cownref = calcLqhBlockRef (cownNodeid);
+ for (Ti = 0; Ti < chostFileSize; Ti++) {
+ ThostPtr.i = Ti;
+ ptrCheckGuard(ThostPtr, chostFileSize, hostRecord);
+ ThostPtr.p->hostLqhBlockRef = calcLqhBlockRef(ThostPtr.i);
+ ThostPtr.p->hostTcBlockRef = calcTcBlockRef(ThostPtr.i);
+ ThostPtr.p->inPackedList = false;
+ ThostPtr.p->noOfPackedWordsLqh = 0;
+ ThostPtr.p->noOfPackedWordsTc = 0;
+ }//for
+ cpackedListIndex = 0;
+ sendNdbSttorryLab(signal);
+ return;
+}//Dblqh::startphase1Lab()
+
+/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+/* +++++++ START PHASE 2 +++++++ */
+/* */
+/* CONNECT LQH WITH ACC AND TUP. */
+/* EVERY CONNECTION RECORD IN LQH IS ASSIGNED TO ONE ACC CONNECTION RECORD */
+/* AND ONE TUP CONNECTION RECORD. */
+/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+void Dblqh::startphase2Lab(Signal* signal, Uint32 config)
+{
+ if (config == 0) {
+ jam();
+ config = 1;
+ } else if (config > 4) {
+ jam();
+ config = 4;
+ }//if
+ cmaxWordsAtNodeRec = MAX_NO_WORDS_OUTSTANDING_COPY_FRAGMENT;
+/* -- ACC AND TUP CONNECTION PROCESS -- */
+ tcConnectptr.i = 0;
+ ptrAss(tcConnectptr, tcConnectionrec);
+ moreconnectionsLab(signal);
+ return;
+}//Dblqh::startphase2Lab()
+
+void Dblqh::moreconnectionsLab(Signal* signal)
+{
+ tcConnectptr.p->tcAccBlockref = caccBlockref;
+ // set TUX block here (no operation is seized in TUX)
+ tcConnectptr.p->tcTuxBlockref = ctuxBlockref;
+/* NO STATE CHECKING IS PERFORMED, ASSUMED TO WORK */
+/* *************** */
+/* ACCSEIZEREQ < */
+/* *************** */
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = cownref;
+ sendSignal(caccBlockref, GSN_ACCSEIZEREQ, signal, 2, JBB);
+ return;
+}//Dblqh::moreconnectionsLab()
+
+/* ***************> */
+/* ACCSEIZECONF > */
+/* ***************> */
+void Dblqh::execACCSEIZECONF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ tcConnectptr.p->accConnectrec = signal->theData[1];
+/* *************** */
+/* TUPSEIZEREQ < */
+/* *************** */
+ tcConnectptr.p->tcTupBlockref = ctupBlockref;
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = cownref;
+ sendSignal(ctupBlockref, GSN_TUPSEIZEREQ, signal, 2, JBB);
+ return;
+}//Dblqh::execACCSEIZECONF()
+
+/* ***************> */
+/* TUPSEIZECONF > */
+/* ***************> */
+void Dblqh::execTUPSEIZECONF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ tcConnectptr.p->tupConnectrec = signal->theData[1];
+/* ------- CHECK IF THERE ARE MORE CONNECTIONS TO BE CONNECTED ------- */
+ tcConnectptr.i = tcConnectptr.p->nextTcConnectrec;
+ if (tcConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ moreconnectionsLab(signal);
+ return;
+ }//if
+/* ALL LQH_CONNECT RECORDS ARE CONNECTED TO ACC AND TUP ---- */
+ sendNdbSttorryLab(signal);
+ return;
+}//Dblqh::execTUPSEIZECONF()
+
+/* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+/* +++++++ START PHASE 4 +++++++ */
+/* */
+/* CONNECT LQH WITH LQH. */
+/* CONNECT EACH LQH WITH EVERY LQH IN THE DATABASE SYSTEM. */
+/* IF INITIAL START THEN CREATE THE FRAGMENT LOG FILES */
+/*IF SYSTEM RESTART OR NODE RESTART THEN OPEN THE FRAGMENT LOG FILES AND */
+/*FIND THE END OF THE LOG FILES. */
+/* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+/* WAIT UNTIL ADD NODE PROCESSES ARE COMPLETED */
+/* IF INITIAL START ALSO WAIT FOR LOG FILES TO INITIALISED */
+/*START TIME SUPERVISION OF LOG FILES. WE HAVE TO WRITE LOG PAGES TO DISK */
+/*EVEN IF THE PAGES ARE NOT FULL TO ENSURE THAT THEY COME TO DISK ASAP. */
+/* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+void Dblqh::startphase3Lab(Signal* signal)
+{
+ LogFileRecordPtr prevLogFilePtr;
+ LogFileRecordPtr zeroLogFilePtr;
+
+ caddNodeState = ZTRUE;
+/* ***************<< */
+/* READ_NODESREQ < */
+/* ***************<< */
+ cinitialStartOngoing = ZTRUE;
+ ndbrequire(cnoLogFiles != 0);
+
+ for (logPartPtr.i = 0; logPartPtr.i < 4; logPartPtr.i++) {
+ jam();
+ ptrAss(logPartPtr, logPartRecord);
+ initLogpart(signal);
+ for (Uint32 fileNo = 0; fileNo < cnoLogFiles; fileNo++) {
+ seizeLogfile(signal);
+ if (fileNo != 0) {
+ jam();
+ prevLogFilePtr.p->nextLogFile = logFilePtr.i;
+ logFilePtr.p->prevLogFile = prevLogFilePtr.i;
+ } else {
+ jam();
+ logPartPtr.p->firstLogfile = logFilePtr.i;
+ logPartPtr.p->currentLogfile = logFilePtr.i;
+ zeroLogFilePtr.i = logFilePtr.i;
+ zeroLogFilePtr.p = logFilePtr.p;
+ }//if
+ prevLogFilePtr.i = logFilePtr.i;
+ prevLogFilePtr.p = logFilePtr.p;
+ initLogfile(signal, fileNo);
+ if ((cstartType == NodeState::ST_INITIAL_START) ||
+ (cstartType == NodeState::ST_INITIAL_NODE_RESTART)) {
+ if (logFilePtr.i == zeroLogFilePtr.i) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*IN AN INITIAL START WE START BY CREATING ALL LOG FILES AND SETTING THEIR */
+/*PROPER SIZE AND INITIALISING PAGE ZERO IN ALL FILES. */
+/*WE START BY CREATING FILE ZERO IN EACH LOG PART AND THEN PROCEED */
+/*SEQUENTIALLY THROUGH ALL LOG FILES IN THE LOG PART. */
+/* ------------------------------------------------------------------------- */
+ openLogfileInit(signal);
+ }//if
+ }//if
+ }//for
+ zeroLogFilePtr.p->prevLogFile = logFilePtr.i;
+ logFilePtr.p->nextLogFile = zeroLogFilePtr.i;
+ }//for
+ if (cstartType != NodeState::ST_INITIAL_START &&
+ cstartType != NodeState::ST_INITIAL_NODE_RESTART) {
+ jam();
+ ndbrequire(cstartType == NodeState::ST_NODE_RESTART ||
+ cstartType == NodeState::ST_SYSTEM_RESTART);
+ /** --------------------------------------------------------------------
+ * THIS CODE KICKS OFF THE SYSTEM RESTART AND NODE RESTART. IT STARTS UP
+ * THE RESTART BY FINDING THE END OF THE LOG AND FROM THERE FINDING THE
+ * INFO ABOUT THE GLOBAL CHECKPOINTS IN THE FRAGMENT LOG.
+ --------------------------------------------------------------------- */
+ for (logPartPtr.i = 0; logPartPtr.i < 4; logPartPtr.i++) {
+ jam();
+ LogFileRecordPtr locLogFilePtr;
+ ptrAss(logPartPtr, logPartRecord);
+ locLogFilePtr.i = logPartPtr.p->firstLogfile;
+ ptrCheckGuard(locLogFilePtr, clogFileFileSize, logFileRecord);
+ locLogFilePtr.p->logFileStatus = LogFileRecord::OPEN_SR_FRONTPAGE;
+ openFileRw(signal, locLogFilePtr);
+ }//for
+ }//if
+
+ signal->theData[0] = cownref;
+ sendSignal(NDBCNTR_REF, GSN_READ_NODESREQ, signal, 1, JBB);
+ return;
+}//Dblqh::startphase3Lab()
+
+/* ****************** */
+/* READ_NODESCONF > */
+/* ****************** */
+void Dblqh::execREAD_NODESCONF(Signal* signal)
+{
+ jamEntry();
+
+ ReadNodesConf * const readNodes = (ReadNodesConf *)&signal->theData[0];
+ cnoOfNodes = readNodes->noOfNodes;
+
+ unsigned ind = 0;
+ unsigned i = 0;
+ for (i = 1; i < MAX_NDB_NODES; i++) {
+ jam();
+ if (NodeBitmask::get(readNodes->allNodes, i)) {
+ jam();
+ cnodeData[ind] = i;
+ cnodeStatus[ind] = NodeBitmask::get(readNodes->inactiveNodes, i);
+ //readNodes->getVersionId(i, readNodes->theVersionIds) not used
+ ind++;
+ }//if
+ }//for
+ ndbrequire(ind == cnoOfNodes);
+ ndbrequire(cnoOfNodes >= 1 && cnoOfNodes < MAX_NDB_NODES);
+ ndbrequire(!(cnoOfNodes == 1 && cstartType == NodeState::ST_NODE_RESTART));
+
+ caddNodeState = ZFALSE;
+ if (cstartType == NodeState::ST_SYSTEM_RESTART) {
+ jam();
+ sendNdbSttorryLab(signal);
+ return;
+ }//if
+ checkStartCompletedLab(signal);
+ return;
+}//Dblqh::execREAD_NODESCONF()
+
+void Dblqh::checkStartCompletedLab(Signal* signal)
+{
+ if (caddNodeState == ZFALSE) {
+ if (cinitialStartOngoing == ZFALSE) {
+ jam();
+ sendNdbSttorryLab(signal);
+ return;
+ }//if
+ }//if
+ return;
+}//Dblqh::checkStartCompletedLab()
+
+void Dblqh::startphase4Lab(Signal* signal)
+{
+ sendNdbSttorryLab(signal);
+ return;
+}//Dblqh::startphase4Lab()
+
+/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+/* SET CONCURRENCY OF LOCAL CHECKPOINTS TO BE USED AFTER SYSTEM RESTART. */
+/* ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+void Dblqh::startphase6Lab(Signal* signal)
+{
+ cstartPhase = ZNIL;
+ cstartType = ZNIL;
+ sendNdbSttorryLab(signal);
+ return;
+}//Dblqh::startphase6Lab()
+
+void Dblqh::sendNdbSttorryLab(Signal* signal)
+{
+ signal->theData[0] = cownref;
+ sendSignal(NDBCNTR_REF, GSN_NDB_STTORRY, signal, 1, JBB);
+ return;
+}//Dblqh::sendNdbSttorryLab()
+
+void Dblqh::sendsttorryLab(Signal* signal)
+{
+/* *********<< */
+/* STTORRY < */
+/* *********<< */
+ signal->theData[0] = csignalKey; /* SIGNAL KEY */
+ signal->theData[1] = 3; /* BLOCK CATEGORY */
+ signal->theData[2] = 2; /* SIGNAL VERSION NUMBER */
+ signal->theData[3] = ZSTART_PHASE1;
+ signal->theData[4] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 5, JBB);
+ return;
+}//Dblqh::sendsttorryLab()
+
+/* ***************>> */
+/* READ_NODESREF > */
+/* ***************>> */
+void Dblqh::execREAD_NODESREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execREAD_NODESREF()
+
+/* *************** */
+/* SIZEALT_REP > */
+/* *************** */
+void Dblqh::execSIZEALT_REP(Signal* signal)
+{
+ jamEntry();
+ cfragrecFileSize = signal->theData[LqhSizeAltReq::IND_FRAG];
+ ctabrecFileSize = signal->theData[LqhSizeAltReq::IND_TABLE];
+ ctcConnectrecFileSize = signal->theData[LqhSizeAltReq::IND_TC_CONNECT];
+ clogFileFileSize = signal->theData[LqhSizeAltReq::IND_LOG_FILES];
+ cscanrecFileSize = signal->theData[LqhSizeAltReq::IND_SCAN];
+ cmaxAccOps = cscanrecFileSize * MAX_PARALLEL_SCANS_PER_FRAG;
+ initRecords();
+ initialiseRecordsLab(signal, 0);
+
+ return;
+}//Dblqh::execSIZEALT_REP()
+
+void Dblqh::returnInitialiseRecordsLab(Signal* signal)
+{
+ signal->theData[0] = DBLQH_REF;
+ sendSignal(CMVMI_REF, GSN_SIZEALT_ACK, signal, 2, JBB);
+
+
+ return;
+}//Dblqh::returnInitialiseRecordsLab()
+
+/* ########################################################################## */
+/* ####### ADD/DELETE FRAGMENT MODULE ####### */
+/* THIS MODULE IS USED BY DICTIONARY TO CREATE NEW FRAGMENTS AND DELETE */
+/* OLD FRAGMENTS. */
+/* */
+/* ########################################################################## */
+/* -------------------------------------------------------------- */
+/* FRAG REQ */
+/* -------------------------------------------------------------- */
+/* *********************************************************> */
+/* LQHFRAGREQ: Create new fragments for a table. Sender DICT */
+/* *********************************************************> */
+void Dblqh::execLQHFRAGREQ(Signal* signal)
+{
+ jamEntry();
+ LqhFragReq * req = (LqhFragReq*)signal->getDataPtr();
+
+ Uint32 retPtr = req->senderData;
+ BlockReference retRef = req->senderRef;
+ Uint32 fragId = req->fragmentId;
+ Uint32 reqinfo = req->requestInfo;
+ tabptr.i = req->tableId;
+ Uint16 tlocalKeylen = req->localKeyLength;
+ Uint32 tmaxLoadFactor = req->maxLoadFactor;
+ Uint32 tminLoadFactor = req->minLoadFactor;
+ Uint8 tk = req->kValue;
+ Uint8 tlhstar = req->lh3DistrBits;
+ Uint8 tlh = req->lh3PageBits;
+ Uint32 tnoOfAttr = req->noOfAttributes;
+ Uint32 tnoOfNull = req->noOfNullAttributes;
+ Uint32 noOfAlloc = req->noOfPagesToPreAllocate;
+ Uint32 tschemaVersion = req->schemaVersion;
+ Uint32 ttupKeyLength = req->keyLength;
+ Uint32 nextLcp = req->nextLCP;
+ Uint32 noOfKeyAttr = req->noOfKeyAttr;
+ Uint32 noOfNewAttr = req->noOfNewAttr;
+ Uint32 checksumIndicator = req->checksumIndicator;
+ Uint32 noOfAttributeGroups = req->noOfAttributeGroups;
+ Uint32 gcpIndicator = req->GCPIndicator;
+ Uint32 startGci = req->startGci;
+ Uint32 tableType = req->tableType;
+ Uint32 primaryTableId = req->primaryTableId;
+
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ bool tempTable = ((reqinfo & LqhFragReq::TemporaryTable) != 0);
+
+ /* Temporary tables set to defined in system restart */
+ if (tabptr.p->tableStatus == Tablerec::NOT_DEFINED){
+ tabptr.p->tableStatus = Tablerec::ADD_TABLE_ONGOING;
+ tabptr.p->tableType = tableType;
+ tabptr.p->primaryTableId = primaryTableId;
+ tabptr.p->schemaVersion = tschemaVersion;
+ }//if
+
+ if (tabptr.p->tableStatus != Tablerec::ADD_TABLE_ONGOING){
+ jam();
+ fragrefLab(signal, retRef, retPtr, ZTAB_STATE_ERROR);
+ return;
+ }//if
+ //--------------------------------------------------------------------
+ // We could arrive here if we create the fragment as part of a take
+ // over by a hot spare node. The table is then is already created
+ // and bit 31 is set, thus indicating that we are creating a fragment
+ // by copy creation. Also since the node has already been started we
+ // know that it is not a node restart ongoing.
+ //--------------------------------------------------------------------
+
+ if (getFragmentrec(signal, fragId)) {
+ jam();
+ fragrefLab(signal, retRef, retPtr, terrorCode);
+ return;
+ }//if
+ if (!insertFragrec(signal, fragId)) {
+ jam();
+ fragrefLab(signal, retRef, retPtr, terrorCode);
+ return;
+ }//if
+ Uint32 copyType = reqinfo & 3;
+ initFragrec(signal, tabptr.i, fragId, copyType);
+ fragptr.p->startGci = startGci;
+ fragptr.p->newestGci = startGci;
+ fragptr.p->tableType = tableType;
+
+ if (DictTabInfo::isOrderedIndex(tableType)) {
+ jam();
+ // find corresponding primary table fragment
+ TablerecPtr tTablePtr;
+ tTablePtr.i = primaryTableId;
+ ptrCheckGuard(tTablePtr, ctabrecFileSize, tablerec);
+ FragrecordPtr tFragPtr;
+ tFragPtr.i = RNIL;
+ for (Uint32 i = 0; i < NO_OF_FRAG_PER_NODE; i++) {
+ if (tTablePtr.p->fragid[i] == fragptr.p->fragId) {
+ jam();
+ tFragPtr.i = tTablePtr.p->fragrec[i];
+ break;
+ }
+ }
+ ndbrequire(tFragPtr.i != RNIL);
+ // store it
+ fragptr.p->tableFragptr = tFragPtr.i;
+ }
+
+ if (tempTable) {
+//--------------------------------------------
+// reqinfo bit 3-4 = 2 means temporary table
+// without logging or checkpointing.
+//--------------------------------------------
+ jam();
+ fragptr.p->logFlag = Fragrecord::STATE_FALSE;
+ fragptr.p->lcpFlag = Fragrecord::LCP_STATE_FALSE;
+ }//if
+
+ fragptr.p->nextLcp = nextLcp;
+//----------------------------------------------
+// For node restarts it is not necessarily zero
+//----------------------------------------------
+ if (cfirstfreeAddfragrec == RNIL) {
+ jam();
+ deleteFragrec(fragId);
+ fragrefLab(signal, retRef, retPtr, ZNO_ADD_FRAGREC);
+ return;
+ }//if
+ seizeAddfragrec(signal);
+ addfragptr.p->addFragid = fragId;
+ addfragptr.p->fragmentPtr = fragptr.i;
+ addfragptr.p->dictBlockref = retRef;
+ addfragptr.p->dictConnectptr = retPtr;
+ addfragptr.p->m_senderAttrPtr = RNIL;
+ addfragptr.p->noOfAttr = tnoOfAttr;
+ addfragptr.p->noOfNull = tnoOfNull;
+ addfragptr.p->noOfAllocPages = noOfAlloc;
+ addfragptr.p->tabId = tabptr.i;
+ addfragptr.p->totalAttrReceived = 0;
+ addfragptr.p->attrSentToTup = ZNIL;/* TO FIND PROGRAMMING ERRORS QUICKLY */
+ addfragptr.p->schemaVer = tschemaVersion;
+ Uint32 tmp = (reqinfo & LqhFragReq::CreateInRunning);
+ addfragptr.p->fragCopyCreation = (tmp == 0 ? 0 : 1);
+ addfragptr.p->addfragErrorCode = 0;
+ addfragptr.p->noOfKeyAttr = noOfKeyAttr;
+ addfragptr.p->noOfNewAttr = noOfNewAttr;
+ addfragptr.p->checksumIndicator = checksumIndicator;
+ addfragptr.p->noOfAttributeGroups = noOfAttributeGroups;
+ addfragptr.p->GCPIndicator = gcpIndicator;
+ addfragptr.p->lh3DistrBits = tlhstar;
+ addfragptr.p->tableType = tableType;
+ addfragptr.p->primaryTableId = primaryTableId;
+
+ if (DictTabInfo::isTable(tableType) ||
+ DictTabInfo::isHashIndex(tableType)) {
+ jam();
+ AccFragReq* const accreq = (AccFragReq*)signal->getDataPtrSend();
+ accreq->userPtr = addfragptr.i;
+ accreq->userRef = cownref;
+ accreq->tableId = tabptr.i;
+ accreq->reqInfo = copyType << 4;
+ accreq->fragId = fragId;
+ accreq->localKeyLen = tlocalKeylen;
+ accreq->maxLoadFactor = tmaxLoadFactor;
+ accreq->minLoadFactor = tminLoadFactor;
+ accreq->kValue = tk;
+ accreq->lhFragBits = tlhstar;
+ accreq->lhDirBits = tlh;
+ accreq->keyLength = ttupKeyLength;
+ /* ----------------------------------------------------------------------- */
+ /* Send ACCFRAGREQ, when confirmation is received send 2 * TUPFRAGREQ to */
+ /* create 2 tuple fragments on this node. */
+ /* ----------------------------------------------------------------------- */
+ addfragptr.p->addfragStatus = AddFragRecord::ACC_ADDFRAG;
+ sendSignal(fragptr.p->accBlockref, GSN_ACCFRAGREQ,
+ signal, AccFragReq::SignalLength, JBB);
+ return;
+ }
+ if (DictTabInfo::isOrderedIndex(tableType)) {
+ jam();
+ // NOTE: next 2 lines stolen from ACC
+ addfragptr.p->fragid1 = (0 << tlhstar) | fragId;
+ addfragptr.p->fragid2 = (1 << tlhstar) | fragId;
+ addfragptr.p->addfragStatus = AddFragRecord::WAIT_TWO_TUP;
+ sendAddFragReq(signal);
+ return;
+ }
+ ndbrequire(false);
+}//Dblqh::execLQHFRAGREQ()
+
+/* *************** */
+/* ACCFRAGCONF > */
+/* *************** */
+void Dblqh::execACCFRAGCONF(Signal* signal)
+{
+ jamEntry();
+ addfragptr.i = signal->theData[0];
+ Uint32 taccConnectptr = signal->theData[1];
+ Uint32 fragId1 = signal->theData[2];
+ Uint32 fragId2 = signal->theData[3];
+ Uint32 accFragPtr1 = signal->theData[4];
+ Uint32 accFragPtr2 = signal->theData[5];
+ Uint32 hashCheckBit = signal->theData[6];
+ ptrCheckGuard(addfragptr, caddfragrecFileSize, addFragRecord);
+ ndbrequire(addfragptr.p->addfragStatus == AddFragRecord::ACC_ADDFRAG);
+
+ addfragptr.p->accConnectptr = taccConnectptr;
+ addfragptr.p->fragid1 = fragId1;
+ addfragptr.p->fragid2 = fragId2;
+ fragptr.i = addfragptr.p->fragmentPtr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ fragptr.p->accFragptr[0] = accFragPtr1;
+ fragptr.p->accFragptr[1] = accFragPtr2;
+ fragptr.p->hashCheckBit = hashCheckBit;
+
+ addfragptr.p->addfragStatus = AddFragRecord::WAIT_TWO_TUP;
+ sendAddFragReq(signal);
+}//Dblqh::execACCFRAGCONF()
+
+/* *************** */
+/* TUPFRAGCONF > */
+/* *************** */
+void Dblqh::execTUPFRAGCONF(Signal* signal)
+{
+ jamEntry();
+ addfragptr.i = signal->theData[0];
+ Uint32 tupConnectptr = signal->theData[1];
+ Uint32 tupFragPtr = signal->theData[2]; /* TUP FRAGMENT POINTER */
+ Uint32 localFragId = signal->theData[3]; /* LOCAL FRAGMENT ID */
+ ptrCheckGuard(addfragptr, caddfragrecFileSize, addFragRecord);
+ fragptr.i = addfragptr.p->fragmentPtr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (localFragId == addfragptr.p->fragid1) {
+ jam();
+ fragptr.p->tupFragptr[0] = tupFragPtr;
+ } else if (localFragId == addfragptr.p->fragid2) {
+ jam();
+ fragptr.p->tupFragptr[1] = tupFragPtr;
+ } else {
+ ndbrequire(false);
+ return;
+ }//if
+ switch (addfragptr.p->addfragStatus) {
+ case AddFragRecord::WAIT_TWO_TUP:
+ jam();
+ fragptr.p->tupFragptr[0] = tupFragPtr;
+ addfragptr.p->tup1Connectptr = tupConnectptr;
+ addfragptr.p->addfragStatus = AddFragRecord::WAIT_ONE_TUP;
+ sendAddFragReq(signal);
+ break;
+ case AddFragRecord::WAIT_ONE_TUP:
+ jam();
+ fragptr.p->tupFragptr[1] = tupFragPtr;
+ addfragptr.p->tup2Connectptr = tupConnectptr;
+ if (DictTabInfo::isOrderedIndex(addfragptr.p->tableType)) {
+ addfragptr.p->addfragStatus = AddFragRecord::WAIT_TWO_TUX;
+ sendAddFragReq(signal);
+ break;
+ }
+ goto done_with_frag;
+ break;
+ case AddFragRecord::WAIT_TWO_TUX:
+ jam();
+ fragptr.p->tuxFragptr[0] = tupFragPtr;
+ addfragptr.p->tux1Connectptr = tupConnectptr;
+ addfragptr.p->addfragStatus = AddFragRecord::WAIT_ONE_TUX;
+ sendAddFragReq(signal);
+ break;
+ case AddFragRecord::WAIT_ONE_TUX:
+ jam();
+ fragptr.p->tuxFragptr[1] = tupFragPtr;
+ addfragptr.p->tux2Connectptr = tupConnectptr;
+ goto done_with_frag;
+ break;
+ done_with_frag:
+ /* ---------------------------------------------------------------- */
+ /* Finished create of fragments. Now ready for creating attributes. */
+ /* ---------------------------------------------------------------- */
+ addfragptr.p->addfragStatus = AddFragRecord::WAIT_ADD_ATTR;
+ {
+ LqhFragConf* conf = (LqhFragConf*)signal->getDataPtrSend();
+ conf->senderData = addfragptr.p->dictConnectptr;
+ conf->lqhFragPtr = addfragptr.i;
+ sendSignal(addfragptr.p->dictBlockref, GSN_LQHFRAGCONF,
+ signal, LqhFragConf::SignalLength, JBB);
+ }
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+}//Dblqh::execTUPFRAGCONF()
+
+/* *************** */
+/* TUXFRAGCONF > */
+/* *************** */
+void Dblqh::execTUXFRAGCONF(Signal* signal)
+{
+ jamEntry();
+ execTUPFRAGCONF(signal);
+}//Dblqh::execTUXFRAGCONF
+
+/*
+ * Add fragment in TUP or TUX. Called up to 4 times.
+ */
+void
+Dblqh::sendAddFragReq(Signal* signal)
+{
+ fragptr.i = addfragptr.p->fragmentPtr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (addfragptr.p->addfragStatus == AddFragRecord::WAIT_TWO_TUP ||
+ addfragptr.p->addfragStatus == AddFragRecord::WAIT_ONE_TUP) {
+ if (DictTabInfo::isTable(addfragptr.p->tableType) ||
+ DictTabInfo::isHashIndex(addfragptr.p->tableType)) {
+ jam();
+ signal->theData[0] = addfragptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = 0; /* ADD TABLE */
+ signal->theData[3] = addfragptr.p->tabId;
+ signal->theData[4] = addfragptr.p->noOfAttr;
+ signal->theData[5] =
+ addfragptr.p->addfragStatus == AddFragRecord::WAIT_TWO_TUP
+ ? addfragptr.p->fragid1 : addfragptr.p->fragid2;
+ signal->theData[6] = (addfragptr.p->noOfAllocPages >> 1) + 1;
+ signal->theData[7] = addfragptr.p->noOfNull;
+ signal->theData[8] = addfragptr.p->schemaVer;
+ signal->theData[9] = addfragptr.p->noOfKeyAttr;
+ signal->theData[10] = addfragptr.p->noOfNewAttr;
+ signal->theData[11] = addfragptr.p->checksumIndicator;
+ signal->theData[12] = addfragptr.p->noOfAttributeGroups;
+ signal->theData[13] = addfragptr.p->GCPIndicator;
+ sendSignal(fragptr.p->tupBlockref, GSN_TUPFRAGREQ,
+ signal, TupFragReq::SignalLength, JBB);
+ return;
+ }
+ if (DictTabInfo::isOrderedIndex(addfragptr.p->tableType)) {
+ jam();
+ signal->theData[0] = addfragptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = 0; /* ADD TABLE */
+ signal->theData[3] = addfragptr.p->tabId;
+ signal->theData[4] = 1; /* ordered index: one array attr */
+ signal->theData[5] =
+ addfragptr.p->addfragStatus == AddFragRecord::WAIT_TWO_TUP
+ ? addfragptr.p->fragid1 : addfragptr.p->fragid2;
+ signal->theData[6] = (addfragptr.p->noOfAllocPages >> 1) + 1;
+ signal->theData[7] = 0; /* ordered index: no nullable */
+ signal->theData[8] = addfragptr.p->schemaVer;
+ signal->theData[9] = 1; /* ordered index: one key */
+ signal->theData[10] = addfragptr.p->noOfNewAttr;
+ signal->theData[11] = addfragptr.p->checksumIndicator;
+ signal->theData[12] = addfragptr.p->noOfAttributeGroups;
+ signal->theData[13] = addfragptr.p->GCPIndicator;
+ sendSignal(fragptr.p->tupBlockref, GSN_TUPFRAGREQ,
+ signal, TupFragReq::SignalLength, JBB);
+ return;
+ }
+ }
+ if (addfragptr.p->addfragStatus == AddFragRecord::WAIT_TWO_TUX ||
+ addfragptr.p->addfragStatus == AddFragRecord::WAIT_ONE_TUX) {
+ if (DictTabInfo::isOrderedIndex(addfragptr.p->tableType)) {
+ jam();
+ TuxFragReq* const tuxreq = (TuxFragReq*)signal->getDataPtrSend();
+ tuxreq->userPtr = addfragptr.i;
+ tuxreq->userRef = cownref;
+ tuxreq->reqInfo = 0; /* ADD TABLE */
+ tuxreq->tableId = addfragptr.p->tabId;
+ ndbrequire(addfragptr.p->noOfAttr >= 2);
+ tuxreq->noOfAttr = addfragptr.p->noOfAttr - 1; /* skip NDB$TNODE */
+ tuxreq->fragId =
+ addfragptr.p->addfragStatus == AddFragRecord::WAIT_TWO_TUX
+ ? addfragptr.p->fragid1 : addfragptr.p->fragid2;
+ tuxreq->fragOff = addfragptr.p->lh3DistrBits;
+ tuxreq->tableType = addfragptr.p->tableType;
+ tuxreq->primaryTableId = addfragptr.p->primaryTableId;
+ sendSignal(fragptr.p->tuxBlockref, GSN_TUXFRAGREQ,
+ signal, TuxFragReq::SignalLength, JBB);
+ return;
+ }
+ }
+ ndbrequire(false);
+}//Dblqh::sendAddFragReq
+
+/* ************************************************************************> */
+/* LQHADDATTRREQ: Request from DICT to create attributes for the new table. */
+/* ************************************************************************> */
+void Dblqh::execLQHADDATTREQ(Signal* signal)
+{
+ jamEntry();
+ LqhAddAttrReq * const req = (LqhAddAttrReq*)signal->getDataPtr();
+
+ addfragptr.i = req->lqhFragPtr;
+ const Uint32 tnoOfAttr = req->noOfAttributes;
+ const Uint32 senderData = req->senderData;
+ const Uint32 senderAttrPtr = req->senderAttrPtr;
+
+ ptrCheckGuard(addfragptr, caddfragrecFileSize, addFragRecord);
+ ndbrequire(addfragptr.p->addfragStatus == AddFragRecord::WAIT_ADD_ATTR);
+ ndbrequire((tnoOfAttr != 0) && (tnoOfAttr <= LqhAddAttrReq::MAX_ATTRIBUTES));
+ addfragptr.p->totalAttrReceived += tnoOfAttr;
+ ndbrequire(addfragptr.p->totalAttrReceived <= addfragptr.p->noOfAttr);
+
+ addfragptr.p->attrReceived = tnoOfAttr;
+ for (Uint32 i = 0; i < tnoOfAttr; i++) {
+ addfragptr.p->attributes[i] = req->attributes[i];
+ }//for
+ addfragptr.p->attrSentToTup = 0;
+ ndbrequire(addfragptr.p->dictConnectptr == senderData);
+ addfragptr.p->m_senderAttrPtr = senderAttrPtr;
+ addfragptr.p->addfragStatus = AddFragRecord::TUP_ATTR_WAIT1;
+ sendAddAttrReq(signal);
+}//Dblqh::execLQHADDATTREQ()
+
+/* *********************>> */
+/* TUP_ADD_ATTCONF > */
+/* *********************>> */
+void Dblqh::execTUP_ADD_ATTCONF(Signal* signal)
+{
+ jamEntry();
+ addfragptr.i = signal->theData[0];
+ ptrCheckGuard(addfragptr, caddfragrecFileSize, addFragRecord);
+ switch (addfragptr.p->addfragStatus) {
+ case AddFragRecord::TUP_ATTR_WAIT1:
+ jam();
+ addfragptr.p->addfragStatus = AddFragRecord::TUP_ATTR_WAIT2;
+ sendAddAttrReq(signal);
+ break;
+ case AddFragRecord::TUP_ATTR_WAIT2:
+ jam();
+ if (DictTabInfo::isOrderedIndex(addfragptr.p->tableType)) {
+ addfragptr.p->addfragStatus = AddFragRecord::TUX_ATTR_WAIT1;
+ sendAddAttrReq(signal);
+ break;
+ }
+ goto done_with_attr;
+ break;
+ case AddFragRecord::TUX_ATTR_WAIT1:
+ jam();
+ addfragptr.p->addfragStatus = AddFragRecord::TUX_ATTR_WAIT2;
+ sendAddAttrReq(signal);
+ break;
+ case AddFragRecord::TUX_ATTR_WAIT2:
+ jam();
+ goto done_with_attr;
+ break;
+ done_with_attr:
+ addfragptr.p->attrSentToTup = addfragptr.p->attrSentToTup + 1;
+ ndbrequire(addfragptr.p->attrSentToTup <= addfragptr.p->attrReceived);
+ ndbrequire(addfragptr.p->totalAttrReceived <= addfragptr.p->noOfAttr);
+ if (addfragptr.p->attrSentToTup < addfragptr.p->attrReceived) {
+ // more in this batch
+ jam();
+ addfragptr.p->addfragStatus = AddFragRecord::TUP_ATTR_WAIT1;
+ sendAddAttrReq(signal);
+ } else if (addfragptr.p->totalAttrReceived < addfragptr.p->noOfAttr) {
+ // more batches to receive
+ jam();
+ addfragptr.p->addfragStatus = AddFragRecord::WAIT_ADD_ATTR;
+ LqhAddAttrConf *const conf = (LqhAddAttrConf*)signal->getDataPtrSend();
+ conf->senderData = addfragptr.p->dictConnectptr;
+ conf->senderAttrPtr = addfragptr.p->m_senderAttrPtr;
+ conf->fragId = addfragptr.p->addFragid;
+ sendSignal(addfragptr.p->dictBlockref, GSN_LQHADDATTCONF,
+ signal, LqhAddAttrConf::SignalLength, JBB);
+ } else {
+ fragptr.i = addfragptr.p->fragmentPtr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ /* ------------------------------------------------------------------
+ * WE HAVE NOW COMPLETED ADDING THIS FRAGMENT. WE NOW NEED TO SET THE
+ * PROPER STATE IN FRAG_STATUS DEPENDENT ON IF WE ARE CREATING A NEW
+ * REPLICA OR IF WE ARE CREATING A TABLE. FOR FRAGMENTS IN COPY
+ * PROCESS WE DO NOT WANT LOGGING ACTIVATED.
+ * ----------------------------------------------------------------- */
+ if (addfragptr.p->fragCopyCreation == 1) {
+ jam();
+ if (! DictTabInfo::isOrderedIndex(addfragptr.p->tableType))
+ fragptr.p->fragStatus = Fragrecord::ACTIVE_CREATION;
+ else
+ fragptr.p->fragStatus = Fragrecord::FSACTIVE;
+ fragptr.p->logFlag = Fragrecord::STATE_FALSE;
+ } else {
+ jam();
+ fragptr.p->fragStatus = Fragrecord::FSACTIVE;
+ }//if
+ LqhAddAttrConf *const conf = (LqhAddAttrConf*)signal->getDataPtrSend();
+ conf->senderData = addfragptr.p->dictConnectptr;
+ conf->senderAttrPtr = addfragptr.p->m_senderAttrPtr;
+ conf->fragId = addfragptr.p->addFragid;
+ sendSignal(addfragptr.p->dictBlockref, GSN_LQHADDATTCONF, signal,
+ LqhAddAttrConf::SignalLength, JBB);
+ releaseAddfragrec(signal);
+ }//if
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+}
+
+/* **********************>> */
+/* TUX_ADD_ATTRCONF > */
+/* **********************>> */
+void Dblqh::execTUX_ADD_ATTRCONF(Signal* signal)
+{
+ jamEntry();
+ execTUP_ADD_ATTCONF(signal);
+}//Dblqh::execTUX_ADD_ATTRCONF
+
+/*
+ * Add attribute in TUP or TUX. Called up to 4 times.
+ */
+void
+Dblqh::sendAddAttrReq(Signal* signal)
+{
+ arrGuard(addfragptr.p->attrSentToTup, LqhAddAttrReq::MAX_ATTRIBUTES);
+ LqhAddAttrReq::Entry& entry =
+ addfragptr.p->attributes[addfragptr.p->attrSentToTup];
+ const Uint32 attrId = entry.attrId & 0xffff;
+ const Uint32 primaryAttrId = entry.attrId >> 16;
+ fragptr.i = addfragptr.p->fragmentPtr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (addfragptr.p->addfragStatus == AddFragRecord::TUP_ATTR_WAIT1 ||
+ addfragptr.p->addfragStatus == AddFragRecord::TUP_ATTR_WAIT2) {
+ if (DictTabInfo::isTable(addfragptr.p->tableType) ||
+ DictTabInfo::isHashIndex(addfragptr.p->tableType) ||
+ (DictTabInfo::isOrderedIndex(addfragptr.p->tableType) &&
+ primaryAttrId == ZNIL)) {
+ jam();
+ TupAddAttrReq* const tupreq = (TupAddAttrReq*)signal->getDataPtrSend();
+ tupreq->tupConnectPtr =
+ addfragptr.p->addfragStatus == AddFragRecord::TUP_ATTR_WAIT1
+ ? addfragptr.p->tup1Connectptr : addfragptr.p->tup2Connectptr;
+ tupreq->notused1 = 0;
+ tupreq->attrId = attrId;
+ tupreq->attrDescriptor = entry.attrDescriptor;
+ sendSignal(fragptr.p->tupBlockref, GSN_TUP_ADD_ATTRREQ,
+ signal, TupAddAttrReq::SignalLength, JBB);
+ return;
+ }
+ if (DictTabInfo::isOrderedIndex(addfragptr.p->tableType) &&
+ primaryAttrId != ZNIL) {
+ // this attribute is not for TUP
+ jam();
+ TupAddAttrConf* tupconf = (TupAddAttrConf*)signal->getDataPtrSend();
+ tupconf->userPtr = addfragptr.i;
+ sendSignal(reference(), GSN_TUP_ADD_ATTCONF,
+ signal, TupAddAttrConf::SignalLength, JBB);
+ return;
+ }
+ }
+ if (addfragptr.p->addfragStatus == AddFragRecord::TUX_ATTR_WAIT1 ||
+ addfragptr.p->addfragStatus == AddFragRecord::TUX_ATTR_WAIT2) {
+ jam();
+ if (DictTabInfo::isOrderedIndex(addfragptr.p->tableType) &&
+ primaryAttrId != ZNIL) {
+ jam();
+ TuxAddAttrReq* const tuxreq = (TuxAddAttrReq*)signal->getDataPtrSend();
+ tuxreq->tuxConnectPtr =
+ addfragptr.p->addfragStatus == AddFragRecord::TUX_ATTR_WAIT1
+ ? addfragptr.p->tux1Connectptr : addfragptr.p->tux2Connectptr;
+ tuxreq->notused1 = 0;
+ tuxreq->attrId = attrId;
+ tuxreq->attrDescriptor = entry.attrDescriptor;
+ tuxreq->extTypeInfo = entry.extTypeInfo;
+ tuxreq->primaryAttrId = primaryAttrId;
+ sendSignal(fragptr.p->tuxBlockref, GSN_TUX_ADD_ATTRREQ,
+ signal, TuxAddAttrReq::SignalLength, JBB);
+ return;
+ }
+ if (DictTabInfo::isOrderedIndex(addfragptr.p->tableType) &&
+ primaryAttrId == ZNIL) {
+ // this attribute is not for TUX
+ jam();
+ TuxAddAttrConf* tuxconf = (TuxAddAttrConf*)signal->getDataPtrSend();
+ tuxconf->userPtr = addfragptr.i;
+ sendSignal(reference(), GSN_TUX_ADD_ATTRCONF,
+ signal, TuxAddAttrConf::SignalLength, JBB);
+ return;
+ }
+ }
+ ndbrequire(false);
+}//Dblqh::sendAddAttrReq
+
+/* ************************************************************************>> */
+/* TAB_COMMITREQ: Commit the new table for use in transactions. Sender DICT. */
+/* ************************************************************************>> */
+void Dblqh::execTAB_COMMITREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 dihPtr = signal->theData[0];
+ BlockReference dihBlockref = signal->theData[1];
+ tabptr.i = signal->theData[2];
+
+ if (tabptr.i >= ctabrecFileSize) {
+ jam();
+ terrorCode = ZTAB_FILE_SIZE;
+ signal->theData[0] = dihPtr;
+ signal->theData[1] = cownNodeid;
+ signal->theData[2] = tabptr.i;
+ signal->theData[3] = terrorCode;
+ sendSignal(dihBlockref, GSN_TAB_COMMITREF, signal, 4, JBB);
+ return;
+ }//if
+ ptrAss(tabptr, tablerec);
+ if (tabptr.p->tableStatus != Tablerec::ADD_TABLE_ONGOING) {
+ jam();
+ terrorCode = ZTAB_STATE_ERROR;
+ signal->theData[0] = dihPtr;
+ signal->theData[1] = cownNodeid;
+ signal->theData[2] = tabptr.i;
+ signal->theData[3] = terrorCode;
+ signal->theData[4] = tabptr.p->tableStatus;
+ sendSignal(dihBlockref, GSN_TAB_COMMITREF, signal, 5, JBB);
+ ndbrequire(false);
+ return;
+ }//if
+ tabptr.p->usageCount = 0;
+ tabptr.p->tableStatus = Tablerec::TABLE_DEFINED;
+ signal->theData[0] = dihPtr;
+ signal->theData[1] = cownNodeid;
+ signal->theData[2] = tabptr.i;
+ sendSignal(dihBlockref, GSN_TAB_COMMITCONF, signal, 3, JBB);
+ return;
+}//Dblqh::execTAB_COMMITREQ()
+
+
+void Dblqh::fragrefLab(Signal* signal,
+ BlockReference fragBlockRef,
+ Uint32 fragConPtr,
+ Uint32 errorCode)
+{
+ LqhFragRef * ref = (LqhFragRef*)signal->getDataPtrSend();
+ ref->senderData = fragConPtr;
+ ref->errorCode = errorCode;
+ sendSignal(fragBlockRef, GSN_LQHFRAGREF, signal,
+ LqhFragRef::SignalLength, JBB);
+ return;
+}//Dblqh::fragrefLab()
+
+/* ************>> */
+/* ACCFRAGREF > */
+/* ************>> */
+void Dblqh::execACCFRAGREF(Signal* signal)
+{
+ jamEntry();
+ addfragptr.i = signal->theData[0];
+ ptrCheckGuard(addfragptr, caddfragrecFileSize, addFragRecord);
+ terrorCode = signal->theData[1];
+ ndbrequire(addfragptr.p->addfragStatus == AddFragRecord::ACC_ADDFRAG);
+ addfragptr.p->addfragErrorCode = terrorCode;
+
+ const Uint32 ref = addfragptr.p->dictBlockref;
+ const Uint32 senderData = addfragptr.p->dictConnectptr;
+ const Uint32 errorCode = addfragptr.p->addfragErrorCode;
+ releaseAddfragrec(signal);
+ fragrefLab(signal, ref, senderData, errorCode);
+
+ return;
+}//Dblqh::execACCFRAGREF()
+
+/* ************>> */
+/* TUPFRAGREF > */
+/* ************>> */
+void Dblqh::execTUPFRAGREF(Signal* signal)
+{
+ jamEntry();
+ addfragptr.i = signal->theData[0];
+ ptrCheckGuard(addfragptr, caddfragrecFileSize, addFragRecord);
+ terrorCode = signal->theData[1];
+ fragptr.i = addfragptr.p->fragmentPtr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ addfragptr.p->addfragErrorCode = terrorCode;
+ const Uint32 ref = addfragptr.p->dictBlockref;
+ const Uint32 senderData = addfragptr.p->dictConnectptr;
+ const Uint32 errorCode = addfragptr.p->addfragErrorCode;
+ releaseAddfragrec(signal);
+ fragrefLab(signal, ref, senderData, errorCode);
+
+}//Dblqh::execTUPFRAGREF()
+
+/* ************>> */
+/* TUXFRAGREF > */
+/* ************>> */
+void Dblqh::execTUXFRAGREF(Signal* signal)
+{
+ jamEntry();
+ execTUPFRAGREF(signal);
+}//Dblqh::execTUXFRAGREF
+
+/* *********************> */
+/* TUP_ADD_ATTREF > */
+/* *********************> */
+void Dblqh::execTUP_ADD_ATTRREF(Signal* signal)
+{
+ jamEntry();
+
+ addfragptr.i = signal->theData[0];
+ ptrCheckGuard(addfragptr, caddfragrecFileSize, addFragRecord);
+ terrorCode = signal->theData[1];
+ addfragptr.p->addfragErrorCode = terrorCode;
+
+ const Uint32 Ref = addfragptr.p->dictBlockref;
+ const Uint32 senderData = addfragptr.p->dictConnectptr;
+ const Uint32 errorCode = addfragptr.p->addfragErrorCode;
+ releaseAddfragrec(signal);
+
+ LqhAddAttrRef *const ref = (LqhAddAttrRef*)signal->getDataPtrSend();
+ ref->senderData = senderData;
+ ref->errorCode = errorCode;
+ sendSignal(Ref, GSN_LQHADDATTREF, signal,
+ LqhAddAttrRef::SignalLength, JBB);
+
+}//Dblqh::execTUP_ADD_ATTRREF()
+
+/* **********************> */
+/* TUX_ADD_ATTRREF > */
+/* **********************> */
+void Dblqh::execTUX_ADD_ATTRREF(Signal* signal)
+{
+ jamEntry();
+ execTUP_ADD_ATTRREF(signal);
+}//Dblqh::execTUX_ADD_ATTRREF
+
+void
+Dblqh::execPREP_DROP_TAB_REQ(Signal* signal){
+ jamEntry();
+
+ PrepDropTabReq* req = (PrepDropTabReq*)signal->getDataPtr();
+
+ Uint32 senderRef = req->senderRef;
+ Uint32 senderData = req->senderData;
+
+ TablerecPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabrecFileSize, tablerec);
+
+ Uint32 errCode = 0;
+ errCode = checkDropTabState(tabPtr.p->tableStatus, GSN_PREP_DROP_TAB_REQ);
+ if(errCode != 0){
+ jam();
+
+ PrepDropTabRef* ref = (PrepDropTabRef*)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = senderData;
+ ref->tableId = tabPtr.i;
+ ref->errorCode = errCode;
+ sendSignal(senderRef, GSN_PREP_DROP_TAB_REF, signal,
+ PrepDropTabRef::SignalLength, JBB);
+ return;
+ }
+
+ tabPtr.p->tableStatus = Tablerec::PREP_DROP_TABLE_ONGOING;
+ tabPtr.p->waitingTC.clear();
+ tabPtr.p->waitingDIH.clear();
+
+ PrepDropTabConf * conf = (PrepDropTabConf*)signal->getDataPtrSend();
+ conf->tableId = tabPtr.i;
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ sendSignal(senderRef, GSN_PREP_DROP_TAB_CONF, signal,
+ PrepDropTabConf::SignalLength, JBB);
+
+ signal->theData[0] = ZPREP_DROP_TABLE;
+ signal->theData[1] = tabPtr.i;
+ signal->theData[2] = senderRef;
+ signal->theData[3] = senderData;
+ checkDropTab(signal);
+}
+
+void
+Dblqh::checkDropTab(Signal* signal){
+
+ TablerecPtr tabPtr;
+ tabPtr.i = signal->theData[1];
+ ptrCheckGuard(tabPtr, ctabrecFileSize, tablerec);
+
+ ndbrequire(tabPtr.p->tableStatus == Tablerec::PREP_DROP_TABLE_ONGOING);
+
+ if(tabPtr.p->usageCount > 0){
+ jam();
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 4);
+ return;
+ }
+
+ bool lcpDone = true;
+ lcpPtr.i = 0;
+ ptrAss(lcpPtr, lcpRecord);
+ if(lcpPtr.p->lcpState != LcpRecord::LCP_IDLE){
+ jam();
+
+ if(lcpPtr.p->currentFragment.lcpFragOrd.tableId == tabPtr.i){
+ jam();
+ lcpDone = false;
+ }
+
+ if(lcpPtr.p->lcpQueued &&
+ lcpPtr.p->queuedFragment.lcpFragOrd.tableId == tabPtr.i){
+ jam();
+ lcpDone = false;
+ }
+ }
+
+ if(!lcpDone){
+ jam();
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 4);
+ return;
+ }
+
+ tabPtr.p->tableStatus = Tablerec::PREP_DROP_TABLE_DONE;
+
+ WaitDropTabConf * conf = (WaitDropTabConf*)signal->getDataPtrSend();
+ conf->tableId = tabPtr.i;
+ conf->senderRef = reference();
+ for(Uint32 i = 1; i<MAX_NDB_NODES; i++){
+ if(tabPtr.p->waitingTC.get(i)){
+ tabPtr.p->waitingTC.clear(i);
+ sendSignal(calcTcBlockRef(i), GSN_WAIT_DROP_TAB_CONF, signal,
+ WaitDropTabConf::SignalLength, JBB);
+ }
+ if(tabPtr.p->waitingDIH.get(i)){
+ tabPtr.p->waitingDIH.clear(i);
+ sendSignal(calcDihBlockRef(i), GSN_WAIT_DROP_TAB_CONF, signal,
+ WaitDropTabConf::SignalLength, JBB);
+ }
+ }
+}
+
+void
+Dblqh::execWAIT_DROP_TAB_REQ(Signal* signal){
+ jamEntry();
+ WaitDropTabReq * req = (WaitDropTabReq*)signal->getDataPtr();
+
+ TablerecPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabrecFileSize, tablerec);
+
+ Uint32 senderRef = req->senderRef;
+ Uint32 nodeId = refToNode(senderRef);
+ Uint32 blockNo = refToBlock(senderRef);
+
+ if(tabPtr.p->tableStatus == Tablerec::PREP_DROP_TABLE_ONGOING){
+ jam();
+ switch(blockNo){
+ case DBTC:
+ tabPtr.p->waitingTC.set(nodeId);
+ break;
+ case DBDIH:
+ tabPtr.p->waitingDIH.set(nodeId);
+ break;
+ default:
+ ndbrequire(false);
+ }
+ return;
+ }
+
+ if(tabPtr.p->tableStatus == Tablerec::PREP_DROP_TABLE_DONE){
+ jam();
+ WaitDropTabConf * conf = (WaitDropTabConf*)signal->getDataPtrSend();
+ conf->tableId = tabPtr.i;
+ conf->senderRef = reference();
+ sendSignal(senderRef, GSN_WAIT_DROP_TAB_CONF, signal,
+ WaitDropTabConf::SignalLength, JBB);
+ return;
+ }
+
+ WaitDropTabRef * ref = (WaitDropTabRef*)signal->getDataPtrSend();
+ ref->tableId = tabPtr.i;
+ ref->senderRef = reference();
+
+ bool ok = false;
+ switch(tabPtr.p->tableStatus){
+ case Tablerec::TABLE_DEFINED:
+ ok = true;
+ ref->errorCode = WaitDropTabRef::IllegalTableState;
+ break;
+ case Tablerec::NOT_DEFINED:
+ ok = true;
+ ref->errorCode = WaitDropTabRef::NoSuchTable;
+ break;
+ case Tablerec::ADD_TABLE_ONGOING:
+ ok = true;
+ ref->errorCode = WaitDropTabRef::IllegalTableState;
+ break;
+ case Tablerec::PREP_DROP_TABLE_ONGOING:
+ case Tablerec::PREP_DROP_TABLE_DONE:
+ // Should have been take care of above
+ ndbrequire(false);
+ }
+ ndbrequire(ok);
+ ref->tableStatus = tabPtr.p->tableStatus;
+ sendSignal(senderRef, GSN_WAIT_DROP_TAB_REF, signal,
+ WaitDropTabRef::SignalLength, JBB);
+ return;
+}
+
+void
+Dblqh::execDROP_TAB_REQ(Signal* signal){
+ jamEntry();
+
+ DropTabReq* req = (DropTabReq*)signal->getDataPtr();
+
+ Uint32 senderRef = req->senderRef;
+ Uint32 senderData = req->senderData;
+
+ TablerecPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabrecFileSize, tablerec);
+
+ do {
+ if(req->requestType == DropTabReq::RestartDropTab){
+ jam();
+ break;
+ }
+
+ if(req->requestType == DropTabReq::OnlineDropTab){
+ jam();
+ Uint32 errCode = 0;
+ errCode = checkDropTabState(tabPtr.p->tableStatus, GSN_DROP_TAB_REQ);
+ if(errCode != 0){
+ jam();
+
+ DropTabRef* ref = (DropTabRef*)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = senderData;
+ ref->tableId = tabPtr.i;
+ ref->errorCode = errCode;
+ sendSignal(senderRef, GSN_DROP_TAB_REF, signal,
+ DropTabRef::SignalLength, JBB);
+ return;
+ }
+ }
+
+ removeTable(tabPtr.i);
+
+ } while(false);
+
+ ndbrequire(tabPtr.p->usageCount == 0);
+ tabPtr.p->tableStatus = Tablerec::NOT_DEFINED;
+
+ DropTabConf * const dropConf = (DropTabConf *)signal->getDataPtrSend();
+ dropConf->senderRef = reference();
+ dropConf->senderData = senderData;
+ dropConf->tableId = tabPtr.i;
+ sendSignal(senderRef, GSN_DROP_TAB_CONF,
+ signal, DropTabConf::SignalLength, JBB);
+}
+
+Uint32
+Dblqh::checkDropTabState(Tablerec::TableStatus status, Uint32 gsn) const{
+
+ if(gsn == GSN_PREP_DROP_TAB_REQ){
+ switch(status){
+ case Tablerec::NOT_DEFINED:
+ jam();
+ // Fall through
+ case Tablerec::ADD_TABLE_ONGOING:
+ jam();
+ return PrepDropTabRef::NoSuchTable;
+ break;
+ case Tablerec::PREP_DROP_TABLE_ONGOING:
+ jam();
+ return PrepDropTabRef::PrepDropInProgress;
+ break;
+ case Tablerec::PREP_DROP_TABLE_DONE:
+ jam();
+ return PrepDropTabRef::DropInProgress;
+ break;
+ case Tablerec::TABLE_DEFINED:
+ jam();
+ return 0;
+ break;
+ }
+ ndbrequire(0);
+ }
+
+ if(gsn == GSN_DROP_TAB_REQ){
+ switch(status){
+ case Tablerec::NOT_DEFINED:
+ jam();
+ // Fall through
+ case Tablerec::ADD_TABLE_ONGOING:
+ jam();
+ return DropTabRef::NoSuchTable;
+ break;
+ case Tablerec::PREP_DROP_TABLE_ONGOING:
+ jam();
+ return DropTabRef::PrepDropInProgress;
+ break;
+ case Tablerec::PREP_DROP_TABLE_DONE:
+ jam();
+ return 0;
+ break;
+ case Tablerec::TABLE_DEFINED:
+ jam();
+ return DropTabRef::DropWoPrep;
+ }
+ ndbrequire(0);
+ }
+ ndbrequire(0);
+ return RNIL;
+}
+
+void Dblqh::removeTable(Uint32 tableId)
+{
+ tabptr.i = tableId;
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+
+ for (Uint32 i = (NO_OF_FRAG_PER_NODE - 1); (Uint32)~i; i--) {
+ jam();
+ if (tabptr.p->fragid[i] != ZNIL) {
+ jam();
+ deleteFragrec(tabptr.p->fragid[i]);
+ }//if
+ }//for
+}//Dblqh::removeTable()
+
+void
+Dblqh::execALTER_TAB_REQ(Signal* signal)
+{
+ jamEntry();
+ AlterTabReq* const req = (AlterTabReq*)signal->getDataPtr();
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 senderData = req->senderData;
+ const Uint32 changeMask = req->changeMask;
+ const Uint32 tableId = req->tableId;
+ const Uint32 tableVersion = req->tableVersion;
+ const Uint32 gci = req->gci;
+ AlterTabReq::RequestType requestType =
+ (AlterTabReq::RequestType) req->requestType;
+
+ TablerecPtr tablePtr;
+ tablePtr.i = tableId;
+ ptrCheckGuard(tablePtr, ctabrecFileSize, tablerec);
+ tablePtr.p->schemaVersion = tableVersion;
+
+ // Request handled successfully
+ AlterTabConf * conf = (AlterTabConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ conf->changeMask = changeMask;
+ conf->tableId = tableId;
+ conf->tableVersion = tableVersion;
+ conf->gci = gci;
+ conf->requestType = requestType;
+ sendSignal(senderRef, GSN_ALTER_TAB_CONF, signal,
+ AlterTabConf::SignalLength, JBB);
+}
+
+/* ************************************************************************>>
+ * TIME_SIGNAL: Handles time-out of local operations. This is a clean-up
+ * handler. If no other measure has succeeded in cleaning up after time-outs
+ * or else then this routine will remove the transaction after 120 seconds of
+ * inactivity. The check is performed once per 10 second. Sender is QMGR.
+ * ************************************************************************>> */
+void Dblqh::execTIME_SIGNAL(Signal* signal)
+{
+ jamEntry();
+ cLqhTimeOutCount++;
+ cLqhTimeOutCheckCount++;
+ if ((cCounterAccCommitBlocked > 0) ||
+ (cCounterTupCommitBlocked > 0)) {
+ jam();
+ signal->theData[0] = EventReport::UndoLogBlocked;
+ signal->theData[1] = cCounterTupCommitBlocked;
+ signal->theData[2] = cCounterAccCommitBlocked;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+
+ cCounterTupCommitBlocked = 0;
+ cCounterAccCommitBlocked = 0;
+ }//if
+ if (cLqhTimeOutCheckCount < 10) {
+ jam();
+ return;
+ }//if
+ cLqhTimeOutCheckCount = 0;
+#ifdef VM_TRACE
+ TcConnectionrecPtr tTcConptr;
+
+ for (tTcConptr.i = 0; tTcConptr.i < ctcConnectrecFileSize;
+ tTcConptr.i++) {
+ jam();
+ ptrAss(tTcConptr, tcConnectionrec);
+ if ((tTcConptr.p->tcTimer != 0) &&
+ ((tTcConptr.p->tcTimer + 120) < cLqhTimeOutCount)) {
+ ndbout << "Dblqh::execTIME_SIGNAL"<<endl
+ << "Timeout found in tcConnectRecord " <<tTcConptr.i<<endl
+ << " cLqhTimeOutCount = " << cLqhTimeOutCount << endl
+ << " tcTimer="<<tTcConptr.p->tcTimer<<endl
+ << " tcTimer+120="<<tTcConptr.p->tcTimer + 120<<endl;
+
+ ndbout << " transactionState = " << tTcConptr.p->transactionState<<endl;
+ ndbout << " operation = " << tTcConptr.p->operation<<endl;
+ ndbout << " tcNodeFailrec = " << tTcConptr.p->tcNodeFailrec
+ << " seqNoReplica = " << tTcConptr.p->seqNoReplica
+ << " simpleRead = " << tTcConptr.p->simpleRead
+ << endl;
+ ndbout << " replicaType = " << tTcConptr.p->replicaType
+ << " reclenAiLqhkey = " << tTcConptr.p->reclenAiLqhkey
+ << " opExec = " << tTcConptr.p->opExec
+ << endl;
+ ndbout << " opSimple = " << tTcConptr.p->opSimple
+ << " nextSeqNoReplica = " << tTcConptr.p->nextSeqNoReplica
+ << " lockType = " << tTcConptr.p->lockType
+ << " localFragptr = " << tTcConptr.p->localFragptr
+ << endl;
+ ndbout << " lastReplicaNo = " << tTcConptr.p->lastReplicaNo
+ << " indTakeOver = " << tTcConptr.p->indTakeOver
+ << " dirtyOp = " << tTcConptr.p->dirtyOp
+ << endl;
+ ndbout << " activeCreat = " << tTcConptr.p->activeCreat
+ << " tcBlockref = " << hex << tTcConptr.p->tcBlockref
+ << " reqBlockref = " << hex << tTcConptr.p->reqBlockref
+ << " primKeyLen = " << tTcConptr.p->primKeyLen
+ << endl;
+ ndbout << " nextReplica = " << tTcConptr.p->nextReplica
+ << " tcBlockref = " << hex << tTcConptr.p->tcBlockref
+ << " reqBlockref = " << hex << tTcConptr.p->reqBlockref
+ << " primKeyLen = " << tTcConptr.p->primKeyLen
+ << endl;
+ ndbout << " logStopPageNo = " << tTcConptr.p->logStopPageNo
+ << " logStartPageNo = " << tTcConptr.p->logStartPageNo
+ << " logStartPageIndex = " << tTcConptr.p->logStartPageIndex
+ << endl;
+ ndbout << " errorCode = " << tTcConptr.p->errorCode
+ << " clientBlockref = " << hex << tTcConptr.p->clientBlockref
+ << " applRef = " << hex << tTcConptr.p->applRef
+ << " totSendlenAi = " << tTcConptr.p->totSendlenAi
+ << endl;
+ ndbout << " totReclenAi = " << tTcConptr.p->totReclenAi
+ << " tcScanRec = " << tTcConptr.p->tcScanRec
+ << " tcScanInfo = " << tTcConptr.p->tcScanInfo
+ << " tcOprec = " << hex << tTcConptr.p->tcOprec
+ << endl;
+ ndbout << " tableref = " << tTcConptr.p->tableref
+ << " simpleTcConnect = " << tTcConptr.p->simpleTcConnect
+ << " storedProcId = " << tTcConptr.p->storedProcId
+ << " schemaVersion = " << tTcConptr.p->schemaVersion
+ << endl;
+ ndbout << " reqinfo = " << tTcConptr.p->reqinfo
+ << " reqRef = " << tTcConptr.p->reqRef
+ << " readlenAi = " << tTcConptr.p->readlenAi
+ << " prevTc = " << tTcConptr.p->prevTc
+ << endl;
+ ndbout << " prevLogTcrec = " << tTcConptr.p->prevLogTcrec
+ << " prevHashRec = " << tTcConptr.p->prevHashRec
+ << " nodeAfterNext0 = " << tTcConptr.p->nodeAfterNext[0]
+ << " nodeAfterNext1 = " << tTcConptr.p->nodeAfterNext[1]
+ << endl;
+ ndbout << " nextTcConnectrec = " << tTcConptr.p->nextTcConnectrec
+ << " nextTc = " << tTcConptr.p->nextTc
+ << " nextTcLogQueue = " << tTcConptr.p->nextTcLogQueue
+ << " nextLogTcrec = " << tTcConptr.p->nextLogTcrec
+ << endl;
+ ndbout << " nextHashRec = " << tTcConptr.p->nextHashRec
+ << " logWriteState = " << tTcConptr.p->logWriteState
+ << " logStartFileNo = " << tTcConptr.p->logStartFileNo
+ << " listState = " << tTcConptr.p->listState
+ << endl;
+ ndbout << " lastAttrinbuf = " << tTcConptr.p->lastAttrinbuf
+ << " lastTupkeybuf = " << tTcConptr.p->lastTupkeybuf
+ << " hashValue = " << tTcConptr.p->hashValue
+ << endl;
+ ndbout << " gci = " << tTcConptr.p->gci
+ << " fragmentptr = " << tTcConptr.p->fragmentptr
+ << " fragmentid = " << tTcConptr.p->fragmentid
+ << " firstTupkeybuf = " << tTcConptr.p->firstTupkeybuf
+ << endl;
+ ndbout << " firstAttrinbuf = " << tTcConptr.p->firstAttrinbuf
+ << " currTupAiLen = " << tTcConptr.p->currTupAiLen
+ << " currReclenAi = " << tTcConptr.p->currReclenAi
+ << endl;
+ ndbout << " tcTimer = " << tTcConptr.p->tcTimer
+ << " clientConnectrec = " << tTcConptr.p->clientConnectrec
+ << " applOprec = " << hex << tTcConptr.p->applOprec
+ << " abortState = " << tTcConptr.p->abortState
+ << endl;
+ ndbout << " transid0 = " << hex << tTcConptr.p->transid[0]
+ << " transid1 = " << hex << tTcConptr.p->transid[1]
+ << " tupkeyData0 = " << tTcConptr.p->tupkeyData[0]
+ << " tupkeyData1 = " << tTcConptr.p->tupkeyData[1]
+ << endl;
+ ndbout << " tupkeyData2 = " << tTcConptr.p->tupkeyData[2]
+ << " tupkeyData3 = " << tTcConptr.p->tupkeyData[3]
+ << endl;
+ switch (tTcConptr.p->transactionState) {
+
+ case TcConnectionrec::SCAN_STATE_USED:
+ if (tTcConptr.p->tcScanRec < cscanrecFileSize){
+ ScanRecordPtr TscanPtr;
+ TscanPtr.i = tTcConptr.p->tcScanRec;
+ ptrCheckGuard(TscanPtr, cscanrecFileSize, scanRecord);
+ ndbout << " scanState = " << TscanPtr.p->scanState << endl;
+ //TscanPtr.p->scanAccOpPtr[16];
+ //TscanPtr.p->scanApiOpPtr[16];
+ //TscanPtr.p->scanOpLength[16];
+ //TscanPtr.p->scanLocalref[2];
+ ndbout << " copyPtr="<<TscanPtr.p->copyPtr
+ << " nextScanrec="<<TscanPtr.p->nextScanrec
+ << " scanAccPtr="<<TscanPtr.p->scanAccPtr
+ << " scanAiLength="<<TscanPtr.p->scanAiLength
+ << endl;
+ ndbout << " scanCompletedOperations="<<
+ TscanPtr.p->scanCompletedOperations
+ << " scanConcurrentOperations="<<
+ TscanPtr.p->scanConcurrentOperations
+ << " scanErrorCounter="<<TscanPtr.p->scanErrorCounter
+ << " scanLocalFragid="<<TscanPtr.p->scanLocalFragid
+ << endl;
+ ndbout << " scanSchemaVersion="<<TscanPtr.p->scanSchemaVersion
+ << " scanSearchCondFalseCount="<<
+ TscanPtr.p->scanSearchCondFalseCount
+ << " scanStoredProcId="<<TscanPtr.p->scanStoredProcId
+ << " scanTcrec="<<TscanPtr.p->scanTcrec
+ << endl;
+ ndbout << " scanType="<<TscanPtr.p->scanType
+ << " scanApiBlockref="<<TscanPtr.p->scanApiBlockref
+ << " scanNodeId="<<TscanPtr.p->scanNodeId
+ << " scanCompletedStatus="<<TscanPtr.p->scanCompletedStatus
+ << endl;
+ ndbout << " scanFlag="<<TscanPtr.p->scanFlag
+ << " scanLockHold="<<TscanPtr.p->scanLockHold
+ << " scanLockMode="<<TscanPtr.p->scanLockMode
+ << " scanNumber="<<TscanPtr.p->scanNumber
+ << endl;
+ ndbout << " scanReleaseCounter="<<TscanPtr.p->scanReleaseCounter
+ << " scanTcWaiting="<<TscanPtr.p->scanTcWaiting
+ << " scanKeyinfoFlag="<<TscanPtr.p->scanKeyinfoFlag
+ << endl;
+ }else{
+ ndbout << "No connected scan record found" << endl;
+ }
+ break;
+ default:
+ break;
+ }//switch
+
+ // Reset the timer
+ tTcConptr.p->tcTimer = 0;
+ }//if
+ }//for
+#endif
+#ifdef VM_TRACE
+ for (lfoPtr.i = 0; lfoPtr.i < clfoFileSize; lfoPtr.i++) {
+ ptrAss(lfoPtr, logFileOperationRecord);
+ if ((lfoPtr.p->lfoTimer != 0) &&
+ ((lfoPtr.p->lfoTimer + 120) < cLqhTimeOutCount)) {
+ ndbout << "We have lost LFO record" << endl;
+ ndbout << "index = " << lfoPtr.i;
+ ndbout << "State = " << lfoPtr.p->lfoState;
+ ndbout << " Page No = " << lfoPtr.p->lfoPageNo;
+ ndbout << " noPagesRw = " << lfoPtr.p->noPagesRw;
+ ndbout << "lfoWordWritten = " << lfoPtr.p->lfoWordWritten << endl;
+ lfoPtr.p->lfoTimer = cLqhTimeOutCount;
+ }//if
+ }//for
+
+#endif
+
+#if 0
+ LcpRecordPtr TlcpPtr;
+ // Print information about the current local checkpoint
+ TlcpPtr.i = 0;
+ ptrAss(TlcpPtr, lcpRecord);
+ ndbout << "Information about LCP in this LQH" << endl
+ << " lcpState="<<TlcpPtr.p->lcpState<<endl
+ << " firstLcpLocAcc="<<TlcpPtr.p->firstLcpLocAcc<<endl
+ << " firstLcpLocTup="<<TlcpPtr.p->firstLcpLocTup<<endl
+ << " lcpAccptr="<<TlcpPtr.p->lcpAccptr<<endl
+ << " lastFragmentFlag="<<TlcpPtr.p->lastFragmentFlag<<endl
+ << " lcpQueued="<<TlcpPtr.p->lcpQueued<<endl
+ << " reportEmptyref="<< TlcpPtr.p->reportEmptyRef<<endl
+ << " reportEmpty="<<TlcpPtr.p->reportEmpty<<endl;
+#endif
+}//Dblqh::execTIME_SIGNAL()
+
+/* ######################################################################### */
+/* ####### EXECUTION MODULE ####### */
+/* THIS MODULE HANDLES THE RECEPTION OF LQHKEYREQ AND ALL PROCESSING */
+/* OF OPERATIONS ON BEHALF OF THIS REQUEST. THIS DOES ALSO INVOLVE */
+/* RECEPTION OF VARIOUS TYPES OF ATTRINFO AND KEYINFO. IT DOES ALSO */
+/* INVOLVE COMMUNICATION WITH ACC AND TUP. */
+/* ######################################################################### */
+
+void Dblqh::noFreeRecordLab(Signal* signal,
+ const LqhKeyReq * lqhKeyReq,
+ Uint32 errCode)
+{
+ jamEntry();
+ const Uint32 transid1 = lqhKeyReq->transId1;
+ const Uint32 transid2 = lqhKeyReq->transId2;
+ const Uint32 reqInfo = lqhKeyReq->requestInfo;
+
+ if(errCode == ZNO_FREE_MARKER_RECORDS_ERROR ||
+ errCode == ZNODE_SHUTDOWN_IN_PROGESS){
+ releaseTcrec(signal, tcConnectptr);
+ }
+
+ if (LqhKeyReq::getSimpleFlag(reqInfo) &&
+ LqhKeyReq::getOperation(reqInfo) == ZREAD){
+ jam();
+ ndbrequire(LqhKeyReq::getApplicationAddressFlag(reqInfo));
+ const Uint32 apiRef = lqhKeyReq->variableData[0];
+ const Uint32 apiOpRec = lqhKeyReq->variableData[1];
+
+ TcKeyRef * const tcKeyRef = (TcKeyRef *) signal->getDataPtrSend();
+
+ tcKeyRef->connectPtr = apiOpRec;
+ tcKeyRef->transId[0] = transid1;
+ tcKeyRef->transId[1] = transid2;
+ tcKeyRef->errorCode = errCode;
+ sendSignal(apiRef, GSN_TCKEYREF, signal, TcKeyRef::SignalLength, JBB);
+ } else {
+ jam();
+
+ const Uint32 clientPtr = lqhKeyReq->clientConnectPtr;
+ Uint32 TcOprec = clientPtr;
+ if(LqhKeyReq::getSameClientAndTcFlag(reqInfo) == 1){
+ if(LqhKeyReq::getApplicationAddressFlag(reqInfo))
+ TcOprec = lqhKeyReq->variableData[2];
+ else
+ TcOprec = lqhKeyReq->variableData[0];
+ }
+
+ LqhKeyRef * const ref = (LqhKeyRef*)signal->getDataPtrSend();
+ ref->userRef = clientPtr;
+ ref->connectPtr = TcOprec;
+ ref->errorCode = errCode;
+ ref->transId1 = transid1;
+ ref->transId2 = transid2;
+ sendSignal(signal->senderBlockRef(), GSN_LQHKEYREF, signal,
+ LqhKeyRef::SignalLength, JBB);
+ }//if
+ return;
+}//Dblqh::noFreeRecordLab()
+
+void Dblqh::LQHKEY_abort(Signal* signal, int errortype)
+{
+ switch (errortype) {
+ case 0:
+ jam();
+ terrorCode = ZCOPY_NODE_ERROR;
+ break;
+ case 1:
+ jam();
+ terrorCode = ZNO_FREE_LQH_CONNECTION;
+ break;
+ case 2:
+ jam();
+ terrorCode = signal->theData[1];
+ break;
+ case 3:
+ jam();
+ ndbrequire((tcConnectptr.p->transactionState == TcConnectionrec::WAIT_ACC_ABORT) ||
+ (tcConnectptr.p->transactionState == TcConnectionrec::ABORT_STOPPED) ||
+ (tcConnectptr.p->transactionState == TcConnectionrec::ABORT_QUEUED));
+ return;
+ break;
+ case 4:
+ jam();
+ if(tabptr.p->tableStatus == Tablerec::NOT_DEFINED){
+ jam();
+ terrorCode = ZTABLE_NOT_DEFINED;
+ } else if (tabptr.p->tableStatus == Tablerec::PREP_DROP_TABLE_ONGOING ||
+ tabptr.p->tableStatus == Tablerec::PREP_DROP_TABLE_DONE){
+ jam();
+ terrorCode = ZDROP_TABLE_IN_PROGRESS;
+ } else {
+ ndbrequire(0);
+ }
+ break;
+ case 5:
+ jam();
+ terrorCode = ZINVALID_SCHEMA_VERSION;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ abortErrorLab(signal);
+}//Dblqh::LQHKEY_abort()
+
+void Dblqh::LQHKEY_error(Signal* signal, int errortype)
+{
+ switch (errortype) {
+ case 0:
+ jam();
+ break;
+ case 1:
+ jam();
+ break;
+ case 2:
+ jam();
+ break;
+ case 3:
+ jam();
+ break;
+ case 4:
+ jam();
+ break;
+ case 5:
+ jam();
+ break;
+ case 6:
+ jam();
+ break;
+ default:
+ jam();
+ break;
+ }//switch
+ ndbrequire(false);
+}//Dblqh::LQHKEY_error()
+
+void Dblqh::execLQHKEYREF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ terrorCode = signal->theData[2];
+ Uint32 transid1 = signal->theData[3];
+ Uint32 transid2 = signal->theData[4];
+ if (tcConnectptr.i >= ctcConnectrecFileSize) {
+ errorReport(signal, 3);
+ return;
+ }//if
+/*------------------------------------------------------------------*/
+/* WE HAVE TO CHECK THAT THE SIGNAL DO NOT BELONG TO SOMETHING*/
+/* REMOVED DUE TO A TIME-OUT. */
+/*------------------------------------------------------------------*/
+ ptrAss(tcConnectptr, tcConnectionrec);
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ switch (regTcPtr->connectState) {
+ case TcConnectionrec::CONNECTED:
+ jam();
+ if ((regTcPtr->transid[0] != transid1) ||
+ (regTcPtr->transid[1] != transid2)) {
+ warningReport(signal, 14);
+ return;
+ }//if
+ if (regTcPtr->abortState != TcConnectionrec::ABORT_IDLE) {
+ warningReport(signal, 15);
+ return;
+ }//if
+ abortErrorLab(signal);
+ return;
+ break;
+ case TcConnectionrec::LOG_CONNECTED:
+ jam();
+ logLqhkeyrefLab(signal);
+ return;
+ break;
+ case TcConnectionrec::COPY_CONNECTED:
+ jam();
+ copyLqhKeyRefLab(signal);
+ return;
+ break;
+ default:
+ warningReport(signal, 16);
+ return;
+ break;
+ }//switch
+}//Dblqh::execLQHKEYREF()
+
+/* -------------------------------------------------------------------------- */
+/* ------- ENTER PACKED_SIGNAL ------- */
+/* Execution of packed signal. The packed signal can contain COMMIT, COMPLETE */
+/* or LQHKEYCONF signals. These signals will be executed by their resp. exec */
+/* functions. */
+/* -------------------------------------------------------------------------- */
+void Dblqh::execPACKED_SIGNAL(Signal* signal)
+{
+ Uint32 Tstep = 0;
+ Uint32 Tlength;
+ Uint32 TpackedData[28];
+ Uint32 sig0, sig1, sig2, sig3 ,sig4, sig5, sig6;
+
+ jamEntry();
+ Tlength = signal->length();
+ ndbrequire(Tlength <= 25);
+ MEMCOPY_NO_WORDS(&TpackedData[0], &signal->theData[0], Tlength);
+ while (Tlength > Tstep) {
+ switch (TpackedData[Tstep] >> 28) {
+ case ZCOMMIT:
+ jam();
+ sig0 = TpackedData[Tstep + 0] & 0x0FFFFFFF;
+ sig1 = TpackedData[Tstep + 1];
+ sig2 = TpackedData[Tstep + 2];
+ sig3 = TpackedData[Tstep + 3];
+ signal->theData[0] = sig0;
+ signal->theData[1] = sig1;
+ signal->theData[2] = sig2;
+ signal->theData[3] = sig3;
+ signal->header.theLength = 4;
+ execCOMMIT(signal);
+ Tstep += 4;
+ break;
+ case ZCOMPLETE:
+ jam();
+ sig0 = TpackedData[Tstep + 0] & 0x0FFFFFFF;
+ sig1 = TpackedData[Tstep + 1];
+ sig2 = TpackedData[Tstep + 2];
+ signal->theData[0] = sig0;
+ signal->theData[1] = sig1;
+ signal->theData[2] = sig2;
+ signal->header.theLength = 3;
+ execCOMPLETE(signal);
+ Tstep += 3;
+ break;
+ case ZLQHKEYCONF: {
+ jam();
+ LqhKeyConf * const lqhKeyConf = (LqhKeyConf *)signal->getDataPtr();
+
+ sig0 = TpackedData[Tstep + 0] & 0x0FFFFFFF;
+ sig1 = TpackedData[Tstep + 1];
+ sig2 = TpackedData[Tstep + 2];
+ sig3 = TpackedData[Tstep + 3];
+ sig4 = TpackedData[Tstep + 4];
+ sig5 = TpackedData[Tstep + 5];
+ sig6 = TpackedData[Tstep + 6];
+ lqhKeyConf->connectPtr = sig0;
+ lqhKeyConf->opPtr = sig1;
+ lqhKeyConf->userRef = sig2;
+ lqhKeyConf->readLen = sig3;
+ lqhKeyConf->transId1 = sig4;
+ lqhKeyConf->transId2 = sig5;
+ lqhKeyConf->noFiredTriggers = sig6;
+ execLQHKEYCONF(signal);
+ Tstep += LqhKeyConf::SignalLength;
+ break;
+ }
+ case ZREMOVE_MARKER:
+ jam();
+ sig0 = TpackedData[Tstep + 1];
+ sig1 = TpackedData[Tstep + 2];
+ signal->theData[0] = sig0;
+ signal->theData[1] = sig1;
+ signal->header.theLength = 2;
+ execREMOVE_MARKER_ORD(signal);
+ Tstep += 3;
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ }//switch
+ }//while
+ ndbrequire(Tlength == Tstep);
+ return;
+}//Dblqh::execPACKED_SIGNAL()
+
+void
+Dblqh::execREMOVE_MARKER_ORD(Signal* signal)
+{
+ CommitAckMarker key;
+ key.transid1 = signal->theData[0];
+ key.transid2 = signal->theData[1];
+ jamEntry();
+
+ CommitAckMarkerPtr removedPtr;
+ m_commitAckMarkerHash.release(removedPtr, key);
+ ndbrequire(removedPtr.i != RNIL);
+}
+
+
+/* -------------------------------------------------------------------------- */
+/* ------- ENTER SEND_PACKED ------- */
+/* Used to force a packed signal to be sent if local signal buffer is not */
+/* empty. */
+/* -------------------------------------------------------------------------- */
+void Dblqh::execSEND_PACKED(Signal* signal)
+{
+ HostRecordPtr Thostptr;
+ UintR i;
+ UintR TpackedListIndex = cpackedListIndex;
+ jamEntry();
+ for (i = 0; i < TpackedListIndex; i++) {
+ Thostptr.i = cpackedList[i];
+ ptrAss(Thostptr, hostRecord);
+ jam();
+ ndbrequire(Thostptr.i - 1 < MAX_NDB_NODES - 1);
+ if (Thostptr.p->noOfPackedWordsLqh > 0) {
+ jam();
+ sendPackedSignalLqh(signal, Thostptr.p);
+ }//if
+ if (Thostptr.p->noOfPackedWordsTc > 0) {
+ jam();
+ sendPackedSignalTc(signal, Thostptr.p);
+ }//if
+ Thostptr.p->inPackedList = false;
+ }//for
+ cpackedListIndex = 0;
+ return;
+}//Dblqh::execSEND_PACKED()
+
+void
+Dblqh::updatePackedList(Signal* signal, HostRecord * ahostptr, Uint16 hostId)
+{
+ Uint32 TpackedListIndex = cpackedListIndex;
+ if (ahostptr->inPackedList == false) {
+ jam();
+ ahostptr->inPackedList = true;
+ cpackedList[TpackedListIndex] = hostId;
+ cpackedListIndex = TpackedListIndex + 1;
+ }//if
+}//Dblqh::updatePackedList()
+
+/* ************>> */
+/* TUPKEYCONF > */
+/* ************>> */
+void Dblqh::execTUPKEYCONF(Signal* signal)
+{
+ TcConnectionrec *regTcConnectionrec = tcConnectionrec;
+ Uint32 ttcConnectrecFileSize = ctcConnectrecFileSize;
+ const TupKeyConf * const tupKeyConf = (TupKeyConf *)signal->getDataPtr();
+ Uint32 tcIndex = tupKeyConf->userPtr;
+ jamEntry();
+ tcConnectptr.i = tcIndex;
+ ptrCheckGuard(tcConnectptr, ttcConnectrecFileSize, regTcConnectionrec);
+ if (tcConnectptr.p->seqNoReplica == 0) // Primary replica
+ tcConnectptr.p->noFiredTriggers = tupKeyConf->noFiredTriggers;
+ switch (tcConnectptr.p->transactionState) {
+ case TcConnectionrec::WAIT_TUP:
+ jam();
+ tupkeyConfLab(signal);
+ break;
+ case TcConnectionrec::COPY_TUPKEY:
+ jam();
+ copyTupkeyConfLab(signal);
+ break;
+ case TcConnectionrec::SCAN_TUPKEY:
+ jam();
+ scanTupkeyConfLab(signal);
+ break;
+ case TcConnectionrec::WAIT_TUP_TO_ABORT:
+ jam();
+/* ------------------------------------------------------------------------- */
+// Abort was not ready to start until this signal came back. Now we are ready
+// to start the abort.
+/* ------------------------------------------------------------------------- */
+ releaseActiveFrag(signal);
+ abortCommonLab(signal);
+ break;
+ case TcConnectionrec::WAIT_ACC_ABORT:
+ case TcConnectionrec::ABORT_QUEUED:
+ jam();
+/* -------------------------------------------------------------------------- */
+/* IGNORE SINCE ABORT OF THIS OPERATION IS ONGOING ALREADY. */
+/* -------------------------------------------------------------------------- */
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+}//Dblqh::execTUPKEYCONF()
+
+/* ************> */
+/* TUPKEYREF > */
+/* ************> */
+void Dblqh::execTUPKEYREF(Signal* signal)
+{
+ const TupKeyRef * const tupKeyRef = (TupKeyRef *)signal->getDataPtr();
+
+ jamEntry();
+ tcConnectptr.i = tupKeyRef->userRef;
+ terrorCode = tupKeyRef->errorCode;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ switch (tcConnectptr.p->transactionState) {
+ case TcConnectionrec::WAIT_TUP:
+ jam();
+ releaseActiveFrag(signal);
+ abortErrorLab(signal);
+ break;
+ case TcConnectionrec::COPY_TUPKEY:
+ ndbrequire(false);
+ break;
+ case TcConnectionrec::SCAN_TUPKEY:
+ jam();
+ scanTupkeyRefLab(signal);
+ break;
+ case TcConnectionrec::WAIT_TUP_TO_ABORT:
+ jam();
+/* ------------------------------------------------------------------------- */
+// Abort was not ready to start until this signal came back. Now we are ready
+// to start the abort.
+/* ------------------------------------------------------------------------- */
+ releaseActiveFrag(signal);
+ abortCommonLab(signal);
+ break;
+ case TcConnectionrec::WAIT_ACC_ABORT:
+ case TcConnectionrec::ABORT_QUEUED:
+ jam();
+/* ------------------------------------------------------------------------- */
+/* IGNORE SINCE ABORT OF THIS OPERATION IS ONGOING ALREADY. */
+/* ------------------------------------------------------------------------- */
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+}//Dblqh::execTUPKEYREF()
+
+void Dblqh::sendPackedSignalLqh(Signal* signal, HostRecord * ahostptr)
+{
+ Uint32 noOfWords = ahostptr->noOfPackedWordsLqh;
+ BlockReference hostRef = ahostptr->hostLqhBlockRef;
+ MEMCOPY_NO_WORDS(&signal->theData[0],
+ &ahostptr->packedWordsLqh[0],
+ noOfWords);
+ sendSignal(hostRef, GSN_PACKED_SIGNAL, signal, noOfWords, JBB);
+ ahostptr->noOfPackedWordsLqh = 0;
+}//Dblqh::sendPackedSignalLqh()
+
+void Dblqh::sendPackedSignalTc(Signal* signal, HostRecord * ahostptr)
+{
+ Uint32 noOfWords = ahostptr->noOfPackedWordsTc;
+ BlockReference hostRef = ahostptr->hostTcBlockRef;
+ MEMCOPY_NO_WORDS(&signal->theData[0],
+ &ahostptr->packedWordsTc[0],
+ noOfWords);
+ sendSignal(hostRef, GSN_PACKED_SIGNAL, signal, noOfWords, JBB);
+ ahostptr->noOfPackedWordsTc = 0;
+}//Dblqh::sendPackedSignalTc()
+
+void Dblqh::sendCommitLqh(Signal* signal, BlockReference alqhBlockref)
+{
+ HostRecordPtr Thostptr;
+ Thostptr.i = refToNode(alqhBlockref);
+ ptrCheckGuard(Thostptr, chostFileSize, hostRecord);
+ if (Thostptr.p->noOfPackedWordsLqh > 21) {
+ jam();
+ sendPackedSignalLqh(signal, Thostptr.p);
+ } else {
+ jam();
+ updatePackedList(signal, Thostptr.p, Thostptr.i);
+ }//if
+ Uint32 pos = Thostptr.p->noOfPackedWordsLqh;
+ Uint32 ptrAndType = tcConnectptr.p->clientConnectrec | (ZCOMMIT << 28);
+ Uint32 gci = tcConnectptr.p->gci;
+ Uint32 transid1 = tcConnectptr.p->transid[0];
+ Uint32 transid2 = tcConnectptr.p->transid[1];
+ Thostptr.p->packedWordsLqh[pos] = ptrAndType;
+ Thostptr.p->packedWordsLqh[pos + 1] = gci;
+ Thostptr.p->packedWordsLqh[pos + 2] = transid1;
+ Thostptr.p->packedWordsLqh[pos + 3] = transid2;
+ Thostptr.p->noOfPackedWordsLqh = pos + 4;
+}//Dblqh::sendCommitLqh()
+
+void Dblqh::sendCompleteLqh(Signal* signal, BlockReference alqhBlockref)
+{
+ HostRecordPtr Thostptr;
+ Thostptr.i = refToNode(alqhBlockref);
+ ptrCheckGuard(Thostptr, chostFileSize, hostRecord);
+ if (Thostptr.p->noOfPackedWordsLqh > 22) {
+ jam();
+ sendPackedSignalLqh(signal, Thostptr.p);
+ } else {
+ jam();
+ updatePackedList(signal, Thostptr.p, Thostptr.i);
+ }//if
+ Uint32 pos = Thostptr.p->noOfPackedWordsLqh;
+ Uint32 ptrAndType = tcConnectptr.p->clientConnectrec | (ZCOMPLETE << 28);
+ Uint32 transid1 = tcConnectptr.p->transid[0];
+ Uint32 transid2 = tcConnectptr.p->transid[1];
+ Thostptr.p->packedWordsLqh[pos] = ptrAndType;
+ Thostptr.p->packedWordsLqh[pos + 1] = transid1;
+ Thostptr.p->packedWordsLqh[pos + 2] = transid2;
+ Thostptr.p->noOfPackedWordsLqh = pos + 3;
+}//Dblqh::sendCompleteLqh()
+
+void Dblqh::sendCommittedTc(Signal* signal, BlockReference atcBlockref)
+{
+ HostRecordPtr Thostptr;
+ Thostptr.i = refToNode(atcBlockref);
+ ptrCheckGuard(Thostptr, chostFileSize, hostRecord);
+ if (Thostptr.p->noOfPackedWordsTc > 22) {
+ jam();
+ sendPackedSignalTc(signal, Thostptr.p);
+ } else {
+ jam();
+ updatePackedList(signal, Thostptr.p, Thostptr.i);
+ }//if
+ Uint32 pos = Thostptr.p->noOfPackedWordsTc;
+ Uint32 ptrAndType = tcConnectptr.p->clientConnectrec | (ZCOMMITTED << 28);
+ Uint32 transid1 = tcConnectptr.p->transid[0];
+ Uint32 transid2 = tcConnectptr.p->transid[1];
+ Thostptr.p->packedWordsTc[pos] = ptrAndType;
+ Thostptr.p->packedWordsTc[pos + 1] = transid1;
+ Thostptr.p->packedWordsTc[pos + 2] = transid2;
+ Thostptr.p->noOfPackedWordsTc = pos + 3;
+}//Dblqh::sendCommittedTc()
+
+void Dblqh::sendCompletedTc(Signal* signal, BlockReference atcBlockref)
+{
+ HostRecordPtr Thostptr;
+ Thostptr.i = refToNode(atcBlockref);
+ ptrCheckGuard(Thostptr, chostFileSize, hostRecord);
+ if (Thostptr.p->noOfPackedWordsTc > 22) {
+ jam();
+ sendPackedSignalTc(signal, Thostptr.p);
+ } else {
+ jam();
+ updatePackedList(signal, Thostptr.p, Thostptr.i);
+ }//if
+ Uint32 pos = Thostptr.p->noOfPackedWordsTc;
+ Uint32 ptrAndType = tcConnectptr.p->clientConnectrec | (ZCOMPLETED << 28);
+ Uint32 transid1 = tcConnectptr.p->transid[0];
+ Uint32 transid2 = tcConnectptr.p->transid[1];
+ Thostptr.p->packedWordsTc[pos] = ptrAndType;
+ Thostptr.p->packedWordsTc[pos + 1] = transid1;
+ Thostptr.p->packedWordsTc[pos + 2] = transid2;
+ Thostptr.p->noOfPackedWordsTc = pos + 3;
+}//Dblqh::sendCompletedTc()
+
+void Dblqh::sendLqhkeyconfTc(Signal* signal, BlockReference atcBlockref)
+{
+ LqhKeyConf* lqhKeyConf;
+ HostRecordPtr Thostptr;
+
+ Thostptr.i = refToNode(atcBlockref);
+ ptrCheckGuard(Thostptr, chostFileSize, hostRecord);
+ if (refToBlock(atcBlockref) == DBTC) {
+ jam();
+/*******************************************************************
+// This signal was intended for DBTC as part of the normal transaction
+// execution.
+********************************************************************/
+ if (Thostptr.p->noOfPackedWordsTc > (25 - LqhKeyConf::SignalLength)) {
+ jam();
+ sendPackedSignalTc(signal, Thostptr.p);
+ } else {
+ jam();
+ updatePackedList(signal, Thostptr.p, Thostptr.i);
+ }//if
+ lqhKeyConf = (LqhKeyConf *)
+ &Thostptr.p->packedWordsTc[Thostptr.p->noOfPackedWordsTc];
+ Thostptr.p->noOfPackedWordsTc += LqhKeyConf::SignalLength;
+ } else {
+ jam();
+/*******************************************************************
+// This signal was intended for DBLQH as part of log execution or
+// node recovery.
+********************************************************************/
+ if (Thostptr.p->noOfPackedWordsLqh > (25 - LqhKeyConf::SignalLength)) {
+ jam();
+ sendPackedSignalLqh(signal, Thostptr.p);
+ } else {
+ jam();
+ updatePackedList(signal, Thostptr.p, Thostptr.i);
+ }//if
+ lqhKeyConf = (LqhKeyConf *)
+ &Thostptr.p->packedWordsLqh[Thostptr.p->noOfPackedWordsLqh];
+ Thostptr.p->noOfPackedWordsLqh += LqhKeyConf::SignalLength;
+ }//if
+ Uint32 ptrAndType = tcConnectptr.i | (ZLQHKEYCONF << 28);
+ Uint32 tcOprec = tcConnectptr.p->tcOprec;
+ Uint32 ownRef = cownref;
+ Uint32 readlenAi = tcConnectptr.p->readlenAi;
+ Uint32 transid1 = tcConnectptr.p->transid[0];
+ Uint32 transid2 = tcConnectptr.p->transid[1];
+ Uint32 noFiredTriggers = tcConnectptr.p->noFiredTriggers;
+ lqhKeyConf->connectPtr = ptrAndType;
+ lqhKeyConf->opPtr = tcOprec;
+ lqhKeyConf->userRef = ownRef;
+ lqhKeyConf->readLen = readlenAi;
+ lqhKeyConf->transId1 = transid1;
+ lqhKeyConf->transId2 = transid2;
+ lqhKeyConf->noFiredTriggers = noFiredTriggers;
+}//Dblqh::sendLqhkeyconfTc()
+
+/* ************************************************************************>>
+ * KEYINFO: Get tuple request from DBTC. Next step is to contact DBACC to get
+ * key to tuple if all key/attrinfo has been received, else for more attrinfo
+ * signals.
+ * ************************************************************************>> */
+void Dblqh::execKEYINFO(Signal* signal)
+{
+ Uint32 tcOprec = signal->theData[0];
+ Uint32 transid1 = signal->theData[1];
+ Uint32 transid2 = signal->theData[2];
+ jamEntry();
+ if (findTransaction(transid1, transid2, tcOprec) != ZOK) {
+ jam();
+ return;
+ }//if
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->transactionState !=
+ TcConnectionrec::WAIT_TUPKEYINFO) {
+ jam();
+/*****************************************************************************/
+/* TRANSACTION WAS ABORTED, THIS IS MOST LIKELY A SIGNAL BELONGING TO THE */
+/* ABORTED TRANSACTION. THUS IGNORE THE SIGNAL. */
+/*****************************************************************************/
+ return;
+ }//if
+ Uint32 errorCode = handleLongTupKey(signal,
+ (Uint32)regTcPtr->save1,
+ (Uint32)regTcPtr->primKeyLen,
+ &signal->theData[3]);
+ if (errorCode != 0) {
+ if (errorCode == 1) {
+ jam();
+ return;
+ }//if
+ jam();
+ terrorCode = errorCode;
+ abortErrorLab(signal);
+ return;
+ }//if
+ FragrecordPtr regFragptr;
+ regFragptr.i = regTcPtr->fragmentptr;
+ ptrCheckGuard(regFragptr, cfragrecFileSize, fragrecord);
+ fragptr = regFragptr;
+ endgettupkeyLab(signal);
+ return;
+}//Dblqh::execKEYINFO()
+
+/* ------------------------------------------------------------------------- */
+/* FILL IN KEY DATA INTO DATA BUFFERS. */
+/* ------------------------------------------------------------------------- */
+Uint32 Dblqh::handleLongTupKey(Signal* signal,
+ Uint32 keyLength,
+ Uint32 primKeyLength,
+ Uint32* dataPtr)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ Uint32 dataPos = 0;
+ while (true) {
+ keyLength += 4;
+ if (cfirstfreeDatabuf == RNIL) {
+ jam();
+ return ZGET_DATAREC_ERROR;
+ }//if
+ seizeTupkeybuf(signal);
+ Databuf * const regDataPtr = databufptr.p;
+ Uint32 data0 = dataPtr[dataPos];
+ Uint32 data1 = dataPtr[dataPos + 1];
+ Uint32 data2 = dataPtr[dataPos + 2];
+ Uint32 data3 = dataPtr[dataPos + 3];
+ regDataPtr->data[0] = data0;
+ regDataPtr->data[1] = data1;
+ regDataPtr->data[2] = data2;
+ regDataPtr->data[3] = data3;
+ dataPos += 4;
+ if (keyLength < primKeyLength) {
+ if (dataPos > 16) {
+ jam();
+/* SAVE STATE AND WAIT FOR KEYINFO */
+ regTcPtr->save1 = keyLength;
+ return 1;
+ }//if
+ } else {
+ jam();
+ return 0;
+ }//if
+ }//while
+}//Dblqh::handleLongTupKey()
+
+/* ------------------------------------------------------------------------- */
+/* ------- HANDLE ATTRINFO SIGNALS ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+/* ************************************************************************>> */
+/* ATTRINFO: Continuation of KEYINFO signal (except for scans that do not use*/
+/* any KEYINFO). When all key and attribute info is received we contact DBACC*/
+/* for index handling. */
+/* ************************************************************************>> */
+void Dblqh::execATTRINFO(Signal* signal)
+{
+ Uint32 tcOprec = signal->theData[0];
+ Uint32 transid1 = signal->theData[1];
+ Uint32 transid2 = signal->theData[2];
+ jamEntry();
+ if (findTransaction(transid1,
+ transid2,
+ tcOprec) != ZOK) {
+ jam();
+ return;
+ }//if
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ Uint32 length = signal->length() - 3;
+ Uint32 totReclenAi = regTcPtr->totReclenAi;
+ Uint32 currReclenAi = regTcPtr->currReclenAi + length;
+ Uint32* dataPtr = &signal->theData[3];
+ regTcPtr->currReclenAi = currReclenAi;
+ if (totReclenAi == currReclenAi) {
+ switch (regTcPtr->transactionState) {
+ case TcConnectionrec::WAIT_ATTR:
+ {
+ Fragrecord *regFragrecord = fragrecord;
+ Uint32 fragIndex = regTcPtr->fragmentptr;
+ Uint32 tfragrecFileSize = cfragrecFileSize;
+ jam();
+ fragptr.i = fragIndex;
+ ptrCheckGuard(fragptr, tfragrecFileSize, regFragrecord);
+ lqhAttrinfoLab(signal, dataPtr, length);
+ endgettupkeyLab(signal);
+ return;
+ break;
+ }
+ case TcConnectionrec::WAIT_SCAN_AI:
+ jam();
+ scanAttrinfoLab(signal, dataPtr, length);
+ return;
+ break;
+ case TcConnectionrec::WAIT_TUP_TO_ABORT:
+ case TcConnectionrec::LOG_ABORT_QUEUED:
+ case TcConnectionrec::ABORT_QUEUED:
+ case TcConnectionrec::ABORT_STOPPED:
+ case TcConnectionrec::WAIT_ACC_ABORT:
+ case TcConnectionrec::WAIT_AI_AFTER_ABORT:
+ jam();
+ aiStateErrorCheckLab(signal, dataPtr,length);
+ return;
+ break;
+ default:
+ jam();
+ ndbrequire(regTcPtr->abortState != TcConnectionrec::ABORT_IDLE);
+ break;
+ }//switch
+ } else if (currReclenAi < totReclenAi) {
+ jam();
+ switch (regTcPtr->transactionState) {
+ case TcConnectionrec::WAIT_ATTR:
+ jam();
+ lqhAttrinfoLab(signal, dataPtr, length);
+ return;
+ break;
+ case TcConnectionrec::WAIT_SCAN_AI:
+ jam();
+ scanAttrinfoLab(signal, dataPtr, length);
+ return;
+ break;
+ case TcConnectionrec::WAIT_TUP_TO_ABORT:
+ case TcConnectionrec::LOG_ABORT_QUEUED:
+ case TcConnectionrec::ABORT_QUEUED:
+ case TcConnectionrec::ABORT_STOPPED:
+ case TcConnectionrec::WAIT_ACC_ABORT:
+ case TcConnectionrec::WAIT_AI_AFTER_ABORT:
+ jam();
+ aiStateErrorCheckLab(signal, dataPtr, length);
+ return;
+ break;
+ default:
+ jam();
+ ndbrequire(regTcPtr->abortState != TcConnectionrec::ABORT_IDLE);
+ break;
+ }//switch
+ } else {
+ switch (regTcPtr->transactionState) {
+ case TcConnectionrec::WAIT_SCAN_AI:
+ jam();
+ scanAttrinfoLab(signal, dataPtr, length);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ }//if
+ return;
+}//Dblqh::execATTRINFO()
+
+/* ************************************************************************>> */
+/* TUP_ATTRINFO: Interpreted execution in DBTUP generates redo-log info */
+/* which is sent back to DBLQH for logging. This is because the decision */
+/* to execute or not is made in DBTUP and thus we cannot start logging until */
+/* DBTUP part has been run. */
+/* ************************************************************************>> */
+void Dblqh::execTUP_ATTRINFO(Signal* signal)
+{
+ TcConnectionrec *regTcConnectionrec = tcConnectionrec;
+ Uint32 length = signal->length() - 3;
+ Uint32 tcIndex = signal->theData[0];
+ Uint32 ttcConnectrecFileSize = ctcConnectrecFileSize;
+ jamEntry();
+ tcConnectptr.i = tcIndex;
+ ptrCheckGuard(tcConnectptr, ttcConnectrecFileSize, regTcConnectionrec);
+ ndbrequire(tcConnectptr.p->transactionState == TcConnectionrec::WAIT_TUP);
+ if (saveTupattrbuf(signal, &signal->theData[3], length) == ZOK) {
+ return;
+ } else {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* WE ARE WAITING FOR RESPONSE FROM TUP HERE. THUS WE NEED TO */
+/* GO THROUGH THE STATE MACHINE FOR THE OPERATION. */
+/* ------------------------------------------------------------------------- */
+ localAbortStateHandlerLab(signal);
+ }//if
+}//Dblqh::execTUP_ATTRINFO()
+
+/* ------------------------------------------------------------------------- */
+/* ------- HANDLE ATTRINFO FROM LQH ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::lqhAttrinfoLab(Signal* signal, Uint32* dataPtr, Uint32 length)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->operation != ZREAD) {
+ if (regTcPtr->opExec != 1) {
+ if (saveTupattrbuf(signal, dataPtr, length) == ZOK) {
+ ;
+ } else {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* WE MIGHT BE WAITING FOR RESPONSE FROM SOME BLOCK HERE. THUS WE NEED TO */
+/* GO THROUGH THE STATE MACHINE FOR THE OPERATION. */
+/* ------------------------------------------------------------------------- */
+ localAbortStateHandlerLab(signal);
+ return;
+ }//if
+ }//if
+ }//if
+ Uint32 sig0 = regTcPtr->tupConnectrec;
+ Uint32 blockNo = refToBlock(regTcPtr->tcTupBlockref);
+ signal->theData[0] = sig0;
+ EXECUTE_DIRECT(blockNo, GSN_ATTRINFO, signal, length + 3);
+ jamEntry();
+}//Dblqh::lqhAttrinfoLab()
+
+/* ------------------------------------------------------------------------- */
+/* ------ FIND TRANSACTION BY USING HASH TABLE ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+int Dblqh::findTransaction(UintR Transid1, UintR Transid2, UintR TcOprec)
+{
+ TcConnectionrec *regTcConnectionrec = tcConnectionrec;
+ Uint32 ttcConnectrecFileSize = ctcConnectrecFileSize;
+ TcConnectionrecPtr locTcConnectptr;
+
+ Uint32 ThashIndex = (Transid1 ^ TcOprec) & 1023;
+ locTcConnectptr.i = ctransidHash[ThashIndex];
+ while (locTcConnectptr.i != RNIL) {
+ ptrCheckGuard(locTcConnectptr, ttcConnectrecFileSize, regTcConnectionrec);
+ if ((locTcConnectptr.p->transid[0] == Transid1) &&
+ (locTcConnectptr.p->transid[1] == Transid2) &&
+ (locTcConnectptr.p->tcOprec == TcOprec)) {
+/* FIRST PART OF TRANSACTION CORRECT */
+/* SECOND PART ALSO CORRECT */
+/* THE OPERATION RECORD POINTER IN TC WAS ALSO CORRECT */
+ jam();
+ tcConnectptr.i = locTcConnectptr.i;
+ tcConnectptr.p = locTcConnectptr.p;
+ return (int)ZOK;
+ }//if
+ jam();
+/* THIS WAS NOT THE TRANSACTION WHICH WAS SOUGHT */
+ locTcConnectptr.i = locTcConnectptr.p->nextHashRec;
+ }//while
+/* WE DID NOT FIND THE TRANSACTION, REPORT NOT FOUND */
+ return (int)ZNOT_FOUND;
+}//Dblqh::findTransaction()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SAVE ATTRINFO FROM TUP IN ATTRINBUF ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+int Dblqh::saveTupattrbuf(Signal* signal, Uint32* dataPtr, Uint32 length)
+{
+ Uint32 tfirstfreeAttrinbuf = cfirstfreeAttrinbuf;
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ Uint32 currTupAiLen = regTcPtr->currTupAiLen;
+ if (tfirstfreeAttrinbuf == RNIL) {
+ jam();
+ terrorCode = ZGET_ATTRINBUF_ERROR;
+ return ZGET_ATTRINBUF_ERROR;
+ }//if
+ seizeAttrinbuf(signal);
+ Attrbuf * const regAttrPtr = attrinbufptr.p;
+ MEMCOPY_NO_WORDS(&regAttrPtr->attrbuf[0], dataPtr, length);
+ regTcPtr->currTupAiLen = currTupAiLen + length;
+ regAttrPtr->attrbuf[ZINBUF_DATA_LEN] = length;
+ return ZOK;
+}//Dblqh::saveTupattrbuf()
+
+/* ==========================================================================
+ * ======= SEIZE ATTRIBUTE IN BUFFER =======
+ *
+ * GET A NEW ATTRINBUF AND SETS ATTRINBUFPTR.
+ * ========================================================================= */
+void Dblqh::seizeAttrinbuf(Signal* signal)
+{
+ AttrbufPtr tmpAttrinbufptr;
+ AttrbufPtr regAttrinbufptr;
+ Attrbuf *regAttrbuf = attrbuf;
+ Uint32 tattrinbufFileSize = cattrinbufFileSize;
+
+ regAttrinbufptr.i = cfirstfreeAttrinbuf;
+ tmpAttrinbufptr.i = tcConnectptr.p->lastAttrinbuf;
+ ptrCheckGuard(regAttrinbufptr, tattrinbufFileSize, regAttrbuf);
+ Uint32 nextFirst = regAttrinbufptr.p->attrbuf[ZINBUF_NEXT];
+ tcConnectptr.p->lastAttrinbuf = regAttrinbufptr.i;
+ regAttrinbufptr.p->attrbuf[ZINBUF_DATA_LEN] = 0;
+ if (tmpAttrinbufptr.i == RNIL) {
+ jam();
+ tcConnectptr.p->firstAttrinbuf = regAttrinbufptr.i;
+ } else {
+ jam();
+ ptrCheckGuard(tmpAttrinbufptr, tattrinbufFileSize, regAttrbuf);
+ tmpAttrinbufptr.p->attrbuf[ZINBUF_NEXT] = regAttrinbufptr.i;
+ }//if
+ regAttrinbufptr.p->attrbuf[ZINBUF_NEXT] = RNIL;
+ cfirstfreeAttrinbuf = nextFirst;
+ attrinbufptr = regAttrinbufptr;
+}//Dblqh::seizeAttrinbuf()
+
+/* ==========================================================================
+ * ======= SEIZE TC CONNECT RECORD =======
+ *
+ * GETS A NEW TC CONNECT RECORD FROM FREELIST.
+ * ========================================================================= */
+void Dblqh::seizeTcrec()
+{
+ TcConnectionrecPtr locTcConnectptr;
+
+ locTcConnectptr.i = cfirstfreeTcConrec;
+ ptrCheckGuard(locTcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ Uint32 nextTc = locTcConnectptr.p->nextTcConnectrec;
+ locTcConnectptr.p->nextTcConnectrec = RNIL;
+ locTcConnectptr.p->clientConnectrec = RNIL;
+ locTcConnectptr.p->clientBlockref = RNIL;
+ locTcConnectptr.p->abortState = TcConnectionrec::ABORT_IDLE;
+ locTcConnectptr.p->tcTimer = cLqhTimeOutCount;
+ locTcConnectptr.p->tableref = RNIL;
+ locTcConnectptr.p->savePointId = 0;
+ cfirstfreeTcConrec = nextTc;
+ tcConnectptr = locTcConnectptr;
+ locTcConnectptr.p->connectState = TcConnectionrec::CONNECTED;
+}//Dblqh::seizeTcrec()
+
+/* ==========================================================================
+ * ======= SEIZE DATA BUFFER =======
+ * ========================================================================= */
+void Dblqh::seizeTupkeybuf(Signal* signal)
+{
+ Databuf *regDatabuf = databuf;
+ DatabufPtr tmpDatabufptr;
+ DatabufPtr regDatabufptr;
+ Uint32 tdatabufFileSize = cdatabufFileSize;
+
+/* ------- GET A DATABUF. ------- */
+ regDatabufptr.i = cfirstfreeDatabuf;
+ tmpDatabufptr.i = tcConnectptr.p->lastTupkeybuf;
+ ptrCheckGuard(regDatabufptr, tdatabufFileSize, regDatabuf);
+ Uint32 nextFirst = regDatabufptr.p->nextDatabuf;
+ tcConnectptr.p->lastTupkeybuf = regDatabufptr.i;
+ if (tmpDatabufptr.i == RNIL) {
+ jam();
+ tcConnectptr.p->firstTupkeybuf = regDatabufptr.i;
+ } else {
+ jam();
+ ptrCheckGuard(tmpDatabufptr, tdatabufFileSize, regDatabuf);
+ tmpDatabufptr.p->nextDatabuf = regDatabufptr.i;
+ }//if
+ cfirstfreeDatabuf = nextFirst;
+ regDatabufptr.p->nextDatabuf = RNIL;
+ databufptr = regDatabufptr;
+}//Dblqh::seizeTupkeybuf()
+
+/* ------------------------------------------------------------------------- */
+/* ------- TAKE CARE OF LQHKEYREQ ------- */
+/* LQHKEYREQ IS THE SIGNAL THAT STARTS ALL OPERATIONS IN THE LQH BLOCK */
+/* THIS SIGNAL CONTAINS A LOT OF INFORMATION ABOUT WHAT TYPE OF OPERATION, */
+/* KEY INFORMATION, ATTRIBUTE INFORMATION, NODE INFORMATION AND A LOT MORE */
+/* ------------------------------------------------------------------------- */
+void Dblqh::execLQHKEYREQ(Signal* signal)
+{
+ UintR sig0, sig1, sig2, sig3, sig4, sig5;
+ Uint8 tfragDistKey;
+
+ const LqhKeyReq * const lqhKeyReq = (LqhKeyReq *)signal->getDataPtr();
+
+ sig0 = lqhKeyReq->clientConnectPtr;
+ if (cfirstfreeTcConrec != RNIL && !ERROR_INSERTED(5031)) {
+ jamEntry();
+ seizeTcrec();
+ } else {
+/* ------------------------------------------------------------------------- */
+/* NO FREE TC RECORD AVAILABLE, THUS WE CANNOT HANDLE THE REQUEST. */
+/* ------------------------------------------------------------------------- */
+ if (ERROR_INSERTED(5031)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ }
+ noFreeRecordLab(signal, lqhKeyReq, ZNO_TC_CONNECT_ERROR);
+ return;
+ }//if
+
+ c_Counters.operations++;
+
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ regTcPtr->clientBlockref = signal->senderBlockRef();
+ regTcPtr->clientConnectrec = sig0;
+ regTcPtr->tcOprec = sig0;
+ regTcPtr->storedProcId = ZNIL;
+
+ UintR TtotReclenAi = lqhKeyReq->attrLen;
+ sig1 = lqhKeyReq->savePointId;
+ sig2 = lqhKeyReq->hashValue;
+ UintR Treqinfo = lqhKeyReq->requestInfo;
+ sig4 = lqhKeyReq->tableSchemaVersion;
+ sig5 = lqhKeyReq->tcBlockref;
+
+ regTcPtr->savePointId = sig1;
+ regTcPtr->hashValue = sig2;
+ const Uint32 schemaVersion = regTcPtr->schemaVersion = LqhKeyReq::getSchemaVersion(sig4);
+ tabptr.i = LqhKeyReq::getTableId(sig4);
+ regTcPtr->tcBlockref = sig5;
+
+ const Uint8 op = LqhKeyReq::getOperation(Treqinfo);
+ if (op == ZREAD && !getAllowRead()){
+ noFreeRecordLab(signal, lqhKeyReq, ZNODE_SHUTDOWN_IN_PROGESS);
+ return;
+ }
+
+ regTcPtr->totReclenAi = LqhKeyReq::getAttrLen(TtotReclenAi);
+ regTcPtr->tcScanInfo = lqhKeyReq->scanInfo;
+ regTcPtr->indTakeOver = LqhKeyReq::getScanTakeOverFlag(TtotReclenAi);
+
+ regTcPtr->readlenAi = 0;
+ regTcPtr->currTupAiLen = 0;
+ regTcPtr->listState = TcConnectionrec::NOT_IN_LIST;
+ regTcPtr->logWriteState = TcConnectionrec::NOT_STARTED;
+ regTcPtr->fragmentptr = RNIL;
+
+ sig0 = lqhKeyReq->fragmentData;
+ sig1 = lqhKeyReq->transId1;
+ sig2 = lqhKeyReq->transId2;
+ sig3 = lqhKeyReq->variableData[0];
+ sig4 = lqhKeyReq->variableData[1];
+
+ regTcPtr->fragmentid = LqhKeyReq::getFragmentId(sig0);
+ regTcPtr->nextReplica = LqhKeyReq::getNextReplicaNodeId(sig0);
+ regTcPtr->transid[0] = sig1;
+ regTcPtr->transid[1] = sig2;
+ regTcPtr->applRef = sig3;
+ regTcPtr->applOprec = sig4;
+
+ regTcPtr->commitAckMarker = RNIL;
+ if(LqhKeyReq::getMarkerFlag(Treqinfo)){
+ jam();
+
+ CommitAckMarkerPtr markerPtr;
+ m_commitAckMarkerHash.seize(markerPtr);
+ if(markerPtr.i == RNIL){
+ noFreeRecordLab(signal, lqhKeyReq, ZNO_FREE_MARKER_RECORDS_ERROR);
+ return;
+ }
+ markerPtr.p->transid1 = sig1;
+ markerPtr.p->transid2 = sig2;
+ markerPtr.p->apiRef = sig3;
+ markerPtr.p->apiOprec = sig4;
+ const NodeId tcNodeId = refToNode(sig5);
+ markerPtr.p->tcNodeId = tcNodeId;
+
+ m_commitAckMarkerHash.add(markerPtr);
+ regTcPtr->commitAckMarker = markerPtr.i;
+ }
+
+ regTcPtr->reqinfo = Treqinfo;
+ regTcPtr->lastReplicaNo = LqhKeyReq::getLastReplicaNo(Treqinfo);
+ regTcPtr->lockType = LqhKeyReq::getLockType(Treqinfo);
+ regTcPtr->dirtyOp = LqhKeyReq::getDirtyFlag(Treqinfo);
+ regTcPtr->opExec = LqhKeyReq::getInterpretedFlag(Treqinfo);
+ regTcPtr->opSimple = LqhKeyReq::getSimpleFlag(Treqinfo);
+ regTcPtr->simpleRead = ((Treqinfo >> 18) & 15);
+ regTcPtr->operation = LqhKeyReq::getOperation(Treqinfo);
+ regTcPtr->seqNoReplica = LqhKeyReq::getSeqNoReplica(Treqinfo);
+ UintR TreclenAiLqhkey = LqhKeyReq::getAIInLqhKeyReq(Treqinfo);
+ regTcPtr->apiVersionNo = 0;
+
+ regTcPtr->reclenAiLqhkey = TreclenAiLqhkey;
+ regTcPtr->currReclenAi = TreclenAiLqhkey;
+ UintR TitcKeyLen = LqhKeyReq::getKeyLen(Treqinfo);
+ regTcPtr->primKeyLen = TitcKeyLen;
+ regTcPtr->noFiredTriggers = lqhKeyReq->noFiredTriggers;
+
+ UintR TapplAddressInd = LqhKeyReq::getApplicationAddressFlag(Treqinfo);
+ UintR nextPos = (TapplAddressInd << 1);
+ UintR TsameClientAndTcOprec = LqhKeyReq::getSameClientAndTcFlag(Treqinfo);
+ if (TsameClientAndTcOprec == 1) {
+ regTcPtr->tcOprec = lqhKeyReq->variableData[nextPos];
+ nextPos++;
+ }//if
+ UintR TnextReplicasIndicator = regTcPtr->lastReplicaNo -
+ regTcPtr->seqNoReplica;
+ if (TnextReplicasIndicator > 1) {
+ regTcPtr->nodeAfterNext[0] = lqhKeyReq->variableData[nextPos] & 0xFFFF;
+ regTcPtr->nodeAfterNext[1] = lqhKeyReq->variableData[nextPos] >> 16;
+ nextPos++;
+ }//if
+ UintR TstoredProcIndicator = LqhKeyReq::getStoredProcFlag(TtotReclenAi);
+ if (TstoredProcIndicator == 1) {
+ regTcPtr->storedProcId = lqhKeyReq->variableData[nextPos] & ZNIL;
+ nextPos++;
+ }//if
+ UintR TreadLenAiIndicator = LqhKeyReq::getReturnedReadLenAIFlag(Treqinfo);
+ if (TreadLenAiIndicator == 1) {
+ regTcPtr->readlenAi = lqhKeyReq->variableData[nextPos] & ZNIL;
+ nextPos++;
+ }//if
+ sig0 = lqhKeyReq->variableData[nextPos + 0];
+ sig1 = lqhKeyReq->variableData[nextPos + 1];
+ sig2 = lqhKeyReq->variableData[nextPos + 2];
+ sig3 = lqhKeyReq->variableData[nextPos + 3];
+
+ regTcPtr->tupkeyData[0] = sig0;
+ regTcPtr->tupkeyData[1] = sig1;
+ regTcPtr->tupkeyData[2] = sig2;
+ regTcPtr->tupkeyData[3] = sig3;
+
+ if (TitcKeyLen > 0) {
+ if (TitcKeyLen < 4) {
+ nextPos += TitcKeyLen;
+ } else {
+ nextPos += 4;
+ }//if
+ } else {
+ LQHKEY_error(signal, 3);
+ return;
+ }//if
+
+ if ((LqhKeyReq::FixedSignalLength + nextPos + TreclenAiLqhkey) !=
+ signal->length()) {
+ LQHKEY_error(signal, 2);
+ return;
+ }//if
+ UintR TseqNoReplica = regTcPtr->seqNoReplica;
+ UintR TlastReplicaNo = regTcPtr->lastReplicaNo;
+ if (TseqNoReplica == TlastReplicaNo) {
+ jam();
+ regTcPtr->nextReplica = ZNIL;
+ } else {
+ if (TseqNoReplica < TlastReplicaNo) {
+ jam();
+ regTcPtr->nextSeqNoReplica = TseqNoReplica + 1;
+ if ((regTcPtr->nextReplica == 0) ||
+ (regTcPtr->nextReplica == cownNodeid)) {
+ LQHKEY_error(signal, 0);
+ }//if
+ } else {
+ LQHKEY_error(signal, 4);
+ return;
+ }//if
+ }//if
+ TcConnectionrecPtr localNextTcConnectptr;
+ Uint32 hashIndex = (regTcPtr->transid[0] ^ regTcPtr->tcOprec) & 1023;
+ localNextTcConnectptr.i = ctransidHash[hashIndex];
+ ctransidHash[hashIndex] = tcConnectptr.i;
+ regTcPtr->prevHashRec = RNIL;
+ regTcPtr->nextHashRec = localNextTcConnectptr.i;
+ if (localNextTcConnectptr.i != RNIL) {
+/* -------------------------------------------------------------------------- */
+/* ENSURE THAT THE NEXT RECORD HAS SET PREVIOUS TO OUR RECORD IF IT EXISTS */
+/* -------------------------------------------------------------------------- */
+ ptrCheckGuard(localNextTcConnectptr,
+ ctcConnectrecFileSize, tcConnectionrec);
+ jam();
+ localNextTcConnectptr.p->prevHashRec = tcConnectptr.i;
+ }//if
+ if (tabptr.i >= ctabrecFileSize) {
+ LQHKEY_error(signal, 5);
+ return;
+ }//if
+ ptrAss(tabptr, tablerec);
+ if(tabptr.p->tableStatus != Tablerec::TABLE_DEFINED){
+ LQHKEY_abort(signal, 4);
+ return;
+ }
+ if(tabptr.p->schemaVersion != schemaVersion){
+ LQHKEY_abort(signal, 5);
+ return;
+ }
+
+ regTcPtr->tableref = tabptr.i;
+ tabptr.p->usageCount++;
+
+ if (!getFragmentrec(signal, regTcPtr->fragmentid)) {
+ LQHKEY_error(signal, 6);
+ return;
+ }//if
+ regTcPtr->localFragptr = (regTcPtr->hashValue >> fragptr.p->hashCheckBit) & 1;
+ Uint8 TcopyType = fragptr.p->fragCopy;
+ tfragDistKey = fragptr.p->fragDistributionKey;
+ if (fragptr.p->fragStatus == Fragrecord::ACTIVE_CREATION) {
+ jam();
+ regTcPtr->activeCreat = ZTRUE;
+ CRASH_INSERTION(5002);
+ } else {
+ regTcPtr->activeCreat = ZFALSE;
+ }//if
+ regTcPtr->replicaType = TcopyType;
+ regTcPtr->fragmentptr = fragptr.i;
+ Uint8 TdistKey = LqhKeyReq::getDistributionKey(TtotReclenAi);
+ if ((tfragDistKey != TdistKey) &&
+ (regTcPtr->seqNoReplica == 0) &&
+ (regTcPtr->dirtyOp == ZFALSE) &&
+ (regTcPtr->simpleRead != ZSIMPLE_READ)) {
+ /* ----------------------------------------------------------------------
+ * WE HAVE DIFFERENT OPINION THAN THE DIH THAT STARTED THE TRANSACTION.
+ * THE REASON COULD BE THAT THIS IS AN OLD DISTRIBUTION WHICH IS NO LONGER
+ * VALID TO USE. THIS MUST BE CHECKED.
+ * ONE IS ADDED TO THE DISTRIBUTION KEY EVERY TIME WE ADD A NEW REPLICA.
+ * FAILED REPLICAS DO NOT AFFECT THE DISTRIBUTION KEY. THIS MEANS THAT THE
+ * MAXIMUM DEVIATION CAN BE ONE BETWEEN THOSE TWO VALUES.
+ * ---------------------------------------------------------------------- */
+ Int32 tmp = TdistKey - tfragDistKey;
+ tmp = (tmp < 0 ? - tmp : tmp);
+ if ((tmp <= 1) || (tfragDistKey == 0)) {
+ LQHKEY_abort(signal, 0);
+ return;
+ }//if
+ LQHKEY_error(signal, 1);
+ }//if
+ if (TreclenAiLqhkey != 0) {
+ if (regTcPtr->operation != ZREAD) {
+ if (regTcPtr->operation != ZDELETE) {
+ if (regTcPtr->opExec != 1) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* */
+/* UPDATES, WRITES AND INSERTS THAT ARE NOT INTERPRETED WILL USE THE */
+/* SAME ATTRINFO IN ALL REPLICAS. THUS WE SAVE THE ATTRINFO ALREADY */
+/* TO SAVE A SIGNAL FROM TUP TO LQH. INTERPRETED EXECUTION IN TUP */
+/* WILL CREATE NEW ATTRINFO FOR THE OTHER REPLICAS AND IT IS THUS NOT */
+/* A GOOD IDEA TO SAVE THE INFORMATION HERE. READS WILL ALSO BE */
+/* UNNECESSARY TO SAVE SINCE THAT ATTRINFO WILL NEVER BE SENT TO ANY */
+/* MORE REPLICAS. */
+/*---------------------------------------------------------------------------*/
+/* READS AND DELETES CAN ONLY HAVE INFORMATION ABOUT WHAT IS TO BE READ. */
+/* NO INFORMATION THAT NEEDS LOGGING. */
+/*---------------------------------------------------------------------------*/
+ sig0 = lqhKeyReq->variableData[nextPos + 0];
+ sig1 = lqhKeyReq->variableData[nextPos + 1];
+ sig2 = lqhKeyReq->variableData[nextPos + 2];
+ sig3 = lqhKeyReq->variableData[nextPos + 3];
+ sig4 = lqhKeyReq->variableData[nextPos + 4];
+
+ regTcPtr->firstAttrinfo[0] = sig0;
+ regTcPtr->firstAttrinfo[1] = sig1;
+ regTcPtr->firstAttrinfo[2] = sig2;
+ regTcPtr->firstAttrinfo[3] = sig3;
+ regTcPtr->firstAttrinfo[4] = sig4;
+ regTcPtr->currTupAiLen = TreclenAiLqhkey;
+ } else {
+ jam();
+ regTcPtr->reclenAiLqhkey = 0;
+ }//if
+ } else {
+ jam();
+ regTcPtr->reclenAiLqhkey = 0;
+ }//if
+ }//if
+ sig0 = lqhKeyReq->variableData[nextPos + 0];
+ sig1 = lqhKeyReq->variableData[nextPos + 1];
+ sig2 = lqhKeyReq->variableData[nextPos + 2];
+ sig3 = lqhKeyReq->variableData[nextPos + 3];
+ sig4 = lqhKeyReq->variableData[nextPos + 4];
+
+ signal->theData[0] = regTcPtr->tupConnectrec;
+ signal->theData[3] = sig0;
+ signal->theData[4] = sig1;
+ signal->theData[5] = sig2;
+ signal->theData[6] = sig3;
+ signal->theData[7] = sig4;
+ EXECUTE_DIRECT(refToBlock(regTcPtr->tcTupBlockref), GSN_ATTRINFO,
+ signal, TreclenAiLqhkey + 3);
+ jamEntry();
+ if (signal->theData[0] == (UintR)-1) {
+ LQHKEY_abort(signal, 2);
+ return;
+ }//if
+ }//if
+/* ------- TAKE CARE OF PRIM KEY DATA ------- */
+ if (regTcPtr->primKeyLen <= 4) {
+ endgettupkeyLab(signal);
+ return;
+ } else {
+ jam();
+/*--------------------------------------------------------------------*/
+/* KEY LENGTH WAS MORE THAN 4 WORDS (WORD = 4 BYTE). THUS WE */
+/* HAVE TO ALLOCATE A DATA BUFFER TO STORE THE KEY DATA AND */
+/* WAIT FOR THE KEYINFO SIGNAL. */
+/*--------------------------------------------------------------------*/
+ regTcPtr->save1 = 4;
+ regTcPtr->transactionState = TcConnectionrec::WAIT_TUPKEYINFO;
+ return;
+ }//if
+ return;
+}//Dblqh::execLQHKEYREQ()
+
+void Dblqh::endgettupkeyLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->totReclenAi == regTcPtr->currReclenAi) {
+ ;
+ } else {
+ jam();
+ ndbrequire(regTcPtr->currReclenAi < regTcPtr->totReclenAi);
+ regTcPtr->transactionState = TcConnectionrec::WAIT_ATTR;
+ return;
+ }//if
+/* ---------------------------------------------------------------------- */
+/* NOW RECEPTION OF LQHKEYREQ IS COMPLETED THE NEXT STEP IS TO START*/
+/* PROCESSING THE MESSAGE. IF THE MESSAGE IS TO A STAND-BY NODE */
+/* WITHOUT NETWORK REDUNDANCY OR PREPARE-TO-COMMIT ACTIVATED THE */
+/* PREPARATION TO SEND TO THE NEXT NODE WILL START IMMEDIATELY. */
+/* */
+/* OTHERWISE THE PROCESSING WILL START AFTER SETTING THE PROPER */
+/* STATE. HOWEVER BEFORE PROCESSING THE MESSAGE */
+/* IT IS NECESSARY TO CHECK THAT THE FRAGMENT IS NOT PERFORMING */
+/* A CHECKPOINT. THE OPERATION SHALL ALSO BE LINKED INTO THE */
+/* FRAGMENT QUEUE OR LIST OF ACTIVE OPERATIONS. */
+/* */
+/* THE FIRST STEP IN PROCESSING THE MESSAGE IS TO CONTACT DBACC. */
+/*------------------------------------------------------------------------*/
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ case Fragrecord::CRASH_RECOVERING:
+ case Fragrecord::ACTIVE_CREATION:
+ linkActiveFrag(signal);
+ prepareContinueAfterBlockedLab(signal);
+ return;
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ regTcPtr->transactionState = TcConnectionrec::STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dblqh::endgettupkeyLab()
+
+void Dblqh::prepareContinueAfterBlockedLab(Signal* signal)
+{
+ UintR ttcScanOp;
+ UintR ttcScanNumber;
+ UintR taccreq;
+
+/* -------------------------------------------------------------------------- */
+/* INPUT: TC_CONNECTPTR ACTIVE CONNECTION RECORD */
+/* FRAGPTR FRAGMENT RECORD */
+/* -------------------------------------------------------------------------- */
+/* -------------------------------------------------------------------------- */
+/* CONTINUE HERE AFTER BEING BLOCKED FOR A WHILE DURING LOCAL CHECKPOINT. */
+/* -------------------------------------------------------------------------- */
+/* ALSO AFTER NORMAL PROCEDURE WE CONTINUE HERE */
+/* -------------------------------------------------------------------------- */
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->indTakeOver == ZTRUE) {
+ jam();
+ ttcScanOp = KeyInfo20::getScanOp(regTcPtr->tcScanInfo);
+ ttcScanNumber = KeyInfo20::getScanNo(regTcPtr->tcScanInfo);
+ scanptr.i = ZNIL;
+ if (ttcScanNumber < NR_MaxRangeScanNo && ttcScanNumber != NR_ScanNo) {
+ jam();
+ // table fragment also when index scan
+ scanptr.i = fragptr.p->fragScanRec[ttcScanNumber];
+ }
+ if (scanptr.i == ZNIL) {
+ jam();
+ releaseActiveFrag(signal);
+ takeOverErrorLab(signal);
+ return;
+ }//if
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ Uint32 accOpPtr = scanptr.p->scanAccOpPtr[ttcScanOp];
+ if (accOpPtr == RNIL) {
+ jam();
+ releaseActiveFrag(signal);
+ takeOverErrorLab(signal);
+ return;
+ }//if
+ signal->theData[1] = accOpPtr;
+ signal->theData[2] = regTcPtr->transid[0];
+ signal->theData[3] = regTcPtr->transid[1];
+ EXECUTE_DIRECT(refToBlock(regTcPtr->tcAccBlockref), GSN_ACC_TO_REQ,
+ signal, 4);
+ if (signal->theData[0] == (UintR)-1) {
+ execACC_TO_REF(signal);
+ return;
+ }//if
+ jamEntry();
+ }//if
+/*-------------------------------------------------------------------*/
+/* IT IS NOW TIME TO CONTACT ACC. THE TUPLE KEY WILL BE SENT */
+/* AND THIS WILL BE TRANSLATED INTO A LOCAL KEY BY USING THE */
+/* LOCAL PART OF THE LH3-ALGORITHM. ALSO PROPER LOCKS ON THE */
+/* TUPLE WILL BE SET. FOR INSERTS AND DELETES THE MESSAGE WILL */
+/* START AN INSERT/DELETE INTO THE HASH TABLE. */
+/* */
+/* BEFORE SENDING THE MESSAGE THE REQUEST INFORMATION IS SET */
+/* PROPERLY. */
+/* ----------------------------------------------------------------- */
+#if 0
+ if (regTcPtr->tableref != 0) {
+ switch (regTcPtr->operation) {
+ case ZREAD: ndbout << "Läsning "; break;
+ case ZUPDATE: ndbout << " Uppdatering "; break;
+ case ZWRITE: ndbout << "Write "; break;
+ case ZINSERT: ndbout << "Inläggning "; break;
+ case ZDELETE: ndbout << "Borttagning "; break;
+ default: ndbout << "????"; break;
+ }
+ ndbout << "med nyckel = " << regTcPtr->tupkeyData[0] << endl;
+ }
+#endif
+
+ regTcPtr->transactionState = TcConnectionrec::WAIT_ACC;
+ taccreq = regTcPtr->operation;
+ taccreq = taccreq + (regTcPtr->opSimple << 3);
+ taccreq = taccreq + (regTcPtr->lockType << 4);
+ taccreq = taccreq + (regTcPtr->dirtyOp << 6);
+ taccreq = taccreq + (regTcPtr->replicaType << 7);
+ taccreq = taccreq + (regTcPtr->apiVersionNo << 9);
+/* ************ */
+/* ACCKEYREQ < */
+/* ************ */
+ ndbrequire(regTcPtr->localFragptr < 2);
+ Uint32 sig0, sig1, sig2, sig3, sig4;
+ sig0 = regTcPtr->accConnectrec;
+ sig1 = fragptr.p->accFragptr[regTcPtr->localFragptr];
+ sig2 = regTcPtr->hashValue;
+ sig3 = regTcPtr->primKeyLen;
+ sig4 = regTcPtr->transid[0];
+ signal->theData[0] = sig0;
+ signal->theData[1] = sig1;
+ signal->theData[2] = taccreq;
+ signal->theData[3] = sig2;
+ signal->theData[4] = sig3;
+ signal->theData[5] = sig4;
+
+ sig0 = regTcPtr->transid[1];
+ sig1 = regTcPtr->tupkeyData[0];
+ sig2 = regTcPtr->tupkeyData[1];
+ sig3 = regTcPtr->tupkeyData[2];
+ sig4 = regTcPtr->tupkeyData[3];
+ signal->theData[6] = sig0;
+ signal->theData[7] = sig1;
+ signal->theData[8] = sig2;
+ signal->theData[9] = sig3;
+ signal->theData[10] = sig4;
+ if (regTcPtr->primKeyLen > 4) {
+ sendKeyinfoAcc(signal);
+ }//if
+ EXECUTE_DIRECT(refToBlock(regTcPtr->tcAccBlockref), GSN_ACCKEYREQ,
+ signal, 7 + regTcPtr->primKeyLen);
+ if (signal->theData[0] < RNIL) {
+ signal->theData[0] = tcConnectptr.i;
+ execACCKEYCONF(signal);
+ return;
+ } else if (signal->theData[0] == RNIL) {
+ ;
+ } else {
+ ndbrequire(signal->theData[0] == (UintR)-1);
+ signal->theData[0] = tcConnectptr.i;
+ execACCKEYREF(signal);
+ }//if
+ return;
+}//Dblqh::prepareContinueAfterBlockedLab()
+
+/* ========================================================================== */
+/* ======= SEND KEYINFO TO ACC ======= */
+/* */
+/* ========================================================================== */
+void Dblqh::sendKeyinfoAcc(Signal* signal)
+{
+ UintR Ti = 11;
+ DatabufPtr regDatabufptr;
+ regDatabufptr.i = tcConnectptr.p->firstTupkeybuf;
+
+ do {
+ jam();
+ ptrCheckGuard(regDatabufptr, cdatabufFileSize, databuf);
+ Uint32 sig0 = regDatabufptr.p->data[0];
+ Uint32 sig1 = regDatabufptr.p->data[1];
+ Uint32 sig2 = regDatabufptr.p->data[2];
+ Uint32 sig3 = regDatabufptr.p->data[3];
+ signal->theData[Ti] = sig0;
+ signal->theData[Ti + 1] = sig1;
+ signal->theData[Ti + 2] = sig2;
+ signal->theData[Ti + 3] = sig3;
+ regDatabufptr.i = regDatabufptr.p->nextDatabuf;
+ Ti += 4;
+ } while (regDatabufptr.i != RNIL);
+}//Dblqh::sendKeyinfoAcc()
+
+void Dblqh::execLQH_ALLOCREQ(Signal* signal)
+{
+ TcConnectionrecPtr regTcPtr;
+ FragrecordPtr regFragptr;
+
+ jamEntry();
+ regTcPtr.i = signal->theData[0];
+ ptrCheckGuard(regTcPtr, ctcConnectrecFileSize, tcConnectionrec);
+
+ regFragptr.i = regTcPtr.p->fragmentptr;
+ ptrCheckGuard(regFragptr, cfragrecFileSize, fragrecord);
+
+ ndbrequire(regTcPtr.p->localFragptr < 2);
+ signal->theData[0] = regTcPtr.p->tupConnectrec;
+ signal->theData[1] = regFragptr.p->tupFragptr[regTcPtr.p->localFragptr];
+ signal->theData[2] = regTcPtr.p->tableref;
+ Uint32 tup = refToBlock(regTcPtr.p->tcTupBlockref);
+ EXECUTE_DIRECT(tup, GSN_TUP_ALLOCREQ, signal, 3);
+}//Dblqh::execTUP_ALLOCREQ()
+
+/* ************>> */
+/* ACCKEYCONF > */
+/* ************>> */
+void Dblqh::execACCKEYCONF(Signal* signal)
+{
+ TcConnectionrec *regTcConnectionrec = tcConnectionrec;
+ Uint32 ttcConnectrecFileSize = ctcConnectrecFileSize;
+ Uint32 tcIndex = signal->theData[0];
+ Uint32 Tfragid = signal->theData[2];
+ Uint32 localKey1 = signal->theData[3];
+ Uint32 localKey2 = signal->theData[4];
+ Uint32 localKeyFlag = signal->theData[5];
+ jamEntry();
+ tcConnectptr.i = tcIndex;
+ ptrCheckGuard(tcConnectptr, ttcConnectrecFileSize, regTcConnectionrec);
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->transactionState != TcConnectionrec::WAIT_ACC) {
+ LQHKEY_abort(signal, 3);
+ return;
+ }//if
+ /* ------------------------------------------------------------------------
+ * Set transaction state and also reset the activeCreat since that is only
+ * valid in cases where the record was not present.
+ * ------------------------------------------------------------------------ */
+ regTcPtr->transactionState = TcConnectionrec::WAIT_TUP;
+ regTcPtr->activeCreat = ZFALSE;
+ /* ------------------------------------------------------------------------
+ * IT IS NOW TIME TO CONTACT THE TUPLE MANAGER. THE TUPLE MANAGER NEEDS THE
+ * INFORMATION ON WHICH TABLE AND FRAGMENT, THE LOCAL KEY AND IT NEEDS TO
+ * KNOW THE TYPE OF OPERATION TO PERFORM. TUP CAN SEND THE ATTRINFO DATA
+ * EITHER TO THE TC BLOCK OR DIRECTLY TO THE APPLICATION. THE SCHEMA VERSION
+ * IS NEEDED SINCE TWO SCHEMA VERSIONS CAN BE ACTIVE SIMULTANEOUSLY ON A
+ * TABLE.
+ * ------------------------------------------------------------------------ */
+ if (regTcPtr->operation == ZWRITE) {
+ if (signal->theData[1] > 0) {
+ /* --------------------------------------------------------------------
+ * ACC did perform an insert and thus we should indicate that the WRITE
+ * is an INSERT otherwise it is an UPDATE.
+ * -------------------------------------------------------------------- */
+ jam();
+ regTcPtr->operation = ZINSERT;
+ } else {
+ jam();
+ tcConnectptr.p->operation = ZUPDATE;
+ }//if
+ }//if
+ ndbrequire(localKeyFlag == 1);
+ localKey2 = localKey1 & MAX_TUPLES_PER_PAGE;
+ localKey1 = localKey1 >> MAX_TUPLES_BITS;
+ Uint32 Ttupreq = regTcPtr->dirtyOp;
+ Ttupreq = Ttupreq + (regTcPtr->opSimple << 1);
+ Ttupreq = Ttupreq + (regTcPtr->operation << 6);
+ Ttupreq = Ttupreq + (regTcPtr->opExec << 10);
+ Ttupreq = Ttupreq + (regTcPtr->apiVersionNo << 11);
+
+ /* ---------------------------------------------------------------------
+ * Clear interpreted mode bit since we do not want the next replica to
+ * use interpreted mode. The next replica will receive a normal write.
+ * --------------------------------------------------------------------- */
+ regTcPtr->opExec = 0;
+ /* ************< */
+ /* TUPKEYREQ < */
+ /* ************< */
+ TupKeyReq * const tupKeyReq = (TupKeyReq *)signal->getDataPtrSend();
+ Uint32 sig0, sig1, sig2, sig3;
+
+ sig0 = regTcPtr->tupConnectrec;
+ sig1 = regTcPtr->tableref;
+ tupKeyReq->connectPtr = sig0;
+ tupKeyReq->request = Ttupreq;
+ tupKeyReq->tableRef = sig1;
+ tupKeyReq->fragId = Tfragid;
+ tupKeyReq->keyRef1 = localKey1;
+ tupKeyReq->keyRef2 = localKey2;
+
+ sig0 = regTcPtr->totReclenAi;
+ sig1 = regTcPtr->applOprec;
+ sig2 = regTcPtr->applRef;
+ sig3 = regTcPtr->schemaVersion;
+ FragrecordPtr regFragptr;
+ regFragptr.i = regTcPtr->fragmentptr;
+ ptrCheckGuard(regFragptr, cfragrecFileSize, fragrecord);
+ tupKeyReq->attrBufLen = sig0;
+ tupKeyReq->opRef = sig1;
+ tupKeyReq->applRef = sig2;
+ tupKeyReq->schemaVersion = sig3;
+
+ ndbrequire(regTcPtr->localFragptr < 2);
+ sig0 = regTcPtr->storedProcId;
+ sig1 = regTcPtr->transid[0];
+ sig2 = regTcPtr->transid[1];
+ sig3 = regFragptr.p->tupFragptr[regTcPtr->localFragptr];
+ Uint32 tup = refToBlock(regTcPtr->tcTupBlockref);
+
+ tupKeyReq->storedProcedure = sig0;
+ tupKeyReq->transId1 = sig1;
+ tupKeyReq->transId2 = sig2;
+ tupKeyReq->fragPtr = sig3;
+ tupKeyReq->primaryReplica = (tcConnectptr.p->seqNoReplica == 0)?true:false;
+ tupKeyReq->coordinatorTC = tcConnectptr.p->tcBlockref;
+ tupKeyReq->tcOpIndex = tcConnectptr.p->tcOprec;
+ tupKeyReq->savePointId = tcConnectptr.p->savePointId;
+
+ EXECUTE_DIRECT(tup, GSN_TUPKEYREQ, signal, TupKeyReq::SignalLength);
+}//Dblqh::execACCKEYCONF()
+
+/* --------------------------------------------------------------------------
+ * ------- ENTER TUP... -------
+ * ENTER TUPKEYCONF WITH
+ * TC_CONNECTPTR,
+ * TDATA2, LOCAL KEY REFERENCE 1, ONLY INTERESTING AFTER INSERT
+ * TDATA3, LOCAL KEY REFERENCE 1, ONLY INTERESTING AFTER INSERT
+ * TDATA4, TOTAL LENGTH OF READ DATA SENT TO TC/APPLICATION
+ * TDATA5 TOTAL LENGTH OF UPDATE DATA SENT TO/FROM TUP
+ * GOTO TUPKEY_CONF
+ *
+ * TAKE CARE OF RESPONSES FROM TUPLE MANAGER.
+ * -------------------------------------------------------------------------- */
+void Dblqh::tupkeyConfLab(Signal* signal)
+{
+/* ---- GET OPERATION TYPE AND CHECK WHAT KIND OF OPERATION IS REQUESTED ---- */
+ const TupKeyConf * const tupKeyConf = (TupKeyConf *)&signal->theData[0];
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->simpleRead == ZSIMPLE_READ) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THE OPERATION IS A SIMPLE READ. WE WILL IMMEDIATELY COMMIT THE OPERATION.
+ * SINCE WE HAVE NOT RELEASED THE FRAGMENT LOCK (FOR LOCAL CHECKPOINTS) YET
+ * WE CAN GO IMMEDIATELY TO COMMIT_CONTINUE_AFTER_BLOCKED.
+ * WE HAVE ALREADY SENT THE RESPONSE SO WE ARE NOT INTERESTED IN READ LENGTH
+ * ---------------------------------------------------------------------- */
+ regTcPtr->gci = cnewestGci;
+ releaseActiveFrag(signal);
+ commitContinueAfterBlockedLab(signal);
+ return;
+ }//if
+ if (tupKeyConf->readLength != 0) {
+ jam();
+
+ /* SET BIT 15 IN REQINFO */
+ LqhKeyReq::setApplicationAddressFlag(regTcPtr->reqinfo, 1);
+
+ regTcPtr->readlenAi = tupKeyConf->readLength;
+ }//if
+ regTcPtr->totSendlenAi = tupKeyConf->writeLength;
+ ndbrequire(regTcPtr->totSendlenAi == regTcPtr->currTupAiLen);
+ rwConcludedLab(signal);
+ return;
+}//Dblqh::tupkeyConfLab()
+
+/* --------------------------------------------------------------------------
+ * THE CODE IS FOUND IN THE SIGNAL RECEPTION PART OF LQH
+ * -------------------------------------------------------------------------- */
+void Dblqh::rwConcludedLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ /* ------------------------------------------------------------------------
+ * WE HAVE NOW CONCLUDED READING/WRITING IN ACC AND TUP FOR THIS OPERATION.
+ * IT IS NOW TIME TO LOG THE OPERATION, SEND REQUEST TO NEXT NODE OR TC AND
+ * FOR SOME TYPES OF OPERATIONS IT IS EVEN TIME TO COMMIT THE OPERATION.
+ * ------------------------------------------------------------------------ */
+ if (regTcPtr->operation == ZREAD) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * A NORMAL READ OPERATION IS NOT LOGGED BUT IS NOT COMMITTED UNTIL THE
+ * COMMIT SIGNAL ARRIVES. THUS WE CONTINUE PACKING THE RESPONSE.
+ * ---------------------------------------------------------------------- */
+ releaseActiveFrag(signal);
+ packLqhkeyreqLab(signal);
+ return;
+ } else {
+ FragrecordPtr regFragptr;
+ regFragptr.i = regTcPtr->fragmentptr;
+ ptrCheckGuard(regFragptr, cfragrecFileSize, fragrecord);
+ if (regFragptr.p->logFlag == Fragrecord::STATE_FALSE){
+ if (regTcPtr->dirtyOp == ZTRUE) {
+ jam();
+ /* ------------------------------------------------------------------
+ * THIS OPERATION WAS A WRITE OPERATION THAT DO NOT NEED LOGGING AND
+ * THAT CAN CAN BE COMMITTED IMMEDIATELY.
+ * ------------------------------------------------------------------ */
+ regTcPtr->gci = cnewestGci;
+ releaseActiveFrag(signal);
+ commitContinueAfterBlockedLab(signal);
+ return;
+ } else {
+ jam();
+ /* ------------------------------------------------------------------
+ * A NORMAL WRITE OPERATION ON A FRAGMENT WHICH DO NOT NEED LOGGING.
+ * WE WILL PACK THE REQUEST/RESPONSE TO THE NEXT NODE/TO TC.
+ * ------------------------------------------------------------------ */
+ regTcPtr->logWriteState = TcConnectionrec::NOT_WRITTEN;
+ releaseActiveFrag(signal);
+ packLqhkeyreqLab(signal);
+ return;
+ }//if
+ } else {
+ jam();
+ /* --------------------------------------------------------------------
+ * A DIRTY OPERATION WHICH NEEDS LOGGING. WE START BY LOGGING THE
+ * REQUEST. IN THIS CASE WE WILL RELEASE THE FRAGMENT LOCK FIRST.
+ * --------------------------------------------------------------------
+ * A NORMAL WRITE OPERATION THAT NEEDS LOGGING AND WILL NOT BE
+ * PREMATURELY COMMITTED.
+ * -------------------------------------------------------------------- */
+ releaseActiveFrag(signal);
+ logLqhkeyreqLab(signal);
+ return;
+ }//if
+ }//if
+}//Dblqh::rwConcludedLab()
+
+void Dblqh::rwConcludedAiLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ fragptr.i = regTcPtr->fragmentptr;
+ /* ------------------------------------------------------------------------
+ * WE HAVE NOW CONCLUDED READING/WRITING IN ACC AND TUP FOR THIS OPERATION.
+ * IT IS NOW TIME TO LOG THE OPERATION, SEND REQUEST TO NEXT NODE OR TC AND
+ * FOR SOME TYPES OF OPERATIONS IT IS EVEN TIME TO COMMIT THE OPERATION.
+ * IN THIS CASE WE HAVE ALREADY RELEASED THE FRAGMENT LOCK.
+ * ERROR CASES AT FRAGMENT CREATION AND STAND-BY NODES ARE THE REASONS FOR
+ * COMING HERE.
+ * ------------------------------------------------------------------------ */
+ if (regTcPtr->operation == ZREAD) {
+ if (regTcPtr->opSimple == 1) {
+ jam();
+ /* --------------------------------------------------------------------
+ * THE OPERATION IS A SIMPLE READ. WE WILL IMMEDIATELY COMMIT THE
+ * OPERATION.
+ * -------------------------------------------------------------------- */
+ regTcPtr->gci = cnewestGci;
+ localCommitLab(signal);
+ return;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------
+ * A NORMAL READ OPERATION IS NOT LOGGED BUT IS NOT COMMITTED UNTIL
+ * THE COMMIT SIGNAL ARRIVES. THUS WE CONTINUE PACKING THE RESPONSE.
+ * -------------------------------------------------------------------- */
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ packLqhkeyreqLab(signal);
+ return;
+ }//if
+ } else {
+ jam();
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (fragptr.p->logFlag == Fragrecord::STATE_FALSE) {
+ if (regTcPtr->dirtyOp == ZTRUE) {
+ /* ------------------------------------------------------------------
+ * THIS OPERATION WAS A WRITE OPERATION THAT DO NOT NEED LOGGING AND
+ * THAT CAN CAN BE COMMITTED IMMEDIATELY.
+ * ------------------------------------------------------------------ */
+ jam();
+ /* ----------------------------------------------------------------
+ * IT MUST BE ACTIVE CREATION OF A FRAGMENT.
+ * ---------------------------------------------------------------- */
+ regTcPtr->gci = cnewestGci;
+ localCommitLab(signal);
+ return;
+ } else {
+ /* ------------------------------------------------------------------
+ * A NORMAL WRITE OPERATION ON A FRAGMENT WHICH DO NOT NEED LOGGING.
+ * WE WILL PACK THE REQUEST/RESPONSE TO THE NEXT NODE/TO TC.
+ * ------------------------------------------------------------------ */
+ jam();
+ /* ---------------------------------------------------------------
+ * IT MUST BE ACTIVE CREATION OF A FRAGMENT.
+ * NOT A DIRTY OPERATION THUS PACK REQUEST/RESPONSE.
+ * ---------------------------------------------------------------- */
+ regTcPtr->logWriteState = TcConnectionrec::NOT_WRITTEN;
+ packLqhkeyreqLab(signal);
+ return;
+ }//if
+ } else {
+ jam();
+ /* --------------------------------------------------------------------
+ * A DIRTY OPERATION WHICH NEEDS LOGGING. WE START BY LOGGING THE
+ * REQUEST. IN THIS CASE WE WILL RELEASE THE FRAGMENT LOCK FIRST.
+ * -------------------------------------------------------------------- */
+ /* A NORMAL WRITE OPERATION THAT NEEDS LOGGING AND WILL NOT BE
+ * PREMATURELY COMMITTED.
+ * -------------------------------------------------------------------- */
+ logLqhkeyreqLab(signal);
+ return;
+ }//if
+ }//if
+}//Dblqh::rwConcludedAiLab()
+
+/* ##########################################################################
+ * ####### LOG MODULE #######
+ *
+ * ##########################################################################
+ * --------------------------------------------------------------------------
+ * THE LOG MODULE HANDLES THE READING AND WRITING OF THE LOG
+ * IT IS ALSO RESPONSIBLE FOR HANDLING THE SYSTEM RESTART.
+ * IT CONTROLS THE SYSTEM RESTART IN TUP AND ACC AS WELL.
+ * -------------------------------------------------------------------------- */
+void Dblqh::logLqhkeyreqLab(Signal* signal)
+{
+ UintR tcurrentFilepage;
+ TcConnectionrecPtr tmpTcConnectptr;
+
+ if (cnoOfLogPages < ZMIN_LOG_PAGES_OPERATION || ERROR_INSERTED(5032)) {
+ jam();
+ if(ERROR_INSERTED(5032)){
+ CLEAR_ERROR_INSERT_VALUE;
+ }
+/*---------------------------------------------------------------------------*/
+// The log disk is having problems in catching up with the speed of execution.
+// We must wait with writing the log of this operation to ensure we do not
+// overload the log.
+/*---------------------------------------------------------------------------*/
+ terrorCode = ZTEMPORARY_REDO_LOG_FAILURE;
+ abortErrorLab(signal);
+ return;
+ }//if
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ logPartPtr.i = regTcPtr->hashValue & 3;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+/* -------------------------------------------------- */
+/* THIS PART IS USED TO WRITE THE LOG */
+/* -------------------------------------------------- */
+/* -------------------------------------------------- */
+/* CHECK IF A LOG OPERATION IS ONGOING ALREADY. */
+/* IF SO THEN QUEUE THE OPERATION FOR LATER */
+/* RESTART WHEN THE LOG PART IS FREE AGAIN. */
+/* -------------------------------------------------- */
+ LogPartRecord * const regLogPartPtr = logPartPtr.p;
+
+ if(ERROR_INSERTED(5033)){
+ jam();
+ CLEAR_ERROR_INSERT_VALUE;
+
+ if ((regLogPartPtr->firstLogQueue != RNIL) &&
+ (regLogPartPtr->LogLqhKeyReqSent == ZFALSE)) {
+ /* -------------------------------------------------- */
+ /* WE HAVE A PROBLEM IN THAT THE LOG HAS NO */
+ /* ROOM FOR ADDITIONAL OPERATIONS AT THE MOMENT.*/
+ /* -------------------------------------------------- */
+ /* -------------------------------------------------- */
+ /* WE MUST STILL RESTART QUEUED OPERATIONS SO */
+ /* THEY ALSO CAN BE ABORTED. */
+ /* -------------------------------------------------- */
+ regLogPartPtr->LogLqhKeyReqSent = ZTRUE;
+ signal->theData[0] = ZLOG_LQHKEYREQ;
+ signal->theData[1] = logPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+
+ terrorCode = ZTAIL_PROBLEM_IN_LOG_ERROR;
+ abortErrorLab(signal);
+ return;
+ }
+
+ if (regLogPartPtr->logPartState == LogPartRecord::IDLE) {
+ ;
+ } else if (regLogPartPtr->logPartState == LogPartRecord::ACTIVE) {
+ jam();
+ linkWaitLog(signal, logPartPtr);
+ regTcPtr->transactionState = TcConnectionrec::LOG_QUEUED;
+ return;
+ } else {
+ if ((regLogPartPtr->firstLogQueue != RNIL) &&
+ (regLogPartPtr->LogLqhKeyReqSent == ZFALSE)) {
+/* -------------------------------------------------- */
+/* WE HAVE A PROBLEM IN THAT THE LOG HAS NO */
+/* ROOM FOR ADDITIONAL OPERATIONS AT THE MOMENT.*/
+/* -------------------------------------------------- */
+/* -------------------------------------------------- */
+/* WE MUST STILL RESTART QUEUED OPERATIONS SO */
+/* THEY ALSO CAN BE ABORTED. */
+/* -------------------------------------------------- */
+ regLogPartPtr->LogLqhKeyReqSent = ZTRUE;
+ signal->theData[0] = ZLOG_LQHKEYREQ;
+ signal->theData[1] = logPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+ if (regLogPartPtr->logPartState == LogPartRecord::TAIL_PROBLEM) {
+ jam();
+ terrorCode = ZTAIL_PROBLEM_IN_LOG_ERROR;
+ } else {
+ ndbrequire(regLogPartPtr->logPartState == LogPartRecord::FILE_CHANGE_PROBLEM);
+ jam();
+ terrorCode = ZFILE_CHANGE_PROBLEM_IN_LOG_ERROR;
+ }//if
+ abortErrorLab(signal);
+ return;
+ }//if
+ regLogPartPtr->logPartState = LogPartRecord::ACTIVE;
+ logFilePtr.i = regLogPartPtr->currentLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+/* -------------------------------------------------- */
+/* CHECK IF A NEW MBYTE IS TO BE STARTED. IF */
+/* SO INSERT A NEXT LOG RECORD, WRITE THE LOG */
+/* AND PLACE THE LOG POINTER ON THE NEW POSITION*/
+/* IF A NEW FILE IS TO BE USED, CHANGE FILE AND */
+/* ALSO START OPENING THE NEXT LOG FILE. IF A */
+/* LAP HAS BEEN COMPLETED THEN ADD ONE TO LAP */
+/* COUNTER. */
+/* -------------------------------------------------- */
+ checkNewMbyte(signal);
+/* -------------------------------------------------- */
+/* INSERT THE OPERATION RECORD LAST IN THE LIST */
+/* OF NOT COMPLETED OPERATIONS. ALSO RECORD THE */
+/* FILE NO, PAGE NO AND PAGE INDEX OF THE START */
+/* OF THIS LOG RECORD. */
+/* IT IS NOT ALLOWED TO INSERT IT INTO THE LIST */
+/* BEFORE CHECKING THE NEW MBYTE SINCE THAT WILL*/
+/* CAUSE THE OLD VALUES OF TC_CONNECTPTR TO BE */
+/* USED IN WRITE_FILE_DESCRIPTOR. */
+/* -------------------------------------------------- */
+ Uint32 tcIndex = tcConnectptr.i;
+ tmpTcConnectptr.i = regLogPartPtr->lastLogTcrec;
+ regLogPartPtr->lastLogTcrec = tcIndex;
+ if (tmpTcConnectptr.i == RNIL) {
+ jam();
+ regLogPartPtr->firstLogTcrec = tcIndex;
+ } else {
+ ptrCheckGuard(tmpTcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ tmpTcConnectptr.p->nextLogTcrec = tcIndex;
+ }//if
+ Uint32 fileNo = logFilePtr.p->fileNo;
+ tcurrentFilepage = logFilePtr.p->currentFilepage;
+ logPagePtr.i = logFilePtr.p->currentLogpage;
+ regTcPtr->nextLogTcrec = RNIL;
+ regTcPtr->prevLogTcrec = tmpTcConnectptr.i;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ Uint32 pageIndex = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ regTcPtr->logStartFileNo = fileNo;
+ regTcPtr->logStartPageNo = tcurrentFilepage;
+ regTcPtr->logStartPageIndex = pageIndex;
+/* -------------------------------------------------- */
+/* WRITE THE LOG HEADER OF THIS OPERATION. */
+/* -------------------------------------------------- */
+ writeLogHeader(signal);
+/* -------------------------------------------------- */
+/* WRITE THE TUPLE KEY OF THIS OPERATION. */
+/* -------------------------------------------------- */
+ writeKey(signal);
+/* -------------------------------------------------- */
+/* WRITE THE ATTRIBUTE INFO OF THIS OPERATION. */
+/* -------------------------------------------------- */
+ writeAttrinfoLab(signal);
+
+ logNextStart(signal);
+/* -------------------------------------------------- */
+/* RESET THE STATE OF THE LOG PART. IF ANY */
+/* OPERATIONS HAVE QUEUED THEN START THE FIRST */
+/* OF THESE. */
+/* -------------------------------------------------- */
+/* -------------------------------------------------- */
+/* CONTINUE WITH PACKING OF LQHKEYREQ */
+/* -------------------------------------------------- */
+ tcurrentFilepage = logFilePtr.p->currentFilepage;
+ if (logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] == ZPAGE_HEADER_SIZE) {
+ jam();
+ tcurrentFilepage--;
+ }//if
+ regTcPtr->logStopPageNo = tcurrentFilepage;
+ regTcPtr->logWriteState = TcConnectionrec::WRITTEN;
+ if (regTcPtr->abortState != TcConnectionrec::ABORT_IDLE) {
+/* -------------------------------------------------- */
+/* AN ABORT HAVE BEEN ORDERED. THE ABORT WAITED */
+/* FOR THE LOG WRITE TO BE COMPLETED. NOW WE */
+/* CAN PROCEED WITH THE NORMAL ABORT HANDLING. */
+/* -------------------------------------------------- */
+ abortCommonLab(signal);
+ return;
+ }//if
+ if (regTcPtr->dirtyOp != ZTRUE) {
+ packLqhkeyreqLab(signal);
+ } else {
+ /* ----------------------------------------------------------------------
+ * I NEED TO INSERT A COMMIT LOG RECORD SINCE WE ARE WRITING LOG IN THIS
+ * TRANSACTION. SINCE WE RELEASED THE LOG LOCK JUST NOW NO ONE ELSE CAN BE
+ * ACTIVE IN WRITING THE LOG. WE THUS WRITE THE LOG WITHOUT GETTING A LOCK
+ * SINCE WE ARE ONLY WRITING A COMMIT LOG RECORD.
+ * ---------------------------------------------------------------------- */
+ writeCommitLog(signal, logPartPtr);
+ /* ----------------------------------------------------------------------
+ * DIRTY OPERATIONS SHOULD COMMIT BEFORE THEY PACK THE REQUEST/RESPONSE.
+ * ---------------------------------------------------------------------- */
+ regTcPtr->gci = cnewestGci;
+ localCommitLab(signal);
+ }//if
+}//Dblqh::logLqhkeyreqLab()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEND LQHKEYREQ */
+/* */
+/* NO STATE CHECKING SINCE THE SIGNAL IS A LOCAL SIGNAL. THE EXECUTION OF */
+/* THE OPERATION IS COMPLETED. IT IS NOW TIME TO SEND THE OPERATION TO THE */
+/* NEXT REPLICA OR TO TC. */
+/* ------------------------------------------------------------------------- */
+void Dblqh::packLqhkeyreqLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->nextReplica == ZNIL) {
+/* ------------------------------------------------------------------------- */
+/* ------- SEND LQHKEYCONF ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+ sendLqhkeyconfTc(signal, regTcPtr->tcBlockref);
+ if (regTcPtr->dirtyOp != ZTRUE) {
+ jam();
+ regTcPtr->transactionState = TcConnectionrec::PREPARED;
+ releaseOprec(signal);
+ } else {
+ jam();
+/*************************************************************>*/
+/* DIRTY WRITES ARE USED IN TWO SITUATIONS. THE FIRST */
+/* SITUATION IS WHEN THEY ARE USED TO UPDATE COUNTERS AND*/
+/* OTHER ATTRIBUTES WHICH ARE NOT SENSITIVE TO CONSISTE- */
+/* NCY. THE SECOND SITUATION IS BY OPERATIONS THAT ARE */
+/* SENT AS PART OF A COPY FRAGMENT PROCESS. */
+/* */
+/* DURING A COPY FRAGMENT PROCESS THERE IS NO LOGGING */
+/* ONGOING SINCE THE FRAGMENT IS NOT COMPLETE YET. THE */
+/* LOGGING STARTS AFTER COMPLETING THE LAST COPY TUPLE */
+/* OPERATION. THE EXECUTION OF THE LAST COPY TUPLE DOES */
+/* ALSO START A LOCAL CHECKPOINT SO THAT THE FRAGMENT */
+/* REPLICA IS RECOVERABLE. THUS GLOBAL CHECKPOINT ID FOR */
+/* THOSE OPERATIONS ARE NOT INTERESTING. */
+/* */
+/* A DIRTY WRITE IS BY DEFINITION NOT CONSISTENT. THUS */
+/* IT CAN USE ANY GLOBAL CHECKPOINT. THE IDEA HERE IS TO */
+/* ALWAYS USE THE LATEST DEFINED GLOBAL CHECKPOINT ID IN */
+/* THIS NODE. */
+/*************************************************************>*/
+ cleanUp(signal);
+ }//if
+ return;
+ }//if
+/* ------------------------------------------------------------------------- */
+/* ------- SEND LQHKEYREQ ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* THERE ARE MORE REPLICAS TO SEND THE OPERATION TO. A NEW LQHKEYREQ WILL BE */
+/* PREPARED FOR THE NEXT REPLICA. */
+/* ------------------------------------------------------------------------- */
+/* CLEAR REPLICA TYPE, ATTRINFO INDICATOR (IN LQHKEYREQ), */
+/* INTERPRETED EXECUTION, SEQUENTIAL NUMBER OF REPLICA. */
+// Set bit indicating Client and TC record not the same.
+// Set readlenAi indicator if readlenAi != 0
+// Stored Procedure Indicator not set.
+/* ------------------------------------------------------------------------- */
+ LqhKeyReq * const lqhKeyReq = (LqhKeyReq *)&signal->theData[0];
+
+ UintR Treqinfo;
+ UintR sig0, sig1, sig2, sig3, sig4, sig5, sig6;
+ Treqinfo = preComputedRequestInfoMask & regTcPtr->reqinfo;
+
+ UintR TapplAddressIndicator = (regTcPtr->nextSeqNoReplica == 0 ? 0 : 1);
+ LqhKeyReq::setApplicationAddressFlag(Treqinfo, TapplAddressIndicator);
+ LqhKeyReq::setInterpretedFlag(Treqinfo, regTcPtr->opExec);
+ LqhKeyReq::setSeqNoReplica(Treqinfo, regTcPtr->nextSeqNoReplica);
+ LqhKeyReq::setAIInLqhKeyReq(Treqinfo, regTcPtr->reclenAiLqhkey);
+ UintR TreadLenAiInd = (regTcPtr->readlenAi == 0 ? 0 : 1);
+ UintR TsameLqhAndClient = (tcConnectptr.i ==
+ regTcPtr->tcOprec ? 0 : 1);
+ LqhKeyReq::setSameClientAndTcFlag(Treqinfo, TsameLqhAndClient);
+ LqhKeyReq::setReturnedReadLenAIFlag(Treqinfo, TreadLenAiInd);
+
+ UintR TotReclenAi = regTcPtr->totSendlenAi;
+/* ------------------------------------------------------------------------- */
+/* WE ARE NOW PREPARED TO SEND THE LQHKEYREQ. WE HAVE TO DECIDE IF ATTRINFO */
+/* IS INCLUDED IN THE LQHKEYREQ SIGNAL AND THEN SEND IT. */
+/* TAKE OVER SCAN OPERATION IS NEVER USED ON BACKUPS, LOG RECORDS AND START-UP*/
+/* OF NEW REPLICA AND THUS ONLY TOT_SENDLEN_AI IS USED THE UPPER 16 BITS ARE */
+/* ZERO. */
+/* ------------------------------------------------------------------------- */
+ sig0 = tcConnectptr.i;
+ sig1 = regTcPtr->savePointId;
+ sig2 = regTcPtr->hashValue;
+ sig4 = regTcPtr->tcBlockref;
+
+ lqhKeyReq->clientConnectPtr = sig0;
+ lqhKeyReq->attrLen = TotReclenAi;
+ lqhKeyReq->savePointId = sig1;
+ lqhKeyReq->hashValue = sig2;
+ lqhKeyReq->requestInfo = Treqinfo;
+ lqhKeyReq->tcBlockref = sig4;
+
+ sig0 = regTcPtr->tableref + (regTcPtr->schemaVersion << 16);
+ sig1 = regTcPtr->fragmentid + (regTcPtr->nodeAfterNext[0] << 16);
+ sig2 = regTcPtr->transid[0];
+ sig3 = regTcPtr->transid[1];
+ sig4 = regTcPtr->applRef;
+ sig5 = regTcPtr->applOprec;
+ sig6 = regTcPtr->tcOprec;
+ UintR nextPos = (TapplAddressIndicator << 1);
+
+ lqhKeyReq->tableSchemaVersion = sig0;
+ lqhKeyReq->fragmentData = sig1;
+ lqhKeyReq->transId1 = sig2;
+ lqhKeyReq->transId2 = sig3;
+ lqhKeyReq->noFiredTriggers = regTcPtr->noFiredTriggers;
+ lqhKeyReq->variableData[0] = sig4;
+ lqhKeyReq->variableData[1] = sig5;
+ lqhKeyReq->variableData[2] = sig6;
+
+ nextPos += TsameLqhAndClient;
+
+ if ((regTcPtr->lastReplicaNo - regTcPtr->nextSeqNoReplica) > 1) {
+ sig0 = (UintR)regTcPtr->nodeAfterNext[1] +
+ (UintR)(regTcPtr->nodeAfterNext[2] << 16);
+ lqhKeyReq->variableData[nextPos] = sig0;
+ nextPos++;
+ }//if
+ sig0 = regTcPtr->readlenAi;
+ sig1 = regTcPtr->tupkeyData[0];
+ sig2 = regTcPtr->tupkeyData[1];
+ sig3 = regTcPtr->tupkeyData[2];
+ sig4 = regTcPtr->tupkeyData[3];
+
+ lqhKeyReq->variableData[nextPos] = sig0;
+ nextPos += TreadLenAiInd;
+ lqhKeyReq->variableData[nextPos] = sig1;
+ lqhKeyReq->variableData[nextPos + 1] = sig2;
+ lqhKeyReq->variableData[nextPos + 2] = sig3;
+ lqhKeyReq->variableData[nextPos + 3] = sig4;
+ UintR TkeyLen = LqhKeyReq::getKeyLen(Treqinfo);
+ if (TkeyLen < 4) {
+ nextPos += TkeyLen;
+ } else {
+ nextPos += 4;
+ }//if
+
+ sig0 = regTcPtr->firstAttrinfo[0];
+ sig1 = regTcPtr->firstAttrinfo[1];
+ sig2 = regTcPtr->firstAttrinfo[2];
+ sig3 = regTcPtr->firstAttrinfo[3];
+ sig4 = regTcPtr->firstAttrinfo[4];
+ UintR TAiLen = regTcPtr->reclenAiLqhkey;
+ BlockReference lqhRef = calcLqhBlockRef(regTcPtr->nextReplica);
+
+ lqhKeyReq->variableData[nextPos] = sig0;
+ lqhKeyReq->variableData[nextPos + 1] = sig1;
+ lqhKeyReq->variableData[nextPos + 2] = sig2;
+ lqhKeyReq->variableData[nextPos + 3] = sig3;
+ lqhKeyReq->variableData[nextPos + 4] = sig4;
+
+ nextPos += TAiLen;
+
+ sendSignal(lqhRef, GSN_LQHKEYREQ, signal,
+ nextPos + LqhKeyReq::FixedSignalLength, JBB);
+ if (regTcPtr->primKeyLen > 4) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* MORE THAN 4 WORDS OF KEY DATA IS IN THE OPERATION. THEREFORE WE NEED TO */
+/* PREPARE A KEYINFO SIGNAL. MORE THAN ONE KEYINFO SIGNAL CAN BE SENT. */
+/* ------------------------------------------------------------------------- */
+ sendTupkey(signal);
+ }//if
+/* ------------------------------------------------------------------------- */
+/* NOW I AM PREPARED TO SEND ALL THE ATTRINFO SIGNALS. AT THE MOMENT A LOOP */
+/* SENDS ALL AT ONCE. LATER WE HAVE TO ADDRESS THE PROBLEM THAT THESE COULD */
+/* LEAD TO BUFFER EXPLOSION => NODE CRASH. */
+/* ------------------------------------------------------------------------- */
+/* NEW CODE TO SEND ATTRINFO IN PACK_LQHKEYREQ */
+/* THIS CODE USES A REAL-TIME BREAK AFTER */
+/* SENDING 16 SIGNALS. */
+/* -------------------------------------------------- */
+ sig0 = regTcPtr->tcOprec;
+ sig1 = regTcPtr->transid[0];
+ sig2 = regTcPtr->transid[1];
+ signal->theData[0] = sig0;
+ signal->theData[1] = sig1;
+ signal->theData[2] = sig2;
+ AttrbufPtr regAttrinbufptr;
+ regAttrinbufptr.i = regTcPtr->firstAttrinbuf;
+ while (regAttrinbufptr.i != RNIL) {
+ ptrCheckGuard(regAttrinbufptr, cattrinbufFileSize, attrbuf);
+ jam();
+ Uint32 dataLen = regAttrinbufptr.p->attrbuf[ZINBUF_DATA_LEN];
+ ndbrequire(dataLen != 0);
+ MEMCOPY_NO_WORDS(&signal->theData[3], &regAttrinbufptr.p->attrbuf[0], dataLen);
+ regAttrinbufptr.i = regAttrinbufptr.p->attrbuf[ZINBUF_NEXT];
+ sendSignal(lqhRef, GSN_ATTRINFO, signal, dataLen + 3, JBB);
+ }//while
+ regTcPtr->transactionState = TcConnectionrec::PREPARED;
+ if (regTcPtr->dirtyOp == ZTRUE) {
+ jam();
+/*************************************************************>*/
+/* DIRTY WRITES ARE USED IN TWO SITUATIONS. THE FIRST */
+/* SITUATION IS WHEN THEY ARE USED TO UPDATE COUNTERS AND*/
+/* OTHER ATTRIBUTES WHICH ARE NOT SENSITIVE TO CONSISTE- */
+/* NCY. THE SECOND SITUATION IS BY OPERATIONS THAT ARE */
+/* SENT AS PART OF A COPY FRAGMENT PROCESS. */
+/* */
+/* DURING A COPY FRAGMENT PROCESS THERE IS NO LOGGING */
+/* ONGOING SINCE THE FRAGMENT IS NOT COMPLETE YET. THE */
+/* LOGGING STARTS AFTER COMPLETING THE LAST COPY TUPLE */
+/* OPERATION. THE EXECUTION OF THE LAST COPY TUPLE DOES */
+/* ALSO START A LOCAL CHECKPOINT SO THAT THE FRAGMENT */
+/* REPLICA IS RECOVERABLE. THUS GLOBAL CHECKPOINT ID FOR */
+/* THOSE OPERATIONS ARE NOT INTERESTING. */
+/* */
+/* A DIRTY WRITE IS BY DEFINITION NOT CONSISTENT. THUS */
+/* IT CAN USE ANY GLOBAL CHECKPOINT. THE IDEA HERE IS TO */
+/* ALWAYS USE THE LATEST DEFINED GLOBAL CHECKPOINT ID IN */
+/* THIS NODE. */
+/*************************************************************>*/
+ cleanUp(signal);
+ return;
+ }//if
+ /* ------------------------------------------------------------------------
+ * ALL INFORMATION NEEDED BY THE COMMIT PHASE AND COMPLETE PHASE IS
+ * KEPT IN THE TC_CONNECT RECORD. TO ENSURE PROPER USE OF MEMORY
+ * RESOURCES WE DEALLOCATE THE ATTRINFO RECORD AND KEY RECORDS
+ * AS SOON AS POSSIBLE.
+ * ------------------------------------------------------------------------ */
+ releaseOprec(signal);
+}//Dblqh::packLqhkeyreqLab()
+
+/* ========================================================================= */
+/* ==== CHECK IF THE LOG RECORD FITS INTO THE CURRENT MBYTE, ======= */
+/* OTHERWISE SWITCH TO NEXT MBYTE. */
+/* */
+/* ========================================================================= */
+void Dblqh::checkNewMbyte(Signal* signal)
+{
+ UintR tcnmTmp;
+ UintR ttotalLogSize;
+
+/* -------------------------------------------------- */
+/* CHECK IF A NEW MBYTE OF LOG RECORD IS TO BE */
+/* OPENED BEFORE WRITING THE LOG RECORD. NO LOG */
+/* RECORDS ARE ALLOWED TO SPAN A MBYTE BOUNDARY */
+/* */
+/* INPUT: TC_CONNECTPTR THE OPERATION */
+/* LOG_FILE_PTR THE LOG FILE */
+/* OUTPUT: LOG_FILE_PTR THE NEW LOG FILE */
+/* -------------------------------------------------- */
+ ttotalLogSize = ZLOG_HEAD_SIZE + tcConnectptr.p->currTupAiLen;
+ ttotalLogSize = ttotalLogSize + tcConnectptr.p->primKeyLen;
+ tcnmTmp = logFilePtr.p->remainingWordsInMbyte;
+ if ((ttotalLogSize + ZNEXT_LOG_SIZE) <= tcnmTmp) {
+ ndbrequire(tcnmTmp >= ttotalLogSize);
+ logFilePtr.p->remainingWordsInMbyte = tcnmTmp - ttotalLogSize;
+ return;
+ } else {
+ jam();
+/* -------------------------------------------------- */
+/* IT WAS NOT ENOUGH SPACE IN THIS MBYTE FOR */
+/* THIS LOG RECORD. MOVE TO NEXT MBYTE */
+/* THIS MIGHT INCLUDE CHANGING LOG FILE */
+/* -------------------------------------------------- */
+/* WE HAVE TO INSERT A NEXT LOG RECORD FIRST */
+/* -------------------------------------------------- */
+/* THEN CONTINUE BY WRITING THE FILE DESCRIPTORS*/
+/* -------------------------------------------------- */
+ logPagePtr.i = logFilePtr.p->currentLogpage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ changeMbyte(signal);
+ tcnmTmp = logFilePtr.p->remainingWordsInMbyte;
+ }//if
+ ndbrequire(tcnmTmp >= ttotalLogSize);
+ logFilePtr.p->remainingWordsInMbyte = tcnmTmp - ttotalLogSize;
+}//Dblqh::checkNewMbyte()
+
+/* --------------------------------------------------------------------------
+ * ------- WRITE OPERATION HEADER TO LOG -------
+ *
+ * SUBROUTINE SHORT NAME: WLH
+ * ------------------------------------------------------------------------- */
+void Dblqh::writeLogHeader(Signal* signal)
+{
+ Uint32 logPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ Uint32 hashValue = tcConnectptr.p->hashValue;
+ Uint32 operation = tcConnectptr.p->operation;
+ Uint32 keyLen = tcConnectptr.p->primKeyLen;
+ Uint32 aiLen = tcConnectptr.p->currTupAiLen;
+ Uint32 totLogLen = aiLen + keyLen + ZLOG_HEAD_SIZE;
+ if ((logPos + ZLOG_HEAD_SIZE) < ZPAGE_SIZE) {
+ Uint32* dataPtr = &logPagePtr.p->logPageWord[logPos];
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPos + ZLOG_HEAD_SIZE;
+ dataPtr[0] = ZPREP_OP_TYPE;
+ dataPtr[1] = totLogLen;
+ dataPtr[2] = hashValue;
+ dataPtr[3] = operation;
+ dataPtr[4] = aiLen;
+ dataPtr[5] = keyLen;
+ } else {
+ writeLogWord(signal, ZPREP_OP_TYPE);
+ writeLogWord(signal, totLogLen);
+ writeLogWord(signal, hashValue);
+ writeLogWord(signal, operation);
+ writeLogWord(signal, aiLen);
+ writeLogWord(signal, keyLen);
+ }//if
+}//Dblqh::writeLogHeader()
+
+/* --------------------------------------------------------------------------
+ * ------- WRITE TUPLE KEY TO LOG -------
+ *
+ * SUBROUTINE SHORT NAME: WK
+ * ------------------------------------------------------------------------- */
+void Dblqh::writeKey(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ Uint32 logPos, endPos, dataLen;
+ Int32 remainingLen;
+ logPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ remainingLen = regTcPtr->primKeyLen;
+ dataLen = remainingLen;
+ if (remainingLen > 4)
+ dataLen = 4;
+ remainingLen -= dataLen;
+ endPos = logPos + dataLen;
+ if (endPos < ZPAGE_SIZE) {
+ MEMCOPY_NO_WORDS(&logPagePtr.p->logPageWord[logPos],
+ &regTcPtr->tupkeyData[0],
+ dataLen);
+ } else {
+ jam();
+ for (Uint32 i = 0; i < dataLen; i++)
+ writeLogWord(signal, regTcPtr->tupkeyData[i]);
+ endPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ }//if
+ DatabufPtr regDatabufptr;
+ regDatabufptr.i = regTcPtr->firstTupkeybuf;
+ while (remainingLen > 0) {
+ logPos = endPos;
+ ptrCheckGuard(regDatabufptr, cdatabufFileSize, databuf);
+ dataLen = remainingLen;
+ if (remainingLen > 4)
+ dataLen = 4;
+ remainingLen -= dataLen;
+ endPos += dataLen;
+ if (endPos < ZPAGE_SIZE) {
+ MEMCOPY_NO_WORDS(&logPagePtr.p->logPageWord[logPos],
+ &regDatabufptr.p->data[0],
+ dataLen);
+ } else {
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPos;
+ for (Uint32 i = 0; i < dataLen; i++)
+ writeLogWord(signal, regDatabufptr.p->data[i]);
+ endPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ }//if
+ regDatabufptr.i = regDatabufptr.p->nextDatabuf;
+ }//while
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = endPos;
+ ndbrequire(regDatabufptr.i == RNIL);
+}//Dblqh::writeKey()
+
+/* --------------------------------------------------------------------------
+ * ------- WRITE ATTRINFO TO LOG -------
+ *
+ * SUBROUTINE SHORT NAME: WA
+ * ------------------------------------------------------------------------- */
+void Dblqh::writeAttrinfoLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ Uint32 totLen = regTcPtr->currTupAiLen;
+ if (totLen == 0)
+ return;
+ Uint32 logPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ Uint32 lqhLen = regTcPtr->reclenAiLqhkey;
+ ndbrequire(totLen >= lqhLen);
+ Uint32 endPos = logPos + lqhLen;
+ totLen -= lqhLen;
+ if (endPos < ZPAGE_SIZE) {
+ MEMCOPY_NO_WORDS(&logPagePtr.p->logPageWord[logPos],
+ &regTcPtr->firstAttrinfo[0],
+ lqhLen);
+ } else {
+ for (Uint32 i = 0; i < lqhLen; i++)
+ writeLogWord(signal, regTcPtr->firstAttrinfo[i]);
+ endPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ }//if
+ AttrbufPtr regAttrinbufptr;
+ regAttrinbufptr.i = regTcPtr->firstAttrinbuf;
+ while (totLen > 0) {
+ logPos = endPos;
+ ptrCheckGuard(regAttrinbufptr, cattrinbufFileSize, attrbuf);
+ Uint32 dataLen = regAttrinbufptr.p->attrbuf[ZINBUF_DATA_LEN];
+ ndbrequire(totLen >= dataLen);
+ ndbrequire(dataLen > 0);
+ totLen -= dataLen;
+ endPos += dataLen;
+ if (endPos < ZPAGE_SIZE) {
+ MEMCOPY_NO_WORDS(&logPagePtr.p->logPageWord[logPos],
+ &regAttrinbufptr.p->attrbuf[0],
+ dataLen);
+ } else {
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPos;
+ for (Uint32 i = 0; i < dataLen; i++)
+ writeLogWord(signal, regAttrinbufptr.p->attrbuf[i]);
+ endPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ }//if
+ regAttrinbufptr.i = regAttrinbufptr.p->attrbuf[ZINBUF_NEXT];
+ }//while
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = endPos;
+ ndbrequire(regAttrinbufptr.i == RNIL);
+}//Dblqh::writeAttrinfoLab()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEND TUPLE KEY IN KEYINFO SIGNAL(S) ------- */
+/* */
+/* SUBROUTINE SHORT NAME: STU */
+/* ------------------------------------------------------------------------- */
+void Dblqh::sendTupkey(Signal* signal)
+{
+ UintR TdataPos = 3;
+ BlockReference lqhRef = calcLqhBlockRef(tcConnectptr.p->nextReplica);
+ signal->theData[0] = tcConnectptr.p->tcOprec;
+ signal->theData[1] = tcConnectptr.p->transid[0];
+ signal->theData[2] = tcConnectptr.p->transid[1];
+ databufptr.i = tcConnectptr.p->firstTupkeybuf;
+ do {
+ ptrCheckGuard(databufptr, cdatabufFileSize, databuf);
+ signal->theData[TdataPos] = databufptr.p->data[0];
+ signal->theData[TdataPos + 1] = databufptr.p->data[1];
+ signal->theData[TdataPos + 2] = databufptr.p->data[2];
+ signal->theData[TdataPos + 3] = databufptr.p->data[3];
+
+ databufptr.i = databufptr.p->nextDatabuf;
+ TdataPos += 4;
+ if (databufptr.i == RNIL) {
+ jam();
+ sendSignal(lqhRef, GSN_KEYINFO, signal, TdataPos, JBB);
+ return;
+ } else if (TdataPos == 23) {
+ jam();
+ sendSignal(lqhRef, GSN_KEYINFO, signal, 23, JBB);
+ TdataPos = 3;
+ }
+ } while (1);
+}//Dblqh::sendTupkey()
+
+void Dblqh::cleanUp(Signal* signal)
+{
+ releaseOprec(signal);
+ deleteTransidHash(signal);
+ releaseTcrec(signal, tcConnectptr);
+}//Dblqh::cleanUp()
+
+/* --------------------------------------------------------------------------
+ * ---- RELEASE ALL RECORDS CONNECTED TO THE OPERATION RECORD AND THE ----
+ * OPERATION RECORD ITSELF
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseOprec(Signal* signal)
+{
+ UintR Tmpbuf;
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+/* ---- RELEASE DATA BUFFERS ------------------- */
+ DatabufPtr regDatabufptr;
+ regDatabufptr.i = regTcPtr->firstTupkeybuf;
+/* --------------------------------------------------------------------------
+ * ------- RELEASE DATA BUFFERS -------
+ *
+ * ------------------------------------------------------------------------- */
+
+ while (regDatabufptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(regDatabufptr, cdatabufFileSize, databuf);
+ Tmpbuf = regDatabufptr.p->nextDatabuf;
+ regDatabufptr.p->nextDatabuf = cfirstfreeDatabuf;
+ cfirstfreeDatabuf = regDatabufptr.i;
+ regDatabufptr.i = Tmpbuf;
+ }//while
+/* ---- RELEASE ATTRINFO BUFFERS ------------------- */
+ AttrbufPtr regAttrinbufptr;
+ regAttrinbufptr.i = regTcPtr->firstAttrinbuf;
+ /* ########################################################################
+ * ####### RELEASE_ATTRINBUF #######
+ *
+ * ####################################################################### */
+ while (regAttrinbufptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(regAttrinbufptr, cattrinbufFileSize, attrbuf);
+ Tmpbuf = regAttrinbufptr.p->attrbuf[ZINBUF_NEXT];
+ regAttrinbufptr.p->attrbuf[ZINBUF_NEXT] = cfirstfreeAttrinbuf;
+ cfirstfreeAttrinbuf = regAttrinbufptr.i;
+ regAttrinbufptr.i = Tmpbuf;
+ }//while
+ regTcPtr->firstAttrinbuf = RNIL;
+ regTcPtr->lastAttrinbuf = RNIL;
+ regTcPtr->firstTupkeybuf = RNIL;
+ regTcPtr->lastTupkeybuf = RNIL;
+}//Dblqh::releaseOprec()
+
+/* ------------------------------------------------------------------------- */
+/* ------ DELETE TRANSACTION ID FROM HASH TABLE ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::deleteTransidHash(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ TcConnectionrecPtr prevHashptr;
+ TcConnectionrecPtr nextHashptr;
+
+ prevHashptr.i = regTcPtr->prevHashRec;
+ nextHashptr.i = regTcPtr->nextHashRec;
+ if (prevHashptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(prevHashptr, ctcConnectrecFileSize, tcConnectionrec);
+ prevHashptr.p->nextHashRec = nextHashptr.i;
+ } else {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* THE OPERATION WAS PLACED FIRST IN THE LIST OF THE HASH TABLE. NEED TO SET */
+/* A NEW LEADER OF THE LIST. */
+/* ------------------------------------------------------------------------- */
+ Uint32 hashIndex = (regTcPtr->transid[0] ^ regTcPtr->tcOprec) & 1023;
+ ctransidHash[hashIndex] = nextHashptr.i;
+ }//if
+ if (nextHashptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(nextHashptr, ctcConnectrecFileSize, tcConnectionrec);
+ nextHashptr.p->prevHashRec = prevHashptr.i;
+ }//if
+}//Dblqh::deleteTransidHash()
+
+/* --------------------------------------------------------------------------
+ * ------- LINK OPERATION IN ACTIVE LIST ON FRAGMENT -------
+ *
+ * SUBROUTINE SHORT NAME: LAF
+// Input Pointers:
+// tcConnectptr
+// fragptr
+ * ------------------------------------------------------------------------- */
+void Dblqh::linkActiveFrag(Signal* signal)
+{
+ TcConnectionrecPtr lafTcConnectptr;
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ Fragrecord * const regFragPtr = fragptr.p;
+ Uint32 tcIndex = tcConnectptr.i;
+ lafTcConnectptr.i = regFragPtr->activeList;
+ regTcPtr->prevTc = RNIL;
+ regFragPtr->activeList = tcIndex;
+ ndbrequire(regTcPtr->listState == TcConnectionrec::NOT_IN_LIST);
+ regTcPtr->nextTc = lafTcConnectptr.i;
+ regTcPtr->listState = TcConnectionrec::IN_ACTIVE_LIST;
+ if (lafTcConnectptr.i == RNIL) {
+ return;
+ } else {
+ jam();
+ ptrCheckGuard(lafTcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ lafTcConnectptr.p->prevTc = tcIndex;
+ }//if
+ return;
+}//Dblqh::linkActiveFrag()
+
+/* -------------------------------------------------------------------------
+ * ------- RELEASE OPERATION FROM ACTIVE LIST ON FRAGMENT -------
+ *
+ * SUBROUTINE SHORT NAME = RAF
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseActiveFrag(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ TcConnectionrecPtr ralTcNextConnectptr;
+ TcConnectionrecPtr ralTcPrevConnectptr;
+ fragptr.i = regTcPtr->fragmentptr;
+ ralTcPrevConnectptr.i = regTcPtr->prevTc;
+ ralTcNextConnectptr.i = regTcPtr->nextTc;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ Fragrecord * const regFragPtr = fragptr.p;
+ ndbrequire(regTcPtr->listState == TcConnectionrec::IN_ACTIVE_LIST);
+ regTcPtr->listState = TcConnectionrec::NOT_IN_LIST;
+
+ if (ralTcNextConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(ralTcNextConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ ralTcNextConnectptr.p->prevTc = ralTcPrevConnectptr.i;
+ }//if
+ if (ralTcPrevConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(ralTcPrevConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ ralTcPrevConnectptr.p->nextTc = regTcPtr->nextTc;
+ } else {
+ jam();
+ /* ----------------------------------------------------------------------
+ * OPERATION RECORD IS FIRST IN ACTIVE LIST
+ * THIS MEANS THAT THERE EXISTS NO PREVIOUS TC THAT NEEDS TO BE UPDATED.
+ * --------------------------------------------------------------------- */
+ regFragPtr->activeList = ralTcNextConnectptr.i;
+ }//if
+ if (regFragPtr->lcpRef != RNIL) {
+ jam();
+ lcpPtr.i = regFragPtr->lcpRef;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ ndbrequire(lcpPtr.p->lcpState == LcpRecord::LCP_WAIT_ACTIVE_FINISH);
+
+ /* --------------------------------------------------------------------
+ * IF A FRAGMENT IS CURRENTLY STARTING A LOCAL CHECKPOINT AND IT
+ * IS WAITING FOR ACTIVE OPERATIONS TO BE COMPLETED WITH THE
+ * CURRENT PHASE, THEN IT IS CHECKED WHETHER THE
+ * LAST ACTIVE OPERATION WAS NOW COMPLETED.
+ * ------------------------------------------------------------------- */
+ if (regFragPtr->activeList == RNIL) {
+ jam();
+ /* ------------------------------------------------------------------
+ * ACTIVE LIST ON FRAGMENT IS EMPTY AND WE ARE WAITING FOR
+ * THIS TO HAPPEN.
+ * WE WILL NOW START THE CHECKPOINT IN TUP AND ACC.
+ * ----------------------------------------------------------------- */
+ /* SEND START LOCAL CHECKPOINT TO ACC AND TUP */
+ /* ----------------------------------------------------------------- */
+ fragptr.p->lcpRef = RNIL;
+ lcpPtr.p->lcpState = LcpRecord::LCP_START_CHKP;
+ sendStartLcp(signal);
+ }//if
+ }//if
+}//Dblqh::releaseActiveFrag()
+
+/* ######################################################################### */
+/* ####### TRANSACTION MODULE ####### */
+/* THIS MODULE HANDLES THE COMMIT AND THE COMPLETE PHASE. */
+/* ######################################################################### */
+void Dblqh::warningReport(Signal* signal, int place)
+{
+ switch (place) {
+ case 0:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMMIT in wrong state in Dblqh" << endl;
+#endif
+ break;
+ case 1:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMMIT with wrong transid in Dblqh" << endl;
+#endif
+ break;
+ case 2:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMPLETE in wrong state in Dblqh" << endl;
+#endif
+ break;
+ case 3:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMPLETE with wrong transid in Dblqh" << endl;
+#endif
+ break;
+ case 4:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMMITREQ in wrong state in Dblqh" << endl;
+#endif
+ break;
+ case 5:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMMITREQ with wrong transid in Dblqh" << endl;
+#endif
+ break;
+ case 6:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMPLETEREQ in wrong state in Dblqh" << endl;
+#endif
+ break;
+ case 7:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMPLETEREQ with wrong transid in Dblqh" << endl;
+#endif
+ break;
+ case 8:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received ABORT with non-existing transid in Dblqh" << endl;
+#endif
+ break;
+ case 9:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received ABORTREQ with non-existing transid in Dblqh" << endl;
+#endif
+ break;
+ case 10:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received ABORTREQ in wrong state in Dblqh" << endl;
+#endif
+ break;
+ case 11:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMMIT when tc-rec released in Dblqh" << endl;
+#endif
+ break;
+ case 12:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received COMPLETE when tc-rec released in Dblqh" << endl;
+#endif
+ break;
+ case 13:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received LQHKEYREF when tc-rec released in Dblqh" << endl;
+#endif
+ break;
+ case 14:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received LQHKEYREF with wrong transid in Dblqh" << endl;
+#endif
+ break;
+ case 15:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "W: Received LQHKEYREF when already aborting in Dblqh" << endl;
+#endif
+ break;
+ case 16:
+ jam();
+ ndbrequire(cstartPhase == ZNIL);
+#ifdef ABORT_TRACE
+ ndbout << "W: Received LQHKEYREF in wrong state in Dblqh" << endl;
+#endif
+ break;
+ default:
+ jam();
+ break;
+ }//switch
+ return;
+}//Dblqh::warningReport()
+
+void Dblqh::errorReport(Signal* signal, int place)
+{
+ switch (place) {
+ case 0:
+ jam();
+ break;
+ case 1:
+ jam();
+ break;
+ case 2:
+ jam();
+ break;
+ case 3:
+ jam();
+ break;
+ default:
+ jam();
+ break;
+ }//switch
+ systemErrorLab(signal);
+ return;
+}//Dblqh::errorReport()
+
+/* ************************************************************************>>
+ * COMMIT: Start commit request from TC. This signal is originally sent as a
+ * packed signal and this function is called from execPACKED_SIGNAL.
+ * This is the normal commit protocol where TC first send this signal to the
+ * backup node which then will send COMMIT to the primary node. If
+ * everything is ok the primary node send COMMITTED back to TC.
+ * ************************************************************************>> */
+void Dblqh::execCOMMIT(Signal* signal)
+{
+ TcConnectionrec *regTcConnectionrec = tcConnectionrec;
+ Uint32 ttcConnectrecFileSize = ctcConnectrecFileSize;
+ Uint32 tcIndex = signal->theData[0];
+ Uint32 gci = signal->theData[1];
+ Uint32 transid1 = signal->theData[2];
+ Uint32 transid2 = signal->theData[3];
+ jamEntry();
+ if (tcIndex >= ttcConnectrecFileSize) {
+ errorReport(signal, 0);
+ return;
+ }//if
+ if (ERROR_INSERTED(5011)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_COMMIT, signal, 2000, 4);
+ return;
+ }//if
+ if (ERROR_INSERTED(5012)) {
+ SET_ERROR_INSERT_VALUE(5017);
+ sendSignalWithDelay(cownref, GSN_COMMIT, signal, 2000, 4);
+ return;
+ }//if
+ tcConnectptr.i = tcIndex;
+ ptrAss(tcConnectptr, regTcConnectionrec);
+ if ((tcConnectptr.p->transid[0] == transid1) &&
+ (tcConnectptr.p->transid[1] == transid2)) {
+ commitReqLab(signal, gci);
+ return;
+ }//if
+ warningReport(signal, 1);
+ return;
+}//Dblqh::execCOMMIT()
+
+/* ************************************************************************>>
+ * COMMITREQ: Commit request from TC. This is the commit protocol used if
+ * one of the nodes is not behaving correctly. TC explicitly sends COMMITREQ
+ * to both the backup and primary node and gets a COMMITCONF back if the
+ * COMMIT was ok.
+ * ************************************************************************>> */
+void Dblqh::execCOMMITREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 reqPtr = signal->theData[0];
+ BlockReference reqBlockref = signal->theData[1];
+ Uint32 gci = signal->theData[2];
+ Uint32 transid1 = signal->theData[3];
+ Uint32 transid2 = signal->theData[4];
+ Uint32 tcOprec = signal->theData[6];
+ if (ERROR_INSERTED(5004)) {
+ systemErrorLab(signal);
+ }
+ if (ERROR_INSERTED(5017)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_COMMITREQ, signal, 2000, 7);
+ return;
+ }//if
+ if (findTransaction(transid1,
+ transid2,
+ tcOprec) != ZOK) {
+ warningReport(signal, 5);
+ return;
+ }//if
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ switch (regTcPtr->transactionState) {
+ case TcConnectionrec::PREPARED:
+ case TcConnectionrec::LOG_COMMIT_QUEUED_WAIT_SIGNAL:
+ case TcConnectionrec::LOG_COMMIT_WRITTEN_WAIT_SIGNAL:
+ jam();
+/*-------------------------------------------------------*/
+/* THE NORMAL CASE. */
+/*-------------------------------------------------------*/
+ regTcPtr->reqBlockref = reqBlockref;
+ regTcPtr->reqRef = reqPtr;
+ regTcPtr->abortState = TcConnectionrec::REQ_FROM_TC;
+ commitReqLab(signal, gci);
+ return;
+ break;
+ case TcConnectionrec::COMMITTED:
+ jam();
+/*---------------------------------------------------------*/
+/* FOR SOME REASON THE COMMIT PHASE HAVE BEEN */
+/* FINISHED AFTER A TIME OUT. WE NEED ONLY SEND A */
+/* COMMITCONF SIGNAL. */
+/*---------------------------------------------------------*/
+ regTcPtr->reqBlockref = reqBlockref;
+ regTcPtr->reqRef = reqPtr;
+ regTcPtr->abortState = TcConnectionrec::REQ_FROM_TC;
+ signal->theData[0] = regTcPtr->reqRef;
+ signal->theData[1] = cownNodeid;
+ signal->theData[2] = regTcPtr->transid[0];
+ signal->theData[3] = regTcPtr->transid[1];
+ sendSignal(regTcPtr->reqBlockref, GSN_COMMITCONF, signal, 4, JBB);
+ break;
+ case TcConnectionrec::COMMIT_STOPPED:
+ jam();
+ regTcPtr->reqBlockref = reqBlockref;
+ regTcPtr->reqRef = reqPtr;
+ regTcPtr->abortState = TcConnectionrec::REQ_FROM_TC;
+ /*empty*/;
+ break;
+ default:
+ jam();
+ warningReport(signal, 4);
+ return;
+ break;
+ }//switch
+ return;
+}//Dblqh::execCOMMITREQ()
+
+/* ************************************************************************>>
+ * COMPLETE : Complete the transaction. Sent as a packed signal from TC.
+ * Works the same way as COMMIT protocol. This is the normal case with both
+ * primary and backup working (See COMMIT).
+ * ************************************************************************>> */
+void Dblqh::execCOMPLETE(Signal* signal)
+{
+ TcConnectionrec *regTcConnectionrec = tcConnectionrec;
+ Uint32 ttcConnectrecFileSize = ctcConnectrecFileSize;
+ Uint32 tcIndex = signal->theData[0];
+ Uint32 transid1 = signal->theData[1];
+ Uint32 transid2 = signal->theData[2];
+ jamEntry();
+ if (tcIndex >= ttcConnectrecFileSize) {
+ errorReport(signal, 1);
+ return;
+ }//if
+ if (ERROR_INSERTED(5013)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_COMPLETE, signal, 2000, 3);
+ return;
+ }//if
+ if (ERROR_INSERTED(5014)) {
+ SET_ERROR_INSERT_VALUE(5018);
+ sendSignalWithDelay(cownref, GSN_COMPLETE, signal, 2000, 3);
+ return;
+ }//if
+ tcConnectptr.i = tcIndex;
+ ptrAss(tcConnectptr, regTcConnectionrec);
+ if ((tcConnectptr.p->transactionState == TcConnectionrec::COMMITTED) &&
+ (tcConnectptr.p->transid[0] == transid1) &&
+ (tcConnectptr.p->transid[1] == transid2)) {
+ if (tcConnectptr.p->seqNoReplica != 0) {
+ jam();
+ localCommitLab(signal);
+ return;
+ } else {
+ jam();
+ completeTransLastLab(signal);
+ return;
+ }//if
+ }//if
+ if (tcConnectptr.p->transactionState != TcConnectionrec::COMMITTED) {
+ warningReport(signal, 2);
+ } else {
+ warningReport(signal, 3);
+ }//if
+}//Dblqh::execCOMPLETE()
+
+/* ************************************************************************>>
+ * COMPLETEREQ: Complete request from TC. Same as COMPLETE but used if one
+ * node is not working ok (See COMMIT).
+ * ************************************************************************>> */
+void Dblqh::execCOMPLETEREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 reqPtr = signal->theData[0];
+ BlockReference reqBlockref = signal->theData[1];
+ Uint32 transid1 = signal->theData[2];
+ Uint32 transid2 = signal->theData[3];
+ Uint32 tcOprec = signal->theData[5];
+ if (ERROR_INSERTED(5005)) {
+ systemErrorLab(signal);
+ }
+ if (ERROR_INSERTED(5018)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_COMPLETEREQ, signal, 2000, 6);
+ return;
+ }//if
+ if (findTransaction(transid1,
+ transid2,
+ tcOprec) != ZOK) {
+ jam();
+/*---------------------------------------------------------*/
+/* FOR SOME REASON THE COMPLETE PHASE STARTED AFTER */
+/* A TIME OUT. THE TRANSACTION IS GONE. WE NEED TO */
+/* REPORT COMPLETION ANYWAY. */
+/*---------------------------------------------------------*/
+ signal->theData[0] = reqPtr;
+ signal->theData[1] = cownNodeid;
+ signal->theData[2] = transid1;
+ signal->theData[3] = transid2;
+ sendSignal(reqBlockref, GSN_COMPLETECONF, signal, 4, JBB);
+ warningReport(signal, 7);
+ return;
+ }//if
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ switch (regTcPtr->transactionState) {
+ case TcConnectionrec::COMMITTED:
+ jam();
+ regTcPtr->reqBlockref = reqBlockref;
+ regTcPtr->reqRef = reqPtr;
+ regTcPtr->abortState = TcConnectionrec::REQ_FROM_TC;
+ /*empty*/;
+ break;
+/*---------------------------------------------------------*/
+/* THE NORMAL CASE. */
+/*---------------------------------------------------------*/
+ case TcConnectionrec::COMMIT_STOPPED:
+ jam();
+/*---------------------------------------------------------*/
+/* FOR SOME REASON THE COMPLETE PHASE STARTED AFTER */
+/* A TIME OUT. WE HAVE SET THE PROPER VARIABLES SUCH */
+/* THAT A COMPLETECONF WILL BE SENT WHEN COMPLETE IS */
+/* FINISHED. */
+/*---------------------------------------------------------*/
+ regTcPtr->reqBlockref = reqBlockref;
+ regTcPtr->reqRef = reqPtr;
+ regTcPtr->abortState = TcConnectionrec::REQ_FROM_TC;
+ return;
+ break;
+ default:
+ jam();
+ warningReport(signal, 6);
+ return;
+ break;
+ }//switch
+ if (regTcPtr->seqNoReplica != 0) {
+ jam();
+ localCommitLab(signal);
+ return;
+ } else {
+ jam();
+ completeTransLastLab(signal);
+ return;
+ }//if
+}//Dblqh::execCOMPLETEREQ()
+
+/* ************> */
+/* COMPLETED > */
+/* ************> */
+void Dblqh::execLQHKEYCONF(Signal* signal)
+{
+ LqhKeyConf * const lqhKeyConf = (LqhKeyConf *)signal->getDataPtr();
+ Uint32 tcIndex = lqhKeyConf->opPtr;
+ Uint32 ttcConnectrecFileSize = ctcConnectrecFileSize;
+ TcConnectionrec *regTcConnectionrec = tcConnectionrec;
+ jamEntry();
+ if (tcIndex >= ttcConnectrecFileSize) {
+ errorReport(signal, 2);
+ return;
+ }//if
+ tcConnectptr.i = tcIndex;
+ ptrAss(tcConnectptr, regTcConnectionrec);
+ switch (tcConnectptr.p->connectState) {
+ case TcConnectionrec::LOG_CONNECTED:
+ jam();
+ completedLab(signal);
+ return;
+ break;
+ case TcConnectionrec::COPY_CONNECTED:
+ jam();
+ copyCompletedLab(signal);
+ return;
+ break;
+ default:
+ jam();
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dblqh::execLQHKEYCONF()
+
+/* ------------------------------------------------------------------------- */
+/* ------- COMMIT PHASE ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::commitReqLab(Signal* signal, Uint32 gci)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ TcConnectionrec::LogWriteState logWriteState = regTcPtr->logWriteState;
+ TcConnectionrec::TransactionState transState = regTcPtr->transactionState;
+ regTcPtr->gci = gci;
+ if (transState == TcConnectionrec::PREPARED) {
+ if (logWriteState == TcConnectionrec::WRITTEN) {
+ jam();
+ regTcPtr->transactionState = TcConnectionrec::PREPARED_RECEIVED_COMMIT;
+ TcConnectionrecPtr saveTcPtr = tcConnectptr;
+ Uint32 blockNo = refToBlock(regTcPtr->tcTupBlockref);
+ signal->theData[0] = regTcPtr->tupConnectrec;
+ signal->theData[1] = gci;
+ EXECUTE_DIRECT(blockNo, GSN_TUP_WRITELOG_REQ, signal, 2);
+ jamEntry();
+ if (regTcPtr->transactionState == TcConnectionrec::LOG_COMMIT_QUEUED) {
+ jam();
+ return;
+ }//if
+ ndbrequire(regTcPtr->transactionState == TcConnectionrec::LOG_COMMIT_WRITTEN);
+ tcConnectptr = saveTcPtr;
+ } else if (logWriteState == TcConnectionrec::NOT_STARTED) {
+ jam();
+ } else if (logWriteState == TcConnectionrec::NOT_WRITTEN) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* IT IS A READ OPERATION OR OTHER OPERATION THAT DO NOT USE THE LOG. */
+/*---------------------------------------------------------------------------*/
+/*---------------------------------------------------------------------------*/
+/* THE LOG HAS NOT BEEN WRITTEN SINCE THE LOG FLAG WAS FALSE. THIS CAN OCCUR */
+/* WHEN WE ARE STARTING A NEW FRAGMENT. */
+/*---------------------------------------------------------------------------*/
+ regTcPtr->logWriteState = TcConnectionrec::NOT_STARTED;
+ } else {
+ ndbrequire(logWriteState == TcConnectionrec::NOT_WRITTEN_WAIT);
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THE STATE WAS SET TO NOT_WRITTEN BY THE OPERATION BUT LATER A SCAN OF ALL */
+/* OPERATION RECORD CHANGED IT INTO NOT_WRITTEN_WAIT. THIS INDICATES THAT WE */
+/* ARE WAITING FOR THIS OPERATION TO COMMIT OR ABORT SO THAT WE CAN FIND THE */
+/* STARTING GLOBAL CHECKPOINT OF THIS NEW FRAGMENT. */
+/*---------------------------------------------------------------------------*/
+ checkScanTcCompleted(signal);
+ }//if
+ } else if (transState == TcConnectionrec::LOG_COMMIT_QUEUED_WAIT_SIGNAL) {
+ jam();
+ regTcPtr->transactionState = TcConnectionrec::LOG_COMMIT_QUEUED;
+ return;
+ } else if (transState == TcConnectionrec::LOG_COMMIT_WRITTEN_WAIT_SIGNAL) {
+ jam();
+ } else {
+ warningReport(signal, 0);
+ return;
+ }//if
+ if (regTcPtr->seqNoReplica != 0) {
+ jam();
+ commitReplyLab(signal);
+ return;
+ }//if
+ localCommitLab(signal);
+ return;
+}//Dblqh::commitReqLab()
+
+void Dblqh::execLQH_WRITELOG_REQ(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ Uint32 gci = signal->theData[1];
+ Uint32 newestGci = cnewestGci;
+ TcConnectionrec::LogWriteState logWriteState = regTcPtr->logWriteState;
+ TcConnectionrec::TransactionState transState = regTcPtr->transactionState;
+ regTcPtr->gci = gci;
+ if (gci > newestGci) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* KEEP TRACK OF NEWEST GLOBAL CHECKPOINT THAT LQH HAS HEARD OF. */
+/* ------------------------------------------------------------------------- */
+ cnewestGci = gci;
+ }//if
+ if (logWriteState == TcConnectionrec::WRITTEN) {
+/*---------------------------------------------------------------------------*/
+/* I NEED TO INSERT A COMMIT LOG RECORD SINCE WE ARE WRITING LOG IN THIS */
+/* TRANSACTION. */
+/*---------------------------------------------------------------------------*/
+ jam();
+ LogPartRecordPtr regLogPartPtr;
+ Uint32 noOfLogPages = cnoOfLogPages;
+ jam();
+ regLogPartPtr.i = regTcPtr->hashValue & 3;
+ ptrCheckGuard(regLogPartPtr, clogPartFileSize, logPartRecord);
+ if ((regLogPartPtr.p->logPartState == LogPartRecord::ACTIVE) ||
+ (noOfLogPages == 0)) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THIS LOG PART WAS CURRENTLY ACTIVE WRITING ANOTHER LOG RECORD. WE MUST */
+/* WAIT UNTIL THIS PART HAS COMPLETED ITS OPERATION. */
+/*---------------------------------------------------------------------------*/
+// We must delay the write of commit info to the log to safe-guard against
+// a crash due to lack of log pages. We temporary stop all log writes to this
+// log part to ensure that we don't get a buffer explosion in the delayed
+// signal buffer instead.
+/*---------------------------------------------------------------------------*/
+ linkWaitLog(signal, regLogPartPtr);
+ if (transState == TcConnectionrec::PREPARED) {
+ jam();
+ regTcPtr->transactionState = TcConnectionrec::LOG_COMMIT_QUEUED_WAIT_SIGNAL;
+ } else {
+ jam();
+ ndbrequire(transState == TcConnectionrec::PREPARED_RECEIVED_COMMIT);
+ regTcPtr->transactionState = TcConnectionrec::LOG_COMMIT_QUEUED;
+ }//if
+ if (regLogPartPtr.p->logPartState == LogPartRecord::IDLE) {
+ jam();
+ regLogPartPtr.p->logPartState = LogPartRecord::ACTIVE;
+ }//if
+ return;
+ }//if
+ writeCommitLog(signal, regLogPartPtr);
+ if (transState == TcConnectionrec::PREPARED) {
+ jam();
+ regTcPtr->transactionState = TcConnectionrec::LOG_COMMIT_WRITTEN_WAIT_SIGNAL;
+ } else {
+ jam();
+ ndbrequire(transState == TcConnectionrec::PREPARED_RECEIVED_COMMIT);
+ regTcPtr->transactionState = TcConnectionrec::LOG_COMMIT_WRITTEN;
+ }//if
+ }//if
+}//Dblqh::execLQH_WRITELOG_REQ()
+
+void Dblqh::localCommitLab(Signal* signal)
+{
+ FragrecordPtr regFragptr;
+ regFragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(regFragptr, cfragrecFileSize, fragrecord);
+ Fragrecord::FragStatus status = regFragptr.p->fragStatus;
+ fragptr = regFragptr;
+ switch (status) {
+ case Fragrecord::FSACTIVE:
+ case Fragrecord::CRASH_RECOVERING:
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ commitContinueAfterBlockedLab(signal);
+ return;
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::COMMIT_STOPPED;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+}//Dblqh::localCommitLab()
+
+void Dblqh::commitContinueAfterBlockedLab(Signal* signal)
+{
+/* ------------------------------------------------------------------------- */
+/*INPUT: TC_CONNECTPTR ACTIVE OPERATION RECORD */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/*CONTINUE HERE AFTER BEING BLOCKED FOR A WHILE DURING LOCAL CHECKPOINT. */
+/*The operation is already removed from the active list since there is no */
+/*chance for any real-time breaks before we need to release it. */
+/* ------------------------------------------------------------------------- */
+/*ALSO AFTER NORMAL PROCEDURE WE CONTINUE */
+/*WE MUST COMMIT TUP BEFORE ACC TO ENSURE THAT NO ONE RACES IN AND SEES A */
+/*DIRTY STATE IN TUP. */
+/* ------------------------------------------------------------------------- */
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ Fragrecord * const regFragptr = fragptr.p;
+ Uint32 operation = regTcPtr->operation;
+ if (regTcPtr->activeCreat == ZFALSE) {
+ if ((cCommitBlocked == true) &&
+ (regFragptr->fragActiveStatus == ZTRUE)) {
+ jam();
+/* ------------------------------------------------------------------------- */
+// TUP and/or ACC have problems in writing the undo log to disk fast enough.
+// We must avoid the commit at this time and try later instead. The fragment
+// is also active with a local checkpoint and this commit can generate UNDO
+// log records that overflow the UNDO log buffer.
+/* ------------------------------------------------------------------------- */
+/*---------------------------------------------------------------------------*/
+// We must delay the write of commit info to the log to safe-guard against
+// a crash due to lack of log pages. We temporary stop all log writes to this
+// log part to ensure that we don't get a buffer explosion in the delayed
+// signal buffer instead.
+/*---------------------------------------------------------------------------*/
+ logPartPtr.i = regTcPtr->hashValue & 3;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ linkWaitLog(signal, logPartPtr);
+ regTcPtr->transactionState = TcConnectionrec::COMMIT_QUEUED;
+ if (logPartPtr.p->logPartState == LogPartRecord::IDLE) {
+ jam();
+ logPartPtr.p->logPartState = LogPartRecord::ACTIVE;
+ }//if
+ return;
+ }//if
+ if (operation != ZREAD) {
+ TupCommitReq * const tupCommitReq =
+ (TupCommitReq *)signal->getDataPtrSend();
+ Uint32 sig0 = regTcPtr->tupConnectrec;
+ Uint32 tup = refToBlock(regTcPtr->tcTupBlockref);
+ jam();
+ tupCommitReq->opPtr = sig0;
+ tupCommitReq->gci = regTcPtr->gci;
+ tupCommitReq->hashValue = regTcPtr->hashValue;
+ EXECUTE_DIRECT(tup, GSN_TUP_COMMITREQ, signal,
+ TupCommitReq::SignalLength);
+ }//if
+ Uint32 acc = refToBlock(regTcPtr->tcAccBlockref);
+ signal->theData[0] = regTcPtr->accConnectrec;
+ EXECUTE_DIRECT(acc, GSN_ACC_COMMITREQ, signal, 1);
+ Uint32 simpleRead = regTcPtr->simpleRead;
+ jamEntry();
+ if (simpleRead == ZSIMPLE_READ) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*THE OPERATION WAS A SIMPLE READ THUS THE COMMIT PHASE IS ONLY NEEDED TO */
+/*RELEASE THE LOCKS. AT THIS POINT IN THE CODE THE LOCKS ARE RELEASED AND WE */
+/*ARE IN A POSITION TO SEND LQHKEYCONF TO TC. WE WILL ALSO RELEASE ALL */
+/*RESOURCES BELONGING TO THIS OPERATION SINCE NO MORE WORK WILL BE */
+/*PERFORMED. */
+/* ------------------------------------------------------------------------- */
+ cleanUp(signal);
+ return;
+ }//if
+ }//if
+ Uint32 dirtyOp = regTcPtr->dirtyOp;
+ Uint32 seqNoReplica = regTcPtr->seqNoReplica;
+ if (regTcPtr->gci > regFragptr->newestGci) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*IT IS THE FIRST TIME THIS GLOBAL CHECKPOINT IS INVOLVED IN UPDATING THIS */
+/*FRAGMENT. UPDATE THE VARIABLE THAT KEEPS TRACK OF NEWEST GCI IN FRAGMENT */
+/* ------------------------------------------------------------------------- */
+ regFragptr->newestGci = regTcPtr->gci;
+ }//if
+ if (dirtyOp != ZTRUE) {
+ if (seqNoReplica != 0) {
+ jam();
+ completeTransNotLastLab(signal);
+ return;
+ }//if
+ commitReplyLab(signal);
+ return;
+ } else {
+/* ------------------------------------------------------------------------- */
+/*WE MUST HANDLE DIRTY WRITES IN A SPECIAL WAY. THESE OPERATIONS WILL NOT */
+/*SEND ANY COMMIT OR COMPLETE MESSAGES TO OTHER NODES. THEY WILL MERELY SEND */
+/*THOSE SIGNALS INTERNALLY. */
+/* ------------------------------------------------------------------------- */
+ if (regTcPtr->abortState == TcConnectionrec::ABORT_IDLE) {
+ jam();
+ packLqhkeyreqLab(signal);
+ } else {
+ ndbrequire(regTcPtr->abortState != TcConnectionrec::NEW_FROM_TC);
+ jam();
+ sendLqhTransconf(signal, LqhTransConf::Committed);
+ cleanUp(signal);
+ }//if
+ }//if
+}//Dblqh::commitContinueAfterBlockedLab()
+
+void Dblqh::commitReplyLab(Signal* signal)
+{
+/* -------------------------------------------------------------- */
+/* BACKUP AND STAND-BY REPLICAS ONLY UPDATE THE TRANSACTION STATE */
+/* -------------------------------------------------------------- */
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ TcConnectionrec::AbortState abortState = regTcPtr->abortState;
+ regTcPtr->transactionState = TcConnectionrec::COMMITTED;
+ if (abortState == TcConnectionrec::ABORT_IDLE) {
+ Uint32 clientBlockref = regTcPtr->clientBlockref;
+ if (regTcPtr->seqNoReplica == 0) {
+ jam();
+ sendCommittedTc(signal, clientBlockref);
+ return;
+ } else {
+ jam();
+ sendCommitLqh(signal, clientBlockref);
+ return;
+ }//if
+ } else if (regTcPtr->abortState == TcConnectionrec::REQ_FROM_TC) {
+ jam();
+ signal->theData[0] = regTcPtr->reqRef;
+ signal->theData[1] = cownNodeid;
+ signal->theData[2] = regTcPtr->transid[0];
+ signal->theData[3] = regTcPtr->transid[1];
+ sendSignal(tcConnectptr.p->reqBlockref, GSN_COMMITCONF, signal, 4, JBB);
+ } else {
+ ndbrequire(regTcPtr->abortState == TcConnectionrec::NEW_FROM_TC);
+ jam();
+ sendLqhTransconf(signal, LqhTransConf::Committed);
+ }//if
+ return;
+}//Dblqh::commitReplyLab()
+
+/* ------------------------------------------------------------------------- */
+/* ------- COMPLETE PHASE ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::completeTransNotLastLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->abortState == TcConnectionrec::ABORT_IDLE) {
+ Uint32 clientBlockref = regTcPtr->clientBlockref;
+ jam();
+ sendCompleteLqh(signal, clientBlockref);
+ cleanUp(signal);
+ return;
+ } else {
+ jam();
+ completeUnusualLab(signal);
+ return;
+ }//if
+}//Dblqh::completeTransNotLastLab()
+
+void Dblqh::completeTransLastLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->abortState == TcConnectionrec::ABORT_IDLE) {
+ Uint32 clientBlockref = regTcPtr->clientBlockref;
+ jam();
+/* ------------------------------------------------------------------------- */
+/*DIRTY WRITES WHICH ARE LAST IN THE CHAIN OF REPLICAS WILL SEND COMPLETED */
+/*INSTEAD OF SENDING PREPARED TO THE TC (OR OTHER INITIATOR OF OPERATION). */
+/* ------------------------------------------------------------------------- */
+ sendCompletedTc(signal, clientBlockref);
+ cleanUp(signal);
+ return;
+ } else {
+ jam();
+ completeUnusualLab(signal);
+ return;
+ }//if
+}//Dblqh::completeTransLastLab()
+
+void Dblqh::completeUnusualLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->abortState == TcConnectionrec::ABORT_FROM_TC) {
+ jam();
+ sendAborted(signal);
+ } else if (regTcPtr->abortState == TcConnectionrec::NEW_FROM_TC) {
+ jam();
+ sendLqhTransconf(signal, LqhTransConf::Committed);
+ } else {
+ ndbrequire(regTcPtr->abortState == TcConnectionrec::REQ_FROM_TC);
+ jam();
+ signal->theData[0] = regTcPtr->reqRef;
+ signal->theData[1] = cownNodeid;
+ signal->theData[2] = regTcPtr->transid[0];
+ signal->theData[3] = regTcPtr->transid[1];
+ sendSignal(regTcPtr->reqBlockref,
+ GSN_COMPLETECONF, signal, 4, JBB);
+ }//if
+ cleanUp(signal);
+ return;
+}//Dblqh::completeUnusualLab()
+
+/* ========================================================================= */
+/* ======= RELEASE TC CONNECT RECORD ======= */
+/* */
+/* RELEASE A TC CONNECT RECORD TO THE FREELIST. */
+/* ========================================================================= */
+void Dblqh::releaseTcrec(Signal* signal, TcConnectionrecPtr locTcConnectptr)
+{
+ jam();
+ locTcConnectptr.p->tcTimer = 0;
+ locTcConnectptr.p->transactionState = TcConnectionrec::TC_NOT_CONNECTED;
+ locTcConnectptr.p->nextTcConnectrec = cfirstfreeTcConrec;
+ cfirstfreeTcConrec = locTcConnectptr.i;
+
+ TablerecPtr tabPtr;
+ tabPtr.i = locTcConnectptr.p->tableref;
+ if(tabPtr.i == RNIL)
+ return;
+
+ ptrCheckGuard(tabPtr, ctabrecFileSize, tablerec);
+
+ /**
+ * Normal case
+ */
+ ndbrequire(tabPtr.p->usageCount > 0);
+ tabPtr.p->usageCount--;
+}//Dblqh::releaseTcrec()
+
+void Dblqh::releaseTcrecLog(Signal* signal, TcConnectionrecPtr locTcConnectptr)
+{
+ jam();
+ locTcConnectptr.p->tcTimer = 0;
+ locTcConnectptr.p->transactionState = TcConnectionrec::TC_NOT_CONNECTED;
+ locTcConnectptr.p->nextTcConnectrec = cfirstfreeTcConrec;
+ cfirstfreeTcConrec = locTcConnectptr.i;
+
+ TablerecPtr tabPtr;
+ tabPtr.i = locTcConnectptr.p->tableref;
+ if(tabPtr.i == RNIL)
+ return;
+
+}//Dblqh::releaseTcrecLog()
+
+/* ------------------------------------------------------------------------- */
+/* ------- ABORT PHASE ------- */
+/* */
+/*THIS PART IS USED AT ERRORS THAT CAUSE ABORT OF TRANSACTION. */
+/* ------------------------------------------------------------------------- */
+/* ***************************************************>> */
+/* ABORT: Abort transaction in connection. Sender TC. */
+/* This is the normal protocol (See COMMIT) */
+/* ***************************************************>> */
+void Dblqh::execABORT(Signal* signal)
+{
+ jamEntry();
+ Uint32 tcOprec = signal->theData[0];
+ BlockReference tcBlockref = signal->theData[1];
+ Uint32 transid1 = signal->theData[2];
+ Uint32 transid2 = signal->theData[3];
+ if (ERROR_INSERTED(5003)) {
+ systemErrorLab(signal);
+ }
+ if (ERROR_INSERTED(5015)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_ABORT, signal, 2000, 4);
+ return;
+ }//if
+ if (findTransaction(transid1,
+ transid2,
+ tcOprec) != ZOK) {
+ jam();
+/* ------------------------------------------------------------------------- */
+// SEND ABORTED EVEN IF NOT FOUND.
+//THE TRANSACTION MIGHT NEVER HAVE ARRIVED HERE.
+/* ------------------------------------------------------------------------- */
+ signal->theData[0] = tcOprec;
+ signal->theData[1] = transid1;
+ signal->theData[2] = transid2;
+ signal->theData[3] = cownNodeid;
+ signal->theData[4] = ZTRUE;
+ sendSignal(tcBlockref, GSN_ABORTED, signal, 5, JBB);
+ warningReport(signal, 8);
+ return;
+ }//if
+/* ------------------------------------------------------------------------- */
+/*A GUIDING DESIGN PRINCIPLE IN HANDLING THESE ERROR SITUATIONS HAVE BEEN */
+/*KEEP IT SIMPLE. THUS WE RATHER INSERT A WAIT AND SET THE ABORT_STATE TO */
+/*ACTIVE RATHER THAN WRITE NEW CODE TO HANDLE EVERY SPECIAL SITUATION. */
+/* ------------------------------------------------------------------------- */
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->nextReplica != ZNIL) {
+/* ------------------------------------------------------------------------- */
+// We will immediately send the ABORT message also to the next LQH node in line.
+/* ------------------------------------------------------------------------- */
+ BlockReference TLqhRef = calcLqhBlockRef(regTcPtr->nextReplica);
+ signal->theData[0] = regTcPtr->tcOprec;
+ signal->theData[1] = regTcPtr->tcBlockref;
+ signal->theData[2] = regTcPtr->transid[0];
+ signal->theData[3] = regTcPtr->transid[1];
+ sendSignal(TLqhRef, GSN_ABORT, signal, 4, JBB);
+ }//if
+ regTcPtr->abortState = TcConnectionrec::ABORT_FROM_TC;
+ regTcPtr->activeCreat = ZFALSE;
+ abortStateHandlerLab(signal);
+ return;
+}//Dblqh::execABORT()
+
+/* ************************************************************************>>
+ * ABORTREQ: Same as ABORT but used in case one node isn't working ok.
+ * (See COMMITREQ)
+ * ************************************************************************>> */
+void Dblqh::execABORTREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 reqPtr = signal->theData[0];
+ BlockReference reqBlockref = signal->theData[1];
+ Uint32 transid1 = signal->theData[2];
+ Uint32 transid2 = signal->theData[3];
+ Uint32 tcOprec = signal->theData[5];
+ if (ERROR_INSERTED(5006)) {
+ systemErrorLab(signal);
+ }
+ if (ERROR_INSERTED(5016)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_ABORTREQ, signal, 2000, 6);
+ return;
+ }//if
+ if (findTransaction(transid1,
+ transid2,
+ tcOprec) != ZOK) {
+ signal->theData[0] = reqPtr;
+ signal->theData[2] = cownNodeid;
+ signal->theData[3] = transid1;
+ signal->theData[4] = transid2;
+ sendSignal(reqBlockref, GSN_ABORTCONF, signal, 5, JBB);
+ warningReport(signal, 9);
+ return;
+ }//if
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->transactionState != TcConnectionrec::PREPARED) {
+ warningReport(signal, 10);
+ return;
+ }//if
+ regTcPtr->reqBlockref = reqBlockref;
+ regTcPtr->reqRef = reqPtr;
+ regTcPtr->abortState = TcConnectionrec::REQ_FROM_TC;
+ regTcPtr->activeCreat = ZFALSE;
+ abortCommonLab(signal);
+ return;
+}//Dblqh::execABORTREQ()
+
+/* ************>> */
+/* ACC_TO_REF > */
+/* ************>> */
+void Dblqh::execACC_TO_REF(Signal* signal)
+{
+ jamEntry();
+ terrorCode = signal->theData[1];
+ releaseActiveFrag(signal);
+ abortErrorLab(signal);
+ return;
+}//Dblqh::execACC_TO_REF()
+
+/* ************> */
+/* ACCKEYREF > */
+/* ************> */
+void Dblqh::execACCKEYREF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ terrorCode = signal->theData[1];
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ TcConnectionrec * const tcPtr = tcConnectptr.p;
+ switch (tcPtr->transactionState) {
+ case TcConnectionrec::WAIT_ACC:
+ jam();
+ releaseActiveFrag(signal);
+ break;
+ case TcConnectionrec::WAIT_ACC_ABORT:
+ case TcConnectionrec::ABORT_STOPPED:
+ case TcConnectionrec::ABORT_QUEUED:
+ jam();
+/* ------------------------------------------------------------------------- */
+/*IGNORE SINCE ABORT OF THIS OPERATION IS ONGOING ALREADY. */
+/* ------------------------------------------------------------------------- */
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ const Uint32 errCode = terrorCode;
+ tcPtr->errorCode = errCode;
+/* ------------------------------------------------------------------------- */
+/*WHEN AN ABORT FROM TC ARRIVES IT COULD ACTUALLY BE A CORRECT BEHAVIOUR */
+/*SINCE THE TUPLE MIGHT NOT HAVE ARRIVED YET OR ALREADY HAVE BEEN INSERTED. */
+/* ------------------------------------------------------------------------- */
+ if (tcPtr->activeCreat == ZTRUE) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*THIS IS A NORMAL EVENT DURING CREATION OF A FRAGMENT. PERFORM ABORT IN */
+/*TUP AND ACC AND THEN CONTINUE WITH NORMAL COMMIT PROCESSING. IF THE ERROR */
+/*HAPPENS TO BE A SERIOUS ERROR THEN PERFORM ABORT PROCESSING AS NORMAL. */
+/* ------------------------------------------------------------------------- */
+ switch (tcPtr->operation) {
+ case ZUPDATE:
+ case ZDELETE:
+ jam();
+ if (errCode != ZNO_TUPLE_FOUND) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*A NORMAL ERROR WILL BE TREATED AS A NORMAL ABORT AND WILL ABORT THE */
+/*TRANSACTION. NO SPECIAL HANDLING IS NEEDED. */
+/* ------------------------------------------------------------------------- */
+ tcPtr->activeCreat = ZFALSE;
+ }//if
+ break;
+ case ZINSERT:
+ jam();
+ if (errCode != ZTUPLE_ALREADY_EXIST) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*A NORMAL ERROR WILL BE TREATED AS A NORMAL ABORT AND WILL ABORT THE */
+/*TRANSACTION. NO SPECIAL HANDLING IS NEEDED. */
+/* ------------------------------------------------------------------------- */
+ tcPtr->activeCreat = ZFALSE;
+ }//if
+ break;
+ default:
+ jam();
+/* ------------------------------------------------------------------------- */
+/*A NORMAL ERROR WILL BE TREATED AS A NORMAL ABORT AND WILL ABORT THE */
+/*TRANSACTION. NO SPECIAL HANDLING IS NEEDED. */
+/* ------------------------------------------------------------------------- */
+ tcPtr->activeCreat = ZFALSE;
+ break;
+ }//switch
+ } else {
+ /**
+ * Only primary replica can get ZTUPLE_ALREADY_EXIST || ZNO_TUPLE_FOUND
+ *
+ * Unless it's a simple or dirty read
+ */
+ ndbrequire
+ (tcPtr->seqNoReplica == 0 ||
+ (errCode != ZTUPLE_ALREADY_EXIST && errCode != ZNO_TUPLE_FOUND) ||
+ (tcPtr->operation == ZREAD && (tcPtr->dirtyOp || tcPtr->opSimple)));
+ }
+ tcPtr->abortState = TcConnectionrec::ABORT_FROM_LQH;
+ abortCommonLab(signal);
+ return;
+}//Dblqh::execACCKEYREF()
+
+void Dblqh::localAbortStateHandlerLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->abortState != TcConnectionrec::ABORT_IDLE) {
+ jam();
+ return;
+ }//if
+ regTcPtr->activeCreat = ZFALSE;
+ regTcPtr->abortState = TcConnectionrec::ABORT_FROM_LQH;
+ regTcPtr->errorCode = terrorCode;
+ abortStateHandlerLab(signal);
+ return;
+}//Dblqh::localAbortStateHandlerLab()
+
+void Dblqh::abortStateHandlerLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ switch (regTcPtr->transactionState) {
+ case TcConnectionrec::PREPARED:
+ jam();
+/* ------------------------------------------------------------------------- */
+/*THE OPERATION IS ALREADY PREPARED AND SENT TO THE NEXT LQH OR BACK TO TC. */
+/*WE CAN SIMPLY CONTINUE WITH THE ABORT PROCESS. */
+/*IF IT WAS A CHECK FOR TRANSACTION STATUS THEN WE REPORT THE STATUS TO THE */
+/*NEW TC AND CONTINUE WITH THE NEXT OPERATION IN LQH. */
+/* ------------------------------------------------------------------------- */
+ if (regTcPtr->abortState == TcConnectionrec::NEW_FROM_TC) {
+ jam();
+ sendLqhTransconf(signal, LqhTransConf::Prepared);
+ return;
+ }//if
+ break;
+ case TcConnectionrec::LOG_COMMIT_WRITTEN_WAIT_SIGNAL:
+ case TcConnectionrec::LOG_COMMIT_QUEUED_WAIT_SIGNAL:
+ jam();
+/* ------------------------------------------------------------------------- */
+// We can only reach these states for multi-updates on a record in a transaction.
+// We know that at least one of those has received the COMMIT signal, thus we
+// declare us only prepared since we then receive the expected COMMIT signal.
+/* ------------------------------------------------------------------------- */
+ ndbrequire(regTcPtr->abortState == TcConnectionrec::NEW_FROM_TC);
+ sendLqhTransconf(signal, LqhTransConf::Prepared);
+ break;
+ case TcConnectionrec::WAIT_TUPKEYINFO:
+ case TcConnectionrec::WAIT_ATTR:
+ jam();
+/* ------------------------------------------------------------------------- */
+/* WE ARE CURRENTLY WAITING FOR MORE INFORMATION. WE CAN START THE ABORT */
+/* PROCESS IMMEDIATELY. THE KEYINFO AND ATTRINFO SIGNALS WILL BE DROPPED */
+/* SINCE THE ABORT STATE WILL BE SET. */
+/* ------------------------------------------------------------------------- */
+ break;
+ case TcConnectionrec::WAIT_TUP:
+ jam();
+/* ------------------------------------------------------------------------- */
+// TUP is currently active. We have to wait for the TUPKEYREF or TUPKEYCONF
+// to arrive since we might otherwise jeopardise the local checkpoint
+// consistency in overload situations.
+/* ------------------------------------------------------------------------- */
+ regTcPtr->transactionState = TcConnectionrec::WAIT_TUP_TO_ABORT;
+ return;
+ case TcConnectionrec::WAIT_ACC:
+ jam();
+ if (regTcPtr->listState == TcConnectionrec::ACC_BLOCK_LIST) {
+ jam();
+/* ------------------------------------------------------------------------- */
+// If the operation is in the ACC Blocked list the operation is not allowed
+// to start yet. We release it from the ACC Blocked list and will go through
+// the gate in abortCommonLab(..) where it will be blocked.
+/* ------------------------------------------------------------------------- */
+ fragptr.i = regTcPtr->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ releaseAccList(signal);
+ } else {
+ jam();
+/* ------------------------------------------------------------------------- */
+// We start the abort immediately since the operation is still in the active
+// list and the fragment cannot have been frozen yet. By sending LCP_HOLDOPCONF
+// as direct signals we avoid the problem that we might find the operation
+// in an unexpected list in ACC.
+// We cannot accept being blocked before aborting ACC here since that would
+// lead to seriously complex issues.
+/* ------------------------------------------------------------------------- */
+ abortContinueAfterBlockedLab(signal, false);
+ return;
+ }//if
+ break;
+ case TcConnectionrec::LOG_QUEUED:
+ jam();
+/* ------------------------------------------------------------------------- */
+/*CURRENTLY QUEUED FOR LOGGING. WAIT UNTIL THE LOG RECORD HAVE BEEN INSERTED */
+/*AND THEN CONTINUE THE ABORT PROCESS. */
+//Could also be waiting for an overloaded log disk. In this case it is easy
+//to abort when CONTINUEB arrives.
+/* ------------------------------------------------------------------------- */
+ return;
+ break;
+ case TcConnectionrec::STOPPED:
+ jam();
+/* ------------------------------------------------------------------------- */
+/*WE ARE CURRENTLY QUEUED FOR ACCESS TO THE FRAGMENT BY A LOCAL CHECKPOINT. */
+/* ------------------------------------------------------------------------- */
+ releaseWaitQueue(signal);
+ break;
+ case TcConnectionrec::WAIT_AI_AFTER_ABORT:
+ jam();
+/* ------------------------------------------------------------------------- */
+/* ABORT OF ACC AND TUP ALREADY COMPLETED. THIS STATE IS ONLY USED WHEN */
+/* CREATING A NEW FRAGMENT. */
+/* ------------------------------------------------------------------------- */
+ continueAbortLab(signal);
+ return;
+ break;
+ case TcConnectionrec::WAIT_TUP_TO_ABORT:
+ case TcConnectionrec::ABORT_STOPPED:
+ case TcConnectionrec::LOG_ABORT_QUEUED:
+ case TcConnectionrec::WAIT_ACC_ABORT:
+ case TcConnectionrec::ABORT_QUEUED:
+ jam();
+/* ------------------------------------------------------------------------- */
+/*ABORT IS ALREADY ONGOING DUE TO SOME ERROR. WE HAVE ALREADY SET THE STATE */
+/*OF THE ABORT SO THAT WE KNOW THAT TC EXPECTS A REPORT. WE CAN THUS SIMPLY */
+/*EXIT. */
+/* ------------------------------------------------------------------------- */
+ return;
+ break;
+ case TcConnectionrec::COMMIT_STOPPED:
+ case TcConnectionrec::LOG_COMMIT_QUEUED:
+ case TcConnectionrec::COMMIT_QUEUED:
+ jam();
+/* ------------------------------------------------------------------------- */
+/*THIS IS ONLY AN ALLOWED STATE IF A DIRTY WRITE OR SIMPLE READ IS PERFORMED.*/
+/*IF WE ARE MERELY CHECKING THE TRANSACTION STATE IT IS ALSO AN ALLOWED STATE*/
+/* ------------------------------------------------------------------------- */
+ if (regTcPtr->dirtyOp == ZTRUE) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*COMPLETE THE DIRTY WRITE AND THEN REPORT COMPLETED BACK TO TC. SINCE IT IS */
+/*A DIRTY WRITE IT IS ALLOWED TO COMMIT EVEN IF THE TRANSACTION ABORTS. */
+/* ------------------------------------------------------------------------- */
+ return;
+ }//if
+ if (regTcPtr->simpleRead == ZSIMPLE_READ) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*A SIMPLE READ IS CURRENTLY RELEASING THE LOCKS OR WAITING FOR ACCESS TO */
+/*ACC TO CLEAR THE LOCKS. COMPLETE THIS PROCESS AND THEN RETURN AS NORMAL. */
+/*NO DATA HAS CHANGED DUE TO THIS SIMPLE READ ANYWAY. */
+/* ------------------------------------------------------------------------- */
+ return;
+ }//if
+ ndbrequire(regTcPtr->abortState == TcConnectionrec::NEW_FROM_TC);
+ jam();
+/* ------------------------------------------------------------------------- */
+/*WE ARE ONLY CHECKING THE STATUS OF THE TRANSACTION. IT IS COMMITTING. */
+/*COMPLETE THE COMMIT LOCALLY AND THEN SEND REPORT OF COMMITTED TO THE NEW TC*/
+/* ------------------------------------------------------------------------- */
+ return;
+ break;
+ case TcConnectionrec::COMMITTED:
+ jam();
+ ndbrequire(regTcPtr->abortState == TcConnectionrec::NEW_FROM_TC);
+/* ------------------------------------------------------------------------- */
+/*WE ARE CHECKING TRANSACTION STATUS. REPORT COMMITTED AND CONTINUE WITH THE */
+/*NEXT OPERATION. */
+/* ------------------------------------------------------------------------- */
+ sendLqhTransconf(signal, LqhTransConf::Committed);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+/* ------------------------------------------------------------------------- */
+/*THE STATE WAS NOT AN ALLOWED STATE ON A NORMAL OPERATION. SCANS AND COPY */
+/*FRAGMENT OPERATIONS SHOULD HAVE EXECUTED IN ANOTHER PATH. */
+/* ------------------------------------------------------------------------- */
+ break;
+ }//switch
+ abortCommonLab(signal);
+ return;
+}//Dblqh::abortStateHandlerLab()
+
+void Dblqh::abortErrorLab(Signal* signal)
+{
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->abortState == TcConnectionrec::ABORT_IDLE) {
+ jam();
+ regTcPtr->abortState = TcConnectionrec::ABORT_FROM_LQH;
+ regTcPtr->errorCode = terrorCode;
+ }//if
+ /* -----------------------------------------------------------------------
+ * ACTIVE CREATION IS RESET FOR ALL ERRORS WHICH SHOULD BE HANDLED
+ * WITH NORMAL ABORT HANDLING.
+ * ----------------------------------------------------------------------- */
+ regTcPtr->activeCreat = ZFALSE;
+ abortCommonLab(signal);
+ return;
+}//Dblqh::abortErrorLab()
+
+void Dblqh::abortCommonLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ const Uint32 commitAckMarker = regTcPtr->commitAckMarker;
+ if(regTcPtr->activeCreat != ZTRUE && commitAckMarker != RNIL){
+ /**
+ * There is no NR ongoing and we have a marker
+ */
+ jam();
+
+ m_commitAckMarkerHash.release(commitAckMarker);
+ regTcPtr->commitAckMarker = RNIL;
+ }
+
+ fragptr.i = regTcPtr->fragmentptr;
+ if (fragptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ case Fragrecord::CRASH_RECOVERING:
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ linkActiveFrag(signal);
+ abortContinueAfterBlockedLab(signal, true);
+ return;
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ regTcPtr->transactionState = TcConnectionrec::ABORT_STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ } else {
+ jam();
+ continueAbortLab(signal);
+ }//if
+}//Dblqh::abortCommonLab()
+
+void Dblqh::abortContinueAfterBlockedLab(Signal* signal, bool canBlock)
+{
+ /* ------------------------------------------------------------------------
+ * INPUT: TC_CONNECTPTR ACTIVE OPERATION RECORD
+ * ------------------------------------------------------------------------
+ * ------------------------------------------------------------------------
+ * CAN COME HERE AS RESTART AFTER BEING BLOCKED BY A LOCAL CHECKPOINT.
+ * ------------------------------------------------------------------------
+ * ALSO AS PART OF A NORMAL ABORT WITHOUT BLOCKING.
+ * WE MUST ABORT TUP BEFORE ACC TO ENSURE THAT NO ONE RACES IN
+ * AND SEES A STATE IN TUP.
+ * ------------------------------------------------------------------------ */
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ fragptr.i = regTcPtr->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if ((cCommitBlocked == true) &&
+ (fragptr.p->fragActiveStatus == ZTRUE) &&
+ (canBlock == true) &&
+ (regTcPtr->operation != ZREAD)) {
+ jam();
+/* ------------------------------------------------------------------------- */
+// TUP and/or ACC have problems in writing the undo log to disk fast enough.
+// We must avoid the abort at this time and try later instead. The fragment
+// is also active with a local checkpoint and this commit can generate UNDO
+// log records that overflow the UNDO log buffer.
+//
+// In certain situations it is simply too complex to insert a wait state here
+// since ACC is active and we cannot release the operation from the active
+// list without causing great complexity.
+/* ------------------------------------------------------------------------- */
+/*---------------------------------------------------------------------------*/
+// We must delay the write of abort info to the log to safe-guard against
+// a crash due to lack of log pages. We temporary stop all log writes to this
+// log part to ensure that we don't get a buffer explosion in the delayed
+// signal buffer instead.
+/*---------------------------------------------------------------------------*/
+ releaseActiveFrag(signal);
+ logPartPtr.i = regTcPtr->hashValue & 3;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ linkWaitLog(signal, logPartPtr);
+ regTcPtr->transactionState = TcConnectionrec::ABORT_QUEUED;
+ if (logPartPtr.p->logPartState == LogPartRecord::IDLE) {
+ jam();
+ logPartPtr.p->logPartState = LogPartRecord::ACTIVE;
+ }//if
+ return;
+ }//if
+ signal->theData[0] = regTcPtr->tupConnectrec;
+ EXECUTE_DIRECT(DBTUP, GSN_TUP_ABORTREQ, signal, 1);
+ regTcPtr->transactionState = TcConnectionrec::WAIT_ACC_ABORT;
+ signal->theData[0] = regTcPtr->accConnectrec;
+ EXECUTE_DIRECT(DBACC, GSN_ACC_ABORTREQ, signal, 1);
+ /* ------------------------------------------------------------------------
+ * We need to insert a real-time break by sending ACC_ABORTCONF through the
+ * job buffer to ensure that we catch any ACCKEYCONF or TUPKEYCONF or
+ * TUPKEYREF that are in the job buffer but not yet processed. Doing
+ * everything without that would race and create a state error when they
+ * are executed.
+ * ----------------------------------------------------------------------- */
+ return;
+}//Dblqh::abortContinueAfterBlockedLab()
+
+/* ******************>> */
+/* ACC_ABORTCONF > */
+/* ******************>> */
+void Dblqh::execACC_ABORTCONF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ ndbrequire(regTcPtr->transactionState == TcConnectionrec::WAIT_ACC_ABORT);
+ if (regTcPtr->activeCreat == ZTRUE) {
+ /* ----------------------------------------------------------------------
+ * A NORMAL EVENT DURING CREATION OF A FRAGMENT. WE NOW NEED TO CONTINUE
+ * WITH NORMAL COMMIT PROCESSING.
+ * ---------------------------------------------------------------------- */
+ if (regTcPtr->currTupAiLen == regTcPtr->totReclenAi) {
+ jam();
+ regTcPtr->abortState = TcConnectionrec::ABORT_IDLE;
+ rwConcludedLab(signal);
+ return;
+ } else {
+ ndbrequire(regTcPtr->currTupAiLen < regTcPtr->totReclenAi);
+ jam();
+ releaseActiveFrag(signal);
+ regTcPtr->transactionState = TcConnectionrec::WAIT_AI_AFTER_ABORT;
+ return;
+ }//if
+ }//if
+ releaseActiveFrag(signal);
+ continueAbortLab(signal);
+ return;
+}//Dblqh::execACC_ABORTCONF()
+
+void Dblqh::continueAbortLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ /* ------------------------------------------------------------------------
+ * AN ERROR OCCURED IN THE ACTIVE CREATION AFTER THE ABORT PHASE.
+ * WE NEED TO CONTINUE WITH A NORMAL ABORT.
+ * ------------------------------------------------------------------------
+ * ALSO USED FOR NORMAL CLEAN UP AFTER A NORMAL ABORT.
+ * ------------------------------------------------------------------------
+ * ALSO USED WHEN NO FRAGMENT WAS SET UP ON OPERATION.
+ * ------------------------------------------------------------------------ */
+ if (regTcPtr->logWriteState == TcConnectionrec::WRITTEN) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * I NEED TO INSERT A ABORT LOG RECORD SINCE WE ARE WRITING LOG IN THIS
+ * TRANSACTION.
+ * ---------------------------------------------------------------------- */
+ initLogPointers(signal);
+ if (logPartPtr.p->logPartState == LogPartRecord::ACTIVE) {
+ jam();
+ /* --------------------------------------------------------------------
+ * A PREPARE OPERATION IS CURRENTLY WRITING IN THE LOG.
+ * WE MUST WAIT ON OUR TURN TO WRITE THE LOG.
+ * IT IS NECESSARY TO WRITE ONE LOG RECORD COMPLETELY
+ * AT A TIME OTHERWISE WE WILL SCRAMBLE THE LOG.
+ * -------------------------------------------------------------------- */
+ linkWaitLog(signal, logPartPtr);
+ regTcPtr->transactionState = TcConnectionrec::LOG_ABORT_QUEUED;
+ return;
+ }//if
+ if (cnoOfLogPages == 0) {
+ jam();
+/*---------------------------------------------------------------------------*/
+// We must delay the write of commit info to the log to safe-guard against
+// a crash due to lack of log pages. We temporary stop all log writes to this
+// log part to ensure that we don't get a buffer explosion in the delayed
+// signal buffer instead.
+/*---------------------------------------------------------------------------*/
+ linkWaitLog(signal, logPartPtr);
+ regTcPtr->transactionState = TcConnectionrec::LOG_ABORT_QUEUED;
+ if (logPartPtr.p->logPartState == LogPartRecord::IDLE) {
+ jam();
+ logPartPtr.p->logPartState = LogPartRecord::ACTIVE;
+ }//if
+ return;
+ }//if
+ writeAbortLog(signal);
+ removeLogTcrec(signal);
+ } else if (regTcPtr->logWriteState == TcConnectionrec::NOT_STARTED) {
+ jam();
+ } else if (regTcPtr->logWriteState == TcConnectionrec::NOT_WRITTEN) {
+ jam();
+ /* ------------------------------------------------------------------
+ * IT IS A READ OPERATION OR OTHER OPERATION THAT DO NOT USE THE LOG.
+ * ------------------------------------------------------------------ */
+ /* ------------------------------------------------------------------
+ * THE LOG HAS NOT BEEN WRITTEN SINCE THE LOG FLAG WAS FALSE.
+ * THIS CAN OCCUR WHEN WE ARE STARTING A NEW FRAGMENT.
+ * ------------------------------------------------------------------ */
+ regTcPtr->logWriteState = TcConnectionrec::NOT_STARTED;
+ } else {
+ ndbrequire(regTcPtr->logWriteState == TcConnectionrec::NOT_WRITTEN_WAIT);
+ jam();
+ /* ----------------------------------------------------------------
+ * THE STATE WAS SET TO NOT_WRITTEN BY THE OPERATION BUT LATER
+ * A SCAN OF ALL OPERATION RECORD CHANGED IT INTO NOT_WRITTEN_WAIT.
+ * THIS INDICATES THAT WE ARE WAITING FOR THIS OPERATION TO COMMIT
+ * OR ABORT SO THAT WE CAN FIND THE
+ * STARTING GLOBAL CHECKPOINT OF THIS NEW FRAGMENT.
+ * ---------------------------------------------------------------- */
+ checkScanTcCompleted(signal);
+ }//if
+ continueAfterLogAbortWriteLab(signal);
+ return;
+}//Dblqh::continueAbortLab()
+
+void Dblqh::continueAfterLogAbortWriteLab(Signal* signal)
+{
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->simpleRead == ZSIMPLE_READ) {
+ jam();
+ TcKeyRef * const tcKeyRef = (TcKeyRef *) signal->getDataPtrSend();
+
+ tcKeyRef->connectPtr = regTcPtr->applOprec;
+ tcKeyRef->transId[0] = regTcPtr->transid[0];
+ tcKeyRef->transId[1] = regTcPtr->transid[1];
+ tcKeyRef->errorCode = regTcPtr->errorCode;
+ sendSignal(regTcPtr->applRef,
+ GSN_TCKEYREF, signal, TcKeyRef::SignalLength, JBB);
+ cleanUp(signal);
+ return;
+ }//if
+ if (regTcPtr->abortState == TcConnectionrec::ABORT_FROM_LQH) {
+ LqhKeyRef * const lqhKeyRef = (LqhKeyRef *)signal->getDataPtrSend();
+
+ jam();
+ lqhKeyRef->userRef = regTcPtr->clientConnectrec;
+ lqhKeyRef->connectPtr = regTcPtr->tcOprec;
+ lqhKeyRef->errorCode = regTcPtr->errorCode;
+ lqhKeyRef->transId1 = regTcPtr->transid[0];
+ lqhKeyRef->transId2 = regTcPtr->transid[1];
+ sendSignal(regTcPtr->clientBlockref, GSN_LQHKEYREF, signal,
+ LqhKeyRef::SignalLength, JBB);
+ } else if (regTcPtr->abortState == TcConnectionrec::ABORT_FROM_TC) {
+ jam();
+ sendAborted(signal);
+ } else if (regTcPtr->abortState == TcConnectionrec::NEW_FROM_TC) {
+ jam();
+ sendLqhTransconf(signal, LqhTransConf::Aborted);
+ } else {
+ ndbrequire(regTcPtr->abortState == TcConnectionrec::REQ_FROM_TC);
+ jam();
+ signal->theData[0] = regTcPtr->reqRef;
+ signal->theData[1] = tcConnectptr.i;
+ signal->theData[2] = cownNodeid;
+ signal->theData[3] = regTcPtr->transid[0];
+ signal->theData[4] = regTcPtr->transid[1];
+ sendSignal(regTcPtr->reqBlockref, GSN_ABORTCONF,
+ signal, 5, JBB);
+ }//if
+ cleanUp(signal);
+}//Dblqh::continueAfterLogAbortWriteLab()
+
+/* ##########################################################################
+ * ####### MODULE TO HANDLE TC FAILURE #######
+ *
+ * ########################################################################## */
+
+/* ************************************************************************>>
+ * NODE_FAILREP: Node failure report. Sender Ndbcntr. Set status of failed
+ * node to down and reply with NF_COMPLETEREP to DIH which will report that
+ * LQH has completed failure handling.
+ * ************************************************************************>> */
+void Dblqh::execNODE_FAILREP(Signal* signal)
+{
+ UintR TfoundNodes = 0;
+ UintR TnoOfNodes;
+ UintR Tdata[MAX_NDB_NODES];
+
+ NodeFailRep * const nodeFail = (NodeFailRep *)&signal->theData[0];
+
+ TnoOfNodes = nodeFail->noOfNodes;
+ UintR index = 0;
+ for (Uint32 i = 1; i < MAX_NDB_NODES; i++) {
+ jam();
+ if(NodeBitmask::get(nodeFail->theNodes, i)){
+ jam();
+ Tdata[index] = i;
+ index++;
+ }//if
+ }//for
+
+ lcpPtr.i = 0;
+ ptrAss(lcpPtr, lcpRecord);
+
+ ndbrequire(index == TnoOfNodes);
+ ndbrequire(cnoOfNodes - 1 < MAX_NDB_NODES);
+ for (Uint32 i = 0; i < TnoOfNodes; i++) {
+ const Uint32 nodeId = Tdata[i];
+ lcpPtr.p->m_EMPTY_LCP_REQ.clear(nodeId);
+
+ for (Uint32 j = 0; j < cnoOfNodes; j++) {
+ jam();
+ if (cnodeData[j] == nodeId){
+ jam();
+ cnodeStatus[j] = ZNODE_DOWN;
+
+ TfoundNodes++;
+ }//if
+ }//for
+ NFCompleteRep * const nfCompRep = (NFCompleteRep *)&signal->theData[0];
+ nfCompRep->blockNo = DBLQH;
+ nfCompRep->nodeId = cownNodeid;
+ nfCompRep->failedNodeId = Tdata[i];
+ sendSignal(DBDIH_REF, GSN_NF_COMPLETEREP, signal,
+ NFCompleteRep::SignalLength, JBB);
+ }//for
+ ndbrequire(TnoOfNodes == TfoundNodes);
+}//Dblqh::execNODE_FAILREP()
+
+/* ************************************************************************>>
+ * LQH_TRANSREQ: Report status of all transactions where TC was coordinated
+ * by a crashed TC
+ * ************************************************************************>> */
+/* ************************************************************************>>
+ * THIS SIGNAL IS RECEIVED AFTER A NODE CRASH.
+ * THE NODE HAD A TC AND COORDINATED A NUMBER OF TRANSACTIONS.
+ * NOW THE MASTER NODE IS PICKING UP THOSE TRANSACTIONS
+ * TO COMPLETE THEM. EITHER ABORT THEM OR COMMIT THEM.
+ * ************************************************************************>> */
+void Dblqh::execLQH_TRANSREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 newTcPtr = signal->theData[0];
+ BlockReference newTcBlockref = signal->theData[1];
+ Uint32 oldNodeId = signal->theData[2];
+ tcNodeFailptr.i = oldNodeId;
+ ptrCheckGuard(tcNodeFailptr, ctcNodeFailrecFileSize, tcNodeFailRecord);
+ if ((tcNodeFailptr.p->tcFailStatus == TcNodeFailRecord::TC_STATE_TRUE) ||
+ (tcNodeFailptr.p->tcFailStatus == TcNodeFailRecord::TC_STATE_BREAK)) {
+ jam();
+ tcNodeFailptr.p->lastNewTcBlockref = newTcBlockref;
+ /* ------------------------------------------------------------------------
+ * WE HAVE RECEIVED A SIGNAL SPECIFYING THAT WE NEED TO HANDLE THE FAILURE
+ * OF A TC. NOW WE RECEIVE ANOTHER SIGNAL WITH THE SAME ORDER. THIS CAN
+ * OCCUR IF THE NEW TC FAILS. WE MUST BE CAREFUL IN THIS CASE SO THAT WE DO
+ * NOT START PARALLEL ACTIVITIES TRYING TO DO THE SAME THING. WE SAVE THE
+ * NEW BLOCK REFERENCE TO THE LAST NEW TC IN A VARIABLE AND ASSIGN TO IT TO
+ * NEW_TC_BLOCKREF WHEN THE OLD PROCESS RETURNS TO LQH_TRANS_NEXT. IT IS
+ * CERTAIN TO COME THERE SINCE THIS IS THE ONLY PATH TO TAKE CARE OF THE
+ * NEXT TC CONNECT RECORD. WE SET THE STATUS TO BREAK TO INDICATE TO THE OLD
+ * PROCESS WHAT IS HAPPENING.
+ * ------------------------------------------------------------------------ */
+ tcNodeFailptr.p->lastNewTcRef = newTcPtr;
+ tcNodeFailptr.p->tcFailStatus = TcNodeFailRecord::TC_STATE_BREAK;
+ return;
+ }//if
+ tcNodeFailptr.p->oldNodeId = oldNodeId;
+ tcNodeFailptr.p->newTcBlockref = newTcBlockref;
+ tcNodeFailptr.p->newTcRef = newTcPtr;
+ tcNodeFailptr.p->tcRecNow = 0;
+ tcNodeFailptr.p->tcFailStatus = TcNodeFailRecord::TC_STATE_TRUE;
+ signal->theData[0] = ZLQH_TRANS_NEXT;
+ signal->theData[1] = tcNodeFailptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+}//Dblqh::execLQH_TRANSREQ()
+
+void Dblqh::lqhTransNextLab(Signal* signal)
+{
+ UintR tend;
+ UintR tstart;
+ UintR guard0;
+
+ if (tcNodeFailptr.p->tcFailStatus == TcNodeFailRecord::TC_STATE_BREAK) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * AN INTERRUPTION TO THIS NODE FAIL HANDLING WAS RECEIVED AND A NEW
+ * TC HAVE BEEN ASSIGNED TO TAKE OVER THE FAILED TC. PROBABLY THE OLD
+ * NEW TC HAVE FAILED.
+ * ---------------------------------------------------------------------- */
+ tcNodeFailptr.p->newTcBlockref = tcNodeFailptr.p->lastNewTcBlockref;
+ tcNodeFailptr.p->newTcRef = tcNodeFailptr.p->lastNewTcRef;
+ tcNodeFailptr.p->tcRecNow = 0;
+ tcNodeFailptr.p->tcFailStatus = TcNodeFailRecord::TC_STATE_TRUE;
+ }//if
+ tstart = tcNodeFailptr.p->tcRecNow;
+ tend = tstart + 200;
+ guard0 = tend;
+ for (tcConnectptr.i = tstart; tcConnectptr.i <= guard0; tcConnectptr.i++) {
+ jam();
+ if (tcConnectptr.i >= ctcConnectrecFileSize) {
+ jam();
+ /**
+ * Finished with scanning operation record
+ *
+ * now scan markers
+ */
+ scanMarkers(signal, tcNodeFailptr.i, 0, RNIL);
+ return;
+ }//if
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ if (tcConnectptr.p->transactionState != TcConnectionrec::IDLE) {
+ if (tcConnectptr.p->transactionState != TcConnectionrec::TC_NOT_CONNECTED) {
+ if (tcConnectptr.p->tcScanRec == RNIL) {
+ if (refToNode(tcConnectptr.p->tcBlockref) == tcNodeFailptr.p->oldNodeId) {
+ if (tcConnectptr.p->operation != ZREAD) {
+ jam();
+ tcConnectptr.p->tcNodeFailrec = tcNodeFailptr.i;
+ tcConnectptr.p->abortState = TcConnectionrec::NEW_FROM_TC;
+ abortStateHandlerLab(signal);
+ return;
+ } else {
+ jam();
+ if (tcConnectptr.p->opSimple != ZTRUE) {
+ jam();
+ tcConnectptr.p->tcNodeFailrec = tcNodeFailptr.i;
+ tcConnectptr.p->abortState = TcConnectionrec::NEW_FROM_TC;
+ abortStateHandlerLab(signal);
+ return;
+ }//if
+ }//if
+ }//if
+ } else {
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ if (scanptr.p->scanType == ScanRecord::COPY) {
+ jam();
+ if (scanptr.p->scanNodeId == tcNodeFailptr.p->oldNodeId) {
+ jam();
+ /* ------------------------------------------------------------
+ * THE RECEIVER OF THE COPY HAVE FAILED.
+ * WE HAVE TO CLOSE THE COPY PROCESS.
+ * ------------------------------------------------------------ */
+ tcConnectptr.p->tcNodeFailrec = tcNodeFailptr.i;
+ tcConnectptr.p->abortState = TcConnectionrec::NEW_FROM_TC;
+ closeCopyRequestLab(signal);
+ return;
+ }//if
+ } else {
+ if (scanptr.p->scanType == ScanRecord::SCAN) {
+ jam();
+ if (refToNode(tcConnectptr.p->tcBlockref) ==
+ tcNodeFailptr.p->oldNodeId) {
+ jam();
+ tcConnectptr.p->tcNodeFailrec = tcNodeFailptr.i;
+ tcConnectptr.p->abortState = TcConnectionrec::NEW_FROM_TC;
+ closeScanRequestLab(signal);
+ return;
+ }//if
+ } else {
+ jam();
+ /* ------------------------------------------------------------
+ * THIS IS AN ERROR THAT SHOULD NOT OCCUR. WE CRASH THE SYSTEM.
+ * ------------------------------------------------------------ */
+ systemErrorLab(signal);
+ return;
+ }//if
+ }//if
+ }//if
+ }//if
+ }//if
+ }//for
+ tcNodeFailptr.p->tcRecNow = tend + 1;
+ signal->theData[0] = ZLQH_TRANS_NEXT;
+ signal->theData[1] = tcNodeFailptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+}//Dblqh::lqhTransNextLab()
+
+void
+Dblqh::scanMarkers(Signal* signal,
+ Uint32 tcNodeFail,
+ Uint32 startBucket,
+ Uint32 i){
+
+ jam();
+
+ TcNodeFailRecordPtr tcNodeFailPtr;
+ tcNodeFailPtr.i = tcNodeFail;
+ ptrCheckGuard(tcNodeFailPtr, ctcNodeFailrecFileSize, tcNodeFailRecord);
+ const Uint32 crashedTcNodeId = tcNodeFailPtr.p->oldNodeId;
+
+ CommitAckMarkerIterator iter;
+ if(i == RNIL){
+ m_commitAckMarkerHash.next(startBucket, iter);
+ } else {
+ jam();
+ iter.curr.i = i;
+ iter.bucket = startBucket;
+ m_commitAckMarkerHash.getPtr(iter.curr);
+ m_commitAckMarkerHash.next(iter);
+ }
+
+ const Uint32 RT_BREAK = 256;
+ for(Uint32 i = 0; i<RT_BREAK || iter.bucket == startBucket; i++){
+ jam();
+
+ if(iter.curr.i == RNIL){
+ /**
+ * Done with iteration
+ */
+ jam();
+
+ tcNodeFailPtr.p->tcFailStatus = TcNodeFailRecord::TC_STATE_FALSE;
+ signal->theData[0] = tcNodeFailPtr.p->newTcRef;
+ signal->theData[1] = cownNodeid;
+ signal->theData[2] = LqhTransConf::LastTransConf;
+ sendSignal(tcNodeFailPtr.p->newTcBlockref, GSN_LQH_TRANSCONF,
+ signal, 3, JBB);
+ return;
+ }
+
+ if(iter.curr.p->tcNodeId == crashedTcNodeId){
+ jam();
+
+ /**
+ * Found marker belonging to crashed node
+ */
+ LqhTransConf * const lqhTransConf = (LqhTransConf *)&signal->theData[0];
+ lqhTransConf->tcRef = tcNodeFailPtr.p->newTcRef;
+ lqhTransConf->lqhNodeId = cownNodeid;
+ lqhTransConf->operationStatus = LqhTransConf::Marker;
+ lqhTransConf->transId1 = iter.curr.p->transid1;
+ lqhTransConf->transId2 = iter.curr.p->transid2;
+ lqhTransConf->apiRef = iter.curr.p->apiRef;
+ lqhTransConf->apiOpRec = iter.curr.p->apiOprec;
+ sendSignal(tcNodeFailPtr.p->newTcBlockref, GSN_LQH_TRANSCONF,
+ signal, 7, JBB);
+
+ signal->theData[0] = ZSCAN_MARKERS;
+ signal->theData[1] = tcNodeFailPtr.i;
+ signal->theData[2] = iter.bucket;
+ signal->theData[3] = iter.curr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 4, JBB);
+ return;
+ }
+
+ m_commitAckMarkerHash.next(iter);
+ }
+
+ signal->theData[0] = ZSCAN_MARKERS;
+ signal->theData[1] = tcNodeFailPtr.i;
+ signal->theData[2] = iter.bucket;
+ signal->theData[3] = RNIL;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 4, JBB);
+}
+
+/* #########################################################################
+ * ####### SCAN MODULE #######
+ *
+ * #########################################################################
+ * -------------------------------------------------------------------------
+ * THIS MODULE CONTAINS THE CODE THAT HANDLES A SCAN OF A PARTICULAR FRAGMENT
+ * IT OPERATES UNDER THE CONTROL OF TC AND ORDERS ACC TO PERFORM A SCAN OF
+ * ALL TUPLES IN THE FRAGMENT. TUP PERFORMS THE NECESSARY SEARCH CONDITIONS
+ * TO ENSURE THAT ONLY VALID TUPLES ARE RETURNED TO THE APPLICATION.
+ * ------------------------------------------------------------------------- */
+
+void Dblqh::execACC_SCAN_INFO(Signal* signal)
+{
+ jamEntry();
+ scanptr.i = signal->theData[0];
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ Uint32 length = signal->theData[3];
+ ndbrequire(length <= 4);
+ accScanInfoEnterLab(signal, &signal->theData[4], length);
+}//Dblqh::execACC_SCAN_INFO()
+
+
+void Dblqh::execACC_SCAN_INFO24(Signal* signal)
+{
+ jamEntry();
+ scanptr.i = signal->theData[0];
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ Uint32 length = signal->theData[3];
+ ndbrequire(length <= 20);
+ accScanInfoEnterLab(signal, &signal->theData[4], length);
+}//Dblqh::execACC_SCAN_INFO24()
+
+void Dblqh::accScanInfoEnterLab(Signal* signal,
+ Uint32* dataPtr,
+ Uint32 length)
+{
+ ndbrequire(length != 0);
+ if (scanptr.p->scanState == ScanRecord::WAIT_SCAN_KEYINFO) {
+ jam();
+ if (keyinfoLab(signal, dataPtr, length)) {
+ jam();
+ nextScanConfLoopLab(signal);
+ }//if
+ } else {
+ ndbrequire(scanptr.p->scanState == ScanRecord::WAIT_COPY_KEYINFO);
+ jam();
+ if (keyinfoLab(signal, dataPtr, length)) {
+ jam();
+ copySendTupkeyReqLab(signal);
+ }//if
+ }//if
+}//Dblqh::accScanInfoEnterLab()
+
+/* *************** */
+/* ACC_SCANCONF > */
+/* *************** */
+void Dblqh::execACC_SCANCONF(Signal* signal)
+{
+ AccScanConf * const accScanConf = (AccScanConf *)&signal->theData[0];
+ jamEntry();
+ scanptr.i = accScanConf->scanPtr;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ if (scanptr.p->scanState == ScanRecord::WAIT_ACC_SCAN) {
+ accScanConfScanLab(signal);
+ } else {
+ ndbrequire(scanptr.p->scanState == ScanRecord::WAIT_ACC_COPY);
+ accScanConfCopyLab(signal);
+ }//if
+}//Dblqh::execACC_SCANCONF()
+
+/* ************>> */
+/* ACC_SCANREF > */
+/* ************>> */
+void Dblqh::execACC_SCANREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execACC_SCANREF()
+
+/* ***************>> */
+/* NEXT_SCANCONF > */
+/* ***************>> */
+void Dblqh::execNEXT_SCANCONF(Signal* signal)
+{
+ NextScanConf * const nextScanConf = (NextScanConf *)&signal->theData[0];
+ jamEntry();
+ scanptr.i = nextScanConf->scanPtr;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ if (nextScanConf->localKeyLength == 1) {
+ jam();
+ nextScanConf->localKey[1] =
+ nextScanConf->localKey[0] & MAX_TUPLES_PER_PAGE;
+ nextScanConf->localKey[0] = nextScanConf->localKey[0] >> MAX_TUPLES_BITS;
+ }//if
+ switch (scanptr.p->scanState) {
+ case ScanRecord::WAIT_CLOSE_SCAN:
+ jam();
+ accScanCloseConfLab(signal);
+ break;
+ case ScanRecord::WAIT_CLOSE_COPY:
+ jam();
+ accCopyCloseConfLab(signal);
+ break;
+ case ScanRecord::WAIT_NEXT_SCAN:
+ jam();
+ nextScanConfScanLab(signal);
+ break;
+ case ScanRecord::WAIT_NEXT_SCAN_COPY:
+ jam();
+ nextScanConfCopyLab(signal);
+ break;
+ case ScanRecord::WAIT_RELEASE_LOCK:
+ jam();
+ ndbrequire(signal->length() == 1);
+ scanLockReleasedLab(signal);
+ break;
+ default:
+ ndbrequire(false);
+ }//switch
+}//Dblqh::execNEXT_SCANCONF()
+
+/* ***************> */
+/* NEXT_SCANREF > */
+/* ***************> */
+void Dblqh::execNEXT_SCANREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Dblqh::execNEXT_SCANREF()
+
+/* ******************> */
+/* STORED_PROCCONF > */
+/* ******************> */
+void Dblqh::execSTORED_PROCCONF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ Uint32 storedProcId = signal->theData[1];
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ switch (scanptr.p->scanState) {
+ case ScanRecord::WAIT_STORED_PROC_SCAN:
+ jam();
+ scanptr.p->scanStoredProcId = storedProcId;
+ storedProcConfScanLab(signal);
+ break;
+ case ScanRecord::WAIT_DELETE_STORED_PROC_ID_SCAN:
+ jam();
+ releaseActiveFrag(signal);
+ tupScanCloseConfLab(signal);
+ break;
+ case ScanRecord::WAIT_STORED_PROC_COPY:
+ jam();
+ scanptr.p->scanStoredProcId = storedProcId;
+ storedProcConfCopyLab(signal);
+ break;
+ case ScanRecord::WAIT_DELETE_STORED_PROC_ID_COPY:
+ jam();
+ releaseActiveFrag(signal);
+ tupCopyCloseConfLab(signal);
+ break;
+ default:
+ ndbrequire(false);
+ }//switch
+}//Dblqh::execSTORED_PROCCONF()
+
+/* ****************** */
+/* STORED_PROCREF > */
+/* ****************** */
+void Dblqh::execSTORED_PROCREF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ Uint32 errorCode = signal->theData[1];
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ switch (scanptr.p->scanState) {
+ case ScanRecord::WAIT_STORED_PROC_SCAN:
+ jam();
+ scanptr.p->scanStoredProcId = signal->theData[2];
+ tcConnectptr.p->errorCode = errorCode;
+ closeScanLab(signal);
+ break;
+ default:
+ ndbrequire(false);
+ }//switch
+}//Dblqh::execSTORED_PROCREF()
+
+/* --------------------------------------------------------------------------
+ * ENTER SCAN_NEXTREQ
+ * --------------------------------------------------------------------------
+ * PRECONDITION:
+ * TRANSACTION_STATE = SCAN_STATE
+ * SCAN_STATE = WAIT_SCAN_NEXTREQ
+ *
+ * Case scanLockHold: ZTRUE = Unlock previous round of
+ * scanned row(s) and fetch next set of rows.
+ * ZFALSE = Fetch new set of rows.
+ * Number of rows to read depends on parallelism and how many rows
+ * left to scan in the fragment. SCAN_NEXTREQ can also be sent with
+ * closeFlag == ZTRUE to close the scan.
+ * ------------------------------------------------------------------------- */
+void Dblqh::execSCAN_NEXTREQ(Signal* signal)
+{
+ jamEntry();
+ const ScanFragNextReq * const nextReq =
+ (ScanFragNextReq*)&signal->theData[0];
+ const Uint32 transid1 = nextReq->transId1;
+ const Uint32 transid2 = nextReq->transId2;
+ const Uint32 senderData = nextReq->senderData;
+
+ if (findTransaction(transid1, transid2, senderData) != ZOK){
+ jam();
+ DEBUG("Received SCAN_NEXTREQ in LQH with close flag when closed");
+ ndbrequire(nextReq->closeFlag == ZTRUE);
+ return;
+ }
+
+ // Crash node if signal sender is same node
+ CRASH_INSERTION2(5021, refToNode(signal->senderBlockRef()) == cownNodeid);
+ // Crash node if signal sender is NOT same node
+ CRASH_INSERTION2(5022, refToNode(signal->senderBlockRef()) != cownNodeid);
+
+ if (ERROR_INSERTED(5023)){
+ // Drop signal if sender is same node
+ if (refToNode(signal->senderBlockRef()) == cownNodeid) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }
+ }//if
+ if (ERROR_INSERTED(5024)){
+ // Drop signal if sender is NOT same node
+ if (refToNode(signal->senderBlockRef()) != cownNodeid) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }
+ }//if
+ if (ERROR_INSERTED(5025)){
+ // Delay signal if sender is NOT same node
+ if (refToNode(signal->senderBlockRef()) != cownNodeid) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_SCAN_NEXTREQ, signal, 1000,
+ signal->length());
+ return;
+ }
+ }//if
+ if (ERROR_INSERTED(5030)){
+ ndbout << "ERROR 5030" << endl;
+ // Drop signal
+ return;
+ }//if
+
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ndbrequire(scanptr.i != RNIL);
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ scanptr.p->scanTcWaiting = ZTRUE;
+
+ /* ------------------------------------------------------------------
+ * If close flag is set this scan should be closed
+ * If we are waiting for SCAN_NEXTREQ set flag to stop scanning and
+ * continue execution else set flags and wait until the scan
+ * completes itself
+ * ------------------------------------------------------------------ */
+ if (nextReq->closeFlag == ZTRUE){
+ jam();
+ closeScanRequestLab(signal);
+ return;
+ }//if
+
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+
+ /* --------------------------------------------------------------------
+ * If scanLockHold = TRUE we need to unlock previous round of
+ * scanned records.
+ * scanReleaseLocks will set states for this and send a NEXT_SCANREQ.
+ * When confirm signal NEXT_SCANCONF arrives we call
+ * continueScanNextReqLab to continue scanning new rows and
+ * acquiring new locks.
+ * -------------------------------------------------------------------- */
+ if ((scanptr.p->scanLockHold == ZTRUE) &&
+ (scanptr.p->scanCompletedOperations > 0)) {
+ jam();
+ scanptr.p->scanReleaseCounter = 1;
+ scanReleaseLocksLab(signal);
+ return;
+ }//if
+
+ /* -----------------------------------------------------------------------
+ * We end up here when scanLockHold = FALSE or no rows was locked from
+ * previous round.
+ * Simply continue scanning.
+ * ----------------------------------------------------------------------- */
+ continueScanNextReqLab(signal);
+}//Dblqh::execSCAN_NEXTREQ()
+
+void Dblqh::continueScanNextReqLab(Signal* signal)
+{
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ jam();
+ closeScanLab(signal);
+ return;
+ }//if
+
+ // Update timer on tcConnectRecord
+ tcConnectptr.p->tcTimer = cLqhTimeOutCount;
+
+ initScanAccOp(signal);
+ scanptr.p->scanCompletedOperations = 0;
+ scanptr.p->scanFlag = NextScanReq::ZSCAN_NEXT;
+ scanNextLoopLab(signal);
+}//Dblqh::continueScanNextReqLab()
+
+/* -------------------------------------------------------------------------
+ * WE NEED TO RELEASE LOCKS BEFORE CONTINUING
+ * ------------------------------------------------------------------------- */
+void Dblqh::scanReleaseLocksLab(Signal* signal)
+{
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ jam();
+ linkActiveFrag(signal);
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_RELEASE_STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ case Fragrecord::CRASH_RECOVERING:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ ndbrequire(false);
+ }//switch
+ continueScanReleaseAfterBlockedLab(signal);
+}//Dblqh::scanReleaseLocksLab()
+
+void Dblqh::continueScanReleaseAfterBlockedLab(Signal* signal)
+{
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ scanptr.p->scanState = ScanRecord::WAIT_RELEASE_LOCK;
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ ndbrequire((scanptr.p->scanReleaseCounter -1) < MAX_PARALLEL_OP_PER_SCAN);
+ signal->theData[1] =
+ scanptr.p->scanAccOpPtr[scanptr.p->scanReleaseCounter -1];
+ signal->theData[2] = NextScanReq::ZSCAN_COMMIT;
+ if (! scanptr.p->rangeScan)
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+ else
+ sendSignal(tcConnectptr.p->tcTuxBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+}//Dblqh::continueScanReleaseAfterBlockedLab()
+
+/* -------------------------------------------------------------------------
+ * ENTER SCAN_NEXTREQ
+ * -------------------------------------------------------------------------
+ * SCAN_NEXT_REQ SIGNAL ARRIVED IN THE MIDDLE OF EXECUTION OF THE SCAN.
+ * IT WAS A REQUEST TO CLOSE THE SCAN. WE WILL CLOSE THE SCAN IN A
+ * CAREFUL MANNER TO ENSURE THAT NO ERROR OCCURS.
+ * -------------------------------------------------------------------------
+ * PRECONDITION:
+ * TRANSACTION_STATE = SCAN_STATE_USED
+ * TSCAN_COMPLETED = ZTRUE
+ * -------------------------------------------------------------------------
+ * WE CAN ALSO ARRIVE AT THIS LABEL AFTER A NODE CRASH OF THE SCAN
+ * COORDINATOR.
+ * ------------------------------------------------------------------------- */
+void Dblqh::closeScanRequestLab(Signal* signal)
+{
+ DEBUG("transactionState = " << tcConnectptr.p->transactionState);
+ switch (tcConnectptr.p->transactionState) {
+ case TcConnectionrec::SCAN_STATE_USED:
+ DEBUG("scanState = " << scanptr.p->scanState);
+ switch (scanptr.p->scanState) {
+ case ScanRecord::WAIT_SCAN_KEYINFO:
+ case ScanRecord::WAIT_NEXT_SCAN:
+ jam();
+ /* -------------------------------------------------------------------
+ * SET COMPLETION STATUS AND WAIT FOR OPPORTUNITY TO STOP THE SCAN.
+ * ------------------------------------------------------------------- */
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ break;
+ case ScanRecord::WAIT_ACC_SCAN:
+ case ScanRecord::WAIT_STORED_PROC_SCAN:
+ jam();
+ /* -------------------------------------------------------------------
+ * WE ARE CURRENTLY STARTING UP THE SCAN. SET COMPLETED STATUS
+ * AND WAIT FOR COMPLETION OF STARTUP.
+ * ------------------------------------------------------------------- */
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ break;
+ case ScanRecord::WAIT_CLOSE_SCAN:
+ case ScanRecord::WAIT_DELETE_STORED_PROC_ID_SCAN:
+ jam();
+ /*empty*/;
+ break;
+ /* -------------------------------------------------------------------
+ * CLOSE IS ALREADY ONGOING. WE NEED NOT DO ANYTHING.
+ * ------------------------------------------------------------------- */
+ case ScanRecord::WAIT_RELEASE_LOCK:
+ jam();
+ /* -------------------------------------------------------------------
+ * WE ARE CURRENTLY RELEASING RECORD LOCKS. AFTER COMPLETING THIS
+ * WE WILL START TO CLOSE THE SCAN.
+ * ------------------------------------------------------------------- */
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ break;
+ case ScanRecord::WAIT_SCAN_NEXTREQ:
+ jam();
+ /* -------------------------------------------------------------------
+ * WE ARE WAITING FOR A SCAN_NEXTREQ FROM SCAN COORDINATOR(TC)
+ * WICH HAVE CRASHED. CLOSE THE SCAN
+ * ------------------------------------------------------------------- */
+ scanptr.p->scanCompletedStatus = ZTRUE;
+
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+
+ if (scanptr.p->scanLockHold == ZTRUE) {
+ if (scanptr.p->scanCompletedOperations > 0) {
+ jam();
+ scanptr.p->scanReleaseCounter = 1;
+ scanReleaseLocksLab(signal);
+ return;
+ }//if
+ }//if
+ closeScanLab(signal);
+ break;
+ default:
+ ndbrequire(false);
+ }//switch
+ break;
+ case TcConnectionrec::WAIT_SCAN_AI:
+ jam();
+ /* ---------------------------------------------------------------------
+ * WE ARE STILL WAITING FOR THE ATTRIBUTE INFORMATION THAT
+ * OBVIOUSLY WILL NOT ARRIVE. WE CAN QUIT IMMEDIATELY HERE.
+ * --------------------------------------------------------------------- */
+ releaseOprec(signal);
+ if (tcConnectptr.p->abortState == TcConnectionrec::NEW_FROM_TC) {
+ jam();
+ tcNodeFailptr.i = tcConnectptr.p->tcNodeFailrec;
+ ptrCheckGuard(tcNodeFailptr, ctcNodeFailrecFileSize, tcNodeFailRecord);
+ tcNodeFailptr.p->tcRecNow = tcConnectptr.i + 1;
+ signal->theData[0] = ZLQH_TRANS_NEXT;
+ signal->theData[1] = tcNodeFailptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ }//if
+ tcConnectptr.p->abortState = TcConnectionrec::ABORT_ACTIVE;
+ scanptr.p->scanCompletedOperations = 0;
+ sendScanFragConf(signal, ZTRUE);
+ break;
+ case TcConnectionrec::SCAN_TUPKEY:
+ case TcConnectionrec::SCAN_FIRST_STOPPED:
+ case TcConnectionrec::SCAN_CHECK_STOPPED:
+ case TcConnectionrec::SCAN_STOPPED:
+ jam();
+ /* ---------------------------------------------------------------------
+ * SET COMPLETION STATUS AND WAIT FOR OPPORTUNITY TO STOP THE SCAN.
+ * --------------------------------------------------------------------- */
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ break;
+ case TcConnectionrec::SCAN_RELEASE_STOPPED:
+ jam();
+ /* ---------------------------------------------------------------------
+ * WE ARE CURRENTLY RELEASING RECORD LOCKS. AFTER COMPLETING
+ * THIS WE WILL START TO CLOSE THE SCAN.
+ * --------------------------------------------------------------------- */
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ break;
+ case TcConnectionrec::SCAN_CLOSE_STOPPED:
+ jam();
+ /* ---------------------------------------------------------------------
+ * CLOSE IS ALREADY ONGOING. WE NEED NOT DO ANYTHING.
+ * --------------------------------------------------------------------- */
+ /*empty*/;
+ break;
+ default:
+ ndbrequire(false);
+ }//switch
+}//Dblqh::closeScanRequestLab()
+
+/* -------------------------------------------------------------------------
+ * ENTER NEXT_SCANCONF
+ * -------------------------------------------------------------------------
+ * PRECONDITION: SCAN_STATE = WAIT_RELEASE_LOCK
+ * ------------------------------------------------------------------------- */
+void Dblqh::scanLockReleasedLab(Signal* signal)
+{
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ releaseActiveFrag(signal);
+ if (scanptr.p->scanReleaseCounter == scanptr.p->scanCompletedOperations) {
+ if ((scanptr.p->scanErrorCounter > 0) ||
+ (scanptr.p->scanCompletedStatus == ZTRUE)) {
+ jam();
+ closeScanLab(signal);
+ } else if ((scanptr.p->scanConcurrentOperations ==
+ scanptr.p->scanCompletedOperations) &&
+ scanptr.p->scanLockHold != ZTRUE) {
+ jam();
+ scanptr.p->scanState = ScanRecord::WAIT_SCAN_NEXTREQ;
+ sendScanFragConf(signal, ZFALSE);
+ } else {
+ jam();
+ continueScanNextReqLab(signal);
+ }//if
+ } else {
+ ndbrequire(scanptr.p->scanReleaseCounter <=
+ scanptr.p->scanCompletedOperations);
+ jam();
+ scanptr.p->scanReleaseCounter++;
+ scanReleaseLocksLab(signal);
+ }//if
+}//Dblqh::scanLockReleasedLab()
+
+/* -------------------------------------------------------------------------
+ * SCAN_FRAGREQ: Request to start scanning the specified fragment of a table.
+ * ------------------------------------------------------------------------- */
+void Dblqh::execSCAN_FRAGREQ(Signal* signal)
+{
+ const ScanFragReq * const scanFragReq = (ScanFragReq *)&signal->theData[0];
+ ScanFragRef * ref;
+ const Uint32 transid1 = scanFragReq->transId1;
+ const Uint32 transid2 = scanFragReq->transId2;
+ Uint32 errorCode;
+ Uint32 senderData;
+ Uint32 hashIndex;
+ TcConnectionrecPtr nextHashptr;
+
+ jamEntry();
+ const Uint32 reqinfo = scanFragReq->requestInfo;
+ const Uint32 fragId = scanFragReq->fragmentNo;
+ tabptr.i = scanFragReq->tableId;
+ const Uint32 scanConcurrentOperations = ScanFragReq::getConcurrency(reqinfo);
+ const Uint32 scanLockMode = ScanFragReq::getLockMode(reqinfo);
+ const Uint8 keyinfo = ScanFragReq::getKeyinfoFlag(reqinfo);
+ const Uint8 rangeScan = ScanFragReq::getRangeScanFlag(reqinfo);
+
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ if(tabptr.p->tableStatus != Tablerec::TABLE_DEFINED){
+ senderData = scanFragReq->senderData;
+ goto error_handler_early_1;
+ }
+
+ if (cfirstfreeTcConrec != RNIL) {
+ seizeTcrec();
+ tcConnectptr.p->clientConnectrec = scanFragReq->senderData;
+ tcConnectptr.p->clientBlockref = signal->senderBlockRef();
+ tcConnectptr.p->savePointId = scanFragReq->savePointId;
+ } else {
+ jam();
+ /* ---------------------------------------------------------------------
+ * NO FREE TC RECORD AVAILABLE, THUS WE CANNOT HANDLE THE REQUEST.
+ * --------------------------------------------------------------------- */
+ errorCode = ZNO_TC_CONNECT_ERROR;
+ senderData = scanFragReq->senderData;
+ goto error_handler_early;
+ }//if
+ /**
+ * A write allways have to get keyinfo
+ */
+ ndbrequire(scanLockMode == 0 || keyinfo);
+
+ ndbrequire(scanConcurrentOperations <= MAX_PARALLEL_OP_PER_SCAN);
+ ndbrequire(scanConcurrentOperations != 0);
+ if (!getFragmentrec(signal, fragId)) {
+ errorCode = __LINE__;
+ goto error_handler;
+ }//if
+
+ // Verify scan type vs table type (both sides are boolean)
+ if (rangeScan != DictTabInfo::isOrderedIndex(fragptr.p->tableType)) {
+ errorCode = __LINE__; // XXX fix
+ goto error_handler;
+ }//if
+
+ // 1 table scan is reserved for node recovery
+ if (! rangeScan && fragptr.p->noActiveScan >= NR_ScanNo){
+ jam();
+ errorCode = ScanFragRef::ZTOO_MANY_ACTIVE_SCAN_ERROR;
+ goto error_handler;
+ }
+ // count is actually not used in range scans
+ fragptr.p->noActiveScan++;
+
+ // 1 scan record is reserved for node recovery
+ if (cscanNoFreeRec < 2) {
+ jam();
+ errorCode = ScanFragRef::ZNO_FREE_SCANREC_ERROR;
+ goto error_handler1;
+ }
+
+ // XXX adjust cmaxAccOps for range scans and remove this comment
+ if ((cbookedAccOps + scanConcurrentOperations) > cmaxAccOps) {
+ jam();
+ errorCode = ScanFragRef::ZSCAN_BOOK_ACC_OP_ERROR;
+ goto error_handler1;
+ }//if
+
+ seizeScanrec(signal);
+ initScanTc(signal,
+ transid1,
+ transid2,
+ fragId,
+ ZNIL);
+ errorCode = initScanrec(scanFragReq);
+ if (errorCode != ZOK) {
+ jam();
+ goto error_handler2;
+ }//if
+ cbookedAccOps += scanConcurrentOperations;
+
+ hashIndex = (tcConnectptr.p->transid[0] ^ tcConnectptr.p->tcOprec) & 1023;
+ nextHashptr.i = ctransidHash[hashIndex];
+ ctransidHash[hashIndex] = tcConnectptr.i;
+ tcConnectptr.p->prevHashRec = RNIL;
+ tcConnectptr.p->nextHashRec = nextHashptr.i;
+ if (nextHashptr.i != RNIL) {
+ jam();
+ /* ---------------------------------------------------------------------
+ * ENSURE THAT THE NEXT RECORD HAS SET PREVIOUS TO OUR RECORD
+ * IF IT EXISTS
+ * --------------------------------------------------------------------- */
+ ptrCheckGuard(nextHashptr, ctcConnectrecFileSize, tcConnectionrec);
+ nextHashptr.p->prevHashRec = tcConnectptr.i;
+ }//if
+ if (scanptr.p->scanAiLength > 0) {
+ jam();
+ tcConnectptr.p->transactionState = TcConnectionrec::WAIT_SCAN_AI;
+ return;
+ }//if
+ continueAfterReceivingAllAiLab(signal);
+ return;
+
+error_handler2:
+ // no scan number allocated
+ releaseScanrec(signal);
+error_handler1:
+ fragptr.p->noActiveScan--;
+error_handler:
+ ref = (ScanFragRef*)&signal->theData[0];
+ tcConnectptr.p->abortState = TcConnectionrec::ABORT_ACTIVE;
+ ref->senderData = tcConnectptr.p->clientConnectrec;
+ ref->transId1 = transid1;
+ ref->transId2 = transid2;
+ ref->errorCode = errorCode;
+ sendSignal(tcConnectptr.p->clientBlockref, GSN_SCAN_FRAGREF, signal,
+ ScanFragRef::SignalLength, JBB);
+ releaseOprec(signal);
+ releaseTcrec(signal, tcConnectptr);
+ return;
+
+ error_handler_early_1:
+ if(tabptr.p->tableStatus == Tablerec::NOT_DEFINED){
+ jam();
+ errorCode = ZTABLE_NOT_DEFINED;
+ } else if (tabptr.p->tableStatus == Tablerec::PREP_DROP_TABLE_ONGOING ||
+ tabptr.p->tableStatus == Tablerec::PREP_DROP_TABLE_DONE){
+ jam();
+ errorCode = ZDROP_TABLE_IN_PROGRESS;
+ } else {
+ ndbrequire(0);
+ }
+ error_handler_early:
+ ref = (ScanFragRef*)&signal->theData[0];
+ ref->senderData = senderData;
+ ref->transId1 = transid1;
+ ref->transId2 = transid2;
+ ref->errorCode = errorCode;
+ sendSignal(signal->senderBlockRef(), GSN_SCAN_FRAGREF, signal,
+ ScanFragRef::SignalLength, JBB);
+}//Dblqh::execSCAN_FRAGREQ()
+
+void Dblqh::continueAfterReceivingAllAiLab(Signal* signal)
+{
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED;
+ scanptr.p->scanState = ScanRecord::WAIT_ACC_SCAN;
+ AccScanReq * req = (AccScanReq*)&signal->theData[0];
+ req->senderData = scanptr.i;
+ req->senderRef = cownref;
+ req->tableId = tcConnectptr.p->tableref;
+ req->fragmentNo = tcConnectptr.p->fragmentid;
+ req->requestInfo = 0;
+ AccScanReq::setLockMode(req->requestInfo, scanptr.p->scanLockMode);
+ AccScanReq::setKeyinfoFlag(req->requestInfo, scanptr.p->scanKeyinfoFlag);
+ AccScanReq::setReadCommittedFlag(req->requestInfo, scanptr.p->readCommitted);
+ req->transId1 = tcConnectptr.p->transid[0];
+ req->transId2 = tcConnectptr.p->transid[1];
+ req->savePointId = tcConnectptr.p->savePointId;
+ // always use if-stmt to switch (instead of setting a "scan block ref")
+ if (! scanptr.p->rangeScan)
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_ACC_SCANREQ, signal,
+ AccScanReq::SignalLength, JBB);
+ else
+ sendSignal(tcConnectptr.p->tcTuxBlockref, GSN_ACC_SCANREQ, signal,
+ AccScanReq::SignalLength, JBB);
+}//Dblqh::continueAfterReceivingAllAiLab()
+
+void Dblqh::scanAttrinfoLab(Signal* signal, Uint32* dataPtr, Uint32 length)
+{
+ if (saveTupattrbuf(signal, dataPtr, length) == ZOK) {
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ if (tcConnectptr.p->currTupAiLen < scanptr.p->scanAiLength) {
+ jam();
+ } else {
+ jam();
+ ndbrequire(tcConnectptr.p->currTupAiLen == scanptr.p->scanAiLength);
+ continueAfterReceivingAllAiLab(signal);
+ }//if
+ return;
+ }//if
+ terrorCode = ZGET_ATTRINBUF_ERROR;
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ finishScanrec(signal);
+ releaseScanrec(signal);
+ fragptr.p->noActiveScan--;
+ tcConnectptr.p->transactionState = TcConnectionrec::IDLE;
+ sendScanFragRefLateLab(signal);
+}//Dblqh::scanAttrinfoLab()
+
+/*---------------------------------------------------------------------*/
+/* Send this 'I am alive' signal to TC when it is received from ACC */
+/* We include the scanPtr.i that comes from ACC in signalData[1], this */
+/* tells TC which fragment record to check for a timeout. */
+/*---------------------------------------------------------------------*/
+void Dblqh::execSCAN_HBREP(Signal* signal)
+{
+ jamEntry();
+ scanptr.i = signal->theData[0];
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ switch(scanptr.p->scanType){
+ case ScanRecord::SCAN:
+ if (scanptr.p->scanTcWaiting == ZTRUE) {
+ jam();
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ const Uint32 transid1 = signal->theData[1];
+ const Uint32 transid2 = signal->theData[2];
+ ndbrequire(transid1 == tcConnectptr.p->transid[0] &&
+ transid2 == tcConnectptr.p->transid[1]);
+
+ // Update counter on tcConnectPtr
+ if (tcConnectptr.p->tcTimer != 0){
+ tcConnectptr.p->tcTimer = cLqhTimeOutCount;
+ } else {
+ jam();
+ //ndbout << "SCAN_HBREP when tcTimer was off" << endl;
+ }
+
+ signal->theData[0] = tcConnectptr.p->clientConnectrec;
+ signal->theData[1] = tcConnectptr.p->transid[0];
+ signal->theData[2] = tcConnectptr.p->transid[1];
+ sendSignal(tcConnectptr.p->clientBlockref,
+ GSN_SCAN_HBREP, signal, 3, JBB);
+ }//if
+ break;
+ case ScanRecord::COPY:
+ // ndbout << "Dblqh::execSCAN_HBREP Dropping SCAN_HBREP" << endl;
+ break;
+ default:
+ ndbrequire(false);
+ }
+}
+
+void Dblqh::sendScanFragRefLateLab(Signal* signal)
+{
+ tcConnectptr.p->abortState = TcConnectionrec::ABORT_ACTIVE;
+ ScanFragRef * ref = (ScanFragRef*)&signal->theData[0];
+ ref->senderData = tcConnectptr.p->clientConnectrec;
+ ref->transId1 = tcConnectptr.p->transid[0];
+ ref->transId2 = tcConnectptr.p->transid[1];
+ ref->errorCode = terrorCode;
+ sendSignal(tcConnectptr.p->clientBlockref, GSN_SCAN_FRAGREF, signal,
+ ScanFragRef::SignalLength, JBB);
+ deleteTransidHash(signal);
+ releaseOprec(signal);
+ releaseTcrec(signal, tcConnectptr);
+}//Dblqh::sendScanFragRefLateLab()
+
+
+void Dblqh::accScanConfScanLab(Signal* signal)
+{
+ AccScanConf * const accScanConf = (AccScanConf *)&signal->theData[0];
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ /* -----------------------------------------------------------------------
+ * PRECONDITION: SCAN_STATE = WAIT_ACC_SCAN
+ * ----------------------------------------------------------------------- */
+ if (accScanConf->flag == AccScanConf::ZEMPTY_FRAGMENT) {
+ jam();
+ /* ---------------------------------------------------------------------
+ * THE FRAGMENT WAS EMPTY.
+ * REPORT SUCCESSFUL COPYING.
+ * --------------------------------------------------------------------- */
+ tupScanCloseConfLab(signal);
+ return;
+ }//if
+ scanptr.p->scanAccPtr = accScanConf->accPtr;
+ AttrbufPtr regAttrinbufptr;
+ regAttrinbufptr.i = tcConnectptr.p->firstAttrinbuf;
+ Uint32 boundAiLength = 0;
+ if (scanptr.p->rangeScan) {
+ jam();
+ // bound info length is in first of the 5 header words
+ ptrCheckGuard(regAttrinbufptr, cattrinbufFileSize, attrbuf);
+ boundAiLength = regAttrinbufptr.p->attrbuf[0];
+ TuxBoundInfo* const req = (TuxBoundInfo*)signal->getDataPtrSend();
+ req->errorCode = RNIL;
+ req->tuxScanPtrI = scanptr.p->scanAccPtr;
+ req->boundAiLength = boundAiLength;
+ Uint32* out = (Uint32*)req + TuxBoundInfo::SignalLength;
+ Uint32 sz = 0;
+ while (sz < boundAiLength) {
+ jam();
+ ptrCheckGuard(regAttrinbufptr, cattrinbufFileSize, attrbuf);
+ Uint32 dataLen = regAttrinbufptr.p->attrbuf[ZINBUF_DATA_LEN];
+ MEMCOPY_NO_WORDS(&out[sz],
+ &regAttrinbufptr.p->attrbuf[0],
+ dataLen);
+ sz += dataLen;
+ regAttrinbufptr.i = regAttrinbufptr.p->attrbuf[ZINBUF_NEXT];
+ ptrCheckGuard(regAttrinbufptr, cattrinbufFileSize, attrbuf);
+ }
+ ndbrequire(sz == boundAiLength);
+ EXECUTE_DIRECT(DBTUX, GSN_TUX_BOUND_INFO,
+ signal, TuxBoundInfo::SignalLength + boundAiLength);
+ if (req->errorCode != 0) {
+ jam();
+ /*
+ * Cannot use STORED_PROCREF to abort since even the REF
+ * returns a stored proc id. So record error and continue.
+ * The scan is already Invalid in TUX and returns empty set.
+ */
+ tcConnectptr.p->errorCode = req->errorCode;
+ }
+ }
+ scanptr.p->scanState = ScanRecord::WAIT_STORED_PROC_SCAN;
+ signal->theData[0] = tcConnectptr.p->tupConnectrec;
+ signal->theData[1] = tcConnectptr.p->tableref;
+ signal->theData[2] = scanptr.p->scanSchemaVersion;
+ signal->theData[3] = ZSTORED_PROC_SCAN;
+ ndbrequire(boundAiLength <= scanptr.p->scanAiLength);
+ signal->theData[4] = scanptr.p->scanAiLength - boundAiLength;
+ sendSignal(tcConnectptr.p->tcTupBlockref,
+ GSN_STORED_PROCREQ, signal, 5, JBB);
+
+ signal->theData[0] = tcConnectptr.p->tupConnectrec;
+ while (regAttrinbufptr.i != RNIL) {
+ ptrCheckGuard(regAttrinbufptr, cattrinbufFileSize, attrbuf);
+ jam();
+ Uint32 dataLen = regAttrinbufptr.p->attrbuf[ZINBUF_DATA_LEN];
+ ndbrequire(dataLen != 0);
+ // first 3 words already set in STORED_PROCREQ
+ MEMCOPY_NO_WORDS(&signal->theData[3],
+ &regAttrinbufptr.p->attrbuf[0],
+ dataLen);
+ sendSignal(tcConnectptr.p->tcTupBlockref,
+ GSN_ATTRINFO, signal, dataLen + 3, JBB);
+ regAttrinbufptr.i = regAttrinbufptr.p->attrbuf[ZINBUF_NEXT];
+ }//while
+ releaseOprec(signal);
+}//Dblqh::accScanConfScanLab()
+
+/* -------------------------------------------------------------------------
+ * ENTER STORED_PROCCONF WITH
+ * TC_CONNECTPTR,
+ * TSTORED_PROC_ID
+ * -------------------------------------------------------------------------
+ * PRECONDITION: SCAN_STATE = WAIT_STORED_PROC_SCAN
+ * ------------------------------------------------------------------------- */
+void Dblqh::storedProcConfScanLab(Signal* signal)
+{
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ jam();
+ // STOP THE SCAN PROCESS IF THIS HAS BEEN REQUESTED.
+ closeScanLab(signal);
+ return;
+ }//if
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ jam();
+ linkActiveFrag(signal);
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_FIRST_STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ case Fragrecord::CRASH_RECOVERING:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ continueFirstScanAfterBlockedLab(signal);
+}//Dblqh::storedProcConfScanLab()
+
+void Dblqh::continueFirstScanAfterBlockedLab(Signal* signal)
+{
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ scanptr.p->scanState = ScanRecord::WAIT_NEXT_SCAN;
+ initScanAccOp(signal);
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ signal->theData[1] = RNIL;
+ signal->theData[2] = NextScanReq::ZSCAN_NEXT;
+ if (! scanptr.p->rangeScan)
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+ else
+ sendSignal(tcConnectptr.p->tcTuxBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+ return;
+}//Dblqh::continueFirstScanAfterBlockedLab()
+
+/* -------------------------------------------------------------------------
+ * When executing a scan we must come up to the surface at times to make
+ * sure we can quickly start local checkpoints.
+ * ------------------------------------------------------------------------- */
+void Dblqh::execCHECK_LCP_STOP(Signal* signal)
+{
+ jamEntry();
+ scanptr.i = signal->theData[0];
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (signal->theData[1] == ZTRUE) {
+ jam();
+ releaseActiveFrag(signal);
+ signal->theData[0] = ZCHECK_LCP_STOP_BLOCKED;
+ signal->theData[1] = scanptr.i;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 10, 2);
+ signal->theData[0] = RNIL;
+ return;
+ }//if
+ if (fragptr.p->fragStatus != Fragrecord::FSACTIVE) {
+ ndbrequire(fragptr.p->fragStatus == Fragrecord::BLOCKED);
+ releaseActiveFrag(signal);
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_CHECK_STOPPED;
+ signal->theData[0] = RNIL;
+ }//if
+}//Dblqh::execCHECK_LCP_STOP()
+
+void Dblqh::checkLcpStopBlockedLab(Signal* signal)
+{
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ jam();
+ linkActiveFrag(signal);
+ continueAfterCheckLcpStopBlocked(signal);
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_CHECK_STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ case Fragrecord::CRASH_RECOVERING:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ ndbrequire(false);
+ }//switch
+}//Dblqh::checkLcpStopBlockedLab()
+
+void Dblqh::continueAfterCheckLcpStopBlocked(Signal* signal)
+{
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ signal->theData[1] = AccCheckScan::ZNOT_CHECK_LCP_STOP;
+ if (! scanptr.p->rangeScan)
+ EXECUTE_DIRECT(DBACC, GSN_ACC_CHECK_SCAN, signal, 2);
+ else
+ EXECUTE_DIRECT(DBTUX, GSN_ACC_CHECK_SCAN, signal, 2);
+}//Dblqh::continueAfterCheckLcpStopBlocked()
+
+/* -------------------------------------------------------------------------
+ * ENTER NEXT_SCANCONF
+ * -------------------------------------------------------------------------
+ * PRECONDITION: SCAN_STATE = WAIT_NEXT_SCAN
+ * ------------------------------------------------------------------------- */
+void Dblqh::nextScanConfScanLab(Signal* signal)
+{
+ NextScanConf * const nextScanConf = (NextScanConf *)&signal->theData[0];
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ if (nextScanConf->fragId == RNIL) {
+ jam();
+ /* ---------------------------------------------------------------------
+ * THERE ARE NO MORE TUPLES TO FETCH. IF WE HAVE ANY
+ * OPERATIONS STILL NEEDING A LOCK WE REPORT TO THE
+ * APPLICATION AND CLOSE THE SCAN WHEN THE NEXT SCAN
+ * REQUEST IS RECEIVED. IF WE DO NOT HAVE ANY NEED FOR
+ * LOCKS WE CAN CLOSE THE SCAN IMMEDIATELY.
+ * --------------------------------------------------------------------- */
+ releaseActiveFrag(signal);
+ /*************************************************************
+ * STOP THE SCAN PROCESS IF THIS HAS BEEN REQUESTED.
+ ************************************************************ */
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ if ((scanptr.p->scanLockHold == ZTRUE) &&
+ (scanptr.p->scanCompletedOperations > 0)) {
+ jam();
+ scanptr.p->scanReleaseCounter = 1;
+ scanReleaseLocksLab(signal);
+ return;
+ }//if
+ jam();
+ closeScanLab(signal);
+ return;
+ }//if
+
+ if (scanptr.p->scanCompletedOperations > 0) {
+ jam();
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ scanptr.p->scanState = ScanRecord::WAIT_SCAN_NEXTREQ;
+ sendScanFragConf(signal, ZFALSE);
+ return;
+ }//if
+ closeScanLab(signal);
+ return;
+ }//if
+
+ // If accOperationPtr == RNIL no record was returned by ACC
+ if (nextScanConf->accOperationPtr == RNIL) {
+ jam();
+ /*************************************************************
+ * STOP THE SCAN PROCESS IF THIS HAS BEEN REQUESTED.
+ ************************************************************ */
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ releaseActiveFrag(signal);
+ if ((scanptr.p->scanLockHold == ZTRUE) &&
+ (scanptr.p->scanCompletedOperations > 0)) {
+ jam();
+ scanptr.p->scanReleaseCounter = 1;
+ scanReleaseLocksLab(signal);
+ return;
+ }//if
+ jam();
+ closeScanLab(signal);
+ return;
+ }//if
+
+ if (scanptr.p->scanCompletedOperations > 0) {
+ jam();
+ releaseActiveFrag(signal);
+ scanptr.p->scanState = ScanRecord::WAIT_SCAN_NEXTREQ;
+ sendScanFragConf(signal, ZFALSE);
+ return;
+ }//if
+
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ signal->theData[1] = AccCheckScan::ZCHECK_LCP_STOP;
+ if (! scanptr.p->rangeScan)
+ sendSignal(tcConnectptr.p->tcAccBlockref,
+ GSN_ACC_CHECK_SCAN, signal, 2, JBB);
+ else
+ sendSignal(tcConnectptr.p->tcTuxBlockref,
+ GSN_ACC_CHECK_SCAN, signal, 2, JBB);
+ return;
+ }//if
+
+ ndbrequire(scanptr.p->scanCompletedOperations < MAX_PARALLEL_OP_PER_SCAN);
+ scanptr.p->scanAccOpPtr[scanptr.p->scanCompletedOperations] =
+ nextScanConf->accOperationPtr;
+ scanptr.p->scanLocalref[0] = nextScanConf->localKey[0];
+ scanptr.p->scanLocalref[1] = nextScanConf->localKey[1];
+ scanptr.p->scanLocalFragid = nextScanConf->fragId;
+ if (scanptr.p->scanKeyinfoFlag) {
+ jam();
+ tcConnectptr.p->primKeyLen = nextScanConf->keyLength;
+ seizeTupkeybuf(signal);
+ databufptr.p->data[0] = nextScanConf->key[0];
+ databufptr.p->data[1] = nextScanConf->key[1];
+ databufptr.p->data[2] = nextScanConf->key[2];
+ databufptr.p->data[3] = nextScanConf->key[3];
+ if (nextScanConf->keyLength > 4) {
+ jam();
+ tcConnectptr.p->save1 = 4;
+ scanptr.p->scanState = ScanRecord::WAIT_SCAN_KEYINFO;
+ return;
+ }//if
+ }//if
+ nextScanConfLoopLab(signal);
+}//Dblqh::nextScanConfScanLab()
+
+void Dblqh::nextScanConfLoopLab(Signal* signal)
+{
+ /* ----------------------------------------------------------------------
+ * STOP THE SCAN PROCESS IF THIS HAS BEEN REQUESTED.
+ * ---------------------------------------------------------------------- */
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ jam();
+ releaseActiveFrag(signal);
+ releaseOprec(signal);
+ if ((scanptr.p->scanLockHold == ZTRUE) &&
+ (scanptr.p->scanCompletedOperations > 0)) {
+ jam();
+ scanptr.p->scanReleaseCounter = 1;
+ scanReleaseLocksLab(signal);
+ return;
+ }//if
+ closeScanLab(signal);
+ return;
+ }//if
+
+ Uint32 tableRef;
+ Uint32 tupFragPtr;
+ Uint32 reqinfo = (scanptr.p->scanLockHold == ZFALSE);
+ reqinfo = reqinfo + (tcConnectptr.p->operation << 6);
+ reqinfo = reqinfo + (tcConnectptr.p->opExec << 10);
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_TUPKEY;
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (! scanptr.p->rangeScan) {
+ tableRef = tcConnectptr.p->tableref;
+ if (fragptr.p->fragId == scanptr.p->scanLocalFragid) {
+ jam();
+ tupFragPtr = fragptr.p->tupFragptr[0];
+ } else {
+ jam();
+ tupFragPtr = fragptr.p->tupFragptr[1];
+ }//if
+ } else {
+ jam();
+ // for ordered index use primary table
+ FragrecordPtr tFragPtr;
+ tFragPtr.i = fragptr.p->tableFragptr;
+ ptrCheckGuard(tFragPtr, cfragrecFileSize, fragrecord);
+ tableRef = tFragPtr.p->tabRef;
+ if (tFragPtr.p->fragId == scanptr.p->scanLocalFragid) {
+ jam();
+ tupFragPtr = tFragPtr.p->tupFragptr[0];
+ } else {
+ jam();
+ tupFragPtr = tFragPtr.p->tupFragptr[1];
+ }//if
+ }
+ {
+ TupKeyReq * const tupKeyReq = (TupKeyReq *)signal->getDataPtrSend();
+
+ tupKeyReq->connectPtr = tcConnectptr.p->tupConnectrec;
+ tupKeyReq->request = reqinfo;
+ tupKeyReq->tableRef = tableRef;
+ tupKeyReq->fragId = scanptr.p->scanLocalFragid;
+ tupKeyReq->keyRef1 = scanptr.p->scanLocalref[0];
+ tupKeyReq->keyRef2 = scanptr.p->scanLocalref[1];
+ tupKeyReq->attrBufLen = 0;
+ ndbrequire(scanptr.p->scanCompletedOperations < MAX_PARALLEL_OP_PER_SCAN);
+ tupKeyReq->opRef =
+ scanptr.p->scanApiOpPtr[scanptr.p->scanCompletedOperations];
+ tupKeyReq->applRef = scanptr.p->scanApiBlockref;
+ tupKeyReq->schemaVersion = scanptr.p->scanSchemaVersion;
+ tupKeyReq->storedProcedure = scanptr.p->scanStoredProcId;
+ tupKeyReq->transId1 = tcConnectptr.p->transid[0];
+ tupKeyReq->transId2 = tcConnectptr.p->transid[1];
+ tupKeyReq->fragPtr = tupFragPtr;
+ tupKeyReq->primaryReplica = (tcConnectptr.p->seqNoReplica == 0)?true:false;
+ tupKeyReq->coordinatorTC = tcConnectptr.p->tcBlockref;
+ tupKeyReq->tcOpIndex = tcConnectptr.p->tcOprec;
+ tupKeyReq->savePointId = tcConnectptr.p->savePointId;
+ Uint32 blockNo = refToBlock(tcConnectptr.p->tcTupBlockref);
+ EXECUTE_DIRECT(blockNo, GSN_TUPKEYREQ, signal,
+ TupKeyReq::SignalLength);
+ }
+}//Dblqh::nextScanConfLoopLab()
+
+/* -------------------------------------------------------------------------
+ * RECEPTION OF FURTHER KEY INFORMATION WHEN KEY SIZE > 16 BYTES.
+ * -------------------------------------------------------------------------
+ * PRECONDITION: SCAN_STATE = WAIT_SCAN_KEYINFO
+ * ------------------------------------------------------------------------- */
+bool Dblqh::keyinfoLab(Signal* signal, Uint32* dataPtr, Uint32 length)
+{
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ Uint32 index = 0;
+ do {
+ jam();
+ seizeTupkeybuf(signal);
+ databufptr.p->data[0] = dataPtr[index];
+ databufptr.p->data[1] = dataPtr[index + 1];
+ databufptr.p->data[2] = dataPtr[index + 2];
+ databufptr.p->data[3] = dataPtr[index + 3];
+ index += 4;
+ tcConnectptr.p->save1 = tcConnectptr.p->save1 + 4;
+ if (tcConnectptr.p->save1 >= tcConnectptr.p->primKeyLen) {
+ jam();
+ return true;
+ }//if
+ if (index >= length) {
+ jam();
+ return false;
+ }//if
+ } while (index < 20);
+ ndbrequire(false);
+ return false;
+}//Dblqh::keyinfoLab()
+
+/* -------------------------------------------------------------------------
+ * ENTER TUPKEYCONF
+ * -------------------------------------------------------------------------
+ * PRECONDITION: TRANSACTION_STATE = SCAN_TUPKEY
+ * ------------------------------------------------------------------------- */
+void Dblqh::scanTupkeyConfLab(Signal* signal)
+{
+ UintR tdata3;
+ UintR tdata4;
+ UintR tdata5;
+
+ tdata3 = signal->theData[2];
+ tdata4 = signal->theData[3];
+ tdata5 = signal->theData[4];
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED;
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ releaseActiveFrag(signal);
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ /* ---------------------------------------------------------------------
+ * STOP THE SCAN PROCESS IF THIS HAS BEEN REQUESTED.
+ * --------------------------------------------------------------------- */
+ releaseOprec(signal);
+ if ((scanptr.p->scanLockHold == ZTRUE) &&
+ (scanptr.p->scanCompletedOperations > 0)) {
+ jam();
+ scanptr.p->scanReleaseCounter = 1;
+ scanReleaseLocksLab(signal);
+ return;
+ }//if
+ jam();
+ closeScanLab(signal);
+ return;
+ }//if
+ if (scanptr.p->scanKeyinfoFlag) {
+ jam();
+ DatabufPtr TdataBuf;
+ TdataBuf.i = tcConnectptr.p->firstTupkeybuf;
+ const Uint32 keyLen = tcConnectptr.p->primKeyLen;
+ const Uint32 dataBufSz = cdatabufFileSize;
+
+ /**
+ * Note that this code requires signal->theData to be big enough for
+ * a entire key
+ */
+ ndbrequire(keyLen * 4 <= sizeof(signal->theData));
+ KeyInfo20 * keyInfo = (KeyInfo20*)&signal->theData[0];
+ for(Uint32 i = 0; i < keyLen; i += 4){
+ ptrCheckGuard(TdataBuf, dataBufSz, databuf);
+ keyInfo->keyData[i + 0] = TdataBuf.p->data[0];
+ keyInfo->keyData[i + 1] = TdataBuf.p->data[1];
+ keyInfo->keyData[i + 2] = TdataBuf.p->data[2];
+ keyInfo->keyData[i + 3] = TdataBuf.p->data[3];
+ TdataBuf.i = TdataBuf.p->nextDatabuf;
+ }
+ sendKeyinfo20(signal, scanptr.p, tcConnectptr.p);
+ releaseOprec(signal);
+ }//if
+ ndbrequire(scanptr.p->scanCompletedOperations < MAX_PARALLEL_OP_PER_SCAN);
+ scanptr.p->scanOpLength[scanptr.p->scanCompletedOperations] = tdata4;
+ scanptr.p->scanCompletedOperations++;
+ if ((scanptr.p->scanCompletedOperations ==
+ scanptr.p->scanConcurrentOperations) &&
+ (scanptr.p->scanLockHold == ZTRUE)) {
+ jam();
+ scanptr.p->scanState = ScanRecord::WAIT_SCAN_NEXTREQ;
+ sendScanFragConf(signal, ZFALSE);
+ return;
+ } else if (scanptr.p->scanCompletedOperations ==
+ scanptr.p->scanConcurrentOperations) {
+ jam();
+ scanptr.p->scanReleaseCounter = scanptr.p->scanCompletedOperations;
+ scanReleaseLocksLab(signal);
+ return;
+ } else if (scanptr.p->scanLockHold == ZTRUE) {
+ jam();
+ scanptr.p->scanFlag = NextScanReq::ZSCAN_NEXT;
+ } else {
+ jam();
+ scanptr.p->scanFlag = NextScanReq::ZSCAN_NEXT_COMMIT;
+ }//if
+ scanNextLoopLab(signal);
+}//Dblqh::scanTupkeyConfLab()
+
+void Dblqh::scanNextLoopLab(Signal* signal)
+{
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ jam();
+ linkActiveFrag(signal);
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ case Fragrecord::CRASH_RECOVERING:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ ndbrequire(false);
+ }//switch
+ continueScanAfterBlockedLab(signal);
+}//Dblqh::scanNextLoopLab()
+
+void Dblqh::continueScanAfterBlockedLab(Signal* signal)
+{
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ Uint32 accOpPtr;
+ if (scanptr.p->scanFlag == NextScanReq::ZSCAN_NEXT_ABORT) {
+ jam();
+ scanptr.p->scanFlag = NextScanReq::ZSCAN_NEXT_COMMIT;
+ accOpPtr = scanptr.p->scanAccOpPtr[scanptr.p->scanCompletedOperations];
+ } else if (scanptr.p->scanFlag == NextScanReq::ZSCAN_NEXT_COMMIT) {
+ jam();
+ accOpPtr = scanptr.p->scanAccOpPtr[scanptr.p->scanCompletedOperations - 1];
+ } else {
+ jam();
+ accOpPtr = RNIL; // The value is not used in ACC
+ }//if
+ scanptr.p->scanState = ScanRecord::WAIT_NEXT_SCAN;
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ signal->theData[1] = accOpPtr;
+ signal->theData[2] = scanptr.p->scanFlag;
+ if (! scanptr.p->rangeScan)
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+ else
+ sendSignal(tcConnectptr.p->tcTuxBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+}//Dblqh::continueScanAfterBlockedLab()
+
+/* -------------------------------------------------------------------------
+ * ENTER TUPKEYREF WITH
+ * TC_CONNECTPTR,
+ * TERROR_CODE
+ * -------------------------------------------------------------------------
+ * PRECONDITION: TRANSACTION_STATE = SCAN_TUPKEY
+ * ------------------------------------------------------------------------- */
+void Dblqh::scanTupkeyRefLab(Signal* signal)
+{
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED;
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ releaseActiveFrag(signal);
+ releaseOprec(signal);
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ /* ---------------------------------------------------------------------
+ * STOP THE SCAN PROCESS IF THIS HAS BEEN REQUESTED.
+ * --------------------------------------------------------------------- */
+ if ((scanptr.p->scanLockHold == ZTRUE) &&
+ (scanptr.p->scanCompletedOperations > 0)) {
+ jam();
+ scanptr.p->scanReleaseCounter = 1;
+ scanReleaseLocksLab(signal);
+ return;
+ }//if
+ jam();
+ closeScanLab(signal);
+ return;
+ }//if
+ if ((terrorCode != ZSEARCH_CONDITION_FALSE) &&
+ (terrorCode != ZNO_TUPLE_FOUND) &&
+ (terrorCode >= ZUSER_ERROR_CODE_LIMIT)) {
+ scanptr.p->scanErrorCounter++;
+ tcConnectptr.p->errorCode = terrorCode;
+
+ if (scanptr.p->scanLockHold == ZTRUE) {
+ jam();
+ scanptr.p->scanReleaseCounter = 1;
+ } else {
+ jam();
+ scanptr.p->scanCompletedOperations++;
+ scanptr.p->scanReleaseCounter = scanptr.p->scanCompletedOperations;
+ }//if
+ /* --------------------------------------------------------------------
+ * WE NEED TO RELEASE ALL LOCKS CURRENTLY
+ * HELD BY THIS SCAN.
+ * -------------------------------------------------------------------- */
+ scanReleaseLocksLab(signal);
+ return;
+ }//if
+ /* -----------------------------------------------------------------------
+ * WE NEED TO ENSURE THAT WE DO NOT SEARCH FOR THE NEXT TUPLE FOR A
+ * LONG TIME WHILE WE KEEP A LOCK ON A FOUND TUPLE. WE RATHER REPORT
+ * THE FOUND TUPLE IF FOUND TUPLES ARE RARE. WE SELECT 20 TUPLES
+ * WHICH SHOULD BE ROUGHLY 10 MS OF LOCK HOLD TIME.
+ * ----------------------------------------------------------------------- */
+ scanptr.p->scanSearchCondFalseCount++;
+#if 0
+ // MASV Uncomment this feature since it forgets
+ // to release on operation record in DBACC
+ // This is the quick fix and should be changed in
+ // the future
+ if (scanptr.p->scanSearchCondFalseCount > 20) {
+ if (scanptr.p->scanCompletedOperations > 0) {
+ jam();
+ scanptr.p->scanState = ScanRecord::WAIT_SCAN_NEXTREQ;
+ sendScanFragConf(signal, ZFALSE);
+ return;
+ }//if
+ }//if
+#endif
+
+ scanptr.p->scanFlag = NextScanReq::ZSCAN_NEXT_ABORT;
+ scanNextLoopLab(signal);
+}//Dblqh::scanTupkeyRefLab()
+
+/* -------------------------------------------------------------------------
+ * THE SCAN HAS BEEN COMPLETED. EITHER BY REACHING THE END OR BY COMMAND
+ * FROM THE APPLICATION OR BY SOME SORT OF ERROR CONDITION.
+ * ------------------------------------------------------------------------- */
+void Dblqh::closeScanLab(Signal* signal)
+{
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ jam();
+ linkActiveFrag(signal);
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_CLOSE_STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ case Fragrecord::CRASH_RECOVERING:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ ndbrequire(false);
+ }//switch
+ continueCloseScanAfterBlockedLab(signal);
+}//Dblqh::closeScanLab()
+
+void Dblqh::continueCloseScanAfterBlockedLab(Signal* signal)
+{
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED;
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ scanptr.p->scanState = ScanRecord::WAIT_CLOSE_SCAN;
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ signal->theData[1] = RNIL;
+ signal->theData[2] = NextScanReq::ZSCAN_CLOSE;
+ if (! scanptr.p->rangeScan)
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+ else
+ sendSignal(tcConnectptr.p->tcTuxBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+}//Dblqh::continueCloseScanAfterBlockedLab()
+
+/* -------------------------------------------------------------------------
+ * ENTER NEXT_SCANCONF
+ * -------------------------------------------------------------------------
+ * PRECONDITION: SCAN_STATE = WAIT_CLOSE_SCAN
+ * ------------------------------------------------------------------------- */
+void Dblqh::accScanCloseConfLab(Signal* signal)
+{
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ scanptr.p->scanState = ScanRecord::WAIT_DELETE_STORED_PROC_ID_SCAN;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ signal->theData[0] = tcConnectptr.p->tupConnectrec;
+ signal->theData[1] = tcConnectptr.p->tableref;
+ signal->theData[2] = scanptr.p->scanSchemaVersion;
+ signal->theData[3] = ZDELETE_STORED_PROC_ID;
+ signal->theData[4] = scanptr.p->scanStoredProcId;
+ sendSignal(tcConnectptr.p->tcTupBlockref,
+ GSN_STORED_PROCREQ, signal, 5, JBB);
+}//Dblqh::accScanCloseConfLab()
+
+/* -------------------------------------------------------------------------
+ * ENTER STORED_PROCCONF WITH
+ * -------------------------------------------------------------------------
+ * PRECONDITION: SCAN_STATE = WAIT_DELETE_STORED_PROC_ID_SCAN
+ * ------------------------------------------------------------------------- */
+void Dblqh::tupScanCloseConfLab(Signal* signal)
+{
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (tcConnectptr.p->abortState == TcConnectionrec::NEW_FROM_TC) {
+ jam();
+ tcNodeFailptr.i = tcConnectptr.p->tcNodeFailrec;
+ ptrCheckGuard(tcNodeFailptr, ctcNodeFailrecFileSize, tcNodeFailRecord);
+ tcNodeFailptr.p->tcRecNow = tcConnectptr.i + 1;
+ signal->theData[0] = ZLQH_TRANS_NEXT;
+ signal->theData[1] = tcNodeFailptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ } else if (tcConnectptr.p->errorCode != 0) {
+ jam();
+ ScanFragRef * ref = (ScanFragRef*)&signal->theData[0];
+ ref->senderData = tcConnectptr.p->clientConnectrec;
+ ref->transId1 = tcConnectptr.p->transid[0];
+ ref->transId2 = tcConnectptr.p->transid[1];
+ ref->errorCode = tcConnectptr.p->errorCode;
+ sendSignal(tcConnectptr.p->clientBlockref, GSN_SCAN_FRAGREF, signal,
+ ScanFragRef::SignalLength, JBB);
+ } else {
+ jam();
+ scanptr.p->scanCompletedOperations = 0;
+ sendScanFragConf(signal, ZSCAN_FRAG_CLOSED);
+ }//if
+ finishScanrec(signal);
+ releaseScanrec(signal);
+ tcConnectptr.p->tcScanRec = RNIL;
+ fragptr.p->noActiveScan = fragptr.p->noActiveScan - 1;
+ deleteTransidHash(signal);
+ releaseOprec(signal);
+ releaseTcrec(signal, tcConnectptr);
+}//Dblqh::tupScanCloseConfLab()
+
+/* =========================================================================
+ * ======= INITIATE SCAN_ACC_OP_PTR TO RNIL IN SCAN RECORD =======
+ *
+ * SUBROUTINE SHORT NAME = ISA
+ * ========================================================================= */
+void Dblqh::initScanAccOp(Signal* signal)
+{
+ UintR tisaIndex;
+
+ for (tisaIndex = 0; tisaIndex < MAX_PARALLEL_OP_PER_SCAN; tisaIndex++) {
+ scanptr.p->scanAccOpPtr[tisaIndex] = RNIL;
+ }//for
+}//Dblqh::initScanAccOp()
+
+/* =========================================================================
+ * ======= INITIATE SCAN RECORD =======
+ *
+ * SUBROUTINE SHORT NAME = ISC
+ * ========================================================================= */
+Uint32 Dblqh::initScanrec(const ScanFragReq* scanFragReq)
+{
+ const Uint32 reqinfo = scanFragReq->requestInfo;
+ const Uint32 scanConcurrentOperations = ScanFragReq::getConcurrency(reqinfo);
+ const Uint32 scanLockMode = ScanFragReq::getLockMode(reqinfo);
+ const Uint32 scanLockHold = ScanFragReq::getHoldLockFlag(reqinfo);
+ const Uint32 keyinfo = ScanFragReq::getKeyinfoFlag(reqinfo);
+ const Uint32 readCommitted = ScanFragReq::getReadCommittedFlag(reqinfo);
+ const Uint32 rangeScan = ScanFragReq::getRangeScanFlag(reqinfo);
+ const Uint32 attrLen = ScanFragReq::getAttrLen(reqinfo);
+
+ scanptr.p->scanKeyinfoFlag = keyinfo;
+ scanptr.p->scanLockHold = scanLockHold;
+ scanptr.p->scanCompletedStatus = ZFALSE;
+ scanptr.p->scanType = ScanRecord::SCAN;
+ scanptr.p->scanApiBlockref = scanFragReq->resultRef;
+ scanptr.p->scanAiLength = attrLen;
+ scanptr.p->scanTcrec = tcConnectptr.i;
+ scanptr.p->scanSchemaVersion = scanFragReq->schemaVersion;
+ scanptr.p->scanCompletedOperations = 0;
+ scanptr.p->scanConcurrentOperations = scanConcurrentOperations;
+ scanptr.p->scanErrorCounter = 0;
+ scanptr.p->scanLockMode = scanLockMode;
+ scanptr.p->readCommitted = readCommitted;
+ scanptr.p->rangeScan = rangeScan;
+ scanptr.p->scanSearchCondFalseCount = 0;
+ scanptr.p->scanState = ScanRecord::SCAN_FREE;
+ scanptr.p->scanFlag = ZFALSE;
+ scanptr.p->scanLocalref[0] = 0;
+ scanptr.p->scanLocalref[1] = 0;
+ scanptr.p->scanLocalFragid = 0;
+ scanptr.p->scanTcWaiting = ZTRUE;
+
+ for (Uint32 i = 0; i < scanConcurrentOperations; i++) {
+ jam();
+ scanptr.p->scanApiOpPtr[i] = scanFragReq->clientOpPtr[i];
+ scanptr.p->scanOpLength[i] = 0;
+ scanptr.p->scanAccOpPtr[i] = 0;
+ }//for
+ if (! rangeScan) {
+ jam();
+ for (Int32 i = NR_ScanNo - 1; i >= 0; i--) {
+ jam();
+ if (fragptr.p->fragScanRec[i] == ZNIL) {
+ jam();
+ scanptr.p->scanNumber = i;
+ fragptr.p->fragScanRec[i] = scanptr.i;
+ return ZOK;
+ }//if
+ }//for
+ } else {
+ jam();
+ // put in second half of fragScanRec of primary table fragment
+ FragrecordPtr tFragPtr;
+ tFragPtr.i = fragptr.p->tableFragptr;
+ ptrCheckGuard(tFragPtr, cfragrecFileSize, fragrecord);
+ for (Uint32 i = NR_MinRangeScanNo; i < NR_MaxRangeScanNo; i++) {
+ if (tFragPtr.p->fragScanRec[i] == ZNIL) {
+ jam();
+ scanptr.p->scanNumber = i;
+ tFragPtr.p->fragScanRec[i] = scanptr.i;
+ return ZOK;
+ }
+ }
+ }
+ return ZNO_FREE_FRAG_SCAN_REC_ERROR;
+}//Dblqh::initScanrec()
+
+/* =========================================================================
+ * ======= INITIATE TC RECORD AT SCAN =======
+ *
+ * SUBROUTINE SHORT NAME = IST
+ * ========================================================================= */
+void Dblqh::initScanTc(Signal* signal,
+ Uint32 transid1,
+ Uint32 transid2,
+ Uint32 fragId,
+ Uint32 nodeId)
+{
+ tcConnectptr.p->transid[0] = transid1;
+ tcConnectptr.p->transid[1] = transid2;
+ tcConnectptr.p->tcScanRec = scanptr.i;
+ tcConnectptr.p->tableref = tabptr.i;
+ tcConnectptr.p->fragmentid = fragId;
+ tcConnectptr.p->fragmentptr = fragptr.i;
+ tcConnectptr.p->tcOprec = tcConnectptr.p->clientConnectrec;
+ tcConnectptr.p->tcBlockref = tcConnectptr.p->clientBlockref;
+ tcConnectptr.p->errorCode = 0;
+ tcConnectptr.p->reclenAiLqhkey = 0;
+ tcConnectptr.p->abortState = TcConnectionrec::ABORT_IDLE;
+ tcConnectptr.p->nextReplica = nodeId;
+ tcConnectptr.p->currTupAiLen = 0;
+ tcConnectptr.p->opExec = 1;
+ tcConnectptr.p->operation = ZREAD;
+ tcConnectptr.p->listState = TcConnectionrec::NOT_IN_LIST;
+
+ tabptr.p->usageCount++;
+}//Dblqh::initScanTc()
+
+/* =========================================================================
+ * ======= FINISH SCAN RECORD =======
+ *
+ * REMOVE SCAN RECORD FROM PER FRAGMENT LIST.
+ * ========================================================================= */
+void Dblqh::finishScanrec(Signal* signal)
+{
+ if (! scanptr.p->rangeScan) {
+ ndbrequire(scanptr.p->scanNumber < NR_ScanNo);
+ fragptr.p->fragScanRec[scanptr.p->scanNumber] = ZNIL;
+ } else {
+ jam();
+ ndbrequire(NR_MinRangeScanNo <= scanptr.p->scanNumber && scanptr.p->scanNumber < NR_MaxRangeScanNo);
+ FragrecordPtr tFragPtr;
+ tFragPtr.i = fragptr.p->tableFragptr;
+ ptrCheckGuard(tFragPtr, cfragrecFileSize, fragrecord);
+ tFragPtr.p->fragScanRec[scanptr.p->scanNumber] = ZNIL;
+ }
+}//Dblqh::finishScanrec()
+
+/* =========================================================================
+ * ======= RELEASE SCAN RECORD =======
+ *
+ * RELEASE A SCAN RECORD TO THE FREELIST.
+ * ========================================================================= */
+void Dblqh::releaseScanrec(Signal* signal)
+{
+ scanptr.p->nextScanrec = cfirstfreeScanrec;
+ cfirstfreeScanrec = scanptr.i;
+ scanptr.p->scanState = ScanRecord::SCAN_FREE;
+ scanptr.p->scanType = ScanRecord::ST_IDLE;
+ scanptr.p->scanTcWaiting = ZFALSE;
+ cbookedAccOps -= scanptr.p->scanConcurrentOperations;
+ cscanNoFreeRec++;
+}//Dblqh::releaseScanrec()
+
+/* =========================================================================
+ * ======= SEIZE SCAN RECORD =======
+ *
+ * GETS A NEW SCAN RECORD FROM FREELIST.
+ * ========================================================================= */
+void Dblqh::seizeScanrec(Signal* signal)
+{
+ scanptr.i = cfirstfreeScanrec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ cfirstfreeScanrec = scanptr.p->nextScanrec;
+ scanptr.p->nextScanrec = RNIL;
+ cscanNoFreeRec--;
+}//Dblqh::seizeScanrec()
+
+/* ------------------------------------------------------------------------
+ * ------- SEND KEYINFO20 TO API -------
+ *
+ * ------------------------------------------------------------------------ */
+void Dblqh::sendKeyinfo20(Signal* signal,
+ ScanRecord * scanP,
+ TcConnectionrec * tcConP)
+{
+ ndbrequire(scanP->scanCompletedOperations < MAX_PARALLEL_OP_PER_SCAN);
+ KeyInfo20 * keyInfo = (KeyInfo20 *)&signal->theData[0];
+
+ const Uint32 scanOp = scanP->scanCompletedOperations;
+ keyInfo->clientOpPtr = scanP->scanApiOpPtr[scanOp];
+ keyInfo->keyLen = tcConP->primKeyLen;
+ keyInfo->scanInfo_Node = KeyInfo20::setScanInfo(scanOp,
+ scanP->scanNumber)+
+ (getOwnNodeId() << 16);
+
+ keyInfo->transId1 = tcConP->transid[0];
+ keyInfo->transId2 = tcConP->transid[1];
+
+ const BlockReference ref = scanP->scanApiBlockref;
+ const Uint32 keyLen = tcConP->primKeyLen;
+ if(refToNode(ref) == getOwnNodeId()){
+ jam();
+ EXECUTE_DIRECT(refToBlock(ref), GSN_KEYINFO20, signal, 5 + keyLen);
+ jamEntry();
+ return;
+ }
+
+ bool connectedToNode = getNodeInfo(refToNode(ref)).m_connected;
+
+ if (ERROR_INSERTED(5029)){
+ // Use error insert to turn routing on
+ jam();
+ connectedToNode = false;
+ }
+
+ if (connectedToNode){
+ jam();
+ Uint32 keyLenLeft = keyLen;
+ Uint32 keyDataIndex = 20;
+ for(; keyLenLeft > 20; keyLenLeft -= 20, keyDataIndex += 20){
+ jam();
+ sendSignal(ref, GSN_KEYINFO20, signal, 25, JBB);
+ for(Uint32 i = 0; i<20; i++)
+ keyInfo->keyData[i] = keyInfo->keyData[keyDataIndex + i];
+ }//for
+ sendSignal(ref, GSN_KEYINFO20, signal, 5 + keyLenLeft, JBB);
+ } else {
+ /**
+ * If this node does not have a direct connection
+ * to the receiving node we want to send the signals
+ * routed via the control node
+ */
+ jam();
+ Uint32 keyLenLeft = keyLen;
+ Uint32 keyDataIndex = 19;
+ BlockReference routeBlockref = tcConP->clientBlockref;
+
+ for(; keyLenLeft > 19; keyLenLeft -= 19, keyDataIndex += 19){
+ jam();
+ // store final destination, but save original value
+ Uint32 saveOne = keyInfo->keyData[19];
+ keyInfo->keyData[19] = ref;
+ sendSignal(routeBlockref, GSN_KEYINFO20_R, signal, 25, JBB);
+ keyInfo->keyData[19] = saveOne;
+ for(Uint32 i = 0; i<19; i++){
+ keyInfo->keyData[i] = keyInfo->keyData[keyDataIndex + i];
+ }
+ }//for
+ keyInfo->keyData[keyLenLeft] = ref;
+ sendSignal(routeBlockref, GSN_KEYINFO20_R, signal, 5 + keyLenLeft + 1, JBB);
+ }
+
+}//Dblqh::sendKeyinfo20()
+
+/* ------------------------------------------------------------------------
+ * ------- SEND SCAN_FRAGCONF TO TC THAT CONTROLS THE SCAN -------
+ *
+ * ------------------------------------------------------------------------ */
+void Dblqh::sendScanFragConf(Signal* signal, Uint32 scanCompleted)
+{
+ scanptr.p->scanSearchCondFalseCount = 0;
+ scanptr.p->scanTcWaiting = ZFALSE;
+ ScanFragConf * conf = (ScanFragConf*)&signal->theData[0];
+
+ conf->senderData = tcConnectptr.p->clientConnectrec;
+ conf->completedOps = scanptr.p->scanCompletedOperations;
+ conf->fragmentCompleted = scanCompleted;
+ for(Uint32 i = 0; i<MAX_PARALLEL_OP_PER_SCAN; i++)
+ conf->opReturnDataLen[i] = scanptr.p->scanOpLength[i];
+ conf->transId1 = tcConnectptr.p->transid[0];
+ conf->transId2 = tcConnectptr.p->transid[1];
+ sendSignal(tcConnectptr.p->clientBlockref, GSN_SCAN_FRAGCONF,
+ signal, ScanFragConf::SignalLength, JBB);
+}//Dblqh::sendScanFragConf()
+
+/* ######################################################################### */
+/* ####### NODE RECOVERY MODULE ####### */
+/* */
+/* ######################################################################### */
+/*---------------------------------------------------------------------------*/
+/* */
+/* THIS MODULE IS USED WHEN A NODE HAS FAILED. IT PERFORMS A COPY OF A */
+/* FRAGMENT TO A NEW REPLICA OF THE FRAGMENT. IT DOES ALSO SHUT DOWN ALL */
+/* CONNECTIONS TO THE FAILED NODE. */
+/*---------------------------------------------------------------------------*/
+void Dblqh::calculateHash(Signal* signal)
+{
+ DatabufPtr locDatabufptr;
+ UintR Ti;
+ UintR Tdata0;
+ UintR Tdata1;
+ UintR Tdata2;
+ UintR Tdata3;
+ UintR* Tdata32;
+ Uint64 Tdata[512];
+
+ Tdata32 = (UintR*)&Tdata[0];
+
+ Tdata0 = tcConnectptr.p->tupkeyData[0];
+ Tdata1 = tcConnectptr.p->tupkeyData[1];
+ Tdata2 = tcConnectptr.p->tupkeyData[2];
+ Tdata3 = tcConnectptr.p->tupkeyData[3];
+ Tdata32[0] = Tdata0;
+ Tdata32[1] = Tdata1;
+ Tdata32[2] = Tdata2;
+ Tdata32[3] = Tdata3;
+ locDatabufptr.i = tcConnectptr.p->firstTupkeybuf;
+ Ti = 4;
+ while (locDatabufptr.i != RNIL) {
+ ptrCheckGuard(locDatabufptr, cdatabufFileSize, databuf);
+ Tdata0 = locDatabufptr.p->data[0];
+ Tdata1 = locDatabufptr.p->data[1];
+ Tdata2 = locDatabufptr.p->data[2];
+ Tdata3 = locDatabufptr.p->data[3];
+ Tdata32[Ti ] = Tdata0;
+ Tdata32[Ti + 1] = Tdata1;
+ Tdata32[Ti + 2] = Tdata2;
+ Tdata32[Ti + 3] = Tdata3;
+ locDatabufptr.i = locDatabufptr.p->nextDatabuf;
+ Ti += 4;
+ }//while
+ tcConnectptr.p->hashValue =
+ md5_hash((Uint64*)&Tdata32[0], (UintR)tcConnectptr.p->primKeyLen);
+}//Dblqh::calculateHash()
+
+/* *************************************** */
+/* COPY_FRAGREQ: Start copying a fragment */
+/* *************************************** */
+void Dblqh::execCOPY_FRAGREQ(Signal* signal)
+{
+ jamEntry();
+ const CopyFragReq * const copyFragReq = (CopyFragReq *)&signal->theData[0];
+ tabptr.i = copyFragReq->tableId;
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ const Uint32 fragId = copyFragReq->fragId;
+ const Uint32 copyPtr = copyFragReq->userPtr;
+ const Uint32 userRef = copyFragReq->userRef;
+ const Uint32 nodeId = copyFragReq->nodeId;
+
+ ndbrequire(cnoActiveCopy < 3);
+ ndbrequire(getFragmentrec(signal, fragId));
+ ndbrequire(fragptr.p->copyFragState == ZIDLE);
+ ndbrequire(fragptr.p->noActiveScan < MAX_PARALLEL_SCANS_PER_FRAG);
+ ndbrequire(cfirstfreeScanrec != RNIL);
+ ndbrequire(cfirstfreeTcConrec != RNIL);
+ ndbrequire(fragptr.p->fragScanRec[NR_ScanNo] == ZNIL);
+
+ fragptr.p->fragDistributionKey = copyFragReq->distributionKey;
+
+ if (DictTabInfo::isOrderedIndex(tabptr.p->tableType)) {
+ jam();
+ /**
+ * Ordered index doesn't need to be copied
+ */
+ CopyFragConf * const conf = (CopyFragConf *)&signal->theData[0];
+ conf->userPtr = copyPtr;
+ conf->sendingNodeId = cownNodeid;
+ conf->startingNodeId = nodeId;
+ conf->tableId = tabptr.i;
+ conf->fragId = fragId;
+ sendSignal(userRef, GSN_COPY_FRAGCONF, signal,
+ CopyFragConf::SignalLength, JBB);
+ return;
+ }//if
+
+ seizeScanrec(signal);
+/* ------------------------------------------------------------------------- */
+// We keep track of how many operation records in ACC that has been booked.
+// Copy fragment has records always booked and thus need not book any. The
+// most operations in parallel use is the scanConcurrentOperations.
+// This variable has to be set-up here since it is used by releaseScanrec
+// to unbook operation records in ACC.
+/* ------------------------------------------------------------------------- */
+ scanptr.p->scanConcurrentOperations = 0;
+ scanptr.p->rangeScan = 0;
+ seizeTcrec();
+
+ /**
+ * Remove implicit cast/usage of CopyFragReq
+ */
+ //initCopyrec(signal);
+ scanptr.p->copyPtr = copyPtr;
+ scanptr.p->scanType = ScanRecord::COPY;
+ scanptr.p->scanApiBlockref = userRef;
+ scanptr.p->scanNodeId = nodeId;
+ scanptr.p->scanTcrec = tcConnectptr.i;
+ scanptr.p->scanSchemaVersion = copyFragReq->schemaVersion;
+ scanptr.p->scanCompletedStatus = ZFALSE;
+ scanptr.p->scanErrorCounter = 0;
+ scanptr.p->scanNumber = NR_ScanNo;
+ fragptr.p->fragScanRec[NR_ScanNo] = scanptr.i;
+ fragptr.p->noActiveScan++;
+
+ initScanTc(signal,
+ 0,
+ (DBLQH << 20) + (cownNodeid << 8),
+ fragId,
+ copyFragReq->nodeId);
+ cactiveCopy[cnoActiveCopy] = fragptr.i;
+ cnoActiveCopy++;
+
+ tcConnectptr.p->copyCountWords = 0;
+ tcConnectptr.p->tcOprec = tcConnectptr.i;
+ tcConnectptr.p->schemaVersion = scanptr.p->scanSchemaVersion;
+ scanptr.p->scanState = ScanRecord::WAIT_ACC_COPY;
+ AccScanReq * req = (AccScanReq*)&signal->theData[0];
+ req->senderData = scanptr.i;
+ req->senderRef = cownref;
+ req->tableId = tabptr.i;
+ req->fragmentNo = fragId;
+ req->requestInfo = 0;
+ AccScanReq::setLockMode(req->requestInfo, 0);
+ AccScanReq::setKeyinfoFlag(req->requestInfo, 1);
+ AccScanReq::setReadCommittedFlag(req->requestInfo, 0);
+ req->transId1 = tcConnectptr.p->transid[0];
+ req->transId2 = tcConnectptr.p->transid[1];
+ req->savePointId = tcConnectptr.p->savePointId;
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_ACC_SCANREQ, signal,
+ AccScanReq::SignalLength, JBB);
+ return;
+}//Dblqh::execCOPY_FRAGREQ()
+
+void Dblqh::accScanConfCopyLab(Signal* signal)
+{
+ AccScanConf * const accScanConf = (AccScanConf *)&signal->theData[0];
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+/*--------------------------------------------------------------------------*/
+/* PRECONDITION: SCAN_STATE = WAIT_ACC_COPY */
+/*--------------------------------------------------------------------------*/
+ if (accScanConf->flag == AccScanConf::ZEMPTY_FRAGMENT) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THE FRAGMENT WAS EMPTY. */
+/* REPORT SUCCESSFUL COPYING. */
+/*---------------------------------------------------------------------------*/
+ tupCopyCloseConfLab(signal);
+ return;
+ }//if
+ scanptr.p->scanAccPtr = accScanConf->accPtr;
+ scanptr.p->scanState = ScanRecord::WAIT_STORED_PROC_COPY;
+ signal->theData[0] = tcConnectptr.p->tupConnectrec;
+ signal->theData[1] = tcConnectptr.p->tableref;
+ signal->theData[2] = scanptr.p->scanSchemaVersion;
+ signal->theData[3] = ZSTORED_PROC_COPY;
+// theData[4] is not used in TUP with ZSTORED_PROC_COPY
+ sendSignal(tcConnectptr.p->tcTupBlockref, GSN_STORED_PROCREQ, signal, 5, JBB);
+ return;
+}//Dblqh::accScanConfCopyLab()
+
+/*---------------------------------------------------------------------------*/
+/* ENTER STORED_PROCCONF WITH */
+/* TC_CONNECTPTR, */
+/* TSTORED_PROC_ID */
+/*---------------------------------------------------------------------------*/
+void Dblqh::storedProcConfCopyLab(Signal* signal)
+{
+/*---------------------------------------------------------------------------*/
+/* PRECONDITION: SCAN_STATE = WAIT_STORED_PROC_COPY */
+/*---------------------------------------------------------------------------*/
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THE COPY PROCESS HAVE BEEN COMPLETED, MOST LIKELY DUE TO A NODE FAILURE.*/
+/*---------------------------------------------------------------------------*/
+ closeCopyLab(signal);
+ return;
+ }//if
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ scanptr.p->scanState = ScanRecord::WAIT_NEXT_SCAN_COPY;
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ jam();
+ linkActiveFrag(signal);
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::COPY_FIRST_STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ case Fragrecord::CRASH_RECOVERING:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ continueFirstCopyAfterBlockedLab(signal);
+ return;
+}//Dblqh::storedProcConfCopyLab()
+
+void Dblqh::continueFirstCopyAfterBlockedLab(Signal* signal)
+{
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ signal->theData[1] = RNIL;
+ signal->theData[2] = NextScanReq::ZSCAN_NEXT;
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+ return;
+}//Dblqh::continueFirstCopyAfterBlockedLab()
+
+/*---------------------------------------------------------------------------*/
+/* ENTER NEXT_SCANCONF WITH */
+/* SCANPTR, */
+/* TFRAGID, */
+/* TACC_OPPTR, */
+/* TLOCAL_KEY1, */
+/* TLOCAL_KEY2, */
+/* TKEY_LENGTH, */
+/* TKEY1, */
+/* TKEY2, */
+/* TKEY3, */
+/* TKEY4 */
+/*---------------------------------------------------------------------------*/
+/* PRECONDITION: SCAN_STATE = WAIT_NEXT_SCAN_COPY */
+/*---------------------------------------------------------------------------*/
+void Dblqh::nextScanConfCopyLab(Signal* signal)
+{
+ NextScanConf * const nextScanConf = (NextScanConf *)&signal->theData[0];
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ if (nextScanConf->fragId == RNIL) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THERE ARE NO MORE TUPLES TO FETCH. WE NEED TO CLOSE */
+/* THE COPY IN ACC AND DELETE THE STORED PROCEDURE IN TUP */
+/*---------------------------------------------------------------------------*/
+ releaseActiveFrag(signal);
+ if (tcConnectptr.p->copyCountWords == 0) {
+ closeCopyLab(signal);
+ return;
+ }//if
+/*---------------------------------------------------------------------------*/
+// Wait until copying is completed also at the starting node before reporting
+// completion. Signal completion through scanCompletedStatus-flag.
+/*---------------------------------------------------------------------------*/
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ return;
+ }//if
+
+ // If accOperationPtr == RNIL no record was returned by ACC
+ if (nextScanConf->accOperationPtr == RNIL) {
+ jam();
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ signal->theData[1] = AccCheckScan::ZCHECK_LCP_STOP;
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_ACC_CHECK_SCAN, signal, 2, JBB);
+ return;
+ }
+
+ scanptr.p->scanAccOpPtr[0] = nextScanConf->accOperationPtr;
+ initCopyTc(signal);
+ if (tcConnectptr.p->primKeyLen > 4) {
+ jam();
+ tcConnectptr.p->save1 = 4;
+ scanptr.p->scanState = ScanRecord::WAIT_COPY_KEYINFO;
+ return;
+ }//if
+ copySendTupkeyReqLab(signal);
+ return;
+}//Dblqh::nextScanConfCopyLab()
+
+void Dblqh::copySendTupkeyReqLab(Signal* signal)
+{
+ Uint32 reqinfo = 0;
+ Uint32 tupFragPtr;
+
+ reqinfo = reqinfo + (tcConnectptr.p->operation << 6);
+ reqinfo = reqinfo + (tcConnectptr.p->opExec << 10);
+ tcConnectptr.p->transactionState = TcConnectionrec::COPY_TUPKEY;
+ scanptr.p->scanState = ScanRecord::WAIT_TUPKEY_COPY;
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (fragptr.p->fragId == scanptr.p->scanLocalFragid) {
+ jam();
+ tupFragPtr = fragptr.p->tupFragptr[0];
+ } else {
+ jam();
+ tupFragPtr = fragptr.p->tupFragptr[1];
+ }//if
+ {
+ TupKeyReq * const tupKeyReq = (TupKeyReq *)signal->getDataPtrSend();
+
+ tupKeyReq->connectPtr = tcConnectptr.p->tupConnectrec;
+ tupKeyReq->request = reqinfo;
+ tupKeyReq->tableRef = tcConnectptr.p->tableref;
+ tupKeyReq->fragId = scanptr.p->scanLocalFragid;
+ tupKeyReq->keyRef1 = scanptr.p->scanLocalref[0];
+ tupKeyReq->keyRef2 = scanptr.p->scanLocalref[1];
+ tupKeyReq->attrBufLen = 0;
+ tupKeyReq->opRef = tcConnectptr.i;
+ tupKeyReq->applRef = cownref;
+ tupKeyReq->schemaVersion = scanptr.p->scanSchemaVersion;
+ tupKeyReq->storedProcedure = scanptr.p->scanStoredProcId;
+ tupKeyReq->transId1 = tcConnectptr.p->transid[0];
+ tupKeyReq->transId2 = tcConnectptr.p->transid[1];
+ tupKeyReq->fragPtr = tupFragPtr;
+ tupKeyReq->primaryReplica = (tcConnectptr.p->seqNoReplica == 0)?true:false;
+ tupKeyReq->coordinatorTC = tcConnectptr.p->tcBlockref;
+ tupKeyReq->tcOpIndex = tcConnectptr.p->tcOprec;
+ tupKeyReq->savePointId = tcConnectptr.p->savePointId;
+ Uint32 blockNo = refToBlock(tcConnectptr.p->tcTupBlockref);
+ EXECUTE_DIRECT(blockNo, GSN_TUPKEYREQ, signal,
+ TupKeyReq::SignalLength);
+ }
+}//Dblqh::copySendTupkeyReqLab()
+
+/*---------------------------------------------------------------------------*/
+/* USED IN COPYING OPERATION TO RECEIVE ATTRINFO FROM TUP. */
+/*---------------------------------------------------------------------------*/
+/* ************>> */
+/* TRANSID_AI > */
+/* ************>> */
+void Dblqh::execTRANSID_AI(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ Uint32 length = signal->length() - 3;
+ ndbrequire(tcConnectptr.p->transactionState == TcConnectionrec::COPY_TUPKEY);
+ Uint32 * src = &signal->theData[3];
+ while(length > 22){
+ if (saveTupattrbuf(signal, &signal->theData[3], 22) == ZOK) {
+ ;
+ } else {
+ jam();
+ tcConnectptr.p->errorCode = ZGET_ATTRINBUF_ERROR;
+ return;
+ }//if
+ src += 22;
+ length -= 22;
+ }
+ if (saveTupattrbuf(signal, src, length) == ZOK) {
+ return;
+ }
+ jam();
+ tcConnectptr.p->errorCode = ZGET_ATTRINBUF_ERROR;
+}//Dblqh::execTRANSID_AI()
+
+/*--------------------------------------------------------------------------*/
+/* ENTER TUPKEYCONF WITH */
+/* TC_CONNECTPTR, */
+/* TDATA2, */
+/* TDATA3, */
+/* TDATA4, */
+/* TDATA5 */
+/*--------------------------------------------------------------------------*/
+/* PRECONDITION: TRANSACTION_STATE = COPY_TUPKEY */
+/*--------------------------------------------------------------------------*/
+void Dblqh::copyTupkeyConfLab(Signal* signal)
+{
+ const TupKeyConf * const tupKeyConf = (TupKeyConf *)signal->getDataPtr();
+
+ UintR readLength = tupKeyConf->readLength;
+
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ releaseActiveFrag(signal);
+ if (tcConnectptr.p->errorCode != 0) {
+ jam();
+ closeCopyLab(signal);
+ return;
+ }//if
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THE COPY PROCESS HAVE BEEN CLOSED. MOST LIKELY A NODE FAILURE. */
+/*---------------------------------------------------------------------------*/
+ closeCopyLab(signal);
+ return;
+ }//if
+ tcConnectptr.p->totSendlenAi = readLength;
+ tcConnectptr.p->connectState = TcConnectionrec::COPY_CONNECTED;
+ calculateHash(signal);
+/*---------------------------------------------------------------------------*/
+// To avoid using up to many operation records in ACC we will increase the
+// constant to ensure that we never send more than 40 records at a time.
+// This is where the constant 56 comes from. For long records this constant
+// will not matter that much. The current maximum is 6000 words outstanding
+// (including a number of those 56 words not really sent). We also have to
+// ensure that there are never more simultaneous usage of these operation
+// records to ensure that node recovery does not fail because of simultaneous
+// scanning.
+/*---------------------------------------------------------------------------*/
+ UintR TnoOfWords = readLength + tcConnectptr.p->primKeyLen;
+ TnoOfWords = TnoOfWords + MAGIC_CONSTANT;
+ TnoOfWords = TnoOfWords + (TnoOfWords >> 2);
+
+ /*-----------------------------------------------------------------
+ * NOTE for transid1!
+ * Transid1 in the tcConnection record is used load regulate the
+ * copy(node recovery) process.
+ * The number of outstanding words are written in the transid1
+ * variable. This will be sent to the starting node in the
+ * LQHKEYREQ signal and when the answer is returned in the LQHKEYCONF
+ * we can reduce the number of outstanding words and check to see
+ * if more LQHKEYREQ signals should be sent.
+ *
+ * However efficient this method is rather unsafe in such way that
+ * it overwrites the transid1 original data.
+ *
+ * Also see TR 587.
+ *----------------------------------------------------------------*/
+ tcConnectptr.p->transid[0] = TnoOfWords; // Data overload, see note!
+ packLqhkeyreqLab(signal);
+ tcConnectptr.p->copyCountWords += TnoOfWords;
+ scanptr.p->scanState = ScanRecord::WAIT_LQHKEY_COPY;
+ if (tcConnectptr.p->copyCountWords < cmaxWordsAtNodeRec) {
+ nextRecordCopy(signal);
+ return;
+ }//if
+ return;
+}//Dblqh::copyTupkeyConfLab()
+
+/*---------------------------------------------------------------------------*/
+/* ENTER LQHKEYCONF */
+/*---------------------------------------------------------------------------*/
+/* PRECONDITION: CONNECT_STATE = COPY_CONNECTED */
+/*---------------------------------------------------------------------------*/
+void Dblqh::copyCompletedLab(Signal* signal)
+{
+ const LqhKeyConf * const lqhKeyConf = (LqhKeyConf *)signal->getDataPtr();
+
+ ndbrequire(tcConnectptr.p->transid[1] == lqhKeyConf->transId2);
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ if (tcConnectptr.p->copyCountWords >= cmaxWordsAtNodeRec) {
+ tcConnectptr.p->copyCountWords -= lqhKeyConf->transId1; // Data overload, see note!
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ jam();
+/*---------------------------------------------------------------------------*/
+// Copy to complete, we will not start any new copying.
+/*---------------------------------------------------------------------------*/
+ closeCopyLab(signal);
+ return;
+ }//if
+ if (tcConnectptr.p->copyCountWords < cmaxWordsAtNodeRec) {
+ jam();
+ nextRecordCopy(signal);
+ }//if
+ return;
+ }//if
+ tcConnectptr.p->copyCountWords -= lqhKeyConf->transId1; // Data overload, see note!
+ ndbrequire(tcConnectptr.p->copyCountWords <= cmaxWordsAtNodeRec);
+ if (tcConnectptr.p->copyCountWords > 0) {
+ jam();
+ return;
+ }//if
+/*---------------------------------------------------------------------------*/
+// No more outstanding copies. We will only start new ones from here if it was
+// stopped before and this only happens when copyCountWords is bigger than the
+// threshold value. Since this did not occur we must be waiting for completion.
+// Check that this is so. If not we crash to find out what is going on.
+/*---------------------------------------------------------------------------*/
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ jam();
+ closeCopyLab(signal);
+ return;
+ }//if
+ if (scanptr.p->scanState == ScanRecord::WAIT_LQHKEY_COPY) {
+ jam();
+/*---------------------------------------------------------------------------*/
+// Make sure that something is in progress. Otherwise we will simply stop
+// and nothing more will happen.
+/*---------------------------------------------------------------------------*/
+ systemErrorLab(signal);
+ return;
+ }//if
+ return;
+}//Dblqh::copyCompletedLab()
+
+void Dblqh::nextRecordCopy(Signal* signal)
+{
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ if (scanptr.p->scanState != ScanRecord::WAIT_LQHKEY_COPY) {
+ jam();
+/*---------------------------------------------------------------------------*/
+// Make sure that nothing is in progress. Otherwise we will have to simultaneous
+// scans on the same record and this will certainly lead to unexpected
+// behaviour.
+/*---------------------------------------------------------------------------*/
+ systemErrorLab(signal);
+ return;
+ }//if
+ scanptr.p->scanState = ScanRecord::WAIT_NEXT_SCAN_COPY;
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ jam();
+ linkActiveFrag(signal);
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::COPY_STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ case Fragrecord::CRASH_RECOVERING:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ continueCopyAfterBlockedLab(signal);
+ return;
+}//Dblqh::nextRecordCopy()
+
+void Dblqh::continueCopyAfterBlockedLab(Signal* signal)
+{
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ tcConnectptr.p->errorCode = 0;
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ signal->theData[1] = scanptr.p->scanAccOpPtr[0];
+ signal->theData[2] = NextScanReq::ZSCAN_NEXT_COMMIT;
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+ return;
+}//Dblqh::continueCopyAfterBlockedLab()
+
+void Dblqh::copyLqhKeyRefLab(Signal* signal)
+{
+ ndbrequire(tcConnectptr.p->transid[1] == signal->theData[4]);
+ tcConnectptr.p->copyCountWords -= signal->theData[3];
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ scanptr.p->scanErrorCounter++;
+ tcConnectptr.p->errorCode = terrorCode;
+ closeCopyLab(signal);
+ return;
+}//Dblqh::copyLqhKeyRefLab()
+
+void Dblqh::closeCopyLab(Signal* signal)
+{
+ if (tcConnectptr.p->copyCountWords > 0) {
+/*---------------------------------------------------------------------------*/
+// We are still waiting for responses from the starting node.
+// Wait until all of those have arrived until we start the
+// close process.
+/*---------------------------------------------------------------------------*/
+ jam();
+ return;
+ }//if
+ tcConnectptr.p->transid[0] = 0;
+ tcConnectptr.p->transid[1] = 0;
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ scanptr.p->scanState = ScanRecord::WAIT_CLOSE_COPY;
+ switch (fragptr.p->fragStatus) {
+ case Fragrecord::FSACTIVE:
+ jam();
+ linkActiveFrag(signal);
+ break;
+ case Fragrecord::BLOCKED:
+ jam();
+ linkFragQueue(signal);
+ tcConnectptr.p->transactionState = TcConnectionrec::COPY_CLOSE_STOPPED;
+ return;
+ break;
+ case Fragrecord::FREE:
+ jam();
+ case Fragrecord::ACTIVE_CREATION:
+ jam();
+ case Fragrecord::CRASH_RECOVERING:
+ jam();
+ case Fragrecord::DEFINED:
+ jam();
+ case Fragrecord::REMOVING:
+ jam();
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ continueCloseCopyAfterBlockedLab(signal);
+ return;
+}//Dblqh::closeCopyLab()
+
+void Dblqh::continueCloseCopyAfterBlockedLab(Signal* signal)
+{
+ scanptr.i = tcConnectptr.p->tcScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ signal->theData[0] = scanptr.p->scanAccPtr;
+ signal->theData[1] = RNIL;
+ signal->theData[2] = ZCOPY_CLOSE;
+ sendSignal(tcConnectptr.p->tcAccBlockref, GSN_NEXT_SCANREQ, signal, 3, JBB);
+ return;
+}//Dblqh::continueCloseCopyAfterBlockedLab()
+
+/*---------------------------------------------------------------------------*/
+/* ENTER NEXT_SCANCONF WITH */
+/* SCANPTR, */
+/* TFRAGID, */
+/* TACC_OPPTR, */
+/* TLOCAL_KEY1, */
+/* TLOCAL_KEY2, */
+/* TKEY_LENGTH, */
+/* TKEY1, */
+/* TKEY2, */
+/* TKEY3, */
+/* TKEY4 */
+/*---------------------------------------------------------------------------*/
+/* PRECONDITION: SCAN_STATE = WAIT_CLOSE_COPY */
+/*---------------------------------------------------------------------------*/
+void Dblqh::accCopyCloseConfLab(Signal* signal)
+{
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ scanptr.p->scanState = ScanRecord::WAIT_DELETE_STORED_PROC_ID_COPY;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ signal->theData[0] = tcConnectptr.p->tupConnectrec;
+ signal->theData[1] = tcConnectptr.p->tableref;
+ signal->theData[2] = scanptr.p->scanSchemaVersion;
+ signal->theData[3] = ZDELETE_STORED_PROC_ID;
+ signal->theData[4] = scanptr.p->scanStoredProcId;
+ sendSignal(tcConnectptr.p->tcTupBlockref, GSN_STORED_PROCREQ, signal, 5, JBB);
+ return;
+}//Dblqh::accCopyCloseConfLab()
+
+/*---------------------------------------------------------------------------*/
+/* ENTER STORED_PROCCONF WITH */
+/* TC_CONNECTPTR, */
+/* TSTORED_PROC_ID */
+/*---------------------------------------------------------------------------*/
+/* PRECONDITION: SCAN_STATE = WAIT_DELETE_STORED_PROC_ID_COPY */
+/*---------------------------------------------------------------------------*/
+void Dblqh::tupCopyCloseConfLab(Signal* signal)
+{
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (tcConnectptr.p->abortState == TcConnectionrec::NEW_FROM_TC) {
+ jam();
+ tcNodeFailptr.i = tcConnectptr.p->tcNodeFailrec;
+ ptrCheckGuard(tcNodeFailptr, ctcNodeFailrecFileSize, tcNodeFailRecord);
+ tcNodeFailptr.p->tcRecNow = tcConnectptr.i + 1;
+ signal->theData[0] = ZLQH_TRANS_NEXT;
+ signal->theData[1] = tcNodeFailptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+
+ CopyFragRef * const ref = (CopyFragRef *)&signal->theData[0];
+ ref->userPtr = scanptr.p->copyPtr;
+ ref->sendingNodeId = cownNodeid;
+ ref->startingNodeId = scanptr.p->scanNodeId;
+ ref->tableId = fragptr.p->tabRef;
+ ref->fragId = fragptr.p->fragId;
+ ref->errorCode = ZNODE_FAILURE_ERROR;
+ sendSignal(scanptr.p->scanApiBlockref, GSN_COPY_FRAGREF, signal,
+ CopyFragRef::SignalLength, JBB);
+ } else {
+ if (scanptr.p->scanErrorCounter > 0) {
+ jam();
+ CopyFragRef * const ref = (CopyFragRef *)&signal->theData[0];
+ ref->userPtr = scanptr.p->copyPtr;
+ ref->sendingNodeId = cownNodeid;
+ ref->startingNodeId = scanptr.p->scanNodeId;
+ ref->tableId = fragptr.p->tabRef;
+ ref->fragId = fragptr.p->fragId;
+ ref->errorCode = tcConnectptr.p->errorCode;
+ sendSignal(scanptr.p->scanApiBlockref, GSN_COPY_FRAGREF, signal,
+ CopyFragRef::SignalLength, JBB);
+ } else {
+ jam();
+ CopyFragConf * const conf = (CopyFragConf *)&signal->theData[0];
+ conf->userPtr = scanptr.p->copyPtr;
+ conf->sendingNodeId = cownNodeid;
+ conf->startingNodeId = scanptr.p->scanNodeId;
+ conf->tableId = tcConnectptr.p->tableref;
+ conf->fragId = tcConnectptr.p->fragmentid;
+ sendSignal(scanptr.p->scanApiBlockref, GSN_COPY_FRAGCONF, signal,
+ CopyFragConf::SignalLength, JBB);
+ }//if
+ }//if
+ releaseActiveCopy(signal);
+ tcConnectptr.p->tcScanRec = RNIL;
+ ndbrequire(scanptr.p->scanNumber < MAX_PARALLEL_SCANS_PER_FRAG);
+ fragptr.p->fragScanRec[scanptr.p->scanNumber] = ZNIL;
+ ndbrequire(fragptr.p->noActiveScan > 0);
+ fragptr.p->noActiveScan--;
+ fragptr.p->copyFragState = ZIDLE;
+ releaseOprec(signal);
+ releaseTcrec(signal, tcConnectptr);
+ releaseScanrec(signal);
+}//Dblqh::tupCopyCloseConfLab()
+
+/*---------------------------------------------------------------------------*/
+/* A NODE FAILURE OCCURRED DURING THE COPY PROCESS. WE NEED TO CLOSE THE */
+/* COPY PROCESS SINCE A NODE FAILURE DURING THE COPY PROCESS WILL ALSO */
+/* FAIL THE NODE THAT IS TRYING TO START-UP. */
+/*---------------------------------------------------------------------------*/
+void Dblqh::closeCopyRequestLab(Signal* signal)
+{
+ scanptr.p->scanErrorCounter++;
+ switch (scanptr.p->scanState) {
+ case ScanRecord::WAIT_TUPKEY_COPY:
+ case ScanRecord::WAIT_COPY_KEYINFO:
+ case ScanRecord::WAIT_NEXT_SCAN_COPY:
+ jam();
+/*---------------------------------------------------------------------------*/
+/* SET COMPLETION STATUS AND WAIT FOR OPPORTUNITY TO STOP THE SCAN. */
+// ALSO SET NO OF WORDS OUTSTANDING TO ZERO TO AVOID ETERNAL WAIT.
+/*---------------------------------------------------------------------------*/
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ tcConnectptr.p->copyCountWords = 0;
+ break;
+ case ScanRecord::WAIT_ACC_COPY:
+ case ScanRecord::WAIT_STORED_PROC_COPY:
+ jam();
+/*---------------------------------------------------------------------------*/
+/* WE ARE CURRENTLY STARTING UP THE SCAN. SET COMPLETED STATUS AND WAIT FOR*/
+/* COMPLETION OF STARTUP. */
+/*---------------------------------------------------------------------------*/
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ break;
+ case ScanRecord::WAIT_CLOSE_COPY:
+ case ScanRecord::WAIT_DELETE_STORED_PROC_ID_COPY:
+ jam();
+/*---------------------------------------------------------------------------*/
+/* CLOSE IS ALREADY ONGOING. WE NEED NOT DO ANYTHING. */
+/*---------------------------------------------------------------------------*/
+ break;
+ case ScanRecord::WAIT_LQHKEY_COPY:
+ jam();
+/*---------------------------------------------------------------------------*/
+/* WE ARE WAITING FOR THE FAILED NODE. THE NODE WILL NEVER COME BACK. */
+// WE NEED TO START THE FAILURE HANDLING IMMEDIATELY.
+// ALSO SET NO OF WORDS OUTSTANDING TO ZERO TO AVOID ETERNAL WAIT.
+/*---------------------------------------------------------------------------*/
+ tcConnectptr.p->copyCountWords = 0;
+ closeCopyLab(signal);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dblqh::closeCopyRequestLab()
+
+/* ****************************************************** */
+/* COPY_ACTIVEREQ: Change state of a fragment to ACTIVE. */
+/* ****************************************************** */
+void Dblqh::execCOPY_ACTIVEREQ(Signal* signal)
+{
+ CRASH_INSERTION(5026);
+
+ const CopyActiveReq * const req = (CopyActiveReq *)&signal->theData[0];
+ jamEntry();
+ Uint32 masterPtr = req->userPtr;
+ BlockReference masterRef = req->userRef;
+ tabptr.i = req->tableId;
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ Uint32 fragId = req->fragId;
+ ndbrequire(getFragmentrec(signal, fragId));
+
+ fragptr.p->fragDistributionKey = req->distributionKey;
+
+ ndbrequire(cnoActiveCopy < 3);
+ cactiveCopy[cnoActiveCopy] = fragptr.i;
+ cnoActiveCopy++;
+ fragptr.p->masterBlockref = masterRef;
+ fragptr.p->masterPtr = masterPtr;
+ if (fragptr.p->fragStatus == Fragrecord::FSACTIVE) {
+ jam();
+/*------------------------------------------------------*/
+/* PROCESS HAVE ALREADY BEEN STARTED BY PREVIOUS */
+/* MASTER. WE HAVE ALREADY SET THE PROPER MASTER */
+/* BLOCK REFERENCE. */
+/*------------------------------------------------------*/
+ if (fragptr.p->activeTcCounter == 0) {
+ jam();
+/*------------------------------------------------------*/
+/* PROCESS WAS EVEN COMPLETED. */
+/*------------------------------------------------------*/
+ sendCopyActiveConf(signal, tabptr.i);
+ }//if
+ return;
+ }//if
+ fragptr.p->fragStatus = Fragrecord::FSACTIVE;
+ if (fragptr.p->lcpFlag == Fragrecord::LCP_STATE_TRUE) {
+ jam();
+ fragptr.p->logFlag = Fragrecord::STATE_TRUE;
+ }//if
+ fragptr.p->activeTcCounter = 1;
+/*------------------------------------------------------*/
+/* SET IT TO ONE TO ENSURE THAT IT IS NOT POSSIBLE*/
+/* TO DECREASE IT TO ZERO UNTIL WE HAVE COMPLETED */
+/* THE SCAN. */
+/*------------------------------------------------------*/
+ signal->theData[0] = ZSCAN_TC_CONNECT;
+ signal->theData[1] = 0;
+ signal->theData[2] = tabptr.i;
+ signal->theData[3] = fragId;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 4, JBB);
+ return;
+}//Dblqh::execCOPY_ACTIVEREQ()
+
+void Dblqh::scanTcConnectLab(Signal* signal, Uint32 tstartTcConnect, Uint32 fragId)
+{
+ Uint32 tendTcConnect;
+
+ ndbrequire(getFragmentrec(signal, fragId));
+ if ((tstartTcConnect + 200) >= ctcConnectrecFileSize) {
+ jam();
+ tendTcConnect = ctcConnectrecFileSize - 1;
+ } else {
+ jam();
+ tendTcConnect = tstartTcConnect + 200;
+ }//if
+ for (tcConnectptr.i = tstartTcConnect;
+ tcConnectptr.i <= tendTcConnect;
+ tcConnectptr.i++) {
+ jam();
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ if (tcConnectptr.p->transactionState != TcConnectionrec::IDLE) {
+ switch (tcConnectptr.p->logWriteState) {
+ case TcConnectionrec::NOT_WRITTEN:
+ jam();
+ if (fragptr.i == tcConnectptr.p->fragmentptr) {
+ jam();
+ fragptr.p->activeTcCounter = fragptr.p->activeTcCounter + 1;
+ tcConnectptr.p->logWriteState = TcConnectionrec::NOT_WRITTEN_WAIT;
+ }//if
+ break;
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }//switch
+ }//if
+ }//for
+ if (tendTcConnect < (ctcConnectrecFileSize - 1)) {
+ jam();
+ signal->theData[0] = ZSCAN_TC_CONNECT;
+ signal->theData[1] = tendTcConnect + 1;
+ signal->theData[2] = tabptr.i;
+ signal->theData[3] = fragId;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 4, JBB);
+ } else {
+ jam();
+/*------------------------------------------------------*/
+/* THE SCAN HAVE BEEN COMPLETED. WE CHECK IF ALL */
+/* OPERATIONS HAVE ALREADY BEEN COMPLETED. */
+/*------------------------------------------------------*/
+ ndbrequire(fragptr.p->activeTcCounter > 0);
+ fragptr.p->activeTcCounter--;
+ if (fragptr.p->activeTcCounter == 0) {
+ jam();
+/*------------------------------------------------------*/
+/* SET START GLOBAL CHECKPOINT TO THE NEXT */
+/* CHECKPOINT WE HAVE NOT YET HEARD ANYTHING ABOUT*/
+/* THIS GCP WILL BE COMPLETELY COVERED BY THE LOG.*/
+/*------------------------------------------------------*/
+ fragptr.p->startGci = cnewestGci + 1;
+ sendCopyActiveConf(signal, tabptr.i);
+ }//if
+ }//if
+ return;
+}//Dblqh::scanTcConnectLab()
+
+/*---------------------------------------------------------------------------*/
+/* A NEW MASTER IS REQUESTING THE STATE IN LQH OF THE COPY FRAGMENT PARTS. */
+/*---------------------------------------------------------------------------*/
+/* ***************>> */
+/* COPY_STATEREQ > */
+/* ***************>> */
+void Dblqh::execCOPY_STATEREQ(Signal* signal)
+{
+ Uint32* dataPtr = &signal->theData[2];
+ jamEntry();
+ BlockReference tmasterBlockref = signal->theData[0];
+ Uint32 tnoCopy = 0;
+ do {
+ jam();
+ arrGuard(tnoCopy, 4);
+ fragptr.i = cactiveCopy[tnoCopy];
+ if (fragptr.i == RNIL) {
+ jam();
+ break;
+ }//if
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (fragptr.p->copyFragState != ZIDLE) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THIS FRAGMENT IS CURRENTLY ACTIVE IN COPYING THE FRAGMENT. */
+/*---------------------------------------------------------------------------*/
+ scanptr.i = fragptr.p->fragScanRec[NR_ScanNo];
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ jam();
+ dataPtr[3 + (tnoCopy << 2)] = ZCOPY_CLOSING;
+ } else {
+ jam();
+ dataPtr[3 + (tnoCopy << 2)] = ZCOPY_ONGOING;
+ }//if
+ dataPtr[2 + (tnoCopy << 2)] = scanptr.p->scanSchemaVersion;
+ scanptr.p->scanApiBlockref = tmasterBlockref;
+ } else {
+ ndbrequire(fragptr.p->activeTcCounter != 0);
+/*---------------------------------------------------------------------------*/
+/* COPY FRAGMENT IS COMPLETED AND WE ARE CURRENTLY GETTING THE STARTING */
+/* GCI OF THE NEW REPLICA OF THIS FRAGMENT. */
+/*---------------------------------------------------------------------------*/
+ fragptr.p->masterBlockref = tmasterBlockref;
+ dataPtr[3 + (tnoCopy << 2)] = ZCOPY_ACTIVATION;
+ }//if
+ dataPtr[tnoCopy << 2] = fragptr.p->tabRef;
+ dataPtr[1 + (tnoCopy << 2)] = fragptr.p->fragId;
+ tnoCopy++;
+ } while (tnoCopy < cnoActiveCopy);
+ signal->theData[0] = cownNodeid;
+ signal->theData[1] = tnoCopy;
+ sendSignal(tmasterBlockref, GSN_COPY_STATECONF, signal, 18, JBB);
+ return;
+}//Dblqh::execCOPY_STATEREQ()
+
+/* ========================================================================= */
+/* ======= INITIATE TC RECORD AT COPY FRAGMENT ======= */
+/* */
+/* SUBROUTINE SHORT NAME = ICT */
+/* ========================================================================= */
+void Dblqh::initCopyTc(Signal* signal)
+{
+ const NextScanConf * const nextScanConf = (NextScanConf *)&signal->theData[0];
+ tcConnectptr.p->primKeyLen = nextScanConf->keyLength;
+ tcConnectptr.p->tupkeyData[0] = nextScanConf->key[0];
+ tcConnectptr.p->tupkeyData[1] = nextScanConf->key[1];
+ tcConnectptr.p->tupkeyData[2] = nextScanConf->key[2];
+ tcConnectptr.p->tupkeyData[3] = nextScanConf->key[3];
+ scanptr.p->scanLocalref[0] = nextScanConf->localKey[0];
+ scanptr.p->scanLocalref[1] = nextScanConf->localKey[1];
+ scanptr.p->scanLocalFragid = nextScanConf->fragId;
+ tcConnectptr.p->operation = ZREAD;
+ tcConnectptr.p->apiVersionNo = 0;
+ tcConnectptr.p->opExec = 0; /* NOT INTERPRETED MODE */
+ tcConnectptr.p->schemaVersion = scanptr.p->scanSchemaVersion;
+ Uint32 reqinfo = 0;
+ LqhKeyReq::setKeyLen(reqinfo, nextScanConf->keyLength);
+ LqhKeyReq::setLockType(reqinfo, ZINSERT);
+ LqhKeyReq::setDirtyFlag(reqinfo, 1);
+ LqhKeyReq::setSimpleFlag(reqinfo, 1);
+ LqhKeyReq::setOperation(reqinfo, ZWRITE);
+ /* AILen in LQHKEYREQ IS ZERO */
+ tcConnectptr.p->reqinfo = reqinfo;
+/* ------------------------------------------------------------------------ */
+/* THE RECEIVING NODE WILL EXPECT THAT IT IS THE LAST NODE AND WILL */
+/* SEND COMPLETED AS THE RESPONSE SIGNAL SINCE DIRTY_OP BIT IS SET. */
+/* ------------------------------------------------------------------------ */
+ tcConnectptr.p->nodeAfterNext[0] = ZNIL;
+ tcConnectptr.p->nodeAfterNext[1] = ZNIL;
+ tcConnectptr.p->tcBlockref = cownref;
+ tcConnectptr.p->readlenAi = 0;
+ tcConnectptr.p->storedProcId = ZNIL;
+ tcConnectptr.p->opExec = 0;
+ tcConnectptr.p->nextSeqNoReplica = 0;
+ tcConnectptr.p->dirtyOp = ZFALSE;
+ tcConnectptr.p->lastReplicaNo = 0;
+ tcConnectptr.p->currTupAiLen = 0;
+ tcConnectptr.p->tcTimer = cLqhTimeOutCount;
+}//Dblqh::initCopyTc()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEND COPY_ACTIVECONF TO MASTER DIH ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::sendCopyActiveConf(Signal* signal, Uint32 tableId)
+{
+ releaseActiveCopy(signal);
+ CopyActiveConf * const conf = (CopyActiveConf *)&signal->theData[0];
+ conf->userPtr = fragptr.p->masterPtr;
+ conf->tableId = tableId;
+ conf->fragId = fragptr.p->fragId;
+ conf->startingNodeId = cownNodeid;
+ conf->startGci = fragptr.p->startGci;
+ sendSignal(fragptr.p->masterBlockref, GSN_COPY_ACTIVECONF, signal,
+ CopyActiveConf::SignalLength, JBB);
+}//Dblqh::sendCopyActiveConf()
+
+/* ##########################################################################
+ * ####### LOCAL CHECKPOINT MODULE #######
+ *
+ * ##########################################################################
+ * --------------------------------------------------------------------------
+ * THIS MODULE HANDLES THE EXECUTION AND CONTROL OF LOCAL CHECKPOINTS
+ * IT CONTROLS THE LOCAL CHECKPOINTS IN TUP AND ACC. IT DOES ALSO INTERACT
+ * WITH DIH TO CONTROL WHICH GLOBAL CHECKPOINTS THAT ARE RECOVERABLE
+ * ------------------------------------------------------------------------- */
+void Dblqh::execEMPTY_LCP_REQ(Signal* signal)
+{
+ jamEntry();
+ CRASH_INSERTION(5008);
+ EmptyLcpReq * const emptyLcpOrd = (EmptyLcpReq*)&signal->theData[0];
+
+ lcpPtr.i = 0;
+ ptrAss(lcpPtr, lcpRecord);
+
+ Uint32 nodeId = refToNode(emptyLcpOrd->senderRef);
+
+ lcpPtr.p->m_EMPTY_LCP_REQ.set(nodeId);
+ lcpPtr.p->reportEmpty = true;
+
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_IDLE){
+ jam();
+ bool ok = false;
+ switch(clcpCompletedState){
+ case LCP_IDLE:
+ ok = true;
+ sendEMPTY_LCP_CONF(signal, true);
+ break;
+ case LCP_RUNNING:
+ ok = true;
+ sendEMPTY_LCP_CONF(signal, false);
+ break;
+ case LCP_CLOSE_STARTED:
+ jam();
+ case ACC_LCP_CLOSE_COMPLETED:
+ jam();
+ case TUP_LCP_CLOSE_COMPLETED:
+ jam();
+ ok = true;
+ break;
+ }
+ ndbrequire(ok);
+
+ }//if
+
+ return;
+}//Dblqh::execEMPTY_LCPREQ()
+
+void Dblqh::execLCP_FRAG_ORD(Signal* signal)
+{
+ jamEntry();
+ CRASH_INSERTION(5010);
+ LcpFragOrd * const lcpFragOrd = (LcpFragOrd *)&signal->theData[0];
+ Uint32 lcpId = lcpFragOrd->lcpId;
+
+ lcpPtr.i = 0;
+ ptrAss(lcpPtr, lcpRecord);
+
+ lcpPtr.p->lastFragmentFlag = lcpFragOrd->lastFragmentFlag;
+ if (lcpFragOrd->lastFragmentFlag) {
+ jam();
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_IDLE) {
+ jam();
+ /* ----------------------------------------------------------
+ * NOW THE COMPLETE LOCAL CHECKPOINT ROUND IS COMPLETED.
+ * -------------------------------------------------------- */
+ if (cnoOfFragsCheckpointed > 0) {
+ jam();
+ completeLcpRoundLab(signal);
+ } else {
+ jam();
+ sendLCP_COMPLETE_REP(signal, lcpId);
+ }//if
+ }
+ return;
+ }//if
+ tabptr.i = lcpFragOrd->tableId;
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+
+ ndbrequire(tabptr.p->tableStatus == Tablerec::PREP_DROP_TABLE_ONGOING ||
+ tabptr.p->tableStatus == Tablerec::PREP_DROP_TABLE_DONE ||
+ tabptr.p->tableStatus == Tablerec::TABLE_DEFINED);
+
+ ndbrequire(getFragmentrec(signal, lcpFragOrd->fragmentId));
+
+ lcpPtr.i = 0;
+ ptrAss(lcpPtr, lcpRecord);
+ ndbrequire(!lcpPtr.p->lcpQueued);
+ if (c_lcpId < lcpFragOrd->lcpId) {
+ jam();
+ /**
+ * A new LCP
+ */
+ c_lcpId = lcpFragOrd->lcpId;
+ ndbrequire(lcpPtr.p->lcpState == LcpRecord::LCP_IDLE);
+ setLogTail(signal, lcpFragOrd->keepGci);
+ ndbrequire(clcpCompletedState == LCP_IDLE);
+ clcpCompletedState = LCP_RUNNING;
+ }//if
+ cnoOfFragsCheckpointed++;
+
+ if(tabptr.p->tableStatus == Tablerec::PREP_DROP_TABLE_DONE){
+ jam();
+ LcpRecord::FragOrd fragOrd;
+ fragOrd.fragPtrI = fragptr.i;
+ fragOrd.lcpFragOrd = * lcpFragOrd;
+ sendLCP_FRAG_REP(signal, fragOrd);
+ return;
+ }
+
+ if (lcpPtr.p->lcpState != LcpRecord::LCP_IDLE) {
+ ndbrequire(lcpPtr.p->lcpQueued == false);
+ lcpPtr.p->lcpQueued = true;
+ lcpPtr.p->queuedFragment.fragPtrI = fragptr.i;
+ lcpPtr.p->queuedFragment.lcpFragOrd = * lcpFragOrd;
+ return;
+ }//if
+
+ lcpPtr.p->currentFragment.fragPtrI = fragptr.i;
+ lcpPtr.p->currentFragment.lcpFragOrd = * lcpFragOrd;
+
+ sendLCP_FRAGIDREQ(signal);
+}//Dblqh::execLCP_FRAGORD()
+
+/* --------------------------------------------------------------------------
+ * PRECONDITION: LCP_PTR:LCP_STATE = WAIT_FRAGID
+ * --------------------------------------------------------------------------
+ * WE NOW HAVE THE LOCAL FRAGMENTS THAT THE LOCAL CHECKPOINT WILL USE.
+ * -------------------------------------------------------------------------- */
+void Dblqh::execLCP_FRAGIDCONF(Signal* signal)
+{
+ UintR Tfragid[4];
+
+ jamEntry();
+
+ lcpPtr.i = signal->theData[0];
+
+ Uint32 TaccPtr = signal->theData[1];
+ Uint32 noLocfrag = signal->theData[2];
+ Tfragid[0] = signal->theData[3];
+ Tfragid[1] = signal->theData[4];
+
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ ndbrequire(lcpPtr.p->lcpState == LcpRecord::LCP_WAIT_FRAGID);
+ /* ------------------------------------------------------------------------
+ * NO ERROR CHECKING OF TNO_LOCFRAG VALUE. OUT OF BOUND WILL IMPLY THAT AN
+ * INDEX OUT OF RANGE WILL CAUSE A SYSTEM RESTART WHICH IS DESIRED.
+ * ------------------------------------------------------------------------ */
+ lcpPtr.p->lcpAccptr = TaccPtr;
+ fragptr.i = lcpPtr.p->currentFragment.fragPtrI;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ ndbrequire(noLocfrag - 1 < 2);
+ for (Uint32 Tindex = 0; Tindex < noLocfrag; Tindex++) {
+ jam();
+ Uint32 fragId = Tfragid[Tindex];
+ /* ----------------------------------------------------------------------
+ * THERE IS NO ERROR CHECKING ON PURPOSE. IT IS POSSIBLE TO CALCULATE HOW
+ * MANY LOCAL LCP RECORDS THERE SHOULD BE. IT SHOULD NEVER HAPPEN THAT
+ * THERE IS NO ONE FREE. IF THERE IS NO ONE IT WILL ALSO BE A POINTER
+ * OUT OF RANGE WHICH IS AN ERROR CODE IN ITSELF. REUSES ERROR HANDLING
+ * IN AXE VM.
+ * ---------------------------------------------------------------------- */
+ seizeLcpLoc(signal);
+ initLcpLocAcc(signal, fragId);
+ seizeLcpLoc(signal);
+ initLcpLocTup(signal, fragId);
+ signal->theData[0] = lcpLocptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lcpPtr.p->currentFragment.lcpFragOrd.tableId;
+ signal->theData[3] = lcpLocptr.p->locFragid;
+ signal->theData[4] = lcpPtr.p->currentFragment.lcpFragOrd.lcpNo;
+ signal->theData[5] = lcpPtr.p->currentFragment.lcpFragOrd.lcpId % MAX_LCP_STORED;
+ sendSignal(fragptr.p->tupBlockref, GSN_TUP_PREPLCPREQ, signal, 6, JBB);
+ }//for
+ lcpPtr.p->lcpState = LcpRecord::LCP_WAIT_TUP_PREPLCP;
+ return;
+}//Dblqh::execLCP_FRAGIDCONF()
+
+/* --------------------------------------------------------------------------
+ * PRECONDITION: LCP_LOCPTR:LCP_STATE = WAIT_TUPPREPLCP
+ * --------------------------------------------------------------------------
+ * WE HAVE NOW PREPARED A LOCAL FRAGMENT IN TUP FOR LCP EXECUTION.
+ * -------------------------------------------------------------------------- */
+void Dblqh::execTUP_PREPLCPCONF(Signal* signal)
+{
+ UintR ttupPtr;
+
+ jamEntry();
+ lcpLocptr.i = signal->theData[0];
+ ttupPtr = signal->theData[1];
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+
+ lcpPtr.i = lcpLocptr.p->masterLcpRec;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ ndbrequire(lcpLocptr.p->lcpLocstate == LcpLocRecord::WAIT_TUP_PREPLCP);
+
+ lcpLocptr.p->tupRef = ttupPtr;
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::IDLE;
+ checkLcpTupprep(signal);
+ if (lcpPtr.p->lcpState != LcpRecord::LCP_WAIT_HOLDOPS) {
+ jam();
+ return;
+ }//if
+ fragptr.i = lcpPtr.p->currentFragment.fragPtrI;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ lcpLocptr.i = lcpPtr.p->firstLcpLocAcc;
+ do {
+ jam();
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::WAIT_LCPHOLDOP;
+ signal->theData[0] = lcpPtr.p->lcpAccptr;
+ signal->theData[1] = lcpLocptr.p->locFragid;
+ signal->theData[2] = 0;
+ signal->theData[3] = lcpLocptr.i;
+ sendSignal(fragptr.p->accBlockref, GSN_LCP_HOLDOPREQ, signal, 4, JBA);
+ lcpLocptr.i = lcpLocptr.p->nextLcpLoc;
+ } while (lcpLocptr.i != RNIL);
+ /* ------------------------------------------------------------------------
+ * SET STATE ON FRAGMENT TO BLOCKED TO ENSURE THAT NO MORE OPERATIONS ARE
+ * STARTED FROM LQH IN TUP AND ACC UNTIL THE START CHECKPOINT HAS BEEN
+ * COMPLETED. ALSO SET THE LOCAL CHECKPOINT STATE TO WAIT FOR
+ * LCP_HOLDOPCONF
+ * ----------------------------------------------------------------------- */
+ fragptr.p->fragStatus = Fragrecord::BLOCKED;
+ fragptr.p->fragActiveStatus = ZTRUE;
+ lcpPtr.p->lcpState = LcpRecord::LCP_WAIT_HOLDOPS;
+ return;
+}//Dblqh::execTUP_PREPLCPCONF()
+
+void Dblqh::execTUP_PREPLCPREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execTUP_PREPLCPREF()
+
+void Dblqh::execLCP_FRAGIDREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execLCP_FRAGIDREF()
+
+/* --------------------------------------------------------------------------
+ * A NUMBER OF OPERATIONS THAT HAVE BEEN SET ON HOLD IN ACC. MOVE THOSE TO
+ * LIST OF BLOCKED ACC OPERATIONS. IF MORE OPERATIONS ARE BLOCKED GET THOSE
+ * OTHERWISE CONTINUE THE LOCAL CHECKPOINT BY REQUESTING TUP AND ACC TO
+ * WRITE THEIR START CHECKPOINT.
+ * --------------------------------------------------------------------------
+ * PRECONDITION: LCP_LOCPTR:LCP_LOCSTATE = WAIT_LCPHOLDOP
+ * ------------------------------------------------------------------------- */
+/* ***************>> */
+/* LCP_HOLDOPCONF > */
+/* ***************>> */
+void Dblqh::execLCP_HOLDOPCONF(Signal* signal)
+{
+ UintR tnoHoldops;
+ Uint32 Tdata[23];
+ Uint32 Tlength;
+
+ jamEntry();
+ lcpLocptr.i = signal->theData[0];
+ Tlength = signal->theData[1];
+ for (Uint32 i = 0; i < 23; i++)
+ Tdata[i] = signal->theData[i + 2];
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ ndbrequire(lcpLocptr.p->lcpLocstate == LcpLocRecord::WAIT_LCPHOLDOP);
+
+ lcpPtr.i = lcpLocptr.p->masterLcpRec;
+ /* ------------------------------------------------------------------------
+ * NO ERROR CHECK ON USING VALUE IN MASTER_LCP_REC. ERROR IN THIS
+ * REFERENCE WILL CAUSE POINTER OUT OF RANGE WHICH CAUSES A SYSTEM RESTART.
+ * ----------------------------------------------------------------------- */
+ tnoHoldops = Tlength & 65535;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ fragptr.i = lcpPtr.p->currentFragment.fragPtrI;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ ndbrequire(tnoHoldops <= 23);
+ for (Uint32 Tindex = 0; Tindex < tnoHoldops; Tindex++) {
+ jam();
+ tcConnectptr.i = Tdata[Tindex];
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ moveActiveToAcc(signal);
+ }//for
+ if ((Tlength >> 16) == 1) {
+ jam();
+ /* MORE HOLDOPS NEEDED */
+ signal->theData[0] = lcpPtr.p->lcpAccptr;
+ signal->theData[1] = lcpLocptr.p->locFragid;
+ signal->theData[2] = 1;
+ signal->theData[3] = lcpLocptr.i;
+ sendSignal(fragptr.p->accBlockref, GSN_LCP_HOLDOPREQ, signal, 4, JBA);
+ return;
+ } else {
+ jam();
+ /* NO MORE HOLDOPS NEEDED */
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::HOLDOP_READY;
+ checkLcpHoldop(signal);
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_WAIT_ACTIVE_FINISH) {
+ if (fragptr.p->activeList == RNIL) {
+ jam();
+ /* ------------------------------------------------------------------
+ * THERE ARE NO MORE ACTIVE OPERATIONS. IT IS NOW OK TO START THE
+ * LOCAL CHECKPOINT IN BOTH TUP AND ACC.
+ * ----------------------------------------------------------------- */
+ sendStartLcp(signal);
+ lcpPtr.p->lcpState = LcpRecord::LCP_START_CHKP;
+ } else {
+ jam();
+ // Set this to signal releaseActiveFrag
+ // that it should check to see if itäs time to call sendStartLcp
+ fragptr.p->lcpRef = lcpPtr.i;
+ }//if
+ }//if
+ }//if
+ /* ----------------------- */
+ /* ELSE */
+ /* ------------------------------------------------------------------------
+ * THERE ARE STILL MORE ACTIVE OPERATIONS. WAIT UNTIL THEY ARE FINSIHED.
+ * THIS IS DISCOVERED WHEN RELEASE_ACTIVE_FRAG IS EXECUTED.
+ * ------------------------------------------------------------------------
+ * DO NOTHING, EXIT IS EXECUTED BELOW
+ * ----------------------------------------------------------------------- */
+ return;
+}//Dblqh::execLCP_HOLDOPCONF()
+
+/* ***************> */
+/* LCP_HOLDOPREF > */
+/* ***************> */
+void Dblqh::execLCP_HOLDOPREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execLCP_HOLDOPREF()
+
+/* ************************************************************************>>
+ * ACC_LCPSTARTED: Confirm that ACC started local checkpoint and undo
+ * logging is on.
+ * ************************************************************************>>
+ * --------------------------------------------------------------------------
+ * PRECONDITION: LCP_LOCPTR:LCP_LOCSTATE = ACC_WAIT_STARTED
+ * ------------------------------------------------------------------------- */
+void Dblqh::execACC_LCPSTARTED(Signal* signal)
+{
+ jamEntry();
+ lcpLocptr.i = signal->theData[0];
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ ndbrequire(lcpLocptr.p->lcpLocstate == LcpLocRecord::ACC_WAIT_STARTED);
+
+ lcpPtr.i = lcpLocptr.p->masterLcpRec;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ /* ------------------------------------------------------------------------
+ * NO ERROR CHECK ON USING VALUE IN MASTER_LCP_REC. ERROR IN THIS
+ * REFERENCE WILL CAUSE POINTER OUT OF RANGE WHICH CAUSES A SYSTEM RESTART.
+ * ----------------------------------------------------------------------- */
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::ACC_STARTED;
+ lcpStartedLab(signal);
+ return;
+}//Dblqh::execACC_LCPSTARTED()
+
+/* ******************************************> */
+/* TUP_LCPSTARTED: Same as above but for TUP. */
+/* ******************************************> */
+/* --------------------------------------------------------------------------
+ * PRECONDITION: LCP_LOCPTR:LCP_LOCSTATE = TUP_WAIT_STARTED
+ * ------------------------------------------------------------------------- */
+void Dblqh::execTUP_LCPSTARTED(Signal* signal)
+{
+ jamEntry();
+ lcpLocptr.i = signal->theData[0];
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ ndbrequire(lcpLocptr.p->lcpLocstate == LcpLocRecord::TUP_WAIT_STARTED);
+
+ lcpPtr.i = lcpLocptr.p->masterLcpRec;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ /* ------------------------------------------------------------------------
+ * NO ERROR CHECK ON USING VALUE IN MASTER_LCP_REC. ERROR IN THIS REFERENCE
+ * WILL CAUSE POINTER OUT OF RANGE WHICH CAUSES A SYSTEM RESTART.
+ * ----------------------------------------------------------------------- */
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::TUP_STARTED;
+ lcpStartedLab(signal);
+ return;
+}//Dblqh::execTUP_LCPSTARTED()
+
+void Dblqh::lcpStartedLab(Signal* signal)
+{
+ checkLcpStarted(signal);
+
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_STARTED) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THE LOCAL CHECKPOINT HAS BEEN STARTED. IT IS NOW TIME TO
+ * RESTART THE TRANSACTIONS WHICH HAVE BEEN BLOCKED.
+ * --------------------------------------------------------------------- */
+ fragptr.i = lcpPtr.p->currentFragment.fragPtrI;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ /* ----------------------------------------------------------------------
+ * UPDATE THE MAX_GCI_IN_LCP AND MAX_GCI_COMPLETED_IN_LCP NOW BEFORE
+ * ACTIVATING THE FRAGMENT AGAIN.
+ * --------------------------------------------------------------------- */
+ ndbrequire(lcpPtr.p->currentFragment.lcpFragOrd.lcpNo < MAX_LCP_STORED);
+ fragptr.p->maxGciInLcp = fragptr.p->newestGci;
+ fragptr.p->maxGciCompletedInLcp = cnewestCompletedGci;
+ sendAccContOp(signal); /* START OPERATIONS IN ACC */
+ moveAccActiveFrag(signal); /* MOVE FROM ACC BLOCKED LIST TO ACTIVE LIST
+ ON FRAGMENT */
+ }//if
+ /*---------------*/
+ /* ELSE */
+ /*-------------------------------------------------------------------------*/
+ /* THE LOCAL CHECKPOINT HAS NOT BEEN STARTED. EXIT AND WAIT FOR
+ * MORE SIGNALS */
+ /*-------------------------------------------------------------------------*/
+ /* DO NOTHING, EXIT IS EXECUTED BELOW */
+ /*-------------------------------------------------------------------------*/
+ return;
+}//Dblqh::lcpStartedLab()
+
+/*---------------------------------------------------------------------------
+ * ACC HAVE RESTARTED THE BLOCKED OPERATIONS AGAIN IN ONE FRAGMENT PART.
+ * IT IS NOW OUR TURN TO RESTART ALL OPERATIONS QUEUED IN LQH IF ALL
+ * FRAGMENT PARTS ARE COMPLETED.
+ *-------------------------------------------------------------------------- */
+void Dblqh::execACC_CONTOPCONF(Signal* signal)
+{
+ jamEntry();
+ lcpLocptr.i = signal->theData[0];
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ lcpLocptr.p->accContCounter = 1;
+
+ lcpPtr.i = lcpLocptr.p->masterLcpRec;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ lcpLocptr.i = lcpPtr.p->firstLcpLocAcc;
+ do {
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (lcpLocptr.p->accContCounter == 0) {
+ jam();
+ return;
+ }//if
+ lcpLocptr.i = lcpLocptr.p->nextLcpLoc;
+ } while (lcpLocptr.i != RNIL);
+ fragptr.i = lcpPtr.p->currentFragment.fragPtrI;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ restartOperationsLab(signal);
+ return;
+}//Dblqh::execACC_CONTOPCONF()
+
+/* ********************************************************* */
+/* LQH_RESTART_OP: Restart operations after beeing blocked. */
+/* ********************************************************* */
+/*---------------------------------------------------------------------------*/
+/* PRECONDITION: FRAG_STATUS = BLOCKED AND LCP_STATE = STARTED */
+/*---------------------------------------------------------------------------*/
+void Dblqh::execLQH_RESTART_OP(Signal* signal)
+{
+ jamEntry();
+ fragptr.i = signal->theData[0];
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+
+ lcpPtr.i = signal->theData[1];
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ if (fragptr.p->fragStatus == Fragrecord::BLOCKED) {
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_STARTED) {
+ jam();
+ /***********************************************************************/
+ /* THIS SIGNAL CAN ONLY BE RECEIVED WHEN FRAGMENT IS BLOCKED AND
+ * THE LOCAL CHECKPOINT HAS BEEN STARTED. THE BLOCKING WILL BE
+ * REMOVED AS SOON AS ALL OPERATIONS HAVE BEEN STARTED.
+ ***********************************************************************/
+ restartOperationsLab(signal);
+ return;
+ } else {
+ jam();
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_BLOCKED_COMP) {
+ jam();
+ /*******************************************************************>
+ * THE CHECKPOINT IS COMPLETED BUT HAS NOT YET STARTED UP
+ * ALL OPERATIONS AGAIN.
+ * WE PERFORM THIS START-UP BEFORE CONTINUING WITH THE NEXT
+ * FRAGMENT OF THE LOCAL CHECKPOINT TO AVOID ANY STRANGE ERRORS.
+ *******************************************************************> */
+ restartOperationsLab(signal);
+ return;
+ }//if
+ }//if
+ }//if
+ ndbrequire(false);
+}//Dblqh::execLQH_RESTART_OP()
+
+void Dblqh::restartOperationsLab(Signal* signal)
+{
+ Uint32 loopCount = 0;
+ tcConnectptr.i = fragptr.p->firstWaitQueue;
+ do {
+ if (tcConnectptr.i != RNIL) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* START UP THE TRANSACTION AGAIN. WE START IT AS A SEPARATE SIGNAL. */
+/*---------------------------------------------------------------------------*/
+ signal->theData[0] = ZRESTART_OPERATIONS_AFTER_STOP;
+ signal->theData[1] = tcConnectptr.i;
+ signal->theData[2] = fragptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ tcConnectptr.i = tcConnectptr.p->nextTc;
+ } else {
+ jam();
+/*--------------------------------------------------------------------------*/
+/* NO MORE OPERATIONS TO RESTART. WE CAN NOW RESET THE STATE TO ACTIVE AND */
+/* RESTART NORMAL ACTIVITIES ON THE FRAGMENT WHILE THE FUZZY PART OF THE */
+/* LOCAL CHECKPOINT IS COMPLETING. */
+/* IF THE CHECKPOINT WAS COMPLETED ALREADY ON THIS FRAGMENT WE PROCEED WITH */
+/* THE NEXT FRAGMENT NOW THAT WE HAVE COMPLETED THIS CHECKPOINT. */
+/*--------------------------------------------------------------------------*/
+ fragptr.p->fragStatus = Fragrecord::FSACTIVE;
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_BLOCKED_COMP) {
+ jam();
+ contChkpNextFragLab(signal);
+ return;
+ }//if
+ return;
+ }//if
+ loopCount++;
+ if (loopCount > 16) {
+ jam();
+ signal->theData[0] = fragptr.i;
+ signal->theData[1] = lcpPtr.i;
+ sendSignal(cownref, GSN_LQH_RESTART_OP, signal, 2, JBB);
+ return;
+ }//if
+ } while (1);
+}//Dblqh::restartOperationsLab()
+
+void Dblqh::restartOperationsAfterStopLab(Signal* signal)
+{
+ /*-------------------------------------------------------------------------
+ * WHEN ARRIVING HERE THE OPERATION IS ALREADY SET IN THE ACTIVE LIST.
+ * THUS WE CAN IMMEDIATELY CALL THE METHODS THAT EXECUTE FROM WHERE
+ * THE OPERATION WAS STOPPED.
+ *------------------------------------------------------------------------- */
+ switch (tcConnectptr.p->transactionState) {
+ case TcConnectionrec::STOPPED:
+ jam();
+ /*-----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND ACCKEYREQ
+ *----------------------------------------------------------------------- */
+ prepareContinueAfterBlockedLab(signal);
+ return;
+ break;
+ case TcConnectionrec::COMMIT_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND ACC_COMMITREQ
+ * ---------------------------------------------------------------------- */
+ releaseActiveFrag(signal);
+ commitContinueAfterBlockedLab(signal);
+ return;
+ break;
+ case TcConnectionrec::ABORT_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND ACC_ABORTREQ
+ * ---------------------------------------------------------------------- */
+ abortContinueAfterBlockedLab(signal, true);
+ return;
+ break;
+ case TcConnectionrec::COPY_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND NEXT_SCANREQ DURING COPY FRAGMENT
+ * ---------------------------------------------------------------------- */
+ continueCopyAfterBlockedLab(signal);
+ return;
+ break;
+ case TcConnectionrec::COPY_FIRST_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND NEXT_SCANREQ DURING COPY FRAGMENT
+ * ---------------------------------------------------------------------- */
+ continueFirstCopyAfterBlockedLab(signal);
+ return;
+ break;
+ case TcConnectionrec::SCAN_FIRST_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND NEXT_SCANREQ DURING SCAN
+ * ---------------------------------------------------------------------- */
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED;
+ continueFirstScanAfterBlockedLab(signal);
+ return;
+ break;
+ case TcConnectionrec::SCAN_CHECK_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND NEXT_SCANREQ DURING SCAN
+ * ---------------------------------------------------------------------- */
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED;
+ continueAfterCheckLcpStopBlocked(signal);
+ return;
+ break;
+ case TcConnectionrec::SCAN_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND NEXT_SCANREQ DURING SCAN
+ * ---------------------------------------------------------------------- */
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED;
+ continueScanAfterBlockedLab(signal);
+ return;
+ break;
+ case TcConnectionrec::SCAN_RELEASE_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND NEXT_SCANREQ DURING RELEASE
+ * LOCKS IN SCAN
+ * ---------------------------------------------------------------------- */
+ tcConnectptr.p->transactionState = TcConnectionrec::SCAN_STATE_USED;
+ continueScanReleaseAfterBlockedLab(signal);
+ return;
+ break;
+ case TcConnectionrec::SCAN_CLOSE_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND NEXT_SCANREQ DURING CLOSE OF SCAN
+ * ---------------------------------------------------------------------- */
+ continueCloseScanAfterBlockedLab(signal);
+ return;
+ break;
+ case TcConnectionrec::COPY_CLOSE_STOPPED:
+ jam();
+ /* ----------------------------------------------------------------------
+ * STOPPED BEFORE TRYING TO SEND NEXT_SCANREQ DURING CLOSE OF COPY
+ * ---------------------------------------------------------------------- */
+ continueCloseCopyAfterBlockedLab(signal);
+ return;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+}//Dblqh::restartOperationsAfterStopLab()
+
+/* *************** */
+/* ACC_LCPCONF > */
+/* *************** */
+/*---------------------------------------------------------------------------
+ * PRECONDITION: LCP_LOCPTR:LCP_LOCSTATE = ACC_STARTED
+ *-------------------------------------------------------------------------- */
+void Dblqh::execACC_LCPCONF(Signal* signal)
+{
+ jamEntry();
+ lcpLocptr.i = signal->theData[0];
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ ndbrequire(lcpLocptr.p->lcpLocstate == LcpLocRecord::ACC_STARTED);
+
+ lcpPtr.i = lcpLocptr.p->masterLcpRec;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ /* ------------------------------------------------------------------------
+ * NO ERROR CHECK ON USING VALUE IN MASTER_LCP_REC. ERROR IN
+ * THIS REFERENCE WILL CAUSE POINTER OUT OF RANGE WHICH CAUSES A
+ * SYSTEM RESTART.
+ * ----------------------------------------------------------------------- */
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::ACC_COMPLETED;
+ lcpCompletedLab(signal);
+ return;
+}//Dblqh::execACC_LCPCONF()
+
+/* *************** */
+/* TUP_LCPCONF > */
+/* *************** */
+/* --------------------------------------------------------------------------
+ * PRECONDITION: LCP_LOCPTR:LCP_LOCSTATE = TUP_STARTED
+ * ------------------------------------------------------------------------- */
+void Dblqh::execTUP_LCPCONF(Signal* signal)
+{
+ jamEntry();
+ lcpLocptr.i = signal->theData[0];
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ ndbrequire(lcpLocptr.p->lcpLocstate == LcpLocRecord::TUP_STARTED);
+
+ lcpPtr.i = lcpLocptr.p->masterLcpRec;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ /* ------------------------------------------------------------------------
+ * NO ERROR CHECK ON USING VALUE IN MASTER_LCP_REC. ERROR IN THIS
+ * REFERENCE WILL CAUSE POINTER OUT OF RANGE WHICH CAUSES A SYSTEM RESTART.
+ * ----------------------------------------------------------------------- */
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::TUP_COMPLETED;
+ lcpCompletedLab(signal);
+ return;
+}//Dblqh::execTUP_LCPCONF()
+
+void Dblqh::lcpCompletedLab(Signal* signal)
+{
+ checkLcpCompleted(signal);
+ if (lcpPtr.p->lcpState != LcpRecord::LCP_COMPLETED) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THE LOCAL CHECKPOINT HAS NOT BEEN COMPLETED, EXIT & WAIT
+ * FOR MORE SIGNALS
+ * --------------------------------------------------------------------- */
+ return;
+ }//if
+ /* ------------------------------------------------------------------------
+ * THE LOCAL CHECKPOINT HAS BEEN COMPLETED. IT IS NOW TIME TO START
+ * A LOCAL CHECKPOINT ON THE NEXT FRAGMENT OR COMPLETE THIS LCP ROUND.
+ * ------------------------------------------------------------------------
+ * WE START BY SENDING LCP_REPORT TO DIH TO REPORT THE COMPLETED LCP.
+ * TO CATER FOR NODE CRASHES WE SEND IT IN PARALLEL TO ALL NODES.
+ * ----------------------------------------------------------------------- */
+ sendLCP_FRAG_REP(signal, lcpPtr.p->currentFragment);
+
+ fragptr.i = lcpPtr.p->currentFragment.fragPtrI;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ fragptr.p->fragActiveStatus = ZFALSE;
+
+ contChkpNextFragLab(signal);
+ return;
+}//Dblqh::lcpCompletedLab()
+
+void
+Dblqh::sendLCP_FRAG_REP(Signal * signal,
+ const LcpRecord::FragOrd & fragOrd) const {
+
+ FragrecordPtr fragPtr;
+ fragPtr.i = fragOrd.fragPtrI;
+ ptrCheckGuard(fragPtr, cfragrecFileSize, fragrecord);
+
+ ndbrequire(fragOrd.lcpFragOrd.lcpNo < MAX_LCP_STORED);
+ LcpFragRep * const lcpReport = (LcpFragRep *)&signal->theData[0];
+ lcpReport->nodeId = cownNodeid;
+ lcpReport->lcpId = fragOrd.lcpFragOrd.lcpId;
+ lcpReport->lcpNo = fragOrd.lcpFragOrd.lcpNo;
+ lcpReport->tableId = fragOrd.lcpFragOrd.tableId;
+ lcpReport->fragId = fragOrd.lcpFragOrd.fragmentId;
+ lcpReport->maxGciCompleted = fragPtr.p->maxGciCompletedInLcp;
+ lcpReport->maxGciStarted = fragPtr.p->maxGciInLcp;
+
+ for (Uint32 i = 0; i < cnoOfNodes; i++) {
+ jam();
+ Uint32 nodeId = cnodeData[i];
+ if(cnodeStatus[i] == ZNODE_UP){
+ jam();
+ BlockReference Tblockref = calcDihBlockRef(nodeId);
+ sendSignal(Tblockref, GSN_LCP_FRAG_REP, signal,
+ LcpFragRep::SignalLength, JBB);
+ }//if
+ }//for
+}
+
+void Dblqh::contChkpNextFragLab(Signal* signal)
+{
+ /* ------------------------------------------------------------------------
+ * UPDATE THE LATEST LOCAL CHECKPOINT COMPLETED ON FRAGMENT.
+ * UPDATE THE LCP_ID OF THIS CHECKPOINT.
+ * REMOVE THE LINK BETWEEN THE FRAGMENT RECORD AND THE LCP RECORD.
+ * ----------------------------------------------------------------------- */
+ if (fragptr.p->fragStatus == Fragrecord::BLOCKED) {
+ jam();
+ lcpPtr.p->lcpState = LcpRecord::LCP_BLOCKED_COMP;
+ return;
+ }//if
+ /* ------------------------------------------------------------------------
+ * WE ALSO RELEASE THE LOCAL LCP RECORDS.
+ * ----------------------------------------------------------------------- */
+ releaseLocalLcps(signal);
+ if (lcpPtr.p->lcpQueued) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * Transfer the state from the queued to the active LCP.
+ * --------------------------------------------------------------------- */
+ lcpPtr.p->lcpQueued = false;
+ lcpPtr.p->currentFragment = lcpPtr.p->queuedFragment;
+
+ /* ----------------------------------------------------------------------
+ * START THE QUEUED LOCAL CHECKPOINT.
+ * --------------------------------------------------------------------- */
+ sendLCP_FRAGIDREQ(signal);
+ return;
+ }//if
+
+ lcpPtr.p->lcpState = LcpRecord::LCP_IDLE;
+ if (lcpPtr.p->lastFragmentFlag){
+ jam();
+ /* ----------------------------------------------------------------------
+ * NOW THE COMPLETE LOCAL CHECKPOINT ROUND IS COMPLETED.
+ * --------------------------------------------------------------------- */
+ completeLcpRoundLab(signal);
+ return;
+ }//if
+
+ if (lcpPtr.p->reportEmpty) {
+ jam();
+ sendEMPTY_LCP_CONF(signal, false);
+ }//if
+ return;
+}//Dblqh::contChkpNextFragLab()
+
+void Dblqh::sendLCP_FRAGIDREQ(Signal* signal)
+{
+ ndbrequire(lcpPtr.p->firstLcpLocTup == RNIL);
+ ndbrequire(lcpPtr.p->firstLcpLocAcc == RNIL);
+
+ TablerecPtr tabPtr;
+ tabPtr.i = lcpPtr.p->currentFragment.lcpFragOrd.tableId;
+ ptrAss(tabPtr, tablerec);
+ if(tabPtr.p->tableStatus == Tablerec::PREP_DROP_TABLE_ONGOING ||
+ tabPtr.p->tableStatus == Tablerec::PREP_DROP_TABLE_DONE){
+ jam();
+ /**
+ * Fake that the fragment is done
+ */
+ lcpCompletedLab(signal);
+ return;
+ }
+
+ ndbrequire(tabPtr.p->tableStatus == Tablerec::TABLE_DEFINED);
+
+ lcpPtr.p->lcpState = LcpRecord::LCP_WAIT_FRAGID;
+ signal->theData[0] = lcpPtr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lcpPtr.p->currentFragment.lcpFragOrd.lcpNo;
+ signal->theData[3] = lcpPtr.p->currentFragment.lcpFragOrd.tableId;
+ signal->theData[4] = lcpPtr.p->currentFragment.lcpFragOrd.fragmentId;
+ signal->theData[5] = lcpPtr.p->currentFragment.lcpFragOrd.lcpId % MAX_LCP_STORED;
+ sendSignal(fragptr.p->accBlockref, GSN_LCP_FRAGIDREQ, signal, 6, JBB);
+}//Dblqh::sendLCP_FRAGIDREQ()
+
+void Dblqh::sendEMPTY_LCP_CONF(Signal* signal, bool idle)
+{
+
+ EmptyLcpConf * const rep = (EmptyLcpConf*)&signal->theData[0];
+ /* ----------------------------------------------------------------------
+ * We have been requested to report when there are no more local
+ * waiting to be started or ongoing. In this signal we also report
+ * the last completed fragments state.
+ * ---------------------------------------------------------------------- */
+ rep->senderNodeId = getOwnNodeId();
+ if(!idle){
+ jam();
+ rep->idle = 0 ;
+ rep->tableId = lcpPtr.p->currentFragment.lcpFragOrd.tableId;
+ rep->fragmentId = lcpPtr.p->currentFragment.lcpFragOrd.fragmentId;
+ rep->lcpNo = lcpPtr.p->currentFragment.lcpFragOrd.lcpNo;
+ rep->lcpId = lcpPtr.p->currentFragment.lcpFragOrd.lcpId;
+ } else {
+ jam();
+ rep->idle = 1;
+ rep->tableId = ~0;
+ rep->fragmentId = ~0;
+ rep->lcpNo = ~0;
+ rep->lcpId = c_lcpId;
+ }
+
+ for (Uint32 i = 0; i < cnoOfNodes; i++) {
+ jam();
+ Uint32 nodeId = cnodeData[i];
+ if (lcpPtr.p->m_EMPTY_LCP_REQ.get(nodeId)) {
+ jam();
+
+ BlockReference blockref = calcDihBlockRef(nodeId);
+ sendSignal(blockref, GSN_EMPTY_LCP_CONF, signal,
+ EmptyLcpConf::SignalLength, JBB);
+ }//if
+ }//for
+
+ lcpPtr.p->reportEmpty = false;
+ lcpPtr.p->m_EMPTY_LCP_REQ.clear();
+}//Dblqh::sendEMPTY_LCPCONF()
+
+void Dblqh::execACC_LCPREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execACC_LCPREF()
+
+void Dblqh::execTUP_LCPREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execTUP_LCPREF()
+
+/* --------------------------------------------------------------------------
+ * THE LOCAL CHECKPOINT ROUND IS NOW COMPLETED. SEND COMPLETED MESSAGE
+ * TO THE MASTER DIH.
+ * ------------------------------------------------------------------------- */
+void Dblqh::completeLcpRoundLab(Signal* signal)
+{
+ clcpCompletedState = LCP_CLOSE_STARTED;
+ signal->theData[0] = caccBlockref;
+ signal->theData[1] = cownref;
+ sendSignal(caccBlockref, GSN_END_LCPREQ, signal, 2, JBB);
+ signal->theData[0] = ctupBlockref;
+ signal->theData[1] = cownref;
+ sendSignal(ctupBlockref, GSN_END_LCPREQ, signal, 2, JBB);
+ return;
+}//Dblqh::completeLcpRoundLab()
+
+void Dblqh::execEND_LCPCONF(Signal* signal)
+{
+ jamEntry();
+ BlockReference userpointer = signal->theData[0];
+ if (userpointer == caccBlockref) {
+ if (clcpCompletedState == LCP_CLOSE_STARTED) {
+ jam();
+ clcpCompletedState = ACC_LCP_CLOSE_COMPLETED;
+ return;
+ } else {
+ jam();
+ ndbrequire(clcpCompletedState == TUP_LCP_CLOSE_COMPLETED);
+ clcpCompletedState = LCP_IDLE;
+ }//if
+ } else {
+ ndbrequire(userpointer == ctupBlockref);
+ if (clcpCompletedState == LCP_CLOSE_STARTED) {
+ jam();
+ clcpCompletedState = TUP_LCP_CLOSE_COMPLETED;
+ return;
+ } else {
+ jam();
+ ndbrequire(clcpCompletedState == ACC_LCP_CLOSE_COMPLETED);
+ clcpCompletedState = LCP_IDLE;
+ }//if
+ }//if
+ sendLCP_COMPLETE_REP(signal, lcpPtr.p->currentFragment.lcpFragOrd.lcpId);
+}//Dblqh::execEND_LCPCONF()
+
+void Dblqh::sendLCP_COMPLETE_REP(Signal* signal, Uint32 lcpId)
+{
+ cnoOfFragsCheckpointed = 0;
+ ndbrequire((cnoOfNodes - 1) < (MAX_NDB_NODES - 1));
+ /* ------------------------------------------------------------------------
+ * WE SEND COMP_LCP_ROUND TO ALL NODES TO PREPARE FOR NODE CRASHES.
+ * ----------------------------------------------------------------------- */
+ lcpPtr.i = 0;
+ ptrAss(lcpPtr, lcpRecord);
+ lcpPtr.p->lastFragmentFlag = false;
+
+ LcpCompleteRep* rep = (LcpCompleteRep*)signal->getDataPtrSend();
+ rep->nodeId = getOwnNodeId();
+ rep->lcpId = lcpId;
+ rep->blockNo = DBLQH;
+
+ for (Uint32 i = 0; i < cnoOfNodes; i++) {
+ jam();
+ Uint32 nodeId = cnodeData[i];
+ if(cnodeStatus[i] == ZNODE_UP){
+ jam();
+
+ BlockReference blockref = calcDihBlockRef(nodeId);
+ sendSignal(blockref, GSN_LCP_COMPLETE_REP, signal,
+ LcpCompleteRep::SignalLength, JBB);
+ }//if
+ }//for
+
+ if(lcpPtr.p->reportEmpty){
+ jam();
+ sendEMPTY_LCP_CONF(signal, true);
+ }
+ return;
+}//Dblqh::sendCOMP_LCP_ROUND()
+
+/* ==========================================================================
+ * ======= CHECK IF ALL PARTS OF A LOCAL CHECKPOINT ARE COMPLETED =======
+ *
+ * SUBROUTINE SHORT NAME = CLC
+ * ========================================================================= */
+void Dblqh::checkLcpCompleted(Signal* signal)
+{
+ LcpLocRecordPtr clcLcpLocptr;
+
+ clcLcpLocptr.i = lcpPtr.p->firstLcpLocAcc;
+ while (clcLcpLocptr.i != RNIL) {
+ ptrCheckGuard(clcLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (clcLcpLocptr.p->lcpLocstate != LcpLocRecord::ACC_COMPLETED) {
+ jam();
+ ndbrequire((clcLcpLocptr.p->lcpLocstate == LcpLocRecord::ACC_WAIT_STARTED) ||
+ (clcLcpLocptr.p->lcpLocstate == LcpLocRecord::ACC_STARTED));
+ return;
+ }//if
+ clcLcpLocptr.i = clcLcpLocptr.p->nextLcpLoc;
+ }
+
+ clcLcpLocptr.i = lcpPtr.p->firstLcpLocTup;
+ while (clcLcpLocptr.i != RNIL){
+ ptrCheckGuard(clcLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (clcLcpLocptr.p->lcpLocstate != LcpLocRecord::TUP_COMPLETED) {
+ jam();
+ ndbrequire((clcLcpLocptr.p->lcpLocstate==LcpLocRecord::TUP_WAIT_STARTED)
+ ||(clcLcpLocptr.p->lcpLocstate == LcpLocRecord::TUP_STARTED));
+ return;
+ }//if
+ clcLcpLocptr.i = clcLcpLocptr.p->nextLcpLoc;
+ }
+
+ lcpPtr.p->lcpState = LcpRecord::LCP_COMPLETED;
+}//Dblqh::checkLcpCompleted()
+
+/* ==========================================================================
+ * ======= CHECK IF ALL HOLD OPERATIONS ARE COMPLETED =======
+ *
+ * SUBROUTINE SHORT NAME = CHO
+ * ========================================================================= */
+void Dblqh::checkLcpHoldop(Signal* signal)
+{
+ LcpLocRecordPtr choLcpLocptr;
+
+ choLcpLocptr.i = lcpPtr.p->firstLcpLocAcc;
+ do {
+ ptrCheckGuard(choLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (choLcpLocptr.p->lcpLocstate != LcpLocRecord::HOLDOP_READY) {
+ ndbrequire(choLcpLocptr.p->lcpLocstate == LcpLocRecord::WAIT_LCPHOLDOP);
+ return;
+ }//if
+ choLcpLocptr.i = choLcpLocptr.p->nextLcpLoc;
+ } while (choLcpLocptr.i != RNIL);
+ lcpPtr.p->lcpState = LcpRecord::LCP_WAIT_ACTIVE_FINISH;
+}//Dblqh::checkLcpHoldop()
+
+/* ==========================================================================
+ * ======= CHECK IF ALL PARTS OF A LOCAL CHECKPOINT ARE STARTED =======
+ *
+ * SUBROUTINE SHORT NAME = CLS
+ * ========================================================================== */
+void Dblqh::checkLcpStarted(Signal* signal)
+{
+ LcpLocRecordPtr clsLcpLocptr;
+
+ terrorCode = ZOK;
+ clsLcpLocptr.i = lcpPtr.p->firstLcpLocAcc;
+ do {
+ ptrCheckGuard(clsLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (clsLcpLocptr.p->lcpLocstate != LcpLocRecord::ACC_STARTED) {
+ ndbrequire((clsLcpLocptr.p->lcpLocstate == LcpLocRecord::ACC_COMPLETED) ||
+ (clsLcpLocptr.p->lcpLocstate == LcpLocRecord::ACC_WAIT_STARTED));
+ return;
+ }//if
+ clsLcpLocptr.i = clsLcpLocptr.p->nextLcpLoc;
+ } while (clsLcpLocptr.i != RNIL);
+
+ clsLcpLocptr.i = lcpPtr.p->firstLcpLocTup;
+ do {
+ ptrCheckGuard(clsLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (clsLcpLocptr.p->lcpLocstate != LcpLocRecord::TUP_STARTED) {
+ ndbrequire((clsLcpLocptr.p->lcpLocstate == LcpLocRecord::TUP_COMPLETED) ||
+ (clsLcpLocptr.p->lcpLocstate == LcpLocRecord::TUP_WAIT_STARTED));
+ return;
+ }//if
+ clsLcpLocptr.i = clsLcpLocptr.p->nextLcpLoc;
+ } while (clsLcpLocptr.i != RNIL);
+ lcpPtr.p->lcpState = LcpRecord::LCP_STARTED;
+}//Dblqh::checkLcpStarted()
+
+/* ==========================================================================
+ * ======= CHECK IF ALL PREPARE TUP OPERATIONS ARE COMPLETED =======
+ *
+ * SUBROUTINE SHORT NAME = CLT
+ * ========================================================================== */
+void Dblqh::checkLcpTupprep(Signal* signal)
+{
+ LcpLocRecordPtr cltLcpLocptr;
+ cltLcpLocptr.i = lcpPtr.p->firstLcpLocTup;
+ do {
+ ptrCheckGuard(cltLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (cltLcpLocptr.p->lcpLocstate != LcpLocRecord::IDLE) {
+ ndbrequire(cltLcpLocptr.p->lcpLocstate == LcpLocRecord::WAIT_TUP_PREPLCP);
+ return;
+ }//if
+ cltLcpLocptr.i = cltLcpLocptr.p->nextLcpLoc;
+ } while (cltLcpLocptr.i != RNIL);
+ lcpPtr.p->lcpState = LcpRecord::LCP_WAIT_HOLDOPS;
+}//Dblqh::checkLcpTupprep()
+
+/* ==========================================================================
+ * ======= INITIATE LCP LOCAL RECORD USED TOWARDS ACC =======
+ *
+ * ========================================================================== */
+void Dblqh::initLcpLocAcc(Signal* signal, Uint32 fragId)
+{
+ lcpLocptr.p->nextLcpLoc = lcpPtr.p->firstLcpLocAcc;
+ lcpPtr.p->firstLcpLocAcc = lcpLocptr.i;
+ lcpLocptr.p->locFragid = fragId;
+ lcpLocptr.p->waitingBlock = LcpLocRecord::ACC;
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::IDLE;
+ lcpLocptr.p->masterLcpRec = lcpPtr.i;
+ lcpLocptr.p->tupRef = RNIL;
+}//Dblqh::initLcpLocAcc()
+
+/* ==========================================================================
+ * ======= INITIATE LCP LOCAL RECORD USED TOWARDS TUP =======
+ *
+ * ========================================================================== */
+void Dblqh::initLcpLocTup(Signal* signal, Uint32 fragId)
+{
+ lcpLocptr.p->nextLcpLoc = lcpPtr.p->firstLcpLocTup;
+ lcpPtr.p->firstLcpLocTup = lcpLocptr.i;
+ lcpLocptr.p->locFragid = fragId;
+ lcpLocptr.p->waitingBlock = LcpLocRecord::TUP;
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::WAIT_TUP_PREPLCP;
+ lcpLocptr.p->masterLcpRec = lcpPtr.i;
+ lcpLocptr.p->tupRef = RNIL;
+}//Dblqh::initLcpLocTup()
+
+/* --------------------------------------------------------------------------
+ * ------- MOVE OPERATION FROM ACC WAITING LIST ON FRAGMENT -------
+ * ------- TO ACTIVE LIST ON FRAGMENT -------
+ *
+ * SUBROUTINE SHORT NAME = MAA
+ * -------------------------------------------------------------------------- */
+void Dblqh::moveAccActiveFrag(Signal* signal)
+{
+ UintR maaTcNextConnectptr;
+
+ tcConnectptr.i = fragptr.p->accBlockedList;
+ fragptr.p->accBlockedList = RNIL;
+ /* ------------------------------------------------------------------------
+ * WE WILL MOVE ALL RECORDS FROM THE ACC BLOCKED LIST AT ONCE.
+ * ------------------------------------------------------------------------ */
+ while (tcConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ maaTcNextConnectptr = tcConnectptr.p->nextTc;
+ ndbrequire(tcConnectptr.p->listState == TcConnectionrec::ACC_BLOCK_LIST);
+ tcConnectptr.p->listState = TcConnectionrec::NOT_IN_LIST;
+ linkActiveFrag(signal);
+ tcConnectptr.i = maaTcNextConnectptr;
+ }//while
+}//Dblqh::moveAccActiveFrag()
+
+/* --------------------------------------------------------------------------
+ * ------- MOVE OPERATION FROM ACTIVE LIST ON FRAGMENT -------
+ * ------- TO ACC BLOCKED LIST ON FRAGMENT -------
+ *
+ * SUBROUTINE SHORT NAME = MAT
+ * -------------------------------------------------------------------------- */
+void Dblqh::moveActiveToAcc(Signal* signal)
+{
+ TcConnectionrecPtr matTcNextConnectptr;
+
+ releaseActiveList(signal);
+ /* ------------------------------------------------------------------------
+ * PUT OPERATION RECORD FIRST IN ACC BLOCKED LIST.
+ * ------------------------------------------------------------------------ */
+ matTcNextConnectptr.i = fragptr.p->accBlockedList;
+ tcConnectptr.p->nextTc = matTcNextConnectptr.i;
+ tcConnectptr.p->prevTc = RNIL;
+ tcConnectptr.p->listState = TcConnectionrec::ACC_BLOCK_LIST;
+ fragptr.p->accBlockedList = tcConnectptr.i;
+ if (matTcNextConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(matTcNextConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ matTcNextConnectptr.p->prevTc = tcConnectptr.i;
+ }//if
+}//Dblqh::moveActiveToAcc()
+
+/* ------------------------------------------------------------------------- */
+/* ---- RELEASE LOCAL LCP RECORDS AFTER COMPLETION OF A LOCAL CHECKPOINT---- */
+/* */
+/* SUBROUTINE SHORT NAME = RLL */
+/* ------------------------------------------------------------------------- */
+void Dblqh::releaseLocalLcps(Signal* signal)
+{
+ lcpLocptr.i = lcpPtr.p->firstLcpLocAcc;
+ while (lcpLocptr.i != RNIL){
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ Uint32 tmp = lcpLocptr.p->nextLcpLoc;
+ releaseLcpLoc(signal);
+ lcpLocptr.i = tmp;
+ }
+ lcpPtr.p->firstLcpLocAcc = RNIL;
+
+ lcpLocptr.i = lcpPtr.p->firstLcpLocTup;
+ while (lcpLocptr.i != RNIL){
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ Uint32 tmp = lcpLocptr.p->nextLcpLoc;
+ releaseLcpLoc(signal);
+ lcpLocptr.i = tmp;
+ }
+ lcpPtr.p->firstLcpLocTup = RNIL;
+
+}//Dblqh::releaseLocalLcps()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEIZE LCP LOCAL RECORD ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::seizeLcpLoc(Signal* signal)
+{
+ lcpLocptr.i = cfirstfreeLcpLoc;
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ cfirstfreeLcpLoc = lcpLocptr.p->nextLcpLoc;
+ lcpLocptr.p->nextLcpLoc = RNIL;
+}//Dblqh::seizeLcpLoc()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEND ACC_CONT_OP ------- */
+/* */
+/* INPUT: LCP_PTR LOCAL CHECKPOINT RECORD */
+/* FRAGPTR FRAGMENT RECORD */
+/* */
+/* SUBROUTINE SHORT NAME = SAC */
+/* ------------------------------------------------------------------------- */
+void Dblqh::sendAccContOp(Signal* signal)
+{
+ LcpLocRecordPtr sacLcpLocptr;
+
+ sacLcpLocptr.i = lcpPtr.p->firstLcpLocAcc;
+ do {
+ ptrCheckGuard(sacLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ sacLcpLocptr.p->accContCounter = 0;
+/* ------------------------------------------------------------------------- */
+/*SEND START OPERATIONS TO ACC AGAIN */
+/* ------------------------------------------------------------------------- */
+ signal->theData[0] = lcpPtr.p->lcpAccptr;
+ signal->theData[1] = sacLcpLocptr.p->locFragid;
+ sendSignal(fragptr.p->accBlockref, GSN_ACC_CONTOPREQ, signal, 2, JBA);
+ sacLcpLocptr.i = sacLcpLocptr.p->nextLcpLoc;
+ } while (sacLcpLocptr.i != RNIL);
+}//Dblqh::sendAccContOp()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEND ACC_LCPREQ AND TUP_LCPREQ ------- */
+/* */
+/* INPUT: LCP_PTR LOCAL CHECKPOINT RECORD */
+/* FRAGPTR FRAGMENT RECORD */
+/* SUBROUTINE SHORT NAME = STL */
+/* ------------------------------------------------------------------------- */
+void Dblqh::sendStartLcp(Signal* signal)
+{
+ LcpLocRecordPtr stlLcpLocptr;
+ stlLcpLocptr.i = lcpPtr.p->firstLcpLocAcc;
+ do {
+ jam();
+ ptrCheckGuard(stlLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ stlLcpLocptr.p->lcpLocstate = LcpLocRecord::ACC_WAIT_STARTED;
+ signal->theData[0] = lcpPtr.p->lcpAccptr;
+ signal->theData[1] = stlLcpLocptr.i;
+ signal->theData[2] = stlLcpLocptr.p->locFragid;
+ sendSignal(fragptr.p->accBlockref, GSN_ACC_LCPREQ, signal, 3, JBA);
+ stlLcpLocptr.i = stlLcpLocptr.p->nextLcpLoc;
+ } while (stlLcpLocptr.i != RNIL);
+
+ stlLcpLocptr.i = lcpPtr.p->firstLcpLocTup;
+ do {
+ jam();
+ ptrCheckGuard(stlLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ stlLcpLocptr.p->lcpLocstate = LcpLocRecord::TUP_WAIT_STARTED;
+ signal->theData[0] = stlLcpLocptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = stlLcpLocptr.p->tupRef;
+ sendSignal(fragptr.p->tupBlockref, GSN_TUP_LCPREQ, signal, 3, JBA);
+ stlLcpLocptr.i = stlLcpLocptr.p->nextLcpLoc;
+ } while (stlLcpLocptr.i != RNIL);
+}//Dblqh::sendStartLcp()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SET THE LOG TAIL IN THE LOG FILES ------- */
+/* */
+/*THIS SUBROUTINE HAVE BEEN BUGGY AND IS RATHER COMPLEX. IT IS IMPORTANT TO */
+/*REMEMBER THAT WE SEARCH FROM THE TAIL UNTIL WE REACH THE HEAD (CURRENT). */
+/*THE TAIL AND HEAD CAN BE ON THE SAME MBYTE. WE SEARCH UNTIL WE FIND A MBYTE*/
+/*THAT WE NEED TO KEEP. WE THEN SET THE TAIL TO BE THE PREVIOUS. IF WE DO */
+/*NOT FIND A MBYTE THAT WE NEED TO KEEP UNTIL WE REACH THE HEAD THEN WE USE */
+/*THE HEAD AS TAIL. FINALLY WE HAVE TO MOVE BACK THE TAIL TO ALSO INCLUDE */
+/*ALL PREPARE RECORDS. THIS MEANS THAT LONG-LIVED TRANSACTIONS ARE DANGEROUS */
+/*FOR SHORT LOGS. */
+/* ------------------------------------------------------------------------- */
+
+// this function has not been verified yet
+Uint32 Dblqh::remainingLogSize(const LogFileRecordPtr &sltCurrLogFilePtr,
+ const LogPartRecordPtr &sltLogPartPtr)
+{
+ Uint32 hf = sltCurrLogFilePtr.p->fileNo*ZNO_MBYTES_IN_FILE+sltCurrLogFilePtr.p->currentMbyte;
+ Uint32 tf = sltLogPartPtr.p->logTailFileNo*ZNO_MBYTES_IN_FILE+sltLogPartPtr.p->logTailMbyte;
+ Uint32 sz = sltLogPartPtr.p->noLogFiles*ZNO_MBYTES_IN_FILE;
+ if (tf > hf) hf += sz;
+ return sz-(hf-tf);
+}
+
+void Dblqh::setLogTail(Signal* signal, Uint32 keepGci)
+{
+ LogPartRecordPtr sltLogPartPtr;
+ LogFileRecordPtr sltLogFilePtr;
+#if 0
+ LogFileRecordPtr sltCurrLogFilePtr;
+#endif
+ UintR tsltMbyte;
+ UintR tsltStartMbyte;
+ UintR tsltIndex;
+ UintR tsltFlag;
+
+ for (sltLogPartPtr.i = 0; sltLogPartPtr.i < 4; sltLogPartPtr.i++) {
+ jam();
+ ptrAss(sltLogPartPtr, logPartRecord);
+ findLogfile(signal, sltLogPartPtr.p->logTailFileNo,
+ sltLogPartPtr, &sltLogFilePtr);
+
+#if 0
+ sltCurrLogFilePtr.i = sltLogPartPtr.p->currentLogfile;
+ ptrCheckGuard(sltCurrLogFilePtr, clogFileFileSize, logFileRecord);
+ infoEvent("setLogTail: Available log file %d size = %d[mbytes]+%d[words]", sltLogPartPtr.i,
+ remainingLogSize(sltCurrLogFilePtr, sltLogPartPtr), sltCurrLogFilePtr.p->remainingWordsInMbyte);
+#endif
+
+ tsltMbyte = sltLogPartPtr.p->logTailMbyte;
+ tsltStartMbyte = tsltMbyte;
+ tsltFlag = ZFALSE;
+ if (sltLogFilePtr.i == sltLogPartPtr.p->currentLogfile) {
+/* ------------------------------------------------------------------------- */
+/*THE LOG AND THE TAIL IS ALREADY IN THE SAME FILE. */
+/* ------------------------------------------------------------------------- */
+ if (sltLogFilePtr.p->currentMbyte >= sltLogPartPtr.p->logTailMbyte) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*THE CURRENT MBYTE IS AHEAD OF OR AT THE TAIL. THUS WE WILL ONLY LOOK FOR */
+/*THE TAIL UNTIL WE REACH THE CURRENT MBYTE WHICH IS IN THIS LOG FILE. */
+/*IF THE LOG TAIL IS AHEAD OF THE CURRENT MBYTE BUT IN THE SAME LOG FILE */
+/*THEN WE HAVE TO SEARCH THROUGH ALL FILES BEFORE WE COME TO THE CURRENT */
+/*MBYTE. WE ALWAYS STOP WHEN WE COME TO THE CURRENT MBYTE SINCE THE TAIL */
+/*CAN NEVER BE BEFORE THE HEAD. */
+/* ------------------------------------------------------------------------- */
+ tsltFlag = ZTRUE;
+ }//if
+ }//if
+
+/* ------------------------------------------------------------------------- */
+/*NOW START SEARCHING FOR THE NEW TAIL, STARTING AT THE CURRENT TAIL AND */
+/*PROCEEDING UNTIL WE FIND A MBYTE WHICH IS NEEDED TO KEEP OR UNTIL WE REACH */
+/*CURRENT MBYTE (THE HEAD). */
+/* ------------------------------------------------------------------------- */
+ SLT_LOOP:
+ for (tsltIndex = tsltStartMbyte;
+ tsltIndex <= ZNO_MBYTES_IN_FILE - 1;
+ tsltIndex++) {
+ if (sltLogFilePtr.p->logMaxGciStarted[tsltIndex] >= keepGci) {
+/* ------------------------------------------------------------------------- */
+/*WE ARE NOT ALLOWED TO STEP THE LOG ANY FURTHER AHEAD */
+/*SET THE NEW LOG TAIL AND CONTINUE WITH NEXT LOG PART. */
+/*THIS MBYTE IS NOT TO BE INCLUDED SO WE NEED TO STEP BACK ONE MBYTE. */
+/* ------------------------------------------------------------------------- */
+ if (tsltIndex != 0) {
+ jam();
+ tsltMbyte = tsltIndex - 1;
+ } else {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*STEPPING BACK INCLUDES ALSO STEPPING BACK TO THE PREVIOUS LOG FILE. */
+/* ------------------------------------------------------------------------- */
+ tsltMbyte = ZNO_MBYTES_IN_FILE - 1;
+ sltLogFilePtr.i = sltLogFilePtr.p->prevLogFile;
+ ptrCheckGuard(sltLogFilePtr, clogFileFileSize, logFileRecord);
+ }//if
+ goto SLT_BREAK;
+ } else {
+ jam();
+ if (tsltFlag == ZTRUE) {
+/* ------------------------------------------------------------------------- */
+/*WE ARE IN THE SAME FILE AS THE CURRENT MBYTE AND WE CAN REACH THE CURRENT */
+/*MBYTE BEFORE WE REACH A NEW TAIL. */
+/* ------------------------------------------------------------------------- */
+ if (tsltIndex == sltLogFilePtr.p->currentMbyte) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*THE TAIL OF THE LOG IS ACTUALLY WITHIN THE CURRENT MBYTE. THUS WE SET THE */
+/*LOG TAIL TO BE THE CURRENT MBYTE. */
+/* ------------------------------------------------------------------------- */
+ tsltMbyte = sltLogFilePtr.p->currentMbyte;
+ goto SLT_BREAK;
+ }//if
+ }//if
+ }//if
+ }//for
+ sltLogFilePtr.i = sltLogFilePtr.p->nextLogFile;
+ ptrCheckGuard(sltLogFilePtr, clogFileFileSize, logFileRecord);
+ if (sltLogFilePtr.i == sltLogPartPtr.p->currentLogfile) {
+ jam();
+ tsltFlag = ZTRUE;
+ }//if
+ tsltStartMbyte = 0;
+ goto SLT_LOOP;
+ SLT_BREAK:
+ jam();
+ {
+ UintR ToldTailFileNo = sltLogPartPtr.p->logTailFileNo;
+ UintR ToldTailMByte = sltLogPartPtr.p->logTailMbyte;
+
+ arrGuard(tsltMbyte, 16);
+ sltLogPartPtr.p->logTailFileNo =
+ sltLogFilePtr.p->logLastPrepRef[tsltMbyte] >> 16;
+/* ------------------------------------------------------------------------- */
+/*SINCE LOG_MAX_GCI_STARTED ONLY KEEP TRACK OF COMMIT LOG RECORDS WE ALSO */
+/*HAVE TO STEP BACK THE TAIL SO THAT WE INCLUDE ALL PREPARE RECORDS */
+/*NEEDED FOR THOSE COMMIT RECORDS IN THIS MBYTE. THIS IS A RATHER */
+/*CONSERVATIVE APPROACH BUT IT WORKS. */
+/* ------------------------------------------------------------------------- */
+ sltLogPartPtr.p->logTailMbyte =
+ sltLogFilePtr.p->logLastPrepRef[tsltMbyte] & 65535;
+ if ((ToldTailFileNo != sltLogPartPtr.p->logTailFileNo) ||
+ (ToldTailMByte != sltLogPartPtr.p->logTailMbyte)) {
+ jam();
+ if (sltLogPartPtr.p->logPartState == LogPartRecord::TAIL_PROBLEM) {
+ if (sltLogPartPtr.p->firstLogQueue == RNIL) {
+ jam();
+ sltLogPartPtr.p->logPartState = LogPartRecord::IDLE;
+ } else {
+ jam();
+ sltLogPartPtr.p->logPartState = LogPartRecord::ACTIVE;
+ }//if
+ }//if
+ }//if
+ }
+#if 0
+ infoEvent("setLogTail: Available log file %d size = %d[mbytes]+%d[words]", sltLogPartPtr.i,
+ remainingLogSize(sltCurrLogFilePtr, sltLogPartPtr), sltCurrLogFilePtr.p->remainingWordsInMbyte);
+#endif
+ }//for
+
+}//Dblqh::setLogTail()
+
+/* ######################################################################### */
+/* ####### GLOBAL CHECKPOINT MODULE ####### */
+/* */
+/* ######################################################################### */
+/*---------------------------------------------------------------------------*/
+/* THIS MODULE HELPS DIH IN DISCOVERING WHEN GLOBAL CHECKPOINTS ARE */
+/* RECOVERABLE. IT HANDLES THE REQUEST GCP_SAVEREQ THAT REQUESTS LQH TO */
+/* SAVE A PARTICULAR GLOBAL CHECKPOINT TO DISK AND RESPOND WHEN COMPLETED. */
+/*---------------------------------------------------------------------------*/
+/* *************** */
+/* GCP_SAVEREQ > */
+/* *************** */
+void Dblqh::execGCP_SAVEREQ(Signal* signal)
+{
+ jamEntry();
+ const GCPSaveReq * const saveReq = (GCPSaveReq *)&signal->theData[0];
+
+ if (ERROR_INSERTED(5000)) {
+ systemErrorLab(signal);
+ }
+
+ if (ERROR_INSERTED(5007)){
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_GCP_SAVEREQ, signal, 10000,
+ signal->length());
+ return;
+ }
+
+ const Uint32 dihBlockRef = saveReq->dihBlockRef;
+ const Uint32 dihPtr = saveReq->dihPtr;
+ const Uint32 gci = saveReq->gci;
+
+ ndbrequire(gci >= cnewestCompletedGci);
+
+ if (gci == cnewestCompletedGci) {
+/*---------------------------------------------------------------------------*/
+/* GLOBAL CHECKPOINT HAVE ALREADY BEEN HANDLED. REQUEST MUST HAVE BEEN SENT */
+/* FROM NEW MASTER DIH. */
+/*---------------------------------------------------------------------------*/
+ if (ccurrentGcprec == RNIL) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THIS INDICATES THAT WE HAVE ALREADY SENT GCP_SAVECONF TO PREVIOUS MASTER. */
+/* WE SIMPLY SEND IT ALSO TO THE NEW MASTER. */
+/*---------------------------------------------------------------------------*/
+ GCPSaveConf * const saveConf = (GCPSaveConf*)&signal->theData[0];
+ saveConf->dihPtr = dihPtr;
+ saveConf->nodeId = getOwnNodeId();
+ saveConf->gci = cnewestCompletedGci;
+ sendSignal(dihBlockRef, GSN_GCP_SAVECONF, signal,
+ GCPSaveConf::SignalLength, JBA);
+ return;
+ }
+ jam();
+/*---------------------------------------------------------------------------*/
+/* WE HAVE NOT YET SENT THE RESPONSE TO THE OLD MASTER. WE WILL SET THE NEW */
+/* RECEIVER OF THE RESPONSE AND THEN EXIT SINCE THE PROCESS IS ALREADY */
+/* STARTED. */
+/*---------------------------------------------------------------------------*/
+ gcpPtr.i = ccurrentGcprec;
+ ptrCheckGuard(gcpPtr, cgcprecFileSize, gcpRecord);
+ gcpPtr.p->gcpUserptr = dihPtr;
+ gcpPtr.p->gcpBlockref = dihBlockRef;
+ return;
+ }//if
+
+ ndbrequire(ccurrentGcprec == RNIL);
+
+
+ if(getNodeState().startLevel >= NodeState::SL_STOPPING_4){
+ GCPSaveRef * const saveRef = (GCPSaveRef*)&signal->theData[0];
+ saveRef->dihPtr = dihPtr;
+ saveRef->nodeId = getOwnNodeId();
+ saveRef->gci = gci;
+ saveRef->errorCode = GCPSaveRef::NodeShutdownInProgress;
+ sendSignal(dihBlockRef, GSN_GCP_SAVEREF, signal,
+ GCPSaveRef::SignalLength, JBB);
+ return;
+ }
+
+ if(getNodeState().getNodeRestartInProgress()){
+ GCPSaveRef * const saveRef = (GCPSaveRef*)&signal->theData[0];
+ saveRef->dihPtr = dihPtr;
+ saveRef->nodeId = getOwnNodeId();
+ saveRef->gci = gci;
+ saveRef->errorCode = GCPSaveRef::NodeRestartInProgress;
+ sendSignal(dihBlockRef, GSN_GCP_SAVEREF, signal,
+ GCPSaveRef::SignalLength, JBB);
+ return;
+ }
+
+ ccurrentGcprec = 0;
+ gcpPtr.i = ccurrentGcprec;
+ ptrCheckGuard(gcpPtr, cgcprecFileSize, gcpRecord);
+
+ cnewestCompletedGci = gci;
+ if (gci > cnewestGci) {
+ jam();
+ cnewestGci = gci;
+ }//if
+
+ gcpPtr.p->gcpBlockref = dihBlockRef;
+ gcpPtr.p->gcpUserptr = dihPtr;
+ gcpPtr.p->gcpId = gci;
+ bool tlogActive = false;
+ for (logPartPtr.i = 0; logPartPtr.i <= 3; logPartPtr.i++) {
+ ptrAss(logPartPtr, logPartRecord);
+ if (logPartPtr.p->logPartState == LogPartRecord::ACTIVE) {
+ jam();
+ logPartPtr.p->waitWriteGciLog = LogPartRecord::WWGL_TRUE;
+ tlogActive = true;
+ } else {
+ jam();
+ logPartPtr.p->waitWriteGciLog = LogPartRecord::WWGL_FALSE;
+ logFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPagePtr.i = logFilePtr.p->currentLogpage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ writeCompletedGciLog(signal);
+ }//if
+ }//for
+ if (tlogActive == true) {
+ jam();
+ return;
+ }//if
+ initGcpRecLab(signal);
+ startTimeSupervision(signal);
+ return;
+}//Dblqh::execGCP_SAVEREQ()
+
+/* ------------------------------------------------------------------------- */
+/* START TIME SUPERVISION OF THE LOG PARTS. */
+/* ------------------------------------------------------------------------- */
+void Dblqh::startTimeSupervision(Signal* signal)
+{
+ for (logPartPtr.i = 0; logPartPtr.i <= 3; logPartPtr.i++) {
+ jam();
+ ptrAss(logPartPtr, logPartRecord);
+/* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+/* WE HAVE TO START CHECKING IF THE LOG IS TO BE WRITTEN EVEN IF PAGES ARE */
+/* FULL. INITIALISE THE VALUES OF WHERE WE ARE IN THE LOG CURRENTLY. */
+/* +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ */
+ logPartPtr.p->logPartTimer = 0;
+ logPartPtr.p->logTimer = 1;
+ signal->theData[0] = ZTIME_SUPERVISION;
+ signal->theData[1] = logPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//for
+}//Dblqh::startTimeSupervision()
+
+/*---------------------------------------------------------------------------*/
+/* WE SET THE GLOBAL CHECKPOINT VARIABLES AFTER WRITING THE COMPLETED GCI LOG*/
+/* RECORD. THIS ENSURES THAT WE WILL ENCOUNTER THE COMPLETED GCI RECORD WHEN */
+/* WE EXECUTE THE FRAGMENT LOG. */
+/*---------------------------------------------------------------------------*/
+void Dblqh::initGcpRecLab(Signal* signal)
+{
+/* ======================================================================== */
+/* ======= INITIATE GCP RECORD ======= */
+/* */
+/* SUBROUTINE SHORT NAME = IGR */
+/* ======================================================================== */
+ for (logPartPtr.i = 0; logPartPtr.i <= 3; logPartPtr.i++) {
+ jam();
+ ptrAss(logPartPtr, logPartRecord);
+/*--------------------------------------------------*/
+/* BY SETTING THE GCPREC = 0 WE START THE */
+/* CHECKING BY CHECK_GCP_COMPLETED. THIS */
+/* CHECKING MUST NOT BE STARTED UNTIL WE HAVE */
+/* INSERTED ALL COMPLETE GCI LOG RECORDS IN */
+/* ALL LOG PARTS. */
+/*--------------------------------------------------*/
+ logPartPtr.p->gcprec = 0;
+ gcpPtr.p->gcpLogPartState[logPartPtr.i] = ZWAIT_DISK;
+ gcpPtr.p->gcpSyncReady[logPartPtr.i] = ZFALSE;
+ logFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ gcpPtr.p->gcpFilePtr[logPartPtr.i] = logFilePtr.i;
+ logPagePtr.i = logFilePtr.p->currentLogpage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ if (logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] == ZPAGE_HEADER_SIZE) {
+ jam();
+/*--------------------------------------------------*/
+/* SINCE THE CURRENT FILEPAGE POINTS AT THE */
+/* NEXT WORD TO BE WRITTEN WE HAVE TO ADJUST */
+/* FOR THIS BY DECREASING THE FILE PAGE BY ONE*/
+/* IF NO WORD HAS BEEN WRITTEN ON THE CURRENT */
+/* FILEPAGE. */
+/*--------------------------------------------------*/
+ gcpPtr.p->gcpPageNo[logPartPtr.i] = logFilePtr.p->currentFilepage - 1;
+ gcpPtr.p->gcpWordNo[logPartPtr.i] = ZPAGE_SIZE - 1;
+ } else {
+ jam();
+ gcpPtr.p->gcpPageNo[logPartPtr.i] = logFilePtr.p->currentFilepage;
+ gcpPtr.p->gcpWordNo[logPartPtr.i] =
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] - 1;
+ }//if
+ }//for
+ return;
+}//Dblqh::initGcpRecLab()
+
+/* ========================================================================= */
+/* ==== CHECK IF ANY GLOBAL CHECKPOINTS ARE COMPLETED AFTER A COMPLETED===== */
+/* DISK WRITE. */
+/* */
+/* SUBROUTINE SHORT NAME = CGC */
+/* ========================================================================= */
+void Dblqh::checkGcpCompleted(Signal* signal,
+ Uint32 tcgcPageWritten,
+ Uint32 tcgcWordWritten)
+{
+ UintR tcgcFlag;
+ UintR tcgcJ;
+
+ gcpPtr.i = logPartPtr.p->gcprec;
+ if (gcpPtr.i != RNIL) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* IF THE GLOBAL CHECKPOINT IS NOT WAITING FOR COMPLETION THEN WE CAN QUIT */
+/* THE SEARCH IMMEDIATELY. */
+/* ------------------------------------------------------------------------- */
+ ptrCheckGuard(gcpPtr, cgcprecFileSize, gcpRecord);
+ if (gcpPtr.p->gcpFilePtr[logPartPtr.i] == logFilePtr.i) {
+/* ------------------------------------------------------------------------- */
+/* IF THE COMPLETED DISK OPERATION WAS ON ANOTHER FILE THAN THE ONE WE ARE */
+/* WAITING FOR, THEN WE CAN ALSO QUIT THE SEARCH IMMEDIATELY. */
+/* ------------------------------------------------------------------------- */
+ if (tcgcPageWritten < gcpPtr.p->gcpPageNo[logPartPtr.i]) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* THIS LOG PART HAVE NOT YET WRITTEN THE GLOBAL CHECKPOINT TO DISK. */
+/* ------------------------------------------------------------------------- */
+ return;
+ } else {
+ if (tcgcPageWritten == gcpPtr.p->gcpPageNo[logPartPtr.i]) {
+ if (tcgcWordWritten < gcpPtr.p->gcpWordNo[logPartPtr.i]) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* THIS LOG PART HAVE NOT YET WRITTEN THE GLOBAL CHECKPOINT TO DISK. */
+/* ------------------------------------------------------------------------- */
+ return;
+ }//if
+ }//if
+ }//if
+/* ------------------------------------------------------------------------- */
+/* THIS LOG PART HAVE WRITTEN THE GLOBAL CHECKPOINT TO DISK. */
+/* ------------------------------------------------------------------------- */
+ logPartPtr.p->gcprec = RNIL;
+ gcpPtr.p->gcpLogPartState[logPartPtr.i] = ZON_DISK;
+ tcgcFlag = ZTRUE;
+ for (tcgcJ = 0; tcgcJ <= 3; tcgcJ++) {
+ jam();
+ if (gcpPtr.p->gcpLogPartState[tcgcJ] != ZON_DISK) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*ALL LOG PARTS HAVE NOT SAVED THIS GLOBAL CHECKPOINT TO DISK YET. WAIT FOR */
+/*THEM TO COMPLETE. */
+/* ------------------------------------------------------------------------- */
+ tcgcFlag = ZFALSE;
+ }//if
+ }//for
+ if (tcgcFlag == ZTRUE) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/*WE HAVE FOUND A COMPLETED GLOBAL CHECKPOINT OPERATION. WE NOW NEED TO SEND */
+/*GCP_SAVECONF, REMOVE THE GCP RECORD FROM THE LIST OF WAITING GCP RECORDS */
+/*ON THIS LOG PART AND RELEASE THE GCP RECORD. */
+// After changing the log implementation we need to perform a FSSYNCREQ on all
+// log files where the last log word resided first before proceeding.
+/* ------------------------------------------------------------------------- */
+ UintR Ti;
+ for (Ti = 0; Ti < 4; Ti++) {
+ LogFileRecordPtr loopLogFilePtr;
+ loopLogFilePtr.i = gcpPtr.p->gcpFilePtr[Ti];
+ ptrCheckGuard(loopLogFilePtr, clogFileFileSize, logFileRecord);
+ if (loopLogFilePtr.p->logFileStatus == LogFileRecord::OPEN) {
+ jam();
+ signal->theData[0] = loopLogFilePtr.p->fileRef;
+ signal->theData[1] = cownref;
+ signal->theData[2] = gcpPtr.p->gcpFilePtr[Ti];
+ sendSignal(NDBFS_REF, GSN_FSSYNCREQ, signal, 3, JBA);
+ } else {
+ ndbrequire((loopLogFilePtr.p->logFileStatus ==
+ LogFileRecord::CLOSED) ||
+ (loopLogFilePtr.p->logFileStatus ==
+ LogFileRecord::CLOSING_WRITE_LOG) ||
+ (loopLogFilePtr.p->logFileStatus ==
+ LogFileRecord::OPENING_WRITE_LOG));
+ signal->theData[0] = loopLogFilePtr.i;
+ execFSSYNCCONF(signal);
+ }//if
+ }//for
+ return;
+ }//if
+ }//if
+ }//if
+}//Dblqh::checkGcpCompleted()
+
+void
+Dblqh::execFSSYNCCONF(Signal* signal)
+{
+ GcpRecordPtr localGcpPtr;
+ LogFileRecordPtr localLogFilePtr;
+ LogPartRecordPtr localLogPartPtr;
+ localLogFilePtr.i = signal->theData[0];
+ ptrCheckGuard(localLogFilePtr, clogFileFileSize, logFileRecord);
+ localLogPartPtr.i = localLogFilePtr.p->logPartRec;
+ localGcpPtr.i = ccurrentGcprec;
+ ptrCheckGuard(localGcpPtr, cgcprecFileSize, gcpRecord);
+ localGcpPtr.p->gcpSyncReady[localLogPartPtr.i] = ZTRUE;
+ UintR Ti;
+ for (Ti = 0; Ti < 4; Ti++) {
+ jam();
+ if (localGcpPtr.p->gcpSyncReady[Ti] == ZFALSE) {
+ jam();
+ return;
+ }//if
+ }//for
+ GCPSaveConf * const saveConf = (GCPSaveConf *)&signal->theData[0];
+ saveConf->dihPtr = localGcpPtr.p->gcpUserptr;
+ saveConf->nodeId = getOwnNodeId();
+ saveConf->gci = localGcpPtr.p->gcpId;
+ sendSignal(localGcpPtr.p->gcpBlockref, GSN_GCP_SAVECONF, signal,
+ GCPSaveConf::SignalLength, JBA);
+ ccurrentGcprec = RNIL;
+}//Dblqh::execFSSYNCCONF()
+
+void
+Dblqh::execFSSYNCREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Dblqh::execFSSYNCREF()
+
+
+/* ######################################################################### */
+/* ####### FILE HANDLING MODULE ####### */
+/* */
+/* ######################################################################### */
+/* THIS MODULE HANDLES RESPONSE MESSAGES FROM THE FILE SYSTEM */
+/* ######################################################################### */
+/* ######################################################################### */
+/* SIGNAL RECEPTION MODULE */
+/* THIS MODULE IS A SUB-MODULE OF THE FILE SYSTEM HANDLING. */
+/* */
+/* THIS MODULE CHECKS THE STATE AND JUMPS TO THE PROPER PART OF THE FILE */
+/* HANDLING MODULE. */
+/* ######################################################################### */
+/* *************** */
+/* FSCLOSECONF > */
+/* *************** */
+void Dblqh::execFSCLOSECONF(Signal* signal)
+{
+ jamEntry();
+ logFilePtr.i = signal->theData[0];
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ switch (logFilePtr.p->logFileStatus) {
+ case LogFileRecord::CLOSE_SR_INVALIDATE_PAGES:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSED;
+ // Set the prev file to check if we shall close it.
+ logFilePtr.i = logFilePtr.p->prevLogFile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ exitFromInvalidate(signal);
+ return;
+ break;
+ case LogFileRecord::CLOSING_INIT:
+ jam();
+ closingInitLab(signal);
+ return;
+ break;
+ case LogFileRecord::CLOSING_SR:
+ jam();
+ closingSrLab(signal);
+ return;
+ break;
+ case LogFileRecord::CLOSING_EXEC_SR:
+ jam();
+ closeExecSrLab(signal);
+ return;
+ break;
+ case LogFileRecord::CLOSING_EXEC_SR_COMPLETED:
+ jam();
+ closeExecSrCompletedLab(signal);
+ return;
+ break;
+ case LogFileRecord::CLOSING_WRITE_LOG:
+ jam();
+ closeWriteLogLab(signal);
+ return;
+ break;
+ case LogFileRecord::CLOSING_EXEC_LOG:
+ jam();
+ closeExecLogLab(signal);
+ return;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+}//Dblqh::execFSCLOSECONF()
+
+/* ************>> */
+/* FSCLOSEREF > */
+/* ************>> */
+void Dblqh::execFSCLOSEREF(Signal* signal)
+{
+ jamEntry();
+ terrorCode = signal->theData[1];
+ systemErrorLab(signal);
+ return;
+}//Dblqh::execFSCLOSEREF()
+
+/* ************>> */
+/* FSOPENCONF > */
+/* ************>> */
+void Dblqh::execFSOPENCONF(Signal* signal)
+{
+ jamEntry();
+ initFsopenconf(signal);
+ switch (logFilePtr.p->logFileStatus) {
+ case LogFileRecord::OPEN_SR_INVALIDATE_PAGES:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ readFileInInvalidate(signal);
+ return;
+ break;
+ case LogFileRecord::OPENING_INIT:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openFileInitLab(signal);
+ return;
+ break;
+ case LogFileRecord::OPEN_SR_FRONTPAGE:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openSrFrontpageLab(signal);
+ return;
+ break;
+ case LogFileRecord::OPEN_SR_LAST_FILE:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openSrLastFileLab(signal);
+ return;
+ break;
+ case LogFileRecord::OPEN_SR_NEXT_FILE:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openSrNextFileLab(signal);
+ return;
+ break;
+ case LogFileRecord::OPEN_EXEC_SR_START:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openExecSrStartLab(signal);
+ return;
+ break;
+ case LogFileRecord::OPEN_EXEC_SR_NEW_MBYTE:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openExecSrNewMbyteLab(signal);
+ return;
+ break;
+ case LogFileRecord::OPEN_SR_FOURTH_PHASE:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openSrFourthPhaseLab(signal);
+ return;
+ break;
+ case LogFileRecord::OPEN_SR_FOURTH_NEXT:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openSrFourthNextLab(signal);
+ return;
+ break;
+ case LogFileRecord::OPEN_SR_FOURTH_ZERO:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openSrFourthZeroLab(signal);
+ return;
+ break;
+ case LogFileRecord::OPENING_WRITE_LOG:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ return;
+ break;
+ case LogFileRecord::OPEN_EXEC_LOG:
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN;
+ openExecLogLab(signal);
+ return;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+}//Dblqh::execFSOPENCONF()
+
+/* ************> */
+/* FSOPENREF > */
+/* ************> */
+void Dblqh::execFSOPENREF(Signal* signal)
+{
+ jamEntry();
+ terrorCode = signal->theData[1];
+ systemErrorLab(signal);
+ return;
+}//Dblqh::execFSOPENREF()
+
+/* ************>> */
+/* FSREADCONF > */
+/* ************>> */
+void Dblqh::execFSREADCONF(Signal* signal)
+{
+ jamEntry();
+ initFsrwconf(signal);
+
+ switch (lfoPtr.p->lfoState) {
+ case LogFileOperationRecord::READ_SR_LAST_MBYTE:
+ jam();
+ releaseLfo(signal);
+ readSrLastMbyteLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_FRONTPAGE:
+ jam();
+ releaseLfo(signal);
+ readSrFrontpageLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_LAST_FILE:
+ jam();
+ releaseLfo(signal);
+ readSrLastFileLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_NEXT_FILE:
+ jam();
+ releaseLfo(signal);
+ readSrNextFileLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_EXEC_SR:
+ jam();
+ readExecSrLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_EXEC_LOG:
+ jam();
+ readExecLogLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_INVALIDATE_PAGES:
+ jam();
+ invalidateLogAfterLastGCI(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_FOURTH_PHASE:
+ jam();
+ releaseLfo(signal);
+ readSrFourthPhaseLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_FOURTH_ZERO:
+ jam();
+ releaseLfo(signal);
+ readSrFourthZeroLab(signal);
+ return;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+}//Dblqh::execFSREADCONF()
+
+/* ************>> */
+/* FSREADCONF > */
+/* ************>> */
+void Dblqh::execFSREADREF(Signal* signal)
+{
+ jamEntry();
+ lfoPtr.i = signal->theData[0];
+ ptrCheckGuard(lfoPtr, clfoFileSize, logFileOperationRecord);
+ terrorCode = signal->theData[1];
+ switch (lfoPtr.p->lfoState) {
+ case LogFileOperationRecord::READ_SR_LAST_MBYTE:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_FRONTPAGE:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_LAST_FILE:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_NEXT_FILE:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_EXEC_SR:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_EXEC_LOG:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_FOURTH_PHASE:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_FOURTH_ZERO:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::READ_SR_INVALIDATE_PAGES:
+ jam()
+ systemErrorLab(signal);
+ return;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ return;
+}//Dblqh::execFSREADREF()
+
+/* *************** */
+/* FSWRITECONF > */
+/* *************** */
+void Dblqh::execFSWRITECONF(Signal* signal)
+{
+ jamEntry();
+ initFsrwconf(signal);
+ switch (lfoPtr.p->lfoState) {
+ case LogFileOperationRecord::WRITE_SR_INVALIDATE_PAGES:
+ jam();
+ invalidateLogAfterLastGCI(signal);
+ return;
+ break;
+ case LogFileOperationRecord::WRITE_PAGE_ZERO:
+ jam();
+ writePageZeroLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::LAST_WRITE_IN_FILE:
+ jam();
+ lastWriteInFileLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::INIT_WRITE_AT_END:
+ jam();
+ initWriteEndLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::INIT_FIRST_PAGE:
+ jam();
+ initFirstPageLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::WRITE_GCI_ZERO:
+ jam();
+ writeGciZeroLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::WRITE_DIRTY:
+ jam();
+ writeDirtyLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::WRITE_INIT_MBYTE:
+ jam();
+ writeInitMbyteLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::ACTIVE_WRITE_LOG:
+ jam();
+ writeLogfileLab(signal);
+ return;
+ break;
+ case LogFileOperationRecord::FIRST_PAGE_WRITE_IN_LOGFILE:
+ jam();
+ firstPageWriteLab(signal);
+ return;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+}//Dblqh::execFSWRITECONF()
+
+/* ************>> */
+/* FSWRITEREF > */
+/* ************>> */
+void Dblqh::execFSWRITEREF(Signal* signal)
+{
+ jamEntry();
+ lfoPtr.i = signal->theData[0];
+ ptrCheckGuard(lfoPtr, clfoFileSize, logFileOperationRecord);
+ terrorCode = signal->theData[1];
+ switch (lfoPtr.p->lfoState) {
+ case LogFileOperationRecord::WRITE_PAGE_ZERO:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case LogFileOperationRecord::LAST_WRITE_IN_FILE:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case LogFileOperationRecord::INIT_WRITE_AT_END:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case LogFileOperationRecord::INIT_FIRST_PAGE:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case LogFileOperationRecord::WRITE_GCI_ZERO:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case LogFileOperationRecord::WRITE_DIRTY:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case LogFileOperationRecord::WRITE_INIT_MBYTE:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case LogFileOperationRecord::ACTIVE_WRITE_LOG:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case LogFileOperationRecord::FIRST_PAGE_WRITE_IN_LOGFILE:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case LogFileOperationRecord::WRITE_SR_INVALIDATE_PAGES:
+ jam();
+ systemErrorLab(signal);
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+}//Dblqh::execFSWRITEREF()
+
+
+/* ========================================================================= */
+/* ======= INITIATE WHEN RECEIVING FSOPENCONF ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initFsopenconf(Signal* signal)
+{
+ logFilePtr.i = signal->theData[0];
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logFilePtr.p->fileRef = signal->theData[1];
+ logPartPtr.i = logFilePtr.p->logPartRec;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ logFilePtr.p->currentMbyte = 0;
+ logFilePtr.p->filePosition = 0;
+ logFilePtr.p->logFilePagesToDiskWithoutSynch = 0;
+}//Dblqh::initFsopenconf()
+
+/* ========================================================================= */
+/* ======= INITIATE WHEN RECEIVING FSREADCONF AND FSWRITECONF ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initFsrwconf(Signal* signal)
+{
+ lfoPtr.i = signal->theData[0];
+ ptrCheckGuard(lfoPtr, clfoFileSize, logFileOperationRecord);
+ logFilePtr.i = lfoPtr.p->logFileRec;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPartPtr.i = logFilePtr.p->logPartRec;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ logPagePtr.i = lfoPtr.p->firstLfoPage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+}//Dblqh::initFsrwconf()
+
+/* ######################################################################### */
+/* NORMAL OPERATION MODULE */
+/* THIS MODULE IS A SUB-MODULE OF THE FILE SYSTEM HANDLING. */
+/* */
+/* THIS PART HANDLES THE NORMAL OPENING, CLOSING AND WRITING OF LOG FILES */
+/* DURING NORMAL OPERATION. */
+/* ######################################################################### */
+/*---------------------------------------------------------------------------*/
+/* THIS SIGNAL IS USED TO SUPERVISE THAT THE LOG RECORDS ARE NOT KEPT IN MAIN*/
+/* MEMORY FOR MORE THAN 1 SECOND TO ACHIEVE THE PROPER RELIABILITY. */
+/*---------------------------------------------------------------------------*/
+void Dblqh::timeSup(Signal* signal)
+{
+ LogPageRecordPtr origLogPagePtr;
+ Uint32 wordWritten;
+
+ jamEntry();
+ logPartPtr.i = signal->theData[0];
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ logFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPagePtr.i = logFilePtr.p->currentLogpage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ if (logPartPtr.p->logPartTimer != logPartPtr.p->logTimer) {
+ jam();
+/*--------------------------------------------------------------------------*/
+/* THIS LOG PART HAS NOT WRITTEN TO DISK DURING THE LAST SECOND. */
+/*--------------------------------------------------------------------------*/
+ switch (logPartPtr.p->logPartState) {
+ case LogPartRecord::FILE_CHANGE_PROBLEM:
+ jam();
+/*--------------------------------------------------------------------------*/
+/* THIS LOG PART HAS PROBLEMS IN CHANGING FILES MAKING IT IMPOSSIBLE */
+// TO WRITE TO THE FILE CURRENTLY. WE WILL COMEBACK LATER AND SEE IF
+// THE PROBLEM HAS BEEN FIXED.
+/*--------------------------------------------------------------------------*/
+ case LogPartRecord::ACTIVE:
+ jam();
+/*---------------------------------------------------------------------------*/
+/* AN OPERATION IS CURRENTLY ACTIVE IN WRITING THIS LOG PART. WE THUS CANNOT */
+/* WRITE ANYTHING TO DISK AT THIS MOMENT. WE WILL SEND A SIGNAL DELAYED FOR */
+/* 10 MS AND THEN TRY AGAIN. POSSIBLY THE LOG PART WILL HAVE BEEN WRITTEN */
+/* UNTIL THEN OR ELSE IT SHOULD BE FREE TO WRITE AGAIN. */
+/*---------------------------------------------------------------------------*/
+ signal->theData[0] = ZTIME_SUPERVISION;
+ signal->theData[1] = logPartPtr.i;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 10, 2);
+ return;
+ break;
+ case LogPartRecord::IDLE:
+ case LogPartRecord::TAIL_PROBLEM:
+ jam();
+/*---------------------------------------------------------------------------*/
+/* IDLE AND NOT WRITTEN TO DISK IN A SECOND. ALSO WHEN WE HAVE A TAIL PROBLEM*/
+/* WE HAVE TO WRITE TO DISK AT TIMES. WE WILL FIRST CHECK WHETHER ANYTHING */
+/* AT ALL HAVE BEEN WRITTEN TO THE PAGES BEFORE WRITING TO DISK. */
+/*---------------------------------------------------------------------------*/
+/* WE HAVE TO WRITE TO DISK IN ALL CASES SINCE THERE COULD BE INFORMATION */
+/* STILL IN THE LOG THAT WAS GENERATED BEFORE THE PREVIOUS TIME SUPERVISION */
+/* BUT AFTER THE LAST DISK WRITE. THIS PREVIOUSLY STOPPED ALL DISK WRITES */
+/* WHEN NO MORE LOG WRITES WERE PERFORMED (THIS HAPPENED WHEN LOG GOT FULL */
+/* AND AFTER LOADING THE INITIAL RECORDS IN INITIAL START). */
+/*---------------------------------------------------------------------------*/
+ if (((logFilePtr.p->currentFilepage + 1) & (ZPAGES_IN_MBYTE -1)) == 0) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THIS IS THE LAST PAGE IN THIS MBYTE. WRITE NEXT LOG AND SWITCH TO NEXT */
+/* MBYTE. */
+/*---------------------------------------------------------------------------*/
+ changeMbyte(signal);
+ } else {
+/*---------------------------------------------------------------------------*/
+/* WRITE THE LOG PAGE TO DISK EVEN IF IT IS NOT FULL. KEEP PAGE AND WRITE A */
+/* COPY. THE ORIGINAL PAGE WILL BE WRITTEN AGAIN LATER ON. */
+/*---------------------------------------------------------------------------*/
+ wordWritten = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] - 1;
+ origLogPagePtr.i = logPagePtr.i;
+ origLogPagePtr.p = logPagePtr.p;
+ seizeLogpage(signal);
+ MEMCOPY_NO_WORDS(&logPagePtr.p->logPageWord[0],
+ &origLogPagePtr.p->logPageWord[0],
+ wordWritten + 1);
+ ndbrequire(wordWritten < ZPAGE_SIZE);
+ if (logFilePtr.p->noLogpagesInBuffer > 0) {
+ jam();
+ completedLogPage(signal, ZENFORCE_WRITE);
+/*---------------------------------------------------------------------------*/
+/*SINCE WE ARE ONLY WRITING PART OF THE LAST PAGE WE HAVE TO UPDATE THE WORD */
+/*WRITTEN TO REFLECT THE REAL LAST WORD WRITTEN. WE ALSO HAVE TO MOVE THE */
+/*FILE POSITION ONE STEP BACKWARDS SINCE WE ARE NOT WRITING THE LAST PAGE */
+/*COMPLETELY. IT WILL BE WRITTEN AGAIN. */
+/*---------------------------------------------------------------------------*/
+ lfoPtr.p->lfoWordWritten = wordWritten;
+ logFilePtr.p->filePosition = logFilePtr.p->filePosition - 1;
+ } else {
+ if (wordWritten == (ZPAGE_HEADER_SIZE - 1)) {
+/*---------------------------------------------------------------------------*/
+/*THIS IS POSSIBLE BUT VERY UNLIKELY. IF THE PAGE WAS COMPLETED AFTER THE LAST*/
+/*WRITE TO DISK THEN NO_LOG_PAGES_IN_BUFFER > 0 AND IF NOT WRITTEN SINCE LAST*/
+/*WRITE TO DISK THEN THE PREVIOUS PAGE MUST HAVE BEEN WRITTEN BY SOME */
+/*OPERATION AND THAT BECAME COMPLETELY FULL. IN ANY CASE WE NEED NOT WRITE AN*/
+/*EMPTY PAGE TO DISK. */
+/*---------------------------------------------------------------------------*/
+ jam();
+ releaseLogpage(signal);
+ } else {
+ jam();
+ writeSinglePage(signal, logFilePtr.p->currentFilepage, wordWritten);
+ lfoPtr.p->lfoState = LogFileOperationRecord::ACTIVE_WRITE_LOG;
+ }//if
+ }//if
+ }//if
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ }//if
+ logPartPtr.p->logTimer++;
+ return;
+}//Dblqh::timeSup()
+
+void Dblqh::writeLogfileLab(Signal* signal)
+{
+/*---------------------------------------------------------------------------*/
+/* CHECK IF ANY GLOBAL CHECKPOINTS ARE COMPLETED DUE TO THIS COMPLETED DISK */
+/* WRITE. */
+/*---------------------------------------------------------------------------*/
+ switch (logFilePtr.p->fileChangeState) {
+ case LogFileRecord::NOT_ONGOING:
+ jam();
+ checkGcpCompleted(signal,
+ ((lfoPtr.p->lfoPageNo + lfoPtr.p->noPagesRw) - 1),
+ lfoPtr.p->lfoWordWritten);
+ break;
+ case LogFileRecord::WRITE_PAGE_ZERO_ONGOING:
+ case LogFileRecord::LAST_WRITE_ONGOING:
+ jam();
+ logFilePtr.p->lastPageWritten = (lfoPtr.p->lfoPageNo + lfoPtr.p->noPagesRw) - 1;
+ logFilePtr.p->lastWordWritten = lfoPtr.p->lfoWordWritten;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ releaseLfoPages(signal);
+ releaseLfo(signal);
+ return;
+}//Dblqh::writeLogfileLab()
+
+void Dblqh::closeWriteLogLab(Signal* signal)
+{
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSED;
+ return;
+}//Dblqh::closeWriteLogLab()
+
+/* ######################################################################### */
+/* FILE CHANGE MODULE */
+/* THIS MODULE IS A SUB-MODULE OF THE FILE SYSTEM HANDLING. */
+/* */
+/*THIS PART OF THE FILE MODULE HANDLES WHEN WE ARE CHANGING LOG FILE DURING */
+/*NORMAL OPERATION. WE HAVE TO BE CAREFUL WHEN WE ARE CHANGING LOG FILE SO */
+/*THAT WE DO NOT COMPLICATE THE SYSTEM RESTART PROCESS TOO MUCH. */
+/*THE IDEA IS THAT WE START BY WRITING THE LAST WRITE IN THE OLD FILE AND WE */
+/*ALSO WRITE THE FIRST PAGE OF THE NEW FILE CONCURRENT WITH THAT. THIS FIRST */
+/*PAGE IN THE NEW FILE DO NOT CONTAIN ANY LOG RECORDS OTHER THAN A DESCRIPTOR*/
+/*CONTAINING INFORMATION ABOUT GCI'S NEEDED AT SYSTEM RESTART AND A NEXT LOG */
+/*RECORD. */
+/* */
+/*WHEN BOTH OF THOSE WRITES HAVE COMPLETED WE ALSO WRITE PAGE ZERO IN FILE */
+/*ZERO. THE ONLY INFORMATION WHICH IS INTERESTING HERE IS THE NEW FILE NUMBER*/
+/* */
+/*IF OPTIMISATIONS ARE NEEDED OF THE LOG HANDLING THEN IT IS POSSIBLE TO */
+/*AVOID WRITING THE FIRST PAGE OF THE NEW PAGE IMMEDIATELY. THIS COMPLICATES */
+/*THE SYSTEM RESTART AND ONE HAS TO TAKE SPECIAL CARE WITH FILE ZERO. IT IS */
+/*HOWEVER NO LARGE PROBLEM TO CHANGE INTO THIS SCENARIO. TO AVOID ALSO THE */
+/*WRITING OF PAGE ZERO IS ALSO POSSIBLE BUT COMPLICATES THE DESIGN EVEN */
+/*FURTHER. IT GETS FAIRLY COMPLEX TO FIND THE END OF THE LOG. SOME SORT OF */
+/*BINARY SEARCH IS HOWEVER MOST LIKELY A GOOD METHODOLOGY FOR THIS. */
+/* ######################################################################### */
+void Dblqh::firstPageWriteLab(Signal* signal)
+{
+ releaseLfo(signal);
+/*---------------------------------------------------------------------------*/
+/* RELEASE PAGE ZERO IF THE FILE IS NOT FILE 0. */
+/*---------------------------------------------------------------------------*/
+ Uint32 fileNo = logFilePtr.p->fileNo;
+ if (fileNo != 0) {
+ jam();
+ releaseLogpage(signal);
+ }//if
+/*---------------------------------------------------------------------------*/
+/* IF A NEW FILE HAS BEEN OPENED WE SHALL ALWAYS ALSO WRITE TO PAGE O IN */
+/* FILE 0. THE AIM IS TO MAKE RESTARTS EASIER BY SPECIFYING WHICH IS THE */
+/* LAST FILE WHERE LOGGING HAS STARTED. */
+/*---------------------------------------------------------------------------*/
+/* FIRST CHECK WHETHER THE LAST WRITE IN THE PREVIOUS FILE HAVE COMPLETED */
+/*---------------------------------------------------------------------------*/
+ if (logFilePtr.p->fileChangeState == LogFileRecord::BOTH_WRITES_ONGOING) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THE LAST WRITE WAS STILL ONGOING. */
+/*---------------------------------------------------------------------------*/
+ logFilePtr.p->fileChangeState = LogFileRecord::LAST_WRITE_ONGOING;
+ return;
+ } else {
+ jam();
+ ndbrequire(logFilePtr.p->fileChangeState == LogFileRecord::FIRST_WRITE_ONGOING);
+/*---------------------------------------------------------------------------*/
+/* WRITE TO PAGE 0 IN IN FILE 0 NOW. */
+/*---------------------------------------------------------------------------*/
+ logFilePtr.p->fileChangeState = LogFileRecord::WRITE_PAGE_ZERO_ONGOING;
+ if (fileNo == 0) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* IF THE NEW FILE WAS 0 THEN WE HAVE ALREADY WRITTEN PAGE ZERO IN FILE 0. */
+/*---------------------------------------------------------------------------*/
+ logFilePtr.p->fileChangeState = LogFileRecord::NOT_ONGOING;
+ return;
+ } else {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* WRITE PAGE ZERO IN FILE ZERO. LOG_FILE_REC WILL REFER TO THE LOG FILE WE */
+/* HAVE JUST WRITTEN PAGE ZERO IN TO GET HOLD OF LOG_FILE_PTR FOR THIS */
+/* RECORD QUICKLY. THIS IS NEEDED TO GET HOLD OF THE FILE_CHANGE_STATE. */
+/* THE ONLY INFORMATION WE WANT TO CHANGE IS THE LAST FILE NUMBER IN THE */
+/* FILE DESCRIPTOR. THIS IS USED AT SYSTEM RESTART TO FIND THE END OF THE */
+/* LOG PART. */
+/*---------------------------------------------------------------------------*/
+ Uint32 currLogFile = logFilePtr.i;
+ logFilePtr.i = logPartPtr.p->firstLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPagePtr.i = logFilePtr.p->logPageZero;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ logPagePtr.p->logPageWord[ZPAGE_HEADER_SIZE + ZPOS_FILE_NO] = fileNo;
+ writeSinglePage(signal, 0, ZPAGE_SIZE - 1);
+ lfoPtr.p->logFileRec = currLogFile;
+ lfoPtr.p->lfoState = LogFileOperationRecord::WRITE_PAGE_ZERO;
+ return;
+ }//if
+ }//if
+}//Dblqh::firstPageWriteLab()
+
+void Dblqh::lastWriteInFileLab(Signal* signal)
+{
+ LogFileRecordPtr locLogFilePtr;
+/*---------------------------------------------------------------------------*/
+/* CHECK IF ANY GLOBAL CHECKPOINTS ARE COMPLETED DUE TO THIS COMPLETED DISK */
+/* WRITE. */
+/*---------------------------------------------------------------------------*/
+ checkGcpCompleted(signal,
+ ((lfoPtr.p->lfoPageNo + lfoPtr.p->noPagesRw) - 1),
+ (ZPAGE_SIZE - 1));
+ releaseLfoPages(signal);
+ releaseLfo(signal);
+/*---------------------------------------------------------------------------*/
+/* IF THE FILE IS NOT IN USE OR THE NEXT FILE TO BE USED WE WILL CLOSE IT. */
+/*---------------------------------------------------------------------------*/
+ locLogFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(locLogFilePtr, clogFileFileSize, logFileRecord);
+ if (logFilePtr.i != locLogFilePtr.i) {
+ if (logFilePtr.i != locLogFilePtr.p->nextLogFile) {
+ if (logFilePtr.p->fileNo != 0) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THE FILE IS NOT FILE ZERO EITHER. WE WILL NOT CLOSE FILE ZERO SINCE WE */
+/* USE IT TO KEEP TRACK OF THE CURRENT LOG FILE BY WRITING PAGE ZERO IN */
+/* FILE ZERO. */
+/*---------------------------------------------------------------------------*/
+/* WE WILL CLOSE THE FILE. */
+/*---------------------------------------------------------------------------*/
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSING_WRITE_LOG;
+ closeFile(signal, logFilePtr);
+ }//if
+ }//if
+ }//if
+/*---------------------------------------------------------------------------*/
+/* IF A NEW FILE HAS BEEN OPENED WE SHALL ALWAYS ALSO WRITE TO PAGE O IN */
+/* FILE 0. THE AIM IS TO MAKE RESTARTS EASIER BY SPECIFYING WHICH IS THE */
+/* LAST FILE WHERE LOGGING HAS STARTED. */
+/*---------------------------------------------------------------------------*/
+/* FIRST CHECK WHETHER THE FIRST WRITE IN THE NEW FILE HAVE COMPLETED */
+/* THIS STATE INFORMATION IS IN THE NEW LOG FILE AND THUS WE HAVE TO MOVE */
+/* THE LOG FILE POINTER TO THIS LOG FILE. */
+/*---------------------------------------------------------------------------*/
+ logFilePtr.i = logFilePtr.p->nextLogFile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ if (logFilePtr.p->fileChangeState == LogFileRecord::BOTH_WRITES_ONGOING) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THE FIRST WRITE WAS STILL ONGOING. */
+/*---------------------------------------------------------------------------*/
+ logFilePtr.p->fileChangeState = LogFileRecord::FIRST_WRITE_ONGOING;
+ return;
+ } else {
+ ndbrequire(logFilePtr.p->fileChangeState == LogFileRecord::LAST_WRITE_ONGOING);
+/*---------------------------------------------------------------------------*/
+/* WRITE TO PAGE 0 IN IN FILE 0 NOW. */
+/*---------------------------------------------------------------------------*/
+ logFilePtr.p->fileChangeState = LogFileRecord::WRITE_PAGE_ZERO_ONGOING;
+ Uint32 fileNo = logFilePtr.p->fileNo;
+ if (fileNo == 0) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* IF THE NEW FILE WAS 0 THEN WE HAVE ALREADY WRITTEN PAGE ZERO IN FILE 0. */
+/*---------------------------------------------------------------------------*/
+ logFilePtr.p->fileChangeState = LogFileRecord::NOT_ONGOING;
+ return;
+ } else {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* WRITE PAGE ZERO IN FILE ZERO. LOG_FILE_REC WILL REFER TO THE LOG FILE WE */
+/* HAVE JUST WRITTEN PAGE ZERO IN TO GET HOLD OF LOG_FILE_PTR FOR THIS */
+/* RECORD QUICKLY. THIS IS NEEDED TO GET HOLD OF THE FILE_CHANGE_STATE. */
+/* THE ONLY INFORMATION WE WANT TO CHANGE IS THE LAST FILE NUMBER IN THE */
+/* FILE DESCRIPTOR. THIS IS USED AT SYSTEM RESTART TO FIND THE END OF THE */
+/* LOG PART. */
+/*---------------------------------------------------------------------------*/
+ Uint32 currLogFile = logFilePtr.i;
+ logFilePtr.i = logPartPtr.p->firstLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPagePtr.i = logFilePtr.p->logPageZero;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ logPagePtr.p->logPageWord[ZPAGE_HEADER_SIZE + ZPOS_FILE_NO] = fileNo;
+ writeSinglePage(signal, 0, ZPAGE_SIZE - 1);
+ lfoPtr.p->logFileRec = currLogFile;
+ lfoPtr.p->lfoState = LogFileOperationRecord::WRITE_PAGE_ZERO;
+ return;
+ }//if
+ }//if
+}//Dblqh::lastWriteInFileLab()
+
+void Dblqh::writePageZeroLab(Signal* signal)
+{
+ logFilePtr.p->fileChangeState = LogFileRecord::NOT_ONGOING;
+/*---------------------------------------------------------------------------*/
+/* IT COULD HAVE ARRIVED PAGE WRITES TO THE CURRENT FILE WHILE WE WERE */
+/* WAITING FOR THIS DISK WRITE TO COMPLETE. THEY COULD NOT CHECK FOR */
+/* COMPLETED GLOBAL CHECKPOINTS. THUS WE SHOULD DO THAT NOW INSTEAD. */
+/*---------------------------------------------------------------------------*/
+ checkGcpCompleted(signal,
+ logFilePtr.p->lastPageWritten,
+ logFilePtr.p->lastWordWritten);
+ releaseLfo(signal);
+ return;
+}//Dblqh::writePageZeroLab()
+
+/* ######################################################################### */
+/* INITIAL START MODULE */
+/* THIS MODULE IS A SUB-MODULE OF THE FILE SYSTEM HANDLING. */
+/* */
+/*THIS MODULE INITIALISES ALL THE LOG FILES THAT ARE NEEDED AT A SYSTEM */
+/*RESTART AND WHICH ARE USED DURING NORMAL OPERATIONS. IT CREATES THE FILES */
+/*AND SETS A PROPER SIZE OF THEM AND INITIALISES THE FIRST PAGE IN EACH FILE */
+/* ######################################################################### */
+void Dblqh::openFileInitLab(Signal* signal)
+{
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN_INIT;
+ seizeLogpage(signal);
+ writeSinglePage(signal, (ZNO_MBYTES_IN_FILE * ZPAGES_IN_MBYTE) - 1, ZPAGE_SIZE - 1);
+ lfoPtr.p->lfoState = LogFileOperationRecord::INIT_WRITE_AT_END;
+ return;
+}//Dblqh::openFileInitLab()
+
+void Dblqh::initWriteEndLab(Signal* signal)
+{
+ releaseLfo(signal);
+ initLogpage(signal);
+ if (logFilePtr.p->fileNo == 0) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* PAGE ZERO IN FILE ZERO MUST SET LOG LAP TO ONE SINCE IT HAS STARTED */
+/* WRITING TO THE LOG, ALSO GLOBAL CHECKPOINTS ARE SET TO ZERO. */
+/*---------------------------------------------------------------------------*/
+ logPagePtr.p->logPageWord[ZPOS_LOG_LAP] = 1;
+ logPagePtr.p->logPageWord[ZPOS_MAX_GCI_STARTED] = 0;
+ logPagePtr.p->logPageWord[ZPOS_MAX_GCI_COMPLETED] = 0;
+ logFilePtr.p->logMaxGciStarted[0] = 0;
+ logFilePtr.p->logMaxGciCompleted[0] = 0;
+ }//if
+/*---------------------------------------------------------------------------*/
+/* REUSE CODE FOR INITIALISATION OF FIRST PAGE IN ALL LOG FILES. */
+/*---------------------------------------------------------------------------*/
+ writeFileHeaderOpen(signal, ZINIT);
+ return;
+}//Dblqh::initWriteEndLab()
+
+void Dblqh::initFirstPageLab(Signal* signal)
+{
+ releaseLfo(signal);
+ if (logFilePtr.p->fileNo == 0) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* IN FILE ZERO WE WILL INSERT A PAGE ONE WHERE WE WILL INSERT A COMPLETED */
+/* GCI RECORD FOR GCI = 0. */
+/*---------------------------------------------------------------------------*/
+ initLogpage(signal);
+ logPagePtr.p->logPageWord[ZPOS_LOG_LAP] = 1;
+ logPagePtr.p->logPageWord[ZPAGE_HEADER_SIZE] = ZCOMPLETED_GCI_TYPE;
+ logPagePtr.p->logPageWord[ZPAGE_HEADER_SIZE + 1] = 1;
+ writeSinglePage(signal, 1, ZPAGE_SIZE - 1);
+ lfoPtr.p->lfoState = LogFileOperationRecord::WRITE_GCI_ZERO;
+ return;
+ }//if
+ logFilePtr.p->currentMbyte = 1;
+ writeInitMbyte(signal);
+ return;
+}//Dblqh::initFirstPageLab()
+
+void Dblqh::writeGciZeroLab(Signal* signal)
+{
+ releaseLfo(signal);
+ logFilePtr.p->currentMbyte = 1;
+ writeInitMbyte(signal);
+ return;
+}//Dblqh::writeGciZeroLab()
+
+void Dblqh::writeInitMbyteLab(Signal* signal)
+{
+ releaseLfo(signal);
+ logFilePtr.p->currentMbyte = logFilePtr.p->currentMbyte + 1;
+ if (logFilePtr.p->currentMbyte == ZNO_MBYTES_IN_FILE) {
+ jam();
+ releaseLogpage(signal);
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSING_INIT;
+ closeFile(signal, logFilePtr);
+ return;
+ }//if
+ writeInitMbyte(signal);
+ return;
+}//Dblqh::writeInitMbyteLab()
+
+void Dblqh::closingInitLab(Signal* signal)
+{
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSED;
+ logPartPtr.i = logFilePtr.p->logPartRec;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ if (logFilePtr.p->nextLogFile == logPartPtr.p->firstLogfile) {
+ jam();
+ checkInitCompletedLab(signal);
+ return;
+ } else {
+ jam();
+ logFilePtr.i = logFilePtr.p->nextLogFile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ openLogfileInit(signal);
+ }//if
+ return;
+}//Dblqh::closingInitLab()
+
+void Dblqh::checkInitCompletedLab(Signal* signal)
+{
+ logPartPtr.p->logPartState = LogPartRecord::SR_FIRST_PHASE_COMPLETED;
+/*---------------------------------------------------------------------------*/
+/* WE HAVE NOW INITIALISED ALL FILES IN THIS LOG PART. WE CAN NOW SET THE */
+/* THE LOG LAP TO ONE SINCE WE WILL START WITH LOG LAP ONE. LOG LAP = ZERO */
+/* MEANS THIS PART OF THE LOG IS NOT WRITTEN YET. */
+/*---------------------------------------------------------------------------*/
+ logPartPtr.p->logLap = 1;
+ logPartPtr.i = 0;
+CHECK_LOG_PARTS_LOOP:
+ ptrAss(logPartPtr, logPartRecord);
+ if (logPartPtr.p->logPartState != LogPartRecord::SR_FIRST_PHASE_COMPLETED) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THIS PART HAS STILL NOT COMPLETED. WAIT FOR THIS TO OCCUR. */
+/*---------------------------------------------------------------------------*/
+ return;
+ }//if
+ if (logPartPtr.i == 3) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* ALL LOG PARTS ARE COMPLETED. NOW WE CAN CONTINUE WITH THE RESTART */
+/* PROCESSING. THE NEXT STEP IS TO PREPARE FOR EXECUTING OPERATIONS. THUS WE */
+/* NEED TO INITIALISE ALL NEEDED DATA AND TO OPEN FILE ZERO AND THE NEXT AND */
+/* TO SET THE CURRENT LOG PAGE TO BE PAGE 1 IN FILE ZERO. */
+/*---------------------------------------------------------------------------*/
+ for (logPartPtr.i = 0; logPartPtr.i <= 3; logPartPtr.i++) {
+ ptrAss(logPartPtr, logPartRecord);
+ signal->theData[0] = ZINIT_FOURTH;
+ signal->theData[1] = logPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//for
+ return;
+ } else {
+ jam();
+ logPartPtr.i = logPartPtr.i + 1;
+ goto CHECK_LOG_PARTS_LOOP;
+ }//if
+}//Dblqh::checkInitCompletedLab()
+
+/* ========================================================================= */
+/* ======= INITIATE LOG FILE OPERATION RECORD WHEN ALLOCATED ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initLfo(Signal* signal)
+{
+ lfoPtr.p->firstLfoPage = RNIL;
+ lfoPtr.p->lfoState = LogFileOperationRecord::IDLE;
+ lfoPtr.p->logFileRec = logFilePtr.i;
+ lfoPtr.p->noPagesRw = 0;
+ lfoPtr.p->lfoPageNo = ZNIL;
+}//Dblqh::initLfo()
+
+/* ========================================================================= */
+/* ======= INITIATE LOG FILE WHEN ALLOCATED ======= */
+/* */
+/* INPUT: TFILE_NO NUMBER OF THE FILE INITIATED */
+/* LOG_PART_PTR NUMBER OF LOG PART */
+/* SUBROUTINE SHORT NAME = IL */
+/* ========================================================================= */
+void Dblqh::initLogfile(Signal* signal, Uint32 fileNo)
+{
+ UintR tilTmp;
+ UintR tilIndex;
+
+ logFilePtr.p->currentFilepage = 0;
+ logFilePtr.p->currentLogpage = RNIL;
+ logFilePtr.p->fileName[0] = (UintR)-1;
+ logFilePtr.p->fileName[1] = (UintR)-1; /* = H'FFFFFFFF = -1 */
+ logFilePtr.p->fileName[2] = fileNo; /* Sfile_no */
+ tilTmp = 1; /* VERSION 1 OF FILE NAME */
+ tilTmp = (tilTmp << 8) + 1; /* FRAGMENT LOG => .FRAGLOG AS EXTENSION */
+ tilTmp = (tilTmp << 8) + (8 + logPartPtr.i); /* DIRECTORY = D(8+Part)/DBLQH */
+ tilTmp = (tilTmp << 8) + 255; /* IGNORE Pxx PART OF FILE NAME */
+ logFilePtr.p->fileName[3] = tilTmp;
+/* ========================================================================= */
+/* FILE NAME BECOMES /D2/DBLQH/Tpart_no/Sfile_no.FRAGLOG */
+/* ========================================================================= */
+ logFilePtr.p->fileNo = fileNo;
+ logFilePtr.p->filePosition = 0;
+ logFilePtr.p->firstLfo = RNIL;
+ logFilePtr.p->lastLfo = RNIL;
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSED;
+ logFilePtr.p->logPartRec = logPartPtr.i;
+ logFilePtr.p->noLogpagesInBuffer = 0;
+ logFilePtr.p->firstFilledPage = RNIL;
+ logFilePtr.p->lastFilledPage = RNIL;
+ logFilePtr.p->lastPageWritten = 0;
+ logFilePtr.p->logPageZero = RNIL;
+ logFilePtr.p->currentMbyte = 0;
+ for (tilIndex = 0; tilIndex <= 15; tilIndex++) {
+ logFilePtr.p->logMaxGciCompleted[tilIndex] = (UintR)-1;
+ logFilePtr.p->logMaxGciStarted[tilIndex] = (UintR)-1;
+ logFilePtr.p->logLastPrepRef[tilIndex] = 0;
+ }//for
+}//Dblqh::initLogfile()
+
+/* ========================================================================= */
+/* ======= INITIATE LOG PAGE WHEN ALLOCATED ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initLogpage(Signal* signal)
+{
+ TcConnectionrecPtr ilpTcConnectptr;
+
+ logPagePtr.p->logPageWord[ZPOS_LOG_LAP] = logPartPtr.p->logLap;
+ logPagePtr.p->logPageWord[ZPOS_MAX_GCI_COMPLETED] =
+ logPartPtr.p->logPartNewestCompletedGCI;
+ logPagePtr.p->logPageWord[ZPOS_MAX_GCI_STARTED] = cnewestGci;
+ logPagePtr.p->logPageWord[ZPOS_VERSION] = NDB_VERSION;
+ logPagePtr.p->logPageWord[ZPOS_NO_LOG_FILES] = logPartPtr.p->noLogFiles;
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = ZPAGE_HEADER_SIZE;
+ ilpTcConnectptr.i = logPartPtr.p->firstLogTcrec;
+ if (ilpTcConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(ilpTcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ logPagePtr.p->logPageWord[ZLAST_LOG_PREP_REF] =
+ (ilpTcConnectptr.p->logStartFileNo << 16) +
+ (ilpTcConnectptr.p->logStartPageNo >> ZTWOLOG_NO_PAGES_IN_MBYTE);
+ } else {
+ jam();
+ logPagePtr.p->logPageWord[ZLAST_LOG_PREP_REF] =
+ (logFilePtr.p->fileNo << 16) +
+ (logFilePtr.p->currentFilepage >> ZTWOLOG_NO_PAGES_IN_MBYTE);
+ }//if
+}//Dblqh::initLogpage()
+
+/* ------------------------------------------------------------------------- */
+/* ------- OPEN LOG FILE FOR READ AND WRITE ------- */
+/* */
+/* SUBROUTINE SHORT NAME = OFR */
+/* ------------------------------------------------------------------------- */
+void Dblqh::openFileRw(Signal* signal, LogFileRecordPtr olfLogFilePtr)
+{
+ signal->theData[0] = cownref;
+ signal->theData[1] = olfLogFilePtr.i;
+ signal->theData[2] = olfLogFilePtr.p->fileName[0];
+ signal->theData[3] = olfLogFilePtr.p->fileName[1];
+ signal->theData[4] = olfLogFilePtr.p->fileName[2];
+ signal->theData[5] = olfLogFilePtr.p->fileName[3];
+ signal->theData[6] = ZOPEN_READ_WRITE;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+}//Dblqh::openFileRw()
+
+/* ------------------------------------------------------------------------- */
+/* ------- OPEN LOG FILE DURING INITIAL START ------- */
+/* */
+/* SUBROUTINE SHORT NAME = OLI */
+/* ------------------------------------------------------------------------- */
+void Dblqh::openLogfileInit(Signal* signal)
+{
+ logFilePtr.p->logFileStatus = LogFileRecord::OPENING_INIT;
+ signal->theData[0] = cownref;
+ signal->theData[1] = logFilePtr.i;
+ signal->theData[2] = logFilePtr.p->fileName[0];
+ signal->theData[3] = logFilePtr.p->fileName[1];
+ signal->theData[4] = logFilePtr.p->fileName[2];
+ signal->theData[5] = logFilePtr.p->fileName[3];
+ signal->theData[6] = 0x302;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+}//Dblqh::openLogfileInit()
+
+/* OPEN FOR READ/WRITE, DO CREATE AND DO TRUNCATE FILE */
+/* ------------------------------------------------------------------------- */
+/* ------- OPEN NEXT LOG FILE ------- */
+/* */
+/* SUBROUTINE SHORT NAME = ONL */
+/* ------------------------------------------------------------------------- */
+void Dblqh::openNextLogfile(Signal* signal)
+{
+ LogFileRecordPtr onlLogFilePtr;
+
+ if (logPartPtr.p->noLogFiles > 2) {
+ jam();
+/* -------------------------------------------------- */
+/* IF ONLY 1 OR 2 LOG FILES EXIST THEN THEY ARE */
+/* ALWAYS OPEN AND THUS IT IS NOT NECESSARY TO */
+/* OPEN THEM NOW. */
+/* -------------------------------------------------- */
+ onlLogFilePtr.i = logFilePtr.p->nextLogFile;
+ ptrCheckGuard(onlLogFilePtr, clogFileFileSize, logFileRecord);
+ if (onlLogFilePtr.p->logFileStatus != LogFileRecord::CLOSED) {
+ ndbrequire(onlLogFilePtr.p->fileNo == 0);
+ return;
+ }//if
+ onlLogFilePtr.p->logFileStatus = LogFileRecord::OPENING_WRITE_LOG;
+ signal->theData[0] = cownref;
+ signal->theData[1] = onlLogFilePtr.i;
+ signal->theData[2] = onlLogFilePtr.p->fileName[0];
+ signal->theData[3] = onlLogFilePtr.p->fileName[1];
+ signal->theData[4] = onlLogFilePtr.p->fileName[2];
+ signal->theData[5] = onlLogFilePtr.p->fileName[3];
+ signal->theData[6] = 2;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+ }//if
+}//Dblqh::openNextLogfile()
+
+ /* OPEN FOR READ/WRITE, DON'T CREATE AND DON'T TRUNCATE FILE */
+/* ------------------------------------------------------------------------- */
+/* ------- RELEASE LFO RECORD ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::releaseLfo(Signal* signal)
+{
+#ifdef VM_TRACE
+ // Check that lfo record isn't already in free list
+ LogFileOperationRecordPtr TlfoPtr;
+ TlfoPtr.i = cfirstfreeLfo;
+ while (TlfoPtr.i != RNIL){
+ ptrCheckGuard(TlfoPtr, clfoFileSize, logFileOperationRecord);
+ ndbrequire(TlfoPtr.i != lfoPtr.i);
+ TlfoPtr.i = TlfoPtr.p->nextLfo;
+ }
+#endif
+ lfoPtr.p->nextLfo = cfirstfreeLfo;
+ lfoPtr.p->lfoTimer = 0;
+ cfirstfreeLfo = lfoPtr.i;
+ lfoPtr.p->lfoState = LogFileOperationRecord::IDLE;
+}//Dblqh::releaseLfo()
+
+/* ------------------------------------------------------------------------- */
+/* ------- RELEASE ALL LOG PAGES CONNECTED TO A LFO RECORD ------- */
+/* */
+/* SUBROUTINE SHORT NAME = RLP */
+/* ------------------------------------------------------------------------- */
+void Dblqh::releaseLfoPages(Signal* signal)
+{
+ LogPageRecordPtr rlpLogPagePtr;
+
+ logPagePtr.i = lfoPtr.p->firstLfoPage;
+RLP_LOOP:
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ rlpLogPagePtr.i = logPagePtr.p->logPageWord[ZNEXT_PAGE];
+ releaseLogpage(signal);
+ if (rlpLogPagePtr.i != RNIL) {
+ jam();
+ logPagePtr.i = rlpLogPagePtr.i;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ goto RLP_LOOP;
+ }//if
+ lfoPtr.p->firstLfoPage = RNIL;
+}//Dblqh::releaseLfoPages()
+
+/* ------------------------------------------------------------------------- */
+/* ------- RELEASE LOG PAGE ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::releaseLogpage(Signal* signal)
+{
+#ifdef VM_TRACE
+ // Check that log page isn't already in free list
+ LogPageRecordPtr TlogPagePtr;
+ TlogPagePtr.i = cfirstfreeLogPage;
+ while (TlogPagePtr.i != RNIL){
+ ptrCheckGuard(TlogPagePtr, clogPageFileSize, logPageRecord);
+ ndbrequire(TlogPagePtr.i != logPagePtr.i);
+ TlogPagePtr.i = TlogPagePtr.p->logPageWord[ZNEXT_PAGE];
+ }
+#endif
+
+ cnoOfLogPages++;
+ logPagePtr.p->logPageWord[ZNEXT_PAGE] = cfirstfreeLogPage;
+ cfirstfreeLogPage = logPagePtr.i;
+}//Dblqh::releaseLogpage()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEIZE LFO RECORD ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::seizeLfo(Signal* signal)
+{
+ lfoPtr.i = cfirstfreeLfo;
+ ptrCheckGuard(lfoPtr, clfoFileSize, logFileOperationRecord);
+ cfirstfreeLfo = lfoPtr.p->nextLfo;
+ lfoPtr.p->nextLfo = RNIL;
+ lfoPtr.p->lfoTimer = cLqhTimeOutCount;
+}//Dblqh::seizeLfo()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEIZE LOG FILE RECORD ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::seizeLogfile(Signal* signal)
+{
+ logFilePtr.i = cfirstfreeLogFile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+/* ------------------------------------------------------------------------- */
+/*IF LIST IS EMPTY THEN A SYSTEM CRASH IS INVOKED SINCE LOG_FILE_PTR = RNIL */
+/* ------------------------------------------------------------------------- */
+ cfirstfreeLogFile = logFilePtr.p->nextLogFile;
+ logFilePtr.p->nextLogFile = RNIL;
+}//Dblqh::seizeLogfile()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEIZE LOG PAGE RECORD ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::seizeLogpage(Signal* signal)
+{
+ cnoOfLogPages--;
+ logPagePtr.i = cfirstfreeLogPage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+/* ------------------------------------------------------------------------- */
+/*IF LIST IS EMPTY THEN A SYSTEM CRASH IS INVOKED SINCE LOG_PAGE_PTR = RNIL */
+/* ------------------------------------------------------------------------- */
+ cfirstfreeLogPage = logPagePtr.p->logPageWord[ZNEXT_PAGE];
+ logPagePtr.p->logPageWord[ZNEXT_PAGE] = RNIL;
+}//Dblqh::seizeLogpage()
+
+/* ------------------------------------------------------------------------- */
+/* ------- WRITE FILE DESCRIPTOR INFORMATION ------- */
+/* */
+/* SUBROUTINE SHORT NAME: WFD */
+// Pointer handling:
+// logFilePtr in
+// logPartPtr in
+/* ------------------------------------------------------------------------- */
+void Dblqh::writeFileDescriptor(Signal* signal)
+{
+ TcConnectionrecPtr wfdTcConnectptr;
+ UintR twfdFileNo;
+ UintR twfdMbyte;
+
+/* -------------------------------------------------- */
+/* START BY WRITING TO LOG FILE RECORD */
+/* -------------------------------------------------- */
+ arrGuard(logFilePtr.p->currentMbyte, 16);
+ logFilePtr.p->logMaxGciCompleted[logFilePtr.p->currentMbyte] =
+ logPartPtr.p->logPartNewestCompletedGCI;
+ logFilePtr.p->logMaxGciStarted[logFilePtr.p->currentMbyte] = cnewestGci;
+ wfdTcConnectptr.i = logPartPtr.p->firstLogTcrec;
+ if (wfdTcConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(wfdTcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ twfdFileNo = wfdTcConnectptr.p->logStartFileNo;
+ twfdMbyte = wfdTcConnectptr.p->logStartPageNo >> ZTWOLOG_NO_PAGES_IN_MBYTE;
+ logFilePtr.p->logLastPrepRef[logFilePtr.p->currentMbyte] =
+ (twfdFileNo << 16) + twfdMbyte;
+ } else {
+ jam();
+ logFilePtr.p->logLastPrepRef[logFilePtr.p->currentMbyte] =
+ (logFilePtr.p->fileNo << 16) + logFilePtr.p->currentMbyte;
+ }//if
+}//Dblqh::writeFileDescriptor()
+
+/* ------------------------------------------------------------------------- */
+/* ------- WRITE THE HEADER PAGE OF A NEW FILE ------- */
+/* */
+/* SUBROUTINE SHORT NAME: WMO */
+/* ------------------------------------------------------------------------- */
+void Dblqh::writeFileHeaderOpen(Signal* signal, Uint32 wmoType)
+{
+ LogFileRecordPtr wmoLogFilePtr;
+ UintR twmoNoLogDescriptors;
+ UintR twmoLoop;
+ UintR twmoIndex;
+
+/* -------------------------------------------------- */
+/* WRITE HEADER INFORMATION IN THE NEW FILE. */
+/* -------------------------------------------------- */
+ logPagePtr.p->logPageWord[ZPAGE_HEADER_SIZE + ZPOS_LOG_TYPE] = ZFD_TYPE;
+ logPagePtr.p->logPageWord[ZPAGE_HEADER_SIZE + ZPOS_FILE_NO] =
+ logFilePtr.p->fileNo;
+ if (logPartPtr.p->noLogFiles > ZMAX_LOG_FILES_IN_PAGE_ZERO) {
+ jam();
+ twmoNoLogDescriptors = ZMAX_LOG_FILES_IN_PAGE_ZERO;
+ } else {
+ jam();
+ twmoNoLogDescriptors = logPartPtr.p->noLogFiles;
+ }//if
+ logPagePtr.p->logPageWord[ZPAGE_HEADER_SIZE + ZPOS_NO_FD] =
+ twmoNoLogDescriptors;
+ wmoLogFilePtr.i = logFilePtr.i;
+ twmoLoop = 0;
+WMO_LOOP:
+ jam();
+ if (twmoLoop < twmoNoLogDescriptors) {
+ jam();
+ ptrCheckGuard(wmoLogFilePtr, clogFileFileSize, logFileRecord);
+ for (twmoIndex = 0; twmoIndex <= ZNO_MBYTES_IN_FILE - 1; twmoIndex++) {
+ jam();
+ arrGuard(((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (twmoLoop * ZFD_PART_SIZE)) + twmoIndex, ZPAGE_SIZE);
+ logPagePtr.p->logPageWord[((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (twmoLoop * ZFD_PART_SIZE)) + twmoIndex] =
+ wmoLogFilePtr.p->logMaxGciCompleted[twmoIndex];
+ arrGuard((((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (twmoLoop * ZFD_PART_SIZE)) + ZNO_MBYTES_IN_FILE) +
+ twmoIndex, ZPAGE_SIZE);
+ logPagePtr.p->logPageWord[(((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (twmoLoop * ZFD_PART_SIZE)) + ZNO_MBYTES_IN_FILE) + twmoIndex] =
+ wmoLogFilePtr.p->logMaxGciStarted[twmoIndex];
+ arrGuard((((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (twmoLoop * ZFD_PART_SIZE)) + (2 * ZNO_MBYTES_IN_FILE)) +
+ twmoIndex, ZPAGE_SIZE);
+ logPagePtr.p->logPageWord[(((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (twmoLoop * ZFD_PART_SIZE)) + (2 * ZNO_MBYTES_IN_FILE)) + twmoIndex] =
+ wmoLogFilePtr.p->logLastPrepRef[twmoIndex];
+ }//for
+ wmoLogFilePtr.i = wmoLogFilePtr.p->prevLogFile;
+ twmoLoop = twmoLoop + 1;
+ goto WMO_LOOP;
+ }//if
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] =
+ (ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (ZFD_PART_SIZE * twmoNoLogDescriptors);
+ arrGuard(logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX], ZPAGE_SIZE);
+ logPagePtr.p->logPageWord[logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX]] =
+ ZNEXT_LOG_RECORD_TYPE;
+/* ------------------------------------------------------- */
+/* THIS IS A SPECIAL WRITE OF THE FIRST PAGE IN THE */
+/* LOG FILE. THIS HAS SPECIAL SIGNIFANCE TO FIND */
+/* THE END OF THE LOG AT SYSTEM RESTART. */
+/* ------------------------------------------------------- */
+ writeSinglePage(signal, 0, ZPAGE_SIZE - 1);
+ if (wmoType == ZINIT) {
+ jam();
+ lfoPtr.p->lfoState = LogFileOperationRecord::INIT_FIRST_PAGE;
+ } else {
+ jam();
+ lfoPtr.p->lfoState = LogFileOperationRecord::FIRST_PAGE_WRITE_IN_LOGFILE;
+ }//if
+ logFilePtr.p->filePosition = 1;
+ if (wmoType == ZNORMAL) {
+ jam();
+/* -------------------------------------------------- */
+/* ALLOCATE A NEW PAGE SINCE THE CURRENT IS */
+/* WRITTEN. */
+/* -------------------------------------------------- */
+ seizeLogpage(signal);
+ initLogpage(signal);
+ logFilePtr.p->currentLogpage = logPagePtr.i;
+ logFilePtr.p->currentFilepage = logFilePtr.p->currentFilepage + 1;
+ }//if
+}//Dblqh::writeFileHeaderOpen()
+
+/* -------------------------------------------------- */
+/* THE NEW FILE POSITION WILL ALWAYS BE 1 SINCE */
+/* WE JUST WROTE THE FIRST PAGE IN THE LOG FILE */
+/* -------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* ------- WRITE A MBYTE HEADER DURING INITIAL START ------- */
+/* */
+/* SUBROUTINE SHORT NAME: WIM */
+/* ------------------------------------------------------------------------- */
+void Dblqh::writeInitMbyte(Signal* signal)
+{
+ initLogpage(signal);
+ writeSinglePage(signal, logFilePtr.p->currentMbyte * ZPAGES_IN_MBYTE, ZPAGE_SIZE - 1);
+ lfoPtr.p->lfoState = LogFileOperationRecord::WRITE_INIT_MBYTE;
+}//Dblqh::writeInitMbyte()
+
+/* ------------------------------------------------------------------------- */
+/* ------- WRITE A SINGLE PAGE INTO A FILE ------- */
+/* */
+/* INPUT: TWSP_PAGE_NO THE PAGE NUMBER WRITTEN */
+/* SUBROUTINE SHORT NAME: WSP */
+/* ------------------------------------------------------------------------- */
+void Dblqh::writeSinglePage(Signal* signal, Uint32 pageNo, Uint32 wordWritten)
+{
+ seizeLfo(signal);
+ initLfo(signal);
+ lfoPtr.p->firstLfoPage = logPagePtr.i;
+ logPagePtr.p->logPageWord[ZNEXT_PAGE] = RNIL;
+
+ // Calculate checksum for page
+ logPagePtr.p->logPageWord[ZPOS_CHECKSUM] = calcPageCheckSum(logPagePtr);
+
+ lfoPtr.p->lfoPageNo = pageNo;
+ lfoPtr.p->lfoWordWritten = wordWritten;
+ lfoPtr.p->noPagesRw = 1;
+/* -------------------------------------------------- */
+/* SET TIMER ON THIS LOG PART TO SIGNIFY THAT A */
+/* LOG RECORD HAS BEEN SENT AT THIS TIME. */
+/* -------------------------------------------------- */
+ logPartPtr.p->logPartTimer = logPartPtr.p->logTimer;
+ signal->theData[0] = logFilePtr.p->fileRef;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lfoPtr.i;
+ signal->theData[3] = ZLIST_OF_PAIRS_SYNCH;
+ signal->theData[4] = ZVAR_NO_LOG_PAGE_WORD;
+ signal->theData[5] = 1; /* ONE PAGE WRITTEN */
+ signal->theData[6] = logPagePtr.i;
+ signal->theData[7] = pageNo;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+}//Dblqh::writeSinglePage()
+
+/* ##########################################################################
+ * SYSTEM RESTART PHASE ONE MODULE
+ * THIS MODULE IS A SUB-MODULE OF THE FILE SYSTEM HANDLING.
+ *
+ * THIS MODULE CONTAINS THE CODE FOR THE FIRST PHASE OF THE SYSTEM RESTART.
+ * THE AIM OF THIS PHASE IS TO FIND THE END OF THE LOG AND TO FIND
+ * INFORMATION ABOUT WHERE GLOBAL CHECKPOINTS ARE COMPLETED AND STARTED
+ * IN THE LOG. THIS INFORMATION IS NEEDED TO START PHASE THREE OF
+ * THE SYSTEM RESTART.
+ * ########################################################################## */
+/* --------------------------------------------------------------------------
+ * A SYSTEM RESTART OR NODE RESTART IS ONGOING. WE HAVE NOW OPENED FILE 0
+ * NOW WE NEED TO READ PAGE 0 TO FIND WHICH LOG FILE THAT WAS OPEN AT
+ * CRASH TIME.
+ * -------------------------------------------------------------------------- */
+void Dblqh::openSrFrontpageLab(Signal* signal)
+{
+ readSinglePage(signal, 0);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_SR_FRONTPAGE;
+ return;
+}//Dblqh::openSrFrontpageLab()
+
+/* -------------------------------------------------------------------------
+ * WE HAVE NOW READ PAGE 0 IN FILE 0. CHECK THE LAST OPEN FILE. ACTUALLY THE
+ * LAST OPEN FILE COULD BE THE NEXT AFTER THAT. CHECK THAT FIRST. WHEN THE
+ * LAST WAS FOUND WE CAN FIND ALL THE NEEDED INFORMATION WHERE TO START AND
+ * STOP READING THE LOG.
+ * -------------------------------------------------------------------------- */
+void Dblqh::readSrFrontpageLab(Signal* signal)
+{
+ Uint32 fileNo = logPagePtr.p->logPageWord[ZPAGE_HEADER_SIZE + ZPOS_FILE_NO];
+ if (fileNo == 0) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * FILE 0 WAS ALSO LAST FILE SO WE DO NOT NEED TO READ IT AGAIN.
+ * ---------------------------------------------------------------------- */
+ readSrLastFileLab(signal);
+ return;
+ }//if
+ /* ------------------------------------------------------------------------
+ * CLOSE FILE 0 SO THAT WE HAVE CLOSED ALL FILES WHEN STARTING TO READ
+ * THE FRAGMENT LOG. ALSO RELEASE PAGE ZERO.
+ * ------------------------------------------------------------------------ */
+ releaseLogpage(signal);
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSING_SR;
+ closeFile(signal, logFilePtr);
+ LogFileRecordPtr locLogFilePtr;
+ findLogfile(signal, fileNo, logPartPtr, &locLogFilePtr);
+ locLogFilePtr.p->logFileStatus = LogFileRecord::OPEN_SR_LAST_FILE;
+ openFileRw(signal, locLogFilePtr);
+ return;
+}//Dblqh::readSrFrontpageLab()
+
+void Dblqh::openSrLastFileLab(Signal* signal)
+{
+ readSinglePage(signal, 0);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_SR_LAST_FILE;
+ return;
+}//Dblqh::openSrLastFileLab()
+
+void Dblqh::readSrLastFileLab(Signal* signal)
+{
+ logPartPtr.p->logLap = logPagePtr.p->logPageWord[ZPOS_LOG_LAP];
+ if (logPartPtr.p->noLogFiles > ZMAX_LOG_FILES_IN_PAGE_ZERO) {
+ jam();
+ initGciInLogFileRec(signal, ZMAX_LOG_FILES_IN_PAGE_ZERO);
+ } else {
+ jam();
+ initGciInLogFileRec(signal, logPartPtr.p->noLogFiles);
+ }//if
+ releaseLogpage(signal);
+ /* ------------------------------------------------------------------------
+ * NOW WE HAVE FOUND THE LAST LOG FILE. WE ALSO NEED TO FIND THE LAST
+ * MBYTE THAT WAS LAST WRITTEN BEFORE THE SYSTEM CRASH.
+ * ------------------------------------------------------------------------ */
+ logPartPtr.p->lastLogfile = logFilePtr.i;
+ readSinglePage(signal, 0);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_SR_LAST_MBYTE;
+ logFilePtr.p->currentMbyte = 0;
+ return;
+}//Dblqh::readSrLastFileLab()
+
+void Dblqh::readSrLastMbyteLab(Signal* signal)
+{
+ if (logPartPtr.p->lastMbyte == ZNIL) {
+ if (logPagePtr.p->logPageWord[ZPOS_LOG_LAP] < logPartPtr.p->logLap) {
+ jam();
+ logPartPtr.p->lastMbyte = logFilePtr.p->currentMbyte - 1;
+ }//if
+ }//if
+ arrGuard(logFilePtr.p->currentMbyte, 16);
+ logFilePtr.p->logMaxGciCompleted[logFilePtr.p->currentMbyte] =
+ logPagePtr.p->logPageWord[ZPOS_MAX_GCI_COMPLETED];
+ logFilePtr.p->logMaxGciStarted[logFilePtr.p->currentMbyte] =
+ logPagePtr.p->logPageWord[ZPOS_MAX_GCI_STARTED];
+ logFilePtr.p->logLastPrepRef[logFilePtr.p->currentMbyte] =
+ logPagePtr.p->logPageWord[ZLAST_LOG_PREP_REF];
+ releaseLogpage(signal);
+ if (logFilePtr.p->currentMbyte < (ZNO_MBYTES_IN_FILE - 1)) {
+ jam();
+ logFilePtr.p->currentMbyte++;
+ readSinglePage(signal, ZPAGES_IN_MBYTE * logFilePtr.p->currentMbyte);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_SR_LAST_MBYTE;
+ return;
+ } else {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THE LOG WAS IN THE LAST MBYTE WHEN THE CRASH OCCURRED SINCE ALL
+ * LOG LAPS ARE EQUAL TO THE CURRENT LOG LAP.
+ * ---------------------------------------------------------------------- */
+ if (logPartPtr.p->lastMbyte == ZNIL) {
+ jam();
+ logPartPtr.p->lastMbyte = ZNO_MBYTES_IN_FILE - 1;
+ }//if
+ }//if
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSING_SR;
+ closeFile(signal, logFilePtr);
+ if (logPartPtr.p->noLogFiles > ZMAX_LOG_FILES_IN_PAGE_ZERO) {
+ Uint32 fileNo;
+ if (logFilePtr.p->fileNo >= ZMAX_LOG_FILES_IN_PAGE_ZERO) {
+ jam();
+ fileNo = logFilePtr.p->fileNo - ZMAX_LOG_FILES_IN_PAGE_ZERO;
+ } else {
+ jam();
+ fileNo =
+ (logPartPtr.p->noLogFiles + logFilePtr.p->fileNo) -
+ ZMAX_LOG_FILES_IN_PAGE_ZERO;
+ }//if
+ if (fileNo == 0) {
+ jam();
+ /* --------------------------------------------------------------------
+ * AVOID USING FILE 0 AGAIN SINCE THAT IS PROBABLY CLOSING AT THE
+ * MOMENT.
+ * -------------------------------------------------------------------- */
+ fileNo = 1;
+ logPartPtr.p->srRemainingFiles =
+ logPartPtr.p->noLogFiles - (ZMAX_LOG_FILES_IN_PAGE_ZERO - 1);
+ } else {
+ jam();
+ logPartPtr.p->srRemainingFiles =
+ logPartPtr.p->noLogFiles - ZMAX_LOG_FILES_IN_PAGE_ZERO;
+ }//if
+ LogFileRecordPtr locLogFilePtr;
+ findLogfile(signal, fileNo, logPartPtr, &locLogFilePtr);
+ locLogFilePtr.p->logFileStatus = LogFileRecord::OPEN_SR_NEXT_FILE;
+ openFileRw(signal, locLogFilePtr);
+ return;
+ }//if
+ /* ------------------------------------------------------------------------
+ * THERE WERE NO NEED TO READ ANY MORE PAGE ZERO IN OTHER FILES.
+ * WE NOW HAVE ALL THE NEEDED INFORMATION ABOUT THE GCI'S THAT WE NEED.
+ * NOW JUST WAIT FOR CLOSE OPERATIONS TO COMPLETE.
+ * ------------------------------------------------------------------------ */
+ return;
+}//Dblqh::readSrLastMbyteLab()
+
+void Dblqh::openSrNextFileLab(Signal* signal)
+{
+ readSinglePage(signal, 0);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_SR_NEXT_FILE;
+ return;
+}//Dblqh::openSrNextFileLab()
+
+void Dblqh::readSrNextFileLab(Signal* signal)
+{
+ if (logPartPtr.p->srRemainingFiles > ZMAX_LOG_FILES_IN_PAGE_ZERO) {
+ jam();
+ initGciInLogFileRec(signal, ZMAX_LOG_FILES_IN_PAGE_ZERO);
+ } else {
+ jam();
+ initGciInLogFileRec(signal, logPartPtr.p->srRemainingFiles);
+ }//if
+ releaseLogpage(signal);
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSING_SR;
+ closeFile(signal, logFilePtr);
+ if (logPartPtr.p->srRemainingFiles > ZMAX_LOG_FILES_IN_PAGE_ZERO) {
+ Uint32 fileNo;
+ if (logFilePtr.p->fileNo >= ZMAX_LOG_FILES_IN_PAGE_ZERO) {
+ jam();
+ fileNo = logFilePtr.p->fileNo - ZMAX_LOG_FILES_IN_PAGE_ZERO;
+ } else {
+ jam();
+ fileNo =
+ (logPartPtr.p->noLogFiles + logFilePtr.p->fileNo) -
+ ZMAX_LOG_FILES_IN_PAGE_ZERO;
+ }//if
+ if (fileNo == 0) {
+ jam();
+ /* --------------------------------------------------------------------
+ * AVOID USING FILE 0 AGAIN SINCE THAT IS PROBABLY CLOSING AT THE MOMENT.
+ * -------------------------------------------------------------------- */
+ fileNo = 1;
+ logPartPtr.p->srRemainingFiles =
+ logPartPtr.p->srRemainingFiles - (ZMAX_LOG_FILES_IN_PAGE_ZERO - 1);
+ } else {
+ jam();
+ logPartPtr.p->srRemainingFiles =
+ logPartPtr.p->srRemainingFiles - ZMAX_LOG_FILES_IN_PAGE_ZERO;
+ }//if
+ LogFileRecordPtr locLogFilePtr;
+ findLogfile(signal, fileNo, logPartPtr, &locLogFilePtr);
+ locLogFilePtr.p->logFileStatus = LogFileRecord::OPEN_SR_NEXT_FILE;
+ openFileRw(signal, locLogFilePtr);
+ }//if
+ /* ------------------------------------------------------------------------
+ * THERE WERE NO NEED TO READ ANY MORE PAGE ZERO IN OTHER FILES.
+ * WE NOW HAVE ALL THE NEEDED INFORMATION ABOUT THE GCI'S THAT WE NEED.
+ * NOW JUST WAIT FOR CLOSE OPERATIONS TO COMPLETE.
+ * ------------------------------------------------------------------------ */
+ return;
+}//Dblqh::readSrNextFileLab()
+
+void Dblqh::closingSrLab(Signal* signal)
+{
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSED;
+ logPartPtr.i = logFilePtr.p->logPartRec;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ logFilePtr.i = logPartPtr.p->firstLogfile;
+ do {
+ jam();
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ if (logFilePtr.p->logFileStatus != LogFileRecord::CLOSED) {
+ jam();
+ /* --------------------------------------------------------------------
+ * EXIT AND WAIT FOR REMAINING LOG FILES TO COMPLETE THEIR WORK.
+ * -------------------------------------------------------------------- */
+ return;
+ }//if
+ logFilePtr.i = logFilePtr.p->nextLogFile;
+ } while (logFilePtr.i != logPartPtr.p->firstLogfile);
+ /* ------------------------------------------------------------------------
+ * ALL FILES IN THIS PART HAVE BEEN CLOSED. THIS INDICATES THAT THE FIRST
+ * PHASE OF THE SYSTEM RESTART HAVE BEEN CONCLUDED FOR THIS LOG PART.
+ * CHECK IF ALL OTHER LOG PARTS ARE ALSO COMPLETED.
+ * ------------------------------------------------------------------------ */
+ logPartPtr.p->logPartState = LogPartRecord::SR_FIRST_PHASE_COMPLETED;
+ for (logPartPtr.i = 0; logPartPtr.i <= 3; logPartPtr.i++) {
+ jam();
+ ptrAss(logPartPtr, logPartRecord);
+ if (logPartPtr.p->logPartState != LogPartRecord::SR_FIRST_PHASE_COMPLETED) {
+ jam();
+ /* --------------------------------------------------------------------
+ * EXIT AND WAIT FOR THE REST OF THE LOG PARTS TO COMPLETE.
+ * -------------------------------------------------------------------- */
+ return;
+ }//if
+ }//for
+ /* ------------------------------------------------------------------------
+ * THE FIRST PHASE HAVE BEEN COMPLETED.
+ * ------------------------------------------------------------------------ */
+ signal->theData[0] = ZSR_PHASE3_START;
+ signal->theData[1] = ZSR_PHASE1_COMPLETED;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+}//Dblqh::closingSrLab()
+
+/* ##########################################################################
+ * ####### SYSTEM RESTART PHASE TWO MODULE #######
+ *
+ * THIS MODULE HANDLES THE SYSTEM RESTART WHERE LQH CONTROLS TUP AND ACC TO
+ * ENSURE THAT THEY HAVE KNOWLEDGE OF ALL FRAGMENTS AND HAVE DONE THE NEEDED
+ * READING OF DATA FROM FILE AND EXECUTION OF LOCAL LOGS. THIS PROCESS
+ * EXECUTES CONCURRENTLY WITH PHASE ONE OF THE SYSTEM RESTART. THIS PHASE
+ * FINDS THE INFORMATION ABOUT THE FRAGMENT LOG NEEDED TO EXECUTE THE FRAGMENT
+ * LOG.
+ * WHEN TUP AND ACC HAVE PREPARED ALL FRAGMENTS THEN LQH ORDERS THOSE LQH'S
+ * THAT ARE RESPONSIBLE TO EXECUTE THE FRAGMENT LOGS TO DO SO. IT IS POSSIBLE
+ * THAT ANOTHER NODE EXECUTES THE LOG FOR A FRAGMENT RESIDING AT THIS NODE.
+ * ########################################################################## */
+/* ***************>> */
+/* START_FRAGREQ > */
+/* ***************>> */
+void Dblqh::execSTART_FRAGREQ(Signal* signal)
+{
+ const StartFragReq * const startFragReq = (StartFragReq *)&signal->theData[0];
+ jamEntry();
+
+ tabptr.i = startFragReq->tableId;
+ Uint32 fragId = startFragReq->fragId;
+
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ if (!getFragmentrec(signal, fragId)) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * FRAGMENT WAS NOT DEFINED YET. PUT IT IN. IF NO LOCAL CHECKPOINT EXISTED
+ * THEN THE FRAGMENT HAS ALREADY BEEN ADDED.
+ * ---------------------------------------------------------------------- */
+ if (!insertFragrec(signal, fragId)) {
+ jam();
+ startFragRefLab(signal);
+ return;
+ }//if
+ }//if
+ tabptr.p->tableStatus = Tablerec::TABLE_DEFINED;
+
+ initFragrec(signal, tabptr.i, fragId, ZPRIMARY_NODE);
+ initFragrecSr(signal);
+ if (startFragReq->lcpNo == ZNIL) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THERE WAS NO LOCAL CHECKPOINT AVAILABLE FOR THIS FRAGMENT. WE DO
+ * NOT NEED TO READ IN THE LOCAL FRAGMENT. WE HAVE ALREADY ADDED THE
+ * FRAGMENT AS AN EMPTY FRAGMENT AT THIS POINT. THUS WE CAN SIMPLY
+ * EXIT AND THE FRAGMENT WILL PARTICIPATE IN THE EXECUTION OF THE LOG.
+ * PUT FRAGMENT ON LIST OF COMPLETED FRAGMENTS FOR EXECUTION OF LOG.
+ * ---------------------------------------------------------------------- */
+ fragptr.p->nextFrag = cfirstCompletedFragSr;
+ cfirstCompletedFragSr = fragptr.i;
+ return;
+ }//if
+ if (cfirstWaitFragSr == RNIL) {
+ jam();
+ lcpPtr.i = 0;
+ ptrAss(lcpPtr, lcpRecord);
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_IDLE) {
+ jam();
+ initLcpSr(signal, startFragReq->lcpNo,
+ startFragReq->lcpId, tabptr.i,
+ fragId, fragptr.i);
+ signal->theData[0] = lcpPtr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lcpPtr.p->currentFragment.lcpFragOrd.lcpNo;
+ signal->theData[3] = lcpPtr.p->currentFragment.lcpFragOrd.tableId;
+ signal->theData[4] = lcpPtr.p->currentFragment.lcpFragOrd.fragmentId;
+ sendSignal(fragptr.p->accBlockref, GSN_SR_FRAGIDREQ, signal, 5, JBB);
+ return;
+ }//if
+ }//if
+ fragptr.p->nextFrag = cfirstWaitFragSr;
+ cfirstWaitFragSr = fragptr.i;
+}//Dblqh::execSTART_FRAGREQ()
+
+void Dblqh::startFragRefLab(Signal* signal)
+{
+ const StartFragReq * const startFragReq = (StartFragReq *)&signal->theData[0];
+ BlockReference userRef = startFragReq->userRef;
+ Uint32 userPtr = startFragReq->userPtr;
+ signal->theData[0] = userPtr;
+ signal->theData[1] = terrorCode;
+ signal->theData[2] = cownNodeid;
+ sendSignal(userRef, GSN_START_FRAGREF, signal, 3, JBB);
+ return;
+}//Dblqh::startFragRefLab()
+
+/* ***************>> */
+/* SR_FRAGIDCONF > */
+/* ***************>> */
+/* --------------------------------------------------------------------------
+ * PRECONDITION: LCP_PTR:LCP_STATE = SR_WAIT_FRAGID
+ * -------------------------------------------------------------------------- */
+void Dblqh::execSR_FRAGIDCONF(Signal* signal)
+{
+ SrFragidConf * const srFragidConf = (SrFragidConf *)&signal->theData[0];
+ jamEntry();
+
+ lcpPtr.i = srFragidConf->lcpPtr;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ ndbrequire(lcpPtr.p->lcpState == LcpRecord::LCP_SR_WAIT_FRAGID);
+ /* ------------------------------------------------------------------------
+ * NO ERROR CHECKING OF TNO_LOCFRAG VALUE. OUT OF BOUND WILL IMPLY THAT AN
+ * INDEX OUT OF RANGE WILL CAUSE A SYSTEM RESTART WHICH IS DESIRED.
+ * ------------------------------------------------------------------------ */
+ lcpPtr.p->lcpAccptr = srFragidConf->accPtr;
+ fragptr.i = lcpPtr.p->currentFragment.fragPtrI;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ fragptr.p->accFragptr[0] = srFragidConf->fragPtr[0];
+ fragptr.p->accFragptr[1] = srFragidConf->fragPtr[1];
+ fragptr.p->hashCheckBit = srFragidConf->hashCheckBit;
+ Uint32 noLocFrag = srFragidConf->noLocFrag;
+ ndbrequire(noLocFrag == 2);
+ Uint32 fragid[2];
+ for (Uint32 i = 0; i < noLocFrag; i++) {
+ fragid[i] = srFragidConf->fragId[i];
+ }//for
+
+ for (Uint32 i = 0; i < noLocFrag; i++) {
+ jam();
+ Uint32 fragId = fragid[i];
+ /* ----------------------------------------------------------------------
+ * THERE IS NO ERROR CHECKING ON PURPOSE. IT IS POSSIBLE TO CALCULATE HOW
+ * MANY LOCAL LCP RECORDS THERE SHOULD BE. IT SHOULD NEVER HAPPEN THAT
+ * THERE IS NO ONE FREE. IF THERE IS NO ONE IT WILL ALSO BE A POINTER
+ * OUT OF RANGE WHICH IS AN ERROR CODE IN ITSELF. REUSES ERROR
+ * HANDLING IN AXE VM.
+ * ---------------------------------------------------------------------- */
+ seizeLcpLoc(signal);
+ initLcpLocAcc(signal, fragId);
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::SR_ACC_STARTED;
+ signal->theData[0] = lcpPtr.p->lcpAccptr;
+ signal->theData[1] = lcpLocptr.i;
+ signal->theData[2] = lcpLocptr.p->locFragid;
+ signal->theData[3] = lcpPtr.p->currentFragment.lcpFragOrd.lcpId % MAX_LCP_STORED;
+ sendSignal(fragptr.p->accBlockref, GSN_ACC_SRREQ, signal, 4, JBB);
+ seizeLcpLoc(signal);
+ initLcpLocTup(signal, fragId);
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::SR_TUP_STARTED;
+ signal->theData[0] = lcpLocptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lcpPtr.p->currentFragment.lcpFragOrd.tableId;
+ signal->theData[3] = lcpLocptr.p->locFragid;
+ signal->theData[4] = lcpPtr.p->currentFragment.lcpFragOrd.lcpNo;
+ sendSignal(fragptr.p->tupBlockref, GSN_TUP_SRREQ, signal, 5, JBB);
+ }//for
+ lcpPtr.p->lcpState = LcpRecord::LCP_SR_STARTED;
+ return;
+}//Dblqh::execSR_FRAGIDCONF()
+
+/* ***************> */
+/* SR_FRAGIDREF > */
+/* ***************> */
+void Dblqh::execSR_FRAGIDREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execSR_FRAGIDREF()
+
+/* ************>> */
+/* ACC_SRCONF > */
+/* ************>> */
+/* --------------------------------------------------------------------------
+ * PRECONDITION: LCP_LOCPTR:LCP_LOCSTATE = SR_ACC_STARTED
+ * -------------------------------------------------------------------------- */
+void Dblqh::execACC_SRCONF(Signal* signal)
+{
+ jamEntry();
+ lcpLocptr.i = signal->theData[0];
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (lcpLocptr.p->lcpLocstate != LcpLocRecord::SR_ACC_STARTED) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+
+ lcpPtr.i = lcpLocptr.p->masterLcpRec;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ /* ------------------------------------------------------------------------
+ * NO ERROR CHECK ON USING VALUE IN MASTER_LCP_REC. ERROR IN THIS REFERENCE
+ * WILL CAUSE POINTER OUT OF RANGE WHICH CAUSES A SYSTEM RESTART.
+ * ------------------------------------------------------------------------ */
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::SR_ACC_COMPLETED;
+ srCompletedLab(signal);
+ return;
+}//Dblqh::execACC_SRCONF()
+
+/* ************> */
+/* ACC_SRREF > */
+/* ************> */
+void Dblqh::execACC_SRREF(Signal* signal)
+{
+ jamEntry();
+ terrorCode = signal->theData[1];
+ systemErrorLab(signal);
+ return;
+}//Dblqh::execACC_SRREF()
+
+/* ************>> */
+/* TUP_SRCONF > */
+/* ************>> */
+/* --------------------------------------------------------------------------
+ * PRECONDITION: LCP_LOCPTR:LCP_LOCSTATE = SR_TUP_STARTED
+ * -------------------------------------------------------------------------- */
+void Dblqh::execTUP_SRCONF(Signal* signal)
+{
+ jamEntry();
+ lcpLocptr.i = signal->theData[0];
+ ptrCheckGuard(lcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ Uint32 tupFragPtr = signal->theData[1];
+ ndbrequire(lcpLocptr.p->lcpLocstate == LcpLocRecord::SR_TUP_STARTED);
+
+ lcpPtr.i = lcpLocptr.p->masterLcpRec;
+ ptrCheckGuard(lcpPtr, clcpFileSize, lcpRecord);
+ /* ------------------------------------------------------------------------
+ * NO ERROR CHECK ON USING VALUE IN MASTER_LCP_REC. ERROR IN THIS REFERENCE
+ * WILL CAUSE POINTER OUT OF RANGE WHICH CAUSES A SYSTEM RESTART.
+ * ------------------------------------------------------------------------ */
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::SR_TUP_COMPLETED;
+ fragptr.i = lcpPtr.p->currentFragment.fragPtrI;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (lcpLocptr.i == lcpPtr.p->firstLcpLocTup) {
+ jam();
+ fragptr.p->tupFragptr[1] = tupFragPtr;
+ } else {
+ jam();
+ fragptr.p->tupFragptr[0] = tupFragPtr;
+ }//if
+ srCompletedLab(signal);
+ return;
+}//Dblqh::execTUP_SRCONF()
+
+void Dblqh::srCompletedLab(Signal* signal)
+{
+ checkSrCompleted(signal);
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_SR_COMPLETED) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THE SYSTEM RESTART OF THIS FRAGMENT HAS BEEN COMPLETED. IT IS NOW
+ * TIME TO START A SYSTEM RESTART ON THE NEXT FRAGMENT OR CONTINUE
+ * WITH THE NEXT STEP OF THE SYSTEM RESTART. THIS STEP IS TO EXECUTE
+ * THE FRAGMENT LOGS.
+ * ----------------------------------------------------------------------
+ * WE RELEASE THE LOCAL LCP RECORDS.
+ * --------------------------------------------------------------------- */
+ releaseLocalLcps(signal);
+ /* ----------------------------------------------------------------------
+ * PUT FRAGMENT ON LIST OF FRAGMENTS WHICH HAVE BEEN STARTED AS PART OF
+ * THE SYSTEM RESTART. THEY ARE NOW WAITING TO EXECUTE THE FRAGMENT LOG.
+ * --------------------------------------------------------------------- */
+ fragptr.i = lcpPtr.p->currentFragment.fragPtrI;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ fragptr.p->nextFrag = cfirstCompletedFragSr;
+ cfirstCompletedFragSr = fragptr.i;
+ if (cfirstWaitFragSr != RNIL) {
+ jam();
+ /* --------------------------------------------------------------------
+ * ANOTHER FRAGMENT IS WAITING FOR SYSTEM RESTART. RESTART THIS
+ * FRAGMENT AS WELL.
+ * -------------------------------------------------------------------- */
+ fragptr.i = cfirstWaitFragSr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ cfirstWaitFragSr = fragptr.p->nextFrag;
+ /* --------------------------------------------------------------------
+ * RETRIEVE DATA FROM THE FRAGMENT RECORD.
+ * -------------------------------------------------------------------- */
+ ndbrequire(fragptr.p->srChkpnr < MAX_LCP_STORED);
+ initLcpSr(signal,
+ fragptr.p->srChkpnr,
+ fragptr.p->lcpId[fragptr.p->srChkpnr],
+ fragptr.p->tabRef,
+ fragptr.p->fragId,
+ fragptr.i);
+ signal->theData[0] = lcpPtr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lcpPtr.p->currentFragment.lcpFragOrd.lcpNo;
+ signal->theData[3] = lcpPtr.p->currentFragment.lcpFragOrd.tableId;
+ signal->theData[4] = lcpPtr.p->currentFragment.lcpFragOrd.fragmentId;
+ sendSignal(fragptr.p->accBlockref, GSN_SR_FRAGIDREQ, signal, 5, JBB);
+ return;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------
+ * NO MORE FRAGMENTS ARE WAITING FOR SYSTEM RESTART.
+ * -------------------------------------------------------------------- */
+ lcpPtr.p->lcpState = LcpRecord::LCP_IDLE;
+ if (cstartRecReq == ZTRUE) {
+ jam();
+ /* ----------------------------------------------------------------
+ * WE HAVE ALSO RECEIVED AN INDICATION THAT NO MORE FRAGMENTS
+ * NEEDS RESTART.
+ * NOW IT IS TIME TO START EXECUTING THE UNDO LOG.
+ * ----------------------------------------------------------------
+ * WE ARE NOW IN A POSITION TO ORDER TUP AND ACC TO START
+ * EXECUTING THEIR UNDO LOGS. THIS MUST BE DONE BEFORE THE
+ * FRAGMENT LOGS CAN BE EXECUTED.
+ * ---------------------------------------------------------------- */
+ csrExecUndoLogState = EULS_STARTED;
+ signal->theData[0] = caccBlockref;
+ signal->theData[1] = cownref;
+ sendSignal(caccBlockref, GSN_START_RECREQ, signal, 2, JBB);
+ signal->theData[0] = ctupBlockref;
+ signal->theData[1] = cownref;
+ sendSignal(ctupBlockref, GSN_START_RECREQ, signal, 2, JBB);
+ return;
+ } else {
+ jam();
+ /* ----------------------------------------------------------------
+ * WE HAVE NOT RECEIVED ALL FRAGMENTS YET OR AT LEAST NOT WE
+ * HAVE NOT RECEIVED THE START_RECREQ SIGNAL. EXIT AND WAIT
+ * FOR MORE.
+ * ---------------------------------------------------------------- */
+ return;
+ }//if
+ }//if
+ }//if
+ /*---------------*/
+ /* ELSE */
+ /*-------------------------------------------------------------------------
+ * THE SYSTEM RESTART ON THIS FRAGMENT HAS NOT BEEN COMPLETED,
+ * EXIT AND WAIT FOR MORE SIGNALS
+ *-------------------------------------------------------------------------
+ * DO NOTHING, EXIT IS EXECUTED BELOW
+ *------------------------------------------------------------------------- */
+ return;
+}//Dblqh::srCompletedLab()
+
+/* ************> */
+/* TUP_SRREF > */
+/* ************> */
+void Dblqh::execTUP_SRREF(Signal* signal)
+{
+ jamEntry();
+ terrorCode = signal->theData[1];
+ systemErrorLab(signal);
+ return;
+}//Dblqh::execTUP_SRREF()
+
+/* ***************> */
+/* START_RECREQ > */
+/* ***************> */
+void Dblqh::execSTART_RECREQ(Signal* signal)
+{
+ CRASH_INSERTION(5027);
+
+ jamEntry();
+ StartRecReq * const req = (StartRecReq*)&signal->theData[0];
+ cmasterDihBlockref = req->senderRef;
+
+ crestartOldestGci = req->keepGci;
+ crestartNewestGci = req->lastCompletedGci;
+ cnewestGci = req->newestGci;
+
+ ndbrequire(req->receivingNodeId == cownNodeid);
+
+ cnewestCompletedGci = cnewestGci;
+ cstartRecReq = ZTRUE;
+ for (logPartPtr.i = 0; logPartPtr.i < 4; logPartPtr.i++) {
+ ptrAss(logPartPtr, logPartRecord);
+ logPartPtr.p->logPartNewestCompletedGCI = cnewestCompletedGci;
+ }//for
+ /* ------------------------------------------------------------------------
+ * WE HAVE TO SET THE OLDEST AND THE NEWEST GLOBAL CHECKPOINT IDENTITY
+ * THAT WILL SURVIVE THIS SYSTEM RESTART. THIS IS NEEDED SO THAT WE CAN
+ * SET THE LOG HEAD AND LOG TAIL PROPERLY BEFORE STARTING THE SYSTEM AGAIN.
+ * WE ALSO NEED TO SET CNEWEST_GCI TO ENSURE THAT LOG RECORDS ARE EXECUTED
+ * WITH A PROPER GCI.
+ *------------------------------------------------------------------------ */
+ if (cstartType == NodeState::ST_NODE_RESTART) {
+ jam();
+ signal->theData[0] = ZSR_PHASE3_START;
+ signal->theData[1] = ZSR_PHASE2_COMPLETED;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ }//if
+ if(cstartType == NodeState::ST_INITIAL_NODE_RESTART){
+ jam();
+ StartRecConf * conf = (StartRecConf*)signal->getDataPtrSend();
+ conf->startingNodeId = getOwnNodeId();
+ sendSignal(cmasterDihBlockref, GSN_START_RECCONF, signal,
+ StartRecConf::SignalLength, JBB);
+ return;
+ }//if
+ if (cfirstWaitFragSr == RNIL) {
+ /* ----------------------------------------------------------------------
+ * THERE ARE NO FRAGMENTS WAITING TO BE RESTARTED.
+ * --------------------------------------------------------------------- */
+ lcpPtr.i = 0;
+ ptrAss(lcpPtr, lcpRecord);
+ if (lcpPtr.p->lcpState == LcpRecord::LCP_IDLE) {
+ jam();
+ /* --------------------------------------------------------------------
+ * THERE ARE NO FRAGMENTS THAT ARE CURRENTLY PERFORMING THEIR
+ * SYSTEM RESTART.
+ * --------------------------------------------------------------------
+ * WE ARE NOW IN A POSITION TO ORDER TUP AND ACC TO START EXECUTING
+ * THEIR UNDO LOGS. THIS MUST BE DONE BEFORE THE FRAGMENT LOGS
+ * CAN BE EXECUTED.
+ * ------------------------------------------------------------------- */
+ csrExecUndoLogState = EULS_STARTED;
+ signal->theData[0] = caccBlockref;
+ signal->theData[1] = cownref;
+ sendSignal(caccBlockref, GSN_START_RECREQ, signal, 2, JBB);
+ signal->theData[0] = ctupBlockref;
+ signal->theData[1] = cownref;
+ sendSignal(ctupBlockref, GSN_START_RECREQ, signal, 2, JBB);
+ }//if
+ }//if
+ /* -----------------------------------------------------------------------
+ * EXIT AND WAIT FOR COMPLETION OF ALL FRAGMENTS.
+ * ----------------------------------------------------------------------- */
+ return;
+}//Dblqh::execSTART_RECREQ()
+
+/* ***************>> */
+/* START_RECCONF > */
+/* ***************>> */
+void Dblqh::execSTART_RECCONF(Signal* signal)
+{
+ jamEntry();
+ BlockReference userRef = signal->theData[0];
+ if (userRef == caccBlockref) {
+ if (csrExecUndoLogState == EULS_STARTED) {
+ jam();
+ csrExecUndoLogState = EULS_ACC_COMPLETED;
+ } else {
+ ndbrequire(csrExecUndoLogState == EULS_TUP_COMPLETED);
+ jam();
+ csrExecUndoLogState = EULS_COMPLETED;
+ /* --------------------------------------------------------------------
+ * START THE FIRST PHASE OF EXECUTION OF THE LOG.
+ * ------------------------------------------------------------------- */
+ startExecSr(signal);
+ }//if
+ } else {
+ ndbrequire(userRef == ctupBlockref);
+ if (csrExecUndoLogState == EULS_STARTED) {
+ jam();
+ csrExecUndoLogState = EULS_TUP_COMPLETED;
+ } else {
+ ndbrequire(csrExecUndoLogState == EULS_ACC_COMPLETED);
+ jam();
+ csrExecUndoLogState = EULS_COMPLETED;
+ /* --------------------------------------------------------------------
+ * START THE FIRST PHASE OF EXECUTION OF THE LOG.
+ * ------------------------------------------------------------------- */
+ startExecSr(signal);
+ }//if
+ }//if
+ return;
+}//Dblqh::execSTART_RECCONF()
+
+/* ***************> */
+/* START_RECREF > */
+/* ***************> */
+void Dblqh::execSTART_RECREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}//Dblqh::execSTART_RECREF()
+
+/* ***************>> */
+/* START_EXEC_SR > */
+/* ***************>> */
+void Dblqh::execSTART_EXEC_SR(Signal* signal)
+{
+ FragrecordPtr prevFragptr;
+ jamEntry();
+ fragptr.i = signal->theData[0];
+ prevFragptr.i = signal->theData[1];
+ if (fragptr.i == RNIL) {
+ jam();
+ ndbrequire(cnoOfNodes < MAX_NDB_NODES);
+ /* ----------------------------------------------------------------------
+ * NO MORE FRAGMENTS TO START EXECUTING THE LOG ON.
+ * SEND EXEC_SRREQ TO ALL LQH TO INDICATE THAT THIS NODE WILL
+ * NOT REQUEST ANY MORE FRAGMENTS TO EXECUTE THE FRAGMENT LOG ON.
+ * ----------------------------------------------------------------------
+ * WE NEED TO SEND THOSE SIGNALS EVEN IF WE HAVE NOT REQUESTED
+ * ANY FRAGMENTS PARTICIPATE IN THIS PHASE.
+ * --------------------------------------------------------------------- */
+ for (Uint32 i = 0; i < cnoOfNodes; i++) {
+ jam();
+ if (cnodeStatus[i] == ZNODE_UP) {
+ jam();
+ ndbrequire(cnodeData[i] < MAX_NDB_NODES);
+ BlockReference ref = calcLqhBlockRef(cnodeData[i]);
+ signal->theData[0] = cownNodeid;
+ sendSignal(ref, GSN_EXEC_SRREQ, signal, 1, JBB);
+ }//if
+ }//for
+ } else {
+ jam();
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (fragptr.p->srNoLognodes > csrPhasesCompleted) {
+ jam();
+ Uint32 index = csrPhasesCompleted;
+ arrGuard(index, 4);
+ BlockReference ref = calcLqhBlockRef(fragptr.p->srLqhLognode[index]);
+ fragptr.p->srStatus = Fragrecord::SS_STARTED;
+ /* --------------------------------------------------------------------
+ * SINCE WE CAN HAVE SEVERAL LQH NODES PER FRAGMENT WE CALCULATE
+ * THE LQH POINTER IN SUCH A WAY THAT WE CAN DEDUCE WHICH OF THE
+ * LQH NODES THAT HAS RESPONDED WHEN EXEC_FRAGCONF IS RECEIVED.
+ * ------------------------------------------------------------------- */
+ ExecFragReq * const execFragReq = (ExecFragReq *)&signal->theData[0];
+ execFragReq->userPtr = fragptr.i;
+ execFragReq->userRef = cownref;
+ execFragReq->tableId = fragptr.p->tabRef;
+ execFragReq->fragId = fragptr.p->fragId;
+ execFragReq->startGci = fragptr.p->srStartGci[index];
+ execFragReq->lastGci = fragptr.p->srLastGci[index];
+ sendSignal(ref, GSN_EXEC_FRAGREQ, signal, ExecFragReq::SignalLength, JBB);
+ prevFragptr.i = fragptr.i;
+ fragptr.i = fragptr.p->nextFrag;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------
+ * THIS FRAGMENT IS NOW FINISHED WITH THE SYSTEM RESTART. IT DOES
+ * NOT NEED TO PARTICIPATE IN ANY MORE PHASES. REMOVE IT FROM THE
+ * LIST OF COMPLETED FRAGMENTS TO EXECUTE THE LOG ON.
+ * ALSO SEND START_FRAGCONF TO DIH AND SET THE STATE TO ACTIVE ON THE
+ * FRAGMENT.
+ * ------------------------------------------------------------------- */
+ if (prevFragptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(prevFragptr, cfragrecFileSize, fragrecord);
+ prevFragptr.p->nextFrag = fragptr.p->nextFrag;
+ } else {
+ jam();
+ cfirstCompletedFragSr = fragptr.p->nextFrag;
+ }//if
+ fragptr.p->fragStatus = Fragrecord::FSACTIVE;
+ fragptr.p->logFlag = Fragrecord::STATE_TRUE;
+ signal->theData[0] = fragptr.p->srUserptr;
+ signal->theData[1] = cownNodeid;
+ sendSignal(fragptr.p->srBlockref, GSN_START_FRAGCONF, signal, 2, JBB);
+ /* --------------------------------------------------------------------
+ * WE HAVE TO ENSURE THAT THIS FRAGMENT IS NOT PUT BACK ON THE LIST BY
+ * MISTAKE. WE DO THIS BY ALSO REMOVING IT AS PREVIOUS IN START_EXEC_SR
+ * THIS IS PERFORMED BY KEEPING PREV_FRAGPTR AS PREV_FRAGPTR BUT MOVING
+ * FRAGPTR TO THE NEXT FRAGMENT IN THE LIST.
+ * ------------------------------------------------------------------- */
+ fragptr.i = fragptr.p->nextFrag;
+ }//if
+ signal->theData[0] = fragptr.i;
+ signal->theData[1] = prevFragptr.i;
+ sendSignal(cownref, GSN_START_EXEC_SR, signal, 2, JBB);
+ }//if
+ return;
+}//Dblqh::execSTART_EXEC_SR()
+
+/* ***************> */
+/* EXEC_FRAGREQ > */
+/* ***************> */
+/* --------------------------------------------------------------------------
+ * THIS SIGNAL IS USED TO REQUEST THAT A FRAGMENT PARTICIPATES IN EXECUTING
+ * THE LOG IN THIS NODE.
+ * ------------------------------------------------------------------------- */
+void Dblqh::execEXEC_FRAGREQ(Signal* signal)
+{
+ ExecFragReq * const execFragReq = (ExecFragReq *)&signal->theData[0];
+ jamEntry();
+ tabptr.i = execFragReq->tableId;
+ Uint32 fragId = execFragReq->fragId;
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ if (!getFragmentrec(signal, fragId)) {
+ jam();
+ if (!insertFragrec(signal, fragId)) {
+ jam();
+ sendExecFragRefLab(signal);
+ return;
+ }//if
+ initFragrec(signal, tabptr.i, fragId, ZLOG_NODE);
+ fragptr.p->execSrStatus = Fragrecord::ACTIVE_REMOVE_AFTER;
+ } else {
+ jam();
+ if (fragptr.p->execSrStatus == Fragrecord::ACTIVE_REMOVE_AFTER) {
+ jam();
+ fragptr.p->execSrStatus = Fragrecord::ACTIVE_REMOVE_AFTER;
+ } else {
+ jam();
+ }//if
+ }//if
+ ndbrequire(fragptr.p->execSrNoReplicas < 4);
+ fragptr.p->execSrBlockref[fragptr.p->execSrNoReplicas] = execFragReq->userRef;
+ fragptr.p->execSrUserptr[fragptr.p->execSrNoReplicas] = execFragReq->userPtr;
+ fragptr.p->execSrStartGci[fragptr.p->execSrNoReplicas] = execFragReq->startGci;
+ fragptr.p->execSrLastGci[fragptr.p->execSrNoReplicas] = execFragReq->lastGci;
+ fragptr.p->execSrStatus = Fragrecord::ACTIVE;
+ fragptr.p->execSrNoReplicas++;
+ cnoFragmentsExecSr++;
+ return;
+}//Dblqh::execEXEC_FRAGREQ()
+
+void Dblqh::sendExecFragRefLab(Signal* signal)
+{
+ ExecFragReq * const execFragReq = (ExecFragReq *)&signal->theData[0];
+ BlockReference retRef = execFragReq->userRef;
+ Uint32 retPtr = execFragReq->userPtr;
+
+ signal->theData[0] = retPtr;
+ signal->theData[1] = terrorCode;
+ sendSignal(retRef, GSN_EXEC_FRAGREF, signal, 2, JBB);
+ return;
+}//Dblqh::sendExecFragRefLab()
+
+/* ***************>> */
+/* EXEC_FRAGCONF > */
+/* ***************>> */
+void Dblqh::execEXEC_FRAGCONF(Signal* signal)
+{
+ jamEntry();
+ fragptr.i = signal->theData[0];
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ fragptr.p->srStatus = Fragrecord::SS_COMPLETED;
+ return;
+}//Dblqh::execEXEC_FRAGCONF()
+
+/* ***************> */
+/* EXEC_FRAGREF > */
+/* ***************> */
+void Dblqh::execEXEC_FRAGREF(Signal* signal)
+{
+ jamEntry();
+ terrorCode = signal->theData[1];
+ systemErrorLab(signal);
+ return;
+}//Dblqh::execEXEC_FRAGREF()
+
+/* *************** */
+/* EXEC_SRCONF > */
+/* *************** */
+void Dblqh::execEXEC_SRCONF(Signal* signal)
+{
+ jamEntry();
+ Uint32 nodeId = signal->theData[0];
+ arrGuard(nodeId, MAX_NDB_NODES);
+ cnodeExecSrState[nodeId] = ZEXEC_SR_COMPLETED;
+ ndbrequire(cnoOfNodes < MAX_NDB_NODES);
+ for (Uint32 i = 0; i < cnoOfNodes; i++) {
+ jam();
+ if (cnodeStatus[i] == ZNODE_UP) {
+ jam();
+ nodeId = cnodeData[i];
+ arrGuard(nodeId, MAX_NDB_NODES);
+ if (cnodeExecSrState[nodeId] != ZEXEC_SR_COMPLETED) {
+ jam();
+ /* ------------------------------------------------------------------
+ * ALL NODES HAVE NOT REPORTED COMPLETION OF EXECUTING FRAGMENT
+ * LOGS YET.
+ * ----------------------------------------------------------------- */
+ return;
+ }//if
+ }//if
+ }//for
+ /* ------------------------------------------------------------------------
+ * CLEAR NODE SYSTEM RESTART EXECUTION STATE TO PREPARE FOR NEXT PHASE OF
+ * LOG EXECUTION.
+ * ----------------------------------------------------------------------- */
+ for (nodeId = 0; nodeId < MAX_NDB_NODES; nodeId++) {
+ cnodeExecSrState[nodeId] = ZSTART_SR;
+ }//for
+ /* ------------------------------------------------------------------------
+ * NOW CHECK IF ALL FRAGMENTS IN THIS PHASE HAVE COMPLETED. IF SO START THE
+ * NEXT PHASE.
+ * ----------------------------------------------------------------------- */
+ fragptr.i = cfirstCompletedFragSr;
+ if (fragptr.i == RNIL) {
+ jam();
+ execSrCompletedLab(signal);
+ return;
+ }//if
+ do {
+ jam();
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ ndbrequire(fragptr.p->srStatus == Fragrecord::SS_COMPLETED);
+ fragptr.i = fragptr.p->nextFrag;
+ } while (fragptr.i != RNIL);
+ execSrCompletedLab(signal);
+ return;
+}//Dblqh::execEXEC_SRCONF()
+
+void Dblqh::execSrCompletedLab(Signal* signal)
+{
+ csrPhasesCompleted++;
+ /* ------------------------------------------------------------------------
+ * ALL FRAGMENTS WERE COMPLETED. THIS PHASE IS COMPLETED. IT IS NOW TIME TO
+ * START THE NEXT PHASE.
+ * ----------------------------------------------------------------------- */
+ if (csrPhasesCompleted >= 4) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THIS WAS THE LAST PHASE. WE HAVE NOW COMPLETED THE EXECUTION THE
+ * FRAGMENT LOGS IN ALL NODES. BEFORE WE SEND START_RECCONF TO THE
+ * MASTER DIH TO INDICATE A COMPLETED SYSTEM RESTART IT IS NECESSARY
+ * TO FIND THE HEAD AND THE TAIL OF THE LOG WHEN NEW OPERATIONS START
+ * TO COME AGAIN.
+ *
+ * THE FIRST STEP IS TO FIND THE HEAD AND TAIL MBYTE OF EACH LOG PART.
+ * TO DO THIS WE REUSE THE CONTINUEB SIGNAL SR_LOG_LIMITS. THEN WE
+ * HAVE TO FIND THE ACTUAL PAGE NUMBER AND PAGE INDEX WHERE TO
+ * CONTINUE WRITING THE LOG AFTER THE SYSTEM RESTART.
+ * --------------------------------------------------------------------- */
+ for (logPartPtr.i = 0; logPartPtr.i < 4; logPartPtr.i++) {
+ jam();
+ ptrAss(logPartPtr, logPartRecord);
+ logPartPtr.p->logPartState = LogPartRecord::SR_FOURTH_PHASE_STARTED;
+ logPartPtr.p->logLastGci = crestartNewestGci;
+ logPartPtr.p->logStartGci = crestartOldestGci;
+ logPartPtr.p->logExecState = LogPartRecord::LES_SEARCH_STOP;
+ if (logPartPtr.p->headFileNo == ZNIL) {
+ jam();
+ /* -----------------------------------------------------------------
+ * IF WE HAVEN'T FOUND ANY HEAD OF THE LOG THEN WE ARE IN SERIOUS
+ * PROBLEM. THIS SHOULD NOT OCCUR. IF IT OCCURS ANYWAY THEN WE
+ * HAVE TO FIND A CURE FOR THIS PROBLEM.
+ * ----------------------------------------------------------------- */
+ systemErrorLab(signal);
+ return;
+ }//if
+ signal->theData[0] = ZSR_LOG_LIMITS;
+ signal->theData[1] = logPartPtr.i;
+ signal->theData[2] = logPartPtr.p->lastLogfile;
+ signal->theData[3] = logPartPtr.p->lastMbyte;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 4, JBB);
+ }//for
+ return;
+ } else {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THERE ARE YET MORE PHASES TO RESTART.
+ * WE MUST INITIALISE DATA FOR NEXT PHASE AND SEND START SIGNAL.
+ * --------------------------------------------------------------------- */
+ startExecSr(signal);
+ }//if
+ return;
+}//Dblqh::execSrCompletedLab()
+
+/* ************>> */
+/* EXEC_SRREQ > */
+/* ************>> */
+void Dblqh::execEXEC_SRREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 nodeId = signal->theData[0];
+ ndbrequire(nodeId < MAX_NDB_NODES);
+ cnodeSrState[nodeId] = ZEXEC_SR_COMPLETED;
+ ndbrequire(cnoOfNodes < MAX_NDB_NODES);
+ for (Uint32 i = 0; i < cnoOfNodes; i++) {
+ jam();
+ if (cnodeStatus[i] == ZNODE_UP) {
+ jam();
+ nodeId = cnodeData[i];
+ if (cnodeSrState[nodeId] != ZEXEC_SR_COMPLETED) {
+ jam();
+ /* ------------------------------------------------------------------
+ * ALL NODES HAVE NOT REPORTED COMPLETION OF SENDING EXEC_FRAGREQ YET.
+ * ----------------------------------------------------------------- */
+ return;
+ }//if
+ }//if
+ }//for
+ /* ------------------------------------------------------------------------
+ * CLEAR NODE SYSTEM RESTART STATE TO PREPARE FOR NEXT PHASE OF LOG
+ * EXECUTION
+ * ----------------------------------------------------------------------- */
+ for (nodeId = 0; nodeId < MAX_NDB_NODES; nodeId++) {
+ cnodeSrState[nodeId] = ZSTART_SR;
+ }//for
+ if (csrPhasesCompleted != 0) {
+ /* ----------------------------------------------------------------------
+ * THE FIRST PHASE MUST ALWAYS EXECUTE THE LOG.
+ * --------------------------------------------------------------------- */
+ if (cnoFragmentsExecSr == 0) {
+ jam();
+ /* --------------------------------------------------------------------
+ * THERE WERE NO FRAGMENTS THAT NEEDED TO EXECUTE THE LOG IN THIS PHASE.
+ * ------------------------------------------------------------------- */
+ srPhase3Comp(signal);
+ return;
+ }//if
+ }//if
+ /* ------------------------------------------------------------------------
+ * NOW ALL NODES HAVE SENT ALL EXEC_FRAGREQ. NOW WE CAN START EXECUTING THE
+ * LOG FROM THE MINIMUM GCI NEEDED UNTIL THE MAXIMUM GCI NEEDED.
+ *
+ * WE MUST FIRST CHECK IF THE FIRST PHASE OF THE SYSTEM RESTART HAS BEEN
+ * COMPLETED. THIS HANDLING IS PERFORMED IN THE FILE SYSTEM MODULE
+ * ----------------------------------------------------------------------- */
+ signal->theData[0] = ZSR_PHASE3_START;
+ signal->theData[1] = ZSR_PHASE2_COMPLETED;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+}//Dblqh::execEXEC_SRREQ()
+
+/* ######################################################################### */
+/* SYSTEM RESTART PHASE THREE MODULE */
+/* THIS MODULE IS A SUB-MODULE OF THE FILE SYSTEM HANDLING. */
+/* */
+/* THIS MODULE IS CONCERNED WITH EXECUTING THE FRAGMENT LOG. IT DOES ALSO */
+/* CONTAIN SIGNAL RECEPTIONS LQHKEYCONF AND LQHKEYREF SINCE LQHKEYREQ IS USED*/
+/* TO EXECUTE THE LOG RECORDS. */
+/* */
+/* BEFORE IT STARTS IT HAS BEEN DECIDED WHERE TO START AND WHERE TO STOP */
+/* READING THE FRAGMENT LOG BY USING THE INFORMATION ABOUT GCI DISCOVERED IN */
+/* PHASE ONE OF THE SYSTEM RESTART. */
+/* ######################################################################### */
+/*---------------------------------------------------------------------------*/
+/* PHASE THREE OF THE SYSTEM RESTART CAN NOW START. ONE OF THE PHASES HAVE */
+/* COMPLETED. */
+/*---------------------------------------------------------------------------*/
+void Dblqh::srPhase3Start(Signal* signal)
+{
+ UintR tsrPhaseStarted;
+
+ jamEntry();
+ tsrPhaseStarted = signal->theData[0];
+ if (csrPhaseStarted == ZSR_NO_PHASE_STARTED) {
+ jam();
+ csrPhaseStarted = tsrPhaseStarted;
+ if (cstartType == NodeState::ST_NODE_RESTART) {
+ ndbrequire(cinitialStartOngoing == ZTRUE);
+ cinitialStartOngoing = ZFALSE;
+ checkStartCompletedLab(signal);
+ }//if
+ return;
+ }//if
+ ndbrequire(csrPhaseStarted != tsrPhaseStarted);
+ ndbrequire(csrPhaseStarted != ZSR_BOTH_PHASES_STARTED);
+
+ csrPhaseStarted = ZSR_BOTH_PHASES_STARTED;
+ for (logPartPtr.i = 0; logPartPtr.i < 4; logPartPtr.i++) {
+ jam();
+ ptrAss(logPartPtr, logPartRecord);
+ logPartPtr.p->logPartState = LogPartRecord::SR_THIRD_PHASE_STARTED;
+ logPartPtr.p->logStartGci = (UintR)-1;
+ if (csrPhasesCompleted == 0) {
+ jam();
+ /* --------------------------------------------------------------------
+ * THE FIRST PHASE WE MUST ENSURE THAT IT REACHES THE END OF THE LOG.
+ * ------------------------------------------------------------------- */
+ logPartPtr.p->logLastGci = crestartNewestGci;
+ } else {
+ jam();
+ logPartPtr.p->logLastGci = 2;
+ }//if
+ }//for
+ if (cstartType == NodeState::ST_NODE_RESTART) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * FOR A NODE RESTART WE HAVE NO FRAGMENTS DEFINED YET.
+ * THUS WE CAN SKIP THAT PART
+ * --------------------------------------------------------------------- */
+ signal->theData[0] = ZSR_GCI_LIMITS;
+ signal->theData[1] = RNIL;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ } else {
+ jam();
+ signal->theData[0] = ZSR_GCI_LIMITS;
+ signal->theData[1] = 0;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+ return;
+}//Dblqh::srPhase3Start()
+
+/* --------------------------------------------------------------------------
+ * WE NOW WE NEED TO FIND THE LIMITS WITHIN WHICH TO EXECUTE
+ * THE FRAGMENT LOG
+ * ------------------------------------------------------------------------- */
+void Dblqh::srGciLimits(Signal* signal)
+{
+ LogPartRecordPtr tmpLogPartPtr;
+
+ jamEntry();
+ fragptr.i = signal->theData[0];
+ Uint32 loopCount = 0;
+ logPartPtr.i = 0;
+ ptrAss(logPartPtr, logPartRecord);
+ while (fragptr.i < cfragrecFileSize) {
+ jam();
+ ptrAss(fragptr, fragrecord);
+ if (fragptr.p->execSrStatus != Fragrecord::IDLE) {
+ jam();
+ ndbrequire(fragptr.p->execSrNoReplicas - 1 < 4);
+ for (Uint32 i = 0; i < fragptr.p->execSrNoReplicas; i++) {
+ jam();
+ if (fragptr.p->execSrStartGci[i] < logPartPtr.p->logStartGci) {
+ jam();
+ logPartPtr.p->logStartGci = fragptr.p->execSrStartGci[i];
+ }//if
+ if (fragptr.p->execSrLastGci[i] > logPartPtr.p->logLastGci) {
+ jam();
+ logPartPtr.p->logLastGci = fragptr.p->execSrLastGci[i];
+ }//if
+ }//for
+ }//if
+ loopCount++;
+ if (loopCount > 20) {
+ jam();
+ signal->theData[0] = ZSR_GCI_LIMITS;
+ signal->theData[1] = fragptr.i + 1;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ } else {
+ jam();
+ fragptr.i++;
+ }//if
+ }//while
+ if (logPartPtr.p->logStartGci == (UintR)-1) {
+ jam();
+ /* --------------------------------------------------------------------
+ * THERE WERE NO FRAGMENTS TO INSTALL WE WILL EXECUTE THE LOG AS
+ * SHORT AS POSSIBLE TO REACH THE END OF THE LOG. THIS WE DO BY
+ * STARTING AT THE STOP GCI.
+ * ------------------------------------------------------------------- */
+ logPartPtr.p->logStartGci = logPartPtr.p->logLastGci;
+ }//if
+ for (tmpLogPartPtr.i = 1; tmpLogPartPtr.i < 4; tmpLogPartPtr.i++) {
+ ptrAss(tmpLogPartPtr, logPartRecord);
+ tmpLogPartPtr.p->logStartGci = logPartPtr.p->logStartGci;
+ tmpLogPartPtr.p->logLastGci = logPartPtr.p->logLastGci;
+ }//for
+ for (logPartPtr.i = 0; logPartPtr.i < 4; logPartPtr.i++) {
+ jam();
+ ptrAss(logPartPtr, logPartRecord);
+ logPartPtr.p->logExecState = LogPartRecord::LES_SEARCH_STOP;
+ signal->theData[0] = ZSR_LOG_LIMITS;
+ signal->theData[1] = logPartPtr.i;
+ signal->theData[2] = logPartPtr.p->lastLogfile;
+ signal->theData[3] = logPartPtr.p->lastMbyte;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 4, JBB);
+ }//for
+}//Dblqh::srGciLimits()
+
+/* --------------------------------------------------------------------------
+ * IT IS NOW TIME TO FIND WHERE TO START EXECUTING THE LOG.
+ * THIS SIGNAL IS SENT FOR EACH LOG PART AND STARTS THE EXECUTION
+ * OF THE LOG FOR THIS PART.
+ *-------------------------------------------------------------------------- */
+void Dblqh::srLogLimits(Signal* signal)
+{
+ Uint32 tlastPrepRef;
+ Uint32 tmbyte;
+
+ jamEntry();
+ logPartPtr.i = signal->theData[0];
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ logFilePtr.i = signal->theData[1];
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ tmbyte = signal->theData[2];
+ Uint32 loopCount = 0;
+ /* ------------------------------------------------------------------------
+ * WE ARE SEARCHING FOR THE START AND STOP MBYTE OF THE LOG THAT IS TO BE
+ * EXECUTED.
+ * ----------------------------------------------------------------------- */
+ while(true) {
+ ndbrequire(tmbyte < 16);
+ if (logPartPtr.p->logExecState == LogPartRecord::LES_SEARCH_STOP) {
+ if (logFilePtr.p->logMaxGciCompleted[tmbyte] < logPartPtr.p->logLastGci) {
+ jam();
+ /* --------------------------------------------------------------------
+ * WE ARE STEPPING BACKWARDS FROM MBYTE TO MBYTE. THIS IS THE FIRST
+ * MBYTE WHICH IS TO BE INCLUDED IN THE LOG EXECUTION. THE STOP GCI
+ * HAS NOT BEEN COMPLETED BEFORE THIS MBYTE. THUS THIS MBYTE HAVE
+ * TO BE EXECUTED.
+ * ------------------------------------------------------------------- */
+ logPartPtr.p->stopLogfile = logFilePtr.i;
+ logPartPtr.p->stopMbyte = tmbyte;
+ logPartPtr.p->logExecState = LogPartRecord::LES_SEARCH_START;
+ }//if
+ }//if
+ /* ------------------------------------------------------------------------
+ * WHEN WE HAVEN'T FOUND THE STOP MBYTE IT IS NOT NECESSARY TO LOOK FOR THE
+ * START MBYTE. THE REASON IS THE FOLLOWING LOGIC CHAIN:
+ * MAX_GCI_STARTED >= MAX_GCI_COMPLETED >= LAST_GCI >= START_GCI
+ * THUS MAX_GCI_STARTED >= START_GCI. THUS MAX_GCI_STARTED < START_GCI CAN
+ * NOT BE TRUE AS WE WILL CHECK OTHERWISE.
+ * ----------------------------------------------------------------------- */
+ if (logPartPtr.p->logExecState == LogPartRecord::LES_SEARCH_START) {
+ if (logFilePtr.p->logMaxGciStarted[tmbyte] < logPartPtr.p->logStartGci) {
+ jam();
+ /* --------------------------------------------------------------------
+ * WE HAVE NOW FOUND THE START OF THE EXECUTION OF THE LOG.
+ * WE STILL HAVE TO MOVE IT BACKWARDS TO ALSO INCLUDE THE
+ * PREPARE RECORDS WHICH WERE STARTED IN A PREVIOUS MBYTE.
+ * ------------------------------------------------------------------- */
+ tlastPrepRef = logFilePtr.p->logLastPrepRef[tmbyte];
+ logPartPtr.p->startMbyte = tlastPrepRef & 65535;
+ LogFileRecordPtr locLogFilePtr;
+ findLogfile(signal, tlastPrepRef >> 16, logPartPtr, &locLogFilePtr);
+ logPartPtr.p->startLogfile = locLogFilePtr.i;
+ logPartPtr.p->logExecState = LogPartRecord::LES_EXEC_LOG;
+ }//if
+ }//if
+ if (logPartPtr.p->logExecState != LogPartRecord::LES_EXEC_LOG) {
+ if (tmbyte == 0) {
+ jam();
+ tmbyte = ZNO_MBYTES_IN_FILE - 1;
+ logFilePtr.i = logFilePtr.p->prevLogFile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ } else {
+ jam();
+ tmbyte--;
+ }//if
+ if (logPartPtr.p->lastLogfile == logFilePtr.i) {
+ ndbrequire(logPartPtr.p->lastMbyte != tmbyte);
+ }//if
+ if (loopCount > 20) {
+ jam();
+ signal->theData[0] = ZSR_LOG_LIMITS;
+ signal->theData[1] = logPartPtr.i;
+ signal->theData[2] = logFilePtr.i;
+ signal->theData[3] = tmbyte;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 4, JBB);
+ return;
+ }//if
+ loopCount++;
+ } else {
+ jam();
+ break;
+ }//if
+ }//while
+ /* ------------------------------------------------------------------------
+ * WE HAVE NOW FOUND BOTH THE START AND THE STOP OF THE LOG. NOW START
+ * EXECUTING THE LOG. THE FIRST ACTION IS TO OPEN THE LOG FILE WHERE TO
+ * START EXECUTING THE LOG.
+ * ----------------------------------------------------------------------- */
+ if (logPartPtr.p->logPartState == LogPartRecord::SR_THIRD_PHASE_STARTED) {
+ jam();
+ logFilePtr.i = logPartPtr.p->startLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN_EXEC_SR_START;
+ openFileRw(signal, logFilePtr);
+ } else {
+ jam();
+ ndbrequire(logPartPtr.p->logPartState == LogPartRecord::SR_FOURTH_PHASE_STARTED);
+ /* --------------------------------------------------------------------
+ * WE HAVE NOW FOUND THE TAIL MBYTE IN THE TAIL FILE.
+ * SET THOSE PARAMETERS IN THE LOG PART.
+ * WE HAVE ALSO FOUND THE HEAD MBYTE. WE STILL HAVE TO SEARCH
+ * FOR THE PAGE NUMBER AND PAGE INDEX WHERE TO SET THE HEAD.
+ * ------------------------------------------------------------------- */
+ logFilePtr.i = logPartPtr.p->startLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPartPtr.p->logTailFileNo = logFilePtr.p->fileNo;
+ logPartPtr.p->logTailMbyte = logPartPtr.p->startMbyte;
+ /* --------------------------------------------------------------------
+ * THE HEAD WE ACTUALLY FOUND DURING EXECUTION OF LOG SO WE USE
+ * THIS INFO HERE RATHER THAN THE MBYTE WE FOUND TO BE THE HEADER.
+ * ------------------------------------------------------------------- */
+ LogFileRecordPtr locLogFilePtr;
+ findLogfile(signal, logPartPtr.p->headFileNo, logPartPtr, &locLogFilePtr);
+ locLogFilePtr.p->logFileStatus = LogFileRecord::OPEN_SR_FOURTH_PHASE;
+ openFileRw(signal, locLogFilePtr);
+ }//if
+ return;
+}//Dblqh::srLogLimits()
+
+void Dblqh::openExecSrStartLab(Signal* signal)
+{
+ logPartPtr.p->currentLogfile = logFilePtr.i;
+ logFilePtr.p->currentMbyte = logPartPtr.p->startMbyte;
+ /* ------------------------------------------------------------------------
+ * WE NEED A TC CONNECT RECORD TO HANDLE EXECUTION OF LOG RECORDS.
+ * ------------------------------------------------------------------------ */
+ seizeTcrec();
+ logPartPtr.p->logTcConrec = tcConnectptr.i;
+ /* ------------------------------------------------------------------------
+ * THE FIRST LOG RECORD TO EXECUTE IS ALWAYS AT A NEW MBYTE.
+ * SET THE NUMBER OF PAGES IN THE MAIN MEMORY BUFFER TO ZERO AS AN INITIAL
+ * VALUE. THIS VALUE WILL BE UPDATED AND ENSURED THAT IT RELEASES PAGES IN
+ * THE SUBROUTINE READ_EXEC_SR.
+ * ----------------------------------------------------------------------- */
+ logPartPtr.p->mmBufferSize = 0;
+ readExecSrNewMbyte(signal);
+ return;
+}//Dblqh::openExecSrStartLab()
+
+/* ---------------------------------------------------------------------------
+ * WE WILL ALWAYS ENSURE THAT WE HAVE AT LEAST 16 KBYTE OF LOG PAGES WHEN WE
+ * START READING A LOG RECORD. THE ONLY EXCEPTION IS WHEN WE COME CLOSE TO A
+ * MBYTE BOUNDARY. SINCE WE KNOW THAT LOG RECORDS ARE NEVER WRITTEN ACROSS A
+ * MBYTE BOUNDARY THIS IS NOT A PROBLEM.
+ *
+ * WE START BY READING 64 KBYTE BEFORE STARTING TO EXECUTE THE LOG RECORDS.
+ * WHEN WE COME BELOW 64 KBYTE WE READ ANOTHER SET OF LOG PAGES. WHEN WE
+ * GO BELOW 16 KBYTE WE WAIT UNTIL THE READ PAGES HAVE ENTERED THE BLOCK.
+ * ------------------------------------------------------------------------- */
+/* --------------------------------------------------------------------------
+ * NEW PAGES FROM LOG FILE DURING EXECUTION OF LOG HAS ARRIVED.
+ * ------------------------------------------------------------------------- */
+void Dblqh::readExecSrLab(Signal* signal)
+{
+ buildLinkedLogPageList(signal);
+ /* ------------------------------------------------------------------------
+ * WE NEED TO SET THE CURRENT PAGE INDEX OF THE FIRST PAGE SINCE IT CAN BE
+ * USED IMMEDIATELY WITHOUT ANY OTHER INITIALISATION. THE REST OF THE PAGES
+ * WILL BE INITIALISED BY READ_LOGWORD.
+ * ----------------------------------------------------------------------- */
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = ZPAGE_HEADER_SIZE;
+ if (logPartPtr.p->logExecState ==
+ LogPartRecord::LES_WAIT_READ_EXEC_SR_NEW_MBYTE) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THIS IS THE FIRST READ DURING THE EXECUTION OF THIS MBYTE. SET THE
+ * NEW CURRENT LOG PAGE TO THE FIRST OF THESE PAGES. CHANGE
+ * LOG_EXEC_STATE TO ENSURE THAT WE START EXECUTION OF THE LOG.
+ * --------------------------------------------------------------------- */
+ logFilePtr.p->currentFilepage = logFilePtr.p->currentMbyte *
+ ZPAGES_IN_MBYTE;
+ logPartPtr.p->prevFilepage = logFilePtr.p->currentFilepage;
+ logFilePtr.p->currentLogpage = lfoPtr.p->firstLfoPage;
+ logPartPtr.p->prevLogpage = logFilePtr.p->currentLogpage;
+ }//if
+ moveToPageRef(signal);
+ releaseLfo(signal);
+ /* ------------------------------------------------------------------------
+ * NOW WE HAVE COMPLETED THE RECEPTION OF THESE PAGES.
+ * NOW CHECK IF WE NEED TO READ MORE PAGES.
+ * ----------------------------------------------------------------------- */
+ checkReadExecSr(signal);
+ if (logPartPtr.p->logExecState == LogPartRecord::LES_EXEC_LOG) {
+ jam();
+ signal->theData[0] = ZEXEC_SR;
+ signal->theData[1] = logPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ }//if
+ return;
+}//Dblqh::readExecSrLab()
+
+void Dblqh::openExecSrNewMbyteLab(Signal* signal)
+{
+ readExecSrNewMbyte(signal);
+ return;
+}//Dblqh::openExecSrNewMbyteLab()
+
+void Dblqh::closeExecSrLab(Signal* signal)
+{
+ LogFileRecordPtr locLogFilePtr;
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSED;
+ logPartPtr.i = logFilePtr.p->logPartRec;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ locLogFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(locLogFilePtr, clogFileFileSize, logFileRecord);
+ locLogFilePtr.p->logFileStatus = LogFileRecord::OPEN_EXEC_SR_NEW_MBYTE;
+ openFileRw(signal, locLogFilePtr);
+ return;
+}//Dblqh::closeExecSrLab()
+
+void Dblqh::writeDirtyLab(Signal* signal)
+{
+ releaseLfo(signal);
+ signal->theData[0] = logPartPtr.i;
+ execSr(signal);
+ return;
+}//Dblqh::writeDirtyLab()
+
+/* --------------------------------------------------------------------------
+ * EXECUTE A LOG RECORD WITHIN THE CURRENT MBYTE.
+ * ------------------------------------------------------------------------- */
+void Dblqh::execSr(Signal* signal)
+{
+ LogFileRecordPtr nextLogFilePtr;
+ LogPageRecordPtr tmpLogPagePtr;
+ Uint32 logWord;
+
+ jamEntry();
+ logPartPtr.i = signal->theData[0];
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+
+ do {
+ jam();
+ logFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPagePtr.i = logPartPtr.p->prevLogpage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ if (logPagePtr.p->logPageWord[ZPOS_DIRTY] == ZDIRTY) {
+ jam();
+ switch (logPartPtr.p->logExecState) {
+ case LogPartRecord::LES_EXEC_LOG_COMPLETED:
+ case LogPartRecord::LES_EXEC_LOG_NEW_FILE:
+ case LogPartRecord::LES_EXEC_LOG_NEW_MBYTE:
+ jam();
+ /* ------------------------------------------------------------------
+ * IN THIS WE HAVE COMPLETED EXECUTION OF THE CURRENT LOG PAGE
+ * AND CAN WRITE IT TO DISK SINCE IT IS DIRTY.
+ * ----------------------------------------------------------------- */
+ writeDirty(signal);
+ return;
+ break;
+ case LogPartRecord::LES_EXEC_LOG:
+ jam();
+ /* --------------------------------------------------------------------
+ * IN THIS CASE WE ONLY WRITE THE PAGE TO DISK IF WE HAVE COMPLETED
+ * EXECUTION OF LOG RECORDS BELONGING TO THIS LOG PAGE.
+ * ------------------------------------------------------------------- */
+ if (logFilePtr.p->currentLogpage != logPartPtr.p->prevLogpage) {
+ jam();
+ writeDirty(signal);
+ return;
+ }//if
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ }//if
+ if (logFilePtr.p->currentLogpage != logPartPtr.p->prevLogpage) {
+ jam();
+ logPartPtr.p->prevLogpage = logPagePtr.p->logPageWord[ZNEXT_PAGE];
+ logPartPtr.p->prevFilepage++;
+ continue;
+ }//if
+ switch (logPartPtr.p->logExecState) {
+ case LogPartRecord::LES_EXEC_LOG_COMPLETED:
+ jam();
+ releaseMmPages(signal);
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSING_EXEC_SR_COMPLETED;
+ closeFile(signal, logFilePtr);
+ return;
+ break;
+ case LogPartRecord::LES_EXEC_LOG_NEW_MBYTE:
+ jam();
+ logFilePtr.p->currentMbyte++;
+ readExecSrNewMbyte(signal);
+ return;
+ break;
+ case LogPartRecord::LES_EXEC_LOG_NEW_FILE:
+ jam();
+ nextLogFilePtr.i = logFilePtr.p->nextLogFile;
+ logPartPtr.p->currentLogfile = nextLogFilePtr.i;
+ ptrCheckGuard(nextLogFilePtr, clogFileFileSize, logFileRecord);
+ nextLogFilePtr.p->currentMbyte = 0;
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSING_EXEC_SR;
+ closeFile(signal, logFilePtr);
+ return;
+ break;
+ case LogPartRecord::LES_EXEC_LOG:
+ jam();
+ /*empty*/;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ logPagePtr.i = logFilePtr.p->currentLogpage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ logPartPtr.p->savePageIndex = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ if (logPartPtr.p->execSrPagesRead < ZMIN_READ_BUFFER_SIZE) {
+ /* --------------------------------------------------------------------
+ * THERE WERE LESS THAN 16 KBYTE OF LOG PAGES REMAINING. WE WAIT UNTIL
+ * THE NEXT 64 KBYTE ARRIVES UNTIL WE CONTINUE AGAIN.
+ * ------------------------------------------------------------------- */
+ if ((logPartPtr.p->execSrPagesRead +
+ logPartPtr.p->execSrPagesExecuted) < ZPAGES_IN_MBYTE) {
+ jam();
+ /* ------------------------------------------------------------------
+ * WE ONLY STOP AND WAIT IF THERE MORE PAGES TO READ. IF IT IS NOT
+ * THEN IT IS THE END OF THE MBYTE AND WE WILL CONTINUE. IT IS NO
+ * RISK THAT A LOG RECORD WE FIND WILL NOT BE READ AT THIS TIME
+ * SINCE THE LOG RECORDS NEVER SPAN OVER A MBYTE BOUNDARY.
+ * ----------------------------------------------------------------- */
+ readExecSr(signal);
+ logPartPtr.p->logExecState = LogPartRecord::LES_WAIT_READ_EXEC_SR;
+ return;
+ }//if
+ }//if
+ logWord = readLogword(signal);
+ switch (logWord) {
+/* ========================================================================= */
+/* ========================================================================= */
+ case ZPREP_OP_TYPE:
+ {
+ logWord = readLogword(signal);
+ stepAhead(signal, logWord - 2);
+ break;
+ }
+/* ========================================================================= */
+/* ========================================================================= */
+ case ZINVALID_COMMIT_TYPE:
+ jam();
+ stepAhead(signal, ZCOMMIT_LOG_SIZE - 1);
+ break;
+/* ========================================================================= */
+/* ========================================================================= */
+ case ZCOMMIT_TYPE:
+ {
+ CommitLogRecord commitLogRecord;
+ jam();
+ tcConnectptr.i = logPartPtr.p->logTcConrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ readCommitLog(signal, &commitLogRecord);
+ if (tcConnectptr.p->gci > crestartNewestGci) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THIS LOG RECORD MUST BE IGNORED. IT IS PART OF A GLOBAL CHECKPOINT WHICH */
+/* WILL BE INVALIDATED BY THE SYSTEM RESTART. IF NOT INVALIDATED IT MIGHT BE */
+/* EXECUTED IN A FUTURE SYSTEM RESTART. */
+/*---------------------------------------------------------------------------*/
+ tmpLogPagePtr.i = logPartPtr.p->prevLogpage;
+ ptrCheckGuard(tmpLogPagePtr, clogPageFileSize, logPageRecord);
+ arrGuard(logPartPtr.p->savePageIndex, ZPAGE_SIZE);
+ tmpLogPagePtr.p->logPageWord[logPartPtr.p->savePageIndex] =
+ ZINVALID_COMMIT_TYPE;
+ tmpLogPagePtr.p->logPageWord[ZPOS_DIRTY] = ZDIRTY;
+ } else {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* CHECK IF I AM SUPPOSED TO EXECUTE THIS LOG RECORD. IF I AM THEN SAVE PAGE */
+/* INDEX IN CURRENT LOG PAGE SINCE IT WILL BE OVERWRITTEN WHEN EXECUTING THE */
+/* LOG RECORD. */
+/*---------------------------------------------------------------------------*/
+ logPartPtr.p->execSrExecuteIndex = 0;
+ Uint32 result = checkIfExecLog(signal);
+ if (result == ZOK) {
+ jam();
+//*---------------------------------------------------------------------------*/
+/* IN A NODE RESTART WE WILL NEVER END UP HERE SINCE NO FRAGMENTS HAVE BEEN */
+/* DEFINED YET. THUS NO EXTRA CHECKING FOR NODE RESTART IS NECESSARY. */
+/*---------------------------------------------------------------------------*/
+ logPartPtr.p->savePageIndex =
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ tcConnectptr.p->fragmentptr = fragptr.i;
+ findPageRef(signal, &commitLogRecord);
+ logPartPtr.p->execSrLogPageIndex = commitLogRecord.startPageIndex;
+ if (logPagePtr.i != RNIL) {
+ jam();
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = commitLogRecord.startPageIndex;
+ logPartPtr.p->execSrLogPage = logPagePtr.i;
+ execLogRecord(signal);
+ return;
+ }//if
+ logPartPtr.p->execSrStartPageNo = commitLogRecord.startPageNo;
+ logPartPtr.p->execSrStopPageNo = commitLogRecord.stopPageNo;
+ findLogfile(signal, commitLogRecord.fileNo, logPartPtr, &logFilePtr);
+ logPartPtr.p->execSrExecLogFile = logFilePtr.i;
+ if (logFilePtr.i == logPartPtr.p->currentLogfile) {
+ jam();
+ readExecLog(signal);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_EXEC_LOG;
+ return;
+ } else {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THE FILE IS CURRENTLY NOT OPEN. WE MUST OPEN IT BEFORE WE CAN READ FROM */
+/* THE FILE. */
+/*---------------------------------------------------------------------------*/
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN_EXEC_LOG;
+ openFileRw(signal, logFilePtr);
+ return;
+ }//if
+ }//if
+ }//if
+ break;
+ }
+/* ========================================================================= */
+/* ========================================================================= */
+ case ZABORT_TYPE:
+ jam();
+ stepAhead(signal, ZABORT_LOG_SIZE - 1);
+ break;
+/* ========================================================================= */
+/* ========================================================================= */
+ case ZFD_TYPE:
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THIS IS THE FIRST ITEM WE ENCOUNTER IN A NEW FILE. AT THIS MOMENT WE SHALL*/
+/* SIMPLY BYPASS IT. IT HAS NO SIGNIFANCE WHEN EXECUTING THE LOG. IT HAS ITS */
+/* SIGNIFANCE WHEN FINDING THE START END THE END OF THE LOG. */
+/* WE HARDCODE THE PAGE INDEX SINCE THIS SHOULD NEVER BE FOUND AT ANY OTHER */
+/* PLACE THAN IN THE FIRST PAGE OF A NEW FILE IN THE FIRST POSITION AFTER THE*/
+/* HEADER. */
+/*---------------------------------------------------------------------------*/
+ ndbrequire(logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] ==
+ (ZPAGE_HEADER_SIZE + ZPOS_NO_FD));
+ {
+ Uint32 noFdDescriptors =
+ logPagePtr.p->logPageWord[ZPAGE_HEADER_SIZE + ZPOS_NO_FD];
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] =
+ (ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (noFdDescriptors * ZFD_PART_SIZE);
+ }
+ break;
+/* ========================================================================= */
+/* ========================================================================= */
+ case ZNEXT_LOG_RECORD_TYPE:
+ jam();
+ stepAhead(signal, ZPAGE_SIZE - logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX]);
+ break;
+/* ========================================================================= */
+/* ========================================================================= */
+ case ZNEXT_MBYTE_TYPE:
+/*---------------------------------------------------------------------------*/
+/* WE WILL SKIP A PART OF THE LOG FILE. ACTUALLY THE NEXT POINTER IS TO */
+/* A NEW MBYTE. THEREFORE WE WILL START UP A NEW MBYTE. THIS NEW MBYTE IS */
+/* HOWEVER ONLY STARTED IF IT IS NOT AFTER THE STOP MBYTE. */
+/* IF WE HAVE REACHED THE END OF THE STOP MBYTE THEN THE EXECUTION OF THE LOG*/
+/* IS COMPLETED. */
+/*---------------------------------------------------------------------------*/
+ if (logPartPtr.p->currentLogfile == logPartPtr.p->stopLogfile) {
+ if (logFilePtr.p->currentMbyte == logPartPtr.p->stopMbyte) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* THIS WAS THE LAST MBYTE TO EXECUTE IN THIS LOG PART. WE SHOULD HAVE FOUND */
+/* A COMPLETED GCI RECORD OF THE LAST GCI BEFORE THIS. FOR SOME REASON THIS */
+/* RECORD WAS NOT AVAILABLE ON THE LOG. CRASH THE SYSTEM, A VERY SERIOUS */
+/* ERROR WHICH WE MUST REALLY WORK HARD TO AVOID. */
+/*---------------------------------------------------------------------------*/
+/*---------------------------------------------------------------------------*/
+/* SEND A SIGNAL TO THE SIGNAL LOG AND THEN CRASH THE SYSTEM. */
+/*---------------------------------------------------------------------------*/
+ signal->theData[0] = RNIL;
+ signal->theData[1] = logPartPtr.i;
+ Uint32 tmp = logFilePtr.p->fileName[3];
+ tmp = (tmp >> 8) & 0xff;// To get the Directory, DXX.
+ signal->theData[2] = tmp;
+ signal->theData[3] = logFilePtr.p->fileNo;
+ signal->theData[4] = logFilePtr.p->currentFilepage;
+ signal->theData[5] = logFilePtr.p->currentMbyte;
+ signal->theData[6] = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ sendSignal(cownref, GSN_DEBUG_SIG, signal, 7, JBA);
+ return;
+ }//if
+ }//if
+/*---------------------------------------------------------------------------*/
+/* START EXECUTION OF A NEW MBYTE IN THE LOG. */
+/*---------------------------------------------------------------------------*/
+ if (logFilePtr.p->currentMbyte < (ZNO_MBYTES_IN_FILE - 1)) {
+ jam();
+ logPartPtr.p->logExecState = LogPartRecord::LES_EXEC_LOG_NEW_MBYTE;
+ } else {
+ ndbrequire(logFilePtr.p->currentMbyte == (ZNO_MBYTES_IN_FILE - 1));
+ jam();
+/*---------------------------------------------------------------------------*/
+/* WE HAVE TO CHANGE FILE. CLOSE THIS ONE AND THEN OPEN THE NEXT. */
+/*---------------------------------------------------------------------------*/
+ logPartPtr.p->logExecState = LogPartRecord::LES_EXEC_LOG_NEW_FILE;
+ }//if
+ break;
+/* ========================================================================= */
+/* ========================================================================= */
+ case ZCOMPLETED_GCI_TYPE:
+ jam();
+ logWord = readLogword(signal);
+ if (logWord == logPartPtr.p->logLastGci) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/* IF IT IS THE LAST GCI TO LIVE AFTER SYSTEM RESTART THEN WE RECORD THE NEXT*/
+/* WORD AS THE NEW HEADER OF THE LOG FILE. OTHERWISE WE SIMPLY IGNORE THIS */
+/* LOG RECORD. */
+/*---------------------------------------------------------------------------*/
+ if (csrPhasesCompleted == 0) {
+ jam();
+/*---------------------------------------------------------------------------*/
+/*WE ONLY RECORD THE HEAD OF THE LOG IN THE FIRST LOG ROUND OF LOG EXECUTION.*/
+/*---------------------------------------------------------------------------*/
+ logPartPtr.p->headFileNo = logFilePtr.p->fileNo;
+ logPartPtr.p->headPageNo = logFilePtr.p->currentFilepage;
+ logPartPtr.p->headPageIndex =
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ }//if
+/*---------------------------------------------------------------------------*/
+/* THERE IS NO NEED OF EXECUTING PAST THIS LINE SINCE THERE WILL ONLY BE LOG */
+/* RECORDS THAT WILL BE OF NO INTEREST. THUS CLOSE THE FILE AND START THE */
+/* NEXT PHASE OF THE SYSTEM RESTART. */
+/*---------------------------------------------------------------------------*/
+ logPartPtr.p->logExecState = LogPartRecord::LES_EXEC_LOG_COMPLETED;
+ }//if
+ break;
+ default:
+ jam();
+/* ========================================================================= */
+/* ========================================================================= */
+/*---------------------------------------------------------------------------*/
+/* SEND A SIGNAL TO THE SIGNAL LOG AND THEN CRASH THE SYSTEM. */
+/*---------------------------------------------------------------------------*/
+ signal->theData[0] = RNIL;
+ signal->theData[1] = logPartPtr.i;
+ Uint32 tmp = logFilePtr.p->fileName[3];
+ tmp = (tmp >> 8) & 0xff;// To get the Directory, DXX.
+ signal->theData[2] = tmp;
+ signal->theData[3] = logFilePtr.p->fileNo;
+ signal->theData[4] = logFilePtr.p->currentMbyte;
+ signal->theData[5] = logFilePtr.p->currentFilepage;
+ signal->theData[6] = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ signal->theData[7] = logWord;
+ sendSignal(cownref, GSN_DEBUG_SIG, signal, 8, JBA);
+ return;
+ break;
+ }//switch
+/*---------------------------------------------------------------------------*/
+// We continue to execute log records until we find a proper one to execute or
+// that we reach a new page.
+/*---------------------------------------------------------------------------*/
+ } while (1);
+}//Dblqh::execSr()
+
+/*---------------------------------------------------------------------------*/
+/* THIS SIGNAL IS ONLY RECEIVED TO BE CAPTURED IN THE SIGNAL LOG. IT IS */
+/* ALSO USED TO CRASH THE SYSTEM AFTER SENDING A SIGNAL TO THE LOG. */
+/*---------------------------------------------------------------------------*/
+void Dblqh::execDEBUG_SIG(Signal* signal)
+{
+/*
+2.5 TEMPORARY VARIABLES
+-----------------------
+*/
+ UintR tdebug;
+
+ jamEntry();
+ logPagePtr.i = signal->theData[0];
+ tdebug = logPagePtr.p->logPageWord[0];
+
+ char buf[100];
+ snprintf(buf, 100,
+ "Error while reading REDO log.\n"
+ "D=%d, F=%d Mb=%d FP=%d W1=%d W2=%d",
+ signal->theData[2], signal->theData[3], signal->theData[4],
+ signal->theData[5], signal->theData[6], signal->theData[7]);
+
+ progError(__LINE__, ERR_SR_REDOLOG, buf);
+
+ return;
+}//Dblqh::execDEBUG_SIG()
+
+/*---------------------------------------------------------------------------*/
+/*---------------------------------------------------------------------------*/
+void Dblqh::closeExecLogLab(Signal* signal)
+{
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSED;
+ signal->theData[0] = ZEXEC_SR;
+ signal->theData[1] = logFilePtr.p->logPartRec;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+}//Dblqh::closeExecLogLab()
+
+void Dblqh::openExecLogLab(Signal* signal)
+{
+ readExecLog(signal);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_EXEC_LOG;
+ return;
+}//Dblqh::openExecLogLab()
+
+void Dblqh::readExecLogLab(Signal* signal)
+{
+ buildLinkedLogPageList(signal);
+ logPartPtr.p->logExecState = LogPartRecord::LES_EXEC_LOGREC_FROM_FILE;
+ logPartPtr.p->execSrLfoRec = lfoPtr.i;
+ logPartPtr.p->execSrLogPage = logPagePtr.i;
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] =
+ logPartPtr.p->execSrLogPageIndex;
+ execLogRecord(signal);
+ return;
+}//Dblqh::readExecLogLab()
+
+/*---------------------------------------------------------------------------*/
+/* THIS CODE IS USED TO EXECUTE A LOG RECORD WHEN IT'S DATA HAVE BEEN LOCATED*/
+/* AND TRANSFERRED INTO MEMORY. */
+/*---------------------------------------------------------------------------*/
+void Dblqh::execLogRecord(Signal* signal)
+{
+ jamEntry();
+
+ tcConnectptr.i = logPartPtr.p->logTcConrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ // Read a log record and prepare it for execution
+ readLogHeader(signal);
+ readKey(signal);
+ readAttrinfo(signal);
+ initReqinfoExecSr(signal);
+ arrGuard(logPartPtr.p->execSrExecuteIndex, 4);
+ BlockReference ref = fragptr.p->execSrBlockref[logPartPtr.p->execSrExecuteIndex];
+ tcConnectptr.p->nextReplica = refToNode(ref);
+ tcConnectptr.p->connectState = TcConnectionrec::LOG_CONNECTED;
+ tcConnectptr.p->tcOprec = tcConnectptr.i;
+ packLqhkeyreqLab(signal);
+ return;
+}//Dblqh::execLogRecord()
+
+//----------------------------------------------------------------------------
+// This function invalidates log pages after the last GCI record in a
+// system/node restart. This is to ensure that the end of the log is
+// consistent. This function is executed last in start phase 3.
+// RT 450. EDTJAMO.
+//----------------------------------------------------------------------------
+void Dblqh::invalidateLogAfterLastGCI(Signal* signal) {
+
+ jam();
+ if (logPartPtr.p->logExecState != LogPartRecord::LES_EXEC_LOG_INVALIDATE) {
+ jam();
+ systemError(signal);
+ }
+
+ if (logFilePtr.p->fileNo != logPartPtr.p->invalidateFileNo) {
+ jam();
+ systemError(signal);
+ }
+
+ switch (lfoPtr.p->lfoState) {
+ case LogFileOperationRecord::WRITE_SR_INVALIDATE_PAGES:
+ jam();
+ releaseLfo(signal);
+ releaseLogpage(signal);
+ if (logPartPtr.p->invalidatePageNo < (ZNO_MBYTES_IN_FILE * ZPAGES_IN_MBYTE - 1)) {
+ // We continue in this file.
+ logPartPtr.p->invalidatePageNo++;
+ } else {
+ // We continue in the next file.
+ logFilePtr.i = logFilePtr.p->nextLogFile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPartPtr.p->invalidateFileNo = logFilePtr.p->fileNo;
+ // Page 0 is used for file descriptors.
+ logPartPtr.p->invalidatePageNo = 1;
+ if (logFilePtr.p->logFileStatus != LogFileRecord::OPEN) {
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN_SR_INVALIDATE_PAGES;
+ openFileRw(signal, logFilePtr);
+ return;
+ break;
+ }
+ }
+ // Read a page from the log file.
+ readFileInInvalidate(signal);
+ return;
+ break;
+
+ case LogFileOperationRecord::READ_SR_INVALIDATE_PAGES:
+ jam();
+ releaseLfo(signal);
+ // Check if this page must be invalidated.
+ // If the log lap number on a page after the head of the tail is the same
+ // as the actual log lap number we must invalidate this page. Otherwise it
+ // could be impossible to find the end of the log in a later system/node
+ // restart.
+ if (logPagePtr.p->logPageWord[ZPOS_LOG_LAP] == logPartPtr.p->logLap) {
+ // This page must be invalidated.
+ logPagePtr.p->logPageWord[ZPOS_LOG_LAP] = 0;
+ // Contact NDBFS. Real time break.
+ writeSinglePage(signal, logPartPtr.p->invalidatePageNo, ZPAGE_SIZE - 1);
+ lfoPtr.p->lfoState = LogFileOperationRecord::WRITE_SR_INVALIDATE_PAGES;
+ } else {
+ // We are done with invalidating. Finish start phase 3.4.
+ exitFromInvalidate(signal);
+ }
+ return;
+ break;
+
+ default:
+ jam();
+ systemError(signal);
+ return;
+ break;
+ }
+
+ return;
+}//Dblqh::invalidateLogAfterLastGCI
+
+void Dblqh::readFileInInvalidate(Signal* signal) {
+ jam();
+ // Contact NDBFS. Real time break.
+ readSinglePage(signal, logPartPtr.p->invalidatePageNo);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_SR_INVALIDATE_PAGES;
+}
+
+void Dblqh::exitFromInvalidate(Signal* signal) {
+ jam();
+ // Close files if necessary. Current file and the next file should be
+ // left open.
+ if (logFilePtr.i != logPartPtr.p->currentLogfile) {
+ LogFileRecordPtr currentLogFilePtr;
+ LogFileRecordPtr nextAfterCurrentLogFilePtr;
+
+ currentLogFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(currentLogFilePtr, clogFileFileSize, logFileRecord);
+
+ nextAfterCurrentLogFilePtr.i = currentLogFilePtr.p->nextLogFile;
+
+ if (logFilePtr.i != nextAfterCurrentLogFilePtr.i) {
+ // This file should be closed.
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSE_SR_INVALIDATE_PAGES;
+ closeFile(signal, logFilePtr);
+ // Return from this function and wait for close confirm. Then come back
+ // and test the previous file for closing.
+ return;
+ }
+ }
+
+ // We are done with closing files, send completed signal and exit this phase.
+ signal->theData[0] = ZSR_FOURTH_COMP;
+ signal->theData[1] = logPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+}
+
+
+/*---------------------------------------------------------------------------*/
+/* THE EXECUTION OF A LOG RECORD IS COMPLETED. RELEASE PAGES IF THEY WERE */
+/* READ FROM DISK FOR THIS PARTICULAR OPERATION. */
+/*---------------------------------------------------------------------------*/
+void Dblqh::completedLab(Signal* signal)
+{
+ Uint32 result = returnExecLog(signal);
+/*---------------------------------------------------------------------------*/
+/* ENTER COMPLETED WITH */
+/* LQH_CONNECTPTR */
+/*---------------------------------------------------------------------------*/
+ if (result == ZOK) {
+ jam();
+ execLogRecord(signal);
+ return;
+ } else if (result == ZNOT_OK) {
+ jam();
+ signal->theData[0] = ZEXEC_SR;
+ signal->theData[1] = logPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ } else {
+ jam();
+ /*empty*/;
+ }//if
+/*---------------------------------------------------------------------------*/
+/* WE HAVE TO WAIT FOR CLOSING OF THE EXECUTED LOG FILE BEFORE PROCEEDING IN */
+/* RARE CASES. */
+/*---------------------------------------------------------------------------*/
+ return;
+}//Dblqh::completedLab()
+
+/*---------------------------------------------------------------------------*/
+/* EXECUTION OF LOG RECORD WAS NOT SUCCESSFUL. CHECK IF IT IS OK ANYWAY, */
+/* THEN EXECUTE THE NEXT LOG RECORD. */
+/*---------------------------------------------------------------------------*/
+void Dblqh::logLqhkeyrefLab(Signal* signal)
+{
+ Uint32 result = returnExecLog(signal);
+ switch (tcConnectptr.p->operation) {
+ case ZUPDATE:
+ case ZDELETE:
+ jam();
+ ndbrequire(terrorCode == ZNO_TUPLE_FOUND);
+ break;
+ case ZINSERT:
+ jam();
+ ndbrequire(terrorCode == ZTUPLE_ALREADY_EXIST);
+ break;
+ default:
+ ndbrequire(false);
+ return;
+ break;
+ }//switch
+ if (result == ZOK) {
+ jam();
+ execLogRecord(signal);
+ return;
+ } else if (result == ZNOT_OK) {
+ jam();
+ signal->theData[0] = ZEXEC_SR;
+ signal->theData[1] = logPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ } else {
+ jam();
+ /*empty*/;
+ }//if
+ /* ------------------------------------------------------------------------
+ * WE HAVE TO WAIT FOR CLOSING OF THE EXECUTED LOG FILE BEFORE
+ * PROCEEDING IN RARE CASES.
+ * ----------------------------------------------------------------------- */
+ return;
+}//Dblqh::logLqhkeyrefLab()
+
+void Dblqh::closeExecSrCompletedLab(Signal* signal)
+{
+ logFilePtr.p->logFileStatus = LogFileRecord::CLOSED;
+ signal->theData[0] = logFilePtr.p->logPartRec;
+ execLogComp(signal);
+ return;
+}//Dblqh::closeExecSrCompletedLab()
+
+/* --------------------------------------------------------------------------
+ * ONE OF THE LOG PARTS HAVE COMPLETED EXECUTING THE LOG. CHECK IF ALL LOG
+ * PARTS ARE COMPLETED. IF SO START SENDING EXEC_FRAGCONF AND EXEC_SRCONF.
+ * ------------------------------------------------------------------------- */
+void Dblqh::execLogComp(Signal* signal)
+{
+ logPartPtr.i = signal->theData[0];
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ logPartPtr.p->logPartState = LogPartRecord::SR_THIRD_PHASE_COMPLETED;
+ /* ------------------------------------------------------------------------
+ * WE MUST RELEASE THE TC CONNECT RECORD HERE SO THAT IT CAN BE REUSED.
+ * ----------------------------------------------------------------------- */
+ tcConnectptr.i = logPartPtr.p->logTcConrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ releaseTcrecLog(signal, tcConnectptr);
+ for (logPartPtr.i = 0; logPartPtr.i <= 3; logPartPtr.i++) {
+ jam();
+ ptrAss(logPartPtr, logPartRecord);
+ if (logPartPtr.p->logPartState != LogPartRecord::SR_THIRD_PHASE_COMPLETED) {
+ if (logPartPtr.p->logPartState != LogPartRecord::SR_THIRD_PHASE_STARTED) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ } else {
+ jam();
+ /* ------------------------------------------------------------------
+ * THIS LOG PART WAS NOT COMPLETED YET. EXIT AND WAIT FOR IT
+ * TO COMPLETE
+ * ----------------------------------------------------------------- */
+ return;
+ }//if
+ }//if
+ }//for
+ /* ------------------------------------------------------------------------
+ * ALL LOG PARTS HAVE COMPLETED THE EXECUTION OF THE LOG. WE CAN NOW START
+ * SENDING THE EXEC_FRAGCONF SIGNALS TO ALL INVOLVED FRAGMENTS.
+ * ----------------------------------------------------------------------- */
+ if (cstartType != NodeState::ST_NODE_RESTART) {
+ jam();
+ signal->theData[0] = ZSEND_EXEC_CONF;
+ signal->theData[1] = 0;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ } else {
+ jam();
+ /* ----------------------------------------------------------------------
+ * FOR NODE RESTART WE CAN SKIP A NUMBER OF STEPS SINCE WE HAVE NO
+ * FRAGMENTS DEFINED AT THIS POINT. OBVIOUSLY WE WILL NOT NEED TO
+ * EXECUTE ANY MORE LOG STEPS EITHER AND THUS WE CAN IMMEDIATELY
+ * START FINDING THE END AND THE START OF THE LOG.
+ * --------------------------------------------------------------------- */
+ csrPhasesCompleted = 3;
+ execSrCompletedLab(signal);
+ return;
+ }//if
+ return;
+}//Dblqh::execLogComp()
+
+/* --------------------------------------------------------------------------
+ * GO THROUGH THE FRAGMENT RECORDS TO DEDUCE TO WHICH SHALL BE SENT
+ * EXEC_FRAGCONF AFTER COMPLETING THE EXECUTION OF THE LOG.
+ * ------------------------------------------------------------------------- */
+void Dblqh::sendExecConf(Signal* signal)
+{
+ jamEntry();
+ fragptr.i = signal->theData[0];
+ Uint32 loopCount = 0;
+ while (fragptr.i < cfragrecFileSize) {
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ if (fragptr.p->execSrStatus != Fragrecord::IDLE) {
+ jam();
+ ndbrequire(fragptr.p->execSrNoReplicas - 1 < 4);
+ for (Uint32 i = 0; i < fragptr.p->execSrNoReplicas; i++) {
+ jam();
+ signal->theData[0] = fragptr.p->execSrUserptr[i];
+ sendSignal(fragptr.p->execSrBlockref[i], GSN_EXEC_FRAGCONF,
+ signal, 1, JBB);
+ }//for
+ if (fragptr.p->execSrStatus == Fragrecord::ACTIVE) {
+ jam();
+ fragptr.p->execSrStatus = Fragrecord::IDLE;
+ } else {
+ ndbrequire(fragptr.p->execSrStatus == Fragrecord::ACTIVE_REMOVE_AFTER);
+ jam();
+ Uint32 fragId = fragptr.p->fragId;
+ tabptr.i = fragptr.p->tabRef;
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ deleteFragrec(fragId);
+ }//if
+ fragptr.p->execSrNoReplicas = 0;
+ }//if
+ loopCount++;
+ if (loopCount > 20) {
+ jam();
+ signal->theData[0] = ZSEND_EXEC_CONF;
+ signal->theData[1] = fragptr.i + 1;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ } else {
+ jam();
+ fragptr.i++;
+ }//if
+ }//while
+ /* ----------------------------------------------------------------------
+ * WE HAVE NOW SENT ALL EXEC_FRAGCONF. NOW IT IS TIME TO SEND
+ * EXEC_SRCONF TO ALL NODES.
+ * --------------------------------------------------------------------- */
+ srPhase3Comp(signal);
+}//Dblqh::sendExecConf()
+
+/* --------------------------------------------------------------------------
+ * PHASE 3 HAS NOW COMPLETED. INFORM ALL OTHER NODES OF THIS EVENT.
+ * ------------------------------------------------------------------------- */
+void Dblqh::srPhase3Comp(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(cnoOfNodes < MAX_NDB_NODES);
+ for (Uint32 i = 0; i < cnoOfNodes; i++) {
+ jam();
+ if (cnodeStatus[i] == ZNODE_UP) {
+ jam();
+ ndbrequire(cnodeData[i] < MAX_NDB_NODES);
+ BlockReference ref = calcLqhBlockRef(cnodeData[i]);
+ signal->theData[0] = cownNodeid;
+ sendSignal(ref, GSN_EXEC_SRCONF, signal, 1, JBB);
+ }//if
+ }//for
+ return;
+}//Dblqh::srPhase3Comp()
+
+/* ##########################################################################
+ * SYSTEM RESTART PHASE FOUR MODULE
+ * THIS MODULE IS A SUB-MODULE OF THE FILE SYSTEM HANDLING.
+ *
+ * THIS MODULE SETS UP THE HEAD AND TAIL POINTERS OF THE LOG PARTS IN THE
+ * FRAGMENT LOG. WHEN IT IS COMPLETED IT REPORTS TO THE MASTER DIH THAT
+ * IT HAS COMPLETED THE PART OF THE SYSTEM RESTART WHERE THE DATABASE IS
+ * LOADED.
+ * IT ALSO OPENS THE CURRENT LOG FILE AND THE NEXT AND SETS UP THE FIRST
+ * LOG PAGE WHERE NEW LOG DATA IS TO BE INSERTED WHEN THE SYSTEM STARTS
+ * AGAIN.
+ *
+ * THIS PART IS ACTUALLY EXECUTED FOR ALL RESTART TYPES.
+ * ######################################################################### */
+void Dblqh::initFourth(Signal* signal)
+{
+ LogFileRecordPtr locLogFilePtr;
+ jamEntry();
+ logPartPtr.i = signal->theData[0];
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ crestartNewestGci = 1;
+ crestartOldestGci = 1;
+ /* ------------------------------------------------------------------------
+ * INITIALISE LOG PART AND LOG FILES AS NEEDED.
+ * ----------------------------------------------------------------------- */
+ logPartPtr.p->headFileNo = 0;
+ logPartPtr.p->headPageNo = 1;
+ logPartPtr.p->headPageIndex = ZPAGE_HEADER_SIZE + 2;
+ logPartPtr.p->logPartState = LogPartRecord::SR_FOURTH_PHASE_STARTED;
+ logPartPtr.p->logTailFileNo = 0;
+ logPartPtr.p->logTailMbyte = 0;
+ locLogFilePtr.i = logPartPtr.p->firstLogfile;
+ ptrCheckGuard(locLogFilePtr, clogFileFileSize, logFileRecord);
+ locLogFilePtr.p->logFileStatus = LogFileRecord::OPEN_SR_FOURTH_PHASE;
+ openFileRw(signal, locLogFilePtr);
+ return;
+}//Dblqh::initFourth()
+
+void Dblqh::openSrFourthPhaseLab(Signal* signal)
+{
+ /* ------------------------------------------------------------------------
+ * WE HAVE NOW OPENED THE HEAD LOG FILE WE WILL NOW START READING IT
+ * FROM THE HEAD MBYTE TO FIND THE NEW HEAD OF THE LOG.
+ * ----------------------------------------------------------------------- */
+ readSinglePage(signal, logPartPtr.p->headPageNo);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_SR_FOURTH_PHASE;
+ return;
+}//Dblqh::openSrFourthPhaseLab()
+
+void Dblqh::readSrFourthPhaseLab(Signal* signal)
+{
+ /* ------------------------------------------------------------------------
+ * INITIALISE ALL LOG PART INFO AND LOG FILE INFO THAT IS NEEDED TO
+ * START UP THE SYSTEM.
+ * ------------------------------------------------------------------------
+ * INITIALISE THE NEWEST GLOBAL CHECKPOINT IDENTITY AND THE NEWEST
+ * COMPLETED GLOBAL CHECKPOINT IDENITY AS THE NEWEST THAT WAS RESTARTED.
+ * ------------------------------------------------------------------------
+ * INITIALISE THE HEAD PAGE INDEX IN THIS PAGE.
+ * ASSIGN IT AS THE CURRENT LOGPAGE.
+ * ASSIGN THE FILE AS THE CURRENT LOG FILE.
+ * ASSIGN THE CURRENT FILE NUMBER FROM THE CURRENT LOG FILE AND THE NEXT
+ * FILE NUMBER FROM THE NEXT LOG FILE.
+ * ASSIGN THE CURRENT FILEPAGE FROM HEAD PAGE NUMBER.
+ * ASSIGN THE CURRENT MBYTE BY DIVIDING PAGE NUMBER BY 128.
+ * INITIALISE LOG LAP TO BE THE LOG LAP AS FOUND IN THE HEAD PAGE.
+ * WE HAVE TO CALCULATE THE NUMBER OF REMAINING WORDS IN THIS MBYTE.
+ * ----------------------------------------------------------------------- */
+ cnewestGci = crestartNewestGci;
+ cnewestCompletedGci = crestartNewestGci;
+ logPartPtr.p->logPartNewestCompletedGCI = cnewestCompletedGci;
+ logPartPtr.p->currentLogfile = logFilePtr.i;
+ logFilePtr.p->filePosition = logPartPtr.p->headPageNo;
+ logFilePtr.p->currentMbyte =
+ logPartPtr.p->headPageNo >> ZTWOLOG_NO_PAGES_IN_MBYTE;
+ logFilePtr.p->fileChangeState = LogFileRecord::NOT_ONGOING;
+ logPartPtr.p->logLap = logPagePtr.p->logPageWord[ZPOS_LOG_LAP];
+ logFilePtr.p->currentFilepage = logPartPtr.p->headPageNo;
+ logFilePtr.p->currentLogpage = logPagePtr.i;
+ initLogpage(signal);
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPartPtr.p->headPageIndex;
+ logFilePtr.p->remainingWordsInMbyte =
+ ((
+ ((logFilePtr.p->currentMbyte + 1) * ZPAGES_IN_MBYTE) -
+ logFilePtr.p->currentFilepage) *
+ (ZPAGE_SIZE - ZPAGE_HEADER_SIZE)) -
+ (logPartPtr.p->headPageIndex - ZPAGE_HEADER_SIZE);
+ /* ------------------------------------------------------------------------
+ * THE NEXT STEP IS TO OPEN THE NEXT LOG FILE (IF THERE IS ONE).
+ * ----------------------------------------------------------------------- */
+ if (logFilePtr.p->nextLogFile != logFilePtr.i) {
+ LogFileRecordPtr locLogFilePtr;
+ jam();
+ locLogFilePtr.i = logFilePtr.p->nextLogFile;
+ ptrCheckGuard(locLogFilePtr, clogFileFileSize, logFileRecord);
+ locLogFilePtr.p->logFileStatus = LogFileRecord::OPEN_SR_FOURTH_NEXT;
+ openFileRw(signal, locLogFilePtr);
+ } else {
+ jam();
+ /* ----------------------------------------------------------------------
+ * THIS CAN ONLY OCCUR IF WE HAVE ONLY ONE LOG FILE. THIS LOG FILE MUST
+ * BE LOG FILE ZERO AND THAT IS THE FILE WE CURRENTLY HAVE READ.
+ * THUS WE CAN CONTINUE IMMEDIATELY TO READ PAGE ZERO IN FILE ZERO.
+ * --------------------------------------------------------------------- */
+ openSrFourthZeroSkipInitLab(signal);
+ return;
+ }//if
+ return;
+}//Dblqh::readSrFourthPhaseLab()
+
+void Dblqh::openSrFourthNextLab(Signal* signal)
+{
+ /* ------------------------------------------------------------------------
+ * WE MUST ALSO HAVE FILE 0 OPEN ALL THE TIME.
+ * ----------------------------------------------------------------------- */
+ logFilePtr.i = logPartPtr.p->firstLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ if (logFilePtr.p->logFileStatus == LogFileRecord::OPEN) {
+ jam();
+ openSrFourthZeroSkipInitLab(signal);
+ return;
+ } else {
+ jam();
+ logFilePtr.p->logFileStatus = LogFileRecord::OPEN_SR_FOURTH_ZERO;
+ openFileRw(signal, logFilePtr);
+ }//if
+ return;
+}//Dblqh::openSrFourthNextLab()
+
+void Dblqh::openSrFourthZeroLab(Signal* signal)
+{
+ openSrFourthZeroSkipInitLab(signal);
+ return;
+}//Dblqh::openSrFourthZeroLab()
+
+void Dblqh::openSrFourthZeroSkipInitLab(Signal* signal)
+{
+ if (logFilePtr.i == logPartPtr.p->currentLogfile) {
+ if (logFilePtr.p->currentFilepage == 0) {
+ jam();
+ /* -------------------------------------------------------------------
+ * THE HEADER PAGE IN THE LOG IS PAGE ZERO IN FILE ZERO.
+ * THIS SHOULD NEVER OCCUR.
+ * ------------------------------------------------------------------- */
+ systemErrorLab(signal);
+ return;
+ }//if
+ }//if
+ readSinglePage(signal, 0);
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_SR_FOURTH_ZERO;
+ return;
+}//Dblqh::openSrFourthZeroSkipInitLab()
+
+void Dblqh::readSrFourthZeroLab(Signal* signal)
+{
+ logFilePtr.p->logPageZero = logPagePtr.i;
+ // --------------------------------------------------------------------
+ // This is moved to invalidateLogAfterLastGCI(), RT453.
+ // signal->theData[0] = ZSR_FOURTH_COMP;
+ // signal->theData[1] = logPartPtr.i;
+ // sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ // --------------------------------------------------------------------
+
+ // Need to invalidate log pages after the head of the log. RT 453. EDTJAMO.
+ // Set the start of the invalidation.
+ logFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPartPtr.p->invalidateFileNo = logPartPtr.p->headFileNo;
+ logPartPtr.p->invalidatePageNo = logPartPtr.p->headPageNo;
+
+ logPartPtr.p->logExecState = LogPartRecord::LES_EXEC_LOG_INVALIDATE;
+ seizeLfo(signal);
+ initLfo(signal);
+ // The state here is a little confusing, but simulates that we return
+ // to invalidateLogAfterLastGCI() from an invalidate write and are ready
+ // to read a page from file.
+ lfoPtr.p->lfoState = LogFileOperationRecord::WRITE_SR_INVALIDATE_PAGES;
+
+ invalidateLogAfterLastGCI(signal);
+ return;
+}//Dblqh::readSrFourthZeroLab()
+
+/* --------------------------------------------------------------------------
+ * ONE OF THE LOG PARTS HAVE COMPLETED PHASE FOUR OF THE SYSTEM RESTART.
+ * CHECK IF ALL LOG PARTS ARE COMPLETED. IF SO SEND START_RECCONF
+ * ------------------------------------------------------------------------- */
+void Dblqh::srFourthComp(Signal* signal)
+{
+ jamEntry();
+ logPartPtr.i = signal->theData[0];
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ logPartPtr.p->logPartState = LogPartRecord::SR_FOURTH_PHASE_COMPLETED;
+ for (logPartPtr.i = 0; logPartPtr.i <= 3; logPartPtr.i++) {
+ jam();
+ ptrAss(logPartPtr, logPartRecord);
+ if (logPartPtr.p->logPartState != LogPartRecord::SR_FOURTH_PHASE_COMPLETED) {
+ if (logPartPtr.p->logPartState != LogPartRecord::SR_FOURTH_PHASE_STARTED) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ } else {
+ jam();
+ /* ------------------------------------------------------------------
+ * THIS LOG PART WAS NOT COMPLETED YET.
+ * EXIT AND WAIT FOR IT TO COMPLETE
+ * ----------------------------------------------------------------- */
+ return;
+ }//if
+ }//if
+ }//for
+ /* ------------------------------------------------------------------------
+ * ALL LOG PARTS HAVE COMPLETED PHASE FOUR OF THE SYSTEM RESTART.
+ * WE CAN NOW SEND START_RECCONF TO THE MASTER DIH IF IT WAS A
+ * SYSTEM RESTART. OTHERWISE WE WILL CONTINUE WITH AN INITIAL START.
+ * SET LOG PART STATE TO IDLE TO
+ * INDICATE THAT NOTHING IS GOING ON IN THE LOG PART.
+ * ----------------------------------------------------------------------- */
+ for (logPartPtr.i = 0; logPartPtr.i <= 3; logPartPtr.i++) {
+ ptrAss(logPartPtr, logPartRecord);
+ logPartPtr.p->logPartState = LogPartRecord::IDLE;
+ }//for
+
+ if ((cstartType == NodeState::ST_INITIAL_START) ||
+ (cstartType == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+
+ ndbrequire(cinitialStartOngoing == ZTRUE);
+ cinitialStartOngoing = ZFALSE;
+
+ checkStartCompletedLab(signal);
+ return;
+ } else if ((cstartType == NodeState::ST_NODE_RESTART) ||
+ (cstartType == NodeState::ST_SYSTEM_RESTART)) {
+ jam();
+ StartRecConf * conf = (StartRecConf*)signal->getDataPtrSend();
+ conf->startingNodeId = getOwnNodeId();
+ sendSignal(cmasterDihBlockref, GSN_START_RECCONF, signal,
+ StartRecConf::SignalLength, JBB);
+ } else {
+ ndbrequire(false);
+ }//if
+ return;
+}//Dblqh::srFourthComp()
+
+/* ######################################################################### */
+/* ####### ERROR MODULE ####### */
+/* */
+/* ######################################################################### */
+void Dblqh::warningHandlerLab(Signal* signal)
+{
+ systemErrorLab(signal);
+ return;
+}//Dblqh::warningHandlerLab()
+
+/*---------------------------------------------------------------------------*/
+/* AN ERROR OCCURRED THAT WE WILL NOT TREAT AS SYSTEM ERROR. MOST OFTEN THIS */
+/* WAS CAUSED BY AN ERRONEUS SIGNAL SENT BY ANOTHER NODE. WE DO NOT WISH TO */
+/* CRASH BECAUSE OF FAULTS IN OTHER NODES. THUS WE ONLY REPORT A WARNING. */
+/* THIS IS CURRENTLY NOT IMPLEMENTED AND FOR THE MOMENT WE GENERATE A SYSTEM */
+/* ERROR SINCE WE WANT TO FIND FAULTS AS QUICKLY AS POSSIBLE IN A TEST PHASE.*/
+/* IN A LATER PHASE WE WILL CHANGE THIS TO BE A WARNING MESSAGE INSTEAD. */
+/*---------------------------------------------------------------------------*/
+/*---------------------------------------------------------------------------*/
+/* THIS TYPE OF ERROR SHOULD NOT GENERATE A SYSTEM ERROR IN A PRODUCT */
+/* RELEASE. THIS IS A TEMPORARY SOLUTION DURING TEST PHASE TO QUICKLY */
+/* FIND ERRORS. NORMALLY THIS SHOULD GENERATE A WARNING MESSAGE ONTO */
+/* SOME ERROR LOGGER. THIS WILL LATER BE IMPLEMENTED BY SOME SIGNAL. */
+/*---------------------------------------------------------------------------*/
+/* ------ SYSTEM ERROR SITUATIONS ------- */
+/* IN SITUATIONS WHERE THE STATE IS ERRONEOUS OR IF THE ERROR OCCURS IN */
+/* THE COMMIT, COMPLETE OR ABORT PHASE, WE PERFORM A CRASH OF THE AXE VM*/
+/*---------------------------------------------------------------------------*/
+
+void Dblqh::systemErrorLab(Signal* signal)
+{
+ progError(0, 0);
+/*************************************************************************>*/
+/* WE WANT TO INVOKE AN IMMEDIATE ERROR HERE SO WE GET THAT BY */
+/* INSERTING A CERTAIN POINTER OUT OF RANGE. */
+/*************************************************************************>*/
+}//Dblqh::systemErrorLab()
+
+/* ------- ERROR SITUATIONS ------- */
+
+void Dblqh::aiStateErrorCheckLab(Signal* signal, Uint32* dataPtr, Uint32 length)
+{
+ ndbrequire(tcConnectptr.p->abortState != TcConnectionrec::ABORT_IDLE);
+ if (tcConnectptr.p->transactionState != TcConnectionrec::IDLE) {
+ jam();
+/*************************************************************************>*/
+/* TRANSACTION ABORT IS ONGOING. IT CAN STILL BE A PART OF AN */
+/* OPERATION THAT SHOULD CONTINUE SINCE THE TUPLE HAS NOT ARRIVED */
+/* YET. THIS IS POSSIBLE IF ACTIVE CREATION OF THE FRAGMENT IS */
+/* ONGOING. */
+/*************************************************************************>*/
+ if (tcConnectptr.p->activeCreat == ZTRUE) {
+ jam();
+/*************************************************************************>*/
+/* ONGOING ABORTS DURING ACTIVE CREATION MUST SAVE THE ATTRIBUTE INFO*/
+/* SO THAT IT CAN BE SENT TO THE NEXT NODE IN THE COMMIT CHAIN. THIS */
+/* IS NEEDED SINCE ALL ABORTS DURING CREATION OF A FRAGMENT ARE NOT */
+/* REALLY ERRORS. A MISSING TUPLE TO BE UPDATED SIMPLY MEANS THAT */
+/* IT HASN'T BEEN TRANSFERRED TO THE NEW REPLICA YET. */
+/*************************************************************************>*/
+/*************************************************************************>*/
+/* AFTER THIS ERROR THE ABORT MUST BE COMPLETED. TO ENSURE THIS SET */
+/* ACTIVE CREATION TO FALSE. THIS WILL ENSURE THAT THE ABORT IS */
+/* COMPLETED. */
+/*************************************************************************>*/
+ if (saveTupattrbuf(signal, dataPtr, length) == ZOK) {
+ jam();
+ if (tcConnectptr.p->transactionState ==
+ TcConnectionrec::WAIT_AI_AFTER_ABORT) {
+ if (tcConnectptr.p->currTupAiLen == tcConnectptr.p->totReclenAi) {
+ jam();
+/*************************************************************************>*/
+/* WE WERE WAITING FOR MORE ATTRIBUTE INFO AFTER A SUCCESSFUL ABORT */
+/* IN ACTIVE CREATION STATE. THE TRANSACTION SHOULD CONTINUE AS IF */
+/* IT WAS COMMITTED. NOW ALL INFO HAS ARRIVED AND WE CAN CONTINUE */
+/* WITH NORMAL PROCESSING AS IF THE TRANSACTION WAS PREPARED. */
+/* SINCE THE FRAGMENT IS UNDER CREATION WE KNOW THAT LOGGING IS */
+/* DISABLED. WE STILL HAVE TO CATER FOR DIRTY OPERATION OR NOT. */
+/*************************************************************************>*/
+ tcConnectptr.p->abortState = TcConnectionrec::ABORT_IDLE;
+ rwConcludedAiLab(signal);
+ return;
+ } else {
+ ndbrequire(tcConnectptr.p->currTupAiLen < tcConnectptr.p->totReclenAi);
+ jam();
+ return; /* STILL WAITING FOR MORE ATTRIBUTE INFO */
+ }//if
+ }//if
+ } else {
+ jam();
+/*************************************************************************>*/
+/* AFTER THIS ERROR THE ABORT MUST BE COMPLETED. TO ENSURE THIS SET */
+/* ACTIVE CREATION TO ABORT. THIS WILL ENSURE THAT THE ABORT IS */
+/* COMPLETED AND THAT THE ERROR CODE IS PROPERLY SET */
+/*************************************************************************>*/
+ tcConnectptr.p->errorCode = terrorCode;
+ tcConnectptr.p->activeCreat = ZFALSE;
+ if (tcConnectptr.p->transactionState ==
+ TcConnectionrec::WAIT_AI_AFTER_ABORT) {
+ jam();
+/*************************************************************************>*/
+/* ABORT IS ALREADY COMPLETED. WE NEED TO RESTART IT FROM WHERE IT */
+/* WAS INTERRUPTED. */
+/*************************************************************************>*/
+ continueAbortLab(signal);
+ return;
+ } else {
+ jam();
+ return;
+/*************************************************************************>*/
+// Abort is ongoing. It will complete since we set the activeCreat = ZFALSE
+/*************************************************************************>*/
+ }//if
+ }//if
+ }//if
+ }//if
+/*************************************************************************>*/
+/* TRANSACTION HAVE BEEN ABORTED. THUS IGNORE ALL SIGNALS BELONGING TO IT. */
+/*************************************************************************>*/
+ return;
+}//Dblqh::aiStateErrorCheckLab()
+
+void Dblqh::takeOverErrorLab(Signal* signal)
+{
+ terrorCode = ZTAKE_OVER_ERROR;
+ abortErrorLab(signal);
+ return;
+}//Dblqh::takeOverErrorLab()
+
+/* ##########################################################################
+ * TEST MODULE
+ * ######################################################################### */
+#ifdef VM_TRACE
+void Dblqh::execTESTSIG(Signal* signal)
+{
+ jamEntry();
+ Uint32 userpointer = signal->theData[0];
+ BlockReference userblockref = signal->theData[1];
+ Uint32 testcase = signal->theData[2];
+
+ signal->theData[0] = userpointer;
+ signal->theData[1] = cownref;
+ signal->theData[2] = testcase;
+ sendSignal(userblockref, GSN_TESTSIG, signal, 25, JBB);
+ return;
+}//Dblqh::execTESTSIG()
+
+/* *************** */
+/* MEMCHECKREQ > */
+/* *************** */
+/* ************************************************************************>>
+ * THIS SIGNAL IS PURELY FOR TESTING PURPOSES. IT CHECKS THE FREE LIST
+ * AND REPORTS THE NUMBER OF FREE RECORDS.
+ * THIS CAN BE DONE TO ENSURE THAT NO RECORDS HAS BEEN LOST
+ * ************************************************************************> */
+void Dblqh::execMEMCHECKREQ(Signal* signal)
+{
+ Uint32* dataPtr = &signal->theData[0];
+ jamEntry();
+ BlockReference userblockref = signal->theData[0];
+ Uint32 index = 0;
+ for (Uint32 i = 0; i < 7; i++)
+ dataPtr[i] = 0;
+ addfragptr.i = cfirstfreeAddfragrec;
+ while (addfragptr.i != RNIL) {
+ ptrCheckGuard(addfragptr, caddfragrecFileSize, addFragRecord);
+ addfragptr.i = addfragptr.p->nextAddfragrec;
+ dataPtr[index]++;
+ }//while
+ index++;
+ attrinbufptr.i = cfirstfreeAttrinbuf;
+ while (attrinbufptr.i != RNIL) {
+ ptrCheckGuard(attrinbufptr, cattrinbufFileSize, attrbuf);
+ attrinbufptr.i = attrinbufptr.p->attrbuf[ZINBUF_NEXT];
+ dataPtr[index]++;
+ }//while
+ index++;
+ databufptr.i = cfirstfreeDatabuf;
+ while (databufptr.i != RNIL) {
+ ptrCheckGuard(databufptr, cdatabufFileSize, databuf);
+ databufptr.i = databufptr.p->nextDatabuf;
+ dataPtr[index]++;
+ }//while
+ index++;
+ fragptr.i = cfirstfreeFragrec;
+ while (fragptr.i != RNIL) {
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ fragptr.i = fragptr.p->nextFrag;
+ dataPtr[index]++;
+ }//while
+ index++;
+ for (tabptr.i = 0;
+ tabptr.i < ctabrecFileSize;
+ tabptr.i++) {
+ ptrAss(tabptr, tablerec);
+ if (tabptr.p->tableStatus == Tablerec::NOT_DEFINED) {
+ dataPtr[index]++;
+ }//if
+ }//for
+ index++;
+ tcConnectptr.i = cfirstfreeTcConrec;
+ while (tcConnectptr.i != RNIL) {
+ ptrCheckGuard(tcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ tcConnectptr.i = tcConnectptr.p->nextTcConnectrec;
+ dataPtr[index]++;
+ }//while
+ sendSignal(userblockref, GSN_MEMCHECKCONF, signal, 10, JBB);
+ return;
+}//Dblqh::execMEMCHECKREQ()
+
+#endif
+
+/* ************************************************************************* */
+/* ************************* STATEMENT BLOCKS ****************************** */
+/* ************************************************************************* */
+/* ========================================================================= */
+/* ====== BUILD LINKED LIST OF LOG PAGES AFTER RECEIVING FSREADCONF ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::buildLinkedLogPageList(Signal* signal)
+{
+ LogPageRecordPtr bllLogPagePtr;
+
+ arrGuard(lfoPtr.p->noPagesRw - 1, 16);
+ arrGuard(lfoPtr.p->noPagesRw, 16);
+ for (UintR tbllIndex = 0; tbllIndex < lfoPtr.p->noPagesRw; tbllIndex++) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * BUILD LINKED LIST BUT ALSO ENSURE THAT PAGE IS NOT SEEN AS DIRTY
+ * INITIALLY.
+ * --------------------------------------------------------------------- */
+ bllLogPagePtr.i = lfoPtr.p->logPageArray[tbllIndex];
+ ptrCheckGuard(bllLogPagePtr, clogPageFileSize, logPageRecord);
+
+// #if VM_TRACE
+// // Check logPage checksum before modifying it
+// Uint32 calcCheckSum = calcPageCheckSum(bllLogPagePtr);
+// Uint32 checkSum = bllLogPagePtr.p->logPageWord[ZPOS_CHECKSUM];
+// if (checkSum != calcCheckSum) {
+// ndbout << "Redolog: Checksum failure." << endl;
+// progError(__LINE__, ERR_NDBREQUIRE, "Redolog: Checksum failure.");
+// }
+// #endif
+
+ bllLogPagePtr.p->logPageWord[ZNEXT_PAGE] =
+ lfoPtr.p->logPageArray[tbllIndex + 1];
+ bllLogPagePtr.p->logPageWord[ZPOS_DIRTY] = ZNOT_DIRTY;
+ }//for
+ bllLogPagePtr.i = lfoPtr.p->logPageArray[lfoPtr.p->noPagesRw - 1];
+ ptrCheckGuard(bllLogPagePtr, clogPageFileSize, logPageRecord);
+ bllLogPagePtr.p->logPageWord[ZNEXT_PAGE] = RNIL;
+}//Dblqh::buildLinkedLogPageList()
+
+/* =========================================================================
+ * ======= CHANGE TO NEXT MBYTE IN LOG =======
+ *
+ * ========================================================================= */
+void Dblqh::changeMbyte(Signal* signal)
+{
+ writeNextLog(signal);
+ writeFileDescriptor(signal);
+}//Dblqh::changeMbyte()
+
+/* ========================================================================= */
+/* ====== CHECK IF THIS COMMIT LOG RECORD IS TO BE EXECUTED ======= */
+/* */
+/* SUBROUTINE SHORT NAME = CEL */
+/* ========================================================================= */
+Uint32 Dblqh::checkIfExecLog(Signal* signal)
+{
+ tabptr.i = tcConnectptr.p->tableref;
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ if (getFragmentrec(signal, tcConnectptr.p->fragmentid) &&
+ (tabptr.p->schemaVersion == tcConnectptr.p->schemaVersion)) {
+ if (fragptr.p->execSrStatus != Fragrecord::IDLE) {
+ if (fragptr.p->execSrNoReplicas > logPartPtr.p->execSrExecuteIndex) {
+ ndbrequire((fragptr.p->execSrNoReplicas - 1) < 4);
+ for (Uint32 i = logPartPtr.p->execSrExecuteIndex;
+ i < fragptr.p->execSrNoReplicas;
+ i++) {
+ jam();
+ if (tcConnectptr.p->gci >= fragptr.p->execSrStartGci[i]) {
+ if (tcConnectptr.p->gci <= fragptr.p->execSrLastGci[i]) {
+ jam();
+ logPartPtr.p->execSrExecuteIndex = i;
+ return ZOK;
+ }//if
+ }//if
+ }//for
+ }//if
+ }//if
+ }//if
+ return ZNOT_OK;
+}//Dblqh::checkIfExecLog()
+
+/* ========================================================================= */
+/* == CHECK IF THERE IS LESS THAN 192 KBYTE IN THE BUFFER PLUS INCOMING === */
+/* READS ALREADY STARTED. IF SO IS THE CASE THEN START ANOTHER READ IF */
+/* THERE ARE MORE PAGES IN THIS MBYTE. */
+/* */
+/* ========================================================================= */
+void Dblqh::checkReadExecSr(Signal* signal)
+{
+ logPartPtr.p->logExecState = LogPartRecord::LES_EXEC_LOG;
+ logPartPtr.p->execSrPagesRead = logPartPtr.p->execSrPagesRead + 8;
+ logPartPtr.p->execSrPagesReading = logPartPtr.p->execSrPagesReading - 8;
+ if ((logPartPtr.p->execSrPagesRead + logPartPtr.p->execSrPagesReading) <
+ ZREAD_AHEAD_SIZE) {
+ jam();
+ /* ----------------------------------------------------------------------
+ * WE HAVE LESS THAN 64 KBYTE OF LOG PAGES REMAINING IN MEMORY OR ON
+ * ITS WAY TO MAIN MEMORY. READ IN 8 MORE PAGES.
+ * --------------------------------------------------------------------- */
+ if ((logPartPtr.p->execSrPagesRead + logPartPtr.p->execSrPagesExecuted) <
+ ZPAGES_IN_MBYTE) {
+ jam();
+ /* --------------------------------------------------------------------
+ * THERE ARE MORE PAGES TO READ IN THIS MBYTE. READ THOSE FIRST
+ * IF >= ZPAGES_IN_MBYTE THEN THERE ARE NO MORE PAGES TO READ. THUS
+ * WE PROCEED WITH EXECUTION OF THE LOG.
+ * ------------------------------------------------------------------- */
+ readExecSr(signal);
+ logPartPtr.p->logExecState = LogPartRecord::LES_WAIT_READ_EXEC_SR;
+ }//if
+ }//if
+}//Dblqh::checkReadExecSr()
+
+/* ========================================================================= */
+/* ==== CHECK IF START OF NEW FRAGMENT IS COMPLETED AND WE CAN ======= */
+/* ==== GET THE START GCI ======= */
+/* */
+/* SUBROUTINE SHORT NAME = CTC */
+/* ========================================================================= */
+void Dblqh::checkScanTcCompleted(Signal* signal)
+{
+ tcConnectptr.p->logWriteState = TcConnectionrec::NOT_STARTED;
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ fragptr.p->activeTcCounter = fragptr.p->activeTcCounter - 1;
+ if (fragptr.p->activeTcCounter == 0) {
+ jam();
+ fragptr.p->startGci = cnewestGci + 1;
+ tabptr.i = tcConnectptr.p->tableref;
+ ptrCheckGuard(tabptr, ctabrecFileSize, tablerec);
+ sendCopyActiveConf(signal, tcConnectptr.p->tableref);
+ }//if
+}//Dblqh::checkScanTcCompleted()
+
+/* ==========================================================================
+ * === CHECK IF ALL PARTS OF A SYSTEM RESTART ON A FRAGMENT ARE COMPLETED ===
+ *
+ * SUBROUTINE SHORT NAME = CSC
+ * ========================================================================= */
+void Dblqh::checkSrCompleted(Signal* signal)
+{
+ LcpLocRecordPtr cscLcpLocptr;
+
+ terrorCode = ZOK;
+ ptrGuard(lcpPtr);
+ cscLcpLocptr.i = lcpPtr.p->firstLcpLocAcc;
+CSC_ACC_DOWHILE:
+ ptrCheckGuard(cscLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (cscLcpLocptr.p->lcpLocstate != LcpLocRecord::SR_ACC_COMPLETED) {
+ jam();
+ if (cscLcpLocptr.p->lcpLocstate != LcpLocRecord::SR_ACC_STARTED) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ return;
+ }//if
+ cscLcpLocptr.i = cscLcpLocptr.p->nextLcpLoc;
+ if (cscLcpLocptr.i != RNIL) {
+ jam();
+ goto CSC_ACC_DOWHILE;
+ }//if
+ cscLcpLocptr.i = lcpPtr.p->firstLcpLocTup;
+CSC_TUP_DOWHILE:
+ ptrCheckGuard(cscLcpLocptr, clcpLocrecFileSize, lcpLocRecord);
+ if (cscLcpLocptr.p->lcpLocstate != LcpLocRecord::SR_TUP_COMPLETED) {
+ jam();
+ if (cscLcpLocptr.p->lcpLocstate != LcpLocRecord::SR_TUP_STARTED) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ return;
+ }//if
+ cscLcpLocptr.i = cscLcpLocptr.p->nextLcpLoc;
+ if (cscLcpLocptr.i != RNIL) {
+ jam();
+ goto CSC_TUP_DOWHILE;
+ }//if
+ lcpPtr.p->lcpState = LcpRecord::LCP_SR_COMPLETED;
+}//Dblqh::checkSrCompleted()
+
+/* ------------------------------------------------------------------------- */
+/* ------ CLOSE A FILE DURING EXECUTION OF FRAGMENT LOG ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::closeFile(Signal* signal, LogFileRecordPtr clfLogFilePtr)
+{
+ signal->theData[0] = clfLogFilePtr.p->fileRef;
+ signal->theData[1] = cownref;
+ signal->theData[2] = clfLogFilePtr.i;
+ signal->theData[3] = ZCLOSE_NO_DELETE;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+}//Dblqh::closeFile()
+
+
+/* ---------------------------------------------------------------- */
+/* ---------------- A LOG PAGE HAVE BEEN COMPLETED ---------------- */
+/* */
+/* SUBROUTINE SHORT NAME = CLP */
+// Input Pointers:
+// logFilePtr
+// logPagePtr
+// logPartPtr
+// Defines lfoPtr
+/* ---------------------------------------------------------------- */
+void Dblqh::completedLogPage(Signal* signal, Uint32 clpType)
+{
+ LogPageRecordPtr clpLogPagePtr;
+ LogPageRecordPtr wlpLogPagePtr;
+ UintR twlpNoPages;
+ UintR twlpType;
+
+ if (logFilePtr.p->firstFilledPage == RNIL) {
+ jam();
+ logFilePtr.p->firstFilledPage = logPagePtr.i;
+ } else {
+ jam();
+ clpLogPagePtr.i = logFilePtr.p->lastFilledPage;
+ ptrCheckGuard(clpLogPagePtr, clogPageFileSize, logPageRecord);
+ clpLogPagePtr.p->logPageWord[ZNEXT_PAGE] = logPagePtr.i;
+ }//if
+ logFilePtr.p->lastFilledPage = logPagePtr.i;
+ logPagePtr.p->logPageWord[ZNEXT_PAGE] = RNIL;
+ logFilePtr.p->noLogpagesInBuffer = logFilePtr.p->noLogpagesInBuffer + 1;
+ if (logFilePtr.p->noLogpagesInBuffer != ZMAX_PAGES_WRITTEN) {
+ if (clpType != ZLAST_WRITE_IN_FILE) {
+ if (clpType != ZENFORCE_WRITE) {
+ jam();
+ return;
+ }//if
+ }//if
+ }//if
+ twlpType = clpType;
+/* ------------------------------------------------------------------------- */
+/* ------ WRITE A SET OF LOG PAGES TO DISK ------- */
+/* */
+/* SUBROUTINE SHORT NAME: WLP */
+/* ------------------------------------------------------------------------- */
+ seizeLfo(signal);
+ initLfo(signal);
+ Uint32* dataPtr = &signal->theData[6];
+ twlpNoPages = 0;
+ wlpLogPagePtr.i = logFilePtr.p->firstFilledPage;
+ do {
+ dataPtr[twlpNoPages] = wlpLogPagePtr.i;
+ twlpNoPages++;
+ ptrCheckGuard(wlpLogPagePtr, clogPageFileSize, logPageRecord);
+
+ // Calculate checksum for page
+ wlpLogPagePtr.p->logPageWord[ZPOS_CHECKSUM] = calcPageCheckSum(wlpLogPagePtr);
+ wlpLogPagePtr.i = wlpLogPagePtr.p->logPageWord[ZNEXT_PAGE];
+ } while (wlpLogPagePtr.i != RNIL);
+ ndbrequire(twlpNoPages < 9);
+ dataPtr[twlpNoPages] = logFilePtr.p->filePosition;
+/* -------------------------------------------------- */
+/* SET TIMER ON THIS LOG PART TO SIGNIFY THAT A */
+/* LOG RECORD HAS BEEN SENT AT THIS TIME. */
+/* -------------------------------------------------- */
+ logPartPtr.p->logPartTimer = logPartPtr.p->logTimer;
+ signal->theData[0] = logFilePtr.p->fileRef;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lfoPtr.i;
+ logFilePtr.p->logFilePagesToDiskWithoutSynch += twlpNoPages;
+ if (twlpType == ZLAST_WRITE_IN_FILE) {
+ jam();
+ logFilePtr.p->logFilePagesToDiskWithoutSynch = 0;
+ signal->theData[3] = ZLIST_OF_MEM_PAGES_SYNCH;
+ } else if (logFilePtr.p->logFilePagesToDiskWithoutSynch >
+ MAX_REDO_PAGES_WITHOUT_SYNCH) {
+ jam();
+ logFilePtr.p->logFilePagesToDiskWithoutSynch = 0;
+ signal->theData[3] = ZLIST_OF_MEM_PAGES_SYNCH;
+ } else {
+ jam();
+ signal->theData[3] = ZLIST_OF_MEM_PAGES;
+ }//if
+ signal->theData[4] = ZVAR_NO_LOG_PAGE_WORD;
+ signal->theData[5] = twlpNoPages;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 15, JBA);
+ if (twlpType == ZNORMAL) {
+ jam();
+ lfoPtr.p->lfoState = LogFileOperationRecord::ACTIVE_WRITE_LOG;
+ } else if (twlpType == ZLAST_WRITE_IN_FILE) {
+ jam();
+ lfoPtr.p->lfoState = LogFileOperationRecord::LAST_WRITE_IN_FILE;
+ } else {
+ ndbrequire(twlpType == ZENFORCE_WRITE);
+ jam();
+ lfoPtr.p->lfoState = LogFileOperationRecord::ACTIVE_WRITE_LOG;
+ }//if
+ /* ----------------------------------------------------------------------- */
+ /* ------ MOVE PAGES FROM LOG FILE TO LFO RECORD ------- */
+ /* */
+ /* ----------------------------------------------------------------------- */
+ /* -------------------------------------------------- */
+ /* MOVE PAGES TO LFO RECORD AND REMOVE THEM */
+ /* FROM LOG FILE RECORD. */
+ /* -------------------------------------------------- */
+ lfoPtr.p->firstLfoPage = logFilePtr.p->firstFilledPage;
+ logFilePtr.p->firstFilledPage = RNIL;
+ logFilePtr.p->lastFilledPage = RNIL;
+ logFilePtr.p->noLogpagesInBuffer = 0;
+
+ lfoPtr.p->noPagesRw = twlpNoPages;
+ lfoPtr.p->lfoPageNo = logFilePtr.p->filePosition;
+ lfoPtr.p->lfoWordWritten = ZPAGE_SIZE - 1;
+ logFilePtr.p->filePosition += twlpNoPages;
+}//Dblqh::completedLogPage()
+
+/* ---------------------------------------------------------------- */
+/* ---------------- DELETE FRAGMENT RECORD ------------------------ */
+/* */
+/* SUBROUTINE SHORT NAME = DFR */
+/* ---------------------------------------------------------------- */
+void Dblqh::deleteFragrec(Uint32 fragId)
+{
+ Uint32 indexFound;
+ fragptr.i = RNIL;
+ for (Uint32 i = (NO_OF_FRAG_PER_NODE - 1); (Uint32)~i; i--) {
+ jam();
+ if (tabptr.p->fragid[i] == fragId) {
+ fragptr.i = tabptr.p->fragrec[i];
+ indexFound = i;
+ break;
+ }//if
+ }//for
+ if (fragptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ tabptr.p->fragid[indexFound] = ZNIL;
+ tabptr.p->fragrec[indexFound] = RNIL;
+ releaseFragrec();
+ }//if
+}//Dblqh::deleteFragrec()
+
+/* ------------------------------------------------------------------------- */
+/* ------- FIND LOG FILE RECORD GIVEN FILE NUMBER ------- */
+/* */
+/* INPUT: TFLF_FILE_NO THE FILE NUMBER */
+/* FLF_LOG_PART_PTR THE LOG PART RECORD */
+/* OUTPUT: FLF_LOG_FILE_PTR THE FOUND LOG FILE RECORD */
+/* SUBROUTINE SHORT NAME = FLF */
+/* ------------------------------------------------------------------------- */
+void Dblqh::findLogfile(Signal* signal,
+ Uint32 fileNo,
+ LogPartRecordPtr flfLogPartPtr,
+ LogFileRecordPtr* parLogFilePtr)
+{
+ LogFileRecordPtr locLogFilePtr;
+ locLogFilePtr.i = flfLogPartPtr.p->firstLogfile;
+ Uint32 loopCount = 0;
+ while (true) {
+ ptrCheckGuard(locLogFilePtr, clogFileFileSize, logFileRecord);
+ if (locLogFilePtr.p->fileNo == fileNo) {
+ jam();
+ ndbrequire(loopCount == fileNo);
+ parLogFilePtr->i = locLogFilePtr.i;
+ parLogFilePtr->p = locLogFilePtr.p;
+ return;
+ }//if
+ locLogFilePtr.i = locLogFilePtr.p->nextLogFile;
+ loopCount++;
+ ndbrequire(loopCount < flfLogPartPtr.p->noLogFiles);
+ }//while
+}//Dblqh::findLogfile()
+
+/* ------------------------------------------------------------------------- */
+/* ------ FIND PAGE REFERENCE IN MEMORY BUFFER AT LOG EXECUTION ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::findPageRef(Signal* signal, CommitLogRecord* commitLogRecord)
+{
+ UintR tfprIndex;
+
+ logPagePtr.i = RNIL;
+ if (ERROR_INSERTED(5020)) {
+ // Force system to read page from disk
+ return;
+ }
+ pageRefPtr.i = logPartPtr.p->lastPageRef;
+ do {
+ ptrCheckGuard(pageRefPtr, cpageRefFileSize, pageRefRecord);
+ if (commitLogRecord->fileNo == pageRefPtr.p->prFileNo) {
+ if (commitLogRecord->startPageNo >= pageRefPtr.p->prPageNo) {
+ if (commitLogRecord->startPageNo < (Uint16) (pageRefPtr.p->prPageNo + 8)) {
+ jam();
+ tfprIndex = commitLogRecord->startPageNo - pageRefPtr.p->prPageNo;
+ logPagePtr.i = pageRefPtr.p->pageRef[tfprIndex];
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ return;
+ }//if
+ }//if
+ }//if
+ pageRefPtr.i = pageRefPtr.p->prPrev;
+ } while (pageRefPtr.i != RNIL);
+}//Dblqh::findPageRef()
+
+/* ------------------------------------------------------------------------- */
+/* ------ GET FIRST OPERATION QUEUED FOR LOGGING ------- */
+/* */
+/* SUBROUTINE SHORT NAME = GFL */
+/* ------------------------------------------------------------------------- */
+void Dblqh::getFirstInLogQueue(Signal* signal)
+{
+ TcConnectionrecPtr gflTcConnectptr;
+/* -------------------------------------------------- */
+/* GET THE FIRST FROM THE LOG QUEUE AND REMOVE */
+/* IT FROM THE QUEUE. */
+/* -------------------------------------------------- */
+ gflTcConnectptr.i = logPartPtr.p->firstLogQueue;
+ ptrCheckGuard(gflTcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ logPartPtr.p->firstLogQueue = gflTcConnectptr.p->nextTcLogQueue;
+ if (logPartPtr.p->firstLogQueue == RNIL) {
+ jam();
+ logPartPtr.p->lastLogQueue = RNIL;
+ }//if
+}//Dblqh::getFirstInLogQueue()
+
+/* ---------------------------------------------------------------- */
+/* ---------------- GET FRAGMENT RECORD --------------------------- */
+/* INPUT: TFRAGID FRAGMENT ID LOOKING FOR */
+/* TABPTR TABLE ID */
+/* SUBROUTINE SHORT NAME = GFR */
+/* ---------------------------------------------------------------- */
+bool Dblqh::getFragmentrec(Signal* signal, Uint32 fragId)
+{
+ for (Uint32 i = (NO_OF_FRAG_PER_NODE - 1); (UintR)~i; i--) {
+ jam();
+ if (tabptr.p->fragid[i] == fragId) {
+ fragptr.i = tabptr.p->fragrec[i];
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ return true;
+ }//if
+ }//for
+ return false;
+}//Dblqh::getFragmentrec()
+
+/* ========================================================================= */
+/* ====== INITIATE FRAGMENT RECORD ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseAddfragrec(Signal* signal)
+{
+ if (caddfragrecFileSize != 0) {
+ for (addfragptr.i = 0; addfragptr.i < caddfragrecFileSize; addfragptr.i++) {
+ ptrAss(addfragptr, addFragRecord);
+ addfragptr.p->addfragStatus = AddFragRecord::FREE;
+ addfragptr.p->nextAddfragrec = addfragptr.i + 1;
+ }//for
+ addfragptr.i = caddfragrecFileSize - 1;
+ ptrAss(addfragptr, addFragRecord);
+ addfragptr.p->nextAddfragrec = RNIL;
+ cfirstfreeAddfragrec = 0;
+ } else {
+ jam();
+ cfirstfreeAddfragrec = RNIL;
+ }//if
+}//Dblqh::initialiseAddfragrec()
+
+/* ========================================================================= */
+/* ====== INITIATE ATTRIBUTE IN AND OUT DATA BUFFER ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseAttrbuf(Signal* signal)
+{
+ if (cattrinbufFileSize != 0) {
+ for (attrinbufptr.i = 0;
+ attrinbufptr.i < cattrinbufFileSize;
+ attrinbufptr.i++) {
+ ptrAss(attrinbufptr, attrbuf);
+ attrinbufptr.p->attrbuf[ZINBUF_NEXT] = attrinbufptr.i + 1;
+ }//for
+ /* NEXT ATTRINBUF */
+ attrinbufptr.i = cattrinbufFileSize - 1;
+ ptrAss(attrinbufptr, attrbuf);
+ attrinbufptr.p->attrbuf[ZINBUF_NEXT] = RNIL; /* NEXT ATTRINBUF */
+ cfirstfreeAttrinbuf = 0;
+ } else {
+ jam();
+ cfirstfreeAttrinbuf = RNIL;
+ }//if
+}//Dblqh::initialiseAttrbuf()
+
+/* ========================================================================= */
+/* ====== INITIATE DATA BUFFER ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseDatabuf(Signal* signal)
+{
+ if (cdatabufFileSize != 0) {
+ for (databufptr.i = 0; databufptr.i < cdatabufFileSize; databufptr.i++) {
+ ptrAss(databufptr, databuf);
+ databufptr.p->nextDatabuf = databufptr.i + 1;
+ }//for
+ databufptr.i = cdatabufFileSize - 1;
+ ptrAss(databufptr, databuf);
+ databufptr.p->nextDatabuf = RNIL;
+ cfirstfreeDatabuf = 0;
+ } else {
+ jam();
+ cfirstfreeDatabuf = RNIL;
+ }//if
+}//Dblqh::initialiseDatabuf()
+
+/* ========================================================================= */
+/* ====== INITIATE FRAGMENT RECORD ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseFragrec(Signal* signal)
+{
+ if (cfragrecFileSize != 0) {
+ for (fragptr.i = 0; fragptr.i < cfragrecFileSize; fragptr.i++) {
+ ptrAss(fragptr, fragrecord);
+ fragptr.p->fragStatus = Fragrecord::FREE;
+ fragptr.p->fragActiveStatus = ZFALSE;
+ fragptr.p->execSrStatus = Fragrecord::IDLE;
+ fragptr.p->srStatus = Fragrecord::SS_IDLE;
+ fragptr.p->nextFrag = fragptr.i + 1;
+ }//for
+ fragptr.i = cfragrecFileSize - 1;
+ ptrAss(fragptr, fragrecord);
+ fragptr.p->nextFrag = RNIL;
+ cfirstfreeFragrec = 0;
+ } else {
+ jam();
+ cfirstfreeFragrec = RNIL;
+ }//if
+}//Dblqh::initialiseFragrec()
+
+/* ========================================================================= */
+/* ====== INITIATE FRAGMENT RECORD ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseGcprec(Signal* signal)
+{
+ UintR tigpIndex;
+
+ if (cgcprecFileSize != 0) {
+ for (gcpPtr.i = 0; gcpPtr.i < cgcprecFileSize; gcpPtr.i++) {
+ ptrAss(gcpPtr, gcpRecord);
+ for (tigpIndex = 0; tigpIndex <= 3; tigpIndex++) {
+ gcpPtr.p->gcpLogPartState[tigpIndex] = ZIDLE;
+ gcpPtr.p->gcpSyncReady[tigpIndex] = ZFALSE;
+ }//for
+ }//for
+ }//if
+}//Dblqh::initialiseGcprec()
+
+/* ========================================================================= */
+/* ====== INITIATE LCP RECORD ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseLcpRec(Signal* signal)
+{
+ if (clcpFileSize != 0) {
+ for (lcpPtr.i = 0; lcpPtr.i < clcpFileSize; lcpPtr.i++) {
+ ptrAss(lcpPtr, lcpRecord);
+ lcpPtr.p->lcpState = LcpRecord::LCP_IDLE;
+ lcpPtr.p->lcpQueued = false;
+ lcpPtr.p->firstLcpLocAcc = RNIL;
+ lcpPtr.p->firstLcpLocTup = RNIL;
+ lcpPtr.p->reportEmpty = false;
+ lcpPtr.p->lastFragmentFlag = false;
+ }//for
+ }//if
+}//Dblqh::initialiseLcpRec()
+
+/* ========================================================================= */
+/* ====== INITIATE LCP LOCAL RECORD ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseLcpLocrec(Signal* signal)
+{
+ if (clcpLocrecFileSize != 0) {
+ for (lcpLocptr.i = 0; lcpLocptr.i < clcpLocrecFileSize; lcpLocptr.i++) {
+ ptrAss(lcpLocptr, lcpLocRecord);
+ lcpLocptr.p->nextLcpLoc = lcpLocptr.i + 1;
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::IDLE;
+ lcpLocptr.p->masterLcpRec = RNIL;
+ lcpLocptr.p->waitingBlock = LcpLocRecord::NONE;
+ }//for
+ lcpLocptr.i = clcpLocrecFileSize - 1;
+ ptrAss(lcpLocptr, lcpLocRecord);
+ lcpLocptr.p->nextLcpLoc = RNIL;
+ cfirstfreeLcpLoc = 0;
+ } else {
+ jam();
+ cfirstfreeLcpLoc = RNIL;
+ }//if
+}//Dblqh::initialiseLcpLocrec()
+
+/* ========================================================================= */
+/* ====== INITIATE LOG FILE OPERATION RECORD ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseLfo(Signal* signal)
+{
+ if (clfoFileSize != 0) {
+ for (lfoPtr.i = 0; lfoPtr.i < clfoFileSize; lfoPtr.i++) {
+ ptrAss(lfoPtr, logFileOperationRecord);
+ lfoPtr.p->lfoState = LogFileOperationRecord::IDLE;
+ lfoPtr.p->lfoTimer = 0;
+ lfoPtr.p->nextLfo = lfoPtr.i + 1;
+ }//for
+ lfoPtr.i = clfoFileSize - 1;
+ ptrAss(lfoPtr, logFileOperationRecord);
+ lfoPtr.p->nextLfo = RNIL;
+ cfirstfreeLfo = 0;
+ } else {
+ jam();
+ cfirstfreeLfo = RNIL;
+ }//if
+}//Dblqh::initialiseLfo()
+
+/* ========================================================================= */
+/* ====== INITIATE LOG FILE RECORD ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseLogFile(Signal* signal)
+{
+ if (clogFileFileSize != 0) {
+ for (logFilePtr.i = 0; logFilePtr.i < clogFileFileSize; logFilePtr.i++) {
+ ptrAss(logFilePtr, logFileRecord);
+ logFilePtr.p->nextLogFile = logFilePtr.i + 1;
+ logFilePtr.p->logFileStatus = LogFileRecord::LFS_IDLE;
+ }//for
+ logFilePtr.i = clogFileFileSize - 1;
+ ptrAss(logFilePtr, logFileRecord);
+ logFilePtr.p->nextLogFile = RNIL;
+ cfirstfreeLogFile = 0;
+ } else {
+ jam();
+ cfirstfreeLogFile = RNIL;
+ }//if
+}//Dblqh::initialiseLogFile()
+
+/* ========================================================================= */
+/* ====== INITIATE LOG PAGES ======= */
+/* */
+/* ========================================================================= */
+void Dblqh::initialiseLogPage(Signal* signal)
+{
+ if (clogPageFileSize != 0) {
+ for (logPagePtr.i = 0; logPagePtr.i < clogPageFileSize; logPagePtr.i++) {
+ ptrAss(logPagePtr, logPageRecord);
+ logPagePtr.p->logPageWord[ZNEXT_PAGE] = logPagePtr.i + 1;
+ }//for
+ logPagePtr.i = clogPageFileSize - 1;
+ ptrAss(logPagePtr, logPageRecord);
+ logPagePtr.p->logPageWord[ZNEXT_PAGE] = RNIL;
+ cfirstfreeLogPage = 0;
+ } else {
+ jam();
+ cfirstfreeLogPage = RNIL;
+ }//if
+ cnoOfLogPages = clogPageFileSize;
+}//Dblqh::initialiseLogPage()
+
+/* =========================================================================
+ * ====== INITIATE LOG PART RECORD =======
+ *
+ * ========================================================================= */
+void Dblqh::initialiseLogPart(Signal* signal)
+{
+ for (logPartPtr.i = 0; logPartPtr.i <= 3; logPartPtr.i++) {
+ ptrAss(logPartPtr, logPartRecord);
+ logPartPtr.p->waitWriteGciLog = LogPartRecord::WWGL_FALSE;
+ logPartPtr.p->LogLqhKeyReqSent = ZFALSE;
+ logPartPtr.p->logPartNewestCompletedGCI = (UintR)-1;
+ }//for
+}//Dblqh::initialiseLogPart()
+
+void Dblqh::initialisePageRef(Signal* signal)
+{
+ if (cpageRefFileSize != 0) {
+ for (pageRefPtr.i = 0;
+ pageRefPtr.i < cpageRefFileSize;
+ pageRefPtr.i++) {
+ ptrAss(pageRefPtr, pageRefRecord);
+ pageRefPtr.p->prNext = pageRefPtr.i + 1;
+ }//for
+ pageRefPtr.i = cpageRefFileSize - 1;
+ ptrAss(pageRefPtr, pageRefRecord);
+ pageRefPtr.p->prNext = RNIL;
+ cfirstfreePageRef = 0;
+ } else {
+ jam();
+ cfirstfreePageRef = RNIL;
+ }//if
+}//Dblqh::initialisePageRef()
+
+/* ==========================================================================
+ * ======= INITIATE RECORDS =======
+ *
+ * TAKES CARE OF INITIATION OF ALL RECORDS IN THIS BLOCK.
+ * ========================================================================= */
+void Dblqh::initialiseRecordsLab(Signal* signal, Uint32 data)
+{
+ switch (data) {
+ case 0:
+ jam();
+ for (Uint32 i = 0; i < MAX_NDB_NODES; i++) {
+ cnodeSrState[i] = ZSTART_SR;
+ cnodeExecSrState[i] = ZSTART_SR;
+ }//for
+ for (Uint32 i = 0; i < 1024; i++) {
+ ctransidHash[i] = RNIL;
+ }//for
+ for (Uint32 i = 0; i < 4; i++) {
+ cactiveCopy[i] = RNIL;
+ }//for
+ cnoActiveCopy = 0;
+ cCounterAccCommitBlocked = 0;
+ cCounterTupCommitBlocked = 0;
+ caccCommitBlocked = false;
+ ctupCommitBlocked = false;
+ cCommitBlocked = false;
+ ccurrentGcprec = RNIL;
+ caddNodeState = ZFALSE;
+ cstartRecReq = ZFALSE;
+ cnewestGci = (UintR)-1;
+ cnewestCompletedGci = (UintR)-1;
+ crestartOldestGci = 0;
+ crestartNewestGci = 0;
+ cfirstWaitFragSr = RNIL;
+ cfirstCompletedFragSr = RNIL;
+ csrPhaseStarted = ZSR_NO_PHASE_STARTED;
+ csrPhasesCompleted = 0;
+ cmasterDihBlockref = 0;
+ cnoFragmentsExecSr = 0;
+ clcpCompletedState = LCP_IDLE;
+ csrExecUndoLogState = EULS_IDLE;
+ c_lcpId = 0;
+ cnoOfFragsCheckpointed = 0;
+ sendInitialiseRecords(signal, data);
+ break;
+ case 1:
+ jam();
+ initialiseAddfragrec(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 2:
+ jam();
+ initialiseAttrbuf(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 3:
+ jam();
+ initialiseDatabuf(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 4:
+ jam();
+ initialiseFragrec(signal);
+ sendInitialiseRecords(signal,data);
+ break;
+ case 5:
+ jam();
+ initialiseGcprec(signal);
+ initialiseLcpRec(signal);
+ initialiseLcpLocrec(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 6:
+ jam();
+ initialiseLogPage(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 7:
+ jam();
+ initialiseLfo(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 8:
+ jam();
+ initialiseLogFile(signal);
+ initialiseLogPart(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 9:
+ jam();
+ initialisePageRef(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 10:
+ jam();
+ initialiseScanrec(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 11:
+ jam();
+ initialiseTabrec(signal);
+ sendInitialiseRecords(signal, data);
+ break;
+ case 12:
+ jam();
+ initialiseTcNodeFailRec(signal);
+ initialiseTcrec(signal);
+ returnInitialiseRecordsLab(signal);
+ return;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dblqh::initialiseRecordsLab()
+
+/* --------------------------------------------------------------------------
+ * SEND REAL-TIME BREAK SIGNAL DURING INITIALISATION IN SYSTEM RESTART.
+ * ------------------------------------------------------------------------- */
+void Dblqh::sendInitialiseRecords(Signal* signal, Uint32 data)
+{
+ signal->theData[0] = ZINITIALISE_RECORDS;
+ signal->theData[1] = data + 1;
+ signal->theData[2] = 0;
+ sendSignal(DBLQH_REF, GSN_CONTINUEB, signal, 3, JBB);
+}//Dblqh::sendInitialiseRecords()
+
+/* ==========================================================================
+ * ======= INITIATE TC CONNECTION RECORD =======
+ *
+ * ========================================================================= */
+void Dblqh::initialiseScanrec(Signal* signal)
+{
+ ndbrequire(cscanrecFileSize > 1);
+ for (scanptr.i = 0; scanptr.i < cscanrecFileSize; scanptr.i++) {
+ ptrAss(scanptr, scanRecord);
+ scanptr.p->nextScanrec = scanptr.i + 1;
+ scanptr.p->scanType = ScanRecord::ST_IDLE;
+ scanptr.p->scanState = ScanRecord::SCAN_FREE;
+ scanptr.p->scanTcWaiting = ZFALSE;
+ }//for
+ scanptr.i = cscanrecFileSize - 1;
+ ptrAss(scanptr, scanRecord);
+ scanptr.p->nextScanrec = RNIL;
+ cfirstfreeScanrec = 0;
+ cscanNoFreeRec = cscanrecFileSize;
+}//Dblqh::initialiseScanrec()
+
+/* ==========================================================================
+ * ======= INITIATE TABLE RECORD =======
+ *
+ * ========================================================================= */
+void Dblqh::initialiseTabrec(Signal* signal)
+{
+ if (ctabrecFileSize != 0) {
+ for (tabptr.i = 0; tabptr.i < ctabrecFileSize; tabptr.i++) {
+ ptrAss(tabptr, tablerec);
+ tabptr.p->tableStatus = Tablerec::NOT_DEFINED;
+ tabptr.p->usageCount = 0;
+ for (Uint32 i = 0; i <= (NO_OF_FRAG_PER_NODE - 1); i++) {
+ tabptr.p->fragid[i] = ZNIL;
+ tabptr.p->fragrec[i] = RNIL;
+ }//for
+ }//for
+ }//if
+}//Dblqh::initialiseTabrec()
+
+/* ==========================================================================
+ * ======= INITIATE TC CONNECTION RECORD =======
+ *
+ * ========================================================================= */
+void Dblqh::initialiseTcrec(Signal* signal)
+{
+ if (ctcConnectrecFileSize != 0) {
+ for (tcConnectptr.i = 0;
+ tcConnectptr.i < ctcConnectrecFileSize;
+ tcConnectptr.i++) {
+ ptrAss(tcConnectptr, tcConnectionrec);
+ tcConnectptr.p->transactionState = TcConnectionrec::TC_NOT_CONNECTED;
+ tcConnectptr.p->tcScanRec = RNIL;
+ tcConnectptr.p->logWriteState = TcConnectionrec::NOT_STARTED;
+ tcConnectptr.p->firstAttrinbuf = RNIL;
+ tcConnectptr.p->lastAttrinbuf = RNIL;
+ tcConnectptr.p->firstTupkeybuf = RNIL;
+ tcConnectptr.p->lastTupkeybuf = RNIL;
+ tcConnectptr.p->tcTimer = 0;
+ tcConnectptr.p->nextTcConnectrec = tcConnectptr.i + 1;
+ }//for
+ tcConnectptr.i = ctcConnectrecFileSize - 1;
+ ptrAss(tcConnectptr, tcConnectionrec);
+ tcConnectptr.p->nextTcConnectrec = RNIL;
+ cfirstfreeTcConrec = 0;
+ } else {
+ jam();
+ cfirstfreeTcConrec = RNIL;
+ }//if
+}//Dblqh::initialiseTcrec()
+
+/* ==========================================================================
+ * ======= INITIATE TC CONNECTION RECORD =======
+ *
+ * ========================================================================= */
+void Dblqh::initialiseTcNodeFailRec(Signal* signal)
+{
+ if (ctcNodeFailrecFileSize != 0) {
+ for (tcNodeFailptr.i = 0;
+ tcNodeFailptr.i < ctcNodeFailrecFileSize;
+ tcNodeFailptr.i++) {
+ ptrAss(tcNodeFailptr, tcNodeFailRecord);
+ tcNodeFailptr.p->tcFailStatus = TcNodeFailRecord::TC_STATE_FALSE;
+ }//for
+ }//if
+}//Dblqh::initialiseTcNodeFailRec()
+
+/* ==========================================================================
+ * ======= INITIATE FRAGMENT RECORD =======
+ *
+ * SUBROUTINE SHORT NAME = IF
+ * ========================================================================= */
+void Dblqh::initFragrec(Signal* signal,
+ Uint32 tableId,
+ Uint32 fragId,
+ Uint32 copyType)
+{
+ fragptr.p->accBlockref = caccBlockref;
+ fragptr.p->accBlockedList = RNIL;
+ fragptr.p->activeList = RNIL;
+ fragptr.p->firstWaitQueue = RNIL;
+ fragptr.p->lastWaitQueue = RNIL;
+ fragptr.p->fragStatus = Fragrecord::DEFINED;
+ fragptr.p->fragCopy = copyType;
+ fragptr.p->tupBlockref = ctupBlockref;
+ fragptr.p->tuxBlockref = ctuxBlockref;
+ fragptr.p->lcpRef = RNIL;
+ fragptr.p->logFlag = Fragrecord::STATE_TRUE;
+ fragptr.p->lcpFlag = Fragrecord::LCP_STATE_TRUE;
+ for (Uint32 i = 0; i < MAX_LCP_STORED; i++) {
+ fragptr.p->lcpId[i] = 0;
+ }//for
+ for (Uint32 i = 0; i < NR_MaxRangeScanNo; i++) {
+ fragptr.p->fragScanRec[i] = ZNIL;
+ }//for
+ fragptr.p->maxGciCompletedInLcp = 0;
+ fragptr.p->maxGciInLcp = 0;
+ fragptr.p->noActiveScan = 0;
+ fragptr.p->copyFragState = ZIDLE;
+ fragptr.p->nextFrag = RNIL;
+ fragptr.p->newestGci = cnewestGci;
+ fragptr.p->nextLcp = 0;
+ fragptr.p->tabRef = tableId;
+ fragptr.p->fragId = fragId;
+ fragptr.p->srStatus = Fragrecord::SS_IDLE;
+ fragptr.p->execSrStatus = Fragrecord::IDLE;
+ fragptr.p->execSrNoReplicas = 0;
+ fragptr.p->fragDistributionKey = 0;
+ fragptr.p->activeTcCounter = 0;
+}//Dblqh::initFragrec()
+
+/* ==========================================================================
+ * ======= INITIATE FRAGMENT RECORD FOR SYSTEM RESTART =======
+ *
+ * SUBROUTINE SHORT NAME = IFS
+ * ========================================================================= */
+void Dblqh::initFragrecSr(Signal* signal)
+{
+ const StartFragReq * const startFragReq = (StartFragReq *)&signal->theData[0];
+ Uint32 lcpNo = startFragReq->lcpNo;
+ Uint32 noOfLogNodes = startFragReq->noOfLogNodes;
+ ndbrequire(noOfLogNodes <= 4);
+ fragptr.p->fragStatus = Fragrecord::CRASH_RECOVERING;
+ fragptr.p->srBlockref = startFragReq->userRef;
+ fragptr.p->srUserptr = startFragReq->userPtr;
+ fragptr.p->srChkpnr = lcpNo;
+ if (lcpNo == (MAX_LCP_STORED - 1)) {
+ jam();
+ fragptr.p->lcpId[lcpNo] = startFragReq->lcpId;
+ fragptr.p->nextLcp = 0;
+ } else if (lcpNo < (MAX_LCP_STORED - 1)) {
+ jam();
+ fragptr.p->lcpId[lcpNo] = startFragReq->lcpId;
+ fragptr.p->nextLcp = lcpNo + 1;
+ } else {
+ ndbrequire(lcpNo == ZNIL);
+ jam();
+ fragptr.p->nextLcp = 0;
+ }//if
+ fragptr.p->srNoLognodes = noOfLogNodes;
+ fragptr.p->logFlag = Fragrecord::STATE_FALSE;
+ fragptr.p->srStatus = Fragrecord::SS_IDLE;
+ if (noOfLogNodes > 0) {
+ jam();
+ for (Uint32 i = 0; i < noOfLogNodes; i++) {
+ jam();
+ fragptr.p->srStartGci[i] = startFragReq->startGci[i];
+ fragptr.p->srLastGci[i] = startFragReq->lastGci[i];
+ fragptr.p->srLqhLognode[i] = startFragReq->lqhLogNode[i];
+ }//for
+ fragptr.p->newestGci = startFragReq->lastGci[noOfLogNodes - 1];
+ } else {
+ fragptr.p->newestGci = cnewestGci;
+ }//if
+}//Dblqh::initFragrecSr()
+
+/* ==========================================================================
+ * ======= INITIATE INFORMATION ABOUT GLOBAL CHECKPOINTS =======
+ * IN LOG FILE RECORDS
+ *
+ * INPUT: LOG_FILE_PTR CURRENT LOG FILE
+ * TNO_FD_DESCRIPTORS THE NUMBER OF FILE DESCRIPTORS
+ * TO READ FROM THE LOG PAGE
+ * LOG_PAGE_PTR PAGE ZERO IN LOG FILE
+ * SUBROUTINE SHORT NAME = IGL
+ * ========================================================================= */
+void Dblqh::initGciInLogFileRec(Signal* signal, Uint32 noFdDescriptors)
+{
+ LogFileRecordPtr iglLogFilePtr;
+ UintR tiglLoop;
+ UintR tiglIndex;
+
+ tiglLoop = 0;
+ iglLogFilePtr.i = logFilePtr.i;
+ iglLogFilePtr.p = logFilePtr.p;
+IGL_LOOP:
+ for (tiglIndex = 0; tiglIndex <= ZNO_MBYTES_IN_FILE - 1; tiglIndex++) {
+ arrGuard(((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (tiglLoop * ZFD_PART_SIZE)) + tiglIndex, ZPAGE_SIZE);
+ iglLogFilePtr.p->logMaxGciCompleted[tiglIndex] =
+ logPagePtr.p->logPageWord[((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (tiglLoop * ZFD_PART_SIZE)) + tiglIndex];
+ arrGuard((((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) + ZNO_MBYTES_IN_FILE) +
+ (tiglLoop * ZFD_PART_SIZE)) + tiglIndex, ZPAGE_SIZE);
+ iglLogFilePtr.p->logMaxGciStarted[tiglIndex] =
+ logPagePtr.p->logPageWord[(((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ ZNO_MBYTES_IN_FILE) +
+ (tiglLoop * ZFD_PART_SIZE)) + tiglIndex];
+ arrGuard((((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (2 * ZNO_MBYTES_IN_FILE)) + (tiglLoop * ZFD_PART_SIZE)) +
+ tiglIndex, ZPAGE_SIZE);
+ iglLogFilePtr.p->logLastPrepRef[tiglIndex] =
+ logPagePtr.p->logPageWord[(((ZPAGE_HEADER_SIZE + ZFD_HEADER_SIZE) +
+ (2 * ZNO_MBYTES_IN_FILE)) +
+ (tiglLoop * ZFD_PART_SIZE)) + tiglIndex];
+ }//for
+ tiglLoop = tiglLoop + 1;
+ if (tiglLoop < noFdDescriptors) {
+ jam();
+ iglLogFilePtr.i = iglLogFilePtr.p->prevLogFile;
+ ptrCheckGuard(iglLogFilePtr, clogFileFileSize, logFileRecord);
+ goto IGL_LOOP;
+ }//if
+}//Dblqh::initGciInLogFileRec()
+
+/* ==========================================================================
+ * ======= INITIATE LCP RECORD WHEN USED FOR SYSTEM RESTART =======
+ *
+ * SUBROUTINE SHORT NAME = ILS
+ * ========================================================================= */
+void Dblqh::initLcpSr(Signal* signal,
+ Uint32 lcpNo,
+ Uint32 lcpId,
+ Uint32 tableId,
+ Uint32 fragId,
+ Uint32 fragPtr)
+{
+ lcpPtr.p->lcpQueued = false;
+ lcpPtr.p->currentFragment.fragPtrI = fragPtr;
+ lcpPtr.p->currentFragment.lcpFragOrd.lcpNo = lcpNo;
+ lcpPtr.p->currentFragment.lcpFragOrd.lcpId = lcpId;
+ lcpPtr.p->currentFragment.lcpFragOrd.tableId = tableId;
+ lcpPtr.p->currentFragment.lcpFragOrd.fragmentId = fragId;
+ lcpPtr.p->lcpState = LcpRecord::LCP_SR_WAIT_FRAGID;
+ lcpPtr.p->firstLcpLocAcc = RNIL;
+ lcpPtr.p->firstLcpLocTup = RNIL;
+ lcpPtr.p->lcpAccptr = RNIL;
+}//Dblqh::initLcpSr()
+
+/* ==========================================================================
+ * ======= INITIATE LOG PART =======
+ *
+ * ========================================================================= */
+void Dblqh::initLogpart(Signal* signal)
+{
+ logPartPtr.p->execSrLogPage = RNIL;
+ logPartPtr.p->execSrLogPageIndex = ZNIL;
+ logPartPtr.p->execSrExecuteIndex = 0;
+ logPartPtr.p->noLogFiles = cnoLogFiles;
+ logPartPtr.p->logLap = 0;
+ logPartPtr.p->logTailFileNo = 0;
+ logPartPtr.p->logTailMbyte = 0;
+ logPartPtr.p->lastMbyte = ZNIL;
+ logPartPtr.p->logPartState = LogPartRecord::SR_FIRST_PHASE;
+ logPartPtr.p->logExecState = LogPartRecord::LES_IDLE;
+ logPartPtr.p->firstLogTcrec = RNIL;
+ logPartPtr.p->lastLogTcrec = RNIL;
+ logPartPtr.p->firstLogQueue = RNIL;
+ logPartPtr.p->lastLogQueue = RNIL;
+ logPartPtr.p->gcprec = RNIL;
+ logPartPtr.p->firstPageRef = RNIL;
+ logPartPtr.p->lastPageRef = RNIL;
+ logPartPtr.p->headFileNo = ZNIL;
+ logPartPtr.p->headPageNo = ZNIL;
+ logPartPtr.p->headPageIndex = ZNIL;
+}//Dblqh::initLogpart()
+
+/* ==========================================================================
+ * ======= INITIATE LOG POINTERS =======
+ *
+ * ========================================================================= */
+void Dblqh::initLogPointers(Signal* signal)
+{
+ logPartPtr.i = tcConnectptr.p->hashValue & 3;
+ ptrCheckGuard(logPartPtr, clogPartFileSize, logPartRecord);
+ logFilePtr.i = logPartPtr.p->currentLogfile;
+ ptrCheckGuard(logFilePtr, clogFileFileSize, logFileRecord);
+ logPagePtr.i = logFilePtr.p->currentLogpage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+}//Dblqh::initLogPointers()
+
+/* ------------------------------------------------------------------------- */
+/* ------- INIT REQUEST INFO BEFORE EXECUTING A LOG RECORD ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::initReqinfoExecSr(Signal* signal)
+{
+ UintR Treqinfo = 0;
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ LqhKeyReq::setKeyLen(Treqinfo, regTcPtr->primKeyLen);
+/* ------------------------------------------------------------------------- */
+/* NUMBER OF BACKUPS AND STANDBYS ARE ZERO AND NEED NOT BE SET. */
+/* REPLICA TYPE IS CLEARED BY SEND_LQHKEYREQ. */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* SET LAST REPLICA NUMBER TO ZERO (BIT 10-11) */
+/* ------------------------------------------------------------------------- */
+/* ------------------------------------------------------------------------- */
+/* SET DIRTY FLAG */
+/* ------------------------------------------------------------------------- */
+ LqhKeyReq::setDirtyFlag(Treqinfo, 1);
+/* ------------------------------------------------------------------------- */
+/* SET SIMPLE TRANSACTION */
+/* ------------------------------------------------------------------------- */
+ LqhKeyReq::setSimpleFlag(Treqinfo, 1);
+/* ------------------------------------------------------------------------- */
+/* SET OPERATION TYPE AND LOCK MODE (NEVER READ OPERATION OR SCAN IN LOG) */
+/* ------------------------------------------------------------------------- */
+ LqhKeyReq::setLockType(Treqinfo, regTcPtr->operation);
+ LqhKeyReq::setOperation(Treqinfo, regTcPtr->operation);
+ regTcPtr->reqinfo = Treqinfo;
+/* ------------------------------------------------------------------------ */
+/* NO OF BACKUP IS SET TO ONE AND NUMBER OF STANDBY NODES IS SET TO ZERO. */
+/* THUS THE RECEIVING NODE WILL EXPECT THAT IT IS THE LAST NODE AND WILL */
+/* SEND COMPLETED AS THE RESPONSE SIGNAL SINCE DIRTY_OP BIT IS SET. */
+/* ------------------------------------------------------------------------ */
+/* ------------------------------------------------------------------------- */
+/* SET REPLICA TYPE TO PRIMARY AND NUMBER OF REPLICA TO ONE */
+/* ------------------------------------------------------------------------- */
+ regTcPtr->lastReplicaNo = 0;
+ regTcPtr->apiVersionNo = 0;
+ regTcPtr->nextSeqNoReplica = 0;
+ regTcPtr->opExec = 0;
+ regTcPtr->storedProcId = ZNIL;
+ regTcPtr->readlenAi = 0;
+ regTcPtr->nodeAfterNext[0] = ZNIL;
+ regTcPtr->nodeAfterNext[1] = ZNIL;
+ regTcPtr->dirtyOp = ZFALSE;
+ regTcPtr->tcBlockref = cownref;
+}//Dblqh::initReqinfoExecSr()
+
+/* --------------------------------------------------------------------------
+ * ------- INSERT FRAGMENT -------
+ *
+ * ------------------------------------------------------------------------- */
+bool Dblqh::insertFragrec(Signal* signal, Uint32 fragId)
+{
+ terrorCode = ZOK;
+ if (cfirstfreeFragrec == RNIL) {
+ jam();
+ terrorCode = ZNO_FREE_FRAGMENTREC;
+ return false;
+ }//if
+ seizeFragmentrec(signal);
+ for (Uint32 i = (NO_OF_FRAG_PER_NODE - 1); (Uint32)~i; i--) {
+ jam();
+ if (tabptr.p->fragid[i] == ZNIL) {
+ jam();
+ tabptr.p->fragid[i] = fragId;
+ tabptr.p->fragrec[i] = fragptr.i;
+ return true;
+ }//if
+ }//for
+ terrorCode = ZTOO_MANY_FRAGMENTS;
+ return false;
+}//Dblqh::insertFragrec()
+
+/* --------------------------------------------------------------------------
+ * ------- LINK OPERATION IN ACTIVE LIST ON FRAGMENT -------
+ *
+ * SUBROUTINE SHORT NAME: LFQ
+// Input Pointers:
+// tcConnectptr
+// fragptr
+* ------------------------------------------------------------------------- */
+void Dblqh::linkFragQueue(Signal* signal)
+{
+ TcConnectionrecPtr lfqTcConnectptr;
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ Fragrecord * const regFragPtr = fragptr.p;
+ Uint32 tcIndex = tcConnectptr.i;
+
+ lfqTcConnectptr.i = regFragPtr->lastWaitQueue;
+ regTcPtr->nextTc = RNIL;
+ regFragPtr->lastWaitQueue = tcIndex;
+ regTcPtr->prevTc = lfqTcConnectptr.i;
+ ndbrequire(regTcPtr->listState == TcConnectionrec::NOT_IN_LIST);
+ regTcPtr->listState = TcConnectionrec::WAIT_QUEUE_LIST;
+ if (lfqTcConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(lfqTcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ lfqTcConnectptr.p->nextTc = tcIndex;
+ } else {
+ regFragPtr->firstWaitQueue = tcIndex;
+ }//if
+ return;
+}//Dblqh::linkFragQueue()
+
+/* -------------------------------------------------------------------------
+ * ------- LINK OPERATION INTO WAITING FOR LOGGING -------
+ *
+ * SUBROUTINE SHORT NAME = LWL
+// Input Pointers:
+// tcConnectptr
+// logPartPtr
+ * ------------------------------------------------------------------------- */
+void Dblqh::linkWaitLog(Signal* signal, LogPartRecordPtr regLogPartPtr)
+{
+ TcConnectionrecPtr lwlTcConnectptr;
+
+/* -------------------------------------------------- */
+/* LINK ACTIVE OPERATION INTO QUEUE WAITING FOR */
+/* ACCESS TO THE LOG PART. */
+/* -------------------------------------------------- */
+ lwlTcConnectptr.i = regLogPartPtr.p->lastLogQueue;
+ if (lwlTcConnectptr.i == RNIL) {
+ jam();
+ regLogPartPtr.p->firstLogQueue = tcConnectptr.i;
+ } else {
+ jam();
+ ptrCheckGuard(lwlTcConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ lwlTcConnectptr.p->nextTcLogQueue = tcConnectptr.i;
+ }//if
+ regLogPartPtr.p->lastLogQueue = tcConnectptr.i;
+ tcConnectptr.p->nextTcLogQueue = RNIL;
+ if (regLogPartPtr.p->LogLqhKeyReqSent == ZFALSE) {
+ jam();
+ regLogPartPtr.p->LogLqhKeyReqSent = ZTRUE;
+ signal->theData[0] = ZLOG_LQHKEYREQ;
+ signal->theData[1] = regLogPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+}//Dblqh::linkWaitLog()
+
+/* --------------------------------------------------------------------------
+ * ------- START THE NEXT OPERATION ON THIS LOG PART IF ANY -------
+ * ------- OPERATIONS ARE QUEUED. -------
+ *
+ * SUBROUTINE SHORT NAME = LNS
+// Input Pointers:
+// tcConnectptr
+// logPartPtr
+ * ------------------------------------------------------------------------- */
+void Dblqh::logNextStart(Signal* signal)
+{
+ LogPartRecordPtr lnsLogPartPtr;
+ UintR tlnsStillWaiting;
+ LogPartRecord * const regLogPartPtr = logPartPtr.p;
+
+ if ((regLogPartPtr->firstLogQueue == RNIL) &&
+ (regLogPartPtr->logPartState == LogPartRecord::ACTIVE) &&
+ (regLogPartPtr->waitWriteGciLog != LogPartRecord::WWGL_TRUE)) {
+// --------------------------------------------------------------------------
+// Optimised route for the common case
+// --------------------------------------------------------------------------
+ regLogPartPtr->logPartState = LogPartRecord::IDLE;
+ return;
+ }//if
+ if (regLogPartPtr->firstLogQueue != RNIL) {
+ jam();
+ if (regLogPartPtr->LogLqhKeyReqSent == ZFALSE) {
+ jam();
+ regLogPartPtr->LogLqhKeyReqSent = ZTRUE;
+ signal->theData[0] = ZLOG_LQHKEYREQ;
+ signal->theData[1] = logPartPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+ } else {
+ if (regLogPartPtr->logPartState == LogPartRecord::ACTIVE) {
+ jam();
+ regLogPartPtr->logPartState = LogPartRecord::IDLE;
+ } else {
+ jam();
+ }//if
+ }//if
+ if (regLogPartPtr->waitWriteGciLog != LogPartRecord::WWGL_TRUE) {
+ jam();
+ return;
+ } else {
+ jam();
+/* --------------------------------------------------------------------------
+ * A COMPLETE GCI LOG RECORD IS WAITING TO BE WRITTEN. WE GIVE THIS HIGHEST
+ * PRIORITY AND WRITE IT IMMEDIATELY. AFTER WRITING IT WE CHECK IF ANY MORE
+ * LOG PARTS ARE WAITING. IF NOT WE SEND A SIGNAL THAT INITIALISES THE GCP
+ * RECORD TO WAIT UNTIL ALL COMPLETE GCI LOG RECORDS HAVE REACHED TO DISK.
+ * -------------------------------------------------------------------------- */
+ writeCompletedGciLog(signal);
+ logPartPtr.p->waitWriteGciLog = LogPartRecord::WWGL_FALSE;
+ tlnsStillWaiting = ZFALSE;
+ for (lnsLogPartPtr.i = 0; lnsLogPartPtr.i < 4; lnsLogPartPtr.i++) {
+ jam();
+ ptrAss(lnsLogPartPtr, logPartRecord);
+ if (lnsLogPartPtr.p->waitWriteGciLog == LogPartRecord::WWGL_TRUE) {
+ jam();
+ tlnsStillWaiting = ZTRUE;
+ }//if
+ }//for
+ if (tlnsStillWaiting == ZFALSE) {
+ jam();
+ signal->theData[0] = ZINIT_GCP_REC;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 1, JBB);
+ }//if
+ }//if
+}//Dblqh::logNextStart()
+
+/* --------------------------------------------------------------------------
+ * ------- MOVE PAGES FROM LFO RECORD TO PAGE REFERENCE RECORD -------
+ * WILL ALWAYS MOVE 8 PAGES TO A PAGE REFERENCE RECORD.
+ *
+ * SUBROUTINE SHORT NAME = MPR
+ * ------------------------------------------------------------------------- */
+void Dblqh::moveToPageRef(Signal* signal)
+{
+ LogPageRecordPtr mprLogPagePtr;
+ PageRefRecordPtr mprPageRefPtr;
+ UintR tmprIndex;
+
+/* --------------------------------------------------------------------------
+ * ------- INSERT PAGE REFERENCE RECORD -------
+ *
+ * INPUT: LFO_PTR LOG FILE OPERATION RECORD
+ * LOG_PART_PTR LOG PART RECORD
+ * PAGE_REF_PTR THE PAGE REFERENCE RECORD TO BE INSERTED.
+ * ------------------------------------------------------------------------- */
+ PageRefRecordPtr iprPageRefPtr;
+
+ if ((logPartPtr.p->mmBufferSize + 8) >= ZMAX_MM_BUFFER_SIZE) {
+ jam();
+ pageRefPtr.i = logPartPtr.p->firstPageRef;
+ ptrCheckGuard(pageRefPtr, cpageRefFileSize, pageRefRecord);
+ releasePrPages(signal);
+ removePageRef(signal);
+ } else {
+ jam();
+ logPartPtr.p->mmBufferSize = logPartPtr.p->mmBufferSize + 8;
+ }//if
+ seizePageRef(signal);
+ if (logPartPtr.p->firstPageRef == RNIL) {
+ jam();
+ logPartPtr.p->firstPageRef = pageRefPtr.i;
+ } else {
+ jam();
+ iprPageRefPtr.i = logPartPtr.p->lastPageRef;
+ ptrCheckGuard(iprPageRefPtr, cpageRefFileSize, pageRefRecord);
+ iprPageRefPtr.p->prNext = pageRefPtr.i;
+ }//if
+ pageRefPtr.p->prPrev = logPartPtr.p->lastPageRef;
+ logPartPtr.p->lastPageRef = pageRefPtr.i;
+
+ pageRefPtr.p->prFileNo = logFilePtr.p->fileNo;
+ pageRefPtr.p->prPageNo = lfoPtr.p->lfoPageNo;
+ tmprIndex = 0;
+ mprLogPagePtr.i = lfoPtr.p->firstLfoPage;
+MPR_LOOP:
+ arrGuard(tmprIndex, 8);
+ pageRefPtr.p->pageRef[tmprIndex] = mprLogPagePtr.i;
+ tmprIndex = tmprIndex + 1;
+ ptrCheckGuard(mprLogPagePtr, clogPageFileSize, logPageRecord);
+ mprLogPagePtr.i = mprLogPagePtr.p->logPageWord[ZNEXT_PAGE];
+ if (mprLogPagePtr.i != RNIL) {
+ jam();
+ goto MPR_LOOP;
+ }//if
+ mprPageRefPtr.i = pageRefPtr.p->prPrev;
+ if (mprPageRefPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(mprPageRefPtr, cpageRefFileSize, pageRefRecord);
+ mprLogPagePtr.i = mprPageRefPtr.p->pageRef[7];
+ ptrCheckGuard(mprLogPagePtr, clogPageFileSize, logPageRecord);
+ mprLogPagePtr.p->logPageWord[ZNEXT_PAGE] = pageRefPtr.p->pageRef[0];
+ }//if
+}//Dblqh::moveToPageRef()
+
+/* ------------------------------------------------------------------------- */
+/* ------- READ THE ATTRINFO FROM THE LOG ------- */
+/* */
+/* SUBROUTINE SHORT NAME = RA */
+/* ------------------------------------------------------------------------- */
+void Dblqh::readAttrinfo(Signal* signal)
+{
+ Uint32 remainingLen = tcConnectptr.p->totSendlenAi;
+ if (remainingLen == 0) {
+ jam();
+ tcConnectptr.p->reclenAiLqhkey = 0;
+ return;
+ }//if
+ Uint32 dataLen = remainingLen;
+ if (remainingLen > 5)
+ dataLen = 5;
+ readLogData(signal, dataLen, &tcConnectptr.p->firstAttrinfo[0]);
+ tcConnectptr.p->reclenAiLqhkey = dataLen;
+ remainingLen -= dataLen;
+ while (remainingLen > 0) {
+ jam();
+ dataLen = remainingLen;
+ if (remainingLen > 22)
+ dataLen = 22;
+ seizeAttrinbuf(signal);
+ readLogData(signal, dataLen, &attrinbufptr.p->attrbuf[0]);
+ attrinbufptr.p->attrbuf[ZINBUF_DATA_LEN] = dataLen;
+ remainingLen -= dataLen;
+ }//while
+}//Dblqh::readAttrinfo()
+
+/* ------------------------------------------------------------------------- */
+/* ------- READ COMMIT LOG ------- */
+/* */
+/* SUBROUTINE SHORT NAME = RCL */
+/* ------------------------------------------------------------------------- */
+void Dblqh::readCommitLog(Signal* signal, CommitLogRecord* commitLogRecord)
+{
+ Uint32 trclPageIndex = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ if ((trclPageIndex + (ZCOMMIT_LOG_SIZE - 1)) < ZPAGE_SIZE) {
+ jam();
+ tcConnectptr.p->tableref = logPagePtr.p->logPageWord[trclPageIndex + 0];
+ tcConnectptr.p->schemaVersion = logPagePtr.p->logPageWord[trclPageIndex + 1];
+ tcConnectptr.p->fragmentid = logPagePtr.p->logPageWord[trclPageIndex + 2];
+ commitLogRecord->fileNo = logPagePtr.p->logPageWord[trclPageIndex + 3];
+ commitLogRecord->startPageNo = logPagePtr.p->logPageWord[trclPageIndex + 4];
+ commitLogRecord->startPageIndex = logPagePtr.p->logPageWord[trclPageIndex + 5];
+ commitLogRecord->stopPageNo = logPagePtr.p->logPageWord[trclPageIndex + 6];
+ tcConnectptr.p->gci = logPagePtr.p->logPageWord[trclPageIndex + 7];
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] =
+ (trclPageIndex + ZCOMMIT_LOG_SIZE) - 1;
+ } else {
+ jam();
+ tcConnectptr.p->tableref = readLogword(signal);
+ tcConnectptr.p->schemaVersion = readLogword(signal);
+ tcConnectptr.p->fragmentid = readLogword(signal);
+ commitLogRecord->fileNo = readLogword(signal);
+ commitLogRecord->startPageNo = readLogword(signal);
+ commitLogRecord->startPageIndex = readLogword(signal);
+ commitLogRecord->stopPageNo = readLogword(signal);
+ tcConnectptr.p->gci = readLogword(signal);
+ }//if
+ tcConnectptr.p->transid[0] = logPartPtr.i + 65536;
+ tcConnectptr.p->transid[1] = (DBLQH << 20) + (cownNodeid << 8);
+}//Dblqh::readCommitLog()
+
+/* ------------------------------------------------------------------------- */
+/* ------- READ LOG PAGES FROM DISK IN ORDER TO EXECUTE A LOG ------- */
+/* RECORD WHICH WAS NOT FOUND IN MAIN MEMORY. */
+/* */
+/* SUBROUTINE SHORT NAME = REL */
+/* ------------------------------------------------------------------------- */
+void Dblqh::readExecLog(Signal* signal)
+{
+ UintR trelIndex;
+ UintR trelI;
+
+ seizeLfo(signal);
+ initLfo(signal);
+ trelI = logPartPtr.p->execSrStopPageNo - logPartPtr.p->execSrStartPageNo;
+ arrGuard(trelI + 1, 16);
+ lfoPtr.p->logPageArray[trelI + 1] = logPartPtr.p->execSrStartPageNo;
+ for (trelIndex = logPartPtr.p->execSrStopPageNo; (trelIndex >= logPartPtr.p->execSrStartPageNo) &&
+ (UintR)~trelIndex; trelIndex--) {
+ jam();
+ seizeLogpage(signal);
+ arrGuard(trelI, 16);
+ lfoPtr.p->logPageArray[trelI] = logPagePtr.i;
+ trelI--;
+ }//for
+ lfoPtr.p->lfoPageNo = logPartPtr.p->execSrStartPageNo;
+ lfoPtr.p->noPagesRw = (logPartPtr.p->execSrStopPageNo -
+ logPartPtr.p->execSrStartPageNo) + 1;
+ lfoPtr.p->firstLfoPage = lfoPtr.p->logPageArray[0];
+ signal->theData[0] = logFilePtr.p->fileRef;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lfoPtr.i;
+ signal->theData[3] = ZLIST_OF_MEM_PAGES; // edtjamo TR509 //ZLIST_OF_PAIRS;
+ signal->theData[4] = ZVAR_NO_LOG_PAGE_WORD;
+ signal->theData[5] = lfoPtr.p->noPagesRw;
+ signal->theData[6] = lfoPtr.p->logPageArray[0];
+ signal->theData[7] = lfoPtr.p->logPageArray[1];
+ signal->theData[8] = lfoPtr.p->logPageArray[2];
+ signal->theData[9] = lfoPtr.p->logPageArray[3];
+ signal->theData[10] = lfoPtr.p->logPageArray[4];
+ signal->theData[11] = lfoPtr.p->logPageArray[5];
+ signal->theData[12] = lfoPtr.p->logPageArray[6];
+ signal->theData[13] = lfoPtr.p->logPageArray[7];
+ signal->theData[14] = lfoPtr.p->logPageArray[8];
+ signal->theData[15] = lfoPtr.p->logPageArray[9];
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 16, JBA);
+}//Dblqh::readExecLog()
+
+/* ------------------------------------------------------------------------- */
+/* ------- READ 64 KBYTES WHEN EXECUTING THE FRAGMENT LOG ------- */
+/* */
+/* SUBROUTINE SHORT NAME = RES */
+/* ------------------------------------------------------------------------- */
+void Dblqh::readExecSrNewMbyte(Signal* signal)
+{
+ logFilePtr.p->currentFilepage = logFilePtr.p->currentMbyte * ZPAGES_IN_MBYTE;
+ logFilePtr.p->filePosition = logFilePtr.p->currentMbyte * ZPAGES_IN_MBYTE;
+ logPartPtr.p->execSrPagesRead = 0;
+ logPartPtr.p->execSrPagesReading = 0;
+ logPartPtr.p->execSrPagesExecuted = 0;
+ readExecSr(signal);
+ logPartPtr.p->logExecState = LogPartRecord::LES_WAIT_READ_EXEC_SR_NEW_MBYTE;
+}//Dblqh::readExecSrNewMbyte()
+
+/* ------------------------------------------------------------------------- */
+/* ------- READ 64 KBYTES WHEN EXECUTING THE FRAGMENT LOG ------- */
+/* */
+/* SUBROUTINE SHORT NAME = RES */
+/* ------------------------------------------------------------------------- */
+void Dblqh::readExecSr(Signal* signal)
+{
+ UintR tresPageid;
+ UintR tresIndex;
+
+ tresPageid = logFilePtr.p->filePosition;
+ seizeLfo(signal);
+ initLfo(signal);
+ for (tresIndex = 7; (UintR)~tresIndex; tresIndex--) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* GO BACKWARDS SINCE WE INSERT AT THE BEGINNING AND WE WANT THAT FIRST PAGE */
+/* SHALL BE FIRST AND LAST PAGE LAST. */
+/* ------------------------------------------------------------------------- */
+ seizeLogpage(signal);
+ lfoPtr.p->logPageArray[tresIndex] = logPagePtr.i;
+ }//for
+ lfoPtr.p->lfoState = LogFileOperationRecord::READ_EXEC_SR;
+ lfoPtr.p->lfoPageNo = tresPageid;
+ logFilePtr.p->filePosition = logFilePtr.p->filePosition + 8;
+ logPartPtr.p->execSrPagesReading = logPartPtr.p->execSrPagesReading + 8;
+ lfoPtr.p->noPagesRw = 8;
+ lfoPtr.p->firstLfoPage = lfoPtr.p->logPageArray[0];
+ signal->theData[0] = logFilePtr.p->fileRef;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lfoPtr.i;
+ signal->theData[3] = ZLIST_OF_MEM_PAGES;
+ signal->theData[4] = ZVAR_NO_LOG_PAGE_WORD;
+ signal->theData[5] = 8;
+ signal->theData[6] = lfoPtr.p->logPageArray[0];
+ signal->theData[7] = lfoPtr.p->logPageArray[1];
+ signal->theData[8] = lfoPtr.p->logPageArray[2];
+ signal->theData[9] = lfoPtr.p->logPageArray[3];
+ signal->theData[10] = lfoPtr.p->logPageArray[4];
+ signal->theData[11] = lfoPtr.p->logPageArray[5];
+ signal->theData[12] = lfoPtr.p->logPageArray[6];
+ signal->theData[13] = lfoPtr.p->logPageArray[7];
+ signal->theData[14] = tresPageid;
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 15, JBA);
+}//Dblqh::readExecSr()
+
+/* ------------------------------------------------------------------------- */
+/* ------------ READ THE PRIMARY KEY FROM THE LOG ---------------- */
+/* */
+/* SUBROUTINE SHORT NAME = RK */
+/* --------------------------------------------------------------------------*/
+void Dblqh::readKey(Signal* signal)
+{
+ Uint32 remainingLen = tcConnectptr.p->primKeyLen;
+ ndbrequire(remainingLen != 0);
+ Uint32 dataLen = remainingLen;
+ if (remainingLen > 4)
+ dataLen = 4;
+ readLogData(signal, dataLen, &tcConnectptr.p->tupkeyData[0]);
+ remainingLen -= dataLen;
+ while (remainingLen > 0) {
+ jam();
+ seizeTupkeybuf(signal);
+ dataLen = remainingLen;
+ if (dataLen > 4)
+ dataLen = 4;
+ readLogData(signal, dataLen, &databufptr.p->data[0]);
+ remainingLen -= dataLen;
+ }//while
+}//Dblqh::readKey()
+
+/* ------------------------------------------------------------------------- */
+/* ------------ READ A NUMBER OF WORDS FROM LOG INTO CDATA ---------------- */
+/* */
+/* SUBROUTINE SHORT NAME = RLD */
+/* --------------------------------------------------------------------------*/
+void Dblqh::readLogData(Signal* signal, Uint32 noOfWords, Uint32* dataPtr)
+{
+ ndbrequire(noOfWords < 32);
+ Uint32 logPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ if ((logPos + noOfWords) >= ZPAGE_SIZE) {
+ for (Uint32 i = 0; i < noOfWords; i++)
+ dataPtr[i] = readLogwordExec(signal);
+ } else {
+ MEMCOPY_NO_WORDS(dataPtr, &logPagePtr.p->logPageWord[logPos], noOfWords);
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPos + noOfWords;
+ }//if
+}//Dblqh::readLogData()
+
+/* ------------------------------------------------------------------------- */
+/* ------------ READ THE LOG HEADER OF A PREPARE LOG HEADER ---------------- */
+/* */
+/* SUBROUTINE SHORT NAME = RLH */
+/* --------------------------------------------------------------------------*/
+void Dblqh::readLogHeader(Signal* signal)
+{
+ Uint32 logPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ if ((logPos + ZLOG_HEAD_SIZE) < ZPAGE_SIZE) {
+ jam();
+ tcConnectptr.p->hashValue = logPagePtr.p->logPageWord[logPos + 2];
+ tcConnectptr.p->operation = logPagePtr.p->logPageWord[logPos + 3];
+ tcConnectptr.p->totSendlenAi = logPagePtr.p->logPageWord[logPos + 4];
+ tcConnectptr.p->primKeyLen = logPagePtr.p->logPageWord[logPos + 5];
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPos + ZLOG_HEAD_SIZE;
+ } else {
+ jam();
+ readLogwordExec(signal); /* IGNORE PREPARE LOG RECORD TYPE */
+ readLogwordExec(signal); /* IGNORE LOG RECORD SIZE */
+ tcConnectptr.p->hashValue = readLogwordExec(signal);
+ tcConnectptr.p->operation = readLogwordExec(signal);
+ tcConnectptr.p->totSendlenAi = readLogwordExec(signal);
+ tcConnectptr.p->primKeyLen = readLogwordExec(signal);
+ }//if
+}//Dblqh::readLogHeader()
+
+/* ------------------------------------------------------------------------- */
+/* ------- READ A WORD FROM THE LOG ------- */
+/* */
+/* OUTPUT: TLOG_WORD */
+/* SUBROUTINE SHORT NAME = RLW */
+/* ------------------------------------------------------------------------- */
+Uint32 Dblqh::readLogword(Signal* signal)
+{
+ Uint32 logPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ ndbrequire(logPos < ZPAGE_SIZE);
+ Uint32 logWord = logPagePtr.p->logPageWord[logPos];
+ logPos++;
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPos;
+ if (logPos >= ZPAGE_SIZE) {
+ jam();
+ logPagePtr.i = logPagePtr.p->logPageWord[ZNEXT_PAGE];
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = ZPAGE_HEADER_SIZE;
+ logFilePtr.p->currentLogpage = logPagePtr.i;
+ logFilePtr.p->currentFilepage++;
+ logPartPtr.p->execSrPagesRead--;
+ logPartPtr.p->execSrPagesExecuted++;
+ }//if
+ return logWord;
+}//Dblqh::readLogword()
+
+/* ------------------------------------------------------------------------- */
+/* ------- READ A WORD FROM THE LOG WHEN EXECUTING A LOG RECORD ------- */
+/* */
+/* OUTPUT: TLOG_WORD */
+/* SUBROUTINE SHORT NAME = RWE */
+/* ------------------------------------------------------------------------- */
+Uint32 Dblqh::readLogwordExec(Signal* signal)
+{
+ Uint32 logPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ ndbrequire(logPos < ZPAGE_SIZE);
+ Uint32 logWord = logPagePtr.p->logPageWord[logPos];
+ logPos++;
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPos;
+ if (logPos >= ZPAGE_SIZE) {
+ jam();
+ logPagePtr.i = logPagePtr.p->logPageWord[ZNEXT_PAGE];
+ if (logPagePtr.i != RNIL){
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = ZPAGE_HEADER_SIZE;
+ } else {
+ // Reading word at the last pos in the last page
+ // Don't step forward to next page!
+ jam();
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX]++;
+ }
+ }//if
+ return logWord;
+}//Dblqh::readLogwordExec()
+
+/* ------------------------------------------------------------------------- */
+/* ------- READ A SINGLE PAGE FROM THE LOG ------- */
+/* */
+/* INPUT: TRSP_PAGE_NO */
+/* SUBROUTINE SHORT NAME = RSP */
+/* ------------------------------------------------------------------------- */
+void Dblqh::readSinglePage(Signal* signal, Uint32 pageNo)
+{
+ seizeLfo(signal);
+ initLfo(signal);
+ seizeLogpage(signal);
+ lfoPtr.p->firstLfoPage = logPagePtr.i;
+ lfoPtr.p->lfoPageNo = pageNo;
+ lfoPtr.p->noPagesRw = 1;
+ signal->theData[0] = logFilePtr.p->fileRef;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lfoPtr.i;
+ signal->theData[3] = ZLIST_OF_PAIRS;
+ signal->theData[4] = ZVAR_NO_LOG_PAGE_WORD;
+ signal->theData[5] = 1;
+ signal->theData[6] = logPagePtr.i;
+ signal->theData[7] = pageNo;
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 8, JBA);
+}//Dblqh::readSinglePage()
+
+/* --------------------------------------------------------------------------
+ * ------- RELEASE OPERATION FROM ACTIVE LIST ON FRAGMENT -------
+ *
+ * SUBROUTINE SHORT NAME = RAC
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseAccList(Signal* signal)
+{
+ TcConnectionrecPtr racTcNextConnectptr;
+ TcConnectionrecPtr racTcPrevConnectptr;
+
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ racTcPrevConnectptr.i = tcConnectptr.p->prevTc;
+ racTcNextConnectptr.i = tcConnectptr.p->nextTc;
+ if (tcConnectptr.p->listState != TcConnectionrec::ACC_BLOCK_LIST) {
+ jam();
+ systemError(signal);
+ }//if
+ tcConnectptr.p->listState = TcConnectionrec::NOT_IN_LIST;
+ if (racTcNextConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(racTcNextConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ racTcNextConnectptr.p->prevTc = racTcPrevConnectptr.i;
+ }//if
+ if (racTcPrevConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(racTcPrevConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ racTcPrevConnectptr.p->nextTc = tcConnectptr.p->nextTc;
+ } else {
+ jam();
+ /* ---------------------------------------------------------------------
+ * OPERATION RECORD IS FIRST IN ACTIVE LIST
+ * THIS MEANS THAT THERE EXISTS NO PREVIOUS TC THAT NEEDS TO BE UPDATED.
+ * --------------------------------------------------------------------- */
+ fragptr.p->accBlockedList = racTcNextConnectptr.i;
+ }//if
+}//Dblqh::releaseAccList()
+
+/* --------------------------------------------------------------------------
+ * ------- REMOVE COPY FRAGMENT FROM ACTIVE COPY LIST -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseActiveCopy(Signal* signal)
+{
+ /* MUST BE 8 BIT */
+ UintR tracFlag;
+ UintR tracIndex;
+
+ tracFlag = ZFALSE;
+ for (tracIndex = 0; tracIndex < 4; tracIndex++) {
+ if (tracFlag == ZFALSE) {
+ jam();
+ if (cactiveCopy[tracIndex] == fragptr.i) {
+ jam();
+ tracFlag = ZTRUE;
+ }//if
+ } else {
+ if (tracIndex < 3) {
+ jam();
+ cactiveCopy[tracIndex - 1] = cactiveCopy[tracIndex];
+ } else {
+ jam();
+ cactiveCopy[3] = RNIL;
+ }//if
+ }//if
+ }//for
+ ndbrequire(tracFlag == ZTRUE);
+ cnoActiveCopy--;
+}//Dblqh::releaseActiveCopy()
+
+/* --------------------------------------------------------------------------
+ * ------- RELEASE OPERATION FROM ACTIVE LIST ON FRAGMENT -------
+ *
+ * SUBROUTINE SHORT NAME = RAL
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseActiveList(Signal* signal)
+{
+ TcConnectionrecPtr ralTcNextConnectptr;
+ TcConnectionrecPtr ralTcPrevConnectptr;
+ ralTcPrevConnectptr.i = tcConnectptr.p->prevTc;
+ ralTcNextConnectptr.i = tcConnectptr.p->nextTc;
+ ndbrequire(tcConnectptr.p->listState == TcConnectionrec::IN_ACTIVE_LIST);
+ tcConnectptr.p->listState = TcConnectionrec::NOT_IN_LIST;
+ if (ralTcNextConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(ralTcNextConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ ralTcNextConnectptr.p->prevTc = ralTcPrevConnectptr.i;
+ }//if
+ if (ralTcPrevConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(ralTcPrevConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ ralTcPrevConnectptr.p->nextTc = tcConnectptr.p->nextTc;
+ } else {
+ jam();
+ /* ----------------------------------------------------------------------
+ * OPERATION RECORD IS FIRST IN ACTIVE LIST
+ * THIS MEANS THAT THERE EXISTS NO PREVIOUS TC THAT NEEDS TO BE UPDATED.
+ * --------------------------------------------------------------------- */
+ fragptr.p->activeList = ralTcNextConnectptr.i;
+ }//if
+}//Dblqh::releaseActiveList()
+
+/* --------------------------------------------------------------------------
+ * ------- RELEASE ADD FRAGMENT RECORD -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseAddfragrec(Signal* signal)
+{
+ addfragptr.p->addfragStatus = AddFragRecord::FREE;
+ addfragptr.p->nextAddfragrec = cfirstfreeAddfragrec;
+ cfirstfreeAddfragrec = addfragptr.i;
+}//Dblqh::releaseAddfragrec()
+
+/* --------------------------------------------------------------------------
+ * ------- RELEASE FRAGMENT RECORD -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseFragrec()
+{
+ fragptr.p->fragStatus = Fragrecord::FREE;
+ fragptr.p->nextFrag = cfirstfreeFragrec;
+ cfirstfreeFragrec = fragptr.i;
+}//Dblqh::releaseFragrec()
+
+/* --------------------------------------------------------------------------
+ * ------- RELEASE LCP LOCAL RECORD -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseLcpLoc(Signal* signal)
+{
+ lcpLocptr.p->lcpLocstate = LcpLocRecord::IDLE;
+ lcpLocptr.p->nextLcpLoc = cfirstfreeLcpLoc;
+ cfirstfreeLcpLoc = lcpLocptr.i;
+}//Dblqh::releaseLcpLoc()
+
+/* --------------------------------------------------------------------------
+ * ------- RELEASE A PAGE REFERENCE RECORD. -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::releasePageRef(Signal* signal)
+{
+ pageRefPtr.p->prNext = cfirstfreePageRef;
+ cfirstfreePageRef = pageRefPtr.i;
+}//Dblqh::releasePageRef()
+
+/* --------------------------------------------------------------------------
+ * --- RELEASE ALL PAGES IN THE MM BUFFER AFTER EXECUTING THE LOG ON IT. ----
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseMmPages(Signal* signal)
+{
+RMP_LOOP:
+ jam();
+ pageRefPtr.i = logPartPtr.p->firstPageRef;
+ if (pageRefPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(pageRefPtr, cpageRefFileSize, pageRefRecord);
+ releasePrPages(signal);
+ removePageRef(signal);
+ goto RMP_LOOP;
+ }//if
+}//Dblqh::releaseMmPages()
+
+/* --------------------------------------------------------------------------
+ * ------- RELEASE A SET OF PAGES AFTER EXECUTING THE LOG ON IT. -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::releasePrPages(Signal* signal)
+{
+ UintR trppIndex;
+
+ for (trppIndex = 0; trppIndex <= 7; trppIndex++) {
+ jam();
+ logPagePtr.i = pageRefPtr.p->pageRef[trppIndex];
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ releaseLogpage(signal);
+ }//for
+}//Dblqh::releasePrPages()
+
+/* --------------------------------------------------------------------------
+ * ------- RELEASE OPERATION FROM WAIT QUEUE LIST ON FRAGMENT -------
+ *
+ * SUBROUTINE SHORT NAME : RWA
+ * ------------------------------------------------------------------------- */
+void Dblqh::releaseWaitQueue(Signal* signal)
+{
+ TcConnectionrecPtr rwaTcNextConnectptr;
+ TcConnectionrecPtr rwaTcPrevConnectptr;
+
+ fragptr.i = tcConnectptr.p->fragmentptr;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ rwaTcPrevConnectptr.i = tcConnectptr.p->prevTc;
+ rwaTcNextConnectptr.i = tcConnectptr.p->nextTc;
+ if (tcConnectptr.p->listState != TcConnectionrec::WAIT_QUEUE_LIST) {
+ jam();
+ systemError(signal);
+ }//if
+ tcConnectptr.p->listState = TcConnectionrec::NOT_IN_LIST;
+ if (rwaTcNextConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(rwaTcNextConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ rwaTcNextConnectptr.p->prevTc = rwaTcPrevConnectptr.i;
+ } else {
+ jam();
+ fragptr.p->lastWaitQueue = rwaTcPrevConnectptr.i;
+ }//if
+ if (rwaTcPrevConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(rwaTcPrevConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ rwaTcPrevConnectptr.p->nextTc = rwaTcNextConnectptr.i;
+ } else {
+ jam();
+ fragptr.p->firstWaitQueue = rwaTcNextConnectptr.i;
+ }//if
+}//Dblqh::releaseWaitQueue()
+
+/* --------------------------------------------------------------------------
+ * ------- REMOVE OPERATION RECORD FROM LIST ON LOG PART OF NOT -------
+ * COMPLETED OPERATIONS IN THE LOG.
+ *
+ * SUBROUTINE SHORT NAME = RLO
+ * ------------------------------------------------------------------------- */
+void Dblqh::removeLogTcrec(Signal* signal)
+{
+ TcConnectionrecPtr rloTcNextConnectptr;
+ TcConnectionrecPtr rloTcPrevConnectptr;
+ rloTcPrevConnectptr.i = tcConnectptr.p->prevLogTcrec;
+ rloTcNextConnectptr.i = tcConnectptr.p->nextLogTcrec;
+ if (rloTcNextConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(rloTcNextConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ rloTcNextConnectptr.p->prevLogTcrec = rloTcPrevConnectptr.i;
+ } else {
+ jam();
+ logPartPtr.p->lastLogTcrec = rloTcPrevConnectptr.i;
+ }//if
+ if (rloTcPrevConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(rloTcPrevConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ rloTcPrevConnectptr.p->nextLogTcrec = rloTcNextConnectptr.i;
+ } else {
+ jam();
+ logPartPtr.p->firstLogTcrec = rloTcNextConnectptr.i;
+ }//if
+}//Dblqh::removeLogTcrec()
+
+/* --------------------------------------------------------------------------
+ * ------- REMOVE PAGE REFERENCE RECORD FROM LIST IN THIS LOG PART -------
+ *
+ * SUBROUTINE SHORT NAME = RPR
+ * ------------------------------------------------------------------------- */
+void Dblqh::removePageRef(Signal* signal)
+{
+ PageRefRecordPtr rprPageRefPtr;
+
+ pageRefPtr.i = logPartPtr.p->firstPageRef;
+ if (pageRefPtr.i != RNIL) {
+ jam();
+ ptrCheckGuard(pageRefPtr, cpageRefFileSize, pageRefRecord);
+ if (pageRefPtr.p->prNext == RNIL) {
+ jam();
+ logPartPtr.p->lastPageRef = RNIL;
+ logPartPtr.p->firstPageRef = RNIL;
+ } else {
+ jam();
+ logPartPtr.p->firstPageRef = pageRefPtr.p->prNext;
+ rprPageRefPtr.i = pageRefPtr.p->prNext;
+ ptrCheckGuard(rprPageRefPtr, cpageRefFileSize, pageRefRecord);
+ rprPageRefPtr.p->prPrev = RNIL;
+ }//if
+ releasePageRef(signal);
+ }//if
+}//Dblqh::removePageRef()
+
+/* ------------------------------------------------------------------------- */
+/* ------- RETURN FROM EXECUTION OF LOG ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+Uint32 Dblqh::returnExecLog(Signal* signal)
+{
+ tcConnectptr.p->connectState = TcConnectionrec::CONNECTED;
+ initLogPointers(signal);
+ logPartPtr.p->execSrExecuteIndex++;
+ Uint32 result = checkIfExecLog(signal);
+ if (result == ZOK) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* THIS LOG RECORD WILL BE EXECUTED AGAIN TOWARDS ANOTHER NODE. */
+/* ------------------------------------------------------------------------- */
+ logPagePtr.i = logPartPtr.p->execSrLogPage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] =
+ logPartPtr.p->execSrLogPageIndex;
+ } else {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* NO MORE EXECUTION OF THIS LOG RECORD. */
+/* ------------------------------------------------------------------------- */
+ if (logPartPtr.p->logExecState ==
+ LogPartRecord::LES_EXEC_LOGREC_FROM_FILE) {
+ jam();
+/* ------------------------------------------------------------------------- */
+/* THE LOG RECORD WAS READ FROM DISK. RELEASE ITS PAGES IMMEDIATELY. */
+/* ------------------------------------------------------------------------- */
+ lfoPtr.i = logPartPtr.p->execSrLfoRec;
+ ptrCheckGuard(lfoPtr, clfoFileSize, logFileOperationRecord);
+ releaseLfoPages(signal);
+ releaseLfo(signal);
+ logPartPtr.p->logExecState = LogPartRecord::LES_EXEC_LOG;
+ if (logPartPtr.p->execSrExecLogFile != logPartPtr.p->currentLogfile) {
+ jam();
+ LogFileRecordPtr clfLogFilePtr;
+ clfLogFilePtr.i = logPartPtr.p->execSrExecLogFile;
+ ptrCheckGuard(clfLogFilePtr, clogFileFileSize, logFileRecord);
+ clfLogFilePtr.p->logFileStatus = LogFileRecord::CLOSING_EXEC_LOG;
+ closeFile(signal, clfLogFilePtr);
+ result = ZCLOSE_FILE;
+ }//if
+ }//if
+ logPartPtr.p->execSrExecuteIndex = 0;
+ logPartPtr.p->execSrLogPage = RNIL;
+ logPartPtr.p->execSrLogPageIndex = ZNIL;
+ logPagePtr.i = logFilePtr.p->currentLogpage;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPartPtr.p->savePageIndex;
+ }//if
+ return result;
+}//Dblqh::returnExecLog()
+
+/* --------------------------------------------------------------------------
+ * ------- SEIZE ADD FRAGMENT RECORD ------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::seizeAddfragrec(Signal* signal)
+{
+ addfragptr.i = cfirstfreeAddfragrec;
+ ptrCheckGuard(addfragptr, caddfragrecFileSize, addFragRecord);
+ cfirstfreeAddfragrec = addfragptr.p->nextAddfragrec;
+}//Dblqh::seizeAddfragrec()
+
+/* --------------------------------------------------------------------------
+ * ------- SEIZE FRAGMENT RECORD -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::seizeFragmentrec(Signal* signal)
+{
+ fragptr.i = cfirstfreeFragrec;
+ ptrCheckGuard(fragptr, cfragrecFileSize, fragrecord);
+ cfirstfreeFragrec = fragptr.p->nextFrag;
+ fragptr.p->nextFrag = RNIL;
+}//Dblqh::seizeFragmentrec()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEIZE A PAGE REFERENCE RECORD. ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dblqh::seizePageRef(Signal* signal)
+{
+ pageRefPtr.i = cfirstfreePageRef;
+ ptrCheckGuard(pageRefPtr, cpageRefFileSize, pageRefRecord);
+ cfirstfreePageRef = pageRefPtr.p->prNext;
+ pageRefPtr.p->prNext = RNIL;
+}//Dblqh::seizePageRef()
+
+/* --------------------------------------------------------------------------
+ * ------- SEND ABORTED -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::sendAborted(Signal* signal)
+{
+ UintR TlastInd;
+ if (tcConnectptr.p->nextReplica == ZNIL) {
+ TlastInd = ZTRUE;
+ } else {
+ TlastInd = ZFALSE;
+ }//if
+ signal->theData[0] = tcConnectptr.p->tcOprec;
+ signal->theData[1] = tcConnectptr.p->transid[0];
+ signal->theData[2] = tcConnectptr.p->transid[1];
+ signal->theData[3] = cownNodeid;
+ signal->theData[4] = TlastInd;
+ sendSignal(tcConnectptr.p->tcBlockref, GSN_ABORTED, signal, 5, JBB);
+ return;
+}//Dblqh::sendAborted()
+
+/* --------------------------------------------------------------------------
+ * ------- SEND LQH_TRANSCONF -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::sendLqhTransconf(Signal* signal, LqhTransConf::OperationStatus stat)
+{
+ tcNodeFailptr.i = tcConnectptr.p->tcNodeFailrec;
+ ptrCheckGuard(tcNodeFailptr, ctcNodeFailrecFileSize, tcNodeFailRecord);
+
+ Uint32 reqInfo = 0;
+ LqhTransConf::setReplicaType(reqInfo, tcConnectptr.p->replicaType);
+ LqhTransConf::setReplicaNo(reqInfo, tcConnectptr.p->seqNoReplica);
+ LqhTransConf::setLastReplicaNo(reqInfo, tcConnectptr.p->lastReplicaNo);
+ LqhTransConf::setSimpleFlag(reqInfo, tcConnectptr.p->opSimple);
+ LqhTransConf::setDirtyFlag(reqInfo, tcConnectptr.p->dirtyOp);
+ LqhTransConf::setOperation(reqInfo, tcConnectptr.p->operation);
+
+ LqhTransConf * const lqhTransConf = (LqhTransConf *)&signal->theData[0];
+ lqhTransConf->tcRef = tcNodeFailptr.p->newTcRef;
+ lqhTransConf->lqhNodeId = cownNodeid;
+ lqhTransConf->operationStatus = stat;
+ lqhTransConf->lqhConnectPtr = tcConnectptr.i;
+ lqhTransConf->transId1 = tcConnectptr.p->transid[0];
+ lqhTransConf->transId2 = tcConnectptr.p->transid[1];
+ lqhTransConf->oldTcOpRec = tcConnectptr.p->tcOprec;
+ lqhTransConf->requestInfo = reqInfo;
+ lqhTransConf->gci = tcConnectptr.p->gci;
+ lqhTransConf->nextNodeId1 = tcConnectptr.p->nextReplica;
+ lqhTransConf->nextNodeId2 = tcConnectptr.p->nodeAfterNext[0];
+ lqhTransConf->nextNodeId3 = tcConnectptr.p->nodeAfterNext[1];
+ lqhTransConf->apiRef = tcConnectptr.p->applRef;
+ lqhTransConf->apiOpRec = tcConnectptr.p->applOprec;
+ lqhTransConf->tableId = tcConnectptr.p->tableref;
+ sendSignal(tcNodeFailptr.p->newTcBlockref, GSN_LQH_TRANSCONF,
+ signal, LqhTransConf::SignalLength, JBB);
+ tcNodeFailptr.p->tcRecNow = tcConnectptr.i + 1;
+ signal->theData[0] = ZLQH_TRANS_NEXT;
+ signal->theData[1] = tcNodeFailptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+}//Dblqh::sendLqhTransconf()
+
+/* --------------------------------------------------------------------------
+ * ------- START ANOTHER PHASE OF LOG EXECUTION -------
+ * RESET THE VARIABLES NEEDED BY THIS PROCESS AND SEND THE START SIGNAL
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::startExecSr(Signal* signal)
+{
+ cnoFragmentsExecSr = 0;
+ signal->theData[0] = cfirstCompletedFragSr;
+ signal->theData[1] = RNIL;
+ sendSignal(cownref, GSN_START_EXEC_SR, signal, 2, JBB);
+}//Dblqh::startExecSr()
+
+/* ¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤
+ * ¤¤¤¤¤¤¤ LOG MODULE ¤¤¤¤¤¤¤
+ * ¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤¤ */
+/* --------------------------------------------------------------------------
+ * ------- STEP FORWARD IN FRAGMENT LOG DURING LOG EXECUTION -------
+ *
+ * ------------------------------------------------------------------------- */
+void Dblqh::stepAhead(Signal* signal, Uint32 stepAheadWords)
+{
+ UintR tsaPos;
+
+ tsaPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ while ((stepAheadWords + tsaPos) >= ZPAGE_SIZE) {
+ jam();
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = ZPAGE_SIZE;
+ stepAheadWords = stepAheadWords - (ZPAGE_SIZE - tsaPos);
+ logFilePtr.p->currentLogpage = logPagePtr.p->logPageWord[ZNEXT_PAGE];
+ logPagePtr.i = logPagePtr.p->logPageWord[ZNEXT_PAGE];
+ logFilePtr.p->currentFilepage++;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = ZPAGE_HEADER_SIZE;
+ logPartPtr.p->execSrPagesRead--;
+ logPartPtr.p->execSrPagesExecuted++;
+ tsaPos = ZPAGE_HEADER_SIZE;
+ }//while
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = stepAheadWords + tsaPos;
+}//Dblqh::stepAhead()
+
+/* --------------------------------------------------------------------------
+ * ------- WRITE A ABORT LOG RECORD -------
+ *
+ * SUBROUTINE SHORT NAME: WAL
+ * ------------------------------------------------------------------------- */
+void Dblqh::writeAbortLog(Signal* signal)
+{
+ if ((ZABORT_LOG_SIZE + ZNEXT_LOG_SIZE) >
+ logFilePtr.p->remainingWordsInMbyte) {
+ jam();
+ changeMbyte(signal);
+ }//if
+ logFilePtr.p->remainingWordsInMbyte =
+ logFilePtr.p->remainingWordsInMbyte - ZABORT_LOG_SIZE;
+ writeLogWord(signal, ZABORT_TYPE);
+ writeLogWord(signal, tcConnectptr.p->transid[0]);
+ writeLogWord(signal, tcConnectptr.p->transid[1]);
+}//Dblqh::writeAbortLog()
+
+/* --------------------------------------------------------------------------
+ * ------- WRITE A COMMIT LOG RECORD -------
+ *
+ * SUBROUTINE SHORT NAME: WCL
+ * ------------------------------------------------------------------------- */
+void Dblqh::writeCommitLog(Signal* signal, LogPartRecordPtr regLogPartPtr)
+{
+ LogFileRecordPtr regLogFilePtr;
+ LogPageRecordPtr regLogPagePtr;
+ TcConnectionrec * const regTcPtr = tcConnectptr.p;
+ regLogFilePtr.i = regLogPartPtr.p->currentLogfile;
+ ptrCheckGuard(regLogFilePtr, clogFileFileSize, logFileRecord);
+ regLogPagePtr.i = regLogFilePtr.p->currentLogpage;
+ Uint32 twclTmp = regLogFilePtr.p->remainingWordsInMbyte;
+ ptrCheckGuard(regLogPagePtr, clogPageFileSize, logPageRecord);
+ logPartPtr = regLogPartPtr;
+ logFilePtr = regLogFilePtr;
+ logPagePtr = regLogPagePtr;
+ if ((ZCOMMIT_LOG_SIZE + ZNEXT_LOG_SIZE) > twclTmp) {
+ jam();
+ changeMbyte(signal);
+ twclTmp = logFilePtr.p->remainingWordsInMbyte;
+ }//if
+
+ Uint32 twclLogPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ Uint32 tableId = regTcPtr->tableref;
+ Uint32 schemaVersion = regTcPtr->schemaVersion;
+ Uint32 fragId = regTcPtr->fragmentid;
+ Uint32 fileNo = regTcPtr->logStartFileNo;
+ Uint32 startPageNo = regTcPtr->logStartPageNo;
+ Uint32 pageIndex = regTcPtr->logStartPageIndex;
+ Uint32 stopPageNo = regTcPtr->logStopPageNo;
+ Uint32 gci = regTcPtr->gci;
+ logFilePtr.p->remainingWordsInMbyte = twclTmp - ZCOMMIT_LOG_SIZE;
+
+ if ((twclLogPos + ZCOMMIT_LOG_SIZE) >= ZPAGE_SIZE) {
+ writeLogWord(signal, ZCOMMIT_TYPE);
+ writeLogWord(signal, tableId);
+ writeLogWord(signal, schemaVersion);
+ writeLogWord(signal, fragId);
+ writeLogWord(signal, fileNo);
+ writeLogWord(signal, startPageNo);
+ writeLogWord(signal, pageIndex);
+ writeLogWord(signal, stopPageNo);
+ writeLogWord(signal, gci);
+ } else {
+ Uint32* dataPtr = &logPagePtr.p->logPageWord[twclLogPos];
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = twclLogPos + ZCOMMIT_LOG_SIZE;
+ dataPtr[0] = ZCOMMIT_TYPE;
+ dataPtr[1] = tableId;
+ dataPtr[2] = schemaVersion;
+ dataPtr[3] = fragId;
+ dataPtr[4] = fileNo;
+ dataPtr[5] = startPageNo;
+ dataPtr[6] = pageIndex;
+ dataPtr[7] = stopPageNo;
+ dataPtr[8] = gci;
+ }//if
+ TcConnectionrecPtr rloTcNextConnectptr;
+ TcConnectionrecPtr rloTcPrevConnectptr;
+ rloTcPrevConnectptr.i = regTcPtr->prevLogTcrec;
+ rloTcNextConnectptr.i = regTcPtr->nextLogTcrec;
+ if (rloTcNextConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(rloTcNextConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ rloTcNextConnectptr.p->prevLogTcrec = rloTcPrevConnectptr.i;
+ } else {
+ regLogPartPtr.p->lastLogTcrec = rloTcPrevConnectptr.i;
+ }//if
+ if (rloTcPrevConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(rloTcPrevConnectptr, ctcConnectrecFileSize, tcConnectionrec);
+ rloTcPrevConnectptr.p->nextLogTcrec = rloTcNextConnectptr.i;
+ } else {
+ regLogPartPtr.p->firstLogTcrec = rloTcNextConnectptr.i;
+ }//if
+}//Dblqh::writeCommitLog()
+
+/* --------------------------------------------------------------------------
+ * ------- WRITE A COMPLETED GCI LOG RECORD -------
+ *
+ * SUBROUTINE SHORT NAME: WCG
+// Input Pointers:
+// logFilePtr
+// logPartPtr
+ * ------------------------------------------------------------------------- */
+void Dblqh::writeCompletedGciLog(Signal* signal)
+{
+ if ((ZCOMPLETED_GCI_LOG_SIZE + ZNEXT_LOG_SIZE) >
+ logFilePtr.p->remainingWordsInMbyte) {
+ jam();
+ changeMbyte(signal);
+ }//if
+ logFilePtr.p->remainingWordsInMbyte =
+ logFilePtr.p->remainingWordsInMbyte - ZCOMPLETED_GCI_LOG_SIZE;
+ writeLogWord(signal, ZCOMPLETED_GCI_TYPE);
+ writeLogWord(signal, cnewestCompletedGci);
+ logPartPtr.p->logPartNewestCompletedGCI = cnewestCompletedGci;
+}//Dblqh::writeCompletedGciLog()
+
+/* --------------------------------------------------------------------------
+ * ------- WRITE A DIRTY PAGE DURING LOG EXECUTION -------
+ *
+ * SUBROUTINE SHORT NAME: WD
+ * ------------------------------------------------------------------------- */
+void Dblqh::writeDirty(Signal* signal)
+{
+ logPagePtr.p->logPageWord[ZPOS_DIRTY] = ZNOT_DIRTY;
+
+ // Calculate checksum for page
+ logPagePtr.p->logPageWord[ZPOS_CHECKSUM] = calcPageCheckSum(logPagePtr);
+
+ seizeLfo(signal);
+ initLfo(signal);
+ lfoPtr.p->lfoPageNo = logPartPtr.p->prevFilepage;
+ lfoPtr.p->noPagesRw = 1;
+ lfoPtr.p->lfoState = LogFileOperationRecord::WRITE_DIRTY;
+ lfoPtr.p->firstLfoPage = logPagePtr.i;
+ signal->theData[0] = logFilePtr.p->fileRef;
+ signal->theData[1] = cownref;
+ signal->theData[2] = lfoPtr.i;
+ signal->theData[3] = ZLIST_OF_PAIRS_SYNCH;
+ signal->theData[4] = ZVAR_NO_LOG_PAGE_WORD;
+ signal->theData[5] = 1;
+ signal->theData[6] = logPagePtr.i;
+ signal->theData[7] = logPartPtr.p->prevFilepage;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+}//Dblqh::writeDirty()
+
+/* --------------------------------------------------------------------------
+ * ------- WRITE A WORD INTO THE LOG, CHECK FOR NEW PAGE -------
+ *
+ * SUBROUTINE SHORT NAME: WLW
+ * ------------------------------------------------------------------------- */
+void Dblqh::writeLogWord(Signal* signal, Uint32 data)
+{
+ Uint32 logPos = logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX];
+ ndbrequire(logPos < ZPAGE_SIZE);
+ logPagePtr.p->logPageWord[logPos] = data;
+ logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] = logPos + 1;
+ if ((logPos + 1) == ZPAGE_SIZE) {
+ jam();
+ completedLogPage(signal, ZNORMAL);
+ seizeLogpage(signal);
+ initLogpage(signal);
+ logFilePtr.p->currentLogpage = logPagePtr.i;
+ logFilePtr.p->currentFilepage++;
+ }//if
+}//Dblqh::writeLogWord()
+
+/* --------------------------------------------------------------------------
+ * ------- WRITE A NEXT LOG RECORD AND CHANGE TO NEXT MBYTE -------
+ *
+ * SUBROUTINE SHORT NAME: WNL
+// Input Pointers:
+// logFilePtr(Redefines)
+// logPagePtr (Redefines)
+// logPartPtr
+ * ------------------------------------------------------------------------- */
+void Dblqh::writeNextLog(Signal* signal)
+{
+ LogFileRecordPtr wnlNextLogFilePtr;
+ UintR twnlNextFileNo;
+ UintR twnlNewMbyte;
+ UintR twnlRemWords;
+ UintR twnlNextMbyte;
+
+/* -------------------------------------------------- */
+/* CALCULATE THE NEW NUMBER OF REMAINING WORDS */
+/* AS 128*2036 WHERE 128 * 8 KBYTE = 1 MBYTE */
+/* AND 2036 IS THE NUMBER OF WORDS IN A PAGE */
+/* THAT IS USED FOR LOG INFORMATION. */
+/* -------------------------------------------------- */
+ twnlRemWords = ZPAGE_SIZE - ZPAGE_HEADER_SIZE;
+ twnlRemWords = twnlRemWords * ZPAGES_IN_MBYTE;
+ wnlNextLogFilePtr.i = logFilePtr.p->nextLogFile;
+ ptrCheckGuard(wnlNextLogFilePtr, clogFileFileSize, logFileRecord);
+/* -------------------------------------------------- */
+/* WRITE THE NEXT LOG RECORD. */
+/* -------------------------------------------------- */
+ ndbrequire(logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX] < ZPAGE_SIZE);
+ logPagePtr.p->logPageWord[logPagePtr.p->logPageWord[ZCURR_PAGE_INDEX]] =
+ ZNEXT_MBYTE_TYPE;
+ if (logFilePtr.p->currentMbyte == (ZNO_MBYTES_IN_FILE - 1)) {
+ jam();
+/* -------------------------------------------------- */
+/* CALCULATE THE NEW REMAINING WORDS WHEN */
+/* CHANGING LOG FILE IS PERFORMED */
+/* -------------------------------------------------- */
+ twnlRemWords = twnlRemWords - (ZPAGE_SIZE - ZPAGE_HEADER_SIZE);
+/* -------------------------------------------------- */
+/* ENSURE THAT THE LOG PAGES ARE WRITTEN AFTER */
+/* WE HAVE CHANGED MBYTE. */
+/* -------------------------------------------------- */
+/* ENSURE LAST PAGE IN PREVIOUS MBYTE IS */
+/* WRITTEN AND THAT THE STATE OF THE WRITE IS */
+/* PROPERLY SET. */
+/* -------------------------------------------------- */
+/* WE HAVE TO CHANGE LOG FILE */
+/* -------------------------------------------------- */
+ completedLogPage(signal, ZLAST_WRITE_IN_FILE);
+ if (wnlNextLogFilePtr.p->fileNo == 0) {
+ jam();
+/* -------------------------------------------------- */
+/* WE HAVE FINALISED A LOG LAP, START FROM LOG */
+/* FILE 0 AGAIN */
+/* -------------------------------------------------- */
+ logPartPtr.p->logLap++;
+ }//if
+ logPartPtr.p->currentLogfile = wnlNextLogFilePtr.i;
+ logFilePtr.i = wnlNextLogFilePtr.i;
+ logFilePtr.p = wnlNextLogFilePtr.p;
+ twnlNewMbyte = 0;
+ } else {
+ jam();
+/* -------------------------------------------------- */
+/* INCREMENT THE CURRENT MBYTE */
+/* SET PAGE INDEX TO PAGE HEADER SIZE */
+/* -------------------------------------------------- */
+ completedLogPage(signal, ZENFORCE_WRITE);
+ twnlNewMbyte = logFilePtr.p->currentMbyte + 1;
+ }//if
+/* -------------------------------------------------- */
+/* CHANGE TO NEW LOG FILE IF NECESSARY */
+/* UPDATE THE FILE POSITION TO THE NEW MBYTE */
+/* FOUND IN PAGE PART OF TNEXT_LOG_PTR */
+/* ALLOCATE AND INITIATE A NEW PAGE SINCE WE */
+/* HAVE SENT THE PREVIOUS PAGE TO DISK. */
+/* SET THE NEW NUMBER OF REMAINING WORDS IN THE */
+/* NEW MBYTE ALLOCATED. */
+/* -------------------------------------------------- */
+ logFilePtr.p->currentMbyte = twnlNewMbyte;
+ logFilePtr.p->filePosition = twnlNewMbyte * ZPAGES_IN_MBYTE;
+ logFilePtr.p->currentFilepage = twnlNewMbyte * ZPAGES_IN_MBYTE;
+ logFilePtr.p->remainingWordsInMbyte = twnlRemWords;
+ seizeLogpage(signal);
+ if (logFilePtr.p->currentMbyte == 0) {
+ jam();
+ logFilePtr.p->lastPageWritten = 0;
+ if (logFilePtr.p->fileNo == 0) {
+ jam();
+ releaseLogpage(signal);
+ logPagePtr.i = logFilePtr.p->logPageZero;
+ ptrCheckGuard(logPagePtr, clogPageFileSize, logPageRecord);
+ }//if
+ }//if
+ initLogpage(signal);
+ logFilePtr.p->currentLogpage = logPagePtr.i;
+ if (logFilePtr.p->currentMbyte == 0) {
+ jam();
+/* -------------------------------------------------- */
+/* THIS IS A NEW FILE, WRITE THE FILE DESCRIPTOR*/
+/* ALSO OPEN THE NEXT LOG FILE TO ENSURE THAT */
+/* THIS FILE IS OPEN WHEN ITS TURN COMES. */
+/* -------------------------------------------------- */
+ writeFileHeaderOpen(signal, ZNORMAL);
+ openNextLogfile(signal);
+ logFilePtr.p->fileChangeState = LogFileRecord::BOTH_WRITES_ONGOING;
+ }//if
+ if (logFilePtr.p->fileNo == logPartPtr.p->logTailFileNo) {
+ if (logFilePtr.p->currentMbyte == logPartPtr.p->logTailMbyte) {
+ jam();
+/* -------------------------------------------------- */
+/* THE HEAD AND TAIL HAS MET. THIS SHOULD NEVER */
+/* OCCUR. CAN HAPPEN IF THE LOCAL CHECKPOINTS */
+/* TAKE FAR TOO LONG TIME. SO TIMING PROBLEMS */
+/* CAN INVOKE THIS SYSTEM CRASH. HOWEVER ONLY */
+/* VERY SERIOUS TIMING PROBLEMS. */
+/* -------------------------------------------------- */
+ systemError(signal);
+ }//if
+ }//if
+ if (logFilePtr.p->currentMbyte == (ZNO_MBYTES_IN_FILE - 1)) {
+ jam();
+ twnlNextMbyte = 0;
+ if (logFilePtr.p->fileChangeState != LogFileRecord::NOT_ONGOING) {
+ jam();
+ logPartPtr.p->logPartState = LogPartRecord::FILE_CHANGE_PROBLEM;
+ }//if
+ twnlNextFileNo = wnlNextLogFilePtr.p->fileNo;
+ } else {
+ jam();
+ twnlNextMbyte = logFilePtr.p->currentMbyte + 1;
+ twnlNextFileNo = logFilePtr.p->fileNo;
+ }//if
+ if (twnlNextFileNo == logPartPtr.p->logTailFileNo) {
+ if (logPartPtr.p->logTailMbyte == twnlNextMbyte) {
+ jam();
+/* -------------------------------------------------- */
+/* THE NEXT MBYTE WILL BE THE TAIL. WE MUST */
+/* STOP LOGGING NEW OPERATIONS. THIS OPERATION */
+/* ALLOWED TO PASS. ALSO COMMIT, NEXT, COMPLETED*/
+/* GCI, ABORT AND FRAGMENT SPLIT IS ALLOWED. */
+/* OPERATIONS ARE ALLOWED AGAIN WHEN THE TAIL */
+/* IS MOVED FORWARD AS A RESULT OF A START_LCP */
+/* _ROUND SIGNAL ARRIVING FROM DBDIH. */
+/* -------------------------------------------------- */
+ logPartPtr.p->logPartState = LogPartRecord::TAIL_PROBLEM;
+ }//if
+ }//if
+}//Dblqh::writeNextLog()
+
+void
+Dblqh::execDUMP_STATE_ORD(Signal* signal)
+{
+ DumpStateOrd * const dumpState = (DumpStateOrd *)&signal->theData[0];
+ if(dumpState->args[0] == DumpStateOrd::CommitAckMarkersSize){
+ infoEvent("LQH: m_commitAckMarkerPool: %d free size: %d",
+ m_commitAckMarkerPool.getNoOfFree(),
+ m_commitAckMarkerPool.getSize());
+ }
+ if(dumpState->args[0] == DumpStateOrd::CommitAckMarkersDump){
+ infoEvent("LQH: m_commitAckMarkerPool: %d free size: %d",
+ m_commitAckMarkerPool.getNoOfFree(),
+ m_commitAckMarkerPool.getSize());
+
+ CommitAckMarkerIterator iter;
+ for(m_commitAckMarkerHash.first(iter); iter.curr.i != RNIL;
+ m_commitAckMarkerHash.next(iter)){
+ infoEvent("CommitAckMarker: i = %d (0x%x, 0x%x)"
+ " ApiRef: 0x%x apiOprec: 0x%x TcNodeId: %d",
+ iter.curr.i,
+ iter.curr.p->transid1,
+ iter.curr.p->transid2,
+ iter.curr.p->apiRef,
+ iter.curr.p->apiOprec,
+ iter.curr.p->tcNodeId);
+ }
+ }
+
+ // Dump info about number of log pages
+ if(dumpState->args[0] == DumpStateOrd::LqhDumpNoLogPages){
+ infoEvent("LQH: Log pages : %d Free: %d",
+ clogPageFileSize,
+ cnoOfLogPages);
+ }
+
+ // Dump all defined tables that LQH knowns about
+ if(dumpState->args[0] == DumpStateOrd::LqhDumpAllDefinedTabs){
+ for(Uint32 i = 0; i<ctabrecFileSize; i++){
+ TablerecPtr tabPtr;
+ tabPtr.i = i;
+ ptrAss(tabPtr, tablerec);
+ if(tabPtr.p->tableStatus != Tablerec::NOT_DEFINED){
+ infoEvent("Table %d Status: %d Usage: %d",
+ i, tabPtr.p->tableStatus, tabPtr.p->usageCount);
+ }
+ }
+ return;
+ }
+
+ // Dump all ScanRecords
+ if (dumpState->args[0] == DumpStateOrd::LqhDumpAllScanRec){
+ Uint32 recordNo = 0;
+ if (signal->length() == 1)
+ infoEvent("LQH: Dump all ScanRecords - size: %d",
+ cscanrecFileSize);
+ else if (signal->length() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ dumpState->args[0] = DumpStateOrd::LqhDumpOneScanRec;
+ dumpState->args[1] = recordNo;
+ execDUMP_STATE_ORD(signal);
+
+ if (recordNo < cscanrecFileSize-1){
+ dumpState->args[0] = DumpStateOrd::LqhDumpAllScanRec;
+ dumpState->args[1] = recordNo+1;
+ sendSignal(reference(), GSN_DUMP_STATE_ORD, signal, 2, JBB);
+ }
+ return;
+ }
+
+ // Dump all active ScanRecords
+ if (dumpState->args[0] == DumpStateOrd::LqhDumpAllActiveScanRec){
+ Uint32 recordNo = 0;
+ if (signal->length() == 1)
+ infoEvent("LQH: Dump active ScanRecord - size: %d",
+ cscanrecFileSize);
+ else if (signal->length() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ ScanRecordPtr sp;
+ sp.i = recordNo;
+ ptrAss(sp, scanRecord);
+ if (sp.p->scanState != ScanRecord::SCAN_FREE){
+ dumpState->args[0] = DumpStateOrd::LqhDumpOneScanRec;
+ dumpState->args[1] = recordNo;
+ execDUMP_STATE_ORD(signal);
+ }
+
+ if (recordNo < cscanrecFileSize-1){
+ dumpState->args[0] = DumpStateOrd::LqhDumpAllActiveScanRec;
+ dumpState->args[1] = recordNo+1;
+ sendSignal(reference(), GSN_DUMP_STATE_ORD, signal, 2, JBB);
+ }
+ return;
+ }
+
+ if(dumpState->args[0] == DumpStateOrd::LqhDumpOneScanRec){
+ Uint32 recordNo = RNIL;
+ if (signal->length() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ if (recordNo >= cscanrecFileSize)
+ return;
+
+ ScanRecordPtr sp;
+ sp.i = recordNo;
+ ptrAss(sp, scanRecord);
+ infoEvent("Dblqh::ScanRecord[%d]: state=%d, type=%d, "
+ "complStatus=%d, scanNodeId=%d",
+ sp.i,
+ sp.p->scanState,
+ sp.p->scanType,
+ sp.p->scanCompletedStatus,
+ sp.p->scanNodeId);
+ infoEvent(" apiBref=0x%x, scanAccPtr=%d",
+ sp.p->scanApiBlockref,
+ sp.p->scanAccPtr);
+ infoEvent(" copyptr=%d, ailen=%d, complOps=%d, concurrOps=%d",
+ sp.p->copyPtr,
+ sp.p->scanAiLength,
+ sp.p->scanCompletedOperations,
+ sp.p->scanConcurrentOperations);
+ infoEvent(" errCnt=%d, localFid=%d, schV=%d, searcCondFalseC=%d",
+ sp.p->scanErrorCounter,
+ sp.p->scanLocalFragid,
+ sp.p->scanSchemaVersion,
+ sp.p->scanSearchCondFalseCount);
+ infoEvent(" stpid=%d, flag=%d, lhold=%d, lmode=%d, num=%d",
+ sp.p->scanStoredProcId,
+ sp.p->scanFlag,
+ sp.p->scanLockHold,
+ sp.p->scanLockMode,
+ sp.p->scanNumber);
+ infoEvent(" relCount=%d, TCwait=%d, TCRec=%d, KIflag=%d",
+ sp.p->scanReleaseCounter,
+ sp.p->scanTcWaiting,
+ sp.p->scanTcrec,
+ sp.p->scanKeyinfoFlag);
+ infoEvent(" next=%d",
+ sp.p->nextScanrec);
+ return;
+ }
+ if(dumpState->args[0] == DumpStateOrd::LqhDumpLcpState){
+
+ infoEvent("== LQH LCP STATE ==");
+ infoEvent(" clcpCompletedState=%d, c_lcpId=%d, cnoOfFragsCheckpointed=%d",
+ clcpCompletedState,
+ c_lcpId,
+ cnoOfFragsCheckpointed);
+
+ LcpRecordPtr TlcpPtr;
+ // Print information about the current local checkpoint
+ TlcpPtr.i = 0;
+ ptrAss(TlcpPtr, lcpRecord);
+ infoEvent(" lcpState=%d firstLcpLocTup=%d firstLcpLocAcc=%d",
+ TlcpPtr.p->lcpState,
+ TlcpPtr.p->firstLcpLocTup,
+ TlcpPtr.p->firstLcpLocAcc);
+ infoEvent(" lcpAccptr=%d lastFragmentFlag=%d",
+ TlcpPtr.p->lcpAccptr,
+ TlcpPtr.p->lastFragmentFlag);
+ infoEvent("currentFragment.fragPtrI=%d",
+ TlcpPtr.p->currentFragment.fragPtrI);
+ infoEvent("currentFragment.lcpFragOrd.tableId=%d",
+ TlcpPtr.p->currentFragment.lcpFragOrd.tableId);
+ infoEvent(" lcpQueued=%d reportEmpty=%d",
+ TlcpPtr.p->lcpQueued,
+ TlcpPtr.p->reportEmpty);
+ char buf[TlcpPtr.p->m_EMPTY_LCP_REQ.TextLength+1];
+ infoEvent(" m_EMPTY_LCP_REQ=%d",
+ TlcpPtr.p->m_EMPTY_LCP_REQ.getText(buf));
+
+ return;
+ }
+
+
+
+}//Dblqh::execDUMP_STATE_ORD()
+
+void Dblqh::execSET_VAR_REQ(Signal* signal)
+{
+
+ SetVarReq* const setVarReq = (SetVarReq*)&signal->theData[0];
+ ConfigParamId var = setVarReq->variable();
+
+ switch (var) {
+
+ case NoOfConcurrentCheckpointsAfterRestart:
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ case NoOfConcurrentCheckpointsDuringRestart:
+ // Valid only during start so value not set.
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ default:
+ sendSignal(CMVMI_REF, GSN_SET_VAR_REF, signal, 1, JBB);
+ } // switch
+
+}//execSET_VAR_REQ()
+
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* ---------------------- TRIGGER HANDLING ------------------------ */
+/* ---------------------------------------------------------------- */
+/* */
+/* All trigger signals from TRIX are forwarded top TUP */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+// Trigger signals
+void
+Dblqh::execCREATE_TRIG_REQ(Signal* signal)
+{
+ jamEntry();
+ NodeId myNodeId = getOwnNodeId();
+ BlockReference tupref = calcTupBlockRef(myNodeId);
+
+ sendSignal(tupref, GSN_CREATE_TRIG_REQ, signal, CreateTrigReq::SignalLength, JBB);
+}
+
+void
+Dblqh::execCREATE_TRIG_CONF(Signal* signal)
+{
+ jamEntry();
+ NodeId myNodeId = getOwnNodeId();
+ BlockReference dictref = calcDictBlockRef(myNodeId);
+
+ sendSignal(dictref, GSN_CREATE_TRIG_CONF, signal, CreateTrigConf::SignalLength, JBB);
+}
+
+void
+Dblqh::execCREATE_TRIG_REF(Signal* signal)
+{
+ jamEntry();
+ NodeId myNodeId = getOwnNodeId();
+ BlockReference dictref = calcDictBlockRef(myNodeId);
+
+ sendSignal(dictref, GSN_CREATE_TRIG_REF, signal, CreateTrigRef::SignalLength, JBB);
+}
+
+void
+Dblqh::execDROP_TRIG_REQ(Signal* signal)
+{
+ jamEntry();
+ NodeId myNodeId = getOwnNodeId();
+ BlockReference tupref = calcTupBlockRef(myNodeId);
+
+ sendSignal(tupref, GSN_DROP_TRIG_REQ, signal, DropTrigReq::SignalLength, JBB);
+}
+
+void
+Dblqh::execDROP_TRIG_CONF(Signal* signal)
+{
+ jamEntry();
+ NodeId myNodeId = getOwnNodeId();
+ BlockReference dictref = calcDictBlockRef(myNodeId);
+
+ sendSignal(dictref, GSN_DROP_TRIG_CONF, signal, DropTrigConf::SignalLength, JBB);
+}
+
+void
+Dblqh::execDROP_TRIG_REF(Signal* signal)
+{
+ jamEntry();
+ NodeId myNodeId = getOwnNodeId();
+ BlockReference dictref = calcDictBlockRef(myNodeId);
+
+ sendSignal(dictref, GSN_DROP_TRIG_REF, signal, DropTrigRef::SignalLength, JBB);
+}
+
+Uint32 Dblqh::calcPageCheckSum(LogPageRecordPtr logP){
+ Uint32 checkSum = 37;
+#ifdef VM_TRACE
+ for (Uint32 i = (ZPOS_CHECKSUM+1); i<ZPAGE_SIZE; i++)
+ checkSum = logP.p->logPageWord[i] ^ checkSum;
+#endif
+ return checkSum;
+ }
+
diff --git a/ndb/src/kernel/blocks/dblqh/Makefile b/ndb/src/kernel/blocks/dblqh/Makefile
new file mode 100644
index 00000000000..520486d8058
--- /dev/null
+++ b/ndb/src/kernel/blocks/dblqh/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := dblqh
+DIRS := redoLogReader
+
+SOURCES = \
+ DblqhInit.cpp \
+ DblqhMain.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dblqh/redoLogReader/Makefile b/ndb/src/kernel/blocks/dblqh/redoLogReader/Makefile
new file mode 100644
index 00000000000..a89b648de77
--- /dev/null
+++ b/ndb/src/kernel/blocks/dblqh/redoLogReader/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+BIN_TARGET := redoLogFileReader
+
+SOURCES := records.cpp redoLogFileReader.cpp
+
+TYPE := util
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dblqh/redoLogReader/records.cpp b/ndb/src/kernel/blocks/dblqh/redoLogReader/records.cpp
new file mode 100644
index 00000000000..092b7840c20
--- /dev/null
+++ b/ndb/src/kernel/blocks/dblqh/redoLogReader/records.cpp
@@ -0,0 +1,312 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "records.hpp"
+
+void printOut(const char *string, Uint32 value) {
+ ndbout_c("%-30s%-12u%-12x", string, value, value);
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+bool AbortTransactionRecord::check() {
+ // Not implemented yet.
+ return true;
+}
+
+Uint32 AbortTransactionRecord::getLogRecordSize() {
+ return ABORTTRANSACTIONRECORDSIZE;
+}
+
+NdbOut& operator<<(NdbOut& no, const AbortTransactionRecord& atr) {
+ no << "----------ABORT TRANSACTION RECORD-------------" << endl << endl;
+ printOut("Record type:", atr.m_recordType);
+ printOut("TransactionId1:", atr.m_transactionId1);
+ printOut("TransactionId2:", atr.m_transactionId2);
+ no << endl;
+ return no;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+bool NextMbyteRecord::check() {
+ // Not implemented yet.
+ return true;
+}
+
+Uint32 NextMbyteRecord::getLogRecordSize() {
+ return NEXTMBYTERECORDSIZE;
+}
+
+NdbOut& operator<<(NdbOut& no, const NextMbyteRecord& nmr) {
+ no << "----------NEXT MBYTE RECORD--------------------" << endl << endl;
+ printOut("Record type:", nmr.m_recordType);
+ no << endl;
+ return no;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+bool CommitTransactionRecord::check() {
+ // Not implemented yet.
+ return true;
+}
+
+Uint32 CommitTransactionRecord::getLogRecordSize() {
+ return COMMITTRANSACTIONRECORDSIZE;
+}
+
+NdbOut& operator<<(NdbOut& no, const CommitTransactionRecord& ctr) {
+ no << "----------COMMIT TRANSACTION RECORD------------" << endl << endl;
+ printOut("Record type:", ctr.m_recordType);
+ printOut("TableId", ctr.m_tableId);
+ printOut("FfragmentId", ctr.m_fragmentId);
+ printOut("File no. of Prep. Op.", ctr.m_fileNumberOfPrepareOperation);
+ printOut("Start page no. of Prep. Op.", ctr.m_startPageNumberOfPrepareOperation);
+ printOut("Start page index of Prep. Op.", ctr.m_startPageIndexOfPrepareOperation);
+ printOut("Stop page no. of Prep. Op.", ctr.m_stopPageNumberOfPrepareOperation);
+ printOut("GlobalCheckpoint", ctr.m_globalCheckpoint);
+
+ no << endl;
+ return no;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+bool InvalidCommitTransactionRecord::check() {
+ // Not implemented yet.
+ return true;
+}
+
+Uint32 InvalidCommitTransactionRecord::getLogRecordSize() {
+ return COMMITTRANSACTIONRECORDSIZE;
+}
+
+NdbOut& operator<<(NdbOut& no, const InvalidCommitTransactionRecord& ictr) {
+ no << "------INVALID COMMIT TRANSACTION RECORD--------" << endl << endl;
+ printOut("Record type:", ictr.m_recordType);
+ printOut("TableId", ictr.m_tableId);
+ printOut("FfragmentId", ictr.m_fragmentId);
+ printOut("File no. of Prep. Op.", ictr.m_fileNumberOfPrepareOperation);
+ printOut("Start page no. of Prep. Op.", ictr.m_startPageNumberOfPrepareOperation);
+ printOut("Start page index of Prep. Op.", ictr.m_startPageIndexOfPrepareOperation);
+ printOut("Stop page no. of Prep. Op.", ictr.m_stopPageNumberOfPrepareOperation);
+ printOut("GlobalCheckpoint", ictr.m_globalCheckpoint);
+
+ no << endl;
+ return no;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+bool PrepareOperationRecord::check() {
+ // Not fully implemented.
+ if (m_operationType == 3 && m_attributeLength != 0)
+ return false;
+
+ if (m_logRecordSize != (m_attributeLength + m_keyLength + 7))
+ return false;
+
+ return true;
+}
+
+Uint32 PrepareOperationRecord::getLogRecordSize() {
+ return m_logRecordSize;
+}
+
+NdbOut& operator<<(NdbOut& no, const PrepareOperationRecord& por) {
+ no << "-----------PREPARE OPERATION RECORD------------" << endl << endl;
+ printOut("Record type:", por.m_recordType);
+ printOut("logRecordSize:", por.m_logRecordSize);
+ printOut("hashValue:", por.m_hashValue);
+ printOut("schemaVersion:", por.m_schemaVersion);
+ switch (por.m_operationType) {
+ case 0:
+ ndbout_c("%-30s%-12u%-6s", "operationType:",
+ por.m_operationType, "read");
+ break;
+ case 1:
+ ndbout_c("%-30s%-12u%-6s", "operationType:",
+ por.m_operationType, "update");
+ break;
+ case 2:
+ ndbout_c("%-30s%-12u%-6s", "operationType:",
+ por.m_operationType, "insert");
+ break;
+ case 3:
+ ndbout_c("%-30s%-12u%-6s", "operationType:",
+ por.m_operationType, "delete");
+ break;
+ default:
+ printOut("operationType:", por.m_operationType);
+ }
+ printOut("attributeLength:", por.m_attributeLength);
+ printOut("keyLength:", por.m_keyLength);
+
+#if 1
+ // Print keydata
+ Uint32* p = (Uint32*)&por.m_keyInfo;
+ for(Uint32 i=0; i < por.m_keyLength; i++){
+ printOut("keydata:", *p);
+ p++;
+ }
+
+ // Print attrdata
+ for(Uint32 i=0; i < por.m_attributeLength; i++){
+ printOut("attrdata:", *p);
+ p++;
+ }
+#endif
+
+ no << endl;
+ return no;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+bool CompletedGCIRecord::check() {
+ // Not implemented yet.
+ return true;
+}
+
+Uint32 CompletedGCIRecord::getLogRecordSize() {
+ return COMPLETEDGCIRECORDSIZE;
+}
+
+NdbOut& operator<<(NdbOut& no, const CompletedGCIRecord& cGCIr) {
+ no << "-----------COMPLETED GCI RECORD----------------" << endl << endl;
+ printOut("Record type:", cGCIr.m_recordType);
+ printOut("Completed GCI:", cGCIr.m_theCompletedGCI);
+ no << endl;
+ return no;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+bool NextLogRecord::check() {
+ // Not implemented yet.
+ return true;
+}
+
+Uint32 NextLogRecord::getLogRecordSize(Uint32 pageIndex) {
+ return PAGESIZE - pageIndex;
+}
+
+NdbOut& operator<<(NdbOut& no, const NextLogRecord& nl) {
+ no << "-----------NEXT LOG RECORD --------------------" << endl << endl;
+ printOut("Record type:", nl.m_recordType);
+ no << endl;
+ return no;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+Uint32 PageHeader::getLogRecordSize() {
+ return PAGEHEADERSIZE;
+}
+
+bool PageHeader::check() {
+ // Not implemented yet.
+ return true;
+}
+
+NdbOut& operator<<(NdbOut& no, const PageHeader& ph) {
+ no << "------------PAGE HEADER------------------------" << endl << endl;
+ ndbout_c("%-30s%-12s%-12s\n", "", "Decimal", "Hex");
+ printOut("Checksum:", ph.m_checksum);
+ printOut("Laps since initial start:", ph.m_lap);
+ printOut("Max gci completed:", ph.m_max_gci_completed);
+ printOut("Max gci started:", ph.m_max_gci_started);
+ printOut("Ptr to next page:", ph.m_next_page);
+ printOut("Ptr to previous page:", ph.m_previous_page);
+ printOut("Ndb version:", ph.m_ndb_version);
+ printOut("Number of log files:", ph.m_number_of_logfiles);
+ printOut("Current page index:", ph.m_current_page_index);
+ printOut("Oldest prepare op. file No.:", ph.m_old_prepare_file_number);
+ printOut("Oldest prepare op. page ref.:", ph.m_old_prepare_page_reference);
+ printOut("Dirty flag:", ph.m_dirty_flag);
+ no << endl;
+ return no;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+Uint32 FileDescriptor::getLogRecordSize() {
+ return FILEDESCRIPTORHEADERSIZE
+ + m_fdHeader.m_noOfDescriptors * FILEDESCRIPTORRECORDSIZE;
+}
+
+NdbOut& operator<<(NdbOut& no, const FileDescriptor& fd) {
+ no << "-------FILE DESCRIPTOR HEADER------------------" << endl << endl;
+ printOut("Record type:", fd.m_fdHeader.m_recordType);
+ printOut("Number of file descriptors:", fd.m_fdHeader.m_noOfDescriptors);
+ printOut("File number:", fd.m_fdHeader.m_fileNo);
+ ndbout << endl;
+ for(Uint32 i = 0; i < fd.m_fdHeader.m_noOfDescriptors; i++) {
+ fd.printARecord(i);
+ }
+ return no;
+}
+
+void FileDescriptor::printARecord( Uint32 recordIndex ) const {
+ ndbout << "------------------FILE DESCRIPTOR " << recordIndex
+ <<" ---------------------" << endl << endl;
+ ndbout_c("%-30s%-12s%-12s\n", "", "Decimal", "Hex");
+
+ for(int i = 1; i <= NO_MBYTE_IN_FILE; i++) {
+ ndbout_c("%s%2d%s%-12u%-12x", "Max GCI completed, mbyte ", i, ": ",
+ m_fdRecord[recordIndex].m_maxGciCompleted[i-1],
+ m_fdRecord[recordIndex].m_maxGciCompleted[i-1]);
+ }
+ for(int i = 1; i <= NO_MBYTE_IN_FILE; i++) {
+ ndbout_c("%s%2d%s%-12u%-12x", "Max GCI started, mbyte ", i, ": ",
+ m_fdRecord[recordIndex].m_maxGciStarted[i-1],
+ m_fdRecord[recordIndex].m_maxGciStarted[i-1]);
+ }
+ for(int i = 1; i <= NO_MBYTE_IN_FILE; i++) {
+ ndbout_c("%s%2d%s%-12u%-12x", "Last prepared ref, mbyte ", i, ": ",
+ m_fdRecord[recordIndex].m_lastPreparedReference[i-1],
+ m_fdRecord[recordIndex].m_lastPreparedReference[i-1]);
+ }
+ ndbout << endl;
+}
+
+bool FileDescriptor::check() {
+ // Not implemented yet.
+ return true;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
diff --git a/ndb/src/kernel/blocks/dblqh/redoLogReader/records.hpp b/ndb/src/kernel/blocks/dblqh/redoLogReader/records.hpp
new file mode 100644
index 00000000000..e73986e4d73
--- /dev/null
+++ b/ndb/src/kernel/blocks/dblqh/redoLogReader/records.hpp
@@ -0,0 +1,235 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <ndb_types.h>
+
+#define ZNEW_PREP_OP_TYPE 0
+#define ZPREP_OP_TYPE 1
+#define ZCOMMIT_TYPE 2
+#define ZABORT_TYPE 3
+#define ZFD_TYPE 4
+#define ZFRAG_SPLIT_TYPE 5
+#define ZNEXT_LOG_RECORD_TYPE 6
+#define ZNEXT_MBYTE_TYPE 7
+#define ZCOMPLETED_GCI_TYPE 8
+#define ZINVALID_COMMIT_TYPE 9
+
+#define MAX_FILE_DESCRIPTORS 40
+#define NO_MBYTE_IN_FILE 16
+
+#define PAGESIZE 8192
+#define NO_PAGES_IN_MBYTE 32
+#define NO_MBYTE_IN_FILE 16
+
+#define COMMITTRANSACTIONRECORDSIZE 8
+#define COMPLETEDGCIRECORDSIZE 2
+#define PAGEHEADERSIZE 32
+#define FILEDESCRIPTORHEADERSIZE 3
+#define FILEDESCRIPTORRECORDSIZE 48
+#define NEXTMBYTERECORDSIZE 1
+#define ABORTTRANSACTIONRECORDSIZE 3
+
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+class AbortTransactionRecord {
+ friend NdbOut& operator<<(NdbOut&, const AbortTransactionRecord&);
+public:
+ bool check();
+ Uint32 getLogRecordSize();
+protected:
+ Uint32 m_recordType;
+ Uint32 m_transactionId1;
+ Uint32 m_transactionId2;
+};
+
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+class NextMbyteRecord {
+ friend NdbOut& operator<<(NdbOut&, const NextMbyteRecord&);
+public:
+ bool check();
+ Uint32 getLogRecordSize();
+protected:
+ Uint32 m_recordType;
+};
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+
+class PrepareOperationRecord {
+ friend NdbOut& operator<<(NdbOut&, const PrepareOperationRecord&);
+public:
+ bool check();
+ Uint32 getLogRecordSize();
+
+protected:
+ Uint32 m_recordType;
+ Uint32 m_logRecordSize;
+ Uint32 m_hashValue;
+ Uint32 m_schemaVersion;
+ Uint32 m_operationType; // 0 READ, 1 UPDATE, 2 INSERT, 3 DELETE
+ Uint32 m_attributeLength;
+ Uint32 m_keyLength;
+ Uint32 *m_keyInfo; // In this order
+ Uint32 *m_attrInfo;// In this order
+};
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+class CompletedGCIRecord {
+ friend NdbOut& operator<<(NdbOut&, const CompletedGCIRecord&);
+public:
+ bool check();
+ Uint32 getLogRecordSize();
+protected:
+ Uint32 m_recordType;
+ Uint32 m_theCompletedGCI;
+};
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+class NextLogRecord {
+ friend NdbOut& operator<<(NdbOut&, const NextLogRecord&);
+public:
+ bool check();
+ Uint32 getLogRecordSize(Uint32);
+protected:
+ Uint32 m_recordType;
+};
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+class PageHeader {
+ friend NdbOut& operator<<(NdbOut&, const PageHeader&);
+public:
+ bool check();
+ Uint32 getLogRecordSize();
+protected:
+ Uint32 m_checksum;
+ Uint32 m_lap;
+ Uint32 m_max_gci_completed;
+ Uint32 m_max_gci_started;
+ Uint32 m_next_page;
+ Uint32 m_previous_page;
+ Uint32 m_ndb_version;
+ Uint32 m_number_of_logfiles;
+ Uint32 m_current_page_index;
+ Uint32 m_old_prepare_file_number;
+ Uint32 m_old_prepare_page_reference;
+ Uint32 m_dirty_flag;
+};
+
+//----------------------------------------------------------------
+// File descriptor.
+//----------------------------------------------------------------
+
+class FileDescriptorHeader {
+public:
+ Uint32 m_recordType;
+ Uint32 m_noOfDescriptors;
+ Uint32 m_fileNo;
+};
+
+class FileDescriptorRecord {
+public:
+ Uint32 m_maxGciCompleted[16];
+ Uint32 m_maxGciStarted[16];
+ Uint32 m_lastPreparedReference[16];
+};
+
+class FileDescriptor {
+ friend NdbOut& operator<<(NdbOut&, const FileDescriptor&);
+public:
+ bool check();
+ Uint32 getLogRecordSize();
+protected:
+ void printARecord( Uint32 ) const;
+ FileDescriptorHeader m_fdHeader;
+ FileDescriptorRecord m_fdRecord[1];
+};
+
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+class CommitTransactionRecord {
+ friend NdbOut& operator<<(NdbOut&, const CommitTransactionRecord&);
+public:
+ bool check();
+ Uint32 getLogRecordSize();
+protected:
+ Uint32 m_recordType;
+ Uint32 m_tableId;
+ Uint32 m_fragmentId;
+ Uint32 m_fileNumberOfPrepareOperation;
+ Uint32 m_startPageNumberOfPrepareOperation;
+ Uint32 m_startPageIndexOfPrepareOperation;
+ Uint32 m_stopPageNumberOfPrepareOperation;
+ Uint32 m_globalCheckpoint;
+};
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+class InvalidCommitTransactionRecord {
+ friend NdbOut& operator<<(NdbOut&, const InvalidCommitTransactionRecord&);
+public:
+ bool check();
+ Uint32 getLogRecordSize();
+protected:
+ Uint32 m_recordType;
+ Uint32 m_tableId;
+ Uint32 m_fragmentId;
+ Uint32 m_fileNumberOfPrepareOperation;
+ Uint32 m_startPageNumberOfPrepareOperation;
+ Uint32 m_startPageIndexOfPrepareOperation;
+ Uint32 m_stopPageNumberOfPrepareOperation;
+ Uint32 m_globalCheckpoint;
+};
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+struct NextLogRec {
+
+};
+
+struct NewPrepareOperation {
+
+};
+
+struct FragmentSplit {
+
+};
diff --git a/ndb/src/kernel/blocks/dblqh/redoLogReader/redoLogFileReader.cpp b/ndb/src/kernel/blocks/dblqh/redoLogReader/redoLogFileReader.cpp
new file mode 100644
index 00000000000..d2d166fa03e
--- /dev/null
+++ b/ndb/src/kernel/blocks/dblqh/redoLogReader/redoLogFileReader.cpp
@@ -0,0 +1,465 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//----------------------------------------------------------------
+// REDOLOGFILEREADER
+// Reads a redo log file and checks it for errors and/or prints
+// the file in a human readable format.
+//
+// Usage: redoLogFileReader <file> [-noprint] [-nocheck]
+// [-mbyte <0-15>] [-mbyteHeaders] [-pageHeaders]
+//
+//----------------------------------------------------------------
+
+
+#include "records.hpp"
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <errno.h>
+#define RETURN_ERROR 1
+#define RETURN_OK 0
+
+#define FROM_BEGINNING 0
+
+void usage(const char * prg);
+Uint32 readRecordOverPageBoundary (Uint32 *, Uint32 , Uint32 , Uint32);
+Uint32 readFromFile(FILE * f, Uint32 *toPtr, Uint32 sizeInWords);
+void readArguments(int argc, const char** argv);
+void doExit();
+
+FILE * f;
+char fileName[256];
+bool thePrintFlag = true;
+bool theCheckFlag = true;
+bool onlyPageHeaders = false;
+bool onlyMbyteHeaders = false;
+bool onlyFileDesc = false;
+bool firstLap = true;
+Uint32 startAtMbyte = 0;
+Uint32 startAtPage = 0;
+Uint32 startAtPageIndex = 0;
+Uint32 *redoLogPage;
+
+NDB_COMMAND(redoLogFileReader, "redoLogFileReader", "redoLogFileReader", "Read a redo log file", 16384) {
+ Uint32 pageIndex = 0;
+ Uint32 oldPageIndex = 0;
+ Uint32 recordType = 1234567890;
+
+ PageHeader *thePageHeader;
+ CompletedGCIRecord *cGCIrecord;
+ PrepareOperationRecord *poRecord;
+ NextLogRecord *nlRecord;
+ FileDescriptor *fdRecord;
+ CommitTransactionRecord *ctRecord;
+ InvalidCommitTransactionRecord *ictRecord;
+ NextMbyteRecord *nmRecord;
+ AbortTransactionRecord *atRecord;
+
+ readArguments(argc, argv);
+
+ f = fopen(fileName, "rb");
+ if(!f){
+ perror("Error: open file");
+ exit(RETURN_ERROR);
+ }
+
+ Uint32 tmpFileOffset = startAtMbyte * PAGESIZE * NO_PAGES_IN_MBYTE * sizeof(Uint32);
+ if (fseek(f, tmpFileOffset, FROM_BEGINNING)) {
+ perror("Error: Move in file");
+ exit(RETURN_ERROR);
+ }
+
+ redoLogPage = new Uint32[PAGESIZE*NO_PAGES_IN_MBYTE];
+
+ // Loop for every mbyte.
+ for (Uint32 j = startAtMbyte; j < NO_MBYTE_IN_FILE; j++) {
+ readFromFile(f, redoLogPage, PAGESIZE*NO_PAGES_IN_MBYTE);
+
+ if (firstLap) {
+ pageIndex = startAtPageIndex;
+ firstLap = false;
+ } else
+ pageIndex = 0;
+
+ // Loop for every page.
+ for (int i = startAtPage; i < NO_PAGES_IN_MBYTE; i++) {
+
+ if (pageIndex == 0) {
+ thePageHeader = (PageHeader *) &redoLogPage[i*PAGESIZE];
+ // Print out mbyte number, page number and page index.
+ ndbout << j << ":" << i << ":" << pageIndex << endl
+ << " " << j*32 + i << ":" << pageIndex << " ";
+ if (thePrintFlag) ndbout << (*thePageHeader);
+ if (theCheckFlag) {
+ if(!thePageHeader->check()) {
+ doExit();
+ }
+
+ Uint32 checkSum = 37;
+ for (int ps = 1; ps < PAGESIZE; ps++)
+ checkSum = redoLogPage[i*PAGESIZE+ps] ^ checkSum;
+
+ if (checkSum != redoLogPage[i*PAGESIZE]){
+ ndbout << "WRONG CHECKSUM: checksum = " << redoLogPage[i*PAGESIZE]
+ << " expected = " << checkSum << endl;
+ doExit();
+ }
+ else
+ ndbout << "expected checksum: " << checkSum << endl;
+
+ }
+ pageIndex += thePageHeader->getLogRecordSize();
+ }
+
+ if (onlyMbyteHeaders) {
+ // Show only the first page header in every mbyte of the file.
+ break;
+ }
+
+ if (onlyPageHeaders) {
+ // Show only page headers. Continue with the next page in this for loop.
+ pageIndex = 0;
+ continue;
+ }
+
+ do {
+ // Print out mbyte number, page number and page index.
+ ndbout << j << ":" << i << ":" << pageIndex << endl
+ << " " << j*32 + i << ":" << pageIndex << " ";
+ recordType = redoLogPage[i*PAGESIZE + pageIndex];
+ switch(recordType) {
+ case ZFD_TYPE:
+ fdRecord = (FileDescriptor *) &redoLogPage[i*PAGESIZE + pageIndex];
+ if (thePrintFlag) ndbout << (*fdRecord);
+ if (theCheckFlag) {
+ if(!fdRecord->check()) {
+ doExit();
+ }
+ }
+ if (onlyFileDesc) {
+ delete [] redoLogPage;
+ exit(RETURN_OK);
+ }
+ pageIndex += fdRecord->getLogRecordSize();
+ break;
+
+ case ZNEXT_LOG_RECORD_TYPE:
+ nlRecord = (NextLogRecord *) (&redoLogPage[i*PAGESIZE] + pageIndex);
+ pageIndex += nlRecord->getLogRecordSize(pageIndex);
+ if (pageIndex <= PAGESIZE) {
+ if (thePrintFlag) ndbout << (*nlRecord);
+ if (theCheckFlag) {
+ if(!nlRecord->check()) {
+ doExit();
+ }
+ }
+ }
+ break;
+
+ case ZCOMPLETED_GCI_TYPE:
+ cGCIrecord = (CompletedGCIRecord *) &redoLogPage[i*PAGESIZE + pageIndex];
+ pageIndex += cGCIrecord->getLogRecordSize();
+ if (pageIndex <= PAGESIZE) {
+ if (thePrintFlag) ndbout << (*cGCIrecord);
+ if (theCheckFlag) {
+ if(!cGCIrecord->check()) {
+ doExit();
+ }
+ }
+ }
+ break;
+
+ case ZPREP_OP_TYPE:
+ poRecord = (PrepareOperationRecord *) &redoLogPage[i*PAGESIZE + pageIndex];
+ pageIndex += poRecord->getLogRecordSize();
+ if (pageIndex <= PAGESIZE) {
+ if (thePrintFlag) ndbout << (*poRecord);
+ if (theCheckFlag) {
+ if(!poRecord->check()) {
+ doExit();
+ }
+ }
+ }
+ else {
+ oldPageIndex = pageIndex - poRecord->getLogRecordSize();
+ }
+ break;
+
+ case ZCOMMIT_TYPE:
+ ctRecord = (CommitTransactionRecord *) &redoLogPage[i*PAGESIZE + pageIndex];
+ pageIndex += ctRecord->getLogRecordSize();
+ if (pageIndex <= PAGESIZE) {
+ if (thePrintFlag) ndbout << (*ctRecord);
+ if (theCheckFlag) {
+ if(!ctRecord->check()) {
+ doExit();
+ }
+ }
+ }
+ else {
+ oldPageIndex = pageIndex - ctRecord->getLogRecordSize();
+ }
+ break;
+
+ case ZINVALID_COMMIT_TYPE:
+ ictRecord = (InvalidCommitTransactionRecord *) &redoLogPage[i*PAGESIZE + pageIndex];
+ pageIndex += ictRecord->getLogRecordSize();
+ if (pageIndex <= PAGESIZE) {
+ if (thePrintFlag) ndbout << (*ictRecord);
+ if (theCheckFlag) {
+ if(!ictRecord->check()) {
+ doExit();
+ }
+ }
+ }
+ else {
+ oldPageIndex = pageIndex - ictRecord->getLogRecordSize();
+ }
+ break;
+
+ case ZNEXT_MBYTE_TYPE:
+ nmRecord = (NextMbyteRecord *) &redoLogPage[i*PAGESIZE + pageIndex];
+ if (thePrintFlag) ndbout << (*nmRecord);
+ i = NO_PAGES_IN_MBYTE;
+ break;
+
+ case ZABORT_TYPE:
+ atRecord = (AbortTransactionRecord *) &redoLogPage[i*PAGESIZE + pageIndex];
+ pageIndex += atRecord->getLogRecordSize();
+ if (pageIndex <= PAGESIZE) {
+ if (thePrintFlag) ndbout << (*atRecord);
+ if (theCheckFlag) {
+ if(!atRecord->check()) {
+ doExit();
+ }
+ }
+ }
+ break;
+
+ case ZNEW_PREP_OP_TYPE:
+ case ZFRAG_SPLIT_TYPE:
+ ndbout << endl << "Record type = " << recordType << " not implemented." << endl;
+ doExit();
+
+ default:
+ ndbout << " ------ERROR: UNKNOWN RECORD TYPE------" << endl;
+
+ // Print out remaining data in this page
+ for (int j = pageIndex; j < PAGESIZE; j++){
+ Uint32 unknown = redoLogPage[i*PAGESIZE + j];
+
+ ndbout_c("%-30d%-12u%-12x", j, unknown, unknown);
+ }
+
+ doExit();
+ }
+ } while(pageIndex < PAGESIZE && i < NO_PAGES_IN_MBYTE);
+
+ if (pageIndex > PAGESIZE) {
+ // The last record overlapped page boundary. Must redo that record.
+ pageIndex = readRecordOverPageBoundary(&redoLogPage[i*PAGESIZE],
+ pageIndex, oldPageIndex, recordType);
+ } else {
+ pageIndex = 0;
+ }
+ ndbout << endl;
+ }//for
+ ndbout << endl;
+ if (startAtMbyte != 0) {
+ break;
+ }
+ }//for
+ fclose(f);
+ delete [] redoLogPage;
+ exit(RETURN_OK);
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+Uint32 readFromFile(FILE * f, Uint32 *toPtr, Uint32 sizeInWords) {
+ Uint32 noOfReadWords;
+ if ( !(noOfReadWords = fread(toPtr, sizeof(Uint32), sizeInWords, f)) ) {
+ ndbout << "Error reading file" << endl;
+ doExit();
+ }
+
+ return noOfReadWords;
+}
+
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+Uint32 readRecordOverPageBoundary(Uint32 *pagePtr, Uint32 pageIndex, Uint32 oldPageIndex, Uint32 recordType) {
+ Uint32 pageHeader[PAGEHEADERSIZE];
+ Uint32 tmpPages[PAGESIZE*10];
+ PageHeader *thePageHeader;
+ Uint32 recordSize = 0;
+
+ PrepareOperationRecord *poRecord;
+ CommitTransactionRecord *ctRecord;
+ InvalidCommitTransactionRecord *ictRecord;
+
+ memcpy(pageHeader, pagePtr + PAGESIZE, PAGEHEADERSIZE*sizeof(Uint32));
+ memcpy(tmpPages, pagePtr + oldPageIndex, (PAGESIZE - oldPageIndex)*sizeof(Uint32));
+ memcpy(tmpPages + PAGESIZE - oldPageIndex ,
+ (pagePtr + PAGESIZE + PAGEHEADERSIZE),
+ (PAGESIZE - PAGEHEADERSIZE)*sizeof(Uint32));
+
+ switch(recordType) {
+ case ZPREP_OP_TYPE:
+ poRecord = (PrepareOperationRecord *) tmpPages;
+ recordSize = poRecord->getLogRecordSize();
+ if (recordSize < (PAGESIZE - PAGEHEADERSIZE)) {
+ if (theCheckFlag) {
+ if(!poRecord->check()) {
+ doExit();
+ }
+ }
+ if (thePrintFlag) ndbout << (*poRecord);
+ } else {
+ ndbout << "Error: Record greater than a Page" << endl;
+ }
+ break;
+
+ case ZCOMMIT_TYPE:
+ ctRecord = (CommitTransactionRecord *) tmpPages;
+ recordSize = ctRecord->getLogRecordSize();
+ if (recordSize < (PAGESIZE - PAGEHEADERSIZE)) {
+ if (theCheckFlag) {
+ if(!ctRecord->check()) {
+ doExit();
+ }
+ }
+ if (thePrintFlag) ndbout << (*ctRecord);
+ } else {
+ ndbout << endl << "Error: Record greater than a Page" << endl;
+ }
+ break;
+
+ case ZINVALID_COMMIT_TYPE:
+ ictRecord = (InvalidCommitTransactionRecord *) tmpPages;
+ recordSize = ictRecord->getLogRecordSize();
+ if (recordSize < (PAGESIZE - PAGEHEADERSIZE)) {
+ if (theCheckFlag) {
+ if(!ictRecord->check()) {
+ doExit();
+ }
+ }
+ if (thePrintFlag) ndbout << (*ictRecord);
+ } else {
+ ndbout << endl << "Error: Record greater than a Page" << endl;
+ }
+ break;
+
+ case ZNEW_PREP_OP_TYPE:
+ case ZABORT_TYPE:
+ case ZFRAG_SPLIT_TYPE:
+ case ZNEXT_MBYTE_TYPE:
+ ndbout << endl << "Record type = " << recordType << " not implemented." << endl;
+ return 0;
+
+ default:
+ ndbout << endl << "Error: Unknown record type. Record type = " << recordType << endl;
+ return 0;
+ }
+
+ thePageHeader = (PageHeader *) (pagePtr + PAGESIZE);
+ if (thePrintFlag) ndbout << (*thePageHeader);
+
+ return PAGEHEADERSIZE - PAGESIZE + oldPageIndex + recordSize;
+}
+
+//----------------------------------------------------------------
+//
+//----------------------------------------------------------------
+
+
+void usage(const char * prg){
+ ndbout << endl << "Usage: " << endl << prg
+ << " <Binary log file> [-noprint] [-nocheck] [-mbyte <0-15>] "
+ << "[-mbyteheaders] [-pageheaders] [-filedescriptors] [-page <0-31>] "
+ << "[-pageindex <12-8191>]"
+ << endl << endl;
+
+}
+void readArguments(int argc, const char** argv)
+{
+ if(argc < 2 || argc > 9){
+ usage(argv[0]);
+ doExit();
+ }
+
+ strcpy(fileName, argv[1]);
+ argc--;
+
+ int i = 2;
+ while (argc > 1)
+ {
+ if (strcmp(argv[i], "-noprint") == 0) {
+ thePrintFlag = false;
+ } else if (strcmp(argv[i], "-nocheck") == 0) {
+ theCheckFlag = false;
+ } else if (strcmp(argv[i], "-mbyteheaders") == 0) {
+ onlyMbyteHeaders = true;
+ } else if (strcmp(argv[i], "-pageheaders") == 0) {
+ onlyPageHeaders = true;
+ } else if (strcmp(argv[i], "-filedescriptors") == 0) {
+ onlyFileDesc = true;
+ } else if (strcmp(argv[i], "-mbyte") == 0) {
+ startAtMbyte = atoi(argv[i+1]);
+ if (startAtMbyte > 15) {
+ usage(argv[0]);
+ doExit();
+ }
+ argc--;
+ i++;
+ } else if (strcmp(argv[i], "-page") == 0) {
+ startAtPage = atoi(argv[i+1]);
+ if (startAtPage > 31) {
+ usage(argv[0]);
+ doExit();
+ }
+ argc--;
+ i++;
+ } else if (strcmp(argv[i], "-pageindex") == 0) {
+ startAtPageIndex = atoi(argv[i+1]);
+ if (startAtPageIndex > 8191 || startAtPageIndex < 12) {
+ usage(argv[0]);
+ doExit();
+ }
+ argc--;
+ i++;
+ } else {
+ usage(argv[0]);
+ doExit();
+ }
+ argc--;
+ i++;
+ }
+
+}
+
+void doExit() {
+ ndbout << "Error in redoLogReader(). Exiting!" << endl;
+ fclose(f);
+ delete [] redoLogPage;
+ exit(RETURN_ERROR);
+}
diff --git a/ndb/src/kernel/blocks/dbtc/Dbtc.hpp b/ndb/src/kernel/blocks/dbtc/Dbtc.hpp
new file mode 100644
index 00000000000..a3a01065429
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtc/Dbtc.hpp
@@ -0,0 +1,1947 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBTC_H
+#define DBTC_H
+
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <SimulatedBlock.hpp>
+#include <DLHashTable.hpp>
+#include <SLList.hpp>
+#include <DLList.hpp>
+#include <DLFifoList.hpp>
+#include <DataBuffer.hpp>
+#include <Bitmask.hpp>
+#include <AttributeList.hpp>
+#include <signaldata/AttrInfo.hpp>
+#include <signaldata/LqhTransConf.hpp>
+#include <signaldata/LqhKey.hpp>
+#include <signaldata/TrigAttrInfo.hpp>
+#include <signaldata/TcIndx.hpp>
+#include <signaldata/TransIdAI.hpp>
+#include <trigger_definitions.h>
+#include <SignalCounter.hpp>
+
+#ifdef DBTC_C
+/*
+ * 2.2 LOCAL SYMBOLS
+ * -----------------
+ */
+#define Z8NIL 255
+#define ZAPI_CONNECT_FILESIZE 20
+#define ZATTRBUF_FILESIZE 4000
+#define ZCLOSED 2
+#define ZCOMMITING 0 /* VALUE FOR TRANSTATUS */
+#define ZCOMMIT_SETUP 2
+#define ZCONTINUE_ABORT_080 4
+#define ZDATABUF_FILESIZE 4000
+#define ZGCP_FILESIZE 10
+#define ZINBUF_DATA_LEN 24 /* POSITION OF 'DATA LENGHT'-VARIABLE. */
+#define ZINBUF_NEXT 27 /* POSITION OF 'NEXT'-VARIABLE. */
+#define ZINBUF_PREV 26 /* POSITION OF 'PREVIOUS'-VARIABLE. */
+#define ZINTSPH1 1
+#define ZINTSPH2 2
+#define ZINTSPH3 3
+#define ZINTSPH6 6
+#define ZLASTPHASE 255
+#define ZMAX_DATA_IN_LQHKEYREQ 12
+#define ZNODEBUF_FILESIZE 2000
+#define ZNR_OF_SEIZE 10
+#define ZSCANREC_FILE_SIZE 100
+#define ZSCAN_FRAGREC_FILE_SIZE 400
+#define ZSCAN_OPREC_FILE_SIZE 400
+#define ZSEND_ATTRINFO 0
+#define ZSPH1 1
+#define ZTABREC_FILESIZE 16
+#define ZTAKE_OVER_ACTIVE 1
+#define ZTAKE_OVER_IDLE 0
+#define ZTC_CONNECT_FILESIZE 200
+#define ZTCOPCONF_SIZE 6
+
+// ----------------------------------------
+// Error Codes for Scan
+// ----------------------------------------
+#define ZNO_CONCURRENCY_ERROR 242
+#define ZTOO_HIGH_CONCURRENCY_ERROR 244
+#define ZNO_SCANREC_ERROR 245
+#define ZNO_FRAGMENT_ERROR 246
+#define ZSCAN_AI_LEN_ERROR 269
+#define ZSCAN_LQH_ERROR 270
+#define ZSCAN_FRAG_LQH_ERROR 274
+
+#define ZSCANTIME_OUT_ERROR 296
+#define ZSCANTIME_OUT_ERROR2 297
+
+// ----------------------------------------
+// Error Codes for transactions
+// ----------------------------------------
+#define ZSTATE_ERROR 202
+#define ZLENGTH_ERROR 207 // Also Scan
+#define ZERO_KEYLEN_ERROR 208
+#define ZSIGNAL_ERROR 209
+#define ZGET_ATTRBUF_ERROR 217 // Also Scan
+#define ZGET_DATAREC_ERROR 218
+#define ZMORE_AI_IN_TCKEYREQ_ERROR 220
+#define ZCOMMITINPROGRESS 230
+#define ZROLLBACKNOTALLOWED 232
+#define ZNO_FREE_TC_CONNECTION 233 // Also Scan
+#define ZABORTINPROGRESS 237
+#define ZPREPAREINPROGRESS 238
+#define ZWRONG_SCHEMA_VERSION_ERROR 241 // Also Scan
+#define ZSCAN_NODE_ERROR 250
+#define ZTRANS_STATUS_ERROR 253
+#define ZTIME_OUT_ERROR 266
+#define ZSIMPLE_READ_WITHOUT_AI 271
+#define ZNO_AI_WITH_UPDATE 272
+#define ZSEIZE_API_COPY_ERROR 275
+#define ZSCANINPROGRESS 276
+#define ZABORT_ERROR 277
+#define ZCOMMIT_TYPE_ERROR 278
+
+#define ZNO_FREE_TC_MARKER 279
+#define ZNODE_SHUTDOWN_IN_PROGRESS 280
+#define ZCLUSTER_SHUTDOWN_IN_PROGRESS 281
+#define ZWRONG_STATE 282
+#define ZCLUSTER_IN_SINGLEUSER_MODE 299
+
+#define ZDROP_TABLE_IN_PROGRESS 283
+#define ZNO_SUCH_TABLE 284
+#define ZUNKNOWN_TABLE_ERROR 285
+#define ZNODEFAIL_BEFORE_COMMIT 286
+#define ZINDEX_CORRUPT_ERROR 287
+
+// ----------------------------------------
+// Seize error
+// ----------------------------------------
+#define ZNO_FREE_API_CONNECTION 219
+#define ZSYSTEM_NOT_STARTED_ERROR 203
+
+// ----------------------------------------
+// Release errors
+// ----------------------------------------
+#define ZINVALID_CONNECTION 229
+
+
+#define ZNOT_FOUND 626
+#define ZALREADYEXIST 630
+#define ZINCONSISTENTHASHINDEX 892
+#endif
+
+class Dbtc: public SimulatedBlock {
+public:
+ enum ConnectionState {
+ CS_CONNECTED = 0,
+ CS_DISCONNECTED = 1,
+ CS_STARTED = 2,
+ CS_RECEIVING = 3,
+ CS_PREPARED = 4,
+ CS_START_PREPARING = 5,
+ CS_REC_PREPARING = 6,
+ CS_RESTART = 7,
+ CS_ABORTING = 8,
+ CS_COMPLETING = 9,
+ CS_COMPLETE_SENT = 10,
+ CS_PREPARE_TO_COMMIT = 11,
+ CS_COMMIT_SENT = 12,
+ CS_START_COMMITTING = 13,
+ CS_COMMITTING = 14,
+ CS_REC_COMMITTING = 15,
+ CS_WAIT_ABORT_CONF = 16,
+ CS_WAIT_COMPLETE_CONF = 17,
+ CS_WAIT_COMMIT_CONF = 18,
+ CS_FAIL_ABORTING = 19,
+ CS_FAIL_ABORTED = 20,
+ CS_FAIL_PREPARED = 21,
+ CS_FAIL_COMMITTING = 22,
+ CS_FAIL_COMMITTED = 23,
+ CS_FAIL_COMPLETED = 24,
+ CS_START_SCAN = 25
+ };
+
+ enum OperationState {
+ OS_CONNECTING_DICT = 0,
+ OS_CONNECTED = 1,
+ OS_OPERATING = 2,
+ OS_PREPARED = 3,
+ OS_COMMITTING = 4,
+ OS_COMMITTED = 5,
+ OS_COMPLETING = 6,
+ OS_COMPLETED = 7,
+ OS_RESTART = 8,
+ OS_ABORTING = 9,
+ OS_ABORT_SENT = 10,
+ OS_TAKE_OVER = 11,
+ OS_WAIT_DIH = 12,
+ OS_WAIT_KEYINFO = 13,
+ OS_WAIT_ATTR = 14,
+ OS_WAIT_COMMIT_CONF = 15,
+ OS_WAIT_ABORT_CONF = 16,
+ OS_WAIT_COMPLETE_CONF = 17
+ };
+
+ enum AbortState {
+ AS_IDLE = 0,
+ AS_ACTIVE = 1
+ };
+
+ enum HostState {
+ HS_ALIVE = 0,
+ HS_DEAD = 1
+ };
+
+ enum LqhTransState {
+ LTS_IDLE = 0,
+ LTS_ACTIVE = 1
+ };
+
+ enum TakeOverState {
+ TOS_NOT_DEFINED = 0,
+ TOS_IDLE = 1,
+ TOS_ACTIVE = 2,
+ TOS_COMPLETED = 3,
+ TOS_NODE_FAILED = 4
+ };
+
+ enum FailState {
+ FS_IDLE = 0,
+ FS_LISTENING = 1,
+ FS_COMPLETING = 2
+ };
+
+ enum SystemStartState {
+ SSS_TRUE = 0,
+ SSS_FALSE = 1
+ };
+
+ enum TimeOutCheckState {
+ TOCS_TRUE = 0,
+ TOCS_FALSE = 1
+ };
+
+ enum ReturnSignal {
+ RS_NO_RETURN = 0,
+ RS_TCKEYCONF = 1,
+ RS_TCKEYREF = 2,
+ RS_TC_COMMITCONF = 3,
+ RS_TCROLLBACKCONF = 4,
+ RS_TCROLLBACKREP = 5
+ };
+
+ enum IndexOperationState {
+ IOS_NOOP = 0,
+ IOS_INDEX_ACCESS = 1,
+ IOS_INDEX_ACCESS_WAIT_FOR_TCKEYCONF = 2,
+ IOS_INDEX_ACCESS_WAIT_FOR_TRANSID_AI = 3,
+ IOS_INDEX_OPERATION = 4
+ };
+
+ enum IndexState {
+ IS_BUILDING = 0, // build in progress, start state at create
+ IS_ONLINE = 1 // ready to use
+ };
+
+
+ /**--------------------------------------------------------------------------
+ * LOCAL SYMBOLS PER 'SYMBOL-VALUED' VARIABLE
+ *
+ *
+ * NSYMB ZAPI_CONNECT_FILESIZE = 20
+ * NSYMB ZTC_CONNECT_FILESIZE = 200
+ * NSYMB ZHOST_FILESIZE = 16
+ * NSYMB ZDATABUF_FILESIZE = 4000
+ * NSYMB ZATTRBUF_FILESIZE = 4000
+ * NSYMB ZGCP_FILESIZE = 10
+ *
+ *
+ * ABORTED CODES
+ * TPHASE NSYMB ZSPH1 = 1
+ * NSYMB ZLASTPHASE = 255
+ *
+ *
+ * LQH_TRANS
+ * NSYMB ZTRANS_ABORTED = 1
+ * NSYMB ZTRANS_PREPARED = 2
+ * NSYMB ZTRANS_COMMITTED = 3
+ * NSYMB ZCOMPLETED_LQH_TRANS = 4
+ * NSYMB ZTRANS_COMPLETED = 5
+ *
+ *
+ * TAKE OVER
+ * NSYMB ZTAKE_OVER_IDLE = 0
+ * NSYMB ZTAKE_OVER_ACTIVE = 1
+ *
+ * ATTRBUF (ATTRBUF_RECORD)
+ * NSYMB ZINBUF_DATA_LEN = 24
+ * NSYMB ZINBUF_NEXTFREE = 25 (NOT USED )
+ * NSYMB ZINBUF_PREV = 26
+ * NSYMB ZINBUF_NEXT = 27
+ -------------------------------------------------------------------------*/
+ /*
+ 2.3 RECORDS AND FILESIZES
+ -------------------------
+ */
+ /* **************************************************************** */
+ /* ---------------------------------------------------------------- */
+ /* ------------------- TRIGGER AND INDEX DATA --------------------- */
+ /* ---------------------------------------------------------------- */
+ /* **************************************************************** */
+ /* ********* DEFINED TRIGGER DATA ********* */
+ /* THIS RECORD FORMS LISTS OF ACTIVE */
+ /* TRIGGERS FOR EACH TABLE. */
+ /* THE RECORDS ARE MANAGED BY A TRIGGER */
+ /* POOL WHERE A TRIGGER RECORD IS SEIZED */
+ /* WHEN A TRIGGER IS ACTIVATED AND RELEASED */
+ /* WHEN THE TRIGGER IS DEACTIVATED. */
+ /* **************************************** */
+ struct TcDefinedTriggerData {
+ /**
+ * Trigger id, used to identify the trigger
+ */
+ UintR triggerId;
+
+ /**
+ * Trigger type, defines what the trigger is used for
+ */
+ TriggerType::Value triggerType;
+
+ /**
+ * Trigger type, defines what the trigger is used for
+ */
+ TriggerEvent::Value triggerEvent;
+
+ /**
+ * Attribute mask, defines what attributes are to be monitored
+ * Can be seen as a compact representation of SQL column name list
+ */
+ Bitmask<MAXNROFATTRIBUTESINWORDS> attributeMask;
+
+ /**
+ * Next ptr (used in pool/list)
+ */
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+
+ /**
+ * Index id, only used by secondary_index triggers. This is same as
+ * index table id in DICT.
+ **/
+ Uint32 indexId;
+
+ /**
+ * Prev pointer (used in list)
+ */
+ Uint32 prevList;
+
+ inline void print(NdbOut & s) const {
+ s << "[DefinedTriggerData = " << triggerId << "]";
+ }
+ };
+ typedef Ptr<TcDefinedTriggerData> DefinedTriggerPtr;
+
+ /**
+ * Pool of trigger data record
+ */
+ ArrayPool<TcDefinedTriggerData> c_theDefinedTriggerPool;
+
+ /**
+ * The list of active triggers
+ */
+ DLList<TcDefinedTriggerData> c_theDefinedTriggers;
+
+ typedef DataBuffer<11> AttributeBuffer;
+
+ AttributeBuffer::DataBufferPool c_theAttributeBufferPool;
+
+ UintR c_transactionBufferSpace;
+
+
+ /* ********** FIRED TRIGGER DATA ********** */
+ /* THIS RECORD FORMS LISTS OF FIRED */
+ /* TRIGGERS FOR A TRANSACTION. */
+ /* THE RECORDS ARE MANAGED BY A TRIGGER */
+ /* POOL WHERE A TRIGGER RECORD IS SEIZED */
+ /* WHEN A TRIGGER IS ACTIVATED AND RELEASED */
+ /* WHEN THE TRIGGER IS DEACTIVATED. */
+ /* **************************************** */
+ struct TcFiredTriggerData {
+ TcFiredTriggerData(AttributeBuffer::DataBufferPool & abp):
+ keyValues(abp),
+ beforeValues(abp),
+ afterValues(abp)
+ {}
+ /**
+ * Trigger id, used to identify the trigger
+ **/
+ Uint32 triggerId;
+
+ /**
+ * The operation that fired the trigger
+ */
+ Uint32 fireingOperation;
+
+ /**
+ * Trigger attribute info, primary key value(s)
+ */
+ AttributeBuffer keyValues;
+
+ /**
+ * Trigger attribute info, attribute value(s) before operation
+ */
+ AttributeBuffer beforeValues;
+
+ /**
+ * Trigger attribute info, attribute value(s) after operation
+ */
+ AttributeBuffer afterValues;
+
+ /**
+ * Next ptr (used in pool/list)
+ */
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+
+ /**
+ * Prev pointer (used in list)
+ */
+ Uint32 prevList;
+
+ inline void print(NdbOut & s) const {
+ s << "[FiredTriggerData = " << triggerId << "]";
+ }
+ };
+ typedef Ptr<TcFiredTriggerData> FiredTriggerPtr;
+
+ /**
+ * Pool of trigger data record
+ */
+ ArrayPool<TcFiredTriggerData> c_theFiredTriggerPool;
+ AttributeBuffer::DataBufferPool c_theTriggerAttrInfoPool;
+
+ Uint32 c_maxNumberOfDefinedTriggers;
+ Uint32 c_maxNumberOfFiredTriggers;
+
+ struct AttrInfoRecord {
+ /**
+ * Pre-allocated AttrInfo signal
+ */
+ AttrInfo attrInfo;
+
+ /**
+ * Next ptr (used in pool/list)
+ */
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ /**
+ * Prev pointer (used in list)
+ */
+ Uint32 prevList;
+ };
+
+
+ /* ************* INDEX DATA *************** */
+ /* THIS RECORD FORMS LISTS OF ACTIVE */
+ /* INDEX FOR EACH TABLE. */
+ /* THE RECORDS ARE MANAGED BY A INDEX */
+ /* POOL WHERE AN INDEX RECORD IS SEIZED */
+ /* WHEN AN INDEX IS CREATED AND RELEASED */
+ /* WHEN THE INDEX IS DROPPED. */
+ /* **************************************** */
+ struct TcIndexData {
+ /**
+ * IndexState
+ */
+ IndexState indexState;
+
+ /**
+ * Index id, same as index table id in DICT
+ */
+ Uint32 indexId;
+
+ /**
+ * Index attribute list. Only the length is used in v21x.
+ */
+ AttributeList attributeList;
+
+ /**
+ * Primary table id, the primary table to be indexed
+ */
+ Uint32 primaryTableId;
+
+ /**
+ * Primary key position in secondary table
+ */
+ Uint32 primaryKeyPos;
+
+ /**
+ * Next ptr (used in pool/list)
+ */
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ /**
+ * Prev pointer (used in list)
+ */
+ Uint32 prevList;
+ };
+
+ typedef Ptr<TcIndexData> TcIndexDataPtr;
+
+ /**
+ * Pool of index data record
+ */
+ ArrayPool<TcIndexData> c_theIndexPool;
+
+ /**
+ * The list of defined indexes
+ */
+ ArrayList<TcIndexData> c_theIndexes;
+ UintR c_maxNumberOfIndexes;
+
+ struct TcIndexOperation {
+ TcIndexOperation(AttributeBuffer::DataBufferPool & abp) :
+ indexOpState(IOS_NOOP),
+ expectedKeyInfo(0),
+ keyInfo(abp),
+ expectedAttrInfo(0),
+ attrInfo(abp),
+ expectedTransIdAI(0),
+ transIdAI(abp),
+ tcIndxReq(new TcIndxReq()),
+ indexReadTcConnect(RNIL)
+ {}
+
+ ~TcIndexOperation()
+ {
+ delete tcIndxReq;
+ }
+
+ // Index data
+ Uint32 indexOpId;
+ IndexOperationState indexOpState; // Used to mark on-going TcKeyReq
+ Uint32 expectedKeyInfo;
+ AttributeBuffer keyInfo; // For accumulating IndxKeyInfo
+ Uint32 expectedAttrInfo;
+ AttributeBuffer attrInfo; // For accumulating IndxAttrInfo
+ Uint32 expectedTransIdAI;
+ AttributeBuffer transIdAI; // For accumulating TransId_AI
+
+ TcIndxReq* tcIndxReq;
+ UintR connectionIndex;
+ UintR indexReadTcConnect; //
+
+ /**
+ * Next ptr (used in pool/list)
+ */
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ /**
+ * Prev pointer (used in list)
+ */
+ Uint32 prevList;
+ };
+
+ typedef Ptr<TcIndexOperation> TcIndexOperationPtr;
+
+ /**
+ * Pool of index data record
+ */
+ ArrayPool<TcIndexOperation> c_theIndexOperationPool;
+
+ /**
+ * The list of index operations
+ */
+ ArrayList<TcIndexOperation> c_theIndexOperations;
+
+ UintR c_maxNumberOfIndexOperations;
+
+ struct TcSeizedIndexOperation {
+ /**
+ * Next ptr (used in pool/list)
+ */
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ /**
+ * Prev pointer (used in list)
+ */
+ Uint32 prevList;
+ };
+
+ /**
+ * Pool of seized index operations
+ */
+ ArrayPool<TcSeizedIndexOperation> c_theSeizedIndexOperationPool;
+
+ typedef Ptr<TcSeizedIndexOperation> TcSeizedIndexOperationPtr;
+
+ /************************** API CONNECT RECORD ***********************
+ * The API connect record contains the connection record to which the
+ * application connects.
+ *
+ * The application can send one operation at a time. It can send a
+ * new operation immediately after sending the previous operation.
+ * Thereby several operations can be active in one transaction within TC.
+ * This is achieved by using the API connect record.
+ * Each active operation is handled by the TC connect record.
+ * As soon as the TC connect record has sent the
+ * request to the LQH it is ready to receive new operations.
+ * The LQH connect record takes care of waiting for an operation to
+ * complete.
+ * When an operation has completed on the LQH connect record,
+ * a new operation can be started on this LQH connect record.
+ *******************************************************************
+ *
+ * API CONNECT RECORD ALIGNED TO BE 256 BYTES
+ ********************************************************************/
+
+ /*******************************************************************>*/
+ // We break out the API Timer for optimisation on scanning rather than
+ // on fast access.
+ /*******************************************************************>*/
+ inline void setApiConTimer(Uint32 apiConPtrI, Uint32 value, Uint32 line){
+ c_apiConTimer[apiConPtrI] = value;
+ c_apiConTimer_line[apiConPtrI] = line;
+ }
+
+ inline Uint32 getApiConTimer(Uint32 apiConPtrI) const {
+ return c_apiConTimer[apiConPtrI];
+ }
+ UintR* c_apiConTimer;
+ UintR* c_apiConTimer_line;
+
+ struct ApiConnectRecord {
+ ApiConnectRecord(ArrayPool<TcFiredTriggerData> & firedTriggerPool,
+ ArrayPool<TcSeizedIndexOperation> & seizedIndexOpPool):
+ theFiredTriggers(firedTriggerPool),
+ isIndexOp(false),
+ theSeizedIndexOperations(seizedIndexOpPool)
+ {}
+
+ //---------------------------------------------------
+ // First 16 byte cache line. Hot variables.
+ //---------------------------------------------------
+ ConnectionState apiConnectstate;
+ UintR transid[2];
+ UintR firstTcConnect;
+
+ //---------------------------------------------------
+ // Second 16 byte cache line. Hot variables.
+ //---------------------------------------------------
+ UintR lqhkeyconfrec;
+ UintR cachePtr;
+ UintR currSavePointId;
+ UintR counter;
+
+ //---------------------------------------------------
+ // Third 16 byte cache line. First and second cache
+ // line plus this will be enough for copy API records.
+ // Variables used in late phases.
+ //---------------------------------------------------
+ UintR nextGcpConnect;
+ UintR prevGcpConnect;
+ UintR gcpPointer;
+ UintR ndbapiConnect;
+
+ //---------------------------------------------------
+ // Fourth 16 byte cache line. Only used in late phases.
+ // Plus 4 bytes of error handling.
+ //---------------------------------------------------
+ UintR nextApiConnect;
+ BlockReference ndbapiBlockref;
+ UintR apiCopyRecord;
+ UintR globalcheckpointid;
+
+ //---------------------------------------------------
+ // Second 64 byte cache line starts. First 16 byte
+ // cache line in this one. Variables primarily used
+ // in early phase.
+ //---------------------------------------------------
+ UintR lastTcConnect;
+ UintR lqhkeyreqrec;
+ AbortState abortState;
+ Uint32 buddyPtr;
+ Uint8 unused;
+ Uint8 unused2;
+ Uint8 takeOverRec;
+ Uint8 currentReplicaNo;
+
+ //---------------------------------------------------
+ // Error Handling variables. If cache line 32 bytes
+ // ensures that cache line is still only read in
+ // early phases.
+ //---------------------------------------------------
+ union {
+ UintR apiScanRec;
+ UintR commitAckMarker;
+ };
+ UintR currentTcConnect;
+ BlockReference tcBlockref;
+ Uint16 returncode;
+ Uint16 takeOverInd;
+
+ //---------------------------------------------------
+ // Second 64 byte cache line. Third 16 byte cache line
+ // in this one. Variables primarily used in early phase
+ // and checked in late phase.
+ // Fourth cache line is the tcSendArray that is used
+ // when two and three operations are responded to in
+ // parallel. The first two entries in tcSendArray is
+ // part of the third cache line.
+ //---------------------------------------------------
+ //---------------------------------------------------
+ // timeOutCounter is used waiting for ABORTCONF, COMMITCONF
+ // and COMPLETECONF
+ //---------------------------------------------------
+ UintR failureNr;
+ Uint8 tckeyrec; // Ändrad från R
+ Uint8 tcindxrec;
+ Uint8 apiFailState; // Ändrad från R
+ ReturnSignal returnsignal;
+ Uint8 timeOutCounter;
+
+ UintR tcSendArray[6];
+
+ // Trigger data
+
+ /**
+ * The list of fired triggers
+ */
+ DLFifoList<TcFiredTriggerData> theFiredTriggers;
+
+ bool triggerPending; // Used to mark waiting for a CONTINUEB
+
+ // Index data
+
+ bool isIndexOp; // Used to mark on-going TcKeyReq as indx table access
+ bool indexOpReturn;
+ UintR noIndexOp; // No outstanding index ops
+
+ // Index op return context
+ UintR indexOp;
+ UintR clientData;
+ UintR attrInfoLen;
+
+ UintR accumulatingIndexOp;
+ UintR executingIndexOp;
+ UintR tcIndxSendArray[6];
+ ArrayList<TcSeizedIndexOperation> theSeizedIndexOperations;
+ };
+
+ typedef Ptr<ApiConnectRecord> ApiConnectRecordPtr;
+
+
+ /************************** TC CONNECT RECORD ************************/
+ /* *******************************************************************/
+ /* TC CONNECT RECORD KEEPS ALL INFORMATION TO CARRY OUT A TRANSACTION*/
+ /* THE TRANSACTION CONTROLLER ESTABLISHES CONNECTIONS TO DIFFERENT */
+ /* BLOCKS TO CARRY OUT THE TRANSACTION. THERE CAN BE SEVERAL RECORDS */
+ /* PER ACTIVE TRANSACTION. THE TC CONNECT RECORD COOPERATES WITH THE */
+ /* API CONNECT RECORD FOR COMMUNICATION WITH THE API AND WITH THE */
+ /* LQH CONNECT RECORD FOR COMMUNICATION WITH THE LQH'S INVOLVED IN */
+ /* THE TRANSACTION. TC CONNECT RECORD IS PERMANENTLY CONNECTED TO A */
+ /* RECORD IN DICT AND ONE IN DIH. IT CONTAINS A LIST OF ACTIVE LQH */
+ /* CONNECT RECORDS AND A LIST OF STARTED BUT NOT ACTIVE LQH CONNECT */
+ /* RECORDS. IT DOES ALSO CONTAIN A LIST OF ALL OPERATIONS THAT ARE */
+ /* EXECUTED WITH THE TC CONNECT RECORD. */
+ /*******************************************************************>*/
+ /* TC_CONNECT RECORD ALIGNED TO BE 128 BYTES */
+ /*******************************************************************>*/
+ struct TcConnectRecord {
+ //---------------------------------------------------
+ // First 16 byte cache line. Those variables are only
+ // used in error cases.
+ //---------------------------------------------------
+ UintR tcOprec; /* TC OPREC of operation being taken over */
+ Uint16 failData[4]; /* Failed nodes when taking over an operation */
+ UintR nextTcFailHash;
+
+ //---------------------------------------------------
+ // Second 16 byte cache line. Those variables are used
+ // from LQHKEYCONF to sending COMMIT and COMPLETED.
+ //---------------------------------------------------
+ UintR lastLqhCon; /* Connect record in last replicas Lqh record */
+ Uint16 lastLqhNodeId; /* Node id of last replicas Lqh */
+ Uint16 m_execAbortOption;/* TcKeyReq::ExecuteAbortOption */
+ UintR commitAckMarker; /* CommitMarker I value */
+
+ //---------------------------------------------------
+ // Third 16 byte cache line. The hottest variables.
+ //---------------------------------------------------
+ OperationState tcConnectstate; /* THE STATE OF THE CONNECT*/
+ UintR apiConnect; /* POINTER TO API CONNECT RECORD */
+ UintR nextTcConnect; /* NEXT TC RECORD*/
+ Uint8 dirtyOp;
+ Uint8 lastReplicaNo; /* NUMBER OF THE LAST REPLICA IN THE OPERATION */
+ Uint8 noOfNodes; /* TOTAL NUMBER OF NODES IN OPERATION */
+ Uint8 operation; /* OPERATION TYPE */
+ /* 0 = READ REQUEST */
+ /* 1 = UPDATE REQUEST */
+ /* 2 = INSERT REQUEST */
+ /* 3 = DELETE REQUEST */
+
+ //---------------------------------------------------
+ // Fourth 16 byte cache line. The mildly hot variables.
+ // tcNodedata expands 4 Bytes into the next cache line
+ // with indexes almost never used.
+ //---------------------------------------------------
+ UintR clientData; /* SENDERS OPERATION POINTER */
+ UintR dihConnectptr; /* CONNECTION TO DIH BLOCK ON THIS NODE */
+ UintR prevTcConnect; /* DOUBLY LINKED LIST OF TC CONNECT RECORDS*/
+ UintR savePointId;
+
+ Uint16 tcNodedata[4];
+
+ // Trigger data
+ FiredTriggerPtr accumulatingTriggerData;
+ UintR noFiredTriggers;
+ UintR noReceivedTriggers;
+ UintR triggerExecutionCount;
+ UintR triggeringOperation;
+ UintR savedState[LqhKeyConf::SignalLength];
+ UintR triggerError;
+
+ // Index data
+ bool isIndexOp; // Used to mark on-going TcKeyReq as index table access
+ UintR indexOp;
+ UintR currentIndexId;
+ UintR attrInfoLen;
+ };
+
+ friend struct TcConnectRecord;
+
+ typedef Ptr<TcConnectRecord> TcConnectRecordPtr;
+
+ // ********************** CACHE RECORD **************************************
+ //---------------------------------------------------------------------------
+ // This record is used between reception of TCKEYREQ and sending of LQHKEYREQ
+ // It is separatedso as to improve the cache hit rate and also to minimise
+ // the necessary memory storage in NDB Cluster.
+ //---------------------------------------------------------------------------
+
+ struct CacheRecord {
+ //---------------------------------------------------
+ // First 16 byte cache line. Variables used by
+ // ATTRINFO processing.
+ //---------------------------------------------------
+ UintR firstAttrbuf; /* POINTER TO LINKED LIST OF ATTRIBUTE BUFFERS */
+ UintR lastAttrbuf; /* POINTER TO LINKED LIST OF ATTRIBUTE BUFFERS */
+ UintR currReclenAi;
+ Uint16 attrlength; /* ATTRIBUTE INFORMATION LENGTH */
+ Uint16 save1;
+
+ //---------------------------------------------------
+ // Second 16 byte cache line. Variables initiated by
+ // TCKEYREQ and used in LQHKEYREQ.
+ //---------------------------------------------------
+ UintR attrinfo15[4];
+
+ //---------------------------------------------------
+ // Third 16 byte cache line. Variables initiated by
+ // TCKEYREQ and used in LQHKEYREQ.
+ //---------------------------------------------------
+ UintR attrinfo0;
+ UintR schemaVersion;/* SCHEMA VERSION USED IN TRANSACTION */
+ UintR tableref; /* POINTER TO THE TABLE IN WHICH THE FRAGMENT EXISTS*/
+ Uint16 apiVersionNo;
+ Uint16 keylen; /* KEY LENGTH SENT BY REQUEST SIGNAL */
+
+ //---------------------------------------------------
+ // Fourth 16 byte cache line. Variables initiated by
+ // TCKEYREQ and used in LQHKEYREQ.
+ //---------------------------------------------------
+ UintR keydata[4]; /* RECEIVES FIRST 16 BYTES OF TUPLE KEY */
+
+ //---------------------------------------------------
+ // First 16 byte cache line in second 64 byte cache
+ // line. Diverse use.
+ //---------------------------------------------------
+ UintR fragmentid; /* THE COMPUTED FRAGMENT ID */
+ UintR hashValue; /* THE HASH VALUE USED TO LOCATE FRAGMENT */
+
+ Uint8 distributionKeyIndicator;
+ Uint8 distributionGroupIndicator;
+ Uint8 distributionGroupType;
+ Uint8 lenAiInTckeyreq; /* LENGTH OF ATTRIBUTE INFORMATION IN TCKEYREQ */
+
+ Uint8 distributionKey;
+
+ /**
+ * EXECUTION MODE OF OPERATION
+ * 0 = NORMAL EXECUTION, 1 = INTERPRETED EXECUTION
+ */
+ Uint8 opExec;
+
+ /**
+ * LOCK TYPE OF OPERATION IF READ OPERATION
+ * 0 = READ LOCK, 1 = WRITE LOCK
+ */
+ Uint8 opLock;
+
+ /**
+ * IS THE OPERATION A SIMPLE TRANSACTION
+ * 0 = NO, 1 = YES
+ */
+ Uint8 opSimple;
+
+ //---------------------------------------------------
+ // Second 16 byte cache line in second 64 byte cache
+ // line. Diverse use.
+ //---------------------------------------------------
+ UintR distributionGroup;
+ UintR nextCacheRec;
+ UintR distributionKeySize;
+ Uint16 scanNode;
+ unsigned scanTakeOverInd : 1;
+ unsigned scanInfo : 15; // 12 bits used currently
+
+ //---------------------------------------------------
+ // Third and fourth 16 byte cache line in second 64
+ // byte cache line. Not used currently.
+ //---------------------------------------------------
+ UintR firstKeybuf; /* POINTER THE LINKED LIST OF KEY BUFFERS */
+ UintR lastKeybuf; /* VARIABLE POINTING TO THE LAST KEY BUFFER */
+ UintR packedCacheVar[6];
+ };
+
+ typedef Ptr<CacheRecord> CacheRecordPtr;
+
+ /* ************************ HOST RECORD ********************************** */
+ /********************************************************/
+ /* THIS RECORD CONTAINS ALIVE-STATUS ON ALL NODES IN THE*/
+ /* SYSTEM */
+ /********************************************************/
+ /* THIS RECORD IS ALIGNED TO BE 128 BYTES. */
+ /********************************************************/
+ struct HostRecord {
+ HostState hostStatus;
+ LqhTransState lqhTransStatus;
+ TakeOverState takeOverStatus;
+ bool inPackedList;
+ UintR ndbVersion;
+ UintR noOfPackedWordsLqh;
+ UintR packedWordsLqh[26];
+ UintR noOfWordsTCKEYCONF;
+ UintR packedWordsTCKEYCONF[30];
+ UintR noOfWordsTCINDXCONF;
+ UintR packedWordsTCINDXCONF[30];
+ BlockReference hostLqhBlockRef;
+ }; /* p2c: size = 128 bytes */
+
+ typedef Ptr<HostRecord> HostRecordPtr;
+
+ /* *********** TABLE RECORD ********************************************* */
+ /********************************************************/
+ /* THIS RECORD CONTAINS THE CURRENT SCHEMA VERSION OF */
+ /* ALL TABLES IN THE SYSTEM. */
+ /********************************************************/
+ struct TableRecord {
+ Uint32 currentSchemaVersion;
+ Uint8 enabled;
+ Uint8 dropping;
+ Uint8 tableType;
+ Uint8 storedTable;
+
+ bool checkTable(Uint32 schemaVersion) const {
+ return enabled && !dropping && (schemaVersion == currentSchemaVersion);
+ }
+
+ Uint32 getErrorCode(Uint32 schemaVersion) const;
+
+ struct DropTable {
+ Uint32 senderRef;
+ Uint32 senderData;
+ SignalCounter waitDropTabCount;
+ } dropTable;
+ };
+ typedef Ptr<TableRecord> TableRecordPtr;
+
+ /**
+ * Each scan allocates one ScanRecord to store information
+ * about the current scan
+ *
+ */
+ struct ScanRecord {
+ /** NOTE! This is the old comment for ScanState. - MASV
+ * STATE TRANSITIONS OF SCAN_STATE. SCAN_STATE IS THE STATE
+ * VARIABLE OF THE RECEIVE AND DELIVERY PROCESS.
+ * THE PROCESS HAS THREE STEPS IT GOES THROUGH.
+ * 1) THE INITIAL STATES WHEN RECEIVING DATA FOR THE SCAN.
+ * - WAIT_SCAN_TAB_INFO
+ * - WAIT_AI
+ * - WAIT_FRAGMENT_COUNT
+ * 2) THE EXECUTION STATES WHEN THE SCAN IS PERFORMED.
+ * - SCAN_NEXT_ORDERED
+ * - DELIVERED
+ * - QUEUED_DELIVERED
+ * 3) THE CLOSING STATE WHEN THE SCAN PROCESS IS CLOSING UP
+ * EVERYTHING.
+ * - CLOSING_SCAN
+ * INITIAL START WHEN SCAN_TABREQ RECEIVED
+ * -> WAIT_SCAN_TAB_INFO (IF ANY SCAN_TABINFO TO BE RECEIVED)
+ * -> WAIT_AI (IF NO SCAN_TAB_INFO BUT ATTRINFO IS RECEIVED)
+ * -> WAIT_FRAGMENT_COUNT (IF NEITHER SCAN_TABINFO OR ATTRINFO
+ * RECEIVED)
+ *
+ * WAIT_SCAN_TAB_INFO TRANSITIONS:
+ * -> WAIT_SCAN_TABINFO (WHEN MORE SCAN_TABINFO RECEIVED)
+ * -> WAIT_AI (WHEN ATTRINFO RECEIVED AFTER RECEIVING ALL
+ * SCAN_TABINFO)
+ * -> WAIT_FRAGMENT_COUNT (WHEN NO ATTRINFO RECEIVED AFTER
+ * RECEIVING ALL SCAN_TABINFO )
+ * WAIT_AI TRANSITIONS:
+ * -> WAIT_AI (WHEN MORE ATTRINFO RECEIVED)
+ * -> WAIT_FRAGMENT_COUNT (WHEN ALL ATTRINFO RECEIVED)
+ *
+ * WAIT_FRAGMENT_COUNT TRANSITIONS:
+ * -> SCAN_NEXT_ORDERED
+ *
+ * SCAN_NEXT_ORDERED TRANSITIONS:
+ * -> DELIVERED (WHEN FIRST SCAN_FRAGCONF ARRIVES WITH OPERATIONS
+ * TO REPORT IN IT)
+ * -> CLOSING_SCAN (WHEN SCAN IS CLOSED BY SCAN_NEXTREQ OR BY SOME
+ * ERROR)
+ *
+ * DELIVERED TRANSITIONS:
+ * -> SCAN_NEXT_ORDERED (IF SCAN_NEXTREQ ARRIVES BEFORE ANY NEW
+ * OPERATIONS TO REPORT ARRIVES)
+ * -> QUEUED_DELIVERED (IF NEW OPERATION TO REPORT ARRIVES BEFORE
+ * SCAN_NEXTREQ)
+ * -> CLOSING_SCAN (WHEN SCAN IS CLOSED BY SCAN_NEXTREQ OR BY SOME
+ * ERROR)
+ *
+ * QUEUED_DELIVERED TRANSITIONS:
+ * -> DELIVERED (WHEN SCAN_NEXTREQ ARRIVES AND QUEUED OPERATIONS
+ * TO REPORT ARE SENT TO THE APPLICATION)
+ * -> CLOSING_SCAN (WHEN SCAN IS CLOSED BY SCAN_NEXTREQ OR BY
+ * SOME ERROR)
+ */
+ enum ScanState {
+ IDLE = 0,
+ WAIT_SCAN_TAB_INFO = 1,
+ WAIT_AI = 2,
+ WAIT_FRAGMENT_COUNT = 3,
+ SCAN_NEXT_ORDERED = 4,
+ QUEUED_DELIVERED = 5,
+ DELIVERED = 6,
+ CLOSING_SCAN = 7
+ };
+ // State of this scan
+ ScanState scanState;
+ // References to ScanFragRecs
+ Uint32 scanFragrec[16];
+ // Refrences to ScanOperationRecords
+ Uint32 scanOprec[16];
+ // Number of ScanOperationRecords allocated
+ Uint32 noScanOprec;
+ // Id of the next fragment to be scanned. Used by scan fragment
+ // processes when they are ready for the next fragment
+ Uint32 scanNextFragId;
+ // Total number of fragments in the table we are scanning
+ Uint32 scanNoFrag;
+ // Index of next ScanRecords when in free list
+ Uint32 nextScan;
+ // Length of expected attribute information
+ Uint32 scanAiLength;
+ // Reference to ApiConnectRecord
+ Uint32 scanApiRec;
+ // Reference to TcConnectRecord
+ Uint32 scanTcrec;
+ // Number of scan frag processes that belong to this scan
+ Uint32 scanParallel;
+ // The number of recieved operations so far
+ Uint32 scanReceivedOperations;
+ // Schema version used by this scan
+ Uint32 scanSchemaVersion;
+ // Index of stored procedure belonging to this scan
+ Uint32 scanStoredProcId;
+ // The index of table that is scanned
+ Uint32 scanTableref;
+ // Number of operation records per scanned fragment
+ Uint16 noOprecPerFrag;
+ // The number of SCAN_TABINFO to receive
+ Uint16 noScanTabInfo;
+ // The number of SCAN_TABINFO received so far
+ Uint16 scanTabInfoReceived;
+ // apiIsClosed indicates if it's ok to release all resources
+ // and send a response to the API
+ // If it's false resources should not be released wait for API
+ // to close the scan
+ bool apiIsClosed;
+ // The number of scan frag processes that have completed their task
+ Uint8 scanProcessesCompleted;
+ // This variable is ZFALSE as long as any scan process is still alive
+ // It is ZTRUE as soon as all scan processes have been stopped
+ Uint8 scanCompletedStatus;
+ // Shall the locks be held until the application have read the
+ // records
+ Uint8 scanLockHold;
+ // Shall the locks be read or write locks
+ Uint8 scanLockMode;
+ // Skip locks by other transactions and read latest committed
+ Uint8 readCommitted;
+ // Scan is on ordered index
+ Uint8 rangeScan;
+ };
+ typedef Ptr<ScanRecord> ScanRecordPtr;
+
+ /**
+ * Each scan has max 16 ScanOperationRecords
+ * they are used for storing data to be sent to the api
+ */
+ struct ScanOperationRecord {
+ // Reference to the scan operation in api
+ Uint32 apiOpptr[16];
+ // Index and length of all recieved operations
+ // They will be cached here until SCAN_TABCONF is sent to api
+ Uint32 scanOpLength[16];
+ // Next ScanOperationRecord when in free list
+ Uint32 nextScanOp;
+ }; /* p2c: size = 132 bytes */
+
+ typedef Ptr<ScanOperationRecord> ScanOperationRecordPtr;
+
+ /**
+ * There is max 16 ScanFragRec's for
+ * each scan started in TC. Each ScanFragRec is used by
+ * a scan fragment "process" that scans one fragment at a time.
+ * It will receive max 16 tuples in each request
+ */
+ struct ScanFragRec {
+ /**
+ * ScanFragState
+ * WAIT_GET_PRIMCONF : Waiting for DIGETPRIMCONF when starting a new
+ * fragment scan
+ * LQH_ACTIVE : The scan process has sent a command to LQH and is
+ * waiting for the response
+ * LQH_ACTIVE_CLOSE : The scan process has sent close to LQH and is
+ * waiting for the response
+ * DELIVERED : The result have been delivered, this scan frag process
+ * are waiting for a SCAN_NEXTREQ to tell us to continue scanning
+ * RETURNING_FROM_DELIVERY : SCAN_NEXTREQ received and continuing scan
+ * soon
+ * QUEUED_FOR_DELIVERY : Result queued in TC and waiting for delivery
+ * to API
+ * COMPLETED : The fragment scan processes has completed and finally
+ * sent a SCAN_PROCCONF
+ */
+ enum ScanFragState {
+ IDLE = 0,
+ WAIT_GET_PRIMCONF = 1,
+ LQH_ACTIVE = 2,
+ LQH_ACTIVE_CLOSE = 3,
+ DELIVERED = 4,
+ RETURNING_FROM_DELIVERY = 5,
+ QUEUED_FOR_DELIVERY = 6,
+ COMPLETED = 7
+ };
+ // Timer for checking timeout of this fragment scan
+ Uint32 scanFragTimer;
+ // Id of the current scanned fragment
+ Uint32 scanFragId;
+ // Blockreference of LQH
+ BlockReference lqhBlockref;
+ // getNodeInfo.m_connectCount, set at seize used so that
+ // I don't accidently kill a starting node
+ Uint32 m_connectCount;
+ // State of this fragment scan
+ ScanFragState scanFragState;
+ // Id of the ScanRecord this fragment scan belongs to
+ Uint32 scanRec;
+ // Index of next ScanFragRec, when in list of
+ // free ScanFragRec's
+ Uint32 nextScanFrag;
+ // Process id of this scan process within the total scan
+ Uint32 scanFragProcId;
+ // Node where current fragment resides
+ NodeId scanFragNodeId;
+ // Index of where to store the result in ScanRecord
+ Uint16 scanIndividual;
+ // The maximum number of operations that can be scanned before
+ // returning to TC
+ Uint16 scanFragConcurrency;
+ // Current status of the fragment scan
+ // * 0 = NOT COMPLETED
+ // * 1 = COMPLETED
+ // * 2 = CLOSED
+ Uint8 scanFragCompletedStatus;
+
+ inline void startFragTimer(Uint32 timeVal){
+ scanFragTimer = timeVal;
+ }
+ inline void stopFragTimer(void){
+ scanFragTimer = 0;
+ }
+ };
+
+ typedef Ptr<ScanFragRec> ScanFragRecPtr;
+
+ /* **********************************************************************$ */
+ /* ******$ DATA BUFFER ******$ */
+ /* */
+ /* THIS BUFFER IS USED AS A GENERAL DATA STORAGE. */
+ /* **********************************************************************$ */
+ struct DatabufRecord {
+ UintR data[4];
+ /* 4 * 1 WORD = 4 WORD */
+ UintR nextDatabuf;
+ }; /* p2c: size = 20 bytes */
+
+ typedef Ptr<DatabufRecord> DatabufRecordPtr;
+
+ /* **********************************************************************$ */
+ /* ******$ ATTRIBUTE INFORMATION RECORD ******$ */
+ /*
+ * CAN CONTAIN ONE (1) ATTRINFO SIGNAL. ONE SIGNAL CONTAINS 24 ATTR.
+ * INFO WORDS. BUT 32 ELEMENTS ARE USED TO MAKE PLEX HAPPY.
+ * SOME OF THE ELEMENTS ARE USED TO THE FOLLOWING THINGS:
+ * DATA LENGHT IN THIS RECORD IS STORED IN THE ELEMENT INDEXED BY
+ * ZINBUF_DATA_LEN.
+ * NEXT FREE ATTRBUF IS POINTED OUT BY THE ELEMENT INDEXED BY
+ * PREVIOUS ATTRBUF IS POINTED OUT BY THE ELEMENT INDEXED BY ZINBUF_PREV
+ * (NOT USED YET).
+ * NEXT ATTRBUF IS POINTED OUT BY THE ELEMENT INDEXED BY ZINBUF_NEXT. */
+ /* ******************************************************************** */
+ struct AttrbufRecord {
+ UintR attrbuf[32];
+ }; /* p2c: size = 128 bytes */
+
+ typedef Ptr<AttrbufRecord> AttrbufRecordPtr;
+
+ /*************************************************************************>*/
+ /* GLOBAL CHECKPOINT INFORMATION RECORD */
+ /* */
+ /* THIS RECORD IS USED TO STORE THE GLOBALCHECKPOINT NUMBER AND A
+ * COUNTER DURING THE COMPLETION PHASE OF THE TRANSACTION */
+ /*************************************************************************>*/
+ /* */
+ /* GCP RECORD ALIGNED TO BE 32 BYTES */
+ /*************************************************************************>*/
+ struct GcpRecord {
+ UintR gcpUnused1[2]; /* p2c: Not used */
+ UintR firstApiConnect;
+ UintR lastApiConnect;
+ UintR gcpId;
+ UintR nextGcp;
+ UintR gcpUnused2; /* p2c: Not used */
+ Uint16 gcpNomoretransRec;
+ }; /* p2c: size = 32 bytes */
+
+ typedef Ptr<GcpRecord> GcpRecordPtr;
+
+ /*************************************************************************>*/
+ /* TC_FAIL_RECORD */
+ /* THIS RECORD IS USED WHEN HANDLING TAKE OVER OF ANOTHER FAILED
+ * TC NODE. */
+ /*************************************************************************>*/
+ struct TcFailRecord {
+ Uint16 queueList[MAX_NDB_NODES];
+ Uint8 takeOverProcState[MAX_NDB_NODES];
+ UintR completedTakeOver;
+ UintR currentHashIndexTakeOver;
+ FailState failStatus;
+ Uint16 queueIndex;
+ Uint16 takeOverNode;
+ }; /* p2c: size = 64 bytes */
+
+ typedef Ptr<TcFailRecord> TcFailRecordPtr;
+
+public:
+ Dbtc(const class Configuration &);
+ virtual ~Dbtc();
+
+private:
+ BLOCK_DEFINES(Dbtc);
+
+ // Transit signals
+ void execPACKED_SIGNAL(Signal* signal);
+ void execABORTED(Signal* signal);
+ void execATTRINFO(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+ void execKEYINFO(Signal* signal);
+ void execSCAN_NEXTREQ(Signal* signal);
+ void execSCAN_PROCREQ(Signal* signal);
+ void execSCAN_PROCCONF(Signal* signal);
+ void execTAKE_OVERTCREQ(Signal* signal);
+ void execTAKE_OVERTCCONF(Signal* signal);
+ void execLQHKEYREF(Signal* signal);
+ void execTRANSID_AI_R(Signal* signal);
+ void execKEYINFO20_R(Signal* signal);
+
+ // Received signals
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execSEND_PACKED(Signal* signal);
+ void execCOMPLETED(Signal* signal);
+ void execCOMMITTED(Signal* signal);
+ void execDIGETNODESREF(Signal* signal);
+ void execDIGETPRIMCONF(Signal* signal);
+ void execDIGETPRIMREF(Signal* signal);
+ void execDISEIZECONF(Signal* signal);
+ void execDIVERIFYCONF(Signal* signal);
+ void execDI_FCOUNTCONF(Signal* signal);
+ void execDI_FCOUNTREF(Signal* signal);
+ void execGCP_NOMORETRANS(Signal* signal);
+ void execLQHKEYCONF(Signal* signal);
+ void execNDB_STTOR(Signal* signal);
+ void execREAD_NODESCONF(Signal* signal);
+ void execREAD_NODESREF(Signal* signal);
+ void execSTTOR(Signal* signal);
+ void execTC_COMMITREQ(Signal* signal);
+ void execTC_CLOPSIZEREQ(Signal* signal);
+ void execTCGETOPSIZEREQ(Signal* signal);
+ void execTCKEYREQ(Signal* signal);
+ void execTCRELEASEREQ(Signal* signal);
+ void execTCSEIZEREQ(Signal* signal);
+ void execTCROLLBACKREQ(Signal* signal);
+ void execTC_HBREP(Signal* signal);
+ void execTC_SCHVERREQ(Signal* signal);
+ void execSCAN_TABREQ(Signal* signal);
+ void execSCAN_TABINFO(Signal* signal);
+ void execSCAN_FRAGCONF(Signal* signal);
+ void execSCAN_FRAGREF(Signal* signal);
+ void execSIZEALT_REP(Signal* signal);
+ void execLQH_TRANSCONF(Signal* signal);
+ void execCOMPLETECONF(Signal* signal);
+ void execCOMMITCONF(Signal* signal);
+ void execABORTCONF(Signal* signal);
+ void execNODE_FAILREP(Signal* signal);
+ void execINCL_NODEREQ(Signal* signal);
+ void execTIME_SIGNAL(Signal* signal);
+ void execAPI_FAILREQ(Signal* signal);
+ void execSCAN_HBREP(Signal* signal);
+ void execSET_VAR_REQ(Signal* signal);
+
+ void execABORT_ALL_REQ(Signal* signal);
+
+ void execCREATE_TRIG_REQ(Signal* signal);
+ void execDROP_TRIG_REQ(Signal* signal);
+ void execFIRE_TRIG_ORD(Signal* signal);
+ void execTRIG_ATTRINFO(Signal* signal);
+ void execCREATE_INDX_REQ(Signal* signal);
+ void execDROP_INDX_REQ(Signal* signal);
+ void execTCINDXREQ(Signal* signal);
+ void execINDXKEYINFO(Signal* signal);
+ void execINDXATTRINFO(Signal* signal);
+ void execALTER_INDX_REQ(Signal* signal);
+
+ // Index table lookup
+ void execTCKEYCONF(Signal* signal);
+ void execTCKEYREF(Signal* signal);
+ void execTRANSID_AI(Signal* signal);
+ void execTCROLLBACKREP(Signal* signal);
+
+ void execCREATE_TAB_REQ(Signal* signal);
+ void execPREP_DROP_TAB_REQ(Signal* signal);
+ void execDROP_TAB_REQ(Signal* signal);
+ void execWAIT_DROP_TAB_CONF(Signal* signal);
+ void checkWaitDropTabFailedLqh(Signal*, Uint32 nodeId, Uint32 tableId);
+ void execALTER_TAB_REQ(Signal* signal);
+ void set_timeout_value(Uint32 timeOut);
+ void set_appl_timeout_value(Uint32 timeOut);
+ void set_no_parallel_takeover(Uint32);
+ void updateBuddyTimer(ApiConnectRecordPtr);
+
+ // Statement blocks
+ void updatePackedList(Signal* signal, HostRecord* ahostptr,
+ Uint16 ahostIndex);
+ void clearTcNodeData(Signal* signal,
+ UintR TLastLqhIndicator,
+ UintR Tstart);
+ void errorReport(Signal* signal, int place);
+ void warningReport(Signal* signal, int place);
+ void printState(Signal* signal, int place);
+ int seizeTcRecord(Signal* signal);
+ int seizeCacheRecord(Signal* signal);
+ void TCKEY_abort(Signal* signal, int place);
+ void copyFromToLen(UintR* sourceBuffer, UintR* destBuffer, UintR copyLen);
+ void reportNodeFailed(Signal* signal, Uint32 nodeId);
+ void sendPackedTCKEYCONF(Signal* signal,
+ HostRecord * ahostptr,
+ UintR hostId);
+ void sendPackedTCINDXCONF(Signal* signal,
+ HostRecord * ahostptr,
+ UintR hostId);
+ void sendPackedSignalLqh(Signal* signal, HostRecord * ahostptr);
+ void sendCommitLqh(Signal* signal,
+ TcConnectRecord * const regTcPtr);
+ void sendCompleteLqh(Signal* signal,
+ TcConnectRecord * const regTcPtr);
+ void sendTCKEY_FAILREF(Signal* signal, const ApiConnectRecord *);
+ void sendTCKEY_FAILCONF(Signal* signal, const ApiConnectRecord *);
+ void checkStartTimeout(Signal* signal);
+ void checkStartFragTimeout(Signal* signal);
+ void timeOutFoundFragLab(Signal* signal, Uint32 TscanConPtr);
+ void timeOutLoopStartFragLab(Signal* signal, Uint32 TscanConPtr);
+ int releaseAndAbort(Signal* signal);
+ void findApiConnectFail(Signal* signal);
+ void findTcConnectFail(Signal* signal);
+ void initApiConnectFail(Signal* signal);
+ void initTcConnectFail(Signal* signal);
+ void initTcFail(Signal* signal);
+ void releaseTakeOver(Signal* signal);
+ void setupFailData(Signal* signal);
+ void updateApiStateFail(Signal* signal);
+ void updateTcStateFail(Signal* signal);
+ void handleApiFailState(Signal* signal, UintR anApiConnectptr);
+ void handleFailedApiNode(Signal* signal,
+ UintR aFailedNode,
+ UintR anApiConnectPtr);
+ void handleScanStop(Signal* signal, UintR aFailedNode);
+ void initScanTcrec(Signal* signal);
+ void initScanApirec(Signal* signal,
+ Uint32 buddyPtr,
+ UintR transid1,
+ UintR transid2);
+ void initScanOprec(Signal* signal);
+ void initScanrec(Signal* signal,
+ const UintR scanParallel,
+ const UintR noOprecPerFrag);
+ void initScanfragrec(Signal* signal);
+ void releaseScanrec(Signal* signal);
+ void releaseScanResources(Signal* signal);
+ void releaseScanFragrec(Signal* signal);
+ void releaseScanOprec(Signal* signal);
+ void seizeScanrec(Signal* signal);
+ void seizeScanFragrec(Signal* signal);
+ void seizeScanOprec(Signal* signal);
+ void sendScanFragReq(Signal* signal);
+ void sendScanTabConf(Signal* signal);
+ void sendScanProcConf(Signal* signal);
+ void setScanReceived(Signal* signal, Uint32 noCompletedOps);
+
+ void checkGcp(Signal* signal);
+ void commitGciHandling(Signal* signal, UintR Tgci);
+ void copyApi(Signal* signal);
+ void DIVER_node_fail_handling(Signal* signal, UintR Tgci);
+ void gcpTcfinished(Signal* signal);
+ void handleGcp(Signal* signal);
+ void hash(Signal* signal);
+ void initApiConnect(Signal* signal);
+ void initApiConnectRec(Signal* signal,
+ ApiConnectRecord * const regApiPtr,
+ bool releaseIndexOperations = false);
+ void initattrbuf(Signal* signal);
+ void initdatabuf(Signal* signal);
+ void initgcp(Signal* signal);
+ void inithost(Signal* signal);
+ void initialiseScanrec(Signal* signal);
+ void initialiseScanFragrec(Signal* signal);
+ void initialiseScanOprec(Signal* signal);
+ void initTable(Signal* signal);
+ void initialiseTcConnect(Signal* signal);
+ void linkApiToGcp(Signal* signal);
+ void linkGciInGcilist(Signal* signal);
+ void linkKeybuf(Signal* signal);
+ void linkTcInConnectionlist(Signal* signal);
+ void releaseAbortResources(Signal* signal);
+ void releaseApiCon(Signal* signal, UintR aApiConnectPtr);
+ void releaseApiConCopy(Signal* signal);
+ void releaseApiConnectFail(Signal* signal);
+ void releaseAttrinfo(Signal* signal);
+ void releaseGcp(Signal* signal);
+ void releaseKeys(Signal* signal);
+ void releaseSimpleRead(Signal* signal);
+ void releaseDirtyWrite(Signal* signal);
+ void releaseTcCon(Signal* signal);
+ void releaseTcConnectFail(Signal* signal);
+ void releaseTransResources(Signal* signal);
+ void saveAttrbuf(Signal* signal);
+ void seizeApiConnect(Signal* signal);
+ void seizeApiConnectCopy(Signal* signal);
+ void seizeApiConnectFail(Signal* signal);
+ void seizeDatabuf(Signal* signal);
+ void seizeGcp(Signal* signal);
+ void seizeTcConnect(Signal* signal);
+ void seizeTcConnectFail(Signal* signal);
+ void sendApiCommit(Signal* signal);
+ void sendAttrinfo(Signal* signal,
+ UintR TattrinfoPtr,
+ AttrbufRecord * const regAttrPtr,
+ UintR TBref);
+ void sendContinueTimeOutControl(Signal* signal, Uint32 TapiConPtr);
+ void sendInitialiseRecords(Signal* signal, UintR Tnext);
+ void sendKeyinfo(Signal* signal, BlockReference TBRef, Uint32 len);
+ void sendlqhkeyreq(Signal* signal, BlockReference TBRef);
+ void sendSystemError(Signal* signal);
+ void sendtckeyconf(Signal* signal, UintR TcommitFlag);
+ void sendTcIndxConf(Signal* signal, UintR TcommitFlag);
+ void unlinkApiConnect(Signal* signal);
+ void unlinkGcp(Signal* signal);
+ void unlinkReadyTcCon(Signal* signal);
+ void handleFailedOperation(Signal* signal,
+ const LqhKeyRef * const lqhKeyRef,
+ bool gotLqhKeyRef);
+ void markOperationAborted(ApiConnectRecord * const regApiPtr,
+ TcConnectRecord * const regTcPtr);
+ void clearCommitAckMarker(ApiConnectRecord * const regApiPtr,
+ TcConnectRecord * const regTcPtr);
+ // Trigger and index handling
+ bool saveINDXKEYINFO(Signal* signal,
+ TcIndexOperation* indexOp,
+ const Uint32 *src,
+ Uint32 len);
+ bool receivedAllINDXKEYINFO(TcIndexOperation* indexOp);
+ bool saveINDXATTRINFO(Signal* signal,
+ TcIndexOperation* indexOp,
+ const Uint32 *src,
+ Uint32 len);
+ bool receivedAllINDXATTRINFO(TcIndexOperation* indexOp);
+ bool saveTRANSID_AI(Signal* signal,
+ TcIndexOperation* indexOp,
+ const Uint32 *src,
+ Uint32 len);
+ bool receivedAllTRANSID_AI(TcIndexOperation* indexOp);
+ void readIndexTable(Signal* signal,
+ ApiConnectRecord* regApiPtr,
+ TcIndexOperation* indexOp);
+ void executeIndexOperation(Signal* signal,
+ ApiConnectRecord* regApiPtr,
+ TcIndexOperation* indexOp);
+ bool seizeIndexOperation(ApiConnectRecord* regApiPtr,
+ TcIndexOperationPtr& indexOpPtr);
+ void releaseIndexOperation(ApiConnectRecord* regApiPtr,
+ TcIndexOperation* indexOp);
+ void releaseAllSeizedIndexOperations(ApiConnectRecord* regApiPtr);
+ void setupIndexOpReturn(ApiConnectRecord* regApiPtr,
+ TcConnectRecord* regTcPtr);
+
+ void saveTriggeringOpState(Signal* signal,
+ TcConnectRecord* trigOp);
+ void restoreTriggeringOpState(Signal* signal,
+ TcConnectRecord* trigOp);
+ void continueTriggeringOp(Signal* signal,
+ TcConnectRecord* trigOp);
+
+ void scheduleFiredTrigger(ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr);
+ void executeTriggers(Signal* signal, ApiConnectRecordPtr* transPtr);
+ void executeTrigger(Signal* signal,
+ TcFiredTriggerData* firedTriggerData,
+ ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr);
+ void executeIndexTrigger(Signal* signal,
+ TcDefinedTriggerData* definedTriggerData,
+ TcFiredTriggerData* firedTriggerData,
+ ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr);
+ void insertIntoIndexTable(Signal* signal,
+ TcFiredTriggerData* firedTriggerData,
+ ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr,
+ TcIndexData* indexData,
+ bool holdOperation = false);
+ void deleteFromIndexTable(Signal* signal,
+ TcFiredTriggerData* firedTriggerData,
+ ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr,
+ TcIndexData* indexData,
+ bool holdOperation = false);
+ void releaseFiredTriggerData(DLFifoList<TcFiredTriggerData>* triggers);
+ // Generated statement blocks
+ void warningHandlerLab(Signal* signal);
+ void systemErrorLab(Signal* signal);
+ void sendSignalErrorRefuseLab(Signal* signal);
+ void scanTabRefLab(Signal* signal, Uint32 errCode);
+ void diFcountReqLab(Signal* signal);
+ void signalErrorRefuseLab(Signal* signal);
+ void abort080Lab(Signal* signal);
+ void packKeyData000Lab(Signal* signal, BlockReference TBRef);
+ void abortScanLab(Signal* signal, Uint32 errCode);
+ void sendAbortedAfterTimeout(Signal* signal, int Tcheck);
+ void abort010Lab(Signal* signal);
+ void abort015Lab(Signal* signal);
+ void packLqhkeyreq(Signal* signal, BlockReference TBRef);
+ void packLqhkeyreq040Lab(Signal* signal,
+ UintR anAttrBufIndex,
+ BlockReference TBRef);
+ void packLqhkeyreq040Lab(Signal* signal);
+ void returnFromQueuedDeliveryLab(Signal* signal);
+ void startTakeOverLab(Signal* signal);
+ void toCompleteHandlingLab(Signal* signal);
+ void toCommitHandlingLab(Signal* signal);
+ void toAbortHandlingLab(Signal* signal);
+ void abortErrorLab(Signal* signal);
+ void nodeTakeOverCompletedLab(Signal* signal);
+ void ndbsttorry010Lab(Signal* signal);
+ void commit020Lab(Signal* signal);
+ void complete010Lab(Signal* signal);
+ void releaseAtErrorLab(Signal* signal);
+ void seizeDatabuferrorLab(Signal* signal);
+ void scanAttrinfoLab(Signal* signal, UintR Tlen);
+ void seizeAttrbuferrorLab(Signal* signal);
+ void attrinfoDihReceivedLab(Signal* signal);
+ void aiErrorLab(Signal* signal);
+ void attrinfo020Lab(Signal* signal);
+ void scanReleaseResourcesLab(Signal* signal);
+ void scanCompletedLab(Signal* signal);
+ void scanFragError(Signal* signal, Uint32 errorCode);
+ void diverify010Lab(Signal* signal);
+ void returnInitialiseRecordsLab(Signal* signal);
+ void intstartphase2x010Lab(Signal* signal);
+ void intstartphase3x010Lab(Signal* signal);
+ void sttorryLab(Signal* signal);
+ void abortBeginErrorLab(Signal* signal);
+ void tabStateErrorLab(Signal* signal);
+ void wrongSchemaVersionErrorLab(Signal* signal);
+ void noFreeConnectionErrorLab(Signal* signal);
+ void tckeyreq050Lab(Signal* signal);
+ void timeOutFoundLab(Signal* signal, UintR anAdd);
+ void completeTransAtTakeOverLab(Signal* signal, UintR TtakeOverInd);
+ void completeTransAtTakeOverDoLast(Signal* signal, UintR TtakeOverInd);
+ void completeTransAtTakeOverDoOne(Signal* signal, UintR TtakeOverInd);
+ void timeOutLoopStartLab(Signal* signal, Uint32 apiConnectPtr);
+ void initialiseRecordsLab(Signal* signal, UintR Tdata0);
+ void tckeyreq020Lab(Signal* signal);
+ void intstartphase2x020Lab(Signal* signal);
+ void intstartphase1x010Lab(Signal* signal);
+ void startphase1x010Lab(Signal* signal);
+
+ void lqhKeyConf_checkTransactionState(Signal * signal,
+ ApiConnectRecord * const regApiPtr);
+
+ void checkDropTab(Signal* signal);
+
+ void checkScanActiveInFailedLqh(Signal* signal,
+ Uint32 scanPtrI,
+ Uint32 failedNodeId);
+
+ // Initialisation
+ void initData();
+ void initRecords();
+
+ // Transit signals
+
+
+ ApiConnectRecord *apiConnectRecord;
+ ApiConnectRecordPtr apiConnectptr;
+ UintR capiConnectFilesize;
+
+ TcConnectRecord *tcConnectRecord;
+ TcConnectRecordPtr tcConnectptr;
+ UintR ctcConnectFilesize;
+
+ CacheRecord *cacheRecord;
+ CacheRecordPtr cachePtr;
+ UintR ccacheFilesize;
+
+ AttrbufRecord *attrbufRecord;
+ AttrbufRecordPtr attrbufptr;
+ UintR cattrbufFilesize;
+
+ HostRecord *hostRecord;
+ HostRecordPtr hostptr;
+ UintR chostFilesize;
+
+ GcpRecord *gcpRecord;
+ GcpRecordPtr gcpPtr;
+ UintR cgcpFilesize;
+
+ TableRecord *tableRecord;
+ UintR ctabrecFilesize;
+
+ UintR thashValue;
+ UintR tdistrHashValue;
+
+ UintR ttransid_ptr;
+ UintR cfailure_nr;
+ UintR coperationsize;
+ UintR ctcTimer;
+
+ ApiConnectRecordPtr tmpApiConnectptr;
+ UintR tcheckGcpId;
+ UintR cconcurrentOp;
+
+ UintR cattrinfoCount;
+ UintR ctransCount;
+ UintR ccommitCount;
+ UintR creadCount;
+
+ UintR csimpleReadCount;
+ UintR cwriteCount;
+ UintR cabortCount;
+ Uint16 cownNodeid;
+ Uint16 terrorCode;
+
+ UintR cfirstfreeAttrbuf;
+ UintR cfirstfreeTcConnect;
+ UintR cfirstfreeApiConnectCopy;
+ UintR cfirstfreeCacheRec;
+
+ UintR cfirstgcp;
+ UintR clastgcp;
+ UintR cfirstfreeGcp;
+ UintR cfirstfreeScanrec;
+
+ TableRecordPtr tabptr;
+ UintR cfirstfreeApiConnectFail;
+ UintR cfirstfreeApiConnect;
+
+ UintR cfirstfreeDatabuf;
+ BlockReference cdihblockref;
+ BlockReference cownref; /* OWN BLOCK REFERENCE */
+
+ ApiConnectRecordPtr timeOutptr;
+
+ ScanRecord *scanRecord;
+ ScanRecordPtr scanptr;
+ UintR cscanrecFileSize;
+
+ ScanOperationRecord *scanOperationRecord;
+ ScanOperationRecordPtr scanOpptr;
+ UintR cscanOprecFileSize;
+
+ ScanFragRec *scanFragmentRecord;
+ ScanFragRecPtr scanFragptr;
+ UintR cscanFragrecFileSize;
+
+ UintR cfirstfreeScanOprec;
+ UintR cnoFreeScanOprec;
+ UintR cfirstfreeScanFragrec;
+ UintR cdatabufFilesize;
+
+ BlockReference cdictblockref;
+ BlockReference cerrorBlockref;
+ BlockReference clqhblockref;
+ BlockReference cndbcntrblockref;
+
+ Uint16 csignalKey;
+ Uint16 csystemnodes;
+ Uint16 cnodes[4];
+ NodeId cmasterNodeId;
+ UintR cnoParallelTakeOver;
+ TimeOutCheckState ctimeOutCheckFragActive;
+
+ UintR ctimeOutCheckFragCounter;
+ UintR ctimeOutCheckCounter;
+ UintR ctimeOutValue;
+ UintR ctimeOutCheckDelay;
+ Uint32 ctimeOutCheckHeartbeat;
+ Uint32 ctimeOutCheckLastHeartbeat;
+ Uint32 ctimeOutMissedHeartbeats;
+ Uint32 c_appl_timeout_value;
+
+ SystemStartState csystemStart;
+ TimeOutCheckState ctimeOutCheckActive;
+
+ BlockReference capiFailRef;
+ UintR cpackedListIndex;
+ Uint16 cpackedList[MAX_NODES];
+ UintR capiConnectClosing[MAX_NODES];
+ UintR con_lineNodes;
+
+ DatabufRecord *databufRecord;
+ DatabufRecordPtr databufptr;
+ DatabufRecordPtr tmpDatabufptr;
+
+ UintR treqinfo;
+ UintR ttransid1;
+ UintR ttransid2;
+
+ UintR tabortInd;
+
+ NodeId tnodeid;
+ BlockReference tblockref;
+
+ LqhTransConf::OperationStatus ttransStatus;
+ UintR ttcOprec;
+ NodeId tfailedNodeId;
+ Uint8 tcurrentReplicaNo;
+ Uint8 tpad1;
+
+ UintR tgci;
+ UintR tapplRef;
+ UintR tapplOprec;
+
+ UintR tindex;
+ UintR tmaxData;
+ UintR tmp;
+
+ UintR tnodes;
+ BlockReference tusersblkref;
+ UintR tuserpointer;
+ UintR tloadCode;
+
+ UintR tconfig1;
+ UintR tconfig2;
+
+ UintR cdata[32];
+ UintR ctransidFailHash[512];
+ UintR ctcConnectFailHash[1024];
+
+ /**
+ * Commit Ack handling
+ */
+public:
+ struct CommitAckMarker {
+ Uint32 transid1;
+ Uint32 transid2;
+ union { Uint32 nextPool; Uint32 nextHash; };
+ Uint32 prevHash;
+ Uint32 apiConnectPtr;
+ Uint16 apiNodeId;
+ Uint16 noOfLqhs;
+ Uint16 lqhNodeId[MAX_REPLICAS];
+
+ inline bool equal(const CommitAckMarker & p) const {
+ return ((p.transid1 == transid1) && (p.transid2 == transid2));
+ }
+
+ inline Uint32 hashValue() const {
+ return transid1;
+ }
+ };
+private:
+ typedef Ptr<CommitAckMarker> CommitAckMarkerPtr;
+ typedef DLHashTable<CommitAckMarker>::Iterator CommitAckMarkerIterator;
+
+ ArrayPool<CommitAckMarker> m_commitAckMarkerPool;
+ DLHashTable<CommitAckMarker> m_commitAckMarkerHash;
+
+ void execTC_COMMIT_ACK(Signal* signal);
+ void sendRemoveMarkers(Signal*, const CommitAckMarker *);
+ void sendRemoveMarker(Signal* signal,
+ NodeId nodeId,
+ Uint32 transid1,
+ Uint32 transid2);
+ void removeMarkerForFailedAPI(Signal* signal, Uint32 nodeId, Uint32 bucket);
+
+ bool getAllowStartTransaction() const {
+ if(getNodeState().getSingleUserMode())
+ return true;
+ return getNodeState().startLevel < NodeState::SL_STOPPING_2;
+ }
+
+ void checkAbortAllTimeout(Signal* signal, Uint32 sleepTime);
+ struct AbortAllRecord {
+ AbortAllRecord(){ clientRef = 0; }
+ Uint32 clientData;
+ BlockReference clientRef;
+
+ Uint32 oldTimeOutValue;
+ };
+ AbortAllRecord c_abortRec;
+
+ /************************** API CONNECT RECORD ***********************/
+ /* *******************************************************************/
+ /* THE API CONNECT RECORD CONTAINS THE CONNECTION RECORD TO WHICH THE*/
+ /* APPLICATION CONNECTS. THE APPLICATION CAN SEND ONE OPERATION AT A */
+ /* TIME. IT CAN SEND A NEW OPERATION IMMEDIATELY AFTER SENDING THE */
+ /* PREVIOUS OPERATION. THEREBY SEVERAL OPERATIONS CAN BE ACTIVE IN */
+ /* ONE TRANSACTION WITHIN TC. THIS IS ACHIEVED BY USING THE API */
+ /* CONNECT RECORD. EACH ACTIVE OPERATION IS HANDLED BY THE TC */
+ /* CONNECT RECORD. AS SOON AS THE TC CONNECT RECORD HAS SENT THE */
+ /* REQUEST TO THE LQH IT IS READY TO RECEIVE NEW OPERATIONS. THE */
+ /* LQH CONNECT RECORD TAKES CARE OF WAITING FOR AN OPERATION TO */
+ /* COMPLETE. WHEN AN OPERATION HAS COMPLETED ON THE LQH CONNECT */
+ /* RECORD A NEW OPERATION CAN BE STARTED ON THIS LQH CONNECT RECORD. */
+ /*******************************************************************>*/
+ /* */
+ /* API CONNECT RECORD ALIGNED TO BE 256 BYTES */
+ /*******************************************************************>*/
+ /************************** TC CONNECT RECORD ************************/
+ /* *******************************************************************/
+ /* TC CONNECT RECORD KEEPS ALL INFORMATION TO CARRY OUT A TRANSACTION*/
+ /* THE TRANSACTION CONTROLLER ESTABLISHES CONNECTIONS TO DIFFERENT */
+ /* BLOCKS TO CARRY OUT THE TRANSACTION. THERE CAN BE SEVERAL RECORDS */
+ /* PER ACTIVE TRANSACTION. THE TC CONNECT RECORD COOPERATES WITH THE */
+ /* API CONNECT RECORD FOR COMMUNICATION WITH THE API AND WITH THE */
+ /* LQH CONNECT RECORD FOR COMMUNICATION WITH THE LQH'S INVOLVED IN */
+ /* THE TRANSACTION. TC CONNECT RECORD IS PERMANENTLY CONNECTED TO A */
+ /* RECORD IN DICT AND ONE IN DIH. IT CONTAINS A LIST OF ACTIVE LQH */
+ /* CONNECT RECORDS AND A LIST OF STARTED BUT NOT ACTIVE LQH CONNECT */
+ /* RECORDS. IT DOES ALSO CONTAIN A LIST OF ALL OPERATIONS THAT ARE */
+ /* EXECUTED WITH THE TC CONNECT RECORD. */
+ /*******************************************************************>*/
+ /* TC_CONNECT RECORD ALIGNED TO BE 128 BYTES */
+ /*******************************************************************>*/
+ UintR cfirstfreeTcConnectFail;
+
+ /* POINTER FOR THE LQH RECORD*/
+ /* ************************ HOST RECORD ********************************* */
+ /********************************************************/
+ /* THIS RECORD CONTAINS ALIVE-STATUS ON ALL NODES IN THE*/
+ /* SYSTEM */
+ /********************************************************/
+ /* THIS RECORD IS ALIGNED TO BE 8 BYTES. */
+ /********************************************************/
+ /* ************************ TABLE RECORD ******************************** */
+ /********************************************************/
+ /* THIS RECORD CONTAINS THE CURRENT SCHEMA VERSION OF */
+ /* ALL TABLES IN THE SYSTEM. */
+ /********************************************************/
+ /*-------------------------------------------------------------------------*/
+ /* THE TC CONNECTION USED BY THIS SCAN. */
+ /*-------------------------------------------------------------------------*/
+ /*-------------------------------------------------------------------------*/
+ /* LENGTH READ FOR A PARTICULAR SCANNED OPERATION. */
+ /*-------------------------------------------------------------------------*/
+ /*-------------------------------------------------------------------------*/
+ /* REFERENCE TO THE SCAN RECORD FOR THIS SCAN PROCESS. */
+ /*-------------------------------------------------------------------------*/
+ /* *********************************************************************** */
+ /* ******$ DATA BUFFER ******$ */
+ /* */
+ /* THIS BUFFER IS USED AS A GENERAL DATA STORAGE. */
+ /* *********************************************************************** */
+ /* *********************************************************************** */
+ /* ******$ ATTRIBUTE INFORMATION RECORD ******$ */
+ /*
+ CAN CONTAIN ONE (1) ATTRINFO SIGNAL. ONE SIGNAL CONTAINS 24 ATTR.
+ INFO WORDS. BUT 32 ELEMENTS ARE USED TO MAKE PLEX HAPPY.
+ SOME OF THE ELEMENTS ARE USED TO THE FOLLOWING THINGS:
+ DATA LENGHT IN THIS RECORD IS STORED IN THE ELEMENT INDEXED BY
+ ZINBUF_DATA_LEN.
+ NEXT FREE ATTRBUF IS POINTED OUT BY THE ELEMENT INDEXED BY
+ PREVIOUS ATTRBUF IS POINTED OUT BY THE ELEMENT INDEXED BY ZINBUF_PREV
+ (NOT USED YET).
+ NEXT ATTRBUF IS POINTED OUT BY THE ELEMENT INDEXED BY ZINBUF_NEXT.
+ */
+ /* ********************************************************************** */
+ /**************************************************************************/
+ /* GLOBAL CHECKPOINT INFORMATION RECORD */
+ /* */
+ /* THIS RECORD IS USED TO STORE THE GCP NUMBER AND A COUNTER */
+ /* DURING THE COMPLETION PHASE OF THE TRANSACTION */
+ /**************************************************************************/
+ /* */
+ /* GCP RECORD ALIGNED TO BE 32 BYTES */
+ /**************************************************************************/
+ /**************************************************************************/
+ /* TC_FAIL_RECORD */
+ /* THIS RECORD IS USED WHEN HANDLING TAKE OVER OF ANOTHER FAILED TC NODE.*/
+ /**************************************************************************/
+ TcFailRecord *tcFailRecord;
+ TcFailRecordPtr tcNodeFailptr;
+ /**************************************************************************/
+ // Temporary variables that are not allowed to use for storage between
+ // signals. They
+ // can only be used in a signal to transfer values between subroutines.
+ // In the long run
+ // those variables should be removed and exchanged for stack
+ // variable communication.
+ /**************************************************************************/
+};
+#endif
diff --git a/ndb/src/kernel/blocks/dbtc/DbtcInit.cpp b/ndb/src/kernel/blocks/dbtc/DbtcInit.cpp
new file mode 100644
index 00000000000..1ac5273188c
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtc/DbtcInit.cpp
@@ -0,0 +1,357 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTC_C
+#include "Dbtc.hpp"
+#include <pc.hpp>
+#include <ndb_limits.h>
+#include <Properties.hpp>
+#include <Configuration.hpp>
+#include <new>
+
+#define DEBUG(x) { ndbout << "TC::" << x << endl; }
+
+
+void Dbtc::initData()
+{
+ cattrbufFilesize = ZATTRBUF_FILESIZE;
+ capiConnectFilesize = ZAPI_CONNECT_FILESIZE;
+ ccacheFilesize = ZAPI_CONNECT_FILESIZE;
+ chostFilesize = MAX_NODES;
+ cdatabufFilesize = ZDATABUF_FILESIZE;
+ cgcpFilesize = ZGCP_FILESIZE;
+ cscanrecFileSize = ZSCANREC_FILE_SIZE;
+ cscanFragrecFileSize = ZSCAN_FRAGREC_FILE_SIZE;
+ cscanOprecFileSize = ZSCAN_OPREC_FILE_SIZE;
+ ctabrecFilesize = ZTABREC_FILESIZE;
+ ctcConnectFilesize = ZTC_CONNECT_FILESIZE;
+ cdihblockref = DBDIH_REF;
+ cdictblockref = DBDICT_REF;
+ clqhblockref = DBLQH_REF;
+ cerrorBlockref = NDBCNTR_REF;
+
+ cacheRecord = 0;
+ apiConnectRecord = 0;
+ tcConnectRecord = 0;
+ hostRecord = 0;
+ tableRecord = 0;
+ scanRecord = 0;
+ scanOperationRecord = 0;
+ scanFragmentRecord = 0;
+ databufRecord = 0;
+ attrbufRecord = 0;
+ gcpRecord = 0;
+ tcFailRecord = 0;
+ c_apiConTimer = 0;
+ c_apiConTimer_line = 0;
+ // Records with constant sizes
+ tcFailRecord = (TcFailRecord*)allocRecord("TcFailRecord",
+ sizeof(TcFailRecord), 1);
+
+ // Variables
+ ctcTimer = 0;
+
+ // Trigger and index pools
+ c_theDefinedTriggerPool.setSize(c_maxNumberOfDefinedTriggers);
+ c_theFiredTriggerPool.setSize(c_maxNumberOfFiredTriggers);
+ c_theIndexPool.setSize(c_maxNumberOfIndexes);
+ c_theIndexOperationPool.setSize(c_maxNumberOfIndexOperations);
+ c_theSeizedIndexOperationPool.setSize(c_maxNumberOfIndexOperations);
+ c_theAttributeBufferPool.setSize(c_transactionBufferSpace);
+}//Dbtc::initData()
+
+void Dbtc::initRecords()
+{
+ // Records with dynamic sizes
+ cacheRecord = (CacheRecord*)allocRecord("CacheRecord",
+ sizeof(CacheRecord),
+ ccacheFilesize);
+
+ apiConnectRecord = (ApiConnectRecord*)allocRecord("ApiConnectRecord",
+ sizeof(ApiConnectRecord),
+ capiConnectFilesize);
+
+ for(unsigned i = 0; i<capiConnectFilesize; i++) {
+ void * p = &apiConnectRecord[i];
+ new (p) ApiConnectRecord(c_theFiredTriggerPool,
+ c_theSeizedIndexOperationPool);
+ }
+ // Init all fired triggers
+ DLFifoList<TcFiredTriggerData> triggers(c_theFiredTriggerPool);
+ FiredTriggerPtr tptr;
+ while(triggers.seize(tptr) == true) {
+ new (tptr.p) TcFiredTriggerData(c_theAttributeBufferPool);
+ }
+ triggers.release();
+
+ /*
+ // Init all index records
+ ArrayList<TcIndexData> indexes(c_theIndexPool);
+ TcIndexDataPtr iptr;
+ while(indexes.seize(iptr) == true) {
+ new (iptr.p) TcIndexData(c_theAttrInfoListPool);
+ }
+ indexes.release();
+ */
+
+ // Init all index operation records
+ ArrayList<TcIndexOperation> indexOps(c_theIndexOperationPool);
+ TcIndexOperationPtr ioptr;
+ while(indexOps.seize(ioptr) == true) {
+ new (ioptr.p) TcIndexOperation(c_theAttributeBufferPool);
+ }
+ indexOps.release();
+
+ c_apiConTimer = (UintR*)allocRecord("ApiConTimer",
+ sizeof(UintR),
+ capiConnectFilesize);
+
+ c_apiConTimer_line = (UintR*)allocRecord("ApiConTimer_line",
+ sizeof(UintR),
+ capiConnectFilesize);
+
+ tcConnectRecord = (TcConnectRecord*)allocRecord("TcConnectRecord",
+ sizeof(TcConnectRecord),
+ ctcConnectFilesize);
+
+ m_commitAckMarkerPool.setSize(capiConnectFilesize);
+ m_commitAckMarkerHash.setSize(512);
+
+ hostRecord = (HostRecord*)allocRecord("HostRecord",
+ sizeof(HostRecord),
+ chostFilesize);
+
+ tableRecord = (TableRecord*)allocRecord("TableRecord",
+ sizeof(TableRecord),
+ ctabrecFilesize);
+
+ scanRecord = (ScanRecord*)allocRecord("ScanRecord",
+ sizeof(ScanRecord),
+ cscanrecFileSize);
+
+ scanOperationRecord = (ScanOperationRecord*)
+ allocRecord("ScanOperationRecord",
+ sizeof(ScanOperationRecord),
+ cscanOprecFileSize);
+
+ scanFragmentRecord = (ScanFragRec*)
+ allocRecord("ScanFragRec",
+ sizeof(ScanFragRec),
+ cscanFragrecFileSize);
+
+ databufRecord = (DatabufRecord*)allocRecord("DatabufRecord",
+ sizeof(DatabufRecord),
+ cdatabufFilesize);
+
+ attrbufRecord = (AttrbufRecord*)allocRecord("AttrbufRecord",
+ sizeof(AttrbufRecord),
+ cattrbufFilesize);
+
+ gcpRecord = (GcpRecord*)allocRecord("GcpRecord",
+ sizeof(GcpRecord),
+ cgcpFilesize);
+
+}//Dbtc::initRecords()
+
+Dbtc::Dbtc(const class Configuration & conf):
+ SimulatedBlock(DBTC, conf),
+ c_theDefinedTriggers(c_theDefinedTriggerPool),
+ c_maxNumberOfDefinedTriggers(0),
+ c_maxNumberOfFiredTriggers(0),
+ c_theIndexes(c_theIndexPool),
+ c_maxNumberOfIndexes(0),
+ c_theIndexOperations(c_theIndexOperationPool),
+ c_maxNumberOfIndexOperations(0),
+ m_commitAckMarkerHash(m_commitAckMarkerPool)
+{
+
+ BLOCK_CONSTRUCTOR(Dbtc);
+
+ const Properties * p = conf.getOwnProperties();
+ ndbrequire(p != 0);
+
+ Uint32 transactionBufferMemory = 0;
+ Uint32 maxNoOfIndexes = 0, maxNoOfConcurrentIndexOperations = 0;
+ Uint32 maxNoOfTriggers = 0, maxNoOfFiredTriggers = 0;
+
+ p->get("TransactionBufferMemory", &transactionBufferMemory);
+ p->get("MaxNoOfIndexes", &maxNoOfIndexes);
+ p->get("MaxNoOfConcurrentIndexOperations", &maxNoOfConcurrentIndexOperations);
+ p->get("MaxNoOfTriggers", &maxNoOfTriggers);
+ p->get("MaxNoOfFiredTriggers", &maxNoOfFiredTriggers);
+
+ c_transactionBufferSpace =
+ transactionBufferMemory / AttributeBuffer::getSegmentSize();
+ c_maxNumberOfIndexes = maxNoOfIndexes;
+ c_maxNumberOfIndexOperations = maxNoOfConcurrentIndexOperations;
+ c_maxNumberOfDefinedTriggers = maxNoOfTriggers;
+ c_maxNumberOfFiredTriggers = maxNoOfFiredTriggers;
+
+ // Transit signals
+ addRecSignal(GSN_PACKED_SIGNAL, &Dbtc::execPACKED_SIGNAL);
+ addRecSignal(GSN_ABORTED, &Dbtc::execABORTED);
+ addRecSignal(GSN_ATTRINFO, &Dbtc::execATTRINFO);
+ addRecSignal(GSN_CONTINUEB, &Dbtc::execCONTINUEB);
+ addRecSignal(GSN_KEYINFO, &Dbtc::execKEYINFO);
+ addRecSignal(GSN_SCAN_TABINFO, &Dbtc::execSCAN_TABINFO);
+ addRecSignal(GSN_SCAN_NEXTREQ, &Dbtc::execSCAN_NEXTREQ);
+ addRecSignal(GSN_SCAN_PROCREQ, &Dbtc::execSCAN_PROCREQ);
+ addRecSignal(GSN_SCAN_PROCCONF, &Dbtc::execSCAN_PROCCONF);
+ addRecSignal(GSN_TAKE_OVERTCREQ, &Dbtc::execTAKE_OVERTCREQ);
+ addRecSignal(GSN_TAKE_OVERTCCONF, &Dbtc::execTAKE_OVERTCCONF);
+ addRecSignal(GSN_LQHKEYREF, &Dbtc::execLQHKEYREF);
+
+ // Received signals
+
+ addRecSignal(GSN_DUMP_STATE_ORD, &Dbtc::execDUMP_STATE_ORD);
+ addRecSignal(GSN_SEND_PACKED, &Dbtc::execSEND_PACKED);
+ addRecSignal(GSN_SCAN_HBREP, &Dbtc::execSCAN_HBREP);
+ addRecSignal(GSN_COMPLETED, &Dbtc::execCOMPLETED);
+ addRecSignal(GSN_COMMITTED, &Dbtc::execCOMMITTED);
+ addRecSignal(GSN_DIGETPRIMCONF, &Dbtc::execDIGETPRIMCONF);
+ addRecSignal(GSN_DIGETPRIMREF, &Dbtc::execDIGETPRIMREF);
+ addRecSignal(GSN_DISEIZECONF, &Dbtc::execDISEIZECONF);
+ addRecSignal(GSN_DIVERIFYCONF, &Dbtc::execDIVERIFYCONF);
+ addRecSignal(GSN_DI_FCOUNTCONF, &Dbtc::execDI_FCOUNTCONF);
+ addRecSignal(GSN_DI_FCOUNTREF, &Dbtc::execDI_FCOUNTREF);
+ addRecSignal(GSN_GCP_NOMORETRANS, &Dbtc::execGCP_NOMORETRANS);
+ addRecSignal(GSN_LQHKEYCONF, &Dbtc::execLQHKEYCONF);
+ addRecSignal(GSN_NDB_STTOR, &Dbtc::execNDB_STTOR);
+ addRecSignal(GSN_READ_NODESCONF, &Dbtc::execREAD_NODESCONF);
+ addRecSignal(GSN_READ_NODESREF, &Dbtc::execREAD_NODESREF);
+ addRecSignal(GSN_STTOR, &Dbtc::execSTTOR);
+ addRecSignal(GSN_TC_COMMITREQ, &Dbtc::execTC_COMMITREQ);
+ addRecSignal(GSN_TC_CLOPSIZEREQ, &Dbtc::execTC_CLOPSIZEREQ);
+ addRecSignal(GSN_TCGETOPSIZEREQ, &Dbtc::execTCGETOPSIZEREQ);
+ addRecSignal(GSN_TCKEYREQ, &Dbtc::execTCKEYREQ);
+ addRecSignal(GSN_TCRELEASEREQ, &Dbtc::execTCRELEASEREQ);
+ addRecSignal(GSN_TCSEIZEREQ, &Dbtc::execTCSEIZEREQ);
+ addRecSignal(GSN_TCROLLBACKREQ, &Dbtc::execTCROLLBACKREQ);
+ addRecSignal(GSN_TC_HBREP, &Dbtc::execTC_HBREP);
+ addRecSignal(GSN_TC_SCHVERREQ, &Dbtc::execTC_SCHVERREQ);
+ addRecSignal(GSN_SCAN_TABREQ, &Dbtc::execSCAN_TABREQ);
+ addRecSignal(GSN_SCAN_FRAGCONF, &Dbtc::execSCAN_FRAGCONF);
+ addRecSignal(GSN_SCAN_FRAGREF, &Dbtc::execSCAN_FRAGREF);
+ addRecSignal(GSN_SIZEALT_REP, &Dbtc::execSIZEALT_REP);
+ addRecSignal(GSN_LQH_TRANSCONF, &Dbtc::execLQH_TRANSCONF);
+ addRecSignal(GSN_COMPLETECONF, &Dbtc::execCOMPLETECONF);
+ addRecSignal(GSN_COMMITCONF, &Dbtc::execCOMMITCONF);
+ addRecSignal(GSN_ABORTCONF, &Dbtc::execABORTCONF);
+ addRecSignal(GSN_NODE_FAILREP, &Dbtc::execNODE_FAILREP);
+ addRecSignal(GSN_INCL_NODEREQ, &Dbtc::execINCL_NODEREQ);
+ addRecSignal(GSN_TIME_SIGNAL, &Dbtc::execTIME_SIGNAL);
+ addRecSignal(GSN_API_FAILREQ, &Dbtc::execAPI_FAILREQ);
+ addRecSignal(GSN_SET_VAR_REQ, &Dbtc::execSET_VAR_REQ);
+
+ addRecSignal(GSN_TC_COMMIT_ACK, &Dbtc::execTC_COMMIT_ACK);
+ addRecSignal(GSN_ABORT_ALL_REQ, &Dbtc::execABORT_ALL_REQ);
+
+ addRecSignal(GSN_CREATE_TRIG_REQ, &Dbtc::execCREATE_TRIG_REQ);
+ addRecSignal(GSN_DROP_TRIG_REQ, &Dbtc::execDROP_TRIG_REQ);
+ addRecSignal(GSN_FIRE_TRIG_ORD, &Dbtc::execFIRE_TRIG_ORD);
+ addRecSignal(GSN_TRIG_ATTRINFO, &Dbtc::execTRIG_ATTRINFO);
+
+ addRecSignal(GSN_CREATE_INDX_REQ, &Dbtc::execCREATE_INDX_REQ);
+ addRecSignal(GSN_DROP_INDX_REQ, &Dbtc::execDROP_INDX_REQ);
+ addRecSignal(GSN_TCINDXREQ, &Dbtc::execTCINDXREQ);
+ addRecSignal(GSN_INDXKEYINFO, &Dbtc::execINDXKEYINFO);
+ addRecSignal(GSN_INDXATTRINFO, &Dbtc::execINDXATTRINFO);
+ addRecSignal(GSN_ALTER_INDX_REQ, &Dbtc::execALTER_INDX_REQ);
+
+ addRecSignal(GSN_TRANSID_AI_R, &Dbtc::execTRANSID_AI_R);
+ addRecSignal(GSN_KEYINFO20_R, &Dbtc::execKEYINFO20_R);
+
+ // Index table lookup
+ addRecSignal(GSN_TCKEYCONF, &Dbtc::execTCKEYCONF);
+ addRecSignal(GSN_TCKEYREF, &Dbtc::execTCKEYREF);
+ addRecSignal(GSN_TRANSID_AI, &Dbtc::execTRANSID_AI);
+ addRecSignal(GSN_TCROLLBACKREP, &Dbtc::execTCROLLBACKREP);
+
+ //addRecSignal(GSN_CREATE_TAB_REQ, &Dbtc::execCREATE_TAB_REQ);
+ addRecSignal(GSN_DROP_TAB_REQ, &Dbtc::execDROP_TAB_REQ);
+ addRecSignal(GSN_PREP_DROP_TAB_REQ, &Dbtc::execPREP_DROP_TAB_REQ);
+ addRecSignal(GSN_WAIT_DROP_TAB_CONF, &Dbtc::execWAIT_DROP_TAB_CONF);
+
+ addRecSignal(GSN_ALTER_TAB_REQ, &Dbtc::execALTER_TAB_REQ);
+
+ initData();
+}//Dbtc::Dbtc()
+
+Dbtc::~Dbtc()
+{
+ // Records with dynamic sizes
+ deallocRecord((void **)&cacheRecord, "CacheRecord",
+ sizeof(CacheRecord),
+ ccacheFilesize);
+
+ deallocRecord((void **)&apiConnectRecord, "ApiConnectRecord",
+ sizeof(ApiConnectRecord),
+ capiConnectFilesize);
+
+ deallocRecord((void **)&tcConnectRecord, "TcConnectRecord",
+ sizeof(TcConnectRecord),
+ ctcConnectFilesize);
+
+ deallocRecord((void **)&hostRecord, "HostRecord",
+ sizeof(HostRecord),
+ chostFilesize);
+
+ deallocRecord((void **)&tableRecord, "TableRecord",
+ sizeof(TableRecord),
+ ctabrecFilesize);
+
+ deallocRecord((void **)&scanRecord, "ScanRecord",
+ sizeof(ScanRecord),
+ cscanrecFileSize);
+
+ deallocRecord((void **)&scanOperationRecord,
+ "ScanOperationRecord",
+ sizeof(ScanOperationRecord),
+ cscanOprecFileSize);
+
+ deallocRecord((void **)&scanFragmentRecord,
+ "ScanFragRec",
+ sizeof(ScanFragRec),
+ cscanFragrecFileSize);
+
+ deallocRecord((void **)&databufRecord, "DatabufRecord",
+ sizeof(DatabufRecord),
+ cdatabufFilesize);
+
+ deallocRecord((void **)&attrbufRecord, "AttrbufRecord",
+ sizeof(AttrbufRecord),
+ cattrbufFilesize);
+
+ deallocRecord((void **)&gcpRecord, "GcpRecord",
+ sizeof(GcpRecord),
+ cgcpFilesize);
+
+ deallocRecord((void **)&tcFailRecord, "TcFailRecord",
+ sizeof(TcFailRecord), 1);
+
+ deallocRecord((void **)&c_apiConTimer, "ApiConTimer",
+ sizeof(UintR),
+ capiConnectFilesize);
+
+ deallocRecord((void **)&c_apiConTimer_line, "ApiConTimer",
+ sizeof(UintR),
+ capiConnectFilesize);
+}//Dbtc::~Dbtc()
+
+BLOCK_FUNCTIONS(Dbtc);
+
+
+
diff --git a/ndb/src/kernel/blocks/dbtc/DbtcMain.cpp b/ndb/src/kernel/blocks/dbtc/DbtcMain.cpp
new file mode 100644
index 00000000000..ac8a130eb83
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtc/DbtcMain.cpp
@@ -0,0 +1,13111 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTC_C
+
+#include "Dbtc.hpp"
+#include "md5_hash.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+
+#include <signaldata/EventReport.hpp>
+#include <signaldata/TcKeyReq.hpp>
+#include <signaldata/TcKeyConf.hpp>
+#include <signaldata/TcKeyRef.hpp>
+#include <signaldata/KeyInfo.hpp>
+#include <signaldata/AttrInfo.hpp>
+#include <signaldata/TransIdAI.hpp>
+#include <signaldata/TcRollbackRep.hpp>
+#include <signaldata/TcSizeAltReq.hpp>
+#include <signaldata/SetVarReq.hpp>
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/ReadNodesConf.hpp>
+#include <signaldata/NFCompleteRep.hpp>
+#include <signaldata/LqhKey.hpp>
+#include <signaldata/TcCommit.hpp>
+#include <signaldata/TcContinueB.hpp>
+#include <signaldata/TcKeyFailConf.hpp>
+#include <signaldata/AbortAll.hpp>
+#include <signaldata/ScanFrag.hpp>
+#include <signaldata/ScanTab.hpp>
+#include <signaldata/PrepDropTab.hpp>
+#include <signaldata/DropTab.hpp>
+#include <signaldata/AlterTab.hpp>
+#include <signaldata/CreateTrig.hpp>
+#include <signaldata/DropTrig.hpp>
+#include <signaldata/FireTrigOrd.hpp>
+#include <signaldata/TrigAttrInfo.hpp>
+#include <signaldata/CreateIndx.hpp>
+#include <signaldata/DropIndx.hpp>
+#include <signaldata/AlterIndx.hpp>
+#include <signaldata/ScanTab.hpp>
+#include <signaldata/SystemError.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+#include <signaldata/DisconnectRep.hpp>
+#include <signaldata/TcHbRep.hpp>
+
+#include <signaldata/PrepDropTab.hpp>
+#include <signaldata/DropTab.hpp>
+#include <signaldata/TcIndx.hpp>
+#include <signaldata/IndxKeyInfo.hpp>
+#include <signaldata/IndxAttrInfo.hpp>
+#include <signaldata/PackedSignal.hpp>
+#include <AttributeHeader.hpp>
+#include <signaldata/DictTabInfo.hpp>
+
+#include <NdbOut.hpp>
+
+// Use DEBUG to print messages that should be
+// seen only when we debug the product
+#ifdef VM_TRACE
+#define DEBUG(x) ndbout << "DBTC: "<< x << endl;
+#else
+#define DEBUG(x)
+#endif
+
+#define INTERNAL_TRIGGER_TCKEYREQ_JBA 0
+
+void
+Dbtc::updateBuddyTimer(ApiConnectRecordPtr apiPtr)
+{
+ if (apiPtr.p->buddyPtr != RNIL) {
+ jam();
+ ApiConnectRecordPtr buddyApiPtr;
+ buddyApiPtr.i = apiPtr.p->buddyPtr;
+ ptrCheckGuard(buddyApiPtr, capiConnectFilesize, apiConnectRecord);
+ if (getApiConTimer(buddyApiPtr.i) != 0) {
+ if ((apiPtr.p->transid[0] == buddyApiPtr.p->transid[0]) &&
+ (apiPtr.p->transid[1] == buddyApiPtr.p->transid[1])) {
+ jam();
+ setApiConTimer(buddyApiPtr.i, ctcTimer, __LINE__);
+ } else {
+ jam();
+ // Not a buddy anymore since not the same transid
+ apiPtr.p->buddyPtr = RNIL;
+ }//if
+ }//if
+ }//if
+}
+
+void Dbtc::execCONTINUEB(Signal* signal)
+{
+ UintR tcase;
+
+ jamEntry();
+ tcase = signal->theData[0];
+ UintR Tdata0 = signal->theData[1];
+ UintR Tdata1 = signal->theData[2];
+ switch (tcase) {
+ case TcContinueB::ZRETURN_FROM_QUEUED_DELIVERY:
+ jam();
+ scanptr.i = Tdata0;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ scanFragptr.i = Tdata1;
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+ ndbrequire(scanFragptr.p->scanFragState ==
+ ScanFragRec::RETURNING_FROM_DELIVERY);
+ returnFromQueuedDeliveryLab(signal);
+ return;
+ case TcContinueB::ZCOMPLETE_TRANS_AT_TAKE_OVER:
+ jam();
+ tcNodeFailptr.i = Tdata0;
+ ptrCheckGuard(tcNodeFailptr, 1, tcFailRecord);
+ completeTransAtTakeOverLab(signal, Tdata1);
+ return;
+ case TcContinueB::ZCONTINUE_TIME_OUT_CONTROL:
+ jam();
+ timeOutLoopStartLab(signal, Tdata0);
+ return;
+ case TcContinueB::ZNODE_TAKE_OVER_COMPLETED:
+ jam();
+ tnodeid = Tdata0;
+ tcNodeFailptr.i = 0;
+ ptrAss(tcNodeFailptr, tcFailRecord);
+ nodeTakeOverCompletedLab(signal);
+ return;
+ case TcContinueB::ZINITIALISE_RECORDS:
+ jam();
+ initialiseRecordsLab(signal, Tdata0);
+ return;
+ case TcContinueB::ZSEND_COMMIT_LOOP:
+ jam();
+ apiConnectptr.i = Tdata0;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ tcConnectptr.i = Tdata1;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ commit020Lab(signal);
+ return;
+ case TcContinueB::ZSEND_COMPLETE_LOOP:
+ jam();
+ apiConnectptr.i = Tdata0;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ tcConnectptr.i = Tdata1;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ complete010Lab(signal);
+ return;
+ case TcContinueB::ZHANDLE_FAILED_API_NODE:
+ jam();
+ handleFailedApiNode(signal, Tdata0, Tdata1);
+ return;
+ case TcContinueB::ZTRANS_EVENT_REP:
+ jam();
+ /* -------------------------------------------------------------------- */
+ // Report information about transaction activity once per second.
+ /* -------------------------------------------------------------------- */
+ if (signal->theData[1] == 0) {
+ signal->theData[0] = EventReport::TransReportCounters;
+ signal->theData[1] = ctransCount;
+ signal->theData[2] = ccommitCount;
+ signal->theData[3] = creadCount;
+ signal->theData[4] = csimpleReadCount;
+ signal->theData[5] = cwriteCount;
+ signal->theData[6] = cattrinfoCount;
+ signal->theData[7] = cconcurrentOp;
+ signal->theData[8] = cabortCount;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 9, JBB);
+ }//if
+ ctransCount = 0;
+ ccommitCount = 0;
+ creadCount = 0;
+ csimpleReadCount = 0;
+ cwriteCount = 0;
+ cattrinfoCount = 0;
+ cabortCount = 0;
+ signal->theData[0] = TcContinueB::ZTRANS_EVENT_REP;
+ signal->theData[1] = 0;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 5000, 2);
+ return;
+ case TcContinueB::ZCONTINUE_TIME_OUT_FRAG_CONTROL:
+ jam();
+ timeOutLoopStartFragLab(signal, Tdata0);
+ return;
+ case TcContinueB::ZABORT_BREAK:
+ jam();
+ tcConnectptr.i = Tdata0;
+ apiConnectptr.i = Tdata1;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ apiConnectptr.p->counter--;
+ abort015Lab(signal);
+ return;
+ case TcContinueB::ZABORT_TIMEOUT_BREAK:
+ jam();
+ tcConnectptr.i = Tdata0;
+ apiConnectptr.i = Tdata1;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ sendAbortedAfterTimeout(signal, 1);
+ return;
+ case TcContinueB::ZHANDLE_FAILED_API_NODE_REMOVE_MARKERS:
+ jam();
+ removeMarkerForFailedAPI(signal, Tdata0, Tdata1);
+ return;
+ case TcContinueB::ZWAIT_ABORT_ALL:
+ jam();
+ checkAbortAllTimeout(signal, Tdata0);
+ return;
+ case TcContinueB::ZCHECK_SCAN_ACTIVE_FAILED_LQH:
+ jam();
+ checkScanActiveInFailedLqh(signal, Tdata0, Tdata1);
+ return;
+ case TcContinueB::CHECK_WAIT_DROP_TAB_FAILED_LQH:
+ jam();
+ checkWaitDropTabFailedLqh(signal, Tdata0, Tdata1);
+ return;
+ case TcContinueB::TRIGGER_PENDING:
+ jam();
+ ApiConnectRecordPtr transPtr;
+ transPtr.i = Tdata0;
+ ptrCheckGuard(transPtr, capiConnectFilesize, apiConnectRecord);
+ transPtr.p->triggerPending = false;
+ executeTriggers(signal, &transPtr);
+ return;
+ default:
+ ndbrequire(false);
+ }//switch
+}
+
+void Dbtc::execDIGETNODESREF(Signal* signal)
+{
+ jamEntry();
+ terrorCode = signal->theData[1];
+ releaseAtErrorLab(signal);
+}
+
+void Dbtc::execINCL_NODEREQ(Signal* signal)
+{
+ jamEntry();
+ tblockref = signal->theData[0];
+ hostptr.i = signal->theData[1];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ hostptr.p->hostStatus = HS_ALIVE;
+ hostptr.p->takeOverStatus = TOS_IDLE;
+ signal->theData[0] = cownref;
+ sendSignal(tblockref, GSN_INCL_NODECONF, signal, 1, JBB);
+}
+
+void Dbtc::execREAD_NODESREF(Signal* signal)
+{
+ jamEntry();
+ ndbrequire(false);
+}
+
+void Dbtc::execTC_SCHVERREQ(Signal* signal)
+{
+ jamEntry();
+ tabptr.i = signal->theData[0];
+ ptrCheckGuard(tabptr, ctabrecFilesize, tableRecord);
+ tabptr.p->currentSchemaVersion = signal->theData[1];
+ tabptr.p->storedTable = (bool)signal->theData[2];
+ BlockReference retRef = signal->theData[3];
+ tabptr.p->tableType = (Uint8)signal->theData[4];
+ BlockReference retPtr = signal->theData[5];
+
+ ndbrequire(tabptr.p->enabled == false);
+ tabptr.p->enabled = true;
+ tabptr.p->dropping = false;
+
+ signal->theData[0] = tabptr.i;
+ signal->theData[1] = retPtr;
+ sendSignal(retRef, GSN_TC_SCHVERCONF, signal, 2, JBB);
+}//Dbtc::execTC_SCHVERREQ()
+
+void
+Dbtc::execPREP_DROP_TAB_REQ(Signal* signal)
+{
+ jamEntry();
+
+ PrepDropTabReq* req = (PrepDropTabReq*)signal->getDataPtr();
+
+ TableRecordPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabrecFilesize, tableRecord);
+
+ Uint32 senderRef = req->senderRef;
+ Uint32 senderData = req->senderData;
+
+ if(!tabPtr.p->enabled){
+ jam();
+ PrepDropTabRef* ref = (PrepDropTabRef*)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = senderData;
+ ref->tableId = tabPtr.i;
+ ref->errorCode = PrepDropTabRef::NoSuchTable;
+ sendSignal(senderRef, GSN_PREP_DROP_TAB_REF, signal,
+ PrepDropTabRef::SignalLength, JBB);
+ return;
+ }
+
+ if(tabPtr.p->dropping){
+ jam();
+ PrepDropTabRef* ref = (PrepDropTabRef*)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = senderData;
+ ref->tableId = tabPtr.i;
+ ref->errorCode = PrepDropTabRef::DropInProgress;
+ sendSignal(senderRef, GSN_PREP_DROP_TAB_REF, signal,
+ PrepDropTabRef::SignalLength, JBB);
+ return;
+ }
+
+ tabPtr.p->dropping = true;
+ tabPtr.p->dropTable.senderRef = senderRef;
+ tabPtr.p->dropTable.senderData = senderData;
+
+ {
+ WaitDropTabReq * req = (WaitDropTabReq*)signal->getDataPtrSend();
+ req->tableId = tabPtr.i;
+ req->senderRef = reference();
+
+ HostRecordPtr hostPtr;
+ tabPtr.p->dropTable.waitDropTabCount.clearWaitingFor();
+ for (hostPtr.i = 1; hostPtr.i < MAX_NDB_NODES; hostPtr.i++) {
+ jam();
+ ptrAss(hostPtr, hostRecord);
+ if (hostPtr.p->hostStatus == HS_ALIVE) {
+ jam();
+ tabPtr.p->dropTable.waitDropTabCount.setWaitingFor(hostPtr.i);
+ sendSignal(calcLqhBlockRef(hostPtr.i), GSN_WAIT_DROP_TAB_REQ,
+ signal, WaitDropTabReq::SignalLength, JBB);
+ }//for
+ }//if
+
+ ndbrequire(tabPtr.p->dropTable.waitDropTabCount.done() != true);
+ }
+}
+
+void
+Dbtc::execWAIT_DROP_TAB_CONF(Signal* signal)
+{
+ jamEntry();
+ WaitDropTabConf * conf = (WaitDropTabConf*)signal->getDataPtr();
+
+ TableRecordPtr tabPtr;
+ tabPtr.i = conf->tableId;
+ ptrCheckGuard(tabPtr, ctabrecFilesize, tableRecord);
+
+ ndbrequire(tabPtr.p->dropping == true);
+ Uint32 nodeId = refToNode(conf->senderRef);
+ tabPtr.p->dropTable.waitDropTabCount.clearWaitingFor(nodeId);
+
+ if(!tabPtr.p->dropTable.waitDropTabCount.done()){
+ jam();
+ return;
+ }
+
+ {
+ PrepDropTabConf* conf = (PrepDropTabConf*)signal->getDataPtrSend();
+ conf->tableId = tabPtr.i;
+ conf->senderRef = reference();
+ conf->senderData = tabPtr.p->dropTable.senderData;
+ sendSignal(tabPtr.p->dropTable.senderRef, GSN_PREP_DROP_TAB_CONF, signal,
+ PrepDropTabConf::SignalLength, JBB);
+ tabPtr.p->dropTable.senderRef = 0;
+ }
+}
+
+void
+Dbtc::checkWaitDropTabFailedLqh(Signal* signal, Uint32 nodeId, Uint32 tableId)
+{
+
+ TableRecordPtr tabPtr;
+ tabPtr.i = tableId;
+
+ WaitDropTabConf * conf = (WaitDropTabConf*)signal->getDataPtr();
+ conf->tableId = tableId;
+
+ const Uint32 RT_BREAK = 16;
+ for(Uint32 i = 0; i<RT_BREAK && tabPtr.i < ctabrecFilesize; i++, tabPtr.i++){
+ jam();
+ ptrAss(tabPtr, tableRecord);
+ if(tabPtr.p->enabled && tabPtr.p->dropping){
+ if(tabPtr.p->dropTable.waitDropTabCount.isWaitingFor(nodeId)){
+ jam();
+ conf->senderRef = calcLqhBlockRef(nodeId);
+ execWAIT_DROP_TAB_CONF(signal);
+ tabPtr.i++;
+ break;
+ }
+ }
+ }
+
+ if(tabPtr.i == ctabrecFilesize){
+ /**
+ * Finished
+ */
+ jam();
+ return;
+ }
+
+ signal->theData[0] = TcContinueB::CHECK_WAIT_DROP_TAB_FAILED_LQH;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = tabPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+}
+
+void
+Dbtc::execDROP_TAB_REQ(Signal* signal)
+{
+ jamEntry();
+
+ DropTabReq* req = (DropTabReq*)signal->getDataPtr();
+
+ TableRecordPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabrecFilesize, tableRecord);
+
+ Uint32 senderRef = req->senderRef;
+ Uint32 senderData = req->senderData;
+ DropTabReq::RequestType rt = (DropTabReq::RequestType)req->requestType;
+
+ if(!tabPtr.p->enabled && rt == DropTabReq::OnlineDropTab){
+ jam();
+ DropTabRef* ref = (DropTabRef*)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = senderData;
+ ref->tableId = tabPtr.i;
+ ref->errorCode = DropTabRef::NoSuchTable;
+ sendSignal(senderRef, GSN_DROP_TAB_REF, signal,
+ DropTabRef::SignalLength, JBB);
+ return;
+ }
+
+ if(!tabPtr.p->dropping && rt == DropTabReq::OnlineDropTab){
+ jam();
+ DropTabRef* ref = (DropTabRef*)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = senderData;
+ ref->tableId = tabPtr.i;
+ ref->errorCode = DropTabRef::DropWoPrep;
+ sendSignal(senderRef, GSN_DROP_TAB_REF, signal,
+ DropTabRef::SignalLength, JBB);
+ return;
+ }
+
+ tabPtr.p->enabled = false;
+ tabPtr.p->dropping = false;
+
+ DropTabConf * conf = (DropTabConf*)signal->getDataPtrSend();
+ conf->tableId = tabPtr.i;
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ sendSignal(senderRef, GSN_DROP_TAB_CONF, signal,
+ PrepDropTabConf::SignalLength, JBB);
+}
+
+void Dbtc::execALTER_TAB_REQ(Signal * signal)
+{
+ AlterTabReq* const req = (AlterTabReq*)signal->getDataPtr();
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 senderData = req->senderData;
+ const Uint32 changeMask = req->changeMask;
+ const Uint32 tableId = req->tableId;
+ const Uint32 tableVersion = req->tableVersion;
+ const Uint32 gci = req->gci;
+ AlterTabReq::RequestType requestType =
+ (AlterTabReq::RequestType) req->requestType;
+
+ TableRecordPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, ctabrecFilesize, tableRecord);
+ tabPtr.p->currentSchemaVersion = tableVersion;
+
+ // Request handled successfully
+ AlterTabConf * conf = (AlterTabConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ conf->changeMask = changeMask;
+ conf->tableId = tableId;
+ conf->tableVersion = tableVersion;
+ conf->gci = gci;
+ conf->requestType = requestType;
+ sendSignal(senderRef, GSN_ALTER_TAB_CONF, signal,
+ AlterTabConf::SignalLength, JBB);
+}
+
+/* ***************************************************************************/
+/* START / RESTART */
+/* ***************************************************************************/
+void Dbtc::execSIZEALT_REP(Signal* signal)
+{
+ jamEntry();
+ tblockref = signal->theData[TcSizeAltReq::IND_BLOCK_REF];
+ const UintR apiConnect = signal->theData[TcSizeAltReq::IND_API_CONNECT];
+ const UintR tcConnect = signal->theData[TcSizeAltReq::IND_TC_CONNECT];
+ const UintR tables = signal->theData[TcSizeAltReq::IND_TABLE];
+ const UintR localScan = signal->theData[TcSizeAltReq::IND_LOCAL_SCAN];
+ const UintR tcScan = signal->theData[TcSizeAltReq::IND_TC_SCAN];
+
+ ccacheFilesize = (apiConnect/3) + 1;
+ capiConnectFilesize = apiConnect;
+ ctcConnectFilesize = tcConnect;
+ ctabrecFilesize = tables;
+ cscanrecFileSize = tcScan;
+ cscanOprecFileSize = localScan;
+ cscanFragrecFileSize = localScan;
+
+ initRecords();
+ initialiseRecordsLab(signal, (UintR)0);
+}//Dbtc::execSIZEALT_REP()
+
+void Dbtc::returnInitialiseRecordsLab(Signal* signal)
+{
+ signal->theData[0] = DBTC_REF;
+ sendSignal(CMVMI_REF, GSN_SIZEALT_ACK, signal, 2, JBB);
+}//Dbtc::returnInitialiseRecordsLab()
+
+void Dbtc::execSTTOR(Signal* signal)
+{
+ Uint16 tphase;
+
+ jamEntry();
+ /* START CASE */
+ tphase = signal->theData[1];
+ csignalKey = signal->theData[6];
+ switch (tphase) {
+ case ZSPH1:
+ jam();
+ startphase1x010Lab(signal);
+ return;
+ default:
+ jam();
+ sttorryLab(signal); /* START PHASE 255 */
+ return;
+ }//switch
+}//Dbtc::execSTTOR()
+
+void Dbtc::sttorryLab(Signal* signal)
+{
+ signal->theData[0] = csignalKey;
+ signal->theData[1] = 3; /* BLOCK CATEGORY */
+ signal->theData[2] = 2; /* SIGNAL VERSION NUMBER */
+ signal->theData[3] = ZSPH1;
+ signal->theData[4] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 5, JBB);
+}//Dbtc::sttorryLab()
+
+/* ***************************************************************************/
+/* INTERNAL START / RESTART */
+/*****************************************************************************/
+void Dbtc::execNDB_STTOR(Signal* signal)
+{
+ Uint16 tndbstartphase;
+ Uint16 tstarttype;
+
+ jamEntry();
+ tusersblkref = signal->theData[0];
+ tnodeid = signal->theData[1];
+ tndbstartphase = signal->theData[2]; /* START PHASE */
+ tstarttype = signal->theData[3]; /* START TYPE */
+ Uint32 config1 = signal->theData[12]; /* CONFIG INFO TC */
+ Uint32 config2 = signal->theData[13]; /* CONFIG INFO TC */
+ switch (tndbstartphase) {
+ case ZINTSPH1:
+ jam();
+ ctimeOutCheckDelay = 50; // 500ms
+ set_timeout_value(config1);
+ set_no_parallel_takeover(config2);
+ intstartphase1x010Lab(signal);
+ return;
+ case ZINTSPH2:
+ jam();
+ set_appl_timeout_value(config2);
+ intstartphase2x010Lab(signal);
+ return;
+ case ZINTSPH3:
+ jam();
+ intstartphase3x010Lab(signal); /* SEIZE CONNECT RECORD IN EACH LQH*/
+// Start transaction event reporting.
+ signal->theData[0] = TcContinueB::ZTRANS_EVENT_REP;
+ signal->theData[1] = 1;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 10, 2);
+ return;
+ case ZINTSPH6:
+ jam();
+ csystemStart = SSS_TRUE;
+ break;
+ default:
+ jam();
+ break;
+ }//switch
+ ndbsttorry010Lab(signal);
+ return;
+}//Dbtc::execNDB_STTOR()
+
+void Dbtc::ndbsttorry010Lab(Signal* signal)
+{
+ signal->theData[0] = cownref;
+ sendSignal(cndbcntrblockref, GSN_NDB_STTORRY, signal, 1, JBB);
+}//Dbtc::ndbsttorry010Lab()
+
+void
+Dbtc::set_timeout_value(Uint32 timeOut)
+{
+ timeOut = timeOut / 10;
+ if (timeOut < 2) {
+ jam();
+ timeOut = 100;
+ }//if
+ ctimeOutValue = timeOut;
+}
+
+void
+Dbtc::set_appl_timeout_value(Uint32 timeOut)
+{
+ timeOut /= 10;
+ if (timeOut < ctimeOutValue) {
+ jam();
+ c_appl_timeout_value = ctimeOutValue;
+ }//if
+ c_appl_timeout_value = timeOut;
+}
+
+void
+Dbtc::set_no_parallel_takeover(Uint32 noParallelTakeOver)
+{
+ if (noParallelTakeOver == 0) {
+ jam();
+ noParallelTakeOver = 1;
+ } else if (noParallelTakeOver > MAX_NDB_NODES) {
+ jam();
+ noParallelTakeOver = MAX_NDB_NODES;
+ }//if
+ cnoParallelTakeOver = noParallelTakeOver;
+}
+
+/* ***************************************************************************/
+/* S T A R T P H A S E 1 X */
+/* INITIALISE BLOCKREF AND BLOCKNUMBERS */
+/* ***************************************************************************/
+void Dbtc::startphase1x010Lab(Signal* signal)
+{
+ csystemStart = SSS_FALSE;
+ ctimeOutCheckCounter = 0;
+ ctimeOutCheckFragCounter = 0;
+ ctimeOutMissedHeartbeats = 0;
+ ctimeOutCheckHeartbeat = 0;
+ ctimeOutCheckLastHeartbeat = 0;
+ ctimeOutCheckActive = TOCS_FALSE;
+ ctimeOutCheckFragActive = TOCS_FALSE;
+ sttorryLab(signal);
+}//Dbtc::startphase1x010Lab()
+
+/*****************************************************************************/
+/* I N T S T A R T P H A S E 1 X */
+/* INITIALISE ALL RECORDS. */
+/*****************************************************************************/
+void Dbtc::intstartphase1x010Lab(Signal* signal)
+{
+ cownNodeid = tnodeid;
+ cownref = calcTcBlockRef(cownNodeid);
+ clqhblockref = calcLqhBlockRef(cownNodeid);
+ cdihblockref = calcDihBlockRef(cownNodeid);
+ cdictblockref = calcDictBlockRef(cownNodeid);
+ cndbcntrblockref = calcNdbCntrBlockRef(cownNodeid);
+ cerrorBlockref = calcNdbCntrBlockRef(cownNodeid);
+ coperationsize = 0;
+ cfailure_nr = 0;
+ ndbsttorry010Lab(signal);
+}//Dbtc::intstartphase1x010Lab()
+
+/*****************************************************************************/
+/* I N T S T A R T P H A S E 2 X */
+/* SET-UP LOCAL CONNECTIONS. */
+/*****************************************************************************/
+void Dbtc::intstartphase2x010Lab(Signal* signal)
+{
+ tcConnectptr.i = cfirstfreeTcConnect;
+ intstartphase2x020Lab(signal);
+}//Dbtc::intstartphase2x010Lab()
+
+void Dbtc::intstartphase2x020Lab(Signal* signal)
+{
+ if (tcConnectptr.i == RNIL) {
+ jam();
+ ndbsttorry010Lab(signal);
+ return;
+ }//if
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ tcConnectptr.p->tcConnectstate = OS_CONNECTING_DICT;
+/* ****************** */
+/* DISEIZEREQ < */
+/* ****************** */
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = cownref;
+ sendSignal(cdihblockref, GSN_DISEIZEREQ, signal, 2, JBB);
+}//Dbtc::intstartphase2x020Lab()
+
+void Dbtc::execDISEIZECONF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ tcConnectptr.p->dihConnectptr = signal->theData[1];
+ tcConnectptr.i = tcConnectptr.p->nextTcConnect;
+ intstartphase2x020Lab(signal);
+}//Dbtc::execDISEIZECONF()
+
+/*****************************************************************************/
+/* I N T S T A R T P H A S E 3 X */
+/* PREPARE DISTRIBUTED CONNECTIONS */
+/*****************************************************************************/
+void Dbtc::intstartphase3x010Lab(Signal* signal)
+{
+ signal->theData[0] = cownref;
+ sendSignal(cndbcntrblockref, GSN_READ_NODESREQ, signal, 1, JBB);
+}//Dbtc::intstartphase3x010Lab()
+
+void Dbtc::execREAD_NODESCONF(Signal* signal)
+{
+ UintR guard0;
+
+ jamEntry();
+
+ ReadNodesConf * const readNodes = (ReadNodesConf *)&signal->theData[0];
+
+ csystemnodes = readNodes->noOfNodes;
+ cmasterNodeId = readNodes->masterNodeId;
+
+ con_lineNodes = 0;
+ arrGuard(csystemnodes, MAX_NDB_NODES);
+ guard0 = csystemnodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES); // Check not zero nodes
+
+ for (unsigned i = 1; i < MAX_NDB_NODES; i++) {
+ jam();
+ if (NodeBitmask::get(readNodes->allNodes, i)) {
+ hostptr.i = i;
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+
+ hostptr.p->takeOverStatus = TOS_IDLE;
+ hostptr.p->ndbVersion = ReadNodesConf::getVersionId
+ (i, readNodes->theVersionIds);
+
+ if (NodeBitmask::get(readNodes->inactiveNodes, i)) {
+ jam();
+ hostptr.p->hostStatus = HS_DEAD;
+ } else {
+ jam();
+ con_lineNodes++;
+ hostptr.p->hostStatus = HS_ALIVE;
+ }//if
+ }//if
+ }//for
+ ndbsttorry010Lab(signal);
+}//Dbtc::execREAD_NODESCONF()
+
+/*****************************************************************************/
+/* A P I _ F A I L R E Q */
+// An API node has failed for some reason. We need to disconnect all API
+// connections to the API node. This also includes
+/*****************************************************************************/
+void Dbtc::execAPI_FAILREQ(Signal* signal)
+{
+ /***************************************************************************
+ * Set the block reference to return API_FAILCONF to. Set the number of api
+ * connects currently closing to one to indicate that we are still in the
+ * process of going through the api connect records. Thus checking for zero
+ * can only be true after all api connect records have been checked.
+ **************************************************************************/
+ jamEntry();
+ capiFailRef = signal->theData[1];
+ arrGuard(signal->theData[0], MAX_NODES);
+ capiConnectClosing[signal->theData[0]] = 1;
+ handleFailedApiNode(signal, signal->theData[0], (UintR)0);
+}
+
+void
+Dbtc::handleFailedApiNode(Signal* signal,
+ UintR TapiFailedNode,
+ UintR TapiConnectPtr)
+{
+ UintR TloopCount = 0;
+ arrGuard(TapiFailedNode, MAX_NODES);
+ apiConnectptr.i = TapiConnectPtr;
+ do {
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ const UintR TapiNode = refToNode(apiConnectptr.p->ndbapiBlockref);
+ if (TapiNode == TapiFailedNode) {
+#ifdef VM_TRACE
+ if (apiConnectptr.p->apiFailState != ZFALSE) {
+ ndbout << "Error in previous API fail handling discovered" << endl
+ << " apiConnectptr.i = " << apiConnectptr.i << endl
+ << " apiConnectstate = " << apiConnectptr.p->apiConnectstate
+ << endl
+ << " ndbapiBlockref = " << hex
+ << apiConnectptr.p->ndbapiBlockref << endl
+ << " apiNode = " << refToNode(apiConnectptr.p->ndbapiBlockref)
+ << endl;
+ if (apiConnectptr.p->lastTcConnect != RNIL){
+ jam();
+ tcConnectptr.i = apiConnectptr.p->lastTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ ndbout << " tcConnectptr.i = " << tcConnectptr.i << endl
+ << " tcConnectstate = " << tcConnectptr.p->tcConnectstate
+ << endl;
+ }
+ }//if
+#endif
+
+ apiConnectptr.p->returnsignal = RS_NO_RETURN;
+ /***********************************************************************/
+ // The connected node is the failed node.
+ /**********************************************************************/
+ switch(apiConnectptr.p->apiConnectstate) {
+ case CS_DISCONNECTED:
+ /*********************************************************************/
+ // These states do not need any special handling.
+ // Simply continue with the next.
+ /*********************************************************************/
+ jam();
+ break;
+ case CS_ABORTING:
+ /*********************************************************************/
+ // This could actually mean that the API connection is already
+ // ready to release if the abortState is IDLE.
+ /*********************************************************************/
+ if (apiConnectptr.p->abortState == AS_IDLE) {
+ jam();
+ releaseApiCon(signal, apiConnectptr.i);
+ } else {
+ jam();
+ capiConnectClosing[TapiFailedNode]++;
+ apiConnectptr.p->apiFailState = ZTRUE;
+ }//if
+ break;
+ case CS_WAIT_ABORT_CONF:
+ case CS_WAIT_COMMIT_CONF:
+ case CS_START_COMMITTING:
+ case CS_PREPARE_TO_COMMIT:
+ case CS_COMMITTING:
+ case CS_COMMIT_SENT:
+ /*********************************************************************/
+ // These states indicate that an abort process or commit process is
+ // already ongoing. We will set a state in the api record indicating
+ // that the API node has failed.
+ // Also we will increase the number of outstanding api records to
+ // wait for before we can respond with API_FAILCONF.
+ /*********************************************************************/
+ jam();
+ capiConnectClosing[TapiFailedNode]++;
+ apiConnectptr.p->apiFailState = ZTRUE;
+ break;
+ case CS_START_SCAN:
+ /*********************************************************************/
+ // The api record was performing a scan operation. We need to check
+ // on the scan state. Since completing a scan process might involve
+ // sending several signals we will increase the loop count by 64.
+ /*********************************************************************/
+ jam();
+ handleScanStop(signal, TapiFailedNode);
+ TloopCount += 64;
+ break;
+ case CS_CONNECTED:
+ /*********************************************************************/
+ // The api record is connected to failed node. We need to release the
+ // connection and set it in a disconnected state.
+ /*********************************************************************/
+ jam();
+ releaseApiCon(signal, apiConnectptr.i);
+ break;
+ case CS_REC_COMMITTING:
+ case CS_RECEIVING:
+ case CS_STARTED:
+ /*********************************************************************/
+ // The api record was in the process of performing a transaction but
+ // had not yet sent all information.
+ // We need to initiate an ABORT since the API will not provide any
+ // more information.
+ // Since the abort can send many signals we will insert a real-time
+ // break after checking this record.
+ /*********************************************************************/
+ jam();
+ apiConnectptr.p->apiFailState = ZTRUE;
+ capiConnectClosing[TapiFailedNode]++;
+ abort010Lab(signal);
+ TloopCount = 256;
+ break;
+ case CS_PREPARED:
+ jam();
+ case CS_REC_PREPARING:
+ jam();
+ case CS_START_PREPARING:
+ jam();
+ /*********************************************************************/
+ // Not implemented yet.
+ /*********************************************************************/
+ systemErrorLab(signal);
+ break;
+ case CS_RESTART:
+ jam();
+ case CS_COMPLETING:
+ jam();
+ case CS_COMPLETE_SENT:
+ jam();
+ case CS_WAIT_COMPLETE_CONF:
+ jam();
+ case CS_FAIL_ABORTING:
+ jam();
+ case CS_FAIL_ABORTED:
+ jam();
+ case CS_FAIL_PREPARED:
+ jam();
+ case CS_FAIL_COMMITTING:
+ jam();
+ case CS_FAIL_COMMITTED:
+ /*********************************************************************/
+ // These states are only valid on copy and fail API connections.
+ /*********************************************************************/
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+ } else {
+ jam();
+ }//if
+ apiConnectptr.i++;
+ if (apiConnectptr.i > ((capiConnectFilesize / 3) - 1)) {
+ jam();
+ capiConnectClosing[TapiFailedNode]--;
+ /**
+ * Finished with scanning connection record
+ *
+ * Now scan markers
+ */
+ removeMarkerForFailedAPI(signal, TapiFailedNode, RNIL); // RNIL = first
+ return;
+ }//if
+ } while (TloopCount++ < 256);
+ signal->theData[0] = TcContinueB::ZHANDLE_FAILED_API_NODE;
+ signal->theData[1] = TapiFailedNode;
+ signal->theData[2] = apiConnectptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+}//Dbtc::handleFailedApiNode()
+
+void
+Dbtc::removeMarkerForFailedAPI(Signal* signal,
+ Uint32 nodeId,
+ Uint32 startBucket)
+{
+ CommitAckMarkerIterator iter;
+ if(startBucket == RNIL){
+ jam();
+ capiConnectClosing[nodeId]++;
+ m_commitAckMarkerHash.next(0, iter);
+ } else {
+ jam();
+ m_commitAckMarkerHash.next(startBucket, iter);
+ }
+
+ const Uint32 RT_BREAK = 256;
+ for(Uint32 i = 0; i<RT_BREAK || iter.bucket == startBucket; i++){
+ jam();
+
+ if(iter.curr.i == RNIL){
+ jam();
+ /**
+ * Done with iteration
+ */
+ capiConnectClosing[nodeId]--;
+ if (capiConnectClosing[nodeId] == 0) {
+ jam();
+ /********************************************************************/
+ // No outstanding ABORT or COMMIT's of this failed API node.
+ // We can respond with API_FAILCONF
+ /********************************************************************/
+ signal->theData[0] = nodeId;
+ signal->theData[1] = cownref;
+ sendSignal(capiFailRef, GSN_API_FAILCONF, signal, 2, JBB);
+ }
+ return;
+ }
+
+ if(iter.curr.p->apiNodeId == nodeId){
+ jam();
+
+ /**
+ * Check so that the record is not still in use
+ *
+ * (This can happen when NF and API Fail happens at the same time)
+ */
+ ApiConnectRecordPtr apiConnectPtr;
+ apiConnectPtr.i = iter.curr.p->apiConnectPtr;
+ ptrCheckGuard(apiConnectPtr, capiConnectFilesize, apiConnectRecord);
+ if(apiConnectPtr.p->commitAckMarker == iter.curr.i){
+ jam();
+ /**
+ * The record is still active
+ *
+ * Don't remove it, but continueb instead
+ */
+ break;
+ }
+
+ sendRemoveMarkers(signal, iter.curr.p);
+ m_commitAckMarkerHash.release(iter.curr);
+
+ break;
+ }
+ m_commitAckMarkerHash.next(iter);
+ }
+
+ signal->theData[0] = TcContinueB::ZHANDLE_FAILED_API_NODE_REMOVE_MARKERS;
+ signal->theData[1] = nodeId;
+ signal->theData[2] = iter.bucket;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+}
+
+void Dbtc::handleApiFailState(Signal* signal, UintR TapiConnectptr)
+{
+ ApiConnectRecordPtr TlocalApiConnectptr;
+ UintR TfailedApiNode;
+
+ TlocalApiConnectptr.i = TapiConnectptr;
+ ptrCheckGuard(TlocalApiConnectptr, capiConnectFilesize, apiConnectRecord);
+ TfailedApiNode = refToNode(TlocalApiConnectptr.p->ndbapiBlockref);
+ arrGuard(TfailedApiNode, MAX_NODES);
+ capiConnectClosing[TfailedApiNode]--;
+ releaseApiCon(signal, TapiConnectptr);
+ TlocalApiConnectptr.p->apiFailState = ZFALSE;
+ if (capiConnectClosing[TfailedApiNode] == 0) {
+ jam();
+ signal->theData[0] = TfailedApiNode;
+ signal->theData[1] = cownref;
+ sendSignal(capiFailRef, GSN_API_FAILCONF, signal, 2, JBB);
+ }//if
+}//Dbtc::handleApiFailState()
+
+/**
+ * Dbtc::handleScanStop
+ * This function is called when an entire scan should be stopped
+ * Check state of the scan and take appropriate action.
+ * The parameter TapiFailedNode indicates if the scan is stopped
+ * because an API node has failed or if it has been stopped because
+ * the scan has timed out.
+ *
+ */
+void Dbtc::handleScanStop(Signal* signal, UintR TapiFailedNode)
+{
+ arrGuard(TapiFailedNode, MAX_NODES);
+
+ scanptr.i = apiConnectptr.p->apiScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+
+ // If api has failed we must release all resources
+ bool apiNodeHasFailed = (TapiFailedNode != 0);
+
+ DEBUG("handleScanStop: scanState = "<< scanptr.p->scanState);
+
+ switch (scanptr.p->scanState) {
+ case ScanRecord::WAIT_SCAN_TAB_INFO:
+ case ScanRecord::WAIT_AI:
+ jam();
+ /**
+ * The scan process is still in the definition phase.
+ * We will release the resources and then release the connection
+ * to the failed API.
+ */
+ releaseScanResources(signal);
+ if (apiNodeHasFailed) {
+ jam();
+ releaseApiCon(signal, apiConnectptr.i);
+ }//if
+ break;
+
+ case ScanRecord::WAIT_FRAGMENT_COUNT:
+ jam();
+ if (!apiNodeHasFailed) {
+ jam();
+ /**
+ * Time-out waiting for a local signal can only happen
+ * if we have a serious problem.
+ */
+ systemErrorLab(signal);
+ }//if
+ capiConnectClosing[TapiFailedNode]++;
+ apiConnectptr.p->apiFailState = ZTRUE;
+ scanptr.p->apiIsClosed = true;
+ break;
+
+ case ScanRecord::CLOSING_SCAN:
+ jam();
+ /**
+ * With CLOSING_SCAN it is enough to set the
+ * fail state such that the connection is released at the end of the
+ * closing process. The close process is already ongoing.
+ * Set apiIsClosed to true to indicate that resources should be released
+ * at the end of the close process.
+ **/
+
+ if (apiNodeHasFailed) {
+ jam();
+ capiConnectClosing[TapiFailedNode]++;
+ apiConnectptr.p->apiFailState = ZTRUE;
+ scanptr.p->apiIsClosed = true;
+ }//if
+ if (apiConnectptr.p->apiFailState == ZTRUE) {
+ jam();
+ handleApiFailState(signal, apiConnectptr.i);
+ return;
+ }//if
+ break;
+
+ case ScanRecord::SCAN_NEXT_ORDERED:
+ /**
+ * In the SCAN_NEXT_ORDERED state we will wait for the next natural place
+ * to receive some action from the API and instead of waiting for the
+ * API here we will start the abort process.
+
+ * After the abort process is completed we will release the connection.
+ */
+ if (apiNodeHasFailed) {
+ jam();
+ capiConnectClosing[TapiFailedNode]++;
+ apiConnectptr.p->apiFailState = ZTRUE;
+ }//if
+ // Release resources and send a response to API
+ scanptr.p->apiIsClosed = true;
+ scanCompletedLab(signal);
+ break;
+
+ case ScanRecord::DELIVERED:
+ case ScanRecord::QUEUED_DELIVERED:
+ /**
+ * A response has been sent to the api but it has not responded
+ */
+
+ if (apiNodeHasFailed) {
+ jam();
+ capiConnectClosing[TapiFailedNode]++;
+ apiConnectptr.p->apiFailState = ZTRUE;
+ scanptr.p->apiIsClosed = true;
+ } else {
+ jam();
+ /*
+ In this case we have received a time-out caused by the application
+ waiting too long to continue the scan. We will check the application
+ time-out instead of the deadlock detetection time-out. If the
+ application time-out hasn't fired we will simply ignore the condition.
+ */
+ if ((ctcTimer - getApiConTimer(apiConnectptr.i)) <= c_appl_timeout_value) {
+ jam();
+ return;
+ }//if
+ // Dont' release, wait until api responds or fails
+ scanptr.p->apiIsClosed = false;
+ }
+ scanCompletedLab(signal);
+ break;
+
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+
+ }//switch
+}//Dbtc::handleScanStop()
+
+/****************************************************************************
+ * T C S E I Z E R E Q
+ * THE APPLICATION SENDS A REQUEST TO SEIZE A CONNECT RECORD TO CARRY OUT A
+ * TRANSACTION
+ * TC BLOCK TAKE OUT A CONNECT RECORD FROM THE FREE LIST AND ESTABLISHES ALL
+ * NECESSARY CONNECTION BEFORE REPLYING TO THE APPLICATION BLOCK
+ ****************************************************************************/
+void Dbtc::execTCSEIZEREQ(Signal* signal)
+{
+ UintR tapiPointer;
+ BlockReference tapiBlockref; /* SENDER BLOCK REFERENCE*/
+
+ jamEntry();
+ tapiPointer = signal->theData[0]; /* REQUEST SENDERS CONNECT RECORD POINTER*/
+ tapiBlockref = signal->theData[1]; /* SENDERS BLOCK REFERENCE*/
+
+ const NodeState::StartLevel sl =
+ (NodeState::StartLevel)getNodeState().startLevel;
+
+ const NodeId senderNodeId = refToNode(tapiBlockref);
+ const bool local = senderNodeId == getOwnNodeId() || senderNodeId == 0;
+
+ if(!(senderNodeId == getNodeState().getSingleUserApi()) &&
+ !getNodeState().getSingleUserMode()) {
+ if(!(sl==NodeState::SL_SINGLEUSER &&
+ senderNodeId == getNodeState().getSingleUserApi())) {
+ if (!(sl == NodeState::SL_STARTED ||
+ (sl == NodeState::SL_STARTING && local == true))) {
+ jam();
+
+ Uint32 errCode;
+ if(!(sl == NodeState::SL_SINGLEUSER && local))
+ {
+ switch(sl){
+ case NodeState::SL_STARTING:
+ errCode = ZSYSTEM_NOT_STARTED_ERROR;
+ break;
+ case NodeState::SL_STOPPING_1:
+ case NodeState::SL_STOPPING_2:
+ case NodeState::SL_STOPPING_3:
+ case NodeState::SL_STOPPING_4:
+ if(getNodeState().stopping.systemShutdown)
+ errCode = ZCLUSTER_SHUTDOWN_IN_PROGRESS;
+ else
+ errCode = ZNODE_SHUTDOWN_IN_PROGRESS;
+ break;
+ case NodeState::SL_SINGLEUSER:
+ errCode = ZCLUSTER_IN_SINGLEUSER_MODE;
+ break;
+ default:
+ errCode = ZWRONG_STATE;
+ break;
+ }
+ signal->theData[0] = tapiPointer;
+ signal->theData[1] = errCode;
+ sendSignal(tapiBlockref, GSN_TCSEIZEREF, signal, 2, JBB);
+ return;
+ }//if (!(sl == SL_SINGLEUSER))
+ } //if
+ }
+ }
+
+ seizeApiConnect(signal);
+ if (terrorCode == ZOK) {
+ jam();
+ apiConnectptr.p->ndbapiConnect = tapiPointer;
+ apiConnectptr.p->ndbapiBlockref = tapiBlockref;
+ signal->theData[0] = apiConnectptr.p->ndbapiConnect;
+ signal->theData[1] = apiConnectptr.i;
+ sendSignal(tapiBlockref, GSN_TCSEIZECONF, signal, 2, JBB);
+ return;
+ }
+
+ signal->theData[0] = tapiPointer;
+ signal->theData[1] = terrorCode;
+ sendSignal(tapiBlockref, GSN_TCSEIZEREF, signal, 2, JBB);
+}//Dbtc::execTCSEIZEREQ()
+
+/****************************************************************************/
+/* T C R E L E A S E Q */
+/* REQUEST TO RELEASE A CONNECT RECORD */
+/****************************************************************************/
+void Dbtc::execTCRELEASEREQ(Signal* signal)
+{
+ UintR tapiPointer;
+ BlockReference tapiBlockref; /* SENDER BLOCK REFERENCE*/
+
+ jamEntry();
+ tapiPointer = signal->theData[0]; /* REQUEST SENDERS CONNECT RECORD POINTER*/
+ tapiBlockref = signal->theData[1];/* SENDERS BLOCK REFERENCE*/
+ tuserpointer = signal->theData[2];
+ if (tapiPointer >= capiConnectFilesize) {
+ jam();
+ signal->theData[0] = tuserpointer;
+ signal->theData[1] = ZINVALID_CONNECTION;
+ sendSignal(tapiBlockref, GSN_TCRELEASEREF, signal, 2, JBB);
+ return;
+ } else {
+ jam();
+ apiConnectptr.i = tapiPointer;
+ }//if
+ ptrAss(apiConnectptr, apiConnectRecord);
+ if (apiConnectptr.p->apiConnectstate == CS_DISCONNECTED) {
+ jam();
+ signal->theData[0] = tuserpointer;
+ sendSignal(tapiBlockref, GSN_TCRELEASECONF, signal, 1, JBB);
+ } else {
+ if (tapiBlockref == apiConnectptr.p->ndbapiBlockref) {
+ if (apiConnectptr.p->apiConnectstate == CS_CONNECTED) {
+ jam(); /* JUST REPLY OK */
+ releaseApiCon(signal, apiConnectptr.i);
+ signal->theData[0] = tuserpointer;
+ sendSignal(apiConnectptr.p->ndbapiBlockref,
+ GSN_TCRELEASECONF, signal, 1, JBB);
+ } else {
+ jam();
+ signal->theData[0] = tuserpointer;
+ signal->theData[1] = ZINVALID_CONNECTION;
+ sendSignal(apiConnectptr.p->ndbapiBlockref,
+ GSN_TCRELEASEREF, signal, 2, JBB);
+ }
+ } else {
+ jam();
+ signal->theData[0] = tuserpointer;
+ signal->theData[1] = ZINVALID_CONNECTION;
+ sendSignal(tapiBlockref, GSN_TCRELEASEREF, signal, 2, JBB);
+ }//if
+ }//if
+}//Dbtc::execTCRELEASEREQ()
+
+/****************************************************************************/
+// Error Handling for TCKEYREQ messages
+/****************************************************************************/
+void Dbtc::signalErrorRefuseLab(Signal* signal)
+{
+ ptrGuard(apiConnectptr);
+ if (apiConnectptr.p->apiConnectstate != CS_DISCONNECTED) {
+ jam();
+ apiConnectptr.p->abortState = AS_IDLE;
+ apiConnectptr.p->apiConnectstate = CS_ABORTING;
+ }//if
+ sendSignalErrorRefuseLab(signal);
+}//Dbtc::signalErrorRefuseLab()
+
+void Dbtc::sendSignalErrorRefuseLab(Signal* signal)
+{
+ ndbassert(false);
+ ptrGuard(apiConnectptr);
+ if (apiConnectptr.p->apiConnectstate != CS_DISCONNECTED) {
+ jam();
+ signal->theData[0] = apiConnectptr.p->ndbapiConnect;
+ signal->theData[1] = signal->theData[ttransid_ptr];
+ signal->theData[2] = signal->theData[ttransid_ptr + 1];
+ signal->theData[3] = ZSIGNAL_ERROR;
+ sendSignal(apiConnectptr.p->ndbapiBlockref, GSN_TCROLLBACKREP,
+ signal, 4, JBB);
+ }
+}//Dbtc::sendSignalErrorRefuseLab()
+
+void Dbtc::abortBeginErrorLab(Signal* signal)
+{
+ apiConnectptr.p->transid[0] = signal->theData[ttransid_ptr];
+ apiConnectptr.p->transid[1] = signal->theData[ttransid_ptr + 1];
+ abortErrorLab(signal);
+}//Dbtc::abortBeginErrorLab()
+
+void Dbtc::printState(Signal* signal, int place)
+{
+#ifdef VM_TRACE // Change to if 0 to disable these printouts
+ ndbout << "-- Dbtc::printState -- " << endl;
+ ndbout << "Received from place = " << place
+ << " apiConnectptr.i = " << apiConnectptr.i
+ << " apiConnectstate = " << apiConnectptr.p->apiConnectstate << endl;
+ ndbout << "ctcTimer = " << ctcTimer
+ << " ndbapiBlockref = " << hex <<apiConnectptr.p->ndbapiBlockref
+ << " Transid = " << apiConnectptr.p->transid[0]
+ << " " << apiConnectptr.p->transid[1] << endl;
+ ndbout << " apiTimer = " << getApiConTimer(apiConnectptr.i)
+ << " counter = " << apiConnectptr.p->counter
+ << " lqhkeyconfrec = " << apiConnectptr.p->lqhkeyconfrec
+ << " lqhkeyreqrec = " << apiConnectptr.p->lqhkeyreqrec << endl;
+ ndbout << "abortState = " << (int)apiConnectptr.p->abortState
+ << " apiScanRec = " << apiConnectptr.p->apiScanRec
+ << " returncode = " << apiConnectptr.p->returncode << endl;
+ ndbout << "tckeyrec = " << apiConnectptr.p->tckeyrec
+ << " returnsignal = " << apiConnectptr.p->returnsignal
+ << " apiFailState = " << apiConnectptr.p->apiFailState << endl;
+ if (apiConnectptr.p->cachePtr != RNIL) {
+ jam();
+ CacheRecord *localCacheRecord = cacheRecord;
+ UintR TcacheFilesize = ccacheFilesize;
+ UintR TcachePtr = apiConnectptr.p->cachePtr;
+ if (TcachePtr < TcacheFilesize) {
+ jam();
+ CacheRecord * const regCachePtr = &localCacheRecord[TcachePtr];
+ ndbout << "currReclenAi = " << regCachePtr->currReclenAi
+ << " attrlength = " << regCachePtr->attrlength
+ << " tableref = " << regCachePtr->tableref
+ << " keylen = " << regCachePtr->keylen << endl;
+ } else {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ }//if
+#endif
+ return;
+}//Dbtc::printState()
+
+void
+Dbtc::TCKEY_abort(Signal* signal, int place)
+{
+ switch (place) {
+ case 0:
+ jam();
+ terrorCode = ZSTATE_ERROR;
+ apiConnectptr.p->firstTcConnect = RNIL;
+ printState(signal, 4);
+ abortBeginErrorLab(signal);
+ return;
+ case 1:
+ jam();
+ printState(signal, 3);
+ sendSignalErrorRefuseLab(signal);
+ return;
+ case 2:{
+ printState(signal, 6);
+ const TcKeyReq * const tcKeyReq = (TcKeyReq *)&signal->theData[0];
+ const Uint32 t1 = tcKeyReq->transId1;
+ const Uint32 t2 = tcKeyReq->transId2;
+ signal->theData[0] = apiConnectptr.p->ndbapiConnect;
+ signal->theData[1] = t1;
+ signal->theData[2] = t2;
+ signal->theData[3] = ZABORT_ERROR;
+ sendSignal(apiConnectptr.p->ndbapiBlockref, GSN_TCROLLBACKREP,
+ signal, 4, JBB);
+ return;
+ }
+ case 3:
+ jam();
+ printState(signal, 7);
+ noFreeConnectionErrorLab(signal);
+ return;
+ case 4:
+ jam();
+ terrorCode = ZERO_KEYLEN_ERROR;
+ releaseAtErrorLab(signal);
+ return;
+ case 5:
+ jam();
+ terrorCode = ZNO_AI_WITH_UPDATE;
+ releaseAtErrorLab(signal);
+ return;
+ case 6:
+ jam();
+ warningHandlerLab(signal);
+ return;
+
+ case 7:
+ jam();
+ tabStateErrorLab(signal);
+ return;
+
+ case 8:
+ jam();
+ wrongSchemaVersionErrorLab(signal);
+ return;
+
+ case 9:
+ jam();
+ terrorCode = ZSTATE_ERROR;
+ releaseAtErrorLab(signal);
+ return;
+
+ case 10:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 11:
+ jam();
+ terrorCode = ZMORE_AI_IN_TCKEYREQ_ERROR;
+ releaseAtErrorLab(signal);
+ return;
+
+ case 12:
+ jam();
+ terrorCode = ZSIMPLE_READ_WITHOUT_AI;
+ releaseAtErrorLab(signal);
+ return;
+
+ case 13:
+ jam();
+ switch (tcConnectptr.p->tcConnectstate) {
+ case OS_WAIT_KEYINFO:
+ jam();
+ printState(signal, 8);
+ terrorCode = ZSTATE_ERROR;
+ abortErrorLab(signal);
+ return;
+ default:
+ jam();
+ /********************************************************************/
+ /* MISMATCH BETWEEN STATE ON API CONNECTION AND THIS */
+ /* PARTICULAR TC CONNECT RECORD. THIS MUST BE CAUSED BY NDB */
+ /* INTERNAL ERROR. */
+ /********************************************************************/
+ systemErrorLab(signal);
+ return;
+ }//switch
+ return;
+
+ case 15:
+ jam();
+ terrorCode = ZSCAN_NODE_ERROR;
+ releaseAtErrorLab(signal);
+ return;
+
+ case 16:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 17:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 18:
+ jam();
+ warningHandlerLab(signal);
+ return;
+
+ case 19:
+ jam();
+ return;
+
+ case 20:
+ jam();
+ warningHandlerLab(signal);
+ return;
+
+ case 21:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 22:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 23:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 24:
+ jam();
+ seizeAttrbuferrorLab(signal);
+ return;
+
+ case 25:
+ jam();
+ warningHandlerLab(signal);
+ return;
+
+ case 26:
+ jam();
+ return;
+
+ case 27:
+ systemErrorLab(signal);
+ jam();
+ return;
+
+ case 28:
+ jam();
+ // NOT USED
+ return;
+
+ case 29:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 30:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 31:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 32:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 33:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 34:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 35:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 36:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 37:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 38:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 39:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 40:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 41:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 42:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 43:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 44:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 45:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 46:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 47:
+ jam();
+ terrorCode = apiConnectptr.p->returncode;
+ releaseAtErrorLab(signal);
+ return;
+
+ case 48:
+ jam();
+ terrorCode = ZCOMMIT_TYPE_ERROR;
+ releaseAtErrorLab(signal);
+ return;
+
+ case 49:
+ jam();
+ abortErrorLab(signal);
+ return;
+
+ case 50:
+ jam();
+ systemErrorLab(signal);
+ return;
+
+ case 51:
+ jam();
+ abortErrorLab(signal);
+ return;
+
+ case 52:
+ jam();
+ abortErrorLab(signal);
+ return;
+
+ case 53:
+ jam();
+ abortErrorLab(signal);
+ return;
+
+ case 54:
+ jam();
+ abortErrorLab(signal);
+ return;
+
+ case 55:
+ jam();
+ printState(signal, 5);
+ sendSignalErrorRefuseLab(signal);
+ return;
+
+ case 56:{
+ jam();
+ terrorCode = ZNO_FREE_TC_MARKER;
+ abortErrorLab(signal);
+ return;
+ }
+ case 57:{
+ jam();
+ /**
+ * Initialize object before starting error handling
+ */
+ initApiConnectRec(signal, apiConnectptr.p, true);
+ switch(getNodeState().startLevel){
+ case NodeState::SL_STOPPING_2:
+ case NodeState::SL_STOPPING_3:
+ case NodeState::SL_STOPPING_4:
+ if(getNodeState().stopping.systemShutdown)
+ terrorCode = ZCLUSTER_SHUTDOWN_IN_PROGRESS;
+ else
+ terrorCode = ZNODE_SHUTDOWN_IN_PROGRESS;
+ break;
+ case NodeState::SL_SINGLEUSER:
+ terrorCode = ZCLUSTER_IN_SINGLEUSER_MODE;
+ break;
+ default:
+ terrorCode = ZWRONG_STATE;
+ break;
+ }
+ abortErrorLab(signal);
+ return;
+ }
+
+ case 58:{
+ jam();
+ releaseAtErrorLab(signal);
+ return;
+ }
+
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//switch
+}
+
+void Dbtc::execKEYINFO(Signal* signal)
+{
+ UintR compare_transid1, compare_transid2;
+ jamEntry();
+ apiConnectptr.i = signal->theData[0];
+ tmaxData = 20;
+ if (apiConnectptr.i >= capiConnectFilesize) {
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }//if
+ ptrAss(apiConnectptr, apiConnectRecord);
+ ttransid_ptr = 1;
+ compare_transid1 = apiConnectptr.p->transid[0] ^ signal->theData[1];
+ compare_transid2 = apiConnectptr.p->transid[1] ^ signal->theData[2];
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ jam();
+ printState(signal, 10);
+ sendSignalErrorRefuseLab(signal);
+ return;
+ }//if
+ switch (apiConnectptr.p->apiConnectstate) {
+ case CS_RECEIVING:
+ case CS_REC_COMMITTING:
+ jam();
+ /*empty*/;
+ break;
+ /* OK */
+ case CS_ABORTING:
+ jam();
+ return; /* IGNORE */
+ case CS_CONNECTED:
+ jam();
+ /****************************************************************>*/
+ /* MOST LIKELY CAUSED BY A MISSED SIGNAL. SEND REFUSE AND */
+ /* SET STATE TO ABORTING. */
+ /****************************************************************>*/
+ printState(signal, 11);
+ signalErrorRefuseLab(signal);
+ return;
+ case CS_STARTED:
+ jam();
+ /****************************************************************>*/
+ /* MOST LIKELY CAUSED BY A MISSED SIGNAL. SEND REFUSE AND */
+ /* SET STATE TO ABORTING. SINCE A TRANSACTION WAS STARTED */
+ /* WE ALSO NEED TO ABORT THIS TRANSACTION. */
+ /****************************************************************>*/
+ terrorCode = ZSIGNAL_ERROR;
+ printState(signal, 2);
+ abortErrorLab(signal);
+ return;
+ default:
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }//switch
+
+ CacheRecord *localCacheRecord = cacheRecord;
+ UintR TcacheFilesize = ccacheFilesize;
+ UintR TcachePtr = apiConnectptr.p->cachePtr;
+ UintR TtcTimer = ctcTimer;
+ CacheRecord * const regCachePtr = &localCacheRecord[TcachePtr];
+ if (TcachePtr >= TcacheFilesize) {
+ TCKEY_abort(signal, 42);
+ return;
+ }//if
+ setApiConTimer(apiConnectptr.i, TtcTimer, __LINE__);
+ cachePtr.i = TcachePtr;
+ cachePtr.p = regCachePtr;
+
+ tcConnectptr.i = apiConnectptr.p->lastTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ switch (tcConnectptr.p->tcConnectstate) {
+ case OS_WAIT_KEYINFO:
+ jam();
+ tckeyreq020Lab(signal);
+ return;
+ default:
+ jam();
+ terrorCode = ZSTATE_ERROR;
+ abortErrorLab(signal);
+ return;
+ }//switch
+}//Dbtc::execKEYINFO()
+
+/*---------------------------------------------------------------------------*/
+/* */
+/* MORE THAN FOUR WORDS OF KEY DATA. WE NEED TO PACK THIS IN KEYINFO SIGNALS.*/
+/* WE WILL ALWAYS PACK 4 WORDS AT A TIME. */
+/*---------------------------------------------------------------------------*/
+void Dbtc::packKeyData000Lab(Signal* signal,
+ BlockReference TBRef)
+{
+ CacheRecord * const regCachePtr = cachePtr.p;
+ UintR Tmp;
+ Uint16 tdataPos;
+
+ jam();
+ tdataPos = 0;
+ Tmp = regCachePtr->keylen;
+ databufptr.i = regCachePtr->firstKeybuf;
+ do {
+ jam();
+ if (tdataPos == 20) {
+ jam();
+ /*---------------------------------------------------------------------*/
+ /* 4 MORE WORDS WILL NOT FIT IN THE 24 DATA WORDS IN A SIGNAL */
+ /*---------------------------------------------------------------------*/
+ sendKeyinfo(signal, TBRef, 20);
+ tdataPos = 0;
+ }//if
+ Tmp = Tmp - 4;
+ ptrCheckGuard(databufptr, cdatabufFilesize, databufRecord);
+ cdata[tdataPos ] = databufptr.p->data[0];
+ cdata[tdataPos + 1] = databufptr.p->data[1];
+ cdata[tdataPos + 2] = databufptr.p->data[2];
+ cdata[tdataPos + 3] = databufptr.p->data[3];
+ tdataPos = tdataPos + 4;
+ if (Tmp <= 4) {
+ jam();
+ /*---------------------------------------------------------------------*/
+ /* LAST PACK OF KEY DATA HAVE BEEN SENT */
+ /*---------------------------------------------------------------------*/
+ /* THERE WERE UNSENT INFORMATION, SEND IT. */
+ /*---------------------------------------------------------------------*/
+ sendKeyinfo(signal, TBRef, tdataPos);
+ releaseKeys(signal);
+ return;
+ }//if
+ databufptr.i = databufptr.p->nextDatabuf;
+ } while (1);
+}//Dbtc::packKeyData000Lab()
+
+void Dbtc::tckeyreq020Lab(Signal* signal)
+{
+ CacheRecord * const regCachePtr = cachePtr.p;
+ UintR TdataPos = 0;
+ UintR TkeyLen = regCachePtr->keylen;
+ UintR Tlen = regCachePtr->save1;
+
+ do {
+ if (cfirstfreeDatabuf == RNIL) {
+ jam();
+ seizeDatabuferrorLab(signal);
+ return;
+ }//if
+ linkKeybuf(signal);
+ arrGuard(TdataPos, 19);
+ databufptr.p->data[0] = signal->theData[TdataPos + 3];
+ databufptr.p->data[1] = signal->theData[TdataPos + 4];
+ databufptr.p->data[2] = signal->theData[TdataPos + 5];
+ databufptr.p->data[3] = signal->theData[TdataPos + 6];
+ Tlen = Tlen + 4;
+ TdataPos = TdataPos + 4;
+ if (Tlen < TkeyLen) {
+ jam();
+ if (TdataPos >= tmaxData) {
+ jam();
+ /*----------------------------------------------------*/
+ /** EXIT AND WAIT FOR SIGNAL KEYINFO OR KEYINFO9 **/
+ /** WHEN EITHER OF THE SIGNALS IS RECEIVED A JUMP **/
+ /** TO LABEL "KEYINFO_LABEL" IS DONE. THEN THE **/
+ /** PROGRAM RETURNS TO LABEL TCKEYREQ020 **/
+ /*----------------------------------------------------*/
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ regCachePtr->save1 = Tlen;
+ tcConnectptr.p->tcConnectstate = OS_WAIT_KEYINFO;
+ return;
+ }//if
+ } else {
+ jam();
+ tckeyreq050Lab(signal);
+ return;
+ }//if
+ } while (1);
+ return;
+}//Dbtc::tckeyreq020Lab()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SAVE ATTRIBUTE INFORMATION IN OPERATION RECORD ------- */
+/* ------------------------------------------------------------------------- */
+void Dbtc::saveAttrbuf(Signal* signal)
+{
+ CacheRecord * const regCachePtr = cachePtr.p;
+ UintR TfirstfreeAttrbuf = cfirstfreeAttrbuf;
+ UintR TattrbufFilesize = cattrbufFilesize;
+ UintR TTcfirstAttrbuf = regCachePtr->firstAttrbuf;
+ UintR Tlen = signal->length() - 3;
+ AttrbufRecord *localAttrbufRecord = attrbufRecord;
+
+ AttrbufRecord * const regAttrPtr = &localAttrbufRecord[TfirstfreeAttrbuf];
+ if (TfirstfreeAttrbuf >= TattrbufFilesize) {
+ TCKEY_abort(signal, 21);
+ return;
+ }//if
+ UintR Tnext = regAttrPtr->attrbuf[ZINBUF_NEXT];
+ if (TTcfirstAttrbuf == RNIL) {
+ jam();
+ regCachePtr->firstAttrbuf = TfirstfreeAttrbuf;
+ } else {
+ AttrbufRecordPtr saAttrbufptr;
+
+ saAttrbufptr.i = regCachePtr->lastAttrbuf;
+ jam();
+ if (saAttrbufptr.i >= TattrbufFilesize) {
+ TCKEY_abort(signal, 22);
+ return;
+ }//if
+ saAttrbufptr.p = &localAttrbufRecord[saAttrbufptr.i];
+ saAttrbufptr.p->attrbuf[ZINBUF_NEXT] = TfirstfreeAttrbuf;
+ }//if
+
+ cfirstfreeAttrbuf = Tnext;
+ regAttrPtr->attrbuf[ZINBUF_NEXT] = RNIL;
+ regCachePtr->lastAttrbuf = TfirstfreeAttrbuf;
+ regAttrPtr->attrbuf[ZINBUF_DATA_LEN] = Tlen;
+
+ UintR Tdata1 = signal->theData[3];
+ UintR Tdata2 = signal->theData[4];
+ UintR Tdata3 = signal->theData[5];
+ UintR Tdata4 = signal->theData[6];
+ UintR Tdata5 = signal->theData[7];
+ UintR Tdata6 = signal->theData[8];
+ UintR Tdata7 = signal->theData[9];
+ UintR Tdata8 = signal->theData[10];
+
+ regAttrPtr->attrbuf[0] = Tdata1;
+ regAttrPtr->attrbuf[1] = Tdata2;
+ regAttrPtr->attrbuf[2] = Tdata3;
+ regAttrPtr->attrbuf[3] = Tdata4;
+ regAttrPtr->attrbuf[4] = Tdata5;
+ regAttrPtr->attrbuf[5] = Tdata6;
+ regAttrPtr->attrbuf[6] = Tdata7;
+ regAttrPtr->attrbuf[7] = Tdata8;
+
+ if (Tlen > 8) {
+
+ Tdata1 = signal->theData[11];
+ Tdata2 = signal->theData[12];
+ Tdata3 = signal->theData[13];
+ Tdata4 = signal->theData[14];
+ Tdata5 = signal->theData[15];
+ Tdata6 = signal->theData[16];
+ Tdata7 = signal->theData[17];
+
+ regAttrPtr->attrbuf[8] = Tdata1;
+ regAttrPtr->attrbuf[9] = Tdata2;
+ regAttrPtr->attrbuf[10] = Tdata3;
+ regAttrPtr->attrbuf[11] = Tdata4;
+ regAttrPtr->attrbuf[12] = Tdata5;
+ regAttrPtr->attrbuf[13] = Tdata6;
+ regAttrPtr->attrbuf[14] = Tdata7;
+ jam();
+ if (Tlen > 15) {
+
+ Tdata1 = signal->theData[18];
+ Tdata2 = signal->theData[19];
+ Tdata3 = signal->theData[20];
+ Tdata4 = signal->theData[21];
+ Tdata5 = signal->theData[22];
+ Tdata6 = signal->theData[23];
+ Tdata7 = signal->theData[24];
+
+ jam();
+ regAttrPtr->attrbuf[15] = Tdata1;
+ regAttrPtr->attrbuf[16] = Tdata2;
+ regAttrPtr->attrbuf[17] = Tdata3;
+ regAttrPtr->attrbuf[18] = Tdata4;
+ regAttrPtr->attrbuf[19] = Tdata5;
+ regAttrPtr->attrbuf[20] = Tdata6;
+ regAttrPtr->attrbuf[21] = Tdata7;
+ }//if
+ }//if
+}//Dbtc::saveAttrbuf()
+
+void Dbtc::execATTRINFO(Signal* signal)
+{
+ UintR compare_transid1, compare_transid2;
+ UintR Tdata1 = signal->theData[0];
+ UintR Tlength = signal->length();
+ UintR TapiConnectFilesize = capiConnectFilesize;
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+
+ jamEntry();
+ apiConnectptr.i = Tdata1;
+ ttransid_ptr = 1;
+ if (Tdata1 >= TapiConnectFilesize) {
+ DEBUG("Drop ATTRINFO, wrong apiConnectptr");
+ TCKEY_abort(signal, 18);
+ return;
+ }//if
+
+ UintR Tdata2 = signal->theData[1];
+ UintR Tdata3 = signal->theData[2];
+ ApiConnectRecord * const regApiPtr = &localApiConnectRecord[Tdata1];
+ compare_transid1 = regApiPtr->transid[0] ^ Tdata2;
+ compare_transid2 = regApiPtr->transid[1] ^ Tdata3;
+ apiConnectptr.p = regApiPtr;
+ compare_transid1 = compare_transid1 | compare_transid2;
+
+ if (compare_transid1 != 0) {
+ DEBUG("Drop ATTRINFO, wrong transid, lenght="<<Tlength
+ << " transid("<<hex<<Tdata2<<", "<<Tdata3);
+ TCKEY_abort(signal, 19);
+ return;
+ }//if
+ if (Tlength < 4) {
+ DEBUG("Drop ATTRINFO, wrong length = " << Tlength);
+ TCKEY_abort(signal, 20);
+ return;
+ }
+ Tlength -= 3;
+ UintR TcompREC_COMMIT = (regApiPtr->apiConnectstate == CS_REC_COMMITTING);
+ UintR TcompRECEIVING = (regApiPtr->apiConnectstate == CS_RECEIVING);
+ UintR TcompBOTH = TcompREC_COMMIT | TcompRECEIVING;
+
+ if (TcompBOTH) {
+ jam();
+ if (ERROR_INSERTED(8015)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ if (ERROR_INSERTED(8016)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ CacheRecord *localCacheRecord = cacheRecord;
+ UintR TcacheFilesize = ccacheFilesize;
+ UintR TcachePtr = regApiPtr->cachePtr;
+ UintR TtcTimer = ctcTimer;
+ CacheRecord * const regCachePtr = &localCacheRecord[TcachePtr];
+ if (TcachePtr >= TcacheFilesize) {
+ TCKEY_abort(signal, 43);
+ return;
+ }//if
+ UintR TfirstfreeAttrbuf = cfirstfreeAttrbuf;
+ UintR TcurrReclenAi = regCachePtr->currReclenAi;
+ UintR TattrLen = regCachePtr->attrlength;
+
+ setApiConTimer(apiConnectptr.i, TtcTimer, __LINE__);
+ cachePtr.i = TcachePtr;
+ cachePtr.p = regCachePtr;
+ TcurrReclenAi = TcurrReclenAi + Tlength;
+ regCachePtr->currReclenAi = TcurrReclenAi;
+ int TattrlengthRemain = TattrLen - TcurrReclenAi;
+
+ if (TfirstfreeAttrbuf == RNIL) {
+ DEBUG("No more attrinfo buffers");
+ TCKEY_abort(signal, 24);
+ return;
+ }//if
+ saveAttrbuf(signal);
+ if (TattrlengthRemain == 0) {
+ /****************************************************************>*/
+ /* HERE WE HAVE FOUND THAT THE LAST SIGNAL BELONGING TO THIS */
+ /* OPERATION HAVE BEEN RECEIVED. THIS MEANS THAT WE CAN NOW REUSE */
+ /* THE API CONNECT RECORD. HOWEVER IF PREPARE OR COMMIT HAVE BEEN */
+ /* RECEIVED THEN IT IS NOT ALLOWED TO RECEIVE ANY FURTHER */
+ /* OPERATIONS. */
+ /****************************************************************>*/
+ UintR TlastConnect = regApiPtr->lastTcConnect;
+ if (TcompRECEIVING) {
+ jam();
+ regApiPtr->apiConnectstate = CS_STARTED;
+ } else {
+ jam();
+ regApiPtr->apiConnectstate = CS_START_COMMITTING;
+ }//if
+ tcConnectptr.i = TlastConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ attrinfoDihReceivedLab(signal);
+ } else if (TattrlengthRemain < 0) {
+ jam();
+ DEBUG("ATTRINFO wrong total length="<<Tlength
+ <<", TattrlengthRemain="<<TattrlengthRemain
+ <<", TattrLen="<<TattrLen
+ <<", TcurrReclenAi="<<TcurrReclenAi);
+ tcConnectptr.i = regApiPtr->lastTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ aiErrorLab(signal);
+ }//if
+ return;
+ } else if (regApiPtr->apiConnectstate == CS_START_SCAN) {
+ jam();
+ scanAttrinfoLab(signal, Tlength);
+ return;
+ } else {
+ switch (regApiPtr->apiConnectstate) {
+ case CS_ABORTING:
+ jam();
+ /* JUST IGNORE THE SIGNAL*/
+ // DEBUG("Drop ATTRINFO, CS_ABORTING");
+ return;
+ case CS_CONNECTED:
+ jam();
+ /* MOST LIKELY CAUSED BY A MISSED SIGNAL.*/
+ // DEBUG("Drop ATTRINFO, CS_CONNECTED");
+ return;
+ case CS_STARTED:
+ jam();
+ /****************************************************************>*/
+ /* MOST LIKELY CAUSED BY A MISSED SIGNAL. SEND REFUSE AND */
+ /* SET STATE TO ABORTING. SINCE A TRANSACTION WAS STARTED */
+ /* WE ALSO NEED TO ABORT THIS TRANSACTION. */
+ /****************************************************************>*/
+ terrorCode = ZSIGNAL_ERROR;
+ printState(signal, 1);
+ abortErrorLab(signal);
+ return;
+ default:
+ jam();
+ /****************************************************************>*/
+ /* SIGNAL RECEIVED IN AN UNEXPECTED STATE. WE IGNORE SIGNAL */
+ /* SINCE WE DO NOT REALLY KNOW WHERE THE ERROR OCCURRED. */
+ /****************************************************************>*/
+ DEBUG("Drop ATTRINFO, illegal state="<<regApiPtr->apiConnectstate);
+ printState(signal, 9);
+ return;
+ }//switch
+ }//if
+}//Dbtc::execATTRINFO()
+
+/* *********************************************************************>> */
+/* */
+/* MODULE: HASH MODULE */
+/* DESCRIPTION: CONTAINS THE HASH VALUE CALCULATION */
+/* *********************************************************************> */
+void Dbtc::hash(Signal* signal)
+{
+ DatabufRecordPtr locDatabufptr;
+ UintR ti;
+ UintR Tdata0;
+ UintR Tdata1;
+ UintR Tdata2;
+ UintR Tdata3;
+ UintR* Tdata32;
+ Uint64 Tdata[512];
+
+ CacheRecord * const regCachePtr = cachePtr.p;
+ Tdata32 = (UintR*)&Tdata[0];
+
+ Tdata0 = regCachePtr->keydata[0];
+ Tdata1 = regCachePtr->keydata[1];
+ Tdata2 = regCachePtr->keydata[2];
+ Tdata3 = regCachePtr->keydata[3];
+ Tdata32[0] = Tdata0;
+ Tdata32[1] = Tdata1;
+ Tdata32[2] = Tdata2;
+ Tdata32[3] = Tdata3;
+ if (regCachePtr->keylen > 4) {
+ locDatabufptr.i = regCachePtr->firstKeybuf;
+ ti = 4;
+ while (locDatabufptr.i != RNIL) {
+ ptrCheckGuard(locDatabufptr, cdatabufFilesize, databufRecord);
+ Tdata0 = locDatabufptr.p->data[0];
+ Tdata1 = locDatabufptr.p->data[1];
+ Tdata2 = locDatabufptr.p->data[2];
+ Tdata3 = locDatabufptr.p->data[3];
+ Tdata32[ti ] = Tdata0;
+ Tdata32[ti + 1] = Tdata1;
+ Tdata32[ti + 2] = Tdata2;
+ Tdata32[ti + 3] = Tdata3;
+ locDatabufptr.i = locDatabufptr.p->nextDatabuf;
+ ti += 4;
+ }//while
+ }//if
+ UintR ThashValue;
+ UintR TdistrHashValue;
+ ThashValue = md5_hash((Uint64*)&Tdata32[0], (UintR)regCachePtr->keylen);
+
+ if (regCachePtr->distributionGroupIndicator == 1) {
+ if (regCachePtr->distributionGroupType == 1) {
+ jam();
+ TdistrHashValue = (regCachePtr->distributionGroup << 6);
+ } else {
+ jam();
+ Tdata32[0] = regCachePtr->distributionGroup;
+ TdistrHashValue = md5_hash((Uint64*)&Tdata32[0], (UintR)1);
+ }//if
+ } else if (regCachePtr->distributionKeyIndicator == 1) {
+ jam();
+ TdistrHashValue = md5_hash((Uint64*)&Tdata32[0],
+ (UintR)regCachePtr->distributionKeySize);
+ } else {
+ jam();
+ TdistrHashValue = ThashValue;
+ }//if
+ thashValue = ThashValue;
+ tdistrHashValue = TdistrHashValue;
+}//Dbtc::hash()
+
+/*
+INIT_API_CONNECT_REC
+---------------------------
+*/
+/* ========================================================================= */
+/* ======= INIT_API_CONNECT_REC ======= */
+/* */
+/* ========================================================================= */
+void Dbtc::initApiConnectRec(Signal* signal,
+ ApiConnectRecord * const regApiPtr,
+ bool releaseIndexOperations)
+{
+ const TcKeyReq * const tcKeyReq = (TcKeyReq *)&signal->theData[0];
+ UintR TfailureNr = cfailure_nr;
+ UintR TtransCount = ctransCount;
+ UintR Ttransid0 = tcKeyReq->transId1;
+ UintR Ttransid1 = tcKeyReq->transId2;
+
+ regApiPtr->returnsignal = RS_TCKEYCONF;
+ regApiPtr->firstTcConnect = RNIL;
+ regApiPtr->lastTcConnect = RNIL;
+ regApiPtr->globalcheckpointid = 0;
+ regApiPtr->lqhkeyconfrec = 0;
+ regApiPtr->lqhkeyreqrec = 0;
+ regApiPtr->tckeyrec = 0;
+ regApiPtr->tcindxrec = 0;
+ regApiPtr->failureNr = TfailureNr;
+ regApiPtr->transid[0] = Ttransid0;
+ regApiPtr->transid[1] = Ttransid1;
+ regApiPtr->commitAckMarker = RNIL;
+ regApiPtr->buddyPtr = RNIL;
+ regApiPtr->currSavePointId = 0;
+ // Trigger data
+ releaseFiredTriggerData(&regApiPtr->theFiredTriggers),
+ // Index data
+ regApiPtr->indexOpReturn = false;
+ regApiPtr->noIndexOp = 0;
+ if(releaseIndexOperations)
+ releaseAllSeizedIndexOperations(regApiPtr);
+
+ ctransCount = TtransCount + 1;
+}//Dbtc::initApiConnectRec()
+
+int
+Dbtc::seizeTcRecord(Signal* signal)
+{
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+ UintR TfirstfreeTcConnect = cfirstfreeTcConnect;
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+ tcConnectptr.i = TfirstfreeTcConnect;
+ if (TfirstfreeTcConnect >= TtcConnectFilesize) {
+ int place = 3;
+ if (TfirstfreeTcConnect != RNIL) {
+ place = 10;
+ }//if
+ TCKEY_abort(signal, place);
+ return 1;
+ }//if
+ //--------------------------------------------------------------------------
+ // Optimised version of ptrAss(tcConnectptr, tcConnectRecord)
+ //--------------------------------------------------------------------------
+ TcConnectRecord * const regTcPtr =
+ &localTcConnectRecord[TfirstfreeTcConnect];
+
+ UintR TconcurrentOp = cconcurrentOp;
+ UintR TlastTcConnect = regApiPtr->lastTcConnect;
+ UintR TtcConnectptrIndex = tcConnectptr.i;
+ TcConnectRecordPtr tmpTcConnectptr;
+
+ cfirstfreeTcConnect = regTcPtr->nextTcConnect;
+ tcConnectptr.p = regTcPtr;
+
+ cconcurrentOp = TconcurrentOp + 1;
+ regTcPtr->prevTcConnect = TlastTcConnect;
+ regTcPtr->nextTcConnect = RNIL;
+ regTcPtr->accumulatingTriggerData.i = RNIL;
+ regTcPtr->accumulatingTriggerData.p = NULL;
+ regTcPtr->noFiredTriggers = 0;
+ regTcPtr->noReceivedTriggers = 0;
+ regTcPtr->triggerExecutionCount = 0;
+ regTcPtr->triggeringOperation = RNIL;
+ regTcPtr->triggerError = 0;
+ regTcPtr->isIndexOp = false;
+ regTcPtr->indexOp = RNIL;
+ regTcPtr->currentIndexId = RNIL;
+
+ regApiPtr->lastTcConnect = TtcConnectptrIndex;
+
+ if (TlastTcConnect == RNIL) {
+ jam();
+ regApiPtr->firstTcConnect = TtcConnectptrIndex;
+ } else {
+ tmpTcConnectptr.i = TlastTcConnect;
+ jam();
+ ptrCheckGuard(tmpTcConnectptr, TtcConnectFilesize, localTcConnectRecord);
+ tmpTcConnectptr.p->nextTcConnect = TtcConnectptrIndex;
+ }//if
+ return 0;
+}//Dbtc::seizeTcRecord()
+
+int
+Dbtc::seizeCacheRecord(Signal* signal)
+{
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ UintR TfirstfreeCacheRec = cfirstfreeCacheRec;
+ UintR TcacheFilesize = ccacheFilesize;
+ CacheRecord *localCacheRecord = cacheRecord;
+ if (TfirstfreeCacheRec >= TcacheFilesize) {
+ TCKEY_abort(signal, 41);
+ return 1;
+ }//if
+ CacheRecord * const regCachePtr = &localCacheRecord[TfirstfreeCacheRec];
+
+ regApiPtr->cachePtr = TfirstfreeCacheRec;
+ cfirstfreeCacheRec = regCachePtr->nextCacheRec;
+ cachePtr.i = TfirstfreeCacheRec;
+ cachePtr.p = regCachePtr;
+
+#ifdef VM_TRACE
+ // This is a good place to check that resources have
+ // been properly released from CacheRecord
+ ndbrequire(regCachePtr->firstKeybuf == RNIL);
+ ndbrequire(regCachePtr->lastKeybuf == RNIL);
+#endif
+ regCachePtr->firstKeybuf = RNIL;
+ regCachePtr->lastKeybuf = RNIL;
+ regCachePtr->firstAttrbuf = RNIL;
+ regCachePtr->lastAttrbuf = RNIL;
+ regCachePtr->currReclenAi = 0;
+ return 0;
+}//Dbtc::seizeCacheRecord()
+
+/*****************************************************************************/
+/* T C K E Y R E Q */
+/* AFTER HAVING ESTABLISHED THE CONNECT, THE APPLICATION BLOCK SENDS AN */
+/* OPERATION REQUEST TO TC. ALL NECESSARY INFORMATION TO CARRY OUT REQUEST */
+/* IS FURNISHED IN PARAMETERS. TC STORES THIS INFORMATION AND ENQUIRES */
+/* FROM DIH ABOUT THE NODES WHICH MAY HAVE THE REQUESTED DATA */
+/*****************************************************************************/
+void Dbtc::execTCKEYREQ(Signal* signal)
+{
+ UintR compare_transid1, compare_transid2;
+ UintR titcLenAiInTckeyreq;
+ UintR TkeyLength;
+ const TcKeyReq * const tcKeyReq = (TcKeyReq *)signal->getDataPtr();
+ UintR Treqinfo;
+
+ jamEntry();
+ /*-------------------------------------------------------------------------
+ * Common error routines are used for several signals, they need to know
+ * where to find the transaction identifier in the signal.
+ *-------------------------------------------------------------------------*/
+ const UintR TapiIndex = tcKeyReq->apiConnectPtr;
+ const UintR TapiMaxIndex = capiConnectFilesize;
+ const UintR TtabIndex = tcKeyReq->tableId;
+ const UintR TtabMaxIndex = ctabrecFilesize;
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+
+ ttransid_ptr = 6;
+ apiConnectptr.i = TapiIndex;
+ if (TapiIndex >= TapiMaxIndex) {
+ TCKEY_abort(signal, 6);
+ return;
+ }//if
+ if (TtabIndex >= TtabMaxIndex) {
+ TCKEY_abort(signal, 7);
+ return;
+ }//if
+
+ Treqinfo = tcKeyReq->requestInfo;
+ //--------------------------------------------------------------------------
+ // Optimised version of ptrAss(tabptr, tableRecord)
+ // Optimised version of ptrAss(apiConnectptr, apiConnectRecord)
+ //--------------------------------------------------------------------------
+ ApiConnectRecord * const regApiPtr = &localApiConnectRecord[TapiIndex];
+ apiConnectptr.p = regApiPtr;
+
+ Uint32 TstartFlag = tcKeyReq->getStartFlag(Treqinfo);
+ bool isIndexOp = regApiPtr->isIndexOp;
+ bool isIndexOpReturn = regApiPtr->indexOpReturn;
+ regApiPtr->isIndexOp = false; // Reset marker
+ switch (regApiPtr->apiConnectstate) {
+ case CS_CONNECTED:{
+ if (TstartFlag == 1 && getAllowStartTransaction() == true){
+ //---------------------------------------------------------------------
+ // Initialise API connect record if transaction is started.
+ //---------------------------------------------------------------------
+ jam();
+ initApiConnectRec(signal, regApiPtr);
+ } else {
+ if(getAllowStartTransaction() == true){
+ /*------------------------------------------------------------------
+ * WE EXPECTED A START TRANSACTION. SINCE NO OPERATIONS HAVE BEEN
+ * RECEIVED WE INDICATE THIS BY SETTING FIRST_TC_CONNECT TO RNIL TO
+ * ENSURE PROPER OPERATION OF THE COMMON ABORT HANDLING.
+ *-----------------------------------------------------------------*/
+ TCKEY_abort(signal, 0);
+ return;
+ } else {
+ /**
+ * getAllowStartTransaction() == false
+ */
+ TCKEY_abort(signal, 57);
+ return;
+ }//if
+ }
+ }
+ break;
+ case CS_STARTED:
+ //------------------------------------------------------------------------
+ // Transaction is started already. Check that the operation is on the same
+ // transaction.
+ //------------------------------------------------------------------------
+ compare_transid1 = regApiPtr->transid[0] ^ tcKeyReq->transId1;
+ compare_transid2 = regApiPtr->transid[1] ^ tcKeyReq->transId2;
+ jam();
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ TCKEY_abort(signal, 1);
+ return;
+ }//if
+ break;
+ case CS_ABORTING:
+ if (regApiPtr->abortState == AS_IDLE) {
+ if (TstartFlag == 1) {
+ //--------------------------------------------------------------------
+ // Previous transaction had been aborted and the abort was completed.
+ // It is then OK to start a new transaction again.
+ //--------------------------------------------------------------------
+ jam();
+ initApiConnectRec(signal, regApiPtr);
+ } else {
+ //--------------------------------------------------------------------
+ // The current transaction was aborted successfully.
+ // We will not do anything before we receive an operation
+ // with a start indicator. We will ignore this signal.
+ //--------------------------------------------------------------------
+ jam();
+ // DEBUG("Drop TCKEYREQ - apiConnectState=CS_ABORTING, ==AS_IDLE");
+ return;
+ }//if
+ } else {
+ //----------------------------------------------------------------------
+ // Previous transaction is still aborting
+ //----------------------------------------------------------------------
+ jam();
+ if (TstartFlag == 1) {
+ //--------------------------------------------------------------------
+ // If a new transaction tries to start while the old is
+ // still aborting, we will report this to the starting API.
+ //--------------------------------------------------------------------
+ TCKEY_abort(signal, 2);
+ return;
+ }//if
+ //----------------------------------------------------------------------
+ // Ignore signals without start indicator set when aborting transaction.
+ //----------------------------------------------------------------------
+ // DEBUG("Drop TCKEYREQ - apiConnectState=CS_ABORTING, !=AS_IDLE");
+ return;
+ }//if
+ break;
+ case CS_START_COMMITTING:
+ jam();
+ if(isIndexOpReturn || TcKeyReq::getExecutingTrigger(Treqinfo)){
+ break;
+ }
+ default:
+ jam();
+ /*----------------------------------------------------------------------
+ * IN THIS CASE THE NDBAPI IS AN UNTRUSTED ENTITY THAT HAS SENT A SIGNAL
+ * WHEN IT WAS NOT EXPECTED TO.
+ * WE MIGHT BE IN A PROCESS TO RECEIVE, PREPARE,
+ * COMMIT OR COMPLETE AND OBVIOUSLY THIS IS NOT A DESIRED EVENT.
+ * WE WILL ALWAYS COMPLETE THE ABORT HANDLING BEFORE WE ALLOW
+ * ANYTHING TO HAPPEN ON THIS CONNECTION AGAIN.
+ * THUS THERE IS NO ACTION FROM THE API THAT CAN SPEED UP THIS PROCESS.
+ *---------------------------------------------------------------------*/
+ TCKEY_abort(signal, 55);
+ return;
+ }//switch
+
+ TableRecordPtr localTabptr;
+ localTabptr.i = TtabIndex;
+ localTabptr.p = &tableRecord[TtabIndex];
+ if (localTabptr.p->checkTable(tcKeyReq->tableSchemaVersion)) {
+ ;
+ } else {
+ /*-----------------------------------------------------------------------*/
+ /* THE API IS WORKING WITH AN OLD SCHEMA VERSION. IT NEEDS REPLACEMENT. */
+ /* COULD ALSO BE THAT THE TABLE IS NOT DEFINED. */
+ /*-----------------------------------------------------------------------*/
+ TCKEY_abort(signal, 8);
+ return;
+ }//if
+
+ //-------------------------------------------------------------------------
+ // Error Insertion for testing purposes. Test to see what happens when no
+ // more TC records available.
+ //-------------------------------------------------------------------------
+ if (ERROR_INSERTED(8032)) {
+ TCKEY_abort(signal, 3);
+ return;
+ }//if
+
+ if (seizeTcRecord(signal) != 0) {
+ return;
+ }//if
+
+ if (seizeCacheRecord(signal) != 0) {
+ return;
+ }//if
+
+ TcConnectRecord * const regTcPtr = tcConnectptr.p;
+ CacheRecord * const regCachePtr = cachePtr.p;
+
+ /*
+ INIT_TC_CONNECT_REC
+ -------------------------
+ */
+ /* ---------------------------------------------------------------------- */
+ /* ------- INIT OPERATION RECORD WITH SIGNAL DATA AND RNILS ------- */
+ /* */
+ /* ---------------------------------------------------------------------- */
+
+ UintR TapiVersionNo = tcKeyReq->getAPIVersion(tcKeyReq->attrLen);
+ UintR Tlqhkeyreqrec = regApiPtr->lqhkeyreqrec;
+ regApiPtr->lqhkeyreqrec = Tlqhkeyreqrec + 1;
+ regCachePtr->apiVersionNo = TapiVersionNo;
+
+ UintR TapiConnectptrIndex = apiConnectptr.i;
+ UintR TsenderData = tcKeyReq->senderData;
+ UintR TattrLen = tcKeyReq->getAttrinfoLen(tcKeyReq->attrLen);
+ UintR TattrinfoCount = cattrinfoCount;
+
+ regTcPtr->apiConnect = TapiConnectptrIndex;
+ regTcPtr->clientData = TsenderData;
+ regTcPtr->commitAckMarker = RNIL;
+ regTcPtr->isIndexOp = isIndexOp;
+ regTcPtr->indexOp = regApiPtr->executingIndexOp;
+ regTcPtr->savePointId = regApiPtr->currSavePointId;
+ regApiPtr->executingIndexOp = RNIL;
+
+ if (TcKeyReq::getExecutingTrigger(Treqinfo)) {
+ // Save the TcOperationPtr for fireing operation
+ regTcPtr->triggeringOperation = TsenderData;
+ }
+
+ if (TcKeyReq::getExecuteFlag(Treqinfo)){
+ Uint32 currSPId = regApiPtr->currSavePointId;
+ regApiPtr->currSavePointId = ++currSPId;
+ }
+
+ regCachePtr->attrlength = TattrLen;
+ cattrinfoCount = TattrinfoCount + TattrLen;
+
+ UintR TtabptrIndex = localTabptr.i;
+ UintR TtableSchemaVersion = tcKeyReq->tableSchemaVersion;
+ Uint8 TOperationType = tcKeyReq->getOperationType(Treqinfo);
+ regCachePtr->tableref = TtabptrIndex;
+ regCachePtr->schemaVersion = TtableSchemaVersion;
+ regTcPtr->operation = TOperationType;
+
+ // Uint8 TSimpleFlag = tcKeyReq->getSimpleFlag(Treqinfo);
+ Uint8 TDirtyFlag = tcKeyReq->getDirtyFlag(Treqinfo);
+ Uint8 TInterpretedFlag = tcKeyReq->getInterpretedFlag(Treqinfo);
+ Uint8 TDistrGroupFlag = tcKeyReq->getDistributionGroupFlag(Treqinfo);
+ Uint8 TDistrGroupTypeFlag = tcKeyReq->getDistributionGroupTypeFlag(Treqinfo);
+ Uint8 TDistrKeyFlag = tcKeyReq->getDistributionKeyFlag(Treqinfo);
+ Uint8 TexecuteFlag = tcKeyReq->getExecuteFlag(Treqinfo);
+
+ //RONM_TEST Disable simple reads temporarily
+ regCachePtr->opSimple = 0;
+ // regCachePtr->opSimple = TSimpleFlag;
+ regTcPtr->dirtyOp = TDirtyFlag;
+ regCachePtr->opExec = TInterpretedFlag;
+
+ regCachePtr->distributionGroupIndicator = TDistrGroupFlag;
+ regCachePtr->distributionGroupType = TDistrGroupTypeFlag;
+ regCachePtr->distributionKeyIndicator = TDistrKeyFlag;
+
+ //-------------------------------------------------------------
+ // The next step is to read the upto three conditional words.
+ //-------------------------------------------------------------
+ Uint32 TkeyIndex;
+ Uint32* TOptionalDataPtr = (Uint32*)&tcKeyReq->scanInfo;
+ {
+ Uint32 TDistrGHIndex = tcKeyReq->getScanIndFlag(Treqinfo);
+ Uint32 TDistrKeyIndex = TDistrGHIndex + TDistrGroupFlag;
+ Uint32 TscanNode = tcKeyReq->getTakeOverScanNode(TOptionalDataPtr[0]);
+ Uint32 TscanInfo = tcKeyReq->getTakeOverScanInfo(TOptionalDataPtr[0]);
+
+ regCachePtr->scanTakeOverInd = TDistrGHIndex;
+ regCachePtr->scanNode = TscanNode;
+ regCachePtr->scanInfo = TscanInfo;
+
+ regCachePtr->distributionGroup = TOptionalDataPtr[TDistrGHIndex];
+ regCachePtr->distributionKeySize = TOptionalDataPtr[TDistrKeyIndex];
+
+ TkeyIndex = TDistrKeyIndex + TDistrKeyFlag;
+ }
+ Uint32* TkeyDataPtr = &TOptionalDataPtr[TkeyIndex];
+
+ UintR Tdata1 = TkeyDataPtr[0];
+ UintR Tdata2 = TkeyDataPtr[1];
+ UintR Tdata3 = TkeyDataPtr[2];
+ UintR Tdata4 = TkeyDataPtr[3];
+ UintR Tdata5;
+
+ regCachePtr->keydata[0] = Tdata1;
+ regCachePtr->keydata[1] = Tdata2;
+ regCachePtr->keydata[2] = Tdata3;
+ regCachePtr->keydata[3] = Tdata4;
+
+ TkeyLength = tcKeyReq->getKeyLength(Treqinfo);
+ Uint32 TAIDataIndex;
+ if (TkeyLength > 8) {
+ TAIDataIndex = TkeyIndex + 8;
+ } else {
+ if (TkeyLength == 0) {
+ TCKEY_abort(signal, 4);
+ return;
+ }//if
+ TAIDataIndex = TkeyIndex + TkeyLength;
+ }//if
+ Uint32* TAIDataPtr = &TOptionalDataPtr[TAIDataIndex];
+
+ titcLenAiInTckeyreq = tcKeyReq->getAIInTcKeyReq(Treqinfo);
+ regCachePtr->keylen = TkeyLength;
+ regCachePtr->lenAiInTckeyreq = titcLenAiInTckeyreq;
+ regCachePtr->currReclenAi = titcLenAiInTckeyreq;
+
+ Tdata1 = TAIDataPtr[0];
+ Tdata2 = TAIDataPtr[1];
+ Tdata3 = TAIDataPtr[2];
+ Tdata4 = TAIDataPtr[3];
+ Tdata5 = TAIDataPtr[4];
+
+ regCachePtr->attrinfo0 = Tdata1;
+ regCachePtr->attrinfo15[0] = Tdata2;
+ regCachePtr->attrinfo15[1] = Tdata3;
+ regCachePtr->attrinfo15[2] = Tdata4;
+ regCachePtr->attrinfo15[3] = Tdata5;
+
+ if (TOperationType == ZREAD) {
+ Uint8 TreadCount = creadCount;
+ jam();
+ regCachePtr->opLock = 0;
+ creadCount = TreadCount + 1;
+ } else if(TOperationType == ZREAD_EX){
+ Uint8 TreadCount = creadCount;
+ jam();
+ TOperationType = ZREAD;
+ regTcPtr->operation = ZREAD;
+ regCachePtr->opLock = ZUPDATE;
+ creadCount = TreadCount + 1;
+ } else {
+ if(regApiPtr->commitAckMarker == RNIL){
+ jam();
+ CommitAckMarkerPtr tmp;
+ if(!m_commitAckMarkerHash.seize(tmp)){
+ TCKEY_abort(signal, 56);
+ return;
+ } else {
+ regTcPtr->commitAckMarker = tmp.i;
+ regApiPtr->commitAckMarker = tmp.i;
+ tmp.p->transid1 = tcKeyReq->transId1;
+ tmp.p->transid2 = tcKeyReq->transId2;
+ tmp.p->apiNodeId = refToNode(regApiPtr->ndbapiBlockref);
+ tmp.p->apiConnectPtr = TapiIndex;
+ tmp.p->noOfLqhs = 0;
+ m_commitAckMarkerHash.add(tmp);
+ }
+ }
+
+ UintR Tattrlength = regCachePtr->attrlength;
+ UintR TwriteCount = cwriteCount;
+ UintR Toperationsize = coperationsize;
+ /* --------------------------------------------------------------------
+ * THIS IS A TEMPORARY TABLE, DON'T UPDATE coperationsize.
+ * THIS VARIABLE CONTROLS THE INTERVAL BETWEEN LCP'S AND
+ * TEMP TABLES DON'T PARTICIPATE.
+ * -------------------------------------------------------------------- */
+ if (localTabptr.p->storedTable) {
+ coperationsize = ((Toperationsize + Tattrlength) + TkeyLength) + 17;
+ }
+ cwriteCount = TwriteCount + 1;
+ switch (TOperationType) {
+ case ZUPDATE:
+ jam();
+ if (Tattrlength == 0) {
+ TCKEY_abort(signal, 5);
+ return;
+ }//if
+ /*---------------------------------------------------------------------*/
+ // The missing break is intentional since we also want to set the opLock
+ // variable also for updates
+ /*---------------------------------------------------------------------*/
+ case ZINSERT:
+ case ZDELETE:
+ jam();
+ regCachePtr->opLock = TOperationType;
+ break;
+ case ZWRITE:
+ jam();
+ // A write operation is originally an insert operation.
+ regCachePtr->opLock = ZINSERT;
+ break;
+ default:
+ TCKEY_abort(signal, 9);
+ return;
+ }//switch
+ }//if
+
+ Uint32 TabortOption = tcKeyReq->getAbortOption(Treqinfo);
+ regTcPtr->m_execAbortOption = TabortOption;
+
+ /*-------------------------------------------------------------------------
+ * Check error handling per operation
+ * If CommitFlag is set state accordingly and check for early abort
+ *------------------------------------------------------------------------*/
+ if (tcKeyReq->getCommitFlag(Treqinfo) == 1) {
+ ndbrequire(TexecuteFlag);
+ regApiPtr->apiConnectstate = CS_REC_COMMITTING;
+ } else {
+ /* ---------------------------------------------------------------------
+ * PREPARE TRANSACTION IS NOT IMPLEMENTED YET.
+ * ---------------------------------------------------------------------
+ * ELSIF (TREQINFO => 3) (*) 1 = 1 THEN
+ * IF PREPARE TRANSACTION THEN
+ * API_CONNECTPTR:API_CONNECTSTATE = REC_PREPARING
+ * SET STATE TO PREPARING
+ * --------------------------------------------------------------------- */
+ if (regApiPtr->apiConnectstate == CS_START_COMMITTING) {
+ jam();
+ // Trigger execution at commit
+
+ regApiPtr->apiConnectstate = CS_REC_COMMITTING;
+ } else {
+ jam();
+ regApiPtr->apiConnectstate = CS_RECEIVING;
+ }//if
+ }//if
+ if (TkeyLength <= 4) {
+ tckeyreq050Lab(signal);
+ return;
+ } else {
+ if (cfirstfreeDatabuf != RNIL) {
+ jam();
+ linkKeybuf(signal);
+ Tdata1 = TkeyDataPtr[4];
+ Tdata2 = TkeyDataPtr[5];
+ Tdata3 = TkeyDataPtr[6];
+ Tdata4 = TkeyDataPtr[7];
+
+ DatabufRecord * const regDataPtr = databufptr.p;
+ regDataPtr->data[0] = Tdata1;
+ regDataPtr->data[1] = Tdata2;
+ regDataPtr->data[2] = Tdata3;
+ regDataPtr->data[3] = Tdata4;
+ } else {
+ jam();
+ seizeDatabuferrorLab(signal);
+ return;
+ }//if
+ if (TkeyLength <= 8) {
+ jam();
+ tckeyreq050Lab(signal);
+ return;
+ } else {
+ jam();
+ /* --------------------------------------------------------------------
+ * THE TCKEYREQ DIDN'T CONTAIN ALL KEY DATA,
+ * SAVE STATE AND WAIT FOR KEYINFO
+ * --------------------------------------------------------------------*/
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ regCachePtr->save1 = 8;
+ regTcPtr->tcConnectstate = OS_WAIT_KEYINFO;
+ return;
+ }//if
+ }//if
+ return;
+}//Dbtc::execTCKEYREQ()
+
+void Dbtc::tckeyreq050Lab(Signal* signal)
+{
+ UintR tnoOfBackup;
+ UintR tnoOfStandby;
+ UintR tnodeinfo;
+
+ hash(signal); /* NOW IT IS TIME TO CALCULATE THE HASH VALUE*/
+
+ CacheRecord * const regCachePtr = cachePtr.p;
+ TcConnectRecord * const regTcPtr = tcConnectptr.p;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+
+ UintR TtcTimer = ctcTimer;
+ UintR ThashValue = thashValue;
+ UintR TdistrHashValue = tdistrHashValue;
+ UintR TdihConnectptr = regTcPtr->dihConnectptr;
+ UintR Ttableref = regCachePtr->tableref;
+
+ TableRecordPtr localTabptr;
+ localTabptr.i = Ttableref;
+ localTabptr.p = &tableRecord[localTabptr.i];
+ Uint32 schemaVersion = regCachePtr->schemaVersion;
+ if(localTabptr.p->checkTable(schemaVersion)){
+ ;
+ } else {
+ terrorCode = localTabptr.p->getErrorCode(schemaVersion);
+ TCKEY_abort(signal, 58);
+ return;
+ }
+
+ setApiConTimer(apiConnectptr.i, TtcTimer, __LINE__);
+ regCachePtr->hashValue = ThashValue;
+
+ signal->theData[0] = TdihConnectptr;
+ signal->theData[1] = Ttableref;
+ signal->theData[2] = TdistrHashValue;
+
+ /*-------------------------------------------------------------*/
+ /* FOR EFFICIENCY REASONS WE AVOID THE SIGNAL SENDING HERE AND */
+ /* PROCEED IMMEDIATELY TO DIH. IN MULTI-THREADED VERSIONS WE */
+ /* HAVE TO INSERT A MUTEX ON DIH TO ENSURE PROPER OPERATION. */
+ /* SINCE THIS SIGNAL AND DIVERIFYREQ ARE THE ONLY SIGNALS SENT */
+ /* TO DIH IN TRAFFIC IT SHOULD BE OK (3% OF THE EXECUTION TIME */
+ /* IS SPENT IN DIH AND EVEN LESS IN REPLICATED NDB. */
+ /*-------------------------------------------------------------*/
+ EXECUTE_DIRECT(DBDIH, GSN_DIGETNODESREQ, signal, 3);
+ UintR TerrorIndicator = signal->theData[0];
+ jamEntry();
+ if (TerrorIndicator != 0) {
+ execDIGETNODESREF(signal);
+ return;
+ }
+ /****************>>*/
+ /* DIGETNODESCONF >*/
+ /* ***************>*/
+
+ UintR Tdata1 = signal->theData[1];
+ UintR Tdata2 = signal->theData[2];
+ UintR Tdata3 = signal->theData[3];
+ UintR Tdata4 = signal->theData[4];
+ UintR Tdata5 = signal->theData[5];
+ UintR Tdata6 = signal->theData[6];
+
+ regCachePtr->fragmentid = Tdata1;
+ tnodeinfo = Tdata2;
+
+ regTcPtr->tcNodedata[0] = Tdata3;
+ regTcPtr->tcNodedata[1] = Tdata4;
+ regTcPtr->tcNodedata[2] = Tdata5;
+ regTcPtr->tcNodedata[3] = Tdata6;
+
+ Uint8 Toperation = regTcPtr->operation;
+ tnoOfBackup = tnodeinfo & 3;
+ tnoOfStandby = (tnodeinfo >> 8) & 3;
+
+ regCachePtr->distributionKey = (tnodeinfo >> 16) & 255;
+ if (Toperation == ZREAD) {
+ if (regCachePtr->opSimple == 1) {
+ jam();
+ /*-------------------------------------------------------------*/
+ /* A SIMPLE READ CAN SELECT ANY OF THE PRIMARY AND */
+ /* BACKUP NODES TO READ. WE WILL TRY TO SELECT THIS */
+ /* NODE IF POSSIBLE TO AVOID UNNECESSARY COMMUNICATION */
+ /* WITH SIMPLE READS. */
+ /*-------------------------------------------------------------*/
+ arrGuard(tnoOfBackup, 4);
+ UintR Tindex;
+ for (Tindex = 1; Tindex <= tnoOfBackup; Tindex++) {
+ UintR Tnode = regTcPtr->tcNodedata[Tindex];
+ UintR TownNode = cownNodeid;
+ jam();
+ if (Tnode == TownNode) {
+ jam();
+ regTcPtr->tcNodedata[0] = Tnode;
+ }//if
+ }//for
+ if (regCachePtr->attrlength == 0) {
+ /*-------------------------------------------------------------*/
+ // A simple read which does not read anything is a strange
+ // creature and we abort rather than continue.
+ /*-------------------------------------------------------------*/
+ TCKEY_abort(signal, 12);
+ return;
+ }//if
+ }//if
+ jam();
+ regTcPtr->lastReplicaNo = 0;
+ regTcPtr->noOfNodes = 1;
+ } else {
+ UintR TlastReplicaNo;
+ jam();
+ TlastReplicaNo = tnoOfBackup + tnoOfStandby;
+ regTcPtr->lastReplicaNo = (Uint8)TlastReplicaNo;
+ regTcPtr->noOfNodes = (Uint8)(TlastReplicaNo + 1);
+ }//if
+ if (regCachePtr->lenAiInTckeyreq == regCachePtr->attrlength) {
+ /****************************************************************>*/
+ /* HERE WE HAVE FOUND THAT THE LAST SIGNAL BELONGING TO THIS */
+ /* OPERATION HAVE BEEN RECEIVED. THIS MEANS THAT WE CAN NOW REUSE */
+ /* THE API CONNECT RECORD. HOWEVER IF PREPARE OR COMMIT HAVE BEEN */
+ /* RECEIVED THEN IT IS NOT ALLOWED TO RECEIVE ANY FURTHER */
+ /* OPERATIONS. WE KNOW THAT WE WILL WAIT FOR DICT NEXT. IT IS NOT */
+ /* POSSIBLE FOR THE TC CONNECTION TO BE READY YET. */
+ /****************************************************************>*/
+ switch (regApiPtr->apiConnectstate) {
+ case CS_RECEIVING:
+ jam();
+ regApiPtr->apiConnectstate = CS_STARTED;
+ break;
+ case CS_REC_COMMITTING:
+ jam();
+ regApiPtr->apiConnectstate = CS_START_COMMITTING;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//switch
+ attrinfoDihReceivedLab(signal);
+ return;
+ } else {
+ if (regCachePtr->lenAiInTckeyreq < regCachePtr->attrlength) {
+ TtcTimer = ctcTimer;
+ jam();
+ setApiConTimer(apiConnectptr.i, TtcTimer, __LINE__);
+ regTcPtr->tcConnectstate = OS_WAIT_ATTR;
+ return;
+ } else {
+ TCKEY_abort(signal, 11);
+ return;
+ }//if
+ }//if
+ return;
+}//Dbtc::tckeyreq050Lab()
+
+void Dbtc::attrinfoDihReceivedLab(Signal* signal)
+{
+ CacheRecord * const regCachePtr = cachePtr.p;
+ TcConnectRecord * const regTcPtr = tcConnectptr.p;
+ Uint16 Tnode = regTcPtr->tcNodedata[0];
+ Uint16 TscanTakeOverInd = regCachePtr->scanTakeOverInd;
+ Uint16 TscanNode = regCachePtr->scanNode;
+
+ TableRecordPtr localTabptr;
+ localTabptr.i = regCachePtr->tableref;
+ localTabptr.p = &tableRecord[localTabptr.i];
+
+ if(localTabptr.p->checkTable(regCachePtr->schemaVersion)){
+ ;
+ } else {
+ terrorCode = localTabptr.p->getErrorCode(regCachePtr->schemaVersion);
+ TCKEY_abort(signal, 58);
+ return;
+ }
+ if ((TscanTakeOverInd == 1) &&
+ (Tnode != TscanNode)) {
+ TCKEY_abort(signal, 15);
+ return;
+ }//if
+ arrGuard(Tnode, MAX_NDB_NODES);
+ packLqhkeyreq(signal, calcLqhBlockRef(Tnode));
+}//Dbtc::attrinfoDihReceivedLab()
+
+void Dbtc::packLqhkeyreq(Signal* signal,
+ BlockReference TBRef)
+{
+ CacheRecord * const regCachePtr = cachePtr.p;
+ UintR Tkeylen = regCachePtr->keylen;
+ UintR TfirstAttrbuf = regCachePtr->firstAttrbuf;
+ sendlqhkeyreq(signal, TBRef);
+ if (Tkeylen > 4) {
+ packKeyData000Lab(signal, TBRef);
+ }//if
+ packLqhkeyreq040Lab(signal,
+ TfirstAttrbuf,
+ TBRef);
+}//Dbtc::packLqhkeyreq()
+
+void Dbtc::sendlqhkeyreq(Signal* signal,
+ BlockReference TBRef)
+{
+ UintR tslrAttrLen;
+ UintR Tdata10;
+ TcConnectRecord * const regTcPtr = tcConnectptr.p;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ CacheRecord * const regCachePtr = cachePtr.p;
+#ifdef ERROR_INSERT
+ if (ERROR_INSERTED(8002)) {
+ systemErrorLab(signal);
+ }//if
+ if (ERROR_INSERTED(8007)) {
+ if (apiConnectptr.p->apiConnectstate == CS_STARTED) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ }//if
+ if (ERROR_INSERTED(8008)) {
+ if (apiConnectptr.p->apiConnectstate == CS_START_COMMITTING) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ }//if
+ if (ERROR_INSERTED(8009)) {
+ if (apiConnectptr.p->apiConnectstate == CS_STARTED) {
+ return;
+ }//if
+ }//if
+ if (ERROR_INSERTED(8010)) {
+ if (apiConnectptr.p->apiConnectstate == CS_START_COMMITTING) {
+ return;
+ }//if
+ }//if
+#endif
+
+ tslrAttrLen = 0;
+ LqhKeyReq::setAttrLen(tslrAttrLen, regCachePtr->attrlength);
+ /* ---------------------------------------------------------------------- */
+ // Bit16 == 0 since StoredProcedures are not yet supported.
+ /* ---------------------------------------------------------------------- */
+ LqhKeyReq::setDistributionKey(tslrAttrLen, regCachePtr->distributionKey);
+ LqhKeyReq::setScanTakeOverFlag(tslrAttrLen, regCachePtr->scanTakeOverInd);
+
+ Tdata10 = 0;
+ LqhKeyReq::setKeyLen(Tdata10, regCachePtr->keylen);
+ LqhKeyReq::setLastReplicaNo(Tdata10, regTcPtr->lastReplicaNo);
+ LqhKeyReq::setLockType(Tdata10, regCachePtr->opLock);
+ /* ---------------------------------------------------------------------- */
+ // Indicate Application Reference is present in bit 15
+ /* ---------------------------------------------------------------------- */
+ LqhKeyReq::setApplicationAddressFlag(Tdata10, 1);
+ LqhKeyReq::setDirtyFlag(Tdata10, regTcPtr->dirtyOp);
+ LqhKeyReq::setInterpretedFlag(Tdata10, regCachePtr->opExec);
+ LqhKeyReq::setSimpleFlag(Tdata10, regCachePtr->opSimple);
+ LqhKeyReq::setOperation(Tdata10, regTcPtr->operation);
+ /* -----------------------------------------------------------------------
+ * Sequential Number of first LQH = 0, bit 22-23
+ * IF ATTRIBUTE INFORMATION IS SENT IN TCKEYREQ,
+ * IT IS ALSO SENT IN LQHKEYREQ
+ * ----------------------------------------------------------------------- */
+ LqhKeyReq::setAIInLqhKeyReq(Tdata10, regCachePtr->lenAiInTckeyreq);
+ /* -----------------------------------------------------------------------
+ * Bit 27 == 0 since TC record is the same as the client record.
+ * Bit 28 == 0 since readLenAi can only be set after reading in LQH.
+ * ----------------------------------------------------------------------- */
+ //LqhKeyReq::setAPIVersion(Tdata10, regCachePtr->apiVersionNo);
+ Uint32 commitAckMarker = regTcPtr->commitAckMarker;
+ if(commitAckMarker != RNIL){
+ jam();
+
+ LqhKeyReq::setMarkerFlag(Tdata10, 1);
+
+ CommitAckMarker * tmp;
+ tmp = m_commitAckMarkerHash.getPtr(commitAckMarker);
+
+ /**
+ * Populate LQH array
+ */
+ const Uint32 noOfLqhs = regTcPtr->noOfNodes;
+ tmp->noOfLqhs = noOfLqhs;
+ for(Uint32 i = 0; i<noOfLqhs; i++){
+ tmp->lqhNodeId[i] = regTcPtr->tcNodedata[i];
+ }
+ }
+
+ /* ************************************************************> */
+ /* NO READ LENGTH SENT FROM TC. SEQUENTIAL NUMBER IS 1 AND IT */
+ /* IS SENT TO A PRIMARY NODE. */
+ /* ************************************************************> */
+ UintR sig0, sig1, sig2, sig3, sig4, sig5, sig6;
+
+ LqhKeyReq * const lqhKeyReq = (LqhKeyReq *)signal->getDataPtrSend();
+
+ sig0 = tcConnectptr.i;
+ sig2 = regCachePtr->hashValue;
+ sig4 = cownref;
+ sig5 = regTcPtr->savePointId;
+
+ lqhKeyReq->clientConnectPtr = sig0;
+ lqhKeyReq->attrLen = tslrAttrLen;
+ lqhKeyReq->hashValue = sig2;
+ lqhKeyReq->requestInfo = Tdata10;
+ lqhKeyReq->tcBlockref = sig4;
+ lqhKeyReq->savePointId = sig5;
+
+ sig0 = regCachePtr->tableref + (regCachePtr->schemaVersion << 16);
+ sig1 = regCachePtr->fragmentid + (regTcPtr->tcNodedata[1] << 16);
+ sig2 = regApiPtr->transid[0];
+ sig3 = regApiPtr->transid[1];
+ sig4 = regApiPtr->ndbapiBlockref;
+ sig5 = regTcPtr->clientData;
+ sig6 = regCachePtr->scanInfo;
+
+ lqhKeyReq->tableSchemaVersion = sig0;
+ lqhKeyReq->fragmentData = sig1;
+ lqhKeyReq->transId1 = sig2;
+ lqhKeyReq->transId2 = sig3;
+ lqhKeyReq->scanInfo = sig6;
+
+ lqhKeyReq->variableData[0] = sig4;
+ lqhKeyReq->variableData[1] = sig5;
+
+ UintR nextPos = 2;
+
+ if (regTcPtr->lastReplicaNo > 1) {
+ sig0 = (UintR)regTcPtr->tcNodedata[2] +
+ (UintR)(regTcPtr->tcNodedata[3] << 16);
+ lqhKeyReq->variableData[nextPos] = sig0;
+ nextPos++;
+ }//if
+
+ sig0 = regCachePtr->keydata[0];
+ sig1 = regCachePtr->keydata[1];
+ sig2 = regCachePtr->keydata[2];
+ sig3 = regCachePtr->keydata[3];
+ UintR Tkeylen = regCachePtr->keylen;
+
+ lqhKeyReq->variableData[nextPos + 0] = sig0;
+ lqhKeyReq->variableData[nextPos + 1] = sig1;
+ lqhKeyReq->variableData[nextPos + 2] = sig2;
+ lqhKeyReq->variableData[nextPos + 3] = sig3;
+
+ if (Tkeylen < 4) {
+ nextPos += Tkeylen;
+ } else {
+ nextPos += 4;
+ }//if
+
+ sig0 = regCachePtr->attrinfo0;
+ sig1 = regCachePtr->attrinfo15[0];
+ sig2 = regCachePtr->attrinfo15[1];
+ sig3 = regCachePtr->attrinfo15[2];
+ sig4 = regCachePtr->attrinfo15[3];
+ UintR TlenAi = regCachePtr->lenAiInTckeyreq;
+
+ lqhKeyReq->variableData[nextPos + 0] = sig0;
+ lqhKeyReq->variableData[nextPos + 1] = sig1;
+ lqhKeyReq->variableData[nextPos + 2] = sig2;
+ lqhKeyReq->variableData[nextPos + 3] = sig3;
+ lqhKeyReq->variableData[nextPos + 4] = sig4;
+
+ nextPos += TlenAi;
+
+ // Reset trigger count
+ regTcPtr->accumulatingTriggerData.i = RNIL;
+ regTcPtr->accumulatingTriggerData.p = NULL;
+ regTcPtr->noFiredTriggers = 0;
+ regTcPtr->triggerExecutionCount = 0;
+
+ sendSignal(TBRef, GSN_LQHKEYREQ, signal,
+ nextPos + LqhKeyReq::FixedSignalLength, JBB);
+}//Dbtc::sendlqhkeyreq()
+
+void Dbtc::packLqhkeyreq040Lab(Signal* signal,
+ UintR anAttrBufIndex,
+ BlockReference TBRef)
+{
+ TcConnectRecord * const regTcPtr = tcConnectptr.p;
+ CacheRecord * const regCachePtr = cachePtr.p;
+#ifdef ERROR_INSERT
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ if (ERROR_INSERTED(8009)) {
+ if (regApiPtr->apiConnectstate == CS_STARTED) {
+ attrbufptr.i = RNIL;
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ }//if
+ if (ERROR_INSERTED(8010)) {
+ if (regApiPtr->apiConnectstate == CS_START_COMMITTING) {
+ attrbufptr.i = RNIL;
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ }//if
+#endif
+
+ UintR TattrbufFilesize = cattrbufFilesize;
+ AttrbufRecord *localAttrbufRecord = attrbufRecord;
+ while (1) {
+ if (anAttrBufIndex == RNIL) {
+ UintR TtcTimer = ctcTimer;
+ UintR Tread = (regTcPtr->operation == ZREAD);
+ UintR Tsimple = (regCachePtr->opSimple == ZTRUE);
+ UintR Tboth = Tread & Tsimple;
+ setApiConTimer(apiConnectptr.i, TtcTimer, __LINE__);
+ jam();
+ /*--------------------------------------------------------------------
+ * WE HAVE SENT ALL THE SIGNALS OF THIS OPERATION. SET STATE AND EXIT.
+ *---------------------------------------------------------------------*/
+ releaseAttrinfo(signal);
+ if (Tboth) {
+ jam();
+ releaseSimpleRead(signal);
+ return;
+ }//if
+ regTcPtr->tcConnectstate = OS_OPERATING;
+ return;
+ }//if
+ if (anAttrBufIndex < TattrbufFilesize) {
+ AttrbufRecord * const regAttrPtr = &localAttrbufRecord[anAttrBufIndex];
+ anAttrBufIndex = regAttrPtr->attrbuf[ZINBUF_NEXT];
+ sendAttrinfo(signal,
+ tcConnectptr.i,
+ regAttrPtr,
+ TBRef);
+ } else {
+ TCKEY_abort(signal, 17);
+ return;
+ }//if
+ }//while
+}//Dbtc::packLqhkeyreq040Lab()
+
+/* ========================================================================= */
+/* ------- RELEASE ALL ATTRINFO RECORDS IN AN OPERATION RECORD ------- */
+/* ========================================================================= */
+void Dbtc::releaseAttrinfo(Signal* signal)
+{
+ UintR Tmp;
+ AttrbufRecordPtr Tattrbufptr;
+ CacheRecord * const regCachePtr = cachePtr.p;
+ UintR TattrbufFilesize = cattrbufFilesize;
+ UintR TfirstfreeAttrbuf = cfirstfreeAttrbuf;
+ Tattrbufptr.i = regCachePtr->firstAttrbuf;
+ AttrbufRecord *localAttrbufRecord = attrbufRecord;
+
+ while (Tattrbufptr.i < TattrbufFilesize) {
+ Tattrbufptr.p = &localAttrbufRecord[Tattrbufptr.i];
+ Tmp = Tattrbufptr.p->attrbuf[ZINBUF_NEXT];
+ Tattrbufptr.p->attrbuf[ZINBUF_NEXT] = TfirstfreeAttrbuf;
+ TfirstfreeAttrbuf = Tattrbufptr.i;
+ Tattrbufptr.i = Tmp;
+ jam();
+ }//while
+ if (Tattrbufptr.i == RNIL) {
+//---------------------------------------------------
+// Now we will release the cache record at the same
+// time as releasing the attrinfo records.
+//---------------------------------------------------
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ UintR TfirstfreeCacheRec = cfirstfreeCacheRec;
+ UintR TCacheIndex = cachePtr.i;
+ cfirstfreeAttrbuf = TfirstfreeAttrbuf;
+ regCachePtr->nextCacheRec = TfirstfreeCacheRec;
+ cfirstfreeCacheRec = TCacheIndex;
+ regApiPtr->cachePtr = RNIL;
+ return;
+ }//if
+ systemErrorLab(signal);
+ return;
+}//Dbtc::releaseAttrinfo()
+
+/* ========================================================================= */
+/* ------- RELEASE ALL RECORDS CONNECTED TO A SIMPLE OPERATION ------- */
+/* ========================================================================= */
+void Dbtc::releaseSimpleRead(Signal* signal)
+{
+ unlinkReadyTcCon(signal);
+ releaseTcCon(signal);
+
+ /**
+ * No LQHKEYCONF in Simple/Dirty read
+ * Therefore decrese no LQHKEYCONF(REF) we are waiting for
+ */
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ UintR TsimpleReadCount = csimpleReadCount;
+ UintR Tlqhkeyreqrec = regApiPtr->lqhkeyreqrec;
+
+ csimpleReadCount = TsimpleReadCount + 1;
+ regApiPtr->lqhkeyreqrec = Tlqhkeyreqrec - 1;
+
+ /**
+ * If start committing and no operation in lists
+ * simply return
+ */
+ if (regApiPtr->apiConnectstate == CS_START_COMMITTING &&
+ regApiPtr->firstTcConnect == RNIL) {
+
+ jam();
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ regApiPtr->apiConnectstate = CS_CONNECTED;
+ return;
+ }//if
+
+ /**
+ * Else Emulate LQHKEYCONF
+ */
+ lqhKeyConf_checkTransactionState(signal, regApiPtr);
+
+}//Dbtc::releaseSimpleRead()
+
+/* ------------------------------------------------------------------------- */
+/* ------- CHECK IF ALL TC CONNECTIONS ARE COMPLETED ------- */
+/* ------------------------------------------------------------------------- */
+void Dbtc::unlinkReadyTcCon(Signal* signal)
+{
+ TcConnectRecordPtr urtTcConnectptr;
+
+ TcConnectRecord * const regTcPtr = tcConnectptr.p;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ if (regTcPtr->prevTcConnect != RNIL) {
+ jam();
+ urtTcConnectptr.i = regTcPtr->prevTcConnect;
+ ptrCheckGuard(urtTcConnectptr, TtcConnectFilesize, localTcConnectRecord);
+ urtTcConnectptr.p->nextTcConnect = regTcPtr->nextTcConnect;
+ } else {
+ jam();
+ regApiPtr->firstTcConnect = regTcPtr->nextTcConnect;
+ }//if
+ if (regTcPtr->nextTcConnect != RNIL) {
+ jam();
+ urtTcConnectptr.i = regTcPtr->nextTcConnect;
+ ptrCheckGuard(urtTcConnectptr, TtcConnectFilesize, localTcConnectRecord);
+ urtTcConnectptr.p->prevTcConnect = regTcPtr->prevTcConnect;
+ } else {
+ jam();
+ regApiPtr->lastTcConnect = tcConnectptr.p->prevTcConnect;
+ }//if
+}//Dbtc::unlinkReadyTcCon()
+
+void Dbtc::releaseTcCon(Signal* signal)
+{
+ TcConnectRecord * const regTcPtr = tcConnectptr.p;
+ UintR TfirstfreeTcConnect = cfirstfreeTcConnect;
+ UintR TconcurrentOp = cconcurrentOp;
+ UintR TtcConnectptrIndex = tcConnectptr.i;
+
+ regTcPtr->tcConnectstate = OS_CONNECTED;
+ regTcPtr->nextTcConnect = TfirstfreeTcConnect;
+ regTcPtr->apiConnect = RNIL;
+ regTcPtr->isIndexOp = false;
+ regTcPtr->indexOp = RNIL;
+ cfirstfreeTcConnect = TtcConnectptrIndex;
+ cconcurrentOp = TconcurrentOp - 1;
+}//Dbtc::releaseTcCon()
+
+void Dbtc::execPACKED_SIGNAL(Signal* signal)
+{
+ LqhKeyConf * const lqhKeyConf = (LqhKeyConf *)signal->getDataPtr();
+
+ UintR Ti;
+ UintR Tstep = 0;
+ UintR Tlength;
+ UintR TpackedData[28];
+ UintR Tdata1, Tdata2, Tdata3, Tdata4;
+
+ jamEntry();
+ Tlength = signal->length();
+ if (Tlength > 25) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ Uint32* TpackDataPtr;
+ for (Ti = 0; Ti < Tlength; Ti += 4) {
+ Uint32* TsigDataPtr = &signal->theData[Ti];
+ Tdata1 = TsigDataPtr[0];
+ Tdata2 = TsigDataPtr[1];
+ Tdata3 = TsigDataPtr[2];
+ Tdata4 = TsigDataPtr[3];
+
+ TpackDataPtr = &TpackedData[Ti];
+ TpackDataPtr[0] = Tdata1;
+ TpackDataPtr[1] = Tdata2;
+ TpackDataPtr[2] = Tdata3;
+ TpackDataPtr[3] = Tdata4;
+ }//for
+ while (Tlength > Tstep) {
+
+ TpackDataPtr = &TpackedData[Tstep];
+ Tdata1 = TpackDataPtr[0];
+ Tdata2 = TpackDataPtr[1];
+ Tdata3 = TpackDataPtr[2];
+
+ lqhKeyConf->connectPtr = Tdata1 & 0x0FFFFFFF;
+ lqhKeyConf->opPtr = Tdata2;
+ lqhKeyConf->userRef = Tdata3;
+
+ switch (Tdata1 >> 28) {
+ case ZCOMMITTED:
+ signal->header.theLength = 3;
+ execCOMMITTED(signal);
+ Tstep += 3;
+ break;
+ case ZCOMPLETED:
+ signal->header.theLength = 3;
+ execCOMPLETED(signal);
+ Tstep += 3;
+ break;
+ case ZLQHKEYCONF:
+ jam();
+ Tdata1 = TpackDataPtr[3];
+ Tdata2 = TpackDataPtr[4];
+ Tdata3 = TpackDataPtr[5];
+ Tdata4 = TpackDataPtr[6];
+
+ lqhKeyConf->readLen = Tdata1;
+ lqhKeyConf->transId1 = Tdata2;
+ lqhKeyConf->transId2 = Tdata3;
+ lqhKeyConf->noFiredTriggers = Tdata4;
+ signal->header.theLength = LqhKeyConf::SignalLength;
+ execLQHKEYCONF(signal);
+ Tstep += LqhKeyConf::SignalLength;
+ break;
+ default:
+ systemErrorLab(signal);
+ return;
+ }//switch
+ }//while
+ return;
+}//Dbtc::execPACKED_SIGNAL()
+
+void Dbtc::execLQHKEYCONF(Signal* signal)
+{
+ const LqhKeyConf * const lqhKeyConf = (LqhKeyConf *)signal->getDataPtr();
+ UintR compare_transid1, compare_transid2;
+ BlockReference tlastLqhBlockref;
+ UintR tlastLqhConnect;
+ UintR treadlenAi;
+ UintR TtcConnectptrIndex;
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+
+ tlastLqhConnect = lqhKeyConf->connectPtr;
+ TtcConnectptrIndex = lqhKeyConf->opPtr;
+ tlastLqhBlockref = lqhKeyConf->userRef;
+ treadlenAi = lqhKeyConf->readLen;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+
+ /*------------------------------------------------------------------------
+ * NUMBER OF EXTERNAL TRIGGERS FIRED IN DATA[6]
+ * OPERATION IS NOW COMPLETED. CHECK FOR CORRECT OPERATION POINTER
+ * TO ENSURE NO CRASHES BECAUSE OF ERRONEUS NODES. CHECK STATE OF
+ * OPERATION. THEN SET OPERATION STATE AND RETRIEVE ALL POINTERS
+ * OF THIS OPERATION. PUT COMPLETED OPERATION IN LIST OF COMPLETED
+ * OPERATIONS ON THE LQH CONNECT RECORD.
+ *------------------------------------------------------------------------
+ * THIS SIGNAL ALWAYS ARRIVE BEFORE THE ABORTED SIGNAL ARRIVES SINCE IT USES
+ * THE SAME PATH BACK TO TC AS THE ABORTED SIGNAL DO. WE DO HOWEVER HAVE A
+ * PROBLEM WHEN WE ENCOUNTER A TIME-OUT WAITING FOR THE ABORTED SIGNAL.
+ * THEN THIS SIGNAL MIGHT ARRIVE WHEN THE TC CONNECT RECORD HAVE BEEN REUSED
+ * BY OTHER TRANSACTION THUS WE CHECK THE TRANSACTION ID OF THE SIGNAL
+ * BEFORE ACCEPTING THIS SIGNAL.
+ * Due to packing of LQHKEYCONF the ABORTED signal can now arrive before
+ * this.
+ * This is more reason to ignore the signal if not all states are correct.
+ *------------------------------------------------------------------------*/
+ if (TtcConnectptrIndex >= TtcConnectFilesize) {
+ TCKEY_abort(signal, 25);
+ return;
+ }//if
+ TcConnectRecord* const regTcPtr = &localTcConnectRecord[TtcConnectptrIndex];
+ OperationState TtcConnectstate = regTcPtr->tcConnectstate;
+ tcConnectptr.i = TtcConnectptrIndex;
+ tcConnectptr.p = regTcPtr;
+ if (TtcConnectstate != OS_OPERATING) {
+ warningReport(signal, 23);
+ return;
+ }//if
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+ UintR TapiConnectptrIndex = regTcPtr->apiConnect;
+ UintR TapiConnectFilesize = capiConnectFilesize;
+ UintR Ttrans1 = lqhKeyConf->transId1;
+ UintR Ttrans2 = lqhKeyConf->transId2;
+ regTcPtr->noFiredTriggers = lqhKeyConf->noFiredTriggers;
+
+ if (TapiConnectptrIndex >= TapiConnectFilesize) {
+ TCKEY_abort(signal, 29);
+ return;
+ }//if
+ ApiConnectRecord * const regApiPtr =
+ &localApiConnectRecord[TapiConnectptrIndex];
+ apiConnectptr.i = TapiConnectptrIndex;
+ apiConnectptr.p = regApiPtr;
+ compare_transid1 = regApiPtr->transid[0] ^ Ttrans1;
+ compare_transid2 = regApiPtr->transid[1] ^ Ttrans2;
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ warningReport(signal, 24);
+ return;
+ }//if
+
+#ifdef ERROR_INSERT
+ if (ERROR_INSERTED(8029)) {
+ systemErrorLab(signal);
+ }//if
+ if (ERROR_INSERTED(8003)) {
+ if (regApiPtr->apiConnectstate == CS_STARTED) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ }//if
+ if (ERROR_INSERTED(8004)) {
+ if (regApiPtr->apiConnectstate == CS_RECEIVING) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ }//if
+ if (ERROR_INSERTED(8005)) {
+ if (regApiPtr->apiConnectstate == CS_REC_COMMITTING) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ }//if
+ if (ERROR_INSERTED(8006)) {
+ if (regApiPtr->apiConnectstate == CS_START_COMMITTING) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ }//if
+ if (ERROR_INSERTED(8023)) {
+ SET_ERROR_INSERT_VALUE(8024);
+ return;
+ }//if
+#endif
+ UintR TtcTimer = ctcTimer;
+ regTcPtr->lastLqhCon = tlastLqhConnect;
+ regTcPtr->lastLqhNodeId = refToNode(tlastLqhBlockref);
+
+ UintR Ttckeyrec = (UintR)regApiPtr->tckeyrec;
+ UintR TclientData = regTcPtr->clientData;
+ UintR TdirtyOp = regTcPtr->dirtyOp;
+ ConnectionState TapiConnectstate = regApiPtr->apiConnectstate;
+ if (Ttckeyrec > (ZTCOPCONF_SIZE - 2)) {
+ TCKEY_abort(signal, 30);
+ return;
+ }
+ if (TapiConnectstate == CS_ABORTING) {
+ warningReport(signal, 27);
+ return;
+ }//if
+
+ setApiConTimer(apiConnectptr.i, TtcTimer, __LINE__);
+
+ if (regTcPtr->isIndexOp) {
+ jam();
+ // This was an internal TCKEYREQ
+ // will be returned unpacked
+ regTcPtr->attrInfoLen = treadlenAi;
+ } else {
+ jam();
+ regApiPtr->tcSendArray[Ttckeyrec] = TclientData;
+ regApiPtr->tcSendArray[Ttckeyrec + 1] = treadlenAi;
+ if ((regTcPtr->noFiredTriggers == 0) &&
+ (regTcPtr->triggeringOperation == RNIL)) {
+ jam();
+ /*
+ Skip counting triggering operations the first round
+ since they will enter execLQHKEYCONF a second time
+ Skip counting internally generated TcKeyReq
+ */
+ regApiPtr->tckeyrec += 2;
+ }//if
+ }//if
+ if (TdirtyOp == ZTRUE) {
+ UintR Tlqhkeyreqrec = regApiPtr->lqhkeyreqrec;
+ jam();
+ releaseDirtyWrite(signal);
+ regApiPtr->lqhkeyreqrec = Tlqhkeyreqrec - 1;
+ } else {
+ jam();
+ if (regTcPtr->noFiredTriggers == 0) {
+ jam();
+ // No triggers to execute
+ UintR Tlqhkeyconfrec = regApiPtr->lqhkeyconfrec;
+ regApiPtr->lqhkeyconfrec = Tlqhkeyconfrec + 1;
+ regTcPtr->tcConnectstate = OS_PREPARED;
+ }
+ }//if
+
+ /**
+ * And now decide what to do next
+ */
+ if (regTcPtr->triggeringOperation != RNIL) {
+ jam();
+ // This operation was created by a trigger execting operation
+ // Restart it if we have executed all it's triggers
+ TcConnectRecordPtr opPtr;
+
+ opPtr.i = regTcPtr->triggeringOperation;
+ ptrCheckGuard(opPtr, ctcConnectFilesize, localTcConnectRecord);
+ opPtr.p->triggerExecutionCount--;
+ if (opPtr.p->triggerExecutionCount == 0) {
+ /*
+ We have completed current trigger execution
+ Continue triggering operation
+ */
+ jam();
+ regTcPtr->triggeringOperation = RNIL;
+ continueTriggeringOp(signal, opPtr.p);
+ }
+ } else if (regTcPtr->noFiredTriggers == 0) {
+ // This operation did not fire any triggers, finish operation
+ jam();
+ if (regTcPtr->isIndexOp) {
+ jam();
+ setupIndexOpReturn(regApiPtr, regTcPtr);
+ }
+ lqhKeyConf_checkTransactionState(signal, regApiPtr);
+ } else {
+ // We have fired triggers
+ jam();
+ saveTriggeringOpState(signal, regTcPtr);
+ if (regTcPtr->noReceivedTriggers == regTcPtr->noFiredTriggers) {
+ ApiConnectRecordPtr transPtr;
+
+ // We have received all data
+ jam();
+ transPtr.i = TapiConnectptrIndex;
+ transPtr.p = regApiPtr;
+ executeTriggers(signal, &transPtr);
+ }
+ // else wait for more trigger data
+ }
+}//Dbtc::execLQHKEYCONF()
+
+
+void Dbtc::setupIndexOpReturn(ApiConnectRecord* regApiPtr,
+ TcConnectRecord* regTcPtr)
+{
+ regApiPtr->indexOpReturn = true;
+ regApiPtr->indexOp = regTcPtr->indexOp;
+ regApiPtr->clientData = regTcPtr->clientData;
+ regApiPtr->attrInfoLen = regTcPtr->attrInfoLen;
+}
+
+/**
+ * lqhKeyConf_checkTransactionState
+ *
+ * This functions checks state variables, and
+ * decides if it should wait for more LQHKEYCONF signals
+ * or if it should start commiting
+ */
+void
+Dbtc::lqhKeyConf_checkTransactionState(Signal * signal,
+ ApiConnectRecord * const apiConnectPtrP)
+{
+/*---------------------------------------------------------------*/
+/* IF THE COMMIT FLAG IS SET IN SIGNAL TCKEYREQ THEN DBTC HAS TO */
+/* SEND TCKEYCONF FOR ALL OPERATIONS EXCEPT THE LAST ONE. WHEN */
+/* THE TRANSACTION THEN IS COMMITTED TCKEYCONF IS SENT FOR THE */
+/* WHOLE TRANSACTION */
+/* IF THE COMMIT FLAG IS NOT RECECIVED DBTC WILL SEND TCKEYCONF */
+/* FOR ALL OPERATIONS, AND THEN WAIT FOR THE API TO CONCLUDE THE */
+/* TRANSACTION */
+/*---------------------------------------------------------------*/
+ ConnectionState TapiConnectstate = apiConnectPtrP->apiConnectstate;
+ UintR Tlqhkeyconfrec = apiConnectPtrP->lqhkeyconfrec;
+ UintR Tlqhkeyreqrec = apiConnectPtrP->lqhkeyreqrec;
+ int TnoOfOutStanding = Tlqhkeyreqrec - Tlqhkeyconfrec;
+
+ switch (TapiConnectstate) {
+ case CS_START_COMMITTING:
+ if (TnoOfOutStanding == 0) {
+ jam();
+ diverify010Lab(signal);
+ return;
+ } else if (TnoOfOutStanding > 0) {
+ if (apiConnectPtrP->tckeyrec == ZTCOPCONF_SIZE) {
+ jam();
+ sendtckeyconf(signal, 0);
+ return;
+ } else if (apiConnectPtrP->indexOpReturn) {
+ jam();
+ sendtckeyconf(signal, 0);
+ return;
+ }//if
+ jam();
+ return;
+ } else {
+ TCKEY_abort(signal, 44);
+ return;
+ }//if
+ return;
+ case CS_STARTED:
+ case CS_RECEIVING:
+ if (TnoOfOutStanding == 0) {
+ jam();
+ sendtckeyconf(signal, 0);
+ return;
+ } else {
+ if (apiConnectPtrP->tckeyrec == ZTCOPCONF_SIZE) {
+ jam();
+ sendtckeyconf(signal, 0);
+ return;
+ } else if (apiConnectPtrP->indexOpReturn) {
+ jam();
+ sendtckeyconf(signal, 0);
+ return;
+ }//if
+ jam();
+ }//if
+ return;
+ case CS_REC_COMMITTING:
+ if (TnoOfOutStanding > 0) {
+ if (apiConnectPtrP->tckeyrec == ZTCOPCONF_SIZE) {
+ jam();
+ sendtckeyconf(signal, 0);
+ return;
+ } else if (apiConnectPtrP->indexOpReturn) {
+ jam();
+ sendtckeyconf(signal, 0);
+ return;
+ }//if
+ jam();
+ return;
+ }//if
+ TCKEY_abort(signal, 45);
+ return;
+ case CS_CONNECTED:
+ jam();
+/*---------------------------------------------------------------*/
+/* WE HAVE CONCLUDED THE TRANSACTION SINCE IT WAS ONLY */
+/* CONSISTING OF DIRTY WRITES AND ALL OF THOSE WERE */
+/* COMPLETED. ENSURE TCKEYREC IS ZERO TO PREVENT ERRORS. */
+/*---------------------------------------------------------------*/
+ apiConnectPtrP->tckeyrec = 0;
+ return;
+ default:
+ TCKEY_abort(signal, 46);
+ return;
+ }//switch
+}//Dbtc::lqhKeyConf_checkTransactionState()
+
+void Dbtc::sendtckeyconf(Signal* signal, UintR TcommitFlag)
+{
+ HostRecordPtr localHostptr;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ const UintR TopWords = (UintR)regApiPtr->tckeyrec;
+ localHostptr.i = refToNode(regApiPtr->ndbapiBlockref);
+ const Uint32 type = getNodeInfo(localHostptr.i).m_type;
+ const bool is_api = (type >= NodeInfo::API && type <= NodeInfo::REP);
+ const BlockNumber TblockNum = refToBlock(regApiPtr->ndbapiBlockref);
+ const Uint32 Tmarker = (regApiPtr->commitAckMarker == RNIL) ? 0 : 1;
+ ptrAss(localHostptr, hostRecord);
+ UintR TcurrLen = localHostptr.p->noOfWordsTCKEYCONF;
+ UintR confInfo = 0;
+ TcKeyConf::setCommitFlag(confInfo, TcommitFlag);
+ TcKeyConf::setMarkerFlag(confInfo, Tmarker);
+ const UintR TpacketLen = 6 + TopWords;
+ regApiPtr->tckeyrec = 0;
+
+ if (regApiPtr->indexOpReturn) {
+ jam();
+ // Return internally generated TCKEY
+ TcKeyConf * const tcKeyConf = (TcKeyConf *)signal->getDataPtrSend();
+ TcKeyConf::setNoOfOperations(confInfo, 1);
+ tcKeyConf->apiConnectPtr = regApiPtr->indexOp;
+ tcKeyConf->gci = regApiPtr->globalcheckpointid;
+ tcKeyConf->confInfo = confInfo;
+ tcKeyConf->transId1 = regApiPtr->transid[0];
+ tcKeyConf->transId2 = regApiPtr->transid[1];
+ tcKeyConf->operations[0].apiOperationPtr = regApiPtr->clientData;
+ tcKeyConf->operations[0].attrInfoLen = regApiPtr->attrInfoLen;
+ Uint32 sigLen = TcKeyConf::StaticLength + TcKeyConf::OperationLength;
+ EXECUTE_DIRECT(DBTC, GSN_TCKEYCONF, signal, sigLen);
+ regApiPtr->indexOpReturn = false;
+ if (TopWords == 0) {
+ jam();
+ return; // No queued TcKeyConf
+ }//if
+ }//if
+
+ TcKeyConf::setNoOfOperations(confInfo, (TopWords >> 1));
+ if ((TpacketLen > 25) || !is_api){
+ TcKeyConf * const tcKeyConf = (TcKeyConf *)signal->getDataPtrSend();
+
+ jam();
+ tcKeyConf->apiConnectPtr = regApiPtr->ndbapiConnect;
+ tcKeyConf->gci = regApiPtr->globalcheckpointid;;
+ tcKeyConf->confInfo = confInfo;
+ tcKeyConf->transId1 = regApiPtr->transid[0];
+ tcKeyConf->transId2 = regApiPtr->transid[1];
+ copyFromToLen(&regApiPtr->tcSendArray[0],
+ (UintR*)&tcKeyConf->operations,
+ (UintR)ZTCOPCONF_SIZE);
+ sendSignal(regApiPtr->ndbapiBlockref,
+ GSN_TCKEYCONF, signal, (TpacketLen - 1), JBB);
+ return;
+ } else if (((TcurrLen + TpacketLen) > 25) && (TcurrLen > 0)) {
+ jam();
+ sendPackedTCKEYCONF(signal, localHostptr.p, localHostptr.i);
+ TcurrLen = 0;
+ } else {
+ jam();
+ updatePackedList(signal, localHostptr.p, localHostptr.i);
+ }//if
+ // -------------------------------------------------------------------------
+ // The header contains the block reference of receiver plus the real signal
+ // length - 3, since we have the real signal length plus one additional word
+ // for the header we have to do - 4.
+ // -------------------------------------------------------------------------
+ UintR Tpack0 = (TblockNum << 16) + (TpacketLen - 4);
+ UintR Tpack1 = regApiPtr->ndbapiConnect;
+ UintR Tpack2 = regApiPtr->globalcheckpointid;
+ UintR Tpack3 = confInfo;
+ UintR Tpack4 = regApiPtr->transid[0];
+ UintR Tpack5 = regApiPtr->transid[1];
+
+ localHostptr.p->noOfWordsTCKEYCONF = TcurrLen + TpacketLen;
+
+ localHostptr.p->packedWordsTCKEYCONF[TcurrLen + 0] = Tpack0;
+ localHostptr.p->packedWordsTCKEYCONF[TcurrLen + 1] = Tpack1;
+ localHostptr.p->packedWordsTCKEYCONF[TcurrLen + 2] = Tpack2;
+ localHostptr.p->packedWordsTCKEYCONF[TcurrLen + 3] = Tpack3;
+ localHostptr.p->packedWordsTCKEYCONF[TcurrLen + 4] = Tpack4;
+ localHostptr.p->packedWordsTCKEYCONF[TcurrLen + 5] = Tpack5;
+
+ UintR Ti;
+ for (Ti = 6; Ti < TpacketLen; Ti++) {
+ localHostptr.p->packedWordsTCKEYCONF[TcurrLen + Ti] =
+ regApiPtr->tcSendArray[Ti - 6];
+ }//for
+}//Dbtc::sendtckeyconf()
+
+void Dbtc::copyFromToLen(UintR* sourceBuffer, UintR* destBuffer, UintR Tlen)
+{
+ UintR Tindex = 0;
+ UintR Ti;
+ while (Tlen >= 4) {
+ UintR Tdata0 = sourceBuffer[Tindex + 0];
+ UintR Tdata1 = sourceBuffer[Tindex + 1];
+ UintR Tdata2 = sourceBuffer[Tindex + 2];
+ UintR Tdata3 = sourceBuffer[Tindex + 3];
+ Tlen -= 4;
+ destBuffer[Tindex + 0] = Tdata0;
+ destBuffer[Tindex + 1] = Tdata1;
+ destBuffer[Tindex + 2] = Tdata2;
+ destBuffer[Tindex + 3] = Tdata3;
+ Tindex += 4;
+ }//while
+ for (Ti = 0; Ti < Tlen; Ti++, Tindex++) {
+ destBuffer[Tindex] = sourceBuffer[Tindex];
+ }//for
+}//Dbtc::copyFromToLen()
+
+void Dbtc::execSEND_PACKED(Signal* signal)
+{
+ HostRecordPtr Thostptr;
+ HostRecord *localHostRecord = hostRecord;
+ UintR i;
+ UintR TpackedListIndex = cpackedListIndex;
+ jamEntry();
+ for (i = 0; i < TpackedListIndex; i++) {
+ Thostptr.i = cpackedList[i];
+ ptrAss(Thostptr, localHostRecord);
+ arrGuard(Thostptr.i - 1, MAX_NODES - 1);
+ UintR TnoOfPackedWordsLqh = Thostptr.p->noOfPackedWordsLqh;
+ UintR TnoOfWordsTCKEYCONF = Thostptr.p->noOfWordsTCKEYCONF;
+ UintR TnoOfWordsTCINDXCONF = Thostptr.p->noOfWordsTCINDXCONF;
+ jam();
+ if (TnoOfPackedWordsLqh > 0) {
+ jam();
+ sendPackedSignalLqh(signal, Thostptr.p);
+ }//if
+ if (TnoOfWordsTCKEYCONF > 0) {
+ jam();
+ sendPackedTCKEYCONF(signal, Thostptr.p, (Uint32)Thostptr.i);
+ }//if
+ if (TnoOfWordsTCINDXCONF > 0) {
+ jam();
+ sendPackedTCINDXCONF(signal, Thostptr.p, (Uint32)Thostptr.i);
+ }//if
+ Thostptr.p->inPackedList = false;
+ }//for
+ cpackedListIndex = 0;
+ return;
+}//Dbtc::execSEND_PACKED()
+
+void
+Dbtc::updatePackedList(Signal* signal, HostRecord* ahostptr, Uint16 ahostIndex)
+{
+ if (ahostptr->inPackedList == false) {
+ UintR TpackedListIndex = cpackedListIndex;
+ jam();
+ ahostptr->inPackedList = true;
+ cpackedList[TpackedListIndex] = ahostIndex;
+ cpackedListIndex = TpackedListIndex + 1;
+ }//if
+}//Dbtc::updatePackedList()
+
+void Dbtc::sendPackedSignalLqh(Signal* signal, HostRecord * ahostptr)
+{
+ UintR Tj;
+ UintR TnoOfWords = ahostptr->noOfPackedWordsLqh;
+ for (Tj = 0; Tj < TnoOfWords; Tj += 4) {
+ UintR sig0 = ahostptr->packedWordsLqh[Tj + 0];
+ UintR sig1 = ahostptr->packedWordsLqh[Tj + 1];
+ UintR sig2 = ahostptr->packedWordsLqh[Tj + 2];
+ UintR sig3 = ahostptr->packedWordsLqh[Tj + 3];
+ signal->theData[Tj + 0] = sig0;
+ signal->theData[Tj + 1] = sig1;
+ signal->theData[Tj + 2] = sig2;
+ signal->theData[Tj + 3] = sig3;
+ }//for
+ ahostptr->noOfPackedWordsLqh = 0;
+ sendSignal(ahostptr->hostLqhBlockRef,
+ GSN_PACKED_SIGNAL,
+ signal,
+ TnoOfWords,
+ JBB);
+}//Dbtc::sendPackedSignalLqh()
+
+void Dbtc::sendPackedTCKEYCONF(Signal* signal,
+ HostRecord * ahostptr,
+ UintR hostId)
+{
+ UintR Tj;
+ UintR TnoOfWords = ahostptr->noOfWordsTCKEYCONF;
+ BlockReference TBref = numberToRef(API_PACKED, hostId);
+ for (Tj = 0; Tj < ahostptr->noOfWordsTCKEYCONF; Tj += 4) {
+ UintR sig0 = ahostptr->packedWordsTCKEYCONF[Tj + 0];
+ UintR sig1 = ahostptr->packedWordsTCKEYCONF[Tj + 1];
+ UintR sig2 = ahostptr->packedWordsTCKEYCONF[Tj + 2];
+ UintR sig3 = ahostptr->packedWordsTCKEYCONF[Tj + 3];
+ signal->theData[Tj + 0] = sig0;
+ signal->theData[Tj + 1] = sig1;
+ signal->theData[Tj + 2] = sig2;
+ signal->theData[Tj + 3] = sig3;
+ }//for
+ ahostptr->noOfWordsTCKEYCONF = 0;
+ sendSignal(TBref, GSN_TCKEYCONF, signal, TnoOfWords, JBB);
+}//Dbtc::sendPackedTCKEYCONF()
+
+void Dbtc::sendPackedTCINDXCONF(Signal* signal,
+ HostRecord * ahostptr,
+ UintR hostId)
+{
+ UintR Tj;
+ UintR TnoOfWords = ahostptr->noOfWordsTCINDXCONF;
+ BlockReference TBref = numberToRef(API_PACKED, hostId);
+ for (Tj = 0; Tj < ahostptr->noOfWordsTCINDXCONF; Tj += 4) {
+ UintR sig0 = ahostptr->packedWordsTCINDXCONF[Tj + 0];
+ UintR sig1 = ahostptr->packedWordsTCINDXCONF[Tj + 1];
+ UintR sig2 = ahostptr->packedWordsTCINDXCONF[Tj + 2];
+ UintR sig3 = ahostptr->packedWordsTCINDXCONF[Tj + 3];
+ signal->theData[Tj + 0] = sig0;
+ signal->theData[Tj + 1] = sig1;
+ signal->theData[Tj + 2] = sig2;
+ signal->theData[Tj + 3] = sig3;
+ }//for
+ ahostptr->noOfWordsTCINDXCONF = 0;
+ sendSignal(TBref, GSN_TCINDXCONF, signal, TnoOfWords, JBB);
+}//Dbtc::sendPackedTCINDXCONF()
+
+/*
+4.3.11 DIVERIFY
+---------------
+*/
+/*****************************************************************************/
+/* D I V E R I F Y */
+/* */
+/*****************************************************************************/
+void Dbtc::diverify010Lab(Signal* signal)
+{
+ UintR TfirstfreeApiConnectCopy = cfirstfreeApiConnectCopy;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ signal->theData[0] = apiConnectptr.i;
+ if (ERROR_INSERTED(8022)) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ if (TfirstfreeApiConnectCopy != RNIL) {
+ seizeApiConnectCopy(signal);
+ regApiPtr->apiConnectstate = CS_PREPARE_TO_COMMIT;
+ /*-----------------------------------------------------------------------
+ * WE COME HERE ONLY IF THE TRANSACTION IS PREPARED ON ALL TC CONNECTIONS.
+ * THUS WE CAN START THE COMMIT PHASE BY SENDING DIVERIFY ON ALL TC
+ * CONNECTIONS AND THEN WHEN ALL DIVERIFYCONF HAVE BEEN RECEIVED THE
+ * COMMIT MESSAGE CAN BE SENT TO ALL INVOLVED PARTS.
+ *-----------------------------------------------------------------------*/
+ EXECUTE_DIRECT(DBDIH, GSN_DIVERIFYREQ, signal, 1);
+ if (signal->theData[2] == 0) {
+ execDIVERIFYCONF(signal);
+ }
+ return;
+ } else {
+ /*-----------------------------------------------------------------------
+ * There were no free copy connections available. We must abort the
+ * transaction since otherwise we will have a problem with the report
+ * to the application.
+ * This should more or less not happen but if it happens we do not want to
+ * crash and we do not want to create code to handle it properly since
+ * it is difficult to test it and will be complex to handle a problem
+ * more or less not occurring.
+ *-----------------------------------------------------------------------*/
+ terrorCode = ZSEIZE_API_COPY_ERROR;
+ abortErrorLab(signal);
+ return;
+ }//if
+}//Dbtc::diverify010Lab()
+
+/* ------------------------------------------------------------------------- */
+/* ------- SEIZE_API_CONNECT ------- */
+/* SEIZE CONNECT RECORD FOR A REQUEST */
+/* ------------------------------------------------------------------------- */
+void Dbtc::seizeApiConnectCopy(Signal* signal)
+{
+ ApiConnectRecordPtr locApiConnectptr;
+
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+ UintR TapiConnectFilesize = capiConnectFilesize;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+
+ locApiConnectptr.i = cfirstfreeApiConnectCopy;
+ ptrCheckGuard(locApiConnectptr, TapiConnectFilesize, localApiConnectRecord);
+ cfirstfreeApiConnectCopy = locApiConnectptr.p->nextApiConnect;
+ locApiConnectptr.p->nextApiConnect = RNIL;
+ regApiPtr->apiCopyRecord = locApiConnectptr.i;
+ regApiPtr->triggerPending = false;
+ regApiPtr->isIndexOp = false;
+}//Dbtc::seizeApiConnectCopy()
+
+void Dbtc::execDIVERIFYCONF(Signal* signal)
+{
+ UintR TapiConnectptrIndex = signal->theData[0];
+ UintR TapiConnectFilesize = capiConnectFilesize;
+ UintR Tgci = signal->theData[1];
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+
+ jamEntry();
+ if (ERROR_INSERTED(8017)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ if (TapiConnectptrIndex >= TapiConnectFilesize) {
+ TCKEY_abort(signal, 31);
+ return;
+ }//if
+ ApiConnectRecord * const regApiPtr =
+ &localApiConnectRecord[TapiConnectptrIndex];
+ ConnectionState TapiConnectstate = regApiPtr->apiConnectstate;
+ UintR TApifailureNr = regApiPtr->failureNr;
+ UintR Tfailure_nr = cfailure_nr;
+ apiConnectptr.i = TapiConnectptrIndex;
+ apiConnectptr.p = regApiPtr;
+ if (TapiConnectstate != CS_PREPARE_TO_COMMIT) {
+ TCKEY_abort(signal, 32);
+ return;
+ }//if
+ /*--------------------------------------------------------------------------
+ * THIS IS THE COMMIT POINT. IF WE ARRIVE HERE THE TRANSACTION IS COMMITTED
+ * UNLESS EVERYTHING CRASHES BEFORE WE HAVE BEEN ABLE TO REPORT THE COMMIT
+ * DECISION. THERE IS NO TURNING BACK FROM THIS DECISION FROM HERE ON.
+ * WE WILL INSERT THE TRANSACTION INTO ITS PROPER QUEUE OF
+ * TRANSACTIONS FOR ITS GLOBAL CHECKPOINT.
+ *-------------------------------------------------------------------------*/
+ if (TApifailureNr != Tfailure_nr) {
+ DIVER_node_fail_handling(signal, Tgci);
+ return;
+ }//if
+ commitGciHandling(signal, Tgci);
+
+ /**************************************************************************
+ * C O M M I T
+ * THE TRANSACTION HAVE NOW BEEN VERIFIED AND NOW THE COMMIT PHASE CAN START
+ **************************************************************************/
+
+ UintR TtcConnectptrIndex = regApiPtr->firstTcConnect;
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+
+ regApiPtr->counter = regApiPtr->lqhkeyconfrec;
+ regApiPtr->apiConnectstate = CS_COMMITTING;
+ if (TtcConnectptrIndex >= TtcConnectFilesize) {
+ TCKEY_abort(signal, 33);
+ return;
+ }//if
+ TcConnectRecord* const regTcPtr = &localTcConnectRecord[TtcConnectptrIndex];
+ tcConnectptr.i = TtcConnectptrIndex;
+ tcConnectptr.p = regTcPtr;
+ commit020Lab(signal);
+}//Dbtc::execDIVERIFYCONF()
+
+/*--------------------------------------------------------------------------*/
+/* COMMIT_GCI_HANDLING */
+/* SET UP GLOBAL CHECKPOINT DATA STRUCTURE AT THE COMMIT POINT. */
+/*--------------------------------------------------------------------------*/
+void Dbtc::commitGciHandling(Signal* signal, UintR Tgci)
+{
+ GcpRecordPtr localGcpPointer;
+
+ UintR TgcpFilesize = cgcpFilesize;
+ UintR Tfirstgcp = cfirstgcp;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ GcpRecord *localGcpRecord = gcpRecord;
+
+ regApiPtr->globalcheckpointid = Tgci;
+ if (Tfirstgcp != RNIL) {
+ /* IF THIS GLOBAL CHECKPOINT ALREADY EXISTS */
+ localGcpPointer.i = Tfirstgcp;
+ ptrCheckGuard(localGcpPointer, TgcpFilesize, localGcpRecord);
+ do {
+ if (regApiPtr->globalcheckpointid == localGcpPointer.p->gcpId) {
+ jam();
+ gcpPtr.i = localGcpPointer.i;
+ gcpPtr.p = localGcpPointer.p;
+ linkApiToGcp(signal);
+ return;
+ } else {
+ localGcpPointer.i = localGcpPointer.p->nextGcp;
+ jam();
+ if (localGcpPointer.i != RNIL) {
+ jam();
+ ptrCheckGuard(localGcpPointer, TgcpFilesize, localGcpRecord);
+ continue;
+ }//if
+ }//if
+ seizeGcp(signal);
+ linkApiToGcp(signal);
+ return;
+ } while (1);
+ } else {
+ jam();
+ seizeGcp(signal);
+ linkApiToGcp(signal);
+ }//if
+}//Dbtc::commitGciHandling()
+
+/* --------------------------------------------------------------------------*/
+/* -LINK AN API CONNECT RECORD IN STATE PREPARED INTO THE LIST WITH GLOBAL - */
+/* CHECKPOINTS. WHEN THE TRANSACTION I COMPLETED THE API CONNECT RECORD IS */
+/* LINKED OUT OF THE LIST. */
+/*---------------------------------------------------------------------------*/
+void Dbtc::linkApiToGcp(Signal* signal)
+{
+ ApiConnectRecordPtr localApiConnectptr;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ GcpRecord * const regGcpPtr = gcpPtr.p;
+ UintR TapiConnectptrIndex = apiConnectptr.i;
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+
+ regApiPtr->nextGcpConnect = RNIL;
+ if (regGcpPtr->firstApiConnect == RNIL) {
+ regGcpPtr->firstApiConnect = TapiConnectptrIndex;
+ jam();
+ } else {
+ UintR TapiConnectFilesize = capiConnectFilesize;
+ localApiConnectptr.i = regGcpPtr->lastApiConnect;
+ jam();
+ ptrCheckGuard(localApiConnectptr,
+ TapiConnectFilesize, localApiConnectRecord);
+ localApiConnectptr.p->nextGcpConnect = TapiConnectptrIndex;
+ }//if
+ UintR TlastApiConnect = regGcpPtr->lastApiConnect;
+ regApiPtr->gcpPointer = gcpPtr.i;
+ regApiPtr->prevGcpConnect = TlastApiConnect;
+ regGcpPtr->lastApiConnect = TapiConnectptrIndex;
+}//Dbtc::linkApiToGcp()
+
+void Dbtc::seizeGcp(Signal* signal)
+{
+ GcpRecordPtr tmpGcpPointer;
+ GcpRecordPtr localGcpPointer;
+
+ UintR Tfirstgcp = cfirstgcp;
+ UintR Tglobalcheckpointid = apiConnectptr.p->globalcheckpointid;
+ UintR TgcpFilesize = cgcpFilesize;
+ GcpRecord *localGcpRecord = gcpRecord;
+
+ localGcpPointer.i = cfirstfreeGcp;
+ ptrCheckGuard(localGcpPointer, TgcpFilesize, localGcpRecord);
+ UintR TfirstfreeGcp = localGcpPointer.p->nextGcp;
+ localGcpPointer.p->gcpId = Tglobalcheckpointid;
+ localGcpPointer.p->nextGcp = RNIL;
+ localGcpPointer.p->firstApiConnect = RNIL;
+ localGcpPointer.p->lastApiConnect = RNIL;
+ localGcpPointer.p->gcpNomoretransRec = ZFALSE;
+ cfirstfreeGcp = TfirstfreeGcp;
+
+ if (Tfirstgcp == RNIL) {
+ jam();
+ cfirstgcp = localGcpPointer.i;
+ } else {
+ tmpGcpPointer.i = clastgcp;
+ jam();
+ ptrCheckGuard(tmpGcpPointer, TgcpFilesize, localGcpRecord);
+ tmpGcpPointer.p->nextGcp = localGcpPointer.i;
+ }//if
+ clastgcp = localGcpPointer.i;
+ gcpPtr = localGcpPointer;
+}//Dbtc::seizeGcp()
+
+/*---------------------------------------------------------------------------*/
+// Send COMMIT messages to all LQH operations involved in the transaction.
+/*---------------------------------------------------------------------------*/
+void Dbtc::commit020Lab(Signal* signal)
+{
+ TcConnectRecordPtr localTcConnectptr;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+
+ localTcConnectptr.p = tcConnectptr.p;
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ UintR Tcount = 0;
+ do {
+ /*-----------------------------------------------------------------------
+ * WE ARE NOW READY TO RELEASE ALL OPERATIONS ON THE LQH
+ *-----------------------------------------------------------------------*/
+ /* *********< */
+ /* COMMIT < */
+ /* *********< */
+ localTcConnectptr.i = localTcConnectptr.p->nextTcConnect;
+ localTcConnectptr.p->tcConnectstate = OS_COMMITTING;
+ sendCommitLqh(signal, localTcConnectptr.p);
+
+ if (localTcConnectptr.i != RNIL) {
+ Tcount = Tcount + 1;
+ if (Tcount < 16) {
+ ptrCheckGuard(localTcConnectptr,
+ TtcConnectFilesize, localTcConnectRecord);
+ jam();
+ continue;
+ } else {
+ jam();
+ if (ERROR_INSERTED(8014)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ signal->theData[0] = TcContinueB::ZSEND_COMMIT_LOOP;
+ signal->theData[1] = apiConnectptr.i;
+ signal->theData[2] = localTcConnectptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ }//if
+ } else {
+ jam();
+ regApiPtr->apiConnectstate = CS_COMMIT_SENT;
+ return;
+ }//if
+ } while (1);
+}//Dbtc::commit020Lab()
+
+void Dbtc::sendCommitLqh(Signal* signal,
+ TcConnectRecord * const regTcPtr)
+{
+ HostRecordPtr Thostptr;
+ UintR ThostFilesize = chostFilesize;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ Thostptr.i = regTcPtr->lastLqhNodeId;
+ ptrCheckGuard(Thostptr, ThostFilesize, hostRecord);
+ if (Thostptr.p->noOfPackedWordsLqh > 21) {
+ jam();
+ sendPackedSignalLqh(signal, Thostptr.p);
+ } else {
+ jam();
+ updatePackedList(signal, Thostptr.p, Thostptr.i);
+ }//if
+ UintR Tindex = Thostptr.p->noOfPackedWordsLqh;
+ UintR* TDataPtr = &Thostptr.p->packedWordsLqh[Tindex];
+ UintR Tdata1 = regTcPtr->lastLqhCon;
+ UintR Tdata2 = regApiPtr->globalcheckpointid;
+ UintR Tdata3 = regApiPtr->transid[0];
+ UintR Tdata4 = regApiPtr->transid[1];
+
+ TDataPtr[0] = Tdata1 | (ZCOMMIT << 28);
+ TDataPtr[1] = Tdata2;
+ TDataPtr[2] = Tdata3;
+ TDataPtr[3] = Tdata4;
+ Thostptr.p->noOfPackedWordsLqh = Tindex + 4;
+}//Dbtc::sendCommitLqh()
+
+void
+Dbtc::DIVER_node_fail_handling(Signal* signal, UintR Tgci)
+{
+ /*------------------------------------------------------------------------
+ * AT LEAST ONE NODE HAS FAILED DURING THE TRANSACTION. WE NEED TO CHECK IF
+ * THIS IS SO SERIOUS THAT WE NEED TO ABORT THE TRANSACTION. IN BOTH THE
+ * ABORT AND THE COMMIT CASES WE NEED TO SET-UP THE DATA FOR THE
+ * ABORT/COMMIT/COMPLETE HANDLING AS ALSO USED BY TAKE OVER FUNCTIONALITY.
+ *------------------------------------------------------------------------*/
+ tabortInd = ZFALSE;
+ setupFailData(signal);
+ if (tabortInd == ZFALSE) {
+ jam();
+ commitGciHandling(signal, Tgci);
+ toCommitHandlingLab(signal);
+ } else {
+ jam();
+ apiConnectptr.p->returnsignal = RS_TCROLLBACKREP;
+ apiConnectptr.p->returncode = ZNODEFAIL_BEFORE_COMMIT;
+ toAbortHandlingLab(signal);
+ }//if
+ return;
+}//Dbtc::DIVER_node_fail_handling()
+
+
+/* ------------------------------------------------------------------------- */
+/* ------- ENTER COMMITTED ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dbtc::execCOMMITTED(Signal* signal)
+{
+ TcConnectRecordPtr localTcConnectptr;
+ ApiConnectRecordPtr localApiConnectptr;
+
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+ UintR TapiConnectFilesize = capiConnectFilesize;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+
+#ifdef ERROR_INSERT
+ if (ERROR_INSERTED(8018)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ if (ERROR_INSERTED(8030)) {
+ systemErrorLab(signal);
+ }//if
+ if (ERROR_INSERTED(8025)) {
+ SET_ERROR_INSERT_VALUE(8026);
+ return;
+ }//if
+ if (ERROR_INSERTED(8041)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_COMMITTED, signal, 2000, 3);
+ return;
+ }//if
+ if (ERROR_INSERTED(8042)) {
+ SET_ERROR_INSERT_VALUE(8046);
+ sendSignalWithDelay(cownref, GSN_COMMITTED, signal, 2000, 4);
+ return;
+ }//if
+#endif
+ localTcConnectptr.i = signal->theData[0];
+ jamEntry();
+ ptrCheckGuard(localTcConnectptr, TtcConnectFilesize, localTcConnectRecord);
+ localApiConnectptr.i = localTcConnectptr.p->apiConnect;
+ if (localTcConnectptr.p->tcConnectstate != OS_COMMITTING) {
+ warningReport(signal, 4);
+ return;
+ }//if
+ ptrCheckGuard(localApiConnectptr, TapiConnectFilesize,
+ localApiConnectRecord);
+ UintR Tcounter = localApiConnectptr.p->counter - 1;
+ ConnectionState TapiConnectstate = localApiConnectptr.p->apiConnectstate;
+ UintR Tdata1 = localApiConnectptr.p->transid[0] - signal->theData[1];
+ UintR Tdata2 = localApiConnectptr.p->transid[1] - signal->theData[2];
+ Tdata1 = Tdata1 | Tdata2;
+ bool TcheckCondition =
+ (TapiConnectstate != CS_COMMIT_SENT) || (Tcounter != 0);
+
+ setApiConTimer(localApiConnectptr.i, ctcTimer, __LINE__);
+ localApiConnectptr.p->counter = Tcounter;
+ localTcConnectptr.p->tcConnectstate = OS_COMMITTED;
+ if (Tdata1 != 0) {
+ warningReport(signal, 5);
+ return;
+ }//if
+ if (TcheckCondition) {
+ jam();
+ /*-------------------------------------------------------*/
+ // We have not sent all COMMIT requests yet. We could be
+ // in the state that all sent are COMMITTED but we are
+ // still waiting for a CONTINUEB to send the rest of the
+ // COMMIT requests.
+ /*-------------------------------------------------------*/
+ return;
+ }//if
+ if (ERROR_INSERTED(8020)) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ /*-------------------------------------------------------*/
+ /* THE ENTIRE TRANSACTION IS NOW COMMITED */
+ /* NOW WE NEED TO SEND THE RESPONSE TO THE APPLICATION. */
+ /* THE APPLICATION CAN THEN REUSE THE API CONNECTION AND */
+ /* THEREFORE WE NEED TO MOVE THE API CONNECTION TO A */
+ /* NEW API CONNECT RECORD. */
+ /*-------------------------------------------------------*/
+
+ apiConnectptr = localApiConnectptr;
+ sendApiCommit(signal);
+
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ localTcConnectptr.i = regApiPtr->firstTcConnect;
+ UintR Tlqhkeyconfrec = regApiPtr->lqhkeyconfrec;
+ ptrCheckGuard(localTcConnectptr, TtcConnectFilesize, localTcConnectRecord);
+ regApiPtr->counter = Tlqhkeyconfrec;
+
+ tcConnectptr = localTcConnectptr;
+ complete010Lab(signal);
+ return;
+
+}//Dbtc::execCOMMITTED()
+
+/*-------------------------------------------------------*/
+/* SEND_API_COMMIT */
+/* SEND COMMIT DECISION TO THE API. */
+/*-------------------------------------------------------*/
+void Dbtc::sendApiCommit(Signal* signal)
+{
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+
+ if (regApiPtr->returnsignal == RS_TCKEYCONF) {
+ sendtckeyconf(signal, 1);
+ } else if (regApiPtr->returnsignal == RS_TC_COMMITCONF) {
+ jam();
+ TcCommitConf * const commitConf = (TcCommitConf *)&signal->theData[0];
+ if(regApiPtr->commitAckMarker == RNIL){
+ jam();
+ commitConf->apiConnectPtr = regApiPtr->ndbapiConnect;
+ } else {
+ jam();
+ commitConf->apiConnectPtr = regApiPtr->ndbapiConnect | 1;
+ }
+ commitConf->transId1 = regApiPtr->transid[0];
+ commitConf->transId2 = regApiPtr->transid[1];
+
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TC_COMMITCONF, signal, 3, JBB);
+ } else if (regApiPtr->returnsignal == RS_NO_RETURN) {
+ jam();
+ } else {
+ TCKEY_abort(signal, 37);
+ return;
+ }//if
+ UintR TapiConnectFilesize = capiConnectFilesize;
+ UintR TcommitCount = ccommitCount;
+ UintR TapiIndex = apiConnectptr.i;
+ UintR TnewApiIndex = regApiPtr->apiCopyRecord;
+ UintR TapiFailState = regApiPtr->apiFailState;
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+
+ tmpApiConnectptr.p = apiConnectptr.p;
+ tmpApiConnectptr.i = TapiIndex;
+ ccommitCount = TcommitCount + 1;
+ apiConnectptr.i = TnewApiIndex;
+ ptrCheckGuard(apiConnectptr, TapiConnectFilesize, localApiConnectRecord);
+ copyApi(signal);
+ if (TapiFailState != ZTRUE) {
+ return;
+ } else {
+ jam();
+ handleApiFailState(signal, tmpApiConnectptr.i);
+ return;
+ }//if
+}//Dbtc::sendApiCommit()
+
+/* ========================================================================= */
+/* ======= COPY_API ======= */
+/* COPY API RECORD ALSO RESET THE OLD API RECORD SO THAT IT */
+/* IS PREPARED TO RECEIVE A NEW TRANSACTION. */
+/*===========================================================================*/
+void Dbtc::copyApi(Signal* signal)
+{
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ ApiConnectRecord * const regTmpApiPtr = tmpApiConnectptr.p;
+
+ UintR TndbapiConnect = regTmpApiPtr->ndbapiConnect;
+ UintR TfirstTcConnect = regTmpApiPtr->firstTcConnect;
+ UintR Ttransid1 = regTmpApiPtr->transid[0];
+ UintR Ttransid2 = regTmpApiPtr->transid[1];
+ UintR Tlqhkeyconfrec = regTmpApiPtr->lqhkeyconfrec;
+ UintR TgcpPointer = regTmpApiPtr->gcpPointer;
+ UintR TgcpFilesize = cgcpFilesize;
+ UintR TcommitAckMarker = regTmpApiPtr->commitAckMarker;
+ GcpRecord *localGcpRecord = gcpRecord;
+
+ regApiPtr->ndbapiBlockref = regTmpApiPtr->ndbapiBlockref;
+ regApiPtr->ndbapiConnect = TndbapiConnect;
+ regApiPtr->firstTcConnect = TfirstTcConnect;
+ regApiPtr->apiConnectstate = CS_COMPLETING;
+ regApiPtr->transid[0] = Ttransid1;
+ regApiPtr->transid[1] = Ttransid2;
+ regApiPtr->lqhkeyconfrec = Tlqhkeyconfrec;
+ regApiPtr->commitAckMarker = TcommitAckMarker;
+
+ gcpPtr.i = TgcpPointer;
+ ptrCheckGuard(gcpPtr, TgcpFilesize, localGcpRecord);
+ unlinkApiConnect(signal);
+ linkApiToGcp(signal);
+ setApiConTimer(tmpApiConnectptr.i, 0, __LINE__);
+ regTmpApiPtr->apiConnectstate = CS_CONNECTED;
+ regTmpApiPtr->commitAckMarker = RNIL;
+}//Dbtc::copyApi()
+
+void Dbtc::unlinkApiConnect(Signal* signal)
+{
+ ApiConnectRecordPtr localApiConnectptr;
+ ApiConnectRecord * const regTmpApiPtr = tmpApiConnectptr.p;
+ UintR TapiConnectFilesize = capiConnectFilesize;
+ UintR TprevGcpConnect = regTmpApiPtr->prevGcpConnect;
+ UintR TnextGcpConnect = regTmpApiPtr->nextGcpConnect;
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+
+ if (TprevGcpConnect == RNIL) {
+ gcpPtr.p->firstApiConnect = TnextGcpConnect;
+ jam();
+ } else {
+ localApiConnectptr.i = TprevGcpConnect;
+ jam();
+ ptrCheckGuard(localApiConnectptr,
+ TapiConnectFilesize, localApiConnectRecord);
+ localApiConnectptr.p->nextGcpConnect = TnextGcpConnect;
+ }//if
+ if (TnextGcpConnect == RNIL) {
+ gcpPtr.p->lastApiConnect = TprevGcpConnect;
+ jam();
+ } else {
+ localApiConnectptr.i = TnextGcpConnect;
+ jam();
+ ptrCheckGuard(localApiConnectptr,
+ TapiConnectFilesize, localApiConnectRecord);
+ localApiConnectptr.p->prevGcpConnect = TprevGcpConnect;
+ }//if
+}//Dbtc::unlinkApiConnect()
+
+void Dbtc::complete010Lab(Signal* signal)
+{
+ TcConnectRecordPtr localTcConnectptr;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+
+ localTcConnectptr.p = tcConnectptr.p;
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ UintR TapiConnectptrIndex = apiConnectptr.i;
+ UintR Tcount = 0;
+ do {
+ localTcConnectptr.p->apiConnect = TapiConnectptrIndex;
+ localTcConnectptr.p->tcConnectstate = OS_COMPLETING;
+
+ /* ************ */
+ /* COMPLETE < */
+ /* ************ */
+ const Uint32 nextTcConnect = localTcConnectptr.p->nextTcConnect;
+ sendCompleteLqh(signal, localTcConnectptr.p);
+ localTcConnectptr.i = nextTcConnect;
+ if (localTcConnectptr.i != RNIL) {
+ Tcount++;
+ if (Tcount < 16) {
+ ptrCheckGuard(localTcConnectptr,
+ TtcConnectFilesize, localTcConnectRecord);
+ jam();
+ continue;
+ } else {
+ jam();
+ if (ERROR_INSERTED(8013)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ signal->theData[0] = TcContinueB::ZSEND_COMPLETE_LOOP;
+ signal->theData[1] = apiConnectptr.i;
+ signal->theData[2] = localTcConnectptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ }//if
+ } else {
+ jam();
+ regApiPtr->apiConnectstate = CS_COMPLETE_SENT;
+ return;
+ }//if
+ } while (1);
+}//Dbtc::complete010Lab()
+
+void Dbtc::sendCompleteLqh(Signal* signal,
+ TcConnectRecord * const regTcPtr)
+{
+ HostRecordPtr Thostptr;
+ UintR ThostFilesize = chostFilesize;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ Thostptr.i = regTcPtr->lastLqhNodeId;
+ ptrCheckGuard(Thostptr, ThostFilesize, hostRecord);
+ if (Thostptr.p->noOfPackedWordsLqh > 22) {
+ jam();
+ sendPackedSignalLqh(signal, Thostptr.p);
+ } else {
+ jam();
+ updatePackedList(signal, Thostptr.p, Thostptr.i);
+ }//if
+
+ UintR Tindex = Thostptr.p->noOfPackedWordsLqh;
+ UintR* TDataPtr = &Thostptr.p->packedWordsLqh[Tindex];
+ UintR Tdata1 = regTcPtr->lastLqhCon | (ZCOMPLETE << 28);
+ UintR Tdata2 = regApiPtr->transid[0];
+ UintR Tdata3 = regApiPtr->transid[1];
+
+ TDataPtr[0] = Tdata1;
+ TDataPtr[1] = Tdata2;
+ TDataPtr[2] = Tdata3;
+ Thostptr.p->noOfPackedWordsLqh = Tindex + 3;
+}//Dbtc::sendCompleteLqh()
+
+void
+Dbtc::execTC_COMMIT_ACK(Signal* signal){
+ jamEntry();
+
+ CommitAckMarker key;
+ key.transid1 = signal->theData[0];
+ key.transid2 = signal->theData[1];
+
+ CommitAckMarkerPtr removedMarker;
+ m_commitAckMarkerHash.release(removedMarker, key);
+ if (removedMarker.i == RNIL) {
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }//if
+ sendRemoveMarkers(signal, removedMarker.p);
+}
+
+void
+Dbtc::sendRemoveMarkers(Signal* signal, const CommitAckMarker * marker){
+ jam();
+ const Uint32 noOfLqhs = marker->noOfLqhs;
+ const Uint32 transId1 = marker->transid1;
+ const Uint32 transId2 = marker->transid2;
+
+ for(Uint32 i = 0; i<noOfLqhs; i++){
+ jam();
+ const NodeId nodeId = marker->lqhNodeId[i];
+ sendRemoveMarker(signal, nodeId, transId1, transId2);
+ }
+}
+
+void
+Dbtc::sendRemoveMarker(Signal* signal,
+ NodeId nodeId,
+ Uint32 transid1,
+ Uint32 transid2){
+ /**
+ * Seize host ptr
+ */
+ HostRecordPtr hostPtr;
+ const UintR ThostFilesize = chostFilesize;
+ hostPtr.i = nodeId;
+ ptrCheckGuard(hostPtr, ThostFilesize, hostRecord);
+
+ if (hostPtr.p->noOfPackedWordsLqh > (25 - 3)){
+ jam();
+ sendPackedSignalLqh(signal, hostPtr.p);
+ } else {
+ jam();
+ updatePackedList(signal, hostPtr.p, hostPtr.i);
+ }//if
+
+ UintR numWord = hostPtr.p->noOfPackedWordsLqh;
+ UintR* dataPtr = &hostPtr.p->packedWordsLqh[numWord];
+
+ dataPtr[0] = (ZREMOVE_MARKER << 28);
+ dataPtr[1] = transid1;
+ dataPtr[2] = transid2;
+ hostPtr.p->noOfPackedWordsLqh = numWord + 3;
+}
+
+void Dbtc::execCOMPLETED(Signal* signal)
+{
+ TcConnectRecordPtr localTcConnectptr;
+ ApiConnectRecordPtr localApiConnectptr;
+
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+ UintR TapiConnectFilesize = capiConnectFilesize;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+
+#ifdef ERROR_INSERT
+ if (ERROR_INSERTED(8031)) {
+ systemErrorLab(signal);
+ }//if
+ if (ERROR_INSERTED(8019)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return;
+ }//if
+ if (ERROR_INSERTED(8027)) {
+ SET_ERROR_INSERT_VALUE(8028);
+ return;
+ }//if
+ if (ERROR_INSERTED(8043)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_COMPLETED, signal, 2000, 3);
+ return;
+ }//if
+ if (ERROR_INSERTED(8044)) {
+ SET_ERROR_INSERT_VALUE(8047);
+ sendSignalWithDelay(cownref, GSN_COMPLETED, signal, 2000, 3);
+ return;
+ }//if
+#endif
+ localTcConnectptr.i = signal->theData[0];
+ jamEntry();
+ ptrCheckGuard(localTcConnectptr, TtcConnectFilesize, localTcConnectRecord);
+ bool Tcond1 = (localTcConnectptr.p->tcConnectstate != OS_COMPLETING);
+ localApiConnectptr.i = localTcConnectptr.p->apiConnect;
+ if (Tcond1) {
+ warningReport(signal, 6);
+ return;
+ }//if
+ ptrCheckGuard(localApiConnectptr, TapiConnectFilesize,
+ localApiConnectRecord);
+ UintR Tdata1 = localApiConnectptr.p->transid[0] - signal->theData[1];
+ UintR Tdata2 = localApiConnectptr.p->transid[1] - signal->theData[2];
+ UintR Tcounter = localApiConnectptr.p->counter - 1;
+ ConnectionState TapiConnectstate = localApiConnectptr.p->apiConnectstate;
+ Tdata1 = Tdata1 | Tdata2;
+ bool TcheckCondition =
+ (TapiConnectstate != CS_COMPLETE_SENT) || (Tcounter != 0);
+ if (Tdata1 != 0) {
+ warningReport(signal, 7);
+ return;
+ }//if
+ setApiConTimer(localApiConnectptr.i, ctcTimer, __LINE__);
+ localApiConnectptr.p->counter = Tcounter;
+ localTcConnectptr.p->tcConnectstate = OS_COMPLETED;
+ localTcConnectptr.p->noOfNodes = 0; // == releaseNodes(signal)
+ if (TcheckCondition) {
+ jam();
+ /*-------------------------------------------------------*/
+ // We have not sent all COMPLETE requests yet. We could be
+ // in the state that all sent are COMPLETED but we are
+ // still waiting for a CONTINUEB to send the rest of the
+ // COMPLETE requests.
+ /*-------------------------------------------------------*/
+ return;
+ }//if
+ if (ERROR_INSERTED(8021)) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ apiConnectptr = localApiConnectptr;
+ releaseTransResources(signal);
+}//Dbtc::execCOMPLETED()
+
+/*---------------------------------------------------------------------------*/
+/* RELEASE_TRANS_RESOURCES */
+/* RELEASE ALL RESOURCES THAT ARE CONNECTED TO THIS TRANSACTION. */
+/*---------------------------------------------------------------------------*/
+void Dbtc::releaseTransResources(Signal* signal)
+{
+ TcConnectRecordPtr localTcConnectptr;
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+
+ localTcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ do {
+ jam();
+ ptrCheckGuard(localTcConnectptr, TtcConnectFilesize, localTcConnectRecord);
+ UintR rtrTcConnectptrIndex = localTcConnectptr.p->nextTcConnect;
+ tcConnectptr.i = localTcConnectptr.i;
+ tcConnectptr.p = localTcConnectptr.p;
+ localTcConnectptr.i = rtrTcConnectptrIndex;
+ releaseTcCon(signal);
+ } while (localTcConnectptr.i != RNIL);
+ handleGcp(signal);
+ releaseFiredTriggerData(&apiConnectptr.p->theFiredTriggers);
+ releaseAllSeizedIndexOperations(apiConnectptr.p);
+ releaseApiConCopy(signal);
+}//Dbtc::releaseTransResources()
+
+/* *********************************************************************>> */
+/* MODULE: HANDLE_GCP */
+/* DESCRIPTION: HANDLES GLOBAL CHECKPOINT HANDLING AT THE COMPLETION */
+/* OF THE COMMIT PHASE AND THE ABORT PHASE. WE MUST ENSURE THAT TC */
+/* SENDS GCP_TCFINISHED WHEN ALL TRANSACTIONS BELONGING TO A CERTAIN */
+/* GLOBAL CHECKPOINT HAVE COMPLETED. */
+/* *********************************************************************>> */
+void Dbtc::handleGcp(Signal* signal)
+{
+ GcpRecord *localGcpRecord = gcpRecord;
+ GcpRecordPtr localGcpPtr;
+ UintR TapiConnectptrIndex = apiConnectptr.i;
+ UintR TgcpFilesize = cgcpFilesize;
+ localGcpPtr.i = apiConnectptr.p->gcpPointer;
+ tmpApiConnectptr.i = TapiConnectptrIndex;
+ tmpApiConnectptr.p = apiConnectptr.p;
+ ptrCheckGuard(localGcpPtr, TgcpFilesize, localGcpRecord);
+ gcpPtr.i = localGcpPtr.i;
+ gcpPtr.p = localGcpPtr.p;
+ unlinkApiConnect(signal);
+ if (localGcpPtr.p->firstApiConnect == RNIL) {
+ if (localGcpPtr.p->gcpNomoretransRec == ZTRUE) {
+ jam();
+ tcheckGcpId = localGcpPtr.p->gcpId;
+ gcpTcfinished(signal);
+ unlinkGcp(signal);
+ }//if
+ }//if
+}//Dbtc::handleGcp()
+
+void Dbtc::releaseApiConCopy(Signal* signal)
+{
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ UintR TfirstfreeApiConnectCopyOld = cfirstfreeApiConnectCopy;
+ cfirstfreeApiConnectCopy = apiConnectptr.i;
+ regApiPtr->nextApiConnect = TfirstfreeApiConnectCopyOld;
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ regApiPtr->apiConnectstate = CS_RESTART;
+}//Dbtc::releaseApiConCopy()
+
+/* ========================================================================= */
+/* ------- RELEASE ALL RECORDS CONNECTED TO A DIRTY WRITE OPERATION ------- */
+/* ========================================================================= */
+void Dbtc::releaseDirtyWrite(Signal* signal)
+{
+ unlinkReadyTcCon(signal);
+ releaseTcCon(signal);
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ if (regApiPtr->apiConnectstate == CS_START_COMMITTING) {
+ if (regApiPtr->firstTcConnect == RNIL) {
+ jam();
+ regApiPtr->apiConnectstate = CS_CONNECTED;
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ sendtckeyconf(signal, 1);
+ }//if
+ }//if
+}//Dbtc::releaseDirtyWrite()
+
+/*****************************************************************************
+ * L Q H K E Y R E F
+ * WHEN LQHKEYREF IS RECEIVED DBTC WILL CHECK IF COMMIT FLAG WAS SENT FROM THE
+ * APPLICATION. IF SO, THE WHOLE TRANSACTION WILL BE ROLLED BACK AND SIGNAL
+ * TCROLLBACKREP WILL BE SENT TO THE API.
+ *
+ * OTHERWISE TC WILL CHECK THE ERRORCODE. IF THE ERRORCODE IS INDICATING THAT
+ * THE "ROW IS NOT FOUND" FOR UPDATE/READ/DELETE OPERATIONS AND "ROW ALREADY
+ * EXISTS" FOR INSERT OPERATIONS, DBTC WILL RELEASE THE OPERATION AND THEN
+ * SEND RETURN SIGNAL TCKEYREF TO THE USER. THE USER THEN HAVE TO SEND
+ * SIGNAL TC_COMMITREQ OR TC_ROLLBACKREQ TO CONCLUDE THE TRANSACTION.
+ * IF ANY TCKEYREQ WITH COMMIT IS RECEIVED AND API_CONNECTSTATE EQUALS
+ * "REC_LQHREFUSE",
+ * THE OPERATION WILL BE TREATED AS AN OPERATION WITHOUT COMMIT. WHEN ANY
+ * OTHER FAULTCODE IS RECEIVED THE WHOLE TRANSACTION MUST BE ROLLED BACK
+ *****************************************************************************/
+void Dbtc::execLQHKEYREF(Signal* signal)
+{
+ const LqhKeyRef * const lqhKeyRef = (LqhKeyRef *)signal->getDataPtr();
+ jamEntry();
+
+ handleFailedOperation(signal, lqhKeyRef, true);
+}
+
+void Dbtc::handleFailedOperation(Signal* signal,
+ const LqhKeyRef * const lqhKeyRef,
+ bool gotLqhKeyRef)
+{
+ UintR compare_transid1, compare_transid2;
+
+ UintR TtcConnectFilesize = ctcConnectFilesize;
+ /*-------------------------------------------------------------------------
+ *
+ * RELEASE NODE BUFFER(S) TO INDICATE THAT THIS OPERATION HAVE NO
+ * TRANSACTION PARTS ACTIVE ANYMORE.
+ * LQHKEYREF HAVE CLEARED ALL PARTS ON ITS PATH BACK TO TC.
+ *-------------------------------------------------------------------------*/
+ if (lqhKeyRef->connectPtr < TtcConnectFilesize) {
+ /*-----------------------------------------------------------------------
+ * WE HAVE TO CHECK THAT THE TRANSACTION IS STILL VALID. FIRST WE CHECK
+ * THAT THE LQH IS STILL CONNECTED TO A TC, IF THIS HOLDS TRUE THEN THE
+ * TC MUST BE CONNECTED TO AN API CONNECT RECORD.
+ * WE MUST ENSURE THAT THE TRANSACTION ID OF THIS API CONNECT
+ * RECORD IS STILL THE SAME AS THE ONE LQHKEYREF REFERS TO.
+ * IF NOT SIMPLY EXIT AND FORGET THE SIGNAL SINCE THE TRANSACTION IS
+ * ALREADY COMPLETED (ABORTED).
+ *-----------------------------------------------------------------------*/
+ tcConnectptr.i = lqhKeyRef->connectPtr;
+ terrorCode = lqhKeyRef->errorCode;
+ ptrAss(tcConnectptr, tcConnectRecord);
+ TcConnectRecord * const regTcPtr = tcConnectptr.p;
+ if (regTcPtr->tcConnectstate == OS_OPERATING) {
+ apiConnectptr.i = regTcPtr->apiConnect;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ compare_transid1 = regApiPtr->transid[0] ^ lqhKeyRef->transId1;
+ compare_transid2 = regApiPtr->transid[1] ^ lqhKeyRef->transId2;
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ warningReport(signal, 25);
+ return;
+ }//if
+
+ if (regTcPtr->triggeringOperation != RNIL) {
+ jam();
+ // This operation was created by a trigger execting operation
+ TcConnectRecordPtr opPtr;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+
+ const Uint32 currentIndexId = regTcPtr->currentIndexId;
+
+ opPtr.i = regTcPtr->triggeringOperation;
+ ptrCheckGuard(opPtr, ctcConnectFilesize, localTcConnectRecord);
+ if (currentIndexId != RNIL)
+ {
+ jam();
+ // The operation executed an index trigger
+ TcIndexData* indexData = NULL;
+ indexData = c_theIndexes.getPtr(currentIndexId);
+ if (regTcPtr->operation == ZDELETE) {
+ if (lqhKeyRef->errorCode == ZNOT_FOUND) {
+ if (indexData->indexState == IS_BUILDING) {
+ jam();
+ /*
+ If an index trigger fail with delete during index
+ build phase it just means that the index build has not
+ yet inserted that tuple
+ Check if operation was a delete and part of trigger execution
+ */
+ } else {
+ jam();
+ ndbassert(false);
+ terrorCode = ZINDEX_CORRUPT_ERROR;
+ abortErrorLab(signal);
+ return;
+ }//if
+ } else {
+ terrorCode = lqhKeyRef->errorCode;
+ abortErrorLab(signal);
+ return;
+ }
+ } else if (regTcPtr->operation == ZINSERT) {
+ if (lqhKeyRef->errorCode == ZALREADYEXIST) {
+ terrorCode = 893; //Constraint violation
+ abortErrorLab(signal);
+ return;
+ } else {
+ terrorCode = lqhKeyRef->errorCode;
+ abortErrorLab(signal);
+ }
+ } else {
+ ndbrequire(false);
+ return;
+ }
+ markOperationAborted(regApiPtr, regTcPtr);
+ if (regApiPtr->apiConnectstate == CS_ABORTING) {
+ jam();
+ return;
+ }
+ unlinkReadyTcCon(signal);
+ releaseTcCon(signal);
+ // Decrease counter as if NOOP
+ regApiPtr->lqhkeyreqrec--;
+ opPtr.p->triggerExecutionCount--;
+ if (opPtr.p->triggerExecutionCount == 0) {
+ jam();
+ /*
+ We have completed current trigger execution
+ continue triggering operation
+ */
+ continueTriggeringOp(signal, opPtr.p);
+ }//if
+ if (!regApiPtr->theFiredTriggers.isEmpty()) {
+ jam();
+ /*
+ There are more triggers
+ Continue with next trigger
+ */
+ executeTriggers(signal, &apiConnectptr);
+ }//if
+ return;
+ } else {
+ /**
+ * Currently the index id is always set for triggering operations
+ since we only support them for unique hash indexes at the moment.
+ */
+ ndbrequire(false);
+ return;
+ }
+ }
+ if (gotLqhKeyRef) {
+ jam();
+ markOperationAborted(regApiPtr, regTcPtr);
+ }//if
+
+ if(regApiPtr->apiConnectstate == CS_ABORTING){
+ /**
+ * We're already aborting' so don't send an "extra" TCKEYREF
+ */
+ jam();
+ return;
+ }
+
+ const Uint32 abortOption = regTcPtr->m_execAbortOption;
+ if (abortOption == TcKeyReq::AbortOnError) {
+ /**
+ * No error is allowed on this operation
+ */
+ TCKEY_abort(signal, 49);
+ return;
+ }//if
+
+ if (regTcPtr->commitAckMarker != RNIL){
+ /**
+ * This was an insert/update/delete/write which failed
+ * that contained the marker
+ * Currently unsupported to place new marker
+ */
+ TCKEY_abort(signal, 49);
+ return;
+ }
+
+ /* *************** */
+ /* TCKEYREF < */
+ /* *************** */
+ TcKeyRef * const tcKeyRef = (TcKeyRef *) signal->getDataPtrSend();
+ tcKeyRef->transId[0] = regApiPtr->transid[0];
+ tcKeyRef->transId[1] = regApiPtr->transid[1];
+ tcKeyRef->errorCode = terrorCode;
+ bool isIndexOp = regTcPtr->isIndexOp;
+ Uint32 indexOp = tcConnectptr.p->indexOp;
+ Uint32 clientData = regTcPtr->clientData;
+ if (gotLqhKeyRef) {
+ unlinkReadyTcCon(signal); /* LINK TC CONNECT RECORD OUT OF */
+ releaseTcCon(signal); /* RELEASE THE TC CONNECT RECORD */
+ }
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ if (isIndexOp) {
+ jam();
+ tcKeyRef->connectPtr = indexOp;
+ EXECUTE_DIRECT(DBTC, GSN_TCKEYREF, signal, TcKeyRef::SignalLength);
+ } else {
+ jam();
+ tcKeyRef->connectPtr = clientData;
+ sendSignal(regApiPtr->ndbapiBlockref,
+ GSN_TCKEYREF, signal, TcKeyRef::SignalLength, JBB);
+ }//if
+
+ /*---------------------------------------------------------------------
+ * SINCE WE ARE NOT ABORTING WE NEED TO UPDATE THE COUNT OF HOW MANY
+ * LQHKEYREQ THAT HAVE RETURNED.
+ * IF NO MORE OUTSTANDING LQHKEYREQ'S THEN WE NEED TO
+ * TCKEYCONF (IF THERE IS ANYTHING TO SEND).
+ *---------------------------------------------------------------------*/
+ if (gotLqhKeyRef) {
+ jam();
+ regApiPtr->lqhkeyreqrec = regApiPtr->lqhkeyreqrec - 1;
+ } else {
+ jam();
+ regApiPtr->lqhkeyconfrec = regApiPtr->lqhkeyconfrec + 1;
+ }//if
+ if (regApiPtr->lqhkeyconfrec == regApiPtr->lqhkeyreqrec) {
+ if ((regApiPtr->lqhkeyconfrec == 0) &&
+ (regApiPtr->apiConnectstate == CS_START_COMMITTING)) {
+
+ if(abortOption == TcKeyReq::IgnoreError){
+ jam();
+ regApiPtr->returnsignal = RS_NO_RETURN;
+ abort010Lab(signal);
+ return;
+ }
+
+ /*----------------------------------------------------------------
+ * Not a single operation was successful.
+ * This we report as an aborted transaction
+ * to avoid performing a commit of zero operations.
+ *----------------------------------------------------------------*/
+ TCKEY_abort(signal, 54);
+ return;
+ }//if
+ if (regApiPtr->apiConnectstate == CS_START_COMMITTING) {
+ jam();
+ diverify010Lab(signal);
+ return;
+ } else if (regApiPtr->tckeyrec > 0) {
+ jam();
+ sendtckeyconf(signal, 0);
+ return;
+ }//if
+ }//if
+ return;
+
+ } else {
+ warningReport(signal, 26);
+ }//if
+ } else {
+ errorReport(signal, 6);
+ }//if
+ return;
+}//Dbtc::execLQHKEYREF()
+
+void Dbtc::clearCommitAckMarker(ApiConnectRecord * const regApiPtr,
+ TcConnectRecord * const regTcPtr)
+{
+ const Uint32 commitAckMarker = regTcPtr->commitAckMarker;
+ if (regApiPtr->commitAckMarker == RNIL)
+ ndbassert(commitAckMarker == RNIL);
+ if (commitAckMarker != RNIL)
+ ndbassert(regApiPtr->commitAckMarker != RNIL);
+ if(commitAckMarker != RNIL){
+ jam();
+ m_commitAckMarkerHash.release(commitAckMarker);
+ regTcPtr->commitAckMarker = RNIL;
+ regApiPtr->commitAckMarker = RNIL;
+ }
+}
+
+void Dbtc::markOperationAborted(ApiConnectRecord * const regApiPtr,
+ TcConnectRecord * const regTcPtr)
+{
+ /*------------------------------------------------------------------------
+ * RELEASE NODES TO INDICATE THAT THE OPERATION IS ALREADY ABORTED IN THE
+ * LQH'S ALSO SET STATE TO ABORTING TO INDICATE THE ABORT IS
+ * ALREADY COMPLETED.
+ *------------------------------------------------------------------------*/
+ regTcPtr->noOfNodes = 0; // == releaseNodes(signal)
+ regTcPtr->tcConnectstate = OS_ABORTING;
+ clearCommitAckMarker(regApiPtr, regTcPtr);
+}
+
+/*--------------------------------------*/
+/* EXIT AND WAIT FOR SIGNAL TCOMMITREQ */
+/* OR TCROLLBACKREQ FROM THE USER TO */
+/* CONTINUE THE TRANSACTION */
+/*--------------------------------------*/
+void Dbtc::execTC_COMMITREQ(Signal* signal)
+{
+ UintR compare_transid1, compare_transid2;
+
+ jamEntry();
+ apiConnectptr.i = signal->theData[0];
+ if (apiConnectptr.i < capiConnectFilesize) {
+ ptrAss(apiConnectptr, apiConnectRecord);
+ compare_transid1 = apiConnectptr.p->transid[0] ^ signal->theData[1];
+ compare_transid2 = apiConnectptr.p->transid[1] ^ signal->theData[2];
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ jam();
+ return;
+ }//if
+
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+
+ const Uint32 apiConnectPtr = regApiPtr->ndbapiConnect;
+ const Uint32 apiBlockRef = regApiPtr->ndbapiBlockref;
+ const Uint32 transId1 = regApiPtr->transid[0];
+ const Uint32 transId2 = regApiPtr->transid[1];
+ Uint32 errorCode = 0;
+
+ switch (regApiPtr->apiConnectstate) {
+ case CS_STARTED:
+ tcConnectptr.i = regApiPtr->firstTcConnect;
+ if (tcConnectptr.i != RNIL) {
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ if (regApiPtr->lqhkeyconfrec == regApiPtr->lqhkeyreqrec) {
+ jam();
+ /*******************************************************************/
+ // The proper case where the application is waiting for commit or
+ // abort order.
+ // Start the commit order.
+ /*******************************************************************/
+ regApiPtr->returnsignal = RS_TC_COMMITCONF;
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ diverify010Lab(signal);
+ return;
+ } else {
+ jam();
+ /*******************************************************************/
+ // The transaction is started but not all operations are completed.
+ // It is not possible to commit the transaction in this state.
+ // We will abort it instead.
+ /*******************************************************************/
+ regApiPtr->returnsignal = RS_NO_RETURN;
+ abort010Lab(signal);
+ errorCode = ZTRANS_STATUS_ERROR;
+ }//if
+ } else {
+ jam();
+ /**
+ * No operations, accept commit
+ */
+ TcCommitConf * const commitConf = (TcCommitConf *)&signal->theData[0];
+ commitConf->apiConnectPtr = apiConnectPtr;
+ commitConf->transId1 = transId1;
+ commitConf->transId2 = transId2;
+
+ sendSignal(apiBlockRef, GSN_TC_COMMITCONF, signal, 3, JBB);
+
+ regApiPtr->returnsignal = RS_NO_RETURN;
+ releaseAbortResources(signal);
+ return;
+ }//if
+ break;
+ case CS_RECEIVING:
+ jam();
+ /***********************************************************************/
+ // A transaction is still receiving data. We cannot commit an unfinished
+ // transaction. We will abort it instead.
+ /***********************************************************************/
+ regApiPtr->returnsignal = RS_NO_RETURN;
+ abort010Lab(signal);
+ errorCode = ZPREPAREINPROGRESS;
+ break;
+
+ case CS_START_COMMITTING:
+ case CS_COMMITTING:
+ case CS_COMMIT_SENT:
+ case CS_COMPLETING:
+ case CS_COMPLETE_SENT:
+ case CS_REC_COMMITTING:
+ case CS_PREPARE_TO_COMMIT:
+ jam();
+ /***********************************************************************/
+ // The transaction is already performing a commit but it is not concluded
+ // yet.
+ /***********************************************************************/
+ errorCode = ZCOMMITINPROGRESS;
+ break;
+ case CS_ABORTING:
+ jam();
+ errorCode = ZABORTINPROGRESS;
+ break;
+ case CS_START_SCAN:
+ jam();
+ /***********************************************************************/
+ // The transaction is a scan. Scans cannot commit
+ /***********************************************************************/
+ errorCode = ZSCANINPROGRESS;
+ break;
+ case CS_PREPARED:
+ jam();
+ return;
+ case CS_START_PREPARING:
+ jam();
+ return;
+ case CS_REC_PREPARING:
+ jam();
+ return;
+ break;
+ default:
+ warningHandlerLab(signal);
+ return;
+ }//switch
+ TcCommitRef * const commitRef = (TcCommitRef*)&signal->theData[0];
+ commitRef->apiConnectPtr = apiConnectPtr;
+ commitRef->transId1 = transId1;
+ commitRef->transId2 = transId2;
+ commitRef->errorCode = errorCode;
+ sendSignal(apiBlockRef, GSN_TC_COMMITREF, signal,
+ TcCommitRef::SignalLength, JBB);
+ return;
+ } else /** apiConnectptr.i < capiConnectFilesize */ {
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }
+}//Dbtc::execTC_COMMITREQ()
+
+void Dbtc::execTCROLLBACKREQ(Signal* signal)
+{
+ UintR compare_transid1, compare_transid2;
+
+ jamEntry();
+ apiConnectptr.i = signal->theData[0];
+ if (apiConnectptr.i >= capiConnectFilesize) {
+ goto TC_ROLL_warning;
+ }//if
+ ptrAss(apiConnectptr, apiConnectRecord);
+ compare_transid1 = apiConnectptr.p->transid[0] ^ signal->theData[1];
+ compare_transid2 = apiConnectptr.p->transid[1] ^ signal->theData[2];
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ jam();
+ return;
+ }//if
+
+ switch (apiConnectptr.p->apiConnectstate) {
+ case CS_STARTED:
+ case CS_RECEIVING:
+ jam();
+ apiConnectptr.p->returnsignal = RS_TCROLLBACKCONF;
+ abort010Lab(signal);
+ return;
+ case CS_CONNECTED:
+ jam();
+ signal->theData[0] = apiConnectptr.p->ndbapiConnect;
+ signal->theData[1] = apiConnectptr.p->transid[0];
+ signal->theData[2] = apiConnectptr.p->transid[1];
+ sendSignal(apiConnectptr.p->ndbapiBlockref, GSN_TCROLLBACKCONF,
+ signal, 3, JBB);
+ break;
+ case CS_START_SCAN:
+ case CS_PREPARE_TO_COMMIT:
+ case CS_COMMITTING:
+ case CS_COMMIT_SENT:
+ case CS_COMPLETING:
+ case CS_COMPLETE_SENT:
+ case CS_WAIT_COMMIT_CONF:
+ case CS_WAIT_COMPLETE_CONF:
+ case CS_RESTART:
+ case CS_DISCONNECTED:
+ case CS_START_COMMITTING:
+ case CS_REC_COMMITTING:
+ jam();
+ /* ***************< */
+ /* TC_ROLLBACKREF < */
+ /* ***************< */
+ signal->theData[0] = apiConnectptr.p->ndbapiConnect;
+ signal->theData[1] = apiConnectptr.p->transid[0];
+ signal->theData[2] = apiConnectptr.p->transid[1];
+ signal->theData[3] = ZROLLBACKNOTALLOWED;
+ sendSignal(apiConnectptr.p->ndbapiBlockref, GSN_TCROLLBACKREF,
+ signal, 4, JBB);
+ break;
+ /* SEND A REFUSAL SIGNAL*/
+ case CS_ABORTING:
+ jam();
+ if (apiConnectptr.p->abortState == AS_IDLE) {
+ jam();
+ signal->theData[0] = apiConnectptr.p->ndbapiConnect;
+ signal->theData[1] = apiConnectptr.p->transid[0];
+ signal->theData[2] = apiConnectptr.p->transid[1];
+ sendSignal(apiConnectptr.p->ndbapiBlockref, GSN_TCROLLBACKCONF,
+ signal, 3, JBB);
+ } else {
+ jam();
+ apiConnectptr.p->returnsignal = RS_TCROLLBACKCONF;
+ }//if
+ break;
+ case CS_WAIT_ABORT_CONF:
+ jam();
+ apiConnectptr.p->returnsignal = RS_TCROLLBACKCONF;
+ break;
+ case CS_START_PREPARING:
+ jam();
+ case CS_PREPARED:
+ jam();
+ case CS_REC_PREPARING:
+ jam();
+ default:
+ goto TC_ROLL_system_error;
+ break;
+ }//switch
+ return;
+
+TC_ROLL_warning:
+ jam();
+ warningHandlerLab(signal);
+ return;
+
+TC_ROLL_system_error:
+ jam();
+ systemErrorLab(signal);
+ return;
+}//Dbtc::execTCROLLBACKREQ()
+
+void Dbtc::execTC_HBREP(Signal* signal)
+{
+ const TcHbRep * const tcHbRep =
+ (TcHbRep *)signal->getDataPtr();
+
+ jamEntry();
+ apiConnectptr.i = tcHbRep->apiConnectPtr;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+
+ if (apiConnectptr.p->transid[0] == tcHbRep->transId1 &&
+ apiConnectptr.p->transid[1] == tcHbRep->transId2){
+
+ if (getApiConTimer(apiConnectptr.i) != 0){
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ } else {
+ DEBUG("TCHBREP received when timer was off apiConnectptr.i="
+ << apiConnectptr.i);
+ }
+ }
+}//Dbtc::execTCHBREP()
+
+/*
+4.3.15 ABORT
+-----------
+*/
+/*****************************************************************************/
+/* A B O R T */
+/* */
+/*****************************************************************************/
+void Dbtc::warningReport(Signal* signal, int place)
+{
+ switch (place) {
+ case 0:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "ABORTED to not active TC record" << endl;
+#endif
+ break;
+ case 1:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "ABORTED to TC record active with new transaction" << endl;
+#endif
+ break;
+ case 2:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "ABORTED to active TC record not expecting ABORTED" << endl;
+#endif
+ break;
+ case 3:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "ABORTED to TC rec active with trans but wrong node" << endl;
+ ndbout << "This is ok when aborting in node failure situations" << endl;
+#endif
+ break;
+ case 4:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMMITTED in wrong state in Dbtc" << endl;
+#endif
+ break;
+ case 5:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMMITTED with wrong transid in Dbtc" << endl;
+#endif
+ break;
+ case 6:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMPLETED in wrong state in Dbtc" << endl;
+#endif
+ break;
+ case 7:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMPLETED with wrong transid in Dbtc" << endl;
+#endif
+ break;
+ case 8:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMMITCONF with tc-rec in wrong state in Dbtc" << endl;
+#endif
+ break;
+ case 9:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMMITCONF with api-rec in wrong state in Dbtc" <<endl;
+#endif
+ break;
+ case 10:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMMITCONF with wrong transid in Dbtc" << endl;
+#endif
+ break;
+ case 11:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMMITCONF from wrong nodeid in Dbtc" << endl;
+#endif
+ break;
+ case 12:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMPLETECONF, tc-rec in wrong state in Dbtc" << endl;
+#endif
+ break;
+ case 13:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMPLETECONF, api-rec in wrong state in Dbtc" << endl;
+#endif
+ break;
+ case 14:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMPLETECONF with wrong transid in Dbtc" << endl;
+#endif
+ break;
+ case 15:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received COMPLETECONF from wrong nodeid in Dbtc" << endl;
+#endif
+ break;
+ case 16:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received ABORTCONF, tc-rec in wrong state in Dbtc" << endl;
+#endif
+ break;
+ case 17:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received ABORTCONF, api-rec in wrong state in Dbtc" << endl;
+#endif
+ break;
+ case 18:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received ABORTCONF with wrong transid in Dbtc" << endl;
+#endif
+ break;
+ case 19:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received ABORTCONF from wrong nodeid in Dbtc" << endl;
+#endif
+ break;
+ case 20:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Time-out waiting for ABORTCONF in Dbtc" << endl;
+#endif
+ break;
+ case 21:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Time-out waiting for COMMITCONF in Dbtc" << endl;
+#endif
+ break;
+ case 22:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Time-out waiting for COMPLETECONF in Dbtc" << endl;
+#endif
+ break;
+ case 23:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received LQHKEYCONF in wrong tc-state in Dbtc" << endl;
+#endif
+ break;
+ case 24:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received LQHKEYREF to wrong transid in Dbtc" << endl;
+#endif
+ break;
+ case 25:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received LQHKEYREF in wrong state in Dbtc" << endl;
+#endif
+ break;
+ case 26:
+ jam();
+#ifdef ABORT_TRACE
+ ndbout << "Received LQHKEYCONF to wrong transid in Dbtc" << endl;
+#endif
+ break;
+ case 27:
+ jam();
+ // printState(signal, 27);
+#ifdef ABORT_TRACE
+ ndbout << "Received LQHKEYCONF in wrong api-state in Dbtc" << endl;
+#endif
+ break;
+ default:
+ jam();
+ break;
+ }//switch
+ return;
+}//Dbtc::warningReport()
+
+void Dbtc::errorReport(Signal* signal, int place)
+{
+ switch (place) {
+ case 0:
+ jam();
+ break;
+ case 1:
+ jam();
+ break;
+ case 2:
+ jam();
+ break;
+ case 3:
+ jam();
+ break;
+ case 4:
+ jam();
+ break;
+ case 5:
+ jam();
+ break;
+ case 6:
+ jam();
+ break;
+ default:
+ jam();
+ break;
+ }//switch
+ systemErrorLab(signal);
+ return;
+}//Dbtc::errorReport()
+
+/* ------------------------------------------------------------------------- */
+/* ------- ENTER ABORTED ------- */
+/* */
+/*-------------------------------------------------------------------------- */
+void Dbtc::execABORTED(Signal* signal)
+{
+ UintR compare_transid1, compare_transid2;
+
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ UintR Tnodeid = signal->theData[3];
+ UintR TlastLqhInd = signal->theData[4];
+
+ if (ERROR_INSERTED(8040)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_ABORTED, signal, 2000, 5);
+ return;
+ }//if
+ /*------------------------------------------------------------------------
+ * ONE PARTICIPANT IN THE TRANSACTION HAS REPORTED THAT IT IS ABORTED.
+ *------------------------------------------------------------------------*/
+ if (tcConnectptr.i >= ctcConnectFilesize) {
+ errorReport(signal, 0);
+ return;
+ }//if
+ /*-------------------------------------------------------------------------
+ * WE HAVE TO CHECK THAT THIS IS NOT AN OLD SIGNAL BELONGING TO A
+ * TRANSACTION ALREADY ABORTED. THIS CAN HAPPEN WHEN TIME-OUT OCCURS
+ * IN TC WAITING FOR ABORTED.
+ *-------------------------------------------------------------------------*/
+ ptrAss(tcConnectptr, tcConnectRecord);
+ if (tcConnectptr.p->tcConnectstate != OS_ABORT_SENT) {
+ warningReport(signal, 2);
+ return;
+ /*-----------------------------------------------------------------------*/
+ // ABORTED reported on an operation not expecting ABORT.
+ /*-----------------------------------------------------------------------*/
+ }//if
+ apiConnectptr.i = tcConnectptr.p->apiConnect;
+ if (apiConnectptr.i >= capiConnectFilesize) {
+ warningReport(signal, 0);
+ return;
+ }//if
+ ptrAss(apiConnectptr, apiConnectRecord);
+ compare_transid1 = apiConnectptr.p->transid[0] ^ signal->theData[1];
+ compare_transid2 = apiConnectptr.p->transid[1] ^ signal->theData[2];
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ warningReport(signal, 1);
+ return;
+ }//if
+ if (ERROR_INSERTED(8024)) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+
+ /**
+ * Release marker
+ */
+ clearCommitAckMarker(apiConnectptr.p, tcConnectptr.p);
+
+ Uint32 i;
+ Uint32 Tfound = 0;
+ for (i = 0; i < tcConnectptr.p->noOfNodes; i++) {
+ jam();
+ if (tcConnectptr.p->tcNodedata[i] == Tnodeid) {
+ /*---------------------------------------------------------------------
+ * We have received ABORTED from one of the participants in this
+ * operation in this aborted transaction.
+ * Record all nodes that have completed abort.
+ * If last indicator is set it means that no more replica has
+ * heard of the operation and are thus also aborted.
+ *---------------------------------------------------------------------*/
+ jam();
+ Tfound = 1;
+ clearTcNodeData(signal, TlastLqhInd, i);
+ }//if
+ }//for
+ if (Tfound == 0) {
+ warningReport(signal, 3);
+ return;
+ }
+ for (i = 0; i < tcConnectptr.p->noOfNodes; i++) {
+ if (tcConnectptr.p->tcNodedata[i] != 0) {
+ /*--------------------------------------------------------------------
+ * There are still outstanding ABORTED's to wait for.
+ *--------------------------------------------------------------------*/
+ jam();
+ return;
+ }//if
+ }//for
+ tcConnectptr.p->noOfNodes = 0;
+ tcConnectptr.p->tcConnectstate = OS_ABORTING;
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ apiConnectptr.p->counter--;
+ if (apiConnectptr.p->counter > 0) {
+ jam();
+ /*----------------------------------------------------------------------
+ * WE ARE STILL WAITING FOR MORE PARTICIPANTS TO SEND ABORTED.
+ *----------------------------------------------------------------------*/
+ return;
+ }//if
+ /*------------------------------------------------------------------------*/
+ /* */
+ /* WE HAVE NOW COMPLETED THE ABORT PROCESS. WE HAVE RECEIVED ABORTED */
+ /* FROM ALL PARTICIPANTS IN THE TRANSACTION. WE CAN NOW RELEASE ALL */
+ /* RESOURCES CONNECTED TO THE TRANSACTION AND SEND THE ABORT RESPONSE */
+ /*------------------------------------------------------------------------*/
+ releaseAbortResources(signal);
+}//Dbtc::execABORTED()
+
+void Dbtc::clearTcNodeData(Signal* signal,
+ UintR TLastLqhIndicator,
+ UintR Tstart)
+{
+ UintR Ti;
+ if (TLastLqhIndicator == ZTRUE) {
+ for (Ti = Tstart ; Ti < tcConnectptr.p->noOfNodes; Ti++) {
+ jam();
+ tcConnectptr.p->tcNodedata[Ti] = 0;
+ }//for
+ } else {
+ jam();
+ tcConnectptr.p->tcNodedata[Tstart] = 0;
+ }//for
+}//clearTcNodeData()
+
+void Dbtc::abortErrorLab(Signal* signal)
+{
+ ptrGuard(apiConnectptr);
+ ApiConnectRecord * transP = apiConnectptr.p;
+ if (transP->apiConnectstate == CS_ABORTING && transP->abortState != AS_IDLE){
+ jam();
+ return;
+ }
+ transP->returnsignal = RS_TCROLLBACKREP;
+ transP->returncode = terrorCode;
+ abort010Lab(signal);
+}//Dbtc::abortErrorLab()
+
+void Dbtc::abort010Lab(Signal* signal)
+{
+ ApiConnectRecord * transP = apiConnectptr.p;
+ if (transP->apiConnectstate == CS_ABORTING && transP->abortState != AS_IDLE){
+ jam();
+ return;
+ }
+ transP->apiConnectstate = CS_ABORTING;
+ /*------------------------------------------------------------------------*/
+ /* AN ABORT DECISION HAS BEEN TAKEN FOR SOME REASON. WE NEED TO ABORT */
+ /* ALL PARTICIPANTS IN THE TRANSACTION. */
+ /*------------------------------------------------------------------------*/
+ transP->abortState = AS_ACTIVE;
+ transP->counter = 0;
+
+ if (transP->firstTcConnect == RNIL) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ /* WE HAVE NO PARTICIPANTS IN THE TRANSACTION. */
+ /*-----------------------------------------------------------------------*/
+ releaseAbortResources(signal);
+ return;
+ }//if
+ tcConnectptr.i = transP->firstTcConnect;
+ abort015Lab(signal);
+}//Dbtc::abort010Lab()
+
+/*--------------------------------------------------------------------------*/
+/* */
+/* WE WILL ABORT ONE NODE PER OPERATION AT A TIME. THIS IS TO KEEP */
+/* ERROR HANDLING OF THIS PROCESS FAIRLY SIMPLE AND TRACTABLE. */
+/* EVEN IF NO NODE OF THIS PARTICULAR NODE NUMBER NEEDS ABORTION WE */
+/* MUST ENSURE THAT ALL NODES ARE CHECKED. THUS A FAULTY NODE DOES */
+/* NOT MEAN THAT ALL NODES IN AN OPERATION IS ABORTED. FOR THIS REASON*/
+/* WE SET THE TCONTINUE_ABORT TO TRUE WHEN A FAULTY NODE IS DETECTED. */
+/*--------------------------------------------------------------------------*/
+void Dbtc::abort015Lab(Signal* signal)
+{
+ Uint32 TloopCount = 0;
+ABORT020:
+ jam();
+ TloopCount++;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ switch (tcConnectptr.p->tcConnectstate) {
+ case OS_WAIT_DIH:
+ case OS_WAIT_KEYINFO:
+ case OS_WAIT_ATTR:
+ jam();
+ /*----------------------------------------------------------------------*/
+ /* WE ARE STILL WAITING FOR MORE KEYINFO/ATTRINFO. WE HAVE NOT CONTACTED*/
+ /* ANY LQH YET AND SO WE CAN SIMPLY SET STATE TO ABORTING. */
+ /*----------------------------------------------------------------------*/
+ tcConnectptr.p->noOfNodes = 0; // == releaseAbort(signal)
+ tcConnectptr.p->tcConnectstate = OS_ABORTING;
+ break;
+ case OS_CONNECTED:
+ jam();
+ /*-----------------------------------------------------------------------
+ * WE ARE STILL IN THE INITIAL PHASE OF THIS OPERATION.
+ * NEED NOT BOTHER ABOUT ANY LQH ABORTS.
+ *-----------------------------------------------------------------------*/
+ tcConnectptr.p->noOfNodes = 0; // == releaseAbort(signal)
+ tcConnectptr.p->tcConnectstate = OS_ABORTING;
+ break;
+ case OS_PREPARED:
+ jam();
+ case OS_OPERATING:
+ jam();
+ /*----------------------------------------------------------------------
+ * WE HAVE SENT LQHKEYREQ AND ARE IN SOME STATE OF EITHER STILL
+ * SENDING THE OPERATION, WAITING FOR REPLIES, WAITING FOR MORE
+ * ATTRINFO OR OPERATION IS PREPARED. WE NEED TO ABORT ALL LQH'S.
+ *----------------------------------------------------------------------*/
+ releaseAndAbort(signal);
+ tcConnectptr.p->tcConnectstate = OS_ABORT_SENT;
+ TloopCount += 127;
+ break;
+ case OS_ABORTING:
+ jam();
+ break;
+ case OS_ABORT_SENT:
+ jam();
+ DEBUG("ABORT_SENT state in abort015Lab(), not expected");
+ systemErrorLab(signal);
+ return;
+ default:
+ jam();
+ DEBUG("tcConnectstate = " << tcConnectptr.p->tcConnectstate);
+ systemErrorLab(signal);
+ return;
+ }//switch
+
+ if (tcConnectptr.p->nextTcConnect != RNIL) {
+ jam();
+ tcConnectptr.i = tcConnectptr.p->nextTcConnect;
+ if (TloopCount < 1024) {
+ goto ABORT020;
+ } else {
+ jam();
+ /*---------------------------------------------------------------------
+ * Reset timer to avoid time-out in real-time break.
+ * Increase counter to ensure that we don't think that all ABORTED have
+ * been received before all have been sent.
+ *---------------------------------------------------------------------*/
+ apiConnectptr.p->counter++;
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ signal->theData[0] = TcContinueB::ZABORT_BREAK;
+ signal->theData[1] = tcConnectptr.i;
+ signal->theData[2] = apiConnectptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ }//if
+ }//if
+ if (apiConnectptr.p->counter > 0) {
+ jam();
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ return;
+ }//if
+ /*-----------------------------------------------------------------------
+ * WE HAVE NOW COMPLETED THE ABORT PROCESS. WE HAVE RECEIVED ABORTED
+ * FROM ALL PARTICIPANTS IN THE TRANSACTION. WE CAN NOW RELEASE ALL
+ * RESOURCES CONNECTED TO THE TRANSACTION AND SEND THE ABORT RESPONSE
+ *------------------------------------------------------------------------*/
+ releaseAbortResources(signal);
+}//Dbtc::abort015Lab()
+
+/*--------------------------------------------------------------------------*/
+/* RELEASE KEY AND ATTRINFO OBJECTS AND SEND ABORT TO THE LQH BLOCK. */
+/*--------------------------------------------------------------------------*/
+int Dbtc::releaseAndAbort(Signal* signal)
+{
+ HostRecordPtr localHostptr;
+ UintR TnoLoops = tcConnectptr.p->noOfNodes;
+
+ apiConnectptr.p->counter++;
+ for (Uint32 Ti = 0; Ti < TnoLoops ; Ti++) {
+ localHostptr.i = tcConnectptr.p->tcNodedata[Ti];
+ ptrCheckGuard(localHostptr, chostFilesize, hostRecord);
+ if (localHostptr.p->hostStatus == HS_ALIVE) {
+ jam();
+ /* ************< */
+ /* ABORT < */
+ /* ************< */
+ tblockref = calcLqhBlockRef(localHostptr.i);
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = apiConnectptr.p->transid[0];
+ signal->theData[3] = apiConnectptr.p->transid[1];
+ sendSignal(tblockref, GSN_ABORT, signal, 4, JBB);
+ return 1;
+ } else {
+ jam();
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = apiConnectptr.p->transid[0];
+ signal->theData[2] = apiConnectptr.p->transid[1];
+ signal->theData[3] = hostptr.i;
+ signal->theData[4] = ZFALSE;
+ sendSignal(cownref, GSN_ABORTED, signal, 5, JBB);
+ }//if
+ }//for
+ return 1;
+}//Dbtc::releaseAndAbort()
+
+/* ------------------------------------------------------------------------- */
+/* ------- ENTER TIME_SIGNAL ------- */
+/* */
+/* ------------------------------------------------------------------------- */
+void Dbtc::execTIME_SIGNAL(Signal* signal)
+{
+
+ jamEntry();
+ ctcTimer++;
+ if (csystemStart != SSS_TRUE) {
+ jam();
+ return;
+ }//if
+ checkStartTimeout(signal);
+ checkStartFragTimeout(signal);
+}//Dbtc::execTIME_SIGNAL()
+
+/*------------------------------------------------*/
+/* Start timeout handling if not already going on */
+/*------------------------------------------------*/
+void Dbtc::checkStartTimeout(Signal* signal)
+{
+ ctimeOutCheckCounter++;
+ if (ctimeOutCheckActive == TOCS_TRUE) {
+ jam();
+ // Check heartbeat of timeout loop
+ if(ctimeOutCheckHeartbeat > ctimeOutCheckLastHeartbeat){
+ jam();
+ ctimeOutMissedHeartbeats = 0;
+ }else{
+ jam();
+ ctimeOutMissedHeartbeats++;
+ if (ctimeOutMissedHeartbeats > 100){
+ jam();
+ systemErrorLab(signal);
+ }
+ }
+ ctimeOutCheckLastHeartbeat = ctimeOutCheckHeartbeat;
+ return;
+ }//if
+ if (ctimeOutCheckCounter < ctimeOutCheckDelay) {
+ jam();
+ /*------------------------------------------------------------------*/
+ /* */
+ /* NO TIME-OUT CHECKED THIS TIME. WAIT MORE. */
+ /*------------------------------------------------------------------*/
+ return;
+ }//if
+ ctimeOutCheckActive = TOCS_TRUE;
+ ctimeOutCheckCounter = 0;
+ timeOutLoopStartLab(signal, 0); // 0 is first api connect record
+ return;
+}//Dbtc::execTIME_SIGNAL()
+
+/*----------------------------------------------------------------*/
+/* Start fragment (scan) timeout handling if not already going on */
+/*----------------------------------------------------------------*/
+void Dbtc::checkStartFragTimeout(Signal* signal)
+{
+ ctimeOutCheckFragCounter++;
+ if (ctimeOutCheckFragActive == TOCS_TRUE) {
+ jam();
+ return;
+ }//if
+ if (ctimeOutCheckFragCounter < ctimeOutCheckDelay) {
+ jam();
+ /*------------------------------------------------------------------*/
+ /* NO TIME-OUT CHECKED THIS TIME. WAIT MORE. */
+ /*------------------------------------------------------------------*/
+ return;
+ }//if
+
+ // Go through the fragment records and look for timeout in a scan.
+ ctimeOutCheckFragActive = TOCS_TRUE;
+ ctimeOutCheckFragCounter = 0;
+ timeOutLoopStartFragLab(signal, 0); // 0 means first scan record
+}//checkStartFragTimeout()
+
+/*------------------------------------------------------------------*/
+/* IT IS NOW TIME TO CHECK WHETHER ANY TRANSACTIONS HAVE */
+/* BEEN DELAYED FOR SO LONG THAT WE ARE FORCED TO PERFORM */
+/* SOME ACTION, EITHER ABORT OR RESEND OR REMOVE A NODE FROM */
+/* THE WAITING PART OF A PROTOCOL. */
+/*------------------------------------------------------------------*/
+void Dbtc::timeOutLoopStartLab(Signal* signal, Uint32 TapiConPtr)
+{
+ UintR texpiredTime[8];
+ UintR TloopCount = 0;
+
+ ctimeOutCheckHeartbeat = ctcTimer;
+
+ const Uint32 TapiConSz = capiConnectFilesize;
+ const Uint32 TtcTimer = ctcTimer;
+ const Uint32 TtimeOutValue = ctimeOutValue;
+
+ while ((TapiConPtr + 8) < TapiConSz) {
+ jam();
+ texpiredTime[0] = TtcTimer - getApiConTimer(TapiConPtr + 0);
+ texpiredTime[1] = TtcTimer - getApiConTimer(TapiConPtr + 1);
+ texpiredTime[2] = TtcTimer - getApiConTimer(TapiConPtr + 2);
+ texpiredTime[3] = TtcTimer - getApiConTimer(TapiConPtr + 3);
+ texpiredTime[4] = TtcTimer - getApiConTimer(TapiConPtr + 4);
+ texpiredTime[5] = TtcTimer - getApiConTimer(TapiConPtr + 5);
+ texpiredTime[6] = TtcTimer - getApiConTimer(TapiConPtr + 6);
+ texpiredTime[7] = TtcTimer - getApiConTimer(TapiConPtr + 7);
+ for (Uint32 Ti = 0; Ti < 8; Ti++) {
+ if (getApiConTimer(TapiConPtr + Ti) != 0) {
+ if (texpiredTime[Ti] > TtimeOutValue) {
+ jam();
+ timeOutFoundLab(signal, TapiConPtr + Ti);
+ return;
+ }//if
+ }//if
+ }//for
+ TapiConPtr += 8;
+ if (TloopCount++ > 128) {
+ jam();
+ sendContinueTimeOutControl(signal, TapiConPtr);
+ return;
+ }//if
+ }//while
+ for ( ; TapiConPtr < TapiConSz; TapiConPtr++) {
+ jam();
+ if (getApiConTimer(TapiConPtr) != 0) {
+ texpiredTime[0] = TtcTimer - getApiConTimer(TapiConPtr);
+ if (texpiredTime[0] > TtimeOutValue) {
+ jam();
+ timeOutFoundLab(signal, TapiConPtr);
+ return;
+ }//if
+ }//if
+ }//for
+ /*------------------------------------------------------------------*/
+ /* */
+ /* WE HAVE NOW CHECKED ALL TRANSACTIONS FOR TIME-OUT AND ALSO */
+ /* STARTED TIME-OUT HANDLING OF THOSE WE FOUND. WE ARE NOW */
+ /* READY AND CAN WAIT FOR THE NEXT TIME-OUT CHECK. */
+ /*------------------------------------------------------------------*/
+ ctimeOutCheckActive = TOCS_FALSE;
+}//Dbtc::timeOutLoopStartLab()
+
+void Dbtc::timeOutFoundLab(Signal* signal, Uint32 TapiConPtr)
+{
+ sendContinueTimeOutControl(signal, TapiConPtr + 1);
+
+ apiConnectptr.i = TapiConPtr;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ /*------------------------------------------------------------------*/
+ /* */
+ /* THIS TRANSACTION HAVE EXPERIENCED A TIME-OUT AND WE NEED TO*/
+ /* FIND OUT WHAT WE NEED TO DO BASED ON THE STATE INFORMATION.*/
+ /*------------------------------------------------------------------*/
+ DEBUG("Time-out in state = " << apiConnectptr.p->apiConnectstate
+ << " apiConnectptr.i = " << apiConnectptr.i);
+ switch (apiConnectptr.p->apiConnectstate) {
+ case CS_STARTED:
+ if(apiConnectptr.p->lqhkeyreqrec == apiConnectptr.p->lqhkeyconfrec){
+ jam();
+ /*
+ We are waiting for application to continue the transaction. In this
+ particular state we will use the application timeout parameter rather
+ than the shorter Deadlock detection timeout.
+ */
+ if ((ctcTimer - getApiConTimer(apiConnectptr.i)) <= c_appl_timeout_value) {
+ jam();
+ return;
+ }//if
+ apiConnectptr.p->returnsignal = RS_NO_RETURN;
+ } else {
+ jam();
+ apiConnectptr.p->returnsignal = RS_TCROLLBACKREP;
+ }
+ apiConnectptr.p->returncode = ZTIME_OUT_ERROR;
+ abort010Lab(signal);
+ return;
+ case CS_RECEIVING:
+ case CS_REC_COMMITTING:
+ case CS_START_COMMITTING:
+ jam();
+ /*------------------------------------------------------------------*/
+ /* WE ARE STILL IN THE PREPARE PHASE AND THE TRANSACTION HAS */
+ /* NOT YET REACHED ITS COMMIT POINT. THUS IT IS NOW OK TO */
+ /* START ABORTING THE TRANSACTION. ALSO START CHECKING THE */
+ /* REMAINING TRANSACTIONS. */
+ /*------------------------------------------------------------------*/
+ terrorCode = ZTIME_OUT_ERROR;
+ abortErrorLab(signal);
+ return;
+ case CS_COMMITTING:
+ jam();
+ /*------------------------------------------------------------------*/
+ // We are simply waiting for a signal in the job buffer. Only extreme
+ // conditions should get us here. We ignore it.
+ /*------------------------------------------------------------------*/
+ case CS_COMPLETING:
+ jam();
+ /*------------------------------------------------------------------*/
+ // We are simply waiting for a signal in the job buffer. Only extreme
+ // conditions should get us here. We ignore it.
+ /*------------------------------------------------------------------*/
+ case CS_PREPARE_TO_COMMIT:
+ jam();
+ /*------------------------------------------------------------------*/
+ /* WE ARE WAITING FOR DIH TO COMMIT THE TRANSACTION. WE SIMPLY*/
+ /* KEEP WAITING SINCE THERE IS NO BETTER IDEA ON WHAT TO DO. */
+ /* IF IT IS BLOCKED THEN NO TRANSACTION WILL PASS THIS GATE. */
+ // To ensure against strange bugs we crash the system if we have passed
+ // time-out period by a factor of 10 and it is also at least 5 seconds.
+ /*------------------------------------------------------------------*/
+ if (((ctcTimer - getApiConTimer(apiConnectptr.i)) > (10 * ctimeOutValue)) &&
+ ((ctcTimer - getApiConTimer(apiConnectptr.i)) > 500)) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ break;
+ case CS_COMMIT_SENT:
+ jam();
+ /*------------------------------------------------------------------*/
+ /* WE HAVE SENT COMMIT TO A NUMBER OF NODES. WE ARE CURRENTLY */
+ /* WAITING FOR THEIR REPLY. WITH NODE RECOVERY SUPPORTED WE */
+ /* WILL CHECK FOR CRASHED NODES AND RESEND THE COMMIT SIGNAL */
+ /* TO THOSE NODES THAT HAVE MISSED THE COMMIT SIGNAL DUE TO */
+ /* A NODE FAILURE. */
+ /*------------------------------------------------------------------*/
+ tabortInd = ZCOMMIT_SETUP;
+ setupFailData(signal);
+ toCommitHandlingLab(signal);
+ return;
+ case CS_COMPLETE_SENT:
+ jam();
+ /*--------------------------------------------------------------------*/
+ /* WE HAVE SENT COMPLETE TO A NUMBER OF NODES. WE ARE CURRENTLY */
+ /* WAITING FOR THEIR REPLY. WITH NODE RECOVERY SUPPORTED WE */
+ /* WILL CHECK FOR CRASHED NODES AND RESEND THE COMPLETE SIGNAL */
+ /* TO THOSE NODES THAT HAVE MISSED THE COMPLETE SIGNAL DUE TO */
+ /* A NODE FAILURE. */
+ /*--------------------------------------------------------------------*/
+ tabortInd = ZCOMMIT_SETUP;
+ setupFailData(signal);
+ toCompleteHandlingLab(signal);
+ return;
+ case CS_ABORTING:
+ jam();
+ /*------------------------------------------------------------------*/
+ /* TIME-OUT DURING ABORT. WE NEED TO SEND ABORTED FOR ALL */
+ /* NODES THAT HAVE FAILED BEFORE SENDING ABORTED. */
+ /*------------------------------------------------------------------*/
+ tcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ sendAbortedAfterTimeout(signal, 0);
+ break;
+ case CS_START_SCAN:
+ jam();
+ apiConnectptr.p->returncode = ZSCANTIME_OUT_ERROR;
+ handleScanStop(signal, 0);
+ break;
+ case CS_WAIT_ABORT_CONF:
+ jam();
+ tcConnectptr.i = apiConnectptr.p->currentTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ arrGuard(apiConnectptr.p->currentReplicaNo, 4);
+ hostptr.i = tcConnectptr.p->tcNodedata[apiConnectptr.p->currentReplicaNo];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ /*------------------------------------------------------------------*/
+ // Time-out waiting for ABORTCONF. We will resend the ABORTREQ just in
+ // case.
+ /*------------------------------------------------------------------*/
+ warningReport(signal, 20);
+ apiConnectptr.p->timeOutCounter++;
+ if (apiConnectptr.p->timeOutCounter > 3) {
+ /*------------------------------------------------------------------*/
+ // 100 time-outs are not acceptable. We will shoot down the node
+ // not responding.
+ /*------------------------------------------------------------------*/
+ reportNodeFailed(signal, hostptr.i);
+ }//if
+ apiConnectptr.p->currentReplicaNo++;
+ }//if
+ tcurrentReplicaNo = (Uint8)Z8NIL;
+ toAbortHandlingLab(signal);
+ return;
+ case CS_WAIT_COMMIT_CONF:
+ jam();
+ tcConnectptr.i = apiConnectptr.p->currentTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ arrGuard(apiConnectptr.p->currentReplicaNo, 4);
+ hostptr.i = tcConnectptr.p->tcNodedata[apiConnectptr.p->currentReplicaNo];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ /*------------------------------------------------------------------*/
+ // Time-out waiting for COMMITCONF. We will resend the COMMITREQ just in
+ // case.
+ /*------------------------------------------------------------------*/
+ warningReport(signal, 21);
+ apiConnectptr.p->timeOutCounter++;
+ if (apiConnectptr.p->timeOutCounter > 3) {
+ /*------------------------------------------------------------------*/
+ // 100 time-outs are not acceptable. We will shoot down the node
+ // not responding.
+ /*------------------------------------------------------------------*/
+ reportNodeFailed(signal, hostptr.i);
+ }//if
+ apiConnectptr.p->currentReplicaNo++;
+ }//if
+ tcurrentReplicaNo = (Uint8)Z8NIL;
+ toCommitHandlingLab(signal);
+ return;
+ case CS_WAIT_COMPLETE_CONF:
+ jam();
+ tcConnectptr.i = apiConnectptr.p->currentTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ arrGuard(apiConnectptr.p->currentReplicaNo, 4);
+ hostptr.i = tcConnectptr.p->tcNodedata[apiConnectptr.p->currentReplicaNo];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ /*------------------------------------------------------------------*/
+ // Time-out waiting for COMPLETECONF. We will resend the COMPLETEREQ
+ // just in case.
+ /*------------------------------------------------------------------*/
+ warningReport(signal, 22);
+ apiConnectptr.p->timeOutCounter++;
+ if (apiConnectptr.p->timeOutCounter > 100) {
+ /*------------------------------------------------------------------*/
+ // 100 time-outs are not acceptable. We will shoot down the node
+ // not responding.
+ /*------------------------------------------------------------------*/
+ reportNodeFailed(signal, hostptr.i);
+ }//if
+ apiConnectptr.p->currentReplicaNo++;
+ }//if
+ tcurrentReplicaNo = (Uint8)Z8NIL;
+ toCompleteHandlingLab(signal);
+ return;
+ case CS_FAIL_PREPARED:
+ jam();
+ case CS_FAIL_COMMITTING:
+ jam();
+ case CS_FAIL_COMMITTED:
+ jam();
+ case CS_REC_PREPARING:
+ jam();
+ case CS_START_PREPARING:
+ jam();
+ case CS_PREPARED:
+ jam();
+ case CS_RESTART:
+ jam();
+ case CS_FAIL_ABORTED:
+ jam();
+ case CS_DISCONNECTED:
+ jam();
+ default:
+ jam();
+ /*------------------------------------------------------------------*/
+ /* AN IMPOSSIBLE STATE IS SET. CRASH THE SYSTEM. */
+ /*------------------------------------------------------------------*/
+ DEBUG("State = " << apiConnectptr.p->apiConnectstate);
+ systemErrorLab(signal);
+ return;
+ }//switch
+ return;
+}//Dbtc::timeOutFoundLab()
+
+void Dbtc::sendAbortedAfterTimeout(Signal* signal, int Tcheck)
+{
+ ApiConnectRecord * transP = apiConnectptr.p;
+ if(transP->abortState == AS_IDLE){
+ jam();
+ warningEvent("TC: %d: %d state=%d abort==IDLE place: %d fop=%d t: %d",
+ __LINE__,
+ apiConnectptr.i,
+ transP->apiConnectstate,
+ c_apiConTimer_line[apiConnectptr.i],
+ transP->firstTcConnect,
+ c_apiConTimer[apiConnectptr.i]
+ );
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ return;
+ }
+
+ OperationState tmp[16];
+
+ Uint32 TloopCount = 0;
+ do {
+ jam();
+ if (tcConnectptr.i == RNIL) {
+ jam();
+ if (Tcheck == 0) {
+ jam();
+ /*------------------------------------------------------------------
+ * All nodes had already reported ABORTED for all tcConnect records.
+ * Crash since it is an error situation that we then received a
+ * time-out.
+ *------------------------------------------------------------------*/
+ char buf[96]; buf[0] = 0;
+ char buf2[96];
+ snprintf(buf, sizeof(buf), "TC %d: %d ops:",
+ __LINE__, apiConnectptr.i);
+ for(Uint32 i = 0; i<TloopCount; i++){
+ snprintf(buf2, sizeof(buf2), "%s %d", buf, tmp[i]);
+ snprintf(buf, sizeof(buf), buf2);
+ }
+ warningEvent(buf);
+ releaseAbortResources(signal);
+ return;
+ }//if
+ return;
+ }//if
+ TloopCount++;
+ if (TloopCount >= 1024) {
+ jam();
+ /*------------------------------------------------------------------*/
+ // Insert a real-time break for large transactions to avoid blowing
+ // away the job buffer.
+ /*------------------------------------------------------------------*/
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ signal->theData[0] = TcContinueB::ZABORT_TIMEOUT_BREAK;
+ signal->theData[1] = tcConnectptr.i;
+ signal->theData[2] = apiConnectptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ }//if
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ if(TloopCount < 16){
+ jam();
+ tmp[TloopCount-1] = tcConnectptr.p->tcConnectstate;
+ }
+
+ if (tcConnectptr.p->tcConnectstate == OS_ABORT_SENT) {
+ jam();
+ /*------------------------------------------------------------------*/
+ // We have sent an ABORT signal to this node but not yet received any
+ // reply. We have to send an ABORTED signal on our own in some cases.
+ // If the node is declared as up and running and still do not respond
+ // in time to the ABORT signal we will declare it as dead.
+ /*------------------------------------------------------------------*/
+ UintR Ti = 0;
+ arrGuard(tcConnectptr.p->noOfNodes, 4);
+ for (Ti = 0; Ti < tcConnectptr.p->noOfNodes; Ti++) {
+ jam();
+ if (tcConnectptr.p->tcNodedata[Ti] != 0) {
+ TloopCount += 31;
+ Tcheck = 1;
+ hostptr.i = tcConnectptr.p->tcNodedata[Ti];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ jam();
+ /*---------------------------------------------------------------
+ * A backup replica has not sent ABORTED.
+ * Could be that a node before him has crashed.
+ * Send an ABORT signal specifically to this node.
+ * We will not send to any more nodes after this
+ * to avoid race problems.
+ * To also ensure that we use this message also as a heartbeat
+ * we will move this node to the primary replica seat.
+ * The primary replica and any failed node after it will
+ * be removed from the node list. Update also number of nodes.
+ * Finally break the loop to ensure we don't mess
+ * things up by executing another loop.
+ * We also update the timer to ensure we don't get time-out
+ * too early.
+ *--------------------------------------------------------------*/
+ BlockReference TBRef = calcLqhBlockRef(hostptr.i);
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = apiConnectptr.p->transid[0];
+ signal->theData[3] = apiConnectptr.p->transid[1];
+ sendSignal(TBRef, GSN_ABORT, signal, 4, JBB);
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ break;
+ } else {
+ jam();
+ /*--------------------------------------------------------------
+ * The node we are waiting for is dead. We will send ABORTED to
+ * ourselves vicarious for the failed node.
+ *--------------------------------------------------------------*/
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = apiConnectptr.p->transid[0];
+ signal->theData[2] = apiConnectptr.p->transid[1];
+ signal->theData[3] = hostptr.i;
+ signal->theData[4] = ZFALSE;
+ sendSignal(cownref, GSN_ABORTED, signal, 5, JBB);
+ }//if
+ }//if
+ }//for
+ }//if
+ tcConnectptr.i = tcConnectptr.p->nextTcConnect;
+ } while (1);
+}//Dbtc::sendAbortedAfterTimeout()
+
+void Dbtc::reportNodeFailed(Signal* signal, Uint32 nodeId)
+{
+ DisconnectRep * const rep = (DisconnectRep *)&signal->theData[0];
+ rep->nodeId = nodeId;
+ rep->err = DisconnectRep::TcReportNodeFailed;
+ sendSignal(QMGR_REF, GSN_DISCONNECT_REP, signal,
+ DisconnectRep::SignalLength, JBB);
+}//Dbtc::reportNodeFailed()
+
+/*-------------------------------------------------*/
+/* Timeout-loop for scanned fragments. */
+/*-------------------------------------------------*/
+void Dbtc::timeOutLoopStartFragLab(Signal* signal, Uint32 TscanConPtr)
+{
+ ScanFragRecPtr timeOutPtr[8];
+ UintR tfragTimer[8];
+ UintR texpiredTime[8];
+ UintR TloopCount = 0;
+ Uint32 TtcTimer = ctcTimer;
+
+ while ((TscanConPtr + 8) < cscanFragrecFileSize) {
+ jam();
+ timeOutPtr[0].i = TscanConPtr + 0;
+ timeOutPtr[1].i = TscanConPtr + 1;
+ timeOutPtr[2].i = TscanConPtr + 2;
+ timeOutPtr[3].i = TscanConPtr + 3;
+ timeOutPtr[4].i = TscanConPtr + 4;
+ timeOutPtr[5].i = TscanConPtr + 5;
+ timeOutPtr[6].i = TscanConPtr + 6;
+ timeOutPtr[7].i = TscanConPtr + 7;
+
+ ptrAss(timeOutPtr[0], scanFragmentRecord);
+ ptrAss(timeOutPtr[1], scanFragmentRecord);
+ ptrAss(timeOutPtr[2], scanFragmentRecord);
+ ptrAss(timeOutPtr[3], scanFragmentRecord);
+ ptrAss(timeOutPtr[4], scanFragmentRecord);
+ ptrAss(timeOutPtr[5], scanFragmentRecord);
+ ptrAss(timeOutPtr[6], scanFragmentRecord);
+ ptrAss(timeOutPtr[7], scanFragmentRecord);
+
+ tfragTimer[0] = timeOutPtr[0].p->scanFragTimer;
+ tfragTimer[1] = timeOutPtr[1].p->scanFragTimer;
+ tfragTimer[2] = timeOutPtr[2].p->scanFragTimer;
+ tfragTimer[3] = timeOutPtr[3].p->scanFragTimer;
+ tfragTimer[4] = timeOutPtr[4].p->scanFragTimer;
+ tfragTimer[5] = timeOutPtr[5].p->scanFragTimer;
+ tfragTimer[6] = timeOutPtr[6].p->scanFragTimer;
+ tfragTimer[7] = timeOutPtr[7].p->scanFragTimer;
+
+ texpiredTime[0] = TtcTimer - tfragTimer[0];
+ texpiredTime[1] = TtcTimer - tfragTimer[1];
+ texpiredTime[2] = TtcTimer - tfragTimer[2];
+ texpiredTime[3] = TtcTimer - tfragTimer[3];
+ texpiredTime[4] = TtcTimer - tfragTimer[4];
+ texpiredTime[5] = TtcTimer - tfragTimer[5];
+ texpiredTime[6] = TtcTimer - tfragTimer[6];
+ texpiredTime[7] = TtcTimer - tfragTimer[7];
+
+ for (Uint32 Ti = 0; Ti < 8; Ti++) {
+ jam();
+ if (tfragTimer[Ti] != 0) {
+
+ if (texpiredTime[Ti] > ctimeOutValue) {
+ jam();
+ DEBUG("Fragment timeout found:"<<
+ " ctimeOutValue=" <<ctimeOutValue
+ <<", texpiredTime="<<texpiredTime[Ti]<<endl
+ <<" tfragTimer="<<tfragTimer[Ti]
+ <<", ctcTimer="<<ctcTimer);
+ timeOutFoundFragLab(signal, TscanConPtr + Ti);
+ return;
+ }//if
+ }//if
+ }//for
+ TscanConPtr += 8;
+ /*----------------------------------------------------------------*/
+ /* We split the process up checking 1024 fragmentrecords at a time*/
+ /* to maintain real time behaviour. */
+ /*----------------------------------------------------------------*/
+ if (TloopCount++ > 128 ) {
+ jam();
+ signal->theData[0] = TcContinueB::ZCONTINUE_TIME_OUT_FRAG_CONTROL;
+ signal->theData[1] = TscanConPtr;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ }//if
+ }//while
+ for ( ; TscanConPtr < cscanFragrecFileSize; TscanConPtr++){
+ jam();
+ timeOutPtr[0].i = TscanConPtr;
+ ptrAss(timeOutPtr[0], scanFragmentRecord);
+ if (timeOutPtr[0].p->scanFragTimer != 0) {
+ texpiredTime[0] = ctcTimer - timeOutPtr[0].p->scanFragTimer;
+ if (texpiredTime[0] > ctimeOutValue) {
+ jam();
+ DEBUG("Fragment timeout found:"<<
+ " ctimeOutValue=" <<ctimeOutValue
+ <<", texpiredTime="<<texpiredTime[0]<<endl
+ <<" tfragTimer="<<tfragTimer[0]
+ <<", ctcTimer="<<ctcTimer);
+ timeOutFoundFragLab(signal, TscanConPtr);
+ return;
+ }//if
+ }//if
+ }//for
+ ctimeOutCheckFragActive = TOCS_FALSE;
+ return;
+}//timeOutLoopStartFragLab()
+
+/*--------------------------------------------------------------------------*/
+/*Handle the heartbeat signal from LQH in a scan process */
+// (Set timer on fragrec.)
+/*--------------------------------------------------------------------------*/
+void Dbtc::execSCAN_HBREP(Signal* signal)
+{
+ jamEntry();
+
+ scanFragptr.i = signal->theData[0];
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+
+ switch (scanFragptr.p->scanFragState){
+ case ScanFragRec::LQH_ACTIVE:
+ case ScanFragRec::LQH_ACTIVE_CLOSE:
+ break;
+
+ default:
+ DEBUG("execSCAN_HBREP: scanFragState="<<scanFragptr.p->scanFragState);
+ systemErrorLab(signal);
+ break;
+ }
+
+ scanptr.i = scanFragptr.p->scanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+
+ if (!(apiConnectptr.p->transid[0] == signal->theData[1] &&
+ apiConnectptr.p->transid[1] == signal->theData[2])){
+ jam();
+ /**
+ * Send signal back to sender so that the crash occurs there
+ */
+ // Save original transid
+ signal->theData[3] = signal->theData[0];
+ signal->theData[4] = signal->theData[1];
+ // Set transid to illegal values
+ signal->theData[1] = RNIL;
+ signal->theData[2] = RNIL;
+
+ sendSignal(signal->senderBlockRef(), GSN_SCAN_HBREP, signal, 5, JBA);
+ DEBUG("SCAN_HBREP with wrong transid("
+ <<signal->theData[3]<<", "<<signal->theData[4]<<")");
+ return;
+ }//if
+
+ // Update timer on ScanFragRec
+ if (scanFragptr.p->scanFragTimer != 0){
+ updateBuddyTimer(apiConnectptr);
+ scanFragptr.p->startFragTimer(ctcTimer);
+ } else {
+ DEBUG("SCAN_HBREP when scanFragTimer was turned off");
+ }
+}//execSCAN_HBREP()
+
+/*--------------------------------------------------------------------------*/
+/* Timeout has occured on a fragment which means a scan has timed out. */
+/* If this is true we have an error in LQH/ACC. */
+/*--------------------------------------------------------------------------*/
+void Dbtc::timeOutFoundFragLab(Signal* signal, UintR TscanConPtr)
+{
+ scanFragptr.i = TscanConPtr;
+ ptrAss(scanFragptr, scanFragmentRecord);
+ DEBUG("timeOutFoundFragLab: scanFragState = "<<scanFragptr.p->scanFragState);
+
+ /*-------------------------------------------------------------------------*/
+ // The scan fragment has expired its timeout. Check its state to decide
+ // what to do.
+ /*-------------------------------------------------------------------------*/
+ switch (scanFragptr.p->scanFragState) {
+
+ case ScanFragRec::WAIT_GET_PRIMCONF:
+ jam();
+ // Crash the system if we do not return from DIGETPRIMREQ in time.
+ systemErrorLab(signal);
+ break;
+
+ case ScanFragRec::LQH_ACTIVE:
+ jam();
+ /**
+ * The LQH expired it's timeout, try to close it
+ */
+ scanFragError(signal, ZSCAN_FRAG_LQH_ERROR);
+ DEBUG(" LQH_ACTIVE - closing the fragment scan in node "
+ <<scanFragptr.p->scanFragNodeId);
+ break;
+
+ case ScanFragRec::LQH_ACTIVE_CLOSE:{
+ jam();
+ /**
+ * The close of LQH expired its time-out. This is not
+ * acceptable behaviour from LQH and thus we will shoot
+ * it down.
+ */
+ Uint32 nodeId = scanFragptr.p->scanFragNodeId;
+ Uint32 cc = scanFragptr.p->m_connectCount;
+ if(getNodeInfo(nodeId).m_connectCount == cc){
+ const BlockReference errRef = calcNdbCntrBlockRef(nodeId);
+ SystemError * const sysErr = (SystemError*)&signal->theData[0];
+ sysErr->errorCode = SystemError::ScanfragTimeout;
+ sysErr->errorRef = reference();
+ sysErr->data1 = scanFragptr.i;
+ sysErr->data2 = scanFragptr.p->scanRec;
+ sendSignal(errRef, GSN_SYSTEM_ERROR, signal,
+ SystemError::SignalLength, JBA);
+ DEBUG(" node " << nodeId << " killed");
+ } else {
+ DEBUG(" node " << nodeId << " not killed as it has restarted");
+ }
+ scanFragptr.p->stopFragTimer();
+ break;
+ }
+
+ case ScanFragRec::DELIVERED:
+ jam();
+ case ScanFragRec::RETURNING_FROM_DELIVERY:
+ jam();
+ case ScanFragRec::IDLE:
+ jam();
+ case ScanFragRec::QUEUED_FOR_DELIVERY:
+ jam();
+ /*-----------------------------------------------------------------------
+ * Should never occur. We will simply report set the timer to zero and
+ * continue. In a debug version we should crash here but not in a release
+ * version. In a release version we will simply set the time-out to zero.
+ *-----------------------------------------------------------------------*/
+#ifdef VM_TRACE
+ systemErrorLab(signal);
+#endif
+ scanFragptr.p->stopFragTimer();
+ break;
+ default:
+ jam();
+ /*-----------------------------------------------------------------------
+ * Non-existent state. Crash.
+ *-----------------------------------------------------------------------*/
+ systemErrorLab(signal);
+ break;
+ }//switch
+
+ signal->theData[0] = TcContinueB::ZCONTINUE_TIME_OUT_FRAG_CONTROL;
+ signal->theData[1] = TscanConPtr + 1;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+}//timeOutFoundFragLab()
+
+
+/*
+ 4.3.16 GCP_NOMORETRANS
+ ----------------------
+*/
+/*****************************************************************************
+ * G C P _ N O M O R E T R A N S
+ *
+ * WHEN DBTC RECEIVES SIGNAL GCP_NOMORETRANS A CHECK IS DONE TO FIND OUT IF
+ * THERE ARE ANY GLOBAL CHECKPOINTS GOING ON - CFIRSTGCP /= RNIL. DBTC THEN
+ * SEARCHES THE GCP_RECORD FILE TO FIND OUT IF THERE ARE ANY TRANSACTIONS NOT
+ * CONCLUDED WITH THIS SPECIFIC CHECKPOINT - GCP_PTR:GCP_ID = TCHECK_GCP_ID.
+ * FOR EACH TRANSACTION WHERE API_CONNECTSTATE EQUALS PREPARED, COMMITTING,
+ * COMMITTED OR COMPLETING SIGNAL CONTINUEB IS SENT WITH A DELAY OF 100 MS,
+ * THE COUNTER GCP_PTR:OUTSTANDINGAPI IS INCREASED. WHEN CONTINUEB IS RECEIVED
+ * THE COUNTER IS DECREASED AND A CHECK IS DONE TO FIND OUT IF ALL
+ * TRANSACTIONS ARE CONCLUDED. IF SO, SIGNAL GCP_TCFINISHED IS SENT.
+ *****************************************************************************/
+void Dbtc::execGCP_NOMORETRANS(Signal* signal)
+{
+ jamEntry();
+ tcheckGcpId = signal->theData[1];
+ if (cfirstgcp != RNIL) {
+ jam();
+ /* A GLOBAL CHECKPOINT IS GOING ON */
+ gcpPtr.i = cfirstgcp; /* SET POINTER TO FIRST GCP IN QUEUE*/
+ ptrCheckGuard(gcpPtr, cgcpFilesize, gcpRecord);
+ if (gcpPtr.p->gcpId == tcheckGcpId) {
+ jam();
+ if (gcpPtr.p->firstApiConnect != RNIL) {
+ jam();
+ gcpPtr.p->gcpNomoretransRec = ZTRUE;
+ } else {
+ jam();
+ gcpTcfinished(signal);
+ unlinkGcp(signal);
+ }//if
+ } else {
+ jam();
+ /*------------------------------------------------------------*/
+ /* IF IT IS NOT THE FIRST THEN THERE SHOULD BE NO */
+ /* RECORD FOR THIS GLOBAL CHECKPOINT. WE ALWAYS REMOVE */
+ /* THE GLOBAL CHECKPOINTS IN ORDER. */
+ /*------------------------------------------------------------*/
+ gcpTcfinished(signal);
+ }//if
+ } else {
+ jam();
+ gcpTcfinished(signal);
+ }//if
+ return;
+}//Dbtc::execGCP_NOMORETRANS()
+
+/*****************************************************************************/
+/* */
+/* TAKE OVER MODULE */
+/* */
+/*****************************************************************************/
+/* */
+/* THIS PART OF TC TAKES OVER THE COMMIT/ABORT OF TRANSACTIONS WHERE THE */
+/* NODE ACTING AS TC HAVE FAILED. IT STARTS BY QUERYING ALL NODES ABOUT */
+/* ANY OPERATIONS PARTICIPATING IN A TRANSACTION WHERE THE TC NODE HAVE */
+/* FAILED. */
+/* */
+/* AFTER RECEIVING INFORMATION FROM ALL NODES ABOUT OPERATION STATUS THIS */
+/* CODE WILL ENSURE THAT ALL AFFECTED TRANSACTIONS ARE PROPERLY ABORTED OR*/
+/* COMMITTED. THE ORIGINATING APPLICATION NODE WILL ALSO BE CONTACTED. */
+/* IF THE ORIGINATING APPLICATION ALSO FAILED THEN THERE IS CURRENTLY NO */
+/* WAY TO FIND OUT WHETHER A TRANSACTION WAS PERFORMED OR NOT. */
+/*****************************************************************************/
+void Dbtc::execNODE_FAILREP(Signal* signal)
+{
+ HostRecordPtr tmpHostptr;
+ jamEntry();
+
+ NodeFailRep * const nodeFail = (NodeFailRep *)&signal->theData[0];
+
+ cfailure_nr = nodeFail->failNo;
+ const Uint32 tnoOfNodes = nodeFail->noOfNodes;
+ const Uint32 tnewMasterId = nodeFail->masterNodeId;
+
+ arrGuard(tnoOfNodes, MAX_NDB_NODES);
+ int index = 0;
+ for (unsigned i = 1; i< MAX_NDB_NODES; i++) {
+ if(NodeBitmask::get(nodeFail->theNodes, i)){
+ cdata[index] = i;
+ index++;
+ }//if
+ }//for
+
+ tcNodeFailptr.i = 0;
+ ptrAss(tcNodeFailptr, tcFailRecord);
+ for (Uint32 tindex = 0; tindex < tnoOfNodes; tindex++) {
+ jam();
+ hostptr.i = cdata[tindex];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ /*------------------------------------------------------------*/
+ /* SET STATUS OF THE FAILED NODE TO DEAD SINCE IT HAS */
+ /* FAILED. */
+ /*------------------------------------------------------------*/
+ hostptr.p->hostStatus = HS_DEAD;
+
+ if (hostptr.p->takeOverStatus == TOS_COMPLETED) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* A VERY UNUSUAL SITUATION. THE TAKE OVER WAS COMPLETED*/
+ /* EVEN BEFORE WE HEARD ABOUT THE NODE FAILURE REPORT. */
+ /* HOWEVER UNUSUAL THIS SITUATION IS POSSIBLE. */
+ /*------------------------------------------------------------*/
+ /* RELEASE THE CURRENTLY UNUSED LQH CONNECTIONS. THE */
+ /* REMAINING WILL BE RELEASED WHEN THE TRANSACTION THAT */
+ /* USED THEM IS COMPLETED. */
+ /*------------------------------------------------------------*/
+ {
+ NFCompleteRep * const nfRep = (NFCompleteRep *)&signal->theData[0];
+ nfRep->blockNo = DBTC;
+ nfRep->nodeId = cownNodeid;
+ nfRep->failedNodeId = hostptr.i;
+ }
+ sendSignal(cdihblockref, GSN_NF_COMPLETEREP, signal,
+ NFCompleteRep::SignalLength, JBB);
+ } else {
+ ndbrequire(hostptr.p->takeOverStatus == TOS_IDLE);
+ hostptr.p->takeOverStatus = TOS_NODE_FAILED;
+ }//if
+
+ if (tcNodeFailptr.p->failStatus == FS_LISTENING) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* THE CURRENT TAKE OVER CAN BE AFFECTED BY THIS NODE */
+ /* FAILURE. */
+ /*------------------------------------------------------------*/
+ if (hostptr.p->lqhTransStatus == LTS_ACTIVE) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* WE WERE WAITING FOR THE FAILED NODE IN THE TAKE OVER */
+ /* PROTOCOL FOR TC. */
+ /*------------------------------------------------------------*/
+ signal->theData[0] = TcContinueB::ZNODE_TAKE_OVER_COMPLETED;
+ signal->theData[1] = hostptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+ }//if
+
+ }//for
+
+ const bool masterFailed = (cmasterNodeId != tnewMasterId);
+ cmasterNodeId = tnewMasterId;
+
+ if(getOwnNodeId() == cmasterNodeId && masterFailed){
+ /**
+ * Master has failed and I'm the new master
+ */
+ jam();
+
+ for (hostptr.i = 1; hostptr.i < MAX_NDB_NODES; hostptr.i++) {
+ jam();
+ ptrAss(hostptr, hostRecord);
+ if (hostptr.p->hostStatus != HS_ALIVE) {
+ jam();
+ if (hostptr.p->takeOverStatus == TOS_COMPLETED) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* SEND TAKE OVER CONFIRMATION TO ALL ALIVE NODES IF */
+ /* TAKE OVER IS COMPLETED. THIS IS PERFORMED TO ENSURE */
+ /* THAT ALL NODES AGREE ON THE IDLE STATE OF THE TAKE */
+ /* OVER. THIS MIGHT BE MISSED IN AN ERROR SITUATION IF */
+ /* MASTER FAILS AFTER SENDING CONFIRMATION TO NEW */
+ /* MASTER BUT FAILING BEFORE SENDING TO ANOTHER NODE */
+ /* WHICH WAS NOT MASTER. IF THIS NODE LATER BECOMES */
+ /* MASTER IT MIGHT START A NEW TAKE OVER EVEN AFTER THE */
+ /* CRASHED NODE HAVE ALREADY RECOVERED. */
+ /*------------------------------------------------------------*/
+ for(tmpHostptr.i = 1; tmpHostptr.i < MAX_NDB_NODES;tmpHostptr.i++) {
+ jam();
+ ptrAss(tmpHostptr, hostRecord);
+ if (tmpHostptr.p->hostStatus == HS_ALIVE) {
+ jam();
+ tblockref = calcTcBlockRef(tmpHostptr.i);
+ signal->theData[0] = hostptr.i;
+ sendSignal(tblockref, GSN_TAKE_OVERTCCONF, signal, 1, JBB);
+ }//if
+ }//for
+ }//if
+ }//if
+ }//for
+ }
+
+ if(getOwnNodeId() == cmasterNodeId){
+ jam();
+ for (hostptr.i = 1; hostptr.i < MAX_NDB_NODES; hostptr.i++) {
+ jam();
+ ptrAss(hostptr, hostRecord);
+ if (hostptr.p->hostStatus != HS_ALIVE) {
+ jam();
+ if (hostptr.p->takeOverStatus == TOS_NODE_FAILED) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* CONCLUDE ALL ACTIVITIES THE FAILED TC DID CONTROL */
+ /* SINCE WE ARE THE MASTER. THIS COULD HAVE BEEN STARTED*/
+ /* BY A PREVIOUS MASTER BUT HAVE NOT BEEN CONCLUDED YET.*/
+ /*------------------------------------------------------------*/
+ hostptr.p->takeOverStatus = TOS_ACTIVE;
+ signal->theData[0] = hostptr.i;
+ sendSignal(cownref, GSN_TAKE_OVERTCREQ, signal, 1, JBB);
+ }//if
+ }//if
+ }//for
+ }//if
+
+ for (Uint32 tindex = 0; tindex < tnoOfNodes; tindex++) {
+ jam();
+ hostptr.i = cdata[tindex];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ /*------------------------------------------------------------*/
+ /* LOOP THROUGH AND ABORT ALL SCANS THAT WHERE */
+ /* CONTROLLED BY THIS TC AND ACTIVE IN THE FAILED */
+ /* NODE'S LQH */
+ /*------------------------------------------------------------*/
+ checkScanActiveInFailedLqh(signal, 0, hostptr.i);
+ checkWaitDropTabFailedLqh(signal, hostptr.i, 0); // nodeid, tableid
+ }//for
+
+}//Dbtc::execNODE_FAILREP()
+
+void Dbtc::checkScanActiveInFailedLqh(Signal* signal,
+ Uint32 scanPtrI,
+ Uint32 failedNodeId){
+
+ for (scanptr.i = scanPtrI; scanptr.i < cscanrecFileSize; scanptr.i++) {
+ jam();
+ ptrAss(scanptr, scanRecord);
+ if (scanptr.p->scanState != ScanRecord::IDLE){
+ for (Uint32 i=0; i<16; i++) {
+ jam();
+ scanFragptr.i = scanptr.p->scanFragrec[i];
+ if (scanFragptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+ if (scanFragptr.p->scanFragNodeId == failedNodeId){
+ switch (scanFragptr.p->scanFragState){
+ case ScanFragRec::LQH_ACTIVE:
+ case ScanFragRec::LQH_ACTIVE_CLOSE:
+ jam();
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize,
+ apiConnectRecord);
+
+ // The connection to this LQH is closed
+ scanFragptr.p->lqhBlockref = RNIL;
+
+ DEBUG("checkScanActiveInFailedLqh: scanFragError");
+ scanFragError(signal, ZSCAN_LQH_ERROR);
+
+ break;
+
+ default:
+ /* empty */
+ jam();
+ break;
+ }// switch
+
+ } //if
+ } //if
+ } //for
+ } //if
+
+ // Send CONTINUEB to continue later
+ signal->theData[0] = TcContinueB::ZCHECK_SCAN_ACTIVE_FAILED_LQH;
+ signal->theData[1] = scanptr.i + 1; // Check next scanptr
+ signal->theData[2] = failedNodeId;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ return;
+ }//for
+}
+
+void Dbtc::execTAKE_OVERTCCONF(Signal* signal)
+{
+ jamEntry();
+ tfailedNodeId = signal->theData[0];
+ hostptr.i = tfailedNodeId;
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ switch (hostptr.p->takeOverStatus) {
+ case TOS_IDLE:
+ jam();
+ /*------------------------------------------------------------*/
+ /* THIS MESSAGE ARRIVED EVEN BEFORE THE NODE_FAILREP */
+ /* MESSAGE. THIS IS POSSIBLE IN EXTREME SITUATIONS. */
+ /* WE SET THE STATE TO TAKE_OVER_COMPLETED AND WAIT */
+ /* FOR THE NODE_FAILREP MESSAGE. */
+ /*------------------------------------------------------------*/
+ hostptr.p->takeOverStatus = TOS_COMPLETED;
+ break;
+ case TOS_NODE_FAILED:
+ case TOS_ACTIVE:
+ jam();
+ /*------------------------------------------------------------*/
+ /* WE ARE NOT MASTER AND THE TAKE OVER IS ACTIVE OR WE */
+ /* ARE MASTER AND THE TAKE OVER IS ACTIVE. IN BOTH */
+ /* WE SET THE STATE TO TAKE_OVER_COMPLETED. */
+ /*------------------------------------------------------------*/
+ /* RELEASE THE CURRENTLY UNUSED LQH CONNECTIONS. THE */
+ /* REMAINING WILL BE RELEASED WHEN THE TRANSACTION THAT */
+ /* USED THEM IS COMPLETED. */
+ /*------------------------------------------------------------*/
+ hostptr.p->takeOverStatus = TOS_COMPLETED;
+ {
+ NFCompleteRep * const nfRep = (NFCompleteRep *)&signal->theData[0];
+ nfRep->blockNo = DBTC;
+ nfRep->nodeId = cownNodeid;
+ nfRep->failedNodeId = hostptr.i;
+ }
+ sendSignal(cdihblockref, GSN_NF_COMPLETEREP, signal,
+ NFCompleteRep::SignalLength, JBB);
+ break;
+ case TOS_COMPLETED:
+ jam();
+ /*------------------------------------------------------------*/
+ /* WE HAVE ALREADY RECEIVED THE CONF SIGNAL. IT IS MOST */
+ /* LIKELY SENT FROM A NEW MASTER WHICH WASN'T SURE IF */
+ /* THIS NODE HEARD THE CONF SIGNAL FROM THE OLD MASTER. */
+ /* WE SIMPLY IGNORE THE MESSAGE. */
+ /*------------------------------------------------------------*/
+ /*empty*/;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//switch
+}//Dbtc::execTAKE_OVERTCCONF()
+
+void Dbtc::execTAKE_OVERTCREQ(Signal* signal)
+{
+ jamEntry();
+ tfailedNodeId = signal->theData[0];
+ tcNodeFailptr.i = 0;
+ ptrAss(tcNodeFailptr, tcFailRecord);
+ if (tcNodeFailptr.p->failStatus != FS_IDLE) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* WE CAN CURRENTLY ONLY HANDLE ONE TAKE OVER AT A TIME */
+ /*------------------------------------------------------------*/
+ /* IF MORE THAN ONE TAKE OVER IS REQUESTED WE WILL */
+ /* QUEUE THE TAKE OVER AND START IT AS SOON AS THE */
+ /* PREVIOUS ARE COMPLETED. */
+ /*------------------------------------------------------------*/
+ arrGuard(tcNodeFailptr.p->queueIndex, MAX_NDB_NODES);
+ tcNodeFailptr.p->queueList[tcNodeFailptr.p->queueIndex] = tfailedNodeId;
+ tcNodeFailptr.p->queueIndex = tcNodeFailptr.p->queueIndex + 1;
+ return;
+ }//if
+ startTakeOverLab(signal);
+}//Dbtc::execTAKE_OVERTCREQ()
+
+/*------------------------------------------------------------*/
+/* INITIALISE THE HASH TABLES FOR STORING TRANSACTIONS */
+/* AND OPERATIONS DURING TC TAKE OVER. */
+/*------------------------------------------------------------*/
+void Dbtc::startTakeOverLab(Signal* signal)
+{
+ for (tindex = 0; tindex <= 511; tindex++) {
+ ctransidFailHash[tindex] = RNIL;
+ }//for
+ for (tindex = 0; tindex <= 1023; tindex++) {
+ ctcConnectFailHash[tindex] = RNIL;
+ }//for
+ tcNodeFailptr.p->failStatus = FS_LISTENING;
+ tcNodeFailptr.p->takeOverNode = tfailedNodeId;
+ for (hostptr.i = 1; hostptr.i < MAX_NDB_NODES; hostptr.i++) {
+ jam();
+ ptrAss(hostptr, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ jam();
+ tblockref = calcLqhBlockRef(hostptr.i);
+ hostptr.p->lqhTransStatus = LTS_ACTIVE;
+ signal->theData[0] = tcNodeFailptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = tfailedNodeId;
+ sendSignal(tblockref, GSN_LQH_TRANSREQ, signal, 3, JBB);
+ }//if
+ }//for
+}//Dbtc::startTakeOverLab()
+
+/*------------------------------------------------------------*/
+/* A REPORT OF AN OPERATION WHERE TC FAILED HAS ARRIVED.*/
+/*------------------------------------------------------------*/
+void Dbtc::execLQH_TRANSCONF(Signal* signal)
+{
+ jamEntry();
+ LqhTransConf * const lqhTransConf = (LqhTransConf *)&signal->theData[0];
+
+ tcNodeFailptr.i = lqhTransConf->tcRef;
+ ptrCheckGuard(tcNodeFailptr, 1, tcFailRecord);
+ tnodeid = lqhTransConf->lqhNodeId;
+ ttransStatus = (LqhTransConf::OperationStatus)lqhTransConf->operationStatus;
+ ttransid1 = lqhTransConf->transId1;
+ ttransid2 = lqhTransConf->transId2;
+ ttcOprec = lqhTransConf->oldTcOpRec;
+ treqinfo = lqhTransConf->requestInfo;
+ tgci = lqhTransConf->gci;
+ cnodes[0] = lqhTransConf->nextNodeId1;
+ cnodes[1] = lqhTransConf->nextNodeId2;
+ cnodes[2] = lqhTransConf->nextNodeId3;
+ const Uint32 ref = tapplRef = lqhTransConf->apiRef;
+ tapplOprec = lqhTransConf->apiOpRec;
+ const Uint32 tableId = lqhTransConf->tableId;
+
+ if (ttransStatus == LqhTransConf::LastTransConf){
+ jam();
+ /*------------------------------------------------------------*/
+ /* A NODE HAS REPORTED COMPLETION OF TAKE OVER REPORTING*/
+ /*------------------------------------------------------------*/
+ nodeTakeOverCompletedLab(signal);
+ return;
+ }//if
+ if (ttransStatus == LqhTransConf::Marker){
+ jam();
+ treqinfo = 0;
+ LqhTransConf::setMarkerFlag(treqinfo, 1);
+ } else {
+ TableRecordPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, ctabrecFilesize, tableRecord);
+ switch((DictTabInfo::TableType)tabPtr.p->tableType){
+ case DictTabInfo::SystemTable:
+ case DictTabInfo::UserTable:
+ break;
+ default:
+ tapplRef = 0;
+ tapplOprec = 0;
+ }
+ }
+
+ findApiConnectFail(signal);
+
+ if(apiConnectptr.p->ndbapiBlockref == 0 && tapplRef != 0){
+ apiConnectptr.p->ndbapiBlockref = ref;
+ apiConnectptr.p->ndbapiConnect = tapplOprec;
+ }
+
+ if (ttransStatus != LqhTransConf::Marker){
+ jam();
+ findTcConnectFail(signal);
+ }
+}//Dbtc::execLQH_TRANSCONF()
+
+/*------------------------------------------------------------*/
+/* A NODE HAS REPORTED COMPLETION OF TAKE OVER REPORTING*/
+/*------------------------------------------------------------*/
+void Dbtc::nodeTakeOverCompletedLab(Signal* signal)
+{
+ Uint32 guard0;
+
+ hostptr.i = tnodeid;
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ hostptr.p->lqhTransStatus = LTS_IDLE;
+ for (hostptr.i = 1; hostptr.i < MAX_NDB_NODES; hostptr.i++) {
+ jam();
+ ptrAss(hostptr, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ if (hostptr.p->lqhTransStatus == LTS_ACTIVE) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* NOT ALL NODES ARE COMPLETED WITH REPORTING IN THE */
+ /* TAKE OVER. */
+ /*------------------------------------------------------------*/
+ return;
+ }//if
+ }//if
+ }//for
+ /*------------------------------------------------------------*/
+ /* ALL NODES HAVE REPORTED ON THE STATUS OF THE VARIOUS */
+ /* OPERATIONS THAT WAS CONTROLLED BY THE FAILED TC. WE */
+ /* ARE NOW IN A POSITION TO COMPLETE ALL OF THOSE */
+ /* TRANSACTIONS EITHER IN A SUCCESSFUL WAY OR IN AN */
+ /* UNSUCCESSFUL WAY. WE WILL ALSO REPORT THIS CONCLUSION*/
+ /* TO THE APPLICATION IF THAT IS STILL ALIVE. */
+ /*------------------------------------------------------------*/
+ tcNodeFailptr.p->currentHashIndexTakeOver = 0;
+ tcNodeFailptr.p->completedTakeOver = 0;
+ tcNodeFailptr.p->failStatus = FS_COMPLETING;
+ guard0 = cnoParallelTakeOver - 1;
+ /*------------------------------------------------------------*/
+ /* WE WILL COMPLETE THE TRANSACTIONS BY STARTING A */
+ /* NUMBER OF PARALLEL ACTIVITIES. EACH ACTIVITY WILL */
+ /* COMPLETE ONE TRANSACTION AT A TIME AND IN THAT */
+ /* TRANSACTION IT WILL COMPLETE ONE OPERATION AT A TIME.*/
+ /* WHEN ALL ACTIVITIES ARE COMPLETED THEN THE TAKE OVER */
+ /* IS COMPLETED. */
+ /*------------------------------------------------------------*/
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (tindex = 0; tindex <= guard0; tindex++) {
+ jam();
+ tcNodeFailptr.p->takeOverProcState[tindex] = ZTAKE_OVER_ACTIVE;
+ signal->theData[0] = TcContinueB::ZCOMPLETE_TRANS_AT_TAKE_OVER;
+ signal->theData[1] = tcNodeFailptr.i;
+ signal->theData[2] = tindex;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ }//for
+}//Dbtc::nodeTakeOverCompletedLab()
+
+/*------------------------------------------------------------*/
+/* COMPLETE A NEW TRANSACTION FROM THE HASH TABLE OF */
+/* TRANSACTIONS TO COMPLETE. */
+/*------------------------------------------------------------*/
+void Dbtc::completeTransAtTakeOverLab(Signal* signal, UintR TtakeOverInd)
+{
+ jam();
+ while (tcNodeFailptr.p->currentHashIndexTakeOver < 512){
+ jam();
+ apiConnectptr.i =
+ ctransidFailHash[tcNodeFailptr.p->currentHashIndexTakeOver];
+ if (apiConnectptr.i != RNIL) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* WE HAVE FOUND A TRANSACTION THAT NEEDS TO BE */
+ /* COMPLETED. REMOVE IT FROM THE HASH TABLE SUCH THAT */
+ /* NOT ANOTHER ACTIVITY ALSO TRIES TO COMPLETE THIS */
+ /* TRANSACTION. */
+ /*------------------------------------------------------------*/
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ ctransidFailHash[tcNodeFailptr.p->currentHashIndexTakeOver] =
+ apiConnectptr.p->nextApiConnect;
+
+ completeTransAtTakeOverDoOne(signal, TtakeOverInd);
+ // One transaction taken care of, return from this function
+ // and wait for the next CONTINUEB to continue processing
+ break;
+
+ } else {
+ if (tcNodeFailptr.p->currentHashIndexTakeOver < 511){
+ jam();
+ tcNodeFailptr.p->currentHashIndexTakeOver++;
+ } else {
+ jam();
+ completeTransAtTakeOverDoLast(signal, TtakeOverInd);
+ tcNodeFailptr.p->currentHashIndexTakeOver++;
+ }//if
+ }//if
+ }//while
+}//Dbtc::completeTransAtTakeOverLab()
+
+
+
+
+void Dbtc::completeTransAtTakeOverDoLast(Signal* signal, UintR TtakeOverInd)
+{
+ Uint32 guard0;
+ /*------------------------------------------------------------*/
+ /* THERE ARE NO MORE TRANSACTIONS TO COMPLETE. THIS */
+ /* ACTIVITY IS COMPLETED. */
+ /*------------------------------------------------------------*/
+ arrGuard(TtakeOverInd, MAX_NDB_NODES);
+ if (tcNodeFailptr.p->takeOverProcState[TtakeOverInd] != ZTAKE_OVER_ACTIVE) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ tcNodeFailptr.p->takeOverProcState[TtakeOverInd] = ZTAKE_OVER_IDLE;
+ tcNodeFailptr.p->completedTakeOver++;
+
+ if (tcNodeFailptr.p->completedTakeOver == cnoParallelTakeOver) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* WE WERE THE LAST ACTIVITY THAT WAS COMPLETED. WE NEED*/
+ /* TO REPORT THE COMPLETION OF THE TAKE OVER TO ALL */
+ /* NODES THAT ARE ALIVE. */
+ /*------------------------------------------------------------*/
+ for (hostptr.i = 1; hostptr.i < MAX_NDB_NODES; hostptr.i++) {
+ jam();
+ ptrAss(hostptr, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ jam();
+ tblockref = calcTcBlockRef(hostptr.i);
+ signal->theData[0] = tcNodeFailptr.p->takeOverNode;
+ sendSignal(tblockref, GSN_TAKE_OVERTCCONF, signal, 1, JBB);
+ }//if
+ }//for
+ if (tcNodeFailptr.p->queueIndex > 0) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* THERE ARE MORE NODES TO TAKE OVER. WE NEED TO START */
+ /* THE TAKE OVER. */
+ /*------------------------------------------------------------*/
+ tfailedNodeId = tcNodeFailptr.p->queueList[0];
+ guard0 = tcNodeFailptr.p->queueIndex - 1;
+ arrGuard(guard0 + 1, MAX_NDB_NODES);
+ for (tindex = 0; tindex <= guard0; tindex++) {
+ jam();
+ tcNodeFailptr.p->queueList[tindex] =
+ tcNodeFailptr.p->queueList[tindex + 1];
+ }//for
+ tcNodeFailptr.p->queueIndex--;
+ startTakeOverLab(signal);
+ return;
+ } else {
+ jam();
+ tcNodeFailptr.p->failStatus = FS_IDLE;
+ }//if
+ }//if
+ return;
+}//Dbtc::completeTransAtTakeOverDoLast()
+
+void Dbtc::completeTransAtTakeOverDoOne(Signal* signal, UintR TtakeOverInd)
+{
+ apiConnectptr.p->takeOverRec = (Uint8)tcNodeFailptr.i;
+ apiConnectptr.p->takeOverInd = TtakeOverInd;
+
+ switch (apiConnectptr.p->apiConnectstate) {
+ case CS_FAIL_COMMITTED:
+ jam();
+ /*------------------------------------------------------------*/
+ /* ALL PARTS OF THE TRANSACTIONS REPORTED COMMITTED. WE */
+ /* HAVE THUS COMPLETED THE COMMIT PHASE. WE CAN REPORT */
+ /* COMMITTED TO THE APPLICATION AND CONTINUE WITH THE */
+ /* COMPLETE PHASE. */
+ /*------------------------------------------------------------*/
+ sendTCKEY_FAILCONF(signal, apiConnectptr.p);
+ tcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ apiConnectptr.p->currentTcConnect = tcConnectptr.i;
+ apiConnectptr.p->currentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ tcurrentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ toCompleteHandlingLab(signal);
+ return;
+ case CS_FAIL_COMMITTING:
+ jam();
+ /*------------------------------------------------------------*/
+ /* AT LEAST ONE PART WAS ONLY PREPARED AND AT LEAST ONE */
+ /* PART WAS COMMITTED. COMPLETE THE COMMIT PHASE FIRST. */
+ /* THEN CONTINUE AS AFTER COMMITTED. */
+ /*------------------------------------------------------------*/
+ tcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ apiConnectptr.p->currentTcConnect = tcConnectptr.i;
+ apiConnectptr.p->currentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ tcurrentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ toCommitHandlingLab(signal);
+ return;
+ case CS_FAIL_ABORTING:
+ case CS_FAIL_PREPARED:
+ jam();
+ /*------------------------------------------------------------*/
+ /* WE WILL ABORT THE TRANSACTION IF IT IS IN A PREPARED */
+ /* STATE IN THIS VERSION. IN LATER VERSIONS WE WILL */
+ /* HAVE TO ADD CODE FOR HANDLING OF PREPARED-TO-COMMIT */
+ /* TRANSACTIONS. THESE ARE NOT ALLOWED TO ABORT UNTIL WE*/
+ /* HAVE HEARD FROM THE TRANSACTION COORDINATOR. */
+ /* */
+ /* IT IS POSSIBLE TO COMMIT TRANSACTIONS THAT ARE */
+ /* PREPARED ACTUALLY. WE WILL LEAVE THIS PROBLEM UNTIL */
+ /* LATER VERSIONS. */
+ /*------------------------------------------------------------*/
+ tcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ apiConnectptr.p->currentTcConnect = tcConnectptr.i;
+ apiConnectptr.p->currentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ tcurrentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ toAbortHandlingLab(signal);
+ return;
+ case CS_FAIL_ABORTED:
+ jam();
+ sendTCKEY_FAILREF(signal, apiConnectptr.p);
+
+ signal->theData[0] = TcContinueB::ZCOMPLETE_TRANS_AT_TAKE_OVER;
+ signal->theData[1] = (UintR)apiConnectptr.p->takeOverRec;
+ signal->theData[2] = apiConnectptr.p->takeOverInd;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ releaseTakeOver(signal);
+ break;
+ case CS_FAIL_COMPLETED:
+ jam();
+ sendTCKEY_FAILCONF(signal, apiConnectptr.p);
+
+ signal->theData[0] = TcContinueB::ZCOMPLETE_TRANS_AT_TAKE_OVER;
+ signal->theData[1] = (UintR)apiConnectptr.p->takeOverRec;
+ signal->theData[2] = apiConnectptr.p->takeOverInd;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ releaseApiConnectFail(signal);
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//switch
+}//Dbtc::completeTransAtTakeOverDoOne()
+
+void
+Dbtc::sendTCKEY_FAILREF(Signal* signal, const ApiConnectRecord * regApiPtr){
+ jam();
+
+ const Uint32 ref = regApiPtr->ndbapiBlockref;
+ if(ref != 0){
+ signal->theData[0] = regApiPtr->ndbapiConnect;
+ signal->theData[1] = regApiPtr->transid[0];
+ signal->theData[2] = regApiPtr->transid[1];
+
+ sendSignal(ref, GSN_TCKEY_FAILREF, signal, 3, JBB);
+ }
+}
+
+void
+Dbtc::sendTCKEY_FAILCONF(Signal* signal, const ApiConnectRecord * regApiPtr){
+ jam();
+ TcKeyFailConf * const failConf = (TcKeyFailConf *)&signal->theData[0];
+
+ if(regApiPtr->commitAckMarker == RNIL){
+ jam();
+ failConf->apiConnectPtr = regApiPtr->ndbapiConnect;
+ } else {
+ jam();
+ failConf->apiConnectPtr = regApiPtr->ndbapiConnect | 1;
+ }
+ failConf->transId1 = regApiPtr->transid[0];
+ failConf->transId2 = regApiPtr->transid[1];
+
+ sendSignal(regApiPtr->ndbapiBlockref,
+ GSN_TCKEY_FAILCONF, signal, TcKeyFailConf::SignalLength, JBB);
+}
+
+
+/*------------------------------------------------------------*/
+/* THIS PART HANDLES THE ABORT PHASE IN THE CASE OF A */
+/* NODE FAILURE BEFORE THE COMMIT DECISION. */
+/*------------------------------------------------------------*/
+/* ABORT REQUEST SUCCESSFULLY COMPLETED ON TNODEID */
+/*------------------------------------------------------------*/
+void Dbtc::execABORTCONF(Signal* signal)
+{
+ UintR compare_transid1, compare_transid2;
+
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ tnodeid = signal->theData[2];
+ if (ERROR_INSERTED(8045)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_ABORTCONF, signal, 2000, 5);
+ return;
+ }//if
+ if (tcConnectptr.i >= ctcConnectFilesize) {
+ errorReport(signal, 5);
+ return;
+ }//if
+ ptrAss(tcConnectptr, tcConnectRecord);
+ if (tcConnectptr.p->tcConnectstate != OS_WAIT_ABORT_CONF) {
+ warningReport(signal, 16);
+ return;
+ }//if
+ apiConnectptr.i = tcConnectptr.p->apiConnect;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ if (apiConnectptr.p->apiConnectstate != CS_WAIT_ABORT_CONF) {
+ warningReport(signal, 17);
+ return;
+ }//if
+ compare_transid1 = apiConnectptr.p->transid[0] ^ signal->theData[3];
+ compare_transid2 = apiConnectptr.p->transid[1] ^ signal->theData[4];
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ warningReport(signal, 18);
+ return;
+ }//if
+ arrGuard(apiConnectptr.p->currentReplicaNo, 4);
+ if (tcConnectptr.p->tcNodedata[apiConnectptr.p->currentReplicaNo] !=
+ tnodeid) {
+ warningReport(signal, 19);
+ return;
+ }//if
+ tcurrentReplicaNo = (Uint8)Z8NIL;
+ tcConnectptr.p->tcConnectstate = OS_ABORTING;
+ toAbortHandlingLab(signal);
+}//Dbtc::execABORTCONF()
+
+void Dbtc::toAbortHandlingLab(Signal* signal)
+{
+ do {
+ if (tcurrentReplicaNo != (Uint8)Z8NIL) {
+ jam();
+ arrGuard(tcurrentReplicaNo, 4);
+ const LqhTransConf::OperationStatus stat =
+ (LqhTransConf::OperationStatus)
+ tcConnectptr.p->failData[tcurrentReplicaNo];
+ switch(stat){
+ case LqhTransConf::InvalidStatus:
+ case LqhTransConf::Aborted:
+ jam();
+ /*empty*/;
+ break;
+ case LqhTransConf::Prepared:
+ jam();
+ hostptr.i = tcConnectptr.p->tcNodedata[tcurrentReplicaNo];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ jam();
+ tblockref = calcLqhBlockRef(hostptr.i);
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ tcConnectptr.p->tcConnectstate = OS_WAIT_ABORT_CONF;
+ apiConnectptr.p->apiConnectstate = CS_WAIT_ABORT_CONF;
+ apiConnectptr.p->timeOutCounter = 0;
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = apiConnectptr.p->transid[0];
+ signal->theData[3] = apiConnectptr.p->transid[1];
+ signal->theData[4] = apiConnectptr.p->tcBlockref;
+ signal->theData[5] = tcConnectptr.p->tcOprec;
+ sendSignal(tblockref, GSN_ABORTREQ, signal, 6, JBB);
+ return;
+ }//if
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//switch
+ }//if
+ if (apiConnectptr.p->currentReplicaNo > 0) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* THERE IS STILL ANOTHER REPLICA THAT NEEDS TO BE */
+ /* ABORTED. */
+ /*------------------------------------------------------------*/
+ apiConnectptr.p->currentReplicaNo--;
+ tcurrentReplicaNo = apiConnectptr.p->currentReplicaNo;
+ } else {
+ /*------------------------------------------------------------*/
+ /* THE LAST REPLICA IN THIS OPERATION HAVE COMMITTED. */
+ /*------------------------------------------------------------*/
+ tcConnectptr.i = tcConnectptr.p->nextTcConnect;
+ if (tcConnectptr.i == RNIL) {
+ /*------------------------------------------------------------*/
+ /* WE HAVE COMPLETED THE ABORT PHASE. WE CAN NOW REPORT */
+ /* THE ABORT STATUS TO THE APPLICATION AND CONTINUE */
+ /* WITH THE NEXT TRANSACTION. */
+ /*------------------------------------------------------------*/
+ if (apiConnectptr.p->takeOverRec != (Uint8)Z8NIL) {
+ jam();
+ sendTCKEY_FAILREF(signal, apiConnectptr.p);
+ const Uint32 marker = apiConnectptr.p->commitAckMarker;
+ if(marker != RNIL){
+ jam();
+
+ CommitAckMarkerPtr tmp;
+ tmp.i = marker;
+ tmp.p = m_commitAckMarkerHash.getPtr(tmp.i);
+
+ m_commitAckMarkerHash.release(tmp);
+ apiConnectptr.p->commitAckMarker = RNIL;
+ }
+
+ /*------------------------------------------------------------*/
+ /* WE HAVE COMPLETED THIS TRANSACTION NOW AND CAN */
+ /* CONTINUE THE PROCESS WITH THE NEXT TRANSACTION. */
+ /*------------------------------------------------------------*/
+ signal->theData[0] = TcContinueB::ZCOMPLETE_TRANS_AT_TAKE_OVER;
+ signal->theData[1] = (UintR)apiConnectptr.p->takeOverRec;
+ signal->theData[2] = apiConnectptr.p->takeOverInd;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ releaseTakeOver(signal);
+ } else {
+ jam();
+ releaseAbortResources(signal);
+ }//if
+ return;
+ }//if
+ apiConnectptr.p->currentTcConnect = tcConnectptr.i;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ apiConnectptr.p->currentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ tcurrentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ }//if
+ } while (1);
+}//Dbtc::toAbortHandlingLab()
+
+/*------------------------------------------------------------*/
+/* THIS PART HANDLES THE COMMIT PHASE IN THE CASE OF A */
+/* NODE FAILURE IN THE MIDDLE OF THE COMMIT PHASE. */
+/*------------------------------------------------------------*/
+/* COMMIT REQUEST SUCCESSFULLY COMPLETED ON TNODEID */
+/*------------------------------------------------------------*/
+void Dbtc::execCOMMITCONF(Signal* signal)
+{
+ UintR compare_transid1, compare_transid2;
+
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ tnodeid = signal->theData[1];
+ if (ERROR_INSERTED(8046)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_COMMITCONF, signal, 2000, 4);
+ return;
+ }//if
+ if (tcConnectptr.i >= ctcConnectFilesize) {
+ errorReport(signal, 4);
+ return;
+ }//if
+ ptrAss(tcConnectptr, tcConnectRecord);
+ if (tcConnectptr.p->tcConnectstate != OS_WAIT_COMMIT_CONF) {
+ warningReport(signal, 8);
+ return;
+ }//if
+ apiConnectptr.i = tcConnectptr.p->apiConnect;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ if (apiConnectptr.p->apiConnectstate != CS_WAIT_COMMIT_CONF) {
+ warningReport(signal, 9);
+ return;
+ }//if
+ compare_transid1 = apiConnectptr.p->transid[0] ^ signal->theData[2];
+ compare_transid2 = apiConnectptr.p->transid[1] ^ signal->theData[3];
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ warningReport(signal, 10);
+ return;
+ }//if
+ arrGuard(apiConnectptr.p->currentReplicaNo, 4);
+ if (tcConnectptr.p->tcNodedata[apiConnectptr.p->currentReplicaNo] !=
+ tnodeid) {
+ warningReport(signal, 11);
+ return;
+ }//if
+ if (ERROR_INSERTED(8026)) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ tcurrentReplicaNo = (Uint8)Z8NIL;
+ tcConnectptr.p->tcConnectstate = OS_COMMITTED;
+ toCommitHandlingLab(signal);
+}//Dbtc::execCOMMITCONF()
+
+void Dbtc::toCommitHandlingLab(Signal* signal)
+{
+ do {
+ if (tcurrentReplicaNo != (Uint8)Z8NIL) {
+ jam();
+ arrGuard(tcurrentReplicaNo, 4);
+ switch (tcConnectptr.p->failData[tcurrentReplicaNo]) {
+ case LqhTransConf::InvalidStatus:
+ jam();
+ /*empty*/;
+ break;
+ case LqhTransConf::Committed:
+ jam();
+ /*empty*/;
+ break;
+ case LqhTransConf::Prepared:
+ jam();
+ /*------------------------------------------------------------*/
+ /* THE NODE WAS PREPARED AND IS WAITING FOR ABORT OR */
+ /* COMMIT REQUEST FROM TC. */
+ /*------------------------------------------------------------*/
+ hostptr.i = tcConnectptr.p->tcNodedata[tcurrentReplicaNo];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ jam();
+ tblockref = calcLqhBlockRef(hostptr.i);
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ apiConnectptr.p->apiConnectstate = CS_WAIT_COMMIT_CONF;
+ apiConnectptr.p->timeOutCounter = 0;
+ tcConnectptr.p->tcConnectstate = OS_WAIT_COMMIT_CONF;
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = apiConnectptr.p->globalcheckpointid;
+ signal->theData[3] = apiConnectptr.p->transid[0];
+ signal->theData[4] = apiConnectptr.p->transid[1];
+ signal->theData[5] = apiConnectptr.p->tcBlockref;
+ signal->theData[6] = tcConnectptr.p->tcOprec;
+ sendSignal(tblockref, GSN_COMMITREQ, signal, 7, JBB);
+ return;
+ }//if
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ }//if
+ if (apiConnectptr.p->currentReplicaNo > 0) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* THERE IS STILL ANOTHER REPLICA THAT NEEDS TO BE */
+ /* COMMITTED. */
+ /*------------------------------------------------------------*/
+ apiConnectptr.p->currentReplicaNo--;
+ tcurrentReplicaNo = apiConnectptr.p->currentReplicaNo;
+ } else {
+ /*------------------------------------------------------------*/
+ /* THE LAST REPLICA IN THIS OPERATION HAVE COMMITTED. */
+ /*------------------------------------------------------------*/
+ tcConnectptr.i = tcConnectptr.p->nextTcConnect;
+ if (tcConnectptr.i == RNIL) {
+ /*------------------------------------------------------------*/
+ /* WE HAVE COMPLETED THE COMMIT PHASE. WE CAN NOW REPORT*/
+ /* THE COMMIT STATUS TO THE APPLICATION AND CONTINUE */
+ /* WITH THE COMPLETE PHASE. */
+ /*------------------------------------------------------------*/
+ if (apiConnectptr.p->takeOverRec != (Uint8)Z8NIL) {
+ jam();
+ sendTCKEY_FAILCONF(signal, apiConnectptr.p);
+ } else {
+ jam();
+ sendApiCommit(signal);
+ }//if
+ apiConnectptr.p->currentTcConnect = apiConnectptr.p->firstTcConnect;
+ tcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ tcurrentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ apiConnectptr.p->currentReplicaNo = tcurrentReplicaNo;
+ toCompleteHandlingLab(signal);
+ return;
+ }//if
+ apiConnectptr.p->currentTcConnect = tcConnectptr.i;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ apiConnectptr.p->currentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ tcurrentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ }//if
+ } while (1);
+}//Dbtc::toCommitHandlingLab()
+
+/*------------------------------------------------------------*/
+/* COMMON PART TO HANDLE COMPLETE PHASE WHEN ANY NODE */
+/* HAVE FAILED. */
+/*------------------------------------------------------------*/
+/* THE NODE WITH TNODEID HAVE COMPLETED THE OPERATION */
+/*------------------------------------------------------------*/
+void Dbtc::execCOMPLETECONF(Signal* signal)
+{
+ UintR compare_transid1, compare_transid2;
+
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ tnodeid = signal->theData[1];
+ if (ERROR_INSERTED(8047)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ sendSignalWithDelay(cownref, GSN_COMPLETECONF, signal, 2000, 4);
+ return;
+ }//if
+ if (tcConnectptr.i >= ctcConnectFilesize) {
+ errorReport(signal, 3);
+ return;
+ }//if
+ ptrAss(tcConnectptr, tcConnectRecord);
+ if (tcConnectptr.p->tcConnectstate != OS_WAIT_COMPLETE_CONF) {
+ warningReport(signal, 12);
+ return;
+ }//if
+ apiConnectptr.i = tcConnectptr.p->apiConnect;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ if (apiConnectptr.p->apiConnectstate != CS_WAIT_COMPLETE_CONF) {
+ warningReport(signal, 13);
+ return;
+ }//if
+ compare_transid1 = apiConnectptr.p->transid[0] ^ signal->theData[2];
+ compare_transid2 = apiConnectptr.p->transid[1] ^ signal->theData[3];
+ compare_transid1 = compare_transid1 | compare_transid2;
+ if (compare_transid1 != 0) {
+ warningReport(signal, 14);
+ return;
+ }//if
+ arrGuard(apiConnectptr.p->currentReplicaNo, 4);
+ if (tcConnectptr.p->tcNodedata[apiConnectptr.p->currentReplicaNo] !=
+ tnodeid) {
+ warningReport(signal, 15);
+ return;
+ }//if
+ if (ERROR_INSERTED(8028)) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ tcConnectptr.p->tcConnectstate = OS_COMPLETED;
+ tcurrentReplicaNo = (Uint8)Z8NIL;
+ toCompleteHandlingLab(signal);
+}//Dbtc::execCOMPLETECONF()
+
+void Dbtc::toCompleteHandlingLab(Signal* signal)
+{
+ do {
+ if (tcurrentReplicaNo != (Uint8)Z8NIL) {
+ jam();
+ arrGuard(tcurrentReplicaNo, 4);
+ switch (tcConnectptr.p->failData[tcurrentReplicaNo]) {
+ case LqhTransConf::InvalidStatus:
+ jam();
+ /*empty*/;
+ break;
+ default:
+ jam();
+ /*------------------------------------------------------------*/
+ /* THIS NODE DID NOT REPORT ANYTHING FOR THIS OPERATION */
+ /* IT MUST HAVE FAILED. */
+ /*------------------------------------------------------------*/
+ /*------------------------------------------------------------*/
+ /* SEND COMPLETEREQ TO THE NEXT REPLICA. */
+ /*------------------------------------------------------------*/
+ hostptr.i = tcConnectptr.p->tcNodedata[tcurrentReplicaNo];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ if (hostptr.p->hostStatus == HS_ALIVE) {
+ jam();
+ tblockref = calcLqhBlockRef(hostptr.i);
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ tcConnectptr.p->tcConnectstate = OS_WAIT_COMPLETE_CONF;
+ apiConnectptr.p->apiConnectstate = CS_WAIT_COMPLETE_CONF;
+ apiConnectptr.p->timeOutCounter = 0;
+ tcConnectptr.p->apiConnect = apiConnectptr.i;
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = cownref;
+ signal->theData[2] = apiConnectptr.p->transid[0];
+ signal->theData[3] = apiConnectptr.p->transid[1];
+ signal->theData[4] = apiConnectptr.p->tcBlockref;
+ signal->theData[5] = tcConnectptr.p->tcOprec;
+ sendSignal(tblockref, GSN_COMPLETEREQ, signal, 6, JBB);
+ return;
+ }//if
+ break;
+ }//switch
+ }//if
+ if (apiConnectptr.p->currentReplicaNo != 0) {
+ jam();
+ /*------------------------------------------------------------*/
+ /* THERE ARE STILL MORE REPLICAS IN THIS OPERATION. WE */
+ /* NEED TO CONTINUE WITH THOSE REPLICAS. */
+ /*------------------------------------------------------------*/
+ apiConnectptr.p->currentReplicaNo--;
+ tcurrentReplicaNo = apiConnectptr.p->currentReplicaNo;
+ } else {
+ tcConnectptr.i = tcConnectptr.p->nextTcConnect;
+ if (tcConnectptr.i == RNIL) {
+ /*------------------------------------------------------------*/
+ /* WE HAVE COMPLETED THIS TRANSACTION NOW AND CAN */
+ /* CONTINUE THE PROCESS WITH THE NEXT TRANSACTION. */
+ /*------------------------------------------------------------*/
+ if (apiConnectptr.p->takeOverRec != (Uint8)Z8NIL) {
+ jam();
+ signal->theData[0] = TcContinueB::ZCOMPLETE_TRANS_AT_TAKE_OVER;
+ signal->theData[1] = (UintR)apiConnectptr.p->takeOverRec;
+ signal->theData[2] = apiConnectptr.p->takeOverInd;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ releaseTakeOver(signal);
+ } else {
+ jam();
+ releaseTransResources(signal);
+ }//if
+ return;
+ }//if
+ /*------------------------------------------------------------*/
+ /* WE HAVE COMPLETED AN OPERATION AND THERE ARE MORE TO */
+ /* COMPLETE. TAKE THE NEXT OPERATION AND START WITH THE */
+ /* FIRST REPLICA SINCE IT IS THE COMPLETE PHASE. */
+ /*------------------------------------------------------------*/
+ apiConnectptr.p->currentTcConnect = tcConnectptr.i;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ tcurrentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ apiConnectptr.p->currentReplicaNo = tcurrentReplicaNo;
+ }//if
+ } while (1);
+}//Dbtc::toCompleteHandlingLab()
+
+/*------------------------------------------------------------*/
+/* */
+/* FIND THE API CONNECT RECORD FOR THIS TRANSACTION */
+/* DURING TAKE OVER FROM A FAILED TC. IF NONE EXISTS */
+/* YET THEN SEIZE A NEW API CONNECT RECORD AND LINK IT */
+/* INTO THE HASH TABLE. */
+/*------------------------------------------------------------*/
+void Dbtc::findApiConnectFail(Signal* signal)
+{
+ ApiConnectRecordPtr fafPrevApiConnectptr;
+ ApiConnectRecordPtr fafNextApiConnectptr;
+ UintR tfafHashNumber;
+
+ tfafHashNumber = ttransid1 & 511;
+ fafPrevApiConnectptr.i = RNIL;
+ ptrNull(fafPrevApiConnectptr);
+ arrGuard(tfafHashNumber, 512);
+ fafNextApiConnectptr.i = ctransidFailHash[tfafHashNumber];
+ ptrCheck(fafNextApiConnectptr, capiConnectFilesize, apiConnectRecord);
+FAF_LOOP:
+ jam();
+ if (fafNextApiConnectptr.i == RNIL) {
+ jam();
+ if (cfirstfreeApiConnectFail == RNIL) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ seizeApiConnectFail(signal);
+ if (fafPrevApiConnectptr.i == RNIL) {
+ jam();
+ ctransidFailHash[tfafHashNumber] = apiConnectptr.i;
+ } else {
+ jam();
+ ptrGuard(fafPrevApiConnectptr);
+ fafPrevApiConnectptr.p->nextApiConnect = apiConnectptr.i;
+ }//if
+ apiConnectptr.p->nextApiConnect = RNIL;
+ initApiConnectFail(signal);
+ } else {
+ jam();
+ fafPrevApiConnectptr.i = fafNextApiConnectptr.i;
+ fafPrevApiConnectptr.p = fafNextApiConnectptr.p;
+ apiConnectptr.i = fafNextApiConnectptr.i;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ fafNextApiConnectptr.i = apiConnectptr.p->nextApiConnect;
+ ptrCheck(fafNextApiConnectptr, capiConnectFilesize, apiConnectRecord);
+ if ((apiConnectptr.p->transid[1] != ttransid2) ||
+ (apiConnectptr.p->transid[0] != ttransid1)) {
+ goto FAF_LOOP;
+ }//if
+ updateApiStateFail(signal);
+ }//if
+}//Dbtc::findApiConnectFail()
+
+/*----------------------------------------------------------*/
+/* FIND THE TC CONNECT AND IF NOT FOUND ALLOCATE A NEW */
+/*----------------------------------------------------------*/
+void Dbtc::findTcConnectFail(Signal* signal)
+{
+ UintR tftfHashNumber;
+
+ tftfHashNumber = (ttransid1 ^ ttcOprec) & 1023;
+ tcConnectptr.i = ctcConnectFailHash[tftfHashNumber];
+ do {
+ if (tcConnectptr.i == RNIL) {
+ jam();
+ if (cfirstfreeTcConnectFail == RNIL) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ seizeTcConnectFail(signal);
+ linkTcInConnectionlist(signal);
+ tcConnectptr.p->nextTcFailHash = ctcConnectFailHash[tftfHashNumber];
+ ctcConnectFailHash[tftfHashNumber] = tcConnectptr.i;
+ initTcConnectFail(signal);
+ return;
+ } else {
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ if (tcConnectptr.p->tcOprec != ttcOprec) {
+ jam(); /* FRAGMENTID = TC_OPREC HERE, LOOP ANOTHER TURN */
+ tcConnectptr.i = tcConnectptr.p->nextTcFailHash;
+ } else {
+ updateTcStateFail(signal);
+ return;
+ }//if
+ }//if
+ } while (1);
+}//Dbtc::findTcConnectFail()
+
+/*----------------------------------------------------------*/
+/* INITIALISE AN API CONNECT FAIL RECORD */
+/*----------------------------------------------------------*/
+void Dbtc::initApiConnectFail(Signal* signal)
+{
+ apiConnectptr.p->transid[0] = ttransid1;
+ apiConnectptr.p->transid[1] = ttransid2;
+ apiConnectptr.p->firstTcConnect = RNIL;
+ apiConnectptr.p->currSavePointId = 0;
+ apiConnectptr.p->lastTcConnect = RNIL;
+ tblockref = calcTcBlockRef(tcNodeFailptr.p->takeOverNode);
+
+ apiConnectptr.p->tcBlockref = tblockref;
+ apiConnectptr.p->ndbapiBlockref = tapplRef;
+ apiConnectptr.p->ndbapiConnect = tapplOprec;
+ apiConnectptr.p->buddyPtr = RNIL;
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ switch(ttransStatus){
+ case LqhTransConf::Committed:
+ jam();
+ apiConnectptr.p->globalcheckpointid = tgci;
+ apiConnectptr.p->apiConnectstate = CS_FAIL_COMMITTED;
+ break;
+ case LqhTransConf::Prepared:
+ jam();
+ apiConnectptr.p->apiConnectstate = CS_FAIL_PREPARED;
+ break;
+ case LqhTransConf::Aborted:
+ jam();
+ apiConnectptr.p->apiConnectstate = CS_FAIL_ABORTED;
+ break;
+ case LqhTransConf::Marker:
+ jam();
+ apiConnectptr.p->apiConnectstate = CS_FAIL_COMPLETED;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ }//if
+ apiConnectptr.p->commitAckMarker = RNIL;
+ if(LqhTransConf::getMarkerFlag(treqinfo)){
+ jam();
+ CommitAckMarkerPtr tmp;
+ m_commitAckMarkerHash.seize(tmp);
+
+ ndbrequire(tmp.i != RNIL);
+
+ apiConnectptr.p->commitAckMarker = tmp.i;
+ tmp.p->transid1 = ttransid1;
+ tmp.p->transid2 = ttransid2;
+ tmp.p->apiNodeId = refToNode(tapplRef);
+ tmp.p->noOfLqhs = 1;
+ tmp.p->lqhNodeId[0] = tnodeid;
+ tmp.p->apiConnectPtr = apiConnectptr.i;
+ m_commitAckMarkerHash.add(tmp);
+ }
+}//Dbtc::initApiConnectFail()
+
+/*------------------------------------------------------------*/
+/* INITIALISE AT TC CONNECT AT TAKE OVER WHEN ALLOCATING*/
+/* THE TC CONNECT RECORD. */
+/*------------------------------------------------------------*/
+void Dbtc::initTcConnectFail(Signal* signal)
+{
+ tcConnectptr.p->apiConnect = apiConnectptr.i;
+ tcConnectptr.p->tcOprec = ttcOprec;
+ Uint32 treplicaNo = LqhTransConf::getReplicaNo(treqinfo);
+ for (Uint32 i = 0; i < MAX_REPLICAS; i++) {
+ tcConnectptr.p->failData[i] = LqhTransConf::InvalidStatus;
+ }//for
+ tcConnectptr.p->tcNodedata[treplicaNo] = tnodeid;
+ tcConnectptr.p->failData[treplicaNo] = ttransStatus;
+ tcConnectptr.p->lastReplicaNo = LqhTransConf::getLastReplicaNo(treqinfo);
+ tcConnectptr.p->dirtyOp = LqhTransConf::getDirtyFlag(treqinfo);
+
+}//Dbtc::initTcConnectFail()
+
+/*----------------------------------------------------------*/
+/* INITIALISE TC NODE FAIL RECORD. */
+/*----------------------------------------------------------*/
+void Dbtc::initTcFail(Signal* signal)
+{
+ tcNodeFailptr.i = 0;
+ ptrAss(tcNodeFailptr, tcFailRecord);
+ tcNodeFailptr.p->queueIndex = 0;
+ tcNodeFailptr.p->failStatus = FS_IDLE;
+}//Dbtc::initTcFail()
+
+/*----------------------------------------------------------*/
+/* RELEASE_TAKE_OVER */
+/*----------------------------------------------------------*/
+void Dbtc::releaseTakeOver(Signal* signal)
+{
+ TcConnectRecordPtr rtoNextTcConnectptr;
+
+ rtoNextTcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ do {
+ jam();
+ tcConnectptr.i = rtoNextTcConnectptr.i;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ rtoNextTcConnectptr.i = tcConnectptr.p->nextTcConnect;
+ releaseTcConnectFail(signal);
+ } while (rtoNextTcConnectptr.i != RNIL);
+ releaseApiConnectFail(signal);
+}//Dbtc::releaseTakeOver()
+
+/*---------------------------------------------------------------------------*/
+/* SETUP_FAIL_DATA */
+/* SETUP DATA TO REUSE TAKE OVER CODE FOR HANDLING ABORT/COMMIT IN NODE */
+/* FAILURE SITUATIONS. */
+/*---------------------------------------------------------------------------*/
+void Dbtc::setupFailData(Signal* signal)
+{
+ tcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ do {
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ switch (tcConnectptr.p->tcConnectstate) {
+ case OS_PREPARED:
+ case OS_COMMITTING:
+ jam();
+ for (tindex = 0; tindex <= tcConnectptr.p->lastReplicaNo; tindex++) {
+ jam();
+ /*-------------------------------------------------------------------
+ * KEYDATA IS USED TO KEEP AN INDICATION OF STATE IN LQH.
+ * IN THIS CASE ALL LQH'S ARE PREPARED AND WAITING FOR
+ * COMMIT/ABORT DECISION.
+ *------------------------------------------------------------------*/
+ arrGuard(tindex, 4);
+ tcConnectptr.p->failData[tindex] = LqhTransConf::Prepared;
+ }//for
+ break;
+ case OS_COMMITTED:
+ case OS_COMPLETING:
+ jam();
+ for (tindex = 0; tindex <= tcConnectptr.p->lastReplicaNo; tindex++) {
+ jam();
+ /*-------------------------------------------------------------------
+ * KEYDATA IS USED TO KEEP AN INDICATION OF STATE IN LQH.
+ * IN THIS CASE ALL LQH'S ARE COMMITTED AND WAITING FOR
+ * COMPLETE MESSAGE.
+ *------------------------------------------------------------------*/
+ arrGuard(tindex, 4);
+ tcConnectptr.p->failData[tindex] = LqhTransConf::Committed;
+ }//for
+ break;
+ case OS_COMPLETED:
+ jam();
+ for (tindex = 0; tindex <= tcConnectptr.p->lastReplicaNo; tindex++) {
+ jam();
+ /*-------------------------------------------------------------------
+ * KEYDATA IS USED TO KEEP AN INDICATION OF STATE IN LQH.
+ * IN THIS CASE ALL LQH'S ARE COMPLETED.
+ *-------------------------------------------------------------------*/
+ arrGuard(tindex, 4);
+ tcConnectptr.p->failData[tindex] = LqhTransConf::InvalidStatus;
+ }//for
+ break;
+ default:
+ jam();
+ sendSystemError(signal);
+ break;
+ }//switch
+ if (tabortInd != ZCOMMIT_SETUP) {
+ jam();
+ for (UintR Ti = 0; Ti <= tcConnectptr.p->lastReplicaNo; Ti++) {
+ hostptr.i = tcConnectptr.p->tcNodedata[Ti];
+ ptrCheckGuard(hostptr, chostFilesize, hostRecord);
+ if (hostptr.p->hostStatus != HS_ALIVE) {
+ jam();
+ /*-----------------------------------------------------------------
+ * FAILURE OF ANY INVOLVED NODE ALWAYS INVOKES AN ABORT DECISION.
+ *-----------------------------------------------------------------*/
+ tabortInd = ZTRUE;
+ }//if
+ }//for
+ }//if
+ tcConnectptr.p->tcConnectstate = OS_TAKE_OVER;
+ tcConnectptr.p->tcOprec = tcConnectptr.i;
+ tcConnectptr.i = tcConnectptr.p->nextTcConnect;
+ } while (tcConnectptr.i != RNIL);
+ apiConnectptr.p->tcBlockref = cownref;
+ apiConnectptr.p->currentTcConnect = apiConnectptr.p->firstTcConnect;
+ tcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ apiConnectptr.p->currentReplicaNo = tcConnectptr.p->lastReplicaNo;
+ tcurrentReplicaNo = tcConnectptr.p->lastReplicaNo;
+}//Dbtc::setupFailData()
+
+/*----------------------------------------------------------*/
+/* UPDATE THE STATE OF THE API CONNECT FOR THIS PART. */
+/*----------------------------------------------------------*/
+void Dbtc::updateApiStateFail(Signal* signal)
+{
+ if(LqhTransConf::getMarkerFlag(treqinfo)){
+ jam();
+ const Uint32 marker = apiConnectptr.p->commitAckMarker;
+ if(marker == RNIL){
+ jam();
+
+ CommitAckMarkerPtr tmp;
+ m_commitAckMarkerHash.seize(tmp);
+ ndbrequire(tmp.i != RNIL);
+
+ apiConnectptr.p->commitAckMarker = tmp.i;
+ tmp.p->transid1 = ttransid1;
+ tmp.p->transid2 = ttransid2;
+ tmp.p->apiNodeId = refToNode(tapplRef);
+ tmp.p->noOfLqhs = 1;
+ tmp.p->lqhNodeId[0] = tnodeid;
+ tmp.p->apiConnectPtr = apiConnectptr.i;
+ m_commitAckMarkerHash.add(tmp);
+ } else {
+ jam();
+
+ CommitAckMarkerPtr tmp;
+ tmp.i = marker;
+ tmp.p = m_commitAckMarkerHash.getPtr(marker);
+
+ const Uint32 noOfLqhs = tmp.p->noOfLqhs;
+ ndbrequire(noOfLqhs < MAX_REPLICAS);
+ tmp.p->lqhNodeId[noOfLqhs] = tnodeid;
+ tmp.p->noOfLqhs = (noOfLqhs + 1);
+ }
+ }
+
+ switch (ttransStatus) {
+ case LqhTransConf::Committed:
+ jam();
+ switch (apiConnectptr.p->apiConnectstate) {
+ case CS_FAIL_COMMITTING:
+ case CS_FAIL_COMMITTED:
+ jam();
+ ndbrequire(tgci == apiConnectptr.p->globalcheckpointid);
+ break;
+ case CS_FAIL_PREPARED:
+ jam();
+ apiConnectptr.p->apiConnectstate = CS_FAIL_COMMITTING;
+ apiConnectptr.p->globalcheckpointid = tgci;
+ break;
+ case CS_FAIL_COMPLETED:
+ jam();
+ apiConnectptr.p->globalcheckpointid = tgci;
+ apiConnectptr.p->apiConnectstate = CS_FAIL_COMMITTED;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+ break;
+ case LqhTransConf::Prepared:
+ jam();
+ switch (apiConnectptr.p->apiConnectstate) {
+ case CS_FAIL_COMMITTED:
+ jam();
+ apiConnectptr.p->apiConnectstate = CS_FAIL_COMMITTING;
+ break;
+ case CS_FAIL_ABORTED:
+ jam();
+ apiConnectptr.p->apiConnectstate = CS_FAIL_ABORTING;
+ break;
+ case CS_FAIL_COMMITTING:
+ case CS_FAIL_PREPARED:
+ case CS_FAIL_ABORTING:
+ jam();
+ /*empty*/;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+ break;
+ case LqhTransConf::Aborted:
+ jam();
+ switch (apiConnectptr.p->apiConnectstate) {
+ case CS_FAIL_COMMITTING:
+ case CS_FAIL_COMMITTED:
+ jam();
+ systemErrorLab(signal);
+ break;
+ case CS_FAIL_PREPARED:
+ jam();
+ apiConnectptr.p->apiConnectstate = CS_FAIL_ABORTING;
+ break;
+ case CS_FAIL_ABORTING:
+ case CS_FAIL_ABORTED:
+ jam();
+ /*empty*/;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+ break;
+ case LqhTransConf::Marker:
+ jam();
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+}//Dbtc::updateApiStateFail()
+
+/*------------------------------------------------------------*/
+/* UPDATE_TC_STATE_FAIL */
+/* */
+/* WE NEED TO UPDATE THE STATUS OF TC_CONNECT RECORD AND*/
+/* WE ALSO NEED TO CHECK THAT THERE IS CONSISTENCY */
+/* BETWEEN THE DIFFERENT REPLICAS. */
+/*------------------------------------------------------------*/
+void Dbtc::updateTcStateFail(Signal* signal)
+{
+ const Uint8 treplicaNo = LqhTransConf::getReplicaNo(treqinfo);
+ const Uint8 tlastReplicaNo = LqhTransConf::getLastReplicaNo(treqinfo);
+ const Uint8 tdirtyOp = LqhTransConf::getDirtyFlag(treqinfo);
+
+ TcConnectRecord * regTcPtr = tcConnectptr.p;
+
+ ndbrequire(regTcPtr->apiConnect == apiConnectptr.i);
+ ndbrequire(regTcPtr->failData[treplicaNo] == LqhTransConf::InvalidStatus);
+ ndbrequire(regTcPtr->lastReplicaNo == tlastReplicaNo);
+ ndbrequire(regTcPtr->dirtyOp == tdirtyOp);
+
+ regTcPtr->tcNodedata[treplicaNo] = tnodeid;
+ regTcPtr->failData[treplicaNo] = ttransStatus;
+}//Dbtc::updateTcStateFail()
+
+void Dbtc::execTCGETOPSIZEREQ(Signal* signal)
+{
+ jamEntry();
+ CRASH_INSERTION(8000);
+
+ UintR Tuserpointer = signal->theData[0]; /* DBDIH POINTER */
+ BlockReference Tusersblkref = signal->theData[1];/* DBDIH BLOCK REFERENCE */
+ signal->theData[0] = Tuserpointer;
+ signal->theData[1] = coperationsize;
+ sendSignal(Tusersblkref, GSN_TCGETOPSIZECONF, signal, 2, JBB);
+}//Dbtc::execTCGETOPSIZEREQ()
+
+void Dbtc::execTC_CLOPSIZEREQ(Signal* signal)
+{
+ jamEntry();
+ CRASH_INSERTION(8001);
+
+ tuserpointer = signal->theData[0];
+ tusersblkref = signal->theData[1];
+ /* DBDIH BLOCK REFERENCE */
+ coperationsize = 0;
+ signal->theData[0] = tuserpointer;
+ sendSignal(tusersblkref, GSN_TC_CLOPSIZECONF, signal, 1, JBB);
+}//Dbtc::execTC_CLOPSIZEREQ()
+
+/* ######################################################################### */
+/* ####### ERROR MODULE ####### */
+/* ######################################################################### */
+void Dbtc::tabStateErrorLab(Signal* signal)
+{
+ terrorCode = ZSTATE_ERROR;
+ releaseAtErrorLab(signal);
+}//Dbtc::tabStateErrorLab()
+
+void Dbtc::wrongSchemaVersionErrorLab(Signal* signal)
+{
+ const TcKeyReq * const tcKeyReq = (TcKeyReq *)&signal->theData[0];
+
+ TableRecordPtr tabPtr;
+ tabPtr.i = tcKeyReq->tableId;
+ const Uint32 schemVer = tcKeyReq->tableSchemaVersion;
+ ptrCheckGuard(tabPtr, ctabrecFilesize, tableRecord);
+
+ terrorCode = tabPtr.p->getErrorCode(schemVer);
+
+ abortErrorLab(signal);
+}//Dbtc::wrongSchemaVersionErrorLab()
+
+void Dbtc::noFreeConnectionErrorLab(Signal* signal)
+{
+ terrorCode = ZNO_FREE_TC_CONNECTION;
+ abortErrorLab(signal); /* RECORD. OTHERWISE GOTO ERRORHANDLING */
+}//Dbtc::noFreeConnectionErrorLab()
+
+void Dbtc::aiErrorLab(Signal* signal)
+{
+ terrorCode = ZLENGTH_ERROR;
+ abortErrorLab(signal);
+}//Dbtc::aiErrorLab()
+
+void Dbtc::seizeAttrbuferrorLab(Signal* signal)
+{
+ terrorCode = ZGET_ATTRBUF_ERROR;
+ abortErrorLab(signal);
+}//Dbtc::seizeAttrbuferrorLab()
+
+void Dbtc::seizeDatabuferrorLab(Signal* signal)
+{
+ terrorCode = ZGET_DATAREC_ERROR;
+ releaseAtErrorLab(signal);
+}//Dbtc::seizeDatabuferrorLab()
+
+void Dbtc::releaseAtErrorLab(Signal* signal)
+{
+ ptrGuard(tcConnectptr);
+ tcConnectptr.p->tcConnectstate = OS_ABORTING;
+ /*-------------------------------------------------------------------------*
+ * A FAILURE OF THIS OPERATION HAS OCCURRED. THIS FAILURE WAS EITHER A
+ * FAULTY PARAMETER OR A RESOURCE THAT WAS NOT AVAILABLE.
+ * WE WILL ABORT THE ENTIRE TRANSACTION SINCE THIS IS THE SAFEST PATH
+ * TO HANDLE THIS PROBLEM.
+ * SINCE WE HAVE NOT YET CONTACTED ANY LQH WE SET NUMBER OF NODES TO ZERO
+ * WE ALSO SET THE STATE TO ABORTING TO INDICATE THAT WE ARE NOT EXPECTING
+ * ANY SIGNALS.
+ *-------------------------------------------------------------------------*/
+ tcConnectptr.p->noOfNodes = 0;
+ abortErrorLab(signal);
+}//Dbtc::releaseAtErrorLab()
+
+void Dbtc::warningHandlerLab(Signal* signal)
+{
+ ndbassert(false);
+}//Dbtc::warningHandlerLab()
+
+void Dbtc::systemErrorLab(Signal* signal)
+{
+ progError(0, 0);
+}//Dbtc::systemErrorLab()
+
+
+/* ######################################################################### *
+ * ####### SCAN MODULE ####### *
+ * ######################################################################### *
+
+ The application orders a scan of a table. We divide the scan into a scan on
+ each fragment. The scan uses the primary replicas since the scan might be
+ used for an update in a separate transaction.
+
+ Scans are always done as a separate transaction. Locks from the scan
+ can be overtaken by another transaction. Scans can never lock the entire
+ table. Locks are released immediately after the read has been verified
+ by the application. There is not even an option to leave the locks.
+ The reason is that this would hurt real-time behaviour too much.
+
+ -# The first step in handling a scan of a table is to receive all signals
+ defining the scan. If failures occur during this step we release all
+ resource and reply with SCAN_TABREF providing the error code.
+ If system load is too high, the request will not be allowed.
+
+ -# The second step retrieves the number of fragments that exist in the
+ table. It also ensures that the table actually exist. After this,
+ the scan is ready to be parallelised. The idea is that the receiving
+ process (hereafter called delivery process) will start up a number
+ of scan processes. Each of these scan processes will
+ independently scan one fragment at a time. The delivery
+ process object is the scan record and the scan process object is
+ the scan fragment record plus the scan operation record.
+
+ -# The third step is thus performed in parallel. In the third step each
+ scan process retrieves the primary replica of the fragment it will
+ scan. Then it starts the scan as soon as the load on that node permits.
+
+ The LQH returns either when it retrieved the maximum number of tuples or
+ when it has retrived at least one tuple and is hindered by a lock to
+ retrieve the next tuple. This is to ensure that a scan process never
+ can be involved in a deadlock situation.
+
+ When the scan process receives a number of tuples to report to the
+ application it checks the state of the delivery process. Only one delivery
+ at a time is handled by the application. Thus if the delivery process
+ has already sent a number of tuples to the application this set of tuples
+ are queued.
+
+ When the application requests the next set of tuples it is immediately
+ delivered if any are queued, otherwise it waits for the next scan
+ process that is ready to deliver.
+
+
+ ERROR HANDLING
+
+ As already mentioned it is rather easy to handle errors before the scan
+ processes have started. In this case it is enough to release the resources
+ and send SCAN_TAB_REF.
+
+ If an error occurs in any of the scan processes then we have to stop all
+ scan processes. We do however only stop the delivery process and ask
+ the api to order us to close the scan. The reason is that we can easily
+ enter into difficult timing problems since the application and this
+ block is out of synch we will thus always start by report the error to
+ the application and wait for a close request. This error report uses the
+ SCAN_TABREF signal with a special error code that the api must check for.
+
+
+ CLOSING AN ACTIVE SCAN
+
+ The application can close a scan for several reasons before it is completed.
+ One reason was mentioned above where an error in a scan process led to a
+ request to close the scan. Another reason could simply be that the
+ application found what it looked for and is thus not interested in the
+ rest of the scan.
+
+ IT COULD ALSO BE DEPENDENT ON INTERNAL ERRORS IN THE API.
+
+ When a close scan request is received, all scan processes are stopped and all
+ resources belonging to those scan processes are released. Stopping the scan
+ processes most often includes communication with an LQH where the local scan
+ is controlled. Finally all resources belonging to the scan is released and
+ the SCAN_TABCONF is sent with an indication of that the scan is closed.
+
+
+ CLOSING A COMPLETED SCAN
+
+ When all scan processes are completed then a report is sent to the
+ application which indicates that no more tuples can be fetched.
+ The application will send a close scan and the same action as when
+ closing an active scan is performed.
+ In this case it will of course not find any active scan processes.
+ It will even find all scan processes already released.
+
+ The reason for requiring the api to close the scan is the same as above.
+ It is to avoid any timing problems due to that the api and this block
+ is out of synch.
+
+ * ######################################################################## */
+void Dbtc::execSCAN_TABREQ(Signal* signal)
+{
+ const ScanTabReq * const scanTabReq = (ScanTabReq *)&signal->theData[0];
+ const UintR reqinfo = scanTabReq->requestInfo;
+ const Uint32 aiLength = scanTabReq->attrLen;
+ const Uint32 schemaVersion = scanTabReq->tableSchemaVersion;
+ const UintR transid1 = scanTabReq->transId1;
+ const UintR transid2 = scanTabReq->transId2;
+ const Uint32 tmpXX = scanTabReq->buddyConPtr;
+ const Uint32 buddyPtr = (tmpXX == 0xFFFFFFFF ? RNIL : tmpXX);
+ Uint32 currSavePointId = 0;
+
+ Uint8 scanConcurrency = scanTabReq->getParallelism(reqinfo);
+ Uint32 scanParallel;
+ Uint32 noOprecPerFrag;
+ Uint32 errCode;
+
+ jamEntry();
+ apiConnectptr.i = scanTabReq->apiConnectPtr;
+ tabptr.i = scanTabReq->tableId;
+ for(int i=0; i<16; i++)
+ cdata[i] = scanTabReq->apiOperationPtr[i];
+
+ if (apiConnectptr.i >= capiConnectFilesize ||
+ tabptr.i >= ctabrecFilesize) {
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }//if
+ ptrAss(apiConnectptr, apiConnectRecord);
+
+ if (apiConnectptr.p->apiConnectstate != CS_CONNECTED) {
+ jam();
+ // could be left over from TCKEYREQ rollback
+ if (apiConnectptr.p->apiConnectstate == CS_ABORTING &&
+ apiConnectptr.p->abortState == AS_IDLE) {
+ jam();
+ } else {
+ jam();
+ errCode = ZSTATE_ERROR;
+ goto SCAN_TAB_error;
+ }
+ }
+ ptrAss(tabptr, tableRecord);
+
+ if (aiLength == 0) {
+ jam()
+ errCode = ZSCAN_AI_LEN_ERROR;
+ goto SCAN_TAB_error;
+ }//if
+ if (!tabptr.p->checkTable(schemaVersion)){
+ jam();
+ goto SCAN_TAB_schema_error;
+ }//if
+ /*****************************************************************
+ * THE CONCURRENCY LEVEL SPECIFIED BY THE APPLICATION. IT MUST BE
+ * BETWEEN 1 AND 240. IF IT IS 16 OR GREATER IT MUST BE A MULTIPLE
+ * OF 16. CONCURRENCY LEVELS UPTO 16 ONLY SCAN ONE FRAGMENT AT A
+ * TIME. IF WE SPECIFY 32 IT WILL SCAN TWO FRAGMENTS AT A TIME AND
+ * SO FORTH. MAXIMUM 15 PARALLEL SCANS ARE ALLOWED
+ ******************************************************************/
+ if (scanConcurrency == 0) {
+ jam();
+ errCode = ZNO_CONCURRENCY_ERROR;
+ goto SCAN_TAB_error;
+ }//if
+ if (scanConcurrency <= 16) {
+ jam();
+ noOprecPerFrag = scanConcurrency;
+ } else {
+ if (scanConcurrency <= 240) {
+ jam();
+ //If scanConcurrency > 16 it must be a multiple of 16
+ if (((scanConcurrency >> 4) << 4) < scanConcurrency) {
+ scanConcurrency = ((scanConcurrency >> 4) << 4) + 16;
+ }//if
+ } else {
+ jam();
+ errCode = ZTOO_HIGH_CONCURRENCY_ERROR;
+ goto SCAN_TAB_error;
+ }//if
+ noOprecPerFrag = 16;
+ }//if
+
+ scanParallel = ((scanConcurrency - 1) >> 4) + 1;
+ /**********************************************************
+ * CALCULATE THE NUMBER OF SCAN_TABINFO SIGNALS THAT WILL
+ * ARRIVE TO DEFINE THIS SCAN. THIS ALSO DEFINES THE NUMBER
+ * OF PARALLEL SCANS AND IT ALSO DEFINES THE NUMBER OF SCAN
+ * OPERATION POINTER RECORDS TO ALLOCATE.
+ **********************************************************/
+ if (cnoFreeScanOprec < scanParallel) {
+ jam();
+ errCode = ZNO_SCANREC_ERROR;
+ goto SCAN_TAB_error;
+ // WE DID NOT HAVE ENOUGH OF FREE SCAN OPERATION POINTER RECORDS.
+ // THUS WE REFUSE THE SCAN OPERATION.
+ }//if
+ if (cfirstfreeTcConnect == RNIL) {
+ jam();
+ errCode = ZNO_FREE_TC_CONNECTION;
+ goto SCAN_TAB_error;
+ }//if
+ if (cfirstfreeScanrec == RNIL) {
+ jam();
+ errCode = ZNO_SCANREC_ERROR;
+ goto SCAN_TAB_error;
+ }//if
+
+ if (buddyPtr != RNIL) {
+ jam();
+
+ ApiConnectRecordPtr buddyApiPtr;
+ buddyApiPtr.i = buddyPtr;
+ ptrCheckGuard(buddyApiPtr, capiConnectFilesize, apiConnectRecord);
+ if ((transid1 == buddyApiPtr.p->transid[0]) &&
+ (transid2 == buddyApiPtr.p->transid[1])) {
+ jam();
+ currSavePointId = buddyApiPtr.p->currSavePointId;
+ buddyApiPtr.p->currSavePointId++;
+ }
+ }
+
+ seizeTcConnect(signal);
+ seizeCacheRecord(signal);
+ seizeScanrec(signal);
+ initScanrec(signal, scanParallel, noOprecPerFrag);
+ initScanTcrec(signal);
+ initScanApirec(signal, buddyPtr, transid1, transid2);
+ cnoFreeScanOprec = cnoFreeScanOprec - scanParallel;
+
+ // The scan is started
+ apiConnectptr.p->apiConnectstate = CS_START_SCAN;
+ apiConnectptr.p->currSavePointId = currSavePointId;
+
+ /**********************************************************
+ * We start the timer on scanRec to be able to discover a
+ * timeout in the API the API now is in charge!
+ ***********************************************************/
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ updateBuddyTimer(apiConnectptr);
+ if (scanptr.p->noScanTabInfo > 1) {
+ jam();
+ scanptr.p->scanState = ScanRecord::WAIT_SCAN_TAB_INFO;
+ return;
+ }//if
+ /***********************************************************
+ * WE HAVE NOW RECEIVED ALL REFERENCES TO SCAN OBJECTS IN
+ * THE API. WE ARE NOW READY TO RECEIVE THE ATTRIBUTE INFO
+ * IF ANY TO RECEIVE.
+ **********************************************************/
+ scanptr.p->scanState = ScanRecord::WAIT_AI;
+ return;
+
+ SCAN_TAB_schema_error:
+ jam();
+ errCode = tabptr.p->getErrorCode(schemaVersion);
+
+ SCAN_TAB_error:
+ jam();
+ ScanTabRef * ref = (ScanTabRef*)&signal->theData[0];
+ ref->apiConnectPtr = apiConnectptr.p->ndbapiConnect;
+ ref->transId1 = transid1;
+ ref->transId2 = transid2;
+ ref->errorCode = errCode;
+ sendSignal(apiConnectptr.p->ndbapiBlockref, GSN_SCAN_TABREF,
+ signal, ScanTabRef::SignalLength, JBB);
+
+ return;
+}//Dbtc::execSCAN_TABREQ()
+
+void Dbtc::initScanTcrec(Signal* signal)
+{
+ tcConnectptr.p->apiConnect = apiConnectptr.i;
+}//Dbtc::initScanTcrec()
+
+void Dbtc::initScanApirec(Signal* signal,
+ Uint32 buddyPtr, UintR transid1, UintR transid2)
+{
+ ApiConnectRecord * apiPtr = apiConnectptr.p;
+ apiPtr->apiScanRec = scanptr.i;
+ apiPtr->returncode = 0;
+ apiPtr->transid[0] = transid1;
+ apiPtr->transid[1] = transid2;
+ apiPtr->buddyPtr = buddyPtr;
+
+}//Dbtc::initScanApirec()
+
+void Dbtc::initScanOprec(Signal* signal)
+{
+ UintR tisoIndex;
+
+ for (tisoIndex = 0; tisoIndex < 16; tisoIndex++) {
+ scanOpptr.p->apiOpptr[tisoIndex] = cdata[tisoIndex];
+ scanOpptr.p->scanOpLength[tisoIndex] = RNIL;
+ }//for
+}//Dbtc::initScanOprec()
+
+void Dbtc::initScanrec(Signal* signal,
+ UintR scanParallel,
+ UintR noOprecPerFrag)
+{
+ const ScanTabReq * const scanTabReq = (ScanTabReq *)&signal->theData[0];
+ const UintR reqinfo = scanTabReq->requestInfo;
+ ndbrequire(scanParallel < 16);
+
+ scanptr.p->scanTcrec = tcConnectptr.i;
+ scanptr.p->scanApiRec = apiConnectptr.i;
+ scanptr.p->scanAiLength = scanTabReq->attrLen;
+ scanptr.p->scanTableref = tabptr.i;
+ scanptr.p->scanSchemaVersion = scanTabReq->tableSchemaVersion;
+ scanptr.p->scanParallel = scanParallel;
+ scanptr.p->noScanOprec = scanParallel;
+ scanptr.p->noScanTabInfo = scanParallel;
+ scanptr.p->scanTabInfoReceived = 1;
+ scanptr.p->scanProcessesCompleted = 0;
+ scanptr.p->scanLockMode = ScanTabReq::getLockMode(reqinfo);
+ scanptr.p->scanLockHold = ScanTabReq::getHoldLockFlag(reqinfo);
+ scanptr.p->readCommitted = ScanTabReq::getReadCommittedFlag(reqinfo);
+ scanptr.p->rangeScan = ScanTabReq::getRangeScanFlag(reqinfo);
+ scanptr.p->scanStoredProcId = scanTabReq->storedProcId;
+ scanptr.p->scanReceivedOperations = 0;
+ scanptr.p->noOprecPerFrag = noOprecPerFrag;
+ scanptr.p->apiIsClosed = false;
+ scanptr.p->scanCompletedStatus = ZFALSE;
+ scanptr.p->scanState = ScanRecord::SCAN_NEXT_ORDERED;
+ for (Uint32 i = 0; i < 16; i++) {
+ if (i < scanParallel){
+ jam();
+ seizeScanOprec(signal);
+ scanptr.p->scanOprec[i] = scanOpptr.i;
+ } else {
+ jam();
+ scanptr.p->scanOprec[i] = RNIL;
+ }
+ scanptr.p->scanFragrec[i] = RNIL;
+ }//for
+ scanOpptr.i = scanptr.p->scanOprec[0];
+ ptrCheckGuard(scanOpptr, cscanOprecFileSize, scanOperationRecord);
+ initScanOprec(signal);
+}//Dbtc::initScanrec()
+
+void Dbtc::scanTabRefLab(Signal* signal, Uint32 errCode)
+{
+ ScanTabRef * ref = (ScanTabRef*)&signal->theData[0];
+ ref->apiConnectPtr = apiConnectptr.p->ndbapiConnect;
+ ref->transId1 = apiConnectptr.p->transid[0];
+ ref->transId2 = apiConnectptr.p->transid[1];
+ ref->errorCode = errCode;
+ sendSignal(apiConnectptr.p->ndbapiBlockref, GSN_SCAN_TABREF,
+ signal, ScanTabRef::SignalLength, JBB);
+}//Dbtc::scanTabRefLab()
+
+/******************************************************
+ * execSCAN_TABINFO
+ ******************************************************/
+void Dbtc::execSCAN_TABINFO(Signal* signal)
+{
+ jamEntry();
+ apiConnectptr.i = signal->theData[0];
+ for(int i=0; i<16; i++)
+ cdata[i] = signal->theData[i+1];
+
+ if (apiConnectptr.i >= capiConnectFilesize) {
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }//if
+ ptrAss(apiConnectptr, apiConnectRecord);
+
+ if (apiConnectptr.p->apiConnectstate != CS_START_SCAN){
+ jam();
+ DEBUG("apiPtr(" << apiConnectptr.i << ") Dropping SCAN_TABINFO, wrong state: " << apiConnectptr.p->apiConnectstate);
+ return;
+ }
+
+ scanptr.i = apiConnectptr.p->apiScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+
+ const Uint32 tscanOprec = scanptr.p->scanTabInfoReceived;
+ scanptr.p->scanTabInfoReceived++;
+ arrGuard(tscanOprec, 16);
+ scanOpptr.i = scanptr.p->scanOprec[tscanOprec];
+ ptrCheckGuard(scanOpptr, cscanOprecFileSize, scanOperationRecord);
+ // Start timer and wait for response from API node.
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ updateBuddyTimer(apiConnectptr);
+
+ initScanOprec(signal);
+ // Start timer and wait for response from API node.
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ updateBuddyTimer(apiConnectptr);
+
+ if (scanptr.p->scanTabInfoReceived == scanptr.p->noScanTabInfo) {
+ jam();
+ /******************************************************************
+ * WE HAVE NOW RECEIVED ALL REFERENCES TO SCAN OBJECTS IN THE API.
+ * WE ARE NOW READY TO RECEIVE THE ATTRIBUTE INFO IF ANY TO RECEIVE.
+ ******************************************************************/
+ scanptr.p->scanState = ScanRecord::WAIT_AI;
+ return;
+ }
+ ndbrequire(scanptr.p->scanTabInfoReceived <= scanptr.p->noScanTabInfo);
+}//Dbtc::execSCAN_TABINFO()
+
+/*---------------------------------------------------------------------------*/
+/* */
+/* RECEPTION OF ATTRINFO FOR SCAN TABLE REQUEST. */
+/*---------------------------------------------------------------------------*/
+void Dbtc::scanAttrinfoLab(Signal* signal, UintR Tlen)
+{
+ scanptr.i = apiConnectptr.p->apiScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ cachePtr.i = apiConnectptr.p->cachePtr;
+ ptrCheckGuard(cachePtr, ccacheFilesize, cacheRecord);
+ CacheRecord * const regCachePtr = cachePtr.p;
+ ndbrequire(scanptr.p->scanState == ScanRecord::WAIT_AI);
+
+ regCachePtr->currReclenAi = regCachePtr->currReclenAi + Tlen;
+ if (regCachePtr->currReclenAi < scanptr.p->scanAiLength) {
+ if (cfirstfreeAttrbuf == RNIL) {
+ goto scanAttrinfo_attrbuf_error;
+ }//if
+ saveAttrbuf(signal);
+ } else {
+ if (regCachePtr->currReclenAi > scanptr.p->scanAiLength) {
+ goto scanAttrinfo_len_error;
+ } else {
+ /* CURR_RECLEN_AI = SCAN_AI_LENGTH */
+ if (cfirstfreeAttrbuf == RNIL) {
+ goto scanAttrinfo_attrbuf2_error;
+ }//if
+ saveAttrbuf(signal);
+ /**************************************************
+ * WE HAVE NOW RECEIVED ALL INFORMATION CONCERNING
+ * THIS SCAN. WE ARE READY TO START THE ACTUAL
+ * EXECUTION OF THE SCAN QUERY
+ **************************************************/
+ diFcountReqLab(signal);
+ return;
+ }//if
+ }//if
+ return;
+
+scanAttrinfo_attrbuf_error:
+ jam();
+ abortScanLab(signal, ZGET_ATTRBUF_ERROR);
+ return;
+
+scanAttrinfo_attrbuf2_error:
+ jam();
+ abortScanLab(signal, ZGET_ATTRBUF_ERROR);
+ return;
+
+scanAttrinfo_len_error:
+ jam();
+ abortScanLab(signal, ZLENGTH_ERROR);
+ return;
+}//Dbtc::scanAttrinfoLab()
+
+void Dbtc::diFcountReqLab(Signal* signal)
+{
+ /**
+ * Check so that the table is not being dropped
+ */
+ TableRecordPtr tabPtr;
+ tabPtr.i = scanptr.p->scanTableref;
+ tabPtr.p = &tableRecord[tabPtr.i];
+ if (tabPtr.p->checkTable(scanptr.p->scanSchemaVersion)){
+ ;
+ } else {
+ abortScanLab(signal, tabPtr.p->getErrorCode(scanptr.p->scanSchemaVersion));
+ return;
+ }
+
+ scanptr.p->scanState = ScanRecord::WAIT_FRAGMENT_COUNT;
+ /*************************************************
+ * THE FIRST STEP TO RECEIVE IS SUCCESSFULLY COMPLETED.
+ * WE MUST FIRST GET THE NUMBER OF FRAGMENTS IN THE TABLE.
+ ***************************************************/
+ signal->theData[0] = tcConnectptr.p->dihConnectptr;
+ signal->theData[1] = scanptr.p->scanTableref;
+ sendSignal(cdihblockref, GSN_DI_FCOUNTREQ, signal, 2, JBB);
+ return;
+}//Dbtc::diFcountReqLab()
+
+/********************************************************************
+ * execDI_FCOUNTCONF
+ *
+ * WE HAVE ASKED DIH ABOUT THE NUMBER OF FRAGMENTS IN THIS TABLE.
+ * WE WILL NOW START A NUMBER OF PARALLEL SCAN PROCESSES. EACH OF
+ * THESE WILL SCAN ONE FRAGMENT AT A TIME. THEY WILL CONTINUE THIS
+ * UNTIL THERE ARE NO MORE FRAGMENTS TO SCAN OR UNTIL THE APPLICATION
+ * CLOSES THE SCAN.
+ ********************************************************************/
+void Dbtc::execDI_FCOUNTCONF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ const UintR tfragCount = signal->theData[1];
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ apiConnectptr.i = tcConnectptr.p->apiConnect;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ scanptr.i = apiConnectptr.p->apiScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ ndbrequire(scanptr.p->scanState == ScanRecord::WAIT_FRAGMENT_COUNT);
+ if (apiConnectptr.p->apiFailState == ZTRUE) {
+ jam();
+ releaseScanResources(signal);
+ handleApiFailState(signal, apiConnectptr.i);
+ return;
+ }//if
+ if (tfragCount == 0) {
+ jam();
+ abortScanLab(signal, ZNO_FRAGMENT_ERROR);
+ return;
+ }//if
+
+ /**
+ * Check so that the table is not being dropped
+ */
+ TableRecordPtr tabPtr;
+ tabPtr.i = scanptr.p->scanTableref;
+ tabPtr.p = &tableRecord[tabPtr.i];
+ if (tabPtr.p->checkTable(scanptr.p->scanSchemaVersion)){
+ ;
+ } else {
+ abortScanLab(signal, tabPtr.p->getErrorCode(scanptr.p->scanSchemaVersion));
+ return;
+ }
+
+ if (tfragCount < scanptr.p->scanParallel) {
+ jam();
+ for (Uint32 i = tfragCount; i < scanptr.p->scanParallel; i++) {
+ jam();
+ arrGuard(i, 16);
+ scanOpptr.i = scanptr.p->scanOprec[i];
+ ptrCheckGuard(scanOpptr, cscanOprecFileSize, scanOperationRecord);
+ releaseScanOprec(signal);
+ scanptr.p->scanOprec[i] = RNIL;
+ }//for
+ scanptr.p->scanParallel = tfragCount;
+ }//if
+ scanptr.p->scanNoFrag = tfragCount;
+ for (UintR i = 0; i < scanptr.p->scanParallel; i++) {
+ jam();
+ // START EACH OF THE PARALLEL SCAN PROCESSES
+ signal->theData[0] = scanptr.i;
+ signal->theData[1] = i;
+ signal->theData[2] = scanptr.p->noOprecPerFrag;
+ sendSignal(cownref, GSN_SCAN_PROCREQ, signal, 3, JBB);
+ }//for
+ // We don't need the timer for checking API anymore, control goes to LQH.
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ scanptr.p->scanNextFragId = scanptr.p->scanParallel;
+ scanptr.p->scanState = ScanRecord::SCAN_NEXT_ORDERED;
+}//Dbtc::execDI_FCOUNTCONF()
+
+/******************************************************
+ * execDI_FCOUNTREF
+ ******************************************************/
+void Dbtc::execDI_FCOUNTREF(Signal* signal)
+{
+ jamEntry();
+ tcConnectptr.i = signal->theData[0];
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ const Uint32 errCode = signal->theData[1];
+ apiConnectptr.i = tcConnectptr.p->apiConnect;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ scanptr.i = apiConnectptr.p->apiScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ ndbrequire(scanptr.p->scanState == ScanRecord::WAIT_FRAGMENT_COUNT);
+ if (apiConnectptr.p->apiFailState == ZTRUE) {
+ jam();
+ releaseScanResources(signal);
+ handleApiFailState(signal, apiConnectptr.i);
+ return;
+ }//if
+ abortScanLab(signal, errCode);
+}//Dbtc::execDI_FCOUNTREF()
+
+void Dbtc::abortScanLab(Signal* signal, Uint32 errCode)
+{
+ releaseScanResources(signal);
+ scanTabRefLab(signal, errCode);
+}//Dbtc::abortScanLab()
+
+void Dbtc::scanReleaseResourcesLab(Signal* signal)
+{
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ if (apiConnectptr.p->returncode != 0) {
+ jam();
+ ScanTabRef * ref = (ScanTabRef*)&signal->theData[0];
+ ref->apiConnectPtr = apiConnectptr.p->ndbapiConnect;
+ ref->transId1 = apiConnectptr.p->transid[0];
+ ref->transId2 = apiConnectptr.p->transid[1];
+ ref->errorCode = apiConnectptr.p->returncode;
+ sendSignal(apiConnectptr.p->ndbapiBlockref,
+ GSN_SCAN_TABREF, signal, ScanTabRef::SignalLength, JBB);
+ } else {
+ jam();
+ sendScanTabConf(signal);
+ }//if
+ releaseScanResources(signal);
+ if (apiConnectptr.p->apiFailState == ZTRUE) {
+ jam();
+ handleApiFailState(signal, apiConnectptr.i);
+ return;
+ }//if
+}//Dbtc::scanReleaseResourcesLab()
+
+void Dbtc::releaseScanResources(Signal* signal)
+{
+ if (apiConnectptr.p->cachePtr != RNIL) {
+ cachePtr.i = apiConnectptr.p->cachePtr;
+ ptrCheckGuard(cachePtr, ccacheFilesize, cacheRecord);
+ releaseAttrinfo(signal);
+ }//if
+ cnoFreeScanOprec = cnoFreeScanOprec + scanptr.p->noScanOprec;
+ scanptr.p->scanCompletedStatus = ZCLOSED;
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ releaseTcCon(signal);
+ for (Uint32 i = 0; i < 16; i++) {
+ jam();
+ scanFragptr.i = scanptr.p->scanFragrec[i];
+ scanptr.p->scanFragrec[i] = RNIL;
+ if (scanFragptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+ releaseScanFragrec(signal);
+ }//if
+ scanOpptr.i = scanptr.p->scanOprec[i];
+ scanptr.p->scanOprec[i] = RNIL;
+ if (scanOpptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(scanOpptr, cscanOprecFileSize, scanOperationRecord);
+ releaseScanOprec(signal);
+ }//if
+ }//for
+ releaseScanrec(signal);
+ apiConnectptr.p->apiScanRec = RNIL;
+ apiConnectptr.p->apiConnectstate = CS_CONNECTED;
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+}//Dbtc::releaseScanResources()
+
+
+/******************************************************
+ * execSCAN_PROCREQ
+ ******************************************************/
+void Dbtc::execSCAN_PROCREQ(Signal* signal)
+{
+ jamEntry();
+ scanptr.i = signal->theData[0];
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+
+ const UintR tscanFragId = signal->theData[1];
+ ndbrequire(tscanFragId < 16);
+ const UintR tscanNoOprec = signal->theData[2];
+
+ ndbrequire(cfirstfreeScanFragrec != RNIL);
+ seizeScanFragrec(signal);
+
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+
+ scanptr.p->scanFragrec[tscanFragId] = scanFragptr.i;
+ scanFragptr.p->scanRec = scanptr.i;
+ scanFragptr.p->scanIndividual = tscanFragId * tscanNoOprec;
+ scanFragptr.p->scanFragProcId = tscanFragId;
+ scanFragptr.p->scanFragId = tscanFragId;
+ scanFragptr.p->scanFragConcurrency = tscanNoOprec;
+ scanFragptr.p->scanFragCompletedStatus = ZFALSE;
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+
+ {
+ /**
+ * Check table
+ */
+ TableRecordPtr tabPtr;
+ tabPtr.i = scanptr.p->scanTableref;
+ ptrAss(tabPtr, tableRecord);
+ Uint32 schemaVersion = scanptr.p->scanSchemaVersion;
+ if(tabPtr.p->checkTable(schemaVersion) == false){
+ jam();
+ scanFragError(signal, tabPtr.p->getErrorCode(schemaVersion));
+ return;
+ }
+ }
+
+ signal->theData[0] = tcConnectptr.p->dihConnectptr;
+ signal->theData[1] = scanFragptr.i;
+ signal->theData[2] = scanptr.p->scanTableref;
+ signal->theData[3] = tscanFragId;
+ sendSignal(cdihblockref, GSN_DIGETPRIMREQ, signal, 4, JBB);
+ scanFragptr.p->scanFragState = ScanFragRec::WAIT_GET_PRIMCONF;
+ updateBuddyTimer(apiConnectptr);
+ scanFragptr.p->startFragTimer(ctcTimer);
+}//Dbtc::execSCAN_PROCREQ()
+
+/****************************************************************
+ * execDIGETPRIMCONF
+ *
+ * WE HAVE RECEIVED THE PRIMARY NODE OF THIS FRAGMENT.
+ * WE ARE NOW READY TO ASK FOR PERMISSION TO LOAD THIS
+ * SPECIFIC NODE WITH A SCAN OPERATION.
+ ****************************************************************/
+void Dbtc::execDIGETPRIMCONF(Signal* signal)
+{
+ jamEntry();
+ // tcConnectptr.i in theData[0] is not used
+ scanFragptr.i = signal->theData[1];
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+
+ tnodeid = signal->theData[2];
+ arrGuard(tnodeid, MAX_NDB_NODES);
+
+ ndbrequire(scanFragptr.p->scanFragState == ScanFragRec::WAIT_GET_PRIMCONF);
+ scanFragptr.p->stopFragTimer();
+ scanFragptr.p->lqhBlockref = RNIL;
+
+ scanptr.i = scanFragptr.p->scanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+
+ {
+ /**
+ * Check table
+ */
+ TableRecordPtr tabPtr;
+ tabPtr.i = scanptr.p->scanTableref;
+ ptrAss(tabPtr, tableRecord);
+ Uint32 schemaVersion = scanptr.p->scanSchemaVersion;
+ if(tabPtr.p->checkTable(schemaVersion) == false){
+ jam();
+ scanFragError(signal, tabPtr.p->getErrorCode(schemaVersion));
+ return;
+ }
+ }
+
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ cachePtr.i = apiConnectptr.p->cachePtr;
+ ptrCheckGuard(cachePtr, ccacheFilesize, cacheRecord);
+ switch (scanptr.p->scanState) {
+ case ScanRecord::CLOSING_SCAN:
+ jam();
+ updateBuddyTimer(apiConnectptr);
+ scanFragptr.p->startFragTimer(ctcTimer);
+ sendScanProcConf(signal);
+ return;
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }//switch
+ scanFragptr.p->scanFragNodeId = tnodeid;
+ scanFragptr.p->lqhBlockref = calcLqhBlockRef(tnodeid);
+ scanFragptr.p->m_connectCount = getNodeInfo(tnodeid).m_connectCount;
+ sendScanFragReq(signal);
+ attrbufptr.i = cachePtr.p->firstAttrbuf;
+ while (attrbufptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(attrbufptr, cattrbufFilesize, attrbufRecord);
+ sendAttrinfo(signal,
+ scanFragptr.i,
+ attrbufptr.p,
+ scanFragptr.p->lqhBlockref);
+ attrbufptr.i = attrbufptr.p->attrbuf[ZINBUF_NEXT];
+ }//while
+ scanFragptr.p->scanFragState = ScanFragRec::LQH_ACTIVE;
+ updateBuddyTimer(apiConnectptr);
+ scanFragptr.p->startFragTimer(ctcTimer);
+ /*********************************************
+ * WE HAVE NOW STARTED A FRAGMENT SCAN. NOW
+ * WAIT FOR THE FIRST SCANNED RECORDS
+ *********************************************/
+}//Dbtc::execDIGETPRIMCONF
+
+/***************************************************
+ * execDIGETPRIMREF
+ *
+ * WE ARE NOW FORCED TO STOP THE SCAN. THIS ERROR
+ * IS NOT RECOVERABLE SINCE THERE IS A PROBLEM WITH
+ * FINDING A PRIMARY REPLICA OF A CERTAIN FRAGMENT.
+ ***************************************************/
+void Dbtc::execDIGETPRIMREF(Signal* signal)
+{
+ jamEntry();
+ // tcConnectptr.i in theData[0] is not used.
+ scanFragptr.i = signal->theData[1];
+ const Uint32 errCode = signal->theData[2];
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+ ndbrequire(scanFragptr.p->scanFragState == ScanFragRec::WAIT_GET_PRIMCONF);
+ scanFragError(signal, errCode);
+}//Dbtc::execDIGETPRIMREF()
+
+/**
+ * Dbtc::execSCAN_FRAGREF
+ * Our attempt to scan a fragment was refused
+ * set error code and close all other fragment
+ * scan's belonging to this scan
+ */
+void Dbtc::execSCAN_FRAGREF(Signal* signal)
+{
+ const ScanFragRef * const ref = (ScanFragRef *)&signal->theData[0];
+
+ jamEntry();
+ const Uint32 errCode = ref->errorCode;
+
+ scanFragptr.i = ref->senderData;
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+
+ scanptr.i = scanFragptr.p->scanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+
+ Uint32 transid1 = apiConnectptr.p->transid[0] ^ ref->transId1;
+ Uint32 transid2 = apiConnectptr.p->transid[1] ^ ref->transId2;
+ transid1 = transid1 | transid2;
+ if (transid1 != 0) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+
+ /**
+ * Set errorcode, close connection to this lqh fragment,
+ * stop fragment timer and call scanFragError to start
+ * close of the other fragment scans
+ */
+ scanFragptr.p->lqhBlockref = RNIL;
+ scanFragError(signal, errCode);
+}//Dbtc::execSCAN_FRAGREF()
+
+/**
+ * Dbtc::scanFragError
+ *
+ * Called when an error occurs during
+ * a scan of a fragment.
+ * NOTE that one scan may consist of several fragment scans.
+ *
+ */
+void Dbtc::scanFragError(Signal* signal, Uint32 errorCode)
+{
+ jam();
+ scanptr.i = scanFragptr.p->scanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ DEBUG("scanFragError, errorCode = "<< errorCode
+ << ", scanState = " << scanptr.p->scanState);
+
+ scanFragptr.p->stopFragTimer();
+
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ if (scanFragptr.p->lqhBlockref == RNIL){
+ // Since the lqh is closed, this scan process should be reported
+ // as completed immediately
+ jam();
+ updateBuddyTimer(apiConnectptr);
+ scanFragptr.p->startFragTimer(ctcTimer);
+ sendScanProcConf(signal);
+ }//if
+
+ // If close of the scan is not already started
+ if (scanptr.p->scanState != ScanRecord::CLOSING_SCAN) {
+ jam();
+ apiConnectptr.p->returncode = errorCode;
+
+ /**
+ * Only set apiIsClosed if API is waiting for an answer
+ */
+ if (scanptr.p->scanState == ScanRecord::SCAN_NEXT_ORDERED){
+ jam();
+ scanptr.p->apiIsClosed = true;
+ }
+ scanCompletedLab(signal);
+ return;
+ }//if
+}//Dbtc::scanFragError()
+
+
+/************************************************************
+ * execSCAN_FRAGCONF
+ *
+ * A NUMBER OF OPERATIONS HAVE BEEN COMPLETED IN THIS
+ * FRAGMENT. TAKE CARE OF AND ISSUE FURTHER ACTIONS.
+ ************************************************************/
+void Dbtc::execSCAN_FRAGCONF(Signal* signal)
+{
+ jamEntry();
+
+ const ScanFragConf * const conf = (ScanFragConf*)&signal->theData[0];
+ const Uint32 noCompletedOps = conf->completedOps;
+
+ for(Uint32 i = 0; i<noCompletedOps; i++)
+ cdata[i] = conf->opReturnDataLen[i];
+ scanFragptr.i = conf->senderData;
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+
+ scanptr.i = scanFragptr.p->scanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+
+ Uint32 transid1 = apiConnectptr.p->transid[0] ^ conf->transId1;
+ Uint32 transid2 = apiConnectptr.p->transid[1] ^ conf->transId2;
+ transid1 = transid1 | transid2;
+ if (transid1 != 0) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+
+ scanFragptr.p->scanFragCompletedStatus = conf->fragmentCompleted;
+ scanFragptr.p->stopFragTimer();
+
+ switch (scanFragptr.p->scanFragCompletedStatus) {
+ case ZFALSE:
+ case ZTRUE:
+ jam();
+ /* empty */
+ break;
+
+ case ZCLOSED:
+ /* The scan has finished this fragment. */
+ jam();
+ returnFromQueuedDeliveryLab(signal);
+ return;
+ break;
+
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+
+ // CHECK THE STATE OF THE DELIVERY PROCESS TO THE APPLICATION.
+ switch (scanptr.p->scanState) {
+ case ScanRecord::SCAN_NEXT_ORDERED:
+ jam();
+ /**
+ * THE APPLICATION HAVE ISSUED A SCAN_NEXTREQ AND IS WAITING
+ * FOR MORE OPERATIONS. SEND OPERATIONS DIRECTLY
+ */
+ if (noCompletedOps > 0) {
+ jam();
+ setScanReceived(signal, noCompletedOps);
+ sendScanTabConf(signal);
+ scanptr.p->scanState = ScanRecord::DELIVERED;
+ scanFragptr.p->scanFragState = ScanFragRec::DELIVERED;
+ return;
+ }//if
+ break;
+
+ case ScanRecord::DELIVERED:
+ case ScanRecord::QUEUED_DELIVERED:
+ jam();
+ /**
+ * THE APPLICATION HAVE ALREADY RECEIVED A DELIVERY.
+ * QUEUE THE RECEIVED SCAN OPERATIONS AND ISSUE THEM
+ * WHEN THE APPLICATION ASKS FOR MORE.
+ */
+ if (noCompletedOps > 0) {
+ jam();
+ setScanReceived(signal, noCompletedOps);
+ scanptr.p->scanState = ScanRecord::QUEUED_DELIVERED;
+ scanFragptr.p->scanFragState = ScanFragRec::QUEUED_FOR_DELIVERY;
+ return;
+ }//if
+ break;
+
+ case ScanRecord::CLOSING_SCAN:
+ jam();
+ /*************************************************
+ * WE ARE CURRENTLY CLOSING THE SCAN.
+ *
+ * WE HAVE ALREADY ORDERED THE FRAGMENT TO CLOSE ITS
+ * SCAN. THIS SIGNAL MUST HAVE BEEN SENT BEFORE THIS
+ * CLOSE SIGNAL ARRIVED. SIMPLY IGNORE THIS SIGNAL.
+ **************************************************/
+ return;
+ break;
+
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+
+ }//switch
+
+ /**
+ * THERE WAS NO TUPLES LEFT TO REPORT IN THIS FRAGMENT. CLOSE SCAN
+ * HAVE NOT BEEN ORDERED. WE CAN CONTINUE THE SCAN PROCESS IMMEDIATELY.
+ * THE COMPLETED STATUS MUST BE TRUE SINCE IT IS NOT CLOSED. IF IT WAS
+ * FALSE IT MUST HAVE BEEN MORE TUPLES TO SCAN AND AT LEAST ONE OF
+ * THOSE SHOULD HAVE BEEN REPORTED.
+ */
+ if (scanFragptr.p->scanFragCompletedStatus == ZFALSE) {
+ jam();
+ /**
+ * THE SENDING NODE IS OUT OF ORDER WE WILL KILL IT BY SENDING SYSTEM
+ * ERROR TO IT
+ */
+ const BlockReference errRef =
+ calcNdbCntrBlockRef(scanFragptr.p->scanFragNodeId);
+ SystemError * const sysErr = (SystemError*)&signal->theData[0];
+ sysErr->errorCode = SystemError::ScanfragStateError;
+ sysErr->errorRef = reference();
+ sendSignal(errRef, GSN_SYSTEM_ERROR, signal,
+ SystemError::SignalLength, JBA);
+ return;
+ }//if
+ returnFromQueuedDeliveryLab(signal);
+}//Dbtc::execSCAN_FRAGCONF()
+
+void Dbtc::returnFromQueuedDeliveryLab(Signal* signal)
+{
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+
+ switch(scanFragptr.p->scanFragCompletedStatus) {
+ case ZFALSE:
+ {
+ /*********************************************************************
+ * WE HAVE SENT THE SCANNED OPERATION TO THE APPLICATION AND WE HAVE
+ * RECEIVED THE ORDER TO CONTINUE SCANNING. THE CURRENT FRAGMENT STILL
+ * CONTAINS MORE TUPLES TO SCAN.
+ *********************************************************************/
+ jam();
+ scanFragptr.p->scanFragState = ScanFragRec::LQH_ACTIVE;
+ ScanFragNextReq * nextReq = (ScanFragNextReq*)&signal->theData[0];
+ nextReq->senderData = scanFragptr.i;
+ nextReq->closeFlag = ZFALSE;
+ nextReq->transId1 = apiConnectptr.p->transid[0];
+ nextReq->transId2 = apiConnectptr.p->transid[1];
+ sendSignal(scanFragptr.p->lqhBlockref, GSN_SCAN_NEXTREQ, signal,
+ ScanFragNextReq::SignalLength, JBB);
+ }
+ break;
+
+ case ZTRUE:
+ {
+ /*********************************************************************
+ * WE HAVE SENT THE SCANNED OPERATION TO THE APPLICATION AND WE HAVE
+ * RECEIVED THE ORDER TO CONTINUE SCANNING. THE CURRENT FRAGMENT HAVE
+ * BEEN COMPLETELY SCANNED AND WE ARE READY TO CLOSE IT.
+ *********************************************************************/
+ jam();
+ scanFragptr.p->scanFragState = ScanFragRec::LQH_ACTIVE_CLOSE;
+ ScanFragNextReq * nextReq = (ScanFragNextReq*)&signal->theData[0];
+ nextReq->senderData = scanFragptr.i;
+ nextReq->closeFlag = ZTRUE;
+ nextReq->transId1 = apiConnectptr.p->transid[0];
+ nextReq->transId2 = apiConnectptr.p->transid[1];
+ sendSignal(scanFragptr.p->lqhBlockref, GSN_SCAN_NEXTREQ, signal,
+ ScanFragNextReq::SignalLength, JBB);
+ }
+ break;
+
+ case ZCLOSED:
+ {
+ /********************************************************************
+ * THE SCANNED FRAGMENT HAVE BEEN CLOSED. IF CLOSE SCAN HAVE BEEN
+ * ORDERED THEN WE CAN REPORT THAT THIS SCAN PROCESS IS COMPLETED.
+ * ALSO IF THERE ARE NO MORE FRAGMENTS TO SCAN WE CAN REPORT THAT
+ * THE SCAN PROCESS IS COMPLETED.
+ ********************************************************************/
+ jam();
+ scanFragptr.p->lqhBlockref = RNIL;
+ if ((scanptr.p->scanState != ScanRecord::CLOSING_SCAN) &&
+ (scanptr.p->scanNextFragId < scanptr.p->scanNoFrag)){
+ jam();
+ scanFragptr.p->scanFragState = ScanFragRec::WAIT_GET_PRIMCONF;
+ tcConnectptr.i = scanptr.p->scanTcrec;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ scanFragptr.p->scanFragId = scanptr.p->scanNextFragId;
+ scanptr.p->scanNextFragId++;
+ signal->theData[0] = tcConnectptr.p->dihConnectptr;
+ signal->theData[1] = scanFragptr.i;
+ signal->theData[2] = scanptr.p->scanTableref;
+ signal->theData[3] = scanFragptr.p->scanFragId;
+ sendSignal(cdihblockref, GSN_DIGETPRIMREQ, signal, 4, JBB);
+ } else {
+ jam();
+ sendScanProcConf(signal);
+ }//if
+ }
+ break;
+
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+
+ updateBuddyTimer(apiConnectptr);
+ scanFragptr.p->startFragTimer(ctcTimer);
+}//Dbtc::returnFromQueuedDeliveryLab()
+
+/**********************************************************
+ * execSCAN_PROCCONF
+ **********************************************************/
+void Dbtc::execSCAN_PROCCONF(Signal* signal)
+{
+ jamEntry();
+
+ scanptr.i = signal->theData[0];
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ scanptr.p->scanProcessesCompleted++;
+ ndbassert(scanptr.p->scanProcessesCompleted <= scanptr.p->scanParallel);
+
+ scanFragptr.i = signal->theData[1];
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+ scanFragptr.p->stopFragTimer();
+ scanFragptr.p->scanFragState = ScanFragRec::COMPLETED;
+
+ if (scanptr.p->scanProcessesCompleted == scanptr.p->scanParallel) {
+ jam();
+
+ // Check that all scan processes are in state COMPLETED
+ for (Uint32 i = 0; i < 16; i++) {
+ scanFragptr.i = scanptr.p->scanFragrec[i];
+ if (scanFragptr.i != RNIL) {
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+ ndbrequire(scanFragptr.p->scanFragState == ScanFragRec::COMPLETED);
+ }
+ }
+
+ // ALL SCAN PROCESSES HAS COMPLETED
+ scanptr.p->scanCompletedStatus = ZTRUE;
+ switch (scanptr.p->scanState) {
+
+ case ScanRecord::CLOSING_SCAN:
+ jam();
+ if (scanptr.p->apiIsClosed == true) {
+ jam();
+ /*
+ * The API has either failed or ordered a close of this scan
+ * it's resources should be released and a response sent
+ */
+ scanReleaseResourcesLab(signal);
+ return;
+ }//if
+
+ /**
+ * The close have been performed but the API is still alive and not
+ * expecting a response, keep resources until API fails or it orders
+ * a close
+ */
+ return;
+ case ScanRecord::SCAN_NEXT_ORDERED:
+ jam();
+ /**
+ * The scan is completed and api is waiting for a response.
+ * Reslease resources and send a response.
+ */
+ scanReleaseResourcesLab(signal);
+ return;
+ case ScanRecord::DELIVERED:
+ case ScanRecord::QUEUED_DELIVERED:
+ jam();
+ /**
+ * All processes have reported completion, wait for a new request from
+ * API and start close of the scan then.
+ */
+ scanptr.p->scanReceivedOperations = 0;
+ scanptr.p->scanState = ScanRecord::CLOSING_SCAN;
+ return;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+ }
+}//Dbtc::execSCAN_PROCCONF()
+
+
+/****************************************************************************
+ * execSCAN_NEXTREQ
+ *
+ * THE APPLICATION HAVE PROCESSED THE TUPLES TRANSFERRED AND IS NOW READY FOR
+ * MORE. THIS SIGNAL IS ALSO USED TO CLOSE THE SCAN.
+ ****************************************************************************/
+void Dbtc::execSCAN_NEXTREQ(Signal* signal)
+{
+ const ScanNextReq * const req = (ScanNextReq *)&signal->theData[0];
+ const UintR transid1 = req->transId1;
+ const UintR transid2 = req->transId2;
+ const UintR stopScan = req->stopScan;
+
+ jamEntry();
+
+ apiConnectptr.i = req->apiConnectPtr;
+ if (apiConnectptr.i >= capiConnectFilesize) {
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }//if
+ ptrAss(apiConnectptr, apiConnectRecord);
+
+ /**
+ * Check transid
+ */
+ const UintR ctransid1 = apiConnectptr.p->transid[0] ^ transid1;
+ const UintR ctransid2 = apiConnectptr.p->transid[1] ^ transid2;
+ if ((ctransid1 | ctransid2) != 0){
+ ScanTabRef * ref = (ScanTabRef*)&signal->theData[0];
+ ref->apiConnectPtr = apiConnectptr.p->ndbapiConnect;
+ ref->transId1 = transid1;
+ ref->transId2 = transid2;
+ ref->errorCode = ZSTATE_ERROR;
+ sendSignal(signal->senderBlockRef(), GSN_SCAN_TABREF,
+ signal, ScanTabRef::SignalLength, JBB);
+ DEBUG("Wrong transid");
+ return;
+ }
+
+ /**
+ * Check state of API connection
+ */
+ if (apiConnectptr.p->apiConnectstate != CS_START_SCAN) {
+ jam();
+ if (apiConnectptr.p->apiConnectstate == CS_CONNECTED) {
+ jam();
+ /*********************************************************************
+ * The application sends a SCAN_NEXTREQ after experiencing a time-out.
+ * We will send a SCAN_TABREF to indicate a time-out occurred.
+ *********************************************************************/
+ DEBUG("scanTabRefLab: ZSCANTIME_OUT_ERROR2");
+ ndbrequire(false); //B2 indication of strange things going on
+ scanTabRefLab(signal, ZSCANTIME_OUT_ERROR2);
+ return;
+ }
+ DEBUG("scanTabRefLab: ZSTATE_ERROR");
+ DEBUG(" apiConnectstate="<<apiConnectptr.p->apiConnectstate);
+ scanTabRefLab(signal, ZSTATE_ERROR);
+ return;
+ }//if
+
+ /*******************************************************
+ * START THE ACTUAL LOGIC OF SCAN_NEXTREQ.
+ ********************************************************/
+ // Stop the timer that is used to check for timeout in the API
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ scanptr.i = apiConnectptr.p->apiScanRec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+
+ if (scanptr.p->apiIsClosed == true) {
+ jam();
+ /**
+ * The close is already started. Api has failed or
+ * has not responded in time so this signal is not allowed
+ */
+ DEBUG("execSCAN_NEXTREQ: apiIsClosed == true");
+ DEBUG(" apiConnectstate="<<apiConnectptr.p->apiConnectstate);
+ DEBUG(" scanState="<<scanptr.p->scanState);
+ return;
+ }//if
+
+
+ if (scanptr.p->scanState == ScanRecord::CLOSING_SCAN) {
+ jam();
+ /*********************************************************************
+ * WE HAVE STARTED A CLOSE OF THIS SCAN OPERATION. NOW WE CAN REPORT
+ * THIS TO THE APPLICATION. BEFORE WE REPORT IT TO THE APPLICATION WE
+ * MUST COMPLETE THE CLOSE FIRST.
+ *********************************************************************/
+ if (scanptr.p->scanCompletedStatus == ZTRUE) {
+ jam();
+ /*********************************************************************
+ * THE SCAN IS ALREADY COMPLETED. WE ARE NOW READY TO COMPLETE THE SCAN
+ * BY RELEASING ALL RESOURCES AND SENDING THE CONFIRMATION TO THE
+ * APPLICATION.
+ *********************************************************************/
+ scanReleaseResourcesLab(signal);
+ return;
+ } else {
+ jam();
+ /*********************************************************************
+ * THE CLOSE IS ONGOING BUT NOT YET COMPLETED. WE WILL SET THE STATE
+ * TO INDICATE THAT THE APPLICATION IS WAITING FOR THE RESPONSE.
+ *********************************************************************/
+ scanptr.p->apiIsClosed = true;
+ return;
+ }//if
+ }//if
+
+ if (stopScan == ZTRUE) {
+ jam();
+ /*********************************************************************
+ * APPLICATION IS CLOSING THE SCAN.
+ **********************************************************************/
+ scanptr.p->apiIsClosed = true;
+ scanCompletedLab(signal);
+ return;
+ }//if
+
+ /*********************************************************************
+ * THOSE SCAN PROCESSES THAT WAS SENT IN PREVIOUS MESSAGE ARE
+ * ACKNOWLEDGED BY THIS REQUEST FOR MORE SCANNED OPERATIONS. WE CAN
+ * THUS RESTART THOSE SCAN PROCESSES.
+ *********************************************************************/
+ for (Uint32 i = 0; i < 16; i++) {
+ jam();
+ scanFragptr.i = scanptr.p->scanFragrec[i];
+ if (scanFragptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+ if (scanFragptr.p->scanFragState == ScanFragRec::DELIVERED) {
+ jam();
+ scanFragptr.p->scanFragState = ScanFragRec::RETURNING_FROM_DELIVERY;
+ signal->theData[0] = TcContinueB::ZRETURN_FROM_QUEUED_DELIVERY;
+ signal->theData[1] = scanptr.i;
+ signal->theData[2] = scanFragptr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 3, JBB);
+ }
+ }//if
+ }//for
+
+ switch (scanptr.p->scanState) {
+ case ScanRecord::QUEUED_DELIVERED:
+ /*********************************************************************
+ * A NUMBER OF SCAN PROCESSES ARE READY TO DELIVER. DELIVER AND SET
+ * STATE TO DELIVERED. ALSO CONTINUE PROCESS QUEUED SCAN PROCESSES.
+ *********************************************************************/
+ jam();
+ sendScanTabConf(signal);
+ scanptr.p->scanState = ScanRecord::DELIVERED;
+ /*********************************************************************
+ * UPDATE STATUS OF THE SCAN PROCESSES THAT WAS NOW SENT TO THE
+ * APPLICATION TO DELIVERED. PREVIOUSLY THEY WERE QUEUED FOR DELIVERY.
+ *********************************************************************/
+ for (Uint32 i = 0; i < 16; i++) {
+ jam();
+ scanFragptr.i = scanptr.p->scanFragrec[i];
+ if (scanFragptr.i != RNIL) {
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+ if (scanFragptr.p->scanFragState == ScanFragRec::QUEUED_FOR_DELIVERY) {
+ jam();
+ scanFragptr.p->scanFragState = ScanFragRec::DELIVERED;
+ }//if
+ }//if
+ }//for
+ return;
+ case ScanRecord::DELIVERED:
+ jam();
+ /*********************************************************************
+ * WE HAVE NOT ANY QUEUED DELIVERIES. SET STATE TO INDICATE IT IS OK
+ * TO SEND SCAN_TABCONF AS SOON AS ANY FRAGMENT IS READY TO DELIVER.
+ *********************************************************************/
+ scanptr.p->scanState = ScanRecord::SCAN_NEXT_ORDERED;
+ return;
+ case ScanRecord::SCAN_NEXT_ORDERED:
+ jam();
+ /* empty */
+ return;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//switch
+}//Dbtc::execSCAN_NEXTREQ()
+
+void Dbtc::scanCompletedLab(Signal* signal) {
+ scanptr.p->scanReceivedOperations = 0;
+ scanptr.p->scanState = ScanRecord::CLOSING_SCAN;
+
+ // Iterate over all fragment scans and check if
+ // they need to be closed in LQH
+ for (Uint32 i = 0; i < 16; i++) {
+ if (scanptr.p->scanFragrec[i] == RNIL) {
+ jam();
+ continue;
+ }
+ scanFragptr.i = scanptr.p->scanFragrec[i];
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+
+ if (scanFragptr.p->lqhBlockref == RNIL){
+ // The connection to this LQH has been closed
+ jam();
+ continue;
+ }
+
+ if (scanFragptr.p->scanFragCompletedStatus == ZCLOSED){
+ // The fragment scan is already completed
+ jam();
+ continue;
+ }
+
+ if (scanFragptr.p->scanFragState == ScanFragRec::RETURNING_FROM_DELIVERY){
+ // The scan process is soon to continue executing
+ // Set scanFragCompletedStatus to ZTRUE so that LQH is properly closed
+ // when this scan process "returns from delivery"
+ jam();
+ DEBUG("scanCompletedLab: setting scanFragCompletedStatus to ZTRUE");
+ scanFragptr.p->scanFragCompletedStatus = ZTRUE;
+ continue;
+ }
+
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+
+ ScanFragNextReq * nextReq = (ScanFragNextReq*)&signal->theData[0];
+ nextReq->senderData = scanFragptr.i;
+ nextReq->closeFlag = ZTRUE;
+ nextReq->transId1 = apiConnectptr.p->transid[0];
+ nextReq->transId2 = apiConnectptr.p->transid[1];
+ sendSignal(scanFragptr.p->lqhBlockref, GSN_SCAN_NEXTREQ, signal,
+ ScanFragNextReq::SignalLength, JBB);
+ updateBuddyTimer(apiConnectptr);
+
+ updateBuddyTimer(apiConnectptr);
+ scanFragptr.p->startFragTimer(ctcTimer);
+ scanFragptr.p->scanFragState = ScanFragRec::LQH_ACTIVE_CLOSE;
+
+ }//for
+}//Dbtc::scanCompletedLab()
+
+void Dbtc::sendScanProcConf(Signal* signal){
+ signal->theData[0] = scanptr.i;
+ signal->theData[1] = scanFragptr.i;
+ sendSignal(cownref, GSN_SCAN_PROCCONF, signal, 2, JBB);
+}
+
+void Dbtc::releaseScanrec(Signal* signal) {
+ scanptr.p->nextScan = cfirstfreeScanrec;
+ scanptr.p->scanState = ScanRecord::IDLE;
+ scanptr.p->scanTcrec = RNIL;
+ cfirstfreeScanrec = scanptr.i;
+}//Dbtc::releaseScanrec()
+
+void Dbtc::releaseScanFragrec(Signal* signal) {
+ scanFragptr.p->nextScanFrag = cfirstfreeScanFragrec;
+ scanFragptr.p->scanFragState = ScanFragRec::IDLE;
+ cfirstfreeScanFragrec = scanFragptr.i;
+ scanFragptr.p->stopFragTimer();
+}//Dbtc::releaseScanFragrec()
+
+void Dbtc::releaseScanOprec(Signal* signal) {
+ scanOpptr.p->nextScanOp = cfirstfreeScanOprec;
+ cfirstfreeScanOprec = scanOpptr.i;
+}//Dbtc::releaseScanOprec()
+
+void Dbtc::seizeScanrec(Signal* signal) {
+ scanptr.i = cfirstfreeScanrec;
+ ptrCheckGuard(scanptr, cscanrecFileSize, scanRecord);
+ cfirstfreeScanrec = scanptr.p->nextScan;
+ scanptr.p->nextScan = RNIL;
+ ndbrequire(scanptr.p->scanState == ScanRecord::IDLE);
+}//Dbtc::seizeScanrec()
+
+void Dbtc::seizeScanFragrec(Signal* signal) {
+ scanFragptr.i = cfirstfreeScanFragrec;
+ ptrCheckGuard(scanFragptr, cscanFragrecFileSize, scanFragmentRecord);
+ cfirstfreeScanFragrec = scanFragptr.p->nextScanFrag;
+ scanFragptr.p->nextScanFrag = RNIL;
+ ndbrequire(scanFragptr.p->scanFragState == ScanFragRec::IDLE);
+}//Dbtc::seizeScanFragrec()
+
+void Dbtc::seizeScanOprec(Signal* signal) {
+ scanOpptr.i = cfirstfreeScanOprec;
+ ptrCheckGuard(scanOpptr, cscanOprecFileSize, scanOperationRecord);
+ cfirstfreeScanOprec = scanOpptr.p->nextScanOp;
+ scanOpptr.p->nextScanOp = RNIL;
+}//Dbtc::seizeScanOprec()
+
+
+void Dbtc::sendScanFragReq(Signal* signal) {
+ arrGuard(scanFragptr.p->scanFragProcId, 16);
+ scanOpptr.i = scanptr.p->scanOprec[scanFragptr.p->scanFragProcId];
+ ptrCheckGuard(scanOpptr, cscanOprecFileSize, scanOperationRecord);
+
+ Uint32 requestInfo = 0;
+ ScanFragReq::setConcurrency(requestInfo, scanFragptr.p->scanFragConcurrency);
+ ScanFragReq::setLockMode(requestInfo, scanptr.p->scanLockMode);
+ ScanFragReq::setHoldLockFlag(requestInfo, scanptr.p->scanLockHold);
+ if(scanptr.p->scanLockMode == 1){ // Not read -> keyinfo
+ jam();
+ ScanFragReq::setKeyinfoFlag(requestInfo, 1);
+ }
+ ScanFragReq::setReadCommittedFlag(requestInfo, scanptr.p->readCommitted);
+ ScanFragReq::setRangeScanFlag(requestInfo, scanptr.p->rangeScan);
+ ScanFragReq::setAttrLen(requestInfo, scanptr.p->scanAiLength);
+ apiConnectptr.i = scanptr.p->scanApiRec;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ ScanFragReq * const req = (ScanFragReq *)&signal->theData[0];
+ req->senderData = scanFragptr.i;
+ req->resultRef = apiConnectptr.p->ndbapiBlockref;
+ req->requestInfo = requestInfo;
+ req->savePointId = apiConnectptr.p->currSavePointId;
+ req->tableId = scanptr.p->scanTableref;
+ req->fragmentNo = scanFragptr.p->scanFragId;
+ req->schemaVersion = scanptr.p->scanSchemaVersion;
+ req->transId1 = apiConnectptr.p->transid[0];
+ req->transId2 = apiConnectptr.p->transid[1];
+ for(int i = 0; i<16; i++){
+ req->clientOpPtr[i] = scanOpptr.p->apiOpptr[i];
+ }
+ sendSignal(scanFragptr.p->lqhBlockref, GSN_SCAN_FRAGREQ, signal, 25, JBB);
+ updateBuddyTimer(apiConnectptr);
+ scanFragptr.p->startFragTimer(ctcTimer);
+ scanFragptr.p->scanFragCompletedStatus = ZFALSE;
+
+}//Dbtc::sendScanFragReq()
+
+
+void Dbtc::sendScanTabConf(Signal* signal) {
+ jam();
+ /*******************************************************
+ * Send SCAN_TABINFO with information about all
+ * received operations
+ *******************************************************/
+ Int32 operationsToSend = scanptr.p->scanReceivedOperations;
+ Uint32 sstOpIndex = 0;
+
+ while (operationsToSend > 0){
+ jam();
+
+ ScanTabInfo * info = (ScanTabInfo*)&signal->theData[0];
+ info->apiConnectPtr = apiConnectptr.p->ndbapiConnect;
+
+ for (int i = 0; i < 16; i++){
+ jam();
+ arrGuard(sstOpIndex, 16);
+ scanOpptr.i = scanptr.p->scanOprec[sstOpIndex];
+ ptrCheckGuard(scanOpptr, cscanOprecFileSize, scanOperationRecord);
+ info->operLenAndIdx[i] = scanOpptr.p->scanOpLength[i];
+ operationsToSend--;
+ scanOpptr.p->scanOpLength[i] = RNIL;
+ }
+ sstOpIndex++;
+ sendSignal(apiConnectptr.p->ndbapiBlockref,
+ GSN_SCAN_TABINFO, signal, ScanTabInfo::SignalLength, JBB);
+ }
+
+ /********************************************************
+ * Send SCAN_TABCONF signaling that a result set have
+ * been sent to the API
+ *********************************************************/
+ Uint32 requestInfo = 0;
+ ScanTabConf::setOperations(requestInfo, scanptr.p->scanReceivedOperations);
+ ScanTabConf::setScanStatus(requestInfo, scanptr.p->scanCompletedStatus);
+
+ ScanTabConf * conf = (ScanTabConf*)&signal->theData[0];
+ conf->apiConnectPtr = apiConnectptr.p->ndbapiConnect;
+ conf->requestInfo = requestInfo;
+ conf->transId1 = apiConnectptr.p->transid[0];
+ conf->transId2 = apiConnectptr.p->transid[1];
+ sendSignal(apiConnectptr.p->ndbapiBlockref,
+ GSN_SCAN_TABCONF, signal, ScanTabConf::SignalLength, JBB);
+
+ scanptr.p->scanReceivedOperations = 0;
+ // Start the scanRec-timer again and wait for response from the API.
+ setApiConTimer(apiConnectptr.i, ctcTimer, __LINE__);
+ updateBuddyTimer(apiConnectptr);
+}//Dbtc::sendScanTabConf()
+
+
+/*
+ * Write index and length of all operations received into
+ * scanOprec->scanOpLength buffer
+ */
+void Dbtc::setScanReceived(Signal* signal, Uint32 noCompletedOps)
+{
+ UintR tssrIndividual;
+ UintR tssrOprecIndex;
+ UintR tssrLengthPlusIndex;
+ UintR tssrOpIndex;
+
+ ndbrequire(noCompletedOps <= 16);
+ tssrIndividual = scanFragptr.p->scanIndividual;
+ for (Uint32 i = 0; i < noCompletedOps; i++) {
+ jam();
+ tssrOprecIndex = scanptr.p->scanReceivedOperations >> 4;
+ arrGuard(tssrOprecIndex, 16);
+ scanOpptr.i = scanptr.p->scanOprec[tssrOprecIndex];
+ ptrCheckGuard(scanOpptr, cscanOprecFileSize, scanOperationRecord);
+ tssrLengthPlusIndex = tssrIndividual << 24;
+ tssrLengthPlusIndex += cdata[i];
+ tssrOpIndex = scanptr.p->scanReceivedOperations & 15;
+ scanOpptr.p->scanOpLength[tssrOpIndex] = tssrLengthPlusIndex;
+ scanptr.p->scanReceivedOperations++;
+ tssrIndividual++;
+ }//for
+}//Dbtc::setScanReceived()
+
+void Dbtc::gcpTcfinished(Signal* signal)
+{
+ signal->theData[1] = tcheckGcpId;
+ sendSignal(cdihblockref, GSN_GCP_TCFINISHED, signal, 2, JBB);
+}//Dbtc::gcpTcfinished()
+
+void Dbtc::initApiConnect(Signal* signal)
+{
+ Uint32 tiacTmp;
+ Uint32 guard4;
+
+ tiacTmp = capiConnectFilesize / 3;
+ ndbrequire(tiacTmp > 0);
+ guard4 = tiacTmp + 1;
+ for (cachePtr.i = 0; cachePtr.i < guard4; cachePtr.i++) {
+ ptrAss(cachePtr, cacheRecord);
+ cachePtr.p->firstAttrbuf = RNIL;
+ cachePtr.p->lastAttrbuf = RNIL;
+ cachePtr.p->firstKeybuf = RNIL;
+ cachePtr.p->lastKeybuf = RNIL;
+ cachePtr.p->nextCacheRec = cachePtr.i + 1;
+ }//for
+ cachePtr.i = tiacTmp;
+ ptrCheckGuard(cachePtr, ccacheFilesize, cacheRecord);
+ cachePtr.p->nextCacheRec = RNIL;
+ cfirstfreeCacheRec = 0;
+
+ guard4 = tiacTmp - 1;
+ for (apiConnectptr.i = 0; apiConnectptr.i <= guard4; apiConnectptr.i++) {
+ jam();
+ ptrAss(apiConnectptr, apiConnectRecord);
+ apiConnectptr.p->apiConnectstate = CS_DISCONNECTED;
+ apiConnectptr.p->apiFailState = ZFALSE;
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ apiConnectptr.p->takeOverRec = (Uint8)Z8NIL;
+ apiConnectptr.p->cachePtr = RNIL;
+ apiConnectptr.p->nextApiConnect = apiConnectptr.i + 1;
+ apiConnectptr.p->ndbapiBlockref = 0xFFFFFFFF; // Invalid ref
+ apiConnectptr.p->commitAckMarker = RNIL;
+ apiConnectptr.p->firstTcConnect = RNIL;
+ apiConnectptr.p->triggerPending = false;
+ apiConnectptr.p->isIndexOp = false;
+ apiConnectptr.p->accumulatingIndexOp = RNIL;
+ apiConnectptr.p->executingIndexOp = RNIL;
+ apiConnectptr.p->buddyPtr = RNIL;
+ apiConnectptr.p->currSavePointId = 0;
+ }//for
+ apiConnectptr.i = tiacTmp - 1;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ apiConnectptr.p->nextApiConnect = RNIL;
+ cfirstfreeApiConnect = 0;
+ guard4 = (2 * tiacTmp) - 1;
+ for (apiConnectptr.i = tiacTmp; apiConnectptr.i <= guard4; apiConnectptr.i++)
+ {
+ jam();
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ apiConnectptr.p->apiConnectstate = CS_RESTART;
+ apiConnectptr.p->apiFailState = ZFALSE;
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ apiConnectptr.p->takeOverRec = (Uint8)Z8NIL;
+ apiConnectptr.p->cachePtr = RNIL;
+ apiConnectptr.p->nextApiConnect = apiConnectptr.i + 1;
+ apiConnectptr.p->ndbapiBlockref = 0xFFFFFFFF; // Invalid ref
+ apiConnectptr.p->commitAckMarker = RNIL;
+ apiConnectptr.p->firstTcConnect = RNIL;
+ apiConnectptr.p->triggerPending = false;
+ apiConnectptr.p->isIndexOp = false;
+ apiConnectptr.p->accumulatingIndexOp = RNIL;
+ apiConnectptr.p->executingIndexOp = RNIL;
+ apiConnectptr.p->buddyPtr = RNIL;
+ apiConnectptr.p->currSavePointId = 0;
+ }//for
+ apiConnectptr.i = (2 * tiacTmp) - 1;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ apiConnectptr.p->nextApiConnect = RNIL;
+ cfirstfreeApiConnectCopy = tiacTmp;
+ guard4 = (3 * tiacTmp) - 1;
+ for (apiConnectptr.i = 2 * tiacTmp; apiConnectptr.i <= guard4;
+ apiConnectptr.i++) {
+ jam();
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ apiConnectptr.p->apiFailState = ZFALSE;
+ apiConnectptr.p->apiConnectstate = CS_RESTART;
+ apiConnectptr.p->takeOverRec = (Uint8)Z8NIL;
+ apiConnectptr.p->cachePtr = RNIL;
+ apiConnectptr.p->nextApiConnect = apiConnectptr.i + 1;
+ apiConnectptr.p->ndbapiBlockref = 0xFFFFFFFF; // Invalid ref
+ apiConnectptr.p->commitAckMarker = RNIL;
+ apiConnectptr.p->firstTcConnect = RNIL;
+ apiConnectptr.p->triggerPending = false;
+ apiConnectptr.p->isIndexOp = false;
+ apiConnectptr.p->accumulatingIndexOp = RNIL;
+ apiConnectptr.p->executingIndexOp = RNIL;
+ apiConnectptr.p->buddyPtr = RNIL;
+ apiConnectptr.p->currSavePointId = 0;
+ }//for
+ apiConnectptr.i = (3 * tiacTmp) - 1;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ apiConnectptr.p->nextApiConnect = RNIL;
+ cfirstfreeApiConnectFail = 2 * tiacTmp;
+}//Dbtc::initApiConnect()
+
+void Dbtc::initattrbuf(Signal* signal)
+{
+ ndbrequire(cattrbufFilesize > 0);
+ for (attrbufptr.i = 0; attrbufptr.i < cattrbufFilesize; attrbufptr.i++) {
+ jam();
+ ptrAss(attrbufptr, attrbufRecord);
+ attrbufptr.p->attrbuf[ZINBUF_NEXT] = attrbufptr.i + 1; /* NEXT ATTRBUF */
+ }//for
+ attrbufptr.i = cattrbufFilesize - 1;
+ ptrAss(attrbufptr, attrbufRecord);
+ attrbufptr.p->attrbuf[ZINBUF_NEXT] = RNIL; /* NEXT ATTRBUF */
+ cfirstfreeAttrbuf = 0;
+}//Dbtc::initattrbuf()
+
+void Dbtc::initdatabuf(Signal* signal)
+{
+ ndbrequire(cdatabufFilesize > 0);
+ for (databufptr.i = 0; databufptr.i < cdatabufFilesize; databufptr.i++) {
+ ptrAss(databufptr, databufRecord);
+ databufptr.p->nextDatabuf = databufptr.i + 1;
+ }//for
+ databufptr.i = cdatabufFilesize - 1;
+ ptrCheckGuard(databufptr, cdatabufFilesize, databufRecord);
+ databufptr.p->nextDatabuf = RNIL;
+ cfirstfreeDatabuf = 0;
+}//Dbtc::initdatabuf()
+
+void Dbtc::initgcp(Signal* signal)
+{
+ ndbrequire(cgcpFilesize > 0);
+ for (gcpPtr.i = 0; gcpPtr.i < cgcpFilesize; gcpPtr.i++) {
+ ptrAss(gcpPtr, gcpRecord);
+ gcpPtr.p->nextGcp = gcpPtr.i + 1;
+ }//for
+ gcpPtr.i = cgcpFilesize - 1;
+ ptrCheckGuard(gcpPtr, cgcpFilesize, gcpRecord);
+ gcpPtr.p->nextGcp = RNIL;
+ cfirstfreeGcp = 0;
+ cfirstgcp = RNIL;
+ clastgcp = RNIL;
+}//Dbtc::initgcp()
+
+void Dbtc::inithost(Signal* signal)
+{
+ cpackedListIndex = 0;
+ ndbrequire(chostFilesize > 0);
+ for (hostptr.i = 0; hostptr.i < chostFilesize; hostptr.i++) {
+ jam();
+ ptrAss(hostptr, hostRecord);
+ hostptr.p->hostStatus = HS_DEAD;
+ hostptr.p->inPackedList = false;
+ hostptr.p->takeOverStatus = TOS_NOT_DEFINED;
+ hostptr.p->lqhTransStatus = LTS_IDLE;
+ hostptr.p->ndbVersion = ZNIL;
+ hostptr.p->noOfWordsTCKEYCONF = 0;
+ hostptr.p->noOfWordsTCINDXCONF = 0;
+ hostptr.p->noOfPackedWordsLqh = 0;
+ hostptr.p->hostLqhBlockRef = calcLqhBlockRef(hostptr.i);
+ }//for
+}//Dbtc::inithost()
+
+void Dbtc::initialiseRecordsLab(Signal* signal, UintR Tdata0)
+{
+ switch (Tdata0) {
+ case 0:
+ jam();
+ initApiConnect(signal);
+ break;
+ case 1:
+ jam();
+ initattrbuf(signal);
+ break;
+ case 2:
+ jam();
+ initdatabuf(signal);
+ break;
+ case 3:
+ jam();
+ initgcp(signal);
+ break;
+ case 4:
+ jam();
+ inithost(signal);
+ break;
+ case 5:
+ jam();
+ // UNUSED Free to initialise something
+ break;
+ case 6:
+ jam();
+ initTable(signal);
+ break;
+ case 7:
+ jam();
+ initialiseScanrec(signal);
+ break;
+ case 8:
+ jam();
+ initialiseScanOprec(signal);
+ break;
+ case 9:
+ jam();
+ initialiseScanFragrec(signal);
+ break;
+ case 10:
+ jam();
+ initialiseTcConnect(signal);
+ break;
+ case 11:
+ jam();
+ initTcFail(signal);
+ returnInitialiseRecordsLab(signal);
+ return;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ sendInitialiseRecords(signal, (Tdata0 + 1));
+ return;
+}//Dbtc::initialiseRecordsLab()
+
+void Dbtc::initialiseScanrec(Signal* signal)
+{
+ ndbrequire(cscanrecFileSize > 0);
+ for (scanptr.i = 0; scanptr.i < cscanrecFileSize; scanptr.i++) {
+ jam();
+ ptrAss(scanptr, scanRecord);
+ scanptr.p->scanState = ScanRecord::IDLE;
+ scanptr.p->nextScan = scanptr.i + 1;
+ }//for
+ scanptr.i = cscanrecFileSize - 1;
+ ptrAss(scanptr, scanRecord);
+ scanptr.p->nextScan = RNIL;
+ cfirstfreeScanrec = 0;
+}//Dbtc::initialiseScanrec()
+
+void Dbtc::initialiseScanFragrec(Signal* signal)
+{
+ ndbrequire(cscanFragrecFileSize > 0);
+ for (scanFragptr.i = 0; scanFragptr.i < cscanFragrecFileSize;
+ scanFragptr.i++) {
+ jam();
+ ptrAss(scanFragptr, scanFragmentRecord);
+ scanFragptr.p->scanFragState = ScanFragRec::IDLE;
+ scanFragptr.p->stopFragTimer();
+ scanFragptr.p->nextScanFrag = scanFragptr.i + 1;
+ }//for
+ scanFragptr.i = cscanFragrecFileSize - 1;
+ ptrAss(scanFragptr, scanFragmentRecord);
+ scanFragptr.p->nextScanFrag = RNIL;
+ cfirstfreeScanFragrec = 0;
+}//Dbtc::initialiseScanFragrec()
+
+void Dbtc::initialiseScanOprec(Signal* signal)
+{
+ ndbrequire(cscanOprecFileSize > 0);
+ for (scanOpptr.i = 0; scanOpptr.i < cscanOprecFileSize; scanOpptr.i++) {
+ jam();
+ ptrAss(scanOpptr, scanOperationRecord);
+ scanOpptr.p->nextScanOp = scanOpptr.i + 1;
+ }//for
+ scanOpptr.i = cscanOprecFileSize - 1;
+ ptrAss(scanOpptr, scanOperationRecord);
+ scanOpptr.p->nextScanOp = RNIL;
+ cfirstfreeScanOprec = 0;
+ cnoFreeScanOprec = cscanOprecFileSize;
+}//Dbtc::initialiseScanOprec()
+
+void Dbtc::initTable(Signal* signal)
+{
+
+ ndbrequire(ctabrecFilesize > 0);
+ for (tabptr.i = 0; tabptr.i < ctabrecFilesize; tabptr.i++) {
+ ptrAss(tabptr, tableRecord);
+ tabptr.p->currentSchemaVersion = 0;
+ tabptr.p->storedTable = true;
+ tabptr.p->tableType = 0;
+ tabptr.p->enabled = false;
+ tabptr.p->dropping = false;
+ }//for
+}//Dbtc::initTable()
+
+void Dbtc::initialiseTcConnect(Signal* signal)
+{
+ ndbrequire(ctcConnectFilesize >= 2);
+
+ // Place half of tcConnectptr's in cfirstfreeTcConnectFail list
+ Uint32 titcTmp = ctcConnectFilesize / 2;
+ for (tcConnectptr.i = 0; tcConnectptr.i < titcTmp; tcConnectptr.i++) {
+ jam();
+ ptrAss(tcConnectptr, tcConnectRecord);
+ tcConnectptr.p->tcConnectstate = OS_RESTART;
+ tcConnectptr.p->apiConnect = RNIL;
+ tcConnectptr.p->noOfNodes = 0;
+ tcConnectptr.p->nextTcConnect = tcConnectptr.i + 1;
+ }//for
+ tcConnectptr.i = titcTmp - 1;
+ ptrAss(tcConnectptr, tcConnectRecord);
+ tcConnectptr.p->nextTcConnect = RNIL;
+ cfirstfreeTcConnectFail = 0;
+
+ // Place other half in cfirstfreeTcConnect list
+ for (tcConnectptr.i = titcTmp; tcConnectptr.i < ctcConnectFilesize;
+ tcConnectptr.i++) {
+ jam();
+ ptrAss(tcConnectptr, tcConnectRecord);
+ tcConnectptr.p->tcConnectstate = OS_RESTART;
+ tcConnectptr.p->apiConnect = RNIL;
+ tcConnectptr.p->noOfNodes = 0;
+ tcConnectptr.p->nextTcConnect = tcConnectptr.i + 1;
+ }//for
+ tcConnectptr.i = ctcConnectFilesize - 1;
+ ptrAss(tcConnectptr, tcConnectRecord);
+ tcConnectptr.p->nextTcConnect = RNIL;
+ cfirstfreeTcConnect = titcTmp;
+ cconcurrentOp = 0;
+}//Dbtc::initialiseTcConnect()
+
+/* ------------------------------------------------------------------------- */
+/* ---- LINK A GLOBAL CHECKPOINT RECORD INTO THE LIST WITH TRANSACTIONS */
+/* WAITING FOR COMPLETION. */
+/* ------------------------------------------------------------------------- */
+void Dbtc::linkGciInGcilist(Signal* signal)
+{
+ GcpRecordPtr tmpGcpPointer;
+ if (cfirstgcp == RNIL) {
+ jam();
+ cfirstgcp = gcpPtr.i;
+ } else {
+ jam();
+ tmpGcpPointer.i = clastgcp;
+ ptrCheckGuard(tmpGcpPointer, cgcpFilesize, gcpRecord);
+ tmpGcpPointer.p->nextGcp = gcpPtr.i;
+ }//if
+ clastgcp = gcpPtr.i;
+}//Dbtc::linkGciInGcilist()
+
+/* ------------------------------------------------------------------------- */
+/* ------- LINK SECONDARY KEY BUFFER IN OPERATION RECORD ------- */
+/* ------------------------------------------------------------------------- */
+void Dbtc::linkKeybuf(Signal* signal)
+{
+ seizeDatabuf(signal);
+ tmpDatabufptr.i = cachePtr.p->lastKeybuf;
+ cachePtr.p->lastKeybuf = databufptr.i;
+ if (tmpDatabufptr.i == RNIL) {
+ jam();
+ cachePtr.p->firstKeybuf = databufptr.i;
+ } else {
+ jam();
+ ptrCheckGuard(tmpDatabufptr, cdatabufFilesize, databufRecord);
+ tmpDatabufptr.p->nextDatabuf = databufptr.i;
+ }//if
+}//Dbtc::linkKeybuf()
+
+/* ------------------------------------------------------------------------- */
+/* ------- LINK A TC CONNECT RECORD INTO THE API LIST OF TC CONNECTIONS --- */
+/* ------------------------------------------------------------------------- */
+void Dbtc::linkTcInConnectionlist(Signal* signal)
+{
+ /* POINTER FOR THE CONNECT_RECORD */
+ TcConnectRecordPtr ltcTcConnectptr;
+
+ tcConnectptr.p->nextTcConnect = RNIL;
+ ltcTcConnectptr.i = apiConnectptr.p->lastTcConnect;
+ ptrCheck(ltcTcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ apiConnectptr.p->lastTcConnect = tcConnectptr.i;
+ if (ltcTcConnectptr.i == RNIL) {
+ jam();
+ apiConnectptr.p->firstTcConnect = tcConnectptr.i;
+ } else {
+ jam();
+ ptrGuard(ltcTcConnectptr);
+ ltcTcConnectptr.p->nextTcConnect = tcConnectptr.i;
+ }//if
+}//Dbtc::linkTcInConnectionlist()
+
+/*---------------------------------------------------------------------------*/
+/* RELEASE_ABORT_RESOURCES */
+/* THIS CODE RELEASES ALL RESOURCES AFTER AN ABORT OF A TRANSACTION AND ALSO */
+/* SENDS THE ABORT DECISION TO THE APPLICATION. */
+/*---------------------------------------------------------------------------*/
+void Dbtc::releaseAbortResources(Signal* signal)
+{
+ TcConnectRecordPtr rarTcConnectptr;
+
+ cabortCount++;
+ if (apiConnectptr.p->cachePtr != RNIL) {
+ cachePtr.i = apiConnectptr.p->cachePtr;
+ ptrCheckGuard(cachePtr, ccacheFilesize, cacheRecord);
+ releaseAttrinfo(signal);
+ releaseKeys(signal);
+ }//if
+ tcConnectptr.i = apiConnectptr.p->firstTcConnect;
+ while (tcConnectptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ // Clear any markers that were set in CS_RECEIVING state
+ clearCommitAckMarker(apiConnectptr.p, tcConnectptr.p);
+ rarTcConnectptr.i = tcConnectptr.p->nextTcConnect;
+ releaseTcCon(signal);
+ tcConnectptr.i = rarTcConnectptr.i;
+ }//while
+ apiConnectptr.p->firstTcConnect = RNIL;
+ apiConnectptr.p->lastTcConnect = RNIL;
+
+ // MASV let state be CS_ABORTING until all
+ // signals in the "air" have been received. Reset to CS_CONNECTED
+ // will be done when a TCKEYREQ with start flag is recieved
+ // or releaseApiCon is called
+ // apiConnectptr.p->apiConnectstate = CS_CONNECTED;
+ apiConnectptr.p->apiConnectstate = CS_ABORTING;
+ apiConnectptr.p->abortState = AS_IDLE;
+
+ bool ok = false;
+ Uint32 blockRef = apiConnectptr.p->ndbapiBlockref;
+ switch(apiConnectptr.p->returnsignal){
+ case RS_TCROLLBACKCONF:
+ jam();
+ ok = true;
+ signal->theData[0] = apiConnectptr.p->ndbapiConnect;
+ signal->theData[1] = apiConnectptr.p->transid[0];
+ signal->theData[2] = apiConnectptr.p->transid[1];
+ sendSignal(blockRef, GSN_TCROLLBACKCONF, signal, 3, JBB);
+ break;
+ case RS_TCROLLBACKREP:{
+ jam();
+ ok = true;
+ TcRollbackRep * const tcRollbackRep =
+ (TcRollbackRep *) signal->getDataPtr();
+
+ tcRollbackRep->connectPtr = apiConnectptr.p->ndbapiConnect;
+ tcRollbackRep->transId[0] = apiConnectptr.p->transid[0];
+ tcRollbackRep->transId[1] = apiConnectptr.p->transid[1];
+ tcRollbackRep->returnCode = apiConnectptr.p->returncode;
+ sendSignal(blockRef, GSN_TCROLLBACKREP, signal,
+ TcRollbackRep::SignalLength, JBB);
+ }
+ break;
+ case RS_NO_RETURN:
+ jam();
+ ok = true;
+ break;
+ case RS_TCKEYCONF:
+ case RS_TCKEYREF:
+ case RS_TC_COMMITCONF:
+ break;
+ }
+ if(!ok){
+ jam();
+ ndbout_c("returnsignal = %d", apiConnectptr.p->returnsignal);
+ sendSystemError(signal);
+ }//if
+
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ apiConnectptr.p->abortState = AS_IDLE;
+ if (apiConnectptr.p->apiFailState == ZTRUE) {
+ jam();
+ handleApiFailState(signal, apiConnectptr.i);
+ return;
+ }//if
+}//Dbtc::releaseAbortResources()
+
+void Dbtc::releaseApiCon(Signal* signal, UintR TapiConnectPtr)
+{
+ ApiConnectRecordPtr TlocalApiConnectptr;
+
+ TlocalApiConnectptr.i = TapiConnectPtr;
+ ptrCheckGuard(TlocalApiConnectptr, capiConnectFilesize, apiConnectRecord);
+ TlocalApiConnectptr.p->nextApiConnect = cfirstfreeApiConnect;
+ cfirstfreeApiConnect = TlocalApiConnectptr.i;
+ setApiConTimer(TlocalApiConnectptr.i, 0, __LINE__);
+ TlocalApiConnectptr.p->apiConnectstate = CS_DISCONNECTED;
+}//Dbtc::releaseApiCon()
+
+void Dbtc::releaseApiConnectFail(Signal* signal)
+{
+ apiConnectptr.p->apiConnectstate = CS_RESTART;
+ apiConnectptr.p->takeOverRec = (Uint8)Z8NIL;
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ apiConnectptr.p->nextApiConnect = cfirstfreeApiConnectFail;
+ cfirstfreeApiConnectFail = apiConnectptr.i;
+}//Dbtc::releaseApiConnectFail()
+
+void Dbtc::releaseGcp(Signal* signal)
+{
+ ptrGuard(gcpPtr);
+ gcpPtr.p->nextGcp = cfirstfreeGcp;
+ cfirstfreeGcp = gcpPtr.i;
+}//Dbtc::releaseGcp()
+
+void Dbtc::releaseKeys(Signal* signal)
+{
+ UintR Tmp;
+ databufptr.i = cachePtr.p->firstKeybuf;
+ while (databufptr.i != RNIL) {
+ jam();
+ ptrCheckGuard(databufptr, cdatabufFilesize, databufRecord);
+ Tmp = databufptr.p->nextDatabuf;
+ databufptr.p->nextDatabuf = cfirstfreeDatabuf;
+ cfirstfreeDatabuf = databufptr.i;
+ databufptr.i = Tmp;
+ }//while
+ cachePtr.p->firstKeybuf = RNIL;
+ cachePtr.p->lastKeybuf = RNIL;
+}//Dbtc::releaseKeys()
+
+void Dbtc::releaseTcConnectFail(Signal* signal)
+{
+ ptrGuard(tcConnectptr);
+ tcConnectptr.p->nextTcConnect = cfirstfreeTcConnectFail;
+ cfirstfreeTcConnectFail = tcConnectptr.i;
+}//Dbtc::releaseTcConnectFail()
+
+void Dbtc::seizeApiConnect(Signal* signal)
+{
+ if (cfirstfreeApiConnect != RNIL) {
+ jam();
+ terrorCode = ZOK;
+ apiConnectptr.i = cfirstfreeApiConnect; /* ASSIGN A FREE RECORD FROM */
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ cfirstfreeApiConnect = apiConnectptr.p->nextApiConnect;
+ apiConnectptr.p->nextApiConnect = RNIL;
+ setApiConTimer(apiConnectptr.i, 0, __LINE__);
+ apiConnectptr.p->apiConnectstate = CS_CONNECTED; /* STATE OF CONNECTION */
+ } else {
+ jam();
+ terrorCode = ZNO_FREE_API_CONNECTION;
+ }//if
+ apiConnectptr.p->triggerPending = false;
+ apiConnectptr.p->isIndexOp = false;
+}//Dbtc::seizeApiConnect()
+
+void Dbtc::seizeApiConnectFail(Signal* signal)
+{
+ apiConnectptr.i = cfirstfreeApiConnectFail;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ cfirstfreeApiConnectFail = apiConnectptr.p->nextApiConnect;
+}//Dbtc::seizeApiConnectFail()
+
+void Dbtc::seizeDatabuf(Signal* signal)
+{
+ databufptr.i = cfirstfreeDatabuf;
+ ptrCheckGuard(databufptr, cdatabufFilesize, databufRecord);
+ cfirstfreeDatabuf = databufptr.p->nextDatabuf;
+ databufptr.p->nextDatabuf = RNIL;
+}//Dbtc::seizeDatabuf()
+
+void Dbtc::seizeTcConnect(Signal* signal)
+{
+ tcConnectptr.i = cfirstfreeTcConnect;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ cfirstfreeTcConnect = tcConnectptr.p->nextTcConnect;
+ cconcurrentOp++;
+ tcConnectptr.p->isIndexOp = false;
+}//Dbtc::seizeTcConnect()
+
+void Dbtc::seizeTcConnectFail(Signal* signal)
+{
+ tcConnectptr.i = cfirstfreeTcConnectFail;
+ ptrCheckGuard(tcConnectptr, ctcConnectFilesize, tcConnectRecord);
+ cfirstfreeTcConnectFail = tcConnectptr.p->nextTcConnect;
+}//Dbtc::seizeTcConnectFail()
+
+void Dbtc::sendAttrinfo(Signal* signal,
+ UintR TattrinfoPtr,
+ AttrbufRecord * const regAttrPtr,
+ UintR TBref)
+{
+ UintR TdataPos;
+ UintR sig0, sig1, sig2, sig3, sig4, sig5, sig6, sig7;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ TdataPos = regAttrPtr->attrbuf[ZINBUF_DATA_LEN];
+ sig0 = TattrinfoPtr;
+ sig1 = regApiPtr->transid[0];
+ sig2 = regApiPtr->transid[1];
+
+ signal->theData[0] = sig0;
+ signal->theData[1] = sig1;
+ signal->theData[2] = sig2;
+
+ sig0 = regAttrPtr->attrbuf[0];
+ sig1 = regAttrPtr->attrbuf[1];
+ sig2 = regAttrPtr->attrbuf[2];
+ sig3 = regAttrPtr->attrbuf[3];
+ sig4 = regAttrPtr->attrbuf[4];
+ sig5 = regAttrPtr->attrbuf[5];
+ sig6 = regAttrPtr->attrbuf[6];
+ sig7 = regAttrPtr->attrbuf[7];
+
+ signal->theData[3] = sig0;
+ signal->theData[4] = sig1;
+ signal->theData[5] = sig2;
+ signal->theData[6] = sig3;
+ signal->theData[7] = sig4;
+ signal->theData[8] = sig5;
+ signal->theData[9] = sig6;
+ signal->theData[10] = sig7;
+
+ if (TdataPos > 8) {
+ sig0 = regAttrPtr->attrbuf[8];
+ sig1 = regAttrPtr->attrbuf[9];
+ sig2 = regAttrPtr->attrbuf[10];
+ sig3 = regAttrPtr->attrbuf[11];
+ sig4 = regAttrPtr->attrbuf[12];
+ sig5 = regAttrPtr->attrbuf[13];
+ sig6 = regAttrPtr->attrbuf[14];
+
+ jam();
+ signal->theData[11] = sig0;
+ signal->theData[12] = sig1;
+ signal->theData[13] = sig2;
+ signal->theData[14] = sig3;
+ signal->theData[15] = sig4;
+ signal->theData[16] = sig5;
+ signal->theData[17] = sig6;
+
+ if (TdataPos > 15) {
+
+ sig0 = regAttrPtr->attrbuf[15];
+ sig1 = regAttrPtr->attrbuf[16];
+ sig2 = regAttrPtr->attrbuf[17];
+ sig3 = regAttrPtr->attrbuf[18];
+ sig4 = regAttrPtr->attrbuf[19];
+ sig5 = regAttrPtr->attrbuf[20];
+ sig6 = regAttrPtr->attrbuf[21];
+
+ jam();
+ signal->theData[18] = sig0;
+ signal->theData[19] = sig1;
+ signal->theData[20] = sig2;
+ signal->theData[21] = sig3;
+ signal->theData[22] = sig4;
+ signal->theData[23] = sig5;
+ signal->theData[24] = sig6;
+ }//if
+ }//if
+ sendSignal(TBref, GSN_ATTRINFO, signal, TdataPos + 3, JBB);
+}//Dbtc::sendAttrinfo()
+
+void Dbtc::sendContinueTimeOutControl(Signal* signal, Uint32 TapiConPtr)
+{
+ signal->theData[0] = TcContinueB::ZCONTINUE_TIME_OUT_CONTROL;
+ signal->theData[1] = TapiConPtr;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+}//Dbtc::sendContinueTimeOutControl()
+
+/* -------------------------------------------------------------------------
+ * SEND REAL-TIME BREAK DURING INITIALISATION OF VARIABLES DURING
+ * SYSTEM RESTART.
+ * ------------------------------------------------------------------------- */
+void Dbtc::sendInitialiseRecords(Signal* signal, UintR Tnext)
+{
+ signal->theData[0] = TcContinueB::ZINITIALISE_RECORDS;
+ signal->theData[1] = Tnext;
+ signal->theData[2] = 0;
+ sendSignal(DBTC_REF, GSN_CONTINUEB, signal, 3, JBB);
+}//Dbtc::sendInitialiseRecords()
+
+void Dbtc::sendKeyinfo(Signal* signal, BlockReference TBRef, Uint32 len)
+{
+ signal->theData[0] = tcConnectptr.i;
+ signal->theData[1] = apiConnectptr.p->transid[0];
+ signal->theData[2] = apiConnectptr.p->transid[1];
+ signal->theData[3] = cdata[0];
+ signal->theData[4] = cdata[1];
+ signal->theData[5] = cdata[2];
+ signal->theData[6] = cdata[3];
+ signal->theData[7] = cdata[4];
+ signal->theData[8] = cdata[5];
+ signal->theData[9] = cdata[6];
+ signal->theData[10] = cdata[7];
+ signal->theData[11] = cdata[8];
+ signal->theData[12] = cdata[9];
+ signal->theData[13] = cdata[10];
+ signal->theData[14] = cdata[11];
+ signal->theData[15] = cdata[12];
+ signal->theData[16] = cdata[13];
+ signal->theData[17] = cdata[14];
+ signal->theData[18] = cdata[15];
+ signal->theData[19] = cdata[16];
+ signal->theData[20] = cdata[17];
+ signal->theData[21] = cdata[18];
+ signal->theData[22] = cdata[19];
+ sendSignal(TBRef, GSN_KEYINFO, signal, 3 + len, JBB);
+}//Dbtc::sendKeyinfo()
+
+void Dbtc::sendSystemError(Signal* signal)
+{
+ progError(0, 0);
+}//Dbtc::sendSystemError()
+
+/* ========================================================================= */
+/* ------- LINK ACTUAL GCP OUT OF LIST ------- */
+/* ------------------------------------------------------------------------- */
+void Dbtc::unlinkGcp(Signal* signal)
+{
+ if (cfirstgcp == gcpPtr.i) {
+ jam();
+ cfirstgcp = gcpPtr.p->nextGcp;
+ if (gcpPtr.i == clastgcp) {
+ jam();
+ clastgcp = RNIL;
+ }//if
+ } else {
+ jam();
+ /* --------------------------------------------------------------------
+ * WE ARE TRYING TO REMOVE A GLOBAL CHECKPOINT WHICH WAS NOT THE OLDEST.
+ * THIS IS A SYSTEM ERROR.
+ * ------------------------------------------------------------------- */
+ sendSystemError(signal);
+ }//if
+ gcpPtr.p->nextGcp = cfirstfreeGcp;
+ cfirstfreeGcp = gcpPtr.i;
+}//Dbtc::unlinkGcp()
+
+void
+Dbtc::execDUMP_STATE_ORD(Signal* signal)
+{
+ DumpStateOrd * const dumpState = (DumpStateOrd *)&signal->theData[0];
+ if(signal->theData[0] == DumpStateOrd::CommitAckMarkersSize){
+ infoEvent("TC: m_commitAckMarkerPool: %d free size: %d",
+ m_commitAckMarkerPool.getNoOfFree(),
+ m_commitAckMarkerPool.getSize());
+ }
+ if(signal->theData[0] == DumpStateOrd::CommitAckMarkersDump){
+ infoEvent("TC: m_commitAckMarkerPool: %d free size: %d",
+ m_commitAckMarkerPool.getNoOfFree(),
+ m_commitAckMarkerPool.getSize());
+
+ CommitAckMarkerIterator iter;
+ for(m_commitAckMarkerHash.first(iter); iter.curr.i != RNIL;
+ m_commitAckMarkerHash.next(iter)){
+ infoEvent("CommitAckMarker: i = %d (0x%x, 0x%x)"
+ " Api: %d Lghs(%d): %d %d %d %d bucket = %d",
+ iter.curr.i,
+ iter.curr.p->transid1,
+ iter.curr.p->transid2,
+ iter.curr.p->apiNodeId,
+ iter.curr.p->noOfLqhs,
+ iter.curr.p->lqhNodeId[0],
+ iter.curr.p->lqhNodeId[1],
+ iter.curr.p->lqhNodeId[2],
+ iter.curr.p->lqhNodeId[3],
+ iter.bucket);
+ }
+ }
+ // Dump all ScanFragRecs
+ if (dumpState->args[0] == DumpStateOrd::TcDumpAllScanFragRec){
+ Uint32 recordNo = 0;
+ if (signal->getLength() == 1)
+ infoEvent("TC: Dump all ScanFragRec - size: %d",
+ cscanFragrecFileSize);
+ else if (signal->getLength() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ dumpState->args[0] = DumpStateOrd::TcDumpOneScanFragRec;
+ dumpState->args[1] = recordNo;
+ execDUMP_STATE_ORD(signal);
+
+ if (recordNo < cscanFragrecFileSize-1){
+ dumpState->args[0] = DumpStateOrd::TcDumpAllScanFragRec;
+ dumpState->args[1] = recordNo+1;
+ sendSignal(reference(), GSN_DUMP_STATE_ORD, signal, 2, JBB);
+ }
+ }
+
+ // Dump one ScanFragRec
+ if (dumpState->args[0] == DumpStateOrd::TcDumpOneScanFragRec){
+ Uint32 recordNo = RNIL;
+ if (signal->getLength() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ if (recordNo >= cscanFragrecFileSize)
+ return;
+
+ ScanFragRecPtr sfp;
+ sfp.i = recordNo;
+ ptrAss(sfp, scanFragmentRecord);
+ infoEvent("Dbtc::ScanFragRec[%d]: state=%d, status=%d, "
+ "fragid=%d, procid=%d, ",
+ sfp.i,
+ sfp.p->scanFragState,
+ sfp.p->scanFragCompletedStatus,
+ sfp.p->scanFragId,
+ sfp.p->scanFragProcId);
+ infoEvent(" nodeid=%d, ind=%d, concurr=%d, timer=%d, next=%d",
+ sfp.p->scanFragNodeId,
+ sfp.p->scanIndividual,
+ sfp.p->scanFragConcurrency,
+ sfp.p->scanFragTimer,
+ sfp.p->nextScanFrag);
+ }
+
+ // Dump all ScanRecords
+ if (dumpState->args[0] == DumpStateOrd::TcDumpAllScanRec){
+ Uint32 recordNo = 0;
+ if (signal->getLength() == 1)
+ infoEvent("TC: Dump all ScanRecord - size: %d",
+ cscanrecFileSize);
+ else if (signal->getLength() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ dumpState->args[0] = DumpStateOrd::TcDumpOneScanRec;
+ dumpState->args[1] = recordNo;
+ execDUMP_STATE_ORD(signal);
+
+ if (recordNo < cscanrecFileSize-1){
+ dumpState->args[0] = DumpStateOrd::TcDumpAllScanRec;
+ dumpState->args[1] = recordNo+1;
+ sendSignal(reference(), GSN_DUMP_STATE_ORD, signal, 2, JBB);
+ }
+ }
+
+ // Dump all active ScanRecords
+ if (dumpState->args[0] == DumpStateOrd::TcDumpAllActiveScanRec){
+ Uint32 recordNo = 0;
+ if (signal->getLength() == 1)
+ infoEvent("TC: Dump active ScanRecord - size: %d",
+ cscanrecFileSize);
+ else if (signal->getLength() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ ScanRecordPtr sp;
+ sp.i = recordNo;
+ ptrAss(sp, scanRecord);
+ if (sp.p->scanState != ScanRecord::IDLE){
+ dumpState->args[0] = DumpStateOrd::TcDumpOneScanRec;
+ dumpState->args[1] = recordNo;
+ execDUMP_STATE_ORD(signal);
+ }
+
+ if (recordNo < cscanrecFileSize-1){
+ dumpState->args[0] = DumpStateOrd::TcDumpAllActiveScanRec;
+ dumpState->args[1] = recordNo+1;
+ sendSignal(reference(), GSN_DUMP_STATE_ORD, signal, 2, JBB);
+ }
+ }
+
+ // Dump one ScanRecord
+ // and associated ScanFragRec and ApiConnectRecord
+ if (dumpState->args[0] == DumpStateOrd::TcDumpOneScanRec){
+ Uint32 recordNo = RNIL;
+ if (signal->getLength() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ if (recordNo >= cscanrecFileSize)
+ return;
+
+ ScanRecordPtr sp;
+ sp.i = recordNo;
+ ptrAss(sp, scanRecord);
+ infoEvent("Dbtc::ScanRecord[%d]: state=%d, scanOprec=%d, "
+ "nextfrag=%d, nofrag=%d",
+ sp.i,
+ sp.p->scanState,
+ sp.p->noScanOprec,
+ sp.p->scanNextFragId,
+ sp.p->scanNoFrag);
+ infoEvent(" ailen=%d, para=%d, receivedop=%d, noOprePperFrag=%d",
+ sp.p->scanAiLength,
+ sp.p->scanParallel,
+ sp.p->scanReceivedOperations,
+ sp.p->noOprecPerFrag);
+ infoEvent(" schv=%d, tab=%d, sproc=%d, noTI=%d, norecTI=%d",
+ sp.p->scanSchemaVersion,
+ sp.p->scanTableref,
+ sp.p->scanStoredProcId,
+ sp.p->noScanTabInfo,
+ sp.p->scanTabInfoReceived);
+ infoEvent(" apiclosed=%d, noProcCompl=%d, "
+ "complStat=%d, lhold=%d, lmode=%d",
+ sp.p->apiIsClosed,
+ sp.p->scanProcessesCompleted,
+ sp.p->scanCompletedStatus,
+ sp.p->scanLockHold,
+ sp.p->scanLockMode);
+ infoEvent(" apiRec=%d, next=%d",
+ sp.p->scanApiRec, sp.p->nextScan);
+
+ if (sp.p->scanState != ScanRecord::IDLE){
+ // Request dump of ScanFragRec
+ for (Uint32 i = 0; i < 16; i++){
+ if (sp.p->scanFragrec[i] != RNIL){
+ dumpState->args[0] = DumpStateOrd::TcDumpOneScanFragRec;
+ dumpState->args[1] = sp.p->scanFragrec[i];
+ execDUMP_STATE_ORD(signal);
+ }
+ }
+ // Request dump of ApiConnectRecord
+ dumpState->args[0] = DumpStateOrd::TcDumpOneApiConnectRec;
+ dumpState->args[1] = sp.p->scanApiRec;
+ execDUMP_STATE_ORD(signal);
+ }
+
+ }
+
+ // Dump all ApiConnectRecord(s)
+ if (dumpState->args[0] == DumpStateOrd::TcDumpAllApiConnectRec){
+ Uint32 recordNo = 0;
+ if (signal->getLength() == 1)
+ infoEvent("TC: Dump all ApiConnectRecord - size: %d",
+ capiConnectFilesize);
+ else if (signal->getLength() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ dumpState->args[0] = DumpStateOrd::TcDumpOneApiConnectRec;
+ dumpState->args[1] = recordNo;
+ execDUMP_STATE_ORD(signal);
+
+ if (recordNo < capiConnectFilesize-1){
+ dumpState->args[0] = DumpStateOrd::TcDumpAllApiConnectRec;
+ dumpState->args[1] = recordNo+1;
+ sendSignal(reference(), GSN_DUMP_STATE_ORD, signal, 2, JBB);
+ }
+ }
+
+ // Dump one ApiConnectRecord
+ if (dumpState->args[0] == DumpStateOrd::TcDumpOneApiConnectRec){
+ Uint32 recordNo = RNIL;
+ if (signal->getLength() == 2)
+ recordNo = dumpState->args[1];
+ else
+ return;
+
+ if (recordNo >= capiConnectFilesize)
+ return;
+
+ ApiConnectRecordPtr ap;
+ ap.i = recordNo;
+ ptrAss(ap, apiConnectRecord);
+ infoEvent("Dbtc::ApiConnectRecord[%d]: state=%d, abortState=%d, "
+ "apiFailState=%d",
+ ap.i,
+ ap.p->apiConnectstate,
+ ap.p->abortState,
+ ap.p->apiFailState);
+ infoEvent(" transid(0x%x, 0x%x), apiBref=0x%x, scanRec=%d",
+ ap.p->transid[0],
+ ap.p->transid[1],
+ ap.p->ndbapiBlockref,
+ ap.p->apiScanRec);
+ infoEvent(" ctcTimer=%d, apiTimer=%d, counter=%d, retcode=%d, "
+ "retsig=%d",
+ ctcTimer, getApiConTimer(ap.i),
+ ap.p->counter,
+ ap.p->returncode,
+ ap.p->returnsignal);
+ infoEvent(" lqhkeyconfrec=%d, lqhkeyreqrec=%d, "
+ "tckeyrec=%d",
+ ap.p->lqhkeyconfrec,
+ ap.p->lqhkeyreqrec,
+ ap.p->tckeyrec);
+ infoEvent(" next=%d ",
+ ap.p->nextApiConnect);
+ }
+
+ if (dumpState->args[0] == DumpStateOrd::TcSetTransactionTimeout){
+ jam();
+ if(signal->getLength() > 1){
+ set_timeout_value(signal->theData[1]);
+ }
+ }
+}//Dbtc::execDUMP_STATE_ORD()
+
+void Dbtc::execSET_VAR_REQ(Signal* signal)
+{
+
+ SetVarReq* const setVarReq = (SetVarReq*)&signal->theData[0];
+ ConfigParamId var = setVarReq->variable();
+ int val = setVarReq->value();
+
+
+ switch (var) {
+
+ case TransactionInactiveTime:
+ jam();
+ set_appl_timeout_value(val);
+ break;
+ case TransactionDeadlockDetectionTimeout:
+ set_timeout_value(val);
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ case NoOfConcurrentProcessesHandleTakeover:
+ set_no_parallel_takeover(val);
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ default:
+ sendSignal(CMVMI_REF, GSN_SET_VAR_REF, signal, 1, JBB);
+ } // switch
+
+}
+
+void Dbtc::execABORT_ALL_REQ(Signal* signal)
+{
+ jamEntry();
+ AbortAllReq * req = (AbortAllReq*)&signal->theData[0];
+ AbortAllRef * ref = (AbortAllRef*)&signal->theData[0];
+
+ const Uint32 senderData = req->senderData;
+ const BlockReference senderRef = req->senderRef;
+
+ if(getAllowStartTransaction() == true && !getNodeState().getSingleUserMode()){
+ jam();
+
+ ref->senderData = senderData;
+ ref->errorCode = AbortAllRef::InvalidState;
+ sendSignal(senderRef, GSN_ABORT_ALL_REF, signal,
+ AbortAllRef::SignalLength, JBB);
+ return;
+ }
+
+ if(c_abortRec.clientRef != 0){
+ jam();
+
+ ref->senderData = senderData;
+ ref->errorCode = AbortAllRef::AbortAlreadyInProgress;
+ sendSignal(senderRef, GSN_ABORT_ALL_REF, signal,
+ AbortAllRef::SignalLength, JBB);
+ return;
+ }
+
+ if(refToNode(senderRef) != getOwnNodeId()){
+ jam();
+
+ ref->senderData = senderData;
+ ref->errorCode = AbortAllRef::FunctionNotImplemented;
+ sendSignal(senderRef, GSN_ABORT_ALL_REF, signal,
+ AbortAllRef::SignalLength, JBB);
+ return;
+ }
+
+ c_abortRec.clientRef = senderRef;
+ c_abortRec.clientData = senderData;
+ c_abortRec.oldTimeOutValue = ctimeOutValue;
+
+ ctimeOutValue = 0;
+
+ const Uint32 sleepTime = (2 * 10 * ctimeOutCheckDelay + 199) / 200;
+
+ checkAbortAllTimeout(signal, (sleepTime == 0 ? 1 : sleepTime));
+}
+
+void Dbtc::checkAbortAllTimeout(Signal* signal, Uint32 sleepTime)
+{
+
+ ndbrequire(c_abortRec.clientRef != 0);
+
+ if(sleepTime > 0){
+ jam();
+
+ sleepTime -= 1;
+ signal->theData[0] = TcContinueB::ZWAIT_ABORT_ALL;
+ signal->theData[1] = sleepTime;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 200, 2);
+ return;
+ }
+
+ AbortAllConf * conf = (AbortAllConf*)&signal->theData[0];
+ conf->senderData = c_abortRec.clientData;
+ sendSignal(c_abortRec.clientRef, GSN_ABORT_ALL_CONF, signal,
+ AbortAllConf::SignalLength, JBB);
+
+ ctimeOutValue = c_abortRec.oldTimeOutValue;
+ c_abortRec.clientRef = 0;
+}
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* ------------------ TRIGGER AND INDEX HANDLING ------------------ */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+void Dbtc::execCREATE_TRIG_REQ(Signal* signal)
+{
+ jamEntry();
+ CreateTrigReq * const createTrigReq =
+ (CreateTrigReq *)&signal->theData[0];
+ TcDefinedTriggerData* triggerData;
+ DefinedTriggerPtr triggerPtr;
+ BlockReference sender = signal->senderBlockRef();
+
+ releaseSections(signal);
+
+ triggerPtr.i = createTrigReq->getTriggerId();
+ if (ERROR_INSERTED(8033) ||
+ !c_theDefinedTriggers.seizeId(triggerPtr,
+ createTrigReq->getTriggerId())) {
+ CLEAR_ERROR_INSERT_VALUE;
+ // Failed to allocate trigger record
+ CreateTrigRef * const createTrigRef =
+ (CreateTrigRef *)&signal->theData[0];
+
+ createTrigRef->setConnectionPtr(createTrigReq->getConnectionPtr());
+ createTrigRef->setErrorCode(CreateTrigRef::TooManyTriggers);
+ sendSignal(sender, GSN_CREATE_TRIG_REF,
+ signal, CreateTrigRef::SignalLength, JBB);
+ return;
+ }
+
+ triggerData = triggerPtr.p;
+ triggerData->triggerId = createTrigReq->getTriggerId();
+ triggerData->triggerType = createTrigReq->getTriggerType();
+ triggerData->triggerEvent = createTrigReq->getTriggerEvent();
+ triggerData->attributeMask = createTrigReq->getAttributeMask();
+ if (triggerData->triggerType == TriggerType::SECONDARY_INDEX)
+ triggerData->indexId = createTrigReq->getIndexId();
+ CreateTrigConf * const createTrigConf =
+ (CreateTrigConf *)&signal->theData[0];
+
+ createTrigConf->setConnectionPtr(createTrigReq->getConnectionPtr());
+ sendSignal(sender, GSN_CREATE_TRIG_CONF,
+ signal, CreateTrigConf::SignalLength, JBB);
+}
+
+
+void Dbtc::execDROP_TRIG_REQ(Signal* signal)
+{
+ jamEntry();
+ DropTrigReq * const dropTrigReq = (DropTrigReq *)&signal->theData[0];
+ BlockReference sender = signal->senderBlockRef();
+
+ if ((c_theDefinedTriggers.getPtr(dropTrigReq->getTriggerId())) == NULL) {
+ jam();
+ // Failed to find find trigger record
+ DropTrigRef * const dropTrigRef = (DropTrigRef *)&signal->theData[0];
+
+ dropTrigRef->setConnectionPtr(dropTrigReq->getConnectionPtr());
+ dropTrigRef->setErrorCode(DropTrigRef::TriggerNotFound);
+ sendSignal(sender, GSN_DROP_TRIG_REF,
+ signal, DropTrigRef::SignalLength, JBB);
+ return;
+ }
+
+ // Release trigger record
+ c_theDefinedTriggers.release(dropTrigReq->getTriggerId());
+
+ DropTrigConf * const dropTrigConf = (DropTrigConf *)&signal->theData[0];
+
+ dropTrigConf->setConnectionPtr(dropTrigReq->getConnectionPtr());
+ sendSignal(sender, GSN_DROP_TRIG_CONF,
+ signal, DropTrigConf::SignalLength, JBB);
+}
+
+void Dbtc::execCREATE_INDX_REQ(Signal* signal)
+{
+ jamEntry();
+ CreateIndxReq * const createIndxReq =
+ (CreateIndxReq *)signal->getDataPtr();
+ TcIndexData* indexData;
+ TcIndexDataPtr indexPtr;
+ BlockReference sender = signal->senderBlockRef();
+
+ if (ERROR_INSERTED(8034) ||
+ !c_theIndexes.seizeId(indexPtr, createIndxReq->getIndexId())) {
+ CLEAR_ERROR_INSERT_VALUE;
+ // Failed to allocate index record
+ CreateIndxRef * const createIndxRef =
+ (CreateIndxRef *)&signal->theData[0];
+
+ createIndxRef->setConnectionPtr(createIndxReq->getConnectionPtr());
+ createIndxRef->setErrorCode(CreateIndxRef::TooManyIndexes);
+ releaseSections(signal);
+ sendSignal(sender, GSN_CREATE_INDX_REF,
+ signal, CreateIndxRef::SignalLength, JBB);
+ return;
+ }
+ indexData = indexPtr.p;
+ // Indexes always start in state IS_BUILDING
+ // Will become IS_ONLINE in execALTER_INDX_REQ
+ indexData->indexState = IS_BUILDING;
+ indexData->indexId = indexPtr.i;
+ indexData->primaryTableId = createIndxReq->getTableId();
+
+ // So far need only attribute count
+ SegmentedSectionPtr ssPtr;
+ signal->getSection(ssPtr, CreateIndxReq::ATTRIBUTE_LIST_SECTION);
+ SimplePropertiesSectionReader r0(ssPtr, getSectionSegmentPool());
+ r0.reset(); // undo implicit first()
+ if (!r0.getWord(&indexData->attributeList.sz) ||
+ !r0.getWords(indexData->attributeList.id, indexData->attributeList.sz)) {
+ ndbrequire(false);
+ }
+ indexData->primaryKeyPos = indexData->attributeList.sz;
+
+ releaseSections(signal);
+
+ CreateIndxConf * const createIndxConf =
+ (CreateIndxConf *)&signal->theData[0];
+
+ createIndxConf->setConnectionPtr(createIndxReq->getConnectionPtr());
+ createIndxConf->setTableId(createIndxReq->getTableId());
+ createIndxConf->setIndexId(createIndxReq->getIndexId());
+ sendSignal(sender, GSN_CREATE_INDX_CONF,
+ signal, CreateIndxConf::SignalLength, JBB);
+}
+
+void Dbtc::execALTER_INDX_REQ(Signal* signal)
+{
+ jamEntry();
+ AlterIndxReq * const alterIndxReq = (AlterIndxReq *)signal->getDataPtr();
+ TcIndexData* indexData;
+ //BlockReference sender = signal->senderBlockRef();
+ BlockReference sender = (BlockReference) alterIndxReq->getUserRef();
+ Uint32 connectionPtr = alterIndxReq->getConnectionPtr();
+ AlterIndxReq::RequestType requestType = alterIndxReq->getRequestType();
+ Uint32 tableId = alterIndxReq->getTableId();
+ Uint32 indexId = alterIndxReq->getIndexId();
+ bool online = (alterIndxReq->getOnline() == 1) ? true : false;
+
+ if ((indexData = c_theIndexes.getPtr(indexId)) == NULL) {
+ jam();
+ // Failed to find index record
+ AlterIndxRef * const alterIndxRef =
+ (AlterIndxRef *)signal->getDataPtrSend();
+
+ alterIndxRef->setUserRef(reference());
+ alterIndxRef->setConnectionPtr(connectionPtr);
+ alterIndxRef->setRequestType(requestType);
+ alterIndxRef->setTableId(tableId);
+ alterIndxRef->setIndexId(indexId);
+ alterIndxRef->setErrorCode(AlterIndxRef::IndexNotFound);
+ alterIndxRef->setErrorLine(__LINE__);
+ alterIndxRef->setErrorNode(getOwnNodeId());
+ sendSignal(sender, GSN_ALTER_INDX_REF,
+ signal, AlterIndxRef::SignalLength, JBB);
+ return;
+ }
+ // Found index record, alter it's state
+ if (online) {
+ jam();
+ indexData->indexState = IS_ONLINE;
+ } else {
+ jam();
+ indexData->indexState = IS_BUILDING;
+ }//if
+ AlterIndxConf * const alterIndxConf =
+ (AlterIndxConf *)signal->getDataPtrSend();
+
+ alterIndxConf->setUserRef(reference());
+ alterIndxConf->setConnectionPtr(connectionPtr);
+ alterIndxConf->setRequestType(requestType);
+ alterIndxConf->setTableId(tableId);
+ alterIndxConf->setIndexId(indexId);
+ sendSignal(sender, GSN_ALTER_INDX_CONF,
+ signal, AlterIndxConf::SignalLength, JBB);
+}
+
+void Dbtc::execFIRE_TRIG_ORD(Signal* signal)
+{
+ jamEntry();
+ FireTrigOrd * const fireTrigOrd = (FireTrigOrd *)signal->getDataPtr();
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+ ApiConnectRecordPtr transPtr;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+ TcConnectRecordPtr opPtr;
+
+ opPtr.i = fireTrigOrd->getConnectionPtr();
+ ptrCheckGuard(opPtr, ctcConnectFilesize, localTcConnectRecord);
+ transPtr.i = opPtr.p->apiConnect;
+ transPtr.p = &localApiConnectRecord[transPtr.i];
+ if(opPtr.p->triggerError == 0){
+ scheduleFiredTrigger(&transPtr, &opPtr);
+ }
+
+ // If we have received complete info of all fired triggers
+ // then execute the triggers
+ if (++(opPtr.p->noReceivedTriggers) == opPtr.p->noFiredTriggers) {
+ jam();
+ if (opPtr.p->triggerError != 0) {
+ jam();
+ // Abort transaction
+ apiConnectptr.i = transPtr.i;
+ terrorCode = opPtr.p->triggerError;
+ abortErrorLab(signal);
+ return;
+ }//if
+ executeTriggers(signal, &transPtr);
+ }//if
+}
+
+void Dbtc::execTRIG_ATTRINFO(Signal* signal)
+{
+ jamEntry();
+ TrigAttrInfo * const trigAttrInfo = (TrigAttrInfo *)signal->getDataPtr();
+ Uint32 attrInfoLength = signal->getLength() - TrigAttrInfo::StaticLength;
+ const Uint32 *src = trigAttrInfo->getData();
+ TcFiredTriggerData* currentTrigger;
+ FiredTriggerPtr firedTrigPtr;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+ TcConnectRecordPtr opPtr;
+
+ opPtr.i = trigAttrInfo->getConnectionPtr();
+ ptrCheckGuard(opPtr, ctcConnectFilesize, localTcConnectRecord);
+
+ if (opPtr.p->accumulatingTriggerData.p) {
+ jam();
+ // We are already accumulating
+ } else {
+ jam();
+ // Allocate new trigger record
+ ApiConnectRecord *localApiConnectRecord = apiConnectRecord;
+ ApiConnectRecordPtr transPtr;
+
+ transPtr.i = opPtr.p->apiConnect;
+ //transPtr.p = &localApiConnectRecord[transPtr.i];
+ ptrCheckGuard(transPtr, capiConnectFilesize, localApiConnectRecord);
+ if (!c_theFiredTriggerPool.seize(firedTrigPtr)) {
+ jam();
+ // Resource shortage, abort transaction
+ // Mark transaction for abortion
+#ifdef VM_TRACE
+ ndbout_c("Dbtc::execTRIG_ATTRINFO: Failed to seize fired triggers\n");
+ ndbout_c("%u: Trigger error = %u\n", __LINE__, 4000);
+#endif
+ opPtr.p->triggerError = 4000;
+ return;
+ }//if
+ ndbrequire(firedTrigPtr.p->keyValues.isEmpty() &&
+ firedTrigPtr.p->beforeValues.isEmpty() &&
+ firedTrigPtr.p->afterValues.isEmpty());
+ firedTrigPtr.p->triggerId = trigAttrInfo->getTriggerId();
+ opPtr.p->accumulatingTriggerData = firedTrigPtr;
+ firedTrigPtr.p->fireingOperation = opPtr.i;
+ }//if
+ currentTrigger = opPtr.p->accumulatingTriggerData.p;
+ switch (trigAttrInfo->getAttrInfoType()) {
+ case(TrigAttrInfo::PRIMARY_KEY):
+ jam();
+ if (currentTrigger->keyValues.append(src, attrInfoLength) == false) {
+ jam();
+ // Mark transaction for abortion
+#ifdef VM_TRACE
+ ndbout_c("Dbtc::execTRIG_ATTRINFO: Failed to seize keyValues\n");
+ ndbout_c("%u: Trigger error = %u\n", __LINE__, 4000);
+#endif
+ opPtr.p->triggerError = 4000;
+ // Return trigger to pool
+ c_theFiredTriggerPool.release(opPtr.p->accumulatingTriggerData.i);
+ return;
+ }
+ break;
+ case(TrigAttrInfo::BEFORE_VALUES):
+ jam();
+ if (currentTrigger->beforeValues.append(src, attrInfoLength) == false) {
+ jam();
+ // Mark transaction for abortion
+#ifdef VM_TRACE
+ ndbout_c("Dbtc::execTRIG_ATTRINFO: Failed to seize beforeValues\n");
+ ndbout_c("%u: Trigger error = %u\n", __LINE__, 4000);
+#endif
+ opPtr.p->triggerError = 4000;
+ // Return trigger to pool
+ c_theFiredTriggerPool.release(opPtr.p->accumulatingTriggerData.i);
+ return;
+ }
+ break;
+ case(TrigAttrInfo::AFTER_VALUES):
+ jam();
+ if (currentTrigger->afterValues.append(src, attrInfoLength) == false) {
+ jam();
+ // Mark transaction for abortion
+#ifdef VM_TRACE
+ ndbout_c("Dbtc::execTRIG_ATTRINFO: Failed to seize afterValues\n");
+ ndbout_c("%u: Trigger error = %u\n", __LINE__, 4000);
+#endif
+ opPtr.p->triggerError = 4000;
+ // Return trigger to pool
+ c_theFiredTriggerPool.release(opPtr.p->accumulatingTriggerData.i);
+ return;
+ }
+ break;
+ }
+}
+
+void Dbtc::execDROP_INDX_REQ(Signal* signal)
+{
+ jamEntry();
+ DropIndxReq * const dropIndxReq = (DropIndxReq *)signal->getDataPtr();
+ TcIndexData* indexData;
+ BlockReference sender = signal->senderBlockRef();
+
+ if ((indexData = c_theIndexes.getPtr(dropIndxReq->getIndexId())) == NULL) {
+ jam();
+ // Failed to find index record
+ DropIndxRef * const dropIndxRef =
+ (DropIndxRef *)signal->getDataPtrSend();
+
+ dropIndxRef->setConnectionPtr(dropIndxReq->getConnectionPtr());
+ dropIndxRef->setErrorCode(DropIndxRef::IndexNotFound);
+ sendSignal(sender, GSN_DROP_INDX_REF,
+ signal, DropIndxRef::SignalLength, JBB);
+ return;
+ }
+ // Release index record
+ c_theIndexes.release(dropIndxReq->getIndexId());
+
+ DropIndxConf * const dropIndxConf =
+ (DropIndxConf *)signal->getDataPtrSend();
+
+ dropIndxConf->setConnectionPtr(dropIndxReq->getConnectionPtr());
+ sendSignal(sender, GSN_DROP_INDX_CONF,
+ signal, DropIndxConf::SignalLength, JBB);
+}
+
+void Dbtc::execTCINDXREQ(Signal* signal)
+{
+ jamEntry();
+
+ TcIndxReq * const tcIndxReq = (TcIndxReq *)signal->getDataPtr();
+ const UintR TapiIndex = tcIndxReq->apiConnectPtr;
+ Uint32 tcIndxRequestInfo = tcIndxReq->requestInfo;
+ Uint32 startFlag = tcIndxReq->getStartFlag(tcIndxRequestInfo);
+ Uint32 * dataPtr = &tcIndxReq->scanInfo;
+ Uint32 indexBufSize = 8; // Maximum for index in TCINDXREQ
+ Uint32 attrBufSize = 5; // Maximum for attrInfo in TCINDXREQ
+ ApiConnectRecordPtr transPtr;
+ transPtr.i = TapiIndex;
+ if (transPtr.i >= capiConnectFilesize) {
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }//if
+ ptrAss(transPtr, apiConnectRecord);
+ ApiConnectRecord * const regApiPtr = transPtr.p;
+ // Seize index operation
+ TcIndexOperationPtr indexOpPtr;
+ if ((startFlag == 1) &&
+ ((regApiPtr->apiConnectstate == CS_CONNECTED) ||
+ ((regApiPtr->apiConnectstate == CS_ABORTING) &&
+ (regApiPtr->abortState == AS_IDLE)))) {
+ jam();
+ // This is a newly started transaction, clean-up
+ releaseAllSeizedIndexOperations(regApiPtr);
+ }//if
+ if (!seizeIndexOperation(regApiPtr, indexOpPtr)) {
+ jam();
+ // Failed to allocate index operation
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ tcIndxRef->connectPtr = tcIndxReq->senderData;
+ tcIndxRef->transId[0] = regApiPtr->transid[0];
+ tcIndxRef->transId[1] = regApiPtr->transid[1];
+ tcIndxRef->errorCode = 4000;
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TCINDXREF, signal,
+ TcIndxRef::SignalLength, JBB);
+ return;
+ }
+ TcIndexOperation* indexOp = indexOpPtr.p;
+ indexOp->indexOpId = indexOpPtr.i;
+
+ // Save original signal
+ *indexOp->tcIndxReq = *tcIndxReq;
+ indexOp->connectionIndex = TapiIndex;
+ regApiPtr->accumulatingIndexOp = indexOp->indexOpId;
+
+ // If operation is readTupleExclusive or updateTuple then read index
+ // table with exclusive lock
+ Uint32 indexLength = TcIndxReq::getIndexLength(tcIndxRequestInfo);
+ Uint32 attrLength = tcIndxReq->attrLen;
+ indexOp->expectedKeyInfo = indexLength;
+ Uint32 includedIndexLength = MIN(indexLength, indexBufSize);
+ indexOp->expectedAttrInfo = attrLength;
+ Uint32 includedAttrLength = MIN(attrLength, attrBufSize);
+ if (saveINDXKEYINFO(signal,
+ indexOp,
+ dataPtr,
+ includedIndexLength)) {
+ jam();
+ // We have received all we need
+ readIndexTable(signal, regApiPtr, indexOp);
+ return;
+ }
+ dataPtr += includedIndexLength;
+ if (saveINDXATTRINFO(signal,
+ indexOp,
+ dataPtr,
+ includedAttrLength)) {
+ jam();
+ // We have received all we need
+ readIndexTable(signal, regApiPtr, indexOp);
+ return;
+ }
+}
+
+
+void Dbtc::sendTcIndxConf(Signal* signal, UintR TcommitFlag)
+{
+ HostRecordPtr localHostptr;
+ ApiConnectRecord * const regApiPtr = apiConnectptr.p;
+ const UintR TopWords = (UintR)regApiPtr->tcindxrec;
+ localHostptr.i = refToNode(regApiPtr->ndbapiBlockref);
+ const Uint32 type = getNodeInfo(localHostptr.i).m_type;
+ const bool is_api = (type >= NodeInfo::API && type <= NodeInfo::REP);
+ const BlockNumber TblockNum = refToBlock(regApiPtr->ndbapiBlockref);
+ const Uint32 Tmarker = (regApiPtr->commitAckMarker == RNIL ? 0 : 1);
+ ptrAss(localHostptr, hostRecord);
+ UintR TcurrLen = localHostptr.p->noOfWordsTCINDXCONF;
+ UintR confInfo = 0;
+ TcIndxConf::setNoOfOperations(confInfo, (TopWords >> 1));
+ TcIndxConf::setCommitFlag(confInfo, TcommitFlag);
+ TcIndxConf::setMarkerFlag(confInfo, Tmarker);
+ const UintR TpacketLen = 6 + TopWords;
+ regApiPtr->tcindxrec = 0;
+
+ if ((TpacketLen > 25) || !is_api){
+ TcIndxConf * const tcIndxConf = (TcIndxConf *)signal->getDataPtrSend();
+
+ jam();
+ tcIndxConf->apiConnectPtr = regApiPtr->ndbapiConnect;
+ tcIndxConf->gci = regApiPtr->globalcheckpointid;;
+ tcIndxConf->confInfo = confInfo;
+ tcIndxConf->transId1 = regApiPtr->transid[0];
+ tcIndxConf->transId2 = regApiPtr->transid[1];
+ copyFromToLen(&regApiPtr->tcIndxSendArray[0],
+ (UintR*)&tcIndxConf->operations,
+ (UintR)ZTCOPCONF_SIZE);
+ sendSignal(regApiPtr->ndbapiBlockref,
+ GSN_TCINDXCONF, signal, (TpacketLen - 1), JBB);
+ return;
+ } else if (((TcurrLen + TpacketLen) > 25) && (TcurrLen > 0)) {
+ jam();
+ sendPackedTCINDXCONF(signal, localHostptr.p, localHostptr.i);
+ TcurrLen = 0;
+ } else {
+ jam();
+ updatePackedList(signal, localHostptr.p, localHostptr.i);
+ }//if
+// -------------------------------------------------------------------------
+// The header contains the block reference of receiver plus the real signal
+// length - 3, since we have the real signal length plus one additional word
+// for the header we have to do - 4.
+// -------------------------------------------------------------------------
+ UintR Tpack0 = (TblockNum << 16) + (TpacketLen - 4);
+ UintR Tpack1 = regApiPtr->ndbapiConnect;
+ UintR Tpack2 = regApiPtr->globalcheckpointid;
+ UintR Tpack3 = confInfo;
+ UintR Tpack4 = regApiPtr->transid[0];
+ UintR Tpack5 = regApiPtr->transid[1];
+
+ localHostptr.p->noOfWordsTCINDXCONF = TcurrLen + TpacketLen;
+
+ localHostptr.p->packedWordsTCINDXCONF[TcurrLen + 0] = Tpack0;
+ localHostptr.p->packedWordsTCINDXCONF[TcurrLen + 1] = Tpack1;
+ localHostptr.p->packedWordsTCINDXCONF[TcurrLen + 2] = Tpack2;
+ localHostptr.p->packedWordsTCINDXCONF[TcurrLen + 3] = Tpack3;
+ localHostptr.p->packedWordsTCINDXCONF[TcurrLen + 4] = Tpack4;
+ localHostptr.p->packedWordsTCINDXCONF[TcurrLen + 5] = Tpack5;
+
+ UintR Ti;
+ for (Ti = 6; Ti < TpacketLen; Ti++) {
+ localHostptr.p->packedWordsTCINDXCONF[TcurrLen + Ti] =
+ regApiPtr->tcIndxSendArray[Ti - 6];
+ }//for
+}//Dbtc::sendTcIndxConf()
+
+void Dbtc::execINDXKEYINFO(Signal* signal)
+{
+ jamEntry();
+ Uint32 keyInfoLength = signal->getLength() - IndxKeyInfo::HeaderLength;
+ IndxKeyInfo * const indxKeyInfo = (IndxKeyInfo *)signal->getDataPtr();
+ const Uint32 *src = indxKeyInfo->getData();
+ const UintR TconnectIndex = indxKeyInfo->connectPtr;
+ ApiConnectRecordPtr transPtr;
+ transPtr.i = TconnectIndex;
+ if (transPtr.i >= capiConnectFilesize) {
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }//if
+ ptrAss(transPtr, apiConnectRecord);
+ ApiConnectRecord * const regApiPtr = transPtr.p;
+ TcIndexOperationPtr indexOpPtr;
+ TcIndexOperation* indexOp;
+
+ indexOpPtr.i = regApiPtr->accumulatingIndexOp;
+ indexOp = c_theIndexOperations.getPtr(indexOpPtr.i);
+ if (saveINDXKEYINFO(signal,
+ indexOp,
+ src,
+ keyInfoLength)) {
+ jam();
+ // We have received all we need
+ readIndexTable(signal, regApiPtr, indexOp);
+ }
+}
+
+void Dbtc::execINDXATTRINFO(Signal* signal)
+{
+ jamEntry();
+ Uint32 attrInfoLength = signal->getLength() - IndxAttrInfo::HeaderLength;
+ IndxAttrInfo * const indxAttrInfo = (IndxAttrInfo *)signal->getDataPtr();
+ const Uint32 *src = indxAttrInfo->getData();
+ const UintR TconnectIndex = indxAttrInfo->connectPtr;
+ ApiConnectRecordPtr transPtr;
+ transPtr.i = TconnectIndex;
+ if (transPtr.i >= capiConnectFilesize) {
+ jam();
+ warningHandlerLab(signal);
+ return;
+ }//if
+ ptrAss(transPtr, apiConnectRecord);
+ ApiConnectRecord * const regApiPtr = transPtr.p;
+ TcIndexOperationPtr indexOpPtr;
+ TcIndexOperation* indexOp;
+
+ indexOpPtr.i = regApiPtr->accumulatingIndexOp;
+ indexOp = c_theIndexOperations.getPtr(indexOpPtr.i);
+ if (saveINDXATTRINFO(signal,
+ indexOp,
+ src,
+ attrInfoLength)) {
+ jam();
+ // We have received all we need
+ readIndexTable(signal, regApiPtr, indexOp);
+ }
+}
+
+/**
+ * Save signal INDXKEYINFO
+ * Return true if we have received all needed data
+ */
+bool Dbtc::saveINDXKEYINFO(Signal* signal,
+ TcIndexOperation* indexOp,
+ const Uint32 *src,
+ Uint32 len)
+{
+ if (!indexOp->keyInfo.append(src, len)) {
+ jam();
+ // Failed to seize keyInfo, abort transaction
+#ifdef VM_TRACE
+ ndbout_c("Dbtc::saveINDXKEYINFO: Failed to seize keyinfo\n");
+#endif
+ // Abort transaction
+ apiConnectptr.i = indexOp->connectionIndex;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ releaseIndexOperation(apiConnectptr.p, indexOp);
+ terrorCode = 4000;
+ abortErrorLab(signal);
+ return true;
+ }
+ if (receivedAllINDXKEYINFO(indexOp) && receivedAllINDXATTRINFO(indexOp)) {
+ jam();
+ return true;
+ }
+ return false;
+}
+
+bool Dbtc::receivedAllINDXKEYINFO(TcIndexOperation* indexOp)
+{
+ return (indexOp->keyInfo.getSize() == indexOp->expectedKeyInfo);
+}
+
+/**
+ * Save signal INDXATTRINFO
+ * Return true if we have received all needed data
+ */
+bool Dbtc::saveINDXATTRINFO(Signal* signal,
+ TcIndexOperation* indexOp,
+ const Uint32 *src,
+ Uint32 len)
+{
+ if (!indexOp->attrInfo.append(src, len)) {
+ jam();
+#ifdef VM_TRACE
+ ndbout_c("Dbtc::saveINDXATTRINFO: Failed to seize attrInfo\n");
+#endif
+ apiConnectptr.i = indexOp->connectionIndex;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ releaseIndexOperation(apiConnectptr.p, indexOp);
+ terrorCode = 4000;
+ abortErrorLab(signal);
+ return true;
+ }
+ if (receivedAllINDXKEYINFO(indexOp) && receivedAllINDXATTRINFO(indexOp)) {
+ jam();
+ return true;
+ }
+ return false;
+}
+
+bool Dbtc::receivedAllINDXATTRINFO(TcIndexOperation* indexOp)
+{
+ return (indexOp->attrInfo.getSize() == indexOp->expectedAttrInfo);
+}
+
+bool Dbtc::saveTRANSID_AI(Signal* signal,
+ TcIndexOperation* indexOp,
+ const Uint32 *src,
+ Uint32 len)
+{
+ Uint32 currentTransIdAILength = indexOp->transIdAI.getSize();
+
+ if (currentTransIdAILength == 0) {
+ jam();
+ // Read first AttributeHeader to get expected size
+ // of the single key attribute expected
+ AttributeHeader* head = (AttributeHeader *) src;
+ indexOp->expectedTransIdAI = head->getHeaderSize() + head->getDataSize();
+ }
+ if (!indexOp->transIdAI.append(src, len)) {
+ jam();
+#ifdef VM_TRACE
+ ndbout_c("Dbtc::saveTRANSID_AI: Failed to seize transIdAI\n");
+#endif
+ apiConnectptr.i = indexOp->connectionIndex;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ releaseIndexOperation(apiConnectptr.p, indexOp);
+ terrorCode = 4000;
+ abortErrorLab(signal);
+ return false;
+ }
+ return true;
+}
+
+bool Dbtc::receivedAllTRANSID_AI(TcIndexOperation* indexOp)
+{
+ return (indexOp->transIdAI.getSize() == indexOp->expectedTransIdAI);
+}
+
+/**
+ * Receive signal TCINDXCONF
+ * This can be either the return of reading an index table
+ * or performing an index operation
+ */
+void Dbtc::execTCKEYCONF(Signal* signal)
+{
+ TcKeyConf * const tcKeyConf = (TcKeyConf *)signal->getDataPtr();
+ TcIndexOperationPtr indexOpPtr;
+
+ jamEntry();
+ indexOpPtr.i = tcKeyConf->apiConnectPtr;
+ TcIndexOperation* indexOp = c_theIndexOperations.getPtr(indexOpPtr.i);
+ Uint32 confInfo = tcKeyConf->confInfo;
+
+ /**
+ * Check on TCKEYCONF wheater the the transaction was committed
+ */
+ Uint32 Tcommit = TcKeyConf::getCommitFlag(confInfo);
+
+ indexOpPtr.p = indexOp;
+ if (!indexOp) {
+ jam();
+ // Missing index operation
+ return;
+ }
+ const UintR TconnectIndex = indexOp->connectionIndex;
+ ApiConnectRecord * const regApiPtr = &apiConnectRecord[TconnectIndex];
+ switch(indexOp->indexOpState) {
+ case(IOS_NOOP): {
+ jam();
+ // Should never happen, abort
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ tcIndxRef->connectPtr = indexOp->tcIndxReq->senderData;
+ tcIndxRef->transId[0] = regApiPtr->transid[0];
+ tcIndxRef->transId[1] = regApiPtr->transid[1];
+ tcIndxRef->errorCode = 4349;
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TCINDXREF, signal,
+ TcIndxRef::SignalLength, JBB);
+ return;
+ }
+ case(IOS_INDEX_ACCESS): {
+ jam();
+ // Wait for TRANSID_AI
+ indexOp->indexOpState = IOS_INDEX_ACCESS_WAIT_FOR_TRANSID_AI;
+ break;
+ }
+ case(IOS_INDEX_ACCESS_WAIT_FOR_TRANSID_AI): {
+ jam();
+ // Double TCKEYCONF, should never happen, abort
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ tcIndxRef->connectPtr = indexOp->tcIndxReq->senderData;
+ tcIndxRef->transId[0] = regApiPtr->transid[0];
+ tcIndxRef->transId[1] = regApiPtr->transid[1];
+ tcIndxRef->errorCode = 4349;
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TCINDXREF, signal,
+ TcIndxRef::SignalLength, JBB);
+ return;
+ }
+ case(IOS_INDEX_ACCESS_WAIT_FOR_TCKEYCONF): {
+ jam();
+ // Continue with index operation
+ executeIndexOperation(signal, regApiPtr, indexOp);
+ break;
+ }
+ case(IOS_INDEX_OPERATION): {
+ // We are done, send TCINDXCONF
+ jam();
+ Uint32 Ttcindxrec = regApiPtr->tcindxrec;
+ // Copy reply from TcKeyConf
+
+ regApiPtr->noIndexOp--; // Decrease count
+ regApiPtr->tcIndxSendArray[Ttcindxrec] = indexOp->tcIndxReq->senderData;
+ regApiPtr->tcIndxSendArray[Ttcindxrec + 1] =
+ tcKeyConf->operations[0].attrInfoLen;
+ regApiPtr->tcindxrec = Ttcindxrec + 2;
+ if (regApiPtr->noIndexOp == 0) {
+ jam();
+ sendTcIndxConf(signal, Tcommit);
+ } else if (regApiPtr->tcindxrec == ZTCOPCONF_SIZE) {
+ jam();
+ sendTcIndxConf(signal, 0);
+ }
+ releaseIndexOperation(regApiPtr, indexOp);
+ break;
+ }
+ }
+}
+
+void Dbtc::execTCKEYREF(Signal* signal)
+{
+ TcKeyRef * const tcKeyRef = (TcKeyRef *)signal->getDataPtr();
+ TcIndexOperationPtr indexOpPtr;
+
+ jamEntry();
+ indexOpPtr.i = tcKeyRef->connectPtr;
+ TcIndexOperation* indexOp = c_theIndexOperations.getPtr(indexOpPtr.i);
+ indexOpPtr.p = indexOp;
+ if (!indexOp) {
+ jam();
+ // Missing index operation
+ return;
+ }
+ const UintR TconnectIndex = indexOp->connectionIndex;
+ ApiConnectRecord * const regApiPtr = &apiConnectRecord[TconnectIndex];
+ Uint32 tcKeyRequestInfo = indexOp->tcIndxReq->requestInfo;
+ Uint32 commitFlg = TcKeyReq::getCommitFlag(tcKeyRequestInfo);
+
+ switch(indexOp->indexOpState) {
+ case(IOS_NOOP): {
+ jam();
+ // Should never happen, abort
+ break;
+ }
+ case(IOS_INDEX_ACCESS):
+ case(IOS_INDEX_ACCESS_WAIT_FOR_TRANSID_AI):
+ case(IOS_INDEX_ACCESS_WAIT_FOR_TCKEYCONF): {
+ jam();
+ // If we fail index access for a non-read operation during commit
+ // we abort transaction
+ if (commitFlg == 1) {
+ jam();
+ releaseIndexOperation(regApiPtr, indexOp);
+ apiConnectptr.i = indexOp->connectionIndex;
+ ptrCheckGuard(apiConnectptr, capiConnectFilesize, apiConnectRecord);
+ terrorCode = tcKeyRef->errorCode;
+ abortErrorLab(signal);
+ break;
+ }
+ // else continue
+ }
+ case(IOS_INDEX_OPERATION): {
+ // Send TCINDXREF
+
+ jam();
+ TcIndxReq * const tcIndxReq = indexOp->tcIndxReq;
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ regApiPtr->noIndexOp--; // Decrease count
+ tcIndxRef->connectPtr = tcIndxReq->senderData;
+ tcIndxRef->transId[0] = tcKeyRef->transId[0];
+ tcIndxRef->transId[1] = tcKeyRef->transId[1];
+ tcIndxRef->errorCode = tcKeyRef->errorCode;
+ sendSignal(regApiPtr->ndbapiBlockref,
+ GSN_TCINDXREF, signal, TcIndxRef::SignalLength, JBB);
+ return;
+ }
+ }
+}
+
+void Dbtc::execTRANSID_AI_R(Signal* signal){
+ TransIdAI * const transIdAI = (TransIdAI *)signal->getDataPtr();
+ Uint32 sigLen = signal->length();
+ Uint32 dataLen = sigLen - TransIdAI::HeaderLength - 1;
+ Uint32 recBlockref = transIdAI->attrData[dataLen];
+
+ jamEntry();
+
+ /**
+ * Forward signal to final destination
+ * Truncate last word since that was used to hold the final dest.
+ */
+ sendSignal(recBlockref, GSN_TRANSID_AI,
+ signal, sigLen - 1, JBB);
+}
+
+void Dbtc::execKEYINFO20_R(Signal* signal){
+ KeyInfo20 * const keyInfo = (KeyInfo20 *)signal->getDataPtr();
+ Uint32 sigLen = signal->length();
+ Uint32 dataLen = sigLen - KeyInfo20::HeaderLength - 1;
+ Uint32 recBlockref = keyInfo->keyData[dataLen];
+
+ jamEntry();
+
+ /**
+ * Forward signal to final destination
+ * Truncate last word since that was used to hold the final dest.
+ */
+ sendSignal(recBlockref, GSN_KEYINFO20,
+ signal, sigLen - 1, JBB);
+}
+
+
+void Dbtc::execTRANSID_AI(Signal* signal)
+{
+ TransIdAI * const transIdAI = (TransIdAI *)signal->getDataPtr();
+
+ jamEntry();
+ TcIndexOperationPtr indexOpPtr;
+ indexOpPtr.i = transIdAI->connectPtr;
+ TcIndexOperation* indexOp = c_theIndexOperations.getPtr(indexOpPtr.i);
+ indexOpPtr.p = indexOp;
+ if (!indexOp) {
+ jam();
+ // Missing index operation
+ }
+ const UintR TconnectIndex = indexOp->connectionIndex;
+ // ApiConnectRecord * const regApiPtr = &apiConnectRecord[TconnectIndex];
+ ApiConnectRecordPtr transPtr;
+
+ transPtr.i = TconnectIndex;
+ ptrCheckGuard(transPtr, capiConnectFilesize, apiConnectRecord);
+ ApiConnectRecord * const regApiPtr = transPtr.p;
+
+ // Acccumulate attribute data
+ if (!saveTRANSID_AI(signal,
+ indexOp,
+ transIdAI->getData(),
+ signal->getLength() - TransIdAI::HeaderLength)) {
+ jam();
+ // Failed to allocate space for TransIdAI
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ tcIndxRef->connectPtr = indexOp->tcIndxReq->senderData;
+ tcIndxRef->transId[0] = regApiPtr->transid[0];
+ tcIndxRef->transId[1] = regApiPtr->transid[1];
+ tcIndxRef->errorCode = 4000;
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TCINDXREF, signal,
+ TcIndxRef::SignalLength, JBB);
+ return;
+ }
+
+ switch(indexOp->indexOpState) {
+ case(IOS_NOOP): {
+ jam();
+ // Should never happen, abort
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ tcIndxRef->connectPtr = indexOp->tcIndxReq->senderData;
+ tcIndxRef->transId[0] = regApiPtr->transid[0];
+ tcIndxRef->transId[1] = regApiPtr->transid[1];
+ tcIndxRef->errorCode = 4349;
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TCINDXREF, signal,
+ TcIndxRef::SignalLength, JBB);
+ return;
+ break;
+ }
+ case(IOS_INDEX_ACCESS): {
+ jam();
+ // Check if all TRANSID_AI have been received
+ if (receivedAllTRANSID_AI(indexOp)) {
+ jam();
+ // Wait for TRANSID_AI
+ indexOp->indexOpState = IOS_INDEX_ACCESS_WAIT_FOR_TCKEYCONF;
+ }
+ break;
+ }
+ case(IOS_INDEX_ACCESS_WAIT_FOR_TCKEYCONF): {
+ jam();
+#ifdef VM_TRACE
+ ndbout_c("Dbtc::execTRANSID_AI: Too many TRANSID_AI, ignore for now\n");
+#endif
+ /*
+ // Too many TRANSID_AI
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ tcIndexRef->connectPtr = indexOp->tcIndxReq->senderData;
+ tcIndxRef->transId[0] = regApiPtr->transid[0];
+ tcIndxRef->transId[1] = regApiPtr->transid[1];
+ tcIndxRef->errorCode = 4349;
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TCINDXREF, signal,
+ TcIndxRef::SignalLength, JBB);
+ */
+ break;
+ }
+ case(IOS_INDEX_ACCESS_WAIT_FOR_TRANSID_AI): {
+ jam();
+ // Check if all TRANSID_AI have been received
+ if (receivedAllTRANSID_AI(indexOp)) {
+ jam();
+ // Continue with index operation
+ executeIndexOperation(signal, regApiPtr, indexOp);
+ }
+ // else continue waiting for more TRANSID_AI
+ break;
+ }
+ case(IOS_INDEX_OPERATION): {
+ // Should never receive TRANSID_AI in this state!!
+ jam();
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ tcIndxRef->connectPtr = indexOp->tcIndxReq->senderData;
+ tcIndxRef->transId[0] = regApiPtr->transid[0];
+ tcIndxRef->transId[1] = regApiPtr->transid[1];
+ tcIndxRef->errorCode = 4349;
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TCINDXREF, signal,
+ TcIndxRef::SignalLength, JBB);
+ return;
+ }
+ }
+}
+
+void Dbtc::execTCROLLBACKREP(Signal* signal)
+{
+ TcRollbackRep* tcRollbackRep = (TcRollbackRep *)signal->getDataPtr();
+ jamEntry();
+ TcIndexOperationPtr indexOpPtr;
+ indexOpPtr.i = tcRollbackRep->connectPtr;
+ TcIndexOperation* indexOp = c_theIndexOperations.getPtr(indexOpPtr.i);
+ indexOpPtr.p = indexOp;
+ tcRollbackRep = (TcRollbackRep *)signal->getDataPtrSend();
+ tcRollbackRep->connectPtr = indexOp->tcIndxReq->senderData;
+ sendSignal(apiConnectptr.p->ndbapiBlockref,
+ GSN_TCROLLBACKREP, signal, TcRollbackRep::SignalLength, JBB);
+}
+
+/**
+ * Read index table with the index attributes as PK
+ */
+void Dbtc::readIndexTable(Signal* signal,
+ ApiConnectRecord* regApiPtr,
+ TcIndexOperation* indexOp)
+{
+ Uint32 keyBufSize = 8; // Maximum for key in TCKEYREQ
+ Uint32 dataPos = 0;
+ TcKeyReq * const tcKeyReq = (TcKeyReq *)signal->getDataPtrSend();
+ Uint32 * dataPtr = &tcKeyReq->scanInfo;
+ Uint32 tcKeyLength = TcKeyReq::StaticLength;
+ Uint32 tcKeyRequestInfo = indexOp->tcIndxReq->requestInfo;
+ AttributeBuffer::DataBufferIterator keyIter;
+ Uint32 keyLength = TcKeyReq::getKeyLength(tcKeyRequestInfo);
+ TcIndexData* indexData;
+ Uint32 transId1 = indexOp->tcIndxReq->transId1;
+ Uint32 transId2 = indexOp->tcIndxReq->transId2;
+
+ const Uint8 opType = TcKeyReq::getOperationType(tcKeyRequestInfo);
+
+ // Find index table
+ if ((indexData = c_theIndexes.getPtr(indexOp->tcIndxReq->indexId)) == NULL) {
+ jam();
+ // Failed to find index record
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ tcIndxRef->connectPtr = indexOp->tcIndxReq->senderData;
+ tcIndxRef->transId[0] = regApiPtr->transid[0];
+ tcIndxRef->transId[1] = regApiPtr->transid[1];
+ tcIndxRef->errorCode = 4000;
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TCINDXREF, signal,
+ TcIndxRef::SignalLength, JBB);
+ return;
+ }
+ tcKeyReq->transId1 = transId1;
+ tcKeyReq->transId2 = transId2;
+ tcKeyReq->tableId = indexData->indexId;
+ tcKeyLength += MIN(keyLength, keyBufSize);
+ tcKeyReq->tableSchemaVersion = indexOp->tcIndxReq->indexSchemaVersion;
+ TcKeyReq::setOperationType(tcKeyRequestInfo,
+ opType == ZREAD ? opType : ZREAD_EX);
+ TcKeyReq::setAIInTcKeyReq(tcKeyRequestInfo, 1); // Allways send one AttrInfo
+ TcKeyReq::setExecutingTrigger(tcKeyRequestInfo, 0);
+ BlockReference originalReceiver = regApiPtr->ndbapiBlockref;
+ regApiPtr->ndbapiBlockref = reference(); // Send result to me
+ tcKeyReq->senderData = indexOp->indexOpId;
+ indexOp->indexOpState = IOS_INDEX_ACCESS;
+ regApiPtr->executingIndexOp = regApiPtr->accumulatingIndexOp;
+ regApiPtr->accumulatingIndexOp = RNIL;
+ regApiPtr->isIndexOp = true;
+
+ Uint32 remainingKey = indexOp->keyInfo.getSize();
+ bool moreKeyData = indexOp->keyInfo.first(keyIter);
+ // *********** KEYINFO in TCKEYREQ ***********
+ while((dataPos < keyBufSize) &&
+ (remainingKey-- != 0)) {
+ *dataPtr++ = *keyIter.data;
+ dataPos++;
+ moreKeyData = indexOp->keyInfo.next(keyIter);
+ }
+ // *********** ATTRINFO in TCKEYREQ ***********
+ tcKeyReq->attrLen = 1; // Primary key is stored as one attribute
+ AttributeHeader::init(dataPtr, indexData->primaryKeyPos, 0);
+ tcKeyLength++;
+ tcKeyReq->requestInfo = tcKeyRequestInfo;
+ EXECUTE_DIRECT(DBTC, GSN_TCKEYREQ, signal, tcKeyLength);
+
+ /**
+ * "Fool" TC not to start commiting transaction since it always will
+ * have one outstanding lqhkeyreq
+ * This is later decreased when the index read is complete
+ */
+ regApiPtr->lqhkeyreqrec++;
+
+ /**
+ * Remember ptr to index read operation
+ * (used to set correct save point id on index operation later)
+ */
+ indexOp->indexReadTcConnect = regApiPtr->lastTcConnect;
+
+ jamEntry();
+ // *********** KEYINFO ***********
+ if (moreKeyData) {
+ jam();
+ // Send KEYINFO sequence
+ KeyInfo * const keyInfo = (KeyInfo *)signal->getDataPtrSend();
+
+ keyInfo->connectPtr = indexOp->tcIndxReq->apiConnectPtr;
+ keyInfo->transId[0] = transId1;
+ keyInfo->transId[1] = transId2;
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ dataPos = 0;
+ while(remainingKey-- != 0) {// If we have not read complete key
+ *dataPtr++ = *keyIter.data;
+ dataPos++;
+ if (dataPos == KeyInfo::DataLength) {
+ // Flush KEYINFO
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength);
+ jamEntry();
+ dataPos = 0;
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ }
+ moreKeyData = indexOp->keyInfo.next(keyIter);
+ }
+ if (dataPos != 0) {
+ // Flush last KEYINFO
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + dataPos);
+ jamEntry();
+ }
+ }
+
+ regApiPtr->ndbapiBlockref = originalReceiver; // reset original receiver
+}
+
+/**
+ * Execute the index operation with the result from
+ * the index table read as PK
+ */
+void Dbtc::executeIndexOperation(Signal* signal,
+ ApiConnectRecord* regApiPtr,
+ TcIndexOperation* indexOp) {
+
+ Uint32 keyBufSize = 8; // Maximum for key in TCKEYREQ
+ Uint32 attrBufSize = 5;
+ Uint32 dataPos = 0;
+ TcIndxReq * const tcIndxReq = indexOp->tcIndxReq;
+ TcKeyReq * const tcKeyReq = (TcKeyReq *)signal->getDataPtrSend();
+ Uint32 * dataPtr = &tcKeyReq->scanInfo;
+ Uint32 tcKeyLength = TcKeyReq::StaticLength;
+ Uint32 tcKeyRequestInfo = tcIndxReq->requestInfo;
+ TcIndexData* indexData;
+ AttributeBuffer::DataBufferIterator attrIter;
+ AttributeBuffer::DataBufferIterator aiIter;
+ bool moreKeyData = indexOp->transIdAI.first(aiIter);
+
+ // Find index table
+ if ((indexData = c_theIndexes.getPtr(tcIndxReq->indexId)) == NULL) {
+ jam();
+ // Failed to find index record
+ TcIndxRef * const tcIndxRef = (TcIndxRef *)signal->getDataPtrSend();
+
+ tcIndxRef->connectPtr = indexOp->tcIndxReq->senderData;
+ tcIndxRef->transId[0] = regApiPtr->transid[0];
+ tcIndxRef->transId[1] = regApiPtr->transid[1];
+ tcIndxRef->errorCode = 4349;
+ sendSignal(regApiPtr->ndbapiBlockref, GSN_TCINDXREF, signal,
+ TcIndxRef::SignalLength, JBB);
+ return;
+ }
+ // Find schema version of primary table
+ TableRecordPtr tabPtr;
+ tabPtr.i = indexData->primaryTableId;
+ ptrCheckGuard(tabPtr, ctabrecFilesize, tableRecord);
+
+ tcKeyReq->apiConnectPtr = tcIndxReq->apiConnectPtr;
+ tcKeyReq->attrLen = tcIndxReq->attrLen;
+ tcKeyReq->tableId = indexData->primaryTableId;
+ tcKeyReq->tableSchemaVersion = tabPtr.p->currentSchemaVersion;
+ tcKeyReq->transId1 = regApiPtr->transid[0];
+ tcKeyReq->transId2 = regApiPtr->transid[1];
+ tcKeyReq->senderData = tcIndxReq->senderData; // Needed for TRANSID_AI to API
+ indexOp->indexOpState = IOS_INDEX_OPERATION;
+ regApiPtr->isIndexOp = true;
+ regApiPtr->executingIndexOp = indexOp->indexOpId;;
+ regApiPtr->noIndexOp++; // Increase count
+
+ // Filter out AttributeHeader:s since this should not be in key
+ AttributeHeader* attrHeader = (AttributeHeader *) aiIter.data;
+
+ Uint32 headerSize = attrHeader->getHeaderSize();
+ Uint32 keySize = attrHeader->getDataSize();
+ TcKeyReq::setKeyLength(tcKeyRequestInfo, keySize);
+ // Skip header
+ if (headerSize == 1) {
+ jam();
+ moreKeyData = indexOp->transIdAI.next(aiIter);
+ } else {
+ jam();
+ moreKeyData = indexOp->transIdAI.next(aiIter, headerSize - 1);
+ }//if
+ while(// If we have not read complete key
+ (keySize != 0) &&
+ (dataPos < keyBufSize)) {
+ *dataPtr++ = *aiIter.data;
+ dataPos++;
+ keySize--;
+ moreKeyData = indexOp->transIdAI.next(aiIter);
+ }
+ tcKeyLength += dataPos;
+
+ Uint32 attributesLength = indexOp->attrInfo.getSize();
+ if (attributesLength <= attrBufSize) {
+ jam();
+ // ATTRINFO fits in TCKEYREQ
+ // Pack ATTRINFO IN TCKEYREQ
+ TcKeyReq::setAIInTcKeyReq(tcKeyRequestInfo, indexOp->attrInfo.getSize());
+ // Insert IndxAttrInfo
+ for(bool moreAttrData = indexOp->attrInfo.first(attrIter);
+ moreAttrData;
+ moreAttrData = indexOp->attrInfo.next(attrIter)) {
+ *dataPtr++ = *attrIter.data;
+ }
+ tcKeyLength += attributesLength;
+ } else {
+ jam();
+ // No ATTRINFO in TCKEYREQ
+ TcKeyReq::setAIInTcKeyReq(tcKeyRequestInfo, 0);
+ }
+
+ TcKeyReq::setCommitFlag(tcKeyRequestInfo, 0);
+ TcKeyReq::setExecuteFlag(tcKeyRequestInfo, 0);
+ TcKeyReq::setExecutingTrigger(tcKeyRequestInfo, 0);
+ tcKeyReq->requestInfo = tcKeyRequestInfo;
+
+ /**
+ * Decrease lqhkeyreqrec to compensate for addition
+ * during read of index table
+ * I.e. let TC start committing when other operations has completed
+ */
+ regApiPtr->lqhkeyreqrec--;
+
+ /**
+ * Fix savepoint id -
+ * fix so that index operation has the same savepoint id
+ * as the read of the index table (TCINDXREQ)
+ */
+ TcConnectRecordPtr tmp;
+ tmp.i = indexOp->indexReadTcConnect;
+ ptrCheckGuard(tmp, ctcConnectFilesize, tcConnectRecord);
+ const Uint32 currSavePointId = regApiPtr->currSavePointId;
+ regApiPtr->currSavePointId = tmp.p->savePointId;
+ EXECUTE_DIRECT(DBTC, GSN_TCKEYREQ, signal, tcKeyLength);
+ regApiPtr->currSavePointId = currSavePointId;
+
+ jamEntry();
+ // *********** KEYINFO ***********
+ if (moreKeyData) {
+ jam();
+ // Send KEYINFO sequence
+ KeyInfo * const keyInfo = (KeyInfo *)signal->getDataPtrSend();
+
+ keyInfo->connectPtr = indexOp->tcIndxReq->apiConnectPtr;
+ keyInfo->transId[0] = regApiPtr->transid[0];
+ keyInfo->transId[1] = regApiPtr->transid[1];
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ dataPos = 0;
+ // Pack any part of a key attribute that did no fit TCKEYREQ
+ while(keySize-- != 0) {// If we have not read complete key
+ *dataPtr++ = *aiIter.data;
+ dataPos++;
+ if (dataPos == KeyInfo::DataLength) {
+ // Flush KEYINFO
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength);
+ jamEntry();
+ dataPos = 0;
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ }
+ moreKeyData = indexOp->transIdAI.next(aiIter);
+ }
+ if (dataPos != 0) {
+ // Flush last KEYINFO
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + dataPos);
+ jamEntry();
+ }
+ }
+
+ // *********** ATTRINFO ***********
+ if (attributesLength > attrBufSize) {
+ jam();
+ // No ATTRINFO in TcKeyReq
+ TcKeyReq::setAIInTcKeyReq(tcKeyReq->requestInfo, 0);
+ // Send ATTRINFO sequence
+ AttrInfo * const attrInfo = (AttrInfo *)signal->getDataPtrSend();
+ Uint32 attrInfoPos = 0;
+
+ attrInfo->connectPtr = indexOp->tcIndxReq->apiConnectPtr;
+ attrInfo->transId[0] = regApiPtr->transid[0];
+ attrInfo->transId[1] = regApiPtr->transid[1];
+ dataPtr = (Uint32 *) &attrInfo->attrData;
+
+
+ // Insert attribute values (insert key values of primary table)
+ for(bool moreAttrData = indexOp->attrInfo.first(attrIter);
+ moreAttrData;
+ moreAttrData = indexOp->attrInfo.next(attrIter)) {
+ *dataPtr++ = *attrIter.data;
+ attrInfoPos++;
+ if (attrInfoPos == AttrInfo::DataLength) {
+ // Flush ATTRINFO
+ EXECUTE_DIRECT(DBTC, GSN_ATTRINFO, signal,
+ AttrInfo::HeaderLength + AttrInfo::DataLength);
+ jamEntry();
+ attrInfoPos = 0;
+ dataPtr = (Uint32 *) &attrInfo->attrData;
+ }
+ }
+ if (attrInfoPos != 0) {
+ // Send last ATTRINFO
+ EXECUTE_DIRECT(DBTC, GSN_ATTRINFO, signal,
+ AttrInfo::HeaderLength + attrInfoPos);
+ jamEntry();
+ }
+ }
+}
+
+bool Dbtc::seizeIndexOperation(ApiConnectRecord* regApiPtr,
+ TcIndexOperationPtr& indexOpPtr)
+{
+ bool seizeOk;
+
+ seizeOk = c_theIndexOperations.seize(indexOpPtr);
+ if (seizeOk) {
+ jam();
+ TcSeizedIndexOperationPtr seizedIndexOpPtr;
+ seizeOk &= regApiPtr->theSeizedIndexOperations.seizeId(seizedIndexOpPtr,
+ indexOpPtr.i);
+ }
+ return seizeOk;
+}
+
+void Dbtc::releaseIndexOperation(ApiConnectRecord* regApiPtr,
+ TcIndexOperation* indexOp)
+{
+ indexOp->indexOpState = IOS_NOOP;
+ indexOp->expectedKeyInfo = 0;
+ indexOp->keyInfo.release();
+ indexOp->expectedAttrInfo = 0;
+ indexOp->attrInfo.release();
+ indexOp->expectedTransIdAI = 0;
+ indexOp->transIdAI.release();
+ regApiPtr->theSeizedIndexOperations.release(indexOp->indexOpId);
+ c_theIndexOperations.release(indexOp->indexOpId);
+}
+
+void Dbtc::releaseAllSeizedIndexOperations(ApiConnectRecord* regApiPtr)
+{
+ TcSeizedIndexOperationPtr seizedIndexOpPtr;
+
+ regApiPtr->theSeizedIndexOperations.first(seizedIndexOpPtr);
+ while(seizedIndexOpPtr.i != RNIL) {
+ jam();
+ TcIndexOperation* indexOp =
+ c_theIndexOperations.getPtr(seizedIndexOpPtr.i);
+
+ indexOp->indexOpState = IOS_NOOP;
+ indexOp->expectedKeyInfo = 0;
+ indexOp->keyInfo.release();
+ indexOp->expectedAttrInfo = 0;
+ indexOp->attrInfo.release();
+ indexOp->expectedTransIdAI = 0;
+ indexOp->transIdAI.release();
+ c_theIndexOperations.release(seizedIndexOpPtr.i);
+ regApiPtr->theSeizedIndexOperations.next(seizedIndexOpPtr);
+ }
+ regApiPtr->theSeizedIndexOperations.release();
+}
+
+void Dbtc::saveTriggeringOpState(Signal* signal, TcConnectRecord* trigOp)
+{
+ LqhKeyConf * lqhKeyConf = (LqhKeyConf *)signal->getDataPtr();
+ copyFromToLen((UintR*)lqhKeyConf,
+ &trigOp->savedState[0],
+ LqhKeyConf::SignalLength);
+}
+
+void Dbtc::restoreTriggeringOpState(Signal* signal, TcConnectRecord* trigOp)
+{
+ LqhKeyConf * lqhKeyConf = (LqhKeyConf *)signal->getDataPtr();
+ copyFromToLen(&trigOp->savedState[0],
+ (UintR*)lqhKeyConf,
+ LqhKeyConf::SignalLength);
+ lqhKeyConf->noFiredTriggers = 0;
+}
+
+void Dbtc::continueTriggeringOp(Signal* signal, TcConnectRecord* trigOp)
+{
+ restoreTriggeringOpState(signal, trigOp);
+ trigOp->noReceivedTriggers = 0;
+ if (trigOp->triggerError != 0) {
+ // A trigger operation has failed
+ LqhKeyConf * lqhKeyConf = (LqhKeyConf *)signal->getDataPtr();
+ LqhKeyRef * lqhKeyRef = (LqhKeyRef *)signal->getDataPtrSend();
+ // Copy fields to avoid overwrite
+ Uint32 opPtr = lqhKeyConf->opPtr;
+ Uint32 userRef = lqhKeyConf->userRef;
+ Uint32 transId1 = lqhKeyConf->transId1;
+ Uint32 transId2 = lqhKeyConf->transId2;
+
+ lqhKeyRef->connectPtr = opPtr;
+ lqhKeyRef->userRef = userRef;
+ if (trigOp->triggerError == 630) { // Tuple already existed
+ jam();
+ lqhKeyRef->errorCode = 893; // Constraint violation
+ } else {
+ jam();
+ lqhKeyRef->errorCode = trigOp->triggerError;
+ }//if
+ lqhKeyRef->transId1 = transId1;
+ lqhKeyRef->transId2 = transId2;
+ handleFailedOperation(signal, lqhKeyRef, false);
+ } else {
+ jam();
+ // All triggers executed successfully, continue operation
+ execLQHKEYCONF(signal);
+ }//if
+}
+
+void Dbtc::scheduleFiredTrigger(ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr)
+{
+ // Set initial values for trigger fireing operation
+ opPtr->p->triggerExecutionCount++;
+ opPtr->p->triggerError = 0;
+ // Insert fired trigger in execution queue
+ transPtr->p->theFiredTriggers.add(opPtr->p->accumulatingTriggerData);
+ opPtr->p->accumulatingTriggerData.i = RNIL;
+ opPtr->p->accumulatingTriggerData.p = NULL;
+}
+
+void Dbtc::executeTriggers(Signal* signal, ApiConnectRecordPtr* transPtr)
+{
+ ApiConnectRecord* regApiPtr = transPtr->p;
+ TcConnectRecord *localTcConnectRecord = tcConnectRecord;
+ TcConnectRecordPtr opPtr;
+ FiredTriggerPtr trigPtr;
+
+ if (!regApiPtr->theFiredTriggers.isEmpty()) {
+ jam();
+ if ((regApiPtr->apiConnectstate == CS_STARTED) ||
+ (regApiPtr->apiConnectstate == CS_START_COMMITTING)) {
+ jam();
+ regApiPtr->theFiredTriggers.first(trigPtr);
+ while (trigPtr.i != RNIL) {
+ jam();
+ // Execute all ready triggers in parallel
+ opPtr.i = trigPtr.p->fireingOperation;
+ ptrCheckGuard(opPtr, ctcConnectFilesize, localTcConnectRecord);
+ FiredTriggerPtr nextTrigPtr;
+ nextTrigPtr.i = trigPtr.i;
+ nextTrigPtr.p = trigPtr.p;
+ regApiPtr->theFiredTriggers.next(nextTrigPtr);
+ if (opPtr.p->noReceivedTriggers == opPtr.p->noFiredTriggers) {
+ jam();
+ // Fireing operation is ready to have a trigger executing
+ executeTrigger(signal, trigPtr.p, transPtr, &opPtr);
+ // Should allow for interleaving here by sending a CONTINUEB and
+ // return
+ // Release trigger records
+ trigPtr.p->keyValues.release();
+ trigPtr.p->beforeValues.release();
+ trigPtr.p->afterValues.release();
+ regApiPtr->theFiredTriggers.release(trigPtr.i);
+ }
+ trigPtr = nextTrigPtr;
+ }
+ return;
+ // No more triggers, continue transaction after last executed trigger has
+ // reurned (in execLQHKEYCONF or execLQHKEYREF)
+ } else {
+ // Wait until transaction is ready to execute a trigger
+ jam();
+ if (!regApiPtr->triggerPending) {
+ jam();
+ regApiPtr->triggerPending = true;
+ signal->theData[0] = TcContinueB::TRIGGER_PENDING;
+ signal->theData[1] = transPtr->i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 3, JBB);
+ }
+ // else
+ // We are already waiting for a pending trigger (CONTINUEB)
+ }
+ }
+}
+
+void Dbtc::executeTrigger(Signal* signal,
+ TcFiredTriggerData* firedTriggerData,
+ ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr)
+{
+ TcDefinedTriggerData* definedTriggerData;
+
+ if ((definedTriggerData =
+ c_theDefinedTriggers.getPtr(firedTriggerData->triggerId))
+ != NULL) {
+ switch(definedTriggerData->triggerType) {
+ case(TriggerType::SECONDARY_INDEX):
+ jam();
+ executeIndexTrigger(signal, definedTriggerData, firedTriggerData,
+ transPtr, opPtr);
+ break;
+ default:
+ ndbrequire(false);
+ }
+ }
+}
+
+void Dbtc::executeIndexTrigger(Signal* signal,
+ TcDefinedTriggerData* definedTriggerData,
+ TcFiredTriggerData* firedTriggerData,
+ ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr)
+{
+ TcIndexData* indexData;
+
+ indexData = c_theIndexes.getPtr(definedTriggerData->indexId);
+ ndbassert(indexData != NULL);
+
+ switch (definedTriggerData->triggerEvent) {
+ case(TriggerEvent::TE_INSERT): {
+ jam();
+ insertIntoIndexTable(signal, firedTriggerData, transPtr, opPtr, indexData);
+ break;
+ }
+ case(TriggerEvent::TE_DELETE): {
+ jam();
+ deleteFromIndexTable(signal, firedTriggerData, transPtr, opPtr, indexData);
+ break;
+ }
+ case(TriggerEvent::TE_UPDATE): {
+ jam();
+ deleteFromIndexTable(signal, firedTriggerData, transPtr, opPtr,
+ indexData, true); // Hold the triggering operation
+ insertIntoIndexTable(signal, firedTriggerData, transPtr, opPtr, indexData);
+ break;
+ }
+ default:
+ ndbrequire(false);
+ }
+}
+
+void Dbtc::releaseFiredTriggerData(DLFifoList<TcFiredTriggerData>* triggers)
+{
+ FiredTriggerPtr trigPtr;
+
+ triggers->first(trigPtr);
+ while (trigPtr.i != RNIL) {
+ jam();
+ // Release trigger records
+ trigPtr.p->keyValues.release();
+ trigPtr.p->beforeValues.release();
+ trigPtr.p->afterValues.release();
+ triggers->next(trigPtr);
+ }
+ triggers->release();
+}
+
+void Dbtc::insertIntoIndexTable(Signal* signal,
+ TcFiredTriggerData* firedTriggerData,
+ ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr,
+ TcIndexData* indexData,
+ bool holdOperation)
+{
+ ApiConnectRecord* regApiPtr = transPtr->p;
+ TcConnectRecord* opRecord = opPtr->p;
+ TcKeyReq * const tcKeyReq = (TcKeyReq *)signal->getDataPtrSend();
+ Uint32 tcKeyRequestInfo = 0;
+ Uint32 tcKeyLength = TcKeyReq::StaticLength;
+ TableRecordPtr indexTabPtr;
+ AttributeBuffer::DataBufferIterator iter;
+ Uint32 attrId = 0;
+ Uint32 keyLength = 0;
+ Uint32 totalPrimaryKeyLength = 0;
+ Uint32 hops;
+
+ indexTabPtr.i = indexData->indexId;
+ ptrCheckGuard(indexTabPtr, ctabrecFilesize, tableRecord);
+ tcKeyReq->apiConnectPtr = transPtr->i;
+ tcKeyReq->senderData = opPtr->i;
+ if (holdOperation) {
+ jam();
+ opRecord->triggerExecutionCount++;
+ }//if
+ // Calculate key length and renumber attribute id:s
+ for(bool moreKeyAttrs = firedTriggerData->afterValues.first(iter);
+ moreKeyAttrs;
+ attrId++) {
+ jam();
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+
+ attrHeader->setAttributeId(attrId);
+ keyLength += attrHeader->getDataSize();
+ hops = attrHeader->getHeaderSize() + attrHeader->getDataSize();
+ moreKeyAttrs = firedTriggerData->afterValues.next(iter, hops);
+ }
+
+ // Filter out single NULL attributes
+ if (attrId == 1) {
+ jam();
+ firedTriggerData->afterValues.first(iter);
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+ if (attrHeader->isNULL() && !firedTriggerData->afterValues.next(iter)) {
+ jam();
+ opRecord->triggerExecutionCount--;
+ if (opRecord->triggerExecutionCount == 0) {
+ /*
+ We have completed current trigger execution
+ Continue triggering operation
+ */
+ jam();
+ continueTriggeringOp(signal, opRecord);
+ }//if
+ return;
+ }//if
+ }//if
+
+ // Calculate total length of primary key to be stored in index table
+ for(bool moreAttrData = firedTriggerData->keyValues.first(iter);
+ (moreAttrData);
+ moreAttrData = firedTriggerData->keyValues.next(iter, hops)) {
+ jam();
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+
+ totalPrimaryKeyLength += attrHeader->getDataSize();
+ hops = attrHeader->getHeaderSize() + attrHeader->getDataSize();
+ }
+ AttributeHeader pkAttrHeader(attrId, totalPrimaryKeyLength);
+
+ TcKeyReq::setKeyLength(tcKeyRequestInfo, keyLength);
+ tcKeyReq->attrLen =
+ firedTriggerData->afterValues.getSize() +
+ pkAttrHeader.getHeaderSize() + pkAttrHeader.getDataSize();
+ tcKeyReq->tableId = indexData->indexId;
+ TcKeyReq::setOperationType(tcKeyRequestInfo, ZINSERT);
+ TcKeyReq::setExecutingTrigger(tcKeyRequestInfo, true);
+ tcKeyReq->tableSchemaVersion = indexTabPtr.p->currentSchemaVersion;
+ tcKeyReq->transId1 = regApiPtr->transid[0];
+ tcKeyReq->transId2 = regApiPtr->transid[1];
+ Uint32 * dataPtr = &tcKeyReq->scanInfo;
+ // Write first part of key in TCKEYREQ
+ Uint32 keyBufSize = 8; // Maximum for key in TCKEYREQ
+ Uint32 attrBufSize = 5; // Maximum for key in TCKEYREQ
+ Uint32 dataPos = 0;
+ // Filter out AttributeHeader:s since this should no be in key
+ bool moreKeyData = firedTriggerData->afterValues.first(iter);
+ Uint32 headerSize = 0, keyAttrSize = 0, dataSize = 0, headAndData = 0;
+
+ while (moreKeyData &&
+ (dataPos < keyBufSize)) {
+ /*
+ If we have not read complete key
+ and it fits in the signal
+ */
+ jam();
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+
+ headerSize = attrHeader->getHeaderSize();
+ keyAttrSize = attrHeader->getDataSize();
+ headAndData = headerSize + attrHeader->getDataSize();
+ // Skip header
+ if (headerSize == 1) {
+ jam();
+ moreKeyData = firedTriggerData->afterValues.next(iter);
+ } else {
+ jam();
+ moreKeyData = firedTriggerData->afterValues.next(iter, headerSize - 1);
+ }//if
+ while((keyAttrSize != 0) &&
+ (dataPos < keyBufSize)) {
+ // If we have not read complete key
+ jam();
+ *dataPtr++ = *iter.data;
+ dataPos++;
+ keyAttrSize--;
+ moreKeyData = firedTriggerData->afterValues.next(iter);
+ }
+ if (keyAttrSize != 0) {
+ jam();
+ break;
+ }//if
+ }
+
+ tcKeyLength += dataPos;
+ Uint32 attributesLength =
+ firedTriggerData->afterValues.getSize() +
+ pkAttrHeader.getHeaderSize() + pkAttrHeader.getDataSize();
+ if (attributesLength <= attrBufSize) {
+ jam();
+ // ATTRINFO fits in TCKEYREQ
+ // Pack ATTRINFO IN TCKEYREQ as one attribute
+ TcKeyReq::setAIInTcKeyReq(tcKeyRequestInfo, attributesLength);
+ bool moreAttrData;
+ // Insert primary key attributes (insert after values of primary table)
+ for(moreAttrData = firedTriggerData->afterValues.first(iter);
+ moreAttrData;
+ moreAttrData = firedTriggerData->afterValues.next(iter)) {
+ *dataPtr++ = *iter.data;
+ }
+ // Insert attribute values (insert key values of primary table)
+ // as one attribute
+ pkAttrHeader.insertHeader(dataPtr);
+ dataPtr += pkAttrHeader.getHeaderSize();
+ moreAttrData = firedTriggerData->keyValues.first(iter);
+ while(moreAttrData) {
+ jam();
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+
+ headerSize = attrHeader->getHeaderSize();
+ dataSize = attrHeader->getDataSize();
+ // Skip header
+ if (headerSize == 1) {
+ jam();
+ moreAttrData = firedTriggerData->keyValues.next(iter);
+ } else {
+ jam();
+ moreAttrData = firedTriggerData->keyValues.next(iter, headerSize - 1);
+ }//if
+ // Copy attribute data
+ while(dataSize-- != 0) {
+ *dataPtr++ = *iter.data;
+ moreAttrData = firedTriggerData->keyValues.next(iter);
+ }
+ }
+ tcKeyLength += attributesLength;
+ } else {
+ jam();
+ // No ATTRINFO in TCKEYREQ
+ TcKeyReq::setAIInTcKeyReq(tcKeyRequestInfo, 0);
+ }
+ tcKeyReq->requestInfo = tcKeyRequestInfo;
+
+ /**
+ * Fix savepoint id -
+ * fix so that insert has same savepoint id as triggering operation
+ */
+ const Uint32 currSavePointId = regApiPtr->currSavePointId;
+ regApiPtr->currSavePointId = opRecord->savePointId;
+ EXECUTE_DIRECT(DBTC, GSN_TCKEYREQ, signal, tcKeyLength);
+ regApiPtr->currSavePointId = currSavePointId;
+ tcConnectptr.p->currentIndexId = indexData->indexId;
+ jamEntry();
+
+ // *********** KEYINFO ***********
+ if (moreKeyData) {
+ jam();
+ // Send KEYINFO sequence
+ KeyInfo * const keyInfo = (KeyInfo *)signal->getDataPtrSend();
+
+ keyInfo->connectPtr = transPtr->i;
+ keyInfo->transId[0] = regApiPtr->transid[0];
+ keyInfo->transId[1] = regApiPtr->transid[1];
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ dataPos = 0;
+ // Pack any part of a key attribute that did no fit TCKEYREQ
+ while((keyAttrSize != 0) &&
+ (dataPos < KeyInfo::DataLength)) {
+ // If we have not read complete key
+ *dataPtr++ = *iter.data;
+ dataPos++;
+ keyAttrSize--;
+ if (dataPos == KeyInfo::DataLength) {
+ jam();
+ // Flush KEYINFO
+#if INTERNAL_TRIGGER_TCKEYREQ_JBA
+ sendSignal(reference(), GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength, JBA);
+#else
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength);
+ jamEntry();
+#endif
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ dataPos = 0;
+ }
+ moreKeyData = firedTriggerData->afterValues.next(iter);
+ }
+
+ while(moreKeyData) {
+ jam();
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+
+ headerSize = attrHeader->getHeaderSize();
+ keyAttrSize = attrHeader->getDataSize();
+ headAndData = headerSize + attrHeader->getDataSize();
+ // Skip header
+ if (headerSize == 1) {
+ jam();
+ moreKeyData = firedTriggerData->afterValues.next(iter);
+ } else {
+ jam();
+ moreKeyData = firedTriggerData->afterValues.next(iter,
+ headerSize - 1);
+ }//if
+ while (keyAttrSize-- != 0) {
+ *dataPtr++ = *iter.data;
+ dataPos++;
+ if (dataPos == KeyInfo::DataLength) {
+ jam();
+ // Flush KEYINFO
+#if INTERNAL_TRIGGER_TCKEYREQ_JBA
+ sendSignal(reference(), GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength, JBA);
+#else
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength);
+ jamEntry();
+#endif
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ dataPos = 0;
+ }
+ moreKeyData = firedTriggerData->afterValues.next(iter);
+ }
+ }
+ if (dataPos != 0) {
+ jam();
+ // Flush last KEYINFO
+#if INTERNAL_TRIGGER_TCKEYREQ_JBA
+ sendSignal(reference(), GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + dataPos, JBA);
+#else
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + dataPos);
+ jamEntry();
+#endif
+ }
+ }
+
+ // *********** ATTRINFO ***********
+ if (attributesLength > attrBufSize) {
+ jam();
+ // No ATTRINFO in TcKeyReq
+ TcKeyReq::setAIInTcKeyReq(tcKeyReq->requestInfo, 0);
+ // Send ATTRINFO sequence
+ AttrInfo * const attrInfo = (AttrInfo *)signal->getDataPtrSend();
+ Uint32 attrInfoPos = 0;
+
+ attrInfo->connectPtr = transPtr->i;
+ attrInfo->transId[0] = regApiPtr->transid[0];
+ attrInfo->transId[1] = regApiPtr->transid[1];
+ dataPtr = (Uint32 *) &attrInfo->attrData;
+
+ bool moreAttrData;
+ // Insert primary key attributes (insert after values of primary table)
+ for(moreAttrData = firedTriggerData->afterValues.first(iter);
+ moreAttrData;
+ moreAttrData = firedTriggerData->afterValues.next(iter)) {
+ *dataPtr++ = *iter.data;
+ attrInfoPos++;
+ if (attrInfoPos == AttrInfo::DataLength) {
+ jam();
+ // Flush ATTRINFO
+#if INTERNAL_TRIGGER_TCKEYREQ_JBA
+ sendSignal(reference(), GSN_ATTRINFO, signal,
+ AttrInfo::HeaderLength + AttrInfo::DataLength, JBA);
+#else
+ EXECUTE_DIRECT(DBTC, GSN_ATTRINFO, signal,
+ AttrInfo::HeaderLength + AttrInfo::DataLength);
+ jamEntry();
+#endif
+ dataPtr = (Uint32 *) &attrInfo->attrData;
+ attrInfoPos = 0;
+ }
+ }
+ // Insert attribute values (insert key values of primary table)
+ // as one attribute
+ pkAttrHeader.insertHeader(dataPtr);
+ dataPtr += pkAttrHeader.getHeaderSize();
+ attrInfoPos += pkAttrHeader.getHeaderSize();
+ moreAttrData = firedTriggerData->keyValues.first(iter);
+ while(moreAttrData) {
+ jam();
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+
+ headerSize = attrHeader->getHeaderSize();
+ dataSize = attrHeader->getDataSize();
+ // Skip header
+ if (headerSize == 1) {
+ jam();
+ moreAttrData = firedTriggerData->keyValues.next(iter);
+ } else {
+ jam();
+ moreAttrData = firedTriggerData->keyValues.next(iter,
+ headerSize - 1);
+ }//if
+ while(dataSize-- != 0) { // If we have not read complete key
+ if (attrInfoPos == AttrInfo::DataLength) {
+ jam();
+ // Flush ATTRINFO
+#if INTERNAL_TRIGGER_TCKEYREQ_JBA
+ sendSignal(reference(), GSN_ATTRINFO, signal,
+ AttrInfo::HeaderLength + AttrInfo::DataLength, JBA);
+#else
+ EXECUTE_DIRECT(DBTC, GSN_ATTRINFO, signal,
+ AttrInfo::HeaderLength + AttrInfo::DataLength);
+ jamEntry();
+#endif
+ dataPtr = (Uint32 *) &attrInfo->attrData;
+ attrInfoPos = 0;
+ }
+ *dataPtr++ = *iter.data;
+ attrInfoPos++;
+ moreAttrData = firedTriggerData->keyValues.next(iter);
+ }
+ }
+ if (attrInfoPos != 0) {
+ jam();
+ // Flush last ATTRINFO
+#if INTERNAL_TRIGGER_TCKEYREQ_JBA
+ sendSignal(reference(), GSN_ATTRINFO, signal,
+ AttrInfo::HeaderLength + attrInfoPos, JBA);
+#else
+ EXECUTE_DIRECT(DBTC, GSN_ATTRINFO, signal,
+ AttrInfo::HeaderLength + attrInfoPos);
+ jamEntry();
+#endif
+ }
+ }
+}
+
+void Dbtc::deleteFromIndexTable(Signal* signal,
+ TcFiredTriggerData* firedTriggerData,
+ ApiConnectRecordPtr* transPtr,
+ TcConnectRecordPtr* opPtr,
+ TcIndexData* indexData,
+ bool holdOperation)
+{
+ ApiConnectRecord* regApiPtr = transPtr->p;
+ TcConnectRecord* opRecord = opPtr->p;
+ TcKeyReq * const tcKeyReq = (TcKeyReq *)signal->getDataPtrSend();
+ Uint32 tcKeyRequestInfo = 0;
+ Uint32 tcKeyLength = 12; // Static length
+ TableRecordPtr indexTabPtr;
+ AttributeBuffer::DataBufferIterator iter;
+ Uint32 attrId = 0;
+ Uint32 keyLength = 0;
+ Uint32 hops;
+
+ indexTabPtr.i = indexData->indexId;
+ ptrCheckGuard(indexTabPtr, ctabrecFilesize, tableRecord);
+ tcKeyReq->apiConnectPtr = transPtr->i;
+ tcKeyReq->senderData = opPtr->i;
+ if (holdOperation) {
+ jam();
+ opRecord->triggerExecutionCount++;
+ }//if
+ // Calculate key length and renumber attribute id:s
+ for(bool moreKeyAttrs = firedTriggerData->beforeValues.first(iter);
+ (moreKeyAttrs);
+ attrId++) {
+ jam();
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+
+ attrHeader->setAttributeId(attrId);
+ keyLength += attrHeader->getDataSize();
+ hops = attrHeader->getHeaderSize() + attrHeader->getDataSize();
+ moreKeyAttrs = firedTriggerData->beforeValues.next(iter, hops);
+ }
+
+ // Filter out single NULL attributes
+ if (attrId == 1) {
+ jam();
+ firedTriggerData->beforeValues.first(iter);
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+ if (attrHeader->isNULL() && !firedTriggerData->beforeValues.next(iter)) {
+ jam();
+ opRecord->triggerExecutionCount--;
+ if (opRecord->triggerExecutionCount == 0) {
+ /*
+ We have completed current trigger execution
+ Continue triggering operation
+ */
+ jam();
+ continueTriggeringOp(signal, opRecord);
+ }//if
+ return;
+ }//if
+ }//if
+
+ TcKeyReq::setKeyLength(tcKeyRequestInfo, keyLength);
+ tcKeyReq->attrLen = 0;
+ tcKeyReq->tableId = indexData->indexId;
+ TcKeyReq::setOperationType(tcKeyRequestInfo, ZDELETE);
+ TcKeyReq::setExecutingTrigger(tcKeyRequestInfo, true);
+ tcKeyReq->tableSchemaVersion = indexTabPtr.p->currentSchemaVersion;
+ tcKeyReq->transId1 = regApiPtr->transid[0];
+ tcKeyReq->transId2 = regApiPtr->transid[1];
+ Uint32 * dataPtr = &tcKeyReq->scanInfo;
+ // Write first part of key in TCKEYREQ
+ Uint32 keyBufSize = 8; // Maximum for key in TCKEYREQ
+ Uint32 dataPos = 0;
+ // Filter out AttributeHeader:s since this should no be in key
+ bool moreKeyData = firedTriggerData->beforeValues.first(iter);
+ Uint32 headerSize = 0, keyAttrSize = 0, headAndData = 0;
+
+ while (moreKeyData &&
+ (dataPos < keyBufSize)) {
+ /*
+ If we have not read complete key
+ and it fits in the signal
+ */
+ jam();
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+
+ headerSize = attrHeader->getHeaderSize();
+ keyAttrSize = attrHeader->getDataSize();
+ headAndData = headerSize + attrHeader->getDataSize();
+ // Skip header
+ if (headerSize == 1) {
+ jam();
+ moreKeyData = firedTriggerData->beforeValues.next(iter);
+ } else {
+ jam();
+ moreKeyData = firedTriggerData->beforeValues.next(iter, headerSize - 1);
+ }//if
+ while((keyAttrSize != 0) &&
+ (dataPos < keyBufSize)) {
+ // If we have not read complete key
+ jam();
+ *dataPtr++ = *iter.data;
+ dataPos++;
+ keyAttrSize--;
+ moreKeyData = firedTriggerData->beforeValues.next(iter);
+ }
+ if (keyAttrSize != 0) {
+ jam();
+ break;
+ }//if
+ }
+
+ tcKeyLength += dataPos;
+ tcKeyReq->requestInfo = tcKeyRequestInfo;
+
+ /**
+ * Fix savepoint id -
+ * fix so that delete has same savepoint id as triggering operation
+ */
+ const Uint32 currSavePointId = regApiPtr->currSavePointId;
+ regApiPtr->currSavePointId = opRecord->savePointId;
+ EXECUTE_DIRECT(DBTC, GSN_TCKEYREQ, signal, tcKeyLength);
+ regApiPtr->currSavePointId = currSavePointId;
+ tcConnectptr.p->currentIndexId = indexData->indexId;
+ jamEntry();
+
+ // *********** KEYINFO ***********
+ if (moreKeyData) {
+ jam();
+ // Send KEYINFO sequence
+ KeyInfo * const keyInfo = (KeyInfo *)signal->getDataPtrSend();
+
+ keyInfo->connectPtr = transPtr->i;
+ keyInfo->transId[0] = regApiPtr->transid[0];
+ keyInfo->transId[1] = regApiPtr->transid[1];
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ dataPos = 0;
+ // Pack any part of a key attribute that did no fit TCKEYREQ
+ while((keyAttrSize != 0) &&
+ (dataPos < KeyInfo::DataLength)) {
+ // If we have not read complete key
+ *dataPtr++ = *iter.data;
+ dataPos++;
+ keyAttrSize--;
+ if (dataPos == KeyInfo::DataLength) {
+ jam();
+ // Flush KEYINFO
+#if INTERNAL_TRIGGER_TCKEYREQ_JBA
+ sendSignal(reference(), GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength, JBA);
+#else
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength);
+ jamEntry();
+#endif
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ dataPos = 0;
+ }
+ moreKeyData = firedTriggerData->beforeValues.next(iter);
+ }
+
+ while(moreKeyData) {
+ jam();
+ AttributeHeader* attrHeader = (AttributeHeader *) iter.data;
+
+ headerSize = attrHeader->getHeaderSize();
+ keyAttrSize = attrHeader->getDataSize();
+ headAndData = headerSize + attrHeader->getDataSize();
+ // Skip header
+ if (headerSize == 1) {
+ jam();
+ moreKeyData = firedTriggerData->beforeValues.next(iter);
+ } else {
+ jam();
+ moreKeyData = firedTriggerData->beforeValues.next(iter,
+ headerSize - 1);
+ }//if
+ while (keyAttrSize-- != 0) {
+ *dataPtr++ = *iter.data;
+ dataPos++;
+ if (dataPos == KeyInfo::DataLength) {
+ jam();
+ // Flush KEYINFO
+#if INTERNAL_TRIGGER_TCKEYREQ_JBA
+ sendSignal(reference(), GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength, JBA);
+#else
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + KeyInfo::DataLength);
+ jamEntry();
+#endif
+ dataPtr = (Uint32 *) &keyInfo->keyData;
+ dataPos = 0;
+ }
+ moreKeyData = firedTriggerData->beforeValues.next(iter);
+ }
+ }
+ if (dataPos != 0) {
+ jam();
+ // Flush last KEYINFO
+#if INTERNAL_TRIGGER_TCKEYREQ_JBA
+ sendSignal(reference(), GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + dataPos, JBA);
+#else
+ EXECUTE_DIRECT(DBTC, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + dataPos);
+ jamEntry();
+#endif
+ }
+ }
+}
+
+Uint32
+Dbtc::TableRecord::getErrorCode(Uint32 schemaVersion) const {
+ if(!enabled)
+ return ZNO_SUCH_TABLE;
+ if(dropping)
+ return ZDROP_TABLE_IN_PROGRESS;
+ if(schemaVersion != currentSchemaVersion)
+ return ZWRONG_SCHEMA_VERSION_ERROR;
+ ErrorReporter::handleAssert("Dbtc::TableRecord::getErrorCode",
+ __FILE__, __LINE__);
+ return 0;
+}
+
diff --git a/ndb/src/kernel/blocks/dbtc/Makefile b/ndb/src/kernel/blocks/dbtc/Makefile
new file mode 100644
index 00000000000..ae876ab1f84
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtc/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := dbtc
+SOURCES = \
+ DbtcInit.cpp \
+ DbtcMain.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/kernel/blocks/dbtup/AttributeOffset.hpp b/ndb/src/kernel/blocks/dbtup/AttributeOffset.hpp
new file mode 100644
index 00000000000..0f3881e9024
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/AttributeOffset.hpp
@@ -0,0 +1,89 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ATTRIBUTE_OFFSET_HPP
+#define ATTRIBUTE_OFFSET_HPP
+
+class AttributeOffset {
+ friend class Dbtup;
+
+private:
+ static void setOffset(Uint32 & desc, Uint32 offset);
+ static void setNullFlagPos(Uint32 & desc, Uint32 offset);
+
+ static Uint32 getOffset(const Uint32 &);
+ static Uint32 getNullFlagPos(const Uint32 &);
+ static Uint32 getNullFlagOffset(const Uint32 &);
+ static Uint32 getNullFlagBitOffset(const Uint32 &);
+ static bool isNULL(const Uint32 &, const Uint32 &);
+};
+
+#define AO_ATTRIBUTE_OFFSET_MASK (0xffff)
+#define AO_NULL_FLAG_POS_MASK (0x7ff)
+#define AO_NULL_FLAG_POS_SHIFT (21)
+#define AO_NULL_FLAG_WORD_MASK (31)
+#define AO_NULL_FLAG_OFFSET_SHIFT (5)
+
+inline
+void
+AttributeOffset::setOffset(Uint32 & desc, Uint32 offset){
+ ASSERT_MAX(offset, AO_ATTRIBUTE_OFFSET_MASK, "AttributeOffset::setOffset");
+ desc |= offset;
+}
+
+inline
+void
+AttributeOffset::setNullFlagPos(Uint32 & desc, Uint32 pos){
+ ASSERT_MAX(pos, AO_NULL_FLAG_POS_MASK, "AttributeOffset::setNullFlagPos");
+ desc |= (pos << AO_NULL_FLAG_POS_SHIFT);
+}
+
+inline
+Uint32
+AttributeOffset::getOffset(const Uint32 & desc)
+{
+ return desc & AO_ATTRIBUTE_OFFSET_MASK;
+}
+
+inline
+Uint32
+AttributeOffset::getNullFlagPos(const Uint32 & desc)
+{
+ return ((desc >> AO_NULL_FLAG_POS_SHIFT) & AO_NULL_FLAG_POS_MASK);
+}
+
+inline
+Uint32
+AttributeOffset::getNullFlagOffset(const Uint32 & desc)
+{
+ return (getNullFlagPos(desc) >> AO_NULL_FLAG_OFFSET_SHIFT);
+}
+
+inline
+Uint32
+AttributeOffset::getNullFlagBitOffset(const Uint32 & desc)
+{
+ return (getNullFlagPos(desc) & AO_NULL_FLAG_WORD_MASK);
+}
+
+inline
+bool
+AttributeOffset::isNULL(const Uint32 & pageWord, const Uint32 & desc)
+{
+ return (((pageWord >> getNullFlagBitOffset(desc)) & 1) == 1);
+}
+
+#endif
diff --git a/ndb/src/kernel/blocks/dbtup/Dbtup.hpp b/ndb/src/kernel/blocks/dbtup/Dbtup.hpp
new file mode 100644
index 00000000000..053b853fa82
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/Dbtup.hpp
@@ -0,0 +1,2359 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBTUP_H
+#define DBTUP_H
+
+#include <pc.hpp>
+#include <SimulatedBlock.hpp>
+#include <ndb_limits.h>
+#include <trigger_definitions.h>
+#include <ArrayList.hpp>
+#include <AttributeHeader.hpp>
+#include <Bitmask.hpp>
+#include <signaldata/TupKey.hpp>
+#include <signaldata/CreateTrig.hpp>
+#include <signaldata/DropTrig.hpp>
+#include <signaldata/TrigAttrInfo.hpp>
+#include <signaldata/BuildIndx.hpp>
+
+#define ZWORDS_ON_PAGE 8192 /* NUMBER OF WORDS ON A PAGE. */
+#define ZATTRBUF_SIZE 32 /* SIZE OF ATTRIBUTE RECORD BUFFER */
+#define ZMIN_PAGE_LIMIT_TUPKEYREQ 5
+#define ZTUP_VERSION_BITS 15
+
+typedef bool (Dbtup::* ReadFunction)(Uint32*,
+ AttributeHeader*,
+ Uint32,
+ Uint32);
+typedef bool (Dbtup::* UpdateFunction)(Uint32*,
+ Uint32,
+ Uint32);
+
+#ifdef DBTUP_C
+//------------------------------------------------------------------
+// Jam Handling:
+//
+// When DBTUP reports lines through jam in the trace files it has to
+// be interpreted. 4024 means as an example line 24 in DbtupCommit.cpp
+// Thus 4000 is added to the line number beacuse it is located in the
+// file DbtupCommit.cpp. The following is the exhaustive list of the
+// added value in the various files. ndbrequire, ptrCheckGuard still
+// only reports the line number in the file it currently is located in.
+//
+// DbtupExecQuery.cpp 0
+// DbtupBuffer.cpp 2000
+// DbtupRoutines.cpp 3000
+// DbtupCommit.cpp 5000
+// DbtupFixAlloc.cpp 6000
+// DbtupTrigger.cpp 7000
+// DbtupAbort.cpp 9000
+// DbtupLCP.cpp 10000
+// DbtupUndoLog.cpp 12000
+// DbtupPageMap.cpp 14000
+// DbtupPagMan.cpp 16000
+// DbtupStoredProcDef.cpp 18000
+// DbtupMeta.cpp 20000
+// DbtupTabDesMan.cpp 22000
+// DbtupGen.cpp 24000
+// DbtupSystemRestart.cpp 26000
+// DbtupIndex.cpp 28000
+// DbtupDebug.cpp 30000
+//------------------------------------------------------------------
+
+/*
+2.2 LOCAL SYMBOLS
+-----------------
+*/
+/* ---------------------------------------------------------------- */
+/* S I Z E O F R E C O R D S */
+/* ---------------------------------------------------------------- */
+#define ZNO_OF_ATTRBUFREC 10000 /* SIZE OF ATTRIBUTE INFO FILE */
+#define ZNO_OF_CONCURRENT_OPEN_OP 40 /* NUMBER OF CONCURRENT OPENS */
+#define ZNO_OF_CONCURRENT_WRITE_OP 80 /* NUMBER OF CONCURRENT DISK WRITES*/
+#define ZNO_OF_FRAGOPREC 20 /* NUMBER OF CONCURRENT ADD FRAG. */
+#define ZNO_OF_FRAGREC 64 /* SIZE OF FRAGMENT FILE. */
+#define ZNO_OF_LCP_REC 10 /* NUMBER OF CONCURRENT CHECKPOINTS*/
+#define ZNO_OF_OPREC 116 /* SIZE OF OPERATION RECORD FILE */
+#define TOT_PAGE_RECORD_SPACE 262144 /* SIZE OF PAGE RECORD FILE. */
+#define ZNO_OF_PAGE TOT_PAGE_RECORD_SPACE/ZWORDS_ON_PAGE
+#define ZNO_OF_PAGE_RANGE_REC 128 /* SIZE OF PAGE RANGE FILE */
+#define ZNO_OF_PARALLELL_UNDO_FILES 16 /* NUMBER OF PARALLEL UNDO FILES */
+#define ZNO_OF_RESTART_INFO_REC 10 /* MAXIMUM PARALLELL RESTART INFOS */
+#define ZNO_OF_TAB_DESCR_REC 484 /* SIZE OF TABLE DESCRIPTOR FILE */
+#define ZNO_OF_TABLEREC 16 /* SIZE OF TABLE RECORD FILE. */
+#ifdef NDB_OSE
+#define ZNO_OF_UNDO_PAGE 80 // Must be multiple of 8
+#else
+#define ZNO_OF_UNDO_PAGE 500 // Must be multiple of 8
+#endif
+ /* 24 SEGMENTS WITH 8 PAGES IN EACH*/
+ /* PLUS ONE UNDO BUFFER CACHE */
+// Undo record identifiers are 32-bits with page index 13-bits
+#define ZUNDO_RECORD_ID_PAGE_INDEX 13 /* 13 BITS = 8192 WORDS/PAGE */
+#define ZUNDO_RECORD_ID_PAGE_INDEX_MASK (ZWORDS_ON_PAGE - 1) /* 1111111111111 */
+
+// Trigger constants
+#define ZDEFAULT_MAX_NO_TRIGGERS_PER_TABLE 16
+
+/* ---------------------------------------------------------------- */
+// VARIABLE NUMBERS OF PAGE_WORD, UNDO_WORD AND LOGIC_WORD FOR
+// COMMUNICATION WITH FILE SYSTEM
+/* ---------------------------------------------------------------- */
+#define ZBASE_ADDR_PAGE_WORD 1 /* BASE ADDRESS OF PAGE_WORD VAR */
+#define ZBASE_ADDR_UNDO_WORD 2 /* BASE ADDRESS OF UNDO_WORD VAR */
+#define ZBASE_ADDR_LOGIC_WORD 3 /* BASE ADDRESS OF LOGIC_WORD VAR */
+
+/* ---------------------------------------------------------------- */
+// NUMBER OF PAGES SENT TO DISK IN DATA BUFFER AND UNDO BUFFER WHEN
+// OPTIMUM PERFORMANCE IS ACHIEVED.
+/* ---------------------------------------------------------------- */
+#define ZUB_SEGMENT_SIZE 8 /* SEGMENT SIZE OF UNDO BUFFER */
+#define ZDB_SEGMENT_SIZE 8 /* SEGMENT SIZE OF DATA BUFFER */
+
+/* ---------------------------------------------------------------- */
+/* A ATTRIBUTE MAY BE NULL, DYNAMIC OR NORMAL. A NORMAL ATTRIBUTE */
+/* IS A ATTRIBUTE THAT IS NOT NULL OR DYNAMIC. A NULL ATTRIBUTE */
+/* MAY HAVE NO VALUE. A DYNAMIC ATTRIBUTE IS A NULL ATTRIBUTE THAT */
+/* DOES NOT HAVE TO BE A MEMBER OF EVERY TUPLE I A CERTAIN TABLE. */
+/* ---------------------------------------------------------------- */
+/**
+ * #defines moved into include/kernel/Interpreter.hpp
+ */
+#define ZMAX_REGISTER 21
+#define ZINSERT_DELETE 0
+/* ---------------------------------------------------------------- */
+/* THE MINIMUM SIZE OF AN 'EMPTY' TUPLE HEADER IN R-WORDS */
+/* ---------------------------------------------------------------- */
+#define ZTUP_HEAD_MINIMUM_SIZE 2
+ /* THE TUPLE HEADER FIELD 'SIZE OF NULL ATTR. FIELD' SPECIFYES */
+ /* THE SIZE OF THE TUPLE HEADER FIELD 'NULL ATTR. FIELD'. */
+ /* THE TUPLE HEADER FIELD 'TYPE' SPECIFYES THE TYPE OF THE TUPLE */
+ /* HEADER. */
+ /* TUPLE ATTRIBUTE INDEX CLUSTERS, ATTRIBUTE */
+ /* CLUSTERS AND A DYNAMIC ATTRIBUTE HEADER. */
+ /* IT MAY ALSO CONTAIN SHORT ATTRIBUTES AND */
+ /* POINTERS TO LONG ATTRIBUTE HEADERS. */
+ /* TUPLE ATTRIBUTE INDEX CLUSTERS, ATTRIBUTE */
+ /* CLUSTERS AND A DYNAMIC ATTRIBUTE HEADER. */
+
+#define ZTH_TYPE3 2 /* TUPLE HEADER THAT MAY HAVE A POINTER TO */
+ /* A DYNAMIC ATTRIBUTE HEADER. IT MAY ALSO */
+ /* CONTAIN SHORT ATTRIBUTES AND POINTERS */
+ /* TO LONG ATTRIBUTE HEADERS. */
+
+ /* DATA STRUCTURE TYPES */
+ /* WHEN ATTRIBUTE INFO IS SENT WITH A ATTRINFO-SIGNAL THE */
+ /* VARIABLE TYPE IS SPECIFYED. THIS MUST BE DONE TO BE ABLE TO */
+ /* NOW HOW MUCH DATA OF A ATTRIBUTE TO READ FROM ATTRINFO. */
+#define ZFIXED_ARRAY 2 /* ZFIXED ARRAY FIELD. */
+#define ZNON_ARRAY 1 /* NORMAL FIELD. */
+#define ZVAR_ARRAY 0 /* VARIABLE ARRAY FIELD */
+#define ZNOT_STORE 3 /* THE ATTR IS STORED IN THE INDEX BLOCK */
+#define ZMAX_SMALL_VAR_ARRAY 256
+
+ /* PLEASE OBSERVE THAT THEESE CONSTANTS CORRESPONDS TO THE NUMBER */
+ /* OF BITS NEEDED TO REPRESENT THEM D O N O T C H A N G E */
+#define Z1BIT_VAR 0 /* 1 BIT VARIABLE. */
+#define Z2BIT_VAR 1 /* 2 BIT VARIABLE. */
+#define Z4BIT_VAR 2 /* 4 BIT VARIABLE. */
+#define Z8BIT_VAR 3 /* 8 BIT VARIABLE. */
+#define Z16BIT_VAR 4 /* 16 BIT VARIABLE. */
+#define Z32BIT_VAR 5 /* 32 BIT VARIABLE. */
+#define Z64BIT_VAR 6 /* 64 BIT VARIABLE. */
+#define Z128BIT_VAR 7 /* 128 BIT VARIABLE. */
+
+ /* WHEN A REQUEST CAN NOT BE EXECUTED BECAUSE OF A ERROR THE */
+ /* ERROR MUST BE IDENTIFYED BY MEANS OF A ERROR CODE AND SENT TO */
+ /* THE REQUESTER. */
+#define ZGET_OPREC_ERROR 804 // TUP_SEIZEREF
+
+#define ZEXIST_FRAG_ERROR 816 // Add fragment
+#define ZFULL_FRAGRECORD_ERROR 817 // Add fragment
+#define ZNO_FREE_PAGE_RANGE_ERROR 818 // Add fragment
+#define ZNOFREE_FRAGOP_ERROR 830 // Add fragment
+#define ZTOO_LARGE_TUPLE_ERROR 851 // Add fragment
+#define ZNO_FREE_TAB_ENTRY_ERROR 852 // Add fragment
+#define ZNO_PAGES_ALLOCATED_ERROR 881 // Add fragment
+
+#define ZGET_REALPID_ERROR 809
+#define ZNOT_IMPLEMENTED_ERROR 812
+#define ZSEIZE_ATTRINBUFREC_ERROR 805
+#define ZTOO_MUCH_ATTRINFO_ERROR 823
+#define ZMEM_NOTABDESCR_ERROR 826
+#define ZMEM_NOMEM_ERROR 827
+#define ZAI_INCONSISTENCY_ERROR 829
+#define ZNO_ILLEGAL_NULL_ATTR 839
+#define ZNOT_NULL_ATTR 840
+#define ZNO_INSTRUCTION_ERROR 871
+#define ZOUTSIDE_OF_PROGRAM_ERROR 876
+#define ZSTORED_PROC_ID_ERROR 877
+#define ZREGISTER_INIT_ERROR 878
+#define ZATTRIBUTE_ID_ERROR 879
+#define ZTRY_TO_READ_TOO_MUCH_ERROR 880
+#define ZTOTAL_LEN_ERROR 882
+#define ZATTR_INTERPRETER_ERROR 883
+#define ZSTACK_OVERFLOW_ERROR 884
+#define ZSTACK_UNDERFLOW_ERROR 885
+#define ZTOO_MANY_INSTRUCTIONS_ERROR 886
+#define ZTRY_TO_UPDATE_ERROR 888
+#define ZCALL_ERROR 890
+#define ZTEMPORARY_RESOURCE_FAILURE 891
+
+#define ZSTORED_SEIZE_ATTRINBUFREC_ERROR 873 // Part of Scan
+
+#define ZREAD_ONLY_CONSTRAINT_VIOLATION 893
+#define ZVAR_SIZED_NOT_SUPPORTED 894
+#define ZINCONSISTENT_NULL_ATTRIBUTE_COUNT 895
+#define ZTUPLE_CORRUPTED_ERROR 896
+#define ZTRY_UPDATE_PRIMARY_KEY 897
+#define ZMUST_BE_ABORTED_ERROR 898
+#define ZTUPLE_DELETED_ERROR 626
+#define ZINSERT_ERROR 630
+
+
+ /* SOME WORD POSITIONS OF FIELDS IN SOME HEADERS */
+#define ZPAGE_STATE_POS 0 /* POSITION OF PAGE STATE */
+#define ZPAGE_NEXT_POS 1 /* POSITION OF THE NEXT POINTER WHEN IN FREELIST */
+#define ZPAGE_PREV_POS 2 /* POSITION OF THE PREVIOUS POINTER WHEN IN FREELIST */
+#define ZFREELIST_HEADER_POS 3 /* POSITION OF THE FIRST FREELIST */
+#define ZPAGE_FRAG_PAGE_ID_POS 4 /* POSITION OF FRAG PAGE ID WHEN USED*/
+#define ZPAGE_NEXT_CLUST_POS 5 /* POSITION OF NEXT FREE SET OF PAGES */
+#define ZPAGE_FIRST_CLUST_POS 2 /* POSITION OF THE POINTER TO THE FIRST PAGE IN A CLUSTER */
+#define ZPAGE_LAST_CLUST_POS 6 /* POSITION OF THE POINTER TO THE LAST PAGE IN A CLUSTER */
+#define ZPAGE_PREV_CLUST_POS 7 /* POSITION OF THE PREVIOUS POINTER */
+#define ZPAGE_HEADER_SIZE 32 /* NUMBER OF WORDS IN MEM PAGEHEADER */
+#define ZDISK_PAGE_HEADER_SIZE 32 /* NUMBER OF WORDS IN DISK PAGEHEADER */
+#define ZNO_OF_FREE_BLOCKS 3 /* NO OF FREE BLOCK IN THE DISK PAGE */
+#define ZDISK_PAGE_ID 8 /* ID OF THE PAGE ON THE DISK */
+#define ZBLOCK_LIST 9
+#define ZCOPY_OF_PAGE 10
+#define ZPAGE_PHYSICAL_INDEX 11
+#define ZNEXT_IN_PAGE_USED_LIST 12
+#define ZPREV_IN_PAGE_USED_LIST 13
+#define ZDISK_USED_TYPE 14
+#define ZFREE_COMMON 1 /* PAGE STATE, PAGE IN COMMON AREA */
+#define ZEMPTY_MM 2 /* PAGE STATE, PAGE IN EMPTY LIST */
+#define ZTH_MM_FREE 3 /* PAGE STATE, TUPLE HEADER PAGE WITH FREE AREA */
+#define ZTH_MM_FULL 4 /* PAGE STATE, TUPLE HEADER PAGE WHICH IS FULL */
+#define ZAC_MM_FREE 5 /* PAGE STATE, ATTRIBUTE CLUSTER PAGE WITH FREE AREA */
+#define ZTH_MM_FREE_COPY 7 /* PAGE STATE, TH COPY PAGE WITH FREE AREA */
+#define ZTH_MM_FULL_COPY 8 /* PAGE STATE, TH COPY PAGE WHICH IS FULL */
+#define ZAC_MM_FREE_COPY 9 /* PAGE STATE, AC COPY PAGE WITH FREE AREA */
+#define ZMAX_NO_COPY_PAGES 4 /* THE MAXIMUM NUMBER OF COPY PAGES ALLOWED PER FRAGMENT */
+
+ /* CONSTANTS USED TO HANDLE TABLE DESCRIPTOR RECORDS */
+ /* ALL POSITIONS AND SIZES IS BASED ON R-WORDS (32-BIT ON APZ 212) */
+#define ZTD_HEADER 0 /* HEADER POSITION */
+#define ZTD_DATASIZE 1 /* SIZE OF THE DATA IN THIS CHUNK */
+#define ZTD_SIZE 2 /* TOTAL SIZE OF TABLE DESCRIPTOR */
+
+ /* TRAILER POSITIONS FROM END OF TABLE DESCRIPTOR RECORD */
+#define ZTD_TR_SIZE 1 /* SIZE DESCRIPTOR POS FROM END+1 */
+#define ZTD_TR_TYPE 2
+#define ZTD_TRAILER_SIZE 2 /* TOTAL SIZE OF TABLE TRAILER */
+#define ZAD_SIZE 2 /* TOTAL SIZE OF ATTR DESCRIPTOR */
+#define ZAD_LOG_SIZE 1 /* TWO LOG OF TOTAL SIZE OF ATTR DESCRIPTOR */
+
+ /* CONSTANTS USED TO HANDLE TABLE DESCRIPTOR AS A FREELIST */
+#define ZTD_FL_HEADER 0 /* HEADER POSITION */
+#define ZTD_FL_SIZE 1 /* TOTAL SIZE OF THIS FREELIST ENTRY */
+#define ZTD_FL_PREV 2 /* PREVIOUS RECORD IN FREELIST */
+#define ZTD_FL_NEXT 3 /* NEXT RECORD IN FREELIST */
+#define ZTD_FREE_SIZE 16 /* SIZE NEEDED TO HOLD ONE FL ENTRY */
+
+ /* CONSTANTS USED IN LSB OF TABLE DESCRIPTOR HEADER DESCRIBING USAGE */
+#define ZTD_TYPE_FREE 0 /* RECORD LINKED INTO FREELIST */
+#define ZTD_TYPE_NORMAL 1 /* RECORD USED AS TABLE DESCRIPTOR */
+ /* ATTRIBUTE OPERATION CONSTANTS */
+#define ZLEAF 1
+#define ZNON_LEAF 2
+
+ /* ATTRINBUFREC VARIABLE POSITIONS. */
+#define ZBUF_PREV 29 /* POSITION OF 'PREV'-VARIABLE (USED BY INTERPRETED EXEC) */
+#define ZBUF_DATA_LEN 30 /* POSITION OF 'DATA LENGTH'-VARIABLE. */
+#define ZBUF_NEXT 31 /* POSITION OF 'NEXT'-VARIABLE. */
+#define ZSAVE_BUF_NEXT 28
+#define ZSAVE_BUF_DATA_LEN 27
+
+ /* RETURN POINTS. */
+ /* RESTART PHASES */
+#define ZSTARTPHASE1 1
+#define ZSTARTPHASE2 2
+#define ZSTARTPHASE3 3
+#define ZSTARTPHASE4 4
+#define ZSTARTPHASE6 6
+
+#define ZADDFRAG 0
+
+ /* CHECKPOINT RECORD TYPES */
+#define ZLCPR_TYPE_INSERT_TH 0 /* INSERT TUPLE HEADER */
+#define ZLCPR_TYPE_DELETE_TH 1 /* DELETE TUPLE HEADER */
+#define ZLCPR_TYPE_UPDATE_TH 2 /* DON'T CREATE IT, JUST UPDETE */
+#define ZLCPR_TYPE_INSERT_TH_NO_DATA 3 /* INSERT TUPLE HEADER */
+#define ZLCPR_ABORT_UPDATE 4 /* UNDO AN UPDATE OPERATION THAT WAS ACTIVE IN LCP */
+#define ZLCPR_ABORT_INSERT 5 /* UNDO AN INSERT OPERATION THAT WAS ACTIVE IN LCP */
+#define ZTABLE_DESCRIPTOR 6 /* TABLE DESCRIPTOR */
+#define ZINDICATE_NO_OP_ACTIVE 7 /* ENSURE THAT NO OPERATION ACTIVE AFTER RESTART */
+#define ZLCPR_UNDO_LOG_PAGE_HEADER 8 /* CHANGE IN PAGE HEADER IS UNDO LOGGED */
+#define ZLCPR_TYPE_UPDATE_GCI 9 /* Update GCI at commit time */
+#define ZNO_CHECKPOINT_RECORDS 10 /* NUMBER OF CHECKPOINTRECORD TYPES */
+
+ /* RESULT CODES */
+ /* ELEMENT POSITIONS IN SYSTEM RESTART INFO PAGE OF THE DATA FILE */
+#define ZSRI_NO_OF_FRAG_PAGES_POS 10 /* NUMBER OF FRAGMENT PAGES WHEN CHECKPOINT STARTED */
+#define ZSRI_TUP_RESERVED_SIZE_POS 11 /* RESERVED SIZE OF THE TUPLE WHEN CP STARTED */
+#define ZSRI_TUP_FIXED_AREA_POS 12 /* SIZE OF THE TUPLE FIXED AREA WHEN CP STARTED */
+#define ZSRI_TAB_DESCR_SIZE 13 /* SIZE OF THE TABLE DESCRIPTOR WHEN CP STARTED */
+#define ZSRI_NO_OF_ATTRIBUTES_POS 14 /* NUMBER OF ATTRIBUTES */
+#define ZSRI_UNDO_LOG_END_REC_ID 15 /* LAST UNDO LOG RECORD ID FOR THIS CHECKPOINT */
+#define ZSRI_UNDO_LOG_END_PAGE_ID 16 /* LAST USED LOG PAGE ID FOR THIS CHECKPOINT */
+#define ZSRI_TH_FREE_FIRST 17 /* FIRST FREE PAGE OF TUPLE HEADERS */
+#define ZSRI_TH_FREE_COPY_FIRST 18 /* FIRST FREE PAGE OF TUPLE HEADER COPIES */
+#define ZSRI_EMPTY_PRIM_PAGE 27 /* FIRST EMPTY PAGE */
+#define ZSRI_NO_COPY_PAGES_ALLOC 28 /* NO COPY PAGES IN FRAGMENT AT LOCAL CHECKPOINT */
+#define ZSRI_UNDO_FILE_VER 29 /* CHECK POINT ID OF THE UNDO FILE */
+#define ZSRI_NO_OF_INDEX_ATTR 30 /* No of index attributes */
+#define ZNO_OF_PAGES_CLUSTER_REC 0
+
+//------------------------------------------------------------
+// TUP_CONTINUEB codes
+//------------------------------------------------------------
+#define ZSTART_EXEC_UNDO_LOG 0
+#define ZCONT_START_SAVE_CL 1
+#define ZCONT_SAVE_DP 2
+#define ZCONT_EXECUTE_LC 3
+#define ZCONT_LOAD_DP 4
+#define ZLOAD_BAL_LCP_TIMER 5
+#define ZINITIALISE_RECORDS 6
+#define ZREL_FRAG 7
+#define ZREPORT_MEMORY_USAGE 8
+#define ZBUILD_INDEX 9
+
+#define ZINDEX_STORAGE 0
+#define ZDATA_WORD_AT_DISK_PAGE 2030
+#define ZALLOC_DISK_PAGE_LAST_INDEX 2047
+#define ZWORD_IN_BLOCK 127 /* NO OF WORD IN A BLOCK */
+#define ZNO_DISK_PAGES_FILE_REC 100
+#define ZMASK_PAGE_INDEX 0x7ff
+#define ZBIT_PAGE_INDEX 11 /* 8 KBYT PAGE = 2048 WORDS */
+#define ZSCAN_PROCEDURE 0
+#define ZCOPY_PROCEDURE 2
+#define ZSTORED_PROCEDURE_DELETE 3
+#define ZSTORED_PROCEDURE_FREE 0xffff
+#define ZMIN_PAGE_LIMIT_TUP_COMMITREQ 2
+#define ZUNDO_PAGE_HEADER_SIZE 2 /* SIZE OF UNDO PAGE HEADER */
+#endif
+
+class Dbtup: public SimulatedBlock {
+public:
+// State values
+enum State {
+ NOT_INITIALIZED = 0,
+ COMMON_AREA_PAGES = 1,
+ UNDO_RESTART_PAGES = 2,
+ UNDO_PAGES = 3,
+ READ_ONE_PAGE = 4,
+ CHECKPOINT_DATA_READ = 7,
+ CHECKPOINT_DATA_READ_PAGE_ZERO = 8,
+ CHECKPOINT_DATA_WRITE = 9,
+ CHECKPOINT_DATA_WRITE_LAST = 10,
+ CHECKPOINT_DATA_WRITE_FLUSH = 11,
+ CHECKPOINT_UNDO_READ = 12,
+ CHECKPOINT_UNDO_READ_FIRST = 13,
+ CHECKPOINT_UNDO_WRITE = 14,
+ CHECKPOINT_UNDO_WRITE_FLUSH = 15,
+ CHECKPOINT_TD_READ = 16,
+ IDLE = 17,
+ ACTIVE = 18,
+ SYSTEM_RESTART = 19,
+ NO_OTHER_OP = 20,
+ COMMIT_DELETE = 21,
+ TO_BE_COMMITTED = 22,
+ ABORTED = 23,
+ ALREADY_ABORTED_INSERT = 24,
+ ALREADY_ABORTED = 25,
+ ABORT_INSERT = 26,
+ ABORT_UPDATE = 27,
+ INIT = 28,
+ INITIAL_READ = 29,
+ INTERPRETED_EXECUTION = 30,
+ FINAL_READ = 31,
+ FINAL_UPDATE = 32,
+ DISCONNECTED = 33,
+ DEFINED = 34,
+ ERROR_WAIT_TUPKEYREQ = 35,
+ STARTED = 36,
+ NOT_DEFINED = 37,
+ COMPLETED = 38,
+ WAIT_ABORT = 39,
+ NORMAL_PAGE = 40,
+ COPY_PAGE = 41,
+ DELETE_BLOCK = 42,
+ WAIT_STORED_PROCEDURE_ATTR_INFO = 43,
+ DATA_FILE_READ = 45,
+ DATA_FILE_WRITE = 46,
+ LCP_DATA_FILE_READ = 47,
+ LCP_DATA_FILE_WRITE = 48,
+ LCP_DATA_FILE_WRITE_WITH_UNDO = 49,
+ LCP_DATA_FILE_CLOSE = 50,
+ LCP_UNDO_FILE_READ = 51,
+ LCP_UNDO_FILE_CLOSE = 52,
+ LCP_UNDO_FILE_WRITE = 53,
+ OPENING_DATA_FILE = 54,
+ INITIATING_RESTART_INFO = 55,
+ INITIATING_FRAGMENT = 56,
+ OPENING_UNDO_FILE = 57,
+ READING_RESTART_INFO = 58,
+ INIT_UNDO_SEGMENTS = 59,
+ READING_TAB_DESCR = 60,
+ READING_DATA_PAGES = 61,
+ WAIT_COPY_PROCEDURE = 62,
+ TOO_MUCH_AI = 63,
+ SAME_PAGE = 64,
+ DEFINING = 65,
+ TUPLE_BLOCKED = 66,
+ ERROR_WAIT_STORED_PROCREQ = 67
+};
+
+// Records
+/* ************** ATTRIBUTE INFO BUFFER RECORD ****************** */
+/* THIS RECORD IS USED AS A BUFFER FOR INCOMING AND OUTGOING DATA */
+/* ************************************************************** */
+struct Attrbufrec {
+ Uint32 attrbuf[ZATTRBUF_SIZE];
+}; /* p2c: size = 128 bytes */
+
+typedef Ptr<Attrbufrec> AttrbufrecPtr;
+
+/* ********** CHECKPOINT INFORMATION ************ */
+/* THIS RECORD HOLDS INFORMATION NEEDED TO */
+/* PERFORM A CHECKPOINT. IT'S POSSIBLE TO RUN */
+/* MULTIPLE CHECKPOINTS AT A TIME. THIS RECORD */
+/* MAKES IT POSSIBLE TO DISTINGER BETWEEN THE */
+/* DIFFERENT CHECKPOINTS. */
+/* ********************************************** */
+struct CheckpointInfo {
+ Uint32 lcpNextRec; /* NEXT RECORD IN FREELIST */
+ Uint32 lcpCheckpointVersion; /* VERSION OF THE CHECKPOINT */
+ Uint32 lcpLocalLogInfoP; /* POINTER TO A LOCAL LOG INFO RECORD */
+ Uint32 lcpUserptr; /* USERPOINTER TO THE BLOCK REQUESTING THE CP */
+ Uint32 lcpFragmentP; /* FRAGMENT POINTER TO WHICH THE CHECKPOINT APPLIES */
+ Uint32 lcpFragmentId; /* FRAGMENT ID */
+ Uint32 lcpTabPtr; /* TABLE POINTER */
+ Uint32 lcpDataBufferSegmentP; /* POINTER TO A DISK BUFFER SEGMENT POINTER (DATA) */
+ Uint32 lcpDataFileHandle; /* FILE HANDLES FOR DATA FILE. LOG FILE HANDLE IN LOCAL_LOG_INFO_RECORD */
+ /* FILE HANDLE TO THE OPEN DATA FILE */
+ Uint32 lcpNoOfPages;
+ Uint32 lcpThFreeFirst;
+ Uint32 lcpThFreeCopyFirst;
+ Uint32 lcpEmptyPrimPage;
+ Uint32 lcpNoCopyPagesAlloc;
+ Uint32 lcpTmpOperPtr; /* TEMPORARY STORAGE OF OPER_PTR DURING SAVE */
+ BlockReference lcpBlockref; /* BLOCKREFERENCE TO THE BLOCK REQUESTING THE CP */
+};
+typedef Ptr<CheckpointInfo> CheckpointInfoPtr;
+
+/* *********** DISK BUFFER SEGMENT INFO ********* */
+/* THIS RECORD HOLDS INFORMATION NEEDED DURING */
+/* A WRITE OF THE DATA BUFFER TO DISK. WHEN THE */
+/* WRITE SIGNAL IS SENT A POINTER TO THIS RECORD */
+/* IS INCLUDED. WHEN THE WRITE IS COMPLETED AND */
+/* CONFIRMED THE PTR TO THIS RECORD IS RETURNED */
+/* AND THE BUFFER PAGES COULD EASILY BE LOCATED */
+/* AND DEALLOCATED. THE CHECKPOINT_INFO_VERSION */
+/* KEEPS TRACK OF THE CHECPOINT_INFO_RECORD THAT */
+/* INITIATED THE WRITE AND THE CP_PAGE_TO_DISK */
+/* ELEMENT COULD BE INCREASED BY THE NUMBER OF */
+/* PAGES WRITTEN. */
+/* ********************************************** */
+struct DiskBufferSegmentInfo {
+ Uint32 pdxDataPage[16]; /* ARRAY OF DATA BUFFER PAGES */
+ Uint32 pdxUndoBufferSet[2];
+ Uint32 pdxNextRec;
+ State pdxBuffertype;
+ State pdxOperation;
+ /*---------------------------------------------------------------------------*/
+ /* PDX_FLAGS BITS AND THEIR USAGE: */
+ /* BIT 0 1 COMMENT */
+ /*---------------------------------------------------------------------------*/
+ /* 0 SEGMENT INVALID SEGMENT VALID USED DURING READS */
+ /* 1-15 NOT USED */
+ /*---------------------------------------------------------------------------*/
+ Uint32 pdxCheckpointInfoP; /* USED DURING LOCAL CHKP */
+ Uint32 pdxRestartInfoP; /* USED DURING RESTART */
+ Uint32 pdxLocalLogInfoP; /* POINTS TO A LOCAL LOG INFO */
+ Uint32 pdxFilePage; /* START PAGE IN FILE */
+ Uint32 pdxNumDataPages; /* NUMBER OF DATA PAGES */
+};
+typedef Ptr<DiskBufferSegmentInfo> DiskBufferSegmentInfoPtr;
+
+struct Fragoperrec {
+ bool definingFragment;
+ Uint32 nextFragoprec;
+ Uint32 lqhPtrFrag;
+ Uint32 fragidFrag;
+ Uint32 tableidFrag;
+ Uint32 fragPointer;
+ Uint32 attributeCount;
+ Uint32 freeNullBit;
+ Uint32 noOfNewAttrCount;
+ BlockReference lqhBlockrefFrag;
+};
+typedef Ptr<Fragoperrec> FragoperrecPtr;
+
+struct Fragrecord {
+ Uint32 nextStartRange;
+ Uint32 currentPageRange;
+ Uint32 rootPageRange;
+ Uint32 noOfPages;
+ Uint32 emptyPrimPage;
+
+ Uint32 firstusedOprec;
+
+ Uint32 thFreeFirst;
+ Uint32 thFreeCopyFirst;
+ Uint32 noCopyPagesAlloc;
+
+ Uint32 checkpointVersion;
+ Uint32 minPageNotWrittenInCheckpoint;
+ Uint32 maxPageWrittenInCheckpoint;
+ State fragStatus;
+ Uint32 fragTableId;
+ Uint32 fragmentId;
+ Uint32 nextfreefrag;
+};
+typedef Ptr<Fragrecord> FragrecordPtr;
+
+ /* ************ LOCAL LOG FILE INFO ************* */
+ /* THIS RECORD HOLDS INFORMATION NEEDED DURING */
+ /* CHECKPOINT AND RESTART. THERE ARE FOUR */
+ /* PARALLELL UNDO LOG FILES, EACH ONE REPRESENTED */
+ /* BY AN ENTITY OF THIS RECORD. */
+ /* BECAUSE EACH FILE IS SHARED BETWEEN FOUR */
+ /* TABLES AND HAS ITS OWN PAGEPOINTERS AND */
+ /* WORDPOINTERS. */
+ /* ********************************************** */
+struct LocalLogInfo {
+ Uint32 lliActiveLcp; /* NUMBER OF ACTIVE LOCAL CHECKPOINTS ON THIS FILE */
+ Uint32 lliEndPageId; /* PAGE IDENTIFIER OF LAST PAGE WITH LOG DATA */
+ Uint32 lliPrevRecordId; /* PREVIOUS RECORD IN THIS LOGFILE */
+ Uint32 lliLogFilePage; /* PAGE IN LOGFILE */
+ Uint32 lliNumFragments; /* NO OF FRAGMENTS RESTARTING FROM THIS LOCAL LOG */
+ Uint32 lliUndoBufferSegmentP; /* POINTER TO A DISK BUFFER SEGMENT POINTER (UNDO) */
+ Uint32 lliUndoFileHandle; /* FILE HANDLE OF UNDO LOG FILE */
+ Uint32 lliUndoPage; /* UNDO PAGE IN BUFFER */
+ Uint32 lliUndoWord;
+ Uint32 lliUndoPagesToDiskWithoutSynch;
+};
+typedef Ptr<LocalLogInfo> LocalLogInfoPtr;
+
+struct Operationrec {
+// Easy to remove (2 words)
+ Uint32 attroutbufLen;
+ Uint32 logSize;
+
+// Needed (20 words)
+ State tupleState;
+ Uint32 prevActiveOp;
+ Uint32 nextActiveOp;
+ Uint32 nextOprecInList;
+ Uint32 prevOprecInList;
+ Uint32 tableRef;
+ Uint32 fragId;
+ Uint32 fragmentPtr;
+ Uint32 fragPageId;
+ Uint32 realPageId;
+ bool undoLogged;
+ Uint32 realPageIdC;
+ Uint32 fragPageIdC;
+ Uint32 firstAttrinbufrec;
+ Uint32 lastAttrinbufrec;
+ Uint32 attrinbufLen;
+ Uint32 currentAttrinbufLen;
+ Uint32 userpointer;
+ State transstate;
+ Uint32 savePointId;
+
+// Easy to remove (3 words)
+ Uint32 tcOperationPtr;
+ Uint32 transid1;
+ Uint32 transid2;
+
+// Needed (2 words)
+ Uint16 pageIndex;
+ Uint16 pageOffset;
+ Uint16 pageOffsetC;
+ Uint16 pageIndexC;
+// Hard to remove
+ Uint16 tupVersion;
+
+// Easy to remove (1.5 word)
+ BlockReference recBlockref;
+ BlockReference userblockref;
+ Uint16 storedProcedureId;
+
+ Uint8 inFragList;
+ Uint8 inActiveOpList;
+ Uint8 deleteInsertFlag;
+
+// Needed (1 word)
+ Uint8 dirtyOp;
+ Uint8 interpretedExec;
+ Uint8 optype;
+ Uint8 opSimple;
+
+// Used by triggers
+ Uint32 primaryReplica;
+ BlockReference coordinatorTC;
+ Uint32 tcOpIndex;
+ Uint32 gci;
+ Uint32 noFiredTriggers;
+ Uint32 hashValue; // only used in TUP_COMMITREQ
+ Bitmask<MAXNROFATTRIBUTESINWORDS> changeMask;
+};
+typedef Ptr<Operationrec> OperationrecPtr;
+
+struct Page {
+ Uint32 pageWord[ZWORDS_ON_PAGE];
+};
+typedef Ptr<Page> PagePtr;
+
+ /* ****************************** PAGE RANGE RECORD ************************** */
+ /* PAGE RANGES AND BASE PAGE ID. EACH RANGE HAS A CORRESPONDING BASE PAGE ID */
+ /* THAT IS USED TO CALCULATE REAL PAGE ID FROM A FRAGMENT PAGE ID AND A TABLE */
+ /* REFERENCE. */
+ /* THE PAGE RANGES ARE ORGANISED IN A B-TREE FASHION WHERE THE VARIABLE TYPE */
+ /* SPECIFIES IF A LEAF NODE HAS BEEN REACHED. IF A LEAF NODE HAS BEEN REACHED */
+ /* THEN BASE_PAGE_ID IS THE BASE_PAGE_ID OF THE SET OF PAGES THAT WAS */
+ /* ALLOCATED IN THAT RANGE. OTHERWISE BASE_PAGE_ID IS THE POINTER TO THE NEXT */
+ /* PAGE_RANGE RECORD. */
+ /* *************************************************************************** */
+struct PageRange {
+ Uint32 startRange[4]; /* START OF RANGE */
+ Uint32 endRange[4]; /* END OF THIS RANGE */
+ Uint32 basePageId[4]; /* BASE PAGE ID. */
+/*---- VARIABLE BASE_PAGE_ID2 (4) 8 DS NEEDED WHEN SUPPORTING 40 BIT PAGE ID -------*/
+ Uint8 type[4]; /* TYPE OF BASE PAGE ID */
+ Uint32 nextFree; /* NEXT FREE PAGE RANGE RECORD */
+ Uint32 parentPtr; /* THE PARENT TO THE PAGE RANGE REC IN THE B-TREE */
+ Uint8 currentIndexPos;
+};
+typedef Ptr<PageRange> PageRangePtr;
+
+ /* *********** PENDING UNDO WRITE INFO ********** */
+ /* THIS RECORD HOLDS INFORMATION NEEDED DURING */
+ /* A FILE OPEN OPERATION */
+ /* IF THE FILE OPEN IS A PART OF A CHECKPOINT THE */
+ /* CHECKPOINT_INFO_P WILL HOLD A POINTER TO THE */
+ /* CHECKPOINT_INFOR_PTR RECORD */
+ /* IF IT IS A PART OF RESTART THE PFO_RESTART_INFO*/
+ /* ELEMENT WILL POINT TO A RESTART INFO RECORD */
+ /* ********************************************** */
+struct PendingFileOpenInfo {
+ Uint32 pfoNextRec;
+ State pfoOpenType;
+ Uint32 pfoCheckpointInfoP;
+ Uint32 pfoRestartInfoP;
+};
+typedef Ptr<PendingFileOpenInfo> PendingFileOpenInfoPtr;
+
+struct RestartInfoRecord {
+ Uint32 sriNextRec;
+ State sriState; /* BLOCKREFERENCE TO THE REQUESTING BLOCK */
+ Uint32 sriUserptr; /* USERPOINTER TO THE REQUESTING BLOCK */
+ Uint32 sriDataBufferSegmentP; /* POINTER TO A DISK BUFFER SEGMENT POINTER (DATA) */
+ Uint32 sriDataFileHandle; /* FILE HANDLE TO THE OPEN DATA FILE */
+ Uint32 sriCheckpointVersion; /* CHECKPOINT VERSION TO RESTART FROM */
+ Uint32 sriFragid; /* FRAGMENT ID */
+ Uint32 sriFragP; /* FRAGMENT POINTER */
+ Uint32 sriTableId; /* TABLE ID */
+ Uint32 sriLocalLogInfoP; /* POINTER TO A LOCAL LOG INFO RECORD */
+ Uint32 sriNumDataPages; /* NUMBER OF DATA PAGES TO READ */
+ Uint32 sriCurDataPageFromBuffer; /* THE CHECKPOINT IS COMPLETED */
+ BlockReference sriBlockref;
+};
+typedef Ptr<RestartInfoRecord> RestartInfoRecordPtr;
+
+ /* ************* TRIGGER DATA ************* */
+ /* THIS RECORD FORMS LISTS OF ACTIVE */
+ /* TRIGGERS FOR EACH TABLE. */
+ /* THE RECORDS ARE MANAGED BY A TRIGGER */
+ /* POOL wHERE A TRIGGER RECORD IS SEIZED */
+ /* WHEN A TRIGGER IS ACTIVATED AND RELEASED */
+ /* WHEN THE TRIGGER IS DEACTIVATED. */
+ /* **************************************** */
+struct TupTriggerData {
+
+ /**
+ * Trigger id, used by DICT/TRIX to identify the trigger
+ */
+ Uint32 triggerId;
+
+ /**
+ * Index id is needed for ordered index.
+ */
+ Uint32 indexId;
+
+ /**
+ * Trigger type etc, defines what the trigger is used for
+ */
+ TriggerType::Value triggerType;
+ TriggerActionTime::Value triggerActionTime;
+ TriggerEvent::Value triggerEvent;
+ /**
+ * Receiver block
+ */
+ Uint32 m_receiverBlock;
+
+ /**
+ * Monitor all replicas, i.e. trigger will fire on all nodes where tuples
+ * are stored
+ */
+ bool monitorReplicas;
+
+ /**
+ * Monitor all attributes, the trigger monitors all changes to attributes
+ * in the table
+ */
+ bool monitorAllAttributes;
+
+ /**
+ * Send only changed attributes at trigger firing time.
+ */
+ bool sendOnlyChangedAttributes;
+
+ /**
+ * Send also before values at trigger firing time.
+ */
+ bool sendBeforeValues;
+
+ /**
+ * Attribute mask, defines what attributes are to be monitored
+ * Can be seen as a compact representation of SQL column name list
+ */
+ Bitmask<MAXNROFATTRIBUTESINWORDS> attributeMask;
+
+ /**
+ * Next ptr (used in pool/list)
+ */
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+
+ /**
+ * Prev pointer (used in list)
+ */
+ Uint32 prevList;
+
+ inline void print(NdbOut & s) const { s << "[TriggerData = " << triggerId << "]"; };
+};
+
+typedef Ptr<TupTriggerData> TriggerPtr;
+
+/**
+ * Pool of trigger data record
+ */
+ArrayPool<TupTriggerData> c_triggerPool;
+
+ /* ************ TABLE RECORD ************ */
+ /* THIS RECORD FORMS A LIST OF TABLE */
+ /* REFERENCE INFORMATION. ONE RECORD */
+ /* PER TABLE REFERENCE. */
+ /* ************************************** */
+struct Tablerec {
+ Tablerec(ArrayPool<TupTriggerData> & triggerPool) :
+ afterInsertTriggers(triggerPool),
+ afterDeleteTriggers(triggerPool),
+ afterUpdateTriggers(triggerPool),
+ subscriptionInsertTriggers(triggerPool),
+ subscriptionDeleteTriggers(triggerPool),
+ subscriptionUpdateTriggers(triggerPool),
+ constraintUpdateTriggers(triggerPool),
+ tuxCustomTriggers(triggerPool)
+ {}
+
+ Bitmask<MAXNROFATTRIBUTESINWORDS> notNullAttributeMask;
+
+ ReadFunction* readFunctionArray;
+ UpdateFunction* updateFunctionArray;
+
+ Uint32 readKeyArray;
+ Uint32 tabDescriptor;
+ Uint32 attributeGroupDescriptor;
+
+ bool GCPIndicator;
+ bool checksumIndicator;
+
+ Uint16 tupheadsize;
+ Uint16 noOfAttr;
+ Uint16 noOfKeyAttr;
+ Uint16 noOfNewAttr;
+ Uint16 noOfNullAttr;
+ Uint16 noOfAttributeGroups;
+
+ Uint8 tupChecksumIndex;
+ Uint8 tupNullIndex;
+ Uint8 tupNullWords;
+ Uint8 tupGCPIndex;
+
+ // Lists of trigger data for active triggers
+ ArrayList<TupTriggerData> afterInsertTriggers;
+ ArrayList<TupTriggerData> afterDeleteTriggers;
+ ArrayList<TupTriggerData> afterUpdateTriggers;
+ ArrayList<TupTriggerData> subscriptionInsertTriggers;
+ ArrayList<TupTriggerData> subscriptionDeleteTriggers;
+ ArrayList<TupTriggerData> subscriptionUpdateTriggers;
+ ArrayList<TupTriggerData> constraintUpdateTriggers;
+
+ // List of ordered indexes
+ ArrayList<TupTriggerData> tuxCustomTriggers;
+
+ Uint32 fragid[2 * NO_OF_FRAG_PER_NODE];
+ Uint32 fragrec[2 * NO_OF_FRAG_PER_NODE];
+
+ struct {
+ Uint32 tabUserPtr;
+ Uint32 tabUserRef;
+ } m_dropTable;
+ State tableStatus;
+};
+
+typedef Ptr<Tablerec> TablerecPtr;
+
+struct storedProc {
+ Uint32 storedLinkFirst;
+ Uint32 storedLinkLast;
+ Uint32 storedCounter;
+ Uint32 nextPool;
+ Uint16 storedCode;
+ Uint16 storedProcLength;
+};
+
+typedef Ptr<storedProc> StoredProcPtr;
+
+ArrayPool<storedProc> c_storedProcPool;
+
+/* **************************** TABLE_DESCRIPTOR RECORD ******************************** */
+/* THIS VARIABLE IS USED TO STORE TABLE DESCRIPTIONS. A TABLE DESCRIPTION IS STORED AS A */
+/* CONTIGUOS ARRAY IN THIS VARIABLE. WHEN A NEW TABLE IS ADDED A CHUNK IS ALLOCATED IN */
+/* THIS RECORD. WHEN ATTRIBUTES ARE ADDED TO THE TABLE, A NEW CHUNK OF PROPER SIZE IS */
+/* ALLOCATED AND ALL DATA IS COPIED TO THIS NEW CHUNK AND THEN THE OLD CHUNK IS PUT IN */
+/* THE FREE LIST. EACH TABLE IS DESCRIBED BY A NUMBER OF TABLE DESCRIPTIVE ATTRIBUTES */
+/* AND A NUMBER OF ATTRIBUTE DESCRIPTORS AS SHOWN IN FIGURE BELOW */
+/* */
+/* WHEN ALLOCATING A TABLE DESCRIPTOR THE SIZE IS ALWAYS A MULTIPLE OF 16 WORDS. */
+/* */
+/* ---------------------------------------------- */
+/* | TRAILER USED FOR ALLOC/DEALLOC | */
+/* ---------------------------------------------- */
+/* | TABLE DESCRIPTIVE ATTRIBUTES | */
+/* ---------------------------------------------- */
+/* | ATTRIBUTE DESCRIPTION 1 | */
+/* ---------------------------------------------- */
+/* | ATTRIBUTE DESCRIPTION 2 | */
+/* ---------------------------------------------- */
+/* | | */
+/* | | */
+/* | | */
+/* ---------------------------------------------- */
+/* | ATTRIBUTE DESCRIPTION N | */
+/* ---------------------------------------------- */
+/* */
+/* THE TABLE DESCRIPTIVE ATTRIBUTES CONTAINS THE FOLLOWING ATTRIBUTES: */
+/* */
+/* ---------------------------------------------- */
+/* | HEADER (TYPE OF INFO) | */
+/* ---------------------------------------------- */
+/* | SIZE OF WHOLE CHUNK (INCL. TRAILER) | */
+/* ---------------------------------------------- */
+/* | TABLE IDENTITY | */
+/* ---------------------------------------------- */
+/* | FRAGMENT IDENTITY | */
+/* ---------------------------------------------- */
+/* | NUMBER OF ATTRIBUTES | */
+/* ---------------------------------------------- */
+/* | SIZE OF FIXED ATTRIBUTES | */
+/* ---------------------------------------------- */
+/* | NUMBER OF NULL FIELDS | */
+/* ---------------------------------------------- */
+/* | NOT USED | */
+/* ---------------------------------------------- */
+/* */
+/* THESE ATTRIBUTES ARE ALL ONE R-VARIABLE IN THE RECORD. */
+/* NORMALLY ONLY ONE TABLE DESCRIPTOR IS USED. DURING SCHEMA CHANGES THERE COULD */
+/* HOWEVER EXIST MORE THAN ONE TABLE DESCRIPTION SINCE THE SCHEMA CHANGE OF VARIOUS */
+/* FRAGMENTS ARE NOT SYNCHRONISED. THIS MEANS THAT ALTHOUGH THE SCHEMA HAS CHANGED */
+/* IN ALL FRAGMENTS, BUT THE FRAGMENTS HAVE NOT REMOVED THE ATTRIBUTES IN THE SAME */
+/* TIME-FRAME. THEREBY SOME ATTRIBUTE INFORMATION MIGHT DIFFER BETWEEN FRAGMENTS. */
+/* EXAMPLES OF ATTRIBUTES THAT MIGHT DIFFER ARE SIZE OF FIXED ATTRIBUTES, NUMBER OF */
+/* ATTRIBUTES, FIELD START WORD, START BIT. */
+/* */
+/* AN ATTRIBUTE DESCRIPTION CONTAINS THE FOLLOWING ATTRIBUTES: */
+/* */
+/* ---------------------------------------------- */
+/* | Field Type, 4 bits (LSB Bits) | */
+/* ---------------------------------------------- */
+/* | Attribute Size, 4 bits | */
+/* ---------------------------------------------- */
+/* | NULL indicator 1 bit | */
+/* ---------------------------------------------- */
+/* | Indicator if TUP stores attr. 1 bit | */
+/* ---------------------------------------------- */
+/* | Not used 6 bits | */
+/* ---------------------------------------------- */
+/* | No. of elements in fixed array 16 bits | */
+/* ---------------------------------------------- */
+/* ---------------------------------------------- */
+/* | Field Start Word, 21 bits (LSB Bits) | */
+/* ---------------------------------------------- */
+/* | NULL Bit, 11 bits | */
+/* ---------------------------------------------- */
+/* */
+/* THE ATTRIBUTE SIZE CAN BE 1,2,4,8,16,32,64 AND 128 BITS. */
+/* */
+/* THE UNUSED PARTS OF THE RECORDS ARE PUT IN A LINKED LIST OF FREE PARTS. EACH OF */
+/* THOSE FREE PARTS HAVE THREE RECORDS ASSIGNED AS SHOWN IN THIS STRUCTURE */
+/* ALL FREE PARTS ARE SET INTO A CHUNK LIST WHERE EACH CHUNK IS AT LEAST 16 WORDS */
+/* */
+/* ---------------------------------------------- */
+/* | HEADER = RNIL | */
+/* ---------------------------------------------- */
+/* | SIZE OF FREE AREA | */
+/* ---------------------------------------------- */
+/* | POINTER TO PREVIOUS FREE AREA | */
+/* ---------------------------------------------- */
+/* | POINTER TO NEXT FREE AREA | */
+/* ---------------------------------------------- */
+/* */
+/* IF THE POINTER TO THE NEXT AREA IS RNIL THEN THIS IS THE LAST FREE AREA. */
+/* */
+/*****************************************************************************************/
+struct TableDescriptor {
+ Uint32 tabDescr;
+};
+typedef Ptr<TableDescriptor> TableDescriptorPtr;
+
+struct HostBuffer {
+ bool inPackedList;
+ Uint32 packetLenRC;
+ Uint32 noOfPacketsRC;
+ Uint32 packetBufferRC[29];
+ Uint32 packetLenTA;
+ Uint32 noOfPacketsTA;
+ Uint32 packetBufferTA[30];
+};
+typedef Ptr<HostBuffer> HostBufferPtr;
+
+ /* **************** UNDO PAGE RECORD ******************* */
+ /* THIS RECORD FORMS AN UNDO PAGE CONTAINING A NUMBER OF */
+ /* DATA WORDS. CURRENTLY THERE ARE 2048 WORDS ON A PAGE */
+ /* EACH OF 32 BITS (4 BYTES) WHICH FORMS AN UNDO PAGE */
+ /* WITH A TOTAL OF 8192 BYTES */
+ /* ***************************************************** */
+struct UndoPage {
+ Uint32 undoPageWord[ZWORDS_ON_PAGE]; /* 32 KB */
+};
+typedef Ptr<UndoPage> UndoPagePtr;
+
+ /*
+ * Build index operation record.
+ */
+ struct BuildIndexRec {
+ // request cannot use signal class due to extra members
+ Uint32 m_request[BuildIndxReq::SignalLength];
+ Uint32 m_triggerPtrI; // the index trigger
+ Uint32 m_fragNo; // fragment number under Tablerec
+ Uint32 m_pageId; // logical fragment page id
+ Uint32 m_tupleNo; // tuple number on page (pageIndex >> 1)
+ BuildIndxRef::ErrorCode m_errorCode;
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+ };
+ typedef Ptr<BuildIndexRec> BuildIndexPtr;
+ ArrayPool<BuildIndexRec> c_buildIndexPool;
+ ArrayList<BuildIndexRec> c_buildIndexList;
+ Uint32 c_noOfBuildIndexRec;
+
+public:
+ Dbtup(const class Configuration &);
+ virtual ~Dbtup();
+
+private:
+ BLOCK_DEFINES(Dbtup);
+
+ // Transit signals
+ void execDEBUG_SIG(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+
+ // Received signals
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execSEND_PACKED(Signal* signal);
+ void execSTTOR(Signal* signal);
+ void execTUP_LCPREQ(Signal* signal);
+ void execEND_LCPREQ(Signal* signal);
+ void execSTART_RECREQ(Signal* signal);
+ void execMEMCHECKREQ(Signal* signal);
+ void execTUPSEIZEREQ(Signal* signal);
+ void execTUPRELEASEREQ(Signal* signal);
+ void execSTORED_PROCREQ(Signal* signal);
+ void execTUPFRAGREQ(Signal* signal);
+ void execTUP_ADD_ATTRREQ(Signal* signal);
+ void execTUP_COMMITREQ(Signal* signal);
+ void execTUP_ABORTREQ(Signal* signal);
+ void execTUP_SRREQ(Signal* signal);
+ void execTUP_PREPLCPREQ(Signal* signal);
+ void execFSOPENCONF(Signal* signal);
+ void execFSOPENREF(Signal* signal);
+ void execFSCLOSECONF(Signal* signal);
+ void execFSCLOSEREF(Signal* signal);
+ void execFSWRITECONF(Signal* signal);
+ void execFSWRITEREF(Signal* signal);
+ void execFSREADCONF(Signal* signal);
+ void execFSREADREF(Signal* signal);
+ void execNDB_STTOR(Signal* signal);
+ void execSIZEALT_REP(Signal* signal);
+ void execSET_VAR_REQ(Signal* signal);
+ void execDROP_TAB_REQ(Signal* signal);
+ void execALTER_TAB_REQ(Signal* signal);
+ void execFSREMOVECONF(Signal* signal);
+ void execFSREMOVEREF(Signal* signal);
+ void execTUP_ALLOCREQ(Signal* signal);
+ void execTUP_DEALLOCREQ(Signal* signal);
+ void execTUP_WRITELOG_REQ(Signal* signal);
+
+ // Ordered index related
+ void execTUP_READ_ATTRS(Signal* signal);
+ void execTUP_QUERY_TH(Signal* signal);
+ void execTUP_STORE_TH(Signal* signal);
+ void execBUILDINDXREQ(Signal* signal);
+ void buildIndex(Signal* signal, Uint32 buildPtrI);
+ void buildIndexReply(Signal* signal, const BuildIndexRec* buildRec);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+// Methods to handle execution of TUPKEYREQ + ATTRINFO.
+//
+// Module Execution Manager
+//
+// The TUPKEYREQ signal is central to this block. This signal is used
+// by everybody that needs to read data residing in DBTUP. The data is
+// read using an interpreter approach.
+//
+// Operations only needing to read execute a simplified version of the
+// interpreter where the only instruction is read Attribute to send.
+// Operations only needing to update the record (insert or update)
+// execute a simplified version of the interpreter where the only
+// instruction is write Attribute.
+//
+// Currently TUPKEYREQ is used in the following situations.
+// 1) Normal transaction execution. Can be any of the types described
+// below.
+// 2) Execution of fragment redo log during system restart.
+// In this situation there will only be normal updates, inserts
+// and deletes performed.
+// 3) A special type of normal transaction execution is to write the
+// records arriving from the primary replica in the node restart
+// processing. This will always be normal write operations which
+// are translated to inserts or updates before arriving to TUP.
+// 4) Scan processing. The scan processing will use normal reads or
+// interpreted reads in their execution. There will be one TUPKEYREQ
+// signal for each record processed.
+// 5) Copy fragment processing. This is a special type of scan used in the
+// primary replica at system restart. It reads the entire reads and
+// converts those to writes to the starting node. In this special case
+// LQH acts as an API node and receives also the ATTRINFO sent in the
+// TRANSID_AI signals.
+//
+// Signal Diagram:
+//
+// In Signals:
+// -----------
+//
+// Logically there is one request TUPKEYREQ which requests to read/write data
+// of one tuple in the database. Since the definition of what to read and write
+// can be bigger than the maximum signal size we segment the signal. The definition
+// of what to read/write/interpreted program is sent before the TUPKEYREQ signal.
+//
+// ---> ATTRINFO
+// ...
+// ---> ATTRINFO
+// ---> TUPKEYREQ
+// The number of ATTRINFO signals can be anything between 0 and upwards.
+// The total size of the ATTRINFO is not allowed to be more than 16384 words.
+// There is always one and only one TUPKEYREQ.
+//
+// Response Signals (successful case):
+//
+// Simple/Dirty Read Operation
+// ---------------------------
+//
+// <---- TRANSID_AI (to API)
+// ...
+// <---- TRANSID_AI (to API)
+// <---- READCONF (to API)
+// <---- TUPKEYCONF (to LQH)
+// There is always exactly one READCONF25 sent last. The number of
+// TRANSID_AI is dependent on how much that was read. The maximum size
+// of the ATTRINFO sent back is 16384 words. The signals are sent
+// directly to the application with an address provided by the
+// TUPKEYREQ signal.
+// A positive response signal is also sent to LQH.
+//
+// Normal Read Operation
+// ---------------------
+//
+// <---- TRANSID_AI (to API)
+// ...
+// <---- TRANSID_AI (to API)
+// <---- TUPKEYCONF (to LQH)
+// The number of TRANSID_AI is dependent on how much that was read.
+// The maximum size of the ATTRINFO sent back is 16384 words. The
+// signals are sent directly to the application with an address
+// provided by the TUPKEYREQ signal.
+// A positive response signal is also sent to LQH.
+//
+// Normal update/insert/delete operation
+// -------------------------------------
+//
+// <---- TUPKEYCONF
+// After successful updating of the tuple LQH is informed of this.
+//
+// Delete with read
+// ----------------
+//
+// Will behave as a normal read although it also prepares the
+// deletion of the tuple.
+//
+// Interpreted Update
+// ------------------
+//
+// <---- TRANSID_AI (to API)
+// ...
+// <---- TRANSID_AI (to API)
+// <---- TUP_ATTRINFO (to LQH)
+// ...
+// <---- TUP_ATTRINFO (to LQH)
+// <---- TUPKEYCONF (to LQH)
+//
+// The interpreted Update contains five sections:
+// The first section performs read Attribute operations
+// that send results back to the API.
+//
+// The second section executes the interpreted program
+// where data from attributes can be updated and it
+// can also read attribute values into the registers.
+//
+// The third section performs unconditional updates of
+// attributes.
+//
+// The fourth section can read the attributes to be sent to the
+// API after updating the record.
+//
+// The fifth section contains subroutines used by the interpreter
+// in the second section.
+//
+// All types of interpreted programs contains the same five sections.
+// The only difference is that only interpreted updates can update
+// attributes. Interpreted inserts are not allowed.
+//
+// Interpreted Updates have to send back the information about the
+// attributes they have updated. This information will be shipped to
+// the log and also to any other replicas. Thus interpreted updates
+// are only performed in the primary replica. The fragment redo log
+// in LQH will contain information so that normal update/inserts/deletes
+// can be performed using TUPKEYREQ.
+//
+// Interpreted Read
+// ----------------
+//
+// From a signalling point of view the Interpreted Read behaves as
+// as a Normal Read. The interpreted Read is often used by Scan's.
+//
+// Interpreted Delete
+// ------------------
+//
+// <---- TUPKEYCONF
+// After successful prepartion to delete the tuple LQH is informed
+// of this.
+//
+// Interpreted Delete with Read
+// ----------------------------
+//
+// From a signalling point of view an interpreted delete with read
+// behaves as a normal read.
+//
+// Continuation after successful case:
+//
+// After a read of any kind the operation record is ready to be used
+// again by a new operation.
+//
+// Any updates, inserts or deletes waits for either of two messages.
+// A commit specifying that the operation is to be performed for real
+// or an abort specifying that the operation is to be rolled back and
+// the record to be restored in its original format.
+//
+// This is handled by the module Transaction Manager.
+//
+// Response Signals (unsuccessful case):
+//
+// <---- TUPKEYREF (to LQH)
+// A signal is sent back to LQH informing about the unsuccessful
+// operation. In this case TUP waits for an abort signal to arrive
+// before the operation record is ready for the next operation.
+// This is handled by the Transaction Manager.
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+
+// *****************************************************************
+// Signal Reception methods.
+// *****************************************************************
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void execTUPKEYREQ(Signal* signal);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void execATTRINFO(Signal* signal);
+
+// Trigger signals
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void execCREATE_TRIG_REQ(Signal* signal);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void execDROP_TRIG_REQ(Signal* signal);
+
+// *****************************************************************
+// Support methods for ATTRINFO.
+// *****************************************************************
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void handleATTRINFOforTUPKEYREQ(Signal* signal,
+ Uint32 length,
+ Operationrec * const regOperPtr);
+
+// *****************************************************************
+// Setting up the environment for reads, inserts, updates and deletes.
+// *****************************************************************
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int handleReadReq(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr,
+ Page* pagePtr);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int handleUpdateReq(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Page* const pagePtr);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int handleInsertReq(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Page* const pagePtr);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int handleDeleteReq(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Page* const pagePtr);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int updateStartLab(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr,
+ Page* const pagePtr);
+
+// *****************************************************************
+// Interpreter Handling methods.
+// *****************************************************************
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int interpreterStartLab(Signal* signal,
+ Page* const pagePtr,
+ Uint32 TupHeadOffset);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int interpreterNextLab(Signal* signal,
+ Page* const pagePtr,
+ Uint32 TupHeadOffset,
+ Uint32* logMemory,
+ Uint32* mainProgram,
+ Uint32 TmainProgLen,
+ Uint32* subroutineProg,
+ Uint32 TsubroutineLen,
+ Uint32 * tmpArea,
+ Uint32 tmpAreaSz);
+
+// *****************************************************************
+// Signal Sending methods.
+// *****************************************************************
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void sendReadAttrinfo(Signal* signal,
+ Uint32 TnoOfData,
+ const Operationrec * const regOperPtr);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void sendLogAttrinfo(Signal* signal,
+ Uint32 TlogSize,
+ Operationrec * const regOperPtr);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void sendTUPKEYCONF(Signal* signal, Operationrec *
+ const regOperPtr,
+ Uint32 TlogSize);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+// *****************************************************************
+// The methods that perform the actual read and update of attributes
+// in the tuple.
+// *****************************************************************
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int readAttributes(Page* const pagePtr,
+ Uint32 TupHeadOffset,
+ Uint32* inBuffer,
+ Uint32 inBufLen,
+ Uint32* outBuffer,
+ Uint32 TmaxRead);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int readAttributesWithoutHeader(Page* const pagePtr,
+ Uint32 TupHeadOffset,
+ Uint32* inBuffer,
+ Uint32 inBufLen,
+ Uint32* outBuffer,
+ Uint32* attrBuffer,
+ Uint32 TmaxRead);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int updateAttributes(Page* const pagePtr,
+ Uint32 TupHeadOffset,
+ Uint32* inBuffer,
+ Uint32 inBufLen);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readFixedSizeTHOneWordNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateFixedSizeTHOneWordNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readFixedSizeTHTwoWordNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateFixedSizeTHTwoWordNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readFixedSizeTHManyWordNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateFixedSizeTHManyWordNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readFixedSizeTHOneWordNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateFixedSizeTHOneWordNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readFixedSizeTHTwoWordNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateFixedSizeTHTwoWordNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readFixedSizeTHManyWordNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readFixedSizeTHZeroWordNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateFixedSizeTHManyWordNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readVariableSizedAttr(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateVariableSizedAttr(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readVarSizeUnlimitedNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateVarSizeUnlimitedNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readVarSizeUnlimitedNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateVarSizeUnlimitedNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readBigVarSizeNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateBigVarSizeNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readBigVarSizeNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateBigVarSizeNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readSmallVarSizeNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateSmallVarSizeNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readSmallVarSizeNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateSmallVarSizeNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readDynFixedSize(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateDynFixedSize(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readDynVarSizeUnlimited(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateDynVarSizeUnlimited(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readDynBigVarSize(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateDynBigVarSize(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool readDynSmallVarSize(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool updateDynSmallVarSize(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ bool nullFlagCheck(Uint32 attrDes2);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void setUpQueryRoutines(Tablerec* const regTabPtr);
+
+// *****************************************************************
+// Service methods.
+// *****************************************************************
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void copyAttrinfo(Signal* signal, Operationrec * const regOperPtr, Uint32* inBuffer);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void initOpConnection(Operationrec* const regOperPtr);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void initOperationrec(Signal* signal);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int initStoredOperationrec(Operationrec* const regOperPtr,
+ Uint32 storedId);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void insertActiveOpList(Signal* signal,
+ OperationrecPtr regOperPtr,
+ Page * const pagePtr,
+ Uint32 pageOffset);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void linkOpIntoFragList(OperationrecPtr regOperPtr,
+ Fragrecord* const regFragPtr);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void bufferREADCONF(Signal* signal, BlockReference aRef, Uint32* buffer, Uint32 Tlen);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void bufferTRANSID_AI(Signal* signal, BlockReference aRef, Uint32* buffer, Uint32 Tlen);
+
+//------------------------------------------------------------------
+// Trigger handling routines
+//------------------------------------------------------------------
+ ArrayList<TupTriggerData>* findTriggerList(Tablerec* table,
+ TriggerType::Value ttype,
+ TriggerActionTime::Value ttime,
+ TriggerEvent::Value tevent);
+
+ bool createTrigger(Tablerec* table, const CreateTrigReq* req);
+
+ Uint32 dropTrigger(Tablerec* table, const DropTrigReq* req);
+
+ void checkImmediateTriggersAfterInsert(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const tablePtr);
+
+ void checkImmediateTriggersAfterUpdate(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const tablePtr);
+
+ void checkImmediateTriggersAfterDelete(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const tablePtr);
+
+#if 0
+ void checkDeferredTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTablePtr);
+#endif
+ void checkDetachedTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTablePtr);
+
+ void fireImmediateTriggers(Signal* signal,
+ ArrayList<TupTriggerData>& triggerList,
+ Operationrec* const regOperPtr);
+
+ void fireDeferredTriggers(Signal* signal,
+ ArrayList<TupTriggerData>& triggerList,
+ Operationrec* const regOperPtr);
+
+ void fireDetachedTriggers(Signal* signal,
+ ArrayList<TupTriggerData>& triggerList,
+ Operationrec* const regOperPtr);
+
+ void executeTriggers(Signal* signal,
+ ArrayList<TupTriggerData>& triggerList,
+ Operationrec* const regOperPtr);
+
+ void executeTrigger(Signal* signal,
+ TupTriggerData* const trigPtr,
+ Operationrec* const regOperPtr);
+
+ bool readTriggerInfo(TupTriggerData* const trigPtr,
+ Operationrec* const regOperPtr,
+ Uint32* const keyBuffer,
+ Uint32& noPrimKey,
+ Uint32* const mainBuffer,
+ Uint32& noMainWords,
+ Uint32* const copyBuffer,
+ Uint32& noCopyWords);
+
+ void sendTrigAttrInfo(Signal* signal,
+ Uint32* data,
+ Uint32 dataLen,
+ bool executeDirect,
+ BlockReference receiverReference);
+
+ Uint32 setAttrIds(Bitmask<MAXNROFATTRIBUTESINWORDS>& attributeMask,
+ Uint32 noOfAttributes,
+ Uint32* inBuffer);
+
+ void sendFireTrigOrd(Signal* signal,
+ Operationrec * const regOperPtr,
+ TupTriggerData* const trigPtr,
+ Uint32 noPrimKeySignals,
+ Uint32 noBeforeSignals,
+ Uint32 noAfterSignals);
+
+ bool primaryKey(Tablerec* const, Uint32);
+
+ // these set terrorCode and return non-zero on error
+
+ int executeTuxInsertTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr);
+
+ int executeTuxUpdateTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr);
+
+ int executeTuxDeleteTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr);
+
+ // these crash the node on error
+
+ void executeTuxCommitTriggers(Signal* signal,
+ Operationrec* regOperPtr,
+ Tablerec* const regTabPtr);
+
+ void executeTuxAbortTriggers(Signal* signal,
+ Operationrec* regOperPtr,
+ Tablerec* const regTabPtr);
+
+// *****************************************************************
+// Error Handling routines.
+// *****************************************************************
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ int TUPKEY_abort(Signal* signal, int error_type);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+ void tupkeyErrorLab(Signal* signal);
+
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+// Methods to handle execution of TUP_COMMITREQ + TUP_ABORTREQ.
+//
+// Module Transaction Manager
+//
+// The Transaction Manager module is responsible for the commit
+// and abort of operations started by the Execution Manager.
+//
+// Commit Operation:
+// ----------------
+//
+// Failures in commit processing is not allowed since that would
+// leave the database in an unreliable state. Thus the only way
+// to handle failures in commit processing is to crash the node.
+//
+// TUP_COMMITREQ can only be received in the wait state after a
+// successful TUPKEYREQ which was not a read operation.
+//
+// Commit of Delete:
+// -----------------
+//
+// This will actually perform the deletion of the record unless
+// other operations also are connected to the record. In this case
+// we will set the delete state on the record that becomes the owner
+// of the record.
+//
+// Commit of Update:
+// ----------------
+//
+// We will release the copy record where the original record was kept.
+// Also here we will take special care if more operations are updating
+// the record simultaneously.
+//
+// Commit of Insert:
+// -----------------
+//
+// Will simply reset the state of the operation record.
+//
+// Signal Diagram:
+// ---> TUP_COMMITREQ (from LQH)
+// <---- TUP_COMMITCONF (to LQH)
+//
+//
+// Abort Operation:
+// ----------------
+//
+// Signal Diagram:
+// ---> TUP_ABORTREQ (from LQH)
+// <---- TUP_ABORTCONF (to LQH)
+//
+// Failures in abort processing is not allowed since that would
+// leave the database in an unreliable state. Thus the only way
+// to handle failures in abort processing is to crash the node.
+//
+// Abort messages can arrive at any time. It can arrive even before
+// anything at all have arrived of the operation. It can arrive after
+// receiving a number of ATTRINFO but before TUPKEYREQ has been received.
+// It must arrive after that we sent TUPKEYREF in response to TUPKEYREQ
+// and finally it can arrive after successfully performing the TUPKEYREQ
+// in all cases including the read case.
+//------------------------------------------------------------------
+//------------------------------------------------------------------
+
+#if 0
+ void checkPages(Fragrecord* const regFragPtr);
+#endif
+ void printoutTuplePage(Uint32 fragid, Uint32 pageid, Uint32 printLimit);
+
+ bool checkUpdateOfPrimaryKey(Uint32* updateBuffer, Tablerec* const regTabPtr);
+
+ void setNullBits(Page* const regPage, Tablerec* const regTabPtr, Uint32 pageOffset);
+ bool checkNullAttributes(Operationrec* const, Tablerec* const);
+ bool getPage(PagePtr& pagePtr,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr);
+
+ bool getPageLastCommitted(Operationrec* const regOperPtr,
+ Operationrec* const leaderOpPtr);
+
+ bool getPageThroughSavePoint(Operationrec* const regOperPtr,
+ Operationrec* const leaderOpPtr);
+
+ Uint32 calculateChecksum(Page* const pagePtr, Uint32 tupHeadOffset, Uint32 tupHeadSize);
+ void setChecksum(Page* const pagePtr, Uint32 tupHeadOffset, Uint32 tupHeadSize);
+
+ void commitSimple(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr);
+
+ void commitRecord(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr);
+
+ void setTupleStatesSetOpType(Operationrec* const regOperPtr,
+ Page* const pagePtr,
+ Uint32& opType,
+ OperationrecPtr& firstOpPtr);
+
+ void findBeforeValueOperation(OperationrecPtr& befOpPtr,
+ OperationrecPtr firstOpPtr);
+
+ void calculateChangeMask(Page* const PagePtr,
+ Tablerec* const regTabPtr,
+ Uint32 pageOffset,
+ Bitmask<MAXNROFATTRIBUTESINWORDS>& attributeMask);
+
+ void updateGcpId(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr);
+
+ void abortUpdate(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr);
+ void commitUpdate(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr);
+
+ void setTupleStateOnPreviousOps(Uint32 prevOpIndex);
+ void copyMem(Signal* signal, Uint32 sourceIndex, Uint32 destIndex);
+
+ void freeAllAttrBuffers(Operationrec* const regOperPtr);
+ void freeAttrinbufrec(Uint32 anAttrBufRec);
+ void removeActiveOpList(Operationrec* const regOperPtr);
+
+ void updatePackedList(Signal* signal, Uint16 ahostIndex);
+
+ void setUpDescriptorReferences(Uint32 descriptorReference,
+ Tablerec* const regTabPtr);
+ void setUpKeyArray(Tablerec* const regTabPtr);
+ bool addfragtotab(Tablerec* const regTabPtr, Uint32 fragId, Uint32 fragIndex);
+ void deleteFragTab(Tablerec* const regTabPtr, Uint32 fragId);
+ void releaseTabDescr(Tablerec* const regTabPtr);
+ void getFragmentrec(FragrecordPtr& regFragPtr, Uint32 fragId, Tablerec* const regTabPtr);
+
+ void initialiseRecordsLab(Signal* signal, Uint32 switchData);
+ void initializeAttrbufrec();
+ void initializeCheckpointInfoRec();
+ void initializeDiskBufferSegmentRecord();
+ void initializeFragoperrec();
+ void initializeFragrecord();
+ void initializeHostBuffer();
+ void initializeLocalLogInfo();
+ void initializeOperationrec();
+ void initializePendingFileOpenInfoRecord();
+ void initializeRestartInfoRec();
+ void initializeTablerec();
+ void initializeTabDescr();
+ void initializeUndoPage();
+
+ void initTab(Tablerec* const regTabPtr);
+
+ void startphase3Lab(Signal* signal, Uint32 config1, Uint32 config2);
+
+ void fragrefuseLab(Signal* signal, FragoperrecPtr fragOperPtr);
+ void fragrefuse1Lab(Signal* signal, FragoperrecPtr fragOperPtr);
+ void fragrefuse2Lab(Signal* signal, FragoperrecPtr fragOperPtr, FragrecordPtr regFragPtr);
+ void fragrefuse3Lab(Signal* signal,
+ FragoperrecPtr fragOperPtr,
+ FragrecordPtr regFragPtr,
+ Tablerec* const regTabPtr,
+ Uint32 fragId);
+ void fragrefuse4Lab(Signal* signal,
+ FragoperrecPtr fragOperPtr,
+ FragrecordPtr regFragPtr,
+ Tablerec* const regTabPtr,
+ Uint32 fragId);
+ void addattrrefuseLab(Signal* signal,
+ FragrecordPtr regFragPtr,
+ FragoperrecPtr fragOperPtr,
+ Tablerec* const regTabPtr,
+ Uint32 fragId);
+
+
+ void checkLcpActiveBufferPage(Uint32 minPageNotWrittenInCheckpoint, DiskBufferSegmentInfoPtr dbsiPtr);
+ void lcpWriteListDataPageSegment(Signal* signal,
+ DiskBufferSegmentInfoPtr dbsiPtr,
+ CheckpointInfoPtr ciPtr,
+ bool flushFlag);
+ void lcpFlushLogLab(Signal* signal, CheckpointInfoPtr ciPtr);
+ void lcpClosedDataFileLab(Signal* signal, CheckpointInfoPtr ciPtr);
+ void lcpEndconfLab(Signal* signal);
+ void lcpSaveDataPageLab(Signal* signal, Uint32 ciIndex);
+ void lcpCompletedLab(Signal* signal, Uint32 ciIndex);
+ void lcpFlushRestartInfoLab(Signal* signal, Uint32 ciIndex);
+ void lcpSaveCopyListLab(Signal* signal, CheckpointInfoPtr ciPtr);
+
+ void sendFSREMOVEREQ(Signal* signal, TablerecPtr tabPtr);
+ void releaseFragment(Signal* signal, Uint32 tableId);
+
+ void allocDataBufferSegment(Signal* signal, DiskBufferSegmentInfoPtr& dbsiPtr);
+ void allocRestartUndoBufferSegment(Signal* signal, DiskBufferSegmentInfoPtr& dbsiPtr, LocalLogInfoPtr lliPtr);
+ void freeDiskBufferSegmentRecord(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr);
+ void freeUndoBufferPages(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr);
+
+ void releaseCheckpointInfoRecord(CheckpointInfoPtr ciPtr);
+ void releaseDiskBufferSegmentRecord(DiskBufferSegmentInfoPtr dbsiPtr);
+ void releaseFragoperrec(FragoperrecPtr fragOperPtr);
+ void releaseFragrec(FragrecordPtr regFragPtr);
+ void releasePendingFileOpenInfoRecord(PendingFileOpenInfoPtr pfoPtr);
+ void releaseRestartInfoRecord(RestartInfoRecordPtr riPtr);
+
+ void seizeDiskBufferSegmentRecord(DiskBufferSegmentInfoPtr& dbsiPtr);
+ void seizeCheckpointInfoRecord(CheckpointInfoPtr& ciPtr);
+ void seizeFragoperrec(FragoperrecPtr& fragOperPtr);
+ void seizeFragrecord(FragrecordPtr& regFragPtr);
+ void seizeOpRec(OperationrecPtr& regOperPtr);
+ void seizePendingFileOpenInfoRecord(PendingFileOpenInfoPtr& pfoiPtr);
+ void seizeRestartInfoRecord(RestartInfoRecordPtr& riPtr);
+
+ // Initialisation
+ void initData();
+ void initRecords();
+
+ void rfrClosedDataFileLab(Signal* signal, Uint32 restartIndex);
+ void rfrCompletedLab(Signal* signal, RestartInfoRecordPtr riPtr);
+ void rfrInitRestartInfoLab(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr);
+ void rfrLoadDataPagesLab(Signal* signal, RestartInfoRecordPtr riPtr, DiskBufferSegmentInfoPtr dbsiPtr);
+ void rfrReadFirstUndoSegment(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr, LocalLogInfoPtr lliPtr);
+ void rfrReadNextDataSegment(Signal* signal, RestartInfoRecordPtr riPtr, DiskBufferSegmentInfoPtr dbsiPtr);
+ void rfrReadNextUndoSegment(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr, LocalLogInfoPtr lliPtr);
+ void rfrReadRestartInfoLab(Signal* signal, RestartInfoRecordPtr riPtr);
+ void rfrReadSecondUndoLogLab(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr);
+
+ void startExecUndoLogLab(Signal* signal, Uint32 lliIndex);
+ void readExecUndoLogLab(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr, LocalLogInfoPtr lliPtr);
+ void closeExecUndoLogLab(Signal* signal, LocalLogInfoPtr lliPtr);
+ void endExecUndoLogLab(Signal* signal, Uint32 lliIndex);
+
+ struct XlcStruct {
+ Uint32 PageId;
+ Uint32 PageIndex;
+ Uint32 LogRecordType;
+ Uint32 FragId;
+ FragrecordPtr FragPtr;
+ LocalLogInfoPtr LliPtr;
+ DiskBufferSegmentInfoPtr DbsiPtr;
+ UndoPagePtr UPPtr;
+ TablerecPtr TabPtr;
+ };
+
+ void xlcGetNextRecordLab(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr, LocalLogInfoPtr lliPtr);
+ void xlcRestartCompletedLab(Signal* signal);
+
+ void xlcCopyData(XlcStruct& xlcStruct, Uint32 pageOffset, Uint32 noOfWords, PagePtr pagePtr);
+ void xlcGetLogHeader(XlcStruct& xlcStruct);
+ Uint32 xlcGetLogWord(XlcStruct& xlcStruct);
+
+ void xlcAbortInsert(Signal* signal, XlcStruct& xlcStruct);
+ void xlcAbortUpdate(Signal* signal, XlcStruct& xlcStruct);
+ void xlcDeleteTh(XlcStruct& xlcStruct);
+ void xlcIndicateNoOpActive(XlcStruct& xlcStruct);
+ void xlcInsertTh(XlcStruct& xlcStruct);
+ void xlcTableDescriptor(XlcStruct& xlcStruct);
+ void xlcUndoLogPageHeader(XlcStruct& xlcStruct);
+ void xlcUpdateTh(XlcStruct& xlcStruct);
+ void xlcUpdateGCI(XlcStruct& xlcStruct);
+
+
+ void cprAddData(Signal* signal,
+ Fragrecord* const regFragPtr,
+ Uint32 pageIndex,
+ Uint32 noOfWords,
+ Uint32 startOffset);
+ void cprAddGCIUpdate(Signal* signal,
+ Uint32 prevGCI,
+ Fragrecord* const regFragPtr);
+ void cprAddLogHeader(Signal* signal,
+ LocalLogInfo* const lliPtr,
+ Uint32 recordType,
+ Uint32 tableId,
+ Uint32 fragId);
+ void cprAddUndoLogPageHeader(Signal* signal,
+ Page* const regPagePtr,
+ Fragrecord* const regFragPtr);
+ void cprAddUndoLogRecord(Signal* signal,
+ Uint32 recordType,
+ Uint32 pageId,
+ Uint32 pageIndex,
+ Uint32 tableId,
+ Uint32 fragId,
+ Uint32 localLogIndex);
+ void cprAddAbortUpdate(Signal* signal,
+ LocalLogInfo* const lliPtr,
+ Operationrec* const regOperPtr);
+ void cprAddUndoLogWord(Signal* signal,
+ LocalLogInfo* const lliPtr,
+ Uint32 undoWord);
+ bool isUndoLoggingNeeded(Fragrecord* const regFragPtr, Uint32 pageId);
+ bool isUndoLoggingActive(Fragrecord* const regFragPtr);
+ bool isUndoLoggingBlocked(Fragrecord* const regFragPtr);
+ bool isPageUndoLogged(Fragrecord* const regFragPtr, Uint32 pageId);
+
+ void seizeUndoBufferSegment(Signal* signal, UndoPagePtr& regUndoPagePtr);
+ void lcpWriteUndoSegment(Signal* signal, LocalLogInfo* const lliPtr, bool flushFlag);
+
+
+ void deleteScanProcedure(Signal* signal, Operationrec* regOperPtr);
+ void copyProcedure(Signal* signal,
+ TablerecPtr regTabPtr,
+ Operationrec* regOperPtr);
+ void scanProcedure(Signal* signal,
+ Operationrec* regOperPtr,
+ Uint32 lenAttrInfo);
+ void storedSeizeAttrinbufrecErrorLab(Signal* signal,
+ Operationrec* regOperPtr);
+ bool storedProcedureAttrInfo(Signal* signal,
+ Operationrec* regOperPtr,
+ Uint32 length,
+ Uint32 firstWord,
+ bool copyProc);
+
+//-----------------------------------------------------------------------------
+// Table Descriptor Memory Manager
+//-----------------------------------------------------------------------------
+
+// Public methods
+ Uint32 allocTabDescr(Uint32 noOfAttributes, Uint32 noOfKeyAttr, Uint32 noOfAttributeGroups);
+ void freeTabDescr(Uint32 retRef, Uint32 retNo);
+ Uint32 getTabDescrWord(Uint32 index);
+ void setTabDescrWord(Uint32 index, Uint32 word);
+
+// Private methods
+ Uint32 sizeOfReadFunction();
+ void removeTdArea(Uint32 tabDesRef, Uint32 list);
+ void insertTdArea(Uint32 sizeOfChunk, Uint32 tabDesRef, Uint32 list);
+ Uint32 itdaMergeTabDescr(Uint32 retRef, Uint32 retNo);
+
+//------------------------------------------------------------------------------------------------------
+// Page Memory Manager
+//------------------------------------------------------------------------------------------------------
+
+// Public methods
+ void allocConsPages(Uint32 noOfPagesToAllocate,
+ Uint32& noOfPagesAllocated,
+ Uint32& allocPageRef);
+ void returnCommonArea(Uint32 retPageRef, Uint32 retNo);
+ void initializePage();
+
+// Private methods
+ void removeCommonArea(Uint32 remPageRef, Uint32 list);
+ void insertCommonArea(Uint32 insPageRef, Uint32 list);
+ void findFreeLeftNeighbours(Uint32& allocPageRef, Uint32& noPagesAllocated, Uint32 noPagesToAllocate);
+ void findFreeRightNeighbours(Uint32& allocPageRef, Uint32& noPagesAllocated, Uint32 noPagesToAllocate);
+ Uint32 nextHigherTwoLog(Uint32 input);
+
+// Private data
+ Uint32 cfreepageList[16];
+
+//------------------------------------------------------------------------------------------------------
+// Page Mapper, convert logical page id's to physical page id's
+// The page mapper also handles the pages allocated to the fragment.
+//------------------------------------------------------------------------------------------------------
+//
+// Public methods
+ Uint32 getRealpid(Fragrecord* const regFragPtr, Uint32 logicalPageId);
+ Uint32 getNoOfPages(Fragrecord* const regFragPtr);
+ void initPageRangeSize(Uint32 size);
+ bool insertPageRangeTab(Fragrecord* const regFragPtr,
+ Uint32 startPageId,
+ Uint32 noPages);
+ void releaseFragPages(Fragrecord* const regFragPtr);
+ void initFragRange(Fragrecord* const regFragPtr);
+ void initializePageRange();
+ Uint32 getEmptyPage(Fragrecord* const regFragPtr);
+ Uint32 allocFragPages(Fragrecord* const regFragPtr, Uint32 noOfPagesAllocated);
+
+// Private methods
+ Uint32 leafPageRangeFull(Fragrecord* const regFragPtr, PageRangePtr currPageRangePtr);
+ void releasePagerange(PageRangePtr regPRPtr);
+ void seizePagerange(PageRangePtr& regPageRangePtr);
+ void errorHandler(Uint32 errorCode);
+ void allocMoreFragPages(Fragrecord* const regFragPtr);
+
+// Private data
+ Uint32 cfirstfreerange;
+ PageRange *pageRange;
+ Uint32 c_noOfFreePageRanges;
+ Uint32 cnoOfPageRangeRec;
+
+//------------------------------------------------------------------------------------------------------
+// Fixed Allocator
+// Allocates and deallocates tuples of fixed size on a fragment.
+//------------------------------------------------------------------------------------------------------
+//
+// Public methods
+ bool allocTh(Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Uint32 pageType,
+ Signal* signal,
+ Uint32& pageOffset,
+ PagePtr& pagePtr);
+
+ void freeThSr(Tablerec* const regTabPtr,
+ Page* const regPagePtr,
+ Uint32 freePageOffset);
+
+ void freeTh(Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Signal* signal,
+ Page* const regPagePtr,
+ Uint32 freePageOffset);
+
+ void getThAtPageSr(Page* const regPagePtr,
+ Uint32& pageOffset);
+
+// Private methods
+ void convertThPage(Uint32 Tupheadsize,
+ Page* const regPagePtr);
+
+ void getThAtPage(Fragrecord* const regFragPtr,
+ Page* const regPagePtr,
+ Signal* signal,
+ Uint32& pageOffset);
+
+ void getEmptyPageThCopy(Fragrecord* const regFragPtr,
+ Signal* signal,
+ Page* const regPagePtr);
+
+ void getEmptyPageTh(Fragrecord* const regFragPtr,
+ Signal* signal,
+ Page* const regPagePtr);
+
+//------------------------------------------------------------------------------------------------------
+// Temporary variables used for storing commonly used variables in certain modules
+//------------------------------------------------------------------------------------------------------
+
+ FragrecordPtr fragptr;
+ OperationrecPtr operPtr;
+ TablerecPtr tabptr;
+
+// readAttributes and updateAttributes module
+ Uint32 tCheckOffset;
+ Uint32 tMaxRead;
+ Uint32 tOutBufIndex;
+ Uint32* tTupleHeader;
+
+// updateAttributes module
+ Uint32 tInBufIndex;
+ Uint32 tInBufLen;
+
+ Uint32 terrorCode;
+
+//------------------------------------------------------------------------------------------------------
+// Common stored variables. Variables that have a valid value always.
+//------------------------------------------------------------------------------------------------------
+ Uint32 cnoOfLcpRec;
+ Uint32 cnoOfParallellUndoFiles;
+ Uint32 cnoOfUndoPage;
+
+ Attrbufrec *attrbufrec;
+ Uint32 cfirstfreeAttrbufrec;
+ Uint32 cnoOfAttrbufrec;
+ Uint32 cnoFreeAttrbufrec;
+
+ CheckpointInfo *checkpointInfo;
+ Uint32 cfirstfreeLcp;
+
+ DiskBufferSegmentInfo *diskBufferSegmentInfo;
+ Uint32 cfirstfreePdx;
+ Uint32 cnoOfConcurrentWriteOp;
+
+ Fragoperrec *fragoperrec;
+ Uint32 cfirstfreeFragopr;
+ Uint32 cnoOfFragoprec;
+
+ Fragrecord *fragrecord;
+ Uint32 cfirstfreefrag;
+ Uint32 cnoOfFragrec;
+
+ HostBuffer *hostBuffer;
+
+ LocalLogInfo *localLogInfo;
+ Uint32 cnoOfLocalLogInfo;
+
+ Uint32 cfirstfreeOprec;
+ Operationrec *operationrec;
+ Uint32 cnoOfOprec;
+
+ Page *page;
+ Uint32 cnoOfPage;
+ Uint32 cnoOfAllocatedPages;
+
+ PendingFileOpenInfo *pendingFileOpenInfo;
+ Uint32 cfirstfreePfo;
+ Uint32 cnoOfConcurrentOpenOp;
+
+ RestartInfoRecord *restartInfoRecord;
+ Uint32 cfirstfreeSri;
+ Uint32 cnoOfRestartInfoRec;
+
+ Tablerec *tablerec;
+ Uint32 cnoOfTablerec;
+
+ TableDescriptor *tableDescriptor;
+ Uint32 cnoOfTabDescrRec;
+
+ UndoPage *undoPage;
+ Uint32 cfirstfreeUndoSeg;
+ Int32 cnoFreeUndoSeg;
+
+
+
+ Uint32 cnoOfDataPagesToDiskWithoutSynch;
+
+ Uint32 cdata[32];
+ Uint32 cdataPages[16];
+ Uint32 cpackedListIndex;
+ Uint32 cpackedList[MAX_NODES];
+ Uint32 cfreeTdList[16];
+ Uint32 clastBitMask;
+ Uint32 clblPageCounter;
+ Uint32 clblPagesPerTick;
+ Uint32 clblPagesPerTickAfterSr;
+ BlockReference clqhBlockref;
+ Uint32 clqhUserpointer;
+ Uint32 cminusOne;
+ BlockReference cndbcntrRef;
+ Uint32 cundoFileVersion;
+ BlockReference cownref;
+ Uint32 cownNodeId;
+ Uint32 czero;
+
+ // A little bit bigger to cover overwrites in copy algorithms (16384 real size).
+#define ZATTR_BUFFER_SIZE 16384
+ Uint32 clogMemBuffer[ZATTR_BUFFER_SIZE + 16];
+ Uint32 coutBuffer[ZATTR_BUFFER_SIZE + 16];
+ Uint32 cinBuffer[ZATTR_BUFFER_SIZE + 16];
+ Uint32 totNoOfPagesAllocated;
+
+ // Trigger variables
+ Uint32 c_maxTriggersPerTable;
+
+ // Counters for num UNDO log records executed
+ Uint32 cSrUndoRecords[9];
+
+ Uint32 c_errorInsert4000TableId;
+
+ void initGlobalTemporaryVars();
+ void reportMemoryUsage(Signal* signal, int incDec);
+#ifdef VM_TRACE
+ struct Th {
+ Uint32 data[1];
+ };
+ friend class NdbOut& operator<<(NdbOut&, const Operationrec&);
+ friend class NdbOut& operator<<(NdbOut&, const Th&);
+#endif
+};
+
+inline
+bool Dbtup::isUndoLoggingNeeded(Fragrecord* const regFragPtr,
+ Uint32 pageId)
+{
+ if ((regFragPtr->checkpointVersion != RNIL) &&
+ (pageId >= regFragPtr->minPageNotWrittenInCheckpoint) &&
+ (pageId < regFragPtr->maxPageWrittenInCheckpoint)) {
+ return true;
+ }//if
+ return false;
+}//Dbtup::isUndoLoggingNeeded()
+
+inline
+bool Dbtup::isUndoLoggingActive(Fragrecord* const regFragPtr)
+{
+ if (regFragPtr->checkpointVersion != RNIL) {
+ return true;
+ }//if
+ return false;
+}//Dbtup::isUndoLoggingNeeded()
+
+inline
+bool Dbtup::isUndoLoggingBlocked(Fragrecord* const regFragPtr)
+{
+ if ((regFragPtr->checkpointVersion != RNIL) &&
+ (cnoFreeUndoSeg < ZMIN_PAGE_LIMIT_TUPKEYREQ)) {
+ return true;
+ }//if
+ return false;
+}//Dbtup::isUndoLoggingNeeded()
+
+inline
+bool Dbtup::isPageUndoLogged(Fragrecord* const regFragPtr,
+ Uint32 pageId)
+{
+ if ((pageId >= regFragPtr->minPageNotWrittenInCheckpoint) &&
+ (pageId < regFragPtr->maxPageWrittenInCheckpoint)) {
+ return true;
+ }//if
+ return false;
+}//Dbtup::isUndoLoggingNeeded()
+
+#endif
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupAbort.cpp b/ndb/src/kernel/blocks/dbtup/DbtupAbort.cpp
new file mode 100644
index 00000000000..1ffc5f06754
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupAbort.cpp
@@ -0,0 +1,473 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+
+#define ljam() { jamLine(9000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(9000 + __LINE__); }
+
+void Dbtup::freeAllAttrBuffers(Operationrec* const regOperPtr)
+{
+ if (regOperPtr->storedProcedureId == ZNIL) {
+ ljam();
+ freeAttrinbufrec(regOperPtr->firstAttrinbufrec);
+ } else {
+ StoredProcPtr storedPtr;
+ c_storedProcPool.getPtr(storedPtr, (Uint32)regOperPtr->storedProcedureId);
+ ndbrequire(storedPtr.p->storedCode == ZSCAN_PROCEDURE);
+ ljam();
+ storedPtr.p->storedCounter--;
+ regOperPtr->storedProcedureId = ZNIL;
+ }//if
+ regOperPtr->firstAttrinbufrec = RNIL;
+ regOperPtr->lastAttrinbufrec = RNIL;
+}//Dbtup::freeAllAttrBuffers()
+
+void Dbtup::freeAttrinbufrec(Uint32 anAttrBuf)
+{
+ Uint32 Ttemp;
+ AttrbufrecPtr localAttrBufPtr;
+ Uint32 RnoFree = cnoFreeAttrbufrec;
+ localAttrBufPtr.i = anAttrBuf;
+ while (localAttrBufPtr.i != RNIL) {
+ ljam();
+ ptrCheckGuard(localAttrBufPtr, cnoOfAttrbufrec, attrbufrec);
+ Ttemp = localAttrBufPtr.p->attrbuf[ZBUF_NEXT];
+ localAttrBufPtr.p->attrbuf[ZBUF_NEXT] = cfirstfreeAttrbufrec;
+ cfirstfreeAttrbufrec = localAttrBufPtr.i;
+ localAttrBufPtr.i = Ttemp;
+ RnoFree++;
+ }//if
+ cnoFreeAttrbufrec = RnoFree;
+}//Dbtup::freeAttrinbufrec()
+
+/* ----------------------------------------------------------------- */
+/* ----------- ABORT THIS PART OF THE TRANSACTION ------------------ */
+/* ----------------------------------------------------------------- */
+void Dbtup::execTUP_ABORTREQ(Signal* signal)
+{
+ OperationrecPtr regOperPtr;
+ FragrecordPtr regFragPtr;
+ TablerecPtr regTabPtr;
+
+ ljamEntry();
+ regOperPtr.i = signal->theData[0];
+ ptrCheckGuard(regOperPtr, cnoOfOprec, operationrec);
+ ndbrequire((regOperPtr.p->transstate == STARTED) ||
+ (regOperPtr.p->transstate == TOO_MUCH_AI) ||
+ (regOperPtr.p->transstate == ERROR_WAIT_TUPKEYREQ) ||
+ (regOperPtr.p->transstate == IDLE));
+ if (regOperPtr.p->optype == ZREAD) {
+ ljam();
+ freeAllAttrBuffers(regOperPtr.p);
+ initOpConnection(regOperPtr.p);
+ return;
+ }//if
+
+ regTabPtr.i = regOperPtr.p->tableRef;
+ ptrCheckGuard(regTabPtr, cnoOfTablerec, tablerec);
+
+ regFragPtr.i = regOperPtr.p->fragmentPtr;
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+
+ // XXX should be integrated into the code that comes after
+ if (!regTabPtr.p->tuxCustomTriggers.isEmpty() &&
+ regOperPtr.p->tupleState == NO_OTHER_OP) {
+ ljam();
+ executeTuxAbortTriggers(signal,
+ regOperPtr.p,
+ regTabPtr.p);
+ OperationrecPtr loopOpPtr;
+ loopOpPtr.i = regOperPtr.p->prevActiveOp;
+ while (loopOpPtr.i != RNIL) {
+ ljam();
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ if (loopOpPtr.p->tupleState != ALREADY_ABORTED) {
+ ljam();
+ executeTuxAbortTriggers(signal,
+ loopOpPtr.p,
+ regTabPtr.p);
+ }
+ loopOpPtr.i = loopOpPtr.p->prevActiveOp;
+ }
+ }
+
+ Uint32 prevActiveOp = regOperPtr.p->prevActiveOp;
+ removeActiveOpList(regOperPtr.p);
+ if (regOperPtr.p->tupleState == NO_OTHER_OP) {
+ if (prevActiveOp == RNIL) {
+ ljam();
+ abortUpdate(signal, regOperPtr.p, regFragPtr.p, regTabPtr.p);
+ } else { //prevActiveOp != RNIL
+ setTupleStateOnPreviousOps(prevActiveOp);
+ if (regOperPtr.p->optype == ZDELETE) {
+ ljam();
+ OperationrecPtr prevOpPtr;
+ prevOpPtr.i = prevActiveOp;
+ ptrCheckGuard(prevOpPtr, cnoOfOprec, operationrec);
+ ndbrequire(prevOpPtr.p->realPageIdC != RNIL);
+ ndbrequire(prevOpPtr.p->optype == ZINSERT);
+ abortUpdate(signal, prevOpPtr.p, regFragPtr.p, regTabPtr.p);
+ } else {
+ jam();
+ abortUpdate(signal, regOperPtr.p, regFragPtr.p, regTabPtr.p);
+ }//if
+ }//if
+ } else {
+ ndbrequire(regOperPtr.p->tupleState == ALREADY_ABORTED);
+ commitUpdate(signal, regOperPtr.p, regFragPtr.p, regTabPtr.p);
+ }//if
+ initOpConnection(regOperPtr.p);
+}//execTUP_ABORTREQ()
+
+void Dbtup::setTupleStateOnPreviousOps(Uint32 prevOpIndex)
+{
+ OperationrecPtr loopOpPtr;
+ loopOpPtr.i = prevOpIndex;
+ do {
+ ljam();
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ loopOpPtr.p->tupleState = ALREADY_ABORTED;
+ loopOpPtr.i = loopOpPtr.p->prevActiveOp;
+ } while (loopOpPtr.i != RNIL);
+}//Dbtup::setTupleStateOnPreviousOps()
+
+/* ---------------------------------------------------------------- */
+/* ------------ PERFORM AN ABORT OF AN UPDATE OPERATION ----------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::abortUpdate(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr)
+{
+ /* RESTORE THE ORIGINAL DATA */
+ /* THE OPER_PTR ALREADY CONTAINS BOTH THE PAGE AND THE COPY PAGE */
+ if (regOperPtr->realPageIdC != RNIL) {
+ ljam();
+ /***********************/
+ /* CHECKPOINT SPECIFIC */
+ /***********************/
+ if (isUndoLoggingNeeded(regFragPtr, regOperPtr->fragPageIdC)) {
+ if (regOperPtr->undoLogged) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* THE UPDATE WAS MADE AFTER THE LOCAL CHECKPOINT STARTED. */
+/* THUS THE ORIGINAL TUPLE WILL BE RESTORED BY A LOG RECORD */
+/* CREATED WHEN UPDATING. THUS IT IS ENOUGH TO LOG THE UNDO */
+/* OF THE COPY RELEASE == INSERT THE COPY TUPLE HEADER WITH */
+/* NO DATA. */
+/* ---------------------------------------------------------------- */
+ cprAddUndoLogRecord(signal,
+ ZLCPR_TYPE_INSERT_TH_NO_DATA,
+ regOperPtr->fragPageIdC,
+ regOperPtr->pageIndexC,
+ regOperPtr->tableRef,
+ regOperPtr->fragId,
+ regFragPtr->checkpointVersion);
+ } else {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* THE UPDATE WAS MADE BEFORE THE LOCAL CHECKPOINT STARTED. */
+/* THE TUPLE WILL THUS BE RESTORED BY COPYING FROM THE COPY. */
+/* THUS WE DO NOT NEED TO RESTORE THE DATA IN THE ORIGINAL. */
+/* WE DO HOWEVER NEED TO ENSURE THAT THE COPY CONTAINS THE */
+/* CORRECT DATA. */
+/* ---------------------------------------------------------------- */
+ cprAddUndoLogRecord(signal,
+ ZLCPR_TYPE_INSERT_TH,
+ regOperPtr->fragPageIdC,
+ regOperPtr->pageIndexC,
+ regOperPtr->tableRef,
+ regOperPtr->fragId,
+ regFragPtr->checkpointVersion);
+ cprAddData(signal,
+ regFragPtr,
+ regOperPtr->realPageIdC,
+ regTabPtr->tupheadsize,
+ regOperPtr->pageOffsetC);
+ }//if
+ }//if
+ Uint32 rpid = regOperPtr->realPageId;
+ Uint32 rpid_copy = regOperPtr->realPageIdC;
+ Uint32 offset = regOperPtr->pageOffset;
+ Uint32 offset_copy = regOperPtr->pageOffsetC;
+ Uint32 tuple_size = regTabPtr->tupheadsize;
+ Uint32 end = offset + tuple_size;
+ Uint32 end_copy = offset_copy + tuple_size;
+ ndbrequire(rpid < cnoOfPage &&
+ rpid_copy < cnoOfPage &&
+ end <= ZWORDS_ON_PAGE &&
+ end_copy <= ZWORDS_ON_PAGE);
+ void* Tdestination = (void*)&page[rpid].pageWord[offset + 1];
+ const void* Tsource = (void*)&page[rpid_copy].pageWord[offset_copy + 1];
+ MEMCOPY_NO_WORDS(Tdestination, Tsource, (tuple_size - 1));
+ {
+ PagePtr pagePtr;
+
+ pagePtr.i = rpid_copy;
+ ptrAss(pagePtr, page);
+ freeTh(regFragPtr,
+ regTabPtr,
+ signal,
+ pagePtr.p,
+ offset_copy);
+ }
+ regOperPtr->realPageIdC = RNIL;
+ regOperPtr->fragPageIdC = RNIL;
+ regOperPtr->pageOffsetC = ZNIL;
+ regOperPtr->pageIndexC = ZNIL;
+ }//if
+}//Dbtup::abortUpdate()
+
+/* **************************************************************** */
+/* ********************** TRANSACTION ERROR MODULE **************** */
+/* **************************************************************** */
+int Dbtup::TUPKEY_abort(Signal* signal, int error_type)
+{
+ switch(error_type) {
+ case 0:
+ ndbrequire(false);
+ break;
+// Not used currently
+
+ case 1:
+//tmupdate_alloc_error:
+ ljam();
+ break;
+
+ case 2:
+ ndbrequire(false);
+ break;
+// Not used currently
+
+ break;
+
+ case 3:
+//tmupdate_alloc_error:
+ ljam();
+ break;
+
+ case 4:
+//Trying to read non-existing attribute identity
+ ljam();
+ terrorCode = ZATTRIBUTE_ID_ERROR;
+ break;
+
+ case 6:
+ ljam();
+ terrorCode = ZTRY_TO_READ_TOO_MUCH_ERROR;
+ break;
+
+ case 7:
+ ljam();
+ terrorCode = ZAI_INCONSISTENCY_ERROR;
+ break;
+
+ case 8:
+ ljam();
+ terrorCode = ZATTR_INTERPRETER_ERROR;
+ break;
+
+ case 9:
+ ljam();
+//Trying to read non-existing attribute identity
+ ljam();
+ terrorCode = ZATTRIBUTE_ID_ERROR;
+ break;
+
+ case 11:
+ ljam();
+ terrorCode = ZATTR_INTERPRETER_ERROR;
+ break;
+
+ case 12:
+ ljam();
+ ndbrequire(false);
+ break;
+
+ case 13:
+ ljam();
+ ndbrequire(false);
+ break;
+
+ case 14:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 15:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 16:
+ ljam();
+ terrorCode = ZTRY_TO_UPDATE_ERROR;
+ break;
+
+ case 17:
+ ljam();
+ terrorCode = ZNO_ILLEGAL_NULL_ATTR;
+ break;
+
+ case 18:
+ ljam();
+ terrorCode = ZNOT_NULL_ATTR;
+ break;
+
+ case 19:
+ ljam();
+ terrorCode = ZTRY_TO_UPDATE_ERROR;
+ break;
+
+ case 20:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 21:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 22:
+ ljam();
+ terrorCode = ZTOTAL_LEN_ERROR;
+ break;
+
+ case 23:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 24:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 25:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 26:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 27:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 28:
+ ljam();
+ terrorCode = ZREGISTER_INIT_ERROR;
+ break;
+
+ case 29:
+ ljam();
+ break;
+
+ case 30:
+ ljam();
+ terrorCode = ZCALL_ERROR;
+ break;
+
+ case 31:
+ ljam();
+ terrorCode = ZSTACK_OVERFLOW_ERROR;
+ break;
+
+ case 32:
+ ljam();
+ terrorCode = ZSTACK_UNDERFLOW_ERROR;
+ break;
+
+ case 33:
+ ljam();
+ terrorCode = ZNO_INSTRUCTION_ERROR;
+ break;
+
+ case 34:
+ ljam();
+ terrorCode = ZOUTSIDE_OF_PROGRAM_ERROR;
+ break;
+
+ case 35:
+ ljam();
+ terrorCode = ZTOO_MANY_INSTRUCTIONS_ERROR;
+ break;
+
+ case 36:
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ break;
+
+ case 37:
+ ljam();
+ terrorCode = ZTEMPORARY_RESOURCE_FAILURE;
+ break;
+
+ case 38:
+ ljam();
+ terrorCode = ZTEMPORARY_RESOURCE_FAILURE;
+ break;
+
+ case 39:
+ ljam();
+ if (operPtr.p->transstate == TOO_MUCH_AI) {
+ ljam();
+ terrorCode = ZTOO_MUCH_ATTRINFO_ERROR;
+ } else if (operPtr.p->transstate == ERROR_WAIT_TUPKEYREQ) {
+ ljam();
+ terrorCode = ZSEIZE_ATTRINBUFREC_ERROR;
+ } else {
+ ndbrequire(false);
+ }//if
+ break;
+
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ tupkeyErrorLab(signal);
+ return -1;
+}//Dbtup::TUPKEY_abort()
+
+void Dbtup::tupkeyErrorLab(Signal* signal)
+{
+ Operationrec * const regOperPtr = operPtr.p;
+
+ freeAllAttrBuffers(regOperPtr);
+ abortUpdate(signal, regOperPtr, fragptr.p, tabptr.p);
+ removeActiveOpList(regOperPtr);
+ initOpConnection(regOperPtr);
+ regOperPtr->transstate = IDLE;
+ regOperPtr->tupleState = NO_OTHER_OP;
+ TupKeyRef * const tupKeyRef = (TupKeyRef *)signal->getDataPtrSend();
+
+ tupKeyRef->userRef = regOperPtr->userpointer;
+ tupKeyRef->errorCode = terrorCode;
+ sendSignal(regOperPtr->userblockref, GSN_TUPKEYREF, signal,
+ TupKeyRef::SignalLength, JBB);
+ return;
+}//Dbtup::tupkeyErrorLab()
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupBuffer.cpp b/ndb/src/kernel/blocks/dbtup/DbtupBuffer.cpp
new file mode 100644
index 00000000000..90c6dbc6802
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupBuffer.cpp
@@ -0,0 +1,329 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <signaldata/TransIdAI.hpp>
+
+#define ljam() { jamLine(2000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(2000 + __LINE__); }
+
+void Dbtup::execSEND_PACKED(Signal* signal)
+{
+ Uint16 hostId;
+ Uint32 i;
+ Uint32 TpackedListIndex = cpackedListIndex;
+ ljamEntry();
+ for (i = 0; i < TpackedListIndex; i++) {
+ ljam();
+ hostId = cpackedList[i];
+ ndbrequire((hostId - 1) < (MAX_NODES - 1)); // Also check not zero
+ Uint32 TpacketTA = hostBuffer[hostId].noOfPacketsTA;
+ Uint32 TpacketRC = hostBuffer[hostId].noOfPacketsRC;
+ if (TpacketTA != 0) {
+ ljam();
+ BlockReference TBref = numberToRef(API_PACKED, hostId);
+ Uint32 TpacketLen = hostBuffer[hostId].packetLenTA;
+ MEMCOPY_NO_WORDS(&signal->theData[0],
+ &hostBuffer[hostId].packetBufferTA[0],
+ TpacketLen);
+ sendSignal(TBref, GSN_TRANSID_AI, signal, TpacketLen, JBB);
+ hostBuffer[hostId].noOfPacketsTA = 0;
+ hostBuffer[hostId].packetLenTA = 0;
+ }//if
+ if (TpacketRC != 0) {
+ ljam();
+ BlockReference TBref = numberToRef(API_PACKED, hostId);
+ Uint32 TpacketLen = hostBuffer[hostId].packetLenRC;
+ MEMCOPY_NO_WORDS(&signal->theData[0],
+ &hostBuffer[hostId].packetBufferRC[0],
+ TpacketLen);
+ sendSignal(TBref, GSN_READCONF, signal, TpacketLen, JBB);
+ hostBuffer[hostId].noOfPacketsRC = 0;
+ hostBuffer[hostId].packetLenRC = 0;
+ }//if
+ hostBuffer[hostId].inPackedList = false;
+ }//for
+ cpackedListIndex = 0;
+}//Dbtup::execSEND_PACKED()
+
+void Dbtup::bufferREADCONF(Signal* signal, BlockReference aRef,
+ Uint32* buffer, Uint32 Tlen)
+{
+ Uint32 hostId = refToNode(aRef);
+ Uint32 Theader = ((refToBlock(aRef) << 16) + (Tlen-3));
+
+ ndbrequire(hostId < MAX_NODES);
+ Uint32 TpacketLen = hostBuffer[hostId].packetLenRC;
+ Uint32 TnoOfPackets = hostBuffer[hostId].noOfPacketsRC;
+ Uint32 sig0 = signal->theData[0];
+ Uint32 sig1 = signal->theData[1];
+ Uint32 sig2 = signal->theData[2];
+ Uint32 sig3 = signal->theData[3];
+
+ BlockReference TBref = numberToRef(API_PACKED, hostId);
+
+ if ((Tlen + TpacketLen + 1) <= 25) {
+// ----------------------------------------------------------------
+// There is still space in the buffer. We will copy it into the
+// buffer.
+// ----------------------------------------------------------------
+ ljam();
+ updatePackedList(signal, hostId);
+ } else if (TnoOfPackets == 1) {
+// ----------------------------------------------------------------
+// The buffer is full and there was only one packet buffered. We
+// will send this as a normal signal.
+// ----------------------------------------------------------------
+ Uint32 TnewRef = numberToRef((hostBuffer[hostId].packetBufferRC[0] >> 16),
+ hostId);
+ MEMCOPY_NO_WORDS(&signal->theData[0],
+ &hostBuffer[hostId].packetBufferRC[1],
+ TpacketLen - 1);
+ sendSignal(TnewRef, GSN_READCONF, signal, (TpacketLen - 1), JBB);
+ TpacketLen = 0;
+ TnoOfPackets = 0;
+ } else {
+// ----------------------------------------------------------------
+// The buffer is full but at least two packets. Send those in
+// packed form.
+// ----------------------------------------------------------------
+ MEMCOPY_NO_WORDS(&signal->theData[0],
+ &hostBuffer[hostId].packetBufferRC[0],
+ TpacketLen);
+ sendSignal(TBref, GSN_READCONF, signal, TpacketLen, JBB);
+ TpacketLen = 0;
+ TnoOfPackets = 0;
+ }//if
+// ----------------------------------------------------------------
+// Copy the signal into the buffer
+// ----------------------------------------------------------------
+ hostBuffer[hostId].packetBufferRC[TpacketLen + 0] = Theader;
+ hostBuffer[hostId].packetBufferRC[TpacketLen + 1] = sig0;
+ hostBuffer[hostId].packetBufferRC[TpacketLen + 2] = sig1;
+ hostBuffer[hostId].packetBufferRC[TpacketLen + 3] = sig2;
+ hostBuffer[hostId].packetBufferRC[TpacketLen + 4] = sig3;
+ hostBuffer[hostId].noOfPacketsRC = TnoOfPackets + 1;
+ hostBuffer[hostId].packetLenRC = Tlen + TpacketLen + 1;
+ MEMCOPY_NO_WORDS(&hostBuffer[hostId].packetBufferRC[TpacketLen + 5],
+ buffer,
+ Tlen - 4);
+}//Dbtup::bufferREADCONF()
+
+void Dbtup::bufferTRANSID_AI(Signal* signal, BlockReference aRef,
+ Uint32* buffer, Uint32 Tlen)
+{
+ Uint32 hostId = refToNode(aRef);
+ Uint32 Theader = ((refToBlock(aRef) << 16)+(Tlen-3));
+
+ ndbrequire(hostId < MAX_NODES);
+ Uint32 TpacketLen = hostBuffer[hostId].packetLenTA;
+ Uint32 TnoOfPackets = hostBuffer[hostId].noOfPacketsTA;
+ Uint32 sig0 = signal->theData[0];
+ Uint32 sig1 = signal->theData[1];
+ Uint32 sig2 = signal->theData[2];
+
+ BlockReference TBref = numberToRef(API_PACKED, hostId);
+
+ if ((Tlen + TpacketLen + 1) <= 25) {
+// ----------------------------------------------------------------
+// There is still space in the buffer. We will copy it into the
+// buffer.
+// ----------------------------------------------------------------
+ ljam();
+ updatePackedList(signal, hostId);
+ } else if (TnoOfPackets == 1) {
+// ----------------------------------------------------------------
+// The buffer is full and there was only one packet buffered. We
+// will send this as a normal signal.
+// ----------------------------------------------------------------
+ Uint32 TnewRef = numberToRef((hostBuffer[hostId].packetBufferTA[0] >> 16),
+ hostId);
+ MEMCOPY_NO_WORDS(&signal->theData[0],
+ &hostBuffer[hostId].packetBufferTA[1],
+ TpacketLen - 1);
+ sendSignal(TnewRef, GSN_TRANSID_AI, signal, (TpacketLen - 1), JBB);
+ TpacketLen = 0;
+ TnoOfPackets = 0;
+ } else {
+// ----------------------------------------------------------------
+// The buffer is full but at least two packets. Send those in
+// packed form.
+// ----------------------------------------------------------------
+ MEMCOPY_NO_WORDS(&signal->theData[0],
+ &hostBuffer[hostId].packetBufferTA[0],
+ TpacketLen);
+ sendSignal(TBref, GSN_TRANSID_AI, signal, TpacketLen, JBB);
+ TpacketLen = 0;
+ TnoOfPackets = 0;
+ }//if
+// ----------------------------------------------------------------
+// Copy the signal into the buffer
+// ----------------------------------------------------------------
+ hostBuffer[hostId].packetBufferTA[TpacketLen + 0] = Theader;
+ hostBuffer[hostId].packetBufferTA[TpacketLen + 1] = sig0;
+ hostBuffer[hostId].packetBufferTA[TpacketLen + 2] = sig1;
+ hostBuffer[hostId].packetBufferTA[TpacketLen + 3] = sig2;
+ hostBuffer[hostId].noOfPacketsTA = TnoOfPackets + 1;
+ hostBuffer[hostId].packetLenTA = Tlen + TpacketLen + 1;
+ MEMCOPY_NO_WORDS(&hostBuffer[hostId].packetBufferTA[TpacketLen + 4],
+ buffer,
+ Tlen - 3);
+}//Dbtup::bufferTRANSID_AI()
+
+void Dbtup::updatePackedList(Signal* signal, Uint16 hostId)
+{
+ if (hostBuffer[hostId].inPackedList == false) {
+ Uint32 TpackedListIndex = cpackedListIndex;
+ ljam();
+ hostBuffer[hostId].inPackedList = true;
+ cpackedList[TpackedListIndex] = hostId;
+ cpackedListIndex = TpackedListIndex + 1;
+ }//if
+}//Dbtup::updatePackedList()
+
+/* ---------------------------------------------------------------- */
+/* ----------------------- SEND READ ATTRINFO --------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::sendReadAttrinfo(Signal* signal,
+ Uint32 ToutBufIndex,
+ const Operationrec * const regOperPtr)
+{
+ const BlockReference recBlockref = regOperPtr->recBlockref;
+ bool toOwnNode = refToNode(recBlockref) == getOwnNodeId();
+ bool connectedToNode = getNodeInfo(refToNode(recBlockref)).m_connected;
+ const Uint32 type = getNodeInfo(refToNode(recBlockref)).m_type;
+ bool is_api = (type >= NodeInfo::API && type <= NodeInfo::REP);
+
+ if (ERROR_INSERTED(4006)){
+ // Use error insert to turn routing on
+ ljam();
+ connectedToNode = false;
+ }
+
+ if (!toOwnNode && !connectedToNode){
+ /**
+ * If this node does not have a direct connection
+ * to the receiving node we want to send the signals
+ * routed via the node that controls this read
+ */
+ Uint32 routeBlockref = regOperPtr->coordinatorTC;
+
+ /**
+ * Fill in a TRANSID_AI signal, use last word to store
+ * final destination and send it to route node
+ * as signal TRANSID_AI_R (R as in Routed)
+ */
+ TransIdAI * const transIdAI = (TransIdAI *)signal->getDataPtr();
+ transIdAI->connectPtr = regOperPtr->tcOperationPtr;
+ transIdAI->transId[0] = regOperPtr->transid1;
+ transIdAI->transId[1] = regOperPtr->transid2;
+
+ Uint32 tot = ToutBufIndex;
+ Uint32 sent = 0;
+ Uint32 maxLen = TransIdAI::DataLength - 1;
+ while (sent < tot) {
+ ljam();
+ Uint32 dataLen = (tot - sent > maxLen) ? maxLen : tot - sent;
+ Uint32 sigLen = dataLen + TransIdAI::HeaderLength + 1;
+ MEMCOPY_NO_WORDS(&transIdAI->attrData,
+ &coutBuffer[sent],
+ dataLen);
+ // Set final destination in last word
+ transIdAI->attrData[dataLen] = recBlockref;
+
+ sendSignal(routeBlockref, GSN_TRANSID_AI_R,
+ signal, sigLen, JBB);
+ sent += dataLen;
+
+ }
+ return;
+ }
+
+ Uint32 TbufIndex = 0;
+ Uint32 sig0 = regOperPtr->tcOperationPtr;
+ Uint32 sig1 = regOperPtr->transid1;
+ Uint32 sig2 = regOperPtr->transid2;
+ signal->theData[0] = sig0;
+ signal->theData[1] = sig1;
+ signal->theData[2] = sig2;
+
+ while (ToutBufIndex > 21) {
+ ljam();
+ MEMCOPY_NO_WORDS(&signal->theData[3],
+ &coutBuffer[TbufIndex],
+ 22);
+ TbufIndex += 22;
+ ToutBufIndex -= 22;
+ const BlockReference sendBref = regOperPtr->recBlockref;
+ if (refToNode(sendBref) != getOwnNodeId()) {
+ ljam();
+ sendSignal(sendBref, GSN_TRANSID_AI, signal, 25, JBB);
+ ljam();
+ } else {
+ ljam();
+ EXECUTE_DIRECT(refToBlock(sendBref), GSN_TRANSID_AI, signal, 25);
+ ljamEntry();
+ }//if
+ }//while
+
+ Uint32 TsigNumber;
+ Uint32 TsigLen;
+ Uint32 TdataIndex;
+ if ((regOperPtr->opSimple == ZTRUE) &&
+ (regOperPtr->optype == ZREAD)) {
+ /* DIRTY OPERATIONS ARE ALSO SIMPLE */
+ ljam();
+ Uint32 sig3 = regOperPtr->attroutbufLen;
+ TdataIndex = 4;
+ TsigLen = 4 + ToutBufIndex;
+ TsigNumber = GSN_READCONF;
+ signal->theData[3] = sig3;
+ if ((TsigLen < 18) && is_api){
+ bufferREADCONF(signal, regOperPtr->recBlockref,
+ &coutBuffer[TbufIndex], TsigLen);
+ return;
+ }//if
+ } else if (ToutBufIndex > 0) {
+ ljam();
+ TdataIndex = 3;
+ TsigLen = 3 + ToutBufIndex;
+ TsigNumber = GSN_TRANSID_AI;
+ if ((TsigLen < 18) && is_api){
+ ljam();
+ bufferTRANSID_AI(signal, regOperPtr->recBlockref,
+ &coutBuffer[TbufIndex], TsigLen);
+ return;
+ }//if
+ } else {
+ ljam();
+ return;
+ }//if
+ MEMCOPY_NO_WORDS(&signal->theData[TdataIndex],
+ &coutBuffer[TbufIndex],
+ ToutBufIndex);
+ const BlockReference sendBref = regOperPtr->recBlockref;
+ if (refToNode(sendBref) != getOwnNodeId()) {
+ ljam();
+ sendSignal(sendBref, TsigNumber, signal, TsigLen, JBB);
+ } else {
+ EXECUTE_DIRECT(refToBlock(sendBref), GSN_TRANSID_AI, signal, TsigLen);
+ ljamEntry();
+ }//if
+}//Dbtup::sendReadAttrinfo()
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupCommit.cpp b/ndb/src/kernel/blocks/dbtup/DbtupCommit.cpp
new file mode 100644
index 00000000000..fa3667b221e
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupCommit.cpp
@@ -0,0 +1,589 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <signaldata/TupCommit.hpp>
+
+#define ljam() { jamLine(5000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(5000 + __LINE__); }
+
+void Dbtup::execTUP_WRITELOG_REQ(Signal* signal)
+{
+ jamEntry();
+ OperationrecPtr loopOpPtr;
+ loopOpPtr.i = signal->theData[0];
+ Uint32 gci = signal->theData[1];
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ while (loopOpPtr.p->nextActiveOp != RNIL) {
+ ljam();
+ loopOpPtr.i = loopOpPtr.p->nextActiveOp;
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ }//while
+ do {
+ Uint32 blockNo = refToBlock(loopOpPtr.p->userblockref);
+ ndbrequire(loopOpPtr.p->transstate == STARTED);
+ signal->theData[0] = loopOpPtr.p->userpointer;
+ signal->theData[1] = gci;
+ if (loopOpPtr.p->prevActiveOp == RNIL) {
+ ljam();
+ EXECUTE_DIRECT(blockNo, GSN_LQH_WRITELOG_REQ, signal, 2);
+ return;
+ }//if
+ ljam();
+ EXECUTE_DIRECT(blockNo, GSN_LQH_WRITELOG_REQ, signal, 2);
+ jamEntry();
+ loopOpPtr.i = loopOpPtr.p->prevActiveOp;
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ } while (true);
+}//Dbtup::execTUP_WRITELOG_REQ()
+
+void Dbtup::execTUP_DEALLOCREQ(Signal* signal)
+{
+ TablerecPtr regTabPtr;
+ FragrecordPtr regFragPtr;
+
+ jamEntry();
+
+ Uint32 fragId = signal->theData[0];
+ regTabPtr.i = signal->theData[1];
+ Uint32 fragPageId = signal->theData[2];
+ Uint32 pageIndex = signal->theData[3];
+
+ ptrCheckGuard(regTabPtr, cnoOfTablerec, tablerec);
+ getFragmentrec(regFragPtr, fragId, regTabPtr.p);
+ ndbrequire(regFragPtr.p != NULL);
+
+ PagePtr pagePtr;
+ pagePtr.i = getRealpid(regFragPtr.p, fragPageId);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 pageIndexScaled = pageIndex >> 1;
+ ndbrequire((pageIndex & 1) == 0);
+ Uint32 pageOffset = ZPAGE_HEADER_SIZE +
+ (regTabPtr.p->tupheadsize * pageIndexScaled);
+//---------------------------------------------------
+/* --- Deallocate a tuple as requested by ACC --- */
+//---------------------------------------------------
+ if (isUndoLoggingNeeded(regFragPtr.p, fragPageId)) {
+ ljam();
+ cprAddUndoLogRecord(signal,
+ ZLCPR_TYPE_INSERT_TH,
+ fragPageId,
+ pageIndex,
+ regTabPtr.i,
+ fragId,
+ regFragPtr.p->checkpointVersion);
+ cprAddData(signal,
+ regFragPtr.p,
+ pagePtr.i,
+ regTabPtr.p->tupheadsize,
+ pageOffset);
+ }//if
+ {
+ freeTh(regFragPtr.p,
+ regTabPtr.p,
+ signal,
+ pagePtr.p,
+ pageOffset);
+ }
+}
+
+/* ---------------------------------------------------------------- */
+/* ------------ PERFORM A COMMIT ON AN UPDATE OPERATION ---------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::commitUpdate(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr)
+{
+ if (regOperPtr->realPageIdC != RNIL) {
+ if (isUndoLoggingNeeded(regFragPtr, regOperPtr->fragPageIdC)) {
+/* ------------------------------------------------------------------------ */
+/* IF THE COPY WAS CREATED WITHIN THIS CHECKPOINT WE ONLY HAVE */
+/* TO LOG THE CREATION OF THE COPY. IF HOWEVER IT WAS CREATED BEFORE SAVE */
+/* THIS CHECKPOINT, WE HAVE TO THE DATA AS WELL. */
+/* ------------------------------------------------------------------------ */
+ if (regOperPtr->undoLogged) {
+ ljam();
+ cprAddUndoLogRecord(signal,
+ ZLCPR_TYPE_INSERT_TH_NO_DATA,
+ regOperPtr->fragPageIdC,
+ regOperPtr->pageIndexC,
+ regOperPtr->tableRef,
+ regOperPtr->fragId,
+ regFragPtr->checkpointVersion);
+ } else {
+ ljam();
+ cprAddUndoLogRecord(signal,
+ ZLCPR_TYPE_INSERT_TH,
+ regOperPtr->fragPageIdC,
+ regOperPtr->pageIndexC,
+ regOperPtr->tableRef,
+ regOperPtr->fragId,
+ regFragPtr->checkpointVersion);
+ cprAddData(signal,
+ regFragPtr,
+ regOperPtr->realPageIdC,
+ regTabPtr->tupheadsize,
+ regOperPtr->pageOffsetC);
+ }//if
+ }//if
+
+ PagePtr copyPagePtr;
+ copyPagePtr.i = regOperPtr->realPageIdC;
+ ptrCheckGuard(copyPagePtr, cnoOfPage, page);
+ freeTh(regFragPtr,
+ regTabPtr,
+ signal,
+ copyPagePtr.p,
+ (Uint32)regOperPtr->pageOffsetC);
+ regOperPtr->realPageIdC = RNIL;
+ regOperPtr->fragPageIdC = RNIL;
+ regOperPtr->pageOffsetC = ZNIL;
+ regOperPtr->pageIndexC = ZNIL;
+ }//if
+}//Dbtup::commitUpdate()
+
+void
+Dbtup::commitSimple(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr)
+{
+ operPtr.p = regOperPtr;
+ fragptr.p = regFragPtr;
+ tabptr.p = regTabPtr;
+
+ // Checking detached triggers
+ checkDetachedTriggers(signal,
+ regOperPtr,
+ regTabPtr);
+
+ removeActiveOpList(regOperPtr);
+ if (regOperPtr->optype == ZUPDATE) {
+ ljam();
+ commitUpdate(signal, regOperPtr, regFragPtr, regTabPtr);
+ if (regTabPtr->GCPIndicator) {
+ updateGcpId(signal, regOperPtr, regFragPtr, regTabPtr);
+ }//if
+ } else if (regOperPtr->optype == ZINSERT) {
+ ljam();
+ if (regTabPtr->GCPIndicator) {
+ updateGcpId(signal, regOperPtr, regFragPtr, regTabPtr);
+ }//if
+ } else {
+ ndbrequire(regOperPtr->optype == ZDELETE);
+ }//if
+}//Dbtup::commitSimple()
+
+void Dbtup::removeActiveOpList(Operationrec* const regOperPtr)
+{
+ if (regOperPtr->inActiveOpList == ZTRUE) {
+ OperationrecPtr raoOperPtr;
+ regOperPtr->inActiveOpList = ZFALSE;
+ if (regOperPtr->prevActiveOp != RNIL) {
+ ljam();
+ raoOperPtr.i = regOperPtr->prevActiveOp;
+ ptrCheckGuard(raoOperPtr, cnoOfOprec, operationrec);
+ raoOperPtr.p->nextActiveOp = regOperPtr->nextActiveOp;
+ } else {
+ ljam();
+ PagePtr pagePtr;
+ pagePtr.i = regOperPtr->realPageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ ndbrequire(regOperPtr->pageOffset < ZWORDS_ON_PAGE);
+ pagePtr.p->pageWord[regOperPtr->pageOffset] = regOperPtr->nextActiveOp;
+ }//if
+ if (regOperPtr->nextActiveOp != RNIL) {
+ ljam();
+ raoOperPtr.i = regOperPtr->nextActiveOp;
+ ptrCheckGuard(raoOperPtr, cnoOfOprec, operationrec);
+ raoOperPtr.p->prevActiveOp = regOperPtr->prevActiveOp;
+ }//if
+ regOperPtr->prevActiveOp = RNIL;
+ regOperPtr->nextActiveOp = RNIL;
+ }//if
+}//Dbtup::removeActiveOpList()
+
+/* ---------------------------------------------------------------- */
+/* INITIALIZATION OF ONE CONNECTION RECORD TO PREPARE FOR NEXT OP. */
+/* ---------------------------------------------------------------- */
+void Dbtup::initOpConnection(Operationrec* const regOperPtr)
+{
+ Uint32 RinFragList = regOperPtr->inFragList;
+ regOperPtr->transstate = IDLE;
+ regOperPtr->currentAttrinbufLen = 0;
+ regOperPtr->optype = ZREAD;
+ if (RinFragList == ZTRUE) {
+ OperationrecPtr tropNextLinkPtr;
+ OperationrecPtr tropPrevLinkPtr;
+/*----------------------------------------------------------------- */
+/* TO ENSURE THAT WE HAVE SUCCESSFUL ABORTS OF FOLLOWING */
+/* OPERATIONS WHICH NEVER STARTED WE SET THE OPTYPE TO READ. */
+/*----------------------------------------------------------------- */
+/* REMOVE IT FROM THE DOUBLY LINKED LIST ON THE FRAGMENT */
+/*----------------------------------------------------------------- */
+ tropPrevLinkPtr.i = regOperPtr->prevOprecInList;
+ tropNextLinkPtr.i = regOperPtr->nextOprecInList;
+ regOperPtr->inFragList = ZFALSE;
+ if (tropPrevLinkPtr.i == RNIL) {
+ ljam();
+ FragrecordPtr regFragPtr;
+ regFragPtr.i = regOperPtr->fragmentPtr;
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+ regFragPtr.p->firstusedOprec = tropNextLinkPtr.i;
+ } else {
+ ljam();
+ ptrCheckGuard(tropPrevLinkPtr, cnoOfOprec, operationrec);
+ tropPrevLinkPtr.p->nextOprecInList = tropNextLinkPtr.i;
+ }//if
+ if (tropNextLinkPtr.i == RNIL) {
+ ;
+ } else {
+ ljam();
+ ptrCheckGuard(tropNextLinkPtr, cnoOfOprec, operationrec);
+ tropNextLinkPtr.p->prevOprecInList = tropPrevLinkPtr.i;
+ }//if
+ regOperPtr->prevOprecInList = RNIL;
+ regOperPtr->nextOprecInList = RNIL;
+ }//if
+}//Dbtup::initOpConnection()
+
+/* ----------------------------------------------------------------- */
+/* --------------- COMMIT THIS PART OF A TRANSACTION --------------- */
+/* ----------------------------------------------------------------- */
+void Dbtup::execTUP_COMMITREQ(Signal* signal)
+{
+ FragrecordPtr regFragPtr;
+ OperationrecPtr regOperPtr;
+ TablerecPtr regTabPtr;
+
+ TupCommitReq * const tupCommitReq = (TupCommitReq *)signal->getDataPtr();
+
+ ljamEntry();
+ regOperPtr.i = tupCommitReq->opPtr;
+ ptrCheckGuard(regOperPtr, cnoOfOprec, operationrec);
+
+ ndbrequire(regOperPtr.p->transstate == STARTED);
+ regOperPtr.p->gci = tupCommitReq->gci;
+ regOperPtr.p->hashValue = tupCommitReq->hashValue;
+
+ regFragPtr.i = regOperPtr.p->fragmentPtr;
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+
+ regTabPtr.i = regOperPtr.p->tableRef;
+ ptrCheckGuard(regTabPtr, cnoOfTablerec, tablerec);
+
+ if (!regTabPtr.p->tuxCustomTriggers.isEmpty()) {
+ ljam();
+ executeTuxCommitTriggers(signal,
+ regOperPtr.p,
+ regTabPtr.p);
+ }
+
+ if (regOperPtr.p->tupleState == NO_OTHER_OP) {
+ if ((regOperPtr.p->prevActiveOp == RNIL) &&
+ (regOperPtr.p->nextActiveOp == RNIL)) {
+ ljam();
+/* ---------------------------------------------------------- */
+// We handle the simple case separately as an optimisation
+/* ---------------------------------------------------------- */
+ commitSimple(signal,
+ regOperPtr.p,
+ regFragPtr.p,
+ regTabPtr.p);
+ } else {
+/* ---------------------------------------------------------- */
+// This is the first commit message of this record in this
+// transaction. We will commit this record completely for this
+// transaction. If there are other operations they will be
+// responsible to release their own resources. Also commit of
+// a delete is postponed until the last operation is committed
+// on the tuple.
+//
+// As part of this commitRecord we will also handle detached
+// triggers and release of resources for this operation.
+/* ---------------------------------------------------------- */
+ ljam();
+ commitRecord(signal,
+ regOperPtr.p,
+ regFragPtr.p,
+ regTabPtr.p);
+ removeActiveOpList(regOperPtr.p);
+ }//if
+ } else {
+ ljam();
+/* ---------------------------------------------------------- */
+// Release any copy tuples
+/* ---------------------------------------------------------- */
+ ndbrequire(regOperPtr.p->tupleState == TO_BE_COMMITTED);
+ commitUpdate(signal, regOperPtr.p, regFragPtr.p, regTabPtr.p);
+ removeActiveOpList(regOperPtr.p);
+ }//if
+ initOpConnection(regOperPtr.p);
+}//execTUP_COMMITREQ()
+
+void
+Dbtup::updateGcpId(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr)
+{
+ PagePtr pagePtr;
+ ljam();
+//--------------------------------------------------------------------
+// Is this code safe for UNDO logging. Not sure currently. RONM
+//--------------------------------------------------------------------
+ pagePtr.i = regOperPtr->realPageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 temp = regOperPtr->pageOffset + regTabPtr->tupGCPIndex;
+ ndbrequire((temp < ZWORDS_ON_PAGE) &&
+ (regTabPtr->tupGCPIndex < regTabPtr->tupheadsize));
+ if (isUndoLoggingNeeded(regFragPtr, regOperPtr->fragPageId)) {
+ Uint32 prevGCI = pagePtr.p->pageWord[temp];
+ ljam();
+ cprAddUndoLogRecord(signal,
+ ZLCPR_TYPE_UPDATE_GCI,
+ regOperPtr->fragPageId,
+ regOperPtr->pageIndex,
+ regOperPtr->tableRef,
+ regOperPtr->fragId,
+ regFragPtr->checkpointVersion);
+ cprAddGCIUpdate(signal,
+ prevGCI,
+ regFragPtr);
+ }//if
+ pagePtr.p->pageWord[temp] = regOperPtr->gci;
+ if (regTabPtr->checksumIndicator) {
+ ljam();
+ setChecksum(pagePtr.p, regOperPtr->pageOffset, regTabPtr->tupheadsize);
+ }//if
+}//Dbtup::updateGcpId()
+
+void
+Dbtup::commitRecord(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr)
+{
+ Uint32 opType;
+ OperationrecPtr firstOpPtr;
+ PagePtr pagePtr;
+
+ pagePtr.i = regOperPtr->realPageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+
+ setTupleStatesSetOpType(regOperPtr, pagePtr.p, opType, firstOpPtr);
+
+ fragptr.p = regFragPtr;
+ tabptr.p = regTabPtr;
+
+ if (opType == ZINSERT_DELETE) {
+ ljam();
+//--------------------------------------------------------------------
+// We started by inserting the tuple and ended by deleting. Seen from
+// transactions point of view no changes were made.
+//--------------------------------------------------------------------
+ commitUpdate(signal, regOperPtr, regFragPtr, regTabPtr);
+ return;
+ } else if (opType == ZINSERT) {
+ ljam();
+//--------------------------------------------------------------------
+// We started by inserting whereafter we made several changes to the
+// tuple that could include updates, deletes and new inserts. The final
+// state of the tuple is the original tuple. This is reached from this
+// operation. We change the optype on this operation to ZINSERT to
+// ensure proper operation of the detached trigger.
+// We restore the optype after executing triggers although not really
+// needed.
+//--------------------------------------------------------------------
+ Uint32 saveOpType = regOperPtr->optype;
+ regOperPtr->optype = ZINSERT;
+ operPtr.p = regOperPtr;
+
+ checkDetachedTriggers(signal,
+ regOperPtr,
+ regTabPtr);
+
+ regOperPtr->optype = saveOpType;
+ } else if (opType == ZUPDATE) {
+ ljam();
+//--------------------------------------------------------------------
+// We want to use the first operation which contains a copy tuple
+// reference. This operation contains the before value of this record
+// for this transaction. Then this operation is used for executing
+// triggers with optype set to update.
+//--------------------------------------------------------------------
+ OperationrecPtr befOpPtr;
+ findBeforeValueOperation(befOpPtr, firstOpPtr);
+
+ Uint32 saveOpType = befOpPtr.p->optype;
+ Bitmask<MAXNROFATTRIBUTESINWORDS> attributeMask;
+ Bitmask<MAXNROFATTRIBUTESINWORDS> saveAttributeMask;
+
+ calculateChangeMask(pagePtr.p,
+ regTabPtr,
+ befOpPtr.p->pageOffset,
+ attributeMask);
+
+ saveAttributeMask.clear();
+ saveAttributeMask.bitOR(befOpPtr.p->changeMask);
+ befOpPtr.p->changeMask.clear();
+ befOpPtr.p->changeMask.bitOR(attributeMask);
+
+ operPtr.p = befOpPtr.p;
+ checkDetachedTriggers(signal,
+ befOpPtr.p,
+ regTabPtr);
+
+ befOpPtr.p->changeMask.clear();
+ befOpPtr.p->changeMask.bitOR(saveAttributeMask);
+
+ befOpPtr.p->optype = saveOpType;
+ } else if (opType == ZDELETE) {
+ ljam();
+//--------------------------------------------------------------------
+// We want to use the first operation which contains a copy tuple.
+// We benefit from the fact that we know that it cannot be a simple
+// delete and it cannot be an insert followed by a delete. Thus there
+// must either be an update or a insert following a delete. In both
+// cases we will find a before value in a copy tuple.
+//
+// An added complexity is that the trigger handling assumes that the
+// before value is located in the original tuple so we have to move the
+// copy tuple reference to the original tuple reference and afterwards
+// restore it again.
+//--------------------------------------------------------------------
+ OperationrecPtr befOpPtr;
+ findBeforeValueOperation(befOpPtr, firstOpPtr);
+ Uint32 saveOpType = befOpPtr.p->optype;
+
+ Uint32 realPageId = befOpPtr.p->realPageId;
+ Uint32 pageOffset = befOpPtr.p->pageOffset;
+ Uint32 fragPageId = befOpPtr.p->fragPageId;
+ Uint32 pageIndex = befOpPtr.p->pageIndex;
+
+ befOpPtr.p->realPageId = befOpPtr.p->realPageIdC;
+ befOpPtr.p->pageOffset = befOpPtr.p->pageOffsetC;
+ befOpPtr.p->fragPageId = befOpPtr.p->fragPageIdC;
+ befOpPtr.p->pageIndex = befOpPtr.p->pageIndexC;
+
+ operPtr.p = befOpPtr.p;
+ checkDetachedTriggers(signal,
+ befOpPtr.p,
+ regTabPtr);
+
+ befOpPtr.p->realPageId = realPageId;
+ befOpPtr.p->pageOffset = pageOffset;
+ befOpPtr.p->fragPageId = fragPageId;
+ befOpPtr.p->pageIndex = pageIndex;
+ befOpPtr.p->optype = saveOpType;
+ } else {
+ ndbrequire(false);
+ }//if
+
+ commitUpdate(signal, regOperPtr, regFragPtr, regTabPtr);
+ if (regTabPtr->GCPIndicator) {
+ updateGcpId(signal, regOperPtr, regFragPtr, regTabPtr);
+ }//if
+}//Dbtup::commitRecord()
+
+void
+Dbtup::setTupleStatesSetOpType(Operationrec* const regOperPtr,
+ Page* const pagePtr,
+ Uint32& opType,
+ OperationrecPtr& firstOpPtr)
+{
+ OperationrecPtr loopOpPtr;
+ OperationrecPtr lastOpPtr;
+
+ ndbrequire(regOperPtr->pageOffset < ZWORDS_ON_PAGE);
+ loopOpPtr.i = pagePtr->pageWord[regOperPtr->pageOffset];
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ lastOpPtr = loopOpPtr;
+ if (loopOpPtr.p->optype == ZDELETE) {
+ ljam();
+ opType = ZDELETE;
+ } else {
+ ljam();
+ opType = ZUPDATE;
+ }//if
+ do {
+ ljam();
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ firstOpPtr = loopOpPtr;
+ loopOpPtr.p->tupleState = TO_BE_COMMITTED;
+ loopOpPtr.i = loopOpPtr.p->nextActiveOp;
+ } while (loopOpPtr.i != RNIL);
+ if (opType == ZDELETE) {
+ ljam();
+ if (firstOpPtr.p->optype == ZINSERT) {
+ ljam();
+ opType = ZINSERT_DELETE;
+ }//if
+ } else {
+ ljam();
+ if (firstOpPtr.p->optype == ZINSERT) {
+ ljam();
+ opType = ZINSERT;
+ }//if
+ }///if
+}//Dbtup::setTupleStatesSetOpType()
+
+void Dbtup::findBeforeValueOperation(OperationrecPtr& befOpPtr,
+ OperationrecPtr firstOpPtr)
+{
+ befOpPtr = firstOpPtr;
+ if (befOpPtr.p->realPageIdC != RNIL) {
+ ljam();
+ return;
+ } else {
+ ljam();
+ befOpPtr.i = befOpPtr.p->prevActiveOp;
+ ptrCheckGuard(befOpPtr, cnoOfOprec, operationrec);
+ ndbrequire(befOpPtr.p->realPageIdC != RNIL);
+ }//if
+}//Dbtup::findBeforeValueOperation()
+
+void
+Dbtup::calculateChangeMask(Page* const pagePtr,
+ Tablerec* const regTabPtr,
+ Uint32 pageOffset,
+ Bitmask<MAXNROFATTRIBUTESINWORDS>& attributeMask)
+{
+ OperationrecPtr loopOpPtr;
+
+ attributeMask.clear();
+ ndbrequire(pageOffset < ZWORDS_ON_PAGE);
+ loopOpPtr.i = pagePtr->pageWord[pageOffset];
+ do {
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ if (loopOpPtr.p->optype == ZUPDATE) {
+ ljam();
+ attributeMask.bitOR(loopOpPtr.p->changeMask);
+ } else if (loopOpPtr.p->optype == ZINSERT) {
+ ljam();
+ attributeMask.set();
+ return;
+ } else {
+ ndbrequire(loopOpPtr.p->optype == ZDELETE);
+ }//if
+ loopOpPtr.i = loopOpPtr.p->nextActiveOp;
+ } while (loopOpPtr.i != RNIL);
+}//Dbtup::calculateChangeMask()
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupDebug.cpp b/ndb/src/kernel/blocks/dbtup/DbtupDebug.cpp
new file mode 100644
index 00000000000..c38fde23404
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupDebug.cpp
@@ -0,0 +1,399 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <signaldata/DropTab.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+#include <signaldata/EventReport.hpp>
+#include <Vector.hpp>
+
+#define ljam() { jamLine(30000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(30000 + __LINE__); }
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* ------------------------ DEBUG MODULE -------------------------- */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+void Dbtup::execDEBUG_SIG(Signal* signal)
+{
+ PagePtr regPagePtr;
+ ljamEntry();
+ regPagePtr.i = signal->theData[0];
+ ptrCheckGuard(regPagePtr, cnoOfPage, page);
+}//Dbtup::execDEBUG_SIG()
+
+#ifdef TEST_MR
+#include <time.h>
+
+void startTimer(struct timespec *tp)
+{
+ clock_gettime(CLOCK_REALTIME, tp);
+}//startTimer()
+
+int stopTimer(struct timespec *tp)
+{
+ double timer_count;
+ struct timespec theStopTime;
+ clock_gettime(CLOCK_REALTIME, &theStopTime);
+ timer_count = (double)(1000000*((double)theStopTime.tv_sec - (double)tp->tv_sec)) +
+ (double)((double)((double)theStopTime.tv_nsec - (double)tp->tv_nsec)/(double)1000);
+ return (int)timer_count;
+}//stopTimer()
+
+#endif // end TEST_MR
+
+struct Chunk {
+ Uint32 pageId;
+ Uint32 pageCount;
+};
+
+void
+Dbtup::reportMemoryUsage(Signal* signal, int incDec){
+ signal->theData[0] = EventReport::MemoryUsage;
+ signal->theData[1] = incDec;
+ signal->theData[2] = sizeof(Page);
+ signal->theData[3] = cnoOfAllocatedPages;
+ signal->theData[4] = cnoOfPage;
+ signal->theData[5] = DBTUP;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 6, JBB);
+}
+
+void
+Dbtup::execDUMP_STATE_ORD(Signal* signal)
+{
+ Uint32 type = signal->theData[0];
+ if(type == DumpStateOrd::DumpPageMemory){
+ reportMemoryUsage(signal, 0);
+ return;
+ }
+ DumpStateOrd * const dumpState = (DumpStateOrd *)&signal->theData[0];
+
+#if 0
+ if (type == 100) {
+ RelTabMemReq * const req = (RelTabMemReq *)signal->getDataPtrSend();
+ req->primaryTableId = 2;
+ req->secondaryTableId = RNIL;
+ req->userPtr = 2;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownref, GSN_REL_TABMEMREQ, signal,
+ RelTabMemReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 101) {
+ RelTabMemReq * const req = (RelTabMemReq *)signal->getDataPtrSend();
+ req->primaryTableId = 4;
+ req->secondaryTableId = 5;
+ req->userPtr = 4;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownref, GSN_REL_TABMEMREQ, signal,
+ RelTabMemReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 102) {
+ RelTabMemReq * const req = (RelTabMemReq *)signal->getDataPtrSend();
+ req->primaryTableId = 6;
+ req->secondaryTableId = 8;
+ req->userPtr = 6;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownref, GSN_REL_TABMEMREQ, signal,
+ RelTabMemReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 103) {
+ DropTabFileReq * const req = (DropTabFileReq *)signal->getDataPtrSend();
+ req->primaryTableId = 2;
+ req->secondaryTableId = RNIL;
+ req->userPtr = 2;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownref, GSN_DROP_TABFILEREQ, signal,
+ DropTabFileReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 104) {
+ DropTabFileReq * const req = (DropTabFileReq *)signal->getDataPtrSend();
+ req->primaryTableId = 4;
+ req->secondaryTableId = 5;
+ req->userPtr = 4;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownref, GSN_DROP_TABFILEREQ, signal,
+ DropTabFileReq::SignalLength, JBB);
+ return;
+ }//if
+ if (type == 105) {
+ DropTabFileReq * const req = (DropTabFileReq *)signal->getDataPtrSend();
+ req->primaryTableId = 6;
+ req->secondaryTableId = 8;
+ req->userPtr = 6;
+ req->userRef = DBDICT_REF;
+ sendSignal(cownref, GSN_DROP_TABFILEREQ, signal,
+ DropTabFileReq::SignalLength, JBB);
+ return;
+ }//if
+#endif
+#ifdef ERROR_INSERT
+ if (type == DumpStateOrd::EnableUndoDelayDataWrite) {
+ ndbout << "Dbtup:: delay write of datapages for table = "
+ << dumpState->args[1]<< endl;
+ c_errorInsert4000TableId = dumpState->args[1];
+ SET_ERROR_INSERT_VALUE(4000);
+ return;
+ }//if
+#endif
+#ifdef VM_TRACE
+ if (type == 1211){
+ ndbout_c("Startar modul test av Page Manager");
+
+ Vector<Chunk> chunks;
+ const Uint32 LOOPS = 1000;
+ for(Uint32 i = 0; i<LOOPS; i++){
+
+ // Case
+ Uint32 c = (rand() % 3);
+ const Uint32 free = cnoOfPage - cnoOfAllocatedPages;
+
+ Uint32 alloc = 0;
+ if(free <= 1){
+ c = 0;
+ alloc = 1;
+ } else
+ alloc = 1 + (rand() % (free - 1));
+
+ if(chunks.size() == 0 && c == 0){
+ c = 1 + rand() % 2;
+ }
+
+ ndbout_c("loop=%d case=%d free=%d alloc=%d", i, c, free, alloc);
+ switch(c){
+ case 0:{ // Release
+ const int ch = rand() % chunks.size();
+ Chunk chunk = chunks[ch];
+ chunks.erase(ch);
+ returnCommonArea(chunk.pageId, chunk.pageCount);
+ }
+ break;
+ case 2: { // Seize(n) - fail
+ alloc += free;
+ // Fall through
+ }
+ case 1: { // Seize(n) (success)
+
+ Chunk chunk;
+ allocConsPages(alloc, chunk.pageCount, chunk.pageId);
+ ndbrequire(chunk.pageCount <= alloc);
+ if(chunk.pageCount != 0){
+ chunks.push_back(chunk);
+ } else {
+ ndbout_c(" Failed to alloc %d pages with %d pages free",
+ alloc, free);
+ }
+
+ for(Uint32 i = 0; i<chunk.pageCount; i++){
+ PagePtr pagePtr;
+ pagePtr.i = chunk.pageId + i;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ pagePtr.p->pageWord[ZPAGE_STATE_POS] = ~ZFREE_COMMON;
+ }
+ }
+ break;
+ }
+ }
+ while(chunks.size() > 0){
+ Chunk chunk = chunks.back();
+ returnCommonArea(chunk.pageId, chunk.pageCount);
+ chunks.erase(chunks.size() - 1);
+ }
+ }
+#endif
+}//Dbtup::execDUMP_STATE_ORD()
+
+/* ---------------------------------------------------------------- */
+/* --------- MEMORY CHECK ----------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::execMEMCHECKREQ(Signal* signal)
+{
+ PagePtr regPagePtr;
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ CheckpointInfoPtr ciPtr;
+ UndoPagePtr regUndoPagePtr;
+ Uint32* data = &signal->theData[0];
+
+ ljamEntry();
+ BlockReference blockref = signal->theData[0];
+ for (Uint32 i = 0; i < 25; i++) {
+ ljam();
+ data[i] = 0;
+ }//for
+ for (Uint32 i = 0; i < 16; i++) {
+ regPagePtr.i = cfreepageList[i];
+ ljam();
+ while (regPagePtr.i != RNIL) {
+ ljam();
+ ptrCheckGuard(regPagePtr, cnoOfPage, page);
+ regPagePtr.i = regPagePtr.p->pageWord[ZPAGE_NEXT_POS];
+ data[0]++;
+ }//while
+ }//for
+ regUndoPagePtr.i = cfirstfreeUndoSeg;
+ while (regUndoPagePtr.i != RNIL) {
+ ljam();
+ ptrCheckGuard(regUndoPagePtr, cnoOfUndoPage, undoPage);
+ regUndoPagePtr.i = regUndoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS];
+ data[1] += ZUB_SEGMENT_SIZE;
+ }//while
+ ciPtr.i = cfirstfreeLcp;
+ while (ciPtr.i != RNIL) {
+ ljam();
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ ciPtr.i = ciPtr.p->lcpNextRec;
+ data[2]++;
+ }//while
+ dbsiPtr.i = cfirstfreePdx;
+ while (dbsiPtr.i != ZNIL) {
+ ljam();
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ dbsiPtr.i = dbsiPtr.p->pdxNextRec;
+ data[3]++;
+ }//while
+ sendSignal(blockref, GSN_MEMCHECKCONF, signal, 25, JBB);
+}//Dbtup::memCheck()
+
+// ------------------------------------------------------------------------
+// Help function to be used when debugging. Prints out a tuple page.
+// printLimit is the number of bytes that is printed out from the page. A
+// page is of size 32768 bytes as of March 2003.
+// ------------------------------------------------------------------------
+void Dbtup::printoutTuplePage(Uint32 fragid, Uint32 pageid, Uint32 printLimit)
+{
+ PagePtr tmpPageP;
+ FragrecordPtr tmpFragP;
+ TablerecPtr tmpTableP;
+ Uint32 tmpTupleSize;
+
+ tmpPageP.i = pageid;
+ ptrCheckGuard(tmpPageP, cnoOfPage, page);
+
+ tmpFragP.i = fragid;
+ ptrCheckGuard(tmpFragP, cnoOfFragrec, fragrecord);
+
+ tmpTableP.i = tmpFragP.p->fragTableId;
+ ptrCheckGuard(tmpTableP, cnoOfTablerec, tablerec);
+
+ tmpTupleSize = tmpTableP.p->tupheadsize;
+
+ ndbout << "Fragid: " << fragid << " Pageid: " << pageid << endl
+ << "----------------------------------------" << endl;
+
+ ndbout << "PageHead : ";
+ for (Uint32 i1 = 0; i1 < ZPAGE_HEADER_SIZE; i1++) {
+ if (i1 == 3)
+ ndbout << (tmpPageP.p->pageWord[i1] >> 16) << "," << (tmpPageP.p->pageWord[i1] & 0xffff) << " ";
+ else if (tmpPageP.p->pageWord[i1] == 4059165169u)
+ ndbout << "F1F1F1F1 ";
+ else if (tmpPageP.p->pageWord[i1] == 268435455u)
+ ndbout << "RNIL ";
+ else
+ ndbout << tmpPageP.p->pageWord[i1] << " ";
+ }//for
+ ndbout << endl;
+ for (Uint32 i = ZPAGE_HEADER_SIZE; i < printLimit; i += tmpTupleSize) {
+ ndbout << "pagepos " << i << " : ";
+
+ for (Uint32 j = i; j < i + tmpTupleSize; j++) {
+ if (tmpPageP.p->pageWord[j] == 4059165169u)
+ ndbout << "F1F1F1F1 ";
+ else if (tmpPageP.p->pageWord[j] == 268435455u)
+ ndbout << "RNIL ";
+ else
+ ndbout << tmpPageP.p->pageWord[j] << " ";
+ }//for
+ ndbout << endl;
+ }//for
+}//Dbtup::printoutTuplePage
+
+#ifdef VM_TRACE
+NdbOut&
+operator<<(NdbOut& out, const Dbtup::Operationrec& op)
+{
+ out << "[Operationrec " << hex << &op;
+ // table
+ out << " [tableRef " << dec << op.tableRef << "]";
+ out << " [fragId " << dec << op.fragId << "]";
+ out << " [fragmentPtr " << hex << op.fragmentPtr << "]";
+ // type
+ out << " [optype " << dec << op.optype << "]";
+ out << " [deleteInsertFlag " << dec << op.deleteInsertFlag << "]";
+ out << " [dirtyOp " << dec << op.dirtyOp << "]";
+ out << " [interpretedExec " << dec << op.interpretedExec << "]";
+ out << " [opSimple " << dec << op.opSimple << "]";
+ // state
+ out << " [tupleState " << dec << op.tupleState << "]";
+ out << " [transstate " << dec << op.transstate << "]";
+ out << " [inFragList " << dec << op.inFragList << "]";
+ out << " [inActiveOpList " << dec << op.inActiveOpList << "]";
+ out << " [undoLogged " << dec << op.undoLogged << "]";
+ // links
+ out << " [prevActiveOp " << hex << op.prevActiveOp << "]";
+ out << " [nextActiveOp " << hex << op.nextActiveOp << "]";
+ // tuples
+ out << " [tupVersion " << hex << op.tupVersion << "]";
+ out << " [fragPageId " << dec << op.fragPageId << "]";
+ out << " [pageIndex " << dec << op.pageIndex << "]";
+ out << " [realPageId " << hex << op.realPageId << "]";
+ out << " [pageOffset " << dec << op.pageOffset << "]";
+ out << " [fragPageIdC " << dec << op.fragPageIdC << "]";
+ out << " [pageIndexC " << dec << op.pageIndexC << "]";
+ out << " [realPageIdC " << hex << op.realPageIdC << "]";
+ out << " [pageOffsetC " << dec << op.pageOffsetC << "]";
+ // trans
+ out << " [transid1 " << hex << op.transid1 << "]";
+ out << " [transid2 " << hex << op.transid2 << "]";
+ out << "]";
+ return out;
+}
+
+// uses global tabptr
+NdbOut&
+operator<<(NdbOut& out, const Dbtup::Th& th)
+{
+ // ugly
+ Dbtup* tup = (Dbtup*)globalData.getBlock(DBTUP);
+ const Dbtup::Tablerec& tab = *tup->tabptr.p;
+ unsigned i = 0;
+ out << "[Th " << hex << &th;
+ out << " [op " << hex << th.data[i++] << "]";
+ out << " [version " << hex << (Uint16)th.data[i++] << "]";
+ if (tab.checksumIndicator)
+ out << " [checksum " << hex << th.data[i++] << "]";
+ out << " [nullbits";
+ for (unsigned j = 0; j < tab.tupNullWords; j++)
+ out << " " << hex << th.data[i++];
+ out << "]";
+ if (tab.GCPIndicator)
+ out << " [gcp " << dec << th.data[i++] << "]";
+ out << " [data";
+ while (i < tab.tupheadsize)
+ out << " " << hex << th.data[i++];
+ out << "]";
+ out << "]";
+ return out;
+}
+#endif
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupExecQuery.cpp b/ndb/src/kernel/blocks/dbtup/DbtupExecQuery.cpp
new file mode 100644
index 00000000000..07bad00acf1
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupExecQuery.cpp
@@ -0,0 +1,2067 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <AttributeDescriptor.hpp>
+#include "AttributeOffset.hpp"
+#include <AttributeHeader.hpp>
+#include <Interpreter.hpp>
+#include <signaldata/TupCommit.hpp>
+#include <signaldata/TupKey.hpp>
+#include <NdbSqlUtil.hpp>
+
+/* ----------------------------------------------------------------- */
+/* ----------- INIT_STORED_OPERATIONREC -------------- */
+/* ----------------------------------------------------------------- */
+int Dbtup::initStoredOperationrec(Operationrec* const regOperPtr,
+ Uint32 storedId)
+{
+ jam();
+ StoredProcPtr storedPtr;
+ c_storedProcPool.getPtr(storedPtr, storedId);
+ if (storedPtr.i != RNIL) {
+ if (storedPtr.p->storedCode == ZSCAN_PROCEDURE) {
+ storedPtr.p->storedCounter++;
+ regOperPtr->firstAttrinbufrec = storedPtr.p->storedLinkFirst;
+ regOperPtr->lastAttrinbufrec = storedPtr.p->storedLinkLast;
+ regOperPtr->attrinbufLen = storedPtr.p->storedProcLength;
+ regOperPtr->currentAttrinbufLen = storedPtr.p->storedProcLength;
+ return ZOK;
+ }//if
+ }//if
+ terrorCode = ZSTORED_PROC_ID_ERROR;
+ return terrorCode;
+}//Dbtup::initStoredOperationrec()
+
+void Dbtup::copyAttrinfo(Signal* signal,
+ Operationrec * const regOperPtr,
+ Uint32* inBuffer)
+{
+ AttrbufrecPtr copyAttrBufPtr;
+ Uint32 RnoOfAttrBufrec = cnoOfAttrbufrec;
+ int RbufLen;
+ Uint32 RinBufIndex = 0;
+ Uint32 Rnext;
+ Uint32 Rfirst;
+ Uint32 TstoredProcedure = (regOperPtr->storedProcedureId != ZNIL);
+ Uint32 RnoFree = cnoFreeAttrbufrec;
+
+//-------------------------------------------------------------------------
+// As a prelude to the execution of the TUPKEYREQ we will copy the program
+// into the inBuffer to enable easy execution without any complex jumping
+// between the buffers. In particular this will make the interpreter less
+// complex. Hopefully it does also improve performance.
+//-------------------------------------------------------------------------
+ copyAttrBufPtr.i = regOperPtr->firstAttrinbufrec;
+ while (copyAttrBufPtr.i != RNIL) {
+ jam();
+ ndbrequire(copyAttrBufPtr.i < RnoOfAttrBufrec);
+ ptrAss(copyAttrBufPtr, attrbufrec);
+ RbufLen = copyAttrBufPtr.p->attrbuf[ZBUF_DATA_LEN];
+ Rnext = copyAttrBufPtr.p->attrbuf[ZBUF_NEXT];
+ Rfirst = cfirstfreeAttrbufrec;
+ MEMCOPY_NO_WORDS(&inBuffer[RinBufIndex],
+ &copyAttrBufPtr.p->attrbuf[0],
+ RbufLen);
+ RinBufIndex += RbufLen;
+ if (!TstoredProcedure) {
+ copyAttrBufPtr.p->attrbuf[ZBUF_NEXT] = Rfirst;
+ cfirstfreeAttrbufrec = copyAttrBufPtr.i;
+ RnoFree++;
+ }//if
+ copyAttrBufPtr.i = Rnext;
+ }//while
+ cnoFreeAttrbufrec = RnoFree;
+ if (TstoredProcedure) {
+ jam();
+ StoredProcPtr storedPtr;
+ c_storedProcPool.getPtr(storedPtr, (Uint32)regOperPtr->storedProcedureId);
+ ndbrequire(storedPtr.p->storedCode == ZSCAN_PROCEDURE);
+ storedPtr.p->storedCounter--;
+ regOperPtr->storedProcedureId = ZNIL;
+ }//if
+ // Release the ATTRINFO buffers
+ regOperPtr->firstAttrinbufrec = RNIL;
+ regOperPtr->lastAttrinbufrec = RNIL;
+}//Dbtup::copyAttrinfo()
+
+void Dbtup::handleATTRINFOforTUPKEYREQ(Signal* signal,
+ Uint32 length,
+ Operationrec * const regOperPtr)
+{
+ AttrbufrecPtr TAttrinbufptr;
+ TAttrinbufptr.i = cfirstfreeAttrbufrec;
+ if ((cfirstfreeAttrbufrec < cnoOfAttrbufrec) &&
+ (cnoFreeAttrbufrec > MIN_ATTRBUF)) {
+ ptrAss(TAttrinbufptr, attrbufrec);
+ MEMCOPY_NO_WORDS(&TAttrinbufptr.p->attrbuf[0],
+ &signal->theData[3],
+ length);
+ Uint32 RnoFree = cnoFreeAttrbufrec;
+ Uint32 Rnext = TAttrinbufptr.p->attrbuf[ZBUF_NEXT];
+ TAttrinbufptr.p->attrbuf[ZBUF_DATA_LEN] = length;
+ TAttrinbufptr.p->attrbuf[ZBUF_NEXT] = RNIL;
+
+ AttrbufrecPtr locAttrinbufptr;
+ Uint32 RnewLen = regOperPtr->currentAttrinbufLen;
+
+ locAttrinbufptr.i = regOperPtr->lastAttrinbufrec;
+ cfirstfreeAttrbufrec = Rnext;
+ cnoFreeAttrbufrec = RnoFree - 1;
+ RnewLen += length;
+ regOperPtr->lastAttrinbufrec = TAttrinbufptr.i;
+ regOperPtr->currentAttrinbufLen = RnewLen;
+ if (locAttrinbufptr.i == RNIL) {
+ regOperPtr->firstAttrinbufrec = TAttrinbufptr.i;
+ return;
+ } else {
+ jam();
+ ptrCheckGuard(locAttrinbufptr, cnoOfAttrbufrec, attrbufrec);
+ locAttrinbufptr.p->attrbuf[ZBUF_NEXT] = TAttrinbufptr.i;
+ }//if
+ if (RnewLen < ZATTR_BUFFER_SIZE) {
+ return;
+ } else {
+ jam();
+ regOperPtr->transstate = TOO_MUCH_AI;
+ return;
+ }//if
+ } else if (cnoFreeAttrbufrec <= MIN_ATTRBUF) {
+ jam();
+ regOperPtr->transstate = ERROR_WAIT_TUPKEYREQ;
+ } else {
+ ndbrequire(false);
+ }//if
+}//Dbtup::handleATTRINFOforTUPKEYREQ()
+
+void Dbtup::execATTRINFO(Signal* signal)
+{
+ OperationrecPtr regOpPtr;
+ Uint32 Rsig0 = signal->theData[0];
+ Uint32 Rlen = signal->length();
+ regOpPtr.i = Rsig0;
+
+ jamEntry();
+
+ ptrCheckGuard(regOpPtr, cnoOfOprec, operationrec);
+ if (regOpPtr.p->transstate == IDLE) {
+ handleATTRINFOforTUPKEYREQ(signal, Rlen - 3, regOpPtr.p);
+ return;
+ } else if (regOpPtr.p->transstate == WAIT_STORED_PROCEDURE_ATTR_INFO) {
+ storedProcedureAttrInfo(signal, regOpPtr.p, Rlen - 3, 3, false);
+ return;
+ }//if
+ switch (regOpPtr.p->transstate) {
+ case ERROR_WAIT_STORED_PROCREQ:
+ jam();
+ case TOO_MUCH_AI:
+ jam();
+ case ERROR_WAIT_TUPKEYREQ:
+ jam();
+ return; /* IGNORE ATTRINFO IN THOSE STATES, WAITING FOR ABORT SIGNAL */
+ break;
+ case DISCONNECTED:
+ jam();
+ case STARTED:
+ jam();
+ default:
+ ndbrequire(false);
+ }//switch
+}//Dbtup::execATTRINFO()
+
+void Dbtup::execTUP_ALLOCREQ(Signal* signal)
+{
+ OperationrecPtr regOperPtr;
+ TablerecPtr regTabPtr;
+ FragrecordPtr regFragPtr;
+
+ jamEntry();
+
+ regOperPtr.i = signal->theData[0];
+ regFragPtr.i = signal->theData[1];
+ regTabPtr.i = signal->theData[2];
+
+ if (!((regOperPtr.i < cnoOfOprec) &&
+ (regFragPtr.i < cnoOfFragrec) &&
+ (regTabPtr.i < cnoOfTablerec))) {
+ ndbrequire(false);
+ }//if
+ ptrAss(regOperPtr, operationrec);
+ ptrAss(regFragPtr, fragrecord);
+ ptrAss(regTabPtr, tablerec);
+
+//---------------------------------------------------
+/* --- Allocate a tuple as requested by ACC --- */
+//---------------------------------------------------
+ PagePtr pagePtr;
+ Uint32 pageOffset;
+ if (!allocTh(regFragPtr.p,
+ regTabPtr.p,
+ NORMAL_PAGE,
+ signal,
+ pageOffset,
+ pagePtr)) {
+ signal->theData[0] = terrorCode; // Indicate failure
+ return;
+ }//if
+ Uint32 fragPageId = pagePtr.p->pageWord[ZPAGE_FRAG_PAGE_ID_POS];
+ Uint32 pageIndex = ((pageOffset - ZPAGE_HEADER_SIZE) /
+ regTabPtr.p->tupheadsize) << 1;
+ regOperPtr.p->tableRef = regTabPtr.i;
+ regOperPtr.p->fragId = regFragPtr.p->fragmentId;
+ regOperPtr.p->realPageId = pagePtr.i;
+ regOperPtr.p->fragPageId = fragPageId;
+ regOperPtr.p->pageOffset = pageOffset;
+ regOperPtr.p->pageIndex = pageIndex;
+ /* -------------------------------------------------------------- */
+ /* AN INSERT IS UNDONE BY FREEING THE DATA OCCUPIED BY THE INSERT */
+ /* THE ONLY DATA WE HAVE TO LOG EXCEPT THE TYPE, PAGE AND INDEX */
+ /* IS THE AMOUNT OF DATA TO FREE */
+ /* -------------------------------------------------------------- */
+ if (isUndoLoggingNeeded(regFragPtr.p, fragPageId)) {
+ jam();
+ cprAddUndoLogRecord(signal,
+ ZLCPR_TYPE_DELETE_TH,
+ fragPageId,
+ pageIndex,
+ regTabPtr.i,
+ regFragPtr.p->fragmentId,
+ regFragPtr.p->checkpointVersion);
+ }//if
+
+ //---------------------------------------------------------------
+ // Initialise Active operation list by setting the list to empty
+ //---------------------------------------------------------------
+ ndbrequire(pageOffset < ZWORDS_ON_PAGE);
+ pagePtr.p->pageWord[pageOffset] = RNIL;
+
+ signal->theData[0] = 0;
+ signal->theData[1] = fragPageId;
+ signal->theData[2] = pageIndex;
+}//Dbtup::execTUP_ALLOCREQ()
+
+void
+Dbtup::setChecksum(Page* const pagePtr, Uint32 tupHeadOffset, Uint32 tupHeadSize)
+{
+ // 2 == regTabPtr.p->tupChecksumIndex
+ pagePtr->pageWord[tupHeadOffset + 2] = 0;
+ Uint32 checksum = calculateChecksum(pagePtr, tupHeadOffset, tupHeadSize);
+ pagePtr->pageWord[tupHeadOffset + 2] = checksum;
+}//Dbtup::setChecksum()
+
+Uint32
+Dbtup::calculateChecksum(Page* pagePtr,
+ Uint32 tupHeadOffset,
+ Uint32 tupHeadSize)
+{
+ Uint32 checksum = 0;
+ Uint32 loopStop = tupHeadOffset + tupHeadSize;
+ ndbrequire(loopStop <= ZWORDS_ON_PAGE);
+ // includes tupVersion
+ for (Uint32 i = tupHeadOffset + 1; i < loopStop; i++) {
+ checksum ^= pagePtr->pageWord[i];
+ }//if
+ return checksum;
+}//Dbtup::calculateChecksum()
+
+/* ----------------------------------------------------------------- */
+/* ----------- INSERT_ACTIVE_OP_LIST -------------- */
+/* ----------------------------------------------------------------- */
+void Dbtup::insertActiveOpList(Signal* signal,
+ OperationrecPtr regOperPtr,
+ Page* const pagePtr,
+ Uint32 pageOffset)
+{
+ OperationrecPtr iaoPrevOpPtr;
+ ndbrequire(regOperPtr.p->inActiveOpList == ZFALSE);
+ regOperPtr.p->inActiveOpList = ZTRUE;
+ ndbrequire(pageOffset < ZWORDS_ON_PAGE);
+ iaoPrevOpPtr.i = pagePtr->pageWord[pageOffset];
+ pagePtr->pageWord[pageOffset] = regOperPtr.i;
+ regOperPtr.p->prevActiveOp = RNIL;
+ regOperPtr.p->nextActiveOp = iaoPrevOpPtr.i;
+ if (iaoPrevOpPtr.i == RNIL) {
+ return;
+ } else {
+ jam();
+ ptrCheckGuard(iaoPrevOpPtr, cnoOfOprec, operationrec);
+ iaoPrevOpPtr.p->prevActiveOp = regOperPtr.i;
+ if (iaoPrevOpPtr.p->optype == ZDELETE &&
+ regOperPtr.p->optype == ZINSERT) {
+ jam();
+ // mark both
+ iaoPrevOpPtr.p->deleteInsertFlag = 1;
+ regOperPtr.p->deleteInsertFlag = 1;
+ }
+ return;
+ }//if
+}//Dbtup::insertActiveOpList()
+
+void Dbtup::linkOpIntoFragList(OperationrecPtr regOperPtr,
+ Fragrecord* const regFragPtr)
+{
+ OperationrecPtr sopTmpOperPtr;
+/* ----------------------------------------------------------------- */
+/* LINK THE OPERATION INTO A DOUBLY LINKED LIST ON THE FRAGMENT*/
+/* PUT IT FIRST IN THIS LIST SINCE IT DOESN'T MATTER WHERE IT */
+/* IS PUT. */
+/* ----------------------------------------------------------------- */
+ ndbrequire(regOperPtr.p->inFragList == ZFALSE);
+ regOperPtr.p->inFragList = ZTRUE;
+ regOperPtr.p->prevOprecInList = RNIL;
+ sopTmpOperPtr.i = regFragPtr->firstusedOprec;
+ regFragPtr->firstusedOprec = regOperPtr.i;
+ regOperPtr.p->nextOprecInList = sopTmpOperPtr.i;
+ if (sopTmpOperPtr.i == RNIL) {
+ return;
+ } else {
+ jam();
+ ptrCheckGuard(sopTmpOperPtr, cnoOfOprec, operationrec);
+ sopTmpOperPtr.p->prevOprecInList = regOperPtr.i;
+ }//if
+}//Dbtup::linkOpIntoFragList()
+
+/*
+This routine is optimised for use from TUPKEYREQ.
+This means that a lot of input data is stored in the operation record.
+The routine expects the following data in the operation record to be
+set-up properly.
+Transaction data
+1) transid1
+2) transid2
+3) savePointId
+
+Operation data
+4) optype
+5) dirtyOp
+
+Tuple address
+6) fragPageId
+7) pageIndex
+
+regFragPtr and regTabPtr are references to the table and fragment data and
+is read-only.
+
+The routine will set up the following data in the operation record if
+returned with success.
+
+Tuple address data
+1) realPageId
+2) fragPageId
+3) pageOffset
+4) pageIndex
+
+Also the pagePtr is an output variable if the routine returns with success.
+It's input value can be undefined.
+*/
+bool
+Dbtup::getPage(PagePtr& pagePtr,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr)
+{
+/* ------------------------------------------------------------------------- */
+// GET THE REFERENCE TO THE TUPLE HEADER BY TRANSLATING THE FRAGMENT PAGE ID
+// INTO A REAL PAGE ID AND BY USING THE PAGE INDEX TO DERIVE THE PROPER INDEX
+// IN THE REAL PAGE.
+/* ------------------------------------------------------------------------- */
+ pagePtr.i = getRealpid(regFragPtr, regOperPtr->fragPageId);
+ regOperPtr->realPageId = pagePtr.i;
+ Uint32 RpageIndex = regOperPtr->pageIndex;
+ Uint32 Rtupheadsize = regTabPtr->tupheadsize;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 RpageIndexScaled = RpageIndex >> 1;
+ ndbrequire((RpageIndex & 1) == 0);
+ regOperPtr->pageOffset = ZPAGE_HEADER_SIZE +
+ (Rtupheadsize * RpageIndexScaled);
+
+ OperationrecPtr leaderOpPtr;
+ ndbrequire(regOperPtr->pageOffset < ZWORDS_ON_PAGE);
+ leaderOpPtr.i = pagePtr.p->pageWord[regOperPtr->pageOffset];
+ if (leaderOpPtr.i == RNIL) {
+ return true;
+ }//if
+ ptrCheckGuard(leaderOpPtr, cnoOfOprec, operationrec);
+ bool dirtyRead = ((regOperPtr->optype == ZREAD) &&
+ (regOperPtr->dirtyOp == 1));
+ if (dirtyRead) {
+ bool sameTrans = ((regOperPtr->transid1 == leaderOpPtr.p->transid1) &&
+ (regOperPtr->transid2 == leaderOpPtr.p->transid2));
+ if (!sameTrans) {
+ if (!getPageLastCommitted(regOperPtr, leaderOpPtr.p)) {
+ return false;
+ }//if
+ pagePtr.i = regOperPtr->realPageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ return true;
+ }//if
+ }//if
+ if (regOperPtr->optype == ZREAD) {
+ /*
+ Read uses savepoint id's to find the correct tuple version.
+ */
+ if (getPageThroughSavePoint(regOperPtr, leaderOpPtr.p)) {
+ jam();
+ pagePtr.i = regOperPtr->realPageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ return true;
+ }
+ return false;
+ }
+//----------------------------------------------------------------------
+// Check that no other operation is already active on the tuple. Also
+// that abort or commit is not ongoing.
+//----------------------------------------------------------------------
+ if (leaderOpPtr.p->tupleState == NO_OTHER_OP) {
+ jam();
+ if ((leaderOpPtr.p->optype == ZDELETE) &&
+ (regOperPtr->optype != ZINSERT)) {
+ jam();
+ terrorCode = ZTUPLE_DELETED_ERROR;
+ return false;
+ }//if
+ return true;
+ } else if (leaderOpPtr.p->tupleState == ALREADY_ABORTED) {
+ jam();
+ terrorCode = ZMUST_BE_ABORTED_ERROR;
+ return false;
+ } else {
+ ndbrequire(false);
+ }//if
+ return true;
+}//Dbtup::getPage()
+
+bool
+Dbtup::getPageThroughSavePoint(Operationrec* regOperPtr,
+ Operationrec* leaderOpPtr)
+{
+ bool found = false;
+ OperationrecPtr loopOpPtr;
+ loopOpPtr.p = leaderOpPtr;
+ while(true) {
+ if (regOperPtr->savePointId > loopOpPtr.p->savePointId) {
+ jam();
+ found = true;
+ break;
+ }
+ if (loopOpPtr.p->nextActiveOp == RNIL) {
+ break;
+ }
+ loopOpPtr.i = loopOpPtr.p->nextActiveOp;
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ jam();
+ }
+ if (!found) {
+ return getPageLastCommitted(regOperPtr, loopOpPtr.p);
+ } else {
+ if (loopOpPtr.p->optype == ZDELETE) {
+ jam();
+ terrorCode = ZTUPLE_DELETED_ERROR;
+ return false;
+ }
+ if (loopOpPtr.p->tupleState == ALREADY_ABORTED) {
+ /*
+ Requested tuple version has already been aborted
+ */
+ jam();
+ terrorCode = ZMUST_BE_ABORTED_ERROR;
+ return false;
+ }
+ bool use_copy;
+ if (loopOpPtr.p->prevActiveOp == RNIL) {
+ jam();
+ /*
+ Use original tuple since we are reading from the last written tuple.
+ We are the
+ */
+ use_copy = false;
+ } else {
+ /*
+ Go forward in time to find a copy of the tuple which this operation
+ produced
+ */
+ loopOpPtr.i = loopOpPtr.p->prevActiveOp;
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ if (loopOpPtr.p->optype == ZDELETE) {
+ /*
+ This operation was a Delete and thus have no copy tuple attached to
+ it. We will move forward to the next that either doesn't exist in
+ which case we will return the original tuple of any operation and
+ otherwise it must be an insert which contains a copy record.
+ */
+ if (loopOpPtr.p->prevActiveOp == RNIL) {
+ jam();
+ use_copy = false;
+ } else {
+ jam();
+ loopOpPtr.i = loopOpPtr.p->prevActiveOp;
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ ndbrequire(loopOpPtr.p->optype == ZINSERT);
+ use_copy = true;
+ }
+ } else if (loopOpPtr.p->optype == ZUPDATE) {
+ jam();
+ /*
+ This operation which was the next in time have a copy which was the
+ result of the previous operation which we want to use. Thus use
+ the copy tuple of this operation.
+ */
+ use_copy = true;
+ } else {
+ /*
+ This operation was an insert that happened after an insert or update.
+ This is not a possible case.
+ */
+ ndbrequire(false);
+ return false;
+ }
+ }
+ if (use_copy) {
+ regOperPtr->realPageId = loopOpPtr.p->realPageIdC;
+ regOperPtr->fragPageId = loopOpPtr.p->fragPageIdC;
+ regOperPtr->pageIndex = loopOpPtr.p->pageIndexC;
+ regOperPtr->pageOffset = loopOpPtr.p->pageOffsetC;
+ } else {
+ regOperPtr->realPageId = loopOpPtr.p->realPageId;
+ regOperPtr->fragPageId = loopOpPtr.p->fragPageId;
+ regOperPtr->pageIndex = loopOpPtr.p->pageIndex;
+ regOperPtr->pageOffset = loopOpPtr.p->pageOffset;
+ }
+ return true;
+ }
+}
+
+bool
+Dbtup::getPageLastCommitted(Operationrec* const regOperPtr,
+ Operationrec* const leaderOpPtr)
+{
+//----------------------------------------------------------------------
+// Dirty reads wants to read the latest committed tuple. The latest
+// tuple value could be not existing or else we have to find the copy
+// tuple. Start by finding the end of the list to find the first operation
+// on the record in the ongoing transaction.
+//----------------------------------------------------------------------
+ jam();
+ OperationrecPtr loopOpPtr;
+ loopOpPtr.p = leaderOpPtr;
+ while (loopOpPtr.p->nextActiveOp != RNIL) {
+ jam();
+ loopOpPtr.i = loopOpPtr.p->nextActiveOp;
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ }//while
+ if (loopOpPtr.p->optype == ZINSERT) {
+ jam();
+//----------------------------------------------------------------------
+// With an insert in the start of the list we know that the tuple did not
+// exist before this transaction was started. We don't care if the current
+// transaction is in the commit phase since the commit is not really
+// completed until the operation is gone from TUP.
+//----------------------------------------------------------------------
+ terrorCode = ZTUPLE_DELETED_ERROR;
+ return false;
+ } else {
+//----------------------------------------------------------------------
+// A successful update and delete as first in the queue means that a tuple
+// exist in the committed world. We need to find it.
+//----------------------------------------------------------------------
+ if (loopOpPtr.p->optype == ZUPDATE) {
+ jam();
+//----------------------------------------------------------------------
+// The first operation was a delete we set our tuple reference to the
+// copy tuple of this operation.
+//----------------------------------------------------------------------
+ regOperPtr->realPageId = loopOpPtr.p->realPageIdC;
+ regOperPtr->fragPageId = loopOpPtr.p->fragPageIdC;
+ regOperPtr->pageIndex = loopOpPtr.p->pageIndexC;
+ regOperPtr->pageOffset = loopOpPtr.p->pageOffsetC;
+ } else if ((loopOpPtr.p->optype == ZDELETE) &&
+ (loopOpPtr.p->prevActiveOp == RNIL)) {
+ jam();
+//----------------------------------------------------------------------
+// There was only a delete. The original tuple still is ok.
+//----------------------------------------------------------------------
+ } else {
+ jam();
+//----------------------------------------------------------------------
+// There was another operation after the delete, this must be an insert
+// and we have found our copy tuple there.
+//----------------------------------------------------------------------
+ loopOpPtr.i = loopOpPtr.p->prevActiveOp;
+ ptrCheckGuard(loopOpPtr, cnoOfOprec, operationrec);
+ ndbrequire(loopOpPtr.p->optype == ZINSERT);
+ regOperPtr->realPageId = loopOpPtr.p->realPageIdC;
+ regOperPtr->fragPageId = loopOpPtr.p->fragPageIdC;
+ regOperPtr->pageIndex = loopOpPtr.p->pageIndexC;
+ regOperPtr->pageOffset = loopOpPtr.p->pageOffsetC;
+ }//if
+ }//if
+ return true;
+}//Dbtup::getPageLastCommitted()
+
+void Dbtup::execTUPKEYREQ(Signal* signal)
+{
+ TupKeyReq * const tupKeyReq = (TupKeyReq *)signal->getDataPtr();
+ Uint32 RoperPtr = tupKeyReq->connectPtr;
+ Uint32 Rtabptr = tupKeyReq->tableRef;
+ Uint32 RfragId = tupKeyReq->fragId;
+ Uint32 Rstoredid = tupKeyReq->storedProcedure;
+ Uint32 Rfragptr = tupKeyReq->fragPtr;
+
+ Uint32 RnoOfOprec = cnoOfOprec;
+ Uint32 RnoOfTablerec = cnoOfTablerec;
+ Uint32 RnoOfFragrec = cnoOfFragrec;
+
+ operPtr.i = RoperPtr;
+ fragptr.i = Rfragptr;
+ tabptr.i = Rtabptr;
+ jamEntry();
+
+ ndbrequire(((RoperPtr < RnoOfOprec) &&
+ (Rtabptr < RnoOfTablerec) &&
+ (Rfragptr < RnoOfFragrec)));
+ ptrAss(operPtr, operationrec);
+ Operationrec * const regOperPtr = operPtr.p;
+ ptrAss(fragptr, fragrecord);
+ Fragrecord * const regFragPtr = fragptr.p;
+ ptrAss(tabptr, tablerec);
+ Tablerec* const regTabPtr = tabptr.p;
+
+ Uint32 TrequestInfo = tupKeyReq->request;
+
+ if (regOperPtr->transstate != IDLE) {
+ TUPKEY_abort(signal, 39);
+ return;
+ }//if
+/* ----------------------------------------------------------------- */
+// Operation is ZREAD when we arrive here so no need to worry about the
+// abort process.
+/* ----------------------------------------------------------------- */
+/* ----------- INITIATE THE OPERATION RECORD -------------- */
+/* ----------------------------------------------------------------- */
+ regOperPtr->fragmentPtr = Rfragptr;
+ regOperPtr->dirtyOp = TrequestInfo & 1;
+ regOperPtr->opSimple = (TrequestInfo >> 1) & 1;
+ regOperPtr->interpretedExec = (TrequestInfo >> 10) & 1;
+ regOperPtr->optype = (TrequestInfo >> 6) & 0xf;
+
+ // Attributes needed by trigger execution
+ regOperPtr->noFiredTriggers = 0;
+ regOperPtr->tableRef = Rtabptr;
+ regOperPtr->tcOperationPtr = tupKeyReq->opRef;
+ regOperPtr->primaryReplica = tupKeyReq->primaryReplica;
+ regOperPtr->coordinatorTC = tupKeyReq->coordinatorTC;
+ regOperPtr->tcOpIndex = tupKeyReq->tcOpIndex;
+ regOperPtr->savePointId = tupKeyReq->savePointId;
+
+ regOperPtr->fragId = RfragId;
+
+ regOperPtr->fragPageId = tupKeyReq->keyRef1;
+ regOperPtr->pageIndex = tupKeyReq->keyRef2;
+ regOperPtr->attrinbufLen = regOperPtr->logSize = tupKeyReq->attrBufLen;
+ regOperPtr->recBlockref = tupKeyReq->applRef;
+
+// Schema Version in tupKeyReq->schemaVersion not used in this version
+ regOperPtr->storedProcedureId = Rstoredid;
+ regOperPtr->transid1 = tupKeyReq->transId1;
+ regOperPtr->transid2 = tupKeyReq->transId2;
+
+ regOperPtr->attroutbufLen = 0;
+/* ----------------------------------------------------------------------- */
+// INITIALISE TO DEFAULT VALUE
+// INIT THE COPY REFERENCE RECORDS TO RNIL TO ENSURE THAT THEIR VALUES
+// ARE VALID IF THEY EXISTS
+// NO PENDING CHECKPOINT WHEN COPY CREATED (DEFAULT)
+// NO TUPLE HAS BEEN ALLOCATED YET
+// NO COPY HAS BEEN CREATED YET
+/* ----------------------------------------------------------------------- */
+ regOperPtr->undoLogged = false;
+ regOperPtr->realPageId = RNIL;
+ regOperPtr->realPageIdC = RNIL;
+ regOperPtr->fragPageIdC = RNIL;
+
+ regOperPtr->pageOffset = ZNIL;
+ regOperPtr->pageOffsetC = ZNIL;
+
+ regOperPtr->pageIndexC = ZNIL;
+
+ // version not yet known
+ regOperPtr->tupVersion = ZNIL;
+ regOperPtr->deleteInsertFlag = 0;
+
+ regOperPtr->tupleState = TUPLE_BLOCKED;
+ regOperPtr->changeMask.clear();
+
+ if (Rstoredid != ZNIL) {
+ ndbrequire(initStoredOperationrec(regOperPtr, Rstoredid) == ZOK);
+ }//if
+ copyAttrinfo(signal, regOperPtr, &cinBuffer[0]);
+
+ PagePtr pagePtr;
+ if (!getPage(pagePtr, regOperPtr, regFragPtr, regTabPtr)) {
+ tupkeyErrorLab(signal);
+ return;
+ }//if
+
+ Uint32 Roptype = regOperPtr->optype;
+ if (Roptype == ZREAD) {
+ jam();
+ if (handleReadReq(signal, regOperPtr, regTabPtr, pagePtr.p) != -1) {
+ sendTUPKEYCONF(signal, regOperPtr, 0);
+/* ------------------------------------------------------------------------- */
+// Read Operations need not to be taken out of any lists. We also do not
+// need to wait for commit since there is no changes to commit. Thus we
+// prepare the operation record already now for the next operation.
+// Write operations have set the state to STARTED above indicating that
+// they are waiting for the Commit or Abort decision.
+/* ------------------------------------------------------------------------- */
+ regOperPtr->transstate = IDLE;
+ regOperPtr->currentAttrinbufLen = 0;
+ }//if
+ return;
+ }//if
+ linkOpIntoFragList(operPtr, regFragPtr);
+ insertActiveOpList(signal,
+ operPtr,
+ pagePtr.p,
+ regOperPtr->pageOffset);
+ if (isUndoLoggingBlocked(regFragPtr)) {
+ TUPKEY_abort(signal, 38);
+ return;
+ }//if
+/* ---------------------------------------------------------------------- */
+// WE SET THE CURRENT ACTIVE OPERATION IN THE TUPLE TO POINT TO OUR
+//OPERATION RECORD. IF SEVERAL OPERATIONS WORK ON THIS TUPLE THEY ARE
+// LINKED TO OUR OPERATION RECORD. DIRTY READS CAN ACCESS THE COPY
+// TUPLE THROUGH OUR OPERATION RECORD.
+/* ---------------------------------------------------------------------- */
+ if (Roptype == ZINSERT) {
+ jam();
+ if (handleInsertReq(signal, regOperPtr,
+ regFragPtr, regTabPtr, pagePtr.p) == -1) {
+ return;
+ }//if
+ if (!regTabPtr->tuxCustomTriggers.isEmpty()) {
+ jam();
+ if (executeTuxInsertTriggers(signal, regOperPtr, regTabPtr) != 0) {
+ jam();
+ tupkeyErrorLab(signal);
+ return;
+ }
+ }
+ checkImmediateTriggersAfterInsert(signal,
+ regOperPtr,
+ regTabPtr);
+ sendTUPKEYCONF(signal, regOperPtr, regOperPtr->logSize);
+ return;
+ }//if
+ if (regTabPtr->checksumIndicator &&
+ (calculateChecksum(pagePtr.p,
+ regOperPtr->pageOffset,
+ regTabPtr->tupheadsize) != 0)) {
+ jam();
+ terrorCode = ZTUPLE_CORRUPTED_ERROR;
+ tupkeyErrorLab(signal);
+ return;
+ }//if
+ if (Roptype == ZUPDATE) {
+ jam();
+ if (handleUpdateReq(signal, regOperPtr,
+ regFragPtr, regTabPtr, pagePtr.p) == -1) {
+ return;
+ }//if
+ // If update operation is done on primary,
+ // check any after op triggers
+ terrorCode = 0;
+ if (!regTabPtr->tuxCustomTriggers.isEmpty()) {
+ jam();
+ if (executeTuxUpdateTriggers(signal, regOperPtr, regTabPtr) != 0) {
+ jam();
+ tupkeyErrorLab(signal);
+ return;
+ }
+ }
+ checkImmediateTriggersAfterUpdate(signal,
+ regOperPtr,
+ regTabPtr);
+ // XXX use terrorCode for now since all methods are void
+ if (terrorCode != 0) {
+ tupkeyErrorLab(signal);
+ return;
+ }
+ sendTUPKEYCONF(signal, regOperPtr, regOperPtr->logSize);
+ return;
+ } else if (Roptype == ZDELETE) {
+ jam();
+ if (handleDeleteReq(signal, regOperPtr,
+ regFragPtr, regTabPtr, pagePtr.p) == -1) {
+ return;
+ }//if
+ // If delete operation is done on primary,
+ // check any after op triggers
+ if (!regTabPtr->tuxCustomTriggers.isEmpty()) {
+ jam();
+ if (executeTuxDeleteTriggers(signal, regOperPtr, regTabPtr) != 0) {
+ jam();
+ tupkeyErrorLab(signal);
+ return;
+ }
+ }
+ checkImmediateTriggersAfterDelete(signal,
+ regOperPtr,
+ regTabPtr);
+ sendTUPKEYCONF(signal, regOperPtr, 0);
+ return;
+ } else {
+ ndbrequire(false);
+ }//if
+}//Dbtup::execTUPKEYREQ()
+
+/* ---------------------------------------------------------------- */
+/* ------------------------ CONFIRM REQUEST ----------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::sendTUPKEYCONF(Signal* signal,
+ Operationrec * const regOperPtr,
+ Uint32 TlogSize)
+{
+ TupKeyConf * const tupKeyConf = (TupKeyConf *)signal->getDataPtrSend();
+
+ Uint32 RuserPointer = regOperPtr->userpointer;
+ Uint32 RfragPageId = regOperPtr->fragPageId;
+ Uint32 RpageIndex = regOperPtr->pageIndex;
+ Uint32 RattroutbufLen = regOperPtr->attroutbufLen;
+ Uint32 RnoFiredTriggers = regOperPtr->noFiredTriggers;
+ BlockReference Ruserblockref = regOperPtr->userblockref;
+
+ regOperPtr->transstate = STARTED;
+ regOperPtr->tupleState = NO_OTHER_OP;
+ tupKeyConf->userPtr = RuserPointer;
+ tupKeyConf->pageId = RfragPageId;
+ tupKeyConf->pageIndex = RpageIndex;
+ tupKeyConf->readLength = RattroutbufLen;
+ tupKeyConf->writeLength = TlogSize;
+ tupKeyConf->noFiredTriggers = RnoFiredTriggers;
+
+ EXECUTE_DIRECT(refToBlock(Ruserblockref), GSN_TUPKEYCONF, signal,
+ TupKeyConf::SignalLength);
+ return;
+}//Dbtup::sendTUPKEYCONF()
+
+/* ---------------------------------------------------------------- */
+/* ----------------------------- READ ---------------------------- */
+/* ---------------------------------------------------------------- */
+int Dbtup::handleReadReq(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr,
+ Page* pagePtr)
+{
+ Uint32 Ttupheadoffset = regOperPtr->pageOffset;
+ if (regTabPtr->checksumIndicator &&
+ (calculateChecksum(pagePtr, Ttupheadoffset,
+ regTabPtr->tupheadsize) != 0)) {
+ jam();
+ terrorCode = ZTUPLE_CORRUPTED_ERROR;
+ tupkeyErrorLab(signal);
+ return -1;
+ }//if
+
+ if (regOperPtr->interpretedExec != 1) {
+ jam();
+ Uint32 TnoOfDataRead = readAttributes(pagePtr,
+ Ttupheadoffset,
+ &cinBuffer[0],
+ regOperPtr->attrinbufLen,
+ &coutBuffer[0],
+ (Uint32)ZATTR_BUFFER_SIZE);
+ if (TnoOfDataRead != (Uint32)-1) {
+/* ------------------------------------------------------------------------- */
+// We have read all data into coutBuffer. Now send it to the API.
+/* ------------------------------------------------------------------------- */
+ jam();
+ regOperPtr->attroutbufLen = TnoOfDataRead;
+ sendReadAttrinfo(signal, TnoOfDataRead, regOperPtr);
+ return 0;
+ }//if
+ jam();
+ tupkeyErrorLab(signal);
+ return -1;
+ } else {
+ jam();
+ if (interpreterStartLab(signal, pagePtr, Ttupheadoffset) != -1) {
+ return 0;
+ }//if
+ return -1;
+ }//if
+}//Dbtup::handleReadReq()
+
+/* ---------------------------------------------------------------- */
+/* ---------------------------- UPDATE ---------------------------- */
+/* ---------------------------------------------------------------- */
+int Dbtup::handleUpdateReq(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Page* const pagePtr)
+{
+ PagePtr copyPagePtr;
+ Uint32 tuple_size = regTabPtr->tupheadsize;
+
+//---------------------------------------------------
+/* --- MAKE A COPY OF THIS TUPLE ON A COPY PAGE --- */
+//---------------------------------------------------
+ Uint32 RpageOffsetC;
+ if (!allocTh(regFragPtr,
+ regTabPtr,
+ COPY_PAGE,
+ signal,
+ RpageOffsetC,
+ copyPagePtr)) {
+ TUPKEY_abort(signal, 1);
+ return -1;
+ }//if
+ Uint32 RpageIdC = copyPagePtr.i;
+ Uint32 RfragPageIdC = copyPagePtr.p->pageWord[ZPAGE_FRAG_PAGE_ID_POS];
+ Uint32 indexC = ((RpageOffsetC - ZPAGE_HEADER_SIZE) / tuple_size) << 1;
+ regOperPtr->pageIndexC = indexC;
+ regOperPtr->fragPageIdC = RfragPageIdC;
+ regOperPtr->realPageIdC = RpageIdC;
+ regOperPtr->pageOffsetC = RpageOffsetC;
+ /* -------------------------------------------------------------- */
+ /* IF WE HAVE AN ONGING CHECKPOINT WE HAVE TO LOG THE ALLOCATION */
+ /* OF THE TUPLE HEADER TO BE ABLE TO DELETE IT UPON RESTART */
+ /* THE ONLY DATA EXCEPT THE TYPE, PAGE, INDEX IS THE SIZE TO FREE */
+ /* -------------------------------------------------------------- */
+ if (isUndoLoggingActive(regFragPtr)) {
+ if (isPageUndoLogged(regFragPtr, RfragPageIdC)) {
+ jam();
+ regOperPtr->undoLogged = true;
+ cprAddUndoLogRecord(signal,
+ ZLCPR_TYPE_DELETE_TH,
+ RfragPageIdC,
+ indexC,
+ regOperPtr->tableRef,
+ regOperPtr->fragId,
+ regFragPtr->checkpointVersion);
+ }//if
+ if (isPageUndoLogged(regFragPtr, regOperPtr->fragPageId)) {
+ jam();
+ cprAddUndoLogRecord(signal,
+ ZLCPR_TYPE_UPDATE_TH,
+ regOperPtr->fragPageId,
+ regOperPtr->pageIndex,
+ regOperPtr->tableRef,
+ regOperPtr->fragId,
+ regFragPtr->checkpointVersion);
+ cprAddData(signal,
+ regFragPtr,
+ regOperPtr->realPageId,
+ tuple_size,
+ regOperPtr->pageOffset);
+ }//if
+ }//if
+ Uint32 RwordCount = tuple_size - 1;
+ Uint32 end_dest = RpageOffsetC + tuple_size;
+ Uint32 offset = regOperPtr->pageOffset;
+ Uint32 end_source = offset + tuple_size;
+ ndbrequire(end_dest <= ZWORDS_ON_PAGE && end_source <= ZWORDS_ON_PAGE);
+ void* Tdestination = (void*)&copyPagePtr.p->pageWord[RpageOffsetC + 1];
+ const void* Tsource = (void*)&pagePtr->pageWord[offset + 1];
+ MEMCOPY_NO_WORDS(Tdestination, Tsource, RwordCount);
+
+ Uint32 prev_tup_version;
+ // nextActiveOp is before this op in event order
+ if (regOperPtr->nextActiveOp == RNIL) {
+ jam();
+ prev_tup_version = ((const Uint32*)Tsource)[0];
+ } else {
+ OperationrecPtr prevOperPtr;
+ jam();
+ prevOperPtr.i = regOperPtr->nextActiveOp;
+ ptrCheckGuard(prevOperPtr, cnoOfOprec, operationrec);
+ prev_tup_version = prevOperPtr.p->tupVersion;
+ }//if
+ regOperPtr->tupVersion = (prev_tup_version + 1) &
+ ((1 << ZTUP_VERSION_BITS) - 1);
+ // global variable alert
+ ndbassert(operationrec + operPtr.i == regOperPtr);
+ copyPagePtr.p->pageWord[RpageOffsetC] = operPtr.i;
+
+ return updateStartLab(signal, regOperPtr, regTabPtr, pagePtr);
+}//Dbtup::handleUpdateReq()
+
+/* ---------------------------------------------------------------- */
+/* ----------------------------- INSERT --------------------------- */
+/* ---------------------------------------------------------------- */
+int Dbtup::handleInsertReq(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Page* const pagePtr)
+{
+ Uint32 ret_value;
+
+ if (regOperPtr->nextActiveOp != RNIL) {
+ jam();
+ OperationrecPtr prevExecOpPtr;
+ prevExecOpPtr.i = regOperPtr->nextActiveOp;
+ ptrCheckGuard(prevExecOpPtr, cnoOfOprec, operationrec);
+ if (prevExecOpPtr.p->optype != ZDELETE) {
+ terrorCode = ZINSERT_ERROR;
+ tupkeyErrorLab(signal);
+ return -1;
+ }//if
+ ret_value = handleUpdateReq(signal, regOperPtr,
+ regFragPtr, regTabPtr, pagePtr);
+ } else {
+ jam();
+ regOperPtr->tupVersion = 0;
+ ret_value = updateStartLab(signal, regOperPtr, regTabPtr, pagePtr);
+ }//if
+ if (ret_value != (Uint32)-1) {
+ if (checkNullAttributes(regOperPtr, regTabPtr)) {
+ jam();
+ return 0;
+ }//if
+ TUPKEY_abort(signal, 17);
+ }//if
+ return -1;
+}//Dbtup::handleInsertReq()
+
+/* ---------------------------------------------------------------- */
+/* ---------------------------- DELETE ---------------------------- */
+/* ---------------------------------------------------------------- */
+int Dbtup::handleDeleteReq(Signal* signal,
+ Operationrec* const regOperPtr,
+ Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Page* const pagePtr)
+{
+ // delete must set but not increment tupVersion
+ if (regOperPtr->nextActiveOp != RNIL) {
+ OperationrecPtr prevExecOpPtr;
+ prevExecOpPtr.i = regOperPtr->nextActiveOp;
+ ptrCheckGuard(prevExecOpPtr, cnoOfOprec, operationrec);
+ regOperPtr->tupVersion = prevExecOpPtr.p->tupVersion;
+ } else {
+ jam();
+ regOperPtr->tupVersion = pagePtr->pageWord[regOperPtr->pageOffset + 1];
+ }
+ if (isUndoLoggingNeeded(regFragPtr, regOperPtr->fragPageId)) {
+ jam();
+ cprAddUndoLogRecord(signal,
+ ZINDICATE_NO_OP_ACTIVE,
+ regOperPtr->fragPageId,
+ regOperPtr->pageIndex,
+ regOperPtr->tableRef,
+ regOperPtr->fragId,
+ regFragPtr->checkpointVersion);
+ }//if
+ if (regOperPtr->attrinbufLen == 0) {
+ return 0;
+ }//if
+/* ------------------------------------------------------------------------ */
+/* THE APPLICATION WANTS TO READ THE TUPLE BEFORE IT IS DELETED. */
+/* ------------------------------------------------------------------------ */
+ return handleReadReq(signal, regOperPtr, regTabPtr, pagePtr);
+}//Dbtup::handleDeleteReq()
+
+int
+Dbtup::updateStartLab(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr,
+ Page* const pagePtr)
+{
+ Uint32 retValue;
+ if (regOperPtr->optype == ZINSERT) {
+ jam();
+ setNullBits(pagePtr, regTabPtr, regOperPtr->pageOffset);
+ }
+ if (regOperPtr->interpretedExec != 1) {
+ jam();
+ retValue = updateAttributes(pagePtr,
+ regOperPtr->pageOffset,
+ &cinBuffer[0],
+ regOperPtr->attrinbufLen);
+ if (retValue == (Uint32)-1) {
+ tupkeyErrorLab(signal);
+ }//if
+ } else {
+ jam();
+ retValue = interpreterStartLab(signal, pagePtr, regOperPtr->pageOffset);
+ }//if
+ ndbrequire(regOperPtr->tupVersion != ZNIL);
+ pagePtr->pageWord[regOperPtr->pageOffset + 1] = regOperPtr->tupVersion;
+ if (regTabPtr->checksumIndicator) {
+ jam();
+ setChecksum(pagePtr, regOperPtr->pageOffset, regTabPtr->tupheadsize);
+ }//if
+ return retValue;
+}//Dbtup::updateStartLab()
+
+void
+Dbtup::setNullBits(Page* const regPage, Tablerec* const regTabPtr, Uint32 pageOffset)
+{
+ Uint32 noOfExtraNullWords = regTabPtr->tupNullWords;
+ Uint32 nullOffsetStart = regTabPtr->tupNullIndex + pageOffset;
+ ndbrequire((noOfExtraNullWords + nullOffsetStart) < ZWORDS_ON_PAGE);
+ for (Uint32 i = 0; i < noOfExtraNullWords; i++) {
+ regPage->pageWord[nullOffsetStart + i] = 0xFFFFFFFF;
+ }//for
+}//Dbtup::setNullBits()
+
+bool
+Dbtup::checkNullAttributes(Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr)
+{
+// Implement checking of updating all not null attributes in an insert here.
+ Bitmask<MAXNROFATTRIBUTESINWORDS> attributeMask;
+ /*
+ * The idea here is maybe that changeMask is not-null attributes
+ * and must contain notNullAttributeMask. But:
+ *
+ * 1. changeMask has all bits set on insert
+ * 2. not-null is checked in each UpdateFunction
+ * 3. the code below does not work except trivially due to 1.
+ *
+ * XXX remove or fix
+ */
+ attributeMask.clear();
+ attributeMask.bitOR(regOperPtr->changeMask);
+ attributeMask.bitAND(regTabPtr->notNullAttributeMask);
+ attributeMask.bitXOR(regTabPtr->notNullAttributeMask);
+ if (!attributeMask.isclear()) {
+ return false;
+ }//if
+ return true;
+}//Dbtup::checkNullAttributes()
+
+/* ---------------------------------------------------------------- */
+/* THIS IS THE START OF THE INTERPRETED EXECUTION OF UPDATES. WE */
+/* START BY LINKING ALL ATTRINFO'S IN A DOUBLY LINKED LIST (THEY ARE*/
+/* ALREADY IN A LINKED LIST). WE ALLOCATE A REGISTER MEMORY (EQUAL */
+/* TO AN ATTRINFO RECORD). THE INTERPRETER GOES THROUGH FOUR PHASES*/
+/* DURING THE FIRST PHASE IT IS ONLY ALLOWED TO READ ATTRIBUTES THAT*/
+/* ARE SENT TO THE CLIENT APPLICATION. DURING THE SECOND PHASE IT IS*/
+/* ALLOWED TO READ FROM ATTRIBUTES INTO REGISTERS, TO UPDATE */
+/* ATTRIBUTES BASED ON EITHER A CONSTANT VALUE OR A REGISTER VALUE, */
+/* A DIVERSE SET OF OPERATIONS ON REGISTERS ARE AVAILABLE AS WELL. */
+/* IT IS ALSO POSSIBLE TO PERFORM JUMPS WITHIN THE INSTRUCTIONS THAT*/
+/* BELONGS TO THE SECOND PHASE. ALSO SUBROUTINES CAN BE CALLED IN */
+/* THIS PHASE. THE THIRD PHASE IS TO AGAIN READ ATTRIBUTES AND */
+/* FINALLY THE FOURTH PHASE READS SELECTED REGISTERS AND SEND THEM */
+/* TO THE CLIENT APPLICATION. */
+/* THERE IS A FIFTH REGION WHICH CONTAINS SUBROUTINES CALLABLE FROM */
+/* THE INTERPRETER EXECUTION REGION. */
+/* THE FIRST FIVE WORDS WILL GIVE THE LENGTH OF THE FIVEE REGIONS */
+/* */
+/* THIS MEANS THAT FROM THE APPLICATIONS POINT OF VIEW THE DATABASE */
+/* CAN HANDLE SUBROUTINE CALLS WHERE THE CODE IS SENT IN THE REQUEST*/
+/* THE RETURN PARAMETERS ARE FIXED AND CAN EITHER BE GENERATED */
+/* BEFORE THE EXECUTION OF THE ROUTINE OR AFTER. */
+/* */
+/* IN LATER VERSIONS WE WILL ADD MORE THINGS LIKE THE POSSIBILITY */
+/* TO ALLOCATE MEMORY AND USE THIS AS LOCAL STORAGE. IT IS ALSO */
+/* IMAGINABLE TO HAVE SPECIAL ROUTINES THAT CAN PERFORM CERTAIN */
+/* OPERATIONS ON BLOB'S DEPENDENT ON WHAT THE BLOB REPRESENTS. */
+/* */
+/* */
+/* ----------------------------------------- */
+/* + INITIAL READ REGION + */
+/* ----------------------------------------- */
+/* + INTERPRETED EXECUTE REGION + */
+/* ----------------------------------------- */
+/* + FINAL UPDATE REGION + */
+/* ----------------------------------------- */
+/* + FINAL READ REGION + */
+/* ----------------------------------------- */
+/* + SUBROUTINE REGION + */
+/* ----------------------------------------- */
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+/* ----------------- INTERPRETED EXECUTION ----------------------- */
+/* ---------------------------------------------------------------- */
+int Dbtup::interpreterStartLab(Signal* signal,
+ Page* const pagePtr,
+ Uint32 TupHeadOffset)
+{
+ Operationrec * const regOperPtr = operPtr.p;
+ Uint32 RtotalLen;
+ Uint32 TnoDataRW;
+
+ Uint32 RinitReadLen = cinBuffer[0];
+ Uint32 RexecRegionLen = cinBuffer[1];
+ Uint32 RfinalUpdateLen = cinBuffer[2];
+ Uint32 RfinalRLen = cinBuffer[3];
+ Uint32 RsubLen = cinBuffer[4];
+
+ Uint32 RattrinbufLen = regOperPtr->attrinbufLen;
+ const BlockReference sendBref = regOperPtr->recBlockref;
+
+ Uint32 * dst = &coutBuffer[0];
+ Uint32 dstLen = sizeof(coutBuffer) / 4;
+ Uint32 * tmp = &signal->theData[3];
+ Uint32 tmpLen = (sizeof(signal->theData) / 4) - 3;
+ bool executeDirect = false;
+ const Uint32 node = refToNode(sendBref);
+ if(node != 0 && node != getOwnNodeId()) {
+ ;
+ } else {
+ jam();
+ /**
+ * execute direct
+ */
+ executeDirect = true;
+ dst = &signal->theData[3];
+ dstLen = (sizeof(signal->theData) / 4) - 3;
+
+ tmp = &coutBuffer[0];
+ tmpLen = sizeof(coutBuffer) / 4;
+ }
+
+ RtotalLen = RinitReadLen;
+ RtotalLen += RexecRegionLen;
+ RtotalLen += RfinalUpdateLen;
+ RtotalLen += RfinalRLen;
+ RtotalLen += RsubLen;
+
+ Uint32 RattroutCounter = 0;
+ Uint32 RinstructionCounter = 5;
+ Uint32 RlogSize = 0;
+
+ if (((RtotalLen + 5) == RattrinbufLen) &&
+ (RattrinbufLen >= 5) &&
+ (RattrinbufLen < ZATTR_BUFFER_SIZE)) {
+ /* ---------------------------------------------------------------- */
+ // We start by checking consistency. We must have the first five
+ // words of the ATTRINFO to give us the length of the regions. The
+ // size of these regions must be the same as the total ATTRINFO
+ // length and finally the total length must be within the limits.
+ /* ---------------------------------------------------------------- */
+
+ if (RinitReadLen > 0) {
+ jam();
+ /* ---------------------------------------------------------------- */
+ // The first step that can be taken in the interpreter is to read
+ // data of the tuple before any updates have been applied.
+ /* ---------------------------------------------------------------- */
+ TnoDataRW = readAttributes(pagePtr,
+ TupHeadOffset,
+ &cinBuffer[5],
+ RinitReadLen,
+ &dst[0],
+ dstLen);
+ if (TnoDataRW != (Uint32)-1) {
+ RattroutCounter = TnoDataRW;
+ RinstructionCounter += RinitReadLen;
+ } else {
+ jam();
+ tupkeyErrorLab(signal);
+ return -1;
+ }//if
+ }//if
+ if (RexecRegionLen > 0) {
+ jam();
+ /* ---------------------------------------------------------------- */
+ // The next step is the actual interpreted execution. This executes
+ // a register-based virtual machine which can read and write attributes
+ // to and from registers.
+ /* ---------------------------------------------------------------- */
+ Uint32 RsubPC = RinstructionCounter + RfinalUpdateLen + RfinalRLen;
+ TnoDataRW = interpreterNextLab(signal,
+ pagePtr,
+ TupHeadOffset,
+ &clogMemBuffer[0],
+ &cinBuffer[RinstructionCounter],
+ RexecRegionLen,
+ &cinBuffer[RsubPC],
+ RsubLen,
+ tmp,
+ tmpLen);
+ if (TnoDataRW != (Uint32)-1) {
+ RinstructionCounter += RexecRegionLen;
+ RlogSize = TnoDataRW;
+ } else {
+ jam();
+ return -1;
+ }//if
+ }//if
+ if (RfinalUpdateLen > 0) {
+ jam();
+ /* ---------------------------------------------------------------- */
+ // We can also apply a set of updates without any conditions as part
+ // of the interpreted execution.
+ /* ---------------------------------------------------------------- */
+ if (regOperPtr->optype == ZUPDATE) {
+ TnoDataRW = updateAttributes(pagePtr,
+ TupHeadOffset,
+ &cinBuffer[RinstructionCounter],
+ RfinalUpdateLen);
+ if (TnoDataRW != (Uint32)-1) {
+ MEMCOPY_NO_WORDS(&clogMemBuffer[RlogSize],
+ &cinBuffer[RinstructionCounter],
+ RfinalUpdateLen);
+ RinstructionCounter += RfinalUpdateLen;
+ RlogSize += RfinalUpdateLen;
+ } else {
+ jam();
+ tupkeyErrorLab(signal);
+ return -1;
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 19);
+ }//if
+ }//if
+ if (RfinalRLen > 0) {
+ jam();
+ /* ---------------------------------------------------------------- */
+ // The final action is that we can also read the tuple after it has
+ // been updated.
+ /* ---------------------------------------------------------------- */
+ TnoDataRW = readAttributes(pagePtr,
+ TupHeadOffset,
+ &cinBuffer[RinstructionCounter],
+ RfinalRLen,
+ &dst[RattroutCounter],
+ (dstLen - RattroutCounter));
+ if (TnoDataRW != (Uint32)-1) {
+ RattroutCounter += TnoDataRW;
+ } else {
+ jam();
+ tupkeyErrorLab(signal);
+ return -1;
+ }//if
+ }//if
+ regOperPtr->logSize = RlogSize;
+ regOperPtr->attroutbufLen = RattroutCounter;
+ if(!executeDirect) {
+ jam();
+ sendReadAttrinfo(signal, RattroutCounter, regOperPtr);
+ } else {
+ jam();
+ Uint32 sig0 = regOperPtr->tcOperationPtr;
+ Uint32 sig1 = regOperPtr->transid1;
+ Uint32 sig2 = regOperPtr->transid2;
+ signal->theData[0] = sig0;
+ signal->theData[1] = sig1;
+ signal->theData[2] = sig2;
+ EXECUTE_DIRECT(refToBlock(sendBref), GSN_TRANSID_AI, signal,
+ 3 + RattroutCounter);
+ }//if
+ if (RlogSize > 0) {
+ sendLogAttrinfo(signal, RlogSize, regOperPtr);
+ }//if
+ return 0;
+ } else {
+ return TUPKEY_abort(signal, 22);
+ }//if
+}//Dbtup::interpreterStartLab()
+
+/* ---------------------------------------------------------------- */
+/* WHEN EXECUTION IS INTERPRETED WE NEED TO SEND SOME ATTRINFO*/
+/* BACK TO LQH FOR LOGGING AND SENDING TO BACKUP AND STANDBY */
+/* NODES. */
+/* INPUT: LOG_ATTRINFOPTR WHERE TO FETCH DATA FROM */
+/* TLOG_START FIRST INDEX TO LOG */
+/* TLOG_END LAST INDEX + 1 TO LOG */
+/* ---------------------------------------------------------------- */
+void Dbtup::sendLogAttrinfo(Signal* signal,
+ Uint32 TlogSize,
+ Operationrec * const regOperPtr)
+
+{
+ Uint32 TbufferIndex = 0;
+ signal->theData[0] = regOperPtr->userpointer;
+ while (TlogSize > 22) {
+ MEMCOPY_NO_WORDS(&signal->theData[3],
+ &clogMemBuffer[TbufferIndex],
+ 22);
+ EXECUTE_DIRECT(refToBlock(regOperPtr->userblockref),
+ GSN_TUP_ATTRINFO, signal, 25);
+ TbufferIndex += 22;
+ TlogSize -= 22;
+ }//while
+ MEMCOPY_NO_WORDS(&signal->theData[3],
+ &clogMemBuffer[TbufferIndex],
+ TlogSize);
+ EXECUTE_DIRECT(refToBlock(regOperPtr->userblockref),
+ GSN_TUP_ATTRINFO, signal, 3 + TlogSize);
+}//Dbtup::sendLogAttrinfo()
+
+inline
+Uint32
+brancher(Uint32 TheInstruction, Uint32 TprogramCounter)
+{
+ Uint32 TbranchDirection = TheInstruction >> 31;
+ Uint32 TbranchLength = (TheInstruction >> 16) & 0x7fff;
+ TprogramCounter--;
+ if (TbranchDirection == 1) {
+ jam();
+ /* ---------------------------------------------------------------- */
+ /* WE JUMP BACKWARDS. */
+ /* ---------------------------------------------------------------- */
+ return (TprogramCounter - TbranchLength);
+ } else {
+ jam();
+ /* ---------------------------------------------------------------- */
+ /* WE JUMP FORWARD. */
+ /* ---------------------------------------------------------------- */
+ return (TprogramCounter + TbranchLength);
+ }//if
+}//brancher()
+
+int Dbtup::interpreterNextLab(Signal* signal,
+ Page* const pagePtr,
+ Uint32 TupHeadOffset,
+ Uint32* logMemory,
+ Uint32* mainProgram,
+ Uint32 TmainProgLen,
+ Uint32* subroutineProg,
+ Uint32 TsubroutineLen,
+ Uint32 * tmpArea,
+ Uint32 tmpAreaSz)
+{
+ register Uint32* TcurrentProgram = mainProgram;
+ register Uint32 TcurrentSize = TmainProgLen;
+ register Uint32 RnoOfInstructions = 0;
+ register Uint32 TprogramCounter = 0;
+ register Uint32 theInstruction;
+ register Uint32 theRegister;
+ Uint32 TdataWritten = 0;
+ Uint32 RstackPtr = 0;
+ Uint32 TregMemBuffer[32];
+ Uint32 TstackMemBuffer[32];
+
+ /* ---------------------------------------------------------------- */
+ // Initialise all 8 registers to contain the NULL value.
+ // In this version we can handle 32 and 64 bit unsigned integers.
+ // They are handled as 64 bit values. Thus the 32 most significant
+ // bits are zeroed for 32 bit values.
+ /* ---------------------------------------------------------------- */
+ TregMemBuffer[0] = 0;
+ TregMemBuffer[4] = 0;
+ TregMemBuffer[8] = 0;
+ TregMemBuffer[12] = 0;
+ TregMemBuffer[16] = 0;
+ TregMemBuffer[20] = 0;
+ TregMemBuffer[24] = 0;
+ TregMemBuffer[28] = 0;
+ Uint32 tmpHabitant = ~0;
+
+ while (RnoOfInstructions < 8000) {
+ /* ---------------------------------------------------------------- */
+ /* EXECUTE THE NEXT INTERPRETER INSTRUCTION. */
+ /* ---------------------------------------------------------------- */
+ RnoOfInstructions++;
+ theInstruction = TcurrentProgram[TprogramCounter];
+ theRegister = Interpreter::getReg1(theInstruction) << 2;
+ if (TprogramCounter < TcurrentSize) {
+ TprogramCounter++;
+ switch (Interpreter::getOpCode(theInstruction)) {
+ case Interpreter::READ_ATTR_INTO_REG:
+ jam();
+ /* ---------------------------------------------------------------- */
+ // Read an attribute from the tuple into a register.
+ // While reading an attribute we allow the attribute to be an array
+ // as long as it fits in the 64 bits of the register.
+ /* ---------------------------------------------------------------- */
+ {
+ Uint32 theAttrinfo = theInstruction;
+ Uint32 TnoDataRW;
+ TnoDataRW = readAttributes(pagePtr,
+ TupHeadOffset,
+ &theAttrinfo,
+ (Uint32)1,
+ &TregMemBuffer[theRegister],
+ (Uint32)3);
+ if (TnoDataRW == 2) {
+ /* ------------------------------------------------------------- */
+ // Two words read means that we get the instruction plus one 32
+ // word read. Thus we set the register to be a 32 bit register.
+ /* ------------------------------------------------------------- */
+ TregMemBuffer[theRegister] = 0x50;
+ TregMemBuffer[theRegister + 2] = 0;
+ } else if (TnoDataRW == 3) {
+ /* ------------------------------------------------------------- */
+ // Three words read means that we get the instruction plus two
+ // 32 words read. Thus we set the register to be a 64 bit register.
+ /* ------------------------------------------------------------- */
+ TregMemBuffer[theRegister] = 0x60;
+ } else if (TnoDataRW == 1) {
+ /* ------------------------------------------------------------- */
+ // One word read means that we must have read a NULL value. We set
+ // the register to indicate a NULL value.
+ /* ------------------------------------------------------------- */
+ TregMemBuffer[theRegister] = 0;
+ } else if (TnoDataRW == (Uint32)-1) {
+ jam();
+ tupkeyErrorLab(signal);
+ return -1;
+ } else {
+ /* ------------------------------------------------------------- */
+ // Any other return value from the read attribute here is not
+ // allowed and will lead to a system crash.
+ /* ------------------------------------------------------------- */
+ ndbrequire(false);
+ }//if
+ break;
+ }
+
+ case Interpreter::WRITE_ATTR_FROM_REG:
+ jam();
+ {
+ Uint32 TattrId = theInstruction >> 16;
+ Uint32 TattrDescrIndex = tabptr.p->tabDescriptor +
+ (TattrId << ZAD_LOG_SIZE);
+ Uint32 TattrDesc1 = tableDescriptor[TattrDescrIndex].tabDescr;
+ Uint32 TregType = TregMemBuffer[theRegister];
+
+ /* --------------------------------------------------------------- */
+ // Calculate the number of words of this attribute.
+ // We allow writes into arrays as long as they fit into the 64 bit
+ // register size.
+ //TEST_MR See to that TattrNoOfWords can be
+ // read faster from attribute description.
+ /* --------------------------------------------------------------- */
+ Uint32 TarraySize = (TattrDesc1 >> 16);
+ Uint32 TattrLogLen = (TattrDesc1 >> 4) & 0xf;
+ Uint32 TattrNoOfBits = TarraySize << TattrLogLen;
+ Uint32 TattrNoOfWords = (TattrNoOfBits + 31) >> 5;
+ Uint32 Toptype = operPtr.p->optype;
+
+ Uint32 TdataForUpdate[3];
+ Uint32 Tlen;
+
+ AttributeHeader& ah = AttributeHeader::init(&TdataForUpdate[0],
+ TattrId, TattrNoOfWords);
+ TdataForUpdate[1] = TregMemBuffer[theRegister + 1];
+ TdataForUpdate[2] = TregMemBuffer[theRegister + 2];
+ Tlen = TattrNoOfWords + 1;
+ if (Toptype == ZUPDATE) {
+ if (TattrNoOfWords <= 2) {
+ if (TregType == 0) {
+ /* --------------------------------------------------------- */
+ // Write a NULL value into the attribute
+ /* --------------------------------------------------------- */
+ ah.setNULL();
+ Tlen = 1;
+ }//if
+ Uint32 TnoDataRW;
+ TnoDataRW = updateAttributes(pagePtr,
+ TupHeadOffset,
+ &TdataForUpdate[0],
+ Tlen);
+ if (TnoDataRW != (Uint32)-1) {
+ /* --------------------------------------------------------- */
+ // Write the written data also into the log buffer so that it
+ // will be logged.
+ /* --------------------------------------------------------- */
+ logMemory[TdataWritten + 0] = TdataForUpdate[0];
+ logMemory[TdataWritten + 1] = TdataForUpdate[1];
+ logMemory[TdataWritten + 2] = TdataForUpdate[2];
+ TdataWritten += Tlen;
+ } else {
+ tupkeyErrorLab(signal);
+ return -1;
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 15);
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 16);
+ }//if
+ break;
+ }
+
+ case Interpreter::LOAD_CONST_NULL:
+ jam();
+ TregMemBuffer[theRegister] = 0; /* NULL INDICATOR */
+ break;
+
+ case Interpreter::LOAD_CONST16:
+ jam();
+ TregMemBuffer[theRegister] = 0x50; /* 32 BIT UNSIGNED CONSTANT */
+ TregMemBuffer[theRegister + 1] = theInstruction >> 16;
+ TregMemBuffer[theRegister + 2] = 0;
+ break;
+
+ case Interpreter::LOAD_CONST32:
+ jam();
+ TregMemBuffer[theRegister] = 0x50; /* 32 BIT UNSIGNED CONSTANT */
+ TregMemBuffer[theRegister + 1] = TcurrentProgram[TprogramCounter];
+ TregMemBuffer[theRegister + 2] = 0;
+ TprogramCounter++;
+ break;
+
+ case Interpreter::LOAD_CONST64:
+ jam();
+ TregMemBuffer[theRegister] = 0x60; /* 64 BIT UNSIGNED CONSTANT */
+ TregMemBuffer[theRegister + 1] = TcurrentProgram[TprogramCounter + 0];
+ TregMemBuffer[theRegister + 2] = TcurrentProgram[TprogramCounter + 1];
+ TprogramCounter += 2;
+ break;
+
+ case Interpreter::ADD_REG_REG:
+ jam();
+ {
+ Uint32 TrightRegister = Interpreter::getReg2(theInstruction) << 2;
+ Uint32 TdestRegister = Interpreter::getReg3(theInstruction) << 2;
+
+ Uint32 TrightType = TregMemBuffer[TrightRegister];
+ Uint32 Tright0 = TregMemBuffer[TrightRegister + 1];
+ Uint32 Tright1 = TregMemBuffer[TrightRegister + 2];
+
+ Uint32 TleftType = TregMemBuffer[theRegister];
+ Uint32 Tleft0 = TregMemBuffer[theRegister + 1];
+ Uint32 Tleft1 = TregMemBuffer[theRegister + 2];
+ Uint32 Tany64bit = (((TleftType | TrightType) & 0x60) == 0x60);
+
+ if ((TleftType | TrightType) != 0) {
+ Uint32 Tdest0 = Tleft0 + Tright0;
+ Uint32 Tdest1 = 0;
+ TregMemBuffer[TdestRegister + 1] = Tdest0;
+ TregMemBuffer[TdestRegister] = 0x50;
+ if (Tany64bit) {
+ TregMemBuffer[TdestRegister] = 0x60;
+ Tdest1 = Tleft1 + Tright1;
+ if (Tdest0 < Tleft0) {
+ Tdest1++;
+ }
+ }//if
+ TregMemBuffer[TdestRegister + 2] = Tdest1;
+ } else {
+ return TUPKEY_abort(signal, 20);
+ }
+ break;
+ }
+
+ case Interpreter::SUB_REG_REG:
+ jam();
+ {
+ Uint32 TrightRegister = Interpreter::getReg2(theInstruction) << 2;
+ Uint32 TdestRegister = Interpreter::getReg3(theInstruction) << 2;
+
+ Uint32 TrightType = TregMemBuffer[TrightRegister];
+ Uint32 Tright0 = TregMemBuffer[TrightRegister + 1];
+ Uint32 Tright1 = TregMemBuffer[TrightRegister + 2];
+
+ Uint32 TleftType = TregMemBuffer[theRegister];
+ Uint32 Tleft0 = TregMemBuffer[theRegister + 1];
+ Uint32 Tleft1 = TregMemBuffer[theRegister + 2];
+ Uint32 Tany64bit = (((TleftType | TrightType) & 0x60) == 0x60);
+
+ if ((TleftType | TrightType) != 0) {
+ Uint32 Tdest0 = Tleft0 - Tright0;
+ Uint32 Tdest1 = 0;
+ TregMemBuffer[TdestRegister + 1] = Tdest0;
+ TregMemBuffer[TdestRegister] = 0x50;
+ if (Tany64bit) {
+ TregMemBuffer[TdestRegister] = 0x60;
+ Tdest1 = Tleft1 - Tright1;
+ if (Tdest0 > Tleft0) {
+ Tdest1--;
+ }//if
+ }//if
+ TregMemBuffer[TdestRegister + 2] = Tdest1;
+ } else {
+ return TUPKEY_abort(signal, 21);
+ }//if
+ break;
+ }
+
+ case Interpreter::BRANCH:
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ break;
+
+ case Interpreter::BRANCH_REG_EQ_NULL:
+ if (TregMemBuffer[theRegister] != 0) {
+ jam();
+ continue;
+ } else {
+ jam();
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ }//if
+ break;
+
+ case Interpreter::BRANCH_REG_NE_NULL:
+ if (TregMemBuffer[theRegister] == 0) {
+ jam();
+ continue;
+ } else {
+ jam();
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ }//if
+ break;
+
+
+ case Interpreter::BRANCH_EQ_REG_REG:
+ {
+ Uint32 TrightRegister = Interpreter::getReg2(theInstruction) << 2;
+
+ Uint32 TleftType = TregMemBuffer[theRegister];
+ Uint32 Tleft0 = TregMemBuffer[theRegister + 1];
+ Uint32 Tleft1 = TregMemBuffer[theRegister + 2];
+
+ Uint32 TrightType = TregMemBuffer[TrightRegister];
+ Uint32 Tright0 = TregMemBuffer[TrightRegister + 1];
+ Uint32 Tright1 = TregMemBuffer[TrightRegister + 2];
+ if ((TrightType | TleftType) != 0) {
+ jam();
+ if ((Tleft0 == Tright0) && (Tleft1 == Tright1)) {
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 23);
+ }//if
+ break;
+ }
+
+ case Interpreter::BRANCH_NE_REG_REG:
+ {
+ Uint32 TrightRegister = Interpreter::getReg2(theInstruction) << 2;
+
+ Uint32 TleftType = TregMemBuffer[theRegister];
+ Uint32 Tleft0 = TregMemBuffer[theRegister + 1];
+ Uint32 Tleft1 = TregMemBuffer[theRegister + 2];
+
+ Uint32 TrightType = TregMemBuffer[TrightRegister];
+ Uint32 Tright0 = TregMemBuffer[TrightRegister + 1];
+ Uint32 Tright1 = TregMemBuffer[TrightRegister + 2];
+ if ((TrightType | TleftType) != 0) {
+ jam();
+ if ((Tleft0 != Tright0) || (Tleft1 != Tright1)) {
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 24);
+ }//if
+ break;
+ }
+
+ case Interpreter::BRANCH_LT_REG_REG:
+ {
+ Uint32 TrightRegister = Interpreter::getReg2(theInstruction) << 2;
+
+ Uint32 TleftType = TregMemBuffer[theRegister];
+ Uint32 Tleft0 = TregMemBuffer[theRegister + 1];
+ Uint32 Tleft1 = TregMemBuffer[theRegister + 2];
+
+ Uint32 TrightType = TregMemBuffer[TrightRegister];
+ Uint32 Tright0 = TregMemBuffer[TrightRegister + 1];
+ Uint32 Tright1 = TregMemBuffer[TrightRegister + 2];
+ if ((TrightType | TleftType) != 0) {
+ jam();
+ if ((Tleft0 < Tright0) || ((Tleft0 == Tright0) &&
+ (Tleft1 < Tright1))) {
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 24);
+ }//if
+ break;
+ }
+
+ case Interpreter::BRANCH_LE_REG_REG:
+ {
+ Uint32 TrightRegister = Interpreter::getReg2(theInstruction) << 2;
+
+ Uint32 TleftType = TregMemBuffer[theRegister];
+ Uint32 Tleft0 = TregMemBuffer[theRegister + 1];
+ Uint32 Tleft1 = TregMemBuffer[theRegister + 2];
+
+ Uint32 TrightType = TregMemBuffer[TrightRegister];
+ Uint32 Tright0 = TregMemBuffer[TrightRegister + 1];
+ Uint32 Tright1 = TregMemBuffer[TrightRegister + 2];
+ if ((TrightType | TleftType) != 0) {
+ jam();
+ if ((Tleft0 < Tright0) || ((Tleft0 == Tright0) &&
+ (Tleft1 <= Tright1))) {
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 26);
+ }//if
+ break;
+ }
+
+ case Interpreter::BRANCH_GT_REG_REG:
+ {
+ Uint32 TrightRegister = Interpreter::getReg2(theInstruction) << 2;
+
+ Uint32 TleftType = TregMemBuffer[theRegister];
+ Uint32 Tleft0 = TregMemBuffer[theRegister + 1];
+ Uint32 Tleft1 = TregMemBuffer[theRegister + 2];
+
+ Uint32 TrightType = TregMemBuffer[TrightRegister];
+ Uint32 Tright0 = TregMemBuffer[TrightRegister + 1];
+ Uint32 Tright1 = TregMemBuffer[TrightRegister + 2];
+ if ((TrightType | TleftType) != 0) {
+ jam();
+ if ((Tleft0 > Tright0) || ((Tleft0 == Tright0) &&
+ (Tleft1 > Tright1))) {
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 27);
+ }//if
+ break;
+ }
+
+ case Interpreter::BRANCH_GE_REG_REG:
+ {
+ Uint32 TrightRegister = Interpreter::getReg2(theInstruction) << 2;
+
+ Uint32 TleftType = TregMemBuffer[theRegister];
+ Uint32 Tleft0 = TregMemBuffer[theRegister + 1];
+ Uint32 Tleft1 = TregMemBuffer[theRegister + 2];
+
+ Uint32 TrightType = TregMemBuffer[TrightRegister];
+ Uint32 Tright0 = TregMemBuffer[TrightRegister + 1];
+ Uint32 Tright1 = TregMemBuffer[TrightRegister + 2];
+ if ((TrightType | TleftType) != 0) {
+ jam();
+ if ((Tleft0 > Tright0) || ((Tleft0 == Tright0) &&
+ (Tleft1 >= Tright1))) {
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 28);
+ }//if
+ break;
+ }
+
+ case Interpreter::BRANCH_ATTR_OP_ARG:{
+ jam();
+ Uint32 cond = Interpreter::getBinaryCondition(theInstruction);
+ Uint32 diff = Interpreter::getArrayLengthDiff(theInstruction);
+ Uint32 vchr = Interpreter::isVarchar(theInstruction);
+ Uint32 nopad =Interpreter::isNopad(theInstruction);
+ Uint32 ins2 = TcurrentProgram[TprogramCounter];
+ Uint32 attrId = Interpreter::getBranchCol_AttrId(ins2) << 16;
+ Uint32 argLen = Interpreter::getBranchCol_Len(ins2);
+
+ if(tmpHabitant != attrId){
+ Int32 TnoDataR = readAttributes(pagePtr,
+ TupHeadOffset,
+ &attrId, 1,
+ tmpArea, tmpAreaSz);
+
+ if (TnoDataR == -1) {
+ jam();
+ tupkeyErrorLab(signal);
+ return -1;
+ }
+ tmpHabitant = attrId;
+ }
+
+ AttributeHeader ah(tmpArea[0]);
+
+ const char* s1 = (char*)&tmpArea[1];
+ const char* s2 = (char*)&TcurrentProgram[TprogramCounter+1];
+ Uint32 attrLen = (4 * ah.getDataSize()) - diff;
+ if (vchr) {
+#if NDB_VERSION_MAJOR >= 3
+ bool vok = false;
+ if (attrLen >= 2) {
+ Uint32 vlen = (s1[0] << 8) | s1[1]; // big-endian
+ s1 += 2;
+ attrLen -= 2;
+ if (attrLen >= vlen) {
+ attrLen = vlen;
+ vok = true;
+ }
+ }
+ if (!vok) {
+ terrorCode = ZREGISTER_INIT_ERROR;
+ tupkeyErrorLab(signal);
+ return -1;
+ }
+#else
+ Uint32 tmp;
+ if (attrLen >= 2) {
+ unsigned char* ss = (unsigned char*)&s1[attrLen - 2];
+ tmp = (ss[0] << 8) | ss[1];
+ if (tmp <= attrLen - 2)
+ attrLen = tmp;
+ }
+ // XXX handle bad data
+#endif
+ }
+ bool res = false;
+
+ switch ((Interpreter::BinaryCondition)cond) {
+ case Interpreter::EQ:
+ res = NdbSqlUtil::char_compare(s1, attrLen, s2, argLen, !nopad) == 0;
+ break;
+ case Interpreter::NE:
+ res = NdbSqlUtil::char_compare(s1, attrLen, s2, argLen, !nopad) != 0;
+ break;
+ // note the condition is backwards
+ case Interpreter::LT:
+ res = NdbSqlUtil::char_compare(s1, attrLen, s2, argLen, !nopad) > 0;
+ break;
+ case Interpreter::LE:
+ res = NdbSqlUtil::char_compare(s1, attrLen, s2, argLen, !nopad) >= 0;
+ break;
+ case Interpreter::GT:
+ res = NdbSqlUtil::char_compare(s1, attrLen, s2, argLen, !nopad) < 0;
+ break;
+ case Interpreter::GE:
+ res = NdbSqlUtil::char_compare(s1, attrLen, s2, argLen, !nopad) <= 0;
+ break;
+ case Interpreter::LIKE:
+ res = NdbSqlUtil::char_like(s1, attrLen, s2, argLen, !nopad);
+ break;
+ case Interpreter::NOT_LIKE:
+ res = ! NdbSqlUtil::char_like(s1, attrLen, s2, argLen, !nopad);
+ break;
+ // XXX handle invalid value
+ }
+#ifdef TRACE_INTERPRETER
+ ndbout_c("cond=%u diff=%d vc=%d nopad=%d attr(%d) = >%.*s<(%d) str=>%.*s<(%d) -> res = %d",
+ cond, diff, vchr, nopad,
+ attrId >> 16, attrLen, s1, attrLen, argLen, s2, argLen, res);
+#endif
+ if (res)
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ else {
+ Uint32 tmp = (Interpreter::mod4(argLen) >> 2) + 1;
+ TprogramCounter += tmp;
+ }
+ break;
+ }
+
+ case Interpreter::BRANCH_ATTR_EQ_NULL:{
+ jam();
+ Uint32 ins2 = TcurrentProgram[TprogramCounter];
+ Uint32 attrId = Interpreter::getBranchCol_AttrId(ins2) << 16;
+
+ if(tmpHabitant != attrId){
+ Int32 TnoDataR = readAttributes(pagePtr,
+ TupHeadOffset,
+ &attrId, 1,
+ tmpArea, tmpAreaSz);
+
+ if (TnoDataR == -1) {
+ jam();
+ tupkeyErrorLab(signal);
+ return -1;
+ }
+ tmpHabitant = attrId;
+ }
+
+ AttributeHeader ah(tmpArea[0]);
+ if(ah.isNULL()){
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ } else {
+ TprogramCounter ++;
+ }
+ break;
+ }
+
+ case Interpreter::BRANCH_ATTR_NE_NULL:{
+ jam();
+ Uint32 ins2 = TcurrentProgram[TprogramCounter];
+ Uint32 attrId = Interpreter::getBranchCol_AttrId(ins2) << 16;
+
+ if(tmpHabitant != attrId){
+ Int32 TnoDataR = readAttributes(pagePtr,
+ TupHeadOffset,
+ &attrId, 1,
+ tmpArea, tmpAreaSz);
+
+ if (TnoDataR == -1) {
+ jam();
+ tupkeyErrorLab(signal);
+ return -1;
+ }
+ tmpHabitant = attrId;
+ }
+
+ AttributeHeader ah(tmpArea[0]);
+ if(ah.isNULL()){
+ TprogramCounter ++;
+ } else {
+ TprogramCounter = brancher(theInstruction, TprogramCounter);
+ }
+ break;
+ }
+
+ case Interpreter::EXIT_OK:
+ case Interpreter::EXIT_OK_LAST:
+ jam();
+#ifdef TRACE_INTERPRETER
+ ndbout_c(" - exit_ok");
+#endif
+ return TdataWritten;
+
+ case Interpreter::EXIT_REFUSE:
+ jam();
+#ifdef TRACE_INTERPRETER
+ ndbout_c(" - exit_nok");
+#endif
+ terrorCode = theInstruction >> 16;
+ return TUPKEY_abort(signal, 29);
+
+ case Interpreter::CALL:
+ jam();
+ RstackPtr++;
+ if (RstackPtr < 32) {
+ TstackMemBuffer[RstackPtr] = TprogramCounter + 1;
+ TprogramCounter = theInstruction >> 16;
+ if (TprogramCounter < TsubroutineLen) {
+ TcurrentProgram = subroutineProg;
+ TcurrentSize = TsubroutineLen;
+ } else {
+ return TUPKEY_abort(signal, 30);
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 31);
+ }//if
+ break;
+
+ case Interpreter::RETURN:
+ jam();
+ if (RstackPtr > 0) {
+ TprogramCounter = TstackMemBuffer[RstackPtr];
+ RstackPtr--;
+ if (RstackPtr == 0) {
+ jam();
+ /* ------------------------------------------------------------- */
+ // We are back to the main program.
+ /* ------------------------------------------------------------- */
+ TcurrentProgram = mainProgram;
+ TcurrentSize = TmainProgLen;
+ }//if
+ } else {
+ return TUPKEY_abort(signal, 32);
+ }//if
+ break;
+
+ default:
+ return TUPKEY_abort(signal, 33);
+ }//switch
+ } else {
+ return TUPKEY_abort(signal, 34);
+ }//if
+ }//while
+ return TUPKEY_abort(signal, 35);
+}//Dbtup::interpreterNextLab()
+
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupFixAlloc.cpp b/ndb/src/kernel/blocks/dbtup/DbtupFixAlloc.cpp
new file mode 100644
index 00000000000..cdd54ba2337
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupFixAlloc.cpp
@@ -0,0 +1,384 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+
+#define ljam() { jamLine(6000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(6000 + __LINE__); }
+
+//
+// Fixed Allocator
+// This module is used to allocate and free fixed size tuples from the
+// set of pages attached to a fragment. The fixed size is preset per
+// fragment and their can only be one such value per fragment in the
+// current implementation.
+//
+// Public methods
+// bool allocTh(Fragrecord* const regFragPtr, # In
+// Tablerec* const regTabPtr, # In
+// Uint32 pageType, # In
+// Signal* signal, # In
+// Uint32& pageOffset, # Out
+// PagePtr& pagePtr) # In/Out
+// This method allocates a fixed size and the pagePtr is a reference
+// to the page and pageOffset is the offset in the page of the tuple.
+//
+// freeTh()
+// This method is used to free a tuple header in normal transaction
+// handling.
+//
+// freeThSr()
+// This method is used to free a tuple as part of executing the undo
+// log records.
+//
+// getThAtPageSr()
+// This method is used to allocate a tuple on a set page as part of
+// undo log execution.
+//
+//
+// Private methods
+// getThAtPage()
+// This method gets a tuple from a page with free tuples.
+//
+// convertThPage()
+// Convert an empty page into a page of free tuples in a linked list.
+//
+// getEmptyPageThCopy()
+// A page recently taken from set of empty pages on fragment is made
+// part of the copy pages.
+//
+// getEmptyPageTh()
+// A page recently taken from the set of empty pages on the fragment is
+// is made part of the set of free pages with fixed size tuples in the
+// fragment.
+//
+bool Dbtup::allocTh(Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Uint32 pageType,
+ Signal* signal,
+ Uint32& pageOffset,
+ PagePtr& pagePtr)
+{
+ if (pageType == SAME_PAGE) {
+ ljam();
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ if (pagePtr.p->pageWord[ZPAGE_STATE_POS] == ZTH_MM_FREE) {
+ ljam();
+ getThAtPage(regFragPtr, pagePtr.p, signal, pageOffset);
+ return true;
+ }//if
+ pageType = NORMAL_PAGE;
+ }//if
+ if (pageType == COPY_PAGE) {
+/* ---------------------------------------------------------------- */
+// Allocate a tuple header for the copy of the tuple header
+/* ---------------------------------------------------------------- */
+ if (regFragPtr->thFreeCopyFirst == RNIL) {
+/* ---------------------------------------------------------------- */
+// No page in list with free tuple header exists
+/* ---------------------------------------------------------------- */
+ if (regFragPtr->noCopyPagesAlloc < ZMAX_NO_COPY_PAGES) {
+ ljam();
+/* ---------------------------------------------------------------- */
+// We have not yet allocated the maximum number of copy pages for
+// this fragment.
+/* ---------------------------------------------------------------- */
+ pagePtr.i = getEmptyPage(regFragPtr);
+ if (pagePtr.i != RNIL) {
+ ljam();
+/* ---------------------------------------------------------------- */
+// We have empty pages already allocated to this fragment. Allocate
+// one of those as copy page.
+/* ---------------------------------------------------------------- */
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ getEmptyPageThCopy(regFragPtr, signal, pagePtr.p);
+/* ---------------------------------------------------------------- */
+// Convert page into a tuple header page.
+/* ---------------------------------------------------------------- */
+ convertThPage(regTabPtr->tupheadsize, pagePtr.p);
+ getThAtPage(regFragPtr, pagePtr.p, signal, pageOffset);
+ return true;
+ }//if
+ }//if
+ } else {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* NORMAL PATH WHEN COPY PAGE REQUESTED, GET PAGE POINTER */
+/* AND THEN GOTO COMMON HANDLING OF GET TUPLE HEADER AT PAGE. */
+/* ---------------------------------------------------------------- */
+ pagePtr.i = getRealpid(regFragPtr, regFragPtr->thFreeCopyFirst);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ getThAtPage(regFragPtr, pagePtr.p, signal, pageOffset);
+ return true;
+ }//if
+ }//if
+/* ---------------------------------------------------------------- */
+/* EITHER NORMAL PAGE REQUESTED OR ALLOCATION FROM COPY PAGE */
+/* FAILED. TRY ALLOCATING FROM NORMAL PAGE. */
+/* ---------------------------------------------------------------- */
+ Uint32 fragPageId = regFragPtr->thFreeFirst;
+ if (fragPageId == RNIL) {
+/* ---------------------------------------------------------------- */
+// No prepared tuple header page with free entries exists.
+/* ---------------------------------------------------------------- */
+ pagePtr.i = getEmptyPage(regFragPtr);
+ if (pagePtr.i != RNIL) {
+ ljam();
+/* ---------------------------------------------------------------- */
+// We found empty pages on the fragment. Allocate an empty page and
+// convert it into a tuple header page and put it in thFreeFirst-list.
+/* ---------------------------------------------------------------- */
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ getEmptyPageTh(regFragPtr, signal, pagePtr.p);
+ convertThPage(regTabPtr->tupheadsize, pagePtr.p);
+ getThAtPage(regFragPtr, pagePtr.p, signal, pageOffset);
+ return true;
+ } else {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* THERE ARE NO EMPTY PAGES. MEMORY CAN NOT BE ALLOCATED. */
+/* ---------------------------------------------------------------- */
+ terrorCode = ZMEM_NOMEM_ERROR;
+ return false;
+ }//if
+ } else {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* THIS SHOULD BE THE COMMON PATH THROUGH THE CODE, FREE */
+/* COPY PAGE EXISTED. */
+/* ---------------------------------------------------------------- */
+ pagePtr.i = getRealpid(regFragPtr, fragPageId);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ getThAtPage(regFragPtr, pagePtr.p, signal, pageOffset);
+ return true;
+ }//if
+ ndbrequire(false); // Dead code
+ return false;
+}//Dbtup::allocTh()
+
+void Dbtup::convertThPage(Uint32 Tupheadsize,
+ Page* const regPagePtr)
+{
+ Uint32 ctpConstant = Tupheadsize << 16;
+ Uint32 nextTuple = ZPAGE_HEADER_SIZE + Tupheadsize;
+ Uint32 endOfList;
+ /*
+ ASSUMES AT LEAST ONE TUPLE HEADER FITS AND THEREFORE NO HANDLING
+ OF ZERO AS EXTREME CASE
+ */
+ do {
+ ljam();
+ endOfList = nextTuple - Tupheadsize;
+ regPagePtr->pageWord[endOfList] = ctpConstant + nextTuple;
+ nextTuple += Tupheadsize;
+ } while (nextTuple <= ZWORDS_ON_PAGE);
+ regPagePtr->pageWord[endOfList] = ctpConstant;
+ Uint32 startOfList = ZPAGE_HEADER_SIZE;
+ regPagePtr->pageWord[ZFREELIST_HEADER_POS] = (startOfList << 16) + endOfList;
+}//Dbtup::convertThPage()
+
+void Dbtup::getEmptyPageTh(Fragrecord* const regFragPtr,
+ Signal* signal,
+ Page* const regPagePtr)
+{
+ if (isUndoLoggingNeeded(regFragPtr, regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS])) {
+ cprAddUndoLogPageHeader(signal,
+ regPagePtr,
+ regFragPtr);
+ }//if
+ regPagePtr->pageWord[ZPAGE_NEXT_POS] = regFragPtr->thFreeFirst;
+ regPagePtr->pageWord[ZPAGE_STATE_POS] = ZTH_MM_FREE;
+ regFragPtr->thFreeFirst = regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS];
+
+ndbrequire(regFragPtr->thFreeFirst != (RNIL -1));
+}//Dbtup::getEmptyPageTh()
+
+void Dbtup::getEmptyPageThCopy(Fragrecord* const regFragPtr,
+ Signal* signal,
+ Page* const regPagePtr)
+{
+ if (isUndoLoggingNeeded(regFragPtr, regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS])) {
+ cprAddUndoLogPageHeader(signal,
+ regPagePtr,
+ regFragPtr);
+ }//if
+ regPagePtr->pageWord[ZPAGE_NEXT_POS] = regFragPtr->thFreeCopyFirst;
+ regPagePtr->pageWord[ZPAGE_STATE_POS] = ZTH_MM_FREE_COPY;
+ regFragPtr->thFreeCopyFirst = regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS];
+ regFragPtr->noCopyPagesAlloc++;
+}//Dbtup::getEmptyPageThCopy()
+
+void Dbtup::getThAtPage(Fragrecord* const regFragPtr,
+ Page* const regPagePtr,
+ Signal* signal,
+ Uint32& pageOffset)
+{
+ Uint32 freeListHeader = regPagePtr->pageWord[ZFREELIST_HEADER_POS];
+ Uint32 startTuple = freeListHeader >> 16;
+ Uint32 endTuple = freeListHeader & 0xffff;
+ pageOffset = startTuple; /* START IS THE ONE ALLOCATED */
+ if (startTuple > 0) {
+ if (startTuple != endTuple) {
+/* ---------------------------------------------------------------- */
+/* NOT THE LAST, SIMPLY RESHUFFLE POINTERS. */
+/* ---------------------------------------------------------------- */
+ ndbrequire(startTuple < ZWORDS_ON_PAGE);
+ startTuple = regPagePtr->pageWord[startTuple] & 0xffff;
+ regPagePtr->pageWord[ZFREELIST_HEADER_POS] = endTuple +
+ (startTuple << 16);
+ return;
+ } else {
+/* ---------------------------------------------------------------- */
+/* THIS WAS THE LAST TUPLE HEADER IN THIS PAGE. REMOVE IT FROM*/
+/* THE TUPLE HEADER FREE LIST OR TH COPY FREE LIST. ALSO SET */
+/* A PROPER PAGE STATE. */
+/* */
+/* WE ALSO HAVE TO INSERT AN UNDO LOG ENTRY TO ENSURE PAGE */
+/* ARE MAINTAINED EVEN AFTER A SYSTEM CRASH. */
+/* ---------------------------------------------------------------- */
+ if (isUndoLoggingNeeded(regFragPtr,
+ regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS])) {
+ cprAddUndoLogPageHeader(signal,
+ regPagePtr,
+ regFragPtr);
+ }//if
+ if (regPagePtr->pageWord[ZPAGE_STATE_POS] == ZTH_MM_FREE) {
+ ljam();
+ regFragPtr->thFreeFirst = regPagePtr->pageWord[ZPAGE_NEXT_POS];
+ regPagePtr->pageWord[ZPAGE_STATE_POS] = ZTH_MM_FULL;
+ } else if (regPagePtr->pageWord[ZPAGE_STATE_POS] == ZTH_MM_FREE_COPY) {
+ ljam();
+ regFragPtr->thFreeCopyFirst = regPagePtr->pageWord[ZPAGE_NEXT_POS];
+ regPagePtr->pageWord[ZPAGE_STATE_POS] = ZTH_MM_FULL_COPY;
+ } else {
+ ndbrequire(false);
+ }//if
+ regPagePtr->pageWord[ZFREELIST_HEADER_POS] = 0;
+ regPagePtr->pageWord[ZPAGE_NEXT_POS] = RNIL;
+ }//if
+ } else {
+ ndbrequire(false);
+ }//if
+ return;
+}//Dbtup::getThAtPage()
+
+void Dbtup::getThAtPageSr(Page* const regPagePtr,
+ Uint32& pageOffset)
+{
+ Uint32 freeListHeader = regPagePtr->pageWord[ZFREELIST_HEADER_POS];
+ Uint32 startTuple = freeListHeader >> 16;
+ Uint32 endTuple = freeListHeader & 0xffff;
+ ndbrequire(startTuple > 0);
+ pageOffset = startTuple; /* START IS THE ONE ALLOCATED */
+ if (startTuple == endTuple) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* THIS WAS THE LAST TUPLE HEADER IN THIS PAGE. SINCE WE ARE */
+/* UNDOING PAGE UPDATES WE SHALL NOT DO ANYTHING ABOUT THE */
+/* PAGE HEADER. THIS IS DONE BY SEPARATE LOG RECORDS. */
+/* ---------------------------------------------------------------- */
+ regPagePtr->pageWord[ZFREELIST_HEADER_POS] = 0;
+ } else {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* NOT THE LAST, SIMPLY RESHUFFLE POINTERS. */
+/* ---------------------------------------------------------------- */
+ ndbrequire(startTuple < ZWORDS_ON_PAGE);
+ startTuple = regPagePtr->pageWord[startTuple] & 0xffff; /* GET NEXT POINTER */
+ regPagePtr->pageWord[ZFREELIST_HEADER_POS] = endTuple + (startTuple << 16);
+ }//if
+}//Dbtup::getThAtPageSr()
+
+void Dbtup::freeTh(Fragrecord* const regFragPtr,
+ Tablerec* const regTabPtr,
+ Signal* signal,
+ Page* const regPagePtr,
+ Uint32 freePageOffset)
+{
+ Uint32 startOfList = regPagePtr->pageWord[ZFREELIST_HEADER_POS] >> 16;
+ Uint32 endOfList = regPagePtr->pageWord[ZFREELIST_HEADER_POS] & 0xffff;
+/* LINK THE NOW FREE TUPLE SPACE INTO BEGINNING OF FREE LIST OF OF THE PAGE */
+/* SET THE SIZE OF THE NEW FREE SPACE AND LINK TO THE OLD START OF FREELIST */
+ ndbrequire(freePageOffset < ZWORDS_ON_PAGE);
+ regPagePtr->pageWord[freePageOffset] = (regTabPtr->tupheadsize << 16) +
+ startOfList;
+ if (endOfList == 0) {
+ ljam();
+ ndbrequire(startOfList == 0);
+/* ---------------------------------------------------------------- */
+/* THE PAGE WAS PREVIOUSLY FULL, NO EMPTY SPACE AT ALL. */
+/* THIS ENTRY WILL THEN BE BOTH THE START AND THE END OF THE */
+/* LIST. IT WILL ALSO BE PUT ON THE PROPER FREE LIST. */
+/* */
+/* UPDATE OF NEXT POINTER AND PAGE STATE MUST BE LOGGED TO */
+/* THE UNDO LOG TO ENSURE THAT FREE LISTS ARE OK AFTER A */
+/* SYSTEM RESTART. */
+/* ---------------------------------------------------------------- */
+ if (isUndoLoggingNeeded(regFragPtr, regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS])) {
+ cprAddUndoLogPageHeader(signal,
+ regPagePtr,
+ regFragPtr);
+ }//if
+ regPagePtr->pageWord[ZFREELIST_HEADER_POS] = (freePageOffset << 16) + freePageOffset;
+ if (regPagePtr->pageWord[ZPAGE_STATE_POS] == ZTH_MM_FULL) {
+ ljam();
+ regPagePtr->pageWord[ZPAGE_NEXT_POS] = regFragPtr->thFreeFirst;
+ regFragPtr->thFreeFirst = regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS];
+ regPagePtr->pageWord[ZPAGE_STATE_POS] = ZTH_MM_FREE;
+ } else {
+ ndbrequire(regPagePtr->pageWord[ZPAGE_STATE_POS] == ZTH_MM_FULL_COPY);
+ ljam();
+ regPagePtr->pageWord[ZPAGE_NEXT_POS] = regFragPtr->thFreeCopyFirst;
+ regFragPtr->thFreeCopyFirst = regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS];
+ regPagePtr->pageWord[ZPAGE_STATE_POS] = ZTH_MM_FREE_COPY;
+ }//if
+ } else {
+ ljam();
+ regPagePtr->pageWord[ZFREELIST_HEADER_POS] = (freePageOffset << 16) + endOfList;
+ }//if
+}//Dbtup::freeTh()
+
+void Dbtup::freeThSr(Tablerec* const regTabPtr,
+ Page* const regPagePtr,
+ Uint32 freePageOffset)
+{
+/* ------------------------------------------------------------------------ */
+/* LINK THE NOW FREE TUPLE SPACE INTO BEGINNING OF FREE LIST OF OF THE PAGE */
+/* SET THE SIZE OF THE NEW FREE SPACE AND LINK TO THE OLD START OF FREELIST */
+/* ------------------------------------------------------------------------ */
+ Uint32 startOfList = regPagePtr->pageWord[ZFREELIST_HEADER_POS] >> 16;
+ Uint32 endOfList = regPagePtr->pageWord[ZFREELIST_HEADER_POS] & 0xffff;
+ ndbrequire(freePageOffset < ZWORDS_ON_PAGE);
+ regPagePtr->pageWord[freePageOffset] = (regTabPtr->tupheadsize << 16) + startOfList;
+ if (endOfList == 0) {
+ ljam();
+ ndbrequire(startOfList == 0);
+/* ---------------------------------------------------------------- */
+/* THE PAGE WAS PREVIOUSLY FULL, NO EMPTY SPACE AT ALL. */
+/* THIS ENTRY WILL THEN BE BOTH THE START AND THE END OF THE */
+/* LIST. IT WILL ALSO BE PUT ON THE PROPER FREE LIST. */
+/* ---------------------------------------------------------------- */
+ regPagePtr->pageWord[ZFREELIST_HEADER_POS] = (freePageOffset << 16) + freePageOffset;
+ } else {
+ ljam();
+ regPagePtr->pageWord[ZFREELIST_HEADER_POS] = (freePageOffset << 16) + endOfList;
+ }//if
+}//Dbtup::freeThSr()
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupGen.cpp b/ndb/src/kernel/blocks/dbtup/DbtupGen.cpp
new file mode 100644
index 00000000000..982e1b8df24
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupGen.cpp
@@ -0,0 +1,1319 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <AttributeDescriptor.hpp>
+#include "AttributeOffset.hpp"
+#include <AttributeHeader.hpp>
+#include <Interpreter.hpp>
+#include <signaldata/FsConf.hpp>
+#include <signaldata/FsRef.hpp>
+#include <signaldata/FsRemoveReq.hpp>
+#include <signaldata/SetVarReq.hpp>
+#include <signaldata/TupCommit.hpp>
+#include <signaldata/TupKey.hpp>
+#include <signaldata/TupSizeAltReq.hpp>
+
+#include <signaldata/DropTab.hpp>
+#include <new>
+
+#define DEBUG(x) { ndbout << "TUP::" << x << endl; }
+
+#define ljam() { jamLine(24000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(24000 + __LINE__); }
+
+void Dbtup::initData()
+{
+ cnoOfAttrbufrec = ZNO_OF_ATTRBUFREC;
+ cnoOfLcpRec = ZNO_OF_LCP_REC;
+ cnoOfConcurrentOpenOp = ZNO_OF_CONCURRENT_OPEN_OP;
+ cnoOfConcurrentWriteOp = ZNO_OF_CONCURRENT_WRITE_OP;
+ cnoOfFragoprec = 2 * NO_OF_FRAG_PER_NODE;
+ cnoOfFragrec = ZNO_OF_FRAGREC;
+ cnoOfOprec = ZNO_OF_OPREC;
+ cnoOfPage = ZNO_OF_PAGE;
+ cnoOfPageRangeRec = ZNO_OF_PAGE_RANGE_REC;
+ cnoOfParallellUndoFiles = ZNO_OF_PARALLELL_UNDO_FILES;
+ cnoOfRestartInfoRec = ZNO_OF_RESTART_INFO_REC;
+ cnoOfTablerec = ZNO_OF_TABLEREC;
+ cnoOfTabDescrRec = ZNO_OF_TAB_DESCR_REC;
+ cnoOfUndoPage = ZNO_OF_UNDO_PAGE;
+ c_maxTriggersPerTable = ZDEFAULT_MAX_NO_TRIGGERS_PER_TABLE;
+ c_noOfBuildIndexRec = 32;
+
+ attrbufrec = 0;
+ checkpointInfo = 0;
+ diskBufferSegmentInfo = 0;
+ fragoperrec = 0;
+ fragrecord = 0;
+ hostBuffer = 0;
+ localLogInfo = 0;
+ operationrec = 0;
+ page = 0;
+ pageRange = 0;
+ pendingFileOpenInfo = 0;
+ restartInfoRecord = 0;
+ tablerec = 0;
+ tableDescriptor = 0;
+ undoPage = 0;
+ totNoOfPagesAllocated = 0;
+ cnoOfAllocatedPages = 0;
+
+ // Records with constant sizes
+}//Dbtup::initData()
+
+Dbtup::Dbtup(const class Configuration & conf)
+ : SimulatedBlock(DBTUP, conf),
+ c_storedProcPool(),
+ c_buildIndexList(c_buildIndexPool)
+{
+
+ BLOCK_CONSTRUCTOR(Dbtup);
+
+ addRecSignal(GSN_DEBUG_SIG, &Dbtup::execDEBUG_SIG);
+ addRecSignal(GSN_CONTINUEB, &Dbtup::execCONTINUEB);
+
+ addRecSignal(GSN_DUMP_STATE_ORD, &Dbtup::execDUMP_STATE_ORD);
+ addRecSignal(GSN_SEND_PACKED, &Dbtup::execSEND_PACKED);
+ addRecSignal(GSN_ATTRINFO, &Dbtup::execATTRINFO);
+ addRecSignal(GSN_STTOR, &Dbtup::execSTTOR);
+ addRecSignal(GSN_TUP_LCPREQ, &Dbtup::execTUP_LCPREQ);
+ addRecSignal(GSN_END_LCPREQ, &Dbtup::execEND_LCPREQ);
+ addRecSignal(GSN_START_RECREQ, &Dbtup::execSTART_RECREQ);
+ addRecSignal(GSN_MEMCHECKREQ, &Dbtup::execMEMCHECKREQ);
+ addRecSignal(GSN_TUPKEYREQ, &Dbtup::execTUPKEYREQ);
+ addRecSignal(GSN_TUPSEIZEREQ, &Dbtup::execTUPSEIZEREQ);
+ addRecSignal(GSN_TUPRELEASEREQ, &Dbtup::execTUPRELEASEREQ);
+ addRecSignal(GSN_STORED_PROCREQ, &Dbtup::execSTORED_PROCREQ);
+ addRecSignal(GSN_TUPFRAGREQ, &Dbtup::execTUPFRAGREQ);
+ addRecSignal(GSN_TUP_ADD_ATTRREQ, &Dbtup::execTUP_ADD_ATTRREQ);
+ addRecSignal(GSN_TUP_COMMITREQ, &Dbtup::execTUP_COMMITREQ);
+ addRecSignal(GSN_TUP_ABORTREQ, &Dbtup::execTUP_ABORTREQ);
+ addRecSignal(GSN_TUP_SRREQ, &Dbtup::execTUP_SRREQ);
+ addRecSignal(GSN_TUP_PREPLCPREQ, &Dbtup::execTUP_PREPLCPREQ);
+ addRecSignal(GSN_FSOPENCONF, &Dbtup::execFSOPENCONF);
+ addRecSignal(GSN_FSOPENREF, &Dbtup::execFSOPENREF);
+ addRecSignal(GSN_FSCLOSECONF, &Dbtup::execFSCLOSECONF);
+ addRecSignal(GSN_FSCLOSEREF, &Dbtup::execFSCLOSEREF);
+ addRecSignal(GSN_FSWRITECONF, &Dbtup::execFSWRITECONF);
+ addRecSignal(GSN_FSWRITEREF, &Dbtup::execFSWRITEREF);
+ addRecSignal(GSN_FSREADCONF, &Dbtup::execFSREADCONF);
+ addRecSignal(GSN_FSREADREF, &Dbtup::execFSREADREF);
+ addRecSignal(GSN_NDB_STTOR, &Dbtup::execNDB_STTOR);
+ addRecSignal(GSN_SIZEALT_REP, &Dbtup::execSIZEALT_REP);
+ addRecSignal(GSN_SET_VAR_REQ, &Dbtup::execSET_VAR_REQ);
+
+ // Trigger Signals
+ addRecSignal(GSN_CREATE_TRIG_REQ, &Dbtup::execCREATE_TRIG_REQ);
+ addRecSignal(GSN_DROP_TRIG_REQ, &Dbtup::execDROP_TRIG_REQ);
+
+ addRecSignal(GSN_DROP_TAB_REQ, &Dbtup::execDROP_TAB_REQ);
+ addRecSignal(GSN_FSREMOVEREF, &Dbtup::execFSREMOVEREF);
+ addRecSignal(GSN_FSREMOVECONF, &Dbtup::execFSREMOVECONF);
+
+ addRecSignal(GSN_TUP_ALLOCREQ, &Dbtup::execTUP_ALLOCREQ);
+ addRecSignal(GSN_TUP_DEALLOCREQ, &Dbtup::execTUP_DEALLOCREQ);
+ addRecSignal(GSN_TUP_WRITELOG_REQ, &Dbtup::execTUP_WRITELOG_REQ);
+
+ // Ordered index related
+ addRecSignal(GSN_TUP_READ_ATTRS, &Dbtup::execTUP_READ_ATTRS);
+ addRecSignal(GSN_TUP_QUERY_TH, &Dbtup::execTUP_QUERY_TH);
+ addRecSignal(GSN_TUP_STORE_TH, &Dbtup::execTUP_STORE_TH);
+ addRecSignal(GSN_BUILDINDXREQ, &Dbtup::execBUILDINDXREQ);
+
+ initData();
+}//Dbtup::Dbtup()
+
+Dbtup::~Dbtup()
+{
+ // Records with dynamic sizes
+ deallocRecord((void **)&attrbufrec,"Attrbufrec",
+ sizeof(Attrbufrec),
+ cnoOfAttrbufrec);
+
+ deallocRecord((void **)&checkpointInfo,"CheckpointInfo",
+ sizeof(CheckpointInfo),
+ cnoOfLcpRec);
+
+ deallocRecord((void **)&diskBufferSegmentInfo,
+ "DiskBufferSegmentInfo",
+ sizeof(DiskBufferSegmentInfo),
+ cnoOfConcurrentWriteOp);
+
+ deallocRecord((void **)&fragoperrec,"Fragoperrec",
+ sizeof(Fragoperrec),
+ cnoOfFragoprec);
+
+ deallocRecord((void **)&fragrecord,"Fragrecord",
+ sizeof(Fragrecord),
+ cnoOfFragrec);
+
+ deallocRecord((void **)&hostBuffer,"HostBuffer",
+ sizeof(HostBuffer),
+ MAX_NODES);
+
+ deallocRecord((void **)&localLogInfo,"LocalLogInfo",
+ sizeof(LocalLogInfo),
+ cnoOfParallellUndoFiles);
+
+ deallocRecord((void **)&operationrec,"Operationrec",
+ sizeof(Operationrec),
+ cnoOfOprec);
+
+ deallocRecord((void **)&page,"Page",
+ sizeof(Page),
+ cnoOfPage);
+
+ deallocRecord((void **)&pageRange,"PageRange",
+ sizeof(PageRange),
+ cnoOfPageRangeRec);
+
+ deallocRecord((void **)&pendingFileOpenInfo,
+ "PendingFileOpenInfo",
+ sizeof(PendingFileOpenInfo),
+ cnoOfConcurrentOpenOp);
+
+ deallocRecord((void **)&restartInfoRecord,
+ "RestartInfoRecord",
+ sizeof(RestartInfoRecord),
+ cnoOfRestartInfoRec);
+
+ deallocRecord((void **)&tablerec,"Tablerec",
+ sizeof(Tablerec),
+ cnoOfTablerec);
+
+ deallocRecord((void **)&tableDescriptor, "TableDescriptor",
+ sizeof(TableDescriptor),
+ cnoOfTabDescrRec);
+
+ deallocRecord((void **)&undoPage,"UndoPage",
+ sizeof(UndoPage),
+ cnoOfUndoPage);
+
+}//Dbtup::~Dbtup()
+
+BLOCK_FUNCTIONS(Dbtup);
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* ----- GENERAL SIGNAL MULTIPLEXER (FS + CONTINUEB) -------------- */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+void Dbtup::execFSCLOSECONF(Signal* signal)
+{
+ PendingFileOpenInfoPtr pfoPtr;
+ ljamEntry();
+ pfoPtr.i = signal->theData[0];
+ ptrCheckGuard(pfoPtr, cnoOfConcurrentOpenOp, pendingFileOpenInfo);
+ switch (pfoPtr.p->pfoOpenType) {
+ case LCP_DATA_FILE_CLOSE:
+ {
+ CheckpointInfoPtr ciPtr;
+ ljam();
+ ciPtr.i = pfoPtr.p->pfoCheckpointInfoP;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ ciPtr.p->lcpDataFileHandle = RNIL;
+ lcpClosedDataFileLab(signal, ciPtr);
+ break;
+ }
+ case LCP_UNDO_FILE_CLOSE:
+ {
+ LocalLogInfoPtr lliPtr;
+ ljam();
+ lliPtr.i = pfoPtr.p->pfoCheckpointInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ lliPtr.p->lliUndoFileHandle = RNIL;
+ lcpEndconfLab(signal);
+ break;
+ }
+ case LCP_UNDO_FILE_READ:
+ ljam();
+ endExecUndoLogLab(signal, pfoPtr.p->pfoCheckpointInfoP);
+ break;
+ case LCP_DATA_FILE_READ:
+ ljam();
+ rfrClosedDataFileLab(signal, pfoPtr.p->pfoCheckpointInfoP);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ releasePendingFileOpenInfoRecord(pfoPtr);
+}//Dbtup::execFSCLOSECONF()
+
+void Dbtup::execFSCLOSEREF(Signal* signal)
+{
+ ljamEntry();
+ ndbrequire(false);
+}//Dbtup::execFSCLOSEREF()
+
+void Dbtup::execFSOPENCONF(Signal* signal)
+{
+ PendingFileOpenInfoPtr pfoPtr;
+
+ ljamEntry();
+ pfoPtr.i = signal->theData[0];
+ Uint32 fileHandle = signal->theData[1];
+ ptrCheckGuard(pfoPtr, cnoOfConcurrentOpenOp, pendingFileOpenInfo);
+ switch (pfoPtr.p->pfoOpenType) {
+ case LCP_DATA_FILE_READ:
+ {
+ RestartInfoRecordPtr riPtr;
+ ljam();
+ riPtr.i = pfoPtr.p->pfoRestartInfoP;
+ ptrCheckGuard(riPtr, cnoOfRestartInfoRec, restartInfoRecord);
+ riPtr.p->sriDataFileHandle = fileHandle;
+ rfrReadRestartInfoLab(signal, riPtr);
+ break;
+ }
+ case LCP_UNDO_FILE_READ:
+ {
+ RestartInfoRecordPtr riPtr;
+ LocalLogInfoPtr lliPtr;
+ DiskBufferSegmentInfoPtr dbsiPtr;
+
+ ljam();
+ riPtr.i = pfoPtr.p->pfoRestartInfoP;
+ ptrCheckGuard(riPtr, cnoOfRestartInfoRec, restartInfoRecord);
+ lliPtr.i = riPtr.p->sriLocalLogInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ lliPtr.p->lliUndoFileHandle = fileHandle;
+ dbsiPtr.i = riPtr.p->sriDataBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ rfrLoadDataPagesLab(signal, riPtr, dbsiPtr);
+ break;
+ }
+ case LCP_DATA_FILE_WRITE_WITH_UNDO:
+ {
+ CheckpointInfoPtr ciPtr;
+ LocalLogInfoPtr lliPtr;
+
+ ljam();
+ ciPtr.i = pfoPtr.p->pfoCheckpointInfoP;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ lliPtr.i = ciPtr.p->lcpLocalLogInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ ciPtr.p->lcpDataFileHandle = fileHandle;
+ if (lliPtr.p->lliUndoFileHandle != RNIL) {
+ ljam();
+ signal->theData[0] = ciPtr.p->lcpUserptr;
+ signal->theData[1] = ciPtr.i;
+ sendSignal(ciPtr.p->lcpBlockref, GSN_TUP_PREPLCPCONF, signal, 2, JBB);
+ }//if
+ break;
+ }
+ case LCP_DATA_FILE_WRITE:
+ {
+ CheckpointInfoPtr ciPtr;
+
+ ljam();
+ ciPtr.i = pfoPtr.p->pfoCheckpointInfoP;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ ciPtr.p->lcpDataFileHandle = fileHandle;
+ signal->theData[0] = ciPtr.p->lcpUserptr;
+ signal->theData[1] = ciPtr.i;
+ sendSignal(ciPtr.p->lcpBlockref, GSN_TUP_PREPLCPCONF, signal, 2, JBB);
+ break;
+ }
+ case LCP_UNDO_FILE_WRITE:
+ {
+ CheckpointInfoPtr ciPtr;
+ LocalLogInfoPtr lliPtr;
+
+ ljam();
+ ciPtr.i = pfoPtr.p->pfoCheckpointInfoP;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ lliPtr.i = ciPtr.p->lcpLocalLogInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ lliPtr.p->lliUndoFileHandle = fileHandle;
+ if (ciPtr.p->lcpDataFileHandle != RNIL) {
+ ljam();
+ signal->theData[0] = ciPtr.p->lcpUserptr;
+ signal->theData[1] = ciPtr.i;
+ sendSignal(ciPtr.p->lcpBlockref, GSN_TUP_PREPLCPCONF, signal, 2, JBB);
+ }//if
+ break;
+ }
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ releasePendingFileOpenInfoRecord(pfoPtr);
+}//Dbtup::execFSOPENCONF()
+
+void Dbtup::execFSOPENREF(Signal* signal)
+{
+ ljamEntry();
+ ndbrequire(false);
+}//Dbtup::execFSOPENREF()
+
+void Dbtup::execFSREADCONF(Signal* signal)
+{
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ ljamEntry();
+ dbsiPtr.i = signal->theData[0];
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ switch (dbsiPtr.p->pdxOperation) {
+ case CHECKPOINT_DATA_READ:
+ {
+ RestartInfoRecordPtr riPtr;
+ ljam();
+ riPtr.i = dbsiPtr.p->pdxRestartInfoP;
+ ptrCheckGuard(riPtr, cnoOfRestartInfoRec, restartInfoRecord);
+/************************************************************/
+/* VERIFY THAT THE PAGES ARE CORRECT, HAVE A CORRECT */
+/* STATE AND A CORRECT PAGE ID. */
+/************************************************************/
+ ndbrequire(dbsiPtr.p->pdxNumDataPages <= 16);
+ for (Uint32 i = 0; i < dbsiPtr.p->pdxNumDataPages; i++) {
+ PagePtr pagePtr;
+ ljam();
+ pagePtr.i = dbsiPtr.p->pdxDataPage[i];
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ ndbrequire(pagePtr.p->pageWord[ZPAGE_STATE_POS] != 0);
+ ndbrequire(pagePtr.p->pageWord[ZPAGE_STATE_POS] <= ZAC_MM_FREE_COPY);
+ ndbrequire(pagePtr.p->pageWord[ZPAGE_FRAG_PAGE_ID_POS] == ((dbsiPtr.p->pdxFilePage - 1) + i));
+ }//for
+ rfrLoadDataPagesLab(signal, riPtr, dbsiPtr);
+ break;
+ }
+ case CHECKPOINT_DATA_READ_PAGE_ZERO:
+ {
+ ljam();
+ rfrInitRestartInfoLab(signal, dbsiPtr);
+ break;
+ }
+ case CHECKPOINT_UNDO_READ:
+ {
+ LocalLogInfoPtr lliPtr;
+ ljam();
+ lliPtr.i = dbsiPtr.p->pdxCheckpointInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ xlcGetNextRecordLab(signal, dbsiPtr, lliPtr);
+ break;
+ }
+ case CHECKPOINT_UNDO_READ_FIRST:
+ ljam();
+ rfrReadSecondUndoLogLab(signal, dbsiPtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+}//Dbtup::execFSREADCONF()
+
+void Dbtup::execFSREADREF(Signal* signal)
+{
+ ljamEntry();
+ ndbrequire(false);
+}//Dbtup::execFSREADREF()
+
+void Dbtup::execFSWRITECONF(Signal* signal)
+{
+ DiskBufferSegmentInfoPtr dbsiPtr;
+
+ ljamEntry();
+ dbsiPtr.i = signal->theData[0];
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ switch (dbsiPtr.p->pdxOperation) {
+ case CHECKPOINT_DATA_WRITE:
+ ljam();
+ lcpSaveDataPageLab(signal, dbsiPtr.p->pdxCheckpointInfoP);
+ break;
+ case CHECKPOINT_DATA_WRITE_LAST:
+ {
+ CheckpointInfoPtr ciPtr;
+ ljam();
+ ciPtr.i = dbsiPtr.p->pdxCheckpointInfoP;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ lcpFlushLogLab(signal, ciPtr);
+ break;
+ }
+ case CHECKPOINT_DATA_WRITE_FLUSH:
+ {
+ ljam();
+ Uint32 ciIndex = dbsiPtr.p->pdxCheckpointInfoP;
+ freeDiskBufferSegmentRecord(signal, dbsiPtr);
+ lcpCompletedLab(signal, ciIndex);
+ break;
+ }
+ case CHECKPOINT_UNDO_WRITE_FLUSH:
+ {
+ ljam();
+ Uint32 ciIndex = dbsiPtr.p->pdxCheckpointInfoP;
+ freeDiskBufferSegmentRecord(signal, dbsiPtr);
+ lcpFlushRestartInfoLab(signal, ciIndex);
+ break;
+ }
+ case CHECKPOINT_UNDO_WRITE:
+ ljam();
+ freeDiskBufferSegmentRecord(signal, dbsiPtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ return;
+}//Dbtup::execFSWRITECONF()
+
+void Dbtup::execFSWRITEREF(Signal* signal)
+{
+ ljamEntry();
+ ndbrequire(false);
+}//Dbtup::execFSWRITEREF()
+
+void Dbtup::execCONTINUEB(Signal* signal)
+{
+ ljamEntry();
+ Uint32 actionType = signal->theData[0];
+ Uint32 dataPtr = signal->theData[1];
+ switch (actionType) {
+ case ZSTART_EXEC_UNDO_LOG:
+ ljam();
+ startExecUndoLogLab(signal, dataPtr);
+ break;
+ case ZCONT_SAVE_DP:
+ ljam();
+ lcpSaveDataPageLab(signal, dataPtr);
+ break;
+ case ZCONT_START_SAVE_CL:
+ {
+ CheckpointInfoPtr ciPtr;
+
+ ljam();
+ ciPtr.i = dataPtr;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ lcpSaveCopyListLab(signal, ciPtr);
+ break;
+ }
+ case ZCONT_EXECUTE_LC:
+ {
+ LocalLogInfoPtr lliPtr;
+ DiskBufferSegmentInfoPtr dbsiPtr;
+
+ ljam();
+ lliPtr.i = dataPtr;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ dbsiPtr.i = lliPtr.p->lliUndoBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ xlcGetNextRecordLab(signal, dbsiPtr, lliPtr);
+ break;
+ }
+ case ZCONT_LOAD_DP:
+ {
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ RestartInfoRecordPtr riPtr;
+
+ ljam();
+ riPtr.i = dataPtr;
+ ptrCheckGuard(riPtr, cnoOfRestartInfoRec, restartInfoRecord);
+ dbsiPtr.i = riPtr.p->sriDataBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ rfrLoadDataPagesLab(signal, riPtr, dbsiPtr);
+ break;
+ }
+ case ZLOAD_BAL_LCP_TIMER:
+ ljam();
+ clblPageCounter = clblPagesPerTick;
+ signal->theData[0] = ZLOAD_BAL_LCP_TIMER;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 400, 1);
+ break;
+ case ZINITIALISE_RECORDS:
+ ljam();
+ initialiseRecordsLab(signal, dataPtr);
+ break;
+ case ZREL_FRAG:
+ ljam();
+ releaseFragment(signal, dataPtr);
+ break;
+ case ZREPORT_MEMORY_USAGE:{
+ ljam();
+ static int c_currentMemUsed = 0;
+ int now = (cnoOfAllocatedPages * 100)/cnoOfPage;
+ const int thresholds[] = { 100, 90, 80, 0 };
+
+ Uint32 i = 0;
+ const Uint32 sz = sizeof(thresholds)/sizeof(thresholds[0]);
+ for(i = 0; i<sz; i++){
+ if(now >= thresholds[i]){
+ now = thresholds[i];
+ break;
+ }
+ }
+
+ if(now != c_currentMemUsed){
+ reportMemoryUsage(signal, now > c_currentMemUsed ? 1 : -1);
+ c_currentMemUsed = now;
+ }
+ signal->theData[0] = ZREPORT_MEMORY_USAGE;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 2000, 1);
+ return;
+ }
+ case ZBUILD_INDEX:
+ ljam();
+ buildIndex(signal, dataPtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+}//Dbtup::execTUP_CONTINUEB()
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* ------------------- SYSTEM RESTART MODULE ---------------------- */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+void Dbtup::execSTTOR(Signal* signal)
+{
+ ljamEntry();
+ Uint32 startPhase = signal->theData[1];
+ Uint32 sigKey = signal->theData[6];
+ switch (startPhase) {
+ case ZSTARTPHASE1:
+ ljam();
+ CLEAR_ERROR_INSERT_VALUE;
+ cownref = calcTupBlockRef(0);
+ break;
+ default:
+ ljam();
+ break;
+ }//switch
+ signal->theData[0] = sigKey;
+ signal->theData[1] = 3;
+ signal->theData[2] = 2;
+ signal->theData[3] = ZSTARTPHASE1;
+ signal->theData[4] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 5, JBB);
+ return;
+}//Dbtup::execSTTOR()
+
+/************************************************************************************************/
+// SIZE_ALTREP INITIALIZE DATA STRUCTURES, FILES AND DS VARIABLES, GET READY FOR EXTERNAL
+// CONNECTIONS.
+/************************************************************************************************/
+void Dbtup::execSIZEALT_REP(Signal* signal)
+{
+ BlockReference tsizealtBlockRef;
+
+ ljamEntry();
+ tsizealtBlockRef = signal->theData[TupSizeAltReq::IND_BLOCK_REF];
+ cnoOfFragrec = signal->theData[TupSizeAltReq::IND_FRAG];
+ cnoOfOprec = signal->theData[TupSizeAltReq::IND_OP_RECS];
+ // MemorySpaceTuples is specified in 8k pages, divide by 4 for 32k pages
+ Uint64 noOfWords = (signal->theData[TupSizeAltReq::IND_PAGE] * 2048) + (ZWORDS_ON_PAGE - 1);
+ Uint64 noOfPages = noOfWords / (Uint64)ZWORDS_ON_PAGE;
+ cnoOfPage = (Uint32)noOfPages;
+ initPageRangeSize(signal->theData[TupSizeAltReq::IND_PAGE_RANGE]);
+ cnoOfTablerec = signal->theData[TupSizeAltReq::IND_TABLE];
+ cnoOfTabDescrRec = signal->theData[TupSizeAltReq::IND_TABLE_DESC];
+ Uint32 noOfStoredProc = signal->theData[TupSizeAltReq::IND_STORED_PROC];
+
+ cnoOfTabDescrRec = (cnoOfTabDescrRec & 0xFFFFFFF0) + 16;
+ c_storedProcPool.setSize(noOfStoredProc);
+ c_buildIndexPool.setSize(c_noOfBuildIndexRec);
+
+ initRecords();
+ czero = 0;
+ cminusOne = czero - 1;
+ clastBitMask = 1;
+ clastBitMask = clastBitMask << 31;
+ cnoOfLocalLogInfo = 0;
+ cnoFreeUndoSeg = 0;
+
+ initialiseRecordsLab(signal, 0);
+}//Dbtup::execSIZEALT_REP()
+
+void Dbtup::initRecords()
+{
+ // Records with dynamic sizes
+ attrbufrec = (Attrbufrec*)allocRecord("Attrbufrec",
+ sizeof(Attrbufrec),
+ cnoOfAttrbufrec);
+
+ checkpointInfo = (CheckpointInfo*)allocRecord("CheckpointInfo",
+ sizeof(CheckpointInfo),
+ cnoOfLcpRec);
+
+ diskBufferSegmentInfo = (DiskBufferSegmentInfo*)
+ allocRecord("DiskBufferSegmentInfo",
+ sizeof(DiskBufferSegmentInfo),
+ cnoOfConcurrentWriteOp);
+
+ fragoperrec = (Fragoperrec*)allocRecord("Fragoperrec",
+ sizeof(Fragoperrec),
+ cnoOfFragoprec);
+
+ fragrecord = (Fragrecord*)allocRecord("Fragrecord",
+ sizeof(Fragrecord),
+ cnoOfFragrec);
+
+ hostBuffer = (HostBuffer*)allocRecord("HostBuffer",
+ sizeof(HostBuffer),
+ MAX_NODES);
+
+ localLogInfo = (LocalLogInfo*)allocRecord("LocalLogInfo",
+ sizeof(LocalLogInfo),
+ cnoOfParallellUndoFiles);
+
+ operationrec = (Operationrec*)allocRecord("Operationrec",
+ sizeof(Operationrec),
+ cnoOfOprec);
+
+ page = (Page*)allocRecord("Page",
+ sizeof(Page),
+ cnoOfPage);
+
+ pageRange = (PageRange*)allocRecord("PageRange",
+ sizeof(PageRange),
+ cnoOfPageRangeRec);
+
+ pendingFileOpenInfo = (PendingFileOpenInfo*)
+ allocRecord("PendingFileOpenInfo",
+ sizeof(PendingFileOpenInfo),
+ cnoOfConcurrentOpenOp);
+
+ restartInfoRecord = (RestartInfoRecord*)
+ allocRecord("RestartInfoRecord",
+ sizeof(RestartInfoRecord),
+ cnoOfRestartInfoRec);
+
+ // Trigger data
+ c_triggerPool.setSize(cnoOfTablerec*c_maxTriggersPerTable);
+
+ tablerec = (Tablerec*)allocRecord("Tablerec",
+ sizeof(Tablerec),
+ cnoOfTablerec);
+
+ for(unsigned i = 0; i<cnoOfTablerec; i++) {
+ void * p = &tablerec[i];
+ new (p) Tablerec(c_triggerPool);
+ }
+
+ tableDescriptor = (TableDescriptor*)
+ allocRecord("TableDescriptor",
+ sizeof(TableDescriptor),
+ cnoOfTabDescrRec);
+
+ undoPage = (UndoPage*)allocRecord("UndoPage",
+ sizeof(UndoPage),
+ cnoOfUndoPage);
+
+
+ // Initialize BAT for interface to file system
+ NewVARIABLE* bat = allocateBat(3);
+ bat[1].WA = &page->pageWord[0];
+ bat[1].nrr = cnoOfPage;
+ bat[1].ClusterSize = sizeof(Page);
+ bat[1].bits.q = 13; /* 8192 words/page */
+ bat[1].bits.v = 5;
+ bat[2].WA = &undoPage->undoPageWord[0];
+ bat[2].nrr = cnoOfUndoPage;
+ bat[2].ClusterSize = sizeof(UndoPage);
+ bat[2].bits.q = 13; /* 8192 words/page */
+ bat[2].bits.v = 5;
+}//Dbtup::initRecords()
+
+void Dbtup::initialiseRecordsLab(Signal* signal, Uint32 switchData)
+{
+ switch (switchData) {
+ case 0:
+ ljam();
+ initializeHostBuffer();
+ break;
+ case 1:
+ ljam();
+ initializeOperationrec();
+ break;
+ case 2:
+ ljam();
+ initializePage();
+ break;
+ case 3:
+ ljam();
+ initializeUndoPage();
+ break;
+ case 4:
+ ljam();
+ initializeTablerec();
+ break;
+ case 5:
+ ljam();
+ initializeCheckpointInfoRec();
+ break;
+ case 6:
+ ljam();
+ initializeFragrecord();
+ break;
+ case 7:
+ ljam();
+ initializeFragoperrec();
+ break;
+ case 8:
+ ljam();
+ initializePageRange();
+ break;
+ case 9:
+ ljam();
+ initializeTabDescr();
+ break;
+ case 10:
+ ljam();
+ initializeDiskBufferSegmentRecord();
+ break;
+ case 11:
+ ljam();
+ initializeLocalLogInfo();
+ break;
+ case 12:
+ ljam();
+ initializeAttrbufrec();
+ break;
+ case 13:
+ ljam();
+ initializePendingFileOpenInfoRecord();
+ break;
+ case 14:
+ ljam();
+ initializeRestartInfoRec();
+ signal->theData[0] = cownref;
+ sendSignal(CMVMI_REF, GSN_SIZEALT_ACK, signal, 1, JBB);
+ return;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+/******************************************************************************/
+/* SEND REAL-TIME BREAK DURING INITIALISATION OF VARIABLES IN SYSTEM RESTART */
+/******************************************************************************/
+ signal->theData[0] = ZINITIALISE_RECORDS;
+ signal->theData[1] = switchData + 1;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+}//Dbtup::initialiseRecordsLab()
+
+void Dbtup::execNDB_STTOR(Signal* signal)
+{
+ ljamEntry();
+ cndbcntrRef = signal->theData[0];
+ Uint32 ownNodeId = signal->theData[1];
+ Uint32 startPhase = signal->theData[2];
+ Uint32 data1 = signal->theData[14];
+ Uint32 data2 = signal->theData[15];
+ switch (startPhase) {
+ case ZSTARTPHASE1:
+ ljam();
+ cownNodeId = ownNodeId;
+ cownref = calcTupBlockRef(ownNodeId);
+ break;
+ case ZSTARTPHASE2:
+ ljam();
+ break;
+ case ZSTARTPHASE3:
+ ljam();
+ startphase3Lab(signal, data1, data2);
+ break;
+ case ZSTARTPHASE4:
+ ljam();
+ break;
+ case ZSTARTPHASE6:
+ ljam();
+/*****************************************/
+/* NOW SET THE DISK WRITE SPEED TO */
+/* PAGES PER TICK AFTER SYSTEM */
+/* RESTART. */
+/*****************************************/
+ clblPagesPerTick = clblPagesPerTickAfterSr;
+
+ signal->theData[0] = ZREPORT_MEMORY_USAGE;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 2000, 1);
+ break;
+ default:
+ ljam();
+ break;
+ }//switch
+ signal->theData[0] = cownref;
+ sendSignal(cndbcntrRef, GSN_NDB_STTORRY, signal, 1, JBB);
+}//Dbtup::execNDB_STTOR()
+
+void Dbtup::startphase3Lab(Signal* signal, Uint32 config1, Uint32 config2)
+{
+ if (config1 > 0) {
+ ljam();
+ clblPagesPerTick = config1;
+ } else {
+ ljam();
+ clblPagesPerTick = 1;
+ }//if
+ if (config2 > 0) {
+ ljam();
+ clblPagesPerTickAfterSr = config2;
+ } else {
+ ljam();
+ clblPagesPerTickAfterSr = 1;
+ }//if
+ clblPageCounter = clblPagesPerTick;
+ signal->theData[0] = ZLOAD_BAL_LCP_TIMER;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 100, 1);
+}//Dbtup::startphase3Lab()
+
+void Dbtup::initializeAttrbufrec()
+{
+ AttrbufrecPtr attrBufPtr;
+ for (attrBufPtr.i = 0;
+ attrBufPtr.i < cnoOfAttrbufrec; attrBufPtr.i++) {
+ ptrAss(attrBufPtr, attrbufrec);
+ attrBufPtr.p->attrbuf[ZBUF_NEXT] = attrBufPtr.i + 1;
+ }//for
+ attrBufPtr.i = cnoOfAttrbufrec - 1;
+ ptrAss(attrBufPtr, attrbufrec);
+ attrBufPtr.p->attrbuf[ZBUF_NEXT] = RNIL;
+ cfirstfreeAttrbufrec = 0;
+ cnoFreeAttrbufrec = cnoOfAttrbufrec;
+}//Dbtup::initializeAttrbufrec()
+
+void Dbtup::initializeCheckpointInfoRec()
+{
+ CheckpointInfoPtr checkpointInfoPtr;
+ for (checkpointInfoPtr.i = 0;
+ checkpointInfoPtr.i < cnoOfLcpRec; checkpointInfoPtr.i++) {
+ ptrAss(checkpointInfoPtr, checkpointInfo);
+ checkpointInfoPtr.p->lcpNextRec = checkpointInfoPtr.i + 1;
+ }//for
+ checkpointInfoPtr.i = cnoOfLcpRec - 1;
+ ptrAss(checkpointInfoPtr, checkpointInfo);
+ checkpointInfoPtr.p->lcpNextRec = RNIL;
+ cfirstfreeLcp = 0;
+}//Dbtup::initializeCheckpointInfoRec()
+
+void Dbtup::initializeDiskBufferSegmentRecord()
+{
+ DiskBufferSegmentInfoPtr diskBufferSegmentPtr;
+ for (diskBufferSegmentPtr.i = 0;
+ diskBufferSegmentPtr.i < cnoOfConcurrentWriteOp; diskBufferSegmentPtr.i++) {
+ ptrAss(diskBufferSegmentPtr, diskBufferSegmentInfo);
+ diskBufferSegmentPtr.p->pdxNextRec = diskBufferSegmentPtr.i + 1;
+ diskBufferSegmentPtr.p->pdxBuffertype = NOT_INITIALIZED;
+ }//for
+ diskBufferSegmentPtr.i = cnoOfConcurrentWriteOp - 1;
+ ptrAss(diskBufferSegmentPtr, diskBufferSegmentInfo);
+ diskBufferSegmentPtr.p->pdxNextRec = RNIL;
+ cfirstfreePdx = 0;
+}//Dbtup::initializeDiskBufferSegmentRecord()
+
+void Dbtup::initializeFragoperrec()
+{
+ FragoperrecPtr fragoperPtr;
+ for (fragoperPtr.i = 0; fragoperPtr.i < cnoOfFragoprec; fragoperPtr.i++) {
+ ptrAss(fragoperPtr, fragoperrec);
+ fragoperPtr.p->nextFragoprec = fragoperPtr.i + 1;
+ }//for
+ fragoperPtr.i = cnoOfFragoprec - 1;
+ ptrAss(fragoperPtr, fragoperrec);
+ fragoperPtr.p->nextFragoprec = RNIL;
+ cfirstfreeFragopr = 0;
+}//Dbtup::initializeFragoperrec()
+
+void Dbtup::initializeFragrecord()
+{
+ FragrecordPtr regFragPtr;
+ for (regFragPtr.i = 0; regFragPtr.i < cnoOfFragrec; regFragPtr.i++) {
+ ptrAss(regFragPtr, fragrecord);
+ regFragPtr.p->nextfreefrag = regFragPtr.i + 1;
+ regFragPtr.p->checkpointVersion = RNIL;
+ regFragPtr.p->firstusedOprec = RNIL;
+ regFragPtr.p->fragStatus = IDLE;
+ }//for
+ regFragPtr.i = cnoOfFragrec - 1;
+ ptrAss(regFragPtr, fragrecord);
+ regFragPtr.p->nextfreefrag = RNIL;
+ cfirstfreefrag = 0;
+}//Dbtup::initializeFragrecord()
+
+void Dbtup::initializeHostBuffer()
+{
+ Uint32 hostId;
+ cpackedListIndex = 0;
+ for (hostId = 0; hostId < MAX_NODES; hostId++) {
+ hostBuffer[hostId].inPackedList = false;
+ hostBuffer[hostId].noOfPacketsTA = 0;
+ hostBuffer[hostId].noOfPacketsRC = 0;
+ hostBuffer[hostId].packetLenTA = 0;
+ hostBuffer[hostId].packetLenRC = 0;
+ }//for
+}//Dbtup::initializeHostBuffer()
+
+void Dbtup::initializeLocalLogInfo()
+{
+ LocalLogInfoPtr localLogInfoPtr;
+ for (localLogInfoPtr.i = 0;
+ localLogInfoPtr.i < cnoOfParallellUndoFiles; localLogInfoPtr.i++) {
+ ptrAss(localLogInfoPtr, localLogInfo);
+ localLogInfoPtr.p->lliActiveLcp = 0;
+ localLogInfoPtr.p->lliUndoFileHandle = RNIL;
+ }//for
+}//Dbtup::initializeLocalLogInfo()
+
+void Dbtup::initializeOperationrec()
+{
+ OperationrecPtr regOpPtr;
+ for (regOpPtr.i = 0; regOpPtr.i < cnoOfOprec; regOpPtr.i++) {
+ ptrAss(regOpPtr, operationrec);
+ regOpPtr.p->firstAttrinbufrec = RNIL;
+ regOpPtr.p->lastAttrinbufrec = RNIL;
+ regOpPtr.p->prevOprecInList = RNIL;
+ regOpPtr.p->nextOprecInList = regOpPtr.i + 1;
+ regOpPtr.p->optype = ZREAD;
+ regOpPtr.p->inFragList = ZFALSE;
+ regOpPtr.p->inActiveOpList = ZFALSE;
+/* FOR ABORT HANDLING BEFORE ANY SUCCESSFUL OPERATION */
+ regOpPtr.p->transstate = DISCONNECTED;
+ regOpPtr.p->storedProcedureId = ZNIL;
+ regOpPtr.p->prevActiveOp = RNIL;
+ regOpPtr.p->nextActiveOp = RNIL;
+ regOpPtr.p->tupVersion = ZNIL;
+ regOpPtr.p->deleteInsertFlag = 0;
+ }//for
+ regOpPtr.i = cnoOfOprec - 1;
+ ptrAss(regOpPtr, operationrec);
+ regOpPtr.p->nextOprecInList = RNIL;
+ cfirstfreeOprec = 0;
+}//Dbtup::initializeOperationrec()
+
+void Dbtup::initializePendingFileOpenInfoRecord()
+{
+ PendingFileOpenInfoPtr pendingFileOpenInfoPtr;
+ for (pendingFileOpenInfoPtr.i = 0;
+ pendingFileOpenInfoPtr.i < cnoOfConcurrentOpenOp; pendingFileOpenInfoPtr.i++) {
+ ptrAss(pendingFileOpenInfoPtr, pendingFileOpenInfo);
+ pendingFileOpenInfoPtr.p->pfoNextRec = pendingFileOpenInfoPtr.i + 1;
+ }//for
+ pendingFileOpenInfoPtr.i = cnoOfConcurrentOpenOp - 1;
+ ptrAss(pendingFileOpenInfoPtr, pendingFileOpenInfo);
+ pendingFileOpenInfoPtr.p->pfoNextRec = RNIL;
+ cfirstfreePfo = 0;
+}//Dbtup::initializePendingFileOpenInfoRecord()
+
+void Dbtup::initializeRestartInfoRec()
+{
+ RestartInfoRecordPtr restartInfoPtr;
+ for (restartInfoPtr.i = 0; restartInfoPtr.i < cnoOfRestartInfoRec; restartInfoPtr.i++) {
+ ptrAss(restartInfoPtr, restartInfoRecord);
+ restartInfoPtr.p->sriNextRec = restartInfoPtr.i + 1;
+ }//for
+ restartInfoPtr.i = cnoOfRestartInfoRec - 1;
+ ptrAss(restartInfoPtr, restartInfoRecord);
+ restartInfoPtr.p->sriNextRec = RNIL;
+ cfirstfreeSri = 0;
+}//Dbtup::initializeRestartInfoRec()
+
+void Dbtup::initializeTablerec()
+{
+ TablerecPtr regTabPtr;
+ for (regTabPtr.i = 0; regTabPtr.i < cnoOfTablerec; regTabPtr.i++) {
+ ljam();
+ ptrAss(regTabPtr, tablerec);
+ initTab(regTabPtr.p);
+ }//for
+}//Dbtup::initializeTablerec()
+
+void
+Dbtup::initTab(Tablerec* const regTabPtr)
+{
+ for (Uint32 i = 0; i < (2 * NO_OF_FRAG_PER_NODE); i++) {
+ regTabPtr->fragid[i] = RNIL;
+ regTabPtr->fragrec[i] = RNIL;
+ }//for
+ regTabPtr->readFunctionArray = NULL;
+ regTabPtr->updateFunctionArray = NULL;
+
+ regTabPtr->tabDescriptor = RNIL;
+ regTabPtr->attributeGroupDescriptor = RNIL;
+ regTabPtr->readKeyArray = RNIL;
+
+ regTabPtr->checksumIndicator = false;
+ regTabPtr->GCPIndicator = false;
+
+ regTabPtr->noOfAttr = 0;
+ regTabPtr->noOfKeyAttr = 0;
+ regTabPtr->noOfNewAttr = 0;
+ regTabPtr->noOfAttributeGroups = 0;
+
+ regTabPtr->tupheadsize = 0;
+ regTabPtr->tupNullIndex = 0;
+ regTabPtr->tupNullWords = 0;
+ regTabPtr->tupChecksumIndex = 0;
+ regTabPtr->tupGCPIndex = 0;
+
+ regTabPtr->m_dropTable.tabUserPtr = RNIL;
+ regTabPtr->m_dropTable.tabUserRef = 0;
+ regTabPtr->tableStatus = NOT_DEFINED;
+
+ // Clear trigger data
+ if (!regTabPtr->afterInsertTriggers.isEmpty())
+ regTabPtr->afterInsertTriggers.release();
+ if (!regTabPtr->afterDeleteTriggers.isEmpty())
+ regTabPtr->afterDeleteTriggers.release();
+ if (!regTabPtr->afterUpdateTriggers.isEmpty())
+ regTabPtr->afterUpdateTriggers.release();
+ if (!regTabPtr->subscriptionInsertTriggers.isEmpty())
+ regTabPtr->subscriptionInsertTriggers.release();
+ if (!regTabPtr->subscriptionDeleteTriggers.isEmpty())
+ regTabPtr->subscriptionDeleteTriggers.release();
+ if (!regTabPtr->subscriptionUpdateTriggers.isEmpty())
+ regTabPtr->subscriptionUpdateTriggers.release();
+ if (!regTabPtr->constraintUpdateTriggers.isEmpty())
+ regTabPtr->constraintUpdateTriggers.release();
+ if (!regTabPtr->tuxCustomTriggers.isEmpty())
+ regTabPtr->tuxCustomTriggers.release();
+}//Dbtup::initTab()
+
+void Dbtup::initializeTabDescr()
+{
+ TableDescriptorPtr regTabDesPtr;
+ for (Uint32 i = 0; i < 16; i++) {
+ cfreeTdList[i] = RNIL;
+ }//for
+ for (regTabDesPtr.i = 0; regTabDesPtr.i < cnoOfTabDescrRec; regTabDesPtr.i++) {
+ ptrAss(regTabDesPtr, tableDescriptor);
+ regTabDesPtr.p->tabDescr = RNIL;
+ }//for
+ freeTabDescr(0, cnoOfTabDescrRec);
+}//Dbtup::initializeTabDescr()
+
+void Dbtup::initializeUndoPage()
+{
+ UndoPagePtr undoPagep;
+ for (undoPagep.i = 0;
+ undoPagep.i < cnoOfUndoPage;
+ undoPagep.i = undoPagep.i + ZUB_SEGMENT_SIZE) {
+ ptrAss(undoPagep, undoPage);
+ undoPagep.p->undoPageWord[ZPAGE_NEXT_POS] = undoPagep.i +
+ ZUB_SEGMENT_SIZE;
+ cnoFreeUndoSeg++;
+ }//for
+ undoPagep.i = cnoOfUndoPage - ZUB_SEGMENT_SIZE;
+ ptrAss(undoPagep, undoPage);
+ undoPagep.p->undoPageWord[ZPAGE_NEXT_POS] = RNIL;
+ cfirstfreeUndoSeg = 0;
+}//Dbtup::initializeUndoPage()
+
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+/* --------------- CONNECT/DISCONNECT MODULE ---------------------- */
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::execTUPSEIZEREQ(Signal* signal)
+{
+ OperationrecPtr regOperPtr;
+ ljamEntry();
+ Uint32 userPtr = signal->theData[0];
+ BlockReference userRef = signal->theData[1];
+ if (cfirstfreeOprec != RNIL) {
+ ljam();
+ seizeOpRec(regOperPtr);
+ } else {
+ ljam();
+ signal->theData[0] = userPtr;
+ signal->theData[1] = ZGET_OPREC_ERROR;
+ sendSignal(userRef, GSN_TUPSEIZEREF, signal, 2, JBB);
+ return;
+ }//if
+ regOperPtr.p->optype = ZREAD;
+ initOpConnection(regOperPtr.p);
+ regOperPtr.p->userpointer = userPtr;
+ regOperPtr.p->userblockref = userRef;
+ signal->theData[0] = regOperPtr.p->userpointer;
+ signal->theData[1] = regOperPtr.i;
+ sendSignal(userRef, GSN_TUPSEIZECONF, signal, 2, JBB);
+ return;
+}//Dbtup::execTUPSEIZEREQ()
+
+#define printFragment(t){ for(Uint32 i = 0; i < (2 * NO_OF_FRAG_PER_NODE);i++){\
+ ndbout_c("table = %d fragid[%d] = %d fragrec[%d] = %d", \
+ t.i, t.p->fragid[i], i, t.p->fragrec[i]); }}
+
+void Dbtup::execTUPRELEASEREQ(Signal* signal)
+{
+ OperationrecPtr regOperPtr;
+ ljamEntry();
+ regOperPtr.i = signal->theData[0];
+ ptrCheckGuard(regOperPtr, cnoOfOprec, operationrec);
+ regOperPtr.p->transstate = DISCONNECTED;
+ regOperPtr.p->nextOprecInList = cfirstfreeOprec;
+ cfirstfreeOprec = regOperPtr.i;
+ signal->theData[0] = regOperPtr.p->userpointer;
+ sendSignal(regOperPtr.p->userblockref, GSN_TUPRELEASECONF, signal, 1, JBB);
+ return;
+}//Dbtup::execTUPRELEASEREQ()
+
+/* ---------------------------------------------------------------- */
+/* ---------------- FREE_DISK_BUFFER_SEGMENT_RECORD --------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* THIS ROUTINE DEALLOCATES A DISK SEGMENT AND ITS DATA PAGES */
+/* */
+/* INPUT: DISK_BUFFER_SEGMENT_PTR THE DISK SEGMENT */
+/* */
+/* -----------------------------------------------------------------*/
+void Dbtup::freeDiskBufferSegmentRecord(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr)
+{
+ switch (dbsiPtr.p->pdxBuffertype) {
+ case UNDO_PAGES:
+ case COMMON_AREA_PAGES:
+ ljam();
+ freeUndoBufferPages(signal, dbsiPtr);
+ break;
+ case UNDO_RESTART_PAGES:
+ ljam();
+ dbsiPtr.p->pdxDataPage[0] = dbsiPtr.p->pdxUndoBufferSet[0];
+ freeUndoBufferPages(signal, dbsiPtr);
+ dbsiPtr.p->pdxDataPage[0] = dbsiPtr.p->pdxUndoBufferSet[1];
+ freeUndoBufferPages(signal, dbsiPtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ releaseDiskBufferSegmentRecord(dbsiPtr);
+}//Dbtup::freeDiskBufferSegmentRecord()
+
+/* ---------------------------------------------------------------- */
+/* -------------------- FREE_UNDO_BUFFER_PAGES -------------------- */
+/* ---------------------------------------------------------------- */
+/* */
+/* THIS ROUTINE DEALLOCATES A SEGMENT OF UNDO PAGES */
+/* */
+/* INPUT: UNDO_PAGEP POINTER TO FIRST PAGE IN SEGMENT */
+/* */
+/* -----------------------------------------------------------------*/
+void Dbtup::freeUndoBufferPages(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr)
+{
+ UndoPagePtr undoPagePtr;
+
+ undoPagePtr.i = dbsiPtr.p->pdxDataPage[0];
+ ptrCheckGuard(undoPagePtr, cnoOfUndoPage, undoPage);
+ undoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS] = cfirstfreeUndoSeg;
+ cfirstfreeUndoSeg = undoPagePtr.i;
+ cnoFreeUndoSeg++;
+ if (cnoFreeUndoSeg == ZMIN_PAGE_LIMIT_TUP_COMMITREQ) {
+ EXECUTE_DIRECT(DBLQH, GSN_TUP_COM_UNBLOCK, signal, 1);
+ ljamEntry();
+ }//if
+}//Dbtup::freeUndoBufferPages()
+
+void Dbtup::releaseCheckpointInfoRecord(CheckpointInfoPtr ciPtr)
+{
+ ciPtr.p->lcpNextRec = cfirstfreeLcp;
+ cfirstfreeLcp = ciPtr.i;
+}//Dbtup::releaseCheckpointInfoRecord()
+
+void Dbtup::releaseDiskBufferSegmentRecord(DiskBufferSegmentInfoPtr dbsiPtr)
+{
+ dbsiPtr.p->pdxNextRec = cfirstfreePdx;
+ cfirstfreePdx = dbsiPtr.i;
+}//Dbtup::releaseDiskBufferSegmentRecord()
+
+void Dbtup::releaseFragrec(FragrecordPtr regFragPtr)
+{
+ regFragPtr.p->nextfreefrag = cfirstfreefrag;
+ cfirstfreefrag = regFragPtr.i;
+}//Dbtup::releaseFragrec()
+
+void Dbtup::releasePendingFileOpenInfoRecord(PendingFileOpenInfoPtr pfoPtr)
+{
+ pfoPtr.p->pfoNextRec = cfirstfreePfo;
+ cfirstfreePfo = pfoPtr.i;
+}//Dbtup::releasePendingFileOpenInfoRecord()
+
+void Dbtup::releaseRestartInfoRecord(RestartInfoRecordPtr riPtr)
+{
+ riPtr.p->sriNextRec = cfirstfreeSri;
+ cfirstfreeSri = riPtr.i;
+}//Dbtup::releaseRestartInfoRecord()
+
+void Dbtup::seizeCheckpointInfoRecord(CheckpointInfoPtr& ciPtr)
+{
+ ciPtr.i = cfirstfreeLcp;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ cfirstfreeLcp = ciPtr.p->lcpNextRec;
+ ciPtr.p->lcpNextRec = RNIL;
+}//Dbtup::seizeCheckpointInfoRecord()
+
+void Dbtup::seizeDiskBufferSegmentRecord(DiskBufferSegmentInfoPtr& dbsiPtr)
+{
+ dbsiPtr.i = cfirstfreePdx;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ cfirstfreePdx = dbsiPtr.p->pdxNextRec;
+ dbsiPtr.p->pdxNextRec = RNIL;
+ for (Uint32 i = 0; i < 16; i++) {
+ dbsiPtr.p->pdxDataPage[i] = RNIL;
+ }//for
+ dbsiPtr.p->pdxCheckpointInfoP = RNIL;
+ dbsiPtr.p->pdxRestartInfoP = RNIL;
+ dbsiPtr.p->pdxLocalLogInfoP = RNIL;
+ dbsiPtr.p->pdxFilePage = 0;
+ dbsiPtr.p->pdxNumDataPages = 0;
+}//Dbtup::seizeDiskBufferSegmentRecord()
+
+void Dbtup::seizeOpRec(OperationrecPtr& regOperPtr)
+{
+ regOperPtr.i = cfirstfreeOprec;
+ ptrCheckGuard(regOperPtr, cnoOfOprec, operationrec);
+ cfirstfreeOprec = regOperPtr.p->nextOprecInList;
+}//Dbtup::seizeOpRec()
+
+void Dbtup::seizePendingFileOpenInfoRecord(PendingFileOpenInfoPtr& pfoiPtr)
+{
+ pfoiPtr.i = cfirstfreePfo;
+ ptrCheckGuard(pfoiPtr, cnoOfConcurrentOpenOp, pendingFileOpenInfo);
+ cfirstfreePfo = pfoiPtr.p->pfoNextRec;
+ pfoiPtr.p->pfoNextRec = RNIL;
+}//Dbtup::seizePendingFileOpenInfoRecord()
+
+void Dbtup::execSET_VAR_REQ(Signal* signal)
+{
+ SetVarReq* const setVarReq = (SetVarReq*)signal->getDataPtrSend();
+ ConfigParamId var = setVarReq->variable();
+ int val = setVarReq->value();
+
+ switch (var) {
+
+ case NoOfDiskPagesToDiskAfterRestartTUP:
+ clblPagesPerTick = val;
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ case NoOfDiskPagesToDiskDuringRestartTUP:
+ // Valid only during start so value not set.
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ default:
+ sendSignal(CMVMI_REF, GSN_SET_VAR_REF, signal, 1, JBB);
+ } // switch
+
+
+}//execSET_VAR_REQ()
+
+
+
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupIndex.cpp b/ndb/src/kernel/blocks/dbtup/DbtupIndex.cpp
new file mode 100644
index 00000000000..3a074f7fe5b
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupIndex.cpp
@@ -0,0 +1,576 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <AttributeDescriptor.hpp>
+#include "AttributeOffset.hpp"
+#include <AttributeHeader.hpp>
+#include <signaldata/TupAccess.hpp>
+#include <signaldata/TuxMaint.hpp>
+
+#define ljam() { jamLine(28000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(28000 + __LINE__); }
+
+// methods used by ordered index
+
+void
+Dbtup::execTUP_READ_ATTRS(Signal* signal)
+{
+ ljamEntry();
+ TupReadAttrs* const sig = (TupReadAttrs*)signal->getDataPtrSend();
+ TupReadAttrs reqCopy = *sig;
+ TupReadAttrs* const req = &reqCopy;
+ req->errorCode = 0;
+ // get table
+ TablerecPtr tablePtr;
+ tablePtr.i = req->tableId;
+ ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
+ // get fragment
+ FragrecordPtr fragPtr;
+ if (req->fragPtrI == RNIL) {
+ ljam();
+ getFragmentrec(fragPtr, req->fragId, tablePtr.p);
+ ndbrequire(fragPtr.i != RNIL);
+ req->fragPtrI = fragPtr.i;
+ } else {
+ fragPtr.i = req->fragPtrI;
+ ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
+ ndbrequire(req->fragId == fragPtr.p->fragmentId);
+ }
+ // get page
+ PagePtr pagePtr;
+ if (req->pageId == RNIL) {
+ ljam();
+ Uint32 fragPageId = req->tupAddr >> MAX_TUPLES_BITS;
+ Uint32 pageIndex = req->tupAddr & ((1 << MAX_TUPLES_BITS ) - 1);
+ ndbrequire((pageIndex & 0x1) == 0);
+ // data returned for original tuple
+ req->pageId = getRealpid(fragPtr.p, fragPageId);
+ req->pageOffset = ZPAGE_HEADER_SIZE + (pageIndex >> 1) * tablePtr.p->tupheadsize;
+ }
+ pagePtr.i = req->pageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 pageOffset = req->pageOffset;
+ // search for tuple version if not original
+ if (! (req->requestInfo & TupReadAttrs::ReadKeys) &&
+ pagePtr.p->pageWord[pageOffset + 1] != req->tupVersion) {
+ ljam();
+ OperationrecPtr opPtr;
+ opPtr.i = pagePtr.p->pageWord[pageOffset];
+ Uint32 loopGuard = 0;
+ while (true) {
+ ptrCheckGuard(opPtr, cnoOfOprec, operationrec);
+ if (opPtr.p->realPageIdC != RNIL) {
+ pagePtr.i = opPtr.p->realPageIdC;
+ pageOffset = opPtr.p->pageOffsetC;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ if (pagePtr.p->pageWord[pageOffset + 1] == req->tupVersion) {
+ ljam();
+ break;
+ }
+ }
+ ljam();
+ // next means before in event order
+ opPtr.i = opPtr.p->nextActiveOp;
+ ndbrequire(++loopGuard < (1 << ZTUP_VERSION_BITS));
+ }
+ }
+ // shared buffer
+ Uint32* buffer = (Uint32*)sig + TupReadAttrs::SignalLength;
+ // if request is for keys then we create input section
+ if (req->requestInfo & TupReadAttrs::ReadKeys) {
+ ljam();
+ buffer[0] = tablePtr.p->noOfKeyAttr;
+ const Uint32* keyArray = &tableDescriptor[tablePtr.p->readKeyArray].tabDescr;
+ MEMCOPY_NO_WORDS(&buffer[1], keyArray, tablePtr.p->noOfKeyAttr);
+ }
+ Uint32 inBufLen = buffer[0];
+ Uint32* inBuffer = &buffer[1];
+ Uint32* outBuffer = &buffer[1 + inBufLen];
+ Uint32 maxRead = ZATTR_BUFFER_SIZE;
+ // save globals
+ TablerecPtr tabptr_old = tabptr;
+ FragrecordPtr fragptr_old = fragptr;
+ OperationrecPtr operPtr_old = operPtr;
+ // new globals
+ tabptr = tablePtr;
+ fragptr = fragPtr;
+ operPtr.i = RNIL; // XXX check later
+ operPtr.p = NULL;
+ int ret = readAttributes(pagePtr.p, pageOffset, inBuffer, inBufLen, outBuffer, maxRead);
+ // restore globals
+ tabptr = tabptr_old;
+ fragptr = fragptr_old;
+ operPtr = operPtr_old;
+ // check error
+ if ((Uint32)ret == (Uint32)-1) {
+ ljam();
+ req->errorCode = terrorCode;
+ }
+ // copy back
+ *sig = *req;
+}
+
+void
+Dbtup::execTUP_QUERY_TH(Signal* signal)
+{
+ ljamEntry();
+ Operationrec tempOp;
+ TupQueryTh* const req = (TupQueryTh*)signal->getDataPtrSend();
+ Uint32 tableId = req->tableId;
+ Uint32 fragId = req->fragId;
+ Uint32 tupAddr = req->tupAddr;
+ Uint32 req_tupVersion = req->tupVersion;
+ Uint32 transid1 = req->transId1;
+ Uint32 transid2 = req->transId2;
+ Uint32 savePointId = req->savePointId;
+ Uint32 ret_result = 0;
+ // get table
+ TablerecPtr tablePtr;
+ tablePtr.i = tableId;
+ ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
+ // get fragment
+ FragrecordPtr fragPtr;
+ getFragmentrec(fragPtr, fragId, tablePtr.p);
+ ndbrequire(fragPtr.i != RNIL);
+ // get page
+ PagePtr pagePtr;
+ Uint32 fragPageId = tupAddr >> MAX_TUPLES_BITS;
+ Uint32 pageIndex = tupAddr & ((1 << MAX_TUPLES_BITS ) - 1);
+
+ tempOp.fragPageId = fragPageId;
+ tempOp.pageIndex = pageIndex;
+ tempOp.transid1 = transid1;
+ tempOp.transid2 = transid2;
+ tempOp.savePointId = savePointId;
+ tempOp.optype = ZREAD;
+ tempOp.dirtyOp = 1;
+ if (getPage(pagePtr, &tempOp, fragPtr.p, tablePtr.p)) {
+ /*
+ We use the normal getPage which will return the tuple to be used
+ for this transaction and savepoint id. If its tuple version equals
+ the requested then we have a visible tuple otherwise not.
+ */
+ jam();
+ Uint32 read_tupVersion = pagePtr.p->pageWord[tempOp.pageOffset + 1];
+ if (read_tupVersion == req_tupVersion) {
+ jam();
+ ret_result = 1;
+ }
+ }
+ req->returnCode = ret_result;
+ return;
+}
+
+void
+Dbtup::execTUP_STORE_TH(Signal* signal)
+{
+ ljamEntry();
+ TupStoreTh* const sig = (TupStoreTh*)signal->getDataPtrSend();
+ TupStoreTh reqCopy = *sig;
+ TupStoreTh* const req = &reqCopy;
+ req->errorCode = 0;
+ ndbrequire(req->tupVersion == 0);
+ // get table
+ TablerecPtr tablePtr;
+ tablePtr.i = req->tableId;
+ ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
+ // offset to attribute 0
+ Uint32 attrDescIndex = tablePtr.p->tabDescriptor + (0 << ZAD_LOG_SIZE);
+ Uint32 attrDataOffset = AttributeOffset::getOffset(tableDescriptor[attrDescIndex + 1].tabDescr);
+ // get fragment
+ FragrecordPtr fragPtr;
+ if (req->fragPtrI == RNIL) {
+ ljam();
+ getFragmentrec(fragPtr, req->fragId, tablePtr.p);
+ ndbrequire(fragPtr.i != RNIL);
+ req->fragPtrI = fragPtr.i;
+ } else {
+ fragPtr.i = req->fragPtrI;
+ ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
+ ndbrequire(req->fragId == fragPtr.p->fragmentId);
+ }
+ // handle each case
+ switch (req->opCode) {
+ case TupStoreTh::OpRead:
+ ljam();
+ {
+ PagePtr pagePtr;
+ if (req->pageId == RNIL) {
+ ljam();
+ Uint32 fragPageId = req->tupAddr >> MAX_TUPLES_BITS;
+ Uint32 pageIndex = req->tupAddr & ((1 << MAX_TUPLES_BITS ) - 1);
+ ndbrequire((pageIndex & 0x1) == 0);
+ req->pageId = getRealpid(fragPtr.p, fragPageId);
+ req->pageOffset = ZPAGE_HEADER_SIZE + (pageIndex >> 1) * tablePtr.p->tupheadsize;
+ }
+ pagePtr.i = req->pageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32* data = &pagePtr.p->pageWord[req->pageOffset] + attrDataOffset;
+ Uint32* buffer = (Uint32*)sig + TupStoreTh::SignalLength;
+ ndbrequire(req->dataOffset + req->dataSize <= tablePtr.p->tupheadsize);
+ memcpy(buffer + req->dataOffset, data + req->dataOffset, req->dataSize << 2);
+ }
+ break;
+ case TupStoreTh::OpInsert:
+ ljam();
+ {
+ PagePtr pagePtr;
+ if (! allocTh(fragPtr.p, tablePtr.p, NORMAL_PAGE, signal, req->pageOffset, pagePtr)) {
+ ljam();
+ req->errorCode = terrorCode;
+ break;
+ }
+ req->pageId = pagePtr.i;
+ Uint32 fragPageId = pagePtr.p->pageWord[ZPAGE_FRAG_PAGE_ID_POS];
+ Uint32 pageIndex = ((req->pageOffset - ZPAGE_HEADER_SIZE) / tablePtr.p->tupheadsize) << 1;
+ req->tupAddr = (fragPageId << MAX_TUPLES_BITS) | pageIndex;
+ ndbrequire(req->dataOffset + req->dataSize <= tablePtr.p->tupheadsize);
+ Uint32* data = &pagePtr.p->pageWord[req->pageOffset] + attrDataOffset;
+ Uint32* buffer = (Uint32*)sig + TupStoreTh::SignalLength;
+ memcpy(data + req->dataOffset, buffer + req->dataOffset, req->dataSize << 2);
+ }
+ break;
+ case TupStoreTh::OpUpdate:
+ ljam();
+ {
+ PagePtr pagePtr;
+ if (req->pageId == RNIL) {
+ ljam();
+ Uint32 fragPageId = req->tupAddr >> MAX_TUPLES_BITS;
+ Uint32 pageIndex = req->tupAddr & ((1 << MAX_TUPLES_BITS ) - 1);
+ ndbrequire((pageIndex & 0x1) == 0);
+ req->pageId = getRealpid(fragPtr.p, fragPageId);
+ req->pageOffset = ZPAGE_HEADER_SIZE + (pageIndex >> 1) * tablePtr.p->tupheadsize;
+ }
+ pagePtr.i = req->pageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32* data = &pagePtr.p->pageWord[req->pageOffset] + attrDataOffset;
+ Uint32* buffer = (Uint32*)sig + TupStoreTh::SignalLength;
+ ndbrequire(req->dataOffset + req->dataSize <= tablePtr.p->tupheadsize);
+ memcpy(data + req->dataOffset, buffer + req->dataOffset, req->dataSize << 2);
+ }
+ break;
+ case TupStoreTh::OpDelete:
+ ljam();
+ {
+ PagePtr pagePtr;
+ if (req->pageId == RNIL) {
+ ljam();
+ Uint32 fragPageId = req->tupAddr >> MAX_TUPLES_BITS;
+ Uint32 pageIndex = req->tupAddr & ((1 << MAX_TUPLES_BITS ) - 1);
+ ndbrequire((pageIndex & 0x1) == 0);
+ req->pageId = getRealpid(fragPtr.p, fragPageId);
+ req->pageOffset = ZPAGE_HEADER_SIZE + (pageIndex >> 1) * tablePtr.p->tupheadsize;
+ }
+ pagePtr.i = req->pageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ freeTh(fragPtr.p, tablePtr.p, signal, pagePtr.p, req->pageOffset);
+ // null location
+ req->tupAddr = (Uint32)-1;
+ req->pageId = RNIL;
+ req->pageOffset = 0;
+ }
+ break;
+ }
+ // copy back
+ *sig = *req;
+}
+
+// ordered index build
+
+//#define TIME_MEASUREMENT
+#ifdef TIME_MEASUREMENT
+ static Uint32 time_events;
+ NDB_TICKS tot_time_passed;
+ Uint32 number_events;
+#endif
+void
+Dbtup::execBUILDINDXREQ(Signal* signal)
+{
+ ljamEntry();
+#ifdef TIME_MEASUREMENT
+ time_events = 0;
+ tot_time_passed = 0;
+ number_events = 1;
+#endif
+ // get new operation
+ BuildIndexPtr buildPtr;
+ if (! c_buildIndexList.seize(buildPtr)) {
+ ljam();
+ BuildIndexRec buildRec;
+ memcpy(buildRec.m_request, signal->theData, sizeof(buildRec.m_request));
+ buildRec.m_errorCode = BuildIndxRef::Busy;
+ buildIndexReply(signal, &buildRec);
+ return;
+ }
+ memcpy(buildPtr.p->m_request, signal->theData, sizeof(buildPtr.p->m_request));
+ // check
+ buildPtr.p->m_errorCode = BuildIndxRef::NoError;
+ do {
+ const BuildIndxReq* buildReq = (const BuildIndxReq*)buildPtr.p->m_request;
+ if (buildReq->getTableId() >= cnoOfTablerec) {
+ ljam();
+ buildPtr.p->m_errorCode = BuildIndxRef::InvalidPrimaryTable;
+ break;
+ }
+ TablerecPtr tablePtr;
+ tablePtr.i = buildReq->getTableId();
+ ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
+ if (tablePtr.p->tableStatus != DEFINED) {
+ ljam();
+ buildPtr.p->m_errorCode = BuildIndxRef::InvalidPrimaryTable;
+ break;
+ }
+ if (! DictTabInfo::isOrderedIndex(buildReq->getIndexType())) {
+ ljam();
+ buildPtr.p->m_errorCode = BuildIndxRef::InvalidIndexType;
+ break;
+ }
+ const ArrayList<TupTriggerData>& triggerList = tablePtr.p->tuxCustomTriggers;
+ TriggerPtr triggerPtr;
+ triggerList.first(triggerPtr);
+ while (triggerPtr.i != RNIL) {
+ if (triggerPtr.p->indexId == buildReq->getIndexId()) {
+ ljam();
+ break;
+ }
+ triggerList.next(triggerPtr);
+ }
+ if (triggerPtr.i == RNIL) {
+ ljam();
+ // trigger was not created
+ buildPtr.p->m_errorCode = BuildIndxRef::InternalError;
+ break;
+ }
+ buildPtr.p->m_triggerPtrI = triggerPtr.i;
+ // set to first tuple position
+ buildPtr.p->m_fragNo = 0;
+ buildPtr.p->m_pageId = 0;
+ buildPtr.p->m_tupleNo = 0;
+ // start build
+ buildIndex(signal, buildPtr.i);
+ return;
+ } while (0);
+ // check failed
+ buildIndexReply(signal, buildPtr.p);
+ c_buildIndexList.release(buildPtr);
+}
+
+void
+Dbtup::buildIndex(Signal* signal, Uint32 buildPtrI)
+{
+ // get build record
+ BuildIndexPtr buildPtr;
+ buildPtr.i = buildPtrI;
+ c_buildIndexList.getPtr(buildPtr);
+ const BuildIndxReq* buildReq = (const BuildIndxReq*)buildPtr.p->m_request;
+ // get table
+ TablerecPtr tablePtr;
+ tablePtr.i = buildReq->getTableId();
+ ptrCheckGuard(tablePtr, cnoOfTablerec, tablerec);
+ // get trigger
+ TriggerPtr triggerPtr;
+ triggerPtr.i = buildPtr.p->m_triggerPtrI;
+ c_triggerPool.getPtr(triggerPtr);
+ ndbrequire(triggerPtr.p->indexId == buildReq->getIndexId());
+#ifdef TIME_MEASUREMENT
+ MicroSecondTimer start;
+ MicroSecondTimer stop;
+ NDB_TICKS time_passed;
+#endif
+ do {
+ // get fragment
+ FragrecordPtr fragPtr;
+ if (buildPtr.p->m_fragNo == 2 * NO_OF_FRAG_PER_NODE) {
+ ljam();
+ // build ready
+ buildIndexReply(signal, buildPtr.p);
+ c_buildIndexList.release(buildPtr);
+ return;
+ }
+ ndbrequire(buildPtr.p->m_fragNo < 2 * NO_OF_FRAG_PER_NODE);
+ fragPtr.i = tablePtr.p->fragrec[buildPtr.p->m_fragNo];
+ if (fragPtr.i == RNIL) {
+ ljam();
+ buildPtr.p->m_fragNo++;
+ buildPtr.p->m_pageId = 0;
+ buildPtr.p->m_tupleNo = 0;
+ break;
+ }
+ ptrCheckGuard(fragPtr, cnoOfFragrec, fragrecord);
+ // get page
+ PagePtr pagePtr;
+ if (buildPtr.p->m_pageId >= fragPtr.p->noOfPages) {
+ ljam();
+ buildPtr.p->m_fragNo++;
+ buildPtr.p->m_pageId = 0;
+ buildPtr.p->m_tupleNo = 0;
+ break;
+ }
+ pagePtr.i = getRealpid(fragPtr.p, buildPtr.p->m_pageId);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ const Uint32 pageState = pagePtr.p->pageWord[ZPAGE_STATE_POS];
+ if (pageState != ZTH_MM_FREE &&
+ pageState != ZTH_MM_FREE_COPY &&
+ pageState != ZTH_MM_FULL &&
+ pageState != ZTH_MM_FULL_COPY) {
+ ljam();
+ buildPtr.p->m_pageId++;
+ buildPtr.p->m_tupleNo = 0;
+ break;
+ }
+ // get tuple
+ const Uint32 tupheadsize = tablePtr.p->tupheadsize;
+ const Uint32 pageOffset = ZPAGE_HEADER_SIZE +
+ buildPtr.p->m_tupleNo * tupheadsize;
+ if (pageOffset + tupheadsize > ZWORDS_ON_PAGE) {
+ ljam();
+ buildPtr.p->m_pageId++;
+ buildPtr.p->m_tupleNo = 0;
+ break;
+ }
+ // skip over free tuple
+ bool isFree = false;
+ if (pageState == ZTH_MM_FREE ||
+ pageState == ZTH_MM_FREE_COPY) {
+ ljam();
+ if ((pagePtr.p->pageWord[pageOffset] >> 16) == tupheadsize) {
+ // verify it really is free XXX far too expensive
+ Uint32 nextTuple = pagePtr.p->pageWord[ZFREELIST_HEADER_POS] >> 16;
+ ndbrequire(nextTuple != 0);
+ while (nextTuple != 0) {
+ ljam();
+ if (nextTuple == pageOffset) {
+ ljam();
+ isFree = true;
+ break;
+ }
+ nextTuple = pagePtr.p->pageWord[nextTuple] & 0xffff;
+ }
+ }
+ }
+ if (isFree) {
+ ljam();
+ buildPtr.p->m_tupleNo++;
+ break;
+ }
+ OperationrecPtr pageOperPtr;
+ pageOperPtr.i = pagePtr.p->pageWord[pageOffset];
+ Uint32 pageId = buildPtr.p->m_pageId;
+ Uint32 pageIndex = buildPtr.p->m_tupleNo << 1;
+ if (pageOperPtr.i != RNIL) {
+ /*
+ If there is an ongoing operation on the tuple then it is either a
+ copy tuple or an original tuple with an ongoing transaction. In
+ both cases fragPageId and pageIndex refers to the original tuple.
+ The tuple address stored in TUX will always be the original tuple
+ but with the tuple version of the tuple we found.
+
+ This is necessary to avoid having to update TUX at abort of
+ update. If an update aborts then the copy tuple is copied to
+ the original tuple. The build will however have found that
+ tuple as a copy tuple. The original tuple is stable and is thus
+ preferrable to store in TUX.
+ */
+ jam();
+ ptrCheckGuard(pageOperPtr, cnoOfOprec, operationrec);
+ pageId = pageOperPtr.p->fragPageId;
+ pageIndex = pageOperPtr.p->pageIndex;
+ }//if
+ Uint32 tup_version = pagePtr.p->pageWord[pageOffset + 1];
+#ifdef TIME_MEASUREMENT
+ NdbTick_getMicroTimer(&start);
+#endif
+ // add to index
+ TuxMaintReq* const req = (TuxMaintReq*)signal->getDataPtrSend();
+ req->errorCode = RNIL;
+ req->tableId = tablePtr.i;
+ req->indexId = triggerPtr.p->indexId;
+ req->fragId = tablePtr.p->fragid[buildPtr.p->m_fragNo];
+ req->tupAddr = (pageId << MAX_TUPLES_BITS) | pageIndex;
+ req->tupVersion = tup_version;
+ req->opInfo = TuxMaintReq::OpAdd;
+ EXECUTE_DIRECT(DBTUX, GSN_TUX_MAINT_REQ,
+ signal, TuxMaintReq::SignalLength);
+ ljamEntry();
+ if (req->errorCode != 0) {
+ switch (req->errorCode) {
+ case TuxMaintReq::NoMemError:
+ ljam();
+ buildPtr.p->m_errorCode = BuildIndxRef::AllocationFailure;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+ buildIndexReply(signal, buildPtr.p);
+ c_buildIndexList.release(buildPtr);
+ return;
+ }
+#ifdef TIME_MEASUREMENT
+ NdbTick_getMicroTimer(&stop);
+ time_passed = NdbTick_getMicrosPassed(start, stop);
+ if (time_passed < 1000) {
+ time_events++;
+ tot_time_passed += time_passed;
+ if (time_events == number_events) {
+ NDB_TICKS mean_time_passed = tot_time_passed / (NDB_TICKS)number_events;
+ ndbout << "Number of events = " << number_events;
+ ndbout << " Mean time passed = " << mean_time_passed << endl;
+ number_events <<= 1;
+ tot_time_passed = (NDB_TICKS)0;
+ time_events = 0;
+ }//if
+ }
+#endif
+ // next tuple
+ buildPtr.p->m_tupleNo++;
+ break;
+ } while (0);
+ signal->theData[0] = ZBUILD_INDEX;
+ signal->theData[1] = buildPtr.i;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 2, JBB);
+}
+
+void
+Dbtup::buildIndexReply(Signal* signal, const BuildIndexRec* buildPtrP)
+{
+ const BuildIndxReq* const buildReq = (const BuildIndxReq*)buildPtrP->m_request;
+ // conf is subset of ref
+ BuildIndxRef* rep = (BuildIndxRef*)signal->getDataPtr();
+ rep->setUserRef(buildReq->getUserRef());
+ rep->setConnectionPtr(buildReq->getConnectionPtr());
+ rep->setRequestType(buildReq->getRequestType());
+ rep->setTableId(buildReq->getTableId());
+ rep->setIndexType(buildReq->getIndexType());
+ rep->setIndexId(buildReq->getIndexId());
+ // conf
+ if (buildPtrP->m_errorCode == BuildIndxRef::NoError) {
+ ljam();
+ sendSignal(rep->getUserRef(), GSN_BUILDINDXCONF,
+ signal, BuildIndxConf::SignalLength, JBB);
+ return;
+ }
+ // ref
+ rep->setErrorCode(buildPtrP->m_errorCode);
+ sendSignal(rep->getUserRef(), GSN_BUILDINDXREF,
+ signal, BuildIndxRef::SignalLength, JBB);
+}
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupLCP.cpp b/ndb/src/kernel/blocks/dbtup/DbtupLCP.cpp
new file mode 100644
index 00000000000..b74b2c00e3e
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupLCP.cpp
@@ -0,0 +1,593 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <signaldata/FsConf.hpp>
+#include <signaldata/FsRef.hpp>
+
+#define ljam() { jamLine(10000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(10000 + __LINE__); }
+
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+/* -------------------- LOCAL CHECKPOINT MODULE ------------------- */
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::execTUP_PREPLCPREQ(Signal* signal)
+{
+ CheckpointInfoPtr ciPtr;
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ FragrecordPtr regFragPtr;
+ LocalLogInfoPtr lliPtr;
+ TablerecPtr regTabPtr;
+
+ ljamEntry();
+ Uint32 userptr = signal->theData[0];
+ BlockReference userblockref = signal->theData[1];
+ regTabPtr.i = signal->theData[2];
+ ptrCheckGuard(regTabPtr, cnoOfTablerec, tablerec);
+ Uint32 fragId = signal->theData[3];
+ Uint32 checkpointNumber = signal->theData[4];
+ cundoFileVersion = signal->theData[5];
+
+ getFragmentrec(regFragPtr, fragId, regTabPtr.p);
+ ndbrequire(regTabPtr.i != RNIL);
+ seizeCheckpointInfoRecord(ciPtr);
+
+ lliPtr.i = (cundoFileVersion << 2) + (regTabPtr.i & 0x3);
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ cnoOfDataPagesToDiskWithoutSynch = 0;
+
+ ciPtr.p->lcpDataFileHandle = RNIL;
+ ciPtr.p->lcpCheckpointVersion = checkpointNumber;
+ ciPtr.p->lcpLocalLogInfoP = lliPtr.i;
+ ciPtr.p->lcpFragmentP = regFragPtr.i; /* SET THE FRAGMENT */
+ ciPtr.p->lcpFragmentId = fragId; /* SAVE THE FRAGMENT IDENTITY */
+ ciPtr.p->lcpTabPtr = regTabPtr.i; /* SET THE TABLE POINTER */
+ ciPtr.p->lcpBlockref = userblockref; /* SET THE BLOCK REFERENCE */
+ ciPtr.p->lcpUserptr = userptr; /* SET THE USERPOINTER */
+
+ /***************************************************************/
+ /* OPEN THE UNDO FILE FOR WRITE */
+ /* UPON FSOPENCONF */
+ /***************************************************************/
+ if (lliPtr.p->lliActiveLcp == 0) { /* IS THE UNDO LOG FILE OPEN? */
+ PendingFileOpenInfoPtr undoPfoiPtr;
+ UndoPagePtr regUndoPagePtr;
+
+ ljam();
+ lliPtr.p->lliPrevRecordId = 0;
+ lliPtr.p->lliLogFilePage = 0;
+ lliPtr.p->lliUndoPagesToDiskWithoutSynch = 0;
+ lliPtr.p->lliUndoWord = ZUNDO_PAGE_HEADER_SIZE;
+
+ seizeUndoBufferSegment(signal, regUndoPagePtr);
+ seizeDiskBufferSegmentRecord(dbsiPtr);
+ dbsiPtr.p->pdxBuffertype = UNDO_PAGES;
+ for (Uint32 i = 0; i < ZUB_SEGMENT_SIZE; i++) {
+ dbsiPtr.p->pdxDataPage[i] = regUndoPagePtr.i + i;
+ }//for
+ dbsiPtr.p->pdxFilePage = lliPtr.p->lliLogFilePage;
+ lliPtr.p->lliUndoPage = regUndoPagePtr.i;
+ lliPtr.p->lliUndoBufferSegmentP = dbsiPtr.i;
+ /* F LEVEL NOT USED */
+ Uint32 fileType = 1; /* VERSION */
+ fileType = (fileType << 8) | 2; /* .LOCLOG */
+ fileType = (fileType << 8) | 6; /* D6 */
+ fileType = (fileType << 8) | 0xff; /* DON'T USE P DIRECTORY LEVEL */
+ Uint32 fileFlag = 0x301; /* CREATE, WRITE ONLY, TRUNCATE */
+
+ seizePendingFileOpenInfoRecord(undoPfoiPtr);
+ undoPfoiPtr.p->pfoOpenType = LCP_UNDO_FILE_WRITE;
+ undoPfoiPtr.p->pfoCheckpointInfoP = ciPtr.i;
+
+ signal->theData[0] = cownref;
+ signal->theData[1] = undoPfoiPtr.i;
+ signal->theData[2] = lliPtr.i;
+ signal->theData[3] = 0xFFFFFFFF;
+ signal->theData[4] = cundoFileVersion;
+ signal->theData[5] = fileType;
+ signal->theData[6] = fileFlag;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+ }//if
+ /***************************************************************/
+ /* OPEN THE DATA FILE FOR WRITE */
+ /* THE FILE HANDLE WILL BE SET IN THE CHECKPOINT_INFO_RECORD */
+ /* UPON FSOPENCONF */
+ /***************************************************************/
+ /* OPEN THE DATA FILE IN THE FOLLOWING FORM */
+ /* D5/DBTUP/T<TABID>/F<FRAGID>/S<CHECKPOINT_NUMBER>.DATA */
+
+ PendingFileOpenInfoPtr dataPfoiPtr;
+
+ Uint32 fileType = 1; /* VERSION */
+ fileType = (fileType << 8) | 0; /* .DATA */
+ fileType = (fileType << 8) | 5; /* D5 */
+ fileType = (fileType << 8) | 0xff; /* DON'T USE P DIRECTORY LEVEL */
+ Uint32 fileFlag = 0x301; /* CREATE, WRITE ONLY, TRUNCATE */
+
+ seizePendingFileOpenInfoRecord(dataPfoiPtr); /* SEIZE A NEW FILE OPEN INFO */
+ if (lliPtr.p->lliActiveLcp == 0) {
+ ljam();
+ dataPfoiPtr.p->pfoOpenType = LCP_DATA_FILE_WRITE_WITH_UNDO;
+ } else {
+ ljam();
+ dataPfoiPtr.p->pfoOpenType = LCP_DATA_FILE_WRITE;
+ }//if
+ dataPfoiPtr.p->pfoCheckpointInfoP = ciPtr.i;
+
+ /* LET'S OPEN THE DATA FILE FOR WRITE */
+ /* INCREASE NUMBER OF ACTIVE CHECKPOINTS */
+ lliPtr.p->lliActiveLcp = 1;
+ signal->theData[0] = cownref;
+ signal->theData[1] = dataPfoiPtr.i;
+ signal->theData[2] = ciPtr.p->lcpTabPtr;
+ signal->theData[3] = ciPtr.p->lcpFragmentId;
+ signal->theData[4] = ciPtr.p->lcpCheckpointVersion;
+ signal->theData[5] = fileType;
+ signal->theData[6] = fileFlag;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+ return;
+}//Dbtup::execTUP_PREPLCPREQ()
+
+/* ---------------------------------------------------------------- */
+/* ------------------------ START CHECKPOINT --------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::execTUP_LCPREQ(Signal* signal)
+{
+ CheckpointInfoPtr ciPtr;
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ FragrecordPtr regFragPtr;
+ LocalLogInfoPtr lliPtr;
+
+ ljamEntry();
+// Uint32 userptr = signal->theData[0];
+// BlockReference userblockref = signal->theData[1];
+ ciPtr.i = signal->theData[2];
+
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ regFragPtr.i = ciPtr.p->lcpFragmentP;
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+
+/* ---------------------------------------------------------------- */
+/* ASSIGNING A VALUE DIFFERENT FROM RNIL TO CHECKPOINT VERSION*/
+/* TRIGGERS THAT UNDO LOGGING WILL START FOR THIS FRAGMENT. */
+/* WE ASSIGN IT THE POINTER TO THE CHECKPOINT RECORD FOR */
+/* OPTIMISATION OF THE WRITING OF THE UNDO LOG. */
+/* ---------------------------------------------------------------- */
+ regFragPtr.p->checkpointVersion = ciPtr.p->lcpLocalLogInfoP; /* MARK START OF UNDO LOGGING */
+
+ regFragPtr.p->maxPageWrittenInCheckpoint = getNoOfPages(regFragPtr.p);
+ regFragPtr.p->minPageNotWrittenInCheckpoint = 0;
+ ndbrequire(getNoOfPages(regFragPtr.p) > 0);
+ allocDataBufferSegment(signal, dbsiPtr);
+
+ dbsiPtr.p->pdxNumDataPages = 0;
+ dbsiPtr.p->pdxFilePage = 1;
+ ciPtr.p->lcpDataBufferSegmentP = dbsiPtr.i;
+ dbsiPtr.p->pdxCheckpointInfoP = ciPtr.i;
+ ciPtr.p->lcpNoOfPages = getNoOfPages(regFragPtr.p);
+ ciPtr.p->lcpNoCopyPagesAlloc = regFragPtr.p->noCopyPagesAlloc;
+ ciPtr.p->lcpEmptyPrimPage = regFragPtr.p->emptyPrimPage;
+ ciPtr.p->lcpThFreeFirst = regFragPtr.p->thFreeFirst;
+ ciPtr.p->lcpThFreeCopyFirst = regFragPtr.p->thFreeCopyFirst;
+ lliPtr.i = ciPtr.p->lcpLocalLogInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+/* ---------------------------------------------------------------- */
+/* --- PERFORM A COPY OF THE TABLE DESCRIPTOR FOR THIS FRAGMENT --- */
+/* ---------------------------------------------------------------- */
+ cprAddLogHeader(signal,
+ lliPtr.p,
+ ZTABLE_DESCRIPTOR,
+ ciPtr.p->lcpTabPtr,
+ ciPtr.p->lcpFragmentId);
+
+/* ---------------------------------------------------------------- */
+/* CONTINUE WITH SAVING ACTIVE OPERATIONS AFTER A REAL-TIME */
+/* BREAK. */
+/* ---------------------------------------------------------------- */
+ ciPtr.p->lcpTmpOperPtr = regFragPtr.p->firstusedOprec;
+ lcpSaveCopyListLab(signal, ciPtr);
+ return;
+}//Dbtup::execTUP_LCPREQ()
+
+void Dbtup::allocDataBufferSegment(Signal* signal, DiskBufferSegmentInfoPtr& dbsiPtr)
+{
+ UndoPagePtr regUndoPagePtr;
+
+ seizeDiskBufferSegmentRecord(dbsiPtr);
+ dbsiPtr.p->pdxBuffertype = COMMON_AREA_PAGES;
+ ndbrequire(cfirstfreeUndoSeg != RNIL);
+ if (cnoFreeUndoSeg == ZMIN_PAGE_LIMIT_TUP_COMMITREQ) {
+ EXECUTE_DIRECT(DBLQH, GSN_TUP_COM_BLOCK, signal, 1);
+ ljamEntry();
+ }//if
+ cnoFreeUndoSeg--;
+ ndbrequire(cnoFreeUndoSeg >= 0);
+
+ regUndoPagePtr.i = cfirstfreeUndoSeg;
+ ptrCheckGuard(regUndoPagePtr, cnoOfUndoPage, undoPage);
+ cfirstfreeUndoSeg = regUndoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS];
+ regUndoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS] = RNIL;
+ for (Uint32 i = 0; i < ZUB_SEGMENT_SIZE; i++) {
+ dbsiPtr.p->pdxDataPage[i] = regUndoPagePtr.i + i;
+ }//for
+}//Dbtup::allocDataBufferSegment()
+
+/* ---------------------------------------------------------------- */
+/* --- PERFORM A COPY OF THE ACTIVE OPERATIONS FOR THIS FRAGMENT -- */
+/* ---------------------------------------------------------------- */
+void Dbtup::lcpSaveCopyListLab(Signal* signal, CheckpointInfoPtr ciPtr)
+{
+ FragrecordPtr regFragPtr;
+ LocalLogInfoPtr lliPtr;
+ OperationrecPtr regOpPtr;
+
+ regFragPtr.i = ciPtr.p->lcpFragmentP;
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+ lliPtr.i = ciPtr.p->lcpLocalLogInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ regOpPtr.i = ciPtr.p->lcpTmpOperPtr;
+
+/* -------------------------------------------------------------------------------- */
+/* TRAVERSE THE ENTIRE BLOCK OF OPERATIONS. CHECK IF THERE ARE EXISTING COPYS OF */
+/* TUPLES IN THE CHECKPOINTED FRAGMENT. SAVE THOSE IN A LIST IN THE FOLLOWING FORM: */
+/* */
+/* SOURCE PAGE */
+/* SOURCE INDEX */
+/* COPY PAGE */
+/* COPY INDEX */
+/* -------------------------------------------------------------------------------- */
+ Uint32 loopCount = 0;
+ while ((regOpPtr.i != RNIL) && (loopCount < 50)) {
+ ljam();
+ ptrCheckGuard(regOpPtr, cnoOfOprec, operationrec);
+ if (regOpPtr.p->realPageId != RNIL) {
+/* ---------------------------------------------------------------- */
+// We ensure that we have actually allocated the tuple header and
+// also found it. Otherwise we will fill the undo log with garbage.
+/* ---------------------------------------------------------------- */
+ if (regOpPtr.p->optype == ZUPDATE) {
+ ljam();
+ if (regOpPtr.p->realPageIdC != RNIL) {
+/* ---------------------------------------------------------------- */
+// We ensure that we have actually allocated the tuple header copy.
+// Otherwise we will fill the undo log with garbage.
+/* ---------------------------------------------------------------- */
+ cprAddLogHeader(signal,
+ lliPtr.p,
+ ZLCPR_ABORT_UPDATE,
+ ciPtr.p->lcpTabPtr,
+ ciPtr.p->lcpFragmentId);
+ cprAddAbortUpdate(signal, lliPtr.p, regOpPtr.p);
+ }//if
+ } else if (regOpPtr.p->optype == ZINSERT) {
+ ljam();
+ cprAddUndoLogRecord(signal,
+ ZLCPR_ABORT_INSERT,
+ regOpPtr.p->fragPageId,
+ regOpPtr.p->pageIndex,
+ regOpPtr.p->tableRef,
+ regOpPtr.p->fragId,
+ regFragPtr.p->checkpointVersion);
+ } else {
+ ndbrequire(regOpPtr.p->optype == ZDELETE);
+ ljam();
+ cprAddUndoLogRecord(signal,
+ ZINDICATE_NO_OP_ACTIVE,
+ regOpPtr.p->fragPageId,
+ regOpPtr.p->pageIndex,
+ regOpPtr.p->tableRef,
+ regOpPtr.p->fragId,
+ regFragPtr.p->checkpointVersion);
+ }//if
+ }//if
+ loopCount++;;
+ regOpPtr.i = regOpPtr.p->nextOprecInList;
+ }//while
+ if (regOpPtr.i == RNIL) {
+ ljam();
+
+ signal->theData[0] = ciPtr.p->lcpUserptr;
+ sendSignal(ciPtr.p->lcpBlockref, GSN_TUP_LCPSTARTED, signal, 1, JBA);
+
+ signal->theData[0] = ZCONT_SAVE_DP;
+ signal->theData[1] = ciPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ } else {
+ ljam();
+ ciPtr.p->lcpTmpOperPtr = regOpPtr.i;
+ signal->theData[0] = ZCONT_START_SAVE_CL;
+ signal->theData[1] = ciPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+}//Dbtup::lcpSaveCopyListLab()
+
+/* ---------------------------------------------------------------- */
+/* ------- PERFORM A COPY OF ONE DATAPAGE DURING CHECKPOINT ------- */
+/* ---------------------------------------------------------------- */
+/* THE RANGE OF DATA PAGES IS INCLUDED IN THE CHECKPOINT_INFO_PTR */
+/* LAST_PAGE_TO_BUFFER ELEMENT IS INCREASED UNTIL ALL PAGES ARE */
+/* COPIED TO THE DISK BUFFER. WHEN A DISK BUFFER SEGMENT IS FULL */
+/* IT WILL BE WRITTEN TO DISK (TYPICALLY EACH 8:TH PAGE) */
+/* ---------------------------------------------------------------- */
+void Dbtup::lcpSaveDataPageLab(Signal* signal, Uint32 ciIndex)
+{
+ CheckpointInfoPtr ciPtr;
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ FragrecordPtr regFragPtr;
+ LocalLogInfoPtr lliPtr;
+ UndoPagePtr undoCopyPagePtr;
+ PagePtr pagePtr;
+
+ ciPtr.i = ciIndex;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+ if (ERROR_INSERTED(4000)){
+ if (ciPtr.p->lcpTabPtr == c_errorInsert4000TableId) {
+ // Delay writing of data pages during LCP
+ ndbout << "Delay writing of data pages during LCP" << endl;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 1000, 2);
+ return;
+ }//if
+ }//if
+ if (clblPageCounter == 0) {
+ ljam();
+ signal->theData[0] = ZCONT_SAVE_DP;
+ signal->theData[1] = ciPtr.i;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 100, 2);
+ return;
+ } else {
+ ljam();
+ clblPageCounter--;
+ }//if
+
+ regFragPtr.i = ciPtr.p->lcpFragmentP;
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+ dbsiPtr.i = ciPtr.p->lcpDataBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+
+ pagePtr.i = getRealpid(regFragPtr.p, regFragPtr.p->minPageNotWrittenInCheckpoint);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ ndbrequire(dbsiPtr.p->pdxNumDataPages < 16);
+ undoCopyPagePtr.i = dbsiPtr.p->pdxDataPage[dbsiPtr.p->pdxNumDataPages];
+ ptrCheckGuard(undoCopyPagePtr, cnoOfUndoPage, undoPage);
+ MEMCOPY_NO_WORDS(&undoCopyPagePtr.p->undoPageWord[0],
+ &pagePtr.p->pageWord[0],
+ ZWORDS_ON_PAGE);
+ regFragPtr.p->minPageNotWrittenInCheckpoint++;
+ dbsiPtr.p->pdxNumDataPages++;
+ if (regFragPtr.p->minPageNotWrittenInCheckpoint == regFragPtr.p->maxPageWrittenInCheckpoint) {
+ /* ---------------------------------------------------------- */
+ /* ALL PAGES ARE COPIED, TIME TO FINISH THE CHECKPOINT */
+ /* SAVE THE END POSITIONS OF THE LOG RECORDS SINCE ALL DATA */
+ /* PAGES ARE NOW SAFE ON DISK AND NO MORE LOGGING WILL APPEAR */
+ /* ---------------------------------------------------------- */
+ ljam();
+ lliPtr.i = ciPtr.p->lcpLocalLogInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ regFragPtr.p->checkpointVersion = RNIL; /* UNDO LOGGING IS SHUT OFF */
+ lcpWriteListDataPageSegment(signal, dbsiPtr, ciPtr, false);
+ dbsiPtr.p->pdxOperation = CHECKPOINT_DATA_WRITE_LAST;
+ } else if (dbsiPtr.p->pdxNumDataPages == ZDB_SEGMENT_SIZE) {
+ ljam();
+ lcpWriteListDataPageSegment(signal, dbsiPtr, ciPtr, false);
+ dbsiPtr.p->pdxOperation = CHECKPOINT_DATA_WRITE;
+ } else {
+ ljam();
+ signal->theData[0] = ZCONT_SAVE_DP;
+ signal->theData[1] = ciPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+}//Dbtup::lcpSaveDataPageLab()
+
+void Dbtup::lcpWriteListDataPageSegment(Signal* signal,
+ DiskBufferSegmentInfoPtr dbsiPtr,
+ CheckpointInfoPtr ciPtr,
+ bool flushFlag)
+{
+ Uint32 flags = 1;
+ cnoOfDataPagesToDiskWithoutSynch += dbsiPtr.p->pdxNumDataPages;
+ if ((cnoOfDataPagesToDiskWithoutSynch > MAX_PAGES_WITHOUT_SYNCH) ||
+ (flushFlag)) {
+ ljam();
+/* ---------------------------------------------------------------- */
+// To avoid synching too big chunks at a time we synch after writing
+// a certain number of data pages. (e.g. 2 MBytes).
+/* ---------------------------------------------------------------- */
+ cnoOfDataPagesToDiskWithoutSynch = 0;
+ flags |= 0x10; //Set synch flag unconditionally
+ }//if
+ signal->theData[0] = ciPtr.p->lcpDataFileHandle;
+ signal->theData[1] = cownref;
+ signal->theData[2] = dbsiPtr.i;
+ signal->theData[3] = flags;
+ signal->theData[4] = ZBASE_ADDR_UNDO_WORD;
+ signal->theData[5] = dbsiPtr.p->pdxNumDataPages;
+ signal->theData[6] = dbsiPtr.p->pdxDataPage[0];
+ signal->theData[7] = dbsiPtr.p->pdxFilePage;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+
+ dbsiPtr.p->pdxFilePage += dbsiPtr.p->pdxNumDataPages;
+ dbsiPtr.p->pdxNumDataPages = 0;
+}//Dbtup::lcpWriteListDataPageSegment()
+
+void Dbtup::lcpFlushLogLab(Signal* signal, CheckpointInfoPtr ciPtr)
+{
+ DiskBufferSegmentInfoPtr oldDbsiPtr;
+ LocalLogInfoPtr lliPtr;
+ UndoPagePtr oldUndoPagePtr;
+ UndoPagePtr newUndoPagePtr;
+
+ lliPtr.i = ciPtr.p->lcpLocalLogInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ oldDbsiPtr.i = lliPtr.p->lliUndoBufferSegmentP;
+ ptrCheckGuard(oldDbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ oldDbsiPtr.p->pdxNumDataPages++;
+ if (clblPageCounter > 0) {
+ ljam();
+ clblPageCounter--;
+ }//if
+ oldUndoPagePtr.i = lliPtr.p->lliUndoPage;
+ ptrCheckGuard(oldUndoPagePtr, cnoOfUndoPage, undoPage);
+ lcpWriteUndoSegment(signal, lliPtr.p, true);
+ oldDbsiPtr.p->pdxOperation = CHECKPOINT_UNDO_WRITE_FLUSH;
+ oldDbsiPtr.p->pdxCheckpointInfoP = ciPtr.i;
+
+/* ---------------------------------------------------------------- */
+/* SINCE LAST PAGE SENT TO DISK WAS NOT FULL YET WE COPY IT */
+/* TO THE NEW LAST PAGE. */
+/* ---------------------------------------------------------------- */
+ newUndoPagePtr.i = lliPtr.p->lliUndoPage;
+ ptrCheckGuard(newUndoPagePtr, cnoOfUndoPage, undoPage);
+ ndbrequire(lliPtr.p->lliUndoWord < ZWORDS_ON_PAGE);
+ MEMCOPY_NO_WORDS(&newUndoPagePtr.p->undoPageWord[0],
+ &oldUndoPagePtr.p->undoPageWord[0],
+ lliPtr.p->lliUndoWord);
+}//Dbtup::lcpFlushLogLab()
+
+void Dbtup::lcpFlushRestartInfoLab(Signal* signal, Uint32 ciIndex)
+{
+ CheckpointInfoPtr ciPtr;
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ LocalLogInfoPtr lliPtr;
+ UndoPagePtr undoCopyPagePtr;
+
+ ciPtr.i = ciIndex;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+
+ lliPtr.i = ciPtr.p->lcpLocalLogInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ dbsiPtr.i = ciPtr.p->lcpDataBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ undoCopyPagePtr.i = dbsiPtr.p->pdxDataPage[0]; /* UNDO INFO STORED AT PAGE 0 */
+ ptrCheckGuard(undoCopyPagePtr, cnoOfUndoPage, undoPage);
+ ndbrequire(ciPtr.p->lcpNoOfPages > 0);
+ undoCopyPagePtr.p->undoPageWord[ZSRI_NO_OF_FRAG_PAGES_POS] = ciPtr.p->lcpNoOfPages;
+ undoCopyPagePtr.p->undoPageWord[ZSRI_NO_COPY_PAGES_ALLOC] = ciPtr.p->lcpNoCopyPagesAlloc;
+ undoCopyPagePtr.p->undoPageWord[ZSRI_EMPTY_PRIM_PAGE] = ciPtr.p->lcpEmptyPrimPage;
+ undoCopyPagePtr.p->undoPageWord[ZSRI_TH_FREE_FIRST] = ciPtr.p->lcpThFreeFirst;
+ undoCopyPagePtr.p->undoPageWord[ZSRI_TH_FREE_COPY_FIRST] = ciPtr.p->lcpThFreeCopyFirst;
+ undoCopyPagePtr.p->undoPageWord[ZSRI_UNDO_LOG_END_REC_ID] = lliPtr.p->lliPrevRecordId;
+ undoCopyPagePtr.p->undoPageWord[ZSRI_UNDO_FILE_VER] = cundoFileVersion;
+ if (lliPtr.p->lliUndoWord == ZUNDO_PAGE_HEADER_SIZE) {
+ ljam();
+ undoCopyPagePtr.p->undoPageWord[ZSRI_UNDO_LOG_END_PAGE_ID] = lliPtr.p->lliLogFilePage - 1;
+ } else {
+ ljam();
+ undoCopyPagePtr.p->undoPageWord[ZSRI_UNDO_LOG_END_PAGE_ID] = lliPtr.p->lliLogFilePage;
+ }//if
+ dbsiPtr.p->pdxNumDataPages = 1;
+ dbsiPtr.p->pdxFilePage = 0;
+ if (clblPageCounter > 0) {
+ ljam();
+ clblPageCounter--;
+ }//if
+ lcpWriteListDataPageSegment(signal, dbsiPtr, ciPtr, true);
+ dbsiPtr.p->pdxOperation = CHECKPOINT_DATA_WRITE_FLUSH;
+ return;
+}//Dbtup::lcpFlushRestartInfoLab()
+
+void Dbtup::lcpCompletedLab(Signal* signal, Uint32 ciIndex)
+{
+ CheckpointInfoPtr ciPtr;
+ PendingFileOpenInfoPtr pfoiPtr;
+/* ---------------------------------------------------------------------- */
+/* INSERT CODE TO CLOSE DATA FILE HERE. DO THIS BEFORE SEND CONF */
+/* ---------------------------------------------------------------------- */
+ ciPtr.i = ciIndex;
+ ptrCheckGuard(ciPtr, cnoOfLcpRec, checkpointInfo);
+
+ seizePendingFileOpenInfoRecord(pfoiPtr);
+ pfoiPtr.p->pfoOpenType = LCP_DATA_FILE_CLOSE;
+ pfoiPtr.p->pfoCheckpointInfoP = ciPtr.i;
+
+ signal->theData[0] = ciPtr.p->lcpDataFileHandle;
+ signal->theData[1] = cownref;
+ signal->theData[2] = pfoiPtr.i;
+ signal->theData[3] = 0;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+ return;
+}//Dbtup::lcpCompletedLab()
+
+void Dbtup::lcpClosedDataFileLab(Signal* signal, CheckpointInfoPtr ciPtr)
+{
+ signal->theData[0] = ciPtr.p->lcpUserptr;
+ sendSignal(ciPtr.p->lcpBlockref, GSN_TUP_LCPCONF, signal, 1, JBB);
+ releaseCheckpointInfoRecord(ciPtr);
+ return;
+}//Dbtup::lcpClosedDataFileLab()
+
+/* ---------------------------------------------------------------------- */
+/* LCP END IS THE LAST STEP IN THE LCP PROCESS IT WILL CLOSE THE LOGFILES */
+/* AND RELEASE THE ALLOCATED CHECKPOINT_INFO_RECORDS */
+/* ---------------------------------------------------------------------- */
+void Dbtup::execEND_LCPREQ(Signal* signal)
+{
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ LocalLogInfoPtr lliPtr;
+ PendingFileOpenInfoPtr pfoiPtr;
+
+ ljamEntry();
+ clqhUserpointer = signal->theData[0];
+ clqhBlockref = signal->theData[1];
+ for (lliPtr.i = 0; lliPtr.i < 16; lliPtr.i++) {
+ ljam();
+ ptrAss(lliPtr, localLogInfo);
+ if (lliPtr.p->lliActiveLcp > 0) {
+ ljam();
+ dbsiPtr.i = lliPtr.p->lliUndoBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ freeDiskBufferSegmentRecord(signal, dbsiPtr);
+
+ seizePendingFileOpenInfoRecord(pfoiPtr); /* SEIZE A NEW FILE OPEN INFO */
+ pfoiPtr.p->pfoOpenType = LCP_UNDO_FILE_CLOSE;
+ pfoiPtr.p->pfoCheckpointInfoP = lliPtr.i;
+
+ signal->theData[0] = lliPtr.p->lliUndoFileHandle;
+ signal->theData[1] = cownref;
+ signal->theData[2] = pfoiPtr.i;
+ signal->theData[3] = 0;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+ lliPtr.p->lliActiveLcp = 0;
+ }//if
+ }//for
+ return;
+}//Dbtup::execEND_LCPREQ()
+
+void Dbtup::lcpEndconfLab(Signal* signal)
+{
+ LocalLogInfoPtr lliPtr;
+ for (lliPtr.i = 0; lliPtr.i < 16; lliPtr.i++) {
+ ljam();
+ ptrAss(lliPtr, localLogInfo);
+ if (lliPtr.p->lliUndoFileHandle != RNIL) {
+ ljam();
+/* ---------------------------------------------------------------------- */
+/* WAIT UNTIL ALL LOG FILES HAVE BEEN CLOSED. */
+/* ---------------------------------------------------------------------- */
+ return;
+ }//if
+ }//for
+ signal->theData[0] = clqhUserpointer;
+ sendSignal(clqhBlockref, GSN_END_LCPCONF, signal, 1, JBB);
+ return;
+}//Dbtup::lcpEndconfLab()
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupMeta.cpp b/ndb/src/kernel/blocks/dbtup/DbtupMeta.cpp
new file mode 100644
index 00000000000..887f82308d6
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupMeta.cpp
@@ -0,0 +1,597 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <signaldata/FsConf.hpp>
+#include <signaldata/FsRemoveReq.hpp>
+#include <signaldata/DropTab.hpp>
+#include <signaldata/AlterTab.hpp>
+#include <AttributeDescriptor.hpp>
+#include "AttributeOffset.hpp"
+
+#define ljam() { jamLine(20000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(20000 + __LINE__); }
+
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+/* --------------- ADD/DROP FRAGMENT TABLE MODULE ----------------- */
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::execTUPFRAGREQ(Signal* signal)
+{
+ FragoperrecPtr fragOperPtr;
+ FragrecordPtr regFragPtr;
+ TablerecPtr regTabPtr;
+
+ ljamEntry();
+ Uint32 userptr = signal->theData[0];
+ Uint32 userblockref = signal->theData[1];
+ Uint32 reqinfo = signal->theData[2];
+ regTabPtr.i = signal->theData[3];
+ Uint32 noOfAttributes = signal->theData[4];
+ Uint32 fragId = signal->theData[5];
+ Uint32 noOfNullAttr = signal->theData[7];
+ Uint32 schemaVersion = signal->theData[8];
+ Uint32 noOfKeyAttr = signal->theData[9];
+
+ Uint32 noOfNewAttr = signal->theData[10];
+ Uint32 checksumIndicator = signal->theData[11];
+ Uint32 noOfAttributeGroups = signal->theData[12];
+ Uint32 globalCheckpointIdIndicator = signal->theData[13];
+
+ ptrCheckGuard(regTabPtr, cnoOfTablerec, tablerec);
+ if (cfirstfreeFragopr == RNIL) {
+ ljam();
+ signal->theData[0] = userptr;
+ signal->theData[1] = ZNOFREE_FRAGOP_ERROR;
+ sendSignal(userblockref, GSN_TUPFRAGREF, signal, 2, JBB);
+ return;
+ }//if
+ seizeFragoperrec(fragOperPtr);
+
+ fragOperPtr.p->nextFragoprec = RNIL;
+ fragOperPtr.p->lqhBlockrefFrag = userblockref;
+ fragOperPtr.p->lqhPtrFrag = userptr;
+ fragOperPtr.p->fragidFrag = fragId;
+ fragOperPtr.p->tableidFrag = regTabPtr.i;
+ fragOperPtr.p->attributeCount = noOfAttributes;
+ fragOperPtr.p->freeNullBit = noOfNullAttr;
+ fragOperPtr.p->noOfNewAttrCount = noOfNewAttr;
+
+ ndbrequire(reqinfo == ZADDFRAG);
+
+ getFragmentrec(regFragPtr, fragId, regTabPtr.p);
+ if (regFragPtr.i != RNIL) {
+ ljam();
+ terrorCode = ZEXIST_FRAG_ERROR; /* THE FRAGMENT ALREADY EXIST */
+ fragrefuse1Lab(signal, fragOperPtr);
+ return;
+ }//if
+ if (cfirstfreefrag != RNIL) {
+ ljam();
+ seizeFragrecord(regFragPtr);
+ } else {
+ ljam();
+ terrorCode = ZFULL_FRAGRECORD_ERROR;
+ fragrefuse1Lab(signal, fragOperPtr);
+ return;
+ }//if
+ initFragRange(regFragPtr.p);
+ if (!addfragtotab(regTabPtr.p, fragId, regFragPtr.i)) {
+ ljam();
+ terrorCode = ZNO_FREE_TAB_ENTRY_ERROR;
+ fragrefuse2Lab(signal, fragOperPtr, regFragPtr);
+ return;
+ }//if
+ if (cfirstfreerange == RNIL) {
+ ljam();
+ terrorCode = ZNO_FREE_PAGE_RANGE_ERROR;
+ fragrefuse3Lab(signal, fragOperPtr, regFragPtr, regTabPtr.p, fragId);
+ return;
+ }//if
+
+ regFragPtr.p->emptyPrimPage = RNIL;
+ regFragPtr.p->thFreeFirst = RNIL;
+ regFragPtr.p->thFreeCopyFirst = RNIL;
+ regFragPtr.p->noCopyPagesAlloc = 0;
+ regFragPtr.p->fragTableId = regTabPtr.i;
+ regFragPtr.p->fragmentId = fragId;
+ regFragPtr.p->checkpointVersion = RNIL;
+
+ Uint32 noAllocatedPages = 2;
+ noAllocatedPages = allocFragPages(regFragPtr.p, noAllocatedPages);
+
+ if (noAllocatedPages == 0) {
+ ljam();
+ terrorCode = ZNO_PAGES_ALLOCATED_ERROR;
+ fragrefuse3Lab(signal, fragOperPtr, regFragPtr, regTabPtr.p, fragId);
+ return;
+ }//if
+
+ if (regTabPtr.p->tableStatus == NOT_DEFINED) {
+ ljam();
+//-------------------------------------------------------------------------------------
+// We are setting up references to the header of the tuple.
+// Active operation This word contains a reference to the operation active on the tuple
+// at the moment. RNIL means no one active at all. Not optional.
+// Tuple version Uses only low 16 bits. Not optional.
+// Checksum The third header word is optional and contains a checksum of the
+// tuple header.
+// Null-bits A number of words to contain null bits for all non-dynamic attributes.
+// Each word contains upto 32 null bits. Each time a new word is needed
+// we allocate the complete word. Zero nullable attributes means that
+// there is no word at all
+// Global Checkpoint id
+// This word is optional. When used it is stored as a 32-bit unsigned
+// attribute with attribute identity 0. Thus the kernel assumes that
+// this is the first word after the header.
+//-------------------------------------------------------------------------------------
+ fragOperPtr.p->definingFragment = true;
+ regTabPtr.p->tableStatus = DEFINING;
+ regTabPtr.p->checksumIndicator = (checksumIndicator != 0 ? true : false);
+ regTabPtr.p->GCPIndicator = (globalCheckpointIdIndicator != 0 ? true : false);
+
+ regTabPtr.p->tupChecksumIndex = 2;
+ regTabPtr.p->tupNullIndex = 2 + (checksumIndicator != 0 ? 1 : 0);
+ regTabPtr.p->tupNullWords = (noOfNullAttr + 31) >> 5;
+ regTabPtr.p->tupGCPIndex = regTabPtr.p->tupNullIndex + regTabPtr.p->tupNullWords;
+ regTabPtr.p->tupheadsize = regTabPtr.p->tupGCPIndex;
+
+ regTabPtr.p->noOfKeyAttr = noOfKeyAttr;
+ regTabPtr.p->noOfAttr = noOfAttributes;
+ regTabPtr.p->noOfNewAttr = noOfNewAttr;
+ regTabPtr.p->noOfNullAttr = noOfNullAttr;
+ regTabPtr.p->noOfAttributeGroups = noOfAttributeGroups;
+
+ regTabPtr.p->notNullAttributeMask.clear();
+
+ Uint32 tableDescriptorRef = allocTabDescr(noOfAttributes, noOfKeyAttr, noOfAttributeGroups);
+ if (tableDescriptorRef == RNIL) {
+ ljam();
+ fragrefuse4Lab(signal, fragOperPtr, regFragPtr, regTabPtr.p, fragId);
+ return;
+ }//if
+ setUpDescriptorReferences(tableDescriptorRef, regTabPtr.p);
+ } else {
+ ljam();
+ fragOperPtr.p->definingFragment = false;
+ }//if
+ signal->theData[0] = fragOperPtr.p->lqhPtrFrag;
+ signal->theData[1] = fragOperPtr.i;
+ signal->theData[2] = regFragPtr.i;
+ signal->theData[3] = fragId;
+ sendSignal(fragOperPtr.p->lqhBlockrefFrag, GSN_TUPFRAGCONF, signal, 4, JBB);
+ return;
+}//Dbtup::execTUPFRAGREQ()
+
+/* -------------------------------------------------------------------- */
+/* ------------------------- ADDFRAGTOTAB ----------------------------- */
+/* PUTS A FRAGMENT POINTER AND FID IN THE TABLE ARRAY OF THE TID RECORD */
+/* -------------------------------------------------------------------- */
+bool Dbtup::addfragtotab(Tablerec* const regTabPtr, Uint32 fragId, Uint32 fragIndex)
+{
+ for (Uint32 i = 0; i < (2 * NO_OF_FRAG_PER_NODE); i++) {
+ ljam();
+ if (regTabPtr->fragid[i] == RNIL) {
+ ljam();
+ regTabPtr->fragid[i] = fragId;
+ regTabPtr->fragrec[i] = fragIndex;
+ return true;
+ }//if
+ }//for
+ return false;
+}//Dbtup::addfragtotab()
+
+void Dbtup::getFragmentrec(FragrecordPtr& regFragPtr, Uint32 fragId, Tablerec* const regTabPtr)
+{
+ for (Uint32 i = 0; i < (2 * NO_OF_FRAG_PER_NODE); i++) {
+ ljam();
+ if (regTabPtr->fragid[i] == fragId) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* A FRAGMENT RECORD HAVE BEEN FOUND FOR THIS OPERATION. */
+/* ---------------------------------------------------------------- */
+ regFragPtr.i = regTabPtr->fragrec[i];
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+ return;
+ }//if
+ }//for
+ regFragPtr.i = RNIL;
+ ptrNull(regFragPtr);
+}//Dbtup::getFragmentrec()
+
+void Dbtup::seizeFragrecord(FragrecordPtr& regFragPtr)
+{
+ regFragPtr.i = cfirstfreefrag;
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+ cfirstfreefrag = regFragPtr.p->nextfreefrag;
+ regFragPtr.p->nextfreefrag = RNIL;
+}//Dbtup::seizeFragrecord()
+
+/* ---------------------------------------------------------------- */
+/* SEIZE A FRAGMENT OPERATION RECORD */
+/* ---------------------------------------------------------------- */
+void Dbtup::seizeFragoperrec(FragoperrecPtr& fragOperPtr)
+{
+ fragOperPtr.i = cfirstfreeFragopr;
+ ptrCheckGuard(fragOperPtr, cnoOfFragoprec, fragoperrec);
+ cfirstfreeFragopr = fragOperPtr.p->nextFragoprec;
+ fragOperPtr.p->nextFragoprec = RNIL;
+}//Dbtup::seizeFragoperrec()
+
+/* **************************************************************** */
+/* ************** TUP_ADD_ATTRREQ ****************** */
+/* **************************************************************** */
+void Dbtup::execTUP_ADD_ATTRREQ(Signal* signal)
+{
+ FragrecordPtr regFragPtr;
+ FragoperrecPtr fragOperPtr;
+ TablerecPtr regTabPtr;
+
+ ljamEntry();
+ fragOperPtr.i = signal->theData[0];
+ ptrCheckGuard(fragOperPtr, cnoOfFragoprec, fragoperrec);
+ Uint32 attrId = signal->theData[2];
+ Uint32 attrDescriptor = signal->theData[3];
+
+ regTabPtr.i = fragOperPtr.p->tableidFrag;
+ ptrCheckGuard(regTabPtr, cnoOfTablerec, tablerec);
+
+ Uint32 fragId = fragOperPtr.p->fragidFrag;
+
+ getFragmentrec(regFragPtr, fragId, regTabPtr.p);
+ ndbrequire(regFragPtr.i != RNIL);
+
+ ndbrequire(fragOperPtr.p->attributeCount > 0);
+ fragOperPtr.p->attributeCount--;
+
+ if ((regTabPtr.p->tableStatus == DEFINING) &&
+ (fragOperPtr.p->definingFragment)) {
+ ljam();
+ Uint32 firstTabDesIndex = regTabPtr.p->tabDescriptor + (attrId * ZAD_SIZE);
+ setTabDescrWord(firstTabDesIndex, attrDescriptor);
+ Uint32 attrLen = AttributeDescriptor::getSize(attrDescriptor);
+ Uint32 nullBitPos = 0; /* Default pos for NOT NULL attributes */
+ if (AttributeDescriptor::getNullable(attrDescriptor)) {
+ if (!AttributeDescriptor::getDynamic(attrDescriptor)) {
+ ljam(); /* NULL ATTR */
+ fragOperPtr.p->freeNullBit--; /* STORE NULL BIT POSTITION */
+ nullBitPos = fragOperPtr.p->freeNullBit;
+ ndbrequire(fragOperPtr.p->freeNullBit < ZNIL); /* Check not below zero */
+ }//if
+ } else {
+ ljam();
+ regTabPtr.p->notNullAttributeMask.set(attrId);
+ }//if
+
+ Uint32 attrDes2 = 0;
+ if (!AttributeDescriptor::getDynamic(attrDescriptor)) {
+ ljam();
+ Uint32 attributePos = regTabPtr.p->tupheadsize;
+ switch (AttributeDescriptor::getArrayType(attrDescriptor)) {
+ case 1:
+ case 2:
+ {
+ ljam();
+ Uint32 bitsUsed = AttributeDescriptor::getArraySize(attrDescriptor) * (1 << attrLen);
+ regTabPtr.p->tupheadsize += ((bitsUsed + 31) >> 5);
+ break;
+ }
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ AttributeOffset::setOffset(attrDes2, attributePos);
+ AttributeOffset::setNullFlagPos(attrDes2, nullBitPos);
+ } else {
+ ndbrequire(false);
+ }//if
+ setTabDescrWord(firstTabDesIndex + 1, attrDes2);
+
+ if (regTabPtr.p->tupheadsize > MAX_TUPLE_SIZE_IN_WORDS) {
+ ljam();
+ terrorCode = ZTOO_LARGE_TUPLE_ERROR;
+ addattrrefuseLab(signal, regFragPtr, fragOperPtr, regTabPtr.p, fragId);
+ return;
+ }//if
+ if ((fragOperPtr.p->attributeCount == 0) &&
+ (fragOperPtr.p->freeNullBit != 0)) {
+ ljam();
+ terrorCode = ZINCONSISTENT_NULL_ATTRIBUTE_COUNT;
+ addattrrefuseLab(signal, regFragPtr, fragOperPtr, regTabPtr.p, fragId);
+ return;
+ }//if
+ }//if
+/* **************************************************************** */
+/* ************** TUP_ADD_ATTCONF ****************** */
+/* **************************************************************** */
+ signal->theData[0] = fragOperPtr.p->lqhPtrFrag;
+ sendSignal(fragOperPtr.p->lqhBlockrefFrag, GSN_TUP_ADD_ATTCONF, signal, 1, JBB);
+ if (fragOperPtr.p->attributeCount > 0) {
+ ljam();
+ return; /* EXIT AND WAIT FOR MORE */
+ }//if
+ regFragPtr.p->fragStatus = ACTIVE;
+ if (regTabPtr.p->tableStatus == DEFINING) {
+ ljam();
+ setUpQueryRoutines(regTabPtr.p);
+ setUpKeyArray(regTabPtr.p);
+ regTabPtr.p->tableStatus = DEFINED;
+ }//if
+ releaseFragoperrec(fragOperPtr);
+ return;
+}//Dbtup::execTUP_ADD_ATTRREQ()
+
+void Dbtup::setUpDescriptorReferences(Uint32 descriptorReference,
+ Tablerec* const regTabPtr)
+{
+ Uint32 noOfAttributes = regTabPtr->noOfAttr;
+ descriptorReference += ZTD_SIZE;
+ ReadFunction * tmp = (ReadFunction*)&tableDescriptor[descriptorReference].tabDescr;
+ regTabPtr->readFunctionArray = tmp;
+ regTabPtr->updateFunctionArray = (UpdateFunction*)(tmp + noOfAttributes);
+
+ TableDescriptor * start = &tableDescriptor[descriptorReference];
+ TableDescriptor * end = (TableDescriptor*)(tmp + 2 * noOfAttributes);
+ regTabPtr->readKeyArray = descriptorReference + (end - start);
+ regTabPtr->attributeGroupDescriptor = regTabPtr->readKeyArray + regTabPtr->noOfKeyAttr;
+ regTabPtr->tabDescriptor = regTabPtr->attributeGroupDescriptor + regTabPtr->noOfAttributeGroups;
+}//Dbtup::setUpDescriptorReferences()
+
+Uint32
+Dbtup::sizeOfReadFunction()
+{
+ ReadFunction* tmp = (ReadFunction*)&tableDescriptor[0];
+ TableDescriptor* start = &tableDescriptor[0];
+ TableDescriptor * end = (TableDescriptor*)(tmp + 1);
+ return (Uint32)(end - start);
+}//Dbtup::sizeOfReadFunction()
+
+void Dbtup::setUpKeyArray(Tablerec* const regTabPtr)
+{
+ ndbrequire((regTabPtr->readKeyArray + regTabPtr->noOfKeyAttr) < cnoOfTabDescrRec);
+ Uint32* keyArray = &tableDescriptor[regTabPtr->readKeyArray].tabDescr;
+ Uint32 countKeyAttr = 0;
+ for (Uint32 i = 0; i < regTabPtr->noOfAttr; i++) {
+ ljam();
+ Uint32 refAttr = regTabPtr->tabDescriptor + (i * ZAD_SIZE);
+ Uint32 attrDescriptor = getTabDescrWord(refAttr);
+ if (AttributeDescriptor::getPrimaryKey(attrDescriptor)) {
+ ljam();
+ AttributeHeader::init(&keyArray[countKeyAttr], i, 0);
+ countKeyAttr++;
+ }//if
+ }//for
+ ndbrequire(countKeyAttr == regTabPtr->noOfKeyAttr);
+}//Dbtup::setUpKeyArray()
+
+void Dbtup::addattrrefuseLab(Signal* signal,
+ FragrecordPtr regFragPtr,
+ FragoperrecPtr fragOperPtr,
+ Tablerec* const regTabPtr,
+ Uint32 fragId)
+{
+ releaseFragPages(regFragPtr.p);
+ deleteFragTab(regTabPtr, fragId);
+ releaseFragrec(regFragPtr);
+ releaseTabDescr(regTabPtr);
+ initTab(regTabPtr);
+
+ signal->theData[0] = fragOperPtr.p->lqhPtrFrag;
+ signal->theData[1] = terrorCode;
+ sendSignal(fragOperPtr.p->lqhBlockrefFrag, GSN_TUP_ADD_ATTRREF, signal, 2, JBB);
+ releaseFragoperrec(fragOperPtr);
+ return;
+}//Dbtup::addattrrefuseLab()
+
+void Dbtup::fragrefuse4Lab(Signal* signal,
+ FragoperrecPtr fragOperPtr,
+ FragrecordPtr regFragPtr,
+ Tablerec* const regTabPtr,
+ Uint32 fragId)
+{
+ releaseFragPages(regFragPtr.p);
+ fragrefuse3Lab(signal, fragOperPtr, regFragPtr, regTabPtr, fragId);
+ initTab(regTabPtr);
+ return;
+}//Dbtup::fragrefuse4Lab()
+
+void Dbtup::fragrefuse3Lab(Signal* signal,
+ FragoperrecPtr fragOperPtr,
+ FragrecordPtr regFragPtr,
+ Tablerec* const regTabPtr,
+ Uint32 fragId)
+{
+ fragrefuse2Lab(signal, fragOperPtr, regFragPtr);
+ deleteFragTab(regTabPtr, fragId);
+ return;
+}//Dbtup::fragrefuse3Lab()
+
+void Dbtup::fragrefuse2Lab(Signal* signal, FragoperrecPtr fragOperPtr, FragrecordPtr regFragPtr)
+{
+ fragrefuse1Lab(signal, fragOperPtr);
+ releaseFragrec(regFragPtr);
+ return;
+}//Dbtup::fragrefuse2Lab()
+
+void Dbtup::fragrefuse1Lab(Signal* signal, FragoperrecPtr fragOperPtr)
+{
+ fragrefuseLab(signal, fragOperPtr);
+ releaseFragoperrec(fragOperPtr);
+ return;
+}//Dbtup::fragrefuse1Lab()
+
+void Dbtup::fragrefuseLab(Signal* signal, FragoperrecPtr fragOperPtr)
+{
+ signal->theData[0] = fragOperPtr.p->lqhPtrFrag;
+ signal->theData[1] = terrorCode;
+ sendSignal(fragOperPtr.p->lqhBlockrefFrag, GSN_TUPFRAGREF, signal, 2, JBB);
+ return;
+}//Dbtup::fragrefuseLab()
+
+void Dbtup::releaseFragoperrec(FragoperrecPtr fragOperPtr)
+{
+ fragOperPtr.p->nextFragoprec = cfirstfreeFragopr;
+ cfirstfreeFragopr = fragOperPtr.i;
+}//Dbtup::releaseFragoperrec()
+
+
+void Dbtup::deleteFragTab(Tablerec* const regTabPtr, Uint32 fragId)
+{
+ for (Uint32 i = 0; i < (2 * NO_OF_FRAG_PER_NODE); i++) {
+ ljam();
+ if (regTabPtr->fragid[i] == fragId) {
+ ljam();
+ regTabPtr->fragid[i] = RNIL;
+ regTabPtr->fragrec[i] = RNIL;
+ return;
+ }//if
+ }//for
+ ndbrequire(false);
+}//Dbtup::deleteFragTab()
+
+void
+Dbtup::execDROP_TAB_REQ(Signal* signal)
+{
+ ljamEntry();
+ DropTabReq* req = (DropTabReq*)signal->getDataPtr();
+
+ TablerecPtr tabPtr;
+ tabPtr.i = req->tableId;
+ ptrCheckGuard(tabPtr, cnoOfTablerec, tablerec);
+
+ tabPtr.p->m_dropTable.tabUserRef = req->senderRef;
+ tabPtr.p->m_dropTable.tabUserPtr = req->senderData;
+
+ signal->theData[0] = ZREL_FRAG;
+ signal->theData[1] = tabPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+}//Dbtup::execDROP_TAB_REQ()
+
+void Dbtup::releaseTabDescr(Tablerec* const regTabPtr)
+{
+ Uint32 descriptor = regTabPtr->readKeyArray;
+ if (descriptor != RNIL) {
+ ljam();
+ regTabPtr->tabDescriptor = RNIL;
+ regTabPtr->readKeyArray = RNIL;
+ regTabPtr->readFunctionArray = NULL;
+ regTabPtr->updateFunctionArray = NULL;
+ regTabPtr->attributeGroupDescriptor= RNIL;
+
+ Uint32 sizeFunctionArrays = 2 * (regTabPtr->noOfAttr * sizeOfReadFunction());
+ descriptor -= (sizeFunctionArrays + ZTD_SIZE);
+ Uint32 retNo = getTabDescrWord(descriptor + ZTD_DATASIZE);
+ ndbrequire(getTabDescrWord(descriptor + ZTD_HEADER) == ZTD_TYPE_NORMAL);
+ ndbrequire(retNo == getTabDescrWord((descriptor + retNo) - ZTD_TR_SIZE));
+ ndbrequire(ZTD_TYPE_NORMAL == getTabDescrWord((descriptor + retNo) - ZTD_TR_TYPE));
+ freeTabDescr(descriptor, retNo);
+ }//if
+}//Dbtup::releaseTabDescr()
+
+void Dbtup::releaseFragment(Signal* signal, Uint32 tableId)
+{
+ TablerecPtr tabPtr;
+ tabPtr.i = tableId;
+ ptrCheckGuard(tabPtr, cnoOfTablerec, tablerec);
+ Uint32 fragIndex = RNIL;
+ Uint32 fragId = RNIL;
+ Uint32 i = 0;
+ for (i = 0; i < (2 * NO_OF_FRAG_PER_NODE); i++) {
+ ljam();
+ if (tabPtr.p->fragid[i] != RNIL) {
+ ljam();
+ fragIndex = tabPtr.p->fragrec[i];
+ fragId = tabPtr.p->fragid[i];
+ break;
+ }//if
+ }//for
+ if (fragIndex != RNIL) {
+ ljam();
+
+ FragrecordPtr regFragPtr;
+ regFragPtr.i = fragIndex;
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+ releaseFragPages(regFragPtr.p);
+
+ tabPtr.p->fragid[i] = RNIL;
+ tabPtr.p->fragrec[i] = RNIL;
+ releaseFragrec(regFragPtr);
+
+ signal->theData[0] = ZREL_FRAG;
+ signal->theData[1] = tableId;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ }//if
+
+ /**
+ * Finished...
+ */
+ sendFSREMOVEREQ(signal, tabPtr);
+}//Dbtup::releaseFragment()
+
+void Dbtup::sendFSREMOVEREQ(Signal* signal, TablerecPtr tabPtr)
+{
+ FsRemoveReq * const fsReq = (FsRemoveReq *)signal->getDataPtrSend();
+ fsReq->userReference = cownref;
+ fsReq->userPointer = tabPtr.i;
+ fsReq->fileNumber[0] = tabPtr.i;
+ fsReq->fileNumber[1] = (Uint32)-1; // Remove all fragments
+ fsReq->fileNumber[2] = (Uint32)-1; // Remove all data files within fragment
+ fsReq->fileNumber[3] = 255 | // No P-value used here
+ (5 << 8) | // Data-files in D5
+ (0 << 16) | // Data-files
+ (1 << 24); // Version 1 of fileNumber
+
+ fsReq->directory = 1;
+ fsReq->ownDirectory = 1;
+ sendSignal(NDBFS_REF, GSN_FSREMOVEREQ, signal,
+ FsRemoveReq::SignalLength, JBA);
+}//Dbtup::sendFSREMOVEREQ()
+
+void Dbtup::execFSREMOVECONF(Signal* signal)
+{
+ ljamEntry();
+
+ FsConf * const fsConf = (FsConf *)signal->getDataPtrSend();
+ TablerecPtr tabPtr;
+ tabPtr.i = fsConf->userPointer;
+ ptrCheckGuard(tabPtr, cnoOfTablerec, tablerec);
+
+
+ DropTabConf * const dropConf = (DropTabConf *)signal->getDataPtrSend();
+ dropConf->senderRef = reference();
+ dropConf->senderData = tabPtr.p->m_dropTable.tabUserPtr;
+ dropConf->tableId = tabPtr.i;
+ sendSignal(tabPtr.p->m_dropTable.tabUserRef, GSN_DROP_TAB_CONF,
+ signal, DropTabConf::SignalLength, JBB);
+
+ releaseTabDescr(tabPtr.p);
+ initTab(tabPtr.p);
+}//Dbtup::execFSREMOVECONF()
+
+void Dbtup::execFSREMOVEREF(Signal* signal)
+{
+ ljamEntry();
+ ndbrequire(false);
+}//Dbtup::execFSREMOVEREF()
+
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupPagMan.cpp b/ndb/src/kernel/blocks/dbtup/DbtupPagMan.cpp
new file mode 100644
index 00000000000..410cafee161
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupPagMan.cpp
@@ -0,0 +1,360 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+
+#define ljam() { jamLine(16000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(16000 + __LINE__); }
+
+/* ---------------------------------------------------------------- */
+// 4) Page Memory Manager (buddy algorithm)
+//
+// The following data structures in Dbtup is used by the Page Memory
+// Manager.
+//
+// cfreepageList[16]
+// Pages with a header
+//
+// The cfreepageList is 16 free lists. Free list 0 contains chunks of
+// pages with 2^0 (=1) pages in each chunk. Free list 1 chunks of 2^1
+// (=2) pages in each chunk and so forth upto free list 15 which
+// contains chunks of 2^15 (=32768) pages in each chunk.
+// The cfreepageList array contains the pointer to the first chunk
+// in each of those lists. The lists are doubly linked where the
+// first page in each chunk contains the next and previous references
+// in position ZPAGE_NEXT_CLUST_POS and ZPAGE_PREV_CLUST_POS in the
+// page header.
+// In addition the leading page and the last page in each chunk is marked
+// with a state (=ZFREE_COMMON) in position ZPAGE_STATE_POS in page
+// header. This state indicates that the page is the leading or last page
+// in a chunk of free pages. Furthermore the leading and last page is
+// also marked with a reference to the leading (=ZPAGE_FIRST_CLUST_POS)
+// and the last page (=ZPAGE_LAST_CLUST_POS) in the chunk.
+//
+// The aim of these data structures is to enable a free area handling of
+// free pages based on a buddy algorithm. When allocating pages it is
+// performed in chunks of pages and the algorithm tries to make the
+// chunks as large as possible.
+// This manager is invoked when fragments lack internal page space to
+// accomodate all the data they are requested to store. It is also
+// invoked when fragments deallocate page space back to the free area.
+//
+// The following routines are part of the external interface:
+// void
+// allocConsPages(Uint32 noOfPagesToAllocate, #In
+// Uint32& noOfPagesAllocated, #Out
+// Uint32& retPageRef) #Out
+// void
+// returnCommonArea(Uint32 retPageRef, #In
+// Uint32 retNoPages) #In
+//
+// allocConsPages tries to allocate noOfPagesToAllocate pages in one chunk.
+// If this fails it delivers a chunk as large as possible. It returns the
+// i-value of the first page in the chunk delivered, if zero pages returned
+// this i-value is undefined. It also returns the size of the chunk actually
+// delivered.
+//
+// returnCommonArea is used when somebody is returning pages to the free area.
+// It is used both from internal routines and external routines.
+//
+// The following routines are private routines used to support the
+// above external interface:
+// removeCommonArea()
+// insertCommonArea()
+// findFreeLeftNeighbours()
+// findFreeRightNeighbours()
+// Uint32
+// nextHigherTwoLog(Uint32 input)
+//
+// nextHigherTwoLog is a support routine which is a mathematical function with
+// an integer as input and an integer as output. It calculates the 2-log of
+// (input + 1). If the 2-log of (input + 1) is larger than 15 then the routine
+// will return 15. It is part of the external interface since it is also used
+// by other similar memory management algorithms.
+//
+// External dependencies:
+// None.
+//
+// Side Effects:
+// Apart from the above mentioned data structures there are no more
+// side effects other than through the subroutine parameters in the
+// external interface.
+//
+/* ---------------------------------------------------------------- */
+
+/* ---------------------------------------------------------------- */
+/* CALCULATE THE 2-LOG + 1 OF TMP AND PUT RESULT INTO TBITS */
+/* ---------------------------------------------------------------- */
+Uint32 Dbtup::nextHigherTwoLog(Uint32 input)
+{
+ input = input | (input >> 8);
+ input = input | (input >> 4);
+ input = input | (input >> 2);
+ input = input | (input >> 1);
+ Uint32 output = (input & 0x5555) + ((input >> 1) & 0x5555);
+ output = (output & 0x3333) + ((output >> 2) & 0x3333);
+ output = output + (output >> 4);
+ output = (output & 0xf) + ((output >> 8) & 0xf);
+ return output;
+}//nextHigherTwoLog()
+
+void Dbtup::initializePage()
+{
+ for (Uint32 i = 0; i < 16; i++) {
+ cfreepageList[i] = RNIL;
+ }//for
+ PagePtr pagePtr;
+ for (pagePtr.i = 0; pagePtr.i < cnoOfPage; pagePtr.i++) {
+ ljam();
+ ptrAss(pagePtr, page);
+ pagePtr.p->pageWord[ZPAGE_PHYSICAL_INDEX] = pagePtr.i;
+ pagePtr.p->pageWord[ZPAGE_NEXT_POS] = pagePtr.i + 1;
+ pagePtr.p->pageWord[ZPAGE_NEXT_CLUST_POS] = RNIL;
+ pagePtr.p->pageWord[ZPAGE_LAST_CLUST_POS] = RNIL;
+ pagePtr.p->pageWord[ZPAGE_PREV_POS] = RNIL;
+ pagePtr.p->pageWord[ZPAGE_STATE_POS] = ZFREE_COMMON;
+ }//for
+ pagePtr.i = cnoOfPage - 1;
+ ptrAss(pagePtr, page);
+ pagePtr.p->pageWord[ZPAGE_NEXT_POS] = RNIL;
+
+ pagePtr.i = 0;
+ ptrAss(pagePtr, page);
+ pagePtr.p->pageWord[ZPAGE_STATE_POS] = ~ZFREE_COMMON;
+
+ returnCommonArea(1, cnoOfPage - 1);
+ cnoOfAllocatedPages = 1;
+}//Dbtup::initializePage()
+
+void Dbtup::allocConsPages(Uint32 noOfPagesToAllocate,
+ Uint32& noOfPagesAllocated,
+ Uint32& allocPageRef)
+{
+ if (noOfPagesToAllocate == 0) {
+ ljam();
+ noOfPagesAllocated = 0;
+ return;
+ }//if
+ Uint32 firstListToCheck = nextHigherTwoLog(noOfPagesToAllocate - 1);
+ for (Uint32 i = firstListToCheck; i < 16; i++) {
+ ljam();
+ if (cfreepageList[i] != RNIL) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* PROPER AMOUNT OF PAGES WERE FOUND. NOW SPLIT THE FOUND */
+/* AREA AND RETURN THE PART NOT NEEDED. */
+/* ---------------------------------------------------------------- */
+ noOfPagesAllocated = noOfPagesToAllocate;
+ allocPageRef = cfreepageList[i];
+ removeCommonArea(allocPageRef, i);
+ Uint32 retNo = (1 << i) - noOfPagesToAllocate;
+ Uint32 retPageRef = allocPageRef + noOfPagesToAllocate;
+ returnCommonArea(retPageRef, retNo);
+ return;
+ }//if
+ }//for
+/* ---------------------------------------------------------------- */
+/* PROPER AMOUNT OF PAGES WERE NOT FOUND. FIND AS MUCH AS */
+/* POSSIBLE. */
+/* ---------------------------------------------------------------- */
+ for (Uint32 j = firstListToCheck; (Uint32)~j; j--) {
+ ljam();
+ if (cfreepageList[j] != RNIL) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* SOME AREA WAS FOUND, ALLOCATE ALL OF IT. */
+/* ---------------------------------------------------------------- */
+ allocPageRef = cfreepageList[j];
+ removeCommonArea(allocPageRef, j);
+ noOfPagesAllocated = 1 << j;
+ findFreeLeftNeighbours(allocPageRef, noOfPagesAllocated,
+ noOfPagesToAllocate);
+ findFreeRightNeighbours(allocPageRef, noOfPagesAllocated,
+ noOfPagesToAllocate);
+
+ return;
+ }//if
+ }//for
+/* ---------------------------------------------------------------- */
+/* NO FREE AREA AT ALL EXISTED. RETURN ZERO PAGES */
+/* ---------------------------------------------------------------- */
+ noOfPagesAllocated = 0;
+ return;
+}//allocConsPages()
+
+void Dbtup::returnCommonArea(Uint32 retPageRef, Uint32 retNo)
+{
+ do {
+ ljam();
+ if (retNo == 0) {
+ ljam();
+ return;
+ }//if
+ Uint32 list = nextHigherTwoLog(retNo) - 1;
+ retNo -= (1 << list);
+ insertCommonArea(retPageRef, list);
+ retPageRef += (1 << list);
+ } while (1);
+}//Dbtup::returnCommonArea()
+
+void Dbtup::findFreeLeftNeighbours(Uint32& allocPageRef,
+ Uint32& noPagesAllocated,
+ Uint32 noOfPagesToAllocate)
+{
+ PagePtr pageFirstPtr, pageLastPtr;
+ Uint32 remainAllocate = noOfPagesToAllocate - noPagesAllocated;
+ while (allocPageRef > 0) {
+ ljam();
+ pageLastPtr.i = allocPageRef - 1;
+ ptrCheckGuard(pageLastPtr, cnoOfPage, page);
+ if (pageLastPtr.p->pageWord[ZPAGE_STATE_POS] != ZFREE_COMMON) {
+ ljam();
+ return;
+ } else {
+ ljam();
+ pageFirstPtr.i = pageLastPtr.p->pageWord[ZPAGE_FIRST_CLUST_POS];
+ ndbrequire(pageFirstPtr.i != RNIL);
+ Uint32 list = nextHigherTwoLog(pageLastPtr.i - pageFirstPtr.i);
+ removeCommonArea(pageFirstPtr.i, list);
+ Uint32 listSize = 1 << list;
+ if (listSize > remainAllocate) {
+ ljam();
+ Uint32 retNo = listSize - remainAllocate;
+ returnCommonArea(pageFirstPtr.i, retNo);
+ allocPageRef = pageFirstPtr.i + retNo;
+ noPagesAllocated = noOfPagesToAllocate;
+ return;
+ } else {
+ ljam();
+ allocPageRef = pageFirstPtr.i;
+ noPagesAllocated += listSize;
+ remainAllocate -= listSize;
+ }//if
+ }//if
+ }//while
+}//Dbtup::findFreeLeftNeighbours()
+
+void Dbtup::findFreeRightNeighbours(Uint32& allocPageRef,
+ Uint32& noPagesAllocated,
+ Uint32 noOfPagesToAllocate)
+{
+ PagePtr pageFirstPtr, pageLastPtr;
+ Uint32 remainAllocate = noOfPagesToAllocate - noPagesAllocated;
+ if (remainAllocate == 0) {
+ ljam();
+ return;
+ }//if
+ while ((allocPageRef + noPagesAllocated) < cnoOfPage) {
+ ljam();
+ pageFirstPtr.i = allocPageRef + noPagesAllocated;
+ ptrCheckGuard(pageFirstPtr, cnoOfPage, page);
+ if (pageFirstPtr.p->pageWord[ZPAGE_STATE_POS] != ZFREE_COMMON) {
+ ljam();
+ return;
+ } else {
+ ljam();
+ pageLastPtr.i = pageFirstPtr.p->pageWord[ZPAGE_LAST_CLUST_POS];
+ ndbrequire(pageLastPtr.i != RNIL);
+ Uint32 list = nextHigherTwoLog(pageLastPtr.i - pageFirstPtr.i);
+ removeCommonArea(pageFirstPtr.i, list);
+ Uint32 listSize = 1 << list;
+ if (listSize > remainAllocate) {
+ ljam();
+ Uint32 retPageRef = pageFirstPtr.i + remainAllocate;
+ Uint32 retNo = listSize - remainAllocate;
+ returnCommonArea(retPageRef, retNo);
+ noPagesAllocated += remainAllocate;
+ return;
+ } else {
+ ljam();
+ noPagesAllocated += listSize;
+ remainAllocate -= listSize;
+ }//if
+ }//if
+ }//while
+}//Dbtup::findFreeRightNeighbours()
+
+void Dbtup::insertCommonArea(Uint32 insPageRef, Uint32 insList)
+{
+ cnoOfAllocatedPages -= (1 << insList);
+ PagePtr pageLastPtr, pageInsPtr;
+
+ pageInsPtr.i = insPageRef;
+ ptrCheckGuard(pageInsPtr, cnoOfPage, page);
+ ndbrequire(insList < 16);
+ pageLastPtr.i = (pageInsPtr.i + (1 << insList)) - 1;
+
+ pageInsPtr.p->pageWord[ZPAGE_NEXT_CLUST_POS] = cfreepageList[insList];
+ pageInsPtr.p->pageWord[ZPAGE_PREV_CLUST_POS] = RNIL;
+ pageInsPtr.p->pageWord[ZPAGE_LAST_CLUST_POS] = pageLastPtr.i;
+ cfreepageList[insList] = pageInsPtr.i;
+
+ ptrCheckGuard(pageLastPtr, cnoOfPage, page);
+ pageLastPtr.p->pageWord[ZPAGE_FIRST_CLUST_POS] = pageInsPtr.i;
+ pageLastPtr.p->pageWord[ZPAGE_NEXT_POS] = RNIL;
+}//Dbtup::insertCommonArea()
+
+void Dbtup::removeCommonArea(Uint32 remPageRef, Uint32 list)
+{
+ cnoOfAllocatedPages += (1 << list);
+ PagePtr pagePrevPtr, pageNextPtr, pageLastPtr, pageSearchPtr, remPagePtr;
+
+ remPagePtr.i = remPageRef;
+ ptrCheckGuard(remPagePtr, cnoOfPage, page);
+ ndbrequire(list < 16);
+ if (cfreepageList[list] == remPagePtr.i) {
+ ljam();
+ cfreepageList[list] = remPagePtr.p->pageWord[ZPAGE_NEXT_CLUST_POS];
+ pageNextPtr.i = cfreepageList[list];
+ if (pageNextPtr.i != RNIL) {
+ ljam();
+ ptrCheckGuard(pageNextPtr, cnoOfPage, page);
+ pageNextPtr.p->pageWord[ZPAGE_PREV_CLUST_POS] = RNIL;
+ }//if
+ } else {
+ pageSearchPtr.i = cfreepageList[list];
+ while (true) {
+ ljam();
+ ptrCheckGuard(pageSearchPtr, cnoOfPage, page);
+ pagePrevPtr = pageSearchPtr;
+ pageSearchPtr.i = pageSearchPtr.p->pageWord[ZPAGE_NEXT_CLUST_POS];
+ if (pageSearchPtr.i == remPagePtr.i) {
+ ljam();
+ break;
+ }//if
+ }//while
+ pageNextPtr.i = remPagePtr.p->pageWord[ZPAGE_NEXT_CLUST_POS];
+ pagePrevPtr.p->pageWord[ZPAGE_NEXT_CLUST_POS] = pageNextPtr.i;
+ if (pageNextPtr.i != RNIL) {
+ ljam();
+ ptrCheckGuard(pageNextPtr, cnoOfPage, page);
+ pageNextPtr.p->pageWord[ZPAGE_PREV_CLUST_POS] = pagePrevPtr.i;
+ }//if
+ }//if
+ remPagePtr.p->pageWord[ZPAGE_NEXT_CLUST_POS] = RNIL;
+ remPagePtr.p->pageWord[ZPAGE_LAST_CLUST_POS] = RNIL;
+ remPagePtr.p->pageWord[ZPAGE_PREV_CLUST_POS] = RNIL;
+
+ pageLastPtr.i = (remPagePtr.i + (1 << list)) - 1;
+ ptrCheckGuard(pageLastPtr, cnoOfPage, page);
+ pageLastPtr.p->pageWord[ZPAGE_FIRST_CLUST_POS] = RNIL;
+}//Dbtup::removeCommonArea()
+
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupPageMap.cpp b/ndb/src/kernel/blocks/dbtup/DbtupPageMap.cpp
new file mode 100644
index 00000000000..1f674876642
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupPageMap.cpp
@@ -0,0 +1,556 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+
+#define ljam() { jamLine(14000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(14000 + __LINE__); }
+
+//
+// PageMap is a service used by Dbtup to map logical page id's to physical
+// page id's. The mapping is needs the fragment and the logical page id to
+// provide the physical id.
+//
+// This is a part of Dbtup which is the exclusive user of a certain set of
+// variables on the fragment record and it is the exclusive user of the
+// struct for page ranges.
+//
+//
+// The following methods operate on the data handled by the page map class.
+//
+// Public methods
+// insertPageRange(Uint32 startPageId, # In
+// Uint32 noPages) # In
+// Inserts a range of pages into the mapping structure.
+//
+// void releaseFragPages()
+// Releases all pages and their mappings belonging to a fragment.
+//
+// Uint32 allocFragPages(Uint32 tafpNoAllocRequested)
+// Allocate a set of pages to the fragment from the page manager
+//
+// Uint32 getEmptyPage()
+// Get an empty page from the pool of empty pages on the fragment.
+// It returns the physical page id of the empty page.
+// Returns RNIL if no empty page is available.
+//
+// Uint32 getRealpid(Uint32 logicalPageId)
+// Return the physical page id provided the logical page id
+//
+// void initializePageRange()
+// Initialise free list of page ranges and initialise the page raneg records.
+//
+// void initFragRange()
+// Initialise the fragment variables when allocating a fragment to a table.
+//
+// void initPageRangeSize(Uint32 size)
+// Initialise the number of page ranges.
+//
+// Uint32 getNoOfPages()
+// Get the number of pages on the fragment currently.
+//
+//
+// Private methods
+// Uint32 leafPageRangeFull(PageRangePtr currPageRangePtr)
+//
+// void errorHandler()
+// Method to crash NDB kernel in case of weird data set-up
+//
+// void allocMoreFragPages()
+// When no more empty pages are attached to the fragment and we need more
+// we allocate more pages from the page manager using this method.
+//
+// Private data
+// On the fragment record
+// currentPageRange # The current page range where to insert the next range
+// rootPageRange # The root of the page ranges owned
+// nextStartRange # The next page id to assign when expanding the
+// # page map
+// noOfPages # The number of pages in the fragment
+// emptyPrimPage # The first page of the empty pages in the fragment
+//
+// The full page range struct
+
+Uint32 Dbtup::getEmptyPage(Fragrecord* const regFragPtr)
+{
+ Uint32 logicalPageId = regFragPtr->emptyPrimPage;
+ if (logicalPageId == RNIL) {
+ ljam();
+ allocMoreFragPages(regFragPtr);
+ logicalPageId = regFragPtr->emptyPrimPage;
+ if (logicalPageId == RNIL) {
+ ljam();
+ return RNIL;
+ }//if
+ }//if
+ Uint32 physicalPageId = getRealpid(regFragPtr, logicalPageId);
+ PagePtr pagePtr;
+ pagePtr.i = physicalPageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ regFragPtr->emptyPrimPage = pagePtr.p->pageWord[ZPAGE_NEXT_POS];
+ return physicalPageId;
+}//Dbtup::getEmptyPage()
+
+Uint32 Dbtup::getRealpid(Fragrecord* const regFragPtr, Uint32 logicalPageId)
+{
+ PageRangePtr grpPageRangePtr;
+ Uint32 loopLimit;
+ Uint32 loopCount = 0;
+ Uint32 pageRangeLimit = cnoOfPageRangeRec;
+
+ grpPageRangePtr.i = regFragPtr->rootPageRange;
+ while (true) {
+ ndbrequire(loopCount++ < 100);
+ ndbrequire(grpPageRangePtr.i < pageRangeLimit);
+ ptrAss(grpPageRangePtr, pageRange);
+ loopLimit = grpPageRangePtr.p->currentIndexPos;
+ ndbrequire(loopLimit <= 3);
+ for (Uint32 i = 0; i <= loopLimit; i++) {
+ ljam();
+ if (grpPageRangePtr.p->startRange[i] <= logicalPageId) {
+ if (grpPageRangePtr.p->endRange[i] >= logicalPageId) {
+ if (grpPageRangePtr.p->type[i] == ZLEAF) {
+ ljam();
+ Uint32 realPageId = (logicalPageId - grpPageRangePtr.p->startRange[i]) +
+ grpPageRangePtr.p->basePageId[i];
+ return realPageId;
+ } else {
+ ndbrequire(grpPageRangePtr.p->type[i] == ZNON_LEAF);
+ grpPageRangePtr.i = grpPageRangePtr.p->basePageId[i];
+ }//if
+ }//if
+ }//if
+ }//for
+ }//while
+ return 0;
+}//Dbtup::getRealpid()
+
+Uint32 Dbtup::getNoOfPages(Fragrecord* const regFragPtr)
+{
+ return regFragPtr->noOfPages;
+}//Dbtup::getNoOfPages()
+
+void Dbtup::initPageRangeSize(Uint32 size)
+{
+ cnoOfPageRangeRec = size;
+}//Dbtup::initPageRangeSize()
+
+/* ---------------------------------------------------------------- */
+/* ----------------------- INSERT_PAGE_RANGE_TAB ------------------ */
+/* ---------------------------------------------------------------- */
+/* INSERT A PAGE RANGE INTO THE FRAGMENT */
+/* */
+/* NOTE: THE METHOD IS ATOMIC. EITHER THE ACTION IS */
+/* PERFORMED FULLY OR NO ACTION IS PERFORMED AT ALL. */
+/* TO SUPPORT THIS THE CODE HAS A CLEANUP PART AFTER */
+/* ERRORS. */
+/* ---------------------------------------------------------------- */
+bool Dbtup::insertPageRangeTab(Fragrecord* const regFragPtr,
+ Uint32 startPageId,
+ Uint32 noPages)
+{
+ PageRangePtr currPageRangePtr;
+ if (cfirstfreerange == RNIL) {
+ ljam();
+ return false;
+ }//if
+ currPageRangePtr.i = regFragPtr->currentPageRange;
+ if (currPageRangePtr.i == RNIL) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* THE FIRST PAGE RANGE IS HANDLED WITH SPECIAL CODE */
+/* ---------------------------------------------------------------- */
+ seizePagerange(currPageRangePtr);
+ regFragPtr->rootPageRange = currPageRangePtr.i;
+ currPageRangePtr.p->currentIndexPos = 0;
+ currPageRangePtr.p->parentPtr = RNIL;
+ } else {
+ ljam();
+ ptrCheckGuard(currPageRangePtr, cnoOfPageRangeRec, pageRange);
+ if (currPageRangePtr.p->currentIndexPos < 3) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* THE SIMPLE CASE WHEN IT IS ONLY NECESSARY TO FILL IN THE */
+/* NEXT EMPTY POSITION IN THE PAGE RANGE RECORD IS TREATED */
+/* BY COMMON CODE AT THE END OF THE SUBROUTINE. */
+/* ---------------------------------------------------------------- */
+ currPageRangePtr.p->currentIndexPos++;
+ } else {
+ ljam();
+ ndbrequire(currPageRangePtr.p->currentIndexPos == 3);
+ currPageRangePtr.i = leafPageRangeFull(regFragPtr, currPageRangePtr);
+ if (currPageRangePtr.i == RNIL) {
+ return false;
+ }//if
+ ptrCheckGuard(currPageRangePtr, cnoOfPageRangeRec, pageRange);
+ }//if
+ }//if
+ currPageRangePtr.p->startRange[currPageRangePtr.p->currentIndexPos] = regFragPtr->nextStartRange;
+/* ---------------------------------------------------------------- */
+/* NOW SET THE LEAF LEVEL PAGE RANGE RECORD PROPERLY */
+/* PAGE_RANGE_PTR REFERS TO LEAF RECORD WHEN ARRIVING HERE */
+/* ---------------------------------------------------------------- */
+ currPageRangePtr.p->endRange[currPageRangePtr.p->currentIndexPos] =
+ (regFragPtr->nextStartRange + noPages) - 1;
+ currPageRangePtr.p->basePageId[currPageRangePtr.p->currentIndexPos] = startPageId;
+ currPageRangePtr.p->type[currPageRangePtr.p->currentIndexPos] = ZLEAF;
+/* ---------------------------------------------------------------- */
+/* WE NEED TO UPDATE THE CURRENT PAGE RANGE IN CASE IT HAS */
+/* CHANGED. WE ALSO NEED TO UPDATE THE NEXT START RANGE */
+/* ---------------------------------------------------------------- */
+ regFragPtr->currentPageRange = currPageRangePtr.i;
+ regFragPtr->nextStartRange += noPages;
+/* ---------------------------------------------------------------- */
+/* WE NEED TO UPDATE THE END RANGE IN ALL PAGE RANGE RECORDS */
+/* UP TO THE ROOT. */
+/* ---------------------------------------------------------------- */
+ PageRangePtr loopPageRangePtr;
+ loopPageRangePtr = currPageRangePtr;
+ while (true) {
+ ljam();
+ loopPageRangePtr.i = loopPageRangePtr.p->parentPtr;
+ if (loopPageRangePtr.i != RNIL) {
+ ljam();
+ ptrCheckGuard(loopPageRangePtr, cnoOfPageRangeRec, pageRange);
+ ndbrequire(loopPageRangePtr.p->currentIndexPos < 4);
+ loopPageRangePtr.p->endRange[loopPageRangePtr.p->currentIndexPos] += noPages;
+ } else {
+ ljam();
+ break;
+ }//if
+ }//while
+ regFragPtr->noOfPages += noPages;
+ return true;
+}//Dbtup::insertPageRangeTab()
+
+
+void Dbtup::releaseFragPages(Fragrecord* const regFragPtr)
+{
+ if (regFragPtr->rootPageRange == RNIL) {
+ ljam();
+ return;
+ }//if
+ PageRangePtr regPRPtr;
+ regPRPtr.i = regFragPtr->rootPageRange;
+ ptrCheckGuard(regPRPtr, cnoOfPageRangeRec, pageRange);
+ while (true) {
+ ljam();
+ const Uint32 indexPos = regPRPtr.p->currentIndexPos;
+ ndbrequire(indexPos < 4);
+
+ const Uint32 basePageId = regPRPtr.p->basePageId[indexPos];
+ regPRPtr.p->basePageId[indexPos] = RNIL;
+ if (basePageId == RNIL) {
+ ljam();
+ /**
+ * Finished with indexPos continue with next
+ */
+ if (indexPos > 0) {
+ ljam();
+ regPRPtr.p->currentIndexPos--;
+ continue;
+ }//if
+
+ /* ---------------------------------------------------------------- */
+ /* THE PAGE RANGE REC IS EMPTY. RELEASE IT. */
+ /*----------------------------------------------------------------- */
+ Uint32 parentPtr = regPRPtr.p->parentPtr;
+ releasePagerange(regPRPtr);
+
+ if (parentPtr != RNIL) {
+ ljam();
+ regPRPtr.i = parentPtr;
+ ptrCheckGuard(regPRPtr, cnoOfPageRangeRec, pageRange);
+ continue;
+ }//if
+
+ ljam();
+ ndbrequire(regPRPtr.i == regFragPtr->rootPageRange);
+ initFragRange(regFragPtr);
+ return;
+ } else {
+ if (regPRPtr.p->type[indexPos] == ZNON_LEAF) {
+ jam();
+ /* ---------------------------------------------------------------- */
+ // A non-leaf node, we must release everything below it before we
+ // release this node.
+ /* ---------------------------------------------------------------- */
+ regPRPtr.i = basePageId;
+ ptrCheckGuard(regPRPtr, cnoOfPageRangeRec, pageRange);
+ } else {
+ jam();
+ ndbrequire(regPRPtr.p->type[indexPos] == ZLEAF);
+ /* ---------------------------------------------------------------- */
+ /* PAGE_RANGE_PTR /= RNIL AND THE CURRENT POS IS NOT A CHLED. */
+ /*----------------------------------------------------------------- */
+ const Uint32 start = regPRPtr.p->startRange[indexPos];
+ const Uint32 stop = regPRPtr.p->endRange[indexPos];
+ ndbrequire(stop >= start);
+ const Uint32 retNo = (stop - start + 1);
+ returnCommonArea(basePageId, retNo);
+ }//if
+ }//if
+ }//while
+}//Dbtup::releaseFragPages()
+
+void Dbtup::initializePageRange()
+{
+ PageRangePtr regPTRPtr;
+ for (regPTRPtr.i = 0;
+ regPTRPtr.i < cnoOfPageRangeRec; regPTRPtr.i++) {
+ ptrAss(regPTRPtr, pageRange);
+ regPTRPtr.p->nextFree = regPTRPtr.i + 1;
+ }//for
+ regPTRPtr.i = cnoOfPageRangeRec - 1;
+ ptrAss(regPTRPtr, pageRange);
+ regPTRPtr.p->nextFree = RNIL;
+ cfirstfreerange = 0;
+ c_noOfFreePageRanges = cnoOfPageRangeRec;
+}//Dbtup::initializePageRange()
+
+void Dbtup::initFragRange(Fragrecord* const regFragPtr)
+{
+ regFragPtr->emptyPrimPage = RNIL;
+ regFragPtr->rootPageRange = RNIL;
+ regFragPtr->currentPageRange = RNIL;
+ regFragPtr->noOfPages = 0;
+ regFragPtr->nextStartRange = 0;
+}//initFragRange()
+
+Uint32 Dbtup::allocFragPages(Fragrecord* const regFragPtr, Uint32 tafpNoAllocRequested)
+{
+ Uint32 tafpPagesAllocated = 0;
+ while (true) {
+ Uint32 noOfPagesAllocated = 0;
+ Uint32 noPagesToAllocate = tafpNoAllocRequested - tafpPagesAllocated;
+ Uint32 retPageRef = RNIL;
+ allocConsPages(noPagesToAllocate, noOfPagesAllocated, retPageRef);
+ if (noOfPagesAllocated == 0) {
+ ljam();
+ return tafpPagesAllocated;
+ }//if
+/* ---------------------------------------------------------------- */
+/* IT IS NOW TIME TO PUT THE ALLOCATED AREA INTO THE PAGE */
+/* RANGE TABLE. */
+/* ---------------------------------------------------------------- */
+ Uint32 startRange = regFragPtr->nextStartRange;
+ if (!insertPageRangeTab(regFragPtr, retPageRef, noOfPagesAllocated)) {
+ ljam();
+ returnCommonArea(retPageRef, noOfPagesAllocated);
+ return tafpPagesAllocated;
+ }//if
+ tafpPagesAllocated += noOfPagesAllocated;
+ Uint32 loopLimit = retPageRef + noOfPagesAllocated;
+ PagePtr loopPagePtr;
+/* ---------------------------------------------------------------- */
+/* SINCE A NUMBER OF PAGES WERE ALLOCATED FROM COMMON AREA */
+/* WITH SUCCESS IT IS NOW TIME TO CHANGE THE STATE OF */
+/* THOSE PAGES TO EMPTY_MM AND LINK THEM INTO THE EMPTY */
+/* PAGE LIST OF THE FRAGMENT. */
+/* ---------------------------------------------------------------- */
+ for (loopPagePtr.i = retPageRef; loopPagePtr.i < loopLimit; loopPagePtr.i++) {
+ ljam();
+ ptrCheckGuard(loopPagePtr, cnoOfPage, page);
+ loopPagePtr.p->pageWord[ZPAGE_STATE_POS] = ZEMPTY_MM;
+ loopPagePtr.p->pageWord[ZPAGE_FRAG_PAGE_ID_POS] = startRange +
+ (loopPagePtr.i - retPageRef);
+ loopPagePtr.p->pageWord[ZPAGE_NEXT_POS] = loopPagePtr.p->pageWord[ZPAGE_FRAG_PAGE_ID_POS] + 1;
+ }//for
+ loopPagePtr.i = (retPageRef + noOfPagesAllocated) - 1;
+ ptrCheckGuard(loopPagePtr, cnoOfPage, page);
+ loopPagePtr.p->pageWord[ZPAGE_NEXT_POS] = regFragPtr->emptyPrimPage;
+ regFragPtr->emptyPrimPage = startRange;
+/* ---------------------------------------------------------------- */
+/* WAS ENOUGH PAGES ALLOCATED OR ARE MORE NEEDED. */
+/* ---------------------------------------------------------------- */
+ if (tafpPagesAllocated < tafpNoAllocRequested) {
+ ljam();
+ } else {
+ ndbrequire(tafpPagesAllocated == tafpNoAllocRequested);
+ ljam();
+ return tafpNoAllocRequested;
+ }//if
+ }//while
+}//Dbtup::allocFragPages()
+
+void Dbtup::allocMoreFragPages(Fragrecord* const regFragPtr)
+{
+ Uint32 noAllocPages = regFragPtr->noOfPages >> 3; // 12.5%
+ noAllocPages += regFragPtr->noOfPages >> 4; // 6.25%
+ noAllocPages += 2;
+/* -----------------------------------------------------------------*/
+// We will grow by 18.75% plus two more additional pages to grow
+// a little bit quicker in the beginning.
+/* -----------------------------------------------------------------*/
+ allocFragPages(regFragPtr, noAllocPages);
+}//Dbtup::allocMoreFragPages()
+
+Uint32 Dbtup::leafPageRangeFull(Fragrecord* const regFragPtr, PageRangePtr currPageRangePtr)
+{
+/* ---------------------------------------------------------------- */
+/* THE COMPLEX CASE WHEN THE LEAF NODE IS FULL. GO UP THE TREE*/
+/* TO FIND THE FIRST RECORD WITH A FREE ENTRY. ALLOCATE NEW */
+/* PAGE RANGE RECORDS THEN ALL THE WAY DOWN TO THE LEAF LEVEL */
+/* AGAIN. THE TREE SHOULD ALWAYS REMAIN BALANCED. */
+/* ---------------------------------------------------------------- */
+ PageRangePtr parentPageRangePtr;
+ PageRangePtr foundPageRangePtr;
+ parentPageRangePtr = currPageRangePtr;
+ Uint32 tiprNoLevels = 1;
+ while (true) {
+ ljam();
+ parentPageRangePtr.i = parentPageRangePtr.p->parentPtr;
+ if (parentPageRangePtr.i == RNIL) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* WE HAVE REACHED THE ROOT. A NEW ROOT MUST BE ALLOCATED. */
+/* ---------------------------------------------------------------- */
+ if (c_noOfFreePageRanges < tiprNoLevels) {
+ ljam();
+ return RNIL;
+ }//if
+ PageRangePtr oldRootPRPtr;
+ PageRangePtr newRootPRPtr;
+ oldRootPRPtr.i = regFragPtr->rootPageRange;
+ ptrCheckGuard(oldRootPRPtr, cnoOfPageRangeRec, pageRange);
+ seizePagerange(newRootPRPtr);
+ regFragPtr->rootPageRange = newRootPRPtr.i;
+ oldRootPRPtr.p->parentPtr = newRootPRPtr.i;
+
+ newRootPRPtr.p->basePageId[0] = oldRootPRPtr.i;
+ newRootPRPtr.p->parentPtr = RNIL;
+ newRootPRPtr.p->startRange[0] = 0;
+ newRootPRPtr.p->endRange[0] = regFragPtr->nextStartRange - 1;
+ newRootPRPtr.p->type[0] = ZNON_LEAF;
+ newRootPRPtr.p->startRange[1] = regFragPtr->nextStartRange;
+ newRootPRPtr.p->endRange[1] = regFragPtr->nextStartRange - 1;
+ newRootPRPtr.p->type[1] = ZNON_LEAF;
+ newRootPRPtr.p->currentIndexPos = 1;
+ foundPageRangePtr = newRootPRPtr;
+ break;
+ } else {
+ ljam();
+ ptrCheckGuard(parentPageRangePtr, cnoOfPageRangeRec, pageRange);
+ if (parentPageRangePtr.p->currentIndexPos < 3) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* WE HAVE FOUND AN EMPTY ENTRY IN A PAGE RANGE RECORD. */
+/* ALLOCATE A NEW PAGE RANGE RECORD, FILL IN THE START RANGE, */
+/* ALLOCATE A NEW PAGE RANGE RECORD AND UPDATE THE POINTERS */
+/* ---------------------------------------------------------------- */
+ parentPageRangePtr.p->currentIndexPos++;
+ parentPageRangePtr.p->startRange[parentPageRangePtr.p->currentIndexPos] = regFragPtr->nextStartRange;
+ parentPageRangePtr.p->endRange[parentPageRangePtr.p->currentIndexPos] = regFragPtr->nextStartRange - 1;
+ parentPageRangePtr.p->type[parentPageRangePtr.p->currentIndexPos] = ZNON_LEAF;
+ foundPageRangePtr = parentPageRangePtr;
+ break;
+ } else {
+ ljam();
+ ndbrequire(parentPageRangePtr.p->currentIndexPos == 3);
+/* ---------------------------------------------------------------- */
+/* THE PAGE RANGE RECORD WAS FULL. FIND THE PARENT RECORD */
+/* AND INCREASE THE NUMBER OF LEVELS WE HAVE TRAVERSED */
+/* GOING UP THE TREE. */
+/* ---------------------------------------------------------------- */
+ tiprNoLevels++;
+ }//if
+ }//if
+ }//while
+/* ---------------------------------------------------------------- */
+/* REMEMBER THE ERROR LEVEL IN CASE OF ALLOCATION ERRORS */
+/* ---------------------------------------------------------------- */
+ PageRangePtr newPageRangePtr;
+ PageRangePtr prevPageRangePtr;
+ prevPageRangePtr = foundPageRangePtr;
+ if (c_noOfFreePageRanges < tiprNoLevels) {
+ ljam();
+ return RNIL;
+ }//if
+/* ---------------------------------------------------------------- */
+/* NOW WE HAVE PERFORMED THE SEARCH UPWARDS AND FILLED IN THE */
+/* PROPER FIELDS IN THE PAGE RANGE RECORD WHERE SOME SPACE */
+/* WAS FOUND. THE NEXT STEP IS TO ALLOCATE PAGE RANGES SO */
+/* THAT WE KEEP THE B-TREE BALANCED. THE NEW PAGE RANGE */
+/* ARE ALSO PROPERLY UPDATED ON THE PATH TO THE LEAF LEVEL. */
+/* ---------------------------------------------------------------- */
+ while (true) {
+ ljam();
+ seizePagerange(newPageRangePtr);
+ tiprNoLevels--;
+ ndbrequire(prevPageRangePtr.p->currentIndexPos < 4);
+ prevPageRangePtr.p->basePageId[prevPageRangePtr.p->currentIndexPos] = newPageRangePtr.i;
+ newPageRangePtr.p->parentPtr = prevPageRangePtr.i;
+ newPageRangePtr.p->currentIndexPos = 0;
+ if (tiprNoLevels > 0) {
+ ljam();
+ newPageRangePtr.p->startRange[0] = regFragPtr->nextStartRange;
+ newPageRangePtr.p->endRange[0] = regFragPtr->nextStartRange - 1;
+ newPageRangePtr.p->type[0] = ZNON_LEAF;
+ prevPageRangePtr = newPageRangePtr;
+ } else {
+ ljam();
+ break;
+ }//if
+ }//while
+ return newPageRangePtr.i;
+}//Dbtup::leafPageRangeFull()
+
+void Dbtup::releasePagerange(PageRangePtr regPRPtr)
+{
+ regPRPtr.p->nextFree = cfirstfreerange;
+ cfirstfreerange = regPRPtr.i;
+ c_noOfFreePageRanges++;
+}//Dbtup::releasePagerange()
+
+void Dbtup::seizePagerange(PageRangePtr& regPageRangePtr)
+{
+ regPageRangePtr.i = cfirstfreerange;
+ ptrCheckGuard(regPageRangePtr, cnoOfPageRangeRec, pageRange);
+ cfirstfreerange = regPageRangePtr.p->nextFree;
+ regPageRangePtr.p->nextFree = RNIL;
+ regPageRangePtr.p->currentIndexPos = 0;
+ regPageRangePtr.p->parentPtr = RNIL;
+ for (Uint32 i = 0; i < 4; i++) {
+ regPageRangePtr.p->startRange[i] = 1;
+ regPageRangePtr.p->endRange[i] = 0;
+ regPageRangePtr.p->type[i] = ZNON_LEAF;
+ regPageRangePtr.p->basePageId[i] = (Uint32)-1;
+ }//for
+ c_noOfFreePageRanges--;
+}//Dbtup::seizePagerange()
+
+void Dbtup::errorHandler(Uint32 errorCode)
+{
+ switch (errorCode) {
+ case 0:
+ ljam();
+ break;
+ case 1:
+ ljam();
+ break;
+ case 2:
+ ljam();
+ break;
+ default:
+ ljam();
+ }
+ ndbrequire(false);
+}//Dbtup::errorHandler()
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupRoutines.cpp b/ndb/src/kernel/blocks/dbtup/DbtupRoutines.cpp
new file mode 100644
index 00000000000..a5f56a356f9
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupRoutines.cpp
@@ -0,0 +1,896 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <AttributeDescriptor.hpp>
+#include "AttributeOffset.hpp"
+#include <AttributeHeader.hpp>
+
+#define ljam() { jamLine(3000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(3000 + __LINE__); }
+
+void
+Dbtup::setUpQueryRoutines(Tablerec* const regTabPtr)
+{
+ Uint32 startDescriptor = regTabPtr->tabDescriptor;
+ ndbrequire((startDescriptor + (regTabPtr->noOfAttr << ZAD_LOG_SIZE)) <= cnoOfTabDescrRec);
+ for (Uint32 i = 0; i < regTabPtr->noOfAttr; i++) {
+ Uint32 attrDescriptorStart = startDescriptor + (i << ZAD_LOG_SIZE);
+ Uint32 attrDescriptor = tableDescriptor[attrDescriptorStart].tabDescr;
+ if (!AttributeDescriptor::getDynamic(attrDescriptor)) {
+ if ((AttributeDescriptor::getArrayType(attrDescriptor) == ZNON_ARRAY) ||
+ (AttributeDescriptor::getArrayType(attrDescriptor) == ZFIXED_ARRAY)) {
+ if (!AttributeDescriptor::getNullable(attrDescriptor)) {
+ if (AttributeDescriptor::getSizeInWords(attrDescriptor) == 1) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readFixedSizeTHOneWordNotNULL;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateFixedSizeTHOneWordNotNULL;
+ } else if (AttributeDescriptor::getSizeInWords(attrDescriptor) == 2) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readFixedSizeTHTwoWordNotNULL;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateFixedSizeTHTwoWordNotNULL;
+ } else if (AttributeDescriptor::getSizeInWords(attrDescriptor) > 2) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readFixedSizeTHManyWordNotNULL;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateFixedSizeTHManyWordNotNULL;
+ } else {
+ ndbrequire(false);
+ }//if
+ } else {
+ if (AttributeDescriptor::getSizeInWords(attrDescriptor) == 1) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readFixedSizeTHOneWordNULLable;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateFixedSizeTHManyWordNULLable;
+ } else if (AttributeDescriptor::getSizeInWords(attrDescriptor) == 2) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readFixedSizeTHTwoWordNULLable;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateFixedSizeTHManyWordNULLable;
+ } else if (AttributeDescriptor::getSizeInWords(attrDescriptor) > 2) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readFixedSizeTHManyWordNULLable;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateFixedSizeTHManyWordNULLable;
+ } else {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readFixedSizeTHZeroWordNULLable;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateFixedSizeTHManyWordNULLable;
+ }//if
+ }//if
+ } else if (AttributeDescriptor::getArrayType(attrDescriptor) == ZVAR_ARRAY) {
+ if (!AttributeDescriptor::getNullable(attrDescriptor)) {
+ if (AttributeDescriptor::getArraySize(attrDescriptor) == 0) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readVarSizeUnlimitedNotNULL;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateVarSizeUnlimitedNotNULL;
+ } else if (AttributeDescriptor::getArraySize(attrDescriptor) > ZMAX_SMALL_VAR_ARRAY) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readBigVarSizeNotNULL;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateBigVarSizeNotNULL;
+ } else {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readSmallVarSizeNotNULL;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateSmallVarSizeNotNULL;
+ }//if
+ } else {
+ if (AttributeDescriptor::getArraySize(attrDescriptor) == 0) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readVarSizeUnlimitedNULLable;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateVarSizeUnlimitedNULLable;
+ } else if (AttributeDescriptor::getArraySize(attrDescriptor) > ZMAX_SMALL_VAR_ARRAY) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readBigVarSizeNULLable;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateBigVarSizeNULLable;
+ } else {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readSmallVarSizeNULLable;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateSmallVarSizeNULLable;
+ }//if
+ }//if
+ } else {
+ ndbrequire(false);
+ }//if
+ } else {
+ if ((AttributeDescriptor::getArrayType(attrDescriptor) == ZNON_ARRAY) ||
+ (AttributeDescriptor::getArrayType(attrDescriptor) == ZFIXED_ARRAY)) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readDynFixedSize;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateDynFixedSize;
+ } else if (AttributeDescriptor::getType(attrDescriptor) == ZVAR_ARRAY) {
+ if (AttributeDescriptor::getArraySize(attrDescriptor) == 0) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readDynVarSizeUnlimited;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateDynVarSizeUnlimited;
+ } else if (AttributeDescriptor::getArraySize(attrDescriptor) > ZMAX_SMALL_VAR_ARRAY) {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readDynBigVarSize;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateDynBigVarSize;
+ } else {
+ ljam();
+ regTabPtr->readFunctionArray[i] = &Dbtup::readDynSmallVarSize;
+ regTabPtr->updateFunctionArray[i] = &Dbtup::updateDynSmallVarSize;
+ }//if
+ } else {
+ ndbrequire(false);
+ }//if
+ }//if
+ }//for
+}//Dbtup::setUpQueryRoutines()
+
+/* ---------------------------------------------------------------- */
+/* THIS ROUTINE IS USED TO READ A NUMBER OF ATTRIBUTES IN THE */
+/* DATABASE AND PLACE THE RESULT IN ATTRINFO RECORDS. */
+//
+// In addition to the parameters used in the call it also relies on the
+// following variables set-up properly.
+//
+// operPtr.p Operation record pointer
+// fragptr.p Fragment record pointer
+// tabptr.p Table record pointer
+/* ---------------------------------------------------------------- */
+int Dbtup::readAttributes(Page* const pagePtr,
+ Uint32 tupHeadOffset,
+ Uint32* inBuffer,
+ Uint32 inBufLen,
+ Uint32* outBuffer,
+ Uint32 maxRead)
+{
+ Tablerec* const regTabPtr = tabptr.p;
+ Uint32 numAttributes = regTabPtr->noOfAttr;
+ Uint32 attrDescriptorStart = regTabPtr->tabDescriptor;
+ Uint32 inBufIndex = 0;
+
+ ndbrequire(attrDescriptorStart + (numAttributes << ZAD_LOG_SIZE) <= cnoOfTabDescrRec);
+
+ tOutBufIndex = 0;
+ tCheckOffset = regTabPtr->tupheadsize;
+ tMaxRead = maxRead;
+ tTupleHeader = &pagePtr->pageWord[tupHeadOffset];
+
+ ndbrequire(tupHeadOffset + tCheckOffset <= ZWORDS_ON_PAGE);
+ while (inBufIndex < inBufLen) {
+ Uint32 tmpAttrBufIndex = tOutBufIndex;
+ AttributeHeader ahIn(inBuffer[inBufIndex]);
+ inBufIndex++;
+ Uint32 attributeId = ahIn.getAttributeId();
+ Uint32 attrDescriptorIndex = attrDescriptorStart + (attributeId << ZAD_LOG_SIZE);
+ ljam();
+
+ AttributeHeader::init(&outBuffer[tmpAttrBufIndex], attributeId, 0);
+ AttributeHeader* ahOut = (AttributeHeader*)&outBuffer[tmpAttrBufIndex];
+ tOutBufIndex = tmpAttrBufIndex + 1;
+ if (attributeId < numAttributes) {
+ Uint32 attributeDescriptor = tableDescriptor[attrDescriptorIndex].tabDescr;
+ Uint32 attributeOffset = tableDescriptor[attrDescriptorIndex + 1].tabDescr;
+ ReadFunction f = regTabPtr->readFunctionArray[attributeId];
+ if ((this->*f)(outBuffer,
+ ahOut,
+ attributeDescriptor,
+ attributeOffset)) {
+ continue;
+ } else {
+ return (Uint32)-1;
+ }//if
+ } else {
+ terrorCode = ZATTRIBUTE_ID_ERROR;
+ return (Uint32)-1;
+ }//if
+ }//while
+ return tOutBufIndex;
+}//Dbtup::readAttributes()
+
+int Dbtup::readAttributesWithoutHeader(Page* const pagePtr,
+ Uint32 tupHeadOffset,
+ Uint32* inBuffer,
+ Uint32 inBufLen,
+ Uint32* outBuffer,
+ Uint32* attrBuffer,
+ Uint32 maxRead)
+{
+ Tablerec* const regTabPtr = tabptr.p;
+ Uint32 numAttributes = regTabPtr->noOfAttr;
+ Uint32 attrDescriptorStart = regTabPtr->tabDescriptor;
+ Uint32 inBufIndex = 0;
+ Uint32 attrBufIndex = 0;
+
+ ndbrequire(attrDescriptorStart + (numAttributes << ZAD_LOG_SIZE) <= cnoOfTabDescrRec);
+
+ tOutBufIndex = 0;
+ tCheckOffset = regTabPtr->tupheadsize;
+ tMaxRead = maxRead;
+ tTupleHeader = &pagePtr->pageWord[tupHeadOffset];
+
+ ndbrequire(tupHeadOffset + tCheckOffset <= ZWORDS_ON_PAGE);
+ while (inBufIndex < inBufLen) {
+ AttributeHeader ahIn(inBuffer[inBufIndex]);
+ inBufIndex++;
+ Uint32 attributeId = ahIn.getAttributeId();
+ Uint32 attrDescriptorIndex = attrDescriptorStart + (attributeId << ZAD_LOG_SIZE);
+ ljam();
+
+ AttributeHeader::init(&attrBuffer[attrBufIndex], attributeId, 0);
+ AttributeHeader* ahOut = (AttributeHeader*)&attrBuffer[attrBufIndex];
+ attrBufIndex++;
+ if (attributeId < numAttributes) {
+ Uint32 attributeDescriptor = tableDescriptor[attrDescriptorIndex].tabDescr;
+ Uint32 attributeOffset = tableDescriptor[attrDescriptorIndex + 1].tabDescr;
+ ReadFunction f = regTabPtr->readFunctionArray[attributeId];
+ if ((this->*f)(outBuffer,
+ ahOut,
+ attributeDescriptor,
+ attributeOffset)) {
+ continue;
+ } else {
+ return (Uint32)-1;
+ }//if
+ } else {
+ terrorCode = ZATTRIBUTE_ID_ERROR;
+ return (Uint32)-1;
+ }//if
+ }//while
+ ndbrequire(attrBufIndex == inBufLen);
+ return tOutBufIndex;
+}//Dbtup::readAttributes()
+
+bool
+Dbtup::readFixedSizeTHOneWordNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ Uint32 indexBuf = tOutBufIndex;
+ Uint32 readOffset = AttributeOffset::getOffset(attrDes2);
+ Uint32 const wordRead = tTupleHeader[readOffset];
+ Uint32 newIndexBuf = indexBuf + 1;
+ Uint32 maxRead = tMaxRead;
+
+ ndbrequire(readOffset < tCheckOffset);
+ if (newIndexBuf <= maxRead) {
+ ljam();
+ outBuffer[indexBuf] = wordRead;
+ ahOut->setDataSize(1);
+ tOutBufIndex = newIndexBuf;
+ return true;
+ } else {
+ ljam();
+ terrorCode = ZTRY_TO_READ_TOO_MUCH_ERROR;
+ return false;
+ }//if
+}//Dbtup::readFixedSizeTHOneWordNotNULL()
+
+bool
+Dbtup::readFixedSizeTHTwoWordNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ Uint32 indexBuf = tOutBufIndex;
+ Uint32 readOffset = AttributeOffset::getOffset(attrDes2);
+ Uint32 const wordReadFirst = tTupleHeader[readOffset];
+ Uint32 const wordReadSecond = tTupleHeader[readOffset + 1];
+ Uint32 newIndexBuf = indexBuf + 2;
+ Uint32 maxRead = tMaxRead;
+
+ ndbrequire(readOffset + 1 < tCheckOffset);
+ if (newIndexBuf <= maxRead) {
+ ljam();
+ ahOut->setDataSize(2);
+ outBuffer[indexBuf] = wordReadFirst;
+ outBuffer[indexBuf + 1] = wordReadSecond;
+ tOutBufIndex = newIndexBuf;
+ return true;
+ } else {
+ ljam();
+ terrorCode = ZTRY_TO_READ_TOO_MUCH_ERROR;
+ return false;
+ }//if
+}//Dbtup::readFixedSizeTHTwoWordNotNULL()
+
+bool
+Dbtup::readFixedSizeTHManyWordNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ Uint32 indexBuf = tOutBufIndex;
+ Uint32 readOffset = AttributeOffset::getOffset(attrDes2);
+ Uint32 attrNoOfWords = AttributeDescriptor::getSizeInWords(attrDescriptor);
+ Uint32 newIndexBuf = indexBuf + attrNoOfWords;
+ Uint32 maxRead = tMaxRead;
+
+ ndbrequire((readOffset + attrNoOfWords - 1) < tCheckOffset);
+ if (newIndexBuf <= maxRead) {
+ ljam();
+ ahOut->setDataSize(attrNoOfWords);
+ MEMCOPY_NO_WORDS(&outBuffer[indexBuf],
+ &tTupleHeader[readOffset],
+ attrNoOfWords);
+ tOutBufIndex = newIndexBuf;
+ return true;
+ } else {
+ ljam();
+ terrorCode = ZTRY_TO_READ_TOO_MUCH_ERROR;
+ return false;
+ }//if
+}//Dbtup::readFixedSizeTHManyWordNotNULL()
+
+bool
+Dbtup::readFixedSizeTHOneWordNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ if (!nullFlagCheck(attrDes2)) {
+ ljam();
+ return readFixedSizeTHOneWordNotNULL(outBuffer,
+ ahOut,
+ attrDescriptor,
+ attrDes2);
+ } else {
+ ljam();
+ ahOut->setNULL();
+ return true;
+ }//if
+}//Dbtup::readFixedSizeTHOneWordNULLable()
+
+bool
+Dbtup::readFixedSizeTHTwoWordNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ if (!nullFlagCheck(attrDes2)) {
+ ljam();
+ return readFixedSizeTHTwoWordNotNULL(outBuffer,
+ ahOut,
+ attrDescriptor,
+ attrDes2);
+ } else {
+ ljam();
+ ahOut->setNULL();
+ return true;
+ }//if
+}//Dbtup::readFixedSizeTHTwoWordNULLable()
+
+bool
+Dbtup::readFixedSizeTHManyWordNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ljam();
+ if (!nullFlagCheck(attrDes2)) {
+ ljam();
+ return readFixedSizeTHManyWordNotNULL(outBuffer,
+ ahOut,
+ attrDescriptor,
+ attrDes2);
+ } else {
+ ljam();
+ ahOut->setNULL();
+ return true;
+ }//if
+}//Dbtup::readFixedSizeTHManyWordNULLable()
+
+bool
+Dbtup::readFixedSizeTHZeroWordNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ if (nullFlagCheck(attrDes2)) {
+ ljam();
+ ahOut->setNULL();
+ }//if
+ return true;
+}//Dbtup::readFixedSizeTHZeroWordNULLable()
+
+bool
+Dbtup::nullFlagCheck(Uint32 attrDes2)
+{
+ Tablerec* const regTabPtr = tabptr.p;
+ Uint32 nullFlagOffsetInTuple = AttributeOffset::getNullFlagOffset(attrDes2);
+ ndbrequire(nullFlagOffsetInTuple < regTabPtr->tupNullWords);
+ nullFlagOffsetInTuple += regTabPtr->tupNullIndex;
+ ndbrequire(nullFlagOffsetInTuple < tCheckOffset);
+
+ return (AttributeOffset::isNULL(tTupleHeader[nullFlagOffsetInTuple], attrDes2));
+}//Dbtup::nullFlagCheck()
+
+bool
+Dbtup::readVariableSizedAttr(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readVariableSizedAttr()
+
+bool
+Dbtup::readVarSizeUnlimitedNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readVarSizeUnlimitedNotNULL()
+
+bool
+Dbtup::readVarSizeUnlimitedNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readVarSizeUnlimitedNULLable()
+
+bool
+Dbtup::readBigVarSizeNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readBigVarSizeNotNULL()
+
+bool
+Dbtup::readBigVarSizeNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readBigVarSizeNULLable()
+
+bool
+Dbtup::readSmallVarSizeNotNULL(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readSmallVarSizeNotNULL()
+
+bool
+Dbtup::readSmallVarSizeNULLable(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readSmallVarSizeNULLable()
+
+bool
+Dbtup::readDynFixedSize(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readDynFixedSize()
+
+bool
+Dbtup::readDynVarSizeUnlimited(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readDynVarSizeUnlimited()
+
+bool
+Dbtup::readDynBigVarSize(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readDynBigVarSize()
+
+bool
+Dbtup::readDynSmallVarSize(Uint32* outBuffer,
+ AttributeHeader* ahOut,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::readDynSmallVarSize()
+
+/* ---------------------------------------------------------------------- */
+/* THIS ROUTINE IS USED TO UPDATE A NUMBER OF ATTRIBUTES. IT IS */
+/* USED BY THE INSERT ROUTINE, THE UPDATE ROUTINE AND IT CAN BE */
+/* CALLED SEVERAL TIMES FROM THE INTERPRETER. */
+// In addition to the parameters used in the call it also relies on the
+// following variables set-up properly.
+//
+// pagep.p Page record pointer
+// fragptr.p Fragment record pointer
+// operPtr.p Operation record pointer
+// tabptr.p Table record pointer
+/* ---------------------------------------------------------------------- */
+int Dbtup::updateAttributes(Page* const pagePtr,
+ Uint32 tupHeadOffset,
+ Uint32* inBuffer,
+ Uint32 inBufLen)
+{
+ Tablerec* const regTabPtr = tabptr.p;
+ Operationrec* const regOperPtr = operPtr.p;
+ Uint32 numAttributes = regTabPtr->noOfAttr;
+ Uint32 attrDescriptorStart = regTabPtr->tabDescriptor;
+ ndbrequire(attrDescriptorStart + (numAttributes << ZAD_LOG_SIZE) <= cnoOfTabDescrRec);
+
+ tCheckOffset = regTabPtr->tupheadsize;
+ tTupleHeader = &pagePtr->pageWord[tupHeadOffset];
+ Uint32 inBufIndex = 0;
+ tInBufIndex = 0;
+ tInBufLen = inBufLen;
+
+ ndbrequire(tupHeadOffset + tCheckOffset <= ZWORDS_ON_PAGE);
+ while (inBufIndex < inBufLen) {
+ AttributeHeader ahIn(inBuffer[inBufIndex]);
+ Uint32 attributeId = ahIn.getAttributeId();
+ Uint32 attrDescriptorIndex = attrDescriptorStart + (attributeId << ZAD_LOG_SIZE);
+ if (attributeId < numAttributes) {
+ Uint32 attrDescriptor = tableDescriptor[attrDescriptorIndex].tabDescr;
+ Uint32 attributeOffset = tableDescriptor[attrDescriptorIndex + 1].tabDescr;
+ if ((AttributeDescriptor::getPrimaryKey(attrDescriptor)) &&
+ (regOperPtr->optype != ZINSERT)) {
+ if (checkUpdateOfPrimaryKey(&inBuffer[inBufIndex], regTabPtr)) {
+ ljam();
+ terrorCode = ZTRY_UPDATE_PRIMARY_KEY;
+ return -1;
+ }//if
+ }//if
+ UpdateFunction f = regTabPtr->updateFunctionArray[attributeId];
+ ljam();
+ regOperPtr->changeMask.set(attributeId);
+ if ((this->*f)(inBuffer,
+ attrDescriptor,
+ attributeOffset)) {
+ inBufIndex = tInBufIndex;
+ continue;
+ } else {
+ ljam();
+ return (Uint32)-1;
+ }//if
+ } else {
+ ljam();
+ terrorCode = ZATTRIBUTE_ID_ERROR;
+ return (Uint32)-1;
+ }//if
+ }//while
+ return 0;
+}//Dbtup::updateAttributes()
+
+bool
+Dbtup::checkUpdateOfPrimaryKey(Uint32* updateBuffer, Tablerec* const regTabPtr)
+{
+ Uint32 keyReadBuffer[MAX_KEY_SIZE_IN_WORDS];
+ Uint32 attributeHeader;
+ AttributeHeader* ahOut = (AttributeHeader*)&attributeHeader;
+ AttributeHeader ahIn(*updateBuffer);
+ Uint32 attributeId = ahIn.getAttributeId();
+ Uint32 attrDescriptorIndex = regTabPtr->tabDescriptor + (attributeId << ZAD_LOG_SIZE);
+ Uint32 attrDescriptor = tableDescriptor[attrDescriptorIndex].tabDescr;
+ Uint32 attributeOffset = tableDescriptor[attrDescriptorIndex + 1].tabDescr;
+ ReadFunction f = regTabPtr->readFunctionArray[attributeId];
+
+ AttributeHeader::init(&attributeHeader, attributeId, 0);
+ tOutBufIndex = 0;
+ tMaxRead = MAX_KEY_SIZE_IN_WORDS;
+
+ ndbrequire((this->*f)(&keyReadBuffer[0], ahOut, attrDescriptor, attributeOffset));
+ ndbrequire(tOutBufIndex == ahOut->getDataSize());
+ if (ahIn.getDataSize() != ahOut->getDataSize()) {
+ ljam();
+ return true;
+ }//if
+ if (memcmp(&keyReadBuffer[0], &updateBuffer[1], tOutBufIndex << 2) != 0) {
+ ljam();
+ return true;
+ }//if
+ return false;
+}//Dbtup::checkUpdateOfPrimaryKey()
+
+#if 0
+void Dbtup::checkPages(Fragrecord* const regFragPtr)
+{
+ Uint32 noPages = getNoOfPages(regFragPtr);
+ for (Uint32 i = 0; i < noPages ; i++) {
+ PagePtr pagePtr;
+ pagePtr.i = getRealpid(regFragPtr, i);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ ndbrequire(pagePtr.p->pageWord[1] != (RNIL - 1));
+ }
+}
+#endif
+
+bool
+Dbtup::updateFixedSizeTHOneWordNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ Uint32 indexBuf = tInBufIndex;
+ Uint32 inBufLen = tInBufLen;
+ Uint32 updateOffset = AttributeOffset::getOffset(attrDes2);
+ AttributeHeader ahIn(inBuffer[indexBuf]);
+ Uint32 nullIndicator = ahIn.isNULL();
+ Uint32 newIndex = indexBuf + 2;
+ ndbrequire(updateOffset < tCheckOffset);
+
+ if (newIndex <= inBufLen) {
+ Uint32 updateWord = inBuffer[indexBuf + 1];
+ if (!nullIndicator) {
+ ljam();
+ tInBufIndex = newIndex;
+ tTupleHeader[updateOffset] = updateWord;
+ return true;
+ } else {
+ ljam();
+ terrorCode = ZNOT_NULL_ATTR;
+ return false;
+ }//if
+ } else {
+ ljam();
+ terrorCode = ZAI_INCONSISTENCY_ERROR;
+ return false;
+ }//if
+ return true;
+}//Dbtup::updateFixedSizeTHOneWordNotNULL()
+
+bool
+Dbtup::updateFixedSizeTHTwoWordNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ Uint32 indexBuf = tInBufIndex;
+ Uint32 inBufLen = tInBufLen;
+ Uint32 updateOffset = AttributeOffset::getOffset(attrDes2);
+ AttributeHeader ahIn(inBuffer[indexBuf]);
+ Uint32 nullIndicator = ahIn.isNULL();
+ Uint32 newIndex = indexBuf + 3;
+ ndbrequire((updateOffset + 1) < tCheckOffset);
+
+ if (newIndex <= inBufLen) {
+ Uint32 updateWord1 = inBuffer[indexBuf + 1];
+ Uint32 updateWord2 = inBuffer[indexBuf + 2];
+ if (!nullIndicator) {
+ ljam();
+ tInBufIndex = newIndex;
+ tTupleHeader[updateOffset] = updateWord1;
+ tTupleHeader[updateOffset + 1] = updateWord2;
+ return true;
+ } else {
+ ljam();
+ terrorCode = ZNOT_NULL_ATTR;
+ return false;
+ }//if
+ } else {
+ ljam();
+ terrorCode = ZAI_INCONSISTENCY_ERROR;
+ return false;
+ }//if
+}//Dbtup::updateFixedSizeTHTwoWordNotNULL()
+
+bool
+Dbtup::updateFixedSizeTHManyWordNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ Uint32 indexBuf = tInBufIndex;
+ Uint32 inBufLen = tInBufLen;
+ Uint32 updateOffset = AttributeOffset::getOffset(attrDes2);
+ AttributeHeader ahIn(inBuffer[indexBuf]);
+ Uint32 nullIndicator = ahIn.isNULL();
+ Uint32 noOfWords = AttributeDescriptor::getSizeInWords(attrDescriptor);
+ Uint32 newIndex = indexBuf + noOfWords + 1;
+ ndbrequire((updateOffset + noOfWords - 1) < tCheckOffset);
+
+ if (newIndex <= inBufLen) {
+ if (!nullIndicator) {
+ ljam();
+ tInBufIndex = newIndex;
+ MEMCOPY_NO_WORDS(&tTupleHeader[updateOffset],
+ &inBuffer[indexBuf + 1],
+ noOfWords);
+ return true;
+ } else {
+ ljam();
+ terrorCode = ZNOT_NULL_ATTR;
+ return false;
+ }//if
+ } else {
+ ljam();
+ terrorCode = ZAI_INCONSISTENCY_ERROR;
+ return false;
+ }//if
+}//Dbtup::updateFixedSizeTHManyWordNotNULL()
+
+bool
+Dbtup::updateFixedSizeTHManyWordNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ Tablerec* const regTabPtr = tabptr.p;
+ AttributeHeader ahIn(inBuffer[tInBufIndex]);
+ Uint32 nullIndicator = ahIn.isNULL();
+ Uint32 nullFlagOffset = AttributeOffset::getNullFlagOffset(attrDes2);
+ Uint32 nullFlagBitOffset = AttributeOffset::getNullFlagBitOffset(attrDes2);
+ Uint32 nullWordOffset = nullFlagOffset + regTabPtr->tupNullIndex;
+ ndbrequire((nullFlagOffset < regTabPtr->tupNullWords) &&
+ (nullWordOffset < tCheckOffset));
+ Uint32 nullBits = tTupleHeader[nullWordOffset];
+
+ if (!nullIndicator) {
+ nullBits &= (~(1 << nullFlagBitOffset));
+ ljam();
+ tTupleHeader[nullWordOffset] = nullBits;
+ return updateFixedSizeTHManyWordNotNULL(inBuffer,
+ attrDescriptor,
+ attrDes2);
+ } else {
+ Uint32 newIndex = tInBufIndex + 1;
+ if (newIndex <= tInBufLen) {
+ nullBits |= (1 << nullFlagBitOffset);
+ ljam();
+ tTupleHeader[nullWordOffset] = nullBits;
+ tInBufIndex = newIndex;
+ return true;
+ } else {
+ ljam();
+ terrorCode = ZAI_INCONSISTENCY_ERROR;
+ return false;
+ }//if
+ }//if
+}//Dbtup::updateFixedSizeTHManyWordNULLable()
+
+bool
+Dbtup::updateVariableSizedAttr(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateVariableSizedAttr()
+
+bool
+Dbtup::updateVarSizeUnlimitedNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateVarSizeUnlimitedNotNULL()
+
+bool
+Dbtup::updateVarSizeUnlimitedNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateVarSizeUnlimitedNULLable()
+
+bool
+Dbtup::updateBigVarSizeNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateBigVarSizeNotNULL()
+
+bool
+Dbtup::updateBigVarSizeNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateBigVarSizeNULLable()
+
+bool
+Dbtup::updateSmallVarSizeNotNULL(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateSmallVarSizeNotNULL()
+
+bool
+Dbtup::updateSmallVarSizeNULLable(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateSmallVarSizeNULLable()
+
+bool
+Dbtup::updateDynFixedSize(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateDynFixedSize()
+
+bool
+Dbtup::updateDynVarSizeUnlimited(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateDynVarSizeUnlimited()
+
+bool
+Dbtup::updateDynBigVarSize(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateDynBigVarSize()
+
+bool
+Dbtup::updateDynSmallVarSize(Uint32* inBuffer,
+ Uint32 attrDescriptor,
+ Uint32 attrDes2)
+{
+ ljam();
+ terrorCode = ZVAR_SIZED_NOT_SUPPORTED;
+ return false;
+}//Dbtup::updateDynSmallVarSize()
+
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupStoredProcDef.cpp b/ndb/src/kernel/blocks/dbtup/DbtupStoredProcDef.cpp
new file mode 100644
index 00000000000..3b957688a1c
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupStoredProcDef.cpp
@@ -0,0 +1,230 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+
+#define ljam() { jamLine(18000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(18000 + __LINE__); }
+
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+/* ------------ADD/DROP STORED PROCEDURE MODULE ------------------- */
+/* ---------------------------------------------------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::execSTORED_PROCREQ(Signal* signal)
+{
+ OperationrecPtr regOperPtr;
+ TablerecPtr regTabPtr;
+ ljamEntry();
+ regOperPtr.i = signal->theData[0];
+ ptrCheckGuard(regOperPtr, cnoOfOprec, operationrec);
+ regTabPtr.i = signal->theData[1];
+ ptrCheckGuard(regTabPtr, cnoOfTablerec, tablerec);
+
+ Uint32 requestInfo = signal->theData[3];
+
+ ndbrequire(regOperPtr.p->transstate == IDLE ||
+ ((regOperPtr.p->transstate == ERROR_WAIT_STORED_PROCREQ) &&
+ (requestInfo == ZSTORED_PROCEDURE_DELETE)));
+ ndbrequire(regTabPtr.p->tableStatus == DEFINED);
+ switch (requestInfo) {
+ case ZSCAN_PROCEDURE:
+ ljam();
+ scanProcedure(signal,
+ regOperPtr.p,
+ signal->theData[4]);
+ break;
+ case ZCOPY_PROCEDURE:
+ ljam();
+ copyProcedure(signal, regTabPtr, regOperPtr.p);
+ break;
+ case ZSTORED_PROCEDURE_DELETE:
+ ljam();
+ deleteScanProcedure(signal, regOperPtr.p);
+ break;
+ default:
+ ndbrequire(false);
+ }//switch
+}//Dbtup::execSTORED_PROCREQ()
+
+void Dbtup::deleteScanProcedure(Signal* signal,
+ Operationrec* regOperPtr)
+{
+ StoredProcPtr storedPtr;
+ Uint32 storedProcId = signal->theData[4];
+ c_storedProcPool.getPtr(storedPtr, storedProcId);
+ ndbrequire(storedPtr.p->storedCode == ZSCAN_PROCEDURE);
+ ndbrequire(storedPtr.p->storedCounter == 0);
+ Uint32 firstAttrinbuf = storedPtr.p->storedLinkFirst;
+ storedPtr.p->storedCode = ZSTORED_PROCEDURE_FREE;
+ storedPtr.p->storedLinkFirst = RNIL;
+ storedPtr.p->storedLinkLast = RNIL;
+ storedPtr.p->storedProcLength = 0;
+ c_storedProcPool.release(storedPtr);
+ freeAttrinbufrec(firstAttrinbuf);
+ regOperPtr->currentAttrinbufLen = 0;
+ regOperPtr->transstate = IDLE;
+ signal->theData[0] = regOperPtr->userpointer;
+ signal->theData[1] = storedProcId;
+ sendSignal(regOperPtr->userblockref, GSN_STORED_PROCCONF, signal, 2, JBB);
+}//Dbtup::deleteScanProcedure()
+
+void Dbtup::scanProcedure(Signal* signal,
+ Operationrec* regOperPtr,
+ Uint32 lenAttrInfo)
+{
+//--------------------------------------------------------
+// We introduce the maxCheck so that there is always one
+// stored procedure entry free for copy procedures. Thus
+// no amount of scanning can cause problems for the node
+// recovery functionality.
+//--------------------------------------------------------
+ StoredProcPtr storedPtr;
+ c_storedProcPool.seize(storedPtr);
+ ndbrequire(storedPtr.i != RNIL);
+ storedPtr.p->storedCode = ZSCAN_PROCEDURE;
+ storedPtr.p->storedCounter = 0;
+ storedPtr.p->storedProcLength = lenAttrInfo;
+ storedPtr.p->storedLinkFirst = RNIL;
+ storedPtr.p->storedLinkLast = RNIL;
+ regOperPtr->transstate = WAIT_STORED_PROCEDURE_ATTR_INFO;
+ regOperPtr->attrinbufLen = lenAttrInfo;
+ regOperPtr->currentAttrinbufLen = 0;
+ regOperPtr->pageOffset = storedPtr.i;
+}//Dbtup::scanProcedure()
+
+void Dbtup::copyProcedure(Signal* signal,
+ TablerecPtr regTabPtr,
+ Operationrec* regOperPtr)
+{
+ Uint32 TnoOfAttributes = regTabPtr.p->noOfAttr;
+ scanProcedure(signal,
+ regOperPtr,
+ TnoOfAttributes);
+
+ Uint32 length = 0;
+ for (Uint32 Ti = 0; Ti < TnoOfAttributes; Ti++) {
+ AttributeHeader::init(&signal->theData[length + 1], Ti, 0);
+ length++;
+ if (length == 24) {
+ ljam();
+ ndbrequire(storedProcedureAttrInfo(signal, regOperPtr, length, 1, true));
+ length = 0;
+ }//if
+ }//for
+ if (length != 0) {
+ ljam();
+ ndbrequire(storedProcedureAttrInfo(signal, regOperPtr, length, 1, true));
+ }//if
+ ndbrequire(regOperPtr->currentAttrinbufLen == 0);
+}//Dbtup::copyProcedure()
+
+bool Dbtup::storedProcedureAttrInfo(Signal* signal,
+ Operationrec* regOperPtr,
+ Uint32 length,
+ Uint32 firstWord,
+ bool copyProcedure)
+{
+ AttrbufrecPtr regAttrPtr;
+ Uint32 RnoFree = cnoFreeAttrbufrec;
+ if (ERROR_INSERTED(4004) && !copyProcedure) {
+ CLEAR_ERROR_INSERT_VALUE;
+ storedSeizeAttrinbufrecErrorLab(signal, regOperPtr);
+ return false;
+ }//if
+ regOperPtr->currentAttrinbufLen += length;
+ ndbrequire(regOperPtr->currentAttrinbufLen <= regOperPtr->attrinbufLen);
+ if ((RnoFree > MIN_ATTRBUF) ||
+ (copyProcedure)) {
+ ljam();
+ regAttrPtr.i = cfirstfreeAttrbufrec;
+ ptrCheckGuard(regAttrPtr, cnoOfAttrbufrec, attrbufrec);
+ regAttrPtr.p->attrbuf[ZBUF_DATA_LEN] = 0;
+ cfirstfreeAttrbufrec = regAttrPtr.p->attrbuf[ZBUF_NEXT];
+ cnoFreeAttrbufrec = RnoFree - 1;
+ regAttrPtr.p->attrbuf[ZBUF_NEXT] = RNIL;
+ } else {
+ ljam();
+ storedSeizeAttrinbufrecErrorLab(signal, regOperPtr);
+ return false;
+ }//if
+ if (regOperPtr->firstAttrinbufrec == RNIL) {
+ ljam();
+ regOperPtr->firstAttrinbufrec = regAttrPtr.i;
+ }//if
+ regAttrPtr.p->attrbuf[ZBUF_NEXT] = RNIL;
+ if (regOperPtr->lastAttrinbufrec != RNIL) {
+ AttrbufrecPtr tempAttrinbufptr;
+ ljam();
+ tempAttrinbufptr.i = regOperPtr->lastAttrinbufrec;
+ ptrCheckGuard(tempAttrinbufptr, cnoOfAttrbufrec, attrbufrec);
+ tempAttrinbufptr.p->attrbuf[ZBUF_NEXT] = regAttrPtr.i;
+ }//if
+ regOperPtr->lastAttrinbufrec = regAttrPtr.i;
+
+ regAttrPtr.p->attrbuf[ZBUF_DATA_LEN] = length;
+ MEMCOPY_NO_WORDS(&regAttrPtr.p->attrbuf[0],
+ &signal->theData[firstWord],
+ length);
+
+ if (regOperPtr->currentAttrinbufLen < regOperPtr->attrinbufLen) {
+ ljam();
+ return true;
+ }//if
+ if (ERROR_INSERTED(4005) && !copyProcedure) {
+ CLEAR_ERROR_INSERT_VALUE;
+ storedSeizeAttrinbufrecErrorLab(signal, regOperPtr);
+ return false;
+ }//if
+
+ StoredProcPtr storedPtr;
+ c_storedProcPool.getPtr(storedPtr, (Uint32)regOperPtr->pageOffset);
+ ndbrequire(storedPtr.p->storedCode == ZSCAN_PROCEDURE);
+
+ regOperPtr->currentAttrinbufLen = 0;
+ storedPtr.p->storedLinkFirst = regOperPtr->firstAttrinbufrec;
+ storedPtr.p->storedLinkLast = regOperPtr->lastAttrinbufrec;
+ regOperPtr->firstAttrinbufrec = RNIL;
+ regOperPtr->lastAttrinbufrec = RNIL;
+ regOperPtr->transstate = IDLE;
+ signal->theData[0] = regOperPtr->userpointer;
+ signal->theData[1] = storedPtr.i;
+ sendSignal(regOperPtr->userblockref, GSN_STORED_PROCCONF, signal, 2, JBB);
+ return true;
+}//Dbtup::storedProcedureAttrInfo()
+
+void Dbtup::storedSeizeAttrinbufrecErrorLab(Signal* signal,
+ Operationrec* regOperPtr)
+{
+ StoredProcPtr storedPtr;
+ c_storedProcPool.getPtr(storedPtr, (Uint32)regOperPtr->pageOffset);
+ ndbrequire(storedPtr.p->storedCode == ZSCAN_PROCEDURE);
+
+ storedPtr.p->storedLinkFirst = regOperPtr->firstAttrinbufrec;
+ regOperPtr->firstAttrinbufrec = RNIL;
+ regOperPtr->lastAttrinbufrec = RNIL;
+ regOperPtr->transstate = ERROR_WAIT_STORED_PROCREQ;
+ signal->theData[0] = regOperPtr->userpointer;
+ signal->theData[1] = ZSTORED_SEIZE_ATTRINBUFREC_ERROR;
+ signal->theData[2] = regOperPtr->pageOffset;
+ sendSignal(regOperPtr->userblockref, GSN_STORED_PROCREF, signal, 3, JBB);
+}//Dbtup::storedSeizeAttrinbufrecErrorLab()
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupSystemRestart.cpp b/ndb/src/kernel/blocks/dbtup/DbtupSystemRestart.cpp
new file mode 100644
index 00000000000..580d764c96f
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupSystemRestart.cpp
@@ -0,0 +1,1003 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/FsConf.hpp>
+#include <signaldata/FsRef.hpp>
+
+#define ljam() { jamLine(26000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(26000 + __LINE__); }
+
+/* **************************************************************** */
+/* ********************* SYSTEM RESTART MANAGER ******************* */
+/* **************************************************************** */
+/***************************************************************/
+/* CHECK RESTART STATE AND SET NEW STATE CALLED IN OPEN, */
+/* READ AND COPY STATES */
+/***************************************************************/
+void Dbtup::execTUP_SRREQ(Signal* signal)
+{
+ RestartInfoRecordPtr riPtr;
+ PendingFileOpenInfoPtr pfoiPtr;
+
+ ljamEntry();
+ Uint32 userPtr = signal->theData[0];
+ Uint32 userBlockref = signal->theData[1];
+ Uint32 tableId = signal->theData[2];
+ Uint32 fragId = signal->theData[3];
+ Uint32 checkpointNumber = signal->theData[4];
+
+ seizeRestartInfoRecord(riPtr);
+
+ riPtr.p->sriUserptr = userPtr;
+ riPtr.p->sriBlockref = userBlockref;
+ riPtr.p->sriState = OPENING_DATA_FILE;
+ riPtr.p->sriCheckpointVersion = checkpointNumber;
+ riPtr.p->sriFragid = fragId;
+ riPtr.p->sriTableId = tableId;
+
+ /* OPEN THE DATA FILE IN THE FOLLOWING FORM */
+ /* D5/DBTUP/T<TABID>/F<FRAGID>/S<CHECKPOINT_NUMBER>.DATA */
+ Uint32 fileType = 1; /* VERSION */
+ fileType = (fileType << 8) | 0; /* .DATA */
+ fileType = (fileType << 8) | 5; /* D5 */
+ fileType = (fileType << 8) | 0xff; /* DON'T USE P DIRECTORY LEVEL */
+ Uint32 fileFlag = 0; /* READ ONLY */
+
+ seizePendingFileOpenInfoRecord(pfoiPtr);
+ pfoiPtr.p->pfoOpenType = LCP_DATA_FILE_READ;
+ pfoiPtr.p->pfoRestartInfoP = riPtr.i;
+
+ signal->theData[0] = cownref;
+ signal->theData[1] = pfoiPtr.i;
+ signal->theData[2] = tableId;
+ signal->theData[3] = fragId;
+ signal->theData[4] = checkpointNumber;
+ signal->theData[5] = fileType;
+ signal->theData[6] = fileFlag;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+ return;
+}//Dbtup::execTUP_SRREQ()
+
+void Dbtup::seizeRestartInfoRecord(RestartInfoRecordPtr& riPtr)
+{
+ riPtr.i = cfirstfreeSri;
+ ptrCheckGuard(riPtr, cnoOfRestartInfoRec, restartInfoRecord);
+ cfirstfreeSri = riPtr.p->sriNextRec;
+ riPtr.p->sriNextRec = RNIL;
+}//Dbtup::seizeRestartInfoRecord()
+
+void Dbtup::rfrReadRestartInfoLab(Signal* signal, RestartInfoRecordPtr riPtr)
+{
+ DiskBufferSegmentInfoPtr dbsiPtr;
+
+ seizeDiskBufferSegmentRecord(dbsiPtr);
+ riPtr.p->sriDataBufferSegmentP = dbsiPtr.i;
+ Uint32 retPageRef;
+ Uint32 noAllocPages = 1;
+ Uint32 noOfPagesAllocated;
+ allocConsPages(noAllocPages, noOfPagesAllocated, retPageRef);
+ ndbrequire(noOfPagesAllocated == 1);
+
+ dbsiPtr.p->pdxDataPage[0] = retPageRef;
+ dbsiPtr.p->pdxNumDataPages = 1;
+ dbsiPtr.p->pdxFilePage = 0;
+ rfrReadNextDataSegment(signal, riPtr, dbsiPtr);
+ dbsiPtr.p->pdxOperation = CHECKPOINT_DATA_READ_PAGE_ZERO;
+}//Dbtup::rfrReadRestartInfoLab()
+
+/***************************************************************/
+/* THE RESTART INFORMATION IS NOW READ INTO THE DATA BUFFER */
+/* USE THE RESTART INFORMATION TO INITIATE THE RESTART RECORD */
+/***************************************************************/
+void
+Dbtup::rfrInitRestartInfoLab(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr)
+{
+ Uint32 TzeroDataPage[64];
+ Uint32 Ti;
+ FragrecordPtr regFragPtr;
+ LocalLogInfoPtr lliPtr;
+ PagePtr pagePtr;
+ RestartInfoRecordPtr riPtr;
+ TablerecPtr regTabPtr;
+
+ riPtr.i = dbsiPtr.p->pdxRestartInfoP;
+ ptrCheckGuard(riPtr, cnoOfRestartInfoRec, restartInfoRecord);
+
+ regTabPtr.i = riPtr.p->sriTableId;
+ ptrCheckGuard(regTabPtr, cnoOfTablerec, tablerec);
+
+ Uint32 fragId = riPtr.p->sriFragid;
+ getFragmentrec(regFragPtr, fragId, regTabPtr.p);
+ riPtr.p->sriFragP = regFragPtr.i;
+
+ /* ----- PAGE ALLOCATION --- */
+ /* ALLOCATE PAGES TO FRAGMENT, INSERT THEM INTO PAGE RANGE TABLE AND */
+ /* ALSO CONVERT THEM INTO EMPTY PAGES AND INSERT THEM INTO THE EMPTY LIST */
+ /* OF THE FRAGMENT. SET ALL LISTS OF FREE PAGES TO RNIL */
+
+ ndbrequire(cfirstfreerange != RNIL);
+ pagePtr.i = dbsiPtr.p->pdxDataPage[0];
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ for (Ti = 0; Ti < 63; Ti++) {
+ /***************************************************************/
+ // Save Important content from Page zero in stack variable so
+ // that we can immediately release page zero.
+ /***************************************************************/
+ TzeroDataPage[Ti] = pagePtr.p->pageWord[Ti];
+ }//for
+ /************************************************************************/
+ /* NOW WE DON'T NEED THE RESTART INFO BUFFER PAGE ANYMORE */
+ /* LETS REMOVE IT AND REUSE THE SEGMENT FOR REAL DATA PAGES */
+ /* REMOVE ONE PAGE ONLY, PAGEP IS ALREADY SET TO THE RESTART INFO PAGE */
+ /************************************************************************/
+ returnCommonArea(pagePtr.i, 1);
+
+ Uint32 undoFileVersion = TzeroDataPage[ZSRI_UNDO_FILE_VER];
+ lliPtr.i = (undoFileVersion << 2) + (regTabPtr.i & 0x3);
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ riPtr.p->sriLocalLogInfoP = lliPtr.i;
+
+ ndbrequire(regFragPtr.p->fragTableId == regTabPtr.i);
+ ndbrequire(regFragPtr.p->fragmentId == fragId);
+
+ regFragPtr.p->fragStatus = SYSTEM_RESTART;
+
+ regFragPtr.p->noCopyPagesAlloc = TzeroDataPage[ZSRI_NO_COPY_PAGES_ALLOC];
+
+ riPtr.p->sriCurDataPageFromBuffer = 0;
+ riPtr.p->sriNumDataPages = TzeroDataPage[ZSRI_NO_OF_FRAG_PAGES_POS];
+
+ ndbrequire(riPtr.p->sriNumDataPages >= regFragPtr.p->noOfPages);
+ const Uint32 pageCount = riPtr.p->sriNumDataPages - regFragPtr.p->noOfPages;
+ if(pageCount > 0){
+ Uint32 noAllocPages = allocFragPages(regFragPtr.p, pageCount);
+ ndbrequire(noAllocPages == pageCount);
+ }//if
+ ndbrequire(getNoOfPages(regFragPtr.p) == riPtr.p->sriNumDataPages);
+
+/***************************************************************/
+// Set the variables on fragment record which might have been
+// affected by allocFragPages.
+/***************************************************************/
+
+ regFragPtr.p->emptyPrimPage = TzeroDataPage[ZSRI_EMPTY_PRIM_PAGE];
+ regFragPtr.p->thFreeFirst = TzeroDataPage[ZSRI_TH_FREE_FIRST];
+ regFragPtr.p->thFreeCopyFirst = TzeroDataPage[ZSRI_TH_FREE_COPY_FIRST];
+
+/***************************************************************/
+/* THE RESTART INFORMATION IS NOW READ INTO THE DATA BUFFER */
+/* USE THE RESTART INFORMATION TO INITIATE THE FRAGMENT */
+/***************************************************************/
+ /**
+ * IF THIS UNDO FILE IS NOT OPEN, IT WILL BE OPENED HERE AND THE EXECUTION
+ * WILL CONTINUE WHEN THE FSOPENCONF IS ENTERED.
+ * IF IT'S ALREADY IN USE THE EXECUTION WILL CONTINUE BY A
+ * CONTINUE B SIGNAL
+ */
+ if (lliPtr.p->lliActiveLcp == 0) {
+ PendingFileOpenInfoPtr pfoiPtr;
+ ljam();
+/***************************************************************/
+/* OPEN THE UNDO FILE FOR READ */
+/* THE FILE HANDLE WILL BE SET IN THE LOCAL_LOG_INFO_REC */
+/* UPON FSOPENCONF */
+/***************************************************************/
+ cnoOfLocalLogInfo++;
+ /* F_LEVEL NOT USED */
+ Uint32 fileType = 1; /* VERSION */
+ fileType = (fileType << 8) | 2; /* .LOCLOG */
+ fileType = (fileType << 8) | 6; /* D6 */
+ fileType = (fileType << 8) | 0xff; /* DON'T USE P DIRECTORY LEVEL */
+ Uint32 fileFlag = 0; /* READ ONLY */
+
+ seizePendingFileOpenInfoRecord(pfoiPtr);
+ pfoiPtr.p->pfoOpenType = LCP_UNDO_FILE_READ;
+ pfoiPtr.p->pfoRestartInfoP = riPtr.i;
+
+ signal->theData[0] = cownref;
+ signal->theData[1] = pfoiPtr.i;
+ signal->theData[2] = lliPtr.i;
+ signal->theData[3] = 0xFFFFFFFF;
+ signal->theData[4] = undoFileVersion;
+ signal->theData[5] = fileType;
+ signal->theData[6] = fileFlag;
+ sendSignal(NDBFS_REF, GSN_FSOPENREQ, signal, 7, JBA);
+
+ lliPtr.p->lliPrevRecordId = 0;
+ lliPtr.p->lliActiveLcp = 1;
+ lliPtr.p->lliNumFragments = 1;
+ } else {
+ ljam();
+ signal->theData[0] = ZCONT_LOAD_DP;
+ signal->theData[1] = riPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ lliPtr.p->lliNumFragments++;
+ }//if
+ /* RETAIN THE HIGH- AND LOWSCORE ID:S OF THE LOGRECORD POSITIONS. WE HAVE TO EXECUTE THE */
+ /* UNDO LOG BETWEEN THE END AND START RECORDS FOR ALL RECORDS THAT INCLUDE FRAGMENTS OF */
+ /* THE RIGHT CHECKPOINT VERSION TO COMPLETE THE OPERATION WE HAVE TO RUN ALL LOGS THAT */
+ /* HAS THE NUMBER OF LCP ELEMENT GREATER THAN 0, I.E. IS INCLUDED. */
+ if (TzeroDataPage[ZSRI_UNDO_LOG_END_REC_ID] > lliPtr.p->lliPrevRecordId) {
+ ljam();
+ lliPtr.p->lliPrevRecordId = TzeroDataPage[ZSRI_UNDO_LOG_END_REC_ID];
+ lliPtr.p->lliEndPageId = TzeroDataPage[ZSRI_UNDO_LOG_END_PAGE_ID];
+ }//if
+ return;
+}//Dbtup::rfrInitRestartInfoLab()
+
+/***************************************************************/
+/* LOAD THE NEXT DATA PAGE SEGMENT INTO MEMORY */
+/***************************************************************/
+void Dbtup::rfrLoadDataPagesLab(Signal* signal, RestartInfoRecordPtr riPtr, DiskBufferSegmentInfoPtr dbsiPtr)
+{
+ FragrecordPtr regFragPtr;
+
+ if (riPtr.p->sriCurDataPageFromBuffer >= riPtr.p->sriNumDataPages) {
+ ljam();
+ rfrCompletedLab(signal, riPtr);
+ return;
+ }//if
+ Uint32 startPage = riPtr.p->sriCurDataPageFromBuffer;
+ Uint32 endPage;
+ if ((startPage + ZDB_SEGMENT_SIZE) < riPtr.p->sriNumDataPages) {
+ ljam();
+ endPage = startPage + ZDB_SEGMENT_SIZE;
+ } else {
+ ljam();
+ endPage = riPtr.p->sriNumDataPages;
+ }//if
+ regFragPtr.i = riPtr.p->sriFragP;
+ ptrCheckGuard(regFragPtr, cnoOfFragrec, fragrecord);
+ ndbrequire((endPage - startPage) <= 16);
+ Uint32 i = 0;
+ for (Uint32 pageId = startPage; pageId < endPage; pageId++) {
+ ljam();
+ dbsiPtr.p->pdxDataPage[i] = getRealpid(regFragPtr.p, pageId);
+ i++;
+ }//for
+ dbsiPtr.p->pdxNumDataPages = endPage - startPage; /* SET THE NUMBER OF DATA PAGES */
+ riPtr.p->sriCurDataPageFromBuffer = endPage;
+ dbsiPtr.p->pdxFilePage = startPage + 1;
+ rfrReadNextDataSegment(signal, riPtr, dbsiPtr);
+ return;
+}//Dbtup::rfrLoadDataPagesLab()
+
+void Dbtup::rfrCompletedLab(Signal* signal, RestartInfoRecordPtr riPtr)
+{
+ PendingFileOpenInfoPtr pfoPtr;
+/* ---------------------------------------------------------------------- */
+/* CLOSE THE DATA FILE BEFORE SENDING TUP_SRCONF */
+/* ---------------------------------------------------------------------- */
+ seizePendingFileOpenInfoRecord(pfoPtr);
+ pfoPtr.p->pfoOpenType = LCP_DATA_FILE_READ;
+ pfoPtr.p->pfoCheckpointInfoP = riPtr.i;
+
+ signal->theData[0] = riPtr.p->sriDataFileHandle;
+ signal->theData[1] = cownref;
+ signal->theData[2] = pfoPtr.i;
+ signal->theData[3] = 0;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+}//Dbtup::rfrCompletedLab()
+
+void Dbtup::rfrClosedDataFileLab(Signal* signal, Uint32 restartIndex)
+{
+ RestartInfoRecordPtr riPtr;
+ DiskBufferSegmentInfoPtr dbsiPtr;
+
+ riPtr.i = restartIndex;
+ ptrCheckGuard(riPtr, cnoOfRestartInfoRec, restartInfoRecord);
+ riPtr.p->sriDataFileHandle = RNIL;
+ dbsiPtr.i = riPtr.p->sriDataBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ releaseDiskBufferSegmentRecord(dbsiPtr);
+ signal->theData[0] = riPtr.p->sriUserptr;
+ signal->theData[1] = riPtr.p->sriFragP;
+ sendSignal(riPtr.p->sriBlockref, GSN_TUP_SRCONF, signal, 2, JBB);
+ releaseRestartInfoRecord(riPtr);
+}//Dbtup::rfrClosedDataFileLab()
+
+/* ---------------------------------------------------------------- */
+/* ---------------------- EXECUTE LOCAL LOG ---------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::execSTART_RECREQ(Signal* signal)
+{
+ ljamEntry();
+ clqhUserpointer = signal->theData[0];
+ clqhBlockref = signal->theData[1];
+
+ for (int i = 0; i < ZNO_CHECKPOINT_RECORDS; i++){
+ cSrUndoRecords[i] = 0;
+ }//for
+
+ if (cnoOfLocalLogInfo == 0) {
+ ljam();
+/* ---------------------------------------------------------------- */
+/* THERE WERE NO LOCAL LOGS TO EXECUTE IN THIS SYSTEM RESTART */
+/* ---------------------------------------------------------------- */
+ xlcRestartCompletedLab(signal);
+ return;
+ }//if
+ LocalLogInfoPtr lliPtr;
+ for (lliPtr.i = 0; lliPtr.i < 16; lliPtr.i++) {
+ ljam();
+ ptrAss(lliPtr, localLogInfo);
+ if (lliPtr.p->lliActiveLcp == 1) {
+ ljam();
+ signal->theData[0] = ZSTART_EXEC_UNDO_LOG;
+ signal->theData[1] = lliPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ }//if
+ }//for
+ return;
+}//Dbtup::execSTART_RECREQ()
+
+void Dbtup::closeExecUndoLogLab(Signal* signal, LocalLogInfoPtr lliPtr)
+{
+ PendingFileOpenInfoPtr pfoPtr;
+/* ---------------------------------------------------------------------- */
+/* CLOSE THE UNDO LOG BEFORE COMPLETION OF THE SYSTEM RESTART */
+/* ---------------------------------------------------------------------- */
+ seizePendingFileOpenInfoRecord(pfoPtr);
+ pfoPtr.p->pfoOpenType = LCP_UNDO_FILE_READ;
+ pfoPtr.p->pfoCheckpointInfoP = lliPtr.i;
+
+ signal->theData[0] = lliPtr.p->lliUndoFileHandle;
+ signal->theData[1] = cownref;
+ signal->theData[2] = pfoPtr.i;
+ signal->theData[3] = 0;
+ sendSignal(NDBFS_REF, GSN_FSCLOSEREQ, signal, 4, JBA);
+ return;
+}//Dbtup::closeExecUndoLogLab()
+
+void Dbtup::endExecUndoLogLab(Signal* signal, Uint32 lliIndex)
+{
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ LocalLogInfoPtr lliPtr;
+
+ lliPtr.i = lliIndex;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+ lliPtr.p->lliUndoFileHandle = RNIL;
+ lliPtr.p->lliActiveLcp = 0;
+/* ---------------------------------------------------------------------- */
+/* WE HAVE NOW CLOSED THE LOG. WE WAIT FOR ALL LOCAL LOGS TO */
+/* COMPLETE LOG EXECUTION BEFORE SENDING THE RESPONSE TO LQH. */
+/* ---------------------------------------------------------------------- */
+ dbsiPtr.i = lliPtr.p->lliUndoBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ freeDiskBufferSegmentRecord(signal, dbsiPtr);
+ lliPtr.p->lliUndoBufferSegmentP = RNIL;
+ for (lliPtr.i = 0; lliPtr.i < 16; lliPtr.i++) {
+ ljam();
+ ptrAss(lliPtr, localLogInfo);
+ if (lliPtr.p->lliActiveLcp == 1) {
+ ljam();
+ return;
+ }//if
+ }//for
+ xlcRestartCompletedLab(signal);
+ return;
+}//Dbtup::endExecUndoLogLab()
+
+void Dbtup::xlcRestartCompletedLab(Signal* signal)
+{
+ cnoOfLocalLogInfo = 0;
+
+ signal->theData[0] = EventReport::UNDORecordsExecuted;
+ signal->theData[1] = DBTUP; // From block
+ signal->theData[2] = 0; // Total records executed
+ for (int i = 0; i < 10; i++) {
+ if (i < ZNO_CHECKPOINT_RECORDS){
+ signal->theData[i+3] = cSrUndoRecords[i];
+ signal->theData[2] += cSrUndoRecords[i];
+ } else {
+ signal->theData[i+3] = 0; // Unsused data
+ }//if
+ }//for
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 12, JBB);
+
+/* ---------------------------------------------------------------------- */
+/* ALL LOCAL LOGS HAVE COMPLETED. WE HAVE COMPLETED OUR PART OF THE */
+/* SYSTEM RESTART. */
+/* ---------------------------------------------------------------------- */
+ signal->theData[0] = clqhUserpointer;
+ sendSignal(clqhBlockref, GSN_START_RECCONF, signal, 1, JBB);
+ return;
+}//Dbtup::xlcRestartCompletedLab()
+
+void Dbtup::startExecUndoLogLab(Signal* signal, Uint32 lliIndex)
+{
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ LocalLogInfoPtr lliPtr;
+
+/* ---------------------------------------------------------------------- */
+/* START EXECUTING THE LOG FOR THIS PART. WE BEGIN BY READING THE */
+/* LAST 16 PAGES. */
+/* ---------------------------------------------------------------------- */
+ /* SET THE PREVIOS RECORD TO THE LAST ONE BECAUSE THAT'S WHERE TO START */
+ lliPtr.i = lliIndex;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+
+ allocRestartUndoBufferSegment(signal, dbsiPtr, lliPtr);
+ lliPtr.p->lliUndoBufferSegmentP = dbsiPtr.i;
+ dbsiPtr.p->pdxCheckpointInfoP = lliPtr.i;
+ if (lliPtr.p->lliEndPageId > ((2 * ZUB_SEGMENT_SIZE) - 1)) {
+ ljam();
+ dbsiPtr.p->pdxNumDataPages = ZUB_SEGMENT_SIZE;
+ dbsiPtr.p->pdxFilePage = lliPtr.p->lliEndPageId - (ZUB_SEGMENT_SIZE - 1);
+ } else if (lliPtr.p->lliEndPageId > (ZUB_SEGMENT_SIZE - 1)) {
+ ljam();
+ dbsiPtr.p->pdxNumDataPages = lliPtr.p->lliEndPageId - (ZUB_SEGMENT_SIZE - 1);
+ dbsiPtr.p->pdxFilePage = ZUB_SEGMENT_SIZE;
+ } else {
+ ljam();
+ dbsiPtr.p->pdxNumDataPages = lliPtr.p->lliEndPageId + 1;
+ dbsiPtr.p->pdxFilePage = 0;
+ rfrReadNextUndoSegment(signal, dbsiPtr, lliPtr);
+ return;
+ }//if
+ rfrReadFirstUndoSegment(signal, dbsiPtr, lliPtr);
+ return;
+}//Dbtup::startExecUndoLogLab()
+
+void Dbtup::rfrReadSecondUndoLogLab(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr)
+{
+ LocalLogInfoPtr lliPtr;
+ lliPtr.i = dbsiPtr.p->pdxCheckpointInfoP;
+ ptrCheckGuard(lliPtr, cnoOfParallellUndoFiles, localLogInfo);
+
+ dbsiPtr.p->pdxNumDataPages = ZUB_SEGMENT_SIZE;
+ dbsiPtr.p->pdxFilePage -= ZUB_SEGMENT_SIZE;
+ rfrReadNextUndoSegment(signal, dbsiPtr, lliPtr);
+ return;
+}//Dbtup::rfrReadSecondUndoLogLab()
+
+void Dbtup::readExecUndoLogLab(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr, LocalLogInfoPtr lliPtr)
+{
+/* ---------------------------------------------------------------------- */
+/* THE NEXT UNDO LOG RECORD HAS NOT BEEN READ FROM DISK YET. WE WILL*/
+/* READ UPTO 8 PAGES BACKWARDS OF THE UNDO LOG FILE. WE WILL KEEP */
+/* THE LAST 8 PAGES TO ENSURE THAT WE WILL BE ABLE TO READ THE NEXT */
+/* LOG RECORD EVEN IF IT SPANS UPTO 8 PAGES. */
+/* ---------------------------------------------------------------------- */
+ if (dbsiPtr.p->pdxFilePage >= ZUB_SEGMENT_SIZE) {
+ ljam();
+ for (Uint32 i = 0; i < ZUB_SEGMENT_SIZE; i++) {
+ ljam();
+ Uint32 savePageId = dbsiPtr.p->pdxDataPage[i + ZUB_SEGMENT_SIZE];
+ dbsiPtr.p->pdxDataPage[i + ZUB_SEGMENT_SIZE] = dbsiPtr.p->pdxDataPage[i];
+ dbsiPtr.p->pdxDataPage[i] = savePageId;
+ }//for
+ dbsiPtr.p->pdxNumDataPages = ZUB_SEGMENT_SIZE;
+ dbsiPtr.p->pdxFilePage = dbsiPtr.p->pdxFilePage - ZUB_SEGMENT_SIZE;
+ } else {
+ ljam();
+ Uint32 dataPages[16];
+ ndbrequire(dbsiPtr.p->pdxFilePage > 0);
+ ndbrequire(dbsiPtr.p->pdxFilePage <= ZUB_SEGMENT_SIZE);
+ for (Uint32 i = 0; i < dbsiPtr.p->pdxFilePage; i++) {
+ ljam();
+ dataPages[i] = dbsiPtr.p->pdxDataPage[i + ZUB_SEGMENT_SIZE];
+ }//for
+ for (Uint32 i = 0; i < ZUB_SEGMENT_SIZE; i++) {
+ ljam();
+ dataPages[i + dbsiPtr.p->pdxFilePage] = dbsiPtr.p->pdxDataPage[i];
+ }//for
+ Uint32 limitLoop = ZUB_SEGMENT_SIZE + dbsiPtr.p->pdxFilePage;
+ for (Uint32 i = 0; i < limitLoop; i++) {
+ ljam();
+ dbsiPtr.p->pdxDataPage[i] = dataPages[i];
+ }//for
+ dbsiPtr.p->pdxNumDataPages = dbsiPtr.p->pdxFilePage;
+ dbsiPtr.p->pdxFilePage = 0;
+ }//if
+ rfrReadNextUndoSegment(signal, dbsiPtr, lliPtr);
+ return;
+}//Dbtup::readExecUndoLogLab()
+
+void Dbtup::rfrReadNextDataSegment(Signal* signal, RestartInfoRecordPtr riPtr, DiskBufferSegmentInfoPtr dbsiPtr)
+{
+ dbsiPtr.p->pdxRestartInfoP = riPtr.i;
+ dbsiPtr.p->pdxOperation = CHECKPOINT_DATA_READ;
+ ndbrequire(dbsiPtr.p->pdxNumDataPages <= 8);
+
+ signal->theData[0] = riPtr.p->sriDataFileHandle;
+ signal->theData[1] = cownref;
+ signal->theData[2] = dbsiPtr.i;
+ signal->theData[3] = 2;
+ signal->theData[4] = ZBASE_ADDR_PAGE_WORD;
+ signal->theData[5] = dbsiPtr.p->pdxNumDataPages;
+ for (Uint32 i = 0; i < dbsiPtr.p->pdxNumDataPages; i++) {
+ ljam();
+ signal->theData[6 + i] = dbsiPtr.p->pdxDataPage[i];
+ }//for
+ signal->theData[6 + dbsiPtr.p->pdxNumDataPages] = dbsiPtr.p->pdxFilePage;
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 15, JBA);
+}//Dbtup::rfrReadNextDataSegment()
+
+/* ---------------------------------------------------------------- */
+/* ------------------- RFR_READ_FIRST_UNDO_SEGMENT ---------------- */
+/* ---------------------------------------------------------------- */
+/* THIS ROUTINE READS IN THE FIRST UNDO SEGMENT INTO THE CURRENTLY */
+/* ACTIVE UNDO BUFFER SEGMENT */
+/* -----------------------------------------------------------------*/
+void Dbtup::rfrReadFirstUndoSegment(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr, LocalLogInfoPtr lliPtr)
+{
+ dbsiPtr.p->pdxOperation = CHECKPOINT_UNDO_READ_FIRST;
+
+ signal->theData[0] = lliPtr.p->lliUndoFileHandle;
+ signal->theData[1] = cownref;
+ signal->theData[2] = dbsiPtr.i;
+ signal->theData[3] = 1;
+ signal->theData[4] = ZBASE_ADDR_UNDO_WORD;
+ signal->theData[5] = dbsiPtr.p->pdxNumDataPages;
+ signal->theData[6] = dbsiPtr.p->pdxDataPage[ZUB_SEGMENT_SIZE];
+ signal->theData[7] = dbsiPtr.p->pdxFilePage;
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 8, JBA);
+}//Dbtup::rfrReadFirstUndoSegment()
+
+/* ---------------------------------------------------------------- */
+/* ------------------- RFR_READ_NEXT_UNDO_SEGMENT ----------------- */
+/* ---------------------------------------------------------------- */
+/* THIS ROUTINE READS IN THE NEXT UNDO SEGMENT INTO THE CURRENTLY */
+/* ACTIVE UNDO BUFFER SEGMENT AND SWITCH TO THE UNACTIVE, READY ONE */
+/* -----------------------------------------------------------------*/
+void Dbtup::rfrReadNextUndoSegment(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr, LocalLogInfoPtr lliPtr)
+{
+ dbsiPtr.p->pdxOperation = CHECKPOINT_UNDO_READ;
+
+ signal->theData[0] = lliPtr.p->lliUndoFileHandle;
+ signal->theData[1] = cownref;
+ signal->theData[2] = dbsiPtr.i;
+ signal->theData[3] = 1;
+ signal->theData[4] = ZBASE_ADDR_UNDO_WORD;
+ signal->theData[5] = dbsiPtr.p->pdxNumDataPages;
+ signal->theData[6] = dbsiPtr.p->pdxDataPage[0];
+ signal->theData[7] = dbsiPtr.p->pdxFilePage;
+ sendSignal(NDBFS_REF, GSN_FSREADREQ, signal, 8, JBA);
+}//Dbtup::rfrReadNextUndoSegment()
+
+void Dbtup::xlcGetNextRecordLab(Signal* signal, DiskBufferSegmentInfoPtr dbsiPtr, LocalLogInfoPtr lliPtr)
+{
+ Uint32 loopCount = 0;
+/* ---------------------------------------------------------------------- */
+/* EXECUTE A NEW SET OF UNDO LOG RECORDS. */
+/* ---------------------------------------------------------------------- */
+ XlcStruct xlcStruct;
+
+ xlcStruct.LliPtr = lliPtr;
+ xlcStruct.DbsiPtr = dbsiPtr;
+
+ do {
+ ljam();
+ loopCount++;
+ if (loopCount == 20) {
+ ljam();
+ signal->theData[0] = ZCONT_EXECUTE_LC;
+ signal->theData[1] = xlcStruct.LliPtr.i;
+ sendSignal(cownref, GSN_CONTINUEB, signal, 2, JBB);
+ return;
+ }//if
+ if (xlcStruct.LliPtr.p->lliPrevRecordId == 0) {
+ ljam();
+ closeExecUndoLogLab(signal, xlcStruct.LliPtr);
+ return;
+ }//if
+ xlcStruct.PageId = xlcStruct.LliPtr.p->lliPrevRecordId >> ZUNDO_RECORD_ID_PAGE_INDEX;
+ xlcStruct.PageIndex = xlcStruct.LliPtr.p->lliPrevRecordId & ZUNDO_RECORD_ID_PAGE_INDEX_MASK;
+ if (xlcStruct.PageId < xlcStruct.DbsiPtr.p->pdxFilePage) {
+ ljam();
+ readExecUndoLogLab(signal, xlcStruct.DbsiPtr, xlcStruct.LliPtr);
+ return;
+ }//if
+ ndbrequire((xlcStruct.PageId - xlcStruct.DbsiPtr.p->pdxFilePage) < 16);
+ xlcStruct.UPPtr.i = xlcStruct.DbsiPtr.p->pdxDataPage[xlcStruct.PageId - xlcStruct.DbsiPtr.p->pdxFilePage];
+ ptrCheckGuard(xlcStruct.UPPtr, cnoOfUndoPage, undoPage);
+ xlcGetLogHeader(xlcStruct);
+ getFragmentrec(xlcStruct.FragPtr, xlcStruct.FragId, xlcStruct.TabPtr.p);
+ if (xlcStruct.FragPtr.i == RNIL) {
+ ljam();
+ continue;
+ }//if
+ if (xlcStruct.FragPtr.p->fragStatus != SYSTEM_RESTART) {
+ ljam();
+ continue;
+ }//if
+ ndbrequire(xlcStruct.LogRecordType < ZNO_CHECKPOINT_RECORDS);
+ cSrUndoRecords[xlcStruct.LogRecordType]++;
+ switch (xlcStruct.LogRecordType) {
+ case ZLCPR_TYPE_INSERT_TH:
+ ljam();
+ xlcInsertTh(xlcStruct);
+ break;
+ case ZLCPR_TYPE_DELETE_TH:
+ ljam();
+ xlcDeleteTh(xlcStruct);
+ break;
+ case ZLCPR_TYPE_UPDATE_TH:
+ ljam();
+ xlcUpdateTh(xlcStruct);
+ break;
+ case ZLCPR_TYPE_INSERT_TH_NO_DATA:
+ ljam();
+ xlcInsertTh(xlcStruct);
+ break;
+ case ZLCPR_ABORT_UPDATE:
+ ljam();
+ xlcAbortUpdate(signal, xlcStruct);
+ break;
+ case ZLCPR_ABORT_INSERT:
+ ljam();
+ xlcAbortInsert(signal, xlcStruct);
+ break;
+ case ZTABLE_DESCRIPTOR:
+ ljam();
+ xlcTableDescriptor(xlcStruct);
+ if (xlcStruct.LliPtr.p->lliNumFragments == 0) {
+ ljam();
+ closeExecUndoLogLab(signal, xlcStruct.LliPtr);
+ return;
+ }//if
+ break;
+ case ZLCPR_UNDO_LOG_PAGE_HEADER:
+ ljam();
+ xlcUndoLogPageHeader(xlcStruct);
+ break;
+ case ZINDICATE_NO_OP_ACTIVE:
+ ljam();
+ xlcIndicateNoOpActive(xlcStruct);
+ break;
+ case ZLCPR_TYPE_UPDATE_GCI:
+ ljam();
+ xlcUpdateGCI(xlcStruct);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+ } while (1);
+}//Dbtup::xlcGetNextRecordLab()
+
+/* ---------------------------------------------------------------- */
+/* ----------------- XLC_GET_LOG_HEADER ---------------------- */
+/* ---------------------------------------------------------------- */
+void Dbtup::xlcGetLogHeader(XlcStruct& xlcStruct)
+{
+ Uint32 pIndex = xlcStruct.PageIndex;
+ Uint32 fragId;
+ Uint32 tableId;
+ Uint32 prevId;
+ if ((pIndex + 4) < ZWORDS_ON_PAGE) {
+ UndoPage* const regUndoPagePtr = xlcStruct.UPPtr.p;
+ ljam();
+ xlcStruct.LogRecordType = regUndoPagePtr->undoPageWord[pIndex];
+ prevId = regUndoPagePtr->undoPageWord[pIndex + 1];
+ tableId = regUndoPagePtr->undoPageWord[pIndex + 2];
+ fragId = regUndoPagePtr->undoPageWord[pIndex + 3];
+ xlcStruct.PageIndex = pIndex + 4;
+ } else {
+ ljam();
+ xlcStruct.LogRecordType = xlcGetLogWord(xlcStruct);
+ prevId = xlcGetLogWord(xlcStruct);
+ tableId = xlcGetLogWord(xlcStruct);
+ fragId = xlcGetLogWord(xlcStruct);
+ }//if
+ xlcStruct.LliPtr.p->lliPrevRecordId = prevId;
+ xlcStruct.FragId = fragId;
+ xlcStruct.TabPtr.i = tableId;
+ ptrCheckGuard(xlcStruct.TabPtr, cnoOfTablerec, tablerec);
+}//Dbtup::xlcGetLogHeader()
+
+/* ------------------------------------------------------------------- */
+/* ---------------------- XLC_GET_LOG_WORD --------------------------- */
+/* ------------------------------------------------------------------- */
+Uint32 Dbtup::xlcGetLogWord(XlcStruct& xlcStruct)
+{
+ Uint32 pIndex = xlcStruct.PageIndex;
+ ndbrequire(xlcStruct.UPPtr.p != NULL);
+ ndbrequire(pIndex < ZWORDS_ON_PAGE);
+ Uint32 logWord = xlcStruct.UPPtr.p->undoPageWord[pIndex];
+ pIndex++;
+ xlcStruct.PageIndex = pIndex;
+ if (pIndex == ZWORDS_ON_PAGE) {
+ ljam();
+ xlcStruct.PageIndex = ZUNDO_PAGE_HEADER_SIZE;
+ xlcStruct.PageId++;
+ if ((xlcStruct.PageId - xlcStruct.DbsiPtr.p->pdxFilePage) >= (2 * ZUB_SEGMENT_SIZE)) {
+ ljam();
+ xlcStruct.UPPtr.i = RNIL;
+ ptrNull(xlcStruct.UPPtr);
+ } else {
+ ljam();
+ Uint32 index = xlcStruct.PageId - xlcStruct.DbsiPtr.p->pdxFilePage;
+ ndbrequire(index < 16);
+ xlcStruct.UPPtr.i = xlcStruct.DbsiPtr.p->pdxDataPage[index];
+ ptrCheckGuard(xlcStruct.UPPtr, cnoOfUndoPage, undoPage);
+ }//if
+ }//if
+ return logWord;
+}//Dbtup::xlcGetLogWord()
+
+ /****************************************************/
+ /* INSERT A TUPLE HEADER THE DATA IS THE TUPLE DATA */
+ /****************************************************/
+void Dbtup::xlcInsertTh(XlcStruct& xlcStruct)
+{
+ PagePtr pagePtr;
+ Fragrecord* const regFragPtr = xlcStruct.FragPtr.p;
+ Tablerec* const regTabPtr = xlcStruct.TabPtr.p;
+
+ Uint32 fragPageId = xlcGetLogWord(xlcStruct);
+ Uint32 pageIndex = xlcGetLogWord(xlcStruct);
+ ndbrequire((pageIndex & 1) == 0);
+ pagePtr.i = getRealpid(regFragPtr, fragPageId);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 pageOffset;
+ getThAtPageSr(pagePtr.p, pageOffset);
+ ndbrequire(pageOffset == (ZPAGE_HEADER_SIZE + (regTabPtr->tupheadsize * (pageIndex >> 1))));
+ if (xlcStruct.LogRecordType == ZLCPR_TYPE_INSERT_TH) {
+ ljam();
+ xlcCopyData(xlcStruct, pageOffset, regTabPtr->tupheadsize, pagePtr);
+ } else {
+ ndbrequire(xlcStruct.LogRecordType == ZLCPR_TYPE_INSERT_TH_NO_DATA);
+ ljam();
+ }//if
+/* ----------------------------------------*/
+/* INDICATE THAT NO OPERATIONS ACTIVE */
+/* ----------------------------------------*/
+ ndbrequire(pageOffset < ZWORDS_ON_PAGE);
+ pagePtr.p->pageWord[pageOffset] = RNIL;
+}//Dbtup::xlcInsertTh()
+
+ /**********************************************/
+ /* DELETE A TUPLE HEADER - NO ADDITIONAL DATA */
+ /**********************************************/
+void Dbtup::xlcDeleteTh(XlcStruct& xlcStruct)
+{
+ PagePtr pagePtr;
+ Fragrecord* const regFragPtr = xlcStruct.FragPtr.p;
+ Tablerec* const regTabPtr = xlcStruct.TabPtr.p;
+
+ Uint32 fragPageId = xlcGetLogWord(xlcStruct);
+ Uint32 pageIndex = xlcGetLogWord(xlcStruct);
+ ndbrequire((pageIndex & 1) == 0);
+ pagePtr.i = getRealpid(regFragPtr, fragPageId);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 pageOffset = ZPAGE_HEADER_SIZE + (regTabPtr->tupheadsize * (pageIndex >> 1));
+ freeThSr(regTabPtr, pagePtr.p, pageOffset);
+}//Dbtup::xlcDeleteTh()
+
+ /*****************************************************/
+ /* UPDATE A TUPLE HEADER, THE DATA IS THE TUPLE DATA */
+ /*****************************************************/
+void Dbtup::xlcUpdateTh(XlcStruct& xlcStruct)
+{
+ PagePtr pagePtr;
+ Fragrecord* const regFragPtr = xlcStruct.FragPtr.p;
+ Tablerec* const regTabPtr = xlcStruct.TabPtr.p;
+
+ Uint32 fragPageId = xlcGetLogWord(xlcStruct);
+ Uint32 pageIndex = xlcGetLogWord(xlcStruct);
+ ndbrequire((pageIndex & 1) == 0);
+ pagePtr.i = getRealpid(regFragPtr, fragPageId);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 pageOffset = ZPAGE_HEADER_SIZE + (regTabPtr->tupheadsize * (pageIndex >> 1));
+ xlcCopyData(xlcStruct, pageOffset, regTabPtr->tupheadsize, pagePtr);
+/* ----------------------------------------*/
+/* INDICATE THAT NO OPERATIONS ACTIVE */
+/* ----------------------------------------*/
+ ndbrequire(pageOffset < ZWORDS_ON_PAGE);
+ pagePtr.p->pageWord[pageOffset] = RNIL;
+}//Dbtup::xlcUpdateTh()
+
+ /**************************************************/
+ /* ABORT AN INSERT OPERATION - NO ADDITIONAL DATA */
+ /**************************************************/
+void Dbtup::xlcAbortInsert(Signal* signal, XlcStruct& xlcStruct)
+{
+ PagePtr pagePtr;
+ Fragrecord* const regFragPtr = xlcStruct.FragPtr.p;
+ Tablerec* const regTabPtr = xlcStruct.TabPtr.p;
+
+ Uint32 fragPageId = xlcGetLogWord(xlcStruct);
+ Uint32 pageIndex = xlcGetLogWord(xlcStruct);
+ ndbrequire((pageIndex & 1) == 0);
+ pagePtr.i = getRealpid(regFragPtr, fragPageId);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 pageOffset = ZPAGE_HEADER_SIZE + (regTabPtr->tupheadsize * (pageIndex >> 1));
+ freeTh(regFragPtr, regTabPtr, signal, pagePtr.p, pageOffset);
+}//Dbtup::xlcAbortInsert()
+
+ /*****************************************************/
+ /* COPY DATA FROM COPY TUPLE TO ORIGINAL TUPLE */
+ /*****************************************************/
+void Dbtup::xlcAbortUpdate(Signal* signal, XlcStruct& xlcStruct)
+{
+ PagePtr pagePtr;
+ Fragrecord* const regFragPtr = xlcStruct.FragPtr.p;
+ Tablerec* const regTabPtr = xlcStruct.TabPtr.p;
+ Uint32 tuple_size = regTabPtr->tupheadsize;
+
+ Uint32 fragPageIdC = xlcGetLogWord(xlcStruct);
+ Uint32 pageIndexC = xlcGetLogWord(xlcStruct);
+ ndbrequire((pageIndexC & 1) == 0);
+ Uint32 TdestPageId = getRealpid(regFragPtr, fragPageIdC);
+ Uint32 TcmDestIndex = ZPAGE_HEADER_SIZE +
+ (tuple_size * (pageIndexC >> 1));
+
+ Uint32 fragPageId = xlcGetLogWord(xlcStruct);
+ Uint32 pageIndex = xlcGetLogWord(xlcStruct);
+ ndbrequire((pageIndex & 1) == 0);
+ Uint32 TsourcePageId = getRealpid(regFragPtr, fragPageId);
+ Uint32 TcmSourceIndex = ZPAGE_HEADER_SIZE +
+ (tuple_size * (pageIndex >> 1));
+ Uint32 end_source = tuple_size + TcmSourceIndex;
+ Uint32 end_dest = tuple_size + TcmDestIndex;
+
+ void* Tdestination = (void*)&page[TdestPageId].pageWord[TcmDestIndex + 1];
+ const void* Tsource =
+ (void*)&page[TsourcePageId].pageWord[TcmSourceIndex + 1];
+
+ ndbrequire(TsourcePageId < cnoOfPage &&
+ TdestPageId < cnoOfPage &&
+ end_source <= ZWORDS_ON_PAGE &&
+ end_dest <= ZWORDS_ON_PAGE);
+ MEMCOPY_NO_WORDS(Tdestination, Tsource, (tuple_size - 1));
+
+ pagePtr.i = TsourcePageId;
+ ptrAss(pagePtr, page);
+ freeTh(regFragPtr, regTabPtr, signal, pagePtr.p, TcmSourceIndex);
+
+ pagePtr.i = TdestPageId;
+ ptrAss(pagePtr, page);
+ pagePtr.p->pageWord[TcmDestIndex] = RNIL;
+}//Dbtup::xlcAbortUpdate()
+
+ /*****************************/
+ /* RESTORE UPDATED GCI VALUE */
+ /*****************************/
+void Dbtup::xlcUpdateGCI(XlcStruct& xlcStruct)
+{
+ PagePtr pagePtr;
+ Fragrecord* const regFragPtr = xlcStruct.FragPtr.p;
+ Tablerec* const regTabPtr = xlcStruct.TabPtr.p;
+
+ Uint32 fragPageId = xlcGetLogWord(xlcStruct);
+ Uint32 pageIndex = xlcGetLogWord(xlcStruct);
+ Uint32 restoredGCI = xlcGetLogWord(xlcStruct);
+
+ ndbrequire((pageIndex & 1) == 0);
+ pagePtr.i = getRealpid(regFragPtr, fragPageId);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 pageOffset = ZPAGE_HEADER_SIZE + (regTabPtr->tupheadsize * (pageIndex >> 1));
+ Uint32 gciOffset = pageOffset + regTabPtr->tupGCPIndex;
+ ndbrequire((gciOffset < ZWORDS_ON_PAGE) &&
+ (regTabPtr->tupGCPIndex < regTabPtr->tupheadsize));
+ pagePtr.p->pageWord[gciOffset] = restoredGCI;
+}//Dbtup::xlcUpdateGCI()
+
+ /*****************************************************/
+ /* READ TABLE DESCRIPTOR FROM UNDO LOG */
+ /*****************************************************/
+void Dbtup::xlcTableDescriptor(XlcStruct& xlcStruct)
+{
+ xlcStruct.LliPtr.p->lliNumFragments--;
+ xlcStruct.FragPtr.p->fragStatus = ACTIVE;
+}//Dbtup::xlcTableDescriptor()
+
+ /********************************************************/
+ /* UPDATE PAGE STATE AND NEXT POINTER IN PAGE */
+ /********************************************************/
+void Dbtup::xlcUndoLogPageHeader(XlcStruct& xlcStruct)
+{
+ Fragrecord* const regFragPtr = xlcStruct.FragPtr.p;
+ PagePtr xlcPagep;
+
+ Uint32 fragPageId = xlcGetLogWord(xlcStruct);
+ xlcPagep.i = getRealpid(regFragPtr, fragPageId);
+ ptrCheckGuard(xlcPagep, cnoOfPage, page);
+ Uint32 logWord = xlcGetLogWord(xlcStruct);
+ ndbrequire(logWord != 0);
+ ndbrequire(logWord <= ZAC_MM_FREE_COPY);
+
+ xlcPagep.p->pageWord[ZPAGE_STATE_POS] = logWord;
+ xlcPagep.p->pageWord[ZPAGE_NEXT_POS] = xlcGetLogWord(xlcStruct);
+}//Dbtup::xlcUndoLogPageHeader()
+
+ /********************************************************/
+ /* INDICATE THAT NO OPERATIONS ACTIVE */
+ /********************************************************/
+void Dbtup::xlcIndicateNoOpActive(XlcStruct& xlcStruct)
+{
+ PagePtr pagePtr;
+ Fragrecord* const regFragPtr = xlcStruct.FragPtr.p;
+ Tablerec* const regTabPtr = xlcStruct.TabPtr.p;
+
+ Uint32 fragPageId = xlcGetLogWord(xlcStruct);
+ Uint32 pageIndex = xlcGetLogWord(xlcStruct);
+ ndbrequire((pageIndex & 1) == 0);
+ pagePtr.i = getRealpid(regFragPtr, fragPageId);
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 pageOffset = ZPAGE_HEADER_SIZE + (regTabPtr->tupheadsize * (pageIndex >> 1));
+/* ----------------------------------------*/
+/* INDICATE THAT NO OPERATIONS ACTIVE */
+/* ----------------------------------------*/
+ ndbrequire(pageOffset < ZWORDS_ON_PAGE);
+ pagePtr.p->pageWord[pageOffset] = RNIL;
+}//Dbtup::xlcIndicateNoOpActive()
+
+ /********************************************************/
+ /* THIS IS THE COMMON ROUTINE TO COPY DATA FROM THE */
+ /* UNDO BUFFER TO THE DATA PAGES. IT USES THE */
+ /* XLC_REQUEST_SEGMENT SUB TO GET MORE DATA WHEN NEEDED */
+ /********************************************************/
+void Dbtup::xlcCopyData(XlcStruct& xlcStruct, Uint32 pageOffset, Uint32 noOfWords, PagePtr pagePtr)
+{
+ ndbrequire((pageOffset + noOfWords - 1) < ZWORDS_ON_PAGE);
+ for (Uint32 i = 1; i < noOfWords; i++) {
+ ljam();
+ pagePtr.p->pageWord[pageOffset + i] = xlcGetLogWord(xlcStruct);
+ }//for
+}//Dbtup::xlcCopyData()
+
+void Dbtup::allocRestartUndoBufferSegment(Signal* signal, DiskBufferSegmentInfoPtr& dbsiPtr, LocalLogInfoPtr lliPtr)
+{
+ UndoPagePtr undoPagePtr;
+
+ ndbrequire(cfirstfreeUndoSeg != RNIL);
+ if (cnoFreeUndoSeg == ZMIN_PAGE_LIMIT_TUP_COMMITREQ) {
+ EXECUTE_DIRECT(DBLQH, GSN_TUP_COM_BLOCK, signal, 1);
+ ljamEntry();
+ }//if
+ cnoFreeUndoSeg--;
+ ndbrequire(cnoFreeUndoSeg >= 0);
+ undoPagePtr.i = cfirstfreeUndoSeg;
+ ptrCheckGuard(undoPagePtr, cnoOfUndoPage, undoPage);
+ cfirstfreeUndoSeg = undoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS];
+ undoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS] = RNIL;
+ seizeDiskBufferSegmentRecord(dbsiPtr);
+ dbsiPtr.p->pdxBuffertype = UNDO_RESTART_PAGES;
+ dbsiPtr.p->pdxUndoBufferSet[0] = undoPagePtr.i;
+ for (Uint32 i = 0; i < ZUB_SEGMENT_SIZE; i++) {
+ dbsiPtr.p->pdxDataPage[i] = undoPagePtr.i + i;
+ }//for
+
+ ndbrequire(cfirstfreeUndoSeg != RNIL);
+ if (cnoFreeUndoSeg == ZMIN_PAGE_LIMIT_TUP_COMMITREQ) {
+ EXECUTE_DIRECT(DBLQH, GSN_TUP_COM_BLOCK, signal, 1);
+ ljamEntry();
+ }//if
+ cnoFreeUndoSeg--;
+ ndbrequire(cnoFreeUndoSeg >= 0);
+ undoPagePtr.i = cfirstfreeUndoSeg;
+ ptrCheckGuard(undoPagePtr, cnoOfUndoPage, undoPage);
+ cfirstfreeUndoSeg = undoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS];
+ undoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS] = RNIL;
+ dbsiPtr.p->pdxUndoBufferSet[1] = undoPagePtr.i;
+// lliPtr.p->lliUndoPage = undoPagePtr.i;
+ for (Uint32 i = ZUB_SEGMENT_SIZE; i < (2 * ZUB_SEGMENT_SIZE); i++) {
+ dbsiPtr.p->pdxDataPage[i] = undoPagePtr.i + (i - ZUB_SEGMENT_SIZE);
+ }//for
+ return;
+}//Dbtup::allocRestartUndoBufferSegment()
+
+
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupTabDesMan.cpp b/ndb/src/kernel/blocks/dbtup/DbtupTabDesMan.cpp
new file mode 100644
index 00000000000..d31ab43f108
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupTabDesMan.cpp
@@ -0,0 +1,191 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+
+#define ljam() { jamLine(22000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(22000 + __LINE__); }
+
+/* **************************************************************** */
+/* *********** TABLE DESCRIPTOR MEMORY MANAGER ******************** */
+/* **************************************************************** */
+/* This module is used to allocate and deallocate table descriptor */
+/* memory attached to fragments (could be allocated per table */
+/* instead. Performs its task by a buddy algorithm. */
+/* **************************************************************** */
+Uint32 Dbtup::allocTabDescr(Uint32 noOfAttributes, Uint32 noOfKeyAttr, Uint32 noOfAttributeGroups)
+{
+ Uint32 reference = RNIL;
+ Uint32 allocSize = (ZTD_SIZE + ZTD_TRAILER_SIZE) + (noOfAttributes * ZAD_SIZE);
+ allocSize += noOfAttributeGroups;
+ allocSize += ((2 * noOfAttributes * sizeOfReadFunction()) + noOfKeyAttr);
+/* ---------------------------------------------------------------- */
+/* ALWAYS ALLOCATE A MULTIPLE OF 16 BYTES */
+/* ---------------------------------------------------------------- */
+ allocSize = (((allocSize - 1) >> 4) + 1) << 4;
+ Uint32 list = nextHigherTwoLog(allocSize - 1); /* CALCULATE WHICH LIST IT BELONGS TO */
+ for (Uint32 i = list; i < 16; i++) {
+ ljam();
+ if (cfreeTdList[i] != RNIL) {
+ ljam();
+ reference = cfreeTdList[i];
+ removeTdArea(reference, i); /* REMOVE THE AREA FROM THE FREELIST */
+ Uint32 retNo = (1 << i) - allocSize; /* CALCULATE THE DIFFERENCE */
+ if (retNo >= ZTD_FREE_SIZE) {
+ ljam();
+ Uint32 retRef = reference + allocSize; /* SET THE RETURN POINTER */
+ retNo = itdaMergeTabDescr(retRef, retNo); /* MERGE WITH POSSIBLE RIGHT NEIGHBOURS */
+ freeTabDescr(retRef, retNo); /* RETURN UNUSED TD SPACE TO THE TD AREA */
+ } else {
+ ljam();
+ allocSize = 1 << i;
+ }//if
+ break;
+ }//if
+ }//for
+ if (reference == RNIL) {
+ ljam();
+ terrorCode = ZMEM_NOTABDESCR_ERROR;
+ return RNIL;
+ } else {
+ ljam();
+ setTabDescrWord((reference + allocSize) - ZTD_TR_TYPE, ZTD_TYPE_NORMAL);
+ setTabDescrWord(reference + ZTD_DATASIZE, allocSize);
+
+ /* INITIALIZE THE TRAILER RECORD WITH TYPE AND SIZE */
+ /* THE TRAILER IS USED TO SIMPLIFY MERGE OF FREE AREAS */
+
+ setTabDescrWord(reference + ZTD_HEADER, ZTD_TYPE_NORMAL);
+ setTabDescrWord((reference + allocSize) - ZTD_TR_SIZE, allocSize);
+ return reference;
+ }//if
+}//Dbtup::allocTabDescr()
+
+void Dbtup::freeTabDescr(Uint32 retRef, Uint32 retNo)
+{
+ while (retNo >= ZTD_FREE_SIZE) {
+ ljam();
+ Uint32 list = nextHigherTwoLog(retNo);
+ list--; /* RETURN TO NEXT LOWER LIST */
+ Uint32 sizeOfChunk = 1 << list;
+ insertTdArea(sizeOfChunk, retRef, list);
+ retRef += sizeOfChunk;
+ retNo -= sizeOfChunk;
+ }//while
+}//Dbtup::freeTabDescr()
+
+Uint32
+Dbtup::getTabDescrWord(Uint32 index)
+{
+ ndbrequire(index < cnoOfTabDescrRec);
+ return tableDescriptor[index].tabDescr;
+}//Dbtup::getTabDescrWord()
+
+void
+Dbtup::setTabDescrWord(Uint32 index, Uint32 word)
+{
+ ndbrequire(index < cnoOfTabDescrRec);
+ tableDescriptor[index].tabDescr = word;
+}//Dbtup::setTabDescrWord()
+
+void Dbtup::insertTdArea(Uint32 sizeOfChunk, Uint32 tabDesRef, Uint32 list)
+{
+ ndbrequire(list < 16);
+ setTabDescrWord(tabDesRef + ZTD_FL_HEADER, ZTD_TYPE_FREE);
+ setTabDescrWord(tabDesRef + ZTD_FL_NEXT, cfreeTdList[list]);
+ if (cfreeTdList[list] != RNIL) {
+ ljam(); /* PREVIOUSLY EMPTY SLOT */
+ setTabDescrWord(cfreeTdList[list] + ZTD_FL_PREV, tabDesRef);
+ }//if
+ cfreeTdList[list] = tabDesRef; /* RELINK THE LIST */
+
+ setTabDescrWord(tabDesRef + ZTD_FL_PREV, RNIL);
+ setTabDescrWord(tabDesRef + ZTD_FL_SIZE, 1 << list);
+ setTabDescrWord((tabDesRef + (1 << list)) - ZTD_TR_TYPE, ZTD_TYPE_FREE);
+ setTabDescrWord((tabDesRef + (1 << list)) - ZTD_TR_SIZE, 1 << list);
+}//Dbtup::insertTdArea()
+
+/* ---------------------------------------------------------------- */
+/* ----------------------- MERGE_TAB_DESCR ------------------------ */
+/* ---------------------------------------------------------------- */
+/* INPUT: TAB_DESCR_PTR POINTING AT THE CURRENT CHUNK */
+/* */
+/* SHORTNAME: MTD */
+/* -----------------------------------------------------------------*/
+Uint32 Dbtup::itdaMergeTabDescr(Uint32 retRef, Uint32 retNo)
+{
+ /* THE SIZE OF THE PART TO MERGE MUST BE OF THE SAME SIZE AS THE INSERTED PART */
+ /* THIS IS TRUE EITHER IF ONE PART HAS THE SAME SIZE OR THE SUM OF BOTH PARTS */
+ /* TOGETHER HAS THE SAME SIZE AS THE PART TO BE INSERTED */
+ /* FIND THE SIZES OF THE PARTS TO THE RIGHT OF THE PART TO BE REINSERTED */
+ while ((retRef + retNo) < cnoOfTabDescrRec) {
+ ljam();
+ Uint32 tabDesRef = retRef + retNo;
+ Uint32 headerWord = getTabDescrWord(tabDesRef + ZTD_FL_HEADER);
+ if (headerWord == ZTD_TYPE_FREE) {
+ ljam();
+ Uint32 sizeOfMergedPart = getTabDescrWord(tabDesRef + ZTD_FL_SIZE);
+
+ retNo += sizeOfMergedPart;
+ Uint32 list = nextHigherTwoLog(sizeOfMergedPart - 1);
+ removeTdArea(tabDesRef, list);
+ } else {
+ ljam();
+ return retNo;
+ }//if
+ }//while
+ ndbrequire((retRef + retNo) == cnoOfTabDescrRec);
+ return retNo;
+}//Dbtup::itdaMergeTabDescr()
+
+/* ---------------------------------------------------------------- */
+/* ------------------------ REMOVE_TD_AREA ------------------------ */
+/* ---------------------------------------------------------------- */
+/* */
+/* THIS ROUTINE REMOVES A TD CHUNK FROM THE POOL OF TD RECORDS */
+/* */
+/* INPUT: TLIST LIST TO USE */
+/* TAB_DESCR_PTR POINTS TO THE CHUNK TO BE REMOVED */
+/* */
+/* SHORTNAME: RMTA */
+/* -----------------------------------------------------------------*/
+void Dbtup::removeTdArea(Uint32 tabDesRef, Uint32 list)
+{
+ ndbrequire(list < 16);
+ Uint32 tabDescrNextPtr = getTabDescrWord(tabDesRef + ZTD_FL_NEXT);
+ Uint32 tabDescrPrevPtr = getTabDescrWord(tabDesRef + ZTD_FL_PREV);
+
+ setTabDescrWord(tabDesRef + ZTD_HEADER, ZTD_TYPE_NORMAL);
+ setTabDescrWord((tabDesRef + (1 << list)) - ZTD_TR_TYPE, ZTD_TYPE_NORMAL);
+
+ if (tabDesRef == cfreeTdList[list]) {
+ ljam();
+ cfreeTdList[list] = tabDescrNextPtr; /* RELINK THE LIST */
+ }//if
+ if (tabDescrNextPtr != RNIL) {
+ ljam();
+ setTabDescrWord(tabDescrNextPtr + ZTD_FL_PREV, tabDescrPrevPtr);
+ }//if
+ if (tabDescrPrevPtr != RNIL) {
+ ljam();
+ setTabDescrWord(tabDescrPrevPtr + ZTD_FL_NEXT, tabDescrNextPtr);
+ }//if
+}//Dbtup::removeTdArea()
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupTrigger.cpp b/ndb/src/kernel/blocks/dbtup/DbtupTrigger.cpp
new file mode 100644
index 00000000000..6fceea31150
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupTrigger.cpp
@@ -0,0 +1,1138 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+#include <AttributeDescriptor.hpp>
+#include "AttributeOffset.hpp"
+#include <AttributeHeader.hpp>
+#include <signaldata/FireTrigOrd.hpp>
+#include <signaldata/CreateTrig.hpp>
+#include <signaldata/TuxMaint.hpp>
+
+#define ljam() { jamLine(7000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(7000 + __LINE__); }
+
+/* **************************************************************** */
+/* ---------------------------------------------------------------- */
+/* ----------------------- TRIGGER HANDLING ----------------------- */
+/* ---------------------------------------------------------------- */
+/* **************************************************************** */
+
+ArrayList<Dbtup::TupTriggerData>*
+Dbtup::findTriggerList(Tablerec* table,
+ TriggerType::Value ttype,
+ TriggerActionTime::Value ttime,
+ TriggerEvent::Value tevent)
+{
+ ArrayList<TupTriggerData>* tlist = NULL;
+ switch (ttype) {
+ case TriggerType::SUBSCRIPTION:
+ case TriggerType::SUBSCRIPTION_BEFORE:
+ switch (tevent) {
+ case TriggerEvent::TE_INSERT:
+ ljam();
+ if (ttime == TriggerActionTime::TA_DETACHED)
+ tlist = &table->subscriptionInsertTriggers;
+ break;
+ case TriggerEvent::TE_UPDATE:
+ ljam();
+ if (ttime == TriggerActionTime::TA_DETACHED)
+ tlist = &table->subscriptionUpdateTriggers;
+ break;
+ case TriggerEvent::TE_DELETE:
+ ljam();
+ if (ttime == TriggerActionTime::TA_DETACHED)
+ tlist = &table->subscriptionDeleteTriggers;
+ break;
+ default:
+ break;
+ }
+ break;
+ case TriggerType::SECONDARY_INDEX:
+ switch (tevent) {
+ case TriggerEvent::TE_INSERT:
+ ljam();
+ if (ttime == TriggerActionTime::TA_AFTER)
+ tlist = &table->afterInsertTriggers;
+ break;
+ case TriggerEvent::TE_UPDATE:
+ ljam();
+ if (ttime == TriggerActionTime::TA_AFTER)
+ tlist = &table->afterUpdateTriggers;
+ break;
+ case TriggerEvent::TE_DELETE:
+ ljam();
+ if (ttime == TriggerActionTime::TA_AFTER)
+ tlist = &table->afterDeleteTriggers;
+ break;
+ default:
+ break;
+ }
+ break;
+ case TriggerType::ORDERED_INDEX:
+ switch (tevent) {
+ case TriggerEvent::TE_CUSTOM:
+ ljam();
+ if (ttime == TriggerActionTime::TA_CUSTOM)
+ tlist = &table->tuxCustomTriggers;
+ break;
+ default:
+ break;
+ }
+ break;
+ case TriggerType::READ_ONLY_CONSTRAINT:
+ switch (tevent) {
+ case TriggerEvent::TE_UPDATE:
+ ljam();
+ if (ttime == TriggerActionTime::TA_AFTER)
+ tlist = &table->constraintUpdateTriggers;
+ break;
+ default:
+ break;
+ }
+ break;
+ default:
+ break;
+ }
+ return tlist;
+}
+
+// Trigger signals
+void
+Dbtup::execCREATE_TRIG_REQ(Signal* signal)
+{
+ ljamEntry();
+ BlockReference senderRef = signal->getSendersBlockRef();
+ const CreateTrigReq reqCopy = *(const CreateTrigReq*)signal->getDataPtr();
+ const CreateTrigReq* const req = &reqCopy;
+
+ // Find table
+ TablerecPtr tabPtr;
+ tabPtr.i = req->getTableId();
+ ptrCheckGuard(tabPtr, cnoOfTablerec, tablerec);
+
+ // Create trigger and associate it with the table
+ if (createTrigger(tabPtr.p, req)) {
+ ljam();
+ // Send conf
+ CreateTrigConf* const conf = (CreateTrigConf*)signal->getDataPtrSend();
+ conf->setUserRef(reference());
+ conf->setConnectionPtr(req->getConnectionPtr());
+ conf->setRequestType(req->getRequestType());
+ conf->setTableId(req->getTableId());
+ conf->setIndexId(req->getIndexId());
+ conf->setTriggerId(req->getTriggerId());
+ conf->setTriggerInfo(req->getTriggerInfo());
+ sendSignal(senderRef, GSN_CREATE_TRIG_CONF,
+ signal, CreateTrigConf::SignalLength, JBB);
+ } else {
+ ljam();
+ // Send ref
+ CreateTrigRef* const ref = (CreateTrigRef*)signal->getDataPtrSend();
+ ref->setUserRef(reference());
+ ref->setConnectionPtr(req->getConnectionPtr());
+ ref->setRequestType(req->getRequestType());
+ ref->setTableId(req->getTableId());
+ ref->setIndexId(req->getIndexId());
+ ref->setTriggerId(req->getTriggerId());
+ ref->setTriggerInfo(req->getTriggerInfo());
+ ref->setErrorCode(CreateTrigRef::TooManyTriggers);
+ sendSignal(senderRef, GSN_CREATE_TRIG_REF,
+ signal, CreateTrigRef::SignalLength, JBB);
+ }
+}//Dbtup::execCREATE_TRIG_REQ()
+
+void
+Dbtup::execDROP_TRIG_REQ(Signal* signal)
+{
+ ljamEntry();
+ BlockReference senderRef = signal->getSendersBlockRef();
+ const DropTrigReq reqCopy = *(const DropTrigReq*)signal->getDataPtr();
+ const DropTrigReq* const req = &reqCopy;
+
+ // Find table
+ TablerecPtr tabPtr;
+ tabPtr.i = req->getTableId();
+ ptrCheckGuard(tabPtr, cnoOfTablerec, tablerec);
+
+ // Drop trigger
+ Uint32 r = dropTrigger(tabPtr.p, req);
+ if (r == 0){
+ // Send conf
+ DropTrigConf* const conf = (DropTrigConf*)signal->getDataPtrSend();
+ conf->setUserRef(senderRef);
+ conf->setConnectionPtr(req->getConnectionPtr());
+ conf->setRequestType(req->getRequestType());
+ conf->setTableId(req->getTableId());
+ conf->setIndexId(req->getIndexId());
+ conf->setTriggerId(req->getTriggerId());
+ sendSignal(senderRef, GSN_DROP_TRIG_CONF,
+ signal, DropTrigConf::SignalLength, JBB);
+ } else {
+ // Send ref
+ DropTrigRef* const ref = (DropTrigRef*)signal->getDataPtrSend();
+ ref->setUserRef(senderRef);
+ ref->setConnectionPtr(req->getConnectionPtr());
+ ref->setRequestType(req->getRequestType());
+ ref->setTableId(req->getTableId());
+ ref->setIndexId(req->getIndexId());
+ ref->setTriggerId(req->getTriggerId());
+ ref->setErrorCode((DropTrigRef::ErrorCode)r);
+ ref->setErrorLine(__LINE__);
+ ref->setErrorNode(refToNode(reference()));
+ sendSignal(senderRef, GSN_DROP_TRIG_REF,
+ signal, DropTrigRef::SignalLength, JBB);
+ }
+}//Dbtup::DROP_TRIG_REQ()
+
+/* ---------------------------------------------------------------- */
+/* ------------------------- createTrigger ------------------------ */
+/* */
+/* Creates a new trigger record by fetching one from the trigger */
+/* pool and associates it with the given table. */
+/* Trigger type can be one of secondary_index, subscription, */
+/* constraint(NYI), foreign_key(NYI), schema_upgrade(NYI), */
+/* api_trigger(NYI) or sql_trigger(NYI). */
+/* Note that this method only checks for total number of allowed */
+/* triggers. Checking the number of allowed triggers per table is */
+/* done by TRIX. */
+/* */
+/* ---------------------------------------------------------------- */
+bool
+Dbtup::createTrigger(Tablerec* table, const CreateTrigReq* req)
+{
+ if (ERROR_INSERTED(4003)) {
+ CLEAR_ERROR_INSERT_VALUE;
+ return false;
+ }
+ TriggerType::Value ttype = req->getTriggerType();
+ TriggerActionTime::Value ttime = req->getTriggerActionTime();
+ TriggerEvent::Value tevent = req->getTriggerEvent();
+
+ ArrayList<TupTriggerData>* tlist = findTriggerList(table, ttype, ttime, tevent);
+ ndbrequire(tlist != NULL);
+
+ TriggerPtr tptr;
+ if (!tlist->seize(tptr))
+ return false;
+
+ // Set trigger id
+ tptr.p->triggerId = req->getTriggerId();
+
+ // ndbout_c("Create TupTrigger %u = %u %u %u %u", tptr.p->triggerId, table, ttype, ttime, tevent);
+
+ // Set index id
+ tptr.p->indexId = req->getIndexId();
+
+ // Set trigger type etc
+ tptr.p->triggerType = ttype;
+ tptr.p->triggerActionTime = ttime;
+ tptr.p->triggerEvent = tevent;
+
+ tptr.p->sendBeforeValues = true;
+ if ((tptr.p->triggerType == TriggerType::SUBSCRIPTION) &&
+ ((tptr.p->triggerEvent == TriggerEvent::TE_UPDATE) ||
+ (tptr.p->triggerEvent == TriggerEvent::TE_DELETE))) {
+ ljam();
+ tptr.p->sendBeforeValues = false;
+ }
+ tptr.p->sendOnlyChangedAttributes = false;
+ if (((tptr.p->triggerType == TriggerType::SUBSCRIPTION) ||
+ (tptr.p->triggerType == TriggerType::SUBSCRIPTION_BEFORE)) &&
+ (tptr.p->triggerEvent == TriggerEvent::TE_UPDATE)) {
+ ljam();
+ tptr.p->sendOnlyChangedAttributes = true;
+ }
+
+ // Set monitor all
+ tptr.p->monitorAllAttributes = req->getMonitorAllAttributes();
+ tptr.p->monitorReplicas = req->getMonitorReplicas();
+ tptr.p->m_receiverBlock = refToBlock(req->getReceiverRef());
+
+ tptr.p->attributeMask.clear();
+ if (tptr.p->monitorAllAttributes) {
+ ljam();
+ for(Uint32 i = 0; i < table->noOfAttr; i++) {
+ if (!primaryKey(table, i)) {
+ ljam();
+ tptr.p->attributeMask.set(i);
+ }
+ }
+ } else {
+ // Set attribute mask
+ ljam();
+ tptr.p->attributeMask = req->getAttributeMask();
+ }
+ return true;
+}//Dbtup::createTrigger()
+
+bool
+Dbtup::primaryKey(Tablerec* const regTabPtr, Uint32 attrId)
+{
+ Uint32 attrDescriptorStart = regTabPtr->tabDescriptor;
+ Uint32 attrDescriptor = getTabDescrWord(attrDescriptorStart + (attrId * ZAD_SIZE));
+ return (bool)AttributeDescriptor::getPrimaryKey(attrDescriptor);
+}//Dbtup::primaryKey()
+
+/* ---------------------------------------------------------------- */
+/* -------------------------- dropTrigger ------------------------- */
+/* */
+/* Deletes a trigger record by disassociating it with the given */
+/* table and returning it to the trigger pool. */
+/* Trigger type can be one of secondary_index, subscription, */
+/* constraint(NYI), foreign_key(NYI), schema_upgrade(NYI), */
+/* api_trigger(NYI) or sql_trigger(NYI). */
+/* */
+/* ---------------------------------------------------------------- */
+Uint32
+Dbtup::dropTrigger(Tablerec* table, const DropTrigReq* req)
+{
+ Uint32 triggerId = req->getTriggerId();
+
+ TriggerType::Value ttype = req->getTriggerType();
+ TriggerActionTime::Value ttime = req->getTriggerActionTime();
+ TriggerEvent::Value tevent = req->getTriggerEvent();
+
+ // ndbout_c("Drop TupTrigger %u = %u %u %u %u", triggerId, table, ttype, ttime, tevent);
+
+ ArrayList<TupTriggerData>* tlist = findTriggerList(table, ttype, ttime, tevent);
+ ndbrequire(tlist != NULL);
+
+ Ptr<TupTriggerData> ptr;
+ for (tlist->first(ptr); !ptr.isNull(); tlist->next(ptr)) {
+ ljam();
+ if (ptr.p->triggerId == triggerId) {
+ ljam();
+ tlist->release(ptr.i);
+ return 0;
+ }
+ }
+ return DropTrigRef::TriggerNotFound;
+}//Dbtup::dropTrigger()
+
+/* ---------------------------------------------------------------- */
+/* -------------- checkImmediateTriggersAfterOp ------------------ */
+/* */
+/* Called after an insert, delete, or update operation takes */
+/* place. Fetches before tuple for deletes and updates and */
+/* after tuple for inserts and updates. */
+/* Executes immediate triggers by sending FIRETRIGORD */
+/* */
+/* ---------------------------------------------------------------- */
+void Dbtup::checkImmediateTriggersAfterInsert(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTablePtr)
+{
+ if(refToBlock(regOperPtr->coordinatorTC) == DBLQH) {
+ return;
+ }
+
+ if ((regOperPtr->primaryReplica) &&
+ (!(regTablePtr->afterInsertTriggers.isEmpty()))) {
+ ljam();
+ fireImmediateTriggers(signal,
+ regTablePtr->afterInsertTriggers,
+ regOperPtr);
+ }//if
+}//Dbtup::checkImmediateTriggersAfterInsert()
+
+void Dbtup::checkImmediateTriggersAfterUpdate(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTablePtr)
+{
+ if(refToBlock(regOperPtr->coordinatorTC) == DBLQH) {
+ return;
+ }
+
+ if ((regOperPtr->primaryReplica) &&
+ (!(regTablePtr->afterUpdateTriggers.isEmpty()))) {
+ ljam();
+ fireImmediateTriggers(signal,
+ regTablePtr->afterUpdateTriggers,
+ regOperPtr);
+ }//if
+ if ((regOperPtr->primaryReplica) &&
+ (!(regTablePtr->constraintUpdateTriggers.isEmpty()))) {
+ ljam();
+ fireImmediateTriggers(signal,
+ regTablePtr->constraintUpdateTriggers,
+ regOperPtr);
+ }//if
+}//Dbtup::checkImmediateTriggersAfterUpdate()
+
+void Dbtup::checkImmediateTriggersAfterDelete(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTablePtr)
+{
+ if(refToBlock(regOperPtr->coordinatorTC) == DBLQH) {
+ return;
+ }
+
+ if ((regOperPtr->primaryReplica) &&
+ (!(regTablePtr->afterDeleteTriggers.isEmpty()))) {
+ ljam();
+ executeTriggers(signal,
+ regTablePtr->afterDeleteTriggers,
+ regOperPtr);
+ }//if
+}//Dbtup::checkImmediateTriggersAfterDelete()
+
+#if 0
+/* ---------------------------------------------------------------- */
+/* --------------------- checkDeferredTriggers -------------------- */
+/* */
+/* Called before commit after an insert, delete, or update */
+/* operation. Fetches before tuple for deletes and updates and */
+/* after tuple for inserts and updates. */
+/* Executes deferred triggers by sending FIRETRIGORD */
+/* */
+/* ---------------------------------------------------------------- */
+void Dbtup::checkDeferredTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTablePtr)
+{
+ ljam();
+ // NYI
+}//Dbtup::checkDeferredTriggers()
+#endif
+
+/* ---------------------------------------------------------------- */
+/* --------------------- checkDetachedTriggers -------------------- */
+/* */
+/* Called at commit after an insert, delete, or update operation. */
+/* Fetches before tuple for deletes and updates and */
+/* after tuple for inserts and updates. */
+/* Executes detached triggers by sending FIRETRIGORD */
+/* */
+/* ---------------------------------------------------------------- */
+void Dbtup::checkDetachedTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTablePtr)
+{
+ switch(regOperPtr->optype) {
+ case(ZINSERT):
+ ljam();
+ if (regTablePtr->subscriptionInsertTriggers.isEmpty()) {
+ // Table has no active triggers monitoring inserts at commit
+ ljam();
+ return;
+ }//if
+
+ // If any fired immediate insert trigger then fetch after tuple
+ fireDetachedTriggers(signal,
+ regTablePtr->subscriptionInsertTriggers,
+ regOperPtr);
+ break;
+ case(ZDELETE):
+ ljam();
+ if (regTablePtr->subscriptionDeleteTriggers.isEmpty()) {
+ // Table has no active triggers monitoring deletes at commit
+ ljam();
+ return;
+ }//if
+
+ // Execute any after delete triggers by sending
+ // FIRETRIGORD with the before tuple
+ executeTriggers(signal,
+ regTablePtr->subscriptionDeleteTriggers,
+ regOperPtr);
+ break;
+ case(ZUPDATE):
+ ljam();
+ if (regTablePtr->subscriptionUpdateTriggers.isEmpty()) {
+ // Table has no active triggers monitoring updates at commit
+ ljam();
+ return;
+ }//if
+
+ // If any fired immediate update trigger then fetch after tuple
+ // and send two FIRETRIGORD one with before tuple and one with after tuple
+ fireDetachedTriggers(signal,
+ regTablePtr->subscriptionUpdateTriggers,
+ regOperPtr);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+}//Dbtup::CheckDetachedTriggers()
+
+void
+Dbtup::fireImmediateTriggers(Signal* signal,
+ ArrayList<TupTriggerData>& triggerList,
+ Operationrec* const regOperPtr)
+{
+ TriggerPtr trigPtr;
+ triggerList.first(trigPtr);
+ while (trigPtr.i != RNIL) {
+ ljam();
+ if (trigPtr.p->monitorAllAttributes ||
+ trigPtr.p->attributeMask.overlaps(regOperPtr->changeMask)) {
+ ljam();
+ executeTrigger(signal,
+ trigPtr.p,
+ regOperPtr);
+ }//if
+ triggerList.next(trigPtr);
+ }//while
+}//Dbtup::fireImmediateTriggers()
+
+#if 0
+void
+Dbtup::fireDeferredTriggers(Signal* signal,
+ ArrayList<TupTriggerData>& triggerList,
+ Operationrec* const regOperPtr)
+{
+ TriggerPtr trigPtr;
+ triggerList.first(trigPtr);
+ while (trigPtr.i != RNIL) {
+ ljam();
+ if (trigPtr.p->monitorAllAttributes ||
+ trigPtr.p->attributeMask.overlaps(regOperPtr->changeMask)) {
+ ljam();
+ executeTrigger(signal,
+ trigPtr,
+ regOperPtr);
+ }//if
+ triggerList.next(trigPtr);
+ }//while
+}//Dbtup::fireDeferredTriggers()
+#endif
+
+void
+Dbtup::fireDetachedTriggers(Signal* signal,
+ ArrayList<TupTriggerData>& triggerList,
+ Operationrec* const regOperPtr)
+{
+ TriggerPtr trigPtr;
+ triggerList.first(trigPtr);
+ while (trigPtr.i != RNIL) {
+ ljam();
+ if ((trigPtr.p->monitorReplicas || regOperPtr->primaryReplica) &&
+ (trigPtr.p->monitorAllAttributes ||
+ trigPtr.p->attributeMask.overlaps(regOperPtr->changeMask))) {
+ ljam();
+ executeTrigger(signal,
+ trigPtr.p,
+ regOperPtr);
+ }//if
+ triggerList.next(trigPtr);
+ }//while
+}//Dbtup::fireDetachedTriggers()
+
+void Dbtup::executeTriggers(Signal* signal,
+ ArrayList<TupTriggerData>& triggerList,
+ Operationrec* regOperPtr)
+{
+ TriggerPtr trigPtr;
+ triggerList.first(trigPtr);
+ while (trigPtr.i != RNIL) {
+ ljam();
+ executeTrigger(signal,
+ trigPtr.p,
+ regOperPtr);
+ triggerList.next(trigPtr);
+
+ }//while
+}//Dbtup::executeTriggers()
+
+void Dbtup::executeTrigger(Signal* signal,
+ TupTriggerData* const trigPtr,
+ Operationrec* const regOperPtr)
+{
+
+ /**
+ * The block below does not work together with GREP.
+ * I have 2 db nodes (2 replicas) -> one node group.
+ * I want to have FIRETRIG_ORD sent to all SumaParticipants,
+ * from all nodes in the node group described above. However,
+ * only one of the nodes in the node group actually sends the
+ * FIRE_TRIG_ORD, and the other node enters this "hack" below.
+ * I don't really know what the code snippet below does, but it
+ * does not work with GREP the way Lars and I want it.
+ * We need to have triggers fired from both the primary and the
+ * backup replica, not only the primary as it is now.
+ *
+ * Note: In Suma, I have changed triggers to be created with
+ * setMonitorReplicas(true).
+ * /Johan
+ *
+ * See RT 709
+ */
+ // XXX quick fix to NR, should fix in LQHKEYREQ instead
+ /*
+ if (refToBlock(regOperPtr->coordinatorTC) == DBLQH) {
+ jam();
+ return;
+ }
+ */
+ BlockReference ref = trigPtr->m_receiverBlock;
+ Uint32* const keyBuffer = &cinBuffer[0];
+ Uint32* const mainBuffer = &coutBuffer[0];
+ Uint32* const copyBuffer = &clogMemBuffer[0];
+
+ Uint32 noPrimKey, noMainWords, noCopyWords;
+
+ if (ref == BACKUP) {
+ ljam();
+ /*
+ In order for the implementation of BACKUP to work even when changing
+ primaries in the middle of the backup we need to set the trigger on
+ all replicas. This check checks whether this is the node where this
+ trigger should be fired. The check should preferably have been put
+ completely in the BACKUP block but it was about five times simpler
+ to put it here and also much faster for the backup (small overhead
+ for everybody else.
+ */
+ signal->theData[0] = trigPtr->triggerId;
+ signal->theData[1] = regOperPtr->fragId;
+ EXECUTE_DIRECT(BACKUP, GSN_BACKUP_TRIG_REQ, signal, 2);
+ ljamEntry();
+ if (signal->theData[0] == 0) {
+ ljam();
+ return;
+ }//if
+ }//if
+ if (!readTriggerInfo(trigPtr,
+ regOperPtr,
+ keyBuffer,
+ noPrimKey,
+ mainBuffer,
+ noMainWords,
+ copyBuffer,
+ noCopyWords)) {
+ ljam();
+ return;
+ }//if
+//--------------------------------------------------------------------
+// Now all data for this trigger has been read. It is now time to send
+// the trigger information consisting of two or three sets of TRIG_
+// ATTRINFO signals and one FIRE_TRIG_ORD signal.
+// We start by setting common header info for all TRIG_ATTRINFO signals.
+//--------------------------------------------------------------------
+ bool executeDirect;
+ TrigAttrInfo* const trigAttrInfo = (TrigAttrInfo *)signal->getDataPtrSend();
+ trigAttrInfo->setConnectionPtr(regOperPtr->tcOpIndex);
+ trigAttrInfo->setTriggerId(trigPtr->triggerId);
+
+ switch(trigPtr->triggerType) {
+ case (TriggerType::SECONDARY_INDEX):
+ ljam();
+ ref = regOperPtr->coordinatorTC;
+ executeDirect = false;
+ break;
+ case (TriggerType::SUBSCRIPTION):
+ case (TriggerType::SUBSCRIPTION_BEFORE):
+ ljam();
+ // Since only backup uses subscription triggers we send to backup directly for now
+ ref = trigPtr->m_receiverBlock;
+ executeDirect = true;
+ break;
+ case (TriggerType::READ_ONLY_CONSTRAINT):
+ terrorCode = ZREAD_ONLY_CONSTRAINT_VIOLATION;
+ // XXX should return status and abort the rest
+ return;
+ default:
+ ndbrequire(false);
+ }//switch
+
+ regOperPtr->noFiredTriggers++;
+
+ trigAttrInfo->setAttrInfoType(TrigAttrInfo::PRIMARY_KEY);
+ sendTrigAttrInfo(signal, keyBuffer, noPrimKey, executeDirect, ref);
+
+ Uint32 noAfter = 0;
+ Uint32 noBefore = 0;
+ switch(regOperPtr->optype) {
+ case(ZINSERT):
+ ljam();
+ // Send AttrInfo signals with new attribute values
+ trigAttrInfo->setAttrInfoType(TrigAttrInfo::AFTER_VALUES);
+ sendTrigAttrInfo(signal, mainBuffer, noMainWords, executeDirect, ref);
+ noAfter = noMainWords;
+ break;
+ case(ZDELETE):
+ if (trigPtr->sendBeforeValues) {
+ ljam();
+ trigAttrInfo->setAttrInfoType(TrigAttrInfo::BEFORE_VALUES);
+ sendTrigAttrInfo(signal, mainBuffer, noMainWords, executeDirect, ref);
+ noBefore = noMainWords;
+ }//if
+ break;
+ case(ZUPDATE):
+ ljam();
+ if (trigPtr->sendBeforeValues) {
+ ljam();
+ trigAttrInfo->setAttrInfoType(TrigAttrInfo::BEFORE_VALUES);
+ sendTrigAttrInfo(signal, copyBuffer, noCopyWords, executeDirect, ref);
+ noBefore = noCopyWords;
+ }//if
+ trigAttrInfo->setAttrInfoType(TrigAttrInfo::AFTER_VALUES);
+ sendTrigAttrInfo(signal, mainBuffer, noMainWords, executeDirect, ref);
+ noAfter = noMainWords;
+ break;
+ default:
+ ndbrequire(false);
+ }//switch
+ sendFireTrigOrd(signal,
+ regOperPtr,
+ trigPtr,
+ noPrimKey,
+ noBefore,
+ noAfter);
+}//Dbtup::executeTrigger()
+
+Uint32 Dbtup::setAttrIds(Bitmask<MAXNROFATTRIBUTESINWORDS>& attributeMask,
+ Uint32 noOfAttributes,
+ Uint32* inBuffer)
+{
+ Uint32 bufIndx = 0;
+ for (Uint32 i = 0; i < noOfAttributes; i++) {
+ ljam();
+ if (attributeMask.get(i)) {
+ ljam();
+ AttributeHeader::init(&inBuffer[bufIndx++], i, 0);
+ }//if
+ }//for
+ return bufIndx;
+}//Dbtup::setAttrIds()
+
+bool Dbtup::readTriggerInfo(TupTriggerData* const trigPtr,
+ Operationrec* const regOperPtr,
+ Uint32* const keyBuffer,
+ Uint32& noPrimKey,
+ Uint32* const mainBuffer,
+ Uint32& noMainWords,
+ Uint32* const copyBuffer,
+ Uint32& noCopyWords)
+{
+ noCopyWords = 0;
+ noMainWords = 0;
+ Uint32 readBuffer[MAX_ATTRIBUTES_IN_TABLE];
+ PagePtr pagep;
+
+//---------------------------------------------------------------------------
+// Set-up variables needed by readAttributes operPtr.p, tabptr.p
+//---------------------------------------------------------------------------
+ operPtr.p = regOperPtr;
+ tabptr.i = regOperPtr->tableRef;
+ ptrCheckGuard(tabptr, cnoOfTablerec, tablerec);
+ Tablerec* const regTabPtr = tabptr.p;
+//--------------------------------------------------------------------
+// Initialise pagep and tuple offset for read of main tuple
+//--------------------------------------------------------------------
+ Uint32 tupheadoffset = regOperPtr->pageOffset;
+ pagep.i = regOperPtr->realPageId;
+ ptrCheckGuard(pagep, cnoOfPage, page);
+
+//--------------------------------------------------------------------
+// Read Primary Key Values
+//--------------------------------------------------------------------
+ noPrimKey = readAttributes(pagep.p,
+ tupheadoffset,
+ &tableDescriptor[regTabPtr->readKeyArray].tabDescr,
+ regTabPtr->noOfKeyAttr,
+ keyBuffer,
+ ZATTR_BUFFER_SIZE);
+ ndbrequire(noPrimKey != (Uint32)-1);
+
+ Uint32 numAttrsToRead;
+ if ((regOperPtr->optype == ZUPDATE) &&
+ (trigPtr->sendOnlyChangedAttributes)) {
+ ljam();
+//--------------------------------------------------------------------
+// Update that sends only changed information
+//--------------------------------------------------------------------
+ Bitmask<MAXNROFATTRIBUTESINWORDS> attributeMask;
+ attributeMask = trigPtr->attributeMask;
+ attributeMask.bitAND(regOperPtr->changeMask);
+ numAttrsToRead = setAttrIds(attributeMask, regTabPtr->noOfAttr, &readBuffer[0]);
+
+ } else if ((regOperPtr->optype == ZDELETE) &&
+ (!trigPtr->sendBeforeValues)) {
+ ljam();
+//--------------------------------------------------------------------
+// Delete without sending before values only read Primary Key
+//--------------------------------------------------------------------
+ return true;
+ } else {
+ ljam();
+//--------------------------------------------------------------------
+// All others send all attributes that are monitored
+//--------------------------------------------------------------------
+ numAttrsToRead = setAttrIds(trigPtr->attributeMask, regTabPtr->noOfAttr, &readBuffer[0]);
+ }//if
+ ndbrequire(numAttrsToRead < MAX_ATTRIBUTES_IN_TABLE);
+//--------------------------------------------------------------------
+// Read Main tuple values
+//--------------------------------------------------------------------
+ if ((regOperPtr->optype != ZDELETE) ||
+ (trigPtr->sendBeforeValues)) {
+ ljam();
+ noMainWords = readAttributes(pagep.p,
+ tupheadoffset,
+ &readBuffer[0],
+ numAttrsToRead,
+ mainBuffer,
+ ZATTR_BUFFER_SIZE);
+ ndbrequire(noMainWords != (Uint32)-1);
+ } else {
+ ljam();
+ noMainWords = 0;
+ }//if
+//--------------------------------------------------------------------
+// Read Copy tuple values for UPDATE's
+//--------------------------------------------------------------------
+// Initialise pagep and tuple offset for read of copy tuple
+//--------------------------------------------------------------------
+ if ((regOperPtr->optype == ZUPDATE) &&
+ (trigPtr->sendBeforeValues)) {
+ ljam();
+
+ tupheadoffset = regOperPtr->pageOffsetC;
+ pagep.i = regOperPtr->realPageIdC;
+ ptrCheckGuard(pagep, cnoOfPage, page);
+
+ noCopyWords = readAttributes(pagep.p,
+ tupheadoffset,
+ &readBuffer[0],
+ numAttrsToRead,
+ copyBuffer,
+ ZATTR_BUFFER_SIZE);
+
+ ndbrequire(noCopyWords != (Uint32)-1);
+ if ((noMainWords == noCopyWords) &&
+ (memcmp(mainBuffer, copyBuffer, noMainWords << 2) == 0)) {
+//--------------------------------------------------------------------
+// Although a trigger was fired it was not necessary since the old
+// value and the new value was exactly the same
+//--------------------------------------------------------------------
+ ljam();
+ return false;
+ }//if
+ }//if
+ return true;
+}//Dbtup::readTriggerInfo()
+
+void Dbtup::sendTrigAttrInfo(Signal* signal,
+ Uint32* data,
+ Uint32 dataLen,
+ bool executeDirect,
+ BlockReference receiverReference)
+{
+ TrigAttrInfo* const trigAttrInfo = (TrigAttrInfo *)signal->getDataPtrSend();
+ Uint32 sigLen;
+ Uint32 dataIndex = 0;
+ do {
+ sigLen = dataLen - dataIndex;
+ if (sigLen > TrigAttrInfo::DataLength) {
+ ljam();
+ sigLen = TrigAttrInfo::DataLength;
+ }//if
+ MEMCOPY_NO_WORDS(trigAttrInfo->getData(),
+ data + dataIndex,
+ sigLen);
+ if (executeDirect) {
+ ljam();
+ EXECUTE_DIRECT(receiverReference,
+ GSN_TRIG_ATTRINFO,
+ signal,
+ TrigAttrInfo::StaticLength + sigLen);
+ ljamEntry();
+ } else {
+ ljam();
+ sendSignal(receiverReference,
+ GSN_TRIG_ATTRINFO,
+ signal,
+ TrigAttrInfo::StaticLength + sigLen,
+ JBB);
+ }//if
+ dataIndex += sigLen;
+ } while (dataLen != dataIndex);
+}//Dbtup::sendTrigAttrInfo()
+
+void Dbtup::sendFireTrigOrd(Signal* signal,
+ Operationrec * const regOperPtr,
+ TupTriggerData* const trigPtr,
+ Uint32 noPrimKeyWords,
+ Uint32 noBeforeValueWords,
+ Uint32 noAfterValueWords)
+{
+ FireTrigOrd* const fireTrigOrd = (FireTrigOrd *)signal->getDataPtrSend();
+
+ fireTrigOrd->setConnectionPtr(regOperPtr->tcOpIndex);
+ fireTrigOrd->setTriggerId(trigPtr->triggerId);
+
+ switch(regOperPtr->optype) {
+ case(ZINSERT):
+ ljam();
+ fireTrigOrd->setTriggerEvent(TriggerEvent::TE_INSERT);
+ break;
+ case(ZDELETE):
+ ljam();
+ fireTrigOrd->setTriggerEvent(TriggerEvent::TE_DELETE);
+ break;
+ case(ZUPDATE):
+ ljam();
+ fireTrigOrd->setTriggerEvent(TriggerEvent::TE_UPDATE);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+
+ fireTrigOrd->setNoOfPrimaryKeyWords(noPrimKeyWords);
+ fireTrigOrd->setNoOfBeforeValueWords(noBeforeValueWords);
+ fireTrigOrd->setNoOfAfterValueWords(noAfterValueWords);
+
+ switch(trigPtr->triggerType) {
+ case (TriggerType::SECONDARY_INDEX):
+ ljam();
+ sendSignal(regOperPtr->coordinatorTC, GSN_FIRE_TRIG_ORD,
+ signal, FireTrigOrd::SignalLength, JBB);
+ break;
+ case (TriggerType::SUBSCRIPTION_BEFORE): // Only Suma
+ ljam();
+ // Since only backup uses subscription triggers we
+ // send to backup directly for now
+ fireTrigOrd->setGCI(regOperPtr->gci);
+ fireTrigOrd->setHashValue(regOperPtr->hashValue);
+ EXECUTE_DIRECT(trigPtr->m_receiverBlock,
+ GSN_FIRE_TRIG_ORD,
+ signal,
+ FireTrigOrd::SignalWithHashValueLength);
+ break;
+ case (TriggerType::SUBSCRIPTION):
+ ljam();
+ // Since only backup uses subscription triggers we
+ // send to backup directly for now
+ fireTrigOrd->setGCI(regOperPtr->gci);
+ EXECUTE_DIRECT(trigPtr->m_receiverBlock,
+ GSN_FIRE_TRIG_ORD,
+ signal,
+ FireTrigOrd::SignalWithGCILength);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }//switch
+}//Dbtup::sendFireTrigOrd()
+
+/*
+ * Ordered index triggers.
+ *
+ * Insert: add entry to index
+ * Update: add entry to index, de|ay remove until commit
+ * Delete: do nothing, delay remove until commit
+ * Commit: remove entry delayed from update and delete
+ * Abort : remove entry added by insert and update
+ *
+ * See Notes.txt for the details.
+ */
+
+int
+Dbtup::executeTuxInsertTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr)
+{
+ TuxMaintReq* const req = (TuxMaintReq*)signal->getDataPtrSend();
+ PagePtr pagePtr;
+ pagePtr.i = regOperPtr->realPageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 tupVersion = pagePtr.p->pageWord[regOperPtr->pageOffset + 1];
+ ndbrequire(tupVersion == regOperPtr->tupVersion);
+ // fill in constant part
+ req->tableId = regOperPtr->tableRef;
+ req->fragId = regOperPtr->fragId;
+ req->tupAddr = (regOperPtr->fragPageId << MAX_TUPLES_BITS) | regOperPtr->pageIndex;
+ req->tupVersion = tupVersion;
+ req->opInfo = TuxMaintReq::OpAdd;
+ // loop over index list
+ const ArrayList<TupTriggerData>& triggerList = regTabPtr->tuxCustomTriggers;
+ TriggerPtr triggerPtr;
+ triggerList.first(triggerPtr);
+ while (triggerPtr.i != RNIL) {
+ ljam();
+ req->indexId = triggerPtr.p->indexId;
+ req->errorCode = RNIL;
+ EXECUTE_DIRECT(DBTUX, GSN_TUX_MAINT_REQ,
+ signal, TuxMaintReq::SignalLength);
+ ljamEntry();
+ if (req->errorCode != 0) {
+ ljam();
+ terrorCode = req->errorCode;
+ return -1;
+ }
+ triggerList.next(triggerPtr);
+ }
+ return 0;
+}
+
+int
+Dbtup::executeTuxUpdateTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr)
+{
+ TuxMaintReq* const req = (TuxMaintReq*)signal->getDataPtrSend();
+ PagePtr pagePtr;
+ pagePtr.i = regOperPtr->realPageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ Uint32 tupVersion = pagePtr.p->pageWord[regOperPtr->pageOffset + 1];
+ ndbrequire(tupVersion == regOperPtr->tupVersion);
+ // fill in constant part
+ req->tableId = regOperPtr->tableRef;
+ req->fragId = regOperPtr->fragId;
+ req->tupAddr = (regOperPtr->fragPageId << MAX_TUPLES_BITS) | regOperPtr->pageIndex;
+ req->tupVersion = tupVersion;
+ req->opInfo = TuxMaintReq::OpAdd;
+ // loop over index list
+ const ArrayList<TupTriggerData>& triggerList = regTabPtr->tuxCustomTriggers;
+ TriggerPtr triggerPtr;
+ triggerList.first(triggerPtr);
+ while (triggerPtr.i != RNIL) {
+ ljam();
+ req->tupAddr = (regOperPtr->fragPageId << MAX_TUPLES_BITS) | regOperPtr->pageIndex;
+ req->indexId = triggerPtr.p->indexId;
+ req->errorCode = RNIL;
+ EXECUTE_DIRECT(DBTUX, GSN_TUX_MAINT_REQ,
+ signal, TuxMaintReq::SignalLength);
+ ljamEntry();
+ if (req->errorCode != 0) {
+ ljam();
+ terrorCode = req->errorCode;
+ return -1;
+ }
+ triggerList.next(triggerPtr);
+ }
+ return 0;
+}
+
+int
+Dbtup::executeTuxDeleteTriggers(Signal* signal,
+ Operationrec* const regOperPtr,
+ Tablerec* const regTabPtr)
+{
+ // do nothing
+ return 0;
+}
+
+void
+Dbtup::executeTuxCommitTriggers(Signal* signal,
+ Operationrec* regOperPtr,
+ Tablerec* const regTabPtr)
+{
+ TuxMaintReq* const req = (TuxMaintReq*)signal->getDataPtrSend();
+ // get version
+ // XXX could add prevTupVersion to Operationrec
+ Uint32 tupVersion;
+ if (regOperPtr->optype == ZINSERT) {
+ if (! regOperPtr->deleteInsertFlag)
+ return;
+ ljam();
+ PagePtr pagePtr;
+ pagePtr.i = regOperPtr->realPageIdC;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ tupVersion = pagePtr.p->pageWord[regOperPtr->pageOffsetC + 1];
+ ndbrequire(tupVersion != regOperPtr->tupVersion);
+ } else if (regOperPtr->optype == ZUPDATE) {
+ ljam();
+ PagePtr pagePtr;
+ pagePtr.i = regOperPtr->realPageIdC;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ tupVersion = pagePtr.p->pageWord[regOperPtr->pageOffsetC + 1];
+ ndbrequire(tupVersion != regOperPtr->tupVersion);
+ } else if (regOperPtr->optype == ZDELETE) {
+ if (regOperPtr->deleteInsertFlag)
+ return;
+ ljam();
+ PagePtr pagePtr;
+ pagePtr.i = regOperPtr->realPageId;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ tupVersion = pagePtr.p->pageWord[regOperPtr->pageOffset + 1];
+ ndbrequire(tupVersion == regOperPtr->tupVersion);
+ } else {
+ ndbrequire(false);
+ }
+ // fill in constant part
+ req->tableId = regOperPtr->tableRef;
+ req->fragId = regOperPtr->fragId;
+ req->tupAddr = (regOperPtr->fragPageId << MAX_TUPLES_BITS) | regOperPtr->pageIndex;
+ req->tupVersion = tupVersion;
+ req->opInfo = TuxMaintReq::OpRemove;
+ // loop over index list
+ const ArrayList<TupTriggerData>& triggerList = regTabPtr->tuxCustomTriggers;
+ TriggerPtr triggerPtr;
+ triggerList.first(triggerPtr);
+ while (triggerPtr.i != RNIL) {
+ ljam();
+ req->indexId = triggerPtr.p->indexId;
+ req->errorCode = RNIL;
+ EXECUTE_DIRECT(DBTUX, GSN_TUX_MAINT_REQ,
+ signal, TuxMaintReq::SignalLength);
+ ljamEntry();
+ // commit must succeed
+ ndbrequire(req->errorCode == 0);
+ triggerList.next(triggerPtr);
+ }
+}
+
+void
+Dbtup::executeTuxAbortTriggers(Signal* signal,
+ Operationrec* regOperPtr,
+ Tablerec* const regTabPtr)
+{
+ TuxMaintReq* const req = (TuxMaintReq*)signal->getDataPtrSend();
+ // get version
+ Uint32 tupVersion;
+ if (regOperPtr->optype == ZINSERT) {
+ ljam();
+ tupVersion = regOperPtr->tupVersion;
+ } else if (regOperPtr->optype == ZUPDATE) {
+ ljam();
+ tupVersion = regOperPtr->tupVersion;
+ } else if (regOperPtr->optype == ZDELETE) {
+ ljam();
+ return;
+ } else {
+ ndbrequire(false);
+ }
+ // fill in constant part
+ req->tableId = regOperPtr->tableRef;
+ req->fragId = regOperPtr->fragId;
+ req->tupAddr = (regOperPtr->fragPageId << MAX_TUPLES_BITS) | regOperPtr->pageIndex;
+ req->tupVersion = tupVersion;
+ req->opInfo = TuxMaintReq::OpRemove;
+ // loop over index list
+ const ArrayList<TupTriggerData>& triggerList = regTabPtr->tuxCustomTriggers;
+ TriggerPtr triggerPtr;
+ triggerList.first(triggerPtr);
+ while (triggerPtr.i != RNIL) {
+ ljam();
+ req->indexId = triggerPtr.p->indexId;
+ req->errorCode = RNIL,
+ EXECUTE_DIRECT(DBTUX, GSN_TUX_MAINT_REQ,
+ signal, TuxMaintReq::SignalLength);
+ ljamEntry();
+ // abort must succeed
+ ndbrequire(req->errorCode == 0);
+ triggerList.next(triggerPtr);
+ }
+}
diff --git a/ndb/src/kernel/blocks/dbtup/DbtupUndoLog.cpp b/ndb/src/kernel/blocks/dbtup/DbtupUndoLog.cpp
new file mode 100644
index 00000000000..869f399583f
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/DbtupUndoLog.cpp
@@ -0,0 +1,284 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define DBTUP_C
+#include "Dbtup.hpp"
+#include <RefConvert.hpp>
+#include <ndb_limits.h>
+#include <pc.hpp>
+
+#define ljam() { jamLine(12000 + __LINE__); }
+#define ljamEntry() { jamEntryLine(12000 + __LINE__); }
+
+void Dbtup::cprAddData(Signal* signal,
+ Fragrecord* const regFragPtr,
+ Uint32 pageIndex,
+ Uint32 noOfWords,
+ Uint32 startOffset)
+{
+ UndoPagePtr undoPagePtr;
+ PagePtr pagePtr;
+ LocalLogInfoPtr regLliPtr;
+
+ regLliPtr.i = regFragPtr->checkpointVersion;
+ ptrCheckGuard(regLliPtr, cnoOfParallellUndoFiles, localLogInfo);
+
+ pagePtr.i = pageIndex;
+ ptrCheckGuard(pagePtr, cnoOfPage, page);
+ undoPagePtr.i = regLliPtr.p->lliUndoPage;
+ ptrCheckGuard(undoPagePtr, cnoOfUndoPage, undoPage);
+
+ startOffset++;
+ noOfWords--;
+ if ((regLliPtr.p->lliUndoWord + noOfWords) < ZWORDS_ON_PAGE) {
+ ljam();
+ MEMCOPY_NO_WORDS(&undoPagePtr.p->undoPageWord[regLliPtr.p->lliUndoWord],
+ &pagePtr.p->pageWord[startOffset],
+ noOfWords);
+ regLliPtr.p->lliUndoWord += noOfWords;
+ } else {
+ for (Uint32 i = 0; i < noOfWords; i++) {
+ ljam();
+ Uint32 undoWord = pagePtr.p->pageWord[startOffset + i];
+ cprAddUndoLogWord(signal, regLliPtr.p, undoWord);
+ }//for
+ }//if
+}//Dbtup::cprAddData()
+
+void Dbtup::cprAddLogHeader(Signal* signal,
+ LocalLogInfo* const lliPtr,
+ Uint32 recordType,
+ Uint32 tableId,
+ Uint32 fragId)
+{
+ Uint32 prevRecId = lliPtr->lliPrevRecordId;
+ lliPtr->lliPrevRecordId = lliPtr->lliUndoWord + (lliPtr->lliLogFilePage << ZUNDO_RECORD_ID_PAGE_INDEX);
+ cprAddUndoLogWord(signal, lliPtr, recordType);
+ cprAddUndoLogWord(signal, lliPtr, prevRecId);
+ cprAddUndoLogWord(signal, lliPtr, tableId);
+ cprAddUndoLogWord(signal, lliPtr, fragId);
+}//Dbtup::cprAddLogHeader()
+
+void Dbtup::cprAddGCIUpdate(Signal* signal,
+ Uint32 prevGCI,
+ Fragrecord* const regFragPtr)
+{
+ LocalLogInfoPtr regLliPtr;
+ regLliPtr.i = regFragPtr->checkpointVersion;
+ ptrCheckGuard(regLliPtr, cnoOfParallellUndoFiles, localLogInfo);
+
+ cprAddUndoLogWord(signal, regLliPtr.p, prevGCI);
+}//Dbtup::cprAddLogHeader()
+
+void Dbtup::cprAddUndoLogPageHeader(Signal* signal,
+ Page* const regPagePtr,
+ Fragrecord* const regFragPtr)
+{
+ UndoPagePtr regUndoPagePtr;
+ LocalLogInfoPtr regLliPtr;
+
+ regLliPtr.i = regFragPtr->checkpointVersion;
+ ptrCheckGuard(regLliPtr, cnoOfParallellUndoFiles, localLogInfo);
+
+ Uint32 prevRecId = regLliPtr.p->lliPrevRecordId;
+ Uint32 lliWord = regLliPtr.p->lliUndoWord;
+ regLliPtr.p->lliPrevRecordId = lliWord +
+ (regLliPtr.p->lliLogFilePage << ZUNDO_RECORD_ID_PAGE_INDEX);
+ if ((lliWord + 7) < ZWORDS_ON_PAGE) {
+ ljam();
+ regUndoPagePtr.i = regLliPtr.p->lliUndoPage;
+ ptrCheckGuard(regUndoPagePtr, cnoOfUndoPage, undoPage);
+
+ regUndoPagePtr.p->undoPageWord[lliWord] = ZLCPR_UNDO_LOG_PAGE_HEADER;
+ regUndoPagePtr.p->undoPageWord[lliWord + 1] = prevRecId;
+ regUndoPagePtr.p->undoPageWord[lliWord + 2] = regFragPtr->fragTableId;
+ regUndoPagePtr.p->undoPageWord[lliWord + 3] = regFragPtr->fragmentId;
+ regUndoPagePtr.p->undoPageWord[lliWord + 4] = regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS];
+ regUndoPagePtr.p->undoPageWord[lliWord + 5] = regPagePtr->pageWord[ZPAGE_STATE_POS];
+ regUndoPagePtr.p->undoPageWord[lliWord + 6] = regPagePtr->pageWord[ZPAGE_NEXT_POS];
+ regLliPtr.p->lliUndoWord = lliWord + 7;
+ } else {
+ ljam();
+ cprAddUndoLogWord(signal, regLliPtr.p, ZLCPR_UNDO_LOG_PAGE_HEADER);
+ cprAddUndoLogWord(signal, regLliPtr.p, prevRecId);
+ cprAddUndoLogWord(signal, regLliPtr.p, regFragPtr->fragTableId);
+ cprAddUndoLogWord(signal, regLliPtr.p, regFragPtr->fragmentId);
+ cprAddUndoLogWord(signal, regLliPtr.p, regPagePtr->pageWord[ZPAGE_FRAG_PAGE_ID_POS]);
+ cprAddUndoLogWord(signal, regLliPtr.p, regPagePtr->pageWord[ZPAGE_STATE_POS]);
+ cprAddUndoLogWord(signal, regLliPtr.p, regPagePtr->pageWord[ZPAGE_NEXT_POS]);
+ }//if
+}//Dbtup::cprAddUndoLogPageHeader()
+
+void Dbtup::cprAddUndoLogRecord(Signal* signal,
+ Uint32 recordType,
+ Uint32 pageId,
+ Uint32 pageIndex,
+ Uint32 tableId,
+ Uint32 fragId,
+ Uint32 localLogIndex)
+{
+ LocalLogInfoPtr regLliPtr;
+ UndoPagePtr regUndoPagePtr;
+
+ regLliPtr.i = localLogIndex;
+ ptrCheckGuard(regLliPtr, cnoOfParallellUndoFiles, localLogInfo);
+
+ Uint32 prevRecId = regLliPtr.p->lliPrevRecordId;
+ Uint32 lliWord = regLliPtr.p->lliUndoWord;
+
+ regLliPtr.p->lliPrevRecordId = lliWord +
+ (regLliPtr.p->lliLogFilePage << ZUNDO_RECORD_ID_PAGE_INDEX);
+ if ((lliWord + 6) < ZWORDS_ON_PAGE) {
+ ljam();
+ regUndoPagePtr.i = regLliPtr.p->lliUndoPage;
+ ptrCheckGuard(regUndoPagePtr, cnoOfUndoPage, undoPage);
+ regUndoPagePtr.p->undoPageWord[lliWord] = recordType;
+ regUndoPagePtr.p->undoPageWord[lliWord + 1] = prevRecId;
+ regUndoPagePtr.p->undoPageWord[lliWord + 2] = tableId;
+ regUndoPagePtr.p->undoPageWord[lliWord + 3] = fragId;
+ regUndoPagePtr.p->undoPageWord[lliWord + 4] = pageId;
+ regUndoPagePtr.p->undoPageWord[lliWord + 5] = pageIndex;
+
+ regLliPtr.p->lliUndoWord = lliWord + 6;
+ } else {
+ ljam();
+ cprAddUndoLogWord(signal, regLliPtr.p, recordType);
+ cprAddUndoLogWord(signal, regLliPtr.p, prevRecId);
+ cprAddUndoLogWord(signal, regLliPtr.p, tableId);
+ cprAddUndoLogWord(signal, regLliPtr.p, fragId);
+ cprAddUndoLogWord(signal, regLliPtr.p, pageId);
+ cprAddUndoLogWord(signal, regLliPtr.p, pageIndex);
+ }//if
+}//Dbtup::cprAddUndoLogRecord()
+
+void Dbtup::cprAddAbortUpdate(Signal* signal,
+ LocalLogInfo* const lliPtr,
+ Operationrec* const regOperPtr)
+{
+ Uint32 lliWord = lliPtr->lliUndoWord;
+ if ((lliWord + 4) < ZWORDS_ON_PAGE) {
+ ljam();
+ UndoPagePtr regUndoPagePtr;
+ regUndoPagePtr.i = lliPtr->lliUndoPage;
+ ptrCheckGuard(regUndoPagePtr, cnoOfUndoPage, undoPage);
+
+ regUndoPagePtr.p->undoPageWord[lliWord] = regOperPtr->fragPageId;
+ regUndoPagePtr.p->undoPageWord[lliWord + 1] = regOperPtr->pageIndex;
+ regUndoPagePtr.p->undoPageWord[lliWord + 2] = regOperPtr->fragPageIdC;
+ regUndoPagePtr.p->undoPageWord[lliWord + 3] = regOperPtr->pageIndexC;
+ lliPtr->lliUndoWord = lliWord + 4;
+ } else {
+ ljam();
+ cprAddUndoLogWord(signal, lliPtr, regOperPtr->fragPageId);
+ cprAddUndoLogWord(signal, lliPtr, regOperPtr->pageIndex);
+ cprAddUndoLogWord(signal, lliPtr, regOperPtr->fragPageIdC);
+ cprAddUndoLogWord(signal, lliPtr, regOperPtr->pageIndexC);
+ }//if
+}//Dbtup::cprAddAbortUpdate()
+
+void Dbtup::cprAddUndoLogWord(Signal* signal, LocalLogInfo* const lliPtr, Uint32 undoWord)
+{
+ DiskBufferSegmentInfoPtr dbsiPtr;
+ UndoPagePtr regUndoPagePtr;
+
+ ljam();
+ regUndoPagePtr.i = lliPtr->lliUndoPage;
+ ptrCheckGuard(regUndoPagePtr, cnoOfUndoPage, undoPage);
+ ndbrequire(lliPtr->lliUndoWord < ZWORDS_ON_PAGE);
+ regUndoPagePtr.p->undoPageWord[lliPtr->lliUndoWord] = undoWord;
+
+ lliPtr->lliUndoWord++;
+ if (lliPtr->lliUndoWord == ZWORDS_ON_PAGE) {
+ ljam();
+ lliPtr->lliUndoWord = ZUNDO_PAGE_HEADER_SIZE;
+ lliPtr->lliUndoPage++;
+ if (clblPageCounter > 0) {
+ ljam();
+ clblPageCounter--;
+ }//if
+ dbsiPtr.i = lliPtr->lliUndoBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ dbsiPtr.p->pdxNumDataPages++;
+ ndbrequire(dbsiPtr.p->pdxNumDataPages < 16);
+ lliPtr->lliLogFilePage++;
+ if (dbsiPtr.p->pdxNumDataPages == ZUB_SEGMENT_SIZE) {
+ ljam();
+ lcpWriteUndoSegment(signal, lliPtr, false);
+ }//if
+ }//if
+}//Dbtup::cprAddUndoLogWord()
+
+void Dbtup::lcpWriteUndoSegment(Signal* signal, LocalLogInfo* const lliPtr, bool flushFlag)
+{
+ DiskBufferSegmentInfoPtr dbsiPtr;
+
+ dbsiPtr.i = lliPtr->lliUndoBufferSegmentP;
+ ptrCheckGuard(dbsiPtr, cnoOfConcurrentWriteOp, diskBufferSegmentInfo);
+ Uint32 flags = 1;
+ lliPtr->lliUndoPagesToDiskWithoutSynch += dbsiPtr.p->pdxNumDataPages;
+ if ((lliPtr->lliUndoPagesToDiskWithoutSynch > MAX_PAGES_WITHOUT_SYNCH) ||
+ (flushFlag)) {
+ ljam();
+/* ---------------------------------------------------------------- */
+// To avoid synching too big chunks at a time we synch after writing
+// a certain number of data pages. (e.g. 2 MBytes).
+/* ---------------------------------------------------------------- */
+ lliPtr->lliUndoPagesToDiskWithoutSynch = 0;
+ flags |= 0x10; //Set synch flag unconditionally
+ }//if
+ dbsiPtr.p->pdxOperation = CHECKPOINT_UNDO_WRITE;
+ signal->theData[0] = lliPtr->lliUndoFileHandle;
+ signal->theData[1] = cownref;
+ signal->theData[2] = dbsiPtr.i;
+ signal->theData[3] = flags;
+ signal->theData[4] = ZBASE_ADDR_UNDO_WORD;
+ signal->theData[5] = dbsiPtr.p->pdxNumDataPages;
+ signal->theData[6] = dbsiPtr.p->pdxDataPage[0];
+ signal->theData[7] = dbsiPtr.p->pdxFilePage;
+ sendSignal(NDBFS_REF, GSN_FSWRITEREQ, signal, 8, JBA);
+
+ DiskBufferSegmentInfoPtr newDbsiPtr;
+ UndoPagePtr newUndoPagePtr;
+
+ seizeUndoBufferSegment(signal, newUndoPagePtr);
+ seizeDiskBufferSegmentRecord(newDbsiPtr);
+ newDbsiPtr.p->pdxBuffertype = UNDO_PAGES;
+ for (Uint32 i = 0; i < ZUB_SEGMENT_SIZE; i++) {
+ newDbsiPtr.p->pdxDataPage[i] = newUndoPagePtr.i + i;
+ }//for
+ newDbsiPtr.p->pdxFilePage = lliPtr->lliLogFilePage;
+ lliPtr->lliUndoPage = newUndoPagePtr.i;
+ lliPtr->lliUndoBufferSegmentP = newDbsiPtr.i;
+}//Dbtup::lcpWriteUndoSegment()
+
+void Dbtup::seizeUndoBufferSegment(Signal* signal, UndoPagePtr& regUndoPagePtr)
+{
+ if (cnoFreeUndoSeg == ZMIN_PAGE_LIMIT_TUP_COMMITREQ) {
+ EXECUTE_DIRECT(DBLQH, GSN_TUP_COM_BLOCK, signal, 1);
+ ljamEntry();
+ }//if
+ cnoFreeUndoSeg--;
+ ndbrequire(cnoFreeUndoSeg >= 0);
+ ndbrequire(cfirstfreeUndoSeg != RNIL);
+ regUndoPagePtr.i = cfirstfreeUndoSeg;
+ ptrCheckGuard(regUndoPagePtr, cnoOfUndoPage, undoPage);
+ cfirstfreeUndoSeg = regUndoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS];
+ regUndoPagePtr.p->undoPageWord[ZPAGE_NEXT_POS] = RNIL;
+}//Dbtup::seizeUndoBufferSegment()
+
+
+
diff --git a/ndb/src/kernel/blocks/dbtup/Makefile b/ndb/src/kernel/blocks/dbtup/Makefile
new file mode 100644
index 00000000000..87146f4b441
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/Makefile
@@ -0,0 +1,26 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := dbtup
+SOURCES = \
+ DbtupExecQuery.cpp \
+ DbtupBuffer.cpp \
+ DbtupRoutines.cpp \
+ DbtupCommit.cpp \
+ DbtupFixAlloc.cpp \
+ DbtupTrigger.cpp \
+ DbtupAbort.cpp \
+ DbtupLCP.cpp \
+ DbtupUndoLog.cpp \
+ DbtupPageMap.cpp \
+ DbtupPagMan.cpp \
+ DbtupStoredProcDef.cpp \
+ DbtupMeta.cpp \
+ DbtupTabDesMan.cpp \
+ DbtupGen.cpp \
+ DbtupSystemRestart.cpp \
+ DbtupIndex.cpp \
+ DbtupDebug.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dbtup/Notes.txt b/ndb/src/kernel/blocks/dbtup/Notes.txt
new file mode 100644
index 00000000000..9d47c591fe8
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtup/Notes.txt
@@ -0,0 +1,183 @@
+Operations, tuples, versions
+============================
+
+Operation types.
+
+INSERT insert new original tuple, or insert after delete
+UPDATE update
+DELETE delete
+
+Following need not be considered here.
+
+READ does not change tuples or versions
+WRITE turns into INSERT or UPDATE in LQH
+
+We use more specific names in some cases:
+
+first/INSERT initial insert of new tuple
+delete/INSERT INSERT preceded by DELETE
+DELETE/last DELETE as last operation
+DELETE/insert DELETE followed by INSERT
+
+Tuple + op Can be followed by
+-------------- ------------------
+does not exist first/INSERT
+tuple exists UPDATE DELETE
+INSERT UPDATE DELETE
+UPDATE UPDATE DELETE
+DELETE delete/INSERT
+
+Operations on same tuple are kept in doubly linked list until
+commit or abort. The links at both ends are RNIL i.e. the list
+is not circular. The links are:
+
+nextActiveOp the operation BEFORE this one, in event order
+prevActiveOp the operation AFTER this one, in event order
+
+Operations are done on the "original tuple" i.e. the tuple is
+modified in place. If an operation is about to write over data
+in original tuple, it first copies the tuple to a "copy tuple".
+
+Operation Copy tuple
+--------- ----------
+first/INSERT no
+delete/INSERT yes (this is in effect an update)
+UPDATE yes
+DELETE no
+
+The operation points to the tuples via:
+
+realPageId page i-value of original tuple
+pageOffset word offset of original tuple on the page
+realPageIdC page i-value of copy tuple or RNIL is no copy exists
+pageOffsetC word offset of copy tuple on the page
+
+The original tuple and the copy tuple (if any) point back to
+the operation via word 0. In copy tuple this pointer is never
+changed. In original tuple however it always points to the LATEST
+existing operation i.e. the one with prevActiveOp == RNIL.
+Thus word 0 of original tuple is changed on 2 occasions:
+
+- when a new operation is added to the list
+- when commit or abort removes the latest operation
+
+Note that commit/abort of operations occurs in random order.
+The list is adjusted accordingly.
+
+Versions
+--------
+
+Tuple version is stored in tuple word 1. A new original tuple
+gets version 0. The version is incremented by each new operation
+which makes a copy tuple. Version number wraps around at 15 bits.
+
+When a copy tuple is made, the version in original tuple is copied
+to copy tuple as part of tuple data. This takes place before
+the version in original tuple is updated.
+
+Each operation record contains tuple version called tupVersion.
+
+- at insert of new original tuple, tupVersion is set to 0
+
+- if tuple already exists, the FIRST operation (in event order)
+ reads tupVersion from tuple word 1. If the operation is
+ not DELETE, the version is incremented
+
+- subsequent operation reads tupVersion from the operation
+ BEFORE it (nextActiveOp). If this subsequent operation is
+ not DELETE, the version is incremented
+
+When the operation writes the tuple it sets word 1 to tupVersion.
+In detail, per operation type, where INSERT is divided into
+insert of new original tuple and insert after delete:
+
+Operation Copy Increment Set version in original
+--------- ---- --------- -----------------------
+first/INSERT no no yes, to 0
+delete/INSERT yes yes yes
+UPDATE yes yes yes
+DELETE no no no
+
+Thus an existing version is incremented if and only if
+a copy tuple is made.
+
+Ordered index maintenance
+-------------------------
+
+Each index entry has logical tuple address and tuple version.
+Index entries are added during prepare phase (when each operation
+is executed) and removed during commit or abort phase.
+
+Access to correct tuple version (original or copy) is required
+in TUX which reads index key values 1) to check that at least one
+is not null 2) to do tree search 3) to set min/max prefixes.
+See "Read attributes" below.
+
+An additional complication is that commit/abort of operations
+arrives in random order. So we cannot check for, for example,
+DELETE/insert by looking at prevActiveOp.
+
+Phase Op Action Version in
+----- -- ------ ----------
+prepare INSERT add op and original
+prepare UPDATE add op and original
+prepare DELETE none -
+
+commit first/INSERT none -
+commit delete/INSERT remove copy tuple 1)
+commit UPDATE remove copy tuple 1)
+commit DELETE/last remove op and original
+commit DELETE/insert none -
+
+abort INSERT remove op
+abort UPDATE remove op
+abort DELETE none -
+
+1) alternatively, store prevTupVersion in operation record.
+
+Read attributes, query status
+-----------------------------
+
+TUP_READ_ATTRS signal (or equivalent direct call) reads attribute
+values. Input is logical address of original tuple and tuple
+version. The steps are:
+
+- Translate logical address to physical address of original tuple.
+
+- If version of original tuple in word 1 is right, stop.
+
+- Otherwise word 0 points to LATEST not yet deleted operation.
+ Walk through operation list via nextActiveOp.
+
+- If an operation on the list has realPageIdC == RNIL, skip it.
+
+- Otherwise find copy tuple via realPageIdC, pageOffsetC.
+ If the version of the copy tuple in word 1 is right, stop.
+
+- Call readAttributes() on the tuple found (original or copy).
+
+In short, the version must exist in some not yet deleted tuple,
+either in original or in some copy.
+
+Note that this must work during all phases since index code
+needs to read index key attributes from correct tuple version in
+each add/remove operation.
+
+TUP_QUERY_TH signal (or equivalent direct call) does same search
+for tuple version. It is called from index scan and returns info
+used to decide if the scan can see the tuple.
+
+This signal may also be called during any phase since commit/abort
+of all operations is not done in one time-slice.
+
+Commit and abort
+----------------
+
+[ hairy stuff ]
+
+Problems
+--------
+
+Current abort code can destroy a tuple version too early. This
+happens in test case "ticuur" (insert-commit-update-update-rollback),
+if abort of first update arrives before abort of second update.
diff --git a/ndb/src/kernel/blocks/dbtux/Dbtux.hpp b/ndb/src/kernel/blocks/dbtux/Dbtux.hpp
new file mode 100644
index 00000000000..c56e455a42a
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/Dbtux.hpp
@@ -0,0 +1,1218 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBTUX_H
+#define DBTUX_H
+
+#include <new>
+#include <ndb_limits.h>
+#include <SimulatedBlock.hpp>
+#include <AttributeHeader.hpp>
+#include <ArrayPool.hpp>
+#include <DataBuffer.hpp>
+#include <md5_hash.hpp>
+
+// signal classes
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/TuxContinueB.hpp>
+#include <signaldata/TuxSizeAltReq.hpp>
+#include <signaldata/BuildIndx.hpp>
+#include <signaldata/TupFrag.hpp>
+#include <signaldata/AlterIndx.hpp>
+#include <signaldata/DropTab.hpp>
+#include <signaldata/TuxMaint.hpp>
+#include <signaldata/TupAccess.hpp>
+#include <signaldata/AccScan.hpp>
+#include <signaldata/TuxBound.hpp>
+#include <signaldata/NextScan.hpp>
+#include <signaldata/AccLock.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+
+// debug
+#ifdef VM_TRACE
+#include <NdbOut.hpp>
+#include <OutputStream.hpp>
+#endif
+
+// jams
+#undef jam
+#undef jamEntry
+#ifdef DBTUX_GEN_CPP
+#define jam() jamLine(10000 + __LINE__)
+#define jamEntry() jamEntryLine(10000 + __LINE__)
+#endif
+#ifdef DBTUX_META_CPP
+#define jam() jamLine(20000 + __LINE__)
+#define jamEntry() jamEntryLine(20000 + __LINE__)
+#endif
+#ifdef DBTUX_MAINT_CPP
+#define jam() jamLine(30000 + __LINE__)
+#define jamEntry() jamEntryLine(30000 + __LINE__)
+#endif
+#ifdef DBTUX_NODE_CPP
+#define jam() jamLine(40000 + __LINE__)
+#define jamEntry() jamEntryLine(40000 + __LINE__)
+#endif
+#ifdef DBTUX_TREE_CPP
+#define jam() jamLine(50000 + __LINE__)
+#define jamEntry() jamEntryLine(50000 + __LINE__)
+#endif
+#ifdef DBTUX_SCAN_CPP
+#define jam() jamLine(60000 + __LINE__)
+#define jamEntry() jamEntryLine(60000 + __LINE__)
+#endif
+#ifdef DBTUX_CMP_CPP
+#define jam() jamLine(70000 + __LINE__)
+#define jamEntry() jamEntryLine(70000 + __LINE__)
+#endif
+#ifdef DBTUX_DEBUG_CPP
+#define jam() jamLine(90000 + __LINE__)
+#define jamEntry() jamEntryLine(90000 + __LINE__)
+#endif
+
+class Configuration;
+
+class Dbtux : public SimulatedBlock {
+public:
+ Dbtux(const Configuration& conf);
+ virtual ~Dbtux();
+
+private:
+ // sizes are in words (Uint32)
+ static const unsigned MaxIndexFragments = 2 * NO_OF_FRAG_PER_NODE;
+ static const unsigned MaxIndexAttributes = MAX_ATTRIBUTES_IN_INDEX;
+#ifdef VM_TRACE
+ static const unsigned MaxNodeHandles = 10000; // More space for printTree
+#else
+ static const unsigned MaxNodeHandles = 128; // enough for 1 operation
+#endif
+ static const unsigned MaxAttrDataSize = 2048;
+ static const unsigned DescPageSize = 256;
+ static const unsigned MaxTreeNodeSize = MAX_TTREE_NODE_SIZE;
+ static const unsigned ScanBoundSegmentSize = 7;
+ static const unsigned MaxAccLockOps = MAX_PARALLEL_OP_PER_SCAN;
+ BLOCK_DEFINES(Dbtux);
+
+ // forward declarations
+ struct DescEnt;
+
+ /*
+ * Pointer to Uint32 data. Interpretation is context dependent.
+ */
+ struct Data {
+ private:
+ Uint32* m_data;
+ public:
+ Data();
+ Data(Uint32* data);
+ Data& operator=(Uint32* data);
+ operator Uint32*() const;
+ Data& operator+=(size_t n);
+ AttributeHeader& ah() const;
+ };
+
+ /*
+ * Pointer to constant Uint32 data.
+ */
+ struct ConstData {
+ private:
+ const Uint32* m_data;
+ public:
+ ConstData();
+ ConstData(const Uint32* data);
+ ConstData& operator=(const Uint32* data);
+ operator const Uint32*() const;
+ ConstData& operator+=(size_t n);
+ const AttributeHeader& ah() const;
+ // non-const pointer can be cast to const pointer
+ ConstData(Data data);
+ ConstData& operator=(Data data);
+ };
+
+ // AttributeHeader size is assumed to be 1 word
+ static const unsigned AttributeHeaderSize = 1;
+
+ /*
+ * Logical tuple address, "local key". Identifies both table tuples
+ * and index tuples. The code assumes it is one word.
+ */
+ typedef Uint32 TupAddr;
+ static const unsigned NullTupAddr = (Uint32)-1;
+
+ /*
+ * Physical tuple address in TUP. Provides fast access to table tuple
+ * or index node. Valid within the db node and across timeslices.
+ * Not valid between db nodes or across restarts.
+ */
+ struct TupLoc {
+ Uint32 m_pageId; // page i-value
+ Uint16 m_pageOffset; // page offset in words
+ TupLoc();
+ };
+
+ // tree definitions
+
+ /*
+ * Tree entry. Points to a tuple in primary table via logical address
+ * of "original" tuple and tuple version. Uses 2 words to get correct
+ * aligment (one byte is wasted currently).
+ */
+ struct TreeEnt {
+ TupAddr m_tupAddr; // address of original tuple
+ Uint16 m_tupVersion; // version
+ Uint8 m_fragBit; // which duplicated table fragment
+ Uint8 unused1;
+ TreeEnt();
+ // methods
+ int cmp(const TreeEnt ent) const;
+ };
+ static const unsigned TreeEntSize = sizeof(TreeEnt) >> 2;
+ static const TreeEnt NullTreeEnt;
+
+ /*
+ * Tree node has 1) fixed part 2) actual table data for min and max
+ * prefix 3) max and min entries 4) rest of entries 5) one extra entry
+ * used as work space.
+ *
+ * struct TreeNode part 1
+ * min prefix part 2, size TreeHead::m_prefSize
+ * max prefix part 2, size TreeHead::m_prefSize
+ * max entry part 3
+ * min entry part 3
+ * rest of entries part 4
+ * work entry part 5
+ *
+ * Occupancy (number of entries) is at least 1 except temporarily when
+ * a node is about to be removed. If occupancy is 1, only max entry
+ * is present but both min and max prefixes are set.
+ */
+ struct TreeNode {
+ TupAddr m_link[3]; // link to 0-left child 1-right child 2-parent
+ Uint8 m_side; // we are 0-left child 1-right child 2-root
+ Uint8 m_occup; // current number of entries
+ Int8 m_balance; // balance -1, 0, +1
+ Uint8 unused1;
+ Uint32 m_nodeScan; // list of scans at this node
+ TreeNode();
+ };
+ static const unsigned NodeHeadSize = sizeof(TreeNode) >> 2;
+
+ /*
+ * Tree nodes are not always accessed fully, for cache reasons. There
+ * are 3 access sizes.
+ */
+ enum AccSize {
+ AccNone = 0,
+ AccHead = 1, // part 1
+ AccPref = 2, // parts 1-3
+ AccFull = 3 // parts 1-5
+ };
+
+ /*
+ * Tree header. There is one in each fragment. Contains tree
+ * parameters and address of root node.
+ */
+ struct TreeHead {
+ Uint8 m_nodeSize; // words in tree node
+ Uint8 m_prefSize; // words in min/max prefix each
+ Uint8 m_minOccup; // min entries in internal node
+ Uint8 m_maxOccup; // max entries in node
+ TupAddr m_root; // root node
+ TreeHead();
+ // methods
+ unsigned getSize(AccSize acc) const;
+ Data getPref(TreeNode* node, unsigned i) const;
+ TreeEnt* getEntList(TreeNode* node) const;
+ };
+
+ /*
+ * Tree position. Specifies node, position within node (from 0 to
+ * m_occup), and whether the position is at an existing entry or
+ * before one (if any). Position m_occup points past the node and is
+ * also represented by position 0 of next node. Includes direction
+ * and copy of entry used by scan.
+ */
+ struct TreePos {
+ TupAddr m_addr; // logical node address
+ TupLoc m_loc; // physical address
+ Uint16 m_pos; // position 0 to m_occup
+ Uint8 m_match; // at an existing entry
+ Uint8 m_dir; // from link (0-2) or within node (3)
+ TreeEnt m_ent; // copy of current entry
+ TreePos();
+ };
+
+ // packed metadata
+
+ /*
+ * Descriptor page. The "hot" metadata for an index is stored as
+ * a contiguous array of words on some page.
+ */
+ struct DescPage {
+ Uint32 m_nextPage;
+ Uint32 m_numFree; // number of free words
+ union {
+ Uint32 m_data[DescPageSize];
+ Uint32 nextPool;
+ };
+ DescPage();
+ };
+ typedef Ptr<DescPage> DescPagePtr;
+ ArrayPool<DescPage> c_descPagePool;
+ Uint32 c_descPageList;
+
+ /*
+ * Header for index metadata. Size must be multiple of word size.
+ */
+ struct DescHead {
+ unsigned m_indexId : 24;
+ unsigned pad1 : 8;
+ };
+ static const unsigned DescHeadSize = sizeof(DescHead) >> 2;
+
+ /*
+ * Attribute metadata. Size must be multiple of word size.
+ */
+ struct DescAttr {
+ unsigned m_primaryAttrId : 16;
+ unsigned m_typeId : 8;
+ unsigned m_nullable : 1;
+ unsigned pad1 : 7;
+ };
+ static const unsigned DescAttrSize = sizeof(DescAttr) >> 2;
+
+ /*
+ * Complete metadata for one index. The array of attributes has
+ * variable size.
+ */
+ struct DescEnt {
+ DescHead m_descHead;
+ DescAttr m_descAttr[1]; // variable size data
+ };
+
+ // range scan
+
+ /*
+ * Scan bounds are stored in linked list of segments.
+ */
+ typedef DataBuffer<ScanBoundSegmentSize> ScanBound;
+ typedef DataBuffer<ScanBoundSegmentSize>::ConstDataBufferIterator ScanBoundIterator;
+ typedef DataBuffer<ScanBoundSegmentSize>::DataBufferPool ScanBoundPool;
+ ScanBoundPool c_scanBoundPool;
+
+ /*
+ * Scan operation.
+ *
+ * Tuples are locked one at a time. The current lock op is set to
+ * RNIL as soon as the lock is obtained and passed to LQH. We must
+ * however remember all locks which LQH has not returned for unlocking
+ * since they must be aborted by us when the scan is closed.
+ *
+ * Scan state describes the entry we are interested in. There is
+ * a separate lock wait flag. It may be for current entry or it may
+ * be for an entry we were moved away from. In any case nothing
+ * happens with current entry before lock wait flag is cleared.
+ */
+ struct ScanOp {
+ enum {
+ Undef = 0,
+ First = 1, // before first entry
+ Current = 2, // at current before locking
+ Blocked = 3, // at current waiting for ACC lock
+ Locked = 4, // at current and locked or no lock needed
+ Next = 5, // looking for next extry
+ Last = 6, // after last entry
+ Aborting = 7, // lock wait at scan close
+ Invalid = 9 // cannot return REF to LQH currently
+ };
+ Uint16 m_state;
+ Uint16 m_lockwait;
+ Uint32 m_userPtr; // scanptr.i in LQH
+ Uint32 m_userRef;
+ Uint32 m_tableId;
+ Uint32 m_indexId;
+ Uint32 m_fragId;
+ Uint32 m_fragPtrI;
+ Uint32 m_transId1;
+ Uint32 m_transId2;
+ Uint32 m_savePointId;
+ // lock waited for or obtained and not yet passed to LQH
+ Uint32 m_accLockOp;
+ // locks obtained and passed to LQH but not yet returned by LQH
+ Uint32 m_accLockOps[MaxAccLockOps];
+ Uint8 m_readCommitted; // no locking
+ Uint8 m_lockMode;
+ Uint8 m_keyInfo;
+ ScanBound m_boundMin;
+ ScanBound m_boundMax;
+ ScanBound* m_bound[2]; // pointers to above 2
+ Uint16 m_boundCnt[2]; // number of bounds in each
+ TreePos m_scanPos; // position
+ TreeEnt m_lastEnt; // last entry returned
+ Uint32 m_nodeScan; // next scan at node (single-linked)
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+ ScanOp(ScanBoundPool& scanBoundPool);
+ };
+ typedef Ptr<ScanOp> ScanOpPtr;
+ ArrayPool<ScanOp> c_scanOpPool;
+
+ // indexes and fragments
+
+ /*
+ * Ordered index. Top level data structure. The primary table (table
+ * being indexed) lives in TUP.
+ */
+ struct Index {
+ enum State {
+ NotDefined = 0,
+ Defining = 1,
+ Online = 2, // triggers activated and build done
+ Dropping = 9
+ };
+ State m_state;
+ DictTabInfo::TableType m_tableType;
+ Uint32 m_tableId;
+ Uint16 m_fragOff; // offset for duplicate fragId bits
+ Uint16 m_numFrags;
+ Uint32 m_fragId[MaxIndexFragments];
+ Uint32 m_fragPtrI[MaxIndexFragments];
+ Uint32 m_descPage; // descriptor page
+ Uint16 m_descOff; // offset within the page
+ Uint16 m_numAttrs;
+ union {
+ Uint32 nextPool;
+ };
+ Index();
+ };
+ typedef Ptr<Index> IndexPtr;
+ ArrayPool<Index> c_indexPool;
+
+ /*
+ * Fragment of an index, as known to DIH/TC. Represents the two
+ * duplicate fragments known to LQH/ACC/TUP. Includes tree header.
+ * There are no maintenance operation records yet.
+ */
+ struct Frag {
+ Uint32 m_tableId; // copy from index level
+ Uint32 m_indexId;
+ Uint16 m_fragOff;
+ Uint16 m_fragId;
+ Uint32 m_descPage; // copy from index level
+ Uint16 m_descOff;
+ Uint16 m_numAttrs;
+ TreeHead m_tree;
+ Uint32 m_nodeList; // node cache of current operation
+ Uint32 m_nodeFree; // one node pre-allocated for insert
+ DLList<ScanOp> m_scanList; // current scans on this fragment
+ union {
+ Uint32 nextPool;
+ };
+ Frag(ArrayPool<ScanOp>& scanOpPool);
+ };
+ typedef Ptr<Frag> FragPtr;
+ ArrayPool<Frag> c_fragPool;
+
+ /*
+ * Fragment metadata operation.
+ */
+ struct FragOp {
+ Uint32 m_userPtr;
+ Uint32 m_userRef;
+ Uint32 m_indexId;
+ Uint32 m_fragId;
+ Uint32 m_fragPtrI;
+ Uint32 m_fragNo; // fragment number starting at zero
+ Uint32 m_numAttrsRecvd;
+ union {
+ Uint32 nextPool;
+ };
+ FragOp();
+ };
+ typedef Ptr<FragOp> FragOpPtr;
+ ArrayPool<FragOp> c_fragOpPool;
+
+ // node handles
+
+ /*
+ * A tree operation builds a cache of accessed nodes. This allows
+ * different implementations of index memory access. The cache is
+ * committed and released at the end of the operation.
+ */
+ struct NodeHandle {
+ enum Flags {
+ // bits 0,1 mark need for left,right prefix
+ DoInsert = (1 << 2),
+ DoDelete = (1 << 3),
+ DoUpdate = (1 << 4)
+ };
+ Dbtux& m_tux; // this block
+ Frag& m_frag; // fragment using the node
+ TupAddr m_addr; // logical node address
+ TupLoc m_loc; // physical node address
+ AccSize m_acc; // accessed size
+ unsigned m_flags; // flags
+ union {
+ Uint32 m_next; // next active node under fragment
+ Uint32 nextPool;
+ };
+ TreeNode* m_node; // pointer to node storage
+ Uint32 m_cache[MaxTreeNodeSize];
+ NodeHandle(Dbtux& tux, Frag& frag);
+ // getters
+ TupAddr getLink(unsigned i);
+ unsigned getChilds(); // cannot spell
+ unsigned getSide();
+ unsigned getOccup();
+ int getBalance();
+ Uint32 getNodeScan();
+ Data getPref(unsigned i);
+ TreeEnt getEnt(unsigned pos);
+ TreeEnt getMinMax(unsigned i);
+ // setters
+ void setLink(unsigned i, TupAddr addr);
+ void setSide(unsigned i);
+ void setOccup(unsigned n);
+ void setBalance(int b);
+ void setNodeScan(Uint32 scanPtrI);
+ // operations XXX maybe these should move to Dbtux level
+ void pushUp(Signal* signal, unsigned pos, const TreeEnt& ent);
+ void popDown(Signal* signal, unsigned pos, TreeEnt& ent);
+ void pushDown(Signal* signal, unsigned pos, TreeEnt& ent);
+ void popUp(Signal* signal, unsigned pos, TreeEnt& ent);
+ void slide(Signal* signal, Ptr<NodeHandle> nodePtr, unsigned i);
+ void linkScan(Dbtux::ScanOpPtr scanPtr);
+ void unlinkScan(Dbtux::ScanOpPtr scanPtr);
+ bool islinkScan(Dbtux::ScanOpPtr scanPtr);
+ // for ndbrequire
+ void progError(int line, int cause, const char* extra);
+ };
+ typedef Ptr<NodeHandle> NodeHandlePtr;
+ ArrayPool<NodeHandle> c_nodeHandlePool;
+ friend class NodeHandle;
+
+ // parameters for methods
+
+ /*
+ * Copy attribute data.
+ */
+ struct CopyPar {
+ unsigned m_items; // number of attributes
+ bool m_headers; // copy headers flag (default true)
+ unsigned m_maxwords; // limit size (default no limit)
+ // output
+ unsigned m_numitems; // number of attributes fully copied
+ unsigned m_numwords; // number of words copied
+ CopyPar();
+ };
+
+ /*
+ * Read index key attributes.
+ */
+ struct ReadPar {
+ TreeEnt m_ent; // tuple to read
+ unsigned m_first; // first index attribute
+ unsigned m_count; // number of consecutive index attributes
+ Data m_data; // set pointer if 0 else copy result to it
+ unsigned m_size; // number of words (set in read keys only)
+ ReadPar();
+ };
+
+ /*
+ * Node storage operation.
+ */
+ struct StorePar {
+ TupStoreTh::OpCode m_opCode;// operation code
+ unsigned m_offset; // data offset in words
+ unsigned m_size; // number of words
+ Uint32 m_errorCode; // terrorCode from TUP
+ StorePar();
+ };
+
+ /*
+ * Tree search for entry.
+ */
+ struct SearchPar {
+ ConstData m_data; // input index key values
+ TreeEnt m_ent; // input tuple and version
+ SearchPar();
+ };
+
+ /*
+ * Attribute data comparison.
+ */
+ struct CmpPar {
+ ConstData m_data1; // full search key
+ ConstData m_data2; // full or prefix data
+ unsigned m_len2; // words in data2 buffer
+ unsigned m_first; // first attribute
+ unsigned m_numEq; // number of initial equal attributes
+ CmpPar();
+ };
+
+ /*
+ * Scan bound comparison.
+ */
+ struct BoundPar {
+ ConstData m_data1; // full bound data
+ ConstData m_data2; // full or prefix data
+ unsigned m_count1; // number of bounds
+ unsigned m_len2; // words in data2 buffer
+ unsigned m_dir; // 0-lower bound 1-upper bound
+ BoundPar();
+ };
+
+ // methods
+
+ /*
+ * DbtuxGen.cpp
+ */
+ void execCONTINUEB(Signal* signal);
+ void execSTTOR(Signal* signal);
+ void execSIZEALT_REP(Signal* signal);
+ // utils
+ void copyAttrs(Data dst, ConstData src, CopyPar& copyPar);
+
+ /*
+ * DbtuxMeta.cpp
+ */
+ void execTUXFRAGREQ(Signal* signal);
+ void execTUX_ADD_ATTRREQ(Signal* signal);
+ void execALTER_INDX_REQ(Signal* signal);
+ void execDROP_TAB_REQ(Signal* signal);
+ bool allocDescEnt(IndexPtr indexPtr);
+ void freeDescEnt(IndexPtr indexPtr);
+ void dropIndex(Signal* signal, IndexPtr indexPtr, Uint32 senderRef, Uint32 senderData);
+ // helpers
+ DescEnt& getDescEnt(Uint32 descPage, Uint32 descOff);
+
+ /*
+ * DbtuxMaint.cpp
+ */
+ void execTUX_MAINT_REQ(Signal* signal);
+ void tupReadAttrs(Signal* signal, const Frag& frag, ReadPar& readPar);
+ void tupReadKeys(Signal* signal, const Frag& frag, ReadPar& readPar);
+ void tupStoreTh(Signal* signal, const Frag& frag, NodeHandlePtr nodePtr, StorePar storePar);
+
+ /*
+ * DbtuxNode.cpp
+ */
+ void seizeNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr);
+ void preallocNode(Signal* signal, Frag& frag, Uint32& errorCode);
+ void findNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, TupAddr addr);
+ void selectNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, TupAddr addr, AccSize acc);
+ void insertNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, AccSize acc);
+ void deleteNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr);
+ void accessNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, AccSize acc);
+ void setNodePref(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, unsigned i);
+ void commitNodes(Signal* signal, Frag& frag, bool updateOk);
+
+ /*
+ * DbtuxTree.cpp
+ */
+ void treeSearch(Signal* signal, Frag& frag, SearchPar searchPar, TreePos& treePos);
+ void treeAdd(Signal* signal, Frag& frag, TreePos treePos, TreeEnt ent);
+ void treeRemove(Signal* signal, Frag& frag, TreePos treePos);
+ void treeRotateSingle(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, unsigned i);
+ void treeRotateDouble(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, unsigned i);
+
+ /*
+ * DbtuxScan.cpp
+ */
+ void execACC_SCANREQ(Signal* signal);
+ void execTUX_BOUND_INFO(Signal* signal);
+ void execNEXT_SCANREQ(Signal* signal);
+ void execACC_CHECK_SCAN(Signal* signal);
+ void execACCKEYCONF(Signal* signal);
+ void execACCKEYREF(Signal* signal);
+ void execACC_ABORTCONF(Signal* signal);
+ void scanFirst(Signal* signal, ScanOpPtr scanPtr);
+ void scanNext(Signal* signal, ScanOpPtr scanPtr);
+ bool scanVisible(Signal* signal, ScanOpPtr scanPtr, TreeEnt ent);
+ void scanClose(Signal* signal, ScanOpPtr scanPtr);
+ void addAccLockOp(ScanOp& scan, Uint32 accLockOp);
+ void removeAccLockOp(ScanOp& scan, Uint32 accLockOp);
+ void releaseScanOp(ScanOpPtr& scanPtr);
+
+ /*
+ * DbtuxCmp.cpp
+ */
+ int cmpTreeAttrs(const Frag& frag, CmpPar& cmpPar);
+ int cmpScanBound(const Frag& frag, const BoundPar boundPar);
+
+ /*
+ * DbtuxDebug.cpp
+ */
+ void execDUMP_STATE_ORD(Signal* signal);
+#ifdef VM_TRACE
+ struct PrintPar {
+ char m_path[100]; // LR prefix
+ unsigned m_side; // expected side
+ TupAddr m_parent; // expected parent address
+ int m_depth; // returned depth
+ unsigned m_occup; // returned occupancy
+ bool m_ok; // returned status
+ PrintPar();
+ };
+ void printTree(Signal* signal, Frag& frag, NdbOut& out);
+ void printNode(Signal* signal, Frag& frag, NdbOut& out, TupAddr addr, PrintPar& par);
+ friend class NdbOut& operator<<(NdbOut&, const TreeEnt&);
+ friend class NdbOut& operator<<(NdbOut&, const TreeNode&);
+ friend class NdbOut& operator<<(NdbOut&, const TreeHead&);
+ friend class NdbOut& operator<<(NdbOut&, const TreePos&);
+ friend class NdbOut& operator<<(NdbOut&, const DescAttr&);
+ friend class NdbOut& operator<<(NdbOut&, const Index&);
+ friend class NdbOut& operator<<(NdbOut&, const Frag&);
+ friend class NdbOut& operator<<(NdbOut&, const NodeHandle&);
+ friend class NdbOut& operator<<(NdbOut&, const ScanOp&);
+ FILE* debugFile;
+ NdbOut debugOut;
+ unsigned debugFlags;
+ enum {
+ DebugMeta = 1, // log create and drop index
+ DebugMaint = 2, // log maintenance ops
+ DebugTree = 4, // log and check tree after each op
+ DebugScan = 8 // log scans
+ };
+#endif
+
+ // start up info
+ Uint32 c_internalStartPhase;
+ Uint32 c_typeOfStart;
+
+ // buffers
+ Data c_keyBuffer; // search key or scan bound
+
+ // small stuff
+ static unsigned min(unsigned x, unsigned y);
+ static unsigned max(unsigned x, unsigned y);
+};
+
+// Dbtux::Data
+
+inline
+Dbtux::Data::Data() :
+ m_data(0)
+{
+}
+
+inline
+Dbtux::Data::Data(Uint32* data) :
+ m_data(data)
+{
+}
+
+inline Dbtux::Data&
+Dbtux::Data::operator=(Uint32* data)
+{
+ m_data = data;
+ return *this;
+}
+
+inline
+Dbtux::Data::operator Uint32*() const
+{
+ return m_data;
+}
+
+inline Dbtux::Data&
+Dbtux::Data::operator+=(size_t n)
+{
+ m_data += n;
+ return *this;
+}
+
+inline AttributeHeader&
+Dbtux::Data::ah() const
+{
+ return *reinterpret_cast<AttributeHeader*>(m_data);
+}
+
+// Dbtux::ConstData
+
+inline
+Dbtux::ConstData::ConstData() :
+ m_data(0)
+{
+}
+
+inline
+Dbtux::ConstData::ConstData(const Uint32* data) :
+ m_data(data)
+{
+}
+
+inline Dbtux::ConstData&
+Dbtux::ConstData::operator=(const Uint32* data)
+{
+ m_data = data;
+ return *this;
+}
+
+inline
+Dbtux::ConstData::operator const Uint32*() const
+{
+ return m_data;
+}
+
+inline Dbtux::ConstData&
+Dbtux::ConstData::operator+=(size_t n)
+{
+ m_data += n;
+ return *this;
+}
+
+inline const AttributeHeader&
+Dbtux::ConstData::ah() const
+{
+ return *reinterpret_cast<const AttributeHeader*>(m_data);
+}
+
+inline
+Dbtux::ConstData::ConstData(Data data) :
+ m_data(static_cast<Uint32*>(data))
+{
+}
+
+inline Dbtux::ConstData&
+Dbtux::ConstData::operator=(Data data)
+{
+ m_data = static_cast<Uint32*>(data);
+ return *this;
+}
+
+// Dbtux::TreeEnt
+
+inline int
+Dbtux::TreeEnt::cmp(const TreeEnt ent) const
+{
+ // compare frags first (not optimal but makes easier to read logs)
+ if (m_fragBit < ent.m_fragBit)
+ return -1;
+ if (m_fragBit > ent.m_fragBit)
+ return +1;
+ if (m_tupAddr < ent.m_tupAddr)
+ return -1;
+ if (m_tupAddr > ent.m_tupAddr)
+ return +1;
+ if (m_tupVersion < ent.m_tupVersion)
+ return -1;
+ if (m_tupVersion > ent.m_tupVersion)
+ return +1;
+ return 0;
+}
+
+// Dbtux::TreeHead
+
+inline unsigned
+Dbtux::TreeHead::getSize(AccSize acc) const
+{
+ switch (acc) {
+ case AccNone:
+ return 0;
+ case AccHead:
+ return NodeHeadSize;
+ case AccPref:
+ return NodeHeadSize + 2 * m_prefSize + 2 * TreeEntSize;
+ case AccFull:
+ return m_nodeSize;
+ }
+ REQUIRE(false, "invalid Dbtux::AccSize");
+ return 0;
+}
+
+inline Dbtux::Data
+Dbtux::TreeHead::getPref(TreeNode* node, unsigned i) const
+{
+ Uint32* ptr = (Uint32*)node + NodeHeadSize + i * m_prefSize;
+ return ptr;
+}
+
+inline Dbtux::TreeEnt*
+Dbtux::TreeHead::getEntList(TreeNode* node) const
+{
+ Uint32* ptr = (Uint32*)node + NodeHeadSize + 2 * m_prefSize;
+ return (TreeEnt*)ptr;
+}
+
+// Dbtux
+
+// constructors
+
+inline
+Dbtux::TupLoc::TupLoc() :
+ m_pageId(RNIL),
+ m_pageOffset(0)
+{
+}
+
+inline
+Dbtux::TreeEnt::TreeEnt() :
+ m_tupAddr(NullTupAddr),
+ m_tupVersion(0),
+ m_fragBit(255),
+ unused1(0)
+{
+}
+
+inline
+Dbtux::TreeNode::TreeNode() :
+ m_side(255),
+ m_occup(0),
+ m_balance(0),
+ unused1(0xa1),
+ m_nodeScan(RNIL)
+{
+ m_link[0] = NullTupAddr;
+ m_link[1] = NullTupAddr;
+ m_link[2] = NullTupAddr;
+}
+
+inline
+Dbtux::TreeHead::TreeHead() :
+ m_nodeSize(0),
+ m_prefSize(0),
+ m_minOccup(0),
+ m_maxOccup(0),
+ m_root(0)
+{
+}
+
+inline
+Dbtux::TreePos::TreePos() :
+ m_addr(NullTupAddr),
+ m_loc(),
+ m_pos(ZNIL),
+ m_match(false),
+ m_dir(255),
+ m_ent()
+{
+}
+
+inline
+Dbtux::DescPage::DescPage() :
+ m_nextPage(RNIL),
+ m_numFree(ZNIL)
+{
+ for (unsigned i = 0; i < DescPageSize; i++) {
+#ifdef VM_TRACE
+ m_data[i] = 0x13571357;
+#else
+ m_data[i] = 0;
+#endif
+ }
+}
+
+inline
+Dbtux::Index::Index() :
+ m_state(NotDefined),
+ m_tableType(DictTabInfo::UndefTableType),
+ m_tableId(RNIL),
+ m_numFrags(0),
+ m_descPage(RNIL),
+ m_descOff(0),
+ m_numAttrs(0)
+{
+ for (unsigned i = 0; i < MaxIndexFragments; i++) {
+ m_fragId[i] = ZNIL;
+ m_fragPtrI[i] = RNIL;
+ };
+};
+
+inline
+Dbtux::Frag::Frag(ArrayPool<ScanOp>& scanOpPool) :
+ m_tableId(RNIL),
+ m_indexId(RNIL),
+ m_fragOff(ZNIL),
+ m_fragId(ZNIL),
+ m_descPage(RNIL),
+ m_descOff(0),
+ m_numAttrs(ZNIL),
+ m_tree(),
+ m_nodeList(RNIL),
+ m_nodeFree(RNIL),
+ m_scanList(scanOpPool)
+{
+}
+
+inline
+Dbtux::FragOp::FragOp() :
+ m_userPtr(RNIL),
+ m_userRef(RNIL),
+ m_indexId(RNIL),
+ m_fragId(ZNIL),
+ m_fragPtrI(RNIL),
+ m_fragNo(ZNIL),
+ m_numAttrsRecvd(ZNIL)
+{
+};
+
+inline
+Dbtux::NodeHandle::NodeHandle(Dbtux& tux, Frag& frag) :
+ m_tux(tux),
+ m_frag(frag),
+ m_addr(NullTupAddr),
+ m_loc(),
+ m_acc(AccNone),
+ m_flags(0),
+ m_next(RNIL),
+ m_node(0)
+{
+}
+
+inline
+Dbtux::ScanOp::ScanOp(ScanBoundPool& scanBoundPool) :
+ m_state(Undef),
+ m_lockwait(false),
+ m_userPtr(RNIL),
+ m_userRef(RNIL),
+ m_tableId(RNIL),
+ m_indexId(RNIL),
+ m_fragPtrI(RNIL),
+ m_transId1(0),
+ m_transId2(0),
+ m_savePointId(0),
+ m_accLockOp(RNIL),
+ m_readCommitted(0),
+ m_lockMode(0),
+ m_keyInfo(0),
+ m_boundMin(scanBoundPool),
+ m_boundMax(scanBoundPool),
+ m_scanPos(),
+ m_lastEnt(),
+ m_nodeScan(RNIL)
+{
+ m_bound[0] = &m_boundMin;
+ m_bound[1] = &m_boundMax;
+ m_boundCnt[0] = 0;
+ m_boundCnt[1] = 0;
+ for (unsigned i = 0; i < MaxAccLockOps; i++) {
+ m_accLockOps[i] = RNIL;
+ }
+}
+
+inline
+Dbtux::CopyPar::CopyPar() :
+ m_items(0),
+ m_headers(true),
+ m_maxwords(~0), // max unsigned
+ // output
+ m_numitems(0),
+ m_numwords(0)
+{
+}
+
+inline
+Dbtux::ReadPar::ReadPar() :
+ m_first(0),
+ m_count(0),
+ m_data(0),
+ m_size(0)
+{
+}
+
+inline
+Dbtux::StorePar::StorePar() :
+ m_opCode(TupStoreTh::OpUndefined),
+ m_offset(0),
+ m_size(0),
+ m_errorCode(0)
+{
+}
+
+inline
+Dbtux::SearchPar::SearchPar() :
+ m_data(0),
+ m_ent()
+{
+}
+
+inline
+Dbtux::CmpPar::CmpPar() :
+ m_data1(0),
+ m_data2(0),
+ m_len2(0),
+ m_first(0),
+ m_numEq(0)
+{
+}
+
+inline
+Dbtux::BoundPar::BoundPar() :
+ m_data1(0),
+ m_data2(0),
+ m_count1(0),
+ m_len2(0),
+ m_dir(255)
+{
+}
+
+#ifdef VM_TRACE
+inline
+Dbtux::PrintPar::PrintPar() :
+ // caller fills in
+ m_path(),
+ m_side(255),
+ m_parent(NullTupAddr),
+ // default return values
+ m_depth(0),
+ m_occup(0),
+ m_ok(true)
+{
+}
+#endif
+
+// node handles
+
+inline Dbtux::TupAddr
+Dbtux::NodeHandle::getLink(unsigned i)
+{
+ ndbrequire(i <= 2);
+ return m_node->m_link[i];
+}
+
+inline unsigned
+Dbtux::NodeHandle::getChilds()
+{
+ return
+ (m_node->m_link[0] != NullTupAddr) +
+ (m_node->m_link[1] != NullTupAddr);
+}
+
+inline Dbtux::TupAddr
+Dbtux::NodeHandle::getSide()
+{
+ return m_node->m_side;
+}
+
+inline unsigned
+Dbtux::NodeHandle::getOccup()
+{
+ return m_node->m_occup;
+}
+
+inline int
+Dbtux::NodeHandle::getBalance()
+{
+ return m_node->m_balance;
+}
+
+inline Uint32
+Dbtux::NodeHandle::getNodeScan()
+{
+ return m_node->m_nodeScan;
+}
+
+inline Dbtux::Data
+Dbtux::NodeHandle::getPref(unsigned i)
+{
+ TreeHead& tree = m_frag.m_tree;
+ ndbrequire(m_acc >= AccPref && i <= 1);
+ return tree.getPref(m_node, i);
+}
+
+inline Dbtux::TreeEnt
+Dbtux::NodeHandle::getEnt(unsigned pos)
+{
+ TreeHead& tree = m_frag.m_tree;
+ TreeEnt* entList = tree.getEntList(m_node);
+ const unsigned occup = m_node->m_occup;
+ ndbrequire(pos < occup);
+ if (pos == 0 || pos == occup - 1) {
+ ndbrequire(m_acc >= AccPref)
+ } else {
+ ndbrequire(m_acc == AccFull)
+ }
+ return entList[(1 + pos) % occup];
+}
+
+inline Dbtux::TreeEnt
+Dbtux::NodeHandle::getMinMax(unsigned i)
+{
+ const unsigned occup = m_node->m_occup;
+ ndbrequire(i <= 1 && occup != 0);
+ return getEnt(i == 0 ? 0 : occup - 1);
+}
+
+inline void
+Dbtux::NodeHandle::setLink(unsigned i, TupAddr addr)
+{
+ ndbrequire(i <= 2);
+ m_node->m_link[i] = addr;
+ m_flags |= DoUpdate;
+}
+
+inline void
+Dbtux::NodeHandle::setSide(unsigned i)
+{
+ // ndbrequire(i <= 1);
+ m_node->m_side = i;
+ m_flags |= DoUpdate;
+}
+
+inline void
+Dbtux::NodeHandle::setOccup(unsigned n)
+{
+ TreeHead& tree = m_frag.m_tree;
+ ndbrequire(n <= tree.m_maxOccup);
+ m_node->m_occup = n;
+ m_flags |= DoUpdate;
+}
+
+inline void
+Dbtux::NodeHandle::setBalance(int b)
+{
+ ndbrequire(abs(b) <= 1);
+ m_node->m_balance = b;
+ m_flags |= DoUpdate;
+}
+
+inline void
+Dbtux::NodeHandle::setNodeScan(Uint32 scanPtrI)
+{
+ m_node->m_nodeScan = scanPtrI;
+ m_flags |= DoUpdate;
+}
+
+// other methods
+
+inline Dbtux::DescEnt&
+Dbtux::getDescEnt(Uint32 descPage, Uint32 descOff)
+{
+ DescPagePtr pagePtr;
+ pagePtr.i = descPage;
+ c_descPagePool.getPtr(pagePtr);
+ ndbrequire(descOff < DescPageSize);
+ DescEnt* descEnt = (DescEnt*)&pagePtr.p->m_data[descOff];
+ return *descEnt;
+}
+
+inline unsigned
+Dbtux::min(unsigned x, unsigned y)
+{
+ return x < y ? x : y;
+}
+
+inline unsigned
+Dbtux::max(unsigned x, unsigned y)
+{
+ return x > y ? x : y;
+}
+
+#endif
diff --git a/ndb/src/kernel/blocks/dbtux/DbtuxCmp.cpp b/ndb/src/kernel/blocks/dbtux/DbtuxCmp.cpp
new file mode 100644
index 00000000000..6404cc66213
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/DbtuxCmp.cpp
@@ -0,0 +1,202 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUX_CMP_CPP
+#include "Dbtux.hpp"
+
+/*
+ * Search key vs tree entry.
+ *
+ * Compare search key and index attribute data. The attribute data may
+ * be partial in which case CmpUnknown may be returned. Also counts how
+ * many (additional) initial attributes were equal.
+ */
+int
+Dbtux::cmpTreeAttrs(const Frag& frag, CmpPar& cmpPar)
+{
+ const DescEnt& descEnt = getDescEnt(frag.m_descPage, frag.m_descOff);
+ ConstData data1 = cmpPar.m_data1;
+ ConstData data2 = cmpPar.m_data2;
+ // number of words of attribute data left
+ unsigned len2 = cmpPar.m_len2;
+ const unsigned numAttrs = frag.m_numAttrs;
+ unsigned index = cmpPar.m_first;
+ ndbrequire(index < numAttrs);
+ // skip to right position in search key XXX do it before the call
+ for (unsigned i = 0; i < index; i++) {
+ jam();
+ data1 += AttributeHeaderSize + data1.ah().getDataSize();
+ }
+ unsigned numEq = 0;
+ int ret = 0;
+ while (index < numAttrs) {
+ if (len2 < AttributeHeaderSize) {
+ jam();
+ ret = NdbSqlUtil::CmpUnknown;
+ break;
+ }
+ len2 -= AttributeHeaderSize;
+ if (! data1.ah().isNULL()) {
+ if (! data2.ah().isNULL()) {
+ jam();
+ // current attribute
+ const DescAttr& descAttr = descEnt.m_descAttr[index];
+ const unsigned typeId = descAttr.m_typeId;
+ // full data size
+ const unsigned size1 = data1.ah().getDataSize();
+ ndbrequire(size1 != 0 && size1 == data2.ah().getDataSize());
+ const unsigned size2 = min(size1, len2);
+ len2 -= size2;
+ // compare
+ const Uint32* const p1 = &data1[AttributeHeaderSize];
+ const Uint32* const p2 = &data2[AttributeHeaderSize];
+ ret = NdbSqlUtil::cmp(typeId, p1, p2, size1, size2);
+ if (ret != 0) {
+ jam();
+ break;
+ }
+ } else {
+ jam();
+ // not NULL < NULL
+ ret = -1;
+ break;
+ }
+ } else {
+ if (! data2.ah().isNULL()) {
+ jam();
+ // NULL > not NULL
+ ret = +1;
+ break;
+ }
+ }
+ data1 += AttributeHeaderSize + data1.ah().getDataSize();
+ data2 += AttributeHeaderSize + data2.ah().getDataSize();
+ numEq++;
+ index++;
+ }
+ // XXX until data format errors are handled
+ ndbrequire(ret != NdbSqlUtil::CmpError);
+ cmpPar.m_numEq += numEq; // add to previous count
+ return ret;
+}
+
+/*
+ * Scan bound vs tree entry.
+ *
+ * Compare lower or upper bound and index attribute data. The attribute
+ * data may be partial in which case CmpUnknown may be returned.
+ * Returns -1 if the boundary is to the left of the compared key and +1 if
+ * the boundary is to the right of the compared key.
+ *
+ * To get this behaviour we treat equality a little bit special.
+ * If the boundary is a lower bound then the boundary is to the left of all
+ * equal keys and if it is an upper bound then the boundary is to the right
+ * of all equal keys.
+ *
+ * When searching for the first key we are using the lower bound to try
+ * to find the first key that is to the right of the boundary.
+ * Then we start scanning from this tuple (including the tuple itself)
+ * until we find the first key which is to the right of the boundary. Then
+ * we stop and do not include that key in the scan result.
+ */
+int
+Dbtux::cmpScanBound(const Frag& frag, const BoundPar boundPar)
+{
+ unsigned type = 4;
+ int ret = 0;
+ /*
+ No boundary means full scan, low boundary is to the right of all keys.
+ Thus we should always return -1. For upper bound we are to the right of
+ all keys, thus we should always return +1. We achieve this behaviour
+ by initialising return value to 0 and set type to 4.
+ */
+ const DescEnt& descEnt = getDescEnt(frag.m_descPage, frag.m_descOff);
+ ConstData data1 = boundPar.m_data1;
+ ConstData data2 = boundPar.m_data2;
+ // direction 0-lower 1-upper
+ const unsigned dir = boundPar.m_dir;
+ ndbrequire(dir <= 1);
+ // number of words of data left
+ unsigned len2 = boundPar.m_len2;
+ for (unsigned i = 0; i < boundPar.m_count1; i++) {
+ if (len2 < AttributeHeaderSize) {
+ jam();
+ return NdbSqlUtil::CmpUnknown;
+ }
+ len2 -= AttributeHeaderSize;
+ // get and skip bound type
+ type = data1[0];
+ data1 += 1;
+ ndbrequire(! data1.ah().isNULL());
+ if (! data2.ah().isNULL()) {
+ jam();
+ // current attribute
+ const unsigned index = data1.ah().getAttributeId();
+ const DescAttr& descAttr = descEnt.m_descAttr[index];
+ const unsigned typeId = descAttr.m_typeId;
+ ndbrequire(data2.ah().getAttributeId() == descAttr.m_primaryAttrId);
+ // full data size
+ const unsigned size1 = data1.ah().getDataSize();
+ ndbrequire(size1 != 0 && size1 == data2.ah().getDataSize());
+ const unsigned size2 = min(size1, len2);
+ len2 -= size2;
+ // compare
+ const Uint32* const p1 = &data1[AttributeHeaderSize];
+ const Uint32* const p2 = &data2[AttributeHeaderSize];
+ ret = NdbSqlUtil::cmp(typeId, p1, p2, size1, size2);
+ if (ret != 0) {
+ jam();
+ return ret;
+ }
+ } else {
+ jam();
+ /*
+ NULL is bigger than any bound, thus the boundary is always to the
+ left of NULL
+ */
+ return -1;
+ }
+ data1 += AttributeHeaderSize + data1.ah().getDataSize();
+ data2 += AttributeHeaderSize + data2.ah().getDataSize();
+ }
+ ndbassert(ret == 0);
+ if (dir == 0) {
+ jam();
+ /*
+ Looking for the lower bound. If strict lower bound then the boundary is
+ to the right of the compared key and otherwise (equal included in range)
+ then the boundary is to the left of the key.
+ */
+ if (type == 1) {
+ jam();
+ return +1;
+ }
+ return -1;
+ } else {
+ jam();
+ /*
+ Looking for the upper bound. If strict upper bound then the boundary is
+ to the left of all equal keys and otherwise (equal included in the
+ range) then the boundary is to the right of all equal keys.
+ */
+ if (type == 3) {
+ jam();
+ return -1;
+ }
+ return +1;
+ }
+}
+
diff --git a/ndb/src/kernel/blocks/dbtux/DbtuxDebug.cpp b/ndb/src/kernel/blocks/dbtux/DbtuxDebug.cpp
new file mode 100644
index 00000000000..c5d24205d1a
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/DbtuxDebug.cpp
@@ -0,0 +1,369 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUX_DEBUG_CPP
+#include "Dbtux.hpp"
+
+/*
+ * 12001 log file 0-close 1-open 2-append 3-append to signal log
+ * 12002 log flags 1-meta 2-maint 4-tree 8-scan
+ */
+void
+Dbtux::execDUMP_STATE_ORD(Signal* signal)
+{
+ jamEntry();
+#ifdef VM_TRACE
+ if (signal->theData[0] == DumpStateOrd::TuxLogToFile) {
+ unsigned flag = signal->theData[1];
+ const char* const tuxlog = "tux.log";
+ FILE* slFile = globalSignalLoggers.getOutputStream();
+ if (flag <= 3) {
+ if (debugFile != 0) {
+ if (debugFile != slFile)
+ fclose(debugFile);
+ debugFile = 0;
+ debugOut = *new NdbOut(*new NullOutputStream());
+ }
+ if (flag == 1)
+ debugFile = fopen(tuxlog, "w");
+ if (flag == 2)
+ debugFile = fopen(tuxlog, "a");
+ if (flag == 3)
+ debugFile = slFile;
+ if (debugFile != 0)
+ debugOut = *new NdbOut(*new FileOutputStream(debugFile));
+ }
+ return;
+ }
+ if (signal->theData[0] == DumpStateOrd::TuxSetLogFlags) {
+ debugFlags = signal->theData[1];
+ return;
+ }
+ if (signal->theData[0] == DumpStateOrd::TuxMetaDataJunk) {
+ // read table definition
+ Uint32 tableId = signal->theData[1];
+ Uint32 tableVersion = signal->theData[2];
+ int ret;
+ MetaData md(this);
+ MetaData::Table table;
+ MetaData::Attribute attribute;
+ infoEvent("md: read table %u %u", tableId, tableVersion);
+ if ((ret = md.lock(false)) < 0) {
+ infoEvent("md.lock error %d", ret);
+ return;
+ }
+ if ((ret = md.getTable(table, tableId, tableVersion)) < 0) {
+ infoEvent("md.getTable error %d", ret);
+ // lock is released by destructor
+ return;
+ }
+ infoEvent("md: %s type=%d attrs=%u", table.tableName, table.tableType, table.noOfAttributes);
+ for (Uint32 i = 0; i < table.noOfAttributes; i++) {
+ if ((ret = md.getAttribute(attribute, table, i)) < 0) {
+ infoEvent("mg.getAttribute %u error %d", i, ret);
+ // lock is released by destructor
+ return;
+ }
+ infoEvent("md: %d %s", attribute.attributeId, attribute.attributeName);
+ }
+ if ((ret = md.unlock(false)) < 0) {
+ infoEvent("md.unlock error %d", ret);
+ return;
+ }
+ return;
+ }
+#endif
+}
+
+#ifdef VM_TRACE
+
+void
+Dbtux::printTree(Signal* signal, Frag& frag, NdbOut& out)
+{
+ TreeHead& tree = frag.m_tree;
+ PrintPar par;
+ strcpy(par.m_path, ".");
+ par.m_side = 2;
+ par.m_parent = NullTupAddr;
+ printNode(signal, frag, out, tree.m_root, par);
+ out.m_out->flush();
+ if (! par.m_ok) {
+ if (debugFile == 0) {
+ signal->theData[0] = 12001;
+ signal->theData[1] = 1;
+ execDUMP_STATE_ORD(signal);
+ if (debugFile != 0) {
+ commitNodes(signal, frag, false);
+ printTree(signal, frag, debugOut);
+ }
+ }
+ ndbrequire(false);
+ }
+ commitNodes(signal, frag, false);
+}
+
+void
+Dbtux::printNode(Signal* signal, Frag& frag, NdbOut& out, TupAddr addr, PrintPar& par)
+{
+ if (addr == NullTupAddr) {
+ par.m_depth = 0;
+ return;
+ }
+ TreeHead& tree = frag.m_tree;
+ NodeHandlePtr nodePtr;
+ selectNode(signal, frag, nodePtr, addr, AccFull);
+ out << par.m_path << " " << *nodePtr.p << endl;
+ // check children
+ PrintPar cpar[2];
+ ndbrequire(strlen(par.m_path) + 1 < sizeof(par.m_path));
+ for (unsigned i = 0; i <= 1; i++) {
+ sprintf(cpar[i].m_path, "%s%c", par.m_path, "LR"[i]);
+ cpar[i].m_side = i;
+ cpar[i].m_depth = 0;
+ cpar[i].m_parent = addr;
+ printNode(signal, frag, out, nodePtr.p->getLink(i), cpar[i]);
+ if (! cpar[i].m_ok) {
+ par.m_ok = false;
+ }
+ }
+ // check child-parent links
+ if (nodePtr.p->getLink(2) != par.m_parent) {
+ par.m_ok = false;
+ out << par.m_path << " *** ";
+ out << "parent addr " << hex << nodePtr.p->getLink(2);
+ out << " should be " << hex << par.m_parent << endl;
+ }
+ if (nodePtr.p->getSide() != par.m_side) {
+ par.m_ok = false;
+ out << par.m_path << " *** ";
+ out << "side " << dec << nodePtr.p->getSide();
+ out << " should be " << dec << par.m_side << endl;
+ }
+ // check balance
+ const int balance = -cpar[0].m_depth + cpar[1].m_depth;
+ if (nodePtr.p->getBalance() != balance) {
+ par.m_ok = false;
+ out << par.m_path << " *** ";
+ out << "balance " << nodePtr.p->getBalance();
+ out << " should be " << balance << endl;
+ }
+ if (abs(nodePtr.p->getBalance()) > 1) {
+ par.m_ok = false;
+ out << par.m_path << " *** ";
+ out << "balance " << nodePtr.p->getBalance() << " is invalid" << endl;
+ }
+ // check occupancy
+ if (nodePtr.p->getOccup() > tree.m_maxOccup) {
+ par.m_ok = false;
+ out << par.m_path << " *** ";
+ out << "occupancy " << nodePtr.p->getOccup();
+ out << " greater than max " << tree.m_maxOccup << endl;
+ }
+ // check for occupancy of interior node
+ if (nodePtr.p->getChilds() == 2 && nodePtr.p->getOccup() < tree.m_minOccup) {
+ par.m_ok = false;
+ out << par.m_path << " *** ";
+ out << "occupancy " << nodePtr.p->getOccup() << " of interior node";
+ out << " less than min " << tree.m_minOccup << endl;
+ }
+ // check missed half-leaf/leaf merge
+ for (unsigned i = 0; i <= 1; i++) {
+ if (nodePtr.p->getLink(i) != NullTupAddr &&
+ nodePtr.p->getLink(1 - i) == NullTupAddr &&
+ nodePtr.p->getOccup() + cpar[i].m_occup <= tree.m_maxOccup) {
+ par.m_ok = false;
+ out << par.m_path << " *** ";
+ out << "missed merge with child " << i << endl;
+ }
+ }
+ // return values
+ par.m_depth = 1 + max(cpar[0].m_depth, cpar[1].m_depth);
+ par.m_occup = nodePtr.p->getOccup();
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbtux::TreeEnt& ent)
+{
+ out << dec << ent.m_fragBit;
+ out << "-" << hex << ent.m_tupAddr;
+ out << "-" << dec << ent.m_tupVersion;
+ return out;
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbtux::TreeNode& node)
+{
+ out << "[TreeNode " << hex << &node;
+ out << " [left " << hex << node.m_link[0] << "]";
+ out << " [right " << hex << node.m_link[1] << "]";
+ out << " [up " << hex << node.m_link[2] << "]";
+ out << " [side " << dec << node.m_side << "]";
+ out << " [occup " << dec << node.m_occup << "]";
+ out << " [balance " << dec << (int)node.m_balance << "]";
+ out << " [nodeScan " << hex << node.m_nodeScan << "]";
+ out << "]";
+ return out;
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbtux::TreeHead& tree)
+{
+ out << "[TreeHead " << hex << &tree;
+ out << " [nodeSize " << dec << tree.m_nodeSize << "]";
+ out << " [prefSize " << dec << tree.m_prefSize << "]";
+ out << " [minOccup " << dec << tree.m_minOccup << "]";
+ out << " [maxOccup " << dec << tree.m_maxOccup << "]";
+ out << " [AccHead " << dec << tree.getSize(Dbtux::AccHead) << "]";
+ out << " [AccPref " << dec << tree.getSize(Dbtux::AccPref) << "]";
+ out << " [AccFull " << dec << tree.getSize(Dbtux::AccFull) << "]";
+ out << " [root " << hex << tree.m_root << "]";
+ out << "]";
+ return out;
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbtux::TreePos& pos)
+{
+ out << "[TreePos " << hex << &pos;
+ out << " [addr " << hex << pos.m_addr << "]";
+ out << " [pos " << dec << pos.m_pos << "]";
+ out << " [match " << dec << pos.m_match << "]";
+ out << " [dir " << dec << pos.m_dir << "]";
+ out << " [ent " << pos.m_ent << "]";
+ out << "]";
+ return out;
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbtux::DescAttr& descAttr)
+{
+ out << "[DescAttr " << hex << &descAttr;
+ out << " [primaryAttrId " << dec << descAttr.m_primaryAttrId << "]";
+ out << " [typeId " << dec << descAttr.m_typeId << "]";
+ out << " [nullable " << dec << descAttr.m_nullable << "]";
+ out << "]";
+ return out;
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbtux::ScanOp& scan)
+{
+ out << "[ScanOp " << hex << &scan;
+ out << " [state " << dec << scan.m_state << "]";
+ out << " [lockwait " << dec << scan.m_lockwait << "]";
+ out << " [indexId " << dec << scan.m_indexId << "]";
+ out << " [fragId " << dec << scan.m_fragId << "]";
+ out << " [transId " << hex << scan.m_transId1 << " " << scan.m_transId2 << "]";
+ out << " [savePointId " << dec << scan.m_savePointId << "]";
+ out << " [accLockOp " << hex << scan.m_accLockOp << "]";
+ out << " [accLockOps";
+ for (unsigned i = 0; i < Dbtux::MaxAccLockOps; i++) {
+ if (scan.m_accLockOps[i] != RNIL)
+ out << " " << hex << scan.m_accLockOps[i];
+ }
+ out << "]";
+ out << " [readCommitted " << dec << scan.m_readCommitted << "]";
+ out << " [lockMode " << dec << scan.m_lockMode << "]";
+ out << " [keyInfo " << dec << scan.m_keyInfo << "]";
+ out << " [pos " << scan.m_scanPos << "]";
+ for (unsigned i = 0; i <= 1; i++) {
+ out << " [bound " << dec << i;
+ Dbtux::ScanBound& bound = *scan.m_bound[i];
+ Dbtux::ScanBoundIterator iter;
+ bound.first(iter);
+ for (unsigned j = 0; j < bound.getSize(); j++) {
+ out << " " << hex << *iter.data;
+ bound.next(iter);
+ }
+ out << "]";
+ }
+ out << "]";
+ return out;
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbtux::Index& index)
+{
+ out << "[Index " << hex << &index;
+ out << " [tableId " << dec << index.m_tableId << "]";
+ out << " [fragOff " << dec << index.m_fragOff << "]";
+ out << " [numFrags " << dec << index.m_numFrags << "]";
+ for (unsigned i = 0; i < index.m_numFrags; i++) {
+ out << " [frag " << dec << i << " ";
+ // dangerous and wrong
+ Dbtux* tux = (Dbtux*)globalData.getBlock(DBTUX);
+ const Dbtux::Frag& frag = *tux->c_fragPool.getPtr(index.m_fragPtrI[i]);
+ out << frag;
+ out << "]";
+ }
+ out << " [descPage " << hex << index.m_descPage << "]";
+ out << " [descOff " << dec << index.m_descOff << "]";
+ out << " [numAttrs " << dec << index.m_numAttrs << "]";
+ out << "]";
+ return out;
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbtux::Frag& frag)
+{
+ out << "[Frag " << hex << &frag;
+ out << " [tableId " << dec << frag.m_tableId << "]";
+ out << " [indexId " << dec << frag.m_indexId << "]";
+ out << " [fragOff " << dec << frag.m_fragOff << "]";
+ out << " [fragId " << dec << frag.m_fragId << "]";
+ out << " [descPage " << hex << frag.m_descPage << "]";
+ out << " [descOff " << dec << frag.m_descOff << "]";
+ out << " [numAttrs " << dec << frag.m_numAttrs << "]";
+ out << " [tree " << frag.m_tree << "]";
+ out << "]";
+ return out;
+}
+
+NdbOut&
+operator<<(NdbOut& out, const Dbtux::NodeHandle& node)
+{
+ const Dbtux::Frag& frag = node.m_frag;
+ const Dbtux::TreeHead& tree = frag.m_tree;
+ out << "[NodeHandle " << hex << &node;
+ out << " [addr " << hex << node.m_addr << "]";
+ out << " [acc " << dec << node.m_acc << "]";
+ out << " [flags " << hex << node.m_flags << "]";
+ out << " [node " << *node.m_node << "]";
+ if (node.m_acc >= Dbtux::AccPref) {
+ for (unsigned i = 0; i <= 1; i++) {
+ out << " [pref " << dec << i;
+ const Uint32* data = (const Uint32*)node.m_node + Dbtux::NodeHeadSize + i * tree.m_prefSize;
+ for (unsigned j = 0; j < node.m_frag.m_tree.m_prefSize; j++)
+ out << " " << hex << data[j];
+ out << "]";
+ }
+ out << " [entList";
+ unsigned numpos = node.m_node->m_occup;
+ if (node.m_acc < Dbtux::AccFull && numpos > 2) {
+ numpos = 2;
+ out << "(" << dec << numpos << ")";
+ }
+ const Uint32* data = (const Uint32*)node.m_node + Dbtux::NodeHeadSize + 2 * tree.m_prefSize;
+ const Dbtux::TreeEnt* entList = (const Dbtux::TreeEnt*)data;
+ for (unsigned pos = 0; pos < numpos; pos++)
+ out << " " << entList[pos];
+ out << "]";
+ }
+ out << "]";
+ return out;
+}
+
+#endif
diff --git a/ndb/src/kernel/blocks/dbtux/DbtuxGen.cpp b/ndb/src/kernel/blocks/dbtux/DbtuxGen.cpp
new file mode 100644
index 00000000000..32d66422d5c
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/DbtuxGen.cpp
@@ -0,0 +1,221 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUX_GEN_CPP
+#include "Dbtux.hpp"
+
+Dbtux::Dbtux(const Configuration& conf) :
+ SimulatedBlock(DBTUX, conf),
+ c_descPageList(RNIL),
+#ifdef VM_TRACE
+ debugFile(0),
+ debugOut(*new NullOutputStream()),
+ debugFlags(0),
+#endif
+ c_internalStartPhase(0),
+ c_typeOfStart(NodeState::ST_ILLEGAL_TYPE),
+ c_keyBuffer(0)
+{
+ BLOCK_CONSTRUCTOR(Dbtux);
+ // verify size assumptions (also when release-compiled)
+ ndbrequire(
+ (sizeof(DescHead) & 0x3) == 0 &&
+ (sizeof(DescAttr) & 0x3) == 0 &&
+ (sizeof(TreeEnt) & 0x3) == 0 &&
+ (sizeof(TreeNode) & 0x3) == 0
+ );
+ /*
+ * DbtuxGen.cpp
+ */
+ addRecSignal(GSN_CONTINUEB, &Dbtux::execCONTINUEB);
+ addRecSignal(GSN_STTOR, &Dbtux::execSTTOR);
+ addRecSignal(GSN_SIZEALT_REP, &Dbtux::execSIZEALT_REP);
+ /*
+ * DbtuxMeta.cpp
+ */
+ addRecSignal(GSN_TUXFRAGREQ, &Dbtux::execTUXFRAGREQ);
+ addRecSignal(GSN_TUX_ADD_ATTRREQ, &Dbtux::execTUX_ADD_ATTRREQ);
+ addRecSignal(GSN_ALTER_INDX_REQ, &Dbtux::execALTER_INDX_REQ);
+ addRecSignal(GSN_DROP_TAB_REQ, &Dbtux::execDROP_TAB_REQ);
+ /*
+ * DbtuxMaint.cpp
+ */
+ addRecSignal(GSN_TUX_MAINT_REQ, &Dbtux::execTUX_MAINT_REQ);
+ /*
+ * DbtuxScan.cpp
+ */
+ addRecSignal(GSN_ACC_SCANREQ, &Dbtux::execACC_SCANREQ);
+ addRecSignal(GSN_TUX_BOUND_INFO, &Dbtux::execTUX_BOUND_INFO);
+ addRecSignal(GSN_NEXT_SCANREQ, &Dbtux::execNEXT_SCANREQ);
+ addRecSignal(GSN_ACC_CHECK_SCAN, &Dbtux::execACC_CHECK_SCAN);
+ addRecSignal(GSN_ACCKEYCONF, &Dbtux::execACCKEYCONF);
+ addRecSignal(GSN_ACCKEYREF, &Dbtux::execACCKEYREF);
+ addRecSignal(GSN_ACC_ABORTCONF, &Dbtux::execACC_ABORTCONF);
+ /*
+ * DbtuxDebug.cpp
+ */
+ addRecSignal(GSN_DUMP_STATE_ORD, &Dbtux::execDUMP_STATE_ORD);
+}
+
+Dbtux::~Dbtux()
+{
+}
+
+void
+Dbtux::execCONTINUEB(Signal* signal)
+{
+ jamEntry();
+ const Uint32* data = signal->getDataPtr();
+ switch (data[0]) {
+ case TuxContinueB::DropIndex:
+ {
+ IndexPtr indexPtr;
+ c_indexPool.getPtr(indexPtr, data[1]);
+ dropIndex(signal, indexPtr, data[2], data[3]);
+ }
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+}
+
+/*
+ * STTOR is sent to one block at a time. In NDBCNTR it triggers
+ * NDB_STTOR to the "old" blocks. STTOR carries start phase (SP) and
+ * NDB_STTOR carries internal start phase (ISP).
+ *
+ * SP ISP activities
+ * 1 none
+ * 2 1
+ * 3 2 recover metadata, activate indexes
+ * 4 3 recover data
+ * 5 4-6
+ * 6 skip
+ * 7 skip
+ * 8 7 build non-logged indexes on SR
+ *
+ * DBTUX catches type of start (IS, SR, NR, INR) at SP 3 and updates
+ * internal start phase at SP 7. These are used to prevent index
+ * maintenance operations caused by redo log at SR.
+ */
+void
+Dbtux::execSTTOR(Signal* signal)
+{
+ jamEntry();
+ Uint32 startPhase = signal->theData[1];
+ switch (startPhase) {
+ case 1:
+ jam();
+ CLEAR_ERROR_INSERT_VALUE;
+ break;
+ case 3:
+ jam();
+ c_typeOfStart = signal->theData[7];
+ break;
+ case 7:
+ c_internalStartPhase = 6;
+ default:
+ jam();
+ break;
+ }
+ signal->theData[0] = 0; // garbage
+ signal->theData[1] = 0; // garbage
+ signal->theData[2] = 0; // garbage
+ signal->theData[3] = 1;
+ signal->theData[4] = 3; // for c_typeOfStart
+ signal->theData[5] = 7; // for c_internalStartPhase
+ signal->theData[6] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 7, JBB);
+}
+
+void
+Dbtux::execSIZEALT_REP(Signal* signal)
+{
+ jamEntry();
+ const Uint32* data = signal->getDataPtr();
+ BlockReference sender = data[TuxSizeAltReq::IND_BLOCK_REF];
+ const Uint32 nIndex = data[TuxSizeAltReq::IND_INDEX];
+ const Uint32 nFragment = data[TuxSizeAltReq::IND_FRAGMENT];
+ const Uint32 nAttribute = data[TuxSizeAltReq::IND_ATTRIBUTE];
+ const Uint32 nDescPage = (nIndex + nAttribute + DescPageSize - 1) / DescPageSize;
+ const Uint32 nScanOp = data[TuxSizeAltReq::IND_SCAN];
+ const Uint32 nScanBoundWords = nScanOp * ScanBoundSegmentSize * 4;
+ // allocate records
+ c_indexPool.setSize(nIndex);
+ c_fragPool.setSize(nFragment);
+ c_descPagePool.setSize(nDescPage);
+ c_fragOpPool.setSize(MaxIndexFragments);
+ c_nodeHandlePool.setSize(MaxNodeHandles);
+ c_scanOpPool.setSize(nScanOp);
+ c_scanBoundPool.setSize(nScanBoundWords);
+ /*
+ * Index id is physical array index. We seize and initialize all
+ * index records now. This assumes ArrayPool is an array.
+ */
+ IndexPtr indexPtr;
+ while (1) {
+ jam();
+ c_indexPool.seize(indexPtr);
+ if (indexPtr.i == RNIL) {
+ jam();
+ break;
+ }
+ new (indexPtr.p) Index();
+ }
+ // allocate buffers
+ c_keyBuffer = (Uint32*)allocRecord("c_keyBuffer", sizeof(Uint64), (MaxAttrDataSize + 1) >> 1);
+ // ack
+ sendSignal(sender, GSN_SIZEALT_ACK, signal, 1, JBB);
+}
+
+// utils
+
+void
+Dbtux::copyAttrs(Data dst, ConstData src, CopyPar& copyPar)
+{
+ CopyPar c = copyPar;
+ c.m_numitems = 0;
+ c.m_numwords = 0;
+ while (c.m_numitems < c.m_items) {
+ jam();
+ if (c.m_headers) {
+ unsigned i = 0;
+ while (i < AttributeHeaderSize) {
+ if (c.m_numwords >= c.m_maxwords) {
+ copyPar = c;
+ return;
+ }
+ dst[c.m_numwords++] = src[i++];
+ }
+ }
+ unsigned size = src.ah().getDataSize();
+ src += AttributeHeaderSize;
+ unsigned i = 0;
+ while (i < size) {
+ if (c.m_numwords >= c.m_maxwords) {
+ copyPar = c;
+ return;
+ }
+ dst[c.m_numwords++] = src[i++];
+ }
+ src += size;
+ c.m_numitems++;
+ }
+ copyPar = c;
+}
+
+BLOCK_FUNCTIONS(Dbtux);
diff --git a/ndb/src/kernel/blocks/dbtux/DbtuxMaint.cpp b/ndb/src/kernel/blocks/dbtux/DbtuxMaint.cpp
new file mode 100644
index 00000000000..a6b2485067c
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/DbtuxMaint.cpp
@@ -0,0 +1,369 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUX_MAINT_CPP
+#include "Dbtux.hpp"
+
+/*
+ * Maintain index.
+ */
+
+void
+Dbtux::execTUX_MAINT_REQ(Signal* signal)
+{
+ jamEntry();
+ TuxMaintReq* const sig = (TuxMaintReq*)signal->getDataPtrSend();
+ // ignore requests from redo log
+ if (c_internalStartPhase < 6 &&
+ c_typeOfStart != NodeState::ST_NODE_RESTART &&
+ c_typeOfStart != NodeState::ST_INITIAL_NODE_RESTART) {
+ jam();
+#ifdef VM_TRACE
+ if (debugFlags & DebugMaint) {
+ debugOut << "opInfo=" << hex << sig->opInfo;
+ debugOut << " tableId=" << dec << sig->tableId;
+ debugOut << " indexId=" << dec << sig->indexId;
+ debugOut << " fragId=" << dec << sig->fragId;
+ debugOut << " tupAddr=" << hex << sig->tupAddr;
+ debugOut << " tupVersion=" << dec << sig->tupVersion;
+ debugOut << " -- ignored at ISP=" << dec << c_internalStartPhase;
+ debugOut << " TOS=" << dec << c_typeOfStart;
+ debugOut << endl;
+ }
+#endif
+ sig->errorCode = 0;
+ return;
+ }
+ TuxMaintReq reqCopy = *sig;
+ TuxMaintReq* const req = &reqCopy;
+ const Uint32 opCode = req->opInfo & 0xFF;
+ const Uint32 opFlag = req->opInfo >> 8;
+ // get the index
+ IndexPtr indexPtr;
+ c_indexPool.getPtr(indexPtr, req->indexId);
+ ndbrequire(indexPtr.p->m_tableId == req->tableId);
+ // get base fragment id and extra bits
+ const Uint32 fragOff = indexPtr.p->m_fragOff;
+ const Uint32 fragId = req->fragId & ((1 << fragOff) - 1);
+ const Uint32 fragBit = req->fragId >> fragOff;
+ // get the fragment
+ FragPtr fragPtr;
+ fragPtr.i = RNIL;
+ for (unsigned i = 0; i < indexPtr.p->m_numFrags; i++) {
+ jam();
+ if (indexPtr.p->m_fragId[i] == fragId) {
+ jam();
+ c_fragPool.getPtr(fragPtr, indexPtr.p->m_fragPtrI[i]);
+ break;
+ }
+ }
+ ndbrequire(fragPtr.i != RNIL);
+ Frag& frag = *fragPtr.p;
+ ndbrequire(frag.m_nodeList == RNIL);
+ // set up index entry
+ TreeEnt ent;
+ ent.m_tupAddr = req->tupAddr;
+ ent.m_tupVersion = req->tupVersion;
+ ent.m_fragBit = fragBit;
+ // read search key
+ ReadPar readPar;
+ readPar.m_ent = ent;
+ readPar.m_first = 0;
+ readPar.m_count = frag.m_numAttrs;
+ // output goes here
+ readPar.m_data = c_keyBuffer;
+ tupReadAttrs(signal, frag, readPar);
+ // check if all keys are null
+ {
+ bool allNull = true;
+ ConstData data = readPar.m_data;
+ for (unsigned i = 0; i < frag.m_numAttrs; i++) {
+ if (! data.ah().isNULL()) {
+ jam();
+ allNull = false;
+ break;
+ }
+ data += AttributeHeaderSize + data.ah().getDataSize();
+ }
+ if (allNull) {
+ jam();
+ req->errorCode = 0;
+ *sig = *req;
+ return;
+ }
+ }
+ // find position in tree
+ SearchPar searchPar;
+ searchPar.m_data = c_keyBuffer;
+ searchPar.m_ent = ent;
+ TreePos treePos;
+#ifdef VM_TRACE
+ if (debugFlags & DebugMaint) {
+ debugOut << "opCode=" << dec << opCode;
+ debugOut << " opFlag=" << dec << opFlag;
+ debugOut << " tableId=" << dec << req->tableId;
+ debugOut << " indexId=" << dec << req->indexId;
+ debugOut << " fragId=" << dec << req->fragId;
+ debugOut << " entry=" << ent;
+ debugOut << endl;
+ }
+#endif
+ treeSearch(signal, frag, searchPar, treePos);
+#ifdef VM_TRACE
+ if (debugFlags & DebugMaint) {
+ debugOut << treePos << endl;
+ }
+#endif
+ // do the operation
+ req->errorCode = 0;
+ switch (opCode) {
+ case TuxMaintReq::OpAdd:
+ jam();
+ if (treePos.m_match) {
+ jam();
+ // there is no "Building" state so this will have to do
+ if (indexPtr.p->m_state == Index::Online) {
+ jam();
+ req->errorCode = TuxMaintReq::SearchError;
+ }
+ break;
+ }
+ /*
+ * At most one new node is inserted in the operation. We keep one
+ * free node pre-allocated so the operation cannot fail. This also
+ * gives a real TupAddr for links to the new node.
+ */
+ if (frag.m_nodeFree == RNIL) {
+ jam();
+ preallocNode(signal, frag, req->errorCode);
+ if (req->errorCode != 0) {
+ jam();
+ break;
+ }
+ ndbrequire(frag.m_nodeFree != RNIL);
+ }
+ treeAdd(signal, frag, treePos, ent);
+ break;
+ case TuxMaintReq::OpRemove:
+ jam();
+ if (! treePos.m_match) {
+ jam();
+ // there is no "Building" state so this will have to do
+ if (indexPtr.p->m_state == Index::Online) {
+ jam();
+ req->errorCode = TuxMaintReq::SearchError;
+ }
+ break;
+ }
+ treeRemove(signal, frag, treePos);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+ // commit and release nodes
+ commitNodes(signal, frag, req->errorCode == 0);
+#ifdef VM_TRACE
+ if (debugFlags & DebugTree) {
+ printTree(signal, frag, debugOut);
+ }
+#endif
+ // copy back
+ *sig = *req;
+}
+
+/*
+ * Read index key attributes from TUP. If buffer is provided the data
+ * is copied to it. Otherwise pointer is set to signal data.
+ */
+void
+Dbtux::tupReadAttrs(Signal* signal, const Frag& frag, ReadPar& readPar)
+{
+ // define the direct signal
+ const TreeEnt ent = readPar.m_ent;
+ TupReadAttrs* const req = (TupReadAttrs*)signal->getDataPtrSend();
+ req->errorCode = RNIL;
+ req->requestInfo = 0;
+ req->tableId = frag.m_tableId;
+ req->fragId = frag.m_fragId | (ent.m_fragBit << frag.m_fragOff);
+ req->fragPtrI = RNIL;
+ req->tupAddr = ent.m_tupAddr;
+ req->tupVersion = ent.m_tupVersion;
+ req->pageId = RNIL;
+ req->pageOffset = 0;
+ req->bufferId = 0;
+ // add count and list of attribute ids
+ Data data = (Uint32*)req + TupReadAttrs::SignalLength;
+ data[0] = readPar.m_count;
+ data += 1;
+ const DescEnt& descEnt = getDescEnt(frag.m_descPage, frag.m_descOff);
+ for (Uint32 i = 0; i < readPar.m_count; i++) {
+ jam();
+ const DescAttr& descAttr = descEnt.m_descAttr[readPar.m_first + i];
+ data.ah() = AttributeHeader(descAttr.m_primaryAttrId, 0);
+ data += 1;
+ }
+ // execute
+ EXECUTE_DIRECT(DBTUP, GSN_TUP_READ_ATTRS, signal, TupReadAttrs::SignalLength);
+ jamEntry();
+ ndbrequire(req->errorCode == 0);
+ // data is at output
+ if (readPar.m_data == 0) {
+ readPar.m_data = data;
+ } else {
+ jam();
+ CopyPar copyPar;
+ copyPar.m_items = readPar.m_count;
+ copyPar.m_headers = true;
+ copyAttrs(readPar.m_data, data, copyPar);
+ }
+}
+
+/*
+ * Read primary keys. Copy the data without attribute headers into the
+ * given buffer. Number of words is returned in ReadPar argument.
+ */
+void
+Dbtux::tupReadKeys(Signal* signal, const Frag& frag, ReadPar& readPar)
+{
+ // define the direct signal
+ const TreeEnt ent = readPar.m_ent;
+ TupReadAttrs* const req = (TupReadAttrs*)signal->getDataPtrSend();
+ req->errorCode = RNIL;
+ req->requestInfo = TupReadAttrs::ReadKeys;
+ req->tableId = frag.m_tableId;
+ req->fragId = frag.m_fragId | (ent.m_fragBit << frag.m_fragOff);
+ req->fragPtrI = RNIL;
+ req->tupAddr = ent.m_tupAddr;
+ req->tupVersion = RNIL; // not used
+ req->pageId = RNIL;
+ req->pageOffset = 0;
+ req->bufferId = 0;
+ // execute
+ EXECUTE_DIRECT(DBTUP, GSN_TUP_READ_ATTRS, signal, TupReadAttrs::SignalLength);
+ jamEntry();
+ ndbrequire(req->errorCode == 0);
+ // copy out in special format
+ ConstData data = (Uint32*)req + TupReadAttrs::SignalLength;
+ const Uint32 numKeys = data[0];
+ data += 1 + numKeys;
+ // copy out without headers
+ ndbrequire(readPar.m_data != 0);
+ CopyPar copyPar;
+ copyPar.m_items = numKeys;
+ copyPar.m_headers = false;
+ copyAttrs(readPar.m_data, data, copyPar);
+ // return counts
+ readPar.m_count = numKeys;
+ readPar.m_size = copyPar.m_numwords;
+}
+
+/*
+ * Operate on index node tuple in TUP. The data is copied between node
+ * cache and index storage via signal data.
+ */
+void
+Dbtux::tupStoreTh(Signal* signal, const Frag& frag, NodeHandlePtr nodePtr, StorePar storePar)
+{
+ const TreeHead& tree = frag.m_tree;
+ // define the direct signal
+ TupStoreTh* req = (TupStoreTh*)signal->getDataPtrSend();
+ req->errorCode = RNIL;
+ req->tableId = frag.m_indexId;
+ req->fragId = frag.m_fragId;
+ req->fragPtrI = RNIL;
+ req->tupAddr = nodePtr.p->m_addr;
+ req->tupVersion = 0;
+ req->pageId = nodePtr.p->m_loc.m_pageId;
+ req->pageOffset = nodePtr.p->m_loc.m_pageOffset;
+ req->bufferId = 0;
+ req->opCode = storePar.m_opCode;
+ ndbrequire(storePar.m_offset + storePar.m_size <= tree.m_nodeSize);
+ req->dataOffset = storePar.m_offset;
+ req->dataSize = storePar.m_size;
+ // the node cache
+ ndbrequire(nodePtr.p->m_node != 0);
+ // the buffer in signal data
+ Uint32* const buffer = (Uint32*)req + TupStoreTh::SignalLength;
+ // copy in data
+ switch (storePar.m_opCode) {
+ case TupStoreTh::OpRead:
+ jam();
+ #ifdef VM_TRACE
+ {
+ Uint32* dst = buffer + storePar.m_offset;
+ memset(dst, 0xa9, storePar.m_size << 2);
+ }
+ #endif
+ break;
+ case TupStoreTh::OpInsert:
+ jam();
+ // fallthru
+ case TupStoreTh::OpUpdate:
+ jam();
+ // copy from cache to signal data
+ {
+ Uint32* dst = buffer + storePar.m_offset;
+ const Uint32* src = (const Uint32*)nodePtr.p->m_node + storePar.m_offset;
+ memcpy(dst, src, storePar.m_size << 2);
+ }
+ break;
+ case TupStoreTh::OpDelete:
+ jam();
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+ // execute
+ EXECUTE_DIRECT(DBTUP, GSN_TUP_STORE_TH, signal, TupStoreTh::SignalLength);
+ jamEntry();
+ if (req->errorCode != 0) {
+ jam();
+ storePar.m_errorCode = req->errorCode;
+ return;
+ }
+ ndbrequire(req->errorCode == 0);
+ // copy out data
+ switch (storePar.m_opCode) {
+ case TupStoreTh::OpRead:
+ jam();
+ {
+ Uint32* dst = (Uint32*)nodePtr.p->m_node + storePar.m_offset;
+ const Uint32* src = (const Uint32*)buffer + storePar.m_offset;
+ memcpy(dst, src, storePar.m_size << 2);
+ }
+ // fallthru
+ case TupStoreTh::OpInsert:
+ jam();
+ // fallthru
+ case TupStoreTh::OpUpdate:
+ jam();
+ nodePtr.p->m_addr = req->tupAddr;
+ nodePtr.p->m_loc.m_pageId = req->pageId;
+ nodePtr.p->m_loc.m_pageOffset = req->pageOffset;
+ break;
+ case TupStoreTh::OpDelete:
+ jam();
+ nodePtr.p->m_addr = NullTupAddr;
+ nodePtr.p->m_loc.m_pageId = RNIL;
+ nodePtr.p->m_loc.m_pageOffset = 0;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+}
diff --git a/ndb/src/kernel/blocks/dbtux/DbtuxMeta.cpp b/ndb/src/kernel/blocks/dbtux/DbtuxMeta.cpp
new file mode 100644
index 00000000000..2ffd599429c
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/DbtuxMeta.cpp
@@ -0,0 +1,427 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUX_META_CPP
+#include "Dbtux.hpp"
+
+/*
+ * Create index.
+ *
+ * For historical reasons it looks like we are adding random fragments
+ * and attributes to existing index. In fact all fragments must be
+ * created at one time and they have identical attributes.
+ */
+
+void
+Dbtux::execTUXFRAGREQ(Signal* signal)
+{
+ jamEntry();
+ const TuxFragReq reqCopy = *(const TuxFragReq*)signal->getDataPtr();
+ const TuxFragReq* const req = &reqCopy;
+ IndexPtr indexPtr;
+ indexPtr.i = RNIL;
+ FragOpPtr fragOpPtr;
+ fragOpPtr.i = RNIL;
+ TuxFragRef::ErrorCode errorCode = TuxFragRef::NoError;
+ do {
+ // get the index record
+ if (req->tableId >= c_indexPool.getSize()) {
+ jam();
+ errorCode = TuxFragRef::InvalidRequest;
+ break;
+ }
+ c_indexPool.getPtr(indexPtr, req->tableId);
+ if (indexPtr.p->m_state != Index::NotDefined &&
+ indexPtr.p->m_state != Index::Defining) {
+ jam();
+ errorCode = TuxFragRef::InvalidRequest;
+ indexPtr.i = RNIL; // leave alone
+ break;
+ }
+ // get new operation record
+ c_fragOpPool.seize(fragOpPtr);
+ if (fragOpPtr.i == RNIL) {
+ jam();
+ errorCode = TuxFragRef::NoFreeFragmentOper;
+ break;
+ }
+ new (fragOpPtr.p) FragOp();
+ fragOpPtr.p->m_userPtr = req->userPtr;
+ fragOpPtr.p->m_userRef = req->userRef;
+ fragOpPtr.p->m_indexId = req->tableId;
+ fragOpPtr.p->m_fragId = req->fragId;
+ fragOpPtr.p->m_fragNo = indexPtr.p->m_numFrags;
+ fragOpPtr.p->m_numAttrsRecvd = 0;
+ // check if index has place for more fragments
+ if (indexPtr.p->m_numFrags == MaxIndexFragments) {
+ jam();
+ errorCode = TuxFragRef::NoFreeIndexFragment;
+ break;
+ }
+ // seize new fragment record
+ FragPtr fragPtr;
+ c_fragPool.seize(fragPtr);
+ if (fragPtr.i == RNIL) {
+ jam();
+ errorCode = TuxFragRef::NoFreeFragment;
+ break;
+ }
+ new (fragPtr.p) Frag(c_scanOpPool);
+ fragPtr.p->m_tableId = req->primaryTableId;
+ fragPtr.p->m_indexId = req->tableId;
+ fragPtr.p->m_fragOff = req->fragOff;
+ fragPtr.p->m_fragId = req->fragId;
+ fragPtr.p->m_numAttrs = req->noOfAttr;
+ // add the fragment to the index
+ indexPtr.p->m_fragId[indexPtr.p->m_numFrags] = req->fragId;
+ indexPtr.p->m_fragPtrI[indexPtr.p->m_numFrags] = fragPtr.i;
+ indexPtr.p->m_numFrags++;
+ // save under operation
+ fragOpPtr.p->m_fragPtrI = fragPtr.i;
+ // prepare to receive attributes
+ if (fragOpPtr.p->m_fragNo == 0) {
+ jam();
+ // receiving first fragment
+ ndbrequire(
+ indexPtr.p->m_state == Index::NotDefined &&
+ DictTabInfo::isOrderedIndex(req->tableType) &&
+ req->noOfAttr > 0 &&
+ req->noOfAttr <= MaxIndexAttributes &&
+ indexPtr.p->m_descPage == RNIL);
+ indexPtr.p->m_state = Index::Defining;
+ indexPtr.p->m_tableType = (DictTabInfo::TableType)req->tableType;
+ indexPtr.p->m_tableId = req->primaryTableId;
+ indexPtr.p->m_fragOff = req->fragOff;
+ indexPtr.p->m_numAttrs = req->noOfAttr;
+ // allocate attribute descriptors
+ if (! allocDescEnt(indexPtr)) {
+ jam();
+ errorCode = TuxFragRef::NoFreeAttributes;
+ break;
+ }
+ } else {
+ // receiving subsequent fragment
+ jam();
+ ndbrequire(
+ indexPtr.p->m_state == Index::Defining &&
+ indexPtr.p->m_tableType == (DictTabInfo::TableType)req->tableType &&
+ indexPtr.p->m_tableId == req->primaryTableId &&
+ indexPtr.p->m_fragOff == req->fragOff &&
+ indexPtr.p->m_numAttrs == req->noOfAttr);
+ }
+ // copy metadata address to each fragment
+ fragPtr.p->m_descPage = indexPtr.p->m_descPage;
+ fragPtr.p->m_descOff = indexPtr.p->m_descOff;
+#ifdef VM_TRACE
+ if (debugFlags & DebugMeta) {
+ debugOut << "Add frag " << fragPtr.i << " " << *fragPtr.p << endl;
+ }
+#endif
+ // success
+ TuxFragConf* const conf = (TuxFragConf*)signal->getDataPtrSend();
+ conf->userPtr = req->userPtr;
+ conf->tuxConnectPtr = fragOpPtr.i;
+ conf->fragPtr = fragPtr.i;
+ conf->fragId = fragPtr.p->m_fragId;
+ sendSignal(req->userRef, GSN_TUXFRAGCONF,
+ signal, TuxFragConf::SignalLength, JBB);
+ return;
+ } while (0);
+ // error
+ TuxFragRef* const ref = (TuxFragRef*)signal->getDataPtrSend();
+ ref->userPtr = req->userPtr;
+ ref->errorCode = errorCode;
+ sendSignal(req->userRef, GSN_TUXFRAGREF,
+ signal, TuxFragRef::SignalLength, JBB);
+ if (fragOpPtr.i != RNIL)
+ c_fragOpPool.release(fragOpPtr);
+ if (indexPtr.i != RNIL)
+ dropIndex(signal, indexPtr, 0, 0);
+}
+
+void
+Dbtux::execTUX_ADD_ATTRREQ(Signal* signal)
+{
+ jamEntry();
+ const TuxAddAttrReq reqCopy = *(const TuxAddAttrReq*)signal->getDataPtr();
+ const TuxAddAttrReq* const req = &reqCopy;
+ // get the records
+ FragOpPtr fragOpPtr;
+ IndexPtr indexPtr;
+ FragPtr fragPtr;
+ c_fragOpPool.getPtr(fragOpPtr, req->tuxConnectPtr);
+ c_indexPool.getPtr(indexPtr, fragOpPtr.p->m_indexId);
+ c_fragPool.getPtr(fragPtr, fragOpPtr.p->m_fragPtrI);
+ TuxAddAttrRef::ErrorCode errorCode = TuxAddAttrRef::NoError;
+ do {
+ // expected attribute id
+ const unsigned attrId = fragOpPtr.p->m_numAttrsRecvd++;
+ ndbrequire(
+ indexPtr.p->m_state == Index::Defining &&
+ attrId < indexPtr.p->m_numAttrs &&
+ attrId == req->attrId);
+ // define the attribute
+ DescEnt& descEnt = getDescEnt(indexPtr.p->m_descPage, indexPtr.p->m_descOff);
+ DescAttr& descAttr = descEnt.m_descAttr[attrId];
+ descAttr.m_primaryAttrId = req->primaryAttrId;
+ descAttr.m_typeId = req->extTypeInfo & 0xFF;
+ descAttr.m_nullable = AttributeDescriptor::getNullable(req->attrDescriptor);
+ descAttr.pad1 = 0;
+#ifdef VM_TRACE
+ if (debugFlags & DebugMeta) {
+ debugOut << "Add frag " << fragPtr.i << " attr " << attrId << " " << descAttr << endl;
+ }
+#endif
+ // check if type is valid and has a comparison method
+ const NdbSqlUtil::Type& type = NdbSqlUtil::type(descAttr.m_typeId);
+ if (type.m_typeId == NdbSqlUtil::Type::Undefined ||
+ type.m_cmp == 0) {
+ jam();
+ errorCode = TuxAddAttrRef::InvalidAttributeType;
+ break;
+ }
+ if (indexPtr.p->m_numAttrs == fragOpPtr.p->m_numAttrsRecvd) {
+ jam();
+ // initialize tree header
+ TreeHead& tree = fragPtr.p->m_tree;
+ // make these configurable later
+ tree.m_nodeSize = MAX_TTREE_NODE_SIZE;
+ tree.m_prefSize = MAX_TTREE_PREF_SIZE;
+#ifdef dbtux_min_occup_less_max_occup
+ const unsigned maxSlack = MAX_TTREE_NODE_SLACK;
+#else
+ const unsigned maxSlack = 0;
+#endif
+ // size up to and including first 2 entries
+ const unsigned pref = tree.getSize(AccPref);
+ if (! (pref <= tree.m_nodeSize)) {
+ jam();
+ errorCode = TuxAddAttrRef::InvalidNodeSize;
+ break;
+ }
+ const unsigned slots = (tree.m_nodeSize - pref) / TreeEntSize;
+ // leave out work space entry
+ tree.m_maxOccup = 2 + slots - 1;
+ // min occupancy of interior node must be at least 2
+ if (! (2 + maxSlack <= tree.m_maxOccup)) {
+ jam();
+ errorCode = TuxAddAttrRef::InvalidNodeSize;
+ break;
+ }
+ tree.m_minOccup = tree.m_maxOccup - maxSlack;
+ // root node does not exist
+ tree.m_root = NullTupAddr;
+ // fragment is defined
+ c_fragOpPool.release(fragOpPtr);
+ }
+ // success
+ TuxAddAttrConf* conf = (TuxAddAttrConf*)signal->getDataPtrSend();
+ conf->userPtr = fragOpPtr.p->m_userPtr;
+ sendSignal(fragOpPtr.p->m_userRef, GSN_TUX_ADD_ATTRCONF,
+ signal, TuxAddAttrConf::SignalLength, JBB);
+ return;
+ } while (0);
+ // error
+ TuxAddAttrRef* ref = (TuxAddAttrRef*)signal->getDataPtrSend();
+ ref->userPtr = fragOpPtr.p->m_userPtr;
+ ref->errorCode = errorCode;
+ sendSignal(fragOpPtr.p->m_userRef, GSN_TUX_ADD_ATTRREF,
+ signal, TuxAddAttrRef::SignalLength, JBB);
+ c_fragOpPool.release(fragOpPtr);
+ dropIndex(signal, indexPtr, 0, 0);
+}
+
+/*
+ * Set index online. Currently at system restart this arrives before
+ * build and is therefore not correct.
+ */
+void
+Dbtux::execALTER_INDX_REQ(Signal* signal)
+{
+ jamEntry();
+ const AlterIndxReq reqCopy = *(const AlterIndxReq*)signal->getDataPtr();
+ const AlterIndxReq* const req = &reqCopy;
+ // set index online after build
+ IndexPtr indexPtr;
+ c_indexPool.getPtr(indexPtr, req->getIndexId());
+ indexPtr.p->m_state = Index::Online;
+#ifdef VM_TRACE
+ if (debugFlags & DebugMeta) {
+ debugOut << "Online index " << indexPtr.i << " " << *indexPtr.p << endl;
+ }
+#endif
+ // success
+ AlterIndxConf* const conf = (AlterIndxConf*)signal->getDataPtrSend();
+ conf->setUserRef(reference());
+ conf->setConnectionPtr(req->getConnectionPtr());
+ conf->setRequestType(req->getRequestType());
+ conf->setTableId(req->getTableId());
+ conf->setIndexId(req->getIndexId());
+ conf->setIndexVersion(req->getIndexVersion());
+ sendSignal(req->getUserRef(), GSN_ALTER_INDX_CONF,
+ signal, AlterIndxConf::SignalLength, JBB);
+}
+
+/*
+ * Drop index.
+ *
+ * Uses same DROP_TAB_REQ signal as normal tables.
+ */
+
+void
+Dbtux::execDROP_TAB_REQ(Signal* signal)
+{
+ jamEntry();
+ const DropTabReq reqCopy = *(const DropTabReq*)signal->getDataPtr();
+ const DropTabReq* const req = &reqCopy;
+ IndexPtr indexPtr;
+ c_indexPool.getPtr(indexPtr, req->tableId);
+ // drop works regardless of index state
+#ifdef VM_TRACE
+ if (debugFlags & DebugMeta) {
+ debugOut << "Drop index " << indexPtr.i << " " << *indexPtr.p << endl;
+ }
+#endif
+ ndbrequire(req->senderRef != 0);
+ dropIndex(signal, indexPtr, req->senderRef, req->senderData);
+}
+
+void
+Dbtux::dropIndex(Signal* signal, IndexPtr indexPtr, Uint32 senderRef, Uint32 senderData)
+{
+ jam();
+ indexPtr.p->m_state = Index::Dropping;
+ // drop one fragment at a time
+ if (indexPtr.p->m_numFrags > 0) {
+ jam();
+ unsigned i = --indexPtr.p->m_numFrags;
+ FragPtr fragPtr;
+ c_fragPool.getPtr(fragPtr, indexPtr.p->m_fragPtrI[i]);
+ Frag& frag = *fragPtr.p;
+ ndbrequire(frag.m_nodeList == RNIL);
+ if (frag.m_nodeFree != RNIL) {
+ c_nodeHandlePool.release(frag.m_nodeFree);
+ frag.m_nodeFree = RNIL;
+ }
+ c_fragPool.release(fragPtr);
+ // the real time break is not used for anything currently
+ signal->theData[0] = TuxContinueB::DropIndex;
+ signal->theData[1] = indexPtr.i;
+ signal->theData[2] = senderRef;
+ signal->theData[3] = senderData;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 4, JBB);
+ return;
+ }
+ // drop attributes
+ if (indexPtr.p->m_descPage != RNIL) {
+ jam();
+ freeDescEnt(indexPtr);
+ indexPtr.p->m_descPage = RNIL;
+ }
+ if (senderRef != 0) {
+ jam();
+ // reply to sender
+ DropTabConf* const conf = (DropTabConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ conf->tableId = indexPtr.i;
+ sendSignal(senderRef, GSN_DROP_TAB_CONF,
+ signal, DropTabConf::SignalLength, JBB);
+ }
+ new (indexPtr.p) Index();
+}
+
+/*
+ * Subroutines.
+ */
+
+bool
+Dbtux::allocDescEnt(IndexPtr indexPtr)
+{
+ jam();
+ const unsigned size = DescHeadSize + indexPtr.p->m_numAttrs * DescAttrSize;
+ DescPagePtr pagePtr;
+ pagePtr.i = c_descPageList;
+ while (pagePtr.i != RNIL) {
+ jam();
+ c_descPagePool.getPtr(pagePtr);
+ if (pagePtr.p->m_numFree >= size) {
+ jam();
+ break;
+ }
+ pagePtr.i = pagePtr.p->m_nextPage;
+ }
+ if (pagePtr.i == RNIL) {
+ jam();
+ if (! c_descPagePool.seize(pagePtr)) {
+ jam();
+ return false;
+ }
+ new (pagePtr.p) DescPage();
+ // add in front of list
+ pagePtr.p->m_nextPage = c_descPageList;
+ c_descPageList = pagePtr.i;
+ pagePtr.p->m_numFree = DescPageSize;
+ }
+ ndbrequire(pagePtr.p->m_numFree >= size);
+ indexPtr.p->m_descPage = pagePtr.i;
+ indexPtr.p->m_descOff = DescPageSize - pagePtr.p->m_numFree;
+ pagePtr.p->m_numFree -= size;
+ DescEnt& descEnt = getDescEnt(indexPtr.p->m_descPage, indexPtr.p->m_descOff);
+ descEnt.m_descHead.m_indexId = indexPtr.i;
+ descEnt.m_descHead.pad1 = 0;
+ return true;
+}
+
+void
+Dbtux::freeDescEnt(IndexPtr indexPtr)
+{
+ DescPagePtr pagePtr;
+ c_descPagePool.getPtr(pagePtr, indexPtr.p->m_descPage);
+ Uint32* const data = pagePtr.p->m_data;
+ const unsigned size = DescHeadSize + indexPtr.p->m_numAttrs * DescAttrSize;
+ unsigned off = indexPtr.p->m_descOff;
+ // move the gap to the free area at the top
+ while (off + size < DescPageSize - pagePtr.p->m_numFree) {
+ jam();
+ // next entry to move over the gap
+ DescEnt& descEnt2 = *(DescEnt*)&data[off + size];
+ Uint32 indexId2 = descEnt2.m_descHead.m_indexId;
+ Index& index2 = *c_indexPool.getPtr(indexId2);
+ unsigned size2 = DescHeadSize + index2.m_numAttrs * DescAttrSize;
+ ndbrequire(
+ index2.m_descPage == pagePtr.i &&
+ index2.m_descOff == off + size);
+ // move the entry (overlapping copy if size < size2)
+ for (unsigned i = 0; i < size2; i++) {
+ jam();
+ data[off + i] = data[off + size + i];
+ }
+ off += size2;
+ // adjust page offset in index and all fragments
+ index2.m_descOff -= size;
+ for (unsigned i = 0; i < index2.m_numFrags; i++) {
+ jam();
+ Frag& frag2 = *c_fragPool.getPtr(index2.m_fragPtrI[i]);
+ frag2.m_descOff -= size;
+ ndbrequire(
+ frag2.m_descPage == index2.m_descPage &&
+ frag2.m_descOff == index2.m_descOff);
+ }
+ }
+ ndbrequire(off + size == DescPageSize - pagePtr.p->m_numFree);
+ pagePtr.p->m_numFree += size;
+}
diff --git a/ndb/src/kernel/blocks/dbtux/DbtuxNode.cpp b/ndb/src/kernel/blocks/dbtux/DbtuxNode.cpp
new file mode 100644
index 00000000000..6733a87da97
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/DbtuxNode.cpp
@@ -0,0 +1,633 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUX_NODE_CPP
+#include "Dbtux.hpp"
+
+/*
+ * Node handles.
+ *
+ * We use the "cache" implementation. Node operations are done on
+ * cached copies. Index memory is updated at the end of the operation.
+ * At most one node is inserted and it is always pre-allocated.
+ *
+ * An alternative "pointer" implementation which writes directly into
+ * index memory is planned for later.
+ */
+
+// Dbtux
+
+void
+Dbtux::seizeNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr)
+{
+ if (! c_nodeHandlePool.seize(nodePtr)) {
+ jam();
+ return;
+ }
+ new (nodePtr.p) NodeHandle(*this, frag);
+ nodePtr.p->m_next = frag.m_nodeList;
+ frag.m_nodeList = nodePtr.i;
+ // node cache used always
+ nodePtr.p->m_node = (TreeNode*)nodePtr.p->m_cache;
+ new (nodePtr.p->m_node) TreeNode();
+#ifdef VM_TRACE
+ TreeHead& tree = frag.m_tree;
+ TreeNode* node = nodePtr.p->m_node;
+ memset(tree.getPref(node, 0), 0xa2, tree.m_prefSize << 2);
+ memset(tree.getPref(node, 1), 0xa2, tree.m_prefSize << 2);
+ TreeEnt* entList = tree.getEntList(node);
+ memset(entList, 0xa4, (tree.m_maxOccup + 1) * (TreeEntSize << 2));
+#endif
+}
+
+void
+Dbtux::preallocNode(Signal* signal, Frag& frag, Uint32& errorCode)
+{
+ ndbrequire(frag.m_nodeFree == RNIL);
+ NodeHandlePtr nodePtr;
+ seizeNode(signal, frag, nodePtr);
+ ndbrequire(nodePtr.i != RNIL);
+ // remove from cache XXX ugly
+ frag.m_nodeFree = frag.m_nodeList;
+ frag.m_nodeList = nodePtr.p->m_next;
+ StorePar storePar;
+ storePar.m_opCode = TupStoreTh::OpInsert;
+ storePar.m_offset = 0;
+ storePar.m_size = 0;
+ tupStoreTh(signal, frag, nodePtr, storePar);
+ if (storePar.m_errorCode != 0) {
+ jam();
+ errorCode = storePar.m_errorCode;
+ c_nodeHandlePool.release(nodePtr);
+ frag.m_nodeFree = RNIL;
+ }
+}
+
+/*
+ * Find node in the cache. XXX too slow, use direct links instead
+ */
+void
+Dbtux::findNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, TupAddr addr)
+{
+ NodeHandlePtr tmpPtr;
+ tmpPtr.i = frag.m_nodeList;
+ while (tmpPtr.i != RNIL) {
+ jam();
+ c_nodeHandlePool.getPtr(tmpPtr);
+ if (tmpPtr.p->m_addr == addr) {
+ jam();
+ nodePtr = tmpPtr;
+ return;
+ }
+ tmpPtr.i = tmpPtr.p->m_next;
+ }
+ nodePtr.i = RNIL;
+ nodePtr.p = 0;
+}
+
+/*
+ * Get handle for existing node.
+ */
+void
+Dbtux::selectNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, TupAddr addr, AccSize acc)
+{
+ ndbrequire(addr != NullTupAddr && acc > AccNone);
+ NodeHandlePtr tmpPtr;
+ // search in cache
+ findNode(signal, frag, tmpPtr, addr);
+ if (tmpPtr.i == RNIL) {
+ jam();
+ // add new node
+ seizeNode(signal, frag, tmpPtr);
+ ndbrequire(tmpPtr.i != RNIL);
+ tmpPtr.p->m_addr = addr;
+ }
+ if (tmpPtr.p->m_acc < acc) {
+ jam();
+ accessNode(signal, frag, tmpPtr, acc);
+ }
+ nodePtr = tmpPtr;
+}
+
+/*
+ * Create new node in the cache and mark it for insert.
+ */
+void
+Dbtux::insertNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, AccSize acc)
+{
+ ndbrequire(acc > AccNone);
+ NodeHandlePtr tmpPtr;
+ // use the pre-allocated node
+ tmpPtr.i = frag.m_nodeFree;
+ frag.m_nodeFree = RNIL;
+ c_nodeHandlePool.getPtr(tmpPtr);
+ // move it to the cache
+ tmpPtr.p->m_next = frag.m_nodeList;
+ frag.m_nodeList = tmpPtr.i;
+ tmpPtr.p->m_acc = acc;
+ tmpPtr.p->m_flags |= NodeHandle::DoInsert;
+ nodePtr = tmpPtr;
+}
+
+/*
+ * Mark existing node for deletion.
+ */
+void
+Dbtux::deleteNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr)
+{
+ NodeHandlePtr tmpPtr = nodePtr;
+ ndbrequire(tmpPtr.p->getOccup() == 0);
+ tmpPtr.p->m_flags |= NodeHandle::DoDelete;
+ // scans have already been moved by popDown or popUp
+}
+
+/*
+ * Access more of the node.
+ */
+void
+Dbtux::accessNode(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, AccSize acc)
+{
+ TreeHead& tree = frag.m_tree;
+ NodeHandlePtr tmpPtr = nodePtr;
+ if (tmpPtr.p->m_acc >= acc)
+ return;
+ if (! (tmpPtr.p->m_flags & NodeHandle::DoInsert)) {
+ jam();
+ StorePar storePar;
+ storePar.m_opCode = TupStoreTh::OpRead;
+ storePar.m_offset = tree.getSize(tmpPtr.p->m_acc);
+ storePar.m_size = tree.getSize(acc) - tree.getSize(tmpPtr.p->m_acc);
+ tmpPtr.p->m_tux.tupStoreTh(signal, frag, tmpPtr, storePar);
+ ndbrequire(storePar.m_errorCode == 0);
+ }
+ tmpPtr.p->m_acc = acc;
+}
+
+/*
+ * Set prefix.
+ */
+void
+Dbtux::setNodePref(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, unsigned i)
+{
+ TreeHead& tree = frag.m_tree;
+ NodeHandlePtr tmpPtr = nodePtr;
+ ReadPar readPar;
+ ndbrequire(i <= 1);
+ readPar.m_ent = tmpPtr.p->getMinMax(i);
+ readPar.m_first = 0;
+ readPar.m_count = frag.m_numAttrs;
+ // leave in signal data
+ readPar.m_data = 0;
+ // XXX implement max words to read
+ tupReadAttrs(signal, frag, readPar);
+ // copy whatever fits
+ CopyPar copyPar;
+ copyPar.m_items = readPar.m_count;
+ copyPar.m_headers = true;
+ copyPar.m_maxwords = tree.m_prefSize;
+ Data pref = tmpPtr.p->getPref(i);
+ copyAttrs(pref, readPar.m_data, copyPar);
+ nodePtr.p->m_flags |= NodeHandle::DoUpdate;
+}
+
+/*
+ * Commit and release nodes at the end of an operation. Used also on
+ * error since no changes have been made (updateOk false).
+ */
+void
+Dbtux::commitNodes(Signal* signal, Frag& frag, bool updateOk)
+{
+ TreeHead& tree = frag.m_tree;
+ NodeHandlePtr nodePtr;
+ nodePtr.i = frag.m_nodeList;
+ frag.m_nodeList = RNIL;
+ while (nodePtr.i != RNIL) {
+ c_nodeHandlePool.getPtr(nodePtr);
+ const unsigned flags = nodePtr.p->m_flags;
+ if (flags & NodeHandle::DoDelete) {
+ jam();
+ ndbrequire(updateOk);
+ // delete
+ StorePar storePar;
+ storePar.m_opCode = TupStoreTh::OpDelete;
+ nodePtr.p->m_tux.tupStoreTh(signal, frag, nodePtr, storePar);
+ ndbrequire(storePar.m_errorCode == 0);
+ } else if (flags & NodeHandle::DoUpdate) {
+ jam();
+ ndbrequire(updateOk);
+ // set prefixes
+ if (flags & (1 << 0)) {
+ jam();
+ setNodePref(signal, frag, nodePtr, 0);
+ }
+ if (flags & (1 << 1)) {
+ jam();
+ setNodePref(signal, frag, nodePtr, 1);
+ }
+ // update
+ StorePar storePar;
+ storePar.m_opCode = TupStoreTh::OpUpdate;
+ storePar.m_offset = 0;
+ storePar.m_size = tree.getSize(nodePtr.p->m_acc);
+ nodePtr.p->m_tux.tupStoreTh(signal, frag, nodePtr, storePar);
+ ndbrequire(storePar.m_errorCode == 0);
+ }
+ // release
+ NodeHandlePtr tmpPtr = nodePtr;
+ nodePtr.i = nodePtr.p->m_next;
+ c_nodeHandlePool.release(tmpPtr);
+ }
+}
+
+// Dbtux::NodeHandle
+
+/*
+ * Add entry at position. Move entries greater than or equal to the old
+ * one (if any) to the right.
+ *
+ * X
+ * v
+ * A B C D E _ _ => A B C X D E _
+ * 0 1 2 3 4 5 6 0 1 2 3 4 5 6
+ */
+void
+Dbtux::NodeHandle::pushUp(Signal* signal, unsigned pos, const TreeEnt& ent)
+{
+ TreeHead& tree = m_frag.m_tree;
+ const unsigned occup = getOccup();
+ ndbrequire(occup < tree.m_maxOccup && pos <= occup);
+ // fix scans
+ ScanOpPtr scanPtr;
+ scanPtr.i = getNodeScan();
+ while (scanPtr.i != RNIL) {
+ jam();
+ m_tux.c_scanOpPool.getPtr(scanPtr);
+ TreePos& scanPos = scanPtr.p->m_scanPos;
+ ndbrequire(scanPos.m_addr == m_addr && scanPos.m_pos < occup);
+ if (scanPos.m_pos >= pos) {
+ jam();
+#ifdef VM_TRACE
+ if (m_tux.debugFlags & m_tux.DebugScan) {
+ m_tux.debugOut << "Fix scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ m_tux.debugOut << "At pushUp pos=" << pos << " " << *this << endl;
+ }
+#endif
+ scanPos.m_pos++;
+ }
+ scanPtr.i = scanPtr.p->m_nodeScan;
+ }
+ // fix node
+ TreeEnt* const entList = tree.getEntList(m_node);
+ entList[occup] = entList[0];
+ TreeEnt* const tmpList = entList + 1;
+ for (unsigned i = occup; i > pos; i--) {
+ jam();
+ tmpList[i] = tmpList[i - 1];
+ }
+ tmpList[pos] = ent;
+ if (occup == 0 || pos == 0)
+ m_flags |= (1 << 0);
+ if (occup == 0 || pos == occup)
+ m_flags |= (1 << 1);
+ entList[0] = entList[occup + 1];
+ setOccup(occup + 1);
+ m_flags |= DoUpdate;
+}
+
+/*
+ * Remove and return entry at position. Move entries greater than the
+ * removed one to the left. This is the opposite of pushUp.
+ *
+ * D
+ * ^ ^
+ * A B C D E F _ => A B C E F _ _
+ * 0 1 2 3 4 5 6 0 1 2 3 4 5 6
+ */
+void
+Dbtux::NodeHandle::popDown(Signal* signal, unsigned pos, TreeEnt& ent)
+{
+ TreeHead& tree = m_frag.m_tree;
+ const unsigned occup = getOccup();
+ ndbrequire(occup <= tree.m_maxOccup && pos < occup);
+ ScanOpPtr scanPtr;
+ // move scans whose entry disappears
+ scanPtr.i = getNodeScan();
+ while (scanPtr.i != RNIL) {
+ jam();
+ m_tux.c_scanOpPool.getPtr(scanPtr);
+ TreePos& scanPos = scanPtr.p->m_scanPos;
+ ndbrequire(scanPos.m_addr == m_addr && scanPos.m_pos < occup);
+ const Uint32 nextPtrI = scanPtr.p->m_nodeScan;
+ if (scanPos.m_pos == pos) {
+ jam();
+#ifdef VM_TRACE
+ if (m_tux.debugFlags & m_tux.DebugScan) {
+ m_tux.debugOut << "Move scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ m_tux.debugOut << "At popDown pos=" << pos << " " << *this << endl;
+ }
+#endif
+ m_tux.scanNext(signal, scanPtr);
+ }
+ scanPtr.i = nextPtrI;
+ }
+ // fix other scans
+ scanPtr.i = getNodeScan();
+ while (scanPtr.i != RNIL) {
+ jam();
+ m_tux.c_scanOpPool.getPtr(scanPtr);
+ TreePos& scanPos = scanPtr.p->m_scanPos;
+ ndbrequire(scanPos.m_addr == m_addr && scanPos.m_pos < occup);
+ ndbrequire(scanPos.m_pos != pos);
+ if (scanPos.m_pos > pos) {
+ jam();
+#ifdef VM_TRACE
+ if (m_tux.debugFlags & m_tux.DebugScan) {
+ m_tux.debugOut << "Fix scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ m_tux.debugOut << "At popDown pos=" << pos << " " << *this << endl;
+ }
+#endif
+ scanPos.m_pos--;
+ }
+ scanPtr.i = scanPtr.p->m_nodeScan;
+ }
+ // fix node
+ TreeEnt* const entList = tree.getEntList(m_node);
+ entList[occup] = entList[0];
+ TreeEnt* const tmpList = entList + 1;
+ ent = tmpList[pos];
+ for (unsigned i = pos; i < occup - 1; i++) {
+ jam();
+ tmpList[i] = tmpList[i + 1];
+ }
+ if (occup != 1 && pos == 0)
+ m_flags |= (1 << 0);
+ if (occup != 1 && pos == occup - 1)
+ m_flags |= (1 << 1);
+ entList[0] = entList[occup - 1];
+ setOccup(occup - 1);
+ m_flags |= DoUpdate;
+}
+
+/*
+ * Add entry at existing position. Move entries less than or equal to
+ * the old one to the left. Remove and return old min entry.
+ *
+ * X A
+ * ^ v ^
+ * A B C D E _ _ => B C D X E _ _
+ * 0 1 2 3 4 5 6 0 1 2 3 4 5 6
+ */
+void
+Dbtux::NodeHandle::pushDown(Signal* signal, unsigned pos, TreeEnt& ent)
+{
+ TreeHead& tree = m_frag.m_tree;
+ const unsigned occup = getOccup();
+ ndbrequire(occup <= tree.m_maxOccup && pos < occup);
+ ScanOpPtr scanPtr;
+ // move scans whose entry disappears
+ scanPtr.i = getNodeScan();
+ while (scanPtr.i != RNIL) {
+ jam();
+ m_tux.c_scanOpPool.getPtr(scanPtr);
+ TreePos& scanPos = scanPtr.p->m_scanPos;
+ ndbrequire(scanPos.m_addr == m_addr && scanPos.m_pos < occup);
+ const Uint32 nextPtrI = scanPtr.p->m_nodeScan;
+ if (scanPos.m_pos == 0) {
+ jam();
+#ifdef VM_TRACE
+ if (m_tux.debugFlags & m_tux.DebugScan) {
+ m_tux.debugOut << "Move scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ m_tux.debugOut << "At pushDown pos=" << pos << " " << *this << endl;
+ }
+#endif
+ // here we may miss a valid entry "X" XXX known bug
+ m_tux.scanNext(signal, scanPtr);
+ }
+ scanPtr.i = nextPtrI;
+ }
+ // fix other scans
+ scanPtr.i = getNodeScan();
+ while (scanPtr.i != RNIL) {
+ jam();
+ m_tux.c_scanOpPool.getPtr(scanPtr);
+ TreePos& scanPos = scanPtr.p->m_scanPos;
+ ndbrequire(scanPos.m_addr == m_addr && scanPos.m_pos < occup);
+ ndbrequire(scanPos.m_pos != 0);
+ if (scanPos.m_pos <= pos) {
+ jam();
+#ifdef VM_TRACE
+ if (m_tux.debugFlags & m_tux.DebugScan) {
+ m_tux.debugOut << "Fix scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ m_tux.debugOut << "At pushDown pos=" << pos << " " << *this << endl;
+ }
+#endif
+ scanPos.m_pos--;
+ }
+ scanPtr.i = scanPtr.p->m_nodeScan;
+ }
+ // fix node
+ TreeEnt* const entList = tree.getEntList(m_node);
+ entList[occup] = entList[0];
+ TreeEnt* const tmpList = entList + 1;
+ TreeEnt oldMin = tmpList[0];
+ for (unsigned i = 0; i < pos; i++) {
+ jam();
+ tmpList[i] = tmpList[i + 1];
+ }
+ tmpList[pos] = ent;
+ ent = oldMin;
+ if (true)
+ m_flags |= (1 << 0);
+ if (occup == 1 || pos == occup - 1)
+ m_flags |= (1 << 1);
+ entList[0] = entList[occup];
+ m_flags |= DoUpdate;
+}
+
+/*
+ * Remove and return entry at position. Move entries less than the
+ * removed one to the right. Replace min entry by the input entry.
+ * This is the opposite of pushDown.
+ *
+ * X D
+ * v ^ ^
+ * A B C D E _ _ => X A B C E _ _
+ * 0 1 2 3 4 5 6 0 1 2 3 4 5 6
+ */
+void
+Dbtux::NodeHandle::popUp(Signal* signal, unsigned pos, TreeEnt& ent)
+{
+ TreeHead& tree = m_frag.m_tree;
+ const unsigned occup = getOccup();
+ ndbrequire(occup <= tree.m_maxOccup && pos < occup);
+ ScanOpPtr scanPtr;
+ // move scans whose entry disappears
+ scanPtr.i = getNodeScan();
+ while (scanPtr.i != RNIL) {
+ jam();
+ m_tux.c_scanOpPool.getPtr(scanPtr);
+ TreePos& scanPos = scanPtr.p->m_scanPos;
+ ndbrequire(scanPos.m_addr == m_addr && scanPos.m_pos < occup);
+ const Uint32 nextPtrI = scanPtr.p->m_nodeScan;
+ if (scanPos.m_pos == pos) {
+ jam();
+#ifdef VM_TRACE
+ if (m_tux.debugFlags & m_tux.DebugScan) {
+ m_tux.debugOut << "Move scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ m_tux.debugOut << "At popUp pos=" << pos << " " << *this << endl;
+ }
+#endif
+ // here we may miss a valid entry "X" XXX known bug
+ m_tux.scanNext(signal, scanPtr);
+ }
+ scanPtr.i = nextPtrI;
+ }
+ // fix other scans
+ scanPtr.i = getNodeScan();
+ while (scanPtr.i != RNIL) {
+ jam();
+ m_tux.c_scanOpPool.getPtr(scanPtr);
+ TreePos& scanPos = scanPtr.p->m_scanPos;
+ ndbrequire(scanPos.m_addr == m_addr && scanPos.m_pos < occup);
+ ndbrequire(scanPos.m_pos != pos);
+ if (scanPos.m_pos < pos) {
+ jam();
+#ifdef VM_TRACE
+ if (m_tux.debugFlags & m_tux.DebugScan) {
+ m_tux.debugOut << "Fix scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ m_tux.debugOut << "At popUp pos=" << pos << " " << *this << endl;
+ }
+#endif
+ scanPos.m_pos++;
+ }
+ scanPtr.i = scanPtr.p->m_nodeScan;
+ }
+ // fix node
+ TreeEnt* const entList = tree.getEntList(m_node);
+ entList[occup] = entList[0];
+ TreeEnt* const tmpList = entList + 1;
+ TreeEnt newMin = ent;
+ ent = tmpList[pos];
+ for (unsigned i = pos; i > 0; i--) {
+ jam();
+ tmpList[i] = tmpList[i - 1];
+ }
+ tmpList[0] = newMin;
+ if (true)
+ m_flags |= (1 << 0);
+ if (occup == 1 || pos == occup - 1)
+ m_flags |= (1 << 1);
+ entList[0] = entList[occup];
+ m_flags |= DoUpdate;
+}
+
+/*
+ * Move all possible entries from another node before the min (i=0) or
+ * after the max (i=1). XXX can be optimized
+ */
+void
+Dbtux::NodeHandle::slide(Signal* signal, NodeHandlePtr nodePtr, unsigned i)
+{
+ ndbrequire(i <= 1);
+ TreeHead& tree = m_frag.m_tree;
+ while (getOccup() < tree.m_maxOccup && nodePtr.p->getOccup() != 0) {
+ TreeEnt ent;
+ nodePtr.p->popDown(signal, i == 0 ? nodePtr.p->getOccup() - 1 : 0, ent);
+ pushUp(signal, i == 0 ? 0 : getOccup(), ent);
+ }
+}
+
+/*
+ * Link scan to the list under the node. The list is single-linked and
+ * ordering does not matter.
+ */
+void
+Dbtux::NodeHandle::linkScan(Dbtux::ScanOpPtr scanPtr)
+{
+#ifdef VM_TRACE
+ if (m_tux.debugFlags & m_tux.DebugScan) {
+ m_tux.debugOut << "Link scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ m_tux.debugOut << "To node " << *this << endl;
+ }
+#endif
+ ndbrequire(! islinkScan(scanPtr) && scanPtr.p->m_nodeScan == RNIL);
+ scanPtr.p->m_nodeScan = getNodeScan();
+ setNodeScan(scanPtr.i);
+}
+
+/*
+ * Unlink a scan from the list under the node.
+ */
+void
+Dbtux::NodeHandle::unlinkScan(Dbtux::ScanOpPtr scanPtr)
+{
+#ifdef VM_TRACE
+ if (m_tux.debugFlags & m_tux.DebugScan) {
+ m_tux.debugOut << "Unlink scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ m_tux.debugOut << "From node " << *this << endl;
+ }
+#endif
+ Dbtux::ScanOpPtr currPtr;
+ currPtr.i = getNodeScan();
+ Dbtux::ScanOpPtr prevPtr;
+ prevPtr.i = RNIL;
+ while (true) {
+ jam();
+ m_tux.c_scanOpPool.getPtr(currPtr);
+ Uint32 nextPtrI = currPtr.p->m_nodeScan;
+ if (currPtr.i == scanPtr.i) {
+ jam();
+ if (prevPtr.i == RNIL) {
+ setNodeScan(nextPtrI);
+ } else {
+ jam();
+ prevPtr.p->m_nodeScan = nextPtrI;
+ }
+ scanPtr.p->m_nodeScan = RNIL;
+ // check for duplicates
+ ndbrequire(! islinkScan(scanPtr));
+ return;
+ }
+ prevPtr = currPtr;
+ currPtr.i = nextPtrI;
+ }
+}
+
+/*
+ * Check if a scan is linked to this node. Only for ndbrequire.
+ */
+bool
+Dbtux::NodeHandle::islinkScan(Dbtux::ScanOpPtr scanPtr)
+{
+ Dbtux::ScanOpPtr currPtr;
+ currPtr.i = getNodeScan();
+ while (currPtr.i != RNIL) {
+ jam();
+ m_tux.c_scanOpPool.getPtr(currPtr);
+ if (currPtr.i == scanPtr.i) {
+ jam();
+ return true;
+ }
+ currPtr.i = currPtr.p->m_nodeScan;
+ }
+ return false;
+}
+
+void
+Dbtux::NodeHandle::progError(int line, int cause, const char* extra)
+{
+ m_tux.progError(line, cause, extra);
+}
diff --git a/ndb/src/kernel/blocks/dbtux/DbtuxScan.cpp b/ndb/src/kernel/blocks/dbtux/DbtuxScan.cpp
new file mode 100644
index 00000000000..eaa539d9cfc
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/DbtuxScan.cpp
@@ -0,0 +1,1124 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUX_SCAN_CPP
+#include "Dbtux.hpp"
+
+void
+Dbtux::execACC_SCANREQ(Signal* signal)
+{
+ jamEntry();
+ const AccScanReq reqCopy = *(const AccScanReq*)signal->getDataPtr();
+ const AccScanReq* const req = &reqCopy;
+ ScanOpPtr scanPtr;
+ scanPtr.i = RNIL;
+ do {
+ // get the index
+ IndexPtr indexPtr;
+ c_indexPool.getPtr(indexPtr, req->tableId);
+ // get the fragment
+ FragPtr fragPtr;
+ fragPtr.i = RNIL;
+ for (unsigned i = 0; i < indexPtr.p->m_numFrags; i++) {
+ jam();
+ if (indexPtr.p->m_fragId[i] == req->fragmentNo) {
+ jam();
+ c_fragPool.getPtr(fragPtr, indexPtr.p->m_fragPtrI[i]);
+ break;
+ }
+ }
+ ndbrequire(fragPtr.i != RNIL);
+ Frag& frag = *fragPtr.p;
+ ndbrequire(frag.m_nodeList == RNIL);
+ // must be normal DIH/TC fragment
+ ndbrequire(frag.m_fragId < (1 << frag.m_fragOff));
+ TreeHead& tree = frag.m_tree;
+ // check for empty fragment
+ if (tree.m_root == NullTupAddr) {
+ jam();
+ AccScanConf* const conf = (AccScanConf*)signal->getDataPtrSend();
+ conf->scanPtr = req->senderData;
+ conf->accPtr = RNIL;
+ conf->flag = AccScanConf::ZEMPTY_FRAGMENT;
+ sendSignal(req->senderRef, GSN_ACC_SCANCONF,
+ signal, AccScanConf::SignalLength, JBB);
+ return;
+ }
+ // seize from pool and link to per-fragment list
+ if (! frag.m_scanList.seize(scanPtr)) {
+ jam();
+ break;
+ }
+ new (scanPtr.p) ScanOp(c_scanBoundPool);
+ scanPtr.p->m_state = ScanOp::First;
+ scanPtr.p->m_userPtr = req->senderData;
+ scanPtr.p->m_userRef = req->senderRef;
+ scanPtr.p->m_tableId = indexPtr.p->m_tableId;
+ scanPtr.p->m_indexId = indexPtr.i;
+ scanPtr.p->m_fragId = fragPtr.p->m_fragId;
+ scanPtr.p->m_fragPtrI = fragPtr.i;
+ scanPtr.p->m_transId1 = req->transId1;
+ scanPtr.p->m_transId2 = req->transId2;
+ scanPtr.p->m_savePointId = req->savePointId;
+ scanPtr.p->m_readCommitted = AccScanReq::getReadCommittedFlag(req->requestInfo);
+ scanPtr.p->m_lockMode = AccScanReq::getLockMode(req->requestInfo);
+ scanPtr.p->m_keyInfo = AccScanReq::getKeyinfoFlag(req->requestInfo);
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "Seize scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ }
+#endif
+ /*
+ * readCommitted lockMode keyInfo
+ * 1 0 0 - read committed (no lock)
+ * 0 0 0 - read latest (read lock)
+ * 0 1 1 - read exclusive (write lock)
+ */
+ // conf
+ AccScanConf* const conf = (AccScanConf*)signal->getDataPtrSend();
+ conf->scanPtr = req->senderData;
+ conf->accPtr = scanPtr.i;
+ conf->flag = AccScanConf::ZNOT_EMPTY_FRAGMENT;
+ sendSignal(req->senderRef, GSN_ACC_SCANCONF,
+ signal, AccScanConf::SignalLength, JBB);
+ return;
+ } while (0);
+ if (scanPtr.i != RNIL) {
+ jam();
+ releaseScanOp(scanPtr);
+ }
+ // LQH does not handle REF
+ signal->theData[0] = 0x313;
+ sendSignal(req->senderRef, GSN_ACC_SCANREF,
+ signal, 1, JBB);
+}
+
+/*
+ * Receive bounds for scan in single direct call. The bounds can arrive
+ * in any order. Attribute ids are those of index table.
+ */
+void
+Dbtux::execTUX_BOUND_INFO(Signal* signal)
+{
+ struct BoundInfo {
+ unsigned offset;
+ unsigned size;
+ int type;
+ };
+ TuxBoundInfo* const sig = (TuxBoundInfo*)signal->getDataPtrSend();
+ const TuxBoundInfo reqCopy = *(const TuxBoundInfo*)sig;
+ const TuxBoundInfo* const req = &reqCopy;
+ // get records
+ ScanOp& scan = *c_scanOpPool.getPtr(req->tuxScanPtrI);
+ Index& index = *c_indexPool.getPtr(scan.m_indexId);
+ // collect bound info for each index attribute
+ BoundInfo boundInfo[MaxIndexAttributes][2];
+ // largest attrId seen plus one
+ Uint32 maxAttrId = 0;
+ // skip 5 words
+ if (req->boundAiLength < 5) {
+ jam();
+ scan.m_state = ScanOp::Invalid;
+ sig->errorCode = TuxBoundInfo::InvalidAttrInfo;
+ return;
+ }
+ const Uint32* const data = (Uint32*)sig + TuxBoundInfo::SignalLength;
+ unsigned offset = 5;
+ // walk through entries
+ while (offset + 2 < req->boundAiLength) {
+ jam();
+ const unsigned type = data[offset];
+ if (type > 4) {
+ jam();
+ scan.m_state = ScanOp::Invalid;
+ sig->errorCode = TuxBoundInfo::InvalidAttrInfo;
+ return;
+ }
+ const AttributeHeader* ah = (const AttributeHeader*)&data[offset + 1];
+ const Uint32 attrId = ah->getAttributeId();
+ const Uint32 dataSize = ah->getDataSize();
+ if (attrId >= index.m_numAttrs) {
+ jam();
+ scan.m_state = ScanOp::Invalid;
+ sig->errorCode = TuxBoundInfo::InvalidAttrInfo;
+ return;
+ }
+ while (maxAttrId <= attrId) {
+ BoundInfo* b = boundInfo[maxAttrId++];
+ b[0].type = b[1].type = -1;
+ }
+ BoundInfo* b = boundInfo[attrId];
+ if (type == 0 || type == 1 || type == 4) {
+ if (b[0].type != -1) {
+ jam();
+ scan.m_state = ScanOp::Invalid;
+ sig->errorCode = TuxBoundInfo::InvalidBounds;
+ return;
+ }
+ b[0].offset = offset;
+ b[0].size = 2 + dataSize;
+ b[0].type = type;
+ }
+ if (type == 2 || type == 3 || type == 4) {
+ if (b[1].type != -1) {
+ jam();
+ scan.m_state = ScanOp::Invalid;
+ sig->errorCode = TuxBoundInfo::InvalidBounds;
+ return;
+ }
+ b[1].offset = offset;
+ b[1].size = 2 + dataSize;
+ b[1].type = type;
+ }
+ // jump to next
+ offset += 2 + dataSize;
+ }
+ if (offset != req->boundAiLength) {
+ jam();
+ scan.m_state = ScanOp::Invalid;
+ sig->errorCode = TuxBoundInfo::InvalidAttrInfo;
+ return;
+ }
+ // save the bounds in index attribute id order
+ scan.m_boundCnt[0] = 0;
+ scan.m_boundCnt[1] = 0;
+ for (unsigned i = 0; i < maxAttrId; i++) {
+ jam();
+ const BoundInfo* b = boundInfo[i];
+ // current limitation - check all but last is equality
+ if (i + 1 < maxAttrId) {
+ if (b[0].type != 4 || b[1].type != 4) {
+ jam();
+ scan.m_state = ScanOp::Invalid;
+ sig->errorCode = TuxBoundInfo::InvalidBounds;
+ return;
+ }
+ }
+ for (unsigned j = 0; j <= 1; j++) {
+ if (b[j].type != -1) {
+ jam();
+ bool ok = scan.m_bound[j]->append(&data[b[j].offset], b[j].size);
+ if (! ok) {
+ jam();
+ scan.m_state = ScanOp::Invalid;
+ sig->errorCode = TuxBoundInfo::OutOfBuffers;
+ return;
+ }
+ scan.m_boundCnt[j]++;
+ }
+ }
+ }
+ // no error
+ sig->errorCode = 0;
+}
+
+void
+Dbtux::execNEXT_SCANREQ(Signal* signal)
+{
+ jamEntry();
+ const NextScanReq reqCopy = *(const NextScanReq*)signal->getDataPtr();
+ const NextScanReq* const req = &reqCopy;
+ ScanOpPtr scanPtr;
+ scanPtr.i = req->accPtr;
+ c_scanOpPool.getPtr(scanPtr);
+ ScanOp& scan = *scanPtr.p;
+ Frag& frag = *c_fragPool.getPtr(scan.m_fragPtrI);
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "NEXT_SCANREQ scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+ ndbrequire(frag.m_nodeList == RNIL);
+ // handle unlock previous and close scan
+ switch (req->scanFlag) {
+ case NextScanReq::ZSCAN_NEXT:
+ jam();
+ break;
+ case NextScanReq::ZSCAN_NEXT_COMMIT:
+ jam();
+ case NextScanReq::ZSCAN_COMMIT:
+ jam();
+ if (! scan.m_readCommitted) {
+ jam();
+ AccLockReq* const lockReq = (AccLockReq*)signal->getDataPtrSend();
+ lockReq->returnCode = RNIL;
+ lockReq->requestInfo = AccLockReq::Unlock;
+ lockReq->accOpPtr = req->accOperationPtr;
+ EXECUTE_DIRECT(DBACC, GSN_ACC_LOCKREQ, signal, AccLockReq::UndoSignalLength);
+ jamEntry();
+ ndbrequire(lockReq->returnCode == AccLockReq::Success);
+ removeAccLockOp(scan, req->accOperationPtr);
+ }
+ if (req->scanFlag == NextScanReq::ZSCAN_COMMIT) {
+ jam();
+ NextScanConf* const conf = (NextScanConf*)signal->getDataPtrSend();
+ conf->scanPtr = scan.m_userPtr;
+ unsigned signalLength = 1;
+ sendSignal(scanPtr.p->m_userRef, GSN_NEXT_SCANCONF,
+ signal, signalLength, JBB);
+ return;
+ }
+ break;
+ case NextScanReq::ZSCAN_CLOSE:
+ jam();
+ // unlink from tree node first to avoid state changes
+ if (scan.m_scanPos.m_addr != NullTupAddr) {
+ jam();
+ const TupAddr addr = scan.m_scanPos.m_addr;
+ NodeHandlePtr nodePtr;
+ selectNode(signal, frag, nodePtr, addr, AccHead);
+ nodePtr.p->unlinkScan(scanPtr);
+ scan.m_scanPos.m_addr = NullTupAddr;
+ }
+ if (scan.m_lockwait) {
+ jam();
+ ndbrequire(scan.m_accLockOp != RNIL);
+ // use ACC_ABORTCONF to flush out any reply in job buffer
+ AccLockReq* const lockReq = (AccLockReq*)signal->getDataPtrSend();
+ lockReq->returnCode = RNIL;
+ lockReq->requestInfo = AccLockReq::AbortWithConf;
+ lockReq->accOpPtr = scan.m_accLockOp;
+ EXECUTE_DIRECT(DBACC, GSN_ACC_LOCKREQ, signal, AccLockReq::UndoSignalLength);
+ jamEntry();
+ ndbrequire(lockReq->returnCode == AccLockReq::Success);
+ scan.m_state = ScanOp::Aborting;
+ commitNodes(signal, frag, true);
+ return;
+ }
+ if (scan.m_state == ScanOp::Locked) {
+ jam();
+ ndbrequire(scan.m_accLockOp != RNIL);
+ AccLockReq* const lockReq = (AccLockReq*)signal->getDataPtrSend();
+ lockReq->returnCode = RNIL;
+ lockReq->requestInfo = AccLockReq::Unlock;
+ lockReq->accOpPtr = scan.m_accLockOp;
+ EXECUTE_DIRECT(DBACC, GSN_ACC_LOCKREQ, signal, AccLockReq::UndoSignalLength);
+ jamEntry();
+ ndbrequire(lockReq->returnCode == AccLockReq::Success);
+ scan.m_accLockOp = RNIL;
+ }
+ scan.m_state = ScanOp::Aborting;
+ scanClose(signal, scanPtr);
+ return;
+ case NextScanReq::ZSCAN_NEXT_ABORT:
+ jam();
+ default:
+ jam();
+ ndbrequire(false);
+ break;
+ }
+ // start looking for next scan result
+ AccCheckScan* checkReq = (AccCheckScan*)signal->getDataPtrSend();
+ checkReq->accPtr = scanPtr.i;
+ checkReq->checkLcpStop = AccCheckScan::ZNOT_CHECK_LCP_STOP;
+ EXECUTE_DIRECT(DBTUX, GSN_ACC_CHECK_SCAN, signal, AccCheckScan::SignalLength);
+ jamEntry();
+}
+
+void
+Dbtux::execACC_CHECK_SCAN(Signal* signal)
+{
+ jamEntry();
+ const AccCheckScan reqCopy = *(const AccCheckScan*)signal->getDataPtr();
+ const AccCheckScan* const req = &reqCopy;
+ ScanOpPtr scanPtr;
+ scanPtr.i = req->accPtr;
+ c_scanOpPool.getPtr(scanPtr);
+ ScanOp& scan = *scanPtr.p;
+ Frag& frag = *c_fragPool.getPtr(scan.m_fragPtrI);
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "ACC_CHECK_SCAN scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+ if (req->checkLcpStop == AccCheckScan::ZCHECK_LCP_STOP) {
+ jam();
+ signal->theData[0] = scan.m_userPtr;
+ signal->theData[1] = true;
+ EXECUTE_DIRECT(DBLQH, GSN_CHECK_LCP_STOP, signal, 2);
+ jamEntry();
+ commitNodes(signal, frag, true);
+ return; // stop
+ }
+ if (scan.m_lockwait) {
+ jam();
+ // LQH asks if we are waiting for lock and we tell it to ask again
+ const TreeEnt ent = scan.m_scanPos.m_ent;
+ NextScanConf* const conf = (NextScanConf*)signal->getDataPtrSend();
+ conf->scanPtr = scan.m_userPtr;
+ conf->accOperationPtr = RNIL; // no tuple returned
+ conf->fragId = frag.m_fragId | (ent.m_fragBit << frag.m_fragOff);
+ unsigned signalLength = 3;
+ // if TC has ordered scan close, it will be detected here
+ sendSignal(scan.m_userRef, GSN_NEXT_SCANCONF,
+ signal, signalLength, JBB);
+ commitNodes(signal, frag, true);
+ return; // stop
+ }
+ if (scan.m_state == ScanOp::First) {
+ jam();
+ // search is done only once in single range scan
+ scanFirst(signal, scanPtr);
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "First scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+ }
+ if (scan.m_state == ScanOp::Next) {
+ jam();
+ // look for next
+ scanNext(signal, scanPtr);
+ }
+ // for reading tuple key in Current or Locked state
+ ReadPar keyPar;
+ keyPar.m_data = 0; // indicates not yet done
+ if (scan.m_state == ScanOp::Current) {
+ // found an entry to return
+ jam();
+ ndbrequire(scan.m_accLockOp == RNIL);
+ if (! scan.m_readCommitted) {
+ jam();
+ const TreeEnt ent = scan.m_scanPos.m_ent;
+ // read tuple key
+ keyPar.m_ent = ent;
+ keyPar.m_data = c_keyBuffer;
+ tupReadKeys(signal, frag, keyPar);
+ // get read lock or exclusive lock
+ AccLockReq* const lockReq = (AccLockReq*)signal->getDataPtrSend();
+ lockReq->returnCode = RNIL;
+ lockReq->requestInfo =
+ scan.m_lockMode == 0 ? AccLockReq::LockShared : AccLockReq::LockExclusive;
+ lockReq->accOpPtr = RNIL;
+ lockReq->userPtr = scanPtr.i;
+ lockReq->userRef = reference();
+ lockReq->tableId = scan.m_tableId;
+ lockReq->fragId = frag.m_fragId | (ent.m_fragBit << frag.m_fragOff);
+ // should cache this at fragment create
+ lockReq->fragPtrI = RNIL;
+ const Uint32* const buf32 = static_cast<Uint32*>(keyPar.m_data);
+ const Uint64* const buf64 = reinterpret_cast<const Uint64*>(buf32);
+ lockReq->hashValue = md5_hash(buf64, keyPar.m_size);
+ lockReq->tupAddr = ent.m_tupAddr;
+ lockReq->transId1 = scan.m_transId1;
+ lockReq->transId2 = scan.m_transId2;
+ // execute
+ EXECUTE_DIRECT(DBACC, GSN_ACC_LOCKREQ, signal, AccLockReq::LockSignalLength);
+ jamEntry();
+ switch (lockReq->returnCode) {
+ case AccLockReq::Success:
+ jam();
+ scan.m_state = ScanOp::Locked;
+ scan.m_accLockOp = lockReq->accOpPtr;
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "Lock immediate scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+ break;
+ case AccLockReq::IsBlocked:
+ jam();
+ // normal lock wait
+ scan.m_state = ScanOp::Blocked;
+ scan.m_lockwait = true;
+ scan.m_accLockOp = lockReq->accOpPtr;
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "Lock wait scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+ // LQH will wake us up
+ signal->theData[0] = scan.m_userPtr;
+ signal->theData[1] = true;
+ EXECUTE_DIRECT(DBLQH, GSN_CHECK_LCP_STOP, signal, 2);
+ jamEntry();
+ commitNodes(signal, frag, true);
+ return; // stop
+ break;
+ case AccLockReq::Refused:
+ jam();
+ // we cannot see deleted tuple (assert only)
+ ndbassert(false);
+ // skip it
+ scan.m_state = ScanOp::Next;
+ signal->theData[0] = scan.m_userPtr;
+ signal->theData[1] = true;
+ EXECUTE_DIRECT(DBLQH, GSN_CHECK_LCP_STOP, signal, 2);
+ jamEntry();
+ commitNodes(signal, frag, true);
+ return; // stop
+ break;
+ case AccLockReq::NoFreeOp:
+ jam();
+ // max ops should depend on max scans (assert only)
+ ndbassert(false);
+ // stay in Current state
+ scan.m_state = ScanOp::Current;
+ signal->theData[0] = scan.m_userPtr;
+ signal->theData[1] = true;
+ EXECUTE_DIRECT(DBLQH, GSN_CHECK_LCP_STOP, signal, 2);
+ jamEntry();
+ commitNodes(signal, frag, true);
+ return; // stop
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+ } else {
+ scan.m_state = ScanOp::Locked;
+ }
+ }
+ if (scan.m_state == ScanOp::Locked) {
+ // we have lock or do not need one
+ jam();
+ // read keys if not already done (uses signal)
+ const TreeEnt ent = scan.m_scanPos.m_ent;
+ if (scan.m_keyInfo) {
+ jam();
+ if (keyPar.m_data == 0) {
+ jam();
+ keyPar.m_ent = ent;
+ keyPar.m_data = c_keyBuffer;
+ tupReadKeys(signal, frag, keyPar);
+ }
+ }
+ // conf signal
+ NextScanConf* const conf = (NextScanConf*)signal->getDataPtrSend();
+ conf->scanPtr = scan.m_userPtr;
+ // the lock is passed to LQH
+ Uint32 accLockOp = scan.m_accLockOp;
+ if (accLockOp != RNIL) {
+ scan.m_accLockOp = RNIL;
+ // remember it until LQH unlocks it
+ addAccLockOp(scan, accLockOp);
+ } else {
+ ndbrequire(scan.m_readCommitted);
+ // operation RNIL in LQH would signal no tuple returned
+ accLockOp = (Uint32)-1;
+ }
+ conf->accOperationPtr = accLockOp;
+ conf->fragId = frag.m_fragId | (ent.m_fragBit << frag.m_fragOff);
+ conf->localKey[0] = ent.m_tupAddr;
+ conf->localKey[1] = 0;
+ conf->localKeyLength = 1;
+ unsigned signalLength = 6;
+ // add key info
+ if (scan.m_keyInfo) {
+ jam();
+ conf->keyLength = keyPar.m_size;
+ // piggy-back first 4 words of key data
+ for (unsigned i = 0; i < 4; i++) {
+ conf->key[i] = i < keyPar.m_size ? keyPar.m_data[i] : 0;
+ }
+ signalLength = 11;
+ }
+ if (! scan.m_readCommitted) {
+ sendSignal(scan.m_userRef, GSN_NEXT_SCANCONF,
+ signal, signalLength, JBB);
+ } else {
+ Uint32 blockNo = refToBlock(scan.m_userRef);
+ EXECUTE_DIRECT(blockNo, GSN_NEXT_SCANCONF, signal, signalLength);
+ }
+ // send rest of key data
+ if (scan.m_keyInfo && keyPar.m_size > 4) {
+ unsigned total = 4;
+ while (total < keyPar.m_size) {
+ jam();
+ unsigned length = keyPar.m_size - total;
+ if (length > 20)
+ length = 20;
+ signal->theData[0] = scan.m_userPtr;
+ signal->theData[1] = 0;
+ signal->theData[2] = 0;
+ signal->theData[3] = length;
+ memcpy(&signal->theData[4], &keyPar.m_data[total], length << 2);
+ sendSignal(scan.m_userRef, GSN_ACC_SCAN_INFO24,
+ signal, 4 + length, JBB);
+ total += length;
+ }
+ }
+ // remember last entry returned
+ scan.m_lastEnt = ent;
+ // next time look for next entry
+ scan.m_state = ScanOp::Next;
+ commitNodes(signal, frag, true);
+ return;
+ }
+ // XXX in ACC this is checked before req->checkLcpStop
+ if (scan.m_state == ScanOp::Last ||
+ scan.m_state == ScanOp::Invalid) {
+ jam();
+ NextScanConf* const conf = (NextScanConf*)signal->getDataPtrSend();
+ conf->scanPtr = scan.m_userPtr;
+ conf->accOperationPtr = RNIL;
+ conf->fragId = RNIL;
+ unsigned signalLength = 3;
+ sendSignal(scanPtr.p->m_userRef, GSN_NEXT_SCANCONF,
+ signal, signalLength, JBB);
+ commitNodes(signal, frag, true);
+ return;
+ }
+ ndbrequire(false);
+}
+
+/*
+ * Lock succeeded (after delay) in ACC. If the lock is for current
+ * entry, set state to Locked. If the lock is for an entry we were
+ * moved away from, simply unlock it. Finally, if we are closing the
+ * scan, do nothing since we have already sent an abort request.
+ */
+void
+Dbtux::execACCKEYCONF(Signal* signal)
+{
+ jamEntry();
+ ScanOpPtr scanPtr;
+ scanPtr.i = signal->theData[0];
+ c_scanOpPool.getPtr(scanPtr);
+ ScanOp& scan = *scanPtr.p;
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "Lock obtained scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+ ndbrequire(scan.m_lockwait && scan.m_accLockOp != RNIL);
+ scan.m_lockwait = false;
+ if (scan.m_state == ScanOp::Blocked) {
+ // the lock wait was for current entry
+ jam();
+ scan.m_state = ScanOp::Locked;
+ // LQH has the ball
+ return;
+ }
+ if (scan.m_state != ScanOp::Aborting) {
+ // we were moved, release lock
+ jam();
+ AccLockReq* const lockReq = (AccLockReq*)signal->getDataPtrSend();
+ lockReq->returnCode = RNIL;
+ lockReq->requestInfo = AccLockReq::Unlock;
+ lockReq->accOpPtr = scan.m_accLockOp;
+ EXECUTE_DIRECT(DBACC, GSN_ACC_LOCKREQ, signal, AccLockReq::UndoSignalLength);
+ jamEntry();
+ ndbrequire(lockReq->returnCode == AccLockReq::Success);
+ scan.m_accLockOp = RNIL;
+ // LQH has the ball
+ return;
+ }
+ // continue at ACC_ABORTCONF
+}
+
+/*
+ * Lock failed (after delay) in ACC. Probably means somebody ahead of
+ * us in lock queue deleted the tuple.
+ */
+void
+Dbtux::execACCKEYREF(Signal* signal)
+{
+ jamEntry();
+ ScanOpPtr scanPtr;
+ scanPtr.i = signal->theData[0];
+ c_scanOpPool.getPtr(scanPtr);
+ ScanOp& scan = *scanPtr.p;
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "Lock refused scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+ ndbrequire(scan.m_lockwait && scan.m_accLockOp != RNIL);
+ scan.m_lockwait = false;
+ if (scan.m_state != ScanOp::Aborting) {
+ jam();
+ // release the operation
+ AccLockReq* const lockReq = (AccLockReq*)signal->getDataPtrSend();
+ lockReq->returnCode = RNIL;
+ lockReq->requestInfo = AccLockReq::Abort;
+ lockReq->accOpPtr = scan.m_accLockOp;
+ EXECUTE_DIRECT(DBACC, GSN_ACC_LOCKREQ, signal, AccLockReq::UndoSignalLength);
+ jamEntry();
+ ndbrequire(lockReq->returnCode == AccLockReq::Success);
+ scan.m_accLockOp = RNIL;
+ // scan position should already have been moved (assert only)
+ if (scan.m_state == ScanOp::Blocked) {
+ jam();
+ ndbassert(false);
+ scan.m_state = ScanOp::Next;
+ }
+ // LQH has the ball
+ return;
+ }
+ // continue at ACC_ABORTCONF
+}
+
+/*
+ * Received when scan is closing. This signal arrives after any
+ * ACCKEYCON or ACCKEYREF which may have been in job buffer.
+ */
+void
+Dbtux::execACC_ABORTCONF(Signal* signal)
+{
+ jamEntry();
+ ScanOpPtr scanPtr;
+ scanPtr.i = signal->theData[0];
+ c_scanOpPool.getPtr(scanPtr);
+ ScanOp& scan = *scanPtr.p;
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "ACC_ABORTCONF scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+ ndbrequire(scan.m_state == ScanOp::Aborting);
+ // most likely we are still in lock wait
+ if (scan.m_lockwait) {
+ jam();
+ scan.m_lockwait = false;
+ scan.m_accLockOp = RNIL;
+ }
+ scanClose(signal, scanPtr);
+}
+
+/*
+ * Find start position for single range scan. If it exists, sets state
+ * to Next and links the scan to the node. The first entry is returned
+ * by scanNext.
+ */
+void
+Dbtux::scanFirst(Signal* signal, ScanOpPtr scanPtr)
+{
+ ScanOp& scan = *scanPtr.p;
+ Frag& frag = *c_fragPool.getPtr(scan.m_fragPtrI);
+ TreeHead& tree = frag.m_tree;
+ if (tree.m_root == NullTupAddr) {
+ // tree may have become empty
+ jam();
+ scan.m_state = ScanOp::Last;
+ return;
+ }
+ TreePos pos;
+ pos.m_addr = tree.m_root;
+ NodeHandlePtr nodePtr;
+ // unpack lower bound
+ const ScanBound& bound = *scan.m_bound[0];
+ ScanBoundIterator iter;
+ bound.first(iter);
+ for (unsigned j = 0; j < bound.getSize(); j++) {
+ jam();
+ c_keyBuffer[j] = *iter.data;
+ bound.next(iter);
+ }
+ // comparison parameters
+ BoundPar boundPar;
+ boundPar.m_data1 = c_keyBuffer;
+ boundPar.m_count1 = scan.m_boundCnt[0];
+ boundPar.m_dir = 0;
+loop: {
+ jam();
+ selectNode(signal, frag, nodePtr, pos.m_addr, AccPref);
+ const unsigned occup = nodePtr.p->getOccup();
+ ndbrequire(occup != 0);
+ for (unsigned i = 0; i <= 1; i++) {
+ jam();
+ // compare prefix
+ boundPar.m_data2 = nodePtr.p->getPref(i);
+ boundPar.m_len2 = tree.m_prefSize;
+ int ret = cmpScanBound(frag, boundPar);
+ if (ret == NdbSqlUtil::CmpUnknown) {
+ jam();
+ // read full value
+ ReadPar readPar;
+ readPar.m_ent = nodePtr.p->getMinMax(i);
+ readPar.m_first = 0;
+ readPar.m_count = frag.m_numAttrs;
+ readPar.m_data = 0; // leave in signal data
+ tupReadAttrs(signal, frag, readPar);
+ // compare full value
+ boundPar.m_data2 = readPar.m_data;
+ boundPar.m_len2 = ZNIL; // big
+ ret = cmpScanBound(frag, boundPar);
+ ndbrequire(ret != NdbSqlUtil::CmpUnknown);
+ }
+ if (i == 0 && ret < 0) {
+ jam();
+ const TupAddr tupAddr = nodePtr.p->getLink(i);
+ if (tupAddr != NullTupAddr) {
+ jam();
+ // continue to left subtree
+ pos.m_addr = tupAddr;
+ goto loop;
+ }
+ // start scanning this node
+ pos.m_pos = 0;
+ pos.m_match = false;
+ pos.m_dir = 3;
+ scan.m_scanPos = pos;
+ scan.m_state = ScanOp::Next;
+ nodePtr.p->linkScan(scanPtr);
+ return;
+ }
+ if (i == 1 && ret > 0) {
+ jam();
+ const TupAddr tupAddr = nodePtr.p->getLink(i);
+ if (tupAddr != NullTupAddr) {
+ jam();
+ // continue to right subtree
+ pos.m_addr = tupAddr;
+ goto loop;
+ }
+ // start scanning upwards
+ pos.m_dir = 1;
+ scan.m_scanPos = pos;
+ scan.m_state = ScanOp::Next;
+ nodePtr.p->linkScan(scanPtr);
+ return;
+ }
+ }
+ // read rest of current node
+ accessNode(signal, frag, nodePtr, AccFull);
+ // look for first entry
+ ndbrequire(occup >= 2);
+ for (unsigned j = 1; j < occup; j++) {
+ jam();
+ ReadPar readPar;
+ readPar.m_ent = nodePtr.p->getEnt(j);
+ readPar.m_first = 0;
+ readPar.m_count = frag.m_numAttrs;
+ readPar.m_data = 0; // leave in signal data
+ tupReadAttrs(signal, frag, readPar);
+ // compare
+ boundPar.m_data2 = readPar.m_data;
+ boundPar.m_len2 = ZNIL; // big
+ int ret = cmpScanBound(frag, boundPar);
+ ndbrequire(ret != NdbSqlUtil::CmpUnknown);
+ if (ret < 0) {
+ jam();
+ // start scanning this node
+ pos.m_pos = j;
+ pos.m_match = false;
+ pos.m_dir = 3;
+ scan.m_scanPos = pos;
+ scan.m_state = ScanOp::Next;
+ nodePtr.p->linkScan(scanPtr);
+ return;
+ }
+ }
+ ndbrequire(false);
+ }
+}
+
+/*
+ * Move to next entry. The scan is already linked to some node. When
+ * we leave, if any entry was found, it will be linked to a possibly
+ * different node. The scan has a direction, one of:
+ *
+ * 0 - coming up from left child
+ * 1 - coming up from right child (proceed to parent immediately)
+ * 2 - coming up from root (the scan ends)
+ * 3 - left to right within node
+ * 4 - coming down from parent to left or right child
+ */
+void
+Dbtux::scanNext(Signal* signal, ScanOpPtr scanPtr)
+{
+ ScanOp& scan = *scanPtr.p;
+ Frag& frag = *c_fragPool.getPtr(scan.m_fragPtrI);
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "Next in scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+ if (scan.m_state == ScanOp::Locked) {
+ jam();
+ // version of a tuple locked by us cannot disappear (assert only)
+ ndbassert(false);
+ AccLockReq* const lockReq = (AccLockReq*)signal->getDataPtrSend();
+ lockReq->returnCode = RNIL;
+ lockReq->requestInfo = AccLockReq::Unlock;
+ lockReq->accOpPtr = scan.m_accLockOp;
+ EXECUTE_DIRECT(DBACC, GSN_ACC_LOCKREQ, signal, AccLockReq::UndoSignalLength);
+ jamEntry();
+ ndbrequire(lockReq->returnCode == AccLockReq::Success);
+ scan.m_accLockOp = RNIL;
+ scan.m_state = ScanOp::Current;
+ }
+ // unpack upper bound
+ const ScanBound& bound = *scan.m_bound[1];
+ ScanBoundIterator iter;
+ bound.first(iter);
+ for (unsigned j = 0; j < bound.getSize(); j++) {
+ jam();
+ c_keyBuffer[j] = *iter.data;
+ bound.next(iter);
+ }
+ // comparison parameters
+ BoundPar boundPar;
+ boundPar.m_data1 = c_keyBuffer;
+ boundPar.m_count1 = scan.m_boundCnt[1];
+ boundPar.m_dir = 1;
+ // use copy of position
+ TreePos pos = scan.m_scanPos;
+ // get and remember original node
+ NodeHandlePtr origNodePtr;
+ selectNode(signal, frag, origNodePtr, pos.m_addr, AccHead);
+ ndbrequire(origNodePtr.p->islinkScan(scanPtr));
+ // current node in loop
+ NodeHandlePtr nodePtr = origNodePtr;
+ while (true) {
+ jam();
+ if (pos.m_dir == 2) {
+ // coming up from root ends the scan
+ jam();
+ pos.m_addr = NullTupAddr;
+ scan.m_state = ScanOp::Last;
+ break;
+ }
+ if (nodePtr.p->m_addr != pos.m_addr) {
+ jam();
+ selectNode(signal, frag, nodePtr, pos.m_addr, AccHead);
+ }
+ if (pos.m_dir == 4) {
+ // coming down from parent proceed to left child
+ jam();
+ TupAddr addr = nodePtr.p->getLink(0);
+ if (addr != NullTupAddr) {
+ jam();
+ pos.m_addr = addr;
+ pos.m_dir = 4; // unchanged
+ continue;
+ }
+ // pretend we came from left child
+ pos.m_dir = 0;
+ }
+ if (pos.m_dir == 0) {
+ // coming from left child scan current node
+ jam();
+ pos.m_pos = 0;
+ pos.m_match = false;
+ pos.m_dir = 3;
+ }
+ if (pos.m_dir == 3) {
+ // within node
+ jam();
+ unsigned occup = nodePtr.p->getOccup();
+ ndbrequire(occup >= 1);
+ // access full node
+ accessNode(signal, frag, nodePtr, AccFull);
+ // advance position
+ if (! pos.m_match)
+ pos.m_match = true;
+ else
+ pos.m_pos++;
+ if (pos.m_pos < occup) {
+ jam();
+ pos.m_ent = nodePtr.p->getEnt(pos.m_pos);
+ pos.m_dir = 3; // unchanged
+ // XXX implement prefix optimization
+ ReadPar readPar;
+ readPar.m_ent = pos.m_ent;
+ readPar.m_first = 0;
+ readPar.m_count = frag.m_numAttrs;
+ readPar.m_data = 0; // leave in signal data
+ tupReadAttrs(signal, frag, readPar);
+ // compare
+ boundPar.m_data2 = readPar.m_data;
+ boundPar.m_len2 = ZNIL; // big
+ int ret = cmpScanBound(frag, boundPar);
+ ndbrequire(ret != NdbSqlUtil::CmpUnknown);
+ if (ret < 0) {
+ jam();
+ // hit upper bound of single range scan
+ pos.m_addr = NullTupAddr;
+ scan.m_state = ScanOp::Last;
+ break;
+ }
+ // can we see it
+ if (! scanVisible(signal, scanPtr, pos.m_ent)) {
+ jam();
+ continue;
+ }
+ // found entry
+ scan.m_state = ScanOp::Current;
+ break;
+ }
+ // after node proceed to right child
+ TupAddr addr = nodePtr.p->getLink(1);
+ if (addr != NullTupAddr) {
+ jam();
+ pos.m_addr = addr;
+ pos.m_dir = 4;
+ continue;
+ }
+ // pretend we came from right child
+ pos.m_dir = 1;
+ }
+ if (pos.m_dir == 1) {
+ // coming from right child proceed to parent
+ jam();
+ pos.m_addr = nodePtr.p->getLink(2);
+ pos.m_dir = nodePtr.p->getSide();
+ continue;
+ }
+ ndbrequire(false);
+ }
+ // copy back position
+ scan.m_scanPos = pos;
+ // relink
+ if (scan.m_state == ScanOp::Current) {
+ ndbrequire(pos.m_addr == nodePtr.p->m_addr);
+ if (origNodePtr.i != nodePtr.i) {
+ jam();
+ origNodePtr.p->unlinkScan(scanPtr);
+ nodePtr.p->linkScan(scanPtr);
+ }
+ } else if (scan.m_state == ScanOp::Last) {
+ jam();
+ ndbrequire(pos.m_addr == NullTupAddr);
+ origNodePtr.p->unlinkScan(scanPtr);
+ } else {
+ ndbrequire(false);
+ }
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "Next out scan " << scanPtr.i << " " << scan << endl;
+ }
+#endif
+}
+
+/*
+ * Check if an entry is visible to the scan.
+ *
+ * There is a special check to never return same tuple twice in a row.
+ * This is faster than asking TUP. It also fixes some special cases
+ * which are not analyzed or handled yet.
+ */
+bool
+Dbtux::scanVisible(Signal* signal, ScanOpPtr scanPtr, TreeEnt ent)
+{
+ TupQueryTh* const req = (TupQueryTh*)signal->getDataPtrSend();
+ const ScanOp& scan = *scanPtr.p;
+ const Frag& frag = *c_fragPool.getPtr(scan.m_fragPtrI);
+ /* Assign table, fragment, tuple address + version */
+ Uint32 tableId = frag.m_tableId;
+ Uint32 fragBit = ent.m_fragBit;
+ Uint32 fragId = frag.m_fragId | (fragBit << frag.m_fragOff);
+ Uint32 tupAddr = ent.m_tupAddr;
+ Uint32 tupVersion = ent.m_tupVersion;
+ /* Check for same tuple twice in row */
+ if (scan.m_lastEnt.m_tupAddr == tupAddr &&
+ scan.m_lastEnt.m_fragBit == fragBit) {
+ jam();
+ return false;
+ }
+ req->tableId = tableId;
+ req->fragId = fragId;
+ req->tupAddr = tupAddr;
+ req->tupVersion = tupVersion;
+ /* Assign transaction info, trans id + savepoint id */
+ Uint32 transId1 = scan.m_transId1;
+ Uint32 transId2 = scan.m_transId2;
+ Uint32 savePointId = scan.m_savePointId;
+ req->transId1 = transId1;
+ req->transId2 = transId2;
+ req->savePointId = savePointId;
+ EXECUTE_DIRECT(DBTUP, GSN_TUP_QUERY_TH, signal, TupQueryTh::SignalLength);
+ jamEntry();
+ return (bool)req->returnCode;
+}
+
+/*
+ * Finish closing of scan and send conf. Any lock wait has been done
+ * already.
+ */
+void
+Dbtux::scanClose(Signal* signal, ScanOpPtr scanPtr)
+{
+ ScanOp& scan = *scanPtr.p;
+ Frag& frag = *c_fragPool.getPtr(scanPtr.p->m_fragPtrI);
+ ndbrequire(! scan.m_lockwait && scan.m_accLockOp == RNIL);
+ // unlock all not unlocked by LQH
+ for (unsigned i = 0; i < MaxAccLockOps; i++) {
+ if (scan.m_accLockOps[i] != RNIL) {
+ jam();
+ AccLockReq* const lockReq = (AccLockReq*)signal->getDataPtrSend();
+ lockReq->returnCode = RNIL;
+ lockReq->requestInfo = AccLockReq::Abort;
+ lockReq->accOpPtr = scan.m_accLockOps[i];
+ EXECUTE_DIRECT(DBACC, GSN_ACC_LOCKREQ, signal, AccLockReq::UndoSignalLength);
+ jamEntry();
+ ndbrequire(lockReq->returnCode == AccLockReq::Success);
+ scan.m_accLockOps[i] = RNIL;
+ }
+ }
+ // send conf
+ NextScanConf* const conf = (NextScanConf*)signal->getDataPtrSend();
+ conf->scanPtr = scanPtr.p->m_userPtr;
+ conf->accOperationPtr = RNIL;
+ conf->fragId = RNIL;
+ unsigned signalLength = 3;
+ sendSignal(scanPtr.p->m_userRef, GSN_NEXT_SCANCONF,
+ signal, signalLength, JBB);
+ releaseScanOp(scanPtr);
+ commitNodes(signal, frag, true);
+}
+
+void
+Dbtux::addAccLockOp(ScanOp& scan, Uint32 accLockOp)
+{
+ ndbrequire(accLockOp != RNIL);
+ Uint32* list = scan.m_accLockOps;
+ bool ok = false;
+ for (unsigned i = 0; i < MaxAccLockOps; i++) {
+ ndbrequire(list[i] != accLockOp);
+ if (! ok && list[i] == RNIL) {
+ list[i] = accLockOp;
+ ok = true;
+ // continue check for duplicates
+ }
+ }
+ ndbrequire(ok);
+}
+
+void
+Dbtux::removeAccLockOp(ScanOp& scan, Uint32 accLockOp)
+{
+ ndbrequire(accLockOp != RNIL);
+ Uint32* list = scan.m_accLockOps;
+ bool ok = false;
+ for (unsigned i = 0; i < MaxAccLockOps; i++) {
+ if (list[i] == accLockOp) {
+ list[i] = RNIL;
+ ok = true;
+ break;
+ }
+ }
+ ndbrequire(ok);
+}
+
+/*
+ * Release allocated records.
+ */
+void
+Dbtux::releaseScanOp(ScanOpPtr& scanPtr)
+{
+#ifdef VM_TRACE
+ if (debugFlags & DebugScan) {
+ debugOut << "Release scan " << scanPtr.i << " " << *scanPtr.p << endl;
+ }
+#endif
+ Frag& frag = *c_fragPool.getPtr(scanPtr.p->m_fragPtrI);
+ scanPtr.p->m_boundMin.release();
+ scanPtr.p->m_boundMax.release();
+ // unlink from per-fragment list and release from pool
+ frag.m_scanList.release(scanPtr);
+}
diff --git a/ndb/src/kernel/blocks/dbtux/DbtuxTree.cpp b/ndb/src/kernel/blocks/dbtux/DbtuxTree.cpp
new file mode 100644
index 00000000000..860aa65414f
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/DbtuxTree.cpp
@@ -0,0 +1,755 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define DBTUX_TREE_CPP
+#include "Dbtux.hpp"
+
+/*
+ * Search for entry.
+ *
+ * Search key is index attribute data and tree entry value. Start from
+ * root node and compare the key to min/max of each node. Use linear
+ * search on the final (bounding) node. Initial attributes which are
+ * same in min/max need not be checked.
+ */
+void
+Dbtux::treeSearch(Signal* signal, Frag& frag, SearchPar searchPar, TreePos& treePos)
+{
+ const TreeHead& tree = frag.m_tree;
+ const unsigned numAttrs = frag.m_numAttrs;
+ treePos.m_addr = tree.m_root;
+ NodeHandlePtr nodePtr;
+ if (treePos.m_addr == NullTupAddr) {
+ // empty tree
+ jam();
+ treePos.m_pos = 0;
+ treePos.m_match = false;
+ return;
+ }
+loop: {
+ jam();
+ selectNode(signal, frag, nodePtr, treePos.m_addr, AccPref);
+ const unsigned occup = nodePtr.p->getOccup();
+ ndbrequire(occup != 0);
+ // number of equal initial attributes in bounding node
+ unsigned numEq = ZNIL;
+ for (unsigned i = 0; i <= 1; i++) {
+ jam();
+ // compare prefix
+ CmpPar cmpPar;
+ cmpPar.m_data1 = searchPar.m_data;
+ cmpPar.m_data2 = nodePtr.p->getPref(i);
+ cmpPar.m_len2 = tree.m_prefSize;
+ cmpPar.m_first = 0;
+ cmpPar.m_numEq = 0;
+ int ret = cmpTreeAttrs(frag, cmpPar);
+ if (ret == NdbSqlUtil::CmpUnknown) {
+ jam();
+ // read full value
+ ReadPar readPar;
+ readPar.m_ent = nodePtr.p->getMinMax(i);
+ ndbrequire(cmpPar.m_numEq < numAttrs);
+ readPar.m_first = cmpPar.m_numEq;
+ readPar.m_count = numAttrs - cmpPar.m_numEq;
+ readPar.m_data = 0; // leave in signal data
+ tupReadAttrs(signal, frag, readPar);
+ // compare full value
+ cmpPar.m_data2 = readPar.m_data;
+ cmpPar.m_len2 = ZNIL; // big
+ cmpPar.m_first = readPar.m_first;
+ ret = cmpTreeAttrs(frag, cmpPar);
+ ndbrequire(ret != NdbSqlUtil::CmpUnknown);
+ }
+ if (numEq > cmpPar.m_numEq)
+ numEq = cmpPar.m_numEq;
+ if (ret == 0) {
+ jam();
+ // keys are equal, compare entry values
+ ret = searchPar.m_ent.cmp(nodePtr.p->getMinMax(i));
+ }
+ if (i == 0 ? (ret < 0) : (ret > 0)) {
+ jam();
+ const TupAddr tupAddr = nodePtr.p->getLink(i);
+ if (tupAddr != NullTupAddr) {
+ jam();
+ // continue to left/right subtree
+ treePos.m_addr = tupAddr;
+ goto loop;
+ }
+ // position is immediately before/after this node
+ // XXX disallow second case
+ treePos.m_pos = (i == 0 ? 0 : occup);
+ treePos.m_match = false;
+ return;
+ }
+ if (ret == 0) {
+ jam();
+ // position is at first/last entry
+ treePos.m_pos = (i == 0 ? 0 : occup - 1);
+ treePos.m_match = true;
+ return;
+ }
+ }
+ // read rest of the bounding node
+ accessNode(signal, frag, nodePtr, AccFull);
+ // position is strictly within the node
+ ndbrequire(occup >= 2);
+ const unsigned numWithin = occup - 2;
+ for (unsigned j = 1; j <= numWithin; j++) {
+ jam();
+ int ret = 0;
+ // compare remaining attributes
+ if (numEq < numAttrs) {
+ jam();
+ ReadPar readPar;
+ readPar.m_ent = nodePtr.p->getEnt(j);
+ readPar.m_first = numEq;
+ readPar.m_count = numAttrs - numEq;
+ readPar.m_data = 0; // leave in signal data
+ tupReadAttrs(signal, frag, readPar);
+ // compare
+ CmpPar cmpPar;
+ cmpPar.m_data1 = searchPar.m_data;
+ cmpPar.m_data2 = readPar.m_data;
+ cmpPar.m_len2 = ZNIL; // big
+ cmpPar.m_first = readPar.m_first;
+ ret = cmpTreeAttrs(frag, cmpPar);
+ ndbrequire(ret != NdbSqlUtil::CmpUnknown);
+ }
+ if (ret == 0) {
+ jam();
+ // keys are equal, compare entry values
+ ret = searchPar.m_ent.cmp(nodePtr.p->getEnt(j));
+ }
+ if (ret <= 0) {
+ jam();
+ // position is before or at this entry
+ treePos.m_pos = j;
+ treePos.m_match = (ret == 0);
+ return;
+ }
+ }
+ // position is before last entry
+ treePos.m_pos = occup - 1;
+ treePos.m_match = false;
+ return;
+ }
+}
+
+/*
+ * Add entry.
+ */
+void
+Dbtux::treeAdd(Signal* signal, Frag& frag, TreePos treePos, TreeEnt ent)
+{
+ TreeHead& tree = frag.m_tree;
+ unsigned pos = treePos.m_pos;
+ NodeHandlePtr nodePtr;
+ // check for empty tree
+ if (treePos.m_addr == NullTupAddr) {
+ jam();
+ insertNode(signal, frag, nodePtr, AccPref);
+ nodePtr.p->pushUp(signal, 0, ent);
+ nodePtr.p->setSide(2);
+ tree.m_root = nodePtr.p->m_addr;
+ return;
+ }
+ // access full node
+ selectNode(signal, frag, nodePtr, treePos.m_addr, AccFull);
+ // check if it is bounding node
+ if (pos != 0 && pos != nodePtr.p->getOccup()) {
+ jam();
+ // check if room for one more
+ if (nodePtr.p->getOccup() < tree.m_maxOccup) {
+ jam();
+ nodePtr.p->pushUp(signal, pos, ent);
+ return;
+ }
+ // returns min entry
+ nodePtr.p->pushDown(signal, pos - 1, ent);
+ // find position to add the removed min entry
+ TupAddr childAddr = nodePtr.p->getLink(0);
+ if (childAddr == NullTupAddr) {
+ jam();
+ // left child will be added
+ pos = 0;
+ } else {
+ jam();
+ // find glb node
+ while (childAddr != NullTupAddr) {
+ jam();
+ selectNode(signal, frag, nodePtr, childAddr, AccHead);
+ childAddr = nodePtr.p->getLink(1);
+ }
+ // access full node again
+ accessNode(signal, frag, nodePtr, AccFull);
+ pos = nodePtr.p->getOccup();
+ }
+ // fall thru to next case
+ }
+ // adding new min or max
+ unsigned i = (pos == 0 ? 0 : 1);
+ ndbrequire(nodePtr.p->getLink(i) == NullTupAddr);
+ // check if the half-leaf/leaf has room for one more
+ if (nodePtr.p->getOccup() < tree.m_maxOccup) {
+ jam();
+ nodePtr.p->pushUp(signal, pos, ent);
+ return;
+ }
+ // add a new node
+ NodeHandlePtr childPtr;
+ insertNode(signal, frag, childPtr, AccPref);
+ childPtr.p->pushUp(signal, 0, ent);
+ // connect parent and child
+ nodePtr.p->setLink(i, childPtr.p->m_addr);
+ childPtr.p->setLink(2, nodePtr.p->m_addr);
+ childPtr.p->setSide(i);
+ // re-balance tree at each node
+ while (true) {
+ // height of subtree i has increased by 1
+ int j = (i == 0 ? -1 : +1);
+ int b = nodePtr.p->getBalance();
+ if (b == 0) {
+ // perfectly balanced
+ jam();
+ nodePtr.p->setBalance(j);
+ // height change propagates up
+ } else if (b == -j) {
+ // height of shorter subtree increased
+ jam();
+ nodePtr.p->setBalance(0);
+ // height of tree did not change - done
+ break;
+ } else if (b == j) {
+ // height of longer subtree increased
+ jam();
+ NodeHandlePtr childPtr;
+ selectNode(signal, frag, childPtr, nodePtr.p->getLink(i), AccHead);
+ int b2 = childPtr.p->getBalance();
+ if (b2 == b) {
+ jam();
+ treeRotateSingle(signal, frag, nodePtr, i);
+ } else if (b2 == -b) {
+ jam();
+ treeRotateDouble(signal, frag, nodePtr, i);
+ } else {
+ // height of subtree increased so it cannot be perfectly balanced
+ ndbrequire(false);
+ }
+ // height of tree did not increase - done
+ break;
+ } else {
+ ndbrequire(false);
+ }
+ TupAddr parentAddr = nodePtr.p->getLink(2);
+ if (parentAddr == NullTupAddr) {
+ jam();
+ // root node - done
+ break;
+ }
+ i = nodePtr.p->getSide();
+ selectNode(signal, frag, nodePtr, parentAddr, AccHead);
+ }
+}
+
+/*
+ * Remove entry.
+ */
+void
+Dbtux::treeRemove(Signal* signal, Frag& frag, TreePos treePos)
+{
+ TreeHead& tree = frag.m_tree;
+ unsigned pos = treePos.m_pos;
+ NodeHandlePtr nodePtr;
+ // access full node
+ selectNode(signal, frag, nodePtr, treePos.m_addr, AccFull);
+ TreeEnt ent;
+ // check interior node first
+ if (nodePtr.p->getChilds() == 2) {
+ jam();
+ ndbrequire(nodePtr.p->getOccup() >= tree.m_minOccup);
+ // check if no underflow
+ if (nodePtr.p->getOccup() > tree.m_minOccup) {
+ jam();
+ nodePtr.p->popDown(signal, pos, ent);
+ return;
+ }
+ // save current handle
+ NodeHandlePtr parentPtr = nodePtr;
+ // find glb node
+ TupAddr childAddr = nodePtr.p->getLink(0);
+ while (childAddr != NullTupAddr) {
+ jam();
+ selectNode(signal, frag, nodePtr, childAddr, AccHead);
+ childAddr = nodePtr.p->getLink(1);
+ }
+ // access full node again
+ accessNode(signal, frag, nodePtr, AccFull);
+ // use glb max as new parent min
+ ent = nodePtr.p->getEnt(nodePtr.p->getOccup() - 1);
+ parentPtr.p->popUp(signal, pos, ent);
+ // set up to remove glb max
+ pos = nodePtr.p->getOccup() - 1;
+ // fall thru to next case
+ }
+ // remove the element
+ nodePtr.p->popDown(signal, pos, ent);
+ ndbrequire(nodePtr.p->getChilds() <= 1);
+ // handle half-leaf
+ for (unsigned i = 0; i <= 1; i++) {
+ jam();
+ TupAddr childAddr = nodePtr.p->getLink(i);
+ if (childAddr != NullTupAddr) {
+ // move to child
+ selectNode(signal, frag, nodePtr, childAddr, AccFull);
+ // balance of half-leaf parent requires child to be leaf
+ break;
+ }
+ }
+ ndbrequire(nodePtr.p->getChilds() == 0);
+ // get parent if any
+ TupAddr parentAddr = nodePtr.p->getLink(2);
+ NodeHandlePtr parentPtr;
+ unsigned i = nodePtr.p->getSide();
+ // move all that fits into parent
+ if (parentAddr != NullTupAddr) {
+ jam();
+ selectNode(signal, frag, parentPtr, nodePtr.p->getLink(2), AccFull);
+ parentPtr.p->slide(signal, nodePtr, i);
+ // fall thru to next case
+ }
+ // non-empty leaf
+ if (nodePtr.p->getOccup() >= 1) {
+ jam();
+ return;
+ }
+ // remove empty leaf
+ deleteNode(signal, frag, nodePtr);
+ if (parentAddr == NullTupAddr) {
+ jam();
+ // tree is now empty
+ tree.m_root = NullTupAddr;
+ return;
+ }
+ nodePtr = parentPtr;
+ nodePtr.p->setLink(i, NullTupAddr);
+#ifdef dbtux_min_occup_less_max_occup
+ // check if we created a half-leaf
+ if (nodePtr.p->getBalance() == 0) {
+ jam();
+ // move entries from the other child
+ TupAddr childAddr = nodePtr.p->getLink(1 - i);
+ NodeHandlePtr childPtr;
+ selectNode(signal, frag, childPtr, childAddr, AccFull);
+ nodePtr.p->slide(signal, childPtr, 1 - i);
+ if (childPtr.p->getOccup() == 0) {
+ jam();
+ deleteNode(signal, frag, childPtr);
+ nodePtr.p->setLink(1 - i, NullTupAddr);
+ // we are balanced again but our parent balance changes by -1
+ parentAddr = nodePtr.p->getLink(2);
+ if (parentAddr == NullTupAddr) {
+ jam();
+ return;
+ }
+ // fix side and become parent
+ i = nodePtr.p->getSide();
+ selectNode(signal, frag, nodePtr, parentAddr, AccHead);
+ }
+ }
+#endif
+ // re-balance tree at each node
+ while (true) {
+ // height of subtree i has decreased by 1
+ int j = (i == 0 ? -1 : +1);
+ int b = nodePtr.p->getBalance();
+ if (b == 0) {
+ // perfectly balanced
+ jam();
+ nodePtr.p->setBalance(-j);
+ // height of tree did not change - done
+ return;
+ } else if (b == j) {
+ // height of longer subtree has decreased
+ jam();
+ nodePtr.p->setBalance(0);
+ // height change propagates up
+ } else if (b == -j) {
+ // height of shorter subtree has decreased
+ jam();
+ NodeHandlePtr childPtr;
+ // child on the other side
+ selectNode(signal, frag, childPtr, nodePtr.p->getLink(1 - i), AccHead);
+ int b2 = childPtr.p->getBalance();
+ if (b2 == b) {
+ jam();
+ treeRotateSingle(signal, frag, nodePtr, 1 - i);
+ // height of tree decreased and propagates up
+ } else if (b2 == -b) {
+ jam();
+ treeRotateDouble(signal, frag, nodePtr, 1 - i);
+ // height of tree decreased and propagates up
+ } else {
+ jam();
+ treeRotateSingle(signal, frag, nodePtr, 1 - i);
+ // height of tree did not change - done
+ return;
+ }
+ } else {
+ ndbrequire(false);
+ }
+ TupAddr parentAddr = nodePtr.p->getLink(2);
+ if (parentAddr == NullTupAddr) {
+ jam();
+ // root node - done
+ return;
+ }
+ i = nodePtr.p->getSide();
+ selectNode(signal, frag, nodePtr, parentAddr, AccHead);
+ }
+}
+
+/*
+ * Single rotation about node 5. One of LL (i=0) or RR (i=1).
+ *
+ * 0 0
+ * | |
+ * 5 ==> 3
+ * / \ / \
+ * 3 6 2 5
+ * / \ / / \
+ * 2 4 1 4 6
+ * /
+ * 1
+ *
+ * In this change 5,3 and 2 must always be there. 0, 1, 2, 4 and 6 are
+ * all optional. If 4 are there it changes side.
+*/
+void
+Dbtux::treeRotateSingle(Signal* signal,
+ Frag& frag,
+ NodeHandlePtr& nodePtr,
+ unsigned i)
+{
+ ndbrequire(i <= 1);
+ /*
+ 5 is the old top node that have been unbalanced due to an insert or
+ delete. The balance is still the old balance before the update.
+ Verify that n5Bal is 1 if RR rotate and -1 if LL rotate.
+ */
+ NodeHandlePtr n5Ptr = nodePtr;
+ const TupAddr n5Addr = n5Ptr.p->m_addr;
+ const int n5Bal = n5Ptr.p->getBalance();
+ const int n5side = n5Ptr.p->getSide();
+ ndbrequire(n5Bal + (1 - i) == i);
+ /*
+ 3 is the new root of this part of the tree which is to swap place with
+ node 5. For an insert to cause this it must have the same balance as 5.
+ For deletes it can have the balance 0.
+ */
+ TupAddr n3Addr = n5Ptr.p->getLink(i);
+ NodeHandlePtr n3Ptr;
+ selectNode(signal, frag, n3Ptr, n3Addr, AccHead);
+ const int n3Bal = n3Ptr.p->getBalance();
+ /*
+ 2 must always be there but is not changed. Thus we mereley check that it
+ exists.
+ */
+ ndbrequire(n3Ptr.p->getLink(i) != NullTupAddr);
+ /*
+ 4 is not necessarily there but if it is there it will move from one
+ side of 3 to the other side of 5. For LL it moves from the right side
+ to the left side and for RR it moves from the left side to the right
+ side. This means that it also changes parent from 3 to 5.
+ */
+ TupAddr n4Addr = n3Ptr.p->getLink(1 - i);
+ NodeHandlePtr n4Ptr;
+ if (n4Addr != NullTupAddr) {
+ jam();
+ selectNode(signal, frag, n4Ptr, n4Addr, AccHead);
+ ndbrequire(n4Ptr.p->getSide() == (1 - i) &&
+ n4Ptr.p->getLink(2) == n3Addr);
+ n4Ptr.p->setSide(i);
+ n4Ptr.p->setLink(2, n5Addr);
+ }//if
+
+ /*
+ Retrieve the address of 5's parent before it is destroyed
+ */
+ TupAddr n0Addr = n5Ptr.p->getLink(2);
+
+ /*
+ The next step is to perform the rotation. 3 will inherit 5's parent
+ and side. 5 will become a child of 3 on the right side for LL and on
+ the left side for RR.
+ 5 will get 3 as the parent. It will get 4 as a child and it will be
+ on the right side of 3 for LL and left side of 3 for RR.
+ The final step of the rotate is to check whether 5 originally had any
+ parent. If it had not then 3 is the new root node.
+ We will also verify some preconditions for the change to occur.
+ 1. 3 must have had 5 as parent before the change.
+ 2. 3's side is left for LL and right for RR before change.
+ */
+ ndbrequire(n3Ptr.p->getLink(2) == n5Addr);
+ ndbrequire(n3Ptr.p->getSide() == i);
+ n3Ptr.p->setLink(1 - i, n5Addr);
+ n3Ptr.p->setLink(2, n0Addr);
+ n3Ptr.p->setSide(n5side);
+ n5Ptr.p->setLink(i, n4Addr);
+ n5Ptr.p->setLink(2, n3Addr);
+ n5Ptr.p->setSide(1 - i);
+ if (n0Addr != NullTupAddr) {
+ jam();
+ NodeHandlePtr n0Ptr;
+ selectNode(signal, frag, n0Ptr, n0Addr, AccHead);
+ n0Ptr.p->setLink(n5side, n3Addr);
+ } else {
+ jam();
+ frag.m_tree.m_root = n3Addr;
+ }//if
+ /* The final step of the change is to update the balance of 3 and
+ 5 that changed places. There are two cases here. The first case is
+ when 3 unbalanced in the same direction by an insert or a delete.
+ In this case the changes will make the tree balanced again for both
+ 3 and 5.
+ The second case only occurs at deletes. In this case 3 starts out
+ balanced. In the figure above this could occur if 4 starts out with
+ a right node and the rotate is triggered by a delete of 6's only child.
+ In this case 5 will change balance but still be unbalanced and 3 will
+ be unbalanced in the opposite direction of 5.
+ */
+ if (n3Bal == n5Bal) {
+ jam();
+ n3Ptr.p->setBalance(0);
+ n5Ptr.p->setBalance(0);
+ } else if (n3Bal == 0) {
+ jam();
+ n3Ptr.p->setBalance(-n5Bal);
+ n5Ptr.p->setBalance(n5Bal);
+ } else {
+ ndbrequire(false);
+ }//if
+ /*
+ Set nodePtr to 3 as return parameter for enabling caller to continue
+ traversing the tree.
+ */
+ nodePtr = n3Ptr;
+}
+
+/*
+ * Double rotation about node 6. One of LR (i=0) or RL (i=1).
+ *
+ * 0 0
+ * | |
+ * 6 ==> 4
+ * / \ / \
+ * 2 7 2 6
+ * / \ / \ / \
+ * 1 4 1 3 5 7
+ * / \
+ * 3 5
+ *
+ * In this change 6, 2 and 4 must be there, all others are optional.
+ * We will start by proving a Lemma.
+ * Lemma:
+ * The height of the sub-trees 1 and 7 and the maximum height of the
+ * threes from 3 and 5 are all the same.
+ * Proof:
+ * maxheight(3,5) is defined as the maximum height of 3 and 5.
+ * If height(7) > maxheight(3,5) then the AVL condition is ok and we
+ * don't need to perform a rotation.
+ * If height(7) < maxheight(3,5) then the balance of 6 would be at least
+ * -3 which cannot happen in an AVL tree even before a rotation.
+ * Thus we conclude that height(7) == maxheight(3,5)
+ *
+ * The next step is to prove that the height of 1 is equal to maxheight(3,5).
+ * If height(1) - 1 > maxheight(3,5) then we would have
+ * balance in 6 equal to -3 at least which cannot happen in an AVL-tree.
+ * If height(1) - 1 = maxheight(3,5) then we should have solved the
+ * unbalance with a single rotate and not with a double rotate.
+ * If height(1) + 1 = maxheight(3,5) then we would be doing a rotate
+ * with node 2 as the root of the rotation.
+ * If height(1) + k = maxheight(3,5) where k >= 2 then the tree could not have
+ * been an AVL-tree before the insert or delete.
+ * Thus we conclude that height(1) = maxheight(3,5)
+ *
+ * Thus we conclude that height(1) = maxheight(3,5) = height(7).
+ *
+ * Observation:
+ * The balance of node 4 before the rotation can be any (-1, 0, +1).
+ *
+ * The following changes are needed:
+ * Node 6:
+ * 1) Changes parent from 0 -> 4
+ * 2) 1 - i link stays the same
+ * 3) i side link is derived from 1 - i side link from 4
+ * 4) Side is set to 1 - i
+ * 5) Balance change:
+ * If balance(4) == 0 then balance(6) = 0
+ * since height(3) = height(5) = maxheight(3,5) = height(7)
+ * If balance(4) == +1 then balance(6) = 0
+ * since height(5) = maxheight(3,5) = height(7)
+ * If balance(4) == -1 then balance(6) = 1
+ * since height(5) + 1 = maxheight(3,5) = height(7)
+ *
+ * Node 2:
+ * 1) Changes parent from 6 -> 4
+ * 2) i side link stays the same
+ * 3) 1 - i side link is derived from i side link of 4
+ * 4) Side is set to i (thus not changed)
+ * 5) Balance change:
+ * If balance(4) == 0 then balance(2) = 0
+ * since height(3) = height(5) = maxheight(3,5) = height(1)
+ * If balance(4) == -1 then balance(2) = 0
+ * since height(3) = maxheight(3,5) = height(1)
+ * If balance(4) == +1 then balance(6) = 1
+ * since height(3) + 1 = maxheight(3,5) = height(1)
+ *
+ * Node 4:
+ * 1) Inherits parent from 6
+ * 2) i side link is 2
+ * 3) 1 - i side link is 6
+ * 4) Side is inherited from 6
+ * 5) Balance(4) = 0 independent of previous balance
+ * Proof:
+ * If height(1) = 0 then only 2, 4 and 6 are involved and then it is
+ * trivially true.
+ * If height(1) >= 1 then we are sure that 1 and 7 exist with the same
+ * height and that if 3 and 5 exist they are of the same height as 1 and
+ * 7 and thus we know that 4 is balanced since newheight(2) = newheight(6).
+ *
+ * If Node 3 exists:
+ * 1) Change parent from 4 to 2
+ * 2) Change side from i to 1 - i
+ *
+ * If Node 5 exists:
+ * 1) Change parent from 4 to 6
+ * 2) Change side from 1 - i to i
+ *
+ * If Node 0 exists:
+ * 1) previous link to 6 is replaced by link to 4 on proper side
+ *
+ * Node 1 and 7 needs no changes at all.
+ *
+ * Some additional requires are that balance(2) = - balance(6) = -1/+1 since
+ * otherwise we would do a single rotate.
+ *
+ * The balance(6) is -1 if i == 0 and 1 if i == 1
+ *
+ */
+void
+Dbtux::treeRotateDouble(Signal* signal, Frag& frag, NodeHandlePtr& nodePtr, unsigned i)
+{
+ // old top node
+ NodeHandlePtr n6Ptr = nodePtr;
+ const TupAddr n6Addr = n6Ptr.p->m_addr;
+ // the un-updated balance
+ const int n6Bal = n6Ptr.p->getBalance();
+ const unsigned n6Side = n6Ptr.p->getSide();
+
+ // level 1
+ TupAddr n2Addr = n6Ptr.p->getLink(i);
+ NodeHandlePtr n2Ptr;
+ selectNode(signal, frag, n2Ptr, n2Addr, AccHead);
+ const int n2Bal = n2Ptr.p->getBalance();
+
+ // level 2
+ TupAddr n4Addr = n2Ptr.p->getLink(1 - i);
+ NodeHandlePtr n4Ptr;
+ selectNode(signal, frag, n4Ptr, n4Addr, AccHead);
+ const int n4Bal = n4Ptr.p->getBalance();
+
+ ndbrequire(i <= 1);
+ ndbrequire(n6Bal + (1 - i) == i);
+ ndbrequire(n2Bal == -n6Bal);
+ ndbrequire(n2Ptr.p->getLink(2) == n6Addr);
+ ndbrequire(n2Ptr.p->getSide() == i);
+ ndbrequire(n4Ptr.p->getLink(2) == n2Addr);
+
+ // level 3
+ TupAddr n3Addr = n4Ptr.p->getLink(i);
+ TupAddr n5Addr = n4Ptr.p->getLink(1 - i);
+
+ // fill up leaf before it becomes internal
+ if (n3Addr == NullTupAddr && n5Addr == NullTupAddr) {
+ jam();
+ TreeHead& tree = frag.m_tree;
+ accessNode(signal, frag, n2Ptr, AccFull);
+ accessNode(signal, frag, n4Ptr, AccFull);
+ n4Ptr.p->slide(signal, n2Ptr, i);
+ // implied by rule of merging half-leaves with leaves
+ ndbrequire(n4Ptr.p->getOccup() >= tree.m_minOccup);
+ ndbrequire(n2Ptr.p->getOccup() != 0);
+ } else {
+ if (n3Addr != NullTupAddr) {
+ jam();
+ NodeHandlePtr n3Ptr;
+ selectNode(signal, frag, n3Ptr, n3Addr, AccHead);
+ n3Ptr.p->setLink(2, n2Addr);
+ n3Ptr.p->setSide(1 - i);
+ }
+ if (n5Addr != NullTupAddr) {
+ jam();
+ NodeHandlePtr n5Ptr;
+ selectNode(signal, frag, n5Ptr, n5Addr, AccHead);
+ n5Ptr.p->setLink(2, n6Ptr.p->m_addr);
+ n5Ptr.p->setSide(i);
+ }
+ }
+ // parent
+ TupAddr n0Addr = n6Ptr.p->getLink(2);
+ NodeHandlePtr n0Ptr;
+ // perform the rotation
+ n6Ptr.p->setLink(i, n5Addr);
+ n6Ptr.p->setLink(2, n4Addr);
+ n6Ptr.p->setSide(1 - i);
+
+ n2Ptr.p->setLink(1 - i, n3Addr);
+ n2Ptr.p->setLink(2, n4Addr);
+
+ n4Ptr.p->setLink(i, n2Addr);
+ n4Ptr.p->setLink(1 - i, n6Addr);
+ n4Ptr.p->setLink(2, n0Addr);
+ n4Ptr.p->setSide(n6Side);
+
+ if (n0Addr != NullTupAddr) {
+ jam();
+ selectNode(signal, frag, n0Ptr, n0Addr, AccHead);
+ n0Ptr.p->setLink(n6Side, n4Addr);
+ } else {
+ jam();
+ frag.m_tree.m_root = n4Addr;
+ }
+ // set balance of changed nodes
+ n4Ptr.p->setBalance(0);
+ if (n4Bal == 0) {
+ jam();
+ n2Ptr.p->setBalance(0);
+ n6Ptr.p->setBalance(0);
+ } else if (n4Bal == -n2Bal) {
+ jam();
+ n2Ptr.p->setBalance(0);
+ n6Ptr.p->setBalance(n2Bal);
+ } else if (n4Bal == n2Bal) {
+ jam();
+ n2Ptr.p->setBalance(-n2Bal);
+ n6Ptr.p->setBalance(0);
+ } else {
+ ndbrequire(false);
+ }
+ // new top node
+ nodePtr = n4Ptr;
+}
diff --git a/ndb/src/kernel/blocks/dbtux/Makefile b/ndb/src/kernel/blocks/dbtux/Makefile
new file mode 100644
index 00000000000..30927c31848
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/Makefile
@@ -0,0 +1,17 @@
+include .defs.mk
+
+TYPE = kernel
+
+ARCHIVE_TARGET = dbtux
+
+SOURCES = \
+ DbtuxGen.cpp \
+ DbtuxMeta.cpp \
+ DbtuxMaint.cpp \
+ DbtuxNode.cpp \
+ DbtuxTree.cpp \
+ DbtuxScan.cpp \
+ DbtuxCmp.cpp \
+ DbtuxDebug.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/dbtux/tuxstatus.html b/ndb/src/kernel/blocks/dbtux/tuxstatus.html
new file mode 100644
index 00000000000..264809cefd3
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbtux/tuxstatus.html
@@ -0,0 +1,120 @@
+<HTML>
+<HEAD>
+<META HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=iso-8859-1">
+<TITLE>NDB Ordered Index Status</TITLE>
+</HEAD>
+<BODY LINK="#0000ff" VLINK="#800080" BGCOLOR="#ffffff">
+<p>
+<h2>NDB Ordered Index Status</h2>
+<p>
+<h3>Alpha release Jan 30, 2004</h3>
+<p>
+<ul>
+ <li>
+ Up to 32 index attributes of any type, possibly nullable.
+ <li>
+ Index build i.e. table need not be empty.
+ <li>
+ Logging NOT done, index rebuilt at system restart.
+ <li>
+ Single range scan with lower and upper bounds.
+ <li>
+ Scan with locking: read latest, read for update.
+ <li>
+ LIMITED number of parallel scans.
+ <li>
+ Total result set NOT in index key order.
+ <li>
+ NDB ODBC optimizer to use ordered index for equality but NOT for ranges.
+ <li>
+ MySQL optimizer to use ordered index for equality and ranges.
+</ul>
+<p>
+As an example, consider following index on integer attributes.
+<p>
+<tt>SQL&gt;create index X on T (A, B, C) nologging;</tt>
+<p>
+Single range scan means that bounds are set on
+an initial sequence of index keys, and all but last is an equality.
+<br>
+For example following scans are supported (the last 2 not via NDB ODBC).
+<p>
+<tt>SQL&gt;select * from T where A = 1;</tt>
+<br>
+<tt>SQL&gt;select * from T where A = 1 and B = 10 and C = 20;</tt>
+<br>
+<tt>SQL&gt;select * from T where A &lt; 10;</tt>
+<br>
+<tt>SQL&gt;select * from T where A = 1 and 10 &lt; B and B &lt; 20;</tt>
+<p>
+Following scans are NOT supported:
+<p>
+<tt>SQL&gt;select * from T where B = 1;</tt>
+<br>
+<tt>SQL&gt;select * from T where A &lt; 10 and B &lt; 20;</tt>
+<br>
+<h3>Features and dates</h3>
+[ Now = Jan 19 ]
+<p>
+<table border=1 cellpadding=1>
+<tr align="left">
+ <th width="40%">Feature</th>
+ <th width="15%">Now</th> <th width="15%">Jan 30</th> <th width="15%">Mar 01</th> <th width="15%">Never</th>
+</tr>
+<tr align=left>
+ <td>Index maintenance</td>
+ <td>X</td> <td>-</td> <td>-</td> <td>-</td>
+</tr>
+<tr align=left>
+ <td>Basic scan</td>
+ <td>X 1)</td> <td>-</td> <td>-</td> <td>-</td>
+</tr>
+<tr align=left>
+ <td>Scan bounds on nullable attributes</td>
+ <td>-</td> <td>X</td> <td>-</td> <td>-</td>
+</tr>
+<tr align=left>
+ <td>Scan with locking</td>
+ <td>-</td> <td>X</td> <td>-</td> <td>-</td>
+</tr>
+<tr align="left">
+ <td>NDB ODBC equality bounds</td>
+ <td>-</td> <td>X</td> <td>-</td> <td>-</td>
+</tr>
+<tr align="left">
+ <td>MySQL integration</td>
+ <td>-</td> <td>X</td> <td>-</td> <td>-</td>
+</tr>
+<tr align=left>
+ <td>Index build</td>
+ <td>2)</td> <td>X</td> <td>-</td> <td>-</td>
+</tr>
+<tr align=left>
+ <td>Unlimited number of scans</td>
+ <td>3)</td> <td>-</td> <td>X</td> <td>-</td>
+</tr>
+<tr align=left>
+ <td>Total ordering</td>
+ <td>-</td> <td>-</td> <td>X</td> <td>-</td>
+</tr>
+<tr align=left>
+ <td>Multiple range scan</td>
+ <td>-</td> <td>-</td> <td>X</td> <td>-</td>
+</tr>
+<tr align="left">
+ <td>NDB ODBC range bounds</td>
+ <td>-</td> <td>-</td> <td>-</td> <td>X</td>
+</tr>
+<tr align=left>
+ <td>Logging</td>
+ <td>-</td> <td>-</td> <td>-</td> <td>X</td>
+</tr>
+</table>
+<p>
+1) No locking and bounds must be on non-nullable key attributes.
+<br>
+2) Currently table must be empty when index is created.
+<br>
+3) Currently limited to 11 simultaneous per fragment.
+</BODY>
+</HTML>
diff --git a/ndb/src/kernel/blocks/dbutil/DbUtil.cpp b/ndb/src/kernel/blocks/dbutil/DbUtil.cpp
new file mode 100644
index 00000000000..3936a211d4b
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbutil/DbUtil.cpp
@@ -0,0 +1,2628 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "DbUtil.hpp"
+
+#include <new>
+#include <stddef.h>
+#include <ndb_version.h>
+
+#include <signaldata/WaitGCP.hpp>
+#include <signaldata/KeyInfo.hpp>
+#include <signaldata/AttrInfo.hpp>
+#include <signaldata/TcKeyConf.hpp>
+#include <signaldata/TcKeyFailConf.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/DictTabInfo.hpp>
+
+#include <signaldata/UtilSequence.hpp>
+#include <signaldata/UtilPrepare.hpp>
+#include <signaldata/UtilRelease.hpp>
+#include <signaldata/UtilExecute.hpp>
+#include <signaldata/UtilLock.hpp>
+
+#include <SectionReader.hpp>
+#include <Interpreter.hpp>
+#include <AttributeHeader.hpp>
+
+#include <NdbTick.h>
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: Startup
+ * ------------------------------------------------------------------------
+ *
+ * Constructors, startup, initializations
+ **************************************************************************/
+
+DbUtil::DbUtil(const Configuration & conf) :
+ SimulatedBlock(DBUTIL, conf),
+ c_runningPrepares(c_preparePool),
+ c_runningPreparedOperations(c_preparedOperationPool),
+ c_seizingTransactions(c_transactionPool),
+ c_runningTransactions(c_transactionPool),
+ c_lockQueues(c_lockQueuePool)
+{
+ BLOCK_CONSTRUCTOR(DbUtil);
+
+ // Add received signals
+ addRecSignal(GSN_STTOR, &DbUtil::execSTTOR);
+ addRecSignal(GSN_NDB_STTOR, &DbUtil::execNDB_STTOR);
+ addRecSignal(GSN_DUMP_STATE_ORD, &DbUtil::execDUMP_STATE_ORD);
+ addRecSignal(GSN_CONTINUEB, &DbUtil::execCONTINUEB);
+
+ //addRecSignal(GSN_TCSEIZEREF, &DbUtil::execTCSEIZEREF);
+ addRecSignal(GSN_TCSEIZECONF, &DbUtil::execTCSEIZECONF);
+ addRecSignal(GSN_TCKEYCONF, &DbUtil::execTCKEYCONF);
+ addRecSignal(GSN_TCKEYREF, &DbUtil::execTCKEYREF);
+ addRecSignal(GSN_TCROLLBACKREP, &DbUtil::execTCROLLBACKREP);
+
+ //addRecSignal(GSN_TCKEY_FAILCONF, &DbUtil::execTCKEY_FAILCONF);
+ //addRecSignal(GSN_TCKEY_FAILREF, &DbUtil::execTCKEY_FAILREF);
+ addRecSignal(GSN_TRANSID_AI, &DbUtil::execTRANSID_AI);
+
+ /**
+ * Sequence Service
+ */
+ addRecSignal(GSN_UTIL_SEQUENCE_REQ, &DbUtil::execUTIL_SEQUENCE_REQ);
+ // Debug
+ addRecSignal(GSN_UTIL_SEQUENCE_REF, &DbUtil::execUTIL_SEQUENCE_REF);
+ addRecSignal(GSN_UTIL_SEQUENCE_CONF, &DbUtil::execUTIL_SEQUENCE_CONF);
+
+ /**
+ * Locking
+ */
+ addRecSignal(GSN_UTIL_CREATE_LOCK_REQ, &DbUtil::execUTIL_CREATE_LOCK_REQ);
+ addRecSignal(GSN_UTIL_DESTROY_LOCK_REQ, &DbUtil::execUTIL_DESTORY_LOCK_REQ);
+ addRecSignal(GSN_UTIL_LOCK_REQ, &DbUtil::execUTIL_LOCK_REQ);
+ addRecSignal(GSN_UTIL_UNLOCK_REQ, &DbUtil::execUTIL_UNLOCK_REQ);
+
+ /**
+ * Backend towards Dict
+ */
+ addRecSignal(GSN_GET_TABINFOREF, &DbUtil::execGET_TABINFOREF);
+ addRecSignal(GSN_GET_TABINFO_CONF, &DbUtil::execGET_TABINFO_CONF);
+
+ /**
+ * Prepare / Execute / Release Services
+ */
+ addRecSignal(GSN_UTIL_PREPARE_REQ, &DbUtil::execUTIL_PREPARE_REQ);
+ addRecSignal(GSN_UTIL_PREPARE_CONF, &DbUtil::execUTIL_PREPARE_CONF);
+ addRecSignal(GSN_UTIL_PREPARE_REF, &DbUtil::execUTIL_PREPARE_REF);
+
+ addRecSignal(GSN_UTIL_EXECUTE_REQ, &DbUtil::execUTIL_EXECUTE_REQ);
+ addRecSignal(GSN_UTIL_EXECUTE_CONF, &DbUtil::execUTIL_EXECUTE_CONF);
+ addRecSignal(GSN_UTIL_EXECUTE_REF, &DbUtil::execUTIL_EXECUTE_REF);
+
+ addRecSignal(GSN_UTIL_RELEASE_REQ, &DbUtil::execUTIL_RELEASE_REQ);
+ addRecSignal(GSN_UTIL_RELEASE_CONF, &DbUtil::execUTIL_RELEASE_CONF);
+ addRecSignal(GSN_UTIL_RELEASE_REF, &DbUtil::execUTIL_RELEASE_REF);
+
+ c_pagePool.setSize(100);
+ c_preparePool.setSize(1); // one parallel prepare at a time
+ c_preparedOperationPool.setSize(5); // three hardcoded, two for test
+ c_operationPool.setSize(64); // 64 parallel operations
+ c_transactionPool.setSize(32); // 16 parallel transactions
+ c_attrMappingPool.setSize(100);
+ c_dataBufPool.setSize(6000); // 6000*11*4 = 264K > 8k+8k*16 = 256k
+ {
+ SLList<Prepare> tmp(c_preparePool);
+ PreparePtr ptr;
+ while(tmp.seize(ptr))
+ new (ptr.p) Prepare(c_pagePool);
+ tmp.release();
+ }
+ {
+ SLList<Operation> tmp(c_operationPool);
+ OperationPtr ptr;
+ while(tmp.seize(ptr))
+ new (ptr.p) Operation(c_dataBufPool, c_dataBufPool, c_dataBufPool);
+ tmp.release();
+ }
+ {
+ SLList<PreparedOperation> tmp(c_preparedOperationPool);
+ PreparedOperationPtr ptr;
+ while(tmp.seize(ptr))
+ new (ptr.p) PreparedOperation(c_attrMappingPool,
+ c_dataBufPool, c_dataBufPool);
+ tmp.release();
+ }
+ {
+ SLList<Transaction> tmp(c_transactionPool);
+ TransactionPtr ptr;
+ while(tmp.seize(ptr))
+ new (ptr.p) Transaction(c_pagePool, c_operationPool);
+ tmp.release();
+ }
+
+ c_lockQueuePool.setSize(5);
+ c_lockElementPool.setSize(5);
+ c_lockQueues.setSize(8);
+}
+
+DbUtil::~DbUtil()
+{
+}
+
+BLOCK_FUNCTIONS(DbUtil);
+
+void
+DbUtil::releasePrepare(PreparePtr prepPtr) {
+ prepPtr.p->preparePages.release();
+ c_runningPrepares.release(prepPtr); // Automatic release in pool
+}
+
+void
+DbUtil::releasePreparedOperation(PreparedOperationPtr prepOpPtr) {
+ prepOpPtr.p->attrMapping.release();
+ prepOpPtr.p->attrInfo.release();
+ prepOpPtr.p->rsInfo.release();
+ prepOpPtr.p->pkBitmask.clear();
+ c_preparedOperationPool.release(prepOpPtr); // No list holding these structs
+}
+
+void
+DbUtil::releaseTransaction(TransactionPtr transPtr){
+ transPtr.p->executePages.release();
+ OperationPtr opPtr;
+ for(transPtr.p->operations.first(opPtr); opPtr.i != RNIL;
+ transPtr.p->operations.next(opPtr)){
+ opPtr.p->attrInfo.release();
+ opPtr.p->keyInfo.release();
+ opPtr.p->rs.release();
+ if (opPtr.p->prepOp != 0 && opPtr.p->prepOp_i != RNIL) {
+ if (opPtr.p->prepOp->releaseFlag) {
+ PreparedOperationPtr prepOpPtr;
+ prepOpPtr.i = opPtr.p->prepOp_i;
+ prepOpPtr.p = opPtr.p->prepOp;
+ releasePreparedOperation(prepOpPtr);
+ }
+ }
+ }
+ transPtr.p->operations.release();
+ c_runningTransactions.release(transPtr);
+}
+
+void
+DbUtil::execSTTOR(Signal* signal)
+{
+ jamEntry();
+
+ const Uint32 startphase = signal->theData[1];
+
+ if(startphase == 1){
+ c_transId[0] = (number() << 20) + (getOwnNodeId() << 8);
+ c_transId[1] = 0;
+ }
+
+ if(startphase == 6){
+ hardcodedPrepare();
+ connectTc(signal);
+ }
+
+ signal->theData[0] = 0;
+ signal->theData[3] = 1;
+ signal->theData[4] = 6;
+ signal->theData[5] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 6, JBB);
+
+ return;
+}
+
+void
+DbUtil::execNDB_STTOR(Signal* signal)
+{
+ (void)signal; // Don't want compiler warning
+
+ jamEntry();
+}
+
+
+/***************************
+ * Seize a number of TC records
+ * to use for Util transactions
+ */
+
+void
+DbUtil::connectTc(Signal* signal){
+
+ TransactionPtr ptr;
+ while(c_seizingTransactions.seize(ptr)){
+ signal->theData[0] = ptr.i << 1; // See TcCommitConf
+ signal->theData[1] = reference();
+ sendSignal(DBTC_REF, GSN_TCSEIZEREQ, signal, 2, JBB);
+ }
+}
+
+void
+DbUtil::execTCSEIZECONF(Signal* signal){
+ jamEntry();
+
+ TransactionPtr ptr;
+ ptr.i = signal->theData[0] >> 1;
+ c_seizingTransactions.getPtr(ptr, signal->theData[0] >> 1);
+ ptr.p->connectPtr = signal->theData[1];
+
+ c_seizingTransactions.release(ptr);
+}
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: Misc
+ * ------------------------------------------------------------------------
+ *
+ * ContinueB, Dump
+ **************************************************************************/
+
+void
+DbUtil::execCONTINUEB(Signal* signal){
+ jamEntry();
+ const Uint32 Tdata0 = signal->theData[0];
+
+ switch(Tdata0){
+ default:
+ ndbrequire(0);
+ }
+}
+
+void
+DbUtil::execDUMP_STATE_ORD(Signal* signal){
+ jamEntry();
+
+ /****************************************************************************
+ * SEQUENCE SERVICE
+ *
+ * 200 : Simple test of Public Sequence Interface
+ * ----------------------------------------------
+ * - Sends a SEQUENCE_REQ signal to Util (itself)
+ */
+ const Uint32 tCase = signal->theData[0];
+ if(tCase == 200){
+ jam()
+ ndbout << "--------------------------------------------------" << endl;
+ UtilSequenceReq * req = (UtilSequenceReq*)signal->getDataPtrSend();
+ Uint32 seqId = 1;
+ Uint32 reqTy = UtilSequenceReq::CurrVal;
+
+ if(signal->length() > 1) seqId = signal->theData[1];
+ if(signal->length() > 2) reqTy = signal->theData[2];
+
+ req->senderData = 12;
+ req->sequenceId = seqId;
+ req->requestType = reqTy;
+
+ sendSignal(DBUTIL_REF, GSN_UTIL_SEQUENCE_REQ,
+ signal, UtilSequenceReq::SignalLength, JBB);
+ }
+
+ /****************************************************************************/
+ /* // Obsolete tests, should be rewritten for long signals!!
+ if(tCase == 210){
+ jam();
+ ndbout << "--------------------------------------------------" << endl;
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0], 128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, UtilPrepareReq::Delete);
+ w.add(UtilPrepareReq::TableName, "sys/def/SYSTAB_0");
+ w.add(UtilPrepareReq::AttributeName, "SYSKEY_0"); // AttrNo = 0
+ Uint32 length = w.getWordsUsed();
+ ndbassert(length <= pageSizeInWords);
+
+ sendUtilPrepareReqSignals(signal, propPage, length);
+ }
+ if(tCase == 211){
+ jam();
+ ndbout << "--------------------------------------------------" << endl;
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0],128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, UtilPrepareReq::Insert);
+ w.add(UtilPrepareReq::TableName, "sys/def/SYSTAB_0");
+ w.add(UtilPrepareReq::AttributeName, "SYSKEY_0"); // AttrNo = 0
+ w.add(UtilPrepareReq::AttributeName, "NEXTID"); // AttrNo = 1
+ Uint32 length = w.getWordsUsed();
+ ndbassert(length <= pageSizeInWords);
+
+ sendUtilPrepareReqSignals(signal, propPage, length);
+ }
+ if(tCase == 212){
+ jam();
+ ndbout << "--------------------------------------------------" << endl;
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0],128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, UtilPrepareReq::Update);
+ w.add(UtilPrepareReq::TableName, "sys/def/SYSTAB_0");
+ w.add(UtilPrepareReq::AttributeName, "SYSKEY_0"); // AttrNo = 0
+ w.add(UtilPrepareReq::AttributeName, "NEXTID"); // AttrNo = 1
+ Uint32 length = w.getWordsUsed();
+ ndbassert(length <= pageSizeInWords);
+
+ sendUtilPrepareReqSignals(signal, propPage, length);
+ }
+ if(tCase == 213){
+ jam();
+ ndbout << "--------------------------------------------------" << endl;
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0],128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, UtilPrepareReq::Read);
+ w.add(UtilPrepareReq::TableName, "sys/def/SYSTAB_0");
+ w.add(UtilPrepareReq::AttributeName, "SYSKEY_0"); // AttrNo = 0
+ Uint32 length = w.getWordsUsed();
+ ndbassert(length <= pageSizeInWords);
+
+ sendUtilPrepareReqSignals(signal, propPage, length);
+ }
+ if(tCase == 214){
+ jam();
+ ndbout << "--------------------------------------------------" << endl;
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0], 128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, UtilPrepareReq::Delete);
+ w.add(UtilPrepareReq::TableId, (unsigned int)0); // SYSTAB_0
+ w.add(UtilPrepareReq::AttributeId, (unsigned int)0);// SYSKEY_0
+ Uint32 length = w.getWordsUsed();
+ ndbassert(length <= pageSizeInWords);
+
+ sendUtilPrepareReqSignals(signal, propPage, length);
+ }
+ if(tCase == 215){
+ jam();
+ ndbout << "--------------------------------------------------" << endl;
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0],128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, UtilPrepareReq::Insert);
+ w.add(UtilPrepareReq::TableId, (unsigned int)0); // SYSTAB_0
+ w.add(UtilPrepareReq::AttributeId, (unsigned int)0); // SYSKEY_0
+ w.add(UtilPrepareReq::AttributeId, 1); // NEXTID
+ Uint32 length = w.getWordsUsed();
+ ndbassert(length <= pageSizeInWords);
+
+ sendUtilPrepareReqSignals(signal, propPage, length);
+ }
+ if(tCase == 216){
+ jam();
+ ndbout << "--------------------------------------------------" << endl;
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0],128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, UtilPrepareReq::Update);
+ w.add(UtilPrepareReq::TableId, (unsigned int)0); // SYSTAB_0
+ w.add(UtilPrepareReq::AttributeId, (unsigned int)0);// SYSKEY_0
+ w.add(UtilPrepareReq::AttributeId, 1); // NEXTID
+ Uint32 length = w.getWordsUsed();
+ ndbassert(length <= pageSizeInWords);
+
+ sendUtilPrepareReqSignals(signal, propPage, length);
+ }
+ if(tCase == 217){
+ jam();
+ ndbout << "--------------------------------------------------" << endl;
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0],128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, UtilPrepareReq::Read);
+ w.add(UtilPrepareReq::TableId, (unsigned int)0); // SYSTAB_0
+ w.add(UtilPrepareReq::AttributeId, (unsigned int)0);// SYSKEY_0
+ Uint32 length = w.getWordsUsed();
+ ndbassert(length <= pageSizeInWords);
+
+ sendUtilPrepareReqSignals(signal, propPage, length);
+ }
+ */
+ /****************************************************************************/
+ /* // Obsolete tests, should be rewritten for long signals!!
+ if(tCase == 220){
+ jam();
+ ndbout << "--------------------------------------------------" << endl;
+ Uint32 prepI = signal->theData[1];
+ Uint32 length = signal->theData[2];
+ Uint32 attributeValue0 = signal->theData[3];
+ Uint32 attributeValue1a = signal->theData[4];
+ Uint32 attributeValue1b = signal->theData[5];
+ ndbrequire(prepI != 0);
+
+ UtilExecuteReq * req = (UtilExecuteReq *)signal->getDataPtrSend();
+
+ req->senderData = 221;
+ req->prepareId = prepI;
+ req->totalDataLen = length; // Including headers
+ req->offset = 0;
+
+ AttributeHeader::init(&req->attrData[0], 0, 1); // AttrNo 0, DataSize
+ req->attrData[1] = attributeValue0; // AttrValue
+ AttributeHeader::init(&req->attrData[2], 1, 2); // AttrNo 1, DataSize
+ req->attrData[3] = attributeValue1a; // AttrValue
+ req->attrData[4] = attributeValue1b; // AttrValue
+
+ printUTIL_EXECUTE_REQ(stdout, signal->getDataPtrSend(), 3 + 5,0);
+ sendSignal(DBUTIL_REF, GSN_UTIL_EXECUTE_REQ, signal, 3 + 5, JBB);
+ }
+*/
+ /****************************************************************************
+ * 230 : PRINT STATE
+ */
+#ifdef ARRAY_GUARD
+ if(tCase == 230){
+ jam();
+
+ ndbout << "--------------------------------------------------" << endl;
+ if (signal->length() <= 1) {
+ ndbout << "Usage: DUMP 230 <recordType> <recordNo>" << endl
+ << "[1] Print Prepare (running) records" << endl
+ << "[2] Print PreparedOperation records" << endl
+ << "[3] Print Transaction records" << endl
+ << "[4] Print Operation records" << endl
+ << "Ex. \"dump 230 1 2\" prints Prepare record no 2." << endl
+ << endl
+ << "210 : PREPARE_REQ DELETE SYSTAB_0 SYSKEY_0" << endl
+ << "211 : PREPARE_REQ INSERT SYSTAB_0 SYSKEY_0 NEXTID" << endl
+ << "212 : PREPARE_REQ UPDATE SYSTAB_0 SYSKEY_0 NEXTID" << endl
+ << "213 : PREPARE_REQ READ SYSTAB_0 SYSKEY_0" << endl
+ << "214 : PREPARE_REQ DELETE SYSTAB_0 SYSKEY_0 using id" << endl
+ << "215 : PREPARE_REQ INSERT SYSTAB_0 SYSKEY_0 NEXTID using id" << endl
+ << "216 : PREPARE_REQ UPDATE SYSTAB_0 SYSKEY_0 NEXTID using id" << endl
+ << "217 : PREPARE_REQ READ SYSTAB_0 SYSKEY_0 using id" << endl
+ << "220 : EXECUTE_REQ <PrepId> <Len> <Val1> <Val2a> <Val2b>" <<endl
+ << "299 : Crash system (using ndbrequire(0))"
+ << endl
+ << "Ex. \"dump 220 3 5 1 0 17 \" prints Prepare record no 2."
+ << endl;
+ return;
+ }
+
+ switch (signal->theData[1]) {
+ case 1:
+ // ** Print a specific record **
+ if (signal->length() >= 3) {
+ PreparePtr prepPtr;
+ if (!c_preparePool.isSeized(signal->theData[2])) {
+ ndbout << "Prepare Id: " << signal->theData[2]
+ << " (Not seized!)" << endl;
+ } else {
+ c_preparePool.getPtr(prepPtr, signal->theData[2]);
+ prepPtr.p->print();
+ }
+ return;
+ }
+
+ // ** Print all records **
+ PreparePtr prepPtr;
+ if (!c_runningPrepares.first(prepPtr)) {
+ ndbout << "No Prepare records exist" << endl;
+ return;
+ }
+
+ while (!prepPtr.isNull()) {
+ prepPtr.p->print();
+ c_runningPrepares.next(prepPtr);
+ }
+ return;
+
+ case 2:
+ // ** Print a specific record **
+ if (signal->length() >= 3) {
+ if (!c_preparedOperationPool.isSeized(signal->theData[2])) {
+ ndbout << "PreparedOperation Id: " << signal->theData[2]
+ << " (Not seized!)" << endl;
+ return;
+ }
+ ndbout << "PreparedOperation Id: " << signal->theData[2] << endl;
+ PreparedOperationPtr prepOpPtr;
+ c_runningPreparedOperations.getPtr(prepOpPtr, signal->theData[2]);
+ prepOpPtr.p->print();
+ return;
+ }
+
+ // ** Print all records **
+ PreparedOperationPtr prepOpPtr;
+ if (!c_runningPreparedOperations.first(prepOpPtr)) {
+ ndbout << "No PreparedOperations exist" << endl;
+ return;
+ }
+ while (!prepOpPtr.isNull()) {
+ ndbout << "[-PreparedOperation no " << prepOpPtr.i << ":";
+ prepOpPtr.p->print();
+ ndbout << "]";
+ c_runningPreparedOperations.next(prepOpPtr);
+ }
+ return;
+
+ case 3:
+ // ** Print a specific record **
+ if (signal->length() >= 3) {
+ ndbout << "Print specific record not implemented." << endl;
+ return;
+ }
+
+ // ** Print all records **
+ ndbout << "Print all records not implemented, specify an Id." << endl;
+ return;
+
+ case 4:
+ ndbout << "Not implemented" << endl;
+ return;
+
+ default:
+ ndbout << "Unknown input (try without any data)" << endl;
+ return;
+ }
+ }
+#endif
+ if(tCase == 240 && signal->getLength() == 2){
+ MutexManager::ActiveMutexPtr ptr;
+ ndbrequire(c_mutexMgr.seize(ptr));
+ ptr.p->m_mutexId = signal->theData[1];
+ Callback c = { safe_cast(&DbUtil::mutex_created), ptr.i };
+ ptr.p->m_callback = c;
+ c_mutexMgr.create(signal, ptr);
+ ndbout_c("c_mutexMgr.create ptrI=%d mutexId=%d", ptr.i, ptr.p->m_mutexId);
+ }
+
+ if(tCase == 241 && signal->getLength() == 2){
+ MutexManager::ActiveMutexPtr ptr;
+ ndbrequire(c_mutexMgr.seize(ptr));
+ ptr.p->m_mutexId = signal->theData[1];
+ Callback c = { safe_cast(&DbUtil::mutex_locked), ptr.i };
+ ptr.p->m_callback = c;
+ c_mutexMgr.lock(signal, ptr);
+ ndbout_c("c_mutexMgr.lock ptrI=%d mutexId=%d", ptr.i, ptr.p->m_mutexId);
+ }
+
+ if(tCase == 242 && signal->getLength() == 2){
+ MutexManager::ActiveMutexPtr ptr;
+ ptr.i = signal->theData[1];
+ c_mutexMgr.getPtr(ptr);
+ Callback c = { safe_cast(&DbUtil::mutex_unlocked), ptr.i };
+ ptr.p->m_callback = c;
+ c_mutexMgr.unlock(signal, ptr);
+ ndbout_c("c_mutexMgr.unlock ptrI=%d mutexId=%d", ptr.i, ptr.p->m_mutexId);
+ }
+
+ if(tCase == 243 && signal->getLength() == 3){
+ MutexManager::ActiveMutexPtr ptr;
+ ndbrequire(c_mutexMgr.seize(ptr));
+ ptr.p->m_mutexId = signal->theData[1];
+ ptr.p->m_mutexKey = signal->theData[2];
+ Callback c = { safe_cast(&DbUtil::mutex_destroyed), ptr.i };
+ ptr.p->m_callback = c;
+ c_mutexMgr.destroy(signal, ptr);
+ ndbout_c("c_mutexMgr.destroy ptrI=%d mutexId=%d key=%d",
+ ptr.i, ptr.p->m_mutexId, ptr.p->m_mutexKey);
+ }
+}
+
+void
+DbUtil::mutex_created(Signal* signal, Uint32 ptrI, Uint32 retVal){
+ MutexManager::ActiveMutexPtr ptr; ptr.i = ptrI;
+ c_mutexMgr.getPtr(ptr);
+ ndbout_c("mutex_created - mutexId=%d, retVal=%d",
+ ptr.p->m_mutexId, retVal);
+ c_mutexMgr.release(ptrI);
+}
+
+void
+DbUtil::mutex_destroyed(Signal* signal, Uint32 ptrI, Uint32 retVal){
+ MutexManager::ActiveMutexPtr ptr; ptr.i = ptrI;
+ c_mutexMgr.getPtr(ptr);
+ ndbout_c("mutex_destroyed - mutexId=%d, retVal=%d",
+ ptr.p->m_mutexId, retVal);
+ c_mutexMgr.release(ptrI);
+}
+
+void
+DbUtil::mutex_locked(Signal* signal, Uint32 ptrI, Uint32 retVal){
+ MutexManager::ActiveMutexPtr ptr; ptr.i = ptrI;
+ c_mutexMgr.getPtr(ptr);
+ ndbout_c("mutex_locked - mutexId=%d, retVal=%d key=%d ptrI=%d",
+ ptr.p->m_mutexId, retVal, ptr.p->m_mutexKey, ptrI);
+ if(retVal)
+ c_mutexMgr.release(ptrI);
+}
+
+void
+DbUtil::mutex_unlocked(Signal* signal, Uint32 ptrI, Uint32 retVal){
+ MutexManager::ActiveMutexPtr ptr; ptr.i = ptrI;
+ c_mutexMgr.getPtr(ptr);
+ ndbout_c("mutex_unlocked - mutexId=%d, retVal=%d",
+ ptr.p->m_mutexId, retVal);
+ if(!retVal)
+ c_mutexMgr.release(ptrI);
+}
+
+void
+DbUtil::execUTIL_SEQUENCE_REF(Signal* signal){
+ jamEntry();
+ ndbout << "UTIL_SEQUENCE_REF" << endl;
+ printUTIL_SEQUENCE_REF(stdout, signal->getDataPtrSend(), signal->length(), 0);
+}
+
+void
+DbUtil::execUTIL_SEQUENCE_CONF(Signal* signal){
+ jamEntry();
+ ndbout << "UTIL_SEQUENCE_CONF" << endl;
+ printUTIL_SEQUENCE_CONF(stdout, signal->getDataPtrSend(), signal->length(),0);
+}
+
+void
+DbUtil::execUTIL_PREPARE_CONF(Signal* signal){
+ jamEntry();
+ ndbout << "UTIL_PREPARE_CONF" << endl;
+ printUTIL_PREPARE_CONF(stdout, signal->getDataPtrSend(), signal->length(), 0);
+}
+
+void
+DbUtil::execUTIL_PREPARE_REF(Signal* signal){
+ jamEntry();
+ ndbout << "UTIL_PREPARE_REF" << endl;
+ printUTIL_PREPARE_REF(stdout, signal->getDataPtrSend(), signal->length(), 0);
+}
+
+void
+DbUtil::execUTIL_EXECUTE_CONF(Signal* signal) {
+ jamEntry();
+ ndbout << "UTIL_EXECUTE_CONF" << endl;
+ printUTIL_EXECUTE_CONF(stdout, signal->getDataPtrSend(), signal->length(), 0);
+}
+
+void
+DbUtil::execUTIL_EXECUTE_REF(Signal* signal) {
+ jamEntry();
+
+ ndbout << "UTIL_EXECUTE_REF" << endl;
+ printUTIL_EXECUTE_REF(stdout, signal->getDataPtrSend(), signal->length(), 0);
+}
+
+void
+DbUtil::execUTIL_RELEASE_CONF(Signal* signal) {
+ jamEntry();
+ ndbout << "UTIL_RELEASE_CONF" << endl;
+}
+
+void
+DbUtil::execUTIL_RELEASE_REF(Signal* signal) {
+ jamEntry();
+
+ ndbout << "UTIL_RELEASE_REF" << endl;
+}
+
+void
+DbUtil::sendUtilPrepareRef(Signal* signal, UtilPrepareRef::ErrorCode error,
+ Uint32 recipient, Uint32 senderData){
+ UtilPrepareRef * ref = (UtilPrepareRef *)signal->getDataPtrSend();
+ ref->errorCode = error;
+ ref->senderData = senderData;
+
+ sendSignal(recipient, GSN_UTIL_PREPARE_REF, signal,
+ UtilPrepareRef::SignalLength, JBB);
+}
+
+void
+DbUtil::sendUtilExecuteRef(Signal* signal, UtilExecuteRef::ErrorCode error,
+ Uint32 TCerror, Uint32 recipient, Uint32 senderData){
+
+ UtilExecuteRef * ref = (UtilExecuteRef *)signal->getDataPtrSend();
+ ref->senderData = senderData;
+ ref->errorCode = error;
+ ref->TCErrorCode = TCerror;
+
+ sendSignal(recipient, GSN_UTIL_EXECUTE_REF, signal,
+ UtilPrepareRef::SignalLength, JBB);
+}
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: Prepare service
+ * ------------------------------------------------------------------------
+ *
+ * Prepares a transaction by storing info in some structs
+ **************************************************************************/
+
+void
+DbUtil::execUTIL_PREPARE_REQ(Signal* signal)
+{
+ jamEntry();
+
+ /****************
+ * Decode Signal
+ ****************/
+ UtilPrepareReq * req = (UtilPrepareReq *)signal->getDataPtr();
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 senderData = req->senderData;
+
+ if(signal->getNoOfSections() == 0) {
+ // Missing prepare data
+ jam();
+ releaseSections(signal);
+ sendUtilPrepareRef(signal, UtilPrepareRef::MISSING_PROPERTIES_SECTION,
+ senderRef, senderData);
+ return;
+ }
+
+ PreparePtr prepPtr;
+ SegmentedSectionPtr ptr;
+
+ jam();
+ if(!c_runningPrepares.seize(prepPtr)) {
+ jam();
+ releaseSections(signal);
+ sendUtilPrepareRef(signal, UtilPrepareRef::PREPARE_SEIZE_ERROR,
+ senderRef, senderData);
+ return;
+ };
+ signal->getSection(ptr, UtilPrepareReq::PROPERTIES_SECTION);
+ const Uint32 noPages = (ptr.sz + sizeof(Page32)) / sizeof(Page32);
+ ndbassert(noPages > 0);
+ if (!prepPtr.p->preparePages.seize(noPages)) {
+ jam();
+ releaseSections(signal);
+ sendUtilPrepareRef(signal, UtilPrepareRef::PREPARE_PAGES_SEIZE_ERROR,
+ senderRef, senderData);
+ c_preparePool.release(prepPtr);
+ return;
+ }
+ // Save SimpleProperties
+ Uint32* target = &prepPtr.p->preparePages.getPtr(0)->data[0];
+ copy(target, ptr);
+ prepPtr.p->prepDataLen = ptr.sz;
+ // Release long signal sections
+ releaseSections(signal);
+ // Check table properties with DICT
+ SimplePropertiesSectionReader reader(ptr, getSectionSegmentPool());
+ prepPtr.p->clientRef = senderRef;
+ prepPtr.p->clientData = senderData;
+ // Release long signal sections
+ releaseSections(signal);
+ readPrepareProps(signal, &reader, prepPtr.i);
+}
+
+void DbUtil::readPrepareProps(Signal* signal,
+ SimpleProperties::Reader* reader,
+ Uint32 senderData)
+{
+ jam();
+#if 0
+ printf("DbUtil::readPrepareProps: Received SimpleProperties:\n");
+ reader->printAll(ndbout);
+#endif
+ ndbrequire(reader->first());
+ ndbrequire(reader->getKey() == UtilPrepareReq::NoOfOperations);
+ ndbrequire(reader->getUint32() == 1); // Only one op/trans implemented
+
+ ndbrequire(reader->next());
+ ndbrequire(reader->getKey() == UtilPrepareReq::OperationType);
+
+ ndbrequire(reader->next());
+ UtilPrepareReq::KeyValue tableKey =
+ (UtilPrepareReq::KeyValue) reader->getKey();
+ ndbrequire((tableKey == UtilPrepareReq::TableName) ||
+ (tableKey == UtilPrepareReq::TableId));
+
+ /************************
+ * Ask Dict for metadata
+ ************************/
+ {
+ GetTabInfoReq * req = (GetTabInfoReq *)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = senderData;
+ if (tableKey == UtilPrepareReq::TableName) {
+ jam();
+ char tableName[MAX_TAB_NAME_SIZE];
+ req->requestType = GetTabInfoReq::RequestByName |
+ GetTabInfoReq::LongSignalConf;
+
+ req->tableNameLen = reader->getValueLen(); // Including trailing \0
+
+ /********************************************
+ * Code signal data and send signals to DICT
+ ********************************************/
+
+ ndbrequire(req->tableNameLen < MAX_TAB_NAME_SIZE);
+ reader->getString((char*)tableName);
+ LinearSectionPtr ptr[1];
+ ptr[0].p = (Uint32*)tableName;
+ ptr[0].sz = req->tableNameLen;
+ sendSignal(DBDICT_REF, GSN_GET_TABINFOREQ, signal,
+ GetTabInfoReq::SignalLength, JBB, ptr,1);
+
+ }
+ else { // (tableKey == UtilPrepareReq::TableId)
+ jam();
+ req->requestType = GetTabInfoReq::RequestById |
+ GetTabInfoReq::LongSignalConf;
+ req->tableId = reader->getUint32();
+ sendSignal(DBDICT_REF, GSN_GET_TABINFOREQ, signal,
+ GetTabInfoReq::SignalLength, JBB);
+ }
+
+ }
+}
+
+/**
+ * @note We assume that this signal comes due to a request related
+ * to a Prepare struct. DictTabInfo:s 'senderData' denotes
+ * the Prepare struct related to the request.
+ */
+void
+DbUtil::execGET_TABINFO_CONF(Signal* signal){
+ jamEntry();
+
+ if(!assembleFragments(signal)){
+ jam();
+ return;
+ }
+
+ /****************
+ * Decode signal
+ ****************/
+ GetTabInfoConf * const conf = (GetTabInfoConf*)signal->getDataPtr();
+ const Uint32 prepI = conf->senderData;
+ const Uint32 totalLen = conf->totalLen;
+
+ SegmentedSectionPtr dictTabInfoPtr;
+ signal->getSection(dictTabInfoPtr, GetTabInfoConf::DICT_TAB_INFO);
+ ndbrequire(dictTabInfoPtr.sz == totalLen);
+
+ PreparePtr prepPtr;
+ c_runningPrepares.getPtr(prepPtr, prepI);
+ prepareOperation(signal, prepPtr);
+}
+
+void
+DbUtil::execGET_TABINFOREF(Signal* signal){
+ jamEntry();
+
+ GetTabInfoRef * ref = (GetTabInfoRef *)signal->getDataPtr();
+ Uint32 prepI = ref->senderData;
+#define EVENT_DEBUG
+#if 0 //def EVENT_DEBUG
+ ndbout << "Signal GET_TABINFOREF received." << endl;
+ ndbout << "Error Code: " << ref->errorCode << endl;
+
+ switch (ref->errorCode) {
+ case GetTabInfoRef::InvalidTableId:
+ ndbout << " Msg: Invalid table id" << endl;
+ break;
+ case GetTabInfoRef::TableNotDefined:
+ ndbout << " Msg: Table not defined" << endl;
+ break;
+ case GetTabInfoRef::TableNameToLong:
+ ndbout << " Msg: Table node too long" << endl;
+ break;
+ default:
+ ndbout << " Msg: Unknown error returned from Dict" << endl;
+ break;
+ }
+#endif
+
+ PreparePtr prepPtr;
+ c_runningPrepares.getPtr(prepPtr, prepI);
+
+ sendUtilPrepareRef(signal, UtilPrepareRef::DICT_TAB_INFO_ERROR,
+ prepPtr.p->clientRef, prepPtr.p->clientData);
+
+ releasePrepare(prepPtr);
+}
+
+
+/******************************************************************************
+ * Prepare Operation
+ *
+ * Using a prepare record, prepare an operation (i.e. create PreparedOperation).
+ * Info from both Pepare request (PreparePages) and DictTabInfo is used.
+ *
+ * Algorithm:
+ * -# Seize AttrbuteMapping
+ * - Lookup in preparePages how many attributes should be prepared
+ * - Seize AttributeMapping
+ * -# For each attributes in preparePages
+ * - Lookup id and isPK in dictInfoPages
+ * - Store "no -> (AttributeId, Position)" in AttributeMapping
+ * -# For each map in AttributeMapping
+ * - if (isPK) then assign offset
+ ******************************************************************************/
+void
+DbUtil::prepareOperation(Signal* signal, PreparePtr prepPtr)
+{
+ jam();
+
+ /*******************************************
+ * Seize and store PreparedOperation struct
+ *******************************************/
+ PreparedOperationPtr prepOpPtr;
+ if(!c_runningPreparedOperations.seize(prepOpPtr)) {
+ jam();
+ releaseSections(signal);
+ sendUtilPrepareRef(signal, UtilPrepareRef::PREPARED_OPERATION_SEIZE_ERROR,
+ prepPtr.p->clientRef, prepPtr.p->clientData);
+ releasePrepare(prepPtr);
+ return;
+ }
+ prepPtr.p->prepOpPtr = prepOpPtr;
+
+ /********************
+ * Read request info
+ ********************/
+ SimplePropertiesLinearReader prepPagesReader(&prepPtr.p->preparePages.getPtr(0)->data[0],
+ prepPtr.p->prepDataLen);
+
+ ndbrequire(prepPagesReader.first());
+ ndbrequire(prepPagesReader.getKey() == UtilPrepareReq::NoOfOperations);
+ const Uint32 noOfOperations = prepPagesReader.getUint32();
+ ndbrequire(noOfOperations == 1);
+
+ ndbrequire(prepPagesReader.next());
+ ndbrequire(prepPagesReader.getKey() == UtilPrepareReq::OperationType);
+ const Uint32 operationType = prepPagesReader.getUint32();
+
+ ndbrequire(prepPagesReader.next());
+
+ char tableName[MAX_TAB_NAME_SIZE];
+ Uint32 tableId;
+ UtilPrepareReq::KeyValue tableKey =
+ (UtilPrepareReq::KeyValue) prepPagesReader.getKey();
+ if (tableKey == UtilPrepareReq::TableId) {
+ jam();
+ tableId = prepPagesReader.getUint32();
+ }
+ else {
+ jam();
+ ndbrequire(prepPagesReader.getKey() == UtilPrepareReq::TableName);
+ ndbrequire(prepPagesReader.getValueLen() <= MAX_TAB_NAME_SIZE);
+ prepPagesReader.getString(tableName);
+ }
+ /******************************************************************
+ * Seize AttributeMapping (by counting no of attribs in prepPages)
+ ******************************************************************/
+ Uint32 noOfAttributes = 0; // No of attributes in PreparePages (used later)
+ while(prepPagesReader.next()) {
+ if (tableKey == UtilPrepareReq::TableName) {
+ jam();
+ ndbrequire(prepPagesReader.getKey() == UtilPrepareReq::AttributeName);
+ } else {
+ jam();
+ ndbrequire(prepPagesReader.getKey() == UtilPrepareReq::AttributeId);
+ }
+ noOfAttributes++;
+ }
+ ndbrequire(prepPtr.p->prepOpPtr.p->attrMapping.seize(noOfAttributes));
+ if (operationType == UtilPrepareReq::Read) {
+ ndbrequire(prepPtr.p->prepOpPtr.p->rsInfo.seize(noOfAttributes));
+ }
+ /***************************************
+ * For each attribute name, lookup info
+ ***************************************/
+ // Goto start of attribute names
+ ndbrequire(prepPagesReader.first() && prepPagesReader.next() &&
+ prepPagesReader.next());
+
+ DictTabInfo::Table tableDesc; tableDesc.init();
+ AttrMappingBuffer::DataBufferIterator attrMappingIt;
+ ndbrequire(prepPtr.p->prepOpPtr.p->attrMapping.first(attrMappingIt));
+
+ ResultSetBuffer::DataBufferIterator rsInfoIt;
+ if (operationType == UtilPrepareReq::Read) {
+ ndbrequire(prepPtr.p->prepOpPtr.p->rsInfo.first(rsInfoIt));
+ }
+
+ Uint32 noOfPKAttribsStored = 0;
+ Uint32 noOfNonPKAttribsStored = 0;
+ Uint32 attrLength = 0;
+ Uint32 pkAttrLength = 0;
+ char attrNameRequested[MAX_ATTR_NAME_SIZE];
+ Uint32 attrIdRequested;
+
+ while(prepPagesReader.next()) {
+ UtilPrepareReq::KeyValue attributeKey =
+ (UtilPrepareReq::KeyValue) prepPagesReader.getKey();
+
+ ndbrequire((attributeKey == UtilPrepareReq::AttributeName) ||
+ (attributeKey == UtilPrepareReq::AttributeId));
+ if (attributeKey == UtilPrepareReq::AttributeName) {
+ jam();
+ ndbrequire(prepPagesReader.getValueLen() <= MAX_ATTR_NAME_SIZE);
+
+ prepPagesReader.getString(attrNameRequested);
+ } else {
+ jam();
+ attrIdRequested = prepPagesReader.getUint32();
+ }
+ /*****************************************
+ * Copy DictTabInfo into tableDesc struct
+ *****************************************/
+
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr, GetTabInfoConf::DICT_TAB_INFO);
+ SimplePropertiesSectionReader dictInfoReader(ptr, getSectionSegmentPool());
+
+ SimpleProperties::UnpackStatus unpackStatus;
+ unpackStatus = SimpleProperties::unpack(dictInfoReader, &tableDesc,
+ DictTabInfo::TableMapping,
+ DictTabInfo::TableMappingSize,
+ true, true);
+ ndbrequire(unpackStatus == SimpleProperties::Break);
+
+ /************************
+ * Lookup in DictTabInfo
+ ************************/
+ DictTabInfo::Attribute attrDesc; attrDesc.init();
+ char attrName[MAX_ATTR_NAME_SIZE];
+ Uint32 attrId;
+ bool attributeFound = false;
+ Uint32 noOfKeysFound = 0; // # PK attrs found before attr in DICTdata
+ Uint32 noOfNonKeysFound = 0; // # nonPK attrs found before attr in DICTdata
+ for (Uint32 i=0; i<tableDesc.NoOfAttributes; i++) {
+ if (tableKey == UtilPrepareReq::TableName) {
+ jam();
+ ndbrequire(dictInfoReader.getKey() == DictTabInfo::AttributeName);
+ ndbrequire(dictInfoReader.getValueLen() <= MAX_ATTR_NAME_SIZE);
+ dictInfoReader.getString(attrName);
+ } else { // (tableKey == UtilPrepareReq::TableId)
+ jam();
+ dictInfoReader.next(); // Skip name
+ ndbrequire(dictInfoReader.getKey() == DictTabInfo::AttributeId);
+ attrId = dictInfoReader.getUint32();
+ }
+ unpackStatus = SimpleProperties::unpack(dictInfoReader, &attrDesc,
+ DictTabInfo::AttributeMapping,
+ DictTabInfo::AttributeMappingSize,
+ true, true);
+ ndbrequire(unpackStatus == SimpleProperties::Break);
+ //attrDesc.print(stdout);
+
+ if (attrDesc.AttributeKeyFlag) { jam(); noOfKeysFound++; }
+ else { jam(); noOfNonKeysFound++; }
+ if (attributeKey == UtilPrepareReq::AttributeName) {
+ if (strcmp(attrName, attrNameRequested) == 0) {
+ attributeFound = true;
+ break;
+ }
+ }
+ else // (attributeKey == UtilPrepareReq::AttributeId)
+ if (attrId == attrIdRequested) {
+ attributeFound = true;
+ break;
+ }
+
+ // Move to next attribute
+ ndbassert(dictInfoReader.getKey() == DictTabInfo::AttributeEnd);
+ dictInfoReader.next();
+ }
+
+ /**********************
+ * Attribute not found
+ **********************/
+ if (!attributeFound) {
+ jam();
+ releaseSections(signal);
+ sendUtilPrepareRef(signal,
+ UtilPrepareRef::DICT_TAB_INFO_ERROR,
+ prepPtr.p->clientRef, prepPtr.p->clientData);
+ infoEvent("UTIL: Unknown attribute requested: %s in table: %s",
+ attrNameRequested, tableName);
+ releasePreparedOperation(prepOpPtr);
+ releasePrepare(prepPtr);
+ return;
+ }
+
+ /**************************************************************
+ * Attribute found - store in mapping (AttributeId, Position)
+ **************************************************************/
+ AttributeHeader & attrMap =
+ AttributeHeader::init(attrMappingIt.data,
+ attrDesc.AttributeId, // 1. Store AttrId
+ 0);
+
+ if (attrDesc.AttributeKeyFlag) {
+ // ** Attribute belongs to PK **
+ prepOpPtr.p->pkBitmask.set(attrDesc.AttributeId);
+ attrMap.setDataSize(noOfKeysFound - 1); // 2. Store Position
+ noOfPKAttribsStored++;
+ } else {
+ attrMap.setDataSize(0x3fff); // 2. Store Position (fake)
+ noOfNonPKAttribsStored++;
+
+ /***********************************************************
+ * Error: Read nonPK Attr before all PK attr have been read
+ ***********************************************************/
+ if (noOfPKAttribsStored != tableDesc.NoOfKeyAttr) {
+ jam();
+ releaseSections(signal);
+ sendUtilPrepareRef(signal,
+ UtilPrepareRef::DICT_TAB_INFO_ERROR,
+ prepPtr.p->clientRef, prepPtr.p->clientData);
+ infoEvent("UTIL: Non-PK attr not allowed before "
+ "all PK attrs have been defined, table: %s",
+ tableName);
+ releasePreparedOperation(prepOpPtr);
+ releasePrepare(prepPtr);
+ return;
+ }
+ }
+#if 0
+ ndbout << "BEFORE: attrLength: " << attrLength << endl;
+#endif
+ {
+ int len = 0;
+ switch (attrDesc.AttributeSize) {
+ case DictTabInfo::an8Bit:
+ len = (attrDesc.AttributeArraySize + 3)/ 4;
+ break;
+ case DictTabInfo::a16Bit:
+ len = (attrDesc.AttributeArraySize + 1) / 2;
+ break;
+ case DictTabInfo::a32Bit:
+ len = attrDesc.AttributeArraySize;
+ break;
+ case DictTabInfo::a64Bit:
+ len = attrDesc.AttributeArraySize * 2;
+ break;
+ case DictTabInfo::a128Bit:
+ len = attrDesc.AttributeArraySize * 4;
+ break;
+ }
+ attrLength += len;
+ if (attrDesc.AttributeKeyFlag)
+ pkAttrLength += len;
+
+ if (operationType == UtilPrepareReq::Read) {
+ AttributeHeader::init(rsInfoIt.data,
+ attrDesc.AttributeId, // 1. Store AttrId
+ len);
+ prepOpPtr.p->rsInfo.next(rsInfoIt, 1);
+ }
+ }
+#if 0
+ ndbout << ": AttributeSize: " << attrDesc.AttributeSize << endl;
+ ndbout << ": AttributeArraySize: " << attrDesc.AttributeArraySize << endl;
+ ndbout << "AFTER: attrLength: " << attrLength << endl;
+#endif
+ //attrMappingIt.print(stdout);
+ //prepPtr.p->prepOpPtr.p->attrMapping.print(stdout);
+ prepPtr.p->prepOpPtr.p->attrMapping.next(attrMappingIt, 1);
+ }
+
+ /***************************
+ * Error: Not all PKs found
+ ***************************/
+ if (noOfPKAttribsStored != tableDesc.NoOfKeyAttr) {
+ jam();
+ releaseSections(signal);
+ sendUtilPrepareRef(signal,
+ UtilPrepareRef::DICT_TAB_INFO_ERROR,
+ prepPtr.p->clientRef, prepPtr.p->clientData);
+ infoEvent("UTIL: Not all primary key attributes requested for table: %s",
+ tableName);
+ releasePreparedOperation(prepOpPtr);
+ releasePrepare(prepPtr);
+ return;
+ }
+
+#if 0
+ AttrMappingBuffer::ConstDataBufferIterator tmpIt;
+ for (prepPtr.p->prepOpPtr.p->attrMapping.first(tmpIt); tmpIt.curr.i != RNIL;
+ prepPtr.p->prepOpPtr.p->attrMapping.next(tmpIt)) {
+ AttributeHeader* ah = (AttributeHeader *) tmpIt.data;
+ ah->print(stdout);
+ }
+#endif
+
+ /**********************************************
+ * Preparing of PreparedOperation signal train
+ **********************************************/
+ Uint32 static_len = TcKeyReq::StaticLength;
+ prepOpPtr.p->tckey.tableId = tableDesc.TableId;
+ prepOpPtr.p->tckey.tableSchemaVersion = tableDesc.TableVersion;
+ prepOpPtr.p->noOfKeyAttr = tableDesc.NoOfKeyAttr;
+ prepOpPtr.p->keyLen = tableDesc.KeyLength; // Total no of words in PK
+ if (prepOpPtr.p->keyLen > TcKeyReq::MaxKeyInfo) {
+ jam();
+ prepOpPtr.p->tckeyLenInBytes = (static_len + TcKeyReq::MaxKeyInfo) * 4;
+ } else {
+ jam();
+ prepOpPtr.p->tckeyLenInBytes = (static_len + prepOpPtr.p->keyLen) * 4;
+ }
+ prepOpPtr.p->keyDataPos = static_len; // Start of keyInfo[] in tckeyreq
+
+ Uint32 requestInfo = 0;
+ TcKeyReq::setAbortOption(requestInfo, TcKeyReq::AbortOnError);
+ TcKeyReq::setKeyLength(requestInfo, tableDesc.KeyLength);
+ switch(operationType) {
+ case(UtilPrepareReq::Read):
+ prepOpPtr.p->rsLen =
+ attrLength +
+ tableDesc.NoOfKeyAttr +
+ noOfNonPKAttribsStored; // Read needs a resultset
+ prepOpPtr.p->noOfAttr = tableDesc.NoOfKeyAttr + noOfNonPKAttribsStored;
+ prepOpPtr.p->tckey.attrLen = prepOpPtr.p->noOfAttr;
+ TcKeyReq::setOperationType(requestInfo, ZREAD);
+ break;
+ case(UtilPrepareReq::Update):
+ prepOpPtr.p->rsLen = 0;
+ prepOpPtr.p->noOfAttr = tableDesc.NoOfKeyAttr + noOfNonPKAttribsStored;
+ prepOpPtr.p->tckey.attrLen = attrLength + prepOpPtr.p->noOfAttr;
+ TcKeyReq::setOperationType(requestInfo, ZUPDATE);
+ break;
+ case(UtilPrepareReq::Insert):
+ prepOpPtr.p->rsLen = 0;
+ prepOpPtr.p->noOfAttr = tableDesc.NoOfKeyAttr + noOfNonPKAttribsStored;
+ prepOpPtr.p->tckey.attrLen = attrLength + prepOpPtr.p->noOfAttr;
+ TcKeyReq::setOperationType(requestInfo, ZINSERT);
+ break;
+ case(UtilPrepareReq::Delete):
+ // The number of attributes should equal the size of the primary key
+ ndbrequire(tableDesc.KeyLength == attrLength);
+ prepOpPtr.p->rsLen = 0;
+ prepOpPtr.p->noOfAttr = tableDesc.NoOfKeyAttr;
+ prepOpPtr.p->tckey.attrLen = 0;
+ TcKeyReq::setOperationType(requestInfo, ZDELETE);
+ break;
+ case(UtilPrepareReq::Write):
+ prepOpPtr.p->rsLen = 0;
+ prepOpPtr.p->noOfAttr = tableDesc.NoOfKeyAttr + noOfNonPKAttribsStored;
+ prepOpPtr.p->tckey.attrLen = attrLength + prepOpPtr.p->noOfAttr;
+ TcKeyReq::setOperationType(requestInfo, ZWRITE);
+ break;
+ }
+ TcKeyReq::setAIInTcKeyReq(requestInfo, 0); // Attrinfo sent separately
+ prepOpPtr.p->tckey.requestInfo = requestInfo;
+
+ if (operationType == UtilPrepareReq::Read) {
+ // ResultSet
+ AttrMappingBuffer::ConstDataBufferIterator tmpIt;
+#if 0 //def EVENT_DEBUG
+ ResultSetBuffer & rs = prepOpPtr.p->rsInfo;
+ ResultSetInfoBuffer::DataBufferIterator it;
+ rs.first(it);
+ for (prepOpPtr.p->attrMapping.first(tmpIt);
+ tmpIt.curr.i != RNIL;
+ prepOpPtr.p->attrMapping.next(tmpIt)) {
+ AttributeHeader* ah = (AttributeHeader *) tmpIt.data;
+ ah->print(stdout);
+ AttributeHeader* rsah = (AttributeHeader *) it.data;
+ rsah->print(stdout);
+ rs.next(it,1);
+ printf("%d\n",it.data);
+ }
+#endif
+ }
+
+ /****************************
+ * Confirm completed prepare
+ ****************************/
+ UtilPrepareConf * conf = (UtilPrepareConf *)signal->getDataPtr();
+ conf->senderData = prepPtr.p->clientData;
+ conf->prepareId = prepPtr.p->prepOpPtr.i;
+
+ releaseSections(signal);
+ sendSignal(prepPtr.p->clientRef, GSN_UTIL_PREPARE_CONF, signal,
+ UtilPrepareConf::SignalLength, JBB);
+
+#if 0
+ prepPtr.p->prepOpPtr.p->print();
+#endif
+ releasePrepare(prepPtr);
+}
+
+
+void
+DbUtil::execUTIL_RELEASE_REQ(Signal* signal){
+ jamEntry();
+
+ UtilReleaseReq * req = (UtilReleaseReq *)signal->getDataPtr();
+ const Uint32 clientRef = signal->senderBlockRef();
+ const Uint32 prepareId = req->prepareId;
+ const Uint32 senderData = req->senderData;
+
+#if 0
+ /**
+ * This only works in when ARRAY_GUARD is defined (debug-mode)
+ */
+ if (!c_preparedOperationPool.isSeized(prepareId)) {
+ UtilReleaseRef * ref = (UtilReleaseRef *)signal->getDataPtr();
+ ref->prepareId = prepareId;
+ ref->errorCode = UtilReleaseRef::NO_SUCH_PREPARE_SEIZED;
+ sendSignal(clientRef, GSN_UTIL_RELEASE_REF, signal,
+ UtilReleaseRef::SignalLength, JBB);
+ }
+#endif
+ PreparedOperationPtr prepOpPtr;
+ c_preparedOperationPool.getPtr(prepOpPtr, prepareId);
+
+ releasePreparedOperation(prepOpPtr);
+
+ UtilReleaseConf * const conf = (UtilReleaseConf*)signal->getDataPtrSend();
+ conf->senderData = senderData;
+ sendSignal(clientRef, GSN_UTIL_RELEASE_CONF, signal,
+ UtilReleaseConf::SignalLength, JBB);
+}
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: Sequence Service
+ * ------------------------------------------------------------------------
+ *
+ * A service with a stored incrementable number
+ **************************************************************************/
+
+void
+DbUtil::hardcodedPrepare() {
+ /**
+ * Prepare SequenceCurrVal (READ)
+ */
+ {
+ PreparedOperationPtr ptr;
+ ndbrequire(c_preparedOperationPool.seizeId(ptr, 0));
+ ptr.p->keyLen = 1;
+ ptr.p->tckey.attrLen = 1;
+ ptr.p->rsLen = 3;
+ ptr.p->tckeyLenInBytes = (TcKeyReq::StaticLength +
+ ptr.p->keyLen + ptr.p->tckey.attrLen) * 4;
+ ptr.p->keyDataPos = TcKeyReq::StaticLength;
+ ptr.p->tckey.tableId = 0;
+ Uint32 requestInfo = 0;
+ TcKeyReq::setAbortOption(requestInfo, TcKeyReq::CommitIfFailFree);
+ TcKeyReq::setOperationType(requestInfo, ZREAD);
+ TcKeyReq::setKeyLength(requestInfo, 1);
+ TcKeyReq::setAIInTcKeyReq(requestInfo, 1);
+ ptr.p->tckey.requestInfo = requestInfo;
+ ptr.p->tckey.tableSchemaVersion = 1;
+
+ // This is actually attr data
+ AttributeHeader::init(&ptr.p->tckey.distrGroupHashValue, 1, 0);
+
+ ndbrequire(ptr.p->rsInfo.seize(1));
+ ResultSetInfoBuffer::DataBufferIterator it;
+ ptr.p->rsInfo.first(it);
+ AttributeHeader::init(it.data, 1, 2); // Attribute 1 - 2 data words
+ }
+
+ /**
+ * Prepare SequenceNextVal (UPDATE)
+ */
+ {
+ PreparedOperationPtr ptr;
+ ndbrequire(c_preparedOperationPool.seizeId(ptr, 1));
+ ptr.p->keyLen = 1;
+ ptr.p->rsLen = 3;
+ ptr.p->tckeyLenInBytes = (TcKeyReq::StaticLength + ptr.p->keyLen + 5) * 4;
+ ptr.p->keyDataPos = TcKeyReq::StaticLength;
+ ptr.p->tckey.attrLen = 11;
+ ptr.p->tckey.tableId = 0;
+ Uint32 requestInfo = 0;
+ TcKeyReq::setAbortOption(requestInfo, TcKeyReq::CommitIfFailFree);
+ TcKeyReq::setOperationType(requestInfo, ZUPDATE);
+ TcKeyReq::setKeyLength(requestInfo, 1);
+ TcKeyReq::setAIInTcKeyReq(requestInfo, 5);
+ TcKeyReq::setInterpretedFlag(requestInfo, 1);
+ ptr.p->tckey.requestInfo = requestInfo;
+ ptr.p->tckey.tableSchemaVersion = 1;
+
+ // Signal is packed, which is why attrInfo is at distrGroupHashValue
+ // position
+ Uint32 * attrInfo = &ptr.p->tckey.distrGroupHashValue;
+ attrInfo[0] = 0; // IntialReadSize
+ attrInfo[1] = 5; // InterpretedSize
+ attrInfo[2] = 0; // FinalUpdateSize
+ attrInfo[3] = 1; // FinalReadSize
+ attrInfo[4] = 0; // SubroutineSize
+
+ { // AttrInfo
+ ndbrequire(ptr.p->attrInfo.seize(6));
+ AttrInfoBuffer::DataBufferIterator it;
+ ptr.p->attrInfo.first(it);
+ * it.data = Interpreter::Read(1, 6);
+ ndbrequire(ptr.p->attrInfo.next(it));
+ * it.data = Interpreter::LoadConst16(7, 1);
+ ndbrequire(ptr.p->attrInfo.next(it));
+ * it.data = Interpreter::Add(7, 6, 7);
+ ndbrequire(ptr.p->attrInfo.next(it));
+ * it.data = Interpreter::Write(1, 7);
+ ndbrequire(ptr.p->attrInfo.next(it));
+ * it.data = Interpreter::ExitOK();
+
+ ndbrequire(ptr.p->attrInfo.next(it));
+ AttributeHeader::init(it.data, 1, 0);
+ }
+
+ { // ResultSet
+ ndbrequire(ptr.p->rsInfo.seize(1));
+ ResultSetInfoBuffer::DataBufferIterator it;
+ ptr.p->rsInfo.first(it);
+ AttributeHeader::init(it.data, 1, 2); // Attribute 1 - 2 data words
+ }
+ }
+
+ /**
+ * Prepare CreateSequence (INSERT)
+ */
+ {
+ PreparedOperationPtr ptr;
+ ndbrequire(c_preparedOperationPool.seizeId(ptr, 2));
+ ptr.p->keyLen = 1;
+ ptr.p->tckey.attrLen = 5;
+ ptr.p->rsLen = 0;
+ ptr.p->tckeyLenInBytes = (TcKeyReq::StaticLength +
+ ptr.p->keyLen + ptr.p->tckey.attrLen) * 4;
+ ptr.p->keyDataPos = TcKeyReq::StaticLength;
+ ptr.p->tckey.tableId = 0;
+ Uint32 requestInfo = 0;
+ TcKeyReq::setAbortOption(requestInfo, TcKeyReq::CommitIfFailFree);
+ TcKeyReq::setOperationType(requestInfo, ZINSERT);
+ TcKeyReq::setKeyLength(requestInfo, 1);
+ TcKeyReq::setAIInTcKeyReq(requestInfo, 0);
+ ptr.p->tckey.requestInfo = requestInfo;
+ ptr.p->tckey.tableSchemaVersion = 1;
+ }
+}
+
+void
+DbUtil::execUTIL_SEQUENCE_REQ(Signal* signal){
+ jamEntry();
+
+ UtilSequenceReq * req = (UtilSequenceReq*)signal->getDataPtr();
+
+ PreparedOperation * prepOp;
+
+ switch(req->requestType){
+ case UtilSequenceReq::CurrVal:
+ prepOp = c_preparedOperationPool.getPtr(0); //c_SequenceCurrVal
+ break;
+ case UtilSequenceReq::NextVal:
+ prepOp = c_preparedOperationPool.getPtr(1); //c_SequenceNextVal
+ break;
+ case UtilSequenceReq::Create:
+ prepOp = c_preparedOperationPool.getPtr(2); //c_CreateSequence
+ break;
+ default:
+ ndbrequire(false);
+ }
+
+ /**
+ * 1 Transaction with 1 operation
+ */
+ TransactionPtr transPtr;
+ ndbrequire(c_runningTransactions.seize(transPtr));
+
+ OperationPtr opPtr;
+ ndbrequire(transPtr.p->operations.seize(opPtr));
+
+ ndbrequire(opPtr.p->rs.seize(prepOp->rsLen));
+ ndbrequire(opPtr.p->keyInfo.seize(prepOp->keyLen));
+
+ transPtr.p->gsn = GSN_UTIL_SEQUENCE_REQ;
+ transPtr.p->clientRef = signal->senderBlockRef();
+ transPtr.p->clientData = req->senderData;
+ transPtr.p->sequence.sequenceId = req->sequenceId;
+ transPtr.p->sequence.requestType = req->requestType;
+
+ opPtr.p->prepOp = prepOp;
+ opPtr.p->prepOp_i = RNIL;
+
+ KeyInfoBuffer::DataBufferIterator it;
+ opPtr.p->keyInfo.first(it);
+ it.data[0] = transPtr.p->sequence.sequenceId;
+
+ if(req->requestType == UtilSequenceReq::Create){
+ ndbrequire(opPtr.p->attrInfo.seize(5));
+ AttrInfoBuffer::DataBufferIterator it;
+
+ opPtr.p->attrInfo.first(it);
+ AttributeHeader::init(it.data, 0, 1);
+
+ ndbrequire(opPtr.p->attrInfo.next(it));
+ * it.data = transPtr.p->sequence.sequenceId;
+
+ ndbrequire(opPtr.p->attrInfo.next(it));
+ AttributeHeader::init(it.data, 1, 2);
+
+ ndbrequire(opPtr.p->attrInfo.next(it));
+ * it.data = 0;
+
+ ndbrequire(opPtr.p->attrInfo.next(it));
+ * it.data = 0;
+ }
+
+ runTransaction(signal, transPtr);
+}
+
+int
+DbUtil::getResultSet(Signal* signal, const Transaction * transP,
+ struct LinearSectionPtr sectionsPtr[]) {
+ OperationPtr opPtr;
+ ndbrequire(transP->operations.first(opPtr));
+ ndbrequire(transP->operations.hasNext(opPtr) == false);
+
+ int noAttr = 0;
+ int dataSz = 0;
+ Uint32* tmpBuf = signal->theData + 25;
+ const Uint32* headerBuffer = tmpBuf;
+
+ const ResultSetBuffer & rs = opPtr.p->rs;
+ ResultSetInfoBuffer::ConstDataBufferIterator it;
+
+ // extract headers
+ for(rs.first(it); it.curr.i != RNIL; ) {
+ *tmpBuf++ = it.data[0];
+ rs.next(it, ((AttributeHeader*)&it.data[0])->getDataSize() + 1);
+ noAttr++;
+ }
+
+ if (noAttr == 0)
+ return 0;
+
+ const Uint32* dataBuffer = tmpBuf;
+
+ // extract data
+ for(rs.first(it); it.curr.i != RNIL; ) {
+ int sz = ((AttributeHeader*)&it.data[0])->getDataSize();
+ rs.next(it,1);
+ for (int i = 0; i < sz; i++) {
+ *tmpBuf++ = *it.data;
+ rs.next(it,1);
+ dataSz++;
+ }
+ }
+
+ sectionsPtr[UtilExecuteReq::HEADER_SECTION].p = (Uint32 *)headerBuffer;
+ sectionsPtr[UtilExecuteReq::HEADER_SECTION].sz = noAttr;
+ sectionsPtr[UtilExecuteReq::DATA_SECTION].p = (Uint32 *)dataBuffer;
+ sectionsPtr[UtilExecuteReq::DATA_SECTION].sz = dataSz;
+
+ return 1;
+}
+
+void
+DbUtil::reportSequence(Signal* signal, const Transaction * transP){
+ OperationPtr opPtr;
+ ndbrequire(transP->operations.first(opPtr));
+ ndbrequire(transP->operations.hasNext(opPtr) == false);
+
+ if(transP->errorCode == 0){
+ jam(); // OK
+
+ UtilSequenceConf * ret = (UtilSequenceConf *)signal->getDataPtrSend();
+ ret->senderData = transP->clientData;
+ ret->sequenceId = transP->sequence.sequenceId;
+ ret->requestType = transP->sequence.requestType;
+
+ bool ok = false;
+ switch(transP->sequence.requestType){
+ case UtilSequenceReq::CurrVal:
+ case UtilSequenceReq::NextVal:{
+ ok = true;
+ ndbrequire(opPtr.p->rsRecv == 3);
+
+ ResultSetBuffer::DataBufferIterator rsit;
+ ndbrequire(opPtr.p->rs.first(rsit));
+
+ ret->sequenceValue[0] = rsit.data[1];
+ ret->sequenceValue[1] = rsit.data[2];
+ break;
+ }
+ case UtilSequenceReq::Create:
+ ok = true;
+ ret->sequenceValue[0] = 0;
+ ret->sequenceValue[1] = 0;
+ break;
+ }
+ ndbrequire(ok);
+ sendSignal(transP->clientRef, GSN_UTIL_SEQUENCE_CONF, signal,
+ UtilSequenceConf::SignalLength, JBB);
+ return;
+ }
+
+ UtilSequenceRef::ErrorCode errCode = UtilSequenceRef::TCError;
+
+ switch(transP->sequence.requestType)
+ {
+ case UtilSequenceReq::CurrVal:
+ case UtilSequenceReq::NextVal:{
+ if (transP->errorCode == 626)
+ errCode = UtilSequenceRef::NoSuchSequence;
+ break;
+ }
+ case UtilSequenceReq::Create:
+ break;
+ }
+
+ UtilSequenceRef * ret = (UtilSequenceRef *)signal->getDataPtrSend();
+ ret->senderData = transP->clientData;
+ ret->sequenceId = transP->sequence.sequenceId;
+ ret->requestType = transP->sequence.requestType;
+ ret->errorCode = (Uint32)errCode;
+ sendSignal(transP->clientRef, GSN_UTIL_SEQUENCE_REF, signal,
+ UtilSequenceRef::SignalLength, JBB);
+}
+#if 0
+ Ndb ndb("ndb","def");
+ NdbConnection* tConnection = ndb.startTransaction();
+ NdbOperation* tOperation = tConnection->getNdbOperation("SYSTAB_0");
+
+ //#if 0 && API_CODE
+ if( tOperation != NULL ) {
+ tOperation->interpretedUpdateTuple();
+ tOperation->equal((U_Int32)0, keyValue );
+ tNextId_Result = tOperation->getValue((U_Int32)1);
+ tOperation->incValue((U_Int32)1, (U_Int32)8192);
+
+ if (tConnection->execute( Commit ) != -1 ) {
+ U_Int64 tValue = tNextId_Result->u_64_value(); // Read result value
+ theFirstTransId = tValue;
+ theLastTransId = tValue + 8191;
+ closeTransaction(tConnection);
+ return startTransactionLocal(aPriority, nodeId);
+ }
+ }
+ /**
+ * IntialReadSize = 0;
+ * InterpretedSize = incValue(1);
+ * FinalUpdateSize = 0;
+ * FinalReadSize = 1; // Read value
+ * SubroutineSize = 0;
+ */
+#endif
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: Transaction execution request
+ * ------------------------------------------------------------------------
+ *
+ * Handle requests to execute a prepared transaction
+ **************************************************************************/
+
+void
+DbUtil::execUTIL_EXECUTE_REQ(Signal* signal)
+{
+ jamEntry();
+
+ UtilExecuteReq * req = (UtilExecuteReq *)signal->getDataPtr();
+ const Uint32 clientRef = req->senderRef;
+ const Uint32 clientData = req->senderData;
+ const Uint32 prepareId = req->getPrepareId();
+ const bool releaseFlag = req->getReleaseFlag();
+
+ if(signal->getNoOfSections() == 0) {
+ // Missing prepare data
+ jam();
+ releaseSections(signal);
+ sendUtilExecuteRef(signal, UtilExecuteRef::MissingDataSection,
+ 0, clientRef, clientData);
+ return;
+ }
+ /*******************************
+ * Get PreparedOperation struct
+ *******************************/
+ PreparedOperationPtr prepOpPtr;
+ c_runningPreparedOperations.first(prepOpPtr);
+ while (!prepOpPtr.isNull() && prepOpPtr.i != prepareId)
+ c_runningPreparedOperations.next(prepOpPtr);
+
+ if (prepOpPtr.i != prepareId) {
+ jam();
+ releaseSections(signal);
+ sendUtilExecuteRef(signal, UtilExecuteRef::IllegalPrepareId,
+ 0, clientRef, clientData);
+ return;
+ }
+
+ prepOpPtr.p->releaseFlag = releaseFlag;
+
+ TransactionPtr transPtr;
+ OperationPtr opPtr;
+ SegmentedSectionPtr headerPtr, dataPtr;
+
+ signal->getSection(headerPtr, UtilExecuteReq::HEADER_SECTION);
+ SectionReader headerReader(headerPtr, getSectionSegmentPool());
+ signal->getSection(dataPtr, UtilExecuteReq::DATA_SECTION);
+ SectionReader dataReader(dataPtr, getSectionSegmentPool());
+
+#if 0 //def EVENT_DEBUG
+ // Debugging
+ printf("DbUtil::execUTIL_EXECUTEL_REQ: Headers (%u): ", headerPtr.sz);
+ Uint32 word;
+ while(headerReader.getWord(&word))
+ printf("H'%.8x ", word);
+ printf("\n");
+ printf("DbUtil::execUTIL_EXECUTEL_REQ: Data (%u): ", dataPtr.sz);
+ headerReader.reset();
+ while(dataReader.getWord(&word))
+ printf("H'%.8x ", word);
+ printf("\n");
+ dataReader.reset();
+#endif
+
+// Uint32 totalDataLen = headerPtr.sz + dataPtr.sz;
+
+ /************************************************************
+ * Seize Transaction record
+ ************************************************************/
+ ndbrequire(c_runningTransactions.seize(transPtr));
+ transPtr.p->gsn = GSN_UTIL_EXECUTE_REQ;
+ transPtr.p->clientRef = clientRef;
+ transPtr.p->clientData = clientData;
+ ndbrequire(transPtr.p->operations.seize(opPtr));
+ opPtr.p->prepOp = prepOpPtr.p;
+ opPtr.p->prepOp_i = prepOpPtr.i;
+
+#if 0 //def EVENT_DEBUG
+ printf("opPtr.p->rs.seize( %u )\n", prepOpPtr.p->rsLen);
+#endif
+ ndbrequire(opPtr.p->rs.seize(prepOpPtr.p->rsLen));
+
+ /***********************************************************
+ * Store signal data on linear memory in Transaction record
+ ***********************************************************/
+ KeyInfoBuffer* keyInfo = &opPtr.p->keyInfo;
+ AttrInfoBuffer* attrInfo = &opPtr.p->attrInfo;
+ AttributeHeader header;
+ Uint32* tempBuf = signal->theData + 25;
+ bool dataComplete = true;
+
+ while(headerReader.getWord((Uint32 *)&header)) {
+ Uint32* bufStart = tempBuf;
+ header.insertHeader(tempBuf++);
+ for(unsigned int i = 0; i < header.getDataSize(); i++) {
+ if (!dataReader.getWord(tempBuf++)) {
+ dataComplete = false;
+ break;
+ }
+ }
+ bool res = true;
+
+#if 0 //def EVENT_DEBUG
+ if (TcKeyReq::getOperationType(prepOpPtr.p->tckey.requestInfo) ==
+ TcKeyReq::Read) {
+ if(prepOpPtr.p->pkBitmask.get(header.getAttributeId()))
+ printf("PrimaryKey\n");
+ }
+ printf("AttrId %u Hdrsz %d Datasz %u \n",
+ header.getAttributeId(),
+ header.getHeaderSize(),
+ header.getDataSize());
+#endif
+
+ if(prepOpPtr.p->pkBitmask.get(header.getAttributeId()))
+ // A primary key attribute
+ res = keyInfo->append(bufStart + header.getHeaderSize(),
+ header.getDataSize());
+
+ switch (TcKeyReq::getOperationType(prepOpPtr.p->tckey.requestInfo)) {
+ case ZREAD:
+ res &= attrInfo->append(bufStart, header.getHeaderSize());
+ break;
+ case ZDELETE:
+ // no attrinfo for Delete
+ break;
+ default:
+ res &= attrInfo->append(bufStart,
+ header.getHeaderSize() + header.getDataSize());
+ }
+
+ if (!res) {
+ // Failed to allocate buffer data
+ jam();
+ releaseSections(signal);
+ sendUtilExecuteRef(signal, UtilExecuteRef::AllocationError,
+ 0, clientRef, clientData);
+ releaseTransaction(transPtr);
+ return;
+ }
+ }
+ if (!dataComplete) {
+ // Missing data in data section
+ jam();
+ releaseSections(signal);
+ sendUtilExecuteRef(signal, UtilExecuteRef::MissingData,
+ 0, clientRef, clientData);
+ releaseTransaction(transPtr);
+ return;
+ }
+
+ const Uint32 l1 = prepOpPtr.p->tckey.attrLen;
+ const Uint32 l2 =
+ prepOpPtr.p->attrInfo.getSize() + opPtr.p->attrInfo.getSize();
+
+ if (TcKeyReq::getOperationType(prepOpPtr.p->tckey.requestInfo) != ZREAD){
+ ndbrequire(l1 == l2);
+ } else {
+#if 0
+ ndbout_c("TcKeyReq::Read");
+#endif
+ }
+
+ releaseSections(signal);
+ runTransaction(signal, transPtr);
+}
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: General transaction machinery
+ * ------------------------------------------------------------------------
+ * Executes a prepared transaction
+ **************************************************************************/
+void
+DbUtil::runTransaction(Signal* signal, TransactionPtr transPtr){
+
+ /* Init transaction */
+ transPtr.p->sent = 0;
+ transPtr.p->recv = 0;
+ transPtr.p->errorCode = 0;
+ getTransId(transPtr.p);
+
+ OperationPtr opPtr;
+ ndbrequire(transPtr.p->operations.first(opPtr));
+
+ /* First operation */
+ Uint32 start = 0;
+ TcKeyReq::setStartFlag(start, 1);
+ runOperation(signal, transPtr, opPtr, start);
+ transPtr.p->sent ++;
+
+ /* Rest of operations */
+ start = 0;
+ while(opPtr.i != RNIL){
+ runOperation(signal, transPtr, opPtr, start);
+ transPtr.p->sent ++;
+ }
+ //transPtr.p->print();
+}
+
+void
+DbUtil::runOperation(Signal* signal, TransactionPtr & transPtr,
+ OperationPtr & opPtr, Uint32 start) {
+ Uint32 opI = opPtr.i;
+ Operation * op = opPtr.p;
+ const PreparedOperation * pop = op->prepOp;
+
+ Uint32 lastFlag = 0;
+ if(!transPtr.p->operations.next(opPtr)){
+ TcKeyReq::setCommitFlag(start, 1); // Last operation
+ TcKeyReq::setExecuteFlag(start, 1);
+ }
+
+#if 0 //def EVENT_DEBUG
+ if (TcKeyReq::getOperationType(pop->tckey.requestInfo) ==
+ TcKeyReq::Read) {
+ printf("TcKeyReq::Read runOperation\n");
+ }
+#endif
+
+ /**
+ * Init operation w.r.t result set
+ */
+ initResultSet(op->rs, pop->rsInfo);
+ op->rs.first(op->rsIterator);
+ op->rsRecv = 0;
+#if 0 //def EVENT_DEBUG
+ printf("pop->rsLen %u\n", pop->rsLen);
+#endif
+ op->rsExpect = 0;
+ op->transPtrI = transPtr.i;
+
+ TcKeyReq * tcKey = (TcKeyReq*)signal->getDataPtrSend();
+ //ndbout << "*** 6 ***"<< endl; pop->print();
+ memcpy(tcKey, &pop->tckey, pop->tckeyLenInBytes);
+ //ndbout << "*** 6b ***"<< endl;
+ //printTCKEYREQ(stdout, signal->getDataPtrSend(),
+ // pop->tckeyLenInBytes >> 2, 0);
+ tcKey->apiConnectPtr = transPtr.p->connectPtr;
+ tcKey->senderData = opI;
+ tcKey->transId1 = transPtr.p->transId[0];
+ tcKey->transId2 = transPtr.p->transId[1];
+ tcKey->requestInfo |= start;
+
+#if 0 //def EVENT_DEBUG
+ // Debugging
+ printf("DbUtil::runOperation: KEYINFO\n");
+ op->keyInfo.print(stdout);
+ printf("DbUtil::runOperation: ATTRINFO\n");
+ op->attrInfo.print(stdout);
+#endif
+
+ /**
+ * Key Info
+ */
+ //KeyInfoBuffer::DataBufferIterator kit;
+ KeyInfoIterator kit;
+ op->keyInfo.first(kit);
+ Uint32 *keyDst = ((Uint32*)tcKey) + pop->keyDataPos;
+ for(Uint32 i = 0; i<8 && kit.curr.i != RNIL; i++, op->keyInfo.next(kit)){
+ keyDst[i] = * kit.data;
+ }
+ //ndbout << "*** 7 ***" << endl;
+ //printTCKEYREQ(stdout, signal->getDataPtrSend(),
+ // pop->tckeyLenInBytes >> 2, 0);
+
+#if 0 //def EVENT_DEBUG
+ printf("DbUtil::runOperation: sendSignal(DBTC_REF, GSN_TCKEYREQ, signal, %d , JBB)\n", pop->tckeyLenInBytes >> 2);
+ printTCKEYREQ(stdout, signal->getDataPtr(), pop->tckeyLenInBytes >> 2,0);
+#endif
+ sendSignal(DBTC_REF, GSN_TCKEYREQ, signal, pop->tckeyLenInBytes >> 2, JBB);
+
+ /**
+ * More the 8 words of key info not implemented
+ */
+ // ndbrequire(kit.curr.i == RNIL); // Yes it is
+
+ /**
+ * KeyInfo
+ */
+ KeyInfo* keyInfo = (KeyInfo *)signal->getDataPtrSend();
+ keyInfo->connectPtr = transPtr.p->connectPtr;
+ keyInfo->transId[0] = transPtr.p->transId[0];
+ keyInfo->transId[1] = transPtr.p->transId[1];
+ sendKeyInfo(signal, keyInfo, op->keyInfo, kit);
+
+ /**
+ * AttrInfo
+ */
+ AttrInfo* attrInfo = (AttrInfo *)signal->getDataPtrSend();
+ attrInfo->connectPtr = transPtr.p->connectPtr;
+ attrInfo->transId[0] = transPtr.p->transId[0];
+ attrInfo->transId[1] = transPtr.p->transId[1];
+
+ AttrInfoIterator ait;
+ pop->attrInfo.first(ait);
+ sendAttrInfo(signal, attrInfo, pop->attrInfo, ait);
+
+ op->attrInfo.first(ait);
+ sendAttrInfo(signal, attrInfo, op->attrInfo, ait);
+}
+
+void
+DbUtil::sendKeyInfo(Signal* signal,
+ KeyInfo* keyInfo,
+ const KeyInfoBuffer & keyBuf,
+ KeyInfoIterator & kit)
+{
+ while(kit.curr.i != RNIL) {
+ Uint32 *keyDst = keyInfo->keyData;
+ Uint32 keyDataLen = 0;
+ for(Uint32 i = 0; i<KeyInfo::DataLength && kit.curr.i != RNIL;
+ i++, keyBuf.next(kit)){
+ keyDst[i] = * kit.data;
+ keyDataLen++;
+ }
+#if 0 //def EVENT_DEBUG
+ printf("DbUtil::sendKeyInfo: sendSignal(DBTC_REF, GSN_KEYINFO, signal, %d , JBB)\n", KeyInfo::HeaderLength + keyDataLen);
+#endif
+ sendSignal(DBTC_REF, GSN_KEYINFO, signal,
+ KeyInfo::HeaderLength + keyDataLen, JBB);
+ }
+}
+
+void
+DbUtil::sendAttrInfo(Signal* signal,
+ AttrInfo* attrInfo,
+ const AttrInfoBuffer & attrBuf,
+ AttrInfoIterator & ait)
+{
+ while(ait.curr.i != RNIL) {
+ Uint32 *attrDst = attrInfo->attrData;
+ Uint32 i = 0;
+ for(i = 0; i<AttrInfo::DataLength && ait.curr.i != RNIL;
+ i++, attrBuf.next(ait)){
+ attrDst[i] = * ait.data;
+ }
+#if 0 //def EVENT_DEBUG
+ printf("DbUtil::sendAttrInfo: sendSignal(DBTC_REF, GSN_ATTRINFO, signal, %d , JBB)\n", AttrInfo::HeaderLength + i);
+#endif
+ sendSignal(DBTC_REF, GSN_ATTRINFO, signal,
+ AttrInfo::HeaderLength + i, JBB);
+ }
+}
+
+void
+DbUtil::initResultSet(ResultSetBuffer & rs,
+ const ResultSetInfoBuffer & rsi){
+
+ ResultSetBuffer::DataBufferIterator rsit;
+ rs.first(rsit);
+
+ ResultSetInfoBuffer::ConstDataBufferIterator rsiit;
+ for(rsi.first(rsiit); rsiit.curr.i != RNIL; rsi.next(rsiit)){
+ ndbrequire(rsit.curr.i != RNIL);
+
+ rsit.data[0] = rsiit.data[0];
+#if 0 //def EVENT_DEBUG
+ printf("Init resultset %u, sz %d\n",
+ rsit.curr.i,
+ ((AttributeHeader*)&rsit.data[0])->getDataSize() + 1);
+#endif
+ rs.next(rsit, ((AttributeHeader*)&rsit.data[0])->getDataSize() + 1);
+ }
+}
+
+void
+DbUtil::getTransId(Transaction * transP){
+
+ Uint32 tmp[2];
+ tmp[0] = c_transId[0];
+ tmp[1] = c_transId[1];
+
+ transP->transId[0] = tmp[0];
+ transP->transId[1] = tmp[1];
+
+ c_transId[1] = tmp[1] + 1;
+}
+
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: Post Execute
+ * ------------------------------------------------------------------------
+ *
+ * Handles result from a sent transaction
+ **************************************************************************/
+
+/**
+ * execTRANSID_AI
+ *
+ * Receive result from transaction
+ *
+ * NOTE: This codes assumes that
+ * TransidAI::DataLength = ResultSetBuffer::getSegmentSize() * n
+ */
+void
+DbUtil::execTRANSID_AI(Signal* signal){
+ jamEntry();
+#if 0 //def EVENT_DEBUG
+ ndbout_c("File: %s line: %u",__FILE__,__LINE__);
+#endif
+
+ const Uint32 opI = signal->theData[0];
+ const Uint32 transId1 = signal->theData[1];
+ const Uint32 transId2 = signal->theData[2];
+ const Uint32 dataLen = signal->length() - 3;
+
+ Operation * opP = c_operationPool.getPtr(opI);
+ TransactionPtr transPtr;
+ c_runningTransactions.getPtr(transPtr, opP->transPtrI);
+
+ ndbrequire(transId1 == transPtr.p->transId[0] &&
+ transId2 == transPtr.p->transId[1]);
+ opP->rsRecv += dataLen;
+
+ /**
+ * Save result
+ */
+ Uint32 srcSz = dataLen;
+ const Uint32 *src = &signal->theData[3];
+ const Uint32 segSize = opP->rs.getSegmentSize();
+
+#if 0 //def EVENT_DEBUG
+ printf("rsRecv %u, dataLen %u, rsExpect %u\n",
+ opP->rsRecv, dataLen, opP->rsExpect);
+#endif
+
+ ResultSetBuffer::DataBufferIterator rs = opP->rsIterator;
+
+#if 0 //def EVENT_DEBUG
+ for(int i = 0; i < dataLen; i++)
+ printf("H'%.8x ", src[i]);
+#endif
+
+ ndbrequire(opP->rs.import(rs,src,dataLen));
+ opP->rs.next(rs, dataLen);
+
+#if 0 // replaced this section with import() above
+ while(srcSz > segSize){
+ ndbrequire(rs.curr.i != RNIL);
+ memcpy(rs.data, src, segSize << 2);
+ opP->rs.next(rs, segSize);
+ srcSz -= segSize;
+ // src += segSize * 4; // Bug?
+ src += segSize;
+ }
+
+ if(srcSz > 0){
+ jam();
+ memcpy(rs.data, src, srcSz << 2);
+ rs.curr.i = RNIL;
+ rs.data = 0;
+ }
+#endif
+
+ opP->rsIterator = rs;
+
+ if(!opP->complete()){
+ jam();
+ return;
+ }
+
+#if 0 //def EVENT_DEBUG
+ printf("op complete\n");
+#endif
+
+ transPtr.p->recv++;
+ if(!transPtr.p->complete()){
+ jam();
+ return;
+ }
+
+#if 0 //def EVENT_DEBUG
+ printf("trans complete\n");
+#endif
+
+ finishTransaction(signal, transPtr);
+}
+
+void
+DbUtil::execTCKEYCONF(Signal* signal){
+ jamEntry();
+#if 0 //def EVENT_DEBUG
+ ndbout_c("File: %s line: %u",__FILE__,__LINE__);
+#endif
+
+ TcKeyConf * keyConf = (TcKeyConf*)signal->getDataPtr();
+
+ //const Uint32 gci = keyConf->gci;
+ const Uint32 transI = keyConf->apiConnectPtr >> 1;
+ const Uint32 confInfo = keyConf->confInfo;
+ const Uint32 transId1 = keyConf->transId1;
+ const Uint32 transId2 = keyConf->transId2;
+
+ Uint32 recv = 0;
+ const Uint32 ops = TcKeyConf::getNoOfOperations(confInfo);
+ for(Uint32 i = 0; i<ops; i++){
+ OperationPtr opPtr;
+ c_operationPool.getPtr(opPtr, keyConf->operations[i].apiOperationPtr);
+
+ ndbrequire(opPtr.p->transPtrI == transI);
+ opPtr.p->rsExpect += keyConf->operations[i].attrInfoLen;
+ if(opPtr.p->complete()){
+ recv++;
+ }
+ }
+
+ /**
+ * Check commit ack marker flag
+ */
+ if (TcKeyConf::getMarkerFlag(confInfo)){
+ signal->theData[0] = transId1;
+ signal->theData[1] = transId2;
+ sendSignal(DBTC_REF, GSN_TC_COMMIT_ACK, signal, 2, JBB);
+ }//if
+
+ TransactionPtr transPtr;
+ c_runningTransactions.getPtr(transPtr, transI);
+ ndbrequire(transId1 == transPtr.p->transId[0] &&
+ transId2 == transPtr.p->transId[1]);
+
+ transPtr.p->recv += recv;
+ if(!transPtr.p->complete()){
+ jam();
+ return;
+ }
+ finishTransaction(signal, transPtr);
+}
+
+void
+DbUtil::execTCKEYREF(Signal* signal){
+ jamEntry();
+#if 0 //def EVENT_DEBUG
+ ndbout_c("File: %s line: %u",__FILE__,__LINE__);
+#endif
+
+ const Uint32 transI = signal->theData[0] >> 1;
+ const Uint32 transId1 = signal->theData[1];
+ const Uint32 transId2 = signal->theData[2];
+ const Uint32 errCode = signal->theData[3];
+
+ TransactionPtr transPtr;
+ c_runningTransactions.getPtr(transPtr, transI);
+ ndbrequire(transId1 == transPtr.p->transId[0] &&
+ transId2 == transPtr.p->transId[1]);
+
+ //if(getClassification(errCode) == PermanentError){
+ //}
+
+ //ndbout << "Transaction error (code: " << errCode << ")" << endl;
+
+ transPtr.p->errorCode = errCode;
+ finishTransaction(signal, transPtr);
+}
+
+void
+DbUtil::execTCROLLBACKREP(Signal* signal){
+ jamEntry();
+#if 0 //def EVENT_DEBUG
+ ndbout_c("File: %s line: %u",__FILE__,__LINE__);
+#endif
+
+ const Uint32 transI = signal->theData[0] >> 1;
+ const Uint32 transId1 = signal->theData[1];
+ const Uint32 transId2 = signal->theData[2];
+ const Uint32 errCode = signal->theData[3];
+
+ TransactionPtr transPtr;
+ c_runningTransactions.getPtr(transPtr, transI);
+ ndbrequire(transId1 == transPtr.p->transId[0] &&
+ transId2 == transPtr.p->transId[1]);
+
+ //if(getClassification(errCode) == PermanentError){
+ //}
+
+#if 0 //def EVENT_DEBUG
+ ndbout << "Transaction error (code: " << errCode << ")" << endl;
+#endif
+
+ transPtr.p->errorCode = errCode;
+ finishTransaction(signal, transPtr);
+}
+
+void
+DbUtil::finishTransaction(Signal* signal, TransactionPtr transPtr){
+#if 0 //def EVENT_DEBUG
+ ndbout_c("Transaction %x %x completed %s",
+ transPtr.p->transId[0],
+ transPtr.p->transId[1],
+ transPtr.p->errorCode == 0 ? "OK" : "FAILED");
+#endif
+
+ /*
+ How to find the correct RS? Could we have multi-RS/transaction?
+
+ Operation * opP = c_operationPool.getPtr(opI);
+
+ ResultSetBuffer::DataBufferIterator rsit;
+ ndbrequire(opP->rs.first(rsit));
+ ndbout << "F Result: " << rsit.data << endl;
+
+ while (opP->rs.next(rsit)) {
+ ndbout << "R Result: " << rsit.data << endl;
+ }
+ */
+
+ switch(transPtr.p->gsn){
+ case GSN_UTIL_SEQUENCE_REQ:
+ jam();
+ reportSequence(signal, transPtr.p);
+ break;
+ case GSN_UTIL_EXECUTE_REQ:
+ if (transPtr.p->errorCode) {
+ UtilExecuteRef * ret = (UtilExecuteRef *)signal->getDataPtrSend();
+ ret->senderData = transPtr.p->clientData;
+ ret->errorCode = UtilExecuteRef::TCError;
+ ret->TCErrorCode = transPtr.p->errorCode;
+ sendSignal(transPtr.p->clientRef, GSN_UTIL_EXECUTE_REF, signal,
+ UtilExecuteRef::SignalLength, JBB);
+ } else {
+ struct LinearSectionPtr sectionsPtr[UtilExecuteReq::NoOfSections];
+ UtilExecuteConf * ret = (UtilExecuteConf *)signal->getDataPtrSend();
+ ret->senderData = transPtr.p->clientData;
+ if (getResultSet(signal, transPtr.p, sectionsPtr)) {
+#if 0 //def EVENT_DEBUG
+ for (int j = 0; j < 2; j++) {
+ printf("Result set %u %u\n", j,sectionsPtr[j].sz);
+ for (int i=0; i < sectionsPtr[j].sz; i++)
+ printf("H'%.8x ", sectionsPtr[j].p[i]);
+ printf("\n");
+ }
+#endif
+ sendSignal(transPtr.p->clientRef, GSN_UTIL_EXECUTE_CONF, signal,
+ UtilExecuteConf::SignalLength, JBB,
+ sectionsPtr, UtilExecuteReq::NoOfSections);
+ } else
+ sendSignal(transPtr.p->clientRef, GSN_UTIL_EXECUTE_CONF, signal,
+ UtilExecuteConf::SignalLength, JBB);
+ }
+ break;
+ default:
+ ndbrequire(0);
+ break;
+ }
+ releaseTransaction(transPtr);
+}
+
+void
+DbUtil::execUTIL_LOCK_REQ(Signal * signal){
+ jamEntry();
+ UtilLockReq * req = (UtilLockReq*)signal->getDataPtr();
+ const Uint32 lockId = req->lockId;
+
+ LockQueuePtr lockQPtr;
+ if(!c_lockQueues.find(lockQPtr, lockId)){
+ jam();
+ sendLOCK_REF(signal, req, UtilLockRef::NoSuchLock);
+ return;
+ }
+
+// const Uint32 requestInfo = req->requestInfo;
+ const Uint32 senderNode = refToNode(req->senderRef);
+ if(senderNode != getOwnNodeId() && senderNode != 0){
+ jam();
+ sendLOCK_REF(signal, req, UtilLockRef::DistributedLockNotSupported);
+ return;
+ }
+
+ LocalDLFifoList<LockQueueElement> queue(c_lockElementPool,
+ lockQPtr.p->m_queue);
+ if(req->requestInfo & UtilLockReq::TryLock && !queue.isEmpty()){
+ jam();
+ sendLOCK_REF(signal, req, UtilLockRef::LockAlreadyHeld);
+ return;
+ }
+
+ LockQueueElementPtr lockEPtr;
+ if(!c_lockElementPool.seize(lockEPtr)){
+ jam();
+ sendLOCK_REF(signal, req, UtilLockRef::OutOfLockRecords);
+ return;
+ }
+
+ lockEPtr.p->m_senderRef = req->senderRef;
+ lockEPtr.p->m_senderData = req->senderData;
+
+ if(queue.isEmpty()){
+ jam();
+ sendLOCK_CONF(signal, lockQPtr.p, lockEPtr.p);
+ }
+
+ queue.add(lockEPtr);
+}
+
+void
+DbUtil::execUTIL_UNLOCK_REQ(Signal* signal){
+ jamEntry();
+
+ UtilUnlockReq * req = (UtilUnlockReq*)signal->getDataPtr();
+ const Uint32 lockId = req->lockId;
+
+ LockQueuePtr lockQPtr;
+ if(!c_lockQueues.find(lockQPtr, lockId)){
+ jam();
+ sendUNLOCK_REF(signal, req, UtilUnlockRef::NoSuchLock);
+ return;
+ }
+
+ LocalDLFifoList<LockQueueElement> queue(c_lockElementPool,
+ lockQPtr.p->m_queue);
+ LockQueueElementPtr lockEPtr;
+ if(!queue.first(lockEPtr)){
+ jam();
+ sendUNLOCK_REF(signal, req, UtilUnlockRef::NotLockOwner);
+ return;
+ }
+
+ if(lockQPtr.p->m_lockKey != req->lockKey){
+ jam();
+ sendUNLOCK_REF(signal, req, UtilUnlockRef::NotLockOwner);
+ return;
+ }
+
+ sendUNLOCK_CONF(signal, lockQPtr.p, lockEPtr.p);
+ queue.release(lockEPtr);
+
+ if(queue.first(lockEPtr)){
+ jam();
+ sendLOCK_CONF(signal, lockQPtr.p, lockEPtr.p);
+ return;
+ }
+}
+
+void
+DbUtil::sendLOCK_REF(Signal* signal,
+ const UtilLockReq * req, UtilLockRef::ErrorCode err){
+ const Uint32 senderData = req->senderData;
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 lockId = req->lockId;
+
+ UtilLockRef * ref = (UtilLockRef*)signal->getDataPtrSend();
+ ref->senderData = senderData;
+ ref->senderRef = reference();
+ ref->lockId = lockId;
+ ref->errorCode = err;
+ sendSignal(senderRef, GSN_UTIL_LOCK_REF, signal,
+ UtilLockRef::SignalLength, JBB);
+}
+
+void
+DbUtil::sendLOCK_CONF(Signal* signal,
+ LockQueue * lockQP,
+ LockQueueElement * lockEP){
+ const Uint32 senderData = lockEP->m_senderData;
+ const Uint32 senderRef = lockEP->m_senderRef;
+ const Uint32 lockId = lockQP->m_lockId;
+ const Uint32 lockKey = ++lockQP->m_lockKey;
+
+ UtilLockConf * conf = (UtilLockConf*)signal->getDataPtrSend();
+ conf->senderData = senderData;
+ conf->senderRef = reference();
+ conf->lockId = lockId;
+ conf->lockKey = lockKey;
+ sendSignal(senderRef, GSN_UTIL_LOCK_CONF, signal,
+ UtilLockConf::SignalLength, JBB);
+}
+
+void
+DbUtil::sendUNLOCK_REF(Signal* signal,
+ const UtilUnlockReq* req, UtilUnlockRef::ErrorCode err){
+
+ const Uint32 senderData = req->senderData;
+ const Uint32 senderRef = req->senderRef;
+ const Uint32 lockId = req->lockId;
+
+ UtilUnlockRef * ref = (UtilUnlockRef*)signal->getDataPtrSend();
+ ref->senderData = senderData;
+ ref->senderRef = reference();
+ ref->lockId = lockId;
+ ref->errorCode = err;
+ sendSignal(senderRef, GSN_UTIL_UNLOCK_REF, signal,
+ UtilUnlockRef::SignalLength, JBB);
+}
+
+void
+DbUtil::sendUNLOCK_CONF(Signal* signal,
+ LockQueue * lockQP,
+ LockQueueElement * lockEP){
+ const Uint32 senderData = lockEP->m_senderData;
+ const Uint32 senderRef = lockEP->m_senderRef;
+ const Uint32 lockId = lockQP->m_lockId;
+ ++lockQP->m_lockKey;
+
+ UtilUnlockConf * conf = (UtilUnlockConf*)signal->getDataPtrSend();
+ conf->senderData = senderData;
+ conf->senderRef = reference();
+ conf->lockId = lockId;
+ sendSignal(senderRef, GSN_UTIL_UNLOCK_CONF, signal,
+ UtilUnlockConf::SignalLength, JBB);
+}
+
+void
+DbUtil::execUTIL_CREATE_LOCK_REQ(Signal* signal){
+ jamEntry();
+ UtilCreateLockReq req = * (UtilCreateLockReq*)signal->getDataPtr();
+
+ UtilCreateLockRef::ErrorCode err = UtilCreateLockRef::OK;
+
+ do {
+ LockQueuePtr lockQPtr;
+ if(c_lockQueues.find(lockQPtr, req.lockId)){
+ jam();
+ err = UtilCreateLockRef::LockIdAlreadyUsed;
+ break;
+ }
+
+ if(req.lockType != UtilCreateLockReq::Mutex){
+ jam();
+ err = UtilCreateLockRef::UnsupportedLockType;
+ break;
+ }
+
+ if(!c_lockQueues.seize(lockQPtr)){
+ jam();
+ err = UtilCreateLockRef::OutOfLockQueueRecords;
+ break;
+ }
+
+ new (lockQPtr.p) LockQueue(req.lockId);
+ c_lockQueues.add(lockQPtr);
+
+ UtilCreateLockConf * conf = (UtilCreateLockConf*)signal->getDataPtrSend();
+ conf->senderData = req.senderData;
+ conf->senderRef = reference();
+ conf->lockId = req.lockId;
+
+ sendSignal(req.senderRef, GSN_UTIL_CREATE_LOCK_CONF, signal,
+ UtilCreateLockConf::SignalLength, JBB);
+ return;
+ } while(false);
+
+ UtilCreateLockRef * ref = (UtilCreateLockRef*)signal->getDataPtrSend();
+ ref->senderData = req.senderData;
+ ref->senderRef = reference();
+ ref->lockId = req.lockId;
+ ref->errorCode = err;
+
+ sendSignal(req.senderRef, GSN_UTIL_CREATE_LOCK_REF, signal,
+ UtilCreateLockRef::SignalLength, JBB);
+}
+
+void
+DbUtil::execUTIL_DESTORY_LOCK_REQ(Signal* signal){
+ jamEntry();
+
+ UtilDestroyLockReq req = * (UtilDestroyLockReq*)signal->getDataPtr();
+ UtilDestroyLockRef::ErrorCode err = UtilDestroyLockRef::OK;
+ do {
+ LockQueuePtr lockQPtr;
+ if(!c_lockQueues.find(lockQPtr, req.lockId)){
+ jam();
+ err = UtilDestroyLockRef::NoSuchLock;
+ break;
+ }
+
+ LocalDLFifoList<LockQueueElement> queue(c_lockElementPool,
+ lockQPtr.p->m_queue);
+ LockQueueElementPtr lockEPtr;
+ if(!queue.first(lockEPtr)){
+ jam();
+ err = UtilDestroyLockRef::NotLockOwner;
+ break;
+ }
+
+ if(lockQPtr.p->m_lockKey != req.lockKey){
+ jam();
+ err = UtilDestroyLockRef::NotLockOwner;
+ break;
+ }
+
+ /**
+ * OK
+ */
+
+ // Inform all in lock queue that queue has been destroyed
+ UtilLockRef * ref = (UtilLockRef*)signal->getDataPtrSend();
+ ref->lockId = req.lockId;
+ ref->errorCode = UtilLockRef::NoSuchLock;
+ ref->senderRef = reference();
+ LockQueueElementPtr loopPtr = lockEPtr;
+ for(queue.next(loopPtr); !loopPtr.isNull(); queue.next(loopPtr)){
+ jam();
+ ref->senderData = loopPtr.p->m_senderData;
+ const Uint32 senderRef = loopPtr.p->m_senderRef;
+ sendSignal(senderRef, GSN_UTIL_LOCK_REF, signal,
+ UtilLockRef::SignalLength, JBB);
+ }
+ queue.release();
+ c_lockQueues.release(lockQPtr);
+
+ // Send Destroy conf
+ UtilDestroyLockConf* conf=(UtilDestroyLockConf*)signal->getDataPtrSend();
+ conf->senderData = req.senderData;
+ conf->senderRef = reference();
+ conf->lockId = req.lockId;
+ sendSignal(req.senderRef, GSN_UTIL_DESTROY_LOCK_CONF, signal,
+ UtilDestroyLockConf::SignalLength, JBB);
+ return;
+ } while(false);
+
+ UtilDestroyLockRef * ref = (UtilDestroyLockRef*)signal->getDataPtrSend();
+ ref->senderData = req.senderData;
+ ref->senderRef = reference();
+ ref->lockId = req.lockId;
+ ref->errorCode = err;
+ sendSignal(req.senderRef, GSN_UTIL_DESTROY_LOCK_REF, signal,
+ UtilDestroyLockRef::SignalLength, JBB);
+}
diff --git a/ndb/src/kernel/blocks/dbutil/DbUtil.hpp b/ndb/src/kernel/blocks/dbutil/DbUtil.hpp
new file mode 100644
index 00000000000..8ab2fe8d8d0
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbutil/DbUtil.hpp
@@ -0,0 +1,484 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBUTIL_H
+#define DBUTIL_H
+
+#include <ndb_limits.h>
+#include <SimulatedBlock.hpp>
+#include <NodeBitmask.hpp>
+
+#include <ArrayList.hpp>
+#include <ArrayPool.hpp>
+#include <SLList.hpp>
+#include <DLList.hpp>
+#include <DLFifoList.hpp>
+#include <DataBuffer.hpp>
+#include <KeyTable.hpp>
+
+#include <signaldata/KeyInfo.hpp>
+#include <signaldata/AttrInfo.hpp>
+#include <signaldata/TcKeyReq.hpp>
+#include <signaldata/UtilPrepare.hpp>
+#include <signaldata/UtilExecute.hpp>
+#include <signaldata/UtilLock.hpp>
+#include <SimpleProperties.hpp>
+
+#define UTIL_WORDS_PER_PAGE 8191
+
+/**
+ * @class DbUtil
+ * @brief Database utilities
+ *
+ * This block implements transactional services which can be used by other
+ * blocks.
+ *
+ * @section secSequence Module: The Sequence Service
+ *
+ * A sequence is a varaible stored in the database. Each time it is
+ * requested with "NextVal" it returns a unique number. If requested
+ * with "CurrVal" it returns the current number.
+ *
+ * - Request: SEQUENCE_REQ
+ * Requests the 'NextVal' or 'CurrVal' for sequence variable 'sequenceId'.
+ *
+ * - Response: SEQUENCE_CONF / REF (if failure)
+ * Returns value requested.
+ */
+class DbUtil : public SimulatedBlock
+{
+public:
+ DbUtil(const class Configuration & conf);
+ virtual ~DbUtil();
+ BLOCK_DEFINES(DbUtil);
+
+protected:
+ /**
+ * Startup & Misc
+ */
+ void execSTTOR(Signal* signal);
+ void execNDB_STTOR(Signal* signal);
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+
+ /**
+ * Sequence Service : Public interface
+ */
+ void execUTIL_SEQUENCE_REQ(Signal* signal);
+ void execUTIL_SEQUENCE_REF(Signal* signal);
+ void execUTIL_SEQUENCE_CONF(Signal* signal);
+
+ /**
+ * Prepare Service : Public interface
+ */
+ void execUTIL_PREPARE_REQ(Signal* signal);
+ void execUTIL_PREPARE_CONF(Signal* signal);
+ void execUTIL_PREPARE_REF(Signal* signal);
+
+ /**
+ * Delete Service : Public interface
+ */
+ void execUTIL_DELETE_REQ(Signal* signal);
+ void execUTIL_DELETE_REF(Signal* signal);
+ void execUTIL_DELETE_CONF(Signal* signal);
+
+ /**
+ * Execute Service : Public interface
+ */
+ void execUTIL_EXECUTE_REQ(Signal* signal);
+ void execUTIL_EXECUTE_REF(Signal* signal);
+ void execUTIL_EXECUTE_CONF(Signal* signal);
+
+ /**
+ * Prepare Release Service : Public interface
+ */
+ void execUTIL_RELEASE_REQ(Signal* signal);
+ void execUTIL_RELEASE_CONF(Signal* signal);
+ void execUTIL_RELEASE_REF(Signal* signal);
+
+ /**
+ * Backend interface to a used TC service
+ */
+ void execTCSEIZECONF(Signal* signal);
+ void execTCKEYCONF(Signal* signal);
+ void execTCKEYREF(Signal* signal);
+ void execTCROLLBACKREP(Signal* signal);
+ void execTCKEY_FAILCONF(Signal* signal);
+ void execTCKEY_FAILREF(Signal* signal);
+ void execTRANSID_AI(Signal* signal);
+
+ /**
+ * Backend interface to a used DICT service
+ */
+ void execGET_TABINFOREF(Signal*);
+ void execGET_TABINFO_CONF(Signal* signal);
+
+private:
+
+public:
+ struct PreparedOperation;
+
+ typedef DataBuffer<11> KeyInfoBuffer;
+ typedef KeyInfoBuffer::ConstDataBufferIterator KeyInfoIterator;
+ typedef DataBuffer<11> AttrInfoBuffer;
+ typedef AttrInfoBuffer::ConstDataBufferIterator AttrInfoIterator;
+ typedef DataBuffer<11> ResultSetBuffer;
+ typedef DataBuffer<11> ResultSetInfoBuffer;
+ typedef DataBuffer<1> AttrMappingBuffer;
+
+ /**
+ * @struct Page32
+ * @brief For storing SimpleProperties objects and similar temporary data
+ */
+ struct Page32 {
+ Uint32 data[UTIL_WORDS_PER_PAGE];
+ Uint32 nextPool; // Note: This used as data when seized
+ };
+
+ /**
+ * @struct Prepare
+ * @brief Info regarding prepare request (contains a prepared operation)
+ *
+ * The prepare phase interprets the table and attribute names sent
+ * in the prepare request from the client and asks DICT for meta
+ * information.
+ */
+ struct Prepare {
+ Prepare(ArrayPool<Page32> & ap) : preparePages(ap) {}
+
+ /*** Client info ***/
+ Uint32 clientRef;
+ Uint32 clientData;
+
+ /**
+ * SimpleProp sent in UTIL_PREPARE_REQ
+ *
+ * Example format:
+ * - UtilPrepareReq::NoOfOperations=1
+ * - UtilPrepareReq::OperationType=UtilPrepareReq::Delete
+ * - UtilPrepareReq::TableName="SYSTAB_0"
+ * - UtilPrepareReq::AttributeName="SYSKEY_0"
+ */
+ Uint32 prepDataLen;
+ Array<Page32> preparePages;
+
+ /*** PreparedOperation constructed in Prepare phase ***/
+ Ptr<PreparedOperation> prepOpPtr;
+
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+
+ void print() const {
+ ndbout << "[-Prepare-" << endl
+ << " clientRef: " << clientRef
+ << ", clientData: " << clientData
+ << "]" << endl;
+ }
+ };
+
+ /**
+ * @struct PreparedOperation
+ * @brief Contains instantiated TcKeyReq signaldata for operation
+ *
+ * The prepare phase is finished by storing the request in a
+ * PreparedOperation record.
+ */
+ struct PreparedOperation {
+ PreparedOperation(AttrMappingBuffer::DataBufferPool & am,
+ AttrInfoBuffer::DataBufferPool & ai,
+ ResultSetInfoBuffer::DataBufferPool & rs) :
+ releaseFlag(false), attrMapping(am), attrInfo(ai), rsInfo(rs)
+ {
+ pkBitmask.clear();
+ }
+
+ /*** Various Operation Info ***/
+ Uint32 keyLen; // Length of primary key (fixed size is assumed)
+ Uint32 rsLen; // Size of result set
+ Uint32 noOfKeyAttr; // Number of key attributes
+ Uint32 noOfAttr; // Number of attributes
+ bool releaseFlag; // flag if operation release after completion
+
+ /**
+ * Attribute Mapping
+ *
+ * This datastructure (buffer of AttributeHeader:s) are used to map
+ * each execute request to a TCKEYREQ train of signals.
+ *
+ * The datastructure contains (AttributeId, Position) pairs, where
+ * - AttributeId is id used in database, and
+ * - Position is position of attribute value in TCKEYREQ keyinfo
+ * part of the train of signals which will be send to TC.
+ * Position == 0x3fff means it should *not* be sent
+ * in keyinfo part.
+ */
+ AttrMappingBuffer attrMapping;
+
+ /*** First signal in tckeyreq train ***/
+ Uint32 tckeyLenInBytes; // TcKeyReq total signal length (in bytes)
+ Uint32 keyDataPos; // Where to store keydata[] in tckey signal
+ // (in #words from base in tckey signal)
+ TcKeyReq tckey; // Signaldata for first signal in train
+
+ /*** Attrinfo signals sent to TC (part of tckeyreq train) ***/
+ AttrInfoBuffer attrInfo;
+
+ /*** Result of executed operation ***/
+ ResultSetInfoBuffer rsInfo;
+
+ Bitmask<MAX_ATTRIBUTES_IN_TABLE> pkBitmask;
+
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+
+ void print() const {
+ ndbout << "[-PreparedOperation-" << endl
+ << " keyLen: " << keyLen
+ << ", rsLen: " << rsLen
+ << ", noOfKeyAttr: " << noOfKeyAttr
+ << ", noOfAttr: " << noOfAttr
+ << ", tckeyLenInBytes: " << tckeyLenInBytes
+ << ", keyDataPos: " << keyDataPos << endl
+ << "-AttrMapping- (AttrId, KeyPos)-pairs "
+ << "(Pos=3fff if non-key attr):" << endl;
+ attrMapping.print(stdout);
+ ndbout << "[-tckey- ";
+ printTCKEYREQ(stdout, (Uint32*)&tckey, 8, 0);
+ ndbout << "[-attrInfo- ";
+ attrInfo.print(stdout);
+ ndbout << "[-rsInfo- ";
+ rsInfo.print(stdout);
+ ndbout << "]]]]" << endl;
+ }
+ };
+
+ /**
+ * @struct Operation
+ * @brief Used in execution (contains resultset and buffers for result)
+ */
+ struct Operation {
+ Operation(KeyInfoBuffer::DataBufferPool & ki,
+ AttrInfoBuffer::DataBufferPool & ai,
+ ResultSetBuffer::DataBufferPool & _rs) :
+ prepOp_i(RNIL), keyInfo(ki), attrInfo(ai), rs(_rs) {}
+
+ PreparedOperation * prepOp;
+ Uint32 prepOp_i;
+ KeyInfoBuffer keyInfo;
+ AttrInfoBuffer attrInfo;
+ ResultSetBuffer rs;
+ ResultSetBuffer::DataBufferIterator rsIterator;
+
+ Uint32 transPtrI;
+
+ Uint32 rsRecv;
+ Uint32 rsExpect;
+ inline bool complete() const { return rsRecv == rsExpect; }
+
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+
+ void print() const {
+ ndbout << "[-Operation-" << endl
+ << " transPtrI: " << transPtrI
+ << ", rsRecv: " << rsRecv;
+ ndbout << "[-PreparedOperation-" << endl;
+ prepOp->print();
+ ndbout << "[-keyInfo-" << endl;
+ keyInfo.print(stdout);
+ ndbout << "[-attrInfo-" << endl;
+ attrInfo.print(stdout);
+ ndbout << "]]" << endl;
+ }
+ };
+
+ /**
+ * @struct Transaction
+ * @brief Used in execution (contains list of operations)
+ */
+ struct Transaction {
+ Transaction(ArrayPool<Page32> & ap, ArrayPool<Operation> & op) :
+ executePages(ap), operations(op) {}
+
+ Uint32 clientRef;
+ Uint32 clientData;
+ Array<Page32> executePages;
+
+ Uint32 gsn; // Request type (SEQUENCE, DELETE, etc)
+ union {
+ /**
+ * Sequence transaction
+ */
+ struct {
+ Uint32 sequenceId;
+ Uint32 requestType;
+ } sequence;
+ };
+
+ Uint32 connectPtr;
+ Uint32 transId[2];
+ SLList<Operation> operations;
+
+ Uint32 errorCode;
+ Uint32 sent; // No of operations sent
+ Uint32 recv; // No of completed operations received
+ inline bool complete() const { return sent == recv; };
+
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+
+ void print() const {
+ ndbout << "[-Transaction-" << endl
+ << " clientRef: " << clientRef
+ << ", clientData: " << clientData
+ << ", gsn: " << gsn
+ << ", errorCode: " << errorCode
+ << endl
+ << " sent: " << sent << " operations"
+ << ", recv: " << recv << " completed operations";
+ OperationPtr opPtr;
+ this->operations.first(opPtr);
+ while(opPtr.i != RNIL){
+ ndbout << "[-Operation-" << endl;
+ opPtr.p->print();
+ this->operations.next(opPtr);
+ }
+ ndbout << "]" << endl;
+ }
+ };
+
+ typedef Ptr<Page32> Page32Ptr;
+ typedef Ptr<Prepare> PreparePtr;
+ typedef Ptr<Transaction> TransactionPtr;
+ typedef Ptr<Operation> OperationPtr;
+ typedef Ptr<PreparedOperation> PreparedOperationPtr;
+
+ Uint32 c_transId[2];
+ ArrayPool<Page32> c_pagePool;
+ ArrayPool<Prepare> c_preparePool;
+ ArrayPool<Operation> c_operationPool;
+ ArrayPool<PreparedOperation> c_preparedOperationPool;
+ ArrayPool<Transaction> c_transactionPool;
+
+ DataBuffer<1>::DataBufferPool c_attrMappingPool;
+ DataBuffer<11>::DataBufferPool c_dataBufPool;
+ DLList<Prepare> c_runningPrepares;
+ DLList<PreparedOperation> c_runningPreparedOperations;
+ DLList<Transaction> c_seizingTransactions; // Being seized at TC
+ DLList<Transaction> c_runningTransactions; // Seized and now exec.
+
+ void getTransId(Transaction *);
+ void initResultSet(ResultSetBuffer &, const ResultSetInfoBuffer &);
+ void runTransaction(Signal* signal, TransactionPtr);
+ void runOperation(Signal* signal, TransactionPtr &, OperationPtr &, Uint32);
+ void sendKeyInfo(Signal* signal,
+ KeyInfo* keyInfo,
+ const KeyInfoBuffer & keyBuf,
+ KeyInfoIterator & kit);
+ void sendAttrInfo(Signal*,
+ AttrInfo* attrInfo,
+ const AttrInfoBuffer & attrInfo,
+ AttrInfoIterator & ait);
+ int getResultSet(Signal* signal, const Transaction * transP,
+ struct LinearSectionPtr sectionsPtr[]);
+ void finishTransaction(Signal*, TransactionPtr);
+ void releaseTransaction(TransactionPtr transPtr);
+ void hardcodedPrepare();
+ void connectTc(Signal* signal);
+ void reportSequence(Signal*, const Transaction *);
+ void readPrepareProps(Signal* signal,
+ SimpleProperties::Reader* reader,
+ Uint32 senderData);
+ void prepareOperation(Signal*, PreparePtr);
+ void sendUtilPrepareRef(Signal*, UtilPrepareRef::ErrorCode, Uint32, Uint32);
+ void sendUtilExecuteRef(Signal*, UtilExecuteRef::ErrorCode,
+ Uint32, Uint32, Uint32);
+ void releasePrepare(PreparePtr);
+ void releasePreparedOperation(PreparedOperationPtr);
+
+ /***************************************************************************
+ * Lock manager
+ */
+ struct LockQueueElement {
+ Uint32 m_senderData;
+ Uint32 m_senderRef;
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+ };
+ typedef Ptr<LockQueueElement> LockQueueElementPtr;
+
+ struct LockQueue {
+ LockQueue(){}
+ LockQueue(Uint32 id) : m_queue() { m_lockId = id; m_lockKey = 0;}
+ union {
+ Uint32 m_lockId;
+ Uint32 key;
+ };
+ Uint32 m_lockKey;
+ DLFifoList<LockQueueElement>::Head m_queue;
+ union {
+ Uint32 nextHash;
+ Uint32 nextPool;
+ };
+ Uint32 prevHash;
+
+ Uint32 hashValue() const {
+ return m_lockId;
+ }
+ bool equal(const LockQueue & rec) const {
+ return m_lockId == rec.m_lockId;
+ }
+ };
+ typedef Ptr<LockQueue> LockQueuePtr;
+
+
+ ArrayPool<LockQueue> c_lockQueuePool;
+ ArrayPool<LockQueueElement> c_lockElementPool;
+ KeyTable<LockQueue> c_lockQueues;
+
+ void execUTIL_CREATE_LOCK_REQ(Signal* signal);
+ void execUTIL_DESTORY_LOCK_REQ(Signal* signal);
+ void execUTIL_LOCK_REQ(Signal* signal);
+ void execUTIL_UNLOCK_REQ(Signal* signal);
+
+ void sendLOCK_REF(Signal*, const UtilLockReq * req, UtilLockRef::ErrorCode);
+ void sendLOCK_CONF(Signal*, LockQueue *, LockQueueElement *);
+
+ void sendUNLOCK_REF(Signal*, const UtilUnlockReq*, UtilUnlockRef::ErrorCode);
+ void sendUNLOCK_CONF(Signal*, LockQueue *, LockQueueElement *);
+
+ // For testing of mutex:es
+ void mutex_created(Signal* signal, Uint32 mutexId, Uint32 retVal);
+ void mutex_destroyed(Signal* signal, Uint32 mutexId, Uint32 retVal);
+ void mutex_locked(Signal* signal, Uint32 mutexId, Uint32 retVal);
+ void mutex_unlocked(Signal* signal, Uint32 mutexId, Uint32 retVal);
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/dbutil/DbUtil.txt b/ndb/src/kernel/blocks/dbutil/DbUtil.txt
new file mode 100644
index 00000000000..cc8c1985009
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbutil/DbUtil.txt
@@ -0,0 +1,68 @@
+UTIL Protocols
+--------------
+Transactions are executed in two phases:
+1) PREPARE
+2) EXECUTE
+
+
+PREPARE PHASE
+-------------
+1) ** REQUEST **
+ Client (any block) requests prepare service from Util:
+
+ Client --UTIL_PREPARE_REQ--> Util
+ ...
+ Client --UTIL_PREPARE_REQ--> Util
+
+2) ** DICTINFO **
+ Util requests Dict for information about table:
+
+ Util --GET_TABINFOREQ--> Dict
+
+ Util <--DICTTABINFO-- Dict
+ ...
+ Util <--DICTTABINFO-- Dict
+
+3) ** PREPARE **
+ Operation (= transaction) is prepared (DbUtil::prepareOperation)
+
+ a) AttrMapping is created (a map used to read of the
+ actual execute request attribute values and put them in KEYINFO)
+
+ b) TC Signal train is prepared
+
+4) ** CONFIRM **
+ Request is confirmed
+
+ Client <--UTIL_PREPARE_CONF-- Util
+
+
+EXECUTE PHASE
+-------------
+1) Client (any block) requests execute service from Util:
+ (Execute can be INSERT, DELETE,...)
+
+ Client --UTIL_EXECUTE_REQ--> Util (Multi-signals not yet implemented)
+ ...
+ Client --UTIL_EXECUTE_REQ--> Util
+
+2) Util --TCKEYREQ--> tc
+
+ Util --KEYINFO--> tc (sometimes) (Not yet implemented)
+ ...
+ Util --KEYINFO--> tc
+
+ Util --ATTRINFO--> tc (sometimes)
+ ...
+ Util --ATTRINFO--> tc
+
+3) Util <--TCKEYCONF-- tc
+
+ Util --TC_COMMIT_ACK-->tc (sometimes)
+
+ (in parallel with)
+
+ Util <--TRANSID_AI-- tc (sometimes)
+ ...
+ Util <--TRANSID_AI-- tc
+
diff --git a/ndb/src/kernel/blocks/dbutil/Makefile b/ndb/src/kernel/blocks/dbutil/Makefile
new file mode 100644
index 00000000000..54b7326e4e5
--- /dev/null
+++ b/ndb/src/kernel/blocks/dbutil/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := dbutil
+SOURCES = DbUtil.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/grep/Grep.cpp b/ndb/src/kernel/blocks/grep/Grep.cpp
new file mode 100644
index 00000000000..093e9a225e6
--- /dev/null
+++ b/ndb/src/kernel/blocks/grep/Grep.cpp
@@ -0,0 +1,2001 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Grep.hpp"
+#include <ndb_version.h>
+
+#include <NdbTCP.h>
+#include <Bitmask.hpp>
+
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/ReadNodesConf.hpp>
+#include <signaldata/CheckNodeGroups.hpp>
+#include <signaldata/GrepImpl.hpp>
+#include <signaldata/RepImpl.hpp>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/WaitGCP.hpp>
+#include <GrepEvent.hpp>
+#include <AttributeHeader.hpp>
+
+#define CONTINUEB_DELAY 500
+#define SSREPBLOCKNO 2
+#define PSREPBLOCKNO 2
+
+//#define DEBUG_GREP
+//#define DEBUG_GREP_SUBSCRIPTION
+//#define DEBUG_GREP_TRANSFER
+//#define DEBUG_GREP_APPLY
+//#define DEBUG_GREP_DELETE
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: STARTUP of GREP Block, etc
+ * ------------------------------------------------------------------------
+ **************************************************************************/
+static Uint32 g_TypeOfStart = NodeState::ST_ILLEGAL_TYPE;
+void
+Grep::getNodeGroupMembers(Signal* signal) {
+ jam();
+ /**
+ * Ask DIH for nodeGroupMembers
+ */
+ CheckNodeGroups * sd = (CheckNodeGroups*)signal->getDataPtrSend();
+ sd->blockRef = reference();
+ sd->requestType =
+ CheckNodeGroups::Direct |
+ CheckNodeGroups::GetNodeGroupMembers;
+ sd->nodeId = getOwnNodeId();
+ EXECUTE_DIRECT(DBDIH, GSN_CHECKNODEGROUPSREQ, signal,
+ CheckNodeGroups::SignalLength);
+ jamEntry();
+
+ c_nodeGroup = sd->output;
+ c_noNodesInGroup = 0;
+ for (int i = 0; i < MAX_NDB_NODES; i++) {
+ if (sd->mask.get(i)) {
+ if (i == getOwnNodeId()) c_idInNodeGroup = c_noNodesInGroup;
+ c_nodesInGroup[c_noNodesInGroup] = i;
+ c_noNodesInGroup++;
+ }
+ }
+ ndbrequire(c_noNodesInGroup >= 0); // at least 1 node in the nodegroup
+
+#ifdef NODEFAIL_DEBUG
+ for (Uint32 i = 0; i < c_noNodesInGroup; i++) {
+ ndbout_c ("Grep: NodeGroup %u, me %u, me in group %u, member[%u] %u",
+ c_nodeGroup, getOwnNodeId(), c_idInNodeGroup,
+ i, c_nodesInGroup[i]);
+ }
+#endif
+}
+
+
+void
+Grep::execSTTOR(Signal* signal)
+{
+ jamEntry();
+ const Uint32 startphase = signal->theData[1];
+ const Uint32 typeOfStart = signal->theData[7];
+ if (startphase == 3)
+ {
+ jam();
+ signal->theData[0] = reference();
+ g_TypeOfStart = typeOfStart;
+ sendSignal(NDBCNTR_REF, GSN_READ_NODESREQ, signal, 1, JBB);
+ return;
+ }
+ if(startphase == 5) {
+ jam();
+ /**
+ * we don't want any log/meta records comming to use
+ * until we are done with the recovery.
+ */
+ if (g_TypeOfStart == NodeState::ST_NODE_RESTART) {
+ jam();
+ pspart.m_recoveryMode = true;
+ getNodeGroupMembers(signal);
+ for (Uint32 i = 0; i < c_noNodesInGroup; i++) {
+ Uint32 ref =numberToRef(GREP, c_nodesInGroup[i]);
+ if (ref != reference())
+ sendSignal(ref, GSN_GREP_START_ME, signal,
+ 1 /*SumaStartMe::SignalLength*/, JBB);
+ }
+ } else pspart.m_recoveryMode = false;
+
+ }
+
+ if(startphase == 7) {
+ jam();
+ if (g_TypeOfStart == NodeState::ST_NODE_RESTART) {
+ pspart.m_recoveryMode = false;
+ }
+ }
+
+ sendSTTORRY(signal);
+}
+
+
+void
+Grep::PSPart::execSTART_ME(Signal* signal)
+{
+ jamEntry();
+ GrepStartMe * me =(GrepStartMe*)signal->getDataPtr();
+ BlockReference ref = me->senderRef;
+ GrepAddSubReq* const addReq = (GrepAddSubReq *)signal->getDataPtr();
+
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.first(c_subPtr);
+ for(; !c_subPtr.isNull(); c_subscriptions.next(c_subPtr)) {
+ jam();
+ subPtr.i = c_subPtr.curr.i;
+ subPtr.p = c_subscriptions.getPtr(subPtr.i);
+ addReq->subscriptionId = subPtr.p->m_subscriptionId;
+ addReq->subscriptionKey = subPtr.p->m_subscriptionKey;
+ addReq->subscriberData = subPtr.p->m_subscriberData;
+ addReq->subscriptionType = subPtr.p->m_subscriptionType;
+ addReq->senderRef = subPtr.p->m_coordinatorRef;
+ addReq->subscriberRef =subPtr.p->m_subscriberRef;
+
+ sendSignal(ref,
+ GSN_GREP_ADD_SUB_REQ,
+ signal,
+ GrepAddSubReq::SignalLength,
+ JBB);
+ }
+
+ addReq->subscriptionId = 0;
+ addReq->subscriptionKey = 0;
+ addReq->subscriberData = 0;
+ addReq->subscriptionType = 0;
+ addReq->senderRef = 0;
+ addReq->subscriberRef = 0;
+
+ sendSignal(ref,
+ GSN_GREP_ADD_SUB_REQ,
+ signal,
+ GrepAddSubReq::SignalLength,
+ JBB);
+}
+
+void
+Grep::PSPart::execGREP_ADD_SUB_REQ(Signal* signal)
+{
+ jamEntry();
+ GrepAddSubReq * const grepReq = (GrepAddSubReq *)signal->getDataPtr();
+ const Uint32 subId = grepReq->subscriptionId;
+ const Uint32 subKey = grepReq->subscriptionKey;
+ const Uint32 subData = grepReq->subscriberData;
+ const Uint32 subType = grepReq->subscriptionType;
+ const Uint32 coordinatorRef = grepReq->senderRef;
+
+ /**
+ * this is ref to the REP node for this subscription.
+ */
+ const Uint32 subRef = grepReq->subscriberRef;
+
+ if(subId!=0 && subKey!=0) {
+ jam();
+ SubscriptionPtr subPtr;
+ ndbrequire( c_subscriptionPool.seize(subPtr));
+ subPtr.p->m_coordinatorRef = coordinatorRef;
+ subPtr.p->m_subscriptionId = subId;
+ subPtr.p->m_subscriptionKey = subKey;
+ subPtr.p->m_subscriberRef = subRef;
+ subPtr.p->m_subscriberData = subData;
+ subPtr.p->m_subscriptionType = subType;
+
+ c_subscriptions.add(subPtr);
+ }
+ else {
+ jam();
+ GrepAddSubConf * conf = (GrepAddSubConf *)grepReq;
+ conf->noOfSub =
+ c_subscriptionPool.getSize()-c_subscriptionPool.getNoOfFree();
+ sendSignal(signal->getSendersBlockRef(),
+ GSN_GREP_ADD_SUB_CONF,
+ signal,
+ GrepAddSubConf::SignalLength,
+ JBB);
+ }
+}
+
+void
+Grep::PSPart::execGREP_ADD_SUB_REF(Signal* signal)
+{
+ /**
+ * @todo fix error stuff
+ */
+}
+
+void
+Grep::PSPart::execGREP_ADD_SUB_CONF(Signal* signal)
+{
+ jamEntry();
+ GrepAddSubConf* const conf = (GrepAddSubConf *)signal->getDataPtr();
+ Uint32 noOfSubscriptions = conf->noOfSub;
+ Uint32 noOfRestoredSubscriptions =
+ c_subscriptionPool.getSize()-c_subscriptionPool.getNoOfFree();
+ if(noOfSubscriptions!=noOfRestoredSubscriptions) {
+ jam();
+ /**
+ *@todo send ref signal
+ */
+ ndbrequire(false);
+ }
+}
+
+void
+Grep::execREAD_NODESCONF(Signal* signal)
+{
+ jamEntry();
+ ReadNodesConf * conf = (ReadNodesConf *)signal->getDataPtr();
+
+#if 0
+ ndbout_c("Grep: Recd READ_NODESCONF");
+#endif
+
+ /******************************
+ * Check which REP nodes exist
+ ******************************/
+ for (Uint32 i = 1; i < MAX_NODES; i++)
+ {
+ jam();
+#if 0
+ ndbout_c("Grep: Found node %d of type %d", i, getNodeInfo(i).getType());
+#endif
+ if (getNodeInfo(i).getType() == NodeInfo::REP)
+ {
+ jam();
+ /**
+ * @todo This should work for more than ONE rep node!
+ */
+ pscoord.m_repRef = numberToRef(PSREPBLOCKNO, i);
+ pspart.m_repRef = numberToRef(PSREPBLOCKNO, i);
+#if 0
+ ndbout_c("Grep: REP node %d detected", i);
+#endif
+ }
+ }
+
+ /*****************************
+ * Check which DB nodes exist
+ *****************************/
+ m_aliveNodes.clear();
+
+ Uint32 count = 0;
+ for(Uint32 i = 0; i<MAX_NDB_NODES; i++)
+ {
+ if (NodeBitmask::get(conf->allNodes, i))
+ {
+ jam();
+ count++;
+
+ NodePtr node;
+ ndbrequire(m_nodes.seize(node));
+
+ node.p->nodeId = i;
+ if (NodeBitmask::get(conf->inactiveNodes, i))
+ {
+ node.p->alive = 0;
+ }
+ else
+ {
+ node.p->alive = 1;
+ m_aliveNodes.set(i);
+ }
+ }
+ }
+ m_masterNodeId = conf->masterNodeId;
+ ndbrequire(count == conf->noOfNodes);
+ sendSTTORRY(signal);
+}
+
+void
+Grep::sendSTTORRY(Signal* signal)
+{
+ signal->theData[0] = 0;
+ signal->theData[3] = 1;
+ signal->theData[4] = 3;
+ signal->theData[5] = 5;
+ signal->theData[6] = 7;
+ signal->theData[7] = 255; // No more start phases from missra
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 8, JBB);
+}
+
+void
+Grep::execNDB_STTOR(Signal* signal)
+{
+ jamEntry();
+}
+
+void
+Grep::execDUMP_STATE_ORD(Signal* signal)
+{
+ jamEntry();
+ //Uint32 tCase = signal->theData[0];
+
+#if 0
+ if(sscoord.m_repRef == 0)
+ {
+ ndbout << "Grep: Recd DUMP signal but has no connection with REP node"
+ << endl;
+ return;
+ }
+#endif
+
+ /*
+ switch (tCase)
+ {
+ case 8100: sscoord.grepReq(signal, GrepReq::START_SUBSCR); break;
+ case 8102: sscoord.grepReq(signal, GrepReq::START_METALOG); break;
+ case 8104: sscoord.grepReq(signal, GrepReq::START_METASCAN); break;
+ case 8106: sscoord.grepReq(signal, GrepReq::START_DATALOG); break;
+ case 8108: sscoord.grepReq(signal, GrepReq::START_DATASCAN); break;
+ case 8110: sscoord.grepReq(signal, GrepReq::STOP_SUBSCR); break;
+ case 8500: sscoord.grepReq(signal, GrepReq::REMOVE_BUFFERS); break;
+ case 8300: sscoord.grepReq(signal, GrepReq::SLOWSTOP); break;
+ case 8400: sscoord.grepReq(signal, GrepReq::FASTSTOP); break;
+ case 8600: sscoord.grepReq(signal, GrepReq::CREATE_SUBSCR); break;
+ case 8700: sscoord.dropTable(signal,(Uint32)signal->theData[1]);break;
+ default: break;
+ }
+ */
+}
+
+/**
+ * Signal received when REP node has failed
+ */
+void
+Grep::execAPI_FAILREQ(Signal* signal)
+{
+ jamEntry();
+ //Uint32 failedApiNode = signal->theData[0];
+ //BlockReference retRef = signal->theData[1];
+
+ /**
+ * @todo We should probably do something smart if the
+ * PS REP node fails???? /Lars
+ */
+
+#if 0
+ ndbout_c("Grep: API_FAILREQ received for API node %d.", failedApiNode);
+#endif
+
+ /**
+ * @note This signal received is NOT allowed to send any CONF
+ * signal, since this would screw up TC/DICT to API
+ * "connections".
+ */
+}
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: GREP Control
+ * ------------------------------------------------------------------------
+ **************************************************************************/
+void
+Grep::execGREP_REQ(Signal* signal)
+{
+ jamEntry();
+
+ //GrepReq * req = (GrepReq *)signal->getDataPtr();
+
+ /**
+ * @todo Fix so that request is redirected to REP Server
+ * Obsolete?
+ * Was: sscoord.grepReq(signal, req->request);
+ */
+ ndbout_c("Warning! REP commands can only be executed at REP SERVER prompt!");
+}
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: NODE STATE HANDLING
+ * ------------------------------------------------------------------------
+ **************************************************************************/
+void
+Grep::execNODE_FAILREP(Signal* signal)
+{
+ jamEntry();
+ NodeFailRep * rep = (NodeFailRep*)signal->getDataPtr();
+ bool changed = false;
+
+ NodePtr nodePtr;
+ for(m_nodes.first(nodePtr); nodePtr.i != RNIL; m_nodes.next(nodePtr))
+ {
+ jam();
+ if (NodeBitmask::get(rep->theNodes, nodePtr.p->nodeId))
+ {
+ jam();
+
+ if (nodePtr.p->alive)
+ {
+ jam();
+ ndbassert(m_aliveNodes.get(nodePtr.p->nodeId));
+ changed = true;
+ }
+ else
+ {
+ ndbassert(!m_aliveNodes.get(nodePtr.p->nodeId));
+ }
+
+ nodePtr.p->alive = 0;
+ m_aliveNodes.clear(nodePtr.p->nodeId);
+ }
+ }
+
+
+ /**
+ * Problem: Fix a node failure running a protocol
+ *
+ * 1. Coordinator node of a protocol dies
+ * - Elect a new coordinator
+ * - send ref to user
+ *
+ * 2. Non-coordinator dies.
+ * - make coordinator aware of this
+ * so that coordinator does not wait for
+ * conf from faulty node
+ * - node recovery will restore the non-coordinator.
+ *
+ */
+}
+
+void
+Grep::execINCL_NODEREQ(Signal* signal)
+{
+ jamEntry();
+
+ //const Uint32 senderRef = signal->theData[0];
+ const Uint32 inclNode = signal->theData[1];
+
+ NodePtr node;
+ for(m_nodes.first(node); node.i != RNIL; m_nodes.next(node))
+ {
+ jam();
+ const Uint32 nodeId = node.p->nodeId;
+ if (inclNode == nodeId) {
+ jam();
+
+ ndbrequire(node.p->alive == 0);
+ ndbassert(!m_aliveNodes.get(nodeId));
+
+ node.p->alive = 1;
+ m_aliveNodes.set(nodeId);
+
+ break;
+ }
+ }
+
+ /**
+ * @todo: if we include this DIH's got to be prepared, later if needed...
+ */
+#if 0
+ signal->theData[0] = reference();
+
+ sendSignal(senderRef, GSN_INCL_NODECONF, signal, 1, JBB);
+#endif
+}
+
+
+/**
+ * Helper methods
+ */
+void
+Grep::PSCoord::prepareOperationRec(SubCoordinatorPtr subPtr,
+ BlockReference subscriber,
+ Uint32 subId,
+ Uint32 subKey,
+ Uint32 request)
+{
+ subPtr.p->m_coordinatorRef = reference();
+ subPtr.p->m_subscriberRef = subscriber;
+ subPtr.p->m_subscriberData = subPtr.i;
+ subPtr.p->m_subscriptionId = subId;
+ subPtr.p->m_subscriptionKey = subKey;
+ subPtr.p->m_outstandingRequest = request;
+}
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: CREATE SUBSCRIPTION ID
+ * ------------------------------------------------------------------------
+ *
+ * Requests SUMA to create a unique subscription id
+ **************************************************************************/
+
+void
+Grep::PSCoord::execGREP_CREATE_SUBID_REQ(Signal* signal)
+{
+ jamEntry();
+
+ CreateSubscriptionIdReq * req =
+ (CreateSubscriptionIdReq*)signal->getDataPtr();
+ BlockReference ref = signal->getSendersBlockRef();
+
+ SubCoordinatorPtr subPtr;
+ if( !c_subCoordinatorPool.seize(subPtr)) {
+ jam();
+ SubCoordinator sub;
+ sub.m_subscriberRef = ref;
+ sub.m_subscriptionId = 0;
+ sub.m_subscriptionKey = 0;
+ sendRefToSS(signal, sub, GrepError::SUBSCRIPTION_ID_NOMEM );
+ return;
+ }
+ prepareOperationRec(subPtr,
+ ref,
+ 0,0,
+ GSN_CREATE_SUBID_REQ);
+
+
+ ndbout_c("SUBID_REQ Ref %d",ref);
+ req->senderData=subPtr.p->m_subscriberData;
+
+ sendSignal(SUMA_REF, GSN_CREATE_SUBID_REQ, signal,
+ SubCreateReq::SignalLength, JBB);
+
+#if 1 //def DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSCoord: Sent CREATE_SUBID_REQ to SUMA");
+#endif
+}
+
+void
+Grep::PSCoord::execCREATE_SUBID_CONF(Signal* signal)
+{
+ jamEntry();
+ CreateSubscriptionIdConf const * conf =
+ (CreateSubscriptionIdConf *)signal->getDataPtr();
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ Uint32 subData = conf->subscriberData;
+
+#if 1 //def DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSCoord: Recd GREP_SUBID_CONF (subId:%d, subKey:%d)",
+ subId, subKey);
+#endif
+
+ SubCoordinatorPtr subPtr;
+ c_subCoordinatorPool.getPtr(subPtr, subData);
+ BlockReference repRef = subPtr.p->m_subscriberRef;
+
+ { // Check that id/key is unique
+ SubCoordinator key;
+ SubCoordinatorPtr tmp;
+ key.m_subscriptionId = subId;
+ key.m_subscriptionKey = subKey;
+ if(c_runningSubscriptions.find(tmp, key)){
+ jam();
+ SubCoordinator sub;
+ sub.m_subscriberRef=repRef;
+ sub.m_subscriptionId = subId;
+ sub.m_subscriptionKey = subKey;
+ sendRefToSS(signal,sub, GrepError::SUBSCRIPTION_ID_NOT_UNIQUE );
+ return;
+ }
+ }
+
+ sendSignal(subPtr.p->m_subscriberRef, GSN_GREP_CREATE_SUBID_CONF, signal,
+ CreateSubscriptionIdConf::SignalLength, JBB);
+ c_subCoordinatorPool.release(subData);
+
+ m_grep->sendEventRep(signal,
+ EventReport::GrepSubscriptionInfo,
+ GrepEvent::GrepPS_CreateSubIdConf,
+ subId,
+ subKey,
+ (Uint32)GrepError::NO_ERROR);
+}
+
+void
+Grep::PSCoord::execCREATE_SUBID_REF(Signal* signal) {
+ jamEntry();
+ CreateSubscriptionIdRef const * ref =
+ (CreateSubscriptionIdRef *)signal->getDataPtr();
+ Uint32 subData = ref->subscriberData;
+ GrepError::Code err;
+
+ Uint32 sendersBlockRef = signal->getSendersBlockRef();
+ if(sendersBlockRef == SUMA_REF)
+ {
+ jam();
+ err = GrepError::SUBSCRIPTION_ID_SUMA_FAILED_CREATE;
+ }
+
+ SubCoordinatorPtr subPtr;
+ c_runningSubscriptions.getPtr(subPtr, subData);
+ BlockReference repref = subPtr.p->m_subscriberRef;
+
+ SubCoordinator sub;
+ sub.m_subscriberRef = repref;
+ sub.m_subscriptionId = 0;
+ sub.m_subscriptionKey = 0;
+ sendRefToSS(signal,sub, err);
+
+}
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: CREATE SUBSCRIPTION
+ * ------------------------------------------------------------------------
+ *
+ * Creates a subscription for every GREP to its local SUMA.
+ * GREP node that executes createSubscription becomes the GREP Coord.
+ **************************************************************************/
+
+/**
+ * Request to create a subscription (sent from SS)
+ */
+void
+Grep::PSCoord::execGREP_SUB_CREATE_REQ(Signal* signal)
+{
+ jamEntry();
+ GrepSubCreateReq const * grepReq = (GrepSubCreateReq *)signal->getDataPtr();
+ Uint32 subId = grepReq->subscriptionId;
+ Uint32 subKey = grepReq->subscriptionKey;
+ Uint32 subType = grepReq->subscriptionType;
+ BlockReference rep = signal->getSendersBlockRef();
+
+ GrepCreateReq * req =(GrepCreateReq*)grepReq;
+
+ SubCoordinatorPtr subPtr;
+
+ if( !c_subCoordinatorPool.seize(subPtr)) {
+ jam();
+ SubCoordinator sub;
+ sub.m_subscriberRef = rep;
+ sub.m_subscriptionId = 0;
+ sub.m_subscriptionKey = 0;
+ sub.m_outstandingRequest = GSN_GREP_CREATE_REQ;
+ sendRefToSS(signal, sub, GrepError::NOSPACE_IN_POOL);
+ return;
+ }
+ prepareOperationRec(subPtr,
+ numberToRef(PSREPBLOCKNO, refToNode(rep)), subId, subKey,
+ GSN_GREP_CREATE_REQ);
+
+ /* Get the payload of the signal.
+ */
+ SegmentedSectionPtr selectedTablesPtr;
+ if(subType == SubCreateReq::SelectiveTableSnapshot) {
+ jam();
+ ndbrequire(signal->getNoOfSections()==1);
+ signal->getSection(selectedTablesPtr,0);
+ signal->header.m_noOfSections = 0;
+ }
+ /**
+ * Prepare the signal to be sent to Grep participatns
+ */
+ subPtr.p->m_subscriptionType = subType;
+ req->senderRef = reference();
+ req->subscriberRef = numberToRef(PSREPBLOCKNO, refToNode(rep));
+ req->subscriberData = subPtr.p->m_subscriberData;
+ req->subscriptionId = subId;
+ req->subscriptionKey = subKey;
+ req->subscriptionType = subType;
+
+ /*add payload if it is a selectivetablesnap*/
+ if(subType == SubCreateReq::SelectiveTableSnapshot) {
+ jam();
+ signal->setSection(selectedTablesPtr, 0);
+ }
+
+ /******************************
+ * Send to all PS participants
+ ******************************/
+ NodeReceiverGroup rg(GREP, m_grep->m_aliveNodes);
+ subPtr.p->m_outstandingParticipants = rg;
+ sendSignal(rg,
+ GSN_GREP_CREATE_REQ, signal,
+ GrepCreateReq::SignalLength, JBB);
+
+
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSCoord: Sent GREP_CREATE_REQ "
+ "(subId:%d, subKey:%d, subData:%d, subType:%d) to parts",
+ subId, subKey, subPtr.p->m_subscriberData, subType);
+#endif
+}
+
+void
+Grep::PSPart::execGREP_CREATE_REQ(Signal* signal)
+{
+ jamEntry();
+ GrepCreateReq * const grepReq = (GrepCreateReq *)signal->getDataPtr();
+ const Uint32 subId = grepReq->subscriptionId;
+ const Uint32 subKey = grepReq->subscriptionKey;
+ const Uint32 subData = grepReq->subscriberData;
+ const Uint32 subType = grepReq->subscriptionType;
+ const Uint32 coordinatorRef = grepReq->senderRef;
+ const Uint32 subRef = grepReq->subscriberRef; //this is ref to the
+ //REP node for this
+ //subscription.
+
+ SubscriptionPtr subPtr;
+ ndbrequire( c_subscriptionPool.seize(subPtr));
+ subPtr.p->m_coordinatorRef = coordinatorRef;
+ subPtr.p->m_subscriptionId = subId;
+ subPtr.p->m_subscriptionKey = subKey;
+ subPtr.p->m_subscriberRef = subRef;
+ subPtr.p->m_subscriberData = subPtr.i;
+ subPtr.p->m_subscriptionType = subType;
+ subPtr.p->m_outstandingRequest = GSN_GREP_CREATE_REQ;
+ subPtr.p->m_operationPtrI = subData;
+
+ c_subscriptions.add(subPtr);
+
+ SegmentedSectionPtr selectedTablesPtr;
+ if(subType == SubCreateReq::SelectiveTableSnapshot) {
+ jam();
+ ndbrequire(signal->getNoOfSections()==1);
+ signal->getSection(selectedTablesPtr,0);// SubCreateReq::TABLE_LIST);
+ signal->header.m_noOfSections = 0;
+ }
+
+ /**
+ * Prepare signal to be sent to SUMA
+ */
+ SubCreateReq * sumaReq = (SubCreateReq *)grepReq;
+ sumaReq->subscriberRef = GREP_REF;
+ sumaReq->subscriberData = subPtr.p->m_subscriberData;
+ sumaReq->subscriptionId = subPtr.p->m_subscriptionId;
+ sumaReq->subscriptionKey = subPtr.p->m_subscriptionKey;
+ sumaReq->subscriptionType = subPtr.p->m_subscriptionType;
+ /*add payload if it is a selectivetablesnap*/
+ if(subType == SubCreateReq::SelectiveTableSnapshot) {
+ jam();
+ signal->setSection(selectedTablesPtr, 0);
+ }
+ sendSignal(SUMA_REF,
+ GSN_SUB_CREATE_REQ,
+ signal,
+ SubCreateReq::SignalLength,
+ JBB);
+}
+
+void
+Grep::PSPart::execSUB_CREATE_CONF(Signal* signal)
+{
+ jamEntry();
+
+ SubCreateConf * const conf = (SubCreateConf *)signal->getDataPtr();
+ Uint32 subData = conf->subscriberData;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+ /**
+ @todo check why this can fuck up -johan
+
+ ndbrequire(subPtr.p->m_subscriptionId == conf->subscriptionId);
+ ndbrequire(subPtr.p->m_subscriptionKey == conf->subscriptionKey);
+ */
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSPart: Recd SUB_CREATE_CONF "
+ "(subId:%d, subKey:%d) from SUMA",
+ conf->subscriptionId, conf->subscriptionKey);
+#endif
+
+ /*********************
+ * Send conf to coord
+ *********************/
+ GrepCreateConf * grepConf = (GrepCreateConf*)conf;
+ grepConf->senderNodeId = getOwnNodeId();
+ grepConf->senderData = subPtr.p->m_operationPtrI;
+ sendSignal(subPtr.p->m_coordinatorRef, GSN_GREP_CREATE_CONF, signal,
+ GrepCreateConf::SignalLength, JBB);
+ subPtr.p->m_outstandingRequest = 0;
+}
+
+/**
+ * Handle errors that either occured in:
+ * 1) PSPart
+ * or
+ * 2) propagated from local SUMA
+ */
+void
+Grep::PSPart::execSUB_CREATE_REF(Signal* signal)
+{
+ jamEntry();
+ SubCreateRef * const ref = (SubCreateRef *)signal->getDataPtr();
+ Uint32 subData = ref->subscriberData;
+ GrepError::Code err = (GrepError::Code)ref->err;
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+ sendRefToPSCoord(signal, *subPtr.p, err /*error*/);
+ subPtr.p->m_outstandingRequest = 0;
+}
+
+void
+Grep::PSCoord::execGREP_CREATE_CONF(Signal* signal)
+{
+ jamEntry();
+ GrepCreateConf const * conf = (GrepCreateConf *)signal->getDataPtr();
+ Uint32 subData = conf->senderData;
+ Uint32 nodeId = conf->senderNodeId;
+
+ SubCoordinatorPtr subPtr;
+ c_subCoordinatorPool.getPtr(subPtr, subData);
+
+ ndbrequire(subPtr.p->m_outstandingRequest == GSN_GREP_CREATE_REQ);
+
+ subPtr.p->m_outstandingParticipants.clearWaitingFor(nodeId);
+
+ if(!subPtr.p->m_outstandingParticipants.done()) return;
+ /********************************
+ * All participants have CONF:ed
+ ********************************/
+ Uint32 subId = subPtr.p->m_subscriptionId;
+ Uint32 subKey = subPtr.p->m_subscriptionKey;
+
+ GrepSubCreateConf * grepConf = (GrepSubCreateConf *)signal->getDataPtr();
+ grepConf->subscriptionId = subId;
+ grepConf->subscriptionKey = subKey;
+ sendSignal(subPtr.p->m_subscriberRef, GSN_GREP_SUB_CREATE_CONF, signal,
+ GrepSubCreateConf::SignalLength, JBB);
+
+ /**
+ * Send event report
+ */
+ m_grep->sendEventRep(signal,
+ EventReport::GrepSubscriptionInfo,
+ GrepEvent::GrepPS_SubCreateConf,
+ subId,
+ subKey,
+ (Uint32)GrepError::NO_ERROR);
+
+ c_subCoordinatorPool.release(subPtr);
+
+}
+
+/**
+ * Handle errors that either occured in:
+ * 1) PSCoord
+ * or
+ * 2) propagated from PSPart
+ */
+void
+Grep::PSCoord::execGREP_CREATE_REF(Signal* signal)
+{
+ jamEntry();
+ GrepCreateRef * const ref = (GrepCreateRef *)signal->getDataPtr();
+ Uint32 subData = ref->senderData;
+ Uint32 err = ref->err;
+ SubCoordinatorPtr subPtr;
+ c_runningSubscriptions.getPtr(subPtr, subData);
+
+ sendRefToSS(signal, *subPtr.p, (GrepError::Code)err /*error*/);
+}
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: START SUBSCRIPTION
+ * ------------------------------------------------------------------------
+ *
+ * Starts a subscription at SUMA.
+ * Each participant starts its own subscription.
+ **************************************************************************/
+
+/**
+ * Request to start subscription (Sent from SS)
+ */
+void
+Grep::PSCoord::execGREP_SUB_START_REQ(Signal* signal)
+{
+ jamEntry();
+ GrepSubStartReq * const subReq = (GrepSubStartReq *)signal->getDataPtr();
+ SubscriptionData::Part part = (SubscriptionData::Part) subReq->part;
+ Uint32 subId = subReq->subscriptionId;
+ Uint32 subKey = subReq->subscriptionKey;
+ BlockReference rep = signal->getSendersBlockRef();
+
+ SubCoordinatorPtr subPtr;
+
+ if(!c_subCoordinatorPool.seize(subPtr)) {
+ jam();
+ SubCoordinator sub;
+ sub.m_subscriberRef = rep;
+ sub.m_subscriptionId = 0;
+ sub.m_subscriptionKey = 0;
+ sub.m_outstandingRequest = GSN_GREP_START_REQ;
+ sendRefToSS(signal, sub, GrepError::NOSPACE_IN_POOL);
+ return;
+ }
+
+ prepareOperationRec(subPtr,
+ numberToRef(PSREPBLOCKNO, refToNode(rep)),
+ subId, subKey,
+ GSN_GREP_START_REQ);
+
+ GrepStartReq * const req = (GrepStartReq *) subReq;
+ req->part = (Uint32) part;
+ req->subscriptionId = subPtr.p->m_subscriptionId;
+ req->subscriptionKey = subPtr.p->m_subscriptionKey;
+ req->senderData = subPtr.p->m_subscriberData;
+
+ /***************************
+ * Send to all participants
+ ***************************/
+ NodeReceiverGroup rg(GREP, m_grep->m_aliveNodes);
+ subPtr.p->m_outstandingParticipants = rg;
+ sendSignal(rg,
+ GSN_GREP_START_REQ,
+ signal,
+ GrepStartReq::SignalLength, JBB);
+
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSCoord: Sent GREP_START_REQ "
+ "(subId:%d, subKey:%d, senderData:%d, part:%d) to all participants",
+ req->subscriptionId, req->subscriptionKey, req->senderData, part);
+#endif
+}
+
+
+void
+Grep::PSPart::execGREP_START_REQ(Signal* signal)
+{
+ jamEntry();
+ GrepStartReq * const grepReq = (GrepStartReq *) signal->getDataPtr();
+ SubscriptionData::Part part = (SubscriptionData::Part)grepReq->part;
+ Uint32 subId = grepReq->subscriptionId;
+ Uint32 subKey = grepReq->subscriptionKey;
+ Uint32 operationPtrI = grepReq->senderData;
+
+ Subscription key;
+ key.m_subscriptionId = subId;
+ key.m_subscriptionKey = subKey;
+ SubscriptionPtr subPtr;
+ ndbrequire(c_subscriptions.find(subPtr, key));;
+ subPtr.p->m_outstandingRequest = GSN_GREP_START_REQ;
+ subPtr.p->m_operationPtrI = operationPtrI;
+ /**
+ * send SUB_START_REQ to local SUMA
+ */
+ SubStartReq * sumaReq = (SubStartReq *) grepReq;
+ sumaReq->subscriptionId = subId;
+ sumaReq->subscriptionKey = subKey;
+ sumaReq->subscriberData = subPtr.i;
+ sumaReq->part = (Uint32) part;
+
+ sendSignal(SUMA_REF, GSN_SUB_START_REQ, signal,
+ SubStartReq::SignalLength, JBB);
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSPart: Sent SUB_START_REQ (subId:%d, subKey:%d, part:%d)",
+ subId, subKey, (Uint32)part);
+#endif
+}
+
+
+void
+Grep::PSPart::execSUB_START_CONF(Signal* signal)
+{
+ jamEntry();
+
+ SubStartConf * const conf = (SubStartConf *) signal->getDataPtr();
+ SubscriptionData::Part part = (SubscriptionData::Part)conf->part;
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ Uint32 subData = conf->subscriberData;
+ Uint32 firstGCI = conf->firstGCI;
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSPart: Recd SUB_START_CONF "
+ "(subId:%d, subKey:%d, subData:%d)",
+ subId, subKey, subData);
+#endif
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+ ndbrequire(subPtr.p->m_subscriptionId == subId);
+ ndbrequire(subPtr.p->m_subscriptionKey == subKey);
+
+ GrepStartConf * grepConf = (GrepStartConf *)conf;
+ grepConf->senderData = subPtr.p->m_operationPtrI;
+ grepConf->part = (Uint32) part;
+ grepConf->subscriptionKey = subKey;
+ grepConf->subscriptionId = subId;
+ grepConf->firstGCI = firstGCI;
+ grepConf->senderNodeId = getOwnNodeId();
+ sendSignal(subPtr.p->m_coordinatorRef, GSN_GREP_START_CONF, signal,
+ GrepStartConf::SignalLength, JBB);
+ subPtr.p->m_outstandingRequest = 0;
+
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSPart: Sent GREP_START_CONF "
+ "(subId:%d, subKey:%d, subData:%d, part:%d)",
+ subId, subKey, subData, part);
+#endif
+}
+
+
+/**
+ * Handle errors that either occured in:
+ * 1) PSPart
+ * or
+ * 2) propagated from local SUMA
+ *
+ * Propagates REF signal to PSCoord
+ */
+void
+Grep::PSPart::execSUB_START_REF(Signal* signal)
+{
+ SubStartRef * const ref = (SubStartRef *)signal->getDataPtr();
+ Uint32 subData = ref->subscriberData;
+ GrepError::Code err = (GrepError::Code)ref->err;
+ SubscriptionData::Part part = (SubscriptionData::Part)ref->part;
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+ sendRefToPSCoord(signal, *subPtr.p, err /*error*/, part);
+ subPtr.p->m_outstandingRequest = 0;
+}
+
+
+/**
+ * Logging has started... (says PS Participant)
+ */
+void
+Grep::PSCoord::execGREP_START_CONF(Signal* signal)
+{
+ jamEntry();
+
+ GrepStartConf * const conf = (GrepStartConf *) signal->getDataPtr();
+ Uint32 subData = conf->senderData;
+ SubscriptionData::Part part = (SubscriptionData::Part)conf->part;
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ Uint32 firstGCI = conf->firstGCI;
+
+ SubCoordinatorPtr subPtr;
+ c_subCoordinatorPool.getPtr(subPtr, subData);
+ ndbrequire(subPtr.p->m_outstandingRequest == GSN_GREP_START_REQ);
+
+ subPtr.p->m_outstandingParticipants.clearWaitingFor(conf->senderNodeId);
+
+ if(!subPtr.p->m_outstandingParticipants.done()) return;
+ jam();
+
+ /*************************
+ * All participants ready
+ *************************/
+ GrepSubStartConf * grepConf = (GrepSubStartConf *) conf;
+ grepConf->part = part;
+ grepConf->subscriptionId = subId;
+ grepConf->subscriptionKey = subKey;
+ grepConf->firstGCI = firstGCI;
+
+ bool ok = false;
+ switch(part) {
+ case SubscriptionData::MetaData:
+ ok = true;
+ sendSignal(subPtr.p->m_subscriberRef, GSN_GREP_SUB_START_CONF, signal,
+ GrepSubStartConf::SignalLength, JBB);
+
+ /**
+ * Send event report
+ */
+ m_grep->sendEventRep(signal,
+ EventReport::GrepSubscriptionInfo,
+ GrepEvent::GrepPS_SubStartMetaConf,
+ subId, subKey,
+ (Uint32)GrepError::NO_ERROR);
+
+ c_subCoordinatorPool.release(subPtr);
+ break;
+ case SubscriptionData::TableData:
+ ok = true;
+ sendSignal(subPtr.p->m_subscriberRef, GSN_GREP_SUB_START_CONF, signal,
+ GrepSubStartConf::SignalLength, JBB);
+
+ /**
+ * Send event report
+ */
+ m_grep->sendEventRep(signal,
+ EventReport::GrepSubscriptionInfo,
+ GrepEvent::GrepPS_SubStartDataConf,
+ subId, subKey,
+ (Uint32)GrepError::NO_ERROR);
+
+
+ c_subCoordinatorPool.release(subPtr);
+ break;
+ }
+ ndbrequire(ok);
+
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSCoord: Recd SUB_START_CONF (subId:%d, subKey:%d, part:%d) "
+ "from all slaves",
+ subId, subKey, (Uint32)part);
+#endif
+}
+
+/**
+ * Handle errors that either occured in:
+ * 1) PSCoord
+ * or
+ * 2) propagated from PSPart
+ */
+void
+Grep::PSCoord::execGREP_START_REF(Signal* signal)
+{
+ jamEntry();
+ GrepStartRef * const ref = (GrepStartRef *)signal->getDataPtr();
+ Uint32 subData = ref->senderData;
+ GrepError::Code err = (GrepError::Code)ref->err;
+ SubscriptionData::Part part = (SubscriptionData::Part)ref->part;
+
+ SubCoordinatorPtr subPtr;
+ c_runningSubscriptions.getPtr(subPtr, subData);
+ sendRefToSS(signal, *subPtr.p, err /*error*/, part);
+}
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: REMOVE SUBSCRIPTION
+ * ------------------------------------------------------------------------
+ *
+ * Remove a subscription at SUMA.
+ * Each participant removes its own subscription.
+ * We start by deleting the subscription inside the requestor
+ * since, we don't know if nodes (REP nodes or DB nodes)
+ * have disconnected after we sent out this and
+ * if we dont delete the sub in the requestor now,
+ * we won't be able to create a new subscription
+ **************************************************************************/
+
+/**
+ * Request to abort subscription (Sent from SS)
+ */
+void
+Grep::PSCoord::execGREP_SUB_REMOVE_REQ(Signal* signal)
+{
+ jamEntry();
+ GrepSubRemoveReq * const subReq = (GrepSubRemoveReq *)signal->getDataPtr();
+ Uint32 subId = subReq->subscriptionId;
+ Uint32 subKey = subReq->subscriptionKey;
+ BlockReference rep = signal->getSendersBlockRef();
+
+ SubCoordinatorPtr subPtr;
+ if( !c_subCoordinatorPool.seize(subPtr)) {
+ jam();
+ SubCoordinator sub;
+ sub.m_subscriberRef = rep;
+ sub.m_subscriptionId = 0;
+ sub.m_subscriptionKey = 0;
+ sub.m_outstandingRequest = GSN_GREP_REMOVE_REQ;
+ sendRefToSS(signal, sub, GrepError::NOSPACE_IN_POOL);
+ return;
+ }
+
+
+ prepareOperationRec(subPtr,
+ numberToRef(PSREPBLOCKNO, refToNode(rep)),
+ subId, subKey,
+ GSN_GREP_REMOVE_REQ);
+
+ c_runningSubscriptions.add(subPtr);
+
+ GrepRemoveReq * req = (GrepRemoveReq *) subReq;
+ req->subscriptionId = subPtr.p->m_subscriptionId;
+ req->subscriptionKey = subPtr.p->m_subscriptionKey;
+ req->senderData = subPtr.p->m_subscriberData;
+ req->senderRef = subPtr.p->m_coordinatorRef;
+
+ /***************************
+ * Send to all participants
+ ***************************/
+ NodeReceiverGroup rg(GREP, m_grep->m_aliveNodes);
+ subPtr.p->m_outstandingParticipants = rg;
+ sendSignal(rg,
+ GSN_GREP_REMOVE_REQ, signal,
+ GrepRemoveReq::SignalLength, JBB);
+}
+
+
+void
+Grep::PSPart::execGREP_REMOVE_REQ(Signal* signal)
+{
+ jamEntry();
+ GrepRemoveReq * const grepReq = (GrepRemoveReq *) signal->getDataPtr();
+ Uint32 subId = grepReq->subscriptionId;
+ Uint32 subKey = grepReq->subscriptionKey;
+ Uint32 subData = grepReq->senderData;
+ Uint32 coordinator = grepReq->senderRef;
+
+ Subscription key;
+ key.m_subscriptionId = subId;
+ key.m_subscriptionKey = subKey;
+ SubscriptionPtr subPtr;
+
+ if(!c_subscriptions.find(subPtr, key))
+ {
+ /**
+ * The subscription was not found, so it must be deleted.
+ * Send CONF back, since it does not exist (thus, it is removed)
+ */
+ GrepRemoveConf * grepConf = (GrepRemoveConf *)grepReq;
+ grepConf->subscriptionKey = subKey;
+ grepConf->subscriptionId = subId;
+ grepConf->senderData = subData;
+ grepConf->senderNodeId = getOwnNodeId();
+ sendSignal(coordinator, GSN_GREP_REMOVE_CONF, signal,
+ GrepRemoveConf::SignalLength, JBB);
+ return;
+ }
+
+ subPtr.p->m_operationPtrI = subData;
+ subPtr.p->m_coordinatorRef = coordinator;
+ subPtr.p->m_outstandingRequest = GSN_GREP_REMOVE_REQ;
+
+ /**
+ * send SUB_REMOVE_REQ to local SUMA
+ */
+ SubRemoveReq * sumaReq = (SubRemoveReq *) grepReq;
+ sumaReq->subscriptionId = subId;
+ sumaReq->subscriptionKey = subKey;
+ sumaReq->senderData = subPtr.i;
+ sendSignal(SUMA_REF, GSN_SUB_REMOVE_REQ, signal,
+ SubStartReq::SignalLength, JBB);
+}
+
+
+/**
+ * SUB_REMOVE_CONF (from local SUMA)
+ */
+void
+Grep::PSPart::execSUB_REMOVE_CONF(Signal* signal)
+{
+ jamEntry();
+ SubRemoveConf * const conf = (SubRemoveConf *) signal->getDataPtr();
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ Uint32 subData = conf->subscriberData;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+ ndbrequire(subPtr.p->m_subscriptionId == subId);
+ ndbrequire(subPtr.p->m_subscriptionKey == subKey);
+ subPtr.p->m_outstandingRequest = 0;
+ GrepRemoveConf * grepConf = (GrepRemoveConf *)conf;
+ grepConf->subscriptionKey = subKey;
+ grepConf->subscriptionId = subId;
+ grepConf->senderData = subPtr.p->m_operationPtrI;
+ grepConf->senderNodeId = getOwnNodeId();
+ sendSignal(subPtr.p->m_coordinatorRef, GSN_GREP_REMOVE_CONF, signal,
+ GrepRemoveConf::SignalLength, JBB);
+ c_subscriptions.release(subPtr);
+
+}
+
+
+/**
+ * SUB_REMOVE_CONF (from local SUMA)
+ */
+void
+Grep::PSPart::execSUB_REMOVE_REF(Signal* signal)
+{
+ jamEntry();
+ SubRemoveRef * const ref = (SubRemoveRef *)signal->getDataPtr();
+ Uint32 subData = ref->subscriberData;
+ GrepError::Code err = (GrepError::Code)ref->err;
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+
+ //sendSubRemoveRef_PSCoord(signal, *subPtr.p, err /*error*/);
+}
+
+
+/**
+ * Aborting has been carried out (says Participants)
+ */
+void
+Grep::PSCoord::execGREP_REMOVE_CONF(Signal* signal)
+{
+ jamEntry();
+ GrepRemoveConf * const conf = (GrepRemoveConf *) signal->getDataPtr();
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ Uint32 senderNodeId = conf->senderNodeId;
+ Uint32 subData = conf->senderData;
+ SubCoordinatorPtr subPtr;
+ c_subCoordinatorPool.getPtr(subPtr, subData);
+
+ ndbrequire(subPtr.p->m_outstandingRequest == GSN_GREP_REMOVE_REQ);
+
+ subPtr.p->m_outstandingParticipants.clearWaitingFor(senderNodeId);
+
+ if(!subPtr.p->m_outstandingParticipants.done()) {
+ jam();
+ return;
+ }
+ jam();
+
+ /*************************
+ * All participants ready
+ *************************/
+
+ m_grep->sendEventRep(signal,
+ EventReport::GrepSubscriptionInfo,
+ GrepEvent::GrepPS_SubRemoveConf,
+ subId, subKey,
+ GrepError::NO_ERROR);
+
+ GrepSubRemoveConf * grepConf = (GrepSubRemoveConf *) conf;
+ grepConf->subscriptionId = subId;
+ grepConf->subscriptionKey = subKey;
+ sendSignal(subPtr.p->m_subscriberRef, GSN_GREP_SUB_REMOVE_CONF, signal,
+ GrepSubRemoveConf::SignalLength, JBB);
+
+ c_subCoordinatorPool.release(subPtr);
+}
+
+
+
+void
+Grep::PSCoord::execGREP_REMOVE_REF(Signal* signal)
+{
+ jamEntry();
+ GrepRemoveRef * const ref = (GrepRemoveRef *)signal->getDataPtr();
+ Uint32 subData = ref->senderData;
+ Uint32 err = ref->err;
+ SubCoordinatorPtr subPtr;
+
+ /**
+ * Get the operationrecord matching subdata and remove it. Subsequent
+ * execGREP_REMOVE_REF will simply be ignored at this stage.
+ */
+ for( c_runningSubscriptions.first(c_subPtr);
+ !c_subPtr.isNull(); c_runningSubscriptions.next(c_subPtr)) {
+ jam();
+ subPtr.i = c_subPtr.curr.i;
+ subPtr.p = c_runningSubscriptions.getPtr(subPtr.i);
+ if(subData == subPtr.i)
+ {
+ sendRefToSS(signal, *subPtr.p, (GrepError::Code)err /*error*/);
+ c_runningSubscriptions.release(subPtr);
+ return;
+ }
+ }
+ return;
+}
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: LOG RECORDS (COMING IN FROM LOCAL SUMA)
+ * ------------------------------------------------------------------------
+ *
+ * After the subscription is started, we get log records from SUMA.
+ * Both table data and meta data log records are received.
+ *
+ * TODO:
+ * @todo Changes in meta data is currently not
+ * allowed during global replication
+ **************************************************************************/
+
+void
+Grep::PSPart::execSUB_META_DATA(Signal* signal)
+{
+ jamEntry();
+ if(m_recoveryMode) {
+ jam();
+ return;
+ }
+ /**
+ * METASCAN and METALOG
+ */
+ SubMetaData * data = (SubMetaData *) signal->getDataPtrSend();
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, data->subscriberData);
+
+ /***************************
+ * Forward data to REP node
+ ***************************/
+ sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_META_DATA, signal,
+ SubMetaData::SignalLength, JBB);
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Grep::PSPart: Sent SUB_META_DATA to REP "
+ "(TableId: %d, SenderData: %d, GCI: %d)",
+ data->tableId, data->senderData, data->gci);
+#endif
+}
+
+/**
+ * Receive table data from SUMA and dispatches it to REP node.
+ */
+void
+Grep::PSPart::execSUB_TABLE_DATA(Signal* signal)
+{
+ jamEntry();
+ if(m_recoveryMode) {
+ jam();
+ return;
+ }
+ ndbrequire(m_repRef!=0);
+
+ if(!assembleFragments(signal)) { jam(); return; }
+
+ /**
+ * Check if it is SCAN or LOG data that has arrived
+ */
+ if(signal->getNoOfSections() == 2)
+ {
+ jam();
+ /**
+ * DATASCAN - Not marked with GCI, so mark with latest seen GCI
+ */
+ if(m_firstScanGCI == 1 && m_lastScanGCI == 0) {
+ m_firstScanGCI = m_latestSeenGCI;
+ m_lastScanGCI = m_latestSeenGCI;
+ }
+ SubTableData * data = (SubTableData*)signal->getDataPtrSend();
+ Uint32 subData = data->senderData;
+ data->gci = m_latestSeenGCI;
+ data->logType = SubTableData::SCAN;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+ sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_TABLE_DATA, signal,
+ SubTableData::SignalLength, JBB);
+#ifdef DEBUG_GREP
+ ndbout_c("Grep::PSPart: Sent SUB_TABLE_DATA (Scan, GCI: %d)",
+ data->gci);
+#endif
+ }
+ else
+ {
+ jam();
+ /**
+ * DATALOG (TRIGGER) - Already marked with GCI
+ */
+ SubTableData * data = (SubTableData*)signal->getDataPtrSend();
+ data->logType = SubTableData::LOG;
+ Uint32 subData = data->senderData;
+ if (data->gci > m_latestSeenGCI) m_latestSeenGCI = data->gci;
+
+ // Reformat to sections and send to replication node.
+ LinearSectionPtr ptr[3];
+ ptr[0].p = signal->theData + 25;
+ ptr[0].sz = data->noOfAttributes;
+ ptr[1].p = signal->theData + 25 + MAX_ATTRIBUTES_IN_TABLE;
+ ptr[1].sz = data->dataSize;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+ sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_TABLE_DATA,
+ signal, SubTableData::SignalLength, JBB, ptr, 2);
+#ifdef DEBUG_GREP
+ ndbout_c("Grep::PSPart: Sent SUB_TABLE_DATA (Log, GCI: %d)",
+ data->gci);
+#endif
+ }
+}
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: START SYNCHRONIZATION
+ * ------------------------------------------------------------------------
+ *
+ *
+ **************************************************************************/
+
+/**
+ * Request to start sync (from Rep SS)
+ */
+void
+Grep::PSCoord::execGREP_SUB_SYNC_REQ(Signal* signal)
+{
+ jamEntry();
+ GrepSubSyncReq * const subReq = (GrepSubSyncReq*)signal->getDataPtr();
+ SubscriptionData::Part part = (SubscriptionData::Part) subReq->part;
+ Uint32 subId = subReq->subscriptionId;
+ Uint32 subKey = subReq->subscriptionKey;
+ BlockReference rep = signal->getSendersBlockRef();
+
+ SubCoordinatorPtr subPtr;
+ if( !c_subCoordinatorPool.seize(subPtr)) {
+ jam();
+ SubCoordinator sub;
+ sub.m_subscriberRef = rep;
+ sub.m_subscriptionId = 0;
+ sub.m_subscriptionKey = 0;
+ sub.m_outstandingRequest = GSN_GREP_SYNC_REQ;
+ sendRefToSS(signal, sub, GrepError::NOSPACE_IN_POOL);
+ return;
+ }
+
+ prepareOperationRec(subPtr,
+ numberToRef(PSREPBLOCKNO, refToNode(rep)),
+ subId, subKey,
+ GSN_GREP_SYNC_REQ);
+
+ GrepSyncReq * req = (GrepSyncReq *)subReq;
+ req->subscriptionId = subPtr.p->m_subscriptionId;
+ req->subscriptionKey = subPtr.p->m_subscriptionKey;
+ req->senderData = subPtr.p->m_subscriberData;
+ req->part = (Uint32)part;
+
+ /***************************
+ * Send to all participants
+ ***************************/
+ NodeReceiverGroup rg(GREP, m_grep->m_aliveNodes);
+ subPtr.p->m_outstandingParticipants = rg;
+ sendSignal(rg,
+ GSN_GREP_SYNC_REQ, signal, GrepSyncReq::SignalLength, JBB);
+}
+
+
+/**
+ * Sync req from Grep::PSCoord to PS particpant
+ */
+void
+Grep::PSPart::execGREP_SYNC_REQ(Signal* signal)
+{
+ jamEntry();
+
+ GrepSyncReq * const grepReq = (GrepSyncReq *) signal->getDataPtr();
+ Uint32 part = grepReq->part;
+ Uint32 subId = grepReq->subscriptionId;
+ Uint32 subKey = grepReq->subscriptionKey;
+ Uint32 subData = grepReq->senderData;
+
+ Subscription key;
+ key.m_subscriptionId = subId;
+ key.m_subscriptionKey = subKey;
+ SubscriptionPtr subPtr;
+ ndbrequire(c_subscriptions.find(subPtr, key));
+ subPtr.p->m_operationPtrI = subData;
+ subPtr.p->m_outstandingRequest = GSN_GREP_SYNC_REQ;
+ /**********************************
+ * Send SUB_SYNC_REQ to local SUMA
+ **********************************/
+ SubSyncReq * sumaReq = (SubSyncReq *)grepReq;
+ sumaReq->subscriptionId = subId;
+ sumaReq->subscriptionKey = subKey;
+ sumaReq->subscriberData = subPtr.i;
+ sumaReq->part = part;
+ sendSignal(SUMA_REF, GSN_SUB_SYNC_REQ, signal,
+ SubSyncReq::SignalLength, JBB);
+}
+
+
+/**
+ * SYNC conf from SUMA
+ */
+void
+Grep::PSPart::execSUB_SYNC_CONF(Signal* signal)
+{
+ jamEntry();
+
+ SubSyncConf * const conf = (SubSyncConf *) signal->getDataPtr();
+ Uint32 part = conf->part;
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ Uint32 subData = conf->subscriberData;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+
+ ndbrequire(subPtr.p->m_subscriptionId == subId);
+ ndbrequire(subPtr.p->m_subscriptionKey == subKey);
+
+ GrepSyncConf * grepConf = (GrepSyncConf *)conf;
+ grepConf->senderNodeId = getOwnNodeId();
+ grepConf->part = part;
+ grepConf->firstGCI = m_firstScanGCI;
+ grepConf->lastGCI = m_lastScanGCI;
+ grepConf->subscriptionId = subId;
+ grepConf->subscriptionKey = subKey;
+ grepConf->senderData = subPtr.p->m_operationPtrI;
+ sendSignal(subPtr.p->m_coordinatorRef, GSN_GREP_SYNC_CONF, signal,
+ GrepSyncConf::SignalLength, JBB);
+
+ m_firstScanGCI = 1;
+ m_lastScanGCI = 0;
+ subPtr.p->m_outstandingRequest = 0;
+}
+
+/**
+ * Handle errors that either occured in:
+ * 1) PSPart
+ * or
+ * 2) propagated from local SUMA
+ *
+ * Propagates REF signal to PSCoord
+ */
+void
+Grep::PSPart::execSUB_SYNC_REF(Signal* signal) {
+ jamEntry();
+ SubSyncRef * const ref = (SubSyncRef *)signal->getDataPtr();
+ Uint32 subData = ref->subscriberData;
+ GrepError::Code err = (GrepError::Code)ref->err;
+ SubscriptionData::Part part = (SubscriptionData::Part)ref->part;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subData);
+ sendRefToPSCoord(signal, *subPtr.p, err /*error*/ ,part);
+ subPtr.p->m_outstandingRequest = 0;
+}
+
+/**
+ * Syncing has started... (says PS Participant)
+ */
+void
+Grep::PSCoord::execGREP_SYNC_CONF(Signal* signal)
+{
+ jamEntry();
+
+ GrepSyncConf const * conf = (GrepSyncConf *)signal->getDataPtr();
+ Uint32 part = conf->part;
+ Uint32 firstGCI = conf->firstGCI;
+ Uint32 lastGCI = conf->lastGCI;
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ Uint32 subData = conf->senderData;
+
+ SubCoordinatorPtr subPtr;
+ c_subCoordinatorPool.getPtr(subPtr, subData);
+ ndbrequire(subPtr.p->m_outstandingRequest == GSN_GREP_SYNC_REQ);
+
+ subPtr.p->m_outstandingParticipants.clearWaitingFor(conf->senderNodeId);
+ if(!subPtr.p->m_outstandingParticipants.done()) return;
+
+ /**
+ * Send event
+ */
+ GrepEvent::Subscription event;
+ if(part == SubscriptionData::MetaData)
+ event = GrepEvent::GrepPS_SubSyncMetaConf;
+ else
+ event = GrepEvent::GrepPS_SubSyncDataConf;
+
+ /* @todo Johan: Add firstGCI here. /Lars */
+ m_grep->sendEventRep(signal, EventReport::GrepSubscriptionInfo,
+ event, subId, subKey,
+ (Uint32)GrepError::NO_ERROR,
+ lastGCI);
+
+ /*************************
+ * All participants ready
+ *************************/
+ GrepSubSyncConf * grepConf = (GrepSubSyncConf *)conf;
+ grepConf->part = part;
+ grepConf->firstGCI = firstGCI;
+ grepConf->lastGCI = lastGCI;
+ grepConf->subscriptionId = subId;
+ grepConf->subscriptionKey = subKey;
+
+ sendSignal(subPtr.p->m_subscriberRef, GSN_GREP_SUB_SYNC_CONF, signal,
+ GrepSubSyncConf::SignalLength, JBB);
+ c_subCoordinatorPool.release(subPtr);
+}
+
+/**
+ * Handle errors that either occured in:
+ * 1) PSCoord
+ * or
+ * 2) propagated from PSPart
+ */
+void
+Grep::PSCoord::execGREP_SYNC_REF(Signal* signal) {
+ jamEntry();
+ GrepSyncRef * const ref = (GrepSyncRef *)signal->getDataPtr();
+ Uint32 subData = ref->senderData;
+ SubscriptionData::Part part = (SubscriptionData::Part)ref->part;
+ GrepError::Code err = (GrepError::Code)ref->err;
+ SubCoordinatorPtr subPtr;
+ c_runningSubscriptions.getPtr(subPtr, subData);
+ sendRefToSS(signal, *subPtr.p, err /*error*/, part);
+}
+
+
+
+void
+Grep::PSCoord::sendRefToSS(Signal * signal,
+ SubCoordinator sub,
+ GrepError::Code err,
+ SubscriptionData::Part part) {
+ /**
+
+ GrepCreateRef * ref = (GrepCreateRef*)signal->getDataPtrSend();
+ ref->senderData = sub.m_subscriberData;
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+ ref->err = err;
+ sendSignal(sub.m_coordinatorRef, GSN_GREP_CREATE_REF, signal,
+ GrepCreateRef::SignalLength, JBB);
+*/
+
+ jam();
+ GrepEvent::Subscription event;
+ switch(sub.m_outstandingRequest) {
+ case GSN_GREP_CREATE_SUBID_REQ:
+ {
+ jam();
+ CreateSubscriptionIdRef * ref =
+ (CreateSubscriptionIdRef*)signal->getDataPtrSend();
+ ref->err = (Uint32)err;
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+ sendSignal(sub.m_subscriberRef,
+ GSN_GREP_CREATE_SUBID_REF,
+ signal,
+ CreateSubscriptionIdRef::SignalLength,
+ JBB);
+ event = GrepEvent::GrepPS_CreateSubIdRef;
+ }
+ break;
+ case GSN_GREP_CREATE_REQ:
+ {
+ jam();
+ GrepSubCreateRef * ref = (GrepSubCreateRef*)signal->getDataPtrSend();
+ ref->err = (Uint32)err;
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+ sendSignal(sub.m_subscriberRef, GSN_GREP_SUB_CREATE_REF, signal,
+ GrepSubCreateRef::SignalLength, JBB);
+ event = GrepEvent::GrepPS_SubCreateRef;
+ }
+ break;
+ case GSN_GREP_SYNC_REQ:
+ {
+ jam();
+ GrepSubSyncRef * ref = (GrepSubSyncRef*)signal->getDataPtrSend();
+ ref->err = (Uint32)err;
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+ ref->part = (SubscriptionData::Part) part;
+ sendSignal(sub.m_subscriberRef,
+ GSN_GREP_SUB_SYNC_REF,
+ signal,
+ GrepSubSyncRef::SignalLength,
+ JBB);
+ if(part == SubscriptionData::MetaData)
+ event = GrepEvent::GrepPS_SubSyncMetaRef;
+ else
+ event = GrepEvent::GrepPS_SubSyncDataRef;
+ }
+ break;
+ case GSN_GREP_START_REQ:
+ {
+ jam();
+ GrepSubStartRef * ref = (GrepSubStartRef*)signal->getDataPtrSend();
+ ref->err = (Uint32)err;
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+
+ sendSignal(sub.m_subscriberRef, GSN_GREP_SUB_START_REF,
+ signal, GrepSubStartRef::SignalLength, JBB);
+ if(part == SubscriptionData::MetaData)
+ event = GrepEvent::GrepPS_SubStartMetaRef;
+ else
+ event = GrepEvent::GrepPS_SubStartDataRef;
+ /**
+ * Send event report
+ */
+ m_grep->sendEventRep(signal,
+ EventReport::GrepSubscriptionAlert,
+ event,
+ sub.m_subscriptionId,
+ sub.m_subscriptionKey,
+ (Uint32)err);
+ }
+ break;
+ case GSN_GREP_REMOVE_REQ:
+ {
+ jam();
+ GrepSubRemoveRef * ref = (GrepSubRemoveRef*)signal->getDataPtrSend();
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+ ref->err = (Uint32)err;
+
+ sendSignal(sub.m_subscriberRef,
+ GSN_GREP_SUB_REMOVE_REF,
+ signal,
+ GrepSubRemoveRef::SignalLength,
+ JBB);
+
+ event = GrepEvent::GrepPS_SubRemoveRef;
+ }
+ break;
+ default:
+ ndbrequire(false);
+ }
+ /**
+ * Finally, send an event.
+ */
+ m_grep->sendEventRep(signal,
+ EventReport::GrepSubscriptionAlert,
+ event,
+ sub.m_subscriptionId,
+ sub.m_subscriptionKey,
+ err);
+
+}
+
+
+void
+Grep::PSPart::sendRefToPSCoord(Signal * signal,
+ Subscription sub,
+ GrepError::Code err,
+ SubscriptionData::Part part) {
+
+ jam();
+ GrepEvent::Subscription event;
+ switch(sub.m_outstandingRequest) {
+
+ case GSN_GREP_CREATE_REQ:
+ {
+ GrepCreateRef * ref = (GrepCreateRef*)signal->getDataPtrSend();
+ ref->senderData = sub.m_subscriberData;
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+ ref->err = err;
+ sendSignal(sub.m_coordinatorRef, GSN_GREP_CREATE_REF, signal,
+ GrepCreateRef::SignalLength, JBB);
+
+ event = GrepEvent::GrepPS_SubCreateRef;
+ }
+ break;
+ case GSN_GREP_SYNC_REQ:
+ {
+ GrepSyncRef * ref = (GrepSyncRef*)signal->getDataPtrSend();
+ ref->senderData = sub.m_subscriberData;
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+ ref->part = part;
+ ref->err = err;
+ sendSignal(sub.m_coordinatorRef,
+ GSN_GREP_SYNC_REF, signal,
+ GrepSyncRef::SignalLength, JBB);
+ if(part == SubscriptionData::MetaData)
+ event = GrepEvent::GrepPS_SubSyncMetaRef;
+ else
+ event = GrepEvent::GrepPS_SubSyncDataRef;
+ }
+ break;
+ case GSN_GREP_START_REQ:
+ {
+ jam();
+ GrepStartRef * ref = (GrepStartRef*)signal->getDataPtrSend();
+ ref->senderData = sub.m_subscriberData;
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+ ref->part = (Uint32) part;
+ ref->err = err;
+ sendSignal(sub.m_coordinatorRef, GSN_GREP_START_REF, signal,
+ GrepStartRef::SignalLength, JBB);
+ if(part == SubscriptionData::MetaData)
+ event = GrepEvent::GrepPS_SubStartMetaRef;
+ else
+ event = GrepEvent::GrepPS_SubStartDataRef;
+ }
+ break;
+
+ case GSN_GREP_REMOVE_REQ:
+ {
+ jamEntry();
+ GrepRemoveRef * ref = (GrepRemoveRef*)signal->getDataPtrSend();
+ ref->senderData = sub.m_operationPtrI;
+ ref->subscriptionId = sub.m_subscriptionId;
+ ref->subscriptionKey = sub.m_subscriptionKey;
+ ref->err = err;
+ sendSignal(sub.m_coordinatorRef, GSN_GREP_REMOVE_REF, signal,
+ GrepCreateRef::SignalLength, JBB);
+
+ }
+ break;
+ default:
+ ndbrequire(false);
+ }
+
+ /**
+ * Finally, send an event.
+ */
+ m_grep->sendEventRep(signal,
+ EventReport::GrepSubscriptionAlert,
+ event,
+ sub.m_subscriptionId,
+ sub.m_subscriptionKey,
+ err);
+
+}
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: GREP PS Coordinator GCP
+ * ------------------------------------------------------------------------
+ *
+ *
+ **************************************************************************/
+
+void
+Grep::PSPart::execSUB_GCP_COMPLETE_REP(Signal* signal)
+{
+ jamEntry();
+ if(m_recoveryMode) {
+ jam();
+ return;
+ }
+ SubGcpCompleteRep * rep = (SubGcpCompleteRep *)signal->getDataPtrSend();
+ rep->senderRef = reference();
+
+ if (rep->gci > m_latestSeenGCI) m_latestSeenGCI = rep->gci;
+ SubscriptionPtr subPtr;
+ c_subscriptions.first(c_subPtr);
+ for(; !c_subPtr.isNull(); c_subscriptions.next(c_subPtr)) {
+
+ subPtr.i = c_subPtr.curr.i;
+ subPtr.p = c_subscriptions.getPtr(subPtr.i);
+ sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_GCP_COMPLETE_REP, signal,
+ SubGcpCompleteRep::SignalLength, JBB);
+ }
+
+#ifdef DEBUG_GREP
+ ndbout_c("Grep::PSPart: Recd SUB_GCP_COMPLETE_REP "
+ "(GCI: %d, nodeId: %d) from SUMA",
+ rep->gci, refToNode(rep->senderRef));
+#endif
+}
+
+
+void
+Grep::PSPart::execSUB_SYNC_CONTINUE_REQ(Signal* signal)
+{
+ jamEntry();
+ SubSyncContinueReq * const req = (SubSyncContinueReq*)signal->getDataPtr();
+ Uint32 subData = req->subscriberData;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr,subData);
+
+ /**
+ * @todo Figure out how to control how much data we can receive?
+ */
+ SubSyncContinueConf * conf = (SubSyncContinueConf*)req;
+ conf->subscriptionId = subPtr.p->m_subscriptionId;
+ conf->subscriptionKey = subPtr.p->m_subscriptionKey;
+ sendSignal(SUMA_REF, GSN_SUB_SYNC_CONTINUE_CONF, signal,
+ SubSyncContinueConf::SignalLength, JBB);
+}
+
+void
+Grep::sendEventRep(Signal * signal,
+ EventReport::EventType type,
+ GrepEvent::Subscription event,
+ Uint32 subId,
+ Uint32 subKey,
+ Uint32 err,
+ Uint32 other) {
+ jam();
+ signal->theData[0] = type;
+ signal->theData[1] = event;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ signal->theData[4] = err;
+
+ if(other==0)
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 5 ,JBB);
+ else {
+ signal->theData[5] = other;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 6 ,JBB);
+ }
+}
diff --git a/ndb/src/kernel/blocks/grep/Grep.hpp b/ndb/src/kernel/blocks/grep/Grep.hpp
new file mode 100644
index 00000000000..ba8f5780522
--- /dev/null
+++ b/ndb/src/kernel/blocks/grep/Grep.hpp
@@ -0,0 +1,548 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GREP_HPP
+#define GREP_HPP
+
+#include <ndb_limits.h>
+#include <SimulatedBlock.hpp>
+
+#include <NodeBitmask.hpp>
+#include <SignalCounter.hpp>
+#include <SLList.hpp>
+
+#include <DLList.hpp>
+
+#include <GrepError.hpp>
+#include <GrepEvent.hpp>
+
+#include <signaldata/EventReport.hpp>
+#include <signaldata/SumaImpl.hpp>
+
+
+/**
+ * Module in block (Should be placed elsewhere)
+ */
+class BlockComponent {
+public:
+ BlockComponent(SimulatedBlock *);
+ BlockReference reference() { return m_sb->reference(); };
+ BlockNumber number() { return m_sb->number(); };
+
+ void sendSignal(NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf ) const {
+ m_sb->sendSignal(rg, gsn, signal, length, jbuf);
+ }
+
+ void sendSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf ) const {
+ m_sb->sendSignal(ref, gsn, signal, length, jbuf);
+ }
+
+ void sendSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections) const {
+ m_sb->sendSignal(ref, gsn, signal, length, jbuf, ptr, noOfSections);
+ }
+
+ void sendSignalWithDelay(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 delayInMilliSeconds,
+ Uint32 length) const {
+
+ m_sb->sendSignalWithDelay(ref, gsn, signal, delayInMilliSeconds, length);
+ }
+
+ NodeId getOwnNodeId() const {
+ return m_sb->getOwnNodeId();
+ }
+
+ bool assembleFragments(Signal * signal) {
+ return m_sb->assembleFragments(signal);
+ }
+
+ void progError(int line, int err_code, const char* extra) {
+ m_sb->progError(line, err_code, extra);
+ }
+
+private:
+ SimulatedBlock * m_sb;
+};
+
+
+
+/**
+ * Participant of GREP Protocols (not necessarily a protocol coordinator)
+ *
+ * This object is only used on primary system
+ */
+#if 0
+class GrepParticipant : public SimulatedBlock
+{
+protected:
+ GrepParticipant(const Configuration & conf);
+ virtual ~GrepParticipant();
+ BLOCK_DEFINES(GrepParticipant);
+
+protected:
+ /***************************************************************************
+ * SUMA Signal Interface
+ ***************************************************************************/
+ void execSUB_CREATE_CONF(Signal*);
+ void execSUB_STARTCONF(Signal*);
+ void execSUB_REMOVE_CONF(Signal*);
+
+ void execSUB_META_DATA(Signal*);
+ void execSUB_TABLE_DATA(Signal*);
+
+ void execSUB_SYNC_CONF(Signal*);
+
+ void execSUB_GCP_COMPLETE_REP(Signal*);
+ void execSUB_SYNC_CONTINUE_REQ(Signal*);
+
+ /***************************************************************************
+ * GREP Coordinator Signal Interface
+ ***************************************************************************/
+ void execGREP_CREATE_REQ(Signal*);
+ void execGREP_START_REQ(Signal*);
+ void execGREP_SYNC_REQ(Signal*);
+ void execGREP_REMOVE_REQ(Signal*);
+
+
+protected:
+ BlockReference m_repRef; ///< Replication node (only one rep node per grep)
+
+private:
+ BlockReference m_coordinator;
+ Uint32 m_latestSeenGCI;
+};
+#endif
+
+
+/**
+ * GREP Coordinator
+ */
+class Grep : public SimulatedBlock //GrepParticipant
+{
+ //BLOCK_DEFINES(Grep);
+
+public:
+ Grep(const Configuration & conf);
+ virtual ~Grep();
+
+private:
+ /***************************************************************************
+ * General Signal Recivers
+ ***************************************************************************/
+ void execSTTOR(Signal*);
+ void sendSTTORRY(Signal*);
+ void execNDB_STTOR(Signal*);
+ void execDUMP_STATE_ORD(Signal*);
+ void execREAD_NODESCONF(Signal*);
+ void execNODE_FAILREP(Signal*);
+ void execINCL_NODEREQ(Signal*);
+ void execGREP_REQ(Signal*);
+ void execAPI_FAILREQ(Signal*);
+ /**
+ * Forwarded to PSCoord
+ */
+ //CONF
+ void fwdGREP_CREATE_CONF(Signal* s) {
+ pscoord.execGREP_CREATE_CONF(s); };
+ void fwdGREP_START_CONF(Signal* s) {
+ pscoord.execGREP_START_CONF(s); };
+ void fwdGREP_SYNC_CONF(Signal* s) {
+ pscoord.execGREP_SYNC_CONF(s); };
+ void fwdGREP_REMOVE_CONF(Signal* s) {
+ pscoord.execGREP_REMOVE_CONF(s); };
+ void fwdCREATE_SUBID_CONF(Signal* s) {
+ pscoord.execCREATE_SUBID_CONF(s); };
+
+ //REF
+
+ void fwdGREP_CREATE_REF(Signal* s) {
+ pscoord.execGREP_CREATE_REF(s); };
+ void fwdGREP_START_REF(Signal* s) {
+ pscoord.execGREP_START_REF(s); };
+ void fwdGREP_SYNC_REF(Signal* s) {
+ pscoord.execGREP_SYNC_REF(s); };
+
+ void fwdGREP_REMOVE_REF(Signal* s) {
+ pscoord.execGREP_REMOVE_REF(s); };
+
+ void fwdCREATE_SUBID_REF(Signal* s) {
+ pscoord.execCREATE_SUBID_REF(s); };
+
+ //REQ
+ void fwdGREP_SUB_CREATE_REQ(Signal* s) {
+ pscoord.execGREP_SUB_CREATE_REQ(s); };
+ void fwdGREP_SUB_START_REQ(Signal* s) {
+ pscoord.execGREP_SUB_START_REQ(s); };
+ void fwdGREP_SUB_SYNC_REQ(Signal* s) {
+ pscoord.execGREP_SUB_SYNC_REQ(s); };
+ void fwdGREP_SUB_REMOVE_REQ(Signal* s) {
+ pscoord.execGREP_SUB_REMOVE_REQ(s); };
+ void fwdGREP_CREATE_SUBID_REQ(Signal* s) {
+ pscoord.execGREP_CREATE_SUBID_REQ(s); };
+
+ /**
+ * Forwarded to PSPart
+ */
+
+ void fwdSTART_ME(Signal* s){
+ pspart.execSTART_ME(s);
+ };
+ void fwdGREP_ADD_SUB_REQ(Signal* s){
+ pspart.execGREP_ADD_SUB_REQ(s);
+ };
+ void fwdGREP_ADD_SUB_REF(Signal* s){
+ pspart.execGREP_ADD_SUB_REF(s);
+ };
+ void fwdGREP_ADD_SUB_CONF(Signal* s){
+ pspart.execGREP_ADD_SUB_CONF(s);
+ };
+
+ //CONF
+ void fwdSUB_CREATE_CONF(Signal* s) {
+ pspart.execSUB_CREATE_CONF(s); };
+ void fwdSUB_START_CONF(Signal* s) {
+ pspart.execSUB_START_CONF(s); };
+ void fwdSUB_REMOVE_CONF(Signal* s) {
+ pspart.execSUB_REMOVE_CONF(s); };
+ void fwdSUB_SYNC_CONF(Signal* s) {
+ pspart.execSUB_SYNC_CONF(s); };
+
+ //REF
+
+ void fwdSUB_CREATE_REF(Signal* s) {
+ pspart.execSUB_CREATE_REF(s); };
+ void fwdSUB_START_REF(Signal* s) {
+ pspart.execSUB_START_REF(s); };
+ void fwdSUB_REMOVE_REF(Signal* s) {
+ pspart.execSUB_REMOVE_REF(s); };
+ void fwdSUB_SYNC_REF(Signal* s) {
+ pspart.execSUB_SYNC_REF(s); };
+
+ //REQ
+ void fwdSUB_SYNC_CONTINUE_REQ(Signal* s) {
+ pspart.execSUB_SYNC_CONTINUE_REQ(s); };
+ void fwdGREP_CREATE_REQ(Signal* s) {
+ pspart.execGREP_CREATE_REQ(s); };
+ void fwdGREP_START_REQ(Signal* s) {
+ pspart.execGREP_START_REQ(s); };
+ void fwdGREP_SYNC_REQ(Signal* s) {
+ pspart.execGREP_SYNC_REQ(s); };
+ void fwdGREP_REMOVE_REQ(Signal* s) {
+ pspart.execGREP_REMOVE_REQ(s); };
+
+ void fwdSUB_META_DATA(Signal* s) {
+ pspart.execSUB_META_DATA(s); };
+ void fwdSUB_TABLE_DATA(Signal* s) {
+ pspart.execSUB_TABLE_DATA(s); };
+
+ void fwdSUB_GCP_COMPLETE_REP(Signal* s) {
+ pspart.execSUB_GCP_COMPLETE_REP(s); };
+
+ void sendEventRep(Signal * signal,
+ EventReport::EventType type,
+ GrepEvent::Subscription event,
+ Uint32 subId,
+ Uint32 subKey,
+ Uint32 err,
+ Uint32 gci=0);
+
+ void getNodeGroupMembers(Signal* signal);
+
+
+ /***************************************************************************
+ * Block Data
+ ***************************************************************************/
+ struct Node {
+ Uint32 nodeId;
+ Uint32 alive;
+ Uint32 nextList;
+ union { Uint32 prevList; Uint32 nextPool; };
+ };
+ typedef Ptr<Node> NodePtr;
+
+ NodeId m_masterNodeId;
+ SLList<Node> m_nodes;
+ NdbNodeBitmask m_aliveNodes;
+ ArrayPool<Node> m_nodePool;
+
+ /**
+ * for all Suma's to keep track of other Suma's in Node group
+ */
+ Uint32 c_nodeGroup;
+ Uint32 c_noNodesInGroup;
+ Uint32 c_idInNodeGroup;
+ NodeId c_nodesInGroup[4];
+
+
+public:
+ /***************************************************************************
+ * GREP PS Coordinator
+ ***************************************************************************/
+ class PSCoord : public BlockComponent {
+
+ private:
+
+ struct SubCoordinator {
+ Uint32 m_subscriberRef;
+ Uint32 m_subscriberData;
+ Uint32 m_coordinatorRef;
+ Uint32 m_subscriptionId;
+ Uint32 m_subscriptionKey;
+ Uint32 m_subscriptionType;
+ NdbNodeBitmask m_participants;
+ Uint32 m_outstandingRequest;
+ SignalCounter m_outstandingParticipants;
+
+ Uint32 nextHash;
+ union { Uint32 prevHash; Uint32 nextPool; };
+
+ Uint32 hashValue() const {
+ return m_subscriptionId + m_subscriptionKey;
+ }
+
+ bool equal(const SubCoordinator & s) const {
+ return
+ m_subscriptionId == s.m_subscriptionId &&
+ m_subscriptionKey == s.m_subscriptionKey;
+ }
+
+ };
+
+ typedef Ptr<SubCoordinator> SubCoordinatorPtr;
+ ArrayPool<SubCoordinator> c_subCoordinatorPool;
+ DLHashTable<SubCoordinator>::Iterator c_subPtr;
+ DLHashTable<SubCoordinator> c_runningSubscriptions;
+
+ void prepareOperationRec(SubCoordinatorPtr ptr,
+ BlockReference subscriber,
+ Uint32 subId,
+ Uint32 subKey,
+ Uint32 request);
+
+ public:
+ PSCoord(class Grep *);
+
+ void execGREP_CREATE_CONF(Signal*);
+ void execGREP_START_CONF(Signal*);
+ void execGREP_SYNC_CONF(Signal*);
+ void execGREP_REMOVE_CONF(Signal*);
+
+ void execGREP_CREATE_REF(Signal*);
+ void execGREP_START_REF(Signal*);
+ void execGREP_SYNC_REF(Signal*);
+ void execGREP_REMOVE_REF(Signal*);
+
+
+ void execCREATE_SUBID_CONF(Signal*); //comes from SUMA
+ void execGREP_CREATE_SUBID_REQ(Signal*);
+
+ void execGREP_SUB_CREATE_REQ(Signal*);
+ void execGREP_SUB_START_REQ(Signal*);
+ void execGREP_SUB_SYNC_REQ(Signal*);
+ void execGREP_SUB_REMOVE_REQ(Signal*);
+
+
+
+ void execCREATE_SUBID_REF(Signal*);
+
+
+
+ void sendCreateSubIdRef_SS(Signal * signal,
+ Uint32 subId,
+ Uint32 subKey,
+ BlockReference to,
+ GrepError::Code err);
+
+
+ void sendSubRemoveRef_SS(Signal * signal,
+ SubCoordinator sub,
+ GrepError::Code err);
+
+ void sendRefToSS(Signal * signal,
+ SubCoordinator sub,
+ GrepError::Code err,
+ SubscriptionData::Part part = (SubscriptionData::Part)0);
+
+ void setRepRef(BlockReference rr) { m_repRef = rr; };
+ //void setAliveNodes(NdbNodeBitmask an) { m_aliveNodes = an; };
+
+ BlockReference m_repRef; ///< Rep node (only one rep node per grep)
+ // NdbNodeBitmask m_aliveNodes;
+
+ Uint32 m_outstandingRequest;
+ SignalCounter m_outstandingParticipants;
+
+ Grep * m_grep;
+ } pscoord;
+ friend class PSCoord;
+
+ /***************************************************************************
+ * GREP PS Participant
+ ***************************************************************************
+ * Participant of GREP Protocols (not necessarily a protocol coordinator)
+ *
+ * This object is only used on primary system
+ ***************************************************************************/
+ class PSPart: public BlockComponent
+ {
+ //protected:
+ //GrepParticipant(const Configuration & conf);
+ //virtual ~GrepParticipant();
+ //BLOCK_DEFINES(GrepParticipant);
+
+ struct Subscription {
+ Uint32 m_subscriberRef;
+ Uint32 m_subscriberData;
+ Uint32 m_subscriptionId;
+ Uint32 m_subscriptionKey;
+ Uint32 m_subscriptionType;
+ Uint32 m_coordinatorRef;
+ Uint32 m_outstandingRequest;
+ Uint32 m_operationPtrI;
+ Uint32 nextHash;
+ union { Uint32 prevHash; Uint32 nextPool; };
+
+ Uint32 hashValue() const {
+ return m_subscriptionId + m_subscriptionKey;
+ }
+
+ bool equal(const Subscription & s) const {
+ return
+ m_subscriptionId == s.m_subscriptionId &&
+ m_subscriptionKey == s.m_subscriptionKey;
+ }
+
+ };
+ typedef Ptr<Subscription> SubscriptionPtr;
+
+ DLHashTable<Subscription> c_subscriptions;
+ DLHashTable<Subscription>::Iterator c_subPtr;
+ ArrayPool<Subscription> c_subscriptionPool;
+
+ public:
+ PSPart(class Grep *);
+
+
+ //protected:
+ /*************************************************************************
+ * SUMA Signal Interface
+ *************************************************************************/
+ void execSUB_CREATE_CONF(Signal*);
+ void execSUB_START_CONF(Signal*);
+ void execSUB_SYNC_CONF(Signal*);
+ void execSUB_REMOVE_CONF(Signal*);
+
+ void execSUB_CREATE_REF(Signal*);
+ void execSUB_START_REF(Signal*);
+ void execSUB_SYNC_REF(Signal*);
+ void execSUB_REMOVE_REF(Signal*);
+
+
+ void execSUB_META_DATA(Signal*);
+ void execSUB_TABLE_DATA(Signal*);
+
+
+ void execSUB_GCP_COMPLETE_REP(Signal*);
+ void execSUB_SYNC_CONTINUE_REQ(Signal*);
+
+ /*************************************************************************
+ * GREP Coordinator Signal Interface
+ *************************************************************************/
+ void execGREP_CREATE_REQ(Signal*);
+ void execGREP_START_REQ(Signal*);
+ void execGREP_SYNC_REQ(Signal*);
+ void execGREP_REMOVE_REQ(Signal*);
+
+ /**
+ * NR/NF signals
+ */
+ void execSTART_ME(Signal *);
+ void execGREP_ADD_SUB_REQ(Signal *);
+ void execGREP_ADD_SUB_REF(Signal *);
+ void execGREP_ADD_SUB_CONF(Signal *);
+
+ /*************************************************************************
+ * GREP Coordinator error handling interface
+ *************************************************************************/
+
+ void sendRefToPSCoord(Signal * signal,
+ Subscription sub,
+ GrepError::Code err,
+ SubscriptionData::Part part = (SubscriptionData::Part)0);
+
+ //protected:
+ BlockReference m_repRef; ///< Replication node
+ ///< (only one rep node per grep)
+ bool m_recoveryMode;
+
+ private:
+ BlockReference m_coordinator;
+ Uint32 m_firstScanGCI;
+ Uint32 m_lastScanGCI;
+ Uint32 m_latestSeenGCI;
+ Grep * m_grep;
+ } pspart;
+ friend class PSPart;
+
+ /***************************************************************************
+ * AddRecSignal Stuff (should maybe be gerneralized)
+ ***************************************************************************/
+ typedef void (Grep::* ExecSignalLocal1) (Signal* signal);
+ typedef void (Grep::PSCoord::* ExecSignalLocal2) (Signal* signal);
+ typedef void (Grep::PSPart::* ExecSignalLocal4) (Signal* signal);
+
+ void
+ addRecSignal(GlobalSignalNumber gsn, ExecSignalLocal1 f, bool force = false){
+ addRecSignalImpl(gsn, (ExecFunction)f, force);
+ }
+ void
+ addRecSignal(GlobalSignalNumber gsn, ExecSignalLocal2 f, bool force = false){
+ addRecSignalImpl(gsn, (ExecFunction)f, force);
+ }
+ void
+ addRecSignal(GlobalSignalNumber gsn, ExecSignalLocal4 f, bool force = false){
+ addRecSignalImpl(gsn, (ExecFunction)f, force);
+ }
+};
+
+
+/*************************************************************************
+ * Requestor
+ *
+ * The following methods are callbacks (registered functions)
+ * for the Requestor. The Requestor calls these when it needs
+ * something to be done.
+ *************************************************************************/
+void startSubscription(void * cbObj, Signal*, int type);
+void scanSubscription(void * cbObj, Signal*, int type);
+
+#endif
diff --git a/ndb/src/kernel/blocks/grep/GrepInit.cpp b/ndb/src/kernel/blocks/grep/GrepInit.cpp
new file mode 100644
index 00000000000..70bf6678754
--- /dev/null
+++ b/ndb/src/kernel/blocks/grep/GrepInit.cpp
@@ -0,0 +1,165 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Grep.hpp"
+#include <new>
+#include <Properties.hpp>
+#include <Configuration.hpp>
+
+/*****************************************************************************
+ * Grep Participant
+ *****************************************************************************/
+#if 0
+GrepParticipant::GrepParticipant(const Configuration & conf) :
+ SimulatedBlock(GREP, conf)
+{
+ BLOCK_CONSTRUCTOR(Grep);
+ //m_repRef = 0;
+ m_latestSeenGCI = 0;
+}
+
+GrepParticipant::~GrepParticipant()
+{
+}
+
+BLOCK_FUNCTIONS(GrepParticipant);
+#endif
+
+/*****************************************************************************
+ * Grep Coordinator
+ *****************************************************************************/
+Grep::Grep(const Configuration & conf) :
+ // GrepParticipant(conf),
+ SimulatedBlock(GREP, conf),
+ m_nodes(m_nodePool),
+ pscoord(this),
+ pspart(this)
+{
+ m_nodePool.setSize(MAX_NDB_NODES);
+ m_masterNodeId = getOwnNodeId();
+
+ /***************************************************************************
+ * General Signals
+ ***************************************************************************/
+ addRecSignal(GSN_STTOR, &Grep::execSTTOR);
+ addRecSignal(GSN_NDB_STTOR, &Grep::execNDB_STTOR);
+ addRecSignal(GSN_DUMP_STATE_ORD, &Grep::execDUMP_STATE_ORD);
+ addRecSignal(GSN_READ_NODESCONF, &Grep::execREAD_NODESCONF);
+ addRecSignal(GSN_NODE_FAILREP, &Grep::execNODE_FAILREP);
+ addRecSignal(GSN_INCL_NODEREQ, &Grep::execINCL_NODEREQ);
+
+ addRecSignal(GSN_GREP_REQ, &Grep::execGREP_REQ);
+ addRecSignal(GSN_API_FAILREQ, &Grep::execAPI_FAILREQ);
+
+
+ /***************************************************************************
+ * Grep::PSCoord Signal Interface
+ ***************************************************************************/
+ /**
+ * From Grep::PSPart
+ */
+ addRecSignal(GSN_GREP_CREATE_CONF, &Grep::fwdGREP_CREATE_CONF);
+ addRecSignal(GSN_GREP_START_CONF, &Grep::fwdGREP_START_CONF);
+ addRecSignal(GSN_GREP_SYNC_CONF, &Grep::fwdGREP_SYNC_CONF);
+ addRecSignal(GSN_GREP_REMOVE_CONF, &Grep::fwdGREP_REMOVE_CONF);
+
+ addRecSignal(GSN_GREP_CREATE_REF, &Grep::fwdGREP_CREATE_REF);
+ addRecSignal(GSN_GREP_START_REF, &Grep::fwdGREP_START_REF);
+ addRecSignal(GSN_GREP_REMOVE_REF, &Grep::fwdGREP_REMOVE_REF);
+
+ /**
+ * From Grep::SSCoord to Grep::PSCoord
+ */
+ addRecSignal(GSN_GREP_SUB_START_REQ, &Grep::fwdGREP_SUB_START_REQ);
+ addRecSignal(GSN_GREP_SUB_CREATE_REQ, &Grep::fwdGREP_SUB_CREATE_REQ);
+ addRecSignal(GSN_GREP_SUB_SYNC_REQ, &Grep::fwdGREP_SUB_SYNC_REQ);
+ addRecSignal(GSN_GREP_SUB_REMOVE_REQ, &Grep::fwdGREP_SUB_REMOVE_REQ);
+ addRecSignal(GSN_GREP_CREATE_SUBID_REQ, &Grep::fwdGREP_CREATE_SUBID_REQ);
+
+ /****************************************************************************
+ * PSPart
+ ***************************************************************************/
+ /**
+ * From SUMA to GREP PS Participant. If suma is not a coodinator
+ */
+ addRecSignal(GSN_SUB_START_CONF, &Grep::fwdSUB_START_CONF);
+ addRecSignal(GSN_SUB_CREATE_CONF, &Grep::fwdSUB_CREATE_CONF);
+ addRecSignal(GSN_SUB_SYNC_CONF, &Grep::fwdSUB_SYNC_CONF);
+ addRecSignal(GSN_SUB_REMOVE_CONF, &Grep::fwdSUB_REMOVE_CONF);
+ addRecSignal(GSN_SUB_CREATE_REF, &Grep::fwdSUB_CREATE_REF);
+ addRecSignal(GSN_SUB_START_REF, &Grep::fwdSUB_START_REF);
+ addRecSignal(GSN_SUB_SYNC_REF, &Grep::fwdSUB_SYNC_REF);
+ addRecSignal(GSN_SUB_REMOVE_REF, &Grep::fwdSUB_REMOVE_REF);
+
+ addRecSignal(GSN_SUB_SYNC_CONTINUE_REQ,
+ &Grep::fwdSUB_SYNC_CONTINUE_REQ);
+
+ /**
+ * From Suma to Grep::PSPart. Data signals.
+ */
+ addRecSignal(GSN_SUB_META_DATA, &Grep::fwdSUB_META_DATA);
+ addRecSignal(GSN_SUB_TABLE_DATA, &Grep::fwdSUB_TABLE_DATA);
+ addRecSignal(GSN_SUB_GCP_COMPLETE_REP, &Grep::fwdSUB_GCP_COMPLETE_REP);
+
+ /**
+ * From Grep::PSCoord to Grep::PSPart
+ */
+ addRecSignal(GSN_GREP_CREATE_REQ, &Grep::fwdGREP_CREATE_REQ);
+ addRecSignal(GSN_GREP_START_REQ, &Grep::fwdGREP_START_REQ);
+ addRecSignal(GSN_GREP_REMOVE_REQ, &Grep::fwdGREP_REMOVE_REQ);
+ addRecSignal(GSN_GREP_SYNC_REQ, &Grep::fwdGREP_SYNC_REQ);
+ addRecSignal(GSN_CREATE_SUBID_CONF, &Grep::fwdCREATE_SUBID_CONF);
+ addRecSignal(GSN_GREP_START_ME, &Grep::fwdSTART_ME);
+ addRecSignal(GSN_GREP_ADD_SUB_REQ, &Grep::fwdGREP_ADD_SUB_REQ);
+ addRecSignal(GSN_GREP_ADD_SUB_REF, &Grep::fwdGREP_ADD_SUB_REF);
+ addRecSignal(GSN_GREP_ADD_SUB_CONF, &Grep::fwdGREP_ADD_SUB_CONF);
+}
+
+Grep::~Grep()
+{
+}
+
+//BLOCK_FUNCTIONS(Grep);
+
+Grep::PSPart::PSPart(Grep * sb) :
+ BlockComponent(sb),
+ c_subscriptions(c_subscriptionPool)
+{
+ m_grep = sb;
+
+ m_firstScanGCI = 1; // Empty interval = [1,0]
+ m_lastScanGCI = 0;
+
+ m_latestSeenGCI = 0;
+
+ c_subscriptions.setSize(10);
+ c_subscriptionPool.setSize(10);
+}
+
+Grep::PSCoord::PSCoord(Grep * sb) :
+ BlockComponent(sb),
+ c_runningSubscriptions(c_subCoordinatorPool)
+{
+ m_grep = sb;
+ c_runningSubscriptions.setSize(10);
+ c_subCoordinatorPool.setSize(2);
+}
+
+//BLOCK_FUNCTIONS(Grep::PSCoord);
+
+BlockComponent::BlockComponent(SimulatedBlock * sb) {
+ m_sb = sb;
+}
diff --git a/ndb/src/kernel/blocks/grep/Makefile b/ndb/src/kernel/blocks/grep/Makefile
new file mode 100644
index 00000000000..5ad5a0bce3b
--- /dev/null
+++ b/ndb/src/kernel/blocks/grep/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := grep
+
+SOURCES = Grep.cpp GrepInit.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/grep/systab_test/Makefile b/ndb/src/kernel/blocks/grep/systab_test/Makefile
new file mode 100644
index 00000000000..bd69e0f3799
--- /dev/null
+++ b/ndb/src/kernel/blocks/grep/systab_test/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := kernel
+
+BIN_TARGET := grep_systab_test
+BIN_TARGET_ARCHIVES := portlib general
+
+CCFLAGS_LOC += -I..
+
+SOURCES = ../GrepSystemTable.cpp grep_systab_test.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/grep/systab_test/grep_systab_test.cpp b/ndb/src/kernel/blocks/grep/systab_test/grep_systab_test.cpp
new file mode 100644
index 00000000000..e3a77af4e4e
--- /dev/null
+++ b/ndb/src/kernel/blocks/grep/systab_test/grep_systab_test.cpp
@@ -0,0 +1,138 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * Unit Test for GrepSystemTable
+ */
+
+#include "../GrepSystemTable.hpp"
+#include <SimulatedBlock.hpp>
+
+#define EXEC(X) ( ndbout << endl, ndbout_c(#X), X )
+
+int
+main () {
+ GrepSystemTable st;
+
+ Uint32 f, l;
+
+ ndbout_c("*************************************");
+ ndbout_c("* GrepSystemTable Unit Test Program *");
+ ndbout_c("*************************************");
+
+ ndbout_c("--------------------------------------------------------");
+ ndbout_c("Test 1: Clear");
+ ndbout_c("--------------------------------------------------------");
+
+ EXEC(st.set(GrepSystemTable::PS, 22, 26));
+ st.print();
+ st.require(GrepSystemTable::PS, 22, 26);
+
+ EXEC(st.clear(GrepSystemTable::PS, 20, 24));
+ st.print();
+ st.require(GrepSystemTable::PS, 25, 26);
+
+ EXEC(st.clear(GrepSystemTable::PS, 0, 100));
+ st.print();
+ st.require(GrepSystemTable::PS, 1, 0);
+
+ EXEC(st.set(GrepSystemTable::PS, 22, 26));
+ st.print();
+ st.require(GrepSystemTable::PS, 22, 26);
+
+ EXEC(st.clear(GrepSystemTable::PS, 24, 28));
+ st.print();
+ st.require(GrepSystemTable::PS, 22, 23);
+
+ EXEC(st.clear(GrepSystemTable::PS, 0, 100));
+ st.print();
+ st.require(GrepSystemTable::PS, 1, 0);
+
+ EXEC(st.set(GrepSystemTable::PS, 22, 26));
+ st.print();
+ st.require(GrepSystemTable::PS, 22, 26);
+
+ EXEC(st.clear(GrepSystemTable::PS, 24, 26));
+ st.print();
+ st.require(GrepSystemTable::PS, 22, 23);
+
+ EXEC(st.clear(GrepSystemTable::PS, 0, 100));
+ st.print();
+ st.require(GrepSystemTable::PS, 1, 0);
+
+ EXEC(st.set(GrepSystemTable::PS, 22, 26));
+ st.print();
+ st.require(GrepSystemTable::PS, 22, 26);
+
+ EXEC(st.clear(GrepSystemTable::PS, 22, 24));
+ st.print();
+ st.require(GrepSystemTable::PS, 25, 26);
+
+ ndbout_c("--------------------------------------------------------");
+ ndbout_c("Test 2: PS --> SSreq");
+ ndbout_c("--------------------------------------------------------");
+
+ EXEC(st.set(GrepSystemTable::PS, 22, 26));
+ st.print();
+ st.require(GrepSystemTable::PS, 22, 26);
+ st.require(GrepSystemTable::SSReq, 1, 0);
+
+ if (!EXEC(st.copy(GrepSystemTable::PS, GrepSystemTable::SSReq, 3, &f, &l)))
+ ndbout_c("%s:%d: Illegal copy!", __FILE__, __FILE__);
+ ndbout_c("f=%d, l=%d", f, l);
+ st.print();
+ st.require(GrepSystemTable::PS, 22, 26);
+ st.require(GrepSystemTable::SSReq, 22, 24);
+
+ EXEC(st.clear(GrepSystemTable::PS, 22, 22));
+ st.print();
+ st.require(GrepSystemTable::PS, 23, 26);
+ st.require(GrepSystemTable::SSReq, 22, 24);
+
+ if (!EXEC(st.copy(GrepSystemTable::PS, GrepSystemTable::SSReq, 2, &f, &l)))
+ ndbout_c("%s:%d: Illegal copy!", __FILE__, __LINE__);
+ ndbout_c("f=%d, l=%d", f, l);
+ st.print();
+ st.require(GrepSystemTable::PS, 23, 26);
+ st.require(GrepSystemTable::SSReq, 22, 26);
+
+ st.set(GrepSystemTable::SS, 7, 9);
+ st.set(GrepSystemTable::InsReq, 7, 9);
+ if (EXEC(st.movable(GrepSystemTable::SS, GrepSystemTable::InsReq)))
+ ndbout_c("%s:%d: Illegal move!", __FILE__, __LINE__);
+ st.print();
+ st.require(GrepSystemTable::SS, 7, 9);
+ st.require(GrepSystemTable::InsReq, 7, 9);
+
+ EXEC(st.intervalMinus(7, 9, 7, 7, &f, &l));
+ ndbout_c("f=%d, l=%d", f, l);
+
+ st.clear(GrepSystemTable::InsReq, 8, 9);
+ st.require(GrepSystemTable::SS, 7, 9);
+ st.require(GrepSystemTable::InsReq, 7, 7);
+ if (EXEC(st.movable(GrepSystemTable::SS, GrepSystemTable::InsReq)) != 2)
+ ndbout_c("%s:%d: Illegal move!", __FILE__, __LINE__);
+ st.print();
+
+ EXEC(st.copy(GrepSystemTable::SS, GrepSystemTable::InsReq, &f));
+ st.print();
+ st.require(GrepSystemTable::SS, 7, 9);
+ st.require(GrepSystemTable::InsReq, 7, 8);
+
+ ndbout_c("--------------------------------------------------------");
+ ndbout_c("Test completed");
+ ndbout_c("--------------------------------------------------------");
+}
diff --git a/ndb/src/kernel/blocks/mutexes.hpp b/ndb/src/kernel/blocks/mutexes.hpp
new file mode 100644
index 00000000000..5c0276fc4fa
--- /dev/null
+++ b/ndb/src/kernel/blocks/mutexes.hpp
@@ -0,0 +1,39 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef KERNEL_MUTEXES_HPP
+#define KERNEL_MUTEXES_HPP
+
+#include <ndb_types.h>
+
+/**
+ * This mutex is used by:
+ * DIH - before sending START_LCP to all participants
+ * DICT - before commiting a CREATE TABLE
+ * BACKUP - before sending DEFINE_BACKUP
+ */
+#define DIH_START_LCP_MUTEX 0
+#define DICT_COMMIT_TABLE_MUTEX 0
+
+/**
+ * This mutex is used by
+ * DIH - before switching primary replica
+ * BACKUP - before sending DEFINE_BACKUP
+ */
+#define DIH_SWITCH_PRIMARY_MUTEX 1
+#define BACKUP_DEFINE_MUTEX 1
+
+#endif
diff --git a/ndb/src/kernel/blocks/ndbcntr/Makefile b/ndb/src/kernel/blocks/ndbcntr/Makefile
new file mode 100644
index 00000000000..8e9c4f01027
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbcntr/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := ndbcntr
+
+SOURCES = \
+ NdbcntrInit.cpp \
+ NdbcntrSysTable.cpp \
+ NdbcntrMain.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/ndbcntr/Ndbcntr.hpp b/ndb/src/kernel/blocks/ndbcntr/Ndbcntr.hpp
new file mode 100644
index 00000000000..a481573e370
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbcntr/Ndbcntr.hpp
@@ -0,0 +1,513 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBCNTR_H
+#define NDBCNTR_H
+
+
+#include <pc.hpp>
+#include <SimulatedBlock.hpp>
+#include <ndb_limits.h>
+#include <signaldata/CmvmiCfgConf.hpp>
+#include <signaldata/StopReq.hpp>
+#include <signaldata/ResumeReq.hpp>
+#include <signaldata/DictTabInfo.hpp>
+
+#include <NodeState.hpp>
+#include <NdbTick.h>
+
+#ifdef NDBCNTR_C
+/*
+2.1 GLOBAL SYMBOLS
+------------------
+*/
+/*
+2.2 LOCAL SYMBOLS
+-----------------
+*/
+#define ZNO_NDB_BLOCKS 6 /* ACC, DICT, DIH, LQH, TC, TUP */
+#define ZDELAY_NODERESTART 5 /* MASTER REFUSED NODERESTART, WAIT SEC */
+#define ZDELAY_START 25 /* WAIT SECONDS FOR OTHER NODES, RESTART*/
+#define ZHB_INTERVAL 10 /* HEART BEAT INTERVAL TIME MS */
+#define ZNO_NDB_NODES 1
+#define ZHB_TYPE 1
+
+//------- ERROR CODES -----------------------------------------
+#define ZERROR_ALREADY_EXISTS 901
+#define ZERROR_DOESNT_EXIST 902
+#define ZERROR_VALUE_OUT_OF_RANGE 903
+#define ZERROR_NOT_STARTED 904
+#define ZERROR_NODE_RESTART 905
+#define ZERROR_NO_RESTART_NODES 906
+#define ZERROR_STARTPHASE_VALUE 907
+#define ZERROR_CNTR_MASTERREQ 908
+#define ZERROR_CNTR_MASTERREF 909
+#define ZERROR_CNTR_MASTERCONF 910
+#define ZERROR_NOT_RUNNING 911
+#define ZERROR_CNTR_WAITREP 912
+#define ZNOT_AVAILABLE 913
+#define ZERROR_DISK_FAILURE 914
+#define ZERROR_TOO_MANY_NODES 915
+#define ZERROR_TYPEOFSTART 916
+#define ZERROR_CTYPE_OF_START 917
+#define ZERROR_ZSTART 918
+#define ZERROR_AT_SELECT_START 919
+#define ZERROR_CNTR_CHANGEREP 920
+#define ZERR_DETECT_NODERESTART_1 921
+#define ZERR_DETECT_NODERESTART_2 922
+#define ZERROR_CONTINUEB 923
+#define ZERR_APPL_REGREF 924
+#define ZERR_DICTADDATTRREF 925
+#define ZERR_DICTPREPAREREF 926
+#define ZERR_DICTRELEASEREF 927
+#define ZERR_DICTTABREF 928
+#define ZERR_DICTSEIZEREF 929
+#define ZERR_NDB_STARTREF 930
+#define ZERR_NODE_STATESREF 931
+#define ZERR_READ_NODESREF 932
+#define ZERR_TCKEYREF 933
+#define ZERR_TCRELEASEREF 934
+#define ZERR_TCSEIZEREF 935
+#define ZNOTVALIDSTATE_1 936
+#define ZNOTVALIDSTATE_2 937
+#define ZNODE_FAILURE_DURING_RESTART 938
+#define ZSTART_IN_PROGRESS_ERROR 939
+#define ZCOULD_NOT_OCCUR_ERROR 940
+#define ZTOO_MANY_NODES_ERROR 941
+#define ZNODE_RESTART_ONGOING_ERROR 942
+#define ZWE_ARE_DECLARED_DEAD_ERROR 943
+#define ZTIME_OUT_ERROR 944
+#define ZTYPE_OF_START_ERROR 945
+#define ZMASTER_CONFLICT_ERROR 946
+#define ZNODE_CONFLICT_ERROR 947
+
+//------- OTHERS ---------------------------------------------
+#define ZCONTINUEB_1 1
+#define ZCONTINUEB_2 2
+#define ZSHUTDOWN 3
+
+#define ZAPPL_SUBTYPE 0
+#define ZNAME_OF_APPL "NDB"
+#define ZVOTING 2
+#define ZSIZE_CFG_BLOCK_REC 8
+#define ZSIZE_NDB_BLOCKS_REC 16 /* MAX BLOCKS IN NDB */
+#define ZSIZE_SYSTAB 2048
+#define ZSTART_PHASE_1 1
+#define ZSTART_PHASE_2 2
+#define ZSTART_PHASE_3 3
+#define ZSTART_PHASE_4 4
+#define ZSTART_PHASE_5 5
+#define ZSTART_PHASE_6 6
+#define ZSTART_PHASE_7 7
+#define ZSTART_PHASE_8 8
+#define ZSTART_PHASE_9 9
+#define ZSTART_PHASE_END 255
+#define ZWAITPOINT_4_1 1
+#define ZWAITPOINT_4_2 2
+#define ZWAITPOINT_5_1 3
+#define ZWAITPOINT_5_2 4
+#define ZWAITPOINT_6_1 5
+#define ZWAITPOINT_6_2 6
+#define ZWAITPOINT_7_1 7
+#define ZWAITPOINT_7_2 8
+#define ZSYSTAB_VERSION 1
+/*
+------- SIGNAL CONSTANTS -----------------------------------
+*/
+#define ZNOT_MASTER 0
+/* REASON OF CNTR_MASTERREF */
+#define ZTOO_FEW_NODES 1
+#define ZNEW_MASTER 0
+#define ZDELETE_NODE 1
+#define ZSYSTAB_EXIST 3
+#define ZVOTE_NEW_NODE 4
+#define ZVARIABLE_NO 1
+#define ZAMOUNT_PAGES 8
+#endif
+
+class Ndbcntr: public SimulatedBlock {
+public:
+ // State values
+ enum State {
+ NOT_ACTIVE = 0,
+ ACTIVE = 1
+ };
+
+// Records
+
+/* FSREADREQ FSWRITEREQ */
+/**
+ * 2.3 RECORDS AND FILESIZES
+ * ------------------------------------------------------------
+ */
+
+/**
+ * CFG_BLOCK_REC CONTAINS ALL CONFIG DATA SENT IN EACH NDB_STTOR
+ * SOME OTHER CONFIG DATA IS STORED IN RECORD 0
+ *
+ * WHEN CFG_BLOCK_PTR = ZSTART_PHASE_X ( CINTERNAL_STARTPHASE )
+ * WORD 0 DICT_1
+ * WORD 1 DICT_2
+ * WORD 2 DIH_1
+ * WORD 3 DIH_2
+ * WORD 4 LQH_1
+ * WORD 5 LQH_2
+ * WORD 6 TC_1
+ * WORD 7 TC_2
+ * WORD 8 TUP_1
+ * WORD 9 TUP_2
+ * WORD 10 ACC_1
+ * WORD 11 ACC_2
+ *
+ * CFG_BLOCK_PTR = 0
+ * WORD 0 CDELAY_START
+ * WORD 1 CDELAY_NODERESTART
+ *------------------------------------------------------------------------*/
+ struct CfgBlockRec {
+ UintR cfgData[CmvmiCfgConf::NO_OF_WORDS];
+ }; /* p2c: size = 64 bytes */
+
+ typedef Ptr<CfgBlockRec> CfgBlockRecPtr;
+
+/*------------------------------------------------------------------------*/
+// CONTAIN INFO ABOUT ALL NODES IN CLUSTER. NODE_PTR ARE USED AS NODE NUMBER.
+// IF THE STATE ARE ZDELETE THEN THE NODE DOESN'T EXIST. NODES ARE ALLOWED
+// TO REGISTER (ZADD) DURING RESTART.
+// WHEN THE SYSTEM IS RUNNING THE MASTER WILL CHECK IF ANY NODE HAS MADE A
+// CNTR_MASTERREQ AND TAKE CARE OF THE
+// REQUEST. TO CONFIRM THE REQ, THE MASTER DEMANDS THAT ALL RUNNING NODES HAS
+// VOTED FOR THE NEW NODE.
+// NODE_PTR:MASTER_REQ IS USED DURING RESTART TO LOG POSTPONED
+// CNTR_MASTERREQ'S
+/*------------------------------------------------------------------------*/
+ struct NodeRec {
+ UintR dynamicId;
+ BlockReference cntrBlockref;
+ Uint16 masterReq;
+ Uint16 state;
+ Uint16 ndbVersion;
+ Uint16 subType;
+ Uint8 votes;
+ Uint8 voter;
+ UintR nodeDefined;
+ }; /* p2c: size = 16 bytes */
+
+ typedef Ptr<NodeRec> NodeRecPtr;
+
+ struct NdbBlocksRec {
+ BlockReference blockref;
+ }; /* p2c: size = 2 bytes */
+
+ typedef Ptr<NdbBlocksRec> NdbBlocksRecPtr;
+
+ /**
+ * Ndbcntr creates and initializes system tables on initial system start.
+ * The tables are defined in static structs in NdbcntrSysTable.cpp.
+ */
+ struct SysColumn {
+ unsigned pos;
+ const char* name;
+ // DictTabInfo
+ DictTabInfo::ExtType type;
+ Uint32 length;
+ bool keyFlag;
+ bool nullable;
+ };
+ struct SysTable {
+ const char* name;
+ unsigned columnCount;
+ const SysColumn* columnList;
+ // DictTabInfo
+ DictTabInfo::TableType tableType;
+ DictTabInfo::FragmentType fragmentType;
+ bool tableLoggedFlag;
+ // saved table id
+ mutable Uint32 tableId;
+ };
+ struct SysIndex {
+ const char* name;
+ const SysTable* primaryTable;
+ Uint32 columnCount;
+ Uint32 columnList[4];
+ // DictTabInfo
+ DictTabInfo::TableType indexType;
+ DictTabInfo::FragmentType fragmentType;
+ bool indexLoggedFlag;
+ // saved index table id
+ mutable Uint32 indexId;
+ };
+ static const SysTable* g_sysTableList[];
+ static const unsigned g_sysTableCount;
+ // the system tables
+ static const SysTable g_sysTable_SYSTAB_0;
+ static const SysTable g_sysTable_NDBEVENTS_0;
+
+public:
+ Ndbcntr(const class Configuration &);
+ virtual ~Ndbcntr();
+
+private:
+ BLOCK_DEFINES(Ndbcntr);
+
+ // Transit signals
+ void execCONTINUEB(Signal* signal);
+ void execREAD_NODESCONF(Signal* signal);
+ void execREAD_NODESREF(Signal* signal);
+ void execCNTR_MASTERREQ(Signal* signal);
+ void execCNTR_MASTERCONF(Signal* signal);
+ void execCNTR_MASTERREF(Signal* signal);
+ void execCNTR_WAITREP(Signal* signal);
+ void execNODE_STATESREQ(Signal* signal);
+ void execNODE_STATESCONF(Signal* signal);
+ void execNODE_STATESREF(Signal* signal);
+ void execNODE_FAILREP(Signal* signal);
+ void execSYSTEM_ERROR(Signal* signal);
+ void execVOTE_MASTERORD(Signal* signal);
+
+ // Received signals
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execSTTOR(Signal* signal);
+ void execTCSEIZECONF(Signal* signal);
+ void execTCSEIZEREF(Signal* signal);
+ void execTCRELEASECONF(Signal* signal);
+ void execTCRELEASEREF(Signal* signal);
+ void execTCKEYCONF(Signal* signal);
+ void execTCKEYREF(Signal* signal);
+ void execTCROLLBACKREP(Signal* signal);
+ void execGETGCICONF(Signal* signal);
+ void execDIH_RESTARTCONF(Signal* signal);
+ void execDIH_RESTARTREF(Signal* signal);
+ void execCREATE_TABLE_REF(Signal* signal);
+ void execCREATE_TABLE_CONF(Signal* signal);
+ void execNDB_STTORRY(Signal* signal);
+ void execNDB_STARTCONF(Signal* signal);
+ void execREAD_NODESREQ(Signal* signal);
+ void execAPPL_REGCONF(Signal* signal);
+ void execAPPL_REGREF(Signal* signal);
+ void execAPPL_CHANGEREP(Signal* signal);
+ void execAPPL_STARTCONF(Signal* signal);
+ void execNDB_STARTREF(Signal* signal);
+ void execCMVMI_CFGCONF(Signal* signal);
+ void execSET_VAR_REQ(Signal* signal);
+
+ void execSTOP_PERM_REF(Signal* signal);
+ void execSTOP_PERM_CONF(Signal* signal);
+
+ void execSTOP_ME_REF(Signal* signal);
+ void execSTOP_ME_CONF(Signal* signal);
+
+ void execWAIT_GCP_REF(Signal* signal);
+ void execWAIT_GCP_CONF(Signal* signal);
+
+ void execSTOP_REQ(Signal* signal);
+ void execRESUME_REQ(Signal* signal);
+
+ void execCHANGE_NODE_STATE_CONF(Signal* signal);
+
+ void execABORT_ALL_REF(Signal* signal);
+ void execABORT_ALL_CONF(Signal* signal);
+
+ // Statement blocks
+ void sendCreateTabReq(Signal* signal, const char* buffer, Uint32 bufLen);
+ void startInsertTransactions(Signal* signal);
+ UintR checkNodelist(Signal* signal, Uint16 TnoRestartNodes);
+ void chooseRestartNodes(Signal* signal);
+ void copyCfgVariables(Signal* signal);
+ void deleteNode(Signal* signal);
+ void detectNoderestart(Signal* signal);
+ void getStartNodes(Signal* signal);
+ void initData(Signal* signal);
+ void replyMasterconfToAll(Signal* signal);
+ void resetStartVariables(Signal* signal);
+ void sendCntrMasterreq(Signal* signal);
+ void sendNdbSttor(Signal* signal);
+ void sendSttorry(Signal* signal);
+
+ // Generated statement blocks
+ void systemErrorLab(Signal* signal);
+
+ void createSystableLab(Signal* signal, unsigned index);
+ void crSystab7Lab(Signal* signal);
+ void crSystab8Lab(Signal* signal);
+ void crSystab9Lab(Signal* signal);
+
+ void startPhase1Lab(Signal* signal);
+ void startPhase2Lab(Signal* signal);
+ void startPhase3Lab(Signal* signal);
+ void startPhase4Lab(Signal* signal);
+ void startPhase5Lab(Signal* signal);
+ // jump 2 to resync phase counters
+ void startPhase8Lab(Signal* signal);
+ void startPhase9Lab(Signal* signal);
+ void ph2ALab(Signal* signal);
+ void ph2CLab(Signal* signal);
+ void ph2ELab(Signal* signal);
+ void ph2FLab(Signal* signal);
+ void ph2GLab(Signal* signal);
+ void ph3ALab(Signal* signal);
+ void ph4ALab(Signal* signal);
+ void ph4BLab(Signal* signal);
+ void ph4CLab(Signal* signal);
+ void ph5ALab(Signal* signal);
+ void ph6ALab(Signal* signal);
+ void ph6BLab(Signal* signal);
+ void ph7ALab(Signal* signal);
+ void ph8ALab(Signal* signal);
+
+ void masterreq010Lab(Signal* signal,
+ Uint16 TnoRestartNodes,
+ Uint16 TuserNodeId);
+ void masterreq020Lab(Signal* signal);
+ void masterreq030Lab(Signal* signal,
+ Uint16 TnoRestartNodes,
+ Uint16 TuserNodeId);
+
+ void waitpoint41Lab(Signal* signal);
+ void waitpoint51Lab(Signal* signal);
+ void waitpoint52Lab(Signal* signal);
+ void waitpoint61Lab(Signal* signal);
+ void waitpoint71Lab(Signal* signal);
+
+ void updateNodeState(Signal* signal, const NodeState & newState) const ;
+ void getNodeGroup(Signal* signal);
+
+ // Initialisation
+ void initData();
+ void initRecords();
+
+ // Variables
+ /**------------------------------------------------------------------------
+ * CONTAIN INFO ABOUT ALL NODES IN CLUSTER. NODE_PTR ARE USED AS NODE NUMBER
+ * IF THE STATE ARE ZDELETE THEN THE NODE DOESN'T EXIST. NODES ARE ALLOWED
+ * TO REGISTER (ZADD) DURING RESTART.
+ *
+ * WHEN THE SYSTEM IS RUNNING THE MASTER WILL CHECK IF ANY NODE HAS MADE
+ * A CNTR_MASTERREQ AND TAKE CARE OF THE REQUEST.
+ * TO CONFIRM THE REQ, THE MASTER DEMANDS THAT ALL RUNNING NODES HAS VOTED
+ * FOR THE NEW NODE.
+ * NODE_PTR:MASTER_REQ IS USED DURING RESTART TO LOG
+ * POSTPONED CNTR_MASTERREQ'S
+ *------------------------------------------------------------------------*/
+ CfgBlockRec *cfgBlockRec;
+ NodeRec *nodeRec;
+ NdbBlocksRec *ndbBlocksRec;
+ NodeRecPtr nodePtr;
+/*
+2.4 COMMON STORED VARIABLES
+*/
+ Uint16 cstartNodes[MAX_NDB_NODES];
+ BlockReference ccmvmiBlockref;
+ BlockReference cqmgrBlockref;
+ BlockReference cdictBlockref;
+ BlockReference cdihBlockref;
+ BlockReference clqhBlockref;
+ BlockReference cownBlockref;
+ BlockReference ctcBlockref;
+ UintR cnoNdbNodes;
+ UintR capplStartconfFlag;
+ UintR cgciSystab;
+ UintR ckey;
+ UintR cnoRegNodes;
+ UintR cnoRunNodes;
+ UintR cnoNeedNodes;
+ UintR cnoWaitrep;
+ UintR cnoWaitrep6;
+ UintR cnoWaitrep7;
+ //UintR csystabId;
+ UintR ctcConnectionP;
+ UintR ctcReqInfo;
+ UintR clastGci;
+ UintR cmasterLastGci;
+ UintR cmasterCurrentId;
+ Uint16 cmasterDihId;
+ Uint16 cresponses;
+ Uint16 cdelayStart;
+
+ Uint16 cinternalStartphase;
+ Uint16 cmasterNodeId;
+ Uint16 cmasterCandidateId;
+ Uint16 cndbBlocksCount;
+ Uint16 cnoStartNodes;
+ Uint16 cnoVoters;
+ Uint16 cstartProgressFlag;
+ Uint16 cqmgrConnectionP;
+ Uint16 csignalKey;
+ NodeState::StartType ctypeOfStart;
+ Uint16 cmasterVoters;
+ Uint16 cdynamicNodeId;
+ Uint8 cwaitContinuebFlag;
+ Uint8 cstartPhase;
+ Uint8 ctransidPhase;
+
+ Uint32 c_fsRemoveCount;
+ Uint32 c_nodeGroup;
+ void clearFilesystem(Signal* signal);
+ void execFSREMOVEREF(Signal* signal);
+ void execFSREMOVECONF(Signal* signal);
+
+public:
+ struct StopRecord {
+ public:
+ StopRecord(Ndbcntr & _cntr) : cntr(_cntr) {
+ stopReq.senderRef = 0;
+ }
+
+ Ndbcntr & cntr;
+ StopReq stopReq; // Signal data
+ NDB_TICKS stopInitiatedTime; // When was the stop initiated
+
+ bool checkNodeFail(Signal* signal);
+ void checkTimeout(Signal* signal);
+ void checkApiTimeout(Signal* signal);
+ void checkTcTimeout(Signal* signal);
+ void checkLqhTimeout_1(Signal* signal);
+ void checkLqhTimeout_2(Signal* signal);
+
+ BlockNumber number() const { return cntr.number(); }
+ void progError(int line, int cause, const char * extra) {
+ cntr.progError(line, cause, extra);
+ }
+ };
+private:
+ StopRecord c_stopRec;
+ friend struct StopRecord;
+
+ struct Missra {
+ Missra(Ndbcntr & ref) : cntr(ref) { }
+
+ Uint32 currentBlockIndex;
+ Uint32 currentStartPhase;
+ Uint32 nextStartPhase[NO_OF_BLOCKS];
+
+ void execSTART_ORD(Signal* signal);
+ void execSTTORRY(Signal* signal);
+ void sendNextSTTOR(Signal* signal);
+
+ BlockNumber number() const { return cntr.number(); }
+ void progError(int line, int cause, const char * extra) {
+ cntr.progError(line, cause, extra);
+ }
+ Ndbcntr & cntr;
+ };
+
+ Missra c_missra;
+ friend struct Missra;
+
+ void execSTTORRY(Signal* signal);
+ void execSTART_ORD(Signal* signal);
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/ndbcntr/NdbcntrInit.cpp b/ndb/src/kernel/blocks/ndbcntr/NdbcntrInit.cpp
new file mode 100644
index 00000000000..9af6359876b
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbcntr/NdbcntrInit.cpp
@@ -0,0 +1,126 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#define NDBCNTR_C
+#include "Ndbcntr.hpp"
+#include <ndb_limits.h>
+
+#define DEBUG(x) { ndbout << "Ndbcntr::" << x << endl; }
+
+
+void Ndbcntr::initData()
+{
+
+ // Records with constant sizes
+ cfgBlockRec = new CfgBlockRec[ZSIZE_CFG_BLOCK_REC];
+ nodeRec = new NodeRec[MAX_NDB_NODES];
+ ndbBlocksRec = new NdbBlocksRec[ZSIZE_NDB_BLOCKS_REC];
+}//Ndbcntr::initData()
+
+void Ndbcntr::initRecords()
+{
+ // Records with dynamic sizes
+}//Ndbcntr::initRecords()
+
+Ndbcntr::Ndbcntr(const class Configuration & conf):
+ SimulatedBlock(NDBCNTR, conf),
+ c_stopRec(* this),
+ c_missra(* this),
+ cnoWaitrep6(0),
+ cnoWaitrep7(0)
+{
+
+ BLOCK_CONSTRUCTOR(Ndbcntr);
+
+ // Transit signals
+ addRecSignal(GSN_CONTINUEB, &Ndbcntr::execCONTINUEB);
+ addRecSignal(GSN_READ_NODESCONF, &Ndbcntr::execREAD_NODESCONF);
+ addRecSignal(GSN_READ_NODESREF, &Ndbcntr::execREAD_NODESREF);
+ addRecSignal(GSN_CNTR_MASTERREQ, &Ndbcntr::execCNTR_MASTERREQ);
+ addRecSignal(GSN_CNTR_MASTERCONF, &Ndbcntr::execCNTR_MASTERCONF);
+ addRecSignal(GSN_CNTR_MASTERREF, &Ndbcntr::execCNTR_MASTERREF);
+ addRecSignal(GSN_CNTR_WAITREP, &Ndbcntr::execCNTR_WAITREP);
+ addRecSignal(GSN_NODE_STATESREQ, &Ndbcntr::execNODE_STATESREQ);
+ addRecSignal(GSN_NODE_STATESCONF, &Ndbcntr::execNODE_STATESCONF);
+ addRecSignal(GSN_NODE_STATESREF, &Ndbcntr::execNODE_STATESREF);
+ addRecSignal(GSN_NODE_FAILREP, &Ndbcntr::execNODE_FAILREP);
+ addRecSignal(GSN_SYSTEM_ERROR , &Ndbcntr::execSYSTEM_ERROR);
+ addRecSignal(GSN_VOTE_MASTERORD, &Ndbcntr::execVOTE_MASTERORD);
+
+ // Received signals
+ addRecSignal(GSN_DUMP_STATE_ORD, &Ndbcntr::execDUMP_STATE_ORD);
+ addRecSignal(GSN_STTOR, &Ndbcntr::execSTTOR);
+ addRecSignal(GSN_TCSEIZECONF, &Ndbcntr::execTCSEIZECONF);
+ addRecSignal(GSN_TCSEIZEREF, &Ndbcntr::execTCSEIZEREF);
+ addRecSignal(GSN_TCRELEASECONF, &Ndbcntr::execTCRELEASECONF);
+ addRecSignal(GSN_TCRELEASEREF, &Ndbcntr::execTCRELEASEREF);
+ addRecSignal(GSN_TCKEYCONF, &Ndbcntr::execTCKEYCONF);
+ addRecSignal(GSN_TCKEYREF, &Ndbcntr::execTCKEYREF);
+ addRecSignal(GSN_TCROLLBACKREP, &Ndbcntr::execTCROLLBACKREP);
+ addRecSignal(GSN_GETGCICONF, &Ndbcntr::execGETGCICONF);
+ addRecSignal(GSN_DIH_RESTARTCONF, &Ndbcntr::execDIH_RESTARTCONF);
+ addRecSignal(GSN_DIH_RESTARTREF, &Ndbcntr::execDIH_RESTARTREF);
+ addRecSignal(GSN_CREATE_TABLE_REF, &Ndbcntr::execCREATE_TABLE_REF);
+ addRecSignal(GSN_CREATE_TABLE_CONF, &Ndbcntr::execCREATE_TABLE_CONF);
+ addRecSignal(GSN_NDB_STTORRY, &Ndbcntr::execNDB_STTORRY);
+ addRecSignal(GSN_NDB_STARTCONF, &Ndbcntr::execNDB_STARTCONF);
+ addRecSignal(GSN_READ_NODESREQ, &Ndbcntr::execREAD_NODESREQ);
+ addRecSignal(GSN_APPL_REGCONF, &Ndbcntr::execAPPL_REGCONF);
+ addRecSignal(GSN_APPL_REGREF, &Ndbcntr::execAPPL_REGREF);
+ addRecSignal(GSN_APPL_CHANGEREP, &Ndbcntr::execAPPL_CHANGEREP);
+ addRecSignal(GSN_APPL_STARTCONF, &Ndbcntr::execAPPL_STARTCONF);
+ addRecSignal(GSN_NDB_STARTREF, &Ndbcntr::execNDB_STARTREF);
+ addRecSignal(GSN_CMVMI_CFGCONF, &Ndbcntr::execCMVMI_CFGCONF);
+ addRecSignal(GSN_SET_VAR_REQ, &Ndbcntr::execSET_VAR_REQ);
+
+ addRecSignal(GSN_STOP_PERM_REF, &Ndbcntr::execSTOP_PERM_REF);
+ addRecSignal(GSN_STOP_PERM_CONF, &Ndbcntr::execSTOP_PERM_CONF);
+
+ addRecSignal(GSN_STOP_ME_REF, &Ndbcntr::execSTOP_ME_REF);
+ addRecSignal(GSN_STOP_ME_CONF, &Ndbcntr::execSTOP_ME_CONF);
+
+ addRecSignal(GSN_STOP_REQ, &Ndbcntr::execSTOP_REQ);
+ addRecSignal(GSN_RESUME_REQ, &Ndbcntr::execRESUME_REQ);
+
+ addRecSignal(GSN_WAIT_GCP_REF, &Ndbcntr::execWAIT_GCP_REF);
+ addRecSignal(GSN_WAIT_GCP_CONF, &Ndbcntr::execWAIT_GCP_CONF);
+ addRecSignal(GSN_CHANGE_NODE_STATE_CONF,
+ &Ndbcntr::execCHANGE_NODE_STATE_CONF);
+
+ addRecSignal(GSN_ABORT_ALL_REF, &Ndbcntr::execABORT_ALL_REF);
+ addRecSignal(GSN_ABORT_ALL_CONF, &Ndbcntr::execABORT_ALL_CONF);
+
+ addRecSignal(GSN_START_ORD, &Ndbcntr::execSTART_ORD);
+ addRecSignal(GSN_STTORRY, &Ndbcntr::execSTTORRY);
+
+ addRecSignal(GSN_FSREMOVEREF, &Ndbcntr::execFSREMOVEREF);
+ addRecSignal(GSN_FSREMOVECONF, &Ndbcntr::execFSREMOVECONF);
+
+ initData();
+ ctypeOfStart = NodeState::ST_ILLEGAL_TYPE;
+}//Ndbcntr::Ndbcntr()
+
+Ndbcntr::~Ndbcntr()
+{
+ delete []cfgBlockRec;
+ delete []nodeRec;
+ delete []ndbBlocksRec;
+
+}//Ndbcntr::~Ndbcntr()
+
+BLOCK_FUNCTIONS(Ndbcntr);
diff --git a/ndb/src/kernel/blocks/ndbcntr/NdbcntrMain.cpp b/ndb/src/kernel/blocks/ndbcntr/NdbcntrMain.cpp
new file mode 100644
index 00000000000..6f30ff2c511
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbcntr/NdbcntrMain.cpp
@@ -0,0 +1,3385 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#define NDBCNTR_C
+#include "Ndbcntr.hpp"
+
+#include <ndb_limits.h>
+#include <ndb_version.h>
+#include <SimpleProperties.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/CreateTable.hpp>
+#include <signaldata/ReadNodesConf.hpp>
+#include <signaldata/CntrMasterReq.hpp>
+#include <signaldata/CntrMasterConf.hpp>
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/TcKeyReq.hpp>
+#include <signaldata/TcKeyConf.hpp>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/SetVarReq.hpp>
+#include <signaldata/NodeStateSignalData.hpp>
+#include <signaldata/StopPerm.hpp>
+#include <signaldata/StopMe.hpp>
+#include <signaldata/WaitGCP.hpp>
+#include <signaldata/CheckNodeGroups.hpp>
+#include <signaldata/StartOrd.hpp>
+#include <signaldata/AbortAll.hpp>
+#include <signaldata/SystemError.hpp>
+#include <signaldata/NdbSttor.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+
+#include <signaldata/FsRemoveReq.hpp>
+
+#include <AttributeHeader.hpp>
+#include <Configuration.hpp>
+#include <DebuggerNames.hpp>
+
+#include <NdbOut.hpp>
+#include <NdbTick.h>
+
+#define ZSYSTEM_RUN 256
+
+/**
+ * ALL_BLOCKS Used during start phases and while changing node state
+ *
+ * NDBFS_REF Has to be before NDBCNTR_REF (due to "ndb -i" stuff)
+ */
+struct BlockInfo {
+ BlockReference Ref; // BlockReference
+ Uint32 NextSP; // Next start phase
+};
+
+static BlockInfo ALL_BLOCKS[] = {
+ { DBTC_REF, 1 },
+ { DBDIH_REF, 1 },
+ { DBLQH_REF, 1 },
+ { DBACC_REF, 1 },
+ { DBTUP_REF, 1 },
+ { DBDICT_REF, 1 },
+ { NDBFS_REF, 0 },
+ { NDBCNTR_REF, 0 },
+ { QMGR_REF, 1 },
+ { CMVMI_REF, 1 },
+ { TRIX_REF, 1 },
+ { BACKUP_REF, 1 },
+ { DBUTIL_REF, 1 },
+ { SUMA_REF, 1 },
+ { GREP_REF, 1 },
+ { DBTUX_REF, 1 }
+};
+
+static const Uint32 ALL_BLOCKS_SZ = sizeof(ALL_BLOCKS)/sizeof(BlockInfo);
+
+/*******************************/
+/* CONTINUEB */
+/*******************************/
+void Ndbcntr::execCONTINUEB(Signal* signal)
+{
+ jamEntry();
+ UintR Ttemp1 = signal->theData[0];
+ switch (Ttemp1) {
+ case ZCONTINUEB_1:
+ jam();
+ if (cwaitContinuebFlag == ZFALSE) {
+ jam();
+/*******************************/
+/* SIGNAL NOT WANTED ANYMORE */
+/*******************************/
+ return;
+ } else {
+ jam();
+/*******************************/
+/* START ALREADY IN PROGRESS */
+/*******************************/
+ if (cstartProgressFlag == ZVOTING) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ if (ctypeOfStart == NodeState::ST_NODE_RESTART) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ ph2ELab(signal);
+ return;
+ }//if
+ break;
+ case ZSHUTDOWN:
+ jam();
+ c_stopRec.checkTimeout(signal);
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+}//Ndbcntr::execCONTINUEB()
+
+/*******************************/
+/* SYSTEM_ERROR */
+/*******************************/
+void Ndbcntr::execSYSTEM_ERROR(Signal* signal)
+{
+ const SystemError * const sysErr = (SystemError *)signal->getDataPtr();
+ char buf[100];
+ int killingNode = refToNode(sysErr->errorRef);
+
+ jamEntry();
+ switch (sysErr->errorCode){
+ case SystemError::StartInProgressError:
+ snprintf(buf, sizeof(buf),
+ "Node %d killed this node because "
+ "master start in progress error",
+ killingNode);
+ break;
+
+ case SystemError::GCPStopDetected:
+ snprintf(buf, sizeof(buf),
+ "Node %d killed this node because "
+ "GCP stop was detected",
+ killingNode);
+ break;
+
+ case SystemError::ScanfragTimeout:
+ snprintf(buf, sizeof(buf),
+ "Node %d killed this node because "
+ "a fragment scan timed out and could not be stopped",
+ killingNode);
+ break;
+
+ case SystemError::ScanfragStateError:
+ snprintf(buf, sizeof(buf),
+ "Node %d killed this node because "
+ "the state of a fragment scan was out of sync.",
+ killingNode);
+ break;
+
+ case SystemError::CopyFragRefError:
+ snprintf(buf, sizeof(buf),
+ "Node %d killed this node because "
+ "it could not copy a fragment during node restart",
+ killingNode);
+ break;
+
+ default:
+ snprintf(buf, sizeof(buf), "System error %d, "
+ " this node was killed by node %d",
+ sysErr->errorCode, killingNode);
+ break;
+ }
+
+ progError(__LINE__,
+ ERR_SYSTEM_ERROR,
+ buf);
+ return;
+}//Ndbcntr::execSYSTEM_ERROR()
+
+/*---------------------------------------------------------------------------*/
+/* The STTOR signal is on level C, we use CONTINUEB to get into level B */
+/*---------------------------------------------------------------------------*/
+/**************************** >----------------------------------------------*/
+/* STTOR > SENDER : MISSRA */
+/**************************** >------------------+ RECEIVER : NDBCNTR */
+ /* INPUT : CSTART_PHASE */
+ /* CSIGNAL_KEY */
+ /*---------------------------*/
+/*******************************/
+/* STTOR */
+/*******************************/
+void Ndbcntr::execSTTOR(Signal* signal)
+{
+ jamEntry();
+ cstartPhase = signal->theData[1];
+ csignalKey = signal->theData[6];
+
+ NodeState newState(NodeState::SL_STARTING, cstartPhase,
+ (NodeState::StartType)ctypeOfStart);
+ updateNodeState(signal, newState);
+
+ switch (cstartPhase) {
+ case 0:
+ if(theConfiguration.getInitialStart()){
+ jam();
+ c_fsRemoveCount = 0;
+ clearFilesystem(signal);
+ return;
+ }
+ sendSttorry(signal);
+ break;
+ case ZSTART_PHASE_1:
+ jam();
+ startPhase1Lab(signal);
+ break;
+ case ZSTART_PHASE_2:
+ jam();
+ startPhase2Lab(signal);
+ break;
+ case ZSTART_PHASE_3:
+ jam();
+ startPhase3Lab(signal);
+ break;
+ case ZSTART_PHASE_4:
+ jam();
+ startPhase4Lab(signal);
+ break;
+ case ZSTART_PHASE_5:
+ jam();
+ startPhase5Lab(signal);
+ break;
+ case 6:
+ jam();
+ getNodeGroup(signal);
+ break;
+ case ZSTART_PHASE_8:
+ jam();
+ startPhase8Lab(signal);
+ break;
+ case ZSTART_PHASE_9:
+ jam();
+ startPhase9Lab(signal);
+ break;
+ default:
+ jam();
+ sendSttorry(signal);
+ break;
+ }//switch
+}//Ndbcntr::execSTTOR()
+
+void
+Ndbcntr::getNodeGroup(Signal* signal){
+ jam();
+ CheckNodeGroups * sd = (CheckNodeGroups*)signal->getDataPtrSend();
+ sd->requestType = CheckNodeGroups::Direct | CheckNodeGroups::GetNodeGroup;
+ EXECUTE_DIRECT(DBDIH, GSN_CHECKNODEGROUPSREQ, signal,
+ CheckNodeGroups::SignalLength);
+ jamEntry();
+ c_nodeGroup = sd->output;
+ sendSttorry(signal);
+}
+
+/*******************************/
+/* NDB_STTORRY */
+/*******************************/
+void Ndbcntr::execNDB_STTORRY(Signal* signal)
+{
+ jamEntry();
+ switch (cstartPhase) {
+ case ZSTART_PHASE_2:
+ jam();
+ ph2GLab(signal);
+ return;
+ break;
+ case ZSTART_PHASE_3:
+ jam();
+ ph3ALab(signal);
+ return;
+ break;
+ case ZSTART_PHASE_4:
+ jam();
+ ph4BLab(signal);
+ return;
+ break;
+ case ZSTART_PHASE_5:
+ jam();
+ ph5ALab(signal);
+ return;
+ break;
+ case ZSTART_PHASE_6:
+ jam();
+ ph6ALab(signal);
+ return;
+ break;
+ case ZSTART_PHASE_7:
+ jam();
+ ph6BLab(signal);
+ return;
+ break;
+ case ZSTART_PHASE_8:
+ jam();
+ ph7ALab(signal);
+ return;
+ break;
+ case ZSTART_PHASE_9:
+ jam();
+ ph8ALab(signal);
+ return;
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+}//Ndbcntr::execNDB_STTORRY()
+
+/*
+4.2 START PHASE 1 */
+/*###########################################################################*/
+/*LOAD OUR BLOCK REFERENCE AND OUR NODE ID. LOAD NODE IDS OF ALL NODES IN */
+/* CLUSTER CALCULATE BLOCK REFERENCES OF ALL BLOCKS IN THIS NODE */
+/*---------------------------------------------------------------------------*/
+/*******************************/
+/* STTOR */
+/*******************************/
+void Ndbcntr::startPhase1Lab(Signal* signal)
+{
+ jamEntry();
+
+ initData(signal);
+ cownBlockref = calcNdbCntrBlockRef(0);
+ cnoRunNodes = 0;
+ cnoRegNodes = 0;
+
+ NdbBlocksRecPtr ndbBlocksPtr;
+
+ cdynamicNodeId = 0;
+ cownBlockref = calcNdbCntrBlockRef(getOwnNodeId());
+ cqmgrBlockref = calcQmgrBlockRef(getOwnNodeId());
+ cdictBlockref = calcDictBlockRef(getOwnNodeId());
+ cdihBlockref = calcDihBlockRef(getOwnNodeId());
+ clqhBlockref = calcLqhBlockRef(getOwnNodeId());
+ ctcBlockref = calcTcBlockRef(getOwnNodeId());
+ ccmvmiBlockref = numberToRef(CMVMI, getOwnNodeId());
+
+ ndbBlocksPtr.i = 0;
+ ptrAss(ndbBlocksPtr, ndbBlocksRec);
+ ndbBlocksPtr.p->blockref = clqhBlockref;
+ ndbBlocksPtr.i = 1;
+ ptrAss(ndbBlocksPtr, ndbBlocksRec);
+ ndbBlocksPtr.p->blockref = cdictBlockref;
+ ndbBlocksPtr.i = 2;
+ ptrAss(ndbBlocksPtr, ndbBlocksRec);
+ ndbBlocksPtr.p->blockref = calcTupBlockRef(getOwnNodeId());
+ ndbBlocksPtr.i = 3;
+ ptrAss(ndbBlocksPtr, ndbBlocksRec);
+ ndbBlocksPtr.p->blockref = calcAccBlockRef(getOwnNodeId());
+ ndbBlocksPtr.i = 4;
+ ptrAss(ndbBlocksPtr, ndbBlocksRec);
+ ndbBlocksPtr.p->blockref = ctcBlockref;
+ ndbBlocksPtr.i = 5;
+ ptrAss(ndbBlocksPtr, ndbBlocksRec);
+ ndbBlocksPtr.p->blockref = cdihBlockref;
+ sendSttorry(signal);
+ return;
+}
+
+/*
+4.3 START PHASE 2 */
+/*###########################################################################*/
+// SEND A REGISTATION REQUEST TO QMGR AND WAIT FOR REPLY APPL_REGCONF OR
+// APPL_REGREF COLLECT ALL OTHER NDB NODES
+// AND THEIR STATES FIND OUT WHAT KIND OF START THIS NODE ARE GOING TO PERFORM
+// IF THIS IS A SYSTEM OR INITIAL
+// RESTART THEN FIND OUT WHO IS THE MASTER IF THIS NODE BECOME THE CNTR MASTER
+// THEN COLLECT CNTR_MASTERREQ FROM
+// ALL OTHER REGISTRATED CNTR THE MASTER WILL SEND BACK A CNTR_MASTERCONF WITH
+// FINAL DECISSION ABOUT WHAT TYPE
+// OF START AND WHICH NODES ARE APPROVED TO PARTICIPATE IN THE START IF THE
+// RECEIVER OF CNTR_MASTERREQ HAVE A
+// BETTER CHOICE OF MASTER THEN SEND CNTR_MASTERREF. NEW NODES ARE ALWAYS
+// ALLOWED TO REGISTER, EVEN DURING
+// RESTART BUT THEY WILL BE IGNORED UNTIL THE START HAVE FINISHED.
+// SEND SIGNAL NDBSTTOR TO ALL BLOCKS, ACC, DICT, DIH, LQH, TC AND TUP
+// SEND SIGNAL APPL_REGREQ TO QMGR IN THIS NODE AND WAIT FOR REPLY
+// APPL_REGCONF OR APPL_REGREF */
+/*--------------------------------------------------------------------------*/
+/*******************************/
+/* READ_NODESREF */
+/*******************************/
+void Ndbcntr::execREAD_NODESREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Ndbcntr::execREAD_NODESREF()
+
+/*******************************/
+/* APPL_REGREF */
+/*******************************/
+void Ndbcntr::execAPPL_REGREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Ndbcntr::execAPPL_REGREF()
+
+/*******************************/
+/* CNTR_MASTERREF */
+/*******************************/
+void Ndbcntr::execCNTR_MASTERREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Ndbcntr::execCNTR_MASTERREF()
+
+/*******************************/
+/* NDB_STARTREF */
+/*******************************/
+void Ndbcntr::execNDB_STARTREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Ndbcntr::execNDB_STARTREF()
+
+/*******************************/
+/* STTOR */
+/*******************************/
+void Ndbcntr::startPhase2Lab(Signal* signal)
+{
+ cinternalStartphase = cstartPhase - 1;
+/*--------------------------------------*/
+/* CASE: CSTART_PHASE = ZSTART_PHASE_2 */
+/*--------------------------------------*/
+ cndbBlocksCount = 0;
+ cwaitContinuebFlag = ZFALSE;
+/* NOT WAITING FOR SIGNAL CONTINUEB */
+
+ clastGci = 0;
+ signal->theData[0] = cownBlockref;
+ sendSignal(cdihBlockref, GSN_DIH_RESTARTREQ, signal, 1, JBB);
+ return;
+}//Ndbcntr::startPhase2Lab()
+
+/*******************************/
+/* DIH_RESTARTCONF */
+/*******************************/
+void Ndbcntr::execDIH_RESTARTCONF(Signal* signal)
+{
+ jamEntry();
+ cmasterDihId = signal->theData[0];
+ clastGci = signal->theData[1];
+ ctypeOfStart = NodeState::ST_SYSTEM_RESTART;
+ ph2ALab(signal);
+ return;
+}//Ndbcntr::execDIH_RESTARTCONF()
+
+/*******************************/
+/* DIH_RESTARTREF */
+/*******************************/
+void Ndbcntr::execDIH_RESTARTREF(Signal* signal)
+{
+ jamEntry();
+ ctypeOfStart = NodeState::ST_INITIAL_START;
+ ph2ALab(signal);
+ return;
+}//Ndbcntr::execDIH_RESTARTREF()
+
+void Ndbcntr::ph2ALab(Signal* signal)
+{
+ /******************************/
+ /* request configured nodes */
+ /* from QMGR */
+ /* READ_NODESREQ */
+ /******************************/
+ signal->theData[0] = cownBlockref;
+ sendSignal(cqmgrBlockref, GSN_READ_NODESREQ, signal, 1, JBB);
+ return;
+}//Ndbcntr::ph2ALab()
+
+/*******************************/
+/* READ_NODESCONF */
+/*******************************/
+void Ndbcntr::execREAD_NODESCONF(Signal* signal)
+{
+ jamEntry();
+ ReadNodesConf * const readNodes = (ReadNodesConf *)&signal->theData[0];
+
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if(NodeBitmask::get(readNodes->allNodes, nodePtr.i)){
+ jam();
+ nodePtr.p->nodeDefined = ZTRUE;
+ } else {
+ jam();
+ nodePtr.p->nodeDefined = ZFALSE;
+ }//if
+ }//for
+
+ CfgBlockRecPtr cfgBlockPtr;
+
+ cfgBlockPtr.i = 0;
+ ptrAss(cfgBlockPtr, cfgBlockRec);
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = cfgBlockPtr.i;
+ sendSignal(ccmvmiBlockref, GSN_CMVMI_CFGREQ, signal, 2, JBB);
+ return;
+}
+
+/*******************************/
+/* CMVMI_CFGCONF */
+/*******************************/
+void Ndbcntr::execCMVMI_CFGCONF(Signal* signal)
+{
+ CfgBlockRecPtr cfgBlockPtr;
+ jamEntry();
+
+ CmvmiCfgConf * const cfgConf = (CmvmiCfgConf *)&signal->theData[0];
+
+ cfgBlockPtr.i = cfgConf->startPhase;
+ ptrCheckGuard(cfgBlockPtr, ZSIZE_CFG_BLOCK_REC, cfgBlockRec);
+ for(unsigned int i = 0; i<CmvmiCfgConf::NO_OF_WORDS; i++)
+ cfgBlockPtr.p->cfgData[i] = cfgConf->theData[i];
+
+ if (cfgBlockPtr.i < 4) {
+ jam();
+ cfgBlockPtr.i = cfgBlockPtr.i + 1;
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = cfgBlockPtr.i;
+ sendSignal(ccmvmiBlockref, GSN_CMVMI_CFGREQ, signal, 2, JBB);
+ return;
+ }
+
+ jam();
+
+ cfgBlockPtr.i = 0;
+ ptrAss(cfgBlockPtr, cfgBlockRec);
+
+ cdelayStart = cfgBlockPtr.p->cfgData[0];
+
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = strlen(ZNAME_OF_APPL) | (ZNAME_OF_APPL[0] << 8);
+ signal->theData[2] = ZNAME_OF_APPL[1] | (ZNAME_OF_APPL[2] << 8);
+ signal->theData[9] = ZAPPL_SUBTYPE;
+ signal->theData[10] = 0; //NDB_VERSION;
+ sendSignal(cqmgrBlockref, GSN_APPL_REGREQ, signal, 11, JBB);
+ return; /* WAIT FOR APPL_REGCONF */
+}//Ndbcntr::execCMVMI_CFGCONF()
+
+/*******************************/
+/* APPL_REGCONF */
+/*******************************/
+void Ndbcntr::execAPPL_REGCONF(Signal* signal)
+{
+ jamEntry();
+ cqmgrConnectionP = signal->theData[0];
+ cnoNdbNodes = signal->theData[1];
+ if(ctypeOfStart == NodeState::ST_INITIAL_START){
+ cmasterCandidateId = signal->theData[2];
+ } else {
+ cmasterCandidateId = ZNIL;
+ }
+
+ nodePtr.i = getOwnNodeId();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+
+ /*----------------------------------------------------------------------*/
+ /* CALCULATE HOW MANY NODES THAT WE NEED TO PERFORM A START. MAKE A */
+ /* DECISION ABOUT WAITING FOR MORE NODES OR TO CONTINUE AT ONCE */
+ /*----------------------------------------------------------------------*/
+ nodePtr.p->state = ZADD;
+ nodePtr.p->ndbVersion = 0; //NDB_VERSION;
+ nodePtr.p->subType = ZAPPL_SUBTYPE;
+ nodePtr.p->dynamicId = signal->theData[3];
+ // Save dynamic nodeid in global variable
+ cdynamicNodeId = nodePtr.p->dynamicId;
+ cnoRegNodes = cnoRegNodes + 1;
+ switch((NodeState::StartType)ctypeOfStart){
+ case NodeState::ST_INITIAL_START:
+ jam();
+ cnoNeedNodes = cnoNdbNodes;
+ break;
+ case NodeState::ST_SYSTEM_RESTART:
+ if (cnoNdbNodes == 2) {
+ jam();
+ /*--------------------------------------*/
+ /* NEED > 50% OF ALL NODES. */
+ /* WE WILL SEND CONTINUEB WHEN THE WE */
+ /* RECEIVE THE FIRST APPL_CHANGEREP. */
+ /*--------------------------------------*/
+ cnoNeedNodes = 1; /* IF ONLY 2 NODES IN CLUSTER, 1 WILL DO*/
+ } else {
+ jam();
+ cnoNeedNodes = (cnoNdbNodes >> 1) + 1;
+ }//if
+ break;
+ case NodeState::ST_NODE_RESTART:
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ break;
+ default:
+ ndbrequire(false);
+ }//if
+
+ /*--------------------------------------------------------------*/
+ /* WE CAN COME HERE ALSO IN A NODE RESTART IF THE */
+ /* REGISTRATION OF A RUNNING NODE HAPPENS TO ARRIVE BEFORE*/
+ /* THE APPL_REGCONF SIGNAL. */
+ /* IN THAT CASE CNO_NEED_NODES = ZNIL IF NOT NODE_STATE */
+ /* SIGNAL HAS RETURNED THE PROPER VALUE. IN BOTH CASES WE */
+ /* DO NOT NEED TO ASSIGN IT HERE. */
+ /*--------------------------------------------------------------*/
+ ph2CLab(signal);
+ return;
+}//Ndbcntr::execAPPL_REGCONF()
+
+/*--------------------------------------------------------------*/
+/* CHECK THAT WE GOT ALL NODES REGISTRATED AS WE NEED FOR THIS */
+/* KIND OF START. WE ALWAYS END UP HERE AFTER HANDLING OF */
+/* APPL_CHANGEREP AND NODE_STATESCONF */
+/*--------------------------------------------------------------*/
+void Ndbcntr::ph2CLab(Signal* signal)
+{
+ NodeRecPtr ownNodePtr;
+ ownNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(ownNodePtr, MAX_NDB_NODES, nodeRec);
+ if (ownNodePtr.p->state != ZADD) {
+ jam();
+ return;
+ }//if
+ switch (ctypeOfStart) {
+ case NodeState::ST_INITIAL_START:
+ jam();
+ if (cnoRegNodes == cnoNeedNodes) {
+ jam();
+ ph2ELab(signal);
+/*******************************/
+/* ALL NODES ADDED */
+/*******************************/
+ return;
+ }//if
+ break;
+ case NodeState::ST_SYSTEM_RESTART:
+ ndbrequire(cnoRunNodes == 0);
+ if (cnoRegNodes == cnoNdbNodes) {
+ jam();
+ /*******************************/
+ /* ALL NODES ADDED */
+ /*******************************/
+ ph2ELab(signal);
+ return;
+ }//if
+ if (cwaitContinuebFlag == ZFALSE) {
+ if (cnoRegNodes == cnoNeedNodes) {
+ jam();
+ /****************************************/
+ /* ENOUGH NODES ADDED, WAIT CDELAY_START*/
+ /****************************************/
+ cwaitContinuebFlag = ZTRUE;
+ /*******************************/
+ /* A DELAY SIGNAL TO MYSELF */
+ /*******************************/
+ signal->theData[0] = ZCONTINUEB_1;
+ sendSignalWithDelay(cownBlockref, GSN_CONTINUEB,
+ signal, cdelayStart * 1000, 1);
+ return;
+ }//if
+ }//if
+ break;
+ case NodeState::ST_NODE_RESTART:
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ jam();
+ if (cnoNeedNodes <= cnoRunNodes) {
+ /*----------------------------------------------*/
+ /* GOT ALL RUNNING NODES */
+ /* " =< " :NODES MAY HAVE FINISHED A NODERESTART*/
+ /* WHILE WE WERE WAITING FOR NODE_STATESCONF */
+ /*----------------------------------------------*/
+ if (cnoRegNodes != (cnoRunNodes + 1)) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ getStartNodes(signal);
+ cwaitContinuebFlag = ZFALSE;
+ cstartProgressFlag = ZTRUE;
+ /*--------------------------------------------------------------*/
+ /* IF SOMEONE ELSE IS PERFORMING NODERESTART THEN WE GOT A REF */
+ /* AND WE HAVE TO MAKE A NEW NODE_STATESREQ */
+ /*--------------------------------------------------------------*/
+ sendCntrMasterreq(signal);
+ }//if
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ /*--------------------------------------------------------------*/
+ /* WAIT FOR THE CONTINUEB SIGNAL */
+ /* AND / OR MORE NODES TO REGISTER */
+ /*--------------------------------------------------------------*/
+ return;
+}//Ndbcntr::ph2CLab()
+
+/*******************************/
+/* CONTINUEB */
+/*******************************/
+/*--------------------------------------------------------------*/
+/* WE COME HERE ONLY IN SYSTEM RESTARTS AND INITIAL START. FOR */
+/* INITIAL START WE HAVE ALREADY CALCULATED THE MASTER. FOR */
+/* SYSTEM RESTART WE NEED TO PERFORM A VOTING SCHEME TO AGREE */
+/* ON A COMMON MASTER. WE GET OUR VOTE FROM DIH AND THE RESTART */
+/* INFORMATION IN DIH. */
+/*--------------------------------------------------------------*/
+void Ndbcntr::ph2ELab(Signal* signal)
+{
+ cwaitContinuebFlag = ZFALSE;
+/*--------------------------------------*/
+/* JMP TO THIS WHEN ENOUGH NO OF */
+/* NODES ADDED */
+/*--------------------------------------*/
+/*--------------------------------------*/
+/* IGNORE CONTINUEB SIGNAL */
+/* CONTINUEB SIGNALS WILL EXIT AT */
+/* SIGNAL RECEPTION */
+/*--------------------------------------*/
+ if (cnoRegNodes >= cnoNeedNodes) {
+ jam();
+ getStartNodes(signal);
+ if (ctypeOfStart == NodeState::ST_INITIAL_START) {
+ if (cmasterCandidateId != getOwnNodeId()) {
+ jam();
+/*--------------------------------------*/
+/* THIS NODE IS NOT THE MASTER */
+/* DON'T SEND ANY MORE CNTR_MASTERREQ */
+/* VOTE FOR MASTER */
+/*--------------------------------------*/
+ cstartProgressFlag = ZTRUE;
+ sendCntrMasterreq(signal);
+ resetStartVariables(signal);
+ } else {
+ jam();
+ masterreq020Lab(signal);
+ }//if
+ } else if (ctypeOfStart == NodeState::ST_SYSTEM_RESTART) {
+ jam();
+/*--------------------------------------------------------------*/
+/* WE START THE SELECTION OF MASTER PROCESS. IF WE HAVE NOT */
+/* COMPLETED THIS BEFORE THE TIME OUT WE WILL TRY A NEW RESTART.*/
+/*--------------------------------------------------------------*/
+ cwaitContinuebFlag = ZTRUE;
+ cstartProgressFlag = ZVOTING;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->state == ZADD) {
+ jam();
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = cmasterDihId;
+ signal->theData[2] = clastGci;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_VOTE_MASTERORD,
+ signal, 3, JBB);
+ }//if
+ }//for
+ } else {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ } else {
+ jam();
+/*--------------------------------------------------------------*/
+/* TOO FEW NODES TO START */
+/* WE HAVE WAITED FOR THE GIVEN TIME OUT AND NOT ENOUGH NODES */
+/* HAS REGISTERED. WE WILL CRASH AND RENEW THE ATTEMPT TO START */
+/* THE SYSTEM. */
+/*--------------------------------------------------------------*/
+ systemErrorLab(signal);
+ }//if
+ return;
+}//Ndbcntr::ph2ELab()
+
+/*******************************/
+/* MASTER NODE CONFIRMS REQ */
+/* CNTR_MASTERCONF */
+/*******************************/
+void Ndbcntr::execCNTR_MASTERCONF(Signal* signal)
+{
+ jamEntry();
+
+ CntrMasterConf * const cntrMasterConf =
+ (CntrMasterConf *)&signal->theData[0];
+
+ cnoStartNodes = cntrMasterConf->noStartNodes;
+ int index = 0;
+ unsigned i;
+ for (i = 1; i < MAX_NDB_NODES; i++) {
+ jam();
+ if (NodeBitmask::get(cntrMasterConf->theNodes, i)) {
+ jam();
+ cstartNodes[index] = i;
+ index++;
+ }//if
+ }//for
+ if (cnoStartNodes != index) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ ph2FLab(signal);
+ return;
+}//Ndbcntr::execCNTR_MASTERCONF()
+
+void Ndbcntr::ph2FLab(Signal* signal)
+{
+/*--------------------------------------------------------------*/
+//The nodes have been selected and we now know which nodes are
+// included in the system restart. We can reset wait for CONTINUEB
+// flag to ensure system is not restarted when CONTINUEB after the
+// delay.
+/*--------------------------------------------------------------*/
+ cmasterNodeId = cmasterCandidateId;
+ cwaitContinuebFlag = ZFALSE;
+ ph2GLab(signal);
+ return;
+}//Ndbcntr::ph2FLab()
+
+/*--------------------------------------*/
+/* RECEIVED CNTR_MASTERCONF */
+/*--------------------------------------*/
+/*******************************/
+/* NDB_STTORRY */
+/*******************************/
+/*---------------------------------------------------------------------------*/
+// NOW WE CAN START NDB START PHASE 1. IN THIS PHASE ALL BLOCKS
+// (EXCEPT DIH THAT INITIALISED WHEN
+// RECEIVING DIH_RESTARTREQ) WILL INITIALISE THEIR DATA, COMMON VARIABLES,
+// LINKED LISTS AND RECORD VARIABLES.
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::ph2GLab(Signal* signal)
+{
+ if (cndbBlocksCount < ZNO_NDB_BLOCKS) {
+ jam();
+ sendNdbSttor(signal);
+ return;
+ }//if
+ sendSttorry(signal);
+ return;
+}//Ndbcntr::ph2GLab()
+
+/*
+4.4 START PHASE 3 */
+/*###########################################################################*/
+// SEND SIGNAL NDBSTTOR TO ALL BLOCKS, ACC, DICT, DIH, LQH, TC AND TUP
+// WHEN ALL BLOCKS HAVE RETURNED THEIR NDB_STTORRY ALL BLOCK HAVE FINISHED
+// THEIR LOCAL CONNECTIONs SUCESSFULLY
+// AND THEN WE CAN SEND APPL_STARTREG TO INFORM QMGR THAT WE ARE READY TO
+// SET UP DISTRIBUTED CONNECTIONS.
+/*--------------------------------------------------------------*/
+// THIS IS NDB START PHASE 3.
+/*--------------------------------------------------------------*/
+/*******************************/
+/* STTOR */
+/*******************************/
+void Ndbcntr::startPhase3Lab(Signal* signal)
+{
+ cinternalStartphase = cstartPhase - 1;
+/*--------------------------------------*/
+/* CASE: CSTART_PHASE = ZSTART_PHASE_3 */
+/*--------------------------------------*/
+ cndbBlocksCount = 0;
+ ph3ALab(signal);
+ return;
+}//Ndbcntr::startPhase3Lab()
+
+/*******************************/
+/* NDB_STTORRY */
+/*******************************/
+void Ndbcntr::ph3ALab(Signal* signal)
+{
+ Uint16 tnoStartNodes;
+
+ if (cndbBlocksCount < ZNO_NDB_BLOCKS) {
+ jam();
+ sendNdbSttor(signal);
+ return;
+ }//if
+/*******************************/
+/*< APPL_STARTREG <*/
+/*******************************/
+ if (ctypeOfStart == NodeState::ST_NODE_RESTART) {
+ jam();
+ tnoStartNodes = 1;
+ } else if (ctypeOfStart == NodeState::ST_INITIAL_NODE_RESTART) {
+ jam();
+ tnoStartNodes = 1;
+ } else {
+ jam();
+ tnoStartNodes = cnoStartNodes;
+ }//if
+ signal->theData[0] = cqmgrConnectionP;
+ signal->theData[1] = tnoStartNodes;
+ sendSignal(cqmgrBlockref, GSN_APPL_STARTREG, signal, 2, JBB);
+ sendSttorry(signal);
+ return;
+}//Ndbcntr::ph3ALab()
+
+/*
+4.5 START PHASE 4 */
+/*###########################################################################*/
+// WAIT FOR ALL NODES IN CLUSTER TO CHANGE STATE INTO ZSTART ,
+// APPL_CHANGEREP IS ALWAYS SENT WHEN SOMEONE HAVE
+// CHANGED THEIR STATE. APPL_STARTCONF INDICATES THAT ALL NODES ARE IN START
+// STATE SEND NDB_STARTREQ TO DIH AND THEN WAIT FOR NDB_STARTCONF
+/*---------------------------------------------------------------------------*/
+/*******************************/
+/* STTOR */
+/*******************************/
+void Ndbcntr::startPhase4Lab(Signal* signal)
+{
+ cinternalStartphase = cstartPhase - 1;
+/*--------------------------------------*/
+/* CASE: CSTART_PHASE = ZSTART_PHASE_4 */
+/*--------------------------------------*/
+ cndbBlocksCount = 0;
+ cnoWaitrep = 0;
+ if (capplStartconfFlag != ZTRUE) {
+ jam();
+/*------------------------------------------------------*/
+/* HAVE WE ALREADY RECEIVED APPL_STARTCONF */
+/*------------------------------------------------------*/
+ return;
+ }//if
+ ph4ALab(signal);
+ return;
+}//Ndbcntr::startPhase4Lab()
+
+/*******************************/
+/* APPL_STARTCONF */
+/*******************************/
+void Ndbcntr::execAPPL_STARTCONF(Signal* signal)
+{
+ jamEntry();
+ if (cstartPhase == ZSTART_PHASE_4) {
+ jam();
+ ph4ALab(signal);
+ return;
+ } else {
+ jam();
+ capplStartconfFlag = ZTRUE;
+//------------------------------------------------
+/* FLAG WILL BE CHECKED WHEN WE RECEIVED STTOR */
+/* SIGNAL MAY BE RECEIVED IN STARTPHASE 3 */
+//------------------------------------------------
+ return;
+ }//if
+}//Ndbcntr::execAPPL_STARTCONF()
+
+void Ndbcntr::ph4ALab(Signal* signal)
+{
+ nodePtr.i = getOwnNodeId();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ nodePtr.p->state = ZSTART;
+ ph4BLab(signal);
+ return;
+}//Ndbcntr::ph4ALab()
+
+/*******************************/
+/* NDB_STTORRY */
+/*******************************/
+void Ndbcntr::ph4BLab(Signal* signal)
+{
+/*--------------------------------------*/
+/* CASE: CSTART_PHASE = ZSTART_PHASE_4 */
+/*--------------------------------------*/
+ if (cndbBlocksCount < ZNO_NDB_BLOCKS) {
+ jam();
+ sendNdbSttor(signal);
+ return;
+ }//if
+ if ((ctypeOfStart == NodeState::ST_NODE_RESTART) ||
+ (ctypeOfStart == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+ sendSttorry(signal);
+ return;
+ }//if
+ waitpoint41Lab(signal);
+ return;
+}//Ndbcntr::ph4BLab()
+
+void Ndbcntr::waitpoint41Lab(Signal* signal)
+{
+ if (getOwnNodeId() == cmasterNodeId) {
+ jam();
+/*--------------------------------------*/
+/* MASTER WAITS UNTIL ALL SLAVES HAS */
+/* SENT THE REPORTS */
+/*--------------------------------------*/
+ cnoWaitrep++;
+ if (cnoWaitrep == cnoStartNodes) {
+ jam();
+/*---------------------------------------------------------------------------*/
+// NDB_STARTREQ STARTS UP ALL SET UP OF DISTRIBUTION INFORMATION IN DIH AND
+// DICT. AFTER SETTING UP THIS
+// DATA IT USES THAT DATA TO SET UP WHICH FRAGMENTS THAT ARE TO START AND
+// WHERE THEY ARE TO START. THEN
+// IT SETS UP THE FRAGMENTS AND RECOVERS THEM BY:
+// 1) READING A LOCAL CHECKPOINT FROM DISK.
+// 2) EXECUTING THE UNDO LOG ON INDEX AND DATA.
+// 3) EXECUTING THE FRAGMENT REDO LOG FROM ONE OR SEVERAL NODES TO
+// RESTORE THE RESTART CONFIGURATION OF DATA IN NDB CLUSTER.
+/*---------------------------------------------------------------------------*/
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = ctypeOfStart;
+ sendSignal(cdihBlockref, GSN_NDB_STARTREQ, signal, 2, JBB);
+ }//if
+ } else {
+ jam();
+/*--------------------------------------*/
+/* SLAVE NODES WILL PASS HERE ONCE AND */
+/* SEND A WAITPOINT REPORT TO MASTER. */
+/* SLAVES WONT DO ANYTHING UNTIL THEY */
+/* RECEIVE A WAIT REPORT FROM THE MASTER*/
+/*--------------------------------------*/
+ nodePtr.i = cmasterNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = ZWAITPOINT_4_1;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_WAITREP, signal, 2, JBB);
+ }//if
+ return;
+}//Ndbcntr::waitpoint41Lab()
+
+/*******************************/
+/* NDB_STARTCONF */
+/*******************************/
+void Ndbcntr::execNDB_STARTCONF(Signal* signal)
+{
+ jamEntry();
+ UintR guard0;
+ UintR Ttemp1;
+
+ guard0 = cnoStartNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (Ttemp1 = 0; Ttemp1 <= guard0; Ttemp1++) {
+ jam();
+ if (cstartNodes[Ttemp1] != getOwnNodeId()) {
+ jam();
+ nodePtr.i = cstartNodes[Ttemp1];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = ZWAITPOINT_4_2;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_WAITREP, signal, 2, JBB);
+ }//if
+ }//for
+ sendSttorry(signal);
+ return;
+}//Ndbcntr::execNDB_STARTCONF()
+
+/*
+4.6 START PHASE 5 */
+/*###########################################################################*/
+// SEND APPL_RUN TO THE QMGR IN THIS BLOCK
+// SEND NDB_STTOR ALL BLOCKS ACC, DICT, DIH, LQH, TC AND TUP THEN WAIT FOR
+// THEIR NDB_STTORRY
+/*---------------------------------------------------------------------------*/
+/*******************************/
+/* STTOR */
+/*******************************/
+void Ndbcntr::startPhase5Lab(Signal* signal)
+{
+ cinternalStartphase = cstartPhase - 1;
+ cndbBlocksCount = 0;
+ cnoWaitrep = 0;
+ ph5ALab(signal);
+ return;
+}//Ndbcntr::startPhase5Lab()
+
+/*******************************/
+/* NDB_STTORRY */
+/*******************************/
+/*---------------------------------------------------------------------------*/
+// THIS IS NDB START PHASE 5.
+/*---------------------------------------------------------------------------*/
+// IN THIS START PHASE TUP INITIALISES DISK FILES FOR DISK STORAGE IF INITIAL
+// START. DIH WILL START UP
+// THE GLOBAL CHECKPOINT PROTOCOL AND WILL CONCLUDE ANY UNFINISHED TAKE OVERS
+// THAT STARTED BEFORE THE SYSTEM CRASH.
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::ph5ALab(Signal* signal)
+{
+ if (cndbBlocksCount < ZNO_NDB_BLOCKS) {
+ jam();
+ sendNdbSttor(signal);
+ return;
+ }//if
+
+ cstartPhase = cstartPhase + 1;
+ cinternalStartphase = cstartPhase - 1;
+ if (getOwnNodeId() == cmasterNodeId) {
+ switch(ctypeOfStart){
+ case NodeState::ST_INITIAL_START:
+ jam();
+ /*--------------------------------------*/
+ /* MASTER CNTR IS RESPONSIBLE FOR */
+ /* CREATING SYSTEM TABLES */
+ /*--------------------------------------*/
+ createSystableLab(signal, 0);
+ return;
+ case NodeState::ST_SYSTEM_RESTART:
+ jam();
+ waitpoint52Lab(signal);
+ return;
+ case NodeState::ST_NODE_RESTART:
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ break;
+ case NodeState::ST_ILLEGAL_TYPE:
+ break;
+ }
+ ndbrequire(false);
+ }
+
+ /**
+ * Not master
+ */
+ NdbSttor * const req = (NdbSttor*)signal->getDataPtrSend();
+ switch(ctypeOfStart){
+ case NodeState::ST_NODE_RESTART:
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ jam();
+ /*----------------------------------------------------------------------*/
+ // SEND NDB START PHASE 5 IN NODE RESTARTS TO COPY DATA TO THE NEWLY
+ // STARTED NODE.
+ /*----------------------------------------------------------------------*/
+ req->senderRef = cownBlockref;
+ req->nodeId = getOwnNodeId();
+ req->internalStartPhase = cinternalStartphase;
+ req->typeOfStart = ctypeOfStart;
+ req->masterNodeId = cmasterNodeId;
+
+#ifdef TRACE_STTOR
+ ndbout_c("sending NDB_STTOR(%d) to DIH", cinternalStartphase);
+#endif
+ sendSignal(cdihBlockref, GSN_NDB_STTOR, signal,
+ NdbSttor::SignalLength, JBB);
+ return;
+ case NodeState::ST_INITIAL_START:
+ case NodeState::ST_SYSTEM_RESTART:
+ jam();
+ /*--------------------------------------*/
+ /* DURING SYSTEMRESTART AND INITALSTART:*/
+ /* SLAVE NODES WILL PASS HERE ONCE AND */
+ /* SEND A WAITPOINT REPORT TO MASTER. */
+ /* SLAVES WONT DO ANYTHING UNTIL THEY */
+ /* RECEIVE A WAIT REPORT FROM THE MASTER*/
+ /* WHEN THE MASTER HAS FINISHED HIS WORK*/
+ /*--------------------------------------*/
+ nodePtr.i = cmasterNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = ZWAITPOINT_5_2;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_WAITREP, signal, 2, JBB);
+ return;
+ default:
+ ndbrequire(false);
+ }
+}//Ndbcntr::ph5ALab()
+
+void Ndbcntr::waitpoint52Lab(Signal* signal)
+{
+ cnoWaitrep = cnoWaitrep + 1;
+/*---------------------------------------------------------------------------*/
+// THIS WAITING POINT IS ONLY USED BY A MASTER NODE. WE WILL EXECUTE NDB START
+// PHASE 5 FOR DIH IN THE
+// MASTER. THIS WILL START UP LOCAL CHECKPOINTS AND WILL ALSO CONCLUDE ANY
+// UNFINISHED LOCAL CHECKPOINTS
+// BEFORE THE SYSTEM CRASH. THIS WILL ENSURE THAT WE ALWAYS RESTART FROM A
+// WELL KNOWN STATE.
+/*---------------------------------------------------------------------------*/
+/*--------------------------------------*/
+/* MASTER WAITS UNTIL HE RECEIVED WAIT */
+/* REPORTS FROM ALL SLAVE CNTR */
+/*--------------------------------------*/
+ if (cnoWaitrep == cnoStartNodes) {
+ jam();
+ NdbSttor * const req = (NdbSttor*)signal->getDataPtrSend();
+ req->senderRef = cownBlockref;
+ req->nodeId = getOwnNodeId();
+ req->internalStartPhase = cinternalStartphase;
+ req->typeOfStart = ctypeOfStart;
+ req->masterNodeId = cmasterNodeId;
+#ifdef TRACE_STTOR
+ ndbout_c("sending NDB_STTOR(%d) to DIH", cinternalStartphase);
+#endif
+ sendSignal(cdihBlockref, GSN_NDB_STTOR, signal,
+ NdbSttor::SignalLength, JBB);
+ }//if
+ return;
+}//Ndbcntr::waitpoint52Lab()
+
+/*******************************/
+/* NDB_STTORRY */
+/*******************************/
+void Ndbcntr::ph6ALab(Signal* signal)
+{
+ UintR guard0;
+ UintR Ttemp1;
+
+ if ((ctypeOfStart == NodeState::ST_NODE_RESTART) ||
+ (ctypeOfStart == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+ waitpoint51Lab(signal);
+ return;
+ }//if
+ guard0 = cnoStartNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (Ttemp1 = 0; Ttemp1 <= guard0; Ttemp1++) {
+ jam();
+ if (cstartNodes[Ttemp1] != getOwnNodeId()) {
+ jam();
+ nodePtr.i = cstartNodes[Ttemp1];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = ZWAITPOINT_5_1;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_WAITREP, signal, 2, JBB);
+ }//if
+ }//for
+ waitpoint51Lab(signal);
+ return;
+}//Ndbcntr::ph6ALab()
+
+void Ndbcntr::waitpoint51Lab(Signal* signal)
+{
+ cstartPhase = cstartPhase + 1;
+/*---------------------------------------------------------------------------*/
+// A FINAL STEP IS NOW TO SEND NDB_STTOR TO TC. THIS MAKES IT POSSIBLE TO
+// CONNECT TO TC FOR APPLICATIONS.
+// THIS IS NDB START PHASE 6 WHICH IS FOR ALL BLOCKS IN ALL NODES.
+/*---------------------------------------------------------------------------*/
+ cinternalStartphase = cstartPhase - 1;
+ cndbBlocksCount = 0;
+ ph6BLab(signal);
+ return;
+}//Ndbcntr::waitpoint51Lab()
+
+void Ndbcntr::ph6BLab(Signal* signal)
+{
+ // c_missra.currentStartPhase - cstartPhase - cinternalStartphase =
+ // 5 - 7 - 6
+ if (cndbBlocksCount < ZNO_NDB_BLOCKS) {
+ jam();
+ sendNdbSttor(signal);
+ return;
+ }//if
+ if ((ctypeOfStart == NodeState::ST_NODE_RESTART) ||
+ (ctypeOfStart == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+ sendSttorry(signal);
+ return;
+ }
+ waitpoint61Lab(signal);
+}
+
+void Ndbcntr::waitpoint61Lab(Signal* signal)
+{
+ if (getOwnNodeId() == cmasterNodeId) {
+ jam();
+ cnoWaitrep6++;
+ if (cnoWaitrep6 == cnoStartNodes) {
+ jam();
+ Uint32 guard0 = cnoStartNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (Uint32 Ttemp1 = 0; Ttemp1 <= guard0; Ttemp1++) {
+ jam();
+ if (cstartNodes[Ttemp1] != getOwnNodeId()) {
+ jam();
+ nodePtr.i = cstartNodes[Ttemp1];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = ZWAITPOINT_6_2;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_WAITREP, signal, 2, JBB);
+ }
+ }
+ sendSttorry(signal);
+ }
+ } else {
+ jam();
+ nodePtr.i = cmasterNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = ZWAITPOINT_6_1;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_WAITREP, signal, 2, JBB);
+ }
+}
+
+// Start phase 8 (internal 7)
+void Ndbcntr::startPhase8Lab(Signal* signal)
+{
+ cinternalStartphase = cstartPhase - 1;
+ cndbBlocksCount = 0;
+ ph7ALab(signal);
+}
+
+void Ndbcntr::ph7ALab(Signal* signal)
+{
+ while (cndbBlocksCount < ZNO_NDB_BLOCKS) {
+ jam();
+ sendNdbSttor(signal);
+ return;
+ }
+ if ((ctypeOfStart == NodeState::ST_NODE_RESTART) ||
+ (ctypeOfStart == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+ sendSttorry(signal);
+ return;
+ }
+ waitpoint71Lab(signal);
+}
+
+void Ndbcntr::waitpoint71Lab(Signal* signal)
+{
+ if (getOwnNodeId() == cmasterNodeId) {
+ jam();
+ cnoWaitrep7++;
+ if (cnoWaitrep7 == cnoStartNodes) {
+ jam();
+ Uint32 guard0 = cnoStartNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (Uint32 Ttemp1 = 0; Ttemp1 <= guard0; Ttemp1++) {
+ jam();
+ if (cstartNodes[Ttemp1] != getOwnNodeId()) {
+ jam();
+ nodePtr.i = cstartNodes[Ttemp1];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = ZWAITPOINT_7_2;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_WAITREP, signal, 2, JBB);
+ }
+ }
+ sendSttorry(signal);
+ }
+ } else {
+ jam();
+ nodePtr.i = cmasterNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = ZWAITPOINT_7_1;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_WAITREP, signal, 2, JBB);
+ }
+}
+
+// Start phase 9 (internal 8)
+void Ndbcntr::startPhase9Lab(Signal* signal)
+{
+ cinternalStartphase = cstartPhase - 1;
+ cndbBlocksCount = 0;
+ ph8ALab(signal);
+}
+
+void Ndbcntr::ph8ALab(Signal* signal)
+{
+/*---------------------------------------------------------------------------*/
+// NODES WHICH PERFORM A NODE RESTART NEEDS TO GET THE DYNAMIC ID'S
+// OF THE OTHER NODES HERE.
+/*---------------------------------------------------------------------------*/
+ signal->theData[0] = cqmgrConnectionP;
+ sendSignal(cqmgrBlockref, GSN_APPL_RUN, signal, 1, JBB);
+ nodePtr.i = getOwnNodeId();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ nodePtr.p->state = ZRUN;
+ cnoRunNodes = cnoRunNodes + 1;
+ sendSttorry(signal);
+ cstartProgressFlag = ZFALSE;
+ ctypeOfStart = (NodeState::StartType)ZSYSTEM_RUN;
+ resetStartVariables(signal);
+ return;
+}//Ndbcntr::ph8BLab()
+
+/*
+4.7 HANDLE GLOBAL EVENTS, NOT BOUNDED TO INITIALSTART OR SYSTEM RESTART */
+/*#######################################################################*/
+/*******************************/
+/* APPL_CHANGEREP */
+/*******************************/
+void Ndbcntr::execAPPL_CHANGEREP(Signal* signal)
+{
+ jamEntry();
+ Uint16 TapplEvent = signal->theData[0];
+ Uint16 TapplVersion = signal->theData[1];
+ Uint16 TapplNodeId = signal->theData[2];
+ Uint16 TapplSubType = signal->theData[3];
+
+ nodePtr.i = TapplNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ nodePtr.p->subType = TapplSubType;
+ nodePtr.p->ndbVersion = TapplVersion;
+ nodePtr.p->dynamicId = signal->theData[4];
+
+ switch (TapplEvent) {
+ case ZADD:
+/*----------------------------*/
+/* ADD A NEW NDB NODE TO FILE */
+/*----------------------------*/
+ if (nodePtr.p->state == ZREMOVE) {
+ jam();
+ if (cnoRegNodes == cnoNdbNodes) {
+ jam();
+/*----------------------------------------------*/
+/* DON'T ACCEPT MORE NODES THAN SYSFILE.CFG SPEC*/
+/*----------------------------------------------*/
+ systemErrorLab(signal);
+ return;
+ }//if
+ nodePtr.p->state = ZADD;
+ cnoRegNodes = cnoRegNodes + 1;
+ } else {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ if (cstartProgressFlag == ZFALSE) {
+/*----------------------------------------------*/
+/* FLAG = TRUE WHEN CNTR_MASTERREQ IS SENT */
+/*----------------------------------------------*/
+ switch (ctypeOfStart) {
+ case NodeState::ST_INITIAL_START:
+ case NodeState::ST_SYSTEM_RESTART:
+ jam();
+ ph2CLab(signal);
+/*----------------------------------------------*/
+/* CHECK IF READY TO MAKE A CNTR_MASTERREQ */
+/*----------------------------------------------*/
+ break;
+ case NodeState::ST_NODE_RESTART:
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ jam();
+/*------------------------------------------------------------------------*/
+/* THIS SHOULD NEVER OCCUR SINCE WE HAVE ALREADY BEEN ALLOWED TO */
+/* START OUR NODE. THE NEXT NODE CANNOT START UNTIL WE ARE FINISHED */
+/*------------------------------------------------------------------------*/
+ systemErrorLab(signal);
+ break;
+ case ZSYSTEM_RUN:
+ jam();
+ /*empty*/;
+ break;
+ default:
+ jam();
+/*------------------------------------------------------------------------*/
+/* NO PARTICULAR ACTION IS NEEDED. THE NODE WILL PERFORM A NODE */
+/* RESTART BUT NO ACTION IS NEEDED AT THIS STAGE IN THE RESTART. */
+/*------------------------------------------------------------------------*/
+ systemErrorLab(signal);
+ break;
+ }//switch
+ } else {
+ jam();
+/*--------------------------------------------------------------------------*/
+// WHEN A RESTART IS IN PROGRESS THERE IS A POSSIBILITY THAT A NODE
+// REGISTER AND
+// THINKS THAT HE WOULD BE THE MASTER (LOWER NODE ID) BUT THE OTHER NODE IS
+// ALREADY RUNNING THE RESTART. THIS WILL BE DETECTED WHEN HE ATTEMPTS A
+// CNTR_MASTERREQ AND RECEIVES A REFUSE SIGNAL IN RETURN. THIS WILL CAUSE HIM
+// TO CRASH. IF HE ATTEMPTS TO JOIN AS A NON-MASTER HE WILL WAIT FOR THE MASTER.
+// IN THIS CASE IT IS BETTER TO SHOT HIM DOWN. FOR SAFETY REASONS WE WILL ALWAYS
+// SHOT HIM DOWN.
+/*--------------------------------------------------------------------------*/
+ const BlockReference tblockref = calcNdbCntrBlockRef(nodePtr.i);
+
+ SystemError * const sysErr = (SystemError*)&signal->theData[0];
+ sysErr->errorCode = SystemError::StartInProgressError;
+ sysErr->errorRef = reference();
+ sendSignal(tblockref, GSN_SYSTEM_ERROR, signal, SystemError::SignalLength, JBA);
+ }//if
+ break;
+ case ZSTART:
+ jam();
+ if (nodePtr.p->state != ZADD) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ nodePtr.p->state = ZSTART;
+ break;
+ case ZRUN:
+ if (nodePtr.p->state == ZREMOVE) {
+ jam();
+ cnoRegNodes = cnoRegNodes + 1;
+ } else {
+ jam();
+ if (nodePtr.p->state != ZSTART) {
+ jam();
+/*----------------------------------------------*/
+/* STATE ZADD OR ZRUN -> ZRUN NOT ALLOWED */
+/*----------------------------------------------*/
+ systemErrorLab(signal);
+ return;
+ }//if
+ }//if
+ cnoRunNodes = cnoRunNodes + 1;
+ nodePtr.p->state = ZRUN;
+ switch (ctypeOfStart) {
+ case NodeState::ST_INITIAL_START:
+ jam();
+ detectNoderestart(signal);
+ if (ctypeOfStart == NodeState::ST_NODE_RESTART) {
+ jam();
+/*--------------------------------------------------------------------------*/
+/* WE DISCOVERED THAT WE ARE TRYING TO PERFORM A INITIAL START WHEN THERE */
+/* ARE ALREADY RUNNING NODES. THIS MEANS THAT THE NODE HAS CLEANED THE */
+/* FILE SYSTEM AND CONTAINS NO DATA. THIS IS AN INITIAL NODE RESTART WHICH */
+/* IS NECESSARY TO START A NODE THAT HAS BEEN TAKEN OVER. */
+/*--------------------------------------------------------------------------*/
+ ctypeOfStart = NodeState::ST_INITIAL_NODE_RESTART;
+ }//if
+ break;
+ case NodeState::ST_SYSTEM_RESTART:
+ jam();
+ detectNoderestart(signal);
+/*----------------------------------------------*/
+/* SHOULD THIS NODE PERFORM A NODE RESTART? */
+/* THEN CHANGE CTYPE_OF_START TO NodeState::ST_NODE_RESTART */
+/* AND SEND NODE_STATESREQ. */
+/* WAIT FOR NODE_STATESCONF. */
+/*----------------------------------------------*/
+ break;
+ case NodeState::ST_NODE_RESTART:
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ jam();
+/*----------------------------------------------*/
+/* IF WE ARE WAITING FOR NODE_STATESCONF, THIS */
+/* JUMP WILL EXIT BECAUSE CNO_NEED_NODES = ZNIL */
+/* UNTIL WE RECEIVE NODE_STATESCONF */
+/*----------------------------------------------*/
+ ph2CLab(signal);
+ break;
+ case ZSYSTEM_RUN:
+ jam();
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+ return;
+}//Ndbcntr::execAPPL_CHANGEREP()
+
+/*--------------------------------------------------------------------------*/
+// A NODE HAS ADDED HAS VOTE ON WHICH MASTER IS TO BE CHOOSEN IN A SYSTEM
+// RESTART. WHEN ALL VOTES HAVE
+// BEEN ADDED THEN WE ARE PREPARED TO CHOOSE MASTER AND CONTINUE WITH THE
+// RESTART PROCESSING.
+/*--------------------------------------------------------------------------*/
+
+/*******************************/
+/* VOT_MASTERORD */
+/*******************************/
+void Ndbcntr::execVOTE_MASTERORD(Signal* signal)
+{
+ jamEntry();
+ nodePtr.i = signal->theData[0];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ UintR TmasterCandidateId = signal->theData[1];
+ UintR TlastGci = signal->theData[2];
+ if (ctypeOfStart != NodeState::ST_SYSTEM_RESTART) {
+ jam();
+ progError(__LINE__,
+ ERR_SR_RESTARTCONFLICT,
+ "One ore more nodes probably requested an initial SR");
+ return;
+ }//if
+ cmasterVoters = cmasterVoters + 1;
+ if (cmasterVoters == 1) {
+ jam();
+ cmasterCurrentId = TmasterCandidateId;
+ cmasterLastGci = TlastGci;
+ } else {
+ if (cmasterLastGci < TlastGci) {
+ jam();
+ cmasterCurrentId = TmasterCandidateId;
+ cmasterLastGci = TlastGci;
+ } else if (cmasterLastGci == TlastGci) {
+ jam();
+ if (cmasterCurrentId != TmasterCandidateId) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ }//if
+ }//if
+ if (cstartProgressFlag == ZVOTING) {
+/*--------------------------------------------------------------------------*/
+// UNLESS START PROGRESS FLAG IS SET TO VOTING WE HAVE NOT YET REACHED A
+// STATE WHERE WE ARE READY TO
+// PROCEED WITH THE SYSTEM RESTART. OUR OWN NOTE HAVE AT LEAST NOT BEEN
+// CAST INTO THE BALLOT YET.
+/*--------------------------------------------------------------------------*/
+ if (cmasterVoters == cnoRegNodes) {
+ cmasterCandidateId = cmasterCurrentId;
+ if (cmasterCandidateId == getOwnNodeId()) {
+ jam();
+ masterreq020Lab(signal);
+ return;
+ } else {
+ jam();
+ cstartProgressFlag = ZTRUE;
+ sendCntrMasterreq(signal);
+ resetStartVariables(signal);
+ }//if
+ }//if
+ }//if
+ return;
+}//Ndbcntr::execVOTE_MASTERORD()
+
+/*******************************/
+/* CNTR_MASTERREQ */
+/*******************************/
+void Ndbcntr::execCNTR_MASTERREQ(Signal* signal)
+{
+ Uint16 ttypeOfStart;
+
+ jamEntry();
+
+ CntrMasterReq * const cntrMasterReq =
+ (CntrMasterReq *)&signal->theData[0];
+
+//-----------------------------------------------
+// cntrMasterReq->userBlockRef NOT USED
+//-----------------------------------------------
+ Uint16 TuserNodeId = cntrMasterReq->userNodeId;
+ ttypeOfStart = cntrMasterReq->typeOfStart;
+ Uint16 TnoRestartNodes = cntrMasterReq->noRestartNodes;
+ int index = 0;
+ unsigned i;
+ for (i = 1; i < MAX_NDB_NODES; i++) {
+ jam();
+ if (NodeBitmask::get(cntrMasterReq->theNodes, i)) {
+ jam();
+ cstartNodes[index] = i;
+ index++;
+ }//if
+ }//for
+ if (TnoRestartNodes != index) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ switch (ttypeOfStart) {
+ case NodeState::ST_INITIAL_START:
+ case NodeState::ST_SYSTEM_RESTART:
+ jam();
+//--------------------------------
+/* ELECTION OF MASTER AT */
+/* INITIAL OR SYSTEM RESTART */
+//--------------------------------
+ masterreq010Lab(signal, TnoRestartNodes, TuserNodeId);
+ break;
+ case NodeState::ST_NODE_RESTART:
+ case NodeState::ST_INITIAL_NODE_RESTART:
+ jam();
+ masterreq030Lab(signal, TnoRestartNodes, TuserNodeId);
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+}//Ndbcntr::execCNTR_MASTERREQ()
+
+/*******************************/
+/* CNTR_WAITREP */
+/*******************************/
+void Ndbcntr::execCNTR_WAITREP(Signal* signal)
+{
+ Uint16 twaitPoint;
+
+ jamEntry();
+ twaitPoint = signal->theData[1];
+ switch (twaitPoint) {
+ case ZWAITPOINT_4_1:
+ jam();
+ waitpoint41Lab(signal);
+ break;
+ case ZWAITPOINT_4_2:
+ jam();
+ sendSttorry(signal);
+ break;
+ case ZWAITPOINT_5_1:
+ jam();
+ waitpoint51Lab(signal);
+ break;
+ case ZWAITPOINT_5_2:
+ jam();
+ waitpoint52Lab(signal);
+ break;
+ case ZWAITPOINT_6_1:
+ jam();
+ waitpoint61Lab(signal);
+ break;
+ case ZWAITPOINT_6_2:
+ jam();
+ sendSttorry(signal);
+ break;
+ case ZWAITPOINT_7_1:
+ jam();
+ waitpoint71Lab(signal);
+ break;
+ case ZWAITPOINT_7_2:
+ jam();
+ sendSttorry(signal);
+ break;
+ default:
+ jam();
+ systemErrorLab(signal);
+ break;
+ }//switch
+}//Ndbcntr::execCNTR_WAITREP()
+
+/*
+4.7.4 MASTERREQ_010 ( CASE: INITIALSTART OR SYSTEMRESTART ) */
+/*--------------------------------------------------------------------------*/
+// ELECTION OF MASTER AND ELECTION OF PARTICIPANTS IN START. SENDER OF
+// CNTR_MASTERREQ THINKS THAT THIS NODE
+// SHOULD BE THE MASTER. WE CAN'T MAKE A DECISION ABOUT WHO THE MASTER
+// SHOULD BE UNTIL TIMELIMIT HAS EXPIRED AND
+// THAT AT LEAST CNO_NEED_NODES ARE ZADD IN NODE_PTR_REC. IF THIS NODE IS
+// MASTER THEN MAKE SURE THAT ALL NODES IN
+// THE CLUSTER COMES TO AN AGREEMENT ABOUT A SUBSET OF NODES THAT SATISFIES
+// THE NUMBER CNO_NEED_NODES. THERE IS
+// A POSSIBILITY THAT THE RECEIVER OF CNTR_MASTERREQ DOESN'T HAS CHOOSEN
+// A MASTER, THEN THE RECEIVER CAN'T
+// EITHER CONFIRM OR REFUSE JUST STORE THE VOTES OF THE CLUSTERMEMBERS.
+// IF THIS NODE BECOME AWARE OF THAT ANOTHER NODE IS MASTER THEN CHECK IF
+// ANYONE HAS VOTED (SENT CNTR_MASTERREQ) */
+// AND THEN SEND THEM CNTR_MASTERREF BACK.
+/*--------------------------------------------------------------------------*/
+void Ndbcntr::masterreq010Lab(Signal* signal,
+ Uint16 TnoRestartNodes,
+ Uint16 TuserNodeId)
+{
+ UintR guard0;
+ UintR Ttemp1;
+
+ nodePtr.i = TuserNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ if (cstartProgressFlag == ZTRUE) {
+ jam();
+/*--------------------------------------*/
+/* RESTART ALREADY IN PROGRESS */
+/*--------------------------------------*/
+ if (ctypeOfStart == NodeState::ST_INITIAL_START) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = getOwnNodeId();
+ signal->theData[2] = ZSTART_IN_PROGRESS_ERROR;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_MASTERREF, signal, 3, JBB);
+ return;
+ }//if
+ cnoVoters = cnoVoters + 1;
+ nodePtr.p->voter = ZTRUE;
+ guard0 = TnoRestartNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (Ttemp1 = 0; Ttemp1 <= guard0; Ttemp1++) {
+ jam();
+ nodePtr.i = cstartNodes[Ttemp1];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ nodePtr.p->votes = nodePtr.p->votes + 1;
+ }//for
+ masterreq020Lab(signal);
+ return;
+}//Ndbcntr::masterreq010Lab()
+
+/*----------------------------------------------------------------------*/
+/* WHEN WE JUST WANT TO CHECK OUR VOTES IT IS POSSIBLE TO JUMP TO THIS */
+/* LABEL. IF WE HAVEN'T RECEIVED ANY VOTES SINCE OUR LASTCHECK WE WILL */
+/* JUST PERFORM AN EXIT */
+/*----------------------------------------------------------------------*/
+void Ndbcntr::masterreq020Lab(Signal* signal)
+{
+ if (cmasterCandidateId == ZNIL) {
+ jam();
+/*--------------------------------------*/
+/* MASTER UNKNOWN */
+/*--------------------------------------*/
+ return;
+ } else if (cmasterCandidateId == getOwnNodeId()) {
+ jam();
+/*--------------------------------------*/
+/* SATISFIED WHEN WE HAVE AS MANY VOTERS*/
+/* AS RESTARTNODES - 1, DIFFERENT NODES?*/
+/* <- CNO_START_NODES, ALL NODES AGREED */
+/* ON THESE CNO_START_NODES */
+/*--------------------------------------*/
+ if ((cnoStartNodes - 1) == cnoVoters) {
+ chooseRestartNodes(signal);
+ if (cnoStartNodes >= cnoNeedNodes) {
+ jam();
+ cstartProgressFlag = ZTRUE;
+/*--------------------------------------*/
+/* DON'T SEND ANY MORE CNTR_MASTERREQ */
+/*--------------------------------------*/
+ replyMasterconfToAll(signal);
+/*--------------------------------------*/
+/* SEND CONF TO ALL PASSED REQ */
+/* DON'T SEND ANYTHING TO REJECTED NODES*/
+/* BLOCK THEM UNTIL SYSTEM IS RUNNING */
+/* CONTINUE RESTART */
+/*--------------------------------------*/
+ ph2FLab(signal);
+ } else {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ }//if
+ } else {
+ jam();
+/*----------------------------------------------------------------------*/
+/* WE RECEIVED A REQUEST TO A MASTER WHILE NOT BEING MASTER. THIS */
+/* MUST BE AN ERROR INDICATION. WE CRASH. */
+/*----------------------------------------------------------------------*/
+ systemErrorLab(signal);
+ }//if
+ return; /* WAIT FOR MORE CNTR_MASTERREQ */
+}//Ndbcntr::masterreq020Lab()
+
+void Ndbcntr::masterreq030Lab(Signal* signal,
+ Uint16 TnoRestartNodes,
+ Uint16 TuserNodeId)
+{
+ UintR TretCode;
+ if (cmasterNodeId == getOwnNodeId()) {
+ jam();
+ TretCode = checkNodelist(signal, TnoRestartNodes);
+ nodePtr.i = TuserNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ if (TretCode == 1) {
+ jam();
+/*******************************************************<*/
+/* CSTART_NODES IS OVERWRITTEN IN RECEIVING BLOCK, */
+/* SO WE MUST SEND CNTR_MASTERCONF TO THE SAME */
+/* CSTART_NODES AS WE RECEIVED IN CNTR_MASTERREQ */
+/*******************************************************<*/
+
+ CntrMasterConf * const cntrMasterConf =
+ (CntrMasterConf *)&signal->theData[0];
+ NodeBitmask::clear(cntrMasterConf->theNodes);
+ for (int i = 0; i < TnoRestartNodes; i++){
+ jam();
+ UintR Tnode = cstartNodes[i];
+ arrGuard(Tnode, MAX_NDB_NODES);
+ NodeBitmask::set(cntrMasterConf->theNodes, Tnode);
+ }//for
+ cntrMasterConf->noStartNodes = TnoRestartNodes;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_MASTERCONF,
+ signal, CntrMasterConf::SignalLength, JBB);
+ } else {
+ jam();
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = getOwnNodeId();
+ signal->theData[2] = ZTOO_FEW_NODES;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_MASTERREF, signal, 3, JBB);
+ }//if
+ } else {
+ jam();
+ nodePtr.i = TuserNodeId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = cownBlockref;
+ signal->theData[1] = getOwnNodeId();
+ signal->theData[2] = ZNOT_MASTER;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_MASTERREF, signal, 3, JBB);
+ }//if
+ return;
+}//Ndbcntr::masterreq030Lab()
+
+/*******************************/
+/* NODE_FAILREP */
+/*******************************/
+void Ndbcntr::execNODE_FAILREP(Signal* signal)
+{
+ UintR TfailureNr;
+ UintR TnoOfNodes;
+ UintR TreadNodes[MAX_NDB_NODES];
+
+ jamEntry();
+
+ const NodeState & st = getNodeState();
+ if(st.startLevel == st.SL_STARTING){
+ if(!st.getNodeRestartInProgress()){
+ progError(__LINE__,
+ ERR_SR_OTHERNODEFAILED,
+ "Unhandled node failure during system restart");
+ }
+ }
+
+ {
+ NodeFailRep * const nodeFail = (NodeFailRep *)&signal->theData[0];
+
+ TfailureNr = nodeFail->failNo;
+ TnoOfNodes = nodeFail->noOfNodes;
+ unsigned index = 0;
+ unsigned i;
+ for (i = 0; i < MAX_NDB_NODES; i++) {
+ jam();
+ if (NodeBitmask::get(nodeFail->theNodes, i)) {
+ jam();
+ TreadNodes[index] = i;
+ index++;
+ ndbrequire(getOwnNodeId() != i);
+ }//if
+ }//for
+ ndbrequire(TnoOfNodes == index);
+ }
+
+ for (Uint32 i = 0; i < TnoOfNodes; i++) {
+ jam();
+ nodePtr.i = TreadNodes[i];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = EventReport::NODE_FAILREP;
+ signal->theData[1] = nodePtr.i;
+ signal->theData[2] = nodePtr.p->state;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+ if (nodePtr.p->state != ZREMOVE) {
+ jam();
+ deleteNode(signal);
+ }//if
+ }//for
+
+/*******************************/
+/*< NODE_FAILREP <*/
+/*******************************/
+ NodeFailRep * const nodeFail = (NodeFailRep *)&signal->theData[0];
+
+ nodeFail->failNo = TfailureNr;
+ nodeFail->masterNodeId = cmasterNodeId;
+
+ nodeFail->noOfNodes = TnoOfNodes;
+ NodeBitmask::clear(nodeFail->theNodes);
+ for (unsigned i = 0; i < TnoOfNodes; i++) {
+ jam();
+ NodeBitmask::set(nodeFail->theNodes, TreadNodes[i]);
+ }//for
+
+ sendSignal(ctcBlockref, GSN_NODE_FAILREP, signal,
+ NodeFailRep::SignalLength, JBB);
+
+ sendSignal(clqhBlockref, GSN_NODE_FAILREP, signal,
+ NodeFailRep::SignalLength, JBB);
+
+ sendSignal(cdihBlockref, GSN_NODE_FAILREP, signal,
+ NodeFailRep::SignalLength, JBB);
+
+ sendSignal(cdictBlockref, GSN_NODE_FAILREP, signal,
+ NodeFailRep::SignalLength, JBB);
+
+ sendSignal(BACKUP_REF, GSN_NODE_FAILREP, signal,
+ NodeFailRep::SignalLength, JBB);
+
+ sendSignal(SUMA_REF, GSN_NODE_FAILREP, signal,
+ NodeFailRep::SignalLength, JBB);
+
+ sendSignal(GREP_REF, GSN_NODE_FAILREP, signal,
+ NodeFailRep::SignalLength, JBB);
+ return;
+}//Ndbcntr::execNODE_FAILREP()
+
+/*******************************/
+/* NODE_STATESCONF */
+/*******************************/
+void Ndbcntr::execNODE_STATESCONF(Signal* signal)
+{
+ jamEntry();
+ cmasterCandidateId = signal->theData[0];
+ cnoNeedNodes = signal->theData[1];
+/*----------------------------------------------------------------------*/
+// Now that we have knowledge of how many nodes are needed we will call
+// ph2CLab to ensure that node restart continues if we already received
+// all APPL_CHANGEREP signals.
+/*----------------------------------------------------------------------*/
+ ph2CLab(signal);
+ return;
+}//Ndbcntr::execNODE_STATESCONF()
+
+/*******************************/
+/* NODE_STATESREF */
+/*******************************/
+void Ndbcntr::execNODE_STATESREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Ndbcntr::execNODE_STATESREF()
+
+/*******************************/
+/* NODE_STATESREQ */
+/*******************************/
+void Ndbcntr::execNODE_STATESREQ(Signal* signal)
+{
+ UintR TnoNeedNodes = 0;
+ NodeRecPtr TNodePtr;
+ jamEntry();
+ BlockReference TuserBlockref = signal->theData[0];
+/*----------------------------------------------------------------------*/
+// IF WE ARE RUNNING, WE WILL ANSWER THIS SIGNAL WITH THE AMOUNT OF NODES
+// THAT ARE IN THE RUN STATE OR START STATE.
+/*----------------------------------------------------------------------*/
+ TNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(TNodePtr, MAX_NDB_NODES, nodeRec);
+ if (TNodePtr.p->state == ZRUN) {
+ jam();
+ for (TNodePtr.i = 1; TNodePtr.i < MAX_NDB_NODES; TNodePtr.i++) {
+ jam();
+ ptrAss(TNodePtr, nodeRec);
+ if ((TNodePtr.p->state == ZRUN) ||
+ (TNodePtr.p->state == ZSTART)) {
+ jam();
+ TnoNeedNodes++;
+ }//if
+ }//for
+ signal->theData[0] = cmasterNodeId;
+ signal->theData[1] = TnoNeedNodes;
+ sendSignal(TuserBlockref, GSN_NODE_STATESCONF, signal, 2, JBB);
+ } else {
+ jam();
+ signal->theData[0] = ZERROR_NOT_RUNNING;
+ sendSignal(TuserBlockref, GSN_NODE_STATESREF, signal, 1, JBB);
+ }//if
+ return;
+}//Ndbcntr::execNODE_STATESREQ()
+
+/*******************************/
+/* READ_NODESREQ */
+/*******************************/
+void Ndbcntr::execREAD_NODESREQ(Signal* signal)
+{
+ UintR TnoNodes = 0;
+ NodeRecPtr TNodePtr;
+ jamEntry();
+
+ /*----------------------------------------------------------------------*/
+ // ANY BLOCK MAY SEND A REQUEST ABOUT NDB NODES AND VERSIONS IN THE
+ // SYSTEM. THIS REQUEST CAN ONLY BE HANDLED IN
+ // ABSOLUTE STARTPHASE 3 OR LATER
+ /*----------------------------------------------------------------------*/
+ BlockReference TuserBlockref = signal->theData[0];
+ ReadNodesConf * const readNodes = (ReadNodesConf *)&signal->theData[0];
+
+ if (cstartPhase > ZSTART_PHASE_2) {
+ ndbrequire(cstartProgressFlag == ZTRUE);
+
+ NodeBitmask::clear(readNodes->allNodes);
+ NodeBitmask::clear(readNodes->inactiveNodes);
+
+ /**
+ * Add started nodes
+ */
+ for (int i = 0; i < cnoStartNodes; i++){
+ jam();
+ TNodePtr.i = cstartNodes[i];
+ ptrCheckGuard(TNodePtr, MAX_NDB_NODES, nodeRec);
+
+ NodeBitmask::set(readNodes->allNodes, TNodePtr.i);
+ readNodes->setVersionId(TNodePtr.i, TNodePtr.p->ndbVersion,
+ readNodes->theVersionIds);
+ TnoNodes++;
+ }//for
+
+ /**
+ * Sometimes add myself
+ */
+ if ((ctypeOfStart == NodeState::ST_NODE_RESTART) ||
+ (ctypeOfStart == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+
+ NodeBitmask::set(readNodes->allNodes, getOwnNodeId());
+ readNodes->setVersionId(getOwnNodeId(), NDB_VERSION,
+ readNodes->theVersionIds);
+ TnoNodes++;
+ }//if
+ /**
+ * Check all nodes which are defined but not already added
+ */
+ for (TNodePtr.i = 1; TNodePtr.i < MAX_NDB_NODES; TNodePtr.i++) {
+ jam();
+ ptrAss(TNodePtr, nodeRec);
+ if ((TNodePtr.p->nodeDefined == ZTRUE) &&
+ (NodeBitmask::get(readNodes->allNodes, TNodePtr.i) == false)){
+ jam();
+
+ NodeBitmask::set(readNodes->allNodes, TNodePtr.i);
+ NodeBitmask::set(readNodes->inactiveNodes, TNodePtr.i);
+ readNodes->setVersionId(TNodePtr.i, NDB_VERSION,
+ readNodes->theVersionIds);
+
+ TnoNodes++;
+ }//if
+ }//for
+
+ readNodes->noOfNodes = TnoNodes;
+ readNodes->masterNodeId = cmasterNodeId;
+ sendSignal(TuserBlockref, GSN_READ_NODESCONF, signal,
+ ReadNodesConf::SignalLength, JBB);
+
+ } else {
+ jam();
+ signal->theData[0] = ZNOT_AVAILABLE;
+ sendSignal(TuserBlockref, GSN_READ_NODESREF, signal, 1, JBB);
+ }//if
+}//Ndbcntr::execREAD_NODESREQ()
+
+/*----------------------------------------------------------------------*/
+// SENDS APPL_ERROR TO QMGR AND THEN SET A POINTER OUT OF BOUNDS
+/*----------------------------------------------------------------------*/
+void Ndbcntr::systemErrorLab(Signal* signal)
+{
+ progError(0, 0); /* BUG INSERTION */
+ return;
+}//Ndbcntr::systemErrorLab()
+
+/*###########################################################################*/
+/* CNTR MASTER CREATES AND INITIALIZES A SYSTEMTABLE AT INITIALSTART */
+/* |-2048| # 1 00000001 | */
+/* | : | : | */
+/* | -1 | # 1 00000001 | */
+/* | 0 | 0 | */
+/* | 1 | 0 | */
+/* | : | : | */
+/* | 2047| 0 | */
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::createSystableLab(Signal* signal, unsigned index)
+{
+ if (index >= g_sysTableCount) {
+ ndbassert(index == g_sysTableCount);
+ startInsertTransactions(signal);
+ return;
+ }
+ const SysTable& table = *g_sysTableList[index];
+ Uint32 propPage[256];
+ LinearWriter w(propPage, 256);
+
+ // XXX remove commented-out lines later
+
+ w.first();
+ w.add(DictTabInfo::TableName, table.name);
+ w.add(DictTabInfo::TableLoggedFlag, table.tableLoggedFlag);
+ //w.add(DictTabInfo::TableKValue, 6);
+ //w.add(DictTabInfo::MinLoadFactor, 70);
+ //w.add(DictTabInfo::MaxLoadFactor, 80);
+ w.add(DictTabInfo::FragmentTypeVal, (Uint32)table.fragmentType);
+ //w.add(DictTabInfo::TableStorageVal, (Uint32)DictTabInfo::MainMemory);
+ //w.add(DictTabInfo::NoOfKeyAttr, 1);
+ w.add(DictTabInfo::NoOfAttributes, (Uint32)table.columnCount);
+ //w.add(DictTabInfo::NoOfNullable, (Uint32)0);
+ //w.add(DictTabInfo::NoOfVariable, (Uint32)0);
+ //w.add(DictTabInfo::KeyLength, 1);
+ w.add(DictTabInfo::TableTypeVal, (Uint32)table.tableType);
+
+ for (unsigned i = 0; i < table.columnCount; i++) {
+ const SysColumn& column = table.columnList[i];
+ ndbassert(column.pos == i);
+ w.add(DictTabInfo::AttributeName, column.name);
+ w.add(DictTabInfo::AttributeId, (Uint32)column.pos);
+ //w.add(DictTabInfo::AttributeType, DictTabInfo::UnSignedType);
+ //w.add(DictTabInfo::AttributeSize, DictTabInfo::a32Bit);
+ //w.add(DictTabInfo::AttributeArraySize, 1);
+ w.add(DictTabInfo::AttributeKeyFlag, (Uint32)column.keyFlag);
+ //w.add(DictTabInfo::AttributeStorage, (Uint32)DictTabInfo::MainMemory);
+ w.add(DictTabInfo::AttributeNullableFlag, (Uint32)column.nullable);
+ // ext type overrides
+ w.add(DictTabInfo::AttributeExtType, (Uint32)column.type);
+ w.add(DictTabInfo::AttributeExtLength, (Uint32)column.length);
+ w.add(DictTabInfo::AttributeEnd, (Uint32)true);
+ }
+ w.add(DictTabInfo::TableEnd, (Uint32)true);
+
+ Uint32 length = w.getWordsUsed();
+ LinearSectionPtr ptr[3];
+ ptr[0].p = &propPage[0];
+ ptr[0].sz = length;
+
+ CreateTableReq* const req = (CreateTableReq*)signal->getDataPtrSend();
+ req->senderData = index;
+ req->senderRef = reference();
+ sendSignal(DBDICT_REF, GSN_CREATE_TABLE_REQ, signal,
+ CreateTableReq::SignalLength, JBB, ptr, 1);
+ return;
+}//Ndbcntr::createSystableLab()
+
+void Ndbcntr::execCREATE_TABLE_REF(Signal* signal)
+{
+ jamEntry();
+ progError(0,0);
+ return;
+}//Ndbcntr::execDICTTABREF()
+
+void Ndbcntr::execCREATE_TABLE_CONF(Signal* signal)
+{
+ jamEntry();
+ CreateTableConf * const conf = (CreateTableConf*)signal->getDataPtrSend();
+ //csystabId = conf->tableId;
+ ndbrequire(conf->senderData < g_sysTableCount);
+ const SysTable& table = *g_sysTableList[conf->senderData];
+ table.tableId = conf->tableId;
+ createSystableLab(signal, conf->senderData + 1);
+ //startInsertTransactions(signal);
+ return;
+}//Ndbcntr::execDICTTABCONF()
+
+/*******************************/
+/* DICTRELEASECONF */
+/*******************************/
+void Ndbcntr::startInsertTransactions(Signal* signal)
+{
+ jamEntry();
+
+ ckey = 1;
+ ctransidPhase = ZTRUE;
+ signal->theData[1] = cownBlockref;
+ sendSignal(ctcBlockref, GSN_TCSEIZEREQ, signal, 2, JBB);
+ return;
+}//Ndbcntr::startInsertTransactions()
+
+/*******************************/
+/* TCSEIZECONF */
+/*******************************/
+void Ndbcntr::execTCSEIZECONF(Signal* signal)
+{
+ jamEntry();
+ ctcConnectionP = signal->theData[1];
+ crSystab7Lab(signal);
+ return;
+}//Ndbcntr::execTCSEIZECONF()
+
+const unsigned int RowsPerCommit = 16;
+void Ndbcntr::crSystab7Lab(Signal* signal)
+{
+ UintR tkey;
+ UintR Tmp;
+
+ TcKeyReq * const tcKeyReq = (TcKeyReq *)&signal->theData[0];
+
+ UintR reqInfo_Start = 0;
+ tcKeyReq->setOperationType(reqInfo_Start, ZINSERT); // Insert
+ tcKeyReq->setKeyLength (reqInfo_Start, 1);
+ tcKeyReq->setAIInTcKeyReq (reqInfo_Start, 5);
+ tcKeyReq->setAbortOption (reqInfo_Start, TcKeyReq::AbortOnError);
+
+/* KEY LENGTH = 1, ATTRINFO LENGTH IN TCKEYREQ = 5 */
+ cresponses = 0;
+ const UintR guard0 = ckey + (RowsPerCommit - 1);
+ for (Tmp = ckey; Tmp <= guard0; Tmp++) {
+ UintR reqInfo = reqInfo_Start;
+ if (Tmp == ckey) { // First iteration, Set start flag
+ jam();
+ tcKeyReq->setStartFlag(reqInfo, 1);
+ } //if
+ if (Tmp == guard0) { // Last iteration, Set commit flag
+ jam();
+ tcKeyReq->setCommitFlag(reqInfo, 1);
+ tcKeyReq->setExecuteFlag(reqInfo, 1);
+ } //if
+ if (ctransidPhase == ZTRUE) {
+ jam();
+ tkey = 0;
+ tkey = tkey - Tmp;
+ } else {
+ jam();
+ tkey = Tmp;
+ }//if
+
+ tcKeyReq->apiConnectPtr = ctcConnectionP;
+ tcKeyReq->attrLen = 5;
+ tcKeyReq->tableId = g_sysTable_SYSTAB_0.tableId;
+ tcKeyReq->requestInfo = reqInfo;
+ tcKeyReq->tableSchemaVersion = ZSYSTAB_VERSION;
+ tcKeyReq->transId1 = 0;
+ tcKeyReq->transId2 = 0;
+
+//-------------------------------------------------------------
+// There is no optional part in this TCKEYREQ. There is one
+// key word and five ATTRINFO words.
+//-------------------------------------------------------------
+ Uint32* tKeyDataPtr = &tcKeyReq->scanInfo;
+ Uint32* tAIDataPtr = &tKeyDataPtr[1];
+
+ tKeyDataPtr[0] = tkey;
+
+ AttributeHeader::init(&tAIDataPtr[0], 0, 1);
+ tAIDataPtr[1] = tkey;
+ AttributeHeader::init(&tAIDataPtr[2], 1, 2);
+ tAIDataPtr[3] = (tkey << 16);
+ tAIDataPtr[4] = 1;
+ sendSignal(ctcBlockref, GSN_TCKEYREQ, signal,
+ TcKeyReq::StaticLength + 6, JBB);
+ }//for
+ ckey = ckey + RowsPerCommit;
+ return;
+}//Ndbcntr::crSystab7Lab()
+
+/*******************************/
+/* TCKEYCONF09 */
+/*******************************/
+void Ndbcntr::execTCKEYCONF(Signal* signal)
+{
+ const TcKeyConf * const keyConf = (TcKeyConf *)&signal->theData[0];
+
+ jamEntry();
+ cgciSystab = keyConf->gci;
+ UintR confInfo = keyConf->confInfo;
+
+ if (TcKeyConf::getMarkerFlag(confInfo)){
+ Uint32 transId1 = keyConf->transId1;
+ Uint32 transId2 = keyConf->transId2;
+ signal->theData[0] = transId1;
+ signal->theData[1] = transId2;
+ sendSignal(ctcBlockref, GSN_TC_COMMIT_ACK, signal, 2, JBB);
+ }//if
+
+ cresponses = cresponses + TcKeyConf::getNoOfOperations(confInfo);
+ if (TcKeyConf::getCommitFlag(confInfo)){
+ jam();
+ ndbrequire(cresponses == RowsPerCommit);
+
+ crSystab8Lab(signal);
+ return;
+ }
+ return;
+}//Ndbcntr::tckeyConfLab()
+
+void Ndbcntr::crSystab8Lab(Signal* signal)
+{
+ if (ckey < ZSIZE_SYSTAB) {
+ jam();
+ crSystab7Lab(signal);
+ return;
+ } else if (ctransidPhase == ZTRUE) {
+ jam();
+ ckey = 1;
+ ctransidPhase = ZFALSE;
+ crSystab7Lab(signal);
+ return;
+ }//if
+ signal->theData[0] = ctcConnectionP;
+ signal->theData[1] = cownBlockref;
+ sendSignal(ctcBlockref, GSN_TCRELEASEREQ, signal, 2, JBB);
+ return;
+}//Ndbcntr::crSystab8Lab()
+
+/*******************************/
+/* TCRELEASECONF */
+/*******************************/
+void Ndbcntr::execTCRELEASECONF(Signal* signal)
+{
+ jamEntry();
+ waitpoint52Lab(signal);
+ return;
+}//Ndbcntr::execTCRELEASECONF()
+
+void Ndbcntr::crSystab9Lab(Signal* signal)
+{
+ signal->theData[1] = cownBlockref;
+ sendSignalWithDelay(cdihBlockref, GSN_GETGCIREQ, signal, 100, 2);
+ return;
+}//Ndbcntr::crSystab9Lab()
+
+/*******************************/
+/* GETGCICONF */
+/*******************************/
+void Ndbcntr::execGETGCICONF(Signal* signal)
+{
+ jamEntry();
+
+#ifndef NO_GCP
+ if (signal->theData[1] < cgciSystab) {
+ jam();
+/*--------------------------------------*/
+/* MAKE SURE THAT THE SYSTABLE IS */
+/* NOW SAFE ON DISK */
+/*--------------------------------------*/
+ crSystab9Lab(signal);
+ return;
+ }//if
+#endif
+ waitpoint52Lab(signal);
+ return;
+}//Ndbcntr::execGETGCICONF()
+
+void Ndbcntr::execTCKEYREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Ndbcntr::execTCKEYREF()
+
+void Ndbcntr::execTCROLLBACKREP(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Ndbcntr::execTCROLLBACKREP()
+
+void Ndbcntr::execTCRELEASEREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Ndbcntr::execTCRELEASEREF()
+
+void Ndbcntr::execTCSEIZEREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Ndbcntr::execTCSEIZEREF()
+
+/*
+4.10 SUBROUTINES */
+/*##########################################################################*/
+/*
+4.10.1 CHECK_NODELIST */
+/*---------------------------------------------------------------------------*/
+/*CHECK THAT ALL THE NEW NODE HAS DETECTED ALL RUNNING NODES */
+/*INPUT: CSTART_NODES */
+/* TNO_RESTART_NODES */
+/* TUSER_NODE_ID */
+/*RET: CNODE_RESTART */
+/*---------------------------------------------------------------------------*/
+UintR Ndbcntr::checkNodelist(Signal* signal, Uint16 TnoRestartNodes)
+{
+ UintR guard1;
+ UintR Ttemp1;
+
+ if (cnoRunNodes == TnoRestartNodes) {
+ jam();
+ guard1 = TnoRestartNodes - 1;
+ arrGuard(guard1, MAX_NDB_NODES);
+ for (Ttemp1 = 0; Ttemp1 <= guard1; Ttemp1++) {
+ jam();
+ nodePtr.i = cstartNodes[Ttemp1];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ if (nodePtr.p->state != ZRUN) {
+ jam();
+ return 0;
+ }//if
+ }//for
+ return 1;
+ }//if
+ return 0;
+}//Ndbcntr::checkNodelist()
+
+/*---------------------------------------------------------------------------*/
+// SELECT NODES THAT ARE IN THE STATE TO PERFORM A INITIALSTART OR
+// SYSTEMRESTART.
+// THIS SUBROUTINE CAN ONLY BE INVOKED BY THE MASTER NODE.
+// TO BE CHOOSEN A NODE NEED AS MANY VOTES AS THERE ARE VOTERS, AND OF
+// COURSE THE NODE HAS TO BE KNOWN BY THE
+// MASTER
+// INPUT: NODE_REC
+// CNO_NEED_NODES
+// RETURN:CNO_START_NODES
+// CSTART_NODES
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::chooseRestartNodes(Signal* signal)
+{
+ cnoStartNodes = 0;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->votes == cnoVoters) {
+ jam();
+ if (nodePtr.p->state == ZADD) {
+ jam();
+ arrGuard(cnoStartNodes, MAX_NDB_NODES);
+ cstartNodes[cnoStartNodes] = nodePtr.i;
+ cnoStartNodes++;
+ }//if
+ } else {
+ jam();
+ if (nodePtr.p->votes > 0) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ }//if
+ }//for
+}//Ndbcntr::chooseRestartNodes()
+
+/*
+4.10.6 DELETE_NODE */
+/*---------------------------------------------------------------------------*/
+// INPUT: NODE_PTR
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::deleteNode(Signal* signal)
+{
+ UintR tminDynamicId;
+
+ if (nodePtr.p->state == ZRUN) {
+ jam();
+ cnoRunNodes = cnoRunNodes - 1;
+ }//if
+ nodePtr.p->state = ZREMOVE;
+ nodePtr.p->votes = 0;
+ nodePtr.p->voter = ZFALSE;
+ cnoRegNodes--;
+ if (nodePtr.i == cmasterNodeId) {
+ jam();
+ cmasterNodeId = ZNIL;
+/*---------------------------------------------------------------------------*/
+// IF MASTER HAVE CRASHED WE NEED TO SELECT A NEW MASTER.
+/*---------------------------------------------------------------------------*/
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->state == ZRUN) {
+ if (cmasterNodeId == ZNIL) {
+ jam();
+ cmasterNodeId = nodePtr.i;
+ tminDynamicId = nodePtr.p->dynamicId;
+ } else {
+ jam();
+ if (nodePtr.p->dynamicId < tminDynamicId) {
+ jam();
+ cmasterNodeId = nodePtr.i;
+ tminDynamicId = nodePtr.p->dynamicId;
+ }//if
+ }//if
+ }//if
+ }//for
+ }//if
+}//Ndbcntr::deleteNode()
+
+/*---------------------------------------------------------------------------*/
+// A NEW NODE TRIES TO DETECT A NODE RESTART. NodeState::ST_NODE_RESTART IS A POSSIBLE
+// STATE ONLY WHEN THE SYSTEM IS RUNNING.
+// IF THE SYSTEM IS RUNNING THEN
+// CTYPE_OF_START = NodeState::ST_SYSTEM_RESTART UNTIL THE FIRST NODE HAS REGISTERED.
+// IF SYSTEM IS */
+// RUNNING THE FIRST NODE TO REGISTER WILL BE ZRUN AND CTYPE_OF_START
+// WILL BE CHANGED */
+// TO NodeState::ST_NODE_RESTART AT PH_2C. WHEN A NodeState::ST_NODE_RESTART IS DETECTED THE NEW NODE
+// HAS TO SEND */
+// A CNTR_MASTERREQ TO THE MASTER
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::detectNoderestart(Signal* signal)
+{
+ NodeRecPtr ownNodePtr;
+ ownNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(ownNodePtr, MAX_NDB_NODES, nodeRec);
+ if (ownNodePtr.p->state != ZADD) {
+ if (ownNodePtr.p->state != ZREMOVE) {
+ jam();
+ return;
+ }//if
+ }//if
+ ctypeOfStart = NodeState::ST_NODE_RESTART;
+/*----------------------------------------------*/
+/* THIS NODE WILL PERFORM A NODE RESTART */
+/* REQUEST OF ALL NODES STATES IN SYSTEM */
+// The purpose of this signal is to ensure that
+// the starting node knows when it has received
+// all APPL_CHANGEREP signals and thus can continue
+// to the next step of the node restart. Thus we
+// need to know the amount of nodes that are in the
+// RUN state and in the START state (more than one
+// node can be copying data simultaneously in the
+// cluster.
+/*----------------------------------------------*/
+ signal->theData[0] = cownBlockref;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_NODE_STATESREQ, signal, 1, JBB);
+ cnoNeedNodes = ZNIL;
+/*---------------------------------*/
+/* PREVENT TO SEND NODE_STATESREQ */
+/*---------------------------------------------------------------------------*/
+/* WE NEED TO WATCH THE NODE RESTART WITH A TIME OUT TO NOT WAIT FOR EVER. */
+/*---------------------------------------------------------------------------*/
+ cwaitContinuebFlag = ZTRUE;
+ signal->theData[0] = ZCONTINUEB_1;
+ sendSignalWithDelay(cownBlockref, GSN_CONTINUEB, signal, 3 * 1000, 1);
+}//Ndbcntr::detectNoderestart()
+
+/*---------------------------------------------------------------------------*/
+// SCAN NODE_REC FOR APPROPRIATE NODES FOR A START.
+// SYSTEMRESTART AND INITALSTART DEMANDS NODES OF STATE ZADD.
+// NODERESTART DEMANDS NODE OF THE STATE ZRUN.
+// INPUT: CTYPE_OF_START, NODE_REC
+// RETURN: CSTART_NODES(), CNO_START_NODES, CMASTER_CANDIDATE_ID
+// (SYSTEMRESTART AND INITALSTART)
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::getStartNodes(Signal* signal)
+{
+ UintR Ttemp1;
+ if ((ctypeOfStart == NodeState::ST_NODE_RESTART) ||
+ (ctypeOfStart == NodeState::ST_INITIAL_NODE_RESTART)) {
+ jam();
+ Ttemp1 = ZRUN;
+ } else {
+ jam();
+/*---------------------------------*/
+/* SYSTEM RESTART AND INITIAL START*/
+/*---------------------------------*/
+ Ttemp1 = ZADD;
+ }//if
+ cnoStartNodes = 0;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->state == Ttemp1) {
+ jam();
+ cstartNodes[cnoStartNodes] = nodePtr.i;/*OVERWRITTEN AT CNTR_MASTERCONF*/
+ cnoStartNodes++;
+ }//if
+ }//for
+}//Ndbcntr::getStartNodes()
+
+/*---------------------------------------------------------------------------*/
+/*INITIALIZE VARIABLES AND RECORDS */
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::initData(Signal* signal)
+{
+ cmasterNodeId = ZNIL;
+ cmasterCandidateId = ZNIL;
+ cmasterVoters = 0;
+ cstartProgressFlag = ZFALSE;
+ capplStartconfFlag = ZFALSE;
+ cnoVoters = 0;
+ cnoStartNodes = 0;
+ for (nodePtr.i = 0; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRec);
+ nodePtr.p->cntrBlockref = calcNdbCntrBlockRef(nodePtr.i);
+ nodePtr.p->state = ZREMOVE;
+ nodePtr.p->dynamicId = 0;
+ nodePtr.p->votes = 0; /* USED BY MASTER */
+ nodePtr.p->voter = ZFALSE; /* USED BY MASTER */
+ nodePtr.p->masterReq = ZFALSE; /* USED BY MASTER */
+ }//for
+}//Ndbcntr::initData()
+
+/*---------------------------------------------------------------------------*/
+// THE MASTER NODE HAS CHOOSEN THE NODES WHO WERE QUALIFIED TO
+// PARTICIPATE IN A INITIALSTART OR SYSTEMRESTART.
+// THIS SUBROTINE SENDS A CNTR_MASTERCONF TO THESE NODES
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::replyMasterconfToAll(Signal* signal)
+{
+ if (cnoStartNodes > 1) {
+ /**
+ * Construct a MasterConf signal
+ */
+
+ CntrMasterConf * const cntrMasterConf =
+ (CntrMasterConf *)&signal->theData[0];
+ NodeBitmask::clear(cntrMasterConf->theNodes);
+
+ cntrMasterConf->noStartNodes = cnoStartNodes;
+
+ for(int i = 0; i<cnoStartNodes; i++)
+ NodeBitmask::set(cntrMasterConf->theNodes, cstartNodes[i]);
+
+ /**
+ * Then distribute it to everyone but myself
+ */
+ for(int i = 0; i<cnoStartNodes; i++){
+ const NodeId nodeId = cstartNodes[i];
+ if(nodeId != getOwnNodeId()){
+ sendSignal(numberToRef(number(), nodeId),
+ GSN_CNTR_MASTERCONF,
+ signal, CntrMasterConf::SignalLength, JBB);
+ }
+ }
+ }
+}//Ndbcntr::replyMasterconfToAll()
+
+/*---------------------------------------------------------------------------*/
+/*RESET VARIABLES USED DURING THE START */
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::resetStartVariables(Signal* signal)
+{
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRec);
+ nodePtr.p->votes = 0;
+ nodePtr.p->voter = ZFALSE;
+ nodePtr.p->masterReq = ZFALSE;
+ }//for
+ cnoVoters = 0;
+ cnoStartNodes = 0;
+ cnoWaitrep6 = cnoWaitrep7 = 0;
+}//Ndbcntr::resetStartVariables()
+
+/*---------------------------------------------------------------------------*/
+// SENDER OF THIS SIGNAL HAS CHOOSEN A MASTER NODE AND SENDS A REQUEST
+// TO THE MASTER_CANDIDATE AS AN VOTE FOR
+// THE MASTER. THE SIGNAL ALSO INCLUDES VOTES FOR NODES WHICH SENDER
+// THINKS SHOULD PARTICIPATE IN THE START.
+// INPUT: CNO_START_NODES
+// CSTART_NODES
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::sendCntrMasterreq(Signal* signal)
+{
+ nodePtr.i = cmasterCandidateId;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+/*--------------------------------------------------------------*/
+/* O:INITIALSTART, 1:SYSTEMRESTART (ELECTION OF MASTER) */
+/* 2:NODE RESTART (SENDER NODE NOT INCLUDED IN CSTART_NODES) */
+/*--------------------------------------------------------------*/
+ CntrMasterReq * const cntrMasterReq = (CntrMasterReq*)&signal->theData[0];
+ NodeBitmask::clear(cntrMasterReq->theNodes);
+ for (int i = 0; i < cnoStartNodes; i++){
+ jam();
+ UintR Tnode = cstartNodes[i];
+ arrGuard(Tnode, MAX_NDB_NODES);
+ NodeBitmask::set(cntrMasterReq->theNodes, Tnode);
+ }//for
+ cntrMasterReq->userBlockRef = cownBlockref;
+ cntrMasterReq->userNodeId = getOwnNodeId();
+ cntrMasterReq->typeOfStart = ctypeOfStart;
+ cntrMasterReq->noRestartNodes = cnoStartNodes;
+ sendSignal(nodePtr.p->cntrBlockref, GSN_CNTR_MASTERREQ,
+ signal, CntrMasterReq::SignalLength, JBB);
+}//Ndbcntr::sendCntrMasterreq()
+
+/*---------------------------------------------------------------------------*/
+// SEND THE SIGNAL
+// INPUT CNDB_BLOCKS_COUNT
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::sendNdbSttor(Signal* signal)
+{
+ CfgBlockRecPtr cfgBlockPtr;
+ NdbBlocksRecPtr ndbBlocksPtr;
+
+ ndbBlocksPtr.i = cndbBlocksCount;
+ ptrCheckGuard(ndbBlocksPtr, ZSIZE_NDB_BLOCKS_REC, ndbBlocksRec);
+ cfgBlockPtr.i = cinternalStartphase;
+ ptrCheckGuard(cfgBlockPtr, ZSIZE_CFG_BLOCK_REC, cfgBlockRec);
+ NdbSttor * const req = (NdbSttor*)signal->getDataPtrSend();
+ req->senderRef = cownBlockref;
+ req->nodeId = getOwnNodeId();
+ req->internalStartPhase = cinternalStartphase;
+ req->typeOfStart = ctypeOfStart;
+ req->masterNodeId = cmasterNodeId;
+
+ for (int i = 0; i < 16; i++) {
+ req->config[i] = cfgBlockPtr.p->cfgData[i];
+ }
+
+ //#define TRACE_STTOR
+ //#define MAX_STARTPHASE 2
+#ifdef TRACE_STTOR
+ ndbout_c("sending NDB_STTOR(%d) to %s",
+ cinternalStartphase,
+ getBlockName( refToBlock(ndbBlocksPtr.p->blockref)));
+#endif
+ sendSignal(ndbBlocksPtr.p->blockref, GSN_NDB_STTOR, signal, 22, JBB);
+ cndbBlocksCount++;
+}//Ndbcntr::sendNdbSttor()
+
+/*---------------------------------------------------------------------------*/
+// JUST SEND THE SIGNAL
+/*---------------------------------------------------------------------------*/
+void Ndbcntr::sendSttorry(Signal* signal)
+{
+ signal->theData[0] = csignalKey;
+ signal->theData[1] = 3;
+ signal->theData[2] = 2;
+ signal->theData[3] = ZSTART_PHASE_1;
+ signal->theData[4] = ZSTART_PHASE_2;
+ signal->theData[5] = ZSTART_PHASE_3;
+ signal->theData[6] = ZSTART_PHASE_4;
+ signal->theData[7] = ZSTART_PHASE_5;
+ signal->theData[8] = ZSTART_PHASE_6;
+ // skip simulated phase 7
+ signal->theData[9] = ZSTART_PHASE_8;
+ signal->theData[10] = ZSTART_PHASE_9;
+ signal->theData[11] = ZSTART_PHASE_END;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 12, JBB);
+}//Ndbcntr::sendSttorry()
+
+void
+Ndbcntr::execDUMP_STATE_ORD(Signal* signal)
+{
+ DumpStateOrd * const & dumpState = (DumpStateOrd *)&signal->theData[0];
+ if(signal->theData[0] == 13){
+ infoEvent("Cntr: cstartPhase = %d, cinternalStartphase = %d, block = %d",
+ cstartPhase, cinternalStartphase, cndbBlocksCount);
+ infoEvent("Cntr: cmasterNodeId = %d, cmasterCandidateId = %d",
+ cmasterNodeId, cmasterCandidateId);
+ }
+
+ if (dumpState->args[0] == DumpStateOrd::NdbcntrTestStopOnError){
+ if (theConfiguration.stopOnError() == true)
+ ((Configuration&)theConfiguration).stopOnError(false);
+
+ const BlockReference tblockref = calcNdbCntrBlockRef(getOwnNodeId());
+
+ SystemError * const sysErr = (SystemError*)&signal->theData[0];
+ sysErr->errorCode = SystemError::TestStopOnError;
+ sysErr->errorRef = reference();
+ sendSignal(tblockref, GSN_SYSTEM_ERROR, signal,
+ SystemError::SignalLength, JBA);
+ }
+
+
+}//Ndbcntr::execDUMP_STATE_ORD()
+
+void Ndbcntr::execSET_VAR_REQ(Signal* signal) {
+ SetVarReq* const setVarReq = (SetVarReq*)&signal->theData[0];
+ ConfigParamId var = setVarReq->variable();
+
+ switch (var) {
+ case TimeToWaitAlive:
+ // Valid only during start so value not set.
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ default:
+ sendSignal(CMVMI_REF, GSN_SET_VAR_REF, signal, 1, JBB);
+ }// switch
+}//Ndbcntr::execSET_VAR_REQ()
+
+void Ndbcntr::updateNodeState(Signal* signal, const NodeState& newState) const{
+ NodeStateRep * const stateRep = (NodeStateRep *)&signal->theData[0];
+
+ stateRep->nodeState = newState;
+ stateRep->nodeState.masterNodeId = cmasterNodeId;
+ stateRep->nodeState.setNodeGroup(c_nodeGroup);
+
+ for(Uint32 i = 0; i<ALL_BLOCKS_SZ; i++){
+ sendSignal(ALL_BLOCKS[i].Ref, GSN_NODE_STATE_REP, signal,
+ NodeStateRep::SignalLength, JBB);
+ }
+}
+
+void
+Ndbcntr::execRESUME_REQ(Signal* signal){
+ //ResumeReq * const req = (ResumeReq *)&signal->theData[0];
+ //ResumeRef * const ref = (ResumeRef *)&signal->theData[0];
+
+ jamEntry();
+ //Uint32 senderData = req->senderData;
+ //BlockReference senderRef = req->senderRef;
+ NodeState newState(NodeState::SL_STARTED);
+ updateNodeState(signal, newState);
+ c_stopRec.stopReq.senderRef=0;
+}
+
+void
+Ndbcntr::execSTOP_REQ(Signal* signal){
+ StopReq * const req = (StopReq *)&signal->theData[0];
+ StopRef * const ref = (StopRef *)&signal->theData[0];
+ Uint32 singleuser = req->singleuser;
+ jamEntry();
+ Uint32 senderData = req->senderData;
+ BlockReference senderRef = req->senderRef;
+ bool abort = StopReq::getStopAbort(req->requestInfo);
+
+ if(getNodeState().startLevel < NodeState::SL_STARTED ||
+ abort && !singleuser){
+ /**
+ * Node is not started yet
+ *
+ * So stop it quickly
+ */
+ jam();
+ const Uint32 reqInfo = req->requestInfo;
+ if(StopReq::getPerformRestart(reqInfo)){
+ jam();
+ StartOrd * startOrd = (StartOrd *)&signal->theData[0];
+ startOrd->restartInfo = reqInfo;
+ sendSignal(CMVMI_REF, GSN_START_ORD, signal, 1, JBA);
+ } else {
+ jam();
+ sendSignal(CMVMI_REF, GSN_STOP_ORD, signal, 1, JBA);
+ }
+ return;
+ }
+
+ if(c_stopRec.stopReq.senderRef != 0 && !singleuser){
+ jam();
+ /**
+ * Requested a system shutdown
+ */
+ if(StopReq::getSystemStop(req->requestInfo)){
+ jam();
+ sendSignalWithDelay(reference(), GSN_STOP_REQ, signal, 100,
+ StopReq::SignalLength);
+ return;
+ }
+
+ /**
+ * Requested a node shutdown
+ */
+ if(StopReq::getSystemStop(c_stopRec.stopReq.requestInfo))
+ ref->errorCode = StopRef::SystemShutdownInProgress;
+ else
+ ref->errorCode = StopRef::NodeShutdownInProgress;
+ ref->senderData = senderData;
+ sendSignal(senderRef, GSN_STOP_REF, signal, StopRef::SignalLength, JBB);
+ return;
+ }
+
+ c_stopRec.stopReq = * req;
+ c_stopRec.stopInitiatedTime = NdbTick_CurrentMillisecond();
+
+ if(StopReq::getSystemStop(c_stopRec.stopReq.requestInfo) && !singleuser) {
+ jam();
+ if(StopReq::getPerformRestart(c_stopRec.stopReq.requestInfo)){
+ ((Configuration&)theConfiguration).stopOnError(false);
+ }
+ }
+ if(!singleuser) {
+ if(!c_stopRec.checkNodeFail(signal)){
+ jam();
+ return;
+ }
+ }
+ NodeState newState(NodeState::SL_STOPPING_1,
+ StopReq::getSystemStop(c_stopRec.stopReq.requestInfo));
+
+ if(singleuser) {
+ newState.setSingleUser(true);
+ newState.setSingleUserApi(c_stopRec.stopReq.singleUserApi);
+ }
+ updateNodeState(signal, newState);
+ signal->theData[0] = ZSHUTDOWN;
+ sendSignalWithDelay(cownBlockref, GSN_CONTINUEB, signal, 100, 1);
+}
+
+void
+Ndbcntr::StopRecord::checkTimeout(Signal* signal){
+ jamEntry();
+
+ if(!cntr.getNodeState().getSingleUserMode())
+ if(!checkNodeFail(signal)){
+ jam();
+ return;
+ }
+
+ switch(cntr.getNodeState().startLevel){
+ case NodeState::SL_STOPPING_1:
+ checkApiTimeout(signal);
+ break;
+ case NodeState::SL_STOPPING_2:
+ checkTcTimeout(signal);
+ break;
+ case NodeState::SL_STOPPING_3:
+ checkLqhTimeout_1(signal);
+ break;
+ case NodeState::SL_STOPPING_4:
+ checkLqhTimeout_2(signal);
+ break;
+ case NodeState::SL_SINGLEUSER:
+ break;
+ default:
+ ndbrequire(false);
+ }
+}
+
+bool
+Ndbcntr::StopRecord::checkNodeFail(Signal* signal){
+ jam();
+ if(StopReq::getSystemStop(stopReq.requestInfo)){
+ jam();
+ return true;
+ }
+
+ /**
+ * Check if I can survive me stopping
+ */
+ NodeBitmask ndbMask; ndbMask.clear();
+ NodeRecPtr aPtr;
+ for(aPtr.i = 1; aPtr.i < MAX_NDB_NODES; aPtr.i++){
+ ptrAss(aPtr, cntr.nodeRec);
+ if(aPtr.i != cntr.getOwnNodeId() && aPtr.p->state == ZRUN){
+ ndbMask.set(aPtr.i);
+ }
+ }
+
+ CheckNodeGroups* sd = (CheckNodeGroups*)&signal->theData[0];
+ sd->blockRef = cntr.reference();
+ sd->requestType = CheckNodeGroups::Direct | CheckNodeGroups::ArbitCheck;
+ sd->mask = ndbMask;
+ cntr.EXECUTE_DIRECT(DBDIH, GSN_CHECKNODEGROUPSREQ, signal,
+ CheckNodeGroups::SignalLength);
+ jamEntry();
+ switch (sd->output) {
+ case CheckNodeGroups::Win:
+ case CheckNodeGroups::Partitioning:
+ return true;
+ break;
+ }
+
+ StopRef * const ref = (StopRef *)&signal->theData[0];
+
+ ref->senderData = stopReq.senderData;
+ ref->errorCode = StopRef::NodeShutdownWouldCauseSystemCrash;
+
+ const BlockReference bref = stopReq.senderRef;
+ cntr.sendSignal(bref, GSN_STOP_REF, signal, StopRef::SignalLength, JBB);
+
+ stopReq.senderRef = 0;
+
+ NodeState newState(NodeState::SL_STARTED);
+
+ cntr.updateNodeState(signal, newState);
+ return false;
+}
+
+void
+Ndbcntr::StopRecord::checkApiTimeout(Signal* signal){
+ const Int32 timeout = stopReq.apiTimeout;
+ const NDB_TICKS alarm = stopInitiatedTime + (NDB_TICKS)timeout;
+ const NDB_TICKS now = NdbTick_CurrentMillisecond();
+ if((timeout >= 0 && now >= alarm)){
+ // || checkWithApiInSomeMagicWay)
+ jam();
+ NodeState newState(NodeState::SL_STOPPING_2,
+ StopReq::getSystemStop(stopReq.requestInfo));
+ if(stopReq.singleuser) {
+ newState.setSingleUser(true);
+ newState.setSingleUserApi(stopReq.singleUserApi);
+ }
+ cntr.updateNodeState(signal, newState);
+
+ stopInitiatedTime = now;
+ }
+
+ signal->theData[0] = ZSHUTDOWN;
+ cntr.sendSignalWithDelay(cntr.reference(), GSN_CONTINUEB, signal, 100, 1);
+}
+
+void
+Ndbcntr::StopRecord::checkTcTimeout(Signal* signal){
+ const Int32 timeout = stopReq.transactionTimeout;
+ const NDB_TICKS alarm = stopInitiatedTime + (NDB_TICKS)timeout;
+ const NDB_TICKS now = NdbTick_CurrentMillisecond();
+ if((timeout >= 0 && now >= alarm)){
+ // || checkWithTcInSomeMagicWay)
+ jam();
+ if(stopReq.getSystemStop(stopReq.requestInfo) || stopReq.singleuser){
+ jam();
+ if(stopReq.singleuser)
+ {
+ jam();
+ AbortAllReq * req = (AbortAllReq*)&signal->theData[0];
+ req->senderRef = cntr.reference();
+ req->senderData = 12;
+ cntr.sendSignal(DBTC_REF, GSN_ABORT_ALL_REQ, signal,
+ AbortAllReq::SignalLength, JBB);
+ }
+ else
+ {
+ WaitGCPReq * req = (WaitGCPReq*)&signal->theData[0];
+ req->senderRef = cntr.reference();
+ req->senderData = 12;
+ req->requestType = WaitGCPReq::CompleteForceStart;
+ cntr.sendSignal(DBDIH_REF, GSN_WAIT_GCP_REQ, signal,
+ WaitGCPReq::SignalLength, JBB);
+ }
+ } else {
+ jam();
+ StopPermReq * req = (StopPermReq*)&signal->theData[0];
+ req->senderRef = cntr.reference();
+ req->senderData = 12;
+ cntr.sendSignal(DBDIH_REF, GSN_STOP_PERM_REQ, signal,
+ StopPermReq::SignalLength, JBB);
+ }
+ return;
+ }
+ signal->theData[0] = ZSHUTDOWN;
+ cntr.sendSignalWithDelay(cntr.reference(), GSN_CONTINUEB, signal, 100, 1);
+}
+
+void Ndbcntr::execSTOP_PERM_REF(Signal* signal){
+ //StopPermRef* const ref = (StopPermRef*)&signal->theData[0];
+
+ jamEntry();
+
+ signal->theData[0] = ZSHUTDOWN;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 1);
+}
+
+void Ndbcntr::execSTOP_PERM_CONF(Signal* signal){
+ jamEntry();
+
+ AbortAllReq * req = (AbortAllReq*)&signal->theData[0];
+ req->senderRef = reference();
+ req->senderData = 12;
+ sendSignal(DBTC_REF, GSN_ABORT_ALL_REQ, signal,
+ AbortAllReq::SignalLength, JBB);
+}
+
+void Ndbcntr::execABORT_ALL_CONF(Signal* signal){
+ jamEntry();
+ if(c_stopRec.stopReq.singleuser) {
+ jam();
+ NodeState newState(NodeState::SL_SINGLEUSER);
+ newState.setSingleUser(true);
+ newState.setSingleUserApi(c_stopRec.stopReq.singleUserApi);
+ updateNodeState(signal, newState);
+ c_stopRec.stopInitiatedTime = NdbTick_CurrentMillisecond();
+
+ }
+ else
+ {
+ jam();
+ NodeState newState(NodeState::SL_STOPPING_3,
+ StopReq::getSystemStop(c_stopRec.stopReq.requestInfo));
+ updateNodeState(signal, newState);
+
+ c_stopRec.stopInitiatedTime = NdbTick_CurrentMillisecond();
+
+ signal->theData[0] = ZSHUTDOWN;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 100, 1);
+ }
+}
+
+void Ndbcntr::execABORT_ALL_REF(Signal* signal){
+ jamEntry();
+ ndbrequire(false);
+}
+
+void
+Ndbcntr::StopRecord::checkLqhTimeout_1(Signal* signal){
+ const Int32 timeout = stopReq.readOperationTimeout;
+ const NDB_TICKS alarm = stopInitiatedTime + (NDB_TICKS)timeout;
+ const NDB_TICKS now = NdbTick_CurrentMillisecond();
+
+ if((timeout >= 0 && now >= alarm)){
+ // || checkWithLqhInSomeMagicWay)
+ jam();
+
+ ChangeNodeStateReq * req = (ChangeNodeStateReq*)&signal->theData[0];
+
+ NodeState newState(NodeState::SL_STOPPING_4,
+ StopReq::getSystemStop(stopReq.requestInfo));
+ req->nodeState = newState;
+ req->senderRef = cntr.reference();
+ req->senderData = 12;
+ cntr.sendSignal(DBLQH_REF, GSN_CHANGE_NODE_STATE_REQ, signal, 2, JBB);
+ return;
+ }
+ signal->theData[0] = ZSHUTDOWN;
+ cntr.sendSignalWithDelay(cntr.reference(), GSN_CONTINUEB, signal, 100, 1);
+}
+
+void Ndbcntr::execCHANGE_NODE_STATE_CONF(Signal* signal){
+ jamEntry();
+ signal->theData[0] = reference();
+ signal->theData[1] = 12;
+ sendSignal(DBDIH_REF, GSN_STOP_ME_REQ, signal, 2, JBB);
+}
+
+void Ndbcntr::execSTOP_ME_REF(Signal* signal){
+ jamEntry();
+ ndbrequire(false);
+}
+
+
+void Ndbcntr::execSTOP_ME_CONF(Signal* signal){
+ jamEntry();
+
+ NodeState newState(NodeState::SL_STOPPING_4,
+ StopReq::getSystemStop(c_stopRec.stopReq.requestInfo));
+ updateNodeState(signal, newState);
+
+ c_stopRec.stopInitiatedTime = NdbTick_CurrentMillisecond();
+ signal->theData[0] = ZSHUTDOWN;
+ sendSignalWithDelay(cownBlockref, GSN_CONTINUEB, signal, 100, 1);
+}
+
+void
+Ndbcntr::StopRecord::checkLqhTimeout_2(Signal* signal){
+ const Int32 timeout = stopReq.operationTimeout;
+ const NDB_TICKS alarm = stopInitiatedTime + (NDB_TICKS)timeout;
+ const NDB_TICKS now = NdbTick_CurrentMillisecond();
+
+ if((timeout >= 0 && now >= alarm)){
+ // || checkWithLqhInSomeMagicWay)
+ jam();
+ if(StopReq::getPerformRestart(stopReq.requestInfo)){
+ jam();
+ StartOrd * startOrd = (StartOrd *)&signal->theData[0];
+ startOrd->restartInfo = stopReq.requestInfo;
+ cntr.sendSignal(CMVMI_REF, GSN_START_ORD, signal, 2, JBA);
+ } else {
+ jam();
+ cntr.sendSignal(CMVMI_REF, GSN_STOP_ORD, signal, 1, JBA);
+ }
+ return;
+ }
+ signal->theData[0] = ZSHUTDOWN;
+ cntr.sendSignalWithDelay(cntr.reference(), GSN_CONTINUEB, signal, 100, 1);
+}
+
+void Ndbcntr::execWAIT_GCP_REF(Signal* signal){
+ jamEntry();
+
+ //WaitGCPRef* const ref = (WaitGCPRef*)&signal->theData[0];
+
+ WaitGCPReq * req = (WaitGCPReq*)&signal->theData[0];
+ req->senderRef = reference();
+ req->senderData = 12;
+ req->requestType = WaitGCPReq::CompleteForceStart;
+ sendSignal(DBDIH_REF, GSN_WAIT_GCP_REQ, signal,
+ WaitGCPReq::SignalLength, JBB);
+}
+
+void Ndbcntr::execWAIT_GCP_CONF(Signal* signal){
+ jamEntry();
+
+ if(StopReq::getPerformRestart(c_stopRec.stopReq.requestInfo)){
+ jam();
+ StartOrd * startOrd = (StartOrd *)&signal->theData[0];
+ startOrd->restartInfo = c_stopRec.stopReq.requestInfo;
+ sendSignalWithDelay(CMVMI_REF, GSN_START_ORD, signal, 500,
+ StartOrd::SignalLength);
+ } else {
+ jam();
+ sendSignalWithDelay(CMVMI_REF, GSN_STOP_ORD, signal, 500, 1);
+ }
+ return;
+}
+
+void Ndbcntr::execSTTORRY(Signal* signal){
+ jamEntry();
+ c_missra.execSTTORRY(signal);
+}
+
+void Ndbcntr::execSTART_ORD(Signal* signal){
+ jamEntry();
+ ndbrequire(NO_OF_BLOCKS == ALL_BLOCKS_SZ);
+ c_missra.execSTART_ORD(signal);
+}
+
+void
+Ndbcntr::clearFilesystem(Signal* signal){
+ FsRemoveReq * req = (FsRemoveReq *)signal->getDataPtrSend();
+ req->userReference = reference();
+ req->userPointer = 0;
+ req->directory = 1;
+ req->ownDirectory = 1;
+ FsOpenReq::setVersion(req->fileNumber, 3);
+ FsOpenReq::setSuffix(req->fileNumber, FsOpenReq::S_CTL); // Can by any...
+ FsOpenReq::v1_setDisk(req->fileNumber, c_fsRemoveCount);
+ sendSignal(NDBFS_REF, GSN_FSREMOVEREQ, signal,
+ FsRemoveReq::SignalLength, JBA);
+ c_fsRemoveCount++;
+}
+
+void
+Ndbcntr::execFSREMOVEREF(Signal* signal){
+ jamEntry();
+ ndbrequire(0);
+}
+
+void
+Ndbcntr::execFSREMOVECONF(Signal* signal){
+ jamEntry();
+ if(c_fsRemoveCount == 13){
+ jam();
+ sendSttorry(signal);
+ } else {
+ jam();
+ ndbrequire(c_fsRemoveCount < 13);
+ clearFilesystem(signal);
+ }//if
+}
+
+void Ndbcntr::Missra::execSTART_ORD(Signal* signal){
+ signal->theData[0] = EventReport::NDBStartStarted;
+ signal->theData[1] = NDB_VERSION;
+ cntr.sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ currentStartPhase = 0;
+ for(Uint32 i = 0; i<NO_OF_BLOCKS; i++){
+ if(ALL_BLOCKS[i].NextSP < currentStartPhase)
+ currentStartPhase = ALL_BLOCKS[i].NextSP;
+ }
+
+ currentBlockIndex = 0;
+
+ sendNextSTTOR(signal);
+}
+
+void Ndbcntr::Missra::execSTTORRY(Signal* signal){
+ const BlockReference ref = signal->senderBlockRef();
+ ndbrequire(refToBlock(ref) == refToBlock(ALL_BLOCKS[currentBlockIndex].Ref));
+
+ /**
+ * Update next start phase
+ */
+ for (Uint32 i = 3; i < 25; i++){
+ jam();
+ if (signal->theData[i] > currentStartPhase){
+ jam();
+ ALL_BLOCKS[currentBlockIndex].NextSP = signal->theData[i];
+ break;
+ }
+ }
+
+ currentBlockIndex++;
+ sendNextSTTOR(signal);
+}
+
+void Ndbcntr::Missra::sendNextSTTOR(Signal* signal){
+
+ for(; currentStartPhase < 255 ; currentStartPhase++){
+ jam();
+
+ const Uint32 start = currentBlockIndex;
+
+ for(; currentBlockIndex < ALL_BLOCKS_SZ; currentBlockIndex++){
+ jam();
+ if(ALL_BLOCKS[currentBlockIndex].NextSP == currentStartPhase){
+ jam();
+ signal->theData[0] = 0;
+ signal->theData[1] = currentStartPhase;
+ signal->theData[2] = 0;
+ signal->theData[3] = 0;
+ signal->theData[4] = 0;
+ signal->theData[5] = 0;
+ signal->theData[6] = 0;
+ signal->theData[7] = cntr.ctypeOfStart;
+
+ const BlockReference ref = ALL_BLOCKS[currentBlockIndex].Ref;
+
+#ifdef MAX_STARTPHASE
+ ndbrequire(currentStartPhase <= MAX_STARTPHASE);
+#endif
+
+#ifdef TRACE_STTOR
+ ndbout_c("sending STTOR(%d) to %s(ref=%x index=%d)",
+ currentStartPhase,
+ getBlockName( refToBlock(ref)),
+ ref,
+ currentBlockIndex);
+#endif
+
+ cntr.sendSignal(ref, GSN_STTOR,
+ signal, 8, JBB);
+ return;
+ }
+ }
+
+ currentBlockIndex = 0;
+
+ if(start != 0){
+ /**
+ * At least one wanted this start phase, report it
+ */
+ jam();
+ signal->theData[0] = EventReport::StartPhaseCompleted;
+ signal->theData[1] = currentStartPhase;
+ signal->theData[2] = cntr.ctypeOfStart;
+ cntr.sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+ }
+ }
+
+ signal->theData[0] = EventReport::NDBStartCompleted;
+ signal->theData[1] = NDB_VERSION;
+ cntr.sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ NodeState newState(NodeState::SL_STARTED);
+ cntr.updateNodeState(signal, newState);
+}
diff --git a/ndb/src/kernel/blocks/ndbcntr/NdbcntrSysTable.cpp b/ndb/src/kernel/blocks/ndbcntr/NdbcntrSysTable.cpp
new file mode 100644
index 00000000000..40e6aa2dcd7
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbcntr/NdbcntrSysTable.cpp
@@ -0,0 +1,94 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Ndbcntr.hpp"
+
+#define arrayLength(x) sizeof(x)/sizeof(x[0])
+
+// SYSTAB_0
+
+static const Ndbcntr::SysColumn
+column_SYSTAB_0[] = {
+ { 0, "SYSKEY_0",
+ DictTabInfo::ExtUnsigned, 1,
+ true, false
+ },
+ { 1, "NEXTID",
+ DictTabInfo::ExtBigunsigned, 1,
+ false, false
+ }
+};
+
+const Ndbcntr::SysTable
+Ndbcntr::g_sysTable_SYSTAB_0 = {
+ "sys/def/SYSTAB_0",
+ arrayLength(column_SYSTAB_0), column_SYSTAB_0,
+ DictTabInfo::SystemTable,
+ DictTabInfo::AllNodesSmallTable,
+ true, ~0
+};
+
+// NDB$EVENTS_0
+
+static const Ndbcntr::SysColumn
+column_NDBEVENTS_0[] = {
+ { 0, "NAME",
+ DictTabInfo::ExtChar, MAX_TAB_NAME_SIZE,
+ true, false
+ },
+ { 1, "EVENT_TYPE",
+ DictTabInfo::ExtUnsigned, 1,
+ false, false
+ },
+ { 2, "TABLE_NAME",
+ DictTabInfo::ExtChar, MAX_TAB_NAME_SIZE,
+ false, false
+ },
+ { 3, "ATTRIBUTE_MASK",
+ DictTabInfo::ExtUnsigned, MAXNROFATTRIBUTESINWORDS,
+ false, false
+ },
+ { 4, "SUBID",
+ DictTabInfo::ExtUnsigned, 1,
+ false, false
+ },
+ { 5, "SUBKEY",
+ DictTabInfo::ExtUnsigned, 1,
+ false, false
+ }
+};
+
+const Ndbcntr::SysTable
+Ndbcntr::g_sysTable_NDBEVENTS_0 = {
+ "sys/def/NDB$EVENTS_0",
+ arrayLength(column_NDBEVENTS_0), column_NDBEVENTS_0,
+ DictTabInfo::SystemTable,
+ DictTabInfo::AllNodesSmallTable,
+ true, ~0
+};
+
+// all
+
+const Ndbcntr::SysTable*
+Ndbcntr::g_sysTableList[] = {
+ &g_sysTable_SYSTAB_0,
+ &g_sysTable_NDBEVENTS_0
+};
+
+//TODO Backup needs this info to allocate appropriate number of records
+//BackupInit.cpp
+const unsigned
+Ndbcntr::g_sysTableCount = arrayLength(Ndbcntr::g_sysTableList);
diff --git a/ndb/src/kernel/blocks/ndbfs/AsyncFile.cpp b/ndb/src/kernel/blocks/ndbfs/AsyncFile.cpp
new file mode 100644
index 00000000000..0e2aa4c6903
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/AsyncFile.cpp
@@ -0,0 +1,1050 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * O_DIRECT
+ */
+#ifdef NDB_LINUX
+#ifndef _GNU_SOURCE
+#define _GNU_SOURCE
+#endif
+#endif
+
+#include "Error.hpp"
+#include "AsyncFile.hpp"
+
+#include <ErrorHandlingMacros.hpp>
+#include <kernel_types.h>
+#include <string.h>
+#include <NdbMem.h>
+#include <NdbThread.h>
+#include <signaldata/FsOpenReq.hpp>
+
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <errno.h>
+
+#ifdef NDB_LINUX
+// This is for pread and pwrite
+#ifndef __USE_UNIX98
+#define __USE_UNIX98
+#endif
+#endif
+
+#include <NdbUnistd.h>
+#if defined NDB_WIN32 || defined NDB_OSE || defined NDB_SOFTOSE
+#include <NdbStdio.h>
+#else
+// For readv and writev
+#include <sys/uio.h>
+#endif
+
+#ifndef NDB_WIN32
+#include <dirent.h>
+#endif
+
+// Use this define if you want printouts from AsyncFile class
+//#define DEBUG_ASYNCFILE
+
+#ifdef DEBUG_ASYNCFILE
+#include <NdbOut.hpp>
+#define DEBUG(x) x
+#define PRINT_ERRORANDFLAGS(f) printErrorAndFlags(f)
+void printErrorAndFlags(Uint32 used_flags);
+#else
+#define DEBUG(x)
+#define PRINT_ERRORANDFLAGS(f)
+#endif
+
+// Define the size of the write buffer (for each thread)
+#if defined NDB_SOFTOSE || defined NDB_OSE
+#define WRITEBUFFERSIZE 65536
+#else
+#define WRITEBUFFERSIZE 262144
+#endif
+
+const char *actionName[] = {
+ "open",
+ "close",
+ "closeRemove",
+ "read",
+ "readv",
+ "write",
+ "writev",
+ "writeSync",
+ "writevSync",
+ "sync",
+ "end" };
+
+static int numAsyncFiles = 0;
+
+extern "C" void * runAsyncFile(void* arg)
+{
+ ((AsyncFile*)arg)->run();
+ return (NULL);
+}
+
+AsyncFile::AsyncFile() :
+ theFileName(),
+#ifdef NDB_WIN32
+ hFile(INVALID_HANDLE_VALUE),
+#else
+ theFd(-1),
+#endif
+ theReportTo(0),
+ theMemoryChannelPtr(NULL)
+{
+}
+
+void
+AsyncFile::doStart(const char * filesystemPath) {
+ theFileName.init(filesystemPath);
+
+ // Stacksize for filesystem threads
+ // An 8k stack should be enough
+ const NDB_THREAD_STACKSIZE stackSize = 8192;
+
+ char buf[16];
+ numAsyncFiles++;
+ snprintf(buf, sizeof(buf), "AsyncFile%d", numAsyncFiles);
+
+ theStartMutexPtr = NdbMutex_Create();
+ theStartConditionPtr = NdbCondition_Create();
+ NdbMutex_Lock(theStartMutexPtr);
+ theStartFlag = false;
+ theThreadPtr = NdbThread_Create(runAsyncFile,
+ (void**)this,
+ stackSize,
+ (char*)&buf,
+ NDB_THREAD_PRIO_MEAN);
+
+ NdbCondition_Wait(theStartConditionPtr,
+ theStartMutexPtr);
+ NdbMutex_Unlock(theStartMutexPtr);
+ NdbMutex_Destroy(theStartMutexPtr);
+ NdbCondition_Destroy(theStartConditionPtr);
+}
+
+AsyncFile::~AsyncFile()
+{
+ void *status;
+ Request request;
+ request.action = Request::end;
+ theMemoryChannelPtr->writeChannel( &request );
+ NdbThread_WaitFor(theThreadPtr, &status);
+ NdbThread_Destroy(&theThreadPtr);
+ delete theMemoryChannelPtr;
+}
+
+void
+AsyncFile::reportTo( MemoryChannel<Request> *reportTo )
+{
+ theReportTo = reportTo;
+}
+
+void AsyncFile::execute(Request* request)
+{
+ theMemoryChannelPtr->writeChannel( request );
+}
+
+void
+AsyncFile::run()
+{
+ Request *request;
+ // Create theMemoryChannel in the thread that will wait for it
+ NdbMutex_Lock(theStartMutexPtr);
+ theMemoryChannelPtr = new MemoryChannel<Request>();
+ theStartFlag = true;
+ // Create write buffer for bigger writes
+ theWriteBufferSize = WRITEBUFFERSIZE;
+ theWriteBuffer = (char *) NdbMem_Allocate(theWriteBufferSize);
+ NdbMutex_Unlock(theStartMutexPtr);
+ NdbCondition_Signal(theStartConditionPtr);
+
+ if (!theWriteBuffer) {
+ DEBUG(ndbout_c("AsyncFile::writeReq, Failed allocating write buffer"));
+ return;
+ }//if
+
+ while (1) {
+ request = theMemoryChannelPtr->readChannel();
+ if (!request) {
+ DEBUG(ndbout_c("Nothing read from Memory Channel in AsyncFile"));
+ endReq();
+ return;
+ }//if
+ switch (request->action) {
+ case Request:: open:
+ openReq(request);
+ break;
+ case Request:: close:
+ closeReq(request);
+ break;
+ case Request:: closeRemove:
+ closeReq(request);
+ removeReq(request);
+ break;
+ case Request:: read:
+ readReq(request);
+ break;
+ case Request:: readv:
+ readvReq(request);
+ break;
+ case Request:: write:
+ writeReq(request);
+ break;
+ case Request:: writev:
+ writevReq(request);
+ break;
+ case Request:: writeSync:
+ writeReq(request);
+ syncReq(request);
+ break;
+ case Request:: writevSync:
+ writevReq(request);
+ syncReq(request);
+ break;
+ case Request:: sync:
+ syncReq(request);
+ break;
+ case Request:: append:
+ appendReq(request);
+ break;
+ case Request::rmrf:
+ rmrfReq(request, (char*)theFileName.c_str(), request->par.rmrf.own_directory);
+ break;
+ case Request:: end:
+ closeReq(request);
+ endReq();
+ return;
+ default:
+ THREAD_REQUIRE(false, "Using default switch in AsyncFile::run");
+ break;
+ }//switch
+ theReportTo->writeChannel(request);
+ }//while
+}//AsyncFile::run()
+
+extern bool Global_useO_SYNC;
+extern bool Global_useO_DIRECT;
+extern bool Global_unlinkO_CREAT;
+extern Uint32 Global_syncFreq;
+
+void AsyncFile::openReq(Request* request)
+{
+ m_openedWithSync = false;
+ m_syncFrequency = 0;
+
+ // for open.flags, see signal FSOPENREQ
+#ifdef NDB_WIN32
+ DWORD dwCreationDisposition;
+ DWORD dwDesiredAccess = 0;
+ DWORD dwShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE;
+ DWORD dwFlagsAndAttributes = FILE_ATTRIBUTE_NORMAL | FILE_FLAG_RANDOM_ACCESS | FILE_FLAG_NO_BUFFERING;
+ const Uint32 flags = request->par.open.flags;
+
+ // Convert file open flags from Solaris to Windows
+ if ((flags & FsOpenReq::OM_CREATE) && (flags & FsOpenReq::OM_TRUNCATE)){
+ dwCreationDisposition = CREATE_ALWAYS;
+ } else if (flags & FsOpenReq::OM_TRUNCATE){
+ dwCreationDisposition = TRUNCATE_EXISTING;
+ } else if (flags & FsOpenReq::OM_CREATE){
+ dwCreationDisposition = CREATE_NEW;
+ } else {
+ dwCreationDisposition = OPEN_EXISTING;
+ }
+
+ switch(flags & 3){
+ case FsOpenReq::OM_READONLY:
+ dwDesiredAccess = GENERIC_READ;
+ break;
+ case FsOpenReq::OM_WRITEONLY:
+ dwDesiredAccess = GENERIC_WRITE;
+ break;
+ case FsOpenReq::OM_READWRITE:
+ dwDesiredAccess = GENERIC_READ | GENERIC_WRITE;
+ break;
+ default:
+ request->error = 1000;
+ break;
+ return;
+ }
+
+ hFile = CreateFile(theFileName.c_str(), dwDesiredAccess, dwShareMode,
+ 0, dwCreationDisposition, dwFlagsAndAttributes, 0);
+
+ if(INVALID_HANDLE_VALUE == hFile) {
+ request->error = GetLastError();
+ if(((ERROR_PATH_NOT_FOUND == request->error) || (ERROR_INVALID_NAME == request->error))
+ && (flags & FsOpenReq::OM_CREATE)) {
+ createDirectories();
+ hFile = CreateFile(theFileName.c_str(), dwDesiredAccess, dwShareMode,
+ 0, dwCreationDisposition, dwFlagsAndAttributes, 0);
+
+ if(INVALID_HANDLE_VALUE == hFile)
+ request->error = GetLastError();
+ else
+ request->error = 0;
+
+ return;
+ }
+ }
+ else {
+ request->error = 0;
+ return;
+ }
+#else
+ const Uint32 flags = request->par.open.flags;
+ Uint32 new_flags = 0;
+
+ // Convert file open flags from Solaris to Liux
+ if(flags & FsOpenReq::OM_CREATE){
+ new_flags |= O_CREAT;
+ }
+
+ if(flags & FsOpenReq::OM_TRUNCATE){
+#if 0
+ if(Global_unlinkO_CREAT){
+ unlink(theFileName.c_str());
+ } else
+#endif
+ new_flags |= O_TRUNC;
+ }
+
+ if(flags & FsOpenReq::OM_APPEND){
+ new_flags |= O_APPEND;
+ }
+
+ if(flags & FsOpenReq::OM_SYNC){
+#if 0
+ if(Global_useO_SYNC){
+ new_flags |= O_SYNC;
+ m_openedWithSync = true;
+ m_syncFrequency = 0;
+ } else {
+#endif
+ m_openedWithSync = false;
+ m_syncCount = 0;
+ m_syncFrequency = Global_syncFreq;
+#if 0
+ }
+#endif
+ } else {
+ m_openedWithSync = false;
+ m_syncFrequency = 0;
+ }
+
+#if 0
+#if NDB_LINUX
+ if(Global_useO_DIRECT){
+ new_flags |= O_DIRECT;
+ }
+#endif
+#endif
+
+ switch(flags & 0x3){
+ case FsOpenReq::OM_READONLY:
+ new_flags |= O_RDONLY;
+ break;
+ case FsOpenReq::OM_WRITEONLY:
+ new_flags |= O_WRONLY;
+ break;
+ case FsOpenReq::OM_READWRITE:
+ new_flags |= O_RDWR;
+ break;
+ default:
+ request->error = 1000;
+ break;
+ return;
+ }
+ const int mode = S_IRUSR | S_IWUSR | S_IRGRP;
+
+ if (-1 == (theFd = ::open(theFileName.c_str(), new_flags, mode))) {
+ PRINT_ERRORANDFLAGS(new_flags);
+ if( (errno == ENOENT ) && (new_flags & O_CREAT ) ) {
+ createDirectories();
+ if (-1 == (theFd = ::open(theFileName.c_str(), new_flags, mode))) {
+ PRINT_ERRORANDFLAGS(new_flags);
+ request->error = errno;
+ }
+ } else {
+ request->error = errno;
+ }
+ }
+#endif
+}
+
+int
+AsyncFile::readBuffer(char * buf, size_t size, off_t offset){
+ int return_value;
+
+#ifdef NDB_WIN32
+ DWORD dwSFP = SetFilePointer(hFile, offset, 0, FILE_BEGIN);
+ if(dwSFP != offset) {
+ return GetLastError();
+ }
+#elif defined NDB_OSE || defined NDB_SOFTOSE
+ return_value = lseek(theFd, offset, SEEK_SET);
+ if (return_value != offset) {
+ return errno;
+ }
+#endif
+
+ while (size > 0) {
+ size_t bytes_read = 0;
+
+#ifdef NDB_WIN32
+ DWORD dwBytesRead;
+ BOOL bRead = ReadFile(hFile,
+ buf,
+ size,
+ &dwBytesRead,
+ 0);
+ if(!bRead){
+ return GetLastError();
+ }
+ bytes_read = dwBytesRead;
+#elif defined NDB_OSE || defined NDB_SOFTOSE
+ return_value = ::read(theFd, buf, size);
+#else // UNIX
+ return_value = ::pread(theFd, buf, size, offset);
+#endif
+#ifndef NDB_WIN32
+ if (return_value == -1 && errno == EINTR) {
+ DEBUG(ndbout_c("EINTR in read"));
+ continue;
+ } else if (return_value == -1){
+ return errno;
+ } else {
+ bytes_read = return_value;
+ }
+#endif
+
+ if(bytes_read == 0){
+ DEBUG(ndbout_c("Read underflow %d %d\n %x\n%d %d",
+ size, offset, buf, bytes_read, return_value));
+ return ERR_ReadUnderflow;
+ }
+
+ if(bytes_read != size){
+ DEBUG(ndbout_c("Warning partial read %d != %d",
+ bytes_read, size));
+ }
+
+ buf += bytes_read;
+ size -= bytes_read;
+ offset += bytes_read;
+ }
+ return 0;
+}
+
+void
+AsyncFile::readReq( Request * request)
+{
+ for(int i = 0; i < request->par.readWrite.numberOfPages ; i++) {
+ off_t offset = request->par.readWrite.pages[i].offset;
+ size_t size = request->par.readWrite.pages[i].size;
+ char * buf = request->par.readWrite.pages[i].buf;
+
+ int err = readBuffer(buf, size, offset);
+ if(err != 0){
+ request->error = err;
+ return;
+ }
+ }
+}
+
+void
+AsyncFile::readvReq( Request * request)
+{
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ readReq(request);
+ return;
+#elif defined NDB_WIN32
+ // ReadFileScatter?
+ readReq(request);
+ return;
+#else
+ int return_value;
+ int length = 0;
+ struct iovec iov[20]; // the parameter in the signal restricts this to 20 deep
+ for(int i=0; i < request->par.readWrite.numberOfPages ; i++) {
+ iov[i].iov_base= request->par.readWrite.pages[i].buf;
+ iov[i].iov_len= request->par.readWrite.pages[i].size;
+ length = length + iov[i].iov_len;
+ }
+ lseek( theFd, request->par.readWrite.pages[0].offset, SEEK_SET );
+ return_value = ::readv(theFd, iov, request->par.readWrite.numberOfPages);
+ if (return_value == -1) {
+ request->error = errno;
+ return;
+ } else if (return_value != length) {
+ request->error = 1011;
+ return;
+ }
+#endif
+}
+
+int
+AsyncFile::extendfile(Request* request) {
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ // Find max size of this file in this request
+ int maxOffset = 0;
+ int maxSize = 0;
+ for(int i=0; i < request->par.readWrite.numberOfPages ; i++) {
+ if (request->par.readWrite.pages[i].offset > maxOffset) {
+ maxOffset = request->par.readWrite.pages[i].offset;
+ maxSize = request->par.readWrite.pages[i].size;
+ }
+ }
+ DEBUG(ndbout_c("extendfile: maxOffset=%d, size=%d", maxOffset, maxSize));
+
+ // Allocate a buffer and fill it with zeros
+ void* pbuf = malloc(maxSize);
+ memset(pbuf, 0, maxSize);
+ for (int p = 0; p <= maxOffset; p = p + maxSize) {
+ int return_value;
+ return_value = lseek(theFd,
+ p,
+ SEEK_SET);
+ if((return_value == -1 ) || (return_value != p)) {
+ return -1;
+ }
+ return_value = ::write(theFd,
+ pbuf,
+ maxSize);
+ if ((return_value == -1) || (return_value != maxSize)) {
+ return -1;
+ }
+ }
+ free(pbuf);
+
+ DEBUG(ndbout_c("extendfile: \"%s\" OK!", theFileName.c_str()));
+ return 0;
+#else
+ request = request;
+ abort();
+ return -1;
+#endif
+}
+
+void
+AsyncFile::writeReq( Request * request)
+{
+ int page_num = 0;
+ bool write_not_complete = true;
+
+ while(write_not_complete) {
+ int totsize = 0;
+ off_t offset = request->par.readWrite.pages[page_num].offset;
+ char* bufptr = theWriteBuffer;
+
+ write_not_complete = false;
+ if (request->par.readWrite.numberOfPages > 1) {
+ off_t page_offset = offset;
+
+ // Multiple page write, copy to buffer for one write
+ for(int i=page_num; i < request->par.readWrite.numberOfPages; i++) {
+ memcpy(bufptr,
+ request->par.readWrite.pages[i].buf,
+ request->par.readWrite.pages[i].size);
+ bufptr += request->par.readWrite.pages[i].size;
+ totsize += request->par.readWrite.pages[i].size;
+ if (((i + 1) < request->par.readWrite.numberOfPages)) {
+ // There are more pages to write
+ // Check that offsets are consequtive
+ if ((page_offset + request->par.readWrite.pages[i].size)
+ !=
+ request->par.readWrite.pages[i+1].offset) {
+ // Next page is not aligned with previous, not allowed
+ DEBUG(ndbout_c("Page offsets are not aligned"));
+ request->error = EINVAL;
+ return;
+ }
+ if ((unsigned)(totsize + request->par.readWrite.pages[i+1].size) > (unsigned)theWriteBufferSize) {
+ // We are not finished and the buffer is full
+ write_not_complete = true;
+ // Start again with next page
+ page_num = i + 1;
+ break;
+ }
+ }
+ page_offset += request->par.readWrite.pages[i].size;
+ }
+ bufptr = theWriteBuffer;
+ } else {
+ // One page write, write page directly
+ bufptr = request->par.readWrite.pages[0].buf;
+ totsize = request->par.readWrite.pages[0].size;
+ }
+ int err = writeBuffer(bufptr, totsize, offset);
+ if(err != 0){
+ request->error = err;
+ return;
+ }
+ } // while(write_not_complete)
+}
+
+int
+AsyncFile::writeBuffer(const char * buf, size_t size, off_t offset,
+ size_t chunk_size)
+{
+ size_t bytes_to_write = chunk_size;
+ int return_value;
+
+#ifdef NDB_WIN32
+ DWORD dwSFP = SetFilePointer(hFile, offset, 0, FILE_BEGIN);
+ if(dwSFP != offset) {
+ return GetLastError();
+ }
+#elif defined NDB_OSE || defined NDB_SOFTOSE
+ return_value = lseek(theFd, offset, SEEK_SET);
+ if (return_value != offset) {
+ DEBUG(ndbout_c("AsyncFile::writeReq, err1: return_value=%d, offset=%d\n",
+ return_value, chunk_offset));
+ PRINT_ERRORANDFLAGS(0);
+ if (errno == 78) {
+ // Could not write beyond end of file, try to extend file
+ DEBUG(ndbout_c("AsyncFile::writeReq, Extend. file! filename=\"%s\" \n",
+ theFileName.c_str()));
+ return_value = extendfile(request);
+ if (return_value == -1) {
+ return errno;
+ }
+ return_value = lseek(theFd, offset, SEEK_SET);
+ if (return_value != offset) {
+ return errno;
+ }
+ } else {
+ return errno;
+ }
+ }
+#endif
+
+ while (size > 0) {
+ if (size < bytes_to_write){
+ // We are at the last chunk
+ bytes_to_write = size;
+ }
+ size_t bytes_written = 0;
+
+#ifdef NDB_WIN32
+ DWORD dwWritten;
+ BOOL bWrite = WriteFile(hFile, buf, bytes_to_write, &dwWritten, 0);
+ if(!bWrite) {
+ return GetLastError();
+ }
+ bytes_written = dwWritten;
+ if (bytes_written != bytes_to_write) {
+ DEBUG(ndbout_c("Warning partial write %d != %d", bytes_written, bytes_to_write));
+ }
+
+#elif defined NDB_OSE || defined NDB_SOFTOSE
+ return_value = ::write(theFd, buf, bytes_to_write);
+#else // UNIX
+ return_value = ::pwrite(theFd, buf, bytes_to_write, offset);
+#endif
+#ifndef NDB_WIN32
+ if (return_value == -1 && errno == EINTR) {
+ bytes_written = 0;
+ DEBUG(ndbout_c("EINTR in write"));
+ } else if (return_value == -1){
+ return errno;
+ } else {
+ bytes_written = return_value;
+
+ if(bytes_written == 0){
+ abort();
+ }
+
+ if(bytes_written != bytes_to_write){
+ DEBUG(ndbout_c("Warning partial write %d != %d",
+ bytes_written, bytes_to_write));
+ }
+ }
+#endif
+
+ buf += bytes_written;
+ size -= bytes_written;
+ offset += bytes_written;
+ }
+ return 0;
+}
+
+void
+AsyncFile::writevReq( Request * request)
+{
+ // WriteFileGather on WIN32?
+ writeReq(request);
+}
+
+
+void
+AsyncFile::closeReq(Request * request)
+{
+ syncReq(request);
+#ifdef NDB_WIN32
+ if(!CloseHandle(hFile)) {
+ request->error = GetLastError();
+ }
+ hFile = INVALID_HANDLE_VALUE;
+#else
+ if (-1 == ::close(theFd)) {
+ request->error = errno;
+ }
+ theFd = -1;
+#endif
+}
+
+bool AsyncFile::isOpen(){
+#ifdef NDB_WIN32
+ return (hFile != INVALID_HANDLE_VALUE);
+#else
+ return (theFd != -1);
+#endif
+}
+
+
+void
+AsyncFile::syncReq(Request * request)
+{
+ if(m_openedWithSync){
+ return;
+ }
+#ifdef NDB_WIN32
+ if(!FlushFileBuffers(hFile)) {
+ request->error = GetLastError();
+ return;
+ }
+#else
+ if (-1 == ::fsync(theFd)){
+ request->error = errno;
+ return;
+ }
+#endif
+ m_syncCount = 0;
+}
+
+void
+AsyncFile::appendReq(Request * request){
+
+ const char * buf = request->par.append.buf;
+ Uint32 size = request->par.append.size;
+
+ m_syncCount += size;
+
+#ifdef NDB_WIN32
+ DWORD dwWritten = 0;
+ while(size > 0){
+ if(!WriteFile(hFile, buf, size, &dwWritten, 0)){
+ request->error = GetLastError();
+ return ;
+ }
+
+ buf += dwWritten;
+ size -= dwWritten;
+ }
+#else
+ while(size > 0){
+ const int n = write(theFd, buf, size);
+ if(n == -1 && errno == EINTR){
+ continue;
+ }
+ if(n == -1){
+ request->error = errno;
+ return;
+ }
+ if(n == 0){
+ abort();
+ }
+ size -= n;
+ buf += n;
+ }
+#endif
+
+ if(m_syncFrequency != 0 && m_syncCount > m_syncFrequency){
+ syncReq(request);
+ request->error = 0;
+ }
+}
+
+void
+AsyncFile::removeReq(Request * request)
+{
+#ifdef NDB_WIN32
+ if(!DeleteFile(theFileName.c_str())) {
+ request->error = GetLastError();
+ }
+#else
+ if (-1 == ::remove(theFileName.c_str())) {
+ request->error = errno;
+
+ }
+#endif
+}
+
+void
+AsyncFile::rmrfReq(Request * request, char * path, bool removePath){
+ Uint32 path_len = strlen(path);
+ Uint32 path_max_copy = PATH_MAX - path_len;
+ char* path_add = &path[path_len];
+#ifndef NDB_WIN32
+ if(!request->par.rmrf.directory){
+ // Remove file
+ if(unlink((const char *)path) != 0 && errno != ENOENT)
+ request->error = errno;
+ return;
+ }
+ // Remove directory
+ DIR* dirp = opendir((const char *)path);
+ if(dirp == 0){
+ if(errno != ENOENT)
+ request->error = errno;
+ return;
+ }
+ struct dirent * dp;
+ while ((dp = readdir(dirp)) != NULL){
+ if ((strcmp(".", dp->d_name) != 0) && (strcmp("..", dp->d_name) != 0)) {
+ snprintf(path_add, (size_t)path_max_copy, "%s%s",
+ DIR_SEPARATOR, dp->d_name);
+ if(remove((const char*)path) == 0){
+ path[path_len] = 0;
+ continue;
+ }
+
+ rmrfReq(request, path, true);
+ path[path_len] = 0;
+ if(request->error != 0){
+ closedir(dirp);
+ return;
+ }
+ }
+ }
+ closedir(dirp);
+ if(removePath && rmdir((const char *)path) != 0){
+ request->error = errno;
+ }
+ return;
+#else
+
+ if(!request->par.rmrf.directory){
+ // Remove file
+ if(!DeleteFile(path)){
+ DWORD dwError = GetLastError();
+ if(dwError!=ERROR_FILE_NOT_FOUND)
+ request->error = dwError;
+ }
+ return;
+ }
+
+ strcat(path, "\\*");
+ WIN32_FIND_DATA ffd;
+ HANDLE hFindFile = FindFirstFile(path, &ffd);
+ path[path_len] = 0;
+ if(INVALID_HANDLE_VALUE==hFindFile){
+ DWORD dwError = GetLastError();
+ if(dwError!=ERROR_PATH_NOT_FOUND)
+ request->error = dwError;
+ return;
+ }
+
+ do {
+ if(0!=strcmp(".", ffd.cFileName) && 0!=strcmp("..", ffd.cFileName)){
+ strcat(path, "\\");
+ strcat(path, ffd.cFileName);
+ if(DeleteFile(path)) {
+ path[path_len] = 0;
+ continue;
+ }//if
+
+ rmrfReq(request, path, true);
+ path[path_len] = 0;
+ if(request->error != 0){
+ FindClose(hFindFile);
+ return;
+ }
+ }
+ } while(FindNextFile(hFindFile, &ffd));
+
+ FindClose(hFindFile);
+
+ if(removePath && !RemoveDirectory(path))
+ request->error = GetLastError();
+
+#endif
+}
+
+void AsyncFile::endReq()
+{
+ // Thread is ended with return
+ if (theWriteBuffer) NdbMem_Free(theWriteBuffer);
+ NdbThread_Exit(0);
+}
+
+
+void AsyncFile::createDirectories()
+{
+ for (int i = 0; i < theFileName.levels(); i++) {
+#ifdef NDB_WIN32
+ CreateDirectory(theFileName.directory(i), 0);
+#else
+ //printf("AsyncFile::createDirectories : \"%s\"\n", theFileName.directory(i));
+ mkdir(theFileName.directory(i), S_IRUSR | S_IWUSR | S_IXUSR | S_IXGRP | S_IRGRP);
+#endif
+ }
+}
+
+#ifdef DEBUG_ASYNCFILE
+void printErrorAndFlags(Uint32 used_flags) {
+ char buf[255];
+ sprintf(buf, "PEAF: errno=%d \"", errno);
+
+ switch(errno) {
+ case EACCES:
+ strcat(buf, "EACCES");
+ break;
+ case EDQUOT:
+ strcat(buf, "EDQUOT");
+ break;
+ case EEXIST :
+ strcat(buf, "EEXIST");
+ break;
+ case EINTR :
+ strcat(buf, "EINTR");
+ break;
+ case EFAULT :
+ strcat(buf, "EFAULT");
+ break;
+ case EIO :
+ strcat(buf, "EIO");
+ break;
+ case EISDIR :
+ strcat(buf, "EISDIR");
+ break;
+ case ELOOP :
+ strcat(buf, "ELOOP");
+ break;
+ case EMFILE :
+ strcat(buf, "EMFILE");
+ break;
+ case ENFILE :
+ strcat(buf, "ENFILE");
+ break;
+ case ENOENT :
+ strcat(buf, "ENOENT ");
+ break;
+ case ENOSPC :
+ strcat(buf, "ENOSPC");
+ break;
+ case ENOTDIR :
+ strcat(buf, "ENOTDIR");
+ break;
+ case ENXIO :
+ strcat(buf, "ENXIO");
+ break;
+ case EOPNOTSUPP:
+ strcat(buf, "EOPNOTSUPP");
+ break;
+#if !defined NDB_OSE && !defined NDB_SOFTOSE
+ case EMULTIHOP :
+ strcat(buf, "EMULTIHOP");
+ break;
+ case ENOLINK :
+ strcat(buf, "ENOLINK");
+ break;
+ case ENOSR :
+ strcat(buf, "ENOSR");
+ break;
+ case EOVERFLOW :
+ strcat(buf, "EOVERFLOW");
+ break;
+#endif
+ case EROFS :
+ strcat(buf, "EROFS");
+ break;
+ case EAGAIN :
+ strcat(buf, "EAGAIN");
+ break;
+ case EINVAL :
+ strcat(buf, "EINVAL");
+ break;
+ case ENOMEM :
+ strcat(buf, "ENOMEM");
+ break;
+ case ETXTBSY :
+ strcat(buf, "ETXTBSY");
+ break;
+ case ENAMETOOLONG:
+ strcat(buf, "ENAMETOOLONG");
+ break;
+ case EBADF:
+ strcat(buf, "EBADF");
+ break;
+ case ESPIPE:
+ strcat(buf, "ESPIPE");
+ break;
+ case ESTALE:
+ strcat(buf, "ESTALE");
+ break;
+ default:
+ strcat(buf, "EOTHER");
+ break;
+ }
+ strcat(buf, "\" ");
+#if defined NDB_OSE
+ strcat(buf, strerror(errno) << " ");
+#endif
+ strcat(buf, " flags: ");
+ switch(used_flags & 3){
+ case O_RDONLY:
+ strcat(buf, "O_RDONLY, ");
+ break;
+ case O_WRONLY:
+ strcat(buf, "O_WRONLY, ");
+ break;
+ case O_RDWR:
+ strcat(buf, "O_RDWR, ");
+ break;
+ default:
+ strcat(buf, "Unknown!!, ");
+ }
+
+ if((used_flags & O_APPEND)==O_APPEND)
+ strcat(buf, "O_APPEND, ");
+ if((used_flags & O_CREAT)==O_CREAT)
+ strcat(buf, "O_CREAT, ");
+ if((used_flags & O_EXCL)==O_EXCL)
+ strcat(buf, "O_EXCL, ");
+ if((used_flags & O_NOCTTY) == O_NOCTTY)
+ strcat(buf, "O_NOCTTY, ");
+ if((used_flags & O_NONBLOCK)==O_NONBLOCK)
+ strcat(buf, "O_NONBLOCK, ");
+ if((used_flags & O_TRUNC)==O_TRUNC)
+ strcat(buf, "O_TRUNC, ");
+#if !defined NDB_OSE && !defined NDB_SOFTOSE
+ if((used_flags & O_DSYNC)==O_DSYNC)
+ strcat(buf, "O_DSYNC, ");
+ if((used_flags & O_NDELAY)==O_NDELAY)
+ strcat(buf, "O_NDELAY, ");
+ if((used_flags & O_RSYNC)==O_RSYNC)
+ strcat(buf, "O_RSYNC, ");
+ if((used_flags & O_SYNC)==O_SYNC)
+ strcat(buf, "O_SYNC, ");
+ DEBUG(ndbout_c(buf));
+#endif
+
+}
+#endif
diff --git a/ndb/src/kernel/blocks/ndbfs/AsyncFile.hpp b/ndb/src/kernel/blocks/ndbfs/AsyncFile.hpp
new file mode 100644
index 00000000000..caa03e52d0c
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/AsyncFile.hpp
@@ -0,0 +1,234 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef AsyncFile_H
+#define AsyncFile_H
+
+//===========================================================================
+//
+// .DESCRIPTION
+// Asynchronous file, All actions are executed concurrently with other
+// activity of the process.
+// Because all action are performed in a seperated thread the result of
+// of a action is send back tru a memory channel.
+// For the asyncronise notivication of a finished request all the calls
+// have a request as paramater, the user can use the userData pointer
+// to add information it needs when the request is send back.
+//
+//
+// .TYPICAL USE:
+// Writing or reading data to/from disk concurrently to other activities.
+//
+//===========================================================================
+//=============================================================================
+//
+// .PUBLIC
+//
+//=============================================================================
+///////////////////////////////////////////////////////////////////////////////
+//
+// AsyncFile( );
+// Description:
+// Initialisation of the class.
+// Parameters:
+// -
+///////////////////////////////////////////////////////////////////////////////
+//
+// ~AsyncFile( );
+// Description:
+// Tell the thread to stop and wait for it to return
+// Parameters:
+// -
+///////////////////////////////////////////////////////////////////////////////
+//
+// doStart( );
+// Description:
+// Spawns the new thread.
+// Parameters:
+// Base path of filesystem
+//
+///////////////////////////////////////////////////////////////////////////////
+//
+// void execute(Request *request);
+// Description:
+// performens the requered action.
+// Parameters:
+// request: request to be called when open is finished.
+// action= open|close|read|write|sync
+// if action is open then:
+// par.open.flags= UNIX open flags, see man open
+// par.open.name= name of the file to open
+// if action is read or write then:
+// par.readWrite.buf= user provided buffer to read/write
+// the data from/to
+// par.readWrite.size= how many bytes must be read/written
+// par.readWrite.offset= absolute offset in file in bytes
+// return:
+// return values are stored in the request error field:
+// error= return state of the action, UNIX error see man open/errno
+// userData= is untouched can be used be user.
+//
+///////////////////////////////////////////////////////////////////////////////
+//
+// void reportTo( MemoryChannel<Request> *reportTo );
+// Description:
+// set the channel where the file must report the result of the
+// actions back to.
+// Parameters:
+// reportTo: the memory channel to use use MemoryChannelMultipleWriter
+// if more
+// than one file uses this channel to report back.
+//
+///////////////////////////////////////////////////////////////////////////////
+
+#include <kernel_types.h>
+#include "MemoryChannel.hpp"
+#include "Filename.hpp"
+
+const int ERR_ReadUnderflow = 1000;
+
+const int WRITECHUNK = 262144;
+
+class AsyncFile;
+
+class Request
+{
+public:
+ enum Action {
+ open,
+ close,
+ closeRemove,
+ read, // Allways leave readv directly after
+ // read because SimblockAsyncFileSystem depends on it
+ readv,
+ write,// Allways leave writev directly after
+ // write because SimblockAsyncFileSystem depends on it
+ writev,
+ writeSync,// Allways leave writevSync directly after
+ // writeSync because SimblockAsyncFileSystem depends on it
+ writevSync,
+ sync,
+ end,
+ append,
+ rmrf
+ };
+ Action action;
+ union {
+ struct {
+ Uint32 flags;
+ } open;
+ struct {
+ int numberOfPages;
+ struct{
+ char *buf;
+ size_t size;
+ off_t offset;
+ } pages[16];
+ } readWrite;
+ struct {
+ const char * buf;
+ size_t size;
+ } append;
+ struct {
+ bool directory;
+ bool own_directory;
+ } rmrf;
+ } par;
+ int error;
+
+ void set(BlockReference userReference,
+ Uint32 userPointer,
+ Uint16 filePointer);
+ BlockReference theUserReference;
+ Uint32 theUserPointer;
+ Uint16 theFilePointer;
+ // Information for open, needed if the first open action fails.
+ AsyncFile* file;
+ Uint32 theTrace;
+};
+
+
+inline
+void
+Request::set(BlockReference userReference,
+ Uint32 userPointer, Uint16 filePointer)
+{
+ theUserReference= userReference;
+ theUserPointer= userPointer;
+ theFilePointer= filePointer;
+}
+
+class AsyncFile
+{
+public:
+ AsyncFile();
+ ~AsyncFile();
+
+ void reportTo( MemoryChannel<Request> *reportTo );
+
+ void execute( Request* request );
+
+ void doStart(const char * fspath);
+ // its a thread so its always running
+ void run();
+
+ bool isOpen();
+
+ Filename theFileName;
+private:
+
+ void openReq(Request *request);
+ void readReq(Request *request);
+ void readvReq(Request *request);
+ void writeReq(Request *request);
+ void writevReq(Request *request);
+
+ void closeReq(Request *request);
+ void syncReq(Request *request);
+ void removeReq(Request *request);
+ void appendReq(Request *request);
+ void rmrfReq(Request *request, char * path, bool removePath);
+ void endReq();
+
+ int readBuffer(char * buf, size_t size, off_t offset);
+ int writeBuffer(const char * buf, size_t size, off_t offset,
+ size_t chunk_size = WRITECHUNK);
+
+ int extendfile(Request* request);
+ void createDirectories();
+
+#ifdef NDB_WIN32
+ HANDLE hFile;
+#else
+ int theFd;
+#endif
+
+ MemoryChannel<Request> *theReportTo;
+ MemoryChannel<Request>* theMemoryChannelPtr;
+
+ struct NdbThread* theThreadPtr;
+ NdbMutex* theStartMutexPtr;
+ NdbCondition* theStartConditionPtr;
+ bool theStartFlag;
+ int theWriteBufferSize;
+ char* theWriteBuffer;
+
+ bool m_openedWithSync;
+ Uint32 m_syncCount;
+ Uint32 m_syncFrequency;
+};
+
+#endif
diff --git a/ndb/src/kernel/blocks/ndbfs/AsyncFileTest/AsyncFileTest.cpp b/ndb/src/kernel/blocks/ndbfs/AsyncFileTest/AsyncFileTest.cpp
new file mode 100644
index 00000000000..b9954ba130f
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/AsyncFileTest/AsyncFileTest.cpp
@@ -0,0 +1,697 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#define TESTDEBUG 1
+
+#include <sys/stat.h>
+#include <fcntl.h>
+
+#include <string.h>
+#include <kernel_types.h>
+#include <Pool.hpp>
+#include "AsyncFile.hpp"
+#include "NdbOut.hpp"
+#include "NdbTick.h"
+#include "NdbThread.h"
+#include "NdbMain.h"
+
+// Test and benchmark functionality of AsyncFile
+// -n Number of files
+// -r Number of simultaneous requests
+// -s Filesize, number of pages
+// -l Number of iterations
+// -remove, remove files after close
+// -reverse, write files in reverse order, start with the last page
+
+#define MAXFILES 255
+#define DEFAULT_NUM_FILES 1
+#define MAXREQUESTS 256
+#define DEFAULT_NUM_REQUESTS 1
+#define MAXFILESIZE 4096
+#define DEFAULT_FILESIZE 2048
+#define FVERSION 0x01000000
+#define PAGESIZE 8192
+
+#define TIMER_START { Uint64 starttick = NdbTick_CurrentMillisecond()
+#define TIMER_PRINT(str, ops) Uint64 stoptick = NdbTick_CurrentMillisecond();\
+ Uint64 totaltime = (stoptick-starttick); \
+ ndbout << ops << " " << str << \
+ " total time " << (int)totaltime << "ms" << endl;\
+ char buf[255];\
+ sprintf(buf, "%d %s/sec\n",(int)((ops*1000)/totaltime), str);\
+ ndbout <<buf << endl;}
+
+static int numberOfFiles = DEFAULT_NUM_FILES;
+static int numberOfRequests = DEFAULT_NUM_REQUESTS;
+static int fileSize = DEFAULT_FILESIZE;
+static int removeFiles = 0;
+static int writeFilesReverse = 0;
+static int numberOfIterations = 1;
+Uint32 FileNameArray[4];
+
+Pool<AsyncFile>* files;
+AsyncFile* openFiles[MAXFILES];
+Pool<Request>* theRequestPool;
+MemoryChannelMultipleWriter<Request>* theReportChannel;
+
+char WritePages[MAXFILES][PAGESIZE];
+char ReadPages[MAXFILES][PAGESIZE];
+
+int readArguments(int argc, const char** argv);
+int openFile(int fileNum);
+int openFileWait();
+int closeFile(int fileNum);
+int closeFileWait();
+int writeFile( int fileNum, int pagenum);
+int writeFileWait();
+int writeSyncFile( int fileNum, int pagenum);
+int writeSyncFileWait();
+int readFile( int fileNum, int pagenum);
+int readFileWait();
+
+
+NDB_COMMAND(aftest, "aftest", "aftest [-n <Number of files>] [-r <Number of simultaneous requests>] [-s <Filesize, number of pages>] [-l <Number of iterations>] [-remove, remove files after close] [-reverse, write files in reverse order, start with the last page]", "Test the AsyncFile class of Ndb", 8192)
+{
+ int s, numReq, numOps;
+
+ readArguments(argc, argv);
+
+ files = new Pool<AsyncFile>(numberOfFiles, 2);
+ theRequestPool = new Pool<Request>;
+ theReportChannel = new MemoryChannelMultipleWriter<Request>;
+
+ ndbout << "AsyncFileTest starting" << endl;
+ ndbout << " " << numberOfFiles << " files" << endl;
+ ndbout << " " << numberOfRequests << " requests" << endl;
+ ndbout << " " << fileSize << " * 8k files" << endl << endl;
+ ndbout << " " << numberOfIterations << " iterations" << endl << endl;
+
+ NdbThread_SetConcurrencyLevel(numberOfFiles+2);
+
+ // initialize data to write to files
+ for (int i = 0; i < MAXFILES; i++) {
+ for (int j = 0; j < PAGESIZE; j++){
+ WritePages[i][j] = (64+i+j)%256;
+ }
+ // memset(&WritePages[i][0], i+64, PAGESIZE);
+ }
+
+ // Set file directory and name
+ // /T27/F27/NDBFS/S27Pnn.data
+ FileNameArray[0] = 27; // T27
+ FileNameArray[1] = 27; // F27
+ FileNameArray[2] = 27; // S27
+ FileNameArray[3] = FVERSION; // Version
+
+ for (int l = 0; l < numberOfIterations; l++)
+ {
+
+ ndbout << "Opening files" << endl;
+ // Open files
+ for (int f = 0; f < numberOfFiles; f++)
+ {
+ openFile(f);
+
+ }
+
+ // Wait for answer
+ openFileWait();
+
+ ndbout << "Files opened!" << endl<< endl;
+
+ // Write to files
+ ndbout << "Started writing" << endl;
+ TIMER_START;
+ s = 0;
+ numReq = 0;
+ numOps = 0;
+ while ( s < fileSize)
+ {
+ for (int r = 0; r < numberOfRequests; r++)
+ {
+ for (int f = 0; f < numberOfFiles; f++)
+ {
+ writeFile(f, s);
+ numReq++;
+ numOps++;
+ }
+
+ s++;
+ }
+
+ while (numReq > 0)
+ {
+ writeFileWait();
+ numReq--;
+ }
+
+ }
+
+ TIMER_PRINT("writes", numOps);
+
+
+ ndbout << "Started reading" << endl;
+ TIMER_START;
+
+ // Read from files
+ s = 0;
+ numReq = 0;
+ numOps = 0;
+ while ( s < fileSize)
+ {
+ for (int r = 0; r < numberOfRequests; r++)
+ {
+ for (int f = 0; f < numberOfFiles; f++)
+ {
+ readFile(f, s);
+ numReq++;
+ numOps++;
+ }
+
+ s++;
+
+ }
+
+ while (numReq > 0)
+ {
+ readFileWait();
+ numReq--;
+ }
+
+ }
+ TIMER_PRINT("reads", numOps);
+
+ ndbout << "Started writing with sync" << endl;
+ TIMER_START;
+
+ // Write to files
+ s = 0;
+ numReq = 0;
+ numOps = 0;
+ while ( s < fileSize)
+ {
+ for (int r = 0; r < numberOfRequests; r++)
+ {
+ for (int f = 0; f < numberOfFiles; f++)
+ {
+ writeSyncFile(f, s);
+ numReq++;
+ numOps++;
+ }
+
+ s++;
+ }
+
+ while (numReq > 0)
+ {
+ writeSyncFileWait();
+ numReq--;
+ }
+
+ }
+
+ TIMER_PRINT("writeSync", numOps);
+
+ // Close files
+ ndbout << "Closing files" << endl;
+ for (int f = 0; f < numberOfFiles; f++)
+ {
+ closeFile(f);
+
+ }
+
+ // Wait for answer
+ closeFileWait();
+
+ ndbout << "Files closed!" << endl<< endl;
+ }
+
+ // Deallocate memory
+ delete files;
+ delete theReportChannel;
+ delete theRequestPool;
+
+ return 0;
+
+}
+
+
+
+int forward( AsyncFile * file, Request* request )
+{
+ file->execute(request);
+ ERROR_CHECK 0;
+ return 1;
+}
+
+int openFile( int fileNum)
+{
+ AsyncFile* file = (AsyncFile *)files->get();
+
+ FileNameArray[3] = fileNum | FVERSION;
+ file->fileName().set( NDBFS_REF, &FileNameArray[0] );
+ ndbout << "openFile: " << file->fileName().c_str() << endl;
+
+ if( ERROR_STATE ) {
+ ERROR_RESET;
+ files->put( file );
+ ndbout << "Failed to set filename" << endl;
+ return 1;
+ }
+ file->reportTo(theReportChannel);
+
+ Request* request = theRequestPool->get();
+ request->action= Request::open;
+ request->error= 0;
+ request->par.open.flags = 0x302; //O_RDWR | O_CREAT | O_TRUNC ; // 770
+ request->set(NDBFS_REF, 0x23456789, fileNum );
+ request->file = file;
+
+ if (!forward(file,request)) {
+ // Something went wrong
+ ndbout << "Could not forward open request" << endl;
+ theRequestPool->put(request);
+ return 1;
+ }
+ return 0;
+}
+
+int closeFile( int fileNum)
+{
+
+ AsyncFile* file = openFiles[fileNum];
+
+ Request* request = theRequestPool->get();
+ if (removeFiles == 1)
+ request->action = Request::closeRemove;
+ else
+ request->action= Request::close;
+
+ request->error= 0;
+ request->set(NDBFS_REF, 0x23456789, fileNum );
+ request->file = file;
+
+ if (!forward(file,request)) {
+ // Something went wrong
+ ndbout << "Could not forward close request" << endl;
+ theRequestPool->put(request);
+ return 1;
+ }
+ return 0;
+}
+
+int writeFile( int fileNum, int pagenum)
+{
+ AsyncFile* file = openFiles[fileNum];
+#ifdef TESTDEBUG
+ ndbout << "writeFile" << fileNum <<": "<<pagenum<<", " << file->fileName().c_str()<< endl;
+#endif
+ Request *request = theRequestPool->get();
+ request->action = Request::write;
+ request->error = 0;
+ request->set(NDBFS_REF, pagenum, fileNum);
+ request->file = openFiles[fileNum];
+
+ // Write only one page, choose the correct page for each file using fileNum
+ request->par.readWrite.pages[0].buf = &WritePages[fileNum][0];
+ request->par.readWrite.pages[0].size = PAGESIZE;
+ if (writeFilesReverse == 1)
+ {
+ // write the last page in the files first
+ // This is a normal way for the Blocks in Ndb to write to a file
+ request->par.readWrite.pages[0].offset = (fileSize - pagenum - 1) * PAGESIZE;
+ }
+ else
+ {
+ request->par.readWrite.pages[0].offset = pagenum * PAGESIZE;
+ }
+ request->par.readWrite.numberOfPages = 1;
+
+ if (!forward(file,request)) {
+ // Something went wrong
+ ndbout << "Could not forward write request" << endl;
+ theRequestPool->put(request);
+ return 1;
+ }
+ return 0;
+
+}
+
+int writeSyncFile( int fileNum, int pagenum)
+{
+ AsyncFile* file = openFiles[fileNum];
+#ifdef TESTDEBUG
+ ndbout << "writeFile" << fileNum <<": "<<pagenum<<", " << file->fileName().c_str() << endl;
+#endif
+ Request *request = theRequestPool->get();
+ request->action = Request::writeSync;
+ request->error = 0;
+ request->set(NDBFS_REF, pagenum, fileNum);
+ request->file = openFiles[fileNum];
+
+ // Write only one page, choose the correct page for each file using fileNum
+ request->par.readWrite.pages[0].buf = &WritePages[fileNum][0];
+ request->par.readWrite.pages[0].size = PAGESIZE;
+ request->par.readWrite.pages[0].offset = pagenum * PAGESIZE;
+ request->par.readWrite.numberOfPages = 1;
+
+ if (!forward(file,request)) {
+ // Something went wrong
+ ndbout << "Could not forward write request" << endl;
+ theRequestPool->put(request);
+ return 1;
+ }
+ return 0;
+
+}
+
+int readFile( int fileNum, int pagenum)
+{
+ AsyncFile* file = openFiles[fileNum];
+#ifdef TESTDEBUG
+ ndbout << "readFile" << fileNum <<": "<<pagenum<<", " << file->fileName().c_str() << endl;
+#endif
+ Request *request = theRequestPool->get();
+ request->action = Request::read;
+ request->error = 0;
+ request->set(NDBFS_REF, pagenum, fileNum);
+ request->file = openFiles[fileNum];
+
+ // Read only one page, choose the correct page for each file using fileNum
+ request->par.readWrite.pages[0].buf = &ReadPages[fileNum][0];
+ request->par.readWrite.pages[0].size = PAGESIZE;
+ request->par.readWrite.pages[0].offset = pagenum * PAGESIZE;
+ request->par.readWrite.numberOfPages = 1;
+
+ if (!forward(file,request)) {
+ // Something went wrong
+ ndbout << "Could not forward read request" << endl;
+ theRequestPool->put(request);
+ return 1;
+ }
+ return 0;
+
+}
+
+int openFileWait()
+{
+ int openedFiles = 0;
+ while (openedFiles < numberOfFiles)
+ {
+ Request* request = theReportChannel->readChannel();
+ if (request)
+ {
+ if (request->action == Request::open)
+ {
+ if (request->error ==0)
+ {
+#ifdef TESTDEBUG
+ ndbout << "Opened file " << request->file->fileName().c_str() << endl;
+#endif
+ openFiles[request->theFilePointer] = request->file;
+ }
+ else
+ {
+ ndbout << "error while opening file" << endl;
+ exit(1);
+ }
+ theRequestPool->put(request);
+ openedFiles++;
+ }
+ else
+ {
+ ndbout << "Unexpected request received" << endl;
+ }
+ }
+ else
+ {
+ ndbout << "Nothing read from theReportChannel" << endl;
+ }
+ }
+ return 0;
+}
+
+int closeFileWait()
+{
+ int closedFiles = 0;
+ while (closedFiles < numberOfFiles)
+ {
+ Request* request = theReportChannel->readChannel();
+ if (request)
+ {
+ if (request->action == Request::close || request->action == Request::closeRemove)
+ {
+ if (request->error ==0)
+ {
+#ifdef TESTDEBUG
+ ndbout << "Closed file " << request->file->fileName().c_str() << endl;
+#endif
+ openFiles[request->theFilePointer] = NULL;
+ files->put(request->file);
+ }
+ else
+ {
+ ndbout << "error while closing file" << endl;
+ exit(1);
+ }
+ theRequestPool->put(request);
+ closedFiles++;
+ }
+ else
+ {
+ ndbout << "Unexpected request received" << endl;
+ }
+ }
+ else
+ {
+ ndbout << "Nothing read from theReportChannel" << endl;
+ }
+ }
+ return 0;
+}
+
+int writeFileWait()
+{
+ Request* request = theReportChannel->readChannel();
+ if (request)
+ {
+ if (request->action == Request::write)
+ {
+ if (request->error == 0)
+ {
+#ifdef TESTDEBUG
+ ndbout << "writeFileWait"<<request->theFilePointer<<", " << request->theUserPointer<<" "<< request->file->fileName().c_str() << endl;
+#endif
+
+ }
+ else
+ {
+ ndbout << "error while writing file, error=" << request->error << endl;
+ exit(1);
+ }
+ theRequestPool->put(request);
+ }
+ else
+ {
+ ndbout << "Unexpected request received" << endl;
+ }
+ }
+ else
+ {
+ ndbout << "Nothing read from theReportChannel" << endl;
+ }
+ return 0;
+}
+
+int writeSyncFileWait()
+{
+ Request* request = theReportChannel->readChannel();
+ if (request)
+ {
+ if (request->action == Request::writeSync)
+ {
+ if (request->error == 0)
+ {
+#ifdef TESTDEBUG
+ ndbout << "writeFileWait"<<request->theFilePointer<<", " << request->theUserPointer<<" "<< request->file->fileName().c_str() << endl;
+#endif
+
+ }
+ else
+ {
+ ndbout << "error while writing file" << endl;
+ exit(1);
+ }
+ theRequestPool->put(request);
+ }
+ else
+ {
+ ndbout << "Unexpected request received" << endl;
+ }
+ }
+ else
+ {
+ ndbout << "Nothing read from theReportChannel" << endl;
+ }
+ return 0;
+}
+
+int readFileWait()
+{
+ Request* request = theReportChannel->readChannel();
+ if (request)
+ {
+ if (request->action == Request::read)
+ {
+ if (request->error == 0)
+ {
+#ifdef TESTDEBUG
+ ndbout << "readFileWait"<<request->theFilePointer<<", " << request->theUserPointer<<" "<< request->file->fileName().c_str() << endl;
+#endif
+ if (memcmp(&(ReadPages[request->theFilePointer][0]), &(WritePages[request->theFilePointer][0]), PAGESIZE)!=0)
+ {
+ ndbout <<"Verification error!" << endl;
+ for (int i = 0; i < PAGESIZE; i++ ){
+ ndbout <<" Compare Page " << i << " : " << ReadPages[request->theFilePointer][i] <<", " <<WritePages[request->theFilePointer][i] << endl;;
+ if( ReadPages[request->theFilePointer][i] !=WritePages[request->theFilePointer][i])
+
+ exit(1);
+ }
+ }
+
+ }
+ else
+ {
+ ndbout << "error while reading file" << endl;
+ exit(1);
+ }
+ theRequestPool->put(request);
+ }
+ else
+ {
+ ndbout << "Unexpected request received" << endl;
+ }
+ }
+ else
+ {
+ ndbout << "Nothing read from theReportChannel" << endl;
+ }
+ return 0;
+}
+
+int readArguments(int argc, const char** argv)
+{
+
+ int i = 1;
+ while (argc > 1)
+ {
+ if (strcmp(argv[i], "-n") == 0)
+ {
+ numberOfFiles = atoi(argv[i+1]);
+ if ((numberOfFiles < 1) || (numberOfFiles > MAXFILES))
+ {
+ ndbout << "Wrong number of files, default = "<<DEFAULT_NUM_FILES << endl;
+ numberOfFiles = DEFAULT_NUM_FILES;
+ }
+ }
+ else if (strcmp(argv[i], "-r") == 0)
+ {
+ numberOfRequests = atoi(argv[i+1]);
+ if ((numberOfRequests < 1) || (numberOfRequests > MAXREQUESTS))
+ {
+ ndbout << "Wrong number of requests, default = "<<DEFAULT_NUM_REQUESTS << endl;
+ numberOfRequests = DEFAULT_NUM_REQUESTS;
+ }
+ }
+ else if (strcmp(argv[i], "-s") == 0)
+ {
+ fileSize = atoi(argv[i+1]);
+ if ((fileSize < 1) || (fileSize > MAXFILESIZE))
+ {
+ ndbout << "Wrong number of 8k pages, default = "<<DEFAULT_FILESIZE << endl;
+ fileSize = DEFAULT_FILESIZE;
+ }
+ }
+ else if (strcmp(argv[i], "-l") == 0)
+ {
+ numberOfIterations = atoi(argv[i+1]);
+ if ((numberOfIterations < 1))
+ {
+ ndbout << "Wrong number of iterations, default = 1" << endl;
+ numberOfIterations = 1;
+ }
+ }
+ else if (strcmp(argv[i], "-remove") == 0)
+ {
+ removeFiles = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-reverse") == 0)
+ {
+ ndbout << "Writing files reversed" << endl;
+ writeFilesReverse = 1;
+ argc++;
+ i--;
+ }
+
+ argc -= 2;
+ i = i + 2;
+ }
+
+ if ((fileSize % numberOfRequests)!= 0)
+ {
+ numberOfRequests = numberOfRequests - (fileSize % numberOfRequests);
+ ndbout <<"numberOfRequest must be modulo of filesize" << endl;
+ ndbout << "New numberOfRequest="<<numberOfRequests<<endl;
+ }
+ return 0;
+}
+
+
+// Needed for linking...
+
+void ErrorReporter::handleError(ErrorCategory type, int messageID,
+ const char* problemData, const char* objRef, NdbShutdownType stype)
+{
+
+ ndbout << "ErrorReporter::handleError activated" << endl;
+ ndbout << "type= " << type << endl;
+ ndbout << "messageID= " << messageID << endl;
+ ndbout << "problemData= " << problemData << endl;
+ ndbout << "objRef= " << objRef << endl;
+
+ exit(1);
+}
+
+void ErrorReporter::handleAssert(const char* message, const char* file, int line)
+{
+ ndbout << "ErrorReporter::handleAssert activated" << endl;
+ ndbout << "message= " << message << endl;
+ ndbout << "file= " << file << endl;
+ ndbout << "line= " << line << endl;
+ exit(1);
+}
+
+
+GlobalData globalData;
+
+
+Signal::Signal()
+{
+
+}
+
diff --git a/ndb/src/kernel/blocks/ndbfs/AsyncFileTest/Makefile b/ndb/src/kernel/blocks/ndbfs/AsyncFileTest/Makefile
new file mode 100644
index 00000000000..b0356e6da68
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/AsyncFileTest/Makefile
@@ -0,0 +1,27 @@
+include .defs.mk
+
+TYPE := kernel
+
+BIN_TARGET := aftest
+BIN_TARGET_ARCHIVES := ndbfs portlib trace signaldataprint
+
+SOURCES = AsyncFileTest.cpp
+
+CFLAGS_AsyncFileTest.cpp = -I../
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+# run basic tests
+run_test :
+ $(NDB_TOP)/bin/$(BIN_TARGET)
+ $(NDB_TOP)/bin/$(BIN_TARGET) -n 8 -r 8 -l 10 -remove
+ $(NDB_TOP)/bin/$(BIN_TARGET) -n 8 -r 8 -l 10 -reverse -remove
+ $(NDB_TOP)/bin/$(BIN_TARGET) -n 8 -r 8 -l 10 -s 512 -remove
+ $(NDB_TOP)/bin/$(BIN_TARGET) -n 8 -r 4 -l 1000
+
+
+
+
+
+
diff --git a/ndb/src/kernel/blocks/ndbfs/CircularIndex.cpp b/ndb/src/kernel/blocks/ndbfs/CircularIndex.cpp
new file mode 100644
index 00000000000..30b40097c9b
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/CircularIndex.cpp
@@ -0,0 +1,20 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "CircularIndex.hpp"
+
+
+
diff --git a/ndb/src/kernel/blocks/ndbfs/CircularIndex.hpp b/ndb/src/kernel/blocks/ndbfs/CircularIndex.hpp
new file mode 100644
index 00000000000..349cccdbcb4
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/CircularIndex.hpp
@@ -0,0 +1,116 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CircularIndex_H
+#define CircularIndex_H
+
+//===========================================================================
+//
+// .DESCRIPTION
+// Building block for circular buffers. It increment as a normal index.
+// untill it it becomes the maximum size then it becomes zero.
+//
+// .TYPICAL USE:
+// to implement a circular buffer.
+//
+// .EXAMPLE:
+// See MemoryChannel.C
+//===========================================================================
+
+///////////////////////////////////////////////////////////////////////////////
+// CircularIndex( int start= 0,int size=256 );
+// Constuctor
+// Parameters:
+// start: where to start to index
+// size : range of the index, will be from 0 to size-1
+///////////////////////////////////////////////////////////////////////////////
+// operator int ();
+// returns the index
+///////////////////////////////////////////////////////////////////////////////
+// void operator ++ ();
+// increments the index with one, of size is reached it is set to zero
+///////////////////////////////////////////////////////////////////////////////
+// friend int full( const CircularIndex& write, const CircularIndex& read );
+// Taken the write index and the read index from a buffer it is calculated
+// if the buffer is full
+// Parameters:
+// write: index used a write index for the buffer
+// read : index used a read index for the buffer
+// return
+// 0 : not full
+// 1 : full
+///////////////////////////////////////////////////////////////////////////////
+// friend int empty( const CircularIndex& write, const CircularIndex& read );
+// Taken the write index and the read index from a buffer it is calculated
+// if the buffer is empty
+// Parameters:
+// write: index used a write index for the buffer
+// read : index used a read index for the buffer
+// return
+// 0 : not empty
+// 1 : empty
+///////////////////////////////////////////////////////////////////////////////
+
+class CircularIndex
+{
+public:
+ inline CircularIndex( int start= 0,int size=256 );
+ operator int ();
+ CircularIndex& operator ++ ();
+ friend int full( const CircularIndex& write, const CircularIndex& read );
+ friend int empty( const CircularIndex& write, const CircularIndex& read );
+private:
+ int theSize;
+ int theIndex;
+};
+
+inline CircularIndex::operator int ()
+{
+ return theIndex;
+}
+
+inline CircularIndex& CircularIndex::operator ++ ()
+{
+ ++theIndex;
+ if( theIndex >= theSize ){
+ theIndex= 0;
+ }
+ return *this;
+}
+
+
+inline int full( const CircularIndex& write, const CircularIndex& read )
+{
+ int readTmp= read.theIndex;
+
+ if( read.theIndex < write.theIndex )
+ readTmp += read.theSize;
+
+ return ( readTmp - write.theIndex) == 1;
+}
+
+inline int empty( const CircularIndex& write, const CircularIndex& read )
+{
+ return read.theIndex == write.theIndex;
+}
+
+
+inline CircularIndex::CircularIndex( int start,int size ):
+ theSize(size),
+ theIndex(start)
+{
+}
+#endif
diff --git a/ndb/src/kernel/blocks/ndbfs/Filename.cpp b/ndb/src/kernel/blocks/ndbfs/Filename.cpp
new file mode 100644
index 00000000000..98ff7c7e4e4
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/Filename.cpp
@@ -0,0 +1,220 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdlib.h>
+#include <string.h>
+#include <NdbStdio.h>
+#include <NdbUnistd.h>
+#include <NdbOut.hpp>
+
+#include "Filename.hpp"
+#include "ErrorHandlingMacros.hpp"
+#include "Error.hpp"
+#include "RefConvert.hpp"
+#include "DebuggerNames.hpp"
+
+#include <signaldata/FsOpenReq.hpp>
+
+static const char* fileExtension[] = {
+ ".Data",
+ ".FragLog",
+ ".LocLog",
+ ".FragList",
+ ".TableList",
+ ".SchemaLog",
+ ".sysfile",
+ ".log",
+ ".ctl"
+};
+
+static const Uint32 noOfExtensions = sizeof(fileExtension)/sizeof(char*);
+
+Filename::Filename() :
+ theLevelDepth(0)
+{
+}
+
+void
+Filename::init(const char * pFileSystemPath){
+ if (pFileSystemPath == NULL) {
+ ERROR_SET(fatal, AFS_ERROR_NOPATH, ""," Filename::init()");
+ return;
+ }
+
+ strncpy(theBaseDirectory, pFileSystemPath, PATH_MAX);
+
+ // the environment variable is set,
+ // check that it is pointing on a valid directory
+ //
+ char buf2[PATH_MAX]; memset(buf2, 0,sizeof(buf2));
+#ifdef NDB_WIN32
+ char* szFilePart;
+ if(!GetFullPathName(theBaseDirectory, sizeof(buf2), buf2, &szFilePart)
+ || (::GetFileAttributes(theBaseDirectory)&FILE_ATTRIBUTE_READONLY))
+#else
+ if((::realpath(theBaseDirectory, buf2) == NULL)||
+ (::access(theBaseDirectory, W_OK) != 0))
+#endif
+ {
+ ERROR_SET(fatal, AFS_ERROR_INVALIDPATH, pFileSystemPath, " Filename::init()");
+ }
+ strncpy(theBaseDirectory, buf2, sizeof(theBaseDirectory));
+ // path seems ok, add delimiter if missing
+ if (strcmp(&theBaseDirectory[strlen(theBaseDirectory) - 1],
+ DIR_SEPARATOR) != 0)
+ strcat(theBaseDirectory, DIR_SEPARATOR);
+
+}
+
+
+Filename::~Filename(){
+}
+
+void
+Filename::set(BlockReference blockReference,
+ const Uint32 filenumber[4], bool dir)
+{
+ char buf[PATH_MAX];
+ theLevelDepth = 0;
+ strncpy(theName, theBaseDirectory, PATH_MAX);
+
+ const Uint32 type = FsOpenReq::getSuffix(filenumber);
+ const Uint32 version = FsOpenReq::getVersion(filenumber);
+ switch(version){
+ case 1 :{
+ const Uint32 diskNo = FsOpenReq::v1_getDisk(filenumber);
+ const Uint32 table = FsOpenReq::v1_getTable(filenumber);
+ const Uint32 frag = FsOpenReq::v1_getFragment(filenumber);
+ const Uint32 S_val = FsOpenReq::v1_getS(filenumber);
+ const Uint32 P_val = FsOpenReq::v1_getP(filenumber);
+
+ if (diskNo < 0xff){
+ snprintf(buf, sizeof(buf), "D%d%s", diskNo, DIR_SEPARATOR);
+ strcat(theName, buf);
+ theLevelDepth++;
+ }
+
+ {
+ const char* blockName = getBlockName( refToBlock(blockReference) );
+ if (blockName == NULL){
+ ERROR_SET(ecError, AFS_ERROR_PARAMETER,"","No Block Name");
+ return;
+ }
+ snprintf(buf, sizeof(buf), "%s%s", blockName, DIR_SEPARATOR);
+ strcat(theName, buf);
+ theLevelDepth++;
+ }
+
+ if (table < 0xffffffff){
+ snprintf(buf, sizeof(buf), "T%d%s", table, DIR_SEPARATOR);
+ strcat(theName, buf);
+ theLevelDepth++;
+ }
+
+ if (frag < 0xffffffff){
+ snprintf(buf, sizeof(buf), "F%d%s", frag, DIR_SEPARATOR);
+ strcat(theName, buf);
+ theLevelDepth++;
+ }
+
+
+ if (S_val < 0xffffffff){
+ snprintf(buf, sizeof(buf), "S%d", S_val);
+ strcat(theName, buf);
+ }
+
+ if (P_val < 0xff){
+ snprintf(buf, sizeof(buf), "P%d", P_val);
+ strcat(theName, buf);
+ }
+
+ }
+ break;
+ case 2:{
+ const Uint32 seq = FsOpenReq::v2_getSequence(filenumber);
+ const Uint32 nodeId = FsOpenReq::v2_getNodeId(filenumber);
+ const Uint32 count = FsOpenReq::v2_getCount(filenumber);
+
+ snprintf(buf, sizeof(buf), "BACKUP%sBACKUP-%d%s",
+ DIR_SEPARATOR, seq, DIR_SEPARATOR);
+ strcat(theName, buf);
+ if(count == 0xffffffff) {
+ snprintf(buf, sizeof(buf), "BACKUP-%d.%d",
+ seq, nodeId); strcat(theName, buf);
+ } else {
+ snprintf(buf, sizeof(buf), "BACKUP-%d-%d.%d",
+ seq, count, nodeId); strcat(theName, buf);
+ }
+ theLevelDepth = 2;
+ break;
+ }
+ break;
+ case 3:{
+ const Uint32 diskNo = FsOpenReq::v1_getDisk(filenumber);
+
+ if(diskNo == 0xFF){
+ ERROR_SET(ecError, AFS_ERROR_PARAMETER,"","Invalid disk specification");
+ }
+
+ snprintf(buf, sizeof(buf), "D%d%s", diskNo, DIR_SEPARATOR);
+ strcat(theName, buf);
+ theLevelDepth++;
+ }
+ break;
+ default:
+ ERROR_SET(ecError, AFS_ERROR_PARAMETER,"","Wrong version");
+ }
+ if (type >= noOfExtensions){
+ ERROR_SET(ecError, AFS_ERROR_PARAMETER,"","File Type doesn't exist");
+ return;
+ }
+ strcat(theName, fileExtension[type]);
+
+ if(dir == true){
+ for(Uint32 l = strlen(theName) - 1; l >= 0; l--){
+ if(theName[l] == DIR_SEPARATOR[0]){
+ theName[l] = 0;
+ break;
+ }
+ }
+ }
+}
+
+/**
+ * Find out directory name on level
+ * Ex:
+ * theName = "/tmp/fs/T0/NDBFS/D0/P0/S27.data"
+ * level = 1
+ * would return "/tmp/fs/T0/NDBFS/
+ */
+const char* Filename::directory(int level)
+{
+ const char* p;
+
+ p = theName;
+ p += strlen(theBaseDirectory);
+
+ for (int i = 0; i <= level; i++){
+ p = strstr(p, DIR_SEPARATOR);
+ p++;
+ }
+
+ strncpy(theDirectory, theName, p - theName - 1);
+ theDirectory[p-theName-1] = 0;
+ return theDirectory;
+}
+
+
diff --git a/ndb/src/kernel/blocks/ndbfs/Filename.hpp b/ndb/src/kernel/blocks/ndbfs/Filename.hpp
new file mode 100644
index 00000000000..4c3569b5485
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/Filename.hpp
@@ -0,0 +1,97 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef Filename_H
+#define Filename_H
+
+//===========================================================================
+//
+// .DESCRIPTION
+// Takes a 128 bits value (done as a array of four longs) and
+// makes a filename out of it acording the following schema
+// Bits 0-31 T
+// Bits 32-63 F
+// Bits 64-95 S
+// Bits 96-103 P
+// Bits 104-111 D
+// Bits 112-119 File Type
+// Bits 120-127 Version number of Filename
+//
+// T, is used to find/create a directory. If T = 0xFFFF then the
+// file is on top level. In that case the F is of no relevance.
+// F, same as T.
+// S, is used to find/create a filename. If S= 0xFFFF then it is ignored.
+// P, same as S
+// D, is used to find/create the root directory, this is the
+// directory before the blockname. If D= 0xFF then it is ignored.
+// File Type
+// 0 => .Data
+// 1 => .FragLog
+// 2 => .LocLog
+// 3 => .FragList
+// 4 => .TableList
+// 5 => .SchemaLog
+// 6 => .sysfile
+// 15=> ignored
+// Version number of Filename, current version is 0x1, must be
+// used for the this style of options.
+//
+//
+//===========================================================================
+
+#include <kernel_types.h>
+#include <NdbUnistd.h>
+
+class Filename
+{
+public:
+ // filenumber is 64 bits but is split in to 4 32bits words
+ Filename();
+ ~Filename();
+ void set(BlockReference blockReference,
+ const Uint32 filenumber[4], bool dir = false);
+ const char* baseDirectory() const;
+ const char* directory(int level);
+ int levels() const;
+ const char* c_str() const;
+
+ void init(const char * fileSystemPath);
+
+private:
+ int theLevelDepth;
+ char theName[PATH_MAX];
+ char theBaseDirectory[PATH_MAX];
+ char theDirectory[PATH_MAX];
+};
+
+// inline methods
+inline const char* Filename::c_str() const{
+ return theName;
+}
+
+inline const char* Filename::baseDirectory() const{
+ return theBaseDirectory;
+}
+
+inline int Filename::levels() const{
+ return theLevelDepth;
+}
+
+#endif
+
+
+
+
diff --git a/ndb/src/kernel/blocks/ndbfs/Makefile b/ndb/src/kernel/blocks/ndbfs/Makefile
new file mode 100644
index 00000000000..58e1458bf16
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := ndbfs
+
+SOURCES = \
+ AsyncFile.cpp \
+ Ndbfs.cpp VoidFs.cpp \
+ Filename.cpp \
+ CircularIndex.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/kernel/blocks/ndbfs/MemoryChannel.cpp b/ndb/src/kernel/blocks/ndbfs/MemoryChannel.cpp
new file mode 100644
index 00000000000..a1aebdef7a1
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/MemoryChannel.cpp
@@ -0,0 +1,18 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#include "MemoryChannel.hpp"
+
diff --git a/ndb/src/kernel/blocks/ndbfs/MemoryChannel.hpp b/ndb/src/kernel/blocks/ndbfs/MemoryChannel.hpp
new file mode 100644
index 00000000000..6e0c2721ca0
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/MemoryChannel.hpp
@@ -0,0 +1,168 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MemoryChannel_H
+#define MemoryChannel_H
+
+//===========================================================================
+//
+// .DESCRIPTION
+// Pointer based communication channel for communication between two
+// thread. It does not copy any data in or out the channel so the
+// item that is put in can not be used untill the other thread has
+// given it back. There is no support for detecting the return of a
+// item. The channel is half-duplex.
+// For comminication between 1 writer and 1 reader use the MemoryChannel
+// class, for comminication between multiple writer and 1 reader use the
+// MemoryChannelMultipleWriter. There is no support for multiple readers.
+//
+// .TYPICAL USE:
+// to communicate between threads.
+//
+// .EXAMPLE:
+// See AsyncFile.C
+//===========================================================================
+//
+//
+// MemoryChannel( int size= 256);
+// Constuctor
+// Parameters:
+// size : amount of pointer it can hold
+//
+// void operator ++ ();
+// increments the index with one, if size is reached it is set to zero
+//
+// virtual void write( T *t);
+// Puts the item in the channel if the channel is full an error is reported.
+// Parameters:
+// t: pointer to item to put in the channel, after this the item
+// is shared with the other thread.
+// errors
+// AFS_ERROR_CHANNALFULL, channel is full
+//
+// T* read();
+// Reads a itemn from the channel, if channel is empty it blocks untill
+// an item can be read.
+// return
+// T : item from the channel
+//
+// T* tryRead();
+// Reads a item from the channel, if channel is empty it returns zero.
+// return
+// T : item from the channel or zero if channel is empty.
+//
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+#include "MemoryChannelOSE.hpp"
+#else
+
+#include "ErrorHandlingMacros.hpp"
+#include "Error.hpp"
+#include "CircularIndex.hpp"
+#include "NdbMutex.h"
+#include "NdbCondition.h"
+#include <NdbOut.hpp>
+
+#include <assert.h>
+
+template <class T>
+class MemoryChannel
+{
+public:
+ MemoryChannel( int size= 256);
+ virtual ~MemoryChannel( );
+
+ virtual void writeChannel( T *t);
+ T* readChannel();
+ T* tryReadChannel();
+
+private:
+ int theSize;
+ T **theChannel;
+ CircularIndex theWriteIndex;
+ CircularIndex theReadIndex;
+ NdbMutex* theMutexPtr;
+ NdbCondition* theConditionPtr;
+
+};
+
+
+template <class T> MemoryChannel<T>::MemoryChannel( int size):
+ theSize(size),
+ theChannel(new T*[size] ),
+ theWriteIndex(0, size),
+ theReadIndex(0, size)
+{
+ theMutexPtr = NdbMutex_Create();
+ theConditionPtr = NdbCondition_Create();
+}
+
+template <class T> MemoryChannel<T>::~MemoryChannel( )
+{
+ NdbMutex_Destroy(theMutexPtr);
+ NdbCondition_Destroy(theConditionPtr);
+ delete [] theChannel;
+}
+
+template <class T> void MemoryChannel<T>::writeChannel( T *t)
+{
+
+ NdbMutex_Lock(theMutexPtr);
+ REQUIRE(!full(theWriteIndex, theReadIndex), "Memory Channel Full");
+ REQUIRE(theChannel != NULL, "Memory Channel Full");
+ theChannel[theWriteIndex]= t;
+ ++theWriteIndex;
+ NdbMutex_Unlock(theMutexPtr);
+ NdbCondition_Signal(theConditionPtr);
+}
+
+
+template <class T> T* MemoryChannel<T>::readChannel()
+{
+ T* tmp;
+
+ NdbMutex_Lock(theMutexPtr);
+ while ( empty(theWriteIndex, theReadIndex) )
+ {
+ NdbCondition_Wait(theConditionPtr,
+ theMutexPtr);
+ }
+
+ tmp= theChannel[theReadIndex];
+ ++theReadIndex;
+ NdbMutex_Unlock(theMutexPtr);
+ return tmp;
+}
+
+template <class T> T* MemoryChannel<T>::tryReadChannel()
+{
+ T* tmp= 0;
+ NdbMutex_Lock(theMutexPtr);
+ NdbCondition_WaitTimeout(theConditionPtr,
+ theMutexPtr, 0);
+ if ( !empty(theWriteIndex, theReadIndex) )
+ {
+ tmp= theChannel[theReadIndex];
+ ++theReadIndex;
+ }
+ NdbMutex_Unlock(theMutexPtr);
+ return tmp;
+}
+
+#endif
+
+#endif // MemoryChannel_H
+
diff --git a/ndb/src/kernel/blocks/ndbfs/MemoryChannelOSE.hpp b/ndb/src/kernel/blocks/ndbfs/MemoryChannelOSE.hpp
new file mode 100644
index 00000000000..9f70efcadf7
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/MemoryChannelOSE.hpp
@@ -0,0 +1,205 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MemoryChannelOSE_H
+#define MemoryChannelOSE_H
+
+//===========================================================================
+//
+// .DESCRIPTION
+// Pointer based communication channel for communication between two
+// thread. It sends the pointer to the other signal via an OSE signal
+//
+// .TYPICAL USE:
+// to communicate between threads.
+//
+// .EXAMPLE:
+// See AsyncFile.C
+//===========================================================================
+//
+//
+// MemoryChannel( int size= 256);
+// Constuctor
+// Parameters:
+// size : is ignored in OSE version
+//
+// void operator ++ ();
+// increments the index with one, if size is reached it is set to zero
+//
+// virtual void write( T *t);
+// Puts the item in the channel if the channel is full an error is reported.
+// Parameters:
+// t: pointer to item to put in the channel, after this the item
+// is shared with the other thread.
+// errors
+// AFS_ERROR_CHANNALFULL, channel is full
+//
+// T* read();
+// Reads a itemn from the channel, if channel is empty it blocks untill
+// an item can be read.
+// return
+// T : item from the channel
+//
+// T* tryRead();
+// Reads a item from the channel, if channel is empty it returns zero.
+// return
+// T : item from the channel or zero if channel is empty.
+//
+
+#include <ose.h>
+#include "ErrorHandlingMacros.hpp"
+#include "Error.hpp"
+#include "NdbMutex.h"
+#include "NdbCondition.h"
+
+#include <assert.h>
+
+
+
+
+template <class T>
+class MemoryChannel
+{
+public:
+ MemoryChannel( int size= 256);
+ virtual ~MemoryChannel( );
+
+ virtual void writeChannel( T *t);
+ T* readChannel();
+ T* tryReadChannel();
+
+private:
+ PROCESS theReceiverPid;
+};
+
+template <class T> class MemoryChannelMultipleWriter:public MemoryChannel<T>
+{
+public:
+ MemoryChannelMultipleWriter( int size= 256);
+ ~MemoryChannelMultipleWriter( );
+ void writeChannel( T *t);
+
+private:
+};
+
+
+#define MEMCHANNEL_SIGBASE 5643
+
+#define MEMCHANNEL_SIGNAL (MEMCHANNEL_SIGBASE + 1) /* !-SIGNO(struct MemChannelSignal)-! */
+
+
+struct MemChannelSignal
+{
+ SIGSELECT sigNo;
+ void* ptr;
+};
+
+union SIGNAL
+{
+ SIGSELECT sigNo;
+ struct MemChannelSignal memChanSig;
+};
+
+template <class T> MemoryChannel<T>::MemoryChannel( int size )
+{
+ // Default receiver for this channel is the creating process
+ theReceiverPid = current_process();
+}
+
+template <class T> MemoryChannel<T>::~MemoryChannel( )
+{
+}
+
+template <class T> void MemoryChannel<T>::writeChannel( T *t)
+{
+ union SIGNAL* sig;
+
+ sig = alloc(sizeof(struct MemChannelSignal), MEMCHANNEL_SIGNAL);
+ ((struct MemChannelSignal*)sig)->ptr = t;
+ send(&sig, theReceiverPid);
+}
+
+
+template <class T> T* MemoryChannel<T>::readChannel()
+{
+ T* tmp;
+
+ static const SIGSELECT sel_mem[] = {1, MEMCHANNEL_SIGNAL};
+ union SIGNAL* sig;
+
+ tmp = NULL; /* Default value */
+
+ sig = receive((SIGSELECT*)sel_mem);
+ if (sig != NIL){
+ if (sig->sigNo == MEMCHANNEL_SIGNAL){
+ tmp = (T*)(((struct MemChannelSignal*)sig)->ptr);
+ }else{
+ assert(1==0);
+ }
+ free_buf(&sig);
+ }
+
+ return tmp;
+}
+
+template <class T> T* MemoryChannel<T>::tryReadChannel()
+{
+ T* tmp;
+
+ static const SIGSELECT sel_mem[] = {1, MEMCHANNEL_SIGNAL};
+ union SIGNAL* sig;
+
+ tmp = NULL; /* Default value */
+
+ sig = receive_w_tmo(0, (SIGSELECT*)sel_mem);
+ if (sig != NIL){
+ if (sig->sigNo == MEMCHANNEL_SIGNAL){
+ tmp = (T*)(((struct MemChannelSignal*)sig)->ptr);
+ }else{
+ assert(1==0);
+ }
+ free_buf(&sig);
+ }
+
+ return tmp;
+}
+
+
+#endif // MemoryChannel_H
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/ndb/src/kernel/blocks/ndbfs/MemoryChannelTest/Makefile b/ndb/src/kernel/blocks/ndbfs/MemoryChannelTest/Makefile
new file mode 100644
index 00000000000..68f71bfc4cd
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/MemoryChannelTest/Makefile
@@ -0,0 +1,13 @@
+include .defs.mk
+
+TYPE := kernel
+
+BIN_TARGET := mctest
+BIN_TARGET_ARCHIVES := portlib
+
+SOURCES = MemoryChannelTest.cpp
+
+CFLAGS_MemoryChannelTest.cpp = -I../
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/kernel/blocks/ndbfs/MemoryChannelTest/MemoryChannelTest.cpp b/ndb/src/kernel/blocks/ndbfs/MemoryChannelTest/MemoryChannelTest.cpp
new file mode 100644
index 00000000000..aeab9f7828d
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/MemoryChannelTest/MemoryChannelTest.cpp
@@ -0,0 +1,197 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "MemoryChannel.hpp"
+#include "NdbThread.h"
+#include "NdbSleep.h"
+#include "NdbOut.hpp"
+#include "NdbMain.h"
+
+
+
+MemoryChannel<int>* theMemoryChannel;
+
+
+extern "C" void* runProducer(void*arg)
+{
+ // The producer will items into the MemoryChannel
+ int count = *(int*)arg;
+ int* p;
+ int i = 0;
+ while (i <= count)
+ {
+ p = new int(i);
+ ndbout << "P: " << *p << endl;
+ theMemoryChannel->writeChannel(p);
+ if (i%5==0)
+ NdbSleep_MilliSleep(i);
+ i++;
+ }
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+extern "C" void* runConsumer(void* arg)
+{
+ // The producer will read items from MemoryChannel and print on screen
+ int count = *(int*)arg;
+ int* p;
+ int i = 0;
+ while (i < count)
+ {
+ p = theMemoryChannel->readChannel();
+ ndbout << "C: " << *p << endl;
+ i = *p;
+ delete p;
+
+ }
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+
+
+class ArgStruct
+{
+public:
+ ArgStruct(int _items, int _no){
+ items=_items;
+ no=_no;
+ };
+ int items;
+ int no;
+};
+
+MemoryChannelMultipleWriter<ArgStruct>* theMemoryChannel2;
+
+extern "C" void* runProducer2(void*arg)
+{
+ // The producer will items into the MemoryChannel
+ ArgStruct* pArg = (ArgStruct*)arg;
+ int count = pArg->items;
+ ArgStruct* p;
+ int i = 0;
+ while (i < count)
+ {
+ p = new ArgStruct(i, pArg->no);
+ ndbout << "P"<<pArg->no<<": " << i << endl;
+ theMemoryChannel2->writeChannel(p);
+ NdbSleep_MilliSleep(i);
+ i++;
+ }
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+extern "C" void* runConsumer2(void* arg)
+{
+ // The producer will read items from MemoryChannel and print on screen
+ ArgStruct* pArg = (ArgStruct*)arg;
+ int count = pArg->items * pArg->no;
+ ArgStruct* p;
+ int i = 0;
+ while (i < count)
+ {
+ p = theMemoryChannel2->readChannel();
+ ndbout << "C: "<< p->no << ", " << p->items << endl;
+ i++;
+ delete p;
+ }
+ ndbout << "Consumer2: " << count << " received" << endl;
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+
+
+
+//#if defined MEMORYCHANNELTEST
+
+//int main(int argc, char **argv)
+NDB_COMMAND(mctest, "mctest", "mctest", "Test the memory channel used in Ndb", 32768)
+{
+
+ ndbout << "==== testing MemoryChannel ====" << endl;
+
+ theMemoryChannel = new MemoryChannel<int>;
+ theMemoryChannel2 = new MemoryChannelMultipleWriter<ArgStruct>;
+
+ NdbThread* consumerThread;
+ NdbThread* producerThread;
+
+ NdbThread_SetConcurrencyLevel(2);
+
+ int numItems = 100;
+ producerThread = NdbThread_Create(runProducer,
+ (void**)&numItems,
+ 4096,
+ (char*)"producer");
+
+ consumerThread = NdbThread_Create(runConsumer,
+ (void**)&numItems,
+ 4096,
+ (char*)"consumer");
+
+
+ void *status;
+ NdbThread_WaitFor(consumerThread, &status);
+ NdbThread_WaitFor(producerThread, &status);
+
+ ndbout << "==== testing MemoryChannelMultipleWriter ====" << endl;
+#define NUM_THREADS2 5
+ NdbThread_SetConcurrencyLevel(NUM_THREADS2+2);
+ NdbThread* producerThreads[NUM_THREADS2];
+
+ ArgStruct *pArg;
+ for (int j = 0; j < NUM_THREADS2; j++)
+ {
+ char buf[25];
+ sprintf((char*)&buf, "producer%d", j);
+ pArg = new ArgStruct(numItems, j);
+ producerThreads[j] = NdbThread_Create(runProducer2,
+ (void**)pArg,
+ 4096,
+ (char*)&buf);
+ }
+
+ pArg = new ArgStruct(numItems, NUM_THREADS2);
+ consumerThread = NdbThread_Create(runConsumer2,
+ (void**)pArg,
+ 4096,
+ (char*)"consumer");
+
+
+ NdbThread_WaitFor(consumerThread, &status);
+ for (int j = 0; j < NUM_THREADS2; j++)
+ {
+ NdbThread_WaitFor(producerThreads[j], &status);
+ }
+
+
+ return 0;
+
+}
+
+void ErrorReporter::handleError(ErrorCategory type, int messageID,
+ const char* problemData, const char* objRef,
+ NdbShutdownType nst)
+{
+
+ ndbout << "ErrorReporter::handleError activated" << endl;
+ exit(1);
+}
+
+//#endif
diff --git a/ndb/src/kernel/blocks/ndbfs/Ndbfs.cpp b/ndb/src/kernel/blocks/ndbfs/Ndbfs.cpp
new file mode 100644
index 00000000000..8992a2104e9
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/Ndbfs.cpp
@@ -0,0 +1,1008 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <limits.h>
+#include <errno.h>
+
+#include "Ndbfs.hpp"
+#include "AsyncFile.hpp"
+#include "Filename.hpp"
+#include "Error.hpp"
+
+#include <signaldata/FsOpenReq.hpp>
+#include <signaldata/FsCloseReq.hpp>
+#include <signaldata/FsReadWriteReq.hpp>
+#include <signaldata/FsAppendReq.hpp>
+#include <signaldata/FsRemoveReq.hpp>
+#include <signaldata/FsConf.hpp>
+#include <signaldata/FsRef.hpp>
+#include <signaldata/NdbfsContinueB.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+
+#include <RefConvert.hpp>
+#include <NdbSleep.h>
+#include <NdbOut.hpp>
+#include <Configuration.hpp>
+
+#define DEBUG(x) { ndbout << "FS::" << x << endl; }
+
+inline
+int pageSize( const NewVARIABLE* baseAddrRef )
+{
+ int log_psize;
+ int log_qsize = baseAddrRef->bits.q;
+ int log_vsize = baseAddrRef->bits.v;
+ if (log_vsize < 3)
+ log_vsize = 3;
+ log_psize = log_qsize + log_vsize - 3;
+ return (1 << log_psize);
+}
+
+
+Ndbfs::Ndbfs(const Configuration & conf) :
+ SimulatedBlock(NDBFS, conf),
+ scanningInProgress(false),
+ theLastId(0),
+ m_maxOpenedFiles(0)
+{
+ theFileSystemPath = conf.fileSystemPath();
+ theRequestPool = new Pool<Request>;
+
+ const Properties * p = conf.getOwnProperties();
+ ndbrequire(p != 0);
+
+ ndbrequire(p->get("MaxNoOfOpenFiles", &m_maxFiles));
+
+ // Create idle AsyncFiles
+ Uint32 noIdleFiles = 16;
+ for (Uint32 i = 0; i < noIdleFiles; i++){
+ theIdleFiles.push_back(createAsyncFile());
+ }
+
+ BLOCK_CONSTRUCTOR(Ndbfs);
+
+ // Set received signals
+ addRecSignal(GSN_DUMP_STATE_ORD, &Ndbfs::execDUMP_STATE_ORD);
+ addRecSignal(GSN_STTOR, &Ndbfs::execSTTOR);
+ addRecSignal(GSN_FSOPENREQ, &Ndbfs::execFSOPENREQ);
+ addRecSignal(GSN_FSCLOSEREQ, &Ndbfs::execFSCLOSEREQ);
+ addRecSignal(GSN_FSWRITEREQ, &Ndbfs::execFSWRITEREQ);
+ addRecSignal(GSN_FSREADREQ, &Ndbfs::execFSREADREQ);
+ addRecSignal(GSN_FSSYNCREQ, &Ndbfs::execFSSYNCREQ);
+ addRecSignal(GSN_CONTINUEB, &Ndbfs::execCONTINUEB);
+ addRecSignal(GSN_FSAPPENDREQ, &Ndbfs::execFSAPPENDREQ);
+ addRecSignal(GSN_FSREMOVEREQ, &Ndbfs::execFSREMOVEREQ);
+ // Set send signals
+}
+
+Ndbfs::~Ndbfs()
+{
+ // Delete all files
+ // AsyncFile destuctor will take care of deleting
+ // the thread it has created
+ for (unsigned i = 0; i < theFiles.size(); i++){
+ AsyncFile* file = theFiles[i];
+ delete file;
+ theFiles[i] = NULL;
+ }//for
+ theFiles.clear();
+
+ delete theRequestPool;
+}
+
+/* Received a restart signal.
+ * Answer it like any other block
+ * PR0 : StartCase
+ * DR0 : StartPhase
+ * DR1 : ?
+ * DR2 : ?
+ * DR3 : ?
+ * DR4 : ?
+ * DR5 : SignalKey
+ */
+void
+Ndbfs::execSTTOR(Signal* signal)
+{
+ jamEntry();
+
+ if(signal->theData[1] == 0){ // StartPhase 0
+ jam();
+ cownref = NDBFS_REF;
+ // close all open files
+ ndbrequire(theOpenFiles.size() == 0);
+
+ scanningInProgress = false;
+
+ signal->theData[0] = NdbfsContinueB::ZSCAN_MEMORYCHANNEL_10MS_DELAY;
+ sendSignalWithDelay(cownref, GSN_CONTINUEB, signal, 10, 1);
+
+ signal->theData[3] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal,4, JBB);
+ return;
+ }
+ ndbrequire(0);
+}
+
+int
+Ndbfs::forward( AsyncFile * file, Request* request)
+{
+ jam();
+ file->execute(request);
+ return 1;
+}
+
+void
+Ndbfs::execFSOPENREQ(Signal* signal)
+{
+ jamEntry();
+ const FsOpenReq * const fsOpenReq = (FsOpenReq *)&signal->theData[0];
+ const BlockReference userRef = fsOpenReq->userReference;
+ AsyncFile* file = getIdleFile();
+ ndbrequire(file != NULL);
+ ndbrequire(signal->getLength() == FsOpenReq::SignalLength)
+ file->theFileName.set( userRef, fsOpenReq->fileNumber);
+ file->reportTo(&theFromThreads);
+
+ Request* request = theRequestPool->get();
+ request->action = Request::open;
+ request->error = 0;
+ request->par.open.flags = fsOpenReq->fileFlags;
+ request->set(userRef, fsOpenReq->userPointer, newId() );
+ request->file = file;
+ request->theTrace = signal->getTrace();
+
+ ndbrequire(forward(file, request));
+}
+
+void
+Ndbfs::execFSREMOVEREQ(Signal* signal)
+{
+ jamEntry();
+ const FsRemoveReq * const req = (FsRemoveReq *)signal->getDataPtr();
+ const BlockReference userRef = req->userReference;
+ AsyncFile* file = getIdleFile();
+ ndbrequire(file != NULL);
+
+ file->theFileName.set( userRef, req->fileNumber, req->directory);
+ file->reportTo(&theFromThreads);
+
+ Request* request = theRequestPool->get();
+ request->action = Request::rmrf;
+ request->par.rmrf.directory = req->directory;
+ request->par.rmrf.own_directory = req->ownDirectory;
+ request->error = 0;
+ request->set(userRef, req->userPointer, newId() );
+ request->file = file;
+ request->theTrace = signal->getTrace();
+
+ ndbrequire(forward(file, request));
+}
+
+/*
+ * PR0: File Pointer DR0: User reference DR1: User Pointer DR2: Flag bit 0= 1
+ * remove file
+ */
+void
+Ndbfs::execFSCLOSEREQ(Signal * signal)
+{
+ jamEntry();
+ const FsCloseReq * const fsCloseReq = (FsCloseReq *)&signal->theData[0];
+ const BlockReference userRef = fsCloseReq->userReference;
+ const Uint16 filePointer = (Uint16)fsCloseReq->filePointer;
+ const UintR userPointer = fsCloseReq->userPointer;
+
+ AsyncFile* openFile = theOpenFiles.find(filePointer);
+ if (openFile == NULL) {
+ // The file was not open, send error back to sender
+ jam();
+ // Initialise FsRef signal
+ FsRef * const fsRef = (FsRef *)&signal->theData[0];
+ fsRef->userPointer = userPointer;
+ fsRef->setErrorCode(fsRef->errorCode, FsRef::fsErrFileDoesNotExist);
+ fsRef->osErrorCode = ~0; // Indicate local error
+ sendSignal(userRef, GSN_FSCLOSEREF, signal, 3, JBB);
+ return;
+ }
+
+ Request *request = theRequestPool->get();
+ if( fsCloseReq->getRemoveFileFlag(fsCloseReq->fileFlag) == true ) {
+ jam();
+ request->action = Request::closeRemove;
+ } else {
+ jam();
+ request->action = Request::close;
+ }
+ request->set(userRef, fsCloseReq->userPointer, filePointer);
+ request->file = openFile;
+ request->error = 0;
+ request->theTrace = signal->getTrace();
+
+ ndbrequire(forward(openFile, request));
+}
+
+void
+Ndbfs::readWriteRequest(int action, Signal * signal)
+{
+ const FsReadWriteReq * const fsRWReq = (FsReadWriteReq *)&signal->theData[0];
+ Uint16 filePointer = (Uint16)fsRWReq->filePointer;
+ const UintR userPointer = fsRWReq->userPointer;
+ const BlockReference userRef = fsRWReq->userReference;
+ const BlockNumber blockNumber = refToBlock(userRef);
+
+ AsyncFile* openFile = theOpenFiles.find(filePointer);
+
+ const NewVARIABLE *myBaseAddrRef = &getBat(blockNumber)[fsRWReq->varIndex];
+ unsigned int tPageSize;
+ unsigned int tClusterSize;
+ unsigned int tNRR;
+ unsigned int tPageOffset;
+ char* tWA;
+ FsRef::NdbfsErrorCodeType errorCode;
+
+ Request *request = theRequestPool->get();
+ request->error = 0;
+ request->set(userRef, userPointer, filePointer);
+ request->file = openFile;
+ request->action = (Request::Action) action;
+ request->theTrace = signal->getTrace();
+
+ if (fsRWReq->numberOfPages == 0) { //Zero pages not allowed
+ jam();
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }
+
+ if (fsRWReq->varIndex >= getBatSize(blockNumber)) {
+ jam();// Ensure that a valid variable is used
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }
+ if (myBaseAddrRef == NULL) {
+ jam(); // Ensure that a valid variable is used
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }
+ if (openFile == NULL) {
+ jam(); //file not open
+ errorCode = FsRef::fsErrFileDoesNotExist;
+ goto error;
+ }
+ tPageSize = pageSize(myBaseAddrRef);
+ tClusterSize = myBaseAddrRef->ClusterSize;
+ tNRR = myBaseAddrRef->nrr;
+ tWA = (char*)myBaseAddrRef->WA;
+
+ switch (fsRWReq->getFormatFlag(fsRWReq->operationFlag)) {
+
+ // List of memory and file pages pairs
+ case FsReadWriteReq::fsFormatListOfPairs: {
+ jam();
+ for (unsigned int i = 0; i < fsRWReq->numberOfPages; i++) {
+ jam();
+ const Uint32 varIndex = fsRWReq->data.listOfPair[i].varIndex;
+ const Uint32 fileOffset = fsRWReq->data.listOfPair[i].fileOffset;
+ if (varIndex >= tNRR) {
+ jam();
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }//if
+ request->par.readWrite.pages[i].buf = &tWA[varIndex * tClusterSize];
+ request->par.readWrite.pages[i].size = tPageSize;
+ request->par.readWrite.pages[i].offset = fileOffset * tPageSize;
+ }//for
+ request->par.readWrite.numberOfPages = fsRWReq->numberOfPages;
+ break;
+ }//case
+
+ // Range of memory page with one file page
+ case FsReadWriteReq::fsFormatArrayOfPages: {
+ if ((fsRWReq->numberOfPages + fsRWReq->data.arrayOfPages.varIndex) > tNRR) {
+ jam();
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }//if
+ const Uint32 varIndex = fsRWReq->data.arrayOfPages.varIndex;
+ const Uint32 fileOffset = fsRWReq->data.arrayOfPages.fileOffset;
+
+ request->par.readWrite.pages[0].offset = fileOffset * tPageSize;
+ request->par.readWrite.pages[0].size = tPageSize * fsRWReq->numberOfPages;
+ request->par.readWrite.numberOfPages = 1;
+ request->par.readWrite.pages[0].buf = &tWA[varIndex * tPageSize];
+ break;
+ }//case
+
+ // List of memory pages followed by one file page
+ case FsReadWriteReq::fsFormatListOfMemPages: {
+
+ tPageOffset = fsRWReq->data.listOfMemPages.varIndex[fsRWReq->numberOfPages];
+ tPageOffset *= tPageSize;
+
+ for (unsigned int i = 0; i < fsRWReq->numberOfPages; i++) {
+ jam();
+ Uint32 varIndex = fsRWReq->data.listOfMemPages.varIndex[i];
+
+ if (varIndex >= tNRR) {
+ jam();
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }//if
+ request->par.readWrite.pages[i].buf = &tWA[varIndex * tClusterSize];
+ request->par.readWrite.pages[i].size = tPageSize;
+ request->par.readWrite.pages[i].offset = tPageOffset + (i*tPageSize);
+ }//for
+ request->par.readWrite.numberOfPages = fsRWReq->numberOfPages;
+ break;
+ // make it a writev or readv
+ }//case
+
+ default: {
+ jam();
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }//default
+
+ }//switch
+
+ ndbrequire(forward(openFile, request));
+ return;
+
+error:
+ theRequestPool->put(request);
+ FsRef * const fsRef = (FsRef *)&signal->theData[0];
+ fsRef->userPointer = userPointer;
+ fsRef->setErrorCode(fsRef->errorCode, errorCode);
+ fsRef->osErrorCode = ~0; // Indicate local error
+ switch (action) {
+ case Request:: write:
+ case Request:: writeSync: {
+ jam();
+ sendSignal(userRef, GSN_FSWRITEREF, signal, 3, JBB);
+ break;
+ }//case
+ case Request:: read: {
+ jam();
+ sendSignal(userRef, GSN_FSREADREF, signal, 3, JBB);
+ }//case
+ }//switch
+ return;
+}
+
+/*
+ PR0: File Pointer , theData[0]
+ DR0: User reference, theData[1]
+ DR1: User Pointer, etc.
+ DR2: Flag
+ DR3: Var number
+ DR4: amount of pages
+ DR5->: Memory Page id and File page id according to Flag
+*/
+void
+Ndbfs::execFSWRITEREQ(Signal* signal)
+{
+ jamEntry();
+ const FsReadWriteReq * const fsWriteReq = (FsReadWriteReq *)&signal->theData[0];
+
+ if (fsWriteReq->getSyncFlag(fsWriteReq->operationFlag) == true){
+ jam();
+ readWriteRequest( Request::writeSync, signal );
+ } else {
+ jam();
+ readWriteRequest( Request::write, signal );
+ }
+}
+
+/*
+ PR0: File Pointer
+ DR0: User reference
+ DR1: User Pointer
+ DR2: Flag
+ DR3: Var number
+ DR4: amount of pages
+ DR5->: Memory Page id and File page id according to Flag
+*/
+void
+Ndbfs::execFSREADREQ(Signal* signal)
+{
+ jamEntry();
+ readWriteRequest( Request::read, signal );
+}
+
+/*
+ * PR0: File Pointer DR0: User reference DR1: User Pointer
+ */
+void
+Ndbfs::execFSSYNCREQ(Signal * signal)
+{
+ jamEntry();
+ Uint16 filePointer = (Uint16)signal->theData[0];
+ BlockReference userRef = signal->theData[1];
+ const UintR userPointer = signal->theData[2];
+ AsyncFile* openFile = theOpenFiles.find(filePointer);
+
+ if (openFile == NULL) {
+ jam(); //file not open
+ FsRef * const fsRef = (FsRef *)&signal->theData[0];
+ fsRef->userPointer = userPointer;
+ fsRef->setErrorCode(fsRef->errorCode, FsRef::fsErrFileDoesNotExist);
+ fsRef->osErrorCode = ~0; // Indicate local error
+ sendSignal(userRef, GSN_FSSYNCREF, signal, 3, JBB);
+ return;
+ }
+
+ Request *request = theRequestPool->get();
+ request->error = 0;
+ request->action = Request::sync;
+ request->set(userRef, userPointer, filePointer);
+ request->file = openFile;
+ request->theTrace = signal->getTrace();
+
+ ndbrequire(forward(openFile,request));
+}
+
+void
+Ndbfs::execFSAPPENDREQ(Signal * signal)
+{
+ const FsAppendReq * const fsReq = (FsAppendReq *)&signal->theData[0];
+ const Uint16 filePointer = (Uint16)fsReq->filePointer;
+ const UintR userPointer = fsReq->userPointer;
+ const BlockReference userRef = fsReq->userReference;
+ const BlockNumber blockNumber = refToBlock(userRef);
+
+ FsRef::NdbfsErrorCodeType errorCode;
+
+ AsyncFile* openFile = theOpenFiles.find(filePointer);
+ const NewVARIABLE *myBaseAddrRef = &getBat(blockNumber)[fsReq->varIndex];
+
+ const Uint32* tWA = (const Uint32*)myBaseAddrRef->WA;
+ const Uint32 tSz = myBaseAddrRef->nrr;
+ const Uint32 offset = fsReq->offset;
+ const Uint32 size = fsReq->size;
+ Request *request = theRequestPool->get();
+
+ if (openFile == NULL) {
+ jam();
+ errorCode = FsRef::fsErrFileDoesNotExist;
+ goto error;
+ }
+
+ if (myBaseAddrRef == NULL) {
+ jam(); // Ensure that a valid variable is used
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }
+
+ if (fsReq->varIndex >= getBatSize(blockNumber)) {
+ jam();// Ensure that a valid variable is used
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }
+
+ if(offset + size > tSz){
+ jam(); // Ensure that a valid variable is used
+ errorCode = FsRef::fsErrInvalidParameters;
+ goto error;
+ }
+
+ request->error = 0;
+ request->set(userRef, userPointer, filePointer);
+ request->file = openFile;
+ request->action = Request::append;
+ request->theTrace = signal->getTrace();
+
+ request->par.append.buf = (const char *)(tWA + offset);
+ request->par.append.size = size << 2;
+
+ ndbrequire(forward(openFile, request));
+ return;
+
+error:
+ jam();
+ theRequestPool->put(request);
+ FsRef * const fsRef = (FsRef *)&signal->theData[0];
+ fsRef->userPointer = userPointer;
+ fsRef->setErrorCode(fsRef->errorCode, errorCode);
+ fsRef->osErrorCode = ~0; // Indicate local error
+
+ jam();
+ sendSignal(userRef, GSN_FSAPPENDREF, signal, 3, JBB);
+ return;
+}
+
+Uint16
+Ndbfs::newId()
+{
+ // finds a new key, eg a new filepointer
+ for (int i = 1; i < SHRT_MAX; i++)
+ {
+ if (theLastId == SHRT_MAX) {
+ jam();
+ theLastId = 1;
+ } else {
+ jam();
+ theLastId++;
+ }
+
+ if(theOpenFiles.find(theLastId) == NULL) {
+ jam();
+ return theLastId;
+ }
+ }
+ ndbrequire(1 == 0);
+ // The program will not reach this point
+ return 0;
+}
+
+AsyncFile*
+Ndbfs::createAsyncFile(){
+
+ // Check limit of open files
+ if (theFiles.size()+1 == m_maxFiles) {
+ // Print info about all open files
+ for (unsigned i = 0; i < theFiles.size(); i++){
+ AsyncFile* file = theFiles[i];
+ ndbout_c("%2d (0x%x): %s", i, file, file->isOpen()?"OPEN":"CLOSED");
+ }
+ ERROR_SET(fatal, AFS_ERROR_MAXOPEN,""," Ndbfs::createAsyncFile");
+ }
+
+ AsyncFile* file = new AsyncFile;
+ file->doStart(theFileSystemPath);
+
+ // Put the file in list of all files
+ theFiles.push_back(file);
+
+#ifdef VM_TRACE
+ infoEvent("NDBFS: Created new file thread %d", theFiles.size());
+#endif
+
+ return file;
+}
+
+AsyncFile*
+Ndbfs::getIdleFile(){
+ AsyncFile* file;
+ if (theIdleFiles.size() > 0){
+ file = theIdleFiles[0];
+ theIdleFiles.erase(0);
+ } else {
+ file = createAsyncFile();
+ }
+ return file;
+}
+
+
+
+void
+Ndbfs::report(Request * request, Signal* signal)
+{
+ const Uint32 orgTrace = signal->getTrace();
+ signal->setTrace(request->theTrace);
+ const BlockReference ref = request->theUserReference;
+ if (request->error) {
+ jam();
+ // Initialise FsRef signal
+ FsRef * const fsRef = (FsRef *)&signal->theData[0];
+ fsRef->userPointer = request->theUserPointer;
+ fsRef->setErrorCode(fsRef->errorCode, translateErrno(request->error));
+ fsRef->osErrorCode = request->error;
+
+ switch (request->action) {
+ case Request:: open: {
+ jam();
+ // Put the file back in idle files list
+ theIdleFiles.push_back(request->file);
+ sendSignal(ref, GSN_FSOPENREF, signal, FsRef::SignalLength, JBB);
+ break;
+ }
+ case Request:: closeRemove:
+ case Request:: close: {
+ jam();
+ sendSignal(ref, GSN_FSCLOSEREF, signal, FsRef::SignalLength, JBB);
+ break;
+ }
+ case Request:: writeSync:
+ case Request:: writevSync:
+ case Request:: write:
+ case Request:: writev: {
+ jam();
+ sendSignal(ref, GSN_FSWRITEREF, signal, FsRef::SignalLength, JBB);
+ break;
+ }
+ case Request:: read:
+ case Request:: readv: {
+ jam();
+ sendSignal(ref, GSN_FSREADREF, signal, FsRef::SignalLength, JBB);
+ break;
+ }
+ case Request:: sync: {
+ jam();
+ sendSignal(ref, GSN_FSSYNCREF, signal, FsRef::SignalLength, JBB);
+ break;
+ }
+ case Request::append: {
+ jam();
+ sendSignal(ref, GSN_FSAPPENDREF, signal, FsRef::SignalLength, JBB);
+ break;
+ }
+ case Request::rmrf: {
+ jam();
+ // Put the file back in idle files list
+ theIdleFiles.push_back(request->file);
+ sendSignal(ref, GSN_FSREMOVEREF, signal, FsRef::SignalLength, JBB);
+ break;
+ }
+
+ case Request:: end: {
+ // Report nothing
+ break;
+ }
+ }//switch
+ } else {
+ jam();
+ FsConf * const fsConf = (FsConf *)&signal->theData[0];
+ fsConf->userPointer = request->theUserPointer;
+ switch (request->action) {
+ case Request:: open: {
+ jam();
+ theOpenFiles.insert(request->file, request->theFilePointer);
+
+ // Keep track on max number of opened files
+ if (theOpenFiles.size() > m_maxOpenedFiles)
+ m_maxOpenedFiles = theOpenFiles.size();
+
+ fsConf->filePointer = request->theFilePointer;
+ sendSignal(ref, GSN_FSOPENCONF, signal, 3, JBB);
+ break;
+ }
+ case Request:: closeRemove:
+ case Request:: close: {
+ jam();
+ // removes the file from OpenFiles list
+ theOpenFiles.erase(request->theFilePointer);
+ // Put the file in idle files list
+ theIdleFiles.push_back(request->file);
+ sendSignal(ref, GSN_FSCLOSECONF, signal, 1, JBB);
+ break;
+ }
+ case Request:: writeSync:
+ case Request:: writevSync:
+ case Request:: write:
+ case Request:: writev: {
+ jam();
+ sendSignal(ref, GSN_FSWRITECONF, signal, 1, JBB);
+ break;
+ }
+ case Request:: read:
+ case Request:: readv: {
+ jam();
+ sendSignal(ref, GSN_FSREADCONF, signal, 1, JBB);
+ break;
+ }
+ case Request:: sync: {
+ jam();
+ sendSignal(ref, GSN_FSSYNCCONF, signal, 1, JBB);
+ break;
+ }//case
+ case Request::append: {
+ jam();
+ signal->theData[1] = request->par.append.size;
+ sendSignal(ref, GSN_FSAPPENDCONF, signal, 2, JBB);
+ break;
+ }
+ case Request::rmrf: {
+ jam();
+ // Put the file in idle files list
+ theIdleFiles.push_back(request->file);
+ sendSignal(ref, GSN_FSREMOVECONF, signal, 1, JBB);
+ break;
+ }
+ case Request:: end: {
+ // Report nothing
+ break;
+ }
+ }
+ }//if
+ signal->setTrace(orgTrace);
+}
+
+
+bool
+Ndbfs::scanIPC(Signal* signal)
+{
+ Request* request = theFromThreads.tryReadChannel();
+ jam();
+ if (request) {
+ jam();
+ report(request, signal);
+ theRequestPool->put(request);
+ return &request;
+ }
+ return false;
+}
+
+#if defined NDB_WIN32
+int Ndbfs::translateErrno(int aErrno)
+{
+ switch (aErrno)
+ {
+ //permission denied
+ case ERROR_ACCESS_DENIED:
+
+ return FsRef::fsErrPermissionDenied;
+ //temporary not accessible
+ case ERROR_PATH_BUSY:
+ case ERROR_NO_MORE_SEARCH_HANDLES:
+
+ return FsRef::fsErrTemporaryNotAccessible;
+ //no space left on device
+ case ERROR_HANDLE_DISK_FULL:
+ case ERROR_DISK_FULL:
+
+ return FsRef::fsErrNoSpaceLeftOnDevice;
+ //none valid parameters
+ case ERROR_INVALID_HANDLE:
+ case ERROR_INVALID_DRIVE:
+ case ERROR_INVALID_ACCESS:
+ case ERROR_HANDLE_EOF:
+ case ERROR_BUFFER_OVERFLOW:
+
+ return FsRef::fsErrInvalidParameters;
+ //environment error
+ case ERROR_CRC:
+ case ERROR_ARENA_TRASHED:
+ case ERROR_BAD_ENVIRONMENT:
+ case ERROR_INVALID_BLOCK:
+ case ERROR_WRITE_FAULT:
+ case ERROR_READ_FAULT:
+ case ERROR_OPEN_FAILED:
+
+ return FsRef::fsErrEnvironmentError;
+
+ //no more process resources
+ case ERROR_TOO_MANY_OPEN_FILES:
+ case ERROR_NOT_ENOUGH_MEMORY:
+ case ERROR_OUTOFMEMORY:
+ return FsRef::fsErrNoMoreResources;
+ //no file
+ case ERROR_FILE_NOT_FOUND:
+ return FsRef::fsErrFileDoesNotExist;
+
+ case ERR_ReadUnderflow:
+ return FsRef::fsErrReadUnderflow;
+
+ default:
+ return FsRef::fsErrUnknown;
+ }
+}
+#elif defined NDB_OSE || defined NDB_SOFTOSE
+int Ndbfs::translateErrno(int aErrno)
+{
+ switch (aErrno)
+ {
+ //permission denied
+ case EACCES:
+ case EROFS:
+ case ENXIO:
+ return FsRef::fsErrPermissionDenied;
+ //temporary not accessible
+ case EAGAIN:
+ case ETIMEDOUT:
+ case ENOLCK:
+ return FsRef::fsErrTemporaryNotAccessible;
+ //no space left on device
+ case ENFILE:
+ case EDQUOT:
+ case ENOSPC:
+ return FsRef::fsErrNoSpaceLeftOnDevice;
+ //none valid parameters
+ case EINVAL:
+ case EFBIG:
+ case EBADF:
+ case ENAMETOOLONG:
+ case EFAULT:
+ case EISDIR:
+ return FsRef::fsErrInvalidParameters;
+ //environment error
+ case EMLINK:
+ case ELOOP:
+ return FsRef::fsErrEnvironmentError;
+
+ //no more process resources
+ case EMFILE:
+ case ENOMEM:
+ return FsRef::fsErrNoMoreResources;
+ //no file
+ case ENOENT:
+ return FsRef::fsErrFileDoesNotExist;
+
+ case ERR_ReadUnderflow:
+ return FsRef::fsErrReadUnderflow;
+
+ default:
+ return FsRef::fsErrUnknown;
+ }
+}
+#else
+int Ndbfs::translateErrno(int aErrno)
+{
+ switch (aErrno)
+ {
+ //permission denied
+ case EACCES:
+ case EROFS:
+ case ENXIO:
+ return FsRef::fsErrPermissionDenied;
+ //temporary not accessible
+ case EAGAIN:
+ case ETIMEDOUT:
+ case ENOLCK:
+ case EINTR:
+ case EIO:
+ return FsRef::fsErrTemporaryNotAccessible;
+ //no space left on device
+ case ENFILE:
+ case EDQUOT:
+#ifndef NDB_MACOSX
+ case ENOSR:
+#endif
+ case ENOSPC:
+ case EFBIG:
+ return FsRef::fsErrNoSpaceLeftOnDevice;
+ //none valid parameters
+ case EINVAL:
+ case EBADF:
+ case ENAMETOOLONG:
+ case EFAULT:
+ case EISDIR:
+ case ENOTDIR:
+ case EEXIST:
+ case ETXTBSY:
+ return FsRef::fsErrInvalidParameters;
+ //environment error
+ case ELOOP:
+#ifndef NDB_MACOSX
+ case ENOLINK:
+ case EMULTIHOP:
+#endif
+#ifndef NDB_LINUX
+ case EOPNOTSUPP:
+ case ESPIPE:
+#endif
+ case EPIPE:
+ return FsRef::fsErrEnvironmentError;
+
+ //no more process resources
+ case EMFILE:
+ case ENOMEM:
+ return FsRef::fsErrNoMoreResources;
+ //no file
+ case ENOENT:
+ return FsRef::fsErrFileDoesNotExist;
+
+ case ERR_ReadUnderflow:
+ return FsRef::fsErrReadUnderflow;
+
+ default:
+ return FsRef::fsErrUnknown;
+ }
+}
+#endif
+
+
+
+void
+Ndbfs::execCONTINUEB(Signal* signal)
+{
+ jamEntry();
+ if (signal->theData[0] == NdbfsContinueB::ZSCAN_MEMORYCHANNEL_10MS_DELAY) {
+ jam();
+
+ // Also send CONTINUEB to ourself in order to scan for
+ // incoming answers from AsyncFile on MemoryChannel theFromThreads
+ signal->theData[0] = NdbfsContinueB::ZSCAN_MEMORYCHANNEL_10MS_DELAY;
+ sendSignalWithDelay(reference(), GSN_CONTINUEB, signal, 10, 1);
+ if (scanningInProgress == true) {
+ jam();
+ return;
+ }
+ }
+ if (scanIPC(signal)) {
+ jam();
+ scanningInProgress = true;
+ signal->theData[0] = NdbfsContinueB::ZSCAN_MEMORYCHANNEL_NO_DELAY;
+ sendSignal(reference(), GSN_CONTINUEB, signal, 1, JBB);
+ } else {
+ jam();
+ scanningInProgress = false;
+ }
+ return;
+}
+
+bool Global_useO_SYNC = false;
+bool Global_useO_DIRECT = false;
+bool Global_unlinkO_CREAT = false;
+Uint32 Global_syncFreq = 1024 * 1024;
+
+void
+Ndbfs::execDUMP_STATE_ORD(Signal* signal)
+{
+ if(signal->theData[0] == 19){
+ if(signal->length() > 1){
+ Global_useO_SYNC = signal->theData[1];
+ }
+ if(signal->length() > 2){
+ Global_syncFreq = signal->theData[2] * 1024 * 1024;
+ }
+ if(signal->length() > 3){
+ Global_unlinkO_CREAT = signal->theData[3];
+ }
+ if(signal->length() > 4){
+ Global_useO_DIRECT = signal->theData[4];
+ }
+ ndbout_c("useO_SYNC = %d syncFreq = %d unlinkO_CREATE = %d O_DIRECT = %d",
+ Global_useO_SYNC,
+ Global_syncFreq,
+ Global_unlinkO_CREAT,
+ Global_useO_DIRECT);
+ return;
+ }
+ if(signal->theData[0] == DumpStateOrd::NdbfsDumpFileStat){
+ infoEvent("NDBFS: Files: %d Open files: %d",
+ theFiles.size(),
+ theOpenFiles.size());
+ infoEvent(" Idle files: %d Max opened files: %d",
+ theIdleFiles.size(),
+ m_maxOpenedFiles);
+ infoEvent(" Max files: %d",
+ m_maxFiles);
+ infoEvent(" Requests: %d",
+ theRequestPool->size());
+
+ return;
+ }
+ if(signal->theData[0] == DumpStateOrd::NdbfsDumpOpenFiles){
+ infoEvent("NDBFS: Dump open files: %d", theOpenFiles.size());
+
+ for (unsigned i = 0; i < theOpenFiles.size(); i++){
+ AsyncFile* file = theOpenFiles.getFile(i);
+ infoEvent("%2d (0x%x): %s", i,file, file->theFileName.c_str());
+ }
+ return;
+ }
+ if(signal->theData[0] == DumpStateOrd::NdbfsDumpAllFiles){
+ infoEvent("NDBFS: Dump all files: %d", theFiles.size());
+
+ for (unsigned i = 0; i < theFiles.size(); i++){
+ AsyncFile* file = theFiles[i];
+ infoEvent("%2d (0x%x): %s", i,file, file->isOpen()?"OPEN":"CLOSED");
+ }
+ return;
+ }
+ if(signal->theData[0] == DumpStateOrd::NdbfsDumpIdleFiles){
+ infoEvent("NDBFS: Dump idle files: %d", theIdleFiles.size());
+
+ for (unsigned i = 0; i < theIdleFiles.size(); i++){
+ AsyncFile* file = theIdleFiles[i];
+ infoEvent("%2d (0x%x): %s", i,file, file->isOpen()?"OPEN":"CLOSED");
+ }
+ return;
+ }
+}//Ndbfs::execDUMP_STATE_ORD()
+
+
+
+BLOCK_FUNCTIONS(Ndbfs);
+
diff --git a/ndb/src/kernel/blocks/ndbfs/Ndbfs.hpp b/ndb/src/kernel/blocks/ndbfs/Ndbfs.hpp
new file mode 100644
index 00000000000..080196a9ea5
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/Ndbfs.hpp
@@ -0,0 +1,126 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SIMBLOCKASYNCFILESYSTEM_H
+#define SIMBLOCKASYNCFILESYSTEM_H
+
+#include <pc.hpp>
+#include <SimulatedBlock.hpp>
+#include "Pool.hpp"
+#include "AsyncFile.hpp"
+#include "OpenFiles.hpp"
+
+
+
+// Because one NDB Signal request can result in multiple requests to
+// AsyncFile one class must be made responsible to keep track
+// of all out standing request and when all are finished the result
+// must be reported to the sending block.
+
+
+class Ndbfs : public SimulatedBlock
+{
+public:
+ Ndbfs(const class Configuration & conf);
+ virtual ~Ndbfs();
+
+protected:
+ BLOCK_DEFINES(Ndbfs);
+
+ // The signal processing functions
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execFSOPENREQ(Signal* signal);
+ void execFSCLOSEREQ(Signal* signal);
+ void execFSWRITEREQ(Signal* signal);
+ void execFSREADREQ(Signal* signal);
+ void execFSSYNCREQ(Signal* signal);
+ void execFSAPPENDREQ(Signal* signal);
+ void execFSREMOVEREQ(Signal* signal);
+ void execSTTOR(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+
+ bool scanningInProgress;
+ Uint16 newId();
+
+private:
+ int forward(AsyncFile *file, Request* Request);
+ void report(Request* request, Signal* signal);
+ bool scanIPC(Signal* signal);
+
+ // Declared but not defined
+ Ndbfs(Ndbfs & );
+ void operator = (Ndbfs &);
+
+ // Used for uniqe number generation
+ Uint16 theLastId;
+ BlockReference cownref;
+
+ // Communication from files
+ MemoryChannel<Request> theFromThreads;
+
+ Pool<Request>* theRequestPool;
+
+ AsyncFile* createAsyncFile();
+ AsyncFile* getIdleFile();
+
+ Vector<AsyncFile*> theFiles; // List all created AsyncFiles
+ Vector<AsyncFile*> theIdleFiles; // List of idle AsyncFiles
+ OpenFiles theOpenFiles; // List of open AsyncFiles
+ const char * theFileSystemPath;
+
+ // Statistics variables
+ Uint32 m_maxOpenedFiles;
+
+ // Limit for max number of AsyncFiles created
+ Uint32 m_maxFiles;
+
+ void readWriteRequest( int action, Signal * signal );
+
+ static int translateErrno(int aErrno);
+};
+
+class VoidFs : public SimulatedBlock
+{
+public:
+ VoidFs(const class Configuration & conf);
+ virtual ~VoidFs();
+
+protected:
+ BLOCK_DEFINES(VoidFs);
+
+ // The signal processing functions
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execFSOPENREQ(Signal* signal);
+ void execFSCLOSEREQ(Signal* signal);
+ void execFSWRITEREQ(Signal* signal);
+ void execFSREADREQ(Signal* signal);
+ void execFSSYNCREQ(Signal* signal);
+ void execFSAPPENDREQ(Signal* signal);
+ void execFSREMOVEREQ(Signal* signal);
+ void execSTTOR(Signal* signal);
+
+private:
+ // Declared but not defined
+ VoidFs(VoidFs & );
+ void operator = (VoidFs &);
+
+ // Used for uniqe number generation
+ Uint32 c_maxFileNo;
+};
+
+#endif
+
+
diff --git a/ndb/src/kernel/blocks/ndbfs/OpenFiles.hpp b/ndb/src/kernel/blocks/ndbfs/OpenFiles.hpp
new file mode 100644
index 00000000000..b944bb5485b
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/OpenFiles.hpp
@@ -0,0 +1,114 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef OPENFILES_H
+#define OPENFILES_H
+
+#include <Vector.hpp>
+
+class OpenFiles
+{
+public:
+ OpenFiles(){ }
+
+ /* Get a pointer to the file with id */
+ AsyncFile* find(Uint16 id);
+ /* Insert file with id */
+ bool insert(AsyncFile* file, Uint16 id);
+ /* Erase file with id */
+ bool erase(Uint16 id);
+ /* Get number of open files */
+ unsigned size();
+
+ Uint16 getId(unsigned i);
+ AsyncFile* getFile(unsigned i);
+
+
+private:
+
+ class OpenFileItem {
+ public:
+ OpenFileItem(): m_file(NULL), m_id(0){};
+
+ AsyncFile* m_file;
+ Uint16 m_id;
+ };
+
+ Vector<OpenFileItem> m_files;
+};
+
+
+//*****************************************************************************
+inline AsyncFile* OpenFiles::find(Uint16 id){
+ for (unsigned i = 0; i < m_files.size(); i++){
+ if (m_files[i].m_id == id){
+ return m_files[i].m_file;
+ }
+ }
+ return NULL;
+}
+
+//*****************************************************************************
+inline bool OpenFiles::erase(Uint16 id){
+ for (unsigned i = 0; i < m_files.size(); i++){
+ if (m_files[i].m_id == id){
+ m_files.erase(i);
+ return true;
+ }
+ }
+ // Item was not found in list
+ return false;
+}
+
+
+//*****************************************************************************
+inline bool OpenFiles::insert(AsyncFile* file, Uint16 id){
+ // Check if file has already been opened
+ for (unsigned i = 0; i < m_files.size(); i++){
+ if(m_files[i].m_file == NULL)
+ continue;
+
+ if(strcmp(m_files[i].m_file->theFileName.c_str(),
+ file->theFileName.c_str()) == 0){
+ ERROR_SET(fatal, AFS_ERROR_ALLREADY_OPEN,"","OpenFiles::insert()");
+ }
+ }
+
+ // Insert the file into vector
+ OpenFileItem openFile;
+ openFile.m_id = id;
+ openFile.m_file = file;
+ m_files.push_back(openFile);
+
+ return true;
+}
+
+//*****************************************************************************
+inline Uint16 OpenFiles::getId(unsigned i){
+ return m_files[i].m_id;
+}
+
+//*****************************************************************************
+inline AsyncFile* OpenFiles::getFile(unsigned i){
+ return m_files[i].m_file;
+}
+
+//*****************************************************************************
+inline unsigned OpenFiles::size(){
+ return m_files.size();
+}
+
+#endif
diff --git a/ndb/src/kernel/blocks/ndbfs/Pool.hpp b/ndb/src/kernel/blocks/ndbfs/Pool.hpp
new file mode 100644
index 00000000000..a26fa730727
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/Pool.hpp
@@ -0,0 +1,262 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FOR_LIB_POOL_H
+#define FOR_LIB_POOL_H
+
+
+//===========================================================================
+//
+// .PUBLIC
+//
+//===========================================================================
+
+////////////////////////////////////////////////////////////////
+//
+// enum { defInitSize = 256, defIncSize = 64 };
+// Description: type to store initial and incremental size in.
+//
+////////////////////////////////////////////////////////////////
+//
+// Pool(int anInitSize = defInitSize, int anIncSize = defIncSize);
+// Description:
+// Constructor. Allocates anInitSize of objects <template argument>.
+// When the pool runs out of elements, anIncSize elements are added to the
+// pool. (When the pool is not optimized to allocate multiple elements
+// more efficient, the anIncSize MUST be set to 1 to get the best
+// performance...
+//
+// Parameters:
+// defInitSize: Initial size of the pool (# of elements in the pool)
+// defIncSize: # of elements added to the pool when a request to an empty
+// pool is made.
+// Return value:
+// _
+// Errors:
+// -
+// Asserts:
+// _
+//
+////////////////////////////////////////////////////////////////
+//
+// virtual ~Pool();
+// Description:
+// Elements in the pool are all deallocated.
+// Parameters:
+// _
+// Return value:
+// _
+// Errors:
+// -
+// Asserts:
+// theEmptyNodeList==0. No elements are in still in use.
+//
+////////////////////////////////////////////////////////////////
+//
+// T& get();
+// Description:
+// get's an element from the Pool.
+// Parameters:
+// _
+// Return value:
+// T& the element extracted from the Pool. (element must be cleared to
+// mimick newly created element)
+// Errors:
+// -
+// Asserts:
+// _
+//
+////////////////////////////////////////////////////////////////
+//
+// void put(T& aT);
+// Description:
+// Returns an element to the pool.
+// Parameters:
+// aT The element to put back in the pool
+// Return value:
+// void
+// Errors:
+// -
+// Asserts:
+// The pool has "empty" elements, to put element back in...
+//
+//===========================================================================
+//
+// .PRIVATE
+//
+//===========================================================================
+
+////////////////////////////////////////////////////////////////
+//
+// void allocate(int aSize);
+// Description:
+// add aSize elements to the pool
+// Parameters:
+// aSize: # of elements to add to the pool
+// Return value:
+// void
+// Errors:
+// -
+// Asserts:
+// _
+//
+////////////////////////////////////////////////////////////////
+//
+// void deallocate();
+// Description:
+// frees all elements kept in the pool.
+// Parameters:
+// _
+// Return value:
+// void
+// Errors:
+// -
+// Asserts:
+// No elements are "empty" i.e. in use.
+//
+//===========================================================================
+//
+// .PRIVATE
+//
+//===========================================================================
+
+////////////////////////////////////////////////////////////////
+//
+// Pool<T>& operator=(const Pool<T>& cp);
+// Description:
+// Prohibit use of assignement operator.
+// Parameters:
+// cp
+// Return value:
+// Pool<T>&
+// Asserts:
+// _
+//
+////////////////////////////////////////////////////////////////
+//
+// Pool(const Pool<T>& cp);
+// Description:
+// Prohibit use of default copy constructor.
+// Parameters:
+// cp
+// Return value:
+// _
+// Errors:
+// -
+// Asserts:
+// _
+//
+////////////////////////////////////////////////////////////////
+//
+// int initSize;
+// Description: size of the initial size of the pool
+//
+////////////////////////////////////////////////////////////////
+//
+// int incSize;
+// Description: # of elements added to the pool when pool is exhausted.
+//
+////////////////////////////////////////////////////////////////
+//
+// PoolElement<T>* theFullNodeList;
+// Description: List to contain all "unused" elements in the pool
+//
+////////////////////////////////////////////////////////////////
+//
+// PoolElement<T>* theEmptyNodeList;
+// Description: List to contain all "in use" elements in the pool
+//
+//-------------------------------------------------------------------------
+
+template <class T>
+class Pool
+{
+public:
+ enum { defInitSize = 256, defIncSize = 64 };
+
+ Pool(int anInitSize = defInitSize, int anIncSize = defIncSize) :
+ theIncSize(anIncSize),
+ theTop(0),
+ theCurrentSize(0),
+ theList(0)
+ {
+ allocate(anInitSize);
+ }
+
+ virtual ~Pool(void)
+ {
+ for (int i=0; i <theTop ; ++i)
+ delete theList[i];
+
+ delete []theList;
+ }
+
+ T* get();
+ void put(T* aT);
+
+ unsigned size(){ return theTop; };
+
+protected:
+ void allocate(int aSize)
+ {
+ T** tList = theList;
+ int i;
+ theList = new T*[aSize+theCurrentSize];
+ REQUIRE(theList != 0, "Allocate in Pool.hpp failed");
+ // allocate full list
+ for (i = 0; i < theTop; i++) {
+ theList[i] = tList[i];
+ }
+ delete []tList;
+ for (; (theTop < aSize); theTop++){
+ theList[theTop] = (T*)new T;
+ }
+ theCurrentSize += aSize;
+ }
+
+private:
+ Pool<T>& operator=(const Pool<T>& cp);
+ Pool(const Pool<T>& cp);
+
+ int theIncSize;
+ int theTop;
+ int theCurrentSize;
+
+ T** theList;
+};
+
+//******************************************************************************
+template <class T> inline T* Pool<T>::get()
+{
+ T* tmp;
+ if( theTop == 0 )
+ {
+ allocate(theIncSize);
+ }
+ --theTop;
+ tmp = theList[theTop];
+ return tmp;
+}
+
+//
+//******************************************************************************
+template <class T> inline void Pool<T>::put(T* aT)
+{
+ theList[theTop]= aT;
+ ++theTop;
+}
+
+#endif
diff --git a/ndb/src/kernel/blocks/ndbfs/VoidFs.cpp b/ndb/src/kernel/blocks/ndbfs/VoidFs.cpp
new file mode 100644
index 00000000000..d3407e8d4e7
--- /dev/null
+++ b/ndb/src/kernel/blocks/ndbfs/VoidFs.cpp
@@ -0,0 +1,200 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <limits.h>
+#include <errno.h>
+
+#include "Ndbfs.hpp"
+#include "AsyncFile.hpp"
+#include "Filename.hpp"
+#include "Error.hpp"
+
+#include <signaldata/FsOpenReq.hpp>
+#include <signaldata/FsCloseReq.hpp>
+#include <signaldata/FsReadWriteReq.hpp>
+#include <signaldata/FsAppendReq.hpp>
+#include <signaldata/FsRemoveReq.hpp>
+#include <signaldata/FsConf.hpp>
+#include <signaldata/FsRef.hpp>
+#include <signaldata/NdbfsContinueB.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+
+#include <RefConvert.hpp>
+#include <NdbSleep.h>
+#include <NdbOut.hpp>
+#include <Configuration.hpp>
+
+#define DEBUG(x) { ndbout << "FS::" << x << endl; }
+
+VoidFs::VoidFs(const Configuration & conf) :
+ SimulatedBlock(NDBFS, conf)
+{
+ BLOCK_CONSTRUCTOR(VoidFs);
+
+ // Set received signals
+ addRecSignal(GSN_DUMP_STATE_ORD, &VoidFs::execDUMP_STATE_ORD);
+ addRecSignal(GSN_STTOR, &VoidFs::execSTTOR);
+ addRecSignal(GSN_FSOPENREQ, &VoidFs::execFSOPENREQ);
+ addRecSignal(GSN_FSCLOSEREQ, &VoidFs::execFSCLOSEREQ);
+ addRecSignal(GSN_FSWRITEREQ, &VoidFs::execFSWRITEREQ);
+ addRecSignal(GSN_FSREADREQ, &VoidFs::execFSREADREQ);
+ addRecSignal(GSN_FSSYNCREQ, &VoidFs::execFSSYNCREQ);
+ addRecSignal(GSN_FSAPPENDREQ, &VoidFs::execFSAPPENDREQ);
+ addRecSignal(GSN_FSREMOVEREQ, &VoidFs::execFSREMOVEREQ);
+ // Set send signals
+}
+
+VoidFs::~VoidFs()
+{
+}
+
+void
+VoidFs::execSTTOR(Signal* signal)
+{
+ jamEntry();
+
+ if(signal->theData[1] == 0){ // StartPhase 0
+ jam();
+ signal->theData[3] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 4, JBB);
+ return;
+ }
+ ndbrequire(0);
+}
+
+void
+VoidFs::execFSOPENREQ(Signal* signal)
+{
+ jamEntry();
+ const FsOpenReq * const fsOpenReq = (FsOpenReq *)&signal->theData[0];
+ const BlockReference userRef = fsOpenReq->userReference;
+ const Uint32 userPointer = fsOpenReq->userPointer;
+
+ Uint32 flags = fsOpenReq->fileFlags;
+ if(flags == FsOpenReq::OM_READONLY){
+ // Initialise FsRef signal
+ FsRef * const fsRef = (FsRef *)&signal->theData[0];
+ fsRef->userPointer = userPointer;
+ fsRef->errorCode = FsRef::fsErrFileDoesNotExist;
+ fsRef->osErrorCode = ~0;
+ sendSignal(userRef, GSN_FSOPENREF, signal, 3, JBB);
+ return;
+ }
+
+ if(flags & FsOpenReq::OM_WRITEONLY || flags & FsOpenReq::OM_READWRITE){
+ signal->theData[0] = userPointer;
+ signal->theData[1] = c_maxFileNo++;
+ sendSignal(userRef, GSN_FSOPENCONF, signal, 2, JBB);
+ }
+}
+
+void
+VoidFs::execFSREMOVEREQ(Signal* signal)
+{
+ jamEntry();
+ const FsRemoveReq * const req = (FsRemoveReq *)signal->getDataPtr();
+ const Uint32 userRef = req->userReference;
+ const Uint32 userPointer = req->userPointer;
+
+ signal->theData[0] = userPointer;
+ sendSignal(userRef, GSN_FSREMOVECONF, signal, 1, JBB);
+}
+
+/*
+ * PR0: File Pointer DR0: User reference DR1: User Pointer DR2: Flag bit 0= 1
+ * remove file
+ */
+void
+VoidFs::execFSCLOSEREQ(Signal * signal)
+{
+ jamEntry();
+
+ const FsCloseReq * const req = (FsCloseReq *)signal->getDataPtr();
+ const Uint32 userRef = req->userReference;
+ const Uint32 userPointer = req->userPointer;
+
+ signal->theData[0] = userPointer;
+ sendSignal(userRef, GSN_FSCLOSECONF, signal, 1, JBB);
+}
+
+void
+VoidFs::execFSWRITEREQ(Signal* signal)
+{
+ jamEntry();
+ const FsReadWriteReq * const req = (FsReadWriteReq *)signal->getDataPtr();
+ const Uint32 userRef = req->userReference;
+ const Uint32 userPointer = req->userPointer;
+
+ signal->theData[0] = userPointer;
+ sendSignal(userRef, GSN_FSWRITECONF, signal, 1, JBB);
+}
+
+void
+VoidFs::execFSREADREQ(Signal* signal)
+{
+ jamEntry();
+
+ const FsReadWriteReq * const req = (FsReadWriteReq *)signal->getDataPtr();
+ const Uint32 userRef = req->userReference;
+ const Uint32 userPointer = req->userPointer;
+
+ signal->theData[0] = userPointer;
+ sendSignal(userRef, GSN_FSREADCONF, signal, 1, JBB);
+#if 0
+ FsRef * const fsRef = (FsRef *)&signal->theData[0];
+ fsRef->userPointer = userPointer;
+ fsRef->errorCode = FsRef::fsErrEnvironmentError;
+ fsRef->osErrorCode = ~0; // Indicate local error
+ sendSignal(userRef, GSN_FSREADREF, signal, 3, JBB);
+#endif
+}
+
+void
+VoidFs::execFSSYNCREQ(Signal * signal)
+{
+ jamEntry();
+
+ BlockReference userRef = signal->theData[1];
+ const UintR userPointer = signal->theData[2];
+
+ signal->theData[0] = userPointer;
+ sendSignal(userRef, GSN_FSSYNCCONF, signal, 1, JBB);
+
+ return;
+}
+
+void
+VoidFs::execFSAPPENDREQ(Signal * signal)
+{
+ const FsAppendReq * const fsReq = (FsAppendReq *)&signal->theData[0];
+ const UintR userPointer = fsReq->userPointer;
+ const BlockReference userRef = fsReq->userReference;
+ const Uint32 size = fsReq->size;
+
+ signal->theData[0] = userPointer;
+ signal->theData[1] = size << 2;
+ sendSignal(userRef, GSN_FSAPPENDCONF, signal, 2, JBB);
+}
+
+void
+VoidFs::execDUMP_STATE_ORD(Signal* signal)
+{
+}//VoidFs::execDUMP_STATE_ORD()
+
+
+
+BLOCK_FUNCTIONS(VoidFs);
+
diff --git a/ndb/src/kernel/blocks/new-block.tar.gz b/ndb/src/kernel/blocks/new-block.tar.gz
new file mode 100644
index 00000000000..327503ea0b1
--- /dev/null
+++ b/ndb/src/kernel/blocks/new-block.tar.gz
Binary files differ
diff --git a/ndb/src/kernel/blocks/qmgr/Makefile b/ndb/src/kernel/blocks/qmgr/Makefile
new file mode 100644
index 00000000000..cd15643ea60
--- /dev/null
+++ b/ndb/src/kernel/blocks/qmgr/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := qmgr
+
+SOURCES = \
+ QmgrInit.cpp \
+ QmgrMain.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/qmgr/Qmgr.hpp b/ndb/src/kernel/blocks/qmgr/Qmgr.hpp
new file mode 100644
index 00000000000..7d2abd34ebe
--- /dev/null
+++ b/ndb/src/kernel/blocks/qmgr/Qmgr.hpp
@@ -0,0 +1,488 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef QMGR_H
+#define QMGR_H
+
+
+#include <pc.hpp>
+#include <NdbTick.h>
+#include <SimulatedBlock.hpp>
+#include <NodeBitmask.hpp>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/ArbitSignalData.hpp>
+#include <signaldata/CmRegSignalData.hpp>
+#include <signaldata/ApiRegSignalData.hpp>
+#include <signaldata/FailRep.hpp>
+
+#include "timer.hpp"
+
+#ifdef QMGR_C
+
+#define NO_REG_APP 1
+/* Boolean flags --------------------------------*/
+#define ZNULL 0xfffe
+
+/* Delay values, ms -----------------------------*/
+#define ZDELAY_REGREQ 1000
+
+/* Phase of QMGR node ------------------------*/
+#define ZINIT 1 /* All nodes start in phase INIT */
+#define ZWAITING 2 /* Node is connecting to cluster */
+#define ZRUNNING 3 /* Node is running in the cluster */
+#define ZBLOCKED 4 /* Node is blocked from the cluster */
+#define ZWAIT_PRESIDENT 5
+#define ZDEAD 6
+#define ZAPI_ACTIVE 7 /* API IS RUNNING IN NODE */
+#define ZFAIL_CLOSING 8 /* API/NDB IS DISCONNECTING */
+#define ZPREPARE_FAIL 9 /* PREPARATION FOR FAILURE */
+#define ZAPI_INACTIVE 10 /* Inactive API */
+
+/* Type of refuse in CM_NODEINFOREF -------------*/
+#define ZNOT_RUNNING 0
+
+/* Type of continue in CONTINUEB ----------------*/
+#define ZREGREQ_TIMELIMIT 0
+#define ZHB_HANDLING 1
+#define ZREGREQ_MASTER_TIMELIMIT 2
+#define ZAPI_HB_HANDLING 3
+#define ZTIMER_HANDLING 4
+#define ZARBIT_HANDLING 5
+
+/* Error Codes ------------------------------*/
+#define ZERRTOOMANY 1101
+#define ZERRALREADYREG 1102
+#define ZERRNHMISSING 1103
+#define ZERRNLMISSING 1104
+#define ZERRAPPMISSING 1105
+#define ZERROR_NOT_IN_CFGFILE 1106
+#define ZERROR_TIMEOUT 1107
+#define ZERROR_NOT_ZINIT 1108
+#define ZERROR_NODEINFOREF 1109
+#define ZERROR_NOTLOCALQMGR 1110
+#define ZERROR_NOTRUNNING 1111
+#define ZCOULD_NOT_OCCUR_ERROR 1112
+#define ZTIME_OUT_ERROR 1113
+#define ZERROR_NOT_DEAD 1114
+#define ZDECLARED_FAIL_ERROR 1115
+#define ZOWN_NODE_ERROR 1116
+#define ZWRONG_STATE_ERROR 1117
+#define ZNODE_ZERO_ERROR 1118
+#define ZWRONG_NODE_ERROR 1119
+
+#endif
+
+
+class Qmgr : public SimulatedBlock {
+public:
+ // State values
+ enum QmgrState {
+ Q_NOT_ACTIVE = 0,
+ Q_ACTIVE = 1
+ };
+
+ enum FailState {
+ NORMAL = 0,
+ WAITING_FOR_FAILCONF1 = 1,
+ WAITING_FOR_FAILCONF2 = 2,
+ WAITING_FOR_NDB_FAILCONF = 3
+ };
+
+ // Records
+ struct NodeRec {
+ UintR ndynamicId;
+ UintR phase;
+ UintR alarmCount;
+
+ bool m_connected;
+ QmgrState sendPrepFailReqStatus;
+ QmgrState sendCommitFailReqStatus;
+ QmgrState sendCmAddPrepStatus;
+ QmgrState sendCmAddCommitStatus;
+ QmgrState sendPresToStatus;
+ FailState failState;
+ BlockReference rcv[2]; // remember which failconf we have received
+ BlockReference blockRef;
+
+ NodeRec() { }
+ }; /* p2c: size = 52 bytes */
+
+ typedef Ptr<NodeRec> NodeRecPtr;
+
+ struct RegApp {
+ NdbNodeBitmask m_runNodes;
+ char name[15 + 1];
+ UintR noofapps;
+ UintR noofpending;
+ BlockReference blockref;
+ Uint16 version;
+ Uint16 activity;
+ };
+
+ typedef Ptr<RegApp> RegAppPtr;
+
+ enum ArbitState {
+ ARBIT_NULL = 0,
+ ARBIT_INIT = 1, // create new ticket
+ ARBIT_FIND = 2, // find candidate arbitrator node
+ ARBIT_PREP1 = 3, // PREP db nodes with null ticket
+ ARBIT_PREP2 = 4, // PREP db nodes with current ticket
+ ARBIT_START = 5, // START arbitrator API thread
+ ARBIT_RUN = 6, // running with arbitrator
+ ARBIT_CHOOSE = 7, // ask arbitrator after network partition
+ ARBIT_CRASH = 8 // crash ourselves
+ };
+
+ struct ArbitRec {
+ ArbitState state; // state
+ bool newstate; // flag to initialize new state
+ unsigned thread; // identifies a continueB "thread"
+ NodeId node; // current arbitrator candidate
+ ArbitTicket ticket; // ticket
+ NodeBitmask apiMask[1+2]; // arbitrators 0=all 1,2=per rank
+ NodeBitmask newMask; // new nodes to process in RUN state
+ Uint8 sendCount; // control send/recv of signals
+ Uint8 recvCount;
+ NodeBitmask recvMask; // left to recv
+ Uint32 code; // code field from signal
+ Uint32 failureNr; // cfailureNr at arbitration start
+ Uint32 timeout; // timeout for CHOOSE state
+ NDB_TICKS timestamp; // timestamp for checking timeouts
+
+ inline bool match(ArbitSignalData* sd) {
+ return
+ node == sd->node &&
+ ticket.match(sd->ticket);
+ }
+
+ inline void setTimestamp() {
+ timestamp = NdbTick_CurrentMillisecond();
+ }
+
+ inline NDB_TICKS getTimediff() {
+ NDB_TICKS now = NdbTick_CurrentMillisecond();
+ return now < timestamp ? 0 : now - timestamp;
+ }
+ };
+
+public:
+ Qmgr(const class Configuration &);
+ virtual ~Qmgr();
+
+private:
+ BLOCK_DEFINES(Qmgr);
+
+ // Transit signals
+ void execDEBUG_SIG(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+ void execCM_HEARTBEAT(Signal* signal);
+ void execCM_ADD(Signal* signal);
+ void execCM_ACKADD(Signal* signal);
+ void execCM_APPCHG(Signal* signal);
+ void execCM_REGREQ(Signal* signal);
+ void execCM_REGCONF(Signal* signal);
+ void execCM_REGREF(Signal* signal);
+ void execCM_NODEINFOREQ(Signal* signal);
+ void execCM_NODEINFOCONF(Signal* signal);
+ void execCM_NODEINFOREF(Signal* signal);
+ void execPREP_FAILREQ(Signal* signal);
+ void execPREP_FAILCONF(Signal* signal);
+ void execPREP_FAILREF(Signal* signal);
+ void execCOMMIT_FAILREQ(Signal* signal);
+ void execCOMMIT_FAILCONF(Signal* signal);
+ void execFAIL_REP(Signal* signal);
+ void execPRES_TOREQ(Signal* signal);
+ void execPRES_TOCONF(Signal* signal);
+ void execDISCONNECT_REP(Signal* signal);
+ void execSYSTEM_ERROR(Signal* signal);
+
+ // Received signals
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execCONNECT_REP(Signal* signal);
+ void execNDB_FAILCONF(Signal* signal);
+ void execSTTOR(Signal* signal);
+ void execAPPL_REGREQ(Signal* signal);
+ void execAPPL_STARTREG(Signal* signal);
+ void execAPPL_RUN(Signal* signal);
+ void execCM_INIT(Signal* signal);
+ void execCM_INFOCONF(Signal* signal);
+ void execCLOSE_COMCONF(Signal* signal);
+ void execAPI_REGREQ(Signal* signal);
+ void execAPI_FAILCONF(Signal* signal);
+ void execREAD_NODESREQ(Signal* signal);
+ void execSET_VAR_REQ(Signal* signal);
+
+
+ void execAPI_VERSION_REQ(Signal* signal);
+
+ // Arbitration signals
+ void execARBIT_CFG(Signal* signal);
+ void execARBIT_PREPREQ(Signal* signal);
+ void execARBIT_PREPCONF(Signal* signal);
+ void execARBIT_PREPREF(Signal* signal);
+ void execARBIT_STARTCONF(Signal* signal);
+ void execARBIT_STARTREF(Signal* signal);
+ void execARBIT_CHOOSECONF(Signal* signal);
+ void execARBIT_CHOOSEREF(Signal* signal);
+ void execARBIT_STOPREP(Signal* signal);
+
+ // Statement blocks
+ void node_failed(Signal* signal, Uint16 aFailedNode);
+ void checkStartInterface(Signal* signal);
+ void applchangerep(Signal* signal,
+ UintR aRegApp,
+ Uint16 aNode,
+ UintR aType,
+ UintR aVersion);
+ void cmappAdd(Signal* signal,
+ UintR aRegApp,
+ Uint16 aNode,
+ UintR aType,
+ UintR aVersion);
+ void cmappStart(Signal* signal,
+ UintR aRegApp,
+ Uint16 aNode,
+ UintR aType,
+ UintR aVersion);
+ void failReport(Signal* signal,
+ Uint16 aFailedNode,
+ UintR aSendFailRep,
+ FailRep::FailCause failCause);
+ void findNeighbours(Signal* signal);
+ Uint16 translateDynamicIdToNodeId(Signal* signal, UintR TdynamicId);
+ UintR getDynamicId(Signal* signal);
+ void initData(Signal* signal);
+ void prepareAdd(Signal* signal, Uint16 addNode);
+ void sendappchg(Signal* signal, UintR aRegApp, Uint16 aNode);
+ void sendCloseComReq(Signal* signal, BlockReference TBRef, Uint16 TfailNo);
+ void sendPrepFailReq(Signal* signal, Uint16 aNode);
+ void sendApiFailReq(Signal* signal, Uint16 aFailedNode);
+ void sendApiRegRef(Signal*, Uint32 ref, ApiRegRef::ErrorCode);
+
+ // Generated statement blocks
+ void electionWon();
+ void cmInfoconf010Lab(Signal* signal);
+ void apiHbHandlingLab(Signal* signal);
+ void timerHandlingLab(Signal* signal);
+ void hbReceivedLab(Signal* signal);
+ void cmAdd010Lab(Signal* signal);
+ void cmAckadd010Lab(Signal* signal);
+ void cmAppchg010Lab(Signal* signal);
+ void sendCmRegrefLab(Signal* signal, BlockReference ref,
+ CmRegRef::ErrorCode);
+ void systemErrorBecauseOtherNodeFailed(Signal* signal, NodeId);
+ void systemErrorLab(Signal* signal,
+ const char* message = NULL);
+ void cmRegref010Lab(Signal* signal);
+ void cmNodeinforeq010Lab(Signal* signal);
+ void cmNodeinfoconf010Lab(Signal* signal);
+ void prepFailReqLab(Signal* signal);
+ void prepFailConfLab(Signal* signal);
+ void prepFailRefLab(Signal* signal);
+ void commitFailReqLab(Signal* signal);
+ void commitFailConfLab(Signal* signal);
+ void failReportLab(Signal* signal, Uint16 aFailedNode,
+ FailRep::FailCause aFailCause);
+ void sendCommitFailReq(Signal* signal);
+ void presToConfLab(Signal* signal);
+ void sendSttorryLab(Signal* signal);
+ void sttor020Lab(Signal* signal);
+ void applRegreq010Lab(Signal* signal);
+ void applStartreg010Lab(Signal* signal);
+ void applRun010Lab(Signal* signal);
+ void cmInit010Lab(Signal* signal);
+ void closeComConfLab(Signal* signal);
+ void apiRegReqLab(Signal* signal);
+ void regreqTimelimitLab(Signal* signal, UintR callTime);
+ void cmRegreq010Lab(Signal* signal);
+ void cmRegconf010Lab(Signal* signal);
+ void sttor010Lab(Signal* signal);
+ void sendHeartbeat(Signal* signal);
+ void checkHeartbeat(Signal* signal);
+ void setHbDelay(UintR aHbDelay);
+ void setHbApiDelay(UintR aHbApiDelay);
+ void setArbitTimeout(UintR aArbitTimeout);
+
+ // Interface to arbitration module
+ void handleArbitStart(Signal* signal);
+ void handleArbitApiFail(Signal* signal, Uint16 nodeId);
+ void handleArbitNdbAdd(Signal* signal, Uint16 nodeId);
+ void handleArbitCheck(Signal* signal);
+
+ // Private arbitration routines
+ Uint32 getArbitDelay();
+ Uint32 getArbitTimeout();
+ void startArbitThread(Signal* signal);
+ void runArbitThread(Signal* signal);
+ void stateArbitInit(Signal* signal);
+ void stateArbitFind(Signal* signal);
+ void stateArbitPrep(Signal* signal);
+ void stateArbitStart(Signal* signal);
+ void stateArbitRun(Signal* signal);
+ void stateArbitChoose(Signal* signal);
+ void stateArbitCrash(Signal* signal);
+ void computeArbitNdbMask(NodeBitmask& aMask);
+ void reportArbitEvent(Signal* signal, EventReport::EventType type);
+
+ // Initialisation
+ void initData();
+ void initRecords();
+
+ // Transit signals
+ // Variables
+
+ bool checkAPIVersion(NodeId, Uint32 nodeVersion, Uint32 ownVersion) const;
+ bool checkNDBVersion(NodeId, Uint32 nodeVersion, Uint32 ownVersion) const;
+
+private:
+ void sendPrepFailReqRef(Signal* signal,
+ Uint32 dstBlockRef,
+ GlobalSignalNumber gsn,
+ Uint32 blockRef,
+ Uint32 failNo,
+ Uint32 noOfNodes,
+ const NodeId theNodes[]);
+
+
+
+ /* Wait this time until we try to join the */
+ /* cluster again */
+
+ /**** Common stored variables ****/
+
+ NodeRec *nodeRec;
+ RegApp * regApp;
+ ArbitRec arbitRec;
+
+ /* Block references ------------------------------*/
+ BlockReference cpdistref; /* Dist. ref of president */
+
+ /* Node numbers. ---------------------------------*/
+ Uint16 cneighbourl; /* Node no. of lower neighbour */
+ Uint16 cneighbourh; /* Node no. of higher neighbour */
+ Uint16 cpresident; /* Node no. of president */
+
+ /* Counters --------------------------------------*/
+ Uint16 cnoOfNodes; /* Static node counter */
+ Uint16 cclustersize; /* Currently not used */
+ /* Status flags ----------------------------------*/
+
+ Uint16 cstartseq; /* Marks what startseq we are in according to
+ STTOR */
+
+ Uint16 cpresidentBusy; /* Only used by the president, ZTRUE / ZFALSE */
+ Uint16 cacceptRegreq; /* Used by president, ZTRUE / ZFALSE */
+ Uint16 cwaitContinuebPhase1;
+ Uint16 cwaitContinuebPhase2;
+ Uint16 creadyDistCom;
+
+ UintR cstartNo;
+ Uint16 c_regReqReqSent;
+ Uint16 c_regReqReqRecv;
+ Uint64 c_stopElectionTime;
+ Uint16 cpresidentCandidate;
+ Uint16 cdelayRegreq;
+ Uint16 cpresidentAlive;
+ Uint16 csignalkey;
+ Uint16 cstartNode;
+ Uint16 cnoFailedNodes;
+ Uint16 cnoPrepFailedNodes;
+ Uint16 cnoCommitFailedNodes;
+ Uint16 cactivateApiCheck;
+ UintR chbApiDelay;
+
+ UintR ccommitFailureNr;
+ UintR cprepareFailureNr;
+ UintR ctoFailureNr;
+ UintR cfailureNr;
+
+ QmgrState ctoStatus;
+ UintR ccm_infoconfCounter;
+ UintR cLqhTimeSignalCount;
+ bool cHbSent;
+ NDB_TICKS clatestTransactionCheck;
+
+ class Timer interface_check_timer;
+ class Timer hb_check_timer;
+ class Timer hb_send_timer;
+ class Timer hb_api_timer;
+
+
+ UintR cnodemask[NdbNodeBitmask::Size];
+ Uint16 cfailedNodes[MAX_NDB_NODES];
+ Uint16 cprepFailedNodes[MAX_NDB_NODES];
+ Uint16 ccommitFailedNodes[MAX_NDB_NODES];
+
+ /***************************************************************************/
+ /* RECORD NODE_REC: The NodeList contains information about all other nodes
+ * in the cluster.
+ * Member variables:
+ * NTYPE [ ZACTIVE,
+ * ZPASSIVE, Marks the level of activity the
+ * node will show in the cluster.
+ * ZLISTENER ]
+ * PHASE [ ZINIT, = Initial face, before node is added
+ * to cluster
+ * ZWAITING, = Node is added to the cluster and
+ * ready to run
+ * ZRUNNING, = Node is up and running.
+ * ZBLOCKED = Node is not in the cluster
+ * ZAPI_ACTIVE = Node has an active application
+ * ZFAIL_CLOSING = Node is disconnecting
+ * ZDEAD ] = Node has been declared as dead
+ * ALARM_COUNT No of times an alarm has been sent before it is
+ * acknowledged
+ ***************************************************************************/
+ /*************************************************************************
+ * RECORD REG_APP: The REG_APP record is used to store information about
+ * each registered application running on the current node.
+ * Member variables:
+ * BLOCKREF Reference of application block to receive cluster
+ * signals
+ * PTR Not used today but may be used by appl. in future
+ * NAME Unique name of application, max 15 char. long
+ * SUBTYPE Provided as a mechanism for applications to have
+ * more than one type running in the same application
+ * ring. i.e. NDB & NDB-API
+ * VERSION Version no. of application. Two different versions
+ * will be handled as different applications.
+ * TYPE [ ZACTIVE,
+ * ZPASSIVE,
+ * ZLISTENER ] Type of member in the cluster
+ * ACTIVITY [ ZADD, Application has been registered on
+ * node.
+ * ZSTART, Application is ready to start
+ * running distributed.
+ * ZRUN, Application is running actively.
+ * ZDELETE ] Application is beeing removed from
+ * the node.
+ * HBDELAY Delay time for periodic intervalls.
+ * STATUS Heartbeat status, indicates if app is responding
+ * to HBREQ.
+ * RUNNODES() If value is ZTRUE -> app. is also running on the
+ * indexed node.
+ * NOOFAPPS No. of applications left to register themselves as
+ * ready to start, STATUS = ZSTART before we can send
+ * APPL_STARTCONF.
+ * NOOFPENDING No. of apps that have registered themselfs as ready
+ * to start before this app has. We need this since
+ * we set NOOFAPPS when we receive the local
+ * APPL_START. NOOFPENDING is subtracted from NOOFAPPS
+ * when NOOFAPPS is set.
+ **************************************************************************/
+
+};
+#endif
diff --git a/ndb/src/kernel/blocks/qmgr/QmgrInit.cpp b/ndb/src/kernel/blocks/qmgr/QmgrInit.cpp
new file mode 100644
index 00000000000..ffc1448548d
--- /dev/null
+++ b/ndb/src/kernel/blocks/qmgr/QmgrInit.cpp
@@ -0,0 +1,115 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#define QMGR_C
+#include "Qmgr.hpp"
+#include <Configuration.hpp>
+
+#define DEBUG(x) { ndbout << "Qmgr::" << x << endl; }
+
+
+void Qmgr::initData()
+{
+ creadyDistCom = ZFALSE;
+
+ // Records with constant sizes
+ nodeRec = new NodeRec[MAX_NODES];
+ regApp = new RegApp[NO_REG_APP];
+
+ cclustersize = 0;
+ cnoCommitFailedNodes = 0;
+}//Qmgr::initData()
+
+void Qmgr::initRecords()
+{
+ // Records with dynamic sizes
+}//Qmgr::initRecords()
+
+Qmgr::Qmgr(const class Configuration & conf)
+ : SimulatedBlock(QMGR, conf)
+{
+ BLOCK_CONSTRUCTOR(Qmgr);
+
+ // Transit signals
+ addRecSignal(GSN_DUMP_STATE_ORD, &Qmgr::execDUMP_STATE_ORD);
+ addRecSignal(GSN_DEBUG_SIG, &Qmgr::execDEBUG_SIG);
+ addRecSignal(GSN_CONTINUEB, &Qmgr::execCONTINUEB);
+ addRecSignal(GSN_CM_HEARTBEAT, &Qmgr::execCM_HEARTBEAT);
+ addRecSignal(GSN_CM_ADD, &Qmgr::execCM_ADD);
+ addRecSignal(GSN_CM_ACKADD, &Qmgr::execCM_ACKADD);
+ addRecSignal(GSN_CM_APPCHG, &Qmgr::execCM_APPCHG);
+ addRecSignal(GSN_CM_REGREQ, &Qmgr::execCM_REGREQ);
+ addRecSignal(GSN_CM_REGCONF, &Qmgr::execCM_REGCONF);
+ addRecSignal(GSN_CM_REGREF, &Qmgr::execCM_REGREF);
+ addRecSignal(GSN_CM_NODEINFOREQ, &Qmgr::execCM_NODEINFOREQ);
+ addRecSignal(GSN_CM_NODEINFOCONF, &Qmgr::execCM_NODEINFOCONF);
+ addRecSignal(GSN_CM_NODEINFOREF, &Qmgr::execCM_NODEINFOREF);
+ addRecSignal(GSN_PREP_FAILREQ, &Qmgr::execPREP_FAILREQ);
+ addRecSignal(GSN_PREP_FAILCONF, &Qmgr::execPREP_FAILCONF);
+ addRecSignal(GSN_PREP_FAILREF, &Qmgr::execPREP_FAILREF);
+ addRecSignal(GSN_COMMIT_FAILREQ, &Qmgr::execCOMMIT_FAILREQ);
+ addRecSignal(GSN_COMMIT_FAILCONF, &Qmgr::execCOMMIT_FAILCONF);
+ addRecSignal(GSN_FAIL_REP, &Qmgr::execFAIL_REP);
+ addRecSignal(GSN_PRES_TOREQ, &Qmgr::execPRES_TOREQ);
+ addRecSignal(GSN_PRES_TOCONF, &Qmgr::execPRES_TOCONF);
+ addRecSignal(GSN_CM_INFOCONF, &Qmgr::execCM_INFOCONF);
+
+ // Received signals
+ addRecSignal(GSN_CONNECT_REP, &Qmgr::execCONNECT_REP);
+ addRecSignal(GSN_NDB_FAILCONF, &Qmgr::execNDB_FAILCONF);
+ addRecSignal(GSN_STTOR, &Qmgr::execSTTOR);
+ addRecSignal(GSN_APPL_REGREQ, &Qmgr::execAPPL_REGREQ);
+ addRecSignal(GSN_APPL_STARTREG, &Qmgr::execAPPL_STARTREG);
+ addRecSignal(GSN_APPL_RUN, &Qmgr::execAPPL_RUN);
+ addRecSignal(GSN_CM_INIT, &Qmgr::execCM_INIT);
+ addRecSignal(GSN_CLOSE_COMCONF, &Qmgr::execCLOSE_COMCONF);
+ addRecSignal(GSN_API_REGREQ, &Qmgr::execAPI_REGREQ);
+ addRecSignal(GSN_API_VERSION_REQ, &Qmgr::execAPI_VERSION_REQ);
+ addRecSignal(GSN_DISCONNECT_REP, &Qmgr::execDISCONNECT_REP);
+ addRecSignal(GSN_API_FAILCONF, &Qmgr::execAPI_FAILCONF);
+ addRecSignal(GSN_READ_NODESREQ, &Qmgr::execREAD_NODESREQ);
+ addRecSignal(GSN_SET_VAR_REQ, &Qmgr::execSET_VAR_REQ);
+
+ // Arbitration signals
+ addRecSignal(GSN_ARBIT_CFG, &Qmgr::execARBIT_CFG);
+ addRecSignal(GSN_ARBIT_PREPREQ, &Qmgr::execARBIT_PREPREQ);
+ addRecSignal(GSN_ARBIT_PREPCONF, &Qmgr::execARBIT_PREPCONF);
+ addRecSignal(GSN_ARBIT_PREPREF, &Qmgr::execARBIT_PREPREF);
+ addRecSignal(GSN_ARBIT_STARTCONF, &Qmgr::execARBIT_STARTCONF);
+ addRecSignal(GSN_ARBIT_STARTREF, &Qmgr::execARBIT_STARTREF);
+ addRecSignal(GSN_ARBIT_CHOOSECONF, &Qmgr::execARBIT_CHOOSECONF);
+ addRecSignal(GSN_ARBIT_CHOOSEREF, &Qmgr::execARBIT_CHOOSEREF);
+ addRecSignal(GSN_ARBIT_STOPREP, &Qmgr::execARBIT_STOPREP);
+
+ initData();
+
+ const ClusterConfiguration::ClusterData & clusterConf =
+ theConfiguration.clusterConfigurationData() ;
+ setHbDelay(clusterConf.ispValues[0][2]); //cmInit->heartbeatDbDb);
+ setHbApiDelay(clusterConf.ispValues[0][3]); //;cmInit->heartbeatDbApi);
+ setArbitTimeout(clusterConf.ispValues[0][5]); //cmInit->arbitTimeout);
+}//Qmgr::Qmgr()
+
+Qmgr::~Qmgr()
+{
+ delete []nodeRec;
+ delete []regApp;
+}//Qmgr::~Qmgr()
+
+
+BLOCK_FUNCTIONS(Qmgr);
diff --git a/ndb/src/kernel/blocks/qmgr/QmgrMain.cpp b/ndb/src/kernel/blocks/qmgr/QmgrMain.cpp
new file mode 100644
index 00000000000..0f82f8def6f
--- /dev/null
+++ b/ndb/src/kernel/blocks/qmgr/QmgrMain.cpp
@@ -0,0 +1,4533 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#define QMGR_C
+#include "Qmgr.hpp"
+#include <pc.hpp>
+#include <NdbTick.h>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/SetVarReq.hpp>
+#include <signaldata/StartOrd.hpp>
+#include <signaldata/CmInit.hpp>
+#include <signaldata/CloseComReqConf.hpp>
+#include <signaldata/PrepFailReqRef.hpp>
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/ReadNodesConf.hpp>
+#include <signaldata/NFCompleteRep.hpp>
+#include <signaldata/CheckNodeGroups.hpp>
+#include <signaldata/ArbitSignalData.hpp>
+#include <signaldata/ApiRegSignalData.hpp>
+#include <signaldata/ApiVersion.hpp>
+#include <signaldata/BlockCommitOrd.hpp>
+#include <signaldata/FailRep.hpp>
+#include <signaldata/DisconnectRep.hpp>
+
+#include <ndb_version.h>
+
+#ifdef DEBUG_ARBIT
+#include <NdbOut.hpp>
+#endif
+
+// Signal entries and statement blocks
+/* 4 P R O G R A M */
+/*******************************/
+/* CMHEART_BEAT */
+/*******************************/
+void Qmgr::execCM_HEARTBEAT(Signal* signal)
+{
+ NodeRecPtr hbNodePtr;
+ jamEntry();
+ hbNodePtr.i = signal->theData[0];
+ ptrCheckGuard(hbNodePtr, MAX_NDB_NODES, nodeRec);
+ hbNodePtr.p->alarmCount = 0;
+ return;
+}//Qmgr::execCM_HEARTBEAT()
+
+/*******************************/
+/* CM_NODEINFOREF */
+/*******************************/
+void Qmgr::execCM_NODEINFOREF(Signal* signal)
+{
+ jamEntry();
+ systemErrorLab(signal);
+ return;
+}//Qmgr::execCM_NODEINFOREF()
+
+/*******************************/
+/* CONTINUEB */
+/*******************************/
+void Qmgr::execCONTINUEB(Signal* signal)
+{
+ UintR tdata0;
+ UintR tcontinuebType;
+
+ jamEntry();
+ tcontinuebType = signal->theData[0];
+ tdata0 = signal->theData[1];
+ switch (tcontinuebType) {
+ case ZREGREQ_TIMELIMIT:
+ jam();
+ if (cstartNo == tdata0) {
+ jam();
+ regreqTimelimitLab(signal, signal->theData[2]);
+ return;
+ }
+ break;
+ case ZREGREQ_MASTER_TIMELIMIT:
+ jam();
+ if (cstartNo != tdata0) {
+ jam();
+ return;
+ }//if
+ if (cpresidentBusy != ZTRUE) {
+ jam();
+ return;
+ }//if
+ failReportLab(signal, cstartNode, FailRep::ZSTART_IN_REGREQ);
+ return;
+ break;
+ case ZTIMER_HANDLING:
+ jam();
+ timerHandlingLab(signal);
+ return;
+ break;
+ case ZARBIT_HANDLING:
+ jam();
+ runArbitThread(signal);
+ return;
+ break;
+ default:
+ jam();
+ // ZCOULD_NOT_OCCUR_ERROR;
+ systemErrorLab(signal);
+ return;
+ break;
+ }//switch
+ return;
+}//Qmgr::execCONTINUEB()
+
+
+void Qmgr::execDEBUG_SIG(Signal* signal)
+{
+ NodeRecPtr debugNodePtr;
+ jamEntry();
+ debugNodePtr.i = signal->theData[0];
+ ptrCheckGuard(debugNodePtr, MAX_NODES, nodeRec);
+ return;
+}//Qmgr::execDEBUG_SIG()
+
+/*******************************/
+/* FAIL_REP */
+/*******************************/
+void Qmgr::execFAIL_REP(Signal* signal)
+{
+ const FailRep * const failRep = (FailRep *)&signal->theData[0];
+ const NodeId failNodeId = failRep->failNodeId;
+ const FailRep::FailCause failCause = (FailRep::FailCause)failRep->failCause;
+
+ jamEntry();
+ failReportLab(signal, failNodeId, failCause);
+ return;
+}//Qmgr::execFAIL_REP()
+
+/*******************************/
+/* PRES_TOREQ */
+/*******************************/
+void Qmgr::execPRES_TOREQ(Signal* signal)
+{
+ jamEntry();
+ BlockReference Tblockref = signal->theData[0];
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = ccommitFailureNr;
+ sendSignal(Tblockref, GSN_PRES_TOCONF, signal, 2, JBA);
+ return;
+}//Qmgr::execPRES_TOREQ()
+
+/*
+4.2 ADD NODE MODULE*/
+/*##########################################################################*/
+/*
+4.2.1 STTOR */
+/**--------------------------------------------------------------------------
+ * Start phase signal, must be handled by all blocks.
+ * QMGR is only interested in the first phase.
+ * During phase one we clear all registered applications.
+ *---------------------------------------------------------------------------*/
+/*******************************/
+/* STTOR */
+/*******************************/
+void Qmgr::execSTTOR(Signal* signal)
+{
+ jamEntry();
+ cstartseq = signal->theData[1];
+ csignalkey = signal->theData[6];
+ if (cstartseq == 1) {
+ jam();
+ initData(signal);
+ }
+
+ setNodeInfo(getOwnNodeId()).m_version = NDB_VERSION;
+
+ sendSttorryLab(signal);
+ return;
+}//Qmgr::execSTTOR()
+
+void Qmgr::sendSttorryLab(Signal* signal)
+{
+/****************************<*/
+/*< STTORRY <*/
+/****************************<*/
+ signal->theData[0] = csignalkey;
+ signal->theData[1] = 3;
+ signal->theData[2] = 2;
+ signal->theData[3] = 2;
+ signal->theData[4] = 255;
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 5, JBB);
+ return;
+}//Qmgr::sendSttorryLab()
+
+/*
+4.2.2 CM_INIT */
+/**--------------------------------------------------------------------------
+ * This signal is sent by the CLUSTERCTRL block.
+ * It initiates the QMGR and provides needed info about the
+ * cluster configuration (read from file).
+ *
+ * The signal starts all QMGR functions.
+ * It is possible to register applications before this but the QMGR will
+ * not be active before the registration face is complete.
+ *
+ * The CM_INIT will result in a one CM_NODEINFOREQ for each ndb node.
+ * We will also send a CONTINUEB to ourselves as a timelimit.
+ * If anyone sends a REF, CONF or a ( REQ with a lower NODENO than us ) during
+ * this time, we are not the president .
+ *--------------------------------------------------------------------------*/
+/*******************************/
+/* CM_INIT */
+/*******************************/
+void Qmgr::execCM_INIT(Signal* signal)
+{
+ jamEntry();
+
+ CmInit * const cmInit = (CmInit *)&signal->theData[0];
+
+ for(unsigned int i = 0; i<NdbNodeBitmask::Size; i++)
+ cnodemask[i] = cmInit->allNdbNodes[i];
+
+ cnoOfNodes = 0;
+ setHbDelay(cmInit->heartbeatDbDb);
+ setHbApiDelay(cmInit->heartbeatDbApi);
+ setArbitTimeout(cmInit->arbitTimeout);
+ arbitRec.state = ARBIT_NULL; // start state for all nodes
+ arbitRec.apiMask[0].clear(); // prepare for ARBIT_CFG
+
+ NodeRecPtr nodePtr;
+ for (nodePtr.i = 0; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (NdbNodeBitmask::get(cnodemask, nodePtr.i)) {
+ jam();
+
+ nodePtr.p->blockRef = calcQmgrBlockRef(nodePtr.i);
+ nodePtr.p->phase = ZINIT; /* Not added to cluster */
+ cnoOfNodes = cnoOfNodes + 1; /* Should never be changed after this loop. */
+ ndbrequire(getNodeInfo(nodePtr.i).m_type == NodeInfo::DB);
+ } else {
+ jam();
+ nodePtr.p->phase = ZBLOCKED;
+ }//if
+ }//for
+ for (nodePtr.i = MAX_NDB_NODES; nodePtr.i < MAX_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ nodePtr.p->phase = ZBLOCKED;
+ }//for
+
+ nodePtr.i = getOwnNodeId();
+ ptrAss(nodePtr, nodeRec);
+ nodePtr.p->phase = ZINIT;
+ nodePtr.p->m_connected = true;
+
+ /****************************<*/
+ /*< CM_INFOREQ <*/
+ /****************************<*/
+ signal->theData[0] = reference();
+ signal->theData[1] = getOwnNodeId();
+ sendSignal(CMVMI_REF, GSN_CM_INFOREQ, signal, 2, JBB);
+ return;
+}//Qmgr::execCM_INIT()
+
+void Qmgr::setHbDelay(UintR aHbDelay)
+{
+ hb_send_timer.setDelay(aHbDelay < 10 ? 10 : aHbDelay);
+ hb_send_timer.reset();
+ hb_check_timer.setDelay(aHbDelay < 10 ? 10 : aHbDelay);
+ hb_check_timer.reset();
+}
+
+void Qmgr::setHbApiDelay(UintR aHbApiDelay)
+{
+ chbApiDelay = (aHbApiDelay < 100 ? 100 : aHbApiDelay);
+ hb_api_timer.setDelay(chbApiDelay);
+ hb_api_timer.reset();
+}
+
+void Qmgr::setArbitTimeout(UintR aArbitTimeout)
+{
+ arbitRec.timeout = (aArbitTimeout < 10 ? 10 : aArbitTimeout);
+}
+
+void Qmgr::execCONNECT_REP(Signal* signal)
+{
+ NodeRecPtr connectNodePtr;
+ connectNodePtr.i = signal->theData[0];
+ ptrCheckGuard(connectNodePtr, MAX_NODES, nodeRec);
+ connectNodePtr.p->m_connected = true;
+
+ return;
+}//Qmgr::execCONNECT_REP()
+
+/*******************************/
+/* CM_INFOCONF */
+/*******************************/
+void Qmgr::execCM_INFOCONF(Signal* signal)
+{
+ cpresident = ZNIL;
+ cpresidentCandidate = getOwnNodeId();
+ cpresidentAlive = ZFALSE;
+ c_stopElectionTime = NdbTick_CurrentMillisecond();
+ c_stopElectionTime += 30000; // 30s
+ cmInfoconf010Lab(signal);
+
+#if 0
+ /*****************************************************/
+ /* Allow the CLUSTER CONTROL to send STTORRY */
+ /* CM_RUN */
+ /* so we can receive APPL_REGREQ from applications. */
+ /*****************************************************/
+ signal->theData[0] = 0;
+ sendSignal(CMVMI_REF, GSN_CM_RUN, signal, 1, JBB);
+#endif
+ return;
+}//Qmgr::execCM_INFOCONF()
+
+void Qmgr::cmInfoconf010Lab(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ c_regReqReqSent = c_regReqReqRecv = 0;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+
+ if(getNodeInfo(nodePtr.i).getType() != NodeInfo::DB)
+ continue;
+
+ if(!nodePtr.p->m_connected)
+ continue;
+
+ c_regReqReqSent++;
+ CmRegReq * const cmRegReq = (CmRegReq *)&signal->theData[0];
+ cmRegReq->blockRef = reference();
+ cmRegReq->nodeId = getOwnNodeId();
+ cmRegReq->version = NDB_VERSION;
+ sendSignal(nodePtr.p->blockRef, GSN_CM_REGREQ, signal,
+ CmRegReq::SignalLength, JBB);
+ }
+ cstartNo = cstartNo + 1;
+
+ //----------------------------------------
+ /* Wait for a while. When it returns */
+ /* we will check if we got any CM_REGREF*/
+ /* or CM_REGREQ (lower nodeid than our */
+ /* own). */
+ //----------------------------------------
+ signal->theData[0] = ZREGREQ_TIMELIMIT;
+ signal->theData[1] = cstartNo;
+ signal->theData[2] = 0;
+ sendSignalWithDelay(QMGR_REF, GSN_CONTINUEB, signal, 3 * cdelayRegreq, 3);
+ cwaitContinuebPhase1 = ZTRUE;
+ creadyDistCom = ZTRUE;
+ return;
+}//Qmgr::cmInfoconf010Lab()
+
+/*
+4.4.11 CM_REGREQ */
+/**--------------------------------------------------------------------------
+ * If this signal is received someone tries to get registrated.
+ * Only the president have the authority make decissions about new nodes,
+ * so only a president or a node that claims to be the president may send a
+ * reply to this signal.
+ * This signal can occur any time after that STTOR was received.
+ * CPRESIDENT: Timelimit has expired and someone has
+ * decided to enter the president role
+ * CPRESIDENT_CANDIDATE:
+ * Assigned when we receive a CM_REGREF, if we got more than one REF
+ * then we always keep the lowest nodenumber.
+ * We accept this nodeno as president when our timelimit expires
+ * We should consider the following cases:
+ * 1- We are the president. If we are busy by adding new nodes to cluster,
+ * then we have to refuse this node to be added.
+ * The refused node will try in ZREFUSE_ADD_TIME seconds again.
+ * If we are not busy then we confirm
+ *
+ * 2- We know the president, we dont bother us about this REQ.
+ * The president has also got this REQ and will take care of it.
+ *
+ * 3- The president are not known. We have received CM_INIT, so we compare the
+ * senders node number to GETOWNNODEID().
+ * If we have a lower number than the sender then we will claim
+ * that we are the president so we send him a refuse signal back.
+ * We have to wait for the CONTINUEB signal before we can enter the
+ * president role. If our GETOWNNODEID() if larger than sender node number,
+ * we are not the president and just have to wait for the
+ * reply signal (REF) to our CM_REGREQ_2.
+ * 4- We havent received the CM_INIT signal so we don't know who we are.
+ * Ignore the request.
+ *--------------------------------------------------------------------------*/
+/*******************************/
+/* CM_REGREQ */
+/*******************************/
+void Qmgr::execCM_REGREQ(Signal* signal)
+{
+ NodeRecPtr addNodePtr;
+ jamEntry();
+
+ CmRegReq * const cmRegReq = (CmRegReq *)&signal->theData[0];
+ const BlockReference Tblockref = cmRegReq->blockRef;
+ const Uint32 startingVersion = cmRegReq->version;
+ addNodePtr.i = cmRegReq->nodeId;
+
+ if (creadyDistCom == ZFALSE) {
+ jam();
+ /* NOT READY FOR DISTRIBUTED COMMUNICATION.*/
+ return;
+ }//if
+
+ if (!ndbCompatible_ndb_ndb(NDB_VERSION, startingVersion)) {
+ jam();
+ sendCmRegrefLab(signal, Tblockref, CmRegRef::ZINCOMPATIBLE_VERSION);
+ return;
+ }
+
+ ptrCheckGuard(addNodePtr, MAX_NDB_NODES, nodeRec);
+
+ if (cpresident != getOwnNodeId()){
+ jam();
+ if (cpresident == ZNIL) {
+ /***
+ * We don't know the president.
+ * If the node to be added has lower node id
+ * than our president cancidate. Set it as
+ * candidate
+ */
+ jam();
+ if (addNodePtr.i < cpresidentCandidate) {
+ jam();
+ cpresidentCandidate = addNodePtr.i;
+ }//if
+ sendCmRegrefLab(signal, Tblockref, CmRegRef::ZELECTION);
+ return;
+ }
+ /**
+ * We are not the president.
+ * We know the president.
+ * President will answer.
+ */
+ sendCmRegrefLab(signal, Tblockref, CmRegRef::ZNOT_PRESIDENT);
+ return;
+ }//if
+
+ if (cpresidentBusy == ZTRUE) {
+ jam();
+ /**
+ * President busy by adding another node
+ */
+ sendCmRegrefLab(signal, Tblockref, CmRegRef::ZBUSY_PRESIDENT);
+ return;
+ }//if
+
+ if (cacceptRegreq == ZFALSE &&
+ getNodeState().startLevel != NodeState::SL_STARTING) {
+ jam();
+ /**
+ * These checks are really confusing!
+ * The variables that is being checked are probably not
+ * set in the correct places.
+ */
+ sendCmRegrefLab(signal, Tblockref, CmRegRef::ZBUSY);
+ return;
+ }//if
+
+ if (ctoStatus == Q_ACTIVE) {
+ jam();
+ /**
+ * Active taking over as president
+ */
+ sendCmRegrefLab(signal, Tblockref, CmRegRef::ZBUSY_TO_PRES);
+ return;
+ }//if
+
+ if (addNodePtr.p->phase == ZBLOCKED) {
+ jam();
+ /**
+ * The new node is not in config file
+ */
+ sendCmRegrefLab(signal, Tblockref, CmRegRef::ZNOT_IN_CFG);
+ return;
+ }
+
+ if (addNodePtr.p->phase != ZINIT) {
+ jam();
+ sendCmRegrefLab(signal, Tblockref, CmRegRef::ZNOT_DEAD);
+ return;
+ }//if
+
+ jam();
+ /**
+ * WE ARE PRESIDENT AND WE ARE NOT BUSY ADDING ANOTHER NODE.
+ * WE WILL TAKE CARE OF THE INCLUSION OF THIS NODE INTO THE CLUSTER.
+ * WE NEED TO START TIME SUPERVISION OF THIS. SINCE WE CANNOT STOP
+ * TIMED SIGNAL IF THE INCLUSION IS INTERRUPTED WE IDENTIFY
+ * EACH INCLUSION WITH A UNIQUE IDENTITY. THIS IS CHECKED WHEN
+ * THE SIGNAL ARRIVES. IF IT HAS CHANGED THEN WE SIMPLY IGNORE
+ * THE TIMED SIGNAL.
+ */
+ cpresidentBusy = ZTRUE;
+
+ /**
+ * Indicates that we are busy with node start/restart and do
+ * not accept another start until this node is up and running
+ * (cpresidentBusy is released a little too early to use for this
+ * purpose).
+ */
+ cacceptRegreq = ZFALSE;
+ cstartNo = cstartNo + 1;
+ cstartNode = addNodePtr.i;
+ signal->theData[0] = ZREGREQ_MASTER_TIMELIMIT;
+ signal->theData[1] = cstartNo;
+ sendSignalWithDelay(QMGR_REF, GSN_CONTINUEB, signal, 30000, 2);
+ UintR TdynId = getDynamicId(signal); /* <- CDYNAMIC_ID */
+ prepareAdd(signal, addNodePtr.i);
+ setNodeInfo(addNodePtr.i).m_version = startingVersion;
+
+ /**
+ * Send "prepare for adding a new node" to all
+ * running nodes in cluster + the new node.
+ * Give permission to the new node to join the
+ * cluster
+ */
+ /*******************************/
+ /*< CM_REGCONF <*/
+ /*******************************/
+
+ CmRegConf * const cmRegConf = (CmRegConf *)&signal->theData[0];
+
+ cmRegConf->presidentBlockRef = reference();
+ cmRegConf->presidentNodeId = getOwnNodeId();
+ cmRegConf->presidentVersion = getNodeInfo(getOwnNodeId()).m_version;
+ cmRegConf->dynamicId = TdynId;
+ for(unsigned int i = 0; i<NdbNodeBitmask::Size; i++)
+ cmRegConf->allNdbNodes[i] = cnodemask[i];
+
+ sendSignal(Tblockref, GSN_CM_REGCONF, signal,
+ CmRegConf::SignalLength, JBB);
+ return;
+}//Qmgr::execCM_REGREQ()
+
+void Qmgr::sendCmRegrefLab(Signal* signal, BlockReference TBRef,
+ CmRegRef::ErrorCode Terror)
+{
+ CmRegRef* ref = (CmRegRef*)signal->getDataPtrSend();
+ ref->blockRef = reference();
+ ref->nodeId = getOwnNodeId();
+ ref->errorCode = Terror;
+ ref->presidentCandidate = cpresidentCandidate;
+ sendSignal(TBRef, GSN_CM_REGREF, signal,
+ CmRegRef::SignalLength, JBB);
+ return;
+}//Qmgr::sendCmRegrefLab()
+
+/*
+4.4.11 CM_REGCONF */
+/**--------------------------------------------------------------------------
+ * President gives permission to a node which wants to join the cluster.
+ * The president will prepare the cluster that a new node will be added to
+ * cluster. When the new node has set up all connections to the cluster,
+ * the president will send commit to all clusternodes so the phase of the
+ * new node can be changed to ZRUNNING.
+ *--------------------------------------------------------------------------*/
+/*******************************/
+/* CM_REGCONF */
+/*******************************/
+void Qmgr::execCM_REGCONF(Signal* signal)
+{
+ NodeRecPtr myNodePtr;
+ NodeRecPtr nodePtr;
+ NodeRecPtr presidentNodePtr;
+ jamEntry();
+
+ CmRegConf * const cmRegConf = (CmRegConf *)&signal->theData[0];
+ cwaitContinuebPhase1 = ZFALSE;
+ cwaitContinuebPhase2 = ZTRUE;
+
+ if (!ndbCompatible_ndb_ndb(NDB_VERSION, cmRegConf->presidentVersion)) {
+ jam();
+ char buf[128];
+ snprintf(buf,sizeof(buf),"incompatible version own=0x%x other=0x%x, shutting down", NDB_VERSION, cmRegConf->presidentVersion);
+ systemErrorLab(signal, buf);
+ return;
+ }
+
+ /**
+ * Check if all necessary connections has been established
+ */
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ if (NodeBitmask::get(cmRegConf->allNdbNodes, nodePtr.i) == true){
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (!nodePtr.p->m_connected) {
+ jam();
+
+ /**
+ * Missing connection
+ */
+#ifdef VM_TRACE
+ ndbout_c("Resending CM_REGCONF, node %d is not connected", nodePtr.i);
+ ndbout << " presidentBlockRef="<<cmRegConf->presidentBlockRef<<endl
+ << " presidentNodeId="<<cmRegConf->presidentNodeId<<endl
+ << " presidentVersion="<<cmRegConf->presidentVersion<<endl
+ << " dynamicId="<<cmRegConf->dynamicId<<endl;
+#endif
+ for(unsigned int i = 0; i<NdbNodeBitmask::Size; i++) {
+ jam();
+#ifdef VM_TRACE
+ ndbout << " " << i << ": "
+ << hex << cmRegConf->allNdbNodes[i]<<endl;
+#endif
+ }
+ sendSignalWithDelay(reference(), GSN_CM_REGCONF, signal, 100,
+ signal->getLength());
+ return;
+ }
+ }
+ }
+
+ cpdistref = cmRegConf->presidentBlockRef;
+ cpresident = cmRegConf->presidentNodeId;
+ UintR TdynamicId = cmRegConf->dynamicId;
+ for(unsigned int i = 0; i<NdbNodeBitmask::Size; i++)
+ cnodemask[i] = cmRegConf->allNdbNodes[i];
+
+/*--------------------------------------------------------------*/
+// Send this as an EVENT REPORT to inform about hearing about
+// other NDB node proclaiming to be president.
+/*--------------------------------------------------------------*/
+ signal->theData[0] = EventReport::CM_REGCONF;
+ signal->theData[1] = getOwnNodeId();
+ signal->theData[2] = cpresident;
+ signal->theData[3] = TdynamicId;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+ myNodePtr.p->ndynamicId = TdynamicId;
+ presidentNodePtr.i = cpresident;
+ ptrCheckGuard(presidentNodePtr, MAX_NDB_NODES, nodeRec);
+ cpdistref = presidentNodePtr.p->blockRef;
+
+ CmNodeInfoReq * const req = (CmNodeInfoReq*)signal->getDataPtrSend();
+ req->nodeId = getOwnNodeId();
+ req->dynamicId = myNodePtr.p->ndynamicId;
+ req->version = getNodeInfo(getOwnNodeId()).m_version;
+
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ if (NdbNodeBitmask::get(cnodemask, nodePtr.i) == true){
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ switch(nodePtr.p->phase){
+ case ZINIT: /* All nodes start in phase INIT */
+ jam();
+ break;
+ case ZWAITING: /* Node is connecting to cluster */
+ jam();
+ break;
+ case ZRUNNING: /* Node is running in the cluster */
+ jam();
+ break;
+ case ZBLOCKED: /* Node is blocked from the cluster */
+ jam();
+ break;
+ case ZWAIT_PRESIDENT:
+ jam();
+ break;
+ case ZDEAD:
+ jam();
+ break;
+ case ZAPI_ACTIVE: /* API IS RUNNING IN NODE */
+ jam();
+ break;
+ case ZFAIL_CLOSING: /* API/NDB IS DISCONNECTING */
+ jam();
+ break;
+ case ZPREPARE_FAIL: /* PREPARATION FOR FAILURE */
+ jam();
+ break;
+ case ZAPI_INACTIVE: /* Inactive API */
+ jam();
+ break;
+ default:
+ jam();
+ ndbout << "phase="<<nodePtr.p->phase<<endl;
+ break;
+ }
+ ndbrequire(nodePtr.p->phase == ZINIT);
+ ndbrequire(nodePtr.i != getOwnNodeId());
+ nodePtr.p->phase = ZWAITING;
+
+ sendSignal(nodePtr.p->blockRef, GSN_CM_NODEINFOREQ,
+ signal, CmNodeInfoReq::SignalLength, JBB);
+ }
+ }
+ return;
+}//Qmgr::execCM_REGCONF()
+
+/*
+4.4.11 CM_REGREF */
+/**--------------------------------------------------------------------------
+ * Only a president or a president candidate can refuse a node to get added to
+ * the cluster.
+ * Refuse reasons:
+ * ZBUSY We know that the sender is the president and we have to
+ * make a new CM_REGREQ.
+ * ZNOT_IN_CFG This node number is not specified in the configfile,
+ * SYSTEM ERROR
+ * ZELECTION Sender is a president candidate, his timelimit
+ * hasn't expired so maybe someone else will show up.
+ * Update the CPRESIDENT_CANDIDATE, then wait for our
+ * timelimit to expire.
+ *---------------------------------------------------------------------------*/
+/*******************************/
+/* CM_REGREF */
+/*******************************/
+void Qmgr::execCM_REGREF(Signal* signal)
+{
+ jamEntry();
+ c_regReqReqRecv++;
+
+ // Ignore block reference in data[0]
+ UintR TaddNodeno = signal->theData[1];
+ UintR TrefuseReason = signal->theData[2];
+ Uint32 candidate = signal->theData[3];
+
+ if(candidate != cpresidentCandidate){
+ jam();
+ c_regReqReqRecv = c_regReqReqSent + 1;
+ }
+
+ switch (TrefuseReason) {
+ case CmRegRef::ZINCOMPATIBLE_VERSION:
+ jam();
+ systemErrorLab(signal, "incompatible version, connection refused by running ndb node");
+ break;
+ case CmRegRef::ZBUSY:
+ case CmRegRef::ZBUSY_TO_PRES:
+ case CmRegRef::ZBUSY_PRESIDENT:
+ jam();
+ cpresidentAlive = ZTRUE;
+ signal->theData[3] = 0;
+ break;
+ case CmRegRef::ZNOT_IN_CFG:
+ jam();
+ progError(__LINE__, ERR_NODE_NOT_IN_CONFIG);
+ break;
+ case CmRegRef::ZNOT_DEAD:
+ jam();
+ if(TaddNodeno == getOwnNodeId() && cpresident == getOwnNodeId()){
+ jam();
+ cwaitContinuebPhase1 = ZFALSE;
+ cwaitContinuebPhase2 = ZFALSE;
+ return;
+ }
+ progError(__LINE__, ERR_NODE_NOT_DEAD);
+ break;
+ case CmRegRef::ZELECTION:
+ jam();
+ if (cwaitContinuebPhase1 == ZFALSE) {
+ jam();
+ signal->theData[3] = 1;
+ } else if (cpresidentCandidate > TaddNodeno) {
+ jam();
+//----------------------------------------
+/* We may already have a candidate */
+/* choose the lowest nodeno */
+//----------------------------------------
+ signal->theData[3] = 2;
+ cpresidentCandidate = TaddNodeno;
+ } else {
+ signal->theData[3] = 4;
+ }//if
+ break;
+ case CmRegRef::ZNOT_PRESIDENT:
+ jam();
+ cpresidentAlive = ZTRUE;
+ signal->theData[3] = 3;
+ break;
+ default:
+ jam();
+ signal->theData[3] = 5;
+ /*empty*/;
+ break;
+ }//switch
+/*--------------------------------------------------------------*/
+// Send this as an EVENT REPORT to inform about hearing about
+// other NDB node proclaiming not to be president.
+/*--------------------------------------------------------------*/
+ signal->theData[0] = EventReport::CM_REGREF;
+ signal->theData[1] = getOwnNodeId();
+ signal->theData[2] = TaddNodeno;
+//-----------------------------------------
+// signal->theData[3] filled in above
+//-----------------------------------------
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+
+ if(cpresidentAlive == ZTRUE){
+ jam();
+ return;
+ }
+
+ if(c_regReqReqSent != c_regReqReqRecv){
+ jam();
+ return;
+ }
+
+ if(cpresidentCandidate != getOwnNodeId()){
+ jam();
+ return;
+ }
+
+ /**
+ * All configured nodes has agreed
+ */
+ Uint64 now = NdbTick_CurrentMillisecond();
+ if((c_regReqReqRecv == cnoOfNodes) || now > c_stopElectionTime){
+ jam();
+
+ electionWon();
+#if 1
+ signal->theData[0] = 0;
+ sendSignal(CMVMI_REF, GSN_CM_RUN, signal, 1, JBB);
+#endif
+ /**
+ * Start timer handling
+ */
+ signal->theData[0] = ZTIMER_HANDLING;
+ sendSignal(QMGR_REF, GSN_CONTINUEB, signal, 10, JBB);
+ }
+
+ return;
+}//Qmgr::execCM_REGREF()
+
+void
+Qmgr::electionWon(){
+ NodeRecPtr myNodePtr;
+ cpresident = getOwnNodeId(); /* This node becomes president. */
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+
+ myNodePtr.p->phase = ZRUNNING;
+ cpdistref = reference();
+ cclustersize = 1;
+ cneighbourl = ZNIL;
+ cneighbourh = ZNIL;
+ myNodePtr.p->ndynamicId = 1;
+
+ cpresidentAlive = ZTRUE;
+ c_stopElectionTime = ~0;
+}
+
+/*
+4.4.11 CONTINUEB */
+/*--------------------------------------------------------------------------*/
+/* */
+/*--------------------------------------------------------------------------*/
+/****************************>---------------------------------------------*/
+/* CONTINUEB > SENDER: Own block, Own node */
+/****************************>-------+INPUT : TCONTINUEB_TYPE */
+/*--------------------------------------------------------------*/
+void Qmgr::regreqTimelimitLab(Signal* signal, UintR callTime)
+{
+ if (cwaitContinuebPhase1 == ZFALSE) {
+ if (cwaitContinuebPhase2 == ZFALSE) {
+ jam();
+ return;
+ } else {
+ jam();
+ if (callTime < 10) {
+ /*-------------------------------------------------------------*/
+ // We experienced a time-out of inclusion. Give it another few
+ // seconds before crashing.
+ /*-------------------------------------------------------------*/
+ signal->theData[0] = ZREGREQ_TIMELIMIT;
+ signal->theData[1] = cstartNo;
+ signal->theData[2] = callTime + 1;
+ sendSignalWithDelay(QMGR_REF, GSN_CONTINUEB, signal, 3000, 3);
+ return;
+ }//if
+ /*-------------------------------------------------------------*/
+ /* WE HAVE COME HERE BECAUSE THE INCLUSION SUFFERED FROM */
+ /* TIME OUT. WE CRASH AND RESTART. */
+ /*-------------------------------------------------------------*/
+ systemErrorLab(signal);
+ return;
+ }//if
+ } else {
+ jam();
+ cwaitContinuebPhase1 = ZFALSE;
+ }//if
+
+ cmInfoconf010Lab(signal);
+}//Qmgr::regreqTimelimitLab()
+
+/**---------------------------------------------------------------------------
+ * The new node will take care of giving information about own node and ask
+ * all other nodes for nodeinfo. The new node will use CM_NODEINFOREQ for
+ * that purpose. When the setup of connections to all running, the president
+ * will send a commit to all running nodes + the new node
+ * INPUT: NODE_PTR1, must be set as ZNIL if we don't enter CONNECT_NODES)
+ * from signal CM_NODEINFOCONF.
+ *---------------------------------------------------------------------------*/
+/*******************************/
+/* CM_NODEINFOCONF */
+/*******************************/
+void Qmgr::execCM_NODEINFOCONF(Signal* signal)
+{
+ NodeRecPtr replyNodePtr;
+ NodeRecPtr nodePtr;
+ jamEntry();
+
+ CmNodeInfoConf * const conf = (CmNodeInfoConf*)signal->getDataPtr();
+
+ replyNodePtr.i = conf->nodeId;
+ ptrCheckGuard(replyNodePtr, MAX_NDB_NODES, nodeRec);
+ replyNodePtr.p->ndynamicId = conf->dynamicId;
+ setNodeInfo(replyNodePtr.i).m_version = conf->version;
+ replyNodePtr.p->phase = ZRUNNING;
+
+ /**
+ * A node in the cluster has replied nodeinfo about himself.
+ * He is already running in the cluster.
+ */
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZWAITING) {
+ if (nodePtr.i != getOwnNodeId()) {
+ jam();
+ return;
+ }//if
+ }//if
+ }//for
+
+ /**********************************************<*/
+ /* Send an ack. back to the president. */
+ /* CM_ACKADD */
+ /* The new node has been registered by all */
+ /* running nodes and has stored nodeinfo about */
+ /* all running nodes. The new node has to wait */
+ /* for CM_ADD (commit) from president to become */
+ /* a running node in the cluster. */
+ /**********************************************<*/
+ CmAckAdd * const cmAckAdd = (CmAckAdd*)signal->getDataPtrSend();
+ cmAckAdd->requestType = CmAdd::Prepare;
+ cmAckAdd->senderNodeId = getOwnNodeId();
+ cmAckAdd->startingNodeId = getOwnNodeId();
+ sendSignal(cpdistref, GSN_CM_ACKADD, signal, CmAckAdd::SignalLength, JBA);
+ return;
+}//Qmgr::execCM_NODEINFOCONF()
+
+/**---------------------------------------------------------------------------
+ * A new node sends nodeinfo about himself. The new node asks for
+ * corresponding nodeinfo back in the CM_NODEINFOCONF.
+ *---------------------------------------------------------------------------*/
+/*******************************/
+/* CM_NODEINFOREQ */
+/*******************************/
+void Qmgr::execCM_NODEINFOREQ(Signal* signal)
+{
+ NodeRecPtr addNodePtr;
+ NodeRecPtr myNodePtr;
+ jamEntry();
+
+ CmNodeInfoReq * const req = (CmNodeInfoReq*)signal->getDataPtr();
+ addNodePtr.i = req->nodeId;
+ ptrCheckGuard(addNodePtr, MAX_NDB_NODES, nodeRec);
+ addNodePtr.p->ndynamicId = req->dynamicId;
+ setNodeInfo(addNodePtr.i).m_version = req->version;
+
+ const BlockReference Tblockref = signal->getSendersBlockRef();
+
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+ if (myNodePtr.p->phase == ZRUNNING) {
+ if (addNodePtr.p->phase == ZWAITING) {
+ jam();
+ /* President have prepared us */
+ /****************************<*/
+ /*< CM_NODEINFOCONF <*/
+ /****************************<*/
+ CmNodeInfoConf * const conf = (CmNodeInfoConf*)signal->getDataPtrSend();
+ conf->nodeId = getOwnNodeId();
+ conf->dynamicId = myNodePtr.p->ndynamicId;
+ conf->version = getNodeInfo(getOwnNodeId()).m_version;
+ sendSignal(Tblockref, GSN_CM_NODEINFOCONF, signal,
+ CmNodeInfoConf::SignalLength, JBB);
+ /****************************************/
+ /* Send an ack. back to the president */
+ /* CM_ACKADD */
+ /****************************************/
+ CmAckAdd * const cmAckAdd = (CmAckAdd*)signal->getDataPtrSend();
+ cmAckAdd->requestType = CmAdd::Prepare;
+ cmAckAdd->senderNodeId = getOwnNodeId();
+ cmAckAdd->startingNodeId = addNodePtr.i;
+ sendSignal(cpdistref, GSN_CM_ACKADD, signal, CmAckAdd::SignalLength, JBA);
+ } else {
+ jam();
+ addNodePtr.p->phase = ZWAIT_PRESIDENT;
+ }//if
+ } else {
+ jam();
+ /****************************<*/
+ /*< CM_NODEINFOREF <*/
+ /****************************<*/
+ signal->theData[0] = myNodePtr.p->blockRef;
+ signal->theData[1] = myNodePtr.i;
+ signal->theData[2] = ZNOT_RUNNING;
+ sendSignal(Tblockref, GSN_CM_NODEINFOREF, signal, 3, JBB);
+ }//if
+ return;
+}//Qmgr::execCM_NODEINFOREQ()
+
+/*
+4.4.11 CM_ADD */
+/**--------------------------------------------------------------------------
+ * Prepare a running node to add a new node to the cluster. The running node
+ * will change phase of the new node fron ZINIT to ZWAITING. The running node
+ * will also mark that we have received a prepare. When the new node has sent
+ * us nodeinfo we can send an acknowledgement back to the president. When all
+ * running nodes has acknowledged the new node, the president will send a
+ * commit and we can change phase of the new node to ZRUNNING. The president
+ * will also send CM_ADD to himself.
+ *---------------------------------------------------------------------------*/
+/*******************************/
+/* CM_ADD */
+/*******************************/
+void Qmgr::execCM_ADD(Signal* signal)
+{
+ NodeRecPtr addNodePtr;
+ NodeRecPtr nodePtr;
+ NodeRecPtr myNodePtr;
+ jamEntry();
+
+ CmAdd * const cmAdd = (CmAdd*)signal->getDataPtr();
+ const CmAdd::RequestType type = (CmAdd::RequestType)cmAdd->requestType;
+ addNodePtr.i = cmAdd->startingNodeId;
+ //const Uint32 startingVersion = cmAdd->startingVersion;
+ ptrCheckGuard(addNodePtr, MAX_NDB_NODES, nodeRec);
+
+ if(addNodePtr.p->phase == ZFAIL_CLOSING){
+ jam();
+#ifdef VM_TRACE
+ ndbout_c("Enabling communication to CM_ADD node state=%d",
+ addNodePtr.p->phase);
+#endif
+ addNodePtr.p->failState = NORMAL;
+ signal->theData[0] = 0;
+ signal->theData[1] = addNodePtr.i;
+ sendSignal(CMVMI_REF, GSN_OPEN_COMREQ, signal, 2, JBA);
+ }
+
+ switch (type) {
+ case CmAdd::Prepare:
+ jam();
+ if (addNodePtr.i != getOwnNodeId()) {
+ jam();
+ if (addNodePtr.p->phase == ZWAIT_PRESIDENT) {
+ jam();
+ /****************************<*/
+ /*< CM_NODEINFOCONF <*/
+ /****************************<*/
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+
+ CmNodeInfoConf * const conf = (CmNodeInfoConf*)signal->getDataPtrSend();
+ conf->nodeId = getOwnNodeId();
+ conf->dynamicId = myNodePtr.p->ndynamicId;
+ conf->version = getNodeInfo(getOwnNodeId()).m_version;
+ sendSignal(addNodePtr.p->blockRef, GSN_CM_NODEINFOCONF, signal,
+ CmNodeInfoConf::SignalLength, JBB);
+ /****************************<*/
+ /* Send an ack. back to the president */
+ /*< CM_ACKADD <*/
+ /****************************<*/
+ CmAckAdd * const cmAckAdd = (CmAckAdd*)signal->getDataPtrSend();
+ cmAckAdd->requestType = CmAdd::Prepare;
+ cmAckAdd->senderNodeId = getOwnNodeId();
+ cmAckAdd->startingNodeId = addNodePtr.i;
+ sendSignal(cpdistref, GSN_CM_ACKADD, signal, CmAckAdd::SignalLength, JBA);
+ }//if
+ // -----------------------------------------
+ /* Wait for the new node's CM_NODEINFOREQ.*/
+ // -----------------------------------------
+ addNodePtr.p->phase = ZWAITING;
+ }//if
+ break;
+ case CmAdd::AddCommit:{
+ jam();
+ addNodePtr.p->phase = ZRUNNING;
+ addNodePtr.p->alarmCount = 0;
+ findNeighbours(signal);
+ /**-----------------------------------------------------------------------
+ * SEND A HEARTBEAT IMMEDIATELY TO DECREASE THE RISK THAT WE MISS EARLY
+ * HEARTBEATS.
+ *-----------------------------------------------------------------------*/
+ sendHeartbeat(signal);
+ /*-----------------------------------------------------------------------*/
+ /* ENABLE COMMUNICATION WITH ALL BLOCKS WITH THE NEWLY ADDED NODE. */
+ /*-----------------------------------------------------------------------*/
+ signal->theData[0] = addNodePtr.i;
+ sendSignal(CMVMI_REF, GSN_ENABLE_COMORD, signal, 1, JBA);
+ /****************************<*/
+ /*< CM_ACKADD <*/
+ /****************************<*/
+ CmAckAdd * const cmAckAdd = (CmAckAdd*)signal->getDataPtrSend();
+ cmAckAdd->requestType = CmAdd::AddCommit;
+ cmAckAdd->senderNodeId = getOwnNodeId();
+ cmAckAdd->startingNodeId = addNodePtr.i;
+ sendSignal(cpdistref, GSN_CM_ACKADD, signal, CmAckAdd::SignalLength, JBA);
+ break;
+ }
+ case CmAdd::CommitNew:{
+ jam();
+ /*-----------------------------------------------------------------------*/
+ /* WE HAVE BEEN INCLUDED IN THE CLUSTER WE CAN START BEING PART OF THE
+ * HEARTBEAT PROTOCOL AND WE WILL ALSO ENABLE COMMUNICATION WITH ALL
+ * NODES IN THE CLUSTER.
+ *-----------------------------------------------------------------------*/
+ addNodePtr.p->phase = ZRUNNING;
+ addNodePtr.p->alarmCount = 0;
+ findNeighbours(signal);
+ /**-----------------------------------------------------------------------
+ * SEND A HEARTBEAT IMMEDIATELY TO DECREASE THE RISK THAT WE MISS EARLY
+ * HEARTBEATS.
+ *-----------------------------------------------------------------------*/
+ sendHeartbeat(signal);
+ cwaitContinuebPhase2 = ZFALSE;
+ /**-----------------------------------------------------------------------
+ * ENABLE COMMUNICATION WITH ALL BLOCKS IN THE CURRENT CLUSTER AND SET
+ * THE NODES IN THE CLUSTER TO BE RUNNING.
+ *-----------------------------------------------------------------------*/
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if ((nodePtr.p->phase == ZRUNNING) &&
+ (nodePtr.i != getOwnNodeId())) {
+ /*-------------------------------------------------------------------*/
+ // Enable full communication to all other nodes. Not really necessary
+ // to open communication to ourself.
+ /*-------------------------------------------------------------------*/
+ jam();
+ signal->theData[0] = nodePtr.i;
+ sendSignal(CMVMI_REF, GSN_ENABLE_COMORD, signal, 1, JBA);
+ }//if
+ }//for
+
+ /****************************<*/
+ /*< CM_ACKADD <*/
+ /****************************<*/
+ CmAckAdd * const cmAckAdd = (CmAckAdd*)signal->getDataPtrSend();
+ cmAckAdd->requestType = CmAdd::CommitNew;
+ cmAckAdd->senderNodeId = getOwnNodeId();
+ cmAckAdd->startingNodeId = addNodePtr.i;
+ sendSignal(cpdistref, GSN_CM_ACKADD, signal, CmAckAdd::SignalLength, JBA);
+
+#if 1
+ /**********************************************<*/
+ /* Allow the CLUSTER CONTROL to send STTORRY */
+ /* so we can receive CM_REG from applications. */
+ /**********************************************<*/
+ signal->theData[0] = 0;
+ sendSignal(CMVMI_REF, GSN_CM_RUN, signal, 1, JBB);
+#endif
+
+ /**
+ * Start timer handling
+ */
+ signal->theData[0] = ZTIMER_HANDLING;
+ sendSignal(QMGR_REF, GSN_CONTINUEB, signal, 10, JBB);
+ }
+ break;
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }//switch
+ return;
+}//Qmgr::execCM_ADD()
+
+/* 4.10.7 CM_ACKADD - PRESIDENT IS RECEIVER - */
+/*---------------------------------------------------------------------------*/
+/* Entry point for an ack add signal.
+ * The TTYPE defines if it is a prepare or a commit. */
+/*---------------------------------------------------------------------------*/
+void Qmgr::execCM_ACKADD(Signal* signal)
+{
+ NodeRecPtr addNodePtr;
+ NodeRecPtr nodePtr;
+ NodeRecPtr senderNodePtr;
+ jamEntry();
+
+ CmAckAdd * const cmAckAdd = (CmAckAdd*)signal->getDataPtr();
+ const CmAdd::RequestType type = (CmAdd::RequestType)cmAckAdd->requestType;
+ addNodePtr.i = cmAckAdd->startingNodeId;
+ senderNodePtr.i = cmAckAdd->senderNodeId;
+ if (cpresident != getOwnNodeId()) {
+ jam();
+ /*-----------------------------------------------------------------------*/
+ /* IF WE ARE NOT PRESIDENT THEN WE SHOULD NOT RECEIVE THIS MESSAGE. */
+ /*------------------------------------------------------------_----------*/
+ return;
+ }//if
+ if (cpresidentBusy != ZTRUE) {
+ jam();
+ /**----------------------------------------------------------------------
+ * WE ARE PRESIDENT BUT WE ARE NOT BUSY ADDING ANY NODE. THUS WE MUST
+ * HAVE STOPPED THIS ADDING OF THIS NODE.
+ *----------------------------------------------------------------------*/
+ return;
+ }//if
+ if (addNodePtr.i != cstartNode) {
+ jam();
+ /*----------------------------------------------------------------------*/
+ /* THIS IS NOT THE STARTING NODE. WE ARE ACTIVE NOW WITH ANOTHER START. */
+ /*----------------------------------------------------------------------*/
+ return;
+ }//if
+ switch (type) {
+ case CmAdd::Prepare:{
+ jam();
+ ptrCheckGuard(senderNodePtr, MAX_NDB_NODES, nodeRec);
+ senderNodePtr.p->sendCmAddPrepStatus = Q_NOT_ACTIVE;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ /* Check if all prepare are acknowledged*/
+ if (nodePtr.p->sendCmAddPrepStatus == Q_ACTIVE) {
+ jam();
+ return; /* Wait for more acknowledge's */
+ }//if
+ }//for
+ /*----------------------------------------------------------------------*/
+ /* ALL RUNNING NODES HAVE PREPARED THE INCLUSION OF THIS NEW NODE. */
+ /*----------------------------------------------------------------------*/
+ CmAdd * const cmAdd = (CmAdd*)signal->getDataPtrSend();
+ cmAdd->requestType = CmAdd::AddCommit;
+ cmAdd->startingNodeId = addNodePtr.i;
+ cmAdd->startingVersion = getNodeInfo(addNodePtr.i).m_version;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ sendSignal(nodePtr.p->blockRef, GSN_CM_ADD, signal,
+ CmAdd::SignalLength, JBA);
+ nodePtr.p->sendCmAddCommitStatus = Q_ACTIVE;
+ }//if
+ }//for
+ return;
+ break;
+ }
+ case CmAdd::AddCommit:{
+ jam();
+ ptrCheckGuard(senderNodePtr, MAX_NDB_NODES, nodeRec);
+ senderNodePtr.p->sendCmAddCommitStatus = Q_NOT_ACTIVE;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ /* Check to see if we need to wait for */
+ if (nodePtr.p->sendCmAddCommitStatus == Q_ACTIVE) {
+ jam();
+ /* any more ack. commit add. */
+ return; /* Exit and continue waiting. */
+ }//if
+ }//for
+ /****************************************/
+ /* Send commit to the new node so he */
+ /* will change PHASE into ZRUNNING */
+ /****************************************/
+ CmAdd * const cmAdd = (CmAdd*)signal->getDataPtrSend();
+ cmAdd->requestType = CmAdd::CommitNew;
+ cmAdd->startingNodeId = addNodePtr.i;
+ cmAdd->startingVersion = getNodeInfo(addNodePtr.i).m_version;
+
+ ptrCheckGuard(addNodePtr, MAX_NDB_NODES, nodeRec);
+ sendSignal(addNodePtr.p->blockRef, GSN_CM_ADD, signal,
+ CmAdd::SignalLength, JBA);
+ break;
+ }
+ case CmAdd::CommitNew:
+ jam();
+ /*----------------------------------------------------------------------*/
+ /* Increment the amount of nodes in the cluster in waiting mode. */
+ /* President now ready for more CM_REGREQ */
+ /*----------------------------------------------------------------------*/
+ cclustersize = cclustersize + 1;
+ /**
+ * Tell arbitration about new node.
+ */
+ handleArbitNdbAdd(signal, addNodePtr.i);
+ cpresidentBusy = ZFALSE;
+ break;
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }//switch
+ return;
+}//Qmgr::execCM_ACKADD()
+
+/**-------------------------------------------------------------------------
+ * WE HAVE BEEN INCLUDED INTO THE CLUSTER. IT IS NOW TIME TO CALCULATE WHICH
+ * ARE OUR LEFT AND RIGHT NEIGHBOURS FOR THE HEARTBEAT PROTOCOL.
+ *--------------------------------------------------------------------------*/
+void Qmgr::findNeighbours(Signal* signal)
+{
+ UintR toldLeftNeighbour;
+ UintR tfnLeftFound;
+ UintR tfnMaxFound;
+ UintR tfnMinFound;
+ UintR tfnRightFound;
+ NodeRecPtr fnNodePtr;
+ NodeRecPtr fnOwnNodePtr;
+
+ toldLeftNeighbour = cneighbourl;
+ tfnLeftFound = 0;
+ tfnMaxFound = 0;
+ tfnMinFound = (UintR)-1;
+ tfnRightFound = (UintR)-1;
+ fnOwnNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(fnOwnNodePtr, MAX_NDB_NODES, nodeRec);
+ for (fnNodePtr.i = 1; fnNodePtr.i < MAX_NDB_NODES; fnNodePtr.i++) {
+ jam();
+ ptrAss(fnNodePtr, nodeRec);
+ if (fnNodePtr.i != fnOwnNodePtr.i) {
+ if (fnNodePtr.p->phase == ZRUNNING) {
+ if (tfnMinFound > fnNodePtr.p->ndynamicId) {
+ jam();
+ tfnMinFound = fnNodePtr.p->ndynamicId;
+ }//if
+ if (tfnMaxFound < fnNodePtr.p->ndynamicId) {
+ jam();
+ tfnMaxFound = fnNodePtr.p->ndynamicId;
+ }//if
+ if (fnOwnNodePtr.p->ndynamicId > fnNodePtr.p->ndynamicId) {
+ jam();
+ if (fnNodePtr.p->ndynamicId > tfnLeftFound) {
+ jam();
+ tfnLeftFound = fnNodePtr.p->ndynamicId;
+ }//if
+ } else {
+ jam();
+ if (fnNodePtr.p->ndynamicId < tfnRightFound) {
+ jam();
+ tfnRightFound = fnNodePtr.p->ndynamicId;
+ }//if
+ }//if
+ }//if
+ }//if
+ }//for
+ if (tfnLeftFound == 0) {
+ if (tfnMinFound == (UintR)-1) {
+ jam();
+ cneighbourl = ZNIL;
+ } else {
+ jam();
+ cneighbourl = translateDynamicIdToNodeId(signal, tfnMaxFound);
+ }//if
+ } else {
+ jam();
+ cneighbourl = translateDynamicIdToNodeId(signal, tfnLeftFound);
+ }//if
+ if (tfnRightFound == (UintR)-1) {
+ if (tfnMaxFound == 0) {
+ jam();
+ cneighbourh = ZNIL;
+ } else {
+ jam();
+ cneighbourh = translateDynamicIdToNodeId(signal, tfnMinFound);
+ }//if
+ } else {
+ jam();
+ cneighbourh = translateDynamicIdToNodeId(signal, tfnRightFound);
+ }//if
+ if (toldLeftNeighbour != cneighbourl) {
+ jam();
+ if (cneighbourl != ZNIL) {
+ jam();
+ /**-------------------------------------------------------------------*/
+ /* WE ARE SUPERVISING A NEW LEFT NEIGHBOUR. WE START WITH ALARM COUNT
+ * EQUAL TO ZERO.
+ *---------------------------------------------------------------------*/
+ fnNodePtr.i = cneighbourl;
+ ptrCheckGuard(fnNodePtr, MAX_NDB_NODES, nodeRec);
+ fnNodePtr.p->alarmCount = 0;
+ }//if
+ }//if
+
+ signal->theData[0] = EventReport::FIND_NEIGHBOURS;
+ signal->theData[1] = getOwnNodeId();
+ signal->theData[2] = cneighbourl;
+ signal->theData[3] = cneighbourh;
+ signal->theData[4] = fnOwnNodePtr.p->ndynamicId;
+ UintR Tlen = 5;
+ for (fnNodePtr.i = 1; fnNodePtr.i < MAX_NDB_NODES; fnNodePtr.i++) {
+ jam();
+ ptrAss(fnNodePtr, nodeRec);
+ if (fnNodePtr.i != fnOwnNodePtr.i) {
+ if (fnNodePtr.p->phase == ZRUNNING) {
+ jam();
+ signal->theData[Tlen] = fnNodePtr.i;
+ signal->theData[Tlen + 1] = fnNodePtr.p->ndynamicId;
+ if (Tlen < 25) {
+ /*----------------------------------------------------------------*/
+ // This code can only report 11 nodes.
+ // We need to update this when increasing the number of nodes
+ // supported.
+ /*-----------------------------------------------------------------*/
+ Tlen += 2;
+ }
+ }//if
+ }//if
+ }//for
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, Tlen, JBB);
+}//Qmgr::findNeighbours()
+
+/*
+4.10.7 INIT_DATA */
+/*---------------------------------------------------------------------------*/
+/*---------------------------------------------------------------------------*/
+void Qmgr::initData(Signal* signal)
+{
+ RegAppPtr localRegAppptr;
+
+ for (localRegAppptr.i = 0;
+ localRegAppptr.i < NO_REG_APP; localRegAppptr.i++) {
+ ptrAss(localRegAppptr, regApp);
+ localRegAppptr.p->version = 0;
+ localRegAppptr.p->blockref = 0;
+ memset(localRegAppptr.p->name, 0, sizeof(localRegAppptr.p->name));
+ localRegAppptr.p->activity = ZREMOVE;
+ localRegAppptr.p->noofapps = 0;
+ localRegAppptr.p->noofpending = 0;
+ localRegAppptr.p->m_runNodes.clear();
+ }//for
+
+ NodeRecPtr nodePtr;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRec);
+ nodePtr.p->ndynamicId = 0;
+ /* Subr NEXT_DYNAMIC_ID will use this to find */
+ /* a unique higher value than any of these */
+
+ /* Not in config file */
+ nodePtr.p->phase = ZBLOCKED;
+ nodePtr.p->alarmCount = 0;
+ nodePtr.p->sendPrepFailReqStatus = Q_NOT_ACTIVE;
+ nodePtr.p->sendCommitFailReqStatus = Q_NOT_ACTIVE;
+ nodePtr.p->sendCmAddPrepStatus = Q_NOT_ACTIVE;
+ nodePtr.p->sendCmAddCommitStatus = Q_NOT_ACTIVE;
+ nodePtr.p->sendPresToStatus = Q_NOT_ACTIVE;
+ nodePtr.p->m_connected = false;
+ nodePtr.p->failState = NORMAL;
+ nodePtr.p->rcv[0] = 0;
+ nodePtr.p->rcv[1] = 0;
+ }//for
+ ccm_infoconfCounter = 0;
+ cfailureNr = 1;
+ ccommitFailureNr = 1;
+ cprepareFailureNr = 1;
+ cnoFailedNodes = 0;
+ cnoPrepFailedNodes = 0;
+ cwaitContinuebPhase1 = ZFALSE;
+ cwaitContinuebPhase2 = ZFALSE;
+ cstartNo = 0;
+ cpresidentBusy = ZFALSE;
+ cacceptRegreq = ZTRUE;
+ creadyDistCom = ZFALSE;
+ cpresident = ZNIL;
+ cpresidentCandidate = ZNIL;
+ cpdistref = 0;
+ cneighbourh = ZNIL;
+ cneighbourl = ZNIL;
+ cdelayRegreq = ZDELAY_REGREQ;
+ cactivateApiCheck = 0;
+ ctoStatus = Q_NOT_ACTIVE;
+
+ interface_check_timer.setDelay(1000);
+ interface_check_timer.reset();
+ clatestTransactionCheck = 0;
+
+ cLqhTimeSignalCount = 0;
+
+ // catch-all for missing initializations
+ memset(&arbitRec, 0, sizeof(arbitRec));
+}//Qmgr::initData()
+
+/*
+4.10.7 PREPARE_ADD */
+/**--------------------------------------------------------------------------
+ * President sends CM_ADD to prepare all running nodes to add a new node.
+ * Even the president node will get a CM_ADD (prepare).
+ * The new node will make REQs to all running nodes after it has received the
+ * CM_REGCONF. The president will just coordinate the adding of new nodes.
+ * The CM_ADD (prepare) is sent to the cluster before the CM_REGCONF signal
+ * to the new node.
+ *
+ * At the same time we will store all running nodes in CNODEMASK,
+ * which will be sent to the new node
+ * Scan the NODE_REC for all running nodes and create a nodemask where
+ * each bit represents a node.
+ * --------------------------------------------------------------------------*/
+void Qmgr::prepareAdd(Signal* signal, Uint16 anAddedNode)
+{
+ NodeRecPtr nodePtr;
+ NdbNodeBitmask::clear(cnodemask);
+
+ CmAdd * const cmAdd = (CmAdd*)signal->getDataPtrSend();
+ cmAdd->requestType = CmAdd::Prepare;
+ cmAdd->startingNodeId = anAddedNode;
+ cmAdd->startingVersion = getNodeInfo(anAddedNode).m_version;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ /* We found a node to prepare. */
+ NdbNodeBitmask::set(cnodemask, nodePtr.i);
+ sendSignal(nodePtr.p->blockRef, GSN_CM_ADD, signal, CmAdd::SignalLength, JBA);
+ nodePtr.p->sendCmAddPrepStatus = Q_ACTIVE;
+ }//if
+ }//for
+
+ NodeRecPtr addNodePtr;
+ addNodePtr.i = anAddedNode;
+ ptrCheckGuard(addNodePtr, MAX_NDB_NODES, nodeRec);
+ /*****************************<
+ * We send to the node to be added a CM_ADD as well.
+ * We want him to send an ack when he has
+ * received all CM_NODEINFOCONF.
+ */
+ sendSignal(addNodePtr.p->blockRef, GSN_CM_ADD, signal, CmAdd::SignalLength, JBA);
+ addNodePtr.p->sendCmAddPrepStatus = Q_ACTIVE;
+}//Qmgr::prepareAdd()
+
+/**---------------------------------------------------------------------------
+ * HERE WE RECEIVE THE JOB TABLE SIGNAL EVERY 10 MILLISECONDS.
+ * WE WILL USE THIS TO CHECK IF IT IS TIME TO CHECK THE NEIGHBOUR NODE.
+ * WE WILL ALSO SEND A SIGNAL TO BLOCKS THAT NEED A TIME SIGNAL AND
+ * DO NOT WANT TO USE JOB TABLE SIGNALS.
+ *---------------------------------------------------------------------------*/
+void Qmgr::timerHandlingLab(Signal* signal)
+{
+ NDB_TICKS TcurrentTime = NdbTick_CurrentMillisecond();
+ NodeRecPtr myNodePtr;
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+
+ if (myNodePtr.p->phase == ZRUNNING) {
+ jam();
+ /**---------------------------------------------------------------------
+ * WE ARE ONLY PART OF HEARTBEAT CLUSTER IF WE ARE UP AND RUNNING.
+ *---------------------------------------------------------------------*/
+ if (hb_send_timer.check(TcurrentTime)) {
+ jam();
+ sendHeartbeat(signal);
+ hb_send_timer.reset();
+ }
+ if (hb_check_timer.check(TcurrentTime)) {
+ jam();
+ checkHeartbeat(signal);
+ hb_check_timer.reset();
+ }
+ }
+
+ if (interface_check_timer.check(TcurrentTime)) {
+ jam();
+ interface_check_timer.reset();
+ checkStartInterface(signal);
+ }
+
+ if (cactivateApiCheck != 0) {
+ jam();
+ if (hb_api_timer.check(TcurrentTime)) {
+ jam();
+ hb_api_timer.reset();
+ apiHbHandlingLab(signal);
+ }//if
+ if (clatestTransactionCheck == 0) {
+ //-------------------------------------------------------------
+ // Initialise the Transaction check timer.
+ //-------------------------------------------------------------
+ clatestTransactionCheck = TcurrentTime;
+ }//if
+ int counter = 0;
+ while (TcurrentTime > ((NDB_TICKS)10 + clatestTransactionCheck)) {
+ jam();
+ clatestTransactionCheck += (NDB_TICKS)10;
+ sendSignal(DBTC_REF, GSN_TIME_SIGNAL, signal, 1, JBB);
+ cLqhTimeSignalCount++;
+ if (cLqhTimeSignalCount >= 100) {
+ cLqhTimeSignalCount = 0;
+ sendSignal(DBLQH_REF, GSN_TIME_SIGNAL, signal, 1, JBB);
+ }//if
+ counter++;
+ if (counter > 1) {
+ jam();
+ break;
+ } else {
+ ;
+ }//if
+ }//while
+ }//if
+
+ //--------------------------------------------------
+ // Resend this signal with 10 milliseconds delay.
+ //--------------------------------------------------
+ signal->theData[0] = ZTIMER_HANDLING;
+ sendSignalWithDelay(QMGR_REF, GSN_CONTINUEB, signal, 10, 1);
+ return;
+}//Qmgr::timerHandlingLab()
+
+/*---------------------------------------------------------------------------*/
+/* THIS MODULE HANDLES THE SENDING AND RECEIVING OF HEARTBEATS. */
+/*---------------------------------------------------------------------------*/
+void Qmgr::sendHeartbeat(Signal* signal)
+{
+ NodeRecPtr localNodePtr;
+ localNodePtr.i = cneighbourh;
+ if (localNodePtr.i == ZNIL) {
+ jam();
+ /**---------------------------------------------------------------------
+ * THERE ARE NO NEIGHBOURS. THIS IS POSSIBLE IF WE ARE THE ONLY NODE IN
+ * THE CLUSTER.IN THIS CASE WE DO NOT NEED TO SEND ANY HEARTBEAT SIGNALS.
+ *-----------------------------------------------------------------------*/
+ return;
+ }//if
+ ptrCheckGuard(localNodePtr, MAX_NDB_NODES, nodeRec);
+ signal->theData[0] = getOwnNodeId();
+
+ sendSignal(localNodePtr.p->blockRef, GSN_CM_HEARTBEAT, signal, 1, JBA);
+#ifdef VM_TRACE
+ signal->theData[0] = EventReport::SentHeartbeat;
+ signal->theData[1] = localNodePtr.i;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+#endif
+}//Qmgr::sendHeartbeat()
+
+void Qmgr::checkHeartbeat(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+
+ nodePtr.i = cneighbourl;
+ if (nodePtr.i == ZNIL) {
+ jam();
+ /**---------------------------------------------------------------------
+ * THERE ARE NO NEIGHBOURS. THIS IS POSSIBLE IF WE ARE THE ONLY NODE IN
+ * THE CLUSTER. IN THIS CASE WE DO NOT NEED TO CHECK ANY HEARTBEATS.
+ *-----------------------------------------------------------------------*/
+ return;
+ }//if
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+
+ nodePtr.p->alarmCount ++;
+ ndbrequire(nodePtr.p->phase == ZRUNNING);
+ ndbrequire(getNodeInfo(nodePtr.i).m_type == NodeInfo::DB);
+
+ if(nodePtr.p->alarmCount > 2){
+ signal->theData[0] = EventReport::MissedHeartbeat;
+ signal->theData[1] = nodePtr.i;
+ signal->theData[2] = nodePtr.p->alarmCount - 1;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+ }
+
+ if (nodePtr.p->alarmCount > 4) {
+ jam();
+ /**----------------------------------------------------------------------
+ * OUR LEFT NEIGHBOUR HAVE KEPT QUIET FOR THREE CONSECUTIVE HEARTBEAT
+ * PERIODS. THUS WE DECLARE HIM DOWN.
+ *----------------------------------------------------------------------*/
+ signal->theData[0] = EventReport::DeadDueToHeartbeat;
+ signal->theData[1] = nodePtr.i;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ failReportLab(signal, nodePtr.i, FailRep::ZHEARTBEAT_FAILURE);
+ return;
+ }//if
+}//Qmgr::checkHeartbeat()
+
+void Qmgr::apiHbHandlingLab(Signal* signal)
+{
+ NodeRecPtr TnodePtr;
+
+ for (TnodePtr.i = 1; TnodePtr.i < MAX_NODES; TnodePtr.i++) {
+ ptrAss(TnodePtr, nodeRec);
+
+ const NodeInfo::NodeType type = getNodeInfo(TnodePtr.i).getType();
+ if(type == NodeInfo::DB)
+ continue;
+
+ if(type == NodeInfo::INVALID)
+ continue;
+
+ if (TnodePtr.p->m_connected && TnodePtr.p->phase != ZAPI_INACTIVE){
+ jam();
+ TnodePtr.p->alarmCount ++;
+
+ if(TnodePtr.p->alarmCount > 2){
+ signal->theData[0] = EventReport::MissedHeartbeat;
+ signal->theData[1] = TnodePtr.i;
+ signal->theData[2] = TnodePtr.p->alarmCount - 1;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 3, JBB);
+ }
+
+ if (TnodePtr.p->alarmCount > 4) {
+ jam();
+ /*------------------------------------------------------------------*/
+ /* THE API NODE HAS NOT SENT ANY HEARTBEAT FOR THREE SECONDS.
+ * WE WILL DISCONNECT FROM IT NOW.
+ *------------------------------------------------------------------*/
+ /*------------------------------------------------------------------*/
+ /* We call node_failed to release all connections for this api node */
+ /*------------------------------------------------------------------*/
+ signal->theData[0] = EventReport::DeadDueToHeartbeat;
+ signal->theData[1] = TnodePtr.i;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 2, JBB);
+
+ node_failed(signal, TnodePtr.i);
+ }//if
+ }//if
+ }//for
+ return;
+}//Qmgr::apiHbHandlingLab()
+
+void Qmgr::checkStartInterface(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ /*------------------------------------------------------------------------*/
+ // This method is called once per second. After a disconnect we wait at
+ // least three seconds before allowing new connects. We will also ensure
+ // that handling of the failure is completed before we allow new connections.
+ /*------------------------------------------------------------------------*/
+ for (nodePtr.i = 1; nodePtr.i < MAX_NODES; nodePtr.i++) {
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZFAIL_CLOSING) {
+ jam();
+ nodePtr.p->alarmCount = nodePtr.p->alarmCount + 1;
+ if (nodePtr.p->m_connected) {
+ jam();
+ /*-------------------------------------------------------------------*/
+ // We need to ensure that the connection is not restored until it has
+ // been disconnected for at least three seconds.
+ /*-------------------------------------------------------------------*/
+ nodePtr.p->alarmCount = 0;
+ }//if
+ if ((nodePtr.p->alarmCount > 3) && (nodePtr.p->failState == NORMAL)) {
+ /**------------------------------------------------------------------
+ * WE HAVE DISCONNECTED THREE SECONDS AGO. WE ARE NOW READY TO
+ * CONNECT AGAIN AND ACCEPT NEW REGISTRATIONS FROM THIS NODE.
+ * WE WILL NOT ALLOW CONNECTIONS OF API NODES UNTIL API FAIL HANDLING
+ * IS COMPLETE.
+ *-------------------------------------------------------------------*/
+ nodePtr.p->failState = NORMAL;
+ if (getNodeInfo(nodePtr.i).m_type != NodeInfo::DB){
+ jam();
+ nodePtr.p->phase = ZBLOCKED;
+ } else {
+ jam();
+ nodePtr.p->phase = ZINIT;
+ }//if
+ nodePtr.p->alarmCount = 0;
+ signal->theData[0] = 0;
+ signal->theData[1] = nodePtr.i;
+ sendSignal(CMVMI_REF, GSN_OPEN_COMREQ, signal, 2, JBA);
+ } else {
+ if(((nodePtr.p->alarmCount + 1) % 60) == 0){
+ char buf[100];
+ snprintf(buf, sizeof(buf),
+ "Failure handling of node %d has not completed in %d min."
+ " - state = %d",
+ nodePtr.i,
+ (nodePtr.p->alarmCount + 1)/60,
+ nodePtr.p->failState);
+ warningEvent(buf);
+ }
+ }
+ }//if
+ }//for
+ return;
+}//Qmgr::checkStartInterface()
+
+/**-------------------------------------------------------------------------
+ * This method is called when a DISCONNECT_REP signal arrived which means that
+ * the API node is gone and we want to release resources in TC/DICT blocks.
+ *---------------------------------------------------------------------------*/
+void Qmgr::sendApiFailReq(Signal* signal, Uint16 failedNodeNo)
+{
+ NodeRecPtr failedNodePtr;
+
+ jamEntry();
+ failedNodePtr.i = failedNodeNo;
+ signal->theData[0] = failedNodePtr.i;
+ signal->theData[1] = QMGR_REF;
+
+ ptrCheckGuard(failedNodePtr, MAX_NODES, nodeRec);
+
+ ndbrequire(failedNodePtr.p->failState == NORMAL);
+
+ failedNodePtr.p->failState = WAITING_FOR_FAILCONF1;
+ sendSignal(DBTC_REF, GSN_API_FAILREQ, signal, 2, JBA);
+ sendSignal(DBDICT_REF, GSN_API_FAILREQ, signal, 2, JBA);
+ sendSignal(SUMA_REF, GSN_API_FAILREQ, signal, 2, JBA);
+ /**
+ * GREP also need the information that an API node
+ * (actually a REP node) has failed.
+ *
+ * GREP does however NOT send a CONF on this signal, i.e.
+ * the API_FAILREQ signal to GREP is like a REP signal
+ * (i.e. without any confirmation).
+ */
+ sendSignal(GREP_REF, GSN_API_FAILREQ, signal, 2, JBA);
+
+ /**-------------------------------------------------------------------------
+ * THE OTHER NODE WAS AN API NODE. THE COMMUNICATION LINK IS ALREADY
+ * BROKEN AND THUS NO ACTION IS NEEDED TO BREAK THE CONNECTION.
+ * WE ONLY NEED TO SET PARAMETERS TO ENABLE A NEW CONNECTION IN A FEW
+ * SECONDS.
+ *-------------------------------------------------------------------------*/
+ failedNodePtr.p->alarmCount = 0;
+
+ CloseComReqConf * const closeCom = (CloseComReqConf *)&signal->theData[0];
+
+ closeCom->xxxBlockRef = reference();
+ closeCom->failNo = 0;
+ closeCom->noOfNodes = 1;
+ NodeBitmask::clear(closeCom->theNodes);
+ NodeBitmask::set(closeCom->theNodes, failedNodePtr.i);
+ sendSignal(CMVMI_REF, GSN_CLOSE_COMREQ, signal,
+ CloseComReqConf::SignalLength, JBA);
+}//Qmgr::sendApiFailReq()
+
+void Qmgr::execAPI_FAILCONF(Signal* signal)
+{
+ NodeRecPtr failedNodePtr;
+
+ jamEntry();
+ failedNodePtr.i = signal->theData[0];
+ ptrCheckGuard(failedNodePtr, MAX_NODES, nodeRec);
+
+ if (failedNodePtr.p->failState == WAITING_FOR_FAILCONF1){
+ jam();
+
+ failedNodePtr.p->rcv[0] = signal->theData[1];
+ failedNodePtr.p->failState = WAITING_FOR_FAILCONF2;
+
+ } else if (failedNodePtr.p->failState == WAITING_FOR_FAILCONF2) {
+ failedNodePtr.p->rcv[1] = signal->theData[1];
+ failedNodePtr.p->failState = NORMAL;
+
+ if (failedNodePtr.p->rcv[0] == failedNodePtr.p->rcv[1]) {
+ jam();
+ systemErrorLab(signal);
+ } else {
+ jam();
+ failedNodePtr.p->rcv[0] = 0;
+ failedNodePtr.p->rcv[1] = 0;
+ }//if
+ } else {
+ jam();
+#ifdef VM_TRACE
+ ndbout << "failedNodePtr.p->failState = " << failedNodePtr.p->failState
+ << endl;
+#endif
+ systemErrorLab(signal);
+ }//if
+ return;
+}//Qmgr::execAPI_FAILCONF()
+
+void Qmgr::execNDB_FAILCONF(Signal* signal)
+{
+ NodeRecPtr failedNodePtr;
+ NodeRecPtr nodePtr;
+
+ jamEntry();
+ failedNodePtr.i = signal->theData[0];
+ ptrCheckGuard(failedNodePtr, MAX_NODES, nodeRec);
+ if (failedNodePtr.p->failState == WAITING_FOR_NDB_FAILCONF){
+ failedNodePtr.p->failState = NORMAL;
+ } else {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ if (cpresident == getOwnNodeId()) {
+ jam();
+ /**
+ * Prepare a NFCompleteRep and send to all connected API's
+ * They can then abort all transaction waiting for response from
+ * the failed node
+ */
+ NFCompleteRep * const nfComp = (NFCompleteRep *)&signal->theData[0];
+ nfComp->blockNo = QMGR_REF;
+ nfComp->nodeId = getOwnNodeId();
+ nfComp->failedNodeId = failedNodePtr.i;
+
+ for (nodePtr.i = 1; nodePtr.i < MAX_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if ((nodePtr.p->phase == ZAPI_ACTIVE) && nodePtr.p->m_connected) {
+ jam();
+ sendSignal(nodePtr.p->blockRef, GSN_NF_COMPLETEREP, signal,
+ NFCompleteRep::SignalLength, JBA);
+ }//if
+ }//for
+ }//if
+ return;
+}//Qmgr::execNDB_FAILCONF()
+
+/*******************************/
+/* DISCONNECT_REP */
+/*******************************/
+void Qmgr::execDISCONNECT_REP(Signal* signal)
+{
+ const DisconnectRep * const rep = (DisconnectRep *)&signal->theData[0];
+ NodeRecPtr failedNodePtr;
+
+ jamEntry();
+ failedNodePtr.i = rep->nodeId;
+ ptrCheckGuard(failedNodePtr, MAX_NODES, nodeRec);
+ failedNodePtr.p->m_connected = false;
+ node_failed(signal, failedNodePtr.i);
+}//DISCONNECT_REP
+
+void Qmgr::node_failed(Signal* signal, Uint16 aFailedNode)
+{
+ NodeRecPtr failedNodePtr;
+ /**------------------------------------------------------------------------
+ * A COMMUNICATION LINK HAS BEEN DISCONNECTED. WE MUST TAKE SOME ACTION
+ * DUE TO THIS.
+ *-----------------------------------------------------------------------*/
+ failedNodePtr.i = aFailedNode;
+ ptrCheckGuard(failedNodePtr, MAX_NODES, nodeRec);
+
+ if (getNodeInfo(failedNodePtr.i).getType() == NodeInfo::DB){
+ jam();
+ /**---------------------------------------------------------------------
+ * THE OTHER NODE IS AN NDB NODE, WE HANDLE IT AS IF A HEARTBEAT
+ * FAILURE WAS DISCOVERED.
+ *---------------------------------------------------------------------*/
+ switch(failedNodePtr.p->phase){
+ case ZRUNNING:
+ jam();
+ failReportLab(signal, aFailedNode, FailRep::ZLINK_FAILURE);
+ return;
+ case ZFAIL_CLOSING:
+ jam();
+ return;
+ default:
+ jam();
+ /*---------------------------------------------------------------------*/
+ // The other node is still not in the cluster but disconnected.
+ // We must restart communication in three seconds.
+ /*---------------------------------------------------------------------*/
+ failedNodePtr.p->failState = NORMAL;
+ failedNodePtr.p->phase = ZFAIL_CLOSING;
+ failedNodePtr.p->alarmCount = 0;
+
+ CloseComReqConf * const closeCom =
+ (CloseComReqConf *)&signal->theData[0];
+
+ closeCom->xxxBlockRef = reference();
+ closeCom->failNo = 0;
+ closeCom->noOfNodes = 1;
+ NodeBitmask::clear(closeCom->theNodes);
+ NodeBitmask::set(closeCom->theNodes, failedNodePtr.i);
+ sendSignal(CMVMI_REF, GSN_CLOSE_COMREQ, signal,
+ CloseComReqConf::SignalLength, JBA);
+ }//if
+ return;
+ }
+
+ /**
+ * API code
+ */
+ jam();
+ if (failedNodePtr.p->phase != ZFAIL_CLOSING){
+ jam();
+ //--------------------------------------------------------------------------
+ // The API was active and has now failed. We need to initiate API failure
+ // handling. If the API had already failed then we can ignore this
+ // discovery.
+ //--------------------------------------------------------------------------
+ failedNodePtr.p->phase = ZFAIL_CLOSING;
+
+ sendApiFailReq(signal, aFailedNode);
+ arbitRec.code = ArbitCode::ApiFail;
+ handleArbitApiFail(signal, aFailedNode);
+ }//if
+ return;
+}//Qmgr::node_failed()
+
+/**--------------------------------------------------------------------------
+ * AN API NODE IS REGISTERING. IF FOR THE FIRST TIME WE WILL ENABLE
+ * COMMUNICATION WITH ALL NDB BLOCKS.
+ *---------------------------------------------------------------------------*/
+/*******************************/
+/* API_REGREQ */
+/*******************************/
+void Qmgr::execAPI_REGREQ(Signal* signal)
+{
+ jamEntry();
+
+ ApiRegReq* req = (ApiRegReq*)signal->getDataPtr();
+ const Uint32 version = req->version;
+ const BlockReference ref = req->ref;
+
+ NodeRecPtr apiNodePtr;
+ apiNodePtr.i = refToNode(ref);
+ ptrCheckGuard(apiNodePtr, MAX_NODES, nodeRec);
+
+#if 0
+ ndbout_c("Qmgr::execAPI_REGREQ: Recd API_REGREQ (NodeId=%d)", apiNodePtr.i);
+#endif
+
+ bool compatability_check;
+ switch(getNodeInfo(apiNodePtr.i).getType()){
+ case NodeInfo::DB:
+ case NodeInfo::INVALID:
+ sendApiRegRef(signal, ref, ApiRegRef::WrongType);
+ return;
+ case NodeInfo::API:
+ compatability_check = ndbCompatible_ndb_api(NDB_VERSION, version);
+ break;
+ case NodeInfo::MGM:
+ compatability_check = ndbCompatible_ndb_mgmt(NDB_VERSION, version);
+ break;
+ case NodeInfo::REP:
+ compatability_check = ndbCompatible_ndb_api(NDB_VERSION, version);
+ break;
+ }
+
+ if (!compatability_check) {
+ jam();
+ apiNodePtr.p->phase = ZAPI_INACTIVE;
+ sendApiRegRef(signal, ref, ApiRegRef::UnsupportedVersion);
+ return;
+ }
+
+ setNodeInfo(apiNodePtr.i).m_version = version;
+
+ apiNodePtr.p->alarmCount = 0;
+
+ ApiRegConf * const apiRegConf = (ApiRegConf *)&signal->theData[0];
+ apiRegConf->qmgrRef = reference();
+ apiRegConf->apiHeartbeatFrequency = (chbApiDelay / 10);
+ apiRegConf->version = NDB_VERSION;
+
+
+ // if(apiNodePtr.i == getNodeState.single. && NodeState::SL_MAINTENANCE)
+ // apiRegConf->nodeState = NodeState::SL_STARTED;
+ //else
+ apiRegConf->nodeState = getNodeState();
+ {
+ NodeRecPtr nodePtr;
+ nodePtr.i = getOwnNodeId();
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ Uint32 dynamicId = nodePtr.p->ndynamicId;
+
+ if(apiRegConf->nodeState.masterNodeId != getOwnNodeId()){
+ jam();
+ apiRegConf->nodeState.dynamicId = dynamicId;
+ } else {
+ apiRegConf->nodeState.dynamicId = -dynamicId;
+ }
+ }
+ sendSignal(ref, GSN_API_REGCONF, signal, ApiRegConf::SignalLength, JBB);
+
+ if ((getNodeState().startLevel == NodeState::SL_STARTED ||
+ getNodeState().getSingleUserMode())
+ && apiNodePtr.p->phase == ZBLOCKED) {
+ jam();
+ /**----------------------------------------------------------------------
+ * THE API NODE IS REGISTERING. WE WILL ACCEPT IT BY CHANGING STATE AND
+ * SENDING A CONFIRM.
+ *----------------------------------------------------------------------*/
+ apiNodePtr.p->phase = ZAPI_ACTIVE;
+ apiNodePtr.p->blockRef = ref;
+ signal->theData[0] = apiNodePtr.i;
+ sendSignal(CMVMI_REF, GSN_ENABLE_COMORD, signal, 1, JBA);
+ }
+ return;
+}//Qmgr::execAPI_REGREQ()
+
+
+void
+Qmgr::execAPI_VERSION_REQ(Signal * signal) {
+ jamEntry();
+ ApiVersionReq * const req = (ApiVersionReq *)signal->getDataPtr();
+
+ Uint32 senderRef = req->senderRef;
+ Uint32 nodeId = req->nodeId;
+
+ ApiVersionConf * conf = (ApiVersionConf *)req;
+ if(getNodeInfo(nodeId).m_connected)
+ conf->version = getNodeInfo(nodeId).m_version;
+ else
+ conf->version = 0;
+ conf->nodeId = nodeId;
+
+ sendSignal(senderRef,
+ GSN_API_VERSION_CONF,
+ signal,
+ ApiVersionConf::SignalLength, JBB);
+
+
+}
+
+
+#if 0
+bool
+Qmgr::checkAPIVersion(NodeId nodeId,
+ Uint32 apiVersion, Uint32 ownVersion) const {
+ bool ret=true;
+ /**
+ * First implementation...
+ */
+ if ((getMajor(apiVersion) < getMajor(ownVersion) ||
+ getMinor(apiVersion) < getMinor(ownVersion)) &&
+ apiVersion >= API_UPGRADE_VERSION) {
+ jam();
+ if ( getNodeInfo(nodeId).getType() != NodeInfo::MGM ) {
+ jam();
+ ret = false;
+ } else {
+ jam();
+ /* we have a software upgrade situation, mgmtsrvr should be
+ * the highest, let him decide what to do
+ */
+ ;
+ }
+ }
+ return ret;
+}
+#endif
+
+void
+Qmgr::sendApiRegRef(Signal* signal, Uint32 Tref, ApiRegRef::ErrorCode err){
+ ApiRegRef* ref = (ApiRegRef*)signal->getDataPtrSend();
+ ref->ref = reference();
+ ref->version = NDB_VERSION;
+ ref->errorCode = err;
+ sendSignal(Tref, GSN_API_REGREF, signal, ApiRegRef::SignalLength, JBB);
+}
+
+/**--------------------------------------------------------------------------
+ * A NODE HAS BEEN DECLARED AS DOWN. WE WILL CLOSE THE COMMUNICATION TO THIS
+ * NODE IF NOT ALREADY DONE. IF WE ARE PRESIDENT OR BECOMES PRESIDENT BECAUSE
+ * OF A FAILED PRESIDENT THEN WE WILL TAKE FURTHER ACTION.
+ *---------------------------------------------------------------------------*/
+void Qmgr::failReportLab(Signal* signal, Uint16 aFailedNode,
+ FailRep::FailCause aFailCause)
+{
+ NodeRecPtr nodePtr;
+ NodeRecPtr failedNodePtr;
+ NodeRecPtr myNodePtr;
+ UintR TnoFailedNodes;
+
+ failedNodePtr.i = aFailedNode;
+ ptrCheckGuard(failedNodePtr, MAX_NODES, nodeRec);
+ if (failedNodePtr.i == getOwnNodeId()) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+ if (myNodePtr.p->phase != ZRUNNING) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+ TnoFailedNodes = cnoFailedNodes;
+ failReport(signal, failedNodePtr.i, (UintR)ZTRUE, aFailCause);
+ if (cpresident == getOwnNodeId()) {
+ jam();
+ if (cpresidentBusy == ZTRUE) {
+ jam();
+/**-------------------------------------------------------------------
+* ALL STARTING NODES ARE CRASHED WHEN AN ALIVE NODE FAILS DURING ITS
+* START-UP. AS PRESIDENT OF THE CLUSTER IT IS OUR DUTY TO INFORM OTHERS
+* ABOUT THIS.
+*---------------------------------------------------------------------*/
+ failReport(signal, cstartNode, (UintR)ZTRUE, FailRep::ZOTHER_NODE_WHEN_WE_START);
+ }//if
+ if (ctoStatus == Q_NOT_ACTIVE) {
+ jam();
+ /**--------------------------------------------------------------------
+ * AS PRESIDENT WE ARE REQUIRED TO START THE EXCLUSION PROCESS SUCH THAT
+ * THE APPLICATION SEE NODE FAILURES IN A CONSISTENT ORDER.
+ * IF WE HAVE BECOME PRESIDENT NOW (CTO_STATUS = ACTIVE) THEN WE HAVE
+ * TO COMPLETE THE PREVIOUS COMMIT FAILED NODE PROCESS BEFORE STARTING
+ * A NEW.
+ * CTO_STATUS = ACTIVE CAN ALSO MEAN THAT WE ARE PRESIDENT AND ARE
+ * CURRENTLY COMMITTING A SET OF NODE CRASHES. IN THIS CASE IT IS NOT
+ * ALLOWED TO START PREPARING NEW NODE CRASHES.
+ *---------------------------------------------------------------------*/
+ if (TnoFailedNodes != cnoFailedNodes) {
+ jam();
+ cfailureNr = cfailureNr + 1;
+ for (nodePtr.i = 1;
+ nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ sendPrepFailReq(signal, nodePtr.i);
+ }//if
+ }//for
+ }//if
+ }//if
+ }//if
+ return;
+}//Qmgr::failReportLab()
+
+/**-------------------------------------------------------------------------
+ * WE HAVE RECEIVED A PREPARE TO EXCLUDE A NUMBER OF NODES FROM THE CLUSTER.
+ * WE WILL FIRST CHECK THAT WE HAVE NOT ANY MORE NODES THAT
+ * WE ALSO HAVE EXCLUDED
+ *--------------------------------------------------------------------------*/
+/*******************************/
+/* PREP_FAILREQ */
+/*******************************/
+void Qmgr::execPREP_FAILREQ(Signal* signal)
+{
+ NodeRecPtr myNodePtr;
+ jamEntry();
+
+ PrepFailReqRef * const prepFail = (PrepFailReqRef *)&signal->theData[0];
+
+ BlockReference Tblockref = prepFail->xxxBlockRef;
+ Uint16 TfailureNr = prepFail->failNo;
+ cnoPrepFailedNodes = prepFail->noOfNodes;
+ UintR arrayIndex = 0;
+ for (Uint32 Tindex = 0; Tindex < MAX_NDB_NODES; Tindex++) {
+ if (NodeBitmask::get(prepFail->theNodes, Tindex)){
+ cprepFailedNodes[arrayIndex] = Tindex;
+ arrayIndex++;
+ }//if
+ }//for
+ UintR guard0;
+
+ /**
+ * Block commit until node failures has stabilized
+ *
+ * @See RT352
+ */
+ BlockCommitOrd* const block = (BlockCommitOrd *)&signal->theData[0];
+ block->failNo = TfailureNr;
+ EXECUTE_DIRECT(DBDIH, GSN_BLOCK_COMMIT_ORD, signal,
+ BlockCommitOrd::SignalLength);
+
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+ if (myNodePtr.p->phase != ZRUNNING) {
+ jam();
+ systemErrorLab(signal);
+ return;
+ }//if
+
+ guard0 = cnoPrepFailedNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (Uint32 Tindex = 0; Tindex <= guard0; Tindex++) {
+ jam();
+ failReport(signal,
+ cprepFailedNodes[Tindex],
+ (UintR)ZFALSE,
+ FailRep::ZIN_PREP_FAIL_REQ);
+ }//for
+ sendCloseComReq(signal, Tblockref, TfailureNr);
+ cnoCommitFailedNodes = 0;
+ cprepareFailureNr = TfailureNr;
+ return;
+}//Qmgr::execPREP_FAILREQ()
+
+/**---------------------------------------------------------------------------
+ * THE CRASHED NODES HAS BEEN EXCLUDED FROM COMMUNICATION.
+ * WE WILL CHECK WHETHER ANY MORE NODES HAVE FAILED DURING THE PREPARE PROCESS.
+ * IF SO WE WILL REFUSE THE PREPARE PHASE AND EXPECT A NEW PREPARE MESSAGE
+ * WITH ALL FAILED NODES INCLUDED.
+ *---------------------------------------------------------------------------*/
+/*******************************/
+/* CLOSE_COMCONF */
+/*******************************/
+void Qmgr::execCLOSE_COMCONF(Signal* signal)
+{
+ jamEntry();
+
+ CloseComReqConf * const closeCom = (CloseComReqConf *)&signal->theData[0];
+
+ BlockReference Tblockref = closeCom->xxxBlockRef;
+ Uint16 TfailureNr = closeCom->failNo;
+
+ cnoPrepFailedNodes = closeCom->noOfNodes;
+ UintR arrayIndex = 0;
+ UintR Tindex = 0;
+ for(Tindex = 0; Tindex < MAX_NDB_NODES; Tindex++){
+ if(NodeBitmask::get(closeCom->theNodes, Tindex)){
+ cprepFailedNodes[arrayIndex] = Tindex;
+ arrayIndex++;
+ }
+ }
+ UintR tprepFailConf;
+ UintR Tindex2;
+ UintR guard0;
+ UintR guard1;
+ UintR Tfound;
+ Uint16 TfailedNodeNo;
+
+ tprepFailConf = ZTRUE;
+ if (cnoFailedNodes > 0) {
+ jam();
+ guard0 = cnoFailedNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (Tindex = 0; Tindex <= guard0; Tindex++) {
+ jam();
+ TfailedNodeNo = cfailedNodes[Tindex];
+ Tfound = ZFALSE;
+ guard1 = cnoPrepFailedNodes - 1;
+ arrGuard(guard1, MAX_NDB_NODES);
+ for (Tindex2 = 0; Tindex2 <= guard1; Tindex2++) {
+ jam();
+ if (TfailedNodeNo == cprepFailedNodes[Tindex2]) {
+ jam();
+ Tfound = ZTRUE;
+ }//if
+ }//for
+ if (Tfound == ZFALSE) {
+ jam();
+ tprepFailConf = ZFALSE;
+ arrGuard(cnoPrepFailedNodes, MAX_NDB_NODES);
+ cprepFailedNodes[cnoPrepFailedNodes] = TfailedNodeNo;
+ cnoPrepFailedNodes = cnoPrepFailedNodes + 1;
+ }//if
+ }//for
+ }//if
+ if (tprepFailConf == ZFALSE) {
+ jam();
+ for (Tindex = 1; Tindex < MAX_NDB_NODES; Tindex++) {
+ cfailedNodes[Tindex] = cprepFailedNodes[Tindex];
+ }//for
+ cnoFailedNodes = cnoPrepFailedNodes;
+ sendPrepFailReqRef(signal,
+ Tblockref,
+ GSN_PREP_FAILREF,
+ reference(),
+ cfailureNr,
+ cnoPrepFailedNodes,
+ cprepFailedNodes);
+ } else {
+ jam();
+ cnoCommitFailedNodes = cnoPrepFailedNodes;
+ guard0 = cnoPrepFailedNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (Tindex = 0; Tindex <= guard0; Tindex++) {
+ jam();
+ arrGuard(Tindex, MAX_NDB_NODES);
+ ccommitFailedNodes[Tindex] = cprepFailedNodes[Tindex];
+ }//for
+ signal->theData[0] = getOwnNodeId();
+ signal->theData[1] = TfailureNr;
+ sendSignal(Tblockref, GSN_PREP_FAILCONF, signal, 2, JBA);
+ }//if
+ return;
+}//Qmgr::execCLOSE_COMCONF()
+
+/*---------------------------------------------------------------------------*/
+/* WE HAVE RECEIVED A CONFIRM OF THAT THIS NODE HAVE PREPARED THE FAILURE. */
+/*---------------------------------------------------------------------------*/
+/*******************************/
+/* PREP_FAILCONF */
+/*******************************/
+void Qmgr::execPREP_FAILCONF(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ NodeRecPtr replyNodePtr;
+ jamEntry();
+ replyNodePtr.i = signal->theData[0];
+ Uint16 TfailureNr = signal->theData[1];
+ if (TfailureNr != cfailureNr) {
+ jam();
+ /**----------------------------------------------------------------------
+ * WE HAVE ALREADY STARTING A NEW ATTEMPT TO EXCLUDE A NUMBER OF NODES.
+ * IGNORE
+ *----------------------------------------------------------------------*/
+ return;
+ }//if
+ ptrCheckGuard(replyNodePtr, MAX_NDB_NODES, nodeRec);
+ replyNodePtr.p->sendPrepFailReqStatus = Q_NOT_ACTIVE;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ if (nodePtr.p->sendPrepFailReqStatus == Q_ACTIVE) {
+ jam();
+ return;
+ }//if
+ }//if
+ }//for
+ /**
+ * Check node count and groups and invoke arbitrator if necessary.
+ * Continues via sendCommitFailReq() if successful.
+ */
+ arbitRec.failureNr = cfailureNr;
+ handleArbitCheck(signal);
+ return;
+}//Qmgr::execPREP_FAILCONF()
+
+void
+Qmgr::sendCommitFailReq(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ jam();
+ if (arbitRec.failureNr != cfailureNr) {
+ jam();
+ /**----------------------------------------------------------------------
+ * WE HAVE ALREADY STARTING A NEW ATTEMPT TO EXCLUDE A NUMBER OF NODES.
+ * IGNORE
+ *----------------------------------------------------------------------*/
+ return;
+ }//if
+ /**-----------------------------------------------------------------------
+ * WE HAVE SUCCESSFULLY PREPARED A SET OF NODE FAILURES. WE WILL NOW COMMIT
+ * THESE NODE FAILURES.
+ *-------------------------------------------------------------------------*/
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ nodePtr.p->sendCommitFailReqStatus = Q_ACTIVE;
+ signal->theData[0] = cpdistref;
+ signal->theData[1] = cfailureNr;
+ sendSignal(nodePtr.p->blockRef, GSN_COMMIT_FAILREQ, signal, 2, JBA);
+ }//if
+ }//for
+ ctoStatus = Q_ACTIVE;
+ cnoFailedNodes = 0;
+ return;
+}//sendCommitFailReq()
+
+/*---------------------------------------------------------------------------*/
+/* SOME NODE HAVE DISCOVERED A NODE FAILURE THAT WE HAVE NOT YET DISCOVERED. */
+/* WE WILL START ANOTHER ROUND OF PREPARING A SET OF NODE FAILURES. */
+/*---------------------------------------------------------------------------*/
+/*******************************/
+/* PREP_FAILREF */
+/*******************************/
+void Qmgr::execPREP_FAILREF(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ jamEntry();
+
+ PrepFailReqRef * const prepFail = (PrepFailReqRef *)&signal->theData[0];
+
+ Uint16 TfailureNr = prepFail->failNo;
+ cnoPrepFailedNodes = prepFail->noOfNodes;
+
+ UintR arrayIndex = 0;
+ UintR Tindex = 0;
+ for(Tindex = 0; Tindex < MAX_NDB_NODES; Tindex++) {
+ jam();
+ if(NodeBitmask::get(prepFail->theNodes, Tindex)){
+ jam();
+ cprepFailedNodes[arrayIndex] = Tindex;
+ arrayIndex++;
+ }//if
+ }//for
+ if (TfailureNr != cfailureNr) {
+ jam();
+ /**---------------------------------------------------------------------
+ * WE HAVE ALREADY STARTING A NEW ATTEMPT TO EXCLUDE A NUMBER OF NODES.
+ * IGNORE
+ *----------------------------------------------------------------------*/
+ return;
+ }//if
+ UintR guard0;
+ UintR Ti;
+
+ cnoFailedNodes = cnoPrepFailedNodes;
+ guard0 = cnoPrepFailedNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ for (Ti = 0; Ti <= guard0; Ti++) {
+ jam();
+ cfailedNodes[Ti] = cprepFailedNodes[Ti];
+ }//for
+ cfailureNr = cfailureNr + 1;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ sendPrepFailReq(signal, nodePtr.i);
+ }//if
+ }//for
+ return;
+}//Qmgr::execPREP_FAILREF()
+
+/*---------------------------------------------------------------------------*/
+/* THE PRESIDENT IS NOW COMMITTING THE PREVIOUSLY PREPARED NODE FAILURE. */
+/*---------------------------------------------------------------------------*/
+/***********************/
+/* COMMIT_FAILREQ */
+/***********************/
+void Qmgr::execCOMMIT_FAILREQ(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ jamEntry();
+ BlockReference Tblockref = signal->theData[0];
+ UintR TfailureNr = signal->theData[1];
+ if (Tblockref != cpdistref) {
+ jam();
+ return;
+ }//if
+ UintR guard0;
+ UintR Ti;
+ UintR Tj;
+ RegAppPtr localRegAppptr;
+
+ /**
+ * Block commit until node failures has stabilized
+ *
+ * @See RT352
+ */
+ UnblockCommitOrd* const unblock = (UnblockCommitOrd *)&signal->theData[0];
+ unblock->failNo = TfailureNr;
+ EXECUTE_DIRECT(DBDIH, GSN_UNBLOCK_COMMIT_ORD, signal,
+ UnblockCommitOrd::SignalLength);
+
+ if ((ccommitFailureNr != TfailureNr) &&
+ (cnoCommitFailedNodes > 0)) {
+ jam();
+ /**-----------------------------------------------------------------------
+ * WE ONLY DO THIS PART OF THE COMMIT HANDLING THE FIRST TIME WE HEAR THIS
+ * SIGNAL. WE CAN HEAR IT SEVERAL TIMES IF THE PRESIDENTS KEEP FAILING.
+ *-----------------------------------------------------------------------*/
+ ccommitFailureNr = TfailureNr;
+ for (localRegAppptr.i = 0;
+ localRegAppptr.i < NO_REG_APP; localRegAppptr.i++) {
+ jam();
+ ptrAss(localRegAppptr, regApp);
+ if (localRegAppptr.p->activity != ZREMOVE) {
+ /*------------------------------------------------------------------*/
+ // We need to remove the failed nodes from the set of running nodes
+ // in the registered application.
+ //------------------------------------------------------------------*/
+ for (Ti = 0; Ti < cnoCommitFailedNodes; Ti++) {
+ jam();
+ arrGuard(ccommitFailedNodes[Ti], MAX_NDB_NODES);
+ localRegAppptr.p->m_runNodes.clear(ccommitFailedNodes[Ti]);
+ }//for
+ /*------------------------------------------------------------------*/
+ // Send a signal to the registered application to inform him of the
+ // node failure(s).
+ /*------------------------------------------------------------------*/
+ NodeFailRep * const nodeFail = (NodeFailRep *)&signal->theData[0];
+
+ nodeFail->failNo = ccommitFailureNr;
+ nodeFail->noOfNodes = cnoCommitFailedNodes;
+ NodeBitmask::clear(nodeFail->theNodes);
+ for(unsigned i = 0; i < cnoCommitFailedNodes; i++) {
+ jam();
+ NodeBitmask::set(nodeFail->theNodes, ccommitFailedNodes[i]);
+ }//if
+ sendSignal(localRegAppptr.p->blockref, GSN_NODE_FAILREP, signal,
+ NodeFailRep::SignalLength, JBB);
+ }//if
+ }//for
+ guard0 = cnoCommitFailedNodes - 1;
+ arrGuard(guard0, MAX_NDB_NODES);
+ /**--------------------------------------------------------------------
+ * WE MUST PREPARE TO ACCEPT THE CRASHED NODE INTO THE CLUSTER AGAIN BY
+ * SETTING UP CONNECTIONS AGAIN AFTER THREE SECONDS OF DELAY.
+ *--------------------------------------------------------------------*/
+ for (Tj = 0; Tj <= guard0; Tj++) {
+ jam();
+ nodePtr.i = ccommitFailedNodes[Tj];
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ nodePtr.p->phase = ZFAIL_CLOSING;
+ nodePtr.p->failState = WAITING_FOR_NDB_FAILCONF;
+ nodePtr.p->alarmCount = 0;
+ }//for
+ /*----------------------------------------------------------------------*/
+ /* WE INFORM THE API'S WE HAVE CONNECTED ABOUT THE FAILED NODES. */
+ /*----------------------------------------------------------------------*/
+ for (nodePtr.i = 1; nodePtr.i < MAX_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZAPI_ACTIVE) {
+ jam();
+
+ NodeFailRep * const nodeFail = (NodeFailRep *)&signal->theData[0];
+
+ nodeFail->failNo = ccommitFailureNr;
+ nodeFail->noOfNodes = cnoCommitFailedNodes;
+ NodeBitmask::clear(nodeFail->theNodes);
+ for(unsigned i = 0; i < cnoCommitFailedNodes; i++) {
+ jam();
+ NodeBitmask::set(nodeFail->theNodes, ccommitFailedNodes[i]);
+ }//for
+ sendSignal(nodePtr.p->blockRef, GSN_NODE_FAILREP, signal,
+ NodeFailRep::SignalLength, JBB);
+ }//if
+ }//for
+ if (cpresident != getOwnNodeId()) {
+ jam();
+ cnoFailedNodes = cnoCommitFailedNodes - cnoFailedNodes;
+ if (cnoFailedNodes > 0) {
+ jam();
+ guard0 = cnoFailedNodes - 1;
+ arrGuard(guard0 + cnoCommitFailedNodes, MAX_NDB_NODES);
+ for (Tj = 0; Tj <= guard0; Tj++) {
+ jam();
+ cfailedNodes[Tj] = cfailedNodes[Tj + cnoCommitFailedNodes];
+ }//for
+ }//if
+ }//if
+ cnoCommitFailedNodes = 0;
+ }//if
+ /**-----------------------------------------------------------------------
+ * WE WILL ALWAYS ACKNOWLEDGE THE COMMIT EVEN WHEN RECEIVING IT MULTIPLE
+ * TIMES SINCE IT WILL ALWAYS COME FROM A NEW PRESIDENT.
+ *------------------------------------------------------------------------*/
+ signal->theData[0] = getOwnNodeId();
+ sendSignal(Tblockref, GSN_COMMIT_FAILCONF, signal, 1, JBA);
+ return;
+}//Qmgr::execCOMMIT_FAILREQ()
+
+/*--------------------------------------------------------------------------*/
+/* WE HAVE RECEIVED A CONFIRM OF THAT THIS NODE HAVE COMMITTED THE FAILURES.*/
+/*--------------------------------------------------------------------------*/
+/*******************************/
+/* COMMIT_FAILCONF */
+/*******************************/
+void Qmgr::execCOMMIT_FAILCONF(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ NodeRecPtr replyNodePtr;
+ jamEntry();
+ replyNodePtr.i = signal->theData[0];
+
+ ptrCheckGuard(replyNodePtr, MAX_NDB_NODES, nodeRec);
+ replyNodePtr.p->sendCommitFailReqStatus = Q_NOT_ACTIVE;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ if (nodePtr.p->sendCommitFailReqStatus == Q_ACTIVE) {
+ jam();
+ return;
+ }//if
+ }//if
+ }//for
+ /*-----------------------------------------------------------------------*/
+ /* WE HAVE SUCCESSFULLY COMMITTED A SET OF NODE FAILURES. */
+ /*-----------------------------------------------------------------------*/
+ ctoStatus = Q_NOT_ACTIVE;
+ if (cnoFailedNodes != 0) {
+ jam();
+ /**----------------------------------------------------------------------
+ * A FAILURE OCCURRED IN THE MIDDLE OF THE COMMIT PROCESS. WE ARE NOW
+ * READY TO START THE FAILED NODE PROCESS FOR THIS NODE.
+ *----------------------------------------------------------------------*/
+ cfailureNr = cfailureNr + 1;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ sendPrepFailReq(signal, nodePtr.i);
+ }//if
+ }//for
+ }//if
+ return;
+}//Qmgr::execCOMMIT_FAILCONF()
+
+/**--------------------------------------------------------------------------
+ * IF THE PRESIDENT FAILS IN THE MIDDLE OF THE COMMIT OF A FAILED NODE THEN
+ * THE NEW PRESIDENT NEEDS TO QUERY THE COMMIT STATUS IN THE RUNNING NODES.
+ *---------------------------------------------------------------------------*/
+/*******************************/
+/* PRES_TOCONF */
+/*******************************/
+void Qmgr::execPRES_TOCONF(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ NodeRecPtr replyNodePtr;
+ jamEntry();
+ replyNodePtr.i = signal->theData[0];
+ UintR TfailureNr = signal->theData[1];
+ if (ctoFailureNr < TfailureNr) {
+ jam();
+ ctoFailureNr = TfailureNr;
+ }//if
+ ptrCheckGuard(replyNodePtr, MAX_NDB_NODES, nodeRec);
+ replyNodePtr.p->sendPresToStatus = Q_NOT_ACTIVE;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->sendPresToStatus == Q_ACTIVE) {
+ jam();
+ return;
+ }//if
+ }//for
+ /*-------------------------------------------------------------------------*/
+ /* WE ARE NOW READY TO DISCOVER WHETHER THE FAILURE WAS COMMITTED OR NOT. */
+ /*-------------------------------------------------------------------------*/
+ if (ctoFailureNr > ccommitFailureNr) {
+ jam();
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ nodePtr.p->sendCommitFailReqStatus = Q_ACTIVE;
+ signal->theData[0] = cpdistref;
+ signal->theData[1] = ctoFailureNr;
+ sendSignal(nodePtr.p->blockRef, GSN_COMMIT_FAILREQ, signal, 2, JBA);
+ }//if
+ }//for
+ return;
+ }//if
+ /*-------------------------------------------------------------------------*/
+ /* WE ARE NOW READY TO START THE NEW NODE FAILURE PROCESS. */
+ /*-------------------------------------------------------------------------*/
+ ctoStatus = Q_NOT_ACTIVE;
+ cfailureNr = cfailureNr + 1;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ sendPrepFailReq(signal, nodePtr.i);
+ }//if
+ }//for
+ return;
+}//Qmgr::execPRES_TOCONF()
+
+/*--------------------------------------------------------------------------*/
+// Provide information about the configured NDB nodes in the system.
+/*--------------------------------------------------------------------------*/
+void Qmgr::execREAD_NODESREQ(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ UintR TnoOfNodes = 0;
+ BlockReference TBref = signal->theData[0];
+
+ ReadNodesConf * const readNodes = (ReadNodesConf *)&signal->theData[0];
+ NodeBitmask::clear(readNodes->allNodes);
+
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (getNodeInfo(nodePtr.i).getType() == NodeInfo::DB){
+ jam();
+ TnoOfNodes++;
+ NodeBitmask::set(readNodes->allNodes, nodePtr.i);
+ }//if
+ }//for
+ readNodes->noOfNodes = TnoOfNodes;
+ sendSignal(TBref, GSN_READ_NODESCONF, signal,
+ ReadNodesConf::SignalLength, JBB);
+}//Qmgr::execREAD_NODESREQ()
+/*--------------------------------------------------------------------------
+ * Signal from an application requesting to be monitored in the cluster.
+ * APPL_REGREQ can be entered at any time during the life of the QMGR.
+ * It can be entered any number of times.
+ * If QMGR is ZRUNNING a CM_APPCHG will be sent to all active nodes.
+ *---------------------------------------------------------------------------*/
+void Qmgr::execAPPL_REGREQ(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ NodeRecPtr myNodePtr;
+ RegAppPtr lRegApptr;
+ char Tappname[16];
+ jamEntry();
+ BlockReference Tappref = signal->theData[0];
+ Tappname[0] = signal->theData[1] >> 8;
+ Tappname[1] = signal->theData[2];
+ Tappname[2] = signal->theData[2] >> 8;
+ Tappname[3] = signal->theData[3];
+ Tappname[4] = signal->theData[3] >> 8;
+ Tappname[5] = signal->theData[4];
+ Tappname[6] = signal->theData[4] >> 8;
+ Tappname[7] = signal->theData[5];
+ Tappname[8] = signal->theData[5] >> 8;
+ Tappname[9] = signal->theData[6];
+ Tappname[10] = signal->theData[6] >> 8;
+ Tappname[11] = signal->theData[7];
+ Tappname[12] = signal->theData[7] >> 8;
+ Tappname[13] = signal->theData[8];
+ Tappname[14] = signal->theData[8] >> 8;
+ Tappname[signal->theData[1] & 0xFF] = 0;
+ UintR Tversion = signal->theData[10];
+ Uint16 Tnodeno = refToNode(Tappref);
+ if (Tnodeno == 0) {
+ jam();
+ /* Fix for all not distributed applications. */
+ Tnodeno = getOwnNodeId();
+ }//if
+ if (getOwnNodeId() == Tnodeno) {
+ jam();
+ /* Local application */
+ UintR Tfound = RNIL;
+ for (lRegApptr.i = NO_REG_APP-1; (Uint16)~lRegApptr.i; lRegApptr.i--) {
+ jam();
+ ptrAss(lRegApptr, regApp);
+ if (lRegApptr.p->activity == ZREMOVE) {
+ Tfound = lRegApptr.i;
+ break;
+ }//if
+ }//for
+ if (Tfound != RNIL) {
+ jam();
+ /* If there was a slot available we */
+ /* register the application */
+ lRegApptr.i = Tfound;
+ ptrCheckGuard(lRegApptr, NO_REG_APP, regApp);
+ lRegApptr.p->blockref = Tappref;
+ strcpy(lRegApptr.p->name, Tappname);
+ lRegApptr.p->version = Tversion;
+ lRegApptr.p->activity = ZADD;
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+ /****************************<*/
+ /*< APPL_REGCONF <*/
+ /****************************<*/
+ signal->theData[0] = lRegApptr.i;
+ signal->theData[1] = cnoOfNodes;
+ signal->theData[2] = cpresident;
+ signal->theData[3] = myNodePtr.p->ndynamicId;
+ sendSignal(lRegApptr.p->blockref, GSN_APPL_REGCONF, signal, 4, JBB);
+ if (myNodePtr.p->phase == ZRUNNING) {
+ jam();
+ /* Check to see if any further action */
+ for (nodePtr.i = 1;
+ nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ /* is needed at this time */
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ sendappchg(signal, lRegApptr.i, nodePtr.i);
+ }//if
+ }//for
+ }//if
+ } else {
+ jam();
+ /****************************<*/
+ /*< APPL_REGREF <*/
+ /****************************<*/
+ signal->theData[0] = ZERRTOOMANY;
+ sendSignal(Tappref, GSN_APPL_REGREF, signal, 1, JBB);
+ }//if
+ } else {
+ jam();
+ /* TOO MANY REGISTERED APPLICATIONS */
+ systemErrorLab(signal);
+ }//if
+ return;
+}//Qmgr::execAPPL_REGREQ()
+
+/*
+4.4.11 APPL_STARTREG */
+/**--------------------------------------------------------------------------
+ * Signal from an application indicating that it is ready to start running
+ * distributed. If the application is running alone or if all other
+ * applications of the same kind already have registered as STARTING then
+ * APPL_STARTCONF will be sent to the application as soon as phase four of
+ * STTOR is reached.
+ *--------------------------------------------------------------------------*/
+/*******************************/
+/* APPL_STARTREG */
+/*******************************/
+void Qmgr::execAPPL_STARTREG(Signal* signal)
+{
+ RegAppPtr lRegApptr;
+ NodeRecPtr myNodePtr;
+ UintR TnodeId;
+ jamEntry();
+ lRegApptr.i = signal->theData[0];
+ ptrCheckGuard(lRegApptr, NO_REG_APP, regApp);
+ UintR Tcounter = signal->theData[1];
+
+ lRegApptr.p->activity = ZSTART;
+ /* Application is ready to start. */
+
+ /* Calculate how many apps we wait for */
+ lRegApptr.p->noofapps = (Tcounter - 1) - lRegApptr.p->noofpending;
+ /* send info to all other running nodes in the */
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+ /* cluster indicating the status change of the */
+ if (myNodePtr.p->phase == ZRUNNING) {
+ /* application. */
+ for (TnodeId = 1; TnodeId < MAX_NDB_NODES; TnodeId++) {
+ jam();
+ if (lRegApptr.p->m_runNodes.get(TnodeId)){
+ jam();
+ sendappchg(signal, lRegApptr.i, TnodeId);
+ }//if
+ }//for
+ }//if
+ /****************************<*/
+ /*< APPL_STARTCONF <*/
+ /****************************<*/
+ if (lRegApptr.p->noofapps == 0) {
+ jam();
+ sendSignal(lRegApptr.p->blockref, GSN_APPL_STARTCONF, signal, 1, JBB);
+ }//if
+ return;
+}//Qmgr::execAPPL_STARTREG()
+
+/*
+ 4.4.11 APPL_RUN */
+/*--------------------------------------------------------------------------*/
+/* Signal from an application announcing that it is running. */
+/*--------------------------------------------------------------------------*/
+/*******************************/
+/* APPL_RUN */
+/*******************************/
+void Qmgr::execAPPL_RUN(Signal* signal)
+{
+ RegAppPtr lRegApptr;
+ NodeRecPtr myNodePtr;
+ UintR TnodeId;
+ jamEntry();
+ lRegApptr.i = signal->theData[0];
+ ptrCheckGuard(lRegApptr, NO_REG_APP, regApp);
+ lRegApptr.p->activity = ZRUN;
+ /* Flag the application as running. */
+ myNodePtr.i = getOwnNodeId();
+ ptrCheckGuard(myNodePtr, MAX_NDB_NODES, nodeRec);
+ if (myNodePtr.p->phase == ZRUNNING) {
+ /* If we are running send the appl. status */
+ for (TnodeId = 1; TnodeId < MAX_NDB_NODES; TnodeId++) {
+ jam();
+ /* change to all other running nodes. */
+ if (lRegApptr.p->m_runNodes.get(TnodeId)){
+ jam();
+ sendappchg(signal, lRegApptr.i, TnodeId);
+ }//if
+ }//for
+ }//if
+ /****************************<*/
+ /*< CM_RUN <*/
+ /****************************<*/
+ /*---------------------------------------------------*/
+ /* Inform the CLUSTER CONTROL of NDB started */
+ /* so we can connect to API nodes. */
+ /*---------------------------------------------------*/
+ signal->theData[0] = 1;
+ sendSignal(CMVMI_REF, GSN_CM_RUN, signal, 1, JBB);
+ cactivateApiCheck = 1;
+ /**
+ * Start arbitration thread. This could be done as soon as
+ * we have all nodes (or a winning majority).
+ */
+ if (cpresident == getOwnNodeId())
+ handleArbitStart(signal);
+ return;
+}//Qmgr::execAPPL_RUN()
+
+
+void Qmgr::systemErrorBecauseOtherNodeFailed(Signal* signal,
+ NodeId failedNodeId) {
+ jam();
+
+ // Broadcast that this node is failing to other nodes
+ failReport(signal, getOwnNodeId(), (UintR)ZTRUE, FailRep::ZOWN_FAILURE);
+
+ char buf[100];
+ snprintf(buf, 100,
+ "Node was shutdown during startup because node %d failed",
+ failedNodeId);
+
+ progError(__LINE__, ERR_SR_OTHERNODEFAILED, buf);
+}
+
+
+void Qmgr::systemErrorLab(Signal* signal, const char * message)
+{
+ jam();
+ // Broadcast that this node is failing to other nodes
+ failReport(signal, getOwnNodeId(), (UintR)ZTRUE, FailRep::ZOWN_FAILURE);
+
+ // If it's known why shutdown occured
+ // an error message has been passed to this function
+ progError(__LINE__, 0, message);
+
+ return;
+}//Qmgr::systemErrorLab()
+
+/*
+4.4.11 CM_APPCHG */
+/*---------------------------------------------------------------------------*/
+/*Signal between two QMGRs used to announce any changes of state for an appl.*/
+/*---------------------------------------------------------------------------*/
+/*******************************/
+/* CM_APPCHG */
+/*******************************/
+void Qmgr::execCM_APPCHG(Signal* signal)
+{
+ RegAppPtr lRegApptr;
+ char Tappname[16];
+ jamEntry();
+ UintR Ttype = signal->theData[0];
+ Uint16 Tnodeno = signal->theData[1];
+ Tappname[0] = signal->theData[2] >> 8;
+ Tappname[1] = signal->theData[3];
+ Tappname[2] = signal->theData[3] >> 8;
+ Tappname[3] = signal->theData[4];
+ Tappname[4] = signal->theData[4] >> 8;
+ Tappname[5] = signal->theData[5];
+ Tappname[6] = signal->theData[5] >> 8;
+ Tappname[7] = signal->theData[6];
+ Tappname[8] = signal->theData[6] >> 8;
+ Tappname[9] = signal->theData[7];
+ Tappname[10] = signal->theData[7] >> 8;
+ Tappname[11] = signal->theData[8];
+ Tappname[12] = signal->theData[8] >> 8;
+ Tappname[13] = signal->theData[9];
+ Tappname[14] = signal->theData[9] >> 8;
+ Tappname[signal->theData[2] & 0xFF] = 0;
+ UintR Tversion = signal->theData[11];
+ switch (Ttype) {
+ case ZADD:
+ jam();
+ /* A new application has started on the sending node */
+ for (lRegApptr.i = NO_REG_APP-1; (Uint16)~lRegApptr.i; lRegApptr.i--) {
+ jam();
+ /* We are hosting this application */
+ ptrAss(lRegApptr, regApp);
+ if (strcmp(lRegApptr.p->name, Tappname) == 0) {
+ cmappAdd(signal, lRegApptr.i, Tnodeno, Ttype, Tversion);
+ }//if
+ }//for
+ break;
+
+ case ZSTART:
+ jam();
+ /* A registered application is ready to start on the sending node */
+ for (lRegApptr.i = NO_REG_APP-1; (Uint16)~lRegApptr.i; lRegApptr.i--) {
+ jam();
+ ptrAss(lRegApptr, regApp);
+ if (strcmp(lRegApptr.p->name, Tappname) == 0) {
+ cmappStart(signal, lRegApptr.i, Tnodeno, Ttype, Tversion);
+ }//if
+ }//for
+ break;
+
+ case ZRUN:
+ /* A registered application on the sending node has started to run */
+ jam();
+ for (lRegApptr.i = NO_REG_APP-1; (Uint16)~lRegApptr.i; lRegApptr.i--) {
+ jam();
+ ptrAss(lRegApptr, regApp);
+ if (strcmp(lRegApptr.p->name, Tappname) == 0) {
+ arrGuard(Tnodeno, MAX_NDB_NODES);
+ lRegApptr.p->m_runNodes.set(Tnodeno);
+ applchangerep(signal, lRegApptr.i, Tnodeno, Ttype, Tversion);
+ }//if
+ }//for
+ cacceptRegreq = ZTRUE; /* We can now start accepting new CM_REGREQ */
+ /* since the new node is running */
+ break;
+
+ case ZREMOVE:
+ /* A registered application has been deleted on the sending node */
+ jam();
+ for (lRegApptr.i = NO_REG_APP-1; (Uint16)~lRegApptr.i; lRegApptr.i--) {
+ jam();
+ ptrAss(lRegApptr, regApp);
+ if (strcmp(lRegApptr.p->name, Tappname) == 0) {
+ applchangerep(signal, lRegApptr.i, Tnodeno, Ttype, Tversion);
+ }//if
+ }//for
+ break;
+
+ default:
+ jam();
+ /*empty*/;
+ break;
+ }//switch
+ return;
+}//Qmgr::execCM_APPCHG()
+
+/**--------------------------------------------------------------------------
+ * INPUT REG_APPPTR
+ * TNODENO
+ *--------------------------------------------------------------------------*/
+void Qmgr::applchangerep(Signal* signal,
+ UintR aRegApp,
+ Uint16 aNode,
+ UintR aType,
+ UintR aVersion)
+{
+ RegAppPtr lRegApptr;
+ NodeRecPtr localNodePtr;
+ lRegApptr.i = aRegApp;
+ ptrCheckGuard(lRegApptr, NO_REG_APP, regApp);
+ if (lRegApptr.p->blockref != 0) {
+ jam();
+ localNodePtr.i = aNode;
+ ptrCheckGuard(localNodePtr, MAX_NDB_NODES, nodeRec);
+ /****************************************/
+ /* Send a report of changes on another */
+ /* node to the local application */
+ /****************************************/
+ signal->theData[0] = aType;
+ signal->theData[1] = aVersion;
+ signal->theData[2] = localNodePtr.i;
+ signal->theData[4] = localNodePtr.p->ndynamicId;
+ sendSignal(lRegApptr.p->blockref, GSN_APPL_CHANGEREP, signal, 5, JBB);
+ }//if
+}//Qmgr::applchangerep()
+
+/*
+ 4.10.7 CMAPP_ADD */
+/**--------------------------------------------------------------------------
+ * We only map applications of the same version. We have the same application
+ * and version locally.
+ * INPUT REG_APPPTR
+ * TNODENO Sending node
+ * TVERSION Version of application
+ * OUTPUT REG_APPPTR, TNODENO ( not changed)
+ *---------------------------------------------------------------------------*/
+void Qmgr::cmappAdd(Signal* signal,
+ UintR aRegApp,
+ Uint16 aNode,
+ UintR aType,
+ UintR aVersion)
+{
+ RegAppPtr lRegApptr;
+ lRegApptr.i = aRegApp;
+ ptrCheckGuard(lRegApptr, NO_REG_APP, regApp);
+ if (lRegApptr.p->version == aVersion) {
+ jam();
+ arrGuard(aNode, MAX_NDB_NODES);
+ if (lRegApptr.p->m_runNodes.get(aNode) == false){
+ jam();
+ /* Check if we already have added it. */
+ /*-------------------------------------------------------*/
+ /* Since we only add remote applications, if we also are */
+ /* hosting them we need to send a reply indicating that */
+ /* we also are hosting the application. */
+ /*-------------------------------------------------------*/
+ sendappchg(signal, lRegApptr.i, aNode);
+ lRegApptr.p->m_runNodes.set(aNode);
+ /*---------------------------------------*/
+ /* Add the remote node to the the local */
+ /* nodes memberlist. */
+ /* Inform the local application of the */
+ /* new application running remotely. */
+ /*---------------------------------------*/
+ applchangerep(signal, lRegApptr.i, aNode, aType, aVersion);
+ }//if
+ }//if
+}//Qmgr::cmappAdd()
+
+/*
+4.10.7 CMAPP_START */
+/**--------------------------------------------------------------------------
+ * Inform the local application of the change in node state on the remote node
+ * INPUT REG_APPPTR
+ * OUTPUT -
+ *---------------------------------------------------------------------------*/
+void Qmgr::cmappStart(Signal* signal,
+ UintR aRegApp,
+ Uint16 aNode,
+ UintR aType,
+ UintR aVersion)
+{
+ RegAppPtr lRegApptr;
+ lRegApptr.i = aRegApp;
+ ptrCheckGuard(lRegApptr, NO_REG_APP, regApp);
+ if (lRegApptr.p->version == aVersion) {
+ applchangerep(signal, lRegApptr.i, aNode, aType, aVersion);
+ if (lRegApptr.p->activity == ZSTART) {
+ jam();
+ //----------------------------------------
+ /* If the local application is already */
+ /* in START face then we do some checks.*/
+ //----------------------------------------
+ if (lRegApptr.p->noofapps > 0) {
+ jam();
+ //----------------------------------------
+ /* Check if we need to decrement the no */
+ /* of apps. */
+ /* This indicates how many startsignals */
+ /* from apps remaining before we can */
+ /* send a APPL_STARTCONF. */
+ //----------------------------------------
+ lRegApptr.p->noofapps--;
+ }//if
+ if (lRegApptr.p->noofapps == 0) {
+ jam();
+ //----------------------------------------
+ /* All applications have registered as */
+ /* ready to start. */
+ //----------------------------------------
+ /****************************<*/
+ /*< APPL_STARTCONF <*/
+ /****************************<*/
+ sendSignal(lRegApptr.p->blockref, GSN_APPL_STARTCONF, signal, 1, JBB);
+ }//if
+ } else {
+ jam();
+ /**--------------------------------------------------------------------
+ * Add the ready node to the nodes pending counter.
+ * This counter is used to see how many remote nodes that are waiting
+ * for this node to enter the start face.
+ * It is used when the appl. sends a APPL_STARTREG signal.
+ *---------------------------------------------------------------------*/
+ if (lRegApptr.p->activity == ZADD) {
+ jam();
+ lRegApptr.p->noofpending++;
+ }//if
+ }//if
+ }//if
+}//Qmgr::cmappStart()
+
+/**---------------------------------------------------------------------------
+ * A FAILURE HAVE BEEN DISCOVERED ON A NODE. WE NEED TO CLEAR A
+ * NUMBER OF VARIABLES.
+ *---------------------------------------------------------------------------*/
+void Qmgr::failReport(Signal* signal,
+ Uint16 aFailedNode,
+ UintR aSendFailRep,
+ FailRep::FailCause aFailCause)
+{
+ UintR tfrMinDynamicId;
+ NodeRecPtr failedNodePtr;
+ NodeRecPtr nodePtr;
+ NodeRecPtr presidentNodePtr;
+
+
+ failedNodePtr.i = aFailedNode;
+ ptrCheckGuard(failedNodePtr, MAX_NDB_NODES, nodeRec);
+ if (((cpresidentBusy == ZTRUE) ||
+ (cacceptRegreq == ZFALSE)) &&
+ (cstartNode == aFailedNode)) {
+ jam();
+/*----------------------------------------------------------------------*/
+// A node crashed keeping the president busy and that ensures that there
+// is no acceptance of regreq's which is not acceptable after its crash.
+/*----------------------------------------------------------------------*/
+ cpresidentBusy = ZFALSE;
+ cacceptRegreq = ZTRUE;
+ }//if
+ if (failedNodePtr.p->phase == ZRUNNING) {
+ jam();
+/* WE ALSO NEED TO ADD HERE SOME CODE THAT GETS OUR NEW NEIGHBOURS. */
+ if (cpresident == getOwnNodeId()) {
+ jam();
+ if (failedNodePtr.p->sendCommitFailReqStatus == Q_ACTIVE) {
+ jam();
+ signal->theData[0] = failedNodePtr.i;
+ sendSignal(QMGR_REF, GSN_COMMIT_FAILCONF, signal, 1, JBA);
+ }//if
+ if (failedNodePtr.p->sendPresToStatus == Q_ACTIVE) {
+ jam();
+ signal->theData[0] = failedNodePtr.i;
+ signal->theData[1] = ccommitFailureNr;
+ sendSignal(QMGR_REF, GSN_PRES_TOCONF, signal, 2, JBA);
+ }//if
+ }//if
+ failedNodePtr.p->phase = ZPREPARE_FAIL;
+ failedNodePtr.p->sendCmAddPrepStatus = Q_NOT_ACTIVE;
+ failedNodePtr.p->sendCmAddCommitStatus = Q_NOT_ACTIVE;
+ failedNodePtr.p->sendPrepFailReqStatus = Q_NOT_ACTIVE;
+ failedNodePtr.p->sendCommitFailReqStatus = Q_NOT_ACTIVE;
+ failedNodePtr.p->sendPresToStatus = Q_NOT_ACTIVE;
+ failedNodePtr.p->alarmCount = 0;
+ if (aSendFailRep == ZTRUE) {
+ jam();
+ if (failedNodePtr.i != getOwnNodeId()) {
+ jam();
+ FailRep * const failRep = (FailRep *)&signal->theData[0];
+ failRep->failNodeId = failedNodePtr.i;
+ failRep->failCause = aFailCause;
+ sendSignal(failedNodePtr.p->blockRef, GSN_FAIL_REP, signal,
+ FailRep::SignalLength, JBA);
+ }//if
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ FailRep * const failRep = (FailRep *)&signal->theData[0];
+ failRep->failNodeId = failedNodePtr.i;
+ failRep->failCause = aFailCause;
+ sendSignal(nodePtr.p->blockRef, GSN_FAIL_REP, signal,
+ FailRep::SignalLength, JBA);
+ }//if
+ }//for
+ }//if
+ if (failedNodePtr.i == getOwnNodeId()) {
+ jam();
+ return;
+ }//if
+ failedNodePtr.p->ndynamicId = 0;
+ findNeighbours(signal);
+ if (failedNodePtr.i == cpresident) {
+ jam();
+ /**--------------------------------------------------------------------
+ * IF PRESIDENT HAVE FAILED WE MUST CALCULATE THE NEW PRESIDENT BY
+ * FINDING THE NODE WITH THE MINIMUM DYNAMIC IDENTITY.
+ *---------------------------------------------------------------------*/
+ tfrMinDynamicId = (UintR)-1;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ if (nodePtr.p->ndynamicId < tfrMinDynamicId) {
+ jam();
+ tfrMinDynamicId = nodePtr.p->ndynamicId;
+ cpresident = nodePtr.i;
+ }//if
+ }//if
+ }//for
+ presidentNodePtr.i = cpresident;
+ ptrCheckGuard(presidentNodePtr, MAX_NDB_NODES, nodeRec);
+ cpdistref = presidentNodePtr.p->blockRef;
+ if (cpresident == getOwnNodeId()) {
+ CRASH_INSERTION(920);
+ cfailureNr = cprepareFailureNr;
+ ctoFailureNr = 0;
+ ctoStatus = Q_ACTIVE;
+ if (cnoCommitFailedNodes > 0) {
+ jam();
+ /**-----------------------------------------------------------------
+ * IN THIS SITUATION WE ARE UNCERTAIN OF WHETHER THE NODE FAILURE
+ * PROCESS WAS COMMITTED. WE NEED TO QUERY THE OTHER NODES ABOUT
+ * THEIR STATUS.
+ *-----------------------------------------------------------------*/
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES;
+ nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ nodePtr.p->sendPresToStatus = Q_ACTIVE;
+ signal->theData[0] = cpdistref;
+ signal->theData[1] = cprepareFailureNr;
+ sendSignal(nodePtr.p->blockRef, GSN_PRES_TOREQ,
+ signal, 1, JBA);
+ }//if
+ }//for
+ } else {
+ jam();
+ /*-----------------------------------------------------------------*/
+ // In this case it could be that a commit process is still ongoing.
+ // If so we must conclude it as the new master.
+ /*-----------------------------------------------------------------*/
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES;
+ nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ jam();
+ nodePtr.p->sendCommitFailReqStatus = Q_ACTIVE;
+ signal->theData[0] = cpdistref;
+ signal->theData[1] = ccommitFailureNr;
+ sendSignal(nodePtr.p->blockRef, GSN_COMMIT_FAILREQ, signal,
+ 2, JBA);
+ }//if
+ }//for
+ }//if
+ }//if
+ }//if
+ arrGuard(cnoFailedNodes, MAX_NDB_NODES);
+ cfailedNodes[cnoFailedNodes] = failedNodePtr.i;
+ cnoFailedNodes = cnoFailedNodes + 1;
+ }//if
+}//Qmgr::failReport()
+
+/*---------------------------------------------------------------------------*/
+/* INPUT: TTDI_DYN_ID */
+/* OUTPUT: TTDI_NODE_ID */
+/*---------------------------------------------------------------------------*/
+Uint16 Qmgr::translateDynamicIdToNodeId(Signal* signal, UintR TdynamicId)
+{
+ NodeRecPtr tdiNodePtr;
+ Uint16 TtdiNodeId = ZNIL;
+
+ for (tdiNodePtr.i = 1; tdiNodePtr.i < MAX_NDB_NODES; tdiNodePtr.i++) {
+ jam();
+ ptrAss(tdiNodePtr, nodeRec);
+ if (tdiNodePtr.p->ndynamicId == TdynamicId) {
+ jam();
+ TtdiNodeId = tdiNodePtr.i;
+ break;
+ }//if
+ }//for
+ if (TtdiNodeId == ZNIL) {
+ jam();
+ systemErrorLab(signal);
+ }//if
+ return TtdiNodeId;
+}//Qmgr::translateDynamicIdToNodeId()
+
+
+/*
+4.10.7 GET_DYNAMIC_ID */
+/**--------------------------------------------------------------------------
+ * FIND THE CLOSEST HIGHER DYNAMIC ID AMONG THE RUNNING NODES. ADD ONE TO
+ * THAT VALUE AND WE HAVE CREATED A NEW, UNIQUE AND HIGHER DYNAMIC VALUE THAN
+ * ANYONE ELSE IN THE CLUSTER.THIS WAY WE DON'T HAVE TO KEEP TRACK OF VARIABLE
+ * THAT HOLDS THE LAST USED DYNAMIC ID, ESPECIALLY WE DON'T NEED TO INFORM
+ * ANY VICE PRESIDENTS ABOUT THAT DYNAMIC VARIABLE.
+ * INPUT -
+ * RET CDYNAMIC_ID USED AS A TEMPORARY VARIABLE TO PASS THE VALUE TO THE
+ * CALLER OF THIS SUBROUTINE
+ *---------------------------------------------------------------------------*/
+UintR Qmgr::getDynamicId(Signal* signal)
+{
+ NodeRecPtr nodePtr;
+ UintR TdynamicId = 0;
+ for (nodePtr.i = 1; nodePtr.i < MAX_NDB_NODES; nodePtr.i++) {
+ jam();
+ ptrAss(nodePtr, nodeRec);
+ if (nodePtr.p->phase == ZRUNNING) {
+ if (nodePtr.p->ndynamicId > TdynamicId) {
+ jam();
+ TdynamicId = nodePtr.p->ndynamicId;
+ }//if
+ }//if
+ }//for
+ TdynamicId++;
+ return TdynamicId;
+}//Qmgr::getDynamicId()
+
+/*
+4.10.7 SENDAPPCHG */
+/*---------------------------------------------------------------------------*/
+/* We only send changes to external nodes. */
+/* INPUT: TNODENO */
+/* REG_APPPTR */
+/*---------------------------------------------------------------------------*/
+void Qmgr::sendappchg(Signal* signal, UintR aRegApp, Uint16 aNode)
+{
+ NodeRecPtr localNodePtr;
+ RegAppPtr lRegApptr;
+ if (aNode != getOwnNodeId()) {
+ jam();
+ localNodePtr.i = aNode;
+ ptrCheckGuard(localNodePtr, MAX_NDB_NODES, nodeRec);
+ lRegApptr.i = aRegApp;
+ ptrCheckGuard(lRegApptr, NO_REG_APP, regApp);
+ /****************************************/
+ /* Signal any application changes to */
+ /* the receiving node */
+ /****************************************/
+ signal->theData[0] = lRegApptr.p->activity;
+ signal->theData[1] = getOwnNodeId();
+ signal->theData[2] = strlen(lRegApptr.p->name)|(lRegApptr.p->name[0] << 8);
+ signal->theData[3] = lRegApptr.p->name[1] | (lRegApptr.p->name[2] << 8);
+ signal->theData[4] = lRegApptr.p->name[3] | (lRegApptr.p->name[4] << 8);
+ signal->theData[5] = lRegApptr.p->name[5] | (lRegApptr.p->name[6] << 8);
+ signal->theData[6] = lRegApptr.p->name[7] | (lRegApptr.p->name[8] << 8);
+ signal->theData[7] = lRegApptr.p->name[9] | (lRegApptr.p->name[10] << 8);
+ signal->theData[8] = lRegApptr.p->name[11] | (lRegApptr.p->name[12] << 8);
+ signal->theData[9] = lRegApptr.p->name[13] | (lRegApptr.p->name[14] << 8);
+ signal->theData[10] = 0;
+ signal->theData[11] = lRegApptr.p->version;
+ sendSignal(localNodePtr.p->blockRef, GSN_CM_APPCHG, signal, 12, JBA);
+ }//if
+}//Qmgr::sendappchg()
+
+/**--------------------------------------------------------------------------
+ * WHEN RECEIVING PREPARE FAILURE REQUEST WE WILL IMMEDIATELY CLOSE
+ * COMMUNICATION WITH ALL THOSE NODES.
+ *--------------------------------------------------------------------------*/
+void Qmgr::sendCloseComReq(Signal* signal, BlockReference TBRef, Uint16 aFailNo)
+{
+ CloseComReqConf * const closeCom = (CloseComReqConf *)&signal->theData[0];
+
+ closeCom->xxxBlockRef = TBRef;
+ closeCom->failNo = aFailNo;
+ closeCom->noOfNodes = cnoPrepFailedNodes;
+
+ NodeBitmask::clear(closeCom->theNodes);
+
+ for(int i = 0; i < cnoPrepFailedNodes; i++) {
+ const NodeId nodeId = cprepFailedNodes[i];
+ jam();
+ NodeBitmask::set(closeCom->theNodes, nodeId);
+ }
+
+ sendSignal(CMVMI_REF, GSN_CLOSE_COMREQ, signal,
+ CloseComReqConf::SignalLength, JBA);
+
+}//Qmgr::sendCloseComReq()
+
+void
+Qmgr::sendPrepFailReqRef(Signal* signal,
+ Uint32 dstBlockRef,
+ GlobalSignalNumber gsn,
+ Uint32 blockRef,
+ Uint32 failNo,
+ Uint32 noOfNodes,
+ const NodeId theNodes[]){
+
+ PrepFailReqRef * const prepFail = (PrepFailReqRef *)&signal->theData[0];
+ prepFail->xxxBlockRef = blockRef;
+ prepFail->failNo = failNo;
+ prepFail->noOfNodes = noOfNodes;
+
+ NodeBitmask::clear(prepFail->theNodes);
+
+ for(Uint32 i = 0; i<noOfNodes; i++){
+ const NodeId nodeId = theNodes[i];
+ NodeBitmask::set(prepFail->theNodes, nodeId);
+ }
+
+ sendSignal(dstBlockRef, gsn, signal, PrepFailReqRef::SignalLength, JBA);
+}
+
+
+/**--------------------------------------------------------------------------
+ * SEND PREPARE FAIL REQUEST FROM PRESIDENT.
+ *---------------------------------------------------------------------------*/
+void Qmgr::sendPrepFailReq(Signal* signal, Uint16 aNode)
+{
+ NodeRecPtr sendNodePtr;
+ sendNodePtr.i = aNode;
+ ptrCheckGuard(sendNodePtr, MAX_NDB_NODES, nodeRec);
+ sendNodePtr.p->sendPrepFailReqStatus = Q_ACTIVE;
+
+ sendPrepFailReqRef(signal,
+ sendNodePtr.p->blockRef,
+ GSN_PREP_FAILREQ,
+ reference(),
+ cfailureNr,
+ cnoFailedNodes,
+ cfailedNodes);
+}//Qmgr::sendPrepFailReq()
+
+/**
+ * Arbitration module. Rest of QMGR calls us only via
+ * the "handle" routines.
+ */
+
+/**
+ * Config signals are logically part of CM_INIT.
+ */
+void
+Qmgr::execARBIT_CFG(Signal* signal)
+{
+ jamEntry();
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ unsigned rank = sd->code;
+ ndbrequire(1 <= rank && rank <= 2);
+ arbitRec.apiMask[0].bitOR(sd->mask);
+ arbitRec.apiMask[rank] = sd->mask;
+}
+
+/**
+ * ContinueB delay (0=JBA 1=JBB)
+ */
+Uint32 Qmgr::getArbitDelay()
+{
+ switch (arbitRec.state) {
+ case ARBIT_NULL:
+ jam();
+ break;
+ case ARBIT_INIT:
+ jam();
+ case ARBIT_FIND:
+ jam();
+ case ARBIT_PREP1:
+ jam();
+ case ARBIT_PREP2:
+ jam();
+ case ARBIT_START:
+ jam();
+ return 100;
+ case ARBIT_RUN:
+ jam();
+ return 1000;
+ case ARBIT_CHOOSE:
+ jam();
+ return 10;
+ case ARBIT_CRASH: // if we could wait
+ jam();
+ return 100;
+ }
+ ndbrequire(false);
+ return (Uint32)-1;
+}
+
+/**
+ * Time to wait for reply. There is only 1 config parameter
+ * (timeout for CHOOSE). XXX The rest are guesses.
+ */
+Uint32 Qmgr::getArbitTimeout()
+{
+ switch (arbitRec.state) {
+ case ARBIT_NULL:
+ jam();
+ break;
+ case ARBIT_INIT: // not used
+ jam();
+ case ARBIT_FIND: // not used
+ jam();
+ return 1000;
+ case ARBIT_PREP1:
+ jam();
+ case ARBIT_PREP2:
+ jam();
+ return 1000 + cnoOfNodes * hb_send_timer.getDelay();
+ case ARBIT_START:
+ jam();
+ return 1000 + arbitRec.timeout;
+ case ARBIT_RUN: // not used (yet)
+ jam();
+ return 1000;
+ case ARBIT_CHOOSE:
+ jam();
+ return arbitRec.timeout;
+ case ARBIT_CRASH: // if we could wait
+ jam();
+ return 100;
+ }
+ ndbrequire(false);
+ return (Uint32)-1;
+}
+
+/**
+ * Start arbitration thread when we are president and database
+ * is opened for the first time.
+ *
+ * XXX Do arbitration check just like on node failure. Since
+ * there is no arbitrator yet, must win on counts alone.
+ */
+void
+Qmgr::handleArbitStart(Signal* signal)
+{
+ jam();
+ ndbrequire(cpresident == getOwnNodeId());
+ ndbrequire(arbitRec.state == ARBIT_NULL);
+ arbitRec.state = ARBIT_INIT;
+ arbitRec.newstate = true;
+ startArbitThread(signal);
+}
+
+/**
+ * Handle API node failure. Called also by non-president nodes.
+ * If we are president go back to INIT state, otherwise to NULL.
+ * Start new thread to save time.
+ */
+void
+Qmgr::handleArbitApiFail(Signal* signal, Uint16 nodeId)
+{
+ if (arbitRec.node != nodeId) {
+ jam();
+ return;
+ }
+ reportArbitEvent(signal, EventReport::ArbitState);
+ arbitRec.node = 0;
+ switch (arbitRec.state) {
+ case ARBIT_NULL: // should not happen
+ jam();
+ case ARBIT_INIT:
+ jam();
+ case ARBIT_FIND:
+ jam();
+ break;
+ case ARBIT_PREP1: // start from beginning
+ jam();
+ case ARBIT_PREP2:
+ jam();
+ case ARBIT_START:
+ jam();
+ case ARBIT_RUN:
+ if (cpresident == getOwnNodeId()) {
+ jam();
+ arbitRec.state = ARBIT_INIT;
+ arbitRec.newstate = true;
+ startArbitThread(signal);
+ } else {
+ jam();
+ arbitRec.state = ARBIT_NULL;
+ }
+ break;
+ case ARBIT_CHOOSE: // XXX too late
+ jam();
+ case ARBIT_CRASH:
+ jam();
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+}
+
+/**
+ * Handle NDB node add. Ignore if arbitration thread not yet
+ * started. If PREP is not ready, go back to INIT. Otherwise
+ * the new node gets arbitrator and ticket once we reach RUN state.
+ * Start new thread to save time.
+ */
+void
+Qmgr::handleArbitNdbAdd(Signal* signal, Uint16 nodeId)
+{
+ jam();
+ ndbrequire(cpresident == getOwnNodeId());
+ switch (arbitRec.state) {
+ case ARBIT_NULL: // before db opened
+ jam();
+ break;
+ case ARBIT_INIT: // start from beginning
+ jam();
+ case ARBIT_FIND:
+ jam();
+ case ARBIT_PREP1:
+ jam();
+ case ARBIT_PREP2:
+ jam();
+ arbitRec.state = ARBIT_INIT;
+ arbitRec.newstate = true;
+ startArbitThread(signal);
+ break;
+ case ARBIT_START: // process in RUN state
+ jam();
+ case ARBIT_RUN:
+ jam();
+ arbitRec.newMask.set(nodeId);
+ break;
+ case ARBIT_CHOOSE: // XXX too late
+ jam();
+ case ARBIT_CRASH:
+ jam();
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+}
+
+/**
+ * Check if current nodeset can survive. The decision is
+ * based on node count, node groups, and on external arbitrator
+ * (if we have one). Always starts a new thread because
+ * 1) CHOOSE cannot wait 2) if we are new president we need
+ * a thread 3) if we are old president it does no harm.
+ */
+void
+Qmgr::handleArbitCheck(Signal* signal)
+{
+ jam();
+ ndbrequire(cpresident == getOwnNodeId());
+ NodeBitmask ndbMask;
+ computeArbitNdbMask(ndbMask);
+ if (2 * ndbMask.count() < cnoOfNodes) {
+ jam();
+ arbitRec.code = ArbitCode::LoseNodes;
+ } else {
+ jam();
+ CheckNodeGroups* sd = (CheckNodeGroups*)&signal->theData[0];
+ sd->blockRef = reference();
+ sd->requestType = CheckNodeGroups::Direct | CheckNodeGroups::ArbitCheck;
+ sd->mask = ndbMask;
+ EXECUTE_DIRECT(DBDIH, GSN_CHECKNODEGROUPSREQ, signal,
+ CheckNodeGroups::SignalLength);
+ jamEntry();
+ switch (sd->output) {
+ case CheckNodeGroups::Win:
+ jam();
+ arbitRec.code = ArbitCode::WinGroups;
+ break;
+ case CheckNodeGroups::Lose:
+ jam();
+ arbitRec.code = ArbitCode::LoseGroups;
+ break;
+ case CheckNodeGroups::Partitioning:
+ jam();
+ arbitRec.code = ArbitCode::Partitioning;
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+ }
+ switch (arbitRec.code) {
+ case ArbitCode::LoseNodes:
+ jam();
+ goto crashme;
+ case ArbitCode::WinGroups:
+ jam();
+ if (arbitRec.state == ARBIT_RUN) {
+ jam();
+ break;
+ }
+ arbitRec.state = ARBIT_INIT;
+ arbitRec.newstate = true;
+ break;
+ case ArbitCode::LoseGroups:
+ jam();
+ goto crashme;
+ case ArbitCode::Partitioning:
+ if (arbitRec.state == ARBIT_RUN) {
+ jam();
+ arbitRec.state = ARBIT_CHOOSE;
+ arbitRec.newstate = true;
+ break;
+ }
+ if (arbitRec.apiMask[0].count() != 0) {
+ jam();
+ arbitRec.code = ArbitCode::LoseNorun;
+ } else {
+ jam();
+ arbitRec.code = ArbitCode::LoseNocfg;
+ }
+ goto crashme;
+ default:
+ crashme:
+ jam();
+ arbitRec.state = ARBIT_CRASH;
+ arbitRec.newstate = true;
+ break;
+ }
+ reportArbitEvent(signal, EventReport::ArbitResult);
+ switch (arbitRec.state) {
+ default:
+ jam();
+ arbitRec.newMask.bitAND(ndbMask); // delete failed nodes
+ arbitRec.recvMask.bitAND(ndbMask);
+ sendCommitFailReq(signal); // start commit of failed nodes
+ break;
+ case ARBIT_CHOOSE:
+ jam();
+ case ARBIT_CRASH:
+ jam();
+ break;
+ }
+ startArbitThread(signal);
+}
+
+/**
+ * Start a new continueB thread. The thread id is incremented
+ * so that any old thread will exit.
+ */
+void
+Qmgr::startArbitThread(Signal* signal)
+{
+ jam();
+ ndbrequire(cpresident == getOwnNodeId());
+ arbitRec.code = ArbitCode::ThreadStart;
+ reportArbitEvent(signal, EventReport::ArbitState);
+ signal->theData[1] = ++arbitRec.thread;
+ runArbitThread(signal);
+}
+
+/**
+ * Handle arbitration thread. The initial thread normally ends
+ * up in RUN state. New thread can be started to save time.
+ */
+void
+Qmgr::runArbitThread(Signal* signal)
+{
+#ifdef DEBUG_ARBIT
+ NodeBitmask ndbMask;
+ computeArbitNdbMask(ndbMask);
+ ndbout << "arbit thread:";
+ ndbout << " state=" << arbitRec.state;
+ ndbout << " newstate=" << arbitRec.newstate;
+ ndbout << " thread=" << arbitRec.thread;
+ ndbout << " node=" << arbitRec.node;
+ ndbout << " ticket=" << arbitRec.ticket.getText();
+ ndbout << " ndbmask=" << ndbMask.getText();
+ ndbout << " sendcount=" << arbitRec.sendCount;
+ ndbout << " recvcount=" << arbitRec.recvCount;
+ ndbout << " recvmask=" << arbitRec.recvMask.getText();
+ ndbout << " code=" << arbitRec.code;
+ ndbout << endl;
+#endif
+ if (signal->theData[1] != arbitRec.thread) {
+ jam();
+ return; // old thread dies
+ }
+ switch (arbitRec.state) {
+ case ARBIT_INIT: // main thread
+ jam();
+ stateArbitInit(signal);
+ break;
+ case ARBIT_FIND:
+ jam();
+ stateArbitFind(signal);
+ break;
+ case ARBIT_PREP1:
+ jam();
+ case ARBIT_PREP2:
+ jam();
+ stateArbitPrep(signal);
+ break;
+ case ARBIT_START:
+ jam();
+ stateArbitStart(signal);
+ break;
+ case ARBIT_RUN:
+ jam();
+ stateArbitRun(signal);
+ break;
+ case ARBIT_CHOOSE: // partitition thread
+ jam();
+ stateArbitChoose(signal);
+ break;
+ case ARBIT_CRASH:
+ jam();
+ stateArbitCrash(signal);
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+ signal->theData[0] = ZARBIT_HANDLING;
+ signal->theData[1] = arbitRec.thread;
+ signal->theData[2] = arbitRec.state; // just for signal log
+ Uint32 delay = getArbitDelay();
+ if (delay == 0) {
+ jam();
+ sendSignal(QMGR_REF, GSN_CONTINUEB, signal, 3, JBA);
+ } else if (delay == 1) {
+ jam();
+ sendSignal(QMGR_REF, GSN_CONTINUEB, signal, 3, JBB);
+ } else {
+ jam();
+ sendSignalWithDelay(QMGR_REF, GSN_CONTINUEB, signal, delay, 3);
+ }//if
+}
+
+/**
+ * Handle INIT state. Generate next ticket. Switch to FIND
+ * state without delay.
+ */
+void
+Qmgr::stateArbitInit(Signal* signal)
+{
+ if (arbitRec.newstate) {
+ jam();
+ CRASH_INSERTION((Uint32)910 + arbitRec.state);
+
+ arbitRec.node = 0;
+ arbitRec.ticket.update();
+ arbitRec.newMask.clear();
+ arbitRec.code = 0;
+ arbitRec.newstate = false;
+ }
+ arbitRec.state = ARBIT_FIND;
+ arbitRec.newstate = true;
+ stateArbitFind(signal);
+}
+
+/**
+ * Handle FIND state. Find first arbitrator which is alive
+ * and invoke PREP state without delay. If none are found,
+ * loop in FIND state. This is forever if no arbitrators
+ * are configured (not the normal case).
+ *
+ * XXX Add adaptive behaviour to avoid getting stuck on API
+ * nodes which are alive but do not respond or die too soon.
+ */
+void
+Qmgr::stateArbitFind(Signal* signal)
+{
+ if (arbitRec.newstate) {
+ jam();
+ CRASH_INSERTION((Uint32)910 + arbitRec.state);
+
+ arbitRec.code = 0;
+ arbitRec.newstate = false;
+ }
+ NodeRecPtr aPtr;
+ for (unsigned rank = 1; rank <= 2; rank++) {
+ jam();
+ aPtr.i = 0;
+ const unsigned stop = NodeBitmask::NotFound;
+ while ((aPtr.i = arbitRec.apiMask[rank].find(aPtr.i + 1)) != stop) {
+ jam();
+ ptrAss(aPtr, nodeRec);
+ if (aPtr.p->phase != ZAPI_ACTIVE)
+ continue;
+ arbitRec.node = aPtr.i;
+ arbitRec.state = ARBIT_PREP1;
+ arbitRec.newstate = true;
+ stateArbitPrep(signal);
+ return;
+ }
+ }
+}
+
+/**
+ * Handle PREP states. First round nulls any existing tickets.
+ * Second round sends new ticket. When all confirms have been
+ * received invoke START state immediately.
+ */
+void
+Qmgr::stateArbitPrep(Signal* signal)
+{
+ if (arbitRec.newstate) {
+ jam();
+ CRASH_INSERTION((Uint32)910 + arbitRec.state);
+
+ arbitRec.sendCount = 0; // send all at once
+ computeArbitNdbMask(arbitRec.recvMask); // to send and recv
+ arbitRec.recvMask.clear(getOwnNodeId());
+ arbitRec.code = 0;
+ arbitRec.newstate = false;
+ }
+ if (! arbitRec.sendCount) {
+ jam();
+ NodeRecPtr aPtr;
+ aPtr.i = 0;
+ const unsigned stop = NodeBitmask::NotFound;
+ while ((aPtr.i = arbitRec.recvMask.find(aPtr.i + 1)) != stop) {
+ jam();
+ ptrAss(aPtr, nodeRec);
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ sd->sender = getOwnNodeId();
+ if (arbitRec.state == ARBIT_PREP1) {
+ jam();
+ sd->code = ArbitCode::PrepPart1;
+ } else {
+ jam();
+ sd->code = ArbitCode::PrepPart2;
+ }
+ sd->node = arbitRec.node;
+ sd->ticket = arbitRec.ticket;
+ sd->mask.clear();
+ sendSignal(aPtr.p->blockRef, GSN_ARBIT_PREPREQ, signal,
+ ArbitSignalData::SignalLength, JBB);
+ }
+ arbitRec.setTimestamp(); // send time
+ arbitRec.sendCount = 1;
+ return;
+ }
+ if (arbitRec.code != 0) { // error
+ jam();
+ arbitRec.state = ARBIT_INIT;
+ arbitRec.newstate = true;
+ return;
+ }
+ if (arbitRec.recvMask.count() == 0) { // recv all
+ if (arbitRec.state == ARBIT_PREP1) {
+ jam();
+ arbitRec.state = ARBIT_PREP2;
+ arbitRec.newstate = true;
+ } else {
+ jam();
+ arbitRec.state = ARBIT_START;
+ arbitRec.newstate = true;
+ stateArbitStart(signal);
+ }
+ return;
+ }
+ if (arbitRec.getTimediff() > getArbitTimeout()) {
+ jam();
+ arbitRec.state = ARBIT_INIT;
+ arbitRec.newstate = true;
+ return;
+ }
+}
+
+void
+Qmgr::execARBIT_PREPREQ(Signal* signal)
+{
+ jamEntry();
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ if (getOwnNodeId() == cpresident) {
+ jam();
+ return; // wrong state
+ }
+ if (sd->sender != cpresident) {
+ jam();
+ return; // wrong state
+ }
+ NodeRecPtr aPtr;
+ aPtr.i = sd->sender;
+ ptrAss(aPtr, nodeRec);
+ switch (sd->code) {
+ case ArbitCode::PrepPart1: // zero them just to be sure
+ jam();
+ arbitRec.node = 0;
+ arbitRec.ticket.clear();
+ break;
+ case ArbitCode::PrepPart2: // non-president enters RUN state
+ jam();
+ case ArbitCode::PrepAtrun:
+ jam();
+ arbitRec.node = sd->node;
+ arbitRec.ticket = sd->ticket;
+ arbitRec.code = sd->code;
+ reportArbitEvent(signal, EventReport::ArbitState);
+ arbitRec.state = ARBIT_RUN;
+ arbitRec.newstate = true;
+ if (sd->code == ArbitCode::PrepAtrun) {
+ jam();
+ return;
+ }
+ break;
+ default:
+ jam();
+ ndbrequire(false);
+ }
+ sd->sender = getOwnNodeId();
+ sd->code = 0;
+ sendSignal(aPtr.p->blockRef, GSN_ARBIT_PREPCONF, signal,
+ ArbitSignalData::SignalLength, JBB);
+}
+
+void
+Qmgr::execARBIT_PREPCONF(Signal* signal)
+{
+ jamEntry();
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ if (! arbitRec.match(sd)) {
+ jam();
+ return; // stray signal
+ }
+ if (arbitRec.state != ARBIT_PREP1 && arbitRec.state != ARBIT_PREP2) {
+ jam();
+ return; // wrong state
+ }
+ if (! arbitRec.recvMask.get(sd->sender)) {
+ jam();
+ return; // wrong state
+ }
+ arbitRec.recvMask.clear(sd->sender);
+ if (arbitRec.code == 0 && sd->code != 0) {
+ jam();
+ arbitRec.code = sd->code;
+ }//if
+}
+
+void
+Qmgr::execARBIT_PREPREF(Signal* signal)
+{
+ jamEntry();
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ if (sd->code == 0) {
+ jam();
+ sd->code = ArbitCode::ErrUnknown;
+ }
+ execARBIT_PREPCONF(signal);
+}
+
+/**
+ * Handle START state. On first call send start request to
+ * the chosen arbitrator. Then wait for a CONF.
+ */
+void
+Qmgr::stateArbitStart(Signal* signal)
+{
+ if (arbitRec.newstate) {
+ jam();
+ CRASH_INSERTION((Uint32)910 + arbitRec.state);
+
+ arbitRec.sendCount = 0;
+ arbitRec.recvCount = 0;
+ arbitRec.code = 0;
+ arbitRec.newstate = false;
+ }
+ if (! arbitRec.sendCount) {
+ jam();
+ BlockReference blockRef = calcApiClusterMgrBlockRef(arbitRec.node);
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ sd->sender = getOwnNodeId();
+ sd->code = 0;
+ sd->node = arbitRec.node;
+ sd->ticket = arbitRec.ticket;
+ sd->mask.clear();
+ sendSignal(blockRef, GSN_ARBIT_STARTREQ, signal,
+ ArbitSignalData::SignalLength, JBB);
+ arbitRec.sendCount = 1;
+ arbitRec.setTimestamp(); // send time
+ return;
+ }
+ if (arbitRec.recvCount) {
+ jam();
+ reportArbitEvent(signal, EventReport::ArbitState);
+ if (arbitRec.code == ArbitCode::ApiStart) {
+ jam();
+ arbitRec.state = ARBIT_RUN;
+ arbitRec.newstate = true;
+ return;
+ }
+ arbitRec.state = ARBIT_INIT;
+ arbitRec.newstate = true;
+ return;
+ }
+ if (arbitRec.getTimediff() > getArbitTimeout()) {
+ jam();
+ arbitRec.code = ArbitCode::ErrTimeout;
+ reportArbitEvent(signal, EventReport::ArbitState);
+ arbitRec.state = ARBIT_INIT;
+ arbitRec.newstate = true;
+ return;
+ }
+}
+
+void
+Qmgr::execARBIT_STARTCONF(Signal* signal)
+{
+ jamEntry();
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ if (! arbitRec.match(sd)) {
+ jam();
+ return; // stray signal
+ }
+ if (arbitRec.state != ARBIT_START) {
+ jam();
+ return; // wrong state
+ }
+ if (arbitRec.recvCount) {
+ jam();
+ return; // wrong state
+ }
+ arbitRec.code = sd->code;
+ arbitRec.recvCount = 1;
+}
+
+void
+Qmgr::execARBIT_STARTREF(Signal* signal)
+{
+ jamEntry();
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ if (sd->code == 0) {
+ jam();
+ sd->code = ArbitCode::ErrUnknown;
+ }
+ execARBIT_STARTCONF(signal);
+}
+
+/**
+ * Handle RUN state. Send ticket to any new nodes which have
+ * appeared after PREP state. We don't care about a CONF.
+ */
+void
+Qmgr::stateArbitRun(Signal* signal)
+{
+ if (arbitRec.newstate) {
+ jam();
+ CRASH_INSERTION((Uint32)910 + arbitRec.state);
+
+ arbitRec.code = 0;
+ arbitRec.newstate = false;
+ }
+ NodeRecPtr aPtr;
+ aPtr.i = 0;
+ const unsigned stop = NodeBitmask::NotFound;
+ while ((aPtr.i = arbitRec.newMask.find(aPtr.i + 1)) != stop) {
+ jam();
+ arbitRec.newMask.clear(aPtr.i);
+ ptrAss(aPtr, nodeRec);
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ sd->sender = getOwnNodeId();
+ sd->code = ArbitCode::PrepAtrun;
+ sd->node = arbitRec.node;
+ sd->ticket = arbitRec.ticket;
+ sd->mask.clear();
+ sendSignal(aPtr.p->blockRef, GSN_ARBIT_PREPREQ, signal,
+ ArbitSignalData::SignalLength, JBB);
+ }
+}
+
+/**
+ * Handle CHOOSE state. Entered only from RUN state when
+ * there is a possible network partitioning. Send CHOOSE to
+ * the arbitrator. On win switch to INIT state because a new
+ * ticket must be created.
+ */
+void
+Qmgr::stateArbitChoose(Signal* signal)
+{
+ if (arbitRec.newstate) {
+ jam();
+ CRASH_INSERTION((Uint32)910 + arbitRec.state);
+
+ arbitRec.sendCount = 0;
+ arbitRec.recvCount = 0;
+ arbitRec.code = 0;
+ arbitRec.newstate = false;
+ }
+ if (! arbitRec.sendCount) {
+ jam();
+ BlockReference blockRef = calcApiClusterMgrBlockRef(arbitRec.node);
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ sd->sender = getOwnNodeId();
+ sd->code = 0;
+ sd->node = arbitRec.node;
+ sd->ticket = arbitRec.ticket;
+ computeArbitNdbMask(sd->mask);
+ sendSignal(blockRef, GSN_ARBIT_CHOOSEREQ, signal,
+ ArbitSignalData::SignalLength, JBA);
+ arbitRec.sendCount = 1;
+ arbitRec.setTimestamp(); // send time
+ return;
+ }
+ if (arbitRec.recvCount) {
+ jam();
+ reportArbitEvent(signal, EventReport::ArbitResult);
+ if (arbitRec.code == ArbitCode::WinChoose) {
+ jam();
+ sendCommitFailReq(signal); // start commit of failed nodes
+ arbitRec.state = ARBIT_INIT;
+ arbitRec.newstate = true;
+ return;
+ }
+ arbitRec.state = ARBIT_CRASH;
+ arbitRec.newstate = true;
+ stateArbitCrash(signal); // do it at once
+ return;
+ }
+ if (arbitRec.getTimediff() > getArbitTimeout()) {
+ jam();
+ arbitRec.code = ArbitCode::ErrTimeout;
+ reportArbitEvent(signal, EventReport::ArbitState);
+ arbitRec.state = ARBIT_CRASH;
+ arbitRec.newstate = true;
+ stateArbitCrash(signal); // do it at once
+ return;
+ }
+}
+
+void
+Qmgr::execARBIT_CHOOSECONF(Signal* signal)
+{
+ jamEntry();
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ if (!arbitRec.match(sd)) {
+ jam();
+ return; // stray signal
+ }
+ if (arbitRec.state != ARBIT_CHOOSE) {
+ jam();
+ return; // wrong state
+ }
+ if (arbitRec.recvCount) {
+ jam();
+ return; // wrong state
+ }
+ arbitRec.recvCount = 1;
+ arbitRec.code = sd->code;
+}
+
+void
+Qmgr::execARBIT_CHOOSEREF(Signal* signal)
+{
+ jamEntry();
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ if (sd->code == 0) {
+ jam();
+ sd->code = ArbitCode::ErrUnknown;
+ }
+ execARBIT_CHOOSECONF(signal);
+}
+
+/**
+ * Handle CRASH state. We must crash immediately. But it
+ * would be nice to wait until event reports have been sent.
+ * XXX tell other nodes in our party to crash too.
+ */
+void
+Qmgr::stateArbitCrash(Signal* signal)
+{
+ jam();
+ if (arbitRec.newstate) {
+ jam();
+ CRASH_INSERTION((Uint32)910 + arbitRec.state);
+
+ arbitRec.setTimestamp();
+ arbitRec.code = 0;
+ arbitRec.newstate = false;
+ }
+#if 0
+ if (! (arbitRec.getTimediff() > getArbitTimeout()))
+ return;
+#endif
+ progError(__LINE__, ERR_ARBIT_SHUTDOWN, "Arbitrator decided to shutdown this node");
+}
+
+/**
+ * Arbitrator may inform us that it will exit. This lets us
+ * start looking sooner for a new one. Handle it like API node
+ * failure.
+ */
+void
+Qmgr::execARBIT_STOPREP(Signal* signal)
+{
+ jamEntry();
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ if (! arbitRec.match(sd)) {
+ jam();
+ return; // stray signal
+ }
+ arbitRec.code = ArbitCode::ApiExit;
+ handleArbitApiFail(signal, arbitRec.node);
+}
+
+void
+Qmgr::computeArbitNdbMask(NodeBitmask& aMask)
+{
+ NodeRecPtr aPtr;
+ aMask.clear();
+ for (aPtr.i = 1; aPtr.i < MAX_NDB_NODES; aPtr.i++) {
+ jam();
+ ptrAss(aPtr, nodeRec);
+ if (getNodeInfo(aPtr.i).getType() == NodeInfo::DB && aPtr.p->phase == ZRUNNING){
+ jam();
+ aMask.set(aPtr.i);
+ }
+ }
+}
+
+/**
+ * Report arbitration event. We use arbitration signal format
+ * where sender (word 0) is event type.
+ */
+void
+Qmgr::reportArbitEvent(Signal* signal, EventReport::EventType type)
+{
+ ArbitSignalData* sd = (ArbitSignalData*)&signal->theData[0];
+ sd->sender = type;
+ sd->code = arbitRec.code | (arbitRec.state << 16);
+ sd->node = arbitRec.node;
+ sd->ticket = arbitRec.ticket;
+ sd->mask.clear();
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal,
+ ArbitSignalData::SignalLength, JBB);
+}
+
+// end of arbitration module
+
+void
+Qmgr::execDUMP_STATE_ORD(Signal* signal)
+{
+ switch (signal->theData[0]) {
+ case 1:
+ infoEvent("creadyDistCom = %d, cpresident = %d, cpresidentBusy = %d\n",
+ creadyDistCom, cpresident, cpresidentBusy);
+ infoEvent("cacceptRegreq = %d, ccm_infoconfCounter = %d\n",
+ cacceptRegreq, ccm_infoconfCounter);
+ infoEvent("cstartNo = %d, cstartNode = %d, cwaitC..phase1 = %d\n",
+ cstartNo, cstartNode, cwaitContinuebPhase1);
+ infoEvent("cwaitC..phase2 = %d, cpresidentAlive = %d, cpresidentCand = %d\n"
+ ,cwaitContinuebPhase2, cpresidentAlive, cpresidentCandidate);
+ infoEvent("ctoStatus = %d\n", ctoStatus);
+ for(Uint32 i = 1; i<MAX_NDB_NODES; i++){
+ if(getNodeInfo(i).getType() == NodeInfo::DB){
+ NodeRecPtr nodePtr;
+ nodePtr.i = i;
+ ptrCheckGuard(nodePtr, MAX_NDB_NODES, nodeRec);
+ char buf[100];
+ switch(nodePtr.p->phase){
+ case ZINIT:
+ sprintf(buf, "Node %d: ZINIT(%d)", i, nodePtr.p->phase);
+ break;
+ case ZBLOCKED:
+ sprintf(buf, "Node %d: ZBLOCKED(%d)", i, nodePtr.p->phase);
+ break;
+ case ZWAITING:
+ sprintf(buf, "Node %d: ZWAITING(%d)", i, nodePtr.p->phase);
+ break;
+ case ZWAIT_PRESIDENT:
+ sprintf(buf, "Node %d: ZWAIT_PRESIDENT(%d)", i, nodePtr.p->phase);
+ break;
+ case ZRUNNING:
+ sprintf(buf, "Node %d: ZRUNNING(%d)", i, nodePtr.p->phase);
+ break;
+ case ZFAIL_CLOSING:
+ sprintf(buf, "Node %d: ZFAIL_CLOSING(%d)", i, nodePtr.p->phase);
+ break;
+ case ZAPI_INACTIVE:
+ sprintf(buf, "Node %d: ZAPI_INACTIVE(%d)", i, nodePtr.p->phase);
+ break;
+ case ZAPI_ACTIVE:
+ sprintf(buf, "Node %d: ZAPI_ACTIVE(%d)", i, nodePtr.p->phase);
+ break;
+ case ZPREPARE_FAIL:
+ sprintf(buf, "Node %d: ZPREPARE_FAIL(%d)", i, nodePtr.p->phase);
+ break;
+ default:
+ sprintf(buf, "Node %d: <UNKNOWN>(%d)", i, nodePtr.p->phase);
+ break;
+ }
+ infoEvent(buf);
+ }
+ }
+ default:
+ ;
+ }//switch
+}//Qmgr::execDUMP_STATE_ORD()
+
+void Qmgr::execSET_VAR_REQ(Signal* signal)
+{
+ SetVarReq* const setVarReq = (SetVarReq*)&signal->theData[0];
+ ConfigParamId var = setVarReq->variable();
+ UintR val = setVarReq->value();
+
+ switch (var) {
+ case HeartbeatIntervalDbDb:
+ setHbDelay(val/10);
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ case HeartbeatIntervalDbApi:
+ setHbApiDelay(val/10);
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ case ArbitTimeout:
+ setArbitTimeout(val);
+ sendSignal(CMVMI_REF, GSN_SET_VAR_CONF, signal, 1, JBB);
+ break;
+
+ default:
+ sendSignal(CMVMI_REF, GSN_SET_VAR_REF, signal, 1, JBB);
+ }// switch
+}//execSET_VAR_REQ()
diff --git a/ndb/src/kernel/blocks/qmgr/timer.hpp b/ndb/src/kernel/blocks/qmgr/timer.hpp
new file mode 100644
index 00000000000..9c35a23766c
--- /dev/null
+++ b/ndb/src/kernel/blocks/qmgr/timer.hpp
@@ -0,0 +1,72 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * @class Timer
+ * @brief A timer class that can't be fooled by NTP:ing the system clock to old time
+ */
+class Timer {
+public:
+ Timer() {
+ m_delay = 10;
+ };
+
+ Timer(NDB_TICKS delay_time) {
+ m_delay = delay_time;
+ }
+
+ /**
+ * Set/Get alarm time of timer
+ */
+ inline void setDelay(NDB_TICKS delay_time) { m_delay = delay_time; }
+ inline NDB_TICKS getDelay() { return m_delay; }
+
+ /**
+ * Start timer
+ */
+ inline void reset() {
+ m_current_time = NdbTick_CurrentMillisecond();
+ m_alarm_time = m_current_time + m_delay;
+ }
+
+ /**
+ * Check for alarm
+ */
+ inline bool check() { return check(NdbTick_CurrentMillisecond()); }
+
+ inline bool check(NDB_TICKS check_time) {
+ /**
+ * Standard alarm check
+ */
+ if (check_time > m_alarm_time) return true;
+
+ /**
+ * Time progressing, but it is not alarm time yet
+ */
+ if (check_time >= m_current_time) return false;
+
+ /**
+ * Time has moved backwards
+ */
+ reset();
+ return false;
+ }
+
+private:
+ NDB_TICKS m_current_time;
+ NDB_TICKS m_alarm_time;
+ NDB_TICKS m_delay;
+};
diff --git a/ndb/src/kernel/blocks/suma/Makefile b/ndb/src/kernel/blocks/suma/Makefile
new file mode 100644
index 00000000000..20014c94670
--- /dev/null
+++ b/ndb/src/kernel/blocks/suma/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := suma
+
+SOURCES = Suma.cpp SumaInit.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/kernel/blocks/suma/Suma.cpp b/ndb/src/kernel/blocks/suma/Suma.cpp
new file mode 100644
index 00000000000..236333f58e4
--- /dev/null
+++ b/ndb/src/kernel/blocks/suma/Suma.cpp
@@ -0,0 +1,3971 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Suma.hpp"
+
+#include <ndb_version.h>
+
+#include <NdbTCP.h>
+#include <Bitmask.hpp>
+#include <SimpleProperties.hpp>
+
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/ReadNodesConf.hpp>
+
+#include <signaldata/ListTables.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/GetTableId.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <signaldata/ScanFrag.hpp>
+#include <signaldata/TransIdAI.hpp>
+#include <signaldata/CreateTrig.hpp>
+#include <signaldata/AlterTrig.hpp>
+#include <signaldata/DropTrig.hpp>
+#include <signaldata/FireTrigOrd.hpp>
+#include <signaldata/TrigAttrInfo.hpp>
+#include <signaldata/CheckNodeGroups.hpp>
+#include <signaldata/GCPSave.hpp>
+#include <GrepError.hpp>
+
+#include <DebuggerNames.hpp>
+
+//#define HANDOVER_DEBUG
+//#define NODEFAIL_DEBUG
+//#define NODEFAIL_DEBUG2
+//#define DEBUG_SUMA_SEQUENCE
+//#define EVENT_DEBUG
+//#define EVENT_PH3_DEBUG
+//#define EVENT_DEBUG2
+
+/**
+ * @todo:
+ * SUMA crashes if an index is created at the same time as
+ * global replication. Very easy to reproduce using testIndex.
+ * Note: This only happens occasionally, but is quite easy to reprod.
+ */
+
+Uint32 g_subPtrI = RNIL;
+static const Uint32 SUMA_SEQUENCE = 0xBABEBABE;
+
+
+/**************************************************************
+ *
+ * Start of suma
+ *
+ */
+
+#define PRINT_ONLY 0
+static Uint32 g_TypeOfStart = NodeState::ST_ILLEGAL_TYPE;
+
+void
+Suma::getNodeGroupMembers(Signal* signal) {
+ jam();
+ /**
+ * Ask DIH for nodeGroupMembers
+ */
+ CheckNodeGroups * sd = (CheckNodeGroups*)signal->getDataPtrSend();
+ sd->blockRef = reference();
+ sd->requestType =
+ CheckNodeGroups::Direct |
+ CheckNodeGroups::GetNodeGroupMembers;
+ sd->nodeId = getOwnNodeId();
+ EXECUTE_DIRECT(DBDIH, GSN_CHECKNODEGROUPSREQ, signal,
+ CheckNodeGroups::SignalLength);
+ jamEntry();
+
+ c_nodeGroup = sd->output;
+ c_noNodesInGroup = 0;
+ for (int i = 0; i < MAX_NDB_NODES; i++) {
+ if (sd->mask.get(i)) {
+ if (i == getOwnNodeId()) c_idInNodeGroup = c_noNodesInGroup;
+ c_nodesInGroup[c_noNodesInGroup] = i;
+ c_noNodesInGroup++;
+ }
+ }
+
+ // ndbout_c("c_noNodesInGroup=%d", c_noNodesInGroup);
+ ndbrequire(c_noNodesInGroup >= 0); // at least 1 node in the nodegroup
+
+#ifdef NODEFAIL_DEBUG
+ for (Uint32 i = 0; i < c_noNodesInGroup; i++) {
+ ndbout_c ("Suma: NodeGroup %u, me %u, me in group %u, member[%u] %u",
+ c_nodeGroup, getOwnNodeId(), c_idInNodeGroup,
+ i, c_nodesInGroup[i]);
+ }
+#endif
+}
+
+void
+Suma::execSTTOR(Signal* signal) {
+ jamEntry();
+
+ const Uint32 startphase = signal->theData[1];
+ const Uint32 typeOfStart = signal->theData[7];
+
+#ifdef NODEFAIL_DEBUG
+ ndbout_c ("SUMA::execSTTOR startphase = %u, typeOfStart = %u",
+ startphase, typeOfStart);
+
+#endif
+
+ if(startphase == 1){
+ jam();
+ c_restartLock = true;
+ }
+
+ if(startphase == 3){
+ jam();
+ g_TypeOfStart = typeOfStart;
+ signal->theData[0] = reference();
+ sendSignal(NDBCNTR_REF, GSN_READ_NODESREQ, signal, 1, JBB);
+
+#if 0
+
+ /**
+ * Debug
+ */
+
+
+ SubscriptionPtr subPtr;
+ Ptr<SyncRecord> syncPtr;
+ ndbrequire(c_subscriptions.seize(subPtr));
+ ndbrequire(c_syncPool.seize(syncPtr));
+
+
+ ndbout_c("Suma: subPtr.i = %d syncPtr.i = %d", subPtr.i, syncPtr.i);
+
+ subPtr.p->m_syncPtrI = syncPtr.i;
+ subPtr.p->m_subscriptionType = SubCreateReq::DatabaseSnapshot;
+ syncPtr.p->m_subscriptionPtrI = subPtr.i;
+ syncPtr.p->ptrI = syncPtr.i;
+ g_subPtrI = subPtr.i;
+ // sendSTTORRY(signal);
+#endif
+ return;
+ }
+
+ if(startphase == 5) {
+ getNodeGroupMembers(signal);
+ if (g_TypeOfStart == NodeState::ST_NODE_RESTART) {
+ jam();
+ for (Uint32 i = 0; i < c_noNodesInGroup; i++) {
+ Uint32 ref = calcSumaBlockRef(c_nodesInGroup[i]);
+ if (ref != reference())
+ sendSignal(ref, GSN_SUMA_START_ME, signal,
+ 1 /*SumaStartMe::SignalLength*/, JBB);
+ }
+ }
+ }
+
+ if(startphase == 7) {
+ c_restartLock = false; // may be set false earlier with HANDOVER_REQ
+
+ if (g_TypeOfStart != NodeState::ST_NODE_RESTART) {
+ for( int i = 0; i < NO_OF_BUCKETS; i++) {
+ if (getResponsibleSumaNodeId(i) == refToNode(reference())) {
+ // I'm running this bucket
+#ifdef EVENT_DEBUG
+ ndbout_c("bucket %u set to true", i);
+#endif
+ c_buckets[i].active = true;
+ }
+ }
+ }
+
+ if(g_TypeOfStart == NodeState::ST_INITIAL_START &&
+ c_masterNodeId == getOwnNodeId()) {
+ jam();
+ createSequence(signal);
+ return;
+ }//if
+ }//if
+
+
+ sendSTTORRY(signal);
+
+ return;
+}
+
+void
+Suma::createSequence(Signal* signal)
+{
+ jam();
+
+ UtilSequenceReq * req = (UtilSequenceReq*)signal->getDataPtrSend();
+
+ req->senderData = RNIL;
+ req->sequenceId = SUMA_SEQUENCE;
+ req->requestType = UtilSequenceReq::Create;
+#ifdef DEBUG_SUMA_SEQUENCE
+ ndbout_c("SUMA: Create sequence");
+#endif
+ sendSignal(DBUTIL_REF, GSN_UTIL_SEQUENCE_REQ,
+ signal, UtilSequenceReq::SignalLength, JBB);
+ // execUTIL_SEQUENCE_CONF will call createSequenceReply()
+}
+
+void
+Suma::createSequenceReply(Signal* signal,
+ UtilSequenceConf * conf,
+ UtilSequenceRef * ref)
+{
+ jam();
+
+ if (ref != NULL)
+ ndbrequire(false);
+
+ sendSTTORRY(signal);
+}
+
+void
+Suma::execREAD_NODESCONF(Signal* signal){
+ jamEntry();
+ ReadNodesConf * const conf = (ReadNodesConf *)signal->getDataPtr();
+
+ c_aliveNodes.clear();
+ c_preparingNodes.clear();
+
+ Uint32 count = 0;
+ for(Uint32 i = 0; i < MAX_NDB_NODES; i++){
+ if(NodeBitmask::get(conf->allNodes, i)){
+ jam();
+
+ count++;
+
+ NodePtr node;
+ ndbrequire(c_nodes.seize(node));
+
+ node.p->nodeId = i;
+ if(NodeBitmask::get(conf->inactiveNodes, i)){
+ jam();
+ node.p->alive = 0;
+ } else {
+ jam();
+ node.p->alive = 1;
+ c_aliveNodes.set(i);
+ }
+ } else
+ jam();
+ }
+ c_masterNodeId = conf->masterNodeId;
+ ndbrequire(count == conf->noOfNodes);
+
+ sendSTTORRY(signal);
+}
+
+void
+Suma::sendSTTORRY(Signal* signal){
+ signal->theData[0] = 0;
+ signal->theData[3] = 1;
+ signal->theData[4] = 3;
+ signal->theData[5] = 5;
+ signal->theData[6] = 7;
+ signal->theData[7] = 255; // No more start phases from missra
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 8, JBB);
+}
+
+void
+Suma::execNDB_STTOR(Signal* signal)
+{
+ jamEntry();
+}
+
+void
+Suma::execCONTINUEB(Signal* signal){
+ jamEntry();
+}
+
+void
+SumaParticipant::execCONTINUEB(Signal* signal)
+{
+ jamEntry();
+}
+
+/*****************************************************************************
+ *
+ * Node state handling
+ *
+ *****************************************************************************/
+
+void Suma::execAPI_FAILREQ(Signal* signal)
+{
+ jamEntry();
+ Uint32 failedApiNode = signal->theData[0];
+ //BlockReference retRef = signal->theData[1];
+
+ c_failedApiNodes.set(failedApiNode);
+ bool found = removeSubscribersOnNode(signal, failedApiNode);
+
+ if(!found){
+ jam();
+ c_failedApiNodes.clear(failedApiNode);
+ }
+}//execAPI_FAILREQ()
+
+bool
+SumaParticipant::removeSubscribersOnNode(Signal *signal, Uint32 nodeId)
+{
+ bool found = false;
+
+ SubscriberPtr i_subbPtr;
+ c_dataSubscribers.first(i_subbPtr);
+ while(!i_subbPtr.isNull()){
+ SubscriberPtr subbPtr = i_subbPtr;
+ c_dataSubscribers.next(i_subbPtr);
+ jam();
+ if (refToNode(subbPtr.p->m_subscriberRef) == nodeId) {
+ jam();
+ c_dataSubscribers.remove(subbPtr);
+ c_removeDataSubscribers.add(subbPtr);
+ found = true;
+ }
+ }
+ if(found){
+ jam();
+ sendSubStopReq(signal);
+ }
+ return found;
+}
+
+void
+SumaParticipant::sendSubStopReq(Signal *signal){
+ static bool remove_lock = false;
+ jam();
+
+ if(remove_lock) {
+ jam();
+ return;
+ }
+ remove_lock = true;
+
+ SubscriberPtr subbPtr;
+ c_removeDataSubscribers.first(subbPtr);
+ if (subbPtr.isNull()){
+ jam();
+#if 0
+ signal->theData[0] = failedApiNode;
+ signal->theData[1] = reference();
+ sendSignal(retRef, GSN_API_FAILCONF, signal, 2, JBB);
+#endif
+ c_failedApiNodes.clear();
+
+ remove_lock = false;
+ return;
+ }
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subbPtr.p->m_subPtrI);
+
+ SubStopReq * const req = (SubStopReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = subbPtr.i;
+ req->subscriberRef = subbPtr.p->m_subscriberRef;
+ req->subscriberData = subbPtr.p->m_subscriberData;
+ req->subscriptionId = subPtr.p->m_subscriptionId;
+ req->subscriptionKey = subPtr.p->m_subscriptionKey;
+ req->part = SubscriptionData::TableData;
+
+ sendSignal(SUMA_REF, GSN_SUB_STOP_REQ, signal, SubStopReq::SignalLength, JBB);
+}
+
+void
+SumaParticipant::execSUB_STOP_CONF(Signal* signal){
+ jamEntry();
+
+ SubStopConf * const conf = (SubStopConf*)signal->getDataPtr();
+
+ // Uint32 subscriberData = conf->subscriberData;
+ // Uint32 subscriberRef = conf->subscriberRef;
+
+ Subscription key;
+ key.m_subscriptionId = conf->subscriptionId;
+ key.m_subscriptionKey = conf->subscriptionKey;
+
+ SubscriptionPtr subPtr;
+ if(c_subscriptions.find(subPtr, key)) {
+ jam();
+ if (subPtr.p->m_markRemove) {
+ jam();
+ ndbrequire(false);
+ ndbrequire(subPtr.p->m_nSubscribers > 0);
+ subPtr.p->m_nSubscribers--;
+ if (subPtr.p->m_nSubscribers == 0){
+ jam();
+ completeSubRemoveReq(signal, subPtr);
+ }
+ }
+ }
+
+ sendSubStopReq(signal);
+}
+
+void
+SumaParticipant::execSUB_STOP_REF(Signal* signal){
+ jamEntry();
+ SubStopRef * const ref = (SubStopRef*)signal->getDataPtr();
+
+ Uint32 subscriptionId = ref->subscriptionId;
+ Uint32 subscriptionKey = ref->subscriptionKey;
+ Uint32 part = ref->part;
+ Uint32 subscriberData = ref->subscriberData;
+ Uint32 subscriberRef = ref->subscriberRef;
+ // Uint32 err = ref->err;
+
+ if(!ref->isTemporary()){
+ ndbrequire(false);
+ }
+
+ SubStopReq * const req = (SubStopReq*)signal->getDataPtrSend();
+ req->subscriberRef = subscriberRef;
+ req->subscriberData = subscriberData;
+ req->subscriptionId = subscriptionId;
+ req->subscriptionKey = subscriptionKey;
+ req->part = part;
+
+ sendSignal(SUMA_REF, GSN_SUB_STOP_REQ, signal, SubStopReq::SignalLength, JBB);
+}
+
+void
+Suma::execNODE_FAILREP(Signal* signal){
+ jamEntry();
+
+ NodeFailRep * const rep = (NodeFailRep*)signal->getDataPtr();
+
+ bool changed = false;
+
+ NodePtr nodePtr;
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma: nodefailrep");
+#endif
+ c_nodeFailGCI = getFirstGCI(signal);
+
+ for(c_nodes.first(nodePtr); nodePtr.i != RNIL; c_nodes.next(nodePtr)){
+ if(NodeBitmask::get(rep->theNodes, nodePtr.p->nodeId)){
+ if(nodePtr.p->alive){
+ ndbassert(c_aliveNodes.get(nodePtr.p->nodeId));
+ changed = true;
+ jam();
+ } else {
+ ndbassert(!c_aliveNodes.get(nodePtr.p->nodeId));
+ jam();
+ }
+
+ if (c_preparingNodes.get(nodePtr.p->nodeId)) {
+ jam();
+ // we are currently preparing this node that died
+ // it's ok just to clear and go back to waiting for it to start up
+ Restart.resetNode(calcSumaBlockRef(nodePtr.p->nodeId));
+ c_preparingNodes.clear(nodePtr.p->nodeId);
+ } else if (c_handoverToDo) {
+ jam();
+ // TODO what if I'm a SUMA that is currently restarting and the SUMA
+ // responsible for restarting me is the one that died?
+
+ // a node has failed whilst handover is going on
+ // let's check if we're in the process of handover with that node
+ c_handoverToDo = false;
+ for( int i = 0; i < NO_OF_BUCKETS; i++) {
+ if (c_buckets[i].handover) {
+ // I'm doing handover, but is it with the dead node?
+ if (getResponsibleSumaNodeId(i) == nodePtr.p->nodeId) {
+ // so it was the dead node, has handover started?
+ if (c_buckets[i].handover_started) {
+ jam();
+ // we're not ok and will have lost data!
+ // set not active to indicate this -
+ // this will generate takeover behaviour
+ c_buckets[i].active = false;
+ c_buckets[i].handover_started = false;
+ } // else we're ok to revert back to state before
+ c_buckets[i].handover = false;
+ } else {
+ jam();
+ // ok, we're doing handover with a different node
+ c_handoverToDo = true;
+ }
+ }
+ }
+ }
+
+ c_failoverBuffer.nodeFailRep();
+
+ nodePtr.p->alive = 0;
+ c_aliveNodes.clear(nodePtr.p->nodeId); // this has to be done after the loop above
+ }
+ }
+}
+
+void
+Suma::execINCL_NODEREQ(Signal* signal){
+ jamEntry();
+
+ //const Uint32 senderRef = signal->theData[0];
+ const Uint32 inclNode = signal->theData[1];
+
+ NodePtr node;
+ for(c_nodes.first(node); node.i != RNIL; c_nodes.next(node)){
+ jam();
+ const Uint32 nodeId = node.p->nodeId;
+ if(inclNode == nodeId){
+ jam();
+
+ ndbrequire(node.p->alive == 0);
+ ndbrequire(!c_aliveNodes.get(nodeId));
+
+ for (Uint32 j = 0; j < c_noNodesInGroup; j++) {
+ jam();
+ if (c_nodesInGroup[j] == nodeId) {
+ // the starting node is part of my node group
+ jam();
+ c_preparingNodes.set(nodeId); // set as being prepared
+ for (Uint32 i = 0; i < c_noNodesInGroup; i++) {
+ jam();
+ if (i == c_idInNodeGroup) {
+ jam();
+ // I'm responsible for restarting this SUMA
+ // ALL dict's should have meta data info so it is ok to start
+ Restart.startNode(signal, calcSumaBlockRef(nodeId));
+ break;
+ }//if
+ if (c_aliveNodes.get(c_nodesInGroup[i])) {
+ jam();
+ break; // another Suma takes care of this
+ }//if
+ }//for
+ break;
+ }//if
+ }//for
+
+ node.p->alive = 1;
+ c_aliveNodes.set(nodeId);
+
+ break;
+ }//if
+ }//for
+
+#if 0 // if we include this DIH's got to be prepared, later if needed...
+ signal->theData[0] = reference();
+
+ sendSignal(senderRef, GSN_INCL_NODECONF, signal, 1, JBB);
+#endif
+}
+
+void
+Suma::execSIGNAL_DROPPED_REP(Signal* signal){
+ jamEntry();
+ ndbrequire(false);
+}
+
+/********************************************************************
+ *
+ * Dump state
+ *
+ */
+
+void
+Suma::execDUMP_STATE_ORD(Signal* signal){
+ jamEntry();
+
+ Uint32 tCase = signal->theData[0];
+ if(tCase < 8000 || tCase > 8004)
+ return;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, g_subPtrI);
+
+ Ptr<SyncRecord> syncPtr;
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+
+ if(tCase == 8000){
+ syncPtr.p->startMeta(signal);
+ }
+
+ if(tCase == 8001){
+ syncPtr.p->startScan(signal);
+ }
+
+ if(tCase == 8002){
+ syncPtr.p->startTrigger(signal);
+ }
+
+ if(tCase == 8003){
+ subPtr.p->m_subscriptionType = SubCreateReq::SingleTableScan;
+ LocalDataBuffer<15> attrs(c_dataBufferPool, syncPtr.p->m_attributeList);
+ Uint32 tab = 0;
+ Uint32 att[] = { 0, 1, 1 };
+ syncPtr.p->m_tableList.append(&tab, 1);
+ attrs.append(att, 3);
+ }
+
+ if(tCase == 8004){
+ infoEvent("Suma: c_subscriberPool size: %d free: %d",
+ c_subscriberPool.getSize(),
+ c_subscriberPool.getNoOfFree());
+
+ infoEvent("Suma: c_tablePool size: %d free: %d",
+ c_tablePool_.getSize(),
+ c_tablePool_.getNoOfFree());
+
+ infoEvent("Suma: c_subscriptionPool size: %d free: %d",
+ c_subscriptionPool.getSize(),
+ c_subscriptionPool.getNoOfFree());
+
+ infoEvent("Suma: c_syncPool size: %d free: %d",
+ c_syncPool.getSize(),
+ c_syncPool.getNoOfFree());
+
+ infoEvent("Suma: c_dataBufferPool size: %d free: %d",
+ c_dataBufferPool.getSize(),
+ c_dataBufferPool.getNoOfFree());
+ }
+}
+
+/********************************************************************
+ *
+ * Convert a table name (db+schema+tablename) to tableId
+ *
+ */
+
+void
+SumaParticipant::convertNameToId(SubscriptionPtr subPtr, Signal * signal)
+{
+ jam();
+ if(subPtr.p->m_currentTable < subPtr.p->m_maxTables) {
+ jam();
+
+ GetTableIdReq * req = (GetTableIdReq *)signal->getDataPtrSend();
+ char * tableName = subPtr.p->m_tableNames[subPtr.p->m_currentTable];
+ const Uint32 strLen = strlen(tableName) + 1; // NULL Terminated
+ req->senderRef = reference();
+ req->senderData = subPtr.i;
+ req->len = strLen;
+
+ LinearSectionPtr ptr[1];
+ ptr[0].p = (Uint32*)tableName;
+ ptr[0].sz = strLen;
+
+ sendSignal(DBDICT_REF,
+ GSN_GET_TABLEID_REQ,
+ signal,
+ GetTableIdReq::SignalLength,
+ JBB,
+ ptr,
+ 1);
+ } else {
+ jam();
+ sendSubCreateConf(signal, subPtr.p->m_subscriberRef, subPtr);
+ }
+}
+
+
+void
+SumaParticipant::addTableId(Uint32 tableId,
+ SubscriptionPtr subPtr, SyncRecord *psyncRec)
+{
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("SumaParticipant::addTableId(%u,%u,%u), current_table=%u",
+ tableId, subPtr.i, psyncRec, subPtr.p->m_currentTable);
+#endif
+ subPtr.p->m_tables[tableId] = 1;
+ subPtr.p->m_currentTable++;
+ if(psyncRec != NULL)
+ psyncRec->m_tableList.append(&tableId, 1);
+}
+
+void
+SumaParticipant::execGET_TABLEID_CONF(Signal * signal)
+{
+ jamEntry();
+
+ GetTableIdConf* conf = (GetTableIdConf *)signal->getDataPtr();
+ Uint32 tableId = conf->tableId;
+ //Uint32 schemaVersion = conf->schemaVersion;
+ Uint32 senderData = conf->senderData;
+
+ SubscriptionPtr subPtr;
+ Ptr<SyncRecord> syncPtr;
+
+ c_subscriptions.getPtr(subPtr, senderData);
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+
+ /*
+ * add to m_tableList
+ */
+ addTableId(tableId, subPtr, syncPtr.p);
+
+ convertNameToId(subPtr, signal);
+}
+
+void
+SumaParticipant::execGET_TABLEID_REF(Signal * signal)
+{
+ jamEntry();
+ GetTableIdRef const * ref = (GetTableIdRef *)signal->getDataPtr();
+ Uint32 senderData = ref->senderData;
+ // Uint32 err = ref->err;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, senderData);
+ Uint32 subData = subPtr.p->m_subscriberData;
+ SubCreateRef * reff = (SubCreateRef*)ref;
+ /**
+ * @todo: map ref->err to GrepError.
+ */
+ reff->err = GrepError::SELECTED_TABLE_NOT_FOUND;
+ reff->subscriberData = subData;
+ sendSignal(subPtr.p->m_subscriberRef,
+ GSN_SUB_CREATE_REF,
+ signal,
+ SubCreateRef::SignalLength,
+ JBB);
+}
+
+/*************************************************************
+ *
+ * Creation of subscription id's
+ *
+ ************************************************************/
+
+void
+Suma::execCREATE_SUBID_REQ(Signal* signal)
+{
+ jamEntry();
+
+ CRASH_INSERTION(13001);
+
+ CreateSubscriptionIdReq const * req =
+ (CreateSubscriptionIdReq*)signal->getDataPtr();
+ SubscriberPtr subbPtr;
+ if(!c_subscriberPool.seize(subbPtr)){
+ jam();
+ sendSubIdRef(signal, GrepError::SUBSCRIPTION_ID_NOMEM);
+ return;
+ }
+
+ subbPtr.p->m_subscriberRef = signal->getSendersBlockRef();
+ subbPtr.p->m_senderData = req->senderData;
+ subbPtr.p->m_subscriberData = subbPtr.i;
+
+ UtilSequenceReq * utilReq = (UtilSequenceReq*)signal->getDataPtrSend();
+
+ utilReq->senderData = subbPtr.p->m_subscriberData;
+ utilReq->sequenceId = SUMA_SEQUENCE;
+ utilReq->requestType = UtilSequenceReq::NextVal;
+ sendSignal(DBUTIL_REF, GSN_UTIL_SEQUENCE_REQ,
+ signal, UtilSequenceReq::SignalLength, JBB);
+}
+
+void
+Suma::execUTIL_SEQUENCE_CONF(Signal* signal)
+{
+ jamEntry();
+
+ CRASH_INSERTION(13002);
+
+ UtilSequenceConf * conf = (UtilSequenceConf*)signal->getDataPtr();
+#ifdef DEBUG_SUMA_SEQUENCE
+ ndbout_c("SUMA: Create sequence conf");
+#endif
+ if(conf->requestType == UtilSequenceReq::Create) {
+ jam();
+ createSequenceReply(signal, conf, NULL);
+ return;
+ }
+
+ Uint32 subId = conf->sequenceValue[0];
+ Uint32 subData = conf->senderData;
+
+ SubscriberPtr subbPtr;
+ c_subscriberPool.getPtr(subbPtr,subData);
+
+
+ CreateSubscriptionIdConf * subconf = (CreateSubscriptionIdConf*)conf;
+ subconf->subscriptionId = subId;
+ subconf->subscriptionKey =(getOwnNodeId() << 16) | (subId & 0xFFFF);
+ subconf->subscriberData = subbPtr.p->m_senderData;
+
+ sendSignal(subbPtr.p->m_subscriberRef, GSN_CREATE_SUBID_CONF, signal,
+ CreateSubscriptionIdConf::SignalLength, JBB);
+
+ c_subscriberPool.release(subbPtr);
+}
+
+void
+Suma::execUTIL_SEQUENCE_REF(Signal* signal)
+{
+ jamEntry();
+ UtilSequenceRef * ref = (UtilSequenceRef*)signal->getDataPtr();
+
+ if(ref->requestType == UtilSequenceReq::Create) {
+ jam();
+ createSequenceReply(signal, NULL, ref);
+ return;
+ }
+
+ Uint32 subData = ref->senderData;
+
+ SubscriberPtr subbPtr;
+ c_subscriberPool.getPtr(subbPtr,subData);
+ sendSubIdRef(signal, GrepError::SEQUENCE_ERROR);
+ c_subscriberPool.release(subbPtr);
+ return;
+}//execUTIL_SEQUENCE_REF()
+
+
+void
+SumaParticipant::sendSubIdRef(Signal* signal, Uint32 errCode){
+ jam();
+ CreateSubscriptionIdRef * ref =
+ (CreateSubscriptionIdRef *)signal->getDataPtrSend();
+
+ ref->err = errCode;
+ sendSignal(signal->getSendersBlockRef(),
+ GSN_CREATE_SUBID_REF,
+ signal,
+ CreateSubscriptionIdRef::SignalLength,
+ JBB);
+
+ releaseSections(signal);
+ return;
+}
+
+/**********************************************************
+ * Suma participant interface
+ *
+ * Creation of subscriptions
+ */
+
+void
+SumaParticipant::execSUB_CREATE_REQ(Signal* signal) {
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("SumaParticipant::execSUB_CREATE_REQ");
+#endif
+ jamEntry();
+
+ CRASH_INSERTION(13003);
+
+ const SubCreateReq req = *(SubCreateReq*)signal->getDataPtr();
+
+ const Uint32 subId = req.subscriptionId;
+ const Uint32 subKey = req.subscriptionKey;
+ const Uint32 subRef = req.subscriberRef;
+ const Uint32 subData = req.subscriberData;
+ const Uint32 type = req.subscriptionType & SubCreateReq::RemoveFlags;
+ const Uint32 flags = req.subscriptionType & SubCreateReq::GetFlags;
+ const bool addTableFlag = (flags & SubCreateReq::AddTableFlag) != 0;
+ const bool restartFlag = (flags & SubCreateReq::RestartFlag) != 0;
+
+ const Uint32 sender = signal->getSendersBlockRef();
+
+ Subscription key;
+ key.m_subscriptionId = subId;
+ key.m_subscriptionKey = subKey;
+
+ SubscriptionPtr subPtr;
+ Ptr<SyncRecord> syncPtr;
+
+ if (addTableFlag) {
+ ndbrequire(restartFlag); //TODO remove this
+
+ if(!c_subscriptions.find(subPtr, key)) {
+ jam();
+ sendSubCreateRef(signal, req, GrepError::SUBSCRIPTION_NOT_FOUND);
+ return;
+ }
+ jam();
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+ } else {
+ // Check that id/key is unique
+ if(c_subscriptions.find(subPtr, key)) {
+ jam();
+ sendSubCreateRef(signal, req, GrepError::SUBSCRIPTION_ID_NOT_UNIQUE);
+ return;
+ }
+ if(!c_subscriptions.seize(subPtr)) {
+ jam();
+ sendSubCreateRef(signal, req, GrepError::NOSPACE_IN_POOL);
+ return;
+ }
+ if(!c_syncPool.seize(syncPtr)) {
+ jam();
+ sendSubCreateRef(signal, req, GrepError::NOSPACE_IN_POOL);
+ return;
+ }
+ jam();
+ subPtr.p->m_subscriberRef = subRef;
+ subPtr.p->m_subscriberData = subData;
+ subPtr.p->m_subscriptionId = subId;
+ subPtr.p->m_subscriptionKey = subKey;
+ subPtr.p->m_subscriptionType = type;
+
+ /**
+ * ok to memset? Support on all compilers
+ * @todo find out if memset is supported by all compilers
+ */
+ memset(subPtr.p->m_tables,0,MAX_TABLES);
+ subPtr.p->m_maxTables = 0;
+ subPtr.p->m_currentTable = 0;
+ subPtr.p->m_syncPtrI = syncPtr.i;
+ subPtr.p->m_markRemove = false;
+ subPtr.p->m_nSubscribers = 0;
+
+ c_subscriptions.add(subPtr);
+
+ syncPtr.p->m_subscriptionPtrI = subPtr.i;
+ syncPtr.p->m_doSendSyncData = true;
+ syncPtr.p->ptrI = syncPtr.i;
+ syncPtr.p->m_locked = false;
+ syncPtr.p->m_error = false;
+ }
+
+ if (restartFlag ||
+ type == SubCreateReq::TableEvent) {
+
+ syncPtr.p->m_doSendSyncData = false;
+
+ ndbrequire(type != SubCreateReq::SingleTableScan);
+ jam();
+
+ if (subPtr.p->m_tables[req.tableId] != 0) {
+ ndbrequire(false); //TODO remove
+ jam();
+ sendSubCreateRef(signal, req, GrepError::SELECTED_TABLE_ALREADY_ADDED);
+ return;
+ }
+ if (addTableFlag) {
+ ndbrequire(type != SubCreateReq::TableEvent);
+ jam();
+ }
+ subPtr.p->m_maxTables++;
+ addTableId(req.tableId, subPtr, syncPtr.p);
+ } else {
+ switch(type){
+ case SubCreateReq::SingleTableScan:
+ {
+ jam();
+ syncPtr.p->m_tableList.append(&req.tableId, 1);
+ if(signal->getNoOfSections() > 0){
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr, SubCreateReq::ATTRIBUTE_LIST);
+ LocalDataBuffer<15> attrBuf(c_dataBufferPool,syncPtr.p->m_attributeList);
+ append(attrBuf, ptr, getSectionSegmentPool());
+ }
+ }
+ break;
+ case SubCreateReq::SelectiveTableSnapshot:
+ /**
+ * Tables specified by the user that does not exist
+ * in the database are just ignored. No error message
+ * is given, nor does the db nodes crash
+ * @todo: Memory is not release here (used tableBuf)
+ */
+ {
+ if(signal->getNoOfSections() == 0 ){
+ jam();
+ sendSubCreateRef(signal, req, GrepError::WRONG_NO_OF_SECTIONS);
+ return;
+ }
+
+ jam();
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr,0);// SubCreateReq::TABLE_LIST);
+ SimplePropertiesSectionReader r0(ptr, getSectionSegmentPool());
+ Uint32 i=0;
+ char table[MAX_TAB_NAME_SIZE];
+ r0.reset();
+ r0.first();
+ while(true){
+ if ((r0.getValueType() != SimpleProperties::StringValue) ||
+ (r0.getValueLen() <= 0)) {
+ releaseSections(signal);
+ ndbrequire(false);
+ }
+ r0.getString(table);
+ strcpy(subPtr.p->m_tableNames[i],table);
+ i++;
+ if(!r0.next())
+ break;
+ }
+ releaseSections(signal);
+ subPtr.p->m_maxTables = i;
+ subPtr.p->m_currentTable = 0;
+ releaseSections(signal);
+ convertNameToId(subPtr, signal);
+ return;
+ }
+ break;
+ case SubCreateReq::DatabaseSnapshot:
+ {
+ jam();
+ }
+ break;
+ default:
+ ndbrequire(false);
+ }
+ }
+
+ sendSubCreateConf(signal, sender, subPtr);
+
+ return;
+}
+
+void
+SumaParticipant::sendSubCreateConf(Signal* signal, Uint32 sender,
+ SubscriptionPtr subPtr)
+{
+ SubCreateConf * const conf = (SubCreateConf*)signal->getDataPtrSend();
+ conf->subscriptionId = subPtr.p->m_subscriptionId;
+ conf->subscriptionKey = subPtr.p->m_subscriptionKey;
+ conf->subscriberData = subPtr.p->m_subscriberData;
+ sendSignal(sender, GSN_SUB_CREATE_CONF, signal,
+ SubCreateConf::SignalLength, JBB);
+}
+
+void
+SumaParticipant::sendSubCreateRef(Signal* signal, const SubCreateReq& req, Uint32 errCode){
+ jam();
+ SubCreateRef * ref = (SubCreateRef *)signal->getDataPtrSend();
+ ref->subscriberRef = reference();
+ ref->subscriberData = req.subscriberData;
+ ref->err = errCode;
+ releaseSections(signal);
+ sendSignal(signal->getSendersBlockRef(), GSN_SUB_CREATE_REF, signal,
+ SubCreateRef::SignalLength, JBB);
+ return;
+}
+
+
+
+
+
+
+
+
+
+
+
+
+Uint32
+SumaParticipant::getFirstGCI(Signal* signal) {
+ if (c_lastCompleteGCI == RNIL) {
+ ndbout_c("WARNING: c_lastCompleteGCI == RNIL");
+ return 0;
+ }
+ return c_lastCompleteGCI+3;
+}
+
+/**********************************************************
+ *
+ * Setting upp trigger for subscription
+ *
+ */
+
+void
+SumaParticipant::execSUB_SYNC_REQ(Signal* signal) {
+ jamEntry();
+
+ CRASH_INSERTION(13004);
+#ifdef EVENT_PH3_DEBUG
+ ndbout_c("SumaParticipant::execSUB_SYNC_REQ");
+#endif
+
+ SubSyncReq * const req = (SubSyncReq*)signal->getDataPtr();
+
+ SubscriptionPtr subPtr;
+ Subscription key;
+ key.m_subscriptionId = req->subscriptionId;
+ key.m_subscriptionKey = req->subscriptionKey;
+
+ if(!c_subscriptions.find(subPtr, key)){
+ jam();
+ sendSubSyncRef(signal, GrepError::SUBSCRIPTION_ID_NOT_FOUND);
+ return;
+ }
+
+ /**
+ * @todo Tomas, do you really need to do this?
+ */
+ if(subPtr.p->m_subscriptionType == SubCreateReq::TableEvent) {
+ jam();
+ subPtr.p->m_subscriberData = req->subscriberData;
+ }
+
+ bool ok = false;
+ SubscriptionData::Part part = (SubscriptionData::Part)req->part;
+
+ Ptr<SyncRecord> syncPtr;
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+ switch(part){
+ case SubscriptionData::MetaData:
+ ok = true;
+ jam();
+ if (subPtr.p->m_subscriptionType == SubCreateReq::DatabaseSnapshot) {
+ TableList::DataBufferIterator it;
+ syncPtr.p->m_tableList.first(it);
+ if(it.isNull()) {
+ /**
+ * Get all tables from dict
+ */
+ ListTablesReq * req = (ListTablesReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = syncPtr.i;
+ req->requestData = 0;
+ /**
+ * @todo: accomodate scan of index tables?
+ */
+ req->setTableType(DictTabInfo::UserTable);
+
+ sendSignal(DBDICT_REF, GSN_LIST_TABLES_REQ, signal,
+ ListTablesReq::SignalLength, JBB);
+ break;
+ }
+ }
+
+ syncPtr.p->startMeta(signal);
+ break;
+ case SubscriptionData::TableData: {
+ ok = true;
+ jam();
+ syncPtr.p->startScan(signal);
+ break;
+ }
+ }
+ ndbrequire(ok);
+}
+
+void
+SumaParticipant::sendSubSyncRef(Signal* signal, Uint32 errCode){
+ jam();
+ SubSyncRef * ref =
+ (SubSyncRef *)signal->getDataPtrSend();
+ ref->err = errCode;
+ sendSignal(signal->getSendersBlockRef(),
+ GSN_SUB_SYNC_REF,
+ signal,
+ SubSyncRef::SignalLength,
+ JBB);
+
+ releaseSections(signal);
+ return;
+}
+
+/**********************************************************
+ * Dict interface
+ */
+
+void
+SumaParticipant::execLIST_TABLES_CONF(Signal* signal){
+ jamEntry();
+ CRASH_INSERTION(13005);
+ ListTablesConf* const conf = (ListTablesConf*)signal->getDataPtr();
+ SyncRecord* tmp = c_syncPool.getPtr(conf->senderData);
+ tmp->runLIST_TABLES_CONF(signal);
+}
+
+
+void
+SumaParticipant::execGET_TABINFOREF(Signal* signal){
+ jamEntry();
+ GetTabInfoRef* const ref = (GetTabInfoRef*)signal->getDataPtr();
+ SyncRecord* tmp = c_syncPool.getPtr(ref->senderData);
+ tmp->runGET_TABINFOREF(signal);
+}
+
+void
+SumaParticipant::execGET_TABINFO_CONF(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13006);
+
+ if(!assembleFragments(signal)){
+ return;
+ }
+
+ GetTabInfoConf* conf = (GetTabInfoConf*)signal->getDataPtr();
+
+ Uint32 tableId = conf->tableId;
+ Uint32 senderData = conf->senderData;
+
+ SyncRecord* tmp = c_syncPool.getPtr(senderData);
+ ndbrequire(parseTable(signal, conf, tableId, tmp));
+ tmp->runGET_TABINFO_CONF(signal);
+}
+
+bool
+SumaParticipant::parseTable(Signal* signal, GetTabInfoConf* conf, Uint32 tableId,
+ SyncRecord* syncPtr_p){
+
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr, GetTabInfoConf::DICT_TAB_INFO);
+
+ SimplePropertiesSectionReader it(ptr, getSectionSegmentPool());
+
+ SimpleProperties::UnpackStatus s;
+ DictTabInfo::Table tableDesc; tableDesc.init();
+ s = SimpleProperties::unpack(it, &tableDesc,
+ DictTabInfo::TableMapping,
+ DictTabInfo::TableMappingSize,
+ true, true);
+
+ ndbrequire(s == SimpleProperties::Break);
+
+ TablePtr tabPtr;
+ c_tables.find(tabPtr, tableId);
+
+ if(!tabPtr.isNull() &&
+ tabPtr.p->m_schemaVersion != tableDesc.TableVersion){
+ jam();
+ // oops wrong schema version in stored tabledesc
+ // we need to find all subscriptions with old table desc
+ // and all subscribers to this
+ // hopefully none
+ c_tables.release(tabPtr);
+ tabPtr.setNull();
+ DLHashTable<SumaParticipant::Subscription>::Iterator i_subPtr;
+ c_subscriptions.first(i_subPtr);
+ SubscriptionPtr subPtr;
+ for(;!i_subPtr.isNull();c_subscriptions.next(i_subPtr)){
+ jam();
+ c_subscriptions.getPtr(subPtr, i_subPtr.curr.i);
+ SyncRecord* tmp = c_syncPool.getPtr(subPtr.p->m_syncPtrI);
+ if (tmp == syncPtr_p) {
+ jam();
+ continue;
+ }
+ if (subPtr.p->m_tables[tableId]) {
+ jam();
+ subPtr.p->m_tables[tableId] = 0; // remove this old table reference
+ TableList::DataBufferIterator it;
+ for(tmp->m_tableList.first(it);!it.isNull();tmp->m_tableList.next(it)) {
+ jam();
+ if (*it.data == tableId){
+ jam();
+ Uint32 *pdata = it.data;
+ tmp->m_tableList.next(it);
+ for(;!it.isNull();tmp->m_tableList.next(it)) {
+ jam();
+ *pdata = *it.data;
+ pdata = it.data;
+ }
+ *pdata = RNIL; // todo remove this last item...
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ if (tabPtr.isNull()) {
+ jam();
+ /**
+ * Uninitialized table record
+ */
+ ndbrequire(c_tables.seize(tabPtr));
+ new (tabPtr.p) Table;
+ tabPtr.p->m_schemaVersion = RNIL;
+ tabPtr.p->m_tableId = tableId;
+ tabPtr.p->m_hasTriggerDefined[0] = 0;
+ tabPtr.p->m_hasTriggerDefined[1] = 0;
+ tabPtr.p->m_hasTriggerDefined[2] = 0;
+ tabPtr.p->m_triggerIds[0] = ILLEGAL_TRIGGER_ID;
+ tabPtr.p->m_triggerIds[1] = ILLEGAL_TRIGGER_ID;
+ tabPtr.p->m_triggerIds[2] = ILLEGAL_TRIGGER_ID;
+#if 0
+ ndbout_c("Get tab info conf %d", tableId);
+#endif
+ c_tables.add(tabPtr);
+ }
+
+ if(tabPtr.p->m_attributes.getSize() != 0){
+ jam();
+ return true;
+ }
+
+ /**
+ * Initialize table object
+ */
+ Uint32 noAttribs = tableDesc.NoOfAttributes;
+ Uint32 notFixed = (tableDesc.NoOfNullable+tableDesc.NoOfVariable);
+ tabPtr.p->m_schemaVersion = tableDesc.TableVersion;
+
+ // The attribute buffer
+ LocalDataBuffer<15> attrBuf(c_dataBufferPool, tabPtr.p->m_attributes);
+
+ // Temporary buffer
+ DataBuffer<15> theRest(c_dataBufferPool);
+
+ if(!attrBuf.seize(noAttribs)){
+ ndbrequire(false);
+ return false;
+ }
+
+ if(!theRest.seize(notFixed)){
+ ndbrequire(false);
+ return false;
+ }
+
+ DataBuffer<15>::DataBufferIterator attrIt; // Fixed not nullable
+ DataBuffer<15>::DataBufferIterator restIt; // variable + nullable
+ attrBuf.first(attrIt);
+ theRest.first(restIt);
+
+ for(Uint32 i = 0; i < noAttribs; i++) {
+ DictTabInfo::Attribute attrDesc; attrDesc.init();
+ s = SimpleProperties::unpack(it, &attrDesc,
+ DictTabInfo::AttributeMapping,
+ DictTabInfo::AttributeMappingSize,
+ true, true);
+ ndbrequire(s == SimpleProperties::Break);
+
+ if (!attrDesc.AttributeNullableFlag
+ /* && !attrDesc.AttributeVariableFlag */) {
+ jam();
+ * attrIt.data = attrDesc.AttributeId;
+ attrBuf.next(attrIt);
+ } else {
+ jam();
+ * restIt.data = attrDesc.AttributeId;
+ theRest.next(restIt);
+ }
+
+ // Move to next attribute
+ it.next();
+ }
+
+ /**
+ * Put the rest in end of attrBuf
+ */
+ theRest.first(restIt);
+ for(; !restIt.isNull(); theRest.next(restIt)){
+ * attrIt.data = * restIt.data;
+ attrBuf.next(attrIt);
+ }
+
+ theRest.release();
+
+ return true;
+}
+
+void
+SumaParticipant::execDI_FCOUNTCONF(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13007);
+
+ const Uint32 senderData = signal->theData[3];
+ SyncRecord* tmp = c_syncPool.getPtr(senderData);
+ tmp->runDI_FCOUNTCONF(signal);
+}
+
+void
+SumaParticipant::execDIGETPRIMCONF(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13008);
+
+ const Uint32 senderData = signal->theData[1];
+ SyncRecord* tmp = c_syncPool.getPtr(senderData);
+ tmp->runDIGETPRIMCONF(signal);
+}
+
+void
+SumaParticipant::execCREATE_TRIG_CONF(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13009);
+
+ CreateTrigConf * const conf = (CreateTrigConf*)signal->getDataPtr();
+
+ const Uint32 senderData = conf->getConnectionPtr();
+ SyncRecord* tmp = c_syncPool.getPtr(senderData);
+ tmp->runCREATE_TRIG_CONF(signal);
+
+ /**
+ * dodido
+ * @todo: I (Johan) dont know what to do here. Jonas, what do you mean?
+ */
+}
+
+void
+SumaParticipant::execCREATE_TRIG_REF(Signal* signal){
+ jamEntry();
+ ndbrequire(false);
+}
+
+void
+SumaParticipant::execDROP_TRIG_CONF(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13010);
+
+ DropTrigConf * const conf = (DropTrigConf*)signal->getDataPtr();
+
+ const Uint32 senderData = conf->getConnectionPtr();
+ SyncRecord* tmp = c_syncPool.getPtr(senderData);
+ tmp->runDROP_TRIG_CONF(signal);
+}
+
+void
+SumaParticipant::execDROP_TRIG_REF(Signal* signal){
+ jamEntry();
+
+ DropTrigRef * const ref = (DropTrigRef*)signal->getDataPtr();
+
+ const Uint32 senderData = ref->getConnectionPtr();
+ SyncRecord* tmp = c_syncPool.getPtr(senderData);
+ tmp->runDROP_TRIG_CONF(signal);
+}
+
+/*************************************************************************
+ *
+ *
+ */
+
+void
+SumaParticipant::SyncRecord::runLIST_TABLES_CONF(Signal* signal){
+ jam();
+
+ ListTablesConf * const conf = (ListTablesConf*)signal->getDataPtr();
+ const Uint32 len = signal->length() - ListTablesConf::HeaderLength;
+
+ SubscriptionPtr subPtr;
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+
+ for (unsigned i = 0; i < len; i++) {
+ subPtr.p->m_maxTables++;
+ suma.addTableId(ListTablesConf::getTableId(conf->tableData[i]), subPtr, this);
+ }
+
+ // for (unsigned i = 0; i < len; i++)
+ // conf->tableData[i] = ListTablesConf::getTableId(conf->tableData[i]);
+ // m_tableList.append(&conf->tableData[0], len);
+
+#if 0
+ TableList::DataBufferIterator it;
+ int i = 0;
+ for(m_tableList.first(it);!it.isNull();m_tableList.next(it)) {
+ ndbout_c("%u listtableconf tableid %d", i++, *it.data);
+ }
+#endif
+
+ if(len == ListTablesConf::DataLength){
+ jam();
+ // we expect more LIST_TABLE_CONF
+ return;
+ }
+
+#if 0
+ subPtr.p->m_currentTable = 0;
+ subPtr.p->m_maxTables = 0;
+
+ TableList::DataBufferIterator it;
+ for(m_tableList.first(it); !it.isNull(); m_tableList.next(it)) {
+ subPtr.p->m_maxTables++;
+ suma.addTableId(*it.data, subPtr, NULL);
+#ifdef NODEFAIL_DEBUG
+ ndbout_c(" listtableconf tableid %d",*it.data);
+#endif
+ }
+#endif
+
+ startMeta(signal);
+}
+
+void
+SumaParticipant::SyncRecord::startMeta(Signal* signal){
+ jam();
+ m_currentTable = 0;
+ nextMeta(signal);
+}
+
+/**
+ * m_tableList only contains UserTables
+ */
+void
+SumaParticipant::SyncRecord::nextMeta(Signal* signal){
+ jam();
+
+ TableList::DataBufferIterator it;
+ if(!m_tableList.position(it, m_currentTable)){
+ completeMeta(signal);
+ return;
+ }
+
+ GetTabInfoReq * req = (GetTabInfoReq *)signal->getDataPtrSend();
+ req->senderRef = suma.reference();
+ req->senderData = ptrI;
+ req->requestType =
+ GetTabInfoReq::RequestById | GetTabInfoReq::LongSignalConf;
+ req->tableId = * it.data;
+
+#if 0
+ ndbout_c("GET_TABINFOREQ id %d", req->tableId);
+#endif
+ suma.sendSignal(DBDICT_REF, GSN_GET_TABINFOREQ, signal,
+ GetTabInfoReq::SignalLength, JBB);
+}
+
+void
+SumaParticipant::SyncRecord::runGET_TABINFOREF(Signal* signal)
+{
+ jam();
+
+ SubscriptionPtr subPtr;
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ ndbrequire(subPtr.p->m_syncPtrI == ptrI);
+
+ Uint32 type = subPtr.p->m_subscriptionType;
+
+ bool do_continue = false;
+ switch (type) {
+ case SubCreateReq::TableEvent:
+ jam();
+ break;
+ case SubCreateReq::DatabaseSnapshot:
+ jam();
+ do_continue = true;
+ break;
+ case SubCreateReq::SelectiveTableSnapshot:
+ jam();
+ do_continue = true;
+ break;
+ case SubCreateReq::SingleTableScan:
+ jam();
+ break;
+ default:
+ ndbrequire(false);
+ break;
+ }
+
+ if (! do_continue) {
+ m_error = true;
+ completeMeta(signal);
+ return;
+ }
+
+ m_currentTable++;
+ nextMeta(signal);
+ return;
+
+ // now we need to clean-up
+}
+
+
+void
+SumaParticipant::SyncRecord::runGET_TABINFO_CONF(Signal* signal){
+ jam();
+
+ GetTabInfoConf * const conf = (GetTabInfoConf*)signal->getDataPtr();
+ // const Uint32 gci = conf->gci;
+ const Uint32 tableId = conf->tableId;
+ TableList::DataBufferIterator it;
+
+ ndbrequire(m_tableList.position(it, m_currentTable));
+ ndbrequire(* it.data == tableId);
+
+ SubscriptionPtr subPtr;
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ ndbrequire(subPtr.p->m_syncPtrI == ptrI);
+
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr, GetTabInfoConf::DICT_TAB_INFO);
+
+ SubMetaData * data = (SubMetaData*)signal->getDataPtrSend();
+ /**
+ * sending lastCompleteGCI. Used by Lars in interval calculations
+ * incremenet by one, since last_CompleteGCI is the not the current gci.
+ */
+ data->gci = suma.c_lastCompleteGCI + 1;
+ data->tableId = tableId;
+ data->senderData = subPtr.p->m_subscriberData;
+#if PRINT_ONLY
+ ndbout_c("GSN_SUB_META_DATA Table %d", tableId);
+#else
+
+ bool okToSend = m_doSendSyncData;
+
+ /*
+ * If it is a selectivetablesnapshot and the table is not part of the
+ * subscription, then do not send anything, just continue.
+ * If it is a tablevent, don't send regardless since the APIs are not
+ * interested in meta data.
+ */
+ if(subPtr.p->m_subscriptionType == SubCreateReq::SelectiveTableSnapshot)
+ if(!subPtr.p->m_tables[tableId])
+ okToSend = false;
+
+ if(okToSend) {
+ if(refToNode(subPtr.p->m_subscriberRef) == 0){
+ jam();
+ suma.EXECUTE_DIRECT(refToBlock(subPtr.p->m_subscriberRef),
+ GSN_SUB_META_DATA,
+ signal,
+ SubMetaData::SignalLength);
+ jamEntry();
+ suma.releaseSections(signal);
+ } else {
+ jam();
+ suma.sendSignal(subPtr.p->m_subscriberRef,
+ GSN_SUB_META_DATA,
+ signal,
+ SubMetaData::SignalLength, JBB);
+ }
+ }
+#endif
+
+ TablePtr tabPtr;
+ ndbrequire(suma.c_tables.find(tabPtr, tableId));
+
+ LocalDataBuffer<15> fragBuf(suma.c_dataBufferPool, tabPtr.p->m_fragments);
+ if(fragBuf.getSize() == 0){
+ /**
+ * We need to gather fragment info
+ */
+ jam();
+ signal->theData[0] = RNIL;
+ signal->theData[1] = tableId;
+ signal->theData[2] = ptrI;
+ suma.sendSignal(DBDIH_REF, GSN_DI_FCOUNTREQ, signal, 3, JBB);
+ return;
+ }
+
+ m_currentTable++;
+ nextMeta(signal);
+}
+
+void
+SumaParticipant::SyncRecord::runDI_FCOUNTCONF(Signal* signal){
+ jam();
+
+ const Uint32 userPtr = signal->theData[0];
+ const Uint32 fragCount = signal->theData[1];
+ const Uint32 tableId = signal->theData[2];
+
+ ndbrequire(userPtr == RNIL && signal->length() == 5);
+
+ TablePtr tabPtr;
+ ndbrequire(suma.c_tables.find(tabPtr, tableId));
+
+ LocalDataBuffer<15> fragBuf(suma.c_dataBufferPool, tabPtr.p->m_fragments);
+ ndbrequire(fragBuf.getSize() == 0);
+
+ m_currentFragment = fragCount;
+ signal->theData[0] = RNIL;
+ signal->theData[1] = ptrI;
+ signal->theData[2] = tableId;
+ signal->theData[3] = 0; // Frag no
+ suma.sendSignal(DBDIH_REF, GSN_DIGETPRIMREQ, signal, 4, JBB);
+}
+
+void
+SumaParticipant::SyncRecord::runDIGETPRIMCONF(Signal* signal){
+ jam();
+
+ const Uint32 userPtr = signal->theData[0];
+ //const Uint32 senderData = signal->theData[1];
+ const Uint32 nodeCount = signal->theData[6];
+ const Uint32 tableId = signal->theData[7];
+ const Uint32 fragNo = signal->theData[8];
+
+ ndbrequire(userPtr == RNIL && signal->length() == 9);
+ ndbrequire(nodeCount > 0 && nodeCount <= MAX_REPLICAS);
+
+ TablePtr tabPtr;
+ ndbrequire(suma.c_tables.find(tabPtr, tableId));
+ LocalDataBuffer<15> fragBuf(suma.c_dataBufferPool, tabPtr.p->m_fragments);
+
+ /**
+ * Add primary node for fragment to list
+ */
+ FragmentDescriptor fd;
+ fd.m_fragDesc.m_nodeId = signal->theData[2];
+ fd.m_fragDesc.m_fragmentNo = fragNo;
+ signal->theData[2] = fd.m_dummy;
+ fragBuf.append(&signal->theData[2], 1);
+
+ const Uint32 nextFrag = fragNo + 1;
+ if(nextFrag == m_currentFragment){
+ /**
+ * Complete frag info for table
+ */
+ m_currentTable++;
+ nextMeta(signal);
+ return;
+ }
+ signal->theData[0] = RNIL;
+ signal->theData[1] = ptrI;
+ signal->theData[2] = tableId;
+ signal->theData[3] = nextFrag; // Frag no
+ suma.sendSignal(DBDIH_REF, GSN_DIGETPRIMREQ, signal, 4, JBB);
+}
+
+void
+SumaParticipant::SyncRecord::completeMeta(Signal* signal){
+ jam();
+ SubscriptionPtr subPtr;
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ ndbrequire(subPtr.p->m_syncPtrI == ptrI);
+
+#if PRINT_ONLY
+ ndbout_c("GSN_SUB_SYNC_CONF (meta)");
+#else
+
+ suma.releaseSections(signal);
+
+ if (m_error) {
+ SubSyncRef * const ref = (SubSyncRef*)signal->getDataPtrSend();
+ ref->subscriptionId = subPtr.p->m_subscriptionId;
+ ref->subscriptionKey = subPtr.p->m_subscriptionKey;
+ ref->part = SubscriptionData::MetaData;
+ ref->subscriberData = subPtr.p->m_subscriberData;
+ ref->errorCode = SubSyncRef::Undefined;
+ suma.sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_SYNC_REF, signal,
+ SubSyncRef::SignalLength, JBB);
+ } else {
+ SubSyncConf * const conf = (SubSyncConf*)signal->getDataPtrSend();
+ conf->subscriptionId = subPtr.p->m_subscriptionId;
+ conf->subscriptionKey = subPtr.p->m_subscriptionKey;
+ conf->part = SubscriptionData::MetaData;
+ conf->subscriberData = subPtr.p->m_subscriberData;
+ suma.sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_SYNC_CONF, signal,
+ SubSyncConf::SignalLength, JBB);
+ }
+#endif
+}
+
+/**********************************************************
+ *
+ * Scan interface
+ *
+ */
+
+void
+SumaParticipant::SyncRecord::startScan(Signal* signal){
+ jam();
+
+ /**
+ * Get fraginfo
+ */
+ m_currentTable = 0;
+ m_currentFragment = 0;
+
+ nextScan(signal);
+}
+
+bool
+SumaParticipant::SyncRecord::getNextFragment(TablePtr * tab,
+ FragmentDescriptor * fd){
+ jam();
+ SubscriptionPtr subPtr;
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ TableList::DataBufferIterator tabIt;
+ DataBuffer<15>::DataBufferIterator fragIt;
+
+ m_tableList.position(tabIt, m_currentTable);
+ for(; !tabIt.curr.isNull(); m_tableList.next(tabIt), m_currentTable++){
+ TablePtr tabPtr;
+ ndbrequire(suma.c_tables.find(tabPtr, * tabIt.data));
+ if(subPtr.p->m_subscriptionType == SubCreateReq::SelectiveTableSnapshot)
+ {
+ if(!subPtr.p->m_tables[tabPtr.p->m_tableId]) {
+ *tab = tabPtr;
+ return true;
+ }
+ }
+ LocalDataBuffer<15> fragBuf(suma.c_dataBufferPool, tabPtr.p->m_fragments);
+
+ fragBuf.position(fragIt, m_currentFragment);
+ for(; !fragIt.curr.isNull(); fragBuf.next(fragIt), m_currentFragment++){
+ FragmentDescriptor tmp;
+ tmp.m_dummy = * fragIt.data;
+ if(tmp.m_fragDesc.m_nodeId == suma.getOwnNodeId()){
+ * fd = tmp;
+ * tab = tabPtr;
+ return true;
+ }
+ }
+ m_currentFragment = 0;
+ }
+ return false;
+}
+
+void
+SumaParticipant::SyncRecord::nextScan(Signal* signal){
+ jam();
+ TablePtr tabPtr;
+ FragmentDescriptor fd;
+ SubscriptionPtr subPtr;
+ if(!getNextFragment(&tabPtr, &fd)){
+ jam();
+ completeScan(signal);
+ return;
+ }
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ ndbrequire(subPtr.p->m_syncPtrI == ptrI);
+
+ if(subPtr.p->m_subscriptionType == SubCreateReq::SelectiveTableSnapshot) {
+ jam();
+ if(!subPtr.p->m_tables[tabPtr.p->m_tableId]) {
+ /*
+ * table is not part of the subscription. Check next table
+ */
+ m_currentTable++;
+ nextScan(signal);
+ return;
+ }
+ }
+
+ DataBuffer<15>::Head head = m_attributeList;
+ if(head.getSize() == 0){
+ head = tabPtr.p->m_attributes;
+ }
+ LocalDataBuffer<15> attrBuf(suma.c_dataBufferPool, head);
+
+ ScanFragReq * req = (ScanFragReq *)signal->getDataPtrSend();
+ const Uint32 parallelism = 16;
+ const Uint32 attrLen = 5 + attrBuf.getSize();
+
+ req->senderData = m_subscriptionPtrI;
+ req->resultRef = suma.reference();
+ req->tableId = tabPtr.p->m_tableId;
+ req->requestInfo = 0;
+ req->savePointId = 0;
+ ScanFragReq::setConcurrency(req->requestInfo, parallelism);
+ ScanFragReq::setLockMode(req->requestInfo, 0);
+ ScanFragReq::setHoldLockFlag(req->requestInfo, 0);
+ ScanFragReq::setKeyinfoFlag(req->requestInfo, 0);
+ ScanFragReq::setAttrLen(req->requestInfo, attrLen);
+ req->fragmentNo = fd.m_fragDesc.m_fragmentNo;
+ req->schemaVersion = tabPtr.p->m_schemaVersion;
+ req->transId1 = 0;
+ req->transId2 = (SUMA << 20) + (suma.getOwnNodeId() << 8);
+
+ for(unsigned int i = 0; i<parallelism; i++){
+ req->clientOpPtr[i] = (ptrI << 16) + (i + 1);
+ }
+ suma.sendSignal(DBLQH_REF, GSN_SCAN_FRAGREQ, signal, 25, JBB);
+
+ signal->theData[0] = ptrI;
+ signal->theData[1] = 0;
+ signal->theData[2] = (SUMA << 20) + (suma.getOwnNodeId() << 8);
+
+ // Return all
+ signal->theData[3] = attrBuf.getSize();
+ signal->theData[4] = 0;
+ signal->theData[5] = 0;
+ signal->theData[6] = 0;
+ signal->theData[7] = 0;
+
+ Uint32 dataPos = 8;
+ DataBuffer<15>::DataBufferIterator it;
+ for(attrBuf.first(it); !it.curr.isNull(); attrBuf.next(it)){
+ AttributeHeader::init(&signal->theData[dataPos++], * it.data, 0);
+ if(dataPos == 25){
+ suma.sendSignal(DBLQH_REF, GSN_ATTRINFO, signal, 25, JBB);
+ dataPos = 3;
+ }
+ }
+ if(dataPos != 3){
+ suma.sendSignal(DBLQH_REF, GSN_ATTRINFO, signal, dataPos, JBB);
+ }
+
+ m_currentTableId = tabPtr.p->m_tableId;
+ m_currentNoOfAttributes = attrBuf.getSize();
+}
+
+
+void
+SumaParticipant::execSCAN_FRAGREF(Signal* signal){
+ jamEntry();
+
+// ScanFragRef * const ref = (ScanFragRef*)signal->getDataPtr();
+ ndbrequire(false);
+}
+
+void
+SumaParticipant::execSCAN_FRAGCONF(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13011);
+
+ ScanFragConf * const conf = (ScanFragConf*)signal->getDataPtr();
+
+ const Uint32 completed = conf->fragmentCompleted;
+ const Uint32 senderData = conf->senderData;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, senderData);
+
+ if(completed != 2){
+ jam();
+
+#if PRINT_ONLY
+ SubSyncContinueConf * const conf =
+ (SubSyncContinueConf*)signal->getDataPtrSend();
+ conf->subscriptionId = subPtr.p->m_subscriptionId;
+ conf->subscriptionKey = subPtr.p->m_subscriptionKey;
+ execSUB_SYNC_CONTINUE_CONF(signal);
+#else
+ SubSyncContinueReq * const req = (SubSyncContinueReq*)signal->getDataPtrSend();
+ req->subscriberData = subPtr.p->m_subscriberData;
+ req->noOfRowsSent = 0; //rowCount;
+ sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_SYNC_CONTINUE_REQ, signal,
+ SubSyncContinueReq::SignalLength, JBB);
+#endif
+ return;
+ }
+
+ SyncRecord* tmp = c_syncPool.getPtr(subPtr.p->m_syncPtrI);
+
+ tmp->m_currentFragment++;
+ tmp->nextScan(signal);
+}
+
+void
+SumaParticipant::execSUB_SYNC_CONTINUE_CONF(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13012);
+
+ SubSyncContinueConf * const conf =
+ (SubSyncContinueConf*)signal->getDataPtr();
+
+ SubscriptionPtr subPtr;
+ Subscription key;
+ key.m_subscriptionId = conf->subscriptionId;
+ key.m_subscriptionKey = conf->subscriptionKey;
+
+ ndbrequire(c_subscriptions.find(subPtr, key));
+
+ ScanFragNextReq * req = (ScanFragNextReq *)signal->getDataPtrSend();
+ req->senderData = subPtr.i;
+ req->closeFlag = 0;
+ req->transId1 = 0;
+ req->transId2 = (SUMA << 20) + (getOwnNodeId() << 8);
+ sendSignal(DBLQH_REF, GSN_SCAN_NEXTREQ, signal,
+ ScanFragNextReq::SignalLength, JBB);
+}
+
+void
+SumaParticipant::SyncRecord::completeScan(Signal* signal){
+ jam();
+ // m_tableList.release();
+
+ SubscriptionPtr subPtr;
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ ndbrequire(subPtr.p->m_syncPtrI == ptrI);
+
+#if PRINT_ONLY
+ ndbout_c("GSN_SUB_SYNC_CONF (data)");
+#else
+ SubSyncConf * const conf = (SubSyncConf*)signal->getDataPtrSend();
+ conf->subscriptionId = subPtr.p->m_subscriptionId;
+ conf->subscriptionKey = subPtr.p->m_subscriptionKey;
+ conf->part = SubscriptionData::TableData;
+ conf->subscriberData = subPtr.p->m_subscriberData;
+ suma.sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_SYNC_CONF, signal,
+ SubSyncConf::SignalLength, JBB);
+#endif
+}
+
+void
+SumaParticipant::execSCAN_HBREP(Signal* signal){
+ jamEntry();
+#if 0
+ ndbout << "execSCAN_HBREP" << endl << hex;
+ for(int i = 0; i<signal->length(); i++){
+ ndbout << signal->theData[i] << " ";
+ if(((i + 1) % 8) == 0)
+ ndbout << endl << hex;
+ }
+ ndbout << endl;
+#endif
+}
+
+/**********************************************************
+ *
+ * Suma participant interface
+ *
+ * Creation of subscriber
+ *
+ */
+
+void
+SumaParticipant::execSUB_START_REQ(Signal* signal){
+ jamEntry();
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::execSUB_START_REQ");
+#endif
+
+ CRASH_INSERTION(13013);
+
+ if (c_restartLock) {
+ jam();
+ // ndbout_c("c_restartLock");
+ if (RtoI(signal->getSendersBlockRef(), false) == RNIL) {
+ jam();
+ sendSubStartRef(signal, /** Error Code */ 0, true);
+ return;
+ }
+ // only allow other Suma's in the nodegroup to come through for restart purposes
+ }
+
+ Subscription key;
+
+ SubStartReq * const req = (SubStartReq*)signal->getDataPtr();
+
+ Uint32 senderRef = req->senderRef;
+ Uint32 senderData = req->senderData;
+ Uint32 subscriberData = req->subscriberData;
+ Uint32 subscriberRef = req->subscriberRef;
+ SubscriptionData::Part part = (SubscriptionData::Part)req->part;
+ key.m_subscriptionId = req->subscriptionId;
+ key.m_subscriptionKey = req->subscriptionKey;
+
+ SubscriptionPtr subPtr;
+ if(!c_subscriptions.find(subPtr, key)){
+ jam();
+ sendSubStartRef(signal, /** Error Code */ 0);
+ return;
+ }
+
+ Ptr<SyncRecord> syncPtr;
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+ if (syncPtr.p->m_locked) {
+ jam();
+#if 0
+ ndbout_c("Locked");
+#endif
+ sendSubStartRef(signal, /** Error Code */ 0, true);
+ return;
+ }
+ syncPtr.p->m_locked = true;
+
+ SubscriberPtr subbPtr;
+ if(!c_subscriberPool.seize(subbPtr)){
+ jam();
+ syncPtr.p->m_locked = false;
+ sendSubStartRef(signal, /** Error Code */ 0);
+ return;
+ }
+
+ Uint32 type = subPtr.p->m_subscriptionType;
+
+ subbPtr.p->m_senderRef = senderRef;
+ subbPtr.p->m_senderData = senderData;
+
+ switch (type) {
+ case SubCreateReq::TableEvent:
+ jam();
+ // we want the data to return to the API not DICT
+ subbPtr.p->m_subscriberRef = subscriberRef;
+ // ndbout_c("start ref = %u", signal->getSendersBlockRef());
+ // ndbout_c("ref = %u", subbPtr.p->m_subscriberRef);
+ // we use the subscription id for now, should really be API choice
+ subbPtr.p->m_subscriberData = subscriberData;
+
+#if 0
+ if (RtoI(signal->getSendersBlockRef(), false) == RNIL) {
+ jam();
+ for (Uint32 i = 0; i < c_noNodesInGroup; i++) {
+ Uint32 ref = calcSumaBlockRef(c_nodesInGroup[i]);
+ if (ref != reference()) {
+ jam();
+ sendSubStartReq(subPtr, subbPtr, signal, ref);
+ } else
+ jam();
+ }
+ }
+#endif
+ break;
+ case SubCreateReq::DatabaseSnapshot:
+ case SubCreateReq::SelectiveTableSnapshot:
+ jam();
+ subbPtr.p->m_subscriberRef = GREP_REF;
+ subbPtr.p->m_subscriberData = subPtr.p->m_subscriberData;
+ break;
+ case SubCreateReq::SingleTableScan:
+ jam();
+ subbPtr.p->m_subscriberRef = subPtr.p->m_subscriberRef;
+ subbPtr.p->m_subscriberData = subPtr.p->m_subscriberData;
+ }
+
+ subbPtr.p->m_subPtrI = subPtr.i;
+ subbPtr.p->m_firstGCI = RNIL;
+ if (type == SubCreateReq::TableEvent)
+ subbPtr.p->m_lastGCI = 0;
+ else
+ subbPtr.p->m_lastGCI = RNIL; // disable usage of m_lastGCI
+ bool ok = false;
+
+ switch(part){
+ case SubscriptionData::MetaData:
+ ok = true;
+ jam();
+ c_metaSubscribers.add(subbPtr);
+ sendSubStartComplete(signal, subbPtr, 0, part);
+ break;
+ case SubscriptionData::TableData:
+ ok = true;
+ jam();
+ c_prepDataSubscribers.add(subbPtr);
+ syncPtr.p->startTrigger(signal);
+ break;
+ }
+ ndbrequire(ok);
+}
+
+void
+SumaParticipant::sendSubStartComplete(Signal* signal,
+ SubscriberPtr subbPtr,
+ Uint32 firstGCI,
+ SubscriptionData::Part part){
+ jam();
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subbPtr.p->m_subPtrI);
+
+ Ptr<SyncRecord> syncPtr;
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+ syncPtr.p->m_locked = false;
+
+ SubStartConf * const conf = (SubStartConf*)signal->getDataPtrSend();
+
+ conf->senderRef = reference();
+ conf->senderData = subbPtr.p->m_senderData;
+ conf->subscriptionId = subPtr.p->m_subscriptionId;
+ conf->subscriptionKey = subPtr.p->m_subscriptionKey;
+ conf->firstGCI = firstGCI;
+ conf->part = (Uint32) part;
+
+ conf->subscriberData = subPtr.p->m_subscriberData;
+ sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_START_CONF, signal,
+ SubStartConf::SignalLength, JBB);
+}
+
+#if 0
+void
+SumaParticipant::sendSubStartRef(SubscriptionPtr subPtr,
+ Signal* signal, Uint32 errCode,
+ bool temporary){
+ jam();
+ SubStartRef * ref = (SubStartRef *)signal->getDataPtrSend();
+ xxx ref->senderRef = reference();
+ xxx ref->senderData = subPtr.p->m_senderData;
+ ref->subscriptionId = subPtr.p->m_subscriptionId;
+ ref->subscriptionKey = subPtr.p->m_subscriptionKey;
+ ref->part = (Uint32) subPtr.p->m_subscriptionType;
+ ref->subscriberData = subPtr.p->m_subscriberData;
+ ref->err = errCode;
+ if (temporary) {
+ jam();
+ ref->setTemporary();
+ }
+ releaseSections(signal);
+ sendSignal(subPtr.p->m_subscriberRef, GSN_SUB_START_REF, signal,
+ SubStartRef::SignalLength, JBB);
+}
+#endif
+void
+SumaParticipant::sendSubStartRef(Signal* signal, Uint32 errCode,
+ bool temporary){
+ jam();
+ SubStartRef * ref = (SubStartRef *)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->err = errCode;
+ if (temporary) {
+ jam();
+ ref->setTemporary();
+ }
+ releaseSections(signal);
+ sendSignal(signal->getSendersBlockRef(), GSN_SUB_START_REF, signal,
+ SubStartRef::SignalLength, JBB);
+}
+
+/**********************************************************
+ *
+ * Trigger admin interface
+ *
+ */
+
+void
+SumaParticipant::SyncRecord::startTrigger(Signal* signal){
+ jam();
+ m_currentTable = 0;
+ m_latestTriggerId = RNIL;
+ nextTrigger(signal);
+}
+
+void
+SumaParticipant::SyncRecord::nextTrigger(Signal* signal){
+ jam();
+
+ TableList::DataBufferIterator it;
+
+ if(!m_tableList.position(it, m_currentTable)){
+ completeTrigger(signal);
+ return;
+ }
+
+ SubscriptionPtr subPtr;
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ ndbrequire(subPtr.p->m_syncPtrI == ptrI);
+ const Uint32 RT_BREAK = 48;
+ Uint32 latestTriggerId = 0;
+ for(Uint32 i = 0; i<RT_BREAK && !it.isNull(); i++, m_tableList.next(it)){
+ TablePtr tabPtr;
+#if 0
+ ndbout_c("nextTrigger tableid %u", *it.data);
+#endif
+ ndbrequire(suma.c_tables.find(tabPtr, *it.data));
+
+ AttributeMask attrMask;
+ createAttributeMask(attrMask, tabPtr.p);
+
+ for(Uint32 j = 0; j<3; j++){
+ i++;
+ latestTriggerId = (tabPtr.p->m_schemaVersion << 18) |
+ (j << 16) | tabPtr.p->m_tableId;
+ if(tabPtr.p->m_hasTriggerDefined[j] == 0) {
+ ndbrequire(tabPtr.p->m_triggerIds[j] == ILLEGAL_TRIGGER_ID);
+#if 0
+ ndbout_c("DEFINING trigger on table %u[%u]", tabPtr.p->m_tableId, j);
+#endif
+ CreateTrigReq * const req = (CreateTrigReq*)signal->getDataPtrSend();
+ req->setUserRef(SUMA_REF);
+ req->setConnectionPtr(ptrI);
+ req->setTriggerType(TriggerType::SUBSCRIPTION_BEFORE);
+ req->setTriggerActionTime(TriggerActionTime::TA_DETACHED);
+ req->setMonitorReplicas(true);
+ req->setMonitorAllAttributes(false);
+ req->setReceiverRef(SUMA_REF);
+ req->setTriggerId(latestTriggerId);
+ req->setTriggerEvent((TriggerEvent::Value)j);
+ req->setTableId(tabPtr.p->m_tableId);
+ req->setAttributeMask(attrMask);
+ suma.sendSignal(DBTUP_REF, GSN_CREATE_TRIG_REQ,
+ signal, CreateTrigReq::SignalLength, JBB);
+
+ } else {
+ /**
+ * Faking that a trigger has been created in order to
+ * simulate the proper behaviour.
+ * Perhaps this should be a dummy signal instead of
+ * (ab)using CREATE_TRIG_CONF.
+ */
+ CreateTrigConf * conf = (CreateTrigConf*)signal->getDataPtrSend();
+ conf->setConnectionPtr(ptrI);
+ conf->setTableId(tabPtr.p->m_tableId);
+ conf->setTriggerId(latestTriggerId);
+ suma.sendSignal(SUMA_REF,GSN_CREATE_TRIG_CONF,
+ signal, CreateTrigConf::SignalLength, JBB);
+
+ }
+
+ }
+ m_currentTable++;
+ }
+ m_latestTriggerId = latestTriggerId;
+}
+
+void
+SumaParticipant::SyncRecord::createAttributeMask(AttributeMask& mask,
+ Table * table){
+ jam();
+ mask.clear();
+ DataBuffer<15>::DataBufferIterator it;
+ LocalDataBuffer<15> attrBuf(suma.c_dataBufferPool, table->m_attributes);
+ for(attrBuf.first(it); !it.curr.isNull(); attrBuf.next(it)){
+ mask.set(* it.data);
+ }
+}
+
+void
+SumaParticipant::SyncRecord::runCREATE_TRIG_CONF(Signal* signal){
+ jam();
+
+ CreateTrigConf * const conf = (CreateTrigConf*)signal->getDataPtr();
+ const Uint32 triggerId = conf->getTriggerId();
+ Uint32 type = (triggerId >> 16) & 0x3;
+ Uint32 tableId = conf->getTableId();
+
+ TablePtr tabPtr;
+ ndbrequire(suma.c_tables.find(tabPtr, tableId));
+
+ ndbrequire(type < 3);
+ tabPtr.p->m_triggerIds[type] = triggerId;
+ tabPtr.p->m_hasTriggerDefined[type]++;
+
+ if(triggerId == m_latestTriggerId){
+ jam();
+ nextTrigger(signal);
+ }
+}
+
+void
+SumaParticipant::SyncRecord::completeTrigger(Signal* signal){
+ jam();
+ SubscriptionPtr subPtr;
+ CRASH_INSERTION(13013);
+#ifdef EVENT_PH3_DEBUG
+ ndbout_c("SumaParticipant: trigger completed");
+#endif
+ Uint32 gci;
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ ndbrequire(subPtr.p->m_syncPtrI == ptrI);
+
+ SubscriberPtr subbPtr;
+ {
+ bool found = false;
+
+ for(suma.c_prepDataSubscribers.first(subbPtr);
+ !subbPtr.isNull(); suma.c_prepDataSubscribers.next(subbPtr)) {
+ jam();
+ if(subbPtr.p->m_subPtrI == subPtr.i) {
+ jam();
+ found = true;
+ break;
+ }
+ }
+ ndbrequire(found);
+ gci = suma.getFirstGCI(signal);
+ subbPtr.p->m_firstGCI = gci;
+ suma.c_prepDataSubscribers.remove(subbPtr);
+ suma.c_dataSubscribers.add(subbPtr);
+ }
+ suma.sendSubStartComplete(signal, subbPtr, gci, SubscriptionData::TableData);
+}
+
+void
+SumaParticipant::SyncRecord::startDropTrigger(Signal* signal){
+ jam();
+ m_currentTable = 0;
+ m_latestTriggerId = RNIL;
+ nextDropTrigger(signal);
+}
+
+void
+SumaParticipant::SyncRecord::nextDropTrigger(Signal* signal){
+ jam();
+
+ TableList::DataBufferIterator it;
+
+ if(!m_tableList.position(it, m_currentTable)){
+ completeDropTrigger(signal);
+ return;
+ }
+
+ SubscriptionPtr subPtr;
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ ndbrequire(subPtr.p->m_syncPtrI == ptrI);
+
+ const Uint32 RT_BREAK = 48;
+ Uint32 latestTriggerId = 0;
+ for(Uint32 i = 0; i<RT_BREAK && !it.isNull(); i++, m_tableList.next(it)){
+ jam();
+ TablePtr tabPtr;
+#if 0
+ ndbout_c("nextDropTrigger tableid %u", *it.data);
+#endif
+ ndbrequire(suma.c_tables.find(tabPtr, * it.data));
+
+ for(Uint32 j = 0; j<3; j++){
+ jam();
+ ndbrequire(tabPtr.p->m_triggerIds[j] != ILLEGAL_TRIGGER_ID);
+ i++;
+ latestTriggerId = tabPtr.p->m_triggerIds[j];
+ if(tabPtr.p->m_hasTriggerDefined[j] == 1) {
+ jam();
+
+ DropTrigReq * const req = (DropTrigReq*)signal->getDataPtrSend();
+ req->setConnectionPtr(ptrI);
+ req->setUserRef(SUMA_REF); // Sending to myself
+ req->setRequestType(DropTrigReq::RT_USER);
+ req->setTriggerType(TriggerType::SUBSCRIPTION_BEFORE);
+ req->setTriggerActionTime(TriggerActionTime::TA_DETACHED);
+ req->setIndexId(RNIL);
+
+ req->setTableId(tabPtr.p->m_tableId);
+ req->setTriggerId(latestTriggerId);
+ req->setTriggerEvent((TriggerEvent::Value)j);
+
+#if 0
+ ndbout_c("DROPPING trigger %u = %u %u %u on table %u[%u]",
+ latestTriggerId,TriggerType::SUBSCRIPTION_BEFORE,
+ TriggerActionTime::TA_DETACHED, j, tabPtr.p->m_tableId, j);
+#endif
+ suma.sendSignal(DBTUP_REF, GSN_DROP_TRIG_REQ,
+ signal, DropTrigReq::SignalLength, JBB);
+ } else {
+ jam();
+ ndbrequire(tabPtr.p->m_hasTriggerDefined[j] > 1);
+ /**
+ * Faking that a trigger has been dropped in order to
+ * simulate the proper behaviour.
+ * Perhaps this should be a dummy signal instead of
+ * (ab)using DROP_TRIG_CONF.
+ */
+ DropTrigConf * conf = (DropTrigConf*)signal->getDataPtrSend();
+ conf->setConnectionPtr(ptrI);
+ conf->setTableId(tabPtr.p->m_tableId);
+ conf->setTriggerId(latestTriggerId);
+ suma.sendSignal(SUMA_REF,GSN_DROP_TRIG_CONF,
+ signal, DropTrigConf::SignalLength, JBB);
+ }
+ }
+ m_currentTable++;
+ }
+ m_latestTriggerId = latestTriggerId;
+}
+
+void
+SumaParticipant::SyncRecord::runDROP_TRIG_REF(Signal* signal){
+ jam();
+ DropTrigRef * const ref = (DropTrigRef*)signal->getDataPtr();
+ if (ref->getErrorCode() != DropTrigRef::TriggerNotFound){
+ ndbrequire(false);
+ }
+ const Uint32 triggerId = ref->getTriggerId();
+ Uint32 tableId = ref->getTableId();
+ runDropTrig(signal, triggerId, tableId);
+}
+
+void
+SumaParticipant::SyncRecord::runDROP_TRIG_CONF(Signal* signal){
+ jam();
+
+ DropTrigConf * const conf = (DropTrigConf*)signal->getDataPtr();
+ const Uint32 triggerId = conf->getTriggerId();
+ Uint32 tableId = conf->getTableId();
+ runDropTrig(signal, triggerId, tableId);
+}
+
+void
+SumaParticipant::SyncRecord::runDropTrig(Signal* signal,
+ Uint32 triggerId,
+ Uint32 tableId){
+ Uint32 type = (triggerId >> 16) & 0x3;
+
+ TablePtr tabPtr;
+ ndbrequire(suma.c_tables.find(tabPtr, tableId));
+
+ ndbrequire(type < 3);
+ ndbrequire(tabPtr.p->m_triggerIds[type] == triggerId);
+ tabPtr.p->m_hasTriggerDefined[type]--;
+ if (tabPtr.p->m_hasTriggerDefined[type] == 0) {
+ jam();
+ tabPtr.p->m_triggerIds[type] = ILLEGAL_TRIGGER_ID;
+ }
+ if(triggerId == m_latestTriggerId){
+ jam();
+ nextDropTrigger(signal);
+ }
+}
+
+void
+SumaParticipant::SyncRecord::completeDropTrigger(Signal* signal){
+ jam();
+ SubscriptionPtr subPtr;
+ CRASH_INSERTION(13014);
+#if 0
+ ndbout_c("trigger completed");
+#endif
+
+ suma.c_subscriptions.getPtr(subPtr, m_subscriptionPtrI);
+ ndbrequire(subPtr.p->m_syncPtrI == ptrI);
+
+ bool found = false;
+ SubscriberPtr subbPtr;
+ for(suma.c_prepDataSubscribers.first(subbPtr);
+ !subbPtr.isNull(); suma.c_prepDataSubscribers.next(subbPtr)) {
+ jam();
+ if(subbPtr.p->m_subPtrI == subPtr.i) {
+ jam();
+ found = true;
+ break;
+ }
+ }
+ ndbrequire(found);
+ suma.sendSubStopComplete(signal, subbPtr);
+}
+
+/**********************************************************
+ * Scan data interface
+ *
+ * Assumption: one execTRANSID_AI contains all attr info
+ *
+ */
+
+#define SUMA_BUF_SZ1 MAX_KEY_SIZE_IN_WORDS + MAX_TUPLE_SIZE_IN_WORDS
+#define SUMA_BUF_SZ MAX_ATTRIBUTES_IN_TABLE + SUMA_BUF_SZ1
+
+static Uint32 f_bufferLock = 0;
+static Uint32 f_buffer[SUMA_BUF_SZ];
+static Uint32 f_trigBufferSize = 0;
+static Uint32 b_bufferLock = 0;
+static Uint32 b_buffer[SUMA_BUF_SZ];
+static Uint32 b_trigBufferSize = 0;
+
+void
+SumaParticipant::execTRANSID_AI(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13015);
+ TransIdAI * const data = (TransIdAI*)signal->getDataPtr();
+ const Uint32 opPtrI = data->connectPtr;
+ const Uint32 length = signal->length() - 3;
+
+ if(f_bufferLock == 0){
+ f_bufferLock = opPtrI;
+ } else {
+ ndbrequire(f_bufferLock == opPtrI);
+ }
+
+ Ptr<SyncRecord> syncPtr;
+ c_syncPool.getPtr(syncPtr, (opPtrI >> 16));
+
+ Uint32 sum = 0;
+ Uint32 * dst = f_buffer + MAX_ATTRIBUTES_IN_TABLE;
+ Uint32 * headers = f_buffer;
+ const Uint32 * src = &data->attrData[0];
+ const Uint32 * const end = &src[length];
+
+ const Uint32 attribs = syncPtr.p->m_currentNoOfAttributes;
+ for(Uint32 i = 0; i<attribs; i++){
+ Uint32 tmp = * src++;
+ * headers++ = tmp;
+ Uint32 len = AttributeHeader::getDataSize(tmp);
+
+ memcpy(dst, src, 4 * len);
+ dst += len;
+ src += len;
+ sum += len;
+ }
+
+ ndbrequire(src == end);
+
+ /**
+ * Send data to subscriber
+ */
+ LinearSectionPtr ptr[3];
+ ptr[0].p = f_buffer;
+ ptr[0].sz = attribs;
+
+ ptr[1].p = f_buffer + MAX_ATTRIBUTES_IN_TABLE;
+ ptr[1].sz = sum;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, syncPtr.p->m_subscriptionPtrI);
+
+ /**
+ * Initialize signal
+ */
+ SubTableData * sdata = (SubTableData*)signal->getDataPtrSend();
+ Uint32 ref = subPtr.p->m_subscriberRef;
+ sdata->tableId = syncPtr.p->m_currentTableId;
+ sdata->senderData = subPtr.p->m_subscriberData;
+ sdata->operation = 3; // Scan
+ sdata->gci = 1; // Undefined
+#if PRINT_ONLY
+ ndbout_c("GSN_SUB_TABLE_DATA (scan) #attr: %d len: %d", attribs, sum);
+#else
+ sendSignal(ref,
+ GSN_SUB_TABLE_DATA,
+ signal,
+ SubTableData::SignalLength, JBB,
+ ptr, 2);
+#endif
+
+ /**
+ * Reset f_bufferLock
+ */
+ f_bufferLock = 0;
+}
+
+/**********************************************************
+ *
+ * Trigger data interface
+ *
+ */
+
+void
+SumaParticipant::execTRIG_ATTRINFO(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13016);
+ TrigAttrInfo* const trg = (TrigAttrInfo*)signal->getDataPtr();
+ const Uint32 trigId = trg->getTriggerId();
+
+ const Uint32 dataLen = signal->length() - TrigAttrInfo::StaticLength;
+
+ if(trg->getAttrInfoType() == TrigAttrInfo::BEFORE_VALUES){
+ jam();
+
+ ndbrequire(b_bufferLock == trigId);
+
+ memcpy(b_buffer + b_trigBufferSize, trg->getData(), 4 * dataLen);
+ b_trigBufferSize += dataLen;
+ // printf("before values %u %u %u\n",trigId, dataLen, b_trigBufferSize);
+ } else {
+ jam();
+
+ if(f_bufferLock == 0){
+ f_bufferLock = trigId;
+ f_trigBufferSize = 0;
+ b_bufferLock = trigId;
+ b_trigBufferSize = 0;
+ } else {
+ ndbrequire(f_bufferLock == trigId);
+ }
+
+ memcpy(f_buffer + f_trigBufferSize, trg->getData(), 4 * dataLen);
+ f_trigBufferSize += dataLen;
+ }
+}
+
+#ifdef NODEFAIL_DEBUG2
+static int theCounts[64] = {0};
+#endif
+
+Uint32
+Suma::getStoreBucket(Uint32 v)
+{
+ // id will contain id to responsible suma or
+ // RNIL if we don't have nodegroup info yet
+
+ const Uint32 N = NO_OF_BUCKETS;
+ const Uint32 D = v % N; // Distibution key
+ return D;
+}
+
+Uint32
+Suma::getResponsibleSumaNodeId(Uint32 D)
+{
+ // id will contain id to responsible suma or
+ // RNIL if we don't have nodegroup info yet
+
+ Uint32 id;
+
+ if (c_restartLock) {
+ jam();
+ // ndbout_c("c_restartLock");
+ id = RNIL;
+ } else {
+ jam();
+ const Uint32 n = c_noNodesInGroup; // Number nodes in node group
+ const Uint32 C1 = D / n;
+ const Uint32 C2 = D - C1*n; // = D % n;
+ const Uint32 C = C2 + C1 % n;
+ for (Uint32 i = 0; i < n; i++) {
+ jam();
+ id = c_nodesInGroup[(C + i) % n];
+ if (c_aliveNodes.get(id) &&
+ !c_preparingNodes.get(id)) {
+ jam();
+ break;
+ }//if
+ }
+#ifdef NODEFAIL_DEBUG2
+ theCounts[id]++;
+ ndbout_c("Suma:responsible n=%u, D=%u, id = %u, count=%u",
+ n,D, id, theCounts[id]);
+#endif
+ }
+ return id;
+}
+
+Uint32
+SumaParticipant::decideWhoToSend(Uint32 nBucket, Uint32 gci){
+ bool replicaFlag = true;
+ Uint32 nId = RNIL;
+
+ // bucket active/not active set by GCP_COMPLETE
+ if (c_buckets[nBucket].active) {
+ if (c_buckets[nBucket].handover && c_buckets[nBucket].handoverGCI <= gci) {
+ jam();
+ replicaFlag = true; // let the other node send this
+ nId = RNIL;
+ // mark this as started, if we get a node failiure now we have some lost stuff
+ c_buckets[nBucket].handover_started = true;
+ } else {
+ jam();
+ replicaFlag = false;
+ nId = refToNode(reference());
+ }
+ } else {
+ nId = getResponsibleSumaNodeId(nBucket);
+ replicaFlag = !(nId == refToNode(reference()));
+
+ if (!replicaFlag) {
+ if (!c_buckets[nBucket].handover) {
+ jam();
+ // appearently a node has failed and we are taking over sending
+ // from that bucket. Now we need to go back to latest completed
+ // GCI. Handling will depend on Subscriber and Subscription
+
+ // TODO, for now we make an easy takeover
+ if (gci < c_nodeFailGCI)
+ c_lastInconsistentGCI = gci;
+
+ // we now have responsability for this bucket and we're actively
+ // sending from that
+ c_buckets[nBucket].active = true;
+#ifdef HANDOVER_DEBUG
+ ndbout_c("Takeover Bucket %u", nBucket);
+#endif
+ } else if (c_buckets[nBucket].handoverGCI > gci) {
+ jam();
+ replicaFlag = true; // handover going on, but don't start sending yet
+ nId = RNIL;
+ } else {
+ jam();
+#ifdef HANDOVER_DEBUG
+ ndbout_c("Possible error: Will send from GCI = %u", gci);
+#endif
+ }
+ }
+ }
+
+#ifdef NODEFAIL_DEBUG2
+ ndbout_c("Suma:bucket %u, responsible id = %u, replicaFlag = %u",
+ nBucket, nId, (Uint32)replicaFlag);
+#endif
+ return replicaFlag;
+}
+
+void
+SumaParticipant::execFIRE_TRIG_ORD(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13016);
+ FireTrigOrd* const trg = (FireTrigOrd*)signal->getDataPtr();
+ const Uint32 trigId = trg->getTriggerId();
+ const Uint32 hashValue = trg->getHashValue();
+ const Uint32 gci = trg->getGCI();
+ const Uint32 event = trg->getTriggerEvent();
+ const Uint32 triggerId = trg->getTriggerId();
+ Uint32 tableId = triggerId & 0xFFFF;
+
+ ndbrequire(f_bufferLock == trigId);
+
+#ifdef EVENT_DEBUG2
+ ndbout_c("SumaParticipant::execFIRE_TRIG_ORD");
+#endif
+
+ Uint32 sz = trg->getNoOfPrimaryKeyWords()+trg->getNoOfAfterValueWords();
+ ndbrequire(sz == f_trigBufferSize);
+
+ /**
+ * Reformat as "all headers" + "all data"
+ */
+ Uint32 dataLen = 0;
+ Uint32 noOfAttrs = 0;
+ Uint32 * src = f_buffer;
+ Uint32 * headers = signal->theData + 25;
+ Uint32 * dst = signal->theData + 25 + MAX_ATTRIBUTES_IN_TABLE;
+
+ LinearSectionPtr ptr[3];
+ int nptr;
+
+ ptr[0].p = headers;
+ ptr[1].p = dst;
+
+ while(sz > 0){
+ jam();
+ Uint32 tmp = * src ++;
+ * headers ++ = tmp;
+ Uint32 len = AttributeHeader::getDataSize(tmp);
+ memcpy(dst, src, 4 * len);
+ dst += len;
+ src += len;
+
+ noOfAttrs++;
+ dataLen += len;
+ sz -= (1 + len);
+ }
+ ndbrequire(sz == 0);
+
+ ptr[0].sz = noOfAttrs;
+ ptr[1].sz = dataLen;
+
+ if (b_trigBufferSize > 0) {
+ jam();
+ ptr[2].p = b_buffer;
+ ptr[2].sz = b_trigBufferSize;
+ nptr = 3;
+ } else {
+ jam();
+ nptr = 2;
+ }
+
+ // right now only for tableEvent
+ bool replicaFlag = decideWhoToSend(getStoreBucket(hashValue), gci);
+
+ /**
+ * Signal to subscriber(s)
+ */
+ SubTableData * data = (SubTableData*)signal->getDataPtrSend();//trg;
+ data->gci = gci;
+ data->tableId = tableId;
+ data->operation = event;
+ data->noOfAttributes = noOfAttrs;
+ data->dataSize = dataLen;
+
+ SubscriberPtr subbPtr;
+ for(c_dataSubscribers.first(subbPtr); !subbPtr.isNull();
+ c_dataSubscribers.next(subbPtr)){
+ if (subbPtr.p->m_firstGCI > gci) {
+#ifdef EVENT_DEBUG
+ ndbout_c("m_firstGCI = %u, gci = %u", subbPtr.p->m_firstGCI, gci);
+#endif
+ jam();
+ // we're either restarting or it's a newly created subscriber
+ // and waiting for the right gci
+ continue;
+ }
+
+ jam();
+
+ const Uint32 ref = subbPtr.p->m_subscriberRef;
+ // ndbout_c("ref = %u", ref);
+ const Uint32 subdata = subbPtr.p->m_subscriberData;
+ data->senderData = subdata;
+ /*
+ * get subscription ptr for this subscriber
+ */
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subbPtr.p->m_subPtrI);
+
+ if(!subPtr.p->m_tables[tableId]) {
+ jam();
+ continue;
+ //continue in for-loop if the table is not part of
+ //the subscription. Otherwise, send data to subscriber.
+ }
+
+ if (subPtr.p->m_subscriptionType == SubCreateReq::TableEvent) {
+ if (replicaFlag) {
+ jam();
+ c_failoverBuffer.subTableData(gci,NULL,0);
+ continue;
+ }
+ jam();
+ Uint32 tmp = data->logType;
+ if (c_lastInconsistentGCI == data->gci) {
+ data->setGCINotConsistent();
+ }
+
+#ifdef HANDOVER_DEBUG
+ {
+ static int aLongGCIName = 0;
+ if (data->gci != aLongGCIName) {
+ aLongGCIName = data->gci;
+ ndbout_c("sent from GCI = %u", aLongGCIName);
+ }
+ }
+#endif
+ sendSignal(ref, GSN_SUB_TABLE_DATA, signal,
+ SubTableData::SignalLength, JBB, ptr, nptr);
+ data->logType = tmp;
+ } else {
+ ndbassert(refToNode(ref) == 0 || refToNode(ref) == getOwnNodeId());
+ jam();
+#if PRINT_ONLY
+ ndbout_c("GSN_SUB_TABLE_DATA to %s: op: %d #attr: %d len: %d",
+ getBlockName(refToBlock(ref)),
+ noOfAttrs, dataLen);
+
+#else
+#ifdef HANDOVER_DEBUG
+ {
+ static int aLongGCIName2 = 0;
+ if (data->gci != aLongGCIName2) {
+ aLongGCIName2 = data->gci;
+ ndbout_c("(EXECUTE_DIRECT) sent from GCI = %u to %u", aLongGCIName2, ref);
+ }
+ }
+#endif
+ EXECUTE_DIRECT(refToBlock(ref), GSN_SUB_TABLE_DATA, signal,
+ SubTableData::SignalLength);
+ jamEntry();
+#endif
+ }
+ }
+
+ /**
+ * Reset f_bufferLock
+ */
+ f_bufferLock = 0;
+ b_bufferLock = 0;
+}
+
+void
+SumaParticipant::execSUB_GCP_COMPLETE_REP(Signal* signal){
+ jamEntry();
+
+ SubGcpCompleteRep * rep = (SubGcpCompleteRep*)signal->getDataPtrSend();
+
+ Uint32 gci = rep->gci;
+ c_lastCompleteGCI = gci;
+
+ /**
+ * always send SUB_GCP_COMPLETE_REP to Grep (so
+ * Lars can do funky stuff calculating intervals,
+ * even before the subscription is started
+ */
+ rep->senderRef = reference();
+ rep->senderData = 0; //ignored in grep
+ EXECUTE_DIRECT(refToBlock(GREP_REF), GSN_SUB_GCP_COMPLETE_REP, signal,
+ SubGcpCompleteRep::SignalLength);
+
+ /**
+ * Signal to subscriber(s)
+ */
+
+ SubscriberPtr subbPtr;
+ SubscriptionPtr subPtr;
+ c_dataSubscribers.first(subbPtr);
+ for(; !subbPtr.isNull(); c_dataSubscribers.next(subbPtr)){
+
+ if (subbPtr.p->m_firstGCI > gci) {
+ jam();
+ // we don't send SUB_GCP_COMPLETE_REP for incomplete GCI's
+ continue;
+ }
+
+ const Uint32 ref = subbPtr.p->m_subscriberRef;
+ rep->senderRef = ref;
+ rep->senderData = subbPtr.p->m_subscriberData;
+
+ c_subscriptions.getPtr(subPtr, subbPtr.p->m_subPtrI);
+#if PRINT_ONLY
+ ndbout_c("GSN_SUB_GCP_COMPLETE_REP to %s:",
+ getBlockName(refToBlock(ref)));
+#else
+ /**
+ * Ignore sending to GREP (since we sent earlier)
+ */
+ if (ref == GREP_REF) {
+ jam();
+ continue;
+ }
+
+ CRASH_INSERTION(13018);
+
+ if (subPtr.p->m_subscriptionType == SubCreateReq::TableEvent)
+ {
+ jam();
+ sendSignal(ref, GSN_SUB_GCP_COMPLETE_REP, signal,
+ SubGcpCompleteRep::SignalLength, JBB);
+ }
+ else
+ {
+ jam();
+ ndbassert(refToNode(ref) == 0 || refToNode(ref) == getOwnNodeId());
+ EXECUTE_DIRECT(refToBlock(ref), GSN_SUB_GCP_COMPLETE_REP, signal,
+ SubGcpCompleteRep::SignalLength);
+ jamEntry();
+ }
+#endif
+ }
+
+ if (c_handoverToDo) {
+ jam();
+ c_handoverToDo = false;
+ for( int i = 0; i < NO_OF_BUCKETS; i++) {
+ if (c_buckets[i].handover) {
+ if (c_buckets[i].handoverGCI > gci) {
+ jam();
+ c_handoverToDo = true; // still waiting for the right GCI
+ break; /* since all handover should happen at the same time
+ * we can break here
+ */
+ } else {
+ c_buckets[i].handover = false;
+#ifdef HANDOVER_DEBUG
+ ndbout_c("Handover Bucket %u", i);
+#endif
+ if (getResponsibleSumaNodeId(i) == refToNode(reference())) {
+ // my bucket to be handed over to me
+ ndbrequire(!c_buckets[i].active);
+ jam();
+ c_buckets[i].active = true;
+ } else {
+ // someone else's bucket to handover to
+ ndbrequire(c_buckets[i].active);
+ jam();
+ c_buckets[i].active = false;
+ }
+ }
+ }
+ }
+ }
+}
+
+/***********************************************************
+ *
+ * Embryo to syncronize the Suma's so as to know if a subscriber
+ * has received a GCP_COMPLETE from all suma's or not
+ *
+ */
+
+void
+SumaParticipant::runSUB_GCP_COMPLETE_ACC(Signal* signal){
+ jam();
+
+ SubGcpCompleteAcc * const acc = (SubGcpCompleteAcc*)signal->getDataPtr();
+
+ Uint32 gci = acc->rep.gci;
+
+#ifdef EVENT_DEBUG
+ ndbout_c("SumaParticipant::runSUB_GCP_COMPLETE_ACC gci = %u", gci);
+#endif
+
+ c_failoverBuffer.subGcpCompleteRep(gci);
+}
+
+void
+Suma::execSUB_GCP_COMPLETE_ACC(Signal* signal){
+ jamEntry();
+
+ if (RtoI(signal->getSendersBlockRef(), false) != RNIL) {
+ jam();
+ // Ack from other SUMA
+ runSUB_GCP_COMPLETE_ACC(signal);
+ return;
+ }
+
+ jam();
+ // Ack from User and not an acc from other SUMA, redistribute in nodegroup
+
+ SubGcpCompleteAcc * const acc = (SubGcpCompleteAcc*)signal->getDataPtr();
+ Uint32 gci = acc->rep.gci;
+ Uint32 senderRef = acc->rep.senderRef;
+ Uint32 subscriberData = acc->rep.subscriberData;
+
+#ifdef EVENT_DEBUG
+ ndbout_c("Suma::execSUB_GCP_COMPLETE_ACC gci = %u", gci);
+#endif
+ bool moreToCome = false;
+
+ SubscriberPtr subbPtr;
+ for(c_dataSubscribers.first(subbPtr);
+ !subbPtr.isNull(); c_dataSubscribers.next(subbPtr)){
+#ifdef EVENT_DEBUG
+ ndbout_c("Suma::execSUB_GCP_COMPLETE_ACC %u == %u && %u == %u",
+ subbPtr.p->m_subscriberRef,
+ senderRef,
+ subbPtr.p->m_subscriberData,
+ subscriberData);
+#endif
+ if (subbPtr.p->m_subscriberRef == senderRef &&
+ subbPtr.p->m_subscriberData == subscriberData) {
+ jam();
+#ifdef EVENT_DEBUG
+ ndbout_c("Suma::execSUB_GCP_COMPLETE_ACC gci = FOUND SUBSCRIBER");
+#endif
+ subbPtr.p->m_lastGCI = gci;
+ } else if (subbPtr.p->m_lastGCI < gci) {
+ jam();
+ if (subbPtr.p->m_firstGCI <= gci)
+ moreToCome = true;
+ } else
+ jam();
+ }
+
+ if (!moreToCome) {
+ // tell the other SUMA's that I'm done with this GCI
+ jam();
+ for (Uint32 i = 0; i < c_noNodesInGroup; i++) {
+ Uint32 id = c_nodesInGroup[i];
+ Uint32 ref = calcSumaBlockRef(id);
+ if ((ref != reference()) && c_aliveNodes.get(id)) {
+ jam();
+ sendSignal(ref, GSN_SUB_GCP_COMPLETE_ACC, signal,
+ SubGcpCompleteAcc::SignalLength, JBB);
+ } else
+ jam();
+ }
+ }
+}
+
+static Uint32 tmpFailoverBuffer[512];
+//SumaParticipant::FailoverBuffer::FailoverBuffer(DataBuffer<15>::DataBufferPool & p)
+// : m_dataList(p),
+SumaParticipant::FailoverBuffer::FailoverBuffer()
+ :
+ c_gcis(tmpFailoverBuffer), c_sz(512), c_first(0), c_next(0), c_full(false)
+{
+}
+
+bool SumaParticipant::FailoverBuffer::subTableData(Uint32 gci, Uint32 *src, int sz)
+{
+ bool ok = true;
+
+ if (c_full) {
+ ok = false;
+#ifdef EVENT_DEBUG
+ ndbout_c("Suma::FailoverBuffer::SubTableData buffer full gci=%u");
+#endif
+ } else {
+ c_gcis[c_next] = gci;
+ c_next++;
+ if (c_next == c_sz) c_next = 0;
+ if (c_next == c_first)
+ c_full = true;
+ // ndbout_c("%u %u %u",c_first,c_next,c_sz);
+ }
+ return ok;
+}
+bool SumaParticipant::FailoverBuffer::subGcpCompleteRep(Uint32 gci)
+{
+ bool ok = true;
+
+ // ndbout_c("Empty");
+ while (true) {
+ if (c_first == c_next && !c_full)
+ break;
+ if (c_gcis[c_first] > gci)
+ break;
+ c_full = false;
+ c_first++;
+ if (c_first == c_sz) c_first = 0;
+ // ndbout_c("%u %u %u : ",c_first,c_next,c_sz);
+ }
+
+ return ok;
+}
+bool SumaParticipant::FailoverBuffer::nodeFailRep()
+{
+ bool ok = true;
+ while (true) {
+ if (c_first == c_next && !c_full)
+ break;
+
+#ifdef EVENT_DEBUG
+ ndbout_c("Suma::FailoverBuffer::NodeFailRep resending gci=%u", c_gcis[c_first]);
+#endif
+ c_full = false;
+ c_first++;
+ if (c_first == c_sz) c_first = 0;
+ }
+ return ok;
+}
+
+/**********************************************************
+ * Suma participant interface
+ *
+ * Stopping and removing of subscriber
+ *
+ */
+
+void
+SumaParticipant::execSUB_STOP_REQ(Signal* signal){
+ jamEntry();
+
+ CRASH_INSERTION(13019);
+
+ SubStopReq * const req = (SubStopReq*)signal->getDataPtr();
+ Uint32 senderRef = signal->getSendersBlockRef();
+ Uint32 senderData = req->senderData;
+ Uint32 subscriberRef = req->subscriberRef;
+ Uint32 subscriberData = req->subscriberData;
+ SubscriptionPtr subPtr;
+ Subscription key;
+ key.m_subscriptionId = req->subscriptionId;
+ key.m_subscriptionKey = req->subscriptionKey;
+ Uint32 part = req->part;
+
+ if (key.m_subscriptionKey == 0 &&
+ key.m_subscriptionId == 0 &&
+ subscriberData == 0) {
+ SubStopConf* conf = (SubStopConf*)signal->getDataPtrSend();
+
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ conf->subscriptionId = key.m_subscriptionId;
+ conf->subscriptionKey = key.m_subscriptionKey;
+ conf->subscriberData = subscriberData;
+
+ sendSignal(senderRef, GSN_SUB_STOP_CONF, signal,
+ SubStopConf::SignalLength, JBB);
+
+ removeSubscribersOnNode(signal, refToNode(subscriberRef));
+ return;
+ }
+
+ if(!c_subscriptions.find(subPtr, key)){
+ jam();
+ sendSubStopRef(signal, GrepError::SUBSCRIPTION_ID_NOT_FOUND);
+ return;
+ }
+
+ ndbrequire(part == SubscriptionData::TableData);
+
+ SubscriberPtr subbPtr;
+ if (senderRef == reference()){
+ jam();
+ c_subscriberPool.getPtr(subbPtr, senderData);
+ ndbrequire(subbPtr.p->m_subPtrI == subPtr.i &&
+ subbPtr.p->m_subscriberRef == subscriberRef &&
+ subbPtr.p->m_subscriberData == subscriberData);
+ c_removeDataSubscribers.remove(subbPtr);
+ } else {
+ bool found = false;
+ jam();
+ c_dataSubscribers.first(subbPtr);
+ for (;!subbPtr.isNull(); c_dataSubscribers.next(subbPtr)){
+ jam();
+ if (subbPtr.p->m_subPtrI == subPtr.i &&
+ subbPtr.p->m_subscriberRef == subscriberRef &&
+ subbPtr.p->m_subscriberData == subscriberData){
+ // ndbout_c("STOP_REQ: before c_dataSubscribers.release");
+ jam();
+ c_dataSubscribers.remove(subbPtr);
+ found = true;
+ break;
+ }
+ }
+ /**
+ * If we didn't find anyone, send ref
+ */
+ if (!found) {
+ jam();
+ sendSubStopRef(signal, GrepError::SUBSCRIBER_NOT_FOUND);
+ return;
+ }
+ }
+
+ subbPtr.p->m_senderRef = senderRef; // store ref to requestor
+ subbPtr.p->m_senderData = senderData; // store ref to requestor
+ c_prepDataSubscribers.add(subbPtr);
+
+ Ptr<SyncRecord> syncPtr;
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+ if (syncPtr.p->m_locked) {
+ jam();
+ sendSubStopRef(signal, /** Error Code */ 0, true);
+ return;
+ }
+ syncPtr.p->m_locked = true;
+
+ syncPtr.p->startDropTrigger(signal);
+}
+
+void
+SumaParticipant::sendSubStopComplete(Signal* signal, SubscriberPtr subbPtr){
+ jam();
+
+ CRASH_INSERTION(13020);
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.getPtr(subPtr, subbPtr.p->m_subPtrI);
+
+ Ptr<SyncRecord> syncPtr;
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+ syncPtr.p->m_locked = false;
+
+ SubStopConf * const conf = (SubStopConf*)signal->getDataPtrSend();
+
+ conf->senderRef = reference();
+ conf->senderData = subbPtr.p->m_senderData;
+ conf->subscriptionId = subPtr.p->m_subscriptionId;
+ conf->subscriptionKey = subPtr.p->m_subscriptionKey;
+ conf->subscriberData = subbPtr.p->m_subscriberData;
+ Uint32 senderRef = subbPtr.p->m_senderRef;
+
+ c_prepDataSubscribers.release(subbPtr);
+ sendSignal(senderRef, GSN_SUB_STOP_CONF, signal,
+ SubStopConf::SignalLength, JBB);
+}
+
+void
+SumaParticipant::sendSubStopRef(Signal* signal, Uint32 errCode,
+ bool temporary){
+ jam();
+ SubStopRef * ref = (SubStopRef *)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->errorCode = errCode;
+ if (temporary) {
+ ref->setTemporary();
+ }
+ sendSignal(signal->getSendersBlockRef(),
+ GSN_SUB_STOP_REF,
+ signal,
+ SubStopRef::SignalLength,
+ JBB);
+ return;
+}
+
+/**************************************************************
+ *
+ * Removing subscription
+ *
+ */
+
+void
+SumaParticipant::execSUB_REMOVE_REQ(Signal* signal) {
+ jamEntry();
+
+ Uint32 senderRef = signal->getSendersBlockRef();
+
+ CRASH_INSERTION(13021);
+
+ const SubRemoveReq req = *(SubRemoveReq*)signal->getDataPtr();
+ SubscriptionPtr subPtr;
+ Subscription key;
+ key.m_subscriptionId = req.subscriptionId;
+ key.m_subscriptionKey = req.subscriptionKey;
+
+ if(!c_subscriptions.find(subPtr, key)) {
+ jam();
+ sendSubRemoveRef(signal, req, (Uint32) GrepError::SUBSCRIPTION_ID_NOT_FOUND);
+ return;
+ }
+
+ int count = 0;
+ {
+ jam();
+ SubscriberPtr i_subbPtr;
+ for(c_prepDataSubscribers.first(i_subbPtr);
+ !i_subbPtr.isNull(); c_prepDataSubscribers.next(i_subbPtr)){
+ jam();
+ if( i_subbPtr.p->m_subPtrI == subPtr.i ) {
+ jam();
+ sendSubRemoveRef(signal, req, /* ErrorCode */ 0, true);
+ return;
+ // c_prepDataSubscribers.release(subbPtr);
+ }
+ }
+ c_dataSubscribers.first(i_subbPtr);
+ while(!i_subbPtr.isNull()){
+ jam();
+ SubscriberPtr subbPtr = i_subbPtr;
+ c_dataSubscribers.next(i_subbPtr);
+ if( subbPtr.p->m_subPtrI == subPtr.i ) {
+ jam();
+ sendSubRemoveRef(signal, req, /* ErrorCode */ 0, true);
+ return;
+ /* Unfinished/untested code. If remove should be possible
+ * even if subscribers are left these have to be stopped
+ * first. See m_markRemove, m_nSubscribers. We need also to
+ * block remove for this subscription so that multiple
+ * removes is not possible...
+ */
+ c_dataSubscribers.remove(subbPtr);
+ c_removeDataSubscribers.add(subbPtr);
+ count++;
+ }
+ }
+ c_metaSubscribers.first(i_subbPtr);
+ while(!i_subbPtr.isNull()){
+ jam();
+ SubscriberPtr subbPtr = i_subbPtr;
+ c_metaSubscribers.next(i_subbPtr);
+ if( subbPtr.p->m_subPtrI == subPtr.i ){
+ jam();
+ c_metaSubscribers.release(subbPtr);
+ }
+ }
+ }
+
+ subPtr.p->m_senderRef = senderRef;
+ subPtr.p->m_senderData = req.senderData;
+
+ if (count > 0){
+ jam();
+ ndbrequire(false); // code not finalized
+ subPtr.p->m_markRemove = true;
+ subPtr.p->m_nSubscribers = count;
+ sendSubStopReq(signal);
+ } else {
+ completeSubRemoveReq(signal, subPtr);
+ }
+}
+
+void
+SumaParticipant::completeSubRemoveReq(Signal* signal, SubscriptionPtr subPtr) {
+ Uint32 subscriptionId = subPtr.p->m_subscriptionId;
+ Uint32 subscriptionKey = subPtr.p->m_subscriptionKey;
+ Uint32 senderRef = subPtr.p->m_senderRef;
+ Uint32 senderData = subPtr.p->m_senderData;
+
+ {
+ Ptr<SyncRecord> syncPtr;
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+
+ syncPtr.p->release();
+ c_syncPool.release(syncPtr);
+ }
+
+ // if (subPtr.p->m_subscriptionType != SubCreateReq::TableEvent) {
+ // jam();
+ // senderRef = subPtr.p->m_subscriberRef;
+ // }
+ c_subscriptions.release(subPtr);
+
+ /**
+ * I was the last subscription to be remove so clear c_tables
+ */
+#if 0
+ ndbout_c("c_subscriptionPool.getSize() %d c_subscriptionPool.getNoOfFree()%d",
+ c_subscriptionPool.getSize(),c_subscriptionPool.getNoOfFree()+1);
+#endif
+
+ if(c_subscriptionPool.getSize() == c_subscriptionPool.getNoOfFree()+1) {
+ jam();
+#if 0
+ ndbout_c("SUB_REMOVE_REQ:Clearing c_tables");
+#endif
+ KeyTable<Table>::Iterator it;
+ for(c_tables.first(it); !it.isNull(); ){
+
+ it.curr.p->release(* this);
+
+ TablePtr tabPtr = it.curr;
+
+ c_tables.next(it);
+ c_tables.release(tabPtr);
+ }
+ }
+
+ SubRemoveConf * const conf = (SubRemoveConf*)signal->getDataPtrSend();
+ conf->senderRef = reference();
+ conf->senderData = senderData;
+ conf->subscriptionId = subscriptionId;
+ conf->subscriptionKey = subscriptionKey;
+
+ sendSignal(senderRef, GSN_SUB_REMOVE_CONF, signal,
+ SubRemoveConf::SignalLength, JBB);
+}
+
+void
+SumaParticipant::sendSubRemoveRef(Signal* signal, const SubRemoveReq& req,
+ Uint32 errCode, bool temporary){
+ jam();
+ SubRemoveRef * ref = (SubRemoveRef *)signal->getDataPtrSend();
+ ref->senderRef = reference();
+ ref->senderData = req.senderData;
+ ref->err = errCode;
+ if (temporary)
+ ref->setTemporary();
+ releaseSections(signal);
+ sendSignal(signal->getSendersBlockRef(), GSN_SUB_REMOVE_REF,
+ signal, SubRemoveRef::SignalLength, JBB);
+ return;
+}
+
+void
+SumaParticipant::Table::release(SumaParticipant & suma){
+ jam();
+
+ LocalDataBuffer<15> attrBuf(suma.c_dataBufferPool, m_attributes);
+ attrBuf.release();
+
+ LocalDataBuffer<15> fragBuf(suma.c_dataBufferPool, m_fragments);
+ fragBuf.release();
+}
+
+void
+SumaParticipant::SyncRecord::release(){
+ jam();
+ m_tableList.release();
+
+ LocalDataBuffer<15> attrBuf(suma.c_dataBufferPool, m_attributeList);
+ attrBuf.release();
+}
+
+
+/**************************************************************
+ *
+ * Restarting remote node functions, master functionality
+ * (slave does nothing special)
+ * - triggered on INCL_NODEREQ calling startNode
+ * - included node will issue START_ME when it's ready to start
+ * the subscribers
+ *
+ */
+
+Suma::Restart::Restart(Suma& s) : suma(s) {
+ for (int i = 0; i < MAX_REPLICAS; i++) {
+ c_okToStart[i] = false;
+ c_waitingToStart[i] = false;
+ }
+};
+
+void
+Suma::Restart::resetNode(Uint32 sumaRef)
+{
+ jam();
+ int I = suma.RtoI(sumaRef);
+ c_okToStart[I] = false;
+ c_waitingToStart[I] = false;
+}
+
+void
+Suma::Restart::startNode(Signal* signal, Uint32 sumaRef)
+{
+ jam();
+ resetNode(sumaRef);
+
+ // right now we can only handle restarting one node
+ // at a time in a node group
+
+ createSubscription(signal, sumaRef);
+}
+
+void
+Suma::Restart::createSubscription(Signal* signal, Uint32 sumaRef) {
+ jam();
+ suma.c_subscriptions.first(c_subPtr);
+ nextSubscription(signal, sumaRef);
+}
+
+void
+Suma::Restart::nextSubscription(Signal* signal, Uint32 sumaRef) {
+ jam();
+ if (c_subPtr.isNull()) {
+ jam();
+ completeSubscription(signal, sumaRef);
+ return;
+ }
+ SubscriptionPtr subPtr;
+ subPtr.i = c_subPtr.curr.i;
+ subPtr.p = suma.c_subscriptions.getPtr(subPtr.i);
+
+ suma.c_subscriptions.next(c_subPtr);
+
+ SubCreateReq * req = (SubCreateReq *)signal->getDataPtrSend();
+
+ req->subscriberRef = suma.reference();
+ req->subscriberData = subPtr.i;
+ req->subscriptionId = subPtr.p->m_subscriptionId;
+ req->subscriptionKey = subPtr.p->m_subscriptionKey;
+ req->subscriptionType = subPtr.p->m_subscriptionType |
+ SubCreateReq::RestartFlag;
+
+ switch (subPtr.p->m_subscriptionType) {
+ case SubCreateReq::TableEvent:
+ case SubCreateReq::SelectiveTableSnapshot:
+ case SubCreateReq::DatabaseSnapshot: {
+ jam();
+
+ Ptr<SyncRecord> syncPtr;
+ suma.c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+ syncPtr.p->m_tableList.first(syncPtr.p->m_tableList_it);
+
+ ndbrequire(!syncPtr.p->m_tableList_it.isNull());
+
+ req->tableId = *syncPtr.p->m_tableList_it.data;
+
+#if 0
+ for (int i = 0; i < MAX_TABLES; i++)
+ if (subPtr.p->m_tables[i]) {
+ req->tableId = i;
+ break;
+ }
+#endif
+
+ suma.sendSignal(sumaRef, GSN_SUB_CREATE_REQ, signal,
+ SubCreateReq::SignalLength+1 /*to get table Id*/, JBB);
+ return;
+ }
+ case SubCreateReq::SingleTableScan :
+ // TODO
+ jam();
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Suma::execSUB_CREATE_CONF(Signal* signal) {
+ jamEntry();
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::execSUB_CREATE_CONF");
+#endif
+
+ const Uint32 senderRef = signal->senderBlockRef();
+
+ SubCreateConf * const conf = (SubCreateConf *)signal->getDataPtr();
+
+ Subscription key;
+ const Uint32 subscriberData = conf->subscriberData;
+ key.m_subscriptionId = conf->subscriptionId;
+ key.m_subscriptionKey = conf->subscriptionKey;
+
+ SubscriptionPtr subPtr;
+ ndbrequire(c_subscriptions.find(subPtr, key));
+
+ switch(subPtr.p->m_subscriptionType) {
+ case SubCreateReq::TableEvent:
+ case SubCreateReq::SelectiveTableSnapshot:
+ case SubCreateReq::DatabaseSnapshot:
+ {
+ Ptr<SyncRecord> syncPtr;
+ c_syncPool.getPtr(syncPtr, subPtr.p->m_syncPtrI);
+
+ syncPtr.p->m_tableList.next(syncPtr.p->m_tableList_it);
+ if (syncPtr.p->m_tableList_it.isNull()) {
+ jam();
+ SubSyncReq *req = (SubSyncReq *)signal->getDataPtrSend();
+
+ req->subscriptionId = key.m_subscriptionId;
+ req->subscriptionKey = key.m_subscriptionKey;
+ req->subscriberData = subscriberData;
+ req->part = (Uint32) SubscriptionData::MetaData;
+
+ sendSignal(senderRef, GSN_SUB_SYNC_REQ, signal,
+ SubSyncReq::SignalLength, JBB);
+ } else {
+ jam();
+ SubCreateReq * req = (SubCreateReq *)signal->getDataPtrSend();
+
+ req->subscriberRef = reference();
+ req->subscriberData = subPtr.i;
+ req->subscriptionId = subPtr.p->m_subscriptionId;
+ req->subscriptionKey = subPtr.p->m_subscriptionKey;
+ req->subscriptionType = subPtr.p->m_subscriptionType |
+ SubCreateReq::RestartFlag |
+ SubCreateReq::AddTableFlag;
+
+ req->tableId = *syncPtr.p->m_tableList_it.data;
+
+ sendSignal(senderRef, GSN_SUB_CREATE_REQ, signal,
+ SubCreateReq::SignalLength+1 /*to get table Id*/, JBB);
+ }
+ }
+ return;
+ case SubCreateReq::SingleTableScan:
+ ndbrequire(false);
+ }
+ ndbrequire(false);
+}
+
+void
+Suma::execSUB_CREATE_REF(Signal* signal) {
+ jamEntry();
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::execSUB_CREATE_REF");
+#endif
+ //ndbrequire(false);
+}
+
+void
+Suma::execSUB_SYNC_CONF(Signal* signal) {
+ jamEntry();
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::execSUB_SYNC_CONF");
+#endif
+ Uint32 sumaRef = signal->getSendersBlockRef();
+
+ SubSyncConf *conf = (SubSyncConf *)signal->getDataPtr();
+ Subscription key;
+
+ key.m_subscriptionId = conf->subscriptionId;
+ key.m_subscriptionKey = conf->subscriptionKey;
+ // SubscriptionData::Part part = (SubscriptionData::Part)conf->part;
+ // const Uint32 subscriberData = conf->subscriberData;
+
+ SubscriptionPtr subPtr;
+ c_subscriptions.find(subPtr, key);
+
+ switch(subPtr.p->m_subscriptionType) {
+ case SubCreateReq::TableEvent:
+ case SubCreateReq::SelectiveTableSnapshot:
+ case SubCreateReq::DatabaseSnapshot:
+ jam();
+ Restart.nextSubscription(signal, sumaRef);
+ return;
+ case SubCreateReq::SingleTableScan:
+ ndbrequire(false);
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Suma::execSUB_SYNC_REF(Signal* signal) {
+ jamEntry();
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::execSUB_SYNC_REF");
+#endif
+ //ndbrequire(false);
+}
+
+void
+Suma::execSUMA_START_ME(Signal* signal) {
+ jamEntry();
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::execSUMA_START_ME");
+#endif
+
+ Restart.runSUMA_START_ME(signal, signal->getSendersBlockRef());
+}
+
+void
+Suma::Restart::runSUMA_START_ME(Signal* signal, Uint32 sumaRef) {
+ int I = suma.RtoI(sumaRef);
+
+ // restarting Suma is ready for SUB_START_REQ
+ if (c_waitingToStart[I]) {
+ // we've waited with startSubscriber since restarting suma was not ready
+ c_waitingToStart[I] = false;
+ startSubscriber(signal, sumaRef);
+ } else {
+ // do startSubscriber as soon as its time
+ c_okToStart[I] = true;
+ }
+}
+
+void
+Suma::Restart::completeSubscription(Signal* signal, Uint32 sumaRef) {
+ jam();
+ int I = suma.RtoI(sumaRef);
+
+ if (c_okToStart[I]) {// otherwise will start when START_ME comes
+ c_okToStart[I] = false;
+ startSubscriber(signal, sumaRef);
+ } else {
+ c_waitingToStart[I] = true;
+ }
+}
+
+void
+Suma::Restart::startSubscriber(Signal* signal, Uint32 sumaRef) {
+ jam();
+ suma.c_dataSubscribers.first(c_subbPtr);
+ nextSubscriber(signal, sumaRef);
+}
+
+void
+Suma::Restart::sendSubStartReq(SubscriptionPtr subPtr, SubscriberPtr subbPtr,
+ Signal* signal, Uint32 sumaRef)
+{
+ jam();
+ SubStartReq * req = (SubStartReq *)signal->getDataPtrSend();
+
+ req->senderRef = suma.reference();
+ req->senderData = subbPtr.p->m_senderData;
+ req->subscriptionId = subPtr.p->m_subscriptionId;
+ req->subscriptionKey = subPtr.p->m_subscriptionKey;
+ req->part = SubscriptionData::TableData;
+ req->subscriberData = subbPtr.p->m_subscriberData;
+ req->subscriberRef = subbPtr.p->m_subscriberRef;
+
+ // restarting suma will not respond to this until startphase 5
+ // since it is not until then data copying has been completed
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::Restart::sendSubStartReq sending GSN_SUB_START_REQ id=%u key=%u",
+ req->subscriptionId, req->subscriptionKey);
+#endif
+ suma.sendSignal(sumaRef, GSN_SUB_START_REQ,
+ signal, SubStartReq::SignalLength2, JBB);
+}
+
+void
+Suma::execSUB_START_CONF(Signal* signal) {
+ jamEntry();
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::execSUB_START_CONF");
+#endif
+ Uint32 sumaRef = signal->getSendersBlockRef();
+ Restart.nextSubscriber(signal, sumaRef);
+}
+
+void
+Suma::execSUB_START_REF(Signal* signal) {
+ jamEntry();
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::execSUB_START_REF");
+#endif
+ //ndbrequire(false);
+}
+
+void
+Suma::Restart::nextSubscriber(Signal* signal, Uint32 sumaRef) {
+ jam();
+ if (c_subbPtr.isNull()) {
+ jam();
+ completeSubscriber(signal, sumaRef);
+ return;
+ }
+
+ SubscriberPtr subbPtr = c_subbPtr;
+ suma.c_dataSubscribers.next(c_subbPtr);
+
+ /*
+ * get subscription ptr for this subscriber
+ */
+
+ SubscriptionPtr subPtr;
+ suma.c_subscriptions.getPtr(subPtr, subbPtr.p->m_subPtrI);
+ switch (subPtr.p->m_subscriptionType) {
+ case SubCreateReq::TableEvent:
+ case SubCreateReq::SelectiveTableSnapshot:
+ case SubCreateReq::DatabaseSnapshot:
+ {
+ jam();
+ sendSubStartReq(subPtr, subbPtr, signal, sumaRef);
+#if 0
+ SubStartReq * req = (SubStartReq *)signal->getDataPtrSend();
+
+ req->senderRef = reference();
+ req->senderData = subbPtr.p->m_senderData;
+ req->subscriptionId = subPtr.p->m_subscriptionId;
+ req->subscriptionKey = subPtr.p->m_subscriptionKey;
+ req->part = SubscriptionData::TableData;
+ req->subscriberData = subbPtr.p->m_subscriberData;
+ req->subscriberRef = subbPtr.p->m_subscriberRef;
+
+ // restarting suma will not respond to this until startphase 5
+ // since it is not until then data copying has been completed
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::nextSubscriber sending GSN_SUB_START_REQ id=%u key=%u",
+ req->subscriptionId, req->subscriptionKey);
+#endif
+ suma.sendSignal(sumaRef, GSN_SUB_START_REQ,
+ signal, SubStartReq::SignalLength2, JBB);
+#endif
+ }
+ return;
+ case SubCreateReq::SingleTableScan:
+ ndbrequire(false);
+ return;
+ }
+ ndbrequire(false);
+}
+
+void
+Suma::Restart::completeSubscriber(Signal* signal, Uint32 sumaRef) {
+ completeRestartingNode(signal, sumaRef);
+}
+
+void
+Suma::Restart::completeRestartingNode(Signal* signal, Uint32 sumaRef) {
+ jam();
+ SumaHandoverReq * req = (SumaHandoverReq *)signal->getDataPtrSend();
+
+ req->gci = suma.getFirstGCI(signal);
+
+ suma.sendSignal(sumaRef, GSN_SUMA_HANDOVER_REQ, signal,
+ SumaHandoverReq::SignalLength, JBB);
+}
+
+// only run on restarting suma
+
+void
+Suma::execSUMA_HANDOVER_REQ(Signal* signal)
+{
+ jamEntry();
+ // Uint32 sumaRef = signal->getSendersBlockRef();
+ SumaHandoverReq const * req = (SumaHandoverReq *)signal->getDataPtr();
+
+ Uint32 gci = req->gci;
+ Uint32 new_gci = getFirstGCI(signal);
+
+ if (new_gci > gci) {
+ gci = new_gci;
+ }
+
+ { // all recreated subscribers at restarting SUMA start at same GCI
+ SubscriberPtr subbPtr;
+ for(c_dataSubscribers.first(subbPtr);
+ !subbPtr.isNull();
+ c_dataSubscribers.next(subbPtr)){
+ subbPtr.p->m_firstGCI = gci;
+ }
+ }
+
+#ifdef NODEFAIL_DEBUG
+ ndbout_c("Suma::execSUMA_HANDOVER_REQ, gci = %u", gci);
+#endif
+
+ c_handoverToDo = false;
+ c_restartLock = false;
+ {
+#ifdef HANDOVER_DEBUG
+ int c = 0;
+#endif
+ for( int i = 0; i < NO_OF_BUCKETS; i++) {
+ jam();
+ if (getResponsibleSumaNodeId(i) == refToNode(reference())) {
+#ifdef HANDOVER_DEBUG
+ c++;
+#endif
+ jam();
+ c_buckets[i].active = false;
+ c_buckets[i].handoverGCI = gci;
+ c_buckets[i].handover = true;
+ c_buckets[i].handover_started = false;
+ c_handoverToDo = true;
+ }
+ }
+#ifdef HANDOVER_DEBUG
+ ndbout_c("prepared handover of bucket %u buckets", c);
+#endif
+ }
+
+ for (Uint32 i = 0; i < c_noNodesInGroup; i++) {
+ jam();
+ Uint32 ref = calcSumaBlockRef(c_nodesInGroup[i]);
+ if (ref != reference()) {
+ jam();
+ sendSignal(ref, GSN_SUMA_HANDOVER_CONF, signal,
+ SumaHandoverConf::SignalLength, JBB);
+ }//if
+ }
+}
+
+// only run on all but restarting suma
+void
+Suma::execSUMA_HANDOVER_CONF(Signal* signal) {
+ jamEntry();
+ Uint32 sumaRef = signal->getSendersBlockRef();
+ SumaHandoverConf const * conf = (SumaHandoverConf *)signal->getDataPtr();
+
+ Uint32 gci = conf->gci;
+
+#ifdef HANDOVER_DEBUG
+ ndbout_c("Suma::execSUMA_HANDOVER_CONF, gci = %u", gci);
+#endif
+
+ /* TODO, if we are restarting several SUMA's (>2 in a nodegroup)
+ * we have to collect all these conf's before proceding
+ */
+
+ // restarting node is now prepared and ready
+ c_preparingNodes.clear(refToNode(sumaRef)); /* !! important to do before
+ * below since it affects
+ * getResponsibleSumaNodeId()
+ */
+
+ c_handoverToDo = false;
+ // mark all active buckets really belonging to restarting SUMA
+ for( int i = 0; i < NO_OF_BUCKETS; i++) {
+ if (c_buckets[i].active) {
+ // I'm running this bucket
+ if (getResponsibleSumaNodeId(i) == refToNode(sumaRef)) {
+ // but it should really be the restarted node
+ c_buckets[i].handoverGCI = gci;
+ c_buckets[i].handover = true;
+ c_buckets[i].handover_started = false;
+ c_handoverToDo = true;
+ }
+ }
+ }
+}
diff --git a/ndb/src/kernel/blocks/suma/Suma.hpp b/ndb/src/kernel/blocks/suma/Suma.hpp
new file mode 100644
index 00000000000..7bd58b30640
--- /dev/null
+++ b/ndb/src/kernel/blocks/suma/Suma.hpp
@@ -0,0 +1,597 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SUMA_H
+#define SUMA_H
+
+#include <ndb_limits.h>
+#include <SimulatedBlock.hpp>
+
+#include <NodeBitmask.hpp>
+
+#include <SLList.hpp>
+#include <DLList.hpp>
+#include <KeyTable.hpp>
+#include <DataBuffer.hpp>
+#include <SignalCounter.hpp>
+#include <AttributeHeader.hpp>
+#include <AttributeList.hpp>
+
+#include <signaldata/UtilSequence.hpp>
+#include <signaldata/SumaImpl.hpp>
+
+class SumaParticipant : public SimulatedBlock {
+protected:
+ SumaParticipant(const Configuration & conf);
+ virtual ~SumaParticipant();
+ BLOCK_DEFINES(SumaParticipant);
+
+protected:
+ /**
+ * Private interface
+ */
+ void execSUB_CREATE_REQ(Signal* signal);
+ void execSUB_REMOVE_REQ(Signal* signal);
+
+ void execSUB_START_REQ(Signal* signal);
+ void execSUB_STOP_REQ(Signal* signal);
+
+ void execSUB_SYNC_REQ(Signal* signal);
+ void execSUB_ABORT_SYNC_REQ(Signal* signal);
+
+ void execSUB_STOP_CONF(Signal* signal);
+ void execSUB_STOP_REF(Signal* signal);
+
+ /**
+ * Dict interface
+ */
+ void execLIST_TABLES_REF(Signal* signal);
+ void execLIST_TABLES_CONF(Signal* signal);
+ void execGET_TABINFOREF(Signal* signal);
+ void execGET_TABINFO_CONF(Signal* signal);
+ void execGET_TABLEID_CONF(Signal* signal);
+ void execGET_TABLEID_REF(Signal* signal);
+
+ /**
+ * Scan interface
+ */
+ void execSCAN_HBREP(Signal* signal);
+ void execSCAN_FRAGREF(Signal* signal);
+ void execSCAN_FRAGCONF(Signal* signal);
+ void execTRANSID_AI(Signal* signal);
+ void execSUB_SYNC_CONTINUE_REF(Signal* signal);
+ void execSUB_SYNC_CONTINUE_CONF(Signal* signal);
+
+ /**
+ * Trigger logging
+ */
+ void execTRIG_ATTRINFO(Signal* signal);
+ void execFIRE_TRIG_ORD(Signal* signal);
+ void execSUB_GCP_COMPLETE_REP(Signal* signal);
+ void runSUB_GCP_COMPLETE_ACC(Signal* signal);
+
+ /**
+ * DIH signals
+ */
+ void execDI_FCOUNTREF(Signal* signal);
+ void execDI_FCOUNTCONF(Signal* signal);
+ void execDIGETPRIMREF(Signal* signal);
+ void execDIGETPRIMCONF(Signal* signal);
+
+ /**
+ * Trigger administration
+ */
+ void execCREATE_TRIG_REF(Signal* signal);
+ void execCREATE_TRIG_CONF(Signal* signal);
+ void execDROP_TRIG_REF(Signal* signal);
+ void execDROP_TRIG_CONF(Signal* signal);
+
+ /**
+ * continueb
+ */
+ void execCONTINUEB(Signal* signal);
+
+public:
+ typedef DataBuffer<15> TableList;
+
+ union FragmentDescriptor {
+ struct {
+ Uint16 m_fragmentNo;
+ Uint16 m_nodeId;
+ } m_fragDesc;
+ Uint32 m_dummy;
+ };
+
+ /**
+ * Used when sending SCAN_FRAG
+ */
+ union AttributeDescriptor {
+ struct {
+ Uint16 attrId;
+ Uint16 unused;
+ } m_attrDesc;
+ Uint32 m_dummy;
+ };
+
+ struct Table {
+ Table() { m_tableId = ~0; }
+ void release(SumaParticipant&);
+
+ union { Uint32 m_tableId; Uint32 key; };
+ Uint32 m_schemaVersion;
+ Uint32 m_hasTriggerDefined[3]; // Insert/Update/Delete
+ Uint32 m_triggerIds[3]; // Insert/Update/Delete
+
+ /**
+ * Default order in which to ask for attributes during scan
+ * 1) Fixed, not nullable
+ * 2) Rest
+ */
+ DataBuffer<15>::Head m_attributes; // Attribute id's
+
+ /**
+ * Fragments
+ */
+ DataBuffer<15>::Head m_fragments; // Fragment descriptors
+
+ /**
+ * Hash table stuff
+ */
+ Uint32 nextHash;
+ union { Uint32 prevHash; Uint32 nextPool; };
+ Uint32 hashValue() const {
+ return m_tableId;
+ }
+ bool equal(const Table& rec) const {
+ return m_tableId == rec.m_tableId;
+ }
+ };
+ typedef Ptr<Table> TablePtr;
+
+ /**
+ * Subscriptions
+ */
+ struct SyncRecord {
+ SyncRecord(SumaParticipant& s, DataBuffer<15>::DataBufferPool & p)
+ : m_locked(false), m_tableList(p), suma(s)
+#ifdef ERROR_INSERT
+ , cerrorInsert(s.cerrorInsert)
+#endif
+ {}
+
+ void release();
+
+ Uint32 m_subscriptionPtrI;
+ bool m_locked;
+ bool m_doSendSyncData;
+ bool m_error;
+ TableList m_tableList; // Tables to sync (snapshoted at beginning)
+ TableList::DataBufferIterator m_tableList_it;
+
+ /**
+ * Sync meta
+ */
+ void startMeta(Signal*);
+ void nextMeta(Signal*);
+ void completeMeta(Signal*);
+
+ /**
+ * Create triggers
+ */
+ Uint32 m_latestTriggerId;
+ void startTrigger(Signal* signal);
+ void nextTrigger(Signal* signal);
+ void completeTrigger(Signal* signal);
+ void createAttributeMask(AttributeMask&, Table*);
+
+ /**
+ * Drop triggers
+ */
+ void startDropTrigger(Signal* signal);
+ void nextDropTrigger(Signal* signal);
+ void completeDropTrigger(Signal* signal);
+
+ /**
+ * Sync data
+ */
+ Uint32 m_currentTable; // Index in m_tableList
+ Uint32 m_currentFragment; // Index in tabPtr.p->m_fragments
+ DataBuffer<15>::Head m_attributeList; // Attribute if other than default
+ DataBuffer<15>::Head m_tabList; // tables if other than default
+
+ Uint32 m_currentTableId; // Current table
+ Uint32 m_currentNoOfAttributes; // No of attributes for current table
+ void startScan(Signal*);
+ void nextScan(Signal*);
+ bool getNextFragment(TablePtr * tab, FragmentDescriptor * fd);
+ void completeScan(Signal*);
+
+ SumaParticipant & suma;
+#ifdef ERROR_INSERT
+ UintR &cerrorInsert;
+#endif
+ BlockNumber number() const { return suma.number(); }
+ void progError(int line, int cause, const char * extra) {
+ suma.progError(line, cause, extra);
+ }
+
+ void runLIST_TABLES_CONF(Signal* signal);
+ void runGET_TABINFO_CONF(Signal* signal);
+ void runGET_TABINFOREF(Signal* signal);
+
+ void runDI_FCOUNTCONF(Signal* signal);
+ void runDIGETPRIMCONF(Signal* signal);
+
+ void runCREATE_TRIG_CONF(Signal* signal);
+ void runDROP_TRIG_CONF(Signal* signal);
+ void runDROP_TRIG_REF(Signal* signal);
+ void runDropTrig(Signal* signal, Uint32 triggerId, Uint32 tableId);
+
+ union { Uint32 nextPool; Uint32 nextList; Uint32 ptrI; };
+ };
+ friend struct SyncRecord;
+
+ struct Subscription {
+ Uint32 m_subscriberRef;
+ Uint32 m_subscriberData;
+ Uint32 m_senderRef;
+ Uint32 m_senderData;
+ Uint32 m_subscriptionId;
+ Uint32 m_subscriptionKey;
+ Uint32 m_subscriptionType;
+ Uint32 m_coordinatorRef;
+ Uint32 m_syncPtrI; // Active sync operation
+ Uint32 m_nSubscribers;
+ bool m_markRemove;
+
+ Uint32 nextHash;
+ union { Uint32 prevHash; Uint32 nextPool; };
+
+ Uint32 hashValue() const {
+ return m_subscriptionId + m_subscriptionKey;
+ }
+
+ bool equal(const Subscription & s) const {
+ return
+ m_subscriptionId == s.m_subscriptionId &&
+ m_subscriptionKey == s.m_subscriptionKey;
+ }
+ /**
+ * The following holds the table names of tables included
+ * in the subscription.
+ */
+ // TODO we've got to fix this, this is to inefficient. Tomas
+ char m_tables[MAX_TABLES];
+ char m_tableNames[MAX_TABLES][MAX_TAB_NAME_SIZE];
+ /**
+ * "Iterator" used to iterate through m_tableNames
+ */
+ Uint32 m_maxTables;
+ Uint32 m_currentTable;
+ };
+ typedef Ptr<Subscription> SubscriptionPtr;
+
+ struct Subscriber {
+ Uint32 m_senderRef;
+ Uint32 m_senderData;
+ Uint32 m_subscriberRef;
+ Uint32 m_subscriberData;
+ Uint32 m_subPtrI; //reference to subscription
+ Uint32 m_firstGCI; // first GCI to send
+ Uint32 m_lastGCI; // last acnowledged GCI
+ Uint32 nextList;
+ union { Uint32 nextPool; Uint32 prevList; };
+ };
+ typedef Ptr<Subscriber> SubscriberPtr;
+
+ struct Bucket {
+ bool active;
+ bool handover;
+ bool handover_started;
+ Uint32 handoverGCI;
+ };
+#define NO_OF_BUCKETS 24
+ struct Bucket c_buckets[NO_OF_BUCKETS];
+ bool c_handoverToDo;
+ Uint32 c_lastCompleteGCI;
+
+ /**
+ *
+ */
+ DLList<Subscriber> c_metaSubscribers;
+ DLList<Subscriber> c_dataSubscribers;
+ DLList<Subscriber> c_prepDataSubscribers;
+ DLList<Subscriber> c_removeDataSubscribers;
+
+ /**
+ * Lists
+ */
+ KeyTable<Table> c_tables;
+ DLHashTable<Subscription> c_subscriptions;
+
+ /**
+ * Pools
+ */
+ ArrayPool<Subscriber> c_subscriberPool;
+ ArrayPool<Table> c_tablePool_;
+ ArrayPool<Subscription> c_subscriptionPool;
+ ArrayPool<SyncRecord> c_syncPool;
+ DataBuffer<15>::DataBufferPool c_dataBufferPool;
+
+ /**
+ * for restarting Suma not to start sending data too early
+ */
+ bool c_restartLock;
+
+ /**
+ * for flagging that a GCI containg inconsistent data
+ * typically due to node failiure
+ */
+
+ Uint32 c_lastInconsistentGCI;
+ Uint32 c_nodeFailGCI;
+
+ NodeBitmask c_failedApiNodes;
+
+ /**
+ * Functions
+ */
+ bool removeSubscribersOnNode(Signal *signal, Uint32 nodeId);
+
+ bool parseTable(Signal* signal, class GetTabInfoConf* conf, Uint32 tableId,
+ SyncRecord* syncPtr_p);
+ bool checkTableTriggers(SegmentedSectionPtr ptr);
+
+ void addTableId(Uint32 TableId,
+ SubscriptionPtr subPtr, SyncRecord *psyncRec);
+
+ void sendSubIdRef(Signal* signal, Uint32 errorCode);
+ void sendSubCreateConf(Signal* signal, Uint32 sender, SubscriptionPtr subPtr);
+ void sendSubCreateRef(Signal* signal, const SubCreateReq& req, Uint32 errorCode);
+ void sendSubStartRef(SubscriptionPtr subPtr, Signal* signal,
+ Uint32 errorCode, bool temporary = false);
+ void sendSubStartRef(Signal* signal,
+ Uint32 errorCode, bool temporary = false);
+ void sendSubStopRef(Signal* signal,
+ Uint32 errorCode, bool temporary = false);
+ void sendSubSyncRef(Signal* signal, Uint32 errorCode);
+ void sendSubRemoveRef(Signal* signal, const SubRemoveReq& ref,
+ Uint32 errorCode, bool temporary = false);
+ void sendSubStartComplete(Signal*, SubscriberPtr, Uint32,
+ SubscriptionData::Part);
+ void sendSubStopComplete(Signal*, SubscriberPtr);
+ void sendSubStopReq(Signal* signal);
+
+ void completeSubRemoveReq(Signal* signal, SubscriptionPtr subPtr);
+
+ Uint32 getFirstGCI(Signal* signal);
+ Uint32 decideWhoToSend(Uint32 nBucket, Uint32 gci);
+
+ virtual Uint32 getStoreBucket(Uint32 v) = 0;
+ virtual Uint32 getResponsibleSumaNodeId(Uint32 D) = 0;
+ virtual Uint32 RtoI(Uint32 sumaRef, bool dieOnNotFound = true) = 0;
+
+ struct FailoverBuffer {
+ // FailoverBuffer(DataBuffer<15>::DataBufferPool & p);
+ FailoverBuffer();
+
+ bool subTableData(Uint32 gci, Uint32 *src, int sz);
+ bool subGcpCompleteRep(Uint32 gci);
+ bool nodeFailRep();
+
+ // typedef DataBuffer<15> GCIDataBuffer;
+ // GCIDataBuffer m_GCIDataBuffer;
+ // GCIDataBuffer::DataBufferIterator m_GCIDataBuffer_it;
+
+ Uint32 *c_gcis;
+ int c_sz;
+
+ // Uint32 *c_buf;
+ // int c_buf_sz;
+
+ int c_first;
+ int c_next;
+ bool c_full;
+ } c_failoverBuffer;
+
+ /**
+ * Table admin
+ */
+ void convertNameToId( SubscriptionPtr subPtr, Signal * signal);
+
+
+};
+
+class Suma : public SumaParticipant {
+ BLOCK_DEFINES(Suma);
+public:
+ Suma(const Configuration & conf);
+ virtual ~Suma();
+private:
+ /**
+ * Public interface
+ */
+ void execCREATE_SUBSCRIPTION_REQ(Signal* signal);
+ void execDROP_SUBSCRIPTION_REQ(Signal* signal);
+
+ void execSTART_SUBSCRIPTION_REQ(Signal* signal);
+ void execSTOP_SUBSCRIPTION_REQ(Signal* signal);
+
+ void execSYNC_SUBSCRIPTION_REQ(Signal* signal);
+ void execABORT_SYNC_REQ(Signal* signal);
+
+ /**
+ * Framework signals
+ */
+
+ void getNodeGroupMembers(Signal* signal);
+
+ void execSTTOR(Signal* signal);
+ void sendSTTORRY(Signal*);
+ void execNDB_STTOR(Signal* signal);
+ void execDUMP_STATE_ORD(Signal* signal);
+ void execREAD_NODESCONF(Signal* signal);
+ void execNODE_FAILREP(Signal* signal);
+ void execINCL_NODEREQ(Signal* signal);
+ void execCONTINUEB(Signal* signal);
+ void execSIGNAL_DROPPED_REP(Signal* signal);
+ void execAPI_FAILREQ(Signal* signal) ;
+
+ void execSUB_GCP_COMPLETE_ACC(Signal* signal);
+
+ /**
+ * Controller interface
+ */
+ void execSUB_CREATE_REF(Signal* signal);
+ void execSUB_CREATE_CONF(Signal* signal);
+
+ void execSUB_DROP_REF(Signal* signal);
+ void execSUB_DROP_CONF(Signal* signal);
+
+ void execSUB_START_REF(Signal* signal);
+ void execSUB_START_CONF(Signal* signal);
+
+ void execSUB_STOP_REF(Signal* signal);
+ void execSUB_STOP_CONF(Signal* signal);
+
+ void execSUB_SYNC_REF(Signal* signal);
+ void execSUB_SYNC_CONF(Signal* signal);
+
+ void execSUB_ABORT_SYNC_REF(Signal* signal);
+ void execSUB_ABORT_SYNC_CONF(Signal* signal);
+
+ void execSUMA_START_ME(Signal* signal);
+ void execSUMA_HANDOVER_REQ(Signal* signal);
+ void execSUMA_HANDOVER_CONF(Signal* signal);
+
+ /**
+ * Subscription generation interface
+ */
+ void createSequence(Signal* signal);
+ void createSequenceReply(Signal* signal,
+ UtilSequenceConf* conf,
+ UtilSequenceRef* ref);
+ void execUTIL_SEQUENCE_CONF(Signal* signal);
+ void execUTIL_SEQUENCE_REF(Signal* signal);
+ void execCREATE_SUBID_REQ(Signal* signal);
+
+ Uint32 getStoreBucket(Uint32 v);
+ Uint32 getResponsibleSumaNodeId(Uint32 D);
+
+ /**
+ * for Suma that is restarting another
+ */
+
+ struct Restart {
+ Restart(Suma& s);
+
+ Suma & suma;
+
+ bool c_okToStart[MAX_REPLICAS];
+ bool c_waitingToStart[MAX_REPLICAS];
+
+ DLHashTable<SumaParticipant::Subscription>::Iterator c_subPtr; // TODO [MAX_REPLICAS]
+ SubscriberPtr c_subbPtr; // TODO [MAX_REPLICAS]
+
+ void progError(int line, int cause, const char * extra) {
+ suma.progError(line, cause, extra);
+ }
+
+ void resetNode(Uint32 sumaRef);
+ void runSUMA_START_ME(Signal*, Uint32 sumaRef);
+ void startNode(Signal*, Uint32 sumaRef);
+
+ void createSubscription(Signal* signal, Uint32 sumaRef);
+ void nextSubscription(Signal* signal, Uint32 sumaRef);
+ void completeSubscription(Signal* signal, Uint32 sumaRef);
+
+ void startSync(Signal* signal, Uint32 sumaRef);
+ void nextSync(Signal* signal, Uint32 sumaRef);
+ void completeSync(Signal* signal, Uint32 sumaRef);
+
+ void sendSubStartReq(SubscriptionPtr subPtr, SubscriberPtr subbPtr,
+ Signal* signal, Uint32 sumaRef);
+ void startSubscriber(Signal* signal, Uint32 sumaRef);
+ void nextSubscriber(Signal* signal, Uint32 sumaRef);
+ void completeSubscriber(Signal* signal, Uint32 sumaRef);
+
+ void completeRestartingNode(Signal* signal, Uint32 sumaRef);
+ } Restart;
+
+private:
+ friend class Restart;
+ struct SubCoordinator {
+ Uint32 m_subscriberRef;
+ Uint32 m_subscriberData;
+
+ Uint32 m_subscriptionId;
+ Uint32 m_subscriptionKey;
+
+ NdbNodeBitmask m_participants;
+
+ Uint32 m_outstandingGsn;
+ SignalCounter m_outstandingRequests;
+
+ Uint32 nextList;
+ union { Uint32 prevList; Uint32 nextPool; };
+ };
+ Ptr<SubCoordinator> SubCoordinatorPtr;
+
+ struct Node {
+ Uint32 nodeId;
+ Uint32 alive;
+ Uint32 nextList;
+ union { Uint32 prevList; Uint32 nextPool; };
+ };
+ typedef Ptr<Node> NodePtr;
+
+ /**
+ * Variables
+ */
+ NodeId c_masterNodeId;
+ SLList<Node> c_nodes;
+ NdbNodeBitmask c_aliveNodes;
+ NdbNodeBitmask c_preparingNodes;
+
+ Uint32 RtoI(Uint32 sumaRef, bool dieOnNotFound = true);
+
+ /**
+ * for all Suma's to keep track of other Suma's in Node group
+ */
+ Uint32 c_nodeGroup;
+ Uint32 c_noNodesInGroup;
+ Uint32 c_idInNodeGroup;
+ NodeId c_nodesInGroup[MAX_REPLICAS];
+
+ /**
+ * don't seem to be used
+ */
+ ArrayPool<Node> c_nodePool;
+ ArrayPool<SubCoordinator> c_subCoordinatorPool;
+ DLList<SubCoordinator> c_runningSubscriptions;
+};
+
+inline Uint32
+Suma::RtoI(Uint32 sumaRef, bool dieOnNotFound) {
+ for (Uint32 i = 0; i < c_noNodesInGroup; i++) {
+ if (sumaRef == calcSumaBlockRef(c_nodesInGroup[i]))
+ return i;
+ }
+ ndbrequire(!dieOnNotFound);
+ return RNIL;
+}
+
+#endif
diff --git a/ndb/src/kernel/blocks/suma/Suma.txt b/ndb/src/kernel/blocks/suma/Suma.txt
new file mode 100644
index 00000000000..eba031226ef
--- /dev/null
+++ b/ndb/src/kernel/blocks/suma/Suma.txt
@@ -0,0 +1,192 @@
+Protocols involving SUMA:
+
+
+
+USER SUMA UTIL
+========================================================
+CREATE_SUBID_REQ
+------------------------->
+ UTIL_SEQUENCE
+ ---------------------->
+ <----------------------
+CREATE_SUBID_CONF
+<-------------------------
+
+
+
+
+USER SUMA DICT
+========================================================
+SUB_CREATE_REQ
+------------------------->
+ case SelectiveTableSnapshot:
+ GET_TABLEID
+ ---------------------->
+ <----------------------
+SUB_CREATE_CONF
+<-------------------------
+
+
+
+
+
+USER SUMA DICT
+========================================================
+SUB_SYNC_REQ::MetaData
+------------------------->
+ case DatabaseSnapshot:
+ LIST_TABLES
+ ---------------------->
+ <----------------------
+for each table...
+ GET_TABINFO
+ ---------------------->
+ <----------------------
+SUB_META_DATA DIH
+<------------------------- =======
+ DI_FCOUNT
+ ---------------------->
+ <----------------------
+ DI_GETPRIM
+ ---------------------->
+ <----------------------
+..end for each table
+SUB_SYNC_CONF
+<-------------------------
+
+
+
+
+USER SUMA LQH
+========================================================
+SUB_SYNC_REQ::TableData
+------------------------->
+for each table...
+ SCAN_FRAG_REQ
+ ---------------------->
+ ATTRINFO
+ ---------------------->
+ SCAN_FRAG_CONF
+ <----------------------
+SUB_SYNC_CONTINUE
+<-------------------------
+------------------------->
+ SCAN_NEXTREQ
+ ---------------------->
+...end for each table
+
+
+
+??????????
+ SCAN_HBREP
+ <----------------------
+
+
+
+USER SUMA
+===============================
+SUB_START_REQ::MetaData
+------------------------->
+SUB_START_CONF
+<-------------------------
+
+
+
+USER SUMA TUP
+========================================================
+SUB_START_REQ::TableData
+------------------------->
+for each table...
+ CREATE_TRIG
+ ---------------------->
+ <----------------------
+...end for each table
+SUB_START_CONF
+<-------------------------
+
+
+USER SUMA XXX
+========================================================
+ TRANSID_AI
+ <----------------------
+SUB_TABLE_DATA
+<-------------------------
+
+
+
+
+
+USER SUMA XXX
+========================================================
+ TRIG_ATTRINFO
+ <----------------------
+ FIRE_TRIG_ORD
+ <----------------------
+SUB_TABLE_DATA
+<-------------------------
+
+
+
+USER SUMA XXX
+========================================================
+ SUB_GCP_COMPLETE_REP
+ <----------------------
+SUB_GCP_COMPLETE_REP
+<-------------------------
+
+for event only:
+SUB_GCP_COMPLETE_ACK
+------------------------->
+ when all subscribers have sent ACK on gci
+ send to all other suma's in node group:
+ SUB_GCP_COMPLETE_ACK
+ ---------------------->
+
+
+USER SUMA
+===============================
+SUB_STOP_REQ
+------------------------->
+SUB_STOP_CONF
+<-------------------------
+
+
+
+USER SUMA
+===============================
+SUB_REMOVE_REQ
+------------------------->
+SUB_REMOVE_CONF
+<-------------------------
+
+
+
+MASTER SUMA RESTARTING SUMA
+=========================================
+INCL_NODEREQ
+<---------------------------------------------------------
+for each subscription...
+SUB_CREATE_REQ
+------------------------->...
+<-------------------------
+SUB_SYNC_REQ::MetaData
+------------------------->...
+<-------------------------
+... end for each subscription
+
+
+ SUMA_START_ME (sent asynchronously in start phase 5 to all suma's in node group)
+<-------------------------
+ ------------------------->
+
+for each subscriber...
+SUB_START_REQ (not before SUMA_START_ME)
+------------------------->
+<-------------------------
+... end for each subscriber
+
+SUMA_HANDOVER_REQ
+------------------------->
+ SUMA_HANDOVER_CONF (to all suma's in node group)
+<-------------------------
+ ------------------------->
diff --git a/ndb/src/kernel/blocks/suma/SumaInit.cpp b/ndb/src/kernel/blocks/suma/SumaInit.cpp
new file mode 100644
index 00000000000..e9fba5e789c
--- /dev/null
+++ b/ndb/src/kernel/blocks/suma/SumaInit.cpp
@@ -0,0 +1,186 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Suma.hpp"
+
+#include <new>
+#include <Properties.hpp>
+#include <Configuration.hpp>
+
+SumaParticipant::SumaParticipant(const Configuration & conf) :
+ SimulatedBlock(SUMA, conf),
+ c_metaSubscribers(c_subscriberPool),
+ c_dataSubscribers(c_subscriberPool),
+ c_prepDataSubscribers(c_subscriberPool),
+ c_removeDataSubscribers(c_subscriberPool),
+ c_tables(c_tablePool_),
+ c_subscriptions(c_subscriptionPool)
+{
+ BLOCK_CONSTRUCTOR(SumaParticipant);
+
+ /**
+ * SUMA participant if
+ */
+ addRecSignal(GSN_SUB_CREATE_REQ, &SumaParticipant::execSUB_CREATE_REQ);
+ addRecSignal(GSN_SUB_REMOVE_REQ, &SumaParticipant::execSUB_REMOVE_REQ);
+ addRecSignal(GSN_SUB_START_REQ, &SumaParticipant::execSUB_START_REQ);
+ addRecSignal(GSN_SUB_STOP_REQ, &SumaParticipant::execSUB_STOP_REQ);
+ addRecSignal(GSN_SUB_SYNC_REQ, &SumaParticipant::execSUB_SYNC_REQ);
+
+ addRecSignal(GSN_SUB_STOP_CONF, &SumaParticipant::execSUB_STOP_CONF);
+ addRecSignal(GSN_SUB_STOP_REF, &SumaParticipant::execSUB_STOP_REF);
+
+ /**
+ * Dict interface
+ */
+ //addRecSignal(GSN_LIST_TABLES_REF, &SumaParticipant::execLIST_TABLES_REF);
+ addRecSignal(GSN_LIST_TABLES_CONF, &SumaParticipant::execLIST_TABLES_CONF);
+ //addRecSignal(GSN_GET_TABINFOREF, &SumaParticipant::execGET_TABINFO_REF);
+ addRecSignal(GSN_GET_TABINFO_CONF, &SumaParticipant::execGET_TABINFO_CONF);
+ addRecSignal(GSN_GET_TABINFOREF, &SumaParticipant::execGET_TABINFOREF);
+ addRecSignal(GSN_GET_TABLEID_CONF, &SumaParticipant::execGET_TABLEID_CONF);
+ addRecSignal(GSN_GET_TABLEID_REF, &SumaParticipant::execGET_TABLEID_REF);
+
+ /**
+ * Dih interface
+ */
+ //addRecSignal(GSN_DI_FCOUNTREF, &SumaParticipant::execDI_FCOUNTREF);
+ addRecSignal(GSN_DI_FCOUNTCONF, &SumaParticipant::execDI_FCOUNTCONF);
+ //addRecSignal(GSN_DIGETPRIMREF, &SumaParticipant::execDIGETPRIMREF);
+ addRecSignal(GSN_DIGETPRIMCONF, &SumaParticipant::execDIGETPRIMCONF);
+
+ /**
+ * Scan interface
+ */
+ addRecSignal(GSN_SCAN_HBREP, &SumaParticipant::execSCAN_HBREP);
+ addRecSignal(GSN_TRANSID_AI, &SumaParticipant::execTRANSID_AI);
+ addRecSignal(GSN_SCAN_FRAGREF, &SumaParticipant::execSCAN_FRAGREF);
+ addRecSignal(GSN_SCAN_FRAGCONF, &SumaParticipant::execSCAN_FRAGCONF);
+#if 0
+ addRecSignal(GSN_SUB_SYNC_CONTINUE_REF,
+ &SumaParticipant::execSUB_SYNC_CONTINUE_REF);
+#endif
+ addRecSignal(GSN_SUB_SYNC_CONTINUE_CONF,
+ &SumaParticipant::execSUB_SYNC_CONTINUE_CONF);
+
+ /**
+ * Trigger stuff
+ */
+ addRecSignal(GSN_TRIG_ATTRINFO, &SumaParticipant::execTRIG_ATTRINFO);
+ addRecSignal(GSN_FIRE_TRIG_ORD, &SumaParticipant::execFIRE_TRIG_ORD);
+
+ addRecSignal(GSN_CREATE_TRIG_REF, &Suma::execCREATE_TRIG_REF);
+ addRecSignal(GSN_CREATE_TRIG_CONF, &Suma::execCREATE_TRIG_CONF);
+ addRecSignal(GSN_DROP_TRIG_REF, &Suma::execDROP_TRIG_REF);
+ addRecSignal(GSN_DROP_TRIG_CONF, &Suma::execDROP_TRIG_CONF);
+
+ addRecSignal(GSN_SUB_GCP_COMPLETE_REP,
+ &SumaParticipant::execSUB_GCP_COMPLETE_REP);
+
+ /**
+ * @todo: fix pool sizes
+ */
+
+ c_tablePool_.setSize(MAX_TABLES);
+ c_tables.setSize(MAX_TABLES);
+
+ c_subscriptions.setSize(20); //10
+ c_subscriberPool.setSize(64);
+
+ c_subscriptionPool.setSize(64); //2
+ c_syncPool.setSize(20); //2
+ c_dataBufferPool.setSize(128);
+
+ {
+ SLList<SyncRecord> tmp(c_syncPool);
+ Ptr<SyncRecord> ptr;
+ while(tmp.seize(ptr))
+ new (ptr.p) SyncRecord(* this, c_dataBufferPool);
+ tmp.release();
+ }
+
+ for( int i = 0; i < NO_OF_BUCKETS; i++) {
+ c_buckets[i].active = false;
+ c_buckets[i].handover = false;
+ c_buckets[i].handover_started = false;
+ c_buckets[i].handoverGCI = 0;
+ }
+ c_handoverToDo = false;
+ c_lastInconsistentGCI = RNIL;
+ c_lastCompleteGCI = RNIL;
+ c_nodeFailGCI = 0;
+
+ c_failedApiNodes.clear();
+}
+
+SumaParticipant::~SumaParticipant()
+{
+}
+
+Suma::Suma(const Configuration & conf) :
+ SumaParticipant(conf),
+ c_nodes(c_nodePool),
+ c_runningSubscriptions(c_subCoordinatorPool),
+ Restart(*this)
+{
+
+ c_nodePool.setSize(MAX_NDB_NODES);
+ c_masterNodeId = getOwnNodeId();
+
+ c_nodeGroup = c_noNodesInGroup = c_idInNodeGroup = 0;
+ for (int i = 0; i < MAX_REPLICAS; i++) {
+ c_nodesInGroup[i] = 0;
+ }
+
+ c_subCoordinatorPool.setSize(10);
+
+ // Add received signals
+ addRecSignal(GSN_STTOR, &Suma::execSTTOR);
+ addRecSignal(GSN_NDB_STTOR, &Suma::execNDB_STTOR);
+ addRecSignal(GSN_DUMP_STATE_ORD, &Suma::execDUMP_STATE_ORD);
+ addRecSignal(GSN_READ_NODESCONF, &Suma::execREAD_NODESCONF);
+ addRecSignal(GSN_API_FAILREQ, &Suma::execAPI_FAILREQ);
+ addRecSignal(GSN_NODE_FAILREP, &Suma::execNODE_FAILREP);
+ addRecSignal(GSN_INCL_NODEREQ, &Suma::execINCL_NODEREQ);
+ addRecSignal(GSN_CONTINUEB, &Suma::execCONTINUEB);
+ addRecSignal(GSN_SIGNAL_DROPPED_REP, &Suma::execSIGNAL_DROPPED_REP, true);
+ addRecSignal(GSN_UTIL_SEQUENCE_CONF, &Suma::execUTIL_SEQUENCE_CONF);
+ addRecSignal(GSN_UTIL_SEQUENCE_REF, &Suma::execUTIL_SEQUENCE_REF);
+ addRecSignal(GSN_CREATE_SUBID_REQ,
+ &Suma::execCREATE_SUBID_REQ);
+
+ addRecSignal(GSN_SUB_CREATE_CONF, &Suma::execSUB_CREATE_CONF);
+ addRecSignal(GSN_SUB_CREATE_REF, &Suma::execSUB_CREATE_REF);
+ addRecSignal(GSN_SUB_SYNC_CONF, &Suma::execSUB_SYNC_CONF);
+ addRecSignal(GSN_SUB_SYNC_REF, &Suma::execSUB_SYNC_REF);
+ addRecSignal(GSN_SUB_START_CONF, &Suma::execSUB_START_CONF);
+ addRecSignal(GSN_SUB_START_REF, &Suma::execSUB_START_REF);
+
+ addRecSignal(GSN_SUMA_START_ME, &Suma::execSUMA_START_ME);
+ addRecSignal(GSN_SUMA_HANDOVER_REQ, &Suma::execSUMA_HANDOVER_REQ);
+ addRecSignal(GSN_SUMA_HANDOVER_CONF, &Suma::execSUMA_HANDOVER_CONF);
+
+ addRecSignal(GSN_SUB_GCP_COMPLETE_ACC,
+ &Suma::execSUB_GCP_COMPLETE_ACC);
+}
+
+Suma::~Suma()
+{
+}
+
+BLOCK_FUNCTIONS(Suma);
+BLOCK_FUNCTIONS(SumaParticipant);
+
diff --git a/ndb/src/kernel/blocks/trix/Makefile b/ndb/src/kernel/blocks/trix/Makefile
new file mode 100644
index 00000000000..5ac0da11f33
--- /dev/null
+++ b/ndb/src/kernel/blocks/trix/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := trix
+SOURCES = Trix.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/blocks/trix/Trix.cpp b/ndb/src/kernel/blocks/trix/Trix.cpp
new file mode 100644
index 00000000000..f058433840c
--- /dev/null
+++ b/ndb/src/kernel/blocks/trix/Trix.cpp
@@ -0,0 +1,967 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Trix.hpp"
+
+#include <string.h>
+#include <kernel_types.h>
+#include <NdbOut.hpp>
+
+#include <signaldata/ReadNodesConf.hpp>
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/BuildIndx.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <signaldata/UtilPrepare.hpp>
+#include <signaldata/UtilExecute.hpp>
+#include <signaldata/UtilRelease.hpp>
+#include <SectionReader.hpp>
+#include <AttributeHeader.hpp>
+
+#define CONSTRAINT_VIOLATION 893
+
+#define DEBUG(x) { ndbout << "TRIX::" << x << endl; }
+
+/**
+ *
+ */
+Trix::Trix(const Configuration & conf) :
+ SimulatedBlock(TRIX, conf),
+ c_theNodes(c_theNodeRecPool),
+ c_masterNodeId(0),
+ c_masterTrixRef(0),
+ c_noNodesFailed(0),
+ c_noActiveNodes(0),
+ c_theSubscriptions(c_theSubscriptionRecPool),
+ c_thePages(c_thePagePool)
+{
+ BLOCK_CONSTRUCTOR(Trix);
+
+ // Add received signals
+ addRecSignal(GSN_STTOR, &Trix::execSTTOR);
+ addRecSignal(GSN_NDB_STTOR, &Trix::execNDB_STTOR); // Forwarded from DICT
+ addRecSignal(GSN_READ_NODESCONF, &Trix::execREAD_NODESCONF);
+ addRecSignal(GSN_READ_NODESREF, &Trix::execREAD_NODESREF);
+ addRecSignal(GSN_NODE_FAILREP, &Trix::execNODE_FAILREP);
+ addRecSignal(GSN_INCL_NODEREQ, &Trix::execINCL_NODEREQ);
+ addRecSignal(GSN_DUMP_STATE_ORD, &Trix::execDUMP_STATE_ORD);
+
+ // Index build
+ addRecSignal(GSN_BUILDINDXREQ, &Trix::execBUILDINDXREQ);
+ // Dump testing
+ addRecSignal(GSN_BUILDINDXCONF, &Trix::execBUILDINDXCONF);
+ addRecSignal(GSN_BUILDINDXREF, &Trix::execBUILDINDXREF);
+
+
+ addRecSignal(GSN_UTIL_PREPARE_CONF, &Trix::execUTIL_PREPARE_CONF);
+ addRecSignal(GSN_UTIL_PREPARE_REF, &Trix::execUTIL_PREPARE_REF);
+ addRecSignal(GSN_UTIL_EXECUTE_CONF, &Trix::execUTIL_EXECUTE_CONF);
+ addRecSignal(GSN_UTIL_EXECUTE_REF, &Trix::execUTIL_EXECUTE_REF);
+ addRecSignal(GSN_UTIL_RELEASE_CONF, &Trix::execUTIL_RELEASE_CONF);
+ addRecSignal(GSN_UTIL_RELEASE_REF, &Trix::execUTIL_RELEASE_REF);
+
+
+ // Suma signals
+ addRecSignal(GSN_SUB_CREATE_CONF, &Trix::execSUB_CREATE_CONF);
+ addRecSignal(GSN_SUB_CREATE_REF, &Trix::execSUB_CREATE_REF);
+ addRecSignal(GSN_SUB_REMOVE_CONF, &Trix::execSUB_REMOVE_CONF);
+ addRecSignal(GSN_SUB_REMOVE_REF, &Trix::execSUB_REMOVE_REF);
+ addRecSignal(GSN_SUB_SYNC_CONF, &Trix::execSUB_SYNC_CONF);
+ addRecSignal(GSN_SUB_SYNC_REF, &Trix::execSUB_SYNC_REF);
+ addRecSignal(GSN_SUB_SYNC_CONTINUE_REQ, &Trix::execSUB_SYNC_CONTINUE_REQ);
+ addRecSignal(GSN_SUB_META_DATA, &Trix::execSUB_META_DATA);
+ addRecSignal(GSN_SUB_TABLE_DATA, &Trix::execSUB_TABLE_DATA);
+
+ // Allocate pool sizes
+ c_theAttrOrderBufferPool.setSize(100);
+ c_theSubscriptionRecPool.setSize(100);
+ c_thePagePool.setSize(16);
+
+ ArrayList<SubscriptionRecord> subscriptions(c_theSubscriptionRecPool);
+ SubscriptionRecPtr subptr;
+ while(subscriptions.seize(subptr) == true) {
+ new (subptr.p) SubscriptionRecord(c_theAttrOrderBufferPool);
+ }
+ subscriptions.release();
+}
+
+/**
+ *
+ */
+Trix::~Trix()
+{
+}
+
+/**
+ *
+ */
+void Trix::execSTTOR(Signal* signal)
+{
+ jamEntry();
+
+ //const Uint32 startphase = signal->theData[1];
+ const Uint32 theSignalKey = signal->theData[6];
+
+ signal->theData[0] = theSignalKey;
+ signal->theData[3] = 1;
+ signal->theData[4] = 255; // No more start phases from missra
+ sendSignal(NDBCNTR_REF, GSN_STTORRY, signal, 5, JBB);
+ return;
+}//Trix::execSTTOR()
+
+/**
+ *
+ */
+void Trix::execNDB_STTOR(Signal* signal)
+{
+ jamEntry();
+ BlockReference ndbcntrRef = signal->theData[0];
+ Uint16 startphase = signal->theData[2]; /* RESTART PHASE */
+ Uint16 mynode = signal->theData[1];
+ //Uint16 restarttype = signal->theData[3];
+ //UintR configInfo1 = signal->theData[6]; /* CONFIGRATION INFO PART 1 */
+ //UintR configInfo2 = signal->theData[7]; /* CONFIGRATION INFO PART 2 */
+ switch (startphase) {
+ case 3:
+ jam();
+ /* SYMBOLIC START PHASE 4 */
+ /* ABSOLUTE PHASE 5 */
+ /* REQUEST NODE IDENTITIES FROM DBDIH */
+ signal->theData[0] = calcTrixBlockRef(mynode);
+ sendSignal(ndbcntrRef, GSN_READ_NODESREQ, signal, 1, JBB);
+ return;
+ break;
+ case 6:
+ break;
+ default:
+ break;
+ }
+}
+
+/**
+ *
+ */
+void Trix::execREAD_NODESCONF(Signal* signal)
+{
+ jamEntry();
+
+ ReadNodesConf * const readNodes = (ReadNodesConf *)signal->getDataPtr();
+ //Uint32 noOfNodes = readNodes->noOfNodes;
+ NodeRecPtr nodeRecPtr;
+
+ c_masterNodeId = readNodes->masterNodeId;
+ c_masterTrixRef = RNIL;
+ c_noNodesFailed = 0;
+
+ for(unsigned i = 0; i < MAX_NDB_NODES; i++) {
+ jam();
+ if(NodeBitmask::get(readNodes->allNodes, i)) {
+ // Node is defined
+ jam();
+ ndbrequire(c_theNodes.seizeId(nodeRecPtr, i));
+ nodeRecPtr.p->trixRef = calcTrixBlockRef(i);
+ if (i == c_masterNodeId) {
+ c_masterTrixRef = nodeRecPtr.p->trixRef;
+ }
+ if(NodeBitmask::get(readNodes->inactiveNodes, i)){
+ // Node is not active
+ jam();
+ /**-----------------------------------------------------------------
+ * THIS NODE IS DEFINED IN THE CLUSTER BUT IS NOT ALIVE CURRENTLY.
+ * WE ADD THE NODE TO THE SET OF FAILED NODES AND ALSO SET THE
+ * BLOCKSTATE TO BUSY TO AVOID ADDING TRIGGERS OR INDEXES WHILE
+ * NOT ALL NODES ARE ALIVE.
+ *------------------------------------------------------------------*/
+ arrGuard(c_noNodesFailed, MAX_NDB_NODES);
+ nodeRecPtr.p->alive = false;
+ c_noNodesFailed++;
+ c_blockState = Trix::NODE_FAILURE;
+ }
+ else {
+ // Node is active
+ jam();
+ c_noActiveNodes++;
+ nodeRecPtr.p->alive = true;
+ }
+ }
+ }
+ if (c_noNodesFailed == 0) {
+ c_blockState = Trix::STARTED;
+ }
+}
+
+/**
+ *
+ */
+void Trix::execREAD_NODESREF(Signal* signal)
+{
+ // NYI
+}
+
+/**
+ *
+ */
+void Trix::execNODE_FAILREP(Signal* signal)
+{
+ jamEntry();
+ NodeFailRep * const nodeFail = (NodeFailRep *) signal->getDataPtr();
+
+ //Uint32 failureNr = nodeFail->failNo;
+ //Uint32 numberNodes = nodeFail->noOfNodes;
+ Uint32 masterNodeId = nodeFail->masterNodeId;
+
+ NodeRecPtr nodeRecPtr;
+
+ for(c_theNodes.first(nodeRecPtr);
+ nodeRecPtr.i != RNIL;
+ c_theNodes.next(nodeRecPtr)) {
+ if(NodeBitmask::get(nodeFail->theNodes, nodeRecPtr.i)) {
+ nodeRecPtr.p->alive = false;
+ c_noNodesFailed++;
+ c_noActiveNodes--;
+ }
+ }
+ if (c_masterNodeId != masterNodeId) {
+ c_masterNodeId = masterNodeId;
+ NodeRecord* nodeRec = c_theNodes.getPtr(masterNodeId);
+ c_masterTrixRef = nodeRec->trixRef;
+ }
+}
+
+/**
+ *
+ */
+void Trix::execINCL_NODEREQ(Signal* signal)
+{
+ jamEntry();
+ UintR node_id = signal->theData[1];
+ NodeRecord* nodeRec = c_theNodes.getPtr(node_id);
+ nodeRec->alive = true;
+ c_noNodesFailed--;
+ c_noActiveNodes++;
+ nodeRec->trixRef = calcTrixBlockRef(node_id);
+ if (c_noNodesFailed == 0) {
+ c_blockState = Trix::STARTED;
+ }
+}
+
+// Debugging
+void
+Trix::execDUMP_STATE_ORD(Signal* signal)
+{
+ jamEntry();
+
+ DumpStateOrd * dumpStateOrd = (DumpStateOrd *)signal->getDataPtr();
+
+ switch(dumpStateOrd->args[0]) {
+ case(300): {// ok
+ // index2 -T; index2 -I -n10000; index2 -c
+ // all dump 300 0 0 0 0 0 4 2
+ // select_count INDEX0000
+ BuildIndxReq * buildIndxReq = (BuildIndxReq *)signal->getDataPtrSend();
+
+ MEMCOPY_NO_WORDS(buildIndxReq,
+ signal->theData + 1,
+ BuildIndxReq::SignalLength);
+ buildIndxReq->setUserRef(reference()); // return to me
+ buildIndxReq->setParallelism(10);
+ Uint32 indexColumns[1] = {1};
+ Uint32 keyColumns[1] = {0};
+ struct LinearSectionPtr orderPtr[2];
+ buildIndxReq->setColumnOrder(indexColumns, 1, keyColumns, 1, orderPtr);
+ sendSignal(reference(),
+ GSN_BUILDINDXREQ,
+ signal,
+ BuildIndxReq::SignalLength,
+ JBB,
+ orderPtr,
+ BuildIndxReq::NoOfSections);
+ break;
+ }
+ case(301): { // ok
+ // index2 -T; index2 -I -n10000; index2 -c -p
+ // all dump 301 0 0 0 0 0 4 2
+ // select_count INDEX0000
+ BuildIndxReq * buildIndxReq = (BuildIndxReq *)signal->getDataPtrSend();
+
+ MEMCOPY_NO_WORDS(buildIndxReq,
+ signal->theData + 1,
+ BuildIndxReq::SignalLength);
+ buildIndxReq->setUserRef(reference()); // return to me
+ buildIndxReq->setParallelism(10);
+ Uint32 indexColumns[2] = {0, 1};
+ Uint32 keyColumns[1] = {0};
+ struct LinearSectionPtr orderPtr[2];
+ buildIndxReq->setColumnOrder(indexColumns, 2, keyColumns, 1, orderPtr);
+ sendSignal(reference(),
+ GSN_BUILDINDXREQ,
+ signal,
+ BuildIndxReq::SignalLength,
+ JBB,
+ orderPtr,
+ BuildIndxReq::NoOfSections);
+ break;
+ }
+ case(302): { // ok
+ // index -T; index -I -n1000; index -c -p
+ // all dump 302 0 0 0 0 0 4 2
+ // select_count PNUMINDEX0000
+ BuildIndxReq * buildIndxReq = (BuildIndxReq *)signal->getDataPtrSend();
+
+ MEMCOPY_NO_WORDS(buildIndxReq,
+ signal->theData + 1,
+ BuildIndxReq::SignalLength);
+ buildIndxReq->setUserRef(reference()); // return to me
+ buildIndxReq->setParallelism(10);
+ Uint32 indexColumns[3] = {0, 3, 5};
+ Uint32 keyColumns[1] = {0};
+ struct LinearSectionPtr orderPtr[2];
+ buildIndxReq->setColumnOrder(indexColumns, 3, keyColumns, 1, orderPtr);
+ sendSignal(reference(),
+ GSN_BUILDINDXREQ,
+ signal,
+ BuildIndxReq::SignalLength,
+ JBB,
+ orderPtr,
+ BuildIndxReq::NoOfSections);
+ break;
+ }
+ case(303): { // ok
+ // index -T -2; index -I -2 -n1000; index -c -p
+ // all dump 303 0 0 0 0 0 4 2
+ // select_count PNUMINDEX0000
+ BuildIndxReq * buildIndxReq = (BuildIndxReq *)signal->getDataPtrSend();
+
+ MEMCOPY_NO_WORDS(buildIndxReq,
+ signal->theData + 1,
+ BuildIndxReq::SignalLength);
+ buildIndxReq->setUserRef(reference()); // return to me
+ buildIndxReq->setParallelism(10);
+ Uint32 indexColumns[3] = {0, 3, 5};
+ Uint32 keyColumns[2] = {0, 1};
+ struct LinearSectionPtr orderPtr[2];
+ buildIndxReq->setColumnOrder(indexColumns, 3, keyColumns, 2, orderPtr);
+ sendSignal(reference(),
+ GSN_BUILDINDXREQ,
+ signal,
+ BuildIndxReq::SignalLength,
+ JBB,
+ orderPtr,
+ BuildIndxReq::NoOfSections);
+ break;
+ }
+ case(304): { // ok
+ // index -T -L; index -I -L -n1000; index -c -p
+ // all dump 304 0 0 0 0 0 4 2
+ // select_count PNUMINDEX0000
+ BuildIndxReq * buildIndxReq = (BuildIndxReq *)signal->getDataPtrSend();
+
+ MEMCOPY_NO_WORDS(buildIndxReq,
+ signal->theData + 1,
+ BuildIndxReq::SignalLength);
+ buildIndxReq->setUserRef(reference()); // return to me
+ buildIndxReq->setParallelism(10);
+ Uint32 indexColumns[3] = {0, 3, 5};
+ Uint32 keyColumns[1] = {0};
+ struct LinearSectionPtr orderPtr[2];
+ buildIndxReq->setColumnOrder(indexColumns, 3, keyColumns, 1, orderPtr);
+ sendSignal(reference(),
+ GSN_BUILDINDXREQ,
+ signal,
+ BuildIndxReq::SignalLength,
+ JBB,
+ orderPtr,
+ BuildIndxReq::NoOfSections);
+ break;
+ }
+ case(305): { // ok
+ // index -T -2 -L; index -I -2 -L -n1000; index -c -p
+ // all dump 305 0 0 0 0 0 4 2
+ // select_count PNUMINDEX0000
+ BuildIndxReq * buildIndxReq = (BuildIndxReq *)signal->getDataPtrSend();
+
+ MEMCOPY_NO_WORDS(buildIndxReq,
+ signal->theData + 1,
+ BuildIndxReq::SignalLength);
+ buildIndxReq->setUserRef(reference()); // return to me
+ buildIndxReq->setParallelism(10);
+ Uint32 indexColumns[3] = {0, 3, 5};
+ Uint32 keyColumns[2] = {0, 1};
+ struct LinearSectionPtr orderPtr[2];
+ buildIndxReq->setColumnOrder(indexColumns, 3, keyColumns, 2, orderPtr);
+ sendSignal(reference(),
+ GSN_BUILDINDXREQ,
+ signal,
+ BuildIndxReq::SignalLength,
+ JBB,
+ orderPtr,
+ BuildIndxReq::NoOfSections);
+ break;
+ }
+ default: {
+ // Ignore
+ }
+ }
+}
+
+// Build index
+/**
+ *
+ */
+void Trix:: execBUILDINDXREQ(Signal* signal)
+{
+ jamEntry();
+ BuildIndxReq * buildIndxReq = (BuildIndxReq *)signal->getDataPtr();
+
+ // Seize a subscription record
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+
+ if (!c_theSubscriptions.seizeId(subRecPtr, buildIndxReq->getBuildId())) {
+ // Failed to allocate subscription record
+ BuildIndxRef * buildIndxRef = (BuildIndxRef *)signal->getDataPtrSend();
+
+ buildIndxRef->setErrorCode(BuildIndxRef::AllocationFailure);
+ releaseSections(signal);
+ sendSignal(buildIndxReq->getUserRef(),
+ GSN_BUILDINDXREF, signal, BuildIndxRef::SignalLength, JBB);
+ return;
+ }
+ subRec = subRecPtr.p;
+ subRec->errorCode = BuildIndxRef::NoError;
+ subRec->userReference = buildIndxReq->getUserRef();
+ subRec->connectionPtr = buildIndxReq->getConnectionPtr();
+ subRec->subscriptionId = buildIndxReq->getBuildId();
+ subRec->subscriptionKey = buildIndxReq->getBuildKey();
+ subRec->indexType = buildIndxReq->getIndexType();
+ subRec->sourceTableId = buildIndxReq->getTableId();
+ subRec->targetTableId = buildIndxReq->getIndexId();
+ subRec->parallelism = buildIndxReq->getParallelism();
+ subRec->expectedConf = 0;
+ subRec->subscriptionCreated = false;
+ subRec->pendingSubSyncContinueConf = false;
+ subRec->prepareId = RNIL;
+
+ // Get column order segments
+ Uint32 noOfSections = signal->getNoOfSections();
+ if(noOfSections > 0) {
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr, BuildIndxReq::INDEX_COLUMNS);
+ append(subRec->attributeOrder, ptr, getSectionSegmentPool());
+ subRec->noOfIndexColumns = ptr.sz;
+ }
+ if(noOfSections > 1) {
+ SegmentedSectionPtr ptr;
+ signal->getSection(ptr, BuildIndxReq::KEY_COLUMNS);
+ append(subRec->attributeOrder, ptr, getSectionSegmentPool());
+ subRec->noOfKeyColumns = ptr.sz;
+ }
+#if 0
+ // Debugging
+ printf("Trix:: execBUILDINDXREQ: Attribute order:\n");
+ subRec->attributeOrder.print(stdout);
+#endif
+ releaseSections(signal);
+ prepareInsertTransactions(signal, subRecPtr);
+}
+
+void Trix:: execBUILDINDXCONF(Signal* signal)
+{
+ printf("Trix:: execBUILDINDXCONF\n");
+}
+
+void Trix:: execBUILDINDXREF(Signal* signal)
+{
+ printf("Trix:: execBUILDINDXREF\n");
+}
+
+void Trix::execUTIL_PREPARE_CONF(Signal* signal)
+{
+ jamEntry();
+ UtilPrepareConf * utilPrepareConf = (UtilPrepareConf *)signal->getDataPtr();
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+
+ subRecPtr.i = utilPrepareConf->senderData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("Trix::execUTIL_PREPARE_CONF: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ subRecPtr.p = subRec;
+ subRec->prepareId = utilPrepareConf->prepareId;
+ setupSubscription(signal, subRecPtr);
+}
+
+void Trix::execUTIL_PREPARE_REF(Signal* signal)
+{
+ jamEntry();
+ UtilPrepareRef * utilPrepareRef = (UtilPrepareRef *)signal->getDataPtr();
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+
+ subRecPtr.i = utilPrepareRef->senderData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("Trix::execUTIL_PREPARE_REF: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ subRecPtr.p = subRec;
+ subRec->errorCode = BuildIndxRef::InternalError;
+}
+
+void Trix::execUTIL_EXECUTE_CONF(Signal* signal)
+{
+ jamEntry();
+ UtilExecuteConf * utilExecuteConf = (UtilExecuteConf *)signal->getDataPtr();
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+
+ subRecPtr.i = utilExecuteConf->senderData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("rix::execUTIL_EXECUTE_CONF: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ subRecPtr.p = subRec;
+ subRec->expectedConf--;
+ checkParallelism(signal, subRec);
+ if (subRec->expectedConf == 0)
+ buildComplete(signal, subRecPtr);
+}
+
+void Trix::execUTIL_EXECUTE_REF(Signal* signal)
+{
+ jamEntry();
+ UtilExecuteRef * utilExecuteRef = (UtilExecuteRef *)signal->getDataPtr();
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+
+ subRecPtr.i = utilExecuteRef->senderData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("Trix::execUTIL_EXECUTE_REF: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ subRecPtr.p = subRec;
+ ndbrequire(utilExecuteRef->errorCode == UtilExecuteRef::TCError);
+ if(utilExecuteRef->TCErrorCode == CONSTRAINT_VIOLATION)
+ buildFailed(signal, subRecPtr, BuildIndxRef::IndexNotUnique);
+ else
+ buildFailed(signal, subRecPtr, BuildIndxRef::InternalError);
+}
+
+void Trix::execSUB_CREATE_CONF(Signal* signal)
+{
+ jamEntry();
+ SubCreateConf * subCreateConf = (SubCreateConf *)signal->getDataPtr();
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+
+ subRecPtr.i = subCreateConf->subscriberData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("Trix::execSUB_CREATE_CONF: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ ndbrequire(subRec->subscriptionId == subCreateConf->subscriptionId);
+ ndbrequire(subRec->subscriptionKey == subCreateConf->subscriptionKey);
+ subRec->subscriptionCreated = true;
+ subRecPtr.p = subRec;
+ setupTableScan(signal, subRecPtr);
+}
+
+void Trix::execSUB_CREATE_REF(Signal* signal)
+{
+ jamEntry();
+ // THIS SIGNAL IS NEVER SENT FROM SUMA?
+ /*
+ SubCreateRef * subCreateRef = (SubCreateRef *)signal->getDataPtr();
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+
+ subRecPtr.i = subCreateRef->subscriberData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("Trix::execSUB_CREATE_REF: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ subRecPtr.p = subRec;
+ buildFailed(signal, subRecPtr, BuildIndxRef::InternalError);
+ */
+}
+
+void Trix::execSUB_SYNC_CONF(Signal* signal)
+{
+ jamEntry();
+ SubSyncConf * subSyncConf = (SubSyncConf *)signal->getDataPtr();
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+
+ subRecPtr.i = subSyncConf->subscriberData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("Trix::execSUB_SYNC_CONF: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ ndbrequire(subRec->subscriptionId == subSyncConf->subscriptionId);
+ ndbrequire(subRec->subscriptionKey == subSyncConf->subscriptionKey);
+ subRecPtr.p = subRec;
+ if(subSyncConf->part == SubscriptionData::MetaData)
+ startTableScan(signal, subRecPtr);
+ else {
+ subRec->expectedConf--;
+ checkParallelism(signal, subRec);
+ if (subRec->expectedConf == 0)
+ buildComplete(signal, subRecPtr);
+ }
+}
+
+void Trix::execSUB_SYNC_REF(Signal* signal)
+{
+ jamEntry();
+ SubSyncRef * subSyncRef = (SubSyncRef *)signal->getDataPtr();
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+
+ subRecPtr.i = subSyncRef->subscriberData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("Trix::execSUB_SYNC_REF: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ subRecPtr.p = subRec;
+ buildFailed(signal, subRecPtr, BuildIndxRef::InternalError);
+}
+
+void Trix::execSUB_SYNC_CONTINUE_REQ(Signal* signal)
+{
+ SubSyncContinueReq * subSyncContinueReq =
+ (SubSyncContinueReq *) signal->getDataPtr();
+
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+ subRecPtr.i = subSyncContinueReq->subscriberData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("Trix::execSUB_SYNC_CONTINUE_REQ: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ subRecPtr.p = subRec;
+ subRec->pendingSubSyncContinueConf = true;
+ checkParallelism(signal, subRec);
+}
+
+void Trix::execSUB_META_DATA(Signal* signal)
+{
+ jamEntry();
+}
+
+void Trix::execSUB_TABLE_DATA(Signal* signal)
+{
+ jamEntry();
+ SubTableData * subTableData = (SubTableData *)signal->getDataPtr();
+ SubscriptionRecPtr subRecPtr;
+ SubscriptionRecord* subRec;
+ subRecPtr.i = subTableData->subscriberData;
+ if ((subRec = c_theSubscriptions.getPtr(subRecPtr.i)) == NULL) {
+ printf("Trix::execSUB_TABLE_DATA: Failed to find subscription data %u\n", subRecPtr.i);
+ return;
+ }
+ subRecPtr.p = subRec;
+ SegmentedSectionPtr headerPtr, dataPtr;
+ if (!signal->getSection(headerPtr, 0)) {
+ printf("Trix::execSUB_TABLE_DATA: Failed to get header section\n");
+ }
+ if (!signal->getSection(dataPtr, 1)) {
+ printf("Trix::execSUB_TABLE_DATA: Failed to get data section\n");
+ }
+ executeInsertTransaction(signal, subRecPtr, headerPtr, dataPtr);
+}
+
+void Trix::setupSubscription(Signal* signal, SubscriptionRecPtr subRecPtr)
+{
+ Uint32 attributeList[MAX_ATTRIBUTES_IN_TABLE * 2];
+ SubCreateReq * subCreateReq = (SubCreateReq *)signal->getDataPtrSend();
+ SubscriptionRecord* subRec = subRecPtr.p;
+// Uint32 listLen = subRec->noOfIndexColumns + subRec->noOfKeyColumns;
+ AttrOrderBuffer::DataBufferIterator iter;
+ Uint32 i = 0;
+
+ jam();
+ bool moreAttributes = subRec->attributeOrder.first(iter);
+ while (moreAttributes) {
+ attributeList[i++] = *iter.data;
+ moreAttributes = subRec->attributeOrder.next(iter);
+ }
+ // Merge index and key column segments
+ struct LinearSectionPtr orderPtr[3];
+ orderPtr[0].p = attributeList;
+ orderPtr[0].sz = subRec->attributeOrder.getSize();
+
+
+ subCreateReq->subscriberRef = reference();
+ subCreateReq->subscriberData = subRecPtr.i;
+ subCreateReq->subscriptionId = subRec->subscriptionId;
+ subCreateReq->subscriptionKey = subRec->subscriptionKey;
+ subCreateReq->tableId = subRec->sourceTableId;
+ subCreateReq->subscriptionType = SubCreateReq::SingleTableScan;
+
+ sendSignal(SUMA_REF, GSN_SUB_CREATE_REQ,
+ signal, SubCreateReq::SignalLength, JBB, orderPtr, 1);
+}
+
+void Trix::setupTableScan(Signal* signal, SubscriptionRecPtr subRecPtr)
+{
+ SubSyncReq * subSyncReq = (SubSyncReq *)signal->getDataPtrSend();
+
+ jam();
+ subSyncReq->subscriptionId = subRecPtr.i;
+ subSyncReq->subscriptionKey = subRecPtr.p->subscriptionKey;
+ subSyncReq->part = SubscriptionData::MetaData;
+ sendSignal(SUMA_REF, GSN_SUB_SYNC_REQ,
+ signal, SubSyncReq::SignalLength, JBB);
+}
+
+void Trix::startTableScan(Signal* signal, SubscriptionRecPtr subRecPtr)
+{
+ jam();
+ subRecPtr.p->expectedConf = 1;
+ SubSyncReq * subSyncReq = (SubSyncReq *)signal->getDataPtrSend();
+
+ subSyncReq->subscriptionId = subRecPtr.i;
+ subSyncReq->subscriptionKey = subRecPtr.p->subscriptionKey;
+ subSyncReq->part = SubscriptionData::TableData;
+ sendSignal(SUMA_REF, GSN_SUB_SYNC_REQ,
+ signal, SubSyncReq::SignalLength, JBB);
+}
+
+void Trix::prepareInsertTransactions(Signal* signal,
+ SubscriptionRecPtr subRecPtr)
+{
+ SubscriptionRecord* subRec = subRecPtr.p;
+ UtilPrepareReq * utilPrepareReq =
+ (UtilPrepareReq *)signal->getDataPtrSend();
+
+ jam();
+ utilPrepareReq->senderRef = reference();
+ utilPrepareReq->senderData = subRecPtr.i;
+
+ const Uint32 pageSizeInWords = 128;
+ Uint32 propPage[pageSizeInWords];
+ LinearWriter w(&propPage[0],128);
+ w.first();
+ w.add(UtilPrepareReq::NoOfOperations, 1);
+ w.add(UtilPrepareReq::OperationType, UtilPrepareReq::Write);
+ w.add(UtilPrepareReq::TableId, subRec->targetTableId);
+ // Add index attributes in increasing order and one PK attribute
+ for(Uint32 i = 0; i < subRec->noOfIndexColumns + 1; i++)
+ w.add(UtilPrepareReq::AttributeId, i);
+
+#if 0
+ // Debugging
+ SimplePropertiesLinearReader reader(propPage, w.getWordsUsed());
+ printf("Trix::prepareInsertTransactions: Sent SimpleProperties:\n");
+ reader.printAll(ndbout);
+#endif
+
+ struct LinearSectionPtr sectionsPtr[UtilPrepareReq::NoOfSections];
+ sectionsPtr[UtilPrepareReq::PROPERTIES_SECTION].p = propPage;
+ sectionsPtr[UtilPrepareReq::PROPERTIES_SECTION].sz = w.getWordsUsed();
+ sendSignal(DBUTIL_REF, GSN_UTIL_PREPARE_REQ, signal,
+ UtilPrepareReq::SignalLength, JBB,
+ sectionsPtr, UtilPrepareReq::NoOfSections);
+}
+
+void Trix::executeInsertTransaction(Signal* signal,
+ SubscriptionRecPtr subRecPtr,
+ SegmentedSectionPtr headerPtr,
+ SegmentedSectionPtr dataPtr)
+{
+ jam();
+ SubscriptionRecord* subRec = subRecPtr.p;
+ UtilExecuteReq * utilExecuteReq =
+ (UtilExecuteReq *)signal->getDataPtrSend();
+ Uint32* headerBuffer = signal->theData + 25;
+ Uint32* dataBuffer = headerBuffer + headerPtr.sz;
+
+ utilExecuteReq->senderRef = reference();
+ utilExecuteReq->senderData = subRecPtr.i;
+ utilExecuteReq->prepareId = subRec->prepareId;
+#if 0
+ printf("Header size %u\n", headerPtr.sz);
+ for(int i = 0; i < headerPtr.sz; i++)
+ printf("H'%.8x ", headerBuffer[i]);
+ printf("\n");
+
+ printf("Data size %u\n", dataPtr.sz);
+ for(int i = 0; i < dataPtr.sz; i++)
+ printf("H'%.8x ", dataBuffer[i]);
+ printf("\n");
+#endif
+ // Save scan result in linear buffers
+ copy(headerBuffer, headerPtr);
+ copy(dataBuffer, dataPtr);
+
+ // Calculate packed key size
+ Uint32 noOfKeyData = 0;
+ for(Uint32 i = 0; i < headerPtr.sz; i++) {
+ AttributeHeader* keyAttrHead = (AttributeHeader *) headerBuffer + i;
+
+ // Filter out single NULL attributes
+ if (keyAttrHead->isNULL() && (i == (Uint32)0) && (headerPtr.sz == (Uint32)2))
+ return;
+
+ if (i < subRec->noOfIndexColumns)
+ // Renumber index attributes in consequtive order
+ keyAttrHead->setAttributeId(i);
+ else
+ // Calculate total size of PK attribute
+ noOfKeyData += keyAttrHead->getDataSize();
+ }
+ // Increase expected CONF count
+ subRec->expectedConf++;
+
+ // Pack key attributes
+ AttributeHeader::init(headerBuffer + subRec->noOfIndexColumns,
+ subRec->noOfIndexColumns,
+ noOfKeyData);
+
+ struct LinearSectionPtr sectionsPtr[UtilExecuteReq::NoOfSections];
+ sectionsPtr[UtilExecuteReq::HEADER_SECTION].p = headerBuffer;
+ sectionsPtr[UtilExecuteReq::HEADER_SECTION].sz =
+ subRec->noOfIndexColumns + 1;
+ sectionsPtr[UtilExecuteReq::DATA_SECTION].p = dataBuffer;
+ sectionsPtr[UtilExecuteReq::DATA_SECTION].sz = dataPtr.sz;
+ sendSignal(DBUTIL_REF, GSN_UTIL_EXECUTE_REQ, signal,
+ UtilExecuteReq::SignalLength, JBB,
+ sectionsPtr, UtilExecuteReq::NoOfSections);
+}
+
+void Trix::buildComplete(Signal* signal, SubscriptionRecPtr subRecPtr)
+{
+ SubRemoveReq * const req = (SubRemoveReq*)signal->getDataPtrSend();
+ req->senderRef = reference();
+ req->senderData = subRecPtr.i;
+ req->subscriptionId = subRecPtr.p->subscriptionId;
+ req->subscriptionKey = subRecPtr.p->subscriptionKey;
+ sendSignal(SUMA_REF, GSN_SUB_REMOVE_REQ, signal,
+ SubRemoveReq::SignalLength, JBB);
+}
+
+void Trix::buildFailed(Signal* signal,
+ SubscriptionRecPtr subRecPtr,
+ BuildIndxRef::ErrorCode errorCode)
+{
+ SubscriptionRecord* subRec = subRecPtr.p;
+
+ subRec->errorCode = errorCode;
+ // Continue accumulating since we currently cannot stop SUMA
+ subRec->expectedConf--;
+ checkParallelism(signal, subRec);
+ if (subRec->expectedConf == 0)
+ buildComplete(signal, subRecPtr);
+}
+
+void
+Trix::execSUB_REMOVE_REF(Signal* signal){
+ jamEntry();
+ //@todo
+ ndbrequire(false);
+}
+
+void
+Trix::execSUB_REMOVE_CONF(Signal* signal){
+ jamEntry();
+
+ SubRemoveConf * const conf = (SubRemoveConf*)signal->getDataPtrSend();
+
+ SubscriptionRecPtr subRecPtr;
+ c_theSubscriptions.getPtr(subRecPtr, conf->senderData);
+
+ if(subRecPtr.p->prepareId != RNIL){
+ jam();
+
+ UtilReleaseReq * const req = (UtilReleaseReq*)signal->getDataPtrSend();
+ req->prepareId = subRecPtr.p->prepareId;
+ req->senderData = subRecPtr.i;
+
+ sendSignal(DBUTIL_REF, GSN_UTIL_RELEASE_REQ, signal,
+ UtilReleaseReq::SignalLength , JBB);
+ return;
+ }
+
+ {
+ UtilReleaseConf * const conf = (UtilReleaseConf*)signal->getDataPtrSend();
+ conf->senderData = subRecPtr.i;
+ execUTIL_RELEASE_CONF(signal);
+ }
+}
+
+void
+Trix::execUTIL_RELEASE_REF(Signal* signal){
+ jamEntry();
+ ndbrequire(false);
+}
+
+void
+Trix::execUTIL_RELEASE_CONF(Signal* signal){
+
+ UtilReleaseConf * const conf = (UtilReleaseConf*)signal->getDataPtrSend();
+
+ SubscriptionRecPtr subRecPtr;
+ c_theSubscriptions.getPtr(subRecPtr, conf->senderData);
+
+ if(subRecPtr.p->errorCode == BuildIndxRef::NoError){
+ // Build is complete, reply to original sender
+ BuildIndxConf * buildIndxConf = (BuildIndxConf *)signal->getDataPtrSend();
+ buildIndxConf->setUserRef(subRecPtr.p->userReference);
+ buildIndxConf->setConnectionPtr(subRecPtr.p->connectionPtr);
+ buildIndxConf->setRequestType(BuildIndxReq::RT_TRIX);
+ buildIndxConf->setIndexType(subRecPtr.p->indexType);
+ buildIndxConf->setTableId(subRecPtr.p->sourceTableId);
+ buildIndxConf->setIndexId(subRecPtr.p->targetTableId);
+
+ sendSignal(subRecPtr.p->userReference, GSN_BUILDINDXCONF, signal,
+ BuildIndxConf::SignalLength , JBB);
+ } else {
+ // Build failed, reply to original sender
+ BuildIndxRef * buildIndxRef = (BuildIndxRef *)signal->getDataPtrSend();
+ buildIndxRef->setUserRef(subRecPtr.p->userReference);
+ buildIndxRef->setConnectionPtr(subRecPtr.p->connectionPtr);
+ buildIndxRef->setRequestType(BuildIndxReq::RT_TRIX);
+ buildIndxRef->setIndexType(subRecPtr.p->indexType);
+ buildIndxRef->setTableId(subRecPtr.p->sourceTableId);
+ buildIndxRef->setIndexId(subRecPtr.p->targetTableId);
+ buildIndxRef->setErrorCode(subRecPtr.p->errorCode);
+
+ sendSignal(subRecPtr.p->userReference, GSN_BUILDINDXREF, signal,
+ BuildIndxRef::SignalLength , JBB);
+ }
+
+ // Release subscription record
+ subRecPtr.p->attributeOrder.release();
+ c_theSubscriptions.release(subRecPtr.i);
+}
+
+void Trix::checkParallelism(Signal* signal, SubscriptionRecord* subRec)
+{
+ if ((subRec->pendingSubSyncContinueConf) &&
+ (subRec->expectedConf < subRec->parallelism)) {
+ SubSyncContinueConf * subSyncContinueConf =
+ (SubSyncContinueConf *) signal->getDataPtrSend();
+ subSyncContinueConf->subscriptionId = subRec->subscriptionId;
+ subSyncContinueConf->subscriptionKey = subRec->subscriptionKey;
+ sendSignal(SUMA_REF, GSN_SUB_SYNC_CONTINUE_CONF, signal,
+ SubSyncContinueConf::SignalLength , JBB);
+ subRec->pendingSubSyncContinueConf = false;
+ }
+}
+
+BLOCK_FUNCTIONS(Trix);
diff --git a/ndb/src/kernel/blocks/trix/Trix.hpp b/ndb/src/kernel/blocks/trix/Trix.hpp
new file mode 100644
index 00000000000..05563559b77
--- /dev/null
+++ b/ndb/src/kernel/blocks/trix/Trix.hpp
@@ -0,0 +1,203 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TRIX_H
+#define TRIX_H
+
+#include <SimulatedBlock.hpp>
+#include <trigger_definitions.h>
+#include <DataBuffer.hpp>
+#include <ArrayList.hpp>
+#include <SimpleProperties.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/CreateTrig.hpp>
+#include <signaldata/BuildIndx.hpp>
+
+// Error codes
+#define INTERNAL_ERROR_ILLEGAL_CALL 4344
+#define INTERNAL_ERROR_TRIX_BUSY 4345
+
+/**
+ * TRIX - This block manages triggers and index (in coop with DICT)
+ */
+class Trix : public SimulatedBlock
+{
+public:
+ Trix(const class Configuration & conf);
+ virtual ~Trix();
+
+private:
+ // Private attributes
+
+ BLOCK_DEFINES(Trix);
+
+ // Declared but not defined
+ //DBtrix(const Trix &obj);
+ //void operator = (const Trix &);
+
+ // Block state
+ enum BlockState {
+ NOT_STARTED,
+ STARTED,
+ NODE_FAILURE,
+ IDLE,
+ BUSY
+ };
+
+ BlockState c_blockState;
+
+ // Node data needed when communicating with remote TRIX:es
+ struct NodeRecord {
+ bool alive;
+ BlockReference trixRef;
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+ };
+
+ typedef Ptr<NodeRecord> NodeRecPtr;
+
+ /**
+ * The pool of node records
+ */
+ ArrayPool<NodeRecord> c_theNodeRecPool;
+
+ /**
+ * The list of other NDB nodes
+ */
+ ArrayList<NodeRecord> c_theNodes;
+
+ Uint32 c_masterNodeId;
+ BlockReference c_masterTrixRef;
+ Uint16 c_noNodesFailed;
+ Uint16 c_noActiveNodes;
+
+ // Subscription data, when communicating with SUMA
+ enum RequestType {
+ TABLE_REORG = 0,
+ INDEX_BUILD = 1
+ };
+
+ typedef DataBuffer<11> AttrOrderBuffer;
+
+ AttrOrderBuffer::DataBufferPool c_theAttrOrderBufferPool;
+
+ struct SubscriptionRecord {
+ SubscriptionRecord(AttrOrderBuffer::DataBufferPool & aop):
+ attributeOrder(aop)
+ {}
+ RequestType requestType;
+ BlockReference userReference; // For user
+ Uint32 connectionPtr; // For user
+ Uint32 subscriptionId; // For Suma
+ Uint32 subscriptionKey; // For Suma
+ Uint32 prepareId; // For DbUtil
+ Uint32 indexType;
+ Uint32 sourceTableId;
+ Uint32 targetTableId;
+ AttrOrderBuffer attributeOrder;
+ Uint32 noOfIndexColumns;
+ Uint32 noOfKeyColumns;
+ Uint32 parallelism;
+ BuildIndxRef::ErrorCode errorCode;
+ bool subscriptionCreated;
+ bool pendingSubSyncContinueConf;
+ Uint32 expectedConf; // Count in n UTIL_EXECUTE_CONF + 1 SUB_SYNC_CONF
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+ };
+
+ typedef Ptr<SubscriptionRecord> SubscriptionRecPtr;
+
+ /**
+ * The pool of node records
+ */
+ ArrayPool<SubscriptionRecord> c_theSubscriptionRecPool;
+
+ /**
+ * The list of other subscriptions
+ */
+ ArrayList<SubscriptionRecord> c_theSubscriptions;
+
+ // Linear memory abstraction
+#define TRIX_WORDS_PER_PAGE 8191
+ struct Page32 {
+ Uint32 data[TRIX_WORDS_PER_PAGE];
+ Uint32 nextPool;
+ };
+ typedef Ptr<Page32> Page32Ptr;
+
+ ArrayPool<Page32> c_thePagePool;
+ Array<Page32> c_thePages;
+
+ // Private methods
+
+ // System start
+ void execSTTOR(Signal* signal);
+ void execNDB_STTOR(Signal* signal);
+
+ // Node management
+ void execREAD_NODESCONF(Signal* signal);
+ void execREAD_NODESREF(Signal* signal);
+ void execNODE_FAILREP(Signal* signal);
+ void execINCL_NODEREQ(Signal* signal);
+ // Debugging
+ void execDUMP_STATE_ORD(Signal* signal);
+
+ // Build index
+ void execBUILDINDXREQ(Signal* signal);
+ void execBUILDINDXCONF(Signal* signal);
+ void execBUILDINDXREF(Signal* signal);
+
+ void execUTIL_PREPARE_CONF(Signal* signal);
+ void execUTIL_PREPARE_REF(Signal* signal);
+ void execUTIL_EXECUTE_CONF(Signal* signal);
+ void execUTIL_EXECUTE_REF(Signal* signal);
+ void execUTIL_RELEASE_CONF(Signal* signal);
+ void execUTIL_RELEASE_REF(Signal* signal);
+
+ // Suma signals
+ void execSUB_CREATE_CONF(Signal* signal);
+ void execSUB_CREATE_REF(Signal* signal);
+ void execSUB_REMOVE_CONF(Signal* signal);
+ void execSUB_REMOVE_REF(Signal* signal);
+ void execSUB_SYNC_CONF(Signal* signal);
+ void execSUB_SYNC_REF(Signal* signal);
+ void execSUB_SYNC_CONTINUE_REQ(Signal* signal);
+ void execSUB_META_DATA(Signal* signal);
+ void execSUB_TABLE_DATA(Signal* signal);
+
+ // Utility functions
+ void setupSubscription(Signal* signal, SubscriptionRecPtr subRecPtr);
+ void setupTableScan(Signal* signal, SubscriptionRecPtr subRecPtr);
+ void startTableScan(Signal* signal, SubscriptionRecPtr subRecPtr);
+ void prepareInsertTransactions(Signal* signal, SubscriptionRecPtr subRecPtr);
+ void executeInsertTransaction(Signal* signal, SubscriptionRecPtr subRecPtr,
+ SegmentedSectionPtr headerPtr,
+ SegmentedSectionPtr dataPtr);
+ void buildComplete(Signal* signal, SubscriptionRecPtr subRecPtr);
+ void buildFailed(Signal* signal,
+ SubscriptionRecPtr subRecPtr,
+ BuildIndxRef::ErrorCode);
+ void checkParallelism(Signal* signal, SubscriptionRecord* subRec);
+};
+
+#endif
diff --git a/ndb/src/kernel/error/Error.hpp b/ndb/src/kernel/error/Error.hpp
new file mode 100644
index 00000000000..d10c69b327a
--- /dev/null
+++ b/ndb/src/kernel/error/Error.hpp
@@ -0,0 +1,85 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ERROR_H
+#define ERROR_H
+
+/**
+ * Errorcodes for NDB
+ *
+ * These errorcodes should be used whenever a condition
+ * is detected where it's necesssary to shutdown NDB.
+ *
+ * Example: When another node fails while a NDB node are performing
+ * a system restart the node should be shutdown. This
+ * is kind of an error but the cause of the error is known
+ * and a proper errormessage describing the problem should
+ * be printed in error.log. It's therefore important to use
+ * the proper errorcode.
+ *
+ * TODO: In the future the errorcodes should be classified
+ *
+ */
+
+typedef enum ErrorCategory
+{
+ warning,
+ ecError,
+ fatal,
+ assert
+};
+
+const int ERR_BASE = 1000;
+
+// Errorcodes for all blocks except filseystem
+const int ERR_ERR_BASE = ERR_BASE + 1300;
+const int ERR_ERROR_PRGERR = ERR_ERR_BASE+1;
+const int ERR_NODE_NOT_IN_CONFIG = ERR_ERR_BASE+2;
+const int ERR_SYSTEM_ERROR = ERR_ERR_BASE+3;
+const int ERR_INDEX_NOTINRANGE = ERR_ERR_BASE+4;
+const int ERR_ARBIT_SHUTDOWN = ERR_ERR_BASE+5;
+const int ERR_POINTER_NOTINRANGE = ERR_ERR_BASE+6;
+const int ERR_PROGRAMERROR = ERR_ERR_BASE+7;
+const int ERR_SR_OTHERNODEFAILED = ERR_ERR_BASE+8;
+const int ERR_NODE_NOT_DEAD = ERR_ERR_BASE+9;
+const int ERR_SR_REDOLOG = ERR_ERR_BASE+10;
+const int ERR_SR_RESTARTCONFLICT = ERR_ERR_BASE+11;
+const int ERR_NO_MORE_UNDOLOG = ERR_ERR_BASE+12;
+const int ERR_SR_UNDOLOG = ERR_ERR_BASE+13;
+const int ERR_MEMALLOC = ERR_ERR_BASE+27;
+const int BLOCK_ERROR_JBUFCONGESTION = ERR_ERR_BASE+34;
+const int ERROR_TIME_QUEUE_SHORT = ERR_ERR_BASE+35;
+const int ERROR_TIME_QUEUE_LONG = ERR_ERR_BASE+36;
+const int ERROR_TIME_QUEUE_DELAY = ERR_ERR_BASE+37;
+const int ERROR_TIME_QUEUE_INDEX = ERR_ERR_BASE+38;
+const int BLOCK_ERROR_BNR_ZERO = ERR_ERR_BASE+39;
+const int ERROR_WRONG_PRIO_LEVEL = ERR_ERR_BASE+40;
+const int ERR_NDBREQUIRE = ERR_ERR_BASE+41;
+const int ERR_ERROR_INSERT = ERR_ERR_BASE+42;
+const int ERR_INVALID_CONFIG = ERR_ERR_BASE+50;
+const int ERR_OUT_OF_LONG_SIGNAL_MEMORY = ERR_ERR_BASE+51;
+
+// Errorcodes for NDB filesystem
+const int AFS_ERR_BASE = ERR_BASE + 1800;
+const int AFS_ERROR_NOPATH = AFS_ERR_BASE+1;
+const int AFS_ERROR_CHANNALFULL = AFS_ERR_BASE+2;
+const int AFS_ERROR_NOMORETHREADS = AFS_ERR_BASE+3;
+const int AFS_ERROR_PARAMETER = AFS_ERR_BASE+4;
+const int AFS_ERROR_INVALIDPATH = AFS_ERR_BASE+5;
+const int AFS_ERROR_MAXOPEN = AFS_ERR_BASE+6;
+const int AFS_ERROR_ALLREADY_OPEN = AFS_ERR_BASE+7;
+
+#endif // ERROR_H
diff --git a/ndb/src/kernel/error/ErrorHandlingMacros.hpp b/ndb/src/kernel/error/ErrorHandlingMacros.hpp
new file mode 100644
index 00000000000..416507fee23
--- /dev/null
+++ b/ndb/src/kernel/error/ErrorHandlingMacros.hpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ERRORHANDLINGMACROS_H
+#define ERRORHANDLINGMACROS_H
+
+#include "ErrorReporter.hpp"
+#include "Error.hpp"
+
+extern const char programName[];
+
+#define ERROR_SET(messageCategory, messageID, problemData, objectRef) \
+ ErrorReporter::handleError(messageCategory, messageID, problemData, objectRef)
+ // Description:
+ // Call ErrorHandler with the supplied arguments. The
+ // ErrorHandler decides how to report the error.
+ // Parameters:
+ // messageCategory IN A hint to the error handler how the
+ // error should be reported. Can be
+ // error, fatal (or warning, use WARNING_SET instead).
+ // messageID IN Code identifying the error. If less
+ // than 1000 a unix error is assumed. If
+ // greater than 1000 the code is treated
+ // as the specific problem code.
+ // problemData IN A (short) text describing the error.
+ // The context information is added to
+ // this text.
+ // objectRef IN The name of the "victim" of the error.
+ // Specify NULL if not applicable.
+ // Return value:
+ // -
+ // Reported errors:
+ // -
+ // Additional information:
+ // -
+
+#endif
diff --git a/ndb/src/kernel/error/ErrorMessages.cpp b/ndb/src/kernel/error/ErrorMessages.cpp
new file mode 100644
index 00000000000..059aa4af61c
--- /dev/null
+++ b/ndb/src/kernel/error/ErrorMessages.cpp
@@ -0,0 +1,75 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "ErrorMessages.hpp"
+
+struct ErrStruct {
+ int fauldId;
+ const char* text;
+};
+
+const ErrStruct errArray[] = {
+
+ {2301, "Assertion, probably a programming error"},
+ {2302, "Own Node Id not a NDB node, configuration error"},
+ {2303, "System error"},
+ {2304, "Index too large"},
+ {2305, "Arbitrator shutdown"},
+ {2306, "Pointer too large"},
+ {2307, "Internal program error"},
+ {2308, "Node failed during system restart"},
+ {2309, "Node state conflict"},
+ {2310, "Error while reading the REDO log"},
+ {2311, "Conflict when selecting restart type"},
+ {2312, "No more free UNDO log"},
+ {2313, "Error while reading the datapages and UNDO log"},
+ {2327, "Memory allocation failure"},
+ {2334, "Job buffer congestion"},
+ {2335, "Error in short time queue"},
+ {2336, "Error in long time queue"},
+ {2337, "Error in time queue, too long delay"},
+ {2338, "Time queue index out of range"},
+ {2339, "Send signal error"},
+ {2340, "Wrong prio level when sending signal"},
+ {2341, "Internal program error (failed ndbrequire)"},
+ {2342, "Error insert executed" },
+ {2350, "Invalid Configuration fetched from Management Server" },
+
+ // Ndbfs error messages
+ {2801, "No file system path"},
+ {2802, "Channel is full"},
+ {2803, "No more threads"},
+ {2804, "Bad parameter"},
+ {2805, "Illegal file system path"},
+ {2806, "Max number of open files exceeded"},
+ {2807, "File has already been opened"},
+
+ // Sentinel
+ {0, "No message slogan found"}
+
+};
+
+const unsigned short NO_OF_ERROR_MESSAGES = sizeof(errArray)/sizeof(ErrStruct);
+
+const char* lookupErrorMessage(int faultId)
+{
+ int i = 0;
+ while (errArray[i].fauldId != faultId && errArray[i].fauldId != 0)
+ i++;
+ return errArray[i].text;
+}
+
+
diff --git a/ndb/src/kernel/error/ErrorMessages.hpp b/ndb/src/kernel/error/ErrorMessages.hpp
new file mode 100644
index 00000000000..38c8eec636b
--- /dev/null
+++ b/ndb/src/kernel/error/ErrorMessages.hpp
@@ -0,0 +1,22 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ERROR_MESSAGES_H
+#define ERROR_MESSAGES_H
+
+const char* lookupErrorMessage(int faultId);
+
+#endif
diff --git a/ndb/src/kernel/error/ErrorReporter.cpp b/ndb/src/kernel/error/ErrorReporter.cpp
new file mode 100644
index 00000000000..1aa937f4675
--- /dev/null
+++ b/ndb/src/kernel/error/ErrorReporter.cpp
@@ -0,0 +1,397 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "Error.hpp"
+#include "ErrorReporter.hpp"
+#include "ErrorMessages.hpp"
+
+#include <FastScheduler.hpp>
+#include <DebuggerNames.hpp>
+#include <NdbUnistd.h>
+#include <NdbStdio.h>
+#include <NdbHost.h>
+#include <NdbConfig.h>
+#include <string.h>
+#include <Configuration.hpp>
+
+#define MESSAGE_LENGTH 400
+
+const char* errorType[] = {
+ "warning",
+ "error",
+ "fatal",
+ "assert"
+};
+
+
+static int WriteMessage(ErrorCategory thrdType, int thrdMessageID,
+ const char* thrdProblemData,
+ const char* thrdObjRef,
+ Uint32 thrdTheEmulatedJamIndex = 0,
+ Uint8 thrdTheEmulatedJam[] = 0);
+
+static void dumpJam(FILE* jamStream,
+ Uint32 thrdTheEmulatedJamIndex,
+ Uint8 thrdTheEmulatedJam[]);
+
+
+const char*
+ErrorReporter::formatTimeStampString(){
+ TimeModule DateTime; /* To create "theDateTimeString" */
+
+ static char theDateTimeString[39];
+ /* Used to store the generated timestamp */
+ /* ex: "Wednesday 18 September 2000 - 18:54:37" */
+
+ DateTime.setTimeStamp();
+
+ snprintf(theDateTimeString, 39, "%s %d %s %d - %s:%s:%s",
+ DateTime.getDayName(), DateTime.getDayOfMonth(),
+ DateTime.getMonthName(), DateTime.getYear(), DateTime.getHour(),
+ DateTime.getMinute(), DateTime.getSecond());
+
+ return (const char *)&theDateTimeString;
+}
+
+void
+ErrorReporter::formatTraceFileName(char* theName, int maxLen){
+
+ FILE *stream;
+ unsigned int traceFileNo;
+ char fileNameBuf[255];
+ char buf[255];
+
+ NdbConfig_HomePath(fileNameBuf, 255);
+ strncat(fileNameBuf, "NextTraceFileNo.log", 255);
+ /*
+ * Read last number from tracefile
+ */
+ stream = fopen(fileNameBuf, "r+");
+ if (stream == NULL){
+ traceFileNo = 1;
+ } else {
+ fgets(buf, 255, stream);
+ const int scan = sscanf(buf, "%u", &traceFileNo);
+ if(scan != 1){
+ traceFileNo = 1;
+ }
+ fclose(stream);
+ traceFileNo++;
+ }
+
+ /**
+ * Wrap tracefile no
+ */
+ Uint32 tmp = globalEmulatorData.theConfiguration->maxNoOfErrorLogs();
+ if (traceFileNo > tmp ) {
+ traceFileNo = 1;
+ }
+
+ /**
+ * Save new number to the file
+ */
+ stream = fopen(fileNameBuf, "w");
+ if(stream != NULL){
+ fprintf(stream, "%u", traceFileNo);
+ fclose(stream);
+ }
+ /**
+ * Format trace file name
+ */
+ snprintf(theName, maxLen, "%sNDB_TraceFile_%u.trace",
+ NdbConfig_HomePath(fileNameBuf, 255), traceFileNo);
+}
+
+
+void
+ErrorReporter::formatMessage(ErrorCategory type,
+ int faultID,
+ const char* problemData,
+ const char* objRef,
+ const char* theNameOfTheTraceFile,
+ char* messptr){
+ int processId;
+
+ processId = NdbHost_GetProcessId();
+
+ snprintf(messptr, MESSAGE_LENGTH,
+ "Date/Time: %s\nType of error: %s\n"
+ "Message: %s\nFault ID: %d\nProblem data: %s"
+ "\nObject of reference: %s\nProgramName: %s\n"
+ "ProcessID: %d\nTraceFile: %s\n***EOM***\n",
+ formatTimeStampString() ,
+ errorType[type],
+ lookupErrorMessage(faultID),
+ faultID,
+ (problemData == NULL) ? "" : problemData,
+ objRef,
+ programName,
+ processId,
+ theNameOfTheTraceFile);
+
+ // Add trailing blanks to get a fixed lenght of the message
+ while (strlen(messptr) <= MESSAGE_LENGTH-3){
+ strcat(messptr, " ");
+ }
+
+ strcat(messptr, "\n");
+
+ return;
+}
+
+void
+ErrorReporter::handleAssert(const char* message, const char* file, int line)
+{
+ char refMessage[100];
+
+#ifdef USE_EMULATED_JAM
+ const Uint32 blockNumber = theEmulatedJamBlockNumber;
+ const char *blockName = getBlockName(blockNumber);
+
+ snprintf(refMessage, 100, "%s line: %d (block: %s)",
+ file, line, blockName);
+
+ WriteMessage(assert, ERR_ERROR_PRGERR, message, refMessage,
+ theEmulatedJamIndex, theEmulatedJam);
+#else
+ snprintf(refMessage, 100, "file: %s lineNo: %d",
+ file, line);
+
+ WriteMessage(assert, ERR_ERROR_PRGERR, message, refMessage);
+#endif
+
+ NdbShutdown(NST_ErrorHandler);
+}
+
+void
+ErrorReporter::handleThreadAssert(const char* message,
+ const char* file,
+ int line)
+{
+ char refMessage[100];
+ snprintf(refMessage, 100, "file: %s lineNo: %d - %s",
+ file, line, message);
+
+ NdbShutdown(NST_ErrorHandler);
+}//ErrorReporter::handleThreadAssert()
+
+
+void
+ErrorReporter::handleError(ErrorCategory type, int messageID,
+ const char* problemData,
+ const char* objRef,
+ NdbShutdownType nst)
+{
+ type = ecError;
+ // The value for type is not always set correctly in the calling function.
+ // So, to correct this, we set it set it to the value corresponding to
+ // the function that is called.
+#ifdef USE_EMULATED_JAM
+ WriteMessage(type, messageID, problemData,
+ objRef, theEmulatedJamIndex, theEmulatedJam);
+#else
+ WriteMessage(type, messageID, problemData, objRef);
+#endif
+ if(messageID == ERR_ERROR_INSERT){
+ NdbShutdown(NST_ErrorInsert);
+ } else {
+ NdbShutdown(nst);
+ }
+}
+
+// This is the function to write the error-message,
+// when the USE_EMULATED_JAM-flag is set
+// during compilation.
+int
+WriteMessage(ErrorCategory thrdType, int thrdMessageID,
+ const char* thrdProblemData, const char* thrdObjRef,
+ Uint32 thrdTheEmulatedJamIndex,
+ Uint8 thrdTheEmulatedJam[]){
+ FILE *stream;
+ unsigned offset;
+ unsigned long maxOffset; // Maximum size of file.
+ char theMessage[MESSAGE_LENGTH];
+ char theTraceFileName[255];
+ char theErrorFileName[255];
+ ErrorReporter::formatTraceFileName(theTraceFileName, 255);
+
+ // The first 69 bytes is info about the current offset
+ Uint32 noMsg = globalEmulatorData.theConfiguration->maxNoOfErrorLogs();
+
+ maxOffset = (69 + (noMsg * MESSAGE_LENGTH));
+
+ NdbConfig_ErrorFileName(theErrorFileName, 255);
+ stream = fopen(theErrorFileName, "r+");
+ if (stream == NULL) { /* If the file could not be opened. */
+
+ // Create a new file, and skip the first 69 bytes,
+ // which are info about the current offset
+ stream = fopen(theErrorFileName, "w");
+ fprintf(stream, "%s%u%s", "Current byte-offset of file-pointer is: ", 69,
+ " \n\n\n");
+
+ // ...and write the error-message...
+ ErrorReporter::formatMessage(thrdType, thrdMessageID,
+ thrdProblemData, thrdObjRef,
+ theTraceFileName, theMessage);
+ fprintf(stream, "%s", theMessage);
+ fflush(stream);
+
+ /* ...and finally, at the beginning of the file,
+ store the position where to
+ start writing the next message. */
+ offset = ftell(stream);
+ // If we have not reached the maximum number of messages...
+ if (offset <= (maxOffset - MESSAGE_LENGTH)){
+ fseek(stream, 40, SEEK_SET);
+ // ...set the current offset...
+ fprintf(stream,"%d", offset);
+ } else {
+ fseek(stream, 40, SEEK_SET);
+ // ...otherwise, start over from the beginning.
+ fprintf(stream, "%u%s", 69, " ");
+ }
+ } else {
+ // Go to the latest position in the file...
+ fseek(stream, 40, SEEK_SET);
+ fscanf(stream, "%u", &offset);
+ fseek(stream, offset, SEEK_SET);
+
+ // ...and write the error-message there...
+ ErrorReporter::formatMessage(thrdType, thrdMessageID,
+ thrdProblemData, thrdObjRef,
+ theTraceFileName, theMessage);
+ fprintf(stream, "%s", theMessage);
+ fflush(stream);
+
+ /* ...and finally, at the beginning of the file,
+ store the position where to
+ start writing the next message. */
+ offset = ftell(stream);
+
+ // If we have not reached the maximum number of messages...
+ if (offset <= (maxOffset - MESSAGE_LENGTH)){
+ fseek(stream, 40, SEEK_SET);
+ // ...set the current offset...
+ fprintf(stream,"%d", offset);
+ } else {
+ fseek(stream, 40, SEEK_SET);
+ // ...otherwise, start over from the beginning.
+ fprintf(stream, "%u%s", 69, " ");
+ }
+ }
+ fflush(stream);
+ fclose(stream);
+
+ // Open the tracefile...
+ FILE *jamStream = fopen(theTraceFileName, "w");
+
+ // ...and "dump the jam" there.
+ // ErrorReporter::dumpJam(jamStream);
+ if(thrdTheEmulatedJam != 0){
+#ifdef USE_EMULATED_JAM
+ dumpJam(jamStream, thrdTheEmulatedJamIndex, thrdTheEmulatedJam);
+#endif
+ }
+
+ /* Dont print the jobBuffers until a way to copy them,
+ like the other variables,
+ is implemented. Otherwise when NDB keeps running,
+ with this function running
+ in the background, the jobBuffers will change during runtime. And when
+ they're printed here, they will not be correct anymore.
+ */
+ globalScheduler.dumpSignalMemory(jamStream);
+
+ fclose(jamStream);
+
+ return 0;
+}
+
+void
+dumpJam(FILE *jamStream,
+ Uint32 thrdTheEmulatedJamIndex,
+ Uint8 thrdTheEmulatedJam[]) {
+#ifdef USE_EMULATED_JAM
+ // print header
+ const int maxaddr = 8;
+ fprintf(jamStream, "JAM CONTENTS up->down left->right ?=not block entry\n");
+ fprintf(jamStream, "%-7s ", "BLOCK");
+ for (int i = 0; i < maxaddr; i++)
+ fprintf(jamStream, "%-6s ", "ADDR");
+ fprintf(jamStream, "\n");
+
+ // treat as array of Uint32
+ const Uint32 *base = (Uint32 *)thrdTheEmulatedJam;
+ const int first = thrdTheEmulatedJamIndex / sizeof(Uint32); // oldest
+ int cnt, idx;
+
+ // look for first block entry
+ for (cnt = 0, idx = first; cnt < EMULATED_JAM_SIZE; cnt++, idx++) {
+ if (idx >= EMULATED_JAM_SIZE)
+ idx = 0;
+ const Uint32 aJamEntry = base[idx];
+ if (aJamEntry > (1 << 20))
+ break;
+ }
+
+ // 1. if first entry is a block entry, it is printed in the main loop
+ // 2. else if any block entry exists, the jam starts in an unknown block
+ // 3. else if no block entry exists, the block is theEmulatedJamBlockNumber
+ // a "?" indicates first addr is not a block entry
+ if (cnt == 0)
+ ;
+ else if (cnt < EMULATED_JAM_SIZE)
+ fprintf(jamStream, "%-7s?", "");
+ else {
+ const Uint32 aBlockNumber = theEmulatedJamBlockNumber;
+ const char *aBlockName = getBlockName(aBlockNumber);
+ if (aBlockName != 0)
+ fprintf(jamStream, "%-7s?", aBlockName);
+ else
+ fprintf(jamStream, "0x%-5X?", aBlockNumber);
+ }
+
+ // loop over all entries
+ int cntaddr = 0;
+ for (cnt = 0, idx = first; cnt < EMULATED_JAM_SIZE; cnt++, idx++) {
+ globalData.incrementWatchDogCounter(4); // watchdog not to kill us ?
+ if (idx >= EMULATED_JAM_SIZE)
+ idx = 0;
+ const Uint32 aJamEntry = base[idx];
+ if (aJamEntry > (1 << 20)) {
+ const Uint32 aBlockNumber = aJamEntry >> 20;
+ const char *aBlockName = getBlockName(aBlockNumber);
+ if (cnt > 0)
+ fprintf(jamStream, "\n");
+ if (aBlockName != 0)
+ fprintf(jamStream, "%-7s ", aBlockName);
+ else
+ fprintf(jamStream, "0x%-5X ", aBlockNumber);
+ cntaddr = 0;
+ }
+ if (cntaddr == maxaddr) {
+ fprintf(jamStream, "\n%-7s ", "");
+ cntaddr = 0;
+ }
+ fprintf(jamStream, "%06u ", aJamEntry & 0xFFFFF);
+ cntaddr++;
+ }
+ fprintf(jamStream, "\n");
+ fflush(jamStream);
+#endif // USE_EMULATED_JAM
+}
diff --git a/ndb/src/kernel/error/ErrorReporter.hpp b/ndb/src/kernel/error/ErrorReporter.hpp
new file mode 100644
index 00000000000..f1428821ab0
--- /dev/null
+++ b/ndb/src/kernel/error/ErrorReporter.hpp
@@ -0,0 +1,98 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ERRORREPORTER_H
+#define ERRORREPORTER_H
+
+#include "TimeModule.hpp"
+#include "Error.hpp"
+#include <stdio.h>
+#include <Emulator.hpp>
+
+
+#ifdef ASSERT
+#undef ASSERT
+#endif
+
+#define REQUIRE(trueToContinue, message) \
+ if ( (trueToContinue) ) { } else { \
+ ErrorReporter::handleAssert(message, __FILE__, __LINE__); }
+
+#define THREAD_REQUIRE(trueToContinue, message) \
+ if ( (trueToContinue) ) { } else { \
+ ErrorReporter::handleThreadAssert(message, __FILE__, __LINE__); }
+
+#ifdef NDEBUG
+
+#define NDB_ASSERT(trueToContinue, message)
+#define THREAD_ASSERT(trueToContinue, message)
+
+#else
+
+#define NDB_ASSERT(trueToContinue, message) \
+ if ( !(trueToContinue) ) { \
+ ErrorReporter::handleAssert(message, __FILE__, __LINE__); }
+
+#define THREAD_ASSERT(trueToContinue, message) \
+ if ( !(trueToContinue) ) { \
+ ErrorReporter::handleThreadAssert(message, __FILE__, __LINE__); }
+
+#endif
+ // Description:
+ // This macro is used to report programming errors.
+ // Parameters:
+ // trueToContinue IN An expression. If it evaluates to 0
+ // execution is stopped.
+ // message IN A message from the programmer
+ // explaining what went wrong.
+
+class ErrorReporter
+{
+public:
+ static void handleAssert(const char* message,
+ const char* file,
+ int line);
+
+ static void handleThreadAssert(const char* message,
+ const char* file,
+ int line);
+
+ static void handleError(ErrorCategory type,
+ int faultID,
+ const char* problemData,
+ const char* objRef,
+ enum NdbShutdownType = NST_ErrorHandler);
+
+ static void handleWarning(ErrorCategory type,
+ int faultID,
+ const char* problemData,
+ const char* objRef);
+
+ static void formatMessage(ErrorCategory type,
+ int faultID,
+ const char* problemData,
+ const char* objRef,
+ const char* theNameOfTheTraceFile,
+ char* messptr);
+
+ static void formatTraceFileName(char* theName, int maxLen);
+
+ static const char* formatTimeStampString();
+
+private:
+};
+
+#endif
diff --git a/ndb/src/kernel/error/Makefile b/ndb/src/kernel/error/Makefile
new file mode 100644
index 00000000000..0fe81f083ce
--- /dev/null
+++ b/ndb/src/kernel/error/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := error
+
+SOURCES = \
+ TimeModule.cpp \
+ ErrorReporter.cpp \
+ ErrorMessages.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/error/TimeModule.cpp b/ndb/src/kernel/error/TimeModule.cpp
new file mode 100644
index 00000000000..c0f4e40858f
--- /dev/null
+++ b/ndb/src/kernel/error/TimeModule.cpp
@@ -0,0 +1,109 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include "TimeModule.hpp"
+#include <time.h>
+
+static const char* cMonth[] = { "x", "January", "February", "Mars", "April", "May", "June",
+ "July", "August", "September", "October", "November", "December"};
+
+static const char* cDay[] = { "x", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday",
+ "Saturday", "Sunday"};
+
+static const char* cHour[] = { "00","01","02","03","04","05","06","07","08","09","10","11","12",
+ "13","14","15","16","17","18","19","20","21","22","23"};
+
+static const char* cMinute[] = { "00","01","02","03","04","05","06","07","08","09","10","11","12",
+ "13","14","15","16","17","18","19","20","21","22","23","24","25",
+ "26","27","28","29","30","31","32","33","34","35","36","37","38",
+ "39","40","41","42","43","44","45","46","47","48","49","50","51",
+ "52","53","54","55","56","57","58","59"};
+
+static const char* cSecond[] = { "00","01","02","03","04","05","06","07","08","09","10","11","12",
+ "13","14","15","16","17","18","19","20","21","22","23","24","25",
+ "26","27","28","29","30","31","32","33","34","35","36","37","38",
+ "39","40","41","42","43","44","45","46","47","48","49","50","51",
+ "52","53","54","55","56","57","58","59"};
+
+
+TimeModule::TimeModule(){
+}
+
+TimeModule::~TimeModule(){
+}
+
+void
+TimeModule::setTimeStamp()
+{
+ struct tm* rightnow;
+ time_t now;
+
+ time(&now);
+
+ rightnow = localtime(&now);
+
+ iYear = rightnow->tm_year+1900; // localtime returns current year -1900
+ iMonth = rightnow->tm_mon+1; // and month 0-11
+ iMonthDay = rightnow->tm_mday;
+ iWeekDay = rightnow->tm_wday;
+ iHour = rightnow->tm_hour;
+ iMinute = rightnow->tm_min;
+ iSecond = rightnow->tm_sec;
+}
+
+int
+TimeModule::getYear() const
+{
+ return iYear;
+}
+
+int
+TimeModule::getMonthNumber() const
+{
+ return iMonth;
+}
+
+const char*
+TimeModule::getMonthName() const {
+ return cMonth[iMonth];
+}
+
+int
+TimeModule::getDayOfMonth() const {
+ return iMonthDay;
+}
+
+const char*
+TimeModule::getDayName() const {
+ return cDay[iWeekDay];
+}
+
+const char*
+TimeModule::getHour() const {
+ return cHour[iHour];
+}
+
+const char*
+TimeModule::getMinute() const {
+ return cMinute[iMinute];
+}
+
+const char*
+TimeModule::getSecond() const {
+ return cSecond[iSecond];
+}
diff --git a/ndb/src/kernel/error/TimeModule.hpp b/ndb/src/kernel/error/TimeModule.hpp
new file mode 100644
index 00000000000..f1414c77af3
--- /dev/null
+++ b/ndb/src/kernel/error/TimeModule.hpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef _TimeModule_
+#define _TimeModule_
+
+class TimeModule {
+public:
+ TimeModule();
+ ~TimeModule();
+
+ void setTimeStamp();
+
+ int getYear() const;
+ int getMonthNumber() const;
+ int getDayOfMonth() const;
+ const char* getMonthName() const;
+ const char* getDayName() const;
+ const char* getHour() const;
+ const char* getMinute() const;
+ const char* getSecond() const;
+
+private:
+ int iYear;
+ int iMonth;
+ int iMonthDay;
+ int iWeekDay;
+ int iHour;
+ int iMinute;
+ int iSecond;
+};
+
+#endif
diff --git a/ndb/src/kernel/ndb-main/Main.cpp b/ndb/src/kernel/ndb-main/Main.cpp
new file mode 100644
index 00000000000..ca3bfa32a59
--- /dev/null
+++ b/ndb/src/kernel/ndb-main/Main.cpp
@@ -0,0 +1,340 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <ndb_version.h>
+#include "Configuration.hpp"
+#include <TransporterRegistry.hpp>
+
+#include "SimBlockList.hpp"
+#include "ThreadConfig.hpp"
+#include <SignalLoggerManager.hpp>
+#include <NdbOut.hpp>
+#include <NdbMain.h>
+#include <NdbDaemon.h>
+#include <NdbConfig.h>
+#include <WatchDog.hpp>
+
+#include <LogLevel.hpp>
+#include <NodeState.hpp>
+
+#if defined NDB_SOLARIS
+#include <sys/types.h> // For system information
+#include <sys/processor.h> // For system informatio
+#endif
+
+#if !defined NDB_SOFTOSE && !defined NDB_OSE
+#include <signal.h> // For process signals
+
+extern "C" {
+ void ndbSignal(int signo, void (*func) (int));
+ void handler(int signo); // for process signal handling
+};
+
+void catchsigs(); // for process signal handling
+
+#endif
+
+// Shows system information
+void systemInfo(const Configuration & conf,
+ const LogLevel & ll);
+
+const char programName[] = "NDB Kernel";
+
+extern int global_ndb_check;
+NDB_MAIN(ndb_kernel){
+
+ global_ndb_check = 1;
+
+ globalEmulatorData.create();
+
+ // Parse command line options
+ Configuration* theConfig = globalEmulatorData.theConfiguration;
+ if(!theConfig->init(argc, argv)){
+ return 0;
+ }
+
+ { // Do configuration
+ theConfig->setupConfiguration();
+ }
+
+ // Get NDB_HOME path
+ char homePath[255];
+ NdbConfig_HomePath(homePath, 255);
+
+#if defined (NDB_LINUX) || defined (NDB_SOLARIS)
+ if (theConfig->getDaemonMode()) {
+ // Become a daemon
+ char lockfile[255], logfile[255];
+ snprintf(lockfile, 255, "%snode%d.pid", homePath, globalData.ownId);
+ snprintf(logfile, 255, "%snode%d.out", homePath, globalData.ownId);
+ if (NdbDaemon_Make(lockfile, logfile, 0) == -1) {
+ ndbout << "Cannot become daemon: " << NdbDaemon_ErrorText << endl;
+ return 1;
+ }
+ }
+#endif
+
+ systemInfo(* theConfig,
+ theConfig->clusterConfigurationData().SizeAltData.logLevel);
+
+ // Load blocks
+ globalEmulatorData.theSimBlockList->load(* theConfig);
+
+ // Set thread concurrency for Solaris' light weight processes
+ int status;
+ status = NdbThread_SetConcurrencyLevel(30);
+ NDB_ASSERT(status == 0, "Can't set appropriate concurrency level.");
+
+#ifdef VM_TRACE
+ // Create a signal logger
+ char buf[255];
+ strcpy(buf, homePath);
+ FILE * signalLog = fopen(strncat(buf,"Signal.log", 255), "a");
+ globalSignalLoggers.setOutputStream(signalLog);
+#endif
+
+#if !defined NDB_SOFTOSE && !defined NDB_OSE
+ catchsigs();
+#endif
+
+ /**
+ * Do startup
+ */
+ switch(globalData.theRestartFlag){
+ case initial_state:
+ globalEmulatorData.theThreadConfig->doStart(NodeState::SL_CMVMI);
+ break;
+ case perform_start:
+ globalEmulatorData.theThreadConfig->doStart(NodeState::SL_CMVMI);
+ globalEmulatorData.theThreadConfig->doStart(NodeState::SL_STARTING);
+ break;
+ default:
+ NDB_ASSERT(0, "Illegal state globalData.theRestartFlag");
+ }
+
+ globalTransporterRegistry.startSending();
+ globalTransporterRegistry.startReceiving();
+ globalEmulatorData.theWatchDog->doStart();
+
+ globalEmulatorData.theThreadConfig->ipControlLoop();
+
+ NdbShutdown(NST_Normal);
+ return 0;
+}
+
+
+void
+systemInfo(const Configuration & config, const LogLevel & logLevel){
+ int processors = 0;
+ int speed;
+#ifdef NDB_WIN32
+ SYSTEM_INFO sinfo;
+ GetSystemInfo(&sinfo);
+ processors = sinfo.dwNumberOfProcessors;
+ HKEY hKey;
+ if(ERROR_SUCCESS==RegOpenKeyEx
+ (HKEY_LOCAL_MACHINE,
+ TEXT("HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\0"),
+ 0, KEY_READ, &hKey)) {
+ DWORD dwMHz;
+ DWORD cbData = sizeof(dwMHz);
+ if(ERROR_SUCCESS==RegQueryValueEx(hKey,
+ "~MHz", 0, 0, (LPBYTE)&dwMHz, &cbData)) {
+ speed = int(dwMHz);
+ }
+ RegCloseKey(hKey);
+ }
+#elif defined NDB_SOLARIS
+ // Search for at max 16 processors among the first 256 processor ids
+ processor_info_t pinfo; memset(&pinfo, 0, sizeof(pinfo));
+ int pid = 0;
+ while(processors < 16 && pid < 256){
+ if(!processor_info(pid++, &pinfo))
+ processors++;
+ }
+ speed = pinfo.pi_clock;
+#endif
+
+ if(logLevel.getLogLevel(LogLevel::llStartUp) > 0){
+ ndbout << "-- NDB Cluster -- DB node " << globalData.ownId
+ << " -- " << NDB_VERSION_STRING << " -- " << endl;
+#ifdef NDB_SOLARIS
+ ndbout << "NDB is running "
+ << " on a machine with " << processors
+ << " processor(s) at " << speed <<" MHz"
+ << endl;
+#endif
+ }
+ if(logLevel.getLogLevel(LogLevel::llStartUp) > 3){
+ Uint32 t = config.timeBetweenWatchDogCheck();
+ ndbout << "WatchDog timer is set to " << t << " ms" << endl;
+ }
+
+}
+
+#if !defined NDB_SOFTOSE && !defined NDB_OSE
+
+#ifdef NDB_WIN32
+
+void
+catchsigs()
+{
+ ndbSignal(SIGINT, handler); // 2
+ ndbSignal(SIGILL, handler); // 4
+ ndbSignal(SIGFPE, handler); // 8
+#ifndef VM_TRACE
+ ndbSignal(SIGSEGV, handler); // 11
+#endif
+ ndbSignal(SIGTERM, handler); // 15
+ ndbSignal(SIGBREAK, handler); // 21
+ ndbSignal(SIGABRT, handler); // 22
+}
+
+#else
+
+void
+catchsigs(){
+ // Makes the main process catch process signals, eg installs a
+ // handler named "handler". "handler" will then be called is instead
+ // of the defualt process signal handler)
+ ndbSignal(SIGHUP, handler); // 1
+ ndbSignal(SIGINT, handler); // 2
+ ndbSignal(SIGQUIT, handler); // 3
+ ndbSignal(SIGILL, handler); // 4
+ ndbSignal(SIGTRAP, handler); // 5
+#ifdef NDB_LINUX
+ ndbSignal(7, handler);
+#elif NDB_SOLARIS
+ ndbSignal(SIGEMT, handler); // 7
+#elif NDB_MACOSX
+ ndbSignal(SIGEMT, handler); // 7
+#endif
+ ndbSignal(SIGFPE, handler); // 8
+ // SIGKILL cannot be caught, 9
+ ndbSignal(SIGBUS, handler); // 10
+ ndbSignal(SIGSEGV, handler); // 11
+ ndbSignal(SIGSYS, handler); // 12
+ ndbSignal(SIGPIPE, handler); // 13
+ ndbSignal(SIGALRM, handler); // 14
+ ndbSignal(SIGTERM, handler); // 15
+ ndbSignal(SIGUSR1, handler); // 16
+ ndbSignal(SIGUSR2, handler); // 17
+#ifndef NDB_MACOSX
+ ndbSignal(SIGPWR, handler); // 19
+ ndbSignal(SIGPOLL, handler); // 22
+#endif
+ // SIGSTOP cannot be caught 23
+ ndbSignal(SIGTSTP, handler); // 24
+ ndbSignal(SIGTTIN, handler); // 26
+ ndbSignal(SIGTTOU, handler); // 27
+ ndbSignal(SIGVTALRM, handler); // 28
+ ndbSignal(SIGPROF, handler); // 29
+ ndbSignal(SIGXCPU, handler); // 30
+ ndbSignal(SIGXFSZ, handler); // 31
+}
+#endif
+
+extern "C"
+void ndbSignal(int signo, void (*func) (int)) {
+#ifdef NDB_WIN32
+ signal(signo, func);
+#else
+ struct sigaction act, oact;
+ act.sa_handler = func;
+ sigemptyset(&act.sa_mask);
+ act.sa_flags = 0;
+ if(signo == SIGALRM) {
+#ifdef SA_INTERRUPT
+ act.sa_flags |= SA_INTERRUPT;
+#endif
+ } else {
+#ifdef SA_RESTART
+ act.sa_flags |= SA_RESTART;
+#endif
+ }
+ sigaction(signo, &act, &oact);
+#endif
+}
+
+
+#ifdef NDB_WIN32
+
+extern "C"
+void
+handler(int sig)
+{
+ switch(sig){
+ case SIGINT: /* 2 - Interrupt */
+ case SIGTERM: /* 15 - Terminate */
+ case SIGBREAK: /* 21 - Ctrl-Break sequence */
+ case SIGABRT: /* 22 - abnormal termination triggered by abort call */
+ globalData.theRestartFlag = perform_stop;
+ break;
+ default:
+ // restart the system
+ char errorData[40];
+ snprintf(errorData, 40, "Signal %d received", sig);
+ ERROR_SET(fatal, 0, errorData, __FILE__);
+ break;
+ }
+}
+
+#else
+
+extern "C"
+void
+handler(int sig){
+ switch(sig){
+ case SIGHUP: /* 1 - Hang up */
+ case SIGINT: /* 2 - Interrupt */
+ case SIGQUIT: /* 3 - Quit */
+ case SIGTERM: /* 15 - Terminate */
+#ifndef NDB_MACOSX
+ case SIGPWR: /* 19 - Power fail */
+ case SIGPOLL: /* 22 */
+#endif
+ case SIGSTOP: /* 23 */
+ case SIGTSTP: /* 24 */
+ case SIGTTIN: /* 26 */
+ case SIGTTOU: /* 27 */
+ globalData.theRestartFlag = perform_stop;
+ break;
+ case SIGPIPE:
+ /**
+ * Can happen in TCP Transporter
+ *
+ * Just ignore
+ */
+ break;
+ default:
+ // restart the system
+ char errorData[40];
+ snprintf(errorData, 40, "Signal %d received", sig);
+ ERROR_SET(fatal, 0, errorData, __FILE__);
+ break;
+ }
+}
+
+#endif
+#endif
+
+
+
+
+
+
+
+
diff --git a/ndb/src/kernel/ndb-main/Makefile b/ndb/src/kernel/ndb-main/Makefile
new file mode 100644
index 00000000000..29b7ea7e708
--- /dev/null
+++ b/ndb/src/kernel/ndb-main/Makefile
@@ -0,0 +1,42 @@
+include .defs.mk
+
+TYPE := kernel
+
+BIN_TARGET := ndb
+BIN_TARGET_ARCHIVES := mgmapi \
+ cmvmi dbacc dbdict dbdih dblqh dbtc \
+ dbtup ndbfs ndbcntr qmgr trix backup dbutil suma grep dbtux \
+ transporter \
+ kernel \
+ error \
+ trace \
+ signaldataprint \
+ mgmsrvcommon \
+ portlib \
+ logger \
+ general
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = \
+ Main.cpp \
+ SimBlockList.cpp
+
+CCFLAGS_LOC = -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/cmvmi) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/dbacc) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/dbdict) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/dbdih) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/dblqh) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/dbtc) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/dbtup) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/ndbfs) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/missra) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/ndbcntr) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/qmgr) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/trix) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/backup) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/dbutil) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/suma) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/grep) \
+ -I$(call fixpath,$(NDB_TOP)/src/kernel/blocks/dbtux)
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/ndb-main/SimBlockList.cpp b/ndb/src/kernel/ndb-main/SimBlockList.cpp
new file mode 100644
index 00000000000..9e1d28a7fce
--- /dev/null
+++ b/ndb/src/kernel/ndb-main/SimBlockList.cpp
@@ -0,0 +1,111 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "SimBlockList.hpp"
+#include <SimulatedBlock.hpp>
+#include <Cmvmi.hpp>
+#include <Ndbfs.hpp>
+#include <Dbacc.hpp>
+#include <Dbdict.hpp>
+#include <Dbdih.hpp>
+#include <Dblqh.hpp>
+#include <Dbtc.hpp>
+#include <Dbtup.hpp>
+#include <Ndbcntr.hpp>
+#include <Qmgr.hpp>
+#include <Trix.hpp>
+#include <Backup.hpp>
+#include <DbUtil.hpp>
+#include <Suma.hpp>
+#include <Grep.hpp>
+#include <Dbtux.hpp>
+
+enum SIMBLOCKLIST_DUMMY { A_VALUE = 0 };
+
+static
+void * operator new (size_t sz, SIMBLOCKLIST_DUMMY dummy){
+ char * tmp = (char *)malloc(sz);
+
+#ifndef NDB_PURIFY
+#ifdef VM_TRACE
+ const int initValue = 0xf3;
+#else
+ const int initValue = 0x0;
+#endif
+
+ const int p = (sz / 4096);
+ const int r = (sz % 4096);
+
+ for(int i = 0; i<p; i++)
+ memset(tmp+(i*4096), initValue, 4096);
+
+ if(r > 0)
+ memset(tmp+p*4096, initValue, r);
+
+#endif
+
+ return tmp;
+}
+
+void
+SimBlockList::load(const Configuration & conf){
+ noOfBlocks = 16;
+ theList = new SimulatedBlock * [noOfBlocks];
+ for(int i = 0; i<noOfBlocks; i++)
+ theList[i] = 0;
+ Dbdict* dbdict = 0;
+ Dbdih* dbdih = 0;
+
+ theList[0] = new (A_VALUE) Dbacc(conf);
+ theList[1] = new (A_VALUE) Cmvmi(conf);
+ theList[2] = new (A_VALUE) Ndbfs(conf);
+ theList[3] = dbdict = new (A_VALUE) Dbdict(conf);
+ theList[4] = dbdih = new (A_VALUE) Dbdih(conf);
+ theList[5] = new (A_VALUE) Dblqh(conf);
+ theList[6] = new (A_VALUE) Dbtc(conf);
+ theList[7] = new (A_VALUE) Dbtup(conf);
+ theList[8] = new (A_VALUE) Ndbcntr(conf);
+ theList[9] = new (A_VALUE) Qmgr(conf);
+ theList[10] = new (A_VALUE) Trix(conf);
+ theList[11] = new (A_VALUE) Backup(conf);
+ theList[12] = new (A_VALUE) DbUtil(conf);
+ theList[13] = new (A_VALUE) Suma(conf);
+ theList[14] = new (A_VALUE) Grep(conf);
+ theList[15] = new (A_VALUE) Dbtux(conf);
+
+ // Metadata common part shared by block instances
+ ptrMetaDataCommon = new MetaData::Common(*dbdict, *dbdih);
+ for (int i = 0; i < noOfBlocks; i++)
+ theList[i]->setMetaDataCommon(ptrMetaDataCommon);
+}
+
+void
+SimBlockList::unload(){
+ if(theList != 0){
+ for(int i = 0; i<noOfBlocks; i++){
+ if(theList[i] != 0){
+ theList[i]->~SimulatedBlock();
+ free(theList[i]);
+ theList[i] = 0;
+ }
+ }
+ delete [] theList;
+ delete ptrMetaDataCommon;
+ theList = 0;
+ noOfBlocks = 0;
+ ptrMetaDataCommon = 0;
+ }
+}
diff --git a/ndb/src/kernel/vm/Array.hpp b/ndb/src/kernel/vm/Array.hpp
new file mode 100644
index 00000000000..97b0a345cb4
--- /dev/null
+++ b/ndb/src/kernel/vm/Array.hpp
@@ -0,0 +1,165 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ARRAY_HPP
+#define ARRAY_HPP
+
+#include "ArrayPool.hpp"
+
+#include <pc.hpp>
+#include <ErrorReporter.hpp>
+
+/**
+ * Template class used for implementing an
+ * array of object retreived from a pool
+ */
+template <class T>
+class Array {
+public:
+ Array(ArrayPool<T> & thePool);
+
+ /**
+ * Allocate an <b>n</b> objects from pool
+ * These can now be addressed with 0 <= ptr.i < n
+ */
+ bool seize(Uint32 i);
+
+ /**
+ * Release all object from array
+ */
+ void release();
+
+ /**
+ * Return current size of array
+ */
+ Uint32 getSize() const;
+
+ /**
+ * empty
+ */
+ inline bool empty() const { return sz == 0;}
+
+ /**
+ * Update i & p value according to <b>i</b>
+ */
+ void getPtr(Ptr<T> &, Uint32 i) const;
+
+ /**
+ * Update p value for ptr according to i value
+ */
+ void getPtr(Ptr<T> &) const ;
+
+ /**
+ * Get pointer for i value
+ */
+ T * getPtr(Uint32 i) const;
+
+private:
+ Uint32 base, sz;
+ ArrayPool<T> & thePool;
+};
+
+template<class T>
+inline
+Array<T>::Array(ArrayPool<T> & _pool)
+ : thePool(_pool)
+{
+ sz = 0;
+ base = RNIL;
+}
+
+template<class T>
+inline
+bool
+Array<T>::seize(Uint32 n){
+ if(base == RNIL && n > 0){
+ base = thePool.seizeN(n);
+ if(base != RNIL){
+ sz = n;
+ return true;
+ }
+ return false;
+ }
+ ErrorReporter::handleAssert("Array<T>::seize failed", __FILE__, __LINE__);
+ return false;
+}
+
+template<class T>
+inline
+void
+Array<T>::release(){
+ if(base != RNIL){
+ thePool.releaseN(base, sz);
+ sz = 0;
+ base = RNIL;
+ return;
+ }
+}
+
+template<class T>
+inline
+Uint32
+Array<T>::getSize() const {
+ return sz;
+}
+
+template <class T>
+inline
+void
+Array<T>::getPtr(Ptr<T> & p, Uint32 i) const {
+ p.i = i;
+#ifdef ARRAY_GUARD
+ if(i < sz && base != RNIL){
+ p.p = thePool.getPtr(i + base);
+ return;
+ } else {
+ ErrorReporter::handleAssert("Array::getPtr failed", __FILE__, __LINE__);
+ }
+#endif
+ p.p = thePool.getPtr(i + base);
+}
+
+template<class T>
+inline
+void
+Array<T>::getPtr(Ptr<T> & ptr) const {
+#ifdef ARRAY_GUARD
+ if(ptr.i < sz && base != RNIL){
+ ptr.p = thePool.getPtr(ptr.i + base);
+ return;
+ } else {
+ ErrorReporter::handleAssert("Array<T>::getPtr failed", __FILE__, __LINE__);
+ }
+#endif
+ ptr.p = thePool.getPtr(ptr.i + base);
+}
+
+template<class T>
+inline
+T *
+Array<T>::getPtr(Uint32 i) const {
+#ifdef ARRAY_GUARD
+ if(i < sz && base != RNIL){
+ return thePool.getPtr(i + base);
+ } else {
+ ErrorReporter::handleAssert("Array<T>::getPtr failed", __FILE__, __LINE__);
+ }
+#endif
+ return thePool.getPtr(i + base);
+}
+
+
+#endif
diff --git a/ndb/src/kernel/vm/ArrayFifoList.hpp b/ndb/src/kernel/vm/ArrayFifoList.hpp
new file mode 100644
index 00000000000..b21bf449734
--- /dev/null
+++ b/ndb/src/kernel/vm/ArrayFifoList.hpp
@@ -0,0 +1,30 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ARRAY_FIFOLIST_HPP
+#define ARRAY_FIFOLIST_HPP
+
+#include "ArrayPool.hpp"
+#include "DLFifoList.hpp"
+#include "Array.hpp"
+
+template <class T>
+class ArrayFifoList : public DLFifoList<T> {
+public:
+ ArrayFifoList(ArrayPool<T> & thePool) : DLFifoList<T>(thePool) { }
+};
+
+#endif
diff --git a/ndb/src/kernel/vm/ArrayList.hpp b/ndb/src/kernel/vm/ArrayList.hpp
new file mode 100644
index 00000000000..4b46347a39b
--- /dev/null
+++ b/ndb/src/kernel/vm/ArrayList.hpp
@@ -0,0 +1,30 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ARRAY_LIST_HPP
+#define ARRAY_LIST_HPP
+
+#include "ArrayPool.hpp"
+#include "DLList.hpp"
+#include "Array.hpp"
+
+template <class T>
+class ArrayList : public DLList<T> {
+public:
+ ArrayList(ArrayPool<T> & thePool) : DLList<T>(thePool) { }
+};
+
+#endif
diff --git a/ndb/src/kernel/vm/ArrayPool.hpp b/ndb/src/kernel/vm/ArrayPool.hpp
new file mode 100644
index 00000000000..4a84047b614
--- /dev/null
+++ b/ndb/src/kernel/vm/ArrayPool.hpp
@@ -0,0 +1,874 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ARRAY_POOL_HPP
+#define ARRAY_POOL_HPP
+
+#include <pc.hpp>
+#include <ErrorReporter.hpp>
+#include <NdbMem.h>
+#include <Bitmask.hpp>
+#include <assert.h>
+#include <stddef.h>
+
+template <class T> class Array;
+template <class T> class SLList;
+template <class T> class DLList;
+template <class T> class DLHashTable;
+
+/**
+ * Template class used for implementing an
+ * pool of object (in an array with a free list)
+ */
+template <class T>
+class ArrayPool {
+public:
+ ArrayPool();
+ ~ArrayPool();
+
+ /**
+ * Set the size of the pool
+ *
+ * Note, can currently only be called once
+ */
+ bool setSize(Uint32 noOfElements);
+
+ inline Uint32 getNoOfFree() const {
+ return noOfFree;
+ }
+
+ inline Uint32 getSize() const {
+ return size;
+ }
+
+ /**
+ * Update p value for ptr according to i value
+ */
+ void getPtr(Ptr<T> &);
+ void getPtr(ConstPtr<T> &) const;
+ void getPtr(Ptr<T> &, bool CrashOnBoundaryError);
+ void getPtr(ConstPtr<T> &, bool CrashOnBoundaryError) const;
+
+ /**
+ * Get pointer for i value
+ */
+ T * getPtr(Uint32 i);
+ const T * getConstPtr(Uint32 i) const;
+ T * getPtr(Uint32 i, bool CrashOnBoundaryError);
+ const T * getConstPtr(Uint32 i, bool CrashOnBoundaryError) const;
+
+ /**
+ * Update p & i value for ptr according to <b>i</b> value
+ */
+ void getPtr(Ptr<T> &, Uint32 i);
+ void getPtr(ConstPtr<T> &, Uint32 i) const;
+ void getPtr(Ptr<T> &, Uint32 i, bool CrashOnBoundaryError);
+ void getPtr(ConstPtr<T> &, Uint32 i, bool CrashOnBoundaryError) const;
+
+ /**
+ * Allocate an object from pool - update Ptr
+ *
+ * Return i
+ */
+ bool seize(Ptr<T> &);
+
+ /**
+ * Allocate object <b>i</b> from pool - update Ptr
+ */
+ bool seizeId(Ptr<T> &, Uint32 i);
+
+ /**
+ * Check if <b>i</b> is allocated.
+ */
+ bool findId(Uint32 i) const;
+
+ /**
+ * Return an object to pool
+ */
+ void release(Uint32 i);
+
+ /**
+ * Return an object to pool
+ */
+ void release(Ptr<T> &);
+
+#ifdef ARRAY_GUARD
+ /**
+ * Checks if i is a correct seized record
+ *
+ * @note Since this is either an expensive method,
+ * or needs bitmask stuff, this method is only
+ * recommended for debugging.
+ *
+ */
+ bool isSeized(Uint32 i) const {
+ if (i>=size) return false;
+ return BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i);
+ }
+#endif
+
+protected:
+ friend class Array<T>;
+ friend class SLList<T>;
+ friend class DLList<T>;
+ friend class DLHashTable<T>;
+
+ /**
+ * Allocate <b>n</b> consecutive object from pool
+ * return base
+ */
+ Uint32 seizeN(Uint32 n);
+
+ /**
+ * Deallocate <b>n<b> consecutive object to pool
+ * starting from base
+ */
+ void releaseN(Uint32 base, Uint32 n);
+
+public:
+ /**
+ * Release a singel linked list in o(1)
+ * @param first i-value of first element in list
+ * @param last i-value of last element in list
+ * @note nextPool must be used as next pointer in list
+ */
+ void releaseList(Uint32 n, Uint32 first, Uint32 last);
+ //private:
+
+ /**
+ * Print
+ * (Run operator NdbOut<< on every element)
+ */
+ void print(NdbOut & out){
+ out << "FirstFree = " << firstFree << endl;
+ for(Uint32 i = 0; i<size; i++){
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ out << "A ";
+ else
+ out << "F ";
+#endif
+ out << i << ": " << theArray[i] << " ";
+ }
+ out << endl;
+ }
+
+#ifdef DEBUG
+ Uint32 getNoOfFree2() const {
+ Uint32 c2 = size;
+ for(Uint32 i = 0; i<((size + 31)>> 5); i++){
+ Uint32 w = theAllocatedBitmask[i];
+ for(Uint32 j = 0; j<32; j++){
+ if((w & 1) == 1){
+ c2--;
+ }
+ w >>= 1;
+ }
+ }
+ return c2;
+ }
+
+ Uint32 getNoOfFree3() const {
+ Uint32 c = 0;
+ Ptr<T> p;
+ p.i = firstFree;
+ while(p.i != RNIL){
+ c++;
+ p.p = &theArray[p.i];
+ p.i = p.p->next;
+ }
+ return c;
+ }
+#endif
+
+protected:
+ Uint32 firstFree;
+ Uint32 size;
+ Uint32 noOfFree;
+ T * theArray;
+ Uint32 bitmaskSz;
+ Uint32 *theAllocatedBitmask;
+};
+
+template <class T>
+inline
+ArrayPool<T>::ArrayPool(){
+ firstFree = RNIL;
+ size = 0;
+ noOfFree = 0;
+ theArray = 0;
+#ifdef ARRAY_GUARD
+ theAllocatedBitmask = 0;
+#endif
+}
+
+template <class T>
+inline
+ArrayPool<T>::~ArrayPool(){
+ if(theArray != 0){
+ NdbMem_Free(theArray);
+ theArray = 0;
+#ifdef ARRAY_GUARD
+ delete []theAllocatedBitmask;
+ theAllocatedBitmask = 0;
+#endif
+ }
+}
+
+/**
+ * Set the size of the pool
+ *
+ * Note, can currently only be called once
+ */
+template <class T>
+inline
+bool
+ArrayPool<T>::setSize(Uint32 noOfElements){
+ if(size == 0){
+ if(noOfElements == 0)
+ return true;
+ theArray = (T *)NdbMem_Allocate(noOfElements * sizeof(T));
+ if(theArray == 0)
+ return false;
+ size = noOfElements;
+ noOfFree = noOfElements;
+
+ /**
+ * Set next pointers
+ */
+ T * t = &theArray[0];
+ for(Uint32 i = 0; i<size; i++){
+ t->nextPool = (i + 1);
+ t++;
+ }
+ theArray[size-1].nextPool = RNIL;
+ firstFree = 0;
+
+#ifdef ARRAY_GUARD
+ bitmaskSz = (noOfElements + 31) >> 5;
+ theAllocatedBitmask = new Uint32[bitmaskSz];
+ BitmaskImpl::clear(bitmaskSz, theAllocatedBitmask);
+#endif
+
+ return true;
+ }
+ return false;
+}
+
+template <class T>
+inline
+void
+ArrayPool<T>::getPtr(Ptr<T> & ptr){
+ Uint32 i = ptr.i;
+ if(i < size){
+ ptr.p = &theArray[i];
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return;
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+#endif
+ } else {
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+ }
+}
+
+template <class T>
+inline
+void
+ArrayPool<T>::getPtr(ConstPtr<T> & ptr) const {
+ Uint32 i = ptr.i;
+ if(i < size){
+ ptr.p = &theArray[i];
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return;
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+#endif
+ } else {
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+ }
+}
+
+template <class T>
+inline
+void
+ArrayPool<T>::getPtr(Ptr<T> & ptr, Uint32 i){
+ ptr.i = i;
+ if(i < size){
+ ptr.p = &theArray[i];
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return;
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+#endif
+ } else {
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+ }
+}
+
+template <class T>
+inline
+void
+ArrayPool<T>::getPtr(ConstPtr<T> & ptr, Uint32 i) const {
+ ptr.i = i;
+ if(i < size){
+ ptr.p = &theArray[i];
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return;
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+#endif
+ } else {
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+ }
+}
+
+template <class T>
+inline
+T *
+ArrayPool<T>::getPtr(Uint32 i){
+ if(i < size){
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return &theArray[i];
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+ return 0;
+#else
+ return &theArray[i];
+#endif
+ } else {
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+ return 0;
+ }
+}
+
+template <class T>
+inline
+const T *
+ArrayPool<T>::getConstPtr(Uint32 i) const {
+ if(i < size){
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return &theArray[i];
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+ return 0;
+#else
+ return &theArray[i];
+#endif
+ } else {
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+ return 0;
+ }
+}
+
+template <class T>
+inline
+void
+ArrayPool<T>::getPtr(Ptr<T> & ptr, bool CrashOnBoundaryError){
+ Uint32 i = ptr.i;
+ if(i < size){
+ ptr.p = &theArray[i];
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return;
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+#endif
+ } else {
+ ptr.i = RNIL;
+ }
+}
+
+template <class T>
+inline
+void
+ArrayPool<T>::getPtr(ConstPtr<T> & ptr, bool CrashOnBoundaryError) const {
+ Uint32 i = ptr.i;
+ if(i < size){
+ ptr.p = &theArray[i];
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return;
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+#endif
+ } else {
+ ptr.i = RNIL;
+ }
+}
+
+template <class T>
+inline
+void
+ArrayPool<T>::getPtr(Ptr<T> & ptr, Uint32 i, bool CrashOnBoundaryError){
+ ptr.i = i;
+ if(i < size){
+ ptr.p = &theArray[i];
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return;
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+#endif
+ } else {
+ ptr.i = RNIL;
+ }
+}
+
+template <class T>
+inline
+void
+ArrayPool<T>::getPtr(ConstPtr<T> & ptr, Uint32 i,
+ bool CrashOnBoundaryError) const {
+ ptr.i = i;
+ if(i < size){
+ ptr.p = &theArray[i];
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return;
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+#endif
+ } else {
+ ptr.i = RNIL;
+ }
+}
+
+template <class T>
+inline
+T *
+ArrayPool<T>::getPtr(Uint32 i, bool CrashOnBoundaryError){
+ if(i < size){
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return &theArray[i];
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getPtr", __FILE__, __LINE__);
+ return 0;
+#else
+ return &theArray[i];
+#endif
+ } else {
+ return 0;
+ }
+}
+
+template <class T>
+inline
+const T *
+ArrayPool<T>::getConstPtr(Uint32 i, bool CrashOnBoundaryError) const {
+ if(i < size){
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i))
+ return &theArray[i];
+ /**
+ * Getting a non-seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::getConstPtr", __FILE__,__LINE__);
+ return 0;
+#else
+ return &theArray[i];
+#endif
+ } else {
+ return 0;
+ }
+}
+
+/**
+ * Allocate an object from pool - update Ptr
+ *
+ * Return i
+ */
+template <class T>
+inline
+bool
+ArrayPool<T>::seize(Ptr<T> & ptr){
+ Uint32 ff = firstFree;
+ if(ff != RNIL){
+ firstFree = theArray[ff].nextPool;
+
+ ptr.i = ff;
+ ptr.p = &theArray[ff];
+#ifdef ARRAY_GUARD
+ if(!BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, ff)){
+ BitmaskImpl::set(bitmaskSz, theAllocatedBitmask, ff);
+ noOfFree--;
+ return true;
+ } else {
+ /**
+ * Seizing an already seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::seize", __FILE__, __LINE__);
+ return false;
+ }
+#else
+ noOfFree--;
+ return true;
+#endif
+ }
+ ptr.i = RNIL;
+ ptr.p = NULL;
+ return false;
+}
+
+template <class T>
+inline
+bool
+ArrayPool<T>::seizeId(Ptr<T> & ptr, Uint32 i){
+ Uint32 ff = firstFree;
+ Uint32 prev = RNIL;
+ while(ff != i && ff != RNIL){
+ prev = ff;
+ ff = theArray[ff].nextPool;
+ }
+
+ if(ff != RNIL){
+ if(prev == RNIL)
+ firstFree = theArray[ff].nextPool;
+ else
+ theArray[prev].nextPool = theArray[ff].nextPool;
+
+ ptr.i = ff;
+ ptr.p = &theArray[ff];
+#ifdef ARRAY_GUARD
+ if(!BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, ff)){
+ BitmaskImpl::set(bitmaskSz, theAllocatedBitmask, ff);
+ noOfFree--;
+ return true;
+ } else {
+ /**
+ * Seizing an already seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::seizeId", __FILE__, __LINE__);
+ return false;
+ }
+#else
+ noOfFree--;
+ return true;
+#endif
+ }
+ ptr.i = RNIL;
+ ptr.p = NULL;
+ return false;
+}
+
+template <class T>
+inline
+bool
+ArrayPool<T>::findId(Uint32 i) const {
+ if (i >= size)
+ return false;
+ Uint32 ff = firstFree;
+ while(ff != i && ff != RNIL){
+ ff = theArray[ff].nextPool;
+ }
+ return (ff == RNIL);
+}
+
+template<class T>
+Uint32
+ArrayPool<T>::seizeN(Uint32 n){
+ Uint32 curr = firstFree;
+ Uint32 prev = RNIL;
+ Uint32 sz = 0;
+ while(sz < n && curr != RNIL){
+ if(theArray[curr].nextPool == (curr + 1)){
+ sz++;
+ } else {
+ sz = 0;
+ prev = curr;
+ }
+ curr = theArray[curr].nextPool;
+ }
+ if(sz != n){
+ return RNIL;
+ }
+ const Uint32 base = curr - n;
+ if(base == firstFree){
+ firstFree = curr;
+ } else {
+ theArray[prev].nextPool = curr;
+ }
+
+ noOfFree -= n;
+#ifdef ARRAY_GUARD
+ for(Uint32 j = base; j<curr; j++){
+ if(!BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, j)){
+ BitmaskImpl::set(bitmaskSz, theAllocatedBitmask, j);
+ } else {
+ /**
+ * Seizing an already seized element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::seize", __FILE__, __LINE__);
+ return RNIL;
+ }
+ }
+#endif
+ return base;
+}
+
+template<class T>
+inline
+void
+ArrayPool<T>::releaseN(Uint32 base, Uint32 n){
+ Uint32 curr = firstFree;
+ Uint32 prev = RNIL;
+ while(curr < base){
+ prev = curr;
+ curr = theArray[curr].nextPool;
+ }
+ if(curr == firstFree){
+ firstFree = base;
+ } else {
+ theArray[prev].nextPool = base;
+ }
+ const Uint32 end = base + n;
+ for(Uint32 i = base; i<end; i++){
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i)){
+ BitmaskImpl::clear(bitmaskSz, theAllocatedBitmask, i);
+ } else {
+ /**
+ * Relesing a already released element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::release", __FILE__, __LINE__);
+ return;
+ }
+#endif
+ theArray[i].nextPool = i + 1;
+ }
+ theArray[end-1].nextPool = curr;
+ noOfFree += n;
+}
+
+template<class T>
+inline
+void
+ArrayPool<T>::releaseList(Uint32 n, Uint32 first, Uint32 last){
+
+ if(first < size && last < size){
+ Uint32 ff = firstFree;
+ firstFree = first;
+ theArray[last].nextPool = ff;
+ noOfFree += n;
+
+#ifdef ARRAY_GUARD
+ Uint32 tmp = first;
+ for(Uint32 i = 0; i<n; i++){
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, tmp)){
+ BitmaskImpl::clear(bitmaskSz, theAllocatedBitmask, tmp);
+ } else {
+ /**
+ * Relesing a already released element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::releaseList",
+ __FILE__, __LINE__);
+ return;
+ }
+ tmp = theArray[tmp].nextPool;
+ }
+#endif
+ return;
+ }
+ ErrorReporter::handleAssert("ArrayPool<T>::releaseList", __FILE__, __LINE__);
+}
+
+/**
+ * Return an object to pool
+ */
+template <class T>
+inline
+void
+ArrayPool<T>::release(Uint32 _i){
+ const Uint32 i = _i;
+ if(i < size){
+ Uint32 ff = firstFree;
+ theArray[i].nextPool = ff;
+ firstFree = i;
+
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i)){
+ BitmaskImpl::clear(bitmaskSz, theAllocatedBitmask, i);
+ noOfFree++;
+ return;
+ }
+ /**
+ * Relesing a already released element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::release", __FILE__, __LINE__);
+#endif
+ noOfFree++;
+ return;
+ }
+ ErrorReporter::handleAssert("ArrayPool<T>::release", __FILE__, __LINE__);
+}
+
+/**
+ * Return an object to pool
+ */
+template <class T>
+inline
+void
+ArrayPool<T>::release(Ptr<T> & ptr){
+ Uint32 i = ptr.i;
+ if(i < size){
+ Uint32 ff = firstFree;
+ theArray[i].nextPool = ff;
+ firstFree = i;
+
+#ifdef ARRAY_GUARD
+ if(BitmaskImpl::get(bitmaskSz, theAllocatedBitmask, i)){
+ BitmaskImpl::clear(bitmaskSz, theAllocatedBitmask, i);
+ //assert(noOfFree() == noOfFree2());
+ noOfFree++;
+ return;
+ }
+ /**
+ * Relesing a already released element
+ */
+ ErrorReporter::handleAssert("ArrayPool<T>::release", __FILE__, __LINE__);
+#endif
+ noOfFree++;
+ return;
+ }
+ ErrorReporter::handleAssert("ArrayPool<T>::release", __FILE__, __LINE__);
+}
+
+template <class T>
+class UnsafeArrayPool : public ArrayPool<T> {
+public:
+ /**
+ * Update p value for ptr according to i value
+ * ignore if it's allocated or not
+ */
+ void getPtrForce(Ptr<T> &);
+ void getPtrForce(ConstPtr<T> &) const;
+ T * getPtrForce(Uint32 i);
+ const T * getConstPtrForce(Uint32 i) const;
+ void getPtrForce(Ptr<T> &, Uint32 i);
+ void getPtrForce(ConstPtr<T> &, Uint32 i) const;
+};
+
+template <class T>
+inline
+void
+UnsafeArrayPool<T>::getPtrForce(Ptr<T> & ptr){
+ Uint32 i = ptr.i;
+ if(i < size){
+ ptr.p = &theArray[i];
+ } else {
+ ErrorReporter::handleAssert("UnsafeArrayPool<T>::getPtr",
+ __FILE__, __LINE__);
+ }
+}
+
+template <class T>
+inline
+void
+UnsafeArrayPool<T>::getPtrForce(ConstPtr<T> & ptr) const{
+ Uint32 i = ptr.i;
+ if(i < size){
+ ptr.p = &theArray[i];
+ } else {
+ ErrorReporter::handleAssert("UnsafeArrayPool<T>::getPtr",
+ __FILE__, __LINE__);
+ }
+}
+
+template <class T>
+inline
+T *
+UnsafeArrayPool<T>::getPtrForce(Uint32 i){
+ if(i < size){
+ return &theArray[i];
+ } else {
+ ErrorReporter::handleAssert("UnsafeArrayPool<T>::getPtr",
+ __FILE__, __LINE__);
+ return 0;
+ }
+}
+
+template <class T>
+inline
+const T *
+UnsafeArrayPool<T>::getConstPtrForce(Uint32 i) const {
+ if(i < size){
+ return &theArray[i];
+ } else {
+ ErrorReporter::handleAssert("UnsafeArrayPool<T>::getPtr",
+ __FILE__, __LINE__);
+ return 0;
+ }
+}
+
+template <class T>
+inline
+void
+UnsafeArrayPool<T>::getPtrForce(Ptr<T> & ptr, Uint32 i){
+ ptr.i = i;
+ if(i < size){
+ ptr.p = &theArray[i];
+ return ;
+ } else {
+ ErrorReporter::handleAssert("UnsafeArrayPool<T>::getPtr",
+ __FILE__, __LINE__);
+ }
+}
+
+template <class T>
+inline
+void
+UnsafeArrayPool<T>::getPtrForce(ConstPtr<T> & ptr, Uint32 i) const{
+ ptr.i = i;
+ if(i < size){
+ ptr.p = &theArray[i];
+ return ;
+ } else {
+ ErrorReporter::handleAssert("UnsafeArrayPool<T>::getPtr",
+ __FILE__, __LINE__);
+ }
+}
+
+
+#endif
diff --git a/ndb/src/kernel/vm/CArray.hpp b/ndb/src/kernel/vm/CArray.hpp
new file mode 100644
index 00000000000..a6e84e2c041
--- /dev/null
+++ b/ndb/src/kernel/vm/CArray.hpp
@@ -0,0 +1,141 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CARRAY_HPP
+#define CARRAY_HPP
+
+/**
+ * Template class used for implementing an c - array
+ */
+template <class T>
+class CArray {
+public:
+ CArray();
+ ~CArray();
+
+ /**
+ * Set the size of the pool
+ *
+ * Note, can currently only be called once
+ */
+ bool setSize(Uint32 noOfElements);
+
+ /**
+ * Get size
+ */
+ Uint32 getSize() const;
+
+ /**
+ * Update p value for ptr according to i value
+ */
+ void getPtr(Ptr<T> &) const;
+
+ /**
+ * Get pointer for i value
+ */
+ T * getPtr(Uint32 i) const;
+
+ /**
+ * Update p & i value for ptr according to <b>i</b> value
+ */
+ void getPtr(Ptr<T> &, Uint32 i) const;
+
+private:
+ Uint32 size;
+ T * theArray;
+};
+
+template <class T>
+inline
+CArray<T>::CArray(){
+ size = 0;
+ theArray = 0;
+}
+
+template <class T>
+inline
+CArray<T>::~CArray(){
+ if(theArray != 0){
+ NdbMem_Free(theArray);
+ theArray = 0;
+ }
+}
+
+/**
+ * Set the size of the pool
+ *
+ * Note, can currently only be called once
+ */
+template <class T>
+inline
+bool
+CArray<T>::setSize(Uint32 noOfElements){
+ if(size == noOfElements)
+ return true;
+
+ theArray = (T *)NdbMem_Allocate(noOfElements * sizeof(T));
+ if(theArray == 0)
+ return false;
+ size = noOfElements;
+ return true;
+}
+
+template<class T>
+inline
+Uint32
+CArray<T>::getSize() const {
+ return size;
+}
+
+template <class T>
+inline
+void
+CArray<T>::getPtr(Ptr<T> & ptr) const {
+ const Uint32 i = ptr.i;
+ if(i < size){
+ ptr.p = &theArray[i];
+ return;
+ } else {
+ ErrorReporter::handleAssert("CArray<T>::getPtr", __FILE__, __LINE__);
+ }
+}
+
+template <class T>
+inline
+T *
+CArray<T>::getPtr(Uint32 i) const {
+ if(i < size){
+ return &theArray[i];
+ } else {
+ ErrorReporter::handleAssert("CArray<T>::getPtr", __FILE__, __LINE__);
+ return 0;
+ }
+}
+
+template <class T>
+inline
+void
+CArray<T>::getPtr(Ptr<T> & ptr, Uint32 i) const {
+ ptr.i = i;
+ if(i < size){
+ ptr.p = &theArray[i];
+ return;
+ } else {
+ ErrorReporter::handleAssert("CArray<T>::getPtr", __FILE__, __LINE__);
+ }
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/Callback.hpp b/ndb/src/kernel/vm/Callback.hpp
new file mode 100644
index 00000000000..bf1ae5968d3
--- /dev/null
+++ b/ndb/src/kernel/vm/Callback.hpp
@@ -0,0 +1,31 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BLOCK_CALLBACK_HPP
+#define BLOCK_CALLBACK_HPP
+
+/**
+ * Block callbacks
+ */
+typedef void (SimulatedBlock::* CallbackFunction)(class Signal*,
+ Uint32 callbackData,
+ Uint32 returnCode);
+struct Callback {
+ CallbackFunction m_callbackFunction;
+ Uint32 m_callbackData;
+};
+
+#endif
diff --git a/ndb/src/kernel/vm/ClusterConfiguration.cpp b/ndb/src/kernel/vm/ClusterConfiguration.cpp
new file mode 100644
index 00000000000..f04081ee3c1
--- /dev/null
+++ b/ndb/src/kernel/vm/ClusterConfiguration.cpp
@@ -0,0 +1,485 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "ClusterConfiguration.hpp"
+#include <ErrorHandlingMacros.hpp>
+
+#include <pc.hpp>
+#include <BlockNumbers.h>
+#include <signaldata/AccSizeAltReq.hpp>
+#include <signaldata/DictSizeAltReq.hpp>
+#include <signaldata/DihSizeAltReq.hpp>
+#include <signaldata/LqhSizeAltReq.hpp>
+#include <signaldata/TcSizeAltReq.hpp>
+#include <signaldata/TupSizeAltReq.hpp>
+#include <signaldata/TuxSizeAltReq.hpp>
+
+#include <string.h>
+#include <assert.h>
+
+ClusterConfiguration::ClusterConfiguration()
+{
+ for (unsigned i= 0; i< MAX_SIZEALT_BLOCKS; i++) // initialize
+ for (unsigned j= 0; j< MAX_SIZEALT_RECORD; j++) {
+ the_clusterData.SizeAltData.varSize[i][j].valid = false;
+ the_clusterData.SizeAltData.varSize[i][j].nrr = 0;
+ }
+
+ for (unsigned i1 = 0; i1< 5; i1++) // initialize
+ for (unsigned j1= 0; j1< CmvmiCfgConf::NO_OF_WORDS; j1++)
+ the_clusterData.ispValues[i1][j1] = 0;
+
+ the_clusterData.SizeAltData.noOfNodes = 0;
+ the_clusterData.SizeAltData.noOfNDBNodes = 0;
+ the_clusterData.SizeAltData.noOfAPINodes = 0;
+ the_clusterData.SizeAltData.noOfMGMNodes = 0;
+}
+
+ClusterConfiguration::~ClusterConfiguration(){
+}
+
+void
+setValue(VarSize* dst, const int index, UintR variableValue){
+ assert(dst != NULL);
+ assert(index >= 0 && index < MAX_SIZEALT_RECORD);
+
+ dst[index].nrr = variableValue;
+ dst[index].valid = true;
+}
+
+void
+ClusterConfiguration::calcSizeAlteration()
+{
+ SizeAlt *size = &the_clusterData.SizeAltData;
+
+ size->noOfTables++; // Remove impact of system table
+ size->noOfTables += size->noOfIndexes; // Indexes are tables too
+ size->noOfAttributes += 2; // ---"----
+
+ size->noOfTables *= 2; // Remove impact of Dict need 2 ids for each table
+
+ Uint32 noOfDBNodes = size->noOfNDBNodes;
+ if (noOfDBNodes > 15) {
+ noOfDBNodes = 15;
+ }//if
+ Uint32 noOfLocalScanRecords = (noOfDBNodes * size->noOfScanRecords) + 1;
+ Uint32 noOfTCScanRecords = size->noOfScanRecords;
+ {
+ /**
+ * Acc Size Alt values
+ */
+ size->blockNo[ACC] = DBACC;
+
+ VarSize * const acc = &(size->varSize[ACC][0]);
+
+ // Can keep 65536 pages (= 0.5 GByte)
+ setValue(acc, AccSizeAltReq::IND_DIR_RANGE,
+ 4 * NO_OF_FRAG_PER_NODE * size->noOfTables* size->noOfReplicas);
+
+ setValue(acc, AccSizeAltReq::IND_DIR_ARRAY,
+ (size->noOfIndexPages >> 8) +
+ 4 * NO_OF_FRAG_PER_NODE * size->noOfTables* size->noOfReplicas);
+
+ setValue(acc, AccSizeAltReq::IND_FRAGMENT,
+ 2 * NO_OF_FRAG_PER_NODE * size->noOfTables* size->noOfReplicas);
+
+ /*-----------------------------------------------------------------------*/
+ // The extra operation records added are used by the scan and node
+ // recovery process.
+ // Node recovery process will have its operations dedicated to ensure
+ // that they never have a problem with allocation of the operation record.
+ // The remainder are allowed for use by the scan processes.
+ /*-----------------------------------------------------------------------*/
+ setValue(acc, AccSizeAltReq::IND_OP_RECS,
+ size->noOfReplicas*((16 * size->noOfOperations) / 10 + 50) +
+ (noOfLocalScanRecords * MAX_PARALLEL_SCANS_PER_FRAG) +
+ NODE_RECOVERY_SCAN_OP_RECORDS);
+
+ setValue(acc, AccSizeAltReq::IND_OVERFLOW_RECS,
+ size->noOfIndexPages +
+ 2 * NO_OF_FRAG_PER_NODE * size->noOfTables* size->noOfReplicas);
+
+ setValue(acc, AccSizeAltReq::IND_PAGE8,
+ size->noOfIndexPages + 32);
+
+ setValue(acc, AccSizeAltReq::IND_ROOT_FRAG,
+ NO_OF_FRAG_PER_NODE * size->noOfTables* size->noOfReplicas);
+
+ setValue(acc, AccSizeAltReq::IND_TABLE,
+ size->noOfTables);
+
+ setValue(acc, AccSizeAltReq::IND_SCAN,
+ noOfLocalScanRecords);
+ }
+
+ {
+ /**
+ * Dict Size Alt values
+ */
+ size->blockNo[DICT] = DBDICT;
+
+ VarSize * const dict = &(size->varSize[DICT][0]);
+
+ setValue(dict, DictSizeAltReq::IND_ATTRIBUTE,
+ size->noOfAttributes);
+
+ setValue(dict, DictSizeAltReq::IND_CONNECT,
+ size->noOfOperations + 32);
+
+ setValue(dict, DictSizeAltReq::IND_FRAG_CONNECT,
+ NO_OF_FRAG_PER_NODE * size->noOfNDBNodes * size->noOfReplicas);
+
+ setValue(dict, DictSizeAltReq::IND_TABLE,
+ size->noOfTables);
+
+ setValue(dict, DictSizeAltReq::IND_TC_CONNECT,
+ 2* size->noOfOperations);
+ }
+
+ {
+ /**
+ * Dih Size Alt values
+ */
+ size->blockNo[DIH] = DBDIH;
+
+ VarSize * const dih = &(size->varSize[DIH][0]);
+
+ setValue(dih, DihSizeAltReq::IND_API_CONNECT,
+ 2 * size->noOfTransactions);
+
+ setValue(dih, DihSizeAltReq::IND_CONNECT,
+ size->noOfOperations + 46);
+
+ setValue(dih, DihSizeAltReq::IND_FRAG_CONNECT,
+ NO_OF_FRAG_PER_NODE * size->noOfTables * size->noOfNDBNodes);
+
+ int temp;
+ temp = size->noOfReplicas - 2;
+ if (temp < 0)
+ temp = 1;
+ else
+ temp++;
+ setValue(dih, DihSizeAltReq::IND_MORE_NODES,
+ temp * NO_OF_FRAG_PER_NODE *
+ size->noOfTables * size->noOfNDBNodes);
+
+ setValue(dih, DihSizeAltReq::IND_REPLICAS,
+ NO_OF_FRAG_PER_NODE * size->noOfTables *
+ size->noOfNDBNodes * size->noOfReplicas);
+
+ setValue(dih, DihSizeAltReq::IND_TABLE,
+ size->noOfTables);
+ }
+
+ {
+ /**
+ * Lqh Size Alt values
+ */
+ size->blockNo[LQH] = DBLQH;
+
+ VarSize * const lqh = &(size->varSize[LQH][0]);
+
+ setValue(lqh, LqhSizeAltReq::IND_FRAG,
+ NO_OF_FRAG_PER_NODE * size->noOfTables * size->noOfReplicas);
+
+ setValue(lqh, LqhSizeAltReq::IND_CONNECT,
+ size->noOfReplicas*((11 * size->noOfOperations) / 10 + 50));
+
+ setValue(lqh, LqhSizeAltReq::IND_TABLE,
+ size->noOfTables);
+
+ setValue(lqh, LqhSizeAltReq::IND_TC_CONNECT,
+ size->noOfReplicas*((16 * size->noOfOperations) / 10 + 50));
+
+ setValue(lqh, LqhSizeAltReq::IND_REPLICAS,
+ size->noOfReplicas);
+
+ setValue(lqh, LqhSizeAltReq::IND_LOG_FILES,
+ (4 * the_clusterData.ispValues[1][4]));
+
+ setValue(lqh, LqhSizeAltReq::IND_SCAN,
+ noOfLocalScanRecords);
+
+ }
+
+ {
+ /**
+ * Tc Size Alt values
+ */
+ size->blockNo[TC] = DBTC;
+
+ VarSize * const tc = &(size->varSize[TC][0]);
+
+ setValue(tc, TcSizeAltReq::IND_API_CONNECT,
+ 3 * size->noOfTransactions);
+
+ setValue(tc, TcSizeAltReq::IND_TC_CONNECT,
+ size->noOfOperations + 16 + size->noOfTransactions);
+
+ setValue(tc, TcSizeAltReq::IND_TABLE,
+ size->noOfTables);
+
+ setValue(tc, TcSizeAltReq::IND_LOCAL_SCAN,
+ noOfLocalScanRecords);
+
+ setValue(tc, TcSizeAltReq::IND_TC_SCAN,
+ noOfTCScanRecords);
+ }
+
+ {
+ /**
+ * Tup Size Alt values
+ */
+ size->blockNo[TUP] = DBTUP;
+
+ VarSize * const tup = &(size->varSize[TUP][0]);
+
+ setValue(tup, TupSizeAltReq::IND_DISK_PAGE_ARRAY,
+ 2 * NO_OF_FRAG_PER_NODE * size->noOfTables* size->noOfReplicas);
+
+ setValue(tup, TupSizeAltReq::IND_DISK_PAGE_REPRESENT,
+ size->noOfDiskClusters);
+
+ setValue(tup, TupSizeAltReq::IND_FRAG,
+ 2 * NO_OF_FRAG_PER_NODE * size->noOfTables* size->noOfReplicas);
+
+ setValue(tup, TupSizeAltReq::IND_PAGE_CLUSTER,
+ size->noOfFreeClusters);
+
+ setValue(tup, TupSizeAltReq::IND_LOGIC_PAGE,
+ size->noOfDiskBufferPages + size->noOfDiskClusters);
+
+ setValue(tup, TupSizeAltReq::IND_OP_RECS,
+ size->noOfReplicas*((16 * size->noOfOperations) / 10 + 50));
+
+ setValue(tup, TupSizeAltReq::IND_PAGE,
+ size->noOfDataPages);
+
+ setValue(tup, TupSizeAltReq::IND_PAGE_RANGE,
+ 4 * NO_OF_FRAG_PER_NODE * size->noOfTables* size->noOfReplicas);
+
+ setValue(tup, TupSizeAltReq::IND_TABLE,
+ size->noOfTables);
+
+ setValue(tup, TupSizeAltReq::IND_TABLE_DESC,
+ 4 * NO_OF_FRAG_PER_NODE * size->noOfAttributes* size->noOfReplicas +
+ 12 * NO_OF_FRAG_PER_NODE * size->noOfTables* size->noOfReplicas );
+
+ setValue(tup, TupSizeAltReq::IND_DELETED_BLOCKS,
+ size->noOfFreeClusters);
+
+ setValue(tup, TupSizeAltReq::IND_STORED_PROC,
+ noOfLocalScanRecords);
+ }
+
+ {
+ /**
+ * Tux Size Alt values
+ */
+ size->blockNo[TUX] = DBTUX;
+
+ VarSize * const tux = &(size->varSize[TUX][0]);
+
+ setValue(tux, TuxSizeAltReq::IND_INDEX,
+ size->noOfTables);
+
+ setValue(tux, TuxSizeAltReq::IND_FRAGMENT,
+ 2 * NO_OF_FRAG_PER_NODE * size->noOfTables * size->noOfReplicas);
+
+ setValue(tux, TuxSizeAltReq::IND_ATTRIBUTE,
+ size->noOfIndexes * 4);
+
+ setValue(tux, TuxSizeAltReq::IND_SCAN,
+ noOfLocalScanRecords);
+ }
+}
+
+const ClusterConfiguration::ClusterData&
+ClusterConfiguration::clusterData() const
+{
+ return the_clusterData;
+}
+
+void ClusterConfiguration::init(const Properties & p, const Properties & db){
+ const char * msg = "Invalid configuration fetched";
+
+ ClusterData & cd = the_clusterData;
+
+ struct AttribStorage { const char * attrib; Uint32 * storage; };
+ AttribStorage tmp[] = {
+ {"MaxNoOfConcurrentScans", &cd.SizeAltData.noOfScanRecords },
+ {"MaxNoOfTables", &cd.SizeAltData.noOfTables },
+ {"MaxNoOfIndexes", &cd.SizeAltData.noOfIndexes },
+ {"NoOfReplicas", &cd.SizeAltData.noOfReplicas },
+ {"MaxNoOfAttributes", &cd.SizeAltData.noOfAttributes },
+ {"MaxNoOfConcurrentOperations", &cd.SizeAltData.noOfOperations },
+ {"MaxNoOfConcurrentTransactions", &cd.SizeAltData.noOfTransactions },
+ {"NoOfIndexPages", &cd.SizeAltData.noOfIndexPages },
+ {"NoOfDataPages", &cd.SizeAltData.noOfDataPages },
+ {"NoOfDiskBufferPages", &cd.SizeAltData.noOfDiskBufferPages },
+ {"NoOfDiskClusters", &cd.SizeAltData.noOfDiskClusters },
+ {"NoOfFreeDiskClusters", &cd.SizeAltData.noOfFreeClusters },
+ {"TimeToWaitAlive", &cd.ispValues[0][0] },
+ {"HeartbeatIntervalDbDb", &cd.ispValues[0][2] },
+ {"HeartbeatIntervalDbApi", &cd.ispValues[0][3] },
+ {"ArbitrationTimeout", &cd.ispValues[0][5] },
+ {"TimeBetweenLocalCheckpoints", &cd.ispValues[1][2] },
+ {"NoOfFragmentLogFiles", &cd.ispValues[1][4] },
+ {"MaxNoOfConcurrentScans", &cd.SizeAltData.noOfScanRecords },
+ {"NoOfConcurrentCheckpointsDuringRestart", &cd.ispValues[1][5] },
+ {"TransactionDeadlockDetectionTimeout", &cd.ispValues[1][6] },
+ {"NoOfConcurrentProcessesHandleTakeover", &cd.ispValues[1][7] },
+ {"TimeBetweenGlobalCheckpoints", &cd.ispValues[2][3] },
+ {"NoOfConcurrentCheckpointsAfterRestart", &cd.ispValues[2][4] },
+ {"TransactionInactiveTimeout", &cd.ispValues[2][7] },
+ {"NoOfDiskPagesToDiskDuringRestartTUP", &cd.ispValues[3][8] },
+ {"NoOfDiskPagesToDiskAfterRestartTUP", &cd.ispValues[3][9] },
+ {"NoOfDiskPagesToDiskDuringRestartACC", &cd.ispValues[3][10] },
+ {"NoOfDiskPagesToDiskAfterRestartACC", &cd.ispValues[3][11] },
+ {"NoOfDiskClustersPerDiskFile", &cd.ispValues[4][8] },
+ {"NoOfDiskFiles", &cd.ispValues[4][9] },
+ {"NoOfReplicas", &cd.ispValues[2][2] }
+ };
+
+
+ const int sz = sizeof(tmp)/sizeof(AttribStorage);
+ for(int i = 0; i<sz; i++){
+ if(!db.get(tmp[i].attrib, tmp[i].storage)){
+ char buf[255];
+ snprintf(buf, sizeof(buf), "%s not found", tmp[i].attrib);
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, buf);
+ }
+ }
+
+ if(!p.get("NoOfNodes", &cd.SizeAltData.noOfNodes)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, "NoOfNodes missing");
+ }
+
+ Properties::Iterator it(&p);
+ const char * name = 0;
+ Uint32 nodeNo = 0;
+ for(name = it.first(); name != NULL; name = it.next()){
+ if(strncmp(name, "Node_", strlen("Node_")) == 0){
+
+ Uint32 nodeId;
+ const char * nodeType;
+ const Properties * node;
+
+ if(!p.get(name, &node)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, "Node data missing");
+ }
+
+ if(!node->get("Id", &nodeId)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, "Node data (Id) missing");
+ }
+
+ if(!node->get("Type", &nodeType)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, "Node data (Type) missing");
+ }
+
+ if(nodeId > MAX_NODES){
+ char buf[255];
+ snprintf(buf, sizeof(buf),
+ "Maximum DB node id allowed is: %d", MAX_NDB_NODES);
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, buf);
+ }
+
+ if(nodeId == 0){
+ char buf[255];
+ snprintf(buf, sizeof(buf),
+ "Minimum node id allowed in the cluster is: 1");
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, buf);
+ }
+
+ for(unsigned j = 0; j<nodeNo; j++){
+ if(cd.nodeData[j].nodeId == nodeId){
+ char buf[255];
+ snprintf(buf, sizeof(buf), "Two node can not have the same node id");
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, buf);
+ }
+ }
+
+ {
+ for(unsigned j = 0; j<LogLevel::LOGLEVEL_CATEGORIES; j++){
+ Uint32 logLevel;
+ if(db.get(LogLevel::LOGLEVEL_CATEGORY_NAME[j].name, &logLevel)){
+ cd.SizeAltData.logLevel.setLogLevel((LogLevel::EventCategory)j,
+ logLevel);
+ }
+ }
+ }
+
+ cd.nodeData[nodeNo].nodeId = nodeId;
+ const char* tmpApiMgmProperties = 0;
+ if(strcmp("DB", nodeType) == 0){
+ cd.nodeData[nodeNo].nodeType = NodeInfo::DB;
+ cd.SizeAltData.noOfNDBNodes++; // No of NDB processes
+
+ if(nodeId > MAX_NDB_NODES){
+ char buf[255];
+ snprintf(buf, sizeof(buf), "Maximum node id for a ndb node is: %d", MAX_NDB_NODES);
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, buf);
+ }
+ if(cd.SizeAltData.noOfNDBNodes > MAX_NDB_NODES){
+ char buf[255];
+ snprintf(buf, sizeof(buf),
+ "Maximum %d ndb nodes is allowed in the cluster",
+ MAX_NDB_NODES);
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, buf);
+ }
+ } else if(strcmp("API", nodeType) == 0){
+ cd.nodeData[nodeNo].nodeType = NodeInfo::API;
+ cd.SizeAltData.noOfAPINodes++; // No of API processes
+ tmpApiMgmProperties = "API";
+ } else if(strcmp("REP", nodeType) == 0){
+ cd.nodeData[nodeNo].nodeType = NodeInfo::REP;
+ //cd.SizeAltData.noOfAPINodes++; // No of API processes
+ tmpApiMgmProperties = "REP";
+ } else if(strcmp("MGM", nodeType) == 0){
+ cd.nodeData[nodeNo].nodeType = NodeInfo::MGM;
+ cd.SizeAltData.noOfMGMNodes++; // No of MGM processes
+ tmpApiMgmProperties = "MGM";
+ } else {
+ ERROR_SET(fatal, ERR_INVALID_CONFIG,
+ "Invalid configuration: Unknown node type",
+ nodeType);
+ }
+
+ if (tmpApiMgmProperties) {
+ /*
+ const Properties* q = 0;
+
+ if (!p.get(tmpApiMgmProperties, nodeId, &q)) {
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, msg, tmpApiMgmProperties);
+ } else {
+ */
+ Uint32 rank = 0;
+ if (node->get("ArbitrationRank", &rank) && rank > 0) {
+ cd.nodeData[nodeNo].arbitRank = rank;
+ // }
+ }
+ } else {
+ cd.nodeData[nodeNo].arbitRank = 0;
+ }
+
+ nodeNo++;
+ }
+ }
+ cd.SizeAltData.exist = true;
+ calcSizeAlteration();
+}
+
+
diff --git a/ndb/src/kernel/vm/ClusterConfiguration.hpp b/ndb/src/kernel/vm/ClusterConfiguration.hpp
new file mode 100644
index 00000000000..cc7000a54ef
--- /dev/null
+++ b/ndb/src/kernel/vm/ClusterConfiguration.hpp
@@ -0,0 +1,105 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ClusterConfiguration_H
+#define ClusterConfiguration_H
+
+#include <kernel_types.h>
+#include <ndb_limits.h>
+#include <Properties.hpp>
+#include <ErrorReporter.hpp>
+#include <signaldata/CmvmiCfgConf.hpp>
+#include <signaldata/SetLogLevelOrd.hpp>
+#include <NodeInfo.hpp>
+
+// MaxNumber of sizealteration records in each block
+// MaxNumber of blocks with sizealteration, (size of array)
+#define MAX_SIZEALT_RECORD 16
+#define MAX_SIZEALT_BLOCKS 8
+
+enum NdbBlockName { ACC = 0, DICT, DIH, LQH, TC, TUP, TUX, NDB_SIZEALT_OFF };
+// NDB_SIZEALT_OFF is used for block without sizealteration
+// IMPORTANT to assign NDB_SIZEALT_OFF as largest value
+
+struct VarSize {
+ int nrr;
+ bool valid;
+};
+
+struct SizeAlt {
+ unsigned int noOfTables;
+ unsigned int noOfIndexes;
+ unsigned int noOfReplicas;
+ unsigned int noOfNDBNodes;
+ unsigned int noOfAPINodes;
+ unsigned int noOfMGMNodes;
+ unsigned int noOfNodes;
+ unsigned int noOfDiskLessNodes;
+ unsigned int noOfAttributes;
+ unsigned int noOfOperations;
+ unsigned int noOfTransactions;
+ unsigned int noOfIndexPages;
+ unsigned int noOfDataPages;
+ unsigned int noOfDiskBufferPages;
+ unsigned int noOfFreeClusters;
+ unsigned int noOfDiskClusters;
+ unsigned int noOfScanRecords;
+ bool exist;
+ VarSize varSize[MAX_SIZEALT_BLOCKS][MAX_SIZEALT_RECORD];
+ unsigned short blockNo[MAX_SIZEALT_BLOCKS];
+ LogLevel logLevel;
+};
+
+
+class ClusterConfiguration
+{
+public:
+
+ struct NodeData {
+ NodeData() {
+ nodeId = MAX_NODES+1;
+ nodeType = NodeInfo::INVALID;
+ arbitRank = ~0;
+ }
+ NodeId nodeId;
+ NodeInfo::NodeType nodeType;
+ unsigned arbitRank;
+ };
+
+ struct ClusterData
+ {
+ SizeAlt SizeAltData;
+ NodeData nodeData[MAX_NODES];
+ Uint32 ispValues[5][CmvmiCfgConf::NO_OF_WORDS];
+ };
+
+ ClusterConfiguration();
+ ~ClusterConfiguration();
+ const ClusterData& clusterData() const;
+
+ void init(const Properties & p, const Properties & db);
+protected:
+
+private:
+
+ ClusterData the_clusterData;
+
+ void calcSizeAlteration();
+
+};
+
+#endif // ClusterConfiguration_H
+
diff --git a/ndb/src/kernel/vm/Configuration.cpp b/ndb/src/kernel/vm/Configuration.cpp
new file mode 100644
index 00000000000..0b680940105
--- /dev/null
+++ b/ndb/src/kernel/vm/Configuration.cpp
@@ -0,0 +1,338 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Configuration.hpp"
+#include <ErrorHandlingMacros.hpp>
+#include "GlobalData.hpp"
+
+#include <ConfigRetriever.hpp>
+#include <IPCConfig.hpp>
+#include <ndb_version.h>
+#include <NdbMem.h>
+#include <NdbOut.hpp>
+#include <WatchDog.hpp>
+#include <NdbString.h>
+
+#include <getarg.h>
+
+extern "C" {
+ void ndbSetOwnVersion();
+}
+
+bool
+Configuration::init(int argc, const char** argv){
+
+ /**
+ * Default values for arguments
+ */
+ int _start = 1;
+ int _initial = 0;
+ const char* _connect_str = NULL;
+ int _deamon = 0;
+ int _help = 0;
+ int _print_version = 0;
+
+ /**
+ * Arguments to NDB process
+ */
+ struct getargs args[] = {
+ { "version", 'v', arg_flag, &_print_version, "Print version", "" },
+ { "start", 's', arg_flag, &_start, "Start ndb immediately", "" },
+ { "nostart", 'n', arg_negative_flag, &_start, "Don't start ndb immediately", "" },
+ { "deamon", 'd', arg_flag, &_deamon, "Start ndb as deamon", "" },
+ { "initial", 'i', arg_flag, &_initial, "Start ndb immediately", "" },
+
+ { "connect-string", 'c', arg_string, &_connect_str, "\"nodeid=<id>;host=<hostname:port>\"\n", "constr" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "The MySQL Cluster kernel";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return false;
+ }
+
+#if 0
+ ndbout << "start=" <<_start<< endl;
+ ndbout << "initial=" <<_initial<< endl;
+ ndbout << "deamon=" <<_deamon<< endl;
+ ndbout << "connect_str="<<_connect_str<<endl;
+ arg_printusage(args, num_args, argv[0], desc);
+ return false;
+#endif
+
+ ndbSetOwnVersion();
+
+ if (_print_version) {
+ ndbPrintVersion();
+ return false;
+ }
+
+ // Check the start flag
+ if (_start)
+ globalData.theRestartFlag = perform_start;
+ else
+ globalData.theRestartFlag = initial_state;
+
+ // Check the initial flag
+ if (_initial)
+ _initialStart = true;
+
+ // Check connectstring
+ if (_connect_str){
+
+ if(_connect_str[0] == '-' ||
+ strstr(_connect_str, "host") == 0 ||
+ strstr(_connect_str, "nodeid") == 0) {
+ ndbout << "Illegal/empty connectString: " << _connect_str << endl;
+ arg_printusage(args, num_args, argv[0], desc);
+ return false;
+ }
+ _connectString = strdup(_connect_str);
+ }
+
+ // Check deamon flag
+ if (_deamon)
+ _daemonMode = true;
+
+ // Save programname
+ if(argc > 0 && argv[0] != 0)
+ _programName = strdup(argv[0]);
+ else
+ _programName = strdup("");
+
+ return true;
+}
+
+Configuration::Configuration():
+ the_clusterConfigurationData()
+{
+ m_ownProperties = 0;
+ _programName = 0;
+ _connectString = 0;
+ _fsPath = 0;
+ _initialStart = false;
+ _daemonMode = false;
+}
+
+Configuration::~Configuration(){
+ delete m_ownProperties;
+
+ if(_programName != NULL)
+ free(_programName);
+
+ if(_fsPath != NULL)
+ free(_fsPath);
+}
+
+const
+ClusterConfiguration&
+Configuration::clusterConfiguration() const {
+ return the_clusterConfigurationData;
+}
+
+void
+Configuration::setupConfiguration(){
+ /**
+ * Fetch configuration from management server
+ */
+ ConfigRetriever cr;
+ cr.setConnectString(_connectString);
+ stopOnError(true);
+ Properties * p = cr.getConfig("DB", NDB_VERSION);
+ if(p == 0){
+ const char * s = cr.getErrorString();
+ if(s == 0)
+ s = "No error given!";
+
+ /* Set stop on error to true otherwise NDB will
+ go into an restart loop...
+ */
+
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Could not fetch configuration"
+ "/invalid configuration", s);
+ }
+
+ /**
+ * Configure transporters
+ */
+ {
+ IPCConfig * theIPC = new IPCConfig(p);
+
+ if(theIPC->init() != 0){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched", "");
+ }
+
+ if(theIPC->configureTransporters(&globalTransporterRegistry) <= 0){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched",
+ "No transporters configured");
+ }
+
+ globalData.ownId = theIPC->ownId();
+ delete theIPC;
+ }
+
+ /**
+ * Setup cluster configuration data
+ */
+ const Properties * db = 0;
+ if (!p->get("Node", globalData.ownId, &db)) {
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched", "DB missing");
+ }
+ const char * type;
+ if(!(db->get("Type", &type) && strcmp(type, "DB") == 0)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched",
+ "I'm wrong type of node");
+ }
+
+ /**
+ * Save properties object to use in getOwnProperties()
+ */
+ m_ownProperties = new Properties(* db);
+
+ the_clusterConfigurationData.init(* p, * db);
+
+ if(!db->get("MaxNoOfSavedMessages", &_maxErrorLogs)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched",
+ "MaxNoOfSavedMessages missing");
+ }
+
+ if(!db->get("LockPagesInMainMemory", &_lockPagesInMainMemory)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched",
+ "LockPagesInMainMemory missing");
+ }
+
+ if(!db->get("TimeBetweenWatchDogCheck", &_timeBetweenWatchDogCheck)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched",
+ "TimeBetweenWatchDogCheck missing");
+ }
+
+ /**
+ * Get filesystem path
+ */
+ {
+ const char* pFileSystemPath = NULL;
+ if(!db->get("FileSystemPath", &pFileSystemPath)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched",
+ "FileSystemPath missing");
+ }
+
+ if(pFileSystemPath == 0 || strlen(pFileSystemPath) == 0){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched",
+ "Configuration does not contain valid filesystem path");
+ }
+
+ if(pFileSystemPath[strlen(pFileSystemPath) - 1] == '/')
+ _fsPath = strdup(pFileSystemPath);
+ else {
+ _fsPath = (char *)malloc(strlen(pFileSystemPath) + 2);
+ strcpy(_fsPath, pFileSystemPath);
+ strcat(_fsPath, "/");
+ }
+ }
+
+ if(!db->get("StopOnError", &_stopOnError)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched",
+ "StopOnError missing");
+ }
+
+ if(!db->get("RestartOnErrorInsert", &m_restartOnErrorInsert)){
+ ERROR_SET(fatal, ERR_INVALID_CONFIG, "Invalid configuration fetched",
+ "RestartOnErrorInsert missing");
+ }
+
+ delete p;
+
+ if (_lockPagesInMainMemory) {
+ NdbMem_MemLockAll();
+ }
+
+ /**
+ * Create the watch dog thread
+ */
+ {
+ Uint32 t = _timeBetweenWatchDogCheck;
+ t = globalEmulatorData.theWatchDog ->setCheckInterval(t);
+ _timeBetweenWatchDogCheck = t;
+ }
+
+}
+
+bool
+Configuration::lockPagesInMainMemory() const {
+ return _lockPagesInMainMemory;
+}
+
+int
+Configuration::timeBetweenWatchDogCheck() const {
+ return _timeBetweenWatchDogCheck;
+}
+
+const
+ClusterConfiguration::ClusterData&
+Configuration::clusterConfigurationData() const {
+ return the_clusterConfigurationData.clusterData();
+}
+
+void
+Configuration::timeBetweenWatchDogCheck(int value) {
+ _timeBetweenWatchDogCheck = value;
+}
+
+int
+Configuration::maxNoOfErrorLogs() const {
+ return _maxErrorLogs;
+}
+
+void
+Configuration::maxNoOfErrorLogs(int val){
+ _maxErrorLogs = val;
+}
+
+bool
+Configuration::stopOnError() const {
+ return _stopOnError;
+}
+
+void
+Configuration::stopOnError(bool val){
+ _stopOnError = val;
+}
+
+const Properties *
+Configuration::getOwnProperties() const {
+ return m_ownProperties;
+}
+
+int
+Configuration::getRestartOnErrorInsert() const {
+ return m_restartOnErrorInsert;
+}
+
+void
+Configuration::setRestartOnErrorInsert(int i){
+ m_restartOnErrorInsert = i;
+}
+
+char *
+Configuration::getConnectStringCopy() const {
+ if(_connectString != 0)
+ return strdup(_connectString);
+ return 0;
+}
diff --git a/ndb/src/kernel/vm/Configuration.hpp b/ndb/src/kernel/vm/Configuration.hpp
new file mode 100644
index 00000000000..e7e3a125394
--- /dev/null
+++ b/ndb/src/kernel/vm/Configuration.hpp
@@ -0,0 +1,112 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef Configuration_H
+#define Configuration_H
+
+#include "ClusterConfiguration.hpp"
+
+class Configuration {
+public:
+ Configuration();
+ ~Configuration();
+
+ /**
+ * Returns false if arguments are invalid
+ */
+ bool init(int argc, const char** argv);
+
+ void setupConfiguration();
+
+ bool lockPagesInMainMemory() const;
+
+ int timeBetweenWatchDogCheck() const ;
+ void timeBetweenWatchDogCheck(int value);
+
+ int maxNoOfErrorLogs() const ;
+ void maxNoOfErrorLogs(int val);
+
+ bool stopOnError() const;
+ void stopOnError(bool val);
+
+ int getRestartOnErrorInsert() const;
+ void setRestartOnErrorInsert(int);
+
+ // Cluster configuration
+ const ClusterConfiguration::ClusterData& clusterConfigurationData() const;
+ const ClusterConfiguration& clusterConfiguration() const;
+
+ const char * programName() const;
+ const char * fileSystemPath() const;
+ char * getConnectStringCopy() const;
+
+ /**
+ * Return Properties for own node
+ */
+ const Properties * getOwnProperties() const;
+
+ /**
+ *
+ */
+ bool getInitialStart() const;
+ bool getDaemonMode() const;
+
+private:
+ Uint32 _stopOnError;
+ Uint32 m_restartOnErrorInsert;
+ Uint32 _maxErrorLogs;
+ Uint32 _lockPagesInMainMemory;
+ Uint32 _timeBetweenWatchDogCheck;
+
+
+ ClusterConfiguration the_clusterConfigurationData;
+ const Properties * m_ownProperties;
+
+ /**
+ * arguments to NDB process
+ */
+ char * _programName;
+ char * _fsPath;
+ bool _initialStart;
+ char * _connectString;
+ bool _daemonMode;
+};
+
+inline
+const char *
+Configuration::programName() const {
+ return _programName;
+}
+
+inline
+const char *
+Configuration::fileSystemPath() const {
+ return _fsPath;
+}
+
+inline
+bool
+Configuration::getInitialStart() const {
+ return _initialStart;
+}
+
+inline
+bool
+Configuration::getDaemonMode() const {
+ return _daemonMode;
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/DLFifoList.hpp b/ndb/src/kernel/vm/DLFifoList.hpp
new file mode 100644
index 00000000000..0b40d00f56e
--- /dev/null
+++ b/ndb/src/kernel/vm/DLFifoList.hpp
@@ -0,0 +1,348 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DLFIFOLIST_HPP
+#define DLFIFOLIST_HPP
+
+#include "ArrayPool.hpp"
+#include <NdbOut.hpp>
+
+/**
+ * Template class used for implementing an
+ * list of object retreived from a pool
+ */
+template <class T>
+class DLFifoList {
+public:
+ /**
+ * List head
+ */
+ struct Head {
+ Head();
+ Uint32 firstItem;
+ Uint32 lastItem;
+ };
+
+ DLFifoList(ArrayPool<T> & thePool);
+
+ /**
+ * Allocate an object from pool - update Ptr
+ *
+ * Return i
+ */
+ bool seize(Ptr<T> &);
+
+ /**
+ * Allocate object <b>i</b> from pool - update Ptr
+ *
+ * Return i
+ */
+ bool seizeId(Ptr<T> &, Uint32 i);
+
+ /**
+ * Add object to list
+ *
+ * @NOTE MUST be seized from correct pool
+ */
+ void add(Ptr<T> &);
+
+ /**
+ * Return an object to pool
+ */
+ void release(Uint32 i);
+
+ /**
+ * Return an object to pool
+ */
+ void release(Ptr<T> &);
+
+ /**
+ * Return all objects to the pool
+ */
+ void release();
+
+ /**
+ * Update i & p value according to <b>i</b>
+ */
+ void getPtr(Ptr<T> &, Uint32 i) const;
+
+ /**
+ * Update p value for ptr according to i value
+ */
+ void getPtr(Ptr<T> &) const ;
+
+ /**
+ * Get pointer for i value
+ */
+ T * getPtr(Uint32 i) const ;
+
+ /**
+ * Update ptr to first element in list
+ *
+ * Return i
+ */
+ bool first(Ptr<T> &) const ;
+
+ /**
+ * Get next element
+ *
+ * NOTE ptr must be both p & i
+ */
+ bool next(Ptr<T> &) const ;
+
+ /**
+ * Check if next exists
+ *
+ * NOTE ptr must be both p & i
+ */
+ bool hasNext(const Ptr<T> &) const;
+
+ Uint32 noOfElements() const {
+ Uint32 c = 0;
+ Uint32 i = head.firstItem;
+ while(i != RNIL){
+ c++;
+ const T * t = thePool.getPtr(i);
+ i = t->nextList;
+ }
+ return c;
+ }
+
+ /**
+ * Print
+ * (Run operator NdbOut<< on every element)
+ */
+ void print(NdbOut & out) {
+ Uint32 i = head.firstItem;
+ while(i != RNIL){
+ T * t = thePool.getPtr(i);
+ out << (unsigned int) t << "[" << i << "]:";
+ t->print(out); out << " ";
+ i = t->nextList;
+ }
+ }
+
+ inline bool isEmpty() const { return head.firstItem == RNIL;}
+
+protected:
+ Head head;
+ ArrayPool<T> & thePool;
+};
+
+template<class T>
+class LocalDLFifoList : public DLFifoList<T> {
+public:
+ LocalDLFifoList(ArrayPool<T> & thePool, typename DLFifoList<T>::Head & _src)
+ : DLFifoList<T>(thePool), src(_src)
+ {
+ head = src;
+ }
+
+ ~LocalDLFifoList(){
+ src = head;
+ }
+private:
+ typename DLFifoList<T>::Head & src;
+};
+
+template <class T>
+inline
+DLFifoList<T>::DLFifoList(ArrayPool<T> & _pool):
+ thePool(_pool){
+}
+
+template<class T>
+inline
+DLFifoList<T>::Head::Head(){
+ firstItem = RNIL;
+ lastItem = RNIL;
+}
+
+/**
+ * Allocate an object from pool - update Ptr
+ *
+ * Return i
+ */
+template <class T>
+inline
+bool
+DLFifoList<T>::seize(Ptr<T> & p){
+ thePool.seize(p);
+ if (p.i != RNIL) {
+ add(p);
+ return true;
+ }
+ p.p = NULL;
+ return false;
+}
+
+/**
+ * Allocate an object from pool - update Ptr
+ *
+ * Return i
+ */
+template <class T>
+inline
+bool
+DLFifoList<T>::seizeId(Ptr<T> & p, Uint32 ir){
+ thePool.seizeId(p, ir);
+ if(p.i != RNIL){
+ add(p);
+ return true;
+ }
+ p.p = NULL;
+ return false;
+}
+
+template <class T>
+inline
+void
+DLFifoList<T>::add(Ptr<T> & p){
+ T * t = p.p;
+ Uint32 last = head.lastItem;
+
+ if(p.i == RNIL)
+ ErrorReporter::handleAssert("DLFifoList<T>::add", __FILE__, __LINE__);
+
+ t->nextList = RNIL;
+ t->prevList = last;
+ if (head.firstItem == RNIL)
+ head.firstItem = p.i;
+ head.lastItem = p.i;
+
+ if(last != RNIL){
+ T * t2 = thePool.getPtr(last);
+ t2->nextList = p.i;
+ }
+}
+
+/**
+ * Return an object to pool
+ */
+template <class T>
+inline
+void
+DLFifoList<T>::release(Uint32 i){
+ Ptr<T> p;
+ p.i = i;
+ p.p = thePool.getPtr(i);
+ release(p);
+}
+
+/**
+ * Return an object to pool
+ */
+template <class T>
+inline
+void
+DLFifoList<T>::release(Ptr<T> & p){
+ T * t = p.p;
+ Uint32 ni = t->nextList;
+ Uint32 pi = t->prevList;
+
+ if(ni != RNIL){
+ T * t = thePool.getPtr(ni);
+ t->prevList = pi;
+ } else {
+ // We are releasing last
+ head.lastItem = pi;
+ }
+
+ if(pi != RNIL){
+ T * t = thePool.getPtr(pi);
+ t->nextList = ni;
+ } else {
+ // We are releasing first
+ head.firstItem = ni;
+ }
+ thePool.release(p.i);
+}
+
+template <class T>
+inline
+void
+DLFifoList<T>::release(){
+ Ptr<T> p;
+ while(head.firstItem != RNIL){
+ p.i = head.firstItem;
+ p.p = thePool.getPtr(head.firstItem);
+ T * t = p.p;
+ head.firstItem = t->nextList;
+ release(p);
+ }
+}
+
+template <class T>
+inline
+void
+DLFifoList<T>::getPtr(Ptr<T> & p, Uint32 i) const {
+ p.i = i;
+ p.p = thePool.getPtr(i);
+}
+
+template <class T>
+inline
+void
+DLFifoList<T>::getPtr(Ptr<T> & p) const {
+ thePool.getPtr(p);
+}
+
+template <class T>
+inline
+T *
+DLFifoList<T>::getPtr(Uint32 i) const {
+ return thePool.getPtr(i);
+}
+
+/**
+ * Update ptr to first element in list
+ *
+ * Return i
+ */
+template <class T>
+inline
+bool
+DLFifoList<T>::first(Ptr<T> & p) const {
+ p.i = head.firstItem;
+ if(p.i != RNIL){
+ p.p = thePool.getPtr(p.i);
+ return true;
+ }
+ p.p = NULL;
+ return false;
+}
+
+template <class T>
+inline
+bool
+DLFifoList<T>::next(Ptr<T> & p) const {
+ p.i = p.p->nextList;
+ if(p.i != RNIL){
+ p.p = thePool.getPtr(p.i);
+ return true;
+ }
+ p.p = NULL;
+ return false;
+}
+
+template <class T>
+inline
+bool
+DLFifoList<T>::hasNext(const Ptr<T> & p) const {
+ return p.p->nextList != RNIL;
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/DLHashTable.hpp b/ndb/src/kernel/vm/DLHashTable.hpp
new file mode 100644
index 00000000000..f7cd7ae5228
--- /dev/null
+++ b/ndb/src/kernel/vm/DLHashTable.hpp
@@ -0,0 +1,494 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DL_HASHTABLE_HPP
+#define DL_HASHTABLE_HPP
+
+#include "ArrayList.hpp"
+#include <assert.h>
+#include <stddef.h>
+
+/**
+ * DLHashTable implements a hashtable using chaining
+ * (with a double linked list)
+ *
+ * The entries in the hashtable must have the following methods:
+ * -# bool equal(const class T &) const;
+ * Which should return equal if the to objects have the same key
+ * -# Uint32 hashValue() const;
+ * Which should return a 32 bit hashvalue
+ */
+template <class T>
+class DLHashTable {
+public:
+ DLHashTable(ArrayPool<T> & thePool);
+ ~DLHashTable();
+
+ /**
+ * Set the no of bucket in the hashtable
+ *
+ * Note, can currently only be called once
+ */
+ bool setSize(Uint32 noOfElements);
+
+ /**
+ * Seize element from pool - return i
+ *
+ * Note must be either added using <b>add</b> or released
+ * using <b>release</b>
+ */
+ bool seize(Ptr<T> &);
+
+ /**
+ * Add an object to the hashtable
+ */
+ void add(Ptr<T> &);
+
+ /**
+ * Find element key in hashtable update Ptr (i & p)
+ * (using key.equal(...))
+ * @return true if found and false otherwise
+ */
+ bool find(Ptr<T> &, const T & key) const;
+
+ /**
+ * Update i & p value according to <b>i</b>
+ */
+ void getPtr(Ptr<T> &, Uint32 i) const;
+
+ /**
+ * Get element using ptr.i (update ptr.p)
+ */
+ void getPtr(Ptr<T> &) const;
+
+ /**
+ * Get P value for i
+ */
+ T * getPtr(Uint32 i) const;
+
+ /**
+ * Remove element (and set Ptr to removed element)
+ * Note does not return to pool
+ */
+ void remove(Ptr<T> &, const T & key);
+
+ /**
+ * Remove element
+ * Note does not return to pool
+ */
+ void remove(Uint32 i);
+
+ /**
+ * Remove element
+ * Note does not return to pool
+ */
+ void remove(Ptr<T> &);
+
+ /**
+ * Remove all elements, but dont return them to pool
+ */
+ void removeAll();
+
+ /**
+ * Remove element (and set Ptr to removed element)
+ * And return element to pool
+ */
+ void release(Ptr<T> &, const T & key);
+
+ /**
+ * Remove element and return to pool
+ */
+ void release(Uint32 i);
+
+ /**
+ * Remove element and return to pool
+ */
+ void release(Ptr<T> &);
+
+ class Iterator {
+ public:
+ Ptr<T> curr;
+ Uint32 bucket;
+ inline bool isNull() const { return curr.isNull();}
+ inline void setNull() { curr.setNull(); }
+ };
+
+ /**
+ * Sets curr.p according to curr.i
+ */
+ void getPtr(Iterator & iter) const ;
+
+ /**
+ * First element in bucket
+ */
+ bool first(Iterator & iter) const;
+
+ /**
+ * Next Element
+ *
+ * param iter - A "fully set" iterator
+ */
+ bool next(Iterator & iter) const;
+
+ /**
+ * Get next element starting from bucket
+ *
+ * @param bucket - Which bucket to start from
+ * @param iter - An "uninitialized" iterator
+ */
+ bool next(Uint32 bucket, Iterator & iter) const;
+
+private:
+ Uint32 mask;
+ Uint32 * hashValues;
+ ArrayPool<T> & thePool;
+};
+
+template<class T>
+inline
+DLHashTable<T>::DLHashTable(ArrayPool<T> & _pool)
+ : thePool(_pool)
+{
+ mask = 0;
+ hashValues = 0;
+}
+
+template<class T>
+inline
+DLHashTable<T>::~DLHashTable(){
+ if(hashValues != 0)
+ delete [] hashValues;
+}
+
+template<class T>
+inline
+bool
+DLHashTable<T>::setSize(Uint32 size){
+ Uint32 i = 1;
+ while(i < size) i *= 2;
+
+ if(mask == (i - 1)){
+ /**
+ * The size is already set to <b>size</b>
+ */
+ return true;
+ }
+
+ if(mask != 0){
+ /**
+ * The mask is already set
+ */
+ return false;
+ }
+
+ mask = (i - 1);
+ hashValues = new Uint32[i];
+ for(Uint32 j = 0; j<i; j++)
+ hashValues[j] = RNIL;
+
+ return true;
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::add(Ptr<T> & obj){
+ const Uint32 hv = obj.p->hashValue() & mask;
+ const Uint32 i = hashValues[hv];
+
+ if(i == RNIL){
+ hashValues[hv] = obj.i;
+ obj.p->nextHash = RNIL;
+ obj.p->prevHash = RNIL;
+ } else {
+
+ T * tmp = thePool.getPtr(i);
+ tmp->prevHash = obj.i;
+ obj.p->nextHash = i;
+ obj.p->prevHash = RNIL;
+
+ hashValues[hv] = obj.i;
+ }
+}
+
+/**
+ * First element
+ */
+template<class T>
+inline
+bool
+DLHashTable<T>::first(Iterator & iter) const {
+ Uint32 i = 0;
+ while(i <= mask && hashValues[i] == RNIL) i++;
+ if(i <= mask){
+ iter.bucket = i;
+ iter.curr.i = hashValues[i];
+ iter.curr.p = thePool.getPtr(iter.curr.i);
+ return true;
+ } else {
+ iter.curr.i = RNIL;
+ }
+ return false;
+}
+
+template<class T>
+inline
+bool
+DLHashTable<T>::next(Iterator & iter) const {
+ if(iter.curr.p->nextHash == RNIL){
+ Uint32 i = iter.bucket + 1;
+ while(i <= mask && hashValues[i] == RNIL) i++;
+ if(i <= mask){
+ iter.bucket = i;
+ iter.curr.i = hashValues[i];
+ iter.curr.p = thePool.getPtr(iter.curr.i);
+ return true;
+ } else {
+ iter.curr.i = RNIL;
+ return false;
+ }
+ }
+
+ iter.curr.i = iter.curr.p->nextHash;
+ iter.curr.p = thePool.getPtr(iter.curr.i);
+ return true;
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::remove(Ptr<T> & ptr, const T & key){
+ const Uint32 hv = key.hashValue() & mask;
+
+ Uint32 i;
+ T * p;
+ Ptr<T> prev;
+ prev.i = RNIL;
+
+ i = hashValues[hv];
+ while(i != RNIL){
+ p = thePool.getPtr(i);
+ if(key.equal(* p)){
+ const Uint32 next = p->nextHash;
+ if(prev.i == RNIL){
+ hashValues[hv] = next;
+ } else {
+ prev.p->nextHash = next;
+ }
+
+ if(next != RNIL){
+ T * nextP = thePool.getPtr(next);
+ nextP->prevHash = prev.i;
+ }
+
+ ptr.i = i;
+ ptr.p = p;
+ return;
+ }
+ prev.p = p;
+ prev.i = i;
+ i = p->nextHash;
+ }
+ ptr.i = RNIL;
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::release(Ptr<T> & ptr, const T & key){
+ const Uint32 hv = key.hashValue() & mask;
+
+ Uint32 i;
+ T * p;
+ Ptr<T> prev;
+ prev.i = RNIL;
+
+ i = hashValues[hv];
+ while(i != RNIL){
+ p = thePool.getPtr(i);
+ if(key.equal(* p)){
+ const Uint32 next = p->nextHash;
+ if(prev.i == RNIL){
+ hashValues[hv] = next;
+ } else {
+ prev.p->nextHash = next;
+ }
+
+ if(next != RNIL){
+ T * nextP = thePool.getPtr(next);
+ nextP->prevHash = prev.i;
+ }
+
+ thePool.release(i);
+ ptr.i = i;
+ ptr.p = p;
+ return;
+ }
+ prev.p = p;
+ prev.i = i;
+ i = p->nextHash;
+ }
+ ptr.i = RNIL;
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::remove(Uint32 i){
+ Ptr<T> tmp;
+ tmp.i = i;
+ tmp.p = thePool.getPtr(i);
+ remove(tmp);
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::release(Uint32 i){
+ Ptr<T> tmp;
+ tmp.i = i;
+ tmp.p = thePool.getPtr(i);
+ release(tmp);
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::remove(Ptr<T> & ptr){
+ const Uint32 next = ptr.p->nextHash;
+ const Uint32 prev = ptr.p->prevHash;
+
+ if(prev != RNIL){
+ T * prevP = thePool.getPtr(prev);
+ prevP->nextHash = next;
+ } else {
+ const Uint32 hv = ptr.p->hashValue() & mask;
+ hashValues[hv] = next;
+ }
+
+ if(next != RNIL){
+ T * nextP = thePool.getPtr(next);
+ nextP->prevHash = prev;
+ }
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::release(Ptr<T> & ptr){
+ const Uint32 next = ptr.p->nextHash;
+ const Uint32 prev = ptr.p->prevHash;
+
+ if(prev != RNIL){
+ T * prevP = thePool.getPtr(prev);
+ prevP->nextHash = next;
+ } else {
+ const Uint32 hv = ptr.p->hashValue() & mask;
+ hashValues[hv] = next;
+ }
+
+ if(next != RNIL){
+ T * nextP = thePool.getPtr(next);
+ nextP->prevHash = prev;
+ }
+
+ thePool.release(ptr.i);
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::removeAll(){
+ for(Uint32 i = 0; i<=mask; i++)
+ hashValues[i] = RNIL;
+}
+
+template<class T>
+inline
+bool
+DLHashTable<T>::next(Uint32 bucket, Iterator & iter) const {
+ while (bucket <= mask && hashValues[bucket] == RNIL)
+ bucket++;
+
+ if (bucket > mask) {
+ iter.bucket = bucket;
+ iter.curr.i = RNIL;
+ return false;
+ }
+
+ iter.bucket = bucket;
+ iter.curr.i = hashValues[bucket];
+ iter.curr.p = thePool.getPtr(iter.curr.i);
+ return true;
+}
+
+template<class T>
+inline
+bool
+DLHashTable<T>::seize(Ptr<T> & ptr){
+ if(thePool.seize(ptr)){
+ ptr.p->nextHash = ptr.p->prevHash = RNIL;
+ return true;
+ }
+ return false;
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::getPtr(Ptr<T> & ptr, Uint32 i) const {
+ ptr.i = i;
+ ptr.p = thePool.getPtr(i);
+}
+
+template<class T>
+inline
+void
+DLHashTable<T>::getPtr(Ptr<T> & ptr) const {
+ thePool.getPtr(ptr);
+}
+
+template<class T>
+inline
+T *
+DLHashTable<T>::getPtr(Uint32 i) const {
+ return thePool.getPtr(i);
+}
+
+template<class T>
+inline
+bool
+DLHashTable<T>::find(Ptr<T> & ptr, const T & key) const {
+ const Uint32 hv = key.hashValue() & mask;
+
+ Uint32 i;
+ T * p;
+
+ i = hashValues[hv];
+ while(i != RNIL){
+ p = thePool.getPtr(i);
+ if(key.equal(* p)){
+ ptr.i = i;
+ ptr.p = p;
+ return true;
+ }
+ i = p->nextHash;
+ }
+ ptr.i = RNIL;
+ ptr.p = NULL;
+ return false;
+}
+#endif
diff --git a/ndb/src/kernel/vm/DLHashTable2.hpp b/ndb/src/kernel/vm/DLHashTable2.hpp
new file mode 100644
index 00000000000..8386790b0a6
--- /dev/null
+++ b/ndb/src/kernel/vm/DLHashTable2.hpp
@@ -0,0 +1,501 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DL_HASHTABLE2_HPP
+#define DL_HASHTABLE2_HPP
+
+#include "ArrayList.hpp"
+#include <assert.h>
+#include <stddef.h>
+#include <new>
+
+/**
+ * DLHashTable2 is a DLHashTable variant meant for cases where different
+ * DLHashTable instances share a common pool (based on a union U).
+ *
+ * Calls T constructor after seize from pool and T destructor before
+ * release (in all forms) into pool.
+ */
+template <class T, class U>
+class DLHashTable2 {
+public:
+ DLHashTable2(ArrayPool<U> & thePool);
+ ~DLHashTable2();
+
+ /**
+ * Set the no of bucket in the hashtable
+ *
+ * Note, can currently only be called once
+ */
+ bool setSize(Uint32 noOfElements);
+
+ /**
+ * Seize element from pool - return i
+ *
+ * Note must be either added using <b>add</b> or released
+ * using <b>release</b>
+ */
+ bool seize(Ptr<T> &);
+
+ /**
+ * Add an object to the hashtable
+ */
+ void add(Ptr<T> &);
+
+ /**
+ * Find element key in hashtable update Ptr (i & p)
+ * (using key.equal(...))
+ * @return true if found and false otherwise
+ */
+ bool find(Ptr<T> &, const T & key) const;
+
+ /**
+ * Update i & p value according to <b>i</b>
+ */
+ void getPtr(Ptr<T> &, Uint32 i) const;
+
+ /**
+ * Get element using ptr.i (update ptr.p)
+ */
+ void getPtr(Ptr<T> &) const;
+
+ /**
+ * Get P value for i
+ */
+ T * getPtr(Uint32 i) const;
+
+ /**
+ * Remove element (and set Ptr to removed element)
+ * Note does not return to pool
+ */
+ void remove(Ptr<T> &, const T & key);
+
+ /**
+ * Remove element
+ * Note does not return to pool
+ */
+ void remove(Uint32 i);
+
+ /**
+ * Remove element
+ * Note does not return to pool
+ */
+ void remove(Ptr<T> &);
+
+ /**
+ * Remove all elements, but dont return them to pool
+ */
+ void removeAll();
+
+ /**
+ * Remove element (and set Ptr to removed element)
+ * And return element to pool
+ */
+ void release(Ptr<T> &, const T & key);
+
+ /**
+ * Remove element and return to pool
+ */
+ void release(Uint32 i);
+
+ /**
+ * Remove element and return to pool
+ */
+ void release(Ptr<T> &);
+
+ class Iterator {
+ public:
+ Ptr<T> curr;
+ Uint32 bucket;
+ inline bool isNull() const { return curr.isNull();}
+ inline void setNull() { curr.setNull(); }
+ };
+
+ /**
+ * Sets curr.p according to curr.i
+ */
+ void getPtr(Iterator & iter) const ;
+
+ /**
+ * First element in bucket
+ */
+ bool first(Iterator & iter) const;
+
+ /**
+ * Next Element
+ *
+ * param iter - A "fully set" iterator
+ */
+ bool next(Iterator & iter) const;
+
+ /**
+ * Get next element starting from bucket
+ *
+ * @param bucket - Which bucket to start from
+ * @param iter - An "uninitialized" iterator
+ */
+ bool next(Uint32 bucket, Iterator & iter) const;
+
+private:
+ Uint32 mask;
+ Uint32 * hashValues;
+ ArrayPool<U> & thePool;
+};
+
+template<class T, class U>
+inline
+DLHashTable2<T, U>::DLHashTable2(ArrayPool<U> & _pool)
+ : thePool(_pool)
+{
+ mask = 0;
+ hashValues = 0;
+}
+
+template<class T, class U>
+inline
+DLHashTable2<T, U>::~DLHashTable2(){
+ if(hashValues != 0)
+ delete [] hashValues;
+}
+
+template<class T, class U>
+inline
+bool
+DLHashTable2<T, U>::setSize(Uint32 size){
+ Uint32 i = 1;
+ while(i < size) i *= 2;
+
+ if(mask == (i - 1)){
+ /**
+ * The size is already set to <b>size</b>
+ */
+ return true;
+ }
+
+ if(mask != 0){
+ /**
+ * The mask is already set
+ */
+ return false;
+ }
+
+ mask = (i - 1);
+ hashValues = new Uint32[i];
+ for(Uint32 j = 0; j<i; j++)
+ hashValues[j] = RNIL;
+
+ return true;
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::add(Ptr<T> & obj){
+ const Uint32 hv = obj.p->hashValue() & mask;
+ const Uint32 i = hashValues[hv];
+
+ if(i == RNIL){
+ hashValues[hv] = obj.i;
+ obj.p->nextHash = RNIL;
+ obj.p->prevHash = RNIL;
+ } else {
+
+ T * tmp = (T*)thePool.getPtr(i); // cast
+ tmp->prevHash = obj.i;
+ obj.p->nextHash = i;
+ obj.p->prevHash = RNIL;
+
+ hashValues[hv] = obj.i;
+ }
+}
+
+/**
+ * First element
+ */
+template<class T, class U>
+inline
+bool
+DLHashTable2<T, U>::first(Iterator & iter) const {
+ Uint32 i = 0;
+ while(i <= mask && hashValues[i] == RNIL) i++;
+ if(i <= mask){
+ iter.bucket = i;
+ iter.curr.i = hashValues[i];
+ iter.curr.p = (T*)thePool.getPtr(iter.curr.i); // cast
+ return true;
+ } else {
+ iter.curr.i = RNIL;
+ }
+ return false;
+}
+
+template<class T, class U>
+inline
+bool
+DLHashTable2<T, U>::next(Iterator & iter) const {
+ if(iter.curr.p->nextHash == RNIL){
+ Uint32 i = iter.bucket + 1;
+ while(i <= mask && hashValues[i] == RNIL) i++;
+ if(i <= mask){
+ iter.bucket = i;
+ iter.curr.i = hashValues[i];
+ iter.curr.p = (T*)thePool.getPtr(iter.curr.i); // cast
+ return true;
+ } else {
+ iter.curr.i = RNIL;
+ return false;
+ }
+ }
+
+ iter.curr.i = iter.curr.p->nextHash;
+ iter.curr.p = (T*)thePool.getPtr(iter.curr.i); // cast
+ return true;
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::remove(Ptr<T> & ptr, const T & key){
+ const Uint32 hv = key.hashValue() & mask;
+
+ Uint32 i;
+ T * p;
+ Ptr<T> prev;
+ prev.i = RNIL;
+
+ i = hashValues[hv];
+ while(i != RNIL){
+ p = (T*)thePool.getPtr(i); // cast
+ if(key.equal(* p)){
+ const Uint32 next = p->nextHash;
+ if(prev.i == RNIL){
+ hashValues[hv] = next;
+ } else {
+ prev.p->nextHash = next;
+ }
+
+ if(next != RNIL){
+ T * nextP = (T*)thePool.getPtr(next); // cast
+ nextP->prevHash = prev.i;
+ }
+
+ ptr.i = i;
+ ptr.p = p;
+ return;
+ }
+ prev.p = p;
+ prev.i = i;
+ i = p->nextHash;
+ }
+ ptr.i = RNIL;
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::release(Ptr<T> & ptr, const T & key){
+ const Uint32 hv = key.hashValue() & mask;
+
+ Uint32 i;
+ T * p;
+ Ptr<T> prev;
+ prev.i = RNIL;
+
+ i = hashValues[hv];
+ while(i != RNIL){
+ p = (T*)thePool.getPtr(i); // cast
+ if(key.equal(* p)){
+ const Uint32 next = p->nextHash;
+ if(prev.i == RNIL){
+ hashValues[hv] = next;
+ } else {
+ prev.p->nextHash = next;
+ }
+
+ if(next != RNIL){
+ T * nextP = (T*)thePool.getPtr(next); // cast
+ nextP->prevHash = prev.i;
+ }
+
+ p->~T(); // dtor
+ thePool.release(i);
+ ptr.i = i;
+ ptr.p = p; // invalid
+ return;
+ }
+ prev.p = p;
+ prev.i = i;
+ i = p->nextHash;
+ }
+ ptr.i = RNIL;
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::remove(Uint32 i){
+ Ptr<T> tmp;
+ tmp.i = i;
+ tmp.p = (T*)thePool.getPtr(i); // cast
+ remove(tmp);
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::release(Uint32 i){
+ Ptr<T> tmp;
+ tmp.i = i;
+ tmp.p = (T*)thePool.getPtr(i); // cast
+ release(tmp);
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::remove(Ptr<T> & ptr){
+ const Uint32 next = ptr.p->nextHash;
+ const Uint32 prev = ptr.p->prevHash;
+
+ if(prev != RNIL){
+ T * prevP = (T*)thePool.getPtr(prev); // cast
+ prevP->nextHash = next;
+ } else {
+ const Uint32 hv = ptr.p->hashValue() & mask;
+ hashValues[hv] = next;
+ }
+
+ if(next != RNIL){
+ T * nextP = (T*)thePool.getPtr(next); // cast
+ nextP->prevHash = prev;
+ }
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::release(Ptr<T> & ptr){
+ const Uint32 next = ptr.p->nextHash;
+ const Uint32 prev = ptr.p->prevHash;
+
+ if(prev != RNIL){
+ T * prevP = (T*)thePool.getPtr(prev); // cast
+ prevP->nextHash = next;
+ } else {
+ const Uint32 hv = ptr.p->hashValue() & mask;
+ hashValues[hv] = next;
+ }
+
+ if(next != RNIL){
+ T * nextP = (T*)thePool.getPtr(next); // cast
+ nextP->prevHash = prev;
+ }
+
+ thePool.release(ptr.i);
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::removeAll(){
+ for(Uint32 i = 0; i<=mask; i++)
+ hashValues[i] = RNIL;
+}
+
+template<class T, class U>
+inline
+bool
+DLHashTable2<T, U>::next(Uint32 bucket, Iterator & iter) const {
+ while (bucket <= mask && hashValues[bucket] == RNIL)
+ bucket++;
+
+ if (bucket > mask) {
+ iter.bucket = bucket;
+ iter.curr.i = RNIL;
+ return false;
+ }
+
+ iter.bucket = bucket;
+ iter.curr.i = hashValues[bucket];
+ iter.curr.p = (T*)thePool.getPtr(iter.curr.i); // cast
+ return true;
+}
+
+template<class T, class U>
+inline
+bool
+DLHashTable2<T, U>::seize(Ptr<T> & ptr){
+ Ptr<U> ptr2;
+ thePool.seize(ptr2);
+ ptr.i = ptr2.i;
+ ptr.p = (T*)ptr2.p; // cast
+ if (ptr.p != NULL){
+ ptr.p->nextHash = RNIL;
+ ptr.p->prevHash = RNIL;
+ new (ptr.p) T; // ctor
+ }
+ return !ptr.isNull();
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::getPtr(Ptr<T> & ptr, Uint32 i) const {
+ ptr.i = i;
+ ptr.p = (T*)thePool.getPtr(i); // cast
+}
+
+template<class T, class U>
+inline
+void
+DLHashTable2<T, U>::getPtr(Ptr<T> & ptr) const {
+ Ptr<U> ptr2;
+ thePool.getPtr(ptr2);
+ ptr.i = ptr2.i;
+ ptr.p = (T*)ptr2.p; // cast
+}
+
+template<class T, class U>
+inline
+T *
+DLHashTable2<T, U>::getPtr(Uint32 i) const {
+ return (T*)thePool.getPtr(i); // cast
+}
+
+template<class T, class U>
+inline
+bool
+DLHashTable2<T, U>::find(Ptr<T> & ptr, const T & key) const {
+ const Uint32 hv = key.hashValue() & mask;
+
+ Uint32 i;
+ T * p;
+
+ i = hashValues[hv];
+ while(i != RNIL){
+ p = (T*)thePool.getPtr(i); // cast
+ if(key.equal(* p)){
+ ptr.i = i;
+ ptr.p = p;
+ return true;
+ }
+ i = p->nextHash;
+ }
+ ptr.i = RNIL;
+ ptr.p = NULL;
+ return false;
+}
+#endif
diff --git a/ndb/src/kernel/vm/DLList.hpp b/ndb/src/kernel/vm/DLList.hpp
new file mode 100644
index 00000000000..f16ccd312f7
--- /dev/null
+++ b/ndb/src/kernel/vm/DLList.hpp
@@ -0,0 +1,369 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DLLIST_HPP
+#define DLLIST_HPP
+
+#include "ArrayPool.hpp"
+#include <NdbOut.hpp>
+
+/**
+ * Template class used for implementing an
+ * list of object retreived from a pool
+ */
+template <class T>
+class DLList {
+public:
+ /**
+ * List head
+ */
+ struct Head {
+ Head();
+ Uint32 firstItem;
+ inline bool isEmpty() const { return firstItem == RNIL; }
+ };
+
+ DLList(ArrayPool<T> & thePool);
+
+ /**
+ * Allocate an object from pool - update Ptr
+ *
+ * Return i
+ */
+ bool seize(Ptr<T> &);
+
+ /**
+ * Allocate object <b>i</b> from pool - update Ptr
+ *
+ * Return i
+ */
+ bool seizeId(Ptr<T> &, Uint32 i);
+
+ /**
+ * Check if <b>i</b> is allocated.
+ */
+ bool findId(Uint32 i) const;
+
+ /**
+ * Allocate <b>n</b>objects from pool
+ *
+ * Return i value of first object allocated or RNIL if fails
+ */
+ bool seizeN(Ptr<T> &, Uint32 n);
+
+ /**
+ * Return an object to pool
+ */
+ void release(Uint32 i);
+
+ /**
+ * Return an object to pool
+ */
+ void release(Ptr<T> &);
+
+ /**
+ * Return all objects to the pool
+ */
+ void release();
+
+ /**
+ * Add object to list
+ *
+ * @NOTE MUST be seized from correct pool
+ */
+ void add(Ptr<T> &);
+
+ /**
+ * Remove object from list
+ *
+ * @NOTE Does not return it to pool
+ */
+ void remove(Ptr<T> &);
+
+ /**
+ * Update i & p value according to <b>i</b>
+ */
+ void getPtr(Ptr<T> &, Uint32 i) const;
+
+ /**
+ * Update p value for ptr according to i value
+ */
+ void getPtr(Ptr<T> &) const ;
+
+ /**
+ * Get pointer for i value
+ */
+ T * getPtr(Uint32 i) const ;
+
+ /**
+ * Update ptr to first element in list
+ *
+ * @return True if element exists, false otherwise
+ */
+ bool first(Ptr<T> &) const ;
+
+ /**
+ * Get next element
+ *
+ * @note ptr must have both p & i values
+ *
+ * @return True if element exists, false otherwise
+ */
+ bool next(Ptr<T> &) const ;
+
+ /**
+ * Check if next exists
+ *
+ * @note ptr must have both p & i values
+ * @return True if element exists, false otherwise
+ */
+ bool hasNext(const Ptr<T> &) const;
+
+ Uint32 noOfElements() const {
+ Uint32 c = 0;
+ Uint32 i = head.firstItem;
+ while(i != RNIL){
+ c++;
+ const T * t = thePool.getPtr(i);
+ i = t->nextList;
+ }
+ return c;
+ }
+
+ /**
+ * Print
+ * (Run operator NdbOut<< on every element)
+ */
+ void print(NdbOut & out) {
+ Uint32 i = head.firstItem;
+ while(i != RNIL){
+ T * t = thePool.getPtr(i);
+ t->print(out); out << " ";
+ i = t->nextList;
+ }
+ }
+
+ inline bool isEmpty() const { return head.firstItem == RNIL;}
+
+protected:
+ Head head;
+ ArrayPool<T> & thePool;
+};
+
+template<class T>
+class LocalDLList : public DLList<T> {
+public:
+ LocalDLList(ArrayPool<T> & thePool, typename DLList<T>::Head & _src)
+ : DLList<T>(thePool), src(_src)
+ {
+ head = src;
+ }
+
+ ~LocalDLList(){
+ src = head;
+ }
+private:
+ typename DLList<T>::Head & src;
+};
+
+template <class T>
+inline
+DLList<T>::DLList(ArrayPool<T> & _pool):
+ thePool(_pool){
+}
+
+template<class T>
+inline
+DLList<T>::Head::Head(){
+ firstItem = RNIL;
+}
+
+/**
+ * Allocate an object from pool - update Ptr
+ *
+ * Return i
+ */
+template <class T>
+inline
+bool
+DLList<T>::seize(Ptr<T> & p){
+ if(thePool.seize(p)){
+ add(p);
+ return true;
+ }
+ return false;
+}
+
+/**
+ * Allocate an object from pool - update Ptr
+ *
+ * Return i
+ */
+template <class T>
+inline
+bool
+DLList<T>::seizeId(Ptr<T> & p, Uint32 ir){
+ if(thePool.seizeId(p, ir)){
+ add(p);
+ return true;
+ }
+ return false;
+}
+
+template <class T>
+inline
+bool
+DLList<T>::findId(Uint32 i) const {
+ return thePool.findId(i);
+}
+
+template <class T>
+inline
+void
+DLList<T>::add(Ptr<T> & p){
+ T * t = p.p;
+ Uint32 ff = head.firstItem;
+
+ t->nextList = ff;
+ t->prevList = RNIL;
+ head.firstItem = p.i;
+
+ if(ff != RNIL){
+ T * t2 = thePool.getPtr(ff);
+ t2->prevList = p.i;
+ }
+}
+
+template <class T>
+inline
+void
+DLList<T>::remove(Ptr<T> & p){
+ T * t = p.p;
+ Uint32 ni = t->nextList;
+ Uint32 pi = t->prevList;
+
+ if(ni != RNIL){
+ T * t = thePool.getPtr(ni);
+ t->prevList = pi;
+ }
+
+ if(pi != RNIL){
+ T * t = thePool.getPtr(pi);
+ t->nextList = ni;
+ } else {
+ head.firstItem = ni;
+ }
+}
+
+/**
+ * Return an object to pool
+ */
+template <class T>
+inline
+void
+DLList<T>::release(Uint32 i){
+ Ptr<T> p;
+ p.i = i;
+ p.p = thePool.getPtr(i);
+ release(p);
+}
+
+/**
+ * Return an object to pool
+ */
+template <class T>
+inline
+void
+DLList<T>::release(Ptr<T> & p){
+ remove(p);
+ thePool.release(p.i);
+}
+
+template <class T>
+inline
+void
+DLList<T>::release(){
+ while(head.firstItem != RNIL){
+ const T * t = thePool.getPtr(head.firstItem);
+ const Uint32 i = head.firstItem;
+ head.firstItem = t->nextList;
+ thePool.release(i);
+ }
+}
+
+template <class T>
+inline
+void
+DLList<T>::getPtr(Ptr<T> & p, Uint32 i) const {
+ p.i = i;
+ p.p = thePool.getPtr(i);
+}
+
+template <class T>
+inline
+void
+DLList<T>::getPtr(Ptr<T> & p) const {
+ thePool.getPtr(p);
+}
+
+template <class T>
+inline
+T *
+DLList<T>::getPtr(Uint32 i) const {
+ return thePool.getPtr(i);
+}
+
+/**
+ * Update ptr to first element in list
+ *
+ * Return i
+ */
+template <class T>
+inline
+bool
+DLList<T>::first(Ptr<T> & p) const {
+ Uint32 i = head.firstItem;
+ p.i = i;
+ if(i != RNIL){
+ p.p = thePool.getPtr(i);
+ return true;
+ }
+ p.p = NULL;
+ return false;
+}
+
+template <class T>
+inline
+bool
+DLList<T>::next(Ptr<T> & p) const {
+ Uint32 i = p.p->nextList;
+ p.i = i;
+ if(i != RNIL){
+ p.p = thePool.getPtr(i);
+ return true;
+ }
+ p.p = NULL;
+ return false;
+}
+
+template <class T>
+inline
+bool
+DLList<T>::hasNext(const Ptr<T> & p) const {
+ return p.p->nextList != RNIL;
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/DataBuffer.hpp b/ndb/src/kernel/vm/DataBuffer.hpp
new file mode 100644
index 00000000000..3425fca76a3
--- /dev/null
+++ b/ndb/src/kernel/vm/DataBuffer.hpp
@@ -0,0 +1,532 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DATA_BUFFER_HPP
+#define DATA_BUFFER_HPP
+
+#include "ArrayPool.hpp"
+
+/**
+ * @class DataBuffer
+ * @brief Buffer of data words
+ *
+ * @note The buffer is divided into segments (of size sz)
+ */
+template <Uint32 sz>
+class DataBuffer {
+public:
+ struct Segment {
+ Uint32 nextPool;
+ Uint32 data[sz];
+ NdbOut& print(NdbOut& out){
+ out << "[DataBuffer<" << sz << ">::Segment this="
+ << hex << (Uint32)this << dec << " nextPool= "
+ << nextPool << " ]";
+ return out;
+ }
+ };
+public:
+ typedef ArrayPool<Segment> DataBufferPool;
+
+ /**
+ * Head/anchor for data buffer
+ */
+ struct Head {
+ Head() ;
+
+ Uint32 used; // Words used
+ Uint32 firstItem; // First segment (or RNIL)
+ Uint32 lastItem; // Last segment (or RNIL)
+
+ /**
+ * Get size of databuffer, in words
+ */
+ Uint32 getSize() const { return used;}
+
+ /**
+ * Get segment size in words (template argument)
+ */
+ static Uint32 getSegmentSize() { return sz;}
+ };
+
+ /** Constructor */
+ DataBuffer(DataBufferPool &);
+
+ /** Seize <b>n</b> words, Release */
+ bool seize(Uint32 n);
+ void release();
+
+ /**
+ * Get size of databuffer, in words
+ */
+ Uint32 getSize() const;
+
+ /**
+ * Check if buffer is empty
+ */
+ bool isEmpty() const;
+
+ /**
+ * Get segment size in words (template argument)
+ */
+ static Uint32 getSegmentSize();
+
+ void print(FILE*) const;
+
+ /* ----------------------------------------------------------------------- */
+
+ struct DataBufferIterator {
+ Ptr<Segment> curr; // Ptr to current segment
+ Uint32* data; // Pointer to current data (word)
+ Uint32 ind; // Word index within a segment
+ Uint32 pos; // Absolute word position within DataBuffer
+
+ void print(FILE* out) {
+ fprintf(out, "[DataBufferIterator curr.i=%d, data=%p, ind=%d, pos=%d]\n",
+ curr.i, (void*) data, ind, pos);
+ };
+
+ inline bool isNull() const { return curr.isNull();}
+ inline void setNull() { curr.setNull(); data = 0; ind = pos = RNIL;}
+ };
+
+ struct ConstDataBufferIterator {
+ ConstPtr<Segment> curr;
+ const Uint32 * data;
+ Uint32 ind;
+ Uint32 pos;
+
+ inline bool isNull() const { return curr.isNull();}
+ inline void setNull() { curr.setNull(); data = 0; ind = pos = RNIL;}
+ };
+
+ /**
+ * Iterator
+ * @parameter hops Number of words to jump forward
+ * @note DataBuffer::next returns false if applied to last word.
+ */
+ bool first(DataBufferIterator &);
+ bool next(DataBufferIterator &);
+ bool next(DataBufferIterator &, Uint32 hops);
+ bool nextPool(DataBufferIterator &);
+
+ /**
+ * Set iterator to position
+ */
+ bool position(DataBufferIterator& it, Uint32 pos);
+
+ /** Iterator */
+ bool first(ConstDataBufferIterator &) const;
+ bool next(ConstDataBufferIterator &) const;
+ bool next(ConstDataBufferIterator &, Uint32 hops) const;
+ bool nextPool(ConstDataBufferIterator &) const;
+
+ /**
+ * Returns true if it is possible to store <em>len</em>
+ * no of words at position given in iterator.
+ */
+ bool importable(const DataBufferIterator, Uint32 len);
+
+ /**
+ * Stores <em>len</em> no of words starting at location <em>src</em> in
+ * databuffer at position given in iterator.
+ *
+ * @return true if success, false otherwise.
+ * @note Iterator is not advanced.
+ */
+ bool import(const DataBufferIterator &, const Uint32* src, Uint32 len);
+
+ /**
+ * Increases size with appends <em>len</em> words
+ * @return true if success, false otherwise.
+ */
+ bool append(const Uint32* src, Uint32 len);
+
+protected:
+ Head head;
+ DataBufferPool & thePool;
+
+private:
+ /**
+ * This is NOT a public method, since the intension is that the import
+ * method using iterators will be more effective in the future
+ */
+ bool import(Uint32 pos, const Uint32* src, Uint32 len);
+};
+
+template<Uint32 sz>
+class LocalDataBuffer : public DataBuffer<sz> {
+public:
+ LocalDataBuffer(typename DataBuffer<sz>::DataBufferPool & thePool,
+ typename DataBuffer<sz>::Head & _src)
+ : DataBuffer<sz>(thePool), src(_src)
+ {
+ head = src;
+ }
+
+ ~LocalDataBuffer(){
+ src = head;
+ }
+private:
+ typename DataBuffer<sz>::Head & src;
+};
+
+template<Uint32 sz>
+inline
+DataBuffer<sz>::Head::Head(){
+ used = 0;
+ firstItem = RNIL;
+ lastItem = RNIL;
+}
+
+template<Uint32 sz>
+inline
+bool DataBuffer<sz>::importable(const DataBufferIterator it, Uint32 len){
+ return (it.pos + len < head.used);
+}
+
+template<Uint32 sz>
+inline
+bool DataBuffer<sz>::position(DataBufferIterator& it, Uint32 p){
+
+ // TODO: The current implementation is not the most effective one.
+ // A more effective implementation would start at the current
+ // position of the iterator.
+
+ if(!first(it)){
+ return false;
+ }
+ return next(it, p);
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::import(const DataBufferIterator & it,
+ const Uint32* src, Uint32 len){
+
+#if 0
+ DataBufferIterator it;
+ position(it, _it.pos);
+
+ for(; len > 0; len--){
+ Uint32 s = * src;
+ * it.data = s;
+ next(it);
+ src++;
+ }
+ return true;
+#else
+ Uint32 ind = (it.pos % sz);
+ Uint32 left = sz - ind;
+ Segment * p = it.curr.p;
+
+ while(len > left){
+ memcpy(&p->data[ind], src, 4 * left);
+ src += left;
+ len -= left;
+ ind = 0;
+ left = sz;
+ p = thePool.getPtr(p->nextPool);
+ }
+
+ memcpy(&p->data[ind], src, 4 * len);
+ return true;
+#endif
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::append(const Uint32* src, Uint32 len){
+ if(len == 0)
+ return true;
+
+ Uint32 pos = head.used;
+ if(!seize(len)){
+ return false;
+ }
+ DataBufferIterator it;
+
+ if(position(it, pos) && import(it, src, len)){
+ return true;
+ }
+ abort();
+ return false;
+}
+
+template<Uint32 sz>
+inline
+void DataBuffer<sz>::print(FILE* out) const {
+ fprintf(out, "[DataBuffer used=%d words, segmentsize=%d words",
+ head.used, sz);
+
+ if (head.firstItem == RNIL) {
+ fprintf(out, ": No segments seized.]\n");
+ return;
+ } else {
+ fprintf(out, "\n");
+ }
+
+ Ptr<Segment> ptr;
+ ptr.i = head.firstItem;
+
+ Uint32 acc = 0;
+ for(; ptr.i != RNIL; ){
+ thePool.getPtr(ptr);
+ const Uint32 * rest = ptr.p->data;
+ for(Uint32 i = 0; i<sz; i++){
+ fprintf(out, " H'%.8x", rest[i]);
+ if(acc++ == 6){
+ acc = 0;
+ fprintf(out, "\n");
+ }
+ }
+ ptr.i = ptr.p->nextPool;
+ }
+ fprintf(out, " ]\n");
+}
+
+template<Uint32 sz>
+inline
+DataBuffer<sz>::DataBuffer(DataBufferPool & p) : thePool(p){
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::seize(Uint32 n){
+ Uint32 rest; // Free space in last segment (currently)
+ Segment* prevPtr;
+
+ if(head.firstItem == RNIL){
+ rest = 0;
+ prevPtr = (Segment*)&head.firstItem;
+ } else {
+ rest = (sz - (head.used % sz)) % sz;
+ prevPtr = thePool.getPtr(head.lastItem);
+ }
+
+ /**
+ * Check for space
+ */
+ Uint32 free = thePool.getNoOfFree() * sz + rest;
+ if(n > free){
+ release();
+ return false;
+ }
+
+ Uint32 used = head.used + n;
+ Ptr<Segment> currPtr;
+ currPtr.i = head.lastItem;
+
+ while(n >= sz){
+ if(0)
+ ndbout_c("n(%d) %c sz(%d)", n, (n>sz?'>':(n<sz?'<':'=')), sz);
+
+ thePool.seize(currPtr); assert(currPtr.i != RNIL);
+ prevPtr->nextPool = currPtr.i;
+
+ prevPtr = currPtr.p;
+ prevPtr->nextPool = RNIL;
+ n -= sz;
+ }
+
+ if(0){
+ Uint32 pos = rest + n;
+ ndbout_c("rest(%d), n(%d) pos=%d %c sz(%d)",
+ rest, n, pos, (pos>sz?'>':(pos<sz?'<':'=')), sz);
+ }
+
+ if(n > rest){
+ thePool.seize(currPtr);
+ assert(currPtr.i != RNIL);
+ prevPtr->nextPool = currPtr.i;
+ currPtr.p->nextPool = RNIL;
+ }
+
+ head.used = used;
+ head.lastItem = currPtr.i;
+
+#if 0
+ {
+ ndbout_c("Before validate - %d", head.used);
+ if(head.used == 0){
+ assert(head.firstItem == RNIL);
+ assert(head.lastItem == RNIL);
+ } else {
+ Ptr<Segment> tmp;
+ tmp.i = head.firstItem;
+ for(Uint32 i = head.used; i > sz; i -= sz){
+ ndbout << tmp.i << " ";
+ tmp.p = thePool.getPtr(tmp.i);
+ tmp.i = tmp.p->nextPool;
+ }
+ ndbout_c("%d", tmp.i);
+ assert(head.lastItem == tmp.i);
+ }
+ ndbout_c("After validate");
+ }
+#endif
+ return true;
+}
+
+template<Uint32 sz>
+inline
+void
+DataBuffer<sz>::release(){
+ Uint32 used = head.used + sz - 1;
+ if(head.firstItem != RNIL){
+ thePool.releaseList(used / sz, head.firstItem, head.lastItem);
+ head.used = 0;
+ head.firstItem = RNIL;
+ head.lastItem = RNIL;
+ }
+}
+
+template<Uint32 sz>
+inline
+Uint32
+DataBuffer<sz>::getSegmentSize(){
+ return sz;
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::first(DataBufferIterator & it){
+ return first((ConstDataBufferIterator&)it);
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::next(DataBufferIterator & it){
+ return next((ConstDataBufferIterator&)it);
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::next(DataBufferIterator & it, Uint32 hops){
+ return next((ConstDataBufferIterator&)it, hops);
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::first(ConstDataBufferIterator & it) const {
+ it.curr.i = head.firstItem;
+ if(it.curr.i == RNIL){
+ it.setNull();
+ return false;
+ }
+ thePool.getPtr(it.curr);
+ it.data = &it.curr.p->data[0];
+ it.ind = 0;
+ it.pos = 0;
+ return true;
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::next(ConstDataBufferIterator & it) const {
+ it.ind ++;
+ it.data ++;
+ it.pos ++;
+ if(it.ind < sz && it.pos < head.used){
+ return true;
+ }
+
+ if(it.pos < head.used){
+ it.curr.i = it.curr.p->nextPool;
+#ifdef ARRAY_GUARD
+ if(it.curr.i == RNIL){
+ /**
+ * This is actually "internal error"
+ * pos can't be less than head.used and at the same time we can't
+ * find next segment
+ *
+ * Note this must not "really" be checked since thePool.getPtr will
+ * abort when trying to get RNIL. That's why the check is within
+ * ARRAY_GUARD
+ */
+ ErrorReporter::handleAssert("DataBuffer<sz>::next", __FILE__, __LINE__);
+ }
+#endif
+ thePool.getPtr(it.curr);
+ it.data = &it.curr.p->data[0];
+ it.ind = 0;
+ return true;
+ }
+ it.setNull();
+ return false;
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::next(ConstDataBufferIterator & it, Uint32 hops) const {
+#if 0
+ for (Uint32 i=0; i<hops; i++) {
+ if (!this->next(it))
+ return false;
+ }
+ return true;
+#else
+ if(it.pos + hops < head.used){
+ while(hops >= sz){
+ it.curr.i = it.curr.p->nextPool;
+ thePool.getPtr(it.curr);
+ hops -= sz;
+ it.pos += sz;
+ }
+
+ it.ind += hops;
+ it.pos += hops;
+ if(it.ind < sz){
+ it.data = &it.curr.p->data[it.ind];
+ return true;
+ }
+
+ it.curr.i = it.curr.p->nextPool;
+ thePool.getPtr(it.curr);
+ it.ind -= sz;
+ it.data = &it.curr.p->data[it.ind];
+ return true;
+ }
+ it.setNull();
+ return false;
+#endif
+}
+
+template<Uint32 sz>
+inline
+Uint32
+DataBuffer<sz>::getSize() const {
+ return head.used;
+}
+
+template<Uint32 sz>
+inline
+bool
+DataBuffer<sz>::isEmpty() const {
+ return (head.used == 0);
+}
+
+#endif
+
diff --git a/ndb/src/kernel/vm/Emulator.cpp b/ndb/src/kernel/vm/Emulator.cpp
new file mode 100644
index 00000000000..43b5619d202
--- /dev/null
+++ b/ndb/src/kernel/vm/Emulator.cpp
@@ -0,0 +1,336 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Emulator.hpp"
+#include <FastScheduler.hpp>
+#include <SignalLoggerManager.hpp>
+#include <TransporterRegistry.hpp>
+#include <TimeQueue.hpp>
+
+#include "Configuration.hpp"
+#include "WatchDog.hpp"
+#include "ThreadConfig.hpp"
+#include "SimBlockList.hpp"
+
+#include <NodeState.hpp>
+
+#include <NdbMem.h>
+#include <NdbOut.hpp>
+#include <NdbMutex.h>
+#include <NdbSleep.h>
+#include <stdlib.h>
+#include <new>
+
+#ifdef NDB_WIN32
+#include <new.h>
+#include <process.h>
+#define execvp _execvp
+#define set_new_handler _set_new_handler
+#endif
+
+/**
+ * Declare the global variables
+ */
+
+#ifdef USE_EMULATED_JAM
+ Uint8 theEmulatedJam[EMULATED_JAM_SIZE * 4];
+ Uint32 theEmulatedJamIndex = 0;
+ Uint32 theEmulatedJamBlockNumber = 0;
+#endif // USE_EMULATED_JAM
+
+ GlobalData globalData;
+
+ TimeQueue globalTimeQueue;
+ FastScheduler globalScheduler;
+ TransporterRegistry globalTransporterRegistry;
+
+#ifdef VM_TRACE
+ SignalLoggerManager globalSignalLoggers;
+#endif
+
+EmulatorData globalEmulatorData;
+NdbMutex * theShutdownMutex = 0;
+
+EmulatorData::EmulatorData(){
+ theConfiguration = 0;
+ theWatchDog = 0;
+ theThreadConfig = 0;
+ theSimBlockList = 0;
+ theShutdownMutex = 0;
+}
+
+void
+ndb_new_handler(){
+ ERROR_SET(fatal, ERR_MEMALLOC, "New handler", "");
+}
+
+void
+EmulatorData::create(){
+ NdbMem_Create();
+
+ theConfiguration = new Configuration();
+ theWatchDog = new WatchDog();
+ theThreadConfig = new ThreadConfig();
+ theSimBlockList = new SimBlockList();
+
+ theShutdownMutex = NdbMutex_Create();
+
+#ifdef NDB_WIN32
+ set_new_handler((_PNH)ndb_new_handler);
+#else
+ std::set_new_handler(ndb_new_handler);
+#endif
+}
+
+void
+EmulatorData::destroy(){
+ if(theConfiguration)
+ delete theConfiguration; theConfiguration = 0;
+ if(theWatchDog)
+ delete theWatchDog; theWatchDog = 0;
+ if(theThreadConfig)
+ delete theThreadConfig; theThreadConfig = 0;
+ if(theSimBlockList)
+ delete theSimBlockList; theSimBlockList = 0;
+
+ NdbMem_Destroy();
+}
+
+void
+NdbRestart(char * programName,
+ NdbRestartType type, char * connString){
+#if ! ( defined NDB_OSE || defined NDB_SOFTOSE)
+ int argc = 2;
+ switch(type){
+ case NRT_NoStart_Restart:
+ case NRT_DoStart_InitialStart:
+ argc = 3;
+ break;
+ case NRT_NoStart_InitialStart:
+ argc = 4;
+ break;
+ case NRT_DoStart_Restart:
+ case NRT_Default:
+ default:
+ argc = 2;
+ break;
+ }
+
+ if(connString != 0){
+ argc += 2;
+ }
+
+ char ** argv = new char * [argc];
+ argv[0] = programName;
+ argv[argc - 1] = 0;
+
+ switch(type){
+ case NRT_NoStart_Restart:
+ argv[1] = "-n";
+ break;
+ case NRT_DoStart_InitialStart:
+ argv[1] = "-i";
+ break;
+ case NRT_NoStart_InitialStart:
+ argv[1] = "-n";
+ argv[2] = "-i";
+ break;
+ case NRT_DoStart_Restart:
+ case NRT_Default:
+ default:
+ break;
+ }
+
+ if(connString != 0){
+ argv[argc-3] = "-c";
+ argv[argc-2] = connString;
+ }
+
+ execvp(programName, argv);
+#endif
+}
+
+void
+NdbShutdown(NdbShutdownType type,
+ NdbRestartType restartType){
+
+ if(type == NST_ErrorInsert){
+ type = NST_Restart;
+ restartType = (NdbRestartType)
+ globalEmulatorData.theConfiguration->getRestartOnErrorInsert();
+ if(restartType == NRT_Default){
+ type = NST_ErrorHandler;
+ globalEmulatorData.theConfiguration->stopOnError(true);
+ }
+ }
+
+ if(NdbMutex_Trylock(theShutdownMutex) == 0){
+ globalData.theRestartFlag = perform_stop;
+
+ bool restart = false;
+ char * progName = 0;
+ char * connString = 0;
+#if ! ( defined NDB_OSE || defined NDB_SOFTOSE)
+ if((type != NST_Normal &&
+ globalEmulatorData.theConfiguration->stopOnError() == false) ||
+ type == NST_Restart) {
+
+ restart = true;
+ progName = strdup(globalEmulatorData.theConfiguration->programName());
+ connString = globalEmulatorData.theConfiguration->getConnectStringCopy();
+ if(type != NST_Restart){
+ /**
+ * If we crash before we started
+ *
+ * Do restart -n
+ */
+ if(globalData.theStartLevel == NodeState::SL_STARTED)
+ restartType = NRT_Default;
+ else
+ restartType = NRT_NoStart_Restart;
+ }
+ }
+#endif
+
+ const char * shutting = "shutting down";
+ if(restart){
+ shutting = "restarting";
+ }
+
+ switch(type){
+ case NST_Normal:
+ ndbout << "Shutdown initiated" << endl;
+ break;
+ case NST_Watchdog:
+ ndbout << "Watchdog " << shutting << " system" << endl;
+ break;
+ case NST_ErrorHandler:
+ ndbout << "Error handler " << shutting << " system" << endl;
+ break;
+ case NST_Restart:
+ ndbout << "Restarting system" << endl;
+ break;
+ default:
+ ndbout << "Error handler " << shutting << " system"
+ << " (unknown type: " << type << ")" << endl;
+ type = NST_ErrorHandler;
+ break;
+ }
+
+ const char * exitAbort = 0;
+#if defined VM_TRACE && ( ! ( defined NDB_OSE || defined NDB_SOFTOSE) )
+ exitAbort = "aborting";
+#else
+ exitAbort = "exiting";
+#endif
+
+ if(type == NST_Watchdog){
+ if(restart){
+ NdbRestart(progName, restartType, connString);
+ }
+
+ /**
+ * Very serious
+ */
+ ndbout << "Watchdog shutdown completed - " << exitAbort << endl;
+#if defined VM_TRACE && ( ! ( defined NDB_OSE || defined NDB_SOFTOSE) )
+ abort();
+#else
+ exit(1);
+#endif
+ }
+
+ globalEmulatorData.theWatchDog->doStop();
+
+#ifdef VM_TRACE
+ FILE * outputStream = globalSignalLoggers.setOutputStream(0);
+ if(outputStream != 0)
+ fclose(outputStream);
+#endif
+
+ globalTransporterRegistry.stopSending();
+ globalTransporterRegistry.stopReceiving();
+
+ globalTransporterRegistry.removeAll();
+
+#ifdef VM_TRACE
+#define UNLOAD (type != NST_ErrorHandler && type != NST_Watchdog)
+#else
+#define UNLOAD true
+#endif
+ if(UNLOAD){
+ globalEmulatorData.theSimBlockList->unload();
+ globalEmulatorData.destroy();
+
+ }
+
+ if(type != NST_Normal &&
+ type != NST_Restart){
+ if(restart){
+ NdbRestart(progName, restartType, connString);
+ }
+
+ ndbout << "Error handler shutdown completed - " << exitAbort << endl;
+#if defined VM_TRACE && ( ! ( defined NDB_OSE || defined NDB_SOFTOSE) )
+ abort();
+#else
+ exit(1);
+#endif
+ }
+
+ /**
+ * This is a normal restart
+ */
+ if(type == NST_Restart){
+ if(restart){
+ NdbRestart(progName, restartType, connString);
+ }
+ /**
+ * What to do if in restart mode, but being unable to do it...
+ */
+#if defined VM_TRACE && ( ! ( defined NDB_OSE || defined NDB_SOFTOSE) )
+ abort();
+#else
+ exit(1);
+#endif
+ }
+
+ /**
+ * This is normal shutdown
+ */
+ ndbout << "Shutdown completed - exiting" << endl;
+ } else {
+ /**
+ * Shutdown is already in progress
+ */
+
+ /**
+ * If this is the watchdog, kill system the hard way
+ */
+ if (type== NST_Watchdog){
+ ndbout << "Watchdog is killing system the hard way" << endl;
+#if defined VM_TRACE && ( ! ( defined NDB_OSE || defined NDB_SOFTOSE) )
+ abort();
+#else
+ exit(1);
+#endif
+ }
+
+ while(true)
+ NdbSleep_MilliSleep(10);
+ }
+}
+
diff --git a/ndb/src/kernel/vm/Emulator.hpp b/ndb/src/kernel/vm/Emulator.hpp
new file mode 100644
index 00000000000..ba533eb873d
--- /dev/null
+++ b/ndb/src/kernel/vm/Emulator.hpp
@@ -0,0 +1,101 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef EMULATOR_H
+#define EMULATOR_H
+
+//===========================================================================
+//
+// .DESCRIPTION
+// This is the main fuction for the AXE VM emulator.
+// It contains some global objects and a run method.
+//
+//===========================================================================
+#include <kernel_types.h>
+
+extern class JobTable globalJobTable;
+extern class TimeQueue globalTimeQueue;
+extern class FastScheduler globalScheduler;
+extern class TransporterRegistry globalTransporterRegistry;
+extern struct GlobalData globalData;
+
+#ifdef VM_TRACE
+extern class SignalLoggerManager globalSignalLoggers;
+#endif
+
+#ifdef USE_EMULATED_JAM
+#define EMULATED_JAM_SIZE 1024
+#define JAM_MASK ((EMULATED_JAM_SIZE * 4) - 1)
+
+extern Uint8 theEmulatedJam[];
+extern Uint32 theEmulatedJamIndex;
+// last block entry, used in dumpJam() if jam contains no block entries
+extern Uint32 theEmulatedJamBlockNumber;
+#endif // USE_EMULATED_JAM
+
+struct EmulatorData {
+ class Configuration * theConfiguration;
+ class WatchDog * theWatchDog;
+ class ThreadConfig * theThreadConfig;
+ class SimBlockList * theSimBlockList;
+
+ /**
+ * Constructor
+ *
+ * Sets all the pointers to NULL
+ */
+ EmulatorData();
+
+ /**
+ * Create all the objects
+ */
+ void create();
+
+ /**
+ * Destroys all the objects
+ */
+ void destroy();
+};
+
+extern struct EmulatorData globalEmulatorData;
+
+enum NdbShutdownType {
+ NST_Normal,
+ NST_Watchdog,
+ NST_ErrorHandler,
+ NST_Restart,
+ NST_ErrorInsert
+};
+
+enum NdbRestartType {
+ NRT_Default = 0,
+ NRT_NoStart_Restart = 1, // -n
+ NRT_DoStart_Restart = 2, //
+ NRT_NoStart_InitialStart = 3, // -n -i
+ NRT_DoStart_InitialStart = 4 // -i
+};
+
+/**
+ * Shutdown/restart Ndb
+ *
+ * @param type - Type of shutdown/restart
+ * @param restartType - Type of restart (only valid if type == NST_Restart)
+ */
+void
+NdbShutdown(NdbShutdownType type,
+ NdbRestartType restartType = NRT_Default);
+
+#endif
diff --git a/ndb/src/kernel/vm/FastScheduler.cpp b/ndb/src/kernel/vm/FastScheduler.cpp
new file mode 100644
index 00000000000..e9ca4834562
--- /dev/null
+++ b/ndb/src/kernel/vm/FastScheduler.cpp
@@ -0,0 +1,483 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "FastScheduler.hpp"
+#include "RefConvert.hpp"
+
+#include "Emulator.hpp"
+#include "VMSignal.hpp"
+#include <Error.hpp>
+
+#include <SignalLoggerManager.hpp>
+#include <BlockNumbers.h>
+#include <GlobalSignalNumbers.h>
+#include <signaldata/EventReport.hpp>
+#include "LongSignal.hpp"
+#include <NdbTick.h>
+
+#define MIN_NUMBER_OF_SIG_PER_DO_JOB 64
+#define MAX_NUMBER_OF_SIG_PER_DO_JOB 2048
+#define EXTRA_SIGNALS_PER_DO_JOB 32
+
+FastScheduler::FastScheduler()
+{
+ // These constants work for sun only, but they should be initated from
+ // Emulator.C as soon as VMTime has been initiated.
+ theJobBuffers[0].newBuffer(JBASIZE);
+ theJobBuffers[1].newBuffer(JBBSIZE);
+ theJobBuffers[2].newBuffer(JBCSIZE);
+ theJobBuffers[3].newBuffer(JBDSIZE);
+ clear();
+}
+
+FastScheduler::~FastScheduler()
+{
+}
+
+void
+FastScheduler::clear()
+{
+ int i;
+ // Make sure the restart signals are not sent too early
+ // the prio is set back in 'main' using the 'ready' method.
+ globalData.highestAvailablePrio = LEVEL_IDLE;
+ globalData.sendPackedActivated = 0;
+ globalData.activateSendPacked = 0;
+ for (i = 0; i < JB_LEVELS; i++){
+ theJobBuffers[i].clear();
+ }
+ globalData.JobCounter = 0;
+ globalData.JobLap = 0;
+ globalData.loopMax = 32;
+ globalData.VMSignals[0].header.theSignalId = 0;
+
+ theDoJobTotalCounter = 0;
+ theDoJobCallCounter = 0;
+}
+
+void
+FastScheduler::activateSendPacked()
+{
+ globalData.sendPackedActivated = 1;
+ globalData.activateSendPacked = 0;
+ globalData.loopMax = 2048;
+}//FastScheduler::activateSendPacked()
+
+void
+FastScheduler::doJob()
+{
+ do{
+ Uint32 loopCount = 0;
+ Uint32 TminLoops = getBOccupancy() + EXTRA_SIGNALS_PER_DO_JOB;
+ Uint32 TloopMax = (Uint32)globalData.loopMax;
+ if (TminLoops < TloopMax) {
+ TloopMax = TminLoops;
+ }//if
+ if (TloopMax < MIN_NUMBER_OF_SIG_PER_DO_JOB) {
+ TloopMax = MIN_NUMBER_OF_SIG_PER_DO_JOB;
+ }//if
+ register Uint32 tHighPrio = globalData.highestAvailablePrio;
+ register Signal* signal = getVMSignals();
+ while ((tHighPrio < LEVEL_IDLE) && (loopCount < TloopMax)) {
+ // signal->garbage_register();
+ // To ensure we find bugs quickly
+ register Uint32 gsnbnr = theJobBuffers[tHighPrio].retrieve(signal);
+ register BlockNumber reg_bnr = gsnbnr & 0xFFF;
+ register GlobalSignalNumber reg_gsn = gsnbnr >> 16;
+ globalData.incrementWatchDogCounter(1);
+ if (reg_bnr > 0) {
+ Uint32 tJobCounter = globalData.JobCounter;
+ Uint32 tJobLap = globalData.JobLap;
+ SimulatedBlock* b = globalData.getBlock(reg_bnr);
+ theJobPriority[tJobCounter] = (Uint8)tHighPrio;
+ globalData.JobCounter = (tJobCounter + 1) & 4095;
+ globalData.JobLap = tJobLap + 1;
+
+#ifdef VM_TRACE_TIME
+ Uint32 us1, us2;
+ Uint64 ms1, ms2;
+ NdbTick_CurrentMicrosecond(&ms1, &us1);
+ b->m_currentGsn = reg_gsn;
+#endif
+
+ getSections(signal->header.m_noOfSections, signal->m_sectionPtr);
+#ifdef VM_TRACE
+ {
+ if (globalData.testOn) {
+ signal->header.theVerId_signalNumber = reg_gsn;
+ signal->header.theReceiversBlockNumber = reg_bnr;
+
+ globalSignalLoggers.executeSignal(signal->header,
+ tHighPrio,
+ &signal->theData[0],
+ globalData.ownId,
+ signal->m_sectionPtr,
+ signal->header.m_noOfSections);
+ }//if
+ }
+#endif
+ b->executeFunction(reg_gsn, signal);
+ releaseSections(signal->header.m_noOfSections, signal->m_sectionPtr);
+ signal->header.m_noOfSections = 0;
+#ifdef VM_TRACE_TIME
+ NdbTick_CurrentMicrosecond(&ms2, &us2);
+ Uint64 diff = ms2;
+ diff -= ms1;
+ diff *= 1000000;
+ diff += us2;
+ diff -= us1;
+ b->addTime(reg_gsn, diff);
+#endif
+ tHighPrio = globalData.highestAvailablePrio;
+ } else {
+ tHighPrio++;
+ globalData.highestAvailablePrio = tHighPrio;
+ }//if
+ loopCount++;
+ }//while
+ if (globalData.sendPackedActivated == 1) {
+ Uint32 t1 = theDoJobTotalCounter;
+ Uint32 t2 = theDoJobCallCounter;
+ t1 += loopCount;
+ t2++;
+ theDoJobTotalCounter = t1;
+ theDoJobCallCounter = t2;
+ if (t2 == 8192) {
+ reportDoJobStatistics(t1 >> 13);
+ theDoJobCallCounter = 0;
+ theDoJobTotalCounter = 0;
+ }//if
+ }//if
+ } while (getBOccupancy() > MAX_OCCUPANCY);
+}//FastScheduler::doJob()
+
+void FastScheduler::sendPacked()
+{
+ if (globalData.sendPackedActivated == 1) {
+ SimulatedBlock* b_lqh = globalData.getBlock(DBLQH);
+ SimulatedBlock* b_tc = globalData.getBlock(DBTC);
+ SimulatedBlock* b_tup = globalData.getBlock(DBTUP);
+ Signal* signal = getVMSignals();
+ b_lqh->executeFunction(GSN_SEND_PACKED, signal);
+ b_tc->executeFunction(GSN_SEND_PACKED, signal);
+ b_tup->executeFunction(GSN_SEND_PACKED, signal);
+ return;
+ } else if (globalData.activateSendPacked == 0) {
+ return;
+ } else {
+ activateSendPacked();
+ }//if
+ return;
+}//FastScheduler::sendPacked()
+
+Uint32
+APZJobBuffer::retrieve(Signal* signal)
+{
+ Uint32 tOccupancy = theOccupancy;
+ Uint32 myRPtr = rPtr;
+ BufferEntry& buf = buffer[myRPtr];
+ Uint32 gsnbnr;
+ Uint32 cond = (++myRPtr == bufSize) - 1;
+ Uint32 tRecBlockNo = buf.header.theReceiversBlockNumber;
+
+ if (tOccupancy != 0) {
+ if (tRecBlockNo != 0) {
+ // Transform protocol to signal.
+ rPtr = myRPtr & cond;
+ theOccupancy = tOccupancy - 1;
+ gsnbnr = buf.header.theVerId_signalNumber << 16 | tRecBlockNo;
+
+ Uint32 tSignalId = globalData.theSignalId;
+ Uint32 tLength = buf.header.theLength;
+ Uint32 tFirstData = buf.theDataRegister[0];
+ signal->header = buf.header;
+
+ // Recall our signal Id for restart purposes
+ buf.header.theSignalId = tSignalId;
+ globalData.theSignalId = tSignalId + 1;
+
+ Uint32* tDataRegPtr = &buf.theDataRegister[0];
+ Uint32* tSigDataPtr = signal->getDataPtrSend();
+ *tSigDataPtr = tFirstData;
+ tDataRegPtr++;
+ tSigDataPtr++;
+ Uint32 tLengthCopied = 1;
+ while (tLengthCopied < tLength) {
+ Uint32 tData0 = tDataRegPtr[0];
+ Uint32 tData1 = tDataRegPtr[1];
+ Uint32 tData2 = tDataRegPtr[2];
+ Uint32 tData3 = tDataRegPtr[3];
+
+ tDataRegPtr += 4;
+ tLengthCopied += 4;
+
+ tSigDataPtr[0] = tData0;
+ tSigDataPtr[1] = tData1;
+ tSigDataPtr[2] = tData2;
+ tSigDataPtr[3] = tData3;
+ tSigDataPtr += 4;
+ }//while
+
+ /**
+ * Copy sections references (copy all without if-statements)
+ */
+ tDataRegPtr = &buf.theDataRegister[tLength];
+ SegmentedSectionPtr * tSecPtr = &signal->m_sectionPtr[0];
+ Uint32 tData0 = tDataRegPtr[0];
+ Uint32 tData1 = tDataRegPtr[1];
+ Uint32 tData2 = tDataRegPtr[2];
+
+ tSecPtr[0].i = tData0;
+ tSecPtr[1].i = tData1;
+ tSecPtr[2].i = tData2;
+
+ //---------------------------------------------------------
+ // Prefetch of buffer[rPtr] is done here. We prefetch for
+ // read both the first cache line and the next 64 byte
+ // entry
+ //---------------------------------------------------------
+ PREFETCH((void*)&buffer[rPtr]);
+ PREFETCH((void*)(((char*)&buffer[rPtr]) + 64));
+ return gsnbnr;
+ } else {
+ bnr_error();
+ return 0; // Will never come here, simply to keep GCC happy.
+ }//if
+ } else {
+ //------------------------------------------------------------
+ // The Job Buffer was empty, signal this by return zero.
+ //------------------------------------------------------------
+ return 0;
+ }//if
+}//APZJobBuffer::retrieve()
+
+void
+APZJobBuffer::signal2buffer(Signal* signal,
+ BlockNumber bnr, GlobalSignalNumber gsn,
+ BufferEntry& buf)
+{
+ Uint32 tSignalId = globalData.theSignalId;
+ Uint32 tFirstData = signal->theData[0];
+ Uint32 tLength = signal->header.theLength;
+ Uint32 tSigId = buf.header.theSignalId;
+
+ buf.header = signal->header;
+ buf.header.theVerId_signalNumber = gsn;
+ buf.header.theReceiversBlockNumber = bnr;
+ buf.header.theSendersSignalId = tSignalId - 1;
+ buf.header.theSignalId = tSigId;
+ buf.theDataRegister[0] = tFirstData;
+
+ Uint32 tLengthCopied = 1;
+ Uint32* tSigDataPtr = &signal->theData[1];
+ Uint32* tDataRegPtr = &buf.theDataRegister[1];
+ while (tLengthCopied < tLength) {
+ Uint32 tData0 = tSigDataPtr[0];
+ Uint32 tData1 = tSigDataPtr[1];
+ Uint32 tData2 = tSigDataPtr[2];
+ Uint32 tData3 = tSigDataPtr[3];
+
+ tLengthCopied += 4;
+ tSigDataPtr += 4;
+
+ tDataRegPtr[0] = tData0;
+ tDataRegPtr[1] = tData1;
+ tDataRegPtr[2] = tData2;
+ tDataRegPtr[3] = tData3;
+ tDataRegPtr += 4;
+ }//while
+
+ /**
+ * Copy sections references (copy all without if-statements)
+ */
+ tDataRegPtr = &buf.theDataRegister[tLength];
+ SegmentedSectionPtr * tSecPtr = &signal->m_sectionPtr[0];
+ Uint32 tData0 = tSecPtr[0].i;
+ Uint32 tData1 = tSecPtr[1].i;
+ Uint32 tData2 = tSecPtr[2].i;
+ tDataRegPtr[0] = tData0;
+ tDataRegPtr[1] = tData1;
+ tDataRegPtr[2] = tData2;
+}//APZJobBuffer::signal2buffer()
+
+void
+APZJobBuffer::insert(const SignalHeader * const sh,
+ const Uint32 * const theData, const Uint32 secPtrI[3]){
+ Uint32 tOccupancy = theOccupancy;
+ Uint32 myWPtr = wPtr;
+ register BufferEntry& buf = buffer[myWPtr];
+
+ if (tOccupancy < bufSize) {
+ Uint32 cond = (++myWPtr == bufSize) - 1;
+ wPtr = myWPtr & cond;
+ theOccupancy = tOccupancy + 1;
+
+ buf.header = * sh;
+ const Uint32 len = buf.header.theLength;
+ memcpy(buf.theDataRegister, theData, 4 * len);
+ memcpy(&buf.theDataRegister[len], &secPtrI[0], 4 * 3);
+ //---------------------------------------------------------
+ // Prefetch of buffer[wPtr] is done here. We prefetch for
+ // write both the first cache line and the next 64 byte
+ // entry
+ //---------------------------------------------------------
+ WRITEHINT((void*)&buffer[wPtr]);
+ WRITEHINT((void*)(((char*)&buffer[wPtr]) + 64));
+
+ } else {
+ jbuf_error();
+ }//if
+}
+APZJobBuffer::APZJobBuffer()
+ : rPtr(0), wPtr(0), theOccupancy(0), bufSize(0), buffer(NULL), memRef(NULL)
+{
+}
+
+APZJobBuffer::~APZJobBuffer()
+{
+ delete [] buffer;
+}
+
+void
+APZJobBuffer::newBuffer(int size)
+{
+ buffer = new BufferEntry[size];
+ if(buffer){
+ ::memset(buffer, 0, (size * sizeof(BufferEntry)));
+ bufSize = size;
+ } else
+ bufSize = 0;
+}
+
+void
+APZJobBuffer::clear()
+{
+ rPtr = 0;
+ wPtr = 0;
+ theOccupancy = 0;
+}
+
+/**
+ * Function prototype for print_restart
+ *
+ * Defined later in this file
+ */
+void print_restart(FILE * output, Signal* signal, Uint32 aLevel);
+
+void FastScheduler::dumpSignalMemory(FILE * output)
+{
+ Signal signal;
+ Uint32 ReadPtr[5];
+ Uint32 tJob;
+ Uint32 tLastJob;
+
+ fprintf(output, "\n");
+
+ if (globalData.JobLap > 4095) {
+ if (globalData.JobCounter != 0)
+ tJob = globalData.JobCounter - 1;
+ else
+ tJob = 4095;
+ tLastJob = globalData.JobCounter;
+ } else {
+ if (globalData.JobCounter == 0)
+ return; // No signals sent
+ else {
+ tJob = globalData.JobCounter - 1;
+ tLastJob = 4095;
+ }
+ }
+ ReadPtr[0] = theJobBuffers[0].getReadPtr();
+ ReadPtr[1] = theJobBuffers[1].getReadPtr();
+ ReadPtr[2] = theJobBuffers[2].getReadPtr();
+ ReadPtr[3] = theJobBuffers[3].getReadPtr();
+
+ do {
+ unsigned char tLevel = theJobPriority[tJob];
+ globalData.incrementWatchDogCounter(4);
+ if (ReadPtr[tLevel] == 0)
+ ReadPtr[tLevel] = theJobBuffers[tLevel].getBufSize() - 1;
+ else
+ ReadPtr[tLevel]--;
+
+ theJobBuffers[tLevel].retrieveDump(&signal, ReadPtr[tLevel]);
+ print_restart(output, &signal, tLevel);
+
+ if (tJob == 0)
+ tJob = 4095;
+ else
+ tJob--;
+
+ } while (tJob != tLastJob);
+ fflush(output);
+}
+
+void
+FastScheduler::prio_level_error()
+{
+ ERROR_SET(ecError, ERROR_WRONG_PRIO_LEVEL,
+ "Wrong Priority Level", "FastScheduler.C");
+}
+
+void
+jbuf_error()
+{
+ ERROR_SET(ecError, BLOCK_ERROR_JBUFCONGESTION,
+ "Job Buffer Full", "APZJobBuffer.C");
+}
+
+void
+bnr_error()
+{
+ ERROR_SET(ecError, BLOCK_ERROR_BNR_ZERO,
+ "Block Number Zero", "FastScheduler.C");
+}
+
+void
+print_restart(FILE * output, Signal* signal, Uint32 aLevel)
+{
+ fprintf(output, "--------------- Signal ----------------\n");
+ SignalLoggerManager::printSignalHeader(output,
+ signal->header,
+ aLevel,
+ globalData.ownId,
+ true);
+ SignalLoggerManager::printSignalData (output,
+ signal->header,
+ &signal->theData[0]);
+}
+
+/**
+ * This method used to be a Cmvmi member function
+ * but is now a "ordinary" function"
+ *
+ * See TransporterCallback.cpp for explanation
+ */
+void
+FastScheduler::reportDoJobStatistics(Uint32 tMeanLoopCount) {
+ Signal signal;
+ memset(&signal.header, 0, sizeof(signal.header));
+
+ signal.theData[0] = EventReport::JobStatistic;
+ signal.theData[1] = tMeanLoopCount;
+
+ signal.header.theLength = 2;
+ signal.header.theSendersSignalId = 0;
+ signal.header.theSendersBlockRef = numberToRef(0, 0);
+
+ execute(&signal, JBA, CMVMI, GSN_EVENT_REP);
+}
+
diff --git a/ndb/src/kernel/vm/FastScheduler.hpp b/ndb/src/kernel/vm/FastScheduler.hpp
new file mode 100644
index 00000000000..586a7ea27ad
--- /dev/null
+++ b/ndb/src/kernel/vm/FastScheduler.hpp
@@ -0,0 +1,353 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef FastScheduler_H
+#define FastScheduler_H
+
+#include <VMSignal.hpp>
+#include <kernel_types.h>
+#include <Prio.hpp>
+#include <SignalLoggerManager.hpp>
+#include <SimulatedBlock.hpp>
+#include <ErrorHandlingMacros.hpp>
+#include <GlobalData.hpp>
+#include <TransporterDefinitions.hpp>
+#include <prefetch.h>
+
+#define MAX_OCCUPANCY 1024
+
+#define JBASIZE 1280 // Jobs which have dead lines to meet use this level
+#define JBBSIZE 4096 // Most jobs use this level
+#define JBCSIZE 64 // Only used by STTOR and STTORRY currently
+#define JBDSIZE 4096 // Time Queue uses this level for storage, not supported
+ // as priority level
+void bnr_error();
+void jbuf_error();
+class Signal;
+class Block;
+
+class BufferEntry
+{
+public:
+ SignalHeader header;
+ Uint32 theDataRegister[28];
+};
+
+class APZJobBuffer
+{
+public:
+ APZJobBuffer();
+ ~APZJobBuffer();
+
+ void newBuffer(int size);
+
+ void insert(Signal* signal, BlockNumber bnr, GlobalSignalNumber gsn);
+ void insert(const SignalHeader * const sh, const Uint32 * const theData, const Uint32 secPtrI[3]);
+ void insert(Signal* signal, BlockNumber bnr, GlobalSignalNumber gsn,
+ Uint32 myWPtr);
+
+ Uint32 retrieve(Signal *signal);
+ void retrieve(Signal *signal, Uint32 myRptr);
+
+ /**
+ * Used when dumping to trace file
+ */
+ void retrieveDump(Signal *signal, Uint32 myRptr);
+
+ void clear();
+ bool isEmpty() const;
+ Uint32 getOccupancy() const;
+
+ Uint32 getReadPtr() const;
+ Uint32 getWritePtr() const;
+ Uint32 getBufSize() const;
+
+private:
+ void signal2buffer(Signal* signal, BlockNumber bnr,
+ GlobalSignalNumber gsn, BufferEntry& buf);
+ Uint32 rPtr;
+ Uint32 wPtr;
+ Uint32 theOccupancy;
+ Uint32 bufSize;
+ BufferEntry* buffer;
+ BufferEntry* memRef;
+};
+
+
+class FastScheduler
+{
+public:
+ FastScheduler();
+ ~FastScheduler();
+
+ void doJob();
+ int checkDoJob();
+
+ void activateSendPacked();
+
+ void execute(Signal* signal,
+ Priority prio,
+ BlockNumber bnr,
+ GlobalSignalNumber gsn);
+
+ void execute(const SignalHeader * const sh,
+ Uint8 prio, const Uint32 * const theData, const Uint32 secPtr[3]);
+
+ void clear();
+ Signal* getVMSignals();
+
+ void dumpSignalMemory(FILE * output);
+ Priority highestAvailablePrio() const;
+ Uint32 getBOccupancy() const;
+ void sendPacked();
+
+ void insertTimeQueue(Signal* aSignal, BlockNumber bnr,
+ GlobalSignalNumber gsn, Uint32 aIndex);
+ void scheduleTimeQueue(Uint32 aIndex);
+
+private:
+ void highestAvailablePrio(Priority prio);
+ void reportJob(Priority aPriority);
+ void prio_level_error();
+
+ Uint32 theDoJobTotalCounter;
+ Uint32 theDoJobCallCounter;
+ Uint8 theJobPriority[4096];
+ APZJobBuffer theJobBuffers[JB_LEVELS];
+
+ void reportDoJobStatistics(Uint32 meanLoopCount);
+};
+
+inline
+Uint32
+FastScheduler::getBOccupancy() const {
+ return theJobBuffers[JBB].getOccupancy();
+}//FastScheduler::getBOccupancy()
+
+inline
+int
+FastScheduler::checkDoJob()
+{
+ /*
+ * Joob buffer overload protetction
+ * If the job buffer B is filled over a certain limit start
+ * to execute the signals in the job buffer's
+ */
+ if (getBOccupancy() < MAX_OCCUPANCY) {
+ return 0;
+ } else {
+ doJob();
+ return 1;
+ }//if
+}//FastScheduler::checkDoJob()
+
+inline
+void
+FastScheduler::reportJob(Priority aPriority)
+{
+ Uint32 tJobCounter = globalData.JobCounter;
+ Uint32 tJobLap = globalData.JobLap;
+ theJobPriority[tJobCounter] = (Uint8)aPriority;
+ globalData.JobCounter = (tJobCounter + 1) & 4095;
+ globalData.JobLap = tJobLap + 1;
+}
+
+inline
+Priority
+FastScheduler::highestAvailablePrio() const
+{
+ return (Priority)globalData.highestAvailablePrio;
+}
+
+inline
+void
+FastScheduler::highestAvailablePrio(Priority prio)
+{
+ globalData.highestAvailablePrio = (Uint32)prio;
+}
+
+inline
+Signal*
+FastScheduler::getVMSignals()
+{
+ return &globalData.VMSignals[0];
+}
+
+
+// Inserts of a protocol object into the Job Buffer.
+inline
+void
+FastScheduler::execute(const SignalHeader * const sh, Uint8 prio,
+ const Uint32 * const theData, const Uint32 secPtrI[3]){
+#ifdef VM_TRACE
+ if (prio >= LEVEL_IDLE)
+ prio_level_error();
+#endif
+
+ theJobBuffers[prio].insert(sh, theData, secPtrI);
+ if (prio < (Uint8)highestAvailablePrio())
+ highestAvailablePrio((Priority)prio);
+}
+
+inline
+void
+FastScheduler::execute(Signal* signal, Priority prio,
+ BlockNumber bnr, GlobalSignalNumber gsn)
+{
+#ifdef VM_TRACE
+ if (prio >= LEVEL_IDLE)
+ prio_level_error();
+#endif
+ theJobBuffers[prio].insert(signal, bnr, gsn);
+ if (prio < highestAvailablePrio())
+ highestAvailablePrio(prio);
+}
+
+inline
+void
+FastScheduler::insertTimeQueue(Signal* signal, BlockNumber bnr,
+ GlobalSignalNumber gsn, Uint32 aIndex)
+{
+ theJobBuffers[3].insert(signal, bnr, gsn, aIndex);
+}
+
+inline
+void
+FastScheduler::scheduleTimeQueue(Uint32 aIndex)
+{
+ Signal* signal = getVMSignals();
+ theJobBuffers[3].retrieve(signal, aIndex);
+ theJobBuffers[0].insert
+ (signal,
+ (BlockNumber)signal->header.theReceiversBlockNumber,
+ (GlobalSignalNumber)signal->header.theVerId_signalNumber);
+ if (highestAvailablePrio() > JBA)
+ highestAvailablePrio(JBA);
+}
+
+inline
+Uint32
+APZJobBuffer::getWritePtr() const
+{
+ return wPtr;
+}
+
+inline
+Uint32
+APZJobBuffer::getReadPtr() const
+{
+ return rPtr;
+}
+
+inline
+Uint32
+APZJobBuffer::getOccupancy() const
+{
+ return theOccupancy;
+}
+
+inline
+Uint32
+APZJobBuffer::getBufSize() const
+{
+ return bufSize;
+}
+
+inline
+void
+APZJobBuffer::retrieve(Signal* signal, Uint32 myRptr)
+{
+ register BufferEntry& buf = buffer[myRptr];
+
+ buf.header.theSignalId = globalData.theSignalId++;
+
+ signal->header = buf.header;
+
+ Uint32 *from = (Uint32*) &buf.theDataRegister[0];
+ Uint32 *to = (Uint32*) &signal->theData[0];
+ Uint32 noOfWords = buf.header.theLength;
+ for(; noOfWords; noOfWords--)
+ *to++ = *from++;
+ // Copy sections references (copy all without if-statements)
+ SegmentedSectionPtr * tSecPtr = &signal->m_sectionPtr[0];
+ tSecPtr[0].i = from[0];
+ tSecPtr[1].i = from[1];
+ tSecPtr[2].i = from[2];
+ return;
+}
+
+inline
+void
+APZJobBuffer::retrieveDump(Signal* signal, Uint32 myRptr)
+{
+ /**
+ * Note that signal id is not taken from global data
+ */
+
+ register BufferEntry& buf = buffer[myRptr];
+ signal->header = buf.header;
+
+ Uint32 *from = (Uint32*) &buf.theDataRegister[0];
+ Uint32 *to = (Uint32*) &signal->theData[0];
+ Uint32 noOfWords = buf.header.theLength;
+ for(; noOfWords; noOfWords--)
+ *to++ = *from++;
+ return;
+}
+
+inline
+void
+APZJobBuffer::insert(Signal* signal,
+ BlockNumber bnr, GlobalSignalNumber gsn)
+{
+ Uint32 tOccupancy = theOccupancy;
+ Uint32 myWPtr = wPtr;
+ if (tOccupancy < bufSize) {
+ register BufferEntry& buf = buffer[myWPtr];
+ Uint32 cond = (++myWPtr == bufSize) - 1;
+ wPtr = myWPtr & cond;
+ theOccupancy = tOccupancy + 1;
+ signal2buffer(signal, bnr, gsn, buf);
+ //---------------------------------------------------------
+ // Prefetch of buffer[wPtr] is done here. We prefetch for
+ // write both the first cache line and the next 64 byte
+ // entry
+ //---------------------------------------------------------
+ WRITEHINT((void*)&buffer[wPtr]);
+ WRITEHINT((void*)(((char*)&buffer[wPtr]) + 64));
+ } else {
+ jbuf_error();
+ }//if
+}
+
+
+inline
+void
+APZJobBuffer::insert(Signal* signal, BlockNumber bnr,
+ GlobalSignalNumber gsn, Uint32 myWPtr)
+{
+ register BufferEntry& buf = buffer[myWPtr];
+ signal2buffer(signal, bnr, gsn, buf);
+}
+
+inline
+bool
+APZJobBuffer::isEmpty() const
+{
+ return (theOccupancy == 0);
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/GlobalData.hpp b/ndb/src/kernel/vm/GlobalData.hpp
new file mode 100644
index 00000000000..ca7dd467750
--- /dev/null
+++ b/ndb/src/kernel/vm/GlobalData.hpp
@@ -0,0 +1,118 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GLOBAL_DATA_H
+#define GLOBAL_DATA_H
+
+#include <kernel_types.h>
+#include "Prio.hpp"
+#include "VMSignal.hpp"
+#include <stdlib.h>
+#include <assert.h>
+
+#include <BlockNumbers.h>
+#include <NodeState.hpp>
+#include <NodeInfo.hpp>
+
+class SimulatedBlock;
+
+enum restartStates {initial_state,
+ perform_start,
+ system_started,
+ perform_stop};
+
+struct GlobalData {
+ NodeInfo m_nodeInfo[MAX_NODES];
+ Signal VMSignals[1]; // Owned by FastScheduler::
+
+ Uint64 internalMillisecCounter; // Owned by ThreadConfig::
+ Uint32 highestAvailablePrio; // Owned by FastScheduler::
+ Uint32 JobCounter; // Owned by FastScheduler
+ Uint64 JobLap; // Owned by FastScheduler
+ Uint32 loopMax; // Owned by FastScheduler
+
+ Uint32 theNextTimerJob; // Owned by TimeQueue::
+ Uint32 theCurrentTimer; // Owned by TimeQueue::
+ Uint32 theShortTQIndex; // Owned by TimeQueue::
+
+ Uint32 theLongTQIndex; // Owned by TimeQueue::
+ Uint32 theCountTimer; // Owned by TimeQueue::
+ Uint32 theFirstFreeTQIndex; // Owned by TimeQueue::
+ Uint32 testOn; // Owned by the Signal Loggers
+
+ NodeId ownId; // Own processor id
+
+ Uint32 theStartLevel;
+ restartStates theRestartFlag;
+ Uint32 theSignalId;
+
+ Uint32 sendPackedActivated;
+ Uint32 activateSendPacked;
+
+ GlobalData(){
+ theSignalId = 0;
+ theStartLevel = NodeState::SL_NOTHING;
+ theRestartFlag = perform_start;
+ }
+ ~GlobalData(){}
+
+ void setBlock(BlockNumber blockNo, SimulatedBlock * block);
+ SimulatedBlock * getBlock(BlockNumber blockNo);
+
+ void incrementWatchDogCounter(Uint32 place);
+ const Uint32 * getWatchDogPtr();
+
+private:
+ Uint32 watchDog;
+ SimulatedBlock* blockTable[NO_OF_BLOCKS]; // Owned by Dispatcher::
+};
+
+extern GlobalData globalData;
+
+#define GLOBAL_TEST_ON (localTestOn)
+#define GET_GLOBAL_TEST_FLAG bool localTestOn = globalData.testOn
+#define SET_GLOBAL_TEST_ON (globalData.testOn = true)
+#define SET_GLOBAL_TEST_OFF (globalData.testOn = false)
+#define TOGGLE_GLOBAL_TEST_FLAG (globalData.testOn = (globalData.testOn == true ? false : true))
+
+inline
+void
+GlobalData::setBlock(BlockNumber blockNo, SimulatedBlock * block){
+ blockNo -= MIN_BLOCK_NO;
+ assert((blockTable[blockNo] == 0) || (blockTable[blockNo] == block));
+ blockTable[blockNo] = block;
+}
+
+inline
+SimulatedBlock *
+GlobalData::getBlock(BlockNumber blockNo){
+ blockNo -= MIN_BLOCK_NO;
+ return blockTable[blockNo];
+}
+
+inline
+void
+GlobalData::incrementWatchDogCounter(Uint32 place){
+ watchDog = place;
+}
+
+inline
+const Uint32 *
+GlobalData::getWatchDogPtr(){
+ return &watchDog;
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/KeyTable.hpp b/ndb/src/kernel/vm/KeyTable.hpp
new file mode 100644
index 00000000000..e78837b5c8a
--- /dev/null
+++ b/ndb/src/kernel/vm/KeyTable.hpp
@@ -0,0 +1,43 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef KEY_TABLE_HPP
+#define KEY_TABLE_HPP
+
+#include <DLHashTable.hpp>
+
+/**
+ * KeyTable2 is DLHashTable2 with hardcoded Uint32 key named "key".
+ */
+template <class T>
+class KeyTable : public DLHashTable<T> {
+public:
+ KeyTable(ArrayPool<T>& pool) :
+ DLHashTable<T>(pool) {
+ }
+
+ bool find(Ptr<T>& ptr, const T& rec) const {
+ return DLHashTable<T>::find(ptr, rec);
+ }
+
+ bool find(Ptr<T>& ptr, Uint32 key) const {
+ T rec;
+ rec.key = key;
+ return DLHashTable<T>::find(ptr, rec);
+ }
+};
+
+#endif
diff --git a/ndb/src/kernel/vm/KeyTable2.hpp b/ndb/src/kernel/vm/KeyTable2.hpp
new file mode 100644
index 00000000000..5c2b3096abe
--- /dev/null
+++ b/ndb/src/kernel/vm/KeyTable2.hpp
@@ -0,0 +1,43 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef KEY_TABLE2_HPP
+#define KEY_TABLE2_HPP
+
+#include <DLHashTable2.hpp>
+
+/**
+ * KeyTable2 is DLHashTable2 with hardcoded Uint32 key named "key".
+ */
+template <class T, class U>
+class KeyTable2 : public DLHashTable2<T, U> {
+public:
+ KeyTable2(ArrayPool<U>& pool) :
+ DLHashTable2<T, U>(pool) {
+ }
+
+ bool find(Ptr<T>& ptr, const T& rec) const {
+ return DLHashTable2<T, U>::find(ptr, rec);
+ }
+
+ bool find(Ptr<T>& ptr, Uint32 key) const {
+ T rec;
+ rec.key = key;
+ return DLHashTable2<T, U>::find(ptr, rec);
+ }
+};
+
+#endif
diff --git a/ndb/src/kernel/vm/LongSignal.hpp b/ndb/src/kernel/vm/LongSignal.hpp
new file mode 100644
index 00000000000..dfbfdb456da
--- /dev/null
+++ b/ndb/src/kernel/vm/LongSignal.hpp
@@ -0,0 +1,79 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LONG_SIGNAL_HPP
+#define LONG_SIGNAL_HPP
+
+#include "pc.hpp"
+#include <ArrayPool.hpp>
+
+/**
+ * Section handling
+ */
+struct SectionSegment {
+
+ static const Uint32 DataLength = 60;
+
+ Uint32 m_ownerRef;
+ Uint32 m_sz;
+ Uint32 m_lastSegment;
+ union {
+ Uint32 m_nextSegment;
+ Uint32 nextPool;
+ };
+ Uint32 theData[DataLength];
+};
+
+/**
+ * Pool for SectionSegments
+ */
+class SectionSegmentPool : public ArrayPool<SectionSegment> {};
+
+/**
+ * And the instance
+ */
+extern SectionSegmentPool g_sectionSegmentPool;
+
+/**
+ * Function prototypes
+ */
+void print(SegmentedSectionPtr ptr, FILE* out);
+void copy(SegmentedSectionPtr dst, Uint32 * src, Uint32 len);
+void copy(Uint32 * dst, SegmentedSectionPtr src);
+
+extern class SectionSegmentPool g_sectionSegmentPool;
+void getSection(SegmentedSectionPtr & ptr, Uint32 id);
+void linkSegments(Uint32 head, Uint32 tail);
+
+void getSections(Uint32 secCount, SegmentedSectionPtr ptr[3]);
+void releaseSections(Uint32 secCount, SegmentedSectionPtr ptr[3]);
+
+
+#include "DataBuffer.hpp"
+
+template<Uint32 sz>
+void
+append(DataBuffer<sz>& dst, SegmentedSectionPtr ptr, SectionSegmentPool& pool){
+ Uint32 len = ptr.sz;
+ while(len > SectionSegment::DataLength){
+ dst.append(ptr.p->theData, SectionSegment::DataLength);
+ ptr.p = pool.getPtr(ptr.p->m_nextSegment);
+ len -= SectionSegment::DataLength;
+ }
+ dst.append(ptr.p->theData, len);
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/Makefile b/ndb/src/kernel/vm/Makefile
new file mode 100644
index 00000000000..3f448b77b17
--- /dev/null
+++ b/ndb/src/kernel/vm/Makefile
@@ -0,0 +1,29 @@
+include .defs.mk
+
+TYPE := kernel
+
+ARCHIVE_TARGET := kernel
+
+SOURCES = \
+ SimulatedBlock.cpp \
+ FastScheduler.cpp \
+ TimeQueue.cpp \
+ VMSignal.cpp \
+ ThreadConfig.cpp \
+ TransporterCallback.cpp \
+ Emulator.cpp \
+ Configuration.cpp \
+ ClusterConfiguration.cpp \
+ WatchDog.cpp \
+ SimplePropertiesSection.cpp \
+ SectionReader.cpp \
+ MetaData.cpp \
+ Mutex.cpp SafeCounter.cpp
+
+DIRS := testCopy testDataBuffer testSimplePropertiesSection
+ifneq ($(USE_EDITLINE), N)
+DIRS += testLongSig
+endif
+
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/vm/MetaData.cpp b/ndb/src/kernel/vm/MetaData.cpp
new file mode 100644
index 00000000000..bcde6c63272
--- /dev/null
+++ b/ndb/src/kernel/vm/MetaData.cpp
@@ -0,0 +1,113 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "MetaData.hpp"
+#include "SimulatedBlock.hpp"
+#include <blocks/dbdict/Dbdict.hpp>
+#include <blocks/dbdih/Dbdih.hpp>
+
+// MetaData::Common
+
+MetaData::Common::Common(Dbdict& dbdict, Dbdih& dbdih) :
+ m_dbdict(dbdict),
+ m_dbdih(dbdih)
+{
+ m_lock[false] = m_lock[true] = 0;
+}
+
+// MetaData::Table
+
+// MetaData
+
+MetaData::MetaData(Common& common) :
+ m_common(common)
+{
+ m_lock[false] = m_lock[true] = 0;
+}
+
+MetaData::MetaData(SimulatedBlock* block) :
+ m_common(*block->getMetaDataCommon())
+{
+ m_lock[false] = m_lock[true] = 0;
+}
+
+MetaData::~MetaData()
+{
+ for (int i = false; i <= true; i++) {
+ NDB_ASSERT(m_common.m_lock[i] >= m_lock[i], "invalid lock count");
+ m_common.m_lock[i] -= m_lock[i];
+ m_lock[i] = 0;
+ }
+}
+
+int
+MetaData::lock(bool exclusive)
+{
+ if (m_common.m_lock[true] > m_lock[true]) {
+ // locked exclusively by another instance
+ return MetaData::Locked;
+ }
+ m_lock[exclusive]++;
+ m_common.m_lock[exclusive]++;
+ return 0;
+}
+
+int
+MetaData::unlock(bool exclusive)
+{
+ if (m_lock[exclusive] == 0) {
+ return MetaData::NotLocked;
+ }
+ m_lock[exclusive]--;
+ m_common.m_lock[exclusive]--;
+ return 0;
+}
+
+int
+MetaData::getTable(MetaData::Table& table, Uint32 tableId, Uint32 tableVersion)
+{
+ if (m_lock[false] + m_lock[true] == 0) {
+ return MetaData::NotLocked;
+ }
+ return m_common.m_dbdict.getMetaTable(table, tableId, tableVersion);
+}
+
+int
+MetaData::getTable(MetaData::Table& table, const char* tableName)
+{
+ if (m_lock[false] + m_lock[true] == 0) {
+ return MetaData::NotLocked;
+ }
+ return m_common.m_dbdict.getMetaTable(table, tableName);
+}
+
+int
+MetaData::getAttribute(MetaData::Attribute& attribute, const MetaData::Table& table, Uint32 attributeId)
+{
+ if (m_lock[false] + m_lock[true] == 0) {
+ return MetaData::NotLocked;
+ }
+ return m_common.m_dbdict.getMetaAttribute(attribute, table, attributeId);
+}
+
+int
+MetaData::getAttribute(MetaData::Attribute& attribute, const MetaData::Table& table, const char* attributeName)
+{
+ if (m_lock[false] + m_lock[true] == 0) {
+ return MetaData::NotLocked;
+ }
+ return m_common.m_dbdict.getMetaAttribute(attribute, table, attributeName);
+}
diff --git a/ndb/src/kernel/vm/MetaData.hpp b/ndb/src/kernel/vm/MetaData.hpp
new file mode 100644
index 00000000000..f6a941e8f9f
--- /dev/null
+++ b/ndb/src/kernel/vm/MetaData.hpp
@@ -0,0 +1,247 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef KERNEL_VM_METADATA_HPP
+#define KERNEL_VM_METADATA_HPP
+
+#include <ndb_types.h>
+#include <ndb_limits.h>
+#include <ErrorReporter.hpp>
+#include <signaldata/DictTabInfo.hpp>
+
+class SimulatedBlock;
+class Dbdict;
+class Dbdih;
+
+/*
+ * Common metadata for all blocks on the node.
+ *
+ * A database node has unique DICT and DIH instances. Parts of their
+ * metadata are described by subclasses of MetaData. Any block can
+ * access the metadata via a MetaData instance.
+ */
+class MetaData {
+public:
+ /*
+ * Methods return < 0 on error.
+ */
+ enum Error {
+ Locked = -1,
+ NotLocked = -2,
+ InvalidArgument = -3,
+ TableNotFound = -4,
+ InvalidTableVersion = -5,
+ AttributeNotFound = -6
+ };
+
+ /*
+ * Common data shared by all metadata instances. Contains DICT and
+ * DIH pointers and counts of shared and exclusive locks.
+ */
+ class Common {
+ public:
+ Common(Dbdict& dbdict, Dbdih& dbdih);
+ private:
+ friend class MetaData;
+ Dbdict& m_dbdict;
+ Dbdih& m_dbdih;
+ unsigned m_lock[2]; // shared: 0 (false), exclusive: 1 (true)
+ };
+
+ /*
+ * Table metadata. A base class of Dbdict::TableRecord. This is
+ * actually fragment metadata but until "alter table" there is no
+ * difference.
+ */
+ class Table {
+ public:
+ /* Table id (array index in DICT and other blocks) */
+ Uint32 tableId;
+
+ /* Table version (incremented when tableId is re-used) */
+ Uint32 tableVersion;
+
+ /* Table name (may not be unique under "alter table") */
+ char tableName[MAX_TAB_NAME_SIZE];
+
+ /* Type of table or index */
+ DictTabInfo::TableType tableType;
+
+ /* Is table or index online (this flag is not used in DICT) */
+ bool online;
+
+ /* Primary table of index otherwise RNIL */
+ Uint32 primaryTableId;
+
+ /* Type of storage (memory/disk, not used) */
+ DictTabInfo::StorageType storageType;
+
+ /* Type of fragmentation (small/medium/large) */
+ DictTabInfo::FragmentType fragmentType;
+
+ /* Key type of fragmentation (pk/dist key/dist group) */
+ DictTabInfo::FragmentKeyType fragmentKeyType;
+
+ /* Global checkpoint identity when table created */
+ Uint32 gciTableCreated;
+
+ /* Number of attibutes in table */
+ Uint16 noOfAttributes;
+
+ /* Number of null attributes in table (should be computed) */
+ Uint16 noOfNullAttr;
+
+ /* Number of primary key attributes (should be computed) */
+ Uint16 noOfPrimkey;
+
+ /* Length of primary key in words (should be computed) */
+ /* For ordered index this is tree node size in words */
+ Uint16 tupKeyLength;
+
+ /* K value for LH**3 algorithm (only 6 allowed currently) */
+ Uint8 kValue;
+
+ /* Local key length in words (currently 1) */
+ Uint8 localKeyLen;
+
+ /*
+ * Parameter for hash algorithm that specifies the load factor in
+ * percentage of fill level in buckets. A high value means we are
+ * splitting early and that buckets are only lightly used. A high
+ * value means that we have fill the buckets more and get more
+ * likelihood of overflow buckets.
+ */
+ Uint8 maxLoadFactor;
+
+ /*
+ * Used when shrinking to decide when to merge buckets. Hysteresis
+ * is thus possible. Should be smaller but not much smaller than
+ * maxLoadFactor
+ */
+ Uint8 minLoadFactor;
+
+ /* Is the table logged (i.e. data survives system restart) */
+ bool storedTable;
+
+ /* Convenience routines */
+ bool isTable() const;
+ bool isIndex() const;
+ bool isUniqueIndex() const;
+ bool isNonUniqueIndex() const;
+ bool isHashIndex() const;
+ bool isOrderedIndex() const;
+ };
+
+ /*
+ * Attribute metadata. A base class of Dbdict::AttributeRecord.
+ */
+ class Attribute {
+ public:
+ /* Attribute id within table (counted from 0) */
+ Uint16 attributeId;
+
+ /* Attribute number within tuple key (counted from 1) */
+ Uint16 tupleKey;
+
+ /* Attribute name (unique within table) */
+ char attributeName[MAX_ATTR_NAME_SIZE];
+
+ /* Attribute description (old-style packed descriptor) */
+ Uint32 attributeDescriptor;
+
+ /* Extended attributes */
+ Uint32 extType;
+ Uint32 extPrecision;
+ Uint32 extScale;
+ Uint32 extLength;
+
+ /* Autoincrement flag, only for ODBC/SQL */
+ bool autoIncrement;
+
+ /* Default value as null-terminated string, only for ODBC/SQL */
+ char defaultValue[MAX_ATTR_DEFAULT_VALUE_SIZE];
+ };
+
+ /*
+ * Metadata is accessed via a MetaData instance. The constructor
+ * needs a reference to MetaData::Common which can be obtained via
+ * the block. The destructor releases any leftover locks.
+ */
+ MetaData(Common& common);
+ MetaData(SimulatedBlock* block);
+ ~MetaData();
+
+ /*
+ * Access methods. Locking can be shared (read) or exclusive (write).
+ * Locking can be recursive (a count is kept). Example (in a block):
+ *
+ * MetaData md(this);
+ * MetaData::Table table;
+ * ret = md.lock(false);
+ * ret = md.getTable(table, "SYSTAB_0");
+ * ret = md.unlock();
+ */
+ int lock(bool exclusive);
+ int unlock(bool exclusive);
+ int getTable(MetaData::Table& table, Uint32 tableId, Uint32 tableVersion);
+ int getTable(MetaData::Table& table, const char* tableName);
+ int getAttribute(MetaData::Attribute& attribute, const MetaData::Table& table, Uint32 attributeId);
+ int getAttribute(MetaData::Attribute& attribute, const MetaData::Table& table, const char* attributeName);
+
+private:
+ Common& m_common;
+ unsigned m_lock[2];
+};
+
+// MetaData::Table
+
+inline bool
+MetaData::Table::isTable() const
+{
+ return DictTabInfo::isTable(tableType);
+}
+
+inline bool
+MetaData::Table::isIndex() const
+{
+ return DictTabInfo::isIndex(tableType);
+}
+
+inline bool
+MetaData::Table::isUniqueIndex() const
+{
+ return DictTabInfo::isUniqueIndex(tableType);
+}
+
+inline bool
+MetaData::Table::isNonUniqueIndex() const
+{
+ return DictTabInfo::isNonUniqueIndex(tableType);
+}
+
+inline bool
+MetaData::Table::isHashIndex() const
+{
+ return DictTabInfo::isHashIndex(tableType);
+}
+
+inline bool
+MetaData::Table::isOrderedIndex() const
+{
+ return DictTabInfo::isOrderedIndex(tableType);
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/Mutex.cpp b/ndb/src/kernel/vm/Mutex.cpp
new file mode 100644
index 00000000000..1dbc6e7ec4a
--- /dev/null
+++ b/ndb/src/kernel/vm/Mutex.cpp
@@ -0,0 +1,282 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "SimulatedBlock.hpp"
+#include "Mutex.hpp"
+#include <signaldata/UtilLock.hpp>
+
+MutexManager::MutexManager(class SimulatedBlock & block)
+ : m_block(block),
+ m_activeMutexes(m_mutexPool) {
+}
+
+bool
+MutexManager::setSize(Uint32 maxNoOfActiveMutexes){
+ return m_mutexPool.setSize(maxNoOfActiveMutexes);
+}
+
+Uint32
+MutexManager::getSize() const {
+ return m_mutexPool.getSize();
+}
+
+bool
+MutexManager::seize(ActiveMutexPtr& ptr){
+ return m_activeMutexes.seize(ptr);
+}
+
+void
+MutexManager::release(Uint32 activeMutexPtrI){
+ m_activeMutexes.release(activeMutexPtrI);
+}
+
+void
+MutexManager::getPtr(ActiveMutexPtr& ptr){
+ m_activeMutexes.getPtr(ptr);
+}
+
+BlockReference
+MutexManager::reference() const {
+ return m_block.reference();
+}
+
+void
+MutexManager::progError(int line, int err_code, const char* extra) {
+ m_block.progError(line, err_code, extra);
+}
+
+void
+MutexManager::create(Signal* signal, ActiveMutexPtr& ptr){
+
+ UtilCreateLockReq * req = (UtilCreateLockReq*)signal->getDataPtrSend();
+ req->senderData = ptr.i;
+ req->senderRef = m_block.reference();
+ req->lockId = ptr.p->m_mutexId;
+ req->lockType = UtilCreateLockReq::Mutex;
+
+ m_block.sendSignal(DBUTIL_REF,
+ GSN_UTIL_CREATE_LOCK_REQ,
+ signal,
+ UtilCreateLockReq::SignalLength,
+ JBB);
+
+ ptr.p->m_gsn = GSN_UTIL_CREATE_LOCK_REQ;
+}
+
+void
+MutexManager::execUTIL_CREATE_LOCK_REF(Signal* signal){
+
+ UtilCreateLockRef * ref = (UtilCreateLockRef*)signal->getDataPtr();
+ ActiveMutexPtr ptr;
+ m_activeMutexes.getPtr(ptr, ref->senderData);
+ ndbrequire(ptr.p->m_gsn == GSN_UTIL_CREATE_LOCK_REQ);
+ ndbrequire(ptr.p->m_mutexId == ref->lockId);
+
+ ptr.p->m_gsn = 0;
+ m_block.execute(signal, ptr.p->m_callback, ref->errorCode);
+}
+
+void
+MutexManager::execUTIL_CREATE_LOCK_CONF(Signal* signal){
+
+ UtilCreateLockConf * conf = (UtilCreateLockConf*)signal->getDataPtr();
+ ActiveMutexPtr ptr;
+ m_activeMutexes.getPtr(ptr, conf->senderData);
+ ndbrequire(ptr.p->m_gsn == GSN_UTIL_CREATE_LOCK_REQ);
+ ndbrequire(ptr.p->m_mutexId == conf->lockId);
+
+ ptr.p->m_gsn = 0;
+ m_block.execute(signal, ptr.p->m_callback, 0);
+}
+
+
+void
+MutexManager::destroy(Signal* signal, ActiveMutexPtr& ptr){
+
+ UtilDestroyLockReq * req = (UtilDestroyLockReq*)signal->getDataPtrSend();
+ req->senderData = ptr.i;
+ req->senderRef = m_block.reference();
+ req->lockId = ptr.p->m_mutexId;
+ req->lockKey = ptr.p->m_mutexKey;
+
+ m_block.sendSignal(DBUTIL_REF,
+ GSN_UTIL_DESTROY_LOCK_REQ,
+ signal,
+ UtilDestroyLockReq::SignalLength,
+ JBB);
+
+ ptr.p->m_gsn = GSN_UTIL_DESTROY_LOCK_REQ;
+}
+
+void
+MutexManager::execUTIL_DESTORY_LOCK_REF(Signal* signal){
+ UtilDestroyLockRef * ref = (UtilDestroyLockRef*)signal->getDataPtr();
+ ActiveMutexPtr ptr;
+ m_activeMutexes.getPtr(ptr, ref->senderData);
+ ndbrequire(ptr.p->m_gsn == GSN_UTIL_DESTROY_LOCK_REQ);
+ ndbrequire(ptr.p->m_mutexId == ref->lockId);
+
+ ptr.p->m_gsn = 0;
+ m_block.execute(signal, ptr.p->m_callback, ref->errorCode);
+}
+
+void
+MutexManager::execUTIL_DESTORY_LOCK_CONF(Signal* signal){
+ UtilDestroyLockConf * conf = (UtilDestroyLockConf*)signal->getDataPtr();
+ ActiveMutexPtr ptr;
+ m_activeMutexes.getPtr(ptr, conf->senderData);
+ ndbrequire(ptr.p->m_gsn == GSN_UTIL_DESTROY_LOCK_REQ);
+ ndbrequire(ptr.p->m_mutexId == conf->lockId);
+
+ ptr.p->m_gsn = 0;
+ m_block.execute(signal, ptr.p->m_callback, 0);
+}
+
+
+void
+MutexManager::lock(Signal* signal, ActiveMutexPtr& ptr){
+
+ UtilLockReq * req = (UtilLockReq*)signal->getDataPtrSend();
+ req->senderData = ptr.i;
+ req->senderRef = m_block.reference();
+ req->lockId = ptr.p->m_mutexId;
+ req->requestInfo = 0;
+
+ m_block.sendSignal(DBUTIL_REF,
+ GSN_UTIL_LOCK_REQ,
+ signal,
+ UtilLockReq::SignalLength,
+ JBB);
+
+ ptr.p->m_gsn = GSN_UTIL_LOCK_REQ;
+}
+
+void
+MutexManager::trylock(Signal* signal, ActiveMutexPtr& ptr){
+
+ UtilLockReq * req = (UtilLockReq*)signal->getDataPtrSend();
+ req->senderData = ptr.i;
+ req->senderRef = m_block.reference();
+ req->lockId = ptr.p->m_mutexId;
+ req->requestInfo = UtilLockReq::TryLock;
+
+ m_block.sendSignal(DBUTIL_REF,
+ GSN_UTIL_LOCK_REQ,
+ signal,
+ UtilLockReq::SignalLength,
+ JBB);
+
+ ptr.p->m_gsn = GSN_UTIL_LOCK_REQ;
+}
+
+void
+MutexManager::execUTIL_LOCK_REF(Signal* signal){
+ UtilLockRef * ref = (UtilLockRef*)signal->getDataPtr();
+ ActiveMutexPtr ptr;
+ m_activeMutexes.getPtr(ptr, ref->senderData);
+ ndbrequire(ptr.p->m_gsn == GSN_UTIL_LOCK_REQ);
+ ndbrequire(ptr.p->m_mutexId == ref->lockId);
+
+ ptr.p->m_gsn = 0;
+ m_block.execute(signal, ptr.p->m_callback, ref->errorCode);
+}
+
+void
+MutexManager::execUTIL_LOCK_CONF(Signal* signal){
+ UtilLockConf * conf = (UtilLockConf*)signal->getDataPtr();
+ ActiveMutexPtr ptr;
+ m_activeMutexes.getPtr(ptr, conf->senderData);
+ ndbrequire(ptr.p->m_gsn == GSN_UTIL_LOCK_REQ);
+ ndbrequire(ptr.p->m_mutexId == conf->lockId);
+
+ ptr.p->m_mutexKey = conf->lockKey;
+
+ ptr.p->m_gsn = 0;
+ m_block.execute(signal, ptr.p->m_callback, 0);
+}
+
+void
+MutexManager::unlock(Signal* signal, ActiveMutexPtr& ptr){
+ UtilUnlockReq * req = (UtilUnlockReq*)signal->getDataPtrSend();
+ req->senderData = ptr.i;
+ req->senderRef = m_block.reference();
+ req->lockId = ptr.p->m_mutexId;
+ req->lockKey = ptr.p->m_mutexKey;
+
+ m_block.sendSignal(DBUTIL_REF,
+ GSN_UTIL_UNLOCK_REQ,
+ signal,
+ UtilUnlockReq::SignalLength,
+ JBB);
+
+ ptr.p->m_gsn = GSN_UTIL_UNLOCK_REQ;
+}
+
+void
+MutexManager::execUTIL_UNLOCK_REF(Signal* signal){
+ UtilUnlockRef * ref = (UtilUnlockRef*)signal->getDataPtr();
+ ActiveMutexPtr ptr;
+ m_activeMutexes.getPtr(ptr, ref->senderData);
+ ndbrequire(ptr.p->m_gsn == GSN_UTIL_UNLOCK_REQ);
+ ndbrequire(ptr.p->m_mutexId == ref->lockId);
+
+ ptr.p->m_gsn = 0;
+ m_block.execute(signal, ptr.p->m_callback, ref->errorCode);
+}
+
+void
+MutexManager::execUTIL_UNLOCK_CONF(Signal* signal){
+ UtilUnlockConf * conf = (UtilUnlockConf*)signal->getDataPtr();
+ ActiveMutexPtr ptr;
+ m_activeMutexes.getPtr(ptr, conf->senderData);
+ ndbrequire(ptr.p->m_gsn == GSN_UTIL_UNLOCK_REQ);
+ ndbrequire(ptr.p->m_mutexId == conf->lockId);
+
+ ptr.p->m_gsn = 0;
+ m_block.execute(signal, ptr.p->m_callback, 0);
+}
+
+void
+Mutex::release(MutexManager& mgr, Uint32 activePtrI, Uint32 mutexId){
+ MutexManager::ActiveMutexPtr ptr;
+ ptr.i = activePtrI;
+ mgr.getPtr(ptr);
+ if(ptr.p->m_gsn == 0 && ptr.p->m_mutexId == mutexId){
+ mgr.release(activePtrI);
+ return;
+ }
+
+ if(ptr.p->m_mutexId != mutexId)
+ ErrorReporter::handleAssert("MutexHandle::release invalid handle",
+ __FILE__, __LINE__);
+ ErrorReporter::handleAssert("MutexHandle::release of mutex inuse",
+ __FILE__, __LINE__);
+}
+
+void
+Mutex::unlock(){
+ if(!m_ptr.isNull()){
+ m_mgr.getPtr(m_ptr);
+ if(m_ptr.p->m_mutexId == m_mutexId){
+ Callback c = { &SimulatedBlock::ignoreMutexUnlockCallback, m_ptr.i };
+ m_ptr.p->m_callback = c;
+ m_mgr.unlock(m_signal, m_ptr);
+ m_ptr.setNull(); // Remove reference
+ }
+ }
+}
+
diff --git a/ndb/src/kernel/vm/Mutex.hpp b/ndb/src/kernel/vm/Mutex.hpp
new file mode 100644
index 00000000000..40e3fb56b4f
--- /dev/null
+++ b/ndb/src/kernel/vm/Mutex.hpp
@@ -0,0 +1,321 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BLOCK_MUTEX_HPP
+#define BLOCK_MUTEX_HPP
+
+#include "Callback.hpp"
+#include "SimulatedBlock.hpp"
+
+class Mutex;
+
+class MutexManager {
+ friend class Mutex;
+ friend class SimulatedBlock;
+ friend class DbUtil;
+public:
+ MutexManager(class SimulatedBlock &);
+
+ bool setSize(Uint32 maxNoOfActiveMutexes);
+ Uint32 getSize() const ; // Get maxNoOfActiveMutexes
+
+private:
+ /**
+ * core interface
+ */
+ struct ActiveMutex {
+ Uint32 m_gsn; // state
+ Uint32 m_mutexId;
+ Uint32 m_mutexKey;
+ Callback m_callback;
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+ };
+ typedef Ptr<ActiveMutex> ActiveMutexPtr;
+
+ bool seize(ActiveMutexPtr& ptr);
+ void release(Uint32 activeMutexPtrI);
+
+ void getPtr(ActiveMutexPtr& ptr);
+
+ void create(Signal*, ActiveMutexPtr&);
+ void destroy(Signal*, ActiveMutexPtr&);
+ void lock(Signal*, ActiveMutexPtr&);
+ void trylock(Signal*, ActiveMutexPtr&);
+ void unlock(Signal*, ActiveMutexPtr&);
+
+private:
+ void execUTIL_CREATE_LOCK_REF(Signal* signal);
+ void execUTIL_CREATE_LOCK_CONF(Signal* signal);
+ void execUTIL_DESTORY_LOCK_REF(Signal* signal);
+ void execUTIL_DESTORY_LOCK_CONF(Signal* signal);
+ void execUTIL_LOCK_REF(Signal* signal);
+ void execUTIL_LOCK_CONF(Signal* signal);
+ void execUTIL_UNLOCK_REF(Signal* signal);
+ void execUTIL_UNLOCK_CONF(Signal* signal);
+
+ SimulatedBlock & m_block;
+ ArrayPool<ActiveMutex> m_mutexPool;
+ DLList<ActiveMutex> m_activeMutexes;
+
+ BlockReference reference() const;
+ void progError(int line, int err_code, const char* extra = 0);
+};
+
+
+/**
+ * MutexHandle - A "reference" to a mutex
+ * - Should be used together with Mutex
+ */
+class MutexHandle {
+ friend class Mutex;
+public:
+ MutexHandle(Uint32 id);
+
+ bool isNull() const;
+ void release(MutexManager & mgr);
+
+private:
+ const Uint32 m_mutexId;
+ Uint32 m_activeMutexPtrI;
+};
+
+/**
+ * MutexHandle2 - A template-based "reference" to a mutex
+ */
+template<Uint32 MutexId>
+class MutexHandle2 {
+ friend class Mutex;
+public:
+ MutexHandle2();
+
+ bool isNull() const;
+ void release(MutexManager & mgr);
+
+private:
+ Uint32 m_activeMutexPtrI;
+};
+
+/**
+ * A mutex - Used together with a MutexHandle to be put on the stack
+ */
+class Mutex {
+public:
+ Mutex(Signal*, MutexManager & mgr, MutexHandle &);
+
+ template<Uint32 MutexId>
+ Mutex(Signal*, MutexManager & mgr, MutexHandle2<MutexId> &);
+
+ ~Mutex();
+
+ void release();
+ bool isNull() const ;
+
+ bool lock(Callback & callback);
+ bool trylock(Callback & callback);
+ void unlock(Callback & callback);
+ void unlock(); // Ignore callback
+
+ bool create(Callback & callback);
+ bool destroy(Callback & callback);
+
+private:
+ Signal* m_signal;
+ MutexManager & m_mgr;
+ const Uint32 m_mutexId;
+ Uint32 & m_srcPtrI;
+ MutexManager::ActiveMutexPtr m_ptr;
+
+public:
+ static void release(MutexManager&, Uint32 activePtrI, Uint32 mutexId);
+};
+
+inline
+MutexHandle::MutexHandle(Uint32 id) : m_mutexId(id) {
+ m_activeMutexPtrI = RNIL;
+}
+
+inline
+bool
+MutexHandle::isNull() const {
+ return m_activeMutexPtrI == RNIL;
+}
+
+inline
+void
+MutexHandle::release(MutexManager & mgr){
+ if(!isNull()){
+ Mutex::release(mgr, m_activeMutexPtrI, m_mutexId);
+ m_activeMutexPtrI = RNIL;
+ }
+}
+
+template<Uint32 MutexId>
+inline
+MutexHandle2<MutexId>::MutexHandle2() {
+ m_activeMutexPtrI = RNIL;
+}
+
+template<Uint32 MutexId>
+inline
+bool
+MutexHandle2<MutexId>::isNull() const {
+ return m_activeMutexPtrI == RNIL;
+}
+
+
+template<Uint32 MutexId>
+inline
+void
+MutexHandle2<MutexId>::release(MutexManager & mgr){
+ if(!isNull()){
+ Mutex::release(mgr, m_activeMutexPtrI, MutexId);
+ m_activeMutexPtrI = RNIL;
+ }
+}
+
+
+inline
+Mutex::Mutex(Signal* signal, MutexManager & mgr, MutexHandle & mh)
+ : m_signal(signal),
+ m_mgr(mgr),
+ m_mutexId(mh.m_mutexId),
+ m_srcPtrI(mh.m_activeMutexPtrI){
+
+ m_ptr.i = m_srcPtrI;
+
+}
+
+template<Uint32 MutexId>
+inline
+Mutex::Mutex(Signal* signal, MutexManager & mgr, MutexHandle2<MutexId> & mh)
+ : m_signal(signal),
+ m_mgr(mgr),
+ m_mutexId(MutexId),
+ m_srcPtrI(mh.m_activeMutexPtrI){
+
+ m_ptr.i = m_srcPtrI;
+
+}
+
+inline
+Mutex::~Mutex(){
+ m_srcPtrI = m_ptr.i;
+}
+
+inline
+void
+Mutex::release(){
+ if(!m_ptr.isNull()){
+ Mutex::release(m_mgr, m_ptr.i, m_mutexId);
+ m_ptr.setNull();
+ }
+}
+
+inline
+bool
+Mutex::isNull() const {
+ return m_ptr.isNull();
+}
+
+inline
+bool
+Mutex::lock(Callback & callback){
+ if(m_ptr.isNull()){
+ if(m_mgr.seize(m_ptr)){
+ m_ptr.p->m_mutexId = m_mutexId;
+ m_ptr.p->m_callback = callback;
+ m_mgr.lock(m_signal, m_ptr);
+ return true;
+ }
+ return false;
+ }
+ ErrorReporter::handleAssert("Mutex::lock mutex alreay inuse",
+ __FILE__, __LINE__);
+ return false;
+}
+
+inline
+bool
+Mutex::trylock(Callback & callback){
+ if(m_ptr.isNull()){
+ if(m_mgr.seize(m_ptr)){
+ m_ptr.p->m_mutexId = m_mutexId;
+ m_ptr.p->m_callback = callback;
+ m_mgr.lock(m_signal, m_ptr);
+ return true;
+ }
+ return false;
+ }
+ ErrorReporter::handleAssert("Mutex::trylock mutex alreay inuse",
+ __FILE__, __LINE__);
+ return false;
+}
+
+inline
+void
+Mutex::unlock(Callback & callback){
+ if(!m_ptr.isNull()){
+ m_mgr.getPtr(m_ptr);
+ if(m_ptr.p->m_mutexId == m_mutexId){
+ m_ptr.p->m_callback = callback;
+ m_mgr.unlock(m_signal, m_ptr);
+ return;
+ }
+ }
+ ErrorReporter::handleAssert("Mutex::unlock invalid mutex",
+ __FILE__, __LINE__);
+}
+
+inline
+bool
+Mutex::create(Callback & callback){
+ if(m_ptr.isNull()){
+ if(m_mgr.seize(m_ptr)){
+ m_ptr.p->m_mutexId = m_mutexId;
+ m_ptr.p->m_callback = callback;
+ m_mgr.create(m_signal, m_ptr);
+ return true;
+ }
+ return false;
+ }
+ ErrorReporter::handleAssert("Mutex::create mutex alreay inuse",
+ __FILE__, __LINE__);
+ return false;
+}
+
+inline
+bool
+Mutex::destroy(Callback & callback){
+ if(m_ptr.isNull()){
+ if(m_mgr.seize(m_ptr)){
+ m_ptr.p->m_mutexId = m_mutexId;
+ m_ptr.p->m_callback = callback;
+ m_mgr.destroy(m_signal, m_ptr);
+ return true;
+ }
+ return false;
+ }
+ ErrorReporter::handleAssert("Mutex::destroy mutex alreay inuse",
+ __FILE__, __LINE__);
+ return false;
+}
+
+
+#endif
diff --git a/ndb/src/kernel/vm/Prio.hpp b/ndb/src/kernel/vm/Prio.hpp
new file mode 100644
index 00000000000..4c9c22b0afe
--- /dev/null
+++ b/ndb/src/kernel/vm/Prio.hpp
@@ -0,0 +1,32 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PRIO_H
+#define PRIO_H
+
+enum JobBufferLevel {
+ JBA = 0,
+ JBB = 1,
+ JBC = 2,
+ JBD = 3, LEVEL_IDLE = 3,
+ JB_LEVELS,
+ ILLEGAL_JB_LEVEL
+};
+
+typedef JobBufferLevel Priority;
+
+
+#endif
diff --git a/ndb/src/kernel/vm/RequestTracker.hpp b/ndb/src/kernel/vm/RequestTracker.hpp
new file mode 100644
index 00000000000..5fd1ae7255a
--- /dev/null
+++ b/ndb/src/kernel/vm/RequestTracker.hpp
@@ -0,0 +1,58 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __REQUEST_TRACKER_HPP
+#define __REQUEST_TRACKER_HPP
+
+#include "SafeCounter.hpp"
+
+class RequestTracker {
+public:
+ RequestTracker(){ init(); }
+
+ void init() { m_confs.clear(); m_nRefs = 0; }
+
+ template<typename SignalClass>
+ void init(SafeCounterManager& mgr,
+ NodeReceiverGroup rg, Uint16 GSN, Uint32 senderData)
+ {
+ init();
+ SafeCounter tmp(mgr, m_sc);
+ tmp.init<SignalClass>(rg, GSN, senderData);
+ }
+
+ bool ignoreRef(SafeCounterManager& mgr, Uint32 nodeId)
+ { return m_sc.clearWaitingFor(mgr, nodeId); }
+
+ bool reportRef(SafeCounterManager& mgr, Uint32 nodeId)
+ { m_nRefs++; return m_sc.clearWaitingFor(mgr, nodeId); }
+
+ bool reportConf(SafeCounterManager& mgr, Uint32 nodeId)
+ { m_confs.set(nodeId); return m_sc.clearWaitingFor(mgr, nodeId); }
+
+ bool hasRef() { return m_nRefs != 0; }
+
+ bool hasConf() { return !m_confs.isclear(); }
+
+ bool done() { return m_sc.done(); }
+
+private:
+ SafeCounterHandle m_sc;
+ NodeBitmask m_confs;
+ Uint8 m_nRefs;
+};
+
+#endif // __REQUEST_TRACKER_HPP
diff --git a/ndb/src/kernel/vm/SLList.hpp b/ndb/src/kernel/vm/SLList.hpp
new file mode 100644
index 00000000000..47bc7b8b241
--- /dev/null
+++ b/ndb/src/kernel/vm/SLList.hpp
@@ -0,0 +1,295 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SLLIST_HPP
+#define SLLIST_HPP
+
+#include "ArrayPool.hpp"
+#include <NdbOut.hpp>
+
+/**
+ * Template class used for implementing an
+ * list of object retreived from a pool
+ */
+template <class T>
+class SLList {
+public:
+ /**
+ * List head
+ */
+ struct Head {
+ Head();
+ Uint32 firstItem;
+ };
+
+ SLList(ArrayPool<T> & thePool);
+
+ /**
+ * Allocate an object from pool - update Ptr
+ *
+ * Return i
+ */
+ bool seize(Ptr<T> &);
+
+ /**
+ * Allocate object <b>i</b> from pool - update Ptr
+ *
+ * Return i
+ */
+ bool seizeId(Ptr<T> &, Uint32 i);
+
+ /**
+ * Allocate <b>n</b>objects from pool
+ *
+ * Return i value of first object allocated or RNIL if fails
+ */
+ bool seizeN(Ptr<T> &, Uint32 n);
+
+ /**
+ * Return all objects to the pool
+ */
+ void release();
+
+ /**
+ * Update i & p value according to <b>i</b>
+ */
+ void getPtr(Ptr<T> &, Uint32 i) const;
+
+ /**
+ * Update p value for ptr according to i value
+ */
+ void getPtr(Ptr<T> &) const ;
+
+ /**
+ * Get pointer for i value
+ */
+ T * getPtr(Uint32 i) const ;
+
+ /**
+ * Update ptr to first element in list
+ *
+ * Return i
+ */
+ bool first(Ptr<T> &) const ;
+
+ /**
+ * Get next element
+ *
+ * NOTE ptr must be both p & i
+ */
+ bool next(Ptr<T> &) const ;
+
+ /**
+ * Check if next exists
+ *
+ * NOTE ptr must be both p & i
+ */
+ bool hasNext(const Ptr<T> &) const;
+
+ Uint32 noOfElements() const {
+ Uint32 c = 0;
+ Uint32 i = head.firstItem;
+ while(i != RNIL){
+ c++;
+ const T * t = thePool.getPtr(i);
+ i = t->nextList;
+ }
+ return c;
+ }
+
+ /**
+ * Print
+ * (Run operator NdbOut<< on every element)
+ */
+ void print(NdbOut & out) {
+ out << "firstItem = " << head.firstItem << endl;
+ Uint32 i = head.firstItem;
+ while(i != RNIL){
+ T * t = thePool.getPtr(i);
+ t->print(out); out << " ";
+ i = t->next;
+ }
+ }
+
+ inline bool empty() const { return head.firstItem == RNIL;}
+
+protected:
+ Head head;
+ ArrayPool<T> & thePool;
+};
+
+template<class T>
+class LocalSLList : public SLList<T> {
+public:
+ LocalSLList(ArrayPool<T> & thePool, typename SLList<T>::Head & _src)
+ : SLList<T>(thePool), src(_src)
+ {
+ head = src;
+ }
+
+ ~LocalSLList(){
+ src = head;
+ }
+private:
+ typename SLList<T>::Head & src;
+};
+
+template <class T>
+inline
+SLList<T>::SLList(ArrayPool<T> & _pool):
+ thePool(_pool){
+}
+
+template<class T>
+inline
+SLList<T>::Head::Head(){
+ firstItem = RNIL;
+}
+
+template <class T>
+inline
+bool
+SLList<T>::seize(Ptr<T> & p){
+ thePool.seize(p);
+ T * t = p.p;
+ Uint32 ff = head.firstItem;
+ if(p.i != RNIL){
+ t->nextList = ff;
+ head.firstItem = p.i;
+ return true;
+ }
+ return false;
+}
+
+template <class T>
+inline
+bool
+SLList<T>::seizeId(Ptr<T> & p, Uint32 ir){
+ thePool.seizeId(p, ir);
+ T * t = p.p;
+ Uint32 ff = head.firstItem;
+ if(p.i != RNIL){
+ t->nextList = ff;
+ head.firstItem = p.i;
+ return true;
+ }
+ return false;
+}
+
+template <class T>
+inline
+bool
+SLList<T>::seizeN(Ptr<T> & p, Uint32 n){
+ for(Uint32 i = 0; i < n; i++){
+ if(seize(p) == RNIL){
+ /**
+ * Failure
+ */
+ for(; i > 0; i--){
+ const Uint32 tmp = head.firstItem;
+ const T * t = thePool.getPtr(tmp);
+ head.firstItem = t->nextList;
+ thePool.release(tmp);
+ }
+ return false;
+ }
+ }
+
+ /**
+ * Success
+ */
+ p.i = head.firstItem;
+ p.p = thePool.getPtr(head.firstItem);
+
+ return true;
+}
+
+
+template <class T>
+inline
+void
+SLList<T>::release(){
+ while(head.firstItem != RNIL){
+ const T * t = thePool.getPtr(head.firstItem);
+ const Uint32 i = head.firstItem;
+ head.firstItem = t->nextList;
+ thePool.release(i);
+ }
+}
+
+template <class T>
+inline
+void
+SLList<T>::getPtr(Ptr<T> & p, Uint32 i) const {
+ p.i = i;
+ p.p = thePool.getPtr(i);
+}
+
+template <class T>
+inline
+void
+SLList<T>::getPtr(Ptr<T> & p) const {
+ thePool.getPtr(p);
+}
+
+template <class T>
+inline
+T *
+SLList<T>::getPtr(Uint32 i) const {
+ return thePool.getPtr(i);
+}
+
+/**
+ * Update ptr to first element in list
+ *
+ * Return i
+ */
+template <class T>
+inline
+bool
+SLList<T>::first(Ptr<T> & p) const {
+ Uint32 i = head.firstItem;
+ p.i = i;
+ if(i != RNIL){
+ p.p = thePool.getPtr(i);
+ return true;
+ }
+ p.p = NULL;
+ return false;
+}
+
+template <class T>
+inline
+bool
+SLList<T>::next(Ptr<T> & p) const {
+ Uint32 i = p.p->nextList;
+ p.i = i;
+ if(i != RNIL){
+ p.p = thePool.getPtr(i);
+ return true;
+ }
+ p.p = NULL;
+ return false;
+}
+
+template <class T>
+inline
+bool
+SLList<T>::hasNext(const Ptr<T> & p) const {
+ return p.p->nextList != RNIL;
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/SafeCounter.cpp b/ndb/src/kernel/vm/SafeCounter.cpp
new file mode 100644
index 00000000000..b09ad08b026
--- /dev/null
+++ b/ndb/src/kernel/vm/SafeCounter.cpp
@@ -0,0 +1,159 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "SimulatedBlock.hpp"
+#include "SafeCounter.hpp"
+#include <signaldata/NodeFailRep.hpp>
+
+SafeCounterManager::SafeCounterManager(class SimulatedBlock & block)
+ : m_block(block),
+ m_activeCounters(m_counterPool)
+{}
+
+bool
+SafeCounterManager::setSize(Uint32 maxNoOfActiveMutexes) {
+ return m_counterPool.setSize(maxNoOfActiveMutexes);
+}
+
+Uint32
+SafeCounterManager::getSize() const {
+ return m_counterPool.getSize();
+}
+
+bool
+SafeCounterManager::seize(ActiveCounterPtr& ptr){
+ return m_activeCounters.seize(ptr);
+}
+
+void
+SafeCounterManager::release(ActiveCounterPtr& ptr){
+ m_activeCounters.release(ptr);
+}
+
+void
+SafeCounterManager::getPtr(ActiveCounterPtr& ptr, Uint32 ptrI){
+ m_activeCounters.getPtr(ptr, ptrI);
+}
+
+
+void
+SafeCounterManager::printNODE_FAILREP(){
+ ActiveCounterPtr ptr;
+
+ NodeBitmask nodes;
+ nodes.clear();
+ // nodes.bitORC(nodes);
+
+ for(m_activeCounters.first(ptr); !ptr.isNull(); m_activeCounters.next(ptr)){
+ ActiveCounter::SignalDesc desc = ptr.p->m_signalDesc;
+ ndbout_c("theData[desc.m_senderDataOffset=%u] = %u",
+ desc.m_senderDataOffset, ptr.p->m_senderData);
+ ndbout_c("theData[desc.m_errorCodeOffset=%u] = %u",
+ desc.m_errorCodeOffset, desc.m_nodeFailErrorCode);
+ Uint32 len = MAX(MAX(desc.m_senderDataOffset, desc.m_errorCodeOffset),
+ desc.m_senderRefOffset);
+
+ NodeBitmask overlapping = ptr.p->m_nodes;
+ Uint32 i = 0;
+ while((i = overlapping.find(i)) != NodeBitmask::NotFound){
+ ndbout_c(" theData[desc.m_senderRefOffset=%u] = %x",
+ desc.m_senderRefOffset, numberToRef(desc.m_block, i));
+ ndbout_c(" sendSignal(%x,%u,signal,%u,JBB",
+ m_block.reference(), desc.m_gsn, len+1);
+ i++;
+ }
+ }
+}
+
+void
+SafeCounterManager::execNODE_FAILREP(Signal* signal){
+ Uint32 * theData = signal->getDataPtrSend();
+ ActiveCounterPtr ptr;
+ NodeBitmask nodes;
+ nodes.assign(NodeBitmask::Size,
+ ((const NodeFailRep*)signal->getDataPtr())->theNodes);
+
+ for(m_activeCounters.first(ptr); !ptr.isNull(); m_activeCounters.next(ptr)){
+ if(nodes.overlaps(ptr.p->m_nodes)){
+ ActiveCounter::SignalDesc desc = ptr.p->m_signalDesc;
+ theData[desc.m_senderDataOffset] = ptr.p->m_senderData;
+ theData[desc.m_errorCodeOffset] = desc.m_nodeFailErrorCode;
+ Uint32 len = MAX(MAX(desc.m_senderDataOffset, desc.m_errorCodeOffset),
+ desc.m_senderRefOffset);
+
+ NodeBitmask overlapping = ptr.p->m_nodes;
+ overlapping.bitAND(nodes);
+ Uint32 i = 0;
+ while((i = overlapping.find(i)) != NodeBitmask::NotFound){
+ theData[desc.m_senderRefOffset] = numberToRef(desc.m_block, i);
+ m_block.sendSignal(m_block.reference(), desc.m_gsn, signal, len+1,JBB);
+ i++;
+ }
+ }
+ }
+}
+
+BlockReference
+SafeCounterManager::reference() const {
+ return m_block.reference();
+}
+
+void
+SafeCounterManager::progError(int line, int err_code, const char* extra){
+ m_block.progError(line, err_code, extra);
+}
+
+bool
+SafeCounterHandle::clearWaitingFor(SafeCounterManager& mgr, Uint32 nodeId)
+{
+ SafeCounterManager::ActiveCounterPtr ptr;
+ mgr.getPtr(ptr, m_activeCounterPtrI);
+ ptr.p->m_nodes.clear(nodeId);
+
+ if (ptr.p->m_nodes.isclear()){
+ mgr.release(ptr);
+ m_activeCounterPtrI = RNIL;
+ return true;
+ }
+ return false;
+}
+
+SafeCounter::~SafeCounter(){
+ bool clear = m_count == 0;
+ bool isnull = m_ptr.i == RNIL;
+
+ m_activeCounterPtrI = m_ptr.i;
+
+ if(clear && isnull)
+ return;
+
+ if(clear && !isnull){
+ m_mgr.release(m_ptr);
+ m_activeCounterPtrI = RNIL;
+ return;
+ }
+
+ /**
+ * !clear && !isnull
+ */
+ if(!isnull){
+ m_ptr.p->m_nodes = m_nodes;
+ return;
+ }
+
+ ErrorReporter::handleAssert("~SafeCounter:: wo/ init", __FILE__, __LINE__);
+}
diff --git a/ndb/src/kernel/vm/SafeCounter.hpp b/ndb/src/kernel/vm/SafeCounter.hpp
new file mode 100644
index 00000000000..1f3cc15c2d6
--- /dev/null
+++ b/ndb/src/kernel/vm/SafeCounter.hpp
@@ -0,0 +1,301 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __SAFE_COUNTER_HPP
+#define __SAFE_COUNTER_HPP
+
+/*************************************************************
+ *
+ * SafeCounter "automates" three way to node-fais safe protocols
+ * for "slave" failures. This is done by registing "fake" signals
+ * to be sent in case of nodefailure.
+ *
+ * init<SignalClass>(..., GSN, senderData);
+ *
+ * It is implemented so that one can replace SignalCounter with
+ * SafeCounter (SignalCounter should probably go away with time)
+ * methods:
+ * clearWaitingFor(nodeId);
+ * done();
+ * etc.
+ *
+ * If included in a new block method
+ * SafeCounterManager::execNODE_FAILREP must included in
+ * <block>::execNODE_FAILREP
+ *
+ * the SignalClass must have senderRef, senderData and errorCode
+ * and also ErrorCode::NF_FakeErrorREF, implemented
+ *
+ * SafeCounter consists of 3 parts:
+ * SafeCounterManager which keeps track of active "counters"
+ * SafeCounterHandle to store "i-value" in your "op record"
+ * SafeCounter as a temporary variable only to use on the stack
+ * for operation
+ *
+ */
+
+#include <NodeBitmask.hpp>
+#include "DLList.hpp"
+#include "VMSignal.hpp"
+
+class SimulatedBlock;
+
+/**
+ *
+ */
+class SafeCounterManager {
+ friend class SafeCounter;
+ friend class SafeCounterHandle;
+ friend class SimulatedBlock;
+public:
+ SafeCounterManager(class SimulatedBlock &);
+
+ bool setSize(Uint32 maxNoOfActiveMutexes);
+ Uint32 getSize() const ;
+
+ void execNODE_FAILREP(Signal*);
+ void printNODE_FAILREP();
+
+private:
+ struct ActiveCounter { /** sizeof = 7words = 28bytes */
+ public:
+ Uint32 m_senderData;
+ NodeBitmask m_nodes;
+ struct SignalDesc {
+ public:
+ Uint16 m_gsn;
+ Uint16 m_block;
+ Uint8 m_senderRefOffset;
+ Uint8 m_senderDataOffset;
+ Uint8 m_errorCodeOffset;
+ Uint8 m_nodeFailErrorCode;
+ } m_signalDesc;
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+ };
+
+ typedef Ptr<ActiveCounter> ActiveCounterPtr;
+
+ bool seize(ActiveCounterPtr& ptr);
+ void release(ActiveCounterPtr& ptr);
+ void getPtr(ActiveCounterPtr& ptr, Uint32 ptrI);
+
+ SimulatedBlock & m_block;
+ ArrayPool<ActiveCounter> m_counterPool;
+ DLList<ActiveCounter> m_activeCounters;
+
+ BlockReference reference() const;
+ void progError(int line, int err_code, const char* extra = 0);
+};
+
+
+class SafeCounterHandle {
+ friend class SafeCounter;
+public:
+ SafeCounterHandle();
+
+ /**
+ * Return if done (no nodes set in bitmask)
+ */
+ bool clearWaitingFor(SafeCounterManager& mgr, Uint32 nodeId);
+
+ bool done() const;
+
+private:
+ Uint32 m_activeCounterPtrI;
+};
+
+class SafeCounter {
+ friend class SafeCounterManager;
+public:
+ SafeCounter(SafeCounterManager&, SafeCounterHandle&);
+
+ template<typename SignalClass>
+ bool init(Uint16 block, Uint16 GSN, Uint32 senderData);
+
+ template<typename SignalClass>
+ bool init(NodeReceiverGroup rg, Uint16 GSN, Uint32 senderData);
+
+ template<typename SignalClass>
+ bool init(NodeReceiverGroup rg, Uint32 senderData);
+
+ ~SafeCounter();
+
+ void clearWaitingFor();
+
+ /**
+ * When sending to different node
+ */
+ void setWaitingFor(Uint32 nodeId);
+ bool clearWaitingFor(Uint32 nodeId);
+ bool forceClearWaitingFor(Uint32 nodeId);
+
+ bool isWaitingFor(Uint32 nodeId) const;
+ bool done() const;
+
+ const char * getText() const; /* ? needed for, some portability issues */
+
+ SafeCounter& operator=(const NdbNodeBitmask&);
+ SafeCounter& operator=(const NodeReceiverGroup&);
+private:
+ Uint32 m_count;
+ NodeBitmask m_nodes;
+
+ SafeCounterManager & m_mgr;
+ SafeCounterManager::ActiveCounterPtr m_ptr;
+
+ Uint32 & m_activeCounterPtrI;
+};
+
+inline
+SafeCounterHandle::SafeCounterHandle(){
+ m_activeCounterPtrI = RNIL;
+}
+
+inline
+bool
+SafeCounterHandle::done() const {
+ return m_activeCounterPtrI == RNIL;
+}
+
+inline
+SafeCounter::SafeCounter(SafeCounterManager& mgr, SafeCounterHandle& handle)
+ : m_mgr(mgr),
+ m_activeCounterPtrI(handle.m_activeCounterPtrI)
+{
+ m_ptr.i = handle.m_activeCounterPtrI;
+ if (m_ptr.i == RNIL) {
+ m_nodes.clear();
+ m_count = 0;
+ } else {
+ m_mgr.getPtr(m_ptr, m_ptr.i);
+ m_nodes = m_ptr.p->m_nodes;
+ m_count = m_nodes.count();
+ }
+}
+
+template<typename Ref>
+inline
+bool
+SafeCounter::init(Uint16 block, Uint16 GSN, Uint32 senderData){
+
+ SafeCounterManager::ActiveCounter::SignalDesc signalDesc;
+ signalDesc.m_gsn = GSN;
+ signalDesc.m_block = block;
+ signalDesc.m_errorCodeOffset = offsetof(Ref, errorCode) >> 2;
+ signalDesc.m_senderRefOffset = offsetof(Ref, senderRef) >> 2;
+ signalDesc.m_senderDataOffset = offsetof(Ref, senderData) >> 2;
+ signalDesc.m_nodeFailErrorCode = Ref::NF_FakeErrorREF;
+ assert(((Uint32)Ref::NF_FakeErrorREF) < 256);
+
+ if(m_ptr.i == RNIL){
+ SafeCounterManager::ActiveCounterPtr ptr;
+ if(m_mgr.seize(ptr)){
+ ptr.p->m_senderData = senderData;
+ ptr.p->m_signalDesc = signalDesc;
+ m_ptr = ptr;
+ return true;
+ }
+ return false;
+ }
+
+ if(m_count == 0){
+ m_ptr.p->m_senderData = senderData;
+ m_ptr.p->m_signalDesc = signalDesc;
+ return true;
+ }
+
+ ErrorReporter::handleAssert("SafeCounter::init twice", __FILE__, __LINE__);
+ return false;
+}
+
+template<typename Ref>
+inline
+bool
+SafeCounter::init(NodeReceiverGroup rg, Uint16 GSN, Uint32 senderData){
+
+ bool b = init<Ref>(rg.m_block, GSN, senderData);
+ m_nodes = rg.m_nodes;
+ m_count = m_nodes.count();
+ return b;
+}
+
+template<typename Ref>
+inline
+bool
+SafeCounter::init(NodeReceiverGroup rg, Uint32 senderData){
+
+ bool b = init<Ref>(rg.m_block, Ref::GSN, senderData);
+ m_nodes = rg.m_nodes;
+ m_count = m_nodes.count();
+ return b;
+}
+
+inline
+void
+SafeCounter::setWaitingFor(Uint32 nodeId) {
+ if(!m_nodes.get(nodeId)){
+ m_nodes.set(nodeId);
+ m_count++;
+ return;
+ }
+ ErrorReporter::handleAssert("SafeCounter::set", __FILE__, __LINE__);
+}
+
+inline
+bool
+SafeCounter::isWaitingFor(Uint32 nodeId) const {
+ return m_nodes.get(nodeId);
+}
+
+inline
+bool
+SafeCounter::done() const {
+ return m_count == 0;
+}
+
+inline
+bool
+SafeCounter::clearWaitingFor(Uint32 nodeId) {
+ if(m_count > 0 && m_nodes.get(nodeId)){
+ m_count--;
+ m_nodes.clear(nodeId);
+ return (m_count == 0);
+ }
+ ErrorReporter::handleAssert("SafeCounter::clear", __FILE__, __LINE__);
+ return false;
+}
+
+inline
+void
+SafeCounter::clearWaitingFor(){
+ m_count = 0;
+ m_nodes.clear();
+}
+
+inline
+bool
+SafeCounter::forceClearWaitingFor(Uint32 nodeId){
+ if(isWaitingFor(nodeId)){
+ return clearWaitingFor(nodeId);
+ }
+ return (m_count == 0);
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/SectionReader.cpp b/ndb/src/kernel/vm/SectionReader.cpp
new file mode 100644
index 00000000000..9e1cbc855e6
--- /dev/null
+++ b/ndb/src/kernel/vm/SectionReader.cpp
@@ -0,0 +1,143 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <SectionReader.hpp>
+#include <TransporterDefinitions.hpp>
+#include "LongSignal.hpp"
+
+#if 0
+ Uint32 m_len;
+ class SectionSegmentPool & m_pool;
+ class SectionSegment * m_head;
+ class SectionSegment * m_currentPos;
+#endif
+
+SectionReader::SectionReader
+(class SegmentedSectionPtr & ptr, class SectionSegmentPool & pool)
+ : m_pool(pool)
+{
+ if(ptr.p == 0){
+ m_pos = 0;
+ m_len = 0;
+ m_head = 0;
+ m_currentSegment = 0;
+ } else {
+ m_pos = 0;
+ m_len = ptr.p->m_sz;
+ m_head = ptr.p;
+ m_currentSegment = ptr.p;
+ }
+}
+
+void
+SectionReader::reset(){
+ m_pos = 0;
+ m_currentSegment = m_head;
+}
+
+bool
+SectionReader::step(Uint32 len){
+ if(m_pos + len >= m_len) {
+ m_pos++;
+ return false;
+ }
+ while(len > SectionSegment::DataLength){
+ m_currentSegment = m_pool.getPtr(m_currentSegment->m_nextSegment);
+
+ len -= SectionSegment::DataLength;
+ m_pos += SectionSegment::DataLength;
+ }
+
+ Uint32 ind = m_pos % SectionSegment::DataLength;
+ while(len > 0){
+ len--;
+ m_pos++;
+
+ ind++;
+ if(ind == SectionSegment::DataLength){
+ ind = 0;
+ m_currentSegment = m_pool.getPtr(m_currentSegment->m_nextSegment);
+ }
+ }
+ return true;
+}
+
+bool
+SectionReader::getWord(Uint32 * dst){
+ if (peekWord(dst)) {
+ step(1);
+ return true;
+ }
+ return false;
+}
+
+bool
+SectionReader::peekWord(Uint32 * dst) const {
+ if(m_pos < m_len){
+ Uint32 ind = m_pos % SectionSegment::DataLength;
+ * dst = m_currentSegment->theData[ind];
+ return true;
+ }
+ return false;
+}
+
+bool
+SectionReader::peekWords(Uint32 * dst, Uint32 len) const {
+ if(m_pos + len > m_len)
+ return false;
+
+ Uint32 ind = (m_pos % SectionSegment::DataLength);
+ Uint32 left = SectionSegment::DataLength - ind;
+ SectionSegment * p = m_currentSegment;
+
+ while(len > left){
+ memcpy(dst, &p->theData[ind], 4 * left);
+ dst += left;
+ len -= left;
+ ind = 0;
+ left = SectionSegment::DataLength;
+ p = m_pool.getPtr(p->m_nextSegment);
+ }
+
+ memcpy(dst, &p->theData[ind], 4 * len);
+ return true;
+}
+
+bool
+SectionReader::getWords(Uint32 * dst, Uint32 len){
+ if(m_pos + len > m_len)
+ return false;
+
+ Uint32 ind = (m_pos % SectionSegment::DataLength);
+ Uint32 left = SectionSegment::DataLength - ind;
+ SectionSegment * p = m_currentSegment;
+
+ while(len > left){
+ memcpy(dst, &p->theData[ind], 4 * left);
+ dst += left;
+ len -= left;
+ ind = 0;
+ left = SectionSegment::DataLength;
+ p = m_pool.getPtr(p->m_nextSegment);
+ }
+
+ memcpy(dst, &p->theData[ind], 4 * len);
+
+ m_pos += len;
+ m_currentSegment = p;
+ return true;
+}
+
diff --git a/ndb/src/kernel/vm/SectionReader.hpp b/ndb/src/kernel/vm/SectionReader.hpp
new file mode 100644
index 00000000000..17eade24a66
--- /dev/null
+++ b/ndb/src/kernel/vm/SectionReader.hpp
@@ -0,0 +1,49 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SECTION_READER_HPP
+#define SECTION_READER_HPP
+
+#include <ndb_types.h>
+
+class SectionReader {
+public:
+ SectionReader(class SegmentedSectionPtr &,
+ class SectionSegmentPool &);
+
+ void reset();
+ bool step(Uint32 len);
+ bool getWord(Uint32 * dst);
+ bool peekWord(Uint32 * dst) const ;
+ bool peekWords(Uint32 * dst, Uint32 len) const;
+ Uint32 getSize() const;
+ bool getWords(Uint32 * dst, Uint32 len);
+
+private:
+ Uint32 m_pos;
+ Uint32 m_len;
+ class SectionSegmentPool & m_pool;
+ class SectionSegment * m_head;
+ class SectionSegment * m_currentSegment;
+};
+
+inline
+Uint32 SectionReader::getSize() const
+{
+ return m_len;
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/SignalCounter.hpp b/ndb/src/kernel/vm/SignalCounter.hpp
new file mode 100644
index 00000000000..b05d0858867
--- /dev/null
+++ b/ndb/src/kernel/vm/SignalCounter.hpp
@@ -0,0 +1,164 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SIGNAL_COUNTER_HPP
+#define SIGNAL_COUNTER_HPP
+
+#include <NodeBitmask.hpp>
+#include <ErrorReporter.hpp>
+
+class SignalCounter {
+private:
+ Uint32 m_count;
+ NdbNodeBitmask m_nodes;
+
+public:
+ SignalCounter() { clearWaitingFor();}
+ void clearWaitingFor();
+
+ /**
+ * When sending to different node
+ */
+ void setWaitingFor(Uint32 nodeId);
+ void clearWaitingFor(Uint32 nodeId);
+ void forceClearWaitingFor(Uint32 nodeId);
+
+ bool isWaitingFor(Uint32 nodeId) const;
+ bool done() const;
+
+ const char * getText() const;
+
+ SignalCounter& operator=(const NdbNodeBitmask & bitmask);
+ SignalCounter& operator=(const NodeReceiverGroup& rg) {
+ return (* this) = rg.m_nodes;
+ }
+
+ /**
+ * When sending to same node
+ */
+ SignalCounter& operator=(Uint32 count);
+ SignalCounter& operator--(int);
+ SignalCounter& operator++(int);
+ SignalCounter& operator+=(Uint32);
+ Uint32 getCount() const;
+};
+
+inline
+void
+SignalCounter::setWaitingFor(Uint32 nodeId) {
+ if(!m_nodes.get(nodeId)){
+ m_nodes.set(nodeId);
+ m_count++;
+ return;
+ }
+ ErrorReporter::handleAssert("SignalCounter::set", __FILE__, __LINE__);
+}
+
+inline
+bool
+SignalCounter::isWaitingFor(Uint32 nodeId) const {
+ return m_nodes.get(nodeId);
+}
+
+inline
+bool
+SignalCounter::done() const {
+ return m_count == 0;
+}
+
+inline
+Uint32
+SignalCounter::getCount() const {
+ return m_count;
+}
+
+inline
+void
+SignalCounter::clearWaitingFor(Uint32 nodeId) {
+ if(m_nodes.get(nodeId) && m_count > 0){
+ m_count--;
+ m_nodes.clear(nodeId);
+ return;
+ }
+ ErrorReporter::handleAssert("SignalCounter::clear", __FILE__, __LINE__);
+}
+
+inline
+void
+SignalCounter::clearWaitingFor(){
+ m_count = 0;
+ m_nodes.clear();
+}
+
+inline
+void
+SignalCounter::forceClearWaitingFor(Uint32 nodeId){
+ if(isWaitingFor(nodeId)){
+ clearWaitingFor(nodeId);
+ }
+}
+
+inline
+SignalCounter&
+SignalCounter::operator=(Uint32 count){
+ m_count = count;
+ m_nodes.clear();
+ return * this;
+}
+
+inline
+SignalCounter&
+SignalCounter::operator--(int){
+ if(m_count > 0){
+ m_count--;
+ return * this;
+ }
+ ErrorReporter::handleAssert("SignalCounter::operator--", __FILE__, __LINE__);
+ return * this;
+}
+
+inline
+SignalCounter&
+SignalCounter::operator++(int){
+ m_count++;
+ return * this;
+}
+
+inline
+SignalCounter&
+SignalCounter::operator+=(Uint32 n){
+ m_count += n;
+ return * this;
+}
+
+inline
+const char *
+SignalCounter::getText() const {
+ static char buf[255];
+ static char nodes[m_nodes.TextLength+1];
+ snprintf(buf, sizeof(buf), "[SignalCounter: m_count=%d %s]", m_count, m_nodes.getText(nodes));
+ return buf;
+}
+
+inline
+SignalCounter&
+SignalCounter::operator=(const NdbNodeBitmask & bitmask){
+ m_nodes = bitmask;
+ m_count = bitmask.count();
+ return * this;
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/SimBlockList.hpp b/ndb/src/kernel/vm/SimBlockList.hpp
new file mode 100644
index 00000000000..40485a37425
--- /dev/null
+++ b/ndb/src/kernel/vm/SimBlockList.hpp
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SimBlockList_H
+#define SimBlockList_H
+
+#include <MetaData.hpp>
+#include <SimulatedBlock.hpp>
+
+class Configuration;
+
+class SimBlockList
+{
+public:
+ SimBlockList();
+ ~SimBlockList();
+
+ void load(const Configuration & conf);
+ void unload();
+private:
+ int noOfBlocks;
+ SimulatedBlock** theList;
+ MetaData::Common* ptrMetaDataCommon;
+};
+
+inline
+SimBlockList::SimBlockList(){
+ noOfBlocks = 0;
+ theList = 0;
+ ptrMetaDataCommon = 0;
+}
+
+inline
+SimBlockList::~SimBlockList(){
+ unload();
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/SimplePropertiesSection.cpp b/ndb/src/kernel/vm/SimplePropertiesSection.cpp
new file mode 100644
index 00000000000..d442ff2e698
--- /dev/null
+++ b/ndb/src/kernel/vm/SimplePropertiesSection.cpp
@@ -0,0 +1,223 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <SimpleProperties.hpp>
+#include <TransporterDefinitions.hpp>
+#include "LongSignal.hpp"
+
+SimplePropertiesSectionReader::SimplePropertiesSectionReader
+(class SegmentedSectionPtr & ptr, class SectionSegmentPool & pool)
+ : m_pool(pool)
+{
+ if(ptr.p == 0){
+ m_pos = 0;
+ m_len = 0;
+ m_head = 0;
+ m_currentSegment = 0;
+ } else {
+ m_pos = 0;
+ m_len = ptr.p->m_sz;
+ m_head = ptr.p;
+ m_currentSegment = ptr.p;
+ }
+ first();
+}
+
+void
+SimplePropertiesSectionReader::reset(){
+ m_pos = 0;
+ m_currentSegment = m_head;
+}
+
+bool
+SimplePropertiesSectionReader::step(Uint32 len){
+ if(m_pos + len >= m_len) {
+ m_pos++;
+ return false;
+ }
+ while(len > SectionSegment::DataLength){
+ m_currentSegment = m_pool.getPtr(m_currentSegment->m_nextSegment);
+
+ len -= SectionSegment::DataLength;
+ m_pos += SectionSegment::DataLength;
+ }
+
+ Uint32 ind = m_pos % SectionSegment::DataLength;
+ while(len > 0){
+ len--;
+ m_pos++;
+
+ ind++;
+ if(ind == SectionSegment::DataLength){
+ ind = 0;
+ m_currentSegment = m_pool.getPtr(m_currentSegment->m_nextSegment);
+ }
+ }
+ return true;
+}
+
+bool
+SimplePropertiesSectionReader::getWord(Uint32 * dst){
+ if (peekWord(dst)) {
+ step(1);
+ return true;
+ }
+ return false;
+}
+
+bool
+SimplePropertiesSectionReader::peekWord(Uint32 * dst) const {
+ if(m_pos < m_len){
+ Uint32 ind = m_pos % SectionSegment::DataLength;
+ * dst = m_currentSegment->theData[ind];
+ return true;
+ }
+ return false;
+}
+
+bool
+SimplePropertiesSectionReader::peekWords(Uint32 * dst, Uint32 len) const {
+ if(m_pos + len > m_len){
+ return false;
+ }
+ Uint32 ind = (m_pos % SectionSegment::DataLength);
+ Uint32 left = (SectionSegment::DataLength - ind);
+ SectionSegment * p = m_currentSegment;
+
+ while(len > left){
+ memcpy(dst, &p->theData[ind], 4 * left);
+ dst += left;
+ len -= left;
+ ind = 0;
+ left = SectionSegment::DataLength;
+ p = m_pool.getPtr(p->m_nextSegment);
+ }
+
+ memcpy(dst, &p->theData[ind], 4 * len);
+ return true;
+}
+
+bool
+SimplePropertiesSectionReader::getWords(Uint32 * dst, Uint32 len){
+ if(peekWords(dst, len)){
+ step(len);
+ return true;
+ }
+ return false;
+}
+
+SimplePropertiesSectionWriter::SimplePropertiesSectionWriter(class SectionSegmentPool & pool)
+ : m_pool(pool)
+{
+ Ptr<SectionSegment> first;
+ if(m_pool.seize(first)){
+ ;
+ } else {
+ m_pos = -1;
+ m_head = 0;
+ m_currentSegment = 0;
+ m_prevPtrI = RNIL;
+ return;
+ }
+ m_sz = 0;
+ m_pos = 0;
+ m_head = first.p;
+ m_head->m_lastSegment = first.i;
+ m_currentSegment = first.p;
+ m_prevPtrI = RNIL;
+}
+
+bool
+SimplePropertiesSectionWriter::reset(){
+ if(m_pos >= 0){
+ m_pos = 0;
+ return true;
+ }
+ return false;
+}
+
+bool
+SimplePropertiesSectionWriter::putWord(Uint32 val){
+ return putWords(&val, 1);
+}
+
+bool
+SimplePropertiesSectionWriter::putWords(const Uint32 * src, Uint32 len){
+ Uint32 left = SectionSegment::DataLength - m_pos;
+
+ while(len >= left){
+ memcpy(&m_currentSegment->theData[m_pos], src, 4 * left);
+ Ptr<SectionSegment> next;
+ if(m_pool.seize(next)){
+
+ m_prevPtrI = m_currentSegment->m_lastSegment;
+ m_currentSegment->m_nextSegment = next.i;
+ next.p->m_lastSegment = next.i;
+ m_currentSegment = next.p;
+
+ len -= left;
+ src += left;
+ m_sz += left;
+
+ left = SectionSegment::DataLength;
+ m_pos = 0;
+ } else {
+ abort();
+ return false;
+ }
+ }
+
+ memcpy(&m_currentSegment->theData[m_pos], src, 4 * len);
+ m_sz += len;
+ m_pos += len;
+
+ assert(m_pos < (int)SectionSegment::DataLength);
+
+ return true;
+}
+
+void
+SimplePropertiesSectionWriter::getPtr(class SegmentedSectionPtr & dst){
+ // Set last ptr and size
+ if(m_pos >= 0){
+ dst.p = m_head;
+ dst.i = m_head->m_lastSegment;
+ dst.sz = m_sz;
+ m_head->m_sz = m_sz;
+ m_head->m_lastSegment = m_currentSegment->m_lastSegment;
+
+ if((m_pos % SectionSegment::DataLength) == 0){
+ m_pool.release(m_currentSegment->m_lastSegment);
+ m_head->m_lastSegment = m_prevPtrI;
+ }
+
+ m_sz = 0;
+ m_pos = -1;
+ m_head = m_currentSegment = 0;
+ m_prevPtrI = RNIL;
+ return ;
+ }
+ dst.p = 0;
+ dst.sz = 0;
+ dst.i = RNIL;
+
+ m_pool.release(m_head->m_lastSegment);
+
+ m_sz = 0;
+ m_pos = -1;
+ m_head = m_currentSegment = 0;
+ m_prevPtrI = RNIL;
+}
diff --git a/ndb/src/kernel/vm/SimulatedBlock.cpp b/ndb/src/kernel/vm/SimulatedBlock.cpp
new file mode 100644
index 00000000000..b9bfcfebc7d
--- /dev/null
+++ b/ndb/src/kernel/vm/SimulatedBlock.cpp
@@ -0,0 +1,1733 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "SimulatedBlock.hpp"
+#include <NdbOut.hpp>
+#include <GlobalData.hpp>
+#include <Emulator.hpp>
+#include <ErrorHandlingMacros.hpp>
+#include <TimeQueue.hpp>
+#include <TransporterRegistry.hpp>
+#include <SignalLoggerManager.hpp>
+#include <FastScheduler.hpp>
+#include <NdbMem.h>
+#include <NdbStdio.h>
+#include <stdarg.h>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/ContinueFragmented.hpp>
+#include <signaldata/NodeStateSignalData.hpp>
+#include <DebuggerNames.hpp>
+#include "LongSignal.hpp"
+
+#include <Properties.hpp>
+#include "Configuration.hpp"
+
+#define ljamEntry() jamEntryLine(30000 + __LINE__)
+#define ljam() jamLine(30000 + __LINE__)
+
+//
+// Constructor, Destructor
+//
+SimulatedBlock::SimulatedBlock(BlockNumber blockNumber,
+ const class Configuration & conf)
+ : theNodeId(globalData.ownId),
+ theNumber(blockNumber),
+ theReference(numberToRef(blockNumber, globalData.ownId)),
+ theConfiguration(conf),
+ c_fragmentInfoHash(c_fragmentInfoPool),
+ c_linearFragmentSendList(c_fragmentSendPool),
+ c_segmentedFragmentSendList(c_fragmentSendPool),
+ c_mutexMgr(* this),
+ c_counterMgr(* this),
+ c_ptrMetaDataCommon(0)
+{
+ NewVarRef = 0;
+
+ globalData.setBlock(blockNumber, this);
+ c_fragmentIdCounter = 1;
+ c_fragSenderRunning = false;
+
+ const Properties * p = conf.getOwnProperties();
+ ndbrequire(p != 0);
+
+ Uint32 count = 10;
+ char buf[255];
+
+ count = 10;
+ snprintf(buf, 255, "%s.FragmentSendPool", getBlockName(blockNumber));
+ if(!p->get(buf, &count))
+ p->get("FragmentSendPool", &count);
+ c_fragmentSendPool.setSize(count);
+
+ count = 10;
+ snprintf(buf, 255, "%s.FragmentInfoPool", getBlockName(blockNumber));
+ if(!p->get(buf, &count))
+ p->get("FragmentInfoPool", &count);
+ c_fragmentInfoPool.setSize(count);
+
+ count = 10;
+ snprintf(buf, 255, "%s.FragmentInfoHash", getBlockName(blockNumber));
+ if(!p->get(buf, &count))
+ p->get("FragmentInfoHash", &count);
+ c_fragmentInfoHash.setSize(count);
+
+ count = 5;
+ snprintf(buf, 255, "%s.ActiveMutexes", getBlockName(blockNumber));
+ if(!p->get(buf, &count))
+ p->get("ActiveMutexes", &count);
+ c_mutexMgr.setSize(count);
+
+ c_counterMgr.setSize(5);
+
+#ifdef VM_TRACE_TIME
+ clearTimes();
+#endif
+
+ for(GlobalSignalNumber i = 0; i<=MAX_GSN; i++)
+ theExecArray[i] = 0;
+ installSimulatedBlockFunctions();
+
+ CLEAR_ERROR_INSERT_VALUE;
+}
+
+SimulatedBlock::~SimulatedBlock()
+{
+ freeBat();
+#ifdef VM_TRACE_TIME
+ printTimes(stdout);
+#endif
+}
+
+void
+SimulatedBlock::installSimulatedBlockFunctions(){
+ ExecFunction * a = theExecArray;
+ a[GSN_NODE_STATE_REP] = &SimulatedBlock::execNODE_STATE_REP;
+ a[GSN_CHANGE_NODE_STATE_REQ] = &SimulatedBlock::execCHANGE_NODE_STATE_REQ;
+ a[GSN_NDB_TAMPER] = &SimulatedBlock::execNDB_TAMPER;
+ a[GSN_SIGNAL_DROPPED_REP] = &SimulatedBlock::execSIGNAL_DROPPED_REP;
+ a[GSN_CONTINUE_FRAGMENTED]= &SimulatedBlock::execCONTINUE_FRAGMENTED;
+ a[GSN_UTIL_CREATE_LOCK_REF] = &SimulatedBlock::execUTIL_CREATE_LOCK_REF;
+ a[GSN_UTIL_CREATE_LOCK_CONF] = &SimulatedBlock::execUTIL_CREATE_LOCK_CONF;
+ a[GSN_UTIL_DESTROY_LOCK_REF] = &SimulatedBlock::execUTIL_DESTORY_LOCK_REF;
+ a[GSN_UTIL_DESTROY_LOCK_CONF] = &SimulatedBlock::execUTIL_DESTORY_LOCK_CONF;
+ a[GSN_UTIL_LOCK_REF] = &SimulatedBlock::execUTIL_LOCK_REF;
+ a[GSN_UTIL_LOCK_CONF] = &SimulatedBlock::execUTIL_LOCK_CONF;
+ a[GSN_UTIL_UNLOCK_REF] = &SimulatedBlock::execUTIL_UNLOCK_REF;
+ a[GSN_UTIL_UNLOCK_CONF] = &SimulatedBlock::execUTIL_UNLOCK_CONF;
+}
+
+void
+SimulatedBlock::addRecSignalImpl(GlobalSignalNumber gsn,
+ ExecFunction f, bool force){
+ REQUIRE(gsn <= MAX_GSN, "Illegal signal added in block (GSN too high)");
+ char probData[255];
+ snprintf(probData, 255,
+ "Signal (%d) already added in block",
+ gsn);
+ REQUIRE(force || theExecArray[gsn] == 0, probData);
+ theExecArray[gsn] = f;
+}
+
+void
+SimulatedBlock::signal_error(Uint32 gsn, Uint32 len, Uint32 recBlockNo,
+ const char* filename, int lineno) const
+{
+ char objRef[255];
+ snprintf(objRef, 255, "%s:%d", filename, lineno);
+ char probData[255];
+ snprintf(probData, 255,
+ "Signal (GSN: %d, Length: %d, Rec Block No: %d)",
+ gsn, len, recBlockNo);
+
+ ErrorReporter::handleError(ecError,
+ BLOCK_ERROR_BNR_ZERO,
+ probData,
+ objRef);
+}
+
+
+extern class SectionSegmentPool g_sectionSegmentPool;
+
+void
+SimulatedBlock::sendSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jobBuffer) const {
+
+ BlockNumber sendBnr = number();
+ BlockReference sendBRef = reference();
+
+ Uint32 noOfSections = signal->header.m_noOfSections;
+ Uint32 recBlock = refToBlock(ref);
+ Uint32 recNode = refToNode(ref);
+ Uint32 ourProcessor = globalData.ownId;
+
+ signal->header.theLength = length;
+ signal->header.theVerId_signalNumber = gsn;
+ signal->header.theReceiversBlockNumber = recBlock;
+
+ Uint32 tSignalId = signal->header.theSignalId;
+
+ if ((length == 0) || (length > 25) || (recBlock == 0)) {
+ signal_error(gsn, length, recBlock, __FILE__, __LINE__);
+ return;
+ }//if
+#ifdef VM_TRACE
+ if(globalData.testOn){
+ Uint16 proc =
+ (recNode == 0 ? globalData.ownId : recNode);
+ signal->header.theSendersBlockRef = sendBRef;
+ globalSignalLoggers.sendSignal(signal->header,
+ jobBuffer,
+ &signal->theData[0],
+ proc,
+ signal->m_sectionPtr,
+ signal->header.m_noOfSections);
+ }
+#endif
+
+ if(recNode == ourProcessor || recNode == 0) {
+ signal->header.theSendersSignalId = tSignalId;
+ signal->header.theSendersBlockRef = sendBRef;
+ signal->header.theLength = length;
+ globalScheduler.execute(signal, jobBuffer, recBlock,
+ gsn);
+ signal->header.m_noOfSections = 0;
+ signal->header.m_fragmentInfo = 0;
+ return;
+ } else {
+ // send distributed Signal
+ SignalHeader sh;
+
+ Uint32 tTrace = signal->getTrace();
+
+ sh.theVerId_signalNumber = gsn;
+ sh.theReceiversBlockNumber = recBlock;
+ sh.theSendersBlockRef = sendBnr;
+ sh.theLength = length;
+ sh.theTrace = tTrace;
+ sh.theSignalId = tSignalId;
+ sh.m_noOfSections = noOfSections;
+ sh.m_fragmentInfo = 0;
+
+#ifdef TRACE_DISTRIBUTED
+ ndbout_c("send: %s(%d) to (%s, %d)",
+ getSignalName(gsn), gsn, getBlockName(recBlock),
+ recNode);
+#endif
+ SendStatus ss = globalTransporterRegistry.prepareSend(&sh, jobBuffer,
+ &signal->theData[0],
+ recNode,
+ g_sectionSegmentPool,
+ signal->m_sectionPtr);
+
+ ndbrequire(ss == SEND_OK || ss == SEND_BLOCKED || ss == SEND_DISCONNECTED);
+ ::releaseSections(noOfSections, signal->m_sectionPtr);
+ signal->header.m_noOfSections = 0;
+ }
+ return;
+}
+
+void
+SimulatedBlock::sendSignal(NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jobBuffer) const {
+
+ Uint32 noOfSections = signal->header.m_noOfSections;
+ Uint32 tSignalId = signal->header.theSignalId;
+ Uint32 tTrace = signal->getTrace();
+ Uint32 tFragInf = signal->header.m_fragmentInfo;
+
+ Uint32 ourProcessor = globalData.ownId;
+ Uint32 recBlock = rg.m_block;
+
+ signal->header.theLength = length;
+ signal->header.theVerId_signalNumber = gsn;
+ signal->header.theReceiversBlockNumber = recBlock;
+ signal->header.theSendersSignalId = tSignalId;
+ signal->header.theSendersBlockRef = reference();
+
+ if ((length == 0) || (length > 25) || (recBlock == 0)) {
+ signal_error(gsn, length, recBlock, __FILE__, __LINE__);
+ return;
+ }//if
+
+ SignalHeader sh;
+
+ sh.theVerId_signalNumber = gsn;
+ sh.theReceiversBlockNumber = recBlock;
+ sh.theSendersBlockRef = number();
+ sh.theLength = length;
+ sh.theTrace = tTrace;
+ sh.theSignalId = tSignalId;
+ sh.m_noOfSections = noOfSections;
+ sh.m_fragmentInfo = tFragInf;
+
+ /**
+ * Check own node
+ */
+ bool release = true;
+ if(rg.m_nodes.get(0) || rg.m_nodes.get(ourProcessor)){
+#ifdef VM_TRACE
+ if(globalData.testOn){
+ globalSignalLoggers.sendSignal(signal->header,
+ jobBuffer,
+ &signal->theData[0],
+ ourProcessor,
+ signal->m_sectionPtr,
+ signal->header.m_noOfSections);
+ }
+#endif
+ globalScheduler.execute(signal, jobBuffer, recBlock, gsn);
+
+ rg.m_nodes.clear((Uint32)0);
+ rg.m_nodes.clear(ourProcessor);
+ release = false;
+ }
+
+ /**
+ * Do the big loop
+ */
+ Uint32 recNode = 0;
+ while(!rg.m_nodes.isclear()){
+ recNode = rg.m_nodes.find(recNode + 1);
+ rg.m_nodes.clear(recNode);
+#ifdef VM_TRACE
+ if(globalData.testOn){
+ globalSignalLoggers.sendSignal(signal->header,
+ jobBuffer,
+ &signal->theData[0],
+ recNode,
+ signal->m_sectionPtr,
+ signal->header.m_noOfSections);
+ }
+#endif
+
+#ifdef TRACE_DISTRIBUTED
+ ndbout_c("send: %s(%d) to (%s, %d)",
+ getSignalName(gsn), gsn, getBlockName(recBlock),
+ recNode);
+#endif
+
+ SendStatus ss = globalTransporterRegistry.prepareSend(&sh, jobBuffer,
+ &signal->theData[0],
+ recNode,
+ g_sectionSegmentPool,
+ signal->m_sectionPtr);
+ ndbrequire(ss == SEND_OK || ss == SEND_BLOCKED || ss == SEND_DISCONNECTED);
+ }
+
+ if(release){
+ ::releaseSections(noOfSections, signal->m_sectionPtr);
+ }
+
+ signal->header.m_noOfSections = 0;
+ signal->header.m_fragmentInfo = 0;
+
+ return;
+}
+
+bool import(Ptr<SectionSegment> & first, const Uint32 * src, Uint32 len);
+
+void
+SimulatedBlock::sendSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jobBuffer,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections) const {
+
+ BlockNumber sendBnr = number();
+ BlockReference sendBRef = reference();
+
+ Uint32 recBlock = refToBlock(ref);
+ Uint32 recNode = refToNode(ref);
+ Uint32 ourProcessor = globalData.ownId;
+
+ ::releaseSections(signal->header.m_noOfSections, signal->m_sectionPtr);
+
+ signal->header.theLength = length;
+ signal->header.theVerId_signalNumber = gsn;
+ signal->header.theReceiversBlockNumber = recBlock;
+ signal->header.m_noOfSections = noOfSections;
+
+ Uint32 tSignalId = signal->header.theSignalId;
+ Uint32 tFragInfo = signal->header.m_fragmentInfo;
+
+ if ((length == 0) || (length > 25) || (recBlock == 0)) {
+ signal_error(gsn, length, recBlock, __FILE__, __LINE__);
+ return;
+ }//if
+#ifdef VM_TRACE
+ if(globalData.testOn){
+ Uint16 proc =
+ (recNode == 0 ? globalData.ownId : recNode);
+ signal->header.theSendersBlockRef = sendBRef;
+ globalSignalLoggers.sendSignal(signal->header,
+ jobBuffer,
+ &signal->theData[0],
+ proc,
+ ptr, noOfSections);
+ }
+#endif
+
+ if(recNode == ourProcessor || recNode == 0) {
+ signal->header.theSendersSignalId = tSignalId;
+ signal->header.theSendersBlockRef = sendBRef;
+
+ /**
+ * We have to copy the data
+ */
+ Ptr<SectionSegment> segptr[3];
+ for(Uint32 i = 0; i<noOfSections; i++){
+ ndbrequire(import(segptr[i], ptr[i].p, ptr[i].sz));
+ signal->m_sectionPtr[i].i = segptr[i].i;
+ }
+
+ globalScheduler.execute(signal, jobBuffer, recBlock,
+ gsn);
+ signal->header.m_noOfSections = 0;
+ return;
+ } else {
+ // send distributed Signal
+ SignalHeader sh;
+
+ Uint32 tTrace = signal->getTrace();
+ Uint32 noOfSections = signal->header.m_noOfSections;
+
+ sh.theVerId_signalNumber = gsn;
+ sh.theReceiversBlockNumber = recBlock;
+ sh.theSendersBlockRef = sendBnr;
+ sh.theLength = length;
+ sh.theTrace = tTrace;
+ sh.theSignalId = tSignalId;
+ sh.m_noOfSections = noOfSections;
+ sh.m_fragmentInfo = tFragInfo;
+
+#ifdef TRACE_DISTRIBUTED
+ ndbout_c("send: %s(%d) to (%s, %d)",
+ getSignalName(gsn), gsn, getBlockName(recBlock),
+ recNode);
+#endif
+
+ SendStatus ss = globalTransporterRegistry.prepareSend(&sh, jobBuffer,
+ &signal->theData[0],
+ recNode,
+ ptr);
+ ndbrequire(ss == SEND_OK || ss == SEND_BLOCKED || ss == SEND_DISCONNECTED);
+ }
+
+ signal->header.m_noOfSections = 0;
+ signal->header.m_fragmentInfo = 0;
+ return;
+}
+
+void
+SimulatedBlock::sendSignal(NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jobBuffer,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections) const {
+
+ Uint32 tSignalId = signal->header.theSignalId;
+ Uint32 tTrace = signal->getTrace();
+ Uint32 tFragInfo = signal->header.m_fragmentInfo;
+
+ Uint32 ourProcessor = globalData.ownId;
+ Uint32 recBlock = rg.m_block;
+
+ ::releaseSections(signal->header.m_noOfSections, signal->m_sectionPtr);
+
+ signal->header.theLength = length;
+ signal->header.theVerId_signalNumber = gsn;
+ signal->header.theReceiversBlockNumber = recBlock;
+ signal->header.theSendersSignalId = tSignalId;
+ signal->header.theSendersBlockRef = reference();
+ signal->header.m_noOfSections = noOfSections;
+
+ if ((length == 0) || (length > 25) || (recBlock == 0)) {
+ signal_error(gsn, length, recBlock, __FILE__, __LINE__);
+ return;
+ }//if
+
+ SignalHeader sh;
+ sh.theVerId_signalNumber = gsn;
+ sh.theReceiversBlockNumber = recBlock;
+ sh.theSendersBlockRef = number();
+ sh.theLength = length;
+ sh.theTrace = tTrace;
+ sh.theSignalId = tSignalId;
+ sh.m_noOfSections = noOfSections;
+ sh.m_fragmentInfo = tFragInfo;
+
+ /**
+ * Check own node
+ */
+ if(rg.m_nodes.get(0) || rg.m_nodes.get(ourProcessor)){
+#ifdef VM_TRACE
+ if(globalData.testOn){
+ globalSignalLoggers.sendSignal(signal->header,
+ jobBuffer,
+ &signal->theData[0],
+ ourProcessor,
+ ptr, noOfSections);
+ }
+#endif
+ /**
+ * We have to copy the data
+ */
+ Ptr<SectionSegment> segptr[3];
+ for(Uint32 i = 0; i<noOfSections; i++){
+ ndbrequire(import(segptr[i], ptr[i].p, ptr[i].sz));
+ signal->m_sectionPtr[i].i = segptr[i].i;
+ }
+ globalScheduler.execute(signal, jobBuffer, recBlock, gsn);
+
+ rg.m_nodes.clear((Uint32)0);
+ rg.m_nodes.clear(ourProcessor);
+ }
+
+ /**
+ * Do the big loop
+ */
+ Uint32 recNode = 0;
+ while(!rg.m_nodes.isclear()){
+ recNode = rg.m_nodes.find(recNode + 1);
+ rg.m_nodes.clear(recNode);
+
+#ifdef VM_TRACE
+ if(globalData.testOn){
+ globalSignalLoggers.sendSignal(signal->header,
+ jobBuffer,
+ &signal->theData[0],
+ recNode,
+ ptr, noOfSections);
+ }
+#endif
+
+#ifdef TRACE_DISTRIBUTED
+ ndbout_c("send: %s(%d) to (%s, %d)",
+ getSignalName(gsn), gsn, getBlockName(recBlock),
+ recNode);
+#endif
+
+ SendStatus ss = globalTransporterRegistry.prepareSend(&sh, jobBuffer,
+ &signal->theData[0],
+ recNode,
+ ptr);
+ ndbrequire(ss == SEND_OK || ss == SEND_BLOCKED || ss == SEND_DISCONNECTED);
+ }
+
+ signal->header.m_noOfSections = 0;
+ signal->header.m_fragmentInfo = 0;
+
+ return;
+}
+
+void
+SimulatedBlock::sendSignalWithDelay(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 delayInMilliSeconds,
+ Uint32 length) const {
+
+ BlockNumber bnr = refToBlock(ref);
+
+ //BlockNumber sendBnr = number();
+ BlockReference sendBRef = reference();
+
+ if (bnr == 0) {
+ bnr_error();
+ }//if
+
+ signal->header.theLength = length;
+ signal->header.theSendersSignalId = signal->header.theSignalId;
+ signal->header.theSendersBlockRef = sendBRef;
+ signal->header.theVerId_signalNumber = gsn;
+ signal->header.theReceiversBlockNumber = bnr;
+
+#ifdef VM_TRACE
+ {
+ if(globalData.testOn){
+ globalSignalLoggers.sendSignalWithDelay(delayInMilliSeconds,
+ signal->header,
+ 0,
+ &signal->theData[0],
+ globalData.ownId,
+ signal->m_sectionPtr,
+ signal->header.m_noOfSections);
+ }
+ }
+#endif
+ globalTimeQueue.insert(signal, bnr, gsn, delayInMilliSeconds);
+
+ signal->header.m_noOfSections = 0;
+ signal->header.m_fragmentInfo = 0;
+
+ // befor 2nd parameter to globalTimeQueue.insert
+ // (Priority)theSendSig[sigIndex].jobBuffer
+}
+
+void
+SimulatedBlock::releaseSections(Signal* signal){
+ ::releaseSections(signal->header.m_noOfSections, signal->m_sectionPtr);
+ signal->header.m_noOfSections = 0;
+}
+
+class SectionSegmentPool&
+SimulatedBlock::getSectionSegmentPool(){
+ return g_sectionSegmentPool;
+}
+
+NewVARIABLE *
+SimulatedBlock::allocateBat(int batSize){
+ NewVARIABLE* bat = NewVarRef;
+ bat = (NewVARIABLE*)realloc(bat, batSize * sizeof(NewVARIABLE));
+ NewVarRef = bat;
+ theBATSize = batSize;
+ return bat;
+}
+
+void
+SimulatedBlock::freeBat(){
+ if(NewVarRef != 0){
+ free(NewVarRef);
+ NewVarRef = 0;
+ }
+}
+
+const NewVARIABLE *
+SimulatedBlock::getBat(Uint16 blockNo){
+ SimulatedBlock * sb = globalData.getBlock(blockNo);
+ if(sb == 0)
+ return 0;
+ return sb->NewVarRef;
+}
+
+Uint16
+SimulatedBlock::getBatSize(Uint16 blockNo){
+ SimulatedBlock * sb = globalData.getBlock(blockNo);
+ if(sb == 0)
+ return 0;
+ return sb->theBATSize;
+}
+
+void*
+SimulatedBlock::allocRecord(const char * type, size_t s, size_t n) const
+{
+
+ void* p = NULL;
+ size_t size = n*s;
+
+ if (size > 0){
+#ifdef VM_TRACE_MEM
+ ndbout_c("%s::allocRecord(%s, %u, %u) = %u bytes",
+ getBlockName(number()),
+ type,
+ s,
+ n,
+ size);
+#endif
+ p = NdbMem_Allocate(size);
+ if (p == NULL){
+ char buf1[255];
+ char buf2[255];
+ snprintf(buf1, sizeof(buf1), "%s could not allocate memory for %s",
+ getBlockName(number()), type);
+ snprintf(buf2, sizeof(buf2), "Requested: %ux%u = %u bytes", (Uint32)s, (Uint32)n, (Uint32)size);
+ ERROR_SET(fatal, ERR_MEMALLOC, buf1, buf2);
+ }
+
+
+ // Set the allocated memory to zero
+#ifndef NDB_PURIFY
+#if defined NDB_OSE
+ int pages = (size / 4096);
+ if ((size % 4096)!=0)
+ pages++;
+
+ char* p2 =(char*) p;
+ for (int i = 0; i < pages; i++){
+ memset(p2, 0, 4096);
+ p2 = p2 + 4096;
+ }
+#elif 1
+ /**
+ * This code should be enabled in order to find logical errors and not
+ * initalised errors in the kernel.
+ *
+ * NOTE! It's not just "uninitialised errors" that are found by doing this
+ * it will also find logical errors that have been hidden by all the zeros.
+ */
+
+ memset(p, 0xF1, size);
+#endif
+#endif
+ }
+ return p;
+}
+
+void
+SimulatedBlock::deallocRecord(void ** ptr,
+ const char * type, size_t s, size_t n) const {
+ (void)type;
+ (void)s;
+ (void)n;
+
+ if(* ptr != 0){
+ NdbMem_Free(* ptr);
+ * ptr = 0;
+ }
+}
+
+void
+SimulatedBlock::progError(int line, int err_code, const char* extra) const {
+ jamLine(line);
+
+ const char *aBlockName = getBlockName(number(), "VM Kernel");
+
+ // Pack status of interesting config variables
+ // so that we can print them in error.log
+ int magicStatus =
+ (theConfiguration.stopOnError()<<1) +
+ (theConfiguration.getInitialStart()<<2) +
+ (theConfiguration.getDaemonMode()<<3);
+
+
+ /* Add line number to block name */
+ char buf[100];
+ snprintf(&buf[0], 100, "%s (Line: %d) 0x%.8x",
+ aBlockName, line, magicStatus);
+
+ ErrorReporter::handleError(ecError, err_code, extra, buf);
+
+}
+
+void
+SimulatedBlock::infoEvent(const char * msg, ...) const {
+ if(msg == 0)
+ return;
+
+ Uint32 theData[25];
+ theData[0] = EventReport::InfoEvent;
+ char * buf = (char *)&(theData[1]);
+
+ va_list ap;
+ va_start(ap, msg);
+ vsnprintf(buf, 96, msg, ap); // 96 = 100 - 4
+ va_end(ap);
+
+ int len = strlen(buf) + 1;
+ if(len > 96){
+ len = 96;
+ buf[95] = 0;
+ }
+
+ /**
+ * Init and put it into the job buffer
+ */
+ SignalHeader sh;
+ memset(&sh, 0, sizeof(SignalHeader));
+
+ const Signal * signal = globalScheduler.getVMSignals();
+ Uint32 tTrace = signal->header.theTrace;
+ Uint32 tSignalId = signal->header.theSignalId;
+
+ sh.theVerId_signalNumber = GSN_EVENT_REP;
+ sh.theReceiversBlockNumber = CMVMI;
+ sh.theSendersBlockRef = reference();
+ sh.theTrace = tTrace;
+ sh.theSignalId = tSignalId;
+ sh.theLength = ((len+3)/4)+1;
+
+ Uint32 secPtrI[3]; // Dummy
+ globalScheduler.execute(&sh, JBB, theData, secPtrI);
+}
+
+void
+SimulatedBlock::warningEvent(const char * msg, ...) const {
+ if(msg == 0)
+ return;
+
+ Uint32 theData[25];
+ theData[0] = EventReport::WarningEvent;
+ char * buf = (char *)&(theData[1]);
+
+ va_list ap;
+ va_start(ap, msg);
+ vsnprintf(buf, 96, msg, ap); // 96 = 100 - 4
+ va_end(ap);
+
+ int len = strlen(buf) + 1;
+ if(len > 96){
+ len = 96;
+ buf[95] = 0;
+ }
+
+ /**
+ * Init and put it into the job buffer
+ */
+ SignalHeader sh;
+ memset(&sh, 0, sizeof(SignalHeader));
+
+ const Signal * signal = globalScheduler.getVMSignals();
+ Uint32 tTrace = signal->header.theTrace;
+ Uint32 tSignalId = signal->header.theSignalId;
+
+ sh.theVerId_signalNumber = GSN_EVENT_REP;
+ sh.theReceiversBlockNumber = CMVMI;
+ sh.theSendersBlockRef = reference();
+ sh.theTrace = tTrace;
+ sh.theSignalId = tSignalId;
+ sh.theLength = ((len+3)/4)+1;
+
+ Uint32 secPtrI[3]; // Dummy
+ globalScheduler.execute(&sh, JBB, theData, secPtrI);
+}
+
+void
+SimulatedBlock::execNODE_STATE_REP(Signal* signal){
+ const NodeStateRep * const rep = (NodeStateRep *)&signal->theData[0];
+
+ this->theNodeState = rep->nodeState;
+}
+
+void
+SimulatedBlock::execCHANGE_NODE_STATE_REQ(Signal* signal){
+ const ChangeNodeStateReq * const req =
+ (ChangeNodeStateReq *)&signal->theData[0];
+
+ this->theNodeState = req->nodeState;
+ const Uint32 senderData = req->senderData;
+ const BlockReference senderRef = req->senderRef;
+
+ /**
+ * Pack return signal
+ */
+ ChangeNodeStateConf * const conf =
+ (ChangeNodeStateConf *)&signal->theData[0];
+
+ conf->senderData = senderData;
+
+ sendSignal(senderRef, GSN_CHANGE_NODE_STATE_CONF, signal,
+ ChangeNodeStateConf::SignalLength, JBB);
+}
+
+void
+SimulatedBlock::execNDB_TAMPER(Signal * signal){
+ SET_ERROR_INSERT_VALUE(signal->theData[0]);
+}
+
+void
+SimulatedBlock::execSIGNAL_DROPPED_REP(Signal * signal){
+ ErrorReporter::handleError(ecError,
+ ERR_OUT_OF_LONG_SIGNAL_MEMORY,
+ "Signal lost, out of long signal memory",
+ __FILE__,
+ NST_ErrorHandler);
+}
+
+void
+SimulatedBlock::execCONTINUE_FRAGMENTED(Signal * signal){
+ ljamEntry();
+
+ Ptr<FragmentSendInfo> fragPtr;
+
+ c_segmentedFragmentSendList.first(fragPtr);
+ for(; !fragPtr.isNull();){
+ ljam();
+ Ptr<FragmentSendInfo> copyPtr = fragPtr;
+ c_segmentedFragmentSendList.next(fragPtr);
+
+ sendNextSegmentedFragment(signal, * copyPtr.p);
+ if(copyPtr.p->m_status == FragmentSendInfo::SendComplete){
+ ljam();
+ if(copyPtr.p->m_callback.m_callbackFunction != 0) {
+ ljam();
+ execute(signal, copyPtr.p->m_callback, 0);
+ }//if
+ c_segmentedFragmentSendList.release(copyPtr);
+ }
+ }
+
+ c_linearFragmentSendList.first(fragPtr);
+ for(; !fragPtr.isNull();){
+ ljam();
+ Ptr<FragmentSendInfo> copyPtr = fragPtr;
+ c_linearFragmentSendList.next(fragPtr);
+
+ sendNextLinearFragment(signal, * copyPtr.p);
+ if(copyPtr.p->m_status == FragmentSendInfo::SendComplete){
+ ljam();
+ if(copyPtr.p->m_callback.m_callbackFunction != 0) {
+ ljam();
+ execute(signal, copyPtr.p->m_callback, 0);
+ }//if
+ c_linearFragmentSendList.release(copyPtr);
+ }
+ }
+
+ if(c_segmentedFragmentSendList.isEmpty() &&
+ c_linearFragmentSendList.isEmpty()){
+ ljam();
+ c_fragSenderRunning = false;
+ return;
+ }
+
+ ContinueFragmented * sig = (ContinueFragmented*)signal->getDataPtrSend();
+ sig->line = __LINE__;
+ sendSignal(reference(), GSN_CONTINUE_FRAGMENTED, signal, 1, JBB);
+}
+
+#ifdef VM_TRACE_TIME
+void
+SimulatedBlock::clearTimes() {
+ for(Uint32 i = 0; i <= MAX_GSN; i++){
+ m_timeTrace[i].cnt = 0;
+ m_timeTrace[i].sum = 0;
+ m_timeTrace[i].sub = 0;
+ }
+}
+
+void
+SimulatedBlock::printTimes(FILE * output){
+ fprintf(output, "-- %s --\n", getBlockName(number()));
+ Uint64 sum = 0;
+ for(Uint32 i = 0; i <= MAX_GSN; i++){
+ Uint32 n = m_timeTrace[i].cnt;
+ if(n != 0){
+ double dn = n;
+
+ double avg = m_timeTrace[i].sum;
+ double avg2 = avg - m_timeTrace[i].sub;
+
+ avg /= dn;
+ avg2 /= dn;
+
+ fprintf(output,
+ //name ; cnt ; loc ; acc
+ "%s ; #%d ; %dus ; %dus ; %dms\n",
+ getSignalName(i), n, (Uint32)avg, (Uint32)avg2,
+ (Uint32)((m_timeTrace[i].sum - m_timeTrace[i].sub + 500)/ 1000));
+
+ sum += (m_timeTrace[i].sum - m_timeTrace[i].sub);
+ }
+ }
+ sum = (sum + 500)/ 1000;
+ fprintf(output, "-- %s : %d --\n", getBlockName(number()), sum);
+ fprintf(output, "\n");
+ fflush(output);
+}
+
+#endif
+
+void release(SegmentedSectionPtr & ptr);
+
+SimulatedBlock::FragmentInfo::FragmentInfo(Uint32 fragId, Uint32 sender){
+ m_fragmentId = fragId;
+ m_senderRef = sender;
+ m_sectionPtrI[0] = RNIL;
+ m_sectionPtrI[1] = RNIL;
+ m_sectionPtrI[2] = RNIL;
+ m_sectionPtrI[3] = RNIL;
+}
+
+SimulatedBlock::FragmentSendInfo::FragmentSendInfo()
+{
+}
+
+bool
+SimulatedBlock::assembleFragments(Signal * signal){
+ Uint32 sigLen = signal->length() - 1;
+ Uint32 fragId = signal->theData[sigLen];
+ Uint32 fragInfo = signal->header.m_fragmentInfo;
+ Uint32 senderRef = signal->getSendersBlockRef();
+
+ if(fragInfo == 0){
+ return true;
+ }
+
+ const Uint32 secs = signal->header.m_noOfSections;
+ const Uint32 * const secNos = &signal->theData[sigLen - secs];
+
+ if(fragInfo == 1){
+ /**
+ * First in train
+ */
+ Ptr<FragmentInfo> fragPtr;
+ if(!c_fragmentInfoHash.seize(fragPtr)){
+ ndbrequire(false);
+ return false;
+ }
+
+ new (fragPtr.p)FragmentInfo(fragId, senderRef);
+ c_fragmentInfoHash.add(fragPtr);
+
+ for(Uint32 i = 0; i<secs; i++){
+ Uint32 sectionNo = secNos[i];
+ ndbassert(sectionNo < 3);
+ fragPtr.p->m_sectionPtrI[sectionNo] = signal->m_sectionPtr[i].i;
+ }
+
+ /**
+ * Don't release allocated segments
+ */
+ signal->header.m_noOfSections = 0;
+ return false;
+ }
+
+ FragmentInfo key(fragId, senderRef);
+ Ptr<FragmentInfo> fragPtr;
+ if(c_fragmentInfoHash.find(fragPtr, key)){
+
+ /**
+ * FragInfo == 2 or 3
+ */
+ for(Uint32 i = 0; i<secs; i++){
+ Uint32 sectionNo = secNos[i];
+ ndbassert(sectionNo < 3);
+ Uint32 sectionPtrI = signal->m_sectionPtr[i].i;
+ if(fragPtr.p->m_sectionPtrI[sectionNo] != RNIL){
+ linkSegments(fragPtr.p->m_sectionPtrI[sectionNo], sectionPtrI);
+ } else {
+ fragPtr.p->m_sectionPtrI[sectionNo] = sectionPtrI;
+ }
+ }
+
+ /**
+ * fragInfo = 2
+ */
+ if(fragInfo == 2){
+ signal->header.m_noOfSections = 0;
+ return false;
+ }
+
+ /**
+ * fragInfo = 3
+ */
+ Uint32 i;
+ for(i = 0; i<3; i++){
+ Uint32 ptrI = fragPtr.p->m_sectionPtrI[i];
+ if(ptrI != RNIL){
+ signal->m_sectionPtr[i].i = ptrI;
+ } else {
+ break;
+ }
+ }
+ signal->setLength(sigLen - i);
+ signal->header.m_noOfSections = i;
+ signal->header.m_fragmentInfo = 0;
+ getSections(i, signal->m_sectionPtr);
+
+ c_fragmentInfoHash.release(fragPtr);
+ return true;
+ }
+
+ /**
+ * Unable to find fragment
+ */
+ ndbrequire(false);
+ return false;
+}
+
+bool
+SimulatedBlock::sendFirstFragment(FragmentSendInfo & info,
+ NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ Uint32 messageSize){
+
+ info.m_sectionPtr[0].m_segmented.i = RNIL;
+ info.m_sectionPtr[1].m_segmented.i = RNIL;
+ info.m_sectionPtr[2].m_segmented.i = RNIL;
+
+ Uint32 totalSize = 0;
+ SectionSegment * p;
+ switch(signal->header.m_noOfSections){
+ case 3:
+ p = signal->m_sectionPtr[2].p;
+ info.m_sectionPtr[2].m_segmented.p = p;
+ info.m_sectionPtr[2].m_segmented.i = signal->m_sectionPtr[2].i;
+ totalSize += p->m_sz;
+ case 2:
+ p = signal->m_sectionPtr[1].p;
+ info.m_sectionPtr[1].m_segmented.p = p;
+ info.m_sectionPtr[1].m_segmented.i = signal->m_sectionPtr[1].i;
+ totalSize += p->m_sz;
+ case 1:
+ p = signal->m_sectionPtr[0].p;
+ info.m_sectionPtr[0].m_segmented.p = p;
+ info.m_sectionPtr[0].m_segmented.i = signal->m_sectionPtr[0].i;
+ totalSize += p->m_sz;
+ }
+
+ if(totalSize <= messageSize + SectionSegment::DataLength){
+ /**
+ * Send signal directly
+ */
+ sendSignal(rg, gsn, signal, length, jbuf);
+ info.m_status = FragmentSendInfo::SendComplete;
+ return true;
+ }
+
+ /**
+ * Consume sections
+ */
+ signal->header.m_noOfSections = 0;
+
+ /**
+ * Setup info object
+ */
+ info.m_status = FragmentSendInfo::SendNotComplete;
+ info.m_prio = (Uint8)jbuf;
+ info.m_gsn = gsn;
+ info.m_fragInfo = 1;
+ info.m_messageSize = messageSize;
+ info.m_fragmentId = c_fragmentIdCounter++;
+ info.m_nodeReceiverGroup = rg;
+ info.m_callback.m_callbackFunction = 0;
+
+ Ptr<SectionSegment> tmp;
+ if(!import(tmp, &signal->theData[0], length)){
+ ndbrequire(false);
+ return false;
+ }
+ info.m_theDataSection.p = &tmp.p->theData[0];
+ info.m_theDataSection.sz = length;
+ tmp.p->theData[length] = tmp.i;
+
+ sendNextSegmentedFragment(signal, info);
+
+ if(c_fragmentIdCounter == 0){
+ /**
+ * Fragment id 0 is invalid
+ */
+ c_fragmentIdCounter = 1;
+ }
+
+ return true;
+}
+
+#if 0
+#define lsout(x) x
+#else
+#define lsout(x)
+#endif
+
+void
+SimulatedBlock::sendNextSegmentedFragment(Signal* signal,
+ FragmentSendInfo & info){
+
+ /**
+ * Store "theData"
+ */
+ const Uint32 sigLen = info.m_theDataSection.sz;
+ memcpy(&signal->theData[0], info.m_theDataSection.p, 4 * sigLen);
+
+ Uint32 sz = 0;
+ Uint32 maxSz = info.m_messageSize;
+
+ Int32 secNo = 2;
+ Uint32 secCount = 0;
+ Uint32 * secNos = &signal->theData[sigLen];
+
+ enum { Unknown = 0, Full = 1 } loop = Unknown;
+ for(; secNo >= 0 && secCount < 3; secNo--){
+ Uint32 ptrI = info.m_sectionPtr[secNo].m_segmented.i;
+ if(ptrI == RNIL)
+ continue;
+
+ info.m_sectionPtr[secNo].m_segmented.i = RNIL;
+
+ SectionSegment * ptrP = info.m_sectionPtr[secNo].m_segmented.p;
+ const Uint32 size = ptrP->m_sz;
+
+ signal->m_sectionPtr[secCount].i = ptrI;
+ signal->m_sectionPtr[secCount].p = ptrP;
+ signal->m_sectionPtr[secCount].sz = size;
+ secNos[secCount] = secNo;
+ secCount++;
+
+ const Uint32 sizeLeft = maxSz - sz;
+ if(size <= sizeLeft){
+ /**
+ * The section fits
+ */
+ sz += size;
+ lsout(ndbout_c("section %d saved as %d", secNo, secCount-1));
+ continue;
+ }
+
+ const Uint32 overflow = size - sizeLeft; // > 0
+ if(overflow <= SectionSegment::DataLength){
+ /**
+ * Only one segment left to send
+ * send even if sizeLeft <= size
+ */
+ lsout(ndbout_c("section %d saved as %d but full over: %d",
+ secNo, secCount-1, overflow));
+ secNo--;
+ break;
+ }
+
+ // size >= 61
+ if(sizeLeft < SectionSegment::DataLength){
+ /**
+ * Less than one segment left (space)
+ * dont bother sending
+ */
+ secCount--;
+ info.m_sectionPtr[secNo].m_segmented.i = ptrI;
+ loop = Full;
+ lsout(ndbout_c("section %d not saved", secNo));
+ break;
+ }
+
+ /**
+ * Split list
+ * 1) Find place to split
+ * 2) Rewrite header (the part that will be sent)
+ * 3) Write new header (for remaining part)
+ * 4) Store new header on FragmentSendInfo - record
+ */
+ // size >= 61 && sizeLeft >= 60
+ Uint32 sum = SectionSegment::DataLength;
+ Uint32 prevPtrI = ptrI;
+ ptrI = ptrP->m_nextSegment;
+ const Uint32 fill = sizeLeft - SectionSegment::DataLength;
+ while(sum < fill){
+ prevPtrI = ptrI;
+ ptrP = g_sectionSegmentPool.getPtr(ptrI);
+ ptrI = ptrP->m_nextSegment;
+ sum += SectionSegment::DataLength;
+ }
+
+ /**
+ * Rewrite header w.r.t size and last
+ */
+ Uint32 prev = secCount - 1;
+ const Uint32 last = signal->m_sectionPtr[prev].p->m_lastSegment;
+ signal->m_sectionPtr[prev].p->m_lastSegment = prevPtrI;
+ signal->m_sectionPtr[prev].p->m_sz = sum;
+ signal->m_sectionPtr[prev].sz = sum;
+
+ /**
+ * Write "new" list header
+ */
+ ptrP = g_sectionSegmentPool.getPtr(ptrI);
+ ptrP->m_lastSegment = last;
+ ptrP->m_sz = size - sum;
+
+ /**
+ * And store it on info-record
+ */
+ info.m_sectionPtr[secNo].m_segmented.i = ptrI;
+ info.m_sectionPtr[secNo].m_segmented.p = ptrP;
+
+ loop = Full;
+ lsout(ndbout_c("section %d split into %d", secNo, prev));
+ break;
+ }
+
+ lsout(ndbout_c("loop: %d secNo: %d secCount: %d sz: %d",
+ loop, secNo, secCount, sz));
+
+ /**
+ * Store fragment id
+ */
+ secNos[secCount] = info.m_fragmentId;
+
+ Uint32 fragInfo = info.m_fragInfo;
+ info.m_fragInfo = 2;
+ switch(loop){
+ case Unknown:
+ if(secNo >= 0){
+ lsout(ndbout_c("Unknown - Full"));
+ /**
+ * Not finished
+ */
+ break;
+ }
+ // Fall through
+ lsout(ndbout_c("Unknown - Done"));
+ info.m_status = FragmentSendInfo::SendComplete;
+ ndbassert(fragInfo == 2);
+ fragInfo = 3;
+ case Full:
+ break;
+ }
+
+ signal->header.m_fragmentInfo = fragInfo;
+ signal->header.m_noOfSections = secCount;
+
+ sendSignal(info.m_nodeReceiverGroup,
+ info.m_gsn,
+ signal,
+ sigLen + secCount + 1,
+ (JobBufferLevel)info.m_prio);
+
+ if(fragInfo == 3){
+ /**
+ * This is the last signal
+ */
+ g_sectionSegmentPool.release(info.m_theDataSection.p[sigLen]);
+ }
+}
+
+bool
+SimulatedBlock::sendFirstFragment(FragmentSendInfo & info,
+ NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections,
+ Uint32 messageSize){
+
+ ::releaseSections(signal->header.m_noOfSections, signal->m_sectionPtr);
+ signal->header.m_noOfSections = 0;
+
+ info.m_sectionPtr[0].m_linear.p = NULL;
+ info.m_sectionPtr[1].m_linear.p = NULL;
+ info.m_sectionPtr[2].m_linear.p = NULL;
+
+ Uint32 totalSize = 0;
+ switch(noOfSections){
+ case 3:
+ info.m_sectionPtr[2].m_linear = ptr[2];
+ totalSize += ptr[2].sz;
+ case 2:
+ info.m_sectionPtr[1].m_linear = ptr[1];
+ totalSize += ptr[1].sz;
+ case 1:
+ info.m_sectionPtr[0].m_linear = ptr[0];
+ totalSize += ptr[0].sz;
+ }
+
+ if(totalSize <= messageSize + SectionSegment::DataLength){
+ /**
+ * Send signal directly
+ */
+ sendSignal(rg, gsn, signal, length, jbuf, ptr, noOfSections);
+ info.m_status = FragmentSendInfo::SendComplete;
+
+ /**
+ * Indicate to sendLinearSignalFragment
+ * that we'r already done
+ */
+ return true;
+ }
+
+ /**
+ * Setup info object
+ */
+ info.m_status = FragmentSendInfo::SendNotComplete;
+ info.m_prio = (Uint8)jbuf;
+ info.m_gsn = gsn;
+ info.m_messageSize = messageSize;
+ info.m_fragInfo = 1;
+ info.m_fragmentId = c_fragmentIdCounter++;
+ info.m_nodeReceiverGroup = rg;
+ info.m_callback.m_callbackFunction = 0;
+
+ Ptr<SectionSegment> tmp;
+ if(!import(tmp, &signal->theData[0], length)){
+ ndbrequire(false);
+ return false;
+ }
+
+ info.m_theDataSection.p = &tmp.p->theData[0];
+ info.m_theDataSection.sz = length;
+ tmp.p->theData[length] = tmp.i;
+
+ sendNextLinearFragment(signal, info);
+
+ if(c_fragmentIdCounter == 0){
+ /**
+ * Fragment id 0 is invalid
+ */
+ c_fragmentIdCounter = 1;
+ }
+
+ return true;
+}
+
+void
+SimulatedBlock::sendNextLinearFragment(Signal* signal,
+ FragmentSendInfo & info){
+
+ /**
+ * Store "theData"
+ */
+ const Uint32 sigLen = info.m_theDataSection.sz;
+ memcpy(&signal->theData[0], info.m_theDataSection.p, 4 * sigLen);
+
+ Uint32 sz = 0;
+ Uint32 maxSz = info.m_messageSize;
+
+ Int32 secNo = 2;
+ Uint32 secCount = 0;
+ Uint32 * secNos = &signal->theData[sigLen];
+ LinearSectionPtr signalPtr[3];
+
+ enum { Unknown = 0, Full = 2 } loop = Unknown;
+ for(; secNo >= 0 && secCount < 3; secNo--){
+ Uint32 * ptrP = info.m_sectionPtr[secNo].m_linear.p;
+ if(ptrP == NULL)
+ continue;
+
+ info.m_sectionPtr[secNo].m_linear.p = NULL;
+ const Uint32 size = info.m_sectionPtr[secNo].m_linear.sz;
+
+ signalPtr[secCount].p = ptrP;
+ signalPtr[secCount].sz = size;
+ secNos[secCount] = secNo;
+ secCount++;
+
+ const Uint32 sizeLeft = maxSz - sz;
+ if(size <= sizeLeft){
+ /**
+ * The section fits
+ */
+ sz += size;
+ lsout(ndbout_c("section %d saved as %d", secNo, secCount-1));
+ continue;
+ }
+
+ const Uint32 overflow = size - sizeLeft; // > 0
+ if(overflow <= SectionSegment::DataLength){
+ /**
+ * Only one segment left to send
+ * send even if sizeLeft <= size
+ */
+ lsout(ndbout_c("section %d saved as %d but full over: %d",
+ secNo, secCount-1, overflow));
+ secNo--;
+ break;
+ }
+
+ // size >= 61
+ if(sizeLeft < SectionSegment::DataLength){
+ /**
+ * Less than one segment left (space)
+ * dont bother sending
+ */
+ secCount--;
+ info.m_sectionPtr[secNo].m_linear.p = ptrP;
+ loop = Full;
+ lsout(ndbout_c("section %d not saved", secNo));
+ break;
+ }
+
+ /**
+ * Split list
+ * 1) Find place to split
+ * 2) Rewrite header (the part that will be sent)
+ * 3) Write new header (for remaining part)
+ * 4) Store new header on FragmentSendInfo - record
+ */
+ Uint32 sum = sizeLeft;
+ sum /= SectionSegment::DataLength;
+ sum *= SectionSegment::DataLength;
+
+ /**
+ * Rewrite header w.r.t size
+ */
+ Uint32 prev = secCount - 1;
+ signalPtr[prev].sz = sum;
+
+ /**
+ * Write/store "new" header
+ */
+ info.m_sectionPtr[secNo].m_linear.p = ptrP + sum;
+ info.m_sectionPtr[secNo].m_linear.sz = size - sum;
+
+ loop = Full;
+ lsout(ndbout_c("section %d split into %d", secNo, prev));
+ break;
+ }
+
+ lsout(ndbout_c("loop: %d secNo: %d secCount: %d sz: %d",
+ loop, secNo, secCount, sz));
+
+ /**
+ * Store fragment id
+ */
+ secNos[secCount] = info.m_fragmentId;
+
+ Uint32 fragInfo = info.m_fragInfo;
+ info.m_fragInfo = 2;
+ switch(loop){
+ case Unknown:
+ if(secNo >= 0){
+ lsout(ndbout_c("Unknown - Full"));
+ /**
+ * Not finished
+ */
+ break;
+ }
+ // Fall through
+ lsout(ndbout_c("Unknown - Done"));
+ info.m_status = FragmentSendInfo::SendComplete;
+ ndbassert(fragInfo == 2);
+ fragInfo = 3;
+ case Full:
+ break;
+ }
+
+ signal->header.m_noOfSections = 0;
+ signal->header.m_fragmentInfo = fragInfo;
+
+ sendSignal(info.m_nodeReceiverGroup,
+ info.m_gsn,
+ signal,
+ sigLen + secCount + 1,
+ (JobBufferLevel)info.m_prio,
+ signalPtr,
+ secCount);
+
+ if(fragInfo == 3){
+ /**
+ * This is the last signal
+ */
+ g_sectionSegmentPool.release(info.m_theDataSection.p[sigLen]);
+ }
+}
+
+void
+SimulatedBlock::sendFragmentedSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ Callback & c,
+ Uint32 messageSize){
+ bool res = true;
+ Ptr<FragmentSendInfo> ptr;
+ res = c_segmentedFragmentSendList.seize(ptr);
+ ndbrequire(res);
+
+ res = sendFirstFragment(* ptr.p,
+ NodeReceiverGroup(ref),
+ gsn,
+ signal,
+ length,
+ jbuf,
+ messageSize);
+ ndbrequire(res);
+
+ if(ptr.p->m_status == FragmentSendInfo::SendComplete){
+ c_segmentedFragmentSendList.release(ptr);
+ if(c.m_callbackFunction != 0)
+ execute(signal, c, 0);
+ return;
+ }
+ ptr.p->m_callback = c;
+
+ if(!c_fragSenderRunning){
+ c_fragSenderRunning = true;
+ ContinueFragmented * sig = (ContinueFragmented*)signal->getDataPtrSend();
+ sig->line = __LINE__;
+ sendSignal(reference(), GSN_CONTINUE_FRAGMENTED, signal, 1, JBB);
+ }
+}
+
+void
+SimulatedBlock::sendFragmentedSignal(NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ Callback & c,
+ Uint32 messageSize){
+ bool res = true;
+ Ptr<FragmentSendInfo> ptr;
+ res = c_segmentedFragmentSendList.seize(ptr);
+ ndbrequire(res);
+
+ res = sendFirstFragment(* ptr.p,
+ rg,
+ gsn,
+ signal,
+ length,
+ jbuf,
+ messageSize);
+ ndbrequire(res);
+
+ if(ptr.p->m_status == FragmentSendInfo::SendComplete){
+ c_segmentedFragmentSendList.release(ptr);
+ if(c.m_callbackFunction != 0)
+ execute(signal, c, 0);
+ return;
+ }
+ ptr.p->m_callback = c;
+
+ if(!c_fragSenderRunning){
+ c_fragSenderRunning = true;
+ ContinueFragmented * sig = (ContinueFragmented*)signal->getDataPtrSend();
+ sig->line = __LINE__;
+ sendSignal(reference(), GSN_CONTINUE_FRAGMENTED, signal, 1, JBB);
+ }
+}
+
+Callback SimulatedBlock::TheEmptyCallback = {0, 0};
+
+void
+SimulatedBlock::sendFragmentedSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections,
+ Callback & c,
+ Uint32 messageSize){
+ bool res = true;
+ Ptr<FragmentSendInfo> tmp;
+ res = c_linearFragmentSendList.seize(tmp);
+ ndbrequire(res);
+
+ res = sendFirstFragment(* tmp.p,
+ NodeReceiverGroup(ref),
+ gsn,
+ signal,
+ length,
+ jbuf,
+ ptr,
+ noOfSections,
+ messageSize);
+ ndbrequire(res);
+
+ if(tmp.p->m_status == FragmentSendInfo::SendComplete){
+ c_linearFragmentSendList.release(tmp);
+ if(c.m_callbackFunction != 0)
+ execute(signal, c, 0);
+ return;
+ }
+ tmp.p->m_callback = c;
+
+ if(!c_fragSenderRunning){
+ c_fragSenderRunning = true;
+ ContinueFragmented * sig = (ContinueFragmented*)signal->getDataPtrSend();
+ sig->line = __LINE__;
+ sendSignal(reference(), GSN_CONTINUE_FRAGMENTED, signal, 1, JBB);
+ }
+}
+
+void
+SimulatedBlock::sendFragmentedSignal(NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections,
+ Callback & c,
+ Uint32 messageSize){
+ bool res = true;
+ Ptr<FragmentSendInfo> tmp;
+ res = c_linearFragmentSendList.seize(tmp);
+ ndbrequire(res);
+
+ res = sendFirstFragment(* tmp.p,
+ rg,
+ gsn,
+ signal,
+ length,
+ jbuf,
+ ptr,
+ noOfSections,
+ messageSize);
+ ndbrequire(res);
+
+ if(tmp.p->m_status == FragmentSendInfo::SendComplete){
+ c_linearFragmentSendList.release(tmp);
+ if(c.m_callbackFunction != 0)
+ execute(signal, c, 0);
+ return;
+ }
+ tmp.p->m_callback = c;
+
+ if(!c_fragSenderRunning){
+ c_fragSenderRunning = true;
+ ContinueFragmented * sig = (ContinueFragmented*)signal->getDataPtrSend();
+ sig->line = __LINE__;
+ sendSignal(reference(), GSN_CONTINUE_FRAGMENTED, signal, 1, JBB);
+ }
+}
+
+NodeInfo &
+SimulatedBlock::setNodeInfo(NodeId nodeId) {
+ ndbrequire(nodeId > 0 && nodeId < MAX_NODES);
+ return globalData.m_nodeInfo[nodeId];
+}
+
+void
+SimulatedBlock::execUTIL_CREATE_LOCK_REF(Signal* signal){
+ ljamEntry();
+ c_mutexMgr.execUTIL_CREATE_LOCK_REF(signal);
+}
+
+void SimulatedBlock::execUTIL_CREATE_LOCK_CONF(Signal* signal){
+ ljamEntry();
+ c_mutexMgr.execUTIL_CREATE_LOCK_CONF(signal);
+}
+
+void SimulatedBlock::execUTIL_DESTORY_LOCK_REF(Signal* signal){
+ ljamEntry();
+ c_mutexMgr.execUTIL_DESTORY_LOCK_REF(signal);
+}
+
+void SimulatedBlock::execUTIL_DESTORY_LOCK_CONF(Signal* signal){
+ ljamEntry();
+ c_mutexMgr.execUTIL_DESTORY_LOCK_CONF(signal);
+}
+
+void SimulatedBlock::execUTIL_LOCK_REF(Signal* signal){
+ ljamEntry();
+ c_mutexMgr.execUTIL_LOCK_REF(signal);
+}
+
+void SimulatedBlock::execUTIL_LOCK_CONF(Signal* signal){
+ ljamEntry();
+ c_mutexMgr.execUTIL_LOCK_CONF(signal);
+}
+
+void SimulatedBlock::execUTIL_UNLOCK_REF(Signal* signal){
+ ljamEntry();
+ c_mutexMgr.execUTIL_UNLOCK_REF(signal);
+}
+
+void SimulatedBlock::execUTIL_UNLOCK_CONF(Signal* signal){
+ ljamEntry();
+ c_mutexMgr.execUTIL_UNLOCK_CONF(signal);
+}
+
+void
+SimulatedBlock::ignoreMutexUnlockCallback(Signal* signal,
+ Uint32 ptrI, Uint32 retVal){
+ c_mutexMgr.release(ptrI);
+}
+
diff --git a/ndb/src/kernel/vm/SimulatedBlock.hpp b/ndb/src/kernel/vm/SimulatedBlock.hpp
new file mode 100644
index 00000000000..42b8a3034f5
--- /dev/null
+++ b/ndb/src/kernel/vm/SimulatedBlock.hpp
@@ -0,0 +1,665 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SIMULATEDBLOCK_H
+#define SIMULATEDBLOCK_H
+
+#include <NdbTick.h>
+#include <kernel_types.h>
+#include <ndb_version.h>
+
+#include "VMSignal.hpp"
+#include <RefConvert.hpp>
+#include <BlockNumbers.h>
+#include <GlobalSignalNumbers.h>
+#include "pc.hpp"
+#include <NodeInfo.hpp>
+#include <NodeState.hpp>
+#include "GlobalData.hpp"
+#include "LongSignal.hpp"
+#include <SignalLoggerManager.hpp>
+
+#include <Error.hpp>
+#include <ErrorReporter.hpp>
+#include <ErrorHandlingMacros.hpp>
+
+#include <new>
+#include "DLList.hpp"
+#include "ArrayPool.hpp"
+#include "DLHashTable.hpp"
+#include "Callback.hpp"
+#include "Mutex.hpp"
+#include "SafeCounter.hpp"
+#include "MetaData.hpp"
+
+/**
+ * Something for filesystem access
+ */
+struct NewBaseAddrBits /* 32 bits */
+{
+ unsigned int q : 4; /* Highest index - 2log */
+ /* Strings are treated as 16 bit indexed */
+ /* variables with the number of characters in */
+ /* index 0, byte 0 */
+ unsigned int v : 3; /* Size in bits - 2log */
+ unsigned int unused : 25 ;
+};
+
+typedef struct NewVar
+{
+ Uint32 * WA;
+ Uint32 nrr;
+ Uint32 ClusterSize; /* Real Cluster size */
+ NewBaseAddrBits bits;
+} NewVARIABLE; /* 128 bits */
+
+class SimulatedBlock {
+ friend class MutexManager;
+ friend class SafeCounter;
+ friend class SafeCounterManager;
+public:
+ friend class BlockComponent;
+ virtual ~SimulatedBlock();
+
+protected:
+ /**
+ * Constructor
+ */
+ SimulatedBlock(BlockNumber blockNumber,
+ const class Configuration & theConfiguration);
+
+ /**********************************************************
+ * Handling of execFunctions
+ */
+ typedef void (SimulatedBlock::* ExecFunction)(Signal* signal);
+ void addRecSignalImpl(GlobalSignalNumber g, ExecFunction fun, bool f = false);
+ void installSimulatedBlockFunctions();
+ ExecFunction theExecArray[MAX_GSN+1];
+public:
+ /**
+ *
+ */
+ inline void executeFunction(GlobalSignalNumber gsn, Signal* signal);
+protected:
+ static Callback TheEmptyCallback;
+ void execute(Signal* signal, Callback & c, Uint32 returnCode);
+
+
+ /**********************************************************
+ * Send signal - dialects
+ */
+
+ void sendSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf ) const ;
+
+ void sendSignal(NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf ) const ;
+
+ void sendSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections) const ;
+
+ void sendSignal(NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections) const ;
+
+ // Send multiple signal with delay. In this VM the jobbufffer level has
+ // no effect on on delayed signals
+ //
+ void sendSignalWithDelay(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 delayInMilliSeconds,
+ Uint32 length) const ;
+
+ void EXECUTE_DIRECT(Uint32 block,
+ Uint32 gsn,
+ Signal* signal,
+ Uint32 len);
+
+ class SectionSegmentPool& getSectionSegmentPool();
+ void releaseSections(Signal* signal);
+
+ /**********************************************************
+ * Fragmented signals
+ */
+
+ /**
+ * Assemble fragments
+ *
+ * @return true if all fragments has arrived
+ * false otherwise
+ */
+ bool assembleFragments(Signal * signal);
+
+ void sendFragmentedSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ Callback & = TheEmptyCallback,
+ Uint32 messageSize = 240);
+
+ void sendFragmentedSignal(NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ Callback & = TheEmptyCallback,
+ Uint32 messageSize = 240);
+
+ void sendFragmentedSignal(BlockReference ref,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections,
+ Callback &,
+ Uint32 messageSize = 240);
+
+ void sendFragmentedSignal(NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections,
+ Callback &,
+ Uint32 messageSize = 240);
+
+ /**********************************************************
+ * Fragmented signals structures
+ */
+
+ /**
+ * Struct used when assembling fragmented long signals at receiver side
+ */
+ struct FragmentInfo {
+ FragmentInfo(Uint32 fragId, Uint32 sender);
+
+ Uint32 m_senderRef;
+ Uint32 m_fragmentId;
+ Uint32 m_sectionPtrI[3];
+ union {
+ Uint32 nextPool;
+ Uint32 nextHash;
+ };
+ Uint32 prevHash;
+
+ inline bool equal(FragmentInfo & p) const {
+ return m_senderRef == p.m_senderRef && m_fragmentId == p.m_fragmentId;
+ }
+
+ inline Uint32 hashValue() const {
+ return m_senderRef + m_fragmentId ;
+ }
+ }; // sizeof() = 32 bytes
+
+ /**
+ * Struct used when sending fragmented signals
+ */
+ struct FragmentSendInfo {
+ FragmentSendInfo();
+
+ enum Status {
+ SendNotComplete = 0,
+ SendComplete = 1
+ };
+ Uint8 m_status;
+ Uint8 m_prio;
+ Uint16 m_fragInfo;
+ Uint16 m_gsn;
+ Uint16 m_messageSize; // Size of each fragment
+ Uint32 m_fragmentId;
+ union {
+ Ptr<struct SectionSegment> m_segmented;
+ LinearSectionPtr m_linear;
+ } m_sectionPtr[3];
+ LinearSectionPtr m_theDataSection;
+ NodeReceiverGroup m_nodeReceiverGroup; // 3
+ Callback m_callback;
+ union {
+ Uint32 nextPool;
+ Uint32 nextList;
+ };
+ Uint32 prevList;
+ };
+
+ /**
+ * setupFragmentSendInfo
+ * Setup a struct to be used with sendSignalFragment
+ * Used by sendFragmentedSignal
+ */
+ bool sendFirstFragment(FragmentSendInfo & info,
+ NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections,
+ Uint32 messageSize = 240);
+
+ bool sendFirstFragment(FragmentSendInfo & info,
+ NodeReceiverGroup rg,
+ GlobalSignalNumber gsn,
+ Signal* signal,
+ Uint32 length,
+ JobBufferLevel jbuf,
+ Uint32 messageSize = 240);
+
+ /**
+ * Send signal fragment
+ *
+ * @see sendFragmentedSignal
+ */
+ void sendNextSegmentedFragment(Signal* signal, FragmentSendInfo & info);
+
+ /**
+ * Send signal fragment
+ *
+ * @see sendFragmentedSignal
+ */
+ void sendNextLinearFragment(Signal* signal, FragmentSendInfo & info);
+
+ BlockNumber number() const;
+ BlockReference reference() const;
+ NodeId getOwnNodeId() const;
+
+ /**
+ * Prog error
+ * This function should be called when this node should be shutdown
+ * If the cause of the shutdown is known use extradata to add an
+ * errormessage describing the problem
+ */
+ void progError(int line, int err_code, const char* extradata=NULL) const ;
+private:
+ void signal_error(Uint32, Uint32, Uint32, const char*, int) const ;
+ const NodeId theNodeId;
+ const BlockNumber theNumber;
+ const BlockReference theReference;
+
+protected:
+ NewVARIABLE* allocateBat(int batSize);
+ void freeBat();
+ static const NewVARIABLE* getBat (BlockNumber blockNo);
+ static Uint16 getBatSize(BlockNumber blockNo);
+
+ static BlockReference calcTcBlockRef (NodeId aNode);
+ static BlockReference calcLqhBlockRef (NodeId aNode);
+ static BlockReference calcAccBlockRef (NodeId aNode);
+ static BlockReference calcTupBlockRef (NodeId aNode);
+ static BlockReference calcTuxBlockRef (NodeId aNode);
+ static BlockReference calcDihBlockRef (NodeId aNode);
+ static BlockReference calcQmgrBlockRef (NodeId aNode);
+ static BlockReference calcDictBlockRef (NodeId aNode);
+ static BlockReference calcNdbCntrBlockRef (NodeId aNode);
+ static BlockReference calcTrixBlockRef (NodeId aNode);
+ static BlockReference calcBackupBlockRef (NodeId aNode);
+ static BlockReference calcSumaBlockRef (NodeId aNode);
+
+ static BlockReference calcApiClusterMgrBlockRef (NodeId aNode);
+
+ /**
+ * allocRecord
+ * Allocates memory for the datastructures where ndb keeps the data
+ *
+ */
+ void* allocRecord(const char * type, size_t s, size_t n) const ;
+
+ /**
+ * Deallocate record
+ *
+ * NOTE: Also resets pointer
+ */
+ void deallocRecord(void **, const char * type, size_t s, size_t n) const ;
+
+ /**
+ * General info event (sent to cluster log)
+ */
+ void infoEvent(const char * msg, ...) const ;
+ void warningEvent(const char * msg, ...) const ;
+
+ /**
+ * The configuration object
+ */
+ const class Configuration & theConfiguration;
+
+ /**
+ * Get node state
+ */
+ const NodeState & getNodeState() const;
+
+ /**
+ * Get node info
+ */
+ const NodeInfo & getNodeInfo(NodeId nodeId) const;
+ NodeInfo & setNodeInfo(NodeId);
+
+private:
+ NewVARIABLE* NewVarRef; /* New Base Address Table for block */
+ Uint16 theBATSize; /* # entries in BAT */
+
+ /**
+ * Node state
+ */
+ NodeState theNodeState;
+ void execNDB_TAMPER(Signal * signal);
+ void execNODE_STATE_REP(Signal* signal);
+ void execCHANGE_NODE_STATE_REQ(Signal* signal);
+
+ void execSIGNAL_DROPPED_REP(Signal* signal);
+ void execCONTINUE_FRAGMENTED(Signal* signal);
+
+ Uint32 c_fragmentIdCounter;
+ ArrayPool<FragmentInfo> c_fragmentInfoPool;
+ DLHashTable<FragmentInfo> c_fragmentInfoHash;
+
+ bool c_fragSenderRunning;
+ ArrayPool<FragmentSendInfo> c_fragmentSendPool;
+ DLList<FragmentSendInfo> c_linearFragmentSendList;
+ DLList<FragmentSendInfo> c_segmentedFragmentSendList;
+
+public:
+ MutexManager c_mutexMgr;
+
+ void ignoreMutexUnlockCallback(Signal* signal, Uint32 ptrI, Uint32 retVal);
+
+ SafeCounterManager c_counterMgr;
+private:
+ void execUTIL_CREATE_LOCK_REF(Signal* signal);
+ void execUTIL_CREATE_LOCK_CONF(Signal* signal);
+ void execUTIL_DESTORY_LOCK_REF(Signal* signal);
+ void execUTIL_DESTORY_LOCK_CONF(Signal* signal);
+ void execUTIL_LOCK_REF(Signal* signal);
+ void execUTIL_LOCK_CONF(Signal* signal);
+ void execUTIL_UNLOCK_REF(Signal* signal);
+ void execUTIL_UNLOCK_CONF(Signal* signal);
+
+protected:
+
+ // Variable for storing inserted errors, see pc.H
+ ERROR_INSERT_VARIABLE;
+
+private:
+ // Metadata common part shared by block instances
+ MetaData::Common* c_ptrMetaDataCommon;
+public:
+ void setMetaDataCommon(MetaData::Common* ptr) { c_ptrMetaDataCommon = ptr; }
+ MetaData::Common* getMetaDataCommon() { return c_ptrMetaDataCommon; }
+
+#ifdef VM_TRACE_TIME
+public:
+ void clearTimes();
+ void printTimes(FILE * output);
+ void addTime(Uint32 gsn, Uint64 time);
+ void subTime(Uint32 gsn, Uint64 time);
+ struct TimeTrace {
+ Uint32 cnt;
+ Uint64 sum, sub;
+ } m_timeTrace[MAX_GSN+1];
+ Uint32 m_currentGsn;
+#endif
+};
+
+inline
+void
+SimulatedBlock::executeFunction(GlobalSignalNumber gsn, Signal* signal){
+ ExecFunction f = theExecArray[gsn];
+ if(gsn <= MAX_GSN && f != 0){
+ (this->*f)(signal);
+ return;
+ }
+
+ /**
+ * This point only passed if an error has occurred
+ */
+ char errorMsg[255];
+ if (!(gsn <= MAX_GSN)) {
+ snprintf(errorMsg, 255, "Illegal signal received (GSN %d too high)", gsn);
+ REQUIRE(false, errorMsg);
+ }
+ if (!(theExecArray[gsn] != 0)) {
+ snprintf(errorMsg, 255, "Illegal signal received (GSN %d not added)", gsn);
+ REQUIRE(false, errorMsg);
+ }
+ ndbrequire(false);
+}
+
+inline
+void
+SimulatedBlock::execute(Signal* signal, Callback & c, Uint32 returnCode){
+ CallbackFunction fun = c.m_callbackFunction;
+ ndbrequire(fun != 0);
+ c.m_callbackFunction = NULL;
+ (this->*fun)(signal, c.m_callbackData, returnCode);
+}
+
+inline
+BlockNumber
+SimulatedBlock::number() const {
+ return theNumber;
+}
+
+inline
+BlockReference
+SimulatedBlock::reference() const {
+ return theReference;
+}
+
+inline
+NodeId
+SimulatedBlock::getOwnNodeId() const {
+ return theNodeId;
+}
+
+inline
+BlockReference
+SimulatedBlock::calcTcBlockRef (NodeId aNodeId){
+ return numberToRef(DBTC, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcLqhBlockRef (NodeId aNodeId){
+return numberToRef(DBLQH, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcAccBlockRef (NodeId aNodeId){
+ return numberToRef(DBACC, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcTupBlockRef (NodeId aNodeId){
+ return numberToRef(DBTUP, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcTuxBlockRef (NodeId aNodeId){
+ return numberToRef(DBTUX, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcDihBlockRef (NodeId aNodeId){
+ return numberToRef(DBDIH, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcDictBlockRef (NodeId aNodeId){
+ return numberToRef(DBDICT, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcQmgrBlockRef (NodeId aNodeId){
+ return numberToRef(QMGR, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcNdbCntrBlockRef (NodeId aNodeId){
+ return numberToRef(NDBCNTR, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcTrixBlockRef (NodeId aNodeId){
+ return numberToRef(TRIX, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcBackupBlockRef (NodeId aNodeId){
+ return numberToRef(BACKUP, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcSumaBlockRef (NodeId aNodeId){
+ return numberToRef(SUMA, aNodeId);
+}
+
+inline
+BlockReference
+SimulatedBlock::calcApiClusterMgrBlockRef (NodeId aNodeId){
+ return numberToRef(API_CLUSTERMGR, aNodeId);
+}
+
+inline
+const NodeState &
+SimulatedBlock::getNodeState() const {
+ return theNodeState;
+}
+
+inline
+const NodeInfo &
+SimulatedBlock::getNodeInfo(NodeId nodeId) const {
+ ndbrequire(nodeId > 0 && nodeId < MAX_NODES);
+ return globalData.m_nodeInfo[nodeId];
+}
+
+inline
+void
+SimulatedBlock::EXECUTE_DIRECT(Uint32 block,
+ Uint32 gsn,
+ Signal* signal,
+ Uint32 len){
+ signal->setLength(len);
+#ifdef VM_TRACE
+ if(globalData.testOn){
+ signal->header.theVerId_signalNumber = gsn;
+ signal->header.theReceiversBlockNumber = block;
+ signal->header.theSendersBlockRef = reference();
+ globalSignalLoggers.executeDirect(signal->header,
+ 0, // in
+ &signal->theData[0],
+ globalData.ownId);
+ }
+#endif
+ SimulatedBlock* b = globalData.getBlock(block);
+#ifdef VM_TRACE_TIME
+ Uint32 us1, us2;
+ Uint64 ms1, ms2;
+ NdbTick_CurrentMicrosecond(&ms1, &us1);
+ Uint32 tGsn = m_currentGsn;
+ b->m_currentGsn = gsn;
+#endif
+ b->executeFunction(gsn, signal);
+#ifdef VM_TRACE_TIME
+ NdbTick_CurrentMicrosecond(&ms2, &us2);
+ Uint64 diff = ms2;
+ diff -= ms1;
+ diff *= 1000000;
+ diff += us2;
+ diff -= us1;
+ b->addTime(gsn, diff);
+ m_currentGsn = tGsn;
+ subTime(tGsn, diff);
+#endif
+#ifdef VM_TRACE
+ if(globalData.testOn){
+ signal->header.theVerId_signalNumber = gsn;
+ signal->header.theReceiversBlockNumber = block;
+ signal->header.theSendersBlockRef = reference();
+ globalSignalLoggers.executeDirect(signal->header,
+ 1, // out
+ &signal->theData[0],
+ globalData.ownId);
+ }
+#endif
+}
+
+#ifdef VM_TRACE_TIME
+inline
+void
+SimulatedBlock::addTime(Uint32 gsn, Uint64 time){
+ m_timeTrace[gsn].cnt ++;
+ m_timeTrace[gsn].sum += time;
+}
+
+inline
+void
+SimulatedBlock::subTime(Uint32 gsn, Uint64 time){
+ m_timeTrace[gsn].sub += time;
+}
+#endif
+
+/**
+ * Defines for backward compatiblility
+ */
+
+#define BLOCK_DEFINES(BLOCK) \
+ typedef void (BLOCK::* ExecSignalLocal) (Signal* signal); \
+ typedef void (BLOCK::* BlockCallback)(Signal*, Uint32 callb, Uint32 retCode); \
+ inline CallbackFunction safe_cast(BlockCallback f){ \
+ return static_cast<CallbackFunction>(f); \
+ } \
+public:\
+private: \
+ void addRecSignal(GlobalSignalNumber gsn, ExecSignalLocal f, bool force = false)
+
+#define BLOCK_CONSTRUCTOR(BLOCK)
+
+#define BLOCK_FUNCTIONS(BLOCK) \
+void \
+BLOCK::addRecSignal(GlobalSignalNumber gsn, ExecSignalLocal f, bool force){ \
+ addRecSignalImpl(gsn, (ExecFunction)f, force);\
+}
+
+
+#endif
+
diff --git a/ndb/src/kernel/vm/ThreadConfig.cpp b/ndb/src/kernel/vm/ThreadConfig.cpp
new file mode 100644
index 00000000000..d18b20a5bb5
--- /dev/null
+++ b/ndb/src/kernel/vm/ThreadConfig.cpp
@@ -0,0 +1,207 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "ThreadConfig.hpp"
+#include "Emulator.hpp"
+#include "GlobalData.hpp"
+#include "TimeQueue.hpp"
+#include "TransporterRegistry.hpp"
+#include "FastScheduler.hpp"
+#include "pc.hpp"
+
+#include <GlobalSignalNumbers.h>
+#include <BlockNumbers.h>
+
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#include <NdbOut.hpp>
+
+#include <signaldata/StartOrd.hpp>
+
+ThreadConfig::ThreadConfig()
+{
+}
+
+ThreadConfig::~ThreadConfig()
+{
+}
+
+/**
+ * For each millisecond that has passed since this function was last called:
+ * Scan the job buffer and increment the internalMillisecCounter
+ * with 1 to keep track of where we are
+ */
+inline
+void
+ThreadConfig::scanTimeQueue()
+{
+ unsigned int maxCounter;
+ Uint64 currMilliSecond;
+ maxCounter = 0;
+ currMilliSecond = NdbTick_CurrentMillisecond();
+ if (currMilliSecond < globalData.internalMillisecCounter) {
+//--------------------------------------------------------------------
+// This could occur around 2036 or if the operator decides to change
+// time backwards. We cannot know how long time has past since last
+// time and we make a best try with 0 milliseconds.
+//--------------------------------------------------------------------
+#ifdef VM_TRACE
+ ndbout << "Time moved backwards with ";
+ ndbout << (globalData.internalMillisecCounter - currMilliSecond);
+ ndbout << " milliseconds" << endl;
+#endif
+ globalData.internalMillisecCounter = currMilliSecond;
+ }//if
+ if (currMilliSecond > (globalData.internalMillisecCounter + 1500)) {
+//--------------------------------------------------------------------
+// Time has moved forward more than a second. Either it could happen
+// if operator changed the time or if the OS has misbehaved badly.
+// We set the new time to one second from the past.
+//--------------------------------------------------------------------
+#ifdef VM_TRACE
+ ndbout << "Time moved forward with ";
+ ndbout << (currMilliSecond - globalData.internalMillisecCounter);
+ ndbout << " milliseconds" << endl;
+#endif
+ globalData.internalMillisecCounter = currMilliSecond - 1000;
+ }//if
+ while (((currMilliSecond - globalData.internalMillisecCounter) > 0) &&
+ (maxCounter < 20)){
+ globalData.internalMillisecCounter++;
+ maxCounter++;
+ globalTimeQueue.scanTable();
+ }//while
+}//ThreadConfig::scanTimeQueue()
+
+
+//--------------------------------------------------------------------
+// ipControlLoop -- The main loop of ndb.
+// Handles the scheduling of signal execution and input/output
+// One lap in the loop should take approximately 10 milli seconds
+// If the jobbuffer is empty and the laptime is less than 10 milliseconds
+// at the end of the loop
+// the TransporterRegistry is called in order to sleep on the IO ports
+// waiting for another incoming signal to wake us up.
+// The timeout value in this call is calculated as (10 ms - laptime)
+// This would make ndb use less cpu while improving response time.
+//--------------------------------------------------------------------
+void ThreadConfig::ipControlLoop()
+{
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+//--------------------------------------------------------------------
+// To let the Cello Watchdog do it's work NDB must sleep a short
+// period every 10 minutes. If this is not done, the watchdog will
+// reboot the board NDB is running on when the load is high.
+//--------------------------------------------------------------------
+ int loopCounter = 0;
+#endif
+
+//--------------------------------------------------------------------
+// initialise the counter that keeps track of the current millisecond
+//--------------------------------------------------------------------
+ globalData.internalMillisecCounter = NdbTick_CurrentMillisecond();
+ Uint32 i = 0;
+ while (globalData.theRestartFlag != perform_stop) {
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ loopCounter++;
+ if(loopCounter > 1000){
+//--------------------------------------------------------------------
+// This is done to allow OSE do a context switch to let the watchdog
+// do it's stuff.
+//--------------------------------------------------------------------
+ NdbSleep_MilliSleep(1);
+ loopCounter = 0;
+ }
+#endif
+
+ Uint32 timeOutMillis = 0;
+ if (LEVEL_IDLE == globalData.highestAvailablePrio) {
+//--------------------------------------------------------------------
+// The buffers are empty, we need to wait for a while until we continue.
+// We cannot wait forever since we can also have timed events.
+//--------------------------------------------------------------------
+//--------------------------------------------------------------------
+// Set the time we will sleep on the sockets before waking up
+// unconditionally to 10 ms. Will never sleep more than 10 milliseconds
+// on a socket.
+//--------------------------------------------------------------------
+ timeOutMillis = 10;
+ }//if
+//--------------------------------------------------------------------
+// Now it is time to check all interfaces. We will send all buffers
+// plus checking for any received messages.
+//--------------------------------------------------------------------
+ if (i++ >= 20) {
+ globalData.incrementWatchDogCounter(5);
+ globalTransporterRegistry.checkConnections();
+ i = 0;
+ }//if
+
+ globalData.incrementWatchDogCounter(6);
+ globalTransporterRegistry.performSend();
+
+ globalData.incrementWatchDogCounter(7);
+ if (globalTransporterRegistry.pollReceive(timeOutMillis)) {
+ globalData.incrementWatchDogCounter(8);
+ globalTransporterRegistry.performReceive();
+ }
+
+//--------------------------------------------------------------------
+// We scan the time queue to see if there are any timed signals that
+// is now ready to be executed.
+//--------------------------------------------------------------------
+ globalData.incrementWatchDogCounter(2);
+ scanTimeQueue();
+
+//--------------------------------------------------------------------
+// This is where the actual execution of signals occur. We execute
+// until all buffers are empty or until we have executed 2048 signals.
+//--------------------------------------------------------------------
+ globalScheduler.doJob();
+
+ globalScheduler.sendPacked();
+
+ }//while
+
+ globalData.incrementWatchDogCounter(6);
+ globalTransporterRegistry.performSend();
+
+}//ThreadConfig::ipControlLoop()
+
+int
+ThreadConfig::doStart(NodeState::StartLevel startLevel){
+
+ SignalHeader sh;
+ memset(&sh, 0, sizeof(SignalHeader));
+
+ sh.theVerId_signalNumber = GSN_START_ORD;
+ sh.theReceiversBlockNumber = CMVMI;
+ sh.theSendersBlockRef = 0;
+ sh.theTrace = 0;
+ sh.theSignalId = 0;
+ sh.theLength = StartOrd::SignalLength;
+
+ Uint32 theData[25];
+ StartOrd * const startOrd = (StartOrd *)&theData[0];
+ startOrd->restartInfo = 0;
+
+ Uint32 secPtrI[3];
+ globalScheduler.execute(&sh, JBA, theData, secPtrI);
+ return 0;
+}
+
diff --git a/ndb/src/kernel/vm/ThreadConfig.hpp b/ndb/src/kernel/vm/ThreadConfig.hpp
new file mode 100644
index 00000000000..91c2cafe0e0
--- /dev/null
+++ b/ndb/src/kernel/vm/ThreadConfig.hpp
@@ -0,0 +1,39 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ThreadConfig_H
+#define ThreadConfig_H
+
+#include <kernel_types.h>
+#include <ErrorReporter.hpp>
+#include <NodeState.hpp>
+
+class IPCConfig;
+
+class ThreadConfig
+{
+public:
+ ThreadConfig();
+ ~ThreadConfig();
+
+ void ipControlLoop();
+
+ int doStart(NodeState::StartLevel startLevel);
+private:
+
+ void scanTimeQueue();
+};
+#endif // ThreadConfig_H
diff --git a/ndb/src/kernel/vm/TimeQueue.cpp b/ndb/src/kernel/vm/TimeQueue.cpp
new file mode 100644
index 00000000000..56988c2e3da
--- /dev/null
+++ b/ndb/src/kernel/vm/TimeQueue.cpp
@@ -0,0 +1,209 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "TimeQueue.hpp"
+#include <ErrorHandlingMacros.hpp>
+#include <GlobalData.hpp>
+#include <FastScheduler.hpp>
+#include <VMSignal.hpp>
+#include <Error.hpp>
+
+static const int MAX_TIME_QUEUE_VALUE = 32000;
+
+TimeQueue::TimeQueue()
+{
+ clear();
+}
+
+TimeQueue::~TimeQueue()
+{
+}
+
+void
+TimeQueue::clear()
+{
+ globalData.theNextTimerJob = 65535;
+ globalData.theCurrentTimer = 0;
+ globalData.theShortTQIndex = 0;
+ globalData.theLongTQIndex = 0;
+ for (int i = 0; i < MAX_NO_OF_TQ; i++)
+ theFreeIndex[i] = i+1;
+ theFreeIndex[MAX_NO_OF_TQ - 1] = NULL_TQ_ENTRY;
+ globalData.theFirstFreeTQIndex = 0;
+}
+
+void
+TimeQueue::insert(Signal* signal, BlockNumber bnr,
+ GlobalSignalNumber gsn, Uint32 delayTime)
+{
+ if (delayTime == 0)
+ delayTime = 1;
+ register Uint32 regCurrentTime = globalData.theCurrentTimer;
+ register Uint32 i;
+ register Uint32 regSave;
+ register TimerEntry newEntry;
+
+ newEntry.time_struct.delay_time = regCurrentTime + delayTime;
+ newEntry.time_struct.job_index = getIndex();
+ regSave = newEntry.copy_struct;
+
+ globalScheduler.insertTimeQueue(signal, bnr, gsn,
+ newEntry.time_struct.job_index);
+
+ if (newEntry.time_struct.delay_time < globalData.theNextTimerJob)
+ globalData.theNextTimerJob = newEntry.time_struct.delay_time;
+ if (delayTime < 100){
+ register Uint32 regShortIndex = globalData.theShortTQIndex;
+ if (regShortIndex == 0){
+ theShortQueue[0].copy_struct = newEntry.copy_struct;
+ } else if (regShortIndex >= MAX_NO_OF_SHORT_TQ - 1) {
+ ERROR_SET(ecError, ERROR_TIME_QUEUE_SHORT,
+ "Too many in Short Time Queue", "TimeQueue.C" );
+ } else {
+ for (i = 0; i < regShortIndex; i++) {
+ if (theShortQueue[i].time_struct.delay_time >
+ newEntry.time_struct.delay_time) {
+
+ regSave = theShortQueue[i].copy_struct;
+ theShortQueue[i].copy_struct = newEntry.copy_struct;
+ break;
+ }
+ }
+ if (i == regShortIndex) {
+ theShortQueue[regShortIndex].copy_struct = regSave;
+ } else {
+ for (i++; i < regShortIndex; i++) {
+ register Uint32 regTmp = theShortQueue[i].copy_struct;
+ theShortQueue[i].copy_struct = regSave;
+ regSave = regTmp;
+ }
+ theShortQueue[regShortIndex].copy_struct = regSave;
+ }
+ }
+ globalData.theShortTQIndex = regShortIndex + 1;
+ } else if (delayTime <= (unsigned)MAX_TIME_QUEUE_VALUE) {
+ register Uint32 regLongIndex = globalData.theLongTQIndex;
+ if (regLongIndex == 0) {
+ theLongQueue[0].copy_struct = newEntry.copy_struct;
+ } else if (regLongIndex >= MAX_NO_OF_LONG_TQ - 1) {
+ ERROR_SET(ecError, ERROR_TIME_QUEUE_LONG,
+ "Too many in Long Time Queue", "TimeQueue.C" );
+ } else {
+ for (i = 0; i < regLongIndex; i++) {
+ if (theLongQueue[i].time_struct.delay_time >
+ newEntry.time_struct.delay_time) {
+
+ regSave = theLongQueue[i].copy_struct;
+ theLongQueue[i].copy_struct = newEntry.copy_struct;
+ break;
+ }
+ }
+ if (i == regLongIndex) {
+ theLongQueue[regLongIndex].copy_struct = regSave;
+ } else {
+ for (i++; i < regLongIndex; i++) {
+ register Uint32 regTmp = theLongQueue[i].copy_struct;
+ theLongQueue[i].copy_struct = regSave;
+ regSave = regTmp;
+ }
+ theLongQueue[regLongIndex].copy_struct = regSave;
+ }
+ }
+ globalData.theLongTQIndex = regLongIndex + 1;
+ } else {
+ ERROR_SET(ecError, ERROR_TIME_QUEUE_DELAY,
+ "Too long delay for Time Queue", "TimeQueue.C" );
+ }
+}
+
+// executes the expired signals;
+void
+TimeQueue::scanTable()
+{
+ register Uint32 i, j;
+
+ globalData.theCurrentTimer++;
+ if (globalData.theCurrentTimer == 32000)
+ recount_timers();
+ if (globalData.theNextTimerJob > globalData.theCurrentTimer)
+ return;
+ globalData.theNextTimerJob = 65535; // If no more timer jobs
+ for (i = 0; i < globalData.theShortTQIndex; i++) {
+ if (theShortQueue[i].time_struct.delay_time > globalData.theCurrentTimer){
+ break;
+ } else {
+ releaseIndex((Uint32)theShortQueue[i].time_struct.job_index);
+ globalScheduler.scheduleTimeQueue(theShortQueue[i].time_struct.job_index);
+ }
+ }
+ if (i > 0) {
+ for (j = i; j < globalData.theShortTQIndex; j++)
+ theShortQueue[j - i].copy_struct = theShortQueue[j].copy_struct;
+ globalData.theShortTQIndex -= i;
+ }
+ if (globalData.theShortTQIndex != 0) // If not empty
+ globalData.theNextTimerJob = theShortQueue[0].time_struct.delay_time;
+ for (i = 0; i < globalData.theLongTQIndex; i++) {
+ if (theLongQueue[i].time_struct.delay_time > globalData.theCurrentTimer) {
+ break;
+ } else {
+ releaseIndex((Uint32)theLongQueue[i].time_struct.job_index);
+ globalScheduler.scheduleTimeQueue(theLongQueue[i].time_struct.job_index);
+ }
+ }
+ if (i > 0) {
+ for (j = i; j < globalData.theLongTQIndex; j++)
+ theLongQueue[j - i].copy_struct = theLongQueue[j].copy_struct;
+ globalData.theLongTQIndex -= i;
+ }
+ if (globalData.theLongTQIndex != 0) // If not empty
+ if (globalData.theNextTimerJob > theLongQueue[0].time_struct.delay_time)
+ globalData.theNextTimerJob = theLongQueue[0].time_struct.delay_time;
+}
+
+void
+TimeQueue::recount_timers()
+{
+ Uint32 i;
+
+ globalData.theCurrentTimer = 0;
+ globalData.theNextTimerJob -= 32000;
+
+ for (i = 0; i < globalData.theShortTQIndex; i++)
+ theShortQueue[i].time_struct.delay_time -= 32000;
+ for (i = 0; i < globalData.theLongTQIndex; i++)
+ theLongQueue[i].time_struct.delay_time -= 32000;
+}
+
+Uint32
+TimeQueue::getIndex()
+{
+ Uint32 retValue = globalData.theFirstFreeTQIndex;
+ globalData.theFirstFreeTQIndex = (Uint32)theFreeIndex[retValue];
+ if (retValue >= MAX_NO_OF_TQ)
+ ERROR_SET(fatal, ERROR_TIME_QUEUE_INDEX,
+ "Index out of range", "TimeQueue.C" );
+ return retValue;
+}
+
+void
+TimeQueue::releaseIndex(Uint32 aIndex)
+{
+ theFreeIndex[aIndex] = globalData.theFirstFreeTQIndex;
+ globalData.theFirstFreeTQIndex = aIndex;
+}
+
+
diff --git a/ndb/src/kernel/vm/TimeQueue.hpp b/ndb/src/kernel/vm/TimeQueue.hpp
new file mode 100644
index 00000000000..1203ace10f5
--- /dev/null
+++ b/ndb/src/kernel/vm/TimeQueue.hpp
@@ -0,0 +1,62 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TimeQueue_H
+#define TimeQueue_H
+
+#include <kernel_types.h>
+#include "Prio.hpp"
+
+#define MAX_NO_OF_SHORT_TQ 512
+#define MAX_NO_OF_LONG_TQ 512
+#define MAX_NO_OF_TQ (MAX_NO_OF_SHORT_TQ + MAX_NO_OF_LONG_TQ)
+#define NULL_TQ_ENTRY 65535
+
+class Signal;
+
+struct TimeStruct
+{
+ Uint16 delay_time;
+ Uint16 job_index;
+};
+
+union TimerEntry
+{
+ struct TimeStruct time_struct;
+ Uint32 copy_struct;
+};
+
+class TimeQueue
+{
+public:
+ TimeQueue();
+ ~TimeQueue();
+
+ void insert(Signal* signal, BlockNumber bnr,
+ GlobalSignalNumber gsn, Uint32 delayTime);
+ void clear();
+ void scanTable(); // Called once per millisecond
+ Uint32 getIndex();
+ void releaseIndex(Uint32 aIndex);
+ void recount_timers();
+
+private:
+ TimerEntry theShortQueue[MAX_NO_OF_SHORT_TQ];
+ TimerEntry theLongQueue[MAX_NO_OF_LONG_TQ];
+ Uint16 theFreeIndex[MAX_NO_OF_TQ];
+};
+
+#endif
diff --git a/ndb/src/kernel/vm/TransporterCallback.cpp b/ndb/src/kernel/vm/TransporterCallback.cpp
new file mode 100644
index 00000000000..1fec4ea86bd
--- /dev/null
+++ b/ndb/src/kernel/vm/TransporterCallback.cpp
@@ -0,0 +1,430 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <TransporterCallback.hpp>
+#include <TransporterRegistry.hpp>
+#include <FastScheduler.hpp>
+#include <Emulator.hpp>
+#include <ErrorHandlingMacros.hpp>
+#include <stdio.h>
+
+#include "LongSignal.hpp"
+
+#include <signaldata/EventReport.hpp>
+#include <signaldata/TestOrd.hpp>
+#include <signaldata/SignalDroppedRep.hpp>
+#include <signaldata/DisconnectRep.hpp>
+
+#include "VMSignal.hpp"
+#include <NdbOut.hpp>
+#include "DataBuffer.hpp"
+
+/**
+ * The instance
+ */
+SectionSegmentPool g_sectionSegmentPool;
+
+static int f(int v){
+ g_sectionSegmentPool.setSize(v);
+ return v;
+}
+
+static int v = f(2048);
+
+bool
+import(Ptr<SectionSegment> & first, const Uint32 * src, Uint32 len){
+ /**
+ * Dummy data used when setting prev.m_nextSegment for first segment of a
+ * section
+ */
+ Uint32 dummyPrev[4];
+
+ first.p = 0;
+ if(g_sectionSegmentPool.seize(first)){
+ ;
+ } else {
+ return false;
+ }
+
+ first.p->m_sz = len;
+ first.p->m_ownerRef = 0;
+
+ Ptr<SectionSegment> prevPtr = { (SectionSegment *)&dummyPrev[0], 0 };
+ Ptr<SectionSegment> currPtr = first;
+
+ while(len > SectionSegment::DataLength){
+ prevPtr.p->m_nextSegment = currPtr.i;
+ memcpy(&currPtr.p->theData[0], src, 4 * SectionSegment::DataLength);
+ src += SectionSegment::DataLength;
+ len -= SectionSegment::DataLength;
+ prevPtr = currPtr;
+ if(g_sectionSegmentPool.seize(currPtr)){
+ ;
+ } else {
+ first.p->m_lastSegment = prevPtr.i;
+ return false;
+ }
+ }
+
+ first.p->m_lastSegment = currPtr.i;
+ currPtr.p->m_nextSegment = RNIL;
+ memcpy(&currPtr.p->theData[0], src, 4 * len);
+ return true;
+}
+
+void
+linkSegments(Uint32 head, Uint32 tail){
+
+ Ptr<SectionSegment> headPtr;
+ g_sectionSegmentPool.getPtr(headPtr, head);
+
+ Ptr<SectionSegment> tailPtr;
+ g_sectionSegmentPool.getPtr(tailPtr, tail);
+
+ Ptr<SectionSegment> oldTailPtr;
+ g_sectionSegmentPool.getPtr(oldTailPtr, headPtr.p->m_lastSegment);
+
+ headPtr.p->m_lastSegment = tailPtr.p->m_lastSegment;
+ headPtr.p->m_sz += tailPtr.p->m_sz;
+
+ oldTailPtr.p->m_nextSegment = tailPtr.i;
+}
+
+void
+copy(Uint32 * & insertPtr,
+ class SectionSegmentPool & thePool, const SegmentedSectionPtr & _ptr){
+
+ Uint32 len = _ptr.sz;
+ SectionSegment * ptrP = _ptr.p;
+
+ while(len > 60){
+ memcpy(insertPtr, &ptrP->theData[0], 4 * 60);
+ len -= 60;
+ insertPtr += 60;
+ ptrP = thePool.getPtr(ptrP->m_nextSegment);
+ }
+ memcpy(insertPtr, &ptrP->theData[0], 4 * len);
+ insertPtr += len;
+}
+
+void
+copy(Uint32 * dst, SegmentedSectionPtr src){
+ copy(dst, g_sectionSegmentPool, src);
+}
+
+void
+getSections(Uint32 secCount, SegmentedSectionPtr ptr[3]){
+ Uint32 tSec0 = ptr[0].i;
+ Uint32 tSec1 = ptr[1].i;
+ Uint32 tSec2 = ptr[2].i;
+ SectionSegment * p;
+ switch(secCount){
+ case 3:
+ p = g_sectionSegmentPool.getPtr(tSec2);
+ ptr[2].p = p;
+ ptr[2].sz = p->m_sz;
+ case 2:
+ p = g_sectionSegmentPool.getPtr(tSec1);
+ ptr[1].p = p;
+ ptr[1].sz = p->m_sz;
+ case 1:
+ p = g_sectionSegmentPool.getPtr(tSec0);
+ ptr[0].p = p;
+ ptr[0].sz = p->m_sz;
+ case 0:
+ return;
+ }
+ char msg[40];
+ sprintf(msg, "secCount=%d", secCount);
+ ErrorReporter::handleAssert(msg, __FILE__, __LINE__);
+}
+
+void
+getSection(SegmentedSectionPtr & ptr, Uint32 i){
+ ptr.i = i;
+ SectionSegment * p = g_sectionSegmentPool.getPtr(i);
+ ptr.p = p;
+ ptr.sz = p->m_sz;
+}
+
+#define relSz(x) ((x + SectionSegment::DataLength - 1) / SectionSegment::DataLength)
+
+void
+release(SegmentedSectionPtr & ptr){
+ g_sectionSegmentPool.releaseList(relSz(ptr.sz),
+ ptr.i,
+ ptr.p->m_lastSegment);
+}
+
+void
+releaseSections(Uint32 secCount, SegmentedSectionPtr ptr[3]){
+ Uint32 tSec0 = ptr[0].i;
+ Uint32 tSz0 = ptr[0].sz;
+ Uint32 tSec1 = ptr[1].i;
+ Uint32 tSz1 = ptr[1].sz;
+ Uint32 tSec2 = ptr[2].i;
+ Uint32 tSz2 = ptr[2].sz;
+ switch(secCount){
+ case 3:
+ g_sectionSegmentPool.releaseList(relSz(tSz2), tSec2,
+ ptr[2].p->m_lastSegment);
+ case 2:
+ g_sectionSegmentPool.releaseList(relSz(tSz1), tSec1,
+ ptr[1].p->m_lastSegment);
+ case 1:
+ g_sectionSegmentPool.releaseList(relSz(tSz0), tSec0,
+ ptr[0].p->m_lastSegment);
+ case 0:
+ return;
+ }
+ char msg[40];
+ sprintf(msg, "secCount=%d", secCount);
+ ErrorReporter::handleAssert(msg, __FILE__, __LINE__);
+}
+
+#include <DebuggerNames.hpp>
+
+void
+execute(void * callbackObj,
+ SignalHeader * const header,
+ Uint8 prio,
+ Uint32 * const theData,
+ LinearSectionPtr ptr[3]){
+
+ const Uint32 secCount = header->m_noOfSections;
+
+#ifdef TRACE_DISTRIBUTED
+ ndbout_c("recv: %s(%d) from (%s, %d)",
+ getSignalName(header->theVerId_signalNumber),
+ header->theVerId_signalNumber,
+ getBlockName(refToBlock(header->theSendersBlockRef)),
+ refToNode(header->theSendersBlockRef));
+#endif
+
+ bool ok = true;
+ Ptr<SectionSegment> secPtr[3];
+ switch(secCount){
+ case 3:
+ ok &= import(secPtr[2], ptr[2].p, ptr[2].sz);
+ case 2:
+ ok &= import(secPtr[1], ptr[1].p, ptr[1].sz);
+ case 1:
+ ok &= import(secPtr[0], ptr[0].p, ptr[0].sz);
+ }
+
+ Uint32 secPtrI[3];
+ if(ok){
+ /**
+ * Normal path
+ */
+ secPtrI[0] = secPtr[0].i;
+ secPtrI[1] = secPtr[1].i;
+ secPtrI[2] = secPtr[2].i;
+ globalScheduler.execute(header, prio, theData, secPtrI);
+ return;
+ }
+
+ /**
+ * Out of memory
+ */
+ for(Uint32 i = 0; i<secCount; i++){
+ if(secPtr[i].p != 0){
+ g_sectionSegmentPool.releaseList(relSz(ptr[i].sz), secPtr[i].i,
+ secPtr[i].p->m_lastSegment);
+ }
+ }
+ Uint32 gsn = header->theVerId_signalNumber;
+ Uint32 len = header->theLength;
+ Uint32 newLen= (len > 22 ? 22 : len);
+ SignalDroppedRep * rep = (SignalDroppedRep*)theData;
+ memmove(rep->originalData, theData, (4 * newLen));
+ rep->originalGsn = gsn;
+ rep->originalLength = len;
+ rep->originalSectionCount = secCount;
+ header->theVerId_signalNumber = GSN_SIGNAL_DROPPED_REP;
+ header->theLength = newLen + 3;
+ header->m_noOfSections = 0;
+ globalScheduler.execute(header, prio, theData, secPtrI);
+}
+
+NdbOut &
+operator<<(NdbOut& out, const SectionSegment & ss){
+ out << "[ last= " << ss.m_lastSegment << " next= " << ss.nextPool << " ]";
+ return out;
+}
+
+void
+print(SectionSegment * s, Uint32 len, FILE* out){
+ for(Uint32 i = 0; i<len; i++){
+ fprintf(out, "H\'0x%.8x ", s->theData[i]);
+ if(((i + 1) % 6) == 0)
+ fprintf(out, "\n");
+ }
+}
+
+void
+print(SegmentedSectionPtr ptr, FILE* out){
+
+ ptr.p = g_sectionSegmentPool.getPtr(ptr.i);
+ Uint32 len = ptr.p->m_sz;
+
+ fprintf(out, "ptr.i = %d(%p) ptr.sz = %d(%d)\n", ptr.i, ptr.p, len, ptr.sz);
+ while(len > SectionSegment::DataLength){
+ print(ptr.p, SectionSegment::DataLength, out);
+
+ len -= SectionSegment::DataLength;
+ fprintf(out, "ptr.i = %d\n", ptr.p->m_nextSegment);
+ ptr.p = g_sectionSegmentPool.getPtr(ptr.p->m_nextSegment);
+ }
+
+ print(ptr.p, len, out);
+ fprintf(out, "\n");
+}
+
+int
+checkJobBuffer() {
+ /**
+ * Check to see if jobbbuffers are starting to get full
+ * and if so call doJob
+ */
+ return globalScheduler.checkDoJob();
+}
+
+void
+reportError(void * callbackObj, NodeId nodeId, TransporterError errorCode){
+#ifdef DEBUG_TRANSPORTER
+ char buf[255];
+ sprintf(buf, "reportError (%d, 0x%x)", nodeId, errorCode);
+ ndbout << buf << endl;
+#endif
+
+ if(errorCode == TE_SIGNAL_LOST_SEND_BUFFER_FULL){
+ ErrorReporter::handleError(ecError,
+ ERR_PROGRAMERROR,
+ "Signal lost, send buffer full",
+ __FILE__,
+ NST_ErrorHandler);
+ }
+
+ if(errorCode == TE_SIGNAL_LOST){
+ ErrorReporter::handleError(ecError,
+ ERR_PROGRAMERROR,
+ "Signal lost (unknown reason)",
+ __FILE__,
+ NST_ErrorHandler);
+ }
+
+ if(errorCode & 0x8000){
+ reportDisconnect(callbackObj, nodeId, errorCode);
+ }
+
+ Signal signal;
+ memset(&signal.header, 0, sizeof(signal.header));
+
+
+ if(errorCode & 0x8000)
+ signal.theData[0] = EventReport::TransporterError;
+ else
+ signal.theData[0] = EventReport::TransporterWarning;
+
+ signal.theData[1] = nodeId;
+ signal.theData[2] = errorCode;
+
+ signal.header.theLength = 3;
+ signal.header.theSendersSignalId = 0;
+ signal.header.theSendersBlockRef = numberToRef(0, globalData.ownId);
+ globalScheduler.execute(&signal, JBA, CMVMI, GSN_EVENT_REP);
+}
+
+/**
+ * Report average send length in bytes (4096 last sends)
+ */
+void
+reportSendLen(void * callbackObj,
+ NodeId nodeId, Uint32 count, Uint64 bytes){
+
+ Signal signal;
+ memset(&signal.header, 0, sizeof(signal.header));
+
+ signal.header.theLength = 3;
+ signal.header.theSendersSignalId = 0;
+ signal.header.theSendersBlockRef = numberToRef(0, globalData.ownId);
+ signal.theData[0] = EventReport::SendBytesStatistic;
+ signal.theData[1] = nodeId;
+ signal.theData[2] = (bytes/count);
+ globalScheduler.execute(&signal, JBA, CMVMI, GSN_EVENT_REP);
+}
+
+/**
+ * Report average receive length in bytes (4096 last receives)
+ */
+void
+reportReceiveLen(void * callbackObj,
+ NodeId nodeId, Uint32 count, Uint64 bytes){
+
+ Signal signal;
+ memset(&signal.header, 0, sizeof(signal.header));
+
+ signal.header.theLength = 3;
+ signal.header.theSendersSignalId = 0;
+ signal.header.theSendersBlockRef = numberToRef(0, globalData.ownId);
+ signal.theData[0] = EventReport::ReceiveBytesStatistic;
+ signal.theData[1] = nodeId;
+ signal.theData[2] = (bytes/count);
+ globalScheduler.execute(&signal, JBA, CMVMI, GSN_EVENT_REP);
+}
+
+/**
+ * Report connection established
+ */
+
+void
+reportConnect(void * callbackObj, NodeId nodeId){
+
+ Signal signal;
+ memset(&signal.header, 0, sizeof(signal.header));
+
+ signal.header.theLength = 1;
+ signal.header.theSendersSignalId = 0;
+ signal.header.theSendersBlockRef = numberToRef(0, globalData.ownId);
+ signal.theData[0] = nodeId;
+
+ globalScheduler.execute(&signal, JBA, CMVMI, GSN_CONNECT_REP);
+}
+
+/**
+ * Report connection broken
+ */
+void
+reportDisconnect(void * callbackObj, NodeId nodeId, Uint32 errNo){
+
+ Signal signal;
+ memset(&signal.header, 0, sizeof(signal.header));
+
+ signal.header.theLength = DisconnectRep::SignalLength;
+ signal.header.theSendersSignalId = 0;
+ signal.header.theSendersBlockRef = numberToRef(0, globalData.ownId);
+ signal.header.theTrace = TestOrd::TraceDisconnect;
+
+ DisconnectRep * const rep = (DisconnectRep *)&signal.theData[0];
+ rep->nodeId = nodeId;
+ rep->err = errNo;
+
+ globalScheduler.execute(&signal, JBA, CMVMI, GSN_DISCONNECT_REP);
+}
+
+
+
diff --git a/ndb/src/kernel/vm/VMSignal.cpp b/ndb/src/kernel/vm/VMSignal.cpp
new file mode 100644
index 00000000000..bffca6f6541
--- /dev/null
+++ b/ndb/src/kernel/vm/VMSignal.cpp
@@ -0,0 +1,34 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "VMSignal.hpp"
+#include <string.h>
+
+Signal::Signal(){
+ memset(&header, sizeof(header), 0);
+ memset(theData, sizeof(theData), 0);
+}
+
+void
+Signal::garbage_register()
+{
+ int i;
+ theData[0] = 0x13579135;
+ header.theLength = 0x13579135;
+ header.theSendersBlockRef = 0x13579135;
+ for (i = 1; i < 24; i++)
+ theData[i] = 0x13579135;
+}
diff --git a/ndb/src/kernel/vm/VMSignal.hpp b/ndb/src/kernel/vm/VMSignal.hpp
new file mode 100644
index 00000000000..d436143c055
--- /dev/null
+++ b/ndb/src/kernel/vm/VMSignal.hpp
@@ -0,0 +1,182 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef VMSignal_H
+#define VMSignal_H
+
+#include <ErrorReporter.hpp>
+#include <NodeBitmask.hpp>
+
+#include <ndb_limits.h>
+#include <kernel_types.h>
+#include <RefConvert.hpp>
+#include <TransporterDefinitions.hpp>
+#include <stdlib.h>
+
+/**
+ * Struct used when sending to multiple blocks
+ */
+struct NodeReceiverGroup {
+ NodeReceiverGroup();
+ NodeReceiverGroup(Uint32 blockRef);
+ NodeReceiverGroup(Uint32 blockNo, const NodeBitmask &);
+
+ NodeReceiverGroup& operator=(BlockReference ref);
+
+ Uint32 m_block;
+ NodeBitmask m_nodes;
+};
+
+/**
+ * class used for passing argumentes to blocks
+ */
+class Signal {
+ friend class SimulatedBlock;
+ friend class APZJobBuffer;
+ friend class FastScheduler;
+public:
+ Signal();
+
+ Uint32 getLength() const;
+ Uint32 getTrace() const;
+ Uint32 getSendersBlockRef() const;
+
+ const Uint32* getDataPtr() const ;
+ Uint32* getDataPtrSend() ;
+
+ void setTrace(Uint32);
+
+ Uint32 getNoOfSections() const;
+ bool getSection(SegmentedSectionPtr & ptr, Uint32 sectionNo);
+ void setSection(SegmentedSectionPtr ptr, Uint32 sectionNo);
+
+ /**
+ * Old depricated methods...
+ */
+ Uint32 length() const { return getLength();}
+ BlockReference senderBlockRef() const { return getSendersBlockRef();}
+
+private:
+ void setLength(Uint32);
+
+public:
+#define VMS_DATA_SIZE \
+ (MAX_ATTRIBUTES_IN_TABLE + MAX_TUPLE_SIZE_IN_WORDS + MAX_KEY_SIZE_IN_WORDS)
+
+ SignalHeader header; // 28 bytes
+ SegmentedSectionPtr m_sectionPtr[3];
+ Uint32 theData[25+VMS_DATA_SIZE]; // 2048 32-bit words -> 8K Bytes
+
+ void garbage_register();
+};
+
+inline
+Uint32
+Signal::getLength() const {
+ return header.theLength;
+}
+
+inline
+Uint32
+Signal::getTrace() const {
+ return header.theTrace;
+}
+
+inline
+Uint32
+Signal::getSendersBlockRef() const {
+ return header.theSendersBlockRef;
+}
+
+inline
+const Uint32*
+Signal::getDataPtr() const {
+ return &theData[0];
+}
+
+inline
+Uint32*
+Signal::getDataPtrSend() {
+ return &theData[0];
+}
+
+inline
+void
+Signal::setLength(Uint32 len){
+ header.theLength = len;
+}
+
+inline
+void
+Signal::setTrace(Uint32 t){
+ header.theTrace = t;
+}
+
+inline
+Uint32
+Signal::getNoOfSections() const {
+ return header.m_noOfSections;
+}
+
+inline
+bool
+Signal::getSection(SegmentedSectionPtr & ptr, Uint32 section){
+ if(section < header.m_noOfSections){
+ ptr = m_sectionPtr[section];
+ return true;
+ }
+ ptr.p = 0;
+ return false;
+}
+
+inline
+void
+Signal::setSection(SegmentedSectionPtr ptr, Uint32 sectionNo){
+ if(sectionNo != header.m_noOfSections || sectionNo > 2){
+ abort();
+ }
+ m_sectionPtr[sectionNo] = ptr;
+ header.m_noOfSections++;
+}
+
+inline
+NodeReceiverGroup::NodeReceiverGroup() : m_block(0){
+ m_nodes.clear();
+}
+
+inline
+NodeReceiverGroup::NodeReceiverGroup(Uint32 blockRef){
+ m_nodes.clear();
+ m_block = refToBlock(blockRef);
+ m_nodes.set(refToNode(blockRef));
+}
+
+inline
+NodeReceiverGroup::NodeReceiverGroup(Uint32 blockNo, const NodeBitmask & nodes){
+ m_block = blockNo;
+ m_nodes = nodes;
+}
+
+inline
+NodeReceiverGroup&
+NodeReceiverGroup::operator=(BlockReference blockRef){
+ m_nodes.clear();
+ m_block = refToBlock(blockRef);
+ m_nodes.set(refToNode(blockRef));
+ return * this;
+}
+
+#endif
diff --git a/ndb/src/kernel/vm/WaitQueue.hpp b/ndb/src/kernel/vm/WaitQueue.hpp
new file mode 100644
index 00000000000..4d7240b6866
--- /dev/null
+++ b/ndb/src/kernel/vm/WaitQueue.hpp
@@ -0,0 +1,35 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef WAIT_QUEUE_HPP
+#define WAIT_QUEUE_HPP
+
+#include "ArrayPool.hpp"
+#include "SimulatedBlock.hpp"
+
+template <class Block,
+ class T,
+ void (Block::* Function)(Signal*, Ptr<T>)>
+class WaitQueue {
+public:
+ WaitQueue(Block & block, ArrayPool<T>& pool){
+ }
+
+ void add(Ptr<T>, void (Block::* Callback)(Signal*, Ptr<T>)) {}
+ void complete(Ptr<T>) {}
+};
+
+#endif
diff --git a/ndb/src/kernel/vm/WatchDog.cpp b/ndb/src/kernel/vm/WatchDog.cpp
new file mode 100644
index 00000000000..a90f63aff37
--- /dev/null
+++ b/ndb/src/kernel/vm/WatchDog.cpp
@@ -0,0 +1,143 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "WatchDog.hpp"
+#include "GlobalData.hpp"
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <ErrorHandlingMacros.hpp>
+
+extern "C"
+void*
+runWatchDog(void* w){
+ ((WatchDog*)w)->run();
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+WatchDog::WatchDog(Uint32 interval) :
+ theIPValue(globalData.getWatchDogPtr())
+{
+ setCheckInterval(interval);
+ theStop = false;
+ theThreadPtr = 0;
+}
+
+WatchDog::~WatchDog(){
+ doStop();
+}
+
+Uint32
+WatchDog::setCheckInterval(Uint32 interval){
+ // An interval of less than 70ms is not acceptable
+ return theInterval = (interval < 70 ? 70 : interval);
+}
+
+void
+WatchDog::doStart(){
+ theStop = false;
+ theThreadPtr = NdbThread_Create(runWatchDog,
+ (void**)this,
+ 32768,
+ "ndb_watchdog",
+ NDB_THREAD_PRIO_HIGH);
+}
+
+void
+WatchDog::doStop(){
+ void *status;
+ theStop = true;
+ if(theThreadPtr){
+ NdbThread_WaitFor(theThreadPtr, &status);
+ NdbThread_Destroy(&theThreadPtr);
+ }
+}
+
+void
+WatchDog::run(){
+ unsigned int anIPValue;
+ unsigned int alerts = 0;
+ unsigned int oldIPValue = 0;
+
+ // WatchDog for the single threaded NDB
+ while(!theStop){
+ Uint32 tmp = theInterval / 500;
+ tmp= (tmp ? tmp : 1);
+
+ while(!theStop && tmp > 0){
+ NdbSleep_MilliSleep(500);
+ tmp--;
+ }
+
+ if(theStop)
+ break;
+
+ // Verify that the IP thread is not stuck in a loop
+ anIPValue = *theIPValue;
+ if(anIPValue != 0) {
+ oldIPValue = anIPValue;
+ globalData.incrementWatchDogCounter(0);
+ alerts = 0;
+ } else {
+ alerts++;
+ ndbout << "Ndb kernel is stuck in: ";
+ switch (oldIPValue) {
+ case 1:
+ ndbout << "Job Handling" << endl;
+ break;
+ case 2:
+ ndbout << "Scanning Timers" << endl;
+ break;
+ case 3:
+ ndbout << "External I/O" << endl;
+ break;
+ case 4:
+ ndbout << "Print Job Buffers at crash" << endl;
+ break;
+ case 5:
+ ndbout << "Checking connections" << endl;
+ break;
+ case 6:
+ ndbout << "Performing Send" << endl;
+ break;
+ case 7:
+ ndbout << "Polling for Receive" << endl;
+ break;
+ case 8:
+ ndbout << "Performing Receive" << endl;
+ break;
+ default:
+ ndbout << "Unknown place" << endl;
+ break;
+ }//switch
+ if(alerts == 3){
+ shutdownSystem();
+ }
+ }
+ }
+ return;
+}
+
+void
+WatchDog::shutdownSystem(){
+
+ ErrorReporter::handleError(ecError,
+ ERR_PROGRAMERROR,
+ "WatchDog terminate",
+ __FILE__,
+ NST_Watchdog);
+}
diff --git a/ndb/src/kernel/vm/WatchDog.hpp b/ndb/src/kernel/vm/WatchDog.hpp
new file mode 100644
index 00000000000..4b44b1a96a2
--- /dev/null
+++ b/ndb/src/kernel/vm/WatchDog.hpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef WatchDog_H
+#define WatchDog_H
+
+#include <kernel_types.h>
+#include <NdbThread.h>
+
+extern "C" void* runWatchDog(void* w);
+
+class WatchDog{
+public:
+ WatchDog(Uint32 interval = 3000);
+ ~WatchDog();
+
+ void doStart();
+ void doStop();
+
+ Uint32 setCheckInterval(Uint32 interval);
+
+protected:
+ /**
+ * Thread function
+ */
+ friend void* runWatchDog(void* w);
+
+ /**
+ * Thread pointer
+ */
+ NdbThread* theThreadPtr;
+
+private:
+ Uint32 theInterval;
+ const Uint32 * theIPValue;
+
+ bool theStop;
+
+ void run();
+ void shutdownSystem();
+};
+
+#endif // WatchDog_H
diff --git a/ndb/src/kernel/vm/al_test/Makefile b/ndb/src/kernel/vm/al_test/Makefile
new file mode 100644
index 00000000000..a7287a341fd
--- /dev/null
+++ b/ndb/src/kernel/vm/al_test/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := kernel
+
+BIN_TARGET := al_test
+BIN_TARGET_ARCHIVES := portlib
+
+SOURCES = main.cpp
+
+CFLAGS_main.cpp = -DDEBUG
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/vm/al_test/arrayListTest.cpp b/ndb/src/kernel/vm/al_test/arrayListTest.cpp
new file mode 100644
index 00000000000..39d8170cfc5
--- /dev/null
+++ b/ndb/src/kernel/vm/al_test/arrayListTest.cpp
@@ -0,0 +1,317 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <ArrayList.hpp>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+#include <NdbTick.h>
+#include <assert.h>
+
+struct A_Listable_Object {
+ Uint32 next;
+ Uint32 prev;
+ char somedata[12];
+
+ void print (NdbOut & out) {
+ out << "ALO: next = " << next
+ << " prev = " << prev << endl;
+ }
+};
+
+extern const int x_AL_Next = offsetof(A_Listable_Object, next);
+extern const int x_AL_Prev = offsetof(A_Listable_Object, prev);
+
+NdbOut &
+operator<<(NdbOut & o, A_Listable_Object & a){
+ a.print(o);
+ return o;
+}
+
+typedef Ptr<A_Listable_Object> A_Listable_ObjectPtr;
+
+#define APool ArrayPool<A_Listable_Object>
+#define AList ArrayList<A_Listable_Object>
+
+APool aGPool;
+AList aGList(aGPool);
+
+class ArrayListTest {
+public:
+ static void tryList0(int listSize){
+ APool aPool;
+ AList aList(aPool);
+
+ if(!aPool.setSize(listSize)){
+ ndbout << "Failed to do aPool.setSize(" << listSize << ")" << endl;
+ return;
+ }
+
+ int * anArray = new int[listSize];
+
+ for(int i = 1; i<listSize; i++){
+ int arrayElements = 0;
+
+
+ for(int j = 0; j<i; j++){
+ A_Listable_ObjectPtr p;
+ const int ret = aList.seize(p);
+ if(ret == RNIL){
+ ndbout << "Failed to seize!!" << endl;
+ ndbout << "Have seized " << j
+ << " out of " << listSize << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ anArray[arrayElements] = ret;
+ arrayElements++;
+ }
+ assert(aList.noOfElements() == i);
+ assert(aPool.noOfFree() == (listSize - i));
+ assert(arrayElements == i);
+
+ for(int j = 0; j<i; j++){
+ aList.release(anArray[j]);
+ }
+
+ assert(aList.noOfElements() == 0);
+ assert(aPool.noOfFree() == listSize);
+ }
+ }
+
+ static void tryList1(int listSize, int iterations){
+ APool aPool;
+ AList aList(aPool);
+
+ if(!aPool.setSize(listSize)){
+ ndbout << "Failed to do aPool.setSize(" << listSize << ")" << endl;
+ return;
+ }
+
+ ndbout << "Seizing/Releaseing " << iterations
+ << " times over list with " << listSize << " elements" << endl;
+
+ int * anArray = new int[listSize];
+ int arrayElements = 0;
+
+ int noOfSeize = 0;
+ int noFailSeize = 0;
+ int noOfRelease = 0;
+
+ for(int i = 0; i<iterations; i++){
+ assert(arrayElements <= listSize);
+ const int r = rand() % (10 * listSize);
+ if(r < (arrayElements - 1)){
+ /**
+ * Release an element
+ */
+ noOfRelease++;
+ aList.release(anArray[r]);
+ arrayElements--;
+ for(int j = r; j<arrayElements; j++)
+ anArray[j] = anArray[j+1];
+
+ } else {
+ /**
+ * Seize an element
+ */
+ A_Listable_ObjectPtr p;
+ const int ret = aList.seize(p);
+ if(ret == RNIL && arrayElements != listSize){
+ ndbout << "Failed to seize!!"
+ << " iteration=" << i << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << listSize << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(arrayElements >= listSize && ret != RNIL){
+ ndbout << "Seize did not fail when it should have"
+ << " iteration=" << i << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << listSize << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(ret != RNIL){
+ noOfSeize++;
+ anArray[arrayElements] = ret;
+ arrayElements++;
+ } else {
+ noFailSeize++;
+ }
+ }
+ }
+ delete []anArray;
+
+ ndbout << "Seized: " << noOfSeize
+ << " Seized with buffer full: " << noFailSeize
+ << " Release: " << noOfRelease << " --- ";
+ ndbout << "(" << noOfSeize << " + " << noFailSeize << " + " << noOfRelease
+ << " = " << (noOfSeize + noFailSeize + noOfRelease) << ")" << endl;
+ }
+
+ static void tryList2(int size, int iter, int fail){
+ APool aPool;
+ AList aList(aPool);
+
+ if(!aPool.setSize(size)){
+ ndbout << "Failed to do aPool.setSize(" << size << ")" << endl;
+ return;
+ }
+
+ ndbout << "doing getPtr(i) where i > size(" << size << ") "
+ << fail << " times mixed with " << iter
+ << " ordinary seize/release" << endl;
+
+ int * anArray = new int[size];
+ int arrayElements = 0;
+
+ int noOfSeize = 0;
+ int noFailSeize = 0;
+ int noOfRelease = 0;
+
+ for(int i = 0; i<iter; i++){
+ assert(arrayElements <= size);
+ const int r = rand() % (10 * size);
+
+ if((i + 1)%(iter/fail) == 0){
+ aList.getPtr(size + r);
+ continue;
+ }
+
+ if(r < (arrayElements - 1)){
+ /**
+ * Release an element
+ */
+ noOfRelease++;
+ aList.release(anArray[r]);
+ arrayElements--;
+ for(int j = r; j<arrayElements; j++)
+ anArray[j] = anArray[j+1];
+
+ } else {
+ /**
+ * Seize an element
+ */
+ A_Listable_ObjectPtr p;
+ const int ret = aList.seize(p);
+ if(ret == RNIL && arrayElements != size){
+ ndbout << "Failed to seize!!"
+ << " iteration=" << i << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << size << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(arrayElements >= size && ret != RNIL){
+ ndbout << "Seize did not fail when it should have"
+ << " iteration=" << i << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << size << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(ret != RNIL){
+ noOfSeize++;
+ anArray[arrayElements] = ret;
+ arrayElements++;
+ } else {
+ noFailSeize++;
+ }
+ }
+ }
+ delete []anArray;
+ }
+
+ static void
+ tryList3(int size, int fail){
+ ndbout << "Failing " << fail << " times " << endl;
+
+ for(int i = 0; i<fail; i++){
+ APool aPool;
+ AList aList(aPool);
+
+ if(!aPool.setSize(size)){
+ ndbout << "Failed to do aPool.setSize(" << size << ")" << endl;
+ return;
+ }
+
+ const int noOfElementsInBufferWhenFail = (i + 1) * (size /(fail + 1));
+
+ int * anArray = new int[size];
+ for(int i = 0; i<size; i++)
+ anArray[i] = i;
+ int arrayElements = 0;
+
+ int noOfSeize = 0;
+ int noFailSeize = 0;
+ int noOfRelease = 0;
+
+ while(true){
+ assert(arrayElements <= size);
+ if(arrayElements == noOfElementsInBufferWhenFail){
+ ndbout << "++ You should get a ErrorReporter::handle... " << endl;
+ aList.release(anArray[arrayElements]);
+ ndbout << "++ Inbetween these lines" << endl << endl;
+ break;
+ }
+ const int r = rand() % (10 * size);
+ if(r < (arrayElements - 1)){
+ /**
+ * Release an element
+ */
+ noOfRelease++;
+ aList.release(anArray[r]);
+ arrayElements--;
+ for(int j = r; j<arrayElements; j++)
+ anArray[j] = anArray[j+1];
+
+ } else {
+ /**
+ * Seize an element
+ */
+ A_Listable_ObjectPtr p;
+ const int ret = aList.seize(p);
+ if(ret == RNIL && arrayElements != size){
+ ndbout << "Failed to seize!!" << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << size << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(arrayElements >= size && ret != RNIL){
+ ndbout << "Seize did not fail when it should have" << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << size << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(ret != RNIL){
+ noOfSeize++;
+ anArray[arrayElements] = ret;
+ arrayElements++;
+ } else {
+ noFailSeize++;
+ }
+ }
+ }
+ delete []anArray;
+ }
+
+ }
+};
diff --git a/ndb/src/kernel/vm/al_test/arrayPoolTest.cpp b/ndb/src/kernel/vm/al_test/arrayPoolTest.cpp
new file mode 100644
index 00000000000..8b554d5bb41
--- /dev/null
+++ b/ndb/src/kernel/vm/al_test/arrayPoolTest.cpp
@@ -0,0 +1,299 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+#include <ArrayList.hpp>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+#include <NdbTick.h>
+#include <assert.h>
+#include <string.h>
+
+struct A_Poolable_Object {
+ Uint32 next;
+ char somedata[12];
+
+ void print (NdbOut & out) {
+ out << "A_Poolable_Object: next = " << next << endl;
+ }
+
+};
+
+
+NdbOut &
+operator<<(NdbOut & o, A_Poolable_Object & a){
+ a.print(o);
+ return o;
+}
+
+typedef Ptr<A_Poolable_Object> A_Poolable_ObjectPtr;
+#if 1
+#define BPool ArrayPool<A_Poolable_Object>
+#else
+#define BPool ArrayPool(A_Poolable_Object, next)
+#endif
+
+class ArrayPoolTest {
+public:
+ static void tryPool1(int poolSize, int iterations){
+ BPool aPool;
+
+ if(!aPool.setSize(poolSize)){
+ ndbout << "Failed to do aPool.setSize(" << poolSize << ")" << endl;
+ return;
+ }
+
+ ndbout << "Seizing/Releaseing " << iterations
+ << " times over pool with " << poolSize << " elements" << endl;
+
+ int * anArray = new int[poolSize];
+ int arrayElements = 0;
+
+ int noOfSeize = 0;
+ int noFailSeize = 0;
+ int noOfRelease = 0;
+
+ for(int i = 0; i<iterations; i++){
+ if(!((arrayElements <= poolSize) &&
+ (aPool.noOfFree() == aPool.noOfFree2()) &&
+ (aPool.noOfFree() == (poolSize - arrayElements)))){
+ ndbout << "Assertion!!"
+ << " iteration=" << i << endl;
+ const int f1 = aPool.noOfFree();
+ const int f2 = aPool.noOfFree2();
+ ndbout << "noOfFree() = " << f1 << endl;
+ ndbout << "noOfFree2() = " << f2 << endl;
+ ndbout << "poolSize = " << poolSize << endl;
+ ndbout << "arrayElemts = " << arrayElements << endl;
+ aPool.print(ndbout);
+ assert(0);
+ }
+
+ const int r = rand() % (10 * poolSize);
+ if(r < (arrayElements - 1)){
+ /**
+ * Release an element
+ */
+ noOfRelease++;
+ aPool.release(anArray[r]);
+ arrayElements--;
+ for(int j = r; j<arrayElements; j++)
+ anArray[j] = anArray[j+1];
+
+ } else {
+ /**
+ * Seize an element
+ */
+ A_Poolable_ObjectPtr p;
+ const int ret = aPool.seize(p);
+ if(ret == RNIL && arrayElements != poolSize){
+ ndbout << "Failed to seize!!"
+ << " iteration=" << i << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << poolSize << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(arrayElements >= poolSize && ret != RNIL){
+ ndbout << "Seize did not fail when it should have"
+ << " iteration=" << i << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << poolSize << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(ret != RNIL){
+ noOfSeize++;
+ anArray[arrayElements] = ret;
+ arrayElements++;
+ memset(p.p, i, sizeof(p.p->somedata));
+ } else {
+ noFailSeize++;
+ }
+ }
+ }
+ delete []anArray;
+
+ ndbout << "Seized: " << noOfSeize
+ << " Seized with buffer full: " << noFailSeize
+ << " Release: " << noOfRelease << " --- ";
+ ndbout << "(" << noOfSeize << " + " << noFailSeize << " + " << noOfRelease
+ << " = " << (noOfSeize + noFailSeize + noOfRelease) << ")" << endl;
+ }
+
+ static void tryPool2(int size, int iter, int fail){
+ BPool aPool;
+
+ if(!aPool.setSize(size)){
+ ndbout << "Failed to do aPool.setSize(" << size << ")" << endl;
+ return;
+ }
+
+ ndbout << "doing getPtr(i) where i > size(" << size << ") "
+ << fail << " times mixed with " << iter
+ << " ordinary seize/release" << endl;
+
+ int * anArray = new int[size];
+ int arrayElements = 0;
+
+ int noOfSeize = 0;
+ int noFailSeize = 0;
+ int noOfRelease = 0;
+
+ for(int i = 0; i<iter; i++){
+ if(!((arrayElements <= size) &&
+ (aPool.noOfFree() == aPool.noOfFree2()) &&
+ (aPool.noOfFree() == (size - arrayElements)))){
+ ndbout << "Assertion!!"
+ << " iteration=" << i << endl;
+ const int f1 = aPool.noOfFree();
+ const int f2 = aPool.noOfFree2();
+ ndbout << "noOfFree() = " << f1 << endl;
+ ndbout << "noOfFree2() = " << f2 << endl;
+ ndbout << "poolSize = " << size << endl;
+ ndbout << "arrayElemts = " << arrayElements << endl;
+ aPool.print(ndbout);
+ assert(0);
+ }
+ const int r = rand() % (10 * size);
+
+ if((i + 1)%(iter/fail) == 0){
+ aPool.getPtr(size + r);
+ continue;
+ }
+
+ if(r < (arrayElements - 1)){
+ /**
+ * Release an element
+ */
+ noOfRelease++;
+ aPool.release(anArray[r]);
+ arrayElements--;
+ for(int j = r; j<arrayElements; j++)
+ anArray[j] = anArray[j+1];
+
+ } else {
+ /**
+ * Seize an element
+ */
+ A_Poolable_ObjectPtr p;
+ const int ret = aPool.seize(p);
+ if(ret == RNIL && arrayElements != size){
+ ndbout << "Failed to seize!!"
+ << " iteration=" << i << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << size << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(arrayElements >= size && ret != RNIL){
+ ndbout << "Seize did not fail when it should have"
+ << " iteration=" << i << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << size << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(ret != RNIL){
+ noOfSeize++;
+ anArray[arrayElements] = ret;
+ arrayElements++;
+ memset(p.p, p.i, sizeof(p.p->somedata));
+ } else {
+ noFailSeize++;
+ }
+ }
+ }
+ delete []anArray;
+ }
+
+ static void
+ tryPool3(int size, int fail){
+ ndbout << "Failing " << fail << " times " << endl;
+
+ for(int i = 0; i<fail; i++){
+ BPool aPool;
+ if(!aPool.setSize(size)){
+ ndbout << "Failed to do aPool.setSize(" << size << ")" << endl;
+ return;
+ }
+
+ const int noOfElementsInBufferWhenFail = (i + 1) * (size /(fail + 1));
+
+ int * anArray = new int[size];
+ for(int i = 0; i<size; i++)
+ anArray[i] = i;
+ int arrayElements = 0;
+
+ int noOfSeize = 0;
+ int noFailSeize = 0;
+ int noOfRelease = 0;
+
+ while(true){
+ assert(arrayElements <= size);
+ if(arrayElements == noOfElementsInBufferWhenFail){
+ ndbout << "++ You should get a ErrorReporter::handle... " << endl;
+ aPool.release(anArray[arrayElements]);
+ ndbout << "++ Inbetween these lines" << endl << endl;
+ break;
+ }
+ const int r = rand() % (10 * size);
+ if(r < (arrayElements - 1)){
+ /**
+ * Release an element
+ */
+ noOfRelease++;
+ aPool.release(anArray[r]);
+ arrayElements--;
+ for(int j = r; j<arrayElements; j++)
+ anArray[j] = anArray[j+1];
+
+ } else {
+ /**
+ * Seize an element
+ */
+ A_Poolable_ObjectPtr p;
+ const int ret = aPool.seize(p);
+ if(ret == RNIL && arrayElements != size){
+ ndbout << "Failed to seize!!" << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << size << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(arrayElements >= size && ret != RNIL){
+ ndbout << "Seize did not fail when it should have" << endl;
+ ndbout << "Have seized " << arrayElements
+ << " out of " << size << endl;
+ ndbout << "Terminating..." << endl;
+ abort();
+ }
+ if(ret != RNIL){
+ noOfSeize++;
+ anArray[arrayElements] = ret;
+ arrayElements++;
+ } else {
+ noFailSeize++;
+ }
+ }
+ }
+ delete []anArray;
+ }
+
+ }
+};
+
diff --git a/ndb/src/kernel/vm/al_test/main.cpp b/ndb/src/kernel/vm/al_test/main.cpp
new file mode 100644
index 00000000000..42c36173b56
--- /dev/null
+++ b/ndb/src/kernel/vm/al_test/main.cpp
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <ArrayList.hpp>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+#include <NdbTick.h>
+#include <assert.h>
+#include <NdbMain.h>
+
+#include "arrayListTest.cpp"
+#include "arrayPoolTest.cpp"
+
+NDB_COMMAND(al_test, "al_test", "al_test", "al_test", 65535)
+{
+ NdbMem_Create();
+ srand(NdbTick_CurrentMillisecond());
+
+#if 1
+ ndbout << endl << endl << "-- Testing basic basic seize/release" << endl;
+ ArrayListTest::tryList0(10);
+
+ ndbout << endl << endl << "-- Testing basic seize/release" << endl;
+ ArrayListTest::tryList1(1000, 1000);
+
+ ndbout << endl << endl << "-- Testing that seize returns RNIL"
+ << endl;
+ ArrayListTest::tryList1(10, 1000000);
+
+ ndbout << endl << endl << "-- Testing access out of array" << endl;
+ ArrayListTest::tryList2(1000, 100000, 5);
+#endif
+
+#if 1
+ ndbout << endl << endl << "-- Testing basic seize/release" << endl;
+ ArrayPoolTest::tryPool1(1000, 1000);
+
+ ndbout << endl << endl << "-- Testing that seize returns RNIL"
+ << endl;
+ ArrayPoolTest::tryPool1(10, 1000000);
+
+ ndbout << endl << endl << "-- Testing access out of array" << endl;
+ ArrayPoolTest::tryPool2(1000, 100000, 5);
+
+ ndbout << endl << endl << "-- Testing releasing none seized element" << endl;
+ ArrayPoolTest::tryPool3(1000, 5);
+#endif
+}
+
+void
+ErrorReporter::handleBlockAssert(int line)
+{
+ ndbout << "ErrorReporter::handleAssert activated - "
+ << " line= " << line << endl;
+ //assert(0);
+}
diff --git a/ndb/src/kernel/vm/pc.hpp b/ndb/src/kernel/vm/pc.hpp
new file mode 100644
index 00000000000..873a986bc35
--- /dev/null
+++ b/ndb/src/kernel/vm/pc.hpp
@@ -0,0 +1,248 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef PC_H
+#define PC_H
+
+
+#include "Emulator.hpp"
+#include <NdbOut.hpp>
+#include <ndb_limits.h>
+
+#ifdef USE_EMULATED_JAM
+
+#ifdef NDB_WIN32
+
+#define jam() { \
+ Uint32 tEmulatedJamIndex = theEmulatedJamIndex; \
+ *(Uint32*)(theEmulatedJam + tEmulatedJamIndex) = __LINE__; \
+ theEmulatedJamIndex = (tEmulatedJamIndex + 4) & JAM_MASK; }
+#define jamLine(line) { \
+ Uint32 tEmulatedJamIndex = theEmulatedJamIndex; \
+ *(Uint32*)(theEmulatedJam + tEmulatedJamIndex) = line; \
+ theEmulatedJamIndex = (tEmulatedJamIndex + 4) & JAM_MASK; }
+#define jamEntry() { \
+ theEmulatedJamBlockNumber = number(); \
+ Uint32 tEmulatedJamIndex = theEmulatedJamIndex; \
+ *(Uint32*)(theEmulatedJam + tEmulatedJamIndex) = \
+ ((theEmulatedJamBlockNumber << 20) | __LINE__); \
+ theEmulatedJamIndex = (tEmulatedJamIndex + 4) & JAM_MASK; }
+#define jamEntryLine(line) { \
+ theEmulatedJamBlockNumber = number(); \
+ Uint32 tEmulatedJamIndex = theEmulatedJamIndex; \
+ *(Uint32*)(theEmulatedJam + tEmulatedJamIndex) = \
+ ((theEmulatedJamBlockNumber << 20) | line); \
+ theEmulatedJamIndex = (tEmulatedJamIndex + 4) & JAM_MASK; }
+
+#else
+
+#define jam() { \
+ Uint32 tEmulatedJamIndex = theEmulatedJamIndex; \
+ *(Uint32*)((UintPtr)theEmulatedJam + (Uint32)tEmulatedJamIndex) = __LINE__; \
+ theEmulatedJamIndex = (tEmulatedJamIndex + 4) & JAM_MASK; }
+#define jamLine(line) { \
+ Uint32 tEmulatedJamIndex = theEmulatedJamIndex; \
+ *(Uint32*)((UintPtr)theEmulatedJam + (Uint32)tEmulatedJamIndex) = line; \
+ theEmulatedJamIndex = (tEmulatedJamIndex + 4) & JAM_MASK; }
+#define jamEntry() { \
+ theEmulatedJamBlockNumber = number(); \
+ Uint32 tEmulatedJamIndex = theEmulatedJamIndex; \
+ *(Uint32*)((UintPtr)theEmulatedJam + (Uint32)tEmulatedJamIndex) = \
+ ((theEmulatedJamBlockNumber << 20) | __LINE__); \
+ theEmulatedJamIndex = (tEmulatedJamIndex + 4) & JAM_MASK; }
+#define jamEntryLine(line) { \
+ theEmulatedJamBlockNumber = number(); \
+ Uint32 tEmulatedJamIndex = theEmulatedJamIndex; \
+ *(Uint32*)((UintPtr)theEmulatedJam + (Uint32)tEmulatedJamIndex) = \
+ ((theEmulatedJamBlockNumber << 20) | line); \
+ theEmulatedJamIndex = (tEmulatedJamIndex + 4) & JAM_MASK; }
+
+#endif
+
+#else
+#define jam()
+#define jamLine(line)
+#define jamEntry()
+#define jamEntryLine(line)
+#endif
+#ifndef NDB_OPT
+#define ptrCheck(ptr, limit, rec) if (ptr.i < (limit)) ptr.p = &rec[ptr.i]; else ptr.p = NULL
+
+/**
+ * Sets the p-value of a ptr-struct to be a pointer to record no i
+ * (where i is the i-value of the ptr-struct)
+ *
+ * @param ptr ptr-struct with a set i-value (the p-value in this gets set)
+ * @param limit max no of records in rec
+ * @param rec pointer to first record in an array of records
+ */
+#define ptrCheckGuard(ptr, limit, rec) {\
+ UintR TxxzLimit; \
+ TxxzLimit = (limit); \
+ UintR TxxxPtr; \
+ TxxxPtr = ptr.i; \
+ ptr.p = &rec[TxxxPtr]; \
+ if (TxxxPtr < (TxxzLimit)) { \
+ ; \
+ } else { \
+ progError(__LINE__, ERR_POINTER_NOTINRANGE, __FILE__); \
+ }}
+
+#define ptrAss(ptr, rec) ptr.p = &rec[ptr.i]
+#define ptrNull(ptr) ptr.p = NULL
+#define ptrGuard(ptr) if (ptr.p == NULL) \
+ progError(__LINE__, ERR_POINTER_NOTINRANGE, __FILE__)
+#define arrGuard(ind, size) if ((ind) >= (size)) \
+ progError(__LINE__, ERR_INDEX_NOTINRANGE, __FILE__)
+#else
+#define ptrCheck(ptr, limit, rec) ptr.p = &rec[ptr.i]
+#define ptrCheckGuard(ptr, limit, rec) ptr.p = &rec[ptr.i]
+#define ptrAss(ptr, rec) ptr.p = &rec[ptr.i]
+#define ptrNull(ptr) ptr.p = NULL
+#define ptrGuard(ptr)
+#define arrGuard(ind, size)
+#endif
+
+// ------- EVENT STATES OF A NODE -----------------------------
+#define ZADD 0 /* New application added */
+#define ZREMOVE 1 /* An application has been removed */
+#define ZSTART 2 /* An application is ready to start */
+#define ZRUN 3 /* An application has started to run */
+
+// -------- ERROR INSERT MACROS -------
+#ifdef ERROR_INSERT
+#define ERROR_INSERT_VARIABLE UintR cerrorInsert
+#define ERROR_INSERTED(x) (cerrorInsert == (x))
+#define SET_ERROR_INSERT_VALUE(x) cerrorInsert = x
+#define CLEAR_ERROR_INSERT_VALUE cerrorInsert = 0
+#else
+#define ERROR_INSERT_VARIABLE typedef void * cerrorInsert // Will generate compiler error if used
+#define ERROR_INSERTED(x) false
+#define SET_ERROR_INSERT_VALUE(x)
+#define CLEAR_ERROR_INSERT_VALUE
+#endif
+
+/* ------------------------------------------------------------------------- */
+/* COMMONLY USED CONSTANTS. */
+/* ------------------------------------------------------------------------- */
+#define ZFALSE 0
+#define ZTRUE 1
+#define ZSET 1
+#define ZOK 0
+#define ZNOT_OK 1
+#define ZCLOSE_FILE 2
+#define ZNIL 0xffff
+#define Z8NIL 255
+
+/* ------------------------------------------------------------------------- */
+// Number of fragments stored per node. Should be settable on a table basis
+// in future version since small tables want small value and large tables
+// need large value.
+/* ------------------------------------------------------------------------- */
+#define NO_OF_FRAG_PER_NODE 8
+
+/* ---------------------------------------------------------------- */
+// To avoid synching too big chunks at a time we synch after writing
+// a certain number of data/UNDO pages. (e.g. 2 MBytes).
+/* ---------------------------------------------------------------- */
+#define MAX_PAGES_WITHOUT_SYNCH 64
+#define MAX_REDO_PAGES_WITHOUT_SYNCH 32
+
+/* ------------------------------------------------------------------ */
+// We have these constants to ensure that we can easily change the
+// parallelism of node recovery and the amount of scan
+// operations needed for node recoovery.
+/* ------------------------------------------------------------------ */
+#define MAX_NO_WORDS_OUTSTANDING_COPY_FRAGMENT 6000
+#define MAGIC_CONSTANT 56
+#define NODE_RECOVERY_SCAN_OP_RECORDS \
+ (4 + ((4*MAX_NO_WORDS_OUTSTANDING_COPY_FRAGMENT)/ \
+ ((MAGIC_CONSTANT + 2) * 5)))
+
+#ifdef NO_CHECKPOINT
+#define NO_LCP
+#define NO_GCP
+#endif
+
+/**
+ * Ndb kernel blocks assertion handling
+ *
+ * Two type of assertions:
+ * - ndbassert - Only used when compiling VM_TRACE
+ * - ndbrequire - Always checked
+ *
+ * If a ndbassert/ndbrequire fails, the system will
+ * shutdown and generate an error log
+ *
+ *
+ * NOTE these may only be used within blocks
+ */
+#if defined VM_TRACE || defined NDB_DEBUG
+#define ndbassert(check) \
+ if((check)){ \
+ } else { \
+ progError(__LINE__, ERR_NDBREQUIRE, __FILE__); \
+ }
+
+#define ndbrequire(check) \
+ if((check)){ \
+ } else { \
+ progError(__LINE__, ERR_NDBREQUIRE, __FILE__); \
+ }
+#else
+#define ndbassert(check)
+
+#define ndbrequire(check) \
+ if((check)){ \
+ } else { \
+ progError(__LINE__, ERR_NDBREQUIRE, __FILE__); \
+ }
+#endif
+
+#define CRASH_INSERTION(errorType) \
+ if (!ERROR_INSERTED((errorType))) { \
+ } else { \
+ progError(__LINE__, ERR_ERROR_INSERT, __FILE__); \
+ }
+
+#define CRASH_INSERTION2(errorNum, condition) \
+ if (!(ERROR_INSERTED(errorNum) && condition)) { \
+ } else { \
+ progError(__LINE__, ERR_ERROR_INSERT, __FILE__); \
+ }
+
+#define MEMCOPY_PAGE(to, from, page_size_in_bytes) \
+ memcpy((void*)(to), (void*)(from), (size_t)(page_size_in_bytes));
+#define MEMCOPY_NO_WORDS(to, from, no_of_words) \
+ memcpy((to), (void*)(from), (size_t)(no_of_words << 2));
+
+template <class T>
+struct Ptr {
+ T * p;
+ Uint32 i;
+ inline bool isNull() const { return i == RNIL; }
+ inline void setNull() { i = RNIL; }
+};
+
+template <class T>
+struct ConstPtr {
+ const T * p;
+ Uint32 i;
+ inline bool isNull() const { return i == RNIL; }
+ inline void setNull() { i = RNIL; }
+};
+
+#endif
diff --git a/ndb/src/kernel/vm/testCopy/Makefile b/ndb/src/kernel/vm/testCopy/Makefile
new file mode 100644
index 00000000000..5abd93eb74f
--- /dev/null
+++ b/ndb/src/kernel/vm/testCopy/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := util
+
+BIN_TARGET := testCopy
+
+SOURCES = testCopy.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/vm/testCopy/rr.cpp b/ndb/src/kernel/vm/testCopy/rr.cpp
new file mode 100644
index 00000000000..2da8383f523
--- /dev/null
+++ b/ndb/src/kernel/vm/testCopy/rr.cpp
@@ -0,0 +1,33 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <sched.h>
+#include <sys/types.h>
+#include <unistd.h>
+#include <stdio.h>
+
+int
+main(int argc, char * const argv[]){
+ struct sched_param p;
+ p.sched_priority = 1;
+
+ int ret = sched_setscheduler(getpid(), SCHED_RR, &p);
+ printf("ref = %d\n", ret);
+
+ execv(argv[1], &argv[1]);
+ return 0;
+}
diff --git a/ndb/src/kernel/vm/testCopy/testCopy.cpp b/ndb/src/kernel/vm/testCopy/testCopy.cpp
new file mode 100644
index 00000000000..1b4b24f5934
--- /dev/null
+++ b/ndb/src/kernel/vm/testCopy/testCopy.cpp
@@ -0,0 +1,343 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbOut.hpp>
+#include <NdbTick.h>
+#include <ndb_types.h>
+#include <string.h>
+#include <stdlib.h>
+
+#ifdef __NDB_FORTE6
+#define HAND
+bool hand = true;
+#else
+bool hand = false;
+#endif
+
+struct Data7 {
+ Uint32 data[7];
+
+#ifdef HAND
+ inline Data7& operator=(const Data7 & o){
+ Uint32 t0 = o.data[0];
+ Uint32 t1 = o.data[1];
+ Uint32 t2 = o.data[2];
+ Uint32 t3 = o.data[3];
+ Uint32 t4 = o.data[4];
+ Uint32 t5 = o.data[5];
+ Uint32 t6 = o.data[6];
+ data[0] = t0;
+ data[1] = t1;
+ data[2] = t2;
+ data[3] = t3;
+ data[4] = t4;
+ data[5] = t5;
+ data[6] = t6;
+ return * this;
+ }
+#endif
+};
+
+struct Data25 {
+ Uint32 data[25];
+};
+
+struct TestSignal {
+
+ Data7 head;
+ Data25 data;
+};
+
+Uint32 g_time = 3000;
+Uint32 g_count = 8*2048;
+
+TestSignal g_signal;
+TestSignal * g_jobBuffer;
+
+template<Uint32 LEN>
+inline
+void
+MEMCOPY(Uint32 * to, const Uint32 * from){
+ Uint32 t0 ;
+ Uint32 t1 ;
+ Uint32 t2 ;
+ Uint32 t3 ;
+ Uint32 len = LEN;
+ while(len > 4){
+ t0 = from[0];
+ t1 = from[1];
+ t2 = from[2];
+ t3 = from[3];
+
+ to[0] = t0;
+ to[1] = t1;
+ to[2] = t2;
+ to[3] = t3;
+
+
+ to += 4;
+ from += 4;
+ len -= 4;
+ }
+
+ //ndbout_c("len = %d", len);
+
+ t0 = from[0];
+ t1 = from[1];
+ t2 = from[2];
+ switch(len & 3){
+ case 3:
+ //ndbout_c("3");
+ to[2] = t2;
+ case 2:
+ //ndbout_c("2");
+ to[1] = t1;
+ case 1:
+ //ndbout_c("1");
+ to[0] = t0;
+ }
+
+}
+
+inline
+void
+MEMCOPY_NO_WORDS(Uint32 * to, const Uint32 * from, Uint32 len){
+ Uint32 t0 ;
+ Uint32 t1 ;
+ Uint32 t2 ;
+ Uint32 t3 ;
+ while(len > 4){
+ t0 = from[0];
+ t1 = from[1];
+ t2 = from[2];
+ t3 = from[3];
+
+ to[0] = t0;
+ to[1] = t1;
+ to[2] = t2;
+ to[3] = t3;
+
+
+ to += 4;
+ from += 4;
+ len -= 4;
+ }
+
+ //ndbout_c("len = %d", len);
+
+ t0 = from[0];
+ t1 = from[1];
+ t2 = from[2];
+ switch(len & 3){
+ case 3:
+ //ndbout_c("3");
+ to[2] = t2;
+ case 2:
+ //ndbout_c("2");
+ to[1] = t1;
+ case 1:
+ //ndbout_c("1");
+ to[0] = t0;
+ }
+}
+
+inline
+void
+copy1(Uint32 i, TestSignal & ts){
+ TestSignal & dst = g_jobBuffer[i];
+
+ Uint32 t0 = ts.head.data[0];
+ Uint32 t1 = ts.head.data[1];
+ Uint32 t2 = ts.head.data[2];
+ Uint32 t3 = ts.head.data[3];
+ Uint32 t4 = ts.head.data[4];
+ Uint32 t5 = ts.head.data[5];
+ Uint32 t6 = ts.head.data[6];
+
+ dst.head.data[0] = t0;
+ dst.head.data[1] = t1;
+ dst.head.data[2] = t2;
+ dst.head.data[3] = t3;
+ dst.head.data[4] = t4;
+ dst.head.data[5] = t5;
+ dst.head.data[6] = t6;
+}
+
+
+
+inline
+void
+copy2(Uint32 i, TestSignal & ts){
+ TestSignal & dst = g_jobBuffer[i];
+
+ Uint32 t0 = ts.head.data[0];
+ Uint32 t1 = ts.head.data[1];
+ Uint32 t2 = ts.head.data[2];
+ Uint32 t3 = ts.head.data[3];
+
+ dst.head.data[0] = t0;
+ dst.head.data[1] = t1;
+ dst.head.data[2] = t2;
+ dst.head.data[3] = t3;
+
+ Uint32 t4 = ts.head.data[4];
+ Uint32 t5 = ts.head.data[5];
+ Uint32 t6 = ts.head.data[6];
+
+ dst.head.data[4] = t4;
+ dst.head.data[5] = t5;
+ dst.head.data[6] = t6;
+}
+
+inline
+void
+copy3(Uint32 i, TestSignal & ts){
+ TestSignal & dst = g_jobBuffer[i];
+
+ dst.head = ts.head;
+}
+
+inline
+void
+copy4(Uint32 i, TestSignal & ts){
+ TestSignal & dst = g_jobBuffer[i];
+
+ memcpy(&dst.head.data[0], &ts.head.data[0], sizeof(Data7));
+}
+
+inline
+void
+copy5(Uint32 i, TestSignal & ts){
+ TestSignal & dst = g_jobBuffer[i];
+
+ MEMCOPY_NO_WORDS(&dst.head.data[0], &ts.head.data[0], 7);
+}
+
+inline
+void
+copy6(Uint32 i, TestSignal & ts){
+ TestSignal & dst = g_jobBuffer[i];
+
+ MEMCOPY<7>(&dst.head.data[0], &ts.head.data[0]);
+}
+
+inline
+void
+copy7(Uint32 i, TestSignal & ts){
+ TestSignal & dst = g_jobBuffer[i];
+
+#if (__GNUC__ >= 3 ) || (__GNUC__ == 2 && __GNUC_MINOR >= 95)
+ __builtin_memcpy(&dst.head.data[0], &ts.head.data[0], sizeof(Data7));
+#else
+ dst.head = ts.head;
+#endif
+}
+
+template<void (* C)(Uint32 i, TestSignal & ts)>
+int
+doTime(Uint32 ms){
+
+ Uint64 ms1, ms2;
+ const Uint32 count = g_count;
+ for(Uint32 i = 0; i<count; i++)
+ C(i, g_signal);
+ for(Uint32 i = 0; i<count; i++)
+ C(i, g_signal);
+
+ Uint32 laps = 0;
+
+ ms1 = NdbTick_CurrentMillisecond();
+ do {
+ for(int j = 100; j>= 0; j--)
+ for(Uint32 i = 0; i<count; i++){
+ C(i, g_signal);
+ }
+ ms2 = NdbTick_CurrentMillisecond();
+ laps += 100;
+ } while((ms2 - ms1) < ms);
+
+ return laps;
+}
+
+
+template<void (* C)(Uint32 i, TestSignal & ts)>
+void doCopyLap(Uint32 laps, const char * title){
+
+ Uint64 ms1, ms2;
+ const Uint32 count = g_count;
+ for(Uint32 i = 0; i<count; i++)
+ C(i, g_signal);
+ laps--;
+ for(Uint32 i = 0; i<count; i++)
+ C(i, g_signal);
+ laps--;
+
+ Uint32 div = laps;
+
+ ms1 = NdbTick_CurrentMillisecond();
+ while(laps > 0){
+ for(Uint32 i = 0; i<count; i++){
+#if (__GNUC__ == 3 && __GNUC_MINOR >= 1)
+ _builtin_prefetch(&g_jobBuffer[i], 1, 0);
+#endif
+ C(i, g_signal);
+ }
+ laps--;
+ }
+ ms2 = NdbTick_CurrentMillisecond();
+
+ ms2 -= ms1;
+ Uint32 diff = ms2;
+ ndbout_c("%s : %d laps in %d millis => %d copies/sec",
+ title, div, diff, (1000*div*g_count+(diff/2))/diff);
+}
+
+int
+main(int argc, const char ** argv){
+
+ if(argc > 1)
+ g_count = atoi(argv[1]);
+
+ if(argc > 2)
+ g_time = atoi(argv[2]);
+
+ ndbout_c("Using %d entries => %d kB ",
+ g_count,
+ g_count * sizeof(TestSignal) / 1024);
+ ndbout_c("Testing for %d ms", g_time);
+
+ ndbout_c("Using %s copy-constructor",
+ (hand ? "hand written" : "compiler generated"));
+
+ g_jobBuffer = new TestSignal[g_count + 1];
+ for(int i = 0; i<10; i++)
+ memset(g_jobBuffer, 0, g_count * sizeof(TestSignal));
+
+ int laps = doTime<copy2>(g_time);
+ ndbout_c("Laps = %d", laps);
+
+ doCopyLap<copy2>(laps, "4 t-variables");
+ doCopyLap<copy3>(laps, "copy constr. ");
+ doCopyLap<copy1>(laps, "7 t-variables");
+ doCopyLap<copy4>(laps, "mem copy ");
+ doCopyLap<copy5>(laps, "mem copy hand");
+ doCopyLap<copy6>(laps, "mem copy temp");
+ doCopyLap<copy7>(laps, "mem copy gcc ");
+
+ delete[] g_jobBuffer;
+ return 0;
+}
diff --git a/ndb/src/kernel/vm/testDataBuffer/Makefile b/ndb/src/kernel/vm/testDataBuffer/Makefile
new file mode 100644
index 00000000000..693989dfe3c
--- /dev/null
+++ b/ndb/src/kernel/vm/testDataBuffer/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := kernel
+
+BIN_TARGET := testKernelDataBuffer
+BIN_TARGET_ARCHIVES := general portlib
+
+SOURCES = testDataBuffer.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/vm/testDataBuffer/testDataBuffer.cpp b/ndb/src/kernel/vm/testDataBuffer/testDataBuffer.cpp
new file mode 100644
index 00000000000..def8387e343
--- /dev/null
+++ b/ndb/src/kernel/vm/testDataBuffer/testDataBuffer.cpp
@@ -0,0 +1,186 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbTick.h>
+#include <DataBuffer.hpp>
+#include <stdlib.h>
+#include <string.h>
+
+struct Buffer {
+ Buffer(Uint32 size){ m_sz = size; buffer = new Uint32[m_sz]; m_len = 0;}
+ ~Buffer(){ delete [] buffer;}
+
+ Uint32 m_sz;
+ Uint32 m_len;
+ Uint32 * buffer;
+};
+
+inline
+void
+require(bool b){
+ if(!b)
+ abort();
+}
+
+template<Uint32 sz>
+void
+compare(DataBuffer<sz> & db, Buffer & buf){
+ typename DataBuffer<sz>::DataBufferIterator it;
+ db.first(it);
+ for(Uint32 i = 0; i<buf.m_len; i++){
+ if(buf.buffer[i] != * it.data){
+ db.print(stdout);
+ abort();
+ }
+ db.next(it);
+ }
+
+ for(Uint32 i = 0; i<buf.m_len; i++){
+ if(!db.position(it, i))
+ abort();
+ if(buf.buffer[i] != * it.data){
+ db.print(stdout);
+ abort();
+ }
+ }
+}
+
+template<Uint32 sz>
+void
+test(Uint32 loops, Uint32 iter){
+
+ ndbout_c("DataBuffer<%d> loops=%d iter=%d", sz, loops, iter);
+
+ while(loops-- > 0){
+ Uint32 size = sz*((10 + (rand() % (10 * sz)) + sz - 1)/sz);
+
+ typename DataBuffer<sz>::DataBufferPool thePool;
+ typename DataBuffer<sz>::DataBufferIterator it;
+ DataBuffer<sz> db(thePool);
+
+ thePool.setSize((size + sz - 1) / sz);
+ Buffer buf(size);
+
+ bool testOverRun = true;
+
+ for(Uint32 i = 0; i<iter; i++){
+ Uint32 c = (rand() % (testOverRun ? 7 : 4));
+ Uint32 free = (size - db.getSize());
+ Uint32 alloc = 0;
+ if(free == 0){
+ c = (testOverRun ? c : 0);
+ if(c >= 1 && c <= 3)
+ c += 3;
+ }
+
+ if(free <= 1)
+ alloc = 1;
+ else
+ alloc = 1 + (rand() % (free - 1));
+
+ //ndbout_c("iter=%d case=%d free=%d alloc=%d", i, c, free, alloc);
+ switch(c){
+ case 0: // Release
+ db.first(it);
+ for(; !it.curr.isNull(); db.next(it))
+ * it.data = 0;
+
+ db.release();
+ buf.m_len = 0;
+ break;
+ case 1:{ // Append (success)
+ for(Uint32 i = 0; i<alloc; i++)
+ buf.buffer[buf.m_len + i] = buf.m_len + i;//rand();
+
+ require(db.append(&buf.buffer[buf.m_len], alloc));
+ buf.m_len += alloc;
+ break;
+ }
+ case 2: { // Seize(1) (success)
+ for(Uint32 i = 0; i<alloc; i++){
+ buf.buffer[buf.m_len + i] = buf.m_len + i;//rand();
+ require(db.seize(1));
+ require(db.position(it, db.getSize()-1));
+ * it.data = buf.buffer[buf.m_len + i];
+ }
+ buf.m_len += alloc;
+ break;
+ }
+ case 3: { // Seize(n) (success)
+ for(Uint32 i = 0; i<alloc; i++){
+ buf.buffer[buf.m_len + i] = buf.m_len + i;//rand();
+ }
+ Uint32 pos = db.getSize();
+ require(db.seize(alloc));
+ require(db.position(it, pos));
+ for(Uint32 i = 0; i<alloc; i++){
+ * it.data = buf.buffer[buf.m_len + i];
+ db.next(it);
+ }
+ buf.m_len += alloc;
+ break;
+ }
+ case 4: { // Append fail
+ require(!db.append(buf.buffer, alloc + free));
+ require(db.getSize() == 0);
+ buf.m_len = 0;
+ break;
+ }
+ case 5: { // Seize(1) - fail
+ for(Uint32 i = 0; i<free; i++){
+ require(db.seize(1));
+ }
+
+ require(!db.seize(1));
+ require(db.getSize() == 0);
+ buf.m_len = 0;
+ break;
+ }
+ case 6: { // Seize(n) - fail
+ require(!db.seize(alloc + free));
+ require(db.getSize() == 0);
+ buf.m_len = 0;
+ break;
+ }
+ }
+ compare(db, buf);
+ }
+ }
+}
+
+int
+main(void){
+
+ srand(NdbTick_CurrentMillisecond());
+
+
+ test<1>(1000, 1000);
+ test<11>(1000, 1000);
+ test<15>(1000, 1000);
+ test<16>(1000, 1000);
+ test<17>(1000, 1000);
+#if 0
+#endif
+ return 0;
+}
+
+void
+ErrorReporter::handleAssert(const char * msg, const char * file, int line)
+{
+ ndbout << "ErrorReporter::handleAssert activated - "
+ << " line= " << line << endl;
+ abort();
+}
diff --git a/ndb/src/kernel/vm/testLongSig/Makefile b/ndb/src/kernel/vm/testLongSig/Makefile
new file mode 100644
index 00000000000..ecf33dca109
--- /dev/null
+++ b/ndb/src/kernel/vm/testLongSig/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := signalsender
+
+BIN_TARGET := testLongSig
+
+SOURCES = testLongSig.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/vm/testLongSig/testLongSig.cpp b/ndb/src/kernel/vm/testLongSig/testLongSig.cpp
new file mode 100644
index 00000000000..6d421268a0a
--- /dev/null
+++ b/ndb/src/kernel/vm/testLongSig/testLongSig.cpp
@@ -0,0 +1,300 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <editline/editline.h>
+#include <SignalSender.hpp>
+#include <assert.h>
+#include <stdlib.h>
+
+void
+print_help(){
+ ndbout << "The test menu" << endl;
+ ndbout << "1 - Sending of long signals w/ segmented sections" << endl;
+ ndbout << "2 - As 1 but using receiver group" << endl;
+ ndbout << "3 - Sending of long signals w/ linear sections" << endl;
+ ndbout << "4 - As 3 but using receiver group" << endl;
+ ndbout << "5 - Sending of manually fragmented signals w/ segmented sections"
+ << endl;
+ ndbout << "6 - As 5 but using receiver group" << endl;
+ ndbout << "7 - Sending of manually fragmented signals w/ linear sections"
+ << endl;
+ ndbout << "8 - As but using receiver group" << endl;
+
+ ndbout << "9 - Sending of CONTINUEB fragmented signals w/ segmented sections"
+ << endl;
+ ndbout << "10 - As 9 but using receiver group" << endl;
+ ndbout << "11 - Sending of CONTINUEB fragmented signals w/ linear sections"
+ << endl;
+ ndbout << "12 - As but using receiver group" << endl;
+ ndbout << "r - Recive signal from anyone" << endl;
+ ndbout << "a - Run tests 1 - 12 with variable sizes - 10 loops" << endl;
+ ndbout << "b - Run tests 1 - 12 with variable sizes - 100 loops" << endl;
+ ndbout << "c - Run tests 1 - 12 with variable sizes - 1000k loops" << endl;
+}
+
+void runTest(SignalSender &, Uint32 i, bool verbose);
+
+static
+int
+randRange(Uint32 min, Uint32 max){
+ float r = rand();
+ float f = (max - min + 1);
+ float d = (float)RAND_MAX + 1.0;
+ return min + (int)(f * r / d);
+}
+
+static
+int
+randRange(const Uint32 odds[], Uint32 count){
+ Uint32 val = randRange((Uint32)0, 100);
+
+ Uint32 i = 0;
+ Uint32 sum = 0;
+ while(sum <= val && i < count){
+ sum += odds[i];
+ i++;
+ }
+ return i - 1;
+}
+
+int
+main(void){
+
+ srand(NdbTick_CurrentMillisecond());
+#if 0
+ for(int i = 0; i<100; i++)
+ ndbout_c("randRange(0, 3) = %d", randRange(0, 3));
+ return 0;
+#endif
+ SignalSender ss;
+
+ ndbout << "Connecting...";
+ if(!ss.connect(30)){
+ ndbout << "failed" << endl << "Exiting" << endl;
+ return 0;
+ }
+ ndbout << "done" << endl;
+ ndbout_c("Connected as block=%d node=%d",
+ refToBlock(ss.getOwnRef()), refToNode(ss.getOwnRef()));
+
+ Uint32 data[25];
+ Uint32 sec0[70];
+ Uint32 sec1[123];
+ Uint32 sec2[10];
+
+ data[0] = ss.getOwnRef();
+ data[1] = 1;
+ data[2] = 76;
+ data[3] = 1;
+ data[4] = 1;
+ data[5] = 70;
+ data[6] = 123;
+ data[7] = 10;
+ const Uint32 theDataLen = 8;
+
+ for(Uint32 i = 0; i<70; i++)
+ sec0[i] = i;
+
+ for(Uint32 i = 0; i<123; i++)
+ sec1[i] = 70+i;
+
+ for(Uint32 i = 0; i<10; i++)
+ sec2[i] = (i + 1)*(i + 1);
+
+ SimpleSignal signal1;
+ signal1.set(ss, 0, CMVMI, GSN_TESTSIG, theDataLen + 2);
+ signal1.header.m_noOfSections = 1;
+ signal1.header.m_fragmentInfo = 1;
+
+ memcpy(&signal1.theData[0], data, 4 * theDataLen );
+ signal1.theData[theDataLen + 0] = 0;
+ signal1.theData[theDataLen + 1] = 7; // FragmentId
+
+ signal1.ptr[0].sz = 60;
+ signal1.ptr[0].p = &sec0[0];
+
+ SimpleSignal signal2;
+
+ Uint32 idx = 0;
+ memcpy(&signal2.theData[0], data, 4 * theDataLen );
+ signal2.theData[theDataLen + idx] = 0; idx++;
+ signal2.theData[theDataLen + idx] = 1; idx++;
+ //signal2.theData[theDataLen + idx] = 2; idx++;
+ signal2.theData[theDataLen + idx] = 7; idx++; // FragmentId
+
+ signal2.set(ss, 0, CMVMI, GSN_TESTSIG, theDataLen + idx);
+ signal2.header.m_fragmentInfo = 3;
+ signal2.header.m_noOfSections = idx - 1;
+
+ signal2.ptr[0].sz = 10;
+ signal2.ptr[0].p = &sec0[60];
+
+ signal2.ptr[1].sz = 123;
+ signal2.ptr[1].p = &sec1[0];
+
+ signal2.ptr[2].sz = 10;
+ signal2.ptr[2].p = &sec2[0];
+
+ char * buf;
+ while((buf = readline("Enter command: "))){
+ add_history(buf);
+ data[1] = atoi(buf);
+ if(strcmp(buf, "r") == 0){
+ SimpleSignal * ret1 = ss.waitFor();
+ (* ret1).print();
+ delete ret1;
+ continue;
+ }
+ if(strcmp(buf, "a") == 0){
+ runTest(ss, 10, true);
+ print_help();
+ continue;
+ }
+ if(strcmp(buf, "b") == 0){
+ runTest(ss, 100, false);
+ print_help();
+ continue;
+ }
+ if(strcmp(buf, "c") == 0){
+ runTest(ss, 1000000, false);
+ print_help();
+ continue;
+ }
+
+ if(data[1] >= 1 && data[1] <= 12){
+ Uint32 nodeId = ss.getAliveNode();
+ ndbout_c("Sending 2 fragmented to node %d", nodeId);
+ ss.sendSignal(nodeId, &signal1);
+ ss.sendSignal(nodeId, &signal2);
+
+ if(data[1] >= 5){
+ continue;
+ }
+ ndbout_c("Waiting for signal from %d", nodeId);
+
+ SimpleSignal * ret1 = ss.waitFor((Uint16)nodeId);
+ (* ret1).print();
+ Uint32 count = ret1->theData[4] - 1;
+ delete ret1;
+ while(count > 0){
+ ndbout << "Waiting for " << count << " signals... ";
+ SimpleSignal * ret1 = ss.waitFor();
+ ndbout_c("received from node %d",
+ refToNode(ret1->header.theSendersBlockRef));
+ (* ret1).print();
+ delete ret1;
+ count--;
+ }
+ } else {
+ print_help();
+ }
+ }
+ ndbout << "Exiting" << endl;
+};
+
+void
+runTest(SignalSender & ss, Uint32 count, bool verbose){
+
+ SimpleSignal sig;
+ Uint32 sec0[256];
+ Uint32 sec1[256];
+ Uint32 sec2[256];
+ for(Uint32 i = 0; i<256; i++){
+ sec0[i] = i;
+ sec1[i] = i + i;
+ sec2[i] = i * i;
+ }
+
+ sig.set(ss, 0, CMVMI, GSN_TESTSIG, 8);
+ sig.theData[0] = ss.getOwnRef();
+ sig.theData[1] = 1; // TestType
+ sig.theData[2] = 128; // FragSize
+ sig.theData[3] = 0; // Print
+ sig.theData[4] = 1; // RetCount
+
+ sig.ptr[0].p = &sec0[0];
+ sig.ptr[1].p = &sec1[0];
+ sig.ptr[2].p = &sec2[0];
+
+ for(unsigned loop = 0; loop < count; loop++){
+ const Uint32 odds[] = { 5, 40, 30, 25 };
+ const Uint32 secs = randRange(odds, 4);
+ sig.ptr[0].sz = randRange(1, 256);
+ sig.ptr[1].sz = randRange(1, 256);
+ sig.ptr[2].sz = randRange(1, 256);
+ sig.header.m_noOfSections = secs;
+ ndbout << "Loop " << loop << " #secs = " << secs << " sizes = [ ";
+ unsigned min = 256;
+ unsigned max = 0;
+ unsigned sum = 0;
+ for(unsigned i = 0; i<secs; i++){
+ const Uint32 sz = sig.ptr[i].sz;
+ ndbout << sz << " ";
+ min = (min < sz ? min : sz);
+ max = (max > sz ? max : sz);
+ sum += sz;
+ sig.theData[5+i] = sz;
+ }
+ ndbout_c("]");
+ for(int test = 1; test <= 12; test++){
+ sig.theData[1] = test;
+ Uint32 nodeId = ss.getAliveNode();
+ if(verbose){
+ ndbout << " Test " << test << " node " << nodeId << "...";
+ fflush(stdout);
+ }
+ SendStatus r = ss.sendSignal(nodeId, &sig);
+ assert(r == SEND_OK);
+ if(test < 5){
+ SimpleSignal * ret1 = ss.waitFor((Uint16)nodeId);
+ Uint32 count = ret1->theData[4] - 1;
+ delete ret1;
+
+ while(count > 0){
+ SimpleSignal * ret1 = ss.waitFor();
+ delete ret1;
+ count--;
+ }
+ if(verbose)
+ ndbout << "done" << endl;
+ } else {
+ Uint32 nodes = ss.getNoOfConnectedNodes();
+ Uint32 sum2 = 0;
+ if((test & 1) == 1)
+ nodes = 1;
+ while(nodes > 0){
+ SimpleSignal * ret = ss.waitFor();
+ if(ret->header.m_fragmentInfo == 0){
+ for(Uint32 i = 0; i<ret->header.m_noOfSections; i++)
+ sum2 += ret->ptr[i].sz;
+ } else {
+ for(Uint32 i = 0; i<ret->header.m_noOfSections; i++)
+ if(ret->theData[i] != 3)
+ sum2 += ret->ptr[i].sz;
+ }
+ if(ret->header.m_fragmentInfo == 0 ||
+ ret->header.m_fragmentInfo == 3){
+ nodes--;
+ }
+ delete ret;
+ }
+ if(verbose)
+ ndbout_c("done sum=%d sum2=%d", sum, sum2);
+ }
+ }
+ }
+}
diff --git a/ndb/src/kernel/vm/testSimplePropertiesSection/Makefile b/ndb/src/kernel/vm/testSimplePropertiesSection/Makefile
new file mode 100644
index 00000000000..fb3aea00507
--- /dev/null
+++ b/ndb/src/kernel/vm/testSimplePropertiesSection/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := kernel
+
+BIN_TARGET := testSimplePropertiesSection
+BIN_TARGET_ARCHIVES := general portlib
+
+SOURCES = test.cpp ../SimplePropertiesSection.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/kernel/vm/testSimplePropertiesSection/test.cpp b/ndb/src/kernel/vm/testSimplePropertiesSection/test.cpp
new file mode 100644
index 00000000000..20a5d5230fb
--- /dev/null
+++ b/ndb/src/kernel/vm/testSimplePropertiesSection/test.cpp
@@ -0,0 +1,170 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbTick.h>
+#include <TransporterDefinitions.hpp>
+#include <SimpleProperties.hpp>
+#include <LongSignal.hpp>
+#include <stdlib.h>
+#include <string.h>
+
+struct Buffer {
+ Buffer(Uint32 size){ m_sz = size; buffer = new Uint32[m_sz]; m_len = 0;}
+ ~Buffer(){ delete [] buffer;}
+
+ Uint32 m_sz;
+ Uint32 m_len;
+ Uint32 * buffer;
+};
+
+inline
+void
+require(bool b){
+ if(!b)
+ abort();
+}
+
+#define relSz(x) ((x + SectionSegment::DataLength - 1) / SectionSegment::DataLength)
+
+void
+release(SectionSegmentPool & thePool, SegmentedSectionPtr & ptr){
+ const Uint32 sz = relSz(ptr.sz);
+ thePool.releaseList(sz,
+ ptr.i,
+ ptr.p->m_lastSegment);
+}
+
+void
+compare(SimplePropertiesSectionReader & db, Buffer & buf){
+
+ {
+ bool fail = false;
+ db.reset();
+ for(Uint32 i = 0; i<buf.m_len; i++){
+ Uint32 tmp;
+ if(!db.getWord(&tmp)){
+ ndbout_c("getWord(...) failed i=%d size=%d", i, buf.m_len);
+ abort();
+ }
+
+ if(tmp != buf.buffer[i]){
+ ndbout_c("getWord(...)=%d != buf[%d]=%d size=%d", tmp, i,
+ buf.buffer[i], buf.m_len);
+ fail = true;
+ }
+ }
+ require(!fail);
+ }
+
+ {
+ db.reset();
+ Buffer buf2(buf.m_sz);
+ if(!db.getWords(buf2.buffer, buf.m_len))
+ abort();
+
+ bool fail = false;
+ for(Uint32 i = 0; i<buf.m_len; i++){
+ if(buf.buffer[i] != buf2.buffer[i]){
+ ndbout_c("getWords(...) buf[%d] != buf2[%d] size=%d", i, i, buf.m_len);
+ fail = true;
+ }
+ }
+ require(!fail);
+ }
+}
+
+
+void
+test(Uint32 sz, Uint32 loops, Uint32 iter){
+
+ ndbout_c("SimplePropertiesSection sz=%d loops=%d iter=%d", sz, loops, iter);
+
+ while(loops-- > 0){
+ Uint32 size = sz*((10 + (rand() % (10 * sz)) + sz - 1)/sz);
+
+ Buffer buf(size);
+ SectionSegmentPool thePool; thePool.setSize(size);
+
+ for(Uint32 i = 0; i<iter; i++){
+ Uint32 c = 0 + (rand() % (2));
+
+ const Uint32 alloc = 1 + (rand() % (size - 1));
+ SegmentedSectionPtr dst;
+
+ if(0)
+ ndbout_c("size: %d loops: %d iter: %d c=%d alloc=%d",
+ size, loops, i, c, alloc);
+
+ switch(c){
+ case 0:{
+ for(Uint32 i = 0; i<alloc; i++)
+ buf.buffer[i] = i; //rand();
+ buf.m_len = alloc;
+
+ SimplePropertiesSectionWriter w(thePool);
+ for(Uint32 i = 0; i<alloc; i++){
+ w.putWord(buf.buffer[i]);
+ }
+ w.getPtr(dst);
+ break;
+ }
+ case 1:{
+ for(Uint32 i = 0; i<alloc; i++)
+ buf.buffer[i] = i; //rand();
+ buf.m_len = alloc;
+
+ SimplePropertiesSectionWriter w(thePool);
+ Uint32 i = 0;
+ while(i < alloc){
+ Uint32 sz = rand() % (alloc - i + 1);
+ w.putWords(&buf.buffer[i], sz);
+ i += sz;
+ }
+ w.getPtr(dst);
+ break;
+ }
+ case 2:{
+ break;
+ }
+ }
+ SimplePropertiesSectionReader r(dst, thePool);
+ compare(r, buf);
+ release(thePool, dst);
+ require(thePool.getSize() == thePool.getNoOfFree());
+ }
+ }
+}
+
+int
+main(void){
+
+ srand(NdbTick_CurrentMillisecond());
+
+ //test( 1, 1000, 1000);
+ test(54, 1000, 1000);
+ test(59, 1000, 1000);
+ test(60, 1000, 1000);
+ test(61, 1000, 1000);
+ return 0;
+}
+
+void
+ErrorReporter::handleAssert(const char * msg, const char * file, int line)
+{
+ ndbout << "ErrorReporter::handleAssert activated - "
+ << " line= " << line << endl;
+ abort();
+}
diff --git a/ndb/src/mgmapi/Makefile b/ndb/src/mgmapi/Makefile
new file mode 100644
index 00000000000..fac852dbba8
--- /dev/null
+++ b/ndb/src/mgmapi/Makefile
@@ -0,0 +1,27 @@
+include .defs.mk
+
+TYPE := util
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := mgmapi
+
+A_LIB := Y
+SO_LIB := Y
+PIC_LIB := Y
+
+DIRS := test
+
+LIB_TARGET := MGM_API
+LIB_TARGET_ARCHIVES := $(ARCHIVE_TARGET) general portlib
+
+# Source files of non-templated classes (.C files)
+SOURCES = mgmapi.cpp
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include/mgmapi) \
+ -I$(call fixpath,$(NDB_TOP)/src/common/mgmcommon)
+
+CCFLAGS += -DNO_DEBUG_MESSAGES
+
+# -I$(NDB_TOP)/src/common/mgmcommon
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/mgmapi/mgmapi.cpp b/ndb/src/mgmapi/mgmapi.cpp
new file mode 100644
index 00000000000..4c1355e8e46
--- /dev/null
+++ b/ndb/src/mgmapi/mgmapi.cpp
@@ -0,0 +1,1465 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbTCP.h>
+#include "mgmapi.h"
+#include "mgmapi_debug.h"
+#include <socket_io.h>
+
+#include <stdlib.h>
+
+#include <string.h>
+#include <NdbStdio.h>
+#include <NdbString.h>
+#include <errno.h>
+#include <NdbOut.hpp>
+#include <SocketServer.hpp>
+#include <Parser.hpp>
+#include <OutputStream.hpp>
+#include <InputStream.hpp>
+
+
+#define MGM_CMD(name, fun, desc) \
+ { name, \
+ 0, \
+ ParserRow<ParserDummy>::Cmd, \
+ ParserRow<ParserDummy>::String, \
+ ParserRow<ParserDummy>::Optional, \
+ ParserRow<ParserDummy>::IgnoreMinMax, \
+ 0, 0, \
+ fun, \
+ desc, 0 }
+
+#define MGM_ARG(name, type, opt, desc) \
+ { name, \
+ 0, \
+ ParserRow<ParserDummy>::Arg, \
+ ParserRow<ParserDummy>::type, \
+ ParserRow<ParserDummy>::opt, \
+ ParserRow<ParserDummy>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ desc, 0 }
+
+#define MGM_END() \
+ { 0, \
+ 0, \
+ ParserRow<ParserDummy>::Arg, \
+ ParserRow<ParserDummy>::Int, \
+ ParserRow<ParserDummy>::Optional, \
+ ParserRow<ParserDummy>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0, 0 }
+
+class ParserDummy : SocketServer::Session
+{
+public:
+ ParserDummy(NDB_SOCKET_TYPE sock);
+};
+
+ParserDummy::ParserDummy(NDB_SOCKET_TYPE sock) : SocketServer::Session(sock)
+{
+}
+
+typedef Parser<ParserDummy> Parser_t;
+
+#define NDB_MGM_MAX_ERR_DESC_SIZE 256
+
+struct ndb_mgm_handle {
+ char * hostname;
+ unsigned short port;
+
+ int connected;
+ int last_error;
+ int last_error_line;
+ char last_error_desc[NDB_MGM_MAX_ERR_DESC_SIZE];
+ int read_timeout;
+ int write_timeout;
+
+ NDB_SOCKET_TYPE socket;
+
+#ifdef MGMAPI_LOG
+ FILE* logfile;
+#endif
+};
+
+#define SET_ERROR(h, e, s) \
+ { \
+ char tmp[NDB_MGM_MAX_ERR_DESC_SIZE]; \
+ snprintf(tmp, NDB_MGM_MAX_ERR_DESC_SIZE, " (mgmapi.cpp:%d)", __LINE__); \
+ strncpy(h->last_error_desc, s, NDB_MGM_MAX_ERR_DESC_SIZE); \
+ strncat(h->last_error_desc, tmp, NDB_MGM_MAX_ERR_DESC_SIZE); \
+ h->last_error = e; \
+ h->last_error_line = __LINE__; \
+ }
+
+#define CHECK_HANDLE(handle, ret) \
+ if(handle == 0) { \
+ SET_ERROR(handle, NDB_MGM_ILLEGAL_SERVER_HANDLE, ""); \
+ return ret; \
+ }
+
+#define CHECK_CONNECTED(handle, ret) \
+ if (handle->connected != 1) { \
+ SET_ERROR(handle, NDB_MGM_SERVER_NOT_CONNECTED , ""); \
+ return ret; \
+ }
+
+#define CHECK_REPLY(reply, ret) \
+ if(reply == NULL) { \
+ SET_ERROR(handle, NDB_MGM_ILLEGAL_SERVER_REPLY, ""); \
+ return ret; \
+ }
+
+/*****************************************************************************
+ * Handles
+ *****************************************************************************/
+
+extern "C"
+NdbMgmHandle
+ndb_mgm_create_handle()
+{
+ NdbMgmHandle h = (NdbMgmHandle)malloc(sizeof(ndb_mgm_handle));
+ h->connected = 0;
+ h->last_error = 0;
+ h->last_error_line = 0;
+ h->hostname = 0;
+ h->socket = -1;
+ h->read_timeout = 50000;
+ h->write_timeout = 100;
+
+ strncpy(h->last_error_desc, "No error", NDB_MGM_MAX_ERR_DESC_SIZE);
+#ifdef MGMAPI_LOG
+ h->logfile = 0;
+#endif
+
+ return h;
+}
+
+/**
+ * Destroy a handle
+ */
+extern "C"
+void
+ndb_mgm_destroy_handle(NdbMgmHandle * handle)
+{
+ if(!handle)
+ return;
+ if((* handle)->connected){
+ ndb_mgm_disconnect(* handle);
+ }
+ if((* handle)->hostname != 0){
+ free((* handle)->hostname);
+ }
+#ifdef MGMAPI_LOG
+ if ((* handle)->logfile != 0){
+ fclose((* handle)->logfile);
+ (* handle)->logfile = 0;
+ }
+#endif
+ free(* handle);
+ * handle = 0;
+}
+
+/*****************************************************************************
+ * Error handling
+ *****************************************************************************/
+
+/**
+ * Get latest error associated with a handle
+ */
+extern "C"
+int
+ndb_mgm_get_latest_error(const NdbMgmHandle h)
+{
+ return h->last_error;
+}
+
+/**
+ * Get latest error line associated with a handle
+ */
+extern "C"
+int
+ndb_mgm_get_latest_error_line(const NdbMgmHandle h)
+{
+ return h->last_error_line;
+}
+
+extern "C"
+const char *
+ndb_mgm_get_latest_error_msg(const NdbMgmHandle h)
+{
+ for (int i=0; i<ndb_mgm_noOfErrorMsgs; i++) {
+ if (ndb_mgm_error_msgs[i].code == h->last_error)
+ return ndb_mgm_error_msgs[i].msg;
+ }
+
+ return "Error"; // Unknown Error message
+}
+
+extern "C"
+const char *
+ndb_mgm_get_latest_error_desc(const NdbMgmHandle h)
+{
+ return h->last_error_desc;
+}
+
+static
+int
+parse_connect_string(const char * connect_string,
+ NdbMgmHandle handle)
+{
+ if(connect_string == 0){
+ SET_ERROR(handle, NDB_MGM_ILLEGAL_CONNECT_STRING, "");
+ return -1;
+ }
+
+ char * line = strdup(connect_string);
+ if(line == 0){
+ SET_ERROR(handle, NDB_MGM_OUT_OF_MEMORY, "");
+ return -1;
+ }
+
+ char * tmp = strchr(line, ':');
+ if(tmp == 0){
+ free(line);
+ SET_ERROR(handle, NDB_MGM_OUT_OF_MEMORY, "");
+ return -1;
+ }
+ * tmp = 0; tmp++;
+
+ int port = 0;
+ if(sscanf(tmp, "%d", &port) != 1){
+ free(line);
+ SET_ERROR(handle, NDB_MGM_ILLEGAL_PORT_NUMBER, "");
+ return -1;
+ }
+
+ if(handle->hostname != 0)
+ free(handle->hostname);
+
+ handle->hostname = strdup(line);
+ handle->port = port;
+ free(line);
+ return 0;
+}
+
+/*
+ * Call an operation, and return the reply
+ */
+static const Properties *
+ndb_mgm_call(NdbMgmHandle handle, const ParserRow<ParserDummy> *command_reply,
+ const char *cmd, const Properties *cmd_args)
+{
+ SocketOutputStream out(handle->socket);
+ SocketInputStream in(handle->socket, handle->read_timeout);
+
+ out.println(cmd);
+#ifdef MGMAPI_LOG
+ /**
+ * Print command to log file
+ */
+ FileOutputStream f(handle->logfile);
+ f.println("OUT: %s", cmd);
+#endif
+
+ if(cmd_args != NULL) {
+ Properties::Iterator iter(cmd_args);
+ const char *name;
+ while((name = iter.next()) != NULL) {
+ PropertiesType t;
+ Uint32 val_i;
+ BaseString val_s;
+
+ cmd_args->getTypeOf(name, &t);
+ switch(t) {
+ case PropertiesType_Uint32:
+ cmd_args->get(name, &val_i);
+ out.println("%s: %d", name, val_i);
+ break;
+ case PropertiesType_char:
+ cmd_args->get(name, val_s);
+ out.println("%s: %s", name, val_s.c_str());
+ break;
+ default:
+ /* Ignore */
+ break;
+ }
+ }
+#ifdef MGMAPI_LOG
+ /**
+ * Print arguments to log file
+ */
+ cmd_args->print(handle->logfile, "OUT: ");
+#endif
+ }
+ out.println("");
+
+ Parser_t::Context ctx;
+ ParserDummy session(handle->socket);
+ Parser_t parser(command_reply, in, true, true, true);
+
+#if 1
+ const Properties* p = parser.parse(ctx, session);
+ if (p == NULL){
+ /**
+ * Print some info about why the parser returns NULL
+ */
+// ndbout << " status=" << ctx.m_status << ", curr="
+// << ctx.m_currentToken << endl;
+ }
+#ifdef MGMAPI_LOG
+ else {
+ /**
+ * Print reply to log file
+ */
+ p->print(handle->logfile, "IN: ");
+ }
+#endif
+ return p;
+#else
+ return parser.parse(ctx, session);
+#endif
+}
+
+/**
+ * Connect to a management server
+ */
+extern "C"
+int
+ndb_mgm_connect(NdbMgmHandle handle, const char * mgmsrv)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_connect");
+ CHECK_HANDLE(handle, -1);
+
+ if(parse_connect_string(mgmsrv, handle) != 0) {
+ SET_ERROR(handle, NDB_MGM_ILLEGAL_CONNECT_STRING, "");
+ return -1;
+ }
+
+#ifdef MGMAPI_LOG
+ /**
+ * Open the log file
+ */
+ char logname[64];
+ snprintf(logname, 64, "mgmapi.log");
+ handle->logfile = fopen(logname, "w");
+#endif
+
+ /**
+ * Do connect
+ */
+ const NDB_SOCKET_TYPE sockfd = socket(AF_INET, SOCK_STREAM, 0);
+ if (sockfd == NDB_INVALID_SOCKET) {
+ SET_ERROR(handle, NDB_MGM_ILLEGAL_SOCKET, "");
+ return -1;
+ }
+
+ struct sockaddr_in servaddr;
+ memset(&servaddr, 0, sizeof(servaddr));
+ servaddr.sin_family = AF_INET;
+ servaddr.sin_port = htons(handle->port);
+ // Convert ip address presentation format to numeric format
+ const int res1 = Ndb_getInAddr(&servaddr.sin_addr, handle->hostname);
+ if (res1 != 0) {
+ SET_ERROR(handle, NDB_MGM_ILLEGAL_IP_ADDRESS, "");
+ return -1;
+ }
+
+ const int res2 = connect(sockfd, (struct sockaddr*) &servaddr,
+ sizeof(servaddr));
+ if (res2 == -1) {
+ NDB_CLOSE_SOCKET(sockfd);
+ SET_ERROR(handle, NDB_MGM_COULD_NOT_CONNECT_TO_SOCKET, "");
+ return -1;
+ }
+
+ handle->socket = sockfd;
+ handle->connected = 1;
+
+ return 0;
+}
+
+/**
+ * Disconnect from a mgm server
+ */
+extern "C"
+int
+ndb_mgm_disconnect(NdbMgmHandle handle)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_disconnect");
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ NDB_CLOSE_SOCKET(handle->socket);
+ handle->socket = -1;
+ handle->connected = 0;
+
+ return 0;
+}
+
+struct ndb_mgm_type_atoi
+{
+ const char * str;
+ enum ndb_mgm_node_type value;
+};
+
+static struct ndb_mgm_type_atoi type_values[] =
+{
+ { "NDB", NDB_MGM_NODE_TYPE_NDB},
+ { "API", NDB_MGM_NODE_TYPE_API },
+ { "MGM", NDB_MGM_NODE_TYPE_MGM }
+};
+
+const int no_of_type_values = (sizeof(type_values) /
+ sizeof(ndb_mgm_type_atoi));
+
+extern "C"
+enum ndb_mgm_node_type
+ndb_mgm_match_node_type(const char * type)
+{
+ if(type == 0)
+ return NDB_MGM_NODE_TYPE_UNKNOWN;
+
+ for(int i = 0; i<no_of_type_values; i++)
+ if(strcmp(type, type_values[i].str) == 0)
+ return type_values[i].value;
+
+ return NDB_MGM_NODE_TYPE_UNKNOWN;
+}
+
+extern "C"
+const char *
+ndb_mgm_get_node_type_string(enum ndb_mgm_node_type type)
+{
+ for(int i = 0; i<no_of_type_values; i++)
+ if(type_values[i].value == type)
+ return type_values[i].str;
+ return 0;
+}
+
+struct ndb_mgm_status_atoi {
+ const char * str;
+ enum ndb_mgm_node_status value;
+};
+
+static struct ndb_mgm_status_atoi status_values[] =
+{
+ { "UNKNOWN", NDB_MGM_NODE_STATUS_UNKNOWN },
+ { "NO_CONTACT", NDB_MGM_NODE_STATUS_NO_CONTACT },
+ { "NOT_STARTED", NDB_MGM_NODE_STATUS_NOT_STARTED },
+ { "STARTING", NDB_MGM_NODE_STATUS_STARTING },
+ { "STARTED", NDB_MGM_NODE_STATUS_STARTED },
+ { "SHUTTING_DOWN", NDB_MGM_NODE_STATUS_SHUTTING_DOWN },
+ { "RESTARTING", NDB_MGM_NODE_STATUS_RESTARTING },
+ { "SINGLE USER MODE", NDB_MGM_NODE_STATUS_SINGLEUSER }
+};
+
+const int no_of_status_values = (sizeof(status_values) /
+ sizeof(ndb_mgm_status_atoi));
+
+extern "C"
+enum ndb_mgm_node_status
+ndb_mgm_match_node_status(const char * status)
+{
+ if(status == 0)
+ return NDB_MGM_NODE_STATUS_UNKNOWN;
+
+ for(int i = 0; i<no_of_status_values; i++)
+ if(strcmp(status, status_values[i].str) == 0)
+ return status_values[i].value;
+
+ return NDB_MGM_NODE_STATUS_UNKNOWN;
+}
+
+extern "C"
+const char *
+ndb_mgm_get_node_status_string(enum ndb_mgm_node_status status)
+{
+ for(int i = 0; i<no_of_status_values; i++)
+ if(status_values[i].value == status)
+ return status_values[i].str;
+
+ for(int i = 0; i<no_of_status_values; i++)
+ if(status_values[i].value == NDB_MGM_NODE_STATUS_UNKNOWN)
+ return status_values[i].str;
+
+ return 0;
+}
+
+static int
+status_ackumulate(struct ndb_mgm_node_state * state,
+ const char * field,
+ const char * value)
+{
+ if(strcmp("type", field) == 0){
+ state->node_type = ndb_mgm_match_node_type(value);
+ } else if(strcmp("status", field) == 0){
+ state->node_status = ndb_mgm_match_node_status(value);
+ } else if(strcmp("startphase", field) == 0){
+ state->start_phase = atoi(value);
+ } else if(strcmp("dynamic_id", field) == 0){
+ state->dynamic_id = atoi(value);
+ } else if(strcmp("node_group", field) == 0){
+ state->node_group = atoi(value);
+ } else if(strcmp("version", field) == 0){
+ state->version = atoi(value);
+ } else {
+ ndbout_c("Unknown field: %s", field);
+ }
+ return 0;
+}
+
+/**
+ * Compare function for qsort() that sorts ndb_mgm_node_state in
+ * node_id order
+ */
+static int
+cmp_state(const void *_a, const void *_b)
+{
+ struct ndb_mgm_node_state *a, *b;
+
+ a = (struct ndb_mgm_node_state *)_a;
+ b = (struct ndb_mgm_node_state *)_b;
+
+ return a->node_id > b->node_id;
+}
+
+extern "C"
+struct ndb_mgm_cluster_state *
+ndb_mgm_get_status(NdbMgmHandle handle)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_get_status");
+ CHECK_HANDLE(handle, NULL);
+ CHECK_CONNECTED(handle, NULL);
+
+ SocketOutputStream out(handle->socket);
+ SocketInputStream in(handle->socket, handle->read_timeout);
+
+ out.println("get status");
+ out.println("");
+
+ char buf[1024];
+ in.gets(buf, sizeof(buf));
+ if(buf[strlen(buf)-1] == '\n')
+ buf[strlen(buf)-1] = '\0';
+
+ if(strcmp("node status", buf) != 0) {
+ SET_ERROR(handle, NDB_MGM_ILLEGAL_NODE_STATUS, "");
+ return NULL;
+ }
+
+ in.gets(buf, sizeof(buf));
+ if(buf[strlen(buf)-1] == '\n')
+ buf[strlen(buf)-1] = '\0';
+
+ BaseString tmp(buf);
+ Vector<BaseString> split;
+ tmp.split(split, ":");
+ if(split.size() != 2){
+ abort();
+ return NULL;
+ }
+
+ if(!(split[0].trim() == "nodes")){
+ abort();
+ return NULL;
+ }
+
+ const int noOfNodes = atoi(split[1].c_str());
+
+ ndb_mgm_cluster_state *state = (ndb_mgm_cluster_state*)
+ malloc(sizeof(ndb_mgm_cluster_state)+
+ noOfNodes*sizeof(ndb_mgm_node_state));
+
+ state->no_of_nodes = noOfNodes;
+ ndb_mgm_node_state * ptr = &state->node_states[0];
+ int nodeId = 0;
+ int i = -1; ptr--;
+ for(; i<noOfNodes; ){
+ in.gets(buf, sizeof(buf));
+ tmp.assign(buf);
+
+ if(tmp.trim() == ""){
+ break;
+ }
+
+ Vector<BaseString> split;
+ tmp.split(split, ":.");
+ if(split.size() != 4)
+ break;
+
+ const int id = atoi(split[1].c_str());
+ if(id != nodeId){
+ ptr++;
+ i++;
+ nodeId = id;
+ ptr->node_id = id;
+ }
+
+ split[3].trim(" \t\n");
+
+ if(status_ackumulate(ptr,split[2].c_str(), split[3].c_str()) != 0) {
+ break;
+ }
+ }
+
+ if(i+1 != noOfNodes){
+ free(state);
+ abort();
+ return NULL;
+ }
+
+ qsort(state->node_states, state->no_of_nodes, sizeof(state->node_states[0]),
+ cmp_state);
+ return state;
+}
+
+extern "C"
+int
+ndb_mgm_enter_single_user(NdbMgmHandle handle,
+ unsigned int nodeId,
+ struct ndb_mgm_reply* /*reply*/)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_enter_single_user");
+ const ParserRow<ParserDummy> enter_single_reply[] = {
+ MGM_CMD("enter single user reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("nodeId", nodeId);
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, enter_single_reply, "enter single user", &args);
+ CHECK_REPLY(reply, -1);
+
+ BaseString result;
+ reply->get("result", result);
+ if(strcmp(result.c_str(), "Ok") != 0) {
+ SET_ERROR(handle, NDB_MGM_COULD_NOT_ENTER_SINGLE_USER_MODE,
+ result.c_str());
+ delete reply;
+ return -1;
+ }
+
+ delete reply;
+ return 0;
+}
+
+
+extern "C"
+int
+ndb_mgm_exit_single_user(NdbMgmHandle handle, struct ndb_mgm_reply* /*reply*/)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_exit_single_user");
+ const ParserRow<ParserDummy> exit_single_reply[] = {
+ MGM_CMD("exit single user reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, exit_single_reply, "exit single user", 0);
+ CHECK_REPLY(reply, -1);
+
+ const char * buf;
+ reply->get("result", &buf);
+ if(strcmp(buf,"Ok")!=0) {
+ SET_ERROR(handle, NDB_MGM_COULD_NOT_EXIT_SINGLE_USER_MODE, buf);
+ delete reply;
+ return -1;
+ }
+
+ delete reply;
+ return 0;
+}
+
+extern "C"
+int
+ndb_mgm_stop(NdbMgmHandle handle, int no_of_nodes, const int * node_list)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_stop");
+ return ndb_mgm_stop2(handle, no_of_nodes, node_list, 0);
+}
+
+
+extern "C"
+int
+ndb_mgm_stop2(NdbMgmHandle handle, int no_of_nodes, const int * node_list,
+ int abort)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_stop2");
+ const ParserRow<ParserDummy> stop_reply[] = {
+ MGM_CMD("stop reply", NULL, ""),
+ MGM_ARG("stopped", Int, Optional, "No of stopped nodes"),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ if(no_of_nodes < 0){
+ SET_ERROR(handle, NDB_MGM_ILLEGAL_NUMBER_OF_NODES,
+ "Negative number of nodes requested to stop");
+ return -1;
+ }
+
+ Uint32 stoppedNoOfNodes = 0;
+ if(no_of_nodes == 0){
+ /**
+ * All database nodes should be stopped
+ */
+ Properties args;
+ args.put("abort", abort);
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, stop_reply, "stop all", &args);
+ CHECK_REPLY(reply, -1);
+
+ if(!reply->get("stopped", &stoppedNoOfNodes)){
+ SET_ERROR(handle, NDB_MGM_STOP_FAILED,
+ "Could not get number of stopped nodes from mgm server");
+ delete reply;
+ return -1;
+ }
+ BaseString result;
+ reply->get("result", result);
+ if(strcmp(result.c_str(), "Ok") != 0) {
+ SET_ERROR(handle, NDB_MGM_STOP_FAILED, result.c_str());
+ delete reply;
+ return -1;
+ }
+ delete reply;
+ return stoppedNoOfNodes;
+ }
+
+ /**
+ * A list of database nodes should be stopped
+ */
+ Properties args;
+
+ BaseString node_list_str;
+ node_list_str.assfmt("%d", node_list[0]);
+ for(int node = 1; node < no_of_nodes; node++)
+ node_list_str.appfmt(" %d", node_list[node]);
+
+ args.put("node", node_list_str.c_str());
+ args.put("abort", abort);
+
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, stop_reply, "stop", &args);
+ CHECK_REPLY(reply, stoppedNoOfNodes);
+ if(!reply->get("stopped", &stoppedNoOfNodes)){
+ SET_ERROR(handle, NDB_MGM_STOP_FAILED,
+ "Could not get number of stopped nodes from mgm server");
+ delete reply;
+ return -1;
+ }
+ BaseString result;
+ reply->get("result", result);
+ if(strcmp(result.c_str(), "Ok") != 0) {
+ SET_ERROR(handle, NDB_MGM_STOP_FAILED, result.c_str());
+ delete reply;
+ return -1;
+ }
+ delete reply;
+ return stoppedNoOfNodes;
+}
+
+extern "C"
+int
+ndb_mgm_restart2(NdbMgmHandle handle, int no_of_nodes, const int * node_list,
+ int initial, int nostart, int abort)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_restart2");
+ Uint32 restarted = 0;
+ const ParserRow<ParserDummy> restart_reply[] = {
+ MGM_CMD("restart reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_ARG("restarted", Int, Optional, "No of restarted nodes"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ if(no_of_nodes < 0){
+ SET_ERROR(handle, NDB_MGM_RESTART_FAILED,
+ "Restart requested of negative number of nodes");
+ return -1;
+ }
+
+ if(no_of_nodes == 0) {
+ Properties args;
+ args.put("abort", abort);
+ args.put("initialstart", initial);
+ args.put("nostart", nostart);
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, restart_reply, "restart all", &args);
+ CHECK_REPLY(reply, -1);
+
+ BaseString result;
+ reply->get("result", result);
+ if(strcmp(result.c_str(), "Ok") != 0) {
+ SET_ERROR(handle, NDB_MGM_RESTART_FAILED, result.c_str());
+ delete reply;
+ return -1;
+ }
+ if(!reply->get("restarted", &restarted)){
+ SET_ERROR(handle, NDB_MGM_RESTART_FAILED,
+ "Could not get restarted number of nodes from mgm server");
+ delete reply;
+ return -1;
+ }
+ delete reply;
+ return restarted;
+ }
+
+ BaseString node_list_str;
+ node_list_str.assfmt("%d", node_list[0]);
+ for(int node = 1; node < no_of_nodes; node++)
+ node_list_str.appfmt(" %d", node_list[node]);
+
+ Properties args;
+
+ args.put("node", node_list_str.c_str());
+ args.put("abort", abort);
+ args.put("initialstart", initial);
+ args.put("nostart", nostart);
+
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, restart_reply, "restart node", &args);
+ if(reply != NULL) {
+ BaseString result;
+ reply->get("result", result);
+ if(strcmp(result.c_str(), "Ok") != 0) {
+ SET_ERROR(handle, NDB_MGM_RESTART_FAILED, result.c_str());
+ delete reply;
+ return -1;
+ }
+ reply->get("restarted", &restarted);
+ delete reply;
+ }
+
+ return restarted;
+}
+
+extern "C"
+int
+ndb_mgm_restart(NdbMgmHandle handle, int no_of_nodes, const int *node_list)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_restart");
+ return ndb_mgm_restart2(handle, no_of_nodes, node_list, 0, 0, 0);
+}
+
+extern "C"
+unsigned int *
+ndb_mgm_get_logfilter(NdbMgmHandle handle)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_get_logfilter");
+ static Uint32 enabled[7] = {0,0,0,0,0,0,0};
+ const ParserRow<ParserDummy> getinfo_reply[] = {
+ MGM_CMD("clusterlog", NULL, ""),
+ MGM_ARG("enabled", Int, Mandatory, ""),
+ MGM_ARG("debug", Int, Mandatory, ""),
+ MGM_ARG("info", Int, Mandatory, ""),
+ MGM_ARG("warning", Int, Mandatory, ""),
+ MGM_ARG("error", Int, Mandatory, ""),
+ MGM_ARG("critical", Int, Mandatory, ""),
+ MGM_ARG("alert", Int, Mandatory, ""),
+ };
+ CHECK_HANDLE(handle, NULL);
+ CHECK_CONNECTED(handle, NULL);
+
+ Properties args;
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, getinfo_reply, "get info clusterlog", &args);
+ CHECK_REPLY(reply, NULL);
+
+ const char *names[] = { "enabled", "debug", "info", "warning", "error",
+ "critical", "alert" };
+ for(int i=0; i < 7; i++) {
+ reply->get(names[i], &enabled[i]);
+ }
+ return enabled;
+}
+
+extern "C"
+int
+ndb_mgm_filter_clusterlog(NdbMgmHandle handle,
+ enum ndb_mgm_clusterlog_level level,
+ struct ndb_mgm_reply* /*reply*/)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_filter_clusterlog");
+ const ParserRow<ParserDummy> filter_reply[] = {
+ MGM_CMD("set logfilter reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ int retval = -1;
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("level", level);
+
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, filter_reply, "set logfilter", &args);
+ CHECK_REPLY(reply, retval);
+
+ BaseString result;
+ reply->get("result", result);
+ if(strcmp(result.c_str(), "1") == 0) {
+ retval = 0;
+ } else {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ retval = -1;
+ }
+ delete reply;
+ return retval;
+}
+
+struct ndb_mgm_event_categories
+{
+ const char* name;
+ enum ndb_mgm_event_category category;
+};
+
+extern "C"
+int
+ndb_mgm_set_loglevel_clusterlog(NdbMgmHandle handle, int nodeId,
+ /*enum ndb_mgm_event_category*/
+ char * category, int level,
+ struct ndb_mgm_reply* /*reply*/)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR,
+ "Executing: ndb_mgm_set_loglevel_clusterlog");
+ const ParserRow<ParserDummy> clusterlog_reply[] = {
+ MGM_CMD("set cluster loglevel reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("node", nodeId);
+ args.put("category", category);
+ args.put("level", level);
+
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, clusterlog_reply,
+ "set cluster loglevel", &args);
+ CHECK_REPLY(reply, -1);
+
+ BaseString result;
+ reply->get("result", result);
+ if(strcmp(result.c_str(), "Ok") != 0) {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ delete reply;
+ return -1;
+ }
+ delete reply;
+ return 0;
+}
+
+extern "C"
+int
+ndb_mgm_set_loglevel_node(NdbMgmHandle handle, int nodeId,
+ /*enum ndb_mgm_event_category category*/
+ char * category, int level,
+ struct ndb_mgm_reply* /*reply*/)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_set_loglevel_node");
+ const ParserRow<ParserDummy> loglevel_reply[] = {
+ MGM_CMD("set loglevel reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("node", nodeId);
+ args.put("category", category);
+ args.put("level", level);
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, loglevel_reply, "set loglevel", &args);
+ CHECK_REPLY(reply, -1);
+
+ BaseString result;
+ reply->get("result", result);
+ if(strcmp(result.c_str(), "Ok") != 0) {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ delete reply;
+ return -1;
+ }
+
+ delete reply;
+ return 0;
+}
+
+extern "C"
+int
+ndb_mgm_get_stat_port(NdbMgmHandle handle, struct ndb_mgm_reply* /*reply*/)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_get_stat_port");
+ const ParserRow<ParserDummy> stat_reply[] = {
+ MGM_CMD("error", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_CMD("get statport reply", NULL, ""),
+ MGM_ARG("tcpport", Int, Mandatory, "TCP port for statistics"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, stat_reply, "get statport", &args);
+ CHECK_REPLY(reply, -1);
+
+ Uint32 port;
+ reply->get("tcpport", &port);
+
+ delete reply;
+ return port;
+}
+
+extern "C"
+int
+ndb_mgm_dump_state(NdbMgmHandle handle, int nodeId, int* _args,
+ int _num_args, struct ndb_mgm_reply* /* reply */)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_dump_state");
+ const ParserRow<ParserDummy> dump_state_reply[] = {
+ MGM_CMD("dump state reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ char buf[256];
+ char buf2[6];
+ buf[0] = 0;
+ for (int i = 0; i < _num_args; i++){
+ snprintf(buf2, 6, "%d ", _args[i]);
+ strncat(buf, buf2, 256);
+ }
+
+ Properties args;
+ args.put("node", nodeId);
+ args.put("args", buf);
+
+ const Properties *prop;
+ prop = ndb_mgm_call(handle, dump_state_reply, "dump state", &args);
+ CHECK_REPLY(prop, -1);
+
+ BaseString result;
+ prop->get("result", result);
+ if(strcmp(result.c_str(), "Ok") != 0) {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ delete prop;
+ return -1;
+ }
+
+ delete prop;
+ return 0;
+}
+
+extern "C"
+int
+ndb_mgm_start_signallog(NdbMgmHandle handle, int nodeId,
+ struct ndb_mgm_reply* reply)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_start_signallog");
+ const ParserRow<ParserDummy> start_signallog_reply[] = {
+ MGM_CMD("start signallog reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ int retval = -1;
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("node", nodeId);
+
+ const Properties *prop;
+ prop = ndb_mgm_call(handle,
+ start_signallog_reply,
+ "start signallog",
+ &args);
+
+ if(prop != NULL) {
+ BaseString result;
+ prop->get("result", result);
+ if(strcmp(result.c_str(), "Ok") == 0) {
+ retval = 0;
+ } else {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ retval = -1;
+ }
+ delete prop;
+ }
+
+ return retval;
+}
+
+extern "C"
+int
+ndb_mgm_stop_signallog(NdbMgmHandle handle, int nodeId,
+ struct ndb_mgm_reply* reply)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_stop_signallog");
+ const ParserRow<ParserDummy> stop_signallog_reply[] = {
+ MGM_CMD("stop signallog reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ int retval = -1;
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("node", nodeId);
+
+ const Properties *prop;
+ prop = ndb_mgm_call(handle, stop_signallog_reply, "stop signallog", &args);
+
+ if(prop != NULL) {
+ BaseString result;
+ prop->get("result", result);
+ if(strcmp(result.c_str(), "Ok") == 0) {
+ retval = 0;
+ } else {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ retval = -1;
+ }
+ delete prop;
+ }
+
+ return retval;
+}
+
+struct ndb_mgm_signal_log_modes
+{
+ const char* name;
+ enum ndb_mgm_signal_log_mode mode;
+};
+
+extern "C"
+int
+ndb_mgm_log_signals(NdbMgmHandle handle, int nodeId,
+ enum ndb_mgm_signal_log_mode mode,
+ const char* blockNames,
+ struct ndb_mgm_reply* reply)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_log_signals");
+ const ParserRow<ParserDummy> stop_signallog_reply[] = {
+ MGM_CMD("log signals reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ int retval = -1;
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("node", nodeId);
+ args.put("blocks", blockNames);
+
+ switch(mode) {
+ case NDB_MGM_SIGNAL_LOG_MODE_IN:
+ args.put("in", (Uint32)1);
+ args.put("out", (Uint32)0);
+ break;
+ case NDB_MGM_SIGNAL_LOG_MODE_OUT:
+ args.put("in", (Uint32)0);
+ args.put("out", (Uint32)1);
+ break;
+ case NDB_MGM_SIGNAL_LOG_MODE_INOUT:
+ args.put("in", (Uint32)1);
+ args.put("out", (Uint32)1);
+ break;
+ case NDB_MGM_SIGNAL_LOG_MODE_OFF:
+ args.put("in", (Uint32)0);
+ args.put("out", (Uint32)0);
+ break;
+ }
+
+ const Properties *prop;
+ prop = ndb_mgm_call(handle, stop_signallog_reply, "log signals", &args);
+
+ if(prop != NULL) {
+ BaseString result;
+ prop->get("result", result);
+ if(strcmp(result.c_str(), "Ok") == 0) {
+ retval = 0;
+ } else {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ retval = -1;
+ }
+ delete prop;
+ }
+
+ return retval;
+}
+
+extern "C"
+int
+ndb_mgm_set_trace(NdbMgmHandle handle, int nodeId, int traceNumber,
+ struct ndb_mgm_reply* reply)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_set_trace");
+ const ParserRow<ParserDummy> set_trace_reply[] = {
+ MGM_CMD("set trace reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ int retval = -1;
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("node", nodeId);
+ args.put("trace", traceNumber);
+
+ const Properties *prop;
+ prop = ndb_mgm_call(handle, set_trace_reply, "set trace", &args);
+
+ if(prop != NULL) {
+ BaseString result;
+ prop->get("result", result);
+ if(strcmp(result.c_str(), "Ok") == 0) {
+ retval = 0;
+ } else {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ retval = -1;
+ }
+ delete prop;
+ }
+
+ return retval;
+}
+
+extern "C"
+int
+ndb_mgm_insert_error(NdbMgmHandle handle, int nodeId, int errorCode,
+ struct ndb_mgm_reply* reply)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_insert_error");
+ const ParserRow<ParserDummy> insert_error_reply[] = {
+ MGM_CMD("insert error reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ int retval = -1;
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("node", nodeId);
+ args.put("error", errorCode);
+
+ const Properties *prop;
+ prop = ndb_mgm_call(handle, insert_error_reply, "insert error", &args);
+
+ if(prop != NULL) {
+ BaseString result;
+ prop->get("result", result);
+ if(strcmp(result.c_str(), "Ok") == 0) {
+ retval = 0;
+ } else {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ retval = -1;
+ }
+ delete prop;
+ }
+
+ return retval;
+}
+
+extern "C"
+int
+ndb_mgm_start(NdbMgmHandle handle, int no_of_nodes, const int * node_list)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_start");
+ const ParserRow<ParserDummy> start_reply[] = {
+ MGM_CMD("start reply", NULL, ""),
+ MGM_ARG("started", Int, Optional, "No of started nodes"),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ int started = 0;
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ if(no_of_nodes < 0){
+ SET_ERROR(handle, EINVAL, "");
+ return -1;
+ }
+
+ if(no_of_nodes == 0){
+ Properties args;
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, start_reply, "start all", &args);
+ CHECK_REPLY(reply, -1);
+
+ Uint32 count = 0;
+ if(!reply->get("started", &count)){
+ delete reply;
+ return -1;
+ }
+ delete reply;
+ return count;
+ }
+
+ for(int node = 0; node < no_of_nodes; node++) {
+ Properties args;
+ args.put("node", node_list[node]);
+
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, start_reply, "start", &args);
+
+ if(reply != NULL) {
+ BaseString result;
+ reply->get("result", result);
+ if(strcmp(result.c_str(), "Ok") == 0) {
+ started++;
+ } else {
+ SET_ERROR(handle, EINVAL, result.c_str());
+ delete reply;
+ return -1;
+ }
+ }
+ delete reply;
+ }
+
+ return started;
+}
+
+/*****************************************************************************
+ * Backup
+ *****************************************************************************/
+extern "C"
+int
+ndb_mgm_start_backup(NdbMgmHandle handle, unsigned int* _backup_id,
+ struct ndb_mgm_reply* /*reply*/)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_start_backup");
+ const ParserRow<ParserDummy> start_backup_reply[] = {
+ MGM_CMD("start backup reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_ARG("id", Int, Optional, "Id of the started backup"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, start_backup_reply, "start backup", &args);
+ CHECK_REPLY(reply, -1);
+
+ BaseString result;
+ reply->get("result", result);
+ reply->get("id", _backup_id);
+ if(strcmp(result.c_str(), "Ok") != 0) {
+ SET_ERROR(handle, NDB_MGM_COULD_NOT_START_BACKUP, result.c_str());
+ delete reply;
+ return -1;
+ }
+
+ delete reply;
+ return 0;
+}
+
+extern "C"
+int
+ndb_mgm_abort_backup(NdbMgmHandle handle, unsigned int backupId,
+ struct ndb_mgm_reply* /*reply*/)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_abort_backup");
+ const ParserRow<ParserDummy> stop_backup_reply[] = {
+ MGM_CMD("abort backup reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("id", backupId);
+
+ const Properties *prop;
+ prop = ndb_mgm_call(handle, stop_backup_reply, "abort backup", &args);
+ CHECK_REPLY(prop, -1);
+
+ const char * buf;
+ prop->get("result", &buf);
+ if(strcmp(buf,"Ok")!=0) {
+ SET_ERROR(handle, NDB_MGM_COULD_NOT_ABORT_BACKUP, buf);
+ delete prop;
+ return -1;
+ }
+
+ delete prop;
+ return 0;
+}
+
+/*****************************************************************************
+ * Global Replication
+ *****************************************************************************/
+
+extern "C"
+int
+ndb_mgm_rep_command(NdbMgmHandle handle, unsigned int request,
+ unsigned int* replication_id,
+ struct ndb_mgm_reply* /*reply*/)
+{
+ SET_ERROR(handle, NDB_MGM_NO_ERROR, "Executing: ndb_mgm_rep_command");
+ const ParserRow<ParserDummy> replication_reply[] = {
+ MGM_CMD("global replication reply", NULL, ""),
+ MGM_ARG("result", String, Mandatory, "Error message"),
+ MGM_ARG("id", Int, Optional, "Id of global replication"),
+ MGM_END()
+ };
+ CHECK_HANDLE(handle, -1);
+ CHECK_CONNECTED(handle, -1);
+
+ Properties args;
+ args.put("request", request);
+ const Properties *reply;
+ reply = ndb_mgm_call(handle, replication_reply, "rep", &args);
+ CHECK_REPLY(reply, -1);
+
+ const char * result;
+ reply->get("result", &result);
+ reply->get("id", replication_id);
+ if(strcmp(result,"Ok")!=0) {
+ delete reply;
+ return -1;
+ }
+
+ delete reply;
+ return 0;
+}
diff --git a/ndb/src/mgmapi/test/Makefile b/ndb/src/mgmapi/test/Makefile
new file mode 100644
index 00000000000..c6d3efa6fcc
--- /dev/null
+++ b/ndb/src/mgmapi/test/Makefile
@@ -0,0 +1,13 @@
+include .defs.mk
+
+TYPE := mgmapiclient util
+
+BIN_TARGET := testMgmapi
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/src/common/mgmcommon)
+#-I$(NDB_TOP)/include/util -I$(NDB_TOP)/include/portlib
+
+# Source files of non-templated classes (.C files)
+SOURCES = keso.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/mgmapi/test/keso.c b/ndb/src/mgmapi/test/keso.c
new file mode 100644
index 00000000000..f4b192e3db8
--- /dev/null
+++ b/ndb/src/mgmapi/test/keso.c
@@ -0,0 +1,457 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <mgmapi.h>
+
+#ifdef VM_TRACE
+#include <mgmapi_debug.h>
+#endif
+
+#include <NdbOut.hpp>
+#include <NdbStdio.h>
+
+#include <stdlib.h>
+
+static int testConnect(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+static int testDisconnect(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+
+static int testStatus(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+#ifdef VM_TRACE
+static int testLogSignals(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+static int testStartSignalLog(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+static int testStopSignalLog(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+static int testSetTrace(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+static int testInsertError(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+static int testDumpState(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+#endif
+static int testFilterClusterLog(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+static int testSetLogLevelClusterLog(NdbMgmHandle h,
+ struct ndb_mgm_reply* reply);
+static int testSetLogLevelNode(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+static int testRestartNode(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+static int testGetStatPort(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+
+typedef int (*FUNC)(NdbMgmHandle h, struct ndb_mgm_reply* reply);
+
+struct test_case {
+ char name[255];
+ FUNC func;
+};
+
+struct test_case test_connect_disconnect[] = {
+ {"testConnect", &testConnect},
+ {"testDisconnect", &testDisconnect}
+};
+
+struct test_case tests[] = {
+ { "testStatus", &testStatus },
+ { "testFilterClusterLog", &testFilterClusterLog },
+ //{ "testSetLogLevelClusterLog", &testSetLogLevelClusterLog },
+ //{ "testSetLogLevelNode", &testSetLogLevelNode },
+ { "testRestartNode", &testRestartNode },
+ { "testGetStatPort", &testGetStatPort },
+#ifdef VM_TRACE
+ { "testLogSignals", &testLogSignals },
+ { "testStartSignalLog", &testStartSignalLog },
+ { "testStopSignalLog", &testStopSignalLog },
+ { "testSetTrace", &testSetTrace },
+ { "testDumpState", &testDumpState },
+ { "testInsertError", &testInsertError }
+#endif
+};
+
+static int no_of_tests = sizeof(tests) / sizeof(struct test_case);
+static int testFailed = 0;
+
+static const char * g_connect_string = "localhost:2200";
+
+int
+main(int argc, const char** argv){
+
+ struct ndb_mgm_reply reply;
+ int i = 0;
+ NdbMgmHandle h = NULL;
+
+ if(argc > 1)
+ g_connect_string = argv[1];
+
+ ndbout_c("Using connectstring: %s", g_connect_string);
+
+ for (i = 0; i < 2; i++) {
+ ndbout_c("-- %s --", test_connect_disconnect[i].name);
+ if (test_connect_disconnect[i].func(h, &reply) == 0) {
+ ndbout_c("-- Passed --");
+ } else {
+ testFailed++;
+ ndbout_c("-- Failed --");
+ }
+ }
+ ndbout_c("-- %d passed, %d failed --", (2 - testFailed), testFailed);
+
+
+ h = ndb_mgm_create_handle();
+ ndb_mgm_connect(h, g_connect_string);
+
+ for (i = 0; i < no_of_tests; i ++) {
+ ndbout_c("-- %s --", tests[i].name);
+ if (tests[i].func(h, &reply) == 0) {
+ ndbout_c("-- Passed --");
+ } else {
+ testFailed++;
+ ndbout_c("-- Failed --");
+ ndb_mgm_disconnect(h);
+ ndb_mgm_connect(h, g_connect_string);
+ }
+ }
+ ndbout_c("-- %d passed, %d failed --", (no_of_tests - testFailed),
+ testFailed);
+
+ ndb_mgm_disconnect(h);
+
+ return 0;
+}
+
+static
+int testConnect(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+ h = ndb_mgm_create_handle();
+ if (h != NULL) {
+ if (ndb_mgm_connect(h, g_connect_string) == -1) {
+ ndbout_c(g_connect_string);
+ /*ndbout_c("last_error: %d", h->last_error); */
+ return -1;
+ } else {
+ ndbout_c("Connected to localhost:37123");
+ }
+
+ } else {
+ ndbout_c("Unable to create a NdbMgmHandle...");
+ return -1;
+ }
+
+ return 0;
+}
+
+static
+int testDisconnect(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+ ndb_mgm_disconnect(h);
+
+ return 0;
+}
+
+static
+int testStatus(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+ int i = 0;
+ struct ndb_mgm_cluster_state* cluster = ndb_mgm_get_status(h);
+ if (cluster != NULL) {
+ ndbout_c("Number of nodes: %d", cluster->no_of_nodes);
+ for (i = 0; i < cluster->no_of_nodes; i++) {
+ struct ndb_mgm_node_state state = cluster->node_states[i];
+ ndbout_c("NodeId: %d (%s)-- %s", state.node_id,
+ ndb_mgm_get_node_type_string(state.node_type),
+ ndb_mgm_get_node_status_string(state.node_status));
+
+ }
+ free(cluster);
+ } else {
+ ndbout_c("Unable to get node status.");
+ return -1;
+ }
+ return 0;
+}
+
+
+#ifdef VM_TRACE
+
+static
+int testLogSignals(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+ int rc = 0;
+ int nodeId = 0;
+ struct ndb_mgm_cluster_state* cluster = ndb_mgm_get_status(h);
+ if (cluster != NULL) {
+ if (cluster->no_of_nodes != 0) {
+ nodeId = cluster->node_states[0].node_id;
+ }
+ free(cluster);
+ } else {
+ ndbout_c("Unable to get node status.");
+ return -1;
+ }
+
+ rc = ndb_mgm_log_signals(h, nodeId, NDB_MGM_SIGNAL_LOG_MODE_INOUT,
+ "CMVMI QMGR",
+ reply);
+ if (rc != 0) {
+ ndbout_c("rc = %d", reply->return_code);
+ }
+
+ ndbout_c("%s", reply->message);
+
+ return rc;
+}
+
+static
+int testStartSignalLog(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+ int rc = 0;
+ int nodeId = 0;
+ struct ndb_mgm_cluster_state* cluster = ndb_mgm_get_status(h);
+ if (cluster != NULL) {
+ if (cluster->no_of_nodes != 0) {
+ nodeId = cluster->node_states[0].node_id;
+ }
+ free(cluster);
+ } else {
+ ndbout_c("Unable to get node status.");
+ return -1;
+ }
+
+ rc = ndb_mgm_start_signallog(h, nodeId, reply);
+ if (rc != 0) {
+ ndbout_c("rc = %d", reply->return_code);
+ }
+
+ ndbout_c("%s", reply->message);
+
+ return rc;
+}
+
+static
+int testStopSignalLog(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+
+ int rc = 0;
+ int nodeId = 0;
+ struct ndb_mgm_cluster_state* cluster = ndb_mgm_get_status(h);
+ if (cluster != NULL) {
+ if (cluster->no_of_nodes != 0) {
+ nodeId = cluster->node_states[0].node_id;
+ }
+ free(cluster);
+ } else {
+ ndbout_c("Unable to get node status.");
+ return -1;
+ }
+
+ rc = ndb_mgm_stop_signallog(h, nodeId, reply);
+ if (rc != 0) {
+ ndbout_c("rc = %d", reply->return_code);
+ }
+
+ ndbout_c("%s", reply->message);
+
+ return rc;
+
+}
+
+static
+int testSetTrace(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+ int rc = 0;
+ int nodeId = 0;
+ struct ndb_mgm_cluster_state* cluster = ndb_mgm_get_status(h);
+ if (cluster != NULL) {
+ if (cluster->no_of_nodes != 0) {
+ nodeId = cluster->node_states[0].node_id;
+ }
+ free(cluster);
+ } else {
+ ndbout_c("Unable to get node status.");
+ return -1;
+ }
+
+ rc = ndb_mgm_set_trace(h, nodeId, 2, reply);
+ if (rc != 0) {
+ ndbout_c("rc = %d", reply->return_code);
+ }
+
+ ndbout_c("%s", reply->message);
+
+ return rc;
+
+}
+
+static
+int testInsertError(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+ int rc = 0;
+ int nodeId = 0;
+ struct ndb_mgm_cluster_state* cluster = ndb_mgm_get_status(h);
+ if (cluster != NULL) {
+ if (cluster->no_of_nodes != 0) {
+ nodeId = cluster->node_states[0].node_id;
+ }
+ free(cluster);
+ } else {
+ ndbout_c("Unable to get node status.");
+ return -1;
+ }
+
+ rc = ndb_mgm_insert_error(h, nodeId, 9999, reply);
+ if (rc != 0) {
+ ndbout_c("rc = %d", reply->return_code);
+ }
+
+ ndbout_c("%s", reply->message);
+
+ return rc;
+
+}
+
+static
+int testDumpState(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+
+ int rc = 0;
+ int nodeId = 0;
+ int dump[3];
+
+ struct ndb_mgm_cluster_state* cluster = ndb_mgm_get_status(h);
+ if (cluster != NULL) {
+ if (cluster->no_of_nodes != 0) {
+ nodeId = cluster->node_states[0].node_id;
+ }
+ free(cluster);
+ } else {
+ ndbout_c("Unable to get node status.");
+ return -1;
+ }
+
+ dump[0] = 1;
+ dump[1] = 2;
+ dump[2] = 3;
+ rc = ndb_mgm_dump_state(h, nodeId, dump, 3, reply);
+
+ if (rc != 0) {
+ ndbout_c("rc = %d", reply->return_code);
+ }
+
+ ndbout_c("%s", reply->message);
+
+ return rc;
+}
+
+#endif
+
+static
+int testFilterClusterLog(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+ int rc = 0;
+
+ rc = ndb_mgm_filter_clusterlog(h, NDB_MGM_CLUSTERLOG_INFO, reply);
+ if (rc == -1) {
+ ndbout_c("rc = %d", reply->return_code);
+ ndbout_c("%s", reply->message);
+ return -1;
+ }
+
+ ndbout_c("%s", reply->message);
+
+ rc = ndb_mgm_filter_clusterlog(h, NDB_MGM_CLUSTERLOG_DEBUG, reply);
+ if (rc == -1) {
+ ndbout_c("rc = %d", reply->return_code);
+ ndbout_c("%s", reply->message);
+ return -1;
+ }
+
+ ndbout_c("%s", reply->message);
+
+ return rc;
+
+}
+
+static
+int testSetLogLevelClusterLog(NdbMgmHandle h,
+ struct ndb_mgm_reply* reply) {
+ int rc = 0;
+ int nodeId = 0;
+ struct ndb_mgm_cluster_state* cluster = ndb_mgm_get_status(h);
+ if (cluster != NULL) {
+ if (cluster->no_of_nodes != 0) {
+ nodeId = cluster->node_states[0].node_id;
+ }
+ free(cluster);
+ } else {
+ ndbout_c("Unable to get node status.");
+ return -1;
+ }
+
+ rc = ndb_mgm_set_loglevel_clusterlog(h, nodeId,
+ NDB_MGM_EVENT_CATEGORY_CHECKPOINT,
+ 5,
+ reply);
+ if (rc != 0) {
+ ndbout_c("rc = %d", reply->return_code);
+ }
+
+ ndbout_c("%s", reply->message);
+
+ return rc;
+
+}
+
+static
+int testSetLogLevelNode(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+
+ int rc = 0;
+ int nodeId = 0;
+ struct ndb_mgm_cluster_state* cluster = ndb_mgm_get_status(h);
+ if (cluster != NULL) {
+ if (cluster->no_of_nodes != 0) {
+ nodeId = cluster->node_states[0].node_id;
+ }
+ free(cluster);
+ } else {
+ ndbout_c("Unable to get node status.");
+ return -1;
+ }
+
+ rc = ndb_mgm_set_loglevel_node(h, nodeId,
+ NDB_MGM_EVENT_CATEGORY_STATISTIC,
+ 15,
+ reply);
+ if (rc != 0) {
+ ndbout_c("rc = %d", reply->return_code);
+ }
+
+ ndbout_c("%s", reply->message);
+
+ return rc;
+
+}
+
+static int
+testRestartNode(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+ int restarts = 0;
+
+ restarts = ndb_mgm_restart(h, 0, 0); /* Restart all */
+ if (restarts == 0) {
+ ndbout_c("No nodes restarted...");
+ return -1;
+ } else {
+ ndbout_c("%d nodes restarted...", restarts);
+ }
+
+ return 0;
+}
+
+static
+int testGetStatPort(NdbMgmHandle h, struct ndb_mgm_reply* reply) {
+
+ int rc = 0;
+ rc = ndb_mgm_get_stat_port(h, reply);
+ if (rc == 0) {
+ ndbout_c("stat port %s", reply->message);
+ } else {
+ ndbout_c("failed");
+ }
+
+ return rc;
+}
diff --git a/ndb/src/mgmapi/test/mgmSrvApi.cpp b/ndb/src/mgmapi/test/mgmSrvApi.cpp
new file mode 100644
index 00000000000..e93c54ae5a7
--- /dev/null
+++ b/ndb/src/mgmapi/test/mgmSrvApi.cpp
@@ -0,0 +1,128 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/****************************************************
+ * Name:
+ * mgmSrvApi.cpp
+ *
+ * Description:
+ * Test the bahaviour of the Management Server
+ * API based on the tests specified in the
+ * "Test Specification for the Management
+ * Server API" document
+ *
+ *****************************************************/
+#include "mgmapi.h"
+#include "mgmapi_commands.h"
+#include <stdlib.h>
+#include <string.h>
+#include <stdio.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+
+/**
+ * The pupose of this test program is to
+ * verify that the Management Server
+ * API is functioning properly, i.e. a handle
+ * can be created/destroyed properly, the
+ * connection to the NDB nodes is established
+ * correctly, and all the errors are handled in
+ * a proper way.
+ * USE: mgmSrvApi -n -i
+ *
+ * @param n Number of nodes to crash
+ *
+ **/
+NDB_COMMAND(mgmSrvApi, "mgmSrvApi", "mgmSrvApi -n <Number of nodes to crash> -i <Node ID to be crashed> -p <IP address:port number>", "Management Server API testing", 65535){
+
+ const char *Addr = 0;
+ int i;
+ int nodesNo = 0; // Number of nodes to crash
+ int ndbID = 0; // Node ID to be crashed
+
+ i = 1;
+ while (argc > 1) {
+ if (strcmp(argv[i], "-n") == 0)
+ nodesNo = atoi(argv[i+1]);
+
+ if (strcmp(argv[i], "-i") == 0)
+ ndbID = atoi(argv[i+1]);
+
+ if (strcmp(argv[i], "-p") == 0)
+ Addr = argv[i+1];
+
+ argc -= 1;
+ i = i + 1;
+ }
+
+ /*
+ * Create a handle
+ */
+ ndbout << "Creating handle..." << endl;
+ NdbMgmHandle h = ndb_mgm_create_handle();
+
+ /*
+ * Perfom the connection
+ */
+ ndbout << "Connecting..." << endl;
+ if (ndb_mgm_connect(h, Addr) == -1) {
+ ndbout << "Connection to " << Addr << " failed" << endl;
+ exit(-1);
+ }
+
+ /*
+ * Get status of a node
+ */
+ ndbout << "Getting status..." << endl;
+
+ struct ndb_mgm_cluster_state * status;
+ struct ndb_mgm_node_state * nodes;
+
+ status = ndb_mgm_get_status(h);
+ nodes = status->node_states;
+ if (nodes->node_status == 1) {
+ ndbout << "No contact established" << endl;
+ // exit(-1);
+ }
+
+ /*
+ * Stop the NDB nodes
+ */
+ ndbout << "Stopping the node(s)" << endl;
+ const int * list;
+
+ if (nodesNo == 0) // all nodes stopped by definition
+ ndbID = 0;
+
+ list = &ndbID;
+ if (ndb_mgm_stop(h, nodesNo, list) != 1) {
+ ndbout << nodesNo << " NDB node(s) not stopped " << endl;
+ }
+
+ /*
+ * Disconnect from the management server
+ */
+ ndbout << "Disconnecting..." << endl;
+ ndb_mgm_disconnect(h);
+
+ /*
+ * Destroy the handle
+ */
+ ndbout << "Destroying the handle..." << endl;
+ ndb_mgm_destroy_handle(&h);
+
+ return 0;
+}
diff --git a/ndb/src/mgmclient/CommandInterpreter.cpp b/ndb/src/mgmclient/CommandInterpreter.cpp
new file mode 100644
index 00000000000..29d25ebf7d3
--- /dev/null
+++ b/ndb/src/mgmclient/CommandInterpreter.cpp
@@ -0,0 +1,2013 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <errno.h>
+#include <string.h>
+#include <ctype.h>
+#include <stdlib.h>
+#include <limits.h>
+
+#include <ndb_version.h>
+
+#include <mgmapi.h>
+#include <mgmapi_debug.h>
+#include <version.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <EventLogger.hpp>
+#include <signaldata/SetLogLevelOrd.hpp>
+#include <signaldata/GrepImpl.hpp>
+#ifdef HAVE_GLOBAL_REPLICATION
+
+#endif // HAVE_GLOBAL_REPLICATION
+#include "MgmtErrorReporter.hpp"
+#include "CommandInterpreter.hpp"
+#include "CpcClient.hpp"
+
+#ifdef NDB_SOLARIS // XXX fix me
+static char* strsep(char** x, const char* y) { return 0; }
+#endif
+
+
+/*****************************************************************************
+ * HELP
+ *****************************************************************************/
+static const char* helpText =
+"---------------------------------------------------------------------------\n"
+" NDB Cluster -- Management Client -- Help\n"
+"---------------------------------------------------------------------------\n"
+"HELP Print help text\n"
+"HELP SHOW Help for SHOW command\n"
+#ifdef HAVE_GLOBAL_REPLICATION
+"HELP REPLICATION Help for global replication\n"
+#endif // HAVE_GLOBAL_REPLICATION
+#ifdef VM_TRACE // DEBUG ONLY
+"HELP DEBUG Help for debug compiled version\n"
+#endif
+"SHOW Print information about cluster\n"
+"SHOW CONFIG Print configuration\n"
+"SHOW PARAMETERS Print configuration parameters\n"
+"START BACKUP Start backup\n"
+"ABORT BACKUP <backup id> Abort backup\n"
+"CLUSTERLOG ON Enable Cluster logging\n"
+"CLUSTERLOG OFF Disable Cluster logging\n"
+"CLUSTERLOG FILTER <severity> Toggle severity filter on/off\n"
+"CLUSTERLOG INFO Print cluster log information\n"
+"<id> START Start DB node (started with -n)\n"
+"<id> RESTART [-n] [-i] Restart DB node\n"
+"<id> STOP Stop DB node\n"
+"ENTER SINGLE USER MODE <api-node> Enter single user mode\n"
+"EXIT SINGLE USER MODE Exit single user mode\n"
+"<id> STATUS Print status\n"
+"<id> CLUSTERLOG {<category>=<level>}+ Set log level for cluster log\n"
+"REP CONNECT <host:port> Connect to REP server on host:port\n"
+"QUIT Quit management client\n"
+;
+
+static const char* helpTextShow =
+"---------------------------------------------------------------------------\n"
+" NDB Cluster -- Management Client -- Help for SHOW command\n"
+"---------------------------------------------------------------------------\n"
+"SHOW prints NDB Cluster information\n\n"
+"SHOW Print information about cluster\n"
+"SHOW CONFIG Print configuration (in initial config file format)\n"
+"SHOW PARAMETERS Print information about configuration parameters\n\n"
+;
+
+#ifdef HAVE_GLOBAL_REPLICATION
+static const char* helpTextRep =
+"---------------------------------------------------------------------------\n"
+" NDB Cluster -- Management Client -- Help for Global Replication\n"
+"---------------------------------------------------------------------------\n"
+"Commands should be executed on the standby NDB Cluster\n"
+"These features are in an experimental release state.\n"
+"\n"
+"Simple Commands:\n"
+"REP START Start Global Replication\n"
+"REP START REQUESTOR Start Global Replication Requestor\n"
+"REP STATUS Show Global Replication status\n"
+"REP STOP Stop Global Replication\n"
+"REP STOP REQUESTOR Stop Global Replication Requestor\n"
+"\n"
+"Advanced Commands:\n"
+"REP START <protocol> Starts protocol\n"
+"REP STOP <protocol> Stops protocol\n"
+"<protocol> = TRANSFER | APPLY | DELETE\n"
+"\n"
+#ifdef VM_TRACE // DEBUG ONLY
+"Debugging commands:\n"
+"REP DELETE Removes epochs stored in primary and standy systems\n"
+"REP DROP <tableid> Drop a table in SS identified by table id\n"
+"REP SLOWSTOP Stop Replication (Tries to synchonize with primary)\n"
+"REP FASTSTOP Stop Replication (Stops in consistent state)\n"
+"<component> = SUBSCRIPTION\n"
+" METALOG | METASCAN | DATALOG | DATASCAN\n"
+" REQUESTOR | TRANSFER | APPLY | DELETE\n"
+#endif
+;
+#endif // HAVE_GLOBAL_REPLICATION
+
+#ifdef VM_TRACE // DEBUG ONLY
+static const char* helpTextDebug =
+"---------------------------------------------------------------------------\n"
+" NDB Cluster -- Management Client -- Help for Debugging (Internal use only)\n"
+"---------------------------------------------------------------------------\n"
+"SHOW PROPERTIES Print config properties object\n"
+"<id> LOGLEVEL {<category>=<level>}+ Set log level\n"
+#ifdef ERROR_INSERT
+"<id> ERROR <errorNo> Inject error into NDB node\n"
+#endif
+"<id> TRACE <traceNo> Set trace number\n"
+"<id> LOG [BLOCK = {ALL|<block>+}] Set logging on in & out signals\n"
+"<id> LOGIN [BLOCK = {ALL|<block>+}] Set logging on in signals\n"
+"<id> LOGOUT [BLOCK = {ALL|<block>+}] Set logging on out signals\n"
+"<id> LOGOFF [BLOCK = {ALL|<block>+}] Unset signal logging\n"
+"<id> TESTON Start signal logging\n"
+"<id> TESTOFF Stop signal logging\n"
+"<id> SET <configParamName> <value> Update configuration variable\n"
+"<id> DUMP <arg> Dump system state to cluster.log\n"
+"<id> GETSTAT Print statistics\n"
+"\n"
+"<id> = ALL | Any database node id\n"
+;
+#endif
+
+static bool
+convert(const char* s, int& val) {
+
+ if (s == NULL)
+ return false;
+
+ if (strlen(s) == 0)
+ return false;
+
+ errno = 0;
+ char* p;
+ long v = strtol(s, &p, 10);
+ if (errno != 0)
+ return false;
+
+ if (p != &s[strlen(s)])
+ return false;
+
+ val = v;
+ return true;
+}
+
+/*
+ * Constructor
+ */
+CommandInterpreter::CommandInterpreter(const char *_host)
+{
+ m_mgmsrv = ndb_mgm_create_handle();
+ if(m_mgmsrv == NULL) {
+ ndbout_c("Cannot create handle to management server.");
+ printError();
+ }
+
+ connected = false;
+ try_reconnect = 0;
+
+ host = strdup(_host);
+#ifdef HAVE_GLOBAL_REPLICATION
+ rep_host = NULL;
+ m_repserver = NULL;
+ rep_connected = false;
+#endif
+}
+
+/*
+ * Destructor
+ */
+CommandInterpreter::~CommandInterpreter()
+{
+ connected = false;
+ ndb_mgm_destroy_handle(&m_mgmsrv);
+ free((char *)host);
+ host = NULL;
+}
+
+bool
+emptyString(const char* s)
+{
+ if (s == NULL) {
+ return true;
+ }
+
+ for (unsigned int i = 0; i < strlen(s); ++i) {
+ if (! isspace(s[i])) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+class AutoPtr
+{
+public:
+ AutoPtr(void * ptr) : m_ptr(ptr) {}
+ ~AutoPtr() { free(m_ptr);}
+private:
+ void * m_ptr;
+};
+
+
+void
+CommandInterpreter::printError()
+{
+ ndbout_c("* %5d: %s",
+ ndb_mgm_get_latest_error(m_mgmsrv),
+ ndb_mgm_get_latest_error_msg(m_mgmsrv));
+ ndbout_c("* %s", ndb_mgm_get_latest_error_desc(m_mgmsrv));
+}
+
+//*****************************************************************************
+//*****************************************************************************
+
+bool
+CommandInterpreter::connect()
+{
+ if(!connected) {
+ int tries = try_reconnect; // tries == 0 => infinite
+ while(!connected) {
+ if(ndb_mgm_connect(m_mgmsrv, host) == -1) {
+ ndbout << "Cannot connect to management server (" << host << ").";
+ tries--;
+ if (tries == 0)
+ break;
+ ndbout << "Retrying in 5 seconds." << endl;
+ NdbSleep_SecSleep(5);
+ } else
+ connected = true;
+ }
+ }
+ return connected;
+}
+
+bool
+CommandInterpreter::disconnect()
+{
+ if (ndb_mgm_disconnect(m_mgmsrv) == -1) {
+ ndbout_c("Could not disconnect from management server");
+ printError();
+ }
+ connected = false;
+ return true;
+}
+
+//*****************************************************************************
+//*****************************************************************************
+
+int
+CommandInterpreter::readAndExecute(int _try_reconnect)
+{
+ if (_try_reconnect >= 0)
+ try_reconnect=_try_reconnect;
+
+ char* _line = readline_gets();
+ char * line;
+ if(_line == NULL) {
+ // ndbout << endl;
+ return false;
+ }
+
+ line = strdup(_line);
+
+ AutoPtr ptr(line);
+
+ if (emptyString(line)) {
+ return true;
+ }
+
+ for (unsigned int i = 0; i < strlen(line); ++i) {
+ line[i] = toupper(line[i]);
+ }
+
+ // if there is anything in the line proceed
+ char* firstToken = strtok(line, " ");
+ char* allAfterFirstToken = strtok(NULL, "");
+
+ if (strcmp(firstToken, "HELP") == 0 ||
+ strcmp(firstToken, "?") == 0) {
+ executeHelp(allAfterFirstToken);
+ return true;
+ }
+ else if (strcmp(firstToken, "SHOW") == 0) {
+ executeShow(allAfterFirstToken);
+ return true;
+ }
+ else if (strcmp(firstToken, "CLUSTERLOG") == 0){
+ executeClusterLog(allAfterFirstToken);
+ return true;
+ }
+ else if(strcmp(firstToken, "START") == 0 &&
+ allAfterFirstToken != NULL &&
+ strncmp(allAfterFirstToken, "BACKUP", sizeof("BACKUP") - 1) == 0){
+ executeStartBackup(allAfterFirstToken);
+ return true;
+ }
+ else if(strcmp(firstToken, "ABORT") == 0 &&
+ allAfterFirstToken != NULL &&
+ strncmp(allAfterFirstToken, "BACKUP", sizeof("BACKUP") - 1) == 0){
+ executeAbortBackup(allAfterFirstToken);
+ return true;
+ }
+#ifdef HAVE_GLOBAL_REPLICATION
+ else if(strcmp(firstToken, "REPLICATION") == 0 ||
+ strcmp(firstToken, "REP") == 0) {
+ executeRep(allAfterFirstToken);
+ return true;
+ }
+#endif // HAVE_GLOBAL_REPLICATION
+ else if(strcmp(firstToken, "ENTER") == 0 &&
+ allAfterFirstToken != NULL &&
+ strncmp(allAfterFirstToken, "SINGLE USER MODE ",
+ sizeof("SINGLE USER MODE") - 1) == 0){
+ executeEnterSingleUser(allAfterFirstToken);
+ return true;
+ }
+ else if(strcmp(firstToken, "EXIT") == 0 &&
+ allAfterFirstToken != NULL &&
+ strncmp(allAfterFirstToken, "SINGLE USER MODE ",
+ sizeof("SINGLE USER MODE") - 1) == 0){
+ executeExitSingleUser(allAfterFirstToken);
+ return true;
+ }
+ else if (strcmp(firstToken, "ALL") == 0) {
+ analyseAfterFirstToken(-1, allAfterFirstToken);
+ }
+ else if((strcmp(firstToken, "QUIT") == 0 ||
+ strcmp(firstToken, "EXIT") == 0 ||
+ strcmp(firstToken, "BYE") == 0) &&
+ allAfterFirstToken == NULL){
+ return false;
+#if 0
+ } else if(strcmp(firstToken, "CPC") == 0) {
+ executeCpc(allAfterFirstToken);
+#endif
+ } else {
+ /**
+ * First token should be a digit, node ID
+ */
+ int nodeId;
+
+ if (! convert(firstToken, nodeId)) {
+ ndbout << "Invalid command: " << line << endl;
+ ndbout << "Type HELP for help." << endl << endl;
+ return true;
+ }
+
+ if (nodeId < 0) {
+ ndbout << "Invalid node ID: " << firstToken << "." << endl;
+ return true;
+ }
+
+ analyseAfterFirstToken(nodeId, allAfterFirstToken);
+
+ }
+ return true;
+}
+
+
+/**
+ * List of commands used as second command argument
+ */
+static const CommandInterpreter::CommandFunctionPair commands[] = {
+ { "START", &CommandInterpreter::executeStart }
+ ,{ "RESTART", &CommandInterpreter::executeRestart }
+ ,{ "STOP", &CommandInterpreter::executeStop }
+ ,{ "STATUS", &CommandInterpreter::executeStatus }
+ ,{ "LOGLEVEL", &CommandInterpreter::executeLogLevel }
+ ,{ "CLUSTERLOG", &CommandInterpreter::executeEventReporting }
+#ifdef ERROR_INSERT
+ ,{ "ERROR", &CommandInterpreter::executeError }
+#endif
+ ,{ "TRACE", &CommandInterpreter::executeTrace }
+ ,{ "LOG", &CommandInterpreter::executeLog }
+ ,{ "LOGIN", &CommandInterpreter::executeLogIn }
+ ,{ "LOGOUT", &CommandInterpreter::executeLogOut }
+ ,{ "LOGOFF", &CommandInterpreter::executeLogOff }
+ ,{ "TESTON", &CommandInterpreter::executeTestOn }
+ ,{ "TESTOFF", &CommandInterpreter::executeTestOff }
+ ,{ "SET", &CommandInterpreter::executeSet }
+ ,{ "GETSTAT", &CommandInterpreter::executeGetStat }
+ ,{ "DUMP", &CommandInterpreter::executeDumpState }
+};
+
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::analyseAfterFirstToken(int processId,
+ char* allAfterFirstToken) {
+
+ if (emptyString(allAfterFirstToken)) {
+ if (processId == -1) {
+ ndbout << "Expected a command after ALL." << endl;
+ }
+ else {
+ ndbout << "Expected a command after node ID." << endl;
+ }
+ return;
+ }
+
+ char* secondToken = strtok(allAfterFirstToken, " ");
+ char* allAfterSecondToken = strtok(NULL, "\0");
+
+ const int tmpSize = sizeof(commands)/sizeof(CommandFunctionPair);
+ ExecuteFunction fun = 0;
+ const char * command = 0;
+ for(int i = 0; i<tmpSize; i++){
+ if(strcmp(secondToken, commands[i].command) == 0){
+ fun = commands[i].executeFunction;
+ command = commands[i].command;
+ break;
+ }
+ }
+
+ if(fun == 0){
+ ndbout << "Invalid command: " << secondToken << endl;
+ ndbout << "Type HELP for help." << endl << endl;
+ return;
+ }
+
+ if(processId == -1){
+ executeForAll(command, fun, allAfterSecondToken);
+ } else {
+ if(strcmp(command, "STATUS") != 0)
+ ndbout_c("Executing %s on node %d.", command, processId);
+ (this->*fun)(processId, allAfterSecondToken, false);
+ ndbout << endl;
+ }
+}
+
+/**
+ * Get next nodeid larger than the give node_id. node_id will be
+ * set to the next node_id in the list. node_id should be set
+ * to 0 (zero) on the first call.
+ *
+ * @param handle the NDB management handle
+ * @param node_id last node_id retreived, 0 at first call
+ * @param type type of node to look for
+ * @return 1 if a node was found, 0 if no more node exist
+ */
+static
+int
+get_next_nodeid(struct ndb_mgm_cluster_state *cl,
+ int *node_id,
+ enum ndb_mgm_node_type type)
+{
+ int i;
+
+ if(cl == NULL)
+ return 0;
+
+ i=0;
+ while((i < cl->no_of_nodes)) {
+ if((*node_id < cl->node_states[i].node_id) &&
+ (cl->node_states[i].node_type == type)) {
+
+ if(i >= cl->no_of_nodes)
+ return 0;
+
+ *node_id = cl->node_states[i].node_id;
+ return 1;
+ }
+ i++;
+ }
+
+ return 0;
+}
+
+void
+CommandInterpreter::executeForAll(const char * cmd, ExecuteFunction fun,
+ const char * allAfterSecondToken)
+{
+ int nodeId = 0;
+ if(strcmp(cmd, "STOP") == 0) {
+ ndbout_c("Executing STOP on all nodes.");
+ (this->*fun)(nodeId, allAfterSecondToken, true);
+ } else if(strcmp(cmd, "RESTART") == 0) {
+ ndbout_c("Executing RESTART on all nodes.");
+ ndbout_c("Starting shutdown. This may take a while. Please wait...");
+ (this->*fun)(nodeId, allAfterSecondToken, true);
+ ndbout_c("Trying to start all nodes of system.");
+ ndbout_c("Use ALL STATUS to see the system start-up phases.");
+ } else {
+ connect();
+ struct ndb_mgm_cluster_state *cl;
+ cl = ndb_mgm_get_status(m_mgmsrv);
+ if(cl == 0){
+ ndbout_c("Unable get status from management server");
+ printError();
+ return;
+ }
+ while(get_next_nodeid(cl, &nodeId, NDB_MGM_NODE_TYPE_NDB)) {
+ if(strcmp(cmd, "STATUS") != 0)
+ ndbout_c("Executing %s on node %d.", cmd, nodeId);
+ (this->*fun)(nodeId, allAfterSecondToken, true);
+ ndbout << endl;
+ } // while
+ free(cl);
+ }
+}
+
+//*****************************************************************************
+//*****************************************************************************
+bool
+CommandInterpreter::parseBlockSpecification(const char* allAfterLog,
+ Vector<const char*>& blocks)
+{
+ // Parse: [BLOCK = {ALL|<blockName>+}]
+
+ if (emptyString(allAfterLog)) {
+ return true;
+ }
+
+ // Copy allAfterLog since strtok will modify it
+ char* newAllAfterLog = strdup(allAfterLog);
+ char* firstTokenAfterLog = strtok(newAllAfterLog, " ");
+ for (unsigned int i = 0; i < strlen(firstTokenAfterLog); ++i) {
+ firstTokenAfterLog[i] = toupper(firstTokenAfterLog[i]);
+ }
+
+ if (strcmp(firstTokenAfterLog, "BLOCK") != 0) {
+ ndbout << "Unexpected value: " << firstTokenAfterLog
+ << ". Expected BLOCK." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+
+ char* allAfterFirstToken = strtok(NULL, "\0");
+ if (emptyString(allAfterFirstToken)) {
+ ndbout << "Expected =." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+
+ char* secondTokenAfterLog = strtok(allAfterFirstToken, " ");
+ if (strcmp(secondTokenAfterLog, "=") != 0) {
+ ndbout << "Unexpected value: " << secondTokenAfterLog
+ << ". Expected =." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+
+ char* blockName = strtok(NULL, " ");
+ bool all = false;
+ if (blockName != NULL && (strcmp(blockName, "ALL") == 0)) {
+ all = true;
+ }
+ while (blockName != NULL) {
+ blocks.push_back(strdup(blockName));
+ blockName = strtok(NULL, " ");
+ }
+
+ if (blocks.size() == 0) {
+ ndbout << "No block specified." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+ if (blocks.size() > 1 && all) {
+ // More than "ALL" specified
+ ndbout << "Nothing expected after ALL." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+
+ free(newAllAfterLog);
+ return true;
+}
+
+
+
+/*****************************************************************************
+ * HELP
+ *****************************************************************************/
+void
+CommandInterpreter::executeHelp(char* parameters)
+{
+ if (emptyString(parameters)) {
+ ndbout << helpText;
+
+ ndbout << endl
+ << "<severity> = "
+ << "ALERT | CRITICAL | ERROR | WARNING | INFO | DEBUG"
+ << endl;
+
+ ndbout << "<category> = ";
+ for(Uint32 i = 0; i<EventLogger::noOfEventCategoryNames; i++){
+ ndbout << EventLogger::eventCategoryNames[i].name;
+ if (i < EventLogger::noOfEventCategoryNames - 1) {
+ ndbout << " | ";
+ }
+ }
+ ndbout << endl;
+
+ ndbout << "<level> = " << "0 - 15" << endl;
+ ndbout << "<id> = " << "ALL | Any database node id" << endl;
+ ndbout << endl;
+ } else if (strcmp(parameters, "SHOW") == 0) {
+ ndbout << helpTextShow;
+#ifdef HAVE_GLOBAL_REPLICATION
+ } else if (strcmp(parameters, "REPLICATION") == 0 ||
+ strcmp(parameters, "REP") == 0) {
+ ndbout << helpTextRep;
+#endif // HAVE_GLOBAL_REPLICATION
+#ifdef VM_TRACE // DEBUG ONLY
+ } else if (strcmp(parameters, "DEBUG") == 0) {
+ ndbout << helpTextDebug;
+#endif
+ } else {
+ ndbout << "Invalid argument: " << parameters << endl;
+ ndbout << "Type HELP for help." << endl << endl;
+ }
+}
+
+
+/*****************************************************************************
+ * SHOW
+ *****************************************************************************/
+
+void
+CommandInterpreter::executeShow(char* parameters)
+{
+ connect();
+ if (emptyString(parameters)) {
+ ndbout << "Cluster Configuration" << endl
+ << "---------------------" << endl;
+
+ ndb_mgm_cluster_state *state = ndb_mgm_get_status(m_mgmsrv);
+ if(state == NULL) {
+ ndbout_c("Could not get status");
+ printError();
+ return;
+ }
+
+ int
+ ndb_nodes = 0,
+ api_nodes = 0,
+ mgm_nodes = 0;
+
+ for(int i=0; i < state->no_of_nodes; i++) {
+ switch(state->node_states[i].node_type) {
+ case NDB_MGM_NODE_TYPE_API:
+ api_nodes++;
+ break;
+ case NDB_MGM_NODE_TYPE_NDB:
+ ndb_nodes++;
+ break;
+ case NDB_MGM_NODE_TYPE_MGM:
+ mgm_nodes++;
+ break;
+ case NDB_MGM_NODE_TYPE_UNKNOWN:
+ ndbout << "Error: Unknown Node Type" << endl;
+ return;
+ }
+ }
+
+ ndbout << ndb_nodes
+ << " NDB Node(s)"
+ << endl;
+
+ for(int i=0; i < state->no_of_nodes; i++) {
+ if(state->node_states[i].node_type == NDB_MGM_NODE_TYPE_NDB) {
+ ndbout << "DB node:\t" << state->node_states[i].node_id;
+ if(state->node_states[i].version != 0) {
+ ndbout << " (Version: "
+ << getMajor(state->node_states[i].version) << "."
+ << getMinor(state->node_states[i].version) << "."
+ << getBuild(state->node_states[i].version) << ")" << endl;
+
+ } else
+ {
+ ndbout << " (not connected) " << endl;
+ }
+
+ }
+ }
+ ndbout << endl;
+
+ ndbout << api_nodes
+ << " API Node(s)"
+ << endl;
+
+ for(int i=0; i < state->no_of_nodes; i++) {
+ if(state->node_states[i].node_type == NDB_MGM_NODE_TYPE_API) {
+ ndbout << "API node:\t" << state->node_states[i].node_id;
+ if(state->node_states[i].version != 0) {
+ ndbout << " (Version: "
+ << getMajor(state->node_states[i].version) << "."
+ << getMinor(state->node_states[i].version) << "."
+ << getBuild(state->node_states[i].version) << ")" << endl;
+
+ } else
+ {
+ ndbout << " (not connected) " << endl;
+ }
+ }
+ }
+ ndbout << endl;
+
+ ndbout << mgm_nodes
+ << " MGM Node(s)"
+ << endl;
+
+ for(int i=0; i < state->no_of_nodes; i++) {
+ if(state->node_states[i].node_type == NDB_MGM_NODE_TYPE_MGM) {
+ ndbout << "MGM node:\t" << state->node_states[i].node_id;
+ if(state->node_states[i].version != 0) {
+ ndbout << " (Version: "
+ << getMajor(state->node_states[i].version) << "."
+ << getMinor(state->node_states[i].version) << "."
+ << getBuild(state->node_states[i].version) << ")" << endl;
+
+ } else
+ {
+ ndbout << " (no version information available) " << endl;
+ }
+ }
+ }
+ ndbout << endl;
+ // ndbout << helpTextShow;
+ return;
+ } else if (strcmp(parameters, "PROPERTIES") == 0 ||
+ strcmp(parameters, "PROP") == 0) {
+ ndbout << "SHOW PROPERTIES is not yet implemented." << endl;
+ // ndbout << "_mgmtSrvr.getConfig()->print();" << endl; /* XXX */
+ } else if (strcmp(parameters, "CONFIGURATION") == 0 ||
+ strcmp(parameters, "CONFIG") == 0){
+ ndbout << "SHOW CONFIGURATION is not yet implemented." << endl;
+ //nbout << "_mgmtSrvr.getConfig()->printConfigFile();" << endl; /* XXX */
+ } else if (strcmp(parameters, "PARAMETERS") == 0 ||
+ strcmp(parameters, "PARAMS") == 0 ||
+ strcmp(parameters, "PARAM") == 0) {
+ ndbout << "SHOW PARAMETERS is not yet implemented." << endl;
+ // ndbout << "_mgmtSrvr.getConfig()->getConfigInfo()->print();"
+ // << endl; /* XXX */
+ } else {
+ ndbout << "Invalid argument." << endl;
+ }
+}
+
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::executeClusterLog(char* parameters)
+{
+ connect();
+ if (parameters != 0 && strlen(parameters) != 0) {
+ enum ndb_mgm_clusterlog_level severity = NDB_MGM_CLUSTERLOG_ALL;
+ int isOk = true;
+ char name[12];
+ bool noArgs = false;
+
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, " ", &tmpPtr);
+
+ /********************
+ * CLUSTERLOG FILTER
+ ********************/
+ if (strcmp(item, "FILTER") == 0) {
+
+ item = strtok_r(NULL, " ", &tmpPtr);
+ if (item == NULL) {
+ noArgs = true;
+ }
+ while (item != NULL) {
+ snprintf(name, sizeof(name), item);
+
+ if (strcmp(item, "ALL") == 0) {
+ severity = NDB_MGM_CLUSTERLOG_ALL;
+ } else if (strcmp(item, "ALERT") == 0) {
+ severity = NDB_MGM_CLUSTERLOG_ALERT;
+ } else if (strcmp(item, "CRITICAL") == 0) {
+ severity = NDB_MGM_CLUSTERLOG_CRITICAL;
+ } else if (strcmp(item, "ERROR") == 0) {
+ severity = NDB_MGM_CLUSTERLOG_ERROR;
+ } else if (strcmp(item, "WARNING") == 0) {
+ severity = NDB_MGM_CLUSTERLOG_WARNING;
+ } else if (strcmp(item, "INFO") == 0) {
+ severity = NDB_MGM_CLUSTERLOG_INFO;
+ } else if (strcmp(item, "DEBUG") == 0) {
+ severity = NDB_MGM_CLUSTERLOG_DEBUG;
+ } else if (strcmp(item, "OFF") == 0) {
+ severity = NDB_MGM_CLUSTERLOG_OFF;
+ } else {
+ isOk = false;
+ }
+
+ item = strtok_r(NULL, " ", &tmpPtr);
+ } // while(item != NULL){
+ free(tmpString);
+
+ if (noArgs) {
+ ndbout << "Missing argument(s)." << endl;
+ } else if (isOk) {
+ if(ndb_mgm_filter_clusterlog(m_mgmsrv, severity, NULL)) {
+ if(strcmp(name, "ALL") == 0 || strcmp(name, "all") == 0) {
+ ndbout << "All severities levels enabled." << endl;
+ } else if(strcmp(name, "OFF") == 0 || strcmp(name, "off") == 0) {
+ ndbout << "Cluster logging enabled." << endl;
+ } else {
+ ndbout << name << " events disabled." << endl;
+ }
+ } else {
+ if(strcmp(name, "ALL") == 0) {
+ ndbout << "All severities levels disabled." << endl;
+ } else if(strcmp(name, "OFF") == 0) {
+ ndbout << "Cluster logging disabled." << endl;
+ } else {
+ ndbout << name << " events enabled." << endl;
+ }
+ }
+ } else {
+ ndbout << "Invalid severity level." << endl;
+ }
+
+ /********************
+ * CLUSTERLOG INFO
+ ********************/
+ } else if (strcmp(item, "INFO") == 0) {
+ Uint32 *enabled = ndb_mgm_get_logfilter(m_mgmsrv);
+ if(enabled == NULL) {
+ ndbout << "Couldn't get status" << endl;
+ printError();
+ return;
+ }
+ const char* names[] = {"ENABLED", "DEBUG", "INFO", "WARNING", "ERROR",
+ "CRITICAL", "ALERT"};
+ if(enabled[0])
+ ndbout << "Cluster logging is disabled." << endl;
+
+
+ for(int i = 0; i<7;i++)
+ printf("enabled[%d] = %d\n", i, enabled[i]);
+ ndbout << "Severities enabled: ";
+ for(int i = 1; i < 7; i++) {
+ if(enabled[i])
+ ndbout << names[i] << " ";
+ }
+ ndbout << endl;
+
+ /********************
+ * CLUSTERLOG OFF
+ ********************/
+ } else if (strcmp(item, "OFF") == 0) {
+ Uint32 *enabled = ndb_mgm_get_logfilter(m_mgmsrv);
+ if(enabled == NULL) {
+ ndbout << "Couldn't get status" << endl;
+ printError();
+ return;
+ }
+ if(!enabled[0]) {
+ ndb_mgm_filter_clusterlog(m_mgmsrv, NDB_MGM_CLUSTERLOG_OFF, NULL);
+ ndbout << "Cluster logging is disabled." << endl;
+ } else {
+ ndbout << "Cluster logging is already disabled." << endl;
+
+ }
+
+ /********************
+ * CLUSTERLOG ON
+ ********************/
+ } else if (strcmp(item, "ON") == 0) {
+ Uint32 *enabled = ndb_mgm_get_logfilter(m_mgmsrv);
+ if(enabled == NULL) {
+ ndbout << "Could not get status" << endl;
+ printError();
+ return;
+ }
+ if(enabled[0]) {
+ ndb_mgm_filter_clusterlog(m_mgmsrv, NDB_MGM_CLUSTERLOG_OFF, NULL);
+ ndbout << "Cluster logging is enabled." << endl;
+ } else {
+ ndbout << "Cluster logging is already enabled." << endl;
+
+ }
+ } else {
+ ndbout << "Invalid argument." << endl;
+ }
+
+ } else {
+ ndbout << "Missing argument." << endl;
+ }
+}
+
+//*****************************************************************************
+//*****************************************************************************
+
+void
+CommandInterpreter::executeStop(int processId, const char *, bool all)
+{
+ connect();
+ int result = 0;
+ if(all) {
+ result = ndb_mgm_stop(m_mgmsrv, 0, 0);
+ } else {
+ result = ndb_mgm_stop(m_mgmsrv, 1, &processId);
+ }
+ if (result <= 0) {
+ ndbout << "Shutdown failed." << endl;
+ printError();
+ } else
+ {
+ if(all)
+ ndbout << "NDB Cluster has shutdown." << endl;
+ else
+ ndbout << "Node " << processId << " has shutdown." << endl;
+ }
+}
+
+void
+CommandInterpreter::executeEnterSingleUser(char* parameters)
+{
+ connect();
+ strtok(parameters, " ");
+ struct ndb_mgm_reply reply;
+ char* id = strtok(NULL, " ");
+ id = strtok(NULL, " ");
+ id = strtok(NULL, "\0");
+ int nodeId = -1;
+ if(id == 0 || sscanf(id, "%d", &nodeId) != 1){
+ ndbout_c("Invalid arguments: expected <NodeId>");
+ ndbout_c("Use SHOW to see what API nodes are configured");
+ return;
+ }
+ int result = ndb_mgm_enter_single_user(m_mgmsrv, nodeId, &reply);
+
+ if (result != 0) {
+ ndbout_c("Entering single user mode for node %d failed", nodeId);
+ printError();
+ } else {
+ ndbout_c("Entering single user mode");
+ ndbout_c("Access will be granted for API node %d only.", nodeId);
+ ndbout_c("Use ALL STATUS to see when single user mode has been entered.");
+ }
+}
+
+void
+CommandInterpreter::executeExitSingleUser(char* parameters)
+{
+ connect();
+ int result = ndb_mgm_exit_single_user(m_mgmsrv, 0);
+ if (result != 0) {
+ ndbout_c("Exiting single user mode failed.");
+ printError();
+ } else {
+ ndbout_c("Exiting single user mode in progress.");
+ ndbout_c("Use ALL STATUS to see when single user mode has been exited.");
+ }
+}
+
+void
+CommandInterpreter::executeStart(int processId, const char* parameters,
+ bool all)
+{
+ connect();
+ int result;
+ if(all) {
+ result = ndb_mgm_start(m_mgmsrv, 0, 0);
+ } else {
+ result = ndb_mgm_start(m_mgmsrv, 1, &processId);
+ }
+
+ if (result <= 0) {
+ ndbout << "Start failed." << endl;
+ printError();
+ } else
+ {
+ if(all)
+ ndbout_c("NDB Cluster is being started.");
+ else
+ ndbout_c("Database node %d is being started.", processId);
+ }
+}
+
+void
+CommandInterpreter::executeRestart(int processId, const char* parameters,
+ bool all)
+{
+ connect();
+ int result;
+ int nostart = 0;
+ int initialstart = 0;
+ int abort = 0;
+
+ if(parameters != 0 && strlen(parameters) != 0){
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, " ", &tmpPtr);
+ while(item != NULL){
+ if(strcmp(item, "-N") == 0)
+ nostart = 1;
+ if(strcmp(item, "-I") == 0)
+ initialstart = 1;
+ if(strcmp(item, "-A") == 0)
+ abort = 1;
+ item = strtok_r(NULL, " ", &tmpPtr);
+ }
+ free(tmpString);
+ }
+
+ if(all) {
+ result = ndb_mgm_restart2(m_mgmsrv, 0, NULL, initialstart, nostart, abort);
+ } else {
+ int v[1];
+ v[0] = processId;
+ result = ndb_mgm_restart2(m_mgmsrv, 1, v, initialstart, nostart, abort);
+ }
+
+ if (result <= 0) {
+ ndbout.println("Restart failed.", result);
+ printError();
+ } else
+ {
+ if(all)
+ ndbout << "NDB Cluster is being restarted." << endl;
+ else
+ ndbout_c("Database node %d is being restarted.", processId);
+ }
+}
+
+void
+CommandInterpreter::executeDumpState(int processId, const char* parameters,
+ bool all)
+{
+ if(parameters == 0 || strlen(parameters) == 0){
+ ndbout << "Expected argument" << endl;
+ return;
+ }
+ connect();
+
+ Uint32 no = 0;
+ int pars[25];
+
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, " ", &tmpPtr);
+ while(item != NULL){
+ if (0x0 <= strtoll(item, NULL, 0) && strtoll(item, NULL, 0) <= 0xffffffff){
+ pars[no] = strtoll(item, NULL, 0);
+ } else {
+ ndbout << "Illegal value in argument to signal." << endl
+ << "(Value must be between 0 and 0xffffffff.)"
+ << endl;
+ return;
+ }
+ no++;
+ item = strtok_r(NULL, " ", &tmpPtr);
+ }
+ ndbout << "Sending dump signal with data:" << endl;
+ for (Uint32 i=0; i<no; i++) {
+ ndbout.setHexFormat(1) << pars[i] << " ";
+ if (!(i+1 & 0x3)) ndbout << endl;
+ }
+ free(tmpString);
+
+ struct ndb_mgm_reply reply;
+ ndb_mgm_dump_state(m_mgmsrv, processId, pars, no, &reply);
+}
+
+void
+CommandInterpreter::executeStatus(int processId,
+ const char* parameters, bool all)
+{
+ if (! emptyString(parameters)) {
+ ndbout << "No parameters expected to this command." << endl;
+ return;
+ }
+
+ connect();
+ ndb_mgm_node_status status;
+ Uint32 startPhase, version;
+ bool system;
+
+ struct ndb_mgm_cluster_state *cl;
+ cl = ndb_mgm_get_status(m_mgmsrv);
+ if(cl == NULL) {
+ ndbout_c("Cannot get status of node %d.", processId);
+ printError();
+ return;
+ }
+
+ int i = 0;
+ while((i < cl->no_of_nodes) && cl->node_states[i].node_id != processId)
+ i++;
+ if(cl->node_states[i].node_id != processId) {
+ ndbout << processId << ": Node not found" << endl;
+ return;
+ }
+ status = cl->node_states[i].node_status;
+ startPhase = cl->node_states[i].start_phase;
+ version = cl->node_states[i].version;
+
+ ndbout << "Node " << processId << ": ";
+ switch(status){
+ case NDB_MGM_NODE_STATUS_NO_CONTACT:
+ ndbout << "No contact" << endl;
+ break;
+ case NDB_MGM_NODE_STATUS_NOT_STARTED:
+ ndbout << "Not started" ;
+ break;
+ case NDB_MGM_NODE_STATUS_STARTING:
+ ndbout << "Starting (Start phase " << startPhase << ")" ;
+ break;
+ case NDB_MGM_NODE_STATUS_STARTED:
+ ndbout << "Started" ;
+ break;
+ case NDB_MGM_NODE_STATUS_SHUTTING_DOWN:
+ ndbout << "Shutting down " << (system == false ? "node" : "system")
+ << " (Phase " << startPhase << ")"
+ ;
+ break;
+ case NDB_MGM_NODE_STATUS_RESTARTING:
+ ndbout << "Restarting" ;
+ break;
+ case NDB_MGM_NODE_STATUS_SINGLEUSER:
+ ndbout << "Single user mode" ;
+ break;
+ default:
+ ndbout << "Unknown state" ;
+ break;
+ }
+ if(status != NDB_MGM_NODE_STATUS_NO_CONTACT)
+ ndbout_c(" (Version %d.%d.%d)",
+ getMajor(version) ,
+ getMinor(version),
+ getBuild(version));
+}
+
+
+//*****************************************************************************
+//*****************************************************************************
+
+void
+CommandInterpreter::executeLogLevel(int processId, const char* parameters,
+ bool all)
+{
+ connect();
+ (void) all;
+ (void) parameters;
+
+ SetLogLevelOrd logLevel; logLevel.clear();
+ LogLevel::EventCategory cat;
+ int level;
+ if (emptyString(parameters) || (strcmp(parameters, "ALL") == 0)) {
+ for(Uint32 i = 0; i<EventLogger::noOfEventCategoryNames; i++)
+ logLevel.setLogLevel(EventLogger::eventCategoryNames[i].category, 7);
+ } else {
+
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, ", ", &tmpPtr);
+ while(item != NULL){
+ char categoryTxt[255];
+ const int m = sscanf(item, "%[^=]=%d", categoryTxt, &level);
+ if(m != 2){
+ free(tmpString);
+ ndbout << "Invalid loglevel specification category=level" << endl;
+ return;
+ }
+
+ if(!EventLogger::matchEventCategory(categoryTxt,
+ &cat)){
+ ndbout << "Invalid loglevel specification, unknown category: "
+ << categoryTxt << endl;
+ free(tmpString);
+ return ;
+ }
+ if(level < 0 || level > 15){
+ ndbout << "Invalid loglevel specification row, level 0-15" << endl;
+ free(tmpString);
+ return ;
+ }
+ logLevel.setLogLevel(cat, level);
+
+ item = strtok_r(NULL, ", ", &tmpPtr);
+ }
+ free(tmpString);
+ }
+
+ struct ndb_mgm_reply reply;
+ int result;
+ result = ndb_mgm_set_loglevel_node(m_mgmsrv,
+ processId, // fast fix - pekka
+ (char*)EventLogger::getEventCategoryName(cat),
+ level,
+ &reply);
+
+ if (result < 0) {
+ ndbout_c("Executing LOGLEVEL on node %d failed.", processId);
+ printError();
+ } else {
+ ndbout << "Executing LOGLEVEL on node " << processId << " OK!"
+ << endl;
+ }
+
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void CommandInterpreter::executeError(int processId,
+ const char* parameters, bool /* all */)
+{
+ if (emptyString(parameters)) {
+ ndbout << "Missing error number." << endl;
+ return;
+ }
+
+ connect();
+ // Copy parameters since strtok will modify it
+ char* newpar = strdup(parameters);
+ char* firstParameter = strtok(newpar, " ");
+
+ int errorNo;
+ if (! convert(firstParameter, errorNo)) {
+ ndbout << "Expected an integer." << endl;
+ free(newpar);
+ return;
+ }
+
+ char* allAfterFirstParameter = strtok(NULL, "\0");
+ if (! emptyString(allAfterFirstParameter)) {
+ ndbout << "Nothing expected after error number." << endl;
+ free(newpar);
+ return;
+ }
+
+ ndb_mgm_insert_error(m_mgmsrv, processId, errorNo, NULL);
+
+ free(newpar);
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::executeTrace(int /*processId*/,
+ const char* /*parameters*/, bool /*all*/)
+{
+#if 0
+ if (emptyString(parameters)) {
+ ndbout << "Missing trace number." << endl;
+ return;
+ }
+
+ char* newpar = strdup(parameters);
+ char* firstParameter = strtok(newpar, " ");
+
+
+ int traceNo;
+ if (! convert(firstParameter, traceNo)) {
+ ndbout << "Expected an integer." << endl;
+ free(newpar);
+ return;
+ }
+ char* allAfterFirstParameter = strtok(NULL, "\0");
+
+ if (! emptyString(allAfterFirstParameter)) {
+ ndbout << "Nothing expected after trace number." << endl;
+ free(newpar);
+ return;
+ }
+
+ int result = _mgmtSrvr.setTraceNo(processId, traceNo);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+ free(newpar);
+#endif
+}
+
+//*****************************************************************************
+//*****************************************************************************
+
+void
+CommandInterpreter::executeLog(int processId,
+ const char* parameters, bool all)
+{
+ connect();
+ struct ndb_mgm_reply reply;
+ Vector<const char *> blocks;
+ if (! parseBlockSpecification(parameters, blocks)) {
+ return;
+ }
+ int len=0;
+ for(Uint32 i=0; i<blocks.size(); i++) {
+ ndbout_c("blocks %s %d",blocks[i], strlen(blocks[i]));
+ len += strlen(blocks[i]);
+ }
+ len += blocks.size()*2;
+ char * blockNames = (char*)malloc(len);
+
+ for(Uint32 i=0; i<blocks.size(); i++) {
+ strcat(blockNames, blocks[i]);
+ strcat(blockNames, "|");
+ }
+ strcat(blockNames, "\0");
+ ndbout_c("blocknames %s", blockNames);
+
+ /*int res =*/ndb_mgm_log_signals(m_mgmsrv,
+ processId,
+ NDB_MGM_SIGNAL_LOG_MODE_INOUT,
+ blockNames,
+ &reply);
+
+#if 0
+ int result =
+ _mgmtSrvr.setSignalLoggingMode(processId, MgmtSrvr::InOut, blocks);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+#endif
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::executeLogIn(int /* processId */,
+ const char* parameters, bool /* all */)
+{
+ Vector<const char*> blocks;
+ if (! parseBlockSpecification(parameters, blocks)) {
+ return;
+ }
+
+#if 0
+ int result = _mgmtSrvr.setSignalLoggingMode(processId, MgmtSrvr::In, blocks);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+#endif
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::executeLogOut(int /*processId*/,
+ const char* parameters, bool /*all*/)
+{
+ Vector<const char*> blocks;
+ if (! parseBlockSpecification(parameters, blocks)) {
+ return;
+ }
+
+
+#if 0
+ int result = _mgmtSrvr.setSignalLoggingMode(processId, MgmtSrvr::Out,
+ blocks);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+#endif
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::executeLogOff(int /*processId*/,
+ const char* parameters, bool /*all*/)
+{
+ Vector<const char*> blocks;
+ if (! parseBlockSpecification(parameters, blocks)) {
+ return;
+ }
+
+
+#if 0
+ int result = _mgmtSrvr.setSignalLoggingMode(processId, MgmtSrvr::Off,
+ blocks);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+#endif
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::executeTestOn(int /*processId*/,
+ const char* parameters, bool /*all*/)
+{
+ if (! emptyString(parameters)) {
+ ndbout << "No parameters expected to this command." << endl;
+ return;
+ }
+
+#if 0
+ int result = _mgmtSrvr.startSignalTracing(processId);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+#endif
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::executeTestOff(int /*processId*/,
+ const char* parameters, bool /*all*/)
+{
+ if (! emptyString(parameters)) {
+ ndbout << "No parameters expected to this command." << endl;
+ return;
+ }
+
+#if 0
+ int result = _mgmtSrvr.stopSignalTracing(processId);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+#endif
+}
+
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::executeSet(int /*processId*/,
+ const char* parameters, bool /*all*/)
+{
+ if (emptyString(parameters)) {
+ ndbout << "Missing parameter name." << endl;
+ return;
+ }
+#if 0
+ // Copy parameters since strtok will modify it
+ char* newpar = strdup(parameters);
+ char* configParameterName = strtok(newpar, " ");
+
+ char* allAfterParameterName = strtok(NULL, "\0");
+ if (emptyString(allAfterParameterName)) {
+ ndbout << "Missing parameter value." << endl;
+ free(newpar);
+ return;
+ }
+
+ char* value = strtok(allAfterParameterName, " ");
+
+ char* allAfterValue = strtok(NULL, "\0");
+ if (! emptyString(allAfterValue)) {
+ ndbout << "Nothing expected after parameter value." << endl;
+ free(newpar);
+ return;
+ }
+
+ bool configBackupFileUpdated;
+ bool configPrimaryFileUpdated;
+
+ // TODO The handling of the primary and backup config files should be
+ // analysed further.
+ // How it should be handled if only the backup is possible to write.
+
+ int result = _mgmtSrvr.updateConfigParam(processId, configParameterName,
+ value, configBackupFileUpdated,
+ configPrimaryFileUpdated);
+ if (result == 0) {
+ if (configBackupFileUpdated && configPrimaryFileUpdated) {
+ ndbout << "The configuration is updated." << endl;
+ }
+ else if (configBackupFileUpdated && !configPrimaryFileUpdated) {
+ ndbout << "The configuration is updated but it was only possible "
+ << "to update the backup configuration file, not the primary."
+ << endl;
+ }
+ else {
+ NDB_ASSERT(false, "");
+ }
+ }
+ else {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ if (configBackupFileUpdated && configPrimaryFileUpdated) {
+ ndbout << "The configuration files are however updated and "
+ << "the value will be used next time the process is restarted."
+ << endl;
+ }
+ else if (configBackupFileUpdated && !configPrimaryFileUpdated) {
+ ndbout << "It was only possible to update the backup "
+ << "configuration file, not the primary." << endl;
+ }
+ else if (!configBackupFileUpdated && !configPrimaryFileUpdated) {
+ ndbout << "The configuration files are not updated." << endl;
+ }
+ else {
+ // The primary is not tried to write if the write of backup file fails
+ NDB_ASSERT(false, "");
+ }
+ }
+ free(newpar);
+#endif
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void CommandInterpreter::executeGetStat(int /*processId*/,
+ const char* parameters, bool /*all*/)
+{
+ if (! emptyString(parameters)) {
+ ndbout << "No parameters expected to this command." << endl;
+ return;
+ }
+
+#if 0
+ MgmtSrvr::Statistics statistics;
+ int result = _mgmtSrvr.getStatistics(processId, statistics);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ return;
+ }
+#endif
+ // Print statistic...
+ /*
+ ndbout << "Number of GETSTAT commands: "
+ << statistics._test1 << endl;
+ */
+}
+
+//*****************************************************************************
+//*****************************************************************************
+
+void
+CommandInterpreter::executeEventReporting(int processId,
+ const char* parameters,
+ bool all)
+{
+ connect();
+ SetLogLevelOrd logLevel; logLevel.clear();
+ char categoryTxt[255];
+ int level;
+ LogLevel::EventCategory cat;
+ if (emptyString(parameters) || (strcmp(parameters, "ALL") == 0)) {
+ for(Uint32 i = 0; i<EventLogger::noOfEventCategoryNames; i++)
+ logLevel.setLogLevel(EventLogger::eventCategoryNames[i].category, 7);
+ } else {
+
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, ", ", &tmpPtr);
+ while(item != NULL){
+ const int m = sscanf(item, "%[^=]=%d", categoryTxt, &level);
+ if(m != 2){
+ free(tmpString);
+ ndbout << "Invalid loglevel specification category=level" << endl;
+ return;
+ }
+
+ if(!EventLogger::matchEventCategory(categoryTxt,
+ &cat)){
+ ndbout << "Invalid loglevel specification, unknown category: "
+ << categoryTxt << endl;
+ free(tmpString);
+ return ;
+ }
+ if(level < 0 || level > 15){
+ ndbout << "Invalid loglevel specification row, level 0-15" << endl;
+ free(tmpString);
+ return ;
+ }
+ logLevel.setLogLevel(cat, level);
+
+ item = strtok_r(NULL, ", ", &tmpPtr);
+ }
+ free(tmpString);
+ }
+ struct ndb_mgm_reply reply;
+ int result;
+
+ result =
+ ndb_mgm_set_loglevel_clusterlog(m_mgmsrv,
+ processId, // fast fix - pekka
+ (char*)
+ EventLogger::getEventCategoryName(cat),
+ level,
+ &reply);
+
+ if (result != 0) {
+ ndbout_c("Executing CLUSTERLOG on node %d failed", processId);
+ printError();
+ } else {
+ ndbout << "Executing CLUSTERLOG on node " << processId << " OK!"
+ << endl;
+ }
+}
+
+/*****************************************************************************
+ * Backup
+ *****************************************************************************/
+void
+CommandInterpreter::executeStartBackup(char* /*parameters*/)
+{
+ connect();
+ struct ndb_mgm_reply reply;
+ unsigned int backupId;
+ int result = ndb_mgm_start_backup(m_mgmsrv, &backupId, &reply);
+ if (result != 0) {
+ ndbout << "Start of backup failed" << endl;
+ printError();
+ } else {
+ ndbout << "Backup started. Backup id " << backupId << "." << endl;
+ }
+}
+
+void
+CommandInterpreter::executeAbortBackup(char* parameters)
+{
+ connect();
+ strtok(parameters, " ");
+ struct ndb_mgm_reply reply;
+ char* id = strtok(NULL, "\0");
+ int bid = -1;
+ if(id == 0 || sscanf(id, "%d", &bid) != 1){
+ ndbout << "Invalid arguments: expected <BackupId>" << endl;
+ return;
+ }
+ int result = ndb_mgm_abort_backup(m_mgmsrv, bid, &reply);
+ if (result != 0) {
+ ndbout << "Abort of backup " << bid << " failed" << endl;
+ printError();
+ } else {
+ ndbout << "Abort of backup " << bid << " ordered" << endl;
+ }
+}
+
+#ifdef HAVE_GLOBAL_REPLICATION
+/*****************************************************************************
+ * Global Replication
+ *
+ * For information about the different commands, see
+ * GrepReq::Request in file signaldata/grepImpl.cpp.
+ *
+ * Below are commands as of 2003-07-05 (may change!):
+ * START = 0, ///< Start Global Replication (all phases)
+ * START_METALOG = 1, ///< Start Global Replication (all phases)
+ * START_METASCAN = 2, ///< Start Global Replication (all phases)
+ * START_DATALOG = 3, ///< Start Global Replication (all phases)
+ * START_DATASCAN = 4, ///< Start Global Replication (all phases)
+ * START_REQUESTOR = 5, ///< Start Global Replication (all phases)
+ * ABORT = 6, ///< Immediate stop (removes subscription)
+ * SLOW_STOP = 7, ///< Stop after finishing applying current GCI epoch
+ * FAST_STOP = 8, ///< Stop after finishing applying all PS GCI epochs
+ * START_TRANSFER = 9, ///< Start SS-PS transfer
+ * STOP_TRANSFER = 10, ///< Stop SS-PS transfer
+ * START_APPLY = 11, ///< Start applying GCI epochs in SS
+ * STOP_APPLY = 12, ///< Stop applying GCI epochs in SS
+ * STATUS = 13, ///< Status
+ * START_SUBSCR = 14,
+ * REMOVE_BUFFERS = 15,
+ * DROP_TABLE = 16
+
+ *****************************************************************************/
+
+void
+CommandInterpreter::executeRep(char* parameters)
+{
+ if (emptyString(parameters)) {
+ ndbout << helpTextRep;
+ return;
+ }
+
+ connect();
+ char * line = strdup(parameters);
+ char * firstToken = strtok(line, " ");
+
+ struct ndb_rep_reply reply;
+ unsigned int repId;
+
+
+ if (!strcmp(firstToken, "CONNECT")) {
+ char * host = strtok(NULL, "\0");
+ for (unsigned int i = 0; i < strlen(host); ++i) {
+ host[i] = tolower(host[i]);
+ }
+
+ if(host == NULL)
+ {
+ ndbout_c("host:port must be specified.");
+ return;
+ }
+
+ if(rep_connected) {
+ if(m_repserver != NULL) {
+ ndb_rep_disconnect(m_repserver);
+ rep_connected = false;
+ }
+ }
+
+ if(m_repserver == NULL)
+ m_repserver = ndb_rep_create_handle();
+ if(ndb_rep_connect(m_repserver, host) < 0)
+ ndbout_c("Failed to connect to %s", host);
+ else
+ rep_connected=true;
+ return;
+
+ if(!rep_connected) {
+ ndbout_c("Not connected to REP server");
+ }
+ }
+
+ /********
+ * START
+ ********/
+ if (!strcmp(firstToken, "START")) {
+
+ unsigned int req;
+ char *startType = strtok(NULL, "\0");
+
+ if (startType == NULL) {
+ req = GrepReq::START;
+ } else if (!strcmp(startType, "SUBSCRIPTION")) {
+ req = GrepReq::START_SUBSCR;
+ } else if (!strcmp(startType, "METALOG")) {
+ req = GrepReq::START_METALOG;
+ } else if (!strcmp(startType, "METASCAN")) {
+ req = GrepReq::START_METASCAN;
+ } else if (!strcmp(startType, "DATALOG")) {
+ req = GrepReq::START_DATALOG;
+ } else if (!strcmp(startType, "DATASCAN")) {
+ req = GrepReq::START_DATASCAN;
+ } else if (!strcmp(startType, "REQUESTOR")) {
+ req = GrepReq::START_REQUESTOR;
+ } else if (!strcmp(startType, "TRANSFER")) {
+ req = GrepReq::START_TRANSFER;
+ } else if (!strcmp(startType, "APPLY")) {
+ req = GrepReq::START_APPLY;
+ } else if (!strcmp(startType, "DELETE")) {
+ req = GrepReq::START_DELETE;
+ } else {
+ ndbout_c("Illegal argument to command 'REPLICATION START'");
+ return;
+ }
+
+ int result = ndb_rep_command(m_repserver, req, &repId, &reply);
+
+ if (result != 0) {
+ ndbout << "Start of Global Replication failed" << endl;
+ } else {
+ ndbout << "Start of Global Replication ordered" << endl;
+ }
+ return;
+ }
+
+ /********
+ * STOP
+ ********/
+ if (!strcmp(firstToken, "STOP")) {
+ unsigned int req;
+ char *startType = strtok(NULL, " ");
+ unsigned int epoch = 0;
+
+ if (startType == NULL) {
+ /**
+ * Stop immediately
+ */
+ req = GrepReq::STOP;
+ } else if (!strcmp(startType, "EPOCH")) {
+ char *strEpoch = strtok(NULL, "\0");
+ if(strEpoch == NULL) {
+ ndbout_c("Epoch expected!");
+ return;
+ }
+ req = GrepReq::STOP;
+ epoch=atoi(strEpoch);
+ } else if (!strcmp(startType, "SUBSCRIPTION")) {
+ req = GrepReq::STOP_SUBSCR;
+ } else if (!strcmp(startType, "METALOG")) {
+ req = GrepReq::STOP_METALOG;
+ } else if (!strcmp(startType, "METASCAN")) {
+ req = GrepReq::STOP_METASCAN;
+ } else if (!strcmp(startType, "DATALOG")) {
+ req = GrepReq::STOP_DATALOG;
+ } else if (!strcmp(startType, "DATASCAN")) {
+ req = GrepReq::STOP_DATASCAN;
+ } else if (!strcmp(startType, "REQUESTOR")) {
+ req = GrepReq::STOP_REQUESTOR;
+ } else if (!strcmp(startType, "TRANSFER")) {
+ req = GrepReq::STOP_TRANSFER;
+ } else if (!strcmp(startType, "APPLY")) {
+ req = GrepReq::STOP_APPLY;
+ } else if (!strcmp(startType, "DELETE")) {
+ req = GrepReq::STOP_DELETE;
+ } else {
+ ndbout_c("Illegal argument to command 'REPLICATION STOP'");
+ return;
+ }
+ int result = ndb_rep_command(m_repserver, req, &repId, &reply, epoch);
+
+ if (result != 0) {
+ ndbout << "Stop command failed" << endl;
+ } else {
+ ndbout << "Stop ordered" << endl;
+ }
+ return;
+ }
+
+ /*********
+ * STATUS
+ *********/
+ if (!strcmp(firstToken, "STATUS")) {
+ struct rep_state repstate;
+ int result =
+ ndb_rep_get_status(m_repserver, &repId, &reply, &repstate);
+
+ if (result != 0) {
+ ndbout << "Status request of Global Replication failed" << endl;
+ } else {
+ ndbout << "Status request of Global Replication ordered" << endl;
+ ndbout << "See printout at one of the DB nodes" << endl;
+ ndbout << "(Better status report is under development.)" << endl;
+ ndbout << " SubscriptionId " << repstate.subid
+ << " SubscriptionKey " << repstate.subkey << endl;
+ }
+ return;
+ }
+
+ /*********
+ * QUERY (see repapi.h for querable counters)
+ *********/
+ if (!strcmp(firstToken, "QUERY")) {
+ char *query = strtok(NULL, "\0");
+ int queryCounter=-1;
+ if(query != NULL) {
+ queryCounter = atoi(query);
+ }
+ struct rep_state repstate;
+ unsigned repId = 0;
+ int result = ndb_rep_query(m_repserver, (QueryCounter)queryCounter,
+ &repId, &reply, &repstate);
+
+ if (result != 0) {
+ ndbout << "Query repserver failed" << endl;
+ } else {
+ ndbout << "Query repserver sucessful" << endl;
+ ndbout_c("repstate : QueryCounter %d, f=%d l=%d"
+ " nodegroups %d" ,
+ repstate.queryCounter,
+ repstate.first[0], repstate.last[0],
+ repstate.no_of_nodegroups );
+ }
+ return;
+ }
+}
+#endif // HAVE_GLOBAL_REPLICATION
+
+
+/*****************************************************************************
+ * CPC
+ *****************************************************************************/
+
+void
+CommandInterpreter::executeCpc(char *parameters)
+{
+ char *host_str = NULL, *port_str = NULL, *end;
+ long port = 1234; /* XXX */
+
+ while((host_str = strsep(&parameters, " \t:")) != NULL &&
+ host_str[0] == '\0');
+
+ if(parameters && parameters[0] != '\0') {
+ while((port_str = strsep(&parameters, " \t:")) != NULL &&
+ port_str[0] == '\0');
+
+ errno = 0;
+ port = strtol(port_str, &end, 0);
+ if(end[0] != '\0')
+ goto error;
+ if((port == LONG_MAX || port == LONG_MIN) &&
+ errno == ERANGE)
+ goto error;
+ }
+
+ {
+ SimpleCpcClient cpc(host_str, port);
+ bool done = false;
+
+ if(cpc.connect() < 0) {
+ ndbout_c("Cannot connect to %s:%d.", cpc.getHost(), cpc.getPort());
+ switch(errno) {
+ case ENOENT:
+ ndbout << ": " << "No such host" << endl;
+ break;
+ default:
+ ndbout << ": " << strerror(errno) << endl;
+ break;
+ }
+ return;
+ }
+
+ while(!done) {
+ char *line = readline("CPC> ");
+ if(line != NULL) {
+ add_history(line);
+
+ char *cmd = strtok(line, " ");
+ char *arg = strtok(NULL, "");
+
+ if(arg != NULL) {
+ while(arg[0] == ' ')
+ arg++;
+ if(strlen(arg) == 0)
+ arg = NULL;
+ }
+
+ if(cmd != NULL) {
+ if(strcmp(cmd, "exit") == 0)
+ done = true;
+ else if(strcmp(cmd, "list") == 0)
+ cpc.cmd_list(arg);
+ else if(strcmp(cmd, "start") == 0)
+ cpc.cmd_start(arg);
+ else if(strcmp(cmd, "stop") == 0)
+ cpc.cmd_stop(arg);
+ else if(strcmp(cmd, "help") == 0)
+ cpc.cmd_help(arg);
+ }
+ } else {
+ done = true;
+ ndbout << endl;
+ }
+ }
+ }
+ return;
+
+ error:
+ ndbout << "Error: expected a tcp port number, got '" << port_str << "'."
+ << endl;
+ return;
+}
+
+#if 0
+static
+void
+CmdBackupCallback(const MgmtSrvr::BackupEvent & event){
+ char str[255];
+
+ ndbout << endl;
+
+ bool ok = false;
+ switch(event.Event){
+ case MgmtSrvr::BackupEvent::BackupStarted:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup %d started", event.Started.BackupId);
+ break;
+ case MgmtSrvr::BackupEvent::BackupFailedToStart:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup failed to start (Error %d)",
+ event.FailedToStart.ErrorCode);
+ break;
+ case MgmtSrvr::BackupEvent::BackupCompleted:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup %d completed",
+ event.Completed.BackupId);
+ ndbout << str << endl;
+
+ snprintf(str, sizeof(str),
+ " StartGCP: %d StopGCP: %d",
+ event.Completed.startGCP, event.Completed.stopGCP);
+ ndbout << str << endl;
+
+ snprintf(str, sizeof(str),
+ " #Records: %d #LogRecords: %d",
+ event.Completed.NoOfRecords, event.Completed.NoOfLogRecords);
+ ndbout << str << endl;
+
+ snprintf(str, sizeof(str),
+ " Data: %d bytes Log: %d bytes",
+ event.Completed.NoOfBytes, event.Completed.NoOfLogBytes);
+ break;
+ case MgmtSrvr::BackupEvent::BackupAborted:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup %d has been aborted reason %d",
+ event.Aborted.BackupId,
+ event.Aborted.Reason);
+ break;
+ }
+ if(!ok){
+ snprintf(str, sizeof(str),
+ "Unknown backup event: %d",
+ event.Event);
+
+ }
+ ndbout << str << endl;
+}
+#endif
diff --git a/ndb/src/mgmclient/CommandInterpreter.hpp b/ndb/src/mgmclient/CommandInterpreter.hpp
new file mode 100644
index 00000000000..9049ef39915
--- /dev/null
+++ b/ndb/src/mgmclient/CommandInterpreter.hpp
@@ -0,0 +1,198 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CommandInterpreter_H
+#define CommandInterpreter_H
+
+#define HAVE_GLOBAL_REPLICATION
+//*****************************************************************************
+// Author: Peter Lind
+//*****************************************************************************
+
+#include <NdbStdio.h>
+#include <ndb_types.h>
+#include <stdlib.h>
+#include <Vector.hpp>
+#include <editline/editline.h>
+
+#ifdef HAVE_GLOBAL_REPLICATION
+#include "../rep/repapi/repapi.h"
+#endif
+
+#include <mgmapi.h>
+
+class MgmtSrvr;
+
+/**
+ * @class CommandInterpreter
+ * @brief Reads command line in management client
+ *
+ * This class has one public method which reads a command line
+ * from a stream. It then interpret that commmand line and calls a suitable
+ * method in the MgmtSrvr class which executes the command.
+ *
+ * For command syntax, see the HELP command.
+ */
+class CommandInterpreter {
+public:
+ /**
+ * Constructor
+ * @param mgmtSrvr: Management server to use when executing commands
+ */
+ CommandInterpreter(const char *);
+ ~CommandInterpreter();
+
+ /**
+ * Reads one line from the stream, parse the line to find
+ * a command and then calls a suitable method which executes
+ * the command.
+ *
+ * @return true until quit/bye/exit has been typed
+ */
+ int readAndExecute(int _try_reconnect=-1);
+
+private:
+ /**
+ * Read a string, and return a pointer to it.
+ *
+ * @return NULL on EOF.
+ */
+ char *readline_gets ()
+ {
+ static char *line_read = (char *)NULL;
+
+ /* If the buffer has already been allocated, return the memory
+ to the free pool. */
+ if (line_read)
+ {
+ free (line_read);
+ line_read = (char *)NULL;
+ }
+
+ /* Get a line from the user. */
+ line_read = readline ("NDB> ");
+
+ /* If the line has any text in it, save it on the history. */
+ if (line_read && *line_read)
+ add_history (line_read);
+
+ return (line_read);
+ }
+
+ void printError();
+
+ /**
+ * Analyse the command line, after the first token.
+ *
+ * @param processId: DB process id to send command to or -1 if
+ * command will be sent to all DB processes.
+ * @param allAfterFirstToken: What the client gave after the
+ * first token on the command line
+ */
+ void analyseAfterFirstToken(int processId, char* allAfterFirstTokenCstr);
+
+ /**
+ * Parse the block specification part of the LOG* commands,
+ * things after LOG*: [BLOCK = {ALL|<blockName>+}]
+ *
+ * @param allAfterLog: What the client gave after the second token
+ * (LOG*) on the command line
+ * @param blocks, OUT: ALL or name of all the blocks
+ * @return: true if correct syntax, otherwise false
+ */
+ bool parseBlockSpecification(const char* allAfterLog,
+ Vector<const char*>& blocks);
+
+ /**
+ * A bunch of execute functions: Executes one of the commands
+ *
+ * @param processId: DB process id to send command to
+ * @param parameters: What the client gave after the command name
+ * on the command line.
+ * For example if complete input from user is: "1 LOGLEVEL 22" then the
+ * parameters argument is the string with everything after LOGLEVEL, in
+ * this case "22". Each function is responsible to check the parameters
+ * argument.
+ */
+ void executeHelp(char* parameters);
+ void executeShow(char* parameters);
+ void executeRun(char* parameters);
+ void executeInfo(char* parameters);
+ void executeClusterLog(char* parameters);
+
+public:
+ void executeStop(int processId, const char* parameters, bool all);
+ void executeEnterSingleUser(char* parameters);
+ void executeExitSingleUser(char* parameters);
+ void executeStart(int processId, const char* parameters, bool all);
+ void executeRestart(int processId, const char* parameters, bool all);
+ void executeLogLevel(int processId, const char* parameters, bool all);
+ void executeError(int processId, const char* parameters, bool all);
+ void executeTrace(int processId, const char* parameters, bool all);
+ void executeLog(int processId, const char* parameters, bool all);
+ void executeLogIn(int processId, const char* parameters, bool all);
+ void executeLogOut(int processId, const char* parameters, bool all);
+ void executeLogOff(int processId, const char* parameters, bool all);
+ void executeTestOn(int processId, const char* parameters, bool all);
+ void executeTestOff(int processId, const char* parameters, bool all);
+ void executeSet(int processId, const char* parameters, bool all);
+ void executeGetStat(int processId, const char* parameters, bool all);
+ void executeStatus(int processId, const char* parameters, bool all);
+ void executeEventReporting(int processId, const char* parameters, bool all);
+ void executeDumpState(int processId, const char* parameters, bool all);
+ void executeStartBackup(char * parameters);
+ void executeAbortBackup(char * parameters);
+
+ void executeRep(char* parameters);
+
+ void executeCpc(char * parameters);
+
+public:
+ bool connect();
+ bool disconnect();
+
+ /**
+ * A execute function definition
+ */
+public:
+ typedef void (CommandInterpreter::* ExecuteFunction)(int processId,
+ const char * param,
+ bool all);
+
+ struct CommandFunctionPair {
+ const char * command;
+ ExecuteFunction executeFunction;
+ };
+private:
+ /**
+ *
+ */
+ void executeForAll(const char * cmd,
+ ExecuteFunction fun,
+ const char * param);
+
+ NdbMgmHandle m_mgmsrv;
+ bool connected;
+ const char *host;
+ int try_reconnect;
+#ifdef HAVE_GLOBAL_REPLICATION
+ NdbRepHandle m_repserver;
+ const char *rep_host;
+ bool rep_connected;
+#endif
+};
+
+#endif // CommandInterpreter_H
diff --git a/ndb/src/mgmclient/CpcClient.cpp b/ndb/src/mgmclient/CpcClient.cpp
new file mode 100644
index 00000000000..24eab7194e9
--- /dev/null
+++ b/ndb/src/mgmclient/CpcClient.cpp
@@ -0,0 +1,564 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <ndb_types.h>
+#include <editline/editline.h>
+
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <netdb.h>
+#include <errno.h>
+
+#include <NdbOut.hpp>
+#include <NdbTCP.h>
+#include "CpcClient.hpp"
+
+#define CPC_CMD(name, value, desc) \
+ { (name), \
+ 0, \
+ ParserRow_t::Cmd, \
+ ParserRow_t::String, \
+ ParserRow_t::Optional, \
+ ParserRow_t::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ (desc), \
+ (void *)(value) }
+
+#define CPC_ARG(name, type, opt, desc) \
+ { (name), \
+ 0, \
+ ParserRow_t::Arg, \
+ ParserRow_t::type, \
+ ParserRow_t::opt, \
+ ParserRow_t::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ (desc) }
+
+#define CPC_END() \
+ { 0, \
+ 0, \
+ ParserRow_t::Arg, \
+ ParserRow_t::Int, \
+ ParserRow_t::Optional, \
+ ParserRow_t::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0 }
+
+#ifdef DEBUG_PRINT_PROPERTIES
+static void printprop(const Properties &p) {
+ Properties::Iterator iter(&p);
+ const char *name;
+ while((name = iter.next()) != NULL) {
+ PropertiesType t;
+ Uint32 val_i;
+ BaseString val_s;
+
+ p.getTypeOf(name, &t);
+ switch(t) {
+ case PropertiesType_Uint32:
+ p.get(name, &val_i);
+ ndbout << name << " (Uint32): " << val_i << endl;
+ break;
+ case PropertiesType_char:
+ p.get(name, val_s);
+ ndbout << name << " (string): " << val_s << endl;
+ break;
+ default:
+ ndbout << "Unknown type " << t << endl;
+ break;
+ }
+ }
+}
+#endif
+
+void
+SimpleCpcClient::cmd_stop(char *arg) {
+ Properties p;
+ Vector<Process> proc_list;
+
+ list_processes(proc_list, p);
+ bool stopped = false;
+
+ for(size_t i = 0; i < proc_list.size(); i++) {
+ if(strcmp(proc_list[i].m_name.c_str(), arg) == 0) {
+ stopped = true;
+ Properties reply;
+ stop_process(proc_list[i].m_id, reply);
+
+ Uint32 status;
+ reply.get("status", &status);
+ if(status != 0) {
+ BaseString msg;
+ reply.get("errormessage", msg);
+ ndbout << "Stop failed: " << msg << endl;
+ }
+ }
+ }
+
+ if(!stopped)
+ ndbout << "No such process" << endl;
+}
+
+int
+SimpleCpcClient::stop_process(Uint32 id, Properties& reply){
+ const ParserRow_t stop_reply[] = {
+ CPC_CMD("stop process", NULL, ""),
+ CPC_ARG("status", Int, Mandatory, ""),
+ CPC_ARG("id", Int, Optional, ""),
+ CPC_ARG("errormessage", String, Optional, ""),
+
+ CPC_END()
+ };
+
+ Properties args;
+ args.put("id", id);
+
+ const Properties* ret = cpc_call("stop process", args, stop_reply);
+ if(ret == 0){
+ reply.put("status", (Uint32)0);
+ reply.put("errormessage", "unknown error");
+ return -1;
+ }
+
+ Uint32 status = 0;
+ ret->get("status", &status);
+ reply.put("status", status);
+ if(status != 0) {
+ BaseString msg;
+ ret->get("errormessage", msg);
+ reply.put("errormessage", msg.c_str());
+ }
+
+ return status;
+}
+
+void
+SimpleCpcClient::cmd_start(char *arg) {
+ Properties p;
+ Vector<Process> proc_list;
+ list_processes(proc_list, p);
+ bool startped = false;
+
+ for(size_t i = 0; i < proc_list.size(); i++) {
+ if(strcmp(proc_list[i].m_name.c_str(), arg) == 0) {
+ startped = true;
+
+ Properties reply;
+ start_process(proc_list[i].m_id, reply);
+
+ Uint32 status;
+ reply.get("status", &status);
+ if(status != 0) {
+ BaseString msg;
+ reply.get("errormessage", msg);
+ ndbout << "Start failed: " << msg << endl;
+ }
+ }
+ }
+
+ if(!startped)
+ ndbout << "No such process" << endl;
+}
+
+int
+SimpleCpcClient::start_process(Uint32 id, Properties& reply){
+ const ParserRow_t start_reply[] = {
+ CPC_CMD("start process", NULL, ""),
+ CPC_ARG("status", Int, Mandatory, ""),
+ CPC_ARG("id", Int, Optional, ""),
+ CPC_ARG("errormessage", String, Optional, ""),
+
+ CPC_END()
+ };
+
+ Properties args;
+ args.put("id", id);
+
+ const Properties* ret = cpc_call("start process", args, start_reply);
+ if(ret == 0){
+ reply.put("status", (Uint32)0);
+ reply.put("errormessage", "unknown error");
+ return -1;
+ }
+
+ Uint32 status = 0;
+ ret->get("status", &status);
+ reply.put("status", status);
+ if(status != 0) {
+ BaseString msg;
+ ret->get("errormessage", msg);
+ reply.put("errormessage", msg.c_str());
+ }
+
+ return status;
+}
+
+int
+SimpleCpcClient::undefine_process(Uint32 id, Properties& reply){
+ const ParserRow_t stop_reply[] = {
+ CPC_CMD("undefine process", NULL, ""),
+ CPC_ARG("status", Int, Mandatory, ""),
+ CPC_ARG("id", Int, Optional, ""),
+ CPC_ARG("errormessage", String, Optional, ""),
+
+ CPC_END()
+ };
+
+ Properties args;
+ args.put("id", id);
+
+ const Properties* ret = cpc_call("undefine process", args, stop_reply);
+ if(ret == 0){
+ reply.put("status", (Uint32)0);
+ reply.put("errormessage", "unknown error");
+ return -1;
+ }
+
+ Uint32 status = 0;
+ ret->get("status", &status);
+ reply.put("status", status);
+ if(status != 0) {
+ BaseString msg;
+ ret->get("errormessage", msg);
+ reply.put("errormessage", msg.c_str());
+ }
+
+ return status;
+}
+
+static void
+printproc(SimpleCpcClient::Process & p) {
+ ndbout.println("Name: %s", p.m_name.c_str());
+ ndbout.println("Id: %d", p.m_id);
+ ndbout.println("Type: %s", p.m_type.c_str());
+ ndbout.println("Group: %s", p.m_group.c_str());
+ ndbout.println("Program path: %s", p.m_path.c_str());
+ ndbout.println("Arguments: %s", p.m_args.c_str());
+ ndbout.println("Environment: %s", p.m_env.c_str());
+ ndbout.println("Working directory: %s", p.m_cwd.c_str());
+ ndbout.println("Owner: %s", p.m_owner.c_str());
+ ndbout.println("Runas: %s", p.m_runas.c_str());
+ ndbout.println("Ulimit: %s", p.m_ulimit.c_str());
+ ndbout.println("");
+}
+
+void
+SimpleCpcClient::cmd_list(char *arg) {
+ Properties p;
+ Vector<Process> proc_list;
+ list_processes(proc_list, p);
+
+ for(size_t i = 0; i < proc_list.size(); i++) {
+ printproc(proc_list[i]);
+ }
+}
+
+static int
+convert(const Properties & src, SimpleCpcClient::Process & dst){
+ bool b = true;
+ b &= src.get("id", (Uint32*)&dst.m_id);
+ b &= src.get("name", dst.m_name);
+ b &= src.get("type", dst.m_type);
+ b &= src.get("status", dst.m_status);
+ b &= src.get("owner", dst.m_owner);
+ b &= src.get("group", dst.m_group);
+ b &= src.get("path", dst.m_path);
+ b &= src.get("args", dst.m_args);
+ b &= src.get("env", dst.m_env);
+ b &= src.get("cwd", dst.m_cwd);
+ b &= src.get("runas", dst.m_runas);
+
+ b &= src.get("stdin", dst.m_stdin);
+ b &= src.get("stdout", dst.m_stdout);
+ b &= src.get("stderr", dst.m_stderr);
+ b &= src.get("ulimit", dst.m_ulimit);
+
+ return b;
+}
+
+static int
+convert(const SimpleCpcClient::Process & src, Properties & dst ){
+ bool b = true;
+ //b &= dst.put("id", (Uint32)src.m_id);
+ b &= dst.put("name", src.m_name.c_str());
+ b &= dst.put("type", src.m_type.c_str());
+ //b &= dst.put("status", src.m_status.c_str());
+ b &= dst.put("owner", src.m_owner.c_str());
+ b &= dst.put("group", src.m_group.c_str());
+ b &= dst.put("path", src.m_path.c_str());
+ b &= dst.put("args", src.m_args.c_str());
+ b &= dst.put("env", src.m_env.c_str());
+ b &= dst.put("cwd", src.m_cwd.c_str());
+ b &= dst.put("runas", src.m_runas.c_str());
+
+ b &= dst.put("stdin", src.m_stdin.c_str());
+ b &= dst.put("stdout", src.m_stdout.c_str());
+ b &= dst.put("stderr", src.m_stderr.c_str());
+ b &= dst.put("ulimit", src.m_ulimit.c_str());
+
+ return b;
+}
+
+int
+SimpleCpcClient::define_process(Process & p, Properties& reply){
+ const ParserRow_t define_reply[] = {
+ CPC_CMD("define process", NULL, ""),
+ CPC_ARG("status", Int, Mandatory, ""),
+ CPC_ARG("id", Int, Optional, ""),
+ CPC_ARG("errormessage", String, Optional, ""),
+
+ CPC_END()
+ };
+
+ Properties args;
+ convert(p, args);
+
+ const Properties* ret = cpc_call("define process", args, define_reply);
+ if(ret == 0){
+ reply.put("status", (Uint32)0);
+ reply.put("errormessage", "unknown error");
+ return -1;
+ }
+
+ Uint32 status = 0;
+ ret->get("status", &status);
+ reply.put("status", status);
+ if(status != 0) {
+ BaseString msg;
+ ret->get("errormessage", msg);
+ reply.put("errormessage", msg.c_str());
+ }
+
+ Uint32 id;
+ if(!ret->get("id", &id)){
+ return -1;
+ }
+
+ p.m_id = id;
+
+ return status;
+}
+
+int
+SimpleCpcClient::list_processes(Vector<Process> &procs, Properties& reply) {
+ enum Proclist {
+ Proclist_Start,
+ Proclist_End,
+ Proclist_Entry
+ };
+ const ParserRow_t list_reply[] = {
+ CPC_CMD("start processes", Proclist_Start, ""),
+
+ CPC_CMD("end processes", Proclist_End, ""),
+
+ CPC_CMD("process", Proclist_Entry, ""),
+ CPC_ARG("id", Int, Mandatory, "Id of process."),
+ CPC_ARG("name", String, Mandatory, "Name of process"),
+ CPC_ARG("group", String, Mandatory, "Group of process"),
+ CPC_ARG("env", String, Mandatory, "Environment variables for process"),
+ CPC_ARG("path", String, Mandatory, "Path to binary"),
+ CPC_ARG("args", String, Mandatory, "Arguments to process"),
+ CPC_ARG("type", String, Mandatory, "Type of process"),
+ CPC_ARG("cwd", String, Mandatory, "Working directory of process"),
+ CPC_ARG("owner", String, Mandatory, "Owner of process"),
+ CPC_ARG("status",String, Mandatory, "Status of process"),
+ CPC_ARG("runas", String, Mandatory, "Run as user"),
+ CPC_ARG("stdin", String, Mandatory, "Redirect stdin"),
+ CPC_ARG("stdout",String, Mandatory, "Redirect stdout"),
+ CPC_ARG("stderr",String, Mandatory, "Redirect stderr"),
+ CPC_ARG("ulimit",String, Mandatory, "ulimit"),
+
+ CPC_END()
+ };
+
+ reply.clear();
+
+ const Properties args;
+
+ cpc_send("list processes", args);
+
+ bool done = false;
+ while(!done) {
+ const Properties *proc;
+ enum Proclist p;
+ cpc_recv(list_reply, &proc, (void **)&p);
+
+ switch(p) {
+ case Proclist_Start:
+ /* do nothing */
+ break;
+ case Proclist_End:
+ done = true;
+ break;
+ case Proclist_Entry:
+ if(proc != NULL){
+ Process p;
+ convert(* proc, p);
+ procs.push_back(p);
+ }
+ break;
+ default:
+ /* ignore */
+ break;
+ }
+ }
+ return 0;
+}
+
+void
+SimpleCpcClient::cmd_help(char *arg) {
+ ndbout
+ << "HELP Print help text" << endl
+ << "LIST List processes" << endl
+ << "START Start process" << endl
+ << "STOP Stop process" << endl;
+}
+
+SimpleCpcClient::SimpleCpcClient(const char *_host, int _port) {
+ host = strdup(_host);
+ port = _port;
+ cpc_sock = -1;
+ cpc_in = NULL;
+ cpc_out = NULL;
+}
+
+SimpleCpcClient::~SimpleCpcClient() {
+ if(host != NULL) {
+ free(host);
+ host = NULL;
+ }
+
+ port = 0;
+
+ if(cpc_sock == -1) {
+ close(cpc_sock);
+ cpc_sock = -1;
+ }
+
+ if(cpc_in != NULL)
+ delete cpc_in;
+
+ if(cpc_out != NULL)
+ delete cpc_out;
+}
+
+int
+SimpleCpcClient::connect() {
+ struct sockaddr_in sa;
+ struct hostent *hp;
+
+ /* Create socket */
+ cpc_sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
+ if(cpc_sock < 0)
+ return -1;
+
+ /* Connect socket */
+ sa.sin_family = AF_INET;
+ hp = gethostbyname(host);
+ if(hp == NULL) {
+ errno = ENOENT;
+ return -1;
+ }
+
+ memcpy(&sa.sin_addr, hp->h_addr, hp->h_length);
+ sa.sin_port = htons(port);
+ if (::connect(cpc_sock, (struct sockaddr*) &sa, sizeof(sa)) < 0)
+ return -1;
+
+ cpc_in = new SocketInputStream(cpc_sock);
+ cpc_out = new SocketOutputStream(cpc_sock);
+
+ return 0;
+}
+
+int
+SimpleCpcClient::cpc_send(const char *cmd,
+ const Properties &args) {
+
+ cpc_out->println(cmd);
+
+ Properties::Iterator iter(&args);
+ const char *name;
+ while((name = iter.next()) != NULL) {
+ PropertiesType t;
+ Uint32 val_i;
+ BaseString val_s;
+
+ args.getTypeOf(name, &t);
+ switch(t) {
+ case PropertiesType_Uint32:
+ args.get(name, &val_i);
+ cpc_out->println("%s: %d", name, val_i);
+ break;
+ case PropertiesType_char:
+ args.get(name, val_s);
+ cpc_out->println("%s: %s", name, val_s.c_str());
+ break;
+ default:
+ /* Silently ignore */
+ break;
+ }
+ }
+ cpc_out->println("");
+
+ return 0;
+}
+
+/**
+ * Receive a response from the CPCD. The argument reply will point
+ * to a Properties object describing the reply. Note that the caller
+ * is responsible for deleting the Properties object returned.
+ */
+SimpleCpcClient::Parser_t::ParserStatus
+SimpleCpcClient::cpc_recv(const ParserRow_t *syntax,
+ const Properties **reply,
+ void **user_value) {
+ Parser_t::Context ctx;
+ ParserDummy session(cpc_sock);
+ Parser_t parser(syntax, *cpc_in, true, true, true);
+ *reply = parser.parse(ctx, session);
+ if(user_value != NULL)
+ *user_value = ctx.m_currentCmd->user_value;
+ return ctx.m_status;
+}
+
+const Properties *
+SimpleCpcClient::cpc_call(const char *cmd,
+ const Properties &args,
+ const ParserRow_t *reply_syntax) {
+ cpc_send(cmd, args);
+
+#if 0
+ Parser_t::Context ctx;
+ ParserDummy session(cpc_sock);
+ Parser_t parser(reply_syntax, *cpc_in, true, true, true);
+ const Properties *ret = parser.parse(ctx, session);
+ return ret;
+#endif
+ const Properties *ret;
+ cpc_recv(reply_syntax, &ret);
+ return ret;
+}
+
+
+SimpleCpcClient::ParserDummy::ParserDummy(NDB_SOCKET_TYPE sock)
+ : SocketServer::Session(sock) {
+}
+
diff --git a/ndb/src/mgmclient/CpcClient.hpp b/ndb/src/mgmclient/CpcClient.hpp
new file mode 100644
index 00000000000..1655bc57b56
--- /dev/null
+++ b/ndb/src/mgmclient/CpcClient.hpp
@@ -0,0 +1,104 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __CPCCLIENT_HPP_INCLUDED__
+#define __CPCCLIENT_HPP_INCLUDED__
+
+#include <Parser.hpp>
+#include <SocketServer.hpp>
+#include <util/InputStream.hpp>
+#include <util/OutputStream.hpp>
+
+/**
+ * Simple CPC client class. The whole management client should be replaced
+ * something smarter and more worked through.
+ */
+class SimpleCpcClient {
+public:
+ SimpleCpcClient(const char *host, int port);
+ ~SimpleCpcClient();
+
+ static void run(SimpleCpcClient &);
+
+ int getPort() const { return port;}
+ const char * getHost() const { return host;}
+
+ struct Process {
+ int m_id;
+ BaseString m_name;
+
+ BaseString m_owner;
+ BaseString m_group;
+ BaseString m_runas;
+
+ BaseString m_cwd;
+ BaseString m_env;
+ BaseString m_path;
+ BaseString m_args;
+
+ BaseString m_type;
+ BaseString m_status;
+
+ BaseString m_stdin;
+ BaseString m_stdout;
+ BaseString m_stderr;
+ BaseString m_ulimit;
+ };
+
+private:
+ class ParserDummy : SocketServer::Session {
+ public:
+ ParserDummy(NDB_SOCKET_TYPE sock);
+ };
+
+ typedef Parser<ParserDummy> Parser_t;
+ typedef ParserRow<ParserDummy> ParserRow_t;
+
+ char *host;
+ int port;
+ NDB_SOCKET_TYPE cpc_sock;
+ InputStream *cpc_in;
+ OutputStream *cpc_out;
+
+public:
+ int connect();
+
+ void cmd_list(char *arg);
+ void cmd_start(char *arg);
+ void cmd_stop(char *arg);
+ void cmd_help(char *arg);
+
+ int list_processes(Vector<Process>&, Properties &reply);
+ int start_process(Uint32 id, Properties& reply);
+ int stop_process(Uint32 id, Properties& reply);
+ int undefine_process(Uint32 id, Properties& reply);
+ int define_process(Process & p, Properties& reply);
+
+private:
+ int cpc_send(const char *cmd,
+ const Properties &args);
+
+
+ Parser_t::ParserStatus cpc_recv(const ParserRow_t *syntax,
+ const Properties **reply,
+ void **user_data = NULL);
+
+ const Properties *cpc_call(const char *cmd,
+ const Properties &args,
+ const ParserRow_t *reply_syntax);
+};
+
+#endif /* !__CPCCLIENT_HPP_INCLUDED__ */
diff --git a/ndb/src/mgmclient/Makefile b/ndb/src/mgmclient/Makefile
new file mode 100644
index 00000000000..9f9a6344ab9
--- /dev/null
+++ b/ndb/src/mgmclient/Makefile
@@ -0,0 +1,25 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := mgmtclient
+BIN_TARGET_LIBS :=
+BIN_TARGET_ARCHIVES := trace logger general mgmapi mgmsrvcommon portlib repapi
+
+BIN_TARGET_ARCHIVES += editline
+
+DIRS = test_cpcd
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = \
+ main.cpp \
+ CommandInterpreter.cpp \
+ CpcClient.cpp
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include/mgmapi) \
+ -I$(call fixpath,$(NDB_TOP)/src/common/mgmcommon)
+
+include $(NDB_TOP)/Epilogue.mk
+
+_bins_mkconfig : $(NDB_TOP)/bin/$(BIN_TARGET)
+
diff --git a/ndb/src/mgmclient/main.cpp b/ndb/src/mgmclient/main.cpp
new file mode 100644
index 00000000000..bbadaeb5206
--- /dev/null
+++ b/ndb/src/mgmclient/main.cpp
@@ -0,0 +1,105 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <sys/param.h>
+
+#include <NdbMain.h>
+#include <NdbHost.h>
+#include <util/getarg.h>
+#include <mgmapi.h>
+#include <LocalConfig.hpp>
+
+#include "CommandInterpreter.hpp"
+
+#include <signal.h>
+
+const char *progname = "mgmtclient";
+
+
+static CommandInterpreter* com;
+
+extern "C"
+void
+handler(int sig){
+ switch(sig){
+ case SIGPIPE:
+ /**
+ * Will happen when connection to mgmsrv is broken
+ * Reset connected flag
+ */
+ com->disconnect();
+ break;
+ }
+}
+
+int main(int argc, const char** argv){
+ int optind = 0;
+ const char *_default_connectstring = "host=localhost:2200;nodeid=0";
+ const char *_host = 0;
+ int _port = 0;
+ int _help = 0;
+ int _try_reconnect = 0;
+
+ struct getargs args[] = {
+ { "try-reconnect", 0, arg_integer, &_try_reconnect, "", "" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" },
+ };
+ int num_args = sizeof(args) / sizeof(args[0]); /* Number of arguments */
+
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, progname, "[host [port]]");
+ exit(1);
+ }
+
+ argv += optind;
+ argc -= optind;
+
+ LocalConfig cfg;
+
+ if(argc >= 1) {
+ _host = argv[0];
+ if(argc >= 2) {
+ _port = atoi(argv[1]);
+ }
+ } else {
+ if(cfg.init(false, 0, 0, _default_connectstring) && cfg.items > 0 && cfg.ids[0]->type == MgmId_TCP){
+ _host = cfg.ids[0]->data.tcp.remoteHost;
+ _port = cfg.ids[0]->data.tcp.port;
+ } else {
+ cfg.printError();
+ cfg.printUsage();
+ return 1;
+ }
+ }
+
+ char buf[MAXHOSTNAMELEN+10];
+ snprintf(buf, sizeof(buf), "%s:%d", _host, _port);
+
+ ndbout << "-- NDB Cluster -- Management Client --" << endl;
+ printf("Connecting to Management Server: %s\n", buf);
+
+ signal(SIGPIPE, handler);
+
+ com = new CommandInterpreter(buf);
+ while(com->readAndExecute(_try_reconnect));
+ delete com;
+
+ return 0;
+}
+
diff --git a/ndb/src/mgmclient/test_cpcd/Makefile b/ndb/src/mgmclient/test_cpcd/Makefile
new file mode 100644
index 00000000000..4ced10cfc59
--- /dev/null
+++ b/ndb/src/mgmclient/test_cpcd/Makefile
@@ -0,0 +1,17 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := test_cpcd
+BIN_TARGET_LIBS := general
+
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = test_cpcd.cpp ../CpcClient.cpp
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include/mgmapi) \
+ -I$(call fixpath,$(NDB_TOP)/src/common/mgmcommon)
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/src/mgmclient/test_cpcd/test_cpcd.cpp b/ndb/src/mgmclient/test_cpcd/test_cpcd.cpp
new file mode 100644
index 00000000000..6b6dc9f1077
--- /dev/null
+++ b/ndb/src/mgmclient/test_cpcd/test_cpcd.cpp
@@ -0,0 +1,152 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "../CpcClient.hpp"
+#include <Vector.hpp>
+#include <assert.h>
+#include <stdlib.h>
+
+SimpleCpcClient g_client("localhost", 1234);
+Vector<SimpleCpcClient::Process> g_procs;
+
+void define();
+void start(SimpleCpcClient::Process & p);
+void stop(SimpleCpcClient::Process & p);
+void undefine(SimpleCpcClient::Process & p);
+void list();
+SimpleCpcClient::Process* find(int id);
+
+#define ABORT() {ndbout_c("ABORT"); while(true); abort();}
+
+int name = 0;
+
+int
+main(void){
+
+ g_client.connect();
+
+ srand(time(0));
+ for(int i = 0; i<1000; i++){
+ int sz = g_procs.size();
+ int test = rand() % 100;
+ if(sz == 0 || test < 10){
+ define();
+ continue;
+ }
+
+ list();
+
+ int proc = rand() % g_procs.size();
+ SimpleCpcClient::Process & p = g_procs[proc];
+ if(p.m_status == "running" && test > 50){
+ ndbout_c("undefine %d: %s (running)", p.m_id, p.m_name.c_str());
+ undefine(p);
+ g_procs.erase(proc);
+ continue;
+ }
+ if(p.m_status == "running" && test <= 50){
+ ndbout_c("stop %d: %s(running)", p.m_id, p.m_name.c_str());
+ stop(p);
+ continue;
+ }
+ if(p.m_status == "stopped" && test > 50){
+ ndbout_c("undefine %d: %s(stopped)", p.m_id, p.m_name.c_str());
+ undefine(p);
+ g_procs.erase(proc);
+ continue;
+ }
+ if(p.m_status == "stopped" && test <= 50){
+ ndbout_c("start %d %s(stopped)", p.m_id, p.m_name.c_str());
+ start(p);
+ continue;
+ }
+ ndbout_c("Unknown: %s", p.m_status.c_str());
+ }
+}
+
+void define(){
+ SimpleCpcClient::Process m_proc;
+ m_proc.m_id = -1;
+ m_proc.m_type = "temporary";
+ m_proc.m_owner = "atrt";
+ m_proc.m_group = "group";
+ //m_proc.m_cwd
+ //m_proc.m_env
+ //proc.m_proc.m_stdout = "log.out";
+ //proc.m_proc.m_stderr = "2>&1";
+ //proc.m_proc.m_runas = proc.m_host->m_user;
+ //proc.m_proc.m_ulimit = "c:unlimited";
+ m_proc.m_name.assfmt("%d-%d-%s", getpid(), name++, "test");
+ m_proc.m_path.assign("/bin/sleep");
+ m_proc.m_args = "600";
+ g_procs.push_back(m_proc);
+
+ Properties reply;
+ if(g_client.define_process(g_procs.back(), reply) != 0){
+ ndbout_c("define %s -> ERR", m_proc.m_name.c_str());
+ reply.print();
+ ABORT();
+ }
+ ndbout_c("define %s -> %d", m_proc.m_name.c_str(), m_proc.m_id);
+}
+
+void start(SimpleCpcClient::Process & p){
+ Properties reply;
+ if(g_client.start_process(p.m_id, reply) != 0){
+ reply.print();
+ ABORT();
+ }
+}
+
+void stop(SimpleCpcClient::Process & p){
+ Properties reply;
+ if(g_client.stop_process(p.m_id, reply) != 0){
+ reply.print();
+ ABORT();
+ }
+}
+
+void undefine(SimpleCpcClient::Process & p){
+ Properties reply;
+ if(g_client.undefine_process(p.m_id, reply) != 0){
+ reply.print();
+ ABORT();
+ }
+}
+
+void list(){
+ Properties reply;
+ Vector<SimpleCpcClient::Process> procs;
+ if(g_client.list_processes(procs, reply) != 0){
+ reply.print();
+ ABORT();
+ }
+
+ for(int i = 0; i<procs.size(); i++){
+ SimpleCpcClient::Process * p = find(procs[i].m_id);
+ if(p != 0){
+ p->m_status = procs[i].m_status;
+ }
+ }
+}
+SimpleCpcClient::Process* find(int id){
+ for(int i = 0; i<g_procs.size(); i++){
+ if(g_procs[i].m_id == id)
+ return &g_procs[i];
+ }
+ return 0;
+}
diff --git a/ndb/src/mgmsrv/CommandInterpreter.cpp b/ndb/src/mgmsrv/CommandInterpreter.cpp
new file mode 100644
index 00000000000..1e608c05042
--- /dev/null
+++ b/ndb/src/mgmsrv/CommandInterpreter.cpp
@@ -0,0 +1,1240 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "CommandInterpreter.hpp"
+
+#include <string.h>
+#include <ctype.h>
+
+#include "MgmtSrvr.hpp"
+#include "MgmtErrorReporter.hpp"
+#include <NdbOut.hpp>
+#include "convertStrToInt.hpp"
+#include <EventLogger.hpp>
+#include <signaldata/SetLogLevelOrd.hpp>
+#include "ConfigInfo.hpp"
+
+extern "C"
+#include <version.h>
+
+
+static const char* helpTexts[] = {
+ "HELP Print help text",
+ "HELP SHOW Help for the SHOW command",
+#ifdef VM_TRACE // DEBUG ONLY
+ "HELP DEBUG Help for debug compiled version",
+#endif
+ "SHOW Print information about cluster",
+ "SHOW CONFIG Print configuration",
+ "SHOW PARAMETERS Print configuration parameters",
+ "START BACKUP Start backup\n"
+ "ABORT BACKUP <backup id> Aborts backup\n"
+ "CLUSTERLOG ON Enable Cluster logging",
+ "CLUSTERLOG OFF Disable Cluster logging",
+ "CLUSTERLOG FILTER <severity> Toggle severity filter on/off",
+ "CLUSTERLOG INFO Print cluster log information",
+ "{<id>|ALL} START Start DB node (started with -n)",
+ "{<id>|ALL} RESTART [-n] [-i] Restart DB node",
+ "{<id>|ALL} STOP Stop DB node",
+ "{<id>|ALL} STATUS Print status",
+ "{<id>|ALL} CLUSTERLOG {<category>=<level>}+ Set log level for cluster log",
+ "QUIT Quit management server",
+};
+static const int noOfHelpTexts = sizeof(helpTexts)/sizeof(const char*);
+
+static const char* helpTextShow =
+"SHOW prints NDB Cluster information\n\n"
+"SHOW Print information about cluster\n"
+"SHOW CONFIG Print configuration (in initial config file format)\n"
+"SHOW PARAMETERS Print information about configuration parameters\n\n"
+;
+
+#ifdef VM_TRACE // DEBUG ONLY
+static const char* helpTextDebug =
+"SHOW PROPERTIES Print config properties object\n"
+"{<id>|ALL} LOGLEVEL {<category>=<level>}+ Set log level\n"
+"{<id>|ALL} ERROR <errorNo> Inject error into NDB node\n"
+"{<id>|ALL} TRACE <traceNo> Set trace number\n"
+"{<id>|ALL} LOG [BLOCK = {ALL|<block>+}] Set logging on in & out signals\n"
+"{<id>|ALL} LOGIN [BLOCK = {ALL|<block>+}] Set logging on in signals\n"
+"{<id>|ALL} LOGOUT [BLOCK = {ALL|<block>+}] Set logging on out signals\n"
+"{<id>|ALL} LOGOFF [BLOCK = {ALL|<block>+}] Unset signal logging\n"
+"{<id>|ALL} TESTON Start signal logging\n"
+"{<id>|ALL} TESTOFF Stop signal logging\n"
+"{<id>|ALL} SET <configParamName> <value> Update configuration variable\n"
+"{<id>|ALL} DUMP <arg> Dump system state to cluster.log\n"
+"{<id>|ALL} GETSTAT Print statistics\n"
+"\n"
+;
+#endif
+
+
+
+//******************************************************************************
+//******************************************************************************
+CommandInterpreter::CommandInterpreter(MgmtSrvr& mgmtSrvr) :
+ _mgmtSrvr(mgmtSrvr) {
+
+ // _mgmtSrvr.setCallback(CmdBackupCallback);
+}
+
+
+bool emptyString(const char* s) {
+ if (s == NULL) {
+ return true;
+ }
+
+ for (unsigned int i = 0; i < strlen(s); ++i) {
+ if (! isspace(s[i])) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+class AutoPtr {
+public:
+ AutoPtr(void * ptr) : m_ptr(ptr) {}
+ ~AutoPtr() { free(m_ptr);}
+private:
+ void * m_ptr;
+};
+
+//*****************************************************************************
+//*****************************************************************************
+int CommandInterpreter::readAndExecute() {
+
+ char* _line = readline_gets();
+ char * line;
+ if(_line == NULL) {
+ ndbout << endl;
+ return true;
+ }
+
+ line = strdup(_line);
+
+ AutoPtr ptr(line);
+
+ if (emptyString(line)) {
+ return true;
+ }
+
+ for (unsigned int i = 0; i < strlen(line); ++i) {
+ line[i] = toupper(line[i]);
+ }
+
+ // if there is anything in the line proceed
+ char* firstToken = strtok(line, " ");
+ char* allAfterFirstToken = strtok(NULL, "\0");
+
+ if (strcmp(firstToken, "HELP") == 0) {
+ executeHelp(allAfterFirstToken);
+ return true;
+ }
+ else if (strcmp(firstToken, "?") == 0) {
+ executeHelp(allAfterFirstToken);
+ return true;
+ }
+ else if (strcmp(firstToken, "SHOW") == 0) {
+ executeShow(allAfterFirstToken);
+ return true;
+ }
+ else if (strcmp(firstToken, "CLUSTERLOG") == 0) {
+ executeClusterLog(allAfterFirstToken);
+ return true;
+ }
+ else if(strcmp(firstToken, "START") == 0 &&
+ allAfterFirstToken != 0 &&
+ strncmp(allAfterFirstToken, "BACKUP", sizeof("BACKUP") - 1) == 0){
+ executeStartBackup(allAfterFirstToken);
+ return true;
+ }
+ else if(strcmp(firstToken, "ABORT") == 0 &&
+ allAfterFirstToken != 0 &&
+ strncmp(allAfterFirstToken, "BACKUP", sizeof("BACKUP") - 1) == 0){
+ executeAbortBackup(allAfterFirstToken);
+ return true;
+ }
+
+ else if(strcmp(firstToken, "ENTER") == 0 &&
+ allAfterFirstToken != 0 &&
+ strncmp(allAfterFirstToken, "SINGLE USER MODE ",
+ sizeof("SINGLE USER MODE") - 1) == 0){
+ executeEnterSingleUser(allAfterFirstToken);
+ return true;
+ }
+
+ else if(strcmp(firstToken, "EXIT") == 0 &&
+ allAfterFirstToken != 0 &&
+ strncmp(allAfterFirstToken, "SINGLE USER MODE ",
+ sizeof("SINGLE USER MODE") - 1) == 0){
+ executeExitSingleUser(allAfterFirstToken);
+ return true;
+ }
+
+ else if (strcmp(firstToken, "ALL") == 0) {
+ analyseAfterFirstToken(-1, allAfterFirstToken);
+ }
+ else if(strcmp(firstToken, "QUIT") == 0 ||
+ strcmp(firstToken, "EXIT") == 0 ||
+ strcmp(firstToken, "BYE") == 0){
+ return false;
+ } else {
+ // First token should be a digit, process ID
+
+ int processId;
+ if (! convert(firstToken, processId)) {
+ ndbout << "Invalid command: " << _line << "." << endl;
+ ndbout << "Type HELP for help." << endl << endl;
+ return true;
+ }
+ if (processId < 0) {
+ ndbout << "Invalid process ID: " << firstToken << "." << endl;
+ return true;
+ }
+
+ analyseAfterFirstToken(processId, allAfterFirstToken);
+
+ } // else
+ return true;
+}
+
+
+static const CommandInterpreter::CommandFunctionPair commands[] = {
+ { "START", &CommandInterpreter::executeStart }
+ ,{ "RESTART", &CommandInterpreter::executeRestart }
+ ,{ "STOP", &CommandInterpreter::executeStop }
+ ,{ "STATUS", &CommandInterpreter::executeStatus }
+ ,{ "LOGLEVEL", &CommandInterpreter::executeLogLevel }
+#ifdef ERROR_INSERT
+ ,{ "ERROR", &CommandInterpreter::executeError }
+#endif
+ ,{ "TRACE", &CommandInterpreter::executeTrace }
+ ,{ "LOG", &CommandInterpreter::executeLog }
+ ,{ "LOGIN", &CommandInterpreter::executeLogIn }
+ ,{ "LOGOUT", &CommandInterpreter::executeLogOut }
+ ,{ "LOGOFF", &CommandInterpreter::executeLogOff }
+ ,{ "TESTON", &CommandInterpreter::executeTestOn }
+ ,{ "TESTOFF", &CommandInterpreter::executeTestOff }
+ ,{ "CLUSTERLOG", &CommandInterpreter::executeEventReporting }
+ ,{ "DUMP", &CommandInterpreter::executeDumpState }
+ ,{ "JONAS", &CommandInterpreter::jonas }
+};
+
+
+//*****************************************************************************
+//*****************************************************************************
+void
+CommandInterpreter::analyseAfterFirstToken(int processId,
+ char* allAfterFirstToken) {
+
+ if (emptyString(allAfterFirstToken)) {
+ if (processId == -1) {
+ ndbout << "Expected a command after ALL." << endl;
+ }
+ else {
+ ndbout << "Expected a command after process ID." << endl;
+ }
+ return;
+ }
+
+
+ char* secondToken = strtok(allAfterFirstToken, " ");
+ char* allAfterSecondToken = strtok(NULL, "\0");
+
+ const int tmpSize = sizeof(commands)/sizeof(CommandFunctionPair);
+ ExecuteFunction fun = 0;
+ const char * command = 0;
+ for(int i = 0; i<tmpSize; i++){
+ if(strcmp(secondToken, commands[i].command) == 0){
+ fun = commands[i].executeFunction;
+ command = commands[i].command;
+ break;
+ }
+ }
+
+ if(fun == 0){
+ ndbout << "Invalid command: " << secondToken << "." << endl;
+ ndbout << "Type HELP for help." << endl << endl;
+ return;
+ }
+
+ if(processId == -1){
+ executeForAll(command, fun, allAfterSecondToken);
+ } else {
+ if(strcmp(command, "STATUS") != 0)
+ ndbout << "Executing " << command << " on node: "
+ << processId << endl << endl;
+ (this->*fun)(processId, allAfterSecondToken, false);
+ ndbout << endl;
+ }
+}
+
+void
+CommandInterpreter::executeForAll(const char * cmd, ExecuteFunction fun,
+ const char * allAfterSecondToken){
+
+ NodeId nodeId = 0;
+ if(strcmp(cmd, "STOP") == 0 ||
+ strcmp(cmd, "RESTART") == 0){
+ ndbout << "Executing " << cmd << " on all nodes" << endl << "\n";
+ (this->*fun)(nodeId, allAfterSecondToken, true);
+ } else {
+ while(_mgmtSrvr.getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)){
+ if(strcmp(cmd, "STATUS") != 0)
+ ndbout << "Executing " << cmd << " on node: "
+ << nodeId << endl << endl;
+ (this->*fun)(nodeId, allAfterSecondToken, true);
+ ndbout << endl;
+ } // for
+ }
+}
+
+//*****************************************************************************
+//*****************************************************************************
+bool CommandInterpreter::parseBlockSpecification(const char* allAfterLog,
+ Vector<BaseString>& blocks) {
+
+ // Parse: [BLOCK = {ALL|<blockName>+}]
+
+ if (emptyString(allAfterLog)) {
+ return true;
+ }
+
+ // Copy allAfterLog since strtok will modify it
+ char* newAllAfterLog = strdup(allAfterLog);
+ char* firstTokenAfterLog = strtok(newAllAfterLog, " ");
+ for (unsigned int i = 0; i < strlen(firstTokenAfterLog); ++i) {
+ firstTokenAfterLog[i] = toupper(firstTokenAfterLog[i]);
+ }
+
+ if (strcmp(firstTokenAfterLog, "BLOCK") != 0) {
+ ndbout << "Unexpected value: " << firstTokenAfterLog
+ << ". Expected BLOCK." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+
+ char* allAfterFirstToken = strtok(NULL, "\0");
+ if (emptyString(allAfterFirstToken)) {
+ ndbout << "Expected =." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+
+ char* secondTokenAfterLog = strtok(allAfterFirstToken, " ");
+ if (strcmp(secondTokenAfterLog, "=") != 0) {
+ ndbout << "Unexpected value: " << secondTokenAfterLog
+ << ". Expected =." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+
+ char* blockName = strtok(NULL, " ");
+ bool all = false;
+ if (blockName != NULL && (strcmp(blockName, "ALL") == 0)) {
+ all = true;
+ }
+ while (blockName != NULL) {
+ blocks.push_back(BaseString(blockName));
+ blockName = strtok(NULL, " ");
+ }
+
+ if (blocks.size() == 0) {
+ ndbout << "No block specified." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+ if (blocks.size() > 1 && all) {
+ // More than "ALL" specified
+ ndbout << "Nothing expected after ALL." << endl;
+ free(newAllAfterLog);
+ return false;
+ }
+
+ free(newAllAfterLog);
+ return true;
+}
+
+
+
+//******************************************************************************
+//******************************************************************************
+void CommandInterpreter::executeHelp(char* parameters) {
+
+ (void)parameters; // Don't want compiler warning
+
+ if (emptyString(parameters)) {
+ for (int i = 0; i<noOfHelpTexts; i++) {
+ ndbout << helpTexts[i] << endl;
+ }
+
+ ndbout << endl
+ << "<severity> = "
+ << "ALERT | CRITICAL | ERROR | WARNING | INFO | DEBUG"
+ << endl;
+
+ ndbout << "<category> = ";
+ for(Uint32 i = 0; i<EventLogger::noOfEventCategoryNames; i++){
+ ndbout << EventLogger::eventCategoryNames[i].name;
+ if (i < EventLogger::noOfEventCategoryNames - 1) {
+ ndbout << " | ";
+ }
+ }
+ ndbout << endl;
+
+ ndbout << "<level> = " << "0 - 15"
+ << endl;
+
+ ndbout << endl;
+ } else if (strcmp(parameters, "SHOW") == 0) {
+ ndbout << helpTextShow;
+#ifdef VM_TRACE // DEBUG ONLY
+ } else if (strcmp(parameters, "DEBUG") == 0) {
+ ndbout << helpTextDebug;
+#endif
+ } else {
+ ndbout << "Invalid argument." << endl;
+ }
+}
+
+//*****************************************************************************
+//*****************************************************************************
+
+void CommandInterpreter::executeShow(char* parameters) {
+
+ if (emptyString(parameters)) {
+ ndbout << "Cluster Configuration" << endl
+ << "---------------------" << endl;
+
+ NodeId nodeId = 0;
+ ndbout << _mgmtSrvr.getNodeCount(NDB_MGM_NODE_TYPE_NDB)
+ << " NDB Node(s) with"
+ << endl;
+ while (_mgmtSrvr.getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)){
+ ndbout << " Node Id = " << nodeId << endl;
+ }
+ ndbout << endl;
+
+ nodeId = 0;
+ ndbout << _mgmtSrvr.getNodeCount(NDB_MGM_NODE_TYPE_API)
+ << " API Node(s) with"
+ << endl;
+ while (_mgmtSrvr.getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_API)){
+ ndbout << " Node Id = " << nodeId << endl;
+ }
+ ndbout << endl;
+
+ nodeId = 0;
+ ndbout << _mgmtSrvr.getNodeCount(NDB_MGM_NODE_TYPE_MGM)
+ << " MGM Node(s) with"
+ << endl;
+ while (_mgmtSrvr.getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_MGM)){
+ ndbout << " Node Id = " << nodeId << endl;
+ }
+ ndbout << endl;
+
+ ndbout << helpTextShow;
+
+ return;
+ } else if (strcmp(parameters, "PROPERTIES") == 0 ||
+ strcmp(parameters, "PROP") == 0) {
+ _mgmtSrvr.getConfig()->print();
+ } else if (strcmp(parameters, "CONFIGURATION") == 0 ||
+ strcmp(parameters, "CONFIG") == 0){
+ _mgmtSrvr.getConfig()->printConfigFile();
+ } else if (strcmp(parameters, "PARAMETERS") == 0 ||
+ strcmp(parameters, "PARAMS") == 0 ||
+ strcmp(parameters, "PARAM") == 0) {
+ _mgmtSrvr.getConfig()->getConfigInfo()->print();
+ } else {
+ ndbout << "Invalid argument." << endl;
+ }
+}
+
+
+//*****************************************************************************
+//*****************************************************************************
+void CommandInterpreter::executeClusterLog(char* parameters) {
+
+ if (parameters != 0 && strlen(parameters) != 0) {
+ int severity = 7;
+ int isOk = true;
+ char name[12];
+ bool noArgs = false;
+
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, " ", &tmpPtr);
+
+ /********************
+ * CLUSTERLOG FILTER
+ ********************/
+ if (strcmp(item, "FILTER") == 0) {
+
+ item = strtok_r(NULL, " ", &tmpPtr);
+ if (item == NULL) {
+ noArgs = true;
+ }
+ while (item != NULL) {
+ snprintf(name, 12, item);
+
+ if (strcmp(item, "ALL") == 0) {
+ severity = 7;
+ } else if (strcmp(item, "ALERT") == 0) {
+ severity = 6;
+ } else if (strcmp(item, "CRITICAL") == 0) {
+ severity = 5;
+ } else if (strcmp(item, "ERROR") == 0) {
+ severity = 4;
+ } else if (strcmp(item, "WARNING") == 0) {
+ severity = 3;
+ } else if (strcmp(item, "INFO") == 0) {
+ severity = 2;
+ } else if (strcmp(item, "DEBUG") == 0) {
+ severity = 1;
+ } else if (strcmp(item, "OFF") == 0) {
+ severity = 0;
+ } else {
+ isOk = false;
+ }
+
+ item = strtok_r(NULL, " ", &tmpPtr);
+ } // while(item != NULL){
+ free(tmpString);
+
+ if (noArgs) {
+ ndbout << "Missing argument(s)." << endl;
+ } else if (isOk) {
+ if (_mgmtSrvr.setEventLogFilter(severity)) {
+ if(strcmp(name, "ALL") == 0 || strcmp(name, "all") == 0) {
+ ndbout << "All severities levels enabled." << endl;
+ } else if(strcmp(name, "OFF") == 0 || strcmp(name, "off") == 0) {
+ ndbout << "Cluster logging disabled." << endl;
+ } else {
+ ndbout << name << " events enabled." << endl;
+ }
+ } else {
+ if(strcmp(name, "ALL") == 0) {
+ ndbout << "All severities levels disabled." << endl;
+ } else if(strcmp(name, "OFF") == 0) {
+ ndbout << "Cluster logging enabled." << endl;
+ } else {
+ ndbout << name << " events disabled." << endl;
+ }
+ }
+ } else {
+ ndbout << "Invalid severity level." << endl;
+ }
+
+ /********************
+ * CLUSTERLOG INFO
+ ********************/
+ } else if (strcmp(item, "INFO") == 0) {
+ const char* names[] = {"DEBUG", "INFO", "WARNING", "ERROR",
+ "CRITICAL", "ALERT"};
+ if (_mgmtSrvr.isEventLogFilterEnabled(0)) { // OFF
+ ndbout << "Cluster logging is disabled." << endl;
+ }
+
+ ndbout << "Severities enabled: ";
+ for (int i = 0; i < 6; i++) {
+ if (_mgmtSrvr.isEventLogFilterEnabled(i + 1)) {
+ ndbout << names[i] << " ";
+ }
+ }
+ ndbout << endl;
+
+ /********************
+ * CLUSTERLOG OFF
+ ********************/
+ } else if (strcmp(item, "OFF") == 0) {
+ if (!_mgmtSrvr.isEventLogFilterEnabled(0)) { // ON
+ if (_mgmtSrvr.setEventLogFilter(0));
+ ndbout << "Cluster logging is disabled." << endl;
+ } else {
+ ndbout << "Cluster logging is already disabled." << endl;
+ }
+
+ /********************
+ * CLUSTERLOG ON
+ ********************/
+ } else if (strcmp(item, "ON") == 0) {
+ if (_mgmtSrvr.isEventLogFilterEnabled(0)) { // OFF
+ if (_mgmtSrvr.setEventLogFilter(0));
+ ndbout << "Cluster logging is enabled." << endl;
+ } else {
+ ndbout << "Cluster logging is already enabled." << endl;
+ }
+
+ } else {
+ ndbout << "Invalid argument." << endl;
+ }
+
+ } else {
+ ndbout << "Missing argument." << endl;
+ }
+}
+
+void
+stopCallback(int nodeId, void * anyData, int errCode){
+ if(errCode == 0){
+ if(nodeId == 0)
+ ndbout << "\nCluster has shutdown" << endl;
+ else
+ ndbout << "\nNode " << nodeId << " has shutdown" << endl;
+ } else {
+ MgmtSrvr * mgm = (MgmtSrvr *)anyData;
+ ndbout << "Node " << nodeId << " has not shutdown: "
+ << mgm->getErrorText(errCode) << endl;
+ }
+}
+
+void
+versionCallback(int nodeId, int version, void * anyData, int errCode){
+ if(errCode == 0){
+ MgmtSrvr * mgm = (MgmtSrvr *)anyData;
+ switch(mgm->getNodeType(nodeId)){
+ case NDB_MGM_NODE_TYPE_MGM:
+ {
+ ndbout << "MGMT node:\t" << nodeId << " ";
+ ndbout_c(" (Version %d.%d.%d)",
+ getMajor(version) ,
+ getMinor(version),
+ getBuild(version));
+ }
+ break;
+ case NDB_MGM_NODE_TYPE_NDB:
+ {
+ ndbout << "DB node:\t" << nodeId << " ";
+ if(version == 0)
+ ndbout << "(no version information available)" << endl;
+ else {
+ ndbout_c(" (Version %d.%d.%d)",
+ getMajor(version) ,
+ getMinor(version),
+ getBuild(version));
+ }
+ }
+ break;
+ case NDB_MGM_NODE_TYPE_API:
+ {
+ ndbout << "API node:\t" << nodeId << " ";
+ if(version == 0)
+ ndbout << "(no version information available)" << endl;
+ else {
+ ndbout_c(" (Version %d.%d.%d)",
+ getMajor(version) ,
+ getMinor(version),
+ getBuild(version));
+ }
+
+ }
+ break;
+ };
+
+ } else {
+ MgmtSrvr * mgm = (MgmtSrvr *)anyData;
+ ndbout << mgm->getErrorText(errCode) << endl;
+ }
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void CommandInterpreter::executeStop(int processId,
+ const char* parameters, bool all) {
+
+ (void)parameters; // Don't want compiler warning
+
+ int result = 0;
+ if(all)
+ result = _mgmtSrvr.stop((int *)0, false, stopCallback, this);
+ else
+ result = _mgmtSrvr.stopNode(processId, false, stopCallback, this);
+
+ if(result != 0)
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+}
+
+
+void CommandInterpreter::executeStart(int processId, const char* parameters,
+ bool all) {
+ (void)all; // Don't want compiler warning
+
+ if (! emptyString(parameters)) {
+ ndbout << "No parameters expected to this command." << endl;
+ return;
+ }
+
+ int result = _mgmtSrvr.start(processId);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+}
+
+void
+CommandInterpreter::executeRestart(int processId, const char* parameters,
+ bool all) {
+
+ bool nostart = false;
+ bool initialstart = false;
+
+ if(parameters != 0 && strlen(parameters) != 0){
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, " ", &tmpPtr);
+ while(item != NULL){
+ if(strcmp(item, "-N") == 0)
+ nostart = true;
+ if(strcmp(item, "-I") == 0)
+ initialstart = true;
+ item = strtok_r(NULL, " ", &tmpPtr);
+ }
+ free(tmpString);
+ }
+ int result;
+ if(all)
+ result = _mgmtSrvr.restart(nostart, initialstart, false,
+ 0, stopCallback, this);
+ else
+ result = _mgmtSrvr.restartNode(processId, nostart, initialstart, false,
+ stopCallback,
+ this);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+}
+
+void
+CommandInterpreter::executeDumpState(int processId, const char* parameters,
+ bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ if(parameters == 0 || strlen(parameters) == 0){
+ ndbout << "Expected argument" << endl;
+ return;
+ }
+
+ Uint32 no = 0;
+ Uint32 pars[25];
+
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, " ", &tmpPtr);
+ while(item != NULL){
+ if (0x0 <= strtoll(item, NULL, 0) && strtoll(item, NULL, 0) <= 0xffffffff) {
+ pars[no] = strtoll(item, NULL, 0);
+ } else {
+ ndbout << "Illegal value in argument to signal." << endl
+ << "(Value must be between 0 and 0xffffffff.)"
+ << endl;
+ return;
+ }
+ no++;
+ item = strtok_r(NULL, " ", &tmpPtr);
+ }
+ ndbout << "Sending dump signal with data:" << endl;
+ for (Uint32 i=0; i<no; i++) {
+ ndbout.setHexFormat(1) << pars[i] << " ";
+ if (!(i+1 & 0x3)) ndbout << endl;
+ }
+ free(tmpString);
+ int result = _mgmtSrvr.dumpState(processId, pars, no);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+}
+
+void CommandInterpreter::executeStatus(int processId,
+ const char* parameters, bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ if (! emptyString(parameters)) {
+ ndbout << "No parameters expected to this command." << endl;
+ return;
+ }
+
+ ndb_mgm_node_status status;
+ Uint32 startPhase, version, dynamicId, nodeGroup;
+ bool system;
+ int result = _mgmtSrvr.status(processId,
+ &status, &version, &startPhase, &system,
+ &dynamicId, &nodeGroup);
+ if(result != 0){
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ return;
+ }
+
+ ndbout << "Node " << processId << ": ";
+ switch(status){
+ case NDB_MGM_NODE_STATUS_NO_CONTACT:
+ ndbout << "No contact" << endl;
+ break;
+ case NDB_MGM_NODE_STATUS_NOT_STARTED:
+ ndbout << "Not started" ;
+ break;
+ case NDB_MGM_NODE_STATUS_STARTING:
+ ndbout << "Starting (Start phase " << startPhase << ")" ;
+ break;
+ case NDB_MGM_NODE_STATUS_STARTED:
+ ndbout << "Started" ;
+ break;
+ case NDB_MGM_NODE_STATUS_SHUTTING_DOWN:
+ ndbout << "Shutting down " << (system == false ? "node" : "system")
+ << " (Phase " << startPhase << ")"
+ ;
+ break;
+ case NDB_MGM_NODE_STATUS_RESTARTING:
+ ndbout << "Restarting" ;
+ break;
+ case NDB_MGM_NODE_STATUS_SINGLEUSER:
+ ndbout << "Single user mode" ;
+ break;
+ default:
+ ndbout << "Unknown state" ;
+ break;
+ }
+ if(status != NDB_MGM_NODE_STATUS_NO_CONTACT){
+
+ ndbout_c(" (Version %d.%d.%d)",
+ getMajor(version) ,
+ getMinor(version),
+ getBuild(version));
+
+ // NOTE It's possible to print dynamicId and nodeGroup here ...
+ // ndbout << ", " <<dynamicId<<", "<<nodeGroup<<endl;
+ }
+}
+
+
+
+//*****************************************************************************
+//*****************************************************************************
+void CommandInterpreter::executeLogLevel(int processId,
+ const char* parameters, bool all) {
+ (void)all; // Don't want compiler warning
+ SetLogLevelOrd logLevel; logLevel.clear();
+
+ if (emptyString(parameters) || (strcmp(parameters, "ALL") == 0)) {
+ for(Uint32 i = 0; i<EventLogger::noOfEventCategoryNames; i++)
+ logLevel.setLogLevel(EventLogger::eventCategoryNames[i].category, 7);
+ } else {
+
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, ", ", &tmpPtr);
+ while(item != NULL){
+ char categoryTxt[255];
+ int level;
+ const int m = sscanf(item, "%[^=]=%d", categoryTxt, &level);
+ if(m != 2){
+ free(tmpString);
+ ndbout << "Invalid loglevel specification category=level" << endl;
+ return;
+ }
+ LogLevel::EventCategory cat;
+ if(!EventLogger::matchEventCategory(categoryTxt,
+ &cat)){
+ ndbout << "Invalid loglevel specification, unknown category: "
+ << categoryTxt << endl;
+ free(tmpString);
+ return ;
+ }
+ if(level < 0 || level > 15){
+ ndbout << "Invalid loglevel specification row, level 0-15" << endl;
+ free(tmpString);
+ return ;
+ }
+ logLevel.setLogLevel(cat, level);
+
+ item = strtok_r(NULL, ", ", &tmpPtr);
+ }
+ free(tmpString);
+ }
+
+ int result = _mgmtSrvr.setNodeLogLevel(processId, logLevel);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+}
+
+
+
+//*****************************************************************************
+//*****************************************************************************
+void CommandInterpreter::executeError(int processId,
+ const char* parameters, bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ if (emptyString(parameters)) {
+ ndbout << "Missing error number." << endl;
+ return;
+ }
+ // Copy parameters since strtok will modify it
+ char* newpar = strdup(parameters);
+ char* firstParameter = strtok(newpar, " ");
+
+ int errorNo;
+ if (! convert(firstParameter, errorNo)) {
+ ndbout << "Expected an integer." << endl;
+ free(newpar);
+ return;
+ }
+
+ char* allAfterFirstParameter = strtok(NULL, "\0");
+ if (! emptyString(allAfterFirstParameter)) {
+ ndbout << "Nothing expected after error number." << endl;
+ free(newpar);
+ return;
+ }
+
+ int result = _mgmtSrvr.insertError(processId, errorNo);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+ free(newpar);
+}
+
+
+
+//******************************************************************************
+//******************************************************************************
+void CommandInterpreter::executeTrace(int processId,
+ const char* parameters, bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ if (emptyString(parameters)) {
+ ndbout << "Missing trace number." << endl;
+ return;
+ }
+
+ char* newpar = strdup(parameters);
+ char* firstParameter = strtok(newpar, " ");
+
+
+ int traceNo;
+ if (! convert(firstParameter, traceNo)) {
+ ndbout << "Expected an integer." << endl;
+ free(newpar);
+ return;
+ }
+
+ char* allAfterFirstParameter = strtok(NULL, "\0");
+
+ if (! emptyString(allAfterFirstParameter)) {
+ ndbout << "Nothing expected after trace number." << endl;
+ free(newpar);
+ return;
+ }
+
+ int result = _mgmtSrvr.setTraceNo(processId, traceNo);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+ free(newpar);
+}
+
+
+
+//******************************************************************************
+//******************************************************************************
+void CommandInterpreter::executeLog(int processId,
+ const char* parameters, bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ Vector<BaseString> blocks;
+ if (! parseBlockSpecification(parameters, blocks)) {
+ return;
+ }
+
+ int result = _mgmtSrvr.setSignalLoggingMode(processId, MgmtSrvr::InOut, blocks);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+
+}
+
+
+
+//******************************************************************************
+//******************************************************************************
+void CommandInterpreter::executeLogIn(int processId,
+ const char* parameters, bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ Vector<BaseString> blocks;
+ if (! parseBlockSpecification(parameters, blocks)) {
+ return;
+ }
+
+ int result = _mgmtSrvr.setSignalLoggingMode(processId, MgmtSrvr::In, blocks);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+}
+
+//******************************************************************************
+//******************************************************************************
+void CommandInterpreter::executeLogOut(int processId,
+ const char* parameters, bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ Vector<BaseString> blocks;
+ if (! parseBlockSpecification(parameters, blocks)) {
+ return;
+ }
+
+
+ int result = _mgmtSrvr.setSignalLoggingMode(processId, MgmtSrvr::Out, blocks);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+
+}
+
+
+//******************************************************************************
+//******************************************************************************
+void CommandInterpreter::executeLogOff(int processId,
+ const char* parameters, bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ Vector<BaseString> blocks;
+ if (! parseBlockSpecification(parameters, blocks)) {
+ return;
+ }
+
+
+ int result = _mgmtSrvr.setSignalLoggingMode(processId, MgmtSrvr::Off, blocks);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+
+}
+
+//******************************************************************************
+//******************************************************************************
+void CommandInterpreter::executeTestOn(int processId,
+ const char* parameters, bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ if (! emptyString(parameters)) {
+ ndbout << "No parameters expected to this command." << endl;
+ return;
+ }
+
+ int result = _mgmtSrvr.startSignalTracing(processId);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+
+}
+
+//******************************************************************************
+//******************************************************************************
+void CommandInterpreter::executeTestOff(int processId,
+ const char* parameters, bool all) {
+
+ (void)all; // Don't want compiler warning
+
+ if (! emptyString(parameters)) {
+ ndbout << "No parameters expected to this command." << endl;
+ return;
+ }
+
+ int result = _mgmtSrvr.stopSignalTracing(processId);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+
+}
+
+//*****************************************************************************
+//*****************************************************************************
+void CommandInterpreter::executeEventReporting(int processId,
+ const char* parameters,
+ bool all) {
+ (void)all; // Don't want compiler warning
+ SetLogLevelOrd logLevel; logLevel.clear();
+
+ if (emptyString(parameters) || (strcmp(parameters, "ALL") == 0)) {
+ for(Uint32 i = 0; i<EventLogger::noOfEventCategoryNames; i++)
+ logLevel.setLogLevel(EventLogger::eventCategoryNames[i].category, 7);
+ } else {
+
+ char * tmpString = strdup(parameters);
+ char * tmpPtr = 0;
+ char * item = strtok_r(tmpString, ", ", &tmpPtr);
+ while(item != NULL){
+ char categoryTxt[255];
+ int level;
+ const int m = sscanf(item, "%[^=]=%d", categoryTxt, &level);
+ if(m != 2){
+ free(tmpString);
+ ndbout << "Invalid loglevel specification category=level" << endl;
+ return;
+ }
+ LogLevel::EventCategory cat;
+ if(!EventLogger::matchEventCategory(categoryTxt,
+ &cat)){
+ ndbout << "Invalid loglevel specification, unknown category: "
+ << categoryTxt << endl;
+ free(tmpString);
+ return ;
+ }
+ if(level < 0 || level > 15){
+ ndbout << "Invalid loglevel specification row, level 0-15" << endl;
+ free(tmpString);
+ return ;
+ }
+ logLevel.setLogLevel(cat, level);
+
+ item = strtok_r(NULL, ", ", &tmpPtr);
+ }
+ free(tmpString);
+ }
+ ndbout_c("processId %d", processId);
+ int result = _mgmtSrvr.setEventReportingLevel(processId, logLevel);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ }
+}
+
+void
+CommandInterpreter::executeStartBackup(char* parameters) {
+ Uint32 backupId;
+ int result = _mgmtSrvr.startBackup(backupId);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ } else {
+ // ndbout << "Start of backup ordered" << endl;
+ }
+}
+
+void
+CommandInterpreter::executeAbortBackup(char* parameters) {
+ strtok(parameters, " ");
+ char* id = strtok(NULL, "\0");
+ int bid = -1;
+ if(id == 0 || sscanf(id, "%d", &bid) != 1){
+ ndbout << "Invalid arguments: expected <BackupId>" << endl;
+ return;
+ }
+ int result = _mgmtSrvr.abortBackup(bid);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ } else {
+ ndbout << "Abort of backup " << bid << " ordered" << endl;
+ }
+}
+
+
+
+void
+CommandInterpreter::executeEnterSingleUser(char* parameters) {
+ strtok(parameters, " ");
+ char* id = strtok(NULL, " ");
+ id = strtok(NULL, " ");
+ id = strtok(NULL, "\0");
+ int nodeId = -1;
+ if(id == 0 || sscanf(id, "%d", &nodeId) != 1){
+ ndbout << "Invalid arguments: expected <NodeId>" << endl;
+ return;
+ }
+ int result = _mgmtSrvr.enterSingleUser(0, nodeId,0,0);
+ if (result != 0) {
+ ndbout << _mgmtSrvr.getErrorText(result) << endl;
+ } else {
+ ndbout << "Entering single user mode, granting access for node "
+ << nodeId << " OK." << endl;
+ }
+}
+
+void CommandInterpreter::executeExitSingleUser(char* parameters) {
+ _mgmtSrvr.exitSingleUser(0,0,0,0);
+}
+
+
+#include <NdbApiSignal.hpp>
+
+void
+CommandInterpreter::jonas(int processId, const char* parameters, bool all) {
+
+ MgmtSrvr::Area51 tmp = _mgmtSrvr.getStuff();
+
+ NdbApiSignal signal(0);
+ Uint32 * theData = signal.getDataPtrSend();
+ Uint32 data[25];
+ Uint32 sec0[70];
+ Uint32 sec1[123];
+
+ data[0] = 12;
+ data[1] = 13;
+
+ for(Uint32 i = 0; i<70; i++)
+ sec0[i] = i;
+
+ for(Uint32 i = 0; i<123; i++)
+ sec1[i] = 70+i;
+
+ signal.set(0, CMVMI, GSN_TESTSIG, 3);
+ signal.m_noOfSections = 2;
+ signal.m_fragmentInfo = 1;
+
+ LinearSectionPtr ptr[3];
+
+ theData[0] = 3;
+ theData[1] = 0;
+ theData[2] = 7; // FragmentId
+
+ ptr[0].sz = 2;
+ ptr[0].p = &data[0];
+
+ ptr[1].sz = 60;
+ ptr[1].p = &sec0[0];
+
+ tmp.theFacade->lock_mutex();
+ tmp.theRegistry->prepareSend(&signal, 1, theData, processId, ptr);
+ tmp.theFacade->unlock_mutex();
+
+ signal.set(0, CMVMI, GSN_TESTSIG, 3);
+ signal.m_noOfSections = 2;
+ signal.m_fragmentInfo = 3;
+
+ theData[0] = 0;
+ theData[1] = 1;
+ theData[2] = 7; // FragmentId
+
+ ptr[0].sz = 10;
+ ptr[0].p = &sec0[60];
+
+ ptr[1].sz = 123;
+ ptr[1].p = &sec1[0];
+
+ tmp.theFacade->lock_mutex();
+ tmp.theRegistry->prepareSend(&signal, 1, theData, processId, ptr);
+ tmp.theFacade->unlock_mutex();
+}
diff --git a/ndb/src/mgmsrv/CommandInterpreter.hpp b/ndb/src/mgmsrv/CommandInterpreter.hpp
new file mode 100644
index 00000000000..e68aa7da084
--- /dev/null
+++ b/ndb/src/mgmsrv/CommandInterpreter.hpp
@@ -0,0 +1,176 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CommandInterpreter_H
+#define CommandInterpreter_H
+
+//*****************************************************************************
+// Author: Peter Lind
+//*****************************************************************************
+
+#include <NdbStdio.h>
+#include <ndb_types.h>
+#include <stdlib.h>
+#include <Vector.hpp>
+#include <editline/editline.h>
+#include <BaseString.hpp>
+
+class MgmtSrvr;
+
+/**
+ * @class CommandInterpreter
+ * @brief Reads command line in management client
+ *
+ * This class has one public method which reads a command line
+ * from a stream. It then interpret that commmand line and calls a suitable
+ * method in the MgmtSrvr class which executes the command.
+ *
+ * For command syntax, see the HELP command.
+ */
+class CommandInterpreter {
+public:
+ /**
+ * Constructor
+ * @param mgmtSrvr: Management server to use when executing commands
+ */
+ CommandInterpreter(MgmtSrvr& mgmtSrvr);
+
+ /**
+ * Reads one line from the stream, parse the line to find
+ * a command and then calls a suitable method which executes
+ * the command.
+ *
+ * @return true until quit/bye/exit has been typed
+ */
+ int readAndExecute();
+
+private:
+ /**
+ * Read a string, and return a pointer to it.
+ *
+ * @return NULL on EOF.
+ */
+ char *readline_gets ()
+ {
+ static char *line_read = (char *)NULL;
+
+ // Disable the default file-name completion action of TAB
+ // rl_bind_key ('\t', rl_insert);
+
+ /* If the buffer has already been allocated, return the memory
+ to the free pool. */
+ if (line_read)
+ {
+ free (line_read);
+ line_read = (char *)NULL;
+ }
+
+ /* Get a line from the user. */
+ line_read = readline ("NDB> ");
+
+ /* If the line has any text in it, save it on the history. */
+ if (line_read && *line_read)
+ add_history (line_read);
+
+ return (line_read);
+ }
+
+ /**
+ * Analyse the command line, after the first token.
+ *
+ * @param processId: DB process id to send command to or -1 if
+ * command will be sent to all DB processes.
+ * @param allAfterFirstToken: What the client gave after the
+ * first token on the command line
+ */
+ void analyseAfterFirstToken(int processId, char* allAfterFirstTokenCstr);
+
+ /**
+ * Parse the block specification part of the LOG* commands,
+ * things after LOG*: [BLOCK = {ALL|<blockName>+}]
+ *
+ * @param allAfterLog: What the client gave after the second token
+ * (LOG*) on the command line
+ * @param blocks, OUT: ALL or name of all the blocks
+ * @return: true if correct syntax, otherwise false
+ */
+ bool parseBlockSpecification(const char* allAfterLog,
+ Vector<BaseString>& blocks);
+
+ /**
+ * A bunch of execute functions: Executes one of the commands
+ *
+ * @param processId: DB process id to send command to
+ * @param parameters: What the client gave after the command name
+ * on the command line.
+ * For example if complete input from user is: "1 LOGLEVEL 22" then the
+ * parameters argument is the string with everything after LOGLEVEL, in this
+ * case "22". Each function is responsible to check the parameters argument.
+ */
+ void executeHelp(char* parameters);
+ void executeShow(char* parameters);
+ void executeRun(char* parameters);
+ void executeInfo(char* parameters);
+ void executeClusterLog(char* parameters);
+
+public:
+ void executeStop(int processId, const char* parameters, bool all);
+ void executeStart(int processId, const char* parameters, bool all);
+ void executeRestart(int processId, const char* parameters, bool all);
+ void executeLogLevel(int processId, const char* parameters, bool all);
+ void executeError(int processId, const char* parameters, bool all);
+ void executeTrace(int processId, const char* parameters, bool all);
+ void executeLog(int processId, const char* parameters, bool all);
+ void executeLogIn(int processId, const char* parameters, bool all);
+ void executeLogOut(int processId, const char* parameters, bool all);
+ void executeLogOff(int processId, const char* parameters, bool all);
+ void executeTestOn(int processId, const char* parameters, bool all);
+ void executeTestOff(int processId, const char* parameters, bool all);
+ void executeStatus(int processId, const char* parameters, bool all);
+ void executeEnterSingleUser(char* parameters);
+ void executeExitSingleUser(char* parameters);
+ void executeEventReporting(int processId, const char* parameters, bool all);
+ void executeDumpState(int processId, const char* parameters, bool all);
+ void executeStartBackup(char * pars);
+ void executeAbortBackup(char * pars);
+
+ void jonas(int processId, const char* parameters, bool all);
+
+ /**
+ * A execute function definition
+ */
+public:
+ typedef void (CommandInterpreter::* ExecuteFunction)(int processId,
+ const char * param,
+ bool all);
+
+ struct CommandFunctionPair {
+ const char * command;
+ ExecuteFunction executeFunction;
+ };
+private:
+ /**
+ *
+ */
+ void executeForAll(const char * cmd, ExecuteFunction fun, const char * param);
+
+ /**
+ * Management server to use when executing commands
+ */
+ MgmtSrvr& _mgmtSrvr;
+};
+
+#endif // CommandInterpreter_H
diff --git a/ndb/src/mgmsrv/Makefile b/ndb/src/mgmsrv/Makefile
new file mode 100644
index 00000000000..ebf50ecb76e
--- /dev/null
+++ b/ndb/src/mgmsrv/Makefile
@@ -0,0 +1,42 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := mgmtsrvr
+BIN_TARGET_LIBS :=
+BIN_TARGET_ARCHIVES := mgmapi NDB_API mgmsrvcommon
+
+LDFLAGS_LOC = -lpthread
+
+ifneq ($(USE_EDITLINE), N)
+BIN_TARGET_ARCHIVES += editline
+DIRS := mkconfig
+endif
+BIN_TARGET_ARCHIVES += general
+
+BIN_FLAGS += $(TERMCAP_LIB)
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = \
+ MgmtSrvr.cpp \
+ MgmtSrvrGeneralSignalHandling.cpp \
+ main.cpp \
+ Services.cpp \
+ convertStrToInt.cpp \
+ NodeLogLevel.cpp \
+ NodeLogLevelList.cpp \
+ SignalQueue.cpp \
+ MgmtSrvrConfig.cpp
+
+ifeq ($(findstring OSE, $(NDB_OS)),)
+SOURCES += CommandInterpreter.cpp
+endif
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/src/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/include/mgmapi) \
+ -I$(call fixpath,$(NDB_TOP)/src/common/mgmcommon)
+
+include $(NDB_TOP)/Epilogue.mk
+
+_bins_mkconfig : $(NDB_TOP)/bin/$(BIN_TARGET)
+
diff --git a/ndb/src/mgmsrv/MgmtSrvr.cpp b/ndb/src/mgmsrv/MgmtSrvr.cpp
new file mode 100644
index 00000000000..45b32169e57
--- /dev/null
+++ b/ndb/src/mgmsrv/MgmtSrvr.cpp
@@ -0,0 +1,2545 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "MgmtSrvr.hpp"
+#include "MgmtErrorReporter.hpp"
+
+#include <ctype.h>
+#include <NdbOut.hpp>
+#include <AttrType.hpp>
+#include <NdbApiSignal.hpp>
+#include <kernel_types.h>
+#include <RefConvert.hpp>
+#include <BlockNumbers.h>
+#include <GlobalSignalNumbers.h>
+#include <signaldata/TestOrd.hpp>
+#include <signaldata/TamperOrd.hpp>
+#include <signaldata/SetVarReq.hpp>
+#include <signaldata/StartOrd.hpp>
+#include <signaldata/ApiVersion.hpp>
+#include <signaldata/ResumeReq.hpp>
+#include <signaldata/SetLogLevelOrd.hpp>
+#include <signaldata/EventSubscribeReq.hpp>
+#include <signaldata/EventReport.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+#include <signaldata/BackupSignalData.hpp>
+#include <signaldata/GrepImpl.hpp>
+#include <signaldata/ManagementServer.hpp>
+#include <NdbSleep.h>
+#include <EventLogger.hpp>
+#include <DebuggerNames.hpp>
+#include <ndb_version.h>
+#include <string.h>
+
+#include "SocketServer.hpp"
+#include "NodeLogLevel.hpp"
+#include <NdbConfig.h>
+
+#include <time.h>
+#include <pthread.h>
+
+//#define MGM_SRV_DEBUG
+#ifdef MGM_SRV_DEBUG
+#define DEBUG(x) do ndbout << x << endl; while(0)
+#else
+#define DEBUG(x)
+#endif
+
+static
+void
+CmdBackupCallback(const MgmtSrvr::BackupEvent & event)
+{
+ char str[255];
+
+ ndbout << endl;
+
+ bool ok = false;
+ switch(event.Event){
+ case MgmtSrvr::BackupEvent::BackupStarted:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup %d started", event.Started.BackupId);
+ break;
+ case MgmtSrvr::BackupEvent::BackupFailedToStart:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup failed to start (Error %d)",
+ event.FailedToStart.ErrorCode);
+ break;
+ case MgmtSrvr::BackupEvent::BackupCompleted:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup %d completed",
+ event.Completed.BackupId);
+ ndbout << str << endl;
+
+ snprintf(str, sizeof(str),
+ " StartGCP: %d StopGCP: %d",
+ event.Completed.startGCP, event.Completed.stopGCP);
+ ndbout << str << endl;
+
+ snprintf(str, sizeof(str),
+ " #Records: %d #LogRecords: %d",
+ event.Completed.NoOfRecords, event.Completed.NoOfLogRecords);
+ ndbout << str << endl;
+
+ snprintf(str, sizeof(str),
+ " Data: %d bytes Log: %d bytes",
+ event.Completed.NoOfBytes, event.Completed.NoOfLogBytes);
+ break;
+ case MgmtSrvr::BackupEvent::BackupAborted:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup %d has been aborted reason %d",
+ event.Aborted.BackupId,
+ event.Aborted.Reason);
+ break;
+ }
+ if(!ok){
+ snprintf(str, sizeof(str), "Unknown backup event: %d", event.Event);
+ }
+ ndbout << str << endl;
+}
+
+
+void *
+MgmtSrvr::logLevelThread_C(void* m)
+{
+ MgmtSrvr *mgm = (MgmtSrvr*)m;
+
+ mgm->logLevelThreadRun();
+
+ NdbThread_Exit(0);
+ /* NOTREACHED */
+ return 0;
+}
+
+void *
+MgmtSrvr::signalRecvThread_C(void *m)
+{
+ MgmtSrvr *mgm = (MgmtSrvr*)m;
+
+ mgm->signalRecvThreadRun();
+
+ NdbThread_Exit(0);
+ /* NOTREACHED */
+ return 0;
+}
+
+class SigMatch
+{
+public:
+ int gsn;
+ void (MgmtSrvr::* function)(NdbApiSignal *signal);
+
+ SigMatch() { gsn = 0; function = NULL; };
+
+ SigMatch(int _gsn,
+ void (MgmtSrvr::* _function)(NdbApiSignal *signal)) {
+ gsn = _gsn;
+ function = _function;
+ };
+
+ bool check(NdbApiSignal *signal) {
+ if(signal->readSignalNumber() == gsn)
+ return true;
+ return false;
+ };
+
+};
+
+void
+MgmtSrvr::signalRecvThreadRun()
+{
+ Vector<SigMatch> siglist;
+ siglist.push_back(SigMatch(GSN_MGM_LOCK_CONFIG_REQ,
+ &MgmtSrvr::handle_MGM_LOCK_CONFIG_REQ));
+ siglist.push_back(SigMatch(GSN_MGM_UNLOCK_CONFIG_REQ,
+ &MgmtSrvr::handle_MGM_UNLOCK_CONFIG_REQ));
+
+ while(1) {
+ SigMatch *handler = NULL;
+ NdbApiSignal *signal = NULL;
+ if(m_signalRecvQueue.waitFor(siglist, handler, signal)) {
+ if(handler->function != 0)
+ (this->*handler->function)(signal);
+ }
+ }
+};
+
+
+EventLogger g_EventLogger;
+
+void
+MgmtSrvr::logLevelThreadRun()
+{
+ NdbMutex* threadMutex = NdbMutex_Create();
+
+ while (!_isStopThread) {
+ if (_startedNodeId != 0) {
+ NdbMutex_Lock(threadMutex);
+
+ // Local node
+ NodeLogLevel* n = NULL;
+ while ((n = _nodeLogLevelList->next()) != NULL) {
+ if (n->getNodeId() == _startedNodeId) {
+ setNodeLogLevel(_startedNodeId, n->getLogLevelOrd(), true);
+ }
+ }
+ // Cluster log
+ while ((n = _clusterLogLevelList->next()) != NULL) {
+ if (n->getNodeId() == _startedNodeId) {
+ setEventReportingLevel(_startedNodeId, n->getLogLevelOrd(), true);
+ }
+ }
+ _startedNodeId = 0;
+
+ NdbMutex_Unlock(threadMutex);
+
+ } // if (_startedNodeId != 0) {
+
+ NdbSleep_MilliSleep(_logLevelThreadSleep);
+ } // while (!_isStopThread)
+
+ NdbMutex_Destroy(threadMutex);
+}
+
+void
+MgmtSrvr::setStatisticsListner(StatisticsListner* listner)
+{
+ m_statisticsListner = listner;
+}
+
+void
+MgmtSrvr::startEventLog()
+{
+ g_EventLogger.setCategory("MgmSrvr");
+ const Properties *mgmProps;
+ _config->get("Node", _ownNodeId, &mgmProps);
+ BaseString logdest;
+ char clusterLog[MAXPATHLEN];
+
+ NdbConfig_ClusterLogFileName(clusterLog, sizeof(clusterLog));
+
+ mgmProps->get("LogDestination", logdest);
+
+ if(logdest.length()==0) {
+ logdest.assfmt("FILE:filename=%s,maxsize=1000000,maxfiles=6", clusterLog);
+ }
+
+ if(!g_EventLogger.addHandler(logdest)) {
+ ndbout << "ERROR: cannot parse \"" << logdest << "\"" << endl;
+ exit(1);
+ }
+}
+
+void
+MgmtSrvr::stopEventLog()
+{
+ // Nothing yet
+}
+
+class ErrorItem
+{
+public:
+ int _errorCode;
+ const BaseString _errorText;
+};
+
+bool
+MgmtSrvr::setEventLogFilter(int severity)
+{
+ bool enabled = true;
+ Logger::LoggerLevel level = (Logger::LoggerLevel)severity;
+ if (g_EventLogger.isEnable(level)) {
+ g_EventLogger.disable(level);
+ enabled = false;
+ } else {
+ g_EventLogger.enable(level);
+ }
+
+ return enabled;
+}
+
+bool
+MgmtSrvr::isEventLogFilterEnabled(int severity)
+{
+ return g_EventLogger.isEnable((Logger::LoggerLevel)severity);
+}
+
+static ErrorItem errorTable[] =
+{
+ {200, "Backup undefined error"},
+ {202, "Backup failed to allocate buffers (check configuration)"},
+ {203, "Backup failed to setup fs buffers (check configuration)"},
+ {204, "Backup failed to allocate tables (check configuration)"},
+ {205, "Backup failed to insert file header (check configuration)"},
+ {206, "Backup failed to insert table list (check configuration)"},
+ {207, "Backup failed to allocate table memory (check configuration)"},
+ {208, "Backup failed to allocate file record (check configuration)"},
+ {209, "Backup failed to allocate attribute record (check configuration)"},
+
+ {MgmtSrvr::NO_CONTACT_WITH_PROCESS, "No contact with the process (dead ?)."},
+ {MgmtSrvr::PROCESS_NOT_CONFIGURED, "The process is not configured."},
+ {MgmtSrvr::WRONG_PROCESS_TYPE,
+ "The process has wrong type. Expected a DB process."},
+ {MgmtSrvr::COULD_NOT_ALLOCATE_MEMORY, "Could not allocate memory."},
+ {MgmtSrvr::SEND_OR_RECEIVE_FAILED, "Send to process or receive failed."},
+ {MgmtSrvr::INVALID_LEVEL, "Invalid level. Should be between 1 and 30."},
+ {MgmtSrvr::INVALID_ERROR_NUMBER, "Invalid error number. Should be >= 0."},
+ {MgmtSrvr::INVALID_TRACE_NUMBER, "Invalid trace number."},
+ {MgmtSrvr::NOT_IMPLEMENTED, "Not implemented."},
+ {MgmtSrvr::INVALID_BLOCK_NAME, "Invalid block name"},
+
+ {MgmtSrvr::CONFIG_PARAM_NOT_EXIST,
+ "The configuration parameter does not exist for the process type."},
+ {MgmtSrvr::CONFIG_PARAM_NOT_UPDATEABLE,
+ "The configuration parameter is not possible to update."},
+ {MgmtSrvr::VALUE_WRONG_FORMAT_INT_EXPECTED,
+ "Incorrect value. Expected integer."},
+ {MgmtSrvr::VALUE_TOO_LOW, "Value is too low."},
+ {MgmtSrvr::VALUE_TOO_HIGH, "Value is too high."},
+ {MgmtSrvr::VALUE_WRONG_FORMAT_BOOL_EXPECTED,
+ "Incorrect value. Expected TRUE or FALSE."},
+
+ {MgmtSrvr::CONFIG_FILE_OPEN_WRITE_ERROR,
+ "Could not open configuration file for writing."},
+ {MgmtSrvr::CONFIG_FILE_OPEN_READ_ERROR,
+ "Could not open configuration file for reading."},
+ {MgmtSrvr::CONFIG_FILE_WRITE_ERROR,
+ "Write error when writing configuration file."},
+ {MgmtSrvr::CONFIG_FILE_READ_ERROR,
+ "Read error when reading configuration file."},
+ {MgmtSrvr::CONFIG_FILE_CLOSE_ERROR, "Could not close configuration file."},
+
+ {MgmtSrvr::CONFIG_CHANGE_REFUSED_BY_RECEIVER,
+ "The change was refused by the receiving process."},
+ {MgmtSrvr::COULD_NOT_SYNC_CONFIG_CHANGE_AGAINST_PHYSICAL_MEDIUM,
+ "The change could not be synced against physical medium."},
+ {MgmtSrvr::CONFIG_FILE_CHECKSUM_ERROR,
+ "The config file is corrupt. Checksum error."},
+ {MgmtSrvr::NOT_POSSIBLE_TO_SEND_CONFIG_UPDATE_TO_PROCESS_TYPE,
+ "It is not possible to send an update of a configuration variable "
+ "to this kind of process."},
+ {5026, "Node shutdown in progress" },
+ {5027, "System shutdown in progress" },
+ {5028, "Node shutdown would cause system crash" },
+ {5029, "Only one shutdown at a time is possible via mgm server" },
+ {5060, "Operation not allowed in single user mode." },
+ {5061, "DB is not in single user mode." },
+ {5062, "The specified node is not an API node." },
+ {5063,
+ "Cannot enter single user mode. DB nodes in inconsistent startlevel."},
+ {MgmtSrvr::NO_CONTACT_WITH_DB_NODES, "No contact with database nodes" }
+};
+
+int MgmtSrvr::translateStopRef(Uint32 errCode)
+{
+ switch(errCode){
+ case StopRef::NodeShutdownInProgress:
+ return 5026;
+ break;
+ case StopRef::SystemShutdownInProgress:
+ return 5027;
+ break;
+ case StopRef::NodeShutdownWouldCauseSystemCrash:
+ return 5028;
+ break;
+ }
+ return 4999;
+}
+
+static int noOfErrorCodes = sizeof(errorTable) / sizeof(ErrorItem);
+
+int
+MgmtSrvr::getNodeCount(enum ndb_mgm_node_type type) const
+{
+ int count = 0;
+ NodeId nodeId = 0;
+
+ while (getNextNodeId(&nodeId, type)) {
+ count++;
+ }
+ return count;
+}
+
+int
+MgmtSrvr::getStatPort() const
+{
+ const Properties *mgmProps;
+ if(!getConfig()->get("Node", _ownNodeId, &mgmProps))
+ return -1;
+
+ int tmp = -1;
+ if(!mgmProps->get("PortNumberStats", (Uint32 *)&tmp))
+ return -1;
+
+ return tmp;
+}
+
+/* Constructor */
+MgmtSrvr::MgmtSrvr(NodeId nodeId,
+ const BaseString &configFilename,
+ const BaseString &ndb_config_filename):
+ _blockNumber(1), // Hard coded block number since it makes it easy to send
+ // signals to other management servers.
+ _ownReference(0),
+ theSignalIdleList(NULL),
+ theWaitState(WAIT_SUBSCRIBE_CONF),
+ theConfCount(0) {
+
+ _ownNodeId = nodeId;
+ _config = NULL;
+ _isStatPortActive = false;
+ _isClusterLogStatActive = false;
+
+ _isStopThread = false;
+ _logLevelThread = NULL;
+ _logLevelThreadSleep = 500;
+ _startedNodeId = 0;
+
+ m_newConfig = NULL;
+ m_configFilename = configFilename;
+ setCallback(CmdBackupCallback);
+ m_localNdbConfigFilename = ndb_config_filename;
+
+ m_nextConfigGenerationNumber = 0;
+
+ _config = readConfig();
+
+ theMgmtWaitForResponseCondPtr = NdbCondition_Create();
+
+ m_configMutex = NdbMutex_Create();
+
+ /**
+ * Fill the nodeTypes array
+ */
+ for(Uint32 i = 0; i<MAX_NODES; i++)
+ nodeTypes[i] = (enum ndb_mgm_node_type)-1;
+
+ Properties::Iterator it(_config);
+ const char * name;
+ for(name = it.first(); name != NULL; name = it.next()){
+ if(strncmp(name, "Node_", strlen("Node_")) == 0){
+ const Properties * tmp;
+
+ _config->get(name, &tmp);
+
+ Uint32 nodeId;
+ BaseString type;
+ MGM_REQUIRE(tmp->get("Id", &nodeId));
+ MGM_REQUIRE(tmp->get("Type", type));
+ MGM_REQUIRE(nodeId < MAX_NODES);
+
+ if(type == "MGM")
+ nodeTypes[nodeId] = NDB_MGM_NODE_TYPE_MGM;
+ if(type == "API")
+ nodeTypes[nodeId] = NDB_MGM_NODE_TYPE_API;
+ if(type == "DB")
+ nodeTypes[nodeId] = NDB_MGM_NODE_TYPE_NDB;
+ }
+ }
+
+ m_statisticsListner = NULL;
+
+ _nodeLogLevelList = new NodeLogLevelList();
+ _clusterLogLevelList = new NodeLogLevelList();
+
+ _props = NULL;
+}
+
+
+//****************************************************************************
+//****************************************************************************
+bool
+MgmtSrvr::check_start()
+{
+ if (_config == 0) {
+ DEBUG("MgmtSrvr.cpp: _config is NULL.");
+ return false;
+ }
+
+ _props = new Config(* _config);
+ if (_props == 0) {
+ DEBUG("MgmtSrvr.cpp: Object props is NULL.");
+ return false;
+ }
+ MGM_REQUIRE(_props->put("LocalNodeId", _ownNodeId, true));
+
+ return true;
+}
+
+bool
+MgmtSrvr::start()
+{
+ if (_props == NULL) {
+ if (!check_start())
+ return false;
+ }
+
+ theFacade = TransporterFacade::start_instance(_props, NULL);
+ delete _props;
+ _props = NULL;
+
+ if(theFacade == 0) {
+ DEBUG("MgmtSrvr.cpp: theFacade is NULL.");
+ return false;
+ }
+
+ MGM_REQUIRE(_blockNumber == 1);
+
+ // Register ourself at TransporterFacade to be able to receive signals
+ // and to be notified when a database process has died.
+ _blockNumber = theFacade->open(this,
+ signalReceivedNotification,
+ nodeStatusNotification);
+
+ if(_blockNumber == -1){
+ DEBUG("MgmtSrvr.cpp: _blockNumber is -1.");
+ theFacade->stop_instance();
+ theFacade = 0;
+ return false;
+ }
+
+ _ownReference = numberToRef(_blockNumber, _ownNodeId);
+
+ startEventLog();
+ // Set the initial confirmation count for subscribe requests confirm
+ // from NDB nodes in the cluster.
+ //
+ theConfCount = getNodeCount(NDB_MGM_NODE_TYPE_NDB);
+
+ // Loglevel thread
+ _logLevelThread = NdbThread_Create(logLevelThread_C,
+ (void**)this,
+ 32768,
+ "MgmtSrvr_Loglevel",
+ NDB_THREAD_PRIO_LOW);
+
+ m_signalRecvThread = NdbThread_Create(signalRecvThread_C,
+ (void **)this,
+ 32768,
+ "MgmtSrvr_Service",
+ NDB_THREAD_PRIO_LOW);
+
+ return true;
+}
+
+
+//****************************************************************************
+//****************************************************************************
+MgmtSrvr::~MgmtSrvr()
+{
+ while (theSignalIdleList != NULL) {
+ freeSignal();
+ }
+
+ if(theFacade != 0){
+ theFacade->stop_instance();
+ theFacade = 0;
+ }
+
+ stopEventLog();
+
+ NdbCondition_Destroy(theMgmtWaitForResponseCondPtr);
+ NdbMutex_Destroy(m_configMutex);
+
+ if(m_newConfig != NULL)
+ delete m_newConfig;
+ delete _nodeLogLevelList;
+ delete _clusterLogLevelList;
+
+ // End set log level thread
+ void* res = 0;
+ _isStopThread = true;
+
+ if (_logLevelThread != NULL) {
+ NdbThread_WaitFor(_logLevelThread, &res);
+ NdbThread_Destroy(&_logLevelThread);
+ }
+}
+
+//****************************************************************************
+//****************************************************************************
+
+int MgmtSrvr::okToSendTo(NodeId processId, bool unCond)
+{
+ if (getNodeType(processId) != NDB_MGM_NODE_TYPE_NDB)
+ return WRONG_PROCESS_TYPE;
+
+ // Check if we have contact with it
+ if(unCond){
+ if(theFacade->theClusterMgr->getNodeInfo(processId).connected)
+ return 0;
+ return NO_CONTACT_WITH_PROCESS;
+ }
+ if (theFacade->get_node_alive(processId) == 0) {
+ return NO_CONTACT_WITH_PROCESS;
+ } else {
+ return 0;
+ }
+}
+
+/*****************************************************************************
+ * Starting and stopping database nodes
+ ****************************************************************************/
+
+int
+MgmtSrvr::start(int processId)
+{
+ int result;
+
+ result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ StartOrd* const startOrd = CAST_PTR(StartOrd, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, CMVMI, GSN_START_ORD, StartOrd::SignalLength);
+
+ startOrd->restartInfo = 0;
+
+ result = sendSignal(processId, NO_WAIT, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ return 0;
+}
+
+/**
+ * Restart one database node
+ */
+int
+MgmtSrvr::restartNode(int processId, bool nostart,
+ bool initalStart, bool abort,
+ StopCallback callback, void * anyData)
+{
+ int result;
+
+ if(m_stopRec.singleUserMode)
+ return 5060;
+
+ if(m_stopRec.inUse){
+ return 5029;
+ }
+
+ result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ StopReq* const stopReq = CAST_PTR(StopReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, NDBCNTR, GSN_STOP_REQ, StopReq::SignalLength);
+
+ stopReq->requestInfo = 0;
+ StopReq::setSystemStop(stopReq->requestInfo, false);
+ StopReq::setPerformRestart(stopReq->requestInfo, true);
+ StopReq::setNoStart(stopReq->requestInfo, nostart);
+ StopReq::setInitialStart(stopReq->requestInfo, initalStart);
+ StopReq::setStopAbort(stopReq->requestInfo, abort);
+ stopReq->singleuser = 0;
+ stopReq->apiTimeout = 5000;
+ stopReq->transactionTimeout = 1000;
+ stopReq->readOperationTimeout = 1000;
+ stopReq->operationTimeout = 1000;
+ stopReq->senderData = 12;
+ stopReq->senderRef = _ownReference;
+
+ m_stopRec.singleUserMode = false;
+ m_stopRec.sentCount = 1;
+ m_stopRec.reply = 0;
+ m_stopRec.nodeId = processId;
+ m_stopRec.anyData = anyData;
+ m_stopRec.callback = callback;
+ m_stopRec.inUse = true;
+
+ if(callback == NULL){
+ Uint32 timeOut = 0;
+ timeOut += stopReq->apiTimeout;
+ timeOut += stopReq->transactionTimeout;
+ timeOut += stopReq->readOperationTimeout;
+ timeOut += stopReq->operationTimeout;
+ timeOut *= 3;
+ result = sendRecSignal(processId, WAIT_STOP, signal, true, timeOut);
+ } else {
+ result = sendSignal(processId, NO_WAIT, signal, true);
+ }
+
+ if (result == -1) {
+ m_stopRec.inUse = false;
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ if(callback == 0){
+ m_stopRec.inUse = false;
+ return m_stopRec.reply;
+ } else {
+ return 0;
+ }
+}
+
+/**
+ * Restart all database nodes
+ */
+int
+MgmtSrvr::restart(bool nostart, bool initalStart, bool abort,
+ int * stopCount, StopCallback callback, void * anyData)
+{
+ if(m_stopRec.singleUserMode)
+ return 5060;
+
+ if(m_stopRec.inUse){
+ return 5029;
+ }
+
+ m_stopRec.singleUserMode = false;
+ m_stopRec.sentCount = 0;
+ m_stopRec.reply = 0;
+ m_stopRec.nodeId = 0;
+ m_stopRec.anyData = anyData;
+ m_stopRec.callback = callback;
+ m_stopRec.inUse = true;
+
+ /**
+ * Restart all database nodes into idle ("no-started") state
+ */
+ Uint32 timeOut = 0;
+ NodeId nodeId = 0;
+ NodeBitmask nodes;
+ while(getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)){
+ if(okToSendTo(nodeId, true) == 0){
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ StopReq* const stopReq = CAST_PTR(StopReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, NDBCNTR, GSN_STOP_REQ,
+ StopReq::SignalLength);
+
+ stopReq->requestInfo = 0;
+ stopReq->singleuser = 0;
+ StopReq::setSystemStop(stopReq->requestInfo, true);
+ StopReq::setPerformRestart(stopReq->requestInfo, true);
+ if (callback == 0) {
+ // Start node in idle ("no-started") state
+ StopReq::setNoStart(stopReq->requestInfo, 1);
+ } else {
+ StopReq::setNoStart(stopReq->requestInfo, nostart);
+ }
+ StopReq::setInitialStart(stopReq->requestInfo, initalStart);
+ StopReq::setStopAbort(stopReq->requestInfo, abort);
+
+ stopReq->apiTimeout = 5000;
+ stopReq->transactionTimeout = 1000;
+ stopReq->readOperationTimeout = 1000;
+ stopReq->operationTimeout = 1000;
+ stopReq->senderData = 12;
+ stopReq->senderRef = _ownReference;
+
+ timeOut += stopReq->apiTimeout;
+ timeOut += stopReq->transactionTimeout;
+ timeOut += stopReq->readOperationTimeout;
+ timeOut += stopReq->operationTimeout;
+ timeOut *= 3;
+
+ m_stopRec.sentCount++;
+ int res;
+ if(callback == 0){
+ res = sendSignal(nodeId, WAIT_STOP, signal, true);
+ } else {
+ res = sendSignal(nodeId, NO_WAIT, signal, true);
+ }
+
+ if(res != -1){
+ nodes.set(nodeId);
+ }
+ }
+ }
+
+ if(stopCount != 0){
+ * stopCount = m_stopRec.sentCount;
+ }
+
+ if(m_stopRec.sentCount == 0){
+ m_stopRec.inUse = false;
+ return 0;
+ }
+
+ if(callback != 0){
+ return 0;
+ }
+
+ TransporterFacade::instance()->lock_mutex();
+ int waitTime = timeOut/m_stopRec.sentCount;
+ if (receiveOptimisedResponse(waitTime) != 0) {
+ m_stopRec.inUse = false;
+ return -1;
+ }
+
+ /**
+ * Here all nodes were correctly stopped,
+ * so we wait for all nodes to be contactable
+ */
+ nodeId = 0;
+ NDB_TICKS maxTime = NdbTick_CurrentMillisecond() + waitTime;
+ while(getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB) && nodes.get(nodeId)) {
+ enum ndb_mgm_node_status s;
+ s = NDB_MGM_NODE_STATUS_NO_CONTACT;
+ while (s == NDB_MGM_NODE_STATUS_NO_CONTACT && waitTime > 0) {
+ Uint32 startPhase = 0, version = 0, dynamicId = 0, nodeGroup = 0;
+ bool system;
+ status(nodeId, &s, &version, &startPhase,
+ &system, &dynamicId, &nodeGroup);
+ NdbSleep_MilliSleep(100);
+ waitTime = (maxTime - NdbTick_CurrentMillisecond());
+ }
+ }
+
+ if(nostart){
+ m_stopRec.inUse = false;
+ return 0;
+ }
+
+ /**
+ * Now we start all database nodes (i.e. we make them non-idle)
+ * We ignore the result we get from the start command.
+ */
+ nodeId = 0;
+ while(getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB) && nodes.get(nodeId)) {
+ int result;
+ result = start(nodeId);
+ DEBUG("Starting node " << nodeId << " with result " << result);
+ /**
+ * Errors from this call are deliberately ignored.
+ * Maybe the user only wanted to restart a subset of the nodes.
+ * It is also easy for the user to check which nodes have
+ * started and which nodes have not.
+ *
+ * if (result != 0) {
+ * m_stopRec.inUse = false;
+ * return result;
+ * }
+ */
+ }
+
+ m_stopRec.inUse = false;
+ return 0;
+}
+
+/*****************************************************************************
+ * Version handling
+ *****************************************************************************/
+
+int
+MgmtSrvr::versionNode(int processId, bool abort,
+ VersionCallback callback, void * anyData)
+{
+ if(m_versionRec.inUse)
+ return OPERATION_IN_PROGRESS;
+
+ m_versionRec.callback = callback;
+ m_versionRec.inUse = true ;
+ ClusterMgr::Node node;
+ int version;
+ if (getNodeType(processId) == NDB_MGM_NODE_TYPE_MGM) {
+ if(m_versionRec.callback != 0)
+ m_versionRec.callback(processId, NDB_VERSION, this,0);
+ }
+
+ if (getNodeType(processId) == NDB_MGM_NODE_TYPE_NDB) {
+ node = theFacade->theClusterMgr->getNodeInfo(processId);
+ version = node.m_info.m_version;
+ if(theFacade->theClusterMgr->getNodeInfo(processId).connected)
+ if(m_versionRec.callback != 0)
+ m_versionRec.callback(processId, version, this,0);
+ else
+ if(m_versionRec.callback != 0)
+ m_versionRec.callback(processId, 0, this,0);
+
+ }
+
+ if (getNodeType(processId) == NDB_MGM_NODE_TYPE_API) {
+ return sendVersionReq(processId);
+ }
+ m_versionRec.inUse = false ;
+ return 0;
+
+}
+
+int
+MgmtSrvr::sendVersionReq(int processId)
+{
+ Uint32 ndbnode=0;
+ int result;
+ for(Uint32 i = 0; i<MAX_NODES; i++) {
+ if (getNodeType(i) == NDB_MGM_NODE_TYPE_NDB) {
+ if(okToSendTo(i, true) == 0)
+ {
+ ndbnode = i;
+ break;
+ }
+ }
+ }
+
+ if (ndbnode == 0) {
+ m_versionRec.inUse = false;
+ if(m_versionRec.callback != 0)
+ m_versionRec.callback(processId, 0, this,0);
+ return NO_CONTACT_WITH_CLUSTER;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ m_versionRec.inUse = false;
+ if(m_versionRec.callback != 0)
+ m_versionRec.callback(processId, 0, this,0);
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+ ApiVersionReq* req = CAST_PTR(ApiVersionReq, signal->getDataPtrSend());
+ req->senderRef = _ownReference;
+ req->nodeId = processId;
+
+ signal->set(TestOrd::TraceAPI, QMGR, GSN_API_VERSION_REQ,
+ ApiVersionReq::SignalLength);
+
+
+ // if(m_versionRec.callback == 0){
+ Uint32 timeOut = 0;
+ timeOut = 10000;
+ result = sendRecSignal(ndbnode, WAIT_VERSION, signal, true, timeOut);
+ //} else {
+ //result = sendSignal(processId, NO_WAIT, signal, true);
+ // }
+
+ if (result == -1) {
+ m_versionRec.inUse = false;
+ if(m_versionRec.callback != 0)
+ m_versionRec.callback(processId, 0, this,0);
+ m_versionRec.version[processId] = 0;
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ m_versionRec.inUse = false;
+ return 0;
+}
+
+int
+MgmtSrvr::version(int * stopCount, bool abort,
+ VersionCallback callback, void * anyData)
+{
+ ClusterMgr::Node node;
+ int version;
+
+ if(m_versionRec.inUse)
+ return 1;
+
+ m_versionRec.callback = callback;
+ m_versionRec.inUse = true ;
+
+ for(Uint32 i = 0; i<MAX_NODES; i++) {
+ if (getNodeType(i) == NDB_MGM_NODE_TYPE_MGM) {
+ m_versionRec.callback(i, NDB_VERSION, this,0);
+ }
+ }
+ for(Uint32 i = 0; i<MAX_NODES; i++) {
+ if (getNodeType(i) == NDB_MGM_NODE_TYPE_NDB) {
+ node =
+ TransporterFacade::instance()->theClusterMgr->getNodeInfo(i);
+ version = node.m_info.m_version;
+ if(theFacade->theClusterMgr->getNodeInfo(i).connected)
+ m_versionRec.callback(i, version, this,0);
+ else
+ m_versionRec.callback(i, 0, this,0);
+
+ }
+ }
+ for(Uint32 i = 0; i<MAX_NODES; i++) {
+ if (getNodeType(i) == NDB_MGM_NODE_TYPE_API) {
+ return sendVersionReq(i);
+ }
+ }
+
+ return 0;
+}
+
+int
+MgmtSrvr::stopNode(int processId, bool abort, StopCallback callback,
+ void * anyData)
+
+{
+ if(m_stopRec.singleUserMode)
+ return 5060;
+
+ if(m_stopRec.inUse)
+ return 5029;
+
+ int result;
+
+ result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ StopReq* const stopReq = CAST_PTR(StopReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, NDBCNTR, GSN_STOP_REQ, StopReq::SignalLength);
+
+ stopReq->requestInfo = 0;
+ stopReq->singleuser = 0;
+ StopReq::setPerformRestart(stopReq->requestInfo, false);
+ StopReq::setSystemStop(stopReq->requestInfo, false);
+ StopReq::setStopAbort(stopReq->requestInfo, abort);
+
+ stopReq->apiTimeout = 5000;
+ stopReq->transactionTimeout = 1000;
+ stopReq->readOperationTimeout = 1000;
+ stopReq->operationTimeout = 1000;
+ stopReq->senderData = 12;
+ stopReq->senderRef = _ownReference;
+
+ m_stopRec.sentCount = 1;
+ m_stopRec.reply = 0;
+ m_stopRec.nodeId = processId;
+ m_stopRec.anyData = anyData;
+ m_stopRec.callback = callback;
+ m_stopRec.inUse = true;
+
+ if(callback == NULL){
+ Uint32 timeOut = 0;
+ timeOut += stopReq->apiTimeout;
+ timeOut += stopReq->transactionTimeout;
+ timeOut += stopReq->readOperationTimeout;
+ timeOut += stopReq->operationTimeout;
+ timeOut *= 3;
+ result = sendRecSignal(processId, WAIT_STOP, signal, true, timeOut);
+ } else {
+ result = sendSignal(processId, NO_WAIT, signal, true);
+ }
+
+ if (result == -1) {
+ m_stopRec.inUse = false;
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ if(callback == 0){
+ m_stopRec.inUse = false;
+ return m_stopRec.reply;
+ } else {
+ return 0;
+ }
+}
+
+int
+MgmtSrvr::stop(int * stopCount, bool abort, StopCallback callback,
+ void * anyData)
+{
+ if(m_stopRec.singleUserMode)
+ return 5060;
+
+ if(m_stopRec.inUse){
+ return 5029;
+ }
+
+ m_stopRec.singleUserMode = false;
+ m_stopRec.sentCount = 0;
+ m_stopRec.reply = 0;
+ m_stopRec.nodeId = 0;
+ m_stopRec.anyData = anyData;
+ m_stopRec.callback = callback;
+ m_stopRec.inUse = true;
+
+ NodeId nodeId = 0;
+ Uint32 timeOut = 0;
+ while(getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)){
+ if(okToSendTo(nodeId, true) == 0){
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ StopReq* const stopReq = CAST_PTR(StopReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, NDBCNTR, GSN_STOP_REQ,
+ StopReq::SignalLength);
+
+ stopReq->requestInfo = 0;
+ stopReq->singleuser = 0;
+ StopReq::setSystemStop(stopReq->requestInfo, true);
+ StopReq::setPerformRestart(stopReq->requestInfo, false);
+ StopReq::setStopAbort(stopReq->requestInfo, abort);
+
+ stopReq->apiTimeout = 5000;
+ stopReq->transactionTimeout = 1000;
+ stopReq->readOperationTimeout = 1000;
+ stopReq->operationTimeout = 1000;
+ stopReq->senderData = 12;
+ stopReq->senderRef = _ownReference;
+
+ timeOut += stopReq->apiTimeout;
+ timeOut += stopReq->transactionTimeout;
+ timeOut += stopReq->readOperationTimeout;
+ timeOut += stopReq->operationTimeout;
+ timeOut *= 3;
+
+ m_stopRec.sentCount++;
+ if(callback == 0)
+ sendSignal(nodeId, WAIT_STOP, signal, true);
+ else
+ sendSignal(nodeId, NO_WAIT, signal, true);
+ }
+ }
+
+ if(stopCount != 0)
+ * stopCount = m_stopRec.sentCount;
+
+ if(m_stopRec.sentCount > 0){
+ if(callback == 0){
+ TransporterFacade::instance()->lock_mutex();
+ receiveOptimisedResponse(timeOut / m_stopRec.sentCount);
+ } else {
+ return 0;
+ }
+ }
+
+ m_stopRec.inUse = false;
+ return m_stopRec.reply;
+}
+
+/*****************************************************************************
+ * Single user mode
+ ****************************************************************************/
+
+int
+MgmtSrvr::enterSingleUser(int * stopCount, Uint32 singleUserNodeId,
+ EnterSingleCallback callback, void * anyData)
+{
+ if(m_stopRec.singleUserMode) {
+ return 5060;
+ }
+
+ if (getNodeType(singleUserNodeId) != NDB_MGM_NODE_TYPE_API) {
+ return 5062;
+ }
+ ClusterMgr::Node node;
+
+ for(Uint32 i = 0; i<MAX_NODES; i++) {
+ if (getNodeType(i) == NDB_MGM_NODE_TYPE_NDB) {
+ node = TransporterFacade::instance()->theClusterMgr->getNodeInfo(i);
+ if((node.m_state.startLevel != NodeState::SL_STARTED) &&
+ (node.m_state.startLevel != NodeState::SL_NOTHING)) {
+ return 5063;
+ }
+ }
+ }
+
+ if(m_stopRec.inUse){
+ return 5029;
+ }
+
+ if(singleUserNodeId == 0)
+ return 1;
+ m_stopRec.singleUserMode = true;
+ m_stopRec.sentCount = 0;
+ m_stopRec.reply = 0;
+ m_stopRec.nodeId = 0;
+ m_stopRec.anyData = anyData;
+ m_stopRec.callback = callback;
+ m_stopRec.inUse = true;
+
+ NodeId nodeId = 0;
+ Uint32 timeOut = 0;
+ while(getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)){
+ if(okToSendTo(nodeId, true) == 0){
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ StopReq* const stopReq = CAST_PTR(StopReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, NDBCNTR, GSN_STOP_REQ,
+ StopReq::SignalLength);
+
+ stopReq->requestInfo = 0;
+ stopReq->singleuser = 1;
+ stopReq->singleUserApi = singleUserNodeId;
+ StopReq::setSystemStop(stopReq->requestInfo, false);
+ StopReq::setPerformRestart(stopReq->requestInfo, false);
+ StopReq::setStopAbort(stopReq->requestInfo, false);
+
+ stopReq->apiTimeout = 5000;
+ stopReq->transactionTimeout = 1000;
+ stopReq->readOperationTimeout = 1000;
+ stopReq->operationTimeout = 1000;
+ stopReq->senderData = 12;
+ stopReq->senderRef = _ownReference;
+ timeOut += stopReq->apiTimeout;
+ timeOut += stopReq->transactionTimeout;
+ timeOut += stopReq->readOperationTimeout;
+ timeOut += stopReq->operationTimeout;
+ timeOut *= 3;
+
+ m_stopRec.sentCount++;
+ if(callback == 0)
+ sendSignal(nodeId, WAIT_STOP, signal, true);
+ else
+ sendSignal(nodeId, NO_WAIT, signal, true);
+ }
+ }
+
+ if(stopCount != 0)
+ * stopCount = m_stopRec.sentCount;
+
+ if(callback == 0){
+ m_stopRec.inUse = false;
+ return 0;
+ // return m_stopRec.reply;
+ } else {
+ return 0;
+ }
+
+ m_stopRec.inUse = false;
+ return m_stopRec.reply;
+}
+
+int
+MgmtSrvr::exitSingleUser(int * stopCount, bool abort,
+ ExitSingleCallback callback, void * anyData)
+{
+ m_stopRec.sentCount = 0;
+ m_stopRec.reply = 0;
+ m_stopRec.nodeId = 0;
+ m_stopRec.anyData = anyData;
+ m_stopRec.callback = callback;
+ m_stopRec.inUse = true;
+
+ NodeId nodeId = 0;
+ while(getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)){
+ if(okToSendTo(nodeId, true) == 0){
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ ResumeReq* const resumeReq =
+ CAST_PTR(ResumeReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, NDBCNTR, GSN_RESUME_REQ,
+ StopReq::SignalLength);
+ resumeReq->senderData = 12;
+ resumeReq->senderRef = _ownReference;
+
+ m_stopRec.sentCount++;
+ if(callback == 0)
+ sendSignal(nodeId, WAIT_STOP, signal, true);
+ else
+ sendSignal(nodeId, NO_WAIT, signal, true);
+ }
+ }
+
+ m_stopRec.singleUserMode = false;
+
+ if(stopCount != 0)
+ * stopCount = m_stopRec.sentCount;
+
+
+ if(callback == 0){
+ m_stopRec.inUse = false;
+ return m_stopRec.reply;
+ } else {
+ return 0;
+ }
+
+ m_stopRec.inUse = false;
+ return m_stopRec.reply;
+}
+
+
+/*****************************************************************************
+ * Status
+ ****************************************************************************/
+
+#include <ClusterMgr.hpp>
+
+int
+MgmtSrvr::status(int processId,
+ ndb_mgm_node_status * _status,
+ Uint32 * version,
+ Uint32 * _phase,
+ bool * _system,
+ Uint32 * dynamic,
+ Uint32 * nodegroup)
+{
+ if (getNodeType(processId) == NDB_MGM_NODE_TYPE_API) {
+ if(versionNode(processId, false,0,0) ==0)
+ * version = m_versionRec.version[processId];
+ else
+ * version = 0;
+ }
+
+ if (getNodeType(processId) == NDB_MGM_NODE_TYPE_MGM) {
+ * version = NDB_VERSION;
+ }
+
+ const ClusterMgr::Node node =
+ TransporterFacade::instance()->theClusterMgr->getNodeInfo(processId);
+
+ if(!node.connected){
+ * _status = NDB_MGM_NODE_STATUS_NO_CONTACT;
+ return 0;
+ }
+
+ if (getNodeType(processId) == NDB_MGM_NODE_TYPE_NDB) {
+ * version = node.m_info.m_version;
+ }
+
+ * dynamic = node.m_state.dynamicId;
+ * nodegroup = node.m_state.nodeGroup;
+
+ switch(node.m_state.startLevel){
+ case NodeState::SL_CMVMI:
+ * _status = NDB_MGM_NODE_STATUS_NOT_STARTED;
+ * _phase = 0;
+ return 0;
+ break;
+ case NodeState::SL_STARTING:
+ * _status = NDB_MGM_NODE_STATUS_STARTING;
+ * _phase = node.m_state.starting.startPhase;
+ return 0;
+ break;
+ case NodeState::SL_STARTED:
+ * _status = NDB_MGM_NODE_STATUS_STARTED;
+ * _phase = 0;
+ return 0;
+ break;
+ case NodeState::SL_STOPPING_1:
+ * _status = NDB_MGM_NODE_STATUS_SHUTTING_DOWN;
+ * _phase = 1;
+ * _system = node.m_state.stopping.systemShutdown != 0;
+ return 0;
+ break;
+ case NodeState::SL_STOPPING_2:
+ * _status = NDB_MGM_NODE_STATUS_SHUTTING_DOWN;
+ * _phase = 2;
+ * _system = node.m_state.stopping.systemShutdown != 0;
+ return 0;
+ break;
+ case NodeState::SL_STOPPING_3:
+ * _status = NDB_MGM_NODE_STATUS_SHUTTING_DOWN;
+ * _phase = 3;
+ * _system = node.m_state.stopping.systemShutdown != 0;
+ return 0;
+ break;
+ case NodeState::SL_STOPPING_4:
+ * _status = NDB_MGM_NODE_STATUS_SHUTTING_DOWN;
+ * _phase = 4;
+ * _system = node.m_state.stopping.systemShutdown != 0;
+ return 0;
+ break;
+ case NodeState::SL_SINGLEUSER:
+ * _status = NDB_MGM_NODE_STATUS_SINGLEUSER;
+ * _phase = 0;
+ return 0;
+ break;
+ default:
+ * _status = NDB_MGM_NODE_STATUS_UNKNOWN;
+ * _phase = 0;
+ return 0;
+ }
+
+ return -1;
+}
+
+
+//****************************************************************************
+//****************************************************************************
+int
+MgmtSrvr::startStatisticEventReporting(int level)
+{
+ SetLogLevelOrd ll;
+ NodeId nodeId = 0;
+
+ ll.clear();
+ ll.setLogLevel(LogLevel::llStatistic, level);
+
+ if (level > 0) {
+ _isStatPortActive = true;
+ } else {
+ _isStatPortActive = false;
+
+ if (_isClusterLogStatActive) {
+ return 0;
+ }
+ }
+
+ while (getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)) {
+ setEventReportingLevelImpl(nodeId, ll);
+ }
+
+ return 0;
+}
+
+int
+MgmtSrvr::setEventReportingLevel(int processId, const SetLogLevelOrd & ll,
+ bool isResend)
+{
+ for (Uint32 i = 0; i < ll.noOfEntries; i++) {
+ if (ll.theCategories[i] == LogLevel::llStatistic) {
+ if (ll.theLevels[i] > 0) {
+ _isClusterLogStatActive = true;
+ break;
+ } else {
+ _isClusterLogStatActive = false;
+
+ if (_isStatPortActive) {
+ return 0;
+ }
+ break;
+ }
+ } // if (ll.theCategories
+ } // for (int i = 0
+
+ return setEventReportingLevelImpl(processId, ll, isResend);
+}
+
+int
+MgmtSrvr::setEventReportingLevelImpl(int processId,
+ const SetLogLevelOrd & ll,
+ bool isResend)
+{
+ for(Uint32 i = 0; i<ll.noOfEntries; i++){
+ // Save log level for the cluster log
+ if (!isResend) {
+ NodeLogLevel* n = NULL;
+ bool found = false;
+ while ((n = _clusterLogLevelList->next()) != NULL) {
+ if (n->getNodeId() == processId &&
+ n->getCategory() == ll.theCategories[i]) {
+
+ n->setLevel(ll.theLevels[i]);
+ found = true;
+ }
+ }
+ if (!found) {
+ _clusterLogLevelList->add(new NodeLogLevel(processId, ll));
+ }
+ }
+ }
+
+ int result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ EventSubscribeReq * dst =
+ CAST_PTR(EventSubscribeReq, signal->getDataPtrSend());
+ for(Uint32 i = 0; i<ll.noOfEntries; i++){
+ dst->theCategories[i] = ll.theCategories[i];
+ dst->theLevels[i] = ll.theLevels[i];
+ }
+
+ dst->noOfEntries = ll.noOfEntries;
+ dst->blockRef = _ownReference;
+
+ signal->set(TestOrd::TraceAPI, CMVMI, GSN_EVENT_SUBSCRIBE_REQ,
+ EventSubscribeReq::SignalLength);
+
+ result = sendSignal(processId, WAIT_SUBSCRIBE_CONF, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+ else {
+ // Increment the conf counter
+ theConfCount++;
+ }
+
+ return 0;
+}
+
+//****************************************************************************
+//****************************************************************************
+int
+MgmtSrvr::setNodeLogLevel(int processId, const SetLogLevelOrd & ll,
+ bool isResend)
+{
+ for(Uint32 i = 0; i<ll.noOfEntries; i++){
+ // Save log level for the cluster log
+ if (!isResend) {
+ NodeLogLevel* n = NULL;
+ bool found = false;
+ while ((n = _clusterLogLevelList->next()) != NULL) {
+ if (n->getNodeId() == processId &&
+ n->getCategory() == ll.theCategories[i]) {
+
+ n->setLevel(ll.theLevels[i]);
+ found = true;
+ }
+ }
+ if (!found) {
+ _clusterLogLevelList->add(new NodeLogLevel(processId, ll));
+ }
+ }
+ }
+
+ int result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ SetLogLevelOrd * dst = CAST_PTR(SetLogLevelOrd, signal->getDataPtrSend());
+
+ for(Uint32 i = 0; i<ll.noOfEntries; i++){
+ dst->theCategories[i] = ll.theCategories[i];
+ dst->theLevels[i] = ll.theLevels[i];
+ }
+
+ dst->noOfEntries = ll.noOfEntries;
+
+ signal->set(TestOrd::TraceAPI, CMVMI, GSN_SET_LOGLEVELORD,
+ SetLogLevelOrd::SignalLength);
+
+ result = sendSignal(processId, NO_WAIT, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ return 0;
+}
+
+
+//****************************************************************************
+//****************************************************************************
+
+int
+MgmtSrvr::insertError(int processId, int errorNo)
+{
+ if (errorNo < 0) {
+ return INVALID_ERROR_NUMBER;
+ }
+
+ int result;
+ result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ TamperOrd* const tamperOrd = CAST_PTR(TamperOrd, signal->getDataPtrSend());
+ tamperOrd->errorNo = errorNo;
+ signal->set(TestOrd::TraceAPI, CMVMI, GSN_TAMPER_ORD,
+ TamperOrd::SignalLength);
+
+ result = sendSignal(processId, NO_WAIT, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ return 0;
+}
+
+
+
+//****************************************************************************
+//****************************************************************************
+
+int
+MgmtSrvr::setTraceNo(int processId, int traceNo)
+{
+ if (traceNo < 0) {
+ return INVALID_TRACE_NUMBER;
+ }
+
+ int result;
+ result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ TestOrd* const testOrd = CAST_PTR(TestOrd, signal->getDataPtrSend());
+ testOrd->clear();
+
+ // Assume TRACE command causes toggling. Not really defined... ? TODO
+ testOrd->setTraceCommand(TestOrd::Toggle,
+ (TestOrd::TraceSpecification)traceNo);
+ signal->set(TestOrd::TraceAPI, CMVMI, GSN_TEST_ORD, TestOrd::SignalLength);
+
+ result = sendSignal(processId, NO_WAIT, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ return 0;
+}
+
+//****************************************************************************
+//****************************************************************************
+
+int
+MgmtSrvr::getBlockNumber(const BaseString &blockName)
+{
+ short bno = getBlockNo(blockName.c_str());
+ if(bno != 0)
+ return bno;
+ return -1;
+}
+
+//****************************************************************************
+//****************************************************************************
+
+int
+MgmtSrvr::setSignalLoggingMode(int processId, LogMode mode,
+ const Vector<BaseString>& blocks)
+{
+ int result;
+ result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ // Convert from MgmtSrvr format...
+
+ TestOrd::Command command;
+ if (mode == Off) {
+ command = TestOrd::Off;
+ }
+ else {
+ command = TestOrd::On;
+ }
+
+ TestOrd::SignalLoggerSpecification logSpec;
+ switch (mode) {
+ case In:
+ logSpec = TestOrd::InputSignals;
+ break;
+ case Out:
+ logSpec = TestOrd::OutputSignals;
+ break;
+ case InOut:
+ logSpec = TestOrd::InputOutputSignals;
+ break;
+ case Off:
+ // In MgmtSrvr interface it's just possible to switch off all logging, both
+ // "in" and "out" (this should probably be changed).
+ logSpec = TestOrd::InputOutputSignals;
+ break;
+ default:
+ NDB_ASSERT(false, "Unexpected value, MgmtSrvr::setSignalLoggingMode");
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ TestOrd* const testOrd = CAST_PTR(TestOrd, signal->getDataPtrSend());
+ testOrd->clear();
+
+ if (blocks.size() == 0 || blocks[0] == "ALL") {
+ // Logg command for all blocks
+ testOrd->addSignalLoggerCommand(command, logSpec);
+ } else {
+ for(unsigned i = 0; i < blocks.size(); i++){
+ int blockNumber = getBlockNumber(blocks[i]);
+ if (blockNumber == -1) {
+ releaseSignal(signal);
+ return INVALID_BLOCK_NAME;
+ }
+ testOrd->addSignalLoggerCommand(blockNumber, command, logSpec);
+ } // for
+ } // else
+
+ signal->set(TestOrd::TraceAPI, CMVMI, GSN_TEST_ORD, TestOrd::SignalLength);
+ result = sendSignal(processId, NO_WAIT, signal, true);
+
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+ return 0;
+}
+
+
+/*****************************************************************************
+ * Signal tracing
+ *****************************************************************************/
+int MgmtSrvr::startSignalTracing(int processId)
+{
+ int result;
+ result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+
+ TestOrd* const testOrd = CAST_PTR(TestOrd, signal->getDataPtrSend());
+ testOrd->clear();
+ testOrd->setTestCommand(TestOrd::On);
+
+ signal->set(TestOrd::TraceAPI, CMVMI, GSN_TEST_ORD, TestOrd::SignalLength);
+ result = sendSignal(processId, NO_WAIT, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ return 0;
+}
+
+int
+MgmtSrvr::stopSignalTracing(int processId)
+{
+ int result;
+ result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ TestOrd* const testOrd = CAST_PTR(TestOrd, signal->getDataPtrSend());
+ testOrd->clear();
+ testOrd->setTestCommand(TestOrd::Off);
+
+ signal->set(TestOrd::TraceAPI, CMVMI, GSN_TEST_ORD, TestOrd::SignalLength);
+ result = sendSignal(processId, NO_WAIT, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ return 0;
+}
+
+
+/*****************************************************************************
+ * Dump state
+ *****************************************************************************/
+
+int
+MgmtSrvr::dumpState(int processId, const char* args)
+{
+ // Convert the space separeted args
+ // string to an int array
+ Uint32 args_array[25];
+ Uint32 numArgs = 0;
+
+ char buf[10];
+ int b = 0;
+ memset(buf, 0, 10);
+ for (size_t i = 0; i <= strlen(args); i++){
+ if (args[i] == ' ' || args[i] == 0){
+ args_array[numArgs] = atoi(buf);
+ numArgs++;
+ memset(buf, 0, 10);
+ b = 0;
+ } else {
+ buf[b] = args[i];
+ b++;
+ }
+ }
+
+ return dumpState(processId, args_array, numArgs);
+}
+
+int
+MgmtSrvr::dumpState(int processId, const Uint32 args[], Uint32 no)
+{
+ int result;
+
+ result = okToSendTo(processId, true);
+ if (result != 0) {
+ return result;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ const Uint32 len = no > 25 ? 25 : no;
+
+ DumpStateOrd * const dumpOrd =
+ CAST_PTR(DumpStateOrd, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, CMVMI, GSN_DUMP_STATE_ORD, len);
+ for(Uint32 i = 0; i<25; i++){
+ if (i < len)
+ dumpOrd->args[i] = args[i];
+ else
+ dumpOrd->args[i] = 0;
+ }
+
+ result = sendSignal(processId, NO_WAIT, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ return 0;
+}
+
+
+//****************************************************************************
+//****************************************************************************
+
+const char* MgmtSrvr::getErrorText(int errorCode)
+{
+ static char text[255];
+
+ for (int i = 0; i < noOfErrorCodes; ++i) {
+ if (errorCode == errorTable[i]._errorCode) {
+ return errorTable[i]._errorText.c_str();
+ }
+ }
+
+ snprintf(text, 255, "Unknown management server error code %d", errorCode);
+ return text;
+}
+
+/*****************************************************************************
+ * Handle reception of various signals
+ *****************************************************************************/
+
+int
+MgmtSrvr::handleSTATISTICS_CONF(NdbApiSignal* signal)
+{
+ //ndbout << "MgmtSrvr::handleSTATISTICS_CONF" << endl;
+
+ int x = signal->readData(1);
+ //ndbout << "MgmtSrvr::handleSTATISTICS_CONF, x: " << x << endl;
+ _statistics._test1 = x;
+ return 0;
+}
+
+void
+MgmtSrvr::handleReceivedSignal(NdbApiSignal* signal)
+{
+ // The way of handling a received signal is taken from the Ndb class.
+ int returnCode;
+
+ int gsn = signal->readSignalNumber();
+ switch (gsn) {
+ case GSN_API_VERSION_CONF: {
+ if (theWaitState == WAIT_VERSION) {
+ const ApiVersionConf * const conf =
+ CAST_CONSTPTR(ApiVersionConf, signal->getDataPtr());
+ if(m_versionRec.callback != 0)
+ m_versionRec.callback(conf->nodeId, conf->version, this, 0);
+ else {
+ m_versionRec.version[conf->nodeId]=conf->version;
+ }
+ } else return;
+ theWaitState = NO_WAIT;
+ }
+ break;
+
+ case GSN_STATISTICS_CONF:
+ if (theWaitState != WAIT_STATISTICS) {
+ g_EventLogger.warning("MgmtSrvr::handleReceivedSignal, unexpected "
+ "signal received, gsn %d, theWaitState = %d",
+ gsn, theWaitState);
+
+ return;
+ }
+ returnCode = handleSTATISTICS_CONF(signal);
+ if (returnCode != -1) {
+ theWaitState = NO_WAIT;
+ }
+ break;
+
+
+ case GSN_SET_VAR_CONF:
+ if (theWaitState != WAIT_SET_VAR) {
+ g_EventLogger.warning("MgmtSrvr::handleReceivedSignal, unexpected "
+ "signal received, gsn %d, theWaitState = %d",
+ gsn, theWaitState);
+ return;
+ }
+ theWaitState = NO_WAIT;
+ _setVarReqResult = 0;
+ break;
+
+ case GSN_SET_VAR_REF:
+ if (theWaitState != WAIT_SET_VAR) {
+ g_EventLogger.warning("MgmtSrvr::handleReceivedSignal, unexpected "
+ "signal received, gsn %d, theWaitState = %d",
+ gsn, theWaitState);
+ return;
+ }
+ theWaitState = NO_WAIT;
+ _setVarReqResult = -1;
+ break;
+
+ case GSN_EVENT_SUBSCRIBE_CONF:
+ theConfCount--; // OK, we've received a conf message
+ if (theConfCount < 0) {
+ g_EventLogger.warning("MgmtSrvr::handleReceivedSignal, unexpected "
+ "signal received, gsn %d, theWaitState = %d",
+ gsn, theWaitState);
+ theConfCount = 0;
+ }
+ break;
+
+ case GSN_EVENT_REP:
+ eventReport(refToNode(signal->theSendersBlockRef), signal->getDataPtr());
+ break;
+
+ case GSN_STOP_REF:{
+ const StopRef * const ref = CAST_CONSTPTR(StopRef, signal->getDataPtr());
+ const NodeId nodeId = refToNode(signal->theSendersBlockRef);
+ handleStopReply(nodeId, ref->errorCode);
+ return;
+ }
+ break;
+
+ case GSN_BACKUP_CONF:{
+ const BackupConf * const conf =
+ CAST_CONSTPTR(BackupConf, signal->getDataPtr());
+ BackupEvent event;
+ event.Event = BackupEvent::BackupStarted;
+ event.Started.BackupId = conf->backupId;
+ event.Started.Nodes = conf->nodes;
+#ifdef VM_TRACE
+ ndbout_c("Backup master is %d", refToNode(signal->theSendersBlockRef));
+#endif
+ backupCallback(event);
+ }
+ break;
+
+ case GSN_BACKUP_REF:{
+ const BackupRef * const ref =
+ CAST_CONSTPTR(BackupRef, signal->getDataPtr());
+ Uint32 errCode = ref->errorCode;
+ if(ref->errorCode == BackupRef::IAmNotMaster){
+ const Uint32 aNodeId = refToNode(ref->masterRef);
+#ifdef VM_TRACE
+ ndbout_c("I'm not master resending to %d", aNodeId);
+#endif
+ NdbApiSignal aSignal(_ownReference);
+ BackupReq* req = CAST_PTR(BackupReq, aSignal.getDataPtrSend());
+ aSignal.set(TestOrd::TraceAPI, BACKUP, GSN_BACKUP_REQ,
+ BackupReq::SignalLength);
+ req->senderData = 19;
+ req->backupDataLen = 0;
+
+ int i = TransporterFacade::instance()->sendSignalUnCond(&aSignal,
+ aNodeId);
+ if(i == 0){
+ return;
+ }
+ errCode = 5030;
+ }
+ BackupEvent event;
+ event.Event = BackupEvent::BackupFailedToStart;
+ event.FailedToStart.ErrorCode = errCode;
+ backupCallback(event);
+ break;
+ }
+
+ case GSN_BACKUP_ABORT_REP:{
+ const BackupAbortRep * const rep =
+ CAST_CONSTPTR(BackupAbortRep, signal->getDataPtr());
+ BackupEvent event;
+ event.Event = BackupEvent::BackupAborted;
+ event.Aborted.Reason = rep->reason;
+ event.Aborted.BackupId = rep->backupId;
+ backupCallback(event);
+ }
+ break;
+
+ case GSN_BACKUP_COMPLETE_REP:{
+ const BackupCompleteRep * const rep =
+ CAST_CONSTPTR(BackupCompleteRep, signal->getDataPtr());
+ BackupEvent event;
+ event.Event = BackupEvent::BackupCompleted;
+ event.Completed.BackupId = rep->backupId;
+
+ event.Completed.NoOfBytes = rep->noOfBytes;
+ event.Completed.NoOfLogBytes = rep->noOfLogBytes;
+ event.Completed.NoOfRecords = rep->noOfRecords;
+ event.Completed.NoOfLogRecords = rep->noOfLogRecords;
+
+ event.Completed.stopGCP = rep->stopGCP;
+ event.Completed.startGCP = rep->startGCP;
+ event.Completed.Nodes = rep->nodes;
+
+ backupCallback(event);
+ }
+ break;
+
+ case GSN_MGM_LOCK_CONFIG_REP:
+ case GSN_MGM_LOCK_CONFIG_REQ:
+ case GSN_MGM_UNLOCK_CONFIG_REP:
+ case GSN_MGM_UNLOCK_CONFIG_REQ: {
+ m_signalRecvQueue.receive(new NdbApiSignal(*signal));
+ break;
+ }
+
+ default:
+ g_EventLogger.error("Unknown signal received. SignalNumber: "
+ "%i from (%d, %x)",
+ gsn,
+ refToNode(signal->theSendersBlockRef),
+ refToBlock(signal->theSendersBlockRef));
+ }
+
+ if (theWaitState == NO_WAIT) {
+ NdbCondition_Signal(theMgmtWaitForResponseCondPtr);
+ }
+}
+
+/**
+ * A database node was either stopped or there was some error
+ */
+void
+MgmtSrvr::handleStopReply(NodeId nodeId, Uint32 errCode)
+{
+ /**
+ * If we are in single user mode and get a stop reply from a
+ * DB node, then we have had a node crash.
+ * If all DB nodes are gone, and we are still in single user mode,
+ * the set m_stopRec.singleUserMode = false;
+ */
+ if(m_stopRec.singleUserMode) {
+ ClusterMgr::Node node;
+ bool failure = true;
+ for(Uint32 i = 0; i<MAX_NODES; i++) {
+ if (getNodeType(i) == NDB_MGM_NODE_TYPE_NDB) {
+ node = TransporterFacade::instance()->theClusterMgr->getNodeInfo(i);
+ if((node.m_state.startLevel == NodeState::SL_NOTHING))
+ failure = true;
+ else
+ failure = false;
+ }
+ }
+ if(failure) {
+ m_stopRec.singleUserMode = false;
+ }
+ }
+ if(m_stopRec.inUse == false)
+ return;
+
+ if(!(m_stopRec.nodeId == 0 || m_stopRec.nodeId == nodeId))
+ goto error;
+
+ if(m_stopRec.sentCount <= 0)
+ goto error;
+
+ if(!(theWaitState == WAIT_STOP || m_stopRec.callback != 0))
+ goto error;
+
+ if(errCode != 0)
+ m_stopRec.reply = translateStopRef(errCode);
+
+ m_stopRec.sentCount --;
+ if(m_stopRec.sentCount == 0){
+ if(theWaitState == WAIT_STOP){
+ theWaitState = NO_WAIT;
+ NdbCondition_Signal(theMgmtWaitForResponseCondPtr);
+ return;
+ }
+ if(m_stopRec.callback != 0){
+ m_stopRec.inUse = false;
+ StopCallback callback = m_stopRec.callback;
+ m_stopRec.callback = NULL;
+ (* callback)(m_stopRec.nodeId,
+ m_stopRec.anyData,
+ m_stopRec.reply);
+ return;
+ }
+ }
+ return;
+
+ error:
+ if(errCode != 0){
+ g_EventLogger.error("Unexpected signal received. SignalNumber: %i from %d",
+ GSN_STOP_REF, nodeId);
+ }
+}
+
+void
+MgmtSrvr::handleStatus(NodeId nodeId, bool alive)
+{
+ if (alive) {
+ _startedNodeId = nodeId; // Used by logLevelThreadRun()
+ Uint32 theData[25];
+ theData[0] = EventReport::Connected;
+ theData[1] = nodeId;
+ } else {
+ handleStopReply(nodeId, 0);
+ theConfCount++; // Increment the event subscr conf count because
+
+ Uint32 theData[25];
+ theData[0] = EventReport::Disconnected;
+ theData[1] = nodeId;
+
+ eventReport(_ownNodeId, theData);
+ g_EventLogger.info("Lost connection to node %d", nodeId);
+ }
+}
+
+//****************************************************************************
+//****************************************************************************
+
+void
+MgmtSrvr::signalReceivedNotification(void* mgmtSrvr,
+ NdbApiSignal* signal,
+ LinearSectionPtr ptr[3])
+{
+ ((MgmtSrvr*)mgmtSrvr)->handleReceivedSignal(signal);
+}
+
+
+//****************************************************************************
+//****************************************************************************
+void
+MgmtSrvr::nodeStatusNotification(void* mgmSrv, NodeId nodeId,
+ bool alive, bool nfComplete)
+{
+ if(!(!alive && nfComplete))
+ ((MgmtSrvr*)mgmSrv)->handleStatus(nodeId, alive);
+}
+
+enum ndb_mgm_node_type
+MgmtSrvr::getNodeType(NodeId nodeId) const
+{
+ if(nodeId >= MAX_NODES)
+ return (enum ndb_mgm_node_type)-1;
+
+ return nodeTypes[nodeId];
+}
+
+bool
+MgmtSrvr::getNextNodeId(NodeId * nodeId, enum ndb_mgm_node_type type) const
+{
+ NodeId tmp = * nodeId;
+
+ tmp++;
+ while(nodeTypes[tmp] != type && tmp < MAX_NODES)
+ tmp++;
+
+ if(tmp == MAX_NODES)
+ return false;
+
+ * nodeId = tmp;
+ return true;
+}
+
+void
+MgmtSrvr::eventReport(NodeId nodeId, const Uint32 * theData)
+{
+ const EventReport * const eventReport = (EventReport *)&theData[0];
+
+ EventReport::EventType type = eventReport->getEventType();
+
+ if (type == EventReport::TransReportCounters ||
+ type == EventReport::OperationReportCounters) {
+
+ if (_isClusterLogStatActive) {
+ g_EventLogger.log(type, theData, nodeId);
+ }
+
+ if (_isStatPortActive) {
+ char theTime[128];
+ struct tm* tm_now;
+ time_t now;
+ now = time((time_t*)NULL);
+#ifdef NDB_WIN32
+ tm_now = localtime(&now);
+#else
+ tm_now = gmtime(&now);
+#endif
+
+ snprintf(theTime, sizeof(theTime),
+ STATISTIC_DATE,
+ tm_now->tm_year + 1900,
+ tm_now->tm_mon,
+ tm_now->tm_mday,
+ tm_now->tm_hour,
+ tm_now->tm_min,
+ tm_now->tm_sec);
+
+ char str[255];
+
+ if (type == EventReport::TransReportCounters) {
+ snprintf(str, sizeof(str),
+ STATISTIC_LINE,
+ theTime,
+ (int)now,
+ nodeId,
+ theData[1],
+ theData[2],
+ theData[3],
+ // theData[4], simple reads
+ theData[5],
+ theData[6],
+ theData[7],
+ theData[8]);
+ } else if (type == EventReport::OperationReportCounters) {
+ snprintf(str, sizeof(str),
+ OP_STATISTIC_LINE,
+ theTime,
+ (int)now,
+ nodeId,
+ theData[1]);
+ }
+
+ if(m_statisticsListner != 0){
+ m_statisticsListner->println_statistics(str);
+ }
+ }
+
+ return;
+
+ } // if (type ==
+
+ // Log event
+ g_EventLogger.log(type, theData, nodeId);
+
+}
+
+/***************************************************************************
+ * Backup
+ ***************************************************************************/
+
+MgmtSrvr::BackupCallback
+MgmtSrvr::setCallback(BackupCallback aCall)
+{
+ BackupCallback ret = m_backupCallback;
+ m_backupCallback = aCall;
+ return ret;
+}
+
+int
+MgmtSrvr::startBackup(Uint32& backupId, bool waitCompleted)
+{
+ bool next;
+ NodeId nodeId = 0;
+ while((next = getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)) == true &&
+ theFacade->get_node_alive(nodeId) == false);
+
+ if(!next) return NO_CONTACT_WITH_DB_NODES;
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ BackupReq* req = CAST_PTR(BackupReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, BACKUP, GSN_BACKUP_REQ,
+ BackupReq::SignalLength);
+
+ req->senderData = 19;
+ req->backupDataLen = 0;
+
+ int result;
+ if (waitCompleted) {
+ result = sendRecSignal(nodeId, WAIT_BACKUP_COMPLETED, signal, true);
+ }
+ else {
+ result = sendRecSignal(nodeId, WAIT_BACKUP_STARTED, signal, true);
+ }
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ if (waitCompleted){
+ switch(m_lastBackupEvent.Event){
+ case BackupEvent::BackupCompleted:
+ backupId = m_lastBackupEvent.Completed.BackupId;
+ break;
+ case BackupEvent::BackupStarted:
+ backupId = m_lastBackupEvent.Started.BackupId;
+ break;
+ case BackupEvent::BackupFailedToStart:
+ return m_lastBackupEvent.FailedToStart.ErrorCode;
+ case BackupEvent::BackupAborted:
+ return m_lastBackupEvent.Aborted.ErrorCode;
+ default:
+ return -1;
+ break;
+ }
+ } else {
+ switch(m_lastBackupEvent.Event){
+ case BackupEvent::BackupCompleted:
+ backupId = m_lastBackupEvent.Completed.BackupId;
+ break;
+ case BackupEvent::BackupStarted:
+ backupId = m_lastBackupEvent.Started.BackupId;
+ break;
+ case BackupEvent::BackupFailedToStart:
+ return m_lastBackupEvent.FailedToStart.ErrorCode;
+ case BackupEvent::BackupAborted:
+ return m_lastBackupEvent.Aborted.ErrorCode;
+ default:
+ return -1;
+ break;
+ }
+ }
+
+ return 0;
+}
+
+int
+MgmtSrvr::abortBackup(Uint32 backupId)
+{
+ bool next;
+ NodeId nodeId = 0;
+ while((next = getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)) == true &&
+ theFacade->get_node_alive(nodeId) == false);
+
+ if(!next){
+ return NO_CONTACT_WITH_DB_NODES;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ AbortBackupOrd* ord = CAST_PTR(AbortBackupOrd, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, BACKUP, GSN_ABORT_BACKUP_ORD,
+ AbortBackupOrd::SignalLength);
+
+ ord->requestType = AbortBackupOrd::ClientAbort;
+ ord->senderData = 19;
+ ord->backupId = backupId;
+
+ int result = sendSignal(nodeId, NO_WAIT, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ return 0;
+}
+
+void
+MgmtSrvr::backupCallback(BackupEvent & event)
+{
+ char str[255];
+
+ bool ok = false;
+ switch(event.Event){
+ case BackupEvent::BackupStarted:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup %d started", event.Started.BackupId);
+ break;
+ case BackupEvent::BackupFailedToStart:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup failed to start (Backup error %d)",
+ event.FailedToStart.ErrorCode);
+ break;
+ case BackupEvent::BackupCompleted:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup %d completed",
+ event.Completed.BackupId);
+ g_EventLogger.info(str);
+
+ snprintf(str, sizeof(str),
+ " StartGCP: %d StopGCP: %d",
+ event.Completed.startGCP, event.Completed.stopGCP);
+ g_EventLogger.info(str);
+
+ snprintf(str, sizeof(str),
+ " #Records: %d #LogRecords: %d",
+ event.Completed.NoOfRecords, event.Completed.NoOfLogRecords);
+ g_EventLogger.info(str);
+
+ snprintf(str, sizeof(str),
+ " Data: %d bytes Log: %d bytes",
+ event.Completed.NoOfBytes, event.Completed.NoOfLogBytes);
+ break;
+ case BackupEvent::BackupAborted:
+ ok = true;
+ snprintf(str, sizeof(str),
+ "Backup %d has been aborted reason %d",
+ event.Aborted.BackupId,
+ event.Aborted.Reason);
+ break;
+ }
+ if(!ok){
+ snprintf(str, sizeof(str),
+ "Unknown backup event: %d",
+ event.Event);
+
+ }
+ g_EventLogger.info(str);
+
+ switch (theWaitState){
+ case WAIT_BACKUP_STARTED:
+ switch(event.Event){
+ case BackupEvent::BackupStarted:
+ case BackupEvent::BackupFailedToStart:
+ m_lastBackupEvent = event;
+ theWaitState = NO_WAIT;
+ break;
+ default:
+ snprintf(str, sizeof(str),
+ "Received event %d in unexpected state WAIT_BACKUP_STARTED",
+ event.Event);
+ g_EventLogger.info(str);
+ return;
+ }
+
+ break;
+ case WAIT_BACKUP_COMPLETED:
+ switch(event.Event){
+ case BackupEvent::BackupCompleted:
+ case BackupEvent::BackupAborted:
+ case BackupEvent::BackupFailedToStart:
+ m_lastBackupEvent = event;
+ theWaitState = NO_WAIT;
+ break;
+ default:
+ snprintf(str, sizeof(str),
+ "Received event %d in unexpected state WAIT_BACKUP_COMPLETED",
+ event.Event);
+ g_EventLogger.info(str);
+ return;
+ }
+ break;
+ default:
+ snprintf(str, sizeof(str), "Received event %d in unexpected state = %d",
+ event.Event, theWaitState);
+ g_EventLogger.info(str);
+ return;
+
+ }
+
+ if(m_backupCallback != 0){
+ (* m_backupCallback)(event);
+ }
+}
+
+
+/*****************************************************************************
+ * Global Replication
+ *****************************************************************************/
+
+int
+MgmtSrvr::repCommand(Uint32* repReqId, Uint32 request, bool waitCompleted)
+{
+ bool next;
+ NodeId nodeId = 0;
+
+ while((next = getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)) == true &&
+ theFacade->get_node_alive(nodeId) == false);
+
+ if(!next){
+ return NO_CONTACT_WITH_DB_NODES;
+ }
+
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ return COULD_NOT_ALLOCATE_MEMORY;
+ }
+
+ GrepReq* req = CAST_PTR(GrepReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI, GREP, GSN_GREP_REQ, GrepReq::SignalLength);
+ req->senderRef = _ownReference;
+ req->request = request;
+
+ int result;
+ if (waitCompleted)
+ result = sendRecSignal(nodeId, NO_WAIT, signal, true);
+ else
+ result = sendRecSignal(nodeId, NO_WAIT, signal, true);
+ if (result == -1) {
+ return SEND_OR_RECEIVE_FAILED;
+ }
+
+ /**
+ * @todo
+ * Maybe add that we should receive a confirmation that the
+ * request was received ok.
+ * Then we should give the user the correct repReqId.
+ */
+
+ *repReqId = 4711;
+
+ return 0;
+}
+
+
+/*****************************************************************************
+ * Area 51 ???
+ *****************************************************************************/
+
+MgmtSrvr::Area51
+MgmtSrvr::getStuff()
+{
+ Area51 ret;
+ ret.theFacade = theFacade;
+ ret.theRegistry = theFacade->theTransporterRegistry;
+ return ret;
+}
+
+NodeId
+MgmtSrvr::getPrimaryNode() const
+{
+ Uint32 tmp;
+ const Properties *prop = NULL;
+
+ getConfig()->get("SYSTEM", &prop);
+ if(prop == NULL)
+ return 0;
+
+ prop->get("PrimaryMGMNode", &tmp);
+
+ return tmp;
+}
diff --git a/ndb/src/mgmsrv/MgmtSrvr.hpp b/ndb/src/mgmsrv/MgmtSrvr.hpp
new file mode 100644
index 00000000000..4fdf3c99d43
--- /dev/null
+++ b/ndb/src/mgmsrv/MgmtSrvr.hpp
@@ -0,0 +1,780 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MgmtSrvr_H
+#define MgmtSrvr_H
+
+#include <kernel_types.h>
+#include "Config.hpp"
+#include <NdbCondition.h>
+#include <NdbConstant.hpp>
+#include <mgmapi.h>
+
+
+#include <Vector.hpp>
+#include <NodeBitmask.hpp>
+#include <signaldata/ManagementServer.hpp>
+#include "SignalQueue.hpp"
+
+#include "NodeLogLevelList.hpp"
+
+/**
+ * @desc Block number for Management server.
+ * @todo This should probably be somewhere else. I don't know where atm.
+ */
+#define MGMSRV 1
+
+class ConfigInfoServer;
+class NdbApiSignal;
+class Config;
+class SetLogLevelOrd;
+class SocketServer;
+
+/**
+ * @class MgmtSrvr
+ * @brief Main class for the management server.
+ *
+ * It has one interface to be used by a local client.
+ * With the methods it's possible to send different kind of commands to
+ * DB processes, as log level, set trace number etc.
+ *
+ * A MgmtSrvr creates a ConfigInfoServer which serves request on TCP sockets.
+ * The requests come typical from DB and API processes which want
+ * to fetch its configuration parameters. The MgmtSrvr knows about the
+ * configuration by reading a configuration file.
+ *
+ * The MgmtSrvr class corresponds in some ways to the Ndb class in API.
+ * It creates a TransporterFacade, receives signals and defines signals
+ * to send and receive.
+ */
+class MgmtSrvr {
+
+public:
+ class StatisticsListner {
+ public:
+ virtual void println_statistics(const BaseString &s) = 0;
+ };
+
+ /**
+ * Set a reference to the socket server.
+ */
+ void setStatisticsListner(StatisticsListner* listner);
+
+ /**
+ * Start/initate the event log.
+ */
+ void startEventLog();
+
+ /**
+ * Stop the event log.
+ */
+ void stopEventLog();
+
+ /**
+ * Enable/disable eventlog log levels/severities.
+ *
+ * @param serverity the log level/serverity.
+ * @return true if the severity was enabled.
+ */
+ bool setEventLogFilter(int severity);
+
+ /**
+ * Returns true if the log level/severity is enabled.
+ *
+ * @param severity the severity level.
+ */
+ bool isEventLogFilterEnabled(int severity);
+
+ STATIC_CONST( NO_CONTACT_WITH_PROCESS = 5000 );
+ STATIC_CONST( PROCESS_NOT_CONFIGURED = 5001 );
+ STATIC_CONST( WRONG_PROCESS_TYPE = 5002 );
+ STATIC_CONST( COULD_NOT_ALLOCATE_MEMORY = 5003 );
+ STATIC_CONST( SEND_OR_RECEIVE_FAILED = 5005 );
+ STATIC_CONST( INVALID_LEVEL = 5006 );
+ STATIC_CONST( INVALID_ERROR_NUMBER = 5007 );
+ STATIC_CONST( INVALID_TRACE_NUMBER = 5008 );
+ STATIC_CONST( NOT_IMPLEMENTED = 5009 );
+ STATIC_CONST( INVALID_BLOCK_NAME = 5010 );
+
+ STATIC_CONST( CONFIG_PARAM_NOT_EXIST = 5011 );
+ STATIC_CONST( CONFIG_PARAM_NOT_UPDATEABLE = 5012 );
+ STATIC_CONST( VALUE_WRONG_FORMAT_INT_EXPECTED = 5013 );
+ STATIC_CONST( VALUE_TOO_LOW = 5014 );
+ STATIC_CONST( VALUE_TOO_HIGH = 5015 );
+ STATIC_CONST( VALUE_WRONG_FORMAT_BOOL_EXPECTED = 5016 );
+
+ STATIC_CONST( CONFIG_FILE_OPEN_WRITE_ERROR = 5017 );
+ STATIC_CONST( CONFIG_FILE_OPEN_READ_ERROR = 5018 );
+ STATIC_CONST( CONFIG_FILE_WRITE_ERROR = 5019 );
+ STATIC_CONST( CONFIG_FILE_READ_ERROR = 5020 );
+ STATIC_CONST( CONFIG_FILE_CLOSE_ERROR = 5021 );
+
+ STATIC_CONST( CONFIG_CHANGE_REFUSED_BY_RECEIVER = 5022 );
+ STATIC_CONST( COULD_NOT_SYNC_CONFIG_CHANGE_AGAINST_PHYSICAL_MEDIUM = 5023 );
+ STATIC_CONST( CONFIG_FILE_CHECKSUM_ERROR = 5024 );
+ STATIC_CONST( NOT_POSSIBLE_TO_SEND_CONFIG_UPDATE_TO_PROCESS_TYPE = 5025 );
+
+ STATIC_CONST( NODE_SHUTDOWN_IN_PROGESS = 5026 );
+ STATIC_CONST( SYSTEM_SHUTDOWN_IN_PROGRESS = 5027 );
+ STATIC_CONST( NODE_SHUTDOWN_WOULD_CAUSE_SYSTEM_CRASH = 5028 );
+ STATIC_CONST( NO_CONTACT_WITH_CLUSTER = 6666 );
+ STATIC_CONST( OPERATION_IN_PROGRESS = 6667 );
+
+ STATIC_CONST( NO_CONTACT_WITH_DB_NODES = 5030 );
+ /**
+ * This class holds all statistical variables fetched with
+ * the getStatistics methods.
+ */
+ class Statistics { // TODO, Real statistic data to be added
+ public:
+ int _test1;
+ };
+
+ /**
+ * This enum specifies the different signal loggig modes possible to set
+ * with the setSignalLoggingMode method.
+ */
+ enum LogMode {In, Out, InOut, Off};
+
+ /* Constructor */
+ MgmtSrvr(NodeId nodeId, /* Local nodeid */
+ const BaseString &config_filename, /* Where to save config */
+ const BaseString &ndb_config_filename); /* Ndb.cfg filename */
+
+ /**
+ * Read (initial) config file, create TransporterFacade,
+ * define signals, create ConfigInfoServer.
+ * @return true if succeeded, otherwise false
+ */
+ bool check_start(); // may be run before start to check that some things are ok
+ bool start();
+
+ ~MgmtSrvr();
+
+ int status(int processId,
+ ndb_mgm_node_status * status,
+ Uint32 * version,
+ Uint32 * phase,
+ bool * systemShutdown,
+ Uint32 * dynamicId,
+ Uint32 * nodeGroup);
+
+ // All the functions below may return any of this error codes:
+ // NO_CONTACT_WITH_PROCESS, PROCESS_NOT_CONFIGURED, WRONG_PROCESS_TYPE,
+ // COULD_NOT_ALLOCATE_MEMORY, SEND_OR_RECEIVE_FAILED
+
+
+ typedef void (* StopCallback)(int nodeId, void * anyData, int errorCode);
+
+ typedef void (* VersionCallback)(int nodeId, int version,
+ void * anyData, int errorCode);
+
+
+ typedef void (* EnterSingleCallback)(int nodeId, void * anyData,
+ int errorCode);
+ typedef void (* ExitSingleCallback)(int nodeId, void * anyData,
+ int errorCode);
+
+ /**
+ * Lock configuration
+ */
+ int lockConf();
+
+ /**
+ * Unlock configuration, and commit it if commit is true
+ */
+ int unlockConf(bool commit);
+
+ /**
+ * Commit new configuration
+ */
+ int commitConfig();
+
+ /**
+ * Rollback configuration
+ */
+ int rollbackConfig();
+
+ /**
+ * Save a configuration to permanent storage
+ */
+ int saveConfig(const Config *);
+
+ /**
+ * Save the running configuration
+ */
+ int saveConfig() {
+ return saveConfig(_config);
+ };
+
+ /**
+ * Read configuration from file, or from another MGM server
+ */
+ Config *readConfig();
+
+ /**
+ * Fetch configuration from another MGM server
+ */
+ Config *fetchConfig();
+
+ /**
+ * Stop a node
+ *
+ * @param processId: Id of the DB process to stop
+ * @return 0 if succeeded, otherwise: as stated above, plus:
+ */
+ int stopNode(int nodeId, bool abort = false, StopCallback = 0, void *any= 0);
+
+ /**
+ * Stop the system
+ */
+ int stop(int * cnt = 0, bool abort = false, StopCallback = 0, void *any = 0);
+
+ /**
+ * print version info about a node
+ *
+ * @param processId: Id of the DB process to stop
+ * @return 0 if succeeded, otherwise: as stated above, plus:
+ */
+ int versionNode(int nodeId, bool abort = false,
+ VersionCallback = 0, void *any= 0);
+
+ /**
+ * print version info about all node in the system
+ */
+ int version(int * cnt = 0, bool abort = false,
+ VersionCallback = 0, void *any = 0);
+
+ /**
+ * Maintenance on the system
+ */
+ int enterSingleUser(int * cnt = 0, Uint32 singleuserNodeId = 0,
+ EnterSingleCallback = 0, void *any = 0);
+
+
+ /**
+ * Resume from maintenance on the system
+ */
+ int exitSingleUser(int * cnt = 0, bool abort = false,
+ ExitSingleCallback = 0, void *any = 0);
+
+ /**
+ * Start DB process.
+ * @param processId: Id of the DB process to start
+ * @return 0 if succeeded, otherwise: as stated above, plus:
+ */
+ int start(int processId);
+
+ /**
+ * Restart a node
+ * @param processId: Id of the DB process to start
+ */
+ int restartNode(int processId, bool nostart, bool initialStart,
+ bool abort = false,
+ StopCallback = 0, void * anyData = 0);
+
+ /**
+ * Restart the system
+ */
+ int restart(bool nostart, bool initialStart,
+ bool abort = false,
+ int * stopCount = 0, StopCallback = 0, void * anyData = 0);
+
+ int setEventReportingLevel(int processId,
+ const class SetLogLevelOrd & logLevel,
+ bool isResend = false);
+
+ int startStatisticEventReporting(int level = 5);
+
+
+ struct BackupEvent {
+ enum Event {
+ BackupStarted = 1,
+ BackupFailedToStart = 2,
+ BackupCompleted = 3,
+ BackupAborted = 4
+ } Event;
+
+ union {
+ struct {
+ Uint32 BackupId;
+ NdbNodeBitmask Nodes;
+ } Started ;
+ struct {
+ Uint32 ErrorCode;
+ } FailedToStart ;
+ struct {
+ Uint32 BackupId;
+ Uint32 NoOfBytes;
+ Uint32 NoOfRecords;
+ Uint32 NoOfLogBytes;
+ Uint32 NoOfLogRecords;
+ NdbNodeBitmask Nodes;
+ Uint32 startGCP;
+ Uint32 stopGCP;
+ } Completed ;
+ struct {
+ Uint32 BackupId;
+ Uint32 Reason;
+ Uint32 ErrorCode;
+ } Aborted ;
+ };
+ };
+
+ /**
+ * Backup functionallity
+ */
+ typedef void (* BackupCallback)(const BackupEvent& Event);
+ BackupCallback setCallback(BackupCallback);
+ int startBackup(Uint32& backupId, bool waitCompleted = false);
+ int abortBackup(Uint32 backupId);
+ int performBackup(Uint32* backupId);
+
+ /**
+ * Global Replication
+ */
+ int repCommand(Uint32* repReqId, Uint32 request, bool waitCompleted = false);
+
+ //**************************************************************************
+ // Description: Set event report level for a DB process
+ // Parameters:
+ // processId: Id of the DB process
+ // level: Event report level
+ // isResend: Flag to indicate for resending log levels during node restart
+ // Returns: 0 if succeeded, otherwise: as stated above, plus:
+ // INVALID_LEVEL
+ //**************************************************************************
+
+ /**
+ * Sets the Node's log level, i.e., its local event reporting.
+ *
+ * @param processId the DB node id.
+ * @param logLevel the log level.
+ * @param isResend Flag to indicate for resending log levels
+ * during node restart
+
+ * @return 0 if successful or NO_CONTACT_WITH_PROCESS,
+ * SEND_OR_RECEIVE_FAILED,
+ * COULD_NOT_ALLOCATE_MEMORY
+ */
+ int setNodeLogLevel(int processId,
+ const class SetLogLevelOrd & logLevel,
+ bool isResend = false);
+
+
+ /**
+ * Insert an error in a DB process.
+ * @param processId: Id of the DB process
+ * @param errorNo: The error number. > 0.
+ * @return 0 if succeeded, otherwise: as stated above, plus:
+ * INVALID_ERROR_NUMBER
+ */
+ int insertError(int processId, int errorNo);
+
+
+
+ int setTraceNo(int processId, int traceNo);
+ //**************************************************************************
+ // Description: Set trace number in a DB process.
+ // Parameters:
+ // processId: Id of the DB process
+ // trace: Trace number
+ // Returns: 0 if succeeded, otherwise: as stated above, plus:
+ // INVALID_TRACE_NUMBER
+ //**************************************************************************
+
+
+ int setSignalLoggingMode(int processId, LogMode mode,
+ const Vector<BaseString> &blocks);
+
+ int setSignalLoggingMode(int processId, LogMode mode,
+ BaseString &block) {
+ Vector<BaseString> v;
+ v.push_back(block);
+ return setSignalLoggingMode(processId, mode, v);
+ }
+ //**************************************************************************
+ // Description: Set signal logging mode for blocks in a DB process.
+ // Parameters:
+ // processId: Id of the DB process
+ // mode: The log mode
+ // blocks: Which blocks to be affected (container of strings)
+ // Returns: 0 if succeeded, otherwise: as stated above, plus:
+ // INVALID_BLOCK_NAME
+ //**************************************************************************
+
+
+ int startSignalTracing(int processId);
+ //**************************************************************************
+ // Description: Start signal tracing for a DB process.
+ // Parameters:
+ // processId: Id of the DB process
+ // Returns: 0 if succeeded, otherwise: as stated above.
+ //**************************************************************************
+
+
+ int stopSignalTracing(int processId);
+ //**************************************************************************
+ // Description: Stop signal tracing for a DB process.
+ // Parameters:
+ // processId: Id of the DB process
+ // Returns: 0 if succeeded, otherwise: as stated above.
+ //**************************************************************************
+
+ /**
+ * Dump State
+ */
+ int dumpState(int processId, const Uint32 args[], Uint32 argNo);
+ int dumpState(int processId, const char* args);
+
+ /**
+ * Get next node id (node id gt that _nodeId)
+ * of specified type and save it in _nodeId
+ *
+ * @return false if none found
+ */
+ bool getNextNodeId(NodeId * _nodeId, enum ndb_mgm_node_type type) const ;
+
+ /**
+ *
+ */
+ enum ndb_mgm_node_type getNodeType(NodeId) const;
+
+ /**
+ * Get error text
+ *
+ * @param errorCode: Error code to get a match error text for.
+ * @return The error text.
+ */
+ const char* getErrorText(int errorCode);
+
+ /**
+ * Get configuration
+ */
+ const Config * getConfig() const;
+
+ /**
+ * Change configuration paramter
+ */
+ bool changeConfig(const BaseString &section,
+ const BaseString &param,
+ const BaseString &value);
+
+ /**
+ * Returns the node count for the specified node type.
+ *
+ * @param type The node type.
+ * @return The number of nodes of the specified type.
+ */
+ int getNodeCount(enum ndb_mgm_node_type type) const;
+
+ /**
+ * Returns the nodeId of the management master
+ */
+ NodeId getPrimaryNode() const;
+
+ /**
+ * Returns the statistics port number.
+ * @return statistic port number.
+ */
+ int getStatPort() const;
+
+
+ //**************************************************************************
+private:
+ //**************************************************************************
+
+ int setEventReportingLevelImpl(int processId,
+ const class SetLogLevelOrd & logLevel,
+ bool isResend = false);
+
+
+ /**
+ * Check if it is possible to send a signal to a (DB) process
+ *
+ * @param processId: Id of the process to send to
+ * @return 0 OK, 1 process dead, 2 API or MGMT process, 3 not configured
+ */
+ int okToSendTo(NodeId nodeId, bool unCond = false);
+
+ /**
+ * Get block number for a block
+ *
+ * @param blockName: Block to get number for
+ * @return -1 if block not found, otherwise block number
+ */
+ int getBlockNumber(const BaseString &blockName);
+
+ //**************************************************************************
+
+ int _blockNumber;
+ NodeId _ownNodeId;
+ BlockReference _ownReference;
+ NdbMutex *m_configMutex;
+ const Config * _config;
+ Config * m_newConfig;
+ BaseString m_configFilename;
+ BaseString m_localNdbConfigFilename;
+ Uint32 m_nextConfigGenerationNumber;
+
+ int _setVarReqResult; // The result of the SET_VAR_REQ response
+ Statistics _statistics; // handleSTATISTICS_CONF store the result here,
+ // and getStatistics reads it.
+
+
+
+ //**************************************************************************
+ // Specific signal handling methods
+ //**************************************************************************
+
+ static void defineSignals(int blockNumber);
+ //**************************************************************************
+ // Description: Define all signals to be sent or received for a block
+ // Parameters:
+ // blockNumber: The block number send/receive
+ // Returns: -
+ //**************************************************************************
+
+ void handleReceivedSignal(NdbApiSignal* signal);
+ //**************************************************************************
+ // Description: This method is called from "another" thread when a signal
+ // is received. If expect the received signal and succeed to handle it
+ // we signal with a condition variable to the waiting
+ // thread (receiveOptimisedResponse) that the signal has arrived.
+ // Parameters:
+ // signal: The recieved signal
+ // Returns: -
+ //**************************************************************************
+
+ void handleStatus(NodeId nodeId, bool alive);
+ //**************************************************************************
+ // Description: Handle the death of a process
+ // Parameters:
+ // processId: Id of the dead process.
+ // Returns: -
+ //**************************************************************************
+
+ int handleSTATISTICS_CONF(NdbApiSignal* signal);
+ //**************************************************************************
+ // Description: Handle reception of signal STATISTICS_CONF
+ // Parameters:
+ // signal: The recieved signal
+ // Returns: TODO, to be defined
+ //**************************************************************************
+
+ void handle_MGM_LOCK_CONFIG_REQ(NdbApiSignal *signal);
+ void handle_MGM_UNLOCK_CONFIG_REQ(NdbApiSignal *signal);
+
+ //**************************************************************************
+ // Specific signal handling data
+ //**************************************************************************
+
+
+ //**************************************************************************
+ //**************************************************************************
+ // General signal handling methods
+ // This functions are more or less copied from the Ndb class.
+
+
+ /**
+ * WaitSignalType defines states where each state define a set of signals
+ * we accept to receive.
+ * The state is set after we have sent a signal.
+ * When a signal arrives we first check current state (handleReceivedSignal)
+ * to verify that we expect the arrived signal.
+ * It's only then we are in state accepting the arrived signal
+ * we handle the signal.
+ */
+ enum WaitSignalType {
+ NO_WAIT, // We don't expect to receive any signal
+ WAIT_STATISTICS, // Accept STATISTICS_CONF
+ WAIT_SET_VAR, // Accept SET_VAR_CONF and SET_VAR_REF
+ WAIT_SUBSCRIBE_CONF, // Accept event subscription confirmation
+ WAIT_STOP,
+ WAIT_BACKUP_STARTED,
+ WAIT_BACKUP_COMPLETED,
+ WAIT_VERSION
+ };
+
+ /**
+ * Get an unused signal
+ * @return A signal if succeeded, NULL otherwise
+ */
+ NdbApiSignal* getSignal();
+
+ /**
+ * Add a signal to the list of unused signals
+ * @param signal: The signal to add
+ */
+ void releaseSignal(NdbApiSignal* signal);
+
+ /**
+ * Remove a signal from the list of unused signals and delete
+ * the memory for it.
+ */
+ void freeSignal();
+
+ /**
+ * Send a signal
+ * @param processId: Id of the receiver process
+ * @param waitState: State denoting a set of signals we accept to receive
+ * @param signal: The signal to send
+ * @return 0 if succeeded, -1 otherwise
+ */
+ int sendSignal(Uint16 processId, WaitSignalType waitState,
+ NdbApiSignal* signal, bool force = false);
+
+ /**
+ * Send a signal and wait for an answer signal
+ * @param processId: Id of the receiver process
+ * @param waitState: State denoting a set of signals we accept to receive.
+ * @param signal: The signal to send
+ * @return 0 if succeeded, -1 otherwise (for example failed to send or
+ * failed to receive expected signal).
+ */
+ int sendRecSignal(Uint16 processId, WaitSignalType waitState,
+ NdbApiSignal* signal, bool force = false,
+ int waitTime = WAIT_FOR_RESPONSE_TIMEOUT);
+
+ /**
+ * Wait for a signal to arrive.
+ * @return 0 if signal arrived, -1 otherwise
+ */
+ int receiveOptimisedResponse(int waitTime);
+
+ /**
+ * This function is called from "outside" of MgmtSrvr
+ * when a signal is sent to MgmtSrvr.
+ * @param mgmtSrvr: The MgmtSrvr object which shall recieve the signal.
+ * @param signal: The received signal.
+ */
+ static void signalReceivedNotification(void* mgmtSrvr,
+ NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]);
+
+ /**
+ * Called from "outside" of MgmtSrvr when a DB process has died.
+ * @param mgmtSrvr: The MgmtSrvr object wreceiveOptimisedResponsehich
+ * shall receive the notification.
+ * @param processId: Id of the dead process.
+ */
+ static void nodeStatusNotification(void* mgmSrv, NodeId nodeId,
+ bool alive, bool nfCompleted);
+
+ /**
+ * An event from <i>nodeId</i> has arrived
+ */
+ void eventReport(NodeId nodeId, const Uint32 * theData);
+
+
+ //**************************************************************************
+ //**************************************************************************
+ // General signal handling data
+
+ static const unsigned int WAIT_FOR_RESPONSE_TIMEOUT = 300000; // Milliseconds
+ // Max time to wait for a signal to arrive
+
+ NdbApiSignal* theSignalIdleList;
+ // List of unused signals
+
+ WaitSignalType theWaitState;
+ // State denoting a set of signals we accept to recieve.
+
+ NdbCondition* theMgmtWaitForResponseCondPtr;
+ // Condition variable used when we wait for a signal to arrive/a
+ // signal arrives.
+ // We wait in receiveOptimisedResponse and signal in handleReceivedSignal.
+
+ class TransporterFacade * theFacade;
+
+ class SignalQueue m_signalRecvQueue;
+
+ enum ndb_mgm_node_type nodeTypes[MAX_NODES];
+
+ int theConfCount; // The number of expected conf signals
+
+ StatisticsListner * m_statisticsListner; // Used for sending statistics info
+ bool _isStatPortActive;
+ bool _isClusterLogStatActive;
+
+ struct StopRecord {
+ StopRecord(){ inUse = false; callback = 0; singleUserMode = false;}
+ bool inUse;
+ bool singleUserMode;
+ int sentCount;
+ int reply;
+ int nodeId;
+ void * anyData;
+ StopCallback callback;
+ };
+ StopRecord m_stopRec;
+
+ struct VersionRecord {
+ VersionRecord(){ inUse = false; callback = 0;}
+ bool inUse;
+ Uint32 version[MAX_NODES];
+ VersionCallback callback;
+ };
+ VersionRecord m_versionRec;
+ int sendVersionReq( int processId);
+
+
+ void handleStopReply(NodeId nodeId, Uint32 errCode);
+ int translateStopRef(Uint32 errCode);
+
+ bool _isStopThread;
+ int _logLevelThreadSleep;
+ int _startedNodeId;
+
+ /**
+ * Handles the thread wich upon a 'Node is started' event will
+ * set the node's previous loglevel settings.
+ */
+ struct NdbThread* _logLevelThread;
+ static void *logLevelThread_C(void *);
+ void logLevelThreadRun();
+
+ struct NdbThread *m_signalRecvThread;
+ static void *signalRecvThread_C(void *);
+ void signalRecvThreadRun();
+
+ NodeLogLevelList* _nodeLogLevelList;
+ NodeLogLevelList* _clusterLogLevelList;
+
+ void backupCallback(BackupEvent &);
+ BackupCallback m_backupCallback;
+ BackupEvent m_lastBackupEvent;
+
+ Config *_props;
+
+public:
+ /**
+ * This method does not exist
+ */
+ struct Area51 {
+ class TransporterFacade * theFacade;
+ class TransporterRegistry * theRegistry;
+ };
+ Area51 getStuff();
+};
+
+inline
+const Config *
+MgmtSrvr::getConfig() const {
+ return _config;
+}
+
+#endif // MgmtSrvr_H
diff --git a/ndb/src/mgmsrv/MgmtSrvrConfig.cpp b/ndb/src/mgmsrv/MgmtSrvrConfig.cpp
new file mode 100644
index 00000000000..f4e53409b30
--- /dev/null
+++ b/ndb/src/mgmsrv/MgmtSrvrConfig.cpp
@@ -0,0 +1,312 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/TestOrd.hpp>
+#include <OutputStream.hpp>
+
+#include "MgmtSrvr.hpp"
+#include "SignalQueue.hpp"
+#include <InitConfigFileParser.hpp>
+#include <ConfigRetriever.hpp>
+#include <ndb_version.h>
+
+void
+MgmtSrvr::handle_MGM_LOCK_CONFIG_REQ(NdbApiSignal *signal) {
+ NodeId sender = refToNode(signal->theSendersBlockRef);
+ const MgmLockConfigReq * const req = CAST_CONSTPTR(MgmLockConfigReq, signal->getDataPtr());
+
+ NdbApiSignal *reply = getSignal();
+ if(signal == NULL)
+ return; /** @todo handle allocation failure */
+
+ reply->set(TestOrd::TraceAPI,
+ MGMSRV,
+ GSN_MGM_LOCK_CONFIG_REP,
+ MgmLockConfigRep::SignalLength);
+
+ MgmLockConfigRep *lockRep = CAST_PTR(MgmLockConfigRep, reply->getDataPtrSend());
+
+ lockRep->errorCode = MgmLockConfigRep::UNKNOWN_ERROR;
+
+ if(req->newConfigGeneration < m_nextConfigGenerationNumber) {
+ lockRep->errorCode = MgmLockConfigRep::GENERATION_MISMATCH;
+ goto done;
+ }
+ NdbMutex_Lock(m_configMutex);
+
+ m_nextConfigGenerationNumber = req->newConfigGeneration+1;
+
+ lockRep->errorCode = MgmLockConfigRep::OK;
+
+ done:
+ sendSignal(sender, NO_WAIT, reply, true);
+ NdbMutex_Unlock(m_configMutex);
+ return;
+}
+
+void
+MgmtSrvr::handle_MGM_UNLOCK_CONFIG_REQ(NdbApiSignal *signal) {
+ NodeId sender = refToNode(signal->theSendersBlockRef);
+ const MgmUnlockConfigReq * const req = CAST_CONSTPTR(MgmUnlockConfigReq, signal->getDataPtr());
+ MgmUnlockConfigRep *unlockRep;
+
+ NdbApiSignal *reply = getSignal();
+ if(signal == NULL)
+ goto error; /** @todo handle allocation failure */
+
+ reply->set(TestOrd::TraceAPI,
+ MGMSRV,
+ GSN_MGM_UNLOCK_CONFIG_REP,
+ MgmUnlockConfigRep::SignalLength);
+
+ unlockRep = CAST_PTR(MgmUnlockConfigRep, reply->getDataPtrSend());
+
+ unlockRep->errorCode = MgmUnlockConfigRep::UNKNOWN_ERROR;
+
+
+ NdbMutex_Lock(m_configMutex);
+
+ if(req->commitConfig == 1) {
+ m_newConfig = fetchConfig();
+ commitConfig();
+ } else
+ rollbackConfig();
+
+ unlockRep->errorCode = MgmUnlockConfigRep::OK;
+
+ sendSignal(sender, NO_WAIT, reply, true);
+ error:
+ NdbMutex_Unlock(m_configMutex);
+ return;
+}
+
+
+/**
+ * Prepare all MGM nodes for configuration changes
+ *
+ * @returns 0 on success, or -1 on failure
+ */
+int
+MgmtSrvr::lockConf() {
+ int result = -1;
+ MgmLockConfigReq* lockReq;
+ NodeId node = 0;
+
+ /* Check if this is the master node */
+ if(getPrimaryNode() != _ownNodeId)
+ goto done;
+
+ if(NdbMutex_Trylock(m_configMutex) != 0)
+ return -1;
+
+ m_newConfig = new Config(*_config); /* copy the existing config */
+ _config = m_newConfig;
+
+ m_newConfig = new Config(*_config);
+
+ m_nextConfigGenerationNumber++;
+
+ /* Make sure the new configuration _always_ is at least one step older */
+ if(m_nextConfigGenerationNumber < m_newConfig->getGenerationNumber()+1)
+ m_nextConfigGenerationNumber = _config->getGenerationNumber()+1;
+
+ m_newConfig->setGenerationNumber(m_nextConfigGenerationNumber);
+
+ node = 0;
+ while(getNextNodeId(&node, NDB_MGM_NODE_TYPE_MGM)) {
+ if(node != _ownNodeId) {
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ result = COULD_NOT_ALLOCATE_MEMORY;
+ goto done;
+ }
+
+ lockReq = CAST_PTR(MgmLockConfigReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI,
+ MGMSRV,
+ GSN_MGM_LOCK_CONFIG_REQ,
+ MgmLockConfigReq::SignalLength);
+
+ lockReq->newConfigGeneration = m_nextConfigGenerationNumber;
+
+ result = sendSignal(node, NO_WAIT, signal, true);
+
+ NdbApiSignal *reply =
+ m_signalRecvQueue.waitFor(GSN_MGM_LOCK_CONFIG_REP, 0);
+
+ if(reply == NULL) {
+ /** @todo handle timeout/error */
+ ndbout << __FILE__ << ":" << __LINE__ << endl;
+ result = -1;
+ goto done;
+ }
+
+ }
+ }
+
+ done:
+ NdbMutex_Unlock(m_configMutex);
+ return result;
+}
+
+/**
+ * Unlocks configuration
+ *
+ * @returns 0 on success, ! 0 on error
+ */
+int
+MgmtSrvr::unlockConf(bool commit) {
+ int result = -1;
+ MgmUnlockConfigReq* unlockReq;
+ NodeId node = 0;
+
+ /* Check if this is the master node */
+ if(getPrimaryNode() != _ownNodeId)
+ goto done;
+
+ errno = 0;
+ if(NdbMutex_Lock(m_configMutex) != 0)
+ return -1;
+
+ if(commit)
+ commitConfig();
+ else
+ rollbackConfig();
+
+ node = 0;
+ while(getNextNodeId(&node, NDB_MGM_NODE_TYPE_MGM)) {
+ if(node != _ownNodeId) {
+ NdbApiSignal* signal = getSignal();
+ if (signal == NULL) {
+ result = COULD_NOT_ALLOCATE_MEMORY;
+ goto done;
+ }
+
+ unlockReq = CAST_PTR(MgmUnlockConfigReq, signal->getDataPtrSend());
+ signal->set(TestOrd::TraceAPI,
+ MGMSRV,
+ GSN_MGM_UNLOCK_CONFIG_REQ,
+ MgmUnlockConfigReq::SignalLength);
+ unlockReq->commitConfig = commit;
+
+ result = sendSignal(node, NO_WAIT, signal, true);
+
+ NdbApiSignal *reply =
+ m_signalRecvQueue.waitFor(GSN_MGM_UNLOCK_CONFIG_REP, 0);
+
+ if(reply == NULL) {
+ /** @todo handle timeout/error */
+ result = -1;
+ goto done;
+ }
+
+ }
+ }
+
+ done:
+ NdbMutex_Unlock(m_configMutex);
+ return result;
+}
+
+/**
+ * Commit the new configuration
+ */
+int
+MgmtSrvr::commitConfig() {
+ int ret = saveConfig(m_newConfig);
+ delete _config;
+ _config = m_newConfig;
+ m_newConfig = NULL;
+ ndbout << "commit " << ret << endl;
+ return ret;
+}
+
+/**
+ * Rollback to the old configuration
+ */
+int
+MgmtSrvr::rollbackConfig() {
+ delete m_newConfig;
+ m_newConfig = NULL;
+ ndbout << "rollback" << endl;
+ return saveConfig(_config);
+}
+
+/**
+ * Save a configuration to the running configuration file
+ */
+int
+MgmtSrvr::saveConfig(const Config *conf) {
+ BaseString newfile;
+ newfile.appfmt("%s.new", m_configFilename.c_str());
+
+ /* Open and write to the new config file */
+ FILE *f = fopen(newfile.c_str(), "w");
+ if(f == NULL) {
+ /** @todo Send something apropriate to the log */
+ return -1;
+ }
+ FileOutputStream stream(f);
+ conf->printConfigFile(stream);
+
+ fclose(f);
+
+ /* Rename file to real name */
+ rename(newfile.c_str(), m_configFilename.c_str());
+
+ return 0;
+}
+
+Config *
+MgmtSrvr::readConfig() {
+ Config *conf = NULL;
+ if(m_configFilename.length() != 0) {
+ /* Use config file */
+ InitConfigFileParser parser(m_configFilename.c_str());
+
+ if(!parser.readConfigFile()) {
+ /* Try to get configuration from other MGM server */
+ ConfigRetriever cr;
+ cr.setLocalConfigFileName(m_localNdbConfigFilename.c_str());
+ conf = new Config(*cr.getConfig("MGM", NDB_VERSION));
+ } else {
+ conf = new Config(*parser.getConfig());
+ }
+
+ if(conf == NULL)
+ return NULL;
+ }
+ return conf;
+}
+
+Config *
+MgmtSrvr::fetchConfig() {
+ Config *conf = NULL;
+ ConfigRetriever cr;
+ cr.setLocalConfigFileName(m_localNdbConfigFilename.c_str());
+ conf = new Config(*cr.getConfig("MGM", NDB_VERSION));
+
+ return conf;
+}
+
+bool
+MgmtSrvr::changeConfig(const BaseString &section,
+ const BaseString &param,
+ const BaseString &value) {
+ if(m_newConfig == NULL)
+ return false;
+ return m_newConfig->change(section, param, value);
+}
diff --git a/ndb/src/mgmsrv/MgmtSrvrGeneralSignalHandling.cpp b/ndb/src/mgmsrv/MgmtSrvrGeneralSignalHandling.cpp
new file mode 100644
index 00000000000..2126c9d358d
--- /dev/null
+++ b/ndb/src/mgmsrv/MgmtSrvrGeneralSignalHandling.cpp
@@ -0,0 +1,140 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//******************************************************************************
+// General signal handling methods
+// All implementations stolen from the Ndb class.
+// Some kind of reuse should be preferred.
+//******************************************************************************
+
+#include "MgmtSrvr.hpp"
+#include <NdbApiSignal.hpp>
+#include <NdbTick.h>
+
+
+NdbApiSignal*
+MgmtSrvr::getSignal()
+{
+ NdbApiSignal* tSignal;
+ tSignal = theSignalIdleList;
+ if (tSignal != NULL){
+ NdbApiSignal* tSignalNext = tSignal->next();
+ tSignal->next(NULL);
+ theSignalIdleList = tSignalNext;
+ return tSignal;
+ } else
+ {
+ tSignal = new NdbApiSignal(_ownReference);
+ if (tSignal != NULL)
+ tSignal->next(NULL);
+ }
+ return tSignal;
+}
+
+
+void
+MgmtSrvr::releaseSignal(NdbApiSignal* aSignal)
+{
+ aSignal->next(theSignalIdleList);
+ theSignalIdleList = aSignal;
+}
+
+
+void
+MgmtSrvr::freeSignal()
+{
+ NdbApiSignal* tSignal = theSignalIdleList;
+ theSignalIdleList = tSignal->next();
+ delete tSignal;
+}
+
+
+int
+MgmtSrvr::sendSignal(Uint16 aNodeId,
+ WaitSignalType aWaitState,
+ NdbApiSignal* aSignal,
+ bool force)
+{
+ int tReturnCode;
+ theFacade->lock_mutex();
+ if(force){
+ tReturnCode = theFacade->sendSignalUnCond(aSignal,
+ aNodeId);
+ } else {
+ tReturnCode = theFacade->sendSignal(aSignal,
+ aNodeId);
+ }
+ releaseSignal(aSignal);
+ if (tReturnCode == -1) {
+ theFacade->unlock_mutex();
+ return -1;
+ }
+ theWaitState = aWaitState;
+ theFacade->unlock_mutex();
+ return 0;
+}
+
+
+int
+MgmtSrvr::sendRecSignal(Uint16 aNodeId,
+ WaitSignalType aWaitState,
+ NdbApiSignal* aSignal,
+ bool force,
+ int waitTime)
+{
+ int tReturnCode;
+ theFacade->lock_mutex();
+ if(force){
+ tReturnCode = theFacade->sendSignalUnCond(aSignal, aNodeId);
+ } else {
+ tReturnCode = theFacade->sendSignalUnCond(aSignal, aNodeId);
+ }
+ releaseSignal(aSignal);
+ if (tReturnCode == -1) {
+ theFacade->unlock_mutex();
+ return -1;
+ }
+ theWaitState = aWaitState;
+ return receiveOptimisedResponse(waitTime);
+}
+
+
+int
+MgmtSrvr::receiveOptimisedResponse(int waitTime)
+{
+ int tResultCode;
+ theFacade->checkForceSend(_blockNumber);
+ NDB_TICKS maxTime = NdbTick_CurrentMillisecond() + waitTime;
+
+ while (theWaitState != NO_WAIT && waitTime > 0) {
+ NdbCondition_WaitTimeout(theMgmtWaitForResponseCondPtr,
+ theFacade->theMutexPtr,
+ waitTime);
+ if(theWaitState == NO_WAIT)
+ break;
+ waitTime = (maxTime - NdbTick_CurrentMillisecond());
+ }//while
+
+ if(theWaitState == NO_WAIT) {
+ tResultCode = 0;
+ } else {
+ tResultCode = -1;
+ }
+ theFacade->unlock_mutex();
+ return tResultCode;
+}
+
+
diff --git a/ndb/src/mgmsrv/NodeLogLevel.cpp b/ndb/src/mgmsrv/NodeLogLevel.cpp
new file mode 100644
index 00000000000..67791ca02bf
--- /dev/null
+++ b/ndb/src/mgmsrv/NodeLogLevel.cpp
@@ -0,0 +1,68 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NodeLogLevel.hpp"
+// TODO_RONM: Clearly getCategory and getLevel is not correctly coded. Must be taken care of.
+
+NodeLogLevel::NodeLogLevel(int nodeId, const SetLogLevelOrd& ll)
+{
+ m_nodeId = nodeId;
+ m_logLevel = ll;
+}
+
+NodeLogLevel::~NodeLogLevel()
+{
+}
+
+int
+NodeLogLevel::getNodeId() const
+{
+ return m_nodeId;
+}
+
+Uint32
+NodeLogLevel::getCategory() const
+{
+ for (Uint32 i = 0; i < m_logLevel.noOfEntries; i++)
+ {
+ return m_logLevel.theCategories[i];
+ }
+}
+
+int
+NodeLogLevel::getLevel() const
+{
+ for (Uint32 i = 0; i < m_logLevel.noOfEntries; i++)
+ {
+ return m_logLevel.theLevels[i];
+ }
+}
+
+void
+NodeLogLevel::setLevel(int level)
+{
+ for (Uint32 i = 0; i < m_logLevel.noOfEntries; i++)
+ {
+ m_logLevel.theLevels[i] = level;
+ }
+
+}
+
+SetLogLevelOrd
+NodeLogLevel::getLogLevelOrd() const
+{
+ return m_logLevel;
+}
diff --git a/ndb/src/mgmsrv/NodeLogLevel.hpp b/ndb/src/mgmsrv/NodeLogLevel.hpp
new file mode 100644
index 00000000000..3e631e57901
--- /dev/null
+++ b/ndb/src/mgmsrv/NodeLogLevel.hpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODELOGLEVEL_H
+#define NODELOGLEVEL_H
+
+#include <portlib/NdbConstant.hpp>
+#include <signaldata/SetLogLevelOrd.hpp>
+
+/**
+ * Holds a DB node's log level settings for both local and event log levels.
+ * It only holds one log level setting even though SetLogLevelOrd can handle
+ * multiple log levels at once, it is not used in that way in the managment
+ * server.
+ *
+ * @version #@ $Id: NodeLogLevel.hpp,v 1.2 2003/07/05 17:40:22 elathal Exp $
+ */
+class NodeLogLevel
+{
+public:
+ NodeLogLevel(int nodeId, const SetLogLevelOrd& ll);
+ ~NodeLogLevel();
+
+ int getNodeId() const;
+ Uint32 getCategory() const;
+ int getLevel() const;
+ void setLevel(int level);
+ SetLogLevelOrd getLogLevelOrd() const;
+
+private:
+ NodeLogLevel();
+ NodeLogLevel(const NodeLogLevel&);
+ bool operator == (const NodeLogLevel&);
+ NodeLogLevel operator = (const NodeLogLevel&);
+
+ int m_nodeId;
+ SetLogLevelOrd m_logLevel;
+};
+
+#endif
diff --git a/ndb/src/mgmsrv/NodeLogLevelList.cpp b/ndb/src/mgmsrv/NodeLogLevelList.cpp
new file mode 100644
index 00000000000..7cf6dcc4b7e
--- /dev/null
+++ b/ndb/src/mgmsrv/NodeLogLevelList.cpp
@@ -0,0 +1,182 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NodeLogLevelList.hpp"
+
+#include "NodeLogLevel.hpp"
+#include "NdbStdio.h"
+
+//
+// PUBLIC
+//
+
+NodeLogLevelList::NodeLogLevelList() :
+ m_size(0),
+ m_pHeadNode(NULL),
+ m_pTailNode(NULL),
+ m_pCurrNode(NULL)
+{
+}
+
+NodeLogLevelList::~NodeLogLevelList()
+{
+ removeAll();
+}
+
+void
+NodeLogLevelList::add(NodeLogLevel* pNewNode)
+{
+ NodeLogLevelNode* pNode = new NodeLogLevelNode();
+
+ if (m_pHeadNode == NULL)
+ {
+ m_pHeadNode = pNode;
+ pNode->pPrev = NULL;
+ }
+ else
+ {
+ m_pTailNode->pNext = pNode;
+ pNode->pPrev = m_pTailNode;
+ }
+ m_pTailNode = pNode;
+ pNode->pNext = NULL;
+ pNode->pHandler = pNewNode;
+
+ m_size++;
+}
+
+bool
+NodeLogLevelList::remove(NodeLogLevel* pRemoveNode)
+{
+ NodeLogLevelNode* pNode = m_pHeadNode;
+ bool removed = false;
+ do
+ {
+ if (pNode->pHandler == pRemoveNode)
+ {
+ removeNode(pNode);
+ removed = true;
+ break;
+ }
+ } while ( (pNode = next(pNode)) != NULL);
+
+ return removed;
+}
+
+void
+NodeLogLevelList::removeAll()
+{
+ while (m_pHeadNode != NULL)
+ {
+ removeNode(m_pHeadNode);
+ }
+}
+
+NodeLogLevel*
+NodeLogLevelList::next()
+{
+ NodeLogLevel* pHandler = NULL;
+ if (m_pCurrNode == NULL)
+ {
+ m_pCurrNode = m_pHeadNode;
+ if (m_pCurrNode != NULL)
+ {
+ pHandler = m_pCurrNode->pHandler;
+ }
+ }
+ else
+ {
+ m_pCurrNode = next(m_pCurrNode); // Next node
+ if (m_pCurrNode != NULL)
+ {
+ pHandler = m_pCurrNode->pHandler;
+ }
+ }
+
+ return pHandler;
+}
+
+int
+NodeLogLevelList::size() const
+{
+ return m_size;
+}
+
+//
+// PRIVATE
+//
+
+NodeLogLevelList::NodeLogLevelNode*
+NodeLogLevelList::next(NodeLogLevelNode* pNode)
+{
+ NodeLogLevelNode* pCurr = pNode;
+ if (pNode->pNext != NULL)
+ {
+ pCurr = pNode->pNext;
+ }
+ else
+ {
+ // Tail
+ pCurr = NULL;
+ }
+ return pCurr;
+}
+
+NodeLogLevelList::NodeLogLevelNode*
+NodeLogLevelList::prev(NodeLogLevelNode* pNode)
+{
+ NodeLogLevelNode* pCurr = pNode;
+ if (pNode->pPrev != NULL) // head
+ {
+ pCurr = pNode->pPrev;
+ }
+ else
+ {
+ // Head
+ pCurr = NULL;
+ }
+
+ return pCurr;
+}
+
+void
+NodeLogLevelList::removeNode(NodeLogLevelNode* pNode)
+{
+ if (pNode->pPrev == NULL) // If head
+ {
+ m_pHeadNode = pNode->pNext;
+ }
+ else
+ {
+ pNode->pPrev->pNext = pNode->pNext;
+ }
+
+ if (pNode->pNext == NULL) // if tail
+ {
+ m_pTailNode = pNode->pPrev;
+ }
+ else
+ {
+ pNode->pNext->pPrev = pNode->pPrev;
+ }
+
+ pNode->pNext = NULL;
+ pNode->pPrev = NULL;
+ delete pNode->pHandler; // Delete log handler
+ delete pNode;
+
+ m_size--;
+}
diff --git a/ndb/src/mgmsrv/NodeLogLevelList.hpp b/ndb/src/mgmsrv/NodeLogLevelList.hpp
new file mode 100644
index 00000000000..4a55ee211e2
--- /dev/null
+++ b/ndb/src/mgmsrv/NodeLogLevelList.hpp
@@ -0,0 +1,93 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODELOGLEVELLIST_H
+#define NODELOGLEVELLIST_H
+
+class NodeLogLevel;
+
+/**
+ * Provides a simple linked list of NodeLogLevel.
+ *
+ * @see NodeLogLevel
+ * @version #@ $Id: NodeLogLevelList.hpp,v 1.1 2002/08/09 12:53:50 eyualex Exp $
+ */
+class NodeLogLevelList
+{
+public:
+ /**
+ * Default Constructor.
+ */
+ NodeLogLevelList();
+
+ /**
+ * Destructor.
+ */
+ ~NodeLogLevelList();
+
+ /**
+ * Adds a new node.
+ *
+ * @param pNewHandler a new NodeLogLevel.
+ */
+ void add(NodeLogLevel* pNewNode);
+
+ /**
+ * Removes a NodeLogLevel from the list and call its destructor.
+ *
+ * @param pRemoveHandler the NodeLogLevel to remove
+ */
+ bool remove(NodeLogLevel* pRemoveNode);
+
+ /**
+ * Removes all items.
+ */
+ void removeAll();
+
+ /**
+ * Returns the next node in the list.
+ * returns a node or NULL.
+ */
+ NodeLogLevel* next();
+
+ /**
+ * Returns the size of the list.
+ */
+ int size() const;
+private:
+ /** List node */
+ struct NodeLogLevelNode
+ {
+ NodeLogLevelNode* pPrev;
+ NodeLogLevelNode* pNext;
+ NodeLogLevel* pHandler;
+ };
+
+ NodeLogLevelNode* next(NodeLogLevelNode* pNode);
+ NodeLogLevelNode* prev(NodeLogLevelNode* pNode);
+
+ void removeNode(NodeLogLevelNode* pNode);
+
+ int m_size;
+
+ NodeLogLevelNode* m_pHeadNode;
+ NodeLogLevelNode* m_pTailNode;
+ NodeLogLevelNode* m_pCurrNode;
+};
+
+#endif
+
+
diff --git a/ndb/src/mgmsrv/Services.cpp b/ndb/src/mgmsrv/Services.cpp
new file mode 100644
index 00000000000..24f41fe64bf
--- /dev/null
+++ b/ndb/src/mgmsrv/Services.cpp
@@ -0,0 +1,1082 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <string.h>
+#include <ctype.h>
+
+#include <uucode.h>
+#include <socket_io.h>
+#include <ndb_version.h>
+#include <mgmapi.h>
+#include <EventLogger.hpp>
+#include <signaldata/SetLogLevelOrd.hpp>
+#include <BaseString.hpp>
+#include <Base64.hpp>
+
+#include "Services.hpp"
+
+static const unsigned int MAX_READ_TIMEOUT = 1000 ;
+static const unsigned int MAX_WRITE_TIMEOUT = 100 ;
+
+/**
+ const char * name;
+ const char * realName;
+ const Type type;
+ const ArgType argType;
+ const ArgRequired argRequired;
+ const ArgMinMax argMinMax;
+ const int minVal;
+ const int maxVal;
+ void (T::* function)(const class Properties & args);
+ const char * description;
+*/
+
+#define MGM_CMD(name, fun, desc) \
+ { name, \
+ 0, \
+ ParserRow<MgmApiSession>::Cmd, \
+ ParserRow<MgmApiSession>::String, \
+ ParserRow<MgmApiSession>::Optional, \
+ ParserRow<MgmApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ fun, \
+ desc, 0 }
+
+#define MGM_ARG(name, type, opt, desc) \
+ { name, \
+ 0, \
+ ParserRow<MgmApiSession>::Arg, \
+ ParserRow<MgmApiSession>::type, \
+ ParserRow<MgmApiSession>::opt, \
+ ParserRow<MgmApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ desc, 0 }
+
+#define MGM_ARG2(name, type, opt, min, max, desc) \
+ { name, \
+ 0, \
+ ParserRow<MgmApiSession>::Arg, \
+ ParserRow<MgmApiSession>::type, \
+ ParserRow<MgmApiSession>::opt, \
+ ParserRow<MgmApiSession>::IgnoreMinMax, \
+ min, max, \
+ 0, \
+ desc, 0 }
+
+#define MGM_END() \
+ { 0, \
+ 0, \
+ ParserRow<MgmApiSession>::Arg, \
+ ParserRow<MgmApiSession>::Int, \
+ ParserRow<MgmApiSession>::Optional, \
+ ParserRow<MgmApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0, 0 }
+
+#define MGM_CMD_ALIAS(name, realName, fun) \
+ { name, \
+ realName, \
+ ParserRow<MgmApiSession>::CmdAlias, \
+ ParserRow<MgmApiSession>::Int, \
+ ParserRow<MgmApiSession>::Optional, \
+ ParserRow<MgmApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0, 0 }
+
+#define MGM_ARG_ALIAS(name, realName, fun) \
+ { name, \
+ realName, \
+ ParserRow<MgmApiSession>::ArgAlias, \
+ ParserRow<MgmApiSession>::Int, \
+ ParserRow<MgmApiSession>::Optional, \
+ ParserRow<MgmApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0, 0 }
+
+const
+ParserRow<MgmApiSession> commands[] = {
+ MGM_CMD("get statport", &MgmApiSession::getStatPort, ""),
+
+ MGM_CMD("get config", &MgmApiSession::getConfig, ""),
+ MGM_ARG("version", Int, Mandatory, "Configuration version number"),
+ MGM_ARG("node", Int, Optional, "Node ID"),
+
+ MGM_CMD("get version", &MgmApiSession::getVersion, ""),
+
+ MGM_CMD("get status", &MgmApiSession::getStatus, ""),
+
+ MGM_CMD("get info clusterlog", &MgmApiSession::getInfoClusterLog, ""),
+
+ MGM_CMD("restart node", &MgmApiSession::restart, ""),
+ MGM_ARG("node", String, Mandatory, "Nodes to restart"),
+ MGM_ARG("initialstart", Int, Optional, "Initial start"),
+ MGM_ARG("nostart", Int, Optional, "No start"),
+ MGM_ARG("abort", Int, Optional, "Abort"),
+
+ MGM_CMD("restart all", &MgmApiSession::restartAll, ""),
+ MGM_ARG("initialstart", Int, Optional, "Initial start"),
+ MGM_ARG("nostart", Int, Optional, "No start"),
+ MGM_ARG("abort", Int, Optional, "Abort"),
+
+ MGM_CMD("insert error", &MgmApiSession::insertError, ""),
+ MGM_ARG("node", Int, Mandatory, "Node to receive error"),
+ MGM_ARG("error", Int, Mandatory, "Errorcode to insert"),
+
+ MGM_CMD("set trace", &MgmApiSession::setTrace, ""),
+ MGM_ARG("node", Int, Mandatory, "Node"),
+ MGM_ARG("trace", Int, Mandatory, "Trace number"),
+
+ MGM_CMD("log signals", &MgmApiSession::logSignals, ""),
+ MGM_ARG("node", Int, Mandatory, "Node"),
+ MGM_ARG("blocks", String, Mandatory, "Blocks (space separated)"),
+ MGM_ARG("in", Int, Mandatory, "Log input signals"),
+ MGM_ARG("out", Int, Mandatory, "Log output signals"),
+
+ MGM_CMD("start signallog", &MgmApiSession::startSignalLog, ""),
+ MGM_ARG("node", Int, Mandatory, "Node"),
+
+ MGM_CMD("stop signallog", &MgmApiSession::stopSignalLog, ""),
+ MGM_ARG("node", Int, Mandatory, "Node"),
+
+ MGM_CMD("dump state", &MgmApiSession::dumpState, ""),
+ MGM_ARG("node", Int, Mandatory ,"Node"),
+ MGM_ARG("args", String, Mandatory, "Args(space separated int's)"),
+
+ MGM_CMD("start backup", &MgmApiSession::startBackup, ""),
+
+ MGM_CMD("abort backup", &MgmApiSession::abortBackup, ""),
+ MGM_ARG("id", Int, Mandatory, "Backup id"),
+
+ /**
+ * Global Replication
+ */
+ MGM_CMD("rep", &MgmApiSession::repCommand, ""),
+ MGM_ARG("request", Int, Mandatory, "Command"),
+
+ MGM_CMD("stop", &MgmApiSession::stop, ""),
+ MGM_ARG("node", String, Mandatory, "Node"),
+ MGM_ARG("abort", Int, Mandatory, "Node"),
+
+ MGM_CMD("stop all", &MgmApiSession::stopAll, ""),
+ MGM_ARG("abort", Int, Mandatory, "Node"),
+
+ MGM_CMD("enter single user", &MgmApiSession::enterSingleUser, ""),
+ MGM_ARG("nodeId", Int, Mandatory, "Node"),
+
+ MGM_CMD("exit single user", &MgmApiSession::exitSingleUser, ""),
+
+
+ MGM_CMD("start", &MgmApiSession::start, ""),
+ MGM_ARG("node", Int, Mandatory, "Node"),
+
+ MGM_CMD("start all", &MgmApiSession::startAll, ""),
+
+ MGM_CMD("bye", &MgmApiSession::bye, ""),
+
+ MGM_CMD("set loglevel", &MgmApiSession::setLogLevel, ""),
+ MGM_ARG("node", Int, Mandatory, "Node"),
+ MGM_ARG("category", String, Mandatory, "Event category"),
+ MGM_ARG("level", Int, Mandatory, "Log level (0-15)"),
+
+ MGM_CMD("set cluster loglevel", &MgmApiSession::setClusterLogLevel, ""),
+ MGM_ARG("node", Int, Mandatory, "Node"),
+ MGM_ARG("category", String, Mandatory, "Event category"),
+ MGM_ARG("level", Int, Mandatory, "Log level (0-15)"),
+
+ MGM_CMD("set logfilter", &MgmApiSession::setLogFilter, ""),
+ MGM_ARG("level", Int, Mandatory, "Severety level"),
+
+ MGM_CMD("config lock", &MgmApiSession::configLock, ""),
+
+ MGM_CMD("config unlock", &MgmApiSession::configUnlock, ""),
+ MGM_ARG("commit", Int, Mandatory, "Commit changes"),
+
+ MGM_CMD("config change", &MgmApiSession::configChange, ""),
+ MGM_ARG("section", String, Mandatory, "Section"),
+ MGM_ARG("parameter", String, Mandatory, "Parameter"),
+ MGM_ARG("value", String, Mandatory, "Value"),
+
+ MGM_END()
+};
+
+MgmApiSession::MgmApiSession(class MgmtSrvr & mgm, NDB_SOCKET_TYPE sock)
+ : SocketServer::Session(sock), m_mgmsrv(mgm) {
+ m_input = new SocketInputStream(sock);
+ m_output = new SocketOutputStream(sock);
+ m_parser = new Parser_t(commands, *m_input, true, true, true);
+}
+
+void
+MgmApiSession::runSession() {
+ Parser_t::Context ctx;
+ while(!m_stop) {
+ m_parser->run(ctx, *this);
+
+ if(ctx.m_currentToken == 0)
+ break;
+
+ switch(ctx.m_status) {
+ case Parser_t::UnknownCommand:
+#ifdef MGM_GET_CONFIG_BACKWARDS_COMPAT
+ /* Backwards compatibility for old NDBs that still use
+ * the old "GET CONFIG" command.
+ */
+
+ for(size_t i=0; i<strlen(ctx.m_currentToken); i++)
+ ctx.m_currentToken[i] = toupper(ctx.m_currentToken[i]);
+
+ if(strncmp("GET CONFIG ",
+ ctx.m_currentToken,
+ strlen("GET CONFIG ")) == 0)
+ getConfig_old(ctx);
+#endif /* MGM_GET_CONFIG_BACKWARDS_COMPAT */
+ break;
+ default:
+ break;
+ }
+ }
+ NDB_CLOSE_SOCKET(m_socket);
+}
+
+#ifdef MGM_GET_CONFIG_BACKWARDS_COMPAT
+void
+MgmApiSession::getConfig_old(Parser_t::Context &ctx) {
+ Properties args;
+
+ Uint32 version, node;
+
+ if(sscanf(ctx.m_currentToken, "GET CONFIG %d %d",
+ (int *)&version, (int *)&node) != 2) {
+ m_output->println("Expected 2 arguments for GET CONFIG");
+ return;
+ }
+
+ /* Put arguments in properties object so we can call the real function */
+ args.put("version", version);
+ args.put("node", node);
+ getConfig_common(ctx, args, true);
+}
+#endif /* MGM_GET_CONFIG_BACKWARDS_COMPAT */
+
+void
+MgmApiSession::getConfig(Parser_t::Context &ctx,
+ const class Properties &args) {
+ getConfig_common(ctx, args);
+}
+
+void
+MgmApiSession::getConfig_common(Parser_t::Context &,
+ const class Properties &args,
+ bool compat) {
+ Uint32 version, node = 0;
+
+ args.get("version", &version);
+ args.get("node", &node);
+
+#if 0
+ if(version != 0) {
+ m_output->println("get config");
+ m_output->println("result: Invalid version number");
+ m_output->println("");
+ return;
+ }
+#endif
+
+ const Config *conf = m_mgmsrv.getConfig();
+ if(conf == NULL) {
+ m_output->println("get config");
+ m_output->println("result: Could not fetch configuration");
+ m_output->println("");
+ return;
+ }
+
+ bool compatible;
+ switch (m_mgmsrv.getNodeType(node)) {
+ case NDB_MGM_NODE_TYPE_NDB:
+ compatible = ndbCompatible_mgmt_ndb(NDB_VERSION, version);
+ break;
+ case NDB_MGM_NODE_TYPE_API:
+ case NDB_MGM_NODE_TYPE_MGM:
+ compatible = ndbCompatible_mgmt_api(NDB_VERSION, version);
+ break;
+ default:
+ m_output->println("get config");
+ m_output->println("result: unrecognignized node type");
+ m_output->println("");
+ return;
+ }
+
+ if (!compatible){
+ m_output->println("get config");
+ m_output->println("result: incompatible version mgmt 0x%x and node 0x%x",
+ NDB_VERSION, version);
+ m_output->println("");
+ return;
+ }
+
+ Properties *reply = new Properties(*conf);
+ reply->put("Version", NDB_VERSION); // reply->put("Version", version);
+ reply->put("LocalNodeId", node);
+
+#ifdef MGM_GET_CONFIG_BACKWARDS_COMPAT
+ if(compat) {
+ const Uint32 size = reply->getPackedSize();
+ Uint32 *buffer = new Uint32[size/4+1];
+
+ reply->pack(buffer);
+ delete reply;
+
+ const int uurows = (size + 44)/45;
+ char * uubuf = new char[uurows * 62+5];
+
+ const int uusz = uuencode_mem(uubuf, (char *)buffer, size);
+ delete[] buffer;
+
+ m_output->println("GET CONFIG %d %d %d %d %d",
+ 0, NDB_VERSION, node, size, uusz);// 0, version, node, size, uusz);
+
+ m_output->println("begin 664 Ndb_cfg.bin");
+
+ /* XXX Need to write directly to the socket, because the uubuf is not
+ * NUL-terminated. This could/should probably be done in a nicer way.
+ */
+ write_socket(m_socket, MAX_WRITE_TIMEOUT, uubuf, uusz);
+ delete[] uubuf;
+
+ m_output->println("end");
+ m_output->println("");
+ } else {
+#endif /* MGM_GET_CONFIG_BACKWARDS_COMPAT */
+
+ UtilBuffer buffer;
+ BaseString str;
+ reply->pack(buffer);
+ delete reply;
+ base64_encode(buffer, str);
+
+ m_output->println("config: %s", str.c_str());
+ m_output->println("");
+#ifdef MGM_GET_CONFIG_BACKWARDS_COMPAT
+ }
+#endif /* MGM_GET_CONFIG_BACKWARDS_COMPAT */
+}
+
+void
+MgmApiSession::getStatPort(Parser_t::Context &,
+ const class Properties &) {
+
+ m_output->println("get statport reply");
+ m_output->println("tcpport: %d", m_mgmsrv.getStatPort());
+ m_output->println("");
+}
+
+void
+MgmApiSession::insertError(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 node = 0, error = 0;
+
+ args.get("node", &node);
+ args.get("error", &error);
+
+ int result = m_mgmsrv.insertError(node, error);
+
+ m_output->println("insert error reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+}
+
+void
+MgmApiSession::setTrace(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 node = 0, trace = 0;
+
+ args.get("node", &node);
+ args.get("trace", &trace);
+
+ int result = m_mgmsrv.setTraceNo(node, trace);
+
+ m_output->println("set trace reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+}
+
+void
+MgmApiSession::getVersion(Parser<MgmApiSession>::Context &,
+ Properties const &) {
+ m_output->println("version");
+ m_output->println("id: %d", NDB_VERSION);
+ m_output->println("major: %d", getMajor(NDB_VERSION));
+ m_output->println("minor: %d", getMinor(NDB_VERSION));
+ m_output->println("string: %s", NDB_VERSION_STRING);
+ m_output->println("");
+}
+#if 0
+
+/*****************************************************************************
+ * BACKUP
+ *****************************************************************************/
+
+int completed;
+MgmtSrvr::BackupEvent globalEvent;
+
+static void
+completedCallback(const MgmtSrvr::BackupEvent & event){
+
+ ndbout << "WaitCallback" << endl;
+ // Save event in the latestEvent var
+
+ switch(event.Event){
+ case MgmtSrvr::BackupEvent::BackupCompleted:
+ case MgmtSrvr::BackupEvent::BackupFailedToStart:
+ globalEvent = event;
+ completed = 1;
+ break;
+ }
+}
+
+void
+MgmApiSession::startBackup(Parser<MgmApiSession>::Context &,
+ Properties const &) {
+ unsigned backupId;
+ int result;
+
+ MgmtSrvr::BackupCallback prevCallback;
+ prevCallback = m_mgmsrv.setCallback(completedCallback);
+ completed = 0;
+ result = m_mgmsrv.startBackup(backupId);
+ if (result == 0){
+
+ // Wait for the callback to call our condition
+ // waitFor();
+ while (completed == 0)
+ NdbSleep_SecSleep(0);
+
+ if (globalEvent.Event == MgmtSrvr::BackupEvent::BackupFailedToStart)
+ result = globalEvent.FailedToStart.ErrorCode;
+ else
+ backupId = globalEvent.Completed.BackupId;
+ }
+
+ // restore old callback
+ m_mgmsrv.setCallback(prevCallback);
+
+ m_output->println("start backup reply");
+ if(result != 0)
+ m_output->println("result: %s(%d)", m_mgmsrv.getErrorText(result), result);
+ else{
+ m_output->println("result: Ok");
+ m_output->println("id: %d", backupId);
+ }
+ m_output->println("");
+
+}
+#endif
+
+void
+MgmApiSession::startBackup(Parser<MgmApiSession>::Context &,
+ Properties const &) {
+ unsigned backupId;
+ int result;
+
+ result = m_mgmsrv.startBackup(backupId, true);
+
+ m_output->println("start backup reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else{
+ m_output->println("result: Ok");
+ m_output->println("id: %d", backupId);
+ }
+ m_output->println("");
+
+}
+
+void
+MgmApiSession::abortBackup(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 id = 0;
+
+ args.get("id", &id);
+
+ int result = m_mgmsrv.abortBackup(id);
+
+ m_output->println("abort backup reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+}
+
+/*****************************************************************************
+ * Global Replication
+ *****************************************************************************/
+
+void
+MgmApiSession::repCommand(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+
+ Uint32 request = 0;
+ args.get("request", &request);
+
+ Uint32 repReqId;
+ int result = m_mgmsrv.repCommand(&repReqId, request, true);
+
+ m_output->println("global replication reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else{
+ m_output->println("result: Ok");
+ m_output->println("id: %d", repReqId);
+ }
+ m_output->println("");
+}
+
+/*****************************************************************************/
+
+void
+MgmApiSession::dumpState(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 node;
+ BaseString args_str;
+
+ args.get("node", &node);
+ args.get("args", args_str);
+
+ int result = m_mgmsrv.dumpState(node, args_str.c_str());
+ m_output->println("dump state reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+}
+
+
+void
+MgmApiSession::bye(Parser<MgmApiSession>::Context &,
+ Properties const &) {
+ m_stop = true;
+}
+
+void
+MgmApiSession::setClusterLogLevel(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 node, level;
+ BaseString categoryName, errorString;
+ SetLogLevelOrd logLevel;
+ int result;
+ logLevel.clear();
+ args.get("node", &node);
+ args.get("category", categoryName);
+ args.get("level", &level);
+
+ /* XXX should use constants for this value */
+ if(level > 15) {
+ errorString.assign("Invalied loglevel");
+ goto error;
+ }
+
+ categoryName.ndb_toupper();
+
+ LogLevel::EventCategory category;
+ if(!EventLogger::matchEventCategory(categoryName.c_str(), &category)) {
+ errorString.assign("Unknown category");
+ goto error;
+ }
+
+ logLevel.setLogLevel(category, level);
+ result = m_mgmsrv.setEventReportingLevel(node, logLevel);
+
+ m_output->println("set cluster loglevel reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+ return;
+ error:
+ m_output->println("set cluster loglevel reply");
+ m_output->println("result: %s", errorString.c_str());
+ m_output->println("");
+}
+
+void
+MgmApiSession::setLogLevel(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 node = 0, level = 0;
+ BaseString categoryName, errorString;
+ SetLogLevelOrd logLevel;
+ int result;
+ logLevel.clear();
+ args.get("node", &node);
+ args.get("category", categoryName);
+ args.get("level", &level);
+
+ /* XXX should use constants for this value */
+ if(level > 15) {
+ errorString.assign("Invalied loglevel");
+ goto error;
+ }
+
+ categoryName.ndb_toupper();
+
+ LogLevel::EventCategory category;
+ if(!EventLogger::matchEventCategory(categoryName.c_str(), &category)) {
+ errorString.assign("Unknown category");
+ goto error;
+ }
+
+ logLevel.setLogLevel(category, level);
+
+ result = m_mgmsrv.setNodeLogLevel(node, logLevel);
+
+ m_output->println("set loglevel reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+ return;
+ error:
+ m_output->println("set loglevel reply");
+ m_output->println("result: %s", errorString.c_str());
+ m_output->println("");
+}
+
+void
+MgmApiSession::stopSignalLog(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 node;
+
+ args.get("node", &node);
+
+ int result = m_mgmsrv.stopSignalTracing(node);
+
+ m_output->println("stop signallog");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+}
+
+void
+MgmApiSession::restart(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32
+ nostart = 0,
+ initialstart = 0,
+ abort = 0;
+ char *nodes_str;
+ Vector<NodeId> nodes;
+
+ args.get("initialstart", &initialstart);
+ args.get("nostart", &nostart);
+ args.get("abort", &abort);
+ args.get("node", (const char **)&nodes_str);
+
+ char *p, *last;
+ for((p = strtok_r(nodes_str, " ", &last));
+ p;
+ (p = strtok_r(NULL, " ", &last))) {
+ nodes.push_back(atoi(p));
+ }
+
+ int restarted = 0;
+ int result = 0;
+
+ for(size_t i = 0; i < nodes.size(); i++)
+ if((result = m_mgmsrv.restartNode(nodes[i],
+ nostart != 0,
+ initialstart != 0,
+ abort != 0)) == 0)
+ restarted++;
+
+ m_output->println("restart reply");
+ if(result != 0){
+ m_output->println("result: %d-%s", result, m_mgmsrv.getErrorText(result));
+ } else
+ m_output->println("result: Ok");
+ m_output->println("restarted: %d", restarted);
+ m_output->println("");
+}
+
+void
+MgmApiSession::restartAll(Parser<MgmApiSession>::Context &,
+ Properties const &args)
+{
+ Uint32 nostart = 0;
+ Uint32 initialstart = 0;
+ Uint32 abort = 0;
+
+ args.get("initialstart", &initialstart);
+ args.get("abort", &abort);
+ args.get("nostart", &nostart);
+
+ int count = 0;
+ int result = m_mgmsrv.restart(nostart, initialstart, abort, &count);
+
+ m_output->println("restart reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("restarted: %d", count);
+ m_output->println("");
+}
+
+static void
+printNodeStatus(OutputStream *output,
+ MgmtSrvr &mgmsrv,
+ enum ndb_mgm_node_type type) {
+ NodeId nodeId = 0;
+ while(mgmsrv.getNextNodeId(&nodeId, type)) {
+ enum ndb_mgm_node_status status;
+ Uint32 startPhase = 0, version = 0, dynamicId = 0, nodeGroup = 0;
+ bool system;
+ mgmsrv.status(nodeId, &status, &version, &startPhase,
+ &system, &dynamicId, &nodeGroup);
+ output->println("node.%d.type: %s",
+ nodeId,
+ ndb_mgm_get_node_type_string(type));
+ output->println("node.%d.status: %s",
+ nodeId,
+ ndb_mgm_get_node_status_string(status));
+ output->println("node.%d.version: %d", nodeId, version);
+ output->println("node.%d.startphase: %d", nodeId, startPhase);
+ output->println("node.%d.dynamic_id: %d", nodeId, dynamicId);
+ output->println("node.%d.node_group: %d", nodeId, nodeGroup);
+ }
+
+}
+
+void
+MgmApiSession::getStatus(Parser<MgmApiSession>::Context &,
+ Properties const &) {
+ int noOfNodes = 0;
+
+ NodeId nodeId = 0;
+ while(m_mgmsrv.getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_NDB)){
+ noOfNodes++;
+ }
+ nodeId = 0;
+ while(m_mgmsrv.getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_API)){
+ noOfNodes++;
+ }
+ nodeId = 0;
+ while(m_mgmsrv.getNextNodeId(&nodeId, NDB_MGM_NODE_TYPE_MGM)){
+ noOfNodes++;
+ }
+
+ m_output->println("node status");
+ m_output->println("nodes: %d", noOfNodes);
+ printNodeStatus(m_output, m_mgmsrv, NDB_MGM_NODE_TYPE_NDB);
+ printNodeStatus(m_output, m_mgmsrv, NDB_MGM_NODE_TYPE_MGM);
+ printNodeStatus(m_output, m_mgmsrv, NDB_MGM_NODE_TYPE_API);
+
+ nodeId = 0;
+
+ m_output->println("");
+}
+
+void
+MgmApiSession::getInfoClusterLog(Parser<MgmApiSession>::Context &,
+ Properties const &) {
+ const char* names[] = { "enabled",
+ "debug",
+ "info",
+ "warning",
+ "error",
+ "critical",
+ "alert" };
+
+ m_output->println("clusterlog");
+ for(int i = 0; i < 7; i++) {
+ m_output->println("%s: %d",
+ names[i], m_mgmsrv.isEventLogFilterEnabled(i));
+ }
+ m_output->println("");
+}
+
+void
+MgmApiSession::stop(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 abort;
+ char *nodes_str;
+ Vector<NodeId> nodes;
+
+ args.get("node", (const char **)&nodes_str);
+ if(nodes_str == NULL)
+ return;
+ args.get("abort", &abort);
+
+ char *p, *last;
+ for((p = strtok_r(nodes_str, " ", &last));
+ p;
+ (p = strtok_r(NULL, " ", &last))) {
+ nodes.push_back(atoi(p));
+ }
+
+ int stopped = 0, result = 0;
+
+ for(size_t i=0; i < nodes.size(); i++)
+ if((result = m_mgmsrv.stopNode(nodes[i], abort != 0)) == 0)
+ stopped++;
+
+ m_output->println("stop reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("stopped: %d", stopped);
+ m_output->println("");
+}
+
+
+void
+MgmApiSession::stopAll(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ int stopped = 0;
+ Uint32 abort;
+ args.get("abort", &abort);
+
+ int result = m_mgmsrv.stop(&stopped, abort != 0);
+
+ m_output->println("stop reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("stopped: %d", stopped);
+ m_output->println("");
+}
+
+void
+MgmApiSession::enterSingleUser(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ int stopped = 0;
+ Uint32 nodeId = 0;
+ args.get("nodeId", &nodeId);
+ int result = m_mgmsrv.enterSingleUser(&stopped, nodeId);
+ m_output->println("enter single user reply");
+ if(result != 0) {
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ }
+ else {
+ m_output->println("result: Ok");
+ }
+ m_output->println("");
+}
+
+void
+MgmApiSession::exitSingleUser(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ int stopped = 0;
+ int result = m_mgmsrv.exitSingleUser(&stopped, false);
+ m_output->println("exit single user reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+}
+
+
+void
+MgmApiSession::startSignalLog(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 node;
+
+ args.get("node", &node);
+
+ int result = m_mgmsrv.startSignalTracing(node);
+
+ m_output->println("start signallog reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+}
+
+void
+MgmApiSession::logSignals(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 node = 0, in = 0, out = 0;
+ // BaseString blocks;
+ BaseString blockList;
+ char * blockName;
+ args.get("node", &node);
+ args.get("in", &in);
+ args.get("out", &out);
+ args.get("blocks", blockList);
+ // fast fix - pekka
+ char buf[200];
+ snprintf(buf, 200, "%s", blockList.c_str());
+ Vector<BaseString> blocks;
+
+ blockName=strtok(buf,"|");
+ while( blockName != NULL)
+ {
+ blocks.push_back(blockName);
+ blockName=strtok(NULL,"|");
+ }
+
+
+ if(in > 1 || out > 1)
+ return; /* Invalid arguments */
+
+ const MgmtSrvr::LogMode modes[] = {
+ MgmtSrvr::Off,
+ MgmtSrvr::Out,
+ MgmtSrvr::In,
+ MgmtSrvr::InOut,
+ };
+ MgmtSrvr::LogMode mode = modes[in<<1 | out];
+
+ int result = m_mgmsrv.setSignalLoggingMode(node, mode, blocks);
+
+ m_output->println("log signals reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+}
+
+void
+MgmApiSession::start(Parser<MgmApiSession>::Context &,
+ Properties const &args) {
+ Uint32 node;
+
+ args.get("node", &node);
+
+ int result = m_mgmsrv.start(node);
+
+ m_output->println("start reply");
+ if(result != 0)
+ m_output->println("result: %s", m_mgmsrv.getErrorText(result));
+ else
+ m_output->println("result: Ok");
+ m_output->println("");
+}
+
+void
+MgmApiSession::startAll(Parser<MgmApiSession>::Context &,
+ Properties const &) {
+ NodeId node = 0;
+ int started = 0;
+
+ while(m_mgmsrv.getNextNodeId(&node, NDB_MGM_NODE_TYPE_NDB))
+ if(m_mgmsrv.start(node) == 0)
+ started++;
+
+ m_output->println("start reply");
+ m_output->println("result: Ok");
+ m_output->println("started: %d", started);
+ m_output->println("");
+}
+
+void
+MgmApiSession::setLogFilter(Parser_t::Context &ctx,
+ const class Properties &args) {
+ Uint32 level;
+
+ args.get("level", &level);
+
+ int result = m_mgmsrv.setEventLogFilter(level);
+
+ m_output->println("set logfilter reply");
+ m_output->println("result: %d", result);
+ m_output->println("");
+}
+
+void
+MgmApiSession::configLock(Parser_t::Context &,
+ Properties const &) {
+ int ret = m_mgmsrv.lockConf();
+ m_output->println("config lock reply");
+ m_output->println("result: %d", ret);
+ m_output->println("");
+}
+
+void
+MgmApiSession::configUnlock(Parser_t::Context &,
+ Properties const &args) {
+ Uint32 commit;
+ args.get("commit", &commit);
+ int ret = m_mgmsrv.unlockConf(commit == 1);
+ m_output->println("config unlock reply");
+ m_output->println("result: %d", ret);
+ m_output->println("");
+}
+
+void
+MgmApiSession::configChange(Parser_t::Context &,
+ Properties const &args) {
+ BaseString section, param, value;
+ args.get("section", section);
+ args.get("parameter", param);
+ args.get("value", value);
+
+ int ret = m_mgmsrv.changeConfig(section.c_str(),
+ param.c_str(),
+ value.c_str());
+ m_output->println("config change reply");
+ m_output->println("result: %d", ret);
+ m_output->println("");
+}
+
+void
+MgmStatService::println_statistics(const BaseString &line){
+ MutexVector<NDB_SOCKET_TYPE> copy(m_sockets.size());
+ m_sockets.lock();
+ for(int i = m_sockets.size() - 1; i >= 0; i--){
+ if(println_socket(m_sockets[i], MAX_WRITE_TIMEOUT, line.c_str()) == -1){
+ copy.push_back(m_sockets[i]);
+ m_sockets.erase(i, false);
+ }
+ }
+ m_sockets.unlock();
+
+ for(int i = copy.size() - 1; i >= 0; i--){
+ NDB_CLOSE_SOCKET(copy[i]);
+ copy.erase(i);
+ }
+ if(m_sockets.size() == 0 || false){
+ m_mgmsrv->startStatisticEventReporting(0);
+ }
+}
+
+void
+MgmStatService::stopSessions(){
+ for(int i = m_sockets.size() - 1; i >= 0; i--){
+ NDB_CLOSE_SOCKET(m_sockets[i]);
+ m_sockets.erase(i);
+ }
+
+}
diff --git a/ndb/src/mgmsrv/Services.hpp b/ndb/src/mgmsrv/Services.hpp
new file mode 100644
index 00000000000..3690f1a5a93
--- /dev/null
+++ b/ndb/src/mgmsrv/Services.hpp
@@ -0,0 +1,125 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MGMAPI_SERVICE_HPP
+#define MGMAPI_SERVICE_HPP
+
+#include <SocketServer.hpp>
+#include <NdbSleep.h>
+#include <Parser.hpp>
+#include <OutputStream.hpp>
+#include <InputStream.hpp>
+
+#include "MgmtSrvr.hpp"
+
+/** Undefine this to remove backwards compatibility for "GET CONFIG". */
+#define MGM_GET_CONFIG_BACKWARDS_COMPAT
+
+class MgmApiSession : public SocketServer::Session {
+private:
+ typedef Parser<MgmApiSession> Parser_t;
+
+ class MgmtSrvr & m_mgmsrv;
+ InputStream *m_input;
+ OutputStream *m_output;
+ Parser_t *m_parser;
+
+ void getConfig_common(Parser_t::Context &ctx,
+ const class Properties &args,
+ bool compat = false);
+
+public:
+ MgmApiSession(class MgmtSrvr & mgm, NDB_SOCKET_TYPE sock);
+ void runSession();
+
+ void getStatPort(Parser_t::Context &ctx, const class Properties &args);
+ void getConfig(Parser_t::Context &ctx, const class Properties &args);
+#ifdef MGM_GET_CONFIG_BACKWARDS_COMPAT
+ void getConfig_old(Parser_t::Context &ctx);
+#endif /* MGM_GET_CONFIG_BACKWARDS_COMPAT */
+
+ void getVersion(Parser_t::Context &ctx, const class Properties &args);
+ void getStatus(Parser_t::Context &ctx, const class Properties &args);
+ void getInfoClusterLog(Parser_t::Context &ctx, const class Properties &args);
+ void restart(Parser_t::Context &ctx, const class Properties &args);
+ void restartAll(Parser_t::Context &ctx, const class Properties &args);
+ void insertError(Parser_t::Context &ctx, const class Properties &args);
+ void setTrace(Parser_t::Context &ctx, const class Properties &args);
+ void logSignals(Parser_t::Context &ctx, const class Properties &args);
+ void startSignalLog(Parser_t::Context &ctx, const class Properties &args);
+ void stopSignalLog(Parser_t::Context &ctx, const class Properties &args);
+ void dumpState(Parser_t::Context &ctx, const class Properties &args);
+ void startBackup(Parser_t::Context &ctx, const class Properties &args);
+ void abortBackup(Parser_t::Context &ctx, const class Properties &args);
+ void enterSingleUser(Parser_t::Context &ctx, const class Properties &args);
+ void exitSingleUser(Parser_t::Context &ctx, const class Properties &args);
+ void stop(Parser_t::Context &ctx, const class Properties &args);
+ void stopAll(Parser_t::Context &ctx, const class Properties &args);
+ void start(Parser_t::Context &ctx, const class Properties &args);
+ void startAll(Parser_t::Context &ctx, const class Properties &args);
+ void bye(Parser_t::Context &ctx, const class Properties &args);
+ void setLogLevel(Parser_t::Context &ctx, const class Properties &args);
+ void setClusterLogLevel(Parser_t::Context &ctx,
+ const class Properties &args);
+ void setLogFilter(Parser_t::Context &ctx, const class Properties &args);
+ void configLock(Parser_t::Context &ctx, const class Properties &args);
+ void configUnlock(Parser_t::Context &ctx, const class Properties &args);
+ void configChange(Parser_t::Context &ctx, const class Properties &args);
+
+ void repCommand(Parser_t::Context &ctx, const class Properties &args);
+};
+
+class MgmApiService : public SocketServer::Service {
+ class MgmtSrvr * m_mgmsrv;
+public:
+ MgmApiService(){
+ m_mgmsrv = 0;
+ }
+
+ void setMgm(class MgmtSrvr * mgmsrv){
+ m_mgmsrv = mgmsrv;
+ }
+
+ MgmApiSession * newSession(NDB_SOCKET_TYPE socket){
+ return new MgmApiSession(* m_mgmsrv, socket);
+ }
+};
+
+class MgmStatService : public SocketServer::Service,
+ public MgmtSrvr::StatisticsListner
+{
+ class MgmtSrvr * m_mgmsrv;
+ MutexVector<NDB_SOCKET_TYPE> m_sockets;
+public:
+ MgmStatService() : m_sockets(5) {
+ m_mgmsrv = 0;
+ }
+
+ void setMgm(class MgmtSrvr * mgmsrv){
+ m_mgmsrv = mgmsrv;
+ }
+
+ SocketServer::Session * newSession(NDB_SOCKET_TYPE socket){
+ m_sockets.push_back(socket);
+ m_mgmsrv->startStatisticEventReporting(5);
+ return 0;
+ }
+
+ void stopSessions();
+
+ void println_statistics(const BaseString &line);
+};
+#endif
diff --git a/ndb/src/mgmsrv/SignalQueue.cpp b/ndb/src/mgmsrv/SignalQueue.cpp
new file mode 100644
index 00000000000..7003f5c0a89
--- /dev/null
+++ b/ndb/src/mgmsrv/SignalQueue.cpp
@@ -0,0 +1,105 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <string.h>
+
+#include "SignalQueue.hpp"
+
+SignalQueue::SignalQueue() {
+ m_mutex = NdbMutex_Create();
+ m_cond = NdbCondition_Create();
+ m_signalQueueHead = NULL;
+}
+
+SignalQueue::~SignalQueue() {
+ {
+ Guard g(m_mutex);
+ while(m_signalQueueHead != NULL)
+ delete pop();
+ }
+ NdbMutex_Destroy(m_mutex);
+ m_mutex = NULL;
+ NdbCondition_Destroy(m_cond);
+ m_cond = NULL;
+}
+
+NdbApiSignal *
+SignalQueue::pop() {
+ NdbApiSignal *ret;
+
+ if(m_signalQueueHead == NULL)
+ return NULL;
+
+ ret = m_signalQueueHead->signal;
+
+ QueueEntry *old = m_signalQueueHead;
+ m_signalQueueHead = m_signalQueueHead->next;
+
+ delete old;
+
+ return ret;
+}
+
+void
+SignalQueue::receive(void *me, NdbApiSignal *signal) {
+ SignalQueue *q = (SignalQueue *)me;
+ q->receive(signal);
+}
+
+void
+SignalQueue::receive(NdbApiSignal *signal) {
+ QueueEntry *n = new QueueEntry();
+ n->signal = signal;
+ n->next = NULL;
+
+ Guard guard(m_mutex);
+
+ if(m_signalQueueHead == NULL) {
+ m_signalQueueHead = n;
+ NdbCondition_Broadcast(m_cond);
+ return;
+ }
+
+ QueueEntry *cur = m_signalQueueHead;
+
+ while(cur->next != NULL)
+ cur = cur->next;
+
+ cur->next = n;
+
+ NdbCondition_Broadcast(m_cond);
+}
+
+NdbApiSignal *
+SignalQueue::waitFor(int gsn, NodeId nodeid, Uint32 timeout) {
+ Guard g(m_mutex);
+
+ if(m_signalQueueHead == NULL)
+ NdbCondition_WaitTimeout(m_cond, m_mutex, timeout);
+
+ if(m_signalQueueHead == NULL)
+ return NULL;
+
+ if(gsn != 0 &&
+ m_signalQueueHead->signal->readSignalNumber() != gsn)
+ return NULL;
+
+ if(nodeid != 0 &&
+ refToNode(m_signalQueueHead->signal->theSendersBlockRef) != nodeid)
+ return NULL;
+
+ return pop();
+}
diff --git a/ndb/src/mgmsrv/SignalQueue.hpp b/ndb/src/mgmsrv/SignalQueue.hpp
new file mode 100644
index 00000000000..76acaf2289a
--- /dev/null
+++ b/ndb/src/mgmsrv/SignalQueue.hpp
@@ -0,0 +1,100 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __SIGNALQUEUE_HPP_INCLUDED__
+#define __SIGNALQUEUE_HPP_INCLUDED__
+
+#include <NdbApiSignal.hpp>
+#include <NdbMutex.h>
+#include <NdbCondition.h>
+#include <Vector.hpp>
+
+/* XXX Look for an already existing definition */
+#define DEFAULT_TIMEOUT 5000
+
+class SignalQueue {
+public:
+ typedef void (* SignalHandler)(void *obj, int gsn, NdbApiSignal *signal);
+
+ SignalQueue();
+ ~SignalQueue();
+
+ /**
+ * Static wrapper making it possible to call receive without knowing the
+ * type of the receiver
+ */
+ static void receive(void *me, NdbApiSignal *signal);
+
+ /**
+ * Enqueues a signal, and notifies any thread waiting for signals.
+ */
+ void receive(NdbApiSignal *signal);
+
+ NdbApiSignal *waitFor(int gsn,
+ NodeId nodeid = 0,
+ Uint32 timeout = DEFAULT_TIMEOUT);
+ template<class T> bool waitFor(Vector<T> &t,
+ T *&handler,
+ NdbApiSignal *&signal,
+ Uint32 timeout = DEFAULT_TIMEOUT);
+private:
+ NdbMutex *m_mutex; /* Locks all data in SignalQueue */
+ NdbCondition *m_cond; /* Notifies about new signal in the queue */
+
+ /**
+ * Returns the last recently received signal. Must be called with
+ * m_mutex locked.
+ * The caller takes responsibility for deleting the returned object.
+ *
+ * @returns NULL if failed, or a received signal
+ */
+ NdbApiSignal *pop();
+
+ class QueueEntry {
+ public:
+ NdbApiSignal *signal;
+ QueueEntry *next;
+ };
+ QueueEntry *m_signalQueueHead; /** Head of the queue.
+ * New entries added on the tail
+ */
+};
+
+template<class T> bool
+SignalQueue::waitFor(Vector<T> &t,
+ T *&handler,
+ NdbApiSignal *&signal,
+ Uint32 timeout) {
+ Guard g(m_mutex);
+
+ if(m_signalQueueHead == NULL)
+ NdbCondition_WaitTimeout(m_cond, m_mutex, timeout);
+
+ if(m_signalQueueHead == NULL)
+ return false;
+
+ for(size_t i = 0; i < t.size(); i++) {
+ if(t[i].check(m_signalQueueHead->signal)) {
+ handler = &t[i];
+ signal = pop();
+ return true;
+ }
+ }
+
+ return false;
+}
+
+#endif /* !__SIGNALQUEUE_HPP_INCLUDED__ */
diff --git a/ndb/src/mgmsrv/convertStrToInt.cpp b/ndb/src/mgmsrv/convertStrToInt.cpp
new file mode 100644
index 00000000000..82bdb8e4f2f
--- /dev/null
+++ b/ndb/src/mgmsrv/convertStrToInt.cpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <errno.h>
+#include <stdlib.h>
+#include <string.h>
+
+bool convert(const char* s, int& val) {
+
+ if (s == NULL) {
+ return false;
+ }
+
+ if (strlen(s) == 0) {
+ return false;
+ }
+
+ errno = 0;
+ char* p;
+ long v = strtol(s, &p, 10);
+ if (errno != 0) {
+ return false;
+ }
+ if (p != &s[strlen(s)]) {
+ return false;
+ }
+
+ val = v;
+ return true;
+}
+
+
diff --git a/ndb/src/mgmsrv/convertStrToInt.hpp b/ndb/src/mgmsrv/convertStrToInt.hpp
new file mode 100644
index 00000000000..0b2a96ed0bf
--- /dev/null
+++ b/ndb/src/mgmsrv/convertStrToInt.hpp
@@ -0,0 +1,25 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//******************************************************************************
+// Description:
+//
+// Author: Peter Lind
+//******************************************************************************
+
+extern bool convert(const char* s, int& val);
+
+
diff --git a/ndb/src/mgmsrv/main.cpp b/ndb/src/mgmsrv/main.cpp
new file mode 100644
index 00000000000..d10ad8e0f4e
--- /dev/null
+++ b/ndb/src/mgmsrv/main.cpp
@@ -0,0 +1,429 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signal.h>
+#include <sys/ioctl.h>
+
+#include "MgmtSrvr.hpp"
+#include "EventLogger.hpp"
+#include "Config.hpp"
+#include "InitConfigFileParser.hpp"
+#include <SocketServer.hpp>
+#include "Services.hpp"
+#include <version.h>
+#include <kernel_types.h>
+#include <Properties.hpp>
+#include <NdbOut.hpp>
+#include <NdbMain.h>
+#include <NdbDaemon.h>
+#include <NdbConfig.h>
+#include <NdbHost.h>
+#include <ndb_version.h>
+#include <ConfigRetriever.hpp>
+#include <getarg.h>
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+#include <efs.h>
+#else
+#include "CommandInterpreter.hpp"
+#endif
+
+#define DEBUG(x) ndbout << x << endl;
+
+const char progname[] = "mgmtsrvr";
+
+
+/**
+ * @struct MgmGlobals
+ * @brief Global Variables used in the management server
+ ******************************************************************************/
+struct MgmGlobals {
+ MgmGlobals();
+ ~MgmGlobals();
+
+ /** Command line arguments */
+ int daemon; // NOT bool, bool need not be int
+ int non_interactive;
+ const char * config_filename;
+ const char * local_config_filename;
+
+ /** Stuff found in environment or in local config */
+ NodeId localNodeId;
+ bool use_specific_ip;
+ char * interface_name;
+ int port;
+ int port_stats;
+
+ /** The configuration of the cluster */
+ Config * cluster_config;
+
+ /** The Mgmt Server */
+ MgmtSrvr * mgmObject;
+
+ /** The Socket Server */
+ SocketServer * socketServer;
+};
+
+static MgmGlobals glob;
+
+
+/******************************************************************************
+ * Function prototypes
+ ******************************************************************************/
+static bool readLocalConfig();
+static bool readGlobalConfig();
+static bool setPortNo();
+
+/**
+ * Global variables
+ */
+bool g_StopServer;
+extern EventLogger g_EventLogger;
+
+extern int global_mgmt_server_check;
+int _print_version = 0;
+
+struct getargs args[] = {
+ { "version", 0, arg_flag, &_print_version,
+ "Print versions"},
+ { NULL, 'c', arg_string, &glob.config_filename,
+ "Running cluster configuration file", "filename" },
+ { NULL, 'd', arg_flag, &glob.daemon,
+ "Daemon mode" },
+ { NULL, 'l', arg_string, &glob.local_config_filename,
+ "Local configuration file (Ndb.cfg)",
+ "filename" },
+ { NULL, 'n', arg_flag, &glob.non_interactive,
+ "Don't run as daemon, but don't read from stdin", "non-interactive" }
+};
+
+int num_args = sizeof(args) / sizeof(args[0]);
+
+/*
+ * MAIN
+ */
+NDB_MAIN(mgmsrv){
+ /**
+ * OSE specific. Enable shared ownership of file system resources.
+ * This is needed in order to use the cluster log since the events
+ * from the cluster is written from the 'ndb_receive'(NDBAPI) thread/process.
+ */
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ efs_segment_share();
+#endif
+
+ global_mgmt_server_check = 1;
+
+ int optind = 0;
+ if(getarg(args, num_args, argc, argv, &optind)) {
+ arg_printusage(args, num_args, progname, "");
+ exit(1);
+ }
+
+ if (_print_version) {
+ ndbPrintVersion();
+ exit(0);
+ }
+
+ if(glob.config_filename == NULL) {
+ fprintf(stderr, "No configuration file specified\n");
+ exit(1);
+ }
+ glob.socketServer = new SocketServer();
+ MgmApiService * mapi = new MgmApiService();
+ MgmStatService * mstat = new MgmStatService();
+
+ /****************************
+ * Read configuration files *
+ ****************************/
+ if (!readLocalConfig())
+ goto error_end;
+ if (!readGlobalConfig())
+ goto error_end;
+
+ if (!setPortNo())
+ goto error_end;
+
+ if(!glob.use_specific_ip){
+ if(!glob.socketServer->tryBind(glob.port, glob.interface_name)){
+ ndbout_c("Unable to setup port: %s:%d!\n"
+ "Please check if the port is already used,\n"
+ "(perhaps a mgmtsrvr is already running),\n"
+ "and if you are executing on the correct computer",
+ glob.interface_name, glob.port);
+ goto error_end;
+ }
+ free(glob.interface_name);
+ glob.interface_name = 0;
+ }
+
+ if(!glob.socketServer->setup(mapi, glob.port, glob.interface_name)){
+ ndbout_c("Unable to setup management port: %d!\n"
+ "Please check if the port is already used,\n"
+ "(perhaps a mgmtsrvr is already running),\n"
+ "and if you are executing on the correct computer",
+ glob.port);
+ delete mapi;
+ goto error_end;
+ }
+
+ if(!glob.socketServer->setup(mstat, glob.port_stats, glob.interface_name)){
+ ndbout_c("Unable to setup statistic port: %d!\nPlease check if the port"
+ " is already used.", glob.port_stats);
+ delete mstat;
+ goto error_end;
+ }
+
+ glob.mgmObject = new MgmtSrvr(glob.localNodeId,
+ BaseString(glob.config_filename),
+ BaseString(glob.local_config_filename == 0 ? "" : glob.local_config_filename));
+
+ if(!glob.mgmObject->check_start()){
+ ndbout_c("Unable to start management server.");
+ ndbout_c("Probably caused by illegal initial configuration file.");
+ goto error_end;
+ }
+
+#if defined (NDB_LINUX) || defined (NDB_SOLARIS)
+ if (glob.daemon) {
+ // Become a daemon
+ char homePath[255],lockfile[255], logfile[255];
+ NdbConfig_HomePath(homePath, 255);
+ snprintf(lockfile, 255, "%snode%d.pid", homePath, glob.localNodeId);
+ snprintf(logfile, 255, "%snode%d.out", homePath, glob.localNodeId);
+ if (NdbDaemon_Make(lockfile, logfile, 0) == -1) {
+ ndbout << "Cannot become daemon: " << NdbDaemon_ErrorText << endl;
+ return 1;
+ }
+ }
+#endif
+
+ if(!glob.mgmObject->start()){
+ ndbout_c("Unable to start management server.");
+ ndbout_c("Probably caused by illegal initial configuration file.");
+ goto error_end;
+ }
+
+ //glob.mgmObject->saveConfig();
+
+ mstat->setMgm(glob.mgmObject);
+ mapi->setMgm(glob.mgmObject);
+ glob.mgmObject->setStatisticsListner(mstat);
+
+ char msg[256];
+ snprintf(msg, sizeof(msg),
+ "NDB Cluster Management Server. %s", NDB_VERSION_STRING);
+ ndbout_c(msg);
+ g_EventLogger.info(msg);
+
+ snprintf(msg, 256, "Command port: %d, Statistics port: %d",
+ glob.port, glob.port_stats);
+ ndbout_c(msg);
+ g_EventLogger.info(msg);
+
+ g_StopServer = false;
+ glob.socketServer->startServer();
+
+#if ! defined NDB_OSE && ! defined NDB_SOFTOSE
+ if(!glob.daemon && !glob.non_interactive){
+ CommandInterpreter com(* glob.mgmObject);
+ while(com.readAndExecute());
+ } else
+#endif
+ {
+ while(g_StopServer != true)
+ NdbSleep_MilliSleep(500);
+ }
+
+ glob.socketServer->stopServer();
+ glob.socketServer->stopSessions();
+
+ return 0;
+ error_end:
+ return 1;
+}
+
+MgmGlobals::MgmGlobals(){
+ // Default values
+ port = 0;
+ port_stats = 0;
+ config_filename = NULL;
+ local_config_filename = NULL;
+ interface_name = 0;
+ cluster_config = 0;
+ daemon = false;
+ non_interactive = 0;
+ socketServer = 0;
+ mgmObject = 0;
+}
+
+MgmGlobals::~MgmGlobals(){
+ if (socketServer)
+ delete socketServer;
+ if (mgmObject)
+ delete mgmObject;
+ if (cluster_config)
+ delete cluster_config;
+ if (interface_name)
+ free(interface_name);
+}
+
+/**
+ * @fn readLocalConfig
+ * @param glob : Global variables
+ * @return true if success, false otherwise.
+ *
+ * How to get LOCAL CONFIGURATION FILE:
+ * 1. Use local config file name (-l)
+ * 2. Use environment NDB_HOME + Ndb.cfg
+ * If NDB_HOME is not set this results in reading from local dir
+ */
+static bool
+readLocalConfig(){
+ // Read local config file
+ ConfigRetriever cr;
+ cr.setLocalConfigFileName(glob.local_config_filename);
+ int nodeid = cr.init(true);
+ if(nodeid == -1){
+ return false;
+ }
+
+ glob.localNodeId = (NodeId)nodeid;
+ return true;
+}
+
+
+/**
+ * @fn readGlobalConfig
+ * @param glob : Global variables
+ * @return true if success, false otherwise.
+ *
+ * How to get the GLOBAL CONFIGURATION:
+ * 1. Use config file name (this is a text file)(-c)
+ * 2. Use name from line 2 of local config file, ex: file:///c/ndb/Ndb_cfg.bin
+ */
+static bool
+readGlobalConfig() {
+ if(glob.config_filename == NULL)
+ return false;
+
+ /* Use config file */
+ InitConfigFileParser parser(glob.config_filename);
+
+ if(parser.readConfigFile()) {
+ glob.cluster_config = new Config(*parser.getConfig());
+ } else {
+ /* Try to get configuration from other MGM server */
+ ConfigRetriever cr;
+ cr.setLocalConfigFileName(glob.local_config_filename);
+ Properties* mgmconf = cr.getConfig("MGM", NDB_VERSION);
+ if (mgmconf == NULL)
+ return false;
+ glob.cluster_config = new Config(*mgmconf);
+ }
+ return true;
+}
+
+/**
+ * @fn setPortNo
+ * @param glob : Global variables
+ * @return true if success, false otherwise.
+ *
+ * Port number:
+ * 2. Use port number from global configuration file
+ * 4. Use port number for statistics from global configuration file
+ */
+static bool
+setPortNo(){
+ const Properties *mgmProps;
+
+ if(!glob.cluster_config->get("Node", glob.localNodeId, &mgmProps)){
+ ndbout << "Could not retrieve configuration for Node "
+ << glob.localNodeId << " in config file." << endl
+ << "Have you set correct NodeId for this node?" << endl;
+ return false;
+ }
+
+ BaseString type;
+ if(!mgmProps->get("Type", type) || strcasecmp(type.c_str(), "MGM") != 0){
+ ndbout << "Local node id " << glob.localNodeId
+ << " is not defined as management server" << endl
+ << "Have you set correct NodeId for this node?" << endl;
+ return false;
+ }
+
+ /************
+ * Set Port *
+ ************/
+ Uint32 tmp = 0;
+ if (!mgmProps->get("PortNumber", &tmp)){
+ ndbout << "Could not find PortNumber in the configuration file." << endl;
+ return false;
+ }
+ glob.port = tmp;
+
+ /*****************
+ * Set Stat Port *
+ *****************/
+ if (!mgmProps->get("PortNumberStats", &tmp)){
+ ndbout << "Could not find PortNumberStats in the configuration file."
+ << endl;
+ return false;
+ }
+ glob.port_stats = tmp;
+
+ BaseString host;
+ if(!mgmProps->get("ExecuteOnComputer", host)){
+ ndbout << "Failed to find \"ExecuteOnComputer\" for my node" << endl;
+ ndbout << "Unable to verify own hostname" << endl;
+ return false;
+ }
+
+ const char * hostname;
+ {
+ const Properties * p;
+ char buf[255];
+ snprintf(buf, sizeof(buf), "Computer_%s", host.c_str());
+ if(!glob.cluster_config->get(buf, &p)){
+ ndbout << "Failed to find computer " << host << " in config" << endl;
+ ndbout << "Unable to verify own hostname" << endl;
+ return false;
+ }
+ if(!p->get("HostName", &hostname)){
+ ndbout << "Failed to find \"HostName\" for computer " << host
+ << " in config" << endl;
+ ndbout << "Unable to verify own hostname" << endl;
+ return false;
+ }
+ if(NdbHost_GetHostName(buf) != 0){
+ ndbout << "Unable to get own hostname" << endl;
+ ndbout << "Unable to verify own hostname" << endl;
+ return false;
+ }
+ }
+
+ const char * ip_address;
+ if(mgmProps->get("IpAddress", &ip_address)){
+ glob.use_specific_ip = true;
+ glob.interface_name = strdup(ip_address);
+ return true;
+ }
+
+ glob.use_specific_ip = false;
+ glob.interface_name = strdup(hostname);
+
+ return true;
+}
diff --git a/ndb/src/mgmsrv/mkconfig/Makefile b/ndb/src/mgmsrv/mkconfig/Makefile
new file mode 100644
index 00000000000..d35f68a5648
--- /dev/null
+++ b/ndb/src/mgmsrv/mkconfig/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+BIN_TARGET := mkconfig
+BIN_TARGET_ARCHIVES := logger general trace mgmsrvcommon portlib
+
+SOURCES := mkconfig.cpp
+
+CCFLAGS_LOC += -I.. -I$(call fixpath,$(NDB_TOP)/src/common/mgmcommon)
+
+OBJECTS_LOC := ../convertStrToInt.o
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/mgmsrv/mkconfig/mkconfig.cpp b/ndb/src/mgmsrv/mkconfig/mkconfig.cpp
new file mode 100644
index 00000000000..0e9397e43c0
--- /dev/null
+++ b/ndb/src/mgmsrv/mkconfig/mkconfig.cpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbMain.h>
+#include <Properties.hpp>
+#include <stdlib.h>
+
+#include "InitConfigFileParser.hpp"
+#include "Config.hpp"
+#include <assert.h>
+#include <NdbUnistd.h>
+#include <ndb_version.h>
+
+void usage(const char * prg){
+ ndbout << "Usage " << prg << ": <Init config> <Binary file>" << endl;
+
+}
+
+NDB_COMMAND(mkconfig,
+ "mkconfig", "mkconfig",
+ "Make a binary configuration from a config file", 16384){
+ if(argc < 3){
+ usage(argv[0]);
+ return 0;
+ }
+
+ InitConfigFileParser parser(argv[1]);
+ Config* cp;
+
+ if (!parser.readConfigFile())
+ return false;
+
+ cp = (Config *) parser.getConfig();
+ if (cp == NULL)
+ return false;
+
+ cp->put("VersionId", (Uint32)NDB_VERSION);
+
+ Uint32 sz = cp->getPackedSize();
+ Uint32 * buf = new Uint32[sz];
+ if(!cp->pack(buf))
+ return -1;
+
+ FILE * f = fopen(argv[2], "w");
+ if(fwrite(buf, 1, sz, f) != sz){
+ fclose(f);
+ unlink(argv[2]);
+ return -1;
+ }
+ fclose(f);
+ return 0;
+}
diff --git a/ndb/src/ndbapi/API.hpp b/ndb/src/ndbapi/API.hpp
new file mode 100644
index 00000000000..05e2d863cb6
--- /dev/null
+++ b/ndb/src/ndbapi/API.hpp
@@ -0,0 +1,26 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef API_H
+#define API_H
+
+#include <BlockNumbers.h>
+#include <GlobalSignalNumbers.h>
+#include <RefConvert.hpp>
+#include "NdbImpl.hpp"
+#include "NdbDictionaryImpl.hpp"
+
+#endif
diff --git a/ndb/src/ndbapi/ClusterMgr.cpp b/ndb/src/ndbapi/ClusterMgr.cpp
new file mode 100644
index 00000000000..f9c553ab63b
--- /dev/null
+++ b/ndb/src/ndbapi/ClusterMgr.cpp
@@ -0,0 +1,772 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "TransporterFacade.hpp"
+#include "ClusterMgr.hpp"
+#include <IPCConfig.hpp>
+#include "AttrType.hpp"
+#include "NdbApiSignal.hpp"
+#include "API.hpp"
+#include <NdbSleep.h>
+#include <NdbOut.hpp>
+#include <NdbTick.h>
+#include <ndb_limits.h>
+
+#include <stdio.h>
+#include <assert.h>
+
+#include <ndb_version.h>
+#include <signaldata/NodeFailRep.hpp>
+#include <signaldata/NFCompleteRep.hpp>
+#include <signaldata/ApiRegSignalData.hpp>
+
+// Just a C wrapper for threadMain
+extern "C"
+void*
+runClusterMgr_C(void * me)
+{
+ ((ClusterMgr*) me)->threadMain();
+ /**
+ * Sleep to allow another thread that is not exiting to take control
+ * of signals allocated by this thread
+ *
+ * see Ndb::~Ndb() in Ndbinit.cpp
+ */
+#ifdef NDB_OSE
+ NdbSleep_MilliSleep(50);
+#endif
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+extern "C" {
+ void ndbSetOwnVersion();
+}
+ClusterMgr::ClusterMgr(TransporterFacade & _facade):
+ theStop(0),
+ theFacade(_facade)
+{
+ ndbSetOwnVersion();
+ clusterMgrThreadMutex = NdbMutex_Create();
+ noOfConnectedNodes = 0;
+}
+
+ClusterMgr::~ClusterMgr(){
+ doStop();
+ NdbMutex_Destroy(clusterMgrThreadMutex);
+}
+
+void
+ClusterMgr::init(const IPCConfig & config){
+ NodeId tmp = 0;
+ while(config.getNextRemoteNodeId(tmp)) {
+ theNodes[tmp].defined = true;
+#if 0
+ ndbout << "--------------------------------------" << endl;
+ config.print();
+ ndbout << "--------------------------------------" << endl;
+ ndbout_c("ClusterMgr: Node %d defined as %s", tmp, config.getNodeType(tmp));
+#endif
+ if(strcmp(config.getNodeType(tmp), "DB") == 0) {
+ theNodes[tmp].m_info.m_type = NodeInfo::DB;
+ } else if(strcmp(config.getNodeType(tmp), "API") == 0) {
+ theNodes[tmp].m_info.m_type = NodeInfo::API;
+ } else if(strcmp(config.getNodeType(tmp), "MGM") == 0) {
+ theNodes[tmp].m_info.m_type = NodeInfo::MGM;
+ } else if(strcmp(config.getNodeType(tmp), "REP") == 0) {
+ theNodes[tmp].m_info.m_type = NodeInfo::REP;
+ } else if(strcmp(config.getNodeType(tmp), "EXTERNAL REP") == 0) {
+ theNodes[tmp].m_info.m_type = NodeInfo::REP;
+ theNodes[tmp].hbFrequency = config.getREPHBFrequency(tmp);
+ assert(100 <= theNodes[tmp].hbFrequency &&
+ theNodes[tmp].hbFrequency < 60 * 60 * 1000);
+ } else {
+#if 0
+ ndbout_c("ClusterMgr: Unknown node type: %s", config.getNodeType(tmp));
+#endif
+ }
+ }
+}
+
+void
+ClusterMgr::startThread() {
+ NdbMutex_Lock(clusterMgrThreadMutex);
+
+ theStop = 0;
+
+ theClusterMgrThread = NdbThread_Create(runClusterMgr_C,
+ (void**)this,
+ 32768,
+ "ndb_clustermgr",
+ NDB_THREAD_PRIO_LOW);
+ NdbMutex_Unlock(clusterMgrThreadMutex);
+}
+
+void
+ClusterMgr::doStop( ){
+ NdbMutex_Lock(clusterMgrThreadMutex);
+
+ if(theStop){
+ NdbMutex_Unlock(clusterMgrThreadMutex);
+ return;
+ }
+
+ void *status;
+ theStop = 1;
+
+ NdbThread_WaitFor(theClusterMgrThread, &status);
+ NdbThread_Destroy(&theClusterMgrThread);
+
+ NdbMutex_Unlock(clusterMgrThreadMutex);
+}
+
+void
+ClusterMgr::threadMain( ){
+ NdbApiSignal signal(numberToRef(API_CLUSTERMGR, theFacade.ownId()));
+
+ signal.theVerId_signalNumber = GSN_API_REGREQ;
+ signal.theReceiversBlockNumber = QMGR;
+ signal.theTrace = 0;
+ signal.theLength = ApiRegReq::SignalLength;
+
+ ApiRegReq * req = CAST_PTR(ApiRegReq, signal.getDataPtrSend());
+ req->ref = numberToRef(API_CLUSTERMGR, theFacade.ownId());
+ req->version = NDB_VERSION;
+
+
+ Uint32 timeSlept = 100;
+ Uint64 now = NdbTick_CurrentMillisecond();
+
+ while(!theStop){
+ /**
+ * Start of Secure area for use of Transporter
+ */
+ theFacade.lock_mutex();
+ for (int i = 1; i < MAX_NODES; i++){
+ /**
+ * Send register request (heartbeat) to all available nodes
+ * at specified timing intervals
+ */
+ const NodeId nodeId = i;
+ Node & theNode = theNodes[nodeId];
+
+ if (theNode.defined == true) {
+#if 0
+ ndbout_c("ClusterMgr: compatible %d", (int)nodeId);
+#endif
+
+ if (theNode.connected == false){
+ theFacade.doConnect(nodeId);
+ continue;
+ }
+
+#if 0
+ ndbout_c("ClusterMgr: connected %d", (int)nodeId);
+#endif
+
+ theNode.hbCounter += timeSlept;
+ if (theNode.hbCounter >= theNode.hbFrequency){
+ /**
+ * It is now time to send a new Heartbeat
+ */
+ theNode.hbSent++;
+ theNode.hbCounter = 0;
+ /**
+ * If the node is of type REP,
+ * then the receiver of the signal should be API_CLUSTERMGR
+ */
+ if (theNode.m_info.m_type == NodeInfo::REP) {
+ signal.theReceiversBlockNumber = API_CLUSTERMGR;
+ }
+#if 0
+ ndbout_c("ClusterMgr: Sending API_REGREQ to node %d", (int)nodeId);
+#endif
+ theFacade.sendSignalUnCond(&signal, nodeId);
+ }//if
+
+ if (theNode.hbSent == 4 && theNode.hbFrequency > 0){
+ reportNodeFailed(i);
+ }//if
+ }//if(defined)
+ }//for
+ /**
+ * End of secure area. Let other threads in
+ */
+ theFacade.unlock_mutex();
+
+ // Sleep for 100 ms between each Registration Heartbeat
+ Uint64 before = now;
+ NdbSleep_MilliSleep(100);
+ now = NdbTick_CurrentMillisecond();
+ timeSlept = (now - before);
+ }
+}
+
+#if 0
+void
+ClusterMgr::showState(NodeId nodeId){
+ ndbout << "-- ClusterMgr - NodeId = " << nodeId << endl;
+ ndbout << "theNodeList = " << theNodeList[nodeId] << endl;
+ ndbout << "theNodeState = " << theNodeState[nodeId] << endl;
+ ndbout << "theNodeCount = " << theNodeCount[nodeId] << endl;
+ ndbout << "theNodeStopDelay = " << theNodeStopDelay[nodeId] << endl;
+ ndbout << "theNodeSendDelay = " << theNodeSendDelay[nodeId] << endl;
+}
+#endif
+
+ClusterMgr::Node::Node()
+ : m_state(NodeState::SL_NOTHING) {
+ compatible = nfCompleteRep = true;
+ connected = defined = m_alive = false;
+}
+
+/******************************************************************************
+ * API_REGREQ and friends
+ ******************************************************************************/
+
+void
+ClusterMgr::execAPI_REGREQ(const Uint32 * theData){
+ const ApiRegReq * const apiRegReq = (ApiRegReq *)&theData[0];
+ const NodeId nodeId = refToNode(apiRegReq->ref);
+
+#if 0
+ ndbout_c("ClusterMgr: Recd API_REGREQ from node %d", nodeId);
+#endif
+
+ assert(nodeId > 0 && nodeId < MAX_NODES);
+
+ Node & node = theNodes[nodeId];
+ assert(node.defined == true);
+ assert(node.connected == true);
+
+ if(node.m_info.m_version != apiRegReq->version){
+ node.m_info.m_version = apiRegReq->version;
+
+ if (getMajor(node.m_info.m_version) < getMajor(NDB_VERSION) ||
+ getMinor(node.m_info.m_version) < getMinor(NDB_VERSION)) {
+ node.compatible = false;
+ } else {
+ node.compatible = true;
+ }
+ }
+
+ NdbApiSignal signal(numberToRef(API_CLUSTERMGR, theFacade.ownId()));
+ signal.theVerId_signalNumber = GSN_API_REGCONF;
+ signal.theReceiversBlockNumber = API_CLUSTERMGR;
+ signal.theTrace = 0;
+ signal.theLength = ApiRegConf::SignalLength;
+
+ ApiRegConf * const conf = CAST_PTR(ApiRegConf, signal.getDataPtrSend());
+ conf->qmgrRef = numberToRef(API_CLUSTERMGR, theFacade.ownId());
+ conf->version = NDB_VERSION;
+ conf->apiHeartbeatFrequency = node.hbFrequency;
+ theFacade.sendSignalUnCond(&signal, nodeId);
+}
+
+int global_mgmt_server_check = 0; // set to one in mgmtsrvr main;
+void
+ClusterMgr::execAPI_REGCONF(const Uint32 * theData){
+ const ApiRegConf * const apiRegConf = (ApiRegConf *)&theData[0];
+ const NodeId nodeId = refToNode(apiRegConf->qmgrRef);
+
+ assert(nodeId > 0 && nodeId < MAX_NODES);
+
+ Node & node = theNodes[nodeId];
+ assert(node.defined == true);
+ assert(node.connected == true);
+ if(node.m_info.m_version != apiRegConf->version){
+ node.m_info.m_version = apiRegConf->version;
+ if (global_mgmt_server_check == 1)
+ node.compatible = ndbCompatible_mgmt_ndb(NDB_VERSION,
+ node.m_info.m_version);
+ else
+ node.compatible = ndbCompatible_api_ndb(NDB_VERSION,
+ node.m_info.m_version);
+ }
+
+ node.m_state = apiRegConf->nodeState;
+ if (node.compatible && (node.m_state.startLevel == NodeState::SL_STARTED ||
+ node.m_state.startLevel == NodeState::SL_SINGLEUSER)){
+ node.m_alive = true;
+ } else {
+ node.m_alive = false;
+ }//if
+ node.hbSent = 0;
+ node.hbCounter = 0;
+ if (node.m_info.m_type != NodeInfo::REP) {
+ node.hbFrequency = (apiRegConf->apiHeartbeatFrequency * 10) - 50;
+ }
+}
+
+void
+ClusterMgr::execAPI_REGREF(const Uint32 * theData){
+
+ ApiRegRef * ref = (ApiRegRef*)theData;
+
+ const NodeId nodeId = refToNode(ref->ref);
+
+ assert(nodeId > 0 && nodeId < MAX_NODES);
+
+ Node & node = theNodes[nodeId];
+ assert(node.connected == true);
+ assert(node.defined == true);
+
+ node.compatible = false;
+ node.m_alive = false;
+ node.m_state = NodeState::SL_NOTHING;
+ node.m_info.m_version = ref->version;
+
+ switch(ref->errorCode){
+ case ApiRegRef::WrongType:
+ ndbout_c("Node %d reports that this node should be a NDB node", nodeId);
+ abort();
+ case ApiRegRef::UnsupportedVersion:
+ default:
+ break;
+ }
+}
+
+void
+ClusterMgr::execNODE_FAILREP(const Uint32 * theData){
+ NodeFailRep * const nodeFail = (NodeFailRep *)&theData[0];
+ for(int i = 1; i<MAX_NODES; i++){
+ if(NodeBitmask::get(nodeFail->theNodes, i)){
+ reportNodeFailed(i);
+ }
+ }
+}
+
+void
+ClusterMgr::execNF_COMPLETEREP(const Uint32 * theData){
+ NFCompleteRep * const nfComp = (NFCompleteRep *)theData;
+
+ const NodeId nodeId = nfComp->failedNodeId;
+ assert(nodeId > 0 && nodeId < MAX_NODES);
+
+ theFacade.ReportNodeFailureComplete(nodeId);
+ theNodes[nodeId].nfCompleteRep = true;
+}
+
+void
+ClusterMgr::reportConnected(NodeId nodeId){
+ /**
+ * Ensure that we are sending heartbeat every 100 ms
+ * until we have got the first reply from NDB providing
+ * us with the real time-out period to use.
+ */
+ assert(nodeId > 0 && nodeId < MAX_NODES);
+
+ noOfConnectedNodes++;
+
+ Node & theNode = theNodes[nodeId];
+ theNode.connected = true;
+ theNode.hbSent = 0;
+ theNode.hbCounter = 0;
+
+ if (theNode.m_info.m_type != NodeInfo::REP) {
+ theNode.hbFrequency = 0;
+ }
+ theNode.m_info.m_version = 0;
+ theNode.compatible = true;
+ theNode.nfCompleteRep = true;
+
+ theFacade.ReportNodeAlive(nodeId);
+}
+
+void
+ClusterMgr::reportDisconnected(NodeId nodeId){
+ assert(nodeId > 0 && nodeId < MAX_NODES);
+ assert(noOfConnectedNodes > 0);
+
+ noOfConnectedNodes--;
+ theNodes[nodeId].connected = false;
+ theNodes[nodeId].m_info.m_connectCount ++;
+ reportNodeFailed(nodeId);
+}
+
+void
+ClusterMgr::reportNodeFailed(NodeId nodeId){
+
+ Node & theNode = theNodes[nodeId];
+
+ theNode.m_alive = false;
+ if(theNode.connected)
+ theFacade.doDisconnect(nodeId);
+
+ const bool report = (theNode.m_state.startLevel != NodeState::SL_NOTHING);
+ theNode.m_state.startLevel = NodeState::SL_NOTHING;
+
+ if(report)
+ theFacade.ReportNodeDead(nodeId);
+
+ theNode.nfCompleteRep = false;
+
+ if(noOfConnectedNodes == 0){
+ Uint32 theData[1];
+ NFCompleteRep * rep = (NFCompleteRep *)&theData[0];
+
+ for(Uint32 i = 1; i<MAX_NODES; i++){
+ if(theNodes[i].defined && theNodes[i].nfCompleteRep == false){
+ rep->failedNodeId = i;
+ execNF_COMPLETEREP(theData);
+ }
+ }
+ }
+}
+
+/******************************************************************************
+ * Arbitrator
+ ******************************************************************************/
+ArbitMgr::ArbitMgr(TransporterFacade & _fac)
+ : theFacade(_fac)
+{
+ theThreadMutex = NdbMutex_Create();
+ theInputCond = NdbCondition_Create();
+ theInputMutex = NdbMutex_Create();
+
+ theRank = 0;
+ theDelay = 0;
+ theThread = 0;
+
+ theInputTimeout = 0;
+ theInputFull = false;
+ memset(&theInputFull, 0, sizeof(theInputFull));
+ theState = StateInit;
+
+ memset(&theStartReq, 0, sizeof(theStartReq));
+ memset(&theChooseReq1, 0, sizeof(theChooseReq1));
+ memset(&theChooseReq2, 0, sizeof(theChooseReq2));
+ memset(&theStopOrd, 0, sizeof(theStopOrd));
+}
+
+ArbitMgr::~ArbitMgr()
+{
+ NdbMutex_Destroy(theThreadMutex);
+ NdbCondition_Destroy(theInputCond);
+ NdbMutex_Destroy(theInputMutex);
+}
+
+// Start arbitrator thread. This is kernel request.
+// First stop any previous thread since it is a left-over
+// which was never used and which now has wrong ticket.
+void
+ArbitMgr::doStart(const Uint32* theData)
+{
+ ArbitSignal aSignal;
+ NdbMutex_Lock(theThreadMutex);
+ if (theThread != NULL) {
+ aSignal.init(GSN_ARBIT_STOPORD, NULL);
+ aSignal.data.code = StopRestart;
+ sendSignalToThread(aSignal);
+ void* value;
+ NdbThread_WaitFor(theThread, &value);
+ theThread = NULL;
+ theState = StateInit;
+ theInputFull = false;
+ }
+ aSignal.init(GSN_ARBIT_STARTREQ, theData);
+ sendSignalToThread(aSignal);
+ theThread = NdbThread_Create(
+ runArbitMgr_C, (void**)this, 32768, "ndb_arbitmgr",
+ NDB_THREAD_PRIO_HIGH);
+ NdbMutex_Unlock(theThreadMutex);
+}
+
+// The "choose me" signal from a candidate.
+void
+ArbitMgr::doChoose(const Uint32* theData)
+{
+ ArbitSignal aSignal;
+ aSignal.init(GSN_ARBIT_CHOOSEREQ, theData);
+ sendSignalToThread(aSignal);
+}
+
+// Stop arbitrator thread via stop signal from the kernel
+// or when exiting API program.
+void
+ArbitMgr::doStop(const Uint32* theData)
+{
+ ArbitSignal aSignal;
+ NdbMutex_Lock(theThreadMutex);
+ if (theThread != NULL) {
+ aSignal.init(GSN_ARBIT_STOPORD, theData);
+ if (theData == 0) {
+ aSignal.data.code = StopExit;
+ } else {
+ aSignal.data.code = StopRequest;
+ }
+ sendSignalToThread(aSignal);
+ void* value;
+ NdbThread_WaitFor(theThread, &value);
+ theThread = NULL;
+ theState = StateInit;
+ }
+ NdbMutex_Unlock(theThreadMutex);
+}
+
+// private methods
+
+extern "C"
+void*
+runArbitMgr_C(void* me)
+{
+ ((ArbitMgr*) me)->threadMain();
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+void
+ArbitMgr::sendSignalToThread(ArbitSignal& aSignal)
+{
+#ifdef DEBUG_ARBIT
+ char buf[17] = "";
+ ndbout << "arbit recv: ";
+ ndbout << " gsn=" << aSignal.gsn;
+ ndbout << " send=" << aSignal.data.sender;
+ ndbout << " code=" << aSignal.data.code;
+ ndbout << " node=" << aSignal.data.node;
+ ndbout << " ticket=" << aSignal.data.ticket.getText(buf, sizeof(buf));
+ ndbout << " mask=" << aSignal.data.mask.getText(buf, sizeof(buf));
+ ndbout << endl;
+#endif
+ aSignal.setTimestamp(); // signal arrival time
+ NdbMutex_Lock(theInputMutex);
+ while (theInputFull) {
+ NdbCondition_WaitTimeout(theInputCond, theInputMutex, 1000);
+ }
+ theInputBuffer = aSignal;
+ theInputFull = true;
+ NdbCondition_Signal(theInputCond);
+ NdbMutex_Unlock(theInputMutex);
+}
+
+void
+ArbitMgr::threadMain()
+{
+ ArbitSignal aSignal;
+ aSignal = theInputBuffer;
+ threadStart(aSignal);
+ bool stop = false;
+ while (! stop) {
+ NdbMutex_Lock(theInputMutex);
+ while (! theInputFull) {
+ NdbCondition_WaitTimeout(theInputCond, theInputMutex, theInputTimeout);
+ threadTimeout();
+ }
+ aSignal = theInputBuffer;
+ theInputFull = false;
+ NdbCondition_Signal(theInputCond);
+ NdbMutex_Unlock(theInputMutex);
+ switch (aSignal.gsn) {
+ case GSN_ARBIT_CHOOSEREQ:
+ threadChoose(aSignal);
+ break;
+ case GSN_ARBIT_STOPORD:
+ stop = true;
+ break;
+ }
+ }
+ threadStop(aSignal);
+}
+
+// handle events in the thread
+
+void
+ArbitMgr::threadStart(ArbitSignal& aSignal)
+{
+ theStartReq = aSignal;
+ sendStartConf(theStartReq, ArbitCode::ApiStart);
+ theState = StateStarted;
+ theInputTimeout = 1000;
+}
+
+void
+ArbitMgr::threadChoose(ArbitSignal& aSignal)
+{
+ switch (theState) {
+ case StateStarted: // first REQ
+ if (! theStartReq.data.match(aSignal.data)) {
+ sendChooseRef(aSignal, ArbitCode::ErrTicket);
+ break;
+ }
+ theChooseReq1 = aSignal;
+ if (theDelay == 0) {
+ sendChooseConf(aSignal, ArbitCode::WinChoose);
+ theState = StateFinished;
+ theInputTimeout = 1000;
+ break;
+ }
+ theState = StateChoose1;
+ theInputTimeout = 1;
+ return;
+ case StateChoose1: // second REQ within Delay
+ if (! theStartReq.data.match(aSignal.data)) {
+ sendChooseRef(aSignal, ArbitCode::ErrTicket);
+ break;
+ }
+ theChooseReq2 = aSignal;
+ theState = StateChoose2;
+ theInputTimeout = 1;
+ return;
+ case StateChoose2: // too many REQs - refuse all
+ if (! theStartReq.data.match(aSignal.data)) {
+ sendChooseRef(aSignal, ArbitCode::ErrTicket);
+ break;
+ }
+ sendChooseRef(theChooseReq1, ArbitCode::ErrToomany);
+ sendChooseRef(theChooseReq2, ArbitCode::ErrToomany);
+ sendChooseRef(aSignal, ArbitCode::ErrToomany);
+ theState = StateFinished;
+ theInputTimeout = 1000;
+ return;
+ default:
+ sendChooseRef(aSignal, ArbitCode::ErrState);
+ break;
+ }
+}
+
+void
+ArbitMgr::threadTimeout()
+{
+ switch (theState) {
+ case StateStarted:
+ break;
+ case StateChoose1:
+ if (theChooseReq1.getTimediff() < theDelay)
+ break;
+ sendChooseConf(theChooseReq1, ArbitCode::WinChoose);
+ theState = StateFinished;
+ theInputTimeout = 1000;
+ break;
+ case StateChoose2:
+ sendChooseConf(theChooseReq1, ArbitCode::WinChoose);
+ sendChooseConf(theChooseReq2, ArbitCode::LoseChoose);
+ theState = StateFinished;
+ theInputTimeout = 1000;
+ break;
+ default:
+ break;
+ }
+}
+
+void
+ArbitMgr::threadStop(ArbitSignal& aSignal)
+{
+ switch (aSignal.data.code) {
+ case StopExit:
+ switch (theState) {
+ case StateStarted:
+ sendStopRep(theStartReq, 0);
+ break;
+ case StateChoose1: // just in time
+ sendChooseConf(theChooseReq1, ArbitCode::WinChoose);
+ break;
+ case StateChoose2:
+ sendChooseConf(theChooseReq1, ArbitCode::WinChoose);
+ sendChooseConf(theChooseReq2, ArbitCode::LoseChoose);
+ break;
+ case StateInit:
+ case StateFinished:
+ //??
+ break;
+ }
+ break;
+ case StopRequest:
+ break;
+ case StopRestart:
+ break;
+ }
+}
+
+// output routines
+
+void
+ArbitMgr::sendStartConf(ArbitSignal& aSignal, Uint32 code)
+{
+ ArbitSignal copySignal = aSignal;
+ copySignal.gsn = GSN_ARBIT_STARTCONF;
+ copySignal.data.code = code;
+ sendSignalToQmgr(copySignal);
+}
+
+void
+ArbitMgr::sendChooseConf(ArbitSignal& aSignal, Uint32 code)
+{
+ ArbitSignal copySignal = aSignal;
+ copySignal.gsn = GSN_ARBIT_CHOOSECONF;
+ copySignal.data.code = code;
+ sendSignalToQmgr(copySignal);
+}
+
+void
+ArbitMgr::sendChooseRef(ArbitSignal& aSignal, Uint32 code)
+{
+ ArbitSignal copySignal = aSignal;
+ copySignal.gsn = GSN_ARBIT_CHOOSEREF;
+ copySignal.data.code = code;
+ sendSignalToQmgr(copySignal);
+}
+
+void
+ArbitMgr::sendStopRep(ArbitSignal& aSignal, Uint32 code)
+{
+ ArbitSignal copySignal = aSignal;
+ copySignal.gsn = GSN_ARBIT_STOPREP;
+ copySignal.data.code = code;
+ sendSignalToQmgr(copySignal);
+}
+
+/**
+ * Send signal to QMGR. The input includes signal number and
+ * signal data. The signal data is normally a copy of a received
+ * signal so it contains expected arbitrator node id and ticket.
+ * The sender in signal data is the QMGR node id.
+ */
+void
+ArbitMgr::sendSignalToQmgr(ArbitSignal& aSignal)
+{
+ NdbApiSignal signal(numberToRef(API_CLUSTERMGR, theFacade.ownId()));
+
+ signal.theVerId_signalNumber = aSignal.gsn;
+ signal.theReceiversBlockNumber = QMGR;
+ signal.theTrace = 0;
+ signal.theLength = ArbitSignalData::SignalLength;
+
+ ArbitSignalData* sd = CAST_PTR(ArbitSignalData, signal.getDataPtrSend());
+
+ sd->sender = numberToRef(API_CLUSTERMGR, theFacade.ownId());
+ sd->code = aSignal.data.code;
+ sd->node = aSignal.data.node;
+ sd->ticket = aSignal.data.ticket;
+ sd->mask = aSignal.data.mask;
+
+#ifdef DEBUG_ARBIT
+ char buf[17] = "";
+ ndbout << "arbit send: ";
+ ndbout << " gsn=" << aSignal.gsn;
+ ndbout << " recv=" << aSignal.data.sender;
+ ndbout << " code=" << aSignal.data.code;
+ ndbout << " node=" << aSignal.data.node;
+ ndbout << " ticket=" << aSignal.data.ticket.getText(buf, sizeof(buf));
+ ndbout << " mask=" << aSignal.data.mask.getText(buf, sizeof(buf));
+ ndbout << endl;
+#endif
+
+ theFacade.lock_mutex();
+ theFacade.sendSignalUnCond(&signal, aSignal.data.sender);
+ theFacade.unlock_mutex();
+}
+
diff --git a/ndb/src/ndbapi/ClusterMgr.hpp b/ndb/src/ndbapi/ClusterMgr.hpp
new file mode 100644
index 00000000000..7b7b947742b
--- /dev/null
+++ b/ndb/src/ndbapi/ClusterMgr.hpp
@@ -0,0 +1,215 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef ClusterMgr_H
+#define ClusterMgr_H
+
+#include "API.hpp"
+#include <ndb_limits.h>
+#include <NdbThread.h>
+#include <NdbMutex.h>
+#include <NdbCondition.h>
+#include <signaldata/ArbitSignalData.hpp>
+#include <signaldata/NodeStateSignalData.hpp>
+#include <NodeInfo.hpp>
+#include <NodeState.hpp>
+
+extern "C" void* runClusterMgr_C(void * me);
+
+
+/**
+ * @class ClusterMgr
+ */
+class ClusterMgr {
+ friend void* runClusterMgr_C(void * me);
+ friend void execute(void *, struct SignalHeader * const,
+ Uint8, Uint32 * const, LinearSectionPtr ptr[3]);
+public:
+ ClusterMgr(class TransporterFacade &);
+ ~ClusterMgr();
+ void init(const IPCConfig & config);
+
+ void reportConnected(NodeId nodeId);
+ void reportDisconnected(NodeId nodeId);
+
+ bool checkUpgradeCompatability(Uint32 nodeVersion);
+
+ void doStop();
+ void startThread();
+
+private:
+ void threadMain();
+
+ int theStop;
+ class TransporterFacade & theFacade;
+
+public:
+ struct Node {
+ Node();
+ bool defined;
+ bool connected; // Transporter connected
+ bool compatible; // Version is compatible
+ bool nfCompleteRep; // NF Complete Rep has arrived
+ bool m_alive; // Node is alive
+
+ NodeInfo m_info;
+ NodeState m_state;
+
+ /**
+ * Heartbeat stuff
+ */
+ Uint32 hbFrequency; // Heartbeat frequence
+ Uint32 hbCounter; // # milliseconds passed since last hb sent
+ Uint32 hbSent; // # heartbeats sent (without answer)
+ };
+
+ const Node & getNodeInfo(NodeId) const;
+ Uint32 getNoOfConnectedNodes() const;
+
+private:
+ Uint32 noOfConnectedNodes;
+ Node theNodes[MAX_NODES];
+ NdbThread* theClusterMgrThread;
+
+ /**
+ * Used for controlling start/stop of the thread
+ */
+ NdbMutex* clusterMgrThreadMutex;
+
+ void showState(NodeId nodeId);
+ void reportNodeFailed(NodeId nodeId);
+
+ /**
+ * Signals received
+ */
+ void execAPI_REGREQ (const Uint32 * theData);
+ void execAPI_REGCONF (const Uint32 * theData);
+ void execAPI_REGREF (const Uint32 * theData);
+ void execNODE_FAILREP (const Uint32 * theData);
+ void execNF_COMPLETEREP(const Uint32 * theData);
+};
+
+inline
+const ClusterMgr::Node &
+ClusterMgr::getNodeInfo(NodeId nodeId) const {
+ return theNodes[nodeId];
+}
+
+inline
+Uint32
+ClusterMgr::getNoOfConnectedNodes() const {
+ return noOfConnectedNodes;
+}
+
+/******************************************************************************/
+
+/**
+ * @class ArbitMgr
+ * Arbitration manager. Runs in separate thread.
+ * Started only by a request from the kernel.
+ */
+
+extern "C" void* runArbitMgr_C(void* me);
+
+class ArbitMgr
+{
+public:
+ ArbitMgr(class TransporterFacade &);
+ ~ArbitMgr();
+
+ inline void setRank(unsigned n) { theRank = n; }
+ inline void setDelay(unsigned n) { theDelay = n; }
+
+ void doStart(const Uint32* theData);
+ void doChoose(const Uint32* theData);
+ void doStop(const Uint32* theData);
+
+ friend void* runArbitMgr_C(void* me);
+
+private:
+ class TransporterFacade & theFacade;
+ unsigned theRank;
+ unsigned theDelay;
+
+ void threadMain();
+ NdbThread* theThread;
+ NdbMutex* theThreadMutex; // not really needed
+
+ struct ArbitSignal {
+ GlobalSignalNumber gsn;
+ ArbitSignalData data;
+ NDB_TICKS timestamp;
+
+ inline void init(GlobalSignalNumber aGsn, const Uint32* aData) {
+ gsn = aGsn;
+ if (aData != NULL)
+ memcpy(&data, aData, sizeof(data));
+ else
+ memset(&data, 0, sizeof(data));
+ }
+
+ inline void setTimestamp() {
+ timestamp = NdbTick_CurrentMillisecond();
+ }
+
+ inline NDB_TICKS getTimediff() {
+ NDB_TICKS now = NdbTick_CurrentMillisecond();
+ return now < timestamp ? 0 : now - timestamp;
+ }
+ };
+
+ NdbMutex* theInputMutex;
+ NdbCondition* theInputCond;
+ int theInputTimeout;
+ bool theInputFull; // the predicate
+ ArbitSignal theInputBuffer; // shared buffer
+
+ void sendSignalToThread(ArbitSignal& aSignal);
+
+ enum State { // thread states
+ StateInit,
+ StateStarted, // thread started
+ StateChoose1, // received one valid REQ
+ StateChoose2, // received two valid REQs
+ StateFinished // finished one way or other
+ };
+ State theState;
+
+ enum Stop { // stop code in ArbitSignal.data.code
+ StopExit = 1, // at API exit
+ StopRequest = 2, // request from kernel
+ StopRestart = 3 // stop before restart
+ };
+
+ void threadStart(ArbitSignal& aSignal); // handle thread events
+ void threadChoose(ArbitSignal& aSignal);
+ void threadTimeout();
+ void threadStop(ArbitSignal& aSignal);
+
+ ArbitSignal theStartReq;
+ ArbitSignal theChooseReq1;
+ ArbitSignal theChooseReq2;
+ ArbitSignal theStopOrd;
+
+ void sendStartConf(ArbitSignal& aSignal, Uint32);
+ void sendChooseRef(ArbitSignal& aSignal, Uint32);
+ void sendChooseConf(ArbitSignal& aSignal, Uint32);
+ void sendStopRep(ArbitSignal& aSignal, Uint32);
+
+ void sendSignalToQmgr(ArbitSignal& aSignal);
+};
+
+#endif
diff --git a/ndb/src/ndbapi/DictCache.cpp b/ndb/src/ndbapi/DictCache.cpp
new file mode 100644
index 00000000000..36fbc85a875
--- /dev/null
+++ b/ndb/src/ndbapi/DictCache.cpp
@@ -0,0 +1,246 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "DictCache.hpp"
+#include "NdbDictionaryImpl.hpp"
+#include <NdbTick.h>
+#include <NdbCondition.h>
+#include <NdbSleep.h>
+#include <stdlib.h>
+
+LocalDictCache::LocalDictCache(){
+ m_tableHash.createHashTable();
+}
+
+LocalDictCache::~LocalDictCache(){
+ m_tableHash.releaseHashTable();
+}
+
+NdbTableImpl *
+LocalDictCache::get(const char * name){
+ const Uint32 len = strlen(name);
+ return m_tableHash.getData(name, len);
+}
+
+void
+LocalDictCache::put(const char * name, NdbTableImpl * tab){
+ const Uint32 id = tab->m_tableId;
+
+ m_tableHash.insertKey(name, strlen(name), id, tab);
+}
+
+void
+LocalDictCache::drop(const char * name){
+ m_tableHash.deleteKey(name, strlen(name));
+}
+
+/*****************************************************************
+ * Global cache
+ */
+GlobalDictCache::GlobalDictCache(){
+ m_tableHash.createHashTable();
+ m_waitForTableCondition = NdbCondition_Create();
+}
+
+GlobalDictCache::~GlobalDictCache(){
+ NdbElement_t<Vector<TableVersion> > * curr = m_tableHash.getNext(0);
+ while(curr != 0){
+ Vector<TableVersion> * vers = curr->theData;
+ const unsigned sz = vers->size();
+ for(unsigned i = 0; i<sz ; i++){
+ if((* vers)[i].m_impl != 0)
+ delete (* vers)[i].m_impl;
+ }
+ delete curr->theData;
+ curr = m_tableHash.getNext(curr);
+ }
+
+ m_tableHash.releaseHashTable();
+ NdbCondition_Destroy(m_waitForTableCondition);
+}
+
+#include <NdbOut.hpp>
+
+NdbTableImpl *
+GlobalDictCache::get(const char * name)
+{
+ const Uint32 len = strlen(name);
+ Vector<TableVersion> * versions = 0;
+ versions = m_tableHash.getData(name, len);
+ if(versions == 0){
+ versions = new Vector<TableVersion>(2);
+ m_tableHash.insertKey(name, len, 0, versions);
+ }
+
+ int waitTime = 100;
+
+ bool retreive = false;
+ while(versions->size() > 0 && !retreive){
+ TableVersion * ver = & versions->back();
+ switch(ver->m_status){
+ case OK:
+ ver->m_refCount++;
+ return ver->m_impl;
+ case DROPPED:
+ retreive = true; // Break loop
+ break;
+ case RETREIVING:
+ NdbCondition_WaitTimeout(m_waitForTableCondition, m_mutex, waitTime);
+ continue;
+ }
+ }
+
+ /**
+ * Create new...
+ */
+ TableVersion tmp;
+ tmp.m_version = 0;
+ tmp.m_impl = 0;
+ tmp.m_status = RETREIVING;
+ tmp.m_refCount = 1; // The one retreiving it
+ versions->push_back(tmp);
+ return 0;
+}
+
+NdbTableImpl *
+GlobalDictCache::put(const char * name, NdbTableImpl * tab)
+{
+ const Uint32 len = strlen(name);
+ Vector<TableVersion> * vers = m_tableHash.getData(name, len);
+ if(vers == 0){
+ // Should always tried to retreive it first
+ // and then there should be a record
+ abort();
+ }
+
+ const Uint32 sz = vers->size();
+ if(sz == 0){
+ // Should always tried to retreive it first
+ // and then there should be a record
+ abort();
+ }
+
+ TableVersion & ver = vers->back();
+ if(ver.m_status != RETREIVING ||
+ ver.m_impl != 0 ||
+ ver.m_version != 0 ||
+ ver.m_refCount == 0){
+ abort();
+ }
+
+ if(tab == 0){
+ // No table found in db
+ vers->erase(sz - 1);
+ } else {
+ ver.m_impl = tab;
+ ver.m_version = tab->m_version;
+ ver.m_status = OK;
+ }
+
+ NdbCondition_Broadcast(m_waitForTableCondition);
+ return tab;
+}
+
+void
+GlobalDictCache::drop(NdbTableImpl * tab)
+{
+ const Uint32 len = strlen(tab->m_internalName.c_str());
+ Vector<TableVersion> * vers =
+ m_tableHash.getData(tab->m_internalName.c_str(), len);
+ if(vers == 0){
+ // Should always tried to retreive it first
+ // and then there should be a record
+ abort();
+ }
+
+ const Uint32 sz = vers->size();
+ if(sz == 0){
+ // Should always tried to retreive it first
+ // and then there should be a record
+ abort();
+ }
+
+ for(unsigned i = 0; i < sz; i++){
+ TableVersion & ver = (* vers)[i];
+ if(ver.m_impl == tab){
+ if(ver.m_refCount == 0 || ver.m_status == RETREIVING ||
+ ver.m_version != tab->m_version){
+ ndbout_c("Dropping with refCount=%d status=%d impl=%p",
+ ver.m_refCount, ver.m_status, ver.m_impl);
+ break;
+ }
+
+ ver.m_refCount--;
+ ver.m_status = DROPPED;
+ if(ver.m_refCount == 0){
+ delete ver.m_impl;
+ vers->erase(i);
+ }
+ return;
+ }
+ }
+
+ for(unsigned i = 0; i<sz; i++){
+ TableVersion & ver = (* vers)[i];
+ ndbout_c("%d: version: %d refCount: %d status: %d impl: %p",
+ i, ver.m_version, ver.m_refCount, ver.m_status, ver.m_impl);
+ }
+
+ abort();
+}
+
+void
+GlobalDictCache::release(NdbTableImpl * tab){
+ const Uint32 len = strlen(tab->m_internalName.c_str());
+ Vector<TableVersion> * vers =
+ m_tableHash.getData(tab->m_internalName.c_str(), len);
+ if(vers == 0){
+ // Should always tried to retreive it first
+ // and then there should be a record
+ abort();
+ }
+
+ const Uint32 sz = vers->size();
+ if(sz == 0){
+ // Should always tried to retreive it first
+ // and then there should be a record
+ abort();
+ }
+
+ for(unsigned i = 0; i < sz; i++){
+ TableVersion & ver = (* vers)[i];
+ if(ver.m_impl == tab){
+ if(ver.m_refCount == 0 || ver.m_status == RETREIVING ||
+ ver.m_version != tab->m_version){
+ ndbout_c("Releasing with refCount=%d status=%d impl=%p",
+ ver.m_refCount, ver.m_status, ver.m_impl);
+ break;
+ }
+
+ ver.m_refCount--;
+ return;
+ }
+ }
+
+ for(unsigned i = 0; i<sz; i++){
+ TableVersion & ver = (* vers)[i];
+ ndbout_c("%d: version: %d refCount: %d status: %d impl: %p",
+ i, ver.m_version, ver.m_refCount, ver.m_status, ver.m_impl);
+ }
+
+ abort();
+}
+
diff --git a/ndb/src/ndbapi/DictCache.hpp b/ndb/src/ndbapi/DictCache.hpp
new file mode 100644
index 00000000000..e59793bbc09
--- /dev/null
+++ b/ndb/src/ndbapi/DictCache.hpp
@@ -0,0 +1,79 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DictCache_H
+#define DictCache_H
+
+#include <ndb_types.h>
+#include <kernel_types.h>
+#include <NdbError.hpp>
+#include <BaseString.hpp>
+#include <Vector.hpp>
+#include <UtilBuffer.hpp>
+#include <NdbDictionary.hpp>
+#include <Ndb.hpp>
+#include "NdbLinHash.hpp"
+
+/**
+ * A non thread safe dict cache
+ */
+class LocalDictCache {
+public:
+ LocalDictCache();
+ ~LocalDictCache();
+
+ NdbTableImpl * get(const char * name);
+
+ void put(const char * name, NdbTableImpl *);
+ void drop(const char * name);
+
+ NdbLinHash<NdbTableImpl> m_tableHash; // On name
+};
+
+/**
+ * A thread safe dict cache
+ */
+class GlobalDictCache : public NdbLockable {
+public:
+ GlobalDictCache();
+ ~GlobalDictCache();
+
+ NdbTableImpl * get(const char * name);
+
+ NdbTableImpl* put(const char * name, NdbTableImpl *);
+ void drop(NdbTableImpl *);
+ void release(NdbTableImpl *);
+private:
+ enum Status {
+ OK = 0,
+ DROPPED = 1,
+ RETREIVING = 2
+ };
+
+ struct TableVersion {
+ Uint32 m_version;
+ Uint32 m_refCount;
+ NdbTableImpl * m_impl;
+ Status m_status;
+ };
+
+ NdbLinHash<Vector<TableVersion> > m_tableHash;
+ NdbCondition * m_waitForTableCondition;
+};
+
+#endif
+
+
diff --git a/ndb/src/ndbapi/Makefile b/ndb/src/ndbapi/Makefile
new file mode 100644
index 00000000000..932fbd844d2
--- /dev/null
+++ b/ndb/src/ndbapi/Makefile
@@ -0,0 +1,61 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+PIC_ARCHIVE := Y
+NONPIC_ARCHIVE := Y
+ARCHIVE_TARGET := ndbapi
+
+A_LIB := Y
+SO_LIB := Y
+PIC_LIB := Y
+LIB_TARGET := NDB_API
+
+LIB_TARGET_ARCHIVES := $(ARCHIVE_TARGET) \
+ transporter \
+ general \
+ signaldataprint \
+ mgmsrvcommon \
+ portlib \
+ logger \
+ trace
+
+DIRS := signal-sender
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = \
+ TransporterFacade.cpp \
+ ClusterMgr.cpp \
+ Ndb.cpp \
+ NdbPoolImpl.cpp NdbPool.cpp \
+ Ndblist.cpp \
+ Ndbif.cpp \
+ Ndbinit.cpp \
+ Ndberror.cpp \
+ NdbConnection.cpp \
+ NdbConnectionScan.cpp \
+ NdbOperation.cpp \
+ NdbOperationSearch.cpp \
+ NdbOperationScan.cpp \
+ NdbOperationInt.cpp \
+ NdbOperationDefine.cpp \
+ NdbOperationExec.cpp \
+ NdbScanReceiver.cpp \
+ NdbResultSet.cpp \
+ NdbCursorOperation.cpp \
+ NdbScanOperation.cpp NdbScanFilter.cpp \
+ NdbIndexOperation.cpp \
+ NdbEventOperation.cpp \
+ NdbEventOperationImpl.cpp \
+ NdbApiSignal.cpp \
+ NdbRecAttr.cpp \
+ NdbSchemaCon.cpp \
+ NdbSchemaOp.cpp \
+ NdbUtil.cpp \
+ NdbReceiver.cpp \
+ NdbDictionary.cpp NdbDictionaryImpl.cpp DictCache.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+###
+# Backward compatible
diff --git a/ndb/src/ndbapi/Ndb.cpp b/ndb/src/ndbapi/Ndb.cpp
new file mode 100644
index 00000000000..d7930f32d72
--- /dev/null
+++ b/ndb/src/ndbapi/Ndb.cpp
@@ -0,0 +1,1285 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+
+/*****************************************************************************
+Name: Ndb.cpp
+******************************************************************************/
+
+#include "NdbApiSignal.hpp"
+#include "NdbImpl.hpp"
+#include "NdbSchemaOp.hpp"
+#include "NdbSchemaCon.hpp"
+#include "NdbOperation.hpp"
+#include "NdbConnection.hpp"
+#include "NdbEventOperation.hpp"
+#include "NdbRecAttr.hpp"
+#include <md5_hash.hpp>
+#include <NdbSleep.h>
+#include <NdbOut.hpp>
+#include <ndb_limits.h>
+#include "API.hpp"
+#include <NdbEnv.h>
+#include <BaseString.hpp>
+
+static bool fullyQualifiedNames = true;
+
+/****************************************************************************
+void connect();
+
+Connect to any node which has no connection at the moment.
+****************************************************************************/
+NdbConnection* Ndb::doConnect(Uint32 tConNode)
+{
+ Uint32 tNode;
+ Uint32 i = 0;;
+ Uint32 tAnyAlive = 0;
+ int TretCode;
+
+ if (tConNode != 0) {
+ TretCode = NDB_connect(tConNode);
+ if ((TretCode == 1) || (TretCode == 2)) {
+//****************************************************************************
+// We have connections now to the desired node. Return
+//****************************************************************************
+ return getConnectedNdbConnection(tConNode);
+ } else if (TretCode != 0) {
+ tAnyAlive = 1;
+ }//if
+ }//if
+//****************************************************************************
+// We will connect to any node. Make sure that we have connections to all
+// nodes.
+//****************************************************************************
+ Uint32 tNoOfDbNodes = theNoOfDBnodes;
+ i = theCurrentConnectIndex;
+ UintR Tcount = 0;
+ do {
+ if (i >= tNoOfDbNodes) {
+ i = 0;
+ }//if
+ Tcount++;
+ tNode = theDBnodes[i];
+ TretCode = NDB_connect(tNode);
+ if ((TretCode == 1) || (TretCode == 2)) {
+//****************************************************************************
+// We have connections now to the desired node. Return
+//****************************************************************************
+ if (theCurrentConnectIndex == i) {
+ theCurrentConnectCounter++;
+ if (theCurrentConnectCounter == 8) {
+ theCurrentConnectCounter = 1;
+ theCurrentConnectIndex++;
+ }//if
+ } else {
+ // Set to 2 because we have already connected to a node
+ // when we get here.
+ theCurrentConnectCounter = 2;
+ theCurrentConnectIndex = i;
+ }//if
+ return getConnectedNdbConnection(tNode);
+ } else if (TretCode != 0) {
+ tAnyAlive = 1;
+ }//if
+ i++;
+ } while (Tcount < tNoOfDbNodes);
+//****************************************************************************
+// We were unable to find a free connection. If no node alive we will report
+// error code for cluster failure otherwise connection failure.
+//****************************************************************************
+ if (tAnyAlive == 1) {
+#ifdef VM_TRACE
+ ndbout << "TretCode = " << TretCode << endl;
+#endif
+ theError.code = 4006;
+ } else {
+ theError.code = 4009;
+ }//if
+ return NULL;
+}
+
+int
+Ndb::NDB_connect(Uint32 tNode)
+{
+//****************************************************************************
+// We will perform seize of a transaction record in DBTC in the specified node.
+//***************************************************************************
+
+ int tReturnCode;
+ TransporterFacade *tp = TransporterFacade::instance();
+
+ bool nodeAvail = tp->get_node_alive(tNode);
+ if(nodeAvail == false){
+ return 0;
+ }
+
+ NdbConnection * tConArray = theConnectionArray[tNode];
+ if (tConArray != NULL) {
+ return 2;
+ }
+
+ NdbConnection * tNdbCon = getNdbCon(); // Get free connection object.
+ if (tNdbCon == NULL) {
+ return 4;
+ }//if
+ NdbApiSignal* tSignal = getSignal(); // Get signal object
+ if (tSignal == NULL) {
+ releaseNdbCon(tNdbCon);
+ return 4;
+ }//if
+ if (tSignal->setSignal(GSN_TCSEIZEREQ) == -1) {
+ releaseNdbCon(tNdbCon);
+ releaseSignal(tSignal);
+ return 4;
+ }//if
+ tSignal->setData(tNdbCon->ptr2int(), 1);
+//************************************************
+// Set connection pointer as NdbConnection object
+//************************************************
+ tSignal->setData(theMyRef, 2); // Set my block reference
+ tNdbCon->Status(Connecting); // Set status to connecting
+ Uint32 nodeSequence;
+ { // send and receive signal
+ tp->lock_mutex();
+ nodeSequence = tp->getNodeSequence(tNode);
+ bool node_is_alive = tp->get_node_alive(tNode);
+ if (node_is_alive) {
+ tReturnCode = tp->sendSignal(tSignal, tNode);
+ releaseSignal(tSignal);
+ if (tReturnCode == -1) {
+ tp->unlock_mutex();
+ } else {
+ theWaiter.m_node = tNode;
+ theWaiter.m_state = WAIT_TC_SEIZE;
+ tReturnCode = receiveResponse();
+ }//if
+ } else {
+ releaseSignal(tSignal);
+ tp->unlock_mutex();
+ tReturnCode = -1;
+ }//if
+ }
+
+ if ((tReturnCode == 0) && (tNdbCon->Status() == Connected)) {
+ //************************************************
+ // Send and receive was successful
+ //************************************************
+ NdbConnection* tPrevFirst = theConnectionArray[tNode];
+ tNdbCon->setConnectedNodeId(tNode, nodeSequence);
+
+ tNdbCon->setMyBlockReference(theMyRef);
+ theConnectionArray[tNode] = tNdbCon;
+ tNdbCon->theNext = tPrevFirst;
+ return 1;
+ } else {
+ releaseNdbCon(tNdbCon);
+//****************************************************************************
+// Unsuccessful connect is indicated by 3.
+//****************************************************************************
+ return 3;
+ }//if
+}//Ndb::NDB_connect()
+
+NdbConnection *
+Ndb::getConnectedNdbConnection(Uint32 nodeId){
+ NdbConnection* next = theConnectionArray[nodeId];
+ theConnectionArray[nodeId] = next->theNext;
+ next->theNext = NULL;
+
+ return next;
+}//Ndb::getConnectedNdbConnection()
+
+/*****************************************************************************
+disconnect();
+
+Remark: Disconnect all connections to the database.
+*****************************************************************************/
+void
+Ndb::doDisconnect()
+{
+ NdbConnection* tNdbCon;
+ CHECK_STATUS_MACRO_VOID;
+
+ Uint32 tNoOfDbNodes = theNoOfDBnodes;
+ UintR i;
+ for (i = 0; i < tNoOfDbNodes; i++) {
+ Uint32 tNode = theDBnodes[i];
+ tNdbCon = theConnectionArray[tNode];
+ while (tNdbCon != NULL) {
+ NdbConnection* tmpNdbCon = tNdbCon;
+ tNdbCon = tNdbCon->theNext;
+ releaseConnectToNdb(tmpNdbCon);
+ }//while
+ }//for
+ tNdbCon = theTransactionList;
+ while (tNdbCon != NULL) {
+ NdbConnection* tmpNdbCon = tNdbCon;
+ tNdbCon = tNdbCon->theNext;
+ releaseConnectToNdb(tmpNdbCon);
+ }//while
+}//Ndb::disconnect()
+
+/*****************************************************************************
+int waitUntilReady(int timeout);
+
+Return Value: Returns 0 if the Ndb is ready within timeout seconds.
+ Returns -1 otherwise.
+Remark: Waits until a node has status != 0
+*****************************************************************************/
+int
+Ndb::waitUntilReady(int timeout)
+{
+ int secondsCounter = 0;
+ int milliCounter = 0;
+ int noChecksSinceFirstAliveFound = 0;
+
+ if (theInitState != Initialised) {
+ // Ndb::init is not called
+ theError.code = 4256;
+ return -1;
+ }
+
+ do {
+ unsigned int foundAliveNode = 0;
+ TransporterFacade *tp = TransporterFacade::instance();
+ tp->lock_mutex();
+ for (unsigned int i = 0; i < theNoOfDBnodes; i++) {
+ const NodeId nodeId = theDBnodes[i];
+ //************************************************
+ // If any node is answering, ndb is answering
+ //************************************************
+ if (tp->get_node_alive(nodeId) != 0) {
+ foundAliveNode++;
+ }//if
+ }//for
+
+ tp->unlock_mutex();
+ if (foundAliveNode == theNoOfDBnodes) {
+ return 0;
+ }//if
+ if (foundAliveNode > 0) {
+ noChecksSinceFirstAliveFound++;
+ }//if
+ if (noChecksSinceFirstAliveFound > 30) {
+ return 0;
+ }//if
+ NdbSleep_MilliSleep(100);
+ milliCounter += 100;
+ if (milliCounter >= 1000) {
+ secondsCounter++;
+ milliCounter = 0;
+ }//if
+ } while ( secondsCounter < timeout );
+ if (noChecksSinceFirstAliveFound > 0) {
+ return 0;
+ }//if
+ return -1;
+}
+
+/*****************************************************************************
+NdbConnection* startTransaction();
+
+Return Value: Returns a pointer to a connection object.
+ Return NULL otherwise.
+Remark: Start transaction. Synchronous.
+*****************************************************************************/
+NdbConnection*
+Ndb::startTransaction(Uint32 aPriority, const char * keyData, Uint32 keyLen)
+{
+
+ if (theInitState == Initialised) {
+ theError.code = 0;
+ checkFailedNode();
+ /**
+ * If the user supplied key data
+ * We will make a qualified quess to which node is the primary for the
+ * the fragment and contact that node
+ */
+ Uint32 nodeId;
+ if(keyData != 0) {
+ Uint32 fragmentId = computeFragmentId(keyData, keyLen);
+ nodeId = guessPrimaryNode(fragmentId);
+ } else {
+ nodeId = 0;
+ }//if
+ return startTransactionLocal(aPriority, nodeId);
+ } else {
+ return NULL;
+ }//if
+}//Ndb::startTransaction()
+
+/*****************************************************************************
+NdbConnection* hupp(NdbConnection* pBuddyTrans);
+
+Return Value: Returns a pointer to a connection object.
+ Connected to the same node as pBuddyTrans
+ and also using the same transction id
+Remark: Start transaction. Synchronous.
+*****************************************************************************/
+NdbConnection*
+Ndb::hupp(NdbConnection* pBuddyTrans)
+{
+ Uint32 aPriority = 0;
+ if (pBuddyTrans == NULL){
+ return startTransaction();
+ }
+
+ if (theInitState == Initialised) {
+ theError.code = 0;
+ checkFailedNode();
+
+ Uint32 nodeId = pBuddyTrans->getConnectedNodeId();
+ NdbConnection* pCon = startTransactionLocal(aPriority, nodeId);
+ if(pCon == NULL)
+ return NULL;
+
+ if (pCon->getConnectedNodeId() != nodeId){
+ // We could not get a connection to the desired node
+ // release the connection and return NULL
+ closeTransaction(pCon);
+ return NULL;
+ }
+ pCon->setTransactionId(pBuddyTrans->getTransactionId());
+ pCon->setBuddyConPtr((Uint32)pBuddyTrans->getTC_ConnectPtr());
+ return pCon;
+ } else {
+ return NULL;
+ }//if
+}//Ndb::hupp()
+
+NdbConnection*
+Ndb::startTransactionDGroup(Uint32 aPriority, const char * keyData, int type)
+{
+
+ char DGroup[4];
+ if ((keyData == NULL) ||
+ (type > 1)) {
+ theError.code = 4118;
+ return NULL;
+ }//if
+ if (theInitState == Initialised) {
+ theError.code = 0;
+ checkFailedNode();
+ /**
+ * If the user supplied key data
+ * We will make a qualified quess to which node is the primary for the
+ * the fragment and contact that node
+ */
+ Uint32 fragmentId;
+ if (type == 0) {
+ DGroup[0] = keyData[0];
+ DGroup[1] = keyData[1];
+ DGroup[2] = 0x30;
+ DGroup[3] = 0x30;
+ fragmentId = computeFragmentId(&DGroup[0], 4);
+ } else {
+ Uint32 hashValue = ((keyData[0] - 0x30) * 10) + (keyData[1] - 0x30);
+ fragmentId = getFragmentId(hashValue);
+ }//if
+ Uint32 nodeId = guessPrimaryNode(fragmentId);
+ return startTransactionLocal(aPriority, nodeId);
+ } else {
+ return NULL;
+ }//if
+}//Ndb::startTransaction()
+
+NdbConnection*
+Ndb::startTransactionLocal(Uint32 aPriority, Uint32 nodeId)
+{
+#ifdef VM_TRACE
+ char buf[255];
+ const char* val = NdbEnv_GetEnv("NDB_TRANSACTION_NODE_ID", buf, 255);
+ if(val != 0){
+ nodeId = atoi(val);
+ }
+#endif
+
+ NdbConnection* tConnection;
+ Uint64 tFirstTransId = theFirstTransId;
+ tConnection = doConnect(nodeId);
+ if (tConnection == NULL) {
+ return NULL;
+ }//if
+ NdbConnection* tConNext = theTransactionList;
+ tConnection->init();
+ theTransactionList = tConnection; // into a transaction list.
+ tConnection->next(tConNext); // Add the active connection object
+ tConnection->setTransactionId(tFirstTransId);
+ tConnection->thePriority = aPriority;
+ if ((tFirstTransId & 0xFFFFFFFF) == 0xFFFFFFFF) {
+ //---------------------------------------------------
+// Transaction id rolling round. We will start from
+// consecutive identity 0 again.
+//---------------------------------------------------
+ theFirstTransId = ((tFirstTransId >> 32) << 32);
+ } else {
+ theFirstTransId = tFirstTransId + 1;
+ }//if
+#ifdef VM_TRACE
+ if (tConnection->theListState != NotInList) {
+ printState("startTransactionLocal %x", tConnection);
+ abort();
+ }
+#endif
+ return tConnection;
+}//Ndb::startTransactionLocal()
+
+/*****************************************************************************
+void closeTransaction(NdbConnection* aConnection);
+
+Parameters: aConnection: the connection used in the transaction.
+Remark: Close transaction by releasing the connection and all operations.
+*****************************************************************************/
+void
+Ndb::closeTransaction(NdbConnection* aConnection)
+{
+ NdbConnection* tCon;
+ NdbConnection* tPreviousCon;
+
+ if (aConnection == NULL) {
+//-----------------------------------------------------
+// closeTransaction called on NULL pointer, destructive
+// application behaviour.
+//-----------------------------------------------------
+#ifdef VM_TRACE
+ printf("NULL into closeTransaction\n");
+#endif
+ return;
+ }//if
+ CHECK_STATUS_MACRO_VOID;
+
+ tCon = theTransactionList;
+
+ if (aConnection == tCon) { // Remove the active connection object
+ theTransactionList = tCon->next(); // from the transaction list.
+ } else {
+ while (aConnection != tCon) {
+ if (tCon == NULL) {
+//-----------------------------------------------------
+// closeTransaction called on non-existing transaction
+//-----------------------------------------------------
+#ifdef VM_TRACE
+ printf("Non-existing transaction into closeTransaction\n");
+ abort();
+#endif
+ return;
+ }//if
+ tPreviousCon = tCon;
+ tCon = tCon->next();
+ }//while
+ tPreviousCon->next(tCon->next());
+ }//if
+
+ aConnection->release();
+
+ if(aConnection->theError.code == 4008){
+ /**
+ * When a SCAN timed-out, returning the NdbConnection leads
+ * to reuse. And TC crashes when the API tries to reuse it to
+ * something else...
+ */
+#ifdef VM_TRACE
+ printf("Scan timeout:ed NdbConnection-> not returning it-> memory leak\n");
+#endif
+ return;
+ }
+
+ if(aConnection->theError.code == 4008){
+ /**
+ * Something timed-out, returning the NdbConnection leads
+ * to reuse. And TC crashes when the API tries to reuse it to
+ * something else...
+ */
+#ifdef VM_TRACE
+ printf("Con timeout:ed NdbConnection-> not returning it-> memory leak\n");
+#endif
+ return;
+ }
+
+ if (aConnection->theReleaseOnClose == false) {
+ /**
+ * Put it back in idle list for that node
+ */
+ Uint32 nodeId = aConnection->getConnectedNodeId();
+ aConnection->theNext = theConnectionArray[nodeId];
+ theConnectionArray[nodeId] = aConnection;
+ return;
+ } else {
+ aConnection->theReleaseOnClose = false;
+ releaseNdbCon(aConnection);
+ }//if
+}//Ndb::closeTransaction()
+
+/*****************************************************************************
+int* NdbTamper(int aAction, int aNode);
+
+Parameters: aAction Specifies what action to be taken
+ 1: Lock global checkpointing Can only be sent to master DIH, Parameter aNode ignored.
+ 2: UnLock global checkpointing Can only be sent to master DIH, Parameter aNode ignored.
+ 3: Crash node
+
+ aNode Specifies which node the action will be taken
+ -1: Master DIH
+ 0-16: Nodnumber
+
+Return Value: -1 Error .
+
+Remark: Sends a signal to DIH.
+*****************************************************************************/
+int
+Ndb::NdbTamper(TamperType aAction, int aNode)
+{
+ NdbConnection* tNdbConn;
+ NdbApiSignal tSignal(theMyRef);
+ int tNode;
+ int tAction;
+ int ret_code;
+
+#ifdef CUSTOMER_RELEASE
+ return -1;
+#else
+ CHECK_STATUS_MACRO;
+ checkFailedNode();
+
+ theRestartGCI = 0;
+ switch (aAction) {
+// Translate enum to integer. This is done because the SCI layer
+// expects integers.
+ case LockGlbChp:
+ tAction = 1;
+ break;
+ case UnlockGlbChp:
+ tAction = 2;
+ break;
+ case CrashNode:
+ tAction = 3;
+ break;
+ case ReadRestartGCI:
+ tAction = 4;
+ break;
+ default:
+ theError.code = 4102;
+ return -1;
+ }
+
+ tNdbConn = getNdbCon(); // Get free connection object
+ if (tNdbConn == NULL) {
+ theError.code = 4000;
+ return -1;
+ }
+ tSignal.setSignal(GSN_DIHNDBTAMPER);
+ tSignal.setData (tAction, 1);
+ tSignal.setData(tNdbConn->ptr2int(),2);
+ tSignal.setData(theMyRef,3); // Set return block reference
+ tNdbConn->Status(Connecting); // Set status to connecting
+ TransporterFacade *tp = TransporterFacade::instance();
+ if (tAction == 3) {
+ tp->lock_mutex();
+ tp->sendSignal(&tSignal, aNode);
+ tp->unlock_mutex();
+ releaseNdbCon(tNdbConn);
+ } else if ( (tAction == 2) || (tAction == 1) ) {
+ tp->lock_mutex();
+ tNode = tp->get_an_alive_node();
+ if (tNode == 0) {
+ theError.code = 4002;
+ releaseNdbCon(tNdbConn);
+ return -1;
+ }//if
+ ret_code = tp->sendSignal(&tSignal,aNode);
+ tp->unlock_mutex();
+ releaseNdbCon(tNdbConn);
+ return ret_code;
+ } else {
+ do {
+ tp->lock_mutex();
+ // Start protected area
+ tNode = tp->get_an_alive_node();
+ tp->unlock_mutex();
+ // End protected area
+ if (tNode == 0) {
+ theError.code = 4009;
+ releaseNdbCon(tNdbConn);
+ return -1;
+ }//if
+ ret_code = sendRecSignal(tNode, WAIT_NDB_TAMPER, &tSignal, 0);
+ if (ret_code == 0) {
+ if (tNdbConn->Status() != Connected) {
+ theRestartGCI = 0;
+ }//if
+ releaseNdbCon(tNdbConn);
+ return theRestartGCI;
+ } else if ((ret_code == -5) || (ret_code == -2)) {
+ TRACE_DEBUG("Continue DIHNDBTAMPER when node failed/stopping");
+ } else {
+ return -1;
+ }//if
+ } while (1);
+ }
+ return 0;
+#endif
+}
+/****************************************************************************
+NdbSchemaCon* startSchemaTransaction();
+
+Return Value: Returns a pointer to a schema connection object.
+ Return NULL otherwise.
+Remark: Start schema transaction. Synchronous.
+****************************************************************************/
+NdbSchemaCon*
+Ndb::startSchemaTransaction()
+{
+ NdbSchemaCon* tSchemaCon;
+ if (theSchemaConToNdbList != NULL) {
+ theError.code = 4321;
+ return NULL;
+ }//if
+ tSchemaCon = new NdbSchemaCon(this);
+ if (tSchemaCon == NULL) {
+ theError.code = 4000;
+ return NULL;
+ }//if
+ theSchemaConToNdbList = tSchemaCon;
+ return tSchemaCon;
+}
+/*****************************************************************************
+void closeSchemaTransaction(NdbSchemaCon* aSchemaCon);
+
+Parameters: aSchemaCon: the schemacon used in the transaction.
+Remark: Close transaction by releasing the schemacon and all schemaop.
+*****************************************************************************/
+void
+Ndb::closeSchemaTransaction(NdbSchemaCon* aSchemaCon)
+{
+ if (theSchemaConToNdbList != aSchemaCon) {
+ abort();
+ return;
+ }//if
+ aSchemaCon->release();
+ delete aSchemaCon;
+ theSchemaConToNdbList = NULL;
+ return;
+}//Ndb::closeSchemaTransaction()
+
+
+/*****************************************************************************
+void RestartGCI(int aRestartGCI);
+
+Remark: Set theRestartGCI on the NDB object
+*****************************************************************************/
+void
+Ndb::RestartGCI(int aRestartGCI)
+{
+ theRestartGCI = aRestartGCI;
+}
+
+/****************************************************************************
+int getBlockNumber(void);
+
+Remark:
+****************************************************************************/
+int
+Ndb::getBlockNumber()
+{
+ return theNdbBlockNumber;
+}
+
+NdbDictionary::Dictionary *
+Ndb::getDictionary() const {
+ return theDictionary;
+}
+
+/****************************************************************************
+int getNodeId();
+
+Remark:
+****************************************************************************/
+int
+Ndb::getNodeId()
+{
+ return theNode;
+}
+
+/****************************************************************************
+Uint64 getTupleIdFromNdb( Uint32 aTableId );
+
+Parameters: aTableId : The TableId.
+Remark: Returns a new TupleId to the application.
+ The TupleId comes from SYSTAB_0 where SYSKEY_0 = TableId.
+ It is initialized to (TableId << 48) + 1 in NdbcntrMain.cpp.
+****************************************************************************/
+#define DEBUG_TRACE(msg) \
+// ndbout << __FILE__ << " line: " << __LINE__ << " msg: " << msg << endl
+
+Uint64
+Ndb::getAutoIncrementValue(const char* aTableName, Uint32 cacheSize)
+{
+ DEBUG_TRACE("getAutoIncrementValue");
+ const NdbTableImpl* table = theDictionary->getTable(aTableName);
+ if (table == 0)
+ return ~0;
+ Uint64 tupleId = getTupleIdFromNdb(table->m_tableId, cacheSize);
+ return tupleId;
+}
+
+Uint64
+Ndb::getTupleIdFromNdb(const char* aTableName, Uint32 cacheSize )
+{
+ const NdbTableImpl* table = theDictionary->getTable(aTableName);
+ if (table == 0)
+ return ~0;
+ return getTupleIdFromNdb(table->m_tableId, cacheSize);
+}
+
+Uint64
+Ndb::getTupleIdFromNdb(Uint32 aTableId, Uint32 cacheSize )
+{
+ if ( theFirstTupleId[aTableId] != theLastTupleId[aTableId] )
+ {
+ theFirstTupleId[aTableId]++;
+ return theFirstTupleId[aTableId];
+ }
+ else // theFirstTupleId == theLastTupleId
+ {
+ return opTupleIdOnNdb(aTableId, cacheSize, 0);
+ }
+}
+
+bool
+Ndb::setAutoIncrementValue(const char* aTableName, Uint64 val)
+{
+ DEBUG_TRACE("setAutoIncrementValue " << val);
+ const NdbTableImpl* table = theDictionary->getTable(aTableName);
+ if (table == 0)
+ return false;
+ return setTupleIdInNdb(table->m_tableId, val);
+}
+
+bool
+Ndb::setTupleIdInNdb(const char* aTableName, Uint64 val )
+{
+ DEBUG_TRACE("setTupleIdInNdb");
+ const NdbTableImpl* table = theDictionary->getTable(aTableName);
+ if (table == 0)
+ return false;
+ return setTupleIdInNdb(table->m_tableId, val);
+}
+
+bool
+Ndb::setTupleIdInNdb(Uint32 aTableId, Uint64 val )
+{
+ DEBUG_TRACE("setTupleIdInNdb");
+ return (opTupleIdOnNdb(aTableId, val, 1) == val);
+}
+
+Uint64
+Ndb::opTupleIdOnNdb(Uint32 aTableId, Uint64 opValue, Uint32 op)
+{
+ DEBUG_TRACE("opTupleIdOnNdb");
+
+ NdbConnection* tConnection;
+ NdbOperation* tOperation;
+ Uint64 tValue;
+ NdbRecAttr* tRecAttrResult;
+
+ Uint64 ret;
+
+ CHECK_STATUS_MACRO_ZERO;
+
+ BaseString currentDb(getDatabaseName());
+ BaseString currentSchema(getDatabaseSchemaName());
+
+ setDatabaseName("sys");
+ setDatabaseSchemaName("def");
+ tConnection = this->startTransaction();
+ if (tConnection == NULL)
+ goto error_return;
+
+ if (usingFullyQualifiedNames())
+ tOperation = tConnection->getNdbOperation("SYSTAB_0");
+ else
+ tOperation = tConnection->getNdbOperation("sys/def/SYSTAB_0");
+ if (tOperation == NULL)
+ goto error_handler;
+
+ switch (op)
+ {
+ case 0:
+ tOperation->interpretedUpdateTuple();
+ tOperation->equal("SYSKEY_0", aTableId );
+ {
+#ifdef NDB_SOLARIS
+ Uint64 cacheSize64 = opValue; // XXX interpreter bug on Uint32
+ tOperation->incValue("NEXTID", cacheSize64);
+#else
+ Uint32 cacheSize32 = opValue; // XXX for little-endian
+ tOperation->incValue("NEXTID", cacheSize32);
+#endif
+ }
+ tRecAttrResult = tOperation->getValue("NEXTID");
+
+ if (tConnection->execute( Commit ) == -1 )
+ goto error_handler;
+
+ tValue = tRecAttrResult->u_64_value();
+
+ theFirstTupleId[aTableId] = tValue - opValue;
+ theLastTupleId[aTableId] = tValue - 1;
+ ret = theFirstTupleId[aTableId];
+ break;
+ case 1:
+ tOperation->updateTuple();
+ tOperation->equal("SYSKEY_0", aTableId );
+ tOperation->setValue("NEXTID", opValue);
+
+ if (tConnection->execute( Commit ) == -1 )
+ goto error_handler;
+
+ theFirstTupleId[aTableId] = ~0;
+ theLastTupleId[aTableId] = ~0;
+ ret = opValue;
+ break;
+ default:
+ goto error_handler;
+ }
+
+ this->closeTransaction(tConnection);
+
+ // Restore current name space
+ setDatabaseName(currentDb.c_str());
+ setDatabaseSchemaName(currentSchema.c_str());
+
+ return ret;
+
+ error_handler:
+ theError.code = tConnection->theError.code;
+ this->closeTransaction(tConnection);
+ error_return:
+ // Restore current name space
+ setDatabaseName(currentDb.c_str());
+ setDatabaseSchemaName(currentSchema.c_str());
+
+ return ~0;
+}
+
+static const Uint32 MAX_KEY_LEN_64_WORDS = 4;
+static const Uint32 MAX_KEY_LEN_32_WORDS = 8;
+static const Uint32 MAX_KEY_LEN_BYTES = 32;
+
+Uint32
+Ndb::computeFragmentId(const char * keyData, Uint32 keyLen)
+{
+ Uint64 tempData[MAX_KEY_LEN_64_WORDS];
+
+ const Uint32 usedKeyLen = (keyLen + 3) >> 2; // In words
+ const char * usedKeyData = 0;
+
+ /**
+ * If key data buffer is not aligned (on 64 bit boundary)
+ * or key len is not a multiple of 4
+ * Use temp data
+ */
+ if(((((UintPtr)keyData) & 7) == 0) && ((keyLen & 3) == 0)) {
+ usedKeyData = keyData;
+ } else {
+ memcpy(&tempData[0], keyData, keyLen);
+ const int slack = keyLen & 3;
+ if(slack > 0) {
+ memset(&((char *)&tempData[0])[keyLen], 0, (4 - slack));
+ }//if
+ usedKeyData = (char *)&tempData[0];
+ }//if
+
+ Uint32 hashValue = md5_hash((Uint64 *)usedKeyData, usedKeyLen);
+
+ hashValue >>= startTransactionNodeSelectionData.kValue;
+ return getFragmentId(hashValue);
+}//Ndb::computeFragmentId()
+
+Uint32
+Ndb::getFragmentId(Uint32 hashValue)
+{
+ Uint32 fragmentId = hashValue &
+ startTransactionNodeSelectionData.hashValueMask;
+ if(fragmentId < startTransactionNodeSelectionData.hashpointerValue) {
+ fragmentId = hashValue &
+ ((startTransactionNodeSelectionData.hashValueMask << 1) + 1);
+ }//if
+ return fragmentId;
+}
+
+Uint32
+Ndb::guessPrimaryNode(Uint32 fragmentId){
+ //ASSERT(((fragmentId > 0) && fragmentId <
+ // startTransactionNodeSelectionData.noOfFragments), "Invalid fragementId");
+
+ return startTransactionNodeSelectionData.fragment2PrimaryNodeMap[fragmentId];
+}
+
+void
+Ndb::StartTransactionNodeSelectionData::init(Uint32 noOfNodes,
+ Uint32 nodeIds[]) {
+ kValue = 6;
+ noOfFragments = 2 * noOfNodes;
+
+ /**
+ * Compute hashValueMask and hashpointerValue
+ */
+ {
+ Uint32 topBit = (1 << 31);
+ for(int i = 31; i>=0; i--){
+ if((noOfFragments & topBit) != 0)
+ break;
+ topBit >>= 1;
+ }
+ hashValueMask = topBit - 1;
+ hashpointerValue = noOfFragments - (hashValueMask + 1);
+ }
+
+ /**
+ * This initialization depends on
+ * the fact that:
+ * primary node for fragment i = i % noOfNodes
+ *
+ * This algorithm should be implemented in Dbdih
+ */
+ {
+ fragment2PrimaryNodeMap = new Uint32[noOfFragments];
+
+ for(Uint32 i = 0; i<noOfNodes; i++){
+ fragment2PrimaryNodeMap[i] = nodeIds[i];
+ }
+
+ // Sort them (bubble sort)
+ for(Uint32 i = 0; i<noOfNodes-1; i++)
+ for(Uint32 j = i+1; j<noOfNodes; j++)
+ if(fragment2PrimaryNodeMap[i] > fragment2PrimaryNodeMap[j]){
+ Uint32 tmp = fragment2PrimaryNodeMap[i];
+ fragment2PrimaryNodeMap[i] = fragment2PrimaryNodeMap[j];
+ fragment2PrimaryNodeMap[j] = tmp;
+ }
+
+ for(Uint32 i = 0; i<noOfNodes; i++){
+ fragment2PrimaryNodeMap[i+noOfNodes] = fragment2PrimaryNodeMap[i];
+ }
+ }
+}
+
+void
+Ndb::StartTransactionNodeSelectionData::release(){
+ delete [] fragment2PrimaryNodeMap;
+ fragment2PrimaryNodeMap = 0;
+}
+
+Uint32
+convertEndian(Uint32 Data)
+{
+#ifdef _BIG_ENDIAN
+ Uint32 t1, t2, t3, t4;
+ t4 = (Data >> 24) & 255;
+ t3 = (Data >> 16) & 255;
+ t4 = t4 + (t3 << 8);
+ t2 = (Data >> 8) & 255;
+ t4 = t4 + (t2 << 16);
+ t1 = Data & 255;
+ t4 = t4 + (t1 << 24);
+ return t4;
+#else
+ return Data;
+#endif
+}
+const char * Ndb::getCatalogName() const
+{
+ return theDataBase;
+}
+
+void Ndb::setCatalogName(const char * a_catalog_name)
+{
+ if (a_catalog_name) {
+ strncpy(theDataBase, a_catalog_name, NDB_MAX_DATABASE_NAME_SIZE);
+ // Prepare prefix for faster operations
+ uint db_len = MIN(strlen(theDataBase), NDB_MAX_DATABASE_NAME_SIZE - 1);
+ uint schema_len =
+ MIN(strlen(theDataBaseSchema), NDB_MAX_SCHEMA_NAME_SIZE - 1);
+ strncpy(prefixName, theDataBase, NDB_MAX_DATABASE_NAME_SIZE - 1);
+ prefixName[db_len] = '/';
+ strncpy(prefixName+db_len+1, theDataBaseSchema,
+ NDB_MAX_SCHEMA_NAME_SIZE - 1);
+ prefixName[db_len+schema_len+1] = '/';
+ prefixName[db_len+schema_len+2] = '\0';
+ prefixEnd = prefixName + db_len+schema_len + 2;
+ }
+}
+
+const char * Ndb::getSchemaName() const
+{
+ return theDataBaseSchema;
+}
+
+void Ndb::setSchemaName(const char * a_schema_name)
+{
+ if (a_schema_name) {
+ strncpy(theDataBaseSchema, a_schema_name, NDB_MAX_SCHEMA_NAME_SIZE);
+ // Prepare prefix for faster operations
+ uint db_len = MIN(strlen(theDataBase), NDB_MAX_DATABASE_NAME_SIZE - 1);
+ uint schema_len =
+ MIN(strlen(theDataBaseSchema), NDB_MAX_SCHEMA_NAME_SIZE - 1);
+ strncpy(prefixName, theDataBase, NDB_MAX_DATABASE_NAME_SIZE - 1);
+ prefixName[db_len] = '/';
+ strncpy(prefixName+db_len+1, theDataBaseSchema,
+ NDB_MAX_SCHEMA_NAME_SIZE - 1);
+ prefixName[db_len+schema_len+1] = '/';
+ prefixName[db_len+schema_len+2] = '\0';
+ prefixEnd = prefixName + db_len+schema_len + 2;
+ }
+}
+
+/*
+Deprecated functions
+*/
+const char * Ndb::getDatabaseName() const
+{
+ return getCatalogName();
+}
+
+void Ndb::setDatabaseName(const char * a_catalog_name)
+{
+ setCatalogName(a_catalog_name);
+}
+
+const char * Ndb::getDatabaseSchemaName() const
+{
+ return getSchemaName();
+}
+
+void Ndb::setDatabaseSchemaName(const char * a_schema_name)
+{
+ setSchemaName(a_schema_name);
+}
+
+void Ndb::useFullyQualifiedNames(bool turnNamingOn)
+{
+ fullyQualifiedNames = turnNamingOn;
+}
+
+bool Ndb::usingFullyQualifiedNames()
+{
+ return fullyQualifiedNames;
+}
+
+const char *
+Ndb::externalizeTableName(const char * internalTableName)
+{
+ if (fullyQualifiedNames) {
+ register const char *ptr = internalTableName;
+
+ // Skip database name
+ while (*ptr && *ptr++ != '/');
+ // Skip schema name
+ while (*ptr && *ptr++ != '/');
+
+ return ptr;
+ }
+ else
+ return internalTableName;
+}
+
+
+const char *
+Ndb::internalizeTableName(const char * externalTableName)
+{
+ if (fullyQualifiedNames) {
+ strncpy(prefixEnd, externalTableName, NDB_MAX_TAB_NAME_SIZE);
+ return prefixName;
+ }
+ else
+ return externalTableName;
+}
+
+const char *
+Ndb::externalizeIndexName(const char * internalIndexName)
+{
+ if (fullyQualifiedNames) {
+ register const char *ptr = internalIndexName;
+
+ // Scan name from the end
+ while (*ptr++); ptr--; // strend
+ while (ptr >= internalIndexName && *ptr != '/')
+ ptr--;
+
+ return ptr + 1;
+ }
+ else
+ return internalIndexName;
+}
+
+const char *
+Ndb::internalizeIndexName(const NdbTableImpl * table,
+ const char * externalIndexName)
+{
+ if (fullyQualifiedNames) {
+ char tableId[10];
+ sprintf(tableId, "%d", table->m_tableId);
+ Uint32 tabIdLen = strlen(tableId);
+ strncpy(prefixEnd, tableId, tabIdLen);
+ prefixEnd[tabIdLen] = '/';
+ strncpy(prefixEnd + tabIdLen + 1,
+ externalIndexName, NDB_MAX_TAB_NAME_SIZE);
+ return prefixName;
+ }
+ else
+ return externalIndexName;
+}
+
+const BaseString
+Ndb::getDatabaseFromInternalName(const char * internalName)
+{
+ char * databaseName = new char[strlen(internalName) + 1];
+ strcpy(databaseName, internalName);
+ register char *ptr = databaseName;
+
+ /* Scan name for the first '/' */
+ while (*ptr && *ptr != '/')
+ ptr++;
+ *ptr = '\0';
+ BaseString ret = BaseString(databaseName);
+ delete [] databaseName;
+ return ret;
+}
+
+const BaseString
+Ndb::getSchemaFromInternalName(const char * internalName)
+{
+ char * schemaName = new char[strlen(internalName)];
+ register const char *ptr1 = internalName;
+
+ /* Scan name for the second '/' */
+ while (*ptr1 && *ptr1 != '/')
+ ptr1++;
+ strcpy(schemaName, ptr1 + 1);
+ register char *ptr = schemaName;
+ while (*ptr && *ptr != '/')
+ ptr++;
+ *ptr = '\0';
+ BaseString ret = BaseString(schemaName);
+ delete [] schemaName;
+ return ret;
+}
+
+NdbEventOperation* Ndb::createEventOperation(const char* eventName,
+ const int bufferLength)
+{
+ NdbEventOperation* tOp;
+
+ tOp = new NdbEventOperation(this, eventName, bufferLength);
+
+ if (tOp->getState() != NdbEventOperation::CREATED) {
+ delete tOp;
+ tOp = NULL;
+ }
+
+ //now we have to look up this event in dict
+
+ return tOp;
+}
+
+int Ndb::dropEventOperation(NdbEventOperation* op) {
+ delete op;
+ return 0;
+}
+
+NdbGlobalEventBufferHandle* Ndb::getGlobalEventBufferHandle()
+{
+ return theGlobalEventBufferHandle;
+}
+
+//void Ndb::monitorEvent(NdbEventOperation *op, NdbEventCallback cb, void* rs)
+//{
+//}
+
+int
+Ndb::pollEvents(int aMillisecondNumber)
+{
+ return NdbEventOperation::wait(theGlobalEventBufferHandle,
+ aMillisecondNumber);
+}
+
+#ifdef VM_TRACE
+#include <NdbMutex.h>
+#include <stdarg.h>
+static NdbMutex print_state_mutex = NDB_MUTEX_INITIALIZER;
+static bool
+checkdups(NdbConnection** list, unsigned no)
+{
+ for (unsigned i = 0; i < no; i++)
+ for (unsigned j = i + 1; j < no; j++)
+ if (list[i] == list[j])
+ return true;
+ return false;
+}
+void
+Ndb::printState(const char* fmt, ...)
+{
+ char buf[200];
+ va_list ap;
+ va_start(ap, fmt);
+ vsprintf(buf, fmt, ap);
+ va_end(ap);
+ NdbMutex_Lock(&print_state_mutex);
+ bool dups = false;
+ ndbout << buf << " ndb=" << hex << this << dec;
+#ifdef NDB_LINUX
+ ndbout << " thread=" << (int)pthread_self();
+#endif
+ ndbout << endl;
+ for (unsigned n = 0; n < MAX_NDB_NODES; n++) {
+ NdbConnection* con = theConnectionArray[n];
+ if (con != 0) {
+ ndbout << "conn " << n << ":" << endl;
+ while (con != 0) {
+ con->printState();
+ con = con->theNext;
+ }
+ }
+ }
+ ndbout << "prepared: " << theNoOfPreparedTransactions<< endl;
+ if (checkdups(thePreparedTransactionsArray, theNoOfPreparedTransactions)) {
+ ndbout << "!! DUPS !!" << endl;
+ dups = true;
+ }
+ for (unsigned i = 0; i < theNoOfPreparedTransactions; i++)
+ thePreparedTransactionsArray[i]->printState();
+ ndbout << "sent: " << theNoOfSentTransactions<< endl;
+ if (checkdups(theSentTransactionsArray, theNoOfSentTransactions)) {
+ ndbout << "!! DUPS !!" << endl;
+ dups = true;
+ }
+ for (unsigned i = 0; i < theNoOfSentTransactions; i++)
+ theSentTransactionsArray[i]->printState();
+ ndbout << "completed: " << theNoOfCompletedTransactions<< endl;
+ if (checkdups(theCompletedTransactionsArray, theNoOfCompletedTransactions)) {
+ ndbout << "!! DUPS !!" << endl;
+ dups = true;
+ }
+ for (unsigned i = 0; i < theNoOfCompletedTransactions; i++)
+ theCompletedTransactionsArray[i]->printState();
+ NdbMutex_Unlock(&print_state_mutex);
+}
+#endif
+
+
diff --git a/ndb/src/ndbapi/NdbApiSignal.cpp b/ndb/src/ndbapi/NdbApiSignal.cpp
new file mode 100644
index 00000000000..a9cd5b1d53a
--- /dev/null
+++ b/ndb/src/ndbapi/NdbApiSignal.cpp
@@ -0,0 +1,300 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/******************************************************************************
+Name: NdbApiSignal.C
+Include:
+Link:
+Author: UABMNST Mona Natterkvist UAB/B/SD
+Date: 970829
+Version: 0.1
+Description: Interface between TIS and NDB
+Documentation:
+Adjust: 971114 UABMNST First version.
+ 000705 QABANAB Update of Protocol2
+******************************************************************************/
+#include "API.hpp"
+#include "NdbApiSignal.hpp"
+#include <AttrType.hpp>
+
+/**
+ * The following include includes
+ * definitions directly from the kernel
+ *
+ * Definitions that is shared between kernel and the API
+ */
+#include <signaldata/TcKeyReq.hpp>
+#include <signaldata/KeyInfo.hpp>
+#include <signaldata/AttrInfo.hpp>
+#include <signaldata/TestOrd.hpp>
+#include <signaldata/CreateIndx.hpp>
+#include <signaldata/DropIndx.hpp>
+#include <signaldata/TcIndx.hpp>
+#include <signaldata/IndxKeyInfo.hpp>
+#include <signaldata/IndxAttrInfo.hpp>
+#include <signaldata/TcHbRep.hpp>
+
+#include <NdbOut.hpp>
+
+/******************************************************************************
+NdbApiSignal();
+
+Return Value: None
+Remark: Creates a NdbApiSignal object.
+******************************************************************************/
+NdbApiSignal::NdbApiSignal(BlockReference ref)
+{
+ theVerId_signalNumber = 0; // 4 bit ver id - 16 bit gsn
+ theReceiversBlockNumber = 0; // Only 16 bit blocknum
+ theSendersBlockRef = refToBlock(ref);
+ theLength = 0;
+ theSendersSignalId = 0;
+ theSignalId = 0;
+ theTrace = 0;
+ m_noOfSections = 0;
+ m_fragmentInfo = 0;
+ for (int i = 0; i < 25; i++)
+ theData[i] = 0x13579753;
+
+ setDataPtr(&theData[0]);
+ theNextSignal = 0;
+}
+
+/**
+ * Copy constructor
+ */
+NdbApiSignal::NdbApiSignal(const NdbApiSignal &src) {
+ copyFrom(&src);
+}
+/******************************************************************************
+~NdbApiSignal();
+
+Return Value: None
+Remark: Delete a NdbApiSignal object.
+******************************************************************************/
+NdbApiSignal::~NdbApiSignal()
+{
+}
+/******************************************************************************
+int setSignal(NdbSignalType aNdbSignalType);
+
+Return Value: Return 0 : setSignal was successful.
+ Return tErrorCode In all other case.
+Parameters: aNdbSignalType: Type of signal.
+Remark: Set signal header and allocate 128 byte.
+******************************************************************************/
+int
+NdbApiSignal::setSignal(int aNdbSignalType)
+{
+ theSendersSignalId = 0;
+ switch (aNdbSignalType)
+ {
+ case GSN_DIHNDBTAMPER:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBDIH;
+ theVerId_signalNumber = GSN_DIHNDBTAMPER;
+ theLength = 3;
+ }
+ break;
+
+ case GSN_TCSEIZEREQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_TCSEIZEREQ;
+ theLength = 2;
+ }
+ break;
+
+ case GSN_TCKEYREQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_TCKEYREQ;
+ theLength = TcKeyReq::SignalLength;
+ }
+ break;
+
+ case GSN_TCRELEASEREQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_TCRELEASEREQ;
+ theLength = 3;
+ }
+ break;
+
+ case GSN_ATTRINFO:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_ATTRINFO;
+ theLength = AttrInfo::MaxSignalLength;
+ }
+ break;
+
+ case GSN_KEYINFO:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_KEYINFO;
+ theLength = KeyInfo::MaxSignalLength;
+ }
+ break;
+
+ case GSN_TCROLLBACKREQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_TCROLLBACKREQ;
+ theLength = 5;
+ }
+ break;
+
+ case GSN_TC_HBREP:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_TC_HBREP;
+ theLength = TcHbRep::SignalLength;
+ }
+ break;
+
+ case GSN_TC_COMMITREQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_TC_COMMITREQ;
+ theLength = 5;
+ }
+ break;
+
+ case GSN_SCAN_TABREQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_SCAN_TABREQ;
+ theLength = 25;
+ }
+ break;
+
+ case GSN_SCAN_TABINFO:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_SCAN_TABINFO;
+ theLength = 17;
+ }
+ break;
+
+ case GSN_SCAN_NEXTREQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_SCAN_NEXTREQ;
+ theLength = 4;
+ }
+ break;
+
+ case GSN_CREATE_INDX_REQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBDICT;
+ theVerId_signalNumber = GSN_CREATE_INDX_REQ;
+ theLength = CreateIndxReq::SignalLength;
+ }
+ break;
+
+ case GSN_DROP_INDX_REQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBDICT;
+ theVerId_signalNumber = GSN_DROP_INDX_REQ;
+ theLength = DropIndxReq::SignalLength;
+ }
+ break;
+
+ case GSN_TCINDXREQ:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_TCINDXREQ;
+ theLength = TcIndxReq::SignalLength;
+ }
+ break;
+
+ case GSN_INDXKEYINFO:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_INDXKEYINFO;
+ theLength = IndxKeyInfo::MaxSignalLength;
+ }
+ break;
+
+ case GSN_INDXATTRINFO:
+ {
+ theTrace = TestOrd::TraceAPI;
+ theReceiversBlockNumber = DBTC;
+ theVerId_signalNumber = GSN_INDXATTRINFO;
+ theLength = IndxAttrInfo::MaxSignalLength;
+ }
+ break;
+
+ default:
+ {
+ return -1;
+ }
+ }
+ return 0;
+}
+
+void
+NdbApiSignal::set(Uint8 trace,
+ Uint16 receiversBlockNumber,
+ Uint16 signalNumber,
+ Uint32 length){
+
+ theTrace = trace;
+ theReceiversBlockNumber = receiversBlockNumber;
+ theVerId_signalNumber = signalNumber;
+ theLength = length;
+}
+
+void
+NdbApiSignal::copyFrom(const NdbApiSignal * src){
+ theVerId_signalNumber = src->theVerId_signalNumber;
+ theReceiversBlockNumber = src->theReceiversBlockNumber;
+ theSendersBlockRef = src->theSendersBlockRef;
+ theLength = src->theLength;
+ theTrace = src->theTrace;
+
+ Uint32 * dstData = getDataPtrSend();
+ const Uint32 * srcData = src->getDataPtr();
+ for (Uint32 i = 0; i < theLength; i++)
+ dstData[i] = srcData[i];
+
+ setDataPtr(dstData);
+
+ /**
+ * NOTE that theSignalId is used as data ptr
+ * and should not be copied
+ * NOTE that theSendersSignalId is used as next pointer
+ * and should not be copied
+ */
+}
diff --git a/ndb/src/ndbapi/NdbApiSignal.hpp b/ndb/src/ndbapi/NdbApiSignal.hpp
new file mode 100644
index 00000000000..76cefe0e882
--- /dev/null
+++ b/ndb/src/ndbapi/NdbApiSignal.hpp
@@ -0,0 +1,212 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**********************************************************************
+ * Name: NdbApiSignal.H
+ * Include:
+ * Link:
+ * Author: UABMNST Mona Natterkvist UAB/B/SD
+ * Date: 97----
+ * Version: 0.1
+ * Description: Interface between TIS and NDB
+ * Documentation:
+ * Adjust: 971204 UABMNST First version.
+ * Adjust: 000705 QABANAB Changes in Protocol2
+ * Comment:
+ *****************************************************************************/
+#ifndef NdbApiSignal_H
+#define NdbApiSignal_H
+
+#include <kernel_types.h>
+#include "AttrType.hpp"
+#include "TransporterFacade.hpp"
+#include <TransporterDefinitions.hpp>
+#include "Ndb.hpp"
+
+#define CAST_PTR(X,Y) static_cast<X*>(static_cast<void*>(Y))
+#define CAST_CONSTPTR(X,Y) static_cast<const X*>(static_cast<const void*>(Y))
+
+/**
+ * A NdbApiSignal : public SignalHeader
+ *
+ * Stores the address to theData in theSignalId
+ */
+class NdbApiSignal : public SignalHeader
+ {
+public:
+ NdbApiSignal(BlockReference myRef);
+ NdbApiSignal(const NdbApiSignal &);
+ NdbApiSignal(const SignalHeader &header)
+ : SignalHeader(header), theNextSignal(0), theRealData(0) {};
+ ~NdbApiSignal();
+
+ void set(Uint8 trace,
+ Uint16 receiversBlockNumber,
+ Uint16 signalNumber,
+ Uint32 length);
+
+
+ void setData(Uint32 aWord, Uint32 aDataNo);
+ Uint32 readData(Uint32 aDataNo) const; // Read word in signal
+
+ int setSignal(int NdbSignalType); // Set signal header
+ int readSignalNumber(); // Read signal number
+ Uint32 getLength() const;
+ void setLength(Uint32 aLength);
+ void next(NdbApiSignal* anApiSignal);
+ NdbApiSignal* next();
+
+ const Uint32 * getDataPtr() const;
+ Uint32 * getDataPtrSend();
+
+ /**
+ * Fragmentation
+ */
+ bool isFirstFragment() const { return m_fragmentInfo <= 1;}
+ bool isLastFragment() const {
+ return m_fragmentInfo == 0 || m_fragmentInfo == 3;
+ }
+
+ Uint32 getFragmentId() const {
+ return (m_fragmentInfo == 0 ? 0 : getDataPtr()[theLength - 1]);
+ }
+
+private:
+ friend void execute(void * callbackObj,
+ struct SignalHeader * const header,
+ Uint8 prio, Uint32 * const theData,
+ LinearSectionPtr ptr[3]);
+
+ void setDataPtr(Uint32 *);
+
+ friend class NdbConnection;
+ friend class NdbScanReceiver;
+ friend class Table;
+ void copyFrom(const NdbApiSignal * src);
+
+ /**
+ * Only used when creating a signal in the api
+ */
+ Uint32 theData[25];
+ NdbApiSignal *theNextSignal;
+ Uint32 *theRealData;
+};
+/**********************************************************************
+void getLength
+Remark: Get the length of the signal.
+******************************************************************************/
+inline
+Uint32
+NdbApiSignal::getLength() const{
+ return theLength;
+}
+
+/**********************************************************************
+void setLength
+Parameters: aLength: Signal length
+Remark: Set the length in the signal.
+******************************************************************************/
+inline
+void
+NdbApiSignal::setLength(Uint32 aLength){
+ theLength = aLength;
+}
+
+/**********************************************************************
+void next(NdbApiSignal* aSignal);
+
+Parameters: aSignal: Signal object.
+Remark: Insert signal rear in a linked list.
+*****************************************************************************/
+inline
+void
+NdbApiSignal::next(NdbApiSignal* aSignal){
+ theNextSignal = aSignal;
+}
+/**********************************************************************
+NdbApiSignal* next();
+
+Return Value: Return theNext signal object if the next was successful.
+ Return NULL: In all other case.
+Remark: Read the theNext in signal.
+*****************************************************************************/
+inline
+NdbApiSignal*
+NdbApiSignal::next(){
+ return theNextSignal;
+}
+/**********************************************************************
+int readSignalNo();
+
+Return Value: Return the signalNumber.
+Remark: Read signal number
+*****************************************************************************/
+inline
+int
+NdbApiSignal::readSignalNumber()
+{
+ return (int)theVerId_signalNumber;
+}
+/**********************************************************************
+Uint32 readData(Uint32 aDataNo);
+
+Return Value: Return Data word in a signal.
+ Return -1: In all other case.
+ aDataNo: Data number in signal.
+Remark: Return the dataWord information in a signal for a dataNo.
+******************************************************************************/
+inline
+Uint32
+NdbApiSignal::readData(Uint32 aDataNo) const {
+ return getDataPtr()[aDataNo-1];
+}
+/**********************************************************************
+int setData(Uint32 aWord, int aDataNo);
+
+Return Value: Return 0 : setData was successful.
+ Return -1: In all other case.
+Parameters: aWord: Data word.
+ aDataNo: Data number in signal.
+Remark: Set Data word in signal 1 - 25
+******************************************************************************/
+inline
+void
+NdbApiSignal::setData(Uint32 aWord, Uint32 aDataNo){
+ getDataPtrSend()[aDataNo -1] = aWord;
+}
+
+/**
+ * Return pointer to data structure
+ */
+inline
+const Uint32 *
+NdbApiSignal::getDataPtr() const {
+ return theRealData;
+}
+
+inline
+Uint32 *
+NdbApiSignal::getDataPtrSend(){
+ return (Uint32*)&theData[0];
+}
+
+inline
+void
+NdbApiSignal::setDataPtr(Uint32 * ptr){
+ theRealData = ptr;
+}
+
+#endif
diff --git a/ndb/src/ndbapi/NdbConnection.cpp b/ndb/src/ndbapi/NdbConnection.cpp
new file mode 100644
index 00000000000..4ec098c3c60
--- /dev/null
+++ b/ndb/src/ndbapi/NdbConnection.cpp
@@ -0,0 +1,1787 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+/*****************************************************************************
+Name: NdbConnection.C
+Include:
+Link:
+Author: UABMNST Mona Natterkvist UAB/B/UL
+Date: 970829
+Version: 0.1
+Description: Interface between TIS and NDB
+Documentation:
+Adjust: 971022 UABMNST First version.
+*****************************************************************************/
+#include "NdbOut.hpp"
+#include "NdbConnection.hpp"
+#include "NdbOperation.hpp"
+#include "NdbScanOperation.hpp"
+#include "NdbIndexOperation.hpp"
+#include "NdbApiSignal.hpp"
+#include "TransporterFacade.hpp"
+#include "API.hpp"
+#include <ndb_limits.h>
+
+#include <signaldata/TcKeyConf.hpp>
+#include <signaldata/TcIndx.hpp>
+#include <signaldata/TcCommit.hpp>
+#include <signaldata/TcKeyFailConf.hpp>
+#include <signaldata/TcHbRep.hpp>
+
+/*****************************************************************************
+NdbConnection( Ndb* aNdb );
+
+Return Value: None
+Parameters: aNdb: Pointers to the Ndb object
+Remark: Creates a connection object.
+*****************************************************************************/
+NdbConnection::NdbConnection( Ndb* aNdb ) :
+ theSendStatus(NotInit),
+ theCallbackFunction(NULL),
+ theCallbackObject(NULL),
+ theTransArrayIndex(0),
+ theStartTransTime(0),
+ theErrorLine(0),
+ theErrorOperation(NULL),
+ theNdb(aNdb),
+ theNext(NULL),
+ theFirstOpInList(NULL),
+ theLastOpInList(NULL),
+ theFirstExecOpInList(NULL),
+ theLastExecOpInList(NULL),
+ theCompletedFirstOp(NULL),
+ theNoOfOpSent(0),
+ theNoOfOpCompleted(0),
+ theNoOfOpFetched(0),
+ theMyRef(0),
+ theTCConPtr(0),
+ theTransactionId(0),
+ theGlobalCheckpointId(0),
+ theStatus(NotConnected),
+ theCompletionStatus(NotCompleted),
+ theCommitStatus(NotStarted),
+ theMagicNumber(0xFE11DC),
+ theTransactionIsStarted(false),
+ theDBnode(0),
+ theReleaseOnClose(false),
+ // Cursor operations
+ m_waitForReply(true),
+ m_theFirstCursorOperation(NULL),
+ m_theLastCursorOperation(NULL),
+ m_firstExecutedCursorOp(NULL),
+ // Scan operations
+ theScanFinished(0),
+ theCurrentScanRec(NULL),
+ thePreviousScanRec(NULL),
+ theScanningOp(NULL),
+ theBuddyConPtr(0xFFFFFFFF)
+{
+ theListState = NotInList;
+ theError.code = 0;
+ theId = theNdb->theNdbObjectIdMap->map(this);
+}//NdbConnection::NdbConnection()
+
+/*****************************************************************************
+~NdbConnection();
+
+Remark: Deletes the connection object.
+*****************************************************************************/
+NdbConnection::~NdbConnection()
+{
+ theNdb->theNdbObjectIdMap->unmap(theId, this);
+}//NdbConnection::~NdbConnection()
+
+/*****************************************************************************
+void init();
+
+Remark: Initialise connection object for new transaction.
+*****************************************************************************/
+void
+NdbConnection::init()
+{
+ theListState = NotInList;
+ theInUseState = true;
+ theTransactionIsStarted = false;
+ theScanFinished = 0;
+ theNext = NULL;
+
+ theFirstOpInList = NULL;
+ theLastOpInList = NULL;
+
+ theScanningOp = NULL;
+
+ theFirstExecOpInList = NULL;
+ theLastExecOpInList = NULL;
+
+ theCurrentScanRec = NULL;
+ thePreviousScanRec = NULL;
+
+ theCompletedFirstOp = NULL;
+
+ theGlobalCheckpointId = 0;
+ theCommitStatus = Started;
+ theCompletionStatus = NotCompleted;
+ m_abortOption = AbortOnError;
+
+ theError.code = 0;
+ theErrorLine = 0;
+ theErrorOperation = NULL;
+
+ theReleaseOnClose = false;
+ theSimpleState = true;
+ theSendStatus = InitState;
+ theMagicNumber = 0x37412619;
+ // Cursor operations
+ m_waitForReply = true;
+ m_theFirstCursorOperation = NULL;
+ m_theLastCursorOperation = NULL;
+ m_firstExecutedCursorOp = 0;
+ theBuddyConPtr = 0xFFFFFFFF;
+}//NdbConnection::init()
+
+/*****************************************************************************
+setOperationErrorCode(int anErrorCode);
+
+Remark: Sets an error code on the connection object from an
+ operation object.
+*****************************************************************************/
+void
+NdbConnection::setOperationErrorCode(int anErrorCode)
+{
+ if (theError.code == 0)
+ theError.code = anErrorCode;
+}//NdbConnection::setOperationErrorCode()
+
+/*****************************************************************************
+setOperationErrorCodeAbort(int anErrorCode);
+
+Remark: Sets an error code on the connection object from an
+ operation object.
+*****************************************************************************/
+void
+NdbConnection::setOperationErrorCodeAbort(int anErrorCode)
+{
+ if (theTransactionIsStarted == false) {
+ theCommitStatus = Aborted;
+ } else if ((m_abortOption == AbortOnError) &&
+ (theCommitStatus != Committed) &&
+ (theCommitStatus != Aborted)) {
+ theCommitStatus = NeedAbort;
+ }//if
+ if (theError.code == 0)
+ theError.code = anErrorCode;
+}//NdbConnection::setOperationErrorCodeAbort()
+
+/*****************************************************************************
+setErrorCode(int anErrorCode);
+
+Remark: Sets an error indication on the connection object.
+*****************************************************************************/
+void
+NdbConnection::setErrorCode(int anErrorCode)
+{
+ if (theError.code == 0)
+ theError.code = anErrorCode;
+}//NdbConnection::setErrorCode()
+
+/*****************************************************************************
+void handleExecuteCompletion(void);
+
+Remark: Handle time-out on a transaction object.
+*****************************************************************************/
+void
+NdbConnection::handleExecuteCompletion()
+{
+
+ if (theCompletionStatus == CompletedFailure) {
+ NdbOperation* tOpTemp = theFirstExecOpInList;
+ while (tOpTemp != NULL) {
+/*****************************************************************************
+ * Ensure that all executing operations report failed for each
+ * read attribute when failure occurs.
+ * We do not want any operations to report both failure and
+ * success on different read attributes.
+ ****************************************************************************/
+ tOpTemp->handleFailedAI_ElemLen();
+ tOpTemp = tOpTemp->next();
+ }//while
+ theReturnStatus = ReturnFailure;
+ }//if
+ /***************************************************************************
+ * Move the NdbOperation objects from the list of executing
+ * operations to list of completed
+ **************************************************************************/
+ NdbOperation* tFirstExecOp = theFirstExecOpInList;
+ NdbOperation* tLastExecOp = theLastExecOpInList;
+ if (tLastExecOp != NULL) {
+ tLastExecOp->next(theCompletedFirstOp);
+ theCompletedFirstOp = tFirstExecOp;
+ theFirstExecOpInList = NULL;
+ theLastExecOpInList = NULL;
+ }//if
+ theSendStatus = InitState;
+ return;
+}//NdbConnection::handleExecuteCompletion()
+
+/*****************************************************************************
+int execute(ExecType aTypeOfExec, CommitType aTypeOfCommit, int forceSend);
+
+Return Value: Return 0 : execute was successful.
+ Return -1: In all other case.
+Parameters : aTypeOfExec: Type of execute.
+Remark: Initialise connection object for new transaction.
+*****************************************************************************/
+int
+NdbConnection::execute(ExecType aTypeOfExec,
+ AbortOption abortOption,
+ int forceSend)
+{
+//------------------------------------------------------------------------
+// We will start by preparing all operations in the transaction defined
+// since last execute or since beginning. If this works ok we will continue
+// by calling the poll with wait method. This method will return when
+// the NDB kernel has completed its task or when 10 seconds have passed.
+// The NdbConnectionCallBack-method will receive the return code of the
+// transaction. The normal methods of reading error codes still apply.
+//------------------------------------------------------------------------
+ Ndb* tNdb = theNdb;
+
+ m_waitForReply = false;
+ executeAsynchPrepare(aTypeOfExec, NULL, NULL, abortOption);
+ if (m_waitForReply){
+ while (1) {
+ int noOfComp = tNdb->sendPollNdb((3 * WAITFOR_RESPONSE_TIMEOUT),
+ 1, forceSend);
+ if (noOfComp == 0) {
+ /**
+ * This timeout situation can occur if NDB crashes.
+ */
+ ndbout << "This timeout should never occur, execute(..)" << endl;
+ setOperationErrorCodeAbort(4012); // Error code for "Cluster Failure"
+ return -1;
+ }//if
+
+ /*
+ * Check that the completed transactions include this one. There
+ * could be another thread running asynchronously. Even in pure
+ * async case rollback is done synchronously.
+ */
+ if (theListState != NotInList)
+ continue;
+#ifdef VM_TRACE
+ unsigned anyway = 0;
+ for (unsigned i = 0; i < theNdb->theNoOfPreparedTransactions; i++)
+ anyway += theNdb->thePreparedTransactionsArray[i] == this;
+ for (unsigned i = 0; i < theNdb->theNoOfSentTransactions; i++)
+ anyway += theNdb->theSentTransactionsArray[i] == this;
+ for (unsigned i = 0; i < theNdb->theNoOfCompletedTransactions; i++)
+ anyway += theNdb->theCompletedTransactionsArray[i] == this;
+ if (anyway) {
+ theNdb->printState("execute %x", this);
+ abort();
+ }
+#endif
+ if (theReturnStatus == ReturnFailure) {
+ return -1;
+ }//if
+ break;
+ }
+ }
+ return 0;
+}//NdbConnection::execute()
+
+/*****************************************************************************
+void executeAsynchPrepare(ExecType aTypeOfExec,
+ NdbAsynchCallback callBack,
+ void* anyObject,
+ CommitType aTypeOfCommit);
+
+Return Value: No return value
+Parameters : aTypeOfExec: Type of execute.
+ anyObject: An object provided in the callback method
+ callBack: The callback method
+ aTypeOfCommit: What to do when read/updated/deleted records
+ are missing or inserted records already exist.
+
+Remark: Prepare a part of a transaction in an asynchronous manner.
+*****************************************************************************/
+void
+NdbConnection::executeAsynchPrepare( ExecType aTypeOfExec,
+ NdbAsynchCallback aCallback,
+ void* anyObject,
+ AbortOption abortOption)
+{
+ /**
+ * Reset error.code on execute
+ */
+ theError.code = 0;
+
+ NdbCursorOperation* tcOp = m_theFirstCursorOperation;
+ if (tcOp != 0){
+ // Execute any cursor operations
+ while (tcOp != NULL) {
+ int tReturnCode;
+ tReturnCode = tcOp->executeCursor(theDBnode);
+ if (tReturnCode == -1) {
+ return;
+ }//if
+ tcOp = (NdbCursorOperation*)tcOp->next();
+ } // while
+ m_theLastCursorOperation->next(m_firstExecutedCursorOp);
+ m_firstExecutedCursorOp = m_theFirstCursorOperation;
+ // Discard cursor operations, since these are also
+ // in the complete operations list we do not need
+ // to release them.
+ m_theFirstCursorOperation = m_theLastCursorOperation = NULL;
+ }
+
+ bool tTransactionIsStarted = theTransactionIsStarted;
+ NdbOperation* tLastOp = theLastOpInList;
+ Ndb* tNdb = theNdb;
+ CommitStatusType tCommitStatus = theCommitStatus;
+ Uint32 tnoOfPreparedTransactions = tNdb->theNoOfPreparedTransactions;
+
+ theReturnStatus = ReturnSuccess;
+ theCallbackFunction = aCallback;
+ theCallbackObject = anyObject;
+ m_abortOption = abortOption;
+ // SendStatusType tSendStatus = theSendStatus;
+
+// if (tSendStatus != InitState) {
+/****************************************************************************
+ * The application is obviously doing strange things. We should probably
+ * report to the application the problem in some manner. Since we don't have
+ * a good way of handling the problem we avoid discovering the problem.
+ * Should be handled at some point in time.
+ ****************************************************************************/
+// return;
+// }
+ m_waitForReply = true;
+ tNdb->thePreparedTransactionsArray[tnoOfPreparedTransactions] = this;
+ theTransArrayIndex = tnoOfPreparedTransactions;
+ theListState = InPreparedList;
+ tNdb->theNoOfPreparedTransactions = tnoOfPreparedTransactions + 1;
+
+ if(tCommitStatus == Committed){
+ tCommitStatus = Started;
+ tTransactionIsStarted = false;
+ }
+
+ if ((tCommitStatus != Started) ||
+ (aTypeOfExec == Rollback)) {
+/*****************************************************************************
+ * Rollback have been ordered on a started transaction. Call rollback.
+ * Could also be state problem or previous problem which leads to the
+ * same action.
+ ****************************************************************************/
+ if (aTypeOfExec == Rollback) {
+ if (theTransactionIsStarted == false) {
+ theCommitStatus = Aborted;
+ theSendStatus = sendCompleted;
+ } else {
+ theSendStatus = sendABORT;
+ }
+ } else {
+ theSendStatus = sendABORTfail;
+ }//if
+ return;
+ }//if
+ if (tTransactionIsStarted == true) {
+ if (tLastOp != NULL) {
+ if (aTypeOfExec == Commit) {
+/*****************************************************************************
+ * Set commit indicator on last operation when commit has been ordered
+ * and also a number of operations.
+******************************************************************************/
+ tLastOp->theCommitIndicator = 1;
+ }//if
+ } else {
+ if (aTypeOfExec == Commit) {
+ /**********************************************************************
+ * A Transaction have been started and no more operations exist.
+ * We will use the commit method.
+ *********************************************************************/
+ theSendStatus = sendCOMMITstate;
+ return;
+ } else {
+ /**********************************************************************
+ * We need to put it into the array of completed transactions to
+ * ensure that we report the completion in a proper way.
+ * We cannot do this here since that would endanger the completed
+ * transaction array since that is also updated from the receiver
+ * thread and thus we need to do it under mutex lock and thus we
+ * set the sendStatus to ensure that the send method will
+ * put it into the completed array.
+ **********************************************************************/
+ theSendStatus = sendCompleted;
+ return; // No Commit with no operations is OK
+ }//if
+ }//if
+ } else if (tTransactionIsStarted == false) {
+ NdbOperation* tFirstOp = theFirstOpInList;
+ if (tLastOp != NULL) {
+ tFirstOp->setStartIndicator();
+ if (aTypeOfExec == Commit) {
+ tLastOp->theCommitIndicator = 1;
+ }//if
+ } else {
+ /***********************************************************************
+ * No operations are defined and we have not started yet.
+ * Simply return OK. Set commit status if Commit.
+ ***********************************************************************/
+ if (aTypeOfExec == Commit) {
+ theCommitStatus = Committed;
+ }//if
+ /***********************************************************************
+ * We need to put it into the array of completed transactions to
+ * ensure that we report the completion in a proper way. We
+ * cannot do this here since that would endanger the completed
+ * transaction array since that is also updated from the
+ * receiver thread and thus we need to do it under mutex lock
+ * and thus we set the sendStatus to ensure that the send method
+ * will put it into the completed array.
+ ***********************************************************************/
+ theSendStatus = sendCompleted;
+ return;
+ }//if
+ }
+
+ NdbOperation* tOp = theFirstOpInList;
+ theCompletionStatus = NotCompleted;
+ while (tOp) {
+ int tReturnCode;
+ NdbOperation* tNextOp = tOp->next();
+
+ tReturnCode = tOp->prepareSend(theTCConPtr, theTransactionId);
+ if (tReturnCode == -1) {
+ theSendStatus = sendABORTfail;
+ return;
+ }//if
+
+ /*************************************************************************
+ * Now that we have successfully prepared the send of this operation we
+ * move it to the list of executing operations and remove it from the
+ * list of defined operations.
+ ************************************************************************/
+ tOp = tNextOp;
+ }
+
+ NdbOperation* tLastOpInList = theLastOpInList;
+ NdbOperation* tFirstOpInList = theFirstOpInList;
+
+ theFirstOpInList = NULL;
+ theLastOpInList = NULL;
+ theFirstExecOpInList = tFirstOpInList;
+ theLastExecOpInList = tLastOpInList;
+
+ theCompletionStatus = CompletedSuccess;
+ theNoOfOpSent = 0;
+ theNoOfOpCompleted = 0;
+ theSendStatus = sendOperations;
+ return;
+}//NdbConnection::executeAsynchPrepare()
+
+void NdbConnection::close()
+{
+ theNdb->closeTransaction(this);
+}
+
+int NdbConnection::refresh(){
+ return sendTC_HBREP();
+}
+
+/*****************************************************************************
+int sendTC_HBREP();
+
+Return Value: No return value.
+Parameters : None.
+Remark: Order NDB to refresh the timeout counter of the transaction.
+******************************************************************************/
+int
+NdbConnection::sendTC_HBREP() // Send a TC_HBREP signal;
+{
+ NdbApiSignal* tSignal;
+ Ndb* tNdb = theNdb;
+ Uint32 tTransId1, tTransId2;
+
+ tSignal = tNdb->getSignal();
+ if (tSignal == NULL) {
+ return -1;
+ }
+
+ if (tSignal->setSignal(GSN_TC_HBREP) == -1) {
+ return -1;
+ }
+
+ TcHbRep * const tcHbRep = CAST_PTR(TcHbRep, tSignal->getDataPtrSend());
+
+ tcHbRep->apiConnectPtr = theTCConPtr;
+
+ tTransId1 = (Uint32) theTransactionId;
+ tTransId2 = (Uint32) (theTransactionId >> 32);
+ tcHbRep->transId1 = tTransId1;
+ tcHbRep->transId2 = tTransId2;
+
+ TransporterFacade *tp = TransporterFacade::instance();
+ tp->lock_mutex();
+ const int res = tp->sendSignal(tSignal,theDBnode);
+ tp->unlock_mutex();
+ tNdb->releaseSignal(tSignal);
+
+ if (res == -1){
+ return -1;
+ }
+
+ return 0;
+}//NdbConnection::sendTC_HBREP()
+
+/*****************************************************************************
+int doSend();
+
+Return Value: Return 0 : send was successful.
+ Return -1: In all other case.
+Remark: Send all operations belonging to this connection.
+ The caller of this method has the responsibility to remove the
+ object from the prepared transactions array on the Ndb-object.
+*****************************************************************************/
+int
+NdbConnection::doSend()
+{
+ /*
+ This method assumes that at least one operation have been defined. This
+ is ensured by the caller of this routine (=execute).
+ */
+
+ switch(theSendStatus){
+ case sendOperations: {
+ NdbOperation * tOp = theFirstExecOpInList;
+ do {
+ NdbOperation* tNextOp = tOp->next();
+ const Uint32 lastFlag = ((tNextOp == NULL) ? 1 : 0);
+ const int tReturnCode = tOp->doSend(theDBnode, lastFlag);
+ if (tReturnCode == -1) {
+ theReturnStatus = ReturnFailure;
+ break;
+ }//if
+ tOp = tNextOp;
+ } while (tOp != NULL);
+ Ndb* tNdb = theNdb;
+ theSendStatus = sendTC_OP;
+ theTransactionIsStarted = true;
+ tNdb->insert_sent_list(this);
+ return 0;
+ }//case
+ case sendABORT:
+ case sendABORTfail:{
+ /***********************************************************************
+ * Rollback have been ordered on a not started transaction.
+ * Simply return OK and set abort status.
+ ***********************************************************************/
+ if (theSendStatus == sendABORTfail) {
+ theReturnStatus = ReturnFailure;
+ }//if
+ if (sendROLLBACK() == 0) {
+ return 0;
+ }//if
+ break;
+ }//case
+ case sendCOMMITstate:
+ if (sendCOMMIT() == 0) {
+ return 0;
+ }//if
+ break;
+ case sendCompleted:
+ theNdb->insert_completed_list(this);
+ return 0;
+ default:
+ ndbout << "Inconsistent theSendStatus = " << theSendStatus << endl;
+ abort();
+ break;
+ }//switch
+ setOperationErrorCodeAbort(4002);
+ theReleaseOnClose = true;
+ theTransactionIsStarted = false;
+ theCommitStatus = Aborted;
+ return -1;
+}//NdbConnection::doSend()
+
+/**************************************************************************
+int sendROLLBACK();
+
+Return Value: Return -1 if send unsuccessful.
+Parameters : None.
+Remark: Order NDB to rollback the transaction.
+**************************************************************************/
+int
+NdbConnection::sendROLLBACK() // Send a TCROLLBACKREQ signal;
+{
+ Ndb* tNdb = theNdb;
+ if ((theTransactionIsStarted == true) &&
+ (theCommitStatus != Committed) &&
+ (theCommitStatus != Aborted)) {
+/**************************************************************************
+ * The user did not perform any rollback but simply closed the
+ * transaction. We must rollback Ndb since Ndb have been contacted.
+ *************************************************************************/
+ NdbApiSignal tSignal(tNdb->theMyRef);
+ Uint32 tTransId1, tTransId2;
+ TransporterFacade *tp = TransporterFacade::instance();
+ int tReturnCode;
+
+ tTransId1 = (Uint32) theTransactionId;
+ tTransId2 = (Uint32) (theTransactionId >> 32);
+ tSignal.setSignal(GSN_TCROLLBACKREQ);
+ tSignal.setData(theTCConPtr, 1);
+ tSignal.setData(tTransId1, 2);
+ tSignal.setData(tTransId2, 3);
+ tReturnCode = tp->sendSignal(&tSignal,theDBnode);
+ if (tReturnCode != -1) {
+ theSendStatus = sendTC_ROLLBACK;
+ tNdb->insert_sent_list(this);
+ return 0;
+ }//if
+ /*********************************************************************
+ * It was not possible to abort the transaction towards the NDB kernel
+ * and thus we put it into the array of completed transactions that
+ * are ready for reporting to the application.
+ *********************************************************************/
+ return -1;
+ } else {
+ /*
+ It is not necessary to abort the transaction towards the NDB kernel and
+ thus we put it into the array of completed transactions that are ready
+ for reporting to the application.
+ */
+ theSendStatus = sendCompleted;
+ tNdb->insert_completed_list(this);
+ return 0;
+ ;
+ }//if
+}//NdbConnection::sendROLLBACK()
+
+/***************************************************************************
+int sendCOMMIT();
+
+Return Value: Return 0 : send was successful.
+ Return -1: In all other case.
+Parameters : None.
+Remark: Order NDB to commit the transaction.
+***************************************************************************/
+int
+NdbConnection::sendCOMMIT() // Send a TC_COMMITREQ signal;
+{
+ NdbApiSignal tSignal(theNdb->theMyRef);
+ Uint32 tTransId1, tTransId2;
+ TransporterFacade *tp = TransporterFacade::instance();
+ int tReturnCode;
+
+ tTransId1 = (Uint32) theTransactionId;
+ tTransId2 = (Uint32) (theTransactionId >> 32);
+ tSignal.setSignal(GSN_TC_COMMITREQ);
+ tSignal.setData(theTCConPtr, 1);
+ tSignal.setData(tTransId1, 2);
+ tSignal.setData(tTransId2, 3);
+
+ tReturnCode = tp->sendSignal(&tSignal,theDBnode);
+ if (tReturnCode != -1) {
+ theSendStatus = sendTC_COMMIT;
+ theNdb->insert_sent_list(this);
+ return 0;
+ } else {
+ return -1;
+ }//if
+}//NdbConnection::sendCOMMIT()
+
+/******************************************************************************
+void release();
+
+Remark: Release all operations.
+******************************************************************************/
+void
+NdbConnection::release(){
+ if (theTransactionIsStarted == true && theScanningOp != NULL )
+ stopScan();
+
+ releaseOperations();
+ if ( (theTransactionIsStarted == true) &&
+ ((theCommitStatus != Committed) &&
+ (theCommitStatus != Aborted))) {
+/****************************************************************************
+ * The user did not perform any rollback but simply closed the
+ * transaction. We must rollback Ndb since Ndb have been contacted.
+******************************************************************************/
+ execute(Rollback);
+ }//if
+ theMagicNumber = 0xFE11DC;
+ theInUseState = false;
+#ifdef VM_TRACE
+ if (theListState != NotInList) {
+ theNdb->printState("release %x", this);
+ abort();
+ }
+#endif
+}//NdbConnection::release()
+
+void
+NdbConnection::releaseOps(NdbOperation* tOp){
+ while (tOp != NULL) {
+ NdbOperation* tmp = tOp;
+ tOp->release();
+ tOp = tOp->next();
+ theNdb->releaseOperation(tmp);
+ }//while
+}
+
+/******************************************************************************
+void releaseOperations();
+
+Remark: Release all operations.
+******************************************************************************/
+void
+NdbConnection::releaseOperations()
+{
+ // Release any open scans
+ releaseCursorOperations(m_theFirstCursorOperation);
+ releaseCursorOperations(m_firstExecutedCursorOp);
+
+ releaseOps(theCompletedFirstOp);
+ releaseOps(theFirstOpInList);
+ releaseOps(theFirstExecOpInList);
+
+ theCompletedFirstOp = NULL;
+ theFirstOpInList = NULL;
+ theFirstExecOpInList = NULL;
+ theLastOpInList = NULL;
+ theLastExecOpInList = NULL;
+ theScanningOp = NULL;
+ m_theFirstCursorOperation = NULL;
+ m_theLastCursorOperation = NULL;
+ m_firstExecutedCursorOp = NULL;
+}//NdbConnection::releaseOperations()
+
+void
+NdbConnection::releaseCompletedOperations()
+{
+ releaseOps(theCompletedFirstOp);
+ theCompletedFirstOp = NULL;
+}//NdbConnection::releaseOperations()
+
+/******************************************************************************
+void releaseCursorOperations();
+
+Remark: Release all cursor operations.
+ (NdbScanOperation and NdbIndexOperation)
+******************************************************************************/
+void
+NdbConnection::releaseCursorOperations(NdbCursorOperation* cursorOp)
+{
+ while(cursorOp != 0){
+ NdbCursorOperation* next = (NdbCursorOperation*)cursorOp->next();
+ cursorOp->release();
+ if (cursorOp->cursorType() == NdbCursorOperation::ScanCursor)
+ theNdb->releaseScanOperation((NdbScanOperation*)cursorOp);
+ else
+ theNdb->releaseOperation(cursorOp);
+ cursorOp = next;
+ }
+}//NdbConnection::releaseCursorOperations()
+
+/*****************************************************************************
+NdbOperation* getNdbOperation(const char* aTableName);
+
+Return Value Return a pointer to a NdbOperation object if getNdbOperation
+ was succesful.
+ Return NULL : In all other case.
+Parameters: aTableName : Name of the database table.
+Remark: Get an operation from NdbOperation idlelist and get the
+ NdbConnection object
+ who was fetch by startTransaction pointing to this operation
+ getOperation will set the theTableId in the NdbOperation object.
+ synchronous
+******************************************************************************/
+NdbOperation*
+NdbConnection::getNdbOperation(const char* aTableName)
+{
+ if (theCommitStatus == Started){
+ NdbTableImpl* table = theNdb->theDictionary->getTable(aTableName);
+ if (table != 0){
+ return getNdbOperation(table);
+ } else {
+ setErrorCode(theNdb->theDictionary->getNdbError().code);
+ return NULL;
+ }//if
+ }
+
+ setOperationErrorCodeAbort(4114);
+
+ return NULL;
+}//NdbConnection::getNdbOperation()
+
+/*****************************************************************************
+NdbOperation* getNdbOperation(const char* anIndexName, const char* aTableName);
+
+Return Value Return a pointer to a NdbOperation object if getNdbOperation
+ was succesful.
+ Return NULL : In all other case.
+Parameters: anIndexName : Name of the index to use.
+ aTableName : Name of the database table.
+Remark: Get an operation from NdbOperation idlelist and get the
+ NdbConnection object
+ who was fetch by startTransaction pointing to this operation
+ getOperation will set the theTableId in the NdbOperation object.
+ synchronous
+******************************************************************************/
+NdbOperation*
+NdbConnection::getNdbOperation(const char* anIndexName, const char* aTableName)
+{
+ if ((theError.code == 0) &&
+ (theCommitStatus == Started)){
+ NdbIndexImpl* index =
+ theNdb->theDictionary->getIndex(anIndexName, aTableName);
+ NdbTableImpl* table = theNdb->theDictionary->getTable(aTableName);
+ NdbTableImpl* indexTable =
+ theNdb->theDictionary->getIndexTable(index, table);
+ if (indexTable != 0){
+ return getNdbOperation(indexTable);
+ } else {
+ setErrorCode(theNdb->theDictionary->getNdbError().code);
+ return NULL;
+ }//if
+ } else {
+ if (theError.code == 0) {
+ setOperationErrorCodeAbort(4114);
+ }//if
+
+ return NULL;
+ }//if
+}//NdbConnection::getNdbOperation()
+
+/*****************************************************************************
+NdbOperation* getNdbOperation(int aTableId);
+
+Return Value Return a pointer to a NdbOperation object if getNdbOperation
+ was succesful.
+ Return NULL: In all other case.
+Parameters: tableId : Id of the database table beeing deleted.
+Remark: Get an operation from NdbOperation object idlelist and
+ get the NdbConnection object who was fetch by
+ startTransaction pointing to this operation
+ getOperation will set the theTableId in the NdbOperation
+ object, synchronous.
+*****************************************************************************/
+NdbOperation*
+NdbConnection::getNdbOperation(NdbTableImpl * tab)
+{
+ NdbOperation* tOp;
+
+ if (theScanningOp != NULL){
+ setErrorCode(4607);
+ return NULL;
+ }
+
+ tOp = theNdb->getOperation();
+ if (tOp == NULL)
+ goto getNdbOp_error1;
+ if (theLastOpInList != NULL) {
+ theLastOpInList->next(tOp);
+ theLastOpInList = tOp;
+ } else {
+ theLastOpInList = tOp;
+ theFirstOpInList = tOp;
+ }//if
+ tOp->next(NULL);
+ if (tOp->init(tab, this) != -1) {
+ return tOp;
+ } else {
+ theNdb->releaseOperation(tOp);
+ }//if
+ return NULL;
+
+ getNdbOp_error1:
+ setOperationErrorCodeAbort(4000);
+ return NULL;
+}//NdbConnection::getNdbOperation()
+
+// NdbScanOperation
+/*****************************************************************************
+NdbScanOperation* getNdbScanOperation(const char* aTableName);
+
+Return Value Return a pointer to a NdbScanOperation object if getNdbScanOperation was succesful.
+ Return NULL : In all other case.
+Parameters: aTableName : Name of the database table.
+Remark: Get an operation from NdbScanOperation idlelist and get the NdbConnection object
+ who was fetch by startTransaction pointing to this operation
+ getOperation will set the theTableId in the NdbOperation object.synchronous
+******************************************************************************/
+NdbScanOperation*
+NdbConnection::getNdbScanOperation(const char* aTableName)
+{
+ if (theCommitStatus == Started){
+ NdbTableImpl* tab = theNdb->theDictionary->getTable(aTableName);
+ if (tab != 0){
+ return getNdbScanOperation(tab);
+ } else {
+ setOperationErrorCodeAbort(theNdb->theError.code);
+ return NULL;
+ }//if
+ }
+
+ setOperationErrorCodeAbort(4114);
+ return NULL;
+}//NdbConnection::getNdbScanOperation()
+
+/*****************************************************************************
+NdbScanOperation* getNdbScanOperation(const char* anIndexName, const char* aTableName);
+
+Return Value Return a pointer to a NdbScanOperation object if getNdbScanOperation was succesful.
+ Return NULL : In all other case.
+Parameters: anIndexName : Name of the index to use.
+ aTableName : Name of the database table.
+Remark: Get an operation from NdbScanOperation idlelist and get the NdbConnection object
+ who was fetch by startTransaction pointing to this operation
+ getOperation will set the theTableId in the NdbOperation object.synchronous
+******************************************************************************/
+NdbScanOperation*
+NdbConnection::getNdbScanOperation(const char* anIndexName, const char* aTableName)
+{
+ if (theCommitStatus == Started){
+ NdbIndexImpl* index =
+ theNdb->theDictionary->getIndex(anIndexName, aTableName);
+ NdbTableImpl* table = theNdb->theDictionary->getTable(aTableName);
+ NdbTableImpl* indexTable =
+ theNdb->theDictionary->getIndexTable(index, table);
+ if (indexTable != 0){
+ return getNdbScanOperation(indexTable);
+ } else {
+ setOperationErrorCodeAbort(theNdb->theError.code);
+ return NULL;
+ }//if
+ }
+
+ setOperationErrorCodeAbort(4114);
+ return NULL;
+}//NdbConnection::getNdbScanOperation()
+
+/*****************************************************************************
+NdbScanOperation* getNdbScanOperation(int aTableId);
+
+Return Value Return a pointer to a NdbOperation object if getNdbOperation was succesful.
+ Return NULL: In all other case.
+Parameters: tableId : Id of the database table beeing deleted.
+Remark: Get an operation from NdbScanOperation object idlelist and get the NdbConnection
+ object who was fetch by startTransaction pointing to this operation
+ getOperation will set the theTableId in the NdbOperation object, synchronous.
+*****************************************************************************/
+NdbScanOperation*
+NdbConnection::getNdbScanOperation(NdbTableImpl * tab)
+{
+ NdbScanOperation* tOp;
+
+ tOp = theNdb->getScanOperation();
+ if (tOp == NULL)
+ goto getNdbOp_error1;
+
+ // Link scan operation into list of cursor operations
+ if (m_theLastCursorOperation == NULL)
+ m_theFirstCursorOperation = m_theLastCursorOperation = tOp;
+ else {
+ m_theLastCursorOperation->next(tOp);
+ m_theLastCursorOperation = tOp;
+ }
+ tOp->next(NULL);
+ if (tOp->init(tab, this) != -1) {
+ return tOp;
+ } else {
+ theNdb->releaseScanOperation(tOp);
+ }//if
+ return NULL;
+
+getNdbOp_error1:
+ setOperationErrorCodeAbort(4000);
+ return NULL;
+}//NdbConnection::getNdbScanOperation()
+
+
+
+// IndexOperation
+/*****************************************************************************
+NdbIndexOperation* getNdbIndexOperation(const char* anIndexName,
+ const char* aTableName);
+
+Return Value Return a pointer to a NdbOperation object if getNdbScanOperation was succesful.
+ Return NULL : In all other case.
+Parameters: aTableName : Name of the database table.
+Remark: Get an operation from NdbScanOperation idlelist and get the NdbConnection object
+ who was fetch by startTransaction pointing to this operation
+ getOperation will set the theTableId in the NdbScanOperation object.synchronous
+******************************************************************************/
+NdbIndexOperation*
+NdbConnection::getNdbIndexOperation(const char* anIndexName,
+ const char* aTableName)
+{
+ if (theCommitStatus == Started) {
+ NdbTableImpl * table = theNdb->theDictionary->getTable(aTableName);
+ NdbIndexImpl * index = theNdb->theDictionary->getIndex(anIndexName,
+ aTableName);
+ if(table != 0 && index != 0){
+ return getNdbIndexOperation(index, table);
+ }
+
+ if(index == 0){
+ setOperationErrorCodeAbort(4243);
+ return NULL;
+ }
+
+ // table == 0
+ setOperationErrorCodeAbort(theNdb->theError.code);
+ return NULL;
+ }
+
+ setOperationErrorCodeAbort(4114);
+ return 0;
+}//NdbConnection::getNdbIndexOperation()
+
+/*****************************************************************************
+NdbIndexOperation* getNdbIndexOperation(int anIndexId, int aTableId);
+
+Return Value Return a pointer to a NdbIndexOperation object if getNdbIndexOperation was succesful.
+ Return NULL: In all other case.
+Parameters: tableId : Id of the database table beeing deleted.
+Remark: Get an operation from NdbIndexOperation object idlelist and get the NdbConnection
+ object who was fetch by startTransaction pointing to this operation
+ getOperation will set the theTableId in the NdbIndexOperation object, synchronous.
+*****************************************************************************/
+NdbIndexOperation*
+NdbConnection::getNdbIndexOperation(NdbIndexImpl * anIndex,
+ NdbTableImpl * aTable)
+{
+ NdbIndexOperation* tOp;
+
+ tOp = theNdb->getIndexOperation();
+ if (tOp == NULL)
+ goto getNdbOp_error1;
+ if (theLastOpInList != NULL) {
+ theLastOpInList->next(tOp);
+ theLastOpInList = tOp;
+ } else {
+ theLastOpInList = tOp;
+ theFirstOpInList = tOp;
+ }//if
+ tOp->next(NULL);
+ if (tOp->indxInit(anIndex, aTable, this)!= -1) {
+ return tOp;
+ } else {
+ theNdb->releaseOperation(tOp);
+ }//if
+ return NULL;
+
+ getNdbOp_error1:
+ setOperationErrorCodeAbort(4000);
+ return NULL;
+}//NdbConnection::getNdbIndexOperation()
+
+/*******************************************************************************
+int receiveDIHNDBTAMPER(NdbApiSignal* aSignal)
+
+Return Value: Return 0 : receiveDIHNDBTAMPER was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark: Sets theRestartGCI in the NDB object.
+*******************************************************************************/
+int
+NdbConnection::receiveDIHNDBTAMPER(NdbApiSignal* aSignal)
+{
+ if (theStatus != Connecting) {
+ return -1;
+ } else {
+ theNdb->RestartGCI((Uint32)aSignal->readData(2));
+ theStatus = Connected;
+ }//if
+ return 0;
+}//NdbConnection::receiveDIHNDBTAMPER()
+
+/*******************************************************************************
+int receiveTCSEIZECONF(NdbApiSignal* aSignal);
+
+Return Value: Return 0 : receiveTCSEIZECONF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark: Sets TC Connect pointer at reception of TCSEIZECONF.
+*******************************************************************************/
+int
+NdbConnection::receiveTCSEIZECONF(NdbApiSignal* aSignal)
+{
+ if (theStatus != Connecting)
+ {
+ return -1;
+ } else
+ {
+ theTCConPtr = (Uint32)aSignal->readData(2);
+ theStatus = Connected;
+ }
+ return 0;
+}//NdbConnection::receiveTCSEIZECONF()
+
+/*******************************************************************************
+int receiveTCSEIZEREF(NdbApiSignal* aSignal);
+
+Return Value: Return 0 : receiveTCSEIZEREF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark: Sets TC Connect pointer.
+*******************************************************************************/
+int
+NdbConnection::receiveTCSEIZEREF(NdbApiSignal* aSignal)
+{
+ if (theStatus != Connecting)
+ {
+ return -1;
+ } else
+ {
+ theStatus = ConnectFailure;
+ theNdb->theError.code = aSignal->readData(2);
+ return 0;
+ }
+}//NdbConnection::receiveTCSEIZEREF()
+
+/*******************************************************************************
+int receiveTCRELEASECONF(NdbApiSignal* aSignal);
+
+Return Value: Return 0 : receiveTCRELEASECONF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark: DisConnect TC Connect pointer to NDBAPI.
+*******************************************************************************/
+int
+NdbConnection::receiveTCRELEASECONF(NdbApiSignal* aSignal)
+{
+ if (theStatus != DisConnecting)
+ {
+ return -1;
+ } else
+ {
+ theStatus = NotConnected;
+ }
+ return 0;
+}//NdbConnection::receiveTCRELEASECONF()
+
+/*******************************************************************************
+int receiveTCRELEASEREF(NdbApiSignal* aSignal);
+
+Return Value: Return 0 : receiveTCRELEASEREF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark: DisConnect TC Connect pointer to NDBAPI Failure.
+*******************************************************************************/
+int
+NdbConnection::receiveTCRELEASEREF(NdbApiSignal* aSignal)
+{
+ if (theStatus != DisConnecting) {
+ return -1;
+ } else {
+ theStatus = ConnectFailure;
+ theNdb->theError.code = aSignal->readData(2);
+ return 0;
+ }//if
+}//NdbConnection::receiveTCRELEASEREF()
+
+/******************************************************************************
+int receiveTC_COMMITCONF(NdbApiSignal* aSignal);
+
+Return Value: Return 0 : receiveTC_COMMITCONF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark:
+******************************************************************************/
+int
+NdbConnection::receiveTC_COMMITCONF(const TcCommitConf * commitConf)
+{
+ if(theStatus != Connected){
+ return -1;
+ }
+ theCommitStatus = Committed;
+ theCompletionStatus = CompletedSuccess;
+ return 0;
+}//NdbConnection::receiveTC_COMMITCONF()
+
+/******************************************************************************
+int receiveTC_COMMITREF(NdbApiSignal* aSignal);
+
+Return Value: Return 0 : receiveTC_COMMITREF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark:
+******************************************************************************/
+int
+NdbConnection::receiveTC_COMMITREF(NdbApiSignal* aSignal)
+{
+ if(theStatus != Connected){
+ return -1;
+ }
+ const TcCommitRef * const ref = CAST_CONSTPTR(TcCommitRef, aSignal->getDataPtr());
+ setOperationErrorCodeAbort(ref->errorCode);
+ theCommitStatus = Aborted;
+ theCompletionStatus = CompletedFailure;
+ return 0;
+}//NdbConnection::receiveTC_COMMITREF()
+
+/*******************************************************************************
+int receiveTCROLLBACKCONF(NdbApiSignal* aSignal);
+
+Return Value: Return 0 : receiveTCROLLBACKCONF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark:
+*******************************************************************************/
+int
+NdbConnection::receiveTCROLLBACKCONF(NdbApiSignal* aSignal)
+{
+ if(theStatus != Connected){
+ return -1;
+ }
+ theCommitStatus = Aborted;
+ theCompletionStatus = CompletedSuccess;
+ return 0;
+}//NdbConnection::receiveTCROLLBACKCONF()
+
+/*******************************************************************************
+int receiveTCROLLBACKREF(NdbApiSignal* aSignal);
+
+Return Value: Return 0 : receiveTCROLLBACKREF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark:
+*******************************************************************************/
+int
+NdbConnection::receiveTCROLLBACKREF(NdbApiSignal* aSignal)
+{
+ if(theStatus != Connected){
+ return -1;
+ }
+ setOperationErrorCodeAbort(aSignal->readData(2));
+ theCommitStatus = Aborted;
+ theCompletionStatus = CompletedFailure;
+ return 0;
+}//NdbConnection::receiveTCROLLBACKREF()
+
+/*****************************************************************************
+int receiveTCROLLBACKREP( NdbApiSignal* aSignal)
+
+Return Value: Return 0 : send was succesful.
+ Return -1: In all other case.
+Parameters: aSignal: the signal object that contains the
+ TCROLLBACKREP signal from TC.
+Remark: Handles the reception of the ROLLBACKREP signal.
+*****************************************************************************/
+int
+NdbConnection::receiveTCROLLBACKREP( NdbApiSignal* aSignal)
+{
+ Uint64 tRecTransId, tCurrTransId;
+ Uint32 tTmp1, tTmp2;
+
+ if (theStatus != Connected) {
+ return -1;
+ }//if
+/*****************************************************************************
+Check that we are expecting signals from this transaction and that it doesn't
+belong to a transaction already completed. Simply ignore messages from other
+transactions.
+******************************************************************************/
+ tTmp1 = aSignal->readData(2);
+ tTmp2 = aSignal->readData(3);
+ tRecTransId = (Uint64)tTmp1 + ((Uint64)tTmp2 << 32);
+ tCurrTransId = this->getTransactionId();
+ if (tCurrTransId != tRecTransId) {
+ return -1;
+ }//if
+ theError.code = aSignal->readData(4); // Override any previous errors
+
+/**********************************************************************/
+/* A serious error has occured. This could be due to deadlock or */
+/* lack of resources or simply a programming error in NDB. This */
+/* transaction will be aborted. Actually it has already been */
+/* and we only need to report completion and return with the */
+/* error code to the application. */
+/**********************************************************************/
+ theCompletionStatus = CompletedFailure;
+ theCommitStatus = Aborted;
+ return 0;
+}//NdbConnection::receiveTCROLLBACKREP()
+
+/*******************************************************************************
+int receiveTCKEYCONF(NdbApiSignal* aSignal, Uint32 long_short_ind);
+
+Return Value: Return 0 : receiveTCKEYCONF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark:
+*******************************************************************************/
+int
+NdbConnection::receiveTCKEYCONF(const TcKeyConf * keyConf, Uint32 aDataLength)
+{
+ Uint64 tRecTransId;
+ NdbOperation* tOp;
+ Uint32 tConditionFlag;
+
+ const Uint32 tTemp = keyConf->confInfo;
+ const Uint32 tTmp1 = keyConf->transId1;
+ const Uint32 tTmp2 = keyConf->transId2;
+/******************************************************************************
+Check that we are expecting signals from this transaction and that it
+doesn't belong to a transaction already completed. Simply ignore messages
+from other transactions.
+******************************************************************************/
+ tRecTransId = (Uint64)tTmp1 + ((Uint64)tTmp2 << 32);
+
+ const Uint32 tNoOfOperations = TcKeyConf::getNoOfOperations(tTemp);
+ const Uint32 tCommitFlag = TcKeyConf::getCommitFlag(tTemp);
+ tConditionFlag = (Uint32)(((aDataLength - 5) >> 1) - tNoOfOperations);
+ tConditionFlag |= (Uint32)(tNoOfOperations > 10);
+ tConditionFlag |= (Uint32)(tNoOfOperations <= 0);
+ tConditionFlag |= (Uint32)(theTransactionId - tRecTransId);
+ tConditionFlag |= (Uint32)(theStatus - Connected);
+
+ if (tConditionFlag == 0) {
+ const Uint32* tPtr = (Uint32 *)&keyConf->operations[0];
+ for (Uint32 i = 0; i < tNoOfOperations ; i++) {
+ tOp = theNdb->void2rec_op(theNdb->int2void(*tPtr));
+ tPtr++;
+ const Uint32 tAttrInfoLen = *tPtr;
+ tPtr++;
+ if (tOp && tOp->checkMagicNumber() != -1) {
+ tOp->TCOPCONF(tAttrInfoLen);
+ } else {
+ return -1;
+ }//if
+ }//for
+ Uint32 tNoComp = theNoOfOpCompleted;
+ Uint32 tNoSent = theNoOfOpSent;
+ Uint32 tGCI = keyConf->gci;
+ if (tCommitFlag == 1) {
+ theCommitStatus = Committed;
+ theGlobalCheckpointId = tGCI;
+ } else if ((tNoComp >= tNoSent) &&
+ (theLastExecOpInList->theCommitIndicator == 1)){
+/**********************************************************************/
+// We sent the transaction with Commit flag set and received a CONF with
+// no Commit flag set. This is clearly an anomaly.
+/**********************************************************************/
+ theError.code = 4011;
+ theCompletionStatus = CompletedFailure;
+ theCommitStatus = Aborted;
+ return 0;
+ }//if
+ if (tNoComp >= tNoSent) {
+ return 0; // No more operations to wait for
+ }//if
+ // Not completed the reception yet.
+ }//if
+ return -1;
+}//NdbConnection::receiveTCKEYCONF()
+
+/*****************************************************************************
+int receiveTCKEY_FAILCONF( NdbApiSignal* aSignal)
+
+Return Value: Return 0 : receive was completed.
+ Return -1: In all other case.
+Parameters: aSignal: the signal object that contains the
+ TCKEY_FAILCONF signal from TC.
+Remark: Handles the reception of the TCKEY_FAILCONF signal.
+*****************************************************************************/
+int
+NdbConnection::receiveTCKEY_FAILCONF(const TcKeyFailConf * failConf)
+{
+ Uint64 tRecTransId, tCurrTransId;
+ Uint32 tTmp1, tTmp2;
+ NdbOperation* tOp;
+ if (theStatus != Connected) {
+ return -1;
+ }//if
+ /*
+ Check that we are expecting signals from this transaction and that it
+ doesn't belong to a transaction already completed. Simply ignore
+ messages from other transactions.
+ */
+ tTmp1 = failConf->transId1;
+ tTmp2 = failConf->transId2;
+ tRecTransId = (Uint64)tTmp1 + ((Uint64)tTmp2 << 32);
+ tCurrTransId = this->getTransactionId();
+ if (tCurrTransId != tRecTransId) {
+ return -1;
+ }//if
+ /*
+ A node failure of the TC node occured. The transaction has
+ been committed.
+ */
+ theCommitStatus = Committed;
+ tOp = theFirstExecOpInList;
+ while (tOp != NULL) {
+ /*
+ Check if the transaction expected read values...
+ If it did some of them might have gotten lost even if we succeeded
+ in committing the transaction.
+ */
+ if (tOp->theAI_ElementLen != 0) {
+ theCompletionStatus = CompletedFailure;
+ setOperationErrorCodeAbort(4115);
+ break;
+ }//if
+ if (tOp->theCurrentRecAttr != NULL) {
+ theCompletionStatus = CompletedFailure;
+ setOperationErrorCodeAbort(4115);
+ break;
+ }//if
+ tOp = tOp->next();
+ }//while
+ theReleaseOnClose = true;
+ return 0;
+}//NdbConnection::receiveTCKEY_FAILCONF()
+
+/*************************************************************************
+int receiveTCKEY_FAILREF( NdbApiSignal* aSignal)
+
+Return Value: Return 0 : receive was completed.
+ Return -1: In all other case.
+Parameters: aSignal: the signal object that contains the
+ TCKEY_FAILREF signal from TC.
+Remark: Handles the reception of the TCKEY_FAILREF signal.
+**************************************************************************/
+int
+NdbConnection::receiveTCKEY_FAILREF(NdbApiSignal* aSignal)
+{
+ Uint64 tRecTransId, tCurrTransId;
+ Uint32 tTmp1, tTmp2;
+
+ if (theStatus != Connected) {
+ return -1;
+ }//if
+ /*
+ Check that we are expecting signals from this transaction and
+ that it doesn't belong to a transaction already
+ completed. Simply ignore messages from other transactions.
+ */
+ tTmp1 = aSignal->readData(2);
+ tTmp2 = aSignal->readData(3);
+ tRecTransId = (Uint64)tTmp1 + ((Uint64)tTmp2 << 32);
+ tCurrTransId = this->getTransactionId();
+ if (tCurrTransId != tRecTransId) {
+ return -1;
+ }//if
+ /*
+ We received an indication of that this transaction was aborted due to a
+ node failure.
+ */
+ if (theSendStatus == sendTC_ROLLBACK) {
+ /*
+ We were in the process of sending a rollback anyways. We will
+ report it as a success.
+ */
+ theCompletionStatus = CompletedSuccess;
+ } else {
+ theCompletionStatus = CompletedFailure;
+ theError.code = 4031;
+ }//if
+ theReleaseOnClose = true;
+ theCommitStatus = Aborted;
+ return 0;
+}//NdbConnection::receiveTCKEY_FAILREF()
+
+/*******************************************************************************
+int receiveTCINDXCONF(NdbApiSignal* aSignal, Uint32 long_short_ind);
+
+Return Value: Return 0 : receiveTCINDXCONF was successful.
+ Return -1: In all other case.
+Parameters: aSignal: The signal object pointer.
+Remark:
+*******************************************************************************/
+int
+NdbConnection::receiveTCINDXCONF(const TcIndxConf * indxConf, Uint32 aDataLength)
+{
+ Uint64 tRecTransId;
+ Uint32 tConditionFlag;
+
+ const Uint32 tTemp = indxConf->confInfo;
+ const Uint32 tTmp1 = indxConf->transId1;
+ const Uint32 tTmp2 = indxConf->transId2;
+/******************************************************************************
+Check that we are expecting signals from this transaction and that it
+doesn't belong to a transaction already completed. Simply ignore messages
+from other transactions.
+******************************************************************************/
+ tRecTransId = (Uint64)tTmp1 + ((Uint64)tTmp2 << 32);
+
+ const Uint32 tNoOfOperations = TcIndxConf::getNoOfOperations(tTemp);
+ const Uint32 tCommitFlag = TcKeyConf::getCommitFlag(tTemp);
+
+ tConditionFlag = (Uint32)(((aDataLength - 5) >> 1) - tNoOfOperations);
+ tConditionFlag |= (Uint32)(tNoOfOperations > 10);
+ tConditionFlag |= (Uint32)(tNoOfOperations <= 0);
+ tConditionFlag |= (Uint32)(theTransactionId - tRecTransId);
+ tConditionFlag |= (Uint32)(theStatus - Connected);
+
+ if (tConditionFlag == 0) {
+ const Uint32* tPtr = (Uint32 *)&indxConf->operations[0];
+ for (Uint32 i = 0; i < tNoOfOperations ; i++) {
+ NdbIndexOperation* tOp = theNdb->void2rec_iop(theNdb->int2void(*tPtr));
+ tPtr++;
+ const Uint32 tAttrInfoLen = *tPtr;
+ tPtr++;
+ if (tOp && tOp->checkMagicNumber() != -1) {
+ tOp->TCOPCONF(tAttrInfoLen);
+ } else {
+ return -1;
+ }//if
+ }//for
+ Uint32 tNoComp = theNoOfOpCompleted;
+ Uint32 tNoSent = theNoOfOpSent;
+ Uint32 tGCI = indxConf->gci;
+ if (tCommitFlag == 1) {
+ theCommitStatus = Committed;
+ theGlobalCheckpointId = tGCI;
+ } else if ((tNoComp >= tNoSent) &&
+ (theLastExecOpInList->theCommitIndicator == 1)){
+/**********************************************************************/
+// We sent the transaction with Commit flag set and received a CONF with
+// no Commit flag set. This is clearly an anomaly.
+/**********************************************************************/
+ theError.code = 4011;
+ theCompletionStatus = CompletedFailure;
+ theCommitStatus = Aborted;
+ return 0;
+ }//if
+ if (tNoComp >= tNoSent) {
+ return 0; // No more operations to wait for
+ }//if
+ // Not completed the reception yet.
+ }//if
+ return -1;
+}//NdbConnection::receiveTCINDXCONF()
+
+/*****************************************************************************
+int receiveTCINDXREF( NdbApiSignal* aSignal)
+
+Return Value: Return 0 : send was succesful.
+ Return -1: In all other case.
+Parameters: aSignal: the signal object that contains the
+ TCINDXREF signal from TC.
+Remark: Handles the reception of the TCINDXREF signal.
+*****************************************************************************/
+int
+NdbConnection::receiveTCINDXREF( NdbApiSignal* aSignal)
+{
+ Uint64 tRecTransId, tCurrTransId;
+ Uint32 tTmp1, tTmp2;
+
+ if (theStatus != Connected) {
+ return -1;
+ }//if
+/*****************************************************************************
+Check that we are expecting signals from this transaction and that it doesn't
+belong to a transaction already completed. Simply ignore messages from other
+transactions.
+******************************************************************************/
+ tTmp1 = aSignal->readData(2);
+ tTmp2 = aSignal->readData(3);
+ tRecTransId = (Uint64)tTmp1 + ((Uint64)tTmp2 << 32);
+ tCurrTransId = this->getTransactionId();
+ if (tCurrTransId != tRecTransId) {
+ return -1;
+ }//if
+ theError.code = aSignal->readData(4); // Override any previous errors
+
+/**********************************************************************/
+/* A serious error has occured. This could be due to deadlock or */
+/* lack of resources or simply a programming error in NDB. This */
+/* transaction will be aborted. Actually it has already been */
+/* and we only need to report completion and return with the */
+/* error code to the application. */
+/**********************************************************************/
+ theCompletionStatus = CompletedFailure;
+ theCommitStatus = Aborted;
+ return 0;
+}//NdbConnection::receiveTCINDXREF()
+
+/*******************************************************************************
+int OpCompletedFailure();
+
+Return Value: Return 0 : OpCompleteSuccess was successful.
+ Return -1: In all other case.
+Parameters: aErrorCode: The error code.
+Remark: An operation was completed with failure.
+*******************************************************************************/
+int
+NdbConnection::OpCompleteFailure()
+{
+ Uint32 tNoComp = theNoOfOpCompleted;
+ Uint32 tNoSent = theNoOfOpSent;
+ theCompletionStatus = CompletedFailure;
+ tNoComp++;
+ theNoOfOpCompleted = tNoComp;
+ if (tNoComp == tNoSent) {
+ //------------------------------------------------------------------------
+ //If the transaction consists of only simple reads we can set
+ //Commit state Aborted. Otherwise this simple operation cannot
+ //decide the success of the whole transaction since a simple
+ //operation is not really part of that transaction.
+ //------------------------------------------------------------------------
+ if (theSimpleState == 1) {
+ theCommitStatus = Aborted;
+ }//if
+ return 0; // Last operation received
+ } else if (tNoComp > tNoSent) {
+ setOperationErrorCodeAbort(4113); // Too many operations,
+ // stop waiting for more
+ return 0;
+ } else {
+ return -1; // Continue waiting for more signals
+ }//if
+}//NdbConnection::OpCompleteFailure()
+
+/******************************************************************************
+int OpCompleteSuccess();
+
+Return Value: Return 0 : OpCompleteSuccess was successful.
+ Return -1: In all other case.
+Remark: An operation was completed with success.
+*******************************************************************************/
+int
+NdbConnection::OpCompleteSuccess()
+{
+ Uint32 tNoComp = theNoOfOpCompleted;
+ Uint32 tNoSent = theNoOfOpSent;
+ tNoComp++;
+ theNoOfOpCompleted = tNoComp;
+ if (tNoComp == tNoSent) { // Last operation completed
+ if (theSimpleState == 1) {
+ theCommitStatus = Committed;
+ }//if
+ return 0;
+ } else if (tNoComp < tNoSent) {
+ return -1; // Continue waiting for more signals
+ } else {
+ setOperationErrorCodeAbort(4113); // Too many operations,
+ // stop waiting for more
+ theCompletionStatus = CompletedFailure;
+ return 0;
+ }//if
+}//NdbConnection::OpCompleteSuccess()
+
+/******************************************************************************
+ int getGCI();
+
+Remark: Get global checkpoint identity of the transaction
+*******************************************************************************/
+int
+NdbConnection::getGCI()
+{
+ if (theCommitStatus == Committed) {
+ return theGlobalCheckpointId;
+ }//if
+ return 0;
+}//NdbConnection::getGCI()
+
+/*******************************************************************************
+Uint64 getTransactionId(void);
+
+Remark: Get the transaction identity.
+*******************************************************************************/
+Uint64
+NdbConnection::getTransactionId()
+{
+ return theTransactionId;
+}//NdbConnection::getTransactionId()
+
+CommitStatusType
+NdbConnection::commitStatus()
+{
+ return theCommitStatus;
+}//NdbConnection::commitStatus()
+
+int
+NdbConnection::getNdbErrorLine()
+{
+ return theErrorLine;
+}
+
+NdbOperation*
+NdbConnection::getNdbErrorOperation()
+{
+ return theErrorOperation;
+}//NdbConnection::getNdbErrorOperation()
+
+const NdbOperation *
+NdbConnection::getNextCompletedOperation(const NdbOperation * current) const {
+ if(current == 0)
+ return theCompletedFirstOp;
+ return current->theNext;
+}
+
+#ifdef VM_TRACE
+#define CASE(x) case x: ndbout << " " << #x; break
+void
+NdbConnection::printState()
+{
+ ndbout << "con=" << hex << this << dec;
+ ndbout << " node=" << getConnectedNodeId();
+ switch (theStatus) {
+ CASE(NotConnected);
+ CASE(Connecting);
+ CASE(Connected);
+ CASE(DisConnecting);
+ CASE(ConnectFailure);
+ default: ndbout << theStatus;
+ }
+ switch (theListState) {
+ CASE(NotInList);
+ CASE(InPreparedList);
+ CASE(InSendList);
+ CASE(InCompletedList);
+ default: ndbout << theListState;
+ }
+ switch (theSendStatus) {
+ CASE(NotInit);
+ CASE(InitState);
+ CASE(sendOperations);
+ CASE(sendCompleted);
+ CASE(sendCOMMITstate);
+ CASE(sendABORT);
+ CASE(sendABORTfail);
+ CASE(sendTC_ROLLBACK);
+ CASE(sendTC_COMMIT);
+ CASE(sendTC_OP);
+ default: ndbout << theSendStatus;
+ }
+ switch (theCommitStatus) {
+ CASE(NotStarted);
+ CASE(Started);
+ CASE(Committed);
+ CASE(Aborted);
+ CASE(NeedAbort);
+ default: ndbout << theCommitStatus;
+ }
+ switch (theCompletionStatus) {
+ CASE(NotCompleted);
+ CASE(CompletedSuccess);
+ CASE(CompletedFailure);
+ CASE(DefinitionFailure);
+ default: ndbout << theCompletionStatus;
+ }
+ ndbout << endl;
+}
+#undef CASE
+#endif
diff --git a/ndb/src/ndbapi/NdbConnectionScan.cpp b/ndb/src/ndbapi/NdbConnectionScan.cpp
new file mode 100644
index 00000000000..67f07d2a8c0
--- /dev/null
+++ b/ndb/src/ndbapi/NdbConnectionScan.cpp
@@ -0,0 +1,572 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/*****************************************************************************
+ * Name: NdbConnectionScan.cpp
+ * Include:
+ * Link:
+ * Author: UABRONM MikaelRonström UAB/M/MT
+ * QABJKAM Jonas Kamf UAB/M/MT
+ * Date: 2000-06-12
+ * Version: 0.1
+ * Description: Interface between Application and NDB
+ * Documentation:
+ * Adjust: 2000-06-12 UABRONM First version.
+ ****************************************************************************/
+#include "Ndb.hpp"
+#include "NdbConnection.hpp"
+#include "NdbOperation.hpp"
+#include "NdbScanOperation.hpp"
+#include "NdbScanReceiver.hpp"
+#include "NdbApiSignal.hpp"
+#include "TransporterFacade.hpp"
+#include "NdbUtil.hpp"
+#include "API.hpp"
+#include "NdbImpl.hpp"
+
+#include <signaldata/ScanTab.hpp>
+
+#include <NdbOut.hpp>
+#include <assert.h>
+
+// time out for next scan result (-1 is infinite)
+// XXX should change default only if non-trivial interpreted program is used
+#define WAITFOR_SCAN_TIMEOUT 120000
+
+
+/*****************************************************************************
+ * int executeScan();
+ *
+ * 1. Check that the transaction is started and other important preconditions
+ * 2. Tell the kernel to start scanning by sending one SCAN_TABREQ, if
+ * parallelism is greater than 16 also send one SCAN_TABINFO for each
+ * additional 16
+ * Define which attributes to scan in ATTRINFO, this signal also holds the
+ * interpreted program
+ * 3. Wait for the answer of the SCAN_TABREQ. This is either a SCAN_TABCONF if
+ * the scan was correctly defined and a SCAN_TABREF if the scan couldn't
+ * be started.
+ * 4. Check the result, if scan was not started return -1
+ *
+ ****************************************************************************/
+int
+NdbConnection::executeScan(){
+ if (theTransactionIsStarted == true){ // Transaction already started.
+ setErrorCode(4600);
+ return -1;
+ }
+ if (theStatus != Connected) { // Lost connection
+ setErrorCode(4601);
+ return -1;
+ }
+ if (theScanningOp == NULL){
+ setErrorCode(4602); // getNdbOperation must be called before executeScan
+ return -1;
+ }
+ TransporterFacade* tp = TransporterFacade::instance();
+ theNoOfOpCompleted = 0;
+ theNoOfSCANTABCONFRecv = 0;
+ tp->lock_mutex();
+ if (tp->get_node_alive(theDBnode) &&
+ (tp->getNodeSequence(theDBnode) == theNodeSequence)) {
+ if (tp->check_send_size(theDBnode, get_send_size())) {
+ theTransactionIsStarted = true;
+ if (sendScanStart() == -1){
+ tp->unlock_mutex();
+ return -1;
+ }//if
+ theNdb->theWaiter.m_node = theDBnode;
+ theNdb->theWaiter.m_state = WAIT_SCAN;
+ int res = theNdb->receiveResponse(WAITFOR_SCAN_TIMEOUT);
+ if (res == 0) {
+ return 0;
+ } else {
+ if (res == -1) {
+ setErrorCode(4008);
+ } else if (res == -2) {
+ theTransactionIsStarted = false;
+ theReleaseOnClose = true;
+ setErrorCode(4028);
+ } else {
+ ndbout << "Impossible return from receiveResponse in executeScan";
+ ndbout << endl;
+ abort();
+ }//if
+ theCommitStatus = Aborted;
+ return -1;
+ }//if
+ } else {
+ TRACE_DEBUG("Start a scan with send buffer full attempted");
+ setErrorCode(4022);
+ theCommitStatus = Aborted;
+ }//if
+ } else {
+ if (!(tp->get_node_stopping(theDBnode) &&
+ (tp->getNodeSequence(theDBnode) == theNodeSequence))) {
+ TRACE_DEBUG("The node is hard dead when attempting to start a scan");
+ setErrorCode(4029);
+ theReleaseOnClose = true;
+ } else {
+ TRACE_DEBUG("The node is stopping when attempting to start a scan");
+ setErrorCode(4030);
+ }//if
+ theCommitStatus = Aborted;
+ }//if
+ tp->unlock_mutex();
+ return -1;
+}
+
+/******************************************************************************
+ * int nextScanResult();
+ * Remark:
+ * This method is used to distribute data received to the application.
+ * Iterate through the list and search for operations that haven't
+ * been distributed yet (status != Finished).
+ * If there are no more operations/records still waiting to be exececuted
+ * we have to send SCAN_NEXTREQ to fetch next set of records.
+ *
+ * TODO - This function should be able to return a value indicating if
+ * there are any more records already fetched from memory or if it has to
+ * ask the db for more. This would mean we could get better performance when
+ * takeOver is used wince we can take over all ops already fetched, put them
+ * in another trans and send them of to the db when there are no more records
+ * already fetched. Maybe use a new argument to the function for this
+******************************************************************************/
+int
+NdbConnection::nextScanResult(bool fetchAllowed){
+
+ if (theTransactionIsStarted != true){ // Transaction not started.
+ setErrorCode(4601);
+ return -1;
+ }
+ // Scan has finished ok but no operations recived = empty recordset.
+ if(theScanFinished == true){
+ return 1; // No more records
+ }
+ if (theStatus != Connected){// Lost connection
+ setErrorCode(4601);
+ return -1;
+ }
+ // Something went wrong, probably we got a SCAN_TABREF earlier.
+ if (theCompletionStatus == CompletedFailure) {
+ return -1;
+ }
+ if (theNoOfOpCompleted == theNoOfOpFetched) {
+ // There are no more records cached in NdbApi
+ if (fetchAllowed == true){
+ // Get some more records from db
+
+ if (fetchNextScanResult() == -1){
+ return -1;
+ }
+ if (theScanFinished == true) { // The scan has finished.
+ return 1; // 1 = No more records
+ }
+ if (theCompletionStatus == CompletedFailure) {
+ return -1; // Something went wrong, probably we got a SCAN_TABREF.
+ }
+ } else {
+ // There where no more cached records in NdbApi
+ // and we where not allowed to go to db and ask for
+ // more
+ return 2;
+ }
+ }
+
+ // It's not allowed to come here without any cached records
+ if (theCurrentScanRec == NULL){
+#ifdef VM_TRACE
+ ndbout << "nextScanResult("<<fetchAllowed<<")"<<endl
+ << " theTransactionIsStarted = " << theTransactionIsStarted << endl
+ << " theScanFinished = " << theScanFinished << endl
+ << " theCommitStatus = " << theCommitStatus << endl
+ << " theStatus = " << theStatus << endl
+ << " theCompletionStatus = " << theCompletionStatus << endl
+ << " theNoOfOpCompleted = " << theNoOfOpCompleted << endl
+ << " theNoOfOpFetched = " << theNoOfOpFetched << endl
+ << " theScanningOp = " << theScanningOp << endl
+ << " theNoOfSCANTABCONFRecv = "<< theNoOfSCANTABCONFRecv << endl
+ << " theNdb->theWaiter.m_node = " <<theNdb->theWaiter.m_node<<endl
+ << " theNdb->theWaiter.m_state = " << theNdb->theWaiter.m_state << endl;
+ abort();
+#endif
+ return -1;
+ }
+
+ // Execute the saved signals for this operation.
+ NdbScanReceiver* tScanRec = theCurrentScanRec;
+ theScanningOp->theCurrRecAI_Len = 0;
+ theScanningOp->theCurrentRecAttr = theScanningOp->theFirstRecAttr;
+ if(tScanRec->executeSavedSignals() != 0)
+ return -1;
+ theNoOfOpCompleted++;
+ // Remember for next iteration and takeOverScanOp
+ thePreviousScanRec = tScanRec;
+ theCurrentScanRec = tScanRec->next();
+ return 0; // 0 = There are more rows to be fetched.
+}
+
+/******************************************************************************
+ * int stopScan()
+ * Remark: By sending SCAN_NEXTREQ with data word 2 set to TRUE we
+ * abort the scan process.
+ *****************************************************************************/
+int
+NdbConnection::stopScan()
+{
+ if(theScanFinished == true){
+ return 0;
+ }
+ if (theCompletionStatus == CompletedFailure){
+ return 0;
+ }
+
+ if (theScanningOp == 0){
+ return 0;
+ }
+
+ theNoOfOpCompleted = 0;
+ theNoOfSCANTABCONFRecv = 0;
+ theScanningOp->prepareNextScanResult();
+ return sendScanNext(1);
+}
+
+
+/********************************************************************
+ * int sendScanStart()
+ *
+ * Send the signals reuired to define and start the scan
+ * 1. Send SCAN_TABREQ
+ * 2. Send SCAN_TABINFO(if any, parallelism must be > 16)
+ * 3. Send ATTRINFO signals
+ *
+ * Returns -1 if an error occurs otherwise 0.
+ *
+ ********************************************************************/
+int
+NdbConnection::sendScanStart(){
+
+ /***** 0. Prepare signals ******************/
+ // This might modify variables and signals
+ if(theScanningOp->prepareSendScan(theTCConPtr,
+ theTransactionId) == -1)
+ return -1;
+
+ /***** 1. Send SCAN_TABREQ **************/
+ /***** 2. Send SCAN_TABINFO *************/
+ /***** 3. Send ATTRINFO signals *********/
+ if (theScanningOp->doSendScan(theDBnode) == -1)
+ return -1;
+ return 0;
+}
+
+
+int
+NdbConnection::fetchNextScanResult(){
+ theNoOfOpCompleted = 0;
+ theNoOfSCANTABCONFRecv = 0;
+ theScanningOp->prepareNextScanResult();
+ return sendScanNext(0);
+}
+
+
+
+/***********************************************************
+ * int sendScanNext(int stopScanFlag)
+ *
+ * ************************************************************/
+int NdbConnection::sendScanNext(bool stopScanFlag){
+ NdbApiSignal tSignal(theNdb->theMyRef);
+ Uint32 tTransId1, tTransId2;
+ tSignal.setSignal(GSN_SCAN_NEXTREQ);
+ tSignal.setData(theTCConPtr, 1);
+ // Set the stop flag in word 2(1 = stop)
+ Uint32 tStopValue;
+ tStopValue = stopScanFlag == true ? 1 : 0;
+ tSignal.setData(tStopValue, 2);
+ tTransId1 = (Uint32) theTransactionId;
+ tTransId2 = (Uint32) (theTransactionId >> 32);
+ tSignal.setData(tTransId1, 3);
+ tSignal.setData(tTransId2, 4);
+ tSignal.setLength(4);
+ Uint32 conn_seq = theNodeSequence;
+ int return_code = theNdb->sendRecSignal(theDBnode,
+ WAIT_SCAN,
+ &tSignal,
+ conn_seq);
+ if (return_code == 0) {
+ return 0;
+ } else if (return_code == -1) { // Time-out
+ TRACE_DEBUG("Time-out when sending sendScanNext");
+ setErrorCode(4024);
+ theTransactionIsStarted = false;
+ theReleaseOnClose = true;
+ theCommitStatus = Aborted;
+ } else if (return_code == -2) { // Node failed
+ TRACE_DEBUG("Node failed when sendScanNext");
+ setErrorCode(4027);
+ theTransactionIsStarted = false;
+ theReleaseOnClose = true;
+ theCommitStatus = Aborted;
+ } else if (return_code == -3) {
+ TRACE_DEBUG("Send failed when sendScanNext");
+ setErrorCode(4033);
+ theTransactionIsStarted = false;
+ theReleaseOnClose = true;
+ theCommitStatus = Aborted;
+ } else if (return_code == -4) {
+ TRACE_DEBUG("Send buffer full when sendScanNext");
+ setErrorCode(4032);
+ } else if (return_code == -5) {
+ TRACE_DEBUG("Node stopping when sendScanNext");
+ setErrorCode(4034);
+ } else {
+ ndbout << "Impossible return from sendRecSignal" << endl;
+ abort();
+ }//if
+ return -1;
+}
+
+
+/***************************************************************************
+ * int receiveSCAN_TABREF(NdbApiSignal* aSignal)
+ *
+ * This means the scan could not be started, set status(s) to indicate
+ * the failure
+ *
+ ****************************************************************************/
+int
+NdbConnection::receiveSCAN_TABREF(NdbApiSignal* aSignal){
+ const ScanTabRef * const scanTabRef = CAST_CONSTPTR(ScanTabRef, aSignal->getDataPtr());
+ if (theStatus != Connected){
+#ifdef VM_TRACE
+ ndbout << "SCAN_TABREF dropped, theStatus = " << theStatus << endl;
+#endif
+ return -1;
+ }
+ if (aSignal->getLength() != ScanTabRef::SignalLength){
+#ifdef VM_TRACE
+ ndbout << "SCAN_TABREF dropped, signal length " << aSignal->getLength() << endl;
+#endif
+ return -1;
+ }
+ const Uint64 tCurrTransId = this->getTransactionId();
+ const Uint64 tRecTransId = (Uint64)scanTabRef->transId1 +
+ ((Uint64)scanTabRef->transId2 << 32);
+ if ((tRecTransId - tCurrTransId) != (Uint64)0){
+#ifdef VM_TRACE
+ ndbout << "SCAN_TABREF dropped, wrong transid" << endl;
+#endif
+ return -1;
+ }
+#if 0
+ ndbout << "SCAN_TABREF, "
+ <<"transid=("<<hex<<scanTabRef->transId1<<", "<<hex<<scanTabRef->transId2<<")"
+ <<", err="<<dec<<scanTabRef->errorCode << endl;
+#endif
+ setErrorCode(scanTabRef->errorCode);
+ theCompletionStatus = CompletedFailure;
+ theCommitStatus = Aborted; // Indicate that this "transaction" was aborted
+ theTransactionIsStarted = false;
+ theScanningOp->releaseSignals();
+ return 0;
+}
+
+/*****************************************************************************
+ * int receiveSCAN_TABCONF(NdbApiSignal* aSignal)
+ *
+ * Receive SCAN_TABCONF
+ * If scanStatus == 0 there is more records to read. Since signals may be
+ * received in any order we have to go through the lists with saved signals
+ * and check if all expected signals are there so that we can start to
+ * execute them.
+ *
+ * If scanStatus > 0 this indicates that the scan is finished and there are
+ * no more data to be read.
+ *
+ *****************************************************************************/
+int
+NdbConnection::receiveSCAN_TABCONF(NdbApiSignal* aSignal)
+{
+ const ScanTabConf * const conf = CAST_CONSTPTR(ScanTabConf, aSignal->getDataPtr());
+ if (theStatus != Connected){
+#ifdef VM_TRACE
+ ndbout << "Dropping SCAN_TABCONF, theStatus = "<< theStatus << endl;
+#endif
+ return -1;
+ }
+ if(aSignal->getLength() != ScanTabConf::SignalLength){
+#ifdef VM_TRACE
+ ndbout << "Dropping SCAN_TABCONF, getLength = "<< aSignal->getLength() << endl;
+#endif
+ return -1;
+ }
+ const Uint64 tCurrTransId = this->getTransactionId();
+ const Uint64 tRecTransId =
+ (Uint64)conf->transId1 + ((Uint64)conf->transId2 << 32);
+ if ((tRecTransId - tCurrTransId) != (Uint64)0){
+#ifdef VM_TRACE
+ ndbout << "Dropping SCAN_TABCONF, wrong transid" << endl;
+#endif
+ return -1;
+ }
+
+ const Uint8 scanStatus =
+ ScanTabConf::getScanStatus(conf->requestInfo);
+
+ if (scanStatus != 0) {
+ theCompletionStatus = CompletedSuccess;
+ theCommitStatus = Committed;
+ theScanFinished = true;
+ return 0;
+ }
+
+ // There can only be one SCANTABCONF
+ assert(theNoOfSCANTABCONFRecv == 0);
+ theNoOfSCANTABCONFRecv++;
+
+ // Save a copy of the signal
+ NdbApiSignal * tCopy = new NdbApiSignal(0);//getSignal();
+ if (tCopy == NULL){
+ setErrorCode(4000);
+ return 2; // theWaiter.m_state = NO_WAIT
+ }
+ tCopy->copyFrom(aSignal);
+ tCopy->next(NULL);
+ theScanningOp->theSCAN_TABCONF_Recv = tCopy;
+
+ return checkNextScanResultComplete();
+
+}
+
+/*****************************************************************************
+ * int receiveSCAN_TABINFO(NdbApiSignal* aSignal)
+ *
+ * Receive SCAN_TABINFO
+ *
+ *****************************************************************************/
+int
+NdbConnection::receiveSCAN_TABINFO(NdbApiSignal* aSignal)
+{
+ if (theStatus != Connected){
+ //ndbout << "SCAN_TABINFO dropped, theStatus = " << theStatus << endl;
+ return -1;
+ }
+ if (aSignal->getLength() != ScanTabInfo::SignalLength){
+ //ndbout << "SCAN_TABINFO dropped, length = " << aSignal->getLength() << endl;
+ return -1;
+ }
+
+ NdbApiSignal * tCopy = new NdbApiSignal(0);//getSignal();
+ if (tCopy == NULL){
+ setErrorCode(4000);
+ return 2; // theWaiter.m_state = NO_WAIT
+ }
+ tCopy->copyFrom(aSignal);
+ tCopy->next(NULL);
+
+ // Put the signal last in list
+ if (theScanningOp->theFirstSCAN_TABINFO_Recv == NULL)
+ theScanningOp->theFirstSCAN_TABINFO_Recv = tCopy;
+ else
+ theScanningOp->theLastSCAN_TABINFO_Recv->next(tCopy);
+ theScanningOp->theLastSCAN_TABINFO_Recv = tCopy;
+
+ return checkNextScanResultComplete();
+}
+
+/******************************************************************************
+ * int checkNextScanResultComplete(NdbApiSignal* aSignal)
+ *
+ * Remark Traverses all the lists that are associated with
+ * this resultset and checks if all signals are there.
+ * If all required signal are received return 0
+ *
+ *
+ *****************************************************************************/
+int
+NdbConnection::checkNextScanResultComplete(){
+
+ if (theNoOfSCANTABCONFRecv != 1) {
+ return -1;
+ }
+
+ Uint32 tNoOfOpFetched = 0;
+ theCurrentScanRec = NULL;
+ thePreviousScanRec = NULL;
+
+ const ScanTabConf * const conf =
+ CAST_CONSTPTR(ScanTabConf, theScanningOp->theSCAN_TABCONF_Recv->getDataPtr());
+ const Uint32 numOperations = ScanTabConf::getOperations(conf->requestInfo);
+ Uint32 sigIndex = 0;
+ NdbApiSignal* tSignal = theScanningOp->theFirstSCAN_TABINFO_Recv;
+ while(tSignal != NULL){
+ const ScanTabInfo * const info = CAST_CONSTPTR(ScanTabInfo, tSignal->getDataPtr());
+ // Loop through the operations for this SCAN_TABINFO
+ // tOpAndLength is allowed to be zero, this means no
+ // TRANSID_AI signals where sent for this record
+ // I.e getValue was called 0 times when defining scan
+
+ // The max number of operations in each signal is 16
+ Uint32 numOpsInSig = numOperations - sigIndex*16;
+ if (numOpsInSig > 16)
+ numOpsInSig = 16;
+ for(Uint32 i = 0; i < numOpsInSig; i++){
+ const Uint32 tOpAndLength = info->operLenAndIdx[i];
+ const Uint32 tOpIndex = ScanTabInfo::getIdx(tOpAndLength);
+ const Uint32 tOpLen = ScanTabInfo::getLen(tOpAndLength);
+
+ assert(tOpIndex < 256);
+ NdbScanReceiver* tScanRec =
+ theScanningOp->theScanReceiversArray[tOpIndex];
+ assert(tScanRec != NULL);
+ if(tScanRec->isCompleted(tOpLen))
+ tScanRec->setCompleted();
+ else{
+ return -1; // At least one receiver was not ready
+ }
+
+ // Build list of scan receivers
+ if (theCurrentScanRec == NULL) {
+ theCurrentScanRec = tScanRec;
+ thePreviousScanRec = tScanRec;
+ } else {
+ thePreviousScanRec->next(tScanRec);
+ thePreviousScanRec = tScanRec;
+ }
+ tNoOfOpFetched++;
+ }
+ tSignal = tSignal->next();
+ sigIndex++;
+ }
+
+ // Check number of operations fetched against value in SCANTAB_CONF
+ if (tNoOfOpFetched != numOperations) {
+ setErrorCode(4113);
+ return 2; // theWaiter.m_state = NO_WAIT
+ }
+
+ // All signals for this resultset recieved
+ // release SCAN_TAB signals
+ theNoOfSCANTABCONFRecv = 0;
+ theScanningOp->releaseSignals();
+
+ // We have received all operations with correct lengths.
+ thePreviousScanRec = NULL;
+ theNoOfOpFetched = tNoOfOpFetched;
+ return 0;
+}
diff --git a/ndb/src/ndbapi/NdbCursorOperation.cpp b/ndb/src/ndbapi/NdbCursorOperation.cpp
new file mode 100644
index 00000000000..e4dd600c57f
--- /dev/null
+++ b/ndb/src/ndbapi/NdbCursorOperation.cpp
@@ -0,0 +1,57 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbCursorOperation.cpp
+ * Include:
+ * Link:
+ * Author: UABMASD Martin Sköld INN/V Alzato
+ * Date: 2002-04-01
+ * Version: 0.1
+ * Description: Cursor support
+ * Documentation:
+ * Adjust: 2002-04-01 UABMASD First version.
+ ****************************************************************************/
+
+#include <NdbCursorOperation.hpp>
+#include <NdbResultSet.hpp>
+
+NdbCursorOperation::NdbCursorOperation(Ndb* aNdb) :
+ NdbOperation(aNdb),
+ m_resultSet(0)
+{
+}
+
+NdbCursorOperation::~NdbCursorOperation()
+{
+ if (m_resultSet)
+ delete m_resultSet;
+}
+
+void NdbCursorOperation::cursInit()
+{
+ // Initialize result set
+}
+
+NdbResultSet* NdbCursorOperation::getResultSet()
+{
+ if (!m_resultSet)
+ m_resultSet = new NdbResultSet(this);
+
+ return m_resultSet;
+}
+
+
diff --git a/ndb/src/ndbapi/NdbDictionary.cpp b/ndb/src/ndbapi/NdbDictionary.cpp
new file mode 100644
index 00000000000..ec9a56cda62
--- /dev/null
+++ b/ndb/src/ndbapi/NdbDictionary.cpp
@@ -0,0 +1,800 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbDictionary.hpp>
+#include "NdbDictionaryImpl.hpp"
+#include <NdbOut.hpp>
+
+/*****************************************************************
+ * Column facade
+ */
+NdbDictionary::Column::Column(const char * name)
+ : m_impl(* new NdbColumnImpl(* this))
+{
+ setName(name);
+}
+
+NdbDictionary::Column::Column(const NdbDictionary::Column & org)
+ : m_impl(* new NdbColumnImpl(* this))
+{
+ m_impl.assign(org.m_impl);
+}
+
+NdbDictionary::Column::Column(NdbColumnImpl& impl)
+ : m_impl(impl)
+{
+}
+
+NdbDictionary::Column::~Column(){
+ NdbColumnImpl * tmp = &m_impl;
+ if(this != tmp){
+ delete tmp;
+ }
+}
+
+NdbDictionary::Column&
+NdbDictionary::Column::operator=(const NdbDictionary::Column& column)
+{
+ m_impl = column.m_impl;
+
+ return *this;
+}
+
+void
+NdbDictionary::Column::setName(const char * name){
+ m_impl.m_name.assign(name);
+}
+
+const char*
+NdbDictionary::Column::getName() const {
+ return m_impl.m_name.c_str();
+}
+
+void
+NdbDictionary::Column::setType(Type t){
+ m_impl.m_type = t;
+}
+
+NdbDictionary::Column::Type
+NdbDictionary::Column::getType() const {
+ return m_impl.m_type;
+}
+
+void
+NdbDictionary::Column::setPrecision(int val){
+ m_impl.m_precision = val;
+}
+
+int
+NdbDictionary::Column::getPrecision() const {
+ return m_impl.m_precision;
+}
+
+void
+NdbDictionary::Column::setScale(int val){
+ m_impl.m_scale = val;
+}
+
+int
+NdbDictionary::Column::getScale() const{
+ return m_impl.m_scale;
+}
+
+void
+NdbDictionary::Column::setLength(int length){
+ m_impl.m_length = length;
+}
+
+int
+NdbDictionary::Column::getLength() const{
+ return m_impl.m_length;
+}
+
+void
+NdbDictionary::Column::setNullable(bool val){
+ m_impl.m_nullable = val;
+}
+
+bool
+NdbDictionary::Column::getNullable() const {
+ return m_impl.m_nullable;
+}
+
+void
+NdbDictionary::Column::setPrimaryKey(bool val){
+ m_impl.m_pk = val;
+}
+
+bool
+NdbDictionary::Column::getPrimaryKey() const {
+ return m_impl.m_pk;
+}
+
+void
+NdbDictionary::Column::setTupleKey(bool val){
+ m_impl.m_tupleKey = val;
+}
+
+bool
+NdbDictionary::Column::getTupleKey() const {
+ return m_impl.m_tupleKey;
+}
+
+void
+NdbDictionary::Column::setDistributionKey(bool val){
+ m_impl.m_distributionKey = val;
+}
+
+bool
+NdbDictionary::Column::getDistributionKey() const{
+ return m_impl.m_distributionKey;
+}
+
+void
+NdbDictionary::Column::setDistributionGroup(bool val, int bits){
+ m_impl.m_distributionGroup = val;
+ m_impl.m_distributionGroupBits = bits;
+}
+
+bool
+NdbDictionary::Column::getDistributionGroup() const {
+ return m_impl.m_distributionGroup;
+}
+
+int
+NdbDictionary::Column::getDistributionGroupBits() const{
+ return m_impl.m_distributionGroupBits;
+}
+
+void
+NdbDictionary::Column::setIndexOnlyStorage(bool val){
+ m_impl.m_indexOnly = val;
+}
+
+bool
+NdbDictionary::Column::getIndexOnlyStorage() const {
+ return m_impl.m_indexOnly;
+}
+
+void
+NdbDictionary::Column::setAutoIncrement(bool val){
+ m_impl.m_autoIncrement = val;
+}
+
+bool
+NdbDictionary::Column::getAutoIncrement() const {
+ return m_impl.m_autoIncrement;
+}
+
+void
+NdbDictionary::Column::setAutoIncrementInitialValue(Uint64 val){
+ m_impl.m_autoIncrementInitialValue = val;
+}
+
+void
+NdbDictionary::Column::setDefaultValue(const char* defaultValue)
+{
+ m_impl.m_defaultValue.assign(defaultValue);
+}
+
+const char*
+NdbDictionary::Column::getDefaultValue() const
+{
+ return m_impl.m_defaultValue.c_str();
+}
+
+int
+NdbDictionary::Column::getColumnNo() const {
+ return m_impl.m_attrId;
+}
+
+bool
+NdbDictionary::Column::equal(const NdbDictionary::Column & col) const {
+ return m_impl.equal(col.m_impl);
+}
+
+/*****************************************************************
+ * Table facade
+ */
+NdbDictionary::Table::Table(const char * name)
+ : m_impl(* new NdbTableImpl(* this))
+{
+ setName(name);
+}
+
+NdbDictionary::Table::Table(const NdbDictionary::Table & org)
+ : m_impl(* new NdbTableImpl(* this))
+{
+ m_impl.assign(org.m_impl);
+}
+
+NdbDictionary::Table::Table(NdbTableImpl & impl)
+ : m_impl(impl)
+{
+}
+
+NdbDictionary::Table::~Table(){
+ NdbTableImpl * tmp = &m_impl;
+ if(this != tmp){
+ delete tmp;
+ }
+}
+
+NdbDictionary::Table&
+NdbDictionary::Table::operator=(const NdbDictionary::Table::Table& table)
+{
+ m_impl.assign(table.m_impl);
+
+ m_impl.m_facade = this;
+ return *this;
+}
+
+void
+NdbDictionary::Table::setName(const char * name){
+ m_impl.setName(name);
+}
+
+const char *
+NdbDictionary::Table::getName() const {
+ return m_impl.getName();
+}
+
+int
+NdbDictionary::Table::getTableId() const {
+ return m_impl.m_tableId;
+}
+
+void
+NdbDictionary::Table::addColumn(const Column & c){
+ // JONAS check this out!!
+ // Memory leak, the new Column will not be freed
+ //NdbDictionary::Column * col = new Column(c);
+ NdbColumnImpl* col = new NdbColumnImpl;
+ col->assign(NdbColumnImpl::getImpl(c));
+ m_impl.m_columns.push_back(col);
+ if(c.getPrimaryKey()){
+ m_impl.m_noOfKeys++;
+ }
+ m_impl.buildColumnHash();
+}
+
+const NdbDictionary::Column*
+NdbDictionary::Table::getColumn(const char * name) const {
+ return m_impl.getColumn(name);
+}
+
+const NdbDictionary::Column*
+NdbDictionary::Table::getColumn(const int attrId) const {
+ return m_impl.getColumn(attrId);
+}
+
+void
+NdbDictionary::Table::setLogging(bool val){
+ m_impl.m_logging = val;
+}
+
+bool
+NdbDictionary::Table::getLogging() const {
+ return m_impl.m_logging;
+}
+
+void
+NdbDictionary::Table::setFragmentType(FragmentType ft){
+ m_impl.m_fragmentType = ft;
+}
+
+NdbDictionary::Object::FragmentType
+NdbDictionary::Table::getFragmentType() const {
+ return m_impl.m_fragmentType;
+}
+
+void
+NdbDictionary::Table::setKValue(int kValue){
+ m_impl.m_kvalue = kValue;
+}
+
+int
+NdbDictionary::Table::getKValue() const {
+ return m_impl.m_kvalue;
+}
+
+void
+NdbDictionary::Table::setMinLoadFactor(int lf){
+ m_impl.m_minLoadFactor = lf;
+}
+
+int
+NdbDictionary::Table::getMinLoadFactor() const {
+ return m_impl.m_minLoadFactor;
+}
+
+void
+NdbDictionary::Table::setMaxLoadFactor(int lf){
+ m_impl.m_maxLoadFactor = lf;
+}
+
+int
+NdbDictionary::Table::getMaxLoadFactor() const {
+ return m_impl.m_maxLoadFactor;
+}
+
+int
+NdbDictionary::Table::getNoOfColumns() const {
+ return m_impl.m_columns.size();
+}
+
+int
+NdbDictionary::Table::getNoOfPrimaryKeys() const {
+ return m_impl.m_noOfKeys;
+}
+
+const char*
+NdbDictionary::Table::getPrimaryKey(int no) const {
+ int count = 0;
+ for (unsigned i = 0; i < m_impl.m_columns.size(); i++) {
+ if (m_impl.m_columns[i]->m_pk) {
+ if (count++ == no)
+ return m_impl.m_columns[i]->m_name.c_str();
+ }
+ }
+ return 0;
+}
+
+const void*
+NdbDictionary::Table::getFrmData() const {
+ return m_impl.m_frm.get_data();
+}
+
+Uint32
+NdbDictionary::Table::getFrmLength() const {
+ return m_impl.m_frm.length();
+}
+
+void
+NdbDictionary::Table::setFrm(const void* data, Uint32 len){
+ m_impl.m_frm.assign(data, len);
+}
+
+NdbDictionary::Object::Status
+NdbDictionary::Table::getObjectStatus() const {
+ return m_impl.m_status;
+}
+
+int
+NdbDictionary::Table::getObjectVersion() const {
+ return m_impl.m_version;
+}
+
+bool
+NdbDictionary::Table::equal(const NdbDictionary::Table & col) const {
+ return m_impl.equal(col.m_impl);
+}
+
+int
+NdbDictionary::Table::getRowSizeInBytes() const {
+ int sz = 0;
+ for(int i = 0; i<getNoOfColumns(); i++){
+ const NdbDictionary::Column * c = getColumn(i);
+ const NdbColumnImpl & col = NdbColumnImpl::getImpl(* c);
+ sz += (((col.m_attrSize * col.m_arraySize) + 3) / 4);
+ }
+ return sz * 4;
+}
+
+int
+NdbDictionary::Table::createTableInDb(Ndb* pNdb, bool equalOk) const {
+ const NdbDictionary::Table * pTab =
+ pNdb->getDictionary()->getTable(getName());
+ if(pTab != 0 && equal(* pTab))
+ return 0;
+ if(pTab != 0 && !equal(* pTab))
+ return -1;
+ return pNdb->getDictionary()->createTable(* this);
+}
+
+/*****************************************************************
+ * Index facade
+ */
+NdbDictionary::Index::Index(const char * name)
+ : m_impl(* new NdbIndexImpl(* this))
+{
+ setName(name);
+}
+
+NdbDictionary::Index::Index(NdbIndexImpl & impl)
+ : m_impl(impl)
+{
+}
+
+NdbDictionary::Index::~Index(){
+ NdbIndexImpl * tmp = &m_impl;
+ if(this != tmp){
+ delete tmp;
+ }
+}
+
+void
+NdbDictionary::Index::setName(const char * name){
+ m_impl.setName(name);
+}
+
+const char *
+NdbDictionary::Index::getName() const {
+ return m_impl.getName();
+}
+
+void
+NdbDictionary::Index::setTable(const char * table){
+ m_impl.setTable(table);
+}
+
+const char *
+NdbDictionary::Index::getTable() const {
+ return m_impl.getTable();
+}
+
+unsigned
+NdbDictionary::Index::getNoOfColumns() const {
+ return m_impl.m_columns.size();
+}
+
+int
+NdbDictionary::Index::getNoOfIndexColumns() const {
+ return m_impl.m_columns.size();
+}
+
+const NdbDictionary::Column *
+NdbDictionary::Index::getColumn(unsigned no) const {
+ if(no < m_impl.m_columns.size())
+ return m_impl.m_columns[no];
+ return NULL;
+}
+
+const char *
+NdbDictionary::Index::getIndexColumn(int no) const {
+ const NdbDictionary::Column* col = getColumn(no);
+
+ if (col)
+ return col->getName();
+ else
+ return NULL;
+}
+
+void
+NdbDictionary::Index::addColumn(const Column & c){
+ // JONAS check this out!!
+ // Memory leak, the new Column will not be freed
+ //NdbDictionary::Column * col = new Column(c);
+ NdbColumnImpl* col = new NdbColumnImpl;
+ col->assign(NdbColumnImpl::getImpl(c));
+ m_impl.m_columns.push_back(col);
+}
+
+void
+NdbDictionary::Index::addColumnName(const char * name){
+ const Column c(name);
+ addColumn(c);
+}
+
+void
+NdbDictionary::Index::addIndexColumn(const char * name){
+ const Column c(name);
+ addColumn(c);
+}
+
+void
+NdbDictionary::Index::addColumnNames(unsigned noOfNames, const char ** names){
+ for(unsigned i = 0; i < noOfNames; i++) {
+ const Column c(names[i]);
+ addColumn(c);
+ }
+}
+
+void
+NdbDictionary::Index::addIndexColumns(int noOfNames, const char ** names){
+ for(int i = 0; i < noOfNames; i++) {
+ const Column c(names[i]);
+ addColumn(c);
+ }
+}
+
+void
+NdbDictionary::Index::setType(NdbDictionary::Index::Type t){
+ m_impl.m_type = t;
+}
+
+NdbDictionary::Index::Type
+NdbDictionary::Index::getType() const {
+ return m_impl.m_type;
+}
+
+void
+NdbDictionary::Index::setLogging(bool val){
+ m_impl.m_logging = val;
+}
+
+bool
+NdbDictionary::Index::getLogging() const {
+ return m_impl.m_logging;
+}
+
+NdbDictionary::Object::Status
+NdbDictionary::Index::getObjectStatus() const {
+ return m_impl.m_status;
+}
+
+int
+NdbDictionary::Index::getObjectVersion() const {
+ return m_impl.m_version;
+}
+
+/*****************************************************************
+ * Event facade
+ */
+NdbDictionary::Event::Event(const char * name)
+ : m_impl(* new NdbEventImpl(* this))
+{
+ setName(name);
+}
+
+NdbDictionary::Event::Event(NdbEventImpl & impl)
+ : m_impl(impl)
+{
+}
+
+NdbDictionary::Event::~Event()
+{
+ NdbEventImpl * tmp = &m_impl;
+ if(this != tmp){
+ delete tmp;
+ }
+}
+
+void
+NdbDictionary::Event::setName(const char * name)
+{
+ m_impl.setName(name);
+}
+
+void
+NdbDictionary::Event::setTable(const char * table)
+{
+ m_impl.setTable(table);
+}
+
+void
+NdbDictionary::Event::addTableEvent(const TableEvent t)
+{
+ m_impl.addTableEvent(t);
+}
+
+void
+NdbDictionary::Event::setDurability(const EventDurability d)
+{
+ m_impl.setDurability(d);
+}
+
+void
+NdbDictionary::Event::addColumn(const Column & c){
+ // JONAS check this out!!
+ // Memory leak, the new Column will not be freed
+ //NdbDictionary::Column * col = new Column(c);
+ NdbColumnImpl* col = new NdbColumnImpl;
+ col->assign(NdbColumnImpl::getImpl(c));
+ m_impl.m_columns.push_back(col);
+}
+
+void
+NdbDictionary::Event::addEventColumn(unsigned attrId)
+{
+ m_impl.m_attrIds.push_back(attrId);
+}
+
+void
+NdbDictionary::Event::addEventColumn(const char * name)
+{
+ const Column c(name);
+ addColumn(c);
+}
+
+void
+NdbDictionary::Event::addEventColumns(int n, const char ** names)
+{
+ for (int i = 0; i < n; i++)
+ addEventColumn(names[i]);
+}
+
+NdbDictionary::Object::Status
+NdbDictionary::Event::getObjectStatus() const
+{
+ return m_impl.m_status;
+}
+
+int
+NdbDictionary::Event::getObjectVersion() const
+{
+ return m_impl.m_version;
+}
+
+void NdbDictionary::Event::print()
+{
+ m_impl.print();
+}
+
+/*****************************************************************
+ * Dictionary facade
+ */
+NdbDictionary::Dictionary::Dictionary(Ndb & ndb)
+ : m_impl(* new NdbDictionaryImpl(ndb, *this))
+{
+}
+
+NdbDictionary::Dictionary::Dictionary(NdbDictionaryImpl & impl)
+ : m_impl(impl)
+{
+}
+#include <assert.h>
+NdbDictionary::Dictionary::~Dictionary(){
+ NdbDictionaryImpl * tmp = &m_impl;
+ if(this != tmp){
+ delete tmp;
+ }
+}
+
+int
+NdbDictionary::Dictionary::createTable(const Table & t){
+ return m_impl.createTable(NdbTableImpl::getImpl(t));
+}
+
+int
+NdbDictionary::Dictionary::dropTable(Table & t){
+ return m_impl.dropTable(NdbTableImpl::getImpl(t));
+}
+
+int
+NdbDictionary::Dictionary::dropTable(const char * name){
+ return m_impl.dropTable(name);
+}
+
+int
+NdbDictionary::Dictionary::alterTable(const Table & t){
+ return m_impl.alterTable(NdbTableImpl::getImpl(t));
+}
+
+const NdbDictionary::Table *
+NdbDictionary::Dictionary::getTable(const char * name){
+ NdbTableImpl * t = m_impl.getTable(name);
+ if(t)
+ return t->m_facade;
+ return 0;
+}
+
+void
+NdbDictionary::Dictionary::invalidateTable(const char * name){
+ NdbTableImpl * t = m_impl.getTable(name);
+ if(t)
+ m_impl.invalidateObject(* t);
+}
+
+void
+NdbDictionary::Dictionary::removeCachedTable(const char * name){
+ NdbTableImpl * t = m_impl.getTable(name);
+ if(t)
+ m_impl.removeCachedObject(* t);
+}
+
+NdbDictionary::Table
+NdbDictionary::Dictionary::getTableForAlteration(const char * name){
+ const NdbDictionary::Table * oldTable = getTable(name);
+ return (oldTable) ?
+ NdbDictionary::Table(*oldTable)
+ : NdbDictionary::Table();
+}
+
+int
+NdbDictionary::Dictionary::createIndex(const Index & ind)
+{
+ return m_impl.createIndex(NdbIndexImpl::getImpl(ind));
+}
+
+int
+NdbDictionary::Dictionary::dropIndex(const char * indexName,
+ const char * tableName)
+{
+ return m_impl.dropIndex(indexName, tableName);
+}
+
+const NdbDictionary::Index *
+NdbDictionary::Dictionary::getIndex(const char * indexName,
+ const char * tableName)
+{
+ NdbIndexImpl * i = m_impl.getIndex(indexName, tableName);
+ if(i)
+ return i->m_facade;
+ return 0;
+}
+
+void
+NdbDictionary::Dictionary::invalidateIndex(const char * indexName,
+ const char * tableName){
+ NdbIndexImpl * i = m_impl.getIndex(indexName, tableName);
+ if(i) {
+ assert(i->m_table != 0);
+ m_impl.invalidateObject(* i->m_table);
+ }
+}
+
+void
+NdbDictionary::Dictionary::removeCachedIndex(const char * indexName,
+ const char * tableName){
+ NdbIndexImpl * i = m_impl.getIndex(indexName, tableName);
+ if(i) {
+ assert(i->m_table != 0);
+ m_impl.removeCachedObject(* i->m_table);
+ }
+}
+
+const NdbDictionary::Table *
+NdbDictionary::Dictionary::getIndexTable(const char * indexName,
+ const char * tableName)
+{
+ NdbIndexImpl * i = m_impl.getIndex(indexName, tableName);
+ NdbTableImpl * t = m_impl.getTable(tableName);
+ if(i && t) {
+ NdbTableImpl * it = m_impl.getIndexTable(i, t);
+ return it->m_facade;
+ }
+ return 0;
+}
+
+
+int
+NdbDictionary::Dictionary::createEvent(const Event & ev)
+{
+ return m_impl.createEvent(NdbEventImpl::getImpl(ev));
+}
+
+int
+NdbDictionary::Dictionary::dropEvent(const char * eventName)
+{
+ return m_impl.dropEvent(eventName);
+}
+
+const NdbDictionary::Event *
+NdbDictionary::Dictionary::getEvent(const char * eventName)
+{
+ NdbEventImpl * t = m_impl.getEvent(eventName);
+ if(t)
+ return t->m_facade;
+ return 0;
+}
+
+int
+NdbDictionary::Dictionary::listObjects(List& list, Object::Type type)
+{
+ return m_impl.listObjects(list, type);
+}
+
+int
+NdbDictionary::Dictionary::listIndexes(List& list, const char * tableName)
+{
+ return m_impl.listIndexes(list, tableName);
+}
+
+const struct NdbError &
+NdbDictionary::Dictionary::getNdbError() const {
+ return m_impl.getNdbError();
+}
diff --git a/ndb/src/ndbapi/NdbDictionaryImpl.cpp b/ndb/src/ndbapi/NdbDictionaryImpl.cpp
new file mode 100644
index 00000000000..bd94ba9b080
--- /dev/null
+++ b/ndb/src/ndbapi/NdbDictionaryImpl.cpp
@@ -0,0 +1,2733 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbDictionaryImpl.hpp"
+#include "API.hpp"
+#include <NdbOut.hpp>
+#include <AttrType.hpp>
+#include "NdbApiSignal.hpp"
+#include "TransporterFacade.hpp"
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/CreateTable.hpp>
+#include <signaldata/CreateIndx.hpp>
+#include <signaldata/CreateEvnt.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <signaldata/DropTable.hpp>
+#include <signaldata/AlterTable.hpp>
+#include <signaldata/DropIndx.hpp>
+#include <signaldata/ListTables.hpp>
+#include <SimpleProperties.hpp>
+#include <Bitmask.hpp>
+#include <AttributeList.hpp>
+#include <NdbEventOperation.hpp>
+#include "NdbEventOperationImpl.hpp"
+#include <assert.h>
+
+#define DEBUG_PRINT 0
+#define INCOMPATIBLE_VERSION -2
+
+//#define EVENT_DEBUG
+
+/**
+ * Column
+ */
+NdbColumnImpl::NdbColumnImpl()
+ : NdbDictionary::Column(* this), m_facade(this)
+{
+ init();
+}
+
+NdbColumnImpl::NdbColumnImpl(NdbDictionary::Column & f)
+ : NdbDictionary::Column(* this), m_facade(&f)
+{
+ init();
+}
+
+NdbColumnImpl::NdbColumnImpl&
+NdbColumnImpl::operator=(const NdbColumnImpl::NdbColumnImpl& col)
+{
+ m_attrId = col.m_attrId;
+ m_name = col.m_name;
+ m_type = col.m_type;
+ m_precision = col.m_precision;
+ m_scale = col.m_scale;
+ m_length = col.m_length;
+ m_pk = col.m_pk;
+ m_tupleKey = col.m_tupleKey;
+ m_distributionKey = col.m_distributionKey;
+ m_distributionGroup = col.m_distributionGroup;
+ m_distributionGroupBits = col.m_distributionGroupBits;
+ m_nullable = col.m_nullable;
+ m_indexOnly = col.m_indexOnly;
+ m_autoIncrement = col.m_autoIncrement;
+ m_autoIncrementInitialValue = col.m_autoIncrementInitialValue;
+ m_defaultValue = col.m_defaultValue;
+ m_attrType = col.m_attrType;
+ m_attrSize = col.m_attrSize;
+ m_arraySize = col.m_arraySize;
+ m_keyInfoPos = col.m_keyInfoPos;
+ // Do not copy m_facade !!
+
+ return *this;
+}
+
+void
+NdbColumnImpl::init()
+{
+ m_attrId = -1;
+ m_type = NdbDictionary::Column::Unsigned;
+ m_pk = false;
+ m_nullable = false;
+ m_tupleKey = false;
+ m_indexOnly = false;
+ m_distributionKey = false;
+ m_distributionGroup = false;
+ m_distributionGroupBits = 8;
+ m_length = 1;
+ m_scale = 5;
+ m_precision = 5;
+ m_keyInfoPos = 0;
+ m_attrSize = 4,
+ m_arraySize = 1,
+ m_autoIncrement = false;
+ m_autoIncrementInitialValue = 1;
+}
+
+NdbColumnImpl::~NdbColumnImpl()
+{
+}
+
+bool
+NdbColumnImpl::equal(const NdbColumnImpl& col) const
+{
+ if(strcmp(m_name.c_str(), col.m_name.c_str()) != 0){
+ return false;
+ }
+ if(m_type != col.m_type){
+ return false;
+ }
+ if(m_pk != col.m_pk){
+ return false;
+ }
+ if(m_nullable != col.m_nullable){
+ return false;
+ }
+ if(m_pk){
+ if(m_tupleKey != col.m_tupleKey){
+ return false;
+ }
+ if(m_indexOnly != col.m_indexOnly){
+ return false;
+ }
+ if(m_distributionKey != col.m_distributionKey){
+ return false;
+ }
+ if(m_distributionGroup != col.m_distributionGroup){
+ return false;
+ }
+ if(m_distributionGroup &&
+ (m_distributionGroupBits != col.m_distributionGroupBits)){
+ return false;
+ }
+ }
+ if(m_length != col.m_length){
+ return false;
+ }
+
+ switch(m_type){
+ case NdbDictionary::Column::Undefined:
+ break;
+ case NdbDictionary::Column::Tinyint:
+ case NdbDictionary::Column::Tinyunsigned:
+ case NdbDictionary::Column::Smallint:
+ case NdbDictionary::Column::Smallunsigned:
+ case NdbDictionary::Column::Mediumint:
+ case NdbDictionary::Column::Mediumunsigned:
+ case NdbDictionary::Column::Int:
+ case NdbDictionary::Column::Unsigned:
+ case NdbDictionary::Column::Float:
+ break;
+ case NdbDictionary::Column::Decimal:
+ if(m_scale != col.m_scale ||
+ m_precision != col.m_precision){
+ return false;
+ }
+ break;
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Binary:
+ case NdbDictionary::Column::Varbinary:
+ if(m_length != col.m_length){
+ return false;
+ }
+ break;
+ case NdbDictionary::Column::Bigint:
+ case NdbDictionary::Column::Bigunsigned:
+ case NdbDictionary::Column::Double:
+ case NdbDictionary::Column::Datetime:
+ case NdbDictionary::Column::Timespec:
+ case NdbDictionary::Column::Blob:
+ break;
+ }
+ if (m_autoIncrement != col.m_autoIncrement){
+ return false;
+ }
+ if(strcmp(m_defaultValue.c_str(), col.m_defaultValue.c_str()) != 0){
+ return false;
+ }
+
+ return true;
+}
+
+void
+NdbColumnImpl::assign(const NdbColumnImpl& org)
+{
+ m_attrId = org.m_attrId;
+ m_name.assign(org.m_name);
+ m_type = org.m_type;
+ m_precision = org.m_precision;
+ m_scale = org.m_scale;
+ m_length = org.m_length;
+ m_pk = org.m_pk;
+ m_tupleKey = org.m_tupleKey;
+ m_distributionKey = org.m_distributionKey;
+ m_distributionGroup = org.m_distributionGroup;
+ m_distributionGroupBits = org.m_distributionGroupBits;
+ m_nullable = org.m_nullable;
+ m_indexOnly = org.m_indexOnly;
+ m_autoIncrement = org.m_autoIncrement;
+ m_autoIncrementInitialValue = org.m_autoIncrementInitialValue;
+ m_defaultValue.assign(org.m_defaultValue);
+ m_keyInfoPos = org.m_keyInfoPos;
+ m_attrSize = org.m_attrSize;
+ m_arraySize = org.m_arraySize;
+}
+
+/**
+ * NdbTableImpl
+ */
+
+NdbTableImpl::NdbTableImpl()
+ : NdbDictionary::Table(* this), m_facade(this)
+{
+ m_noOfKeys = 0;
+ m_index = 0;
+ init();
+}
+
+NdbTableImpl::NdbTableImpl(NdbDictionary::Table & f)
+ : NdbDictionary::Table(* this), m_facade(&f)
+{
+ init();
+}
+
+NdbTableImpl::~NdbTableImpl()
+{
+ if (m_index != 0) {
+ delete m_index;
+ m_index = 0;
+ }
+ for (unsigned i = 0; i < m_columns.size(); i++)
+ delete m_columns[i];
+}
+
+void
+NdbTableImpl::init(){
+ clearNewProperties();
+ m_frm.clear();
+ m_fragmentType = NdbDictionary::Object::FragAllMedium;
+ m_logging = true;
+ m_kvalue = 6;
+ m_minLoadFactor = 78;
+ m_maxLoadFactor = 80;
+
+ m_index = 0;
+ m_indexType = NdbDictionary::Index::Undefined;
+
+ m_noOfKeys = 0;
+}
+
+bool
+NdbTableImpl::equal(const NdbTableImpl& obj) const
+{
+ if ((m_internalName.c_str() == NULL) ||
+ (strcmp(m_internalName.c_str(), "") == 0) ||
+ (obj.m_internalName.c_str() == NULL) ||
+ (strcmp(obj.m_internalName.c_str(), "") == 0)) {
+ // Shallow equal
+ if(strcmp(getName(), obj.getName()) != 0){
+ return false;
+ }
+ } else
+ // Deep equal
+ if(strcmp(m_internalName.c_str(), obj.m_internalName.c_str()) != 0){
+ return false;
+ }
+
+ if(m_fragmentType != obj.m_fragmentType){
+ return false;
+ }
+
+ if(m_columns.size() != obj.m_columns.size()){
+ return false;
+ }
+
+ for(unsigned i = 0; i<obj.m_columns.size(); i++){
+ if(!m_columns[i]->equal(* obj.m_columns[i])){
+ return false;
+ }
+ }
+
+ if(m_logging != obj.m_logging){
+ return false;
+ }
+
+ if(m_kvalue != obj.m_kvalue){
+ return false;
+ }
+
+ if(m_minLoadFactor != obj.m_minLoadFactor){
+ return false;
+ }
+
+ if(m_maxLoadFactor != obj.m_maxLoadFactor){
+ return false;
+ }
+
+ return true;
+}
+
+void
+NdbTableImpl::assign(const NdbTableImpl& org)
+{
+ m_tableId = org.m_tableId;
+ m_internalName.assign(org.m_internalName);
+ m_externalName.assign(org.m_externalName);
+ m_newExternalName.assign(org.m_newExternalName);
+ m_frm.assign(org.m_frm.get_data(), org.m_frm.length());
+ m_fragmentType = org.m_fragmentType;
+
+ for(unsigned i = 0; i<org.m_columns.size(); i++){
+ NdbColumnImpl * col = new NdbColumnImpl();
+ const NdbColumnImpl * iorg = org.m_columns[i];
+ col->assign(* iorg);
+ m_columns.push_back(col);
+ }
+
+ m_logging = org.m_logging;
+ m_kvalue = org.m_kvalue;
+ m_minLoadFactor = org.m_minLoadFactor;
+ m_maxLoadFactor = org.m_maxLoadFactor;
+
+ if (m_index != 0)
+ delete m_index;
+ m_index = org.m_index;
+
+ m_noOfKeys = org.m_noOfKeys;
+
+ m_version = org.m_version;
+ m_status = org.m_status;
+}
+
+void NdbTableImpl::setName(const char * name)
+{
+ m_newExternalName.assign(name);
+}
+
+const char *
+NdbTableImpl::getName() const
+{
+ if (m_newExternalName.empty())
+ return m_externalName.c_str();
+ else
+ return m_newExternalName.c_str();
+}
+
+void NdbTableImpl::clearNewProperties()
+{
+ m_newExternalName.assign("");
+ m_changeMask = 0;
+}
+
+void NdbTableImpl::copyNewProperties()
+{
+ if (!m_newExternalName.empty()) {
+ m_externalName.assign(m_newExternalName);
+ AlterTableReq::setNameFlag(m_changeMask, true);
+ }
+}
+
+void
+NdbTableImpl::buildColumnHash(){
+ const Uint32 size = m_columns.size();
+
+ for(size_t i = 31; i >= 0; i--){
+ if(((1 << i) & size) != 0){
+ m_columnHashMask = (1 << (i + 1)) - 1;
+ break;
+ }
+ }
+
+ Vector<Uint32> hashValues;
+ Vector<Vector<Uint32> > chains; chains.fill(size, hashValues);
+ for(size_t i = 0; i<size; i++){
+ Uint32 hv = Hash(m_columns[i]->getName()) & 0xFFFE;
+ Uint32 bucket = hv & m_columnHashMask;
+ bucket = (bucket < size ? bucket : bucket - size);
+ assert(bucket < size);
+ hashValues.push_back(hv);
+ chains[bucket].push_back(i);
+ }
+
+ m_columnHash.clear();
+ Uint32 tmp = 1;
+ m_columnHash.fill((unsigned)size-1, tmp); // Default no chaining
+
+ Uint32 pos = 0; // In overflow vector
+ for(size_t i = 0; i<size; i++){
+ Uint32 sz = chains[i].size();
+ if(sz == 1){
+ Uint32 col = chains[i][0];
+ Uint32 hv = hashValues[col];
+ Uint32 bucket = hv & m_columnHashMask;
+ bucket = (bucket < size ? bucket : bucket - size);
+ m_columnHash[bucket] = (col << 16) | hv | 1;
+ } else if(sz > 1){
+ Uint32 col = chains[i][0];
+ Uint32 hv = hashValues[col];
+ Uint32 bucket = hv & m_columnHashMask;
+ bucket = (bucket < size ? bucket : bucket - size);
+ m_columnHash[bucket] = (sz << 16) | (((size - bucket) + pos) << 1);
+ for(size_t j = 0; j<sz; j++, pos++){
+ Uint32 col = chains[i][j];
+ Uint32 hv = hashValues[col];
+ m_columnHash.push_back((col << 16) | hv);
+ }
+ }
+ }
+
+ m_columnHash.push_back(0); // Overflow when looping in end of array
+
+#if 0
+ for(size_t i = 0; i<m_columnHash.size(); i++){
+ Uint32 tmp = m_columnHash[i];
+ int col = -1;
+ if(i < size && (tmp & 1) == 1){
+ col = (tmp >> 16);
+ } else if(i >= size){
+ col = (tmp >> 16);
+ }
+ ndbout_c("m_columnHash[%d] %s = %x",
+ i, col > 0 ? m_columns[col]->getName() : "" , m_columnHash[i]);
+ }
+#endif
+}
+
+/**
+ * NdbIndexImpl
+ */
+
+NdbIndexImpl::NdbIndexImpl() :
+ NdbDictionary::Index(* this),
+ m_facade(this)
+{
+ m_logging = true;
+}
+
+NdbIndexImpl::NdbIndexImpl(NdbDictionary::Index & f) :
+ NdbDictionary::Index(* this),
+ m_facade(&f)
+{
+ m_logging = true;
+}
+
+NdbIndexImpl::~NdbIndexImpl(){
+ for (unsigned i = 0; i < m_columns.size(); i++)
+ delete m_columns[i];
+}
+
+void NdbIndexImpl::setName(const char * name)
+{
+ m_externalName.assign(name);
+}
+
+const char *
+NdbIndexImpl::getName() const
+{
+ return m_externalName.c_str();
+}
+
+void
+NdbIndexImpl::setTable(const char * table)
+{
+ m_tableName.assign(table);
+}
+
+const char *
+NdbIndexImpl::getTable() const
+{
+ return m_tableName.c_str();
+}
+
+/**
+ * NdbEventImpl
+ */
+
+NdbEventImpl::NdbEventImpl() :
+ NdbDictionary::Event(* this),
+ m_facade(this)
+{
+ mi_type = 0;
+ m_dur = NdbDictionary::Event::ED_UNDEFINED;
+ eventOp = NULL;
+ m_tableImpl = NULL;
+}
+
+NdbEventImpl::NdbEventImpl(NdbDictionary::Event & f) :
+ NdbDictionary::Event(* this),
+ m_facade(&f)
+{
+ mi_type = 0;
+ m_dur = NdbDictionary::Event::ED_UNDEFINED;
+ eventOp = NULL;
+ m_tableImpl = NULL;
+}
+
+NdbEventImpl::~NdbEventImpl()
+{
+ for (unsigned i = 0; i < m_columns.size(); i++)
+ delete m_columns[i];
+}
+
+void NdbEventImpl::setName(const char * name)
+{
+ m_externalName.assign(name);
+}
+
+void
+NdbEventImpl::setTable(const char * table)
+{
+ m_tableName.assign(table);
+}
+
+const char *
+NdbEventImpl::getTable() const
+{
+ return m_tableName.c_str();
+}
+
+const char *
+NdbEventImpl::getName() const
+{
+ return m_externalName.c_str();
+}
+
+void
+NdbEventImpl::addTableEvent(const NdbDictionary::Event::TableEvent t = NdbDictionary::Event::TE_ALL)
+{
+ switch (t) {
+ case NdbDictionary::Event::TE_INSERT : mi_type |= 1; break;
+ case NdbDictionary::Event::TE_DELETE : mi_type |= 2; break;
+ case NdbDictionary::Event::TE_UPDATE : mi_type |= 4; break;
+ default: mi_type = 4 | 2 | 1; // all types
+ }
+}
+
+void
+NdbEventImpl::setDurability(const NdbDictionary::Event::EventDurability d)
+{
+ m_dur = d;
+}
+
+/**
+ * NdbDictionaryImpl
+ */
+
+NdbDictionaryImpl::NdbDictionaryImpl(Ndb &ndb)
+ : NdbDictionary::Dictionary(* this),
+ m_facade(this),
+ m_receiver(m_error),
+ m_ndb(ndb)
+{
+ m_globalHash = 0;
+}
+
+NdbDictionaryImpl::NdbDictionaryImpl(Ndb &ndb,
+ NdbDictionary::Dictionary & f)
+ : NdbDictionary::Dictionary(* this),
+ m_facade(&f),
+ m_receiver(m_error),
+ m_ndb(ndb)
+{
+ m_globalHash = 0;
+}
+
+NdbDictionaryImpl::~NdbDictionaryImpl()
+{
+ NdbElement_t<NdbTableImpl> * curr = m_localHash.m_tableHash.getNext(0);
+ while(curr != 0){
+ m_globalHash->lock();
+ m_globalHash->release(curr->theData);
+ m_globalHash->unlock();
+
+ curr = m_localHash.m_tableHash.getNext(curr);
+ }
+}
+
+void
+initDict(NdbDictionary::Dictionary & d)
+{
+ TransporterFacade * tf = TransporterFacade::instance();
+ NdbDictionaryImpl & impl = NdbDictionaryImpl::getImpl(d);
+
+ impl.m_receiver.setTransporter(tf);
+}
+
+bool
+NdbDictionaryImpl::setTransporter(class TransporterFacade * tf)
+{
+ if(tf != 0){
+ m_globalHash = &tf->m_globalDictCache;
+ return m_receiver.setTransporter(tf);
+ }
+
+ return false;
+}
+
+bool
+NdbDictionaryImpl::setTransporter(class Ndb* ndb,
+ class TransporterFacade * tf)
+{
+ m_globalHash = &tf->m_globalDictCache;
+ return m_receiver.setTransporter(ndb, tf);
+}
+
+NdbTableImpl *
+NdbDictionaryImpl::getIndexTable(NdbIndexImpl * index,
+ NdbTableImpl * table)
+{
+ const char * internalName =
+ m_ndb.internalizeIndexName(table, index->getName());
+
+ return getTable(Ndb::externalizeTableName(internalName));
+}
+
+bool
+NdbDictInterface::setTransporter(class TransporterFacade * tf)
+{
+ if(tf == 0)
+ return false;
+
+ Guard g(tf->theMutexPtr);
+
+ m_blockNumber = tf->open(this,
+ execSignal,
+ execNodeStatus);
+
+ if ( m_blockNumber == -1 ) {
+ m_error.code = 4105;
+ return false; // no more free blocknumbers
+ }//if
+ Uint32 theNode = tf->ownId();
+ m_reference = numberToRef(m_blockNumber, theNode);
+ m_transporter = tf;
+ m_waiter.m_mutex = tf->theMutexPtr;
+
+ return true;
+}
+
+bool
+NdbDictInterface::setTransporter(class Ndb* ndb, class TransporterFacade * tf)
+{
+ m_blockNumber = -1;
+ m_reference = ndb->getReference();
+ m_transporter = tf;
+ m_waiter.m_mutex = tf->theMutexPtr;
+
+ return true;
+}
+
+NdbDictInterface::~NdbDictInterface()
+{
+ if (m_transporter != NULL){
+ if (m_blockNumber != -1)
+ m_transporter->close(m_blockNumber);
+ }
+}
+
+void
+NdbDictInterface::execSignal(void* dictImpl,
+ class NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3])
+{
+ NdbDictInterface * tmp = (NdbDictInterface*)dictImpl;
+
+ const Uint32 gsn = signal->readSignalNumber();
+ switch(gsn){
+ case GSN_GET_TABINFOREF:
+ tmp->execGET_TABINFO_REF(signal, ptr);
+ break;
+ case GSN_GET_TABINFO_CONF:
+ tmp->execGET_TABINFO_CONF(signal, ptr);
+ break;
+ case GSN_CREATE_TABLE_REF:
+ tmp->execCREATE_TABLE_REF(signal, ptr);
+ break;
+ case GSN_CREATE_TABLE_CONF:
+ tmp->execCREATE_TABLE_CONF(signal, ptr);
+ break;
+ case GSN_DROP_TABLE_REF:
+ tmp->execDROP_TABLE_REF(signal, ptr);
+ break;
+ case GSN_DROP_TABLE_CONF:
+ tmp->execDROP_TABLE_CONF(signal, ptr);
+ break;
+ case GSN_ALTER_TABLE_REF:
+ tmp->execALTER_TABLE_REF(signal, ptr);
+ break;
+ case GSN_ALTER_TABLE_CONF:
+ tmp->execALTER_TABLE_CONF(signal, ptr);
+ break;
+ case GSN_CREATE_INDX_REF:
+ tmp->execCREATE_INDX_REF(signal, ptr);
+ break;
+ case GSN_CREATE_INDX_CONF:
+ tmp->execCREATE_INDX_CONF(signal, ptr);
+ break;
+ case GSN_DROP_INDX_REF:
+ tmp->execDROP_INDX_REF(signal, ptr);
+ break;
+ case GSN_DROP_INDX_CONF:
+ tmp->execDROP_INDX_CONF(signal, ptr);
+ break;
+ case GSN_CREATE_EVNT_REF:
+ tmp->execCREATE_EVNT_REF(signal, ptr);
+ break;
+ case GSN_CREATE_EVNT_CONF:
+ tmp->execCREATE_EVNT_CONF(signal, ptr);
+ break;
+ case GSN_SUB_START_CONF:
+ tmp->execSUB_START_CONF(signal, ptr);
+ break;
+ case GSN_SUB_START_REF:
+ tmp->execSUB_START_REF(signal, ptr);
+ break;
+ case GSN_SUB_TABLE_DATA:
+ tmp->execSUB_TABLE_DATA(signal, ptr);
+ break;
+ case GSN_SUB_GCP_COMPLETE_REP:
+ tmp->execSUB_GCP_COMPLETE_REP(signal, ptr);
+ break;
+ case GSN_SUB_STOP_CONF:
+ tmp->execSUB_STOP_CONF(signal, ptr);
+ break;
+ case GSN_SUB_STOP_REF:
+ tmp->execSUB_STOP_REF(signal, ptr);
+ break;
+ case GSN_DROP_EVNT_REF:
+ tmp->execDROP_EVNT_REF(signal, ptr);
+ break;
+ case GSN_DROP_EVNT_CONF:
+ tmp->execDROP_EVNT_CONF(signal, ptr);
+ break;
+ case GSN_LIST_TABLES_CONF:
+ tmp->execLIST_TABLES_CONF(signal, ptr);
+ break;
+ default:
+ abort();
+ }
+}
+
+void
+NdbDictInterface::execNodeStatus(void* dictImpl, NodeId aNode,
+ bool alive, bool nfCompleted)
+{
+ NdbDictInterface * tmp = (NdbDictInterface*)dictImpl;
+
+ if(!alive && !nfCompleted){
+ return;
+ }
+
+ if (!alive && nfCompleted){
+ tmp->m_waiter.nodeFail(aNode);
+ }
+}
+
+int
+NdbDictInterface::dictSignal(NdbApiSignal* signal,
+ LinearSectionPtr ptr[3],int noLSP,
+ const int useMasterNodeId,
+ const Uint32 RETRIES,
+ const WaitSignalType wst,
+ const int theWait,
+ const int *errcodes,
+ const int noerrcodes,
+ const int temporaryMask)
+{
+ for(Uint32 i = 0; i<RETRIES; i++){
+ //if (useMasterNodeId == 0)
+ m_buffer.clear();
+
+ // Protected area
+ m_transporter->lock_mutex();
+ Uint32 aNodeId;
+ if (useMasterNodeId) {
+ if ((m_masterNodeId == 0) ||
+ (!m_transporter->get_node_alive(m_masterNodeId))) {
+ m_masterNodeId = m_transporter->get_an_alive_node();
+ }//if
+ aNodeId = m_masterNodeId;
+ } else {
+ aNodeId = m_transporter->get_an_alive_node();
+ }
+ if(aNodeId == 0){
+ m_error.code = 4009;
+ m_transporter->unlock_mutex();
+ return -1;
+ }
+ {
+ int r;
+ if (ptr) {
+#ifdef EVENT_DEBUG
+ printf("Long signal %d ptr", noLSP);
+ for (int q=0;q<noLSP;q++) {
+ printf(" sz %d", ptr[q].sz);
+ }
+ printf("\n");
+#endif
+ r = m_transporter->sendFragmentedSignal(signal, aNodeId, ptr, noLSP);
+ } else {
+#ifdef EVENT_DEBUG
+ printf("Short signal\n");
+#endif
+ r = m_transporter->sendSignal(signal, aNodeId);
+ }
+ if(r != 0){
+ m_transporter->unlock_mutex();
+ continue;
+ }
+ }
+
+ m_error.code = 0;
+
+ m_waiter.m_node = aNodeId;
+ m_waiter.m_state = wst;
+
+ m_waiter.wait(theWait);
+ // End of Protected area
+
+ if(m_waiter.m_state == NO_WAIT && m_error.code == 0){
+ // Normal return
+ return 0;
+ }
+
+ /**
+ * Handle error codes
+ */
+ if(m_waiter.m_state == WAIT_NODE_FAILURE)
+ continue;
+
+ if ( (temporaryMask & m_error.code) != 0 ) {
+ continue;
+ }
+ if (errcodes) {
+ int doContinue = 0;
+ for (int j=0; j < noerrcodes; j++)
+ if(m_error.code == errcodes[j]) {
+ doContinue = 1;
+ continue;
+ }
+ if (doContinue)
+ continue;
+ }
+
+ return -1;
+ }
+ return -1;
+}
+
+/*****************************************************************
+ * get tab info
+ */
+NdbTableImpl *
+NdbDictInterface::getTable(int tableId)
+{
+ NdbApiSignal tSignal(m_reference);
+ GetTabInfoReq * const req = CAST_PTR(GetTabInfoReq, tSignal.getDataPtrSend());
+
+ req->senderRef = m_reference;
+ req->senderData = 0;
+ req->requestType =
+ GetTabInfoReq::RequestById | GetTabInfoReq::LongSignalConf;
+ req->tableId = tableId;
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_GET_TABINFOREQ;
+ tSignal.theLength = GetTabInfoReq::SignalLength;
+
+ return getTable(&tSignal, 0, 0);
+}
+
+NdbTableImpl *
+NdbDictInterface::getTable(const char * name)
+{
+ NdbApiSignal tSignal(m_reference);
+ GetTabInfoReq * const req = CAST_PTR(GetTabInfoReq, tSignal.getDataPtrSend());
+
+ const Uint32 strLen = strlen(name) + 1; // NULL Terminated
+ if(strLen > MAX_TAB_NAME_SIZE) {//sizeof(req->tableName)){
+ m_error.code = 4307;
+ return 0;
+ }
+
+ req->senderRef = m_reference;
+ req->senderData = 0;
+ req->requestType =
+ GetTabInfoReq::RequestByName | GetTabInfoReq::LongSignalConf;
+ req->tableNameLen = strLen;
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_GET_TABINFOREQ;
+ // tSignal.theLength = GetTabInfoReq::HeaderLength + ((strLen + 3) / 4);
+ tSignal.theLength = GetTabInfoReq::SignalLength;
+ LinearSectionPtr ptr[1];
+ ptr[0].p = (Uint32*)name;
+ ptr[0].sz = strLen;
+
+ return getTable(&tSignal, ptr, 1);
+}
+
+NdbTableImpl *
+NdbDictInterface::getTable(class NdbApiSignal * signal,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections)
+{
+ //GetTabInfoReq * const req = CAST_PTR(GetTabInfoReq, signal->getDataPtrSend());
+ int r = dictSignal(signal,ptr,noOfSections,
+ 0/*do not use masternode id*/,
+ 100,
+ WAIT_GET_TAB_INFO_REQ,
+ WAITFOR_RESPONSE_TIMEOUT,
+ NULL,0);
+ if (r) return 0;
+
+ NdbTableImpl * rt = 0;
+ m_error.code = parseTableInfo(&rt,
+ (Uint32*)m_buffer.get_data(),
+ m_buffer.length() / 4);
+ rt->buildColumnHash();
+ return rt;
+}
+
+void
+NdbDictInterface::execGET_TABINFO_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ const GetTabInfoConf* conf = CAST_CONSTPTR(GetTabInfoConf, signal->getDataPtr());
+ if(signal->isFirstFragment()){
+ m_fragmentId = signal->getFragmentId();
+ m_buffer.grow(4 * conf->totalLen);
+ } else {
+ if(m_fragmentId != signal->getFragmentId()){
+ abort();
+ }
+ }
+
+ const Uint32 i = GetTabInfoConf::DICT_TAB_INFO;
+ m_buffer.append(ptr[i].p, 4 * ptr[i].sz);
+
+ if(!signal->isLastFragment()){
+ return;
+ }
+
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execGET_TABINFO_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ const GetTabInfoRef* ref = CAST_CONSTPTR(GetTabInfoRef, signal->getDataPtr());
+
+ m_error.code = ref->errorCode;
+ m_waiter.signal(NO_WAIT);
+}
+
+/*****************************************************************
+ * Pack/Unpack tables
+ */
+struct ApiKernelMapping {
+ Int32 kernelConstant;
+ Int32 apiConstant;
+};
+
+Uint32
+getApiConstant(Int32 kernelConstant, const ApiKernelMapping map[], Uint32 def)
+{
+ int i = 0;
+ while(map[i].kernelConstant != kernelConstant){
+ if(map[i].kernelConstant == -1 &&
+ map[i].apiConstant == -1){
+ return def;
+ }
+ i++;
+ }
+ return map[i].apiConstant;
+}
+
+Uint32
+getKernelConstant(Int32 apiConstant, const ApiKernelMapping map[], Uint32 def)
+{
+ int i = 0;
+ while(map[i].apiConstant != apiConstant){
+ if(map[i].kernelConstant == -1 &&
+ map[i].apiConstant == -1){
+ return def;
+ }
+ i++;
+ }
+ return map[i].kernelConstant;
+}
+
+static const
+ApiKernelMapping
+fragmentTypeMapping[] = {
+ { DictTabInfo::AllNodesSmallTable, NdbDictionary::Object::FragAllSmall },
+ { DictTabInfo::AllNodesMediumTable, NdbDictionary::Object::FragAllMedium },
+ { DictTabInfo::AllNodesLargeTable, NdbDictionary::Object::FragAllLarge },
+ { DictTabInfo::SingleFragment, NdbDictionary::Object::FragSingle },
+ { -1, -1 }
+};
+
+static const
+ApiKernelMapping
+objectTypeMapping[] = {
+ { DictTabInfo::SystemTable, NdbDictionary::Object::SystemTable },
+ { DictTabInfo::UserTable, NdbDictionary::Object::UserTable },
+ { DictTabInfo::UniqueHashIndex, NdbDictionary::Object::UniqueHashIndex },
+ { DictTabInfo::HashIndex, NdbDictionary::Object::HashIndex },
+ { DictTabInfo::UniqueOrderedIndex, NdbDictionary::Object::UniqueOrderedIndex },
+ { DictTabInfo::OrderedIndex, NdbDictionary::Object::OrderedIndex },
+ { DictTabInfo::HashIndexTrigger, NdbDictionary::Object::HashIndexTrigger },
+ { DictTabInfo::IndexTrigger, NdbDictionary::Object::IndexTrigger },
+ { DictTabInfo::SubscriptionTrigger,NdbDictionary::Object::SubscriptionTrigger },
+ { DictTabInfo::ReadOnlyConstraint ,NdbDictionary::Object::ReadOnlyConstraint },
+ { -1, -1 }
+};
+
+static const
+ApiKernelMapping
+objectStateMapping[] = {
+ { DictTabInfo::StateOffline, NdbDictionary::Object::StateOffline },
+ { DictTabInfo::StateBuilding, NdbDictionary::Object::StateBuilding },
+ { DictTabInfo::StateDropping, NdbDictionary::Object::StateDropping },
+ { DictTabInfo::StateOnline, NdbDictionary::Object::StateOnline },
+ { DictTabInfo::StateBroken, NdbDictionary::Object::StateBroken },
+ { -1, -1 }
+};
+
+static const
+ApiKernelMapping
+objectStoreMapping[] = {
+ { DictTabInfo::StoreTemporary, NdbDictionary::Object::StoreTemporary },
+ { DictTabInfo::StorePermanent, NdbDictionary::Object::StorePermanent },
+ { -1, -1 }
+};
+
+static const
+ApiKernelMapping
+indexTypeMapping[] = {
+ { DictTabInfo::UniqueHashIndex, NdbDictionary::Index::UniqueHashIndex },
+ { DictTabInfo::HashIndex, NdbDictionary::Index::HashIndex },
+ { DictTabInfo::UniqueOrderedIndex, NdbDictionary::Index::UniqueOrderedIndex},
+ { DictTabInfo::OrderedIndex, NdbDictionary::Index::OrderedIndex },
+ { -1, -1 }
+};
+
+static const
+ApiKernelMapping
+columnTypeMapping[] = {
+ { DictTabInfo::ExtTinyint, NdbDictionary::Column::Tinyint },
+ { DictTabInfo::ExtTinyunsigned, NdbDictionary::Column::Tinyunsigned },
+ { DictTabInfo::ExtSmallint, NdbDictionary::Column::Smallint },
+ { DictTabInfo::ExtSmallunsigned, NdbDictionary::Column::Smallunsigned },
+ { DictTabInfo::ExtMediumint, NdbDictionary::Column::Mediumint },
+ { DictTabInfo::ExtMediumunsigned, NdbDictionary::Column::Mediumunsigned },
+ { DictTabInfo::ExtInt, NdbDictionary::Column::Int },
+ { DictTabInfo::ExtUnsigned, NdbDictionary::Column::Unsigned },
+ { DictTabInfo::ExtBigint, NdbDictionary::Column::Bigint },
+ { DictTabInfo::ExtBigunsigned, NdbDictionary::Column::Bigunsigned },
+ { DictTabInfo::ExtFloat, NdbDictionary::Column::Float },
+ { DictTabInfo::ExtDouble, NdbDictionary::Column::Double },
+ { DictTabInfo::ExtDecimal, NdbDictionary::Column::Decimal },
+ { DictTabInfo::ExtChar, NdbDictionary::Column::Char },
+ { DictTabInfo::ExtVarchar, NdbDictionary::Column::Varchar },
+ { DictTabInfo::ExtBinary, NdbDictionary::Column::Binary },
+ { DictTabInfo::ExtVarbinary, NdbDictionary::Column::Varbinary },
+ { DictTabInfo::ExtDatetime, NdbDictionary::Column::Datetime },
+ { DictTabInfo::ExtTimespec, NdbDictionary::Column::Timespec },
+ { -1, -1 }
+};
+
+int
+NdbDictInterface::parseTableInfo(NdbTableImpl ** ret,
+ const Uint32 * data, Uint32 len)
+{
+ SimplePropertiesLinearReader it(data, len);
+ DictTabInfo::Table tableDesc; tableDesc.init();
+ SimpleProperties::UnpackStatus s;
+ s = SimpleProperties::unpack(it, &tableDesc,
+ DictTabInfo::TableMapping,
+ DictTabInfo::TableMappingSize,
+ true, true);
+
+ if(s != SimpleProperties::Break){
+ return 703;
+ }
+ const char * internalName = tableDesc.TableName;
+ const char * externalName = Ndb::externalizeTableName(internalName);
+
+ NdbTableImpl * impl = new NdbTableImpl();
+ impl->m_tableId = tableDesc.TableId;
+ impl->m_version = tableDesc.TableVersion;
+ impl->m_status = NdbDictionary::Object::Retrieved;
+ impl->m_internalName.assign(internalName);
+ impl->m_externalName.assign(externalName);
+
+ impl->m_frm.assign(tableDesc.FrmData, tableDesc.FrmLen);
+
+ impl->m_fragmentType = (NdbDictionary::Object::FragmentType)
+ getApiConstant(tableDesc.FragmentType,
+ fragmentTypeMapping,
+ (Uint32)NdbDictionary::Object::FragUndefined);
+
+ impl->m_logging = tableDesc.TableLoggedFlag;
+ impl->m_kvalue = tableDesc.TableKValue;
+ impl->m_minLoadFactor = tableDesc.MinLoadFactor;
+ impl->m_maxLoadFactor = tableDesc.MaxLoadFactor;
+
+ impl->m_indexType = (NdbDictionary::Index::Type)
+ getApiConstant(tableDesc.TableType,
+ indexTypeMapping,
+ NdbDictionary::Index::Undefined);
+
+ if(impl->m_indexType == NdbDictionary::Index::Undefined){
+ } else {
+ const char * externalPrimary =
+ Ndb::externalizeTableName(tableDesc.PrimaryTable);
+ impl->m_primaryTable.assign(externalPrimary);
+ }
+
+ Uint32 keyInfoPos = 0;
+ Uint32 keyCount = 0;
+
+ for(Uint32 i = 0; i < tableDesc.NoOfAttributes; i++) {
+ DictTabInfo::Attribute attrDesc; attrDesc.init();
+ s = SimpleProperties::unpack(it,
+ &attrDesc,
+ DictTabInfo::AttributeMapping,
+ DictTabInfo::AttributeMappingSize,
+ true, true);
+ if(s != SimpleProperties::Break){
+ delete impl;
+ return 703;
+ }
+
+ NdbColumnImpl * col = new NdbColumnImpl();
+ col->m_attrId = attrDesc.AttributeId;
+ col->setName(attrDesc.AttributeName);
+ col->m_type = (NdbDictionary::Column::Type)
+ getApiConstant(attrDesc.AttributeExtType,
+ columnTypeMapping,
+ NdbDictionary::Column::Undefined);
+ if (col->m_type == NdbDictionary::Column::Undefined) {
+ delete impl;
+ return 703;
+ }
+ col->m_extType = attrDesc.AttributeExtType;
+ col->m_precision = attrDesc.AttributeExtPrecision;
+ col->m_scale = attrDesc.AttributeExtScale;
+ col->m_length = attrDesc.AttributeExtLength;
+
+ // translate to old kernel types and sizes
+ if (! attrDesc.translateExtType()) {
+ delete impl;
+ return 703;
+ }
+ col->m_attrType =attrDesc.AttributeType;
+ col->m_attrSize = (1 << attrDesc.AttributeSize) / 8;
+ col->m_arraySize = attrDesc.AttributeArraySize;
+
+ col->m_pk = attrDesc.AttributeKeyFlag;
+ col->m_tupleKey = 0;
+ col->m_distributionKey = attrDesc.AttributeDKey;
+ col->m_distributionGroup = attrDesc.AttributeDGroup;
+ col->m_distributionGroupBits = 16;
+ col->m_nullable = attrDesc.AttributeNullableFlag;
+ col->m_indexOnly = (attrDesc.AttributeStoredInd ? false : true);
+ col->m_autoIncrement = (attrDesc.AttributeAutoIncrement ? true : false);
+ col->m_autoIncrementInitialValue = ~0;
+ col->m_defaultValue.assign(attrDesc.AttributeDefaultValue);
+
+ if(attrDesc.AttributeKeyFlag){
+ col->m_keyInfoPos = keyInfoPos + 1;
+ keyInfoPos += ((col->m_attrSize * col->m_arraySize + 3) / 4);
+ keyCount++;
+ } else {
+ col->m_keyInfoPos = 0;
+ }
+
+ NdbColumnImpl * null = 0;
+ impl->m_columns.fill(attrDesc.AttributeId, null);
+ if(impl->m_columns[attrDesc.AttributeId] != 0){
+ delete col;
+ delete impl;
+ return 703;
+ }
+ impl->m_columns[attrDesc.AttributeId] = col;
+ it.next();
+ }
+ impl->m_noOfKeys = keyCount;
+ * ret = impl;
+ return 0;
+}
+
+/*****************************************************************
+ * Create table and alter table
+ */
+int
+NdbDictInterface::createTable(Ndb & ndb,
+ NdbTableImpl & impl)
+{
+ return createOrAlterTable(ndb, impl, false);
+}
+
+int NdbDictionaryImpl::alterTable(NdbTableImpl &impl)
+{
+ BaseString internalName = impl.m_internalName;
+ const char * originalInternalName = internalName.c_str();
+ BaseString externalName = impl.m_externalName;
+ const char * originalExternalName = externalName.c_str();
+ NdbTableImpl * oldTab = getTable(originalExternalName);
+
+ if(!oldTab){
+ m_error.code = 709;
+ return -1;
+ }
+ // Alter the table
+ int ret = m_receiver.alterTable(m_ndb, impl);
+
+ if(ret == 0){
+ // Remove cached information and let it be refreshed at next access
+ if (m_localHash.get(originalInternalName) != NULL) {
+ m_localHash.drop(originalInternalName);
+ NdbTableImpl * cachedImpl = m_globalHash->get(originalInternalName);
+ // If in local cache it must be in global
+ if (!cachedImpl)
+ abort();
+ m_globalHash->lock();
+ m_globalHash->drop(cachedImpl);
+ m_globalHash->unlock();
+ }
+ }
+
+ return ret;
+}
+
+int
+NdbDictInterface::alterTable(Ndb & ndb,
+ NdbTableImpl & impl)
+{
+ return createOrAlterTable(ndb, impl, true);
+}
+
+int
+NdbDictInterface::createOrAlterTable(Ndb & ndb,
+ NdbTableImpl & impl,
+ bool alter)
+{
+ if((unsigned)impl.getNoOfPrimaryKeys() > MAXNROFTUPLEKEY){
+ m_error.code = 4317;
+ return -1;
+ }
+ unsigned sz = impl.m_columns.size();
+ if (sz > NDB_MAX_ATTRIBUTES_IN_TABLE){
+ m_error.code = 4318;
+ return -1;
+ }
+
+ impl.copyNewProperties();
+ //validate();
+ //aggregate();
+
+ const char * internalName =
+ ndb.internalizeTableName(impl.m_externalName.c_str());
+ impl.m_internalName.assign(internalName);
+ UtilBufferWriter w(m_buffer);
+ DictTabInfo::Table tmpTab; tmpTab.init();
+ snprintf(tmpTab.TableName,
+ sizeof(tmpTab.TableName),
+ internalName);
+
+ bool haveAutoIncrement = false;
+ Uint64 autoIncrementValue;
+ for(unsigned i = 0; i<sz; i++){
+ const NdbColumnImpl * col = impl.m_columns[i];
+ if(col == 0)
+ continue;
+ if (col->m_autoIncrement) {
+ if (haveAutoIncrement) {
+ m_error.code = 4335;
+ return -1;
+ }
+ haveAutoIncrement = true;
+ autoIncrementValue = col->m_autoIncrementInitialValue;
+ }
+ }
+
+ // Check max length of frm data
+ if (impl.m_frm.length() > MAX_FRM_DATA_SIZE){
+ m_error.code = 1229;
+ return -1;
+ }
+ tmpTab.FrmLen = impl.m_frm.length();
+ memcpy(tmpTab.FrmData, impl.m_frm.get_data(), impl.m_frm.length());
+
+ tmpTab.TableLoggedFlag = impl.m_logging;
+ tmpTab.TableKValue = impl.m_kvalue;
+ tmpTab.MinLoadFactor = impl.m_minLoadFactor;
+ tmpTab.MaxLoadFactor = impl.m_maxLoadFactor;
+ tmpTab.TableType = DictTabInfo::UserTable;
+ tmpTab.NoOfAttributes = sz;
+
+ tmpTab.FragmentType = getKernelConstant(impl.m_fragmentType,
+ fragmentTypeMapping,
+ DictTabInfo::AllNodesSmallTable);
+ tmpTab.TableVersion = rand();
+
+ SimpleProperties::UnpackStatus s;
+ s = SimpleProperties::pack(w,
+ &tmpTab,
+ DictTabInfo::TableMapping,
+ DictTabInfo::TableMappingSize, true);
+
+ if(s != SimpleProperties::Eof){
+ abort();
+ }
+
+ for(unsigned i = 0; i<sz; i++){
+ const NdbColumnImpl * col = impl.m_columns[i];
+ if(col == 0)
+ continue;
+
+ DictTabInfo::Attribute tmpAttr; tmpAttr.init();
+ snprintf(tmpAttr.AttributeName, sizeof(tmpAttr.AttributeName),
+ col->m_name.c_str());
+ tmpAttr.AttributeId = i;
+ tmpAttr.AttributeKeyFlag = col->m_pk || col->m_tupleKey;
+ tmpAttr.AttributeNullableFlag = col->m_nullable;
+ tmpAttr.AttributeStoredInd = (col->m_indexOnly ? 0 : 1);
+ tmpAttr.AttributeDKey = col->m_distributionKey;
+ tmpAttr.AttributeDGroup = col->m_distributionGroup;
+
+ tmpAttr.AttributeExtType =
+ getKernelConstant(col->m_type,
+ columnTypeMapping,
+ DictTabInfo::ExtUndefined);
+ tmpAttr.AttributeExtPrecision = col->m_precision;
+ tmpAttr.AttributeExtScale = col->m_scale;
+ tmpAttr.AttributeExtLength = col->m_length;
+
+ // DICT will ignore and recompute this
+ (void)tmpAttr.translateExtType();
+
+ tmpAttr.AttributeAutoIncrement = col->m_autoIncrement;
+ snprintf(tmpAttr.AttributeDefaultValue,
+ sizeof(tmpAttr.AttributeDefaultValue),
+ col->m_defaultValue.c_str());
+ s = SimpleProperties::pack(w,
+ &tmpAttr,
+ DictTabInfo::AttributeMapping,
+ DictTabInfo::TableMappingSize, true);
+ w.add(DictTabInfo::AttributeEnd, 1);
+ }
+
+ NdbApiSignal tSignal(m_reference);
+ tSignal.theReceiversBlockNumber = DBDICT;
+ if (alter) {
+ AlterTableReq * const req =
+ CAST_PTR(AlterTableReq, tSignal.getDataPtrSend());
+
+ req->senderRef = m_reference;
+ req->senderData = 0;
+ req->changeMask = impl.m_changeMask;
+ req->tableId = impl.m_tableId;
+ req->tableVersion = impl.m_version;;
+ tSignal.theVerId_signalNumber = GSN_ALTER_TABLE_REQ;
+ tSignal.theLength = AlterTableReq::SignalLength;
+ }
+ else {
+ CreateTableReq * const req =
+ CAST_PTR(CreateTableReq, tSignal.getDataPtrSend());
+
+ req->senderRef = m_reference;
+ req->senderData = 0;
+ tSignal.theVerId_signalNumber = GSN_CREATE_TABLE_REQ;
+ tSignal.theLength = CreateTableReq::SignalLength;
+ }
+
+ LinearSectionPtr ptr[3];
+ ptr[0].p = (Uint32*)m_buffer.get_data();
+ ptr[0].sz = m_buffer.length() / 4;
+
+ int ret = (alter) ?
+ alterTable(&tSignal, ptr)
+ : createTable(&tSignal, ptr);
+
+ if (haveAutoIncrement) {
+ // if (!ndb.setAutoIncrementValue(impl.m_internalName.c_str(), autoIncrementValue)) {
+ if (!ndb.setAutoIncrementValue(impl.m_externalName.c_str(), autoIncrementValue)) {
+ m_error.code = 4336;
+ ndb.theError = m_error;
+ ret = -1; // errorcode set in initialize_autoincrement
+ }
+ }
+ return ret;
+}
+
+int
+NdbDictInterface::createTable(NdbApiSignal* signal, LinearSectionPtr ptr[3])
+{
+#if DEBUG_PRINT
+ ndbout_c("BufferLen = %d", ptr[0].sz);
+ SimplePropertiesLinearReader r(ptr[0].p, ptr[0].sz);
+ r.printAll(ndbout);
+#endif
+
+ const int noErrCodes = 2;
+ int errCodes[noErrCodes] =
+ {CreateTableRef::Busy,
+ CreateTableRef::NotMaster};
+ return dictSignal(signal,ptr,1,
+ 1/*use masternode id*/,
+ 100,
+ WAIT_CREATE_INDX_REQ,
+ WAITFOR_RESPONSE_TIMEOUT,
+ errCodes,noErrCodes);
+}
+
+
+void
+NdbDictInterface::execCREATE_TABLE_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ //CreateTableConf* const conf = CAST_PTR(CreateTableConf, signal->getDataPtr());
+
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execCREATE_TABLE_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ const CreateTableRef* const ref = CAST_CONSTPTR(CreateTableRef, signal->getDataPtr());
+ m_error.code = ref->errorCode;
+ m_masterNodeId = ref->masterNodeId;
+ m_waiter.signal(NO_WAIT);
+}
+
+int
+NdbDictInterface::alterTable(NdbApiSignal* signal, LinearSectionPtr ptr[3])
+{
+#if DEBUG_PRINT
+ ndbout_c("BufferLen = %d", ptr[0].sz);
+ SimplePropertiesLinearReader r(ptr[0].p, ptr[0].sz);
+ r.printAll(ndbout);
+#endif
+
+ const int noErrCodes = 2;
+ int errCodes[noErrCodes] =
+ {AlterTableRef::NotMaster,
+ AlterTableRef::Busy};
+ int r = dictSignal(signal,ptr,1,
+ 1/*use masternode id*/,
+ 100,WAIT_ALTER_TAB_REQ,
+ WAITFOR_RESPONSE_TIMEOUT,
+ errCodes, noErrCodes);
+ if(m_error.code == AlterTableRef::InvalidTableVersion) {
+ // Clear caches and try again
+ return INCOMPATIBLE_VERSION;
+ }
+
+ return r;
+}
+
+void
+NdbDictInterface::execALTER_TABLE_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ //AlterTableConf* const conf = CAST_CONSTPTR(AlterTableConf, signal->getDataPtr());
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execALTER_TABLE_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ const AlterTableRef * const ref =
+ CAST_CONSTPTR(AlterTableRef, signal->getDataPtr());
+ m_error.code = ref->errorCode;
+ m_masterNodeId = ref->masterNodeId;
+ m_waiter.signal(NO_WAIT);
+}
+
+/*****************************************************************
+ * Drop table
+ */
+int
+NdbDictionaryImpl::dropTable(const char * name)
+{
+ NdbTableImpl * tab = getTable(name);
+ if(tab == 0){
+ return -1;
+ }
+ int ret = dropTable(* tab);
+ // If table stored in cache is incompatible with the one in the kernel
+ // we must clear the cache and try again
+ if (ret == INCOMPATIBLE_VERSION) {
+ const char * internalTableName = m_ndb.internalizeTableName(name);
+
+ m_localHash.drop(internalTableName);
+
+ m_globalHash->lock();
+ m_globalHash->drop(tab);
+ m_globalHash->unlock();
+ return dropTable(name);
+ }
+
+ return ret;
+}
+
+int
+NdbDictionaryImpl::dropTable(NdbTableImpl & impl)
+{
+ const char * name = impl.getName();
+ if(impl.m_status == NdbDictionary::Object::New){
+ return dropTable(name);
+ }
+
+ if (impl.m_indexType != NdbDictionary::Index::Undefined) {
+ m_receiver.m_error.code = 1228;
+ return -1;
+ }
+
+ List list;
+ if (listIndexes(list, name) == -1)
+ return -1;
+ for (unsigned i = 0; i < list.count; i++) {
+ const List::Element& element = list.elements[i];
+ if (dropIndex(element.name, name) == -1)
+ return -1;
+ }
+ int ret = m_receiver.dropTable(impl);
+ if(ret == 0){
+ const char * internalTableName = impl.m_internalName.c_str();
+
+ m_localHash.drop(internalTableName);
+
+ m_globalHash->lock();
+ m_globalHash->drop(&impl);
+ m_globalHash->unlock();
+ }
+
+ return ret;
+}
+
+int
+NdbDictInterface::dropTable(const NdbTableImpl & impl)
+{
+ NdbApiSignal tSignal(m_reference);
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_DROP_TABLE_REQ;
+ tSignal.theLength = DropTableReq::SignalLength;
+
+ DropTableReq * const req = CAST_PTR(DropTableReq, tSignal.getDataPtrSend());
+ req->senderRef = m_reference;
+ req->senderData = 0;
+ req->tableId = impl.m_tableId;
+ req->tableVersion = impl.m_version;
+
+ return dropTable(&tSignal, 0);
+}
+
+int
+NdbDictInterface::dropTable(NdbApiSignal* signal, LinearSectionPtr ptr[3])
+{
+ const int noErrCodes = 3;
+ int errCodes[noErrCodes] =
+ {DropTableRef::NoDropTableRecordAvailable,
+ DropTableRef::NotMaster,
+ DropTableRef::Busy};
+ int r = dictSignal(signal,NULL,0,
+ 1/*use masternode id*/,
+ 100,WAIT_DROP_TAB_REQ,
+ WAITFOR_RESPONSE_TIMEOUT,
+ errCodes, noErrCodes);
+ if(m_error.code == DropTableRef::InvalidTableVersion) {
+ // Clear caches and try again
+ return INCOMPATIBLE_VERSION;
+ }
+ return r;
+}
+
+void
+NdbDictInterface::execDROP_TABLE_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ //DropTableConf* const conf = CAST_CONSTPTR(DropTableConf, signal->getDataPtr());
+
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execDROP_TABLE_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ const DropTableRef* const ref = CAST_CONSTPTR(DropTableRef, signal->getDataPtr());
+ m_error.code = ref->errorCode;
+ m_masterNodeId = ref->masterNodeId;
+ m_waiter.signal(NO_WAIT);
+}
+
+int
+NdbDictionaryImpl::invalidateObject(NdbTableImpl & impl)
+{
+ const char * internalTableName = impl.m_internalName.c_str();
+
+ m_localHash.drop(internalTableName);
+ m_globalHash->lock();
+ m_globalHash->drop(&impl);
+ m_globalHash->unlock();
+ return 0;
+}
+
+int
+NdbDictionaryImpl::removeCachedObject(NdbTableImpl & impl)
+{
+ const char * internalTableName = impl.m_internalName.c_str();
+
+ m_localHash.drop(internalTableName);
+ m_globalHash->lock();
+ m_globalHash->release(&impl);
+ m_globalHash->unlock();
+ return 0;
+}
+
+/*****************************************************************
+ * Get index info
+ */
+NdbIndexImpl*
+NdbDictionaryImpl::getIndexImpl(const char * externalName,
+ const char * internalName)
+{
+ NdbTableImpl* tab = getTableImpl(internalName);
+ if(tab == 0){
+ m_error.code = 4243;
+ return 0;
+ }
+
+ if(tab->m_indexType == NdbDictionary::Index::Undefined){
+ // Not an index
+ m_error.code = 4243;
+ return 0;
+ }
+
+ NdbTableImpl* primTab = getTable(tab->m_primaryTable.c_str());
+ if(primTab == 0){
+ m_error.code = 4243;
+ return 0;
+ }
+
+ /**
+ * Create index impl
+ */
+ NdbIndexImpl* idx = new NdbIndexImpl();
+ idx->m_version = tab->m_version;
+ idx->m_status = tab->m_status;
+ idx->m_indexId = tab->m_tableId;
+ idx->m_internalName.assign(internalName);
+ idx->m_externalName.assign(externalName);
+ idx->m_tableName.assign(primTab->m_externalName);
+ idx->m_type = tab->m_indexType;
+ // skip last attribute (NDB$PK or NDB$TNODE)
+ for(unsigned i = 0; i+1<tab->m_columns.size(); i++){
+ NdbColumnImpl* col = new NdbColumnImpl;
+ // Copy column definition
+ *col = *tab->m_columns[i];
+ idx->m_columns.push_back(col);
+ }
+
+ idx->m_table = tab;
+ // TODO Assign idx to tab->m_index
+ // Don't do it right now since assign can't asign a table with index
+ // tab->m_index = idx;
+ return idx;
+}
+
+/*****************************************************************
+ * Create index
+ */
+int
+NdbDictionaryImpl::createIndex(NdbIndexImpl &ix)
+{
+ NdbTableImpl* tab = getTable(ix.getTable());
+ if(tab == 0){
+ m_error.code = 4249;
+ return -1;
+ }
+
+ return m_receiver.createIndex(m_ndb, ix, * tab);
+}
+
+int
+NdbDictInterface::createIndex(Ndb & ndb,
+ NdbIndexImpl & impl,
+ const NdbTableImpl & table)
+{
+ //validate();
+ //aggregate();
+
+ UtilBufferWriter w(m_buffer);
+ const size_t len = strlen(impl.m_externalName.c_str()) + 1;
+ if(len > MAX_TAB_NAME_SIZE) {
+ m_error.code = 4241;
+ return -1;
+ }
+ const char * internalName =
+ ndb.internalizeIndexName(&table, impl.getName());
+
+ impl.m_internalName.assign(internalName);
+
+ w.add(DictTabInfo::TableName, internalName);
+ w.add(DictTabInfo::TableLoggedFlag, impl.m_logging);
+
+ NdbApiSignal tSignal(m_reference);
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_CREATE_INDX_REQ;
+ tSignal.theLength = CreateIndxReq::SignalLength;
+
+ CreateIndxReq * const req = CAST_PTR(CreateIndxReq, tSignal.getDataPtrSend());
+
+ req->setUserRef(m_reference);
+ req->setConnectionPtr(0);
+ req->setRequestType(CreateIndxReq::RT_USER);
+
+ Uint32 it = getKernelConstant(impl.m_type,
+ indexTypeMapping,
+ DictTabInfo::UndefTableType);
+
+ if(it == DictTabInfo::UndefTableType){
+ m_error.code = 4250;
+ return -1;
+ }
+ req->setIndexType((DictTabInfo::TableType) it);
+
+ req->setTableId(table.m_tableId);
+ req->setOnline(true);
+ AttributeList attributeList;
+ attributeList.sz = impl.m_columns.size();
+ for(unsigned i = 0; i<attributeList.sz; i++){
+ const NdbColumnImpl* col =
+ table.getColumn(impl.m_columns[i]->m_name.c_str());
+ if(col == 0){
+ m_error.code = 4247;
+ return -1;
+ }
+ // Copy column definition
+ *impl.m_columns[i] = *col;
+
+ if(col->m_pk && col->m_indexOnly){
+ m_error.code = 4245;
+ return -1;
+ }
+
+ if (it == DictTabInfo::UniqueHashIndex &&
+ (col->m_nullable) && (attributeList.sz > 1)) {
+ // We only support one NULL attribute
+ m_error.code = 4246;
+ return -1;
+ }
+ attributeList.id[i] = col->m_attrId;
+ }
+ if (it == DictTabInfo::UniqueHashIndex) {
+ // Sort index attributes according to primary table (using insertion sort)
+ for(unsigned i = 1; i < attributeList.sz; i++) {
+ unsigned int temp = attributeList.id[i];
+ unsigned int j = i;
+ while((j > 0) && (attributeList.id[j - 1] > temp)) {
+ attributeList.id[j] = attributeList.id[j - 1];
+ j--;
+ }
+ attributeList.id[j] = temp;
+ }
+ // Check for illegal duplicate attributes
+ for(unsigned i = 0; i<attributeList.sz; i++) {
+ if ((i != (attributeList.sz - 1)) &&
+ (attributeList.id[i] == attributeList.id[i+1])) {
+ m_error.code = 4258;
+ return -1;
+ }
+ }
+ }
+ LinearSectionPtr ptr[3];
+ ptr[0].p = (Uint32*)&attributeList;
+ ptr[0].sz = 1 + attributeList.sz;
+ ptr[1].p = (Uint32*)m_buffer.get_data();
+ ptr[1].sz = m_buffer.length() >> 2; //BUG?
+ return createIndex(&tSignal, ptr);
+}
+
+int
+NdbDictInterface::createIndex(NdbApiSignal* signal,
+ LinearSectionPtr ptr[3])
+{
+ const int noErrCodes = 1;
+ int errCodes[noErrCodes] = {CreateIndxRef::Busy};
+ return dictSignal(signal,ptr,2,
+ 1 /*use masternode id*/,
+ 100,
+ WAIT_CREATE_INDX_REQ,
+ -1,
+ errCodes,noErrCodes);
+}
+
+void
+NdbDictInterface::execCREATE_INDX_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ //CreateTableConf* const conf = CAST_CONSTPTR(CreateTableConf, signal->getDataPtr());
+
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execCREATE_INDX_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ const CreateIndxRef* const ref = CAST_CONSTPTR(CreateIndxRef, signal->getDataPtr());
+ m_error.code = ref->getErrorCode();
+ m_waiter.signal(NO_WAIT);
+}
+
+/*****************************************************************
+ * Drop index
+ */
+int
+NdbDictionaryImpl::dropIndex(const char * indexName,
+ const char * tableName)
+{
+ NdbIndexImpl * idx = getIndex(indexName, tableName);
+ if (idx == 0) {
+ m_error.code = 4243;
+ return -1;
+ }
+ int ret = dropIndex(*idx, tableName);
+ // If index stored in cache is incompatible with the one in the kernel
+ // we must clear the cache and try again
+ if (ret == INCOMPATIBLE_VERSION) {
+ const char * internalIndexName = (tableName)
+ ?
+ m_ndb.internalizeIndexName(getTable(tableName), indexName)
+ :
+ m_ndb.internalizeTableName(indexName); // Index is also a table
+
+ m_localHash.drop(internalIndexName);
+
+ m_globalHash->lock();
+ m_globalHash->drop(idx->m_table);
+ m_globalHash->unlock();
+ return dropIndex(indexName, tableName);
+ }
+
+ return ret;
+}
+
+int
+NdbDictionaryImpl::dropIndex(NdbIndexImpl & impl, const char * tableName)
+{
+ const char * indexName = impl.getName();
+ if (tableName || Ndb::usingFullyQualifiedNames()) {
+ NdbTableImpl * timpl = impl.m_table;
+
+ if (timpl == 0) {
+ m_error.code = 709;
+ return -1;
+ }
+
+ const char * internalIndexName = (tableName)
+ ?
+ m_ndb.internalizeIndexName(getTable(tableName), indexName)
+ :
+ m_ndb.internalizeTableName(indexName); // Index is also a table
+
+ if(impl.m_status == NdbDictionary::Object::New){
+ return dropIndex(indexName, tableName);
+ }
+
+ int ret = m_receiver.dropIndex(impl, *timpl);
+ if(ret == 0){
+ m_localHash.drop(internalIndexName);
+
+ m_globalHash->lock();
+ m_globalHash->drop(impl.m_table);
+ m_globalHash->unlock();
+ }
+
+ return ret;
+ }
+
+ m_error.code = 4243;
+ return -1;
+}
+
+int
+NdbDictInterface::dropIndex(const NdbIndexImpl & impl,
+ const NdbTableImpl & timpl)
+{
+ NdbApiSignal tSignal(m_reference);
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_DROP_INDX_REQ;
+ tSignal.theLength = DropIndxReq::SignalLength;
+
+ DropIndxReq * const req = CAST_PTR(DropIndxReq, tSignal.getDataPtrSend());
+ req->setUserRef(m_reference);
+ req->setConnectionPtr(0);
+ req->setRequestType(DropIndxReq::RT_USER);
+ req->setTableId(~0); // DICT overwrites
+ req->setIndexId(timpl.m_tableId);
+ req->setIndexVersion(timpl.m_version);
+
+ return dropIndex(&tSignal, 0);
+}
+
+int
+NdbDictInterface::dropIndex(NdbApiSignal* signal, LinearSectionPtr ptr[3])
+{
+ const int noErrCodes = 1;
+ int errCodes[noErrCodes] = {DropIndxRef::Busy};
+ int r = dictSignal(signal,NULL,0,
+ 1/*Use masternode id*/,
+ 100,
+ WAIT_DROP_INDX_REQ,
+ WAITFOR_RESPONSE_TIMEOUT,
+ errCodes,noErrCodes);
+ if(m_error.code == DropIndxRef::InvalidIndexVersion) {
+ // Clear caches and try again
+ return INCOMPATIBLE_VERSION;
+ }
+ return r;
+}
+
+void
+NdbDictInterface::execDROP_INDX_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execDROP_INDX_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ const DropIndxRef* const ref = CAST_CONSTPTR(DropIndxRef, signal->getDataPtr());
+ m_error.code = ref->getErrorCode();
+ m_waiter.signal(NO_WAIT);
+}
+
+/*****************************************************************
+ * Create event
+ */
+
+int
+NdbDictionaryImpl::createEvent(NdbEventImpl & evnt)
+{
+ NdbTableImpl* tab = getTable(evnt.getTable());
+
+ if(tab == 0){
+ // m_error.code = 3249;
+ ndbout_c(":createEvent: table %s not found", evnt.getTable());
+#ifdef EVENT_DEBUG
+ ndbout_c("NdbDictionaryImpl::createEvent: table not found: %s", evnt.getTable());
+#endif
+ return -1;
+ }
+
+ evnt.m_tableId = tab->m_tableId;
+ evnt.m_tableImpl = tab;
+#ifdef EVENT_DEBUG
+ ndbout_c("Event on tableId=%d", evnt.m_tableId);
+#endif
+
+ NdbTableImpl &table = *evnt.m_tableImpl;
+
+
+ int attributeList_sz = evnt.m_attrIds.size();
+
+ for (int i = 0; i < attributeList_sz; i++) {
+ NdbColumnImpl *col_impl = table.getColumn(evnt.m_attrIds[i]);
+ if (col_impl) {
+ evnt.m_facade->addColumn(*(col_impl->m_facade));
+ } else {
+ ndbout_c("Attr id %u in table %s not found", evnt.m_attrIds[i],
+ evnt.getTable());
+ return -1;
+ }
+ }
+
+ evnt.m_attrIds.clear();
+
+ attributeList_sz = evnt.m_columns.size();
+#ifdef EVENT_DEBUG
+ ndbout_c("creating event %s", evnt.m_externalName.c_str());
+ ndbout_c("no of columns %d", evnt.m_columns.size());
+#endif
+ int pk_count = 0;
+ evnt.m_attrListBitmask.clear();
+
+ for(int i = 0; i<attributeList_sz; i++){
+ const NdbColumnImpl* col =
+ table.getColumn(evnt.m_columns[i]->m_name.c_str());
+ if(col == 0){
+ m_error.code = 4247;
+ return -1;
+ }
+ // Copy column definition
+ *evnt.m_columns[i] = *col;
+
+ if(col->m_pk){
+ pk_count++;
+ }
+
+ evnt.m_attrListBitmask.set(col->m_attrId);
+ }
+
+ // Sort index attributes according to primary table (using insertion sort)
+ for(int i = 1; i < attributeList_sz; i++) {
+ NdbColumnImpl* temp = evnt.m_columns[i];
+ unsigned int j = i;
+ while((j > 0) && (evnt.m_columns[j - 1]->m_attrId > temp->m_attrId)) {
+ evnt.m_columns[j] = evnt.m_columns[j - 1];
+ j--;
+ }
+ evnt.m_columns[j] = temp;
+ }
+ // Check for illegal duplicate attributes
+ for(int i = 1; i<attributeList_sz; i++) {
+ if (evnt.m_columns[i-1]->m_attrId == evnt.m_columns[i]->m_attrId) {
+ m_error.code = 4258;
+ return -1;
+ }
+ }
+
+#ifdef EVENT_DEBUG
+ char buf[128] = {0};
+ evnt.m_attrListBitmask.getText(buf);
+ ndbout_c("createEvent: mask = %s", buf);
+#endif
+
+ // NdbDictInterface m_receiver;
+ return m_receiver.createEvent(m_ndb, evnt, 0 /* getFlag unset */);
+}
+
+int
+NdbDictInterface::createEvent(class Ndb & ndb,
+ NdbEventImpl & evnt,
+ int getFlag)
+{
+ NdbApiSignal tSignal(m_reference);
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_CREATE_EVNT_REQ;
+ if (getFlag)
+ tSignal.theLength = CreateEvntReq::SignalLengthGet;
+ else
+ tSignal.theLength = CreateEvntReq::SignalLengthCreate;
+
+ CreateEvntReq * const req = CAST_PTR(CreateEvntReq, tSignal.getDataPtrSend());
+
+ req->setUserRef(m_reference);
+ req->setUserData(0);
+
+ if (getFlag) {
+ // getting event from Dictionary
+ req->setRequestType(CreateEvntReq::RT_USER_GET);
+ } else {
+ // creating event in Dictionary
+ req->setRequestType(CreateEvntReq::RT_USER_CREATE);
+ req->setTableId(evnt.m_tableId);
+ req->setAttrListBitmask(evnt.m_attrListBitmask);
+ req->setEventType(evnt.mi_type);
+ }
+
+ UtilBufferWriter w(m_buffer);
+
+ const size_t len = strlen(evnt.m_externalName.c_str()) + 1;
+ if(len > MAX_TAB_NAME_SIZE) {
+ m_error.code = 4241;
+ return -1;
+ }
+
+ w.add(SimpleProperties::StringValue, evnt.m_externalName.c_str());
+
+ if (getFlag == 0)
+ w.add(SimpleProperties::StringValue,
+ ndb.internalizeTableName(evnt.m_tableName.c_str()));
+
+ LinearSectionPtr ptr[3];
+ ptr[0].p = (Uint32*)m_buffer.get_data();
+ ptr[0].sz = (m_buffer.length()+3) >> 2;
+
+ int ret = createEvent(&tSignal, ptr, 1);
+
+ if (ret) {
+ return ret;
+ }
+
+ char *dataPtr = (char *)m_buffer.get_data();
+ unsigned int lenCreateEvntConf = *((unsigned int *)dataPtr);
+ dataPtr += sizeof(lenCreateEvntConf);
+ CreateEvntConf const * evntConf = (CreateEvntConf *)dataPtr;
+ dataPtr += lenCreateEvntConf;
+
+ // NdbEventImpl *evntImpl = (NdbEventImpl *)evntConf->getUserData();
+
+ if (getFlag) {
+ evnt.m_tableId = evntConf->getTableId();
+ evnt.m_attrListBitmask = evntConf->getAttrListBitmask();
+ evnt.mi_type = evntConf->getEventType();
+ evnt.setTable(dataPtr);
+ } else {
+ if (evnt.m_tableId != evntConf->getTableId() ||
+ //evnt.m_attrListBitmask != evntConf->getAttrListBitmask() ||
+ evnt.mi_type != evntConf->getEventType()) {
+ ndbout_c("ERROR*************");
+ return 1;
+ }
+ }
+
+ evnt.m_eventId = evntConf->getEventId();
+ evnt.m_eventKey = evntConf->getEventKey();
+
+ return ret;
+}
+
+int
+NdbDictInterface::createEvent(NdbApiSignal* signal,
+ LinearSectionPtr ptr[3], int noLSP)
+{
+ const int noErrCodes = 1;
+ int errCodes[noErrCodes] = {CreateEvntRef::Busy};
+ return dictSignal(signal,ptr,noLSP,
+ 1 /*use masternode id*/,
+ 100,
+ WAIT_CREATE_INDX_REQ /*WAIT_CREATE_EVNT_REQ*/,
+ -1,
+ errCodes,noErrCodes, CreateEvntRef::Temporary);
+}
+
+int
+NdbDictionaryImpl::executeSubscribeEvent(NdbEventImpl & ev)
+{
+ // NdbDictInterface m_receiver;
+ return m_receiver.executeSubscribeEvent(m_ndb, ev);
+}
+
+int
+NdbDictInterface::executeSubscribeEvent(class Ndb & ndb,
+ NdbEventImpl & evnt)
+{
+ NdbApiSignal tSignal(m_reference);
+ // tSignal.theReceiversBlockNumber = SUMA;
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_SUB_START_REQ;
+ tSignal.theLength = SubStartReq::SignalLength2;
+
+ SubStartReq * sumaStart = CAST_PTR(SubStartReq, tSignal.getDataPtrSend());
+
+ sumaStart->subscriptionId = evnt.m_eventId;
+ sumaStart->subscriptionKey = evnt.m_eventKey;
+ sumaStart->part = SubscriptionData::TableData;
+ sumaStart->subscriberData = evnt.m_bufferId & 0xFF;
+ sumaStart->subscriberRef = m_reference;
+
+ return executeSubscribeEvent(&tSignal, NULL);
+}
+
+int
+NdbDictInterface::executeSubscribeEvent(NdbApiSignal* signal,
+ LinearSectionPtr ptr[3])
+{
+ return dictSignal(signal,NULL,0,
+ 1 /*use masternode id*/,
+ 100,
+ WAIT_CREATE_INDX_REQ /*WAIT_CREATE_EVNT_REQ*/,
+ -1,
+ NULL,0);
+}
+
+int
+NdbDictionaryImpl::stopSubscribeEvent(NdbEventImpl & ev)
+{
+ // NdbDictInterface m_receiver;
+ return m_receiver.stopSubscribeEvent(m_ndb, ev);
+}
+
+int
+NdbDictInterface::stopSubscribeEvent(class Ndb & ndb,
+ NdbEventImpl & evnt)
+{
+#ifdef EVENT_DEBUG
+ ndbout_c("SUB_STOP_REQ");
+#endif
+
+ NdbApiSignal tSignal(m_reference);
+ // tSignal.theReceiversBlockNumber = SUMA;
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_SUB_STOP_REQ;
+ tSignal.theLength = SubStopReq::SignalLength;
+
+ SubStopReq * sumaStop = CAST_PTR(SubStopReq, tSignal.getDataPtrSend());
+
+ sumaStop->subscriptionId = evnt.m_eventId;
+ sumaStop->subscriptionKey = evnt.m_eventKey;
+ sumaStop->subscriberData = evnt.m_bufferId & 0xFF;
+ sumaStop->part = (Uint32) SubscriptionData::TableData;
+ sumaStop->subscriberRef = m_reference;
+
+ return stopSubscribeEvent(&tSignal, NULL);
+}
+
+int
+NdbDictInterface::stopSubscribeEvent(NdbApiSignal* signal,
+ LinearSectionPtr ptr[3])
+{
+ return dictSignal(signal,NULL,0,
+ 1 /*use masternode id*/,
+ 100,
+ WAIT_CREATE_INDX_REQ /*WAIT_SUB_STOP__REQ*/,
+ -1,
+ NULL,0);
+}
+
+NdbEventImpl *
+NdbDictionaryImpl::getEvent(const char * eventName)
+{
+ NdbEventImpl *ev = new NdbEventImpl();
+
+ if (ev == NULL) {
+ return NULL;
+ }
+
+ ev->setName(eventName);
+
+ int ret = m_receiver.createEvent(m_ndb, *ev, 1 /* getFlag set */);
+
+ if (ret) {
+ delete ev;
+ return NULL;
+ }
+
+ // We only have the table name with internal name
+ ev->setTable(m_ndb.externalizeTableName(ev->getTable()));
+ ev->m_tableImpl = getTable(ev->getTable());
+
+ // get the columns from the attrListBitmask
+
+ NdbTableImpl &table = *ev->m_tableImpl;
+ AttributeMask & mask = ev->m_attrListBitmask;
+ int attributeList_sz = mask.count();
+ int id = -1;
+
+#ifdef EVENT_DEBUG
+ ndbout_c("NdbDictionaryImpl::getEvent attributeList_sz = %d",
+ attributeList_sz);
+ char buf[128] = {0};
+ mask.getText(buf);
+ ndbout_c("mask = %s", buf);
+#endif
+
+ for(int i = 0; i < attributeList_sz; i++) {
+ id++; while (!mask.get(id)) id++;
+
+ const NdbColumnImpl* col = table.getColumn(id);
+ if(col == 0) {
+#ifdef EVENT_DEBUG
+ ndbout_c("NdbDictionaryImpl::getEvent could not find column id %d", id);
+#endif
+ m_error.code = 4247;
+ delete ev;
+ return NULL;
+ }
+ NdbColumnImpl* new_col = new NdbColumnImpl;
+ // Copy column definition
+ *new_col = *col;
+
+ ev->m_columns.push_back(new_col);
+ }
+
+ return ev;
+}
+
+void
+NdbDictInterface::execCREATE_EVNT_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+#ifdef EVENT_DEBUG
+ ndbout << "NdbDictionaryImpl.cpp: execCREATE_EVNT_CONF" << endl;
+#endif
+ m_buffer.clear();
+ unsigned int len = signal->getLength() << 2;
+ m_buffer.append((char *)&len, sizeof(len));
+ m_buffer.append(signal->getDataPtr(), len);
+
+ if (signal->m_noOfSections > 0) {
+ m_buffer.append((char *)ptr[0].p, strlen((char *)ptr[0].p)+1);
+ }
+
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execCREATE_EVNT_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+#ifdef EVENT_DEBUG
+ ndbout << "NdbDictionaryImpl.cpp: execCREATE_EVNT_REF" << endl;
+ ndbout << "Exiting" << endl;
+ exit(-1);
+#endif
+
+ const CreateEvntRef* const ref = CAST_CONSTPTR(CreateEvntRef, signal->getDataPtr());
+ m_error.code = ref->getErrorCode();
+#ifdef EVENT_DEBUG
+ ndbout_c("execCREATE_EVNT_REF");
+ ndbout_c("ErrorCode %u", ref->getErrorCode());
+ ndbout_c("Errorline %u", ref->getErrorLine());
+ ndbout_c("ErrorNode %u", ref->getErrorNode());
+#endif
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execSUB_STOP_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+#ifdef EVENT_DEBUG
+ ndbout << "Got GSN_SUB_STOP_CONF" << endl;
+#endif
+ // SubRemoveConf * const sumaRemoveConf = CAST_CONSTPTR(SubRemoveConf, signal->getDataPtr());
+
+ // Uint32 subscriptionId = sumaRemoveConf->subscriptionId;
+ // Uint32 subscriptionKey = sumaRemoveConf->subscriptionKey;
+ // Uint32 senderData = sumaRemoveConf->senderData;
+
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execSUB_STOP_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+#ifdef EVENT_DEBUG
+ ndbout << "Got GSN_SUB_STOP_REF" << endl;
+#endif
+ // SubRemoveConf * const sumaRemoveRef = CAST_CONSTPTR(SubRemoveRef, signal->getDataPtr());
+
+ // Uint32 subscriptionId = sumaRemoveRef->subscriptionId;
+ // Uint32 subscriptionKey = sumaRemoveRef->subscriptionKey;
+ // Uint32 senderData = sumaRemoveRef->senderData;
+
+ m_error.code = 1;
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execSUB_START_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+#ifdef EVENT_DEBUG
+ ndbout << "Got GSN_SUB_START_CONF" << endl;
+#endif
+ const SubStartConf * const sumaStartConf = CAST_CONSTPTR(SubStartConf, signal->getDataPtr());
+
+ // Uint32 subscriptionId = sumaStartConf->subscriptionId;
+ // Uint32 subscriptionKey = sumaStartConf->subscriptionKey;
+ SubscriptionData::Part part =
+ (SubscriptionData::Part)sumaStartConf->part;
+ // Uint32 subscriberData = sumaStartConf->subscriberData;
+
+ switch(part) {
+ case SubscriptionData::MetaData: {
+#ifdef EVENT_DEBUG
+ ndbout << "SubscriptionData::MetaData" << endl;
+#endif
+ m_error.code = 1;
+ break;
+ }
+ case SubscriptionData::TableData: {
+#ifdef EVENT_DEBUG
+ ndbout << "SubscriptionData::TableData" << endl;
+#endif
+ break;
+ }
+ default: {
+#ifdef EVENT_DEBUG
+ ndbout_c("NdbDictInterface::execSUB_START_CONF wrong data");
+#endif
+ m_error.code = 1;
+ break;
+ }
+ }
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execSUB_START_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+#ifdef EVENT_DEBUG
+ ndbout << "Got GSN_SUB_START_REF" << endl;
+#endif
+ m_error.code = 1;
+ m_waiter.signal(NO_WAIT);
+}
+void
+NdbDictInterface::execSUB_GCP_COMPLETE_REP(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+ const SubGcpCompleteRep * const rep = CAST_CONSTPTR(SubGcpCompleteRep, signal->getDataPtr());
+
+ const Uint32 gci = rep->gci;
+ // const Uint32 senderRef = rep->senderRef;
+ const Uint32 subscriberData = rep->subscriberData;
+
+ const Uint32 bufferId = subscriberData;
+
+ const Uint32 ref = signal->theSendersBlockRef;
+
+ NdbApiSignal tSignal(m_reference);
+ SubGcpCompleteAcc * acc = CAST_PTR(SubGcpCompleteAcc, tSignal.getDataPtrSend());
+
+ acc->rep = *rep;
+
+ tSignal.theReceiversBlockNumber = refToBlock(ref);
+ tSignal.theVerId_signalNumber = GSN_SUB_GCP_COMPLETE_ACC;
+ tSignal.theLength = SubGcpCompleteAcc::SignalLength;
+
+ Uint32 aNodeId = refToNode(ref);
+
+ // m_transporter->lock_mutex();
+ int r;
+ r = m_transporter->sendSignal(&tSignal, aNodeId);
+ // m_transporter->unlock_mutex();
+
+ NdbGlobalEventBufferHandle::latestGCI(bufferId, gci);
+}
+
+void
+NdbDictInterface::execSUB_TABLE_DATA(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+#ifdef EVENT_DEBUG
+ const char * FNAME = "NdbDictInterface::execSUB_TABLE_DATA";
+#endif
+ //TODO
+ const SubTableData * const sdata = CAST_CONSTPTR(SubTableData, signal->getDataPtr());
+
+ // const Uint32 gci = sdata->gci;
+ // const Uint32 operation = sdata->operation;
+ // const Uint32 tableId = sdata->tableId;
+ // const Uint32 noOfAttrs = sdata->noOfAttributes;
+ // const Uint32 dataLen = sdata->dataSize;
+ const Uint32 subscriberData = sdata->subscriberData;
+ // const Uint32 logType = sdata->logType;
+
+ for (int i=signal->m_noOfSections;i < 3; i++) {
+ ptr[i].p = NULL;
+ ptr[i].sz = 0;
+ }
+#ifdef EVENT_DEBUG
+ ndbout_c("%s: senderData %d, gci %d, operation %d, tableId %d, noOfAttrs %d, dataLen %d",
+ FNAME, subscriberData, gci, operation, tableId, noOfAttrs, dataLen);
+ ndbout_c("ptr[0] %u %u ptr[1] %u %u ptr[2] %u %u\n",
+ ptr[0].p,ptr[0].sz,ptr[1].p,ptr[1].sz,ptr[2].p,ptr[2].sz);
+#endif
+ const Uint32 bufferId = subscriberData;
+
+ NdbGlobalEventBufferHandle::insertDataL(bufferId,
+ sdata, ptr);
+}
+
+/*****************************************************************
+ * Drop event
+ */
+int
+NdbDictionaryImpl::dropEvent(const char * eventName)
+{
+ NdbEventImpl *ev = new NdbEventImpl();
+ ev->setName(eventName);
+ int ret = m_receiver.dropEvent(*ev);
+ delete ev;
+
+ // printf("__________________RET %u\n", ret);
+ return ret;
+}
+
+int
+NdbDictInterface::dropEvent(const NdbEventImpl &evnt)
+{
+ NdbApiSignal tSignal(m_reference);
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_DROP_EVNT_REQ;
+ tSignal.theLength = DropEvntReq::SignalLength;
+
+ DropEvntReq * const req = CAST_PTR(DropEvntReq, tSignal.getDataPtrSend());
+
+ req->setUserRef(m_reference);
+ req->setUserData(0);
+
+ UtilBufferWriter w(m_buffer);
+
+ w.add(SimpleProperties::StringValue, evnt.m_externalName.c_str());
+
+ LinearSectionPtr ptr[1];
+ ptr[0].p = (Uint32*)m_buffer.get_data();
+ ptr[0].sz = (m_buffer.length()+3) >> 2;
+
+ return dropEvent(&tSignal, ptr, 1);
+}
+
+int
+NdbDictInterface::dropEvent(NdbApiSignal* signal,
+ LinearSectionPtr ptr[3], int noLSP)
+{
+ //TODO
+ const int noErrCodes = 1;
+ int errCodes[noErrCodes] = {DropEvntRef::Busy};
+ return dictSignal(signal,ptr,noLSP,
+ 1 /*use masternode id*/,
+ 100,
+ WAIT_CREATE_INDX_REQ /*WAIT_CREATE_EVNT_REQ*/,
+ -1,
+ errCodes,noErrCodes, DropEvntRef::Temporary);
+}
+void
+NdbDictInterface::execDROP_EVNT_CONF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+#ifdef EVENT_DEBUG
+ ndbout << "NdbDictionaryImpl.cpp: execDROP_EVNT_CONF" << endl;
+#endif
+
+ m_waiter.signal(NO_WAIT);
+}
+
+void
+NdbDictInterface::execDROP_EVNT_REF(NdbApiSignal * signal,
+ LinearSectionPtr ptr[3])
+{
+#ifdef EVENT_DEBUG
+ ndbout << "NdbDictionaryImpl.cpp: execDROP_EVNT_REF" << endl;
+#endif
+ const DropEvntRef* const ref = CAST_CONSTPTR(DropEvntRef, signal->getDataPtr());
+ m_error.code = ref->getErrorCode();
+
+#if 0
+ ndbout_c("execDROP_EVNT_REF");
+ ndbout_c("ErrorCode %u", ref->getErrorCode());
+ ndbout_c("Errorline %u", ref->getErrorLine());
+ ndbout_c("ErrorNode %u", ref->getErrorNode());
+#endif
+
+ m_waiter.signal(NO_WAIT);
+}
+
+/*****************************************************************
+ * List objects or indexes
+ */
+int
+NdbDictionaryImpl::listObjects(List& list, NdbDictionary::Object::Type type)
+{
+ ListTablesReq req;
+ req.requestData = 0;
+ req.setTableType(getKernelConstant(type, objectTypeMapping, 0));
+ req.setListNames(true);
+ return m_receiver.listObjects(list, req.requestData);
+}
+
+int
+NdbDictionaryImpl::listIndexes(List& list, const char * tableName)
+{
+ ListTablesReq
+ req;
+ NdbTableImpl* impl = getTable(tableName);
+ if (impl == 0)
+ return -1;
+ req.requestData = 0;
+ req.setTableId(impl->m_tableId);
+ req.setListNames(true);
+ req.setListIndexes(true);
+ return m_receiver.listObjects(list, req.requestData);
+}
+
+int
+NdbDictInterface::listObjects(NdbDictionary::Dictionary::List& list,
+ Uint32 requestData)
+{
+ NdbApiSignal tSignal(m_reference);
+ ListTablesReq* const req = CAST_PTR(ListTablesReq, tSignal.getDataPtrSend());
+ req->senderRef = m_reference;
+ req->senderData = 0;
+ req->requestData = requestData;
+ tSignal.theReceiversBlockNumber = DBDICT;
+ tSignal.theVerId_signalNumber = GSN_LIST_TABLES_REQ;
+ tSignal.theLength = ListTablesReq::SignalLength;
+ if (listObjects(&tSignal) != 0)
+ return -1;
+ // count
+ const Uint32* data = (const Uint32*)m_buffer.get_data();
+ const unsigned length = m_buffer.length() / 4;
+ list.count = 0;
+ bool ok = true;
+ unsigned pos, count;
+ pos = count = 0;
+ while (pos < length) {
+ // table id - name length - name
+ pos++;
+ if (pos >= length) {
+ ok = false;
+ break;
+ }
+ Uint32 n = (data[pos++] + 3) >> 2;
+ pos += n;
+ if (pos > length) {
+ ok = false;
+ break;
+ }
+ count++;
+ }
+ if (! ok) {
+ // bad signal data
+ m_error.code = 4213;
+ return -1;
+ }
+ list.count = count;
+ list.elements = new NdbDictionary::Dictionary::List::Element[count];
+ pos = count = 0;
+ while (pos < length) {
+ NdbDictionary::Dictionary::List::Element& element = list.elements[count];
+ Uint32 d = data[pos++];
+ element.id = ListTablesConf::getTableId(d);
+ element.type = (NdbDictionary::Object::Type)
+ getApiConstant(ListTablesConf::getTableType(d), objectTypeMapping, 0);
+ element.state = (NdbDictionary::Object::State)
+ getApiConstant(ListTablesConf::getTableState(d), objectStateMapping, 0);
+ element.store = (NdbDictionary::Object::Store)
+ getApiConstant(ListTablesConf::getTableStore(d), objectStoreMapping, 0);
+ // table or index name
+ Uint32 n = (data[pos++] + 3) >> 2;
+ BaseString databaseName;
+ BaseString schemaName;
+ BaseString objectName;
+ if ((element.type == NdbDictionary::Object::UniqueHashIndex) ||
+ (element.type == NdbDictionary::Object::HashIndex) ||
+ (element.type == NdbDictionary::Object::UniqueOrderedIndex) ||
+ (element.type == NdbDictionary::Object::OrderedIndex)) {
+ char * indexName = new char[n << 2];
+ memcpy(indexName, &data[pos], n << 2);
+ databaseName = Ndb::getDatabaseFromInternalName(indexName);
+ schemaName = Ndb::getSchemaFromInternalName(indexName);
+ objectName = BaseString(Ndb::externalizeIndexName(indexName));
+ delete [] indexName;
+ } else if ((element.type == NdbDictionary::Object::SystemTable) ||
+ (element.type == NdbDictionary::Object::UserTable)) {
+ char * tableName = new char[n << 2];
+ memcpy(tableName, &data[pos], n << 2);
+ databaseName = Ndb::getDatabaseFromInternalName(tableName);
+ schemaName = Ndb::getSchemaFromInternalName(tableName);
+ objectName = BaseString(Ndb::externalizeTableName(tableName));
+ delete [] tableName;
+ }
+ else {
+ char * otherName = new char[n << 2];
+ memcpy(otherName, &data[pos], n << 2);
+ objectName = BaseString(otherName);
+ delete [] otherName;
+ }
+ element.database = new char[databaseName.length() + 1];
+ strcpy(element.database, databaseName.c_str());
+ element.schema = new char[schemaName.length() + 1];
+ strcpy(element.schema, schemaName.c_str());
+ element.name = new char[objectName.length() + 1];
+ strcpy(element.name, objectName.c_str());
+ pos += n;
+ count++;
+ }
+ return 0;
+}
+
+int
+NdbDictInterface::listObjects(NdbApiSignal* signal)
+{
+ const Uint32 RETRIES = 100;
+ for (Uint32 i = 0; i < RETRIES; i++) {
+ m_buffer.clear();
+ // begin protected
+ m_transporter->lock_mutex();
+ Uint16 aNodeId = m_transporter->get_an_alive_node();
+ if (aNodeId == 0) {
+ m_error.code = 4009;
+ m_transporter->unlock_mutex();
+ return -1;
+ }
+ if (m_transporter->sendSignal(signal, aNodeId) != 0) {
+ m_transporter->unlock_mutex();
+ continue;
+ }
+ m_error.code = 0;
+ m_waiter.m_node = aNodeId;
+ m_waiter.m_state = WAIT_LIST_TABLES_CONF;
+ m_waiter.wait(WAITFOR_RESPONSE_TIMEOUT);
+ // end protected
+ if (m_waiter.m_state == NO_WAIT && m_error.code == 0)
+ return 0;
+ if (m_waiter.m_state == WAIT_NODE_FAILURE)
+ continue;
+ return -1;
+ }
+ return -1;
+}
+
+void
+NdbDictInterface::execLIST_TABLES_CONF(NdbApiSignal* signal,
+ LinearSectionPtr ptr[3])
+{
+ const unsigned off = ListTablesConf::HeaderLength;
+ const unsigned len = (signal->getLength() - off);
+ m_buffer.append(signal->getDataPtr() + off, len << 2);
+ if (signal->getLength() < ListTablesConf::SignalLength) {
+ // last signal has less than full length
+ m_waiter.signal(NO_WAIT);
+ }
+}
diff --git a/ndb/src/ndbapi/NdbDictionaryImpl.hpp b/ndb/src/ndbapi/NdbDictionaryImpl.hpp
new file mode 100644
index 00000000000..f6b0644ea15
--- /dev/null
+++ b/ndb/src/ndbapi/NdbDictionaryImpl.hpp
@@ -0,0 +1,653 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbDictionaryImpl_H
+#define NdbDictionaryImpl_H
+
+#include <ndb_types.h>
+#include <kernel_types.h>
+#include <ndb_limits.h>
+#include <NdbError.hpp>
+#include <BaseString.hpp>
+#include <Vector.hpp>
+#include <UtilBuffer.hpp>
+#include <NdbDictionary.hpp>
+#include <Bitmask.hpp>
+#include <AttributeList.hpp>
+#include <Ndb.hpp>
+#include "NdbImpl.hpp"
+#include "DictCache.hpp"
+
+class NdbDictObjectImpl {
+public:
+ int m_version;
+ NdbDictionary::Object::Status m_status;
+
+ bool change();
+protected:
+ NdbDictObjectImpl() :
+ m_status(NdbDictionary::Object::New) {
+ }
+};
+
+/**
+ * Column
+ */
+class NdbColumnImpl : public NdbDictionary::Column {
+public:
+ NdbColumnImpl();
+ NdbColumnImpl(NdbDictionary::Column &); // This is not a copy constructor
+ ~NdbColumnImpl();
+ NdbColumnImpl& operator=(const NdbColumnImpl&);
+ void init();
+
+ int m_attrId;
+ BaseString m_name;
+ NdbDictionary::Column::Type m_type;
+ int m_precision;
+ int m_scale;
+ int m_length;
+
+ bool m_pk;
+ bool m_tupleKey;
+ bool m_distributionKey;
+ bool m_distributionGroup;
+ int m_distributionGroupBits;
+ bool m_nullable;
+ bool m_indexOnly;
+ bool m_autoIncrement;
+ Uint64 m_autoIncrementInitialValue;
+ BaseString m_defaultValue;
+
+ /**
+ * Internal types and sizes, and aggregates
+ */
+ Uint32 m_attrType; // type outsize API and DICT
+ Uint32 m_attrSize; // element size (size when arraySize==1)
+ Uint32 m_arraySize; // length or length+2 for Var* types
+ Uint32 m_keyInfoPos;
+ Uint32 m_extType; // used by restore (kernel type in versin v2x)
+ bool getInterpretableType() const ;
+
+ /**
+ * Equality/assign
+ */
+ bool equal(const NdbColumnImpl&) const;
+ void assign(const NdbColumnImpl&);
+
+ static NdbColumnImpl & getImpl(NdbDictionary::Column & t);
+ static const NdbColumnImpl & getImpl(const NdbDictionary::Column & t);
+ NdbDictionary::Column * m_facade;
+};
+
+class NdbTableImpl : public NdbDictionary::Table, public NdbDictObjectImpl {
+public:
+ NdbTableImpl();
+ NdbTableImpl(NdbDictionary::Table &);
+ ~NdbTableImpl();
+
+ void init();
+ void setName(const char * name);
+ const char * getName() const;
+
+ Uint32 m_changeMask;
+ Uint32 m_tableId;
+ BaseString m_internalName;
+ BaseString m_externalName;
+ BaseString m_newExternalName; // Used for alter table
+ UtilBuffer m_frm;
+ NdbDictionary::Object::FragmentType m_fragmentType;
+
+ /**
+ *
+ */
+ Uint32 m_columnHashMask;
+ Vector<Uint32> m_columnHash;
+ Vector<NdbColumnImpl *> m_columns;
+ void buildColumnHash();
+
+ bool m_logging;
+ int m_kvalue;
+ int m_minLoadFactor;
+ int m_maxLoadFactor;
+
+ NdbDictionaryImpl * m_dictionary;
+ NdbIndexImpl * m_index;
+ NdbColumnImpl * getColumn(unsigned attrId);
+ NdbColumnImpl * getColumn(const char * name);
+ const NdbColumnImpl * getColumn(unsigned attrId) const;
+ const NdbColumnImpl * getColumn(const char * name) const;
+
+ /**
+ * Index only stuff
+ */
+ BaseString m_primaryTable;
+ NdbDictionary::Index::Type m_indexType;
+
+ /**
+ * Aggregates
+ */
+ Uint32 m_noOfKeys;
+
+ /**
+ * Equality/assign
+ */
+ bool equal(const NdbTableImpl&) const;
+ void assign(const NdbTableImpl&);
+ void clearNewProperties();
+ void copyNewProperties();
+
+ static NdbTableImpl & getImpl(NdbDictionary::Table & t);
+ static NdbTableImpl & getImpl(const NdbDictionary::Table & t);
+ NdbDictionary::Table * m_facade;
+};
+
+class NdbIndexImpl : public NdbDictionary::Index, public NdbDictObjectImpl {
+public:
+ NdbIndexImpl();
+ NdbIndexImpl(NdbDictionary::Index &);
+ ~NdbIndexImpl();
+
+ void setName(const char * name);
+ const char * getName() const;
+ void setTable(const char * table);
+ const char * getTable() const;
+
+ Uint32 m_indexId;
+ BaseString m_internalName;
+ BaseString m_externalName;
+ BaseString m_tableName;
+ Vector<NdbColumnImpl *> m_columns;
+ NdbDictionary::Index::Type m_type;
+
+ bool m_logging;
+
+ NdbTableImpl * m_table;
+
+ static NdbIndexImpl & getImpl(NdbDictionary::Index & t);
+ static NdbIndexImpl & getImpl(const NdbDictionary::Index & t);
+ NdbDictionary::Index * m_facade;
+};
+
+class NdbEventImpl : public NdbDictionary::Event, public NdbDictObjectImpl {
+public:
+ NdbEventImpl();
+ NdbEventImpl(NdbDictionary::Event &);
+ ~NdbEventImpl();
+
+ void setName(const char * name);
+ const char * getName() const;
+ void setTable(const char * table);
+ const char * getTable() const;
+ void addTableEvent(const NdbDictionary::Event::TableEvent t);
+ void setDurability(const NdbDictionary::Event::EventDurability d);
+ void addEventColumn(const NdbColumnImpl &c);
+
+ void print() {
+ ndbout_c("NdbEventImpl: id=%d, key=%d",
+ m_eventId,
+ m_eventKey);
+ };
+
+ Uint32 m_eventId;
+ Uint32 m_eventKey;
+ Uint32 m_tableId;
+ AttributeMask m_attrListBitmask;
+ //BaseString m_internalName;
+ BaseString m_externalName;
+ Uint32 mi_type;
+ NdbDictionary::Event::EventDurability m_dur;
+
+
+ NdbTableImpl *m_tableImpl;
+ BaseString m_tableName;
+ Vector<NdbColumnImpl *> m_columns;
+ Vector<unsigned> m_attrIds;
+
+ int m_bufferId;
+
+ NdbEventOperation *eventOp;
+
+ static NdbEventImpl & getImpl(NdbDictionary::Event & t);
+ static NdbEventImpl & getImpl(const NdbDictionary::Event & t);
+ NdbDictionary::Event * m_facade;
+};
+
+
+class NdbDictInterface {
+public:
+ NdbDictInterface(NdbError& err) : m_error(err) {
+ m_reference = 0;
+ m_masterNodeId = 0;
+ m_blockNumber = -1;
+ m_transporter= NULL;
+ }
+ ~NdbDictInterface();
+
+ bool setTransporter(class Ndb * ndb, class TransporterFacade * tf);
+ bool setTransporter(class TransporterFacade * tf);
+
+ // To abstract the stuff thats made in all create/drop/lists below
+ int
+ dictSignal(NdbApiSignal* signal,
+ LinearSectionPtr ptr[3], int noLPTR,
+ const int useMasterNodeId,
+ const Uint32 RETRIES,
+ const WaitSignalType wst,
+ const int theWait,
+ const int *errcodes,
+ const int noerrcodes,
+ const int temporaryMask = 0);
+
+ int createOrAlterTable(class Ndb & ndb, NdbTableImpl &, bool alter);
+
+ int createTable(class Ndb & ndb, NdbTableImpl &);
+ int createTable(NdbApiSignal* signal, LinearSectionPtr ptr[3]);
+
+ int alterTable(class Ndb & ndb, NdbTableImpl &);
+ int alterTable(NdbApiSignal* signal, LinearSectionPtr ptr[3]);
+
+ int createIndex(class Ndb & ndb,
+ NdbIndexImpl &,
+ const NdbTableImpl &);
+ int createIndex(NdbApiSignal* signal, LinearSectionPtr ptr[3]);
+
+ int createEvent(class Ndb & ndb, NdbEventImpl &, int getFlag);
+ int createEvent(NdbApiSignal* signal, LinearSectionPtr ptr[3], int noLSP);
+
+ int dropTable(const NdbTableImpl &);
+ int dropTable(NdbApiSignal* signal, LinearSectionPtr ptr[3]);
+
+ int dropIndex(const NdbIndexImpl &, const NdbTableImpl &);
+ int dropIndex(NdbApiSignal* signal, LinearSectionPtr ptr[3]);
+
+ int dropEvent(const NdbEventImpl &);
+ int dropEvent(NdbApiSignal* signal, LinearSectionPtr ptr[3], int noLSP);
+
+ int executeSubscribeEvent(class Ndb & ndb, NdbEventImpl &);
+ int executeSubscribeEvent(NdbApiSignal* signal, LinearSectionPtr ptr[3]);
+
+ int stopSubscribeEvent(class Ndb & ndb, NdbEventImpl &);
+ int stopSubscribeEvent(NdbApiSignal* signal, LinearSectionPtr ptr[3]);
+
+ int listObjects(NdbDictionary::Dictionary::List& list, Uint32 requestData);
+ int listObjects(NdbApiSignal* signal);
+
+ NdbTableImpl * getTable(int tableId);
+ NdbTableImpl * getTable(const char * name);
+ NdbTableImpl * getTable(class NdbApiSignal * signal,
+ LinearSectionPtr ptr[3],
+ Uint32 noOfSections);
+
+ static int parseTableInfo(NdbTableImpl ** dst,
+ const Uint32 * data, Uint32 len);
+
+ NdbError & m_error;
+private:
+ Uint32 m_reference;
+ Uint32 m_masterNodeId;
+ int m_blockNumber;
+
+ NdbWaiter m_waiter;
+ class TransporterFacade * m_transporter;
+
+ friend class Ndb;
+ static void execSignal(void* dictImpl,
+ class NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]);
+
+ static void execNodeStatus(void* dictImpl, NodeId,
+ bool alive, bool nfCompleted);
+
+ void execGET_TABINFO_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execGET_TABINFO_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execCREATE_TABLE_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execCREATE_TABLE_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execALTER_TABLE_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execALTER_TABLE_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+
+ void execCREATE_INDX_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execCREATE_INDX_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execDROP_INDX_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execDROP_INDX_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+
+ void execCREATE_EVNT_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execCREATE_EVNT_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execSUB_START_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execSUB_START_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execSUB_TABLE_DATA(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execSUB_GCP_COMPLETE_REP(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execSUB_STOP_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execSUB_STOP_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execDROP_EVNT_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execDROP_EVNT_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+
+ void execDROP_TABLE_REF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execDROP_TABLE_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+ void execLIST_TABLES_CONF(NdbApiSignal *, LinearSectionPtr ptr[3]);
+
+ Uint32 m_fragmentId;
+ UtilBuffer m_buffer;
+};
+
+class NdbDictionaryImpl : public NdbDictionary::Dictionary {
+public:
+ NdbDictionaryImpl(Ndb &ndb);
+ NdbDictionaryImpl(Ndb &ndb, NdbDictionary::Dictionary & f);
+ ~NdbDictionaryImpl();
+
+ bool setTransporter(class Ndb * ndb, class TransporterFacade * tf);
+ bool setTransporter(class TransporterFacade * tf);
+
+ int createTable(NdbTableImpl &t)
+ {
+ return m_receiver.createTable(m_ndb, t);
+ }
+ int alterTable(NdbTableImpl &t);
+ int dropTable(const char * name);
+ int dropTable(NdbTableImpl &);
+ int invalidateObject(NdbTableImpl &);
+ int removeCachedObject(NdbTableImpl &);
+
+ int createIndex(NdbIndexImpl &ix);
+ int dropIndex(const char * indexName,
+ const char * tableName);
+ int dropIndex(NdbIndexImpl &, const char * tableName);
+ NdbTableImpl * getIndexTable(NdbIndexImpl * index,
+ NdbTableImpl * table);
+
+ int createEvent(NdbEventImpl &);
+ int dropEvent(const char * eventName);
+
+ int executeSubscribeEvent(NdbEventImpl &);
+ int stopSubscribeEvent(NdbEventImpl &);
+
+ int listObjects(List& list, NdbDictionary::Object::Type type);
+ int listIndexes(List& list, const char * tableName);
+
+ NdbTableImpl * getTable(const char * tableName);
+ NdbTableImpl * getTableImpl(const char * internalName);
+ NdbIndexImpl * getIndex(const char * indexName,
+ const char * tableName);
+ NdbIndexImpl * getIndexImpl(const char * name, const char * internalName);
+ NdbEventImpl * getEvent(const char * eventName);
+ NdbEventImpl * getEventImpl(const char * internalName);
+
+ const NdbError & getNdbError() const;
+ NdbError m_error;
+
+ LocalDictCache m_localHash;
+ GlobalDictCache * m_globalHash;
+
+ static NdbDictionaryImpl & getImpl(NdbDictionary::Dictionary & t);
+ static const NdbDictionaryImpl & getImpl(const NdbDictionary::Dictionary &t);
+ NdbDictionary::Dictionary * m_facade;
+
+ NdbDictInterface m_receiver;
+ Ndb & m_ndb;
+};
+
+inline
+NdbEventImpl &
+NdbEventImpl::getImpl(const NdbDictionary::Event & t){
+ return t.m_impl;
+}
+
+inline
+NdbEventImpl &
+NdbEventImpl::getImpl(NdbDictionary::Event & t){
+ return t.m_impl;
+}
+
+inline
+NdbColumnImpl &
+NdbColumnImpl::getImpl(NdbDictionary::Column & t){
+ return t.m_impl;
+}
+
+inline
+const NdbColumnImpl &
+NdbColumnImpl::getImpl(const NdbDictionary::Column & t){
+ return t.m_impl;
+}
+
+inline
+bool
+NdbColumnImpl::getInterpretableType() const {
+ return (m_type == NdbDictionary::Column::Unsigned ||
+ m_type == NdbDictionary::Column::Bigunsigned);
+}
+
+inline
+NdbTableImpl &
+NdbTableImpl::getImpl(NdbDictionary::Table & t){
+ return t.m_impl;
+}
+
+inline
+NdbTableImpl &
+NdbTableImpl::getImpl(const NdbDictionary::Table & t){
+ return t.m_impl;
+}
+
+inline
+NdbColumnImpl *
+NdbTableImpl::getColumn(unsigned attrId){
+ if(m_columns.size() > attrId){
+ return m_columns[attrId];
+ }
+ return 0;
+}
+
+inline
+Uint32
+Hash( const char* str ){
+ Uint32 h = 0;
+ Uint32 len = strlen(str);
+ while(len >= 4){
+ h = (h << 5) + h + str[0];
+ h = (h << 5) + h + str[1];
+ h = (h << 5) + h + str[2];
+ h = (h << 5) + h + str[3];
+ len -= 4;
+ str += 4;
+ }
+
+ switch(len){
+ case 3:
+ h = (h << 5) + h + *str++;
+ case 2:
+ h = (h << 5) + h + *str++;
+ case 1:
+ h = (h << 5) + h + *str++;
+ }
+ return h + h;
+}
+
+
+inline
+NdbColumnImpl *
+NdbTableImpl::getColumn(const char * name){
+
+ Uint32 sz = m_columns.size();
+ NdbColumnImpl** cols = m_columns.getBase();
+ const Uint32 * hashtable = m_columnHash.getBase();
+
+ if(sz > 5 && false){
+ Uint32 hashValue = Hash(name) & 0xFFFE;
+ Uint32 bucket = hashValue & m_columnHashMask;
+ bucket = (bucket < sz ? bucket : bucket - sz);
+ hashtable += bucket;
+ Uint32 tmp = * hashtable;
+ if((tmp & 1) == 1 ){ // No chaining
+ sz = 1;
+ } else {
+ sz = (tmp >> 16);
+ hashtable += (tmp & 0xFFFE) >> 1;
+ tmp = * hashtable;
+ }
+ do {
+ if(hashValue == (tmp & 0xFFFE)){
+ NdbColumnImpl* col = cols[tmp >> 16];
+ if(strcmp(name, col->m_name.c_str()) == 0){
+ return col;
+ }
+ }
+ hashtable++;
+ tmp = * hashtable;
+ } while(--sz > 0);
+#if 0
+ Uint32 dir = m_columnHash[bucket];
+ Uint32 pos = bucket + ((dir & 0xFFFE) >> 1);
+ Uint32 cnt = dir >> 16;
+ ndbout_c("col: %s hv: %x bucket: %d dir: %x pos: %d cnt: %d tmp: %d -> 0",
+ name, hashValue, bucket, dir, pos, cnt, tmp);
+#endif
+ return 0;
+ } else {
+ for(Uint32 i = 0; i<sz; i++){
+ NdbColumnImpl* col = * cols++;
+ if(col != 0 && strcmp(name, col->m_name.c_str()) == 0)
+ return col;
+ }
+ }
+ return 0;
+}
+
+inline
+const NdbColumnImpl *
+NdbTableImpl::getColumn(unsigned attrId) const {
+ if(m_columns.size() > attrId){
+ return m_columns[attrId];
+ }
+ return 0;
+}
+
+inline
+const NdbColumnImpl *
+NdbTableImpl::getColumn(const char * name) const {
+ Uint32 sz = m_columns.size();
+ NdbColumnImpl* const * cols = m_columns.getBase();
+ for(Uint32 i = 0; i<sz; i++, cols++){
+ NdbColumnImpl* col = * cols;
+ if(col != 0 && strcmp(name, col->m_name.c_str()) == 0)
+ return col;
+ }
+ return 0;
+}
+
+inline
+NdbIndexImpl &
+NdbIndexImpl::getImpl(NdbDictionary::Index & t){
+ return t.m_impl;
+}
+
+inline
+NdbIndexImpl &
+NdbIndexImpl::getImpl(const NdbDictionary::Index & t){
+ return t.m_impl;
+}
+
+inline
+NdbDictionaryImpl &
+NdbDictionaryImpl::getImpl(NdbDictionary::Dictionary & t){
+ return t.m_impl;
+}
+
+inline
+const NdbDictionaryImpl &
+NdbDictionaryImpl::getImpl(const NdbDictionary::Dictionary & t){
+ return t.m_impl;
+}
+
+/*****************************************************************
+ * Inline:d getters
+ */
+
+inline
+NdbTableImpl *
+NdbDictionaryImpl::getTable(const char * tableName)
+{
+ const char * internalTableName = m_ndb.internalizeTableName(tableName);
+
+ return getTableImpl(internalTableName);
+}
+
+inline
+NdbTableImpl *
+NdbDictionaryImpl::getTableImpl(const char * internalTableName)
+{
+ NdbTableImpl *ret = m_localHash.get(internalTableName);
+
+ if (ret != 0) {
+ return ret; // autoincrement already initialized
+ }
+
+ m_globalHash->lock();
+ ret = m_globalHash->get(internalTableName);
+ m_globalHash->unlock();
+
+ if (ret == 0){
+ ret = m_receiver.getTable(internalTableName);
+
+ m_globalHash->lock();
+ m_globalHash->put(internalTableName, ret);
+ m_globalHash->unlock();
+
+ if(ret == 0){
+ return 0;
+ }
+ }
+ m_localHash.put(internalTableName, ret);
+
+ m_ndb.theFirstTupleId[ret->getTableId()] = ~0;
+ m_ndb.theLastTupleId[ret->getTableId()] = ~0;
+
+ return ret;
+}
+
+inline
+NdbIndexImpl *
+NdbDictionaryImpl::getIndex(const char * indexName,
+ const char * tableName)
+{
+ if (tableName || Ndb::usingFullyQualifiedNames()) {
+ const char * internalIndexName = 0;
+ if (tableName) {
+ NdbTableImpl * t = getTable(tableName);
+ if (t != 0)
+ internalIndexName = m_ndb.internalizeIndexName(t, indexName);
+ } else {
+ internalIndexName = m_ndb.internalizeTableName(indexName); // Index is also a table
+ }
+ if (internalIndexName) {
+ NdbTableImpl * tab = getTableImpl(internalIndexName);
+
+ if (tab) {
+ if (tab->m_index == 0)
+ tab->m_index = getIndexImpl(indexName, internalIndexName);
+ if (tab->m_index != 0)
+ tab->m_index->m_table = tab;
+ return tab->m_index;
+ }
+ }
+ }
+
+ m_error.code = 4243;
+ return 0;
+}
+
+#endif
diff --git a/ndb/src/ndbapi/NdbEventOperation.cpp b/ndb/src/ndbapi/NdbEventOperation.cpp
new file mode 100644
index 00000000000..ebdebaffd61
--- /dev/null
+++ b/ndb/src/ndbapi/NdbEventOperation.cpp
@@ -0,0 +1,125 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/*****************************************************************************
+ * Name: NdbEventOperation.cpp
+ * Include:
+ * Link:
+ * Author: Tomas Ulin MySQL AB
+ * Date: 2003-11-21
+ * Version: 0.1
+ * Description: Event support
+ * Documentation:
+ * Adjust: 2003-11-21 Tomas Ulin First version.
+ ****************************************************************************/
+
+#include <Ndb.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <portlib/NdbMem.h>
+#include <transporter/TransporterDefinitions.hpp>
+#include <NdbEventOperation.hpp>
+#include "NdbEventOperationImpl.hpp"
+#include "NdbDictionaryImpl.hpp"
+
+NdbEventOperation::NdbEventOperation(Ndb *theNdb,
+ const char* eventName,
+ const int bufferLength)
+ : m_impl(* new NdbEventOperationImpl(*this,theNdb,
+ eventName,
+ bufferLength))
+{
+}
+
+NdbEventOperation::~NdbEventOperation()
+{
+ NdbEventOperationImpl * tmp = &m_impl;
+ if (this != tmp)
+ delete tmp;
+}
+
+NdbEventOperation::State NdbEventOperation::getState()
+{
+ return m_impl.getState();
+}
+
+NdbRecAttr *
+NdbEventOperation::getValue(const char *colName, char *aValue)
+{
+ return m_impl.getValue(colName, aValue, 0);
+}
+
+NdbRecAttr *
+NdbEventOperation::getPreValue(const char *colName, char *aValue)
+{
+ return m_impl.getValue(colName, aValue, 1);
+}
+
+int
+NdbEventOperation::execute()
+{
+ return m_impl.execute();
+}
+
+int
+NdbEventOperation::next(int *pOverrun)
+{
+ return m_impl.next(pOverrun);
+}
+
+bool
+NdbEventOperation::isConsistent()
+{
+ return m_impl.isConsistent();
+}
+
+Uint32
+NdbEventOperation::getGCI()
+{
+ return m_impl.getGCI();
+}
+
+Uint32
+NdbEventOperation::getLatestGCI()
+{
+ return m_impl.getLatestGCI();
+}
+
+NdbDictionary::Event::TableEvent
+NdbEventOperation::getEventType()
+{
+ return m_impl.getEventType();
+}
+
+void
+NdbEventOperation::print()
+{
+ m_impl.print();
+}
+
+/*
+ * Private members
+ */
+
+int
+NdbEventOperation::wait(void *p, int aMillisecondNumber)
+{
+ return NdbEventOperationImpl::wait(p, aMillisecondNumber);
+}
+
+NdbEventOperation::NdbEventOperation(NdbEventOperationImpl& impl)
+ : m_impl(impl) {};
+
diff --git a/ndb/src/ndbapi/NdbEventOperationImpl.cpp b/ndb/src/ndbapi/NdbEventOperationImpl.cpp
new file mode 100644
index 00000000000..d167b8205a2
--- /dev/null
+++ b/ndb/src/ndbapi/NdbEventOperationImpl.cpp
@@ -0,0 +1,1305 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <stdio.h>
+
+
+#include "NdbDictionaryImpl.hpp"
+#include "API.hpp"
+#include <NdbOut.hpp>
+#include <AttrType.hpp>
+#include "NdbApiSignal.hpp"
+#include "TransporterFacade.hpp"
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/CreateTable.hpp>
+#include <signaldata/CreateIndx.hpp>
+#include <signaldata/CreateEvnt.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <signaldata/DropTable.hpp>
+#include <signaldata/DropIndx.hpp>
+#include <signaldata/ListTables.hpp>
+#include <SimpleProperties.hpp>
+#include <Bitmask.hpp>
+#include <AttributeHeader.hpp>
+#include <AttributeList.hpp>
+#include <ndb_types.h>
+#include <kernel_types.h>
+#include <NdbError.hpp>
+#include <BaseString.hpp>
+#include <UtilBuffer.hpp>
+#include <NdbDictionary.hpp>
+#include <Ndb.hpp>
+#include "NdbImpl.hpp"
+#include "DictCache.hpp"
+#include <portlib/NdbMem.h>
+#include <NdbRecAttr.hpp>
+#include <NdbEventOperation.hpp>
+#include "NdbEventOperationImpl.hpp"
+
+/*
+ * Class NdbEventOperationImpl
+ *
+ *
+ */
+
+//#define EVENT_DEBUG
+
+
+NdbEventOperationImpl::NdbEventOperationImpl(NdbEventOperation &N,
+ Ndb *theNdb,
+ const char* eventName,
+ const int bufferLength)
+ : NdbEventOperation(*this), m_ndb(theNdb),
+ m_state(ERROR), m_bufferL(bufferLength)
+{
+
+ m_eventId = 0;
+ theFirstRecAttrs[0] = NULL;
+ theCurrentRecAttrs[0] = NULL;
+ theFirstRecAttrs[1] = NULL;
+ theCurrentRecAttrs[1] = NULL;
+ sdata = NULL;
+ ptr[0].p = NULL;
+ ptr[1].p = NULL;
+ ptr[2].p = NULL;
+
+ // we should lookup id in Dictionary, TODO
+ // also make sure we only have one listener on each event
+
+ if (!m_ndb) { ndbout_c("m_ndb=NULL"); return; }
+
+ NdbDictionary::Dictionary *myDict = m_ndb->getDictionary();
+ if (!myDict) { ndbout_c("getDictionary=NULL"); return; }
+
+ const NdbDictionary::Event *myEvnt = myDict->getEvent(eventName);
+ if (!myEvnt) { ndbout_c("getEvent()=NULL"); return; }
+
+ m_eventImpl = &myEvnt->m_impl;
+ if (!m_eventImpl) { ndbout_c("m_impl=NULL"); return; }
+
+ m_bufferHandle = m_ndb->getGlobalEventBufferHandle();
+ if (m_bufferHandle->m_bufferL > 0)
+ m_bufferL =m_bufferHandle->m_bufferL;
+ else
+ m_bufferHandle->m_bufferL = m_bufferL;
+
+ m_state = CREATED;
+}
+
+NdbEventOperationImpl::~NdbEventOperationImpl()
+{
+ if (sdata) NdbMem_Free(sdata);
+ for (int i=0 ; i<3; i++) {
+ if (ptr[i].p) NdbMem_Free(ptr[i].p);
+ }
+ for (int i=0 ; i<2; i++) {
+ NdbRecAttr *p = theFirstRecAttrs[i];
+ while (p) {
+ NdbRecAttr *p_next = p->next();
+ m_ndb->releaseRecAttr(p);
+ p = p_next;
+ }
+ }
+ if (m_state == NdbEventOperation::EXECUTING) {
+ stop();
+ // m_bufferHandle->dropSubscribeEvent(m_bufferId);
+ ; // We should send stop signal here
+ }
+}
+
+NdbEventOperation::State
+NdbEventOperationImpl::getState()
+{
+ return m_state;
+}
+
+NdbRecAttr*
+NdbEventOperationImpl::getValue(const char *colName, char *aValue, int n)
+{
+ if (m_state != NdbEventOperation::CREATED) {
+ ndbout_c("NdbEventOperationImpl::getValue may only be called between instantiation and execute()");
+ return NULL;
+ }
+
+ NdbColumnImpl *tAttrInfo = m_eventImpl->m_tableImpl->getColumn(colName);
+
+ if (tAttrInfo == NULL) {
+ ndbout_c("NdbEventOperationImpl::getValue attribute %s not found",colName);
+ return NULL;
+ }
+
+ return NdbEventOperationImpl::getValue(tAttrInfo, aValue, n);
+}
+
+NdbRecAttr*
+NdbEventOperationImpl::getValue(const NdbColumnImpl *tAttrInfo, char *aValue, int n)
+{
+ // Insert Attribute Id into ATTRINFO part.
+ NdbRecAttr *&theFirstRecAttr = theFirstRecAttrs[n];
+ NdbRecAttr *&theCurrentRecAttr = theCurrentRecAttrs[n];
+
+ /************************************************************************
+ * Get a Receive Attribute object and link it into the operation object.
+ ************************************************************************/
+ NdbRecAttr *tRecAttr = m_ndb->getRecAttr();
+ if (tRecAttr == NULL) {
+ exit(-1);
+ //setErrorCodeAbort(4000);
+ return NULL;
+ }
+
+ /**********************************************************************
+ * Now set the attribute identity and the pointer to the data in
+ * the RecAttr object
+ * Also set attribute size, array size and attribute type
+ ********************************************************************/
+ if (tRecAttr->setup(tAttrInfo, aValue)) {
+ //setErrorCodeAbort(4000);
+ m_ndb->releaseRecAttr(tRecAttr);
+ exit(-1);
+ return NULL;
+ }
+ //theErrorLine++;
+
+ tRecAttr->setUNDEFINED();
+
+ // We want to keep the list sorted to make data insertion easier later
+ if (theFirstRecAttr == NULL) {
+ theFirstRecAttr = tRecAttr;
+ theCurrentRecAttr = tRecAttr;
+ tRecAttr->next(NULL);
+ } else {
+ Uint32 tAttrId = tAttrInfo->m_attrId;
+ if (tAttrId > theCurrentRecAttr->attrId()) { // right order
+ theCurrentRecAttr->next(tRecAttr);
+ tRecAttr->next(NULL);
+ theCurrentRecAttr = tRecAttr;
+ } else if (theFirstRecAttr->next() == NULL || // only one in list
+ theFirstRecAttr->attrId() > tAttrId) {// or first
+ tRecAttr->next(theFirstRecAttr);
+ theFirstRecAttr = tRecAttr;
+ } else { // at least 2 in list and not first and not last
+ NdbRecAttr *p = theFirstRecAttr;
+ NdbRecAttr *p_next = p->next();
+ while (tAttrId > p_next->attrId()) {
+ p = p_next;
+ p_next = p->next();
+ }
+ if (tAttrId == p_next->attrId()) { // Using same attribute twice
+ tRecAttr->release(); // do I need to do this?
+ m_ndb->releaseRecAttr(tRecAttr);
+ exit(-1);
+ return NULL;
+ }
+ // this is it, between p and p_next
+ p->next(tRecAttr);
+ tRecAttr->next(p_next);
+ }
+ }
+
+ return tRecAttr;
+}
+
+int
+NdbEventOperationImpl::execute()
+{
+ NdbDictionary::Dictionary *myDict = m_ndb->getDictionary();
+ if (!myDict) {
+ ndbout_c("NdbEventOperation::execute(): getDictionary=NULL");
+ return 0;
+ }
+
+ if (theFirstRecAttrs[0] == NULL) { // defaults to get all
+
+ }
+
+ NdbDictionaryImpl & myDictImpl = NdbDictionaryImpl::getImpl(*myDict);
+
+
+ int hasSubscriber;
+ m_bufferId =
+ m_bufferHandle->prepareAddSubscribeEvent(m_eventImpl->m_eventId,
+ hasSubscriber /* return value */);
+
+ m_eventImpl->m_bufferId = m_bufferId;
+
+ int r = -1;
+ if (m_bufferId >= 0) {
+ // now we check if there's already a subscriber
+
+ if (hasSubscriber == 0) { // only excute if there's no other subscribers
+ r = myDictImpl.executeSubscribeEvent(*m_eventImpl);
+ } else {
+ r = 0;
+ }
+ if (r) {
+ //Error
+ m_bufferHandle->unprepareAddSubscribeEvent(m_bufferId);
+ m_state = NdbEventOperation::ERROR;
+ } else {
+ m_bufferHandle->addSubscribeEvent(m_bufferId, this);
+ m_state = NdbEventOperation::EXECUTING;
+ }
+ } else {
+ //Error
+ m_state = NdbEventOperation::ERROR;
+ }
+ return r;
+}
+
+int
+NdbEventOperationImpl::stop()
+{
+ if (m_state != NdbEventOperation::EXECUTING)
+ return -1;
+
+ // ndbout_c("NdbEventOperation::stopping()");
+
+ NdbDictionary::Dictionary *myDict = m_ndb->getDictionary();
+ if (!myDict) {
+ ndbout_c("NdbEventOperation::stop(): getDictionary=NULL");
+ return 0;
+ }
+
+ NdbDictionaryImpl & myDictImpl = NdbDictionaryImpl::getImpl(*myDict);
+
+ int hasSubscriber;
+ int ret =
+ m_bufferHandle->prepareDropSubscribeEvent(m_bufferId,
+ hasSubscriber /* return value */);
+
+ if (ret < 0) {
+ ndbout_c("prepareDropSubscribeEvent failed");
+ return -1;
+ }
+ // m_eventImpl->m_bufferId = m_bufferId;
+
+ int r = -1;
+
+ if (hasSubscriber == 0) { // only excute if there's no other subscribers
+ r = myDictImpl.stopSubscribeEvent(*m_eventImpl);
+#ifdef EVENT_DEBUG
+ ndbout_c("NdbEventOperation::stopping() done");
+#endif
+ } else
+ r = 0;
+
+ if (r) {
+ //Error
+ m_bufferHandle->unprepareDropSubscribeEvent(m_bufferId);
+ m_state = NdbEventOperation::ERROR;
+ } else {
+#ifdef EVENT_DEBUG
+ ndbout_c("NdbEventOperation::dropping()");
+#endif
+ m_bufferHandle->dropSubscribeEvent(m_bufferId);
+ m_state = NdbEventOperation::CREATED;
+ }
+
+
+ return r;
+}
+
+bool
+NdbEventOperationImpl::isConsistent()
+{
+ return sdata->isGCIConsistent();
+}
+
+Uint32
+NdbEventOperationImpl::getGCI()
+{
+ return sdata->gci;
+}
+
+Uint32
+NdbEventOperationImpl::getLatestGCI()
+{
+ return NdbGlobalEventBufferHandle::getLatestGCI();
+}
+
+int
+NdbEventOperationImpl::next(int *pOverrun)
+{
+ int nr = 10000; // a high value
+ int tmpOverrun = 0;
+ int *ptmpOverrun;
+ if (pOverrun) {
+ ptmpOverrun = &tmpOverrun;
+ } else
+ ptmpOverrun = NULL;
+
+ while (nr > 0) {
+ int r=NdbGlobalEventBufferHandle::getDataL(m_bufferId, sdata,
+ ptr, pOverrun);
+ if (pOverrun) {
+ tmpOverrun += *pOverrun;
+ *pOverrun = tmpOverrun;
+ }
+
+ if (r <= 0) return r; // no data
+
+ if (r < nr) r = nr; else nr--; // we don't want to be stuck here forever
+
+#ifdef EVENT_DEBUG
+ ndbout_c("!!!!!!!sdata->operation %u", (Uint32)sdata->operation);
+#endif
+
+ // now move the data into the RecAttrs
+ if ((theFirstRecAttrs[0] == NULL) &&
+ (theFirstRecAttrs[1] == NULL)) return r;
+ // no copying since no RecAttr's
+
+
+ Uint32 *aAttrPtr = ptr[0].p;
+ Uint32 *aAttrEndPtr = aAttrPtr + ptr[0].sz;
+ Uint32 *aDataPtr = ptr[1].p;
+
+#ifdef EVENT_DEBUG
+ printf("after values sz=%u\n", ptr[1].sz);
+ for(int i=0; i < ptr[1].sz; i++)
+ printf ("H'%.8X ",ptr[1].p[i]);
+ printf("\n");
+ printf("before values sz=%u\n", ptr[2].sz);
+ for(int i=0; i < ptr[2].sz; i++)
+ printf ("H'%.8X ",ptr[2].p[i]);
+ printf("\n");
+#endif
+
+ NdbRecAttr *tWorkingRecAttr = theFirstRecAttrs[0];
+
+ // copy data into the RecAttr's
+ // we assume that the respective attribute lists are sorted
+
+ Uint32 tRecAttrId;
+ Uint32 tAttrId;
+ Uint32 tDataSz;
+ int hasSomeData=0;
+ while ((aAttrPtr < aAttrEndPtr) && (tWorkingRecAttr != NULL)) {
+ tRecAttrId = tWorkingRecAttr->attrId();
+ tAttrId = AttributeHeader(*aAttrPtr).getAttributeId();
+ tDataSz = AttributeHeader(*aAttrPtr).getDataSize();
+
+ while (tAttrId > tRecAttrId) {
+ //printf("[%u] %u %u [%u]\n", tAttrId, tDataSz, *aDataPtr, tRecAttrId);
+ tWorkingRecAttr->setUNDEFINED();
+ tWorkingRecAttr = tWorkingRecAttr->next();
+ if (tWorkingRecAttr == NULL)
+ break;
+ tRecAttrId = tWorkingRecAttr->attrId();
+ }
+ if (tWorkingRecAttr == NULL)
+ break;
+
+ //printf("[%u] %u %u [%u]\n", tAttrId, tDataSz, *aDataPtr, tRecAttrId);
+
+ if (tAttrId == tRecAttrId) {
+ tWorkingRecAttr->setNotNULL();
+ if (!m_eventImpl->m_tableImpl->getColumn(tRecAttrId)->getPrimaryKey())
+ hasSomeData++;
+
+ //printf("set!\n");
+
+ Uint32 *theRef = (Uint32*)tWorkingRecAttr->aRef();
+ Uint32 *theEndRef = theRef + tDataSz;
+ while (theRef < theEndRef)
+ *theRef++ = *aDataPtr++;
+
+ // move forward, data has already moved forward
+ aAttrPtr++;
+ tWorkingRecAttr = tWorkingRecAttr->next();
+ } else {
+ // move only attr forward
+ aAttrPtr++;
+ aDataPtr += tDataSz;
+ }
+ }
+
+ while (tWorkingRecAttr != NULL) {
+ tRecAttrId = tWorkingRecAttr->attrId();
+ //printf("set undefined [%u] %u %u [%u]\n", tAttrId, tDataSz, *aDataPtr, tRecAttrId);
+ tWorkingRecAttr->setUNDEFINED();
+ tWorkingRecAttr = tWorkingRecAttr->next();
+ }
+
+ tWorkingRecAttr = theFirstRecAttrs[1];
+ aDataPtr = ptr[2].p;
+ Uint32 *aDataEndPtr = aDataPtr + ptr[2].sz;
+ while ((aDataPtr < aDataEndPtr) && (tWorkingRecAttr != NULL)) {
+ tRecAttrId = tWorkingRecAttr->attrId();
+ tAttrId = AttributeHeader(*aDataPtr).getAttributeId();
+ tDataSz = AttributeHeader(*aDataPtr).getDataSize();
+ aDataPtr++;
+ while (tAttrId > tRecAttrId) {
+ tWorkingRecAttr->setUNDEFINED();
+ tWorkingRecAttr = tWorkingRecAttr->next();
+ if (tWorkingRecAttr == NULL)
+ break;
+ tRecAttrId = tWorkingRecAttr->attrId();
+ }
+ if (tWorkingRecAttr == NULL)
+ break;
+ if (tAttrId == tRecAttrId) {
+ tWorkingRecAttr->setNotNULL();
+
+ if (!m_eventImpl->m_tableImpl->getColumn(tRecAttrId)->getPrimaryKey())
+ hasSomeData++;
+
+ Uint32 *theRef = (Uint32*)tWorkingRecAttr->aRef();
+ Uint32 *theEndRef = theRef + tDataSz;
+ while (theRef < theEndRef)
+ *theRef++ = *aDataPtr++;
+
+ // move forward, data+attr has already moved forward
+ tWorkingRecAttr = tWorkingRecAttr->next();
+ } else {
+ // move only data+attr forward
+ aDataPtr += tDataSz;
+ }
+ }
+ while (tWorkingRecAttr != NULL) {
+ tWorkingRecAttr->setUNDEFINED();
+ tWorkingRecAttr = tWorkingRecAttr->next();
+ }
+
+ if (hasSomeData)
+ return r;
+ }
+ return 0;
+}
+
+NdbDictionary::Event::TableEvent
+NdbEventOperationImpl::getEventType()
+{
+ switch (sdata->operation) {
+ case TriggerEvent::TE_INSERT:
+ return NdbDictionary::Event::TE_INSERT;
+ case TriggerEvent::TE_DELETE:
+ return NdbDictionary::Event::TE_DELETE;
+ case TriggerEvent::TE_UPDATE:
+ return NdbDictionary::Event::TE_UPDATE;
+ default:
+ return NdbDictionary::Event::TE_ALL;
+ }
+}
+
+void
+NdbEventOperationImpl::printRecAttr(NdbRecAttr *p)
+{
+ int size = p->attrSize();
+ int aSize = p->arraySize();
+
+ switch(p->attrType()){
+ case UnSigned:
+ switch(size) {
+ case 8: ndbout << p->u_64_value(); break;
+ case 4: ndbout << p->u_32_value(); break;
+ case 2: ndbout << p->u_short_value(); break;
+ case 1: ndbout << (unsigned) p->u_char_value(); break;
+ default: ndbout << "Unknown size" << endl;
+ }
+ break;
+
+ case Signed:
+ switch(size) {
+ case 8: ndbout << p->int64_value(); break;
+ case 4: ndbout << p->int32_value(); break;
+ case 2: ndbout << p->short_value(); break;
+ case 1: ndbout << (int) p->char_value(); break;
+ default: ndbout << "Unknown size" << endl;
+ }
+ break;
+
+ case String:
+ {
+ char* buf = new char[aSize+1];
+ memcpy(buf, p->aRef(), aSize);
+ buf[aSize] = 0;
+ ndbout << buf;
+ delete [] buf;
+ }
+ break;
+
+ case Float:
+ ndbout << p->float_value();
+ break;
+
+ default:
+ ndbout << "Unknown";
+ break;
+ }
+}
+
+void
+NdbEventOperationImpl::print()
+{
+ ndbout << "EventId " << m_eventId << "\n";
+
+ for (int i = 0; i < 2; i++) {
+ NdbRecAttr *p = theFirstRecAttrs[i];
+ ndbout << " %u " << i;
+ while (p) {
+ ndbout << " : " << p->attrId() << " = ";
+ printRecAttr(p);
+ p = p->next();
+ }
+ ndbout << "\n";
+ }
+}
+
+void
+NdbEventOperationImpl::printAll()
+{
+ Uint32 *aAttrPtr = ptr[0].p;
+ Uint32 *aAttrEndPtr = aAttrPtr + ptr[0].sz;
+ Uint32 *aDataPtr = ptr[1].p;
+
+ //tRecAttr->setup(tAttrInfo, aValue)) {
+
+ Uint32 tAttrId;
+ Uint32 tDataSz;
+ for (; aAttrPtr < aAttrEndPtr; ) {
+ tAttrId = AttributeHeader(*aAttrPtr).getAttributeId();
+ tDataSz = AttributeHeader(*aAttrPtr).getDataSize();
+
+ aAttrPtr++;
+ aDataPtr += tDataSz;
+ }
+}
+
+
+int NdbEventOperationImpl::wait(void *p, int aMillisecondNumber)
+{
+ return ((NdbGlobalEventBufferHandle*)p)->wait(aMillisecondNumber);
+}
+
+/*
+ * Global variable ndbGlobalEventBuffer
+ * Class NdbGlobalEventBufferHandle
+ * Class NdbGlobalEventBuffer
+ *
+ */
+
+#define ADD_DROP_LOCK_GUARDR(TYPE, FN) \
+{ \
+ ndbGlobalEventBuffer->add_drop_lock(); \
+ ndbGlobalEventBuffer->lock(); \
+ TYPE r = ndbGlobalEventBuffer->FN; \
+ ndbGlobalEventBuffer->unlock(); \
+ if (r < 0) { \
+ ndbGlobalEventBuffer->add_drop_unlock(); \
+ } \
+ return r;\
+}
+#define GUARDR(TYPE, FN) \
+{ \
+ ndbGlobalEventBuffer->lock(); \
+ TYPE r = ndbGlobalEventBuffer->FN; \
+ ndbGlobalEventBuffer->unlock(); \
+ return r;\
+}
+#define GUARD(FN) \
+{ \
+ ndbGlobalEventBuffer->lock(); \
+ ndbGlobalEventBuffer->FN; \
+ ndbGlobalEventBuffer->unlock(); \
+}
+#define ADD_DROP_UNLOCK_GUARD(FN) \
+{ \
+ GUARD(FN); \
+ ndbGlobalEventBuffer->add_drop_unlock(); \
+}
+#define GUARDBLOCK(BLOCK) \
+{ \
+ ndbGlobalEventBuffer->lock(); \
+ BLOCK \
+ ndbGlobalEventBuffer->unlock(); \
+}
+
+/*
+ * Global variable ndbGlobalEventBuffer
+ *
+ */
+
+static NdbGlobalEventBuffer *ndbGlobalEventBuffer=NULL;
+#ifdef NDB_WIN32
+static NdbMutex & ndbGlobalEventBufferMutex = * NdbMutex_Create();
+#else
+static NdbMutex ndbGlobalEventBufferMutex = NDB_MUTEX_INITIALIZER;
+#endif
+
+/*
+ * Class NdbGlobalEventBufferHandle
+ * Each Ndb object has a Handle. This Handle is used to access the
+ * global NdbGlobalEventBuffer instance ndbGlobalEventBuffer
+ */
+
+NdbGlobalEventBufferHandle *
+NdbGlobalEventBuffer_init(int n)
+{
+ return new NdbGlobalEventBufferHandle(n);
+ // return NdbGlobalEventBufferHandle::init(n);
+}
+
+void
+NdbGlobalEventBuffer_drop(NdbGlobalEventBufferHandle *h)
+{
+ delete h;
+}
+
+NdbGlobalEventBufferHandle::NdbGlobalEventBufferHandle
+(int MAX_NUMBER_ACTIVE_EVENTS) : m_bufferL(0), m_nids(0)
+{
+ if ((p_cond = NdbCondition_Create()) == NULL) {
+ ndbout_c("NdbGlobalEventBufferHandle: NdbCondition_Create() failed");
+ exit(-1);
+ }
+
+ NdbMutex_Lock(&ndbGlobalEventBufferMutex);
+ if (ndbGlobalEventBuffer == NULL) {
+ if (ndbGlobalEventBuffer == NULL) {
+ ndbGlobalEventBuffer = new NdbGlobalEventBuffer();
+ if (!ndbGlobalEventBuffer) {
+ NdbMutex_Unlock(&ndbGlobalEventBufferMutex);
+ ndbout_c("NdbGlobalEventBufferHandle:: failed to allocate ndbGlobalEventBuffer");
+ exit(-1);
+ }
+ }
+ }
+ NdbMutex_Unlock(&ndbGlobalEventBufferMutex);
+
+ GUARD(real_init(this,MAX_NUMBER_ACTIVE_EVENTS));
+}
+
+NdbGlobalEventBufferHandle::~NdbGlobalEventBufferHandle()
+{
+ NdbCondition_Destroy(p_cond);
+
+ ndbGlobalEventBuffer->lock();
+ ndbGlobalEventBuffer->real_remove(this);
+ ndbGlobalEventBuffer->unlock();
+
+ NdbMutex_Lock(&ndbGlobalEventBufferMutex);
+ if (ndbGlobalEventBuffer->m_handlers.size() == 0) {
+ delete ndbGlobalEventBuffer;
+ ndbGlobalEventBuffer = NULL;
+ }
+ NdbMutex_Unlock(&ndbGlobalEventBufferMutex);
+}
+
+void
+NdbGlobalEventBufferHandle::addBufferId(int bufferId)
+{
+ if (m_nids >= NDB_MAX_ACTIVE_EVENTS) {
+ ndbout_c("NdbGlobalEventBufferHandle::addBufferId error in paramerer setting");
+ exit(-1);
+ }
+ m_bufferIds[m_nids] = bufferId;
+ m_nids++;
+}
+
+void
+NdbGlobalEventBufferHandle::dropBufferId(int bufferId)
+{
+ for (int i = 0; i < m_nids; i++)
+ if (m_bufferIds[i] == bufferId) {
+ m_nids--;
+ for (; i < m_nids; i++)
+ m_bufferIds[i] = m_bufferIds[i+1];
+ return;
+ }
+ ndbout_c("NdbGlobalEventBufferHandle::dropBufferId %d does not exist",
+ bufferId);
+ exit(-1);
+}
+/*
+NdbGlobalEventBufferHandle *
+NdbGlobalEventBufferHandle::init (int MAX_NUMBER_ACTIVE_EVENTS)
+{
+ return new NdbGlobalEventBufferHandle();
+}
+void
+NdbGlobalEventBufferHandle::drop(NdbGlobalEventBufferHandle *handle)
+{
+ delete handle;
+}
+*/
+int
+NdbGlobalEventBufferHandle::prepareAddSubscribeEvent(Uint32 eventId,
+ int& hasSubscriber)
+{
+ ADD_DROP_LOCK_GUARDR(int,real_prepareAddSubscribeEvent(this, eventId, hasSubscriber));
+}
+void
+NdbGlobalEventBufferHandle::addSubscribeEvent
+(int bufferId, NdbEventOperationImpl *ndbEventOperationImpl)
+{
+ ADD_DROP_UNLOCK_GUARD(real_addSubscribeEvent(bufferId, ndbEventOperationImpl));
+}
+void
+NdbGlobalEventBufferHandle::unprepareAddSubscribeEvent(int bufferId)
+{
+ ADD_DROP_UNLOCK_GUARD(real_unprepareAddSubscribeEvent(bufferId));
+}
+
+int
+NdbGlobalEventBufferHandle::prepareDropSubscribeEvent(int bufferId,
+ int& hasSubscriber)
+{
+ ADD_DROP_LOCK_GUARDR(int,real_prepareDropSubscribeEvent(bufferId, hasSubscriber));
+}
+
+void
+NdbGlobalEventBufferHandle::unprepareDropSubscribeEvent(int bufferId)
+{
+ ADD_DROP_UNLOCK_GUARD(real_unprepareDropSubscribeEvent(bufferId));
+}
+
+void
+NdbGlobalEventBufferHandle::dropSubscribeEvent(int bufferId)
+{
+ ADD_DROP_UNLOCK_GUARD(real_dropSubscribeEvent(bufferId));
+}
+
+int
+NdbGlobalEventBufferHandle::insertDataL(int bufferId,
+ const SubTableData * const sdata,
+ LinearSectionPtr ptr[3])
+{
+ GUARDR(int,real_insertDataL(bufferId,sdata,ptr));
+}
+
+void
+NdbGlobalEventBufferHandle::latestGCI(int bufferId, Uint32 gci)
+{
+ GUARD(real_latestGCI(bufferId,gci));
+}
+
+Uint32
+NdbGlobalEventBufferHandle::getLatestGCI()
+{
+ GUARDR(Uint32, real_getLatestGCI());
+}
+
+inline void
+NdbGlobalEventBufferHandle::group_lock()
+{
+ ndbGlobalEventBuffer->group_lock();
+}
+
+inline void
+NdbGlobalEventBufferHandle::group_unlock()
+{
+ ndbGlobalEventBuffer->group_unlock();
+}
+
+int
+NdbGlobalEventBufferHandle::wait(int aMillisecondNumber)
+{
+ GUARDR(int, real_wait(this, aMillisecondNumber));
+}
+
+int NdbGlobalEventBufferHandle::getDataL(const int bufferId,
+ SubTableData * &sdata,
+ LinearSectionPtr ptr[3],
+ int *pOverrun)
+{
+ GUARDR(int,real_getDataL(bufferId,sdata,ptr,pOverrun));
+}
+
+/*
+ * Class NdbGlobalEventBuffer
+ *
+ *
+ */
+
+
+void
+NdbGlobalEventBuffer::lock()
+{
+ if (!m_group_lock_flag)
+ NdbMutex_Lock(&ndbGlobalEventBufferMutex);
+}
+void
+NdbGlobalEventBuffer::unlock()
+{
+ if (!m_group_lock_flag)
+ NdbMutex_Unlock(&ndbGlobalEventBufferMutex);
+}
+void
+NdbGlobalEventBuffer::add_drop_lock()
+{
+ NdbMutex_Lock(p_add_drop_mutex);
+}
+void
+NdbGlobalEventBuffer::add_drop_unlock()
+{
+ NdbMutex_Unlock(p_add_drop_mutex);
+}
+inline void
+NdbGlobalEventBuffer::group_lock()
+{
+ lock();
+ m_group_lock_flag = 1;
+}
+
+inline void
+NdbGlobalEventBuffer::group_unlock()
+{
+ m_group_lock_flag = 0;
+ unlock();
+}
+
+void
+NdbGlobalEventBuffer::lockB(int bufferId)
+{
+ NdbMutex_Lock(m_buf[ID(bufferId)].p_buf_mutex);
+}
+void
+NdbGlobalEventBuffer::unlockB(int bufferId)
+{
+ NdbMutex_Lock(m_buf[ID(bufferId)].p_buf_mutex);
+}
+
+// Private methods
+
+NdbGlobalEventBuffer::NdbGlobalEventBuffer() :
+ m_handlers(),
+ m_group_lock_flag(0),
+ m_latestGCI(0),
+ m_no(0) // must start at ZERO!
+{
+ if ((p_add_drop_mutex = NdbMutex_Create()) == NULL) {
+ ndbout_c("NdbGlobalEventBuffer: NdbMutex_Create() failed");
+ exit(-1);
+ }
+}
+
+NdbGlobalEventBuffer::~NdbGlobalEventBuffer()
+{
+ NdbMutex_Destroy(p_add_drop_mutex);
+ // NdbMem_Deallocate(m_eventBufferIdToEventId);
+}
+void
+NdbGlobalEventBuffer::real_init (NdbGlobalEventBufferHandle *h,
+ int MAX_NUMBER_ACTIVE_EVENTS)
+{
+ if (m_handlers.size() == 0) { // First init
+ m_max = MAX_NUMBER_ACTIVE_EVENTS;
+ m_buf = new BufItem[m_max];
+ // (BufItem *)NdbMem_Allocate(m_max*sizeof(BufItem));
+
+ for (int i=0; i<m_max; i++) {
+ m_buf[i].gId = 0;
+ }
+ }
+ // TODO make sure we don't hit roof
+ // m_handlers[m_nhandlers] = h;
+ m_handlers.push_back(h);
+ // ndbout_c("NdbGlobalEventBuffer::real_init(), m_handles=%u %u", m_nhandlers, h);
+}
+void
+NdbGlobalEventBuffer::real_remove(NdbGlobalEventBufferHandle *h)
+{
+ // ndbout_c("NdbGlobalEventBuffer::real_init_remove(), m_handles=%u %u", m_nhandlers, h);
+ for (Uint32 i=0 ; i < m_handlers.size(); i++) {
+ // ndbout_c("%u %u %u", i, m_handlers[i], h);
+ if (m_handlers[i] == h) {
+ m_handlers.erase(i);
+ if (m_handlers.size() == 0) {
+ // ndbout_c("last to go");
+ delete[] m_buf;
+ m_buf = NULL;
+ // NdbMem_Free((char*)m_buf);
+ }
+ return;
+ }
+ }
+ ndbout_c("NdbGlobalEventBuffer::real_init_remove() non-existing handle");
+ exit(-1);
+}
+
+int
+NdbGlobalEventBuffer::real_prepareAddSubscribeEvent
+(NdbGlobalEventBufferHandle *aHandle, Uint32 eventId, int& hasSubscriber)
+{
+ int i;
+ int bufferId = -1;
+
+ // add_drop_lock(); // only one thread can do add or drop at a time
+
+ // Find place where eventId already set
+ for (i=0; i<m_no; i++) {
+ if (m_buf[i].gId == eventId) {
+ bufferId = i;
+ break;
+ }
+ }
+ if (bufferId < 0) {
+ // find space for new bufferId
+ for (i=0; i<m_no; i++) {
+ if (m_buf[i].gId == 0) {
+ bufferId = i; // we found an empty spot
+ break;
+ }
+ }
+ if (bufferId < 0 &&
+ m_no < m_max) {
+ // room for more so get that
+ bufferId=m_no;
+ m_buf[m_no].gId = 0;
+ m_no++;
+ } else {
+ ndbout_c("prepareAddSubscribeEvent: Can't accept more subscribers");
+ // add_drop_unlock();
+ return -1;
+ }
+ }
+
+ BufItem &b = m_buf[ID(bufferId)];
+
+ if (b.gId == 0) { // first subscriber needs some initialization
+
+ bufferId = NO_ID(0, bufferId);
+
+ b.gId = eventId;
+
+ if ((b.p_buf_mutex = NdbMutex_Create()) == NULL) {
+ ndbout_c("NdbGlobalEventBuffer: NdbMutex_Create() failed");
+ exit(-1);
+ }
+
+ b.subs = 0;
+ b.f = 0;
+ b.sz = 0;
+ b.max_sz = aHandle->m_bufferL;
+ b.data =
+ (BufItem::Data *)NdbMem_Allocate(b.max_sz*sizeof(BufItem::Data));
+ for (int i = 0; i < b.max_sz; i++) {
+ b.data[i].sdata = NULL;
+ b.data[i].ptr[0].p = NULL;
+ b.data[i].ptr[1].p = NULL;
+ b.data[i].ptr[2].p = NULL;
+ }
+ } else {
+#ifdef EVENT_DEBUG
+ ndbout_c("NdbGlobalEventBuffer::prepareAddSubscribeEvent: TRYING handle one subscriber per event b.subs = %u", b.subs);
+#endif
+
+ int ni = -1;
+ for(int i=0; i < b.subs;i++) {
+ if (b.ps[i].theHandle == NULL) {
+ ni = i;
+ break;
+ }
+ }
+ if (ni < 0) {
+ if (b.subs < MAX_SUBSCRIBERS_PER_EVENT) {
+ ni = b.subs;
+ } else {
+ ndbout_c("prepareAddSubscribeEvent: Can't accept more subscribers");
+ // add_drop_unlock();
+ return -1;
+ }
+ }
+ bufferId = NO_ID(ni, bufferId);
+ }
+
+ // initialize BufItem::Ps
+ {
+ int n = NO(bufferId);
+ NdbGlobalEventBuffer::BufItem::Ps &e = b.ps[n];
+ e.theHandle = aHandle;
+ e.b=0;
+ e.bufferempty = 1;
+ e.overrun=0; // set to -1 to handle first insert
+ }
+
+ if (b.subs > 0)
+ hasSubscriber = 1;
+ else
+ hasSubscriber = 0;
+
+#ifdef EVENT_DEBUG
+ ndbout_c("prepareAddSubscribeEvent: handed out bufferId %d for eventId %d",
+ bufferId, eventId);
+#endif
+
+ /* we now have a lock on the prepare so that no one can mess with this
+ * unlock comes in unprepareAddSubscribeEvent or addSubscribeEvent
+ */
+ return bufferId;
+}
+
+void
+NdbGlobalEventBuffer::real_unprepareAddSubscribeEvent(int bufferId)
+{
+ BufItem &b = m_buf[ID(bufferId)];
+ int n = NO(bufferId);
+
+ b.ps[n].theHandle = NULL;
+
+ // remove subscribers from the end,
+ // we have to keep gaps since the position
+ // has been handed out in bufferId
+ for (int i = b.subs-1; i >= 0; i--)
+ if (b.ps[i].theHandle == NULL)
+ b.subs--;
+ else
+ break;
+
+ if (b.subs == 0) {
+#ifdef EVENT_DEBUG
+ ndbout_c("unprepareAddSubscribeEvent: no more subscribers left on eventId %d", b.gId);
+#endif
+ b.gId = 0; // We don't have any subscribers, reuse BufItem
+ if (b.data) {
+ NdbMem_Free((void *)b.data);
+ b.data = NULL;
+ }
+ if (b.p_buf_mutex) {
+ NdbMutex_Destroy(b.p_buf_mutex);
+ b.p_buf_mutex = NULL;
+ }
+ }
+ // add_drop_unlock();
+}
+
+void
+NdbGlobalEventBuffer::real_addSubscribeEvent(int bufferId,
+ void *ndbEventOperation)
+{
+ BufItem &b = m_buf[ID(bufferId)];
+ int n = NO(bufferId);
+
+ b.subs++;
+ b.ps[n].theHandle->addBufferId(bufferId);
+
+ // add_drop_unlock();
+#ifdef EVENT_DEBUG
+ ndbout_c("addSubscribeEvent:: added bufferId %d", bufferId);
+#endif
+}
+
+void
+NdbGlobalEventBuffer::real_unprepareDropSubscribeEvent(int bufferId)
+{
+ // add_drop_unlock(); // only one thread can do add or drop at a time
+}
+
+int
+NdbGlobalEventBuffer::real_prepareDropSubscribeEvent(int bufferId,
+ int& hasSubscriber)
+{
+ // add_drop_lock(); // only one thread can do add or drop at a time
+
+ BufItem &b = m_buf[ID(bufferId)];
+
+ int n = 0;
+ for(int i=0; i < b.subs;i++) {
+ if (b.ps[i].theHandle != NULL)
+ n++;
+ }
+
+ if (n > 1)
+ hasSubscriber = 1;
+ else if (n == 1)
+ hasSubscriber = 0;
+ else
+ return -1;
+
+ return 0;
+}
+
+void
+NdbGlobalEventBuffer::real_dropSubscribeEvent(int bufferId)
+{
+ // add_drop_lock(); // only one thread can do add-drop at a time
+
+ BufItem &b = m_buf[ID(bufferId)];
+ int n = NO(bufferId);
+
+ b.ps[n].overrun=0;
+ b.ps[n].bufferempty=1;
+ b.ps[n].b=0;
+ b.ps[n].theHandle->dropBufferId(bufferId);
+
+ real_unprepareAddSubscribeEvent(bufferId); // does add_drop_unlock();
+
+#ifdef EVENT_DEBUG
+ ndbout_c("dropSubscribeEvent:: dropped bufferId %d", bufferId);
+#endif
+}
+
+void
+NdbGlobalEventBuffer::real_latestGCI(int bufferId, Uint32 gci)
+{
+ if (gci > m_latestGCI)
+ m_latestGCI = gci;
+ else if ((m_latestGCI-gci) > 0xffff) // If NDB stays up :-)
+ m_latestGCI = gci;
+}
+
+Uint32
+NdbGlobalEventBuffer::real_getLatestGCI()
+{
+ return m_latestGCI;
+}
+
+int
+NdbGlobalEventBuffer::real_insertDataL(int bufferId,
+ const SubTableData * const sdata,
+ LinearSectionPtr ptr[3])
+{
+ BufItem &b = m_buf[ID(bufferId)];
+#ifdef EVENT_DEBUG
+ int n = NO(bufferId);
+#endif
+ {
+ if (b.subs) {
+#ifdef EVENT_DEBUG
+ ndbout_c("data insertion in buffer %d with eventId %d", bufferId, b.gId);
+#endif
+ // move front forward
+ if (copy_data_alloc(sdata, ptr,
+ b.data[b.f].sdata, b.data[b.f].ptr))
+ return -1;
+ for (int i=0; i < b.subs; i++) {
+ NdbGlobalEventBuffer::BufItem::Ps &e = b.ps[i];
+ if (e.theHandle) { // active subscriber
+ if (b.f == e.b) { // next-to-read == written
+ if (e.bufferempty == 0) {
+ e.overrun++; // another item has been overwritten
+ e.b++; // move next-to-read next since old item was overwritten
+ if (e.b == b.max_sz) e.b = 0; // start from beginning
+ }
+ }
+ e.bufferempty = 0;
+ // signal subscriber that there's more to get
+ NdbCondition_Signal(e.theHandle->p_cond);
+ }
+ }
+ b.f++; // move next-to-write
+ if (b.f == b.max_sz) b.f = 0; // start from beginning
+#ifdef EVENT_DEBUG
+ ndbout_c("Front= %d Back = %d overun = %d", b.f,
+ b.ps[n].b, b.ps[n].overrun);
+#endif
+ } else {
+#ifdef EVENT_DEBUG
+ ndbout_c("Data arrived before ready eventId", b.gId);
+#endif
+ }
+ }
+ return 0;
+}
+
+int NdbGlobalEventBuffer::hasData(int bufferId) {
+ BufItem &b = m_buf[ID(bufferId)];
+ int n = NO(bufferId);
+ NdbGlobalEventBuffer::BufItem::Ps &e = b.ps[n];
+
+ if(e.bufferempty)
+ return 0;
+
+ if (b.f <= e.b)
+ return b.max_sz-e.b + b.f;
+ else
+ return b.f-e.b;
+}
+
+int NdbGlobalEventBuffer::real_getDataL(const int bufferId,
+ SubTableData * &sdata,
+ LinearSectionPtr ptr[3],
+ int *pOverrun)
+{
+ BufItem &b = m_buf[ID(bufferId)];
+ int n = NO(bufferId);
+ NdbGlobalEventBuffer::BufItem::Ps &e = b.ps[n];
+
+ if (pOverrun) {
+ *pOverrun = e.overrun;
+ e.overrun = 0; // if pOverrun is returned to user reset e.overrun
+ }
+
+ if (e.bufferempty)
+ return 0; // nothing to get
+
+ if (copy_data_alloc(b.data[e.b].sdata, b.data[e.b].ptr,
+ sdata, ptr))
+ return -1;
+
+ e.b++; if (e.b == b.max_sz) e.b = 0; // move next-to-read forward
+
+ if (b.f == e.b) // back has cought up with front
+ e.bufferempty = 1;
+
+#ifdef EVENT_DEBUG
+ ndbout_c("getting data from buffer %d with eventId %d", bufferId, b.gId);
+#endif
+
+ return hasData(bufferId)+1;
+}
+int
+NdbGlobalEventBuffer::copy_data_alloc(const SubTableData * const f_sdata,
+ LinearSectionPtr f_ptr[3],
+ SubTableData * &t_sdata,
+ LinearSectionPtr t_ptr[3])
+{
+ if (t_sdata == NULL) {
+ t_sdata = (SubTableData *)NdbMem_Allocate(sizeof(SubTableData));
+ }
+ memcpy(t_sdata,f_sdata,sizeof(SubTableData));
+ for (int i = 0; i < 3; i++) {
+ LinearSectionPtr & f_p = f_ptr[i];
+ LinearSectionPtr & t_p = t_ptr[i];
+ if (f_p.sz > 0) {
+ if (t_p.p == NULL) {
+ t_p.p = (Uint32 *)NdbMem_Allocate(sizeof(Uint32)*f_p.sz);
+ } else if (t_p.sz != f_p.sz) {
+ NdbMem_Free(t_p.p);
+ t_p.p = (Uint32 *)NdbMem_Allocate(sizeof(Uint32)*f_p.sz);
+ }
+ memcpy(t_p.p, f_p.p, sizeof(Uint32)*f_p.sz);
+ } else if (t_p.p != NULL) {
+ NdbMem_Free(t_p.p);
+ t_p.p = NULL;
+ }
+ t_p.sz = f_p.sz;
+ }
+ return 0;
+}
+int
+NdbGlobalEventBuffer::real_wait(NdbGlobalEventBufferHandle *h,
+ int aMillisecondNumber)
+{
+ // check if there are anything in any of the buffers
+ int n = 0;
+ for (int i = 0; i < h->m_nids; i++)
+ n += hasData(h->m_bufferIds[i]);
+ if (n) return n;
+
+ int r = NdbCondition_WaitTimeout(h->p_cond, &ndbGlobalEventBufferMutex, aMillisecondNumber);
+ if (r > 0)
+ return -1;
+
+ n = 0;
+ for (int i = 0; i < h->m_nids; i++)
+ n += hasData(h->m_bufferIds[i]);
+ return n;
+}
diff --git a/ndb/src/ndbapi/NdbEventOperationImpl.hpp b/ndb/src/ndbapi/NdbEventOperationImpl.hpp
new file mode 100644
index 00000000000..b7dee084a9f
--- /dev/null
+++ b/ndb/src/ndbapi/NdbEventOperationImpl.hpp
@@ -0,0 +1,206 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbEventOperationImpl.hpp
+ * Include:
+ * Link:
+ * Author: Tomas Ulin MySQL AB
+ * Date: 2003-11-21
+ * Version: 0.1
+ * Description: Event support
+ * Documentation:
+ * Adjust: 2003-11-21 Tomas Ulin First version.
+ ****************************************************************************/
+
+#ifndef NdbEventOperationImpl_H
+#define NdbEventOperationImpl_H
+
+class NdbGlobalEventBufferHandle;
+class NdbEventOperationImpl : public NdbEventOperation {
+public:
+ NdbEventOperationImpl(NdbEventOperation &N,
+ Ndb *theNdb,
+ const char* eventName,
+ const int bufferLength);
+ ~NdbEventOperationImpl();
+
+ NdbEventOperation::State getState();
+
+ int execute();
+ int stop();
+ NdbRecAttr *getValue(const char *colName, char *aValue, int n);
+ NdbRecAttr *getValue(const NdbColumnImpl *, char *aValue, int n);
+ static int wait(void *p, int aMillisecondNumber);
+ int next(int *pOverrun);
+ bool isConsistent();
+ Uint32 getGCI();
+ Uint32 getLatestGCI();
+
+ NdbDictionary::Event::TableEvent getEventType();
+
+ /*
+ getOperation();
+ getGCI();
+ getLogType();
+ */
+
+ void print();
+ void printAll();
+ void printRecAttr(NdbRecAttr *);
+
+ Ndb *m_ndb;
+ NdbEventImpl *m_eventImpl;
+ NdbGlobalEventBufferHandle *m_bufferHandle;
+
+ NdbRecAttr *theFirstRecAttrs[2];
+ NdbRecAttr *theCurrentRecAttrs[2];
+
+ NdbEventOperation::State m_state;
+ Uint32 m_eventId;
+ int m_bufferId;
+ int m_bufferL;
+ SubTableData *sdata;
+ LinearSectionPtr ptr[3];
+};
+
+class NdbGlobalEventBuffer;
+class NdbGlobalEventBufferHandle {
+public:
+ NdbGlobalEventBufferHandle (int MAX_NUMBER_ACTIVE_EVENTS);
+ ~NdbGlobalEventBufferHandle ();
+ //static NdbGlobalEventBufferHandle *init(int MAX_NUMBER_ACTIVE_EVENTS);
+
+ // returns bufferId 0-N if ok otherwise -1
+ int prepareAddSubscribeEvent(Uint32 eventId, int& hasSubscriber);
+ void unprepareAddSubscribeEvent(int bufferId);
+ void addSubscribeEvent(int bufferId,
+ NdbEventOperationImpl *ndbEventOperationImpl);
+
+ void unprepareDropSubscribeEvent(int bufferId);
+ int prepareDropSubscribeEvent(int bufferId, int& hasSubscriber);
+ void dropSubscribeEvent(int bufferId);
+
+ static int getDataL(const int bufferId,
+ SubTableData * &sdata,
+ LinearSectionPtr ptr[3],
+ int *pOverrun);
+ static int insertDataL(int bufferId,
+ const SubTableData * const sdata,
+ LinearSectionPtr ptr[3]);
+ static void latestGCI(int bufferId, Uint32 gci);
+ static Uint32 getLatestGCI();
+ static Uint32 getEventId(int bufferId);
+
+ void group_lock();
+ void group_unlock();
+ int wait(int aMillisecondNumber);
+ int m_bufferL;
+private:
+ friend class NdbGlobalEventBuffer;
+ void addBufferId(int bufferId);
+ void dropBufferId(int bufferId);
+
+ struct NdbCondition *p_cond;
+ int m_nids;
+ int m_bufferIds[NDB_MAX_ACTIVE_EVENTS];
+};
+
+class NdbGlobalEventBuffer {
+private:
+ friend class NdbGlobalEventBufferHandle;
+ void lockB(int bufferId);
+ void unlockB(int bufferId);
+ void group_lock();
+ void group_unlock();
+ void lock();
+ void unlock();
+ void add_drop_lock();
+ void add_drop_unlock();
+
+ NdbGlobalEventBuffer();
+ ~NdbGlobalEventBuffer();
+
+ void real_remove(NdbGlobalEventBufferHandle *h);
+ void real_init(NdbGlobalEventBufferHandle *h,
+ int MAX_NUMBER_ACTIVE_EVENTS);
+
+ int real_prepareAddSubscribeEvent(NdbGlobalEventBufferHandle *h,
+ Uint32 eventId, int& hasSubscriber);
+ void real_unprepareAddSubscribeEvent(int bufferId);
+ void real_addSubscribeEvent(int bufferId, void *ndbEventOperation);
+
+ void real_unprepareDropSubscribeEvent(int bufferId);
+ int real_prepareDropSubscribeEvent(int bufferId,
+ int& hasSubscriber);
+ void real_dropSubscribeEvent(int bufferId);
+
+ int real_getDataL(const int bufferId,
+ SubTableData * &sdata,
+ LinearSectionPtr ptr[3],
+ int *pOverrun);
+ int real_insertDataL(int bufferId,
+ const SubTableData * const sdata,
+ LinearSectionPtr ptr[3]);
+ void real_latestGCI(int bufferId, Uint32 gci);
+ Uint32 real_getLatestGCI();
+ int copy_data_alloc(const SubTableData * const f_sdata,
+ LinearSectionPtr f_ptr[3],
+ SubTableData * &t_sdata,
+ LinearSectionPtr t_ptr[3]);
+
+ int real_wait(NdbGlobalEventBufferHandle *, int aMillisecondNumber);
+ int hasData(int bufferId);
+ int ID (int bufferId) {return bufferId & 0xFF;};
+ int NO (int bufferId) {return bufferId >> 16;};
+ int NO_ID (int n, int bufferId) {return (n << 16) | ID(bufferId);};
+
+ Vector<NdbGlobalEventBufferHandle*> m_handlers;
+
+ // Global Mutex used for some things
+ NdbMutex *p_add_drop_mutex;
+
+ int m_group_lock_flag;
+ Uint32 m_latestGCI;
+
+ int m_no;
+ int m_max;
+#define MAX_SUBSCRIBERS_PER_EVENT 16
+ struct BufItem {
+ // local mutex for each event/buffer
+ NdbMutex *p_buf_mutex;
+ Uint32 gId;
+ struct Data {
+ SubTableData *sdata;
+ LinearSectionPtr ptr[3];
+ } * data;
+
+ struct Ps {
+ NdbGlobalEventBufferHandle *theHandle;
+ int b;
+ int overrun;
+ int bufferempty;
+ //void *ndbEventOperation;
+ } ps[MAX_SUBSCRIBERS_PER_EVENT]; // only supports 1 subscriber so far
+
+ int subs;
+ int f;
+ int sz;
+ int max_sz;
+ };
+ BufItem *m_buf;
+};
+#endif
diff --git a/ndb/src/ndbapi/NdbImpl.hpp b/ndb/src/ndbapi/NdbImpl.hpp
new file mode 100644
index 00000000000..cd05335b337
--- /dev/null
+++ b/ndb/src/ndbapi/NdbImpl.hpp
@@ -0,0 +1,173 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_IMPL_HPP
+#define NDB_IMPL_HPP
+
+#include <Vector.hpp>
+#include "ObjectMap.hpp"
+
+/**
+ * Private parts of the Ndb object (corresponding to Ndb.hpp in public API)
+ */
+class NdbImpl {
+public:
+ Vector<class NdbTableImpl *> m_invalidTables;
+
+ void checkErrorCode(Uint32 i);
+ void checkInvalidTable(class NdbDictionaryImpl * dict);
+};
+
+#include <Ndb.hpp>
+#include <NdbError.hpp>
+#include <NdbCondition.h>
+#include <NdbReceiver.hpp>
+
+#include <NdbTick.h>
+
+#ifdef VM_TRACE
+#define TRACE_DEBUG(x) ndbout << x << endl;
+#else
+#define TRACE_DEBUG(x)
+#endif
+
+#define CHECK_STATUS_MACRO \
+ {if (checkInitState() == -1) { theError.code = 4100; return -1;}}
+#define CHECK_STATUS_MACRO_VOID \
+ {if (checkInitState() == -1) { theError.code = 4100; return;}}
+#define CHECK_STATUS_MACRO_ZERO \
+ {if (checkInitState() == -1) { theError.code = 4100; return 0;}}
+#define CHECK_STATUS_MACRO_NULL \
+ {if (checkInitState() == -1) { theError.code = 4100; return NULL;}}
+
+inline
+void *
+Ndb::int2void(Uint32 val){
+ return theNdbObjectIdMap->getObject(val);
+}
+
+inline
+NdbReceiver *
+Ndb::void2rec(void* val){
+ return (NdbReceiver*)val;
+}
+
+inline
+NdbConnection *
+Ndb::void2con(void* val){
+ return (NdbConnection*)val;
+}
+
+inline
+NdbOperation*
+Ndb::void2rec_op(void* val){
+ return (NdbOperation*)(void2rec(val)->getOwner());
+}
+
+inline
+NdbIndexOperation*
+Ndb::void2rec_iop(void* val){
+ return (NdbIndexOperation*)(void2rec(val)->getOwner());
+}
+
+inline
+NdbScanReceiver*
+Ndb::void2rec_srec(void* val){
+ return (NdbScanReceiver*)(void2rec(val)->getOwner());
+}
+
+inline
+int
+Ndb::checkInitState()
+{
+ theError.code = 0;
+
+ if (theInitState != Initialised)
+ return -1;
+ return 0;
+}
+
+Uint32 convertEndian(Uint32 Data);
+
+enum WaitSignalType {
+ NO_WAIT = 0,
+ WAIT_NODE_FAILURE = 1, // Node failure during wait
+ WAIT_TIMEOUT = 2, // Timeout during wait
+
+ WAIT_TC_SEIZE = 3,
+ WAIT_TC_RELEASE = 4,
+ WAIT_NDB_TAMPER = 5,
+ WAIT_SCAN = 6,
+
+ // DICT stuff
+ WAIT_GET_TAB_INFO_REQ = 11,
+ WAIT_CREATE_TAB_REQ = 12,
+ WAIT_DROP_TAB_REQ = 13,
+ WAIT_ALTER_TAB_REQ = 14,
+ WAIT_CREATE_INDX_REQ = 15,
+ WAIT_DROP_INDX_REQ = 16,
+ WAIT_LIST_TABLES_CONF = 17
+};
+
+enum LockMode {
+ Read,
+ Update,
+ Insert,
+ Delete
+};
+
+#include <NdbOut.hpp>
+
+inline
+void
+NdbWaiter::wait(int waitTime)
+{
+ const bool forever = (waitTime == -1);
+ const NDB_TICKS maxTime = NdbTick_CurrentMillisecond() + waitTime;
+ while (1) {
+ if (m_state == NO_WAIT || m_state == WAIT_NODE_FAILURE)
+ break;
+ if (forever) {
+ NdbCondition_Wait(m_condition, (NdbMutex*)m_mutex);
+ } else {
+ if (waitTime <= 0) {
+ m_state = WAIT_TIMEOUT;
+ break;
+ }
+ NdbCondition_WaitTimeout(m_condition, (NdbMutex*)m_mutex, waitTime);
+ waitTime = maxTime - NdbTick_CurrentMillisecond();
+ }
+ }
+ NdbMutex_Unlock((NdbMutex*)m_mutex);
+}
+
+inline
+void
+NdbWaiter::nodeFail(Uint32 aNodeId){
+ if (m_state != NO_WAIT && m_node == aNodeId){
+ m_state = WAIT_NODE_FAILURE;
+ NdbCondition_Signal(m_condition);
+ }
+}
+
+inline
+void
+NdbWaiter::signal(Uint32 state){
+ m_state = state;
+ NdbCondition_Signal(m_condition);
+}
+
+#endif
diff --git a/ndb/src/ndbapi/NdbIndexOperation.cpp b/ndb/src/ndbapi/NdbIndexOperation.cpp
new file mode 100644
index 00000000000..ee5491d72a8
--- /dev/null
+++ b/ndb/src/ndbapi/NdbIndexOperation.cpp
@@ -0,0 +1,719 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbIndexOperation.cpp
+ * Include:
+ * Link:
+ * Author: UABMASD Martin Sköld INN/V Alzato
+ * Date: 2002-04-01
+ * Version: 0.1
+ * Description: Secondary index support
+ * Documentation:
+ * Adjust: 2002-04-01 UABMASD First version.
+ ****************************************************************************/
+
+#include <NdbIndexOperation.hpp>
+#include <NdbResultSet.hpp>
+#include <Ndb.hpp>
+#include <NdbConnection.hpp>
+#include "NdbApiSignal.hpp"
+#include <AttributeHeader.hpp>
+#include <signaldata/TcIndx.hpp>
+#include <signaldata/TcKeyReq.hpp>
+#include <signaldata/IndxKeyInfo.hpp>
+#include <signaldata/IndxAttrInfo.hpp>
+
+#define CHECK_NULL(v) assert(v == NULL); v = NULL;
+
+NdbIndexOperation::NdbIndexOperation(Ndb* aNdb) :
+ NdbOperation(aNdb),
+ m_theIndex(NULL),
+ m_theIndexLen(0),
+ m_theNoOfIndexDefined(0)
+{
+ m_tcReqGSN = GSN_TCINDXREQ;
+ m_attrInfoGSN = GSN_INDXATTRINFO;
+ m_keyInfoGSN = GSN_INDXKEYINFO;
+
+ /**
+ * Change receiver type
+ */
+ theReceiver.init(NdbReceiver::NDB_INDEX_OPERATION, this);
+}
+
+NdbIndexOperation::~NdbIndexOperation()
+{
+}
+
+/*****************************************************************************
+ * int indxInit();
+ *
+ * Return Value: Return 0 : init was successful.
+ * Return -1: In all other case.
+ * Remark: Initiates operation record after allocation.
+ *****************************************************************************/
+int
+NdbIndexOperation::indxInit(NdbIndexImpl * anIndex,
+ NdbTableImpl * aTable,
+ NdbConnection* myConnection)
+{
+ NdbOperation::init(aTable, myConnection);
+
+ switch (anIndex->m_type) {
+ case(NdbDictionary::Index::UniqueHashIndex):
+ break;
+ case(NdbDictionary::Index::Undefined):
+ case(NdbDictionary::Index::HashIndex):
+ case(NdbDictionary::Index::UniqueOrderedIndex):
+ case(NdbDictionary::Index::OrderedIndex):
+ setErrorCodeAbort(4003);
+ return -1;
+ }
+ m_theIndex = anIndex;
+ m_accessTable = anIndex->m_table;
+ m_theIndexLen = 0;
+ m_theNoOfIndexDefined = 0;
+ for (Uint32 i=0; i<MAXNROFTUPLEKEY; i++)
+ for (int j=0; j<3; j++)
+ m_theIndexDefined[i][j] = false;
+
+ TcIndxReq * const tcIndxReq = CAST_PTR(TcIndxReq, theTCREQ->getDataPtrSend());
+ tcIndxReq->scanInfo = 0;
+ theKEYINFOptr = &tcIndxReq->keyInfo[0];
+ theATTRINFOptr = &tcIndxReq->attrInfo[0];
+ return 0;
+}
+
+int NdbIndexOperation::readTuple()
+{
+ // First check that index is unique
+
+ return NdbOperation::readTuple();
+}
+
+int NdbIndexOperation::readTupleExclusive()
+{
+ // First check that index is unique
+
+ return NdbOperation::readTupleExclusive();
+}
+
+int NdbIndexOperation::simpleRead()
+{
+ // First check that index is unique
+
+ return NdbOperation::simpleRead();
+}
+
+int NdbIndexOperation::dirtyRead()
+{
+ // First check that index is unique
+
+ return NdbOperation::dirtyRead();
+}
+
+int NdbIndexOperation::committedRead()
+{
+ // First check that index is unique
+
+ return NdbOperation::committedRead();
+}
+
+int NdbIndexOperation::updateTuple()
+{
+ // First check that index is unique
+
+ return NdbOperation::updateTuple();
+}
+
+int NdbIndexOperation::deleteTuple()
+{
+ // First check that index is unique
+
+ return NdbOperation::deleteTuple();
+}
+
+int NdbIndexOperation::dirtyUpdate()
+{
+ // First check that index is unique
+
+ return NdbOperation::dirtyUpdate();
+}
+
+int NdbIndexOperation::interpretedUpdateTuple()
+{
+ // First check that index is unique
+
+ return NdbOperation::interpretedUpdateTuple();
+}
+
+int NdbIndexOperation::interpretedDeleteTuple()
+{
+ // First check that index is unique
+
+ return NdbOperation::interpretedDeleteTuple();
+}
+
+int NdbIndexOperation::equal_impl(const NdbColumnImpl* tAttrInfo,
+ const char* aValuePassed,
+ Uint32 aVariableKeyLen)
+{
+ register Uint32 tAttrId;
+
+ Uint32 tData;
+ Uint32 tKeyInfoPosition;
+ const char* aValue = aValuePassed;
+ Uint32 tempData[1024];
+
+ if ((theStatus == OperationDefined) &&
+ (aValue != NULL) &&
+ (tAttrInfo != NULL )) {
+ /************************************************************************
+ * Start by checking that the attribute is an index key.
+ * This value is also the word order in the tuple key of this
+ * tuple key attribute.
+ * Then check that this tuple key has not already been defined.
+ * Finally check if all tuple key attributes have been defined. If
+ * this is true then set Operation state to tuple key defined.
+ ************************************************************************/
+ tAttrId = tAttrInfo->m_attrId;
+ tKeyInfoPosition = tAttrInfo->m_keyInfoPos;
+ Uint32 i = 0;
+
+ // Check that the attribute is part if the index attributes
+ // by checking if it is a primary key attribute of index table
+ if (tAttrInfo->m_pk) {
+ Uint32 tKeyDefined = theTupleKeyDefined[0][2];
+ Uint32 tKeyAttrId = theTupleKeyDefined[0][0];
+ do {
+ if (tKeyDefined == false) {
+ goto keyEntryFound;
+ } else {
+ if (tKeyAttrId != tAttrId) {
+ /******************************************************************
+ * We read the key defined variable in advance.
+ * It could potentially read outside its area when
+ * i = MAXNROFTUPLEKEY - 1,
+ * it is not a problem as long as the variable
+ * theTupleKeyDefined is defined
+ * in the middle of the object.
+ * Reading wrong data and not using it causes no problems.
+ *****************************************************************/
+ i++;
+ tKeyAttrId = theTupleKeyDefined[i][0];
+ tKeyDefined = theTupleKeyDefined[i][2];
+ continue;
+ } else {
+ goto equal_error2;
+ }//if
+ }//if
+ } while (i < MAXNROFTUPLEKEY);
+ goto equal_error2;
+ } else {
+ goto equal_error1;
+ }
+ /**************************************************************************
+ * Now it is time to retrieve the tuple key data from the pointer supplied
+ * by the application.
+ * We have to retrieve the size of the attribute in words and bits.
+ *************************************************************************/
+ keyEntryFound:
+ m_theIndexDefined[i][0] = tAttrId;
+ m_theIndexDefined[i][1] = tKeyInfoPosition;
+ m_theIndexDefined[i][2] = true;
+
+ Uint32 sizeInBytes = tAttrInfo->m_attrSize * tAttrInfo->m_arraySize;
+ Uint32 bitsInLastWord = 8 * (sizeInBytes & 3) ;
+ Uint32 totalSizeInWords = (sizeInBytes + 3)/4;// Inc. bits in last word
+ Uint32 sizeInWords = sizeInBytes / 4; // Exc. bits in last word
+
+ if (true){ //tArraySize != 0) {
+ Uint32 tIndexLen = m_theIndexLen;
+
+ m_theIndexLen = tIndexLen + totalSizeInWords;
+ if ((aVariableKeyLen == sizeInBytes) ||
+ (aVariableKeyLen == 0)) {
+ ;
+ } else {
+ goto equal_error3;
+ }
+ }
+#if 0
+ else {
+ /************************************************************************
+ * The attribute is a variable array. We need to use the length parameter
+ * to know the size of this attribute in the key information and
+ * variable area. A key is however not allowed to be larger than 4
+ * kBytes and this is checked for variable array attributes
+ * used as keys.
+ ***********************************************************************/
+ Uint32 tMaxVariableKeyLenInWord = (MAXTUPLEKEYLENOFATTERIBUTEINWORD -
+ tKeyInfoPosition);
+ tAttrSizeInBits = aVariableKeyLen << 3;
+ tAttrSizeInWords = tAttrSizeInBits >> 5;
+ tAttrBitsInLastWord = tAttrSizeInBits - (tAttrSizeInWords << 5);
+ tAttrLenInWords = ((tAttrSizeInBits + 31) >> 5);
+ if (tAttrLenInWords > tMaxVariableKeyLenInWord) {
+ setErrorCodeAbort(4207);
+ return -1;
+ }//if
+ m_theIndexLen = m_theIndexLen + tAttrLenInWords;
+ }//if
+#endif
+
+ /*************************************************************************
+ * Check if the pointer of the value passed is aligned on a 4 byte
+ * boundary. If so only assign the pointer to the internal variable
+ * aValue. If it is not aligned then we start by copying the value to
+ * tempData and use this as aValue instead.
+ *************************************************************************/
+ const int attributeSize = sizeInBytes;
+ const int slack = sizeInBytes & 3;
+ int tDistrKey = tAttrInfo->m_distributionKey;
+ int tDistrGroup = tAttrInfo->m_distributionGroup;
+ if ((((UintPtr)aValue & 3) != 0) || (slack != 0)){
+ memcpy(&tempData[0], aValue, attributeSize);
+ aValue = (char*)&tempData[0];
+ if(slack != 0) {
+ char * tmp = (char*)&tempData[0];
+ memset(&tmp[attributeSize], 0, (4 - slack));
+ }//if
+ }//if
+ OperationType tOpType = theOperationType;
+ if ((tDistrKey != 1) && (tDistrGroup != 1)) {
+ ;
+ } else if (tDistrKey == 1) {
+ theDistrKeySize += totalSizeInWords;
+ theDistrKeyIndicator = 1;
+ } else {
+ Uint32 TsizeInBytes = sizeInBytes;
+ Uint32 TbyteOrderFix = 0;
+ char* TcharByteOrderFix = (char*)&TbyteOrderFix;
+ if (tAttrInfo->m_distributionGroupBits == 8) {
+ char tFirstChar = aValue[TsizeInBytes - 2];
+ char tSecondChar = aValue[TsizeInBytes - 2];
+ TcharByteOrderFix[0] = tFirstChar;
+ TcharByteOrderFix[1] = tSecondChar;
+ TcharByteOrderFix[2] = 0x30;
+ TcharByteOrderFix[3] = 0x30;
+ theDistrGroupType = 0;
+ } else {
+ TbyteOrderFix = ((aValue[TsizeInBytes - 2] - 0x30) * 10)
+ + (aValue[TsizeInBytes - 1] - 0x30);
+ theDistrGroupType = 1;
+ }//if
+ theDistributionGroup = TbyteOrderFix;
+ theDistrGroupIndicator = 1;
+ }//if
+ /**************************************************************************
+ * If the operation is an insert request and the attribute is stored then
+ * we also set the value in the stored part through putting the
+ * information in the INDXATTRINFO signals.
+ *************************************************************************/
+ if ((tOpType == InsertRequest) ||
+ (tOpType == WriteRequest)) {
+ if (!tAttrInfo->m_indexOnly){
+ Uint32 ahValue;
+ Uint32 sz = totalSizeInWords;
+ AttributeHeader::init(&ahValue, tAttrId, sz);
+ insertATTRINFO( ahValue );
+ insertATTRINFOloop((Uint32*)aValue, sizeInWords);
+ if (bitsInLastWord != 0) {
+ tData = *(Uint32*)(aValue + (sizeInWords << 2));
+ tData = convertEndian(tData);
+ tData = tData & ((1 << bitsInLastWord) - 1);
+ tData = convertEndian(tData);
+ insertATTRINFO( tData );
+ }//if
+ }//if
+ }//if
+
+ /**************************************************************************
+ * Store the Key information in the TCINDXREQ and INDXKEYINFO signals.
+ *************************************************************************/
+ if (insertKEYINFO(aValue, tKeyInfoPosition,
+ totalSizeInWords, bitsInLastWord) != -1) {
+ /************************************************************************
+ * Add one to number of tuple key attributes defined.
+ * If all have been defined then set the operation state to indicate
+ * that tuple key is defined.
+ * Thereby no more search conditions are allowed in this version.
+ ***********************************************************************/
+ Uint32 tNoIndexDef = m_theNoOfIndexDefined;
+ Uint32 tErrorLine = theErrorLine;
+ int tNoIndexAttrs = m_theIndex->m_columns.size();
+ unsigned char tInterpretInd = theInterpretIndicator;
+ tNoIndexDef++;
+ m_theNoOfIndexDefined = tNoIndexDef;
+ tErrorLine++;
+ theErrorLine = tErrorLine;
+ if (int(tNoIndexDef) == tNoIndexAttrs) {
+ if (tOpType == UpdateRequest) {
+ if (tInterpretInd == 1) {
+ theStatus = GetValue;
+ } else {
+ theStatus = SetValue;
+ }//if
+ return 0;
+ } else if ((tOpType == ReadRequest) || (tOpType == DeleteRequest) ||
+ (tOpType == ReadExclusive)) {
+ theStatus = GetValue;
+ return 0;
+ } else if ((tOpType == InsertRequest) || (tOpType == WriteRequest)) {
+ theStatus = SetValue;
+ return 0;
+ } else {
+ setErrorCodeAbort(4005);
+ return -1;
+ }//if
+ }//if
+ return 0;
+ } else {
+
+ return -1;
+ }//if
+ } else {
+ if (theStatus != OperationDefined) {
+ return -1;
+ }//if
+
+ if (aValue == NULL) {
+ setErrorCodeAbort(4505);
+ return -1;
+ }//if
+
+ if ( tAttrInfo == NULL ) {
+ setErrorCodeAbort(4004);
+ return -1;
+ }//if
+ }//if
+ return -1;
+
+ equal_error1:
+ setErrorCodeAbort(4205);
+ return -1;
+
+ equal_error2:
+ setErrorCodeAbort(4206);
+ return -1;
+
+ equal_error3:
+ setErrorCodeAbort(4209);
+
+ return -1;
+}
+
+int NdbIndexOperation::executeCursor(int aProcessorId)
+{
+ printf("NdbIndexOperation::executeCursor NYI\n");
+ // NYI
+ return -1;
+}
+void
+NdbIndexOperation::setLastFlag(NdbApiSignal* signal, Uint32 lastFlag)
+{
+ TcIndxReq * const req = CAST_PTR(TcIndxReq, signal->getDataPtrSend());
+ TcKeyReq::setExecuteFlag(req->requestInfo, lastFlag);
+}
+
+int
+NdbIndexOperation::prepareSend(Uint32 aTC_ConnectPtr, Uint64 aTransactionId)
+{
+ Uint32 tTransId1, tTransId2;
+ Uint32 tReqInfo;
+ Uint32 tSignalCount = 0;
+ Uint32 tInterpretInd = theInterpretIndicator;
+
+ theErrorLine = 0;
+
+ if (tInterpretInd != 1) {
+ OperationType tOpType = theOperationType;
+ OperationStatus tStatus = theStatus;
+ if ((tOpType == UpdateRequest) ||
+ (tOpType == InsertRequest) ||
+ (tOpType == WriteRequest)) {
+ if (tStatus != SetValue) {
+ setErrorCodeAbort(4506);
+ return -1;
+ }//if
+ } else if ((tOpType == ReadRequest) || (tOpType == ReadExclusive) ||
+ (tOpType == DeleteRequest)) {
+ if (tStatus != GetValue) {
+ setErrorCodeAbort(4506);
+ return -1;
+ }//if
+ } else {
+ setErrorCodeAbort(4507);
+ return -1;
+ }//if
+ } else {
+ if (prepareSendInterpreted() == -1) {
+ return -1;
+ }//if
+ }//if
+
+//-------------------------------------------------------------
+// We start by filling in the first 8 unconditional words of the
+// TCINDXREQ signal.
+//-------------------------------------------------------------
+ TcIndxReq * const tcIndxReq =
+ CAST_PTR(TcIndxReq, theTCREQ->getDataPtrSend());
+
+ Uint32 tTotalCurrAI_Len = theTotalCurrAI_Len;
+ Uint32 tIndexId = m_theIndex->m_indexId;
+ Uint32 tSchemaVersion = m_theIndex->m_version;
+
+ tcIndxReq->apiConnectPtr = aTC_ConnectPtr;
+ tcIndxReq->senderData = ptr2int();
+ tcIndxReq->attrLen = tTotalCurrAI_Len;
+ tcIndxReq->indexId = tIndexId;
+ tcIndxReq->indexSchemaVersion = tSchemaVersion;
+
+ tTransId1 = (Uint32) aTransactionId;
+ tTransId2 = (Uint32) (aTransactionId >> 32);
+
+//-------------------------------------------------------------
+// Simple is simple if simple or both start and commit is set.
+//-------------------------------------------------------------
+// Temporarily disable simple stuff
+ Uint8 tSimpleIndicator = 0;
+// Uint8 tSimpleIndicator = theSimpleIndicator;
+ Uint8 tCommitIndicator = theCommitIndicator;
+ Uint8 tStartIndicator = theStartIndicator;
+// if ((theNdbCon->theLastOpInList == this) && (theCommitIndicator == 0))
+// abort();
+// Temporarily disable simple stuff
+ Uint8 tSimpleAlt = 0;
+// Uint8 tSimpleAlt = tStartIndicator & tCommitIndicator;
+ tSimpleIndicator = tSimpleIndicator | tSimpleAlt;
+
+//-------------------------------------------------------------
+// Simple state is set if start and commit is set and it is
+// a read request. Otherwise it is set to zero.
+//-------------------------------------------------------------
+ Uint8 tReadInd = (theOperationType == ReadRequest);
+ Uint8 tSimpleState = tReadInd & tSimpleAlt;
+ theNdbCon->theSimpleState = tSimpleState;
+
+ tcIndxReq->transId1 = tTransId1;
+ tcIndxReq->transId2 = tTransId2;
+
+ tReqInfo = 0;
+
+ if (tTotalCurrAI_Len <= TcIndxReq::MaxAttrInfo) {
+ tcIndxReq->setAIInTcIndxReq(tReqInfo, tTotalCurrAI_Len);
+ } else {
+ tcIndxReq->setAIInTcIndxReq(tReqInfo, TcIndxReq::MaxAttrInfo);
+ }//if
+
+ tcIndxReq->setSimpleFlag(tReqInfo, tSimpleIndicator);
+ tcIndxReq->setCommitFlag(tReqInfo, tCommitIndicator);
+ tcIndxReq->setStartFlag(tReqInfo, tStartIndicator);
+ const Uint8 tInterpretIndicator = theInterpretIndicator;
+ tcIndxReq->setInterpretedFlag(tReqInfo, tInterpretIndicator);
+
+ Uint8 tDirtyIndicator = theDirtyIndicator;
+ OperationType tOperationType = theOperationType;
+ Uint32 tIndexLen = m_theIndexLen;
+ Uint8 abortOption = theNdbCon->m_abortOption;
+
+ tcIndxReq->setDirtyFlag(tReqInfo, tDirtyIndicator);
+ tcIndxReq->setOperationType(tReqInfo, tOperationType);
+ tcIndxReq->setIndexLength(tReqInfo, tIndexLen);
+ tcIndxReq->setCommitType(tReqInfo, abortOption);
+
+ Uint8 tDistrKeyIndicator = theDistrKeyIndicator;
+ Uint8 tDistrGroupIndicator = theDistrGroupIndicator;
+ Uint8 tDistrGroupType = theDistrGroupType;
+ Uint8 tScanIndicator = theScanInfo & 1;
+
+ tcIndxReq->setDistributionGroupFlag(tReqInfo, tDistrGroupIndicator);
+ tcIndxReq->setDistributionGroupTypeFlag(tReqInfo, tDistrGroupType);
+ tcIndxReq->setDistributionKeyFlag(tReqInfo, tDistrKeyIndicator);
+ tcIndxReq->setScanIndFlag(tReqInfo, tScanIndicator);
+
+ tcIndxReq->requestInfo = tReqInfo;
+
+//-------------------------------------------------------------
+// The next step is to fill in the upto three conditional words.
+//-------------------------------------------------------------
+ Uint32* tOptionalDataPtr = &tcIndxReq->scanInfo;
+ Uint32 tDistrGHIndex = tScanIndicator;
+ Uint32 tDistrKeyIndex = tDistrGHIndex + tDistrGroupIndicator;
+
+ Uint32 tScanInfo = theScanInfo;
+ Uint32 tDistributionGroup = theDistributionGroup;
+ Uint32 tDistrKeySize = theDistrKeySize;
+
+ tOptionalDataPtr[0] = tScanInfo;
+ tOptionalDataPtr[tDistrGHIndex] = tDistributionGroup;
+ tOptionalDataPtr[tDistrKeyIndex] = tDistrKeySize;
+
+//-------------------------------------------------------------
+// The next is step is to compress the key data part of the
+// TCKEYREQ signal.
+//-------------------------------------------------------------
+ Uint32 tKeyIndex = tDistrKeyIndex + tDistrKeyIndicator;
+ Uint32* tKeyDataPtr = &tOptionalDataPtr[tKeyIndex];
+ Uint32 Tdata1 = tcIndxReq->keyInfo[0];
+ Uint32 Tdata2 = tcIndxReq->keyInfo[1];
+ Uint32 Tdata3 = tcIndxReq->keyInfo[2];
+ Uint32 Tdata4 = tcIndxReq->keyInfo[3];
+ Uint32 Tdata5;
+
+ tKeyDataPtr[0] = Tdata1;
+ tKeyDataPtr[1] = Tdata2;
+ tKeyDataPtr[2] = Tdata3;
+ tKeyDataPtr[3] = Tdata4;
+ if (tIndexLen > 4) {
+ Tdata1 = tcIndxReq->keyInfo[4];
+ Tdata2 = tcIndxReq->keyInfo[5];
+ Tdata3 = tcIndxReq->keyInfo[6];
+ Tdata4 = tcIndxReq->keyInfo[7];
+
+ tKeyDataPtr[4] = Tdata1;
+ tKeyDataPtr[5] = Tdata2;
+ tKeyDataPtr[6] = Tdata3;
+ tKeyDataPtr[7] = Tdata4;
+ }//if
+//-------------------------------------------------------------
+// Finally we also compress the INDXATTRINFO part of the signal.
+// We optimise by using the if-statement for sending INDXKEYINFO
+// signals to calculating the new Attrinfo Index.
+//-------------------------------------------------------------
+ Uint32 tAttrInfoIndex;
+
+ if (tIndexLen > TcIndxReq::MaxKeyInfo) {
+ /**
+ * Set transid and TC connect ptr in the INDXKEYINFO signals
+ */
+ NdbApiSignal* tSignal = theFirstKEYINFO;
+ Uint32 remainingKey = tIndexLen - TcIndxReq::MaxKeyInfo;
+
+ do {
+ Uint32* tSigDataPtr = tSignal->getDataPtrSend();
+ NdbApiSignal* tnextSignal = tSignal->next();
+ tSignalCount++;
+ tSigDataPtr[0] = aTC_ConnectPtr;
+ tSigDataPtr[1] = tTransId1;
+ tSigDataPtr[2] = tTransId2;
+ if (remainingKey > IndxKeyInfo::DataLength) {
+ // The signal is full
+ tSignal->setLength(IndxKeyInfo::MaxSignalLength);
+ remainingKey -= IndxKeyInfo::DataLength;
+ }
+ else {
+ // Last signal
+ tSignal->setLength(IndxKeyInfo::HeaderLength + remainingKey);
+ remainingKey = 0;
+ }
+ tSignal = tnextSignal;
+ } while (tSignal != NULL);
+ tAttrInfoIndex = tKeyIndex + TcIndxReq::MaxKeyInfo;
+ } else {
+ tAttrInfoIndex = tKeyIndex + tIndexLen;
+ }//if
+
+//-------------------------------------------------------------
+// Perform the Attrinfo packing in the TCKEYREQ signal started
+// above.
+//-------------------------------------------------------------
+ Uint32* tAIDataPtr = &tOptionalDataPtr[tAttrInfoIndex];
+ Tdata1 = tcIndxReq->attrInfo[0];
+ Tdata2 = tcIndxReq->attrInfo[1];
+ Tdata3 = tcIndxReq->attrInfo[2];
+ Tdata4 = tcIndxReq->attrInfo[3];
+ Tdata5 = tcIndxReq->attrInfo[4];
+
+ theTCREQ->setLength(tcIndxReq->getAIInTcIndxReq(tReqInfo) +
+ tAttrInfoIndex + TcIndxReq::StaticLength);
+ tAIDataPtr[0] = Tdata1;
+ tAIDataPtr[1] = Tdata2;
+ tAIDataPtr[2] = Tdata3;
+ tAIDataPtr[3] = Tdata4;
+ tAIDataPtr[4] = Tdata5;
+
+/***************************************************
+* Send the INDXATTRINFO signals.
+***************************************************/
+ if (tTotalCurrAI_Len > 5) {
+ // Set the last signal's length.
+ NdbApiSignal* tSignal = theFirstATTRINFO;
+ theCurrentATTRINFO->setLength(theAI_LenInCurrAI);
+ do {
+ Uint32* tSigDataPtr = tSignal->getDataPtrSend();
+ NdbApiSignal* tnextSignal = tSignal->next();
+ tSignalCount++;
+ tSigDataPtr[0] = aTC_ConnectPtr;
+ tSigDataPtr[1] = tTransId1;
+ tSigDataPtr[2] = tTransId2;
+ tSignal = tnextSignal;
+ } while (tSignal != NULL);
+ }//if
+ NdbRecAttr* tRecAttrObject = theFirstRecAttr;
+ theStatus = WaitResponse;
+ theCurrentRecAttr = tRecAttrObject;
+
+ return 0;
+}
+
+void NdbIndexOperation::closeScan()
+{
+ printf("NdbIndexOperation::closeScan NYI\n");
+}
+
+/***************************************************************************
+int receiveTCINDXREF( NdbApiSignal* aSignal)
+
+Return Value: Return 0 : send was succesful.
+ Return -1: In all other case.
+Parameters: aSignal: the signal object that contains the TCINDXREF signal from TC.
+Remark: Handles the reception of the TCKEYREF signal.
+***************************************************************************/
+int
+NdbIndexOperation::receiveTCINDXREF( NdbApiSignal* aSignal)
+{
+ const TcIndxRef * const tcIndxRef = CAST_CONSTPTR(TcIndxRef, aSignal->getDataPtr());
+
+ if (checkState_TransId(aSignal) == -1) {
+ return -1;
+ }//if
+
+ theStatus = Finished;
+
+ theNdbCon->theReturnStatus = ReturnFailure;
+ //--------------------------------------------------------------------------//
+ // If the transaction this operation belongs to consists only of simple reads
+ // we set the error code on the transaction object.
+ // If the transaction consists of other types of operations we set
+ // the error code only on the operation since the simple read is not really
+ // part of this transaction and we can not decide the status of the whole
+ // transaction based on this operation.
+ //--------------------------------------------------------------------------//
+ Uint32 errorCode = tcIndxRef->errorCode;
+ if (theNdbCon->theSimpleState == 0) {
+ theError.code = errorCode;
+ theNdbCon->setOperationErrorCodeAbort(errorCode);
+ return theNdbCon->OpCompleteFailure();
+ } else {
+ theError.code = errorCode;
+ return theNdbCon->OpCompleteSuccess();
+ }
+}//NdbIndexOperation::receiveTCINDXREF()
+
+
+
diff --git a/ndb/src/ndbapi/NdbLinHash.hpp b/ndb/src/ndbapi/NdbLinHash.hpp
new file mode 100644
index 00000000000..f67d4e60200
--- /dev/null
+++ b/ndb/src/ndbapi/NdbLinHash.hpp
@@ -0,0 +1,446 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbLinHash_H
+#define NdbLinHash_H
+
+#include <ndb_types.h>
+
+#define SEGMENTSIZE 64
+#define SEGMENTLOGSIZE 6
+#define DIRECTORYSIZE 64
+#define DIRINDEX(adress) ((adress) >> SEGMENTLOGSIZE)
+#define SEGINDEX(adress) ((adress) & (SEGMENTSIZE-1))
+
+#if !defined(MAXLOADFCTR)
+#define MAXLOADFCTR 2
+#endif
+#if !defined(MINLOADFCTR)
+#define MINLOADFCTR (MAXLOADFCTR/2)
+#endif
+
+template<class C>
+class NdbElement_t {
+public:
+ NdbElement_t();
+ ~NdbElement_t();
+
+ Uint32 len;
+ Uint32 hash;
+ Uint32 localkey1;
+ Uint32 *str;
+ NdbElement_t<C> *next;
+ C* theData;
+private:
+ NdbElement_t(const NdbElement_t<C> & aElement_t);
+ NdbElement_t & operator = (const NdbElement_t<C> & aElement_t);
+};
+
+
+template <class C>
+class NdbLinHash {
+public:
+ NdbLinHash();
+ ~NdbLinHash();
+ void createHashTable(void);
+ void releaseHashTable(void);
+
+ int insertKey(const char * str, Uint32 len, Uint32 lkey1, C* data);
+ int deleteKey(const char * str, Uint32 len);
+
+ C* getData(const char *, Uint32);
+ Uint32* getKey(const char *, Uint32);
+
+ void shrinkTable(void);
+ void expandHashTable(void);
+
+ Uint32 Hash(const char *str, Uint32 len);
+ Uint32 Hash(Uint32 h);
+
+ NdbElement_t<C> * getNext(NdbElement_t<C> * curr);
+
+private:
+ void getBucket(Uint32 hash, int * dirindex, int * segindex);
+
+ struct Segment_t {
+ NdbElement_t<C> * elements[SEGMENTSIZE];
+ };
+
+ Uint32 p; /*bucket to be split*/
+ Uint32 max; /*max is the upper bound*/
+ Int32 slack; /*number of insertions before splits*/
+ Segment_t * directory[DIRECTORYSIZE];
+
+ NdbLinHash(const NdbLinHash<C> & aLinHash);
+ NdbLinHash<C> & operator = (const NdbLinHash<C> & aLinHash);
+};
+
+// All template methods must be inline
+
+template <class C>
+inline
+NdbLinHash<C>::NdbLinHash() {
+}
+
+template <class C>
+inline
+NdbLinHash<C>::NdbLinHash(const NdbLinHash<C>& aLinHash)
+{
+}
+
+template <class C>
+inline
+NdbLinHash<C>::~NdbLinHash()
+{
+}
+
+template <class C>
+inline
+Uint32
+NdbLinHash<C>::Hash( const char* str, Uint32 len )
+{
+ Uint32 h = 0;
+ while(len >= 4){
+ h = (h << 5) + h + str[0];
+ h = (h << 5) + h + str[1];
+ h = (h << 5) + h + str[2];
+ h = (h << 5) + h + str[3];
+ len -= 4;
+ str += 4;
+ }
+
+ while(len > 0){
+ h = (h << 5) + h + *str++;
+ len--;
+ }
+ return h;
+}
+
+template <class C>
+inline
+Uint32
+NdbLinHash<C>::Hash( Uint32 h ){
+ return h;
+}
+
+template <class C>
+inline
+NdbElement_t<C>::NdbElement_t() :
+ len(0),
+ hash(0),
+ localkey1(0),
+ str(NULL),
+ next(NULL),
+ theData(NULL)
+{
+}
+
+template <class C>
+inline
+NdbElement_t<C>::~NdbElement_t()
+{
+ delete []str;
+}
+
+
+/* Initialize the hashtable HASH_T */
+template <class C>
+inline
+void
+NdbLinHash<C>::createHashTable() {
+ p = 0;
+ max = SEGMENTSIZE - 1;
+ slack = SEGMENTSIZE * MAXLOADFCTR;
+ directory[0] = new Segment_t();
+
+ /* The first segment cleared before used */
+ for(int i = 0; i < SEGMENTSIZE; i++ )
+ directory[0]->elements[i] = 0;
+
+ /* clear the rest of the directory */
+ for( int i = 1; i < DIRECTORYSIZE; i++)
+ directory[i] = 0;
+}
+
+template <class C>
+inline
+void
+NdbLinHash<C>::getBucket(Uint32 hash, int * dir, int * seg){
+ Uint32 adress = hash & max;
+ if(adress < p)
+ adress = hash & (2 * max + 1);
+
+ * dir = DIRINDEX(adress);
+ * seg = SEGINDEX(adress);
+}
+
+template <class C>
+inline
+Int32
+NdbLinHash<C>::insertKey( const char* str, Uint32 len, Uint32 lkey1, C* data )
+{
+ const Uint32 hash = Hash(str, len);
+ int dir, seg;
+ getBucket(hash, &dir, &seg);
+
+ NdbElement_t<C> **chainp = &directory[dir]->elements[seg];
+
+ /**
+ * Check if the string already are in the hash table
+ * chain=chainp will copy the contents of HASH_T into chain
+ */
+ NdbElement_t<C> * oldChain = 0;
+ for(NdbElement_t<C> * chain = *chainp; chain != 0; chain = chain->next){
+ if(chain->len == len && !memcmp(chain->str, str, len))
+ return -1; /* Element already exists */
+ else
+ oldChain = chain;
+ }
+
+ /* New entry */
+ NdbElement_t<C> * chain = new NdbElement_t<C>();
+ chain->len = len;
+ chain->hash = hash;
+ chain->localkey1 = lkey1;
+ chain->next = 0;
+ chain->theData = data;
+ chain->str = new Uint32[((len + 3) >> 2)];
+ memcpy( &chain->str[0], str, len );
+ if (oldChain != 0)
+ oldChain->next = chain;
+ else
+ *chainp = chain;
+
+#if 0
+ if(--(slack) < 0)
+ expandHashTable();
+#endif
+
+ return chain->localkey1;
+}
+
+
+template <class C>
+inline
+Uint32*
+NdbLinHash<C>::getKey( const char* str, Uint32 len )
+{
+ const Uint32 tHash = Hash(str, len);
+ int dir, seg;
+ getBucket(tHash, &dir, &seg);
+
+ NdbElement_t<C> ** keyp = &directory[dir]->elements[seg];
+
+ /*Check if the string are in the hash table*/
+ for(NdbElement_t<C> * key = *keyp; key != 0; key = key->next ) {
+ if(key->len == len && !memcmp(key->str, str, len)) {
+ return &key->localkey1;
+ }
+ }
+ return NULL ; /* The key was not found */
+}
+
+template <class C>
+inline
+C*
+NdbLinHash<C>::getData( const char* str, Uint32 len ){
+
+ const Uint32 tHash = Hash(str, len);
+ int dir, seg;
+ getBucket(tHash, &dir, &seg);
+
+ NdbElement_t<C> ** keyp = &directory[dir]->elements[seg];
+
+ /*Check if the string are in the hash table*/
+ for(NdbElement_t<C> * key = *keyp; key != 0; key = key->next ) {
+ if(key->len == len && !memcmp(key->str, str, len)) {
+ return key->theData;
+ }
+ }
+ return NULL ; /* The key was not found */
+}
+
+template <class C>
+inline
+int
+NdbLinHash<C>::deleteKey ( const char* str, Uint32 len){
+ const Uint32 hash = Hash(str, len);
+ int dir, seg;
+ getBucket(hash, &dir, &seg);
+
+ NdbElement_t<C> *oldChain = 0;
+ NdbElement_t<C> **chainp = &directory[dir]->elements[seg];
+ for(NdbElement_t<C> * chain = *chainp; chain != 0; chain = chain->next){
+ if(chain->len == len && !memcmp(chain->str, str, len)){
+ if (oldChain == 0) {
+ delete chain;
+ * chainp = 0;
+ return 1;
+ } else {
+ oldChain->next = chain->next;
+ delete chain;
+ return 1;
+ }
+ } else {
+ oldChain = chain;
+ }
+ }
+ return -1; /* Element doesn't exist */
+}
+
+template <class C>
+inline
+void
+NdbLinHash<C>::releaseHashTable( void ){
+ NdbElement_t<C>* tNextElement;
+ NdbElement_t<C>* tElement;
+
+ //Traverse the whole directory structure
+ for(int countd = 0; countd < DIRECTORYSIZE;countd++ ){
+ if (directory[countd] != 0) {
+ //Traverse whole hashtable
+ for(int counts = 0; counts < SEGMENTSIZE; counts++ )
+ if (directory[countd]->elements[counts] != 0) {
+ tElement = directory[countd]->elements[counts];
+ //Delete all elements even those who is linked
+ do {
+ tNextElement = tElement->next;
+ delete tElement;
+ tElement = tNextElement;
+ } while (tNextElement != 0);
+ }
+ delete directory[countd];
+ }
+ }
+}
+
+template <class C>
+inline
+void
+NdbLinHash<C>::shrinkTable( void )
+{
+ Segment_t *lastseg;
+ NdbElement_t<C> **chainp;
+ Uint32 oldlast = p + max;
+
+ if( oldlast == 0 )
+ return;
+
+ // Adjust the state variables.
+ if( p == 0 ) {
+ max >>= 1;
+ p = max;
+ }
+ else
+ --(p);
+
+ // Update slack after shrink.
+
+ slack -= MAXLOADFCTR;
+
+ // Insert the chain oldlast at the end of chain p.
+
+ chainp = &directory[DIRINDEX(p)]->elements[SEGINDEX(p)];
+ while( *chainp != 0 ) {
+ chainp = &((*chainp)->next);
+ lastseg = directory[DIRINDEX(oldlast)];
+ *chainp = lastseg->elements[SEGINDEX(oldlast)];
+
+ // If necessary free last segment.
+ if( SEGINDEX(oldlast) == 0)
+ delete lastseg;
+ }
+}
+
+template <class C>
+inline
+void
+NdbLinHash<C>::expandHashTable( void )
+{
+
+ NdbElement_t<C> **oldbucketp, *chain, *headofold, *headofnew, *next;
+ Uint32 maxp = max + 1;
+ Uint32 newadress = maxp + p;
+
+
+ // Still room in the adress space?
+ if( newadress >= DIRECTORYSIZE * SEGMENTSIZE ) {
+ return;
+ }
+
+ // If necessary, create a new segment.
+ if( SEGINDEX(newadress) == 0 )
+ directory[DIRINDEX(newadress)] = new Segment_t();
+
+ // Locate the old (to be split) bucket.
+ oldbucketp = &directory[DIRINDEX(p)]->elements[SEGINDEX(p)];
+
+ // Adjust the state variables.
+ p++;
+ if( p > max ) {
+ max = 2 *max + 1;
+ p = 0;
+ }
+
+ // Update slack after expandation.
+ slack += MAXLOADFCTR;
+
+ // Relocate records to the new bucket.
+ headofold = 0;
+ headofnew = 0;
+
+ for( chain = *oldbucketp; chain != 0; chain = next ) {
+ next = chain->next;
+ if( chain->hash & maxp ) {
+ chain->next = headofnew;
+ headofnew = chain;
+ }
+ else {
+ chain->next = headofold;
+ headofold = chain;
+ }
+ }
+ *oldbucketp = headofold;
+ directory[DIRINDEX(newadress)]->elements[SEGINDEX(newadress)] = headofnew;
+}
+
+template <class C>
+inline
+NdbElement_t<C> *
+NdbLinHash<C>::getNext(NdbElement_t<C> * curr){
+ if(curr != 0 && curr->next != 0)
+ return curr->next;
+
+ int dir = 0, seg = 0;
+
+ if(curr != 0){
+ getBucket(curr->hash, &dir, &seg);
+ }
+
+ for(int countd = dir; countd < DIRECTORYSIZE;countd++ ){
+ if (directory[countd] != 0) {
+ for(int counts = seg + 1; counts < SEGMENTSIZE; counts++ ){
+ if (directory[countd]->elements[counts] != 0) {
+ return directory[countd]->elements[counts];
+ }
+ }
+ }
+ }
+
+ return 0;
+}
+
+#endif
diff --git a/ndb/src/ndbapi/NdbOperation.cpp b/ndb/src/ndbapi/NdbOperation.cpp
new file mode 100644
index 00000000000..eaa2b35965b
--- /dev/null
+++ b/ndb/src/ndbapi/NdbOperation.cpp
@@ -0,0 +1,432 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/*****************************************************************************
+ * Name: NdbOperation.C
+ * Include:
+ * Link:
+ * Author: UABMNST Mona Natterkvist UAB/B/SD
+ * Date: 970829
+ * Version: 0.1
+ * Description: Interface between TIS and NDB
+ * Documentation:
+ * Adjust: 971022 UABMNST First version.
+ ****************************************************************************/
+#include "NdbConnection.hpp"
+#include "NdbOperation.hpp"
+#include "NdbApiSignal.hpp"
+#include "NdbRecAttr.hpp"
+#include "NdbUtil.hpp"
+
+#include <signaldata/TcKeyReq.hpp>
+#include "NdbDictionaryImpl.hpp"
+
+#include "API.hpp"
+#include <NdbOut.hpp>
+#include <assert.h>
+
+
+/******************************************************************************
+ * NdbOperation(Ndb* aNdb, Table* aTable);
+ *
+ * Return Value: None
+ * Parameters: aNdb: Pointers to the Ndb object.
+ * aTable: Pointers to the Table object
+ * Remark: Creat an object of NdbOperation.
+ ****************************************************************************/
+NdbOperation::NdbOperation(Ndb* aNdb) :
+ theReceiver(aNdb),
+ theErrorLine(0),
+ theNdb(aNdb),
+ //theTable(aTable),
+ theNdbCon(NULL),
+ theNext(NULL),
+ theNextScanOp(NULL),
+ theTCREQ(NULL),
+ theFirstATTRINFO(NULL),
+ theCurrentATTRINFO(NULL),
+ theTotalCurrAI_Len(0),
+ theAI_LenInCurrAI(0),
+ theFirstKEYINFO(NULL),
+ theLastKEYINFO(NULL),
+ theFirstRecAttr(NULL),
+ theCurrentRecAttr(NULL),
+
+ theFirstLabel(NULL),
+ theLastLabel(NULL),
+ theFirstBranch(NULL),
+ theLastBranch(NULL),
+ theFirstCall(NULL),
+ theLastCall(NULL),
+ theFirstSubroutine(NULL),
+ theLastSubroutine(NULL),
+ theNoOfLabels(0),
+ theNoOfSubroutines(0),
+
+ theTotalRecAI_Len(0),
+ theCurrRecAI_Len(0),
+ theAI_ElementLen(0),
+ theCurrElemPtr(NULL),
+ m_currentTable(NULL), //theTableId(0xFFFF),
+ m_accessTable(NULL), //theAccessTableId(0xFFFF),
+ //theSchemaVersion(0),
+ theTotalNrOfKeyWordInSignal(8),
+ theTupKeyLen(0),
+ theNoOfTupKeyDefined(0),
+ theOperationType(NotDefined),
+ theStatus(Init),
+ theMagicNumber(0xFE11D0),
+ theScanInfo(0),
+ theDistrKeySize(0),
+ theDistributionGroup(0),
+ m_tcReqGSN(GSN_TCKEYREQ),
+ m_keyInfoGSN(GSN_KEYINFO),
+ m_attrInfoGSN(GSN_ATTRINFO),
+ theParallelism(0),
+ theScanReceiversArray(NULL),
+ theSCAN_TABREQ(NULL),
+ theFirstSCAN_TABINFO_Send(NULL),
+ theLastSCAN_TABINFO_Send(NULL),
+ theFirstSCAN_TABINFO_Recv(NULL),
+ theLastSCAN_TABINFO_Recv(NULL),
+ theSCAN_TABCONF_Recv(NULL),
+ theBoundATTRINFO(NULL)
+{
+ theReceiver.init(NdbReceiver::NDB_OPERATION, this);
+ theError.code = 0;
+}
+/*****************************************************************************
+ * ~NdbOperation();
+ *
+ * Remark: Delete tables for connection pointers (id).
+ *****************************************************************************/
+NdbOperation::~NdbOperation( )
+{
+}
+/******************************************************************************
+ *void setErrorCode(int anErrorCode);
+ *
+ * Remark: Set an Error Code on operation and
+ * on connection set an error status.
+ *****************************************************************************/
+void
+NdbOperation::setErrorCode(int anErrorCode)
+{
+ theError.code = anErrorCode;
+ theNdbCon->theErrorLine = theErrorLine;
+ theNdbCon->theErrorOperation = this;
+ theNdbCon->setOperationErrorCode(anErrorCode);
+}
+
+/******************************************************************************
+ * void setErrorCodeAbort(int anErrorCode);
+ *
+ * Remark: Set an Error Code on operation and on connection set
+ * an error status.
+ *****************************************************************************/
+void
+NdbOperation::setErrorCodeAbort(int anErrorCode)
+{
+ theError.code = anErrorCode;
+ theNdbCon->theErrorLine = theErrorLine;
+ theNdbCon->theErrorOperation = this;
+ theNdbCon->setOperationErrorCodeAbort(anErrorCode);
+}
+
+/*****************************************************************************
+ * int init();
+ *
+ * Return Value: Return 0 : init was successful.
+ * Return -1: In all other case.
+ * Remark: Initiates operation record after allocation.
+ *****************************************************************************/
+
+int
+NdbOperation::init(NdbTableImpl* tab, NdbConnection* myConnection){
+ NdbApiSignal* tSignal;
+ theStatus = Init;
+ theError.code = 0;
+ theErrorLine = 1;
+ m_currentTable = m_accessTable = tab;
+
+ theNdbCon = myConnection;
+ for (Uint32 i=0; i<MAXNROFTUPLEKEY; i++)
+ for (int j=0; j<3; j++)
+ theTupleKeyDefined[i][j] = false;
+
+ theFirstATTRINFO = NULL;
+ theCurrentATTRINFO = NULL;
+ theFirstKEYINFO = NULL;
+ theLastKEYINFO = NULL;
+
+
+ theTupKeyLen = 0;
+ theNoOfTupKeyDefined = 0;
+ theTotalCurrAI_Len = 0;
+ theAI_LenInCurrAI = 0;
+ theTotalRecAI_Len = 0;
+ theDistrKeySize = 0;
+ theDistributionGroup = 0;
+ theCurrRecAI_Len = 0;
+ theAI_ElementLen = 0;
+ theStartIndicator = 0;
+ theCommitIndicator = 0;
+ theSimpleIndicator = 0;
+ theDirtyIndicator = 0;
+ theInterpretIndicator = 0;
+ theDistrGroupIndicator= 0;
+ theDistrGroupType = 0;
+ theDistrKeyIndicator = 0;
+ theScanInfo = 0;
+ theFirstRecAttr = NULL;
+ theCurrentRecAttr = NULL;
+ theCurrElemPtr = NULL;
+ theTotalNrOfKeyWordInSignal = 8;
+ theMagicNumber = 0xABCDEF01;
+ theBoundATTRINFO = NULL;
+
+ tSignal = theNdb->getSignal();
+ if (tSignal == NULL)
+ {
+ setErrorCode(4000);
+ return -1;
+ }
+ theTCREQ = tSignal;
+ theTCREQ->setSignal(m_tcReqGSN);
+
+ theAI_LenInCurrAI = 20;
+ TcKeyReq * const tcKeyReq = CAST_PTR(TcKeyReq, theTCREQ->getDataPtrSend());
+ tcKeyReq->scanInfo = 0;
+ theKEYINFOptr = &tcKeyReq->keyInfo[0];
+ theATTRINFOptr = &tcKeyReq->attrInfo[0];
+ return 0;
+}
+
+
+/******************************************************************************
+ * void release();
+ *
+ * Remark: Release all objects connected to the operation object.
+ *****************************************************************************/
+void
+NdbOperation::release()
+{
+ NdbApiSignal* tSignal;
+ NdbApiSignal* tSaveSignal;
+ NdbRecAttr* tRecAttr;
+ NdbRecAttr* tSaveRecAttr;
+ NdbBranch* tBranch;
+ NdbBranch* tSaveBranch;
+ NdbLabel* tLabel;
+ NdbLabel* tSaveLabel;
+ NdbCall* tCall;
+ NdbCall* tSaveCall;
+ NdbSubroutine* tSubroutine;
+ NdbSubroutine* tSaveSubroutine;
+
+ if (theTCREQ != NULL)
+ {
+ theNdb->releaseSignal(theTCREQ);
+ }
+ theTCREQ = NULL;
+ tSignal = theFirstATTRINFO;
+ while (tSignal != NULL)
+ {
+ tSaveSignal = tSignal;
+ tSignal = tSignal->next();
+ theNdb->releaseSignal(tSaveSignal);
+ }
+ theFirstATTRINFO = NULL;
+ theCurrentATTRINFO = NULL;
+ tSignal = theFirstKEYINFO;
+ while (tSignal != NULL)
+ {
+ tSaveSignal = tSignal;
+ tSignal = tSignal->next();
+ theNdb->releaseSignal(tSaveSignal);
+ }
+ theFirstKEYINFO = NULL;
+ theLastKEYINFO = NULL;
+ tRecAttr = theFirstRecAttr;
+ while (tRecAttr != NULL)
+ {
+ tSaveRecAttr = tRecAttr;
+ tRecAttr = tRecAttr->next();
+ theNdb->releaseRecAttr(tSaveRecAttr);
+ }
+ theFirstRecAttr = NULL;
+ theCurrentRecAttr = NULL;
+ if (theInterpretIndicator == 1)
+ {
+ tBranch = theFirstBranch;
+ while (tBranch != NULL)
+ {
+ tSaveBranch = tBranch;
+ tBranch = tBranch->theNext;
+ theNdb->releaseNdbBranch(tSaveBranch);
+ }
+ tLabel = theFirstLabel;
+ while (tLabel != NULL)
+ {
+ tSaveLabel = tLabel;
+ tLabel = tLabel->theNext;
+ theNdb->releaseNdbLabel(tSaveLabel);
+ }
+ tCall = theFirstCall;
+ while (tCall != NULL)
+ {
+ tSaveCall = tCall;
+ tCall = tCall->theNext;
+ theNdb->releaseNdbCall(tSaveCall);
+ }
+ tSubroutine = theFirstSubroutine;
+ while (tSubroutine != NULL)
+ {
+ tSaveSubroutine = tSubroutine;
+ tSubroutine = tSubroutine->theNext;
+ theNdb->releaseNdbSubroutine(tSaveSubroutine);
+ }
+ tSignal = theBoundATTRINFO;
+ while (tSignal != NULL)
+ {
+ tSaveSignal = tSignal;
+ tSignal = tSignal->next();
+ theNdb->releaseSignal(tSaveSignal);
+ }
+ theBoundATTRINFO = NULL;
+ }
+ releaseScan();
+}
+
+NdbRecAttr*
+NdbOperation::getValue(const char* anAttrName, char* aValue)
+{
+ return getValue(m_currentTable->getColumn(anAttrName), aValue);
+}
+
+NdbRecAttr*
+NdbOperation::getValue(Uint32 anAttrId, char* aValue)
+{
+ return getValue(m_currentTable->getColumn(anAttrId), aValue);
+}
+
+int
+NdbOperation::equal(const char* anAttrName,
+ const char* aValuePassed,
+ Uint32 aVariableKeyLen)
+{
+ return equal_impl(m_accessTable->getColumn(anAttrName), aValuePassed, aVariableKeyLen);
+}
+
+int
+NdbOperation::equal(Uint32 anAttrId,
+ const char* aValuePassed,
+ Uint32 aVariableKeyLen)
+{
+ return equal_impl(m_accessTable->getColumn(anAttrId), aValuePassed, aVariableKeyLen);
+}
+
+int
+NdbOperation::setValue( const char* anAttrName,
+ const char* aValuePassed,
+ Uint32 len)
+{
+ return setValue(m_currentTable->getColumn(anAttrName), aValuePassed, len);
+}
+
+
+int
+NdbOperation::setValue( Uint32 anAttrId,
+ const char* aValuePassed,
+ Uint32 len)
+{
+ return setValue(m_currentTable->getColumn(anAttrId), aValuePassed, len);
+}
+
+int
+NdbOperation::incValue(const char* anAttrName, Uint32 aValue)
+{
+ return incValue(m_currentTable->getColumn(anAttrName), aValue);
+}
+
+int
+NdbOperation::incValue(const char* anAttrName, Uint64 aValue)
+{
+ return incValue(m_currentTable->getColumn(anAttrName), aValue);
+}
+
+int
+NdbOperation::incValue(Uint32 anAttrId, Uint32 aValue)
+{
+ return incValue(m_currentTable->getColumn(anAttrId), aValue);
+}
+
+int
+NdbOperation::incValue(Uint32 anAttrId, Uint64 aValue)
+{
+ return incValue(m_currentTable->getColumn(anAttrId), aValue);
+}
+
+int
+NdbOperation::subValue( const char* anAttrName, Uint32 aValue)
+{
+ return subValue(m_currentTable->getColumn(anAttrName), aValue);
+}
+
+int
+NdbOperation::subValue(Uint32 anAttrId, Uint32 aValue)
+{
+ return subValue(m_currentTable->getColumn(anAttrId), aValue);
+}
+
+int
+NdbOperation::read_attr(const char* anAttrName, Uint32 RegDest)
+{
+ return read_attr(m_currentTable->getColumn(anAttrName), RegDest);
+}
+
+int
+NdbOperation::read_attr(Uint32 anAttrId, Uint32 RegDest)
+{
+ return read_attr(m_currentTable->getColumn(anAttrId), RegDest);
+}
+
+int
+NdbOperation::write_attr(const char* anAttrName, Uint32 RegDest)
+{
+ return write_attr(m_currentTable->getColumn(anAttrName), RegDest);
+}
+
+int
+NdbOperation::write_attr(Uint32 anAttrId, Uint32 RegDest)
+{
+ return write_attr(m_currentTable->getColumn(anAttrId), RegDest);
+}
+
+int
+NdbOperation::setBound(const char* anAttrName, int type, const void* aValue, Uint32 len)
+{
+ return setBound(m_accessTable->getColumn(anAttrName), type, aValue, len);
+}
+
+int
+NdbOperation::setBound(Uint32 anAttrId, int type, const void* aValue, Uint32 len)
+{
+ return setBound(m_accessTable->getColumn(anAttrId), type, aValue, len);
+}
+
+
diff --git a/ndb/src/ndbapi/NdbOperationDefine.cpp b/ndb/src/ndbapi/NdbOperationDefine.cpp
new file mode 100644
index 00000000000..a1ce25f19d1
--- /dev/null
+++ b/ndb/src/ndbapi/NdbOperationDefine.cpp
@@ -0,0 +1,769 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/*****************************************************************************
+ * Name: NdbOperationDefine.C
+ * Include:
+ * Link:
+ * Author: UABMNST Mona Natterkvist UAB/B/SD
+ * Date: 970829
+ * Version: 0.1
+ * Description: Interface between TIS and NDB
+ * Documentation:
+ * Adjust: 971022 UABMNST First version.
+ *****************************************************************************/
+#include "NdbOperation.hpp"
+#include "NdbApiSignal.hpp"
+#include "NdbConnection.hpp"
+#include "Ndb.hpp"
+#include "NdbRecAttr.hpp"
+#include "AttrType.hpp"
+#include "NdbUtil.hpp"
+#include "NdbOut.hpp"
+#include "NdbImpl.hpp"
+
+#include <Interpreter.hpp>
+
+#include <AttributeHeader.hpp>
+#include <signaldata/TcKeyReq.hpp>
+
+/*****************************************************************************
+ * int insertTuple();
+ *****************************************************************************/
+int
+NdbOperation::insertTuple()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ theOperationType = InsertRequest;
+ tNdbCon->theSimpleState = 0;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::insertTuple()
+/******************************************************************************
+ * int updateTuple();
+ *****************************************************************************/
+int
+NdbOperation::updateTuple()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ tNdbCon->theSimpleState = 0;
+ theOperationType = UpdateRequest;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::updateTuple()
+/*****************************************************************************
+ * int writeTuple();
+ *****************************************************************************/
+int
+NdbOperation::writeTuple()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ tNdbCon->theSimpleState = 0;
+ theOperationType = WriteRequest;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::writeTuple()
+/******************************************************************************
+ * int readTuple();
+ *****************************************************************************/
+int
+NdbOperation::readTuple()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ tNdbCon->theSimpleState = 0;
+ theOperationType = ReadRequest;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::readTuple()
+
+/*****************************************************************************
+ * int deleteTuple();
+ *****************************************************************************/
+int
+NdbOperation::deleteTuple()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ tNdbCon->theSimpleState = 0;
+ theOperationType = DeleteRequest;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::deleteTuple()
+
+/******************************************************************************
+ * int readTupleExclusive();
+ *****************************************************************************/
+int
+NdbOperation::readTupleExclusive()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ tNdbCon->theSimpleState = 0;
+ theOperationType = ReadExclusive;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::readTupleExclusive()
+
+/*****************************************************************************
+ * int simpleRead();
+ *****************************************************************************/
+int
+NdbOperation::simpleRead()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ theOperationType = ReadRequest;
+ theSimpleIndicator = 1;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::simpleRead()
+
+/*****************************************************************************
+ * int dirtyRead();
+ *****************************************************************************/
+int
+NdbOperation::dirtyRead()
+{
+ return committedRead();
+}//NdbOperation::dirtyRead()
+
+/*****************************************************************************
+ * int committedRead();
+ *****************************************************************************/
+int
+NdbOperation::committedRead()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ theOperationType = ReadRequest;
+ theSimpleIndicator = 1;
+ theDirtyIndicator = 1;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::committedRead()
+
+/*****************************************************************************
+ * int dirtyUpdate();
+ ****************************************************************************/
+int
+NdbOperation::dirtyUpdate()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ theOperationType = UpdateRequest;
+ tNdbCon->theSimpleState = 0;
+ theSimpleIndicator = 1;
+ theDirtyIndicator = 1;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::dirtyUpdate()
+
+/******************************************************************************
+ * int dirtyWrite();
+ *****************************************************************************/
+int
+NdbOperation::dirtyWrite()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ theOperationType = WriteRequest;
+ tNdbCon->theSimpleState = 0;
+ theSimpleIndicator = 1;
+ theDirtyIndicator = 1;
+ theErrorLine = tErrorLine++;
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::dirtyWrite()
+
+/******************************************************************************
+ * int interpretedUpdateTuple();
+ ****************************************************************************/
+int
+NdbOperation::interpretedUpdateTuple()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ tNdbCon->theSimpleState = 0;
+ theOperationType = UpdateRequest;
+ theInterpretIndicator = 1;
+ theAI_LenInCurrAI = 25;
+
+ theErrorLine = tErrorLine++;
+ theTotalCurrAI_Len = 5;
+ theSubroutineSize = 0;
+ theInitialReadSize = 0;
+ theInterpretedSize = 0;
+ theFinalUpdateSize = 0;
+ theFinalReadSize = 0;
+
+ theFirstLabel = NULL;
+ theLastLabel = NULL;
+ theFirstBranch = NULL;
+ theLastBranch = NULL;
+
+ theFirstCall = NULL;
+ theLastCall = NULL;
+ theFirstSubroutine = NULL;
+ theLastSubroutine = NULL;
+
+ theNoOfLabels = 0;
+ theNoOfSubroutines = 0;
+
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::interpretedUpdateTuple()
+
+/*****************************************************************************
+ * int interpretedDeleteTuple();
+ *****************************************************************************/
+int
+NdbOperation::interpretedDeleteTuple()
+{
+ NdbConnection* tNdbCon = theNdbCon;
+ int tErrorLine = theErrorLine;
+ if (theStatus == Init) {
+ theStatus = OperationDefined;
+ tNdbCon->theSimpleState = 0;
+ theOperationType = DeleteRequest;
+ theInterpretIndicator = 1;
+
+ theErrorLine = tErrorLine++;
+ theAI_LenInCurrAI = 25;
+ theTotalCurrAI_Len = 5;
+ theSubroutineSize = 0;
+ theInitialReadSize = 0;
+ theInterpretedSize = 0;
+ theFinalUpdateSize = 0;
+ theFinalReadSize = 0;
+
+ theFirstLabel = NULL;
+ theLastLabel = NULL;
+ theFirstBranch = NULL;
+ theLastBranch = NULL;
+
+ theFirstCall = NULL;
+ theLastCall = NULL;
+ theFirstSubroutine = NULL;
+ theLastSubroutine = NULL;
+
+ theNoOfLabels = 0;
+ theNoOfSubroutines = 0;
+
+ return 0;
+ } else {
+ setErrorCode(4200);
+ return -1;
+ }//if
+}//NdbOperation::interpretedDeleteTuple()
+
+
+
+/******************************************************************************
+ * int getValue(AttrInfo* tAttrInfo, char* aRef )
+ *
+ * Return Value Return 0 : GetValue was successful.
+ * Return -1: In all other case.
+ * Parameters: tAttrInfo : Attribute object of the retrieved attribute
+ * value.
+ * Remark: Define an attribute to retrieve in query.
+ *****************************************************************************/
+NdbRecAttr*
+NdbOperation::getValue(const NdbColumnImpl* tAttrInfo, char* aValue)
+{
+ NdbRecAttr* tRecAttr;
+ if ((tAttrInfo != NULL) &&
+ (!tAttrInfo->m_indexOnly) &&
+ (theStatus != Init)){
+ if (theStatus == SetBound) {
+ saveBoundATTRINFO();
+ theStatus = GetValue;
+ }
+ if (theStatus != GetValue) {
+ if (theInterpretIndicator == 1) {
+ if (theStatus == FinalGetValue) {
+ ; // Simply continue with getValue
+ } else if (theStatus == ExecInterpretedValue) {
+ if (insertATTRINFO(Interpreter::EXIT_OK_LAST) == -1)
+ return NULL;
+ theInterpretedSize = theTotalCurrAI_Len -
+ (theInitialReadSize + 5);
+ } else if (theStatus == SetValueInterpreted) {
+ theFinalUpdateSize = theTotalCurrAI_Len -
+ (theInitialReadSize + theInterpretedSize + 5);
+ } else {
+ setErrorCodeAbort(4230);
+ return NULL;
+ }//if
+ // MASV - How would execution come here?
+ theStatus = FinalGetValue;
+ } else {
+ setErrorCodeAbort(4230);
+ return NULL;
+ }//if
+ }//if
+ Uint32 ah;
+ AttributeHeader::init(&ah, tAttrInfo->m_attrId, 0);
+ if (insertATTRINFO(ah) != -1) {
+ // Insert Attribute Id into ATTRINFO part.
+
+ /************************************************************************
+ * Get a Receive Attribute object and link it into the operation object.
+ ************************************************************************/
+ tRecAttr = theNdb->getRecAttr();
+ if (tRecAttr != NULL) {
+ if (theFirstRecAttr == NULL)
+ theFirstRecAttr = tRecAttr;
+ else
+ theCurrentRecAttr->next(tRecAttr);
+ theCurrentRecAttr = tRecAttr;
+ tRecAttr->next(NULL);
+
+ /**********************************************************************
+ * Now set the attribute identity and the pointer to the data in
+ * the RecAttr object
+ * Also set attribute size, array size and attribute type
+ ********************************************************************/
+ if (tRecAttr->setup(tAttrInfo, aValue) == 0) {
+ theErrorLine++;
+ return tRecAttr;
+ } else {
+ setErrorCodeAbort(4000);
+ return NULL;
+ }
+ } else {
+ setErrorCodeAbort(4000);
+ return NULL;
+ }//if getRecAttr failure
+ } else {
+ return NULL;
+ }//if insertATTRINFO failure
+ } else {
+ if (tAttrInfo == NULL) {
+ setErrorCodeAbort(4004);
+ return NULL;
+ }//if
+ if (tAttrInfo->m_indexOnly){
+ setErrorCodeAbort(4208);
+ return NULL;
+ }//if
+ }//if
+ setErrorCodeAbort(4200);
+ return NULL;
+}
+
+/*****************************************************************************
+ * int setValue(AttrInfo* tAttrInfo, char* aValue, Uint32 len)
+ *
+ * Return Value: Return 0 : SetValue was succesful.
+ * Return -1: In all other case.
+ * Parameters: tAttrInfo : Attribute object where the attribute
+ * info exists.
+ * aValue : Reference to the variable with the new value.
+ * len : Length of the value
+ * Remark: Define a attribute to set in a query.
+******************************************************************************/
+int
+NdbOperation::setValue( const NdbColumnImpl* tAttrInfo,
+ const char* aValuePassed, Uint32 len)
+{
+ int tReturnCode;
+ Uint32 tAttrId;
+ Uint32 tData;
+ Uint32 tempData[2000];
+ OperationType tOpType = theOperationType;
+ OperationStatus tStatus = theStatus;
+
+
+ if ((tOpType == UpdateRequest) ||
+ (tOpType == WriteRequest)) {
+ if (theInterpretIndicator == 0) {
+ if (tStatus == SetValue) {
+ ;
+ } else {
+ setErrorCodeAbort(4234);
+ return -1;
+ }//if
+ } else {
+ if (tStatus == GetValue) {
+ theInitialReadSize = theTotalCurrAI_Len - 5;
+ } else if (tStatus == ExecInterpretedValue) {
+ //--------------------------------------------------------------------
+ // We insert an exit from interpretation since we are now starting
+ // to set values in the tuple by setValue.
+ //--------------------------------------------------------------------
+ if (insertATTRINFO(Interpreter::EXIT_OK_LAST) == -1){
+ return -1;
+ }
+ theInterpretedSize = theTotalCurrAI_Len -
+ (theInitialReadSize + 5);
+ } else if (tStatus == SetValueInterpreted) {
+ ; // Simply continue adding new setValue
+ } else {
+ //--------------------------------------------------------------------
+ // setValue used in the wrong context. Application coding error.
+ //-------------------------------------------------------------------
+ setErrorCodeAbort(4234); //Wrong error code
+ return -1;
+ }//if
+ theStatus = SetValueInterpreted;
+ }//if
+ } else if (tOpType == InsertRequest) {
+ if ((theStatus != SetValue) && (theStatus != OperationDefined)) {
+ setErrorCodeAbort(4234);
+ return -1;
+ }//if
+ } else if (tOpType == ReadRequest || tOpType == ReadExclusive) {
+ setErrorCodeAbort(4504);
+ return -1;
+ } else if (tOpType == DeleteRequest) {
+ setErrorCodeAbort(4504);
+ return -1;
+ } else if (tOpType == OpenScanRequest || tOpType == OpenRangeScanRequest) {
+ setErrorCodeAbort(4228);
+ return -1;
+ } else {
+ //---------------------------------------------------------------------
+ // setValue with undefined operation type.
+ // Probably application coding error.
+ //---------------------------------------------------------------------
+ setErrorCodeAbort(4108);
+ return -1;
+ }//if
+ if (tAttrInfo == NULL) {
+ setErrorCodeAbort(4004);
+ return -1;
+ }//if
+ if (tAttrInfo->m_pk) {
+ if (theOperationType == InsertRequest) {
+ return equal_impl(tAttrInfo, aValuePassed, len);
+ } else {
+ setErrorCodeAbort(4202);
+ return -1;
+ }//if
+ }//if
+ if (len > 8000) {
+ setErrorCodeAbort(4216);
+ return -1;
+ }//if
+
+ tAttrId = tAttrInfo->m_attrId;
+ const char *aValue = aValuePassed;
+ Uint32 ahValue;
+ AttributeHeader& ah = AttributeHeader::init(&ahValue, tAttrId, 0);
+ if (aValue == NULL) {
+ if (tAttrInfo->m_nullable) {
+ ah.setNULL();
+ insertATTRINFO(ahValue);
+ // Insert Attribute Id with the value
+ // NULL into ATTRINFO part.
+ return 0;
+ } else {
+ /***********************************************************************
+ * Setting a NULL value on a NOT NULL attribute is not allowed.
+ **********************************************************************/
+ setErrorCodeAbort(4203);
+ return -1;
+ }//if
+ }//if
+
+ // Insert Attribute Id into ATTRINFO part.
+ const Uint32 sizeInBytes = tAttrInfo->m_attrSize * tAttrInfo->m_arraySize;
+#if 0
+ tAttrSize = tAttrInfo->theAttrSize;
+ tArraySize = tAttrInfo->theArraySize;
+ if (tArraySize == 0) {
+ setErrorCodeAbort(4201);
+ return -1;
+ }//if
+ tAttrSizeInBits = tAttrSize*tArraySize;
+ tAttrSizeInWords = tAttrSizeInBits >> 5;
+#endif
+ const Uint32 bitsInLastWord = 8 * (sizeInBytes & 3) ;
+ if (len != sizeInBytes && (len != 0)) {
+ setErrorCodeAbort(4209);
+ return -1;
+ }//if
+ const Uint32 totalSizeInWords = (sizeInBytes + 3)/4; // Including bits in last word
+ const Uint32 sizeInWords = sizeInBytes / 4; // Excluding bits in last word
+ ah.setDataSize(totalSizeInWords);
+ insertATTRINFO( ahValue );
+
+ /***********************************************************************
+ * Check if the pointer of the value passed is aligned on a 4 byte boundary.
+ * If so only assign the pointer to the internal variable aValue.
+ * If it is not aligned then we start by copying the value to tempData and
+ * use this as aValue instead.
+ *************************************************************************/
+ const int attributeSize = sizeInBytes;
+ const int slack = sizeInBytes & 3;
+
+ if (((UintPtr)aValue & 3) != 0 || (slack != 0)){
+ memcpy(&tempData[0], aValue, attributeSize);
+ aValue = (char*)&tempData[0];
+ if(slack != 0) {
+ char * tmp = (char*)&tempData[0];
+ memset(&tmp[attributeSize], 0, (4 - slack));
+ }//if
+ }//if
+
+ tReturnCode = insertATTRINFOloop((Uint32*)aValue, sizeInWords);
+ if (tReturnCode == -1) {
+ return tReturnCode;
+ }//if
+ if (bitsInLastWord != 0) {
+ tData = *(Uint32*)(aValue + sizeInWords*4);
+ tData = convertEndian(tData);
+ tData = tData & ((1 << bitsInLastWord) - 1);
+ tData = convertEndian(tData);
+ tReturnCode = insertATTRINFO(tData);
+ if (tReturnCode == -1) {
+ return tReturnCode;
+ }//if
+ }//if
+ theErrorLine++;
+ return 0;
+}//NdbOperation::setValue()
+
+/*
+ * Define bound on index column in range scan.
+ */
+int
+NdbOperation::setBound(const NdbColumnImpl* tAttrInfo, int type, const void* aValue, Uint32 len)
+{
+ if (theOperationType == OpenRangeScanRequest &&
+ theStatus == SetBound &&
+ (0 <= type && type <= 4) &&
+ aValue != NULL &&
+ len <= 8000) {
+ // bound type
+ insertATTRINFO(type);
+ // attribute header
+ Uint32 sizeInBytes = tAttrInfo->m_attrSize * tAttrInfo->m_arraySize;
+ if (len != sizeInBytes && (len != 0)) {
+ setErrorCodeAbort(4209);
+ return -1;
+ }
+ len = sizeInBytes;
+ Uint32 tIndexAttrId = tAttrInfo->m_attrId;
+ Uint32 sizeInWords = (len + 3) / 4;
+ AttributeHeader ah(tIndexAttrId, sizeInWords);
+ insertATTRINFO(ah.m_value);
+ // attribute data
+ if ((UintPtr(aValue) & 0x3) == 0 && (len & 0x3) == 0)
+ insertATTRINFOloop((const Uint32*)aValue, sizeInWords);
+ else {
+ Uint32 temp[2000];
+ memcpy(temp, aValue, len);
+ while ((len & 0x3) != 0)
+ ((char*)temp)[len++] = 0;
+ insertATTRINFOloop(temp, sizeInWords);
+ }
+ return 0;
+ } else {
+ setErrorCodeAbort(4228); // XXX wrong code
+ return -1;
+ }
+}
+
+/****************************************************************************
+ * int insertATTRINFO( Uint32 aData );
+ *
+ * Return Value: Return 0 : insertATTRINFO was succesful.
+ * Return -1: In all other case.
+ * Parameters: aData: the data to insert into ATTRINFO.
+ * Remark: Puts the the data into either TCKEYREQ signal or
+ * ATTRINFO signal.
+ *****************************************************************************/
+int
+NdbOperation::insertATTRINFO( Uint32 aData )
+{
+ NdbApiSignal* tSignal;
+ register Uint32 tAI_LenInCurrAI = theAI_LenInCurrAI;
+ register Uint32* tAttrPtr = theATTRINFOptr;
+ register Uint32 tTotCurrAILen = theTotalCurrAI_Len;
+
+ if (tAI_LenInCurrAI >= 25) {
+ Ndb* tNdb = theNdb;
+ NdbApiSignal* tFirstAttrinfo = theFirstATTRINFO;
+ tAI_LenInCurrAI = 3;
+ tSignal = tNdb->getSignal();
+ if (tSignal != NULL) {
+ tSignal->setSignal(m_attrInfoGSN);
+ tAttrPtr = &tSignal->getDataPtrSend()[3];
+ if (tFirstAttrinfo == NULL) {
+ tSignal->next(NULL);
+ theFirstATTRINFO = tSignal;
+ theCurrentATTRINFO = tSignal;
+ } else {
+ NdbApiSignal* tCurrentAttrinfoBeforeUpdate = theCurrentATTRINFO;
+ tSignal->next(NULL);
+ theCurrentATTRINFO = tSignal;
+ tCurrentAttrinfoBeforeUpdate->next(tSignal);
+ }//if
+ } else {
+ goto insertATTRINFO_error1;
+ }//if
+ }//if
+ *tAttrPtr = aData;
+ tAttrPtr++;
+ tTotCurrAILen++;
+ tAI_LenInCurrAI++;
+ theTotalCurrAI_Len = tTotCurrAILen;
+ theAI_LenInCurrAI = tAI_LenInCurrAI;
+ theATTRINFOptr = tAttrPtr;
+ return 0;
+
+insertATTRINFO_error1:
+ setErrorCodeAbort(4000);
+ return -1;
+
+}//NdbOperation::insertATTRINFO()
+
+/*****************************************************************************
+ * int insertATTRINFOloop(Uint32* aDataPtr, Uint32 aLength );
+ *
+ * Return Value: Return 0 : insertATTRINFO was succesful.
+ * Return -1: In all other case.
+ * Parameters: aDataPtr: Pointer to the data to insert into ATTRINFO.
+ * aLength: Length of data to be copied
+ * Remark: Puts the the data into either TCKEYREQ signal or
+ * ATTRINFO signal.
+ *****************************************************************************/
+int
+NdbOperation::insertATTRINFOloop(register const Uint32* aDataPtr,
+ register Uint32 aLength)
+{
+ NdbApiSignal* tSignal;
+ register Uint32 tAI_LenInCurrAI = theAI_LenInCurrAI;
+ register Uint32 tTotCurrAILen = theTotalCurrAI_Len;
+ register Uint32* tAttrPtr = theATTRINFOptr;
+ Ndb* tNdb = theNdb;
+
+ while (aLength > 0) {
+ if (tAI_LenInCurrAI >= 25) {
+ NdbApiSignal* tFirstAttrinfo = theFirstATTRINFO;
+ tAI_LenInCurrAI = 3;
+ tSignal = tNdb->getSignal();
+ if (tSignal != NULL) {
+ tSignal->setSignal(m_attrInfoGSN);
+ tAttrPtr = &tSignal->getDataPtrSend()[3];
+ if (tFirstAttrinfo == NULL) {
+ tSignal->next(NULL);
+ theFirstATTRINFO = tSignal;
+ theCurrentATTRINFO = tSignal;
+ } else {
+ NdbApiSignal* tCurrentAttrinfoBeforeUpdate = theCurrentATTRINFO;
+ tSignal->next(NULL);
+ theCurrentATTRINFO = tSignal;
+ tCurrentAttrinfoBeforeUpdate->next(tSignal);
+ }//if
+ } else {
+ goto insertATTRINFO_error1;
+ }//if
+ }//if
+ {
+ register Uint32 tData = *aDataPtr;
+ aDataPtr++;
+ aLength--;
+ tAI_LenInCurrAI++;
+ *tAttrPtr = tData;
+ tAttrPtr++;
+ tTotCurrAILen++;
+ }
+ }//while
+ theATTRINFOptr = tAttrPtr;
+ theTotalCurrAI_Len = tTotCurrAILen;
+ theAI_LenInCurrAI = tAI_LenInCurrAI;
+ return 0;
+
+insertATTRINFO_error1:
+ setErrorCodeAbort(4000);
+ return -1;
+
+}//NdbOperation::insertATTRINFOloop()
+
+
+
+
diff --git a/ndb/src/ndbapi/NdbOperationExec.cpp b/ndb/src/ndbapi/NdbOperationExec.cpp
new file mode 100644
index 00000000000..1b0ad68b1eb
--- /dev/null
+++ b/ndb/src/ndbapi/NdbOperationExec.cpp
@@ -0,0 +1,967 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/***************************************************************************
+Name: NdbOperationExec.C
+Include:
+Link:
+Author: UABRONM Mikael Ronström UAB/M/MT Jonas Kamf UAB/M/MT
+Date: 2001-10-16
+Version: 1.2
+Description:
+Documentation:
+***************************************************************************/
+#include "NdbOperation.hpp"
+#include "NdbConnection.hpp"
+#include "NdbApiSignal.hpp"
+#include "Ndb.hpp"
+#include "NdbRecAttr.hpp"
+#include "NdbUtil.hpp"
+
+#include "Interpreter.hpp"
+#include <AttributeHeader.hpp>
+#include <signaldata/TcKeyReq.hpp>
+#include <signaldata/KeyInfo.hpp>
+#include <signaldata/AttrInfo.hpp>
+#include <signaldata/ScanTab.hpp>
+
+#include <ndb_version.h>
+
+#include "API.hpp"
+#include <NdbOut.hpp>
+
+
+/******************************************************************************
+int doSend()
+
+Return Value: Return >0 : send was succesful, returns number of signals sent
+ Return -1: In all other case.
+Parameters: aProcessorId: Receiving processor node
+Remark: Sends the ATTRINFO signal(s)
+******************************************************************************/
+int
+NdbOperation::doSendScan(int aProcessorId)
+{
+ Uint32 tSignalCount = 0;
+ NdbApiSignal* tSignal;
+
+ if (theInterpretIndicator != 1 ||
+ (theOperationType != OpenScanRequest &&
+ theOperationType != OpenRangeScanRequest)) {
+ setErrorCodeAbort(4005);
+ return -1;
+ }
+
+ assert(theSCAN_TABREQ != NULL);
+ tSignal = theSCAN_TABREQ;
+ if (tSignal->setSignal(GSN_SCAN_TABREQ) == -1) {
+ setErrorCode(4001);
+ return -1;
+ }
+ // Update the "attribute info length in words" in SCAN_TABREQ before
+ // sending it. This could not be done in openScan because
+ // we created the ATTRINFO signals after the SCAN_TABREQ signal.
+ ScanTabReq * const scanTabReq = CAST_PTR(ScanTabReq, tSignal->getDataPtrSend());
+ scanTabReq->attrLen = theTotalCurrAI_Len;
+ if (theOperationType == OpenRangeScanRequest)
+ scanTabReq->attrLen += theTotalBoundAI_Len;
+ TransporterFacade *tp = TransporterFacade::instance();
+ if (tp->sendSignal(tSignal, aProcessorId) == -1) {
+ setErrorCode(4002);
+ return -1;
+ }
+ tSignalCount++;
+
+ tSignal = theFirstSCAN_TABINFO_Send;
+ while (tSignal != NULL){
+ if (tp->sendSignal(tSignal, aProcessorId)) {
+ setErrorCode(4002);
+ return -1;
+ }
+ tSignalCount++;
+ tSignal = tSignal->next();
+ }
+
+ if (theOperationType == OpenRangeScanRequest) {
+ // must have at least one signal since it contains attrLen for bounds
+ assert(theBoundATTRINFO != NULL);
+ tSignal = theBoundATTRINFO;
+ while (tSignal != NULL) {
+ if (tp->sendSignal(tSignal,aProcessorId) == -1){
+ setErrorCode(4002);
+ return -1;
+ }
+ tSignalCount++;
+ tSignal = tSignal->next();
+ }
+ }
+
+ tSignal = theFirstATTRINFO;
+ while (tSignal != NULL) {
+ if (tp->sendSignal(tSignal,aProcessorId) == -1){
+ setErrorCode(4002);
+ return -1;
+ }
+ tSignalCount++;
+ tSignal = tSignal->next();
+ }
+ theStatus = WaitResponse;
+ return tSignalCount;
+}//NdbOperation::doSendScan()
+
+void
+NdbOperation::setLastFlag(NdbApiSignal* signal, Uint32 lastFlag)
+{
+ TcKeyReq * const req = CAST_PTR(TcKeyReq, signal->getDataPtrSend());
+ TcKeyReq::setExecuteFlag(req->requestInfo, lastFlag);
+}
+
+/******************************************************************************
+int doSend()
+
+Return Value: Return >0 : send was succesful, returns number of signals sent
+ Return -1: In all other case.
+Parameters: aProcessorId: Receiving processor node
+Remark: Sends the TCKEYREQ signal and optional KEYINFO and ATTRINFO
+ signals.
+******************************************************************************/
+int
+NdbOperation::doSend(int aNodeId, Uint32 lastFlag)
+{
+ int tReturnCode;
+ int tSignalCount = 0;
+ assert(theTCREQ != NULL);
+ setLastFlag(theTCREQ, lastFlag);
+ TransporterFacade *tp = TransporterFacade::instance();
+ tReturnCode = tp->sendSignal(theTCREQ, aNodeId);
+ tSignalCount++;
+ if (tReturnCode == -1) {
+ return -1;
+ }
+ NdbApiSignal *tSignal = theFirstKEYINFO;
+ while (tSignal != NULL) {
+ NdbApiSignal* tnextSignal = tSignal->next();
+ tReturnCode = tp->sendSignal(tSignal, aNodeId);
+ tSignal = tnextSignal;
+ if (tReturnCode == -1) {
+ return -1;
+ }
+ tSignalCount++;
+ }//while
+ tSignal = theFirstATTRINFO;
+ while (tSignal != NULL) {
+ NdbApiSignal* tnextSignal = tSignal->next();
+ tReturnCode = tp->sendSignal(tSignal, aNodeId);
+ tSignal = tnextSignal;
+ if (tReturnCode == -1) {
+ return -1;
+ }
+ tSignalCount++;
+ }//while
+ theNdbCon->OpSent();
+ return tSignalCount;
+}//NdbOperation::doSend()
+
+/***************************************************************************
+int prepareSendScan(Uint32 aTC_ConnectPtr,
+ Uint64 aTransactionId)
+
+Return Value: Return 0 : preparation of send was succesful.
+ Return -1: In all other case.
+Parameters: aTC_ConnectPtr: the Connect pointer to TC.
+ aTransactionId: the Transaction identity of the transaction.
+Remark: Puts the the final data into ATTRINFO signal(s) after this
+ we know the how many signal to send and their sizes
+***************************************************************************/
+int NdbOperation::prepareSendScan(Uint32 aTC_ConnectPtr,
+ Uint64 aTransactionId){
+
+ if (theInterpretIndicator != 1 ||
+ (theOperationType != OpenScanRequest &&
+ theOperationType != OpenRangeScanRequest)) {
+ setErrorCodeAbort(4005);
+ return -1;
+ }
+
+ if (theStatus == SetBound) {
+ saveBoundATTRINFO();
+ theStatus = GetValue;
+ }
+
+ theErrorLine = 0;
+
+ // In preapareSendInterpreted we set the sizes (word 4-8) in the
+ // first ATTRINFO signal.
+ if (prepareSendInterpreted() == -1)
+ return -1;
+
+ const Uint32 transId1 = (Uint32) (aTransactionId & 0xFFFFFFFF);
+ const Uint32 transId2 = (Uint32) (aTransactionId >> 32);
+
+ if (theOperationType == OpenRangeScanRequest) {
+ NdbApiSignal* tSignal = theBoundATTRINFO;
+ do{
+ tSignal->setData(aTC_ConnectPtr, 1);
+ tSignal->setData(transId1, 2);
+ tSignal->setData(transId2, 3);
+ tSignal = tSignal->next();
+ } while (tSignal != NULL);
+ }
+ theCurrentATTRINFO->setLength(theAI_LenInCurrAI);
+ NdbApiSignal* tSignal = theFirstATTRINFO;
+ do{
+ tSignal->setData(aTC_ConnectPtr, 1);
+ tSignal->setData(transId1, 2);
+ tSignal->setData(transId2, 3);
+ tSignal = tSignal->next();
+ } while (tSignal != NULL);
+ return 0;
+}
+
+/***************************************************************************
+int prepareSend(Uint32 aTC_ConnectPtr,
+ Uint64 aTransactionId)
+
+Return Value: Return 0 : preparation of send was succesful.
+ Return -1: In all other case.
+Parameters: aTC_ConnectPtr: the Connect pointer to TC.
+ aTransactionId: the Transaction identity of the transaction.
+Remark: Puts the the data into TCKEYREQ signal and optional KEYINFO and ATTRINFO signals.
+***************************************************************************/
+int
+NdbOperation::prepareSend(Uint32 aTC_ConnectPtr, Uint64 aTransId)
+{
+ Uint32 tTransId1, tTransId2;
+ Uint32 tReqInfo;
+ Uint32 tInterpretInd = theInterpretIndicator;
+
+ theErrorLine = 0;
+
+ if (tInterpretInd != 1) {
+ OperationType tOpType = theOperationType;
+ OperationStatus tStatus = theStatus;
+ if ((tOpType == UpdateRequest) ||
+ (tOpType == InsertRequest) ||
+ (tOpType == WriteRequest)) {
+ if (tStatus != SetValue) {
+ setErrorCodeAbort(4116);
+ return -1;
+ }//if
+ } else if ((tOpType == ReadRequest) || (tOpType == ReadExclusive) ||
+ (tOpType == DeleteRequest)) {
+ if (tStatus != GetValue) {
+ setErrorCodeAbort(4116);
+ return -1;
+ }//if
+ } else {
+ setErrorCodeAbort(4005);
+ return -1;
+ }//if
+ } else {
+ if (prepareSendInterpreted() == -1) {
+ return -1;
+ }//if
+ }//if
+
+//-------------------------------------------------------------
+// We start by filling in the first 9 unconditional words of the
+// TCKEYREQ signal.
+//-------------------------------------------------------------
+ TcKeyReq * const tcKeyReq = CAST_PTR(TcKeyReq, theTCREQ->getDataPtrSend());
+
+ Uint32 tTotalCurrAI_Len = theTotalCurrAI_Len;
+ Uint32 tTableId = m_currentTable->m_tableId;
+ Uint32 tSchemaVersion = m_currentTable->m_version;
+
+ tcKeyReq->apiConnectPtr = aTC_ConnectPtr;
+ tcKeyReq->apiOperationPtr = ptr2int();
+ // Check if too much attrinfo have been defined
+ if (tTotalCurrAI_Len > TcKeyReq::MaxTotalAttrInfo){
+ setErrorCodeAbort(4257);
+ return -1;
+ }
+ Uint32 TattrLen = 0;
+ tcKeyReq->setAttrinfoLen(TattrLen, tTotalCurrAI_Len);
+ tcKeyReq->setAPIVersion(TattrLen, NDB_VERSION);
+ tcKeyReq->attrLen = TattrLen;
+
+ tcKeyReq->tableId = tTableId;
+ tcKeyReq->tableSchemaVersion = tSchemaVersion;
+ tTransId1 = (Uint32) aTransId;
+ tTransId2 = (Uint32) (aTransId >> 32);
+
+//-------------------------------------------------------------
+// Simple is simple if simple or both start and commit is set.
+//-------------------------------------------------------------
+// Temporarily disable simple stuff
+ Uint8 tSimpleIndicator = 0;
+// Uint8 tSimpleIndicator = theSimpleIndicator;
+ Uint8 tCommitIndicator = theCommitIndicator;
+ Uint8 tStartIndicator = theStartIndicator;
+// if ((theNdbCon->theLastOpInList == this) && (theCommitIndicator == 0))
+// abort();
+// Temporarily disable simple stuff
+ Uint8 tSimpleAlt = 0;
+// Uint8 tSimpleAlt = tStartIndicator & tCommitIndicator;
+ tSimpleIndicator = tSimpleIndicator | tSimpleAlt;
+
+//-------------------------------------------------------------
+// Simple state is set if start and commit is set and it is
+// a read request. Otherwise it is set to zero.
+//-------------------------------------------------------------
+ Uint8 tReadInd = (theOperationType == ReadRequest);
+ Uint8 tSimpleState = tReadInd & tSimpleAlt;
+ theNdbCon->theSimpleState = tSimpleState;
+
+ tcKeyReq->transId1 = tTransId1;
+ tcKeyReq->transId2 = tTransId2;
+
+ tReqInfo = 0;
+ if (tTotalCurrAI_Len <= TcKeyReq::MaxAttrInfo) {
+ tcKeyReq->setAIInTcKeyReq(tReqInfo, tTotalCurrAI_Len);
+ } else {
+ tcKeyReq->setAIInTcKeyReq(tReqInfo, TcKeyReq::MaxAttrInfo);
+ }//if
+
+ tcKeyReq->setSimpleFlag(tReqInfo, tSimpleIndicator);
+ tcKeyReq->setCommitFlag(tReqInfo, tCommitIndicator);
+ tcKeyReq->setStartFlag(tReqInfo, tStartIndicator);
+ const Uint8 tInterpretIndicator = theInterpretIndicator;
+ tcKeyReq->setInterpretedFlag(tReqInfo, tInterpretIndicator);
+
+ Uint8 tDirtyIndicator = theDirtyIndicator;
+ OperationType tOperationType = theOperationType;
+ Uint32 tTupKeyLen = theTupKeyLen;
+ Uint8 abortOption = theNdbCon->m_abortOption;
+
+ tcKeyReq->setDirtyFlag(tReqInfo, tDirtyIndicator);
+ tcKeyReq->setOperationType(tReqInfo, tOperationType);
+ tcKeyReq->setKeyLength(tReqInfo, tTupKeyLen);
+ tcKeyReq->setAbortOption(tReqInfo, abortOption);
+
+ Uint8 tDistrKeyIndicator = theDistrKeyIndicator;
+ Uint8 tDistrGroupIndicator = theDistrGroupIndicator;
+ Uint8 tDistrGroupType = theDistrGroupType;
+ Uint8 tScanIndicator = theScanInfo & 1;
+
+ tcKeyReq->setDistributionGroupFlag(tReqInfo, tDistrGroupIndicator);
+ tcKeyReq->setDistributionGroupTypeFlag(tReqInfo, tDistrGroupType);
+ tcKeyReq->setDistributionKeyFlag(tReqInfo, tDistrKeyIndicator);
+ tcKeyReq->setScanIndFlag(tReqInfo, tScanIndicator);
+
+ tcKeyReq->requestInfo = tReqInfo;
+
+//-------------------------------------------------------------
+// The next step is to fill in the upto three conditional words.
+//-------------------------------------------------------------
+ Uint32* tOptionalDataPtr = &tcKeyReq->scanInfo;
+ Uint32 tDistrGHIndex = tScanIndicator;
+ Uint32 tDistrKeyIndex = tDistrGHIndex + tDistrGroupIndicator;
+
+ Uint32 tScanInfo = theScanInfo;
+ Uint32 tDistributionGroup = theDistributionGroup;
+ Uint32 tDistrKeySize = theDistrKeySize;
+
+ tOptionalDataPtr[0] = tScanInfo;
+ tOptionalDataPtr[tDistrGHIndex] = tDistributionGroup;
+ tOptionalDataPtr[tDistrKeyIndex] = tDistrKeySize;
+
+//-------------------------------------------------------------
+// The next is step is to compress the key data part of the
+// TCKEYREQ signal.
+//-------------------------------------------------------------
+ Uint32 tKeyIndex = tDistrKeyIndex + tDistrKeyIndicator;
+ Uint32* tKeyDataPtr = &tOptionalDataPtr[tKeyIndex];
+ Uint32 Tdata1 = tcKeyReq->keyInfo[0];
+ Uint32 Tdata2 = tcKeyReq->keyInfo[1];
+ Uint32 Tdata3 = tcKeyReq->keyInfo[2];
+ Uint32 Tdata4 = tcKeyReq->keyInfo[3];
+ Uint32 Tdata5;
+
+ tKeyDataPtr[0] = Tdata1;
+ tKeyDataPtr[1] = Tdata2;
+ tKeyDataPtr[2] = Tdata3;
+ tKeyDataPtr[3] = Tdata4;
+ if (tTupKeyLen > 4) {
+ Tdata1 = tcKeyReq->keyInfo[4];
+ Tdata2 = tcKeyReq->keyInfo[5];
+ Tdata3 = tcKeyReq->keyInfo[6];
+ Tdata4 = tcKeyReq->keyInfo[7];
+
+ tKeyDataPtr[4] = Tdata1;
+ tKeyDataPtr[5] = Tdata2;
+ tKeyDataPtr[6] = Tdata3;
+ tKeyDataPtr[7] = Tdata4;
+ }//if
+//-------------------------------------------------------------
+// Finally we also compress the ATTRINFO part of the signal.
+// We optimise by using the if-statement for sending KEYINFO
+// signals to calculating the new Attrinfo Index.
+//-------------------------------------------------------------
+ Uint32 tAttrInfoIndex;
+
+ if (tTupKeyLen > TcKeyReq::MaxKeyInfo) {
+ /**
+ * Set transid, TC connect ptr and length in the KEYINFO signals
+ */
+ NdbApiSignal* tSignal = theFirstKEYINFO;
+ Uint32 remainingKey = tTupKeyLen - TcKeyReq::MaxKeyInfo;
+ do {
+ Uint32* tSigDataPtr = tSignal->getDataPtrSend();
+ NdbApiSignal* tnextSignal = tSignal->next();
+ tSigDataPtr[0] = aTC_ConnectPtr;
+ tSigDataPtr[1] = tTransId1;
+ tSigDataPtr[2] = tTransId2;
+ if (remainingKey > KeyInfo::DataLength) {
+ // The signal is full
+ tSignal->setLength(KeyInfo::MaxSignalLength);
+ remainingKey -= KeyInfo::DataLength;
+ }
+ else {
+ // Last signal
+ tSignal->setLength(KeyInfo::HeaderLength + remainingKey);
+ remainingKey = 0;
+ }
+ tSignal = tnextSignal;
+ } while (tSignal != NULL);
+ tAttrInfoIndex = tKeyIndex + TcKeyReq::MaxKeyInfo;
+ } else {
+ tAttrInfoIndex = tKeyIndex + tTupKeyLen;
+ }//if
+
+//-------------------------------------------------------------
+// Perform the Attrinfo packing in the TCKEYREQ signal started
+// above.
+//-------------------------------------------------------------
+ Uint32* tAIDataPtr = &tOptionalDataPtr[tAttrInfoIndex];
+ Tdata1 = tcKeyReq->attrInfo[0];
+ Tdata2 = tcKeyReq->attrInfo[1];
+ Tdata3 = tcKeyReq->attrInfo[2];
+ Tdata4 = tcKeyReq->attrInfo[3];
+ Tdata5 = tcKeyReq->attrInfo[4];
+
+ theTCREQ->setLength(tcKeyReq->getAIInTcKeyReq(tReqInfo) +
+ tAttrInfoIndex + TcKeyReq::StaticLength);
+ tAIDataPtr[0] = Tdata1;
+ tAIDataPtr[1] = Tdata2;
+ tAIDataPtr[2] = Tdata3;
+ tAIDataPtr[3] = Tdata4;
+ tAIDataPtr[4] = Tdata5;
+
+/***************************************************
+* Send the ATTRINFO signals.
+***************************************************/
+ if (tTotalCurrAI_Len > 5) {
+ // Set the last signal's length.
+ NdbApiSignal* tSignal = theFirstATTRINFO;
+ theCurrentATTRINFO->setLength(theAI_LenInCurrAI);
+ do {
+ Uint32* tSigDataPtr = tSignal->getDataPtrSend();
+ NdbApiSignal* tnextSignal = tSignal->next();
+ tSigDataPtr[0] = aTC_ConnectPtr;
+ tSigDataPtr[1] = tTransId1;
+ tSigDataPtr[2] = tTransId2;
+ tSignal = tnextSignal;
+ } while (tSignal != NULL);
+ }//if
+ NdbRecAttr* tRecAttrObject = theFirstRecAttr;
+ theStatus = WaitResponse;
+ theCurrentRecAttr = tRecAttrObject;
+ return 0;
+}//NdbOperation::prepareSend()
+
+/***************************************************************************
+int prepareSendInterpreted()
+
+Make preparations to send an interpreted operation.
+Return Value: Return 0 : succesful.
+ Return -1: In all other case.
+***************************************************************************/
+int
+NdbOperation::prepareSendInterpreted()
+{
+ Uint32 tTotalCurrAI_Len = theTotalCurrAI_Len;
+ Uint32 tInitReadSize = theInitialReadSize;
+ if (theStatus == ExecInterpretedValue) {
+ if (insertATTRINFO(Interpreter::EXIT_OK_LAST) != -1) {
+//-------------------------------------------------------------------------
+// Since we read the total length before inserting the last entry in the
+// signals we need to add one to the total length.
+//-------------------------------------------------------------------------
+
+ theInterpretedSize = (tTotalCurrAI_Len + 1) -
+ (tInitReadSize + 5);
+
+ } else {
+ return -1;
+ }//if
+ } else if (theStatus == FinalGetValue) {
+
+ theFinalReadSize = tTotalCurrAI_Len -
+ (tInitReadSize + theInterpretedSize + theFinalUpdateSize + 5);
+
+ } else if (theStatus == SetValueInterpreted) {
+
+ theFinalUpdateSize = tTotalCurrAI_Len -
+ (tInitReadSize + theInterpretedSize + 5);
+
+ } else if (theStatus == SubroutineEnd) {
+
+ theSubroutineSize = tTotalCurrAI_Len -
+ (tInitReadSize + theInterpretedSize +
+ theFinalUpdateSize + theFinalReadSize + 5);
+
+ } else if (theStatus == GetValue) {
+ theInitialReadSize = tTotalCurrAI_Len - 5;
+ } else {
+ setErrorCodeAbort(4116);
+ return -1;
+ }
+
+ while (theFirstBranch != NULL) {
+ Uint32 tRelAddress;
+ Uint32 tLabelAddress = 0;
+ int tAddress = -1;
+ NdbBranch* tNdbBranch = theFirstBranch;
+ Uint32 tBranchLabel = tNdbBranch->theBranchLabel;
+ NdbLabel* tNdbLabel = theFirstLabel;
+ if (tBranchLabel >= theNoOfLabels) {
+ setErrorCodeAbort(4221);
+ return -1;
+ }//if
+
+ // Find the label address
+ while (tNdbLabel != NULL) {
+ for(tLabelAddress = 0; tLabelAddress<16; tLabelAddress++){
+ const Uint32 labelNo = tNdbLabel->theLabelNo[tLabelAddress];
+ if(tBranchLabel == labelNo){
+ tAddress = tNdbLabel->theLabelAddress[tLabelAddress];
+ break;
+ }
+ }
+
+ if(tAddress != -1)
+ break;
+ tNdbLabel = tNdbLabel->theNext;
+ }//while
+ if (tAddress == -1) {
+//-------------------------------------------------------------------------
+// We were unable to find any label which the branch refers to. This means
+// that the application have not programmed the interpreter program correctly.
+//-------------------------------------------------------------------------
+ setErrorCodeAbort(4222);
+ return -1;
+ }//if
+ if (tNdbLabel->theSubroutine[tLabelAddress] != tNdbBranch->theSubroutine) {
+ setErrorCodeAbort(4224);
+ return -1;
+ }//if
+ // Now it is time to update the signal data with the relative branch jump.
+ if (tAddress < int(tNdbBranch->theBranchAddress)) {
+ tRelAddress = (tNdbBranch->theBranchAddress - tAddress) << 16;
+
+ // Indicate backward jump direction
+ tRelAddress = tRelAddress + (1 << 31);
+
+ } else if (tAddress > int(tNdbBranch->theBranchAddress)) {
+ tRelAddress = (tAddress - tNdbBranch->theBranchAddress) << 16;
+ } else {
+ setErrorCodeAbort(4223);
+ return -1;
+ }//if
+ NdbApiSignal* tSignal = tNdbBranch->theSignal;
+ Uint32 tReadData = tSignal->readData(tNdbBranch->theSignalAddress);
+ tSignal->setData((tRelAddress + tReadData), tNdbBranch->theSignalAddress);
+
+ theFirstBranch = theFirstBranch->theNext;
+ theNdb->releaseNdbBranch(tNdbBranch);
+ }//while
+
+ while (theFirstCall != NULL) {
+ Uint32 tSubroutineCount = 0;
+ int tAddress = -1;
+ NdbSubroutine* tNdbSubroutine;
+ NdbCall* tNdbCall = theFirstCall;
+ if (tNdbCall->theSubroutine >= theNoOfSubroutines) {
+ setErrorCodeAbort(4221);
+ return -1;
+ }//if
+// Find the subroutine address
+ tNdbSubroutine = theFirstSubroutine;
+ while (tNdbSubroutine != NULL) {
+ tSubroutineCount += 16;
+ if (tNdbCall->theSubroutine < tSubroutineCount) {
+// Subroutine Found
+ Uint32 tSubroutineAddress = tNdbCall->theSubroutine - (tSubroutineCount - 16);
+ tAddress = tNdbSubroutine->theSubroutineAddress[tSubroutineAddress];
+ break;
+ }//if
+ tNdbSubroutine = tNdbSubroutine->theNext;
+ }//while
+ if (tAddress == -1) {
+ setErrorCodeAbort(4222);
+ return -1;
+ }//if
+// Now it is time to update the signal data with the relative branch jump.
+ NdbApiSignal* tSignal = tNdbCall->theSignal;
+ Uint32 tReadData = tSignal->readData(tNdbCall->theSignalAddress);
+ tSignal->setData(((tAddress << 16) + tReadData), tNdbCall->theSignalAddress);
+
+ theFirstCall = theFirstCall->theNext;
+ theNdb->releaseNdbCall(tNdbCall);
+ }//while
+
+ Uint32 tInitialReadSize = theInitialReadSize;
+ Uint32 tInterpretedSize = theInterpretedSize;
+ Uint32 tFinalUpdateSize = theFinalUpdateSize;
+ Uint32 tFinalReadSize = theFinalReadSize;
+ Uint32 tSubroutineSize = theSubroutineSize;
+ if (theOperationType != OpenScanRequest &&
+ theOperationType != OpenRangeScanRequest) {
+ TcKeyReq * const tcKeyReq = CAST_PTR(TcKeyReq, theTCREQ->getDataPtrSend());
+
+ tcKeyReq->attrInfo[0] = tInitialReadSize;
+ tcKeyReq->attrInfo[1] = tInterpretedSize;
+ tcKeyReq->attrInfo[2] = tFinalUpdateSize;
+ tcKeyReq->attrInfo[3] = tFinalReadSize;
+ tcKeyReq->attrInfo[4] = tSubroutineSize;
+ } else {
+ // If a scan is defined we use the first ATTRINFO instead of TCKEYREQ.
+ theFirstATTRINFO->setData(tInitialReadSize, 4);
+ theFirstATTRINFO->setData(tInterpretedSize, 5);
+ theFirstATTRINFO->setData(tFinalUpdateSize, 6);
+ theFirstATTRINFO->setData(tFinalReadSize, 7);
+ theFirstATTRINFO->setData(tSubroutineSize, 8);
+ }//if
+ return 0;
+}//NdbOperation::prepareSendInterpreted()
+
+/***************************************************************************
+int TCOPCONF(int anAttrInfoLen)
+
+Return Value: Return 0 : send was succesful.
+ Return -1: In all other case.
+Parameters: anAttrInfoLen: The length of the attribute information from TC.
+Remark: Handles the reception of the TC[KEY/INDX]CONF signal.
+***************************************************************************/
+void
+NdbOperation::TCOPCONF(Uint32 anAttrInfoLen)
+{
+ Uint32 tCurrRecLen = theCurrRecAI_Len;
+ if (theStatus == WaitResponse) {
+ theTotalRecAI_Len = anAttrInfoLen;
+ if (anAttrInfoLen == tCurrRecLen) {
+ Uint32 tAI_ElemLen = theAI_ElementLen;
+ NdbRecAttr* tCurrRecAttr = theCurrentRecAttr;
+ theStatus = Finished;
+
+ if ((tAI_ElemLen == 0) &&
+ (tCurrRecAttr == NULL)) {
+ NdbRecAttr* tRecAttr = theFirstRecAttr;
+ while (tRecAttr != NULL) {
+ if (tRecAttr->copyoutRequired()) // copy to application buffer
+ tRecAttr->copyout();
+ tRecAttr = tRecAttr->next();
+ }
+ theNdbCon->OpCompleteSuccess();
+ return;
+ } else if (tAI_ElemLen != 0) {
+ setErrorCode(4213);
+ theNdbCon->OpCompleteFailure();
+ return;
+ } else {
+ setErrorCode(4214);
+ theNdbCon->OpCompleteFailure();
+ return;
+ }//if
+ } else if (anAttrInfoLen > tCurrRecLen) {
+ return;
+ } else {
+ theStatus = Finished;
+
+ if (theAI_ElementLen != 0) {
+ setErrorCode(4213);
+ theNdbCon->OpCompleteFailure();
+ return;
+ }//if
+ if (theCurrentRecAttr != NULL) {
+ setErrorCode(4214);
+ theNdbCon->OpCompleteFailure();
+ return;
+ }//if
+ theNdbCon->OpCompleteFailure();
+ return;
+ }//if
+ } else {
+ setErrorCode(4004);
+ }//if
+ return;
+}//NdbOperation::TCKEYOPCONF()
+
+int
+NdbOperation::checkState_TransId(NdbApiSignal* aSignal)
+{
+ Uint64 tRecTransId, tCurrTransId;
+ Uint32 tTmp1, tTmp2;
+
+ if (theStatus != WaitResponse) {
+#ifdef NDB_NO_DROPPED_SIGNAL
+ abort();
+#endif
+ return -1;
+ }//if
+
+ tTmp1 = aSignal->readData(2);
+ tTmp2 = aSignal->readData(3);
+
+ tRecTransId = (Uint64)tTmp1 + ((Uint64)tTmp2 << 32);
+ tCurrTransId = theNdbCon->getTransactionId();
+ if (tCurrTransId != tRecTransId) {
+#ifdef NDB_NO_DROPPED_SIGNAL
+ abort();
+#endif
+ return -1;
+ }//if
+ return 0;
+}//NdbOperation::checkState_TransId()
+
+/***************************************************************************
+int receiveTCKEYREF( NdbApiSignal* aSignal)
+
+Return Value: Return 0 : send was succesful.
+ Return -1: In all other case.
+Parameters: aSignal: the signal object that contains the TCKEYREF signal from TC.
+Remark: Handles the reception of the TCKEYREF signal.
+***************************************************************************/
+int
+NdbOperation::receiveTCKEYREF( NdbApiSignal* aSignal)
+{
+ if (checkState_TransId(aSignal) == -1) {
+ return -1;
+ }//if
+
+ theStatus = Finished;
+
+ theNdbCon->theReturnStatus = ReturnFailure;
+ //-------------------------------------------------------------------------//
+ // If the transaction this operation belongs to consists only of simple reads
+ // we set the error code on the transaction object.
+ // If the transaction consists of other types of operations we set
+ // the error code only on the operation since the simple read is not really
+ // part of this transaction and we can not decide the status of the whole
+ // transaction based on this operation.
+ //-------------------------------------------------------------------------//
+ if (theNdbCon->theSimpleState == 0) {
+ theError.code = aSignal->readData(4);
+ theNdbCon->setOperationErrorCodeAbort(aSignal->readData(4));
+ return theNdbCon->OpCompleteFailure();
+ } else {
+ theError.code = aSignal->readData(4);
+ return theNdbCon->OpCompleteSuccess();
+ }
+
+}//NdbOperation::receiveTCKEYREF()
+
+/***************************************************************************
+int receiveREAD_CONF( NdbApiSignal* aSignal)
+
+Return Value: Return 0 : send was succesful.
+ Return -1: In all other case.
+Parameters: aSignal: the signal object that contains the READCONF signal from TUP.
+Remark: Handles the reception of the READCONF signal.
+***************************************************************************/
+int
+NdbOperation::receiveREAD_CONF(const Uint32* aDataPtr, Uint32 aDataLength)
+{
+ Uint64 tRecTransId, tCurrTransId;
+ Uint32 tCondFlag = (Uint32)(theStatus - WaitResponse);
+ Uint32 tTotLen = aDataPtr[3];
+
+ tRecTransId = (Uint64)aDataPtr[1] + ((Uint64)aDataPtr[2] << 32);
+ tCurrTransId = theNdbCon->getTransactionId();
+ tCondFlag |= (Uint32)((tRecTransId - tCurrTransId) != (Uint64)0);
+ tCondFlag |= (Uint32)(aDataLength < 4);
+
+ if (tCondFlag == 0) {
+ theTotalRecAI_Len = tTotLen;
+ int tRetValue = receiveREAD_AI((Uint32*)&aDataPtr[4], (aDataLength - 4));
+ if (theStatus == Finished) {
+ return tRetValue;
+ } else {
+ theStatus = Finished;
+ return theNdbCon->OpCompleteFailure();
+ }//if
+ }//if
+#ifdef NDB_NO_DROPPED_SIGNAL
+ abort();
+#endif
+ return -1;
+}//NdbOperation::receiveREAD_CONF()
+
+/***************************************************************************
+int receiveTRANSID_AI( NdbApiSignal* aSignal)
+
+Return Value: Return 0 : send was succesful.
+ Return -1: In all other case.
+Parameters: aSignal: the signal object that contains the TRANSID_AI signal.
+Remark: Handles the reception of the TRANSID_AI signal.
+***************************************************************************/
+int
+NdbOperation::receiveTRANSID_AI(const Uint32* aDataPtr, Uint32 aDataLength)
+{
+ Uint64 tRecTransId, tCurrTransId;
+ Uint32 tCondFlag = (Uint32)(theStatus - WaitResponse);
+
+ tRecTransId = (Uint64)aDataPtr[1] + ((Uint64)aDataPtr[2] << 32);
+ tCurrTransId = theNdbCon->getTransactionId();
+ tCondFlag |= (Uint32)((tRecTransId - tCurrTransId) != (Uint64)0);
+ tCondFlag |= (Uint32)(aDataLength < 3);
+
+ if (tCondFlag == 0) {
+ return receiveREAD_AI((Uint32*)&aDataPtr[3], (aDataLength - 3));
+ }//if
+#ifdef NDB_NO_DROPPED_SIGNAL
+ abort();
+#endif
+ return -1;
+}//NdbOperation::receiveTRANSID_AI()
+
+/***************************************************************************
+int receiveREAD_AI( NdbApiSignal* aSignal, int aLength, int aStartPos)
+
+Return Value: Return 0 : send was succesoccurredful.
+ Return -1: In all other case.
+Parameters: aSignal: the signal object that contains the LEN_ATTRINFO11 signal.
+ aLength:
+ aStartPos:
+Remark: Handles the reception of the LEN_ATTRINFO11 signal.
+***************************************************************************/
+int
+NdbOperation::receiveREAD_AI(Uint32* aDataPtr, Uint32 aLength)
+{
+
+ register Uint32 tAI_ElementLen = theAI_ElementLen;
+ register Uint32* tCurrElemPtr = theCurrElemPtr;
+ if (theError.code == 0) {
+ // If inconsistency error occurred we will still continue
+ // receiving signals since we need to know whether commit
+ // has occurred.
+
+ register Uint32 tData;
+ for (register Uint32 i = 0; i < aLength ; i++, aDataPtr++)
+ {
+ // Code to receive Attribute Information
+ tData = *aDataPtr;
+ if (tAI_ElementLen != 0) {
+ tAI_ElementLen--;
+ *tCurrElemPtr = tData;
+ tCurrElemPtr++;
+ continue;
+ } else {
+ // Waiting for a new attribute element
+ NdbRecAttr* tWorkingRecAttr;
+
+ tWorkingRecAttr = theCurrentRecAttr;
+ AttributeHeader ah(tData);
+ const Uint32 tAttrId = ah.getAttributeId();
+ const Uint32 tAttrSize = ah.getDataSize();
+ if ((tWorkingRecAttr != NULL) &&
+ (tWorkingRecAttr->attrId() == tAttrId)) {
+ ;
+ } else {
+ setErrorCode(4211);
+ break;
+ }//if
+ theCurrentRecAttr = tWorkingRecAttr->next();
+ NdbColumnImpl * col = m_currentTable->getColumn(tAttrId);
+ if (ah.isNULL()) {
+ // Return a Null value from the NDB to the attribute.
+ if(col != 0 && col->m_nullable) {
+ tWorkingRecAttr->setNULL();
+ tAI_ElementLen = 0;
+ } else {
+ setErrorCode(4212);
+ break;
+ }//if
+ } else {
+ // Return a value from the NDB to the attribute.
+ tWorkingRecAttr->setNotNULL();
+ const Uint32 sizeInBytes = col->m_attrSize * col->m_arraySize;
+ const Uint32 sizeInWords = (sizeInBytes + 3) / 4;
+ tAI_ElementLen = tAttrSize;
+ tCurrElemPtr = (Uint32*)tWorkingRecAttr->aRef();
+ if (sizeInWords == tAttrSize){
+ continue;
+ } else {
+ setErrorCode(4201);
+ break;
+ }//if
+ }//if
+ }//if
+ }//for
+ }//if
+ Uint32 tCurrRecLen = theCurrRecAI_Len;
+ Uint32 tTotRecLen = theTotalRecAI_Len;
+ theAI_ElementLen = tAI_ElementLen;
+ theCurrElemPtr = tCurrElemPtr;
+ tCurrRecLen = tCurrRecLen + aLength;
+ theCurrRecAI_Len = tCurrRecLen; // Update Current Received AI Length
+ if (tTotRecLen == tCurrRecLen){ // Operation completed
+ NdbRecAttr* tCurrRecAttr = theCurrentRecAttr;
+ theStatus = Finished;
+
+ NdbConnection* tNdbCon = theNdbCon;
+ if ((tAI_ElementLen == 0) &&
+ (tCurrRecAttr == NULL)) {
+ NdbRecAttr* tRecAttr = theFirstRecAttr;
+ while (tRecAttr != NULL) {
+ if (tRecAttr->copyoutRequired()) // copy to application buffer
+ tRecAttr->copyout();
+ tRecAttr = tRecAttr->next();
+ }
+ return tNdbCon->OpCompleteSuccess();
+ } else if (tAI_ElementLen != 0) {
+ setErrorCode(4213);
+ return tNdbCon->OpCompleteFailure();
+ } else {
+ setErrorCode(4214);
+ return tNdbCon->OpCompleteFailure();
+ }//if
+ }
+ else if ((tCurrRecLen > tTotRecLen) &&
+ (tTotRecLen > 0)) { /* == 0 if TCKEYCONF not yet received */
+ setErrorCode(4215);
+ theStatus = Finished;
+
+ return theNdbCon->OpCompleteFailure();
+ }//if
+ return -1; // Continue waiting for more signals of this operation
+}//NdbOperation::receiveREAD_AI()
+
+void
+NdbOperation::handleFailedAI_ElemLen()
+{
+ NdbRecAttr* tRecAttr = theFirstRecAttr;
+ while (tRecAttr != NULL) {
+ tRecAttr->setUNDEFINED();
+ tRecAttr = tRecAttr->next();
+ }//while
+}//NdbOperation::handleFailedAI_ElemLen()
+
+
+
+
diff --git a/ndb/src/ndbapi/NdbOperationInt.cpp b/ndb/src/ndbapi/NdbOperationInt.cpp
new file mode 100644
index 00000000000..be23a1c274c
--- /dev/null
+++ b/ndb/src/ndbapi/NdbOperationInt.cpp
@@ -0,0 +1,1130 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/************************************************************************************************
+Name: NdbOperationInt.C
+Include:
+Link:
+Author: UABRONM Mikael Ronström UAB/M/MT
+Date: 991029
+Version: 0.1
+Description: Interpreted operations in NDB API
+Documentation:
+Adjust: 991029 UABRONM First version.
+************************************************************************************************/
+#include "NdbOperation.hpp"
+#include "NdbApiSignal.hpp"
+#include "NdbConnection.hpp"
+#include "Ndb.hpp"
+#include "NdbRecAttr.hpp"
+#include "AttrType.hpp"
+#include "NdbUtil.hpp"
+#include "Interpreter.hpp"
+
+#ifdef VM_TRACE
+#include <NdbEnv.h>
+#define INT_DEBUG(x) \
+ { const char* tmp = NdbEnv_GetEnv("INT_DEBUG", (char*)0, 0); \
+ if (tmp != 0 && strlen(tmp) != 0) { ndbout << "INT:"; ndbout_c x; } }
+#else
+#define INT_DEBUG(x)
+#endif
+
+int
+NdbOperation::incCheck(const NdbColumnImpl* tNdbColumnImpl)
+{
+ if ((theInterpretIndicator == 1)) {
+ if ((tNdbColumnImpl == NULL) ||
+ (theOperationType == OpenScanRequest ||
+ theOperationType == OpenRangeScanRequest))
+ goto inc_check_error1;
+ if ((tNdbColumnImpl->getInterpretableType() != true) ||
+ (tNdbColumnImpl->m_pk != false) ||
+ (tNdbColumnImpl->m_nullable))
+ goto inc_check_error2;
+ if (theStatus == ExecInterpretedValue) {
+ ; // Simply continue with interpretation
+ } else if (theStatus == GetValue) {
+ theInitialReadSize = theTotalCurrAI_Len - 5;
+ theStatus = ExecInterpretedValue;
+ } else if (theStatus == SubroutineExec) {
+ ; // Simply continue with interpretation
+ } else {
+ setErrorCodeAbort(4231);
+ return -1;
+ }
+ return tNdbColumnImpl->m_attrId;
+ } else {
+ if (theNdbCon->theCommitStatus == Started)
+ setErrorCodeAbort(4200);
+ }
+ return -1;
+
+ inc_check_error1:
+ if (theOperationType == OpenScanRequest ||
+ theOperationType == OpenRangeScanRequest) {
+ setErrorCodeAbort(4228);
+ return -1;
+ }
+ setErrorCodeAbort(4004);
+ return -1;
+
+ inc_check_error2:
+ if (tNdbColumnImpl->m_pk){
+ setErrorCodeAbort(4202);
+ return -1;
+ }//if
+ if (!tNdbColumnImpl->getInterpretableType()){
+ setErrorCodeAbort(4217);
+ return -1;
+ }//if
+ if (tNdbColumnImpl->m_nullable){
+ setErrorCodeAbort(4218);
+ return -1;
+ }//if
+ setErrorCodeAbort(4219);
+ return -1;
+}
+
+int
+NdbOperation::write_attrCheck(const NdbColumnImpl* tNdbColumnImpl)
+{
+ if ((theInterpretIndicator == 1)) {
+ if ((tNdbColumnImpl == NULL) ||
+ (theOperationType == OpenScanRequest ||
+ theOperationType == OpenRangeScanRequest))
+ goto write_attr_check_error1;
+ if ((tNdbColumnImpl->getInterpretableType() == false) ||
+ (tNdbColumnImpl->m_pk))
+ goto write_attr_check_error2;
+ if (theStatus == ExecInterpretedValue) {
+ ; // Simply continue with interpretation
+ } else if (theStatus == SubroutineExec) {
+ ; // Simply continue with interpretation
+ } else {
+ setErrorCodeAbort(4231);
+ return -1;
+ }
+ return tNdbColumnImpl->m_attrId;
+ } else {
+ if (theNdbCon->theCommitStatus == Started)
+ setErrorCodeAbort(4200);
+ }
+ return -1;
+
+write_attr_check_error1:
+ if (theOperationType == OpenScanRequest ||
+ theOperationType == OpenRangeScanRequest) {
+ setErrorCodeAbort(4228);
+ return -1;
+ }
+ setErrorCodeAbort(4004);
+ return -1;
+
+write_attr_check_error2:
+ if (tNdbColumnImpl->m_pk) {
+ setErrorCodeAbort(4202);
+ return -1;
+ }//if
+ if (tNdbColumnImpl->getInterpretableType() == false){
+ setErrorCodeAbort(4217);
+ return -1;
+ }//if
+ setErrorCodeAbort(4219);
+ return -1;
+}
+
+int
+NdbOperation::read_attrCheck(const NdbColumnImpl* tNdbColumnImpl)
+{
+ if ((theInterpretIndicator == 1)) {
+ if (tNdbColumnImpl == NULL)
+ goto read_attr_check_error1;
+ if (tNdbColumnImpl->getInterpretableType() == false)
+ goto read_attr_check_error2;
+ if (theStatus == ExecInterpretedValue) {
+ ; // Simply continue with interpretation
+ } else if (theStatus == GetValue) {
+ theInitialReadSize = theTotalCurrAI_Len - 5;
+ theStatus = ExecInterpretedValue;
+ } else if (theStatus == SubroutineExec) {
+ ; // Simply continue with interpretation
+ } else {
+ setErrorCodeAbort(4231);
+ return -1;
+ }
+ return tNdbColumnImpl->m_attrId;
+ } else {
+ if (theNdbCon->theCommitStatus == Started)
+ setErrorCodeAbort(4200);
+ }
+ return -1;
+
+ read_attr_check_error1:
+ setErrorCodeAbort(4004);
+ return -1;
+
+ read_attr_check_error2:
+ if (tNdbColumnImpl->getInterpretableType() == false)
+ setErrorCodeAbort(4217);
+ else
+ setErrorCodeAbort(4219);
+ return -1;
+
+}
+
+int
+NdbOperation::initial_interpreterCheck()
+{
+ if ((theInterpretIndicator == 1)) {
+ if (theStatus == SetBound) {
+ saveBoundATTRINFO();
+ theStatus = GetValue;
+ }
+ if (theStatus == ExecInterpretedValue) {
+ return 0; // Simply continue with interpretation
+ } else if (theStatus == GetValue) {
+ theInitialReadSize = theTotalCurrAI_Len - 5;
+ theStatus = ExecInterpretedValue;
+ return 0;
+ } else if (theStatus == SubroutineExec) {
+ return 0; // Simply continue with interpretation
+ } else {
+ setErrorCodeAbort(4231);
+ return -1;
+ }
+ return 0;
+ } else {
+ if (theNdbCon->theCommitStatus == Started)
+ setErrorCodeAbort(4200);
+ }
+ return -1;
+}
+
+int
+NdbOperation::labelCheck()
+{
+ if ((theInterpretIndicator == 1)) {
+ if (theStatus == ExecInterpretedValue) {
+ return 0; // Simply continue with interpretation
+ } else if (theStatus == GetValue) {
+ theInitialReadSize = theTotalCurrAI_Len - 5;
+ theStatus = ExecInterpretedValue;
+ return 0;
+ } else if (theStatus == SubroutineExec) {
+ return 0; // Simply continue with interpretation
+ } else if (theStatus == SubroutineEnd) {
+ theStatus = SubroutineExec;
+ } else {
+ setErrorCodeAbort(4231);
+ return -1;
+ }
+ return 0;
+ } else {
+ if (theNdbCon->theCommitStatus == Started)
+ setErrorCodeAbort(4200);
+ }
+ return -1;
+}
+
+int
+NdbOperation::intermediate_interpreterCheck()
+{
+ if ((theInterpretIndicator == 1)) {
+ if (theStatus == ExecInterpretedValue) {
+ return 0; // Simply continue with interpretation
+ } else if (theStatus == SubroutineExec) {
+ return 0; // Simply continue with interpretation
+ } else {
+ setErrorCodeAbort(4231);
+ return -1;
+ }
+ return 0;
+ } else {
+ if (theNdbCon->theCommitStatus == Started)
+ setErrorCodeAbort(4200);
+ }
+ return -1;
+}
+
+/*****************************************************************************
+ * int incValue(const char* anAttrName, char* aValue, Uint32 aValue)
+ *
+ * Return Value: Return 0 : incValue was succesful.
+ * Return -1: In all other case.
+ * Parameters: anAttrName : Attribute name where the attribute value
+ * will be save.
+ * aValue : The constant to increment the attribute value with.
+ *****************************************************************************/
+int
+NdbOperation::incValue(const NdbColumnImpl* tNdbColumnImpl, Uint32 aValue)
+{
+ INT_DEBUG(("incValue32 %d %u", tNdbColumnImpl->m_attrId, aValue));
+ int tAttrId;
+
+ tAttrId = incCheck(tNdbColumnImpl);
+ if (tAttrId == -1)
+ goto incValue_error1;
+
+// Load Attribute into register 6
+
+ if (insertATTRINFO( Interpreter::Read(tAttrId, 6)) == -1)
+ goto incValue_error1;
+// Load aValue into register 7
+ if (aValue < 65536)
+ {
+ if (insertATTRINFO(Interpreter::LoadConst16(7, aValue)) == -1)
+ goto incValue_error1;
+ } else {
+ if (insertATTRINFO(Interpreter::LoadConst32(7)) == -1)
+ goto incValue_error1;
+ if (insertATTRINFO(aValue) == -1)
+ goto incValue_error1;
+ }
+ // Add register 6 and 7 and put result in register 7
+
+ if (insertATTRINFO( Interpreter::Add(7, 6, 7)) == -1)
+ goto incValue_error1;
+ if (insertATTRINFO( Interpreter::Write(tAttrId, 7)) == -1)
+ goto incValue_error1;
+
+ theErrorLine++;
+ return 0;
+
+incValue_error1:
+ return -1;
+}
+
+/*****************************************************************************
+ * int subValue(const char* anAttrName, char* aValue, Uint32 aValue)
+ *
+ * Return Value: Return 0 : incValue was succesful.
+ * Return -1: In all other case.
+ * Parameters: anAttrName : Attribute name where the attribute value
+ * will be save.
+ * aValue : The constant to increment the attribute value with.
+******************************************************************************/
+int
+NdbOperation::subValue(const NdbColumnImpl* tNdbColumnImpl, Uint32 aValue)
+{
+ INT_DEBUG(("subValue32 %d %u", tNdbColumnImpl->m_attrId, aValue));
+ int tAttrId;
+
+ tAttrId = incCheck(tNdbColumnImpl);
+ if (tAttrId == -1)
+ goto subValue_error1;
+
+// Load Attribute into register 6
+
+ if (insertATTRINFO( Interpreter::Read(tAttrId, 6)) == -1)
+ goto subValue_error1;
+// Load aValue into register 7
+ if (aValue < 65536)
+ {
+ if (insertATTRINFO( Interpreter::LoadConst16(7, aValue)) == -1)
+ goto subValue_error1;
+ } else {
+ if (insertATTRINFO( Interpreter::LoadConst32(7)) == -1)
+ goto subValue_error1;
+ if (insertATTRINFO(aValue) == -1)
+ goto subValue_error1;
+ }
+ // Subtract register 6 and 7 and put result in register 7
+
+ if (insertATTRINFO( Interpreter::Sub(7, 6, 7)) == -1)
+ goto subValue_error1;
+ if (insertATTRINFO( Interpreter::Write(tAttrId, 7)) == -1)
+ goto subValue_error1;
+
+ theErrorLine++;
+ return 0;
+
+ subValue_error1:
+ return -1;
+}
+
+/******************************************************************************
+ * int incValue(const char* anAttrName, char* aValue, Uint64 aValue)
+ *
+ * Return Value: Return 0 : incValue was succesful.
+ * Return -1: In all other case.
+ * Parameters: anAttrName : Attribute name where the attribute value will
+ * be save.
+ * aValue : The constant to increment the attribute value with.
+ *****************************************************************************/
+int
+NdbOperation::incValue(const NdbColumnImpl* tNdbColumnImpl, Uint64 aValue)
+{
+ INT_DEBUG(("incValue64 %d %llu", tNdbColumnImpl->m_attrId, aValue));
+ int tAttrId;
+
+ tAttrId = incCheck(tNdbColumnImpl);
+ if (tAttrId == -1)
+ goto incValue_error1;
+
+// Load Attribute into register 6
+
+ if (insertATTRINFO( Interpreter::Read(tAttrId, 6)) == -1)
+ goto incValue_error1;
+// Load aValue into register 7
+ if (insertATTRINFO( Interpreter::LoadConst64(7)) == -1)
+ goto incValue_error1;
+ if (insertATTRINFO((Uint32)(aValue >> 32)) == -1)
+ goto incValue_error1;
+ if (insertATTRINFO(Uint32(aValue & 0xFFFFFFFF)) == -1)
+ goto incValue_error1;
+ // Add register 6 and 7 and put result in register 7
+ if (insertATTRINFO( Interpreter::Add(7, 6, 7)) == -1)
+ goto incValue_error1;
+ if (insertATTRINFO( Interpreter::Write(tAttrId, 7)) == -1)
+ goto incValue_error1;
+
+ theErrorLine++;
+ return 0;
+
+incValue_error1:
+ return -1;
+}
+
+/*****************************************************************************
+ * int subValue(const char* anAttrName, char* aValue, Uint64 aValue)
+ *
+ * Return Value: Return 0 : incValue was succesful.
+ * Return -1: In all other case.
+ * Parameters: anAttrName : Attribute name where the attribute value will
+ * be save.
+ * aValue : The constant to increment the attribute value with.
+******************************************************************************/
+int
+NdbOperation::subValue(const NdbColumnImpl* tNdbColumnImpl, Uint64 aValue)
+{
+ INT_DEBUG(("subValue64 %d %llu", tNdbColumnImpl->m_attrId, aValue));
+ int tAttrId;
+
+ tAttrId = incCheck(tNdbColumnImpl);
+ if (tAttrId == -1)
+ goto subValue_error1;
+
+// Load Attribute into register 6
+
+ if (insertATTRINFO( Interpreter::Read(6, tAttrId)) == -1)
+ goto subValue_error1;
+// Load aValue into register 7
+ if (insertATTRINFO( Interpreter::LoadConst64(7)) == -1)
+ goto subValue_error1;
+ if (insertATTRINFO((Uint32)(aValue >> 32)) == -1)
+ goto subValue_error1;
+ if (insertATTRINFO(Uint32(aValue & 0xFFFFFFFF)) == -1)
+ goto subValue_error1;
+// Subtract register 6 and 7 and put result in register 7
+ if (insertATTRINFO( Interpreter::Sub(7, 6, 7)) == -1)
+ goto subValue_error1;
+ if (insertATTRINFO( Interpreter::Write(tAttrId, 7)) == -1)
+ goto subValue_error1;
+
+ theErrorLine++;
+ return 0;
+
+subValue_error1:
+ return -1;
+}
+
+/*****************************************************************************
+ * int NdbOperation::def_label()
+ *****************************************************************************/
+int
+NdbOperation::def_label(int tLabelNo)
+{
+ INT_DEBUG(("def_label %d", tLabelNo));
+ Uint32 tLabelIndex;
+ if (labelCheck() == -1)
+ return -1;
+
+ tLabelIndex = theNoOfLabels - ((theNoOfLabels >> 4) << 4);
+ if (tLabelIndex == 0)
+ {
+ NdbLabel* tNdbLabel = theNdb->getNdbLabel();
+ if (tNdbLabel == NULL)
+ {
+ setErrorCodeAbort(4000);
+ return -1;
+ }
+ if (theFirstLabel == NULL)
+ theFirstLabel = tNdbLabel;
+ else
+ theLastLabel->theNext = tNdbLabel;
+
+ theLastLabel = tNdbLabel;
+ tNdbLabel->theNext = NULL;
+ }
+
+ /**
+ * Here we set the address that the label should point to (jump address),
+ * the first 5 words are excluded since they are length specifications and
+ * not part of the data.
+ * We need to add 1 to the current ATTRINFO length since the last inserted
+ * item is not where we want to jump to.
+ * Later on we will update the branch items with this address, this is done in
+ * the NdbOperation::prepareSendInterpreted method.
+ */
+
+ theLastLabel->theLabelNo[tLabelIndex] = tLabelNo;
+ theLastLabel->theLabelAddress[tLabelIndex] = (theTotalCurrAI_Len + 1) - (theInitialReadSize + 5);
+ theLastLabel->theSubroutine[tLabelIndex] = theNoOfSubroutines;
+ theNoOfLabels++;
+ theErrorLine++;
+ return (theNoOfLabels - 1);
+}
+
+/************************************************************************************************
+int NdbOperation::def_subroutine()
+
+************************************************************************************************/
+int
+NdbOperation::def_subroutine(int tSubNo)
+{
+ INT_DEBUG(("def_subroutine %d", tSubNo));
+ Uint32 tSubroutineIndex;
+
+ if (theInterpretIndicator != 1)
+ {
+ setErrorCodeAbort(4200);
+ return -1;
+ }
+
+ if (int(theNoOfSubroutines) != tSubNo)
+ {
+ setErrorCodeAbort(4227);
+ return -1;
+ }
+ if (theStatus == FinalGetValue)
+ {
+ theFinalReadSize = theTotalCurrAI_Len -
+ (theInitialReadSize + theInterpretedSize +
+ theFinalUpdateSize + 5);
+
+ } else if (theStatus == SubroutineEnd)
+ {
+ ; // Correct Status, last call was ret_sub()
+ } else if (theStatus == ExecInterpretedValue)
+ {
+ if (insertATTRINFO(Interpreter::EXIT_OK) == -1)
+ return -1;
+ theInterpretedSize = theTotalCurrAI_Len - (theInitialReadSize + 5);
+ } else if (theStatus == SetValueInterpreted)
+ {
+ theFinalUpdateSize = theTotalCurrAI_Len -
+ (theInitialReadSize + theInterpretedSize + 5);
+
+ } else if (theStatus == GetValue)
+ {
+
+ theInitialReadSize = theTotalCurrAI_Len - 5;
+
+ } else
+ {
+ setErrorCodeAbort(4200);
+ return -1;
+ }
+ theStatus = SubroutineExec;
+ tSubroutineIndex = theNoOfSubroutines - ((theNoOfSubroutines >> 4) << 4);
+ if (tSubroutineIndex == 0)
+ {
+ NdbSubroutine* tNdbSubroutine = theNdb->getNdbSubroutine();
+ if (tNdbSubroutine == NULL)
+ {
+ setErrorCodeAbort(4000);
+ return -1;
+ }
+ if (theFirstSubroutine == NULL)
+ theFirstSubroutine = tNdbSubroutine;
+ else
+ theLastSubroutine->theNext = tNdbSubroutine;
+
+ theLastSubroutine = tNdbSubroutine;
+ tNdbSubroutine->theNext = NULL;
+ }
+ theLastSubroutine->theSubroutineAddress[tSubroutineIndex] = theTotalCurrAI_Len -
+ (theInitialReadSize + theInterpretedSize +
+ theFinalUpdateSize + theFinalReadSize);
+ theNoOfSubroutines++;
+ theErrorLine++;
+ return (theNoOfSubroutines - 1);
+}
+
+/************************************************************************************************
+int NdbOperation::add_reg(Uint32 RegSource1, Uint32 RegSource2, Uint32 RegDest)
+
+************************************************************************************************/
+int
+NdbOperation::add_reg(Uint32 RegSource1, Uint32 RegSource2, Uint32 RegDest)
+{
+ INT_DEBUG(("add_reg %u %u %u", RegSource1, RegSource2, RegDest));
+ if (intermediate_interpreterCheck() == -1)
+ return -1;
+
+ if (RegSource1 >= 8)
+ {
+ setErrorCodeAbort(4229);
+ return -1;
+ }
+ if (RegSource2 >= 8)
+ {
+ setErrorCodeAbort(4229);
+ return -1;
+ }
+ if (RegDest >= 8)
+ {
+ setErrorCodeAbort(4229);
+ return -1;
+ }
+ if (insertATTRINFO( Interpreter::Add(RegDest, RegSource1, RegSource2)) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+/************************************************************************************************
+int NdbOperation::sub_reg(Uint32 RegSource1, Uint32 RegSource2, Uint32 RegDest)
+
+************************************************************************************************/
+int
+NdbOperation::sub_reg(Uint32 RegSource1, Uint32 RegSource2, Uint32 RegDest)
+{
+ INT_DEBUG(("sub_reg %u %u %u", RegSource1, RegSource2, RegDest));
+ if (intermediate_interpreterCheck() == -1)
+ return -1;
+
+ if (RegSource1 >= 8)
+ {
+ setErrorCodeAbort(4229);
+ return -1;
+ }
+ if (RegSource2 >= 8)
+ {
+ setErrorCodeAbort(4229);
+ return -1;
+ }
+ if (RegDest >= 8)
+ {
+ setErrorCodeAbort(4229);
+ return -1;
+ }
+ if (insertATTRINFO( Interpreter::Sub(RegDest, RegSource1, RegSource2)) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+/************************************************************************************************
+int NdbOperation::load_const_u32(Uint32 RegDest, Uint32 Constant)
+
+************************************************************************************************/
+int
+NdbOperation::load_const_u32(Uint32 RegDest, Uint32 Constant)
+{
+ INT_DEBUG(("load_const_u32 %u %u", RegDest, Constant));
+ if (initial_interpreterCheck() == -1)
+ goto l_u32_error1;
+ if (RegDest >= 8)
+ goto l_u32_error2;
+ if (insertATTRINFO( Interpreter::LoadConst32(RegDest)) == -1)
+ goto l_u32_error1;
+ if (insertATTRINFO(Constant) == -1)
+ goto l_u32_error1;
+ theErrorLine++;
+ return 0;
+
+ l_u32_error1:
+ return -1;
+
+ l_u32_error2:
+ setErrorCodeAbort(4229);
+ return -1;
+}
+
+/************************************************************************************************
+int NdbOperation::load_const_u64(Uint32 RegDest, Uint64 Constant)
+
+************************************************************************************************/
+int
+NdbOperation::load_const_u64(Uint32 RegDest, Uint64 Constant)
+{
+ INT_DEBUG(("load_const_u64 %u %llu", RegDest, Constant));
+ Uint32 tTemp1;
+ Uint32 tTemp2;
+ if (initial_interpreterCheck() == -1)
+ return -1;
+ if (RegDest >= 8)
+ {
+ setErrorCodeAbort(4229);
+ return -1;
+ }
+ tTemp1 = (Uint32)(Constant & 0xFFFFFFFF);
+ tTemp2 = (Uint32)(Constant >> 32);
+
+ // 64 bit value
+ if (insertATTRINFO( Interpreter::LoadConst64(RegDest)) == -1)
+ return -1;
+ if (insertATTRINFO(tTemp1) == -1)
+ return -1;
+ if (insertATTRINFO(tTemp2) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+/************************************************************************************************
+int NdbOperation::load_const_null(Uint32 RegDest)
+
+************************************************************************************************/
+int
+NdbOperation::load_const_null(Uint32 RegDest)
+{
+ INT_DEBUG(("load_const_null %u", RegDest));
+ if (initial_interpreterCheck() == -1)
+ return -1;
+ if (RegDest >= 8)
+ {
+ setErrorCodeAbort(4229);
+ return -1;
+ }
+ if (insertATTRINFO( Interpreter::LOAD_CONST_NULL) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+/************************************************************************************************
+int NdbOperation::read_attr(const char* anAttrName, Uint32 RegDest)
+
+************************************************************************************************/
+int
+NdbOperation::read_attr(const NdbColumnImpl* anAttrObject, Uint32 RegDest)
+{
+ INT_DEBUG(("read_attr %d %u", anAttrObject->m_attrId, RegDest));
+ int tAttrId = read_attrCheck(anAttrObject);
+ if (tAttrId == -1)
+ goto read_attr_error1;
+ if (RegDest >= 8)
+ goto read_attr_error2;
+ if (insertATTRINFO( Interpreter::Read(tAttrId, RegDest)) != -1) {
+ return 0;
+ theErrorLine++;
+ }//if
+ return -1;
+
+read_attr_error1:
+ return -1;
+
+read_attr_error2:
+ setErrorCodeAbort(4229);
+ return -1;
+}
+
+/************************************************************************************************
+int NdbOperation::write_attr(const char* anAttrName, Uint32 RegSource)
+
+************************************************************************************************/
+int
+NdbOperation::write_attr(const NdbColumnImpl* anAttrObject, Uint32 RegSource)
+{
+ INT_DEBUG(("write_attr %d %u", anAttrObject->m_attrId, RegSource));
+ int tAttrId = write_attrCheck(anAttrObject);
+ if (tAttrId == -1)
+ return -1;
+ if (insertATTRINFO( Interpreter::Write(tAttrId, RegSource)) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::branch_reg_reg(Uint32 type,
+ Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label)
+{
+ if (intermediate_interpreterCheck() == -1)
+ return -1;
+ if (insertATTRINFO(Interpreter::Branch(type, RegLvalue, RegRvalue)) == -1)
+ return -1;
+ if (insertBranch(Label) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::branch_ge(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label)
+{
+ INT_DEBUG(("branch_ge %u %u %u", RegLvalue, RegRvalue, Label));
+ return branch_reg_reg(Interpreter::BRANCH_GE_REG_REG,
+ RegLvalue, RegRvalue, Label);
+}
+
+int
+NdbOperation::branch_gt(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label)
+{
+ INT_DEBUG(("branch_gt %u %u %u", RegLvalue, RegRvalue, Label));
+ return branch_reg_reg(Interpreter::BRANCH_GT_REG_REG,
+ RegLvalue, RegRvalue, Label);
+}
+
+int
+NdbOperation::branch_le(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label)
+{
+ INT_DEBUG(("branch_le %u %u %u", RegLvalue, RegRvalue, Label));
+ return branch_reg_reg(Interpreter::BRANCH_LE_REG_REG,
+ RegLvalue, RegRvalue, Label);
+}
+
+int
+NdbOperation::branch_lt(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label)
+{
+ INT_DEBUG(("branch_lt %u %u %u", RegLvalue, RegRvalue, Label));
+ return branch_reg_reg(Interpreter::BRANCH_LT_REG_REG,
+ RegLvalue, RegRvalue, Label);
+}
+
+int
+NdbOperation::branch_eq(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label)
+{
+ INT_DEBUG(("branch_eq %u %u %u", RegLvalue, RegRvalue, Label));
+ return branch_reg_reg(Interpreter::BRANCH_EQ_REG_REG,
+ RegLvalue, RegRvalue, Label);
+}
+
+int
+NdbOperation::branch_ne(Uint32 RegLvalue, Uint32 RegRvalue, Uint32 Label)
+{
+ INT_DEBUG(("branch_ne %u %u %u", RegLvalue, RegRvalue, Label));
+ return branch_reg_reg(Interpreter::BRANCH_NE_REG_REG,
+ RegLvalue, RegRvalue, Label);
+}
+
+int
+NdbOperation::branch_ne_null(Uint32 RegLvalue, Uint32 Label)
+{
+ INT_DEBUG(("branch_ne_null %u %u", RegLvalue, Label));
+ if (intermediate_interpreterCheck() == -1)
+ return -1;
+ if (insertATTRINFO((RegLvalue << 6) + Interpreter::BRANCH_REG_NE_NULL) == -1)
+ return -1;
+ if (insertBranch(Label) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::branch_eq_null(Uint32 RegLvalue, Uint32 Label)
+{
+ INT_DEBUG(("branch_eq_null %u %u", RegLvalue, Label));
+ if (intermediate_interpreterCheck() == -1)
+ return -1;
+ if (insertATTRINFO((RegLvalue << 6) + Interpreter::BRANCH_REG_EQ_NULL) == -1)
+ return -1;
+ if (insertBranch(Label) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::branch_label(Uint32 Label)
+{
+ INT_DEBUG(("branch_label %u", Label));
+ if (initial_interpreterCheck() == -1)
+ return -1;
+ if (insertATTRINFO(Interpreter::BRANCH) == -1)
+ return -1;
+ if (insertBranch(Label) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+/************************************************************************************************
+int NdbOperation::interpret_exit_ok()
+
+************************************************************************************************/
+int
+NdbOperation::interpret_exit_ok()
+{
+ INT_DEBUG(("interpret_exit_ok"));
+ if (initial_interpreterCheck() == -1)
+ return -1;
+ if (insertATTRINFO(Interpreter::EXIT_OK) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+/************************************************************************************************
+int NdbOperation::interpret_exit_nok(Uint32 ErrorCode)
+
+************************************************************************************************/
+int
+NdbOperation::interpret_exit_nok(Uint32 ErrorCode)
+{
+ INT_DEBUG(("interpret_exit_nok %u", ErrorCode));
+ if (initial_interpreterCheck() == -1)
+ return -1;
+ if (insertATTRINFO( (ErrorCode << 16) + Interpreter::EXIT_REFUSE) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::interpret_exit_nok()
+{
+ INT_DEBUG(("interpret_exit_nok"));
+ Uint32 ErrorCode = 899;
+
+ if (initial_interpreterCheck() == -1)
+ return -1;
+ if (insertATTRINFO( (ErrorCode << 16) + Interpreter::EXIT_REFUSE) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::call_sub(Uint32 Subroutine)
+{
+ INT_DEBUG(("call_sub %u", Subroutine));
+ if (initial_interpreterCheck() == -1)
+ return -1;
+ if (insertATTRINFO( (Subroutine << 16) + Interpreter::CALL) == -1)
+ return -1;
+ if (insertCall(Subroutine) == -1)
+ return -1;
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::ret_sub()
+{
+ INT_DEBUG(("ret_sub"));
+ if (theInterpretIndicator != 1)
+ {
+ setErrorCodeAbort(4200);
+ return -1;
+ }
+ if (theStatus == SubroutineExec)
+ {
+ ; // Simply continue with interpretation
+ } else
+ {
+ setErrorCodeAbort(4200);
+ return -1;
+ }
+ if (insertATTRINFO(Interpreter::RETURN) == -1)
+ return -1;
+ theStatus = SubroutineEnd;
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::insertBranch(Uint32 aLabel)
+{
+ Uint32 tAddress;
+ NdbBranch* tBranch = theNdb->getNdbBranch();
+ if (tBranch == NULL)
+ goto insertBranch_error1;
+ if (theFirstBranch == NULL)
+ theFirstBranch = tBranch;
+ else
+ theLastBranch->theNext = tBranch;
+ theLastBranch = tBranch;
+ if (theNoOfSubroutines == 0)
+ tAddress = theTotalCurrAI_Len -
+ (theInitialReadSize + 5);
+ else
+ tAddress = theTotalCurrAI_Len -
+ (theInitialReadSize + theInterpretedSize +
+ theFinalUpdateSize + theFinalReadSize + 5);
+
+ tBranch->theBranchAddress = tAddress;
+ tBranch->theSignal = theCurrentATTRINFO;
+ tBranch->theSignalAddress = theAI_LenInCurrAI; // + 1; theAI_LenInCurrAI has already been updated in
+ tBranch->theSubroutine = theNoOfSubroutines; // insertATTRINFO which was done before insertBranch!!
+ tBranch->theBranchLabel = aLabel;
+ return 0;
+
+insertBranch_error1:
+ setErrorCodeAbort(4000);
+ return -1;
+}
+
+int
+NdbOperation::insertCall(Uint32 aCall)
+{
+ NdbCall* tCall = theNdb->getNdbCall();
+ if (tCall == NULL)
+ {
+ setErrorCodeAbort(4000);
+ return -1;
+ }
+ if (theFirstCall == NULL)
+ theFirstCall = tCall;
+ else
+ theLastCall->theNext = tCall;
+ theLastCall = tCall;
+
+ tCall->theSignal = theCurrentATTRINFO;
+ tCall->theSignalAddress = theAI_LenInCurrAI;
+ tCall->theSubroutine = aCall;
+ return 0;
+}
+
+int
+NdbOperation::branch_col(Uint32 type,
+ Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label){
+
+ if (initial_interpreterCheck() == -1)
+ return -1;
+
+ Interpreter::BinaryCondition c = (Interpreter::BinaryCondition)type;
+
+ const NdbDictionary::Column * col =
+ m_currentTable->getColumn(ColId);
+
+ if(col == 0){
+ abort();
+ }
+
+ Uint32 vc = col->getType() == NdbDictionary::Column::Varchar;
+ Uint32 colLen = col->getLength() + 2 * vc;
+ Uint32 al = (4 - (colLen & 3)) & 0x3;
+
+ if (insertATTRINFO(Interpreter::BranchCol(c, al, vc, nopad)) == -1)
+ return -1;
+
+ if (insertBranch(Label) == -1)
+ return -1;
+
+ if (insertATTRINFO(Interpreter::BranchCol_2(ColId, len)))
+ return -1;
+
+ Uint32 len2 = Interpreter::mod4(len);
+ if(len2 == len){
+ insertATTRINFOloop((Uint32*)val, len2 >> 2);
+ } else {
+ len2 -= 4;
+ insertATTRINFOloop((Uint32*)val, len2 >> 2);
+ Uint32 tmp = 0;
+ for (Uint32 i = 0; i < len-len2; i++) {
+ char* p = (char*)&tmp;
+ p[i] = val[len2+i];
+ }
+ insertATTRINFO(tmp);
+ }
+
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::branch_col_eq(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label){
+ INT_DEBUG(("branch_col_eq %u %.*s(%u,%d) -> %u", ColId, len, val, len, nopad, Label));
+ return branch_col(Interpreter::EQ, ColId, val, len, nopad, Label);
+}
+
+int
+NdbOperation::branch_col_ne(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label){
+ INT_DEBUG(("branch_col_ne %u %.*s(%u,%d) -> %u", ColId, len, val, len, nopad, Label));
+ return branch_col(Interpreter::NE, ColId, val, len, nopad, Label);
+}
+int
+NdbOperation::branch_col_lt(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label){
+ INT_DEBUG(("branch_col_lt %u %.*s(%u,%d) -> %u", ColId, len, val, len, nopad, Label));
+ return branch_col(Interpreter::LT, ColId, val, len, nopad, Label);
+}
+int
+NdbOperation::branch_col_le(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label){
+ INT_DEBUG(("branch_col_le %u %.*s(%u,%d) -> %u", ColId, len, val, len, nopad, Label));
+ return branch_col(Interpreter::LE, ColId, val, len, nopad, Label);
+}
+int
+NdbOperation::branch_col_gt(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label){
+ INT_DEBUG(("branch_col_gt %u %.*s(%u,%d) -> %u", ColId, len, val, len, nopad, Label));
+ return branch_col(Interpreter::GT, ColId, val, len, nopad, Label);
+}
+
+int
+NdbOperation::branch_col_ge(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label){
+ INT_DEBUG(("branch_col_ge %u %.*s(%u,%d) -> %u", ColId, len, val, len, nopad, Label));
+ return branch_col(Interpreter::GE, ColId, val, len, nopad, Label);
+}
+
+int
+NdbOperation::branch_col_like(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label){
+ INT_DEBUG(("branch_col_like %u %.*s(%u,%d) -> %u", ColId, len, val, len, nopad, Label));
+ return branch_col(Interpreter::LIKE, ColId, val, len, nopad, Label);
+}
+
+int
+NdbOperation::branch_col_notlike(Uint32 ColId, const char * val, Uint32 len,
+ bool nopad, Uint32 Label){
+ INT_DEBUG(("branch_col_notlike %u %.*s(%u,%d) -> %u", ColId,len,val,len,nopad,Label));
+ return branch_col(Interpreter::NOT_LIKE, ColId, val, len, nopad, Label);
+}
+
+int
+NdbOperation::branch_col_null(Uint32 type, Uint32 ColId, Uint32 Label){
+
+ if (initial_interpreterCheck() == -1)
+ return -1;
+
+ if (insertATTRINFO(type) == -1)
+ return -1;
+
+ if (insertBranch(Label) == -1)
+ return -1;
+
+ if (insertATTRINFO(Interpreter::BranchCol_2(ColId)))
+ return -1;
+
+ theErrorLine++;
+ return 0;
+}
+
+int
+NdbOperation::branch_col_eq_null(Uint32 ColId, Uint32 Label){
+
+ INT_DEBUG(("branch_col_eq_null %u -> %u", ColId, Label));
+ return branch_col_null(Interpreter::BRANCH_ATTR_EQ_NULL, ColId, Label);
+}
+
+int
+NdbOperation::branch_col_ne_null(Uint32 ColId, Uint32 Label){
+
+ INT_DEBUG(("branch_col_ne_null %u -> %u", ColId, Label));
+ return branch_col_null(Interpreter::BRANCH_ATTR_NE_NULL, ColId, Label);
+}
+
diff --git a/ndb/src/ndbapi/NdbOperationScan.cpp b/ndb/src/ndbapi/NdbOperationScan.cpp
new file mode 100644
index 00000000000..df4f2421ec0
--- /dev/null
+++ b/ndb/src/ndbapi/NdbOperationScan.cpp
@@ -0,0 +1,576 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbOperation.hpp"
+#include "NdbScanReceiver.hpp"
+
+#include <signaldata/TcKeyReq.hpp>
+#include <signaldata/ScanTab.hpp>
+#include <signaldata/ScanFrag.hpp>
+#include <signaldata/KeyInfo.hpp>
+
+
+/******************************************************************************
+ * int openScanRead();
+ *****************************************************************************/
+int
+NdbOperation::openScanRead(Uint32 aParallelism)
+{
+ aParallelism = checkParallelism(aParallelism);
+
+ if ((theNdbCon->theCommitStatus != Started) &&
+ (theStatus != Init) &&
+ (aParallelism == 0)) {
+ setErrorCode(4200);
+ return -1;
+ }
+ return openScan(aParallelism, false, false, false);
+}
+
+/****************************************************************************
+ * int openScanExclusive();
+ ****************************************************************************/
+int
+NdbOperation::openScanExclusive(Uint32 aParallelism)
+{
+ aParallelism = checkParallelism(aParallelism);
+
+ if ((theNdbCon->theCommitStatus != Started) &&
+ (theStatus != Init) &&
+ (aParallelism == 0)) {
+ setErrorCode(4200);
+ return -1;
+ }
+ return openScan(aParallelism, true, true, false);
+}
+
+/******************************************************************************
+ * int openScanReadHoldLock();
+ *****************************************************************************/
+int
+NdbOperation::openScanReadHoldLock(Uint32 aParallelism)
+{
+ aParallelism = checkParallelism(aParallelism);
+
+ if ((theNdbCon->theCommitStatus != Started) &&
+ (theStatus != Init) &&
+ (aParallelism == 0)) {
+ setErrorCode(4200);
+ return -1;
+ }
+ return openScan(aParallelism, false, true, false);
+}
+
+/******************************************************************************
+ * int openScanReadCommitted();
+ *****************************************************************************/
+int
+NdbOperation::openScanReadCommitted(Uint32 aParallelism)
+{
+ aParallelism = checkParallelism(aParallelism);
+
+ if ((theNdbCon->theCommitStatus != Started) &&
+ (theStatus != Init) &&
+ (aParallelism == 0)) {
+ setErrorCode(4200);
+ return -1;
+ }
+ return openScan(aParallelism, false, false, true);
+}
+
+/****************************************************************************
+ * int checkParallelism();
+ * Remark If the parallelism is set wrong the number of scan-operations
+ * will not correspond to the number of TRANSID_AI signals returned
+ * from NDB and the result will be a crash, therefore
+ * we adjust it or return an error if the value is totally wrong.
+ ****************************************************************************/
+int
+NdbOperation::checkParallelism(Uint32 aParallelism)
+{
+ if (aParallelism == 0) {
+ setErrorCodeAbort(4232);
+ return 0;
+ }
+ if (aParallelism > 16) {
+ if (aParallelism <= 240) {
+
+ /**
+ * If tscanConcurrency > 16 it must be a multiple of 16
+ */
+ if (((aParallelism >> 4) << 4) < aParallelism) {
+ aParallelism = ((aParallelism >> 4) << 4) + 16;
+ }//if
+
+ /*---------------------------------------------------------------*/
+ /* We cannot have a parallelism > 16 per node */
+ /*---------------------------------------------------------------*/
+ if ((aParallelism / theNdb->theNoOfDBnodes) > 16) {
+ aParallelism = theNdb->theNoOfDBnodes * 16;
+ }//if
+
+ } else {
+ setErrorCodeAbort(4232);
+ aParallelism = 0;
+ }//if
+ }//if
+ return aParallelism;
+}//NdbOperation::checkParallelism()
+
+/**********************************************************************
+ * int openScan();
+ *************************************************************************/
+int
+NdbOperation::openScan(Uint32 aParallelism,
+ bool lockMode, bool lockHoldMode, bool readCommitted)
+{
+ aParallelism = checkParallelism(aParallelism);
+ if(aParallelism == 0){
+ return 0;
+ }
+ NdbScanReceiver* tScanRec;
+ // It is only possible to call openScan if
+ // 1. this transcation don't already contain another scan operation
+ // 2. this transaction don't already contain other operations
+ // 3. theScanOp contains a NdbScanOperation
+ if (theNdbCon->theScanningOp != NULL){
+ setErrorCode(4605);
+ return -1;
+ }
+
+ if ((theNdbCon->theFirstOpInList != this) ||
+ (theNdbCon->theLastOpInList != this)) {
+ setErrorCode(4603);
+ return -1;
+ }
+ theNdbCon->theScanningOp = this;
+
+ initScan();
+ theParallelism = aParallelism;
+
+ // If the scan is on ordered index then it is a range scan
+ if (m_currentTable->m_indexType == NdbDictionary::Index::OrderedIndex ||
+ m_currentTable->m_indexType == NdbDictionary::Index::UniqueOrderedIndex) {
+ assert(m_currentTable == m_accessTable);
+ m_currentTable = theNdb->theDictionary->getTable(m_currentTable->m_primaryTable.c_str());
+ assert(m_currentTable != NULL);
+ // Modify operation state
+ theStatus = SetBound;
+ theOperationType = OpenRangeScanRequest;
+ }
+
+ theScanReceiversArray = new NdbScanReceiver* [aParallelism];
+ if (theScanReceiversArray == NULL){
+ setErrorCodeAbort(4000);
+ return -1;
+ }
+
+ for (Uint32 i = 0; i < aParallelism; i ++) {
+ tScanRec = theNdb->getNdbScanRec();
+ if (tScanRec == NULL) {
+ setErrorCodeAbort(4000);
+ return -1;
+ }//if
+ tScanRec->init(this, lockMode);
+ theScanReceiversArray[i] = tScanRec;
+ }
+
+ theSCAN_TABREQ = theNdb->getSignal();
+ if (theSCAN_TABREQ == NULL) {
+ setErrorCodeAbort(4000);
+ return -1;
+ }//if
+ ScanTabReq * const scanTabReq = CAST_PTR(ScanTabReq, theSCAN_TABREQ->getDataPtrSend());
+ scanTabReq->apiConnectPtr = theNdbCon->theTCConPtr;
+ scanTabReq->tableId = m_accessTable->m_tableId;
+ scanTabReq->tableSchemaVersion = m_accessTable->m_version;
+ scanTabReq->storedProcId = 0xFFFF;
+ scanTabReq->buddyConPtr = theNdbCon->theBuddyConPtr;
+
+ Uint32 reqInfo = 0;
+ ScanTabReq::setParallelism(reqInfo, aParallelism);
+ ScanTabReq::setLockMode(reqInfo, lockMode);
+ ScanTabReq::setHoldLockFlag(reqInfo, lockHoldMode);
+ ScanTabReq::setReadCommittedFlag(reqInfo, readCommitted);
+ if (theOperationType == OpenRangeScanRequest)
+ ScanTabReq::setRangeScanFlag(reqInfo, true);
+ scanTabReq->requestInfo = reqInfo;
+
+ Uint64 transId = theNdbCon->getTransactionId();
+ scanTabReq->transId1 = (Uint32) transId;
+ scanTabReq->transId2 = (Uint32) (transId >> 32);
+
+ for (Uint32 i = 0; i < 16 && i < aParallelism ; i++) {
+ scanTabReq->apiOperationPtr[i] = theScanReceiversArray[i]->ptr2int();
+ }//for
+
+ // Create one additional SCAN_TABINFO for each
+ // 16 of parallelism
+ NdbApiSignal* tSignal;
+ Uint32 tParallelism = aParallelism;
+ while (tParallelism > 16) {
+ tSignal = theNdb->getSignal();
+ if (tSignal == NULL) {
+ setErrorCodeAbort(4000);
+ return -1;
+ }//if
+ if (tSignal->setSignal(GSN_SCAN_TABINFO) == -1) {
+ setErrorCode(4001);
+ return -1;
+ }
+ tSignal->next(theFirstSCAN_TABINFO_Send);
+ theFirstSCAN_TABINFO_Send = tSignal;
+ tParallelism -= 16;
+ }//while
+
+ // Format all SCAN_TABINFO signals
+ tParallelism = 16;
+ tSignal = theFirstSCAN_TABINFO_Send;
+ while (tSignal != NULL) {
+ tSignal->setData(theNdbCon->theTCConPtr, 1);
+ for (int i = 0; i < 16 ; i++) {
+ tSignal->setData(theScanReceiversArray[i + tParallelism]->ptr2int(), i + 2);
+ }//for
+ tSignal = tSignal->next();
+ tParallelism += 16;
+ }//while
+
+ getFirstATTRINFOScan();
+ return 0;
+}//NdbScanOperation::openScan()
+
+/*****************************************************************************
+ * int getFirstATTRINFOScan( U_int32 aData )
+ *
+ * Return Value: Return 0: Successful
+ * Return -1: All other cases
+ * Parameters: None: Only allocate the first signal.
+ * Remark: When a scan is defined we need to use this method instead
+ * of insertATTRINFO for the first signal.
+ * This is because we need not to mess up the code in
+ * insertATTRINFO with if statements since we are not
+ * interested in the TCKEYREQ signal.
+ *****************************************************************************/
+int
+NdbOperation::getFirstATTRINFOScan()
+{
+ NdbApiSignal* tSignal;
+
+ tSignal = theNdb->getSignal();
+ if (tSignal == NULL){
+ setErrorCodeAbort(4000);
+ return -1;
+ }
+ tSignal->setSignal(m_attrInfoGSN);
+ theAI_LenInCurrAI = 8;
+ theATTRINFOptr = &tSignal->getDataPtrSend()[8];
+ theFirstATTRINFO = tSignal;
+ theCurrentATTRINFO = tSignal;
+ theCurrentATTRINFO->next(NULL);
+ return 0;
+}
+
+/*
+ * After setBound() are done, move the accumulated ATTRINFO signals to
+ * a separate list. Then continue with normal scan.
+ */
+int
+NdbOperation::saveBoundATTRINFO()
+{
+ theCurrentATTRINFO->setLength(theAI_LenInCurrAI);
+ theBoundATTRINFO = theFirstATTRINFO;
+ theTotalBoundAI_Len = theTotalCurrAI_Len;
+ theTotalCurrAI_Len = 5;
+ theBoundATTRINFO->setData(theTotalBoundAI_Len, 4);
+ theBoundATTRINFO->setData(0, 5);
+ theBoundATTRINFO->setData(0, 6);
+ theBoundATTRINFO->setData(0, 7);
+ theBoundATTRINFO->setData(0, 8);
+ theStatus = GetValue;
+ return getFirstATTRINFOScan();
+}
+
+/*****************************************************************************
+ * void releaseScan()
+ *
+ * Return Value No return value.
+ * Parameters: No parameters.
+ * Remark: Release objects after scanning.
+ *****************************************************************************/
+void
+NdbOperation::releaseScan()
+{
+ NdbScanReceiver* tScanRec;
+ TransporterFacade::instance()->lock_mutex();
+ for (Uint32 i = 0; i < theParallelism && theScanReceiversArray != NULL; i++) {
+ tScanRec = theScanReceiversArray[i];
+ if (tScanRec != NULL) {
+ tScanRec->release();
+ tScanRec->next(NULL);
+ }
+ }
+ TransporterFacade::instance()->unlock_mutex();
+ releaseSignals();
+
+ if (theScanReceiversArray != NULL) {
+ for (Uint32 i = 0; i < theParallelism; i++) {
+ NdbScanReceiver* tScanRec;
+ tScanRec = theScanReceiversArray[i];
+ if (tScanRec != NULL) {
+ theNdb->releaseNdbScanRec(tScanRec);
+ theScanReceiversArray[i] = NULL;
+ }
+ }
+
+ delete [] theScanReceiversArray;
+ }//if
+ theScanReceiversArray = NULL;
+
+ if (theSCAN_TABREQ != NULL){
+ theNdb->releaseSignal(theSCAN_TABREQ);
+ theSCAN_TABREQ = NULL;
+ }
+}
+
+void NdbOperation::releaseSignals(){
+ theNdb->releaseSignalsInList(&theFirstSCAN_TABINFO_Send);
+ theFirstSCAN_TABINFO_Send = NULL;
+ theLastSCAN_TABINFO_Send = NULL;
+ // theNdb->releaseSignalsInList(&theFirstSCAN_TABINFO_Recv);
+
+ while(theFirstSCAN_TABINFO_Recv != NULL){
+ NdbApiSignal* tmp = theFirstSCAN_TABINFO_Recv;
+ theFirstSCAN_TABINFO_Recv = tmp->next();
+ delete tmp;
+ }
+ theFirstSCAN_TABINFO_Recv = NULL;
+ theLastSCAN_TABINFO_Recv = NULL;
+ if (theSCAN_TABCONF_Recv != NULL){
+ // theNdb->releaseSignal(theSCAN_TABCONF_Recv);
+ delete theSCAN_TABCONF_Recv;
+ theSCAN_TABCONF_Recv = NULL;
+ }
+}
+
+
+void NdbOperation::prepareNextScanResult(){
+ NdbScanReceiver* tScanRec;
+ for (Uint32 i = 0; i < theParallelism; i++) {
+ tScanRec = theScanReceiversArray[i];
+ assert(tScanRec != NULL);
+ tScanRec->prepareNextScanResult();
+ tScanRec->next(NULL);
+ }
+ releaseSignals();
+}
+
+/******************************************************************************
+ * void initScan();
+ *
+ * Return Value: Return 0 : init was successful.
+ * Return -1: In all other case.
+ * Remark: Initiates operation record after allocation.
+ *****************************************************************************/
+void
+NdbOperation::initScan()
+{
+ theTotalRecAI_Len = 0;
+ theCurrRecAI_Len = 0;
+ theStatus = GetValue;
+ theOperationType = OpenScanRequest;
+ theCurrentRecAttr = theFirstRecAttr;
+ theScanInfo = 0;
+ theMagicNumber = 0xABCDEF01;
+ theTotalCurrAI_Len = 5;
+
+ theFirstLabel = NULL;
+ theLastLabel = NULL;
+ theFirstBranch = NULL;
+ theLastBranch = NULL;
+
+ theFirstCall = NULL;
+ theLastCall = NULL;
+ theFirstSubroutine = NULL;
+ theLastSubroutine = NULL;
+
+ theNoOfLabels = 0;
+ theNoOfSubroutines = 0;
+
+ theSubroutineSize = 0;
+ theInitialReadSize = 0;
+ theInterpretedSize = 0;
+ theFinalUpdateSize = 0;
+ theFinalReadSize = 0;
+ theInterpretIndicator = 1;
+
+
+ theFirstSCAN_TABINFO_Send = NULL;
+ theLastSCAN_TABINFO_Send = NULL;
+ theFirstSCAN_TABINFO_Recv = NULL;
+ theLastSCAN_TABINFO_Recv = NULL;
+ theSCAN_TABCONF_Recv = NULL;
+
+ theScanReceiversArray = NULL;
+
+ theTotalBoundAI_Len = 0;
+ theBoundATTRINFO = NULL;
+ return;
+}
+
+NdbOperation* NdbOperation::takeOverForDelete(NdbConnection* updateTrans){
+ return takeOverScanOp(DeleteRequest, updateTrans);
+}
+
+NdbOperation* NdbOperation::takeOverForUpdate(NdbConnection* updateTrans){
+ return takeOverScanOp(UpdateRequest, updateTrans);
+}
+/******************************************************************************
+ * NdbOperation* takeOverScanOp(NdbConnection* updateTrans);
+ *
+ * Parameters: The update transactions NdbConnection pointer.
+ * Return Value: A reference to the transferred operation object
+ * or NULL if no success.
+ * Remark: Take over the scanning transactions NdbOperation
+ * object for a tuple to an update transaction,
+ * which is the last operation read in nextScanResult()
+ * (theNdbCon->thePreviousScanRec)
+ *
+ * FUTURE IMPLEMENTATION: (This note was moved from header file.)
+ * In the future, it will even be possible to transfer
+ * to a NdbConnection on another Ndb-object.
+ * In this case the receiving NdbConnection-object must call
+ * a method receiveOpFromScan to actually receive the information.
+ * This means that the updating transactions can be placed
+ * in separate threads and thus increasing the parallelism during
+ * the scan process.
+ *****************************************************************************/
+NdbOperation*
+NdbOperation::takeOverScanOp(OperationType opType, NdbConnection* updateTrans)
+{
+ if (opType != UpdateRequest && opType != DeleteRequest) {
+ setErrorCode(4604);
+ return NULL;
+ }
+
+ const NdbScanReceiver* tScanRec = theNdbCon->thePreviousScanRec;
+ if (tScanRec == NULL){
+ // No operation read by nextScanResult
+ setErrorCode(4609);
+ return NULL;
+ }
+
+ if (tScanRec->theFirstKEYINFO20_Recv == NULL){
+ // No KEYINFO20 received
+ setErrorCode(4608);
+ return NULL;
+ }
+
+ NdbOperation * newOp = updateTrans->getNdbOperation(m_currentTable);
+ if (newOp == NULL){
+ return NULL;
+ }
+
+ /**
+ * Copy and caclulate attributes from the scanned operation to the
+ * new operation
+ */
+ const KeyInfo20 * const firstKeyInfo20 =
+ CAST_CONSTPTR(KeyInfo20, tScanRec->theFirstKEYINFO20_Recv->getDataPtr());
+ const Uint32 totalKeyLen = firstKeyInfo20->keyLen;
+ newOp->theTupKeyLen = totalKeyLen;
+
+ newOp->theOperationType = opType;
+ if (opType == DeleteRequest) {
+ newOp->theStatus = GetValue;
+ } else {
+ newOp->theStatus = SetValue;
+ }
+ const Uint32 tScanInfo = firstKeyInfo20->scanInfo_Node & 0xFFFF;
+ const Uint32 tTakeOverNode = firstKeyInfo20->scanInfo_Node >> 16;
+ {
+ UintR scanInfo = 0;
+ TcKeyReq::setTakeOverScanFlag(scanInfo, 1);
+ TcKeyReq::setTakeOverScanNode(scanInfo, tTakeOverNode);
+ TcKeyReq::setTakeOverScanInfo(scanInfo, tScanInfo);
+ newOp->theScanInfo = scanInfo;
+ }
+
+ /**
+ * Copy received KEYINFO20 signals into TCKEYREQ and KEYINFO signals
+ * put them in list of the new op
+ */
+ TcKeyReq * const tcKeyReq =
+ CAST_PTR(TcKeyReq, newOp->theTCREQ->getDataPtrSend());
+
+ // Copy the first 8 words of key info from KEYINF20 into TCKEYREQ
+ for (Uint32 i = 0; i < TcKeyReq::MaxKeyInfo; i++) {
+ tcKeyReq->keyInfo[i] = firstKeyInfo20->keyData[i];
+ }
+ if (totalKeyLen > TcKeyReq::MaxKeyInfo) {
+
+ Uint32 keyWordsCopied = TcKeyReq::MaxKeyInfo;
+
+ // Create KEYINFO signals in newOp
+ for (Uint32 i = keyWordsCopied; i < totalKeyLen; i += KeyInfo::DataLength){
+ NdbApiSignal* tSignal = theNdb->getSignal();
+ if (tSignal == NULL){
+ setErrorCodeAbort(4000);
+ return NULL;
+ }
+ if (tSignal->setSignal(GSN_KEYINFO) == -1){
+ setErrorCodeAbort(4001);
+ return NULL;
+ }
+ tSignal->next(newOp->theFirstKEYINFO);
+ newOp->theFirstKEYINFO = tSignal;
+ }
+
+ // Init pointers to KEYINFO20 signal
+ NdbApiSignal* currKeyInfo20 = tScanRec->theFirstKEYINFO20_Recv;
+ const KeyInfo20 * keyInfo20 =
+ CAST_CONSTPTR(KeyInfo20, currKeyInfo20->getDataPtr());
+ Uint32 posInKeyInfo20 = keyWordsCopied;
+
+ // Init pointers to KEYINFO signal
+ NdbApiSignal* currKeyInfo = newOp->theFirstKEYINFO;
+ KeyInfo * keyInfo = CAST_PTR(KeyInfo, currKeyInfo->getDataPtrSend());
+ Uint32 posInKeyInfo = 0;
+
+ // Copy from KEYINFO20 to KEYINFO
+ while(keyWordsCopied < totalKeyLen){
+ keyInfo->keyData[posInKeyInfo++] = keyInfo20->keyData[posInKeyInfo20++];
+ keyWordsCopied++;
+ if(keyWordsCopied >= totalKeyLen)
+ break;
+ if (posInKeyInfo20 >=
+ (currKeyInfo20->getLength()-KeyInfo20::HeaderLength)){
+ currKeyInfo20 = currKeyInfo20->next();
+ keyInfo20 = CAST_CONSTPTR(KeyInfo20, currKeyInfo20->getDataPtr());
+ posInKeyInfo20 = 0;
+ }
+ if (posInKeyInfo >= KeyInfo::DataLength){
+ currKeyInfo = currKeyInfo->next();
+ keyInfo = CAST_PTR(KeyInfo, currKeyInfo->getDataPtrSend());
+ posInKeyInfo = 0;
+ }
+ }
+ }
+
+ return newOp;
+}
+
+
+
diff --git a/ndb/src/ndbapi/NdbOperationSearch.cpp b/ndb/src/ndbapi/NdbOperationSearch.cpp
new file mode 100644
index 00000000000..42f2b1d10d8
--- /dev/null
+++ b/ndb/src/ndbapi/NdbOperationSearch.cpp
@@ -0,0 +1,500 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/******************************************************************************
+Name: NdbOperationSearch.C
+Include:
+Link:
+Author: UABMNST Mona Natterkvist UAB/B/SD
+Date: 970829
+Version: 0.1
+Description: Interface between TIS and NDB
+Documentation:
+Adjust: 971022 UABMNST First version.
+ 971206 UABRONM
+ *****************************************************************************/
+#include "API.hpp"
+
+#include <NdbOperation.hpp>
+#include "NdbApiSignal.hpp"
+#include <NdbConnection.hpp>
+#include <Ndb.hpp>
+#include "NdbImpl.hpp"
+#include <NdbOut.hpp>
+#include "AttrType.hpp"
+
+#include <AttributeHeader.hpp>
+#include <signaldata/TcKeyReq.hpp>
+#include "NdbDictionaryImpl.hpp"
+
+/******************************************************************************
+CondIdType equal(const char* anAttrName, char* aValue, Uint32 aVarKeylen);
+
+Return Value Return 0 : Equal was successful.
+ Return -1: In all other case.
+Parameters: anAttrName : Attribute name for search condition..
+ aValue : Referense to the search value.
+ aVariableKeylen : The length of key in bytes
+Remark: Defines search condition with equality anAttrName.
+******************************************************************************/
+int
+NdbOperation::equal_impl(const NdbColumnImpl* tAttrInfo,
+ const char* aValuePassed,
+ Uint32 aVariableKeyLen)
+{
+ register Uint32 tAttrId;
+
+ Uint32 tData;
+ Uint32 tKeyInfoPosition;
+ const char* aValue = aValuePassed;
+ Uint32 tempData[1024];
+
+ if ((theStatus == OperationDefined) &&
+ (aValue != NULL) &&
+ (tAttrInfo != NULL )) {
+/******************************************************************************
+ * Start by checking that the attribute is a tuple key.
+ * This value is also the word order in the tuple key of this
+ * tuple key attribute.
+ * Then check that this tuple key has not already been defined.
+ * Finally check if all tuple key attributes have been defined. If
+ * this is true then set Operation state to tuple key defined.
+ *****************************************************************************/
+ tAttrId = tAttrInfo->m_attrId;
+ tKeyInfoPosition = tAttrInfo->m_keyInfoPos;
+ Uint32 i = 0;
+ if (tAttrInfo->m_pk) {
+ Uint32 tKeyDefined = theTupleKeyDefined[0][2];
+ Uint32 tKeyAttrId = theTupleKeyDefined[0][0];
+ do {
+ if (tKeyDefined == false) {
+ goto keyEntryFound;
+ } else {
+ if (tKeyAttrId != tAttrId) {
+ /******************************************************************
+ * We read the key defined variable in advance.
+ * It could potentially read outside its area when
+ * i = MAXNROFTUPLEKEY - 1,
+ * it is not a problem as long as the variable
+ * theTupleKeyDefined is defined
+ * in the middle of the object.
+ * Reading wrong data and not using it causes no problems.
+ *****************************************************************/
+ i++;
+ tKeyAttrId = theTupleKeyDefined[i][0];
+ tKeyDefined = theTupleKeyDefined[i][2];
+ continue;
+ } else {
+ goto equal_error2;
+ }//if
+ }//if
+ } while (i < MAXNROFTUPLEKEY);
+ goto equal_error2;
+ } else {
+ goto equal_error1;
+ }
+ /*************************************************************************
+ * Now it is time to retrieve the tuple key data from the pointer supplied
+ * by the application.
+ * We have to retrieve the size of the attribute in words and bits.
+ *************************************************************************/
+ keyEntryFound:
+ theTupleKeyDefined[i][0] = tAttrId;
+ theTupleKeyDefined[i][1] = tKeyInfoPosition;
+ theTupleKeyDefined[i][2] = true;
+
+ Uint32 sizeInBytes = tAttrInfo->m_attrSize * tAttrInfo->m_arraySize;
+ Uint32 bitsInLastWord = 8 * (sizeInBytes & 3) ;
+ Uint32 totalSizeInWords = (sizeInBytes + 3)/4; // Inc. bits in last word
+ Uint32 sizeInWords = sizeInBytes / 4; // Exc. bits in last word
+
+ if (true){ //tArraySize != 0) {
+ Uint32 tTupKeyLen = theTupKeyLen;
+
+ theTupKeyLen = tTupKeyLen + totalSizeInWords;
+ if ((aVariableKeyLen == sizeInBytes) ||
+ (aVariableKeyLen == 0)) {
+ ;
+ } else {
+ goto equal_error3;
+ }
+ }
+#if 0
+ else {
+ /************************************************************************
+ * The attribute is a variable array. We need to use the length parameter
+ * to know the size of this attribute in the key information and
+ * variable area. A key is however not allowed to be larger than 4
+ * kBytes and this is checked for variable array attributes
+ * used as keys.
+ ************************************************************************/
+ Uint32 tMaxVariableKeyLenInWord = (MAXTUPLEKEYLENOFATTERIBUTEINWORD -
+ tKeyInfoPosition);
+ tAttrSizeInBits = aVariableKeyLen << 3;
+ tAttrSizeInWords = tAttrSizeInBits >> 5;
+ tAttrBitsInLastWord = tAttrSizeInBits - (tAttrSizeInWords << 5);
+ tAttrLenInWords = ((tAttrSizeInBits + 31) >> 5);
+ if (tAttrLenInWords > tMaxVariableKeyLenInWord) {
+ setErrorCodeAbort(4207);
+ return -1;
+ }//if
+ theTupKeyLen = theTupKeyLen + tAttrLenInWords;
+ }//if
+#endif
+ /***************************************************************************
+ * Check if the pointer of the value passed is aligned on a 4 byte
+ * boundary. If so only assign the pointer to the internal variable
+ * aValue. If it is not aligned then we start by copying the value to
+ * tempData and use this as aValue instead.
+ *****************************************************************************/
+ const int attributeSize = sizeInBytes;
+ const int slack = sizeInBytes & 3;
+ int tDistrKey = tAttrInfo->m_distributionKey;
+ int tDistrGroup = tAttrInfo->m_distributionGroup;
+ if ((((UintPtr)aValue & 3) != 0) || (slack != 0)){
+ memcpy(&tempData[0], aValue, attributeSize);
+ aValue = (char*)&tempData[0];
+ if(slack != 0) {
+ char * tmp = (char*)&tempData[0];
+ memset(&tmp[attributeSize], 0, (4 - slack));
+ }//if
+ }//if
+ OperationType tOpType = theOperationType;
+ if ((tDistrKey != 1) && (tDistrGroup != 1)) {
+ ;
+ } else if (tDistrKey == 1) {
+ theDistrKeySize += totalSizeInWords;
+ theDistrKeyIndicator = 1;
+ } else {
+ Uint32 TsizeInBytes = sizeInBytes;
+ Uint32 TbyteOrderFix = 0;
+ char* TcharByteOrderFix = (char*)&TbyteOrderFix;
+ if (tAttrInfo->m_distributionGroupBits == 8) {
+ char tFirstChar = aValue[TsizeInBytes - 2];
+ char tSecondChar = aValue[TsizeInBytes - 2];
+ TcharByteOrderFix[0] = tFirstChar;
+ TcharByteOrderFix[1] = tSecondChar;
+ TcharByteOrderFix[2] = 0x30;
+ TcharByteOrderFix[3] = 0x30;
+ theDistrGroupType = 0;
+ } else {
+ TbyteOrderFix = ((aValue[TsizeInBytes - 2] - 0x30) * 10)
+ + (aValue[TsizeInBytes - 1] - 0x30);
+ theDistrGroupType = 1;
+ }//if
+ theDistributionGroup = TbyteOrderFix;
+ theDistrGroupIndicator = 1;
+ }//if
+ /******************************************************************************
+ * If the operation is an insert request and the attribute is stored then
+ * we also set the value in the stored part through putting the
+ * information in the ATTRINFO signals.
+ *****************************************************************************/
+ if ((tOpType == InsertRequest) ||
+ (tOpType == WriteRequest)) {
+ if (!tAttrInfo->m_indexOnly){
+ Uint32 ahValue;
+ const Uint32 sz = totalSizeInWords;
+ AttributeHeader::init(&ahValue, tAttrId, sz);
+ insertATTRINFO( ahValue );
+ insertATTRINFOloop((Uint32*)aValue, sizeInWords);
+ if (bitsInLastWord != 0) {
+ tData = *(Uint32*)(aValue + (sizeInWords << 2));
+ tData = convertEndian(tData);
+ tData = tData & ((1 << bitsInLastWord) - 1);
+ tData = convertEndian(tData);
+ insertATTRINFO( tData );
+ }//if
+ }//if
+ }//if
+
+ /***************************************************************************
+ * Store the Key information in the TCKEYREQ and KEYINFO signals.
+ **************************************************************************/
+ if (insertKEYINFO(aValue, tKeyInfoPosition,
+ totalSizeInWords, bitsInLastWord) != -1) {
+ /*************************************************************************
+ * Add one to number of tuple key attributes defined.
+ * If all have been defined then set the operation state to indicate
+ * that tuple key is defined.
+ * Thereby no more search conditions are allowed in this version.
+ ************************************************************************/
+ Uint32 tNoKeysDef = theNoOfTupKeyDefined;
+ Uint32 tErrorLine = theErrorLine;
+ int tNoTableKeys = m_currentTable->m_noOfKeys;
+ unsigned char tInterpretInd = theInterpretIndicator;
+ tNoKeysDef++;
+ theNoOfTupKeyDefined = tNoKeysDef;
+ tErrorLine++;
+ theErrorLine = tErrorLine;
+ if (int(tNoKeysDef) == tNoTableKeys) {
+ if (tOpType == UpdateRequest) {
+ if (tInterpretInd == 1) {
+ theStatus = GetValue;
+ } else {
+ theStatus = SetValue;
+ }//if
+ return 0;
+ } else if ((tOpType == ReadRequest) || (tOpType == DeleteRequest) ||
+ (tOpType == ReadExclusive)) {
+ theStatus = GetValue;
+ return 0;
+ } else if ((tOpType == InsertRequest) || (tOpType == WriteRequest)) {
+ theStatus = SetValue;
+ return 0;
+ } else {
+ setErrorCodeAbort(4005);
+ return -1;
+ }//if
+ }//if
+ return 0;
+ } else {
+ return -1;
+ }//if
+ }
+
+ if (aValue == NULL) {
+ // NULL value in primary key
+ setErrorCodeAbort(4505);
+ return -1;
+ }//if
+
+ if ( tAttrInfo == NULL ) {
+ // Attribute name not found in table
+ setErrorCodeAbort(4004);
+ return -1;
+ }//if
+
+ if (theStatus == GetValue || theStatus == SetValue){
+ // All pk's defined
+ setErrorCodeAbort(4225);
+ return -1;
+ }//if
+
+ // If we come here, set a general errorcode
+ // and exit
+ setErrorCodeAbort(4200);
+ return -1;
+
+ equal_error1:
+ setErrorCodeAbort(4205);
+ return -1;
+
+ equal_error2:
+ setErrorCodeAbort(4206);
+ return -1;
+
+ equal_error3:
+ setErrorCodeAbort(4209);
+ return -1;
+}
+
+/******************************************************************************
+ * Uint64 setTupleId( void )
+ *
+ * Return Value: Return > 0: OK
+ * Return 0 : setTupleId failed
+ * Parameters:
+ * Remark:
+ *****************************************************************************/
+Uint64
+NdbOperation::setTupleId()
+{
+ if (theStatus != OperationDefined)
+ {
+ return 0;
+ }
+ Uint64 tTupleId = theNdb->getTupleIdFromNdb(m_currentTable->m_tableId);
+ if (tTupleId == ~0){
+ setErrorCodeAbort(theNdb->theError.code);
+ return 0;
+ }
+ if (equal((Uint32)0, tTupleId) == -1)
+ return 0;
+
+ return tTupleId;
+}
+
+/******************************************************************************
+ * int insertKEYINFO(const char* aValue, aStartPosition,
+ * anAttrSizeInWords, Uint32 anAttrBitsInLastWord);
+ *
+ * Return Value: Return 0 : insertKEYINFO was succesful.
+ * Return -1: In all other case.
+ * Parameters: aValue: the data to insert into KEYINFO.
+ * aStartPosition : Start position for Tuplekey in
+ * KEYINFO (TCKEYREQ).
+ * aKeyLenInByte : Length of tuplekey or part of tuplekey
+ * anAttrBitsInLastWord : Nr of bits in last word.
+ * Remark: Puts the the data into either TCKEYREQ signal
+ * or KEYINFO signal.
+ *****************************************************************************/
+int
+NdbOperation::insertKEYINFO(const char* aValue,
+ register Uint32 aStartPosition,
+ register Uint32 anAttrSizeInWords,
+ register Uint32 anAttrBitsInLastWord)
+{
+ NdbApiSignal* tSignal;
+ NdbApiSignal* tCurrentKEYINFO;
+ //register NdbApiSignal* tTCREQ = theTCREQ;
+ register Uint32 tAttrPos;
+ Uint32 tPosition;
+ Uint32 tEndPos;
+ Uint32 tPos;
+ Uint32 signalCounter;
+ Uint32 tData;
+
+/*****************************************************************************
+ * Calculate the end position of the attribute in the key information. *
+ * Since the first attribute starts at position one we need to subtract *
+ * one to get the correct end position. *
+ * We must also remember the last word with only partial information. *
+ *****************************************************************************/
+ tEndPos = aStartPosition + anAttrSizeInWords - 1;
+
+ if ((tEndPos < 9) && (anAttrBitsInLastWord == 0)) {
+ register Uint32 tkeyData = *(Uint32*)aValue;
+ //TcKeyReq* tcKeyReq = CAST_PTR(TcKeyReq, tTCREQ->getDataPtrSend());
+ register Uint32* tDataPtr = (Uint32*)aValue;
+ tAttrPos = 1;
+ register Uint32* tkeyDataPtr = theKEYINFOptr + aStartPosition - 1;
+ // (Uint32*)&tcKeyReq->keyInfo[aStartPosition - 1];
+ do {
+ tDataPtr++;
+ *tkeyDataPtr = tkeyData;
+ if (tAttrPos < anAttrSizeInWords) {
+ ;
+ } else {
+ return 0;
+ }//if
+ tkeyData = *tDataPtr;
+ tkeyDataPtr++;
+ tAttrPos++;
+ } while (1);
+ return 0;
+ }//if
+/*****************************************************************************
+ * Allocate all the KEYINFO signals needed for this key before starting *
+ * to fill the signals with data. This simplifies error handling and *
+ * avoids duplication of code. *
+ *****************************************************************************/
+ tAttrPos = 0;
+ signalCounter = 1;
+ while(tEndPos > theTotalNrOfKeyWordInSignal)
+ {
+ tSignal = theNdb->getSignal();
+ if (tSignal == NULL)
+ {
+ setErrorCodeAbort(4000);
+ return -1;
+ }
+ if (tSignal->setSignal(m_keyInfoGSN) == -1)
+ {
+ setErrorCodeAbort(4001);
+ return -1;
+ }
+ if (theFirstKEYINFO != NULL)
+ theLastKEYINFO->next(tSignal);
+ else
+ theFirstKEYINFO = tSignal;
+ theLastKEYINFO = tSignal;
+ theLastKEYINFO->next(NULL);
+ theTotalNrOfKeyWordInSignal += 20;
+ }
+
+/*****************************************************************************
+ * Change to variable tPosition for more appropriate naming of rest of *
+ * the code. We must set up current KEYINFO already here if the last *
+ * word is a word which is set at LastWordLabel and at the same time *
+ * this is the first word in a KEYINFO signal. *
+ *****************************************************************************/
+ tPosition = aStartPosition;
+ tCurrentKEYINFO = theFirstKEYINFO;
+
+/*****************************************************************************
+ * Start by filling up Key information in the 8 words allocated in the *
+ * TC[KEY/INDX]REQ signal. *
+ *****************************************************************************/
+ while (tPosition < 9)
+ {
+ theKEYINFOptr[tPosition-1] = * (Uint32*)(aValue + (tAttrPos << 2));
+ tAttrPos++;
+ if (anAttrSizeInWords == tAttrPos)
+ goto LastWordLabel;
+ tPosition++;
+ }
+
+/*****************************************************************************
+ * We must set up the start position of the writing of Key information *
+ * before we start the writing of KEYINFO signals. If the start is not *
+ * the first word of the first KEYINFO signals then we must step forward*
+ * to the proper KEYINFO signal and set the signalCounter properly. *
+ * signalCounter is set to the actual position in the signal ( = 4 for *
+ * first key word in KEYINFO signal. *
+ *****************************************************************************/
+ tPos = 8;
+ while ((tPosition - tPos) > 20)
+ {
+ tCurrentKEYINFO = tCurrentKEYINFO->next();
+ tPos += 20;
+ }
+ signalCounter = tPosition - tPos + 3;
+
+/*****************************************************************************
+ * The loop that actually fills in the Key information into the KEYINFO *
+ * signals. Can be optimised by writing larger chunks than 4 bytes at a *
+ * time. *
+ *****************************************************************************/
+ do
+ {
+ if (signalCounter > 23)
+ {
+ tCurrentKEYINFO = tCurrentKEYINFO->next();
+ signalCounter = 4;
+ }
+ tCurrentKEYINFO->setData(*(Uint32*)(aValue + (tAttrPos << 2)),
+ signalCounter);
+ tAttrPos++;
+ if (anAttrSizeInWords == tAttrPos)
+ goto LastWordLabel;
+ tPosition++;
+ signalCounter++;
+ } while (1);
+
+LastWordLabel:
+
+/*****************************************************************************
+ * There could be a last word that only contains partial data. This word*
+ * will contain zeroes in the rest of the bits since the index expects *
+ * a certain number of words and do not care for parts of words. *
+ *****************************************************************************/
+ if (anAttrBitsInLastWord != 0) {
+ tData = *(Uint32*)(aValue + (anAttrSizeInWords - 1) * 4);
+ tData = convertEndian(tData);
+ tData = tData & ((1 << anAttrBitsInLastWord) - 1);
+ tData = convertEndian(tData);
+ if (tPosition > 8) {
+ tCurrentKEYINFO->setData(tData, signalCounter);
+ signalCounter++;
+ } else {
+ theTCREQ->setData(tData, (12 + tPosition));
+ }//if
+ }//if
+
+ return 0;
+}
diff --git a/ndb/src/ndbapi/NdbPool.cpp b/ndb/src/ndbapi/NdbPool.cpp
new file mode 100644
index 00000000000..ba58520c1dc
--- /dev/null
+++ b/ndb/src/ndbapi/NdbPool.cpp
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <Ndb.hpp>
+#include "NdbPoolImpl.hpp"
+#include <NdbPool.hpp>
+
+static NdbPool* m_pool = 0;
+
+bool
+create_instance(Uint32 max_ndb_objects,
+ Uint32 no_conn_obj,
+ Uint32 init_no_ndb_objects)
+{
+ if (m_pool != NULL) {
+ return false;
+ }
+ m_pool = NdbPool::create_instance(max_ndb_objects,
+ no_conn_obj,
+ init_no_ndb_objects);
+ if (m_pool == NULL) {
+ return false;
+ }
+ return true;
+}
+
+void
+drop_instance()
+{
+ if (m_pool == NULL) {
+ return;
+ }
+ NdbPool::drop_instance();
+ m_pool = NULL;
+}
+
+Ndb*
+get_ndb_object(Uint32 &hint_id,
+ const char* a_catalog_name,
+ const char* a_schema_name)
+{
+ if (m_pool == NULL) {
+ return NULL;
+ }
+ return m_pool->get_ndb_object(hint_id, a_catalog_name, a_schema_name);
+}
+
+void
+return_ndb_object(Ndb* returned_object, Uint32 id)
+{
+ if (m_pool == NULL) {
+ return;
+ }
+ m_pool->return_ndb_object(returned_object, id);
+}
+
diff --git a/ndb/src/ndbapi/NdbPoolImpl.cpp b/ndb/src/ndbapi/NdbPoolImpl.cpp
new file mode 100644
index 00000000000..08252d26d79
--- /dev/null
+++ b/ndb/src/ndbapi/NdbPoolImpl.cpp
@@ -0,0 +1,527 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbPoolImpl.hpp"
+#include <assert.h>
+#include <string.h>
+
+NdbMutex *NdbPool::pool_mutex = NULL;
+NdbPool *the_pool = NULL;
+
+NdbPool*
+NdbPool::create_instance(Uint32 max_ndb_obj,
+ Uint32 no_conn_obj,
+ Uint32 init_no_ndb_objects)
+{
+ if (!initPoolMutex()) {
+ return NULL;
+ }
+ NdbMutex_Lock(pool_mutex);
+ NdbPool* a_pool;
+ if (the_pool != NULL) {
+ a_pool = NULL;
+ } else {
+ the_pool = new NdbPool(max_ndb_obj, no_conn_obj);
+ if (!the_pool->init(init_no_ndb_objects)) {
+ delete the_pool;
+ the_pool = NULL;
+ }
+ a_pool = the_pool;
+ }
+ NdbMutex* temp = pool_mutex;
+ if (a_pool == NULL) {
+ pool_mutex = NULL;
+ }
+ NdbMutex_Unlock(pool_mutex);
+ if (a_pool == NULL) {
+ NdbMutex_Destroy(temp);
+ }
+ return a_pool;
+}
+
+void
+NdbPool::drop_instance()
+{
+ if (pool_mutex == NULL) {
+ return;
+ }
+ NdbMutex_Lock(pool_mutex);
+ the_pool->release_all();
+ delete the_pool;
+ the_pool = NULL;
+ NdbMutex* temp = pool_mutex;
+ NdbMutex_Unlock(temp);
+ NdbMutex_Destroy(temp);
+}
+
+bool
+NdbPool::initPoolMutex()
+{
+ bool ret_result = false;
+ if (pool_mutex == NULL) {
+ pool_mutex = NdbMutex_Create();
+ ret_result = ((pool_mutex == NULL) ? false : true);
+ }
+ return ret_result;
+}
+
+NdbPool::NdbPool(Uint32 max_no_objects,
+ Uint32 no_conn_objects)
+{
+ if (no_conn_objects > 1024) {
+ no_conn_objects = 1024;
+ }
+ if (max_no_objects > MAX_NDB_OBJECTS) {
+ max_no_objects = MAX_NDB_OBJECTS;
+ } else if (max_no_objects == 0) {
+ max_no_objects = 1;
+ }
+ m_max_ndb_objects = max_no_objects;
+ m_no_of_conn_objects = no_conn_objects;
+ m_no_of_objects = 0;
+ m_waiting = 0;
+ m_pool_reference = NULL;
+ m_hash_entry = NULL;
+ m_first_free = NULL_POOL;
+ m_first_not_in_use = NULL_POOL;
+ m_last_free = NULL_POOL;
+ input_pool_cond = NULL;
+ output_pool_cond = NULL;
+ m_output_queue = 0;
+ m_input_queue = 0;
+ m_signal_count = 0;
+}
+
+NdbPool::~NdbPool()
+{
+ NdbCondition_Destroy(input_pool_cond);
+ NdbCondition_Destroy(output_pool_cond);
+}
+
+void
+NdbPool::release_all()
+{
+ int i;
+ for (i = 0; i < m_max_ndb_objects + 1; i++) {
+ if (m_pool_reference[i].ndb_reference != NULL) {
+ assert(m_pool_reference[i].in_use);
+ assert(m_pool_reference[i].free_entry);
+ delete m_pool_reference[i].ndb_reference;
+ }
+ }
+ delete [] m_pool_reference;
+ delete [] m_hash_entry;
+ m_pool_reference = NULL;
+ m_hash_entry = NULL;
+}
+
+bool
+NdbPool::init(Uint32 init_no_objects)
+{
+ bool ret_result = false;
+ int i;
+ do {
+ input_pool_cond = NdbCondition_Create();
+ output_pool_cond = NdbCondition_Create();
+ if (input_pool_cond == NULL || output_pool_cond == NULL) {
+ break;
+ }
+ if (init_no_objects > m_max_ndb_objects) {
+ init_no_objects = m_max_ndb_objects;
+ }
+ if (init_no_objects == 0) {
+ init_no_objects = 1;
+ }
+ m_pool_reference = new NdbPool::POOL_STRUCT[m_max_ndb_objects + 1];
+ m_hash_entry = new Uint8[POOL_HASH_TABLE_SIZE];
+
+ if ((m_pool_reference == NULL) || (m_hash_entry == NULL)) {
+ delete [] m_pool_reference;
+ delete [] m_hash_entry;
+ break;
+ }
+ for (i = 0; i < m_max_ndb_objects + 1; i++) {
+ m_pool_reference[i].ndb_reference = NULL;
+ m_pool_reference[i].in_use = false;
+ m_pool_reference[i].next_free_object = i+1;
+ m_pool_reference[i].prev_free_object = i-1;
+ m_pool_reference[i].next_db_object = NULL_POOL;
+ m_pool_reference[i].prev_db_object = NULL_POOL;
+ }
+ for (i = 0; i < POOL_HASH_TABLE_SIZE; i++) {
+ m_hash_entry[i] = NULL_HASH;
+ }
+ m_pool_reference[m_max_ndb_objects].next_free_object = NULL_POOL;
+ m_pool_reference[1].prev_free_object = NULL_POOL;
+ m_first_not_in_use = 1;
+ m_no_of_objects = init_no_objects;
+ for (i = init_no_objects; i > 0 ; i--) {
+ Uint32 fake_id;
+ if (!allocate_ndb(fake_id, (const char*)NULL, (const char*)NULL)) {
+ release_all();
+ break;
+ }
+ }
+ ret_result = true;
+ break;
+ } while (1);
+ return ret_result;
+}
+
+/*
+Get an Ndb object.
+Input:
+hint_id: 0 = no hint, otherwise a hint of which Ndb object the thread
+ used the last time.
+a_db_name: NULL = don't check for database specific Ndb object, otherwise
+ a hint of which database is preferred.
+Output:
+hint_id: Returns id of Ndb object returned
+Return value: Ndb object pointer
+*/
+Ndb*
+NdbPool::get_ndb_object(Uint32 &hint_id,
+ const char* a_catalog_name,
+ const char* a_schema_name)
+{
+ Ndb* ret_ndb = NULL;
+ Uint32 hash_entry = compute_hash(a_schema_name);
+ NdbMutex_Lock(pool_mutex);
+ while (1) {
+ /*
+ We start by checking if we can use the hinted Ndb object.
+ */
+ if ((ret_ndb = get_hint_ndb(hint_id, hash_entry)) != NULL) {
+ break;
+ }
+ /*
+ The hinted Ndb object was not free. We need to allocate another object.
+ We start by checking for a free Ndb object connected to the same database.
+ */
+ if (a_schema_name && (ret_ndb = get_db_hash(hint_id,
+ hash_entry,
+ a_catalog_name,
+ a_schema_name))) {
+ break;
+ }
+ /*
+ No Ndb object connected to the preferred database was found.
+ We look for a free Ndb object in general.
+ */
+ if ((ret_ndb = get_free_list(hint_id, hash_entry)) != NULL) {
+ break;
+ }
+ /*
+ No free Ndb object was found. If we haven't allocated objects up until the
+ maximum number yet then we can allocate a new Ndb object here.
+ */
+ if (m_no_of_objects < m_max_ndb_objects) {
+ if (allocate_ndb(hint_id, a_catalog_name, a_schema_name)) {
+ assert((ret_ndb = get_hint_ndb(hint_id, hash_entry)) != NULL);
+ break;
+ }
+ }
+ /*
+ We need to wait until an Ndb object becomes
+ available.
+ */
+ if ((ret_ndb = wait_free_ndb(hint_id)) != NULL) {
+ break;
+ }
+ /*
+ Not even after waiting were we able to get hold of an Ndb object. We
+ return NULL to indicate this problem.
+ */
+ ret_ndb = NULL;
+ break;
+ }
+ NdbMutex_Unlock(pool_mutex);
+ if (ret_ndb != NULL) {
+ /*
+ We need to set the catalog and schema name of the Ndb object before
+ returning it to the caller.
+ */
+ ret_ndb->setCatalogName(a_catalog_name);
+ ret_ndb->setSchemaName(a_schema_name);
+ }
+ return ret_ndb;
+}
+
+void
+NdbPool::return_ndb_object(Ndb* returned_ndb, Uint32 id)
+{
+ NdbMutex_Lock(pool_mutex);
+ assert(id <= m_max_ndb_objects);
+ assert(id != 0);
+ assert(returned_ndb == m_pool_reference[id].ndb_reference);
+ bool wait_cond = m_waiting;
+ if (wait_cond) {
+ NdbCondition* pool_cond;
+ if (m_signal_count > 0) {
+ pool_cond = output_pool_cond;
+ m_signal_count--;
+ } else {
+ pool_cond = input_pool_cond;
+ }
+ add_wait_list(id);
+ NdbMutex_Unlock(pool_mutex);
+ NdbCondition_Signal(pool_cond);
+ } else {
+ add_free_list(id);
+ add_db_hash(id);
+ NdbMutex_Unlock(pool_mutex);
+ }
+}
+
+bool
+NdbPool::allocate_ndb(Uint32 &id,
+ const char* a_catalog_name,
+ const char* a_schema_name)
+{
+ Ndb* a_ndb;
+ if (m_first_not_in_use == NULL_POOL) {
+ return false;
+ }
+ if (a_schema_name) {
+ a_ndb = new Ndb(a_schema_name, a_catalog_name);
+ } else {
+ a_ndb = new Ndb("");
+ }
+ if (a_ndb == NULL) {
+ return false;
+ }
+ a_ndb->init(m_no_of_conn_objects);
+ m_no_of_objects++;
+
+ id = m_first_not_in_use;
+ Uint32 allocated_id = m_first_not_in_use;
+ m_first_not_in_use = m_pool_reference[allocated_id].next_free_object;
+
+ m_pool_reference[allocated_id].ndb_reference = a_ndb;
+ m_pool_reference[allocated_id].in_use = true;
+ m_pool_reference[allocated_id].free_entry = false;
+
+ add_free_list(allocated_id);
+ add_db_hash(allocated_id);
+ return true;
+}
+
+void
+NdbPool::add_free_list(Uint32 id)
+{
+ assert(!m_pool_reference[id].free_entry);
+ assert(m_pool_reference[id].in_use);
+ m_pool_reference[id].free_entry = true;
+ m_pool_reference[id].next_free_object = m_first_free;
+ m_pool_reference[id].prev_free_object = (Uint8)NULL_POOL;
+ m_first_free = (Uint8)id;
+ if (m_last_free == (Uint8)NULL_POOL) {
+ m_last_free = (Uint8)id;
+ }
+}
+
+void
+NdbPool::add_db_hash(Uint32 id)
+{
+ Ndb* t_ndb = m_pool_reference[id].ndb_reference;
+ const char* schema_name = t_ndb->getSchemaName();
+ Uint32 hash_entry = compute_hash(schema_name);
+ Uint8 next_db_entry = m_hash_entry[hash_entry];
+ m_pool_reference[id].next_db_object = next_db_entry;
+ m_pool_reference[id].prev_db_object = (Uint8)NULL_HASH;
+ m_hash_entry[hash_entry] = (Uint8)id;
+}
+
+Ndb*
+NdbPool::get_free_list(Uint32 &id, Uint32 hash_entry)
+{
+ if (m_first_free == NULL_POOL) {
+ return NULL;
+ }
+ id = m_first_free;
+ Ndb* ret_ndb = get_hint_ndb(m_first_free, hash_entry);
+ assert(ret_ndb != NULL);
+ return ret_ndb;
+}
+
+Ndb*
+NdbPool::get_db_hash(Uint32 &id,
+ Uint32 hash_entry,
+ const char *a_catalog_name,
+ const char *a_schema_name)
+{
+ Uint32 entry_id = m_hash_entry[hash_entry];
+ bool found = false;
+ while (entry_id != NULL_HASH) {
+ Ndb* t_ndb = m_pool_reference[entry_id].ndb_reference;
+ const char *a_ndb_catalog_name = t_ndb->getCatalogName();
+ if (strcmp(a_catalog_name, a_ndb_catalog_name) == 0) {
+ const char *a_ndb_schema_name = t_ndb->getSchemaName();
+ if (strcmp(a_schema_name, a_ndb_schema_name) == 0) {
+ found = true;
+ break;
+ }
+ }
+ entry_id = m_pool_reference[entry_id].next_db_object;
+ }
+ if (found) {
+ id = entry_id;
+ Ndb* ret_ndb = get_hint_ndb(entry_id, hash_entry);
+ assert(ret_ndb != NULL);
+ return ret_ndb;
+ }
+ return NULL;
+}
+
+Ndb*
+NdbPool::get_hint_ndb(Uint32 hint_id, Uint32 hash_entry)
+{
+ Ndb* ret_ndb = NULL;
+ do {
+ if ((hint_id != 0) &&
+ (hint_id <= m_max_ndb_objects) &&
+ (m_pool_reference[hint_id].in_use) &&
+ (m_pool_reference[hint_id].free_entry)) {
+ ret_ndb = m_pool_reference[hint_id].ndb_reference;
+ if (ret_ndb != NULL) {
+ break;
+ } else {
+ assert(false);
+ }
+ }
+ return NULL;
+ } while (1);
+ /*
+ This is where we remove the entry from the free list and from the db hash
+ table.
+ */
+ remove_free_list(hint_id);
+ remove_db_hash(hint_id, hash_entry);
+ return ret_ndb;
+}
+
+void
+NdbPool::remove_free_list(Uint32 id)
+{
+ Uint8 next_free_entry = m_pool_reference[id].next_free_object;
+ Uint8 prev_free_entry = m_pool_reference[id].prev_free_object;
+ if (prev_free_entry == (Uint8)NULL_POOL) {
+ m_first_free = next_free_entry;
+ } else {
+ m_pool_reference[prev_free_entry].next_free_object = next_free_entry;
+ }
+ if (next_free_entry == (Uint8)NULL_POOL) {
+ m_last_free = prev_free_entry;
+ } else {
+ m_pool_reference[next_free_entry].prev_free_object = prev_free_entry;
+ }
+ m_pool_reference[id].next_free_object = NULL_POOL;
+ m_pool_reference[id].prev_free_object = NULL_POOL;
+ m_pool_reference[id].free_entry = false;
+}
+
+void
+NdbPool::remove_db_hash(Uint32 id, Uint32 hash_entry)
+{
+ Uint8 next_free_entry = m_pool_reference[id].next_db_object;
+ Uint8 prev_free_entry = m_pool_reference[id].prev_db_object;
+ if (prev_free_entry == (Uint8)NULL_HASH) {
+ m_hash_entry[hash_entry] = next_free_entry;
+ } else {
+ m_pool_reference[prev_free_entry].next_db_object = next_free_entry;
+ }
+ if (next_free_entry == (Uint8)NULL_HASH) {
+ ;
+ } else {
+ m_pool_reference[next_free_entry].prev_db_object = prev_free_entry;
+ }
+ m_pool_reference[id].next_db_object = NULL_HASH;
+ m_pool_reference[id].prev_db_object = NULL_HASH;
+}
+
+Uint32
+NdbPool::compute_hash(const char *a_schema_name)
+{
+ Uint32 len = strlen(a_schema_name);
+ Uint32 h = 147;
+ for (Uint32 i = 0; i < len; i++) {
+ Uint32 c = a_schema_name[i];
+ h = (h << 5) + h + c;
+ }
+ h &= (POOL_HASH_TABLE_SIZE - 1);
+ return h;
+}
+
+Ndb*
+NdbPool::wait_free_ndb(Uint32 &id)
+{
+ int res;
+ int time_out = 3500;
+ do {
+ NdbCondition* tmp = input_pool_cond;
+ m_waiting++;
+ m_input_queue++;
+ time_out -= 500;
+ res = NdbCondition_WaitTimeout(input_pool_cond, pool_mutex, time_out);
+ if (tmp == input_pool_cond) {
+ m_input_queue--;
+ } else {
+ m_output_queue--;
+ if (m_output_queue == 0) {
+ switch_condition_queue();
+ }
+ }
+ m_waiting--;
+ } while (res == 0 && m_first_wait == NULL_POOL);
+ if (res != 0 && m_first_wait == NULL_POOL) {
+ return NULL;
+ }
+ id = m_first_wait;
+ remove_wait_list();
+ assert(m_waiting != 0 || m_first_wait == NULL_POOL);
+ return m_pool_reference[id].ndb_reference;
+}
+
+void
+NdbPool::remove_wait_list()
+{
+ Uint32 id = m_first_wait;
+ m_first_wait = m_pool_reference[id].next_free_object;
+ m_pool_reference[id].next_free_object = NULL_POOL;
+ m_pool_reference[id].prev_free_object = NULL_POOL;
+ m_pool_reference[id].free_entry = false;
+}
+
+void
+NdbPool::add_wait_list(Uint32 id)
+{
+ m_pool_reference[id].next_free_object = m_first_wait;
+ m_first_wait = id;
+}
+
+void
+NdbPool::switch_condition_queue()
+{
+ m_signal_count = m_input_queue;
+ Uint8 move_queue = m_input_queue;
+ m_input_queue = m_output_queue;
+ m_output_queue = move_queue;
+
+ NdbCondition* move_cond = input_pool_cond;
+ input_pool_cond = output_pool_cond;
+ output_pool_cond = move_cond;
+}
+
diff --git a/ndb/src/ndbapi/NdbPoolImpl.hpp b/ndb/src/ndbapi/NdbPoolImpl.hpp
new file mode 100644
index 00000000000..af6cf4708cf
--- /dev/null
+++ b/ndb/src/ndbapi/NdbPoolImpl.hpp
@@ -0,0 +1,162 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ @section ndbPool Pooling of NDB objects
+ This class implements pooling of NDB objects to support MySQL, ODBC and
+ any application with a great number of threads.
+
+ The general idea is to let the NdbPool class administer all Ndb objects.
+ When a thread needs a Ndb object it request a Ndb object from the Pool.
+ This interface contains some hints to ensure that the proper Ndb object
+ is returned.
+
+ The object contains an array of references to all Ndb objects together with
+ an indication of whether the object is free or not.
+
+ The idea is that the thread should keep track of the Ndb object it used the
+ last time. If this object is still free it will simply get this object
+ back. If the number of threads do not exceed the number of Ndb objects this
+ will always be successful. In certain situations the number of threads will
+ be much greater than the number of Ndb objects. In this situation the Pool
+ will attempt to provide an object that is attached to the same schema
+ as the thread is connected to. If this is not possible it will attempt to
+ get any free Ndb object. If not even this is possible the Pool will wait
+ until an Ndb object becomes free. If an Ndb object becomes available in
+ time it will deliver this Ndb object. In the worst case the call will
+ time-out and return NULL to indicate no free Ndb object was found in time.
+
+ The implementation uses an array of structs which contain a reference to a
+ Ndb object, whether it is in use currently and a number of references to
+ set up linked lists of
+ 1) Free objects on a schema
+ 2) Free objects in LIFO order
+
+ Usage:
+ The class is a singleton.
+ The first step is to call create_instance(..). If successful this will
+ create the NdbPool object and return a reference to it. When completed
+ drop_instance is called to remove the NdbPool object and all memory and
+ other resources attached to it.
+
+ After initialising the NdbPool object all threads can now start using the
+ NdbPool. There are two methods in normal usage mode. The first
+ get_ndb_object gets a Ndb object and the second return_ndb_object returns
+ an Ndb object. The user of the NdbPool must keep track of the identity
+ of the Ndb object. The idea is that this identity can also be used to
+ find the object quickly again unless another thread have taken it. If the
+ user wants any Ndb object it requests identity 0 which means any here.
+
+ When constructing the NdbPool one can set the number of NdbConnection
+ objects which are allowed in all Ndb objects. For use in synchronous
+ applications such as the MySQL server 4 objects should be enough. When
+ using the NdbPool for asynchronous applications one should use 1024 to
+ enable a high level of parallelism. It is also possible to set the
+ maximum number of Ndb objects in the pool and the initial number of
+ Ndb objects allocated.
+*/
+
+#ifndef NdbPool_H
+#define NdbPool_H
+
+#include <Ndb.hpp>
+#include <NdbMutex.h>
+#include <NdbCondition.h>
+#include <NdbOut.hpp>
+
+class NdbPool {
+#define NULL_POOL 0
+#define NULL_HASH 0xFF
+#define POOL_HASH_TABLE_SIZE 32
+#define MAX_NDB_OBJECTS 240
+ struct POOL_STRUCT {
+ Ndb* ndb_reference;
+ bool in_use;
+ bool free_entry;
+ Uint16 next_free_object;
+ Uint16 prev_free_object;
+ Uint16 next_db_object;
+ Uint16 prev_db_object;
+ };
+ public:
+ static NdbPool* create_instance(Uint32 max_ndb_objects = 240,
+ Uint32 no_conn_obj = 4,
+ Uint32 init_no_ndb_objects = 8);
+ static void drop_instance();
+ Ndb* get_ndb_object(Uint32 &hint_id,
+ const char* a_catalog_name,
+ const char* a_schema_name);
+ void return_ndb_object(Ndb* returned_object, Uint32 id);
+ private:
+ bool init(Uint32 initial_no_of_ndb_objects = 8);
+ void release_all();
+ static bool initPoolMutex();
+ NdbPool(Uint32 max_no_of_ndb_objects, Uint32 no_conn_objects);
+ ~NdbPool();
+ /*
+ We have three lists:
+ 1) A list for entries not in use
+ 2) A list for free entries
+ 3) A hash table with schema name and database name as key
+
+ These lists are all initialised in the init call.
+ The list for entries not in use is very simple since the current
+ implementation have not yet any handling of dropping Ndb objects
+ until all Ndb objects are dropped.
+ */
+ void add_free_list(Uint32 id);
+ void remove_free_list(Uint32 id);
+ Ndb* get_free_list(Uint32 &id, Uint32 hash_entry);
+
+ void add_db_hash(Uint32 id);
+ void remove_db_hash(Uint32 id, Uint32 hash_entry);
+ Ndb* get_db_hash(Uint32 &id,
+ Uint32 hash_entry,
+ const char* a_catalog_name,
+ const char* a_schema_name);
+
+ bool allocate_ndb(Uint32 &id,
+ const char* a_catalog_name,
+ const char* a_schema_name);
+ Ndb* get_hint_ndb(Uint32 id, Uint32 hash_entry);
+ Ndb* wait_free_ndb(Uint32 &id);
+ Uint32 compute_hash(const char *a_schema_name);
+ void add_wait_list(Uint32 id);
+ void remove_wait_list();
+ void switch_condition_queue();
+
+ static NdbMutex *pool_mutex;
+ struct NdbCondition *input_pool_cond;
+ struct NdbCondition *output_pool_cond;
+
+ POOL_STRUCT *m_pool_reference;
+ Uint8 *m_hash_entry;
+
+ bool m_inited;
+ Uint32 m_no_of_conn_objects;
+
+ Uint16 m_no_of_objects;
+ Uint16 m_max_ndb_objects;
+ Uint16 m_first_free;
+ Uint16 m_last_free;
+ Uint16 m_first_not_in_use;
+ Uint16 m_waiting;
+ Uint16 m_first_wait;
+ Uint16 m_input_queue;
+ Uint16 m_output_queue;
+ Uint16 m_signal_count;
+};
+#endif
diff --git a/ndb/src/ndbapi/NdbRecAttr.cpp b/ndb/src/ndbapi/NdbRecAttr.cpp
new file mode 100644
index 00000000000..11f36fbd2c4
--- /dev/null
+++ b/ndb/src/ndbapi/NdbRecAttr.cpp
@@ -0,0 +1,126 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/************************************************************************************************
+Name: NdbRecAttr.C
+Include:
+Link:
+Author: UABRONM Mikael Ronström UAB/B/SD
+Date: 971206
+Version: 0.1
+Description: Interface between TIS and NDB
+Documentation:
+Adjust: 971206 UABRONM First version
+************************************************************************************************/
+#include "NdbRecAttr.hpp"
+#include <stdlib.h>
+#include "NdbDictionaryImpl.hpp"
+
+NdbRecAttr::NdbRecAttr() :
+ theStorageX(NULL),
+ theValue(NULL),
+ theRef(NULL),
+ theNext(NULL),
+ theAttrId(0xFFFF),
+ theNULLind(-1)
+{
+}
+
+NdbRecAttr::~NdbRecAttr()
+{
+ release();
+}
+
+int
+NdbRecAttr::setup(const NdbColumnImpl* anAttrInfo, char* aValue)
+{
+ Uint32 tAttrSize = anAttrInfo->m_attrSize;
+ Uint32 tArraySize = anAttrInfo->m_arraySize;
+ Uint32 tAttrByteSize = tAttrSize * tArraySize;
+
+ m_column = anAttrInfo;
+ theAttrId = anAttrInfo->m_attrId;
+ theAttrSize = tAttrSize;
+ theArraySize = tArraySize;
+ theValue = aValue;
+
+ // check alignment to signal data
+ // a future version could check alignment per data type as well
+
+ if (aValue != NULL && (UintPtr(aValue)&3) == 0 && (tAttrByteSize&3) == 0) {
+ theStorageX = NULL;
+ theRef = aValue;
+ return 0;
+ }
+ if (tAttrByteSize <= 32) {
+ theStorageX = NULL;
+ theStorage[0] = 0;
+ theStorage[1] = 0;
+ theStorage[2] = 0;
+ theStorage[3] = 0;
+ theRef = theStorage;
+ return 0;
+ }
+ Uint32 tSize = (tAttrByteSize + 7) >> 3;
+ Uint64* tRef = new Uint64[tSize];
+ if (tRef != NULL) {
+ for (Uint32 i = 0; i < tSize; i++) {
+ tRef[i] = 0;
+ }
+ theStorageX = tRef;
+ theRef = tRef;
+ return 0;
+ }
+ return -1;
+}
+
+void
+NdbRecAttr::copyout()
+{
+ char* tRef = (char*)theRef;
+ char* tValue = theValue;
+ if (tRef != tValue && tRef != NULL && tValue != NULL) {
+ Uint32 n = theAttrSize * theArraySize;
+ while (n-- > 0) {
+ *tValue++ = *tRef++;
+ }
+ }
+}
+
+NdbRecAttr *
+NdbRecAttr::clone() const {
+ NdbRecAttr * ret = new NdbRecAttr();
+
+ ret->theAttrId = theAttrId;
+ ret->theNULLind = theNULLind;
+ ret->theAttrSize = theAttrSize;
+ ret->theArraySize = theArraySize;
+ ret->m_column = m_column;
+
+ Uint32 n = theAttrSize * theArraySize;
+ if(n <= 32){
+ ret->theRef = (char*)&ret->theStorage[0];
+ ret->theStorageX = 0;
+ ret->theValue = 0;
+ } else {
+ ret->theStorageX = new Uint64[((n + 7) >> 3)];
+ ret->theRef = (char*)ret->theStorageX;
+ ret->theValue = 0;
+ }
+ memcpy(ret->theRef, theRef, n);
+ return ret;
+}
diff --git a/ndb/src/ndbapi/NdbReceiver.cpp b/ndb/src/ndbapi/NdbReceiver.cpp
new file mode 100644
index 00000000000..4c461698a4a
--- /dev/null
+++ b/ndb/src/ndbapi/NdbReceiver.cpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbImpl.hpp"
+#include <NdbReceiver.hpp>
+
+NdbReceiver::NdbReceiver(Ndb *aNdb) :
+ theMagicNumber(0),
+ m_ndb(aNdb),
+ m_id(NdbObjectIdMap::InvalidId),
+ m_type(NDB_UNINITIALIZED),
+ m_owner(0)
+{
+}
+
+void
+NdbReceiver::init(ReceiverType type, void* owner)
+{
+ theMagicNumber = 0x11223344;
+ m_type = type;
+ m_owner = owner;
+ if (m_id == NdbObjectIdMap::InvalidId) {
+ if (m_ndb)
+ m_id = m_ndb->theNdbObjectIdMap->map(this);
+ }
+}
+
+NdbReceiver::~NdbReceiver()
+{
+ if (m_id != NdbObjectIdMap::InvalidId) {
+ m_ndb->theNdbObjectIdMap->unmap(m_id, this);
+ }
+}
diff --git a/ndb/src/ndbapi/NdbResultSet.cpp b/ndb/src/ndbapi/NdbResultSet.cpp
new file mode 100644
index 00000000000..8397d5eef91
--- /dev/null
+++ b/ndb/src/ndbapi/NdbResultSet.cpp
@@ -0,0 +1,102 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbResultSet.cpp
+ * Include:
+ * Link:
+ * Author: UABMASD Martin Sköld INN/V Alzato
+ * Date: 2002-04-01
+ * Version: 0.1
+ * Description: Cursor class
+ * Documentation:
+ * Adjust: 2002-04-01 UABMASD First version.
+ ****************************************************************************/
+
+#include <Ndb.hpp>
+#include <NdbConnection.hpp>
+#include <NdbResultSet.hpp>
+
+NdbResultSet::NdbResultSet(NdbCursorOperation *owner)
+: m_operation(owner)
+{
+}
+
+NdbResultSet::~NdbResultSet()
+{
+}
+
+void NdbResultSet::init()
+{
+}
+
+int NdbResultSet::nextResult(bool fetchAllowed)
+{
+ return m_operation->nextResult(fetchAllowed);
+}
+
+void NdbResultSet::close()
+{
+ m_operation->closeScan();
+}
+
+NdbOperation*
+NdbResultSet::updateTuple(){
+ if(m_operation->cursorType() != NdbCursorOperation::ScanCursor){
+ m_operation->setErrorCode(4003);
+ return 0;
+ }
+
+ NdbScanOperation * op = (NdbScanOperation*)(m_operation);
+ return op->takeOverScanOp(UpdateRequest, op->m_transConnection);
+}
+
+NdbOperation*
+NdbResultSet::updateTuple(NdbConnection* takeOverTrans){
+ if(m_operation->cursorType() != NdbCursorOperation::ScanCursor){
+ m_operation->setErrorCode(4003);
+ return 0;
+ }
+
+ return m_operation->takeOverScanOp(UpdateRequest, takeOverTrans);
+}
+
+int
+NdbResultSet::deleteTuple(){
+ if(m_operation->cursorType() != NdbCursorOperation::ScanCursor){
+ m_operation->setErrorCode(4003);
+ return 0;
+ }
+
+ NdbScanOperation * op = (NdbScanOperation*)(m_operation);
+ void * res = op->takeOverScanOp(DeleteRequest, op->m_transConnection);
+ if(res == 0)
+ return -1;
+ return 0;
+}
+
+int
+NdbResultSet::deleteTuple(NdbConnection * takeOverTrans){
+ if(m_operation->cursorType() != NdbCursorOperation::ScanCursor){
+ m_operation->setErrorCode(4003);
+ return 0;
+ }
+
+ void * res = m_operation->takeOverScanOp(DeleteRequest, takeOverTrans);
+ if(res == 0)
+ return -1;
+ return 0;
+}
diff --git a/ndb/src/ndbapi/NdbScanFilter.cpp b/ndb/src/ndbapi/NdbScanFilter.cpp
new file mode 100644
index 00000000000..9542b226d7d
--- /dev/null
+++ b/ndb/src/ndbapi/NdbScanFilter.cpp
@@ -0,0 +1,779 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbScanFilter.hpp>
+#include <NdbOperation.hpp>
+#include "NdbDictionaryImpl.hpp"
+#include <Vector.hpp>
+#include <NdbOut.hpp>
+#include <Interpreter.hpp>
+
+#ifdef VM_TRACE
+#include <NdbEnv.h>
+#define INT_DEBUG(x) \
+ { const char* tmp = NdbEnv_GetEnv("INT_DEBUG", (char*)0, 0); \
+ if (tmp != 0 && strlen(tmp) != 0) { ndbout << "INT:"; ndbout_c x; } }
+#else
+#define INT_DEBUG(x)
+#endif
+
+class NdbScanFilterImpl {
+public:
+ struct State {
+ NdbScanFilter::Group m_group;
+ Uint32 m_popCount;
+ Uint32 m_ownLabel;
+ Uint32 m_trueLabel;
+ Uint32 m_falseLabel;
+ };
+
+ int m_label;
+ State m_current;
+ Vector<State> m_stack;
+ NdbOperation * m_operation;
+ Uint32 m_latestAttrib;
+
+ int cond_col(Interpreter::UnaryCondition, Uint32 attrId);
+
+ template<typename T>
+ int cond_col_const(Interpreter::BinaryCondition, Uint32 attrId, T value);
+
+ int cond_col_const(Interpreter::BinaryCondition, Uint32 attrId,
+ const char * value, Uint32 len, bool nopad);
+};
+
+const Uint32 LabelExit = ~0;
+
+
+NdbScanFilter::NdbScanFilter(class NdbOperation * op)
+ : m_impl(* new NdbScanFilterImpl())
+{
+ m_impl.m_current.m_group = (NdbScanFilter::Group)0;
+ m_impl.m_current.m_popCount = 0;
+ m_impl.m_current.m_ownLabel = 0;
+ m_impl.m_current.m_trueLabel = ~0;
+ m_impl.m_current.m_falseLabel = ~0;
+ m_impl.m_label = 0;
+ m_impl.m_latestAttrib = ~0;
+ m_impl.m_operation = op;
+}
+
+NdbScanFilter::~NdbScanFilter(){
+ delete &m_impl;
+}
+
+int
+NdbScanFilter::begin(Group group){
+
+ switch(group){
+ case NdbScanFilter::AND:
+ INT_DEBUG(("Begin(AND)"));
+ break;
+ case NdbScanFilter::OR:
+ INT_DEBUG(("Begin(OR)"));
+ break;
+ case NdbScanFilter::NAND:
+ INT_DEBUG(("Begin(NAND)"));
+ break;
+ case NdbScanFilter::NOR:
+ INT_DEBUG(("Begin(NOR)"));
+ break;
+ }
+
+ if(group == m_impl.m_current.m_group){
+ switch(group){
+ case NdbScanFilter::AND:
+ case NdbScanFilter::OR:
+ m_impl.m_current.m_popCount++;
+ return 0;
+ case NdbScanFilter::NOR:
+ case NdbScanFilter::NAND:
+ break;
+ }
+ }
+
+ NdbScanFilterImpl::State tmp = m_impl.m_current;
+ m_impl.m_stack.push_back(m_impl.m_current);
+ m_impl.m_current.m_group = group;
+ m_impl.m_current.m_ownLabel = m_impl.m_label++;
+ m_impl.m_current.m_popCount = 0;
+
+ switch(group){
+ case NdbScanFilter::AND:
+ case NdbScanFilter::NAND:
+ m_impl.m_current.m_falseLabel = m_impl.m_current.m_ownLabel;
+ m_impl.m_current.m_trueLabel = tmp.m_trueLabel;
+ break;
+ case NdbScanFilter::OR:
+ case NdbScanFilter::NOR:
+ m_impl.m_current.m_falseLabel = tmp.m_falseLabel;
+ m_impl.m_current.m_trueLabel = m_impl.m_current.m_ownLabel;
+ break;
+ default:
+ m_impl.m_operation->setErrorCodeAbort(4260);
+ return -1;
+ }
+
+ return 0;
+}
+
+int
+NdbScanFilter::end(){
+
+ switch(m_impl.m_current.m_group){
+ case NdbScanFilter::AND:
+ INT_DEBUG(("End(AND pc=%d)", m_impl.m_current.m_popCount));
+ break;
+ case NdbScanFilter::OR:
+ INT_DEBUG(("End(OR pc=%d)", m_impl.m_current.m_popCount));
+ break;
+ case NdbScanFilter::NAND:
+ INT_DEBUG(("End(NAND pc=%d)", m_impl.m_current.m_popCount));
+ break;
+ case NdbScanFilter::NOR:
+ INT_DEBUG(("End(NOR pc=%d)", m_impl.m_current.m_popCount));
+ break;
+ }
+
+ if(m_impl.m_current.m_popCount > 0){
+ m_impl.m_current.m_popCount--;
+ return 0;
+ }
+
+ NdbScanFilterImpl::State tmp = m_impl.m_current;
+ m_impl.m_current = m_impl.m_stack.back();
+ m_impl.m_stack.erase(m_impl.m_stack.size() - 1);
+
+ switch(tmp.m_group){
+ case NdbScanFilter::AND:
+ if(tmp.m_trueLabel == (Uint32)~0){
+ m_impl.m_operation->interpret_exit_ok();
+ } else {
+ m_impl.m_operation->branch_label(tmp.m_trueLabel);
+ }
+ break;
+ case NdbScanFilter::NAND:
+ if(tmp.m_trueLabel == (Uint32)~0){
+ m_impl.m_operation->interpret_exit_nok();
+ } else {
+ m_impl.m_operation->branch_label(tmp.m_falseLabel);
+ }
+ break;
+ case NdbScanFilter::OR:
+ if(tmp.m_falseLabel == (Uint32)~0){
+ m_impl.m_operation->interpret_exit_nok();
+ } else {
+ m_impl.m_operation->branch_label(tmp.m_falseLabel);
+ }
+ break;
+ case NdbScanFilter::NOR:
+ if(tmp.m_falseLabel == (Uint32)~0){
+ m_impl.m_operation->interpret_exit_ok();
+ } else {
+ m_impl.m_operation->branch_label(tmp.m_trueLabel);
+ }
+ break;
+ default:
+ m_impl.m_operation->setErrorCodeAbort(4260);
+ return -1;
+ }
+
+ m_impl.m_operation->def_label(tmp.m_ownLabel);
+
+ if(m_impl.m_stack.size() == 0){
+ switch(tmp.m_group){
+ case NdbScanFilter::AND:
+ case NdbScanFilter::NOR:
+ m_impl.m_operation->interpret_exit_nok();
+ break;
+ case NdbScanFilter::OR:
+ case NdbScanFilter::NAND:
+ m_impl.m_operation->interpret_exit_ok();
+ break;
+ default:
+ m_impl.m_operation->setErrorCodeAbort(4260);
+ return -1;
+ }
+ }
+
+ return 0;
+}
+
+int
+NdbScanFilter::istrue(){
+ if(m_impl.m_current.m_group < NdbScanFilter::AND ||
+ m_impl.m_current.m_group > NdbScanFilter::NOR){
+ m_impl.m_operation->setErrorCodeAbort(4260);
+ return -1;
+ }
+
+ if(m_impl.m_current.m_trueLabel == (Uint32)~0){
+ return m_impl.m_operation->interpret_exit_ok();
+ } else {
+ return m_impl.m_operation->branch_label(m_impl.m_current.m_trueLabel);
+ }
+}
+
+int
+NdbScanFilter::isfalse(){
+ if(m_impl.m_current.m_group < NdbScanFilter::AND ||
+ m_impl.m_current.m_group > NdbScanFilter::NOR){
+ m_impl.m_operation->setErrorCodeAbort(4260);
+ return -1;
+ }
+
+ if(m_impl.m_current.m_falseLabel == (Uint32)~0){
+ return m_impl.m_operation->interpret_exit_nok();
+ } else {
+ return m_impl.m_operation->branch_label(m_impl.m_current.m_falseLabel);
+ }
+}
+
+
+#define action(x, y, z)
+
+
+typedef int (NdbOperation:: * Branch1)(Uint32, Uint32 label);
+typedef int (NdbOperation:: * Branch2)(Uint32, Uint32, Uint32 label);
+typedef int (NdbOperation:: * StrBranch2)(Uint32, const char*,Uint32,bool,Uint32);
+
+struct tab {
+ Branch2 m_branches[5];
+};
+
+static const tab table[] = {
+ /**
+ * EQ (AND, OR, NAND, NOR)
+ */
+ { { 0,
+ &NdbOperation::branch_ne,
+ &NdbOperation::branch_eq,
+ &NdbOperation::branch_eq,
+ &NdbOperation::branch_ne } }
+
+ /**
+ * NEQ
+ */
+ ,{ { 0,
+ &NdbOperation::branch_eq,
+ &NdbOperation::branch_ne,
+ &NdbOperation::branch_ne,
+ &NdbOperation::branch_eq } }
+
+ /**
+ * LT
+ */
+ ,{ { 0,
+ &NdbOperation::branch_le,
+ &NdbOperation::branch_gt,
+ &NdbOperation::branch_gt,
+ &NdbOperation::branch_le } }
+
+ /**
+ * LE
+ */
+ ,{ { 0,
+ &NdbOperation::branch_lt,
+ &NdbOperation::branch_ge,
+ &NdbOperation::branch_ge,
+ &NdbOperation::branch_lt } }
+
+ /**
+ * GT
+ */
+ ,{ { 0,
+ &NdbOperation::branch_ge,
+ &NdbOperation::branch_lt,
+ &NdbOperation::branch_lt,
+ &NdbOperation::branch_ge } }
+
+ /**
+ * GE
+ */
+ ,{ { 0,
+ &NdbOperation::branch_gt,
+ &NdbOperation::branch_le,
+ &NdbOperation::branch_le,
+ &NdbOperation::branch_gt } }
+};
+
+struct tab2 {
+ Branch1 m_branches[5];
+};
+
+static const tab2 table2[] = {
+ /**
+ * IS NULL
+ */
+ { { 0,
+ &NdbOperation::branch_col_ne_null,
+ &NdbOperation::branch_col_eq_null,
+ &NdbOperation::branch_col_eq_null,
+ &NdbOperation::branch_col_ne_null } }
+
+ /**
+ * IS NOT NULL
+ */
+ ,{ { 0,
+ &NdbOperation::branch_col_eq_null,
+ &NdbOperation::branch_col_ne_null,
+ &NdbOperation::branch_col_ne_null,
+ &NdbOperation::branch_col_eq_null } }
+};
+
+const int tab_sz = sizeof(table)/sizeof(table[0]);
+const int tab2_sz = sizeof(table2)/sizeof(table2[0]);
+
+template<typename T>
+int
+matchType(const NdbDictionary::Column * col){
+ return 1;
+}
+
+template<typename T> int load_const(NdbOperation* op, T value, Uint32 reg);
+
+template<>
+int
+load_const(NdbOperation* op, Uint32 value, Uint32 reg){
+ return op->load_const_u32(reg, value);
+}
+
+template<>
+int
+load_const(NdbOperation* op, Uint64 value, Uint32 reg){
+ return op->load_const_u64(reg, value);
+}
+
+template<typename T>
+int
+NdbScanFilterImpl::cond_col_const(Interpreter::BinaryCondition op,
+ Uint32 AttrId, T value){
+
+ if(op < 0 || op >= tab_sz){
+ m_operation->setErrorCodeAbort(4262);
+ return -1;
+ }
+
+ if(m_current.m_group < NdbScanFilter::AND ||
+ m_current.m_group > NdbScanFilter::NOR){
+ m_operation->setErrorCodeAbort(4260);
+ return -1;
+ }
+
+ Branch2 branch = table[op].m_branches[m_current.m_group];
+ const NdbDictionary::Column * col =
+ m_operation->m_currentTable->getColumn(AttrId);
+
+ if(col == 0){
+ m_operation->setErrorCodeAbort(4261);
+ return -1;
+ }
+
+ if(!matchType<T>(col)){
+ /**
+ * Code not reached
+ */
+ return -1;
+ }
+
+ if(m_latestAttrib != AttrId){
+ m_operation->read_attr(&NdbColumnImpl::getImpl(* col), 4);
+ m_latestAttrib = AttrId;
+ }
+
+ load_const<T>(m_operation, value, 5);
+ (m_operation->* branch)(4, 5, m_current.m_ownLabel);
+
+ return 0;
+};
+
+int
+NdbScanFilter::eq(int AttrId, Uint32 value){
+ return m_impl.cond_col_const(Interpreter::EQ, AttrId, value);
+}
+
+int
+NdbScanFilter::ne(int AttrId, Uint32 value){
+ return m_impl.cond_col_const(Interpreter::NE, AttrId, value);
+}
+
+int
+NdbScanFilter::lt(int AttrId, Uint32 value){
+ return m_impl.cond_col_const(Interpreter::LT, AttrId, value);
+}
+
+int
+NdbScanFilter::le(int AttrId, Uint32 value){
+ return m_impl.cond_col_const(Interpreter::LE, AttrId, value);
+}
+
+int
+NdbScanFilter::gt(int AttrId, Uint32 value){
+ return m_impl.cond_col_const(Interpreter::GT, AttrId, value);
+}
+
+int
+NdbScanFilter::ge(int AttrId, Uint32 value){
+ return m_impl.cond_col_const(Interpreter::GE, AttrId, value);
+}
+
+
+int
+NdbScanFilter::eq(int AttrId, Uint64 value){
+ return m_impl.cond_col_const(Interpreter::EQ, AttrId, value);
+}
+
+int
+NdbScanFilter::ne(int AttrId, Uint64 value){
+ return m_impl.cond_col_const(Interpreter::NE, AttrId, value);
+}
+
+int
+NdbScanFilter::lt(int AttrId, Uint64 value){
+ return m_impl.cond_col_const(Interpreter::LT, AttrId, value);
+}
+
+int
+NdbScanFilter::le(int AttrId, Uint64 value){
+ return m_impl.cond_col_const(Interpreter::LE, AttrId, value);
+}
+
+int
+NdbScanFilter::gt(int AttrId, Uint64 value){
+ return m_impl.cond_col_const(Interpreter::GT, AttrId, value);
+}
+
+int
+NdbScanFilter::ge(int AttrId, Uint64 value){
+ return m_impl.cond_col_const(Interpreter::GE, AttrId, value);
+}
+
+
+int
+NdbScanFilterImpl::cond_col(Interpreter::UnaryCondition op, Uint32 AttrId){
+
+ if(op < 0 || op >= tab2_sz){
+ m_operation->setErrorCodeAbort(4262);
+ return -1;
+ }
+
+ if(m_current.m_group < NdbScanFilter::AND ||
+ m_current.m_group > NdbScanFilter::NOR){
+ m_operation->setErrorCodeAbort(4260);
+ return -1;
+ }
+
+ Branch1 branch = table2[op].m_branches[m_current.m_group];
+ (m_operation->* branch)(AttrId, m_current.m_ownLabel);
+ return 0;
+};
+
+int
+NdbScanFilter::isnull(int AttrId){
+ return m_impl.cond_col(Interpreter::IS_NULL, AttrId);
+}
+
+int
+NdbScanFilter::isnotnull(int AttrId){
+ return m_impl.cond_col(Interpreter::IS_NOT_NULL, AttrId);
+}
+
+struct tab3 {
+ StrBranch2 m_branches[5];
+};
+
+static const tab3 table3[] = {
+ /**
+ * EQ (AND, OR, NAND, NOR)
+ */
+ { { 0,
+ &NdbOperation::branch_col_ne,
+ &NdbOperation::branch_col_eq,
+ &NdbOperation::branch_col_ne,
+ &NdbOperation::branch_col_eq } }
+
+ /**
+ * NEQ
+ */
+ ,{ { 0,
+ &NdbOperation::branch_col_eq,
+ &NdbOperation::branch_col_ne,
+ &NdbOperation::branch_col_eq,
+ &NdbOperation::branch_col_ne } }
+
+ /**
+ * LT
+ */
+ ,{ { 0,
+ &NdbOperation::branch_col_le,
+ &NdbOperation::branch_col_gt,
+ &NdbOperation::branch_col_le,
+ &NdbOperation::branch_col_gt } }
+
+ /**
+ * LE
+ */
+ ,{ { 0,
+ &NdbOperation::branch_col_lt,
+ &NdbOperation::branch_col_ge,
+ &NdbOperation::branch_col_lt,
+ &NdbOperation::branch_col_ge } }
+
+ /**
+ * GT
+ */
+ ,{ { 0,
+ &NdbOperation::branch_col_ge,
+ &NdbOperation::branch_col_lt,
+ &NdbOperation::branch_col_ge,
+ &NdbOperation::branch_col_lt } }
+
+ /**
+ * GE
+ */
+ ,{ { 0,
+ &NdbOperation::branch_col_gt,
+ &NdbOperation::branch_col_le,
+ &NdbOperation::branch_col_gt,
+ &NdbOperation::branch_col_le } }
+
+ /**
+ * LIKE
+ */
+ ,{ { 0,
+ &NdbOperation::branch_col_notlike,
+ &NdbOperation::branch_col_like,
+ &NdbOperation::branch_col_notlike,
+ &NdbOperation::branch_col_like } }
+
+ /**
+ * NOT LIKE
+ */
+ ,{ { 0,
+ &NdbOperation::branch_col_like,
+ &NdbOperation::branch_col_notlike,
+ &NdbOperation::branch_col_like,
+ &NdbOperation::branch_col_notlike } }
+};
+
+const int tab3_sz = sizeof(table3)/sizeof(table3[0]);
+
+
+int
+NdbScanFilterImpl::cond_col_const(Interpreter::BinaryCondition op,
+ Uint32 AttrId,
+ const char * value, Uint32 len, bool nopad){
+ if(op < 0 || op >= tab3_sz){
+ m_operation->setErrorCodeAbort(4260);
+ return -1;
+ }
+
+ if(m_current.m_group < NdbScanFilter::AND ||
+ m_current.m_group > NdbScanFilter::NOR){
+ m_operation->setErrorCodeAbort(4260);
+ return -1;
+ }
+
+ StrBranch2 branch = table3[op].m_branches[m_current.m_group];
+ const NdbDictionary::Column * col =
+ m_operation->m_currentTable->getColumn(AttrId);
+
+ if(col == 0){
+ m_operation->setErrorCodeAbort(4261);
+ return -1;
+ }
+
+ (m_operation->* branch)(AttrId, value, len, nopad, m_current.m_ownLabel);
+ return 0;
+}
+
+int
+NdbScanFilter::eq(int ColId, const char * val, Uint32 len, bool nopad){
+ return m_impl.cond_col_const(Interpreter::EQ, ColId, val, len, nopad);
+}
+
+int
+NdbScanFilter::ne(int ColId, const char * val, Uint32 len, bool nopad){
+ return m_impl.cond_col_const(Interpreter::NE, ColId, val, len, nopad);
+}
+
+int
+NdbScanFilter::lt(int ColId, const char * val, Uint32 len, bool nopad){
+ return m_impl.cond_col_const(Interpreter::LT, ColId, val, len, nopad);
+}
+
+int
+NdbScanFilter::le(int ColId, const char * val, Uint32 len, bool nopad){
+ return m_impl.cond_col_const(Interpreter::LE, ColId, val, len, nopad);
+}
+
+int
+NdbScanFilter::gt(int ColId, const char * val, Uint32 len, bool nopad){
+ return m_impl.cond_col_const(Interpreter::GT, ColId, val, len, nopad);
+}
+
+int
+NdbScanFilter::ge(int ColId, const char * val, Uint32 len, bool nopad){
+ return m_impl.cond_col_const(Interpreter::GE, ColId, val, len, nopad);
+}
+
+int
+NdbScanFilter::like(int ColId, const char * val, Uint32 len, bool nopad){
+ return m_impl.cond_col_const(Interpreter::LIKE, ColId, val, len, nopad);
+}
+
+int
+NdbScanFilter::notlike(int ColId, const char * val, Uint32 len, bool nopad){
+ return m_impl.cond_col_const(Interpreter::NOT_LIKE, ColId, val, len, nopad);
+}
+
+#if 0
+int
+main(void){
+ if(0)
+ {
+ ndbout << "a > 7 AND b < 9 AND c = 4" << endl;
+ NdbScanFilter f(0);
+ f.begin(NdbScanFilter::AND);
+ f.gt(0, 7);
+ f.lt(1, 9);
+ f.eq(2, 4);
+ f.end();
+ ndbout << endl;
+ }
+
+ if(0)
+ {
+ ndbout << "a > 7 OR b < 9 OR c = 4" << endl;
+ NdbScanFilter f(0);
+ f.begin(NdbScanFilter::OR);
+ f.gt(0, 7);
+ f.lt(1, 9);
+ f.eq(2, 4);
+ f.end();
+ ndbout << endl;
+ }
+
+ if(0)
+ {
+ ndbout << "a > 7 AND (b < 9 OR c = 4)" << endl;
+ NdbScanFilter f(0);
+ f.begin(NdbScanFilter::AND);
+ f.gt(0, 7);
+ f.begin(NdbScanFilter::OR);
+ f.lt(1, 9);
+ f.eq(2, 4);
+ f.end();
+ f.end();
+ ndbout << endl;
+ }
+
+ if(0)
+ {
+ ndbout << "a > 7 AND (b < 9 AND c = 4)" << endl;
+ NdbScanFilter f(0);
+ f.begin(NdbScanFilter::AND);
+ f.gt(0, 7);
+ f.begin(NdbScanFilter::AND);
+ f.lt(1, 9);
+ f.eq(2, 4);
+ f.end();
+ f.end();
+ ndbout << endl;
+ }
+
+ if(0)
+ {
+ ndbout << "(a > 7 AND b < 9) AND c = 4" << endl;
+ NdbScanFilter f(0);
+ f.begin(NdbScanFilter::AND);
+ f.begin(NdbScanFilter::AND);
+ f.gt(0, 7);
+ f.lt(1, 9);
+ f.end();
+ f.eq(2, 4);
+ f.end();
+ ndbout << endl;
+ }
+
+ if(1)
+ {
+ ndbout << "(a > 7 OR b < 9) AND (c = 4 OR c = 5)" << endl;
+ NdbScanFilter f(0);
+ f.begin(NdbScanFilter::AND);
+ f.begin(NdbScanFilter::OR);
+ f.gt(0, 7);
+ f.lt(1, 9);
+ f.end();
+ f.begin(NdbScanFilter::OR);
+ f.eq(2, 4);
+ f.eq(2, 5);
+ f.end();
+ f.end();
+ ndbout << endl;
+ }
+
+ if(1)
+ {
+ ndbout << "(a > 7 AND b < 9) OR (c = 4 AND c = 5)" << endl;
+ NdbScanFilter f(0);
+ f.begin(NdbScanFilter::OR);
+ f.begin(NdbScanFilter::AND);
+ f.gt(0, 7);
+ f.lt(1, 9);
+ f.end();
+ f.begin(NdbScanFilter::AND);
+ f.eq(2, 4);
+ f.eq(2, 5);
+ f.end();
+ f.end();
+ ndbout << endl;
+ }
+
+ if(1)
+ {
+ ndbout <<
+ "((a > 7 AND b < 9) OR (c = 4 AND d = 5)) AND "
+ "((e > 6 AND f < 8) OR (g = 2 AND h = 3)) " << endl;
+ NdbScanFilter f(0);
+ f.begin(NdbScanFilter::AND);
+ f.begin(NdbScanFilter::OR);
+ f.begin(NdbScanFilter::AND);
+ f.gt(0, 7);
+ f.lt(1, 9);
+ f.end();
+ f.begin(NdbScanFilter::AND);
+ f.eq(2, 4);
+ f.eq(3, 5);
+ f.end();
+ f.end();
+
+ f.begin(NdbScanFilter::OR);
+ f.begin(NdbScanFilter::AND);
+ f.gt(4, 6);
+ f.lt(5, 8);
+ f.end();
+ f.begin(NdbScanFilter::AND);
+ f.eq(6, 2);
+ f.eq(7, 3);
+ f.end();
+ f.end();
+ f.end();
+ }
+
+ return 0;
+}
+#endif
diff --git a/ndb/src/ndbapi/NdbScanOperation.cpp b/ndb/src/ndbapi/NdbScanOperation.cpp
new file mode 100644
index 00000000000..f753d2f6b34
--- /dev/null
+++ b/ndb/src/ndbapi/NdbScanOperation.cpp
@@ -0,0 +1,647 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*****************************************************************************
+ * Name: NdbScanOperation.cpp
+ * Include:
+ * Link:
+ * Author: UABMASD Martin Sköld INN/V Alzato
+ * Date: 2002-04-01
+ * Version: 0.1
+ * Description: Table scan support
+ * Documentation:
+ * Adjust: 2002-04-01 UABMASD First version.
+ ****************************************************************************/
+
+#include <Ndb.hpp>
+#include <NdbScanOperation.hpp>
+#include <NdbConnection.hpp>
+#include <NdbResultSet.hpp>
+#include "NdbApiSignal.hpp"
+#include <NdbOut.hpp>
+#include "NdbDictionaryImpl.hpp"
+#include <NdbString.h>
+#ifndef NDB_MACOSX
+#include <malloc.h>
+#else
+#include <stdlib.h>
+#endif
+
+NdbScanOperation::NdbScanOperation(Ndb* aNdb) :
+ NdbCursorOperation(aNdb),
+ m_transConnection(NULL),
+ m_autoExecute(false),
+ m_updateOp(false),
+ m_deleteOp(false),
+ m_setValueList(new SetValueRecList())
+{
+}
+
+NdbScanOperation::~NdbScanOperation()
+{
+ if (m_setValueList) delete m_setValueList;
+}
+
+NdbCursorOperation::CursorType
+NdbScanOperation::cursorType()
+{
+ return NdbCursorOperation::ScanCursor;
+}
+
+void
+NdbScanOperation::setErrorCode(int aErrorCode){
+ NdbConnection* tmp = theNdbCon;
+ theNdbCon = m_transConnection;
+ NdbOperation::setErrorCode(aErrorCode);
+ theNdbCon = tmp;
+}
+
+void
+NdbScanOperation::setErrorCodeAbort(int aErrorCode){
+ NdbConnection* tmp = theNdbCon;
+ theNdbCon = m_transConnection;
+ NdbOperation::setErrorCodeAbort(aErrorCode);
+ theNdbCon = tmp;
+}
+
+
+/*****************************************************************************
+ * int init();
+ *
+ * Return Value: Return 0 : init was successful.
+ * Return -1: In all other case.
+ * Remark: Initiates operation record after allocation.
+ *****************************************************************************/
+int
+NdbScanOperation::init(NdbTableImpl* tab, NdbConnection* myConnection)
+{
+ m_transConnection = myConnection;
+ //NdbConnection* aScanConnection = theNdb->startTransaction(myConnection);
+ NdbConnection* aScanConnection = theNdb->hupp(myConnection);
+ if (!aScanConnection)
+ return -1;
+ aScanConnection->theFirstOpInList = this;
+ aScanConnection->theLastOpInList = this;
+ NdbCursorOperation::cursInit();
+ // NOTE! The hupped trans becomes the owner of the operation
+ return NdbOperation::init(tab, aScanConnection);
+}
+
+NdbResultSet* NdbScanOperation::readTuples(Uint32 parallell,
+ NdbCursorOperation::LockMode lm)
+{
+ int res = 0;
+ switch(lm){
+ case NdbCursorOperation::LM_Read:
+ parallell = (parallell == 0 ? 240 : parallell);
+ res = openScan(parallell, false, true, false);
+ break;
+ case NdbCursorOperation::LM_Exclusive:
+ parallell = (parallell == 0 ? 1 : parallell);
+ res = openScan(parallell, true, /*irrelevant*/true, /*irrelevant*/false);
+ break;
+ case NdbCursorOperation::LM_Dirty:
+ parallell = (parallell == 0 ? 240 : parallell);
+ res = openScan(parallell, true, /*irrelevant*/true, /*irrelevant*/false);
+ break;
+ default:
+ res = -1;
+ setErrorCode(4003);
+ }
+ if(res == -1){
+ return NULL;
+ }
+ theNdbCon->theFirstOpInList = 0;
+ theNdbCon->theLastOpInList = 0;
+
+ return getResultSet();
+}
+
+int NdbScanOperation::updateTuples(Uint32 parallelism)
+{
+ if (openScanExclusive(parallelism) == -1) {
+ return -1;
+ }
+ theNdbCon->theFirstOpInList = 0;
+ theNdbCon->theLastOpInList = 0;
+
+ m_updateOp = true;
+
+ return 0;
+}
+
+int NdbScanOperation::deleteTuples(Uint32 parallelism)
+{
+ if (openScanExclusive(parallelism) == -1) {
+ return -1;
+ }
+ theNdbCon->theFirstOpInList = 0;
+ theNdbCon->theLastOpInList = 0;
+
+ m_deleteOp = true;
+
+ return 0;
+}
+
+int NdbScanOperation::setValue(const char* anAttrName, const char* aValue, Uint32 len)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrName) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrName, aValue, len);
+ return 0;
+}
+
+int NdbScanOperation::setValue(const char* anAttrName, Int32 aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrName) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrName, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(const char* anAttrName, Uint32 aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrName) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrName, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(const char* anAttrName, Uint64 aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrName) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrName, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(const char* anAttrName, Int64 aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrName) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrName, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(const char* anAttrName, float aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrName) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrName, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(const char* anAttrName, double aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrName) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrName, aValue);
+ return 0;
+}
+
+
+int NdbScanOperation::setValue(Uint32 anAttrId, const char* aValue, Uint32 len)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrId) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrId, aValue, len);
+ return 0;
+}
+
+int NdbScanOperation::setValue(Uint32 anAttrId, Int32 aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrId) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrId, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(Uint32 anAttrId, Uint32 aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrId) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrId, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(Uint32 anAttrId, Uint64 aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrId) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrId, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(Uint32 anAttrId, Int64 aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrId) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrId, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(Uint32 anAttrId, float aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrId) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrId, aValue);
+ return 0;
+}
+
+int NdbScanOperation::setValue(Uint32 anAttrId, double aValue)
+{
+ // Check if attribute exist
+ if (m_currentTable->getColumn(anAttrId) == NULL)
+ return -1;
+
+ m_setValueList->add(anAttrId, aValue);
+ return 0;
+}
+
+// Private methods
+
+int NdbScanOperation::executeCursor(int ProcessorId)
+{
+ int result = theNdbCon->executeScan();
+ // If the scan started ok and we are updating or deleting
+ // iterate over all tuples
+ if ((m_updateOp) || (m_deleteOp)) {
+ NdbOperation* newOp;
+
+ while ((result != -1) && (nextResult() == 0)) {
+ if (m_updateOp) {
+ newOp = takeOverScanOp(UpdateRequest, m_transConnection);
+ // Pass setValues from scan operation to new operation
+ m_setValueList->iterate(SetValueRecList::callSetValueFn, *newOp);
+ // No need to call updateTuple since scan was taken over for update
+ // it should be the same with delete - MASV
+ // newOp->updateTuple();
+ }
+ else if (m_deleteOp) {
+ newOp = takeOverScanOp(DeleteRequest, m_transConnection);
+ // newOp->deleteTuple();
+ }
+#if 0
+ // takeOverScanOp will take over the lock that scan aquired
+ // the lock is released when nextScanResult is called
+ // That means that the "takeover" has to be sent to the kernel
+ // before nextScanresult is called - MASV
+ if (m_autoExecute){
+ m_transConnection->execute(NoCommit);
+ }
+#else
+ m_transConnection->execute(NoCommit);
+#endif
+ }
+ closeScan();
+ }
+
+ return result;
+}
+
+int NdbScanOperation::nextResult(bool fetchAllowed)
+{
+ int result = theNdbCon->nextScanResult(fetchAllowed);
+ if (result == -1){
+ // Move the error code from hupped transaction
+ // to the real trans
+ const NdbError err = theNdbCon->getNdbError();
+ m_transConnection->setOperationErrorCode(err.code);
+ }
+ return result;
+}
+
+int
+NdbScanOperation::prepareSend(Uint32 TC_ConnectPtr, Uint64 TransactionId)
+{
+ printf("NdbScanOperation::prepareSend\n");
+ return 0;
+}
+
+int
+NdbScanOperation::doSend(int ProcessorId)
+{
+ printf("NdbScanOperation::doSend\n");
+ return 0;
+}
+
+void NdbScanOperation::closeScan()
+{
+ if(theNdbCon){
+ if (theNdbCon->stopScan() == -1)
+ theError = theNdbCon->getNdbError();
+ theNdb->closeTransaction(theNdbCon);
+ theNdbCon = 0;
+ }
+ m_transConnection = NULL;
+}
+
+void NdbScanOperation::release(){
+ closeScan();
+ NdbCursorOperation::release();
+}
+
+void SetValueRecList::add(const char* anAttrName, const char* aValue, Uint32 len)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_STRING_ATTR1;
+ newSetValueRec->anAttrName = strdup(anAttrName);
+ newSetValueRec->stringStruct.aStringValue = (char *) malloc(len);
+ strlcpy(newSetValueRec->stringStruct.aStringValue, aValue, len);
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(const char* anAttrName, Int32 aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_INT32_ATTR1;
+ newSetValueRec->anAttrName = strdup(anAttrName);
+ newSetValueRec->anInt32Value = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(const char* anAttrName, Uint32 aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_UINT32_ATTR1;
+ newSetValueRec->anAttrName = strdup(anAttrName);
+ newSetValueRec->anUint32Value = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(const char* anAttrName, Int64 aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_INT64_ATTR1;
+ newSetValueRec->anAttrName = strdup(anAttrName);
+ newSetValueRec->anInt64Value = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(const char* anAttrName, Uint64 aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_UINT64_ATTR1;
+ newSetValueRec->anAttrName = strdup(anAttrName);
+ newSetValueRec->anUint64Value = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(const char* anAttrName, float aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_FLOAT_ATTR1;
+ newSetValueRec->anAttrName = strdup(anAttrName);
+ newSetValueRec->aFloatValue = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(const char* anAttrName, double aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_DOUBLE_ATTR1;
+ newSetValueRec->anAttrName = strdup(anAttrName);
+ newSetValueRec->aDoubleValue = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(Uint32 anAttrId, const char* aValue, Uint32 len)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_STRING_ATTR2;
+ newSetValueRec->anAttrId = anAttrId;
+ newSetValueRec->stringStruct.aStringValue = (char *) malloc(len);
+ strlcpy(newSetValueRec->stringStruct.aStringValue, aValue, len);
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(Uint32 anAttrId, Int32 aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_INT32_ATTR2;
+ newSetValueRec->anAttrId = anAttrId;
+ newSetValueRec->anInt32Value = aValue;
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+}
+
+void SetValueRecList::add(Uint32 anAttrId, Uint32 aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_UINT32_ATTR2;
+ newSetValueRec->anAttrId = anAttrId;
+ newSetValueRec->anUint32Value = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(Uint32 anAttrId, Int64 aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_INT64_ATTR2;
+ newSetValueRec->anAttrId = anAttrId;
+ newSetValueRec->anInt64Value = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(Uint32 anAttrId, Uint64 aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_UINT64_ATTR2;
+ newSetValueRec->anAttrId = anAttrId;
+ newSetValueRec->anUint64Value = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(Uint32 anAttrId, float aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_FLOAT_ATTR2;
+ newSetValueRec->anAttrId = anAttrId;
+ newSetValueRec->aFloatValue = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void SetValueRecList::add(Uint32 anAttrId, double aValue)
+{
+ SetValueRec* newSetValueRec = new SetValueRec();
+
+ newSetValueRec->stype = SetValueRec::SET_DOUBLE_ATTR2;
+ newSetValueRec->anAttrId = anAttrId;
+ newSetValueRec->aDoubleValue = aValue;
+ if (!last)
+ first = last = newSetValueRec;
+ else {
+ last->next = newSetValueRec;
+ last = newSetValueRec;
+ }
+}
+
+void
+SetValueRecList::callSetValueFn(SetValueRec& aSetValueRec, NdbOperation& oper)
+{
+ switch(aSetValueRec.stype) {
+ case(SetValueRec::SET_STRING_ATTR1):
+ oper.setValue(aSetValueRec.anAttrName, aSetValueRec.stringStruct.aStringValue, aSetValueRec.stringStruct.len);
+ break;
+ case(SetValueRec::SET_INT32_ATTR1):
+ oper.setValue(aSetValueRec.anAttrName, aSetValueRec.anInt32Value);
+ break;
+ case(SetValueRec::SET_UINT32_ATTR1):
+ oper.setValue(aSetValueRec.anAttrName, aSetValueRec.anUint32Value);
+ break;
+ case(SetValueRec::SET_INT64_ATTR1):
+ oper.setValue(aSetValueRec.anAttrName, aSetValueRec.anInt64Value);
+ break;
+ case(SetValueRec::SET_UINT64_ATTR1):
+ oper.setValue(aSetValueRec.anAttrName, aSetValueRec.anUint64Value);
+ break;
+ case(SetValueRec::SET_FLOAT_ATTR1):
+ oper.setValue(aSetValueRec.anAttrName, aSetValueRec.aFloatValue);
+ break;
+ case(SetValueRec::SET_DOUBLE_ATTR1):
+ oper.setValue(aSetValueRec.anAttrName, aSetValueRec.aDoubleValue);
+ break;
+ case(SetValueRec::SET_STRING_ATTR2):
+ oper.setValue(aSetValueRec.anAttrId, aSetValueRec.stringStruct.aStringValue, aSetValueRec.stringStruct.len);
+ break;
+ case(SetValueRec::SET_INT32_ATTR2):
+ oper.setValue(aSetValueRec.anAttrId, aSetValueRec.anInt32Value);
+ break;
+ case(SetValueRec::SET_UINT32_ATTR2):
+ oper.setValue(aSetValueRec.anAttrId, aSetValueRec.anUint32Value);
+ break;
+ case(SetValueRec::SET_INT64_ATTR2):
+ oper.setValue(aSetValueRec.anAttrId, aSetValueRec.anInt64Value);
+ break;
+ case(SetValueRec::SET_UINT64_ATTR2):
+ oper.setValue(aSetValueRec.anAttrId, aSetValueRec.anUint64Value);
+ break;
+ case(SetValueRec::SET_FLOAT_ATTR2):
+ oper.setValue(aSetValueRec.anAttrId, aSetValueRec.aFloatValue);
+ break;
+ case(SetValueRec::SET_DOUBLE_ATTR2):
+ oper.setValue(aSetValueRec.anAttrId, aSetValueRec.aDoubleValue);
+ break;
+ }
+}
+
+int
+NdbScanOperation::equal_impl(const NdbColumnImpl* anAttrObject,
+ const char* aValue,
+ Uint32 len){
+ return setBound(anAttrObject, BoundEQ, aValue, len);
+}
+
+
diff --git a/ndb/src/ndbapi/NdbScanReceiver.cpp b/ndb/src/ndbapi/NdbScanReceiver.cpp
new file mode 100644
index 00000000000..6c8c16c3ecf
--- /dev/null
+++ b/ndb/src/ndbapi/NdbScanReceiver.cpp
@@ -0,0 +1,187 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbScanReceiver.hpp"
+#include <NdbRecAttr.hpp>
+
+#include <signaldata/ScanFrag.hpp>
+
+#include <NdbOut.hpp>
+
+
+/***************************************************************************
+ * int receiveKEYINFO20( NdbApiSignal* aSignal)
+ *
+ * Remark: Handles the reception of the KEYINFO20 signal.
+ * Save a copy of the signal in list
+ *
+ ***************************************************************************/
+int
+NdbScanReceiver::receiveKEYINFO20( NdbApiSignal* aSignal){
+ const KeyInfo20 * const keyInfo = CAST_CONSTPTR(KeyInfo20, aSignal->getDataPtr());
+ if (theStatus != Waiting){
+ //ndbout << "Dropping KEYINFO20, theStatus="<<theStatus << endl;
+ return -1;
+ }
+ if (aSignal->getLength() < 5){
+ //ndbout << "Dropping KEYINFO20, length="<<aSignal->getLength() << endl;
+ }
+ Uint64 tCurrTransId = theNdbOp->theNdbCon->getTransactionId();
+ Uint64 tRecTransId = (Uint64)keyInfo->transId1 + ((Uint64)keyInfo->transId2 << 32);
+ if ((tRecTransId - tCurrTransId) != (Uint64)0){
+ //ndbout << "Dropping KEYINFO20 wrong transid" << endl;
+ return -1;
+ }
+
+ NdbApiSignal * tCopy = new NdbApiSignal(0);//getSignal();
+ if (tCopy == NULL) {
+ theNdbOp->setErrorCode(4000);
+ return 2; // theWaitState = NO_WAIT
+ }
+ // Put copy last in list of KEYINFO20 signals
+ tCopy->copyFrom(aSignal);
+ tCopy->next(NULL);
+ if (theFirstKEYINFO20_Recv == NULL)
+ theFirstKEYINFO20_Recv = tCopy;
+ else
+ theLastKEYINFO20_Recv->next(tCopy);
+ theLastKEYINFO20_Recv = tCopy;
+
+ theTotalKI_Len = keyInfo->keyLen; // This is the total length of all signals
+ theTotalRecKI_Len += aSignal->getLength() - 5;
+ return theNdbOp->theNdbCon->checkNextScanResultComplete();
+}
+
+/***************************************************************************
+ * int receiveTRANSID_AI_SCAN( NdbApiSignal* aSignal)
+ *
+ * Remark: Handles the reception of the TRANSID_AI_signal with
+ * 22 signal data words.
+ * Save a copy of the signal in list and check if all
+ * signals belonging to this resultset is receieved.
+ *
+ ***************************************************************************/
+int
+NdbScanReceiver::receiveTRANSID_AI_SCAN( NdbApiSignal* aSignal)
+{
+ const Uint32* aDataPtr = aSignal->getDataPtr();
+ if (theStatus != Waiting){
+ //ndbout << "Dropping TRANSID_AI, theStatus="<<theStatus << endl;
+ return -1;
+ }
+ if (aSignal->getLength() < 3){
+ //ndbout << "Dropping TRANSID_AI, length="<<aSignal->getLength() << endl;
+ return -1;
+ }
+ if (theNdbOp == NULL){
+ //ndbout << "Dropping TRANSID_AI, theNdbOp == NULL" << endl;
+ return -1;
+ }
+ if (theNdbOp->theNdbCon == NULL){
+ //ndbout << "Dropping TRANSID_AI, theNdbOp->theNdbCon == NULL" << endl;
+ return -1;
+ }
+ Uint64 tCurrTransId = theNdbOp->theNdbCon->getTransactionId();
+ Uint64 tRecTransId = (Uint64)aDataPtr[1] + ((Uint64)aDataPtr[2] << 32);
+ if ((tRecTransId - tCurrTransId) != (Uint64)0){
+ //ndbout << "Dropping TRANSID_AI wrong transid" << endl;
+ return -1;
+ }
+
+ NdbApiSignal * tCopy = new NdbApiSignal(0);//getSignal();
+ if (tCopy == NULL){
+ theNdbOp->setErrorCode(4000);
+ return 2; // theWaitState = NO_WAIT
+ }
+ tCopy->copyFrom(aSignal);
+ tCopy->next(NULL);
+ if (theFirstTRANSID_AI_Recv == NULL)
+ theFirstTRANSID_AI_Recv = tCopy;
+ else
+ theLastTRANSID_AI_Recv->next(tCopy);
+ theLastTRANSID_AI_Recv = tCopy;
+ theTotalRecAI_Len += aSignal->getLength() - 3;
+
+ return theNdbOp->theNdbCon->checkNextScanResultComplete();
+}
+
+/***************************************************************************
+ * int executeSavedSignals()
+ *
+ * Remark: Execute all saved TRANSID_AI signals into the parent NdbOperation
+ *
+ *
+ ***************************************************************************/
+int
+NdbScanReceiver::executeSavedSignals(){
+
+ NdbApiSignal* tSignal = theFirstTRANSID_AI_Recv;
+ while (tSignal != NULL) {
+ const Uint32* tDataPtr = tSignal->getDataPtr();
+
+ int tRet = theNdbOp->receiveREAD_AI((Uint32*)&tDataPtr[3],
+ tSignal->getLength() - 3);
+ if (tRet != -1){
+ // -1 means that more signals are wanted ?
+ // Make sure there are no more signals in the list
+ assert(tSignal->next() == NULL);
+ }
+ tSignal = tSignal->next();
+ }
+ // receiveREAD_AI may not copy to application buffers
+ NdbRecAttr* tRecAttr = theNdbOp->theFirstRecAttr;
+ while (tRecAttr != NULL) {
+ if (tRecAttr->copyoutRequired()) // copy to application buffer
+ tRecAttr->copyout();
+ tRecAttr = tRecAttr->next();
+ }
+ // Release TRANSID_AI signals for this receiver
+ while(theFirstTRANSID_AI_Recv != NULL){
+ NdbApiSignal* tmp = theFirstTRANSID_AI_Recv;
+ theFirstTRANSID_AI_Recv = tmp->next();
+ delete tmp;
+ }
+
+ // theNdbOp->theNdb->releaseSignalsInList(&theFirstTRANSID_AI_Recv);
+ theFirstTRANSID_AI_Recv = NULL;
+ theLastTRANSID_AI_Recv = NULL;
+ theStatus = Executed;
+
+ return 0;
+}
+
+
+void
+NdbScanReceiver::prepareNextScanResult(){
+ if(theStatus == Executed){
+
+ // theNdbOp->theNdb->releaseSignalsInList(&theFirstKEYINFO20_Recv);
+ while(theFirstKEYINFO20_Recv != NULL){
+ NdbApiSignal* tmp = theFirstKEYINFO20_Recv;
+ theFirstKEYINFO20_Recv = tmp->next();
+ delete tmp;
+ }
+ theFirstKEYINFO20_Recv = NULL;
+ theLastKEYINFO20_Recv = NULL;
+ theTotalRecAI_Len = 0;
+ theTotalRecKI_Len = 0;
+ if (theLockMode == true)
+ theTotalKI_Len = 0xFFFFFFFF;
+ else
+ theTotalKI_Len = 0;
+ theStatus = Waiting;
+ }
+}
diff --git a/ndb/src/ndbapi/NdbScanReceiver.hpp b/ndb/src/ndbapi/NdbScanReceiver.hpp
new file mode 100644
index 00000000000..5e316719194
--- /dev/null
+++ b/ndb/src/ndbapi/NdbScanReceiver.hpp
@@ -0,0 +1,211 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NdbScanReceiver_H
+#define NdbScanReceiver_H
+
+#include "Ndb.hpp"
+#include "NdbConnection.hpp"
+#include "NdbOperation.hpp"
+#include "NdbApiSignal.hpp"
+#include "NdbReceiver.hpp"
+#include <NdbOut.hpp>
+
+#include <assert.h>
+
+class NdbScanReceiver
+{
+ enum ReceiverStatus { Init,
+ Waiting,
+ Completed,
+ Executed,
+ Released };
+
+ friend class Ndb;
+ friend class NdbOperation;
+public:
+ NdbScanReceiver(Ndb *aNdb) :
+ theReceiver(aNdb),
+ theNdbOp(NULL),
+ theFirstTRANSID_AI_Recv(NULL),
+ theLastTRANSID_AI_Recv(NULL),
+ theFirstKEYINFO20_Recv(NULL),
+ theLastKEYINFO20_Recv(NULL),
+ theTotalRecAI_Len(0),
+ theTotalKI_Len(0xFFFFFFFF),
+ theTotalRecKI_Len(0),
+ theStatus(Init),
+ theNextScanRec(NULL)
+ {
+ theReceiver.init(NdbReceiver::NDB_SCANRECEIVER, this);
+ }
+
+ int checkMagicNumber();
+ int receiveTRANSID_AI_SCAN(NdbApiSignal*);
+ int receiveKEYINFO20(NdbApiSignal*);
+ int executeSavedSignals();
+ void prepareNextScanResult();
+
+ NdbScanReceiver* next();
+ void next(NdbScanReceiver*);
+
+ bool isCompleted(Uint32 aiLenToReceive);
+ void setCompleted();
+
+ void init(NdbOperation* aNdbOp, bool lockMode);
+
+ Uint32 ptr2int() { return theReceiver.getId(); };
+private:
+ NdbScanReceiver();
+ void release();
+
+ NdbReceiver theReceiver;
+
+ NdbOperation* theNdbOp;
+ NdbApiSignal* theFirstTRANSID_AI_Recv;
+ NdbApiSignal* theLastTRANSID_AI_Recv;
+ NdbApiSignal* theFirstKEYINFO20_Recv;
+ NdbApiSignal* theLastKEYINFO20_Recv;
+
+ Uint32 theTotalRecAI_Len;
+ Uint32 theTotalKI_Len;
+ Uint32 theTotalRecKI_Len;
+ ReceiverStatus theStatus;
+ Uint32 theMagicNumber;
+ NdbScanReceiver* theNextScanRec;
+ bool theLockMode;
+
+};
+
+inline
+void
+NdbScanReceiver::init(NdbOperation* aNdbOp, bool lockMode){
+ assert(theStatus == Init || theStatus == Released);
+ theNdbOp = aNdbOp;
+ theMagicNumber = 0xA0B1C2D3;
+ theTotalRecAI_Len = 0;
+
+ /* If we are locking the records for take over
+ * KI_len to receive is at least 1, since we don't know yet
+ * how much KI we are expecting(this is written in the first KI signal)
+ * set theTotalKI_Len to FFFFFFFF, this will make the ScanReciever wait for
+ * at least the first KI, and when that is received we will know if
+ * we are expecting another one
+ */
+ theLockMode = lockMode;
+ if (theLockMode == true)
+ theTotalKI_Len = 0xFFFFFFFF;
+ else
+ theTotalKI_Len = 0;
+ theTotalRecKI_Len = 0;
+
+ assert(theNextScanRec == NULL);
+ theNextScanRec = NULL;
+ assert(theFirstTRANSID_AI_Recv == NULL);
+ theFirstTRANSID_AI_Recv = NULL;
+ assert(theLastTRANSID_AI_Recv == NULL);
+ theLastTRANSID_AI_Recv = NULL;
+ assert(theFirstKEYINFO20_Recv == NULL);
+ theFirstKEYINFO20_Recv = NULL;
+ theLastKEYINFO20_Recv = NULL;
+
+ theStatus = Waiting;
+};
+
+
+inline
+void
+NdbScanReceiver::release(){
+ theStatus = Released;
+ // theNdbOp->theNdb->releaseSignalsInList(&theFirstTRANSID_AI_Recv);
+ while(theFirstTRANSID_AI_Recv != NULL){
+ NdbApiSignal* tmp = theFirstTRANSID_AI_Recv;
+ theFirstTRANSID_AI_Recv = tmp->next();
+ delete tmp;
+ }
+ theFirstTRANSID_AI_Recv = NULL;
+ theLastTRANSID_AI_Recv = NULL;
+ // theNdbOp->theNdb->releaseSignalsInList(&theFirstKEYINFO20_Recv);
+ while(theFirstKEYINFO20_Recv != NULL){
+ NdbApiSignal* tmp = theFirstKEYINFO20_Recv;
+ theFirstKEYINFO20_Recv = tmp->next();
+ delete tmp;
+ }
+ theFirstKEYINFO20_Recv = NULL;
+ theLastKEYINFO20_Recv = NULL;
+ theNdbOp = NULL;
+ theTotalRecAI_Len = 0;
+ theTotalRecKI_Len = 0;
+ theTotalKI_Len = 0xFFFFFFFF;
+};
+
+inline
+int
+NdbScanReceiver::checkMagicNumber()
+{
+ if (theMagicNumber != 0xA0B1C2D3)
+ return -1;
+ return 0;
+}
+
+inline
+NdbScanReceiver*
+NdbScanReceiver::next(){
+ return theNextScanRec;
+}
+
+inline
+void
+NdbScanReceiver::next(NdbScanReceiver* aScanRec){
+ theNextScanRec = aScanRec;
+}
+
+inline
+bool
+NdbScanReceiver::isCompleted(Uint32 aiLenToReceive){
+ assert(theStatus == Waiting || theStatus == Completed);
+#if 0
+ ndbout << "NdbScanReceiver::isCompleted"<<endl
+ << " theStatus = " << theStatus << endl
+ << " theTotalRecAI_Len = " << theTotalRecAI_Len << endl
+ << " aiLenToReceive = " << aiLenToReceive << endl
+ << " theTotalRecKI_Len = "<< theTotalRecKI_Len << endl
+ << " theTotalKI_Len = "<< theTotalKI_Len << endl;
+#endif
+ // Have we already receive everything
+ if(theStatus == Completed)
+ return true;
+
+ // Check that we have received AI
+ if(theTotalRecAI_Len < aiLenToReceive)
+ return false;
+
+ // Check that we have recieved KI
+ if (theTotalRecKI_Len < theTotalKI_Len)
+ return false;
+
+ // We should not have recieved more AI
+ assert(theTotalRecAI_Len <= aiLenToReceive);
+ return true;
+}
+
+inline
+void
+NdbScanReceiver::setCompleted(){
+ theStatus = Completed;
+}
+
+#endif
diff --git a/ndb/src/ndbapi/NdbSchemaCon.cpp b/ndb/src/ndbapi/NdbSchemaCon.cpp
new file mode 100644
index 00000000000..fbf30c70d12
--- /dev/null
+++ b/ndb/src/ndbapi/NdbSchemaCon.cpp
@@ -0,0 +1,163 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*********************************************************************
+Name: NdbSchemaCon.C
+Include:
+Link:
+Author: UABMNST Mona Natterkvist UAB/B/SD
+ EMIKRON Mikael Ronstrom
+Date: 020826
+Version: 2.0
+Description: Interface between application and NDB
+Documentation:
+Adjust: 980126 UABMNST First version.
+ 020826 EMIKRON New version adapted to new DICT version
+************************************************************************************************/
+#include "NdbSchemaCon.hpp"
+#include "NdbSchemaOp.hpp"
+#include "NdbApiSignal.hpp"
+#include "TransporterFacade.hpp"
+#include <RefConvert.hpp>
+#include <signaldata/CreateIndx.hpp>
+#include <signaldata/DropIndx.hpp>
+#include <signaldata/CreateTable.hpp>
+#include <NdbOut.hpp>
+
+/*********************************************************************
+NdbSchemaCon(Ndb* aNdb);
+
+Parameters: aNdb: Pointers to the Ndb object
+Remark: Creates a schemacon object.
+************************************************************************************************/
+NdbSchemaCon::NdbSchemaCon( Ndb* aNdb ) :
+ theNdb(aNdb),
+ theFirstSchemaOpInList(NULL),
+ theMagicNumber(0x75318642)
+{
+ theError.code = 0;
+}//NdbSchemaCon::NdbSchemaCon()
+
+/*********************************************************************
+~NdbSchemaCon();
+
+Remark: Deletes the connection object.
+************************************************************************************************/
+NdbSchemaCon::~NdbSchemaCon()
+{
+}//NdbSchemaCon::~NdbSchemaCon()
+
+/*********************************************************************
+NdbSchemaOp* getNdbSchemaOp();
+
+Return Value Return a pointer to a NdbSchemaOp object if getNdbSchemaOp was sussesful.
+ Return NULL: In all other case.
+Parameters: tableId : Id of the database table beeing deleted.
+************************************************************************************************/
+NdbSchemaOp*
+NdbSchemaCon::getNdbSchemaOp()
+{
+ NdbSchemaOp* tSchemaOp;
+ if (theFirstSchemaOpInList != NULL) {
+ theError.code = 4401; // Only support one add table per transaction
+ return NULL;
+ }//if
+ tSchemaOp = new NdbSchemaOp(theNdb);
+ if ( tSchemaOp == NULL ) {
+ theError.code = 4000; // Could not allocate schema operation
+ return NULL;
+ }//if
+ theFirstSchemaOpInList = tSchemaOp;
+ int retValue = tSchemaOp->init(this);
+ if (retValue == -1) {
+ release();
+ theError.code = 4000; // Could not allocate buffer in schema operation
+ return NULL;
+ }//if
+ return tSchemaOp;
+}//NdbSchemaCon::getNdbSchemaOp()
+
+/*********************************************************************
+int execute();
+
+Return Value: Return 0 : execute was successful.
+ Return -1: In all other case.
+Parameters : aTypeOfExec: Type of execute.
+Remark: Initialise connection object for new transaction.
+************************************************************************************************/
+int
+NdbSchemaCon::execute()
+{
+ if(theError.code != 0) {
+ return -1;
+ }//if
+
+ NdbSchemaOp* tSchemaOp;
+
+ tSchemaOp = theFirstSchemaOpInList;
+ if (tSchemaOp == NULL) {
+ theError.code = 4402;
+ return -1;
+ }//if
+
+ if ((tSchemaOp->sendRec() == -1) || (theError.code != 0)) {
+ // Error Code already set in other place
+ return -1;
+ }//if
+
+ return 0;
+}//NdbSchemaCon::execute()
+
+/*********************************************************************
+void release();
+
+Remark: Release all schemaop.
+************************************************************************************************/
+void
+NdbSchemaCon::release()
+{
+ NdbSchemaOp* tSchemaOp;
+ tSchemaOp = theFirstSchemaOpInList;
+ if (tSchemaOp != NULL) {
+ tSchemaOp->release();
+ delete tSchemaOp;
+ }//if
+ theFirstSchemaOpInList = NULL;
+ return;
+}//NdbSchemaCon::release()
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/ndb/src/ndbapi/NdbSchemaOp.cpp b/ndb/src/ndbapi/NdbSchemaOp.cpp
new file mode 100644
index 00000000000..9f4d7fbcfd4
--- /dev/null
+++ b/ndb/src/ndbapi/NdbSchemaOp.cpp
@@ -0,0 +1,202 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/*****************************************************************************
+Name: NdbSchemaOp.C
+Include:
+Link:
+Author: UABMNST Mona Natterkvist UAB/B/SD
+ EMIKRON Mikael Ronstrom
+Date: 020826
+Version: 2.0
+Description: Interface between application and NDB
+Documentation: Handles createTable and createAttribute calls
+
+Adjust: 980125 UABMNST First version.
+ 020826 EMIKRON New version for new DICT
+*****************************************************************************/
+#include "NdbSchemaOp.hpp"
+#include "NdbSchemaCon.hpp"
+#include "API.hpp"
+
+/*****************************************************************************
+NdbSchemaOp(Ndb* aNdb, Table* aTable);
+
+Return Value: None
+Parameters: aNdb: Pointers to the Ndb object.
+ aTable: Pointers to the Table object
+Remark: Creat an object of NdbSchemaOp.
+*****************************************************************************/
+NdbSchemaOp::NdbSchemaOp(Ndb* aNdb) :
+ theNdb(aNdb),
+ theSchemaCon(NULL),
+ m_currentTable(NULL)
+{
+}//NdbSchemaOp::NdbSchemaOp()
+
+/*****************************************************************************
+~NdbSchemaOp();
+
+Remark: Delete tables for connection pointers (id).
+*****************************************************************************/
+NdbSchemaOp::~NdbSchemaOp( )
+{
+}//~NdbSchemaOp::NdbSchemaOp()
+
+/*****************************************************************************
+int createTable( const char* tableName )
+*****************************************************************************/
+int
+NdbSchemaOp::createTable(const char* aTableName,
+ Uint32 aTableSize,
+ KeyType aTupleKey,
+ int aNrOfPages,
+ FragmentType aFragmentType,
+ int aKValue,
+ int aMinLoadFactor,
+ int aMaxLoadFactor,
+ int aMemoryType,
+ bool aStoredTable)
+{
+ if(m_currentTable != 0){
+ return -1;
+ }
+
+ m_currentTable = new NdbDictionary::Table(aTableName);
+ m_currentTable->setKValue(aKValue);
+ m_currentTable->setMinLoadFactor(aMinLoadFactor);
+ m_currentTable->setMaxLoadFactor(aMaxLoadFactor);
+ m_currentTable->setLogging(aStoredTable);
+ m_currentTable->setFragmentType(NdbDictionary::Object::FragAllMedium);
+ return 0;
+}//NdbSchemaOp::createTable()
+
+/******************************************************************************
+int createAttribute( const char* anAttrName,
+ KeyType aTupleyKey,
+ int anAttrSize,
+ int anArraySize,
+ AttrType anAttrType,
+ SafeType aSafeType,
+ StorageMode aStorageMode,
+ int aNullAttr,
+ int aStorageAttr );
+
+******************************************************************************/
+int
+NdbSchemaOp::createAttribute( const char* anAttrName,
+ KeyType aTupleKey,
+ int anAttrSize,
+ int anArraySize,
+ AttrType anAttrType,
+ StorageMode aStorageMode,
+ bool nullable,
+ StorageAttributeType aStorageAttr,
+ int aDistributionKeyFlag,
+ int aDistributionGroupFlag,
+ int aDistributionGroupNoOfBits,
+ bool aAutoIncrement,
+ const char* aDefaultValue)
+{
+ if (m_currentTable == 0){
+ return -1;
+ }//if
+
+ NdbDictionary::Column col(anAttrName);
+ switch(anAttrType){
+ case Signed:
+ if(anAttrSize == 64)
+ col.setType(NdbDictionary::Column::Bigint);
+ else
+ col.setType(NdbDictionary::Column::Int);
+ break;
+ case UnSigned:
+ if(anAttrSize == 64)
+ col.setType(NdbDictionary::Column::Bigunsigned);
+ else
+ col.setType(NdbDictionary::Column::Unsigned);
+ break;
+ case Float:
+ if(anAttrSize == 64)
+ col.setType(NdbDictionary::Column::Double);
+ else
+ col.setType(NdbDictionary::Column::Float);
+ break;
+ case String:
+ col.setType(NdbDictionary::Column::Char);
+ break;
+ }
+ col.setLength(anArraySize);
+ col.setNullable(nullable);
+ if(aTupleKey != NoKey)
+ col.setPrimaryKey(true);
+ else
+ col.setPrimaryKey(false);
+
+ col.setDistributionKey(aDistributionKeyFlag);
+ col.setDistributionGroup(aDistributionGroupFlag,aDistributionGroupNoOfBits);
+ col.setAutoIncrement(aAutoIncrement);
+ col.setDefaultValue(aDefaultValue != 0 ? aDefaultValue : "");
+
+ m_currentTable->addColumn(col);
+ return 0;
+}
+
+/******************************************************************************
+void release();
+
+Remark: Release all objects connected to the schemaop object.
+******************************************************************************/
+void
+NdbSchemaOp::release(){
+}//NdbSchemaOp::release()
+
+/******************************************************************************
+int sendRec()
+
+Return Value: Return 0 : send was succesful.
+ Return -1: In all other case.
+Parameters:
+Remark: Send and receive signals for schema transaction based on state
+******************************************************************************/
+int
+NdbSchemaOp::sendRec(){
+ int retVal = 0;
+ if(m_currentTable == 0){
+ retVal = -1;
+ } else {
+ retVal = theNdb->getDictionary()->createTable(* m_currentTable);
+ delete m_currentTable;
+ theSchemaCon->theError.code = theNdb->getDictionary()->getNdbError().code;
+ }
+
+ return retVal;
+}//NdbSchemaOp::sendRec()
+
+/******************************************************************************
+int init();
+
+Return Value: Return 0 : init was successful.
+ Return -1: In all other case.
+Remark: Initiates SchemaOp record after allocation.
+******************************************************************************/
+int
+NdbSchemaOp::init(NdbSchemaCon* aSchemaCon)
+{
+ theSchemaCon = aSchemaCon;
+ return 0;
+}//NdbSchemaOp::init()
diff --git a/ndb/src/ndbapi/NdbUtil.cpp b/ndb/src/ndbapi/NdbUtil.cpp
new file mode 100644
index 00000000000..5c74d251ff9
--- /dev/null
+++ b/ndb/src/ndbapi/NdbUtil.cpp
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/************************************************************************************************
+Name: NdbUtil.C
+Include:
+Link:
+Author: UABRONM Mikael Ronström UAB/B/SD
+Date: 991029
+Version: 0.4
+Description: Utility classes for NDB API
+Documentation:
+Adjust: 991029 UABRONM First version.
+Comment:
+************************************************************************************************/
+
+#include "NdbUtil.hpp"
+
+NdbLabel::NdbLabel() :
+ theNext(NULL)
+{
+}
+
+NdbLabel::~NdbLabel()
+{
+}
+
+NdbSubroutine::NdbSubroutine() :
+ theNext(NULL)
+{
+}
+
+NdbSubroutine::~NdbSubroutine()
+{
+}
+
+NdbBranch::NdbBranch() :
+ theSignal(NULL),
+ theNext(NULL)
+{
+}
+
+NdbBranch::~NdbBranch()
+{
+}
+
+NdbCall::NdbCall() :
+ theSignal(NULL),
+ theNext(NULL)
+{
+}
+
+NdbCall::~NdbCall()
+{
+}
diff --git a/ndb/src/ndbapi/NdbUtil.hpp b/ndb/src/ndbapi/NdbUtil.hpp
new file mode 100644
index 00000000000..eeee087d548
--- /dev/null
+++ b/ndb/src/ndbapi/NdbUtil.hpp
@@ -0,0 +1,99 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/************************************************************************************************
+Name: NdbUtil.H
+Include:
+Link:
+Author: UABRONM Mikael Ronström UAB/B/SD
+Date: 991029
+Version: 0.4
+Description: Utility classes for NDB API
+Documentation:
+Adjust: 991029 UABRONM First version.
+Comment:
+************************************************************************************************/
+#ifndef NdbUtil_H
+#define NdbUtil_H
+
+#include <stdlib.h>
+#include <ndb_types.h>
+#include "AttrType.hpp"
+
+class NdbApiSignal;
+class NdbOperation;
+
+class NdbLabel
+{
+friend class NdbOperation;
+friend class Ndb;
+
+private:
+ NdbLabel();
+ ~NdbLabel();
+
+ NdbLabel* theNext;
+ Uint32 theSubroutine[16];
+ Uint32 theLabelAddress[16];
+ Uint32 theLabelNo[16];
+};
+
+class NdbSubroutine
+{
+friend class NdbOperation;
+friend class Ndb;
+
+private:
+ NdbSubroutine();
+ ~NdbSubroutine();
+
+ NdbSubroutine* theNext;
+ Uint32 theSubroutineAddress[16];
+};
+
+class NdbBranch
+{
+friend class NdbOperation;
+friend class Ndb;
+
+private:
+ NdbBranch();
+ ~NdbBranch();
+
+ NdbApiSignal* theSignal;
+ Uint32 theSignalAddress;
+ Uint32 theBranchAddress;
+ Uint32 theBranchLabel;
+ Uint32 theSubroutine;
+ NdbBranch* theNext;
+};
+
+class NdbCall
+{
+friend class NdbOperation;
+friend class Ndb;
+
+private:
+ NdbCall();
+ ~NdbCall();
+
+ NdbApiSignal* theSignal;
+ Uint32 theSignalAddress;
+ Uint32 theSubroutine;
+ NdbCall* theNext;
+};
+
+#endif
diff --git a/ndb/src/ndbapi/Ndberror.cpp b/ndb/src/ndbapi/Ndberror.cpp
new file mode 100644
index 00000000000..2aa890b0918
--- /dev/null
+++ b/ndb/src/ndbapi/Ndberror.cpp
@@ -0,0 +1,635 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbError.hpp>
+#include <NdbStdio.h>
+#include <stdarg.h>
+
+#include <assert.h>
+
+struct ErrorBundle {
+ int code;
+ NdbError::Classification classification;
+ const char * message;
+};
+
+/**
+ * Shorter names in table below
+ */
+static const NdbError::Classification NE = NdbError::NoError;
+static const NdbError::Classification AE = NdbError::ApplicationError;
+static const NdbError::Classification ND = NdbError::NoDataFound;
+static const NdbError::Classification CV = NdbError::ConstraintViolation;
+static const NdbError::Classification SE = NdbError::SchemaError;
+static const NdbError::Classification UD = NdbError::UserDefinedError;
+
+static const NdbError::Classification IS = NdbError::InsufficientSpace;
+static const NdbError::Classification TR = NdbError::TemporaryResourceError;
+static const NdbError::Classification NR = NdbError::NodeRecoveryError;
+static const NdbError::Classification OL = NdbError::OverloadError;
+static const NdbError::Classification TO = NdbError::TimeoutExpired;
+static const NdbError::Classification NS = NdbError::NodeShutdown;
+
+static const NdbError::Classification UR = NdbError::UnknownResultError;
+
+static const NdbError::Classification IE = NdbError::InternalError;
+static const NdbError::Classification NI = NdbError::FunctionNotImplemented;
+static const NdbError::Classification UE = NdbError::UnknownErrorCode;
+
+static
+const
+ErrorBundle ErrorCodes[] = {
+ /**
+ * No error
+ */
+ { 0, NE, "No error" },
+
+ /**
+ * NoDataFound
+ */
+ { 626, ND, "Tuple did not exist" },
+
+ /**
+ * ConstraintViolation
+ */
+ { 630, CV, "Tuple already existed when attempting to insert" },
+ { 840, CV, "Trying to set a NOT NULL attribute to NULL" },
+ { 893, CV, "Constraint violation e.g. duplicate value in unique index" },
+
+ /**
+ * Node recovery errors
+ */
+ { 286, NR, "Node failure caused abort of transaction" },
+ { 250, NR, "Node where lock was held crashed, restart scan transaction" },
+ { 499, NR, "Scan take over error, restart scan transaction" },
+ { 1204, NR, "Temporary failure, distribution changed" },
+ { 4002, NR, "Send to NDB failed" },
+ { 4010, NR, "Node failure caused abort of transaction" },
+ { 4025, NR, "Node failure caused abort of transaction" },
+ { 4027, NR, "Node failure caused abort of transaction" },
+ { 4028, NR, "Node failure caused abort of transaction" },
+ { 4029, NR, "Node failure caused abort of transaction" },
+ { 4031, NR, "Node failure caused abort of transaction" },
+ { 4033, NR, "Send to NDB failed" },
+
+ /**
+ * Node shutdown
+ */
+ { 280, NS, "Transaction aborted due to node shutdown" },
+ // This scan trans had an active fragment scan in a LQH which have crashed
+ { 270, NS, "Transaction aborted due to node shutdown" },
+ { 1223, NS, "Read operation aborted due to node shutdown" },
+ { 4023, NS, "Transaction aborted due to node shutdown" },
+ { 4030, NS, "Transaction aborted due to node shutdown" },
+ { 4034, NS, "Transaction aborted due to node shutdown" },
+
+
+
+ /**
+ * Unknown result
+ */
+ { 4008, UR, "Receive from NDB failed" },
+ { 4009, UR, "Cluster Failure" },
+ { 4012, UR,
+ "Time-out, most likely caused by simple read or cluster failure" },
+ { 4024, UR,
+ "Time-out, most likely caused by simple read or cluster failure" },
+ { 4115, UR,
+ "Transaction was committed but all read information was not "
+ "received due to node crash" },
+
+ /**
+ * TemporaryResourceError
+ */
+ { 217, TR, "217" },
+ { 218, TR, "218" },
+ { 219, TR, "219" },
+ { 233, TR, "Out of operation records in transaction coordinator" },
+ { 275, TR, "275" },
+ { 279, TR, "Out of transaction markers in transaction coordinator" },
+ { 414, TR, "414" },
+ { 418, TR, "Out of transaction buffers in LQH" },
+ { 419, TR, "419" },
+ { 245, TR, "Too many active scans" },
+ { 488, TR, "Too many active scans" },
+ { 490, TR, "Too many active scans" },
+ { 805, TR, "Out of attrinfo records in tuple manager" },
+ { 830, TR, "Out of add fragment operation records" },
+ { 873, TR, "Out of attrinfo records for scan in tuple manager" },
+ { 1217, TR, "1217" },
+ { 1219, TR, "Out of operation records in local data manager" },
+ { 1220, TR, "1220" },
+ { 1222, TR, "Out of transaction markers in LQH" },
+ { 4021, TR, "Out of Send Buffer space in NDB API" },
+ { 4022, TR, "Out of Send Buffer space in NDB API" },
+ { 4032, TR, "Out of Send Buffer space in NDB API" },
+
+ /**
+ * InsufficientSpace
+ */
+ { 623, IS, "623" },
+ { 624, IS, "624" },
+ { 625, IS, "Out of memory in Ndb Kernel, index part" },
+ { 826, IS, "826" },
+ { 827, IS, "Out of memory in Ndb Kernel, data part" },
+ { 832, IS, "832" },
+
+ /**
+ * TimeoutExpired
+ */
+ { 266, TO, "Time-out in NDB, probably caused by deadlock" },
+ { 274, TO, "Time-out in NDB, probably caused by deadlock" }, // Scan trans timeout
+ { 296, TO, "Time-out in NDB, probably caused by deadlock" }, // Scan trans timeout
+ { 297, TO, "Time-out in NDB, probably caused by deadlock" }, // Scan trans timeout, temporary!!
+ { 237, TO, "Transaction had timed out when trying to commit it" },
+
+
+ /**
+ * OverloadError
+ */
+ { 410, OL, "Out of log file space temporarily" },
+ { 677, OL, "Index UNDO buffers overloaded" },
+ { 891, OL, "Data UNDO buffers overloaded" },
+ { 1221, OL, "REDO log buffers overloaded" },
+ { 4006, AE, "Connect failure - out of connection objects" },
+
+
+
+ /**
+ * Internal errors
+ */
+ { 892, IE, "Inconsistent hash index. The index needs to be dropped and recreated" },
+ { 895, IE, "Inconsistent ordered index. The index needs to be dropped and recreated" },
+ { 202, IE, "202" },
+ { 203, IE, "203" },
+ { 207, IE, "207" },
+ { 208, IE, "208" },
+ { 209, IE, "Communication problem, signal error" },
+ { 220, IE, "220" },
+ { 230, IE, "230" },
+ { 232, IE, "232" },
+ { 238, IE, "238" },
+ { 271, IE, "Simple Read transaction without any attributes to read" },
+ { 272, IE, "Update operation without any attributes to update" },
+ { 276, IE, "276" },
+ { 277, IE, "277" },
+ { 278, IE, "278" },
+ { 287, IE, "Index corrupted" },
+ { 631, IE, "631" },
+ { 632, IE, "632" },
+ { 702, IE, "Request to non-master" },
+ { 706, IE, "Inconsistency during table creation" },
+ { 809, IE, "809" },
+ { 812, IE, "812" },
+ { 829, IE, "829" },
+ { 833, IE, "833" },
+ { 839, IE, "Illegal null attribute" },
+ { 871, IE, "871" },
+ { 882, IE, "882" },
+ { 883, IE, "883" },
+ { 887, IE, "887" },
+ { 888, IE, "888" },
+ { 890, IE, "890" },
+ { 4000, IE, "MEMORY ALLOCATION ERROR" },
+ { 4001, IE, "Signal Definition Error" },
+ { 4005, IE, "Internal Error in NdbApi" },
+ { 4011, IE, "Internal Error in NdbApi" },
+ { 4107, IE, "Simple Transaction and Not Start" },
+ { 4108, IE, "Faulty operation type" },
+ { 4109, IE, "Faulty primary key attribute length" },
+ { 4110, IE, "Faulty length in ATTRINFO signal" },
+ { 4111, IE, "Status Error in NdbConnection" },
+ { 4113, IE, "Too many operations received" },
+ { 4320, IE, "Cannot use the same object twice to create table" },
+ { 4321, IE, "Trying to start two schema transactions" },
+ { 4344, IE, "Only DBDICT and TRIX can send requests to TRIX" },
+ { 4345, IE, "TRIX block is not available yet, probably due to node failure" },
+ { 4346, IE, "Internal error at index create/build" },
+ { 4347, IE, "Bad state at alter index" },
+ { 4348, IE, "Inconsistency detected at alter index" },
+ { 4349, IE, "Inconsistency detected at index usage" },
+
+ /**
+ * Application error
+ */
+ { 823, AE, "Too much attrinfo from application in tuple manager" },
+ { 876, AE, "876" },
+ { 877, AE, "877" },
+ { 878, AE, "878" },
+ { 879, AE, "879" },
+ { 884, AE, "Stack overflow in interpreter" },
+ { 885, AE, "Stack underflow in interpreter" },
+ { 886, AE, "More than 65535 instructions executed in interpreter" },
+ { 4256, AE, "Must call Ndb::init() before this function" },
+ { 880, AE, "Tried to read too much - too many getValue calls" },
+ { 4257, AE, "Tried to read too much - too many getValue calls" },
+
+ /**
+ * Scan application errors
+ */
+ { 242, AE, "Zero concurrency in scan"},
+ { 244, AE, "Too high concurrency in scan"},
+ { 269, AE, "No condition and attributes to read in scan"},
+ { 4600, AE, "Transaction is already started"},
+ { 4601, AE, "Transaction is not started"},
+ { 4602, AE, "You must call getNdbOperation before executeScan" },
+ { 4603, AE, "There can only be ONE operation in a scan transaction" },
+ { 4604, AE, "takeOverScanOp, opType must be UpdateRequest or DeleteRequest" },
+ { 4605, AE, "You may only call openScanRead or openScanExclusive once for each operation"},
+ { 4607, AE, "There may only be one operation in a scan transaction"},
+ { 4608, AE, "You can not takeOverScan unless you have used openScanExclusive"},
+ { 4609, AE, "You must call nextScanResult before trying to takeOverScan"},
+ { 4232, AE, "Parallelism can only be between 1 and 240" },
+ { 290, AE, "Scan not started or has been closed by kernel due to timeout" },
+
+ /**
+ * SchemaError
+ */
+ { 701, SE, "System busy with other schema operation" },
+ { 703, SE, "Invalid table format" },
+ { 704, SE, "Attribute name too long" },
+ { 705, SE, "Table name too long" },
+ { 707, SE, "No more table metadata records" },
+ { 708, SE, "No more attribute metadata records" },
+ { 709, SE, "No such table existed" },
+ { 720, SE, "Attribute name reused in table definition" },
+ { 721, SE, "Table or index with given name already exists" },
+ { 723, SE, "No such table existed" },
+ { 736, SE, "Wrong attribute size" },
+ { 737, SE, "Attribute array size too big" },
+ { 738, SE, "Record too big" },
+ { 739, SE, "Unsupported primary key length" },
+ { 740, SE, "Nullable primary key not supported" },
+ { 741, SE, "Unsupported alter table" },
+ { 241, SE, "Invalid schema object version" },
+ { 283, SE, "Table is being dropped" },
+ { 284, SE, "Table not defined in transaction coordinator" },
+ { 285, SE, "Unknown table error in transaction coordinator" },
+ { 881, SE, "Unable to create table, out of data pages" },
+ { 1225, SE, "Table not defined in local query handler" },
+ { 1226, SE, "Table is being dropped" },
+ { 1228, SE, "Cannot use drop table for drop index" },
+ { 1229, SE, "Too long frm data supplied" },
+
+ /**
+ * FunctionNotImplemented
+ */
+ { 4003, NI, "Function not implemented yet" },
+
+ /**
+ * Still uncategorized
+ */
+ { 4004, AE, "Attribute name not found in the Table" },
+
+ { 4100, AE, "Status Error in NDB" },
+ { 4101, AE, "No connections to NDB available and connect failed" },
+ { 4102, AE, "Type in NdbTamper not correct" },
+ { 4103, AE, "No schema connections to NDB available and connect failed" },
+ { 4104, AE, "Ndb Init in wrong state, destroy Ndb object and create a new" },
+ { 4105, AE, "Too many Ndb objects" },
+ { 4106, AE, "All Not NULL attribute have not been defined" },
+ { 4114, AE, "Transaction is already completed" },
+ { 4116, AE, "Operation was not defined correctly, probably missing a key" },
+ { 4117, AE, "Could not start transporter, configuration error"},
+ { 4118, AE, "Parameter error in API call" },
+ { 4300, AE, "Tuple Key Type not correct" },
+ { 4301, AE, "Fragment Type not correct" },
+ { 4302, AE, "Minimum Load Factor not correct" },
+ { 4303, AE, "Maximum Load Factor not correct" },
+ { 4304, AE, "Maximum Load Factor smaller than Minimum" },
+ { 4305, AE, "K value must currently be set to 6" },
+ { 4306, AE, "Memory Type not correct" },
+ { 4307, AE, "Invalid table name" },
+ { 4308, AE, "Attribute Size not correct" },
+ { 4309, AE, "Fixed array too large, maximum 64000 bytes" },
+ { 4310, AE, "Attribute Type not correct" },
+ { 4311, AE, "Storage Mode not correct" },
+ { 4312, AE, "Null Attribute Type not correct" },
+ { 4313, AE, "Index only storage for non-key attribute" },
+ { 4314, AE, "Storage Type of attribute not correct" },
+ { 4315, AE, "No more key attributes allowed after defining variable length key attribute" },
+ { 4316, AE, "Key attributes are not allowed to be NULL attributes" },
+ { 4317, AE, "Too many primary keys defined in table" },
+ { 4318, AE, "Invalid attribute name" },
+ { 4319, AE, "createAttribute called at erroneus place" },
+ { 4322, AE, "Attempt to define distribution key when not prepared to" },
+ { 4323, AE, "Distribution Key set on table but not defined on first attribute" },
+ { 4324, AE, "Attempt to define distribution group when not prepared to" },
+ { 4325, AE, "Distribution Group set on table but not defined on first attribute" },
+ { 4326, AE, "Distribution Group with erroneus number of bits" },
+ { 4327, AE, "Distribution Group with 1 byte attribute is not allowed" },
+ { 4328, AE, "Disk memory attributes not yet supported" },
+ { 4329, AE, "Variable stored attributes not yet supported" },
+ { 4330, AE, "Table names limited to 127 bytes" },
+ { 4331, AE, "Attribute names limited to 31 bytes" },
+ { 4332, AE, "Maximum 2000 attributes in a table" },
+ { 4333, AE, "Maximum 4092 bytes long keys allowed" },
+ { 4334, AE, "Attribute properties length limited to 127 bytes" },
+
+ { 4400, AE, "Status Error in NdbSchemaCon" },
+ { 4401, AE, "Only one schema operation per schema transaction" },
+ { 4402, AE, "No schema operation defined before calling execute" },
+
+ { 4500, AE, "Cannot handle more than 2048 tables in NdbApi" },
+ { 4501, AE, "Insert in hash table failed when getting table information from Ndb" },
+ { 4502, AE, "GetValue not allowed in Update operation" },
+ { 4503, AE, "GetValue not allowed in Insert operation" },
+ { 4504, AE, "SetValue not allowed in Read operation" },
+ { 4505, AE, "NULL value not allowed in primary key search" },
+ { 4506, AE, "Missing getValue/setValue when calling execute" },
+ { 4507, AE, "Missing operation request when calling execute" },
+
+ { 4200, AE, "Status Error when defining an operation" },
+ { 4201, AE, "Variable Arrays not yet supported" },
+ { 4202, AE, "Set value on tuple key attribute is not allowed" },
+ { 4203, AE, "Trying to set a NOT NULL attribute to NULL" },
+ { 4204, AE, "Set value and Read/Delete Tuple is incompatible" },
+ { 4205, AE, "No Key attribute used to define tuple" },
+ { 4206, AE, "Not allowed to equal key attribute twice" },
+ { 4207, AE, "Key size is limited to 4092 bytes" },
+ { 4208, AE, "Trying to read a non-stored attribute" },
+ { 4209, AE, "Length parameter in equal/setValue is incorrect" },
+ { 4210, AE, "Ndb sent more info than the length he specified" },
+ { 4211, AE, "Inconsistency in list of NdbRecAttr-objects" },
+ { 4212, AE, "Ndb reports NULL value on Not NULL attribute" },
+ { 4213, AE, "Not all data of an attribute has been received" },
+ { 4214, AE, "Not all attributes have been received" },
+ { 4215, AE, "More data received than reported in TCKEYCONF message" },
+ { 4216, AE, "More than 8052 bytes in setValue cannot be handled" },
+ { 4217, AE, "It is not allowed to increment any other than unsigned ints" },
+ { 4218, AE, "Currently not allowed to increment NULL-able attributes" },
+ { 4219, AE, "Maximum size of interpretative attributes are 64 bits" },
+ { 4220, AE, "Maximum size of interpretative attributes are 64 bits" },
+ { 4221, AE, "Trying to jump to a non-defined label" },
+ { 4222, AE, "Label was not found, internal error" },
+ { 4223, AE, "Not allowed to create jumps to yourself" },
+ { 4224, AE, "Not allowed to jump to a label in a different subroutine" },
+ { 4225, AE, "All primary keys defined, call setValue/getValue"},
+ { 4226, AE, "Bad number when defining a label" },
+ { 4227, AE, "Bad number when defining a subroutine" },
+ { 4228, AE, "Illegal interpreter function in scan definition" },
+ { 4229, AE, "Illegal register in interpreter function definition" },
+ { 4230, AE, "Illegal state when calling getValue, probably not a read" },
+ { 4231, AE, "Illegal state when calling interpreter routine" },
+ { 4233, AE, "Calling execute (synchronous) when already prepared asynchronous transaction exists" },
+ { 4234, AE, "Illegal to call setValue in this state" },
+ { 4235, AE, "No callback from execute" },
+ { 4236, AE, "Trigger name too long" },
+ { 4237, AE, "Too many triggers" },
+ { 4238, AE, "Trigger not found" },
+ { 4239, AE, "Trigger with given name already exists"},
+ { 4240, AE, "Unsupported trigger type"},
+ { 4241, AE, "Index name too long" },
+ { 4242, AE, "Too many indexes" },
+ { 4243, AE, "Index not found" },
+ { 4244, AE, "Index or table with given name already exists" },
+ { 4245, AE, "Index attribute must be defined as stored, i.e. the StorageAttributeType must be defined as NormalStorageAttribute"},
+ { 4246, AE, "Combined index attributes are not allowed to be NULL attributes" },
+ { 4247, AE, "Illegal index/trigger create/drop/alter request" },
+ { 4248, AE, "Trigger/index name invalid" },
+ { 4249, AE, "Invalid table" },
+ { 4250, AE, "Invalid index type or index logging option" },
+ { 4251, AE, "Cannot create unique index, duplicate keys found" },
+ { 4252, AE, "Failed to allocate space for index" },
+ { 4253, AE, "Failed to create index table" },
+ { 4254, AE, "Table not an index table" },
+ { 4255, AE, "Hash index attributes must be specified in same order as table attributes" },
+ { 4258, AE, "Cannot create unique index, duplicate attributes found in definition" },
+ { 4259, AE, "Invalid set of range scan bounds" },
+ { 4260, UD, "NdbScanFilter: Operator is not defined in NdbScanFilter::Group"},
+ { 4261, UD, "NdbScanFilter: Column is NULL"},
+ { 4262, UD, "NdbScanFilter: Condition is out of bounds"}
+
+};
+
+static
+const
+int NbErrorCodes = sizeof(ErrorCodes)/sizeof(ErrorBundle);
+
+struct ErrorStatusClassification {
+ NdbError::Status status;
+ NdbError::Classification classification;
+};
+
+/**
+ * Mapping between classification and status
+ */
+static
+const
+ErrorStatusClassification StatusClassificationMapping[] = {
+ { NdbError::Success, NdbError::NoError },
+ { NdbError::PermanentError, NdbError::ApplicationError },
+ { NdbError::PermanentError, NdbError::NoDataFound },
+ { NdbError::PermanentError, NdbError::ConstraintViolation },
+ { NdbError::PermanentError, NdbError::SchemaError },
+ { NdbError::PermanentError, NdbError::UserDefinedError },
+ { NdbError::PermanentError, NdbError::InsufficientSpace },
+
+ { NdbError::TemporaryError, NdbError::TemporaryResourceError },
+ { NdbError::TemporaryError, NdbError::NodeRecoveryError },
+ { NdbError::TemporaryError, NdbError::OverloadError },
+ { NdbError::TemporaryError, NdbError::TimeoutExpired },
+ { NdbError::TemporaryError, NdbError::NodeShutdown },
+
+ { NdbError::UnknownResult , NdbError::UnknownResultError },
+ { NdbError::UnknownResult , NdbError::UnknownErrorCode },
+
+ { NdbError::PermanentError, NdbError::InternalError },
+ { NdbError::PermanentError, NdbError::FunctionNotImplemented }
+};
+
+static
+const
+int Nb = sizeof(StatusClassificationMapping)/sizeof(ErrorStatusClassification);
+
+/**
+ * Complete all fields of an NdbError given the error code
+ * and details
+ */
+static
+void
+set(NdbError & error, int code, const char * details, ...){
+ error.code = code;
+
+ va_list ap;
+ va_start(ap, details);
+ vsnprintf(error.details, sizeof(error.details), details, ap);
+ va_end(ap);
+}
+
+static
+void
+update(const NdbError & _err){
+ NdbError & error = (NdbError &) _err;
+
+ bool found = false;
+ for(int i = 0; i<NbErrorCodes; i++){
+ if(ErrorCodes[i].code == error.code){
+ error.classification = ErrorCodes[i].classification;
+ error.message = ErrorCodes[i].message;
+ found = true;
+ break;
+ }
+ }
+
+ if(!found){
+ error.classification = NdbError::UnknownErrorCode;
+ error.message = "Unknown error code";
+ }
+
+ found = false;
+ for(int i = 0; i<Nb; i++){
+ if(StatusClassificationMapping[i].classification == error.classification){
+ error.status = StatusClassificationMapping[i].status;
+ found = true;
+ break;
+ }
+ }
+ if(!found){
+ error.status = NdbError::UnknownResult;
+ }
+
+ error.details = 0;
+}
+
+bool
+checkErrorCodes(){
+ for(int i = 0; i<NbErrorCodes; i++)
+ for(int j = i+1; j<NbErrorCodes; j++)
+ if(ErrorCodes[i].code == ErrorCodes[j].code){
+ printf("ErrorCode %d is defined multiple times!!\n",
+ ErrorCodes[i].code);
+ assert(0);
+ }
+
+ return true;
+}
+
+static const bool a = checkErrorCodes();
+
+#if CHECK_ERRORCODES
+int main(void){
+ checkErrorCodes();
+ return 0;
+}
+#endif
+
+#include <NdbOut.hpp>
+
+/**
+ * operators
+ */
+NdbOut &
+operator<<(NdbOut & out, const NdbError & error){
+ if(error.message != 0)
+ out << error.code << ": " << error.message;
+ else
+ out << error.code << ": ";
+ return out;
+}
+
+NdbOut &
+operator<<(NdbOut & out, const NdbError::Status & status){
+ switch(status) {
+ case NdbError::Success: out << "Success"; break;
+ case NdbError::TemporaryError: out << "Temporary error"; break;
+ case NdbError::PermanentError: out << "Permanent error"; break;
+ case NdbError::UnknownResult: out << "Unknown result"; break;
+ }
+ return out;
+}
+
+NdbOut &
+operator<<(NdbOut & out, const NdbError::Classification & classification){
+ switch(classification) {
+ case NdbError::NoError: out << "No error"; break;
+ case NdbError::ApplicationError: out << "Application error"; break;
+ case NdbError::NoDataFound: out << "No data found"; break;
+ case NdbError::ConstraintViolation: out << "Constraint violation"; break;
+ case NdbError::SchemaError: out << "Schema error"; break;
+ case NdbError::UserDefinedError: out << "User defined error"; break;
+ case NdbError::InsufficientSpace: out << "Insufficient space"; break;
+ case NdbError::TemporaryResourceError: out << "Temporary Resource error";
+ break;
+ case NdbError::NodeRecoveryError: out << "Node Recovery error"; break;
+ case NdbError::OverloadError: out << "Overload error"; break;
+ case NdbError::TimeoutExpired: out << "Timeout expired"; break;
+ case NdbError::UnknownResultError: out << "Unknown result error"; break;
+ case NdbError::InternalError: out << "Internal error"; break;
+ case NdbError::FunctionNotImplemented: out << "Function not implemented";
+ break;
+ case NdbError::UnknownErrorCode: out << "Unknown error code"; break;
+ case NdbError::NodeShutdown: out << "Node shutdown"; break;
+ }
+ return out;
+}
+
+/******************************************************
+ *
+ */
+#include "NdbImpl.hpp"
+#include "NdbDictionaryImpl.hpp"
+#include <NdbSchemaCon.hpp>
+#include <NdbOperation.hpp>
+#include <NdbConnection.hpp>
+
+
+const
+NdbError &
+Ndb::getNdbError(int code){
+ theError.code = code;
+ update(theError);
+ return theError;
+}
+
+const
+NdbError &
+Ndb::getNdbError() const {
+ update(theError);
+ return theError;
+}
+
+const
+NdbError &
+NdbDictionaryImpl::getNdbError() const {
+ update(m_error);
+ return m_error;
+}
+
+const
+NdbError &
+NdbConnection::getNdbError() const {
+ update(theError);
+ return theError;
+}
+
+const
+NdbError &
+NdbOperation::getNdbError() const {
+ update(theError);
+ return theError;
+}
+
+const
+NdbError &
+NdbSchemaCon::getNdbError() const {
+ update(theError);
+ return theError;
+}
+
+
+
diff --git a/ndb/src/ndbapi/Ndbif.cpp b/ndb/src/ndbapi/Ndbif.cpp
new file mode 100644
index 00000000000..e334c1bcc39
--- /dev/null
+++ b/ndb/src/ndbapi/Ndbif.cpp
@@ -0,0 +1,1356 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbApiSignal.hpp"
+#include "AttrType.hpp"
+#include "NdbImpl.hpp"
+#include "NdbSchemaOp.hpp"
+#include "NdbSchemaCon.hpp"
+#include "NdbOperation.hpp"
+#include "NdbIndexOperation.hpp"
+#include "NdbScanReceiver.hpp"
+#include "NdbConnection.hpp"
+#include "NdbRecAttr.hpp"
+#include "NdbReceiver.hpp"
+#include "API.hpp"
+
+#include <signaldata/TcCommit.hpp>
+#include <signaldata/TcKeyFailConf.hpp>
+#include <signaldata/TcKeyConf.hpp>
+#include <signaldata/TestOrd.hpp>
+#include <signaldata/CreateIndx.hpp>
+#include <signaldata/DropIndx.hpp>
+#include <signaldata/TcIndx.hpp>
+
+#include <ndb_limits.h>
+#include <NdbOut.hpp>
+#include <NdbTick.h>
+
+#include <assert.h>
+
+/******************************************************************************
+ * int init( int aNrOfCon, int aNrOfOp );
+ *
+ * Return Value: Return 0 : init was successful.
+ * Return -1: In all other case.
+ * Parameters: aNrOfCon : Number of connections offered to the application.
+ * aNrOfOp : Number of operations offered to the application.
+ * Remark: Create pointers and idle list Synchronous.
+ ****************************************************************************/
+int
+Ndb::init(int aMaxNoOfTransactions)
+{
+ int i;
+ int aNrOfCon;
+ int aNrOfOp;
+ int tMaxNoOfTransactions;
+ NdbApiSignal* tSignal[16]; // Initiate free list of 16 signal objects
+ if (theInitState != NotInitialised) {
+ switch(theInitState){
+ case InitConfigError:
+ theError.code = 4117;
+ break;
+ default:
+ theError.code = 4104;
+ break;
+ }
+ return -1;
+ }//if
+ theInitState = StartingInit;
+ TransporterFacade * theFacade = TransporterFacade::instance();
+ theFacade->lock_mutex();
+
+ const int tBlockNo = theFacade->open(this,
+ executeMessage,
+ statusMessage);
+
+
+ if ( tBlockNo == -1 ) {
+ theError.code = 4105;
+ theFacade->unlock_mutex();
+ return -1; // no more free blocknumbers
+ }//if
+
+ theNdbBlockNumber = tBlockNo;
+
+ theNode = theFacade->ownId();
+ theMyRef = numberToRef(theNdbBlockNumber, theNode);
+
+ for (i = 1; i < MAX_NDB_NODES; i++){
+ if (theFacade->getIsNodeDefined(i)){
+ theDBnodes[theNoOfDBnodes] = i;
+ theNoOfDBnodes++;
+ }
+ }
+
+ theFirstTransId = ((Uint64)theNdbBlockNumber << 52)+((Uint64)theNode << 40);
+ theFacade->unlock_mutex();
+
+
+ theDictionary = new NdbDictionaryImpl(*this);
+ if (theDictionary == NULL) {
+ theError.code = 4000;
+ return -1;
+ }
+ theDictionary->setTransporter(this, theFacade);
+
+ aNrOfCon = theNoOfDBnodes;
+ aNrOfOp = 2*theNoOfDBnodes;
+
+ // Create connection object in a linked list
+ if((createConIdleList(aNrOfCon)) == -1){
+ theError.code = 4000;
+ goto error_handler;
+ }
+
+ // Create operations in a linked list
+ if((createOpIdleList(aNrOfOp)) == -1){
+ theError.code = 4000;
+ goto error_handler;
+ }
+
+ tMaxNoOfTransactions = aMaxNoOfTransactions * 3;
+ if (tMaxNoOfTransactions > 1024) {
+ tMaxNoOfTransactions = 1024;
+ }//if
+ theMaxNoOfTransactions = tMaxNoOfTransactions;
+
+ thePreparedTransactionsArray = new NdbConnection* [tMaxNoOfTransactions];
+ theSentTransactionsArray = new NdbConnection* [tMaxNoOfTransactions];
+ theCompletedTransactionsArray = new NdbConnection* [tMaxNoOfTransactions];
+
+ if ((thePreparedTransactionsArray == NULL) ||
+ (theSentTransactionsArray == NULL) ||
+ (theCompletedTransactionsArray == NULL)) {
+ goto error_handler;
+ }//if
+
+ for (i = 0; i < tMaxNoOfTransactions; i++) {
+ thePreparedTransactionsArray[i] = NULL;
+ theSentTransactionsArray[i] = NULL;
+ theCompletedTransactionsArray[i] = NULL;
+ }//for
+
+ startTransactionNodeSelectionData.init(theNoOfDBnodes, theDBnodes);
+
+ for (i = 0; i < 16; i++){
+ tSignal[i] = getSignal();
+ if(tSignal[i] == NULL) {
+ theError.code = 4000;
+ goto error_handler;
+ }
+ }
+ for (i = 0; i < 16; i++)
+ releaseSignal(tSignal[i]);
+
+ theInitState = Initialised;
+
+ theCommitAckSignal = new NdbApiSignal(theMyRef);
+ return 0;
+
+error_handler:
+ ndbout << "error_handler" << endl;
+ releaseTransactionArrays();
+ while ( theConIdleList != NULL )
+ freeNdbCon();
+ while ( theSignalIdleList != NULL )
+ freeSignal();
+ while (theRecAttrIdleList != NULL)
+ freeRecAttr();
+ while (theOpIdleList != NULL)
+ freeOperation();
+
+ delete theDictionary;
+ TransporterFacade::instance()->close(theNdbBlockNumber);
+ return -1;
+}
+
+void
+Ndb::releaseTransactionArrays()
+{
+ if (thePreparedTransactionsArray != NULL) {
+ delete [] thePreparedTransactionsArray;
+ }//if
+ if (theSentTransactionsArray != NULL) {
+ delete [] theSentTransactionsArray;
+ }//if
+ if (theCompletedTransactionsArray != NULL) {
+ delete [] theCompletedTransactionsArray;
+ }//if
+}//Ndb::releaseTransactionArrays()
+
+void
+Ndb::executeMessage(void* NdbObject,
+ NdbApiSignal * aSignal,
+ LinearSectionPtr ptr[3])
+{
+ Ndb* tNdb = (Ndb*)NdbObject;
+ tNdb->handleReceivedSignal(aSignal, ptr);
+}
+
+void
+Ndb::statusMessage(void* NdbObject, NodeId a_node, bool alive, bool nfComplete)
+{
+ Ndb* tNdb = (Ndb*)NdbObject;
+ if (alive) {
+ if (nfComplete) {
+ assert(0);
+ }//if
+ } else {
+ if (nfComplete) {
+ tNdb->report_node_failure_completed(a_node);
+ } else {
+ tNdb->report_node_failure(a_node);
+ }//if
+ }//if
+ NdbDictInterface::execNodeStatus(&tNdb->theDictionary->m_receiver,
+ a_node, alive, nfComplete);
+}
+
+void
+Ndb::report_node_failure(Uint32 node_id)
+{
+ /**
+ * We can only set the state here since this object can execute
+ * simultaneously.
+ *
+ * This method is only called by ClusterMgr (via lots of methods)
+ */
+ the_release_ind[node_id] = 1;
+ theWaiter.nodeFail(node_id);
+}//Ndb::report_node_failure()
+
+
+void
+Ndb::report_node_failure_completed(Uint32 node_id)
+{
+ abortTransactionsAfterNodeFailure(node_id);
+
+}//Ndb::report_node_failure_completed()
+
+/***************************************************************************
+void abortTransactionsAfterNodeFailure();
+
+Remark: Abort all transactions in theSentTransactionsArray after connection
+ to one node has failed
+****************************************************************************/
+void
+Ndb::abortTransactionsAfterNodeFailure(Uint16 aNodeId)
+{
+ Uint32 tNoSentTransactions = theNoOfSentTransactions;
+ for (int i = tNoSentTransactions - 1; i >= 0; i--) {
+ NdbConnection* localCon = theSentTransactionsArray[i];
+ if (localCon->getConnectedNodeId() == aNodeId ) {
+ const SendStatusType sendStatus = localCon->theSendStatus;
+ if (sendStatus == sendTC_OP || sendStatus == sendTC_COMMIT) {
+ /*
+ A transaction was interrupted in the prepare phase by a node
+ failure. Since the transaction was not found in the phase
+ after the node failure it cannot have been committed and
+ we report a normal node failure abort.
+ */
+ localCon->setOperationErrorCodeAbort(4010);
+ localCon->theCompletionStatus = CompletedFailure;
+ } else if (sendStatus == sendTC_ROLLBACK) {
+ /*
+ We aimed for abort and abort we got even if it was by a node
+ failure. We will thus report it as a success.
+ */
+ localCon->theCompletionStatus = CompletedSuccess;
+ } else {
+#ifdef VM_TRACE
+ printState("abortTransactionsAfterNodeFailure %x", this);
+ abort();
+#endif
+ }//
+ /*
+ All transactions arriving here have no connection to the kernel
+ intact since the node was failing and they were aborted. Thus we
+ set commit state to Aborted and set state to release on close.
+ */
+ localCon->theCommitStatus = Aborted;
+ localCon->theReleaseOnClose = true;
+ completedTransaction(localCon);
+ }//if
+ }//for
+ return;
+}//Ndb::abortTransactionsAfterNodeFailure()
+
+/****************************************************************************
+void handleReceivedSignal(NdbApiSignal* aSignal);
+
+Parameters: aSignal: The signal object.
+Remark: Send all operations belonging to this connection.
+*****************************************************************************/
+void
+Ndb::handleReceivedSignal(NdbApiSignal* aSignal, LinearSectionPtr ptr[3])
+{
+ NdbOperation* tOp;
+ NdbIndexOperation* tIndexOp;
+ NdbConnection* tCon;
+ int tReturnCode;
+ const Uint32* tDataPtr = aSignal->getDataPtr();
+ const Uint32 tWaitState = theWaiter.m_state;
+ const Uint32 tSignalNumber = aSignal->readSignalNumber();
+ const Uint32 tFirstData = *tDataPtr;
+
+ /*
+ In order to support 64 bit processes in the application we need to use
+ id's rather than a direct pointer to the object used. It is also a good
+ idea that one cannot corrupt the application code by sending a corrupt
+ memory pointer.
+
+ All signals received by the API requires the first data word to be such
+ an id to the receiving object.
+ */
+
+ switch (tSignalNumber){
+ case GSN_TCKEYCONF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ const TcKeyConf * const keyConf = (TcKeyConf *)tDataPtr;
+ const BlockReference aTCRef = aSignal->theSendersBlockRef;
+
+ tCon = void2con(tFirstDataPtr);
+ if ((tCon->checkMagicNumber() == 0) &&
+ (tCon->theSendStatus == sendTC_OP)) {
+ tReturnCode = tCon->receiveTCKEYCONF(keyConf, aSignal->getLength());
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ }//if
+
+ if(TcKeyConf::getMarkerFlag(keyConf->confInfo)){
+ NdbConnection::sendTC_COMMIT_ACK(theCommitAckSignal,
+ keyConf->transId1,
+ keyConf->transId2,
+ aTCRef);
+ }
+
+ return;
+ }//if
+ goto InvalidSignal;
+
+ return;
+ }
+ case GSN_READCONF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ tOp = void2rec_op(tFirstDataPtr);
+ if (tOp->checkMagicNumber() == 0) {
+ tCon = tOp->theNdbCon;
+ if (tCon != NULL) {
+ if (tCon->theSendStatus == sendTC_OP) {
+ tReturnCode = tOp->receiveREAD_CONF(tDataPtr,
+ aSignal->getLength());
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ }//if
+ }//if
+ }//if
+ }//if
+ return;
+ }
+ case GSN_TRANSID_AI:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ // ndbout << "*** GSN_TRANSID_AI ***" << endl;
+ NdbReceiver* tRec = void2rec(tFirstDataPtr);
+ if (tRec->getType() == NdbReceiver::NDB_OPERATION){
+ // tOp = (NdbOperation*)tRec->getOwner();
+ tOp = void2rec_op(tFirstDataPtr);
+ // ndbout << "NDB_OPERATION" << endl;
+ if (tOp->checkMagicNumber() == 0) {
+ tCon = tOp->theNdbCon;
+ if (tCon != NULL) {
+ if (tCon->theSendStatus == sendTC_OP) {
+ tReturnCode = tOp->receiveTRANSID_AI(tDataPtr,
+ aSignal->getLength());
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ break;
+ }
+ }
+ }
+ }
+ } else if (tRec->getType() == NdbReceiver::NDB_INDEX_OPERATION){
+ // tOp = (NdbIndexOperation*)tRec->getOwner();
+ tOp = void2rec_iop(tFirstDataPtr);
+ // ndbout << "NDB_INDEX_OPERATION" << endl;
+ if (tOp->checkMagicNumber() == 0) {
+ tCon = tOp->theNdbCon;
+ if (tCon != NULL) {
+ if (tCon->theSendStatus == sendTC_OP) {
+ tReturnCode = tOp->receiveTRANSID_AI(tDataPtr,
+ aSignal->getLength());
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ break;
+ }
+ }
+ }
+ }
+ } else if (tRec->getType() == NdbReceiver::NDB_SCANRECEIVER) {
+ // NdbScanReceiver* tScanRec = (NdbScanReceiver*)tRec->getOwner();
+ // NdbScanReceiver* tScanRec =
+ // (NdbScanReceiver*)(void2rec(tFirstDataPtr)->getOwner());
+ NdbScanReceiver* tScanRec = void2rec_srec(tFirstDataPtr);
+ // ndbout << "NDB_SCANRECEIVER" << endl;
+ if(tScanRec->checkMagicNumber() == 0){
+ tReturnCode = tScanRec->receiveTRANSID_AI_SCAN(aSignal);
+ if (tReturnCode != -1) {
+ theWaiter.m_state = NO_WAIT;
+ break;
+ }
+ }
+ } else {
+#ifdef NDB_NO_DROPPED_SIGNAL
+ abort();
+#endif
+ goto InvalidSignal;
+ }
+ return;
+ }
+ case GSN_TCKEY_FAILCONF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ const TcKeyFailConf * const failConf = (TcKeyFailConf *)tDataPtr;
+ const BlockReference aTCRef = aSignal->theSendersBlockRef;
+
+ tOp = void2rec_op(tFirstDataPtr);
+
+ if (tOp->checkMagicNumber() == 0) {
+ tCon = tOp->theNdbCon;
+ if (tCon != NULL) {
+ if ((tCon->theSendStatus == sendTC_OP) ||
+ (tCon->theSendStatus == sendTC_COMMIT)) {
+ tReturnCode = tCon->receiveTCKEY_FAILCONF(failConf);
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ }//if
+ }//if
+ }//if
+ }//if
+
+ if(tFirstData & 1){
+ NdbConnection::sendTC_COMMIT_ACK(theCommitAckSignal,
+ failConf->transId1,
+ failConf->transId2,
+ aTCRef);
+ }
+ return;
+ }
+ case GSN_TCKEY_FAILREF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ tOp = void2rec_op(tFirstDataPtr);
+ if (tOp->checkMagicNumber() == 0) {
+ tCon = tOp->theNdbCon;
+ if (tCon != NULL) {
+ if ((tCon->theSendStatus == sendTC_OP) ||
+ (tCon->theSendStatus == sendTC_ROLLBACK)) {
+ tReturnCode = tCon->receiveTCKEY_FAILREF(aSignal);
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ return;
+ }//if
+ }//if
+ }//if
+ }//if
+ return;
+ }
+ case GSN_TCKEYREF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ tOp = void2rec_op(tFirstDataPtr);
+ if (tOp->checkMagicNumber() == 0) {
+ tCon = tOp->theNdbCon;
+ if (tCon != NULL) {
+ if (tCon->theSendStatus == sendTC_OP) {
+ tReturnCode = tOp->receiveTCKEYREF(aSignal);
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ }//if
+ return;
+ }//if
+ }//if
+ } //if
+ goto InvalidSignal;
+ return;
+ }
+ case GSN_TC_COMMITCONF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ const TcCommitConf * const commitConf = (TcCommitConf *)tDataPtr;
+ const BlockReference aTCRef = aSignal->theSendersBlockRef;
+
+ tCon = void2con(tFirstDataPtr);
+ if ((tCon->checkMagicNumber() == 0) &&
+ (tCon->theSendStatus == sendTC_COMMIT)) {
+ tReturnCode = tCon->receiveTC_COMMITCONF(commitConf);
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ }//if
+
+ if(tFirstData & 1){
+ NdbConnection::sendTC_COMMIT_ACK(theCommitAckSignal,
+ commitConf->transId1,
+ commitConf->transId2,
+ aTCRef);
+ }
+ return;
+ }
+ goto InvalidSignal;
+ return;
+ }
+
+ case GSN_TC_COMMITREF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ tCon = void2con(tFirstDataPtr);
+ if ((tCon->checkMagicNumber() == 0) &&
+ (tCon->theSendStatus == sendTC_COMMIT)) {
+ tReturnCode = tCon->receiveTC_COMMITREF(aSignal);
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ return;
+ }//if
+ }//if
+ return;
+ }
+ case GSN_TCROLLBACKCONF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ tCon = void2con(tFirstDataPtr);
+ if ((tCon->checkMagicNumber() == 0) &&
+ (tCon->theSendStatus == sendTC_ROLLBACK)) {
+ tReturnCode = tCon->receiveTCROLLBACKCONF(aSignal);
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ }//if
+ }//if
+ return;
+ }
+ case GSN_TCROLLBACKREF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ tCon = void2con(tFirstDataPtr);
+ if ((tCon->checkMagicNumber() == 0) &&
+ (tCon->theSendStatus == sendTC_ROLLBACK)) {
+ tReturnCode = tCon->receiveTCROLLBACKREF(aSignal);
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ return;
+ }//if
+ }//if
+ return;
+ }
+ case GSN_TCROLLBACKREP:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ tCon = void2con(tFirstDataPtr);
+ if (tCon->checkMagicNumber() == 0) {
+ tReturnCode = tCon->receiveTCROLLBACKREP(aSignal);
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ }//if
+ }//if
+ return;
+ }
+ case GSN_TCSEIZECONF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ if (tWaitState != WAIT_TC_SEIZE) {
+ return;
+ }//if
+ tCon = void2con(tFirstDataPtr);
+ if (tCon->checkMagicNumber() != 0) {
+ return;
+ }//if
+ tReturnCode = tCon->receiveTCSEIZECONF(aSignal);
+ if (tReturnCode != -1) {
+ theWaiter.m_state = NO_WAIT;
+ } else {
+ return;
+ }//if
+ break;
+ }
+ case GSN_TCSEIZEREF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ if (tWaitState != WAIT_TC_SEIZE) {
+ return;
+ }//if
+ tCon = void2con(tFirstDataPtr);
+ if (tCon->checkMagicNumber() != 0) {
+ return;
+ }//if
+ tReturnCode = tCon->receiveTCSEIZEREF(aSignal);
+ if (tReturnCode != -1) {
+ theWaiter.m_state = NO_WAIT;
+ } else {
+ return;
+ }//if
+ break;
+ }
+ case GSN_TCRELEASECONF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ if (tWaitState != WAIT_TC_RELEASE) {
+ goto InvalidSignal;
+ }//if
+ tCon = void2con(tFirstDataPtr);
+ if (tCon->checkMagicNumber() != 0) {
+ goto InvalidSignal;
+ }//if
+ tReturnCode = tCon->receiveTCRELEASECONF(aSignal);
+ if (tReturnCode != -1) {
+ theWaiter.m_state = NO_WAIT;
+ }//if
+ break;
+ }
+ case GSN_TCRELEASEREF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ if (tWaitState != WAIT_TC_RELEASE) {
+ goto InvalidSignal;
+ }//if
+ tCon = void2con(tFirstDataPtr);
+ if (tCon->checkMagicNumber() != 0) {
+ goto InvalidSignal;
+ }//if
+ tReturnCode = tCon->receiveTCRELEASEREF(aSignal);
+ if (tReturnCode != -1) {
+ theWaiter.m_state = NO_WAIT;
+ }//if
+ break;
+ }
+
+ case GSN_GET_TABINFOREF:
+ case GSN_GET_TABINFO_CONF:
+ case GSN_CREATE_TABLE_REF:
+ case GSN_CREATE_TABLE_CONF:
+ case GSN_DROP_TABLE_CONF:
+ case GSN_DROP_TABLE_REF:
+ case GSN_ALTER_TABLE_CONF:
+ case GSN_ALTER_TABLE_REF:
+ case GSN_CREATE_INDX_CONF:
+ case GSN_CREATE_INDX_REF:
+ case GSN_DROP_INDX_CONF:
+ case GSN_DROP_INDX_REF:
+ case GSN_CREATE_EVNT_CONF:
+ case GSN_CREATE_EVNT_REF:
+ case GSN_DROP_EVNT_CONF:
+ case GSN_DROP_EVNT_REF:
+ case GSN_LIST_TABLES_CONF:
+ NdbDictInterface::execSignal(&theDictionary->m_receiver,
+ aSignal, ptr);
+ break;
+
+ case GSN_SUB_META_DATA:
+ case GSN_SUB_REMOVE_CONF:
+ case GSN_SUB_REMOVE_REF:
+ break; // ignore these signals
+ case GSN_SUB_GCP_COMPLETE_REP:
+ case GSN_SUB_START_CONF:
+ case GSN_SUB_START_REF:
+ case GSN_SUB_TABLE_DATA:
+ case GSN_SUB_STOP_CONF:
+ case GSN_SUB_STOP_REF:
+ NdbDictInterface::execSignal(&theDictionary->m_receiver,
+ aSignal, ptr);
+ break;
+
+ case GSN_DIHNDBTAMPER:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ if (tWaitState != WAIT_NDB_TAMPER)
+ return;
+ tCon = void2con(tFirstDataPtr);
+ if (tCon->checkMagicNumber() != 0)
+ return;
+ tReturnCode = tCon->receiveDIHNDBTAMPER(aSignal);
+ if (tReturnCode != -1)
+ theWaiter.m_state = NO_WAIT;
+ break;
+ }
+ case GSN_SCAN_TABCONF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ //ndbout << "*** GSN_SCAN_TABCONF *** " << endl;
+ if (tWaitState != WAIT_SCAN){
+ return;
+ }
+ tCon = void2con(tFirstDataPtr);
+ if (tCon->checkMagicNumber() != 0)
+ return;
+ tReturnCode = tCon->receiveSCAN_TABCONF(aSignal);
+ if (tReturnCode != -1)
+ theWaiter.m_state = NO_WAIT;
+ break;
+ }
+ case GSN_SCAN_TABREF:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ if (tWaitState == WAIT_SCAN){
+ tCon = void2con(tFirstDataPtr);
+ if (tCon->checkMagicNumber() == 0){
+ tReturnCode = tCon->receiveSCAN_TABREF(aSignal);
+ if (tReturnCode != -1){
+ theWaiter.m_state = NO_WAIT;
+ }
+ break;
+ }
+ }
+ goto InvalidSignal;
+ }
+ case GSN_SCAN_TABINFO:
+ {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ //ndbout << "*** GSN_SCAN_TABINFO ***" << endl;
+ if (tWaitState != WAIT_SCAN)
+ return;
+ tCon = void2con(tFirstDataPtr);
+ if (tCon->checkMagicNumber() != 0)
+ return;
+ tReturnCode = tCon->receiveSCAN_TABINFO(aSignal);
+ if (tReturnCode != -1)
+ theWaiter.m_state = NO_WAIT;
+ break;
+ }
+ case GSN_KEYINFO20: {
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ //ndbout << "*** GSN_KEYINFO20 ***" << endl;
+ NdbScanReceiver* tScanRec = void2rec_srec(tFirstDataPtr);
+ if (tScanRec->checkMagicNumber() != 0)
+ return;
+ tReturnCode = tScanRec->receiveKEYINFO20(aSignal);
+ if (tReturnCode != -1)
+ theWaiter.m_state = NO_WAIT;
+ break;
+ }
+ case GSN_TCINDXCONF:{
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ const TcIndxConf * const indxConf = (TcIndxConf *)tDataPtr;
+ const BlockReference aTCRef = aSignal->theSendersBlockRef;
+ tCon = void2con(tFirstDataPtr);
+ if ((tCon->checkMagicNumber() == 0) &&
+ (tCon->theSendStatus == sendTC_OP)) {
+ tReturnCode = tCon->receiveTCINDXCONF(indxConf, aSignal->getLength());
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ }//if
+ }//if
+
+ if(TcIndxConf::getMarkerFlag(indxConf->confInfo)){
+ NdbConnection::sendTC_COMMIT_ACK(theCommitAckSignal,
+ indxConf->transId1,
+ indxConf->transId2,
+ aTCRef);
+ }
+ break;
+ }
+ case GSN_TCINDXREF:{
+ void* tFirstDataPtr = int2void(tFirstData);
+ if (tFirstDataPtr == 0) goto InvalidSignal;
+
+ tIndexOp = void2rec_iop(tFirstDataPtr);
+ if (tIndexOp->checkMagicNumber() == 0) {
+ tCon = tIndexOp->theNdbCon;
+ if (tCon != NULL) {
+ if (tCon->theSendStatus == sendTC_OP) {
+ tReturnCode = tIndexOp->receiveTCINDXREF(aSignal);
+ if (tReturnCode != -1) {
+ completedTransaction(tCon);
+ }//if
+ return;
+ }//if
+ }//if
+ }//if
+ goto InvalidSignal;
+ return;
+ }
+ default:
+ goto InvalidSignal;
+ }//switch
+
+ if (theWaiter.m_state == NO_WAIT) {
+ // Wake up the thread waiting for response
+ NdbCondition_Signal(theWaiter.m_condition);
+ }//if
+ return;
+
+ InvalidSignal:
+#ifdef VM_TRACE
+ ndbout_c("Ndbif: Error Ndb::handleReceivedSignal "
+ "(GSN=%d, theWaiter.m_state=%d)"
+ " sender = (Block: %d Node: %d)",
+ tSignalNumber,
+ tWaitState,
+ refToBlock(aSignal->theSendersBlockRef),
+ refToNode(aSignal->theSendersBlockRef));
+#endif
+#ifdef NDB_NO_DROPPED_SIGNAL
+ abort();
+#endif
+
+ return;
+}//Ndb::handleReceivedSignal()
+
+
+/*****************************************************************************
+void completedTransaction(NdbConnection* aCon);
+
+Remark: One transaction has been completed.
+ Remove it from send array and put it into the completed
+ transaction array. Finally check if it is time to wake
+ up a poller.
+******************************************************************************/
+void
+Ndb::completedTransaction(NdbConnection* aCon)
+{
+ Uint32 tTransArrayIndex = aCon->theTransArrayIndex;
+ Uint32 tNoSentTransactions = theNoOfSentTransactions;
+ Uint32 tNoCompletedTransactions = theNoOfCompletedTransactions;
+ if ((tNoSentTransactions > 0) && (aCon->theListState == InSendList) &&
+ (tTransArrayIndex < tNoSentTransactions)) {
+ NdbConnection* tMoveCon = theSentTransactionsArray[tNoSentTransactions - 1];
+
+ theCompletedTransactionsArray[tNoCompletedTransactions] = aCon;
+ aCon->theTransArrayIndex = tNoCompletedTransactions;
+ if (tMoveCon != aCon) {
+ tMoveCon->theTransArrayIndex = tTransArrayIndex;
+ theSentTransactionsArray[tTransArrayIndex] = tMoveCon;
+ }//if
+ theSentTransactionsArray[tNoSentTransactions - 1] = NULL;
+ theNoOfCompletedTransactions = tNoCompletedTransactions + 1;
+
+ theNoOfSentTransactions = tNoSentTransactions - 1;
+ aCon->theListState = InCompletedList;
+ aCon->handleExecuteCompletion();
+ if ((theMinNoOfEventsToWakeUp != 0) &&
+ (theNoOfCompletedTransactions >= theMinNoOfEventsToWakeUp)) {
+ theMinNoOfEventsToWakeUp = 0;
+ NdbCondition_Signal(theWaiter.m_condition);
+ return;
+ }//if
+ } else {
+ ndbout << "theNoOfSentTransactions = " << theNoOfSentTransactions;
+ ndbout << " theListState = " << aCon->theListState;
+ ndbout << " theTransArrayIndex = " << aCon->theTransArrayIndex;
+ ndbout << endl << flush;
+#ifdef VM_TRACE
+ printState("completedTransaction abort");
+#endif
+ abort();
+ }//if
+}//Ndb::completedTransaction()
+
+/*****************************************************************************
+void reportCallback(NdbConnection** aCopyArray, Uint32 aNoOfCompletedTrans);
+
+Remark: Call the callback methods of the completed transactions.
+******************************************************************************/
+void
+Ndb::reportCallback(NdbConnection** aCopyArray, Uint32 aNoOfCompletedTrans)
+{
+ Uint32 i;
+ if (aNoOfCompletedTrans > 0) {
+ for (i = 0; i < aNoOfCompletedTrans; i++) {
+ void* anyObject = aCopyArray[i]->theCallbackObject;
+ NdbAsynchCallback aCallback = aCopyArray[i]->theCallbackFunction;
+ int tResult = 0;
+ if (aCallback != NULL) {
+ if (aCopyArray[i]->theReturnStatus == ReturnFailure) {
+ tResult = -1;
+ }//if
+ (*aCallback)(tResult, aCopyArray[i], anyObject);
+ }//if
+ }//for
+ }//if
+}//Ndb::reportCallback()
+
+/*****************************************************************************
+Uint32 pollCompleted(NdbConnection** aCopyArray);
+
+Remark: Transfer the data from the completed transaction to a local array.
+ This support is used by a number of the poll-methods.
+******************************************************************************/
+Uint32
+Ndb::pollCompleted(NdbConnection** aCopyArray)
+{
+ check_send_timeout();
+ Uint32 i;
+ Uint32 tNoCompletedTransactions = theNoOfCompletedTransactions;
+ if (tNoCompletedTransactions > 0) {
+ for (i = 0; i < tNoCompletedTransactions; i++) {
+ aCopyArray[i] = theCompletedTransactionsArray[i];
+ if (aCopyArray[i]->theListState != InCompletedList) {
+ ndbout << "pollCompleted error ";
+ ndbout << aCopyArray[i]->theListState << endl;
+ abort();
+ }//if
+ theCompletedTransactionsArray[i] = NULL;
+ aCopyArray[i]->theListState = NotInList;
+ }//for
+ }//if
+ theNoOfCompletedTransactions = 0;
+ return tNoCompletedTransactions;
+}//Ndb::pollCompleted()
+
+void
+Ndb::check_send_timeout()
+{
+ NDB_TICKS current_time = NdbTick_CurrentMillisecond();
+ if (current_time - the_last_check_time > 1000) {
+ the_last_check_time = current_time;
+ Uint32 no_of_sent = theNoOfSentTransactions;
+ for (Uint32 i = 0; i < no_of_sent; i++) {
+ NdbConnection* a_con = theSentTransactionsArray[i];
+ if ((current_time - a_con->theStartTransTime) >
+ WAITFOR_RESPONSE_TIMEOUT) {
+#ifdef VM_TRACE
+ a_con->printState();
+#endif
+ a_con->setOperationErrorCodeAbort(4012);
+ a_con->theCommitStatus = Aborted;
+ a_con->theCompletionStatus = CompletedFailure;
+ a_con->handleExecuteCompletion();
+ remove_sent_list(i);
+ insert_completed_list(a_con);
+ no_of_sent--;
+ i--;
+ }//if
+ }//for
+ }//if
+}
+
+void
+Ndb::remove_sent_list(Uint32 list_index)
+{
+ Uint32 last_index = theNoOfSentTransactions - 1;
+ if (list_index < last_index) {
+ NdbConnection* t_con = theSentTransactionsArray[last_index];
+ theSentTransactionsArray[list_index] = t_con;
+ }//if
+ theNoOfSentTransactions = last_index;
+ theSentTransactionsArray[last_index] = 0;
+}
+
+Uint32
+Ndb::insert_completed_list(NdbConnection* a_con)
+{
+ Uint32 no_of_comp = theNoOfCompletedTransactions;
+ theCompletedTransactionsArray[no_of_comp] = a_con;
+ theNoOfCompletedTransactions = no_of_comp + 1;
+ a_con->theListState = InCompletedList;
+ a_con->theTransArrayIndex = no_of_comp;
+ return no_of_comp;
+}
+
+Uint32
+Ndb::insert_sent_list(NdbConnection* a_con)
+{
+ Uint32 no_of_sent = theNoOfSentTransactions;
+ theSentTransactionsArray[no_of_sent] = a_con;
+ theNoOfSentTransactions = no_of_sent + 1;
+ a_con->theListState = InSendList;
+ a_con->theTransArrayIndex = no_of_sent;
+ return no_of_sent;
+}
+
+/*****************************************************************************
+void sendPrepTrans(int forceSend);
+
+Remark: Send a batch of transactions prepared for sending to the NDB kernel.
+******************************************************************************/
+void
+Ndb::sendPrepTrans(int forceSend)
+{
+ // Always called when holding mutex on TransporterFacade
+ /*
+ We will send a list of transactions to the NDB kernel. Before
+ sending we check the following.
+ 1) Node connected to is still alive
+ Checked by both checking node status and node sequence
+ 2) Send buffer can handle the size of messages we are planning to send
+ So far this is just a fake check but will soon be a real check
+ When the connected node has failed we abort the transaction without
+ responding anymore to the node since the kernel will clean up
+ automatically.
+ When sendBuffer cannot handle anymore messages then we will also abort
+ transaction but by communicating to the kernel since it is still alive
+ and we keep a small space for messages like that.
+ */
+ Uint32 i;
+ TransporterFacade* tp = TransporterFacade::instance();
+ Uint32 no_of_prep_trans = theNoOfPreparedTransactions;
+ for (i = 0; i < no_of_prep_trans; i++) {
+ NdbConnection * a_con = thePreparedTransactionsArray[i];
+ thePreparedTransactionsArray[i] = NULL;
+ Uint32 node_id = a_con->getConnectedNodeId();
+ if ((tp->getNodeSequence(node_id) == a_con->theNodeSequence) &&
+ tp->get_node_alive(node_id) ||
+ (tp->get_node_stopping(node_id) &&
+ ((a_con->theSendStatus == sendABORT) ||
+ (a_con->theSendStatus == sendABORTfail) ||
+ (a_con->theSendStatus == sendCOMMITstate) ||
+ (a_con->theSendStatus == sendCompleted)))) {
+ /*
+ We will send if
+ 1) Node is alive and sequences are correct OR
+ 2) Node is stopping and we only want to commit or abort
+ In a graceful stop situation we want to ensure quick aborts
+ of all transactions and commits and thus we allow aborts and
+ commits to continue but not normal operations.
+ */
+ if (tp->check_send_size(node_id, a_con->get_send_size())) {
+ if (a_con->doSend() == 0) {
+ NDB_TICKS current_time = NdbTick_CurrentMillisecond();
+ a_con->theStartTransTime = current_time;
+ continue;
+ } else {
+ /*
+ Although all precautions we did not manage to send the operations
+ Must have been a dropped connection on the transporter side.
+ We don't expect to be able to continue using this connection so
+ we will treat it as a node failure.
+ */
+ TRACE_DEBUG("Send problem even after checking node status");
+ }//if
+ } else {
+ /*
+ The send buffer is currently full or at least close to. We will
+ not allow a send to continue. We will set the connection so that
+ it is indicated that we need to abort the transaction. If we were
+ trying to commit or abort and got a send buffer we will not try
+ again and will thus set the state to Aborted to avoid a more or
+ less eternal loop of tries.
+ */
+ if (a_con->theSendStatus == sendOperations) {
+ a_con->setOperationErrorCodeAbort(4021);
+ a_con->theCommitStatus = NeedAbort;
+ TRACE_DEBUG("Send buffer full and sendOperations");
+ } else {
+ a_con->setOperationErrorCodeAbort(4026);
+ a_con->theCommitStatus = Aborted;
+ TRACE_DEBUG("Send buffer full, set state to Aborted");
+ }//if
+ }//if
+ } else {
+#ifdef VM_TRACE
+ a_con->printState();
+#endif
+ if ((tp->getNodeSequence(node_id) == a_con->theNodeSequence) &&
+ tp->get_node_stopping(node_id)) {
+ /*
+ The node we are connected to is currently in an early stopping phase
+ of a graceful stop. We will not send the prepared transactions. We
+ will simply refuse and let the application code handle the abort.
+ */
+ TRACE_DEBUG("Abort a transaction when stopping a node");
+ a_con->setOperationErrorCodeAbort(4023);
+ a_con->theCommitStatus = NeedAbort;
+ } else {
+ /*
+ The node is hard dead and we cannot continue. We will also release
+ the connection to the free pool.
+ */
+ TRACE_DEBUG("The node was stone dead, inform about abort");
+ a_con->setOperationErrorCodeAbort(4025);
+ a_con->theReleaseOnClose = true;
+ a_con->theTransactionIsStarted = false;
+ a_con->theCommitStatus = Aborted;
+ }//if
+ }//if
+ a_con->theCompletionStatus = CompletedFailure;
+ a_con->handleExecuteCompletion();
+ insert_completed_list(a_con);
+ }//for
+ theNoOfPreparedTransactions = 0;
+ if (forceSend == 0) {
+ tp->checkForceSend(theNdbBlockNumber);
+ } else if (forceSend == 1) {
+ tp->forceSend(theNdbBlockNumber);
+ }//if
+ return;
+}//Ndb::sendPrepTrans()
+
+/*****************************************************************************
+void waitCompletedTransactions(int aMilliSecondsToWait, int noOfEventsToWaitFor);
+
+Remark: First send all prepared operations and then check if there are any
+ transactions already completed. Do not wait for not completed
+ transactions.
+******************************************************************************/
+void
+Ndb::waitCompletedTransactions(int aMilliSecondsToWait,
+ int noOfEventsToWaitFor)
+{
+ theWaiter.m_state = NO_WAIT;
+ /**
+ * theWaiter.m_state = NO_WAIT;
+ * To ensure no messup with synchronous node fail handling
+ * (see ReportFailure)
+ */
+ int waitTime = aMilliSecondsToWait;
+ NDB_TICKS maxTime = NdbTick_CurrentMillisecond() + (NDB_TICKS)waitTime;
+ theMinNoOfEventsToWakeUp = noOfEventsToWaitFor;
+ do {
+ if (waitTime < 1000) waitTime = 1000;
+ NdbCondition_WaitTimeout(theWaiter.m_condition,
+ (NdbMutex*)theWaiter.m_mutex,
+ waitTime);
+ if (theNoOfCompletedTransactions >= (Uint32)noOfEventsToWaitFor) {
+ break;
+ }//if
+ theMinNoOfEventsToWakeUp = noOfEventsToWaitFor;
+ waitTime = (int)(maxTime - NdbTick_CurrentMillisecond());
+ } while (waitTime > 0);
+ return;
+}//Ndb::waitCompletedTransactions()
+
+/*****************************************************************************
+void sendPreparedTransactions(int forceSend = 0);
+
+Remark: First send all prepared operations and then check if there are any
+ transactions already completed. Do not wait for not completed
+ transactions.
+******************************************************************************/
+void
+Ndb::sendPreparedTransactions(int forceSend)
+{
+ TransporterFacade::instance()->lock_mutex();
+ sendPrepTrans(forceSend);
+ TransporterFacade::instance()->unlock_mutex();
+ return;
+}//Ndb::sendPreparedTransactions()
+
+/*****************************************************************************
+int sendPollNdb(int aMillisecondNumber, int minNoOfEventsToWakeup = 1, int forceSend = 0);
+
+Remark: First send all prepared operations and then check if there are any
+ transactions already completed. Wait for not completed
+ transactions until the specified number have completed or until the
+ timeout has occured. Timeout zero means no waiting time.
+******************************************************************************/
+int
+Ndb::sendPollNdb(int aMillisecondNumber, int minNoOfEventsToWakeup, int forceSend)
+{
+ NdbConnection* tConArray[1024];
+ Uint32 tNoCompletedTransactions;
+
+ //theCurrentConnectCounter = 0;
+ //theCurrentConnectIndex++;
+ TransporterFacade::instance()->lock_mutex();
+ sendPrepTrans(forceSend);
+ if ((minNoOfEventsToWakeup <= 0) ||
+ ((Uint32)minNoOfEventsToWakeup > theNoOfSentTransactions)) {
+ minNoOfEventsToWakeup = theNoOfSentTransactions;
+ }//if
+ if ((theNoOfCompletedTransactions < (Uint32)minNoOfEventsToWakeup) &&
+ (aMillisecondNumber > 0)) {
+ waitCompletedTransactions(aMillisecondNumber, minNoOfEventsToWakeup);
+ tNoCompletedTransactions = pollCompleted(tConArray);
+ } else {
+ tNoCompletedTransactions = pollCompleted(tConArray);
+ }//if
+ TransporterFacade::instance()->unlock_mutex();
+ reportCallback(tConArray, tNoCompletedTransactions);
+ return tNoCompletedTransactions;
+}//Ndb::sendPollNdb()
+
+/*****************************************************************************
+int pollNdb(int aMillisecondNumber, int minNoOfEventsToWakeup);
+
+Remark: Check if there are any transactions already completed. Wait for not
+ completed transactions until the specified number have completed or
+ until the timeout has occured. Timeout zero means no waiting time.
+******************************************************************************/
+int
+Ndb::pollNdb(int aMillisecondNumber, int minNoOfEventsToWakeup)
+{
+ NdbConnection* tConArray[1024];
+ Uint32 tNoCompletedTransactions;
+
+ //theCurrentConnectCounter = 0;
+ //theCurrentConnectIndex++;
+ TransporterFacade::instance()->lock_mutex();
+ if ((minNoOfEventsToWakeup == 0) ||
+ ((Uint32)minNoOfEventsToWakeup > theNoOfSentTransactions)) {
+ minNoOfEventsToWakeup = theNoOfSentTransactions;
+ }//if
+ if ((theNoOfCompletedTransactions < (Uint32)minNoOfEventsToWakeup) &&
+ (aMillisecondNumber > 0)) {
+ waitCompletedTransactions(aMillisecondNumber, minNoOfEventsToWakeup);
+ tNoCompletedTransactions = pollCompleted(tConArray);
+ } else {
+ tNoCompletedTransactions = pollCompleted(tConArray);
+ }//if
+ TransporterFacade::instance()->unlock_mutex();
+ reportCallback(tConArray, tNoCompletedTransactions);
+ return tNoCompletedTransactions;
+}//Ndb::sendPollNdbWithoutWait()
+
+/*****************************************************************************
+int receiveOptimisedResponse();
+
+Return: 0 - Response received
+ -1 - Timeout occured waiting for response
+ -2 - Node failure interupted wait for response
+
+******************************************************************************/
+int
+Ndb::receiveResponse(int waitTime)
+{
+ int tResultCode;
+ TransporterFacade::instance()->checkForceSend(theNdbBlockNumber);
+
+ theWaiter.wait(waitTime);
+
+ if(theWaiter.m_state == NO_WAIT) {
+ tResultCode = 0;
+ } else {
+
+#ifdef VM_TRACE
+ ndbout << "ERR: receiveResponse - theWaiter.m_state = ";
+ ndbout << theWaiter.m_state << endl;
+#endif
+
+ if (theWaiter.m_state == WAIT_NODE_FAILURE){
+ tResultCode = -2;
+ } else {
+ tResultCode = -1;
+ }
+ theWaiter.m_state = NO_WAIT;
+ }
+ return tResultCode;
+}//Ndb::receiveResponse()
+
+int
+Ndb::sendRecSignal(Uint16 node_id,
+ Uint32 aWaitState,
+ NdbApiSignal* aSignal,
+ Uint32 conn_seq)
+{
+ /*
+ In most situations 0 is returned.
+ In error cases we have 5 different cases
+ -1: Send ok, time out in waiting for reply
+ -2: Node has failed
+ -3: Send buffer not full, send failed yet
+ -4: Send buffer full
+ -5: Node is currently stopping
+ */
+
+ int return_code;
+ TransporterFacade* tp = TransporterFacade::instance();
+ Uint32 send_size = 1; // Always sends one signal only
+ tp->lock_mutex();
+ // Protected area
+ if ((tp->get_node_alive(node_id)) &&
+ ((tp->getNodeSequence(node_id) == conn_seq) ||
+ (conn_seq == 0))) {
+ if (tp->check_send_size(node_id, send_size)) {
+ return_code = tp->sendSignal(aSignal, node_id);
+ if (return_code != -1) {
+ theWaiter.m_node = node_id;
+ theWaiter.m_state = aWaitState;
+ return receiveResponse();
+ // End of protected area
+ }//if
+ return_code = -3;
+ } else {
+ return_code = -4;
+ }//if
+ } else {
+ if ((tp->get_node_stopping(node_id)) &&
+ ((tp->getNodeSequence(node_id) == conn_seq) ||
+ (conn_seq == 0))) {
+ return_code = -5;
+ } else {
+ return_code = -2;
+ }//if
+ }//if
+ tp->unlock_mutex();
+ // End of protected area
+ return return_code;
+}//Ndb::sendRecSignal()
+
+void
+NdbConnection::sendTC_COMMIT_ACK(NdbApiSignal * aSignal,
+ Uint32 transId1, Uint32 transId2,
+ Uint32 aTCRef){
+#if 0
+ ndbout_c("Sending TC_COMMIT_ACK(0x%x, 0x%x) to -> %d",
+ transId1,
+ transId2,
+ refToNode(aTCRef));
+#endif
+ TransporterFacade *tp = TransporterFacade::instance();
+ aSignal->theTrace = TestOrd::TraceAPI;
+ aSignal->theReceiversBlockNumber = DBTC;
+ aSignal->theVerId_signalNumber = GSN_TC_COMMIT_ACK;
+ aSignal->theLength = 2;
+
+ Uint32 * dataPtr = aSignal->getDataPtrSend();
+ dataPtr[0] = transId1;
+ dataPtr[1] = transId2;
+
+ tp->sendSignal(aSignal, refToNode(aTCRef));
+}
diff --git a/ndb/src/ndbapi/Ndbinit.cpp b/ndb/src/ndbapi/Ndbinit.cpp
new file mode 100644
index 00000000000..9afbbf0df1f
--- /dev/null
+++ b/ndb/src/ndbapi/Ndbinit.cpp
@@ -0,0 +1,285 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbApiSignal.hpp"
+#include "NdbImpl.hpp"
+#include "NdbSchemaOp.hpp"
+#include "NdbSchemaCon.hpp"
+#include "NdbOperation.hpp"
+#include "NdbConnection.hpp"
+#include "NdbRecAttr.hpp"
+#include "IPCConfig.hpp"
+#include "TransporterFacade.hpp"
+#include "ConfigRetriever.hpp"
+#include <ndb_limits.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include "ObjectMap.hpp"
+
+class NdbGlobalEventBufferHandle;
+NdbGlobalEventBufferHandle *NdbGlobalEventBuffer_init(int);
+void NdbGlobalEventBuffer_drop(NdbGlobalEventBufferHandle *);
+
+/**
+ * Static object for NDB
+ */
+static int theNoOfNdbObjects = 0;
+
+static char *ndbConnectString = 0;
+
+#ifdef NDB_WIN32
+static NdbMutex & createNdbMutex = * NdbMutex_Create();
+#else
+static NdbMutex createNdbMutex = NDB_MUTEX_INITIALIZER;
+#endif
+
+
+/***************************************************************************
+Ndb(const char* aDataBase);
+
+Parameters: aDataBase : Name of the database.
+Remark: Connect to the database.
+***************************************************************************/
+Ndb::Ndb( const char* aDataBase , const char* aDataBaseSchema) :
+ thePreparedTransactionsArray(NULL),
+ theSentTransactionsArray(NULL),
+ theCompletedTransactionsArray(NULL),
+ theNoOfPreparedTransactions(0),
+ theNoOfSentTransactions(0),
+ theNoOfCompletedTransactions(0),
+ theNoOfAllocatedTransactions(0),
+ theMaxNoOfTransactions(0),
+ theMinNoOfEventsToWakeUp(0),
+ prefixEnd(NULL),
+ theImpl(NULL),
+ theDictionary(NULL),
+ theConIdleList(NULL),
+ theOpIdleList(NULL),
+ theScanOpIdleList(NULL),
+ theIndexOpIdleList(NULL),
+ theSchemaConIdleList(NULL),
+ theSchemaConToNdbList(NULL),
+ theTransactionList(NULL),
+ theConnectionArray(NULL),
+ theRecAttrIdleList(NULL),
+ theSignalIdleList(NULL),
+ theLabelList(NULL),
+ theBranchList(NULL),
+ theSubroutineList(NULL),
+ theCallList(NULL),
+ theScanList(NULL),
+ theNoOfDBnodes(0),
+ theDBnodes(NULL),
+ the_release_ind(NULL),
+ the_last_check_time(0),
+ theFirstTransId(0),
+ theRestartGCI(0),
+ theNdbBlockNumber(-1),
+ theInitState(NotConstructed),
+ theNdbObjectIdMap(0)
+{
+ cgetSignals =0;
+ cfreeSignals = 0;
+ cnewSignals = 0;
+ creleaseSignals = 0;
+ theError.code = 0;
+
+ theNdbObjectIdMap = new NdbObjectIdMap(1024,1024);
+ theConnectionArray = new NdbConnection * [MAX_NDB_NODES];
+ theDBnodes = new Uint32[MAX_NDB_NODES];
+ the_release_ind = new Uint8[MAX_NDB_NODES];
+ theCommitAckSignal = NULL;
+
+ theCurrentConnectCounter = 1;
+ theCurrentConnectIndex = 0;
+ for (int i = 0; i < MAX_NDB_NODES ; i++) {
+ theConnectionArray[i] = NULL;
+ the_release_ind[i] = 0;
+ theDBnodes[i] = 0;
+ }//forg
+ for (int i = 0; i < 2048 ; i++) {
+ theFirstTupleId[i] = 0;
+ theLastTupleId[i] = 0;
+ }//for
+
+ if (aDataBase)
+ strncpy(theDataBase, aDataBase, NDB_MAX_DATABASE_NAME_SIZE);
+ else
+ memset(theDataBase, 0, sizeof(theDataBase));
+ strncpy(theDataBaseSchema, aDataBaseSchema, NDB_MAX_SCHEMA_NAME_SIZE);
+ // Prepare prefix for faster operations
+ uint db_len = MIN(strlen(theDataBase), NDB_MAX_DATABASE_NAME_SIZE - 1);
+ uint schema_len =
+ MIN(strlen(theDataBaseSchema), NDB_MAX_SCHEMA_NAME_SIZE - 1);
+ strncpy(prefixName, theDataBase, NDB_MAX_DATABASE_NAME_SIZE - 1);
+ prefixName[db_len] = '/';
+ strncpy(prefixName+db_len+1, theDataBaseSchema,
+ NDB_MAX_SCHEMA_NAME_SIZE - 1);
+ prefixName[db_len+schema_len+1] = '/';
+ prefixName[db_len+schema_len+2] = '\0';
+ prefixEnd = prefixName + db_len+schema_len + 2;
+
+ NdbMutex_Lock(&createNdbMutex);
+
+ TransporterFacade * m_facade = 0;
+ if(theNoOfNdbObjects == 0){
+ if ((m_facade = TransporterFacade::start_instance(0,ndbConnectString)) == 0)
+ theInitState = InitConfigError;
+ } else {
+ m_facade = TransporterFacade::instance();
+ }
+
+ if(m_facade != 0){
+ theWaiter.m_mutex = m_facade->theMutexPtr;
+ }
+
+ // For keeping track of how many Ndb objects that exists.
+ theNoOfNdbObjects += 1;
+
+ // Signal that the constructor has finished OK
+ if (theInitState == NotConstructed)
+ theInitState = NotInitialised;
+
+ theImpl = new NdbImpl();
+
+ {
+ NdbGlobalEventBufferHandle *h=
+ NdbGlobalEventBuffer_init(NDB_MAX_ACTIVE_EVENTS);
+ if (h == NULL) {
+ ndbout_c("Failed NdbGlobalEventBuffer_init(%d)",NDB_MAX_ACTIVE_EVENTS);
+ exit(-1);
+ }
+ theGlobalEventBufferHandle = h;
+ }
+
+ NdbMutex_Unlock(&createNdbMutex);
+}
+
+
+void Ndb::setConnectString(const char * connectString)
+{
+ if (ndbConnectString != 0) {
+ free(ndbConnectString);
+ ndbConnectString = 0;
+ }
+ if (connectString)
+ ndbConnectString = strdup(connectString);
+}
+
+/*****************************************************************************
+ * ~Ndb();
+ *
+ * Remark: Disconnect with the database.
+ *****************************************************************************/
+Ndb::~Ndb()
+{
+ doDisconnect();
+
+ delete theDictionary;
+ delete theImpl;
+
+ NdbGlobalEventBuffer_drop(theGlobalEventBufferHandle);
+
+ if (TransporterFacade::instance() != NULL && theNdbBlockNumber > 0){
+ TransporterFacade::instance()->close(theNdbBlockNumber);
+ }
+
+ NdbMutex_Lock(&createNdbMutex);
+
+ theNoOfNdbObjects -= 1;
+ if(theNoOfNdbObjects == 0){
+ TransporterFacade::stop_instance();
+ }//if
+
+ NdbMutex_Unlock(&createNdbMutex);
+
+ if (theSchemaConToNdbList != NULL)
+ closeSchemaTransaction(theSchemaConToNdbList);
+ while ( theConIdleList != NULL )
+ freeNdbCon();
+ while ( theSignalIdleList != NULL )
+ freeSignal();
+ while (theRecAttrIdleList != NULL)
+ freeRecAttr();
+ while (theOpIdleList != NULL)
+ freeOperation();
+ while (theScanOpIdleList != NULL)
+ freeScanOperation();
+ while (theIndexOpIdleList != NULL)
+ freeIndexOperation();
+ while (theLabelList != NULL)
+ freeNdbLabel();
+ while (theBranchList != NULL)
+ freeNdbBranch();
+ while (theSubroutineList != NULL)
+ freeNdbSubroutine();
+ while (theCallList != NULL)
+ freeNdbCall();
+ while (theScanList != NULL)
+ freeNdbScanRec();
+
+ releaseTransactionArrays();
+ startTransactionNodeSelectionData.release();
+
+ delete []theConnectionArray;
+ delete []theDBnodes;
+ delete []the_release_ind;
+ if(theCommitAckSignal != NULL){
+ delete theCommitAckSignal;
+ theCommitAckSignal = NULL;
+ }
+
+ if(theNdbObjectIdMap != 0)
+ delete theNdbObjectIdMap;
+
+ /**
+ * This sleep is to make sure that the transporter
+ * send thread will come in and send any
+ * signal buffers that this thread may have allocated.
+ * If that doesn't happen an error will occur in OSE
+ * when trying to restore a signal buffer allocated by a thread
+ * that have been killed.
+ */
+#ifdef NDB_OSE
+ NdbSleep_MilliSleep(50);
+#endif
+
+#ifdef POORMANSPURIFY
+#ifdef POORMANSGUI
+ ndbout << "cnewSignals=" << cnewSignals << endl;
+ ndbout << "cfreeSignals=" << cfreeSignals << endl;
+ ndbout << "cgetSignals=" << cgetSignals << endl;
+ ndbout << "creleaseSignals=" << creleaseSignals << endl;
+#endif
+ // Poor mans purifier
+ assert(cnewSignals == cfreeSignals);
+ assert(cgetSignals == creleaseSignals);
+#endif
+}
+
+NdbWaiter::NdbWaiter(){
+ m_node = 0;
+ m_state = NO_WAIT;
+ m_mutex = 0;
+ m_condition = NdbCondition_Create();
+}
+
+NdbWaiter::~NdbWaiter(){
+ NdbCondition_Destroy(m_condition);
+}
+
+
diff --git a/ndb/src/ndbapi/Ndblist.cpp b/ndb/src/ndbapi/Ndblist.cpp
new file mode 100644
index 00000000000..3839cc3291b
--- /dev/null
+++ b/ndb/src/ndbapi/Ndblist.cpp
@@ -0,0 +1,798 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include "Ndb.hpp"
+#include "NdbSchemaOp.hpp"
+#include "NdbSchemaCon.hpp"
+#include "NdbOperation.hpp"
+#include "NdbScanOperation.hpp"
+#include "NdbIndexOperation.hpp"
+#include "NdbConnection.hpp"
+#include "NdbApiSignal.hpp"
+#include "NdbRecAttr.hpp"
+#include "NdbScanReceiver.hpp"
+#include "NdbUtil.hpp"
+#include "API.hpp"
+
+void
+Ndb::checkFailedNode()
+{
+ for (NodeId i = 0; i < theNoOfDBnodes; i ++){
+ const NodeId node_id = theDBnodes[i];
+
+ NdbConnection * tNdbCon = theConnectionArray[node_id];
+ if (the_release_ind[node_id] == 1){
+
+ /**
+ * Release all connections in idle list (for node)
+ */
+ theConnectionArray[node_id] = NULL;
+ while (tNdbCon != NULL) {
+ NdbConnection* tempNdbCon = tNdbCon;
+ tNdbCon = tNdbCon->next();
+ releaseNdbCon(tempNdbCon);
+ }//while
+ the_release_ind[node_id] = 0;
+ }//if
+ }//for
+}
+
+#if 0
+void
+NdbImpl::checkInvalidTable(NdbDictionaryImpl * dict){
+ Uint32 sz = m_invalidTables.size();
+ for(Int32 i = sz - 1; i >= 0; i--){
+ NdbTableImpl * tab = m_invalidTables[i];
+ m_invalidTables.erase(i);
+ dict->tableDropped(* tab);
+ }
+}
+
+void
+NdbImpl::checkErrorCode(Uint32 i, NdbTableImpl * tab){
+ switch(i){
+ case 241:
+ case 283:
+ case 284:
+ case 285:
+ case 1225:
+ case 1226:
+
+ }
+}
+#endif
+
+/***************************************************************************
+ * int createConIdleList(int aNrOfCon);
+ *
+ * Return Value: Return the number of created connection object
+ * if createConIdleList was succesful
+ * Return -1: In all other case.
+ * Parameters: aNrOfCon : Number of connections offered to the application.
+ * Remark: Create connection idlelist with NdbConnection objects.
+ ***************************************************************************/
+int
+Ndb::createConIdleList(int aNrOfCon)
+{
+ for (int i = 0; i < aNrOfCon; i++)
+ {
+ NdbConnection* tNdbCon = new NdbConnection(this);
+ if (tNdbCon == NULL)
+ {
+ return -1;
+ }
+ if (theConIdleList == NULL)
+ {
+ theConIdleList = tNdbCon;
+ theConIdleList->next(NULL);
+ } else
+ {
+ tNdbCon->next(theConIdleList);
+ theConIdleList = tNdbCon;
+ }
+ tNdbCon->Status(NotConnected);
+ }
+ theNoOfAllocatedTransactions = aNrOfCon;
+ return aNrOfCon;
+}
+
+/***************************************************************************
+ * int createOpIdleList(int aNrOfOp);
+ *
+ * Return Value: Return the number of created operation object if
+ * createOpIdleList was succesful.
+ * Return -1: In all other case.
+ * Parameters: aNrOfOp: Number of operations offered to the application.
+ * Remark: Create operation idlelist with NdbOperation objects..
+ ***************************************************************************/
+int
+Ndb::createOpIdleList(int aNrOfOp)
+{
+ for (int i = 0; i < aNrOfOp; i++){
+ NdbOperation* tOp = new NdbOperation(this);
+ if ( tOp == NULL ){
+ return -1;
+ }
+ if (theOpIdleList == NULL){
+ theOpIdleList = tOp;
+ theOpIdleList->next(NULL);
+ } else{
+ tOp->next(theOpIdleList);
+ theOpIdleList = tOp;
+ }
+ }
+ return aNrOfOp;
+}
+
+/***************************************************************************
+ * NdbBranch* NdbBranch();
+ *
+ * Return Value: Return a NdbBranch if the getNdbBranch was successful.
+ * Return NULL : In all other case.
+ * Remark: Get a NdbBranch from theBranchList and return the object .
+ ***************************************************************************/
+NdbBranch*
+Ndb::getNdbBranch()
+{
+ NdbBranch* tNdbBranch;
+ if ( theBranchList == NULL )
+ {
+ tNdbBranch = new NdbBranch;
+ if (tNdbBranch == NULL)
+ {
+ return NULL;
+ }
+ tNdbBranch->theNext = NULL;
+ } else
+ {
+ tNdbBranch = theBranchList;
+ theBranchList = tNdbBranch->theNext;
+ tNdbBranch->theNext = NULL;
+ }
+ return tNdbBranch;
+}
+
+/***************************************************************************
+ * NdbCall* NdbCall();
+ *
+ * Return Value: Return a NdbCall if the getNdbCall was successful.
+ * Return NULL : In all other case.
+ * Remark: Get a NdbCall from theCallList and return the object .
+ ***************************************************************************/
+NdbCall*
+Ndb::getNdbCall()
+{
+ NdbCall* tNdbCall;
+ if ( theCallList == NULL )
+ {
+ tNdbCall = new NdbCall;
+ if (tNdbCall == NULL)
+ {
+ return NULL;
+ }
+ tNdbCall->theNext = NULL;
+ } else
+ {
+ tNdbCall = theCallList;
+ theCallList = tNdbCall->theNext;
+ tNdbCall->theNext = NULL;
+ }
+ return tNdbCall;
+}
+
+/***************************************************************************
+ * NdbConnection* getNdbCon();
+ *
+ * Return Value: Return a connection if the getNdbCon was successful.
+ * Return NULL : In all other case.
+ * Remark: Get a connection from theConIdleList and return the object .
+ ***************************************************************************/
+NdbConnection*
+Ndb::getNdbCon()
+{
+ NdbConnection* tNdbCon;
+ if ( theConIdleList == NULL ) {
+ if (theNoOfAllocatedTransactions < theMaxNoOfTransactions) {
+ tNdbCon = new NdbConnection(this);
+ if (tNdbCon == NULL) {
+ return NULL;
+ }//if
+ theNoOfAllocatedTransactions++;
+ } else {
+ ndbout << "theNoOfAllocatedTransactions = " << theNoOfAllocatedTransactions << " theMaxNoOfTransactions = " << theMaxNoOfTransactions << endl;
+ return NULL;
+ }//if
+ tNdbCon->next(NULL);
+ } else
+ {
+ tNdbCon = theConIdleList;
+ theConIdleList = tNdbCon->next();
+ tNdbCon->next(NULL);
+ }
+ tNdbCon->theMagicNumber = 0x37412619;
+ return tNdbCon;
+}
+
+/***************************************************************************
+ * NdbLabel* getNdbLabel();
+ *
+ * Return Value: Return a NdbLabel if the getNdbLabel was successful.
+ * Return NULL : In all other case.
+ * Remark: Get a NdbLabel from theLabelList and return the object .
+ ***************************************************************************/
+NdbLabel*
+Ndb::getNdbLabel()
+{
+ NdbLabel* tNdbLabel;
+ if ( theLabelList == NULL )
+ {
+ tNdbLabel = new NdbLabel;
+ if (tNdbLabel == NULL)
+ {
+ return NULL;
+ }
+ tNdbLabel->theNext = NULL;
+ } else
+ {
+ tNdbLabel = theLabelList;
+ theLabelList = tNdbLabel->theNext;
+ tNdbLabel->theNext = NULL;
+ }
+ return tNdbLabel;
+}
+
+/***************************************************************************
+ * NdbScanReceiver* getNdbScanRec()
+ *
+ * Return Value: Return a NdbScanReceiver
+ * Return NULL : In all other case.
+ * Remark: Get a NdbScanReceiver from theScanRecList and return the
+ * object .
+ ****************************************************************************/
+NdbScanReceiver*
+Ndb::getNdbScanRec()
+{
+ NdbScanReceiver* tNdbScanRec;
+ if ( theScanList == NULL )
+ {
+ tNdbScanRec = new NdbScanReceiver(this);
+ if (tNdbScanRec == NULL)
+ {
+ return NULL;
+ }
+ tNdbScanRec->next(NULL);
+ } else
+ {
+ tNdbScanRec = theScanList;
+ theScanList = tNdbScanRec->next();
+ tNdbScanRec->next(NULL);
+ }
+
+ return tNdbScanRec;
+}
+
+/***************************************************************************
+ * NdbSubroutine* getNdbSubroutine();
+ *
+ * Return Value: Return a NdbSubroutine if the getNdbSubroutine was successful.
+ * Return NULL : In all other case.
+ * Remark: Get a NdbSubroutine from theSubroutineList and return the object .
+ ***************************************************************************/
+NdbSubroutine*
+Ndb::getNdbSubroutine()
+{
+ NdbSubroutine* tNdbSubroutine;
+ if ( theSubroutineList == NULL )
+ {
+ tNdbSubroutine = new NdbSubroutine;
+ if (tNdbSubroutine == NULL)
+ {
+ return NULL;
+ }
+ tNdbSubroutine->theNext = NULL;
+ } else
+ {
+ tNdbSubroutine = theSubroutineList;
+ theSubroutineList = tNdbSubroutine->theNext;
+ tNdbSubroutine->theNext = NULL;
+ }
+ return tNdbSubroutine;
+}
+
+/***************************************************************************
+NdbOperation* getOperation();
+
+Return Value: Return theOpList : if the getOperation was succesful.
+ Return NULL : In all other case.
+Remark: Get an operation from theOpIdleList and return the object .
+***************************************************************************/
+NdbOperation*
+Ndb::getOperation()
+{
+ NdbOperation* tOp = theOpIdleList;
+ if (tOp != NULL ) {
+ NdbOperation* tOpNext = tOp->next();
+ tOp->next(NULL);
+ theOpIdleList = tOpNext;
+ return tOp;
+ } else {
+ tOp = new NdbOperation(this);
+ if (tOp != NULL)
+ tOp->next(NULL);
+ }
+ return tOp;
+}
+
+/***************************************************************************
+NdbScanOperation* getScanOperation();
+
+Return Value: Return theOpList : if the getScanOperation was succesful.
+ Return NULL : In all other case.
+Remark: Get an operation from theScanOpIdleList and return the object .
+***************************************************************************/
+NdbScanOperation*
+Ndb::getScanOperation()
+{
+ NdbScanOperation* tOp = theScanOpIdleList;
+ if (tOp != NULL ) {
+ NdbScanOperation* tOpNext = (NdbScanOperation*) tOp->next();
+ tOp->next(NULL);
+ theScanOpIdleList = tOpNext;
+ return tOp;
+ } else {
+ tOp = new NdbScanOperation(this);
+ if (tOp != NULL)
+ tOp->next(NULL);
+ }
+ return tOp;
+}
+
+/***************************************************************************
+NdbIndexOperation* getIndexOperation();
+
+Return Value: Return theOpList : if the getIndexOperation was succesful.
+ Return NULL : In all other case.
+Remark: Get an operation from theIndexOpIdleList and return the object .
+***************************************************************************/
+NdbIndexOperation*
+Ndb::getIndexOperation()
+{
+ NdbIndexOperation* tOp = theIndexOpIdleList;
+ if (tOp != NULL ) {
+ NdbIndexOperation* tOpNext = (NdbIndexOperation*) tOp->next();
+ tOp->next(NULL);
+ theIndexOpIdleList = tOpNext;
+ return tOp;
+ } else {
+ tOp = new NdbIndexOperation(this);
+ if (tOp != NULL)
+ tOp->next(NULL);
+ }
+ return tOp;
+}
+
+/***************************************************************************
+NdbRecAttr* getRecAttr();
+
+Return Value: Return a reference to a receive attribute object.
+ Return NULL if it's not possible to get a receive attribute object.
+***************************************************************************/
+NdbRecAttr*
+Ndb::getRecAttr()
+{
+ NdbRecAttr* tRecAttr;
+ tRecAttr = theRecAttrIdleList;
+ if (tRecAttr != NULL) {
+ NdbRecAttr* tRecAttrNext = tRecAttr->next();
+ tRecAttr->init();
+ theRecAttrIdleList = tRecAttrNext;
+ return tRecAttr;
+ } else {
+ tRecAttr = new NdbRecAttr;
+ if (tRecAttr == NULL)
+ return NULL;
+ tRecAttr->next(NULL);
+ }//if
+ tRecAttr->init();
+ return tRecAttr;
+}
+
+/***************************************************************************
+NdbApiSignal* getSignal();
+
+Return Value: Return a reference to a signal object.
+ Return NULL if not possible to get a signal object.
+***************************************************************************/
+NdbApiSignal*
+Ndb::getSignal()
+{
+ NdbApiSignal* tSignal = theSignalIdleList;
+ if (tSignal != NULL){
+ NdbApiSignal* tSignalNext = tSignal->next();
+ tSignal->next(NULL);
+ theSignalIdleList = tSignalNext;
+ } else {
+ tSignal = new NdbApiSignal(theMyRef);
+ cnewSignals++;
+ if (tSignal != NULL)
+ tSignal->next(NULL);
+ }
+ cgetSignals++;
+ return tSignal;
+}
+
+/***************************************************************************
+void releaseNdbBranch(NdbBranch* aNdbBranch);
+
+Parameters: NdbBranch: The NdbBranch object.
+Remark: Add a NdbBranch object into the Branch idlelist.
+***************************************************************************/
+void
+Ndb::releaseNdbBranch(NdbBranch* aNdbBranch)
+{
+ aNdbBranch->theNext = theBranchList;
+ theBranchList = aNdbBranch;
+}
+
+/***************************************************************************
+void releaseNdbCall(NdbCall* aNdbCall);
+
+Parameters: NdbBranch: The NdbBranch object.
+Remark: Add a NdbBranch object into the Branch idlelist.
+***************************************************************************/
+void
+Ndb::releaseNdbCall(NdbCall* aNdbCall)
+{
+ aNdbCall->theNext = theCallList;
+ theCallList = aNdbCall;
+}
+
+/***************************************************************************
+void releaseNdbCon(NdbConnection* aNdbCon);
+
+Parameters: aNdbCon: The NdbConnection object.
+Remark: Add a Connection object into the signal idlelist.
+***************************************************************************/
+void
+Ndb::releaseNdbCon(NdbConnection* aNdbCon)
+{
+ aNdbCon->next(theConIdleList);
+ aNdbCon->theMagicNumber = 0xFE11DD;
+ theConIdleList = aNdbCon;
+}
+
+/***************************************************************************
+void releaseNdbLabel(NdbLabel* aNdbLabel);
+
+Parameters: NdbLabel: The NdbLabel object.
+Remark: Add a NdbLabel object into the Label idlelist.
+***************************************************************************/
+void
+Ndb::releaseNdbLabel(NdbLabel* aNdbLabel)
+{
+ aNdbLabel->theNext = theLabelList;
+ theLabelList = aNdbLabel;
+}
+
+/***************************************************************************
+void releaseNdbScanRec(NdbScanReceiver* aNdbScanRec);
+
+Parameters: aNdbScanRec: The NdbScanReceiver object.
+Remark: Add a NdbScanReceiver object into the Scan idlelist.
+***************************************************************************/
+void
+Ndb::releaseNdbScanRec(NdbScanReceiver* aNdbScanRec)
+{
+ aNdbScanRec->next(theScanList);
+ theScanList = aNdbScanRec;
+}
+
+/***************************************************************************
+void releaseNdbSubroutine(NdbSubroutine* aNdbSubroutine);
+
+Parameters: NdbSubroutine: The NdbSubroutine object.
+Remark: Add a NdbSubroutine object into theSubroutine idlelist.
+***************************************************************************/
+void
+Ndb::releaseNdbSubroutine(NdbSubroutine* aNdbSubroutine)
+{
+ aNdbSubroutine->theNext = theSubroutineList;
+ theSubroutineList = aNdbSubroutine;
+}
+
+/***************************************************************************
+void releaseOperation(NdbOperation* anOperation);
+
+Parameters: anOperation : The released NdbOperation object.
+Remark: Add a NdbOperation object into the signal idlelist.
+***************************************************************************/
+void
+Ndb::releaseOperation(NdbOperation* anOperation)
+{
+ if(anOperation->m_tcReqGSN == GSN_TCKEYREQ){
+ anOperation->next(theOpIdleList);
+ anOperation->theNdbCon = NULL;
+ anOperation->theMagicNumber = 0xFE11D0;
+ theOpIdleList = anOperation;
+ } else {
+ assert(anOperation->m_tcReqGSN == GSN_TCINDXREQ);
+ anOperation->next(theIndexOpIdleList);
+ anOperation->theNdbCon = NULL;
+ anOperation->theMagicNumber = 0xFE11D1;
+ theIndexOpIdleList = (NdbIndexOperation*)anOperation;
+ }
+}
+
+/***************************************************************************
+void releaseScanOperation(NdbScanOperation* aScanOperation);
+
+Parameters: aScanOperation : The released NdbScanOperation object.
+Remark: Add a NdbScanOperation object into the signal idlelist.
+***************************************************************************/
+void
+Ndb::releaseScanOperation(NdbScanOperation* aScanOperation)
+{
+ aScanOperation->next(theScanOpIdleList);
+ aScanOperation->theNdbCon = NULL;
+ aScanOperation->theMagicNumber = 0xFE11D2;
+ theScanOpIdleList = (NdbScanOperation*)aScanOperation;
+}
+
+/***************************************************************************
+void releaseRecAttr(NdbRecAttr* aRecAttr);
+
+Parameters: aRecAttr : The released NdbRecAttr object.
+Remark: Add a NdbRecAttr object into the RecAtt idlelist.
+***************************************************************************/
+void
+Ndb::releaseRecAttr(NdbRecAttr* aRecAttr)
+{
+ aRecAttr->release();
+ aRecAttr->next(theRecAttrIdleList);
+ theRecAttrIdleList = aRecAttr;
+}
+
+/***************************************************************************
+void releaseSignal(NdbApiSignal* aSignal);
+
+Parameters: aSignal : The released NdbApiSignal object.
+Remark: Add a NdbApiSignal object into the signal idlelist.
+***************************************************************************/
+void
+Ndb::releaseSignal(NdbApiSignal* aSignal)
+{
+#if defined VM_TRACE
+ // Check that signal is not null
+ assert(aSignal != NULL);
+
+ // Check that signal is not already in list
+ NdbApiSignal* tmp = theSignalIdleList;
+ while (tmp != NULL){
+ assert(tmp != aSignal);
+ tmp = tmp->next();
+ }
+#endif
+ creleaseSignals++;
+ aSignal->next(theSignalIdleList);
+ theSignalIdleList = aSignal;
+}
+
+void
+Ndb::releaseSignalsInList(NdbApiSignal** pList){
+ NdbApiSignal* tmp;
+ while (*pList != NULL){
+ tmp = *pList;
+ *pList = (*pList)->next();
+ releaseSignal(tmp);
+ }
+}
+
+/***************************************************************************
+void freeOperation();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeOperation()
+{
+ NdbOperation* tOp = theOpIdleList;
+ theOpIdleList = theOpIdleList->next();
+ delete tOp;
+}
+
+/***************************************************************************
+void freeScanOperation();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeScanOperation()
+{
+ NdbScanOperation* tOp = theScanOpIdleList;
+ theScanOpIdleList = (NdbScanOperation *) theScanOpIdleList->next();
+ delete tOp;
+}
+
+/***************************************************************************
+void freeIndexOperation();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeIndexOperation()
+{
+ NdbIndexOperation* tOp = theIndexOpIdleList;
+ theIndexOpIdleList = (NdbIndexOperation *) theIndexOpIdleList->next();
+ delete tOp;
+}
+
+/***************************************************************************
+void freeNdbBranch();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeNdbBranch()
+{
+ NdbBranch* tNdbBranch = theBranchList;
+ theBranchList = theBranchList->theNext;
+ delete tNdbBranch;
+}
+
+/***************************************************************************
+void freeNdbCall();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeNdbCall()
+{
+ NdbCall* tNdbCall = theCallList;
+ theCallList = theCallList->theNext;
+ delete tNdbCall;
+}
+
+/***************************************************************************
+void freeNdbScanRec();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeNdbScanRec()
+{
+ NdbScanReceiver* tNdbScanRec = theScanList;
+ theScanList = theScanList->next();
+ delete tNdbScanRec;
+}
+
+/***************************************************************************
+void freeNdbCon();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeNdbCon()
+{
+ NdbConnection* tNdbCon = theConIdleList;
+ theConIdleList = theConIdleList->next();
+ delete tNdbCon;
+}
+
+/***************************************************************************
+void freeNdbLabel();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeNdbLabel()
+{
+ NdbLabel* tNdbLabel = theLabelList;
+ theLabelList = theLabelList->theNext;
+ delete tNdbLabel;
+}
+
+/***************************************************************************
+void freeNdbSubroutine();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeNdbSubroutine()
+{
+ NdbSubroutine* tNdbSubroutine = theSubroutineList;
+ theSubroutineList = theSubroutineList->theNext;
+ delete tNdbSubroutine;
+}
+
+/***************************************************************************
+void freeRecAttr();
+
+Remark: Always release the first item in the free list
+***************************************************************************/
+void
+Ndb::freeRecAttr()
+{
+ NdbRecAttr* tRecAttr = theRecAttrIdleList;
+ theRecAttrIdleList = theRecAttrIdleList->next();
+ delete tRecAttr;
+}
+
+/***************************************************************************
+void freeSignal();
+
+Remark: Delete a signal object from the signal idlelist.
+***************************************************************************/
+void
+Ndb::freeSignal()
+{
+ NdbApiSignal* tSignal = theSignalIdleList;
+ theSignalIdleList = tSignal->next();
+ delete tSignal;
+ cfreeSignals++;
+}
+
+/****************************************************************************
+int releaseConnectToNdb(NdbConnection* aConnectConnection);
+
+Return Value: -1 if error
+Parameters: aConnectConnection : Seized schema connection to DBTC
+Remark: Release and disconnect from DBTC a connection and seize it to theConIdleList.
+*****************************************************************************/
+void
+Ndb::releaseConnectToNdb(NdbConnection* a_con)
+{
+ NdbApiSignal tSignal(theMyRef);
+ int tConPtr;
+
+// I need to close the connection irrespective of whether I
+// manage to reach NDB or not.
+
+ if (a_con == NULL)
+ return;
+
+ Uint32 node_id = a_con->getConnectedNodeId();
+ Uint32 conn_seq = a_con->theNodeSequence;
+ tSignal.setSignal(GSN_TCRELEASEREQ);
+ tSignal.setData((tConPtr = a_con->getTC_ConnectPtr()), 1);
+ tSignal.setData(theMyRef, 2);
+ tSignal.setData(a_con->ptr2int(), 3);
+ a_con->Status(DisConnecting);
+ a_con->theMagicNumber = 0x37412619;
+ int ret_code = sendRecSignal(node_id,
+ WAIT_TC_RELEASE,
+ &tSignal,
+ conn_seq);
+ if (ret_code == 0) {
+ ;
+ } else if (ret_code == -1) {
+ TRACE_DEBUG("Time-out when TCRELEASE sent");
+ } else if (ret_code == -2) {
+ TRACE_DEBUG("Node failed when TCRELEASE sent");
+ } else if (ret_code == -3) {
+ TRACE_DEBUG("Send failed when TCRELEASE sent");
+ } else if (ret_code == -4) {
+ TRACE_DEBUG("Send buffer full when TCRELEASE sent");
+ } else if (ret_code == -5) {
+ TRACE_DEBUG("Node stopping when TCRELEASE sent");
+ } else {
+ ndbout << "Impossible return from sendRecSignal when TCRELEASE" << endl;
+ abort();
+ }//if
+ releaseNdbCon(a_con);
+ return;
+}
+
diff --git a/ndb/src/ndbapi/ObjectMap.hpp b/ndb/src/ndbapi/ObjectMap.hpp
new file mode 100644
index 00000000000..a2a8d00b48f
--- /dev/null
+++ b/ndb/src/ndbapi/ObjectMap.hpp
@@ -0,0 +1,145 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_OBJECT_ID_MAP_HPP
+#define NDB_OBJECT_ID_MAP_HPP
+
+#include <ndb_types.h>
+//#include <NdbMutex.h>
+#include <stdlib.h>
+#include <string.h>
+#include <NdbOut.hpp>
+
+//#define DEBUG_OBJECTMAP
+
+/**
+ * Global ObjectMap
+ */
+class NdbObjectIdMap //: NdbLockable
+{
+public:
+ static const Uint32 InvalidId = ~0;
+ NdbObjectIdMap(Uint32 initalSize = 128, Uint32 expandSize = 10);
+ ~NdbObjectIdMap();
+
+ Uint32 map(void * object);
+ void * unmap(Uint32 id, void *object);
+
+ void * getObject(Uint32 id);
+private:
+ Uint32 m_size;
+ Uint32 m_expandSize;
+ Uint32 m_firstFree;
+ union MapEntry {
+ Uint32 m_next;
+ void * m_obj;
+ } * m_map;
+
+ void expand(Uint32 newSize);
+};
+
+inline
+NdbObjectIdMap::NdbObjectIdMap(Uint32 sz, Uint32 eSz) {
+ m_size = 0;
+ m_firstFree = InvalidId;
+ m_map = 0;
+ m_expandSize = eSz;
+ expand(sz);
+#ifdef DEBUG_OBJECTMAP
+ ndbout_c("NdbObjectIdMap:::NdbObjectIdMap(%u)", sz);
+#endif
+}
+
+inline
+NdbObjectIdMap::~NdbObjectIdMap(){
+ free(m_map);
+}
+
+inline
+Uint32
+NdbObjectIdMap::map(void * object){
+
+ // lock();
+
+ if(m_firstFree == InvalidId){
+ expand(m_expandSize);
+ }
+
+ Uint32 ff = m_firstFree;
+ m_firstFree = m_map[ff].m_next;
+ m_map[ff].m_obj = object;
+
+ // unlock();
+
+#ifdef DEBUG_OBJECTMAP
+ ndbout_c("NdbObjectIdMap::map(0x%x) %u", object, ff<<2);
+#endif
+
+ return ff<<2;
+}
+
+inline
+void *
+NdbObjectIdMap::unmap(Uint32 id, void *object){
+
+ int i = id>>2;
+
+ // lock();
+
+ void * obj = m_map[i].m_obj;
+ if (object == obj) {
+ m_map[i].m_next = m_firstFree;
+ m_firstFree = i;
+ } else {
+ ndbout_c("Error: NdbObjectIdMap::::unmap(%u, 0x%x) obj=0x%x", id, object, obj);
+ return 0;
+ }
+
+ // unlock();
+
+#ifdef DEBUG_OBJECTMAP
+ ndbout_c("NdbObjectIdMap::unmap(%u) obj=0x%x", id, obj);
+#endif
+
+ return obj;
+}
+
+inline void *
+NdbObjectIdMap::getObject(Uint32 id){
+#ifdef DEBUG_OBJECTMAP
+ ndbout_c("NdbObjectIdMap::getObject(%u) obj=0x%x", id, m_map[id>>2].m_obj);
+#endif
+ return m_map[id>>2].m_obj;
+}
+
+inline void
+NdbObjectIdMap::expand(Uint32 incSize){
+ Uint32 newSize = m_size + incSize;
+ MapEntry * tmp = (MapEntry*)malloc(newSize * sizeof(MapEntry));
+
+ memcpy(tmp, m_map, m_size * sizeof(MapEntry));
+ free(m_map);
+ m_map = tmp;
+
+ for(Uint32 i = m_size; i<newSize; i++){
+ m_map[i].m_next = i + 1;
+ }
+ m_firstFree = m_size;
+ m_map[newSize-1].m_next = InvalidId;
+ m_size = newSize;
+}
+
+#endif
diff --git a/ndb/src/ndbapi/ScanOperation.txt b/ndb/src/ndbapi/ScanOperation.txt
new file mode 100644
index 00000000000..7197cf66f7e
--- /dev/null
+++ b/ndb/src/ndbapi/ScanOperation.txt
@@ -0,0 +1,10 @@
+x) NdbConnection (main)
+m_theFirstCursorOperation -> y
+
+y) NdbScanOperation
+m_transConnection -> x
+theNdbCon -> z
+
+z) NdbConnection (scan)
+theScanningOp -> y
+theFirstOpInList -> y (until after openScan)
diff --git a/ndb/src/ndbapi/TransporterFacade.cpp b/ndb/src/ndbapi/TransporterFacade.cpp
new file mode 100644
index 00000000000..746ab169b41
--- /dev/null
+++ b/ndb/src/ndbapi/TransporterFacade.cpp
@@ -0,0 +1,989 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <ndb_limits.h>
+#include "AttrType.hpp"
+#include "TransporterFacade.hpp"
+#include "ClusterMgr.hpp"
+#include <IPCConfig.hpp>
+#include <TransporterCallback.hpp>
+#include <TransporterRegistry.hpp>
+#include "NdbApiSignal.hpp"
+#include <NdbOut.hpp>
+#include <NdbEnv.h>
+#include <NdbSleep.h>
+#include <assert.h>
+
+#include "API.hpp"
+#include <ConfigRetriever.hpp>
+#include <NdbConfig.h>
+#include <ndb_version.h>
+#include <SignalLoggerManager.hpp>
+#include <stdlib.h>
+
+#if !defined NDB_OSE && !defined NDB_SOFTOSE
+#include <signal.h>
+#endif
+
+//#define REPORT_TRANSPORTER
+
+#if defined DEBUG_TRANSPORTER
+#define TRP_DEBUG(t) ndbout << __FILE__ << ":" << __LINE__ << ":" << t << endl;
+#else
+#define TRP_DEBUG(t)
+#endif
+
+TransporterFacade* TransporterFacade::theFacadeInstance = NULL;
+
+
+
+/*****************************************************************************
+ * Call back functions
+ *****************************************************************************/
+
+void
+reportError(void * callbackObj, NodeId nodeId, TransporterError errorCode){
+#ifdef REPORT_TRANSPORTER
+ ndbout_c("REPORT_TRANSP: reportError (nodeId=%d, errorCode=%d)",
+ (int)nodeId, (int)errorCode);
+#endif
+ if(errorCode & 0x8000) {
+ ndbout_c("reportError (%d, %d)\n", (int)nodeId, (int)errorCode);
+ ((TransporterFacade*)(callbackObj))->doDisconnect(nodeId);
+ }
+}
+
+/**
+ * Report average send length in bytes (4096 last sends)
+ */
+void
+reportSendLen(void * callbackObj, NodeId nodeId, Uint32 count, Uint64 bytes){
+#ifdef REPORT_TRANSPORTER
+ ndbout_c("REPORT_TRANSP: reportSendLen (nodeId=%d, bytes/count=%d)",
+ (int)nodeId, (Uint32)(bytes/count));
+#endif
+ (void)nodeId;
+ (void)count;
+ (void)bytes;
+}
+
+/**
+ * Report average receive length in bytes (4096 last receives)
+ */
+void
+reportReceiveLen(void * callbackObj,
+ NodeId nodeId, Uint32 count, Uint64 bytes){
+#ifdef REPORT_TRANSPORTER
+ ndbout_c("REPORT_TRANSP: reportReceiveLen (nodeId=%d, bytes/count=%d)",
+ (int)nodeId, (Uint32)(bytes/count));
+#endif
+ (void)nodeId;
+ (void)count;
+ (void)bytes;
+}
+
+/**
+ * Report connection established
+ */
+void
+reportConnect(void * callbackObj, NodeId nodeId){
+#ifdef REPORT_TRANSPORTER
+ ndbout_c("REPORT_TRANSP: API reportConnect (nodeId=%d)", (int)nodeId);
+#endif
+ ((TransporterFacade*)(callbackObj))->reportConnected(nodeId);
+ // TransporterFacade::instance()->reportConnected(nodeId);
+}
+
+/**
+ * Report connection broken
+ */
+void
+reportDisconnect(void * callbackObj, NodeId nodeId, Uint32 error){
+#ifdef REPORT_TRANSPORTER
+ ndbout_c("REPORT_TRANSP: API reportDisconnect (nodeId=%d)", (int)nodeId);
+#endif
+ ((TransporterFacade*)(callbackObj))->reportDisconnected(nodeId);
+ //TransporterFacade::instance()->reportDisconnected(nodeId);
+}
+
+
+/****************************************************************************
+ *
+ *****************************************************************************/
+
+/**
+ * Report connection broken
+ */
+int checkJobBuffer() {
+ return 0;
+}
+
+#ifdef API_TRACE
+static const char * API_SIGNAL_LOG = "API_SIGNAL_LOG";
+static const char * apiSignalLog = 0;
+static SignalLoggerManager signalLogger;
+static
+inline
+bool
+setSignalLog(){
+ signalLogger.flushSignalLog();
+
+ const char * tmp = NdbEnv_GetEnv(API_SIGNAL_LOG, (char *)0, 0);
+ if(tmp != 0 && apiSignalLog != 0 && strcmp(tmp,apiSignalLog) == 0){
+ return true;
+ } else if(tmp == 0 && apiSignalLog == 0){
+ return false;
+ } else if(tmp == 0 && apiSignalLog != 0){
+ signalLogger.setOutputStream(0);
+ apiSignalLog = tmp;
+ return false;
+ } else if(tmp !=0){
+ if (strcmp(tmp, "-") == 0)
+ signalLogger.setOutputStream(stdout);
+ else
+ signalLogger.setOutputStream(fopen(tmp, "w"));
+ apiSignalLog = tmp;
+ return true;
+ }
+ return false;
+}
+#endif
+
+// These symbols are needed, but not used in the API
+int g_sectionSegmentPool;
+struct ErrorReporter {
+ void handleAssert(const char*, const char*, int);
+};
+void ErrorReporter::handleAssert(const char* message, const char* file, int line) {}
+
+/**
+ * The execute function : Handle received signal
+ */
+void
+execute(void * callbackObj, SignalHeader * const header,
+ Uint8 prio, Uint32 * const theData,
+ LinearSectionPtr ptr[3]){
+
+ TransporterFacade * theFacade = (TransporterFacade*)callbackObj;
+ TransporterFacade::ThreadData::Object_Execute oe;
+ Uint32 tRecBlockNo = header->theReceiversBlockNumber;
+
+#ifdef API_TRACE
+ if(setSignalLog()){
+ // header->theVerId_signalNumber != GSN_API_REGREQ &&
+ // header->theVerId_signalNumber != GSN_API_REGCONF){
+ signalLogger.executeSignal(* header,
+ prio,
+ theData,
+ theFacade->ownId(),
+ ptr, header->m_noOfSections);
+ signalLogger.flushSignalLog();
+ }
+#endif
+
+ if (tRecBlockNo >= MIN_API_BLOCK_NO) {
+ oe = theFacade->m_threads.get(tRecBlockNo);
+ if (oe.m_object != 0 && oe.m_executeFunction != 0) {
+ /**
+ * Handle received signal immediately to avoid any unnecessary
+ * copying of data, allocation of memory and other things. Copying
+ * of data could be interesting to support several priority levels
+ * and to support a special memory structure when executing the
+ * signals. Neither of those are interesting when receiving data
+ * in the NDBAPI. The NDBAPI will thus read signal data directly as
+ * it was written by the sender (SCI sender is other node, Shared
+ * memory sender is other process and TCP/IP sender is the OS that
+ * writes the TCP/IP message into a message buffer).
+ */
+ NdbApiSignal tmpSignal(*header);
+ NdbApiSignal * tSignal = &tmpSignal;
+ tSignal->setDataPtr(theData);
+ (* oe.m_executeFunction) (oe.m_object, tSignal, ptr);
+ }//if
+ } else if (tRecBlockNo == API_PACKED) {
+ /**
+ * Block number == 2047 is used to signal a signal that consists of
+ * multiple instances of the same signal. This is an effort to
+ * package the signals so as to avoid unnecessary communication
+ * overhead since TCP/IP has a great performance impact.
+ */
+ Uint32 Tlength = header->theLength;
+ Uint32 Tsent = 0;
+ /**
+ * Since it contains at least two data packets we will first
+ * copy the signal data to safe place.
+ */
+ while (Tsent < Tlength) {
+ Uint32 Theader = theData[Tsent];
+ Tsent++;
+ Uint32 TpacketLen = (Theader & 0x1F) + 3;
+ tRecBlockNo = Theader >> 16;
+ if (TpacketLen <= 25) {
+ if ((TpacketLen + Tsent) <= Tlength) {
+ /**
+ * Set the data length of the signal and the receivers block
+ * reference and then call the API.
+ */
+ header->theLength = TpacketLen;
+ header->theReceiversBlockNumber = tRecBlockNo;
+ Uint32* tDataPtr = &theData[Tsent];
+ Tsent += TpacketLen;
+ if (tRecBlockNo >= MIN_API_BLOCK_NO) {
+ oe = theFacade->m_threads.get(tRecBlockNo);
+ if(oe.m_object != 0 && oe.m_executeFunction != 0){
+ NdbApiSignal tmpSignal(*header);
+ NdbApiSignal * tSignal = &tmpSignal;
+ tSignal->setDataPtr(tDataPtr);
+ (*oe.m_executeFunction)(oe.m_object, tSignal, 0);
+ }
+ }
+ }
+ }
+ }
+ return;
+ } else if (tRecBlockNo == API_CLUSTERMGR) {
+ /**
+ * The signal was aimed for the Cluster Manager.
+ * We handle it immediately here.
+ */
+ ClusterMgr * clusterMgr = theFacade->theClusterMgr;
+ const Uint32 gsn = header->theVerId_signalNumber;
+
+ switch (gsn){
+ case GSN_API_REGREQ:
+ clusterMgr->execAPI_REGREQ(theData);
+ break;
+
+ case GSN_API_REGCONF:
+ clusterMgr->execAPI_REGCONF(theData);
+ break;
+
+ case GSN_API_REGREF:
+ clusterMgr->execAPI_REGREF(theData);
+ break;
+
+ case GSN_NODE_FAILREP:
+ clusterMgr->execNODE_FAILREP(theData);
+ break;
+
+ case GSN_NF_COMPLETEREP:
+ clusterMgr->execNF_COMPLETEREP(theData);
+ break;
+
+ case GSN_ARBIT_STARTREQ:
+ if (theFacade->theArbitMgr != NULL)
+ theFacade->theArbitMgr->doStart(theData);
+ break;
+
+ case GSN_ARBIT_CHOOSEREQ:
+ if (theFacade->theArbitMgr != NULL)
+ theFacade->theArbitMgr->doChoose(theData);
+ break;
+
+ case GSN_ARBIT_STOPORD:
+ if(theFacade->theArbitMgr != NULL)
+ theFacade->theArbitMgr->doStop(theData);
+ break;
+
+ default:
+ break;
+
+ }
+ return;
+ } else {
+ ; // Ignore all other block numbers.
+ if(header->theVerId_signalNumber!=3) {
+ TRP_DEBUG( "TransporterFacade received signal to unknown block no." );
+ ndbout << "BLOCK NO: " << tRecBlockNo << " sig "
+ << header->theVerId_signalNumber << endl;
+ abort();
+ }
+ }
+}
+
+void
+copy(Uint32 * & insertPtr,
+ class SectionSegmentPool & thePool, const SegmentedSectionPtr & _ptr){
+ abort();
+}
+
+extern "C"
+void
+atexit_stop_instance(){
+ TransporterFacade::stop_instance();
+}
+
+/**
+ * Note that this function need no locking since its
+ * only called from the constructor of Ndb (the NdbObject)
+ *
+ * Which is protected by a mutex
+ */
+TransporterFacade*
+TransporterFacade::start_instance(Properties* props, const char *connectString)
+{
+ bool ownProps = false;
+ if (props == NULL) {
+ // TransporterFacade used from API get config from mgmt srvr
+ ConfigRetriever configRetriever;
+ configRetriever.setConnectString(connectString);
+ props = configRetriever.getConfig("API", NDB_VERSION);
+ if (props == 0) {
+ ndbout << "Configuration error: ";
+ const char* erString = configRetriever.getErrorString();
+ if (erString == 0) {
+ erString = "No error specified!";
+ }
+ ndbout << erString << endl;
+ return NULL;
+ }
+ props->put("LocalNodeId", configRetriever.getOwnNodeId());
+ props->put("LocalNodeType", "API");
+
+ ownProps = true;
+ }
+ TransporterFacade* tf = new TransporterFacade();
+
+ if (! tf->init(props)) {
+ delete tf;
+ return NULL;
+ }
+
+ if (ownProps) {
+ delete props;
+ }
+
+ /**
+ * Install atexit handler
+ */
+ atexit(atexit_stop_instance);
+
+ /**
+ * Install signal handler for SIGPIPE
+ *
+ * This due to the fact that a socket connection might have
+ * been closed in between a select and a corresponding send
+ */
+#if !defined NDB_OSE && !defined NDB_SOFTOSE && !defined NDB_WIN32
+ signal(SIGPIPE, SIG_IGN);
+#endif
+
+ if(theFacadeInstance == NULL){
+ theFacadeInstance = tf;
+ }
+
+ return tf;
+}
+
+/**
+ * Note that this function need no locking since its
+ * only called from the destructor of Ndb (the NdbObject)
+ *
+ * Which is protected by a mutex
+ */
+void
+TransporterFacade::stop_instance(){
+ if(theFacadeInstance == NULL){
+ /**
+ * We are called from atexit function
+ */
+ return;
+ }
+
+ theFacadeInstance->doStop();
+
+ delete theFacadeInstance; theFacadeInstance = NULL;
+}
+
+void
+TransporterFacade::doStop(){
+ /**
+ * First stop the ClusterMgr because it needs to send one more signal
+ * and also uses theFacadeInstance to lock/unlock theMutexPtr
+ */
+ if (theClusterMgr != NULL) theClusterMgr->doStop();
+ if (theArbitMgr != NULL) theArbitMgr->doStop(NULL);
+
+ /**
+ * Now stop the send and receive threads
+ */
+ void *status;
+ theStopReceive = 1;
+ NdbThread_WaitFor(theReceiveThread, &status);
+ NdbThread_WaitFor(theSendThread, &status);
+ NdbThread_Destroy(&theReceiveThread);
+ NdbThread_Destroy(&theSendThread);
+}
+
+extern "C"
+void*
+runSendRequest_C(void * me)
+{
+ ((TransporterFacade*) me)->threadMainSend();
+ NdbThread_Exit(0);
+ return me;
+}
+
+void TransporterFacade::threadMainSend(void)
+{
+ theTransporterRegistry->startSending();
+ while(!theStopReceive) {
+ NdbSleep_MilliSleep(10);
+ NdbMutex_Lock(theMutexPtr);
+ if (sendPerformedLastInterval == 0) {
+ theTransporterRegistry->performSend();
+ }
+ sendPerformedLastInterval = 0;
+ NdbMutex_Unlock(theMutexPtr);
+ }
+ theTransporterRegistry->stopSending();
+}
+
+extern "C"
+void*
+runReceiveResponse_C(void * me)
+{
+ ((TransporterFacade*) me)->threadMainReceive();
+ NdbThread_Exit(0);
+ return me;
+}
+
+void TransporterFacade::threadMainReceive(void)
+{
+ theTransporterRegistry->startReceiving();
+ NdbMutex_Lock(theMutexPtr);
+ theTransporterRegistry->checkConnections();
+ NdbMutex_Unlock(theMutexPtr);
+ while(!theStopReceive) {
+ for(int i = 0; i<10; i++){
+ const int res = theTransporterRegistry->pollReceive(10);
+ if(res > 0){
+ NdbMutex_Lock(theMutexPtr);
+ theTransporterRegistry->performReceive();
+ NdbMutex_Unlock(theMutexPtr);
+ }
+ }
+ NdbMutex_Lock(theMutexPtr);
+ theTransporterRegistry->checkConnections();
+ NdbMutex_Unlock(theMutexPtr);
+ }//while
+ theTransporterRegistry->stopReceiving();
+}
+
+TransporterFacade::TransporterFacade() :
+ theTransporterRegistry(0),
+ theStopReceive(0),
+ theSendThread(NULL),
+ theReceiveThread(NULL)
+{
+ theMutexPtr = NdbMutex_Create();
+ sendPerformedLastInterval = 0;
+
+ checkCounter = 4;
+ currentSendLimit = 1;
+ theClusterMgr = NULL;
+ theArbitMgr = NULL;
+ theStartNodeId = 1;
+}
+
+bool
+TransporterFacade::init(Properties* props)
+{
+ IPCConfig config(props);
+
+ if (config.init() != 0) {
+ TRP_DEBUG( "IPCConfig object config failed to init()" );
+ return false;
+ }
+ theOwnId = config.ownId();
+
+ theTransporterRegistry = new TransporterRegistry(this);
+ if(config.configureTransporters(theTransporterRegistry) <= 0) {
+ TRP_DEBUG( "configureTransporters returned 0 or less" );
+ return false;
+ }
+
+ theClusterMgr = new ClusterMgr(* this);
+ theClusterMgr->init(config);
+
+ theReceiveThread = NdbThread_Create(runReceiveResponse_C,
+ (void**)this,
+ 32768,
+ "ndb_receive",
+ NDB_THREAD_PRIO_LOW);
+
+ theSendThread = NdbThread_Create(runSendRequest_C,
+ (void**)this,
+ 32768,
+ "ndb_send",
+ NDB_THREAD_PRIO_LOW);
+
+ theClusterMgr->startThread();
+
+ /**
+ * Unless there is a "Name", the initiated transporter is within
+ * an NDB Cluster. (If "Name" is defined, then the transporter
+ * is used to connect to a different system, i.e. NDB Cluster.)
+ */
+ if (!props->contains("Name")) {
+ const Properties* p = 0;
+ if(!props->get("Node", ownId(), &p)) {
+ TRP_DEBUG( "Node info missing from config." );
+ return false;
+ }
+
+ Uint32 rank = 0;
+ if (p->get("ArbitrationRank", &rank) && rank > 0) {
+ theArbitMgr = new ArbitMgr(* this);
+ theArbitMgr->setRank(rank);
+ Uint32 delay = 0;
+ p->get("ArbitrationDelay", &delay);
+ theArbitMgr->setDelay(delay);
+ }
+ }
+
+#ifdef API_TRACE
+ signalLogger.logOn(true, 0, SignalLoggerManager::LogInOut);
+#endif
+
+ return true;
+}
+
+
+void
+TransporterFacade::ReportNodeDead(NodeId tNodeId)
+{
+ /**
+ * When a node fails we must report this to each Ndb object.
+ * The function that is used for communicating node failures is called.
+ * This is to ensure that the Ndb objects do not think their connections
+ * are correct after a failure followed by a restart.
+ * After the restart the node is up again and the Ndb object
+ * might not have noticed the failure.
+ */
+ Uint32 sz = m_threads.m_statusNext.size();
+ for (Uint32 i = 0; i < sz ; i ++) {
+ if (m_threads.getInUse(i)){
+ void * obj = m_threads.m_objectExecute[i].m_object;
+ NodeStatusFunction RegPC = m_threads.m_statusFunction[i];
+ (*RegPC) (obj, tNodeId, false, false);
+ }
+ }
+}
+
+void
+TransporterFacade::ReportNodeFailureComplete(NodeId tNodeId)
+{
+ /**
+ * When a node fails we must report this to each Ndb object.
+ * The function that is used for communicating node failures is called.
+ * This is to ensure that the Ndb objects do not think their connections
+ * are correct after a failure followed by a restart.
+ * After the restart the node is up again and the Ndb object
+ * might not have noticed the failure.
+ */
+ Uint32 sz = m_threads.m_statusNext.size();
+ for (Uint32 i = 0; i < sz ; i ++) {
+ if (m_threads.getInUse(i)){
+ void * obj = m_threads.m_objectExecute[i].m_object;
+ NodeStatusFunction RegPC = m_threads.m_statusFunction[i];
+ (*RegPC) (obj, tNodeId, false, true);
+ }
+ }
+}
+
+void
+TransporterFacade::ReportNodeAlive(NodeId tNodeId)
+{
+ /**
+ * When a node fails we must report this to each Ndb object.
+ * The function that is used for communicating node failures is called.
+ * This is to ensure that the Ndb objects do not think there connections
+ * are correct after a failure
+ * followed by a restart.
+ * After the restart the node is up again and the Ndb object
+ * might not have noticed the failure.
+ */
+ Uint32 sz = m_threads.m_statusNext.size();
+ for (Uint32 i = 0; i < sz ; i ++) {
+ if (m_threads.getInUse(i)){
+ void * obj = m_threads.m_objectExecute[i].m_object;
+ NodeStatusFunction RegPC = m_threads.m_statusFunction[i];
+ (*RegPC) (obj, tNodeId, true, false);
+ }
+ }
+}
+
+int
+TransporterFacade::close(BlockNumber blockNumber)
+{
+ NdbMutex_Lock(theMutexPtr);
+ close_local(blockNumber);
+ NdbMutex_Unlock(theMutexPtr);
+ return 0;
+}
+
+int
+TransporterFacade::close_local(BlockNumber blockNumber){
+ m_threads.close(blockNumber);
+ return 0;
+}
+
+int
+TransporterFacade::open(void* objRef,
+ ExecuteFunction fun,
+ NodeStatusFunction statusFun)
+{
+ return m_threads.open(objRef, fun, statusFun);
+}
+
+TransporterFacade::~TransporterFacade(){
+
+ NdbMutex_Lock(theMutexPtr);
+ delete theClusterMgr;
+ delete theArbitMgr;
+ delete theTransporterRegistry;
+ NdbMutex_Unlock(theMutexPtr);
+ NdbMutex_Destroy(theMutexPtr);
+#ifdef API_TRACE
+ signalLogger.setOutputStream(0);
+#endif
+}
+
+void
+TransporterFacade::calculateSendLimit()
+{
+ Uint32 Ti;
+ Uint32 TthreadCount = 0;
+
+ Uint32 sz = m_threads.m_statusNext.size();
+ for (Ti = 0; Ti < sz; Ti++) {
+ if (m_threads.m_statusNext[Ti] == (ThreadData::ACTIVE)){
+ TthreadCount++;
+ m_threads.m_statusNext[Ti] = ThreadData::INACTIVE;
+ }
+ }
+ currentSendLimit = TthreadCount;
+ if (currentSendLimit == 0) {
+ currentSendLimit = 1;
+ }
+ checkCounter = currentSendLimit << 2;
+}
+
+
+//-------------------------------------------------
+// Force sending but still report the sending to the
+// adaptive algorithm.
+//-------------------------------------------------
+void TransporterFacade::forceSend(Uint32 block_number) {
+ checkCounter--;
+ m_threads.m_statusNext[block_number - MIN_API_BLOCK_NO] = ThreadData::ACTIVE;
+ sendPerformedLastInterval = 1;
+ if (checkCounter < 0) {
+ calculateSendLimit();
+ }
+ theTransporterRegistry->forceSendCheck(0);
+}
+
+//-------------------------------------------------
+// Improving API performance
+//-------------------------------------------------
+void
+TransporterFacade::checkForceSend(Uint32 block_number) {
+ m_threads.m_statusNext[block_number - MIN_API_BLOCK_NO] = ThreadData::ACTIVE;
+ //-------------------------------------------------
+ // This code is an adaptive algorithm to discover when
+ // the API should actually send its buffers. The reason
+ // is that the performance is highly dependent on the
+ // size of the writes over the communication network.
+ // Thus we try to ensure that the send size is as big
+ // as possible. At the same time we don't want response
+ // time to increase so therefore we have to keep track of
+ // how the users are performing adaptively.
+ //-------------------------------------------------
+
+ if (theTransporterRegistry->forceSendCheck(currentSendLimit) == 1) {
+ sendPerformedLastInterval = 1;
+ }
+ checkCounter--;
+ if (checkCounter < 0) {
+ calculateSendLimit();
+ }
+}
+
+
+/******************************************************************************
+ * SEND SIGNAL METHODS
+ ******************************************************************************/
+
+int
+TransporterFacade::sendSignal(NdbApiSignal * aSignal, NodeId aNode){
+ Uint32* tDataPtr = aSignal->getDataPtrSend();
+ Uint32 Tlen = aSignal->theLength;
+ Uint32 TBno = aSignal->theReceiversBlockNumber;
+ if(getIsNodeSendable(aNode) == true){
+#ifdef API_TRACE
+ if(setSignalLog()){
+ // aSignal->theVerId_signalNumber != GSN_API_REGREQ &&
+ // aSignal->theVerId_signalNumber != GSN_API_REGCONF){
+ Uint32 tmp = aSignal->theSendersBlockRef;
+ aSignal->theSendersBlockRef = numberToRef(tmp, theOwnId);
+ LinearSectionPtr ptr[3];
+ signalLogger.sendSignal(* aSignal,
+ 1,
+ aSignal->getDataPtr(),
+ aNode, ptr, 0);
+ signalLogger.flushSignalLog();
+ aSignal->theSendersBlockRef = tmp;
+ }
+#endif
+ if ((Tlen != 0) && (Tlen <= 25) && (TBno != 0)) {
+ SendStatus ss = theTransporterRegistry->prepareSend(aSignal,
+ 1, // JBB
+ tDataPtr,
+ aNode,
+ 0);
+ //if (ss != SEND_OK) ndbout << ss << endl;
+ return (ss == SEND_OK ? 0 : -1);
+ } else {
+ ndbout << "ERR: SigLen = " << Tlen << " BlockRec = " << TBno;
+ ndbout << " SignalNo = " << aSignal->theVerId_signalNumber << endl;
+ assert(0);
+ }//if
+ }
+ //const ClusterMgr::Node & node = theClusterMgr->getNodeInfo(aNode);
+ //const Uint32 startLevel = node.m_state.startLevel;
+ return -1; // Node Dead
+}
+
+int
+TransporterFacade::sendSignalUnCond(NdbApiSignal * aSignal, NodeId aNode){
+#ifdef API_TRACE
+ if(setSignalLog()){
+ //aSignal->theVerId_signalNumber != GSN_API_REGREQ &&
+ //aSignal->theVerId_signalNumber != GSN_API_REGCONF
+ Uint32 tmp = aSignal->theSendersBlockRef;
+ aSignal->theSendersBlockRef = numberToRef(tmp, theOwnId);
+ LinearSectionPtr ptr[3];
+ signalLogger.sendSignal(* aSignal,
+ 0,
+ aSignal->getDataPtr(),
+ aNode, ptr, 0);
+ signalLogger.flushSignalLog();
+ aSignal->theSendersBlockRef = tmp;
+ }
+#endif
+ assert((aSignal->theLength != 0) &&
+ (aSignal->theLength <= 25) &&
+ (aSignal->theReceiversBlockNumber != 0));
+ SendStatus ss = theTransporterRegistry->prepareSend(aSignal,
+ 0,
+ aSignal->getDataPtr(),
+ aNode,
+ 0);
+
+ return (ss == SEND_OK ? 0 : -1);
+}
+
+int
+TransporterFacade::sendFragmentedSignal(NdbApiSignal* aSignal, NodeId aNode,
+ LinearSectionPtr ptr[3], Uint32 secs){
+ aSignal->m_noOfSections = secs;
+ if(getIsNodeSendable(aNode) == true){
+#ifdef API_TRACE
+ if(setSignalLog()){
+ Uint32 tmp = aSignal->theSendersBlockRef;
+ aSignal->theSendersBlockRef = numberToRef(tmp, theOwnId);
+ signalLogger.sendSignal(* aSignal,
+ 1,
+ aSignal->getDataPtrSend(),
+ aNode,
+ ptr, secs);
+ signalLogger.flushSignalLog();
+ aSignal->theSendersBlockRef = tmp;
+ }
+#endif
+ SendStatus ss = theTransporterRegistry->prepareSend
+ (aSignal,
+ 1, // JBB
+ aSignal->getDataPtrSend(),
+ aNode,
+ ptr);
+ assert(ss != SEND_MESSAGE_TOO_BIG);
+ aSignal->m_noOfSections = 0;
+ return (ss == SEND_OK ? 0 : -1);
+ }
+ aSignal->m_noOfSections = 0;
+ return -1;
+}
+
+
+
+int
+TransporterFacade::sendFragmentedSignalUnCond(NdbApiSignal* aSignal,
+ NodeId aNode,
+ LinearSectionPtr ptr[3],
+ Uint32 secs){
+ aSignal->m_noOfSections = secs;
+
+#ifdef API_TRACE
+ if(setSignalLog()){
+ Uint32 tmp = aSignal->theSendersBlockRef;
+ aSignal->theSendersBlockRef = numberToRef(tmp, theOwnId);
+ signalLogger.sendSignal(* aSignal,
+ 1,
+ aSignal->getDataPtrSend(),
+ aNode,
+ ptr, secs);
+ aSignal->theSendersBlockRef = tmp;
+ }
+#endif
+ SendStatus ss = theTransporterRegistry->prepareSend(aSignal,
+ 1, // JBB
+ aSignal->getDataPtrSend(),
+ aNode,
+ ptr);
+ assert(ss != SEND_MESSAGE_TOO_BIG);
+ aSignal->m_noOfSections = 0;
+ return (ss == SEND_OK ? 0 : -1);
+}
+
+
+
+/******************************************************************************
+ * CONNECTION METHODS Etc
+ ******************************************************************************/
+
+void
+TransporterFacade::doConnect(int aNodeId){
+ theTransporterRegistry->setIOState(aNodeId, NoHalt);
+ theTransporterRegistry->setPerformState(aNodeId, PerformConnect);
+}
+
+void
+TransporterFacade::doDisconnect(int aNodeId)
+{
+ theTransporterRegistry->setPerformState(aNodeId, PerformDisconnect);
+}
+
+void
+TransporterFacade::reportConnected(int aNodeId)
+{
+ theClusterMgr->reportConnected(aNodeId);
+ return;
+}
+
+void
+TransporterFacade::reportDisconnected(int aNodeId)
+{
+ theClusterMgr->reportDisconnected(aNodeId);
+ return;
+}
+
+NodeId
+TransporterFacade::ownId() const
+{
+ return theOwnId;
+}
+
+bool
+TransporterFacade::isConnected(NodeId aNodeId){
+ return theTransporterRegistry->performState(aNodeId) == PerformIO;
+}
+
+NodeId
+TransporterFacade::get_an_alive_node()
+{
+#ifdef VM_TRACE
+ const char* p = NdbEnv_GetEnv("NDB_ALIVE_NODE_ID", (char*)0, 0);
+ if (p != 0 && *p != 0)
+ return atoi(p);
+#endif
+ NodeId i;
+ for (i = theStartNodeId; i < MAX_NDB_NODES; i++) {
+ if (get_node_alive(i)){
+ theStartNodeId = ((i + 1) % MAX_NDB_NODES);
+ return i;
+ }
+ }
+ for (i = 1; i < theStartNodeId; i++) {
+ if (get_node_alive(i)){
+ theStartNodeId = ((i + 1) % MAX_NDB_NODES);
+ return i;
+ }
+ }
+ return (NodeId)0;
+}
+
+TransporterFacade::ThreadData::ThreadData(Uint32 size){
+ m_firstFree = END_OF_LIST;
+ expand(size);
+}
+
+void
+TransporterFacade::ThreadData::expand(Uint32 size){
+ Object_Execute oe = { 0 ,0 };
+ NodeStatusFunction fun = 0;
+
+ const Uint32 sz = m_statusNext.size();
+ m_objectExecute.fill(sz + size, oe);
+ m_statusFunction.fill(sz + size, fun);
+ for(Uint32 i = 0; i<size; i++){
+ m_statusNext.push_back(sz + i + 1);
+ }
+
+ m_statusNext.back() = m_firstFree;
+ m_firstFree = m_statusNext.size() - size;
+}
+
+int
+TransporterFacade::ThreadData::open(void* objRef,
+ ExecuteFunction fun,
+ NodeStatusFunction fun2){
+
+ Uint32 nextFree = m_firstFree;
+
+ if(m_statusNext.size() >= MAX_NO_THREADS && nextFree == END_OF_LIST){
+ return -1;
+ }
+
+ if(nextFree == END_OF_LIST){
+ expand(10);
+ nextFree = m_firstFree;
+ }
+
+ m_firstFree = m_statusNext[nextFree];
+
+ Object_Execute oe = { objRef , fun };
+
+ m_statusNext[nextFree] = INACTIVE;
+ m_objectExecute[nextFree] = oe;
+ m_statusFunction[nextFree] = fun2;
+
+ return nextFree + MIN_API_BLOCK_NO;
+}
+
+int
+TransporterFacade::ThreadData::close(int number){
+ number -= MIN_API_BLOCK_NO;
+ assert(getInUse(number));
+ m_statusNext[number] = m_firstFree;
+ m_firstFree = number;
+ Object_Execute oe = { 0, 0 };
+ m_objectExecute[number] = oe;
+ m_statusFunction[number] = 0;
+ return 0;
+}
diff --git a/ndb/src/ndbapi/TransporterFacade.hpp b/ndb/src/ndbapi/TransporterFacade.hpp
new file mode 100644
index 00000000000..d9d2dbbcf0f
--- /dev/null
+++ b/ndb/src/ndbapi/TransporterFacade.hpp
@@ -0,0 +1,316 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TransporterFacade_H
+#define TransporterFacade_H
+
+#include <AttrType.hpp>
+#include <kernel_types.h>
+#include <ndb_limits.h>
+#include <NdbThread.h>
+#include <TransporterRegistry.hpp>
+#include <NdbMutex.h>
+#include "DictCache.hpp"
+#include <BlockNumbers.h>
+
+class ClusterMgr;
+class ArbitMgr;
+class Properties;
+class IPCConfig;
+
+class Ndb;
+class NdbApiSignal;
+
+typedef void (* ExecuteFunction)(void *, NdbApiSignal *, LinearSectionPtr ptr[3]);
+typedef void (* NodeStatusFunction)(void *, NodeId, bool nodeAlive, bool nfComplete);
+
+extern "C" {
+ void* runSendRequest_C(void*);
+ void* runReceiveResponse_C(void*);
+ void atexit_stop_instance();
+}
+
+class TransporterFacade
+{
+public:
+ TransporterFacade();
+ virtual ~TransporterFacade();
+ bool init(Properties* props);
+
+ static TransporterFacade* instance();
+ static TransporterFacade* start_instance(Properties* ipcConfig, const char *connectString);
+ static void stop_instance();
+
+ /**
+ * Register this block for sending/receiving signals
+ * @return BlockNumber or -1 for failure
+ */
+ int open(void* objRef, ExecuteFunction, NodeStatusFunction);
+
+ // Close this block number
+ int close(BlockNumber blockNumber);
+
+ // Only sends to nodes which are alive
+ int sendSignal(NdbApiSignal * signal, NodeId nodeId);
+ int sendFragmentedSignal(NdbApiSignal*, NodeId,
+ LinearSectionPtr ptr[3], Uint32 secs);
+
+ //Dirrrrty
+ int sendFragmentedSignalUnCond(NdbApiSignal*, NodeId,
+ LinearSectionPtr ptr[3], Uint32 secs);
+
+
+ // Is node available for running transactions
+ bool get_node_alive(NodeId nodeId) const;
+ bool get_node_stopping(NodeId nodeId) const;
+ bool getIsNodeDefined(NodeId nodeId) const;
+ bool getIsNodeSendable(NodeId nodeId) const;
+ Uint32 getNodeGrp(NodeId nodeId) const;
+ Uint32 getNodeSequence(NodeId nodeId) const;
+
+ // Is there space in sendBuffer to send messages
+ bool check_send_size(Uint32 node_id, Uint32 send_size);
+
+ // My own processor id
+ NodeId ownId() const;
+
+ void doConnect(int NodeId);
+ void reportConnected(int NodeId);
+ void doDisconnect(int NodeId);
+ void reportDisconnected(int NodeId);
+
+ NodeId get_an_alive_node();
+ void ReportNodeAlive(NodeId nodeId);
+ void ReportNodeDead(NodeId nodeId);
+ void ReportNodeFailureComplete(NodeId nodeId);
+
+ void lock_mutex();
+ void unlock_mutex();
+
+ // Improving the API performance
+ void forceSend(Uint32 block_number);
+ void checkForceSend(Uint32 block_number);
+
+ // Close this block number
+ int close_local(BlockNumber blockNumber);
+ void setState(Uint32 aNodeId, PerformState aState);
+
+private:
+ /**
+ * Send a signal unconditional of node status (used by ClusterMgr)
+ */
+ friend class ClusterMgr;
+ friend class ArbitMgr;
+ friend class MgmtSrvr;
+ friend class SignalSender;
+ friend class GrepPS;
+ friend class ExtSender; ///< @todo Hack to be able to sendSignalUnCond
+ friend class GrepSS;
+ friend class Ndb;
+
+ int sendSignalUnCond(NdbApiSignal *, NodeId nodeId);
+
+ bool isConnected(NodeId aNodeId);
+ void doStop();
+
+ TransporterRegistry* theTransporterRegistry;
+ int sendPerformedLastInterval;
+ int theOwnId;
+
+ NodeId theStartNodeId;
+
+ ClusterMgr* theClusterMgr;
+ ArbitMgr* theArbitMgr;
+
+ // Improving the API response time
+ int checkCounter;
+ Uint32 currentSendLimit;
+
+ void calculateSendLimit();
+
+ // Declarations for the receive and send thread
+ int theStopReceive;
+
+ void threadMainSend(void);
+ NdbThread* theSendThread;
+ void threadMainReceive(void);
+ NdbThread* theReceiveThread;
+
+ friend void* runSendRequest_C(void*);
+ friend void* runReceiveResponse_C(void*);
+ friend void atexit_stop_instance();
+
+ /**
+ * Block number handling
+ */
+ struct ThreadData {
+ static const Uint32 ACTIVE = (1 << 16) | 1;
+ static const Uint32 INACTIVE = (1 << 16);
+ static const Uint32 END_OF_LIST = MAX_NO_THREADS + 1;
+
+ ThreadData(Uint32 initialSize = 32);
+
+ /**
+ * Split "object" into 3 list
+ * This to improve locality
+ * when iterating over lists
+ */
+ struct Object_Execute {
+ void * m_object;
+ ExecuteFunction m_executeFunction;
+ };
+ struct NodeStatus_NextFree {
+ NodeStatusFunction m_statusFunction;
+ };
+
+ Uint32 m_firstFree;
+ Vector<Uint32> m_statusNext;
+ Vector<Object_Execute> m_objectExecute;
+ Vector<NodeStatusFunction> m_statusFunction;
+
+ int open(void* objRef, ExecuteFunction, NodeStatusFunction);
+ int close(int number);
+ void expand(Uint32 size);
+
+ inline Object_Execute get(Uint16 blockNo) const {
+ blockNo -= MIN_API_BLOCK_NO;
+ if(blockNo < m_objectExecute.size()){
+ return m_objectExecute[blockNo];
+ }
+ Object_Execute oe = { 0, 0 };
+ return oe;
+ }
+
+ /**
+ * Is the block number used currently
+ */
+ inline bool getInUse(Uint16 index) const {
+ return (m_statusNext[index] & (1 << 16)) != 0;
+ }
+ } m_threads;
+
+ /**
+ * execute function
+ */
+ friend void execute(void * callbackObj, SignalHeader * const header,
+ Uint8 prio,
+ Uint32 * const theData, LinearSectionPtr ptr[3]);
+
+public:
+ NdbMutex* theMutexPtr;
+private:
+ static TransporterFacade* theFacadeInstance;
+
+public:
+ GlobalDictCache m_globalDictCache;
+};
+
+inline
+TransporterFacade*
+TransporterFacade::instance()
+{
+ return theFacadeInstance;
+}
+
+inline
+void
+TransporterFacade::lock_mutex()
+{
+ NdbMutex_Lock(theMutexPtr);
+}
+
+inline
+void
+TransporterFacade::unlock_mutex()
+{
+ NdbMutex_Unlock(theMutexPtr);
+}
+
+#include "ClusterMgr.hpp"
+
+inline
+bool
+TransporterFacade::check_send_size(Uint32 node_id, Uint32 send_size)
+{
+ return true;
+}
+
+inline
+bool
+TransporterFacade::getIsNodeDefined(NodeId n) const {
+ return theClusterMgr->getNodeInfo(n).defined;
+}
+
+inline
+Uint32
+TransporterFacade::getNodeGrp(NodeId n) const {
+ return theClusterMgr->getNodeInfo(n).m_state.nodeGroup;
+}
+
+
+inline
+bool
+TransporterFacade::get_node_alive(NodeId n) const {
+
+ const ClusterMgr::Node & node = theClusterMgr->getNodeInfo(n);
+ return node.m_alive;
+}
+
+inline
+bool
+TransporterFacade::get_node_stopping(NodeId n) const {
+ const ClusterMgr::Node & node = theClusterMgr->getNodeInfo(n);
+ return ((node.m_state.startLevel == NodeState::SL_STOPPING_1) ||
+ (node.m_state.startLevel == NodeState::SL_STOPPING_2));
+}
+
+inline
+bool
+TransporterFacade::getIsNodeSendable(NodeId n) const {
+ const ClusterMgr::Node & node = theClusterMgr->getNodeInfo(n);
+ const Uint32 startLevel = node.m_state.startLevel;
+
+ if (node.m_info.m_type == NodeInfo::DB) {
+ if(node.m_state.singleUserMode &&
+ ownId() == node.m_state.singleUserApi) {
+ return (node.compatible &&
+ (node.m_state.startLevel == NodeState::SL_STOPPING_1 ||
+ node.m_state.startLevel == NodeState::SL_STARTED ||
+ node.m_state.startLevel == NodeState::SL_SINGLEUSER));
+ }
+ else
+ return node.compatible && (startLevel == NodeState::SL_STARTED ||
+ startLevel == NodeState::SL_STOPPING_1);
+ } else if (node.m_info.m_type == NodeInfo::REP) {
+ /**
+ * @todo Check that REP node actually has received API_REG_REQ
+ */
+ return node.compatible;
+ } else {
+ ndbout_c("TransporterFacade::getIsNodeSendable: Illegal node type: "
+ "%d of node: %d",
+ node.m_info.m_type, n);
+ abort();
+ }
+}
+
+inline
+Uint32
+TransporterFacade::getNodeSequence(NodeId n) const {
+ return theClusterMgr->getNodeInfo(n).m_info.m_connectCount;
+}
+
+#endif // TransporterFacade_H
diff --git a/ndb/src/ndbapi/signal-sender/Makefile b/ndb/src/ndbapi/signal-sender/Makefile
new file mode 100644
index 00000000000..56e6ce1eac0
--- /dev/null
+++ b/ndb/src/ndbapi/signal-sender/Makefile
@@ -0,0 +1,19 @@
+include .defs.mk
+
+TYPE := ndbapi
+
+NONPIC_ARCHIVE := Y
+ARCHIVE_TARGET := signal-sender
+
+BIN_TARGET_LIBS := # -lkalle
+BIN_TARGET_ARCHIVES := portlib # $(NDB_TOP)/lib/libkalle.a
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = SignalSender.cpp
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/src/ndbapi)
+
+include $(NDB_TOP)/Epilogue.mk
+
+###
+# Backward compatible
diff --git a/ndb/src/ndbapi/signal-sender/SignalSender.cpp b/ndb/src/ndbapi/signal-sender/SignalSender.cpp
new file mode 100644
index 00000000000..d60f6240a9c
--- /dev/null
+++ b/ndb/src/ndbapi/signal-sender/SignalSender.cpp
@@ -0,0 +1,238 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <assert.h>
+#include "SignalSender.hpp"
+#include "ConfigRetriever.hpp"
+#include <NdbSleep.h>
+#include <SignalLoggerManager.hpp>
+
+SimpleSignal::SimpleSignal(bool dealloc){
+ memset(this, 0, sizeof(* this));
+ deallocSections = dealloc;
+}
+
+SimpleSignal::~SimpleSignal(){
+ if(!deallocSections)
+ return;
+ if(ptr[0].p != 0) delete []ptr[0].p;
+ if(ptr[1].p != 0) delete []ptr[1].p;
+ if(ptr[2].p != 0) delete []ptr[2].p;
+}
+
+void
+SimpleSignal::set(class SignalSender& ss,
+ Uint8 trace, Uint16 recBlock, Uint16 gsn, Uint32 len){
+
+ header.theTrace = trace;
+ header.theReceiversBlockNumber = recBlock;
+ header.theVerId_signalNumber = gsn;
+ header.theLength = len;
+ header.theSendersBlockRef = refToBlock(ss.getOwnRef());
+}
+
+void
+SimpleSignal::print(FILE * out){
+ fprintf(out, "---- Signal ----------------\n");
+ SignalLoggerManager::printSignalHeader(out, header, 0, 0, false);
+ SignalLoggerManager::printSignalData(out, header, theData);
+ for(Uint32 i = 0; i<header.m_noOfSections; i++){
+ Uint32 len = ptr[i].sz;
+ fprintf(out, " --- Section %d size=%d ---\n", i, len);
+ Uint32 * signalData = ptr[i].p;
+ while(len >= 7){
+ fprintf(out,
+ " H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x H\'%.8x\n",
+ signalData[0], signalData[1], signalData[2], signalData[3],
+ signalData[4], signalData[5], signalData[6]);
+ len -= 7;
+ signalData += 7;
+ }
+ if(len > 0){
+ fprintf(out, " H\'%.8x", signalData[0]);
+ for(Uint32 i = 1; i<len; i++)
+ fprintf(out, " H\'%.8x", signalData[i]);
+ fprintf(out, "\n");
+ }
+ }
+}
+
+SignalSender::SignalSender(const char * connectString){
+ m_cond = NdbCondition_Create();
+ theFacade = TransporterFacade::start_instance(0,connectString);
+ m_blockNo = theFacade->open(this, execSignal, execNodeStatus);
+ assert(m_blockNo > 0);
+}
+
+SignalSender::~SignalSender(){
+ theFacade->close(m_blockNo);
+ theFacade->stop_instance();
+ NdbCondition_Destroy(m_cond);
+}
+
+Uint32
+SignalSender::getOwnRef() const {
+ return numberToRef(m_blockNo, theFacade->ownId());
+}
+
+bool
+SignalSender::connectOne(Uint32 timeOutMillis){
+ NDB_TICKS start = NdbTick_CurrentMillisecond();
+ NDB_TICKS now = start;
+ while(theFacade->theClusterMgr->getNoOfConnectedNodes() == 0 &&
+ (timeOutMillis == 0 || (now - start) < timeOutMillis)){
+ NdbSleep_MilliSleep(100);
+ }
+ return theFacade->theClusterMgr->getNoOfConnectedNodes() > 0;
+}
+
+bool
+SignalSender::connectAll(Uint32 timeOutMillis){
+ NDB_TICKS start = NdbTick_CurrentMillisecond();
+ NDB_TICKS now = start;
+ while(theFacade->theClusterMgr->getNoOfConnectedNodes() < 1 &&
+ (timeOutMillis == 0 || (now - start) < timeOutMillis)){
+ NdbSleep_MilliSleep(100);
+ }
+ return theFacade->theClusterMgr->getNoOfConnectedNodes() >= 1;
+}
+
+
+Uint32
+SignalSender::getAliveNode(){
+ return theFacade->get_an_alive_node();
+}
+
+const ClusterMgr::Node &
+SignalSender::getNodeInfo(Uint16 nodeId) const {
+ return theFacade->theClusterMgr->getNodeInfo(nodeId);
+}
+
+Uint32
+SignalSender::getNoOfConnectedNodes() const {
+ return theFacade->theClusterMgr->getNoOfConnectedNodes();
+}
+
+SendStatus
+SignalSender::sendSignal(Uint16 nodeId, const SimpleSignal * s){
+ return theFacade->theTransporterRegistry->prepareSend(&s->header,
+ 1, // JBB
+ &s->theData[0],
+ nodeId,
+ &s->ptr[0]);
+}
+
+template<class T>
+SimpleSignal *
+SignalSender::waitFor(Uint32 timeOutMillis, T & t){
+
+ Guard g(theFacade->theMutexPtr);
+
+ SimpleSignal * s = t.check(m_jobBuffer);
+ if(s != 0){
+ return s;
+ }
+
+ NDB_TICKS now = NdbTick_CurrentMillisecond();
+ NDB_TICKS stop = now + timeOutMillis;
+ Uint32 wait = (timeOutMillis == 0 ? 10 : timeOutMillis);
+ do {
+ NdbCondition_WaitTimeout(m_cond,
+ theFacade->theMutexPtr,
+ wait);
+
+
+ SimpleSignal * s = t.check(m_jobBuffer);
+ if(s != 0){
+ return s;
+ }
+
+ now = NdbTick_CurrentMillisecond();
+ wait = (timeOutMillis == 0 ? 10 : stop - now);
+ } while(stop > now || timeOutMillis == 0);
+
+ return 0;
+}
+
+class WaitForAny {
+public:
+ SimpleSignal * check(Vector<SimpleSignal*> & m_jobBuffer){
+ if(m_jobBuffer.size() > 0){
+ SimpleSignal * s = m_jobBuffer[0];
+ m_jobBuffer.erase(0);
+ return s;
+ }
+ return 0;
+ }
+};
+
+SimpleSignal *
+SignalSender::waitFor(Uint32 timeOutMillis){
+
+ WaitForAny w;
+ return waitFor(timeOutMillis, w);
+}
+
+class WaitForNode {
+public:
+ Uint32 m_nodeId;
+ SimpleSignal * check(Vector<SimpleSignal*> & m_jobBuffer){
+ Uint32 len = m_jobBuffer.size();
+ for(Uint32 i = 0; i<len; i++){
+ if(refToNode(m_jobBuffer[i]->header.theSendersBlockRef) == m_nodeId){
+ SimpleSignal * s = m_jobBuffer[i];
+ m_jobBuffer.erase(i);
+ return s;
+ }
+ }
+ return 0;
+ }
+};
+
+SimpleSignal *
+SignalSender::waitFor(Uint16 nodeId, Uint32 timeOutMillis){
+
+ WaitForNode w;
+ w.m_nodeId = nodeId;
+ return waitFor(timeOutMillis, w);
+}
+
+#include <NdbApiSignal.hpp>
+
+void
+SignalSender::execSignal(void* signalSender,
+ NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]){
+ SimpleSignal * s = new SimpleSignal(true);
+ s->header = * signal;
+ memcpy(&s->theData[0], signal->getDataPtr(), 4 * s->header.theLength);
+ for(Uint32 i = 0; i<s->header.m_noOfSections; i++){
+ s->ptr[i].p = new Uint32[ptr[i].sz];
+ s->ptr[i].sz = ptr[i].sz;
+ memcpy(s->ptr[i].p, ptr[i].p, 4 * ptr[i].sz);
+ }
+ SignalSender * ss = (SignalSender*)signalSender;
+ ss->m_jobBuffer.push_back(s);
+ NdbCondition_Signal(ss->m_cond);
+}
+
+void
+SignalSender::execNodeStatus(void* signalSender,
+ Uint16 NodeId,
+ bool alive,
+ bool nfCompleted){
+}
+
diff --git a/ndb/src/ndbapi/signal-sender/SignalSender.hpp b/ndb/src/ndbapi/signal-sender/SignalSender.hpp
new file mode 100644
index 00000000000..fffe027dbdd
--- /dev/null
+++ b/ndb/src/ndbapi/signal-sender/SignalSender.hpp
@@ -0,0 +1,82 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SIGNAL_SENDER_HPP
+#define SIGNAL_SENDER_HPP
+
+#include <TransporterDefinitions.hpp>
+#include <TransporterFacade.hpp>
+#include <ClusterMgr.hpp>
+#include <Vector.hpp>
+#include <stdio.h>
+
+struct SimpleSignal {
+public:
+ SimpleSignal(bool dealloc = false);
+ ~SimpleSignal();
+
+ void set(class SignalSender&,
+ Uint8 trace, Uint16 recBlock, Uint16 gsn, Uint32 len);
+
+ struct SignalHeader header;
+ Uint32 theData[25];
+ LinearSectionPtr ptr[3];
+
+ void print(FILE * out = stdout);
+private:
+ bool deallocSections;
+};
+
+class SignalSender {
+public:
+ SignalSender(const char * connectString = 0);
+ virtual ~SignalSender();
+
+ bool connectOne(Uint32 timeOutMillis = 0);
+ bool connectAll(Uint32 timeOutMillis = 0);
+ bool connect(Uint32 timeOutMillis = 0) { return connectAll(timeOutMillis);}
+
+ Uint32 getOwnRef() const;
+
+ Uint32 getAliveNode();
+ Uint32 getNoOfConnectedNodes() const;
+ const ClusterMgr::Node & getNodeInfo(Uint16 nodeId) const;
+
+ SendStatus sendSignal(Uint16 nodeId, const SimpleSignal *);
+
+ SimpleSignal * waitFor(Uint32 timeOutMillis = 0);
+ SimpleSignal * waitFor(Uint16 nodeId, Uint32 timeOutMillis = 0);
+ SimpleSignal * waitFor(Uint16 nodeId, Uint16 gsn, Uint32 timeOutMillis = 0);
+
+private:
+ int m_blockNo;
+ TransporterFacade * theFacade;
+
+ static void execSignal(void* signalSender,
+ NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]);
+
+ static void execNodeStatus(void* signalSender, NodeId,
+ bool alive, bool nfCompleted);
+
+ struct NdbCondition * m_cond;
+ Vector<SimpleSignal *> m_jobBuffer;
+
+ template<class T>
+ SimpleSignal * waitFor(Uint32 timeOutMillis, T & t);
+};
+
+#endif
diff --git a/ndb/src/ndbbaseclient/Makefile b/ndb/src/ndbbaseclient/Makefile
new file mode 100644
index 00000000000..f4c49a95ffa
--- /dev/null
+++ b/ndb/src/ndbbaseclient/Makefile
@@ -0,0 +1,29 @@
+include .defs.mk
+
+TYPE := *
+
+PIC_ARCHIVE := Y
+NONPIC_ARCHIVE := Y
+ARCHIVE_TARGET := ndbbaseclient
+
+A_LIB := Y
+SO_LIB := Y
+PIC_LIB := Y
+LIB_TARGET := ndbclient
+
+LIB_TARGET_ARCHIVES := $(ARCHIVE_TARGET) \
+ ndbapi \
+ mgmapi \
+ newtonapi \
+ transporter \
+ general \
+ signaldataprint \
+ mgmsrvcommon \
+ portlib \
+ logger \
+ trace
+
+SOURCES = ndbbaseclient_dummy.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/ndbbaseclient/ndbbaseclient_dummy.cpp b/ndb/src/ndbbaseclient/ndbbaseclient_dummy.cpp
new file mode 100644
index 00000000000..e69de29bb2d
--- /dev/null
+++ b/ndb/src/ndbbaseclient/ndbbaseclient_dummy.cpp
diff --git a/ndb/src/ndbclient/Makefile b/ndb/src/ndbclient/Makefile
new file mode 100644
index 00000000000..2c597eccfa1
--- /dev/null
+++ b/ndb/src/ndbclient/Makefile
@@ -0,0 +1,37 @@
+include .defs.mk
+
+TYPE := *
+
+PIC_ARCHIVE := Y
+NONPIC_ARCHIVE := Y
+ARCHIVE_TARGET := ndbclient
+
+A_LIB := N
+SO_LIB := Y
+PIC_LIB := Y
+LIB_TARGET := ndbclient_extra
+
+LDFLAGS_LAST = -lstdc++ -lm
+
+LIB_TARGET_ARCHIVES := $(ARCHIVE_TARGET) \
+ ndbapi \
+ mgmapi \
+ newtonapi \
+ transporter \
+ general \
+ signaldataprint \
+ mgmsrvcommon \
+ portlib \
+ logger \
+ trace \
+ odbcdriver \
+ odbchandles \
+ odbcdictionary \
+ odbccodegen \
+ odbcexecutor \
+ odbccommon
+
+SOURCES = ndbclient_dummy.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/src/ndbclient/ndbclient_dummy.cpp b/ndb/src/ndbclient/ndbclient_dummy.cpp
new file mode 100644
index 00000000000..e69de29bb2d
--- /dev/null
+++ b/ndb/src/ndbclient/ndbclient_dummy.cpp
diff --git a/ndb/src/newtonapi/Makefile b/ndb/src/newtonapi/Makefile
new file mode 100644
index 00000000000..bed179046a5
--- /dev/null
+++ b/ndb/src/newtonapi/Makefile
@@ -0,0 +1,27 @@
+include .defs.mk
+
+TYPE := ndbapiclient
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := newtonapi
+
+A_LIB := Y
+SO_LIB := Y
+PIC_LIB := Y
+
+LIB_TARGET := NEWTON_API
+LIB_TARGET_ARCHIVES := $(ARCHIVE_TARGET) NDB_API
+
+SOURCES = \
+ dba_binding.cpp \
+ dba_process.cpp \
+ dba_dac.cpp \
+ dba_init.cpp \
+ dba_schema.cpp \
+ dba_bulkread.cpp \
+ dba_error.cpp \
+ dba_config.cpp
+
+CCFLAGS_LOC += -I../include -I$(call fixpath,$(NDB_TOP)/include/portlib) -I$(call fixpath,$(NDB_TOP)/include/util) -I$(call fixpath,$(NDB_TOP)/include/newtonapi) -DDEBUG
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/newtonapi/dba_binding.cpp b/ndb/src/newtonapi/dba_binding.cpp
new file mode 100644
index 00000000000..724f54c0e4b
--- /dev/null
+++ b/ndb/src/newtonapi/dba_binding.cpp
@@ -0,0 +1,433 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "dba_internal.hpp"
+
+static bool matchType(NdbDictionary::Column::Type, DBA_DataTypes_t);
+static bool matchSize(NdbDictionary::Column::Type, unsigned, Size_t);
+static int computeChecksum(const DBA_Binding_t * bindings);
+
+struct DBA__Array {
+ int count;
+ int data[1];
+
+ bool exists(int value) const {
+ for(int i = 0; i<count; i++)
+ if(data[i] == value)
+ return true;
+ return false;
+ }
+
+ void insert(int value){
+ data[count] = value;
+ count++;
+ }
+};
+
+/**
+ * createBindings
+ */
+static
+DBA_Binding_t *
+createBinding(const char* TableName,
+ int NbCol,
+ const DBA_ColumnBinding_t ColsBindings[],
+ Size_t StructSz,
+ const NdbDictionary::Table * theTable,
+ struct DBA__Array * keys,
+ struct DBA__Array * columns);
+
+extern "C"
+DBA_Binding_t *
+DBA_CreateBinding( const char* TableName,
+ int NbCol,
+ const DBA_ColumnBinding_t ColsBindings[],
+ Size_t StructSz ){
+
+ NdbDictionary::Dictionary * dict = DBA__TheNdb->getDictionary();
+ if(dict == 0){
+ DBA__SetLatestError(DBA_NDB_ERROR, 0,
+ "Internal NDB error: No dictionary");
+ return 0;
+ }
+
+ const NdbDictionary::Table * table = dict->getTable(TableName);
+ if(table == 0){
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0,
+ "No such table: %s", TableName);
+ return 0;
+ }
+
+ /**
+ * Keys/Columns in table
+ */
+ const int tabColumns = table->getNoOfColumns();
+ const int tabKeys = table->getNoOfPrimaryKeys();
+
+ /**
+ * Ok, ok... I alloc four bytes extra so what...
+ */
+ struct DBA__Array * keys = (struct DBA__Array *)malloc
+ (sizeof(struct DBA__Array)+tabKeys*sizeof(int));
+
+ if(keys == 0){
+ DBA__SetLatestError(DBA_ERROR, 0,
+ "malloc(%d) failed",
+ sizeof(struct DBA__Array)+tabKeys*sizeof(int));
+ return 0;
+ }
+
+ struct DBA__Array * columns = (struct DBA__Array *)malloc
+ (sizeof(struct DBA__Array)+tabColumns*sizeof(int));
+
+ if(columns == 0){
+ DBA__SetLatestError(DBA_ERROR, 0,
+ "malloc(%d) failed",
+ sizeof(struct DBA__Array)+tabColumns*sizeof(int));
+ free(keys);
+ return 0;
+ }
+
+ columns->count = 0;
+ keys->count = 0;
+
+ DBA_Binding_t * bindings = createBinding(TableName,
+ NbCol,
+ ColsBindings,
+ StructSz,
+ table,
+ keys,
+ columns);
+
+ for(int i = 0; i<tabColumns; i++){
+ const NdbDictionary::Column * col = table->getColumn(i);
+ if(col->getPrimaryKey()){
+ if(!keys->exists(i)){
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0,
+ "Key column: %s not specified in binding",
+ col->getName());
+
+ free(keys); free(columns);
+ DBA_DestroyBinding(bindings);
+ return 0;
+ }
+ }
+ }
+
+ free(keys); free(columns);
+
+ DBA__ValidBinding(bindings);
+
+ return bindings;
+}
+
+DBA_Binding_t *
+createBinding(const char* TableName,
+ int NbCol,
+ const DBA_ColumnBinding_t ColsBindings[],
+ Size_t StructSz,
+ const NdbDictionary::Table * table,
+ struct DBA__Array * keys,
+ struct DBA__Array * columns){
+ /**
+ * Counters for this part of binding
+ */
+ int noOfKeys = 0;
+ int noOfColumns = 0;
+ int noOfSubBindings = 0;
+
+ /**
+ * Check names and types and sizes
+ */
+ for(int i = 0; i<NbCol; i++){
+ if(ColsBindings[i].Ptr){
+ /**
+ * Pointer binding
+ */
+ noOfSubBindings ++;
+
+ DBA_Binding_t * tmp = createBinding(TableName,
+ ColsBindings[i].Size,
+ ColsBindings[i].SubBinding,
+ StructSz,
+ table,
+ keys,
+ columns);
+ DBA__ValidBinding(tmp);
+
+ if(tmp == 0){
+ // createBindings have already set latestError
+ return 0;
+ }
+
+ DBA_DestroyBinding(tmp);
+ } else {
+ const NdbDictionary::Column * col =
+ table->getColumn(ColsBindings[i].Name);
+ const Uint32 attrId = col->getColumnNo();
+
+ if(col == 0){
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0,
+ "Unknown column: %s", ColsBindings[i].Name);
+ return 0;
+ }
+ const NdbDictionary::Column::Type type = col->getType();
+ if(!matchType(type, ColsBindings[i].DataType)){
+ DBA_DEBUG("Incorrect type for: " << ColsBindings[i].Name);
+ DBA_DEBUG("type: " << type);
+ DBA_DEBUG("ColsBindings[i].DataType: " << ColsBindings[i].DataType);
+
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0,
+ "Incorrect type for column: %s",
+ ColsBindings[i].Name);
+
+ return 0;
+ }
+
+ if(!matchSize(type, col->getLength(), ColsBindings[i].Size)){
+ DBA_DEBUG("Incorrect size for: " << ColsBindings[i].Name);
+ DBA_DEBUG("type: " << type);
+ DBA_DEBUG("length: " << col->getLength());
+ DBA_DEBUG("ColsBindings[i].Size" << (Uint64)ColsBindings[i].Size);
+
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0,
+ "Incorrect size for column: %s",
+ ColsBindings[i].Name);
+ return 0;
+ }
+
+ if(col->getPrimaryKey()){
+ noOfKeys++;
+ } else {
+ noOfColumns++;
+ }
+
+ /**
+ * Check only in "validate" phase
+ */
+ if(columns != 0 && keys != 0){
+ if(columns->exists(attrId) || keys->exists(attrId)){
+ DBA_DEBUG("Column bound multiple times: " << ColsBindings[i].Name);
+
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0,
+ "Column bound multiple times: %s",
+ ColsBindings[i].Name);
+ return 0;
+ }
+
+ if(col->getPrimaryKey()){
+ keys->insert(attrId);
+ } else {
+ columns->insert(attrId);
+ }
+ }
+ }
+ }
+
+ /**
+ * Validation is all set
+ */
+
+ /**
+ * Allocate memory
+ */
+ const int szOfStruct =
+ sizeof(DBA_Binding_t)
+ + strlen(TableName) + 4
+ + (2 * sizeof(int) * noOfKeys)
+ + (2 * sizeof(int) * noOfColumns)
+ + ((sizeof(struct DBA_Binding *) + sizeof(int)) * noOfSubBindings)
+ - 4;
+
+ DBA_Binding * ret = (DBA_Binding *)malloc(szOfStruct);
+ if(ret == 0){
+ DBA__SetLatestError(DBA_ERROR, 0,
+ "malloc(%d) failed", szOfStruct);
+ return 0;
+ }
+
+ for(int i = 0; i<DBA__MagicLength; i++)
+ ret->magic[i] = DBA__TheMagic[i];
+
+ ret->noOfKeys = noOfKeys;
+ ret->noOfColumns = noOfColumns;
+ ret->noOfSubBindings = noOfSubBindings;
+
+ ret->keyIds = (int *)&(ret->data[0]);
+ ret->keyOffsets = ret->keyIds + noOfKeys;
+
+ ret->columnIds = ret->keyOffsets + noOfKeys;
+ ret->columnOffsets = ret->columnIds + noOfColumns;
+
+ ret->subBindingOffsets = ret->columnOffsets + noOfColumns;
+ ret->subBindings = (DBA_Binding **)
+ (ret->subBindingOffsets + noOfSubBindings);
+
+ ret->tableName = (char *)(ret->subBindings + noOfSubBindings);
+ ret->structSz = StructSz;
+ ret->checkSum = computeChecksum(ret);
+
+ /**
+ * Populate arrays
+ */
+ strcpy(ret->tableName, TableName);
+
+ int k = 0;
+ int c = 0;
+ int p = 0;
+
+ for(int i = 0; i<NbCol; i++){
+ if(ColsBindings[i].Ptr){
+ ret->subBindings[p] = createBinding(TableName,
+ ColsBindings[i].Size,
+ ColsBindings[i].SubBinding,
+ StructSz,
+ table,
+ 0,
+ 0);
+
+ DBA__ValidBinding(ret->subBindings[p]);
+
+ ret->subBindingOffsets[p] = ColsBindings[i].Offset;
+ p++;
+ } else {
+ const NdbDictionary::Column * col =
+ table->getColumn(ColsBindings[i].Name);
+
+ if(col->getPrimaryKey()){
+ ret->keyIds[k] = col->getColumnNo();
+ ret->keyOffsets[k] = ColsBindings[i].Offset;
+ k++;
+ } else {
+ ret->columnIds[c] = col->getColumnNo();
+ ret->columnOffsets[c] = ColsBindings[i].Offset;
+ c++;
+ }
+ }
+ }
+
+ return ret;
+}
+
+
+extern "C"
+DBA_Error_t
+DBA_DestroyBinding( DBA_Binding_t* Binding ){
+
+ for(int i = 0; i<Binding->noOfSubBindings; i++)
+ DBA_DestroyBinding(Binding->subBindings[i]);
+
+ free(Binding);
+
+ return DBA_NO_ERROR;
+}
+
+static
+bool
+matchType(NdbDictionary::Column::Type t1, DBA_DataTypes_t t2){
+ for(int i = 0; i<DBA__NoOfMappings; i++)
+ if(DBA__DataTypesMappings[i].newtonType == t2 &&
+ DBA__DataTypesMappings[i].ndbType == t1)
+ return true;
+ return false;
+}
+
+static
+bool
+matchSize(NdbDictionary::Column::Type t, unsigned b, Size_t s) {
+ switch(t){
+ case NdbDictionary::Column::Int:
+ case NdbDictionary::Column::Unsigned:
+ case NdbDictionary::Column::Float:
+ return (4 * b) == s;
+ case NdbDictionary::Column::Bigint:
+ case NdbDictionary::Column::Bigunsigned:
+ case NdbDictionary::Column::Double:
+ return (8 * b) == s;
+ case NdbDictionary::Column::Decimal:
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Binary:
+ return (1 * b) == s;
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Varbinary:
+ case NdbDictionary::Column::Datetime:
+ case NdbDictionary::Column::Timespec:
+ case NdbDictionary::Column::Blob:
+ case NdbDictionary::Column::Undefined:
+ return false;
+ }
+ return false;
+}
+
+bool
+DBA__ValidBinding(const DBA_Binding_t * bindings){
+ if(bindings == 0){
+ DBA_DEBUG("Null pointer passed to validBinding");
+ return false;
+ }
+
+ for(int i = 0; i<DBA__MagicLength; i++)
+ if(bindings->magic[i] != DBA__TheMagic[i]){
+ DBA_DEBUG("Invalid magic in validBinding");
+ return false;
+ }
+
+ const int cs = computeChecksum(bindings);
+ if(cs != bindings->checkSum){
+ DBA_DEBUG("Invalid checksum in validBinding");
+ DBA_DEBUG("cs = " << cs << " b->cs= " << bindings->checkSum);
+ return false;
+ }
+
+ return true;
+}
+
+bool
+DBA__ValidBindings(const DBA_Binding_t * const * pBindings, int n){
+ for(int i = 0; i<n; i++)
+ if(!DBA__ValidBinding(pBindings[i]))
+ return false;
+ return true;
+}
+
+/**
+ * Note: currently only checksum "static" part of struct
+ */
+static
+int
+computeChecksum(const DBA_Binding_t * bindings){
+ int sum = 0;
+ int pos = 0;
+ const char * ptr = ((const char *)bindings)+DBA__MagicLength+sizeof(int);
+ const int sz = sizeof(DBA_Binding_t) - DBA__MagicLength - sizeof(int) - 4;
+
+ for(int i = 0; i<sz; i++){
+ sum += ((int)ptr[i]) << pos;
+ pos += 8;
+ if(pos == 32)
+ pos = 0;
+ }
+
+ return sum;
+}
+
+int
+DBA__GetStructSize(const DBA_Binding_t * bind){
+ if(!DBA__ValidBinding(bind))
+ return 0;
+ return bind->structSz;
+}
diff --git a/ndb/src/newtonapi/dba_bulkread.cpp b/ndb/src/newtonapi/dba_bulkread.cpp
new file mode 100644
index 00000000000..1f75037046b
--- /dev/null
+++ b/ndb/src/newtonapi/dba_bulkread.cpp
@@ -0,0 +1,267 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "dba_internal.hpp"
+
+struct DBA__BulkReadData {
+ const DBA_Binding_t * const * pBindings; // The bindings
+ DBA_BulkReadResultSet_t * pData; // The data
+
+ int NbRows; // NbRows per binding
+ int NbBindings; // NbBindings
+ int TotalRows; // Total rows (NbRows*NbBindings)
+
+ DBA_AsyncCallbackFn_t CbFunc; // Users callback
+ DBA_ReqId_t RequestId; // Users request id
+ DBA_Error_t Status; // Request status
+ DBA_ErrorCode_t ErrorCode; /**< Request error
+ Only valid if request is
+ aborted */
+
+ int RowsSubmitted; // No of read sent to NDB
+ int RowsAcknowledged; // No of read responses
+ int OpPerTrans; // Operations per transaction
+
+ struct Index {
+ int binding;
+ int row;
+ int datarow;
+
+ void init() { row = binding = datarow = 0;}
+ void next(int rows) {
+ datarow++; row++;
+ if(row == rows){ row = 0; binding++; }
+ }
+ };
+ Index lastSend;
+ Index nextSend;
+
+ /**
+ * If "simple" bulkread
+ * use this storage
+ */
+ const DBA_Binding_t * bindings[1];
+
+ DBA__BulkReadData() {
+ RequestId = DBA_INVALID_REQID;
+ }
+ void ProcessBulkRead();
+ bool ProcessCallback(int errorCode, NdbConnection * connection);
+};
+
+static
+void
+NewtonCallback(int errorCode,
+ NdbConnection * connection,
+ void * anyObject){
+
+ DBA__BulkReadData * brd = (DBA__BulkReadData*)anyObject;
+
+ brd->ProcessCallback(errorCode, connection);
+
+ DBA__TheNdb->closeTransaction(connection);
+
+ if(brd->RowsSubmitted == brd->TotalRows){
+
+ /**
+ * The entire bulk read is finished,
+ * call users callback
+ */
+ DBA_ReqId_t reqId = brd->RequestId;
+
+ // Invalidate BulkReadData
+ brd->RequestId = DBA_INVALID_REQID;
+
+ brd->CbFunc(reqId, brd->Status, brd->ErrorCode);
+ return;
+ }
+
+ brd->ProcessBulkRead();
+}
+
+/**
+ * A BulkReadData structure
+ */
+static DBA__BulkReadData theBRD;
+
+#define CHECK_BINDINGS(Bindings) \
+ if(!DBA__ValidBinding(Bindings)){ \
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0, "Invalid bindings"); \
+ return DBA_INVALID_REQID; \
+ }
+
+#define CHECK_BINDINGS2(Bindings, NbBindings) \
+ if(!DBA__ValidBindings(Bindings, NbBindings)){ \
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0, "Invalid bindings"); \
+ return DBA_INVALID_REQID; \
+ }
+
+DBA_ReqId_t
+DBA_BulkReadRows(const DBA_Binding_t * pBindings,
+ DBA_BulkReadResultSet_t pData[],
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ){
+
+ CHECK_BINDINGS(pBindings);
+
+ DBA__BulkReadData * brd = &theBRD;
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+
+ if(brd->RequestId != DBA_INVALID_REQID){
+ DBA__SetLatestError(DBA_ERROR, 0,
+ "DBA only permits 1 concurrent bulkread");
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return DBA_ERROR;
+ }
+
+ theBRD.RequestId = 1;
+
+ /**
+ *
+ */
+ brd->bindings[0] = pBindings;
+ brd->pBindings = brd->bindings;
+ brd->pData = pData;
+
+ /**
+ * Control data
+ */
+ brd->NbRows = NbRows;
+ brd->NbBindings = 1;
+ brd->TotalRows = NbRows;
+ brd->CbFunc = CbFunc;
+ brd->Status = DBA_NO_ERROR;
+ brd->ErrorCode = 0;
+ brd->OpPerTrans = DBA__BulkReadCount;
+
+ brd->RowsSubmitted = 0;
+ brd->RowsAcknowledged = 0;
+
+ brd->lastSend.init();
+ brd->nextSend.init();
+
+ brd->ProcessBulkRead();
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+
+ return brd->RequestId;
+}
+
+DBA_ReqId_t
+DBA_BulkMultiReadRows(const DBA_Binding_t * const * pBindings,
+ DBA_BulkReadResultSet_t pData[],
+ int NbBindings,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ){
+
+ CHECK_BINDINGS2(pBindings, NbBindings);
+
+ DBA__BulkReadData * brd = &theBRD;
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+
+ if(brd->RequestId != DBA_INVALID_REQID){
+ DBA__SetLatestError(DBA_ERROR, 0,
+ "DBA only permits 1 concurrent bulkread");
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return DBA_ERROR;
+ }
+
+ brd->RequestId = 1;
+
+ /**
+ *
+ */
+ brd->pBindings = pBindings;
+ brd->pData = pData;
+
+ /**
+ * Control data
+ */
+ brd->NbRows = NbRows;
+ brd->NbBindings = NbBindings;
+ brd->TotalRows = (NbRows * NbBindings);
+ brd->CbFunc = CbFunc;
+ brd->Status = DBA_NO_ERROR;
+ brd->ErrorCode = 0;
+ brd->OpPerTrans = DBA__BulkReadCount;
+
+ brd->RowsSubmitted = 0;
+ brd->RowsAcknowledged = 0;
+
+ brd->lastSend.init();
+ brd->nextSend.init();
+
+ brd->ProcessBulkRead();
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+
+ return brd->RequestId;
+}
+
+bool
+DBA__BulkReadData::ProcessCallback(int errorCode, NdbConnection * con){
+
+ Index tmp = lastSend;
+ const NdbOperation * op = con->getNextCompletedOperation(0);
+
+ for(int i = 0; i<OpPerTrans && RowsAcknowledged < RowsSubmitted; i++){
+ require(op != 0);
+ if(op->getNdbError().code == 0)
+ pData[tmp.datarow].RowFoundIndicator = 1;
+ else
+ pData[tmp.datarow].RowFoundIndicator = 0;
+
+ RowsAcknowledged++;
+ tmp.next(NbRows);
+ op = con->getNextCompletedOperation(op);
+ }
+ return true;
+}
+
+void
+DBA__BulkReadData::ProcessBulkRead(){
+
+ NdbConnection * con = DBA__TheNdb->startTransaction();
+
+ Index tmp = nextSend;
+
+ for(int i = 0; i<OpPerTrans && RowsSubmitted < TotalRows; i++){
+
+ const DBA_Binding_t * binding = pBindings[tmp.binding];
+ void * data = pData[tmp.datarow].DataPtr;
+
+ NdbOperation * op = con->getNdbOperation(binding->tableName);
+
+ op->simpleRead();
+
+ require(DBA__EqualGetValue(op, binding, data));
+
+ RowsSubmitted++;
+ tmp.next(NbRows);
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)this,
+ CommitAsMuchAsPossible);
+
+ lastSend = nextSend;
+ nextSend = tmp;
+}
diff --git a/ndb/src/newtonapi/dba_config.cpp b/ndb/src/newtonapi/dba_config.cpp
new file mode 100644
index 00000000000..d84386a9438
--- /dev/null
+++ b/ndb/src/newtonapi/dba_config.cpp
@@ -0,0 +1,115 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "dba_internal.hpp"
+
+int DBA__NBP_Intervall = 10;
+int DBA__BulkReadCount = 1000;
+int DBA__StartTransactionTimout = 0;
+int DBA__NBP_Force = 1;
+
+struct DBA__Config {
+ int ParamId;
+ int * Param;
+ int min;
+ int max;
+ const char * Description;
+};
+
+static
+DBA__Config Parameters[] = {
+ { 0, &DBA__NBP_Intervall, 4, INT_MAX,
+ "Newton Batch Process Interval(ms)" },
+ { 1, &DBA__BulkReadCount, 1, 5000,
+ "Operations per transaction during bulkread" },
+ { 2, &DBA__StartTransactionTimout, 0, INT_MAX,
+ "Start transaction timeout(ms)" },
+ { 3, &DBA__NBP_Force, 0, 2,
+ "Newton Batch Process Force send algorithm" }
+};
+
+static const int Params = sizeof(Parameters)/sizeof(DBA__Config);
+
+static
+DBA__Config *
+getParam(int id){
+ for(int i = 0; i<Params; i++)
+ if(Parameters[i].ParamId == id)
+ return &Parameters[i];
+ return 0;
+}
+
+
+extern "C"
+DBA_Error_t
+DBA_SetParameter(int ParameterId, int Value){
+ if(Value == -1){
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0, "Node id is not modifyable");
+ return DBA_APPLICATION_ERROR;
+ }
+
+ DBA__Config * p = getParam(ParameterId);
+
+ if(p == 0){
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0, "Invalid parameter id: %d",
+ ParameterId);
+ return DBA_APPLICATION_ERROR;
+ }
+
+ if(Value < p->min){
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0,
+ "Value too small for parameter %d (min = %d)",
+ Value, p->min);
+ return DBA_APPLICATION_ERROR;
+ }
+
+ if(Value > p->max){
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0,
+ "Value too big for parameter %d (max = %d)",
+ Value, p->max);
+ return DBA_APPLICATION_ERROR;
+ }
+
+ * p->Param = Value;
+ return DBA_NO_ERROR;
+}
+
+extern "C"
+DBA_Error_t
+DBA_GetParameter(int ParameterId, int * Value){
+ if(ParameterId == -1){
+ if(DBA__TheNdb == 0){
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0, "DBA_Open() is not called"
+ );
+ return DBA_APPLICATION_ERROR;
+ }
+ * Value = DBA__TheNdb->getNodeId();
+ return DBA_NO_ERROR;
+ }
+
+ DBA__Config * p = getParam(ParameterId);
+ if(p == 0){
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0, "Invalid parameter id: %d",
+ ParameterId);
+ return DBA_APPLICATION_ERROR;
+ }
+
+ * Value = * p->Param;
+
+ return DBA_NO_ERROR;
+}
+
diff --git a/ndb/src/newtonapi/dba_dac.cpp b/ndb/src/newtonapi/dba_dac.cpp
new file mode 100644
index 00000000000..fcb4e676e46
--- /dev/null
+++ b/ndb/src/newtonapi/dba_dac.cpp
@@ -0,0 +1,842 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "dba_internal.hpp"
+#include <NdbSleep.h>
+
+static void
+DBA__ErrorMapping(const NdbError & err, DBA_Error_t * status){
+ switch(err.classification){
+ case NdbError::ConstraintViolation:
+ * status = DBA_CONSTRAINT_VIOLATION;
+ break;
+
+ case NdbError::NoDataFound:
+ * status = DBA_NO_DATA;
+ break;
+
+ case NdbError::TemporaryResourceError:
+ case NdbError::NodeRecoveryError:
+ * status = DBA_TEMPORARY_ERROR;
+ break;
+
+ case NdbError::InsufficientSpace:
+ * status = DBA_INSUFFICIENT_SPACE;
+ break;
+
+ case NdbError::UnknownResultError:
+ * status = DBA_UNKNOWN_RESULT;
+ break;
+
+ case NdbError::OverloadError:
+ * status = DBA_OVERLOAD;
+ break;
+
+ case NdbError::TimeoutExpired:
+ * status = DBA_TIMEOUT;
+ break;
+
+ case NdbError::SchemaError:
+ * status = DBA_SCHEMA_ERROR;
+ break;
+
+ case NdbError::ApplicationError:
+ * status = DBA_APPLICATION_ERROR;
+ break;
+
+ case NdbError::InternalError:
+ default:
+ * status = DBA_NDB_ERROR;
+ break;
+ }
+}
+
+/**
+ * Map between NDB error codes and DBA error codes
+ */
+static
+void
+DBA__CallbackErrorCodeMapping(int errorCode,
+ NdbConnection * connection,
+ DBA_Error_t * status,
+ DBA_ErrorCode_t * errCode) {
+ if(errorCode == 0){
+ * status = DBA_NO_ERROR;
+ * errCode = 0;
+ return;
+ }
+ const NdbError & err = connection->getNdbError();
+ DBA__ErrorMapping(err, status);
+ * errCode = err.code;
+}
+
+/**
+ * When startTransaction fails
+ */
+static
+void
+DBA__ConnectionErrorMapping(Ndb * theNdb){
+ const NdbError & err = theNdb->getNdbError();
+
+ DBA_Error_t status;
+ DBA__ErrorMapping(err, &status);
+
+ DBA__SetLatestError(status, err.code,
+ err.message);
+}
+
+/**
+ * When getNdbOperation fails
+ */
+static
+void
+DBA__OperationErrorMapping(Ndb * theNdb, NdbConnection * con){
+ const NdbError & err = theNdb->getNdbError();
+
+ DBA_Error_t status;
+ DBA__ErrorMapping(err, &status);
+
+ DBA__SetLatestError(status, err.code,
+ err.message);
+}
+
+/**
+ * When equal/get/set value fails
+ */
+static
+void
+DBA__EqualErrorMapping(Ndb * theNdb, NdbConnection * con, NdbOperation * op){
+ const NdbError & err = theNdb->getNdbError();
+
+ DBA_Error_t status;
+ DBA__ErrorMapping(err, &status);
+
+ DBA__SetLatestError(status, err.code,
+ err.message);
+}
+
+static
+void
+NewtonCallback(int errorCode,
+ NdbConnection * connection,
+ void * anyObject){
+
+ DBA_AsyncCallbackFn_t CbFunc = (DBA_AsyncCallbackFn_t)anyObject;
+ DBA_ReqId_t ReqId = (DBA_ReqId_t) connection;
+
+ DBA_Error_t Status = (DBA_Error_t) errorCode;
+ DBA_ErrorCode_t Impl_Status ;
+
+ DBA__CallbackErrorCodeMapping(errorCode, connection, &Status, &Impl_Status);
+
+ DBA__TheNdb->closeTransaction(connection);
+
+ DBA__RecvTransactions++;
+
+ CbFunc(ReqId, Status, Impl_Status);
+}
+
+/**
+ * Start transaction
+ */
+NdbConnection *
+startTransaction(){
+ NdbConnection * con = DBA__TheNdb->startTransaction();
+ if(con != 0)
+ return con;
+
+ const int _t = (DBA__SentTransactions - DBA__RecvTransactions);
+ const int t = (_t>0?_t:-_t);
+
+ if(!(DBA__TheNdb->getNdbError().code == 4006 && t > 1000)){
+ DBA_DEBUG("DBA__TheNdb->getNdbError() = " <<
+ DBA__TheNdb->getNdbError());
+ }
+
+ int sum = 0;
+ int sleepTime = 10;
+ for(; con == 0 && sum < DBA__StartTransactionTimout; ){
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ NdbSleep_MilliSleep(sleepTime);
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ con = DBA__TheNdb->startTransaction();
+
+ sum += sleepTime;
+ sleepTime += 10;
+ }
+
+ return con;
+}
+
+#define CHECK_BINDINGS(Bindings) \
+ if(!DBA__ValidBinding(Bindings)){ \
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0, "Invalid bindings"); \
+ return DBA_INVALID_REQID; \
+ }
+
+#define CHECK_BINDINGS2(Bindings, NbBindings) \
+ if(!DBA__ValidBindings(Bindings, NbBindings)){ \
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0, "Invalid bindings"); \
+ return DBA_INVALID_REQID; \
+ }
+
+#define CHECK_CONNECTION(Connection) \
+ if(Connection == 0){ \
+ DBA__ConnectionErrorMapping(DBA__TheNdb); \
+ NdbMutex_Unlock(DBA__TheNewtonMutex); \
+ return DBA_INVALID_REQID; \
+ }
+
+#define CHECK_OPERATION(Connection, Operation) \
+ if(Operation == 0){ \
+ DBA__OperationErrorMapping(DBA__TheNdb, Connection); \
+ DBA__TheNdb->closeTransaction(Connection); \
+ NdbMutex_Unlock(DBA__TheNewtonMutex); \
+ return DBA_INVALID_REQID; \
+ }
+
+#define EQUAL_ERROR(Connection, Operation) { \
+ DBA__EqualErrorMapping(DBA__TheNdb, Connection, Operation); \
+ DBA__TheNdb->closeTransaction(Connection); \
+ NdbMutex_Unlock(DBA__TheNewtonMutex); \
+ return DBA_INVALID_REQID; \
+ }
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+extern "C"
+DBA_ReqId_t
+DBA_ReadRows( const DBA_Binding_t* pBindings, void* const * _pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->simpleRead();
+
+ void * pData = _pData[i];
+
+ if(!DBA__EqualGetValue(op, pBindings, pData)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_ArrayReadRows( const DBA_Binding_t* pBindings, void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ){
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->simpleRead();
+
+ if(!DBA__EqualGetValue(op, pBindings,
+ ((char*)pData)+i*pBindings->structSz)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_MultiReadRow(const DBA_Binding_t * const * pBindings,
+ void * const * pData,
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+ CHECK_BINDINGS2(pBindings, NbBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbBindings; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings[i]->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->simpleRead();
+
+ if(!DBA__EqualGetValue(op, pBindings[i], pData[i])){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+extern "C"
+DBA_ReqId_t
+DBA_InsertRows( const DBA_Binding_t* pBindings, const void * const * _pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->insertTuple();
+
+ const void * pData = _pData[i];
+
+ if(!DBA__EqualSetValue(op, pBindings, pData)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_ArrayInsertRows( const DBA_Binding_t* pBindings, const void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ){
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->insertTuple();
+
+ if(!DBA__EqualSetValue(op, pBindings,
+ ((char*)pData)+i*pBindings->structSz)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_MultiInsertRow(const DBA_Binding_t * const * pBindings,
+ const void * const * pData,
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+ CHECK_BINDINGS2(pBindings, NbBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbBindings; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings[i]->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->insertTuple();
+
+ if(!DBA__EqualSetValue(op, pBindings[i], pData[i])){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+extern "C"
+DBA_ReqId_t
+DBA_UpdateRows( const DBA_Binding_t* pBindings, const void * const * _pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->updateTuple();
+
+ const void * pData = _pData[i];
+
+ if(!DBA__EqualSetValue(op, pBindings, pData)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_ArrayUpdateRows( const DBA_Binding_t* pBindings, const void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ){
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->updateTuple();
+
+ if(!DBA__EqualSetValue(op, pBindings,
+ ((char*)pData)+i*pBindings->structSz)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_MultiUpdateRow(const DBA_Binding_t * const * pBindings,
+ const void * const * pData,
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+ CHECK_BINDINGS2(pBindings, NbBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbBindings; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings[i]->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->updateTuple();
+
+ if(!DBA__EqualSetValue(op, pBindings[i], pData[i])){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+extern "C"
+DBA_ReqId_t
+DBA_WriteRows( const DBA_Binding_t* pBindings, const void * const * _pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->writeTuple();
+
+ const void * pData = _pData[i];
+
+ if(!DBA__EqualSetValue(op, pBindings, pData)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_ArrayWriteRows( const DBA_Binding_t* pBindings, const void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ){
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->writeTuple();
+
+ if(!DBA__EqualSetValue(op, pBindings,
+ ((char*)pData)+i*pBindings->structSz)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_MultiWriteRow(const DBA_Binding_t * const * pBindings,
+ const void * const * pData,
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+ CHECK_BINDINGS2(pBindings, NbBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbBindings; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings[i]->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->writeTuple();
+
+ if(!DBA__EqualSetValue(op, pBindings[i], pData[i])){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+extern "C"
+DBA_ReqId_t
+DBA_DeleteRows( const DBA_Binding_t* pBindings, const void * const * _pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->deleteTuple();
+
+ const void * pData = _pData[i];
+
+ if(!DBA__Equal(op, pBindings, pData)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_ArrayDeleteRows( const DBA_Binding_t* pBindings, const void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc ){
+ CHECK_BINDINGS(pBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbRows; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->deleteTuple();
+
+ if(!DBA__Equal(op, pBindings,
+ ((char*)pData)+i*pBindings->structSz)){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+extern "C"
+DBA_ReqId_t
+DBA_MultiDeleteRow(const DBA_Binding_t * const * pBindings,
+ const void * const * pData,
+ int NbBindings,
+ DBA_AsyncCallbackFn_t CbFunc ) {
+ CHECK_BINDINGS2(pBindings, NbBindings);
+
+ NdbMutex_Lock(DBA__TheNewtonMutex);
+ NdbConnection * con = startTransaction();
+
+ CHECK_CONNECTION(con);
+
+ for(int i = 0; i<NbBindings; i++){
+ NdbOperation * op = con->getNdbOperation(pBindings[i]->tableName);
+
+ CHECK_OPERATION(con, op);
+
+ op->deleteTuple();
+
+ if(!DBA__Equal(op, pBindings[i], pData[i])){
+ EQUAL_ERROR(con, op);
+ }
+ }
+
+ con->executeAsynchPrepare(Commit,
+ NewtonCallback,
+ (void*)CbFunc);
+
+ DBA__SentTransactions++;
+
+ NdbMutex_Unlock(DBA__TheNewtonMutex);
+ return (DBA_ReqId_t) con;
+}
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+bool
+DBA__EqualGetValue(NdbOperation * op,
+ const DBA_Binding_t* pBindings,
+ void * pData){
+ for(int i = 0; i<pBindings->noOfKeys; i++){
+ if(op->equal(pBindings->keyIds[i],
+ (const char*)pData+pBindings->keyOffsets[i]) == -1){
+ return false;
+ }
+ }
+
+ for(int i = 0; i<pBindings->noOfColumns; i++){
+ if(op->getValue(pBindings->columnIds[i],
+ (char*)pData+pBindings->columnOffsets[i]) == 0){
+ return false;
+ }
+ }
+
+ for(int i = 0; i<pBindings->noOfSubBindings; i++){
+ void * tData = *(void**)((char *)pData+pBindings->subBindingOffsets[i]);
+ const DBA_Binding_t * tBinding = pBindings->subBindings[i];
+ if(!DBA__EqualGetValue(op, tBinding, tData))
+ return false;
+ }
+
+ return true;
+}
+
+bool
+DBA__EqualSetValue(NdbOperation * op,
+ const DBA_Binding_t* pBindings,
+ const void * pData){
+
+ for(int i = 0; i<pBindings->noOfKeys; i++){
+ if(op->equal(pBindings->keyIds[i],
+ (const char*)pData+pBindings->keyOffsets[i]) == -1){
+ return false;
+ }
+ }
+
+ for(int i = 0; i<pBindings->noOfColumns; i++){
+ if(op->setValue(pBindings->columnIds[i],
+ (char*)pData+pBindings->columnOffsets[i]) == -1){
+ return false;
+ }
+ }
+
+ for(int i = 0; i<pBindings->noOfSubBindings; i++){
+ void * tData = * (void**)((char *)pData+pBindings->subBindingOffsets[i]);
+ const DBA_Binding_t * tBinding = pBindings->subBindings[i];
+ if(!DBA__EqualSetValue(op, tBinding, tData))
+ return false;
+ }
+
+ return true;
+}
+
+bool
+DBA__Equal(NdbOperation * op,
+ const DBA_Binding_t* pBindings,
+ const void * pData){
+
+ for(int i = 0; i<pBindings->noOfKeys; i++)
+ if(op->equal(pBindings->keyIds[i],
+ (const char*)pData+pBindings->keyOffsets[i]) == -1){
+ return false;
+ }
+
+ for(int i = 0; i<pBindings->noOfSubBindings; i++){
+ void * tData = *(void**)((char *)pData+pBindings->subBindingOffsets[i]);
+ const DBA_Binding_t * tBinding = pBindings->subBindings[i];
+ if(!DBA__Equal(op, tBinding, tData))
+ return false;
+ }
+
+ return true;
+}
+
diff --git a/ndb/src/newtonapi/dba_error.cpp b/ndb/src/newtonapi/dba_error.cpp
new file mode 100644
index 00000000000..0a154ac1314
--- /dev/null
+++ b/ndb/src/newtonapi/dba_error.cpp
@@ -0,0 +1,118 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "dba_internal.hpp"
+#include <NdbStdio.h>
+#include <stdarg.h>
+
+static DBA_Error_t latestError = DBA_NO_ERROR;
+static DBA_ErrorCode_t latestNdbError = 0;
+static char latestMsg[1024];
+
+/**
+ * Private
+ */
+void
+DBA__SetLatestError(DBA_Error_t le,
+ DBA_ErrorCode_t lnb,
+ const char * msg, ...){
+
+ require(msg != 0);
+
+ latestError = le;
+ latestNdbError = lnb;
+
+ va_list ap;
+
+ va_start(ap, msg);
+ vsnprintf(latestMsg, sizeof(latestMsg)-1, msg, ap);
+ va_end(ap);
+}
+
+/**
+ * Get latest DBA error
+ */
+extern "C"
+DBA_Error_t
+DBA_GetLatestError(){
+ return latestError;
+}
+
+/**
+ * Get latest error string associated with GetLatestError
+ *
+ * String must not be free by caller of this method
+ */
+extern "C"
+const char *
+DBA_GetLatestErrorMsg(){
+ return latestMsg;
+}
+
+/**
+ * Get the latest NDB error
+ *
+ * Note only applicable to synchronous methods
+ */
+extern "C"
+DBA_ErrorCode_t
+DBA_GetLatestNdbError(){
+ return latestNdbError;
+}
+
+extern "C"
+const
+char *
+DBA_GetNdbErrorMsg(DBA_ErrorCode_t code){
+ return DBA__TheNdb->getNdbError(code).message;
+}
+
+struct DBA_ErrorTxtMap {
+ DBA_Error_t Error;
+ const char * Msg;
+};
+
+static
+const DBA_ErrorTxtMap errMap[] = {
+ { DBA_NO_ERROR, "No error" },
+ { DBA_NOT_IMPLEMENTED, "Function Not Implemented" },
+ { DBA_NDB_ERROR, "Uncategorised NDB error" },
+ { DBA_ERROR, "Uncategorised DBA implementation error" },
+ { DBA_APPLICATION_ERROR,
+ "Function called with invalid argument(s)/invalid sequence(s)" },
+ { DBA_NO_DATA, "No row with specified PK existed" },
+ { DBA_CONSTRAINT_VIOLATION, "There already existed a row with that PK" },
+
+ { DBA_TEMPORARY_ERROR, "Request failed due to temporary reasons" },
+ { DBA_INSUFFICIENT_SPACE,
+ "The DB is full" },
+ { DBA_OVERLOAD, "Request was rejected in NDB due to high load situation" },
+ { DBA_TIMEOUT, "The request timed out, probably due to dead-lock" }
+};
+
+static const int ErrMsgs = sizeof(errMap)/sizeof(DBA_ErrorTxtMap);
+
+extern "C"
+const
+char *
+DBA_GetErrorMsg(DBA_Error_t e){
+ for(int i = 0; i<ErrMsgs; i++)
+ if(errMap[i].Error == e)
+ return errMap[i].Msg;
+ return "Invalid error code";
+}
+
diff --git a/ndb/src/newtonapi/dba_init.cpp b/ndb/src/newtonapi/dba_init.cpp
new file mode 100644
index 00000000000..aa5fef1171c
--- /dev/null
+++ b/ndb/src/newtonapi/dba_init.cpp
@@ -0,0 +1,86 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "dba_internal.hpp"
+#include "dba_process.hpp"
+#include <NdbOut.hpp>
+
+
+#ifdef NDB_WIN32
+static NdbMutex & DBA__InitMutex = * NdbMutex_Create();
+#else
+static NdbMutex DBA__InitMutex = NDB_MUTEX_INITIALIZER;
+#endif
+
+Ndb * DBA__TheNdb = 0;
+NdbMutex * DBA__TheNewtonMutex = 0;
+unsigned DBA__SentTransactions = 0;
+unsigned DBA__RecvTransactions = 0;
+NewtonBatchProcess * DBA__TheNBP = 0;
+
+extern "C"
+DBA_Error_t
+DBA_Open( ) {
+ NdbMutex_Lock(&DBA__InitMutex);
+
+ if(DBA__TheNdb != 0){
+ NdbMutex_Unlock(&DBA__InitMutex);
+ return DBA_NO_ERROR;
+ }
+
+ DBA__TheNdb = new Ndb("Newton");
+ DBA__TheNdb->init(1024);
+ if(DBA__TheNdb->waitUntilReady() != 0){
+ delete DBA__TheNdb; DBA__TheNdb = 0;
+ NdbMutex_Unlock(&DBA__InitMutex);
+ return DBA_NDB_ERROR;
+ }
+ DBA__TheNewtonMutex = NdbMutex_Create();
+ DBA__TheNBP = new NewtonBatchProcess(* DBA__TheNdb, * DBA__TheNewtonMutex);
+ DBA__TheNBP->doStart();
+ NdbMutex_Unlock(&DBA__InitMutex);
+ return DBA_NO_ERROR;
+}
+
+
+/**
+ * Closes the database.
+ *
+ * @return Error status
+ */
+extern "C"
+DBA_Error_t
+DBA_Close(void){
+
+ NdbMutex_Lock(&DBA__InitMutex);
+
+ if(DBA__TheNBP != 0)
+ DBA__TheNBP->doStop(true);
+ delete DBA__TheNBP;
+ DBA__TheNBP = 0;
+
+ if(DBA__TheNdb != 0)
+ delete DBA__TheNdb;
+ DBA__TheNdb = 0;
+
+ if(DBA__TheNewtonMutex != 0)
+ NdbMutex_Destroy(DBA__TheNewtonMutex);
+ DBA__TheNewtonMutex = 0;
+
+ NdbMutex_Unlock(&DBA__InitMutex);
+ return DBA_NO_ERROR;
+}
diff --git a/ndb/src/newtonapi/dba_internal.hpp b/ndb/src/newtonapi/dba_internal.hpp
new file mode 100644
index 00000000000..a021db40a7d
--- /dev/null
+++ b/ndb/src/newtonapi/dba_internal.hpp
@@ -0,0 +1,123 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBA_INTERNAL_HPP
+#define DBA_INTERNAL_HPP
+
+extern "C" {
+#include "dba.h"
+}
+
+#include <NdbApi.hpp>
+#include <NdbMutex.h>
+#include <string.h>
+#include <NdbOut.hpp>
+
+#include <stdlib.h>
+
+#ifndef INT_MAX
+#define INT_MAX 2147483647
+#endif
+
+#ifdef DEBUG
+#define DBA_DEBUG(x) ndbout << x << endl
+#else
+#define DBA_DEBUG(x)
+#endif
+
+extern Ndb * DBA__TheNdb;
+extern NdbMutex * DBA__TheNewtonMutex;
+
+extern unsigned DBA__SentTransactions;
+extern unsigned DBA__RecvTransactions;
+
+/**
+ * Configuration
+ */
+extern int DBA__NBP_Intervall; // Param 0
+extern int DBA__BulkReadCount; // Param 1
+extern int DBA__StartTransactionTimout; // Param 2
+extern int DBA__NBP_Force; // Param 3
+
+/**
+ * Error handling
+ */
+void DBA__SetLatestError(DBA_Error_t, DBA_ErrorCode_t, const char *, ...);
+
+/**
+ * Magic string
+ *
+ * Used to make sure that user passes correct pointers
+ */
+const int DBA__MagicLength = 4;
+const char DBA__TheMagic[DBA__MagicLength] = { 'K', 'E', 'S', 'O' };
+
+struct DBA_Binding {
+ char magic[DBA__MagicLength];
+ int checkSum;
+
+ char * tableName;
+ int structSz;
+
+ int noOfKeys;
+ int *keyIds;
+ int *keyOffsets;
+
+ int noOfColumns;
+ int *columnIds;
+ int *columnOffsets;
+
+ int noOfSubBindings;
+ struct DBA_Binding **subBindings;
+ int * subBindingOffsets;
+
+ int data[1];
+};
+
+struct DBA__DataTypesMapping {
+ DBA_DataTypes_t newtonType;
+ NdbDictionary::Column::Type ndbType;
+};
+
+const DBA__DataTypesMapping DBA__DataTypesMappings[] = {
+ { DBA_CHAR, NdbDictionary::Column::Char },
+ { DBA_INT, NdbDictionary::Column::Int }
+};
+
+const int DBA__NoOfMappings = sizeof(DBA__DataTypesMappings)/
+ sizeof(DBA__DataTypesMapping);
+
+/**
+ * Validate magic string and checksum of a binding
+ */
+bool DBA__ValidBinding(const DBA_Binding_t * bindings);
+bool DBA__ValidBindings(const DBA_Binding_t * const * pBindings, int n);
+
+/**
+ * Recursive equalGetValue (used for read)
+ * equalSetValue (used for write)
+ * equal (used for delete)
+ */
+bool DBA__EqualGetValue(NdbOperation *, const DBA_Binding_t *, void *);
+bool DBA__EqualSetValue(NdbOperation *, const DBA_Binding_t *, const void *);
+bool DBA__Equal (NdbOperation *, const DBA_Binding_t *, const void *);
+
+inline void require(bool test){
+ if(!test)
+ abort();
+}
+
+#endif
diff --git a/ndb/src/newtonapi/dba_process.cpp b/ndb/src/newtonapi/dba_process.cpp
new file mode 100644
index 00000000000..ddb6e62f180
--- /dev/null
+++ b/ndb/src/newtonapi/dba_process.cpp
@@ -0,0 +1,123 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "dba_process.hpp"
+
+NewtonBatchProcess::NewtonBatchProcess(Ndb & ndb, NdbMutex & mutex) :
+ theNdb(ndb),
+ theMutex(mutex)
+{
+ theThread = 0;
+ startStopMutex = NdbMutex_Create();
+
+ _running = false;
+ _stop = false;
+}
+
+NewtonBatchProcess::~NewtonBatchProcess(){
+ doStop(true);
+
+ if(theThread != 0)
+ NdbThread_Destroy(&theThread);
+
+ if(startStopMutex != 0)
+ NdbMutex_Destroy(startStopMutex);
+ startStopMutex = 0;
+}
+
+extern "C"
+void*
+runNDB_C(void * _nbp){
+ NewtonBatchProcess * nbp = (NewtonBatchProcess*)_nbp;
+ nbp->_running = true;
+ nbp->run();
+ nbp->_running = false;
+
+ /**
+ * This sleep is to make sure that the transporter
+ * send thread will come in and send any
+ * signal buffers that this thread may have allocated.
+ * If that doesn't happen an error will occur in OSE
+ * when trying to restore a signal buffer allocated by a thread
+ * that have been killed.
+ */
+ NdbSleep_MilliSleep(50);
+ NdbThread_Exit(0);
+ return 0;
+}
+
+void
+NewtonBatchProcess::doStart(){
+ NdbMutex_Lock(startStopMutex);
+ if(_running && !_stop){
+ NdbMutex_Unlock(startStopMutex);
+ return ;
+ }
+
+ while(_running){
+ NdbMutex_Unlock(startStopMutex);
+ NdbSleep_MilliSleep(200);
+ NdbMutex_Lock(startStopMutex);
+ }
+
+ require(!_running);
+ _stop = false;
+
+ if(theThread != 0)
+ NdbThread_Destroy(&theThread);
+
+ theThread = NdbThread_Create(runNDB_C,
+ (void**)this,
+ 65535,
+ "Newton_BP",
+ NDB_THREAD_PRIO_LOWEST);
+
+ NdbMutex_Unlock(startStopMutex);
+}
+
+void
+NewtonBatchProcess::doStop(bool wait){
+ NdbMutex_Lock(startStopMutex);
+ _stop = true;
+
+ if(wait){
+ while(_running){
+ NdbSleep_MilliSleep(200);
+ }
+ }
+ NdbMutex_Unlock(startStopMutex);
+}
+
+bool
+NewtonBatchProcess::isRunning() const {
+ return _running;
+}
+
+bool
+NewtonBatchProcess::isStopping() const {
+ return _stop;
+}
+
+void
+NewtonBatchProcess::run(){
+ while(!_stop){
+ NdbMutex_Lock(&theMutex);
+ theNdb.sendPollNdb(0, 1, DBA__NBP_Force);
+ NdbMutex_Unlock(&theMutex);
+ NdbSleep_MilliSleep(DBA__NBP_Intervall);
+ }
+}
diff --git a/ndb/src/newtonapi/dba_process.hpp b/ndb/src/newtonapi/dba_process.hpp
new file mode 100644
index 00000000000..ef24fbd9142
--- /dev/null
+++ b/ndb/src/newtonapi/dba_process.hpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NEWTON_BP_HPP
+#define NEWTON_BP_HPP
+
+#include "dba_internal.hpp"
+
+#include <NdbThread.h>
+#include <NdbMutex.h>
+#include <NdbSleep.h>
+
+extern "C" void* runNDB_C(void * nbp);
+
+/**
+ * This class implements the NewtonBatchProcess
+ */
+class NewtonBatchProcess {
+ friend void* runNDB_C(void * nbp);
+public:
+ NewtonBatchProcess(Ndb &, NdbMutex &);
+ ~NewtonBatchProcess();
+
+ void doStart();
+ void doStop(bool wait);
+
+ bool isRunning() const ;
+ bool isStopping() const ;
+
+private:
+ void run();
+
+ bool _running;
+ bool _stop;
+
+ Ndb & theNdb;
+ NdbMutex & theMutex;
+
+ NdbThread * theThread;
+ NdbMutex * startStopMutex;
+};
+
+#endif
diff --git a/ndb/src/newtonapi/dba_schema.cpp b/ndb/src/newtonapi/dba_schema.cpp
new file mode 100644
index 00000000000..9fa99cb5d43
--- /dev/null
+++ b/ndb/src/newtonapi/dba_schema.cpp
@@ -0,0 +1,149 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "dba_internal.hpp"
+
+static bool getNdbAttr(DBA_DataTypes_t,
+ Size_t,
+ int * attrSize,
+ int * arraySize,
+ AttrType * attrType);
+
+extern "C"
+DBA_Error_t
+DBA_CreateTable(const char* TableName,
+ int NbColumns,
+ const DBA_ColumnDesc_t Columns[] ){
+
+ if(DBA_TableExists(TableName))
+ return DBA_NO_ERROR;
+
+ NdbSchemaCon * schemaCon = DBA__TheNdb->startSchemaTransaction();
+ if(schemaCon == 0){
+ DBA__SetLatestError(DBA_NDB_ERROR, 0,
+ "Internal NDB error: No schema transaction");
+ return DBA_NDB_ERROR;
+ }
+
+ NdbSchemaOp * schemaOp = schemaCon->getNdbSchemaOp();
+ if(schemaOp == 0){
+ DBA__TheNdb->closeSchemaTransaction(schemaCon);
+ DBA__SetLatestError(DBA_NDB_ERROR, 0,
+ "Internal NDB error: No schema op");
+ return DBA_NDB_ERROR;
+ }
+
+ if(schemaOp->createTable( TableName,
+ 8, // Data Size
+ TupleKey,
+ 2, // Index size
+ All,
+ 6,
+ 78,
+ 80,
+ 1,
+ false) == -1){
+ DBA__TheNdb->closeSchemaTransaction(schemaCon);
+ DBA__SetLatestError(DBA_NDB_ERROR, 0,
+ "Internal NDB error: Create table failed");
+ return DBA_NDB_ERROR;
+ }
+
+ for (int i = 0; i < NbColumns; i++){
+ int attrSize;
+ int arraySize;
+ AttrType attrType;
+
+ if(!getNdbAttr(Columns[i].DataType, Columns[i].Size,
+ &attrSize,
+ &arraySize,
+ &attrType)){
+ DBA__TheNdb->closeSchemaTransaction(schemaCon);
+ DBA__SetLatestError(DBA_APPLICATION_ERROR, 0,
+ "Invalid datatype/size combination");
+ return DBA_APPLICATION_ERROR;
+ }
+
+ if(schemaOp->createAttribute( Columns[i].Name,
+ Columns[i].IsKey ? TupleKey : NoKey,
+ attrSize,
+ arraySize,
+ attrType) == -1){
+ DBA__TheNdb->closeSchemaTransaction(schemaCon);
+ DBA__SetLatestError(DBA_NDB_ERROR, 0,
+ "Internal NDB error: Create attribute failed");
+ return DBA_NDB_ERROR;
+ }
+ }
+
+ if(schemaCon->execute() == -1){
+ DBA__TheNdb->closeSchemaTransaction(schemaCon);
+ DBA__SetLatestError(DBA_NDB_ERROR, 0,
+ "Internal NDB error: Execute schema failed");
+ return DBA_NDB_ERROR;
+ }
+
+ DBA__TheNdb->closeSchemaTransaction(schemaCon);
+
+ return DBA_NO_ERROR;
+}
+
+DBA_Error_t
+DBA_DropTable( char* TableName ){
+ return DBA_NOT_IMPLEMENTED;
+}
+
+Boolean_t
+DBA_TableExists( const char* TableName ){
+ NdbDictionary::Dictionary * dict = DBA__TheNdb->getDictionary();
+ if(dict == 0){
+ return 0;
+ }
+
+ const NdbDictionary::Table * tab = dict->getTable(TableName);
+ if(tab == 0){
+ return 0;
+ }
+ return 1;
+}
+
+static
+bool
+getNdbAttr(DBA_DataTypes_t type,
+ Size_t size,
+ int * attrSize,
+ int * arraySize,
+ AttrType * attrType) {
+
+ if(type == DBA_CHAR){
+ * attrType = String;
+ * attrSize = 8;
+ * arraySize = size;
+ return true;
+ }
+
+ * attrType = Signed;
+ if((size % 4) == 0){
+ * attrSize = 32;
+ * arraySize = size / 4;
+ return true;
+ }
+
+ * attrSize = 8;
+ * arraySize = size;
+
+ return true;
+}
diff --git a/ndb/src/rep/ExtSender.cpp b/ndb/src/rep/ExtSender.cpp
new file mode 100644
index 00000000000..cf31001a85f
--- /dev/null
+++ b/ndb/src/rep/ExtSender.cpp
@@ -0,0 +1,149 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "ExtSender.hpp"
+
+/*****************************************************************************
+ * Constructor / Destructor / Init / Get / Set
+ *****************************************************************************/
+
+/**
+ * @todo: signalErrorHandler is not finished. Just infrastructure.
+ */
+
+ExtSender::ExtSender() {
+ m_tf = NULL;
+ m_nodeId = 0;
+ m_ownRef = 0;
+}
+
+ExtSender::~ExtSender() {
+ if (m_tf) delete m_tf;
+}
+
+void
+ExtSender::setNodeId(Uint32 nodeId)
+{
+#if 0
+ ndbout_c("ExtSender: Set nodeid to %d", nodeId);
+#endif
+
+ m_nodeId = nodeId;
+}
+
+Uint32
+ExtSender::getOwnRef() const
+{
+ if(!m_ownRef) REPABORT("No m_ownRef set");
+
+ return m_ownRef;
+}
+
+void
+ExtSender::setOwnRef(Uint32 ref)
+{
+ // Can only be set once
+ if (m_ownRef != 0) REPABORT("Trying to change m_ownRef");
+
+ m_ownRef = ref;
+}
+
+/*****************************************************************************
+ * Usage
+ *****************************************************************************/
+
+int
+ExtSender::sendSignal(class NdbApiSignal * s) {
+#if 0
+ ndbout_c("ExtSender: Sending signal %d to %d",
+ s->readSignalNumber(), m_nodeId);
+#endif
+
+ if (m_tf == NULL || m_nodeId == 0 || s==0) abort();
+ m_tf->lock_mutex();
+ int retvalue = m_tf->sendSignal(s, m_nodeId);
+ if (retvalue) {
+ RLOG(("sendSignal returned %d for send to node %d", retvalue, m_nodeId));
+ }
+#if 0
+ ndbout_c("ExtSender: Sent signal to %d", m_nodeId);
+#endif
+ m_tf->unlock_mutex();
+ return retvalue;
+}
+
+int
+ExtSender::sendFragmentedSignal(NdbApiSignal * s,
+ LinearSectionPtr ptr[3],
+ Uint32 sections) {
+ if (m_tf == NULL || m_nodeId == 0) abort();
+ m_tf->lock_mutex();
+ int retvalue = m_tf->sendFragmentedSignal(s, m_nodeId, ptr, sections);
+ if (retvalue) {
+ RLOG(("sendFragmentedSignal returned %d for send to node %d",
+ retvalue, m_nodeId));
+ }
+ m_tf->unlock_mutex();
+ return retvalue;
+}
+
+/**
+ * Check that TransporterFacade is connected to at least one DB node
+ */
+bool
+ExtSender::connected(Uint32 timeOutMillis){
+#if 0
+ ndbout_c("ExtSender: Waiting for remote component to be ready!");
+#endif
+
+ NDB_TICKS start = NdbTick_CurrentMillisecond();
+ NDB_TICKS now = start;
+ // while(m_tf->theClusterMgr->getNoOfConnectedNodes() == 0 &&
+ while((m_tf->get_an_alive_node() == 0) &&
+ (timeOutMillis == 0 || (now - start) < timeOutMillis)){
+ NdbSleep_MilliSleep(100);
+ now = NdbTick_CurrentMillisecond();
+ }
+ return m_tf->theClusterMgr->getNoOfConnectedNodes() > 0;
+}
+
+bool
+ExtSender::connected(Uint32 timeOutMillis, Uint32 nodeId){
+ NDB_TICKS start = NdbTick_CurrentMillisecond();
+ NDB_TICKS now = start;
+
+ // while(m_tf->theClusterMgr->getNoOfConnectedNodes() == 0 &&
+ while((m_tf->get_node_alive(nodeId) != 0) &&
+ (timeOutMillis == 0 || (now - start) < timeOutMillis)){
+ NdbSleep_MilliSleep(100);
+ now = NdbTick_CurrentMillisecond();
+ }
+ return m_tf->theClusterMgr->getNoOfConnectedNodes() > 0;
+}
+
+NdbApiSignal *
+ExtSender::getSignal()
+{
+ /**
+ * @todo This should use some kind of list of NdbApiSignals,
+ * similar to the NDBAPI and the MGRSRVR.
+ * The best thing would be to have set of code
+ * shared between the programs.
+ * Thus the NDBAPI and MGMSRVR should be refactored.
+ * /Lars
+ */
+ return new NdbApiSignal(getOwnRef());
+}
diff --git a/ndb/src/rep/ExtSender.hpp b/ndb/src/rep/ExtSender.hpp
new file mode 100644
index 00000000000..0bdabd68f37
--- /dev/null
+++ b/ndb/src/rep/ExtSender.hpp
@@ -0,0 +1,76 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef EXT_SENDER_HPP
+#define EXT_SENDER_HPP
+
+#include <NdbSleep.h>
+#include <TransporterFacade.hpp>
+#include <NdbApiSignal.hpp>
+#include <rep/rep_version.hpp>
+
+/**
+ * @todo Johan comment:
+ *
+ * ext->sendSignal should return something if send failed.
+ * I.e., i think all methods sending a signal should return int
+ * so that we can take care of errors. ALternatively take care of
+ * the error like this:
+ * if(ext->sendSignal(..) < 0 )
+ * handleSignalError(...)
+ *
+ * or a combination....
+ *
+ * Should go through all places that sends signals and check that
+ * they do correct error handling.
+ */
+
+/**
+ * @class ExtSender
+ * @brief Manages connection to a transporter facade
+ */
+class ExtSender {
+public:
+ /***************************************************************************
+ * Constructor / Destructor / Init / Get / Set (Only set once!)
+ ***************************************************************************/
+ ExtSender();
+ ~ExtSender();
+
+ void setTransporterFacade(TransporterFacade * tf) { m_tf = tf; }
+ void setNodeId(Uint32 nodeId);
+ Uint32 getOwnRef() const;
+ void setOwnRef(Uint32 ref);
+
+ /***************************************************************************
+ * Usage
+ ***************************************************************************/
+ int sendSignal(NdbApiSignal * s);
+ int sendFragmentedSignal(NdbApiSignal * s, LinearSectionPtr ptr[3],
+ Uint32 sections);
+
+ bool connected(Uint32 TimeOutInMilliSeconds);
+ bool connected(Uint32 TimeOutInMilliSeconds, Uint32 nodeId);
+
+ NdbApiSignal * getSignal();
+
+private:
+ TransporterFacade * m_tf;
+ Uint32 m_nodeId;
+ Uint32 m_ownRef;
+};
+
+#endif
diff --git a/ndb/src/rep/Makefile b/ndb/src/rep/Makefile
new file mode 100644
index 00000000000..29482b72687
--- /dev/null
+++ b/ndb/src/rep/Makefile
@@ -0,0 +1,30 @@
+include .defs.mk
+
+#
+# This "kernel" type should be removed (only need types)
+#
+TYPE := repserver kernel
+
+DIRS := adapters storage state transfer repapi
+
+BIN_TARGET := ndb_rep
+
+BIN_TARGET_LIBS :=
+BIN_TARGET_ARCHIVES += editline repstorage repadapters reprequestor reptransfer mgmapi NDB_API mgmsrvcommon
+
+LDFLAGS_LOC = -lpthread
+
+SOURCES = \
+ RepMain.cpp \
+ Requestor.cpp \
+ RequestorSubscriptions.cpp \
+ \
+ RepComponents.cpp \
+ RepCommandInterpreter.cpp \
+ RepApiService.cpp \
+ RepApiInterpreter.cpp \
+ SignalQueue.cpp \
+ ExtSender.cpp \
+ dbug_hack.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/rep/NodeConnectInfo.hpp b/ndb/src/rep/NodeConnectInfo.hpp
new file mode 100644
index 00000000000..403f92a5999
--- /dev/null
+++ b/ndb/src/rep/NodeConnectInfo.hpp
@@ -0,0 +1,29 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODE_CONNECTINFO_HPP
+#define NODE_CONNECTINFO_HPP
+
+#include <ndb_types.h>
+
+struct NodeConnectInfo {
+ NodeConnectInfo(Uint16 n, bool c): nodeId(n), connected(c) {};
+ Uint32 nodeId;
+ bool connected;
+};
+
+
+#endif
diff --git a/ndb/src/rep/README b/ndb/src/rep/README
new file mode 100644
index 00000000000..7be5e230eb3
--- /dev/null
+++ b/ndb/src/rep/README
@@ -0,0 +1,147 @@
+ ===========================================
+ MySQL Replication Servers
+ Lars Thalmann and Johan Andersson
+ 2003 MySQL AB
+ ===========================================
+
+-------------------------------------------------------------------------------
+
+ PRIMARY SYSTEM STANDBY SYSTEM
+ REPLICATION SERVER REPLICATION SERVER
+ (PS or SOURCE SYSTEM) (SS or DESTINATION SYSTEM)
+ +------------------+ +-------------------------+
+ | RepMain | | RepMain [Requests] |
+ | | +-------------------------+
+ | | | Requestor [Executes] |
+ +------------------+ +-------------------------+
+ PS --- | ExtNDB | TransPS | --- | TransSS | AppNDB | --- SS
+ +------------------+ +-------------------------+
+ (GCIContainer) (GCIContainer)
+ (RepState)
+
+ Figure 1: Replication Server Threads
+
+Component List
+--------------
+RepMain
+ Main thread that runs command-line interpreter [Requests]
+
+Requestor
+ Thread that runs RepState::execute [Executes]
+
+ExtNDB
+ Extracts transaction epochs from NDB Cluster
+
+TransPS, TransSS
+ Transfers information (control and epoch buffers) between
+ Replication Servers.
+
+AppNDB
+ Applies transaction epochs to NDB Cluster
+
+-------------------------------------------------------------------------------
+
+ RepState Control
+ Object
+ +------------------+
+ | RepState |
+ | [Requests] |
+ | [Executes] |
+ +------------------+
+ | RepStateRequest | --- ExtSender
+ +------------------+
+
+ Figure 2: RepState Object
+
+
+The RepState object is shared by all components.
+
+
+-------------------------------------------------------------------------------
+
+Dependent Directories
+---------------------
+rep/adapters Appliers and Extractors
+ All code dependent on the database system
+
+rep/transfer
+ Depends on NDB transporters
+
+rep/state
+ Shared resources for all components
+
+Independent Directories
+-----------------------
+rep/storage Storage of epochs
+ Should not depend on any transporters/NDB specific
+
+rep/repstate
+ Should only have a reference to an ExtSender for the external PS REP node
+
+
+-------------------------------------------------------------------------------
+
+Replication Teminology
+----------------------
+GLOBAL CHECKPOINT <global checkpoint id - gci>
+A global checkpoint is a point in time when all database server
+are synchronized.
+
+NODE
+A database server with information.
+
+NODE GROUP <node group id>
+A set of database servers, all storing the same information.
+
+SUBSCRIPTION <subscription id>.
+A "subscription" is a collection of services that a source system
+provides. The main to services belonging to a subscription are
+"log" and "scan". Log provides the replication servers with
+log entries (epochs) and scan provides the replication servers
+with scanned data (also stored in epochs).
+
+EPOCH <subscription id, gci>
+An "epoch" is a log of all database changes between two time points.
+(An epoch can have redundant log entries.) An epoch is named by the
+number of the time slice between the two time points.
+
+EPOCH BUFFER <subscription id, gci, node group id>
+An "epoch buffer" is a part of the log belonging to an epoch. An
+epoch buffer does not contain any redundancy.
+
+Two epoch buffers with the same subscription id and gci can be
+"complements" or "duplicates" to each other. If they are complements,
+they store different information, if they are duplicates then they
+store equivalent information (the information need not be identical,
+but it is equivalent for the purpose of restoring the original
+information). If they are duplicates then they have the same name,
+i.e. same subscription id, gci, and node group id.
+
+CHANNEL <subscription id>
+A "channel" is a collection of epoch buffers belonging to
+a specific subscription. (The channel can exist before it is
+assigned to a subscription.)
+
+SUBSCRIPTION CONSISTENT
+A database is "subscription consistent" or "consistent with respect
+to a subscription" if ...
+
+Architectural Terminology
+-------------------------
+ADAPTER
+An "adapter" is either an applier or an extractor.
+
+APPLIER
+An "applier" is a a collection of threads in the replication server
+that applies epochs to a destination database system.
+
+EXTRACTOR
+An "extractor" is a collection of theads in the replication server
+that receives epochs from a source database system.
+
+TRANSFER COMPONENT
+A "transfer component" is a thread in the replication server that is
+responsible for the connection with another replication server.
+
+REQUESTOR
+A thread in the replication server that controls replication.
diff --git a/ndb/src/rep/RepApiInterpreter.cpp b/ndb/src/rep/RepApiInterpreter.cpp
new file mode 100644
index 00000000000..6e6f150713a
--- /dev/null
+++ b/ndb/src/rep/RepApiInterpreter.cpp
@@ -0,0 +1,80 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "RepApiInterpreter.hpp"
+#include <signaldata/GrepImpl.hpp>
+
+RepApiInterpreter::RepApiInterpreter(RepComponents * comps, int port)
+{
+ m_repComponents = comps;
+ m_repState = comps->getRepState();
+ m_port = port;
+ ss = new SocketServer();
+ serv = new RepApiService(*this);
+}
+
+
+RepApiInterpreter::~RepApiInterpreter()
+{
+}
+
+void
+RepApiInterpreter::startInterpreter()
+{
+ if(!ss->setup(serv, m_port)){
+ sleep(1);
+ delete ss;
+ delete serv;
+ }
+ ss->startServer();
+}
+
+
+void
+RepApiInterpreter::stopInterpreter()
+{
+ delete ss;
+}
+
+
+Properties *
+RepApiInterpreter::execCommand(const Properties & props)
+{
+ Properties * result = new Properties();
+ Uint32 req = 0;
+ Uint32 epoch = 0;
+ props.get("request", &req);
+ props.get("epoch", &epoch);
+ GrepError::Code err = m_repState->protectedRequest((GrepReq::Request)req,
+ epoch);
+ result->put("err", err);
+ return result;
+}
+
+Properties *
+RepApiInterpreter::getStatus()
+{
+
+ return m_repState->getStatus();
+}
+
+
+Properties *
+RepApiInterpreter::query(Uint32 counter, Uint32 replicationId)
+{
+ return m_repState->query((QueryCounter)counter, replicationId);
+}
+
diff --git a/ndb/src/rep/RepApiInterpreter.hpp b/ndb/src/rep/RepApiInterpreter.hpp
new file mode 100644
index 00000000000..78f190156b3
--- /dev/null
+++ b/ndb/src/rep/RepApiInterpreter.hpp
@@ -0,0 +1,54 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REP_API_INTERPRETER_HPP
+#define REP_API_INTERPRETER_HPP
+
+#include <editline/editline.h>
+
+#include <rep/RepComponents.hpp>
+#include <rep/state/RepState.hpp>
+#include <rep/RepApiService.hpp>
+#include <signaldata/GrepImpl.hpp>
+#include <Properties.hpp>
+
+/**
+ * @class RepCommandInterpreter
+ * @brief
+ */
+
+class RepApiInterpreter {
+public:
+ RepApiInterpreter(class RepComponents * comps, int port);
+ ~RepApiInterpreter();
+ void startInterpreter();
+ void stopInterpreter();
+ Properties * execCommand(const Properties & props);
+ Properties * getStatus();
+ Properties * query(Uint32 counter, Uint32 replicationId);
+ bool readAndExecute();
+
+private:
+ char * readline_gets() const;
+ void request(Uint32 request);
+ int m_port;
+ class RepComponents * m_repComponents;
+ class RepState * m_repState;
+ SocketServer * ss;
+ RepApiService * serv;
+};
+
+#endif
diff --git a/ndb/src/rep/RepApiService.cpp b/ndb/src/rep/RepApiService.cpp
new file mode 100644
index 00000000000..f5d51f7990e
--- /dev/null
+++ b/ndb/src/rep/RepApiService.cpp
@@ -0,0 +1,320 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <Parser.hpp>
+#include <NdbOut.hpp>
+#include <Properties.hpp>
+#include <socket_io.h>
+#include "RepApiService.hpp"
+#include "RepApiInterpreter.hpp"
+#include "repapi/repapi.h"
+#include <NdbMutex.h>
+#include <NdbStdio.h>
+#include <OutputStream.hpp>
+
+#include <NdbString.h>
+/**
+ const char * name;
+ const char * realName;
+ const Type type;
+ const ArgType argType;
+ const ArgRequired argRequired;
+ const ArgMinMax argMinMax;
+ const int minVal;
+ const int maxVal;
+ void (T::* function)(const class Properties & args);
+ const char * description;
+*/
+
+#define REP_CMD(name, fun, desc) \
+ { name, \
+ 0, \
+ ParserRow<RepApiSession>::Cmd, \
+ ParserRow<RepApiSession>::String, \
+ ParserRow<RepApiSession>::Optional, \
+ ParserRow<RepApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ fun, \
+ desc }
+
+#define REP_ARG(name, type, opt, desc) \
+ { name, \
+ 0, \
+ ParserRow<RepApiSession>::Arg, \
+ ParserRow<RepApiSession>::type, \
+ ParserRow<RepApiSession>::opt, \
+ ParserRow<RepApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ desc }
+
+#define REP_ARG2(name, type, opt, min, max, desc) \
+ { name, \
+ 0, \
+ ParserRow<RepApiSession>::Arg, \
+ ParserRow<RepApiSession>::type, \
+ ParserRow<RepApiSession>::opt, \
+ ParserRow<RepApiSession>::IgnoreMinMax, \
+ min, max, \
+ 0, \
+ desc }
+
+#define REP_END() \
+ { 0, \
+ 0, \
+ ParserRow<RepApiSession>::Arg, \
+ ParserRow<RepApiSession>::Int, \
+ ParserRow<RepApiSession>::Optional, \
+ ParserRow<RepApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0 }
+
+#define REP_CMD_ALIAS(name, realName, fun) \
+ { name, \
+ realName, \
+ ParserRow<RepApiSession>::CmdAlias, \
+ ParserRow<RepApiSession>::Int, \
+ ParserRow<RepApiSession>::Optional, \
+ ParserRow<RepApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0 }
+
+#define REP_ARG_ALIAS(name, realName, fun) \
+ { name, \
+ realName, \
+ ParserRow<RepApiSession>::ArgAlias, \
+ ParserRow<RepApiSession>::Int, \
+ ParserRow<RepApiSession>::Optional, \
+ ParserRow<RepApiSession>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0 }
+
+
+const
+ParserRow<RepApiSession> commands[] =
+{
+
+ REP_CMD("rep" , &RepApiSession::execCommand, ""),
+ REP_ARG("request", Int, Mandatory, "Grep::Request."),
+ REP_ARG("id", Int, Mandatory, "Replication id "),
+ REP_ARG("epoch", Int, Optional, "Epoch. Used by stop epoch ..."),
+
+ REP_CMD("rep status" , &RepApiSession::getStatus, ""),
+ REP_ARG("request", Int, Optional, "Grep::Request."),
+
+ REP_CMD("rep query" , &RepApiSession::query, ""),
+ REP_ARG("id", Int, Mandatory, "Replication Id"),
+ REP_ARG("counter", Int, Mandatory, "QueryCounter."),
+ REP_ARG("request", Int, Mandatory, "Grep::Request."),
+
+ REP_END()
+};
+RepApiSession::RepApiSession(NDB_SOCKET_TYPE sock,
+ class RepApiInterpreter & rep)
+ : SocketServer::Session(sock)
+ , m_rep(rep)
+{
+ m_input = new SocketInputStream(sock);
+ m_output = new SocketOutputStream(sock);
+ m_parser = new Parser<RepApiSession>(commands, *m_input, true, true, true);
+}
+
+RepApiSession::RepApiSession(FILE * f, class RepApiInterpreter & rep)
+ : SocketServer::Session(1)
+ , m_rep(rep)
+{
+ m_input = new FileInputStream(f);
+ m_parser = new Parser<RepApiSession>(commands, *m_input, true, true, true);
+}
+
+RepApiSession::~RepApiSession()
+{
+ delete m_input;
+ delete m_parser;
+}
+
+void
+RepApiSession::runSession()
+{
+ Parser_t::Context ctx;
+ while(!m_stop){
+ m_parser->run(ctx, * this);
+ if(ctx.m_currentToken == 0)
+ break;
+
+ switch(ctx.m_status){
+ case Parser_t::Ok:
+ for(size_t i = 0; i<ctx.m_aliasUsed.size(); i++)
+ ndbout_c("Used alias: %s -> %s",
+ ctx.m_aliasUsed[i]->name, ctx.m_aliasUsed[i]->realName);
+ break;
+ case Parser_t::NoLine:
+ case Parser_t::EmptyLine:
+ break;
+ default:
+ break;
+ }
+ }
+ NDB_CLOSE_SOCKET(m_socket);
+}
+
+void
+RepApiSession::execCommand(Parser_t::Context & /* unused */,
+ const class Properties & args)
+{
+ Uint32 err;
+ Uint32 replicationId;
+ args.get("id", &replicationId);
+ Properties * result = m_rep.execCommand(args);
+ if(result == NULL) {
+ m_output->println("global replication reply");
+ m_output->println("result: %d", -1);
+ m_output->println("id: %d",replicationId);
+ m_output->println("");
+ return;
+ }
+ result->get("err", &err);
+ m_output->println("global replication reply");
+ m_output->println("result: %d", err);
+ m_output->println("id: %d", 0);
+ m_output->println("");
+ delete result;
+}
+
+
+void
+RepApiSession::getStatus(Parser_t::Context & /* unused */,
+ const class Properties & args)
+{
+ Uint32 err;
+ Properties * result = m_rep.getStatus();
+ result->get("err", &err);
+ Uint32 subId;
+ result->get("subid", &subId);
+ Uint32 subKey;
+ result->get("subkey", &subKey);
+ Uint32 connected_rep;
+ result->get("connected_rep", &connected_rep);
+ Uint32 connected_db;
+ result->get("connected_db", &connected_db);
+ Uint32 state;
+ result->get("state", &state);
+ Uint32 state_sub;
+ result->get("state", &state_sub);
+
+ m_output->println("global replication status reply");
+ m_output->println("result: %d",0);
+ m_output->println("id: %d",0);
+ m_output->println("subid: %d", subId);
+ m_output->println("subkey: %d", subKey);
+ m_output->println("connected_rep: %d", connected_rep);
+ m_output->println("connected_db: %d", connected_db);
+ m_output->println("state_sub: %d", state_sub);
+ m_output->println("state: %d", state);
+ m_output->println("");
+ delete result;
+}
+
+
+void
+RepApiSession::query(Parser_t::Context & /* unused */,
+ const class Properties & args)
+{
+ Uint32 err;
+ Uint32 counter, replicationId;
+ args.get("counter", &counter);
+ args.get("id", &replicationId);
+ Properties * result = m_rep.query(counter, replicationId);
+ if(result == NULL) {
+ m_output->println("global replication query reply");
+ m_output->println("result: %s","Failed");
+ m_output->println("id: %d",replicationId);
+ m_output->println("");
+ return;
+ }
+
+ BaseString first;
+ BaseString last;
+ Uint32 subid = 0, subkey = 0, no_of_nodegroups = 0;
+ Uint32 connected_rep = 0, connected_db = 0;
+ Uint32 state = 0 , state_sub = 0;
+ result->get("err", &err);
+ result->get("no_of_nodegroups", &no_of_nodegroups);
+ result->get("subid", &subid);
+ result->get("subkey", &subkey);
+ result->get("connected_rep", &connected_rep);
+ result->get("connected_db", &connected_db);
+ result->get("first", first);
+ result->get("last", last);
+ result->get("state", &state);
+ result->get("state_sub", &state_sub);
+ m_output->println("global replication query reply");
+ m_output->println("result: %s","Ok");
+ m_output->println("id: %d",replicationId);
+ m_output->println("no_of_nodegroups: %d",no_of_nodegroups);
+ m_output->println("subid: %d", subid);
+ m_output->println("subkey: %d", subkey);
+ m_output->println("connected_rep: %d", connected_rep);
+ m_output->println("connected_db: %d", connected_db);
+ m_output->println("state_sub: %d", state_sub);
+ m_output->println("state: %d", state);
+ m_output->println("first: %s", first.c_str());
+ m_output->println("last: %s", last.c_str());
+ m_output->println("");
+ delete result;
+}
+
+
+
+static const char *
+propToString(Properties *prop, const char *key) {
+ static char buf[32];
+ const char *retval = NULL;
+ PropertiesType pt;
+
+ prop->getTypeOf(key, &pt);
+ switch(pt) {
+ case PropertiesType_Uint32:
+ Uint32 val;
+ prop->get(key, &val);
+ snprintf(buf, sizeof buf, "%d", val);
+ retval = buf;
+ break;
+ case PropertiesType_char:
+ const char *str;
+ prop->get(key, &str);
+ retval = str;
+ break;
+ default:
+ snprintf(buf, sizeof buf, "(unknown)");
+ retval = buf;
+ }
+ return retval;
+}
+
+void
+RepApiSession::printProperty(Properties *prop, const char *key) {
+ m_output->println("%s: %s", key, propToString(prop, key));
+}
+
+void
+RepApiSession::stopSession(){
+
+}
diff --git a/ndb/src/rep/RepApiService.hpp b/ndb/src/rep/RepApiService.hpp
new file mode 100644
index 00000000000..e1137e53258
--- /dev/null
+++ b/ndb/src/rep/RepApiService.hpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REP_APISERVICE_HPP
+#define REP_APISERVICE_HPP
+
+#include <Parser.hpp>
+#include <InputStream.hpp>
+#include <SocketServer.hpp>
+
+class RepApiInterpreter;
+
+class RepApiSession : public SocketServer::Session {
+ typedef Parser<RepApiSession> Parser_t;
+
+ class RepApiInterpreter & m_rep;
+ InputStream *m_input;
+ OutputStream *m_output;
+ Parser_t *m_parser;
+
+void printProperty(Properties *prop, const char *key);
+public:
+ RepApiSession(NDB_SOCKET_TYPE, class RepApiInterpreter &);
+ RepApiSession(FILE * f, class RepApiInterpreter & rep);
+ ~RepApiSession();
+
+ virtual void runSession();
+ virtual void stopSession();
+
+ void execCommand(Parser_t::Context & ctx, const class Properties & args);
+ void getStatus(Parser_t::Context & ctx, const class Properties & args);
+ void query(Parser_t::Context & ctx, const class Properties & args);
+
+};
+
+class RepApiService : public SocketServer::Service {
+ class RepApiInterpreter & m_rep;
+public:
+ RepApiService(class RepApiInterpreter & rep) : m_rep(rep) {}
+
+ RepApiSession * newSession(NDB_SOCKET_TYPE theSock){
+ return new RepApiSession(theSock, m_rep);
+ }
+};
+
+#endif
diff --git a/ndb/src/rep/RepCommandInterpreter.cpp b/ndb/src/rep/RepCommandInterpreter.cpp
new file mode 100644
index 00000000000..a0daf9529ab
--- /dev/null
+++ b/ndb/src/rep/RepCommandInterpreter.cpp
@@ -0,0 +1,456 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "RepCommandInterpreter.hpp"
+
+static const char*
+helpTextRep =
+"+-------------------------------------------------------------------------+\n"
+"| MySQL Replication Server |\n"
+"| Commands should be executed on the standby Replication Server |\n"
+"+-------------------------------------------------------------------------+\n"
+"| Simple Commands |\n"
+"+-------------------------------------------------------------------------+\n"
+"| START Start replication |\n"
+"| STATUS Show replication status |\n"
+"+-------------------------------------------------------------------------+\n"
+"| Advanced Commands |\n"
+"+-------------------------------------------------------------------------+\n"
+"| STOP <epoch no> Stop replication after epoch number <epoch no> |\n"
+"| STOP IMMEDIATELY Stop replication after applying the current epoch |\n"
+"| ADD TABLE <db>/<schema>/<tablename> |\n"
+"| Note: <db>/<schema>/<tablename> is case sensitive! |\n"
+"| Use 'STATUS' to see added tables. |\n"
+"| REMOVE TABLE <db>/<schema>/<tablename> |\n"
+"| Note: <db>/<schema>/<tablename> is case sensitive! |\n"
+"| ENABLE <protocol> Starts protocol |\n"
+"| DISABLE <protocol> Stops protocol |\n"
+"| DEBUG Toggle logging of replication messages on console |\n"
+"| |\n"
+"| <protocol> ::= REQUESTOR | TRANSFER | APPLY | DELETE |\n"
+"+-------------------------------------------------------------------------+\n"
+;
+
+/**
+ * @todo
+"| <protocol> ::= SUBID | SUBSCRIPTION |\n"
+"| <protocol> ::= METALOG | METASCAN | DATALOG | DATASCAN |\n"
+"| <system> ::= PRIMARY | STANDBY | TWOWAY |\n"
+"| CONNECT <system> Connects to NDB Cluster and other replication server |\n"
+"| DELETE Removes all epochs stored in replication servers |\n"
+"| DROP <tableid> Drops table in standby system identified by table id |\n"
+"| <epoch> ::= Any integer (naming the last epoch to be applied) |\n"
+*/
+
+RepCommandInterpreter::RepCommandInterpreter(RepComponents * comps)
+{
+ m_repComponents = comps;
+ m_repState = comps->getRepState();
+}
+
+RepCommandInterpreter::~RepCommandInterpreter()
+{
+}
+
+/**
+ * Read a string, and return a pointer to it.
+ *
+ * @return NULL on EOF.
+ */
+char *
+RepCommandInterpreter::readline_gets() const
+{
+ static char *line_read = (char *)NULL;
+
+ // Disable the default file-name completion action of TAB
+ // rl_bind_key ('\t', rl_insert);
+
+ /* If the buffer has already been allocated, return the memory
+ to the free pool. */
+ if (line_read)
+ {
+ NdbMem_Free(line_read);
+ line_read = (char *)NULL;
+ }
+
+ /* Get a line from the user. */
+ line_read = readline ("REP> ");
+
+ /* If the line has any text in it, save it on the history. */
+ if (line_read && *line_read)
+ add_history (line_read);
+
+ return (line_read);
+}
+
+bool emptyString(const char* s)
+{
+ if (s == NULL) {
+ return true;
+ }
+
+ for (unsigned int i = 0; i < strlen(s); ++i) {
+ if (! isspace(s[i])) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+/**
+ * Converts a string to a Uint32 pointed value!
+ */
+bool convert(const char* s, Uint32 * val)
+{
+ if (s == NULL) {
+ return false;
+ }
+
+ if (strlen(s) == 0) {
+ return false;
+ }
+
+ errno = 0;
+ char* p;
+ long v = strtol(s, &p, 10);
+ if (errno != 0) {
+ return false;
+ }
+ if (p != &s[strlen(s)]) {
+ return false;
+ }
+
+ *val = v;
+ return true;
+}
+
+void
+printError(GrepError::Code err)
+{
+ if (err == GrepError::NO_ERROR) { ndbout << "Ok" << endl; }
+ else { ndbout << GrepError::getErrorDesc(err) << endl; }
+}
+
+bool
+RepCommandInterpreter::readAndExecute()
+{
+ GrepError::Code err;
+
+ char* _line = readline_gets();
+ char * line;
+ if(_line == NULL) {
+ ndbout << endl;
+ return true;
+ }
+
+ line = strdup(_line);
+
+ if (emptyString(line)) {
+ return true;
+ }
+
+ /* I have to uncomment this, since otherwise <db>/<schema>/<table>
+ is converted to capitals, but it is case sensitive!
+ for (unsigned int i = 0; i < strlen(line); ++i) {
+ line[i] = toupper(line[i]);
+ }
+ */
+ // if there is anything in the line proceed
+ char* firstToken = strtok(line, " ");
+ for (unsigned int i = 0; i < strlen(firstToken); ++i) {
+ firstToken[i] = toupper(firstToken[i]);
+ }
+ char* allAfterFirstToken = strtok(NULL, "\0");
+
+ /**
+ * Commands for REP Client only
+ */
+ if (strcmp(firstToken, "ADD") == 0) {
+ if (m_repState->m_channel.getStateSub() !=
+ Channel::NO_SUBSCRIPTION_EXISTS) {
+ ndbout_c("Subscription already exists");
+ ndbout_c("Tables must be added before subscription exists");
+ return true;
+ }
+ char * secondToken = strtok(allAfterFirstToken, " ");
+ char * fullTableName = strtok(NULL, "\0");
+ if(fullTableName == NULL) {
+ ndbout_c("Table name not specified");
+ return true;
+ }
+ for (unsigned int i = 0; i < strlen(secondToken); ++i) {
+ secondToken[i] = toupper(secondToken[i]);
+ }
+
+ if (strcmp(secondToken, "TABLE") == 0) {
+ err = m_repState->protectedAddTable(fullTableName);
+ printError(err);
+ return true;
+ }
+ return true;
+ }
+ if (strcmp(firstToken, "REMOVE") == 0) {
+ if (m_repState->m_channel.getStateSub() !=
+ Channel::NO_SUBSCRIPTION_EXISTS) {
+ ndbout_c("Subscription already exists");
+ ndbout_c("Tables can not be removed after subscription is created");
+ return true;
+ }
+ char * secondToken = strtok(allAfterFirstToken, " ");
+ char * fullTableName = strtok(NULL, "\0");
+ if(fullTableName == NULL) {
+ ndbout_c("Table name not specified");
+ return true;
+ }
+ for (unsigned int i = 0; i < strlen(secondToken); ++i) {
+ secondToken[i] = toupper(secondToken[i]);
+ }
+
+ if (strcmp(secondToken, "TABLE") == 0) {
+ err = m_repState->protectedRemoveTable(fullTableName);
+ printError(err);
+ return true;
+ }
+ return true;
+ }
+ /**
+ * now, we can convert allAfterFirstToken to capitals
+ */
+ if(allAfterFirstToken != 0) {
+ for (unsigned int i = 0; i < strlen(allAfterFirstToken); ++i) {
+ allAfterFirstToken[i] = toupper(allAfterFirstToken[i]);
+ }
+ }
+ if (strcmp(firstToken, "CONNECT") == 0) {
+
+ if (strcmp(allAfterFirstToken, "PRIMARY") == 0) {
+ m_repComponents->connectPS();
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "STANDBY") == 0) {
+ m_repComponents->connectPS();
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "TWOWAY") == 0) {
+ m_repComponents->connectPS();
+ return true;
+ }
+ ndbout_c("Unknown argument: %s to command: %s",
+ allAfterFirstToken, firstToken);
+ return true;
+ }
+
+ if (strcmp(firstToken, "HELP") == 0) {
+ ndbout << helpTextRep;
+ return true;
+ }
+
+ if (strcmp(firstToken, "QUIT") == 0 ||
+ strcmp(firstToken, "BYE") == 0 ||
+ strcmp(firstToken, "EXIT") == 0) {
+ return false;
+ }
+
+ /**
+ * Commands for REP Server API
+ */
+ if (strcmp(firstToken, "STATUS") == 0 ||
+ strcmp(firstToken, "INFO") == 0 ||
+ strcmp(firstToken, "I") == 0) {
+ m_repState->protectedRequest(GrepReq::STATUS, 0);
+ return true;
+ }
+
+ if (strcmp(firstToken, "DEBUG") == 0) {
+ if (replogEnabled)
+ {
+ ndbout_c("Debugging disabled.");
+ replogEnabled = false;
+ }
+ else
+ {
+ ndbout_c("Debugging enabled.");
+ replogEnabled = true;
+ }
+ return true;
+ }
+
+ if (strcmp(firstToken, "ENABLE") == 0) {
+ if (strcmp(allAfterFirstToken, "REQUESTOR") == 0) {
+ err = m_repState->protectedRequest(GrepReq::START_REQUESTOR, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "TRANSFER") == 0) {
+ err = m_repState->protectedRequest(GrepReq::START_TRANSFER, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "APPLY") == 0) {
+ err = m_repState->protectedRequest(GrepReq::START_APPLY, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "DELETE") == 0) {
+ err = m_repState->protectedRequest(GrepReq::START_DELETE, 0);
+ printError(err);
+ return true;
+ }
+ ndbout_c("Unknown argument: %s to command: %s",
+ allAfterFirstToken, firstToken);
+ return true;
+ }
+
+ if (strcmp(firstToken, "DISABLE") == 0) {
+ if (strcmp(allAfterFirstToken, "REQUESTOR") == 0) {
+ err = m_repState->protectedRequest(GrepReq::STOP_REQUESTOR, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "TRANSFER") == 0) {
+ err = m_repState->protectedRequest(GrepReq::STOP_TRANSFER, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "APPLY") == 0) {
+ err = m_repState->protectedRequest(GrepReq::STOP_APPLY, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "DELETE") == 0) {
+ err = m_repState->protectedRequest(GrepReq::STOP_DELETE, 0);
+ printError(err);
+ return true;
+ }
+ ndbout_c("Unknown argument: %s to command: %s",
+ allAfterFirstToken, firstToken);
+ return true;
+ }
+
+ if (strcmp(firstToken, "START") == 0) {
+ if (allAfterFirstToken == NULL) {
+ err = m_repState->protectedRequest(GrepReq::START, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "SUBID") == 0) {
+ err = m_repState->protectedRequest(GrepReq::CREATE_SUBSCR, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "SUBSCR") == 0 ||
+ strcmp(allAfterFirstToken, "SUBSCRIPTION") == 0) {
+ err = m_repState->protectedRequest(GrepReq::START_SUBSCR, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "METALOG") == 0) {
+ err = m_repState->protectedRequest(GrepReq::START_METALOG, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "METASCAN") == 0) {
+ err = m_repState->protectedRequest(GrepReq::START_METASCAN, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "DATALOG") == 0) {
+ err = m_repState->protectedRequest(GrepReq::START_DATALOG, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "DATASCAN") == 0) {
+ err = m_repState->protectedRequest(GrepReq::START_DATASCAN, 0);
+ printError(err);
+ return true;
+ }
+ ndbout_c("Unknown argument: %s to command: %s",
+ allAfterFirstToken, firstToken);
+ return true;
+ }
+
+ if (strcmp(firstToken, "STOP") == 0) {
+ if (allAfterFirstToken == NULL) {
+ ndbout_c("Please use either 'STOP IMMEDIATELY' or 'STOP <epoch no>', "
+ "where\n<epoch no> is greater than or equal to "
+ "the last applied epoch.");
+ return true;
+ }
+
+ char * secondToken = strtok(allAfterFirstToken, " ");
+ char * subscription = strtok(NULL, "\0");
+ if (strcmp(secondToken, "SUBSCR") == 0 ||
+ strcmp(secondToken, "SUBSCRIPTION") == 0) {
+ char * sSubId = strtok(subscription," ");
+ char * sSubKey = strtok(NULL, "\0");
+ int subId = atoi(sSubId);
+ int subKey = atoi(sSubKey);
+ err = m_repState->protectedRequest(GrepReq::STOP_SUBSCR, subId, subKey );
+ printError(err);
+ return true;
+ }
+
+ if (strcmp(allAfterFirstToken, "SUBID") == 0) {
+ ndbout_c("Not implemented");
+ return true;
+ }
+
+
+ if (strcmp(allAfterFirstToken, "METALOG") == 0) {
+ err = m_repState->protectedRequest(GrepReq::STOP_METALOG, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "METASCAN") == 0) {
+ err = m_repState->protectedRequest(GrepReq::STOP_METASCAN, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "DATALOG") == 0) {
+ err = m_repState->protectedRequest(GrepReq::STOP_DATALOG, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "DATASCAN") == 0) {
+ err = m_repState->protectedRequest(GrepReq::STOP_DATASCAN, 0);
+ printError(err);
+ return true;
+ }
+ if (strcmp(allAfterFirstToken, "IM") == 0 ||
+ strcmp(allAfterFirstToken, "IMM") == 0 ||
+ strcmp(allAfterFirstToken, "IMMEDIATELY") == 0) {
+ err = m_repState->protectedRequest(GrepReq::STOP, 0);
+ printError(err);
+ return true;
+ }
+ Uint32 stopEpoch;
+ if (convert(allAfterFirstToken, &stopEpoch)) {
+ err = m_repState->protectedRequest(GrepReq::STOP, stopEpoch);
+ printError(err);
+ return true;
+ }
+
+ ndbout_c("Unknown argument: %s to command: %s",
+ allAfterFirstToken, firstToken);
+ return true;
+ }
+
+ ndbout_c("Unknown Command: %s", firstToken);
+ ndbout_c("Type HELP for help.");
+ ndbout << endl;
+ return true;
+}
diff --git a/ndb/src/rep/RepCommandInterpreter.hpp b/ndb/src/rep/RepCommandInterpreter.hpp
new file mode 100644
index 00000000000..398a7c0318c
--- /dev/null
+++ b/ndb/src/rep/RepCommandInterpreter.hpp
@@ -0,0 +1,45 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REP_COMMAND_INTERPRETER_HPP
+#define REP_COMMAND_INTERPRETER_HPP
+
+#include <editline/editline.h>
+
+#include <rep/RepComponents.hpp>
+#include <rep/state/RepState.hpp>
+
+/**
+ * @class RepCommandInterpreter
+ * @brief
+ */
+
+class RepCommandInterpreter {
+public:
+ RepCommandInterpreter(class RepComponents * comps);
+ ~RepCommandInterpreter();
+
+ bool readAndExecute();
+
+private:
+ char * readline_gets() const;
+ void request(Uint32 request);
+
+ class RepComponents * m_repComponents;
+ class RepState * m_repState;
+};
+
+#endif
diff --git a/ndb/src/rep/RepComponents.cpp b/ndb/src/rep/RepComponents.cpp
new file mode 100644
index 00000000000..04b2e0e5fa5
--- /dev/null
+++ b/ndb/src/rep/RepComponents.cpp
@@ -0,0 +1,138 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "RepComponents.hpp"
+
+RepComponents::RepComponents()
+{
+ /**
+ * @todo Fix proper reporting of errors
+ */
+ m_connectStringPS = NULL;
+ m_connectStringSS = NULL;
+
+ /**
+ * Phase 1: Containers, RepState
+ */
+ m_gciContainer = new GCIContainer(MAX_NODE_GROUPS);
+ if (!m_gciContainer) REPABORT("Could not allocate object");
+ m_gciContainerPS = new GCIContainerPS(MAX_NODE_GROUPS);
+ if (!m_gciContainerPS) REPABORT("Could not allocate object");
+ m_repState = new RepState();
+ if (!m_repState) REPABORT("Could not allocate object");
+
+ /**
+ * Phase 2: PS
+ */
+ m_transPS = new TransPS(m_gciContainerPS);
+ if (!m_transPS) REPABORT("Could not allocate object");
+
+
+ m_extAPI = new ExtAPI();
+ if (!m_extAPI) REPABORT("Could not allocate object");
+
+ m_extNDB = new ExtNDB(m_gciContainerPS, m_extAPI);
+ if (!m_extNDB) REPABORT("Could not allocate object");
+
+ /**
+ * Phase 3: SS
+ */
+ m_transSS = new TransSS(m_gciContainer, m_repState);
+ if (!m_transSS) REPABORT("Could not allocate object");
+ m_appNDB = new AppNDB(m_gciContainer, m_repState);
+ if (!m_appNDB) REPABORT("Could not allocate object");
+
+ /**
+ * Phase 4: Requestor
+ */
+ m_requestor = new Requestor(m_gciContainer, m_appNDB, m_repState);
+ if (!m_requestor) REPABORT("Could not allocate object");
+
+ /**
+ * Phase 5
+ */
+ m_repState->init(m_transSS->getRepSender());
+ m_repState->setApplier(m_appNDB);
+ m_repState->setGCIContainer(m_gciContainer);
+
+ m_requestor->setRepSender(m_transSS->getRepSender());
+
+ m_extNDB->setRepSender(m_transPS->getRepSender());
+
+ m_transPS->setGrepSender(m_extNDB->getGrepSender());
+}
+
+RepComponents::~RepComponents()
+{
+ if (m_requestor) delete m_requestor;
+
+ if (m_appNDB) delete m_appNDB;
+ if (m_extNDB) delete m_extNDB;
+ if (m_extAPI) delete m_extAPI;
+
+ if (m_repState) delete m_repState;
+
+ if (m_transPS) delete m_transPS;
+ if (m_transSS) delete m_transSS;
+
+ if (m_gciContainer) delete m_gciContainer;
+ if (m_gciContainerPS) delete m_gciContainerPS;
+}
+
+int
+RepComponents::connectPS()
+{
+ /**
+ * @todo Fix return values of this function
+ */
+
+ /**
+ * Phase 1: TransporterFacade 1, Block number: 2 (PS)
+ */
+ if (!m_extNDB->init(m_connectStringPS)) return -1;
+
+ /**
+ * Phase 2: TransporterFacade 2, Block number: 2 (PS)
+ */
+ m_transPS->init(m_transSS->getTransporterFacade(), m_connectStringPS);
+
+ return 0;
+}
+
+int
+RepComponents::connectSS()
+{
+ /**
+ * @todo Fix return values of this function
+ */
+
+ /**
+ * Phase 1: TransporterFacade 1, Block number: 1 (SS)
+ */
+ m_appNDB->init(m_connectStringSS);
+
+ /**
+ * Phase 2: TransporterFacade 2, Block number: 1 (SS)
+ */
+ m_transSS->init(m_connectStringSS);
+
+ /**
+ * Phase 3: Has no TransporterFacade, just starts thread
+ */
+ m_requestor->init();
+
+ return 0;
+}
diff --git a/ndb/src/rep/RepComponents.hpp b/ndb/src/rep/RepComponents.hpp
new file mode 100644
index 00000000000..8b24858271b
--- /dev/null
+++ b/ndb/src/rep/RepComponents.hpp
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REPCOMPONENTS_HPP
+#define REPCOMPONENTS_HPP
+
+#include <rep/adapters/ExtNDB.hpp>
+#include <rep/adapters/AppNDB.hpp>
+#include <rep/transfer/TransPS.hpp>
+#include <rep/transfer/TransSS.hpp>
+#include <rep/Requestor.hpp>
+#include <rep/state/RepState.hpp>
+
+#include <rep/rep_version.hpp>
+
+
+/**
+ * Connection data
+ */
+class RepComponents {
+public:
+ RepComponents();
+ ~RepComponents();
+
+ int connectPS();
+ int connectSS();
+
+ ExtNDB * m_extNDB;
+ ExtAPI * m_extAPI;
+ TransPS * m_transPS;
+
+ TransSS * m_transSS;
+ AppNDB * m_appNDB;
+
+ Requestor * m_requestor;
+
+ GCIContainer * m_gciContainer;
+ GCIContainerPS * m_gciContainerPS;
+
+ char * m_connectStringPS;
+ char * m_connectStringSS;
+
+ RepState * getRepState() { return m_repState; }
+private:
+ RepState * m_repState;
+};
+
+#endif
diff --git a/ndb/src/rep/RepMain.cpp b/ndb/src/rep/RepMain.cpp
new file mode 100644
index 00000000000..e00f6c0040c
--- /dev/null
+++ b/ndb/src/rep/RepMain.cpp
@@ -0,0 +1,98 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <assert.h>
+#include <stdlib.h>
+
+#include <NdbApiSignal.hpp>
+#include <getarg.h>
+
+#include <rep/RepComponents.hpp>
+
+#include "rep_version.hpp"
+#include <rep/RepCommandInterpreter.hpp>
+#include <rep/RepApiInterpreter.hpp>
+
+
+int
+main(int argc, const char **argv)
+{
+ RepComponents comps;
+ RepCommandInterpreter cmd(&comps);
+
+
+ int helpFlag = false;
+ int noConnectFlag = false;
+ int onlyPrimaryFlag = false;
+ int onlyStandbyFlag = false;
+ int port = 18000;
+ replogEnabled = false;
+
+ struct getargs args[] = {
+ { "psc", '1', arg_string, &comps.m_connectStringPS,
+ "Connect string", "connectstring" },
+ { "ssc", '2', arg_string, &comps.m_connectStringSS,
+ "Connect string", "connectstring" },
+ { "port", 'p', arg_integer, &port,
+ "port for rep api. Default 18000", "" },
+ { "usage", '?', arg_flag, &helpFlag,
+ "Print help", "" },
+/* @todo
+ { "noConnect", 'n', arg_flag, &noConnectFlag,
+ "Do not connect adapters", "" },
+*/
+ { "debug", 'd', arg_flag, &replogEnabled,
+ "Enable debug printouts on console", "" },
+ { "onlyStandby", 's', arg_flag, &onlyStandbyFlag,
+ "Let Replication Server view DBMS as standby (destination) system only",
+ "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "\nWhen working as a primary system node, this program receives\n"\
+ "records from the primary NDB Cluster and forwards them to\n"\
+ "the standby system.\n\n"\
+ "When working as a standby system node, this program receives\n"\
+ "records from another replication node and inserts them into\n"\
+ "the standby NDB Cluster.\n\n"\
+ "Example: ndb_rep --psc=\"nodeid=3;host=localhost:10000\"\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ //argv[optind] == NULL ||
+ helpFlag)
+ {
+ arg_printusage(args, num_args, argv[0], desc);
+ return -1; //NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ RepApiInterpreter api(&comps,port);
+ api.startInterpreter();
+
+ /**************************
+ * Command-line interface *
+ **************************/
+ if (!noConnectFlag && !onlyPrimaryFlag) comps.connectSS();
+ if (!noConnectFlag && !onlyStandbyFlag) comps.connectPS();
+
+
+ while (true) {
+ if(!cmd.readAndExecute()) {
+ api.stopInterpreter();
+ exit(1);
+ }
+ }
+}
diff --git a/ndb/src/rep/Requestor.cpp b/ndb/src/rep/Requestor.cpp
new file mode 100644
index 00000000000..af16fc33844
--- /dev/null
+++ b/ndb/src/rep/Requestor.cpp
@@ -0,0 +1,225 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <assert.h>
+#include "Requestor.hpp"
+#include "ConfigRetriever.hpp"
+
+#include <NdbApiSignal.hpp>
+
+#include <signaldata/RepImpl.hpp>
+#include <signaldata/GrepImpl.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/SumaImpl.hpp>
+
+#include <AttributeHeader.hpp>
+#include <rep/rep_version.hpp>
+
+#define TIME_BETWEEN_EXECUTES_MS 250
+
+/*
+ * @todo The requestor still has a TF, but this is not used...
+ * (We will need a (set of) TF(s) for REP-REP
+ * on the same system though....)
+ */
+
+
+/*****************************************************************************
+ * Constructor / Destructor / Init
+ *****************************************************************************/
+Requestor::Requestor(GCIContainer * gciContainer,
+ AppNDB * appNDB,
+ RepState * repState)
+{
+ m_gciContainer = gciContainer;
+ m_applier = appNDB;
+ m_repState = repState;
+
+ //m_grepSender = new ExtSender();
+ //if (!m_grepSender) REPABORT("");
+
+ m_repState->setSubscriptionRequests(&requestCreateSubscriptionId,
+ &requestCreateSubscription,
+ &requestRemoveSubscription);
+ m_repState->setIntervalRequests(&requestTransfer,
+ &requestApply,
+ &requestDeleteSS,
+ &requestDeletePS);
+ m_repState->setStartRequests(&requestStartMetaLog,
+ &requestStartDataLog,
+ &requestStartMetaScan,
+ &requestStartDataScan,
+ &requestEpochInfo);
+}
+
+Requestor::~Requestor() {
+ //delete m_grepSender;
+}
+
+bool
+Requestor::init(const char * connectString)
+{
+ m_signalExecThread = NdbThread_Create(signalExecThread_C,
+ (void **)this,
+ 32768,
+ "Requestor_Service",
+ NDB_THREAD_PRIO_LOW);
+
+ if (m_signalExecThread == NULL)
+ return false;
+
+ return true;
+}
+
+/*****************************************************************************
+ * Signal Queue Executor
+ *****************************************************************************/
+
+void *
+Requestor::signalExecThread_C(void *g) {
+
+ Requestor *requestor = (Requestor*)g;
+ requestor->signalExecThreadRun();
+ NdbThread_Exit(0);
+
+ /* NOTREACHED */
+ return 0;
+}
+
+class SigMatch
+{
+public:
+ int gsn;
+ void (Requestor::* function)(NdbApiSignal *signal);
+
+ SigMatch() { gsn = 0; function = NULL; };
+
+ SigMatch(int _gsn, void (Requestor::* _function)(NdbApiSignal *signal)) {
+ gsn = _gsn;
+ function = _function;
+ };
+
+ bool check(NdbApiSignal *signal) {
+ if(signal->readSignalNumber() == gsn)
+ return true;
+ return false;
+ };
+};
+
+void
+Requestor::signalExecThreadRun()
+{
+ while(1)
+ {
+ /**
+ * @todo Here we would like to measure the usage size of the
+ * receive buffer of TransSS. If the buffer contains
+ * more than X signals (maybe 1k or 10k), then we should
+ * not do a protectedExecute.
+ * By having the usage size measure thingy,
+ * we avoid having the Requestor requesting more
+ * things than the TransSS can handle.
+ * /Lars
+ *
+ * @todo A different implementation of this functionality
+ * would be to send a signal to myself when the protected
+ * execute is finished. This solution could be
+ * discussed.
+ * /Lars
+ */
+ m_repState->protectedExecute();
+ NdbSleep_MilliSleep(TIME_BETWEEN_EXECUTES_MS);
+ }
+}
+
+void
+Requestor::sendSignalRep(NdbApiSignal * s) {
+ m_repSender->sendSignal(s);
+}
+
+void
+Requestor::execSignal(void* executorObj, NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]){
+
+ Requestor * executor = (Requestor*)executorObj;
+
+ const Uint32 gsn = signal->readSignalNumber();
+ const Uint32 len = signal->getLength();
+
+ NdbApiSignal * s = new NdbApiSignal(executor->m_ownRef);
+ switch (gsn) {
+ case GSN_REP_GET_GCI_CONF:
+ case GSN_REP_GET_GCI_REQ:
+ case GSN_REP_GET_GCIBUFFER_REQ:
+ case GSN_REP_INSERT_GCIBUFFER_REQ:
+ case GSN_REP_CLEAR_SS_GCIBUFFER_REQ:
+ case GSN_REP_CLEAR_PS_GCIBUFFER_REQ:
+ case GSN_REP_DROP_TABLE_REQ:
+ case GSN_GREP_SUB_CREATE_REQ:
+ case GSN_GREP_SUB_START_REQ:
+ case GSN_GREP_SUB_SYNC_REQ:
+ case GSN_GREP_SUB_REMOVE_REQ:
+ case GSN_GREP_CREATE_SUBID_REQ:
+ s->set(0, PSREPBLOCKNO, gsn, len);
+ memcpy(s->getDataPtrSend(), signal->getDataPtr(), 4 * len);
+ executor->m_signalRecvQueue.receive(s);
+ break;
+ default:
+ REPABORT1("Illegal signal received in execSignal", gsn);
+ }
+#if 0
+ ndbout_c("Requestor: Inserted signal into queue (GSN: %d, Len: %d)",
+ signal->readSignalNumber(), len);
+#endif
+}
+
+void
+Requestor::execNodeStatus(void* obj, Uint16 nodeId,
+ bool alive, bool nfCompleted)
+{
+ //Requestor * thisObj = (Requestor*)obj;
+
+ RLOG(("Node changed status (NodeId %d, Alive %d, nfCompleted %d)",
+ nodeId, alive, nfCompleted));
+
+ if(alive) {
+ /**
+ * Connected - set node as connected
+ *
+ * @todo Make it possible to have multiple External REP nodes
+ */
+#if 0
+ for(Uint32 i=0; i<thisObj->m_nodeConnectList.size(); i++) {
+ if(thisObj->m_nodeConnectList[i]->nodeId == nodeId)
+ thisObj->m_nodeConnectList[i]->connected = true;
+ }
+ thisObj->m_grepSender->setNodeId(thisObj->m_nodeConnectList[0]->nodeId);
+#endif
+ }
+
+ if(!alive && !nfCompleted){
+ /**
+ * ???
+ */
+ }
+
+ if(!alive && nfCompleted){
+ /**
+ * Re-connect
+ */
+ }
+}
diff --git a/ndb/src/rep/Requestor.hpp b/ndb/src/rep/Requestor.hpp
new file mode 100644
index 00000000000..ba753be60f2
--- /dev/null
+++ b/ndb/src/rep/Requestor.hpp
@@ -0,0 +1,153 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REQUESTOR_HPP
+#define REQUESTOR_HPP
+
+#include <TransporterDefinitions.hpp>
+#include <TransporterFacade.hpp>
+#include <ClusterMgr.hpp>
+#include <API.hpp>
+#include <Vector.hpp>
+#include <stdio.h>
+#include <GrepError.hpp>
+
+#include <rep/storage/GCIContainer.hpp>
+
+/**
+ * @todo Remove this dependency
+ */
+#include <rep/adapters/AppNDB.hpp>
+
+#include <rep/SignalQueue.hpp>
+#include <rep/ExtSender.hpp>
+
+
+/**
+ * @class Requestor
+ * @brief Connects to GREP Coordinator on the standby system
+ */
+class Requestor {
+public:
+ /***************************************************************************
+ * Constructor / Destructor / Init
+ ***************************************************************************/
+ Requestor(GCIContainer * gciContainer, AppNDB * applier, RepState * repSt);
+ ~Requestor();
+ bool init(const char * connectString = NULL);
+
+ /***************************************************************************
+ * Public Methods
+ ***************************************************************************/
+ void setRepSender(ExtSender * es) { m_repSender = es; };
+
+private:
+ static void * signalExecThread_C(void *); ///< SignalQueue executor thread
+ void signalExecThreadRun();
+
+ static void execSignal(void* executorObj, NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]);
+ static void execNodeStatus(void* executorObj, NodeId, bool alive,
+ bool nfCompleted);
+
+ void sendSignalRep(NdbApiSignal *);
+ void sendSignalGrep(NdbApiSignal *);
+
+ void connectToNdb();
+
+ /***************************************************************************
+ * Signal Executors
+ ***************************************************************************/
+ void execREP_GET_GCIBUFFER_CONF(NdbApiSignal*);
+ void execREP_CLEAR_GCIBUFFER_REP(NdbApiSignal*);
+ void execREP_INSERT_GCIBUFFER_REQ(NdbApiSignal*);
+ void execREP_CLEAR_SS_GCIBUFFER_REQ(NdbApiSignal*);
+ void execREP_DROP_TABLE_REQ(NdbApiSignal*);
+
+ /***************************************************************************
+ * Signal Executors 2
+ ***************************************************************************/
+ void execGREP_CREATE_SUBID_CONF(NdbApiSignal*);
+ void execGREP_CREATE_SUBID_REF(NdbApiSignal*);
+ void createSubscription(NdbApiSignal*);
+ void createSubscriptionId(NdbApiSignal*);
+ void execGREP_SUB_CREATE_CONF(NdbApiSignal*);
+ void execGREP_SUB_CREATE_REF(NdbApiSignal*);
+ void execGREP_SUB_START_CONF(NdbApiSignal*);
+ void execGREP_SUB_START_REF(NdbApiSignal*);
+ void removeSubscription(NdbApiSignal*);
+ void execGREP_SUB_REMOVE_REF(NdbApiSignal*);
+ void execGREP_SUB_SYNC_CONF(NdbApiSignal*);
+ void execGREP_SUB_SYNC_REF(NdbApiSignal*);
+ void execREP_CLEAR_SS_GCIBUFFER_CONF(NdbApiSignal*);
+ void execREP_CLEAR_SS_GCIBUFFER_REF(NdbApiSignal*);
+ void execREP_GET_GCIBUFFER_REF(NdbApiSignal*);
+ void execREP_DISCONNECT_REP(NdbApiSignal*);
+
+ /***************************************************************************
+ * Ref signal senders
+ ***************************************************************************/
+ void sendREP_INSERT_GCIBUFFER_REF(NdbApiSignal * signal,
+ Uint32 gci,
+ Uint32 nodeGrp,
+ GrepError::Code err);
+
+ void sendREP_CLEAR_SS_GCIBUFFER_REF(NdbApiSignal* signal,
+ Uint32 firstGCI,
+ Uint32 lastGCI,
+ Uint32 currentGCI,
+ Uint32 nodeGrp,
+ GrepError::Code err);
+
+ /***************************************************************************
+ * Private Variables
+ ***************************************************************************/
+ class SignalQueue m_signalRecvQueue;
+ struct NdbThread * m_signalExecThread;
+
+ RepState * m_repState;
+
+ Uint32 m_ownNodeId; ///< NodeId of this node
+ Uint32 m_ownBlockNo; ///< BlockNo of this "block"
+ BlockReference m_ownRef; ///< Reference to this
+
+ TransporterFacade * m_transporterFacade;
+
+ GCIContainer * m_gciContainer;
+
+ AppNDB * m_applier;
+ ExtSender * m_repSender;
+
+ friend void startSubscription(void * cbObj, NdbApiSignal* signal, int type);
+ friend void scanSubscription(void * cbObj, NdbApiSignal* signal, int type);
+
+ friend RepState::FuncRequestCreateSubscriptionId requestCreateSubscriptionId;
+ friend RepState::FuncRequestCreateSubscription requestCreateSubscription;
+ friend RepState::FuncRequestRemoveSubscription requestRemoveSubscription;
+
+ friend RepState::FuncRequestTransfer requestTransfer;
+ friend RepState::FuncRequestApply requestApply;
+ friend RepState::FuncRequestDeleteSS requestDeleteSS;
+ friend RepState::FuncRequestDeletePS requestDeletePS;
+
+ friend RepState::FuncRequestStartMetaLog requestStartMetaLog;
+ friend RepState::FuncRequestStartDataLog requestStartDataLog;
+ friend RepState::FuncRequestStartMetaScan requestStartMetaScan;
+ friend RepState::FuncRequestStartDataScan requestStartDataScan;
+ friend RepState::FuncRequestEpochInfo requestEpochInfo;
+};
+
+#endif
diff --git a/ndb/src/rep/RequestorSubscriptions.cpp b/ndb/src/rep/RequestorSubscriptions.cpp
new file mode 100644
index 00000000000..75b41fae037
--- /dev/null
+++ b/ndb/src/rep/RequestorSubscriptions.cpp
@@ -0,0 +1,60 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Requestor.hpp"
+
+#include <signaldata/GrepImpl.hpp>
+#include <signaldata/SumaImpl.hpp>
+
+#include <rep/rep_version.hpp>
+
+/*****************************************************************************
+ * Create Subscription Id
+ *****************************************************************************/
+
+
+/*****************************************************************************
+ * Create Subscription
+ *****************************************************************************/
+
+
+/*****************************************************************************
+ * Start Subscription
+ *****************************************************************************/
+
+/*****************************************************************************
+ * Remove Subscription
+ *****************************************************************************/
+
+void
+Requestor::execGREP_SUB_REMOVE_REF(NdbApiSignal* signal)
+{
+#if 0
+ GrepSubRemoveRef * const ref = (GrepSubRemoveRef *)signal->getDataPtr();
+ Uint32 subId = ref->subscriptionId;
+ Uint32 subKey = ref->subscriptionKey;
+ Uint32 err = ref->err;
+
+ signal->theData[0] = EventReport::GrepSubscriptionAlert;
+ signal->theData[1] = GrepEvent::GrepSS_SubRemoveRef;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ signal->theData[4] = (Uint32)err;
+ sendSignal(CMVMI_REF,GSN_EVENT_REP,signal, 5, JBB);
+#endif
+}
+
+
diff --git a/ndb/src/rep/SignalQueue.cpp b/ndb/src/rep/SignalQueue.cpp
new file mode 100644
index 00000000000..9b356a14b7d
--- /dev/null
+++ b/ndb/src/rep/SignalQueue.cpp
@@ -0,0 +1,106 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <string.h>
+
+#include "SignalQueue.hpp"
+
+SignalQueue::SignalQueue() {
+ m_mutex = NdbMutex_Create();
+ m_cond = NdbCondition_Create();
+ m_signalQueueHead = NULL;
+ m_queueSize = 0;
+}
+
+SignalQueue::~SignalQueue() {
+ {
+ Guard g(m_mutex);
+ while(m_signalQueueHead != NULL)
+ delete pop();
+ }
+ NdbMutex_Destroy(m_mutex);
+ m_mutex = NULL;
+ NdbCondition_Destroy(m_cond);
+ m_cond = NULL;
+}
+
+NdbApiSignal *
+SignalQueue::pop() {
+ NdbApiSignal *ret;
+
+ if(m_signalQueueHead == NULL)
+ return NULL;
+
+ ret = m_signalQueueHead->signal;
+
+ QueueEntry *old = m_signalQueueHead;
+ m_signalQueueHead = m_signalQueueHead->next;
+
+ delete old;
+ m_queueSize--;
+ return ret;
+}
+
+void
+SignalQueue::receive(void *me, NdbApiSignal *signal) {
+ SignalQueue *q = (SignalQueue *)me;
+ q->receive(signal);
+}
+
+void
+SignalQueue::receive(NdbApiSignal *signal) {
+ QueueEntry *n = new QueueEntry();
+ n->signal = signal;
+ n->next = NULL;
+
+ Guard guard(m_mutex);
+
+ if(m_signalQueueHead == NULL) {
+ m_signalQueueHead = n;
+ m_queueSize++;
+ NdbCondition_Broadcast(m_cond);
+ return;
+ }
+
+ QueueEntry *cur = m_signalQueueHead;
+
+ while(cur->next != NULL)
+ cur = cur->next;
+
+ cur->next = n;
+ m_queueSize++;
+ NdbCondition_Broadcast(m_cond);
+}
+
+NdbApiSignal *
+SignalQueue::waitFor(int gsn, NodeId nodeid, Uint32 timeout) {
+ Guard g(m_mutex);
+
+ if(m_signalQueueHead == NULL)
+ NdbCondition_WaitTimeout(m_cond, m_mutex, timeout);
+
+ if(m_signalQueueHead == NULL)
+ return NULL;
+
+ if(gsn != 0 && m_signalQueueHead->signal->readSignalNumber() != gsn)
+ return NULL;
+
+ if(nodeid != 0 &&
+ refToNode(m_signalQueueHead->signal->theSendersBlockRef) != nodeid)
+ return NULL;
+
+ return pop();
+}
diff --git a/ndb/src/rep/SignalQueue.hpp b/ndb/src/rep/SignalQueue.hpp
new file mode 100644
index 00000000000..5ebc78a3a37
--- /dev/null
+++ b/ndb/src/rep/SignalQueue.hpp
@@ -0,0 +1,117 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef __SIGNALQUEUE_HPP_INCLUDED__
+#define __SIGNALQUEUE_HPP_INCLUDED__
+
+#include <NdbApiSignal.hpp>
+#include <NdbMutex.h>
+#include <NdbCondition.h>
+#include <Vector.hpp>
+
+/* XXX Look for an already existing definition */
+#define DEFAULT_TIMEOUT 10000
+
+/**
+ * @class SignalQueue
+ * @brief
+ */
+class SignalQueue {
+public:
+ typedef void (* SignalHandler)(void *obj, int gsn, NdbApiSignal *signal);
+
+ SignalQueue();
+ ~SignalQueue();
+
+ /**
+ * Static wrapper making it possible to call receive without knowing the
+ * type of the receiver
+ */
+ static void receive(void *me, NdbApiSignal *signal);
+
+ /**
+ * Enqueues a signal, and notifies any thread waiting for signals.
+ */
+ void receive(NdbApiSignal *signal);
+
+ NdbApiSignal *waitFor(int gsn,
+ NodeId nodeid = 0,
+ Uint32 timeout = DEFAULT_TIMEOUT);
+ template<class T> bool waitFor(Vector<T> &t,
+ T *&handler,
+ NdbApiSignal *&signal,
+ Uint32 timeout = DEFAULT_TIMEOUT);
+
+ /**
+ * size()
+ */
+
+ Uint32 size() {return m_queueSize;};
+
+private:
+ NdbMutex *m_mutex; /* Locks all data in SignalQueue */
+ NdbCondition *m_cond; /* Notifies about new signal in the queue */
+
+ /**
+ * Returns the last recently received signal.
+ * Must be called with m_mutex locked.
+ *
+ * The caller takes responsibility for deleting the returned object.
+ *
+ * @returns NULL if failed, or a received signal
+ */
+ NdbApiSignal *pop();
+
+ class QueueEntry {
+ public:
+ NdbApiSignal *signal;
+ QueueEntry *next;
+ };
+ QueueEntry *m_signalQueueHead; /** Head of the queue.
+ * New entries added on the tail
+ */
+ Uint32 m_queueSize;
+};
+
+template<class T> bool
+SignalQueue::waitFor(Vector<T> &t,
+ T *&handler,
+ NdbApiSignal *&signal,
+ Uint32 timeout) {
+ Guard g(m_mutex);
+
+ if(m_signalQueueHead == NULL)
+ NdbCondition_WaitTimeout(m_cond, m_mutex, timeout);
+
+ if(m_signalQueueHead == NULL)
+ return false;
+
+ for(size_t i = 0; i < t.size(); i++) {
+ if(t[i].check(m_signalQueueHead->signal)) {
+ handler = &t[i];
+ signal = pop();
+ return true;
+ }
+ }
+
+ ndbout_c("SignalQueue: Queued signal without true check function (GSN: %d)",
+ m_signalQueueHead->signal->theVerId_signalNumber);
+ abort();
+
+ return false;
+}
+
+#endif /* !__SIGNALQUEUE_HPP_INCLUDED__ */
diff --git a/ndb/src/rep/TODO b/ndb/src/rep/TODO
new file mode 100644
index 00000000000..a2462fae6cd
--- /dev/null
+++ b/ndb/src/rep/TODO
@@ -0,0 +1,119 @@
+REQUIREMENTS
+------------
+- It should be possible to run two systems with replication using the
+ same configuration file on both systems.
+
+FEATURES TO IMPLEMENT
+---------------------
+- Fix so that execute and command uses ExtSender.
+ None of them should have their own signals, this should
+ instead by abstacted to the RepStateRequest layer.
+- Delete signals
+ GSN_REP_INSERT_GCIBUFFER_CONF
+ GSN_REP_INSERT_GCIBUFFER_REF
+- Fix so that all ExtSenders are set at one point in the code only.
+- Verify the following signals:
+ GSN_REP_INSERT_GCIBUFFER_REQ
+ GSN_REP_CLEAR_SS_GCIBUFFER_REQ
+ GSN_REP_DROP_TABLE_REQ
+- Fix all @todo's in the code
+- Remove all #if 1, #if 0 etc.
+- Fix correct usage of dbug package used in MySQL source code.
+- System table storing all info about channels
+- Think about how channels, subscriptions etc map to SUMA Subscriptions
+- TableInfoPS must be secured if SS REP is restarted and PS REP still
+ has all log records needed to sync. (This could be saved in a system
+ table instead of using the struct.)
+
+KNOWN BUGS AND LIMITATIONS
+--------------------------
+- REP#1: Non-consistency due to non-logging stop [LIMITATION]
+ Problem:
+ - Stopping replication in state other than "Logging" can
+ lead to a non-consistent state of the destination database
+ Suggested solution:
+ - Implement a cleanData flag (= false) that indicates that
+ this has happend.
+
+- REP#2: PS REP uses epochs from old subscription [BUG]
+ The following scenario can lead to a non-correct replication:
+ - Start replication X
+ - Wait until replication is in "Logging" state
+ - Kill SS REP
+ - Let PS REP be alive
+ - Start new replication Y
+ - Replication Y can use old PS REP epochs from replication X.
+ Suggested solution:
+ - Mark PS buffers with channel ids
+ - Make sure that all epoch requests use channel number in the requests.
+
+- REP#3: When having two node groups, there is sometimes 626 [FIXED]
+ Problem:
+ - Sometimes (when doing updated) there is 626 error code when
+ using 2 node groups.
+ - 626 = Tuple does not exists.
+ - Current code in RepState.cpp is:
+ if(s == Channel::App &&
+ m_channel.getState() == Channel::DATASCAN_COMPLETED &&
+ i.last() >= m_channel.getDataScanEpochs().last() &&
+ i.last() >= m_channel.getMetaScanEpochs().last())
+ {
+ m_channel.setState(Channel::LOG);
+ disableAutoStart();
+ }
+ When the system gets into LOG state, force flag is turned off
+ Suggested solution:
+ - During DATASCAN, force=true (i.e. updates are treated as writes,
+ deletes error due to non-existing tuple are ignored)
+ - The code above must take ALL node groups into account.
+
+- REP#4: User requests sometime vanish when DB node is down [LIMITATION]
+ Problem:
+ - PS REP node does not always REF when no connection to GREP exists
+ Suggested solution:
+ - All REP->GREP signalsends should be checked. If they return <0,
+ then a REF signal should be returned.
+
+- REP#5: User requests sometime vanish when PS REP is down [BUG]
+ Scenario:
+ - Execute "Start" with PS REP node down
+ Solution:
+ - When start is executed, the connect flag should be checked
+
+- REP#6: No warning if table exists [Lars, BUG!]
+ Problem:
+ - There is no warning if a replicated table already exists in the
+ database.
+ Suggested solution:
+ - Print warning
+ - Set cleanData = false
+
+- REP#7: Starting 2nd subscription crashes DB node (Grep.cpp:994) [FIXED]
+ Scenario:
+ - Start replication
+ - Wait until replication is in "Logging" state
+ - Kill SS REP
+ - Let PS REP be alive
+ - Start new replication
+ - Now GREP crashes in Grep.cpp:994.
+ Suggested fix:
+ - If a new subscription is requested with same subscriberData
+ as already exists, then SUMA (or GREP) sends a REF signal
+ indicating that SUMA does not allow a new subscription to be
+ created. [Now no senderData is sent from REP.]
+
+- REP#8: Dangling subscriptions in GREP/SUMA [Johan,LIMITATION]
+ Problem:
+ - If both REP nodes die, then there is no possibility to remove
+ subscriptions from GREP/SUMA
+ Suggested solution 1:
+ - Fix so that GREP/SUMA can receive a subscription removal
+ signal with subid 0. This means that ALL subscriptions are
+ removed. This meaning should be documented in the
+ signaldata class.
+ - A new user command "STOP ALL" is implemented that sends
+ a request to delete all subscriptions.
+ Suggested solution 2:
+ - When GREP detects that ALL PS REP nodes associated with a s
+ subscription are killed, then that subscription should be
+ deleted.
diff --git a/ndb/src/rep/adapters/AppNDB.cpp b/ndb/src/rep/adapters/AppNDB.cpp
new file mode 100644
index 00000000000..abb146d921f
--- /dev/null
+++ b/ndb/src/rep/adapters/AppNDB.cpp
@@ -0,0 +1,581 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "AppNDB.hpp"
+#include <ConfigRetriever.hpp>
+#include <AttributeHeader.hpp>
+#include <NdbOperation.hpp>
+#include <NdbDictionaryImpl.hpp>
+
+#include <signaldata/RepImpl.hpp>
+#include <TransporterFacade.hpp>
+#include <trigger_definitions.h>
+#include <rep/storage/GCIPage.hpp>
+#include <rep/storage/GCIBuffer.hpp>
+#include <rep/rep_version.hpp>
+
+/*****************************************************************************
+ * Constructor / Destructor / Init
+ *****************************************************************************/
+
+AppNDB::~AppNDB()
+{
+ delete m_tableInfoPs;
+ delete m_ndb;
+ m_tableInfoPs = 0;
+}
+
+AppNDB::AppNDB(GCIContainer * gciContainer, RepState * repState)
+{
+ m_gciContainer = gciContainer;
+ m_repState = repState;
+ m_cond = NdbCondition_Create();
+ m_started = true;
+}
+
+void
+AppNDB::init(const char* connectString) {
+
+ // NdbThread_SetConcurrencyLevel(1+ 2);
+ Ndb::useFullyQualifiedNames(false);
+
+ m_ndb = new Ndb("");
+ m_ndb->setConnectString(connectString);
+ /**
+ * @todo Set proper max no of transactions?? needed?? Default 12??
+ */
+ m_ndb->init(2048);
+ m_dict = m_ndb->getDictionary();
+
+ m_ownNodeId = m_ndb->getNodeId();
+
+ ndbout << "-- NDB Cluster -- REP node " << m_ownNodeId << " -- Version "
+ << REP_VERSION_ID << " --" << endl;
+ ndbout_c("Connecting to NDB Cluster...");
+ if (m_ndb->waitUntilReady() != 0){
+ REPABORT("NDB Cluster not ready for connections");
+ }
+ ndbout_c("Phase 1 (AppNDB): Connection 1 to NDB Cluster opened (Applier)");
+
+ m_tableInfoPs = new TableInfoPs();
+
+ m_applierThread = NdbThread_Create(runAppNDB_C,
+ (void**)this,
+ 32768,
+ "AppNDBThread",
+ NDB_THREAD_PRIO_LOW);
+}
+
+
+/*****************************************************************************
+ * Threads
+ *****************************************************************************/
+
+extern "C"
+void*
+runAppNDB_C(void * me)
+{
+ ((AppNDB *) me)->threadMainAppNDB();
+ NdbThread_Exit(0);
+ return me;
+}
+
+void
+AppNDB::threadMainAppNDB() {
+ MetaRecord * mr;
+ LogRecord * lr;
+ GCIBuffer::iterator * itBuffer;
+ GCIPage::iterator * itPage;
+ GCIBuffer * buffer;
+ GCIPage * page;
+ Uint32 gci=0;
+
+ bool force;
+ while(true){
+
+ m_gciBufferList.lock();
+ if(m_gciBufferList.size()==0)
+ NdbCondition_Wait(m_cond, m_gciBufferList.getMutex());
+ m_gciBufferList.unlock();
+
+ /**
+ * Do nothing if we are not started!
+ */
+ if(!m_started)
+ continue;
+
+ if(m_gciBufferList.size()>0) {
+ m_gciBufferList.lock();
+ buffer = m_gciBufferList[0];
+ assert(buffer!=0);
+ if(buffer==0) {
+ m_gciBufferList.unlock();
+// stopApplier(GrepError::REP_APPLY_NULL_GCIBUFFER);
+ return;
+ }
+ m_gciBufferList.unlock();
+
+ RLOG(("Applying %d:[%d]", buffer->getId(), buffer->getGCI()));
+ gci = buffer->getGCI();
+ /**
+ * Do stuff with buffer
+ */
+
+ force = buffer->m_force;
+ itBuffer = new GCIBuffer::iterator(buffer);
+ page = itBuffer->first();
+
+ Record * record;
+ while(page!=0 && m_started) {
+
+ itPage = new GCIPage::iterator(page);
+ record = itPage->first();
+
+ while(record!=0 && m_started) {
+ switch(Record::RecordType(record->recordType)) {
+ case Record::META:
+ mr = (MetaRecord*)record;
+ if(applyMetaRecord(mr, gci) < 0){
+ /**
+ * If we fail with a meta record then
+ * we should fail the replication!
+ */
+ //stopApplier(GrepError::REP_APPLY_METARECORD_FAILED);
+ }
+ break;
+ case Record::LOG:
+ lr = (LogRecord*)record;
+ if(applyLogRecord(lr, force, gci) < 0) {
+ /**
+ * If we fail to apply a log record AND
+ * we have sent a ref to repstate event,
+ * then we should not try to apply another one!
+ */
+// stopApplier(GrepError::REP_APPLY_LOGRECORD_FAILED);
+ }
+ break;
+ default:
+ REPABORT("Illegal record type");
+ };
+ record = itPage->next();
+ }
+ delete itPage;
+ itPage = 0;
+ page = itBuffer->next();
+ }
+
+ m_gciBufferList.erase(0, true);
+ /**
+ * "callback" to RepState to send REP_INSERT_GCIBUFFER_CONF
+ */
+ m_repState->eventInsertConf(buffer->getGCI(), buffer->getId());
+ delete itBuffer;
+ itBuffer = 0;
+ mr = 0;
+ lr = 0;
+ page = 0;
+ buffer = 0;
+ }
+ }
+
+
+}
+
+void AppNDB::startApplier(){
+ m_started = true;
+}
+
+
+void AppNDB::stopApplier(GrepError::Code err){
+ m_started = false;
+ m_repState->eventInsertRef(0,0,0, err);
+}
+
+
+GrepError::Code
+AppNDB::applyBuffer(Uint32 nodeGrp, Uint32 epoch, Uint32 force)
+{
+ m_gciBufferList.lock();
+
+ GCIBuffer * buffer = m_gciContainer->getGCIBuffer(epoch, nodeGrp);
+ if (buffer == NULL) {
+ RLOG(("WARNING! Request to apply NULL buffer %d[%d]. Force %d",
+ nodeGrp, epoch, force));
+ return GrepError::NO_ERROR;
+ }
+ if (!buffer->isComplete()) {
+ RLOG(("WARNING! Request to apply non-complete buffer %d[%d]. Force %d",
+ nodeGrp, epoch, force));
+ return GrepError::REP_APPLY_NONCOMPLETE_GCIBUFFER;
+ }
+ buffer->m_force = force;
+
+ assert(buffer!=0);
+ m_gciBufferList.push_back(buffer, false);
+ NdbCondition_Broadcast(m_cond);
+ m_gciBufferList.unlock();
+ return GrepError::NO_ERROR;
+}
+
+int
+AppNDB::applyLogRecord(LogRecord* lr, bool force, Uint32 gci)
+{
+#if 0
+ RLOG(("Applying log record (force %d, Op %d, GCI %d)",
+ force, lr->operation, gci));
+#endif
+
+ int retries =0;
+ retry:
+ if(retries == 10) {
+ m_repState->eventInsertRef(gci, 0, lr->tableId,
+ GrepError::REP_APPLIER_EXECUTE_TRANSACTION);
+ return -1;
+ }
+ NdbConnection * trans = m_ndb->startTransaction();
+ if (trans == NULL) {
+ /**
+ * Transaction could not be started
+ * @todo Handle the error by:
+ * 1. Return error code
+ * 2. Print log message
+ * 3. On higher level indicate that DB has been tainted
+ */
+ ndbout_c("AppNDB: Send the following error msg to NDB Cluster support");
+ reportNdbError("Cannot start transaction!", trans->getNdbError());
+ m_repState->eventInsertRef(gci, 0, 0,
+ GrepError::REP_APPLIER_START_TRANSACTION);
+ REPABORT("Can not start transaction");
+ }
+
+ /**
+ * Resolve table name based on table id
+ */
+ const Uint32 tableId = lr->tableId;
+ const char * tableName = m_tableInfoPs->getTableName(tableId);
+
+ /**
+ * Close trans and return if it is systab_0.
+ */
+ if (tableId == 0) {
+ RLOG(("WARNING! System table log record received"));
+ m_ndb->closeTransaction(trans);
+ return -1;
+ }
+
+ if (tableName==0) {
+ /**
+ * Table probably does not exist
+ * (Under normal operation this should not happen
+ * since log records should not appear unless the
+ * table has been created.)
+ *
+ * @todo Perhaps the table is not cached due to a restart,
+ * so let's check in the dictionary if it exists.
+ */
+ m_ndb->closeTransaction(trans);
+ m_repState->eventInsertRef(gci, 0, tableId,
+ GrepError::REP_APPLIER_NO_TABLE);
+ return -1;
+ }
+
+ const NdbDictionary::Table * table = m_dict->getTable(tableName);
+
+ NdbOperation * op = trans->getNdbOperation(tableName);
+ if (op == NULL) {
+ ndbout_c("AppNDB: Send the following error msg to NDB Cluster support");
+ reportNdbError("Cannot get NdbOperation record",
+ trans->getNdbError());
+ m_repState->eventInsertRef(gci,0,tableId,
+ GrepError::REP_APPLIER_NO_OPERATION);
+ REPABORT("Can not get NdbOperation record");
+ }
+
+ int check=0;
+ switch(lr->operation) {
+ case TriggerEvent::TE_INSERT: // INSERT
+ check = op->insertTuple();
+ break;
+ case TriggerEvent::TE_DELETE: // DELETE
+ check = op->deleteTuple();
+ break;
+ case TriggerEvent::TE_UPDATE: // UPDATE
+ if (force) {
+ check = op->writeTuple();
+ } else {
+ check = op->updateTuple();
+ }
+ break;
+ case TriggerEvent::TE_CUSTOM: //SCAN
+ check = op->writeTuple();
+ break;
+ default:
+ m_ndb->closeTransaction(trans);
+ return -1;
+ };
+
+ if (check<0) {
+ ndbout_c("AppNDB: Something is weird");
+ }
+
+ /**
+ * @todo index inside LogRecord struct somewhat prettier
+ * Now it 4 (sizeof(Uint32)), and 9 the position inside the struct
+ * where the data starts.
+ */
+ AttributeHeader * ah=(AttributeHeader *)((char *)lr + sizeof(Uint32) * 9);
+ AttributeHeader *end = (AttributeHeader *)(ah + lr->attributeHeaderWSize);
+ Uint32 * dataPtr = (Uint32 *)(end);
+
+ /**
+ * @note attributeheader for operaration insert includes a duplicate
+ * p.k. The quick fix for this problem/bug is to skip the first set of
+ * of p.k, and start from the other set of P.Ks. Data is duplicated for
+ * the p.k.
+ */
+ if (lr->operation == 0) {
+ for(int i = 0; i< table->getNoOfPrimaryKeys(); i++) {
+ ah+=ah->getHeaderSize();
+ dataPtr = dataPtr + ah->getDataSize();
+ }
+ }
+
+ while (ah < end) {
+ const NdbDictionary::Column * column =
+ table->getColumn(ah->getAttributeId());
+ /**
+ * @todo: Here is a limitation. I don't care if it is a tuplekey
+ * that is autogenerated or an ordinary pk. I just whack it in.
+ * However, this must be examined.
+ */
+ if(column->getPrimaryKey()) {
+ if(op->equal(ah->getAttributeId(), (const char *)dataPtr) < 0) {
+ ndbout_c("AppNDB: Equal failed id %d op %d name %s, gci %d force %d",
+ ah->getAttributeId(),
+ lr->operation,
+ column->getName(), gci, force);
+ reportNdbError("Equal!", trans->getNdbError());
+ }
+
+ } else {
+ if(op->setValue(ah->getAttributeId(), (const char *)dataPtr) < 0)
+ ndbout_c("AppNDB: setvalue failed id %d op %d name %s, gci %d force %d",
+ ah->getAttributeId(),
+ lr->operation,
+ column->getName(), gci, force);
+ }
+
+ dataPtr = dataPtr + ah->getDataSize();
+ ah = ah + ah->getHeaderSize() ;
+ }
+
+ if(trans->execute(Commit) != 0) {
+ /**
+ * Transaction commit failure
+ */
+ const NdbError err = trans->getNdbError();
+ m_ndb->closeTransaction(trans);
+ switch(err.status){
+ case NdbError::Success:
+ {
+ m_repState->eventInsertRef(gci, 0, tableId,
+ GrepError::REP_APPLIER_EXECUTE_TRANSACTION);
+ return -1;
+ }
+ break;
+ case NdbError::TemporaryError:
+ {
+ NdbSleep_MilliSleep(50);
+ retries++;
+ goto retry;
+ }
+ break;
+ case NdbError::UnknownResult:
+ {
+ ndbout_c("AppNDB: Send the following error msg to NDB Cluster support");
+ reportNdbError("Execute transaction failed!",
+ trans->getNdbError());
+ m_repState->eventInsertRef(gci, 0, tableId,
+ GrepError::REP_APPLIER_EXECUTE_TRANSACTION);
+ return -1;
+ }
+ break;
+ case NdbError::PermanentError:
+ {
+ if(err.code == 626) {
+ if(force && lr->operation == TriggerEvent::TE_DELETE) /**delete*/ {
+ /**tuple was not found. Ignore this, since
+ * we are trying to apply a "delete a tuple"-log record before
+ * having applied the scan data.
+ */
+ return -1;
+ }
+ }
+
+ ndbout_c("AppNDB: Send the following error msg to NDB Cluster support"); reportNdbError("Execute transaction failed!",
+ trans->getNdbError());
+ ndbout_c("\n\nAppNDB: RepNode will now crash.");
+ m_ndb->closeTransaction(trans);
+ m_repState->eventInsertRef(gci, 0, tableId,
+ GrepError::REP_APPLIER_EXECUTE_TRANSACTION);
+ return -1;
+ }
+ break;
+ }
+ }
+
+ /**
+ * No errors. Close transaction and continue in applierThread.
+ */
+ m_ndb->closeTransaction(trans);
+ return 1;
+}
+
+
+int
+AppNDB::applyMetaRecord(MetaRecord* mr, Uint32 gci)
+{
+ /**
+ * Validate table id
+ */
+ Uint32 tableId = mr->tableId;
+ if (tableId==0) {
+ RLOG(("WARNING! Meta record contained record with tableId 0"));
+ return 0;
+ }
+
+ /**
+ * Prepare meta record
+ */
+ NdbDictionary::Table * table = prepareMetaRecord(mr);
+ if(table == 0) {
+ RLOG(("WARNING! Prepare table meta record failed for table %d", tableId));
+ m_dict->getNdbError();
+ m_repState->eventInsertRef(gci,0,tableId,
+ GrepError::REP_APPLIER_PREPARE_TABLE);
+ return -1;
+ }
+
+ /**
+ * Table does not exist in TableInfoPs -> add it
+ */
+ if(m_tableInfoPs->getTableName(tableId)==0) {
+ RLOG(("Table %d:%s added to m_tableInfoPs", tableId, table->getName()));
+ m_tableInfoPs->insert(tableId,table->getName());
+ }
+
+ /**
+ * Validate that table does not exist in Dict
+ */
+
+ const NdbDictionary::Table * tmpTable = m_dict->getTable(table->getName());
+ if(tmpTable !=0) {
+ /**
+ * Oops, a table with the same name exists
+ */
+ if(tmpTable->getObjectVersion()!=table->getObjectVersion()) {
+ char buf[100];
+ sprintf(buf,"WARNING! Another version of table %d:%s already exists."
+ "Currently, we dont support versions, so will abort now!",
+ tableId, table->getName());
+
+ REPABORT(buf);
+
+ }
+ RLOG(("WARNING! An identical table %d:%s already exists.",
+ tableId, table->getName()));
+ return -1;
+ }
+
+
+ /**
+ * @todo WARNING! Should scan table MR for columns that are not supported
+ */
+ /*
+ NdbDictionary::Column * column;
+
+ for(int i=0; i<table->getNoOfColumns(); i++) {
+ column = table->getColumn(i);
+ if(column->getAutoIncrement()) {
+ reportWarning(table->getName(), column->getName(),
+ "Uses AUTOINCREMENT of PK");
+ }
+ }
+ */
+
+
+ /**
+ * Create table
+ */
+ if(m_dict->createTable(*table)<0) {
+ ndbout_c("AppNDB: Send the following error msg to NDB Cluster support");
+ reportNdbError("Create table failed!", m_dict->getNdbError());
+ m_repState->eventCreateTableRef(gci,
+ tableId,
+ table->getName(),
+ GrepError::REP_APPLIER_CREATE_TABLE);
+ return -1;
+ }
+
+ RLOG(("Table %d:%s created", tableId, table->getName()));
+ return 0;
+}
+
+NdbDictionary::Table*
+AppNDB::prepareMetaRecord(MetaRecord* mr) {
+ NdbTableImpl * tmp = 0;
+ NdbDictionary::Table * table =0;
+ Uint32 * data =(Uint32*)( ((char*)mr + sizeof(Uint32)*6));
+ int res = NdbDictInterface::parseTableInfo(&tmp, data, mr->dataLen);
+ if(res == 0) {
+ table = tmp;
+ return table;
+ } else{
+ return 0;
+ }
+}
+
+void
+AppNDB::reportNdbError(const char * msg, const NdbError & err) {
+ ndbout_c("%s : Error code %d , error message %s",
+ msg, err.code,
+ (err.message ? err.message : ""));
+}
+
+void
+AppNDB::reportWarning(const char * tableName, const char * message) {
+ ndbout_c("WARNING: Table %s, %s", tableName, message);
+}
+
+void
+AppNDB::reportWarning(const char * tableName, const char * columnName,
+ const char * message) {
+ ndbout_c("WARNING: Table %s, column %s, %s", tableName, columnName,message);
+}
+
+int
+AppNDB::dropTable(Uint32 tableId)
+{
+ char * tableName = m_tableInfoPs->getTableName(tableId);
+ if(tableName == 0) return -1;
+ ndbout_c("AppNDB: Dropping table ");
+ if(m_dict->dropTable(tableName) != 0) {
+ reportNdbError("Failed dropping table",m_dict->getNdbError());
+ return -1;
+ }
+ m_tableInfoPs->del(tableId);
+ return 1;
+}
diff --git a/ndb/src/rep/adapters/AppNDB.hpp b/ndb/src/rep/adapters/AppNDB.hpp
new file mode 100644
index 00000000000..c24774d4ed3
--- /dev/null
+++ b/ndb/src/rep/adapters/AppNDB.hpp
@@ -0,0 +1,145 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef APPNDB_HPP
+#define APPNDB_HPP
+#include "NdbApi.hpp"
+
+#include <string.h>
+#include <assert.h>
+
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbStdio.h>
+#include <NdbTick.h>
+
+#include <NdbThread.h>
+#include <Vector.hpp>
+
+#include "TableInfoPs.hpp"
+#include <rep/storage/GCIContainer.hpp>
+#include <rep/storage/GCIBuffer.hpp>
+
+#include <rep/state/RepState.hpp>
+
+extern "C" {
+ void * runAppNDB_C(void *);
+}
+
+/**
+ * @class AppNDB
+ * @brief Connects to NDB and appliers log records into standby system
+ */
+class AppNDB {
+public:
+ /***************************************************************************
+ * Constructor / Destructor / Init
+ ***************************************************************************/
+ AppNDB(class GCIContainer * gciContainer, class RepState * repState);
+ ~AppNDB();
+
+ void init(const char * connectString);
+
+ GrepError::Code
+ applyBuffer(Uint32 nodeGrp, Uint32 first, Uint32 force);
+
+ /**
+ * Takes a table id and drops it.
+ * @param tableId Name of table to be dropped
+ * @return Returns 1 = ok, -1 failed
+ *
+ * @todo Fix: 0 usually means ok...
+ */
+ int dropTable(Uint32 tableId);
+ void startApplier();
+ void stopApplier(GrepError::Code err);
+private:
+ /***************************************************************************
+ * Methods
+ ***************************************************************************/
+ friend void* runAppNDB_C(void*);
+
+ void threadMainAppNDB(void);
+
+ /**
+ * Takes a log records and does the operation specified in the log record
+ * on NDB.
+ * @param - lr (LogRecord)
+ * @param - force true if GREP:SSCoord is in phase STARTING.
+ * Ignore "Execute" errors if true.
+ */
+ int applyLogRecord(LogRecord * lr, bool force, Uint32 gci);
+
+ /**
+ * Applies a table based on a meta record and creates the table
+ * in NDB.
+ * @param - meta record
+ * @return - 0 on success, -1 if something went wrong
+ */
+ int applyMetaRecord(MetaRecord * mr, Uint32 gci);
+
+ /**
+ * Takes a meta record and uses NdbDictionaryXXX::parseInfoTable
+ * and returns a table
+ * @param mr - MetaRecord
+ * @return - a table based on the meta record
+ */
+ NdbDictionary::Table* prepareMetaRecord(MetaRecord * mr);
+
+ /**
+ * Prints out an NDB error message if a ndb operation went wrong.
+ * @param msg - text explaining the error
+ * @param err - NDB error type
+ */
+ void reportNdbError(const char * msg, const NdbError & err);
+
+ /**
+ * Prints out a warning message. Used if support for something
+ * is not implemented.
+ * @param tableName - the name of the table this warning occured on
+ * @param message - warning message
+ */
+ void reportWarning(const char * tableName, const char * message);
+
+ /**
+ * Prints out a warning message. Used if support for something
+ * is not implemented.
+ * @param tableName - the name of the table this warning occured on
+ * @param columnName - the name of the column this warning occured on
+ * @param message - warning message
+ */
+ void reportWarning(const char * tableName, const char * columnName,
+ const char * message);
+
+
+ /***************************************************************************
+ * Variables
+ ***************************************************************************/
+ GCIContainer * m_gciContainer;
+ RepState * m_repState;
+
+ Ndb* m_ndb;
+ NdbDictionary::Dictionary * m_dict;
+ NodeId m_ownNodeId;
+ bool m_started;
+ TableInfoPs * m_tableInfoPs;
+ NdbThread* m_applierThread;
+ NdbCondition * m_cond;
+ MutexVector<GCIBuffer*> m_gciBufferList;
+};
+
+#endif
diff --git a/ndb/src/rep/adapters/ExtAPI.cpp b/ndb/src/rep/adapters/ExtAPI.cpp
new file mode 100644
index 00000000000..0dcd1e85465
--- /dev/null
+++ b/ndb/src/rep/adapters/ExtAPI.cpp
@@ -0,0 +1,31 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "ExtAPI.hpp"
+
+GrepError::Code
+ExtAPI::eventSubscriptionIdCreated(Uint32 subId, Uint32 subKey)
+{
+ NdbApiSignal* signal = m_repSender->getSignal();
+ CreateSubscriptionIdConf * conf =
+ (CreateSubscriptionIdConf *)signal->getDataPtrSend();
+ conf->subscriptionId = subId;
+ conf->subscriptionKey = subKey;
+ signal->set(0, SSREPBLOCKNO, GSN_GREP_CREATE_SUBID_CONF,
+ CreateSubscriptionIdConf::SignalLength);
+ m_repSender->sendSignal(signal);
+ return GrepError::NO_ERROR;
+}
diff --git a/ndb/src/rep/adapters/ExtAPI.hpp b/ndb/src/rep/adapters/ExtAPI.hpp
new file mode 100644
index 00000000000..f10b6c7d682
--- /dev/null
+++ b/ndb/src/rep/adapters/ExtAPI.hpp
@@ -0,0 +1,107 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef EXTAPI_HPP
+#define EXTAPI_HPP
+
+#include <signaldata/RepImpl.hpp>
+#include <signaldata/GrepImpl.hpp>
+#include <signaldata/SumaImpl.hpp>
+
+#include <rep/ExtSender.hpp>
+
+/**
+ * The abstract class for all extractors
+ */
+class ExtAPI
+{
+public:
+ /***************************************************************************
+ * Constructor / Destructor
+ ***************************************************************************/
+#if 0
+ bool init(const char * connectString = NULL);
+
+ GrepError::Code dataLogStarted(Uint32 epoch,
+ Uint32 subId, Uint32 subKey) = 0;
+ GrepError::Code metaLogStarted(Uint32 epoch,
+ Uint32 subId, Uint32 subKey) = 0;
+ GrepError::Code epochComleted() = 0;
+ GrepError::Code subscriptionCreated() = 0;
+ GrepError::Code subscriptionRemoved() = 0;
+ GrepError::Code metaScanCompleted() = 0;
+ GrepError::Code dataScanCompleted() = 0;
+ GrepError::Code subscriptionRemoveFailed() = 0;
+ GrepError::Code metaScanFailed() = 0;
+ GrepError::Code dataScanFailed() = 0;
+ GrepError::Code subscriptiodIdCreateFailed() = 0;
+ GrepError::Code dataLogFailed() = 0;
+ GrepError::Code metaLogFailed() = 0;
+ GrepError::Code subscriptionCreateFailed() = 0;
+
+ /**Above to be deleted*/
+#endif
+
+ virtual GrepError::Code
+ eventSubscriptionIdCreated(Uint32 subId, Uint32 subKey) ;
+
+#if 0
+ GrepError::Code
+ eventSubscriptionDeleted(Uint32 subId, Uint32 subKey);
+
+ GrepError::Code
+ eventMetaLogStarted(NdbApiSignal*, Uint32 subId, Uint32 subKey);
+
+ GrepError::Code
+ eventDataLogStarted(NdbApiSignal*, Uint32 subId, Uint32 subKey);
+
+ GrepError::Code
+ eventMetaScanCompleted(NdbApiSignal*, Uint32 subId, Uint32 subKey,
+ Interval epochs);
+
+ GrepError::Code
+ eventDataScanCompleted(NdbApiSignal*, Uint32 subId, Uint32 subKey,
+ Interval epochs);
+
+ GrepError::Code
+ eventMetaScanFailed(Uint32 subId, Uint32 subKey, GrepError::Code error);
+
+ GrepError::Code
+ eventDataScanFailed(Uint32 subId, Uint32 subKey, GrepError::Code error);
+#endif
+
+ /***************************************************************************
+ * Public Methods
+ ***************************************************************************/
+ void setRepSender(ExtSender * es) { m_repSender = es; };
+ //void signalErrorHandler(NdbApiSignal * s, Uint32 nodeId);
+
+protected:
+ ExtSender * m_repSender;
+};
+
+
+#if 0
+class TestExtAPI : public ExtAPI
+{
+ GrepError::Code
+ eventSubscriptionIdCreated(Uint32 subId, Uint32 subKey) {
+ ndbout_c("Received subscription:%d-%d");
+ };
+};
+#endif
+
+#endif // EXTAPI_HPP
diff --git a/ndb/src/rep/adapters/ExtNDB.cpp b/ndb/src/rep/adapters/ExtNDB.cpp
new file mode 100644
index 00000000000..5ba6bfbbe6e
--- /dev/null
+++ b/ndb/src/rep/adapters/ExtNDB.cpp
@@ -0,0 +1,552 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <assert.h>
+#include "ExtNDB.hpp"
+#include "ConfigRetriever.hpp"
+#include <NdbSleep.h>
+
+#include <NdbApiSignal.hpp>
+
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <AttributeHeader.hpp>
+#include <rep/rep_version.hpp>
+#include <ndb_limits.h>
+
+/*****************************************************************************
+ * Constructor / Destructor / Init
+ *****************************************************************************/
+ExtNDB::ExtNDB(GCIContainerPS * gciContainer, ExtAPI * extAPI)
+{
+ m_grepSender = new ExtSender();
+ if (!m_grepSender) REPABORT("Could not allocate object");
+ m_gciContainerPS = gciContainer;
+
+ m_nodeGroupInfo = new NodeGroupInfo();
+ m_gciContainerPS->setNodeGroupInfo(m_nodeGroupInfo);
+
+ m_doneSetGrepSender = false;
+ m_subId = 0;
+ m_subKey = 0;
+ m_firstGCI = 0;
+ m_dataLogStarted = false;
+
+ m_extAPI = extAPI;
+ if (!m_extAPI) REPABORT("Could not allocate object");
+}
+
+ExtNDB::~ExtNDB()
+{
+ delete m_grepSender;
+ delete m_nodeGroupInfo;
+}
+
+void
+ExtNDB::signalErrorHandler(NdbApiSignal * signal, Uint32 nodeId)
+{
+ //const Uint32 gsn = signal->readSignalNumber();
+ //const Uint32 len = signal->getLength();
+ RLOG(("Send signal failed. Signal %p", signal));
+}
+
+bool
+ExtNDB::init(const char * connectString)
+{
+ m_signalExecThread = NdbThread_Create(signalExecThread_C,
+ (void **)this,
+ 32768,
+ "ExtNDB_Service",
+ NDB_THREAD_PRIO_LOW);
+
+ ConfigRetriever configRetriever;
+ configRetriever.setConnectString(connectString);
+
+ Properties* config = configRetriever.getConfig("REP", REP_VERSION_ID);
+ if (config == 0) {
+ ndbout << "ExtNDB: Configuration error: ";
+ const char* erString = configRetriever.getErrorString();
+ if (erString == 0) {
+ erString = "No error specified!";
+ }
+ ndbout << erString << endl;
+ return false;
+ }
+ m_ownNodeId = configRetriever.getOwnNodeId();
+ config->put("LocalNodeId", m_ownNodeId);
+ config->put("LocalNodeType", "REP");
+
+ /**
+ * Check which GREPs to connect to (in configuration)
+ *
+ * @note SYSTEM LIMITATION: Only connects to one GREP
+ */
+ Uint32 noOfConnections=0;
+ NodeId grepNodeId=0;
+ const Properties * connection;
+
+ config->get("NoOfConnections", &noOfConnections);
+ for (Uint32 i=0; i<noOfConnections; i++) {
+ Uint32 nodeId1, nodeId2;
+ config->get("Connection", i, &connection);
+ connection->get("NodeId1", &nodeId1);
+ connection->get("NodeId2", &nodeId2);
+ if (!connection->contains("System1") &&
+ !connection->contains("System2") &&
+ (nodeId1 == m_ownNodeId || nodeId2 == m_ownNodeId)) {
+ /**
+ * Found connection
+ */
+ if (nodeId1 == m_ownNodeId) {
+ grepNodeId = nodeId2;
+ } else {
+ grepNodeId = nodeId1;
+ }
+ }
+ }
+
+ m_transporterFacade = TransporterFacade::instance();
+
+ assert(m_transporterFacade != 0);
+
+ m_ownBlockNo = m_transporterFacade->open(this, execSignal, execNodeStatus);
+ assert(m_ownBlockNo > 0);
+ m_ownRef = numberToRef(m_ownBlockNo, m_ownNodeId);
+ ndbout_c("EXTNDB blockno %d ownref %d ", m_ownBlockNo, m_ownRef);
+ assert(m_ownNodeId == m_transporterFacade->ownId());
+
+ m_grepSender->setOwnRef(m_ownRef);
+ m_grepSender->setTransporterFacade(m_transporterFacade);
+
+ if(!m_grepSender->connected(50000)){
+ ndbout_c("ExtNDB: Failed to connect to DB nodes!");
+ ndbout_c("ExtNDB: Tried to create transporter as (node %d, block %d).",
+ m_ownNodeId, m_ownBlockNo);
+ ndbout_c("ExtNDB: Check that DB nodes are started.");
+ return false;
+ }
+ ndbout_c("Phase 3 (ExtNDB): Connection %d to NDB Cluster opened (Extractor)",
+ m_ownBlockNo);
+
+ for (Uint32 i=1; i<MAX_NDB_NODES; i++) {
+ if (m_transporterFacade->getIsNodeDefined(i) &&
+ m_transporterFacade->getIsNodeSendable(i))
+ {
+ Uint32 nodeGrp = m_transporterFacade->getNodeGrp(i);
+ m_nodeGroupInfo->addNodeToNodeGrp(i, true, nodeGrp);
+ Uint32 nodeId = m_nodeGroupInfo->getFirstConnectedNode(nodeGrp);
+ m_grepSender->setNodeId(nodeId);
+ if(m_nodeGroupInfo->getPrimaryNode(nodeGrp) == 0) {
+ m_nodeGroupInfo->setPrimaryNode(nodeGrp, nodeId);
+ }
+ m_doneSetGrepSender = true;
+#if 0
+ RLOG(("Added node %d to node group %d", i, nodeGrp));
+#endif
+ }
+ }
+
+ return true;
+}
+
+/*****************************************************************************
+ * Signal Queue Executor
+ *****************************************************************************/
+
+class SigMatch
+{
+public:
+ int gsn;
+ void (ExtNDB::* function)(NdbApiSignal *signal);
+
+ SigMatch() { gsn = 0; function = NULL; };
+
+ SigMatch(int _gsn, void (ExtNDB::* _function)(NdbApiSignal *signal)) {
+ gsn = _gsn;
+ function = _function;
+ };
+
+ bool check(NdbApiSignal *signal) {
+ if(signal->readSignalNumber() == gsn)
+ return true;
+ return false;
+ };
+};
+
+void *
+ExtNDB::signalExecThread_C(void *r)
+{
+ ExtNDB *grepps = (ExtNDB*)r;
+
+ grepps->signalExecThreadRun();
+
+ NdbThread_Exit(0);
+ /* NOTREACHED */
+ return 0;
+}
+
+void
+ExtNDB::signalExecThreadRun()
+{
+ Vector<SigMatch> sl;
+
+ /**
+ * Signals to be executed
+ */
+ sl.push_back(SigMatch(GSN_SUB_GCP_COMPLETE_REP,
+ &ExtNDB::execSUB_GCP_COMPLETE_REP));
+
+ /**
+ * Is also forwarded to SSCoord
+ */
+ sl.push_back(SigMatch(GSN_GREP_SUB_START_CONF,
+ &ExtNDB::execGREP_SUB_START_CONF));
+ sl.push_back(SigMatch(GSN_GREP_SUB_CREATE_CONF,
+ &ExtNDB::execGREP_SUB_CREATE_CONF));
+ sl.push_back(SigMatch(GSN_GREP_SUB_REMOVE_CONF,
+ &ExtNDB::execGREP_SUB_REMOVE_CONF));
+ /**
+ * Signals to be forwarded
+ */
+ sl.push_back(SigMatch(GSN_GREP_CREATE_SUBID_CONF,
+ &ExtNDB::execGREP_CREATE_SUBID_CONF));
+
+ sl.push_back(SigMatch(GSN_GREP_SUB_SYNC_CONF, &ExtNDB::sendSignalRep));
+
+ sl.push_back(SigMatch(GSN_GREP_SUB_REMOVE_REF, &ExtNDB::sendSignalRep));
+ sl.push_back(SigMatch(GSN_GREP_SUB_SYNC_REF, &ExtNDB::sendSignalRep));
+ sl.push_back(SigMatch(GSN_GREP_CREATE_SUBID_REF, &ExtNDB::sendSignalRep));
+
+ sl.push_back(SigMatch(GSN_GREP_SUB_START_REF, &ExtNDB::sendSignalRep));
+ sl.push_back(SigMatch(GSN_GREP_SUB_CREATE_REF, &ExtNDB::sendSignalRep));
+
+ while(1) {
+ SigMatch *handler = NULL;
+ NdbApiSignal *signal = NULL;
+ if(m_signalRecvQueue.waitFor(sl, handler, signal)) {
+#if 0
+ RLOG(("Removed signal from queue (GSN: %d, QSize: %d)",
+ signal->readSignalNumber(), m_signalRecvQueue.size()));
+#endif
+ if(handler->function != 0) {
+ (this->*handler->function)(signal);
+ delete signal; signal = 0;
+ } else {
+ REPABORT("Illegal handler for signal");
+ }
+ }
+ }
+}
+
+void
+ExtNDB::sendSignalRep(NdbApiSignal * s)
+{
+ if(m_repSender->sendSignal(s) == -1)
+ {
+ signalErrorHandler(s, 0);
+ }
+}
+
+void
+ExtNDB::execSignal(void* executorObj, NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3])
+{
+ ExtNDB * executor = (ExtNDB*)executorObj;
+
+ const Uint32 gsn = signal->readSignalNumber();
+ const Uint32 len = signal->getLength();
+
+ NdbApiSignal * s = new NdbApiSignal(executor->m_ownRef);
+ switch(gsn){
+ case GSN_SUB_GCP_COMPLETE_REP:
+ case GSN_GREP_CREATE_SUBID_CONF:
+ case GSN_GREP_SUB_CREATE_CONF:
+ case GSN_GREP_SUB_START_CONF:
+ case GSN_GREP_SUB_SYNC_CONF:
+ case GSN_GREP_SUB_REMOVE_CONF:
+ case GSN_GREP_CREATE_SUBID_REF:
+ case GSN_GREP_SUB_CREATE_REF:
+ case GSN_GREP_SUB_START_REF:
+ case GSN_GREP_SUB_SYNC_REF:
+ case GSN_GREP_SUB_REMOVE_REF:
+ s->set(0, SSREPBLOCKNO, gsn, len);
+ memcpy(s->getDataPtrSend(), signal->getDataPtr(), 4 * len);
+ executor->m_signalRecvQueue.receive(s);
+ break;
+ case GSN_SUB_TABLE_DATA:
+ executor->execSUB_TABLE_DATA(signal, ptr);
+ delete s; s=0;
+ break;
+ case GSN_SUB_META_DATA:
+ executor->execSUB_META_DATA(signal, ptr);
+ delete s; s=0;
+ break;
+ default:
+ REPABORT1("Illegal signal received in execSignal", gsn);
+ }
+ s=0;
+#if 0
+ ndbout_c("ExtNDB: Inserted signal into queue (GSN: %d, Len: %d)",
+ signal->readSignalNumber(), len);
+#endif
+}
+
+void
+ExtNDB::execNodeStatus(void* obj, Uint16 nodeId, bool alive, bool nfCompleted)
+{
+ ExtNDB * thisObj = (ExtNDB*)obj;
+
+ RLOG(("Changed node status (Id %d, Alive %d, nfCompleted %d)",
+ nodeId, alive, nfCompleted));
+
+ if(alive) {
+ /**
+ * Connected
+ */
+ Uint32 nodeGrp = thisObj->m_transporterFacade->getNodeGrp(nodeId);
+ RLOG(("DB node %d of node group %d connected", nodeId, nodeGrp));
+
+ thisObj->m_nodeGroupInfo->addNodeToNodeGrp(nodeId, true, nodeGrp);
+ Uint32 firstNode = thisObj->m_nodeGroupInfo->getPrimaryNode(nodeGrp);
+
+ if(firstNode == 0)
+ thisObj->m_nodeGroupInfo->setPrimaryNode(nodeGrp, nodeId);
+
+ if (!thisObj->m_doneSetGrepSender) {
+ thisObj->m_grepSender->setNodeId(firstNode);
+ thisObj->m_doneSetGrepSender = true;
+ }
+
+ RLOG(("Connect: First connected node in nodegroup: %d",
+ thisObj->m_nodeGroupInfo->getPrimaryNode(nodeGrp)));
+
+ } else if (!nfCompleted) {
+
+ /**
+ * Set node as "disconnected" in m_nodeGroupInfo until
+ * node comes up again.
+ */
+ Uint32 nodeGrp = thisObj->m_transporterFacade->getNodeGrp(nodeId);
+ RLOG(("DB node %d of node group %d disconnected",
+ nodeId, nodeGrp));
+ thisObj->m_nodeGroupInfo->setConnectStatus(nodeId, false);
+ /**
+ * The node that crashed was also the primary node, the we must change
+ * primary node
+ */
+ if(nodeId == thisObj->m_nodeGroupInfo->getPrimaryNode(nodeGrp)) {
+ Uint32 node = thisObj->m_nodeGroupInfo->getFirstConnectedNode(nodeGrp);
+ if(node > 0) {
+ thisObj->m_grepSender->setNodeId(node);
+ thisObj->m_nodeGroupInfo->setPrimaryNode(nodeGrp, node);
+ }
+ else {
+ thisObj->sendDisconnectRep(nodeGrp);
+ }
+ }
+ RLOG(("Disconnect: First connected node in nodegroup: %d",
+ thisObj->m_nodeGroupInfo->getPrimaryNode(nodeGrp)));
+
+ } else if(nfCompleted) {
+ } else {
+ REPABORT("Function execNodeStatus with wrong parameters");
+ }
+}
+
+/*****************************************************************************
+ * Signal Receivers for LOG and SCAN
+ *****************************************************************************/
+
+/**
+ * Receive datalog/datascan from GREP/SUMA
+ */
+void
+ExtNDB::execSUB_TABLE_DATA(NdbApiSignal * signal, LinearSectionPtr ptr[3])
+{
+ SubTableData * const data = (SubTableData*)signal->getDataPtr();
+ Uint32 tableId = data->tableId;
+ Uint32 operation = data->operation;
+ Uint32 gci = data->gci;
+ Uint32 nodeId = refToNode(signal->theSendersBlockRef);
+
+ if((SubTableData::LogType)data->logType == SubTableData::SCAN)
+ {
+ Uint32 nodeGrp = m_nodeGroupInfo->findNodeGroup(nodeId);
+
+ NodeGroupInfo::iterator * it;
+ it = new NodeGroupInfo::iterator(nodeGrp, m_nodeGroupInfo);
+ for(NodeConnectInfo * nci=it->first(); it->exists();nci=it->next()) {
+ m_gciContainerPS->insertLogRecord(nci->nodeId, tableId,
+ operation, ptr, gci);
+ }
+ delete it; it = 0;
+ } else {
+ m_gciContainerPS->insertLogRecord(nodeId, tableId, operation, ptr, gci);
+ }
+}
+
+/**
+ * Receive metalog/metascan from GREP/SUMA
+ */
+void
+ExtNDB::execSUB_META_DATA(NdbApiSignal * signal, LinearSectionPtr ptr[3])
+{
+ Uint32 nodeId = refToNode(signal->theSendersBlockRef);
+ SubMetaData * const data = (SubMetaData*)signal->getDataPtr();
+ Uint32 tableId = data->tableId;
+ Uint32 gci = data->gci;
+
+ Uint32 nodeGrp = m_nodeGroupInfo->findNodeGroup(nodeId);
+
+ NodeGroupInfo::iterator * it;
+ it = new NodeGroupInfo::iterator(nodeGrp, m_nodeGroupInfo);
+ for(NodeConnectInfo * nci=it->first(); it->exists();nci=it->next()) {
+ m_gciContainerPS->insertMetaRecord(nci->nodeId, tableId, ptr, gci);
+ RLOG(("Received meta record in %d[%d]", nci->nodeId, gci));
+ }
+
+ delete it; it = 0;
+}
+
+
+/*****************************************************************************
+ * Signal Receivers (Signals that are actually just forwarded to SS REP)
+ *****************************************************************************/
+
+void
+ExtNDB::execGREP_CREATE_SUBID_CONF(NdbApiSignal * signal)
+{
+ CreateSubscriptionIdConf const * conf =
+ (CreateSubscriptionIdConf *)signal->getDataPtr();
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ ndbout_c("GREP_CREATE_SUBID_CONF m_extAPI=%p\n", m_extAPI);
+ m_extAPI->eventSubscriptionIdCreated(subId, subKey);
+}
+
+/*****************************************************************************
+ * Signal Receivers
+ *****************************************************************************/
+
+/**
+ * Receive information about completed GCI from GREP/SUMA
+ *
+ * GCI completed, i.e. no more unsent log records exists in SUMA
+ * @todo use node id to identify buffers?
+ */
+void
+ExtNDB::execSUB_GCP_COMPLETE_REP(NdbApiSignal * signal)
+{
+ SubGcpCompleteRep * const rep = (SubGcpCompleteRep*)signal->getDataPtr();
+ const Uint32 gci = rep->gci;
+ Uint32 nodeId = refToNode(rep->senderRef);
+
+ RLOG(("Epoch %d completed at node %d", gci, nodeId));
+ m_gciContainerPS->setCompleted(gci, nodeId);
+
+ if(m_firstGCI == gci && !m_dataLogStarted) {
+ sendGREP_SUB_START_CONF(signal, m_firstGCI);
+ m_dataLogStarted = true;
+ }
+}
+
+/**
+ * Send info that scan is competed to SS REP
+ *
+ * @todo Use node id to identify buffers?
+ */
+void
+ExtNDB::sendGREP_SUB_START_CONF(NdbApiSignal * signal, Uint32 gci)
+{
+ RLOG(("Datalog started (Epoch %d)", gci));
+ GrepSubStartConf * conf = (GrepSubStartConf *)signal->getDataPtrSend();
+ conf->firstGCI = gci;
+ conf->subscriptionId = m_subId;
+ conf->subscriptionKey = m_subKey;
+ conf->part = SubscriptionData::TableData;
+ signal->m_noOfSections = 0;
+ signal->set(0, SSREPBLOCKNO, GSN_GREP_SUB_START_CONF,
+ GrepSubStartConf::SignalLength);
+ sendSignalRep(signal);
+}
+
+/**
+ * Scan is completed... says SUMA/GREP
+ *
+ * @todo Use node id to identify buffers?
+ */
+void
+ExtNDB::execGREP_SUB_START_CONF(NdbApiSignal * signal)
+{
+ GrepSubStartConf * const conf = (GrepSubStartConf *)signal->getDataPtr();
+ Uint32 part = conf->part;
+ //Uint32 nodeId = refToNode(conf->senderRef);
+ m_firstGCI = conf->firstGCI;
+
+ if (part == SubscriptionData::TableData) {
+ RLOG(("Datalog started (Epoch %d)", m_firstGCI));
+ return;
+ }
+ RLOG(("Metalog started (Epoch %d)", m_firstGCI));
+
+ signal->set(0, SSREPBLOCKNO, GSN_GREP_SUB_START_CONF,
+ GrepSubStartConf::SignalLength);
+ sendSignalRep(signal);
+}
+
+/**
+ * Receive no of node groups that PS has and pass signal on to SS
+ */
+void
+ExtNDB::execGREP_SUB_CREATE_CONF(NdbApiSignal * signal)
+{
+ GrepSubCreateConf * conf = (GrepSubCreateConf *)signal->getDataPtrSend();
+ m_subId = conf->subscriptionId;
+ m_subKey = conf->subscriptionKey;
+
+ conf->noOfNodeGroups = m_nodeGroupInfo->getNoOfNodeGroups();
+ sendSignalRep(signal);
+}
+
+/**
+ * Receive conf that subscription has been remove in GREP/SUMA
+ *
+ * Pass signal on to TransPS
+ */
+void
+ExtNDB::execGREP_SUB_REMOVE_CONF(NdbApiSignal * signal)
+{
+ m_gciContainerPS->reset();
+ sendSignalRep(signal);
+}
+
+/**
+ * If all PS nodes has disconnected, then remove all epochs
+ * for this subscription.
+ */
+void
+ExtNDB::sendDisconnectRep(Uint32 nodeId)
+{
+ NdbApiSignal * signal = new NdbApiSignal(m_ownRef);
+ signal->set(0, SSREPBLOCKNO, GSN_REP_DISCONNECT_REP,
+ RepDisconnectRep::SignalLength);
+ RepDisconnectRep * rep = (RepDisconnectRep*) signal->getDataPtrSend();
+ rep->nodeId = nodeId;
+ rep->subId = m_subId;
+ rep->subKey = m_subKey;
+ sendSignalRep(signal);
+}
diff --git a/ndb/src/rep/adapters/ExtNDB.hpp b/ndb/src/rep/adapters/ExtNDB.hpp
new file mode 100644
index 00000000000..c69f94d9a7e
--- /dev/null
+++ b/ndb/src/rep/adapters/ExtNDB.hpp
@@ -0,0 +1,113 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef EXTNDB_HPP
+#define EXTNDB_HPP
+
+#include <TransporterDefinitions.hpp>
+#include <TransporterFacade.hpp>
+#include <ClusterMgr.hpp>
+#include <API.hpp>
+#include <Vector.hpp>
+#include <stdio.h>
+
+#include <signaldata/RepImpl.hpp>
+#include <signaldata/GrepImpl.hpp>
+
+#include <rep/SignalQueue.hpp>
+#include <rep/ExtSender.hpp>
+
+#include <rep/storage/GCIContainerPS.hpp>
+#include "ExtAPI.hpp"
+
+/**
+ * @class ExtNDB
+ * @brief Class responsible for connection to primary system GREP
+ */
+class ExtNDB
+{
+public:
+ /***************************************************************************
+ * Constructor / Destructor
+ ***************************************************************************/
+ ExtNDB(GCIContainerPS * gciContainer, ExtAPI * extAPI);
+ ~ExtNDB();
+ bool init(const char * connectString = NULL);
+
+ /***************************************************************************
+ * Public Methods
+ ***************************************************************************/
+ void setGrepSender(ExtSender * es) { m_grepSender = es; };
+ ExtSender * getGrepSender() { return m_grepSender; };
+ void setRepSender(ExtSender * es) {
+ m_extAPI->setRepSender(es); m_repSender = es; };
+ void signalErrorHandler(NdbApiSignal * s, Uint32 nodeId);
+
+private:
+ static void execSignal(void* signalSender, NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]);
+
+ static void execNodeStatus(void* signalSender, NodeId,
+ bool alive, bool nfCompleted);
+
+ void signalExecThreadRun();
+ static void * signalExecThread_C(void *);
+
+ void sendSignalRep(NdbApiSignal *);
+ void sendDisconnectRep(Uint32 nodeId);
+
+ /***************************************************************************
+ * Signal Executors
+ ***************************************************************************/
+ void execSUB_GCP_COMPLETE_REP(NdbApiSignal*);
+ void execGREP_SUB_CREATE_CONF(NdbApiSignal * signal);
+ void execGREP_SUB_REMOVE_CONF(NdbApiSignal * signal);
+ void execGREP_SUB_START_CONF(NdbApiSignal * signal);
+ void sendGREP_SUB_START_CONF(NdbApiSignal * signal, Uint32 gci);
+ void execSUB_TABLE_DATA(NdbApiSignal * signal,LinearSectionPtr ptr[3]);
+ void execSUB_META_DATA(NdbApiSignal * signal,LinearSectionPtr ptr[3]);
+
+ // Signals that are actually just fowarded to REP
+ void execGREP_CREATE_SUBID_CONF(NdbApiSignal *);
+
+ /***************************************************************************
+ * Private Variables
+ ***************************************************************************/
+ struct NdbThread * m_signalExecThread;
+ class SignalQueue m_signalRecvQueue;
+
+ Uint32 m_ownNodeId; ///< NodeId of this node
+ Uint32 m_ownBlockNo; ///< BlockNo of this "block"
+ BlockReference m_ownRef; ///< Reference to this
+
+ ExtSender * m_grepSender; ///< Responsible send to GREP
+ ExtSender * m_repSender; ///< Responsible send to SS REP
+
+ NodeGroupInfo * m_nodeGroupInfo;
+ GCIContainerPS * m_gciContainerPS; ///< Interface to GCICotainer
+ ///< seen by PS
+ TransporterFacade * m_transporterFacade;
+
+ bool m_doneSetGrepSender; ///< Only done once
+ bool m_dataLogStarted;
+ Uint32 m_subId;
+ Uint32 m_subKey;
+ Uint32 m_firstGCI;
+
+ ExtAPI * m_extAPI;
+};
+
+#endif
diff --git a/ndb/src/rep/adapters/Makefile b/ndb/src/rep/adapters/Makefile
new file mode 100644
index 00000000000..bdd711510c3
--- /dev/null
+++ b/ndb/src/rep/adapters/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapi repserver kernel
+
+ARCHIVE_TARGET := repadapters
+
+SOURCES = ExtNDB.cpp \
+ AppNDB.cpp \
+ ExtAPI.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/rep/adapters/TableInfoPs.hpp b/ndb/src/rep/adapters/TableInfoPs.hpp
new file mode 100644
index 00000000000..3fa25979255
--- /dev/null
+++ b/ndb/src/rep/adapters/TableInfoPs.hpp
@@ -0,0 +1,118 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TABLEINFO_PS_HPP
+#define TABLEINFO_PS_HPP
+
+#include <Vector.hpp>
+#include <ndb_types.h>
+#include <string.h>
+#include <NdbMem.h>
+
+struct TableInfo {
+ Uint32 tableId;
+ char* tableName;
+};
+
+/**
+ * @class TableInfoPS
+ * @brief Meta information about tables stored on PS
+ */
+class TableInfoPs {
+public:
+ inline void insert(const Uint32 tableId, const char * tableName);
+
+ inline bool del(const Uint32 tableId);
+
+ inline char * getTableName(const Uint32 tableId) const;
+
+private:
+ Vector<struct TableInfo*> tableInfo;
+
+ inline TableInfo * lookup(const Uint32 tableId) const;
+ inline TableInfo * lookup(const Uint32 tableId , Uint32 * pos) const;
+};
+
+inline
+TableInfo *
+TableInfoPs::lookup(const Uint32 tableId) const{
+ TableInfo * table;
+ Uint32 i=0;
+
+ while(i<tableInfo.size()) {
+ table=tableInfo[i];
+ if(table->tableId == tableId)
+ return table;
+ i++;
+ }
+ return 0;
+}
+
+inline
+TableInfo *
+TableInfoPs::lookup(const Uint32 tableId, Uint32 * pos ) const{
+ TableInfo * table;
+ Uint32 i=0;
+ while(i<tableInfo.size()) {
+ table=tableInfo[i];
+ if(table->tableId == tableId) {
+ *pos=i;
+ return table;
+ }
+ i++;
+ }
+ return 0;
+}
+
+
+inline
+char *
+TableInfoPs::getTableName(const Uint32 tableId) const{
+ TableInfo * table;
+ table=lookup(tableId);
+ if(table!=0)
+ return table->tableName;
+ return 0;
+}
+
+
+inline
+void
+TableInfoPs::insert(const Uint32 tableId, const char * tableName) {
+ TableInfo * table = new TableInfo;
+ table->tableId=tableId;
+ table->tableName=strdup(tableName);
+ tableInfo.push_back(table);
+}
+
+inline
+bool
+TableInfoPs::del(const Uint32 tableId) {
+
+ TableInfo * table;
+ Uint32 i=0;
+ table = lookup(tableId, &i);
+
+ if(table!=0) {
+ NdbMem_Free(table->tableName);
+ delete table;
+ tableInfo.erase(i);
+ return true;
+ }
+ return false;
+}
+
+#endif
diff --git a/ndb/src/rep/dbug_hack.cpp b/ndb/src/rep/dbug_hack.cpp
new file mode 100644
index 00000000000..364325b55ae
--- /dev/null
+++ b/ndb/src/rep/dbug_hack.cpp
@@ -0,0 +1,73 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdarg.h>
+#include <string.h>
+#include <OutputStream.hpp>
+#include "NdbOut.hpp"
+#include "rep_version.hpp"
+
+int replogEnabled;
+
+/**
+ * @todo This should be implemented using MySQLs dbug library
+ */
+extern "C"
+void
+DBUG_PRINT(const char * fmt, ...)
+{
+#ifdef DBUG
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ ndbout << buf << endl;
+ va_end(ap);
+#endif
+}
+
+extern "C"
+void
+replog(const char * fmt, ...)
+{
+ if (replogEnabled)
+ {
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ ndbout << buf << endl;
+ va_end(ap);
+ }
+}
+
+extern "C"
+void
+rlog(const char * fmt, ...)
+{
+ va_list ap;
+ char buf[1000];
+
+ va_start(ap, fmt);
+ if (fmt != 0)
+ vsnprintf(buf, sizeof(buf)-1, fmt, ap);
+ ndbout << buf;
+ va_end(ap);
+}
diff --git a/ndb/src/rep/rep_version.hpp b/ndb/src/rep/rep_version.hpp
new file mode 100644
index 00000000000..a6af131f4ef
--- /dev/null
+++ b/ndb/src/rep/rep_version.hpp
@@ -0,0 +1,86 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REP_VERSION_HPP
+#define REP_VERSION_HPP
+
+/**
+ * Block number for REP
+ */
+#define SSREPBLOCKNO 1
+#define PSREPBLOCKNO 2
+
+#define DBUG
+
+extern "C"
+void
+DBUG_PRINT(const char * fmt, ...);
+
+extern "C"
+void
+replog(const char * fmt, ...);
+
+extern "C"
+void
+rlog(const char * fmt, ...);
+
+#define RLOG(ARGS) \
+ do { if (replogEnabled) { \
+ rlog ARGS; \
+ ndbout << " (" << __FILE__ << ":" << __LINE__ << ")" << endl; \
+ } \
+ } while (0)
+
+/**
+ * Replication logging on or off
+ */
+extern int replogEnabled;
+
+/**
+ * Used for config id
+ */
+#define REP_VERSION_ID 0
+
+#define MAX_NODE_GROUPS 6
+
+#define REPABORT(string) \
+ { \
+ ndbout_c("\nInternal error in %s:%d: %s", __FILE__, __LINE__, string); \
+ abort(); \
+ }
+#define REPABORT1(string, data1) \
+ { \
+ ndbout_c("\nInternal error in %s:%d: %s" \
+ "\n (data1: %d)", \
+ __FILE__, __LINE__, string, data1); \
+ abort(); \
+ }
+#define REPABORT2(string, data1, data2) \
+ { \
+ ndbout_c("\nInternal error in %s:%d: %s" \
+ "\n (data1: %d, data2: %d)", \
+ __FILE__, __LINE__, string, data1, data2); \
+ abort(); \
+ }
+#define REPABORT3(string, data1, data2, data3) \
+ { \
+ ndbout_c("\nInternal error in %s:%d: %s" \
+ "\n (data1: %d, data2: %d data3: %d)", \
+ __FILE__, __LINE__, string, data1, data2, data3); \
+ abort(); \
+ }
+
+#endif
diff --git a/ndb/src/rep/repapi/Makefile b/ndb/src/rep/repapi/Makefile
new file mode 100644
index 00000000000..fdd153f1060
--- /dev/null
+++ b/ndb/src/rep/repapi/Makefile
@@ -0,0 +1,25 @@
+include .defs.mk
+
+TYPE := util
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := repapi
+
+A_LIB := Y
+SO_LIB := Y
+PIC_LIB := Y
+
+#DIRS := test
+
+LIB_TARGET := REP_API
+LIB_TARGET_ARCHIVES := $(ARCHIVE_TARGET) general portlib
+
+# Source files of non-templated classes (.C files)
+SOURCES = repapi.cpp
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include/mgmapi) \
+ -I$(call fixpath,$(NDB_TOP)/src/common/mgmcommon)
+
+CCFLAGS += -DNO_DEBUG_MESSAGES
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/rep/repapi/repapi.cpp b/ndb/src/rep/repapi/repapi.cpp
new file mode 100644
index 00000000000..80274896004
--- /dev/null
+++ b/ndb/src/rep/repapi/repapi.cpp
@@ -0,0 +1,603 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbTCP.h>
+#include "repapi.h"
+//#include "mgmapi_debug.h"
+#include <socket_io.h>
+
+#include <stdlib.h>
+
+#include <string.h>
+#include <NdbStdio.h>
+#include <NdbString.h>
+#include <errno.h>
+#include <NdbOut.hpp>
+#include <SocketServer.hpp>
+#include <Parser.hpp>
+#include <OutputStream.hpp>
+#include <InputStream.hpp>
+
+#if defined VM_TRACE && !defined NO_DEBUG_MESSAGES
+#define DEBUG(x) ndbout << x << endl;
+#elif defined NO_DEBUG_MESSAGES
+#define DEBUG(x)
+#endif
+
+#ifdef NDB_WIN32
+#define EBADMSG EFAULT
+#endif
+
+
+
+class ParserDummy2 : SocketServer::Session {
+public:
+ ParserDummy2(NDB_SOCKET_TYPE sock);
+};
+
+ParserDummy2::ParserDummy2(NDB_SOCKET_TYPE sock) : SocketServer::Session(sock) {
+
+}
+
+typedef Parser<ParserDummy2> Parser_t;
+
+
+#define REP_CMD(name, fun, desc) \
+ { name, \
+ 0, \
+ ParserRow<ParserDummy2>::Cmd, \
+ ParserRow<ParserDummy2>::String, \
+ ParserRow<ParserDummy2>::Optional, \
+ ParserRow<ParserDummy2>::IgnoreMinMax, \
+ 0, 0, \
+ fun, \
+ desc, 0 }
+
+#define REP_ARG(name, type, opt, desc) \
+ { name, \
+ 0, \
+ ParserRow<ParserDummy2>::Arg, \
+ ParserRow<ParserDummy2>::type, \
+ ParserRow<ParserDummy2>::opt, \
+ ParserRow<ParserDummy2>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ desc, 0 }
+
+#define REP_END() \
+ { 0, \
+ 0, \
+ ParserRow<ParserDummy2>::Arg, \
+ ParserRow<ParserDummy2>::Int, \
+ ParserRow<ParserDummy2>::Optional, \
+ ParserRow<ParserDummy2>::IgnoreMinMax, \
+ 0, 0, \
+ 0, \
+ 0, 0 }
+
+struct ndb_rep_handle {
+ char * hostname;
+ unsigned short port;
+
+ int connected;
+ int last_error;
+ int last_error_line;
+ int read_timeout;
+ int write_timeout;
+
+ NDB_SOCKET_TYPE socket;
+
+#ifdef REPAPI_LOG
+ FILE* logfile;
+#endif
+};
+
+#define SET_ERROR(h, e) \
+ h->last_error = e; \
+ h->last_error_line = __LINE__;
+
+extern "C"
+NdbRepHandle
+ndb_rep_create_handle(){
+ NdbRepHandle h = (NdbRepHandle)malloc(sizeof(ndb_rep_handle));
+ h->connected = 0;
+ h->last_error = 0;
+ h->last_error_line = 0;
+ h->hostname = 0;
+ h->socket = -1;
+ h->read_timeout = 50000;
+ h->write_timeout = 100;
+
+#ifdef REPAPI_LOG
+ h->logfile = 0;
+#endif
+
+ return h;
+}
+
+/**
+ * Destroy a handle
+ */
+extern "C"
+void
+ndb_rep_destroy_handle(NdbRepHandle * handle){
+ if(!handle)
+ return;
+ if((* handle)->connected){
+ ndb_rep_disconnect(* handle);
+ }
+ if((* handle)->hostname != 0){
+ free((* handle)->hostname);
+ }
+#ifdef REPAPI_LOG
+ if ((* handle)->logfile != 0){
+ fclose((* handle)->logfile);
+ (* handle)->logfile = 0;
+ }
+#endif
+ free(* handle);
+ * handle = 0;
+}
+
+/**
+ * Get latest error associated with a handle
+ */
+extern "C"
+int
+ndb_rep_get_latest_error(const NdbRepHandle h){
+ return h->last_error;
+}
+
+/**
+ * Get latest error line associated with a handle
+ */
+extern "C"
+int
+ndb_rep_get_latest_error_line(const NdbRepHandle h){
+ return h->last_error_line;
+}
+
+static
+int
+parse_connect_string(const char * connect_string,
+ NdbRepHandle handle){
+
+ if(connect_string == 0){
+ DEBUG("connect_string == 0");
+ SET_ERROR(handle, EINVAL);
+ return -1;
+ }
+
+ char * line = strdup(connect_string);
+ if(line == 0){
+ DEBUG("line == 0");
+ SET_ERROR(handle, ENOMEM);
+ return -1;
+ }
+
+ char * tmp = strchr(line, ':');
+ if(tmp == 0){
+ DEBUG("tmp == 0");
+ free(line);
+ SET_ERROR(handle, EINVAL);
+ return -1;
+ }
+ * tmp = 0; tmp++;
+
+ int port = 0;
+ if(sscanf(tmp, "%d", &port) != 1){
+ DEBUG("sscanf() != 1");
+ free(line);
+ SET_ERROR(handle, EINVAL);
+ return -1;
+ }
+
+ if(handle->hostname != 0)
+ free(handle->hostname);
+
+ handle->hostname = strdup(line);
+ handle->port = port;
+ free(line);
+ return 0;
+}
+
+/*
+ * Call an operation, and return the reply
+ */
+static const Properties *
+ndb_rep_call(NdbRepHandle handle,
+ const ParserRow<ParserDummy2> *command_reply,
+ const char *cmd,
+ const Properties *cmd_args) {
+ SocketOutputStream out(handle->socket);
+ SocketInputStream in(handle->socket, handle->read_timeout);
+
+ out.println(cmd);
+#ifdef REPAPI_LOG
+ /**
+ * Print command to log file
+ */
+ FileOutputStream f(handle->logfile);
+ f.println("OUT: %s", cmd);
+#endif
+
+ if(cmd_args != NULL) {
+ Properties::Iterator iter(cmd_args);
+ const char *name;
+ while((name = iter.next()) != NULL) {
+ PropertiesType t;
+ Uint32 val_i;
+ BaseString val_s;
+
+ cmd_args->getTypeOf(name, &t);
+ switch(t) {
+ case PropertiesType_Uint32:
+ cmd_args->get(name, &val_i);
+ out.println("%s: %d", name, val_i);
+ break;
+ case PropertiesType_char:
+ cmd_args->get(name, val_s);
+ out.println("%s: %s", name, val_s.c_str());
+ break;
+ default:
+ /* Ignore */
+ break;
+ }
+ }
+#ifdef REPAPI_LOG
+ /**
+ * Print arguments to log file
+ */
+ cmd_args->print(handle->logfile, "OUT: ");
+#endif
+ }
+ out.println("");
+
+ Parser_t::Context ctx;
+ ParserDummy2 session(handle->socket);
+ Parser_t parser(command_reply, in, true, true, true);
+
+#if 1
+ const Properties* p = parser.parse(ctx, session);
+ if (p == NULL){
+ /**
+ * Print some info about why the parser returns NULL
+ */
+ ndbout << " status=" << ctx.m_status << ", curr="<<ctx.m_currentToken << endl;
+ }
+#ifdef REPAPI_LOG
+ else {
+ /**
+ * Print reply to log file
+ */
+ p->print(handle->logfile, "IN: ");
+ }
+#endif
+ return p;
+#else
+ return parser.parse(ctx, session);
+#endif
+}
+
+/**
+ * Connect to a rep server
+ *
+ * Returns 0 if OK, sets ndb_rep_handle->last_error otherwise
+ */
+extern "C"
+int
+ndb_rep_connect(NdbRepHandle handle, const char * repsrv){
+
+ if(handle == 0)
+ return -1;
+
+ if(parse_connect_string(repsrv, handle) != 0)
+ return -1;
+
+
+#ifdef REPAPI_LOG
+ /**
+ * Open the log file
+ */
+ char logname[64];
+ snprintf(logname, 64, "repapi.log");
+ handle->logfile = fopen(logname, "w");
+#endif
+
+ /**
+ * Do connect
+ */
+ const NDB_SOCKET_TYPE sockfd = socket(AF_INET, SOCK_STREAM, 0);
+ if (sockfd == NDB_INVALID_SOCKET) {
+ DEBUG("socket() == INVALID_SOCKET");
+ return -1;
+ }
+
+ struct sockaddr_in servaddr;
+ memset(&servaddr, 0, sizeof(servaddr));
+ servaddr.sin_family = AF_INET;
+ servaddr.sin_port = htons(handle->port);
+ // Convert ip address presentation format to numeric format
+ const int res1 = Ndb_getInAddr(&servaddr.sin_addr, handle->hostname);
+ if (res1 != 0) {
+ DEBUG("Ndb_getInAddr(...) == -1");
+ return -1;
+ }
+
+ const int res2 = connect(sockfd, (struct sockaddr*) &servaddr,
+ sizeof(servaddr));
+ if (res2 == -1) {
+ DEBUG("connect() == -1");
+ NDB_CLOSE_SOCKET(sockfd);
+ return -1;
+ }
+
+ handle->socket = sockfd;
+ handle->connected = 1;
+
+ return 0;
+}
+
+/**
+ * Disconnect from a rep server
+ */
+extern "C"
+void
+ndb_rep_disconnect(NdbRepHandle handle){
+ if(handle == 0)
+ return;
+
+ if(handle->connected != 1){
+ return;
+ }
+
+ NDB_CLOSE_SOCKET(handle->socket);
+ handle->socket = -1;
+ handle->connected = 0;
+
+ return;
+}
+
+
+
+/******************************************************************************
+ * Global Replication
+ ******************************************************************************/
+extern "C"
+int ndb_rep_command(NdbRepHandle handle,
+ unsigned int request,
+ unsigned int* replication_id,
+ struct ndb_rep_reply* /*reply*/,
+ unsigned int epoch) {
+
+ *replication_id = 0;
+
+ const ParserRow<ParserDummy2> replication_reply[] = {
+ REP_CMD("global replication reply", NULL, ""),
+ REP_ARG("result", Int, Mandatory, "Error message"),
+ REP_ARG("id", Int, Optional, "Id of global replication"),
+ REP_END()
+ };
+
+ if (handle == 0) {
+ return -1;
+ }
+
+ if (handle->connected != 1) {
+ handle->last_error = EINVAL;
+ return -1;
+ }
+
+ Properties args;
+ args.put("request", request);
+ args.put("id", *replication_id);
+ if(epoch > 0)
+ args.put("epoch",epoch);
+ else
+ args.put("epoch",(unsigned int)0);
+
+ const Properties *reply;
+ reply = ndb_rep_call(handle, replication_reply, "rep", &args);
+
+ if(reply == NULL) {
+ handle->last_error = EIO;
+ return -1;
+ }
+
+ reply->get("id", replication_id);
+ Uint32 result;
+ reply->get("result", &result);
+ delete reply;
+ return result;
+}
+
+extern "C"
+int convert2int(char * first, char * last, unsigned int f[], unsigned int l[])
+{
+ char * ftok = strtok(first, ",");
+ char * ltok = strtok(last, ",");
+ Uint32 i=0;
+ while(ftok!=NULL && ltok!=NULL)
+ {
+ f[i] = atoi(ftok);
+ l[i] = atoi(ltok);
+ ftok = strtok(NULL, ",");
+ ltok = strtok(NULL, ",");
+ i++;
+ }
+
+ return 0;
+}
+
+
+int ndb_rep_query(NdbRepHandle handle,
+ QueryCounter counter,
+ unsigned int* replicationId,
+ struct ndb_rep_reply* /*reply*/,
+ struct rep_state * state)
+{
+ *replicationId = 0; // not used currently.
+
+ if(state == 0)
+ return -1;
+
+ const ParserRow<ParserDummy2> replication_reply[] = {
+ REP_CMD("global replication query reply", NULL, ""),
+ REP_ARG("result", String, Mandatory, "Error message"),
+ REP_ARG("id", Int, Mandatory, "replicationId"),
+ REP_ARG("no_of_nodegroups", Int, Optional, "number of nodegroups"),
+ REP_ARG("subid", Int, Optional, "Id of subscription"),
+ REP_ARG("subkey", Int, Optional, "Key of subscription"),
+ REP_ARG("connected_rep", Int, Optional, "connected to rep"),
+ REP_ARG("connected_db", Int, Optional, "connected to db"),
+ REP_ARG("first", String, Optional, ""),
+ REP_ARG("last", String, Optional, ""),
+ REP_ARG("state_sub", Int, Optional, "state of subsription"),
+ REP_ARG("state", Int, Optional, "state"),
+ REP_END()
+ };
+
+ if (handle == 0) {
+ return -1;
+ }
+
+ if (handle->connected != 1) {
+ handle->last_error = EINVAL;
+ return -1;
+ }
+
+ const Properties *props;
+ Properties args;
+ Uint32 request = 0;
+ args.put("request", request);
+ args.put("id", *replicationId);
+ args.put("counter" , (Uint32)counter);
+ props = ndb_rep_call(handle, replication_reply, "rep query", &args);
+
+ BaseString result;
+ props->get("result", result);
+ if(strcmp(result.c_str(), "Ok") != 0)
+ {
+ delete props;
+ return 1;
+ }
+ state->queryCounter = counter;
+ unsigned int no_of_nodegroups;
+ props->get("no_of_nodegroups", &no_of_nodegroups);
+ state->no_of_nodegroups = no_of_nodegroups;
+
+ if(counter >= 0)
+ {
+ BaseString first, last;
+ props->get("first", first);
+ props->get("last", last);
+ convert2int((char*)first.c_str(), (char*)last.c_str(),
+ state->first , state->last );
+ } else
+ {
+ for(Uint32 i = 0; i<REPAPI_MAX_NODE_GROUPS; i++) {
+ state->first[i] = 0;
+ state->last[i] = 0;
+ }
+ }
+
+ unsigned int connected_rep = 0;
+ props->get("connected_rep", &connected_rep);
+ state->connected_rep = connected_rep;
+
+ unsigned int connected_db = 0;
+ props->get("connected_rep", &connected_db);
+ state->connected_db = connected_db;
+
+ unsigned int subid;
+ props->get("subid", &subid);
+ state->subid = subid;
+
+ unsigned int subkey;
+ props->get("subkey", &subkey);
+ state->subkey = subkey;
+
+ unsigned int _state;
+ props->get("state", &_state);
+ state->state = _state;
+
+ unsigned int state_sub;
+ props->get("state_sub", &state_sub);
+ state->state_sub = state_sub;
+
+ if(props == NULL) {
+ handle->last_error = EIO;
+ return -1;
+ }
+ delete props;
+
+ return 0;
+}
+
+
+extern "C"
+int
+ndb_rep_get_status(NdbRepHandle handle,
+ unsigned int* replication_id,
+ struct ndb_rep_reply* /*reply*/,
+ struct rep_state * repstate) {
+
+ const ParserRow<ParserDummy2> replication_reply[] = {
+ REP_CMD("global replication status reply", NULL, ""),
+ REP_ARG("result", String, Mandatory, "Error message"),
+ REP_ARG("id", Int, Optional, "Error message"),
+ REP_ARG("subid", Int, Optional, "Id of subscription"),
+ REP_ARG("subkey", Int, Optional, "Key of subscription"),
+ REP_ARG("connected_rep", Int, Optional, "connected to rep"),
+ REP_ARG("connected_db", Int, Optional, "connected to db"),
+ REP_ARG("state_sub", Int, Optional, "state of subsription"),
+ REP_ARG("state", Int, Optional, "state"),
+ REP_END()
+ };
+
+ if (handle == 0) {
+ return -1;
+ }
+
+ if (handle->connected != 1) {
+ handle->last_error = EINVAL;
+ return -1;
+ }
+
+ const Properties *reply;
+ Properties args;
+ Uint32 request = 0;
+ args.put("request", request);
+ reply = ndb_rep_call(handle, replication_reply, "rep status", &args);
+
+ if(reply == NULL) {
+ handle->last_error = EIO;
+ return -1;
+ }
+
+ Uint32 result;
+ reply->get("result", &result);
+ reply->get("id", replication_id);
+ reply->get("subid", (Uint32*)&repstate->subid);
+ reply->get("subkey", (Uint32*)&repstate->subkey);
+ reply->get("connected_rep", (Uint32*)&repstate->connected_rep);
+ reply->get("connected_db", (Uint32*)&repstate->connected_db);
+ reply->get("state", (Uint32*)&repstate->state);
+ reply->get("state_sub", (Uint32*)&repstate->state_sub);
+
+ delete reply;
+ return result;
+}
diff --git a/ndb/src/rep/repapi/repapi.h b/ndb/src/rep/repapi/repapi.h
new file mode 100644
index 00000000000..170e493cd86
--- /dev/null
+++ b/ndb/src/rep/repapi/repapi.h
@@ -0,0 +1,216 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REPAPI_H
+#define REPAPI_H
+
+/**
+ * @mainpage NDB Cluster REP API
+ *
+ * The NDB Cluster Replication API (REP API) consists of a C API
+ * which is used to:
+ * - Start and stop replication
+ * - Other administrative tasks
+ *
+ * The functions use simple ASCII based
+ * commands to interact with thw Replication Server.
+ *
+ *
+ * @section General Concepts
+ *
+ * Each REP API function call needs an rep_C_Api::NdbRepHandle
+ * which initally is created by
+ * calling the function ndb_rep_create_handle().
+ *
+ * A function can return:
+ * -# An integer value. If it returns 0 then this indicates success.
+ * -# A pointer value. If it returns NULL then check the latest error.
+ * If it didn't return NULL, then "something" is returned.
+ * This "something" has to be free:ed by the user of the REP API.
+ */
+
+/** @addtogroup REP_C_API
+ * @{
+ */
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#define REPAPI_MAX_NODE_GROUPS 4
+ /**
+ * The NdbRepHandle.
+ */
+ typedef struct ndb_rep_handle * NdbRepHandle;
+
+
+ /**
+ * Default reply from the server
+ */
+ struct ndb_rep_reply {
+ int return_code; ///< 0 if successful,
+ ///< otherwise error code
+ char message[256]; ///< Error or reply message.
+ };
+
+ enum QueryCounter {
+ PS = 0, ///< Stored on Primary System REP
+ SSReq = 1, ///< Requested for transfer to Standby System
+ SS = 2, ///< Stored on Standby System REP
+ AppReq = 3, ///< Requested to be applied to Standby System
+ App = 4, ///< Has been applied to Standby System
+ DelReq = 5, ///< Has been requested to be deleted on PS REP & SS REP
+ Subscription = 6,
+ ConnectionRep = 7,
+ ConnectionDb = 8
+ };
+
+
+ struct rep_state {
+ QueryCounter queryCounter;
+ unsigned int no_of_nodegroups;
+ unsigned int connected_rep;
+ unsigned int connected_db;
+ unsigned int subid;
+ unsigned int subkey;
+ unsigned int state;
+ unsigned int state_sub;
+ unsigned int first[REPAPI_MAX_NODE_GROUPS]; //4 = max no of nodegroups
+ unsigned int last[REPAPI_MAX_NODE_GROUPS]; //4 = max no of nodegroups
+ };
+
+
+
+
+
+
+ /***************************************************************************
+ * FUNCTIONS
+ ***************************************************************************/
+ /**
+ * Create a handle
+ *
+ * @return A handle != 0
+ * or 0 if failed to create one. (Check errno then).
+ */
+ NdbRepHandle ndb_rep_create_handle();
+
+ /**
+ * Destroy a handle
+ *
+ * @param handle Rep server handle
+ */
+ void ndb_rep_destroy_handle(NdbRepHandle * handle);
+
+ /**
+ * Get latest error associated with a handle
+ *
+ * @param handle Rep server handle
+ * @return Latest error.
+ */
+ int ndb_rep_get_latest_error(const NdbRepHandle handle);
+
+ /**
+ * Get latest error line associated with a handle
+ *
+ * @param handle Rep server handle.
+ * @return Latest error line.
+ */
+ int ndb_rep_get_latest_error_line(const NdbRepHandle handle);
+
+ /**
+ * Connect to a REP server
+ *
+ * @param handle Rep server handle.
+ * @param repsrv Hostname and port of the REP server,
+ * "hostname:port".
+ * @return 0 if OK, sets ndb_rep_handle->last_error otherwise.
+ */
+ int ndb_rep_connect(NdbRepHandle handle, const char * repsrv);
+
+ /**
+ * Disconnect from a REP server
+ *
+ * @param handle Rep server handle.
+ */
+ void ndb_rep_disconnect(NdbRepHandle handle);
+
+
+ /**
+ * Global Replication Command
+ *
+ * @param handle NDB REP handle.
+ * @param request Type of request
+ * @param replicationId Replication id is returned from function.
+ * @param reply Reply message.
+ * @param epoch Currenty used to STOP at a certain EPOCH
+ * @return 0 if successful, error code otherwise.
+ */
+ int ndb_rep_command(NdbRepHandle handle,
+ unsigned int request,
+ unsigned int* replicationId,
+ struct ndb_rep_reply* reply,
+ unsigned int epoch = 0);
+
+
+ /**
+ * Global Replication Command
+ *
+ * @param handle NDB REP handle.
+ * @param counter Type of request. If <0, then
+ "first" and "last" in repstate
+ is set to 0;x
+ * @param replicationId Replication id is returned from function.
+ * @param reply Reply message.
+ * @param repstate Struct containing queried data. (Note!
+ * All values are set in the struct, regardless
+ which QueryCounter that has been set
+ * @return 0 if successful, error code otherwise.
+ */
+ int ndb_rep_query(NdbRepHandle handle,
+ QueryCounter counter,
+ unsigned int* replicationId,
+ struct ndb_rep_reply* reply,
+ struct rep_state * repstate);
+
+
+/**
+ * @deprecated (will probably be). Can use ndb_rep_query instead.
+ */
+ int ndb_rep_get_status(NdbRepHandle handle,
+ unsigned int* replication_id,
+ struct ndb_rep_reply* /*reply*/,
+ struct rep_state * repstate);
+
+
+
+ enum RequestStatusCode {
+ OK = 0, ///< Everything OK
+ Error = 1, ///< Generic error
+ AlreadyExists = 2, ///< Entry already exists in list
+ NotExists = 3, ///< Entry does not exist in list
+ AlreadyStopped = 4
+ };
+
+
+
+#ifdef __cplusplus
+}
+#endif
+
+/** @} */
+
+#endif
diff --git a/ndb/src/rep/state/Channel.cpp b/ndb/src/rep/state/Channel.cpp
new file mode 100644
index 00000000000..1d573bad2f5
--- /dev/null
+++ b/ndb/src/rep/state/Channel.cpp
@@ -0,0 +1,487 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Channel.hpp"
+
+Channel::Channel()
+{
+ reset();
+}
+
+Channel::~Channel()
+{
+ /**
+ * Destroy list of selected tables
+ */
+ for(Uint32 i=0; i < m_selectedTables.size(); i++) {
+ delete m_selectedTables[i];
+ m_selectedTables[i] = 0;
+ }
+ m_selectedTables=0;
+}
+
+void
+Channel::reset()
+{
+ for (Uint32 i=0; i<MAX_NO_OF_NODE_GROUPS; i++) {
+ for (Uint32 j=0; j<NO_OF_POSITIONS; j++) {
+ state[i][j].set(1,0);
+ }
+ }
+ m_noOfNodeGroups = 0;
+ m_requestorEnabled = true;
+ m_transferEnabled = true;
+ m_applyEnabled = true;
+ m_deleteEnabled = true;
+ m_autoStartEnabled = false;
+ m_stopEpochId = intervalMax;
+ setSubKey(0);
+ setSubId(0);
+ m_stateSub = NO_SUBSCRIPTION_EXISTS;
+ m_stateRep = CONSISTENT;
+ m_metaScanEpochs = emptyInterval;
+ m_dataScanEpochs = emptyInterval;
+}
+
+bool
+Channel::requestTransfer(Uint32 nodeGrp, Interval * i)
+{
+ invariant();
+ Interval tmp1, tmp2;
+
+ // i = PS - SSReq - SS - App
+ intervalLeftMinus(state[nodeGrp][PS], state[nodeGrp][SSReq], &tmp1);
+ intervalLeftMinus(tmp1, state[nodeGrp][SS], &tmp2);
+ intervalLeftMinus(tmp2, state[nodeGrp][App], i);
+
+ i->onlyLeft(GREP_SYSTEM_TABLE_MAX_RANGE);
+ i->onlyUpToValue(m_stopEpochId);
+ if (i->isEmpty()) return false;
+
+ add(SSReq, nodeGrp, *i);
+ invariant();
+ return true;
+}
+
+bool
+Channel::requestApply(Uint32 nodeGrp, Uint32 * epoch)
+{
+ invariant();
+ Interval tmp1, tmp2;
+
+ // tmp2 = SS - AppReq - App
+ intervalLeftMinus(state[nodeGrp][SS], state[nodeGrp][AppReq], &tmp1);
+ intervalLeftMinus(tmp1, state[nodeGrp][App], &tmp2);
+
+ tmp2.onlyUpToValue(m_stopEpochId);
+ if (tmp2.isEmpty()) return false;
+ tmp2.onlyLeft(1);
+
+ // Check that all GCI Buffers for epoch exists in SS
+ for (Uint32 i=0; i<m_noOfNodeGroups; i++) {
+ if (!state[nodeGrp][SS].inInterval(tmp2.first()))
+ return false;
+ }
+
+ invariant();
+ add(AppReq, nodeGrp, tmp2);
+ invariant();
+ *epoch = tmp2.first();
+ return true;
+}
+
+bool
+Channel::requestDelete(Uint32 nodeGrp, Interval * i)
+{
+ invariant();
+ Interval tmp1;
+
+ // i = (App cut PS) - DelReq
+ intervalCut(state[nodeGrp][App], state[nodeGrp][PS], &tmp1);
+ intervalLeftMinus(tmp1, state[nodeGrp][DelReq], i);
+
+ if (i->isEmpty()) return false;
+ i->onlyLeft(GREP_SYSTEM_TABLE_MAX_RANGE);
+
+ invariant();
+ add(DelReq, nodeGrp, *i);
+ invariant();
+ return true;
+}
+
+void
+Channel::add(Position pos, Uint32 nodeGrp, const Interval i)
+{
+ Interval r;
+ intervalAdd(state[nodeGrp][pos], i, &r);
+ state[nodeGrp][pos].set(r);
+}
+
+void
+Channel::clear(Position p, Uint32 nodeGrp, const Interval i)
+{
+ Interval r;
+ intervalLeftMinus(state[nodeGrp][p], i, &r);
+ state[nodeGrp][p].set(r);
+}
+
+bool
+Channel::isSynchable(Uint32 nodeGrp)
+{
+ return true;
+ /*
+ @todo This should be implemented...
+
+ Interval tmp1, tmp2;
+ intervalAdd(state[nodeGrp][PS], state[nodeGrp][SSReq], &tmp1);
+ intervalAdd(tmp1, state[nodeGrp][SSReq], &tmp2);
+ intervalAdd(tmp2, state[nodeGrp][SS], &tmp1);
+ intervalAdd(tmp1, state[nodeGrp][AppReq], &tmp2);
+ intervalAdd(tmp2, state[nodeGrp][App], &tmp1);
+ if (intervalInclude(state[nodeGrp][PS], tmp1.right()))
+ return true;
+ else
+ return false;
+ */
+}
+
+/**
+ * Return the cut of all App:s.
+ */
+void
+Channel::getFullyAppliedEpochs(Interval * interval)
+{
+ if (m_noOfNodeGroups < 1) {
+ *interval = emptyInterval;
+ return;
+ }
+
+ *interval = universeInterval;
+ for (Uint32 i=0; i<m_noOfNodeGroups; i++) {
+ if (state[i][App].isEmpty()) {
+ *interval = emptyInterval;
+ return;
+ }
+
+ if (interval->first() < state[i][App].first()) {
+ interval->setFirst(state[i][App].first());
+ }
+ if (state[i][App].last() < interval->last()) {
+ interval->setLast(state[i][App].last());
+ }
+ }
+ interval->normalize();
+ return;
+}
+
+/**
+ * Return true if it is ok to remove the subscription and then stop channel
+ */
+bool
+Channel::isStoppable()
+{
+ /**
+ * Check that AppReq are empty for all nodegrps
+ */
+ for (Uint32 i=0; i<m_noOfNodeGroups; i++) {
+ if (!state[i][AppReq].isEmpty()) {
+ RLOG(("Stop disallowed. AppReq is non-empty"));
+ return false;
+ }
+ }
+
+ /**
+ * If stop immediately, then it is ok to stop now
+ */
+ if (m_stopEpochId == 0) {
+ RLOG(("Stop allowed. AppReq empty and immediate stop requested"));
+ return true;
+ }
+
+ /**
+ * If stop on a certain epoch, then
+ * check that stopEpochId is equal to the last applied epoch
+ */
+ Interval interval;
+ getFullyAppliedEpochs(&interval);
+ if (m_stopEpochId > interval.last()) {
+ RLOG(("Stop disallowed. AppReq empty. Stop %d, LastApplied %d",
+ m_stopEpochId, interval.last()));
+ return false;
+ }
+
+ return true;
+}
+
+GrepError::Code
+Channel::setStopEpochId(Uint32 n)
+{
+ /**
+ * If n equal to zero, use next possible epoch (max(App, AppReq))
+ */
+ if (n == 0) {
+ for (Uint32 i=0; i<m_noOfNodeGroups; i++) {
+ n = (state[i][App].last() > n) ? state[i][App].last() : n;
+ n = (state[i][AppReq].last() > n) ? state[i][AppReq].last() : n;
+ }
+ }
+
+ /**
+ * If n >= max(App, AppReq) then set value, else return error code
+ */
+ for (Uint32 i=0; i<m_noOfNodeGroups; i++) {
+ if (n < state[i][App].last()) return GrepError::ILLEGAL_STOP_EPOCH_ID;
+ if (n < state[i][AppReq].last()) return GrepError::ILLEGAL_STOP_EPOCH_ID;
+ }
+
+ m_stopEpochId = n;
+ return GrepError::NO_ERROR;
+};
+
+bool
+Channel::shouldStop()
+{
+ /**
+ * If (m_stopEpochId == App) then channel should stop
+ */
+ for (Uint32 i=0; i<m_noOfNodeGroups; i++) {
+ if(m_stopEpochId != state[i][App].last()) return false;
+ }
+ return true;
+}
+
+/*****************************************************************************
+ * SELECTIVE TABLE INTERFACE
+ *****************************************************************************/
+
+GrepError::Code
+Channel::addTable(const char * tableName)
+{
+ if(strlen(tableName)>MAX_TAB_NAME_SIZE)
+ return GrepError::REP_NOT_PROPER_TABLE;
+ /**
+ * No of separators are the number of '/' found in tableName
+ * since a table is defined as <db>/<schema>/tablename.
+ * if noOfSeparators is not equal to 2, then it is not a valid
+ * table name.
+ */
+ Uint32 noOfSeps = 0;
+ if(strlen(tableName) < 5)
+ return GrepError::REP_NOT_PROPER_TABLE;
+ for(Uint32 i =0; i < strlen(tableName); i++)
+ if(tableName[i]=='/')
+ noOfSeps++;
+ if(noOfSeps!=2)
+ return GrepError::REP_NOT_PROPER_TABLE;
+ table * t= new table(tableName);
+ for(Uint32 i=0; i<m_selectedTables.size(); i++) {
+ if(strcmp(tableName, m_selectedTables[i]->tableName)==0)
+ return GrepError::REP_TABLE_ALREADY_SELECTED;
+ }
+ m_selectedTables.push_back(t);
+ return GrepError::NO_ERROR;
+}
+
+GrepError::Code
+Channel::removeTable(const char * tableName)
+{
+ if(strlen(tableName)>MAX_TAB_NAME_SIZE)
+ return GrepError::REP_NOT_PROPER_TABLE;
+ /**
+ * No of separators are the number of '/' found in tableName
+ * since a table is defined as <db>/<schema>/tablename.
+ * If noOfSeparators is not equal to 2,
+ * then it is not a valid table name.
+ */
+ Uint32 noOfSeps = 0;
+ if(strlen(tableName) < 5)
+ return GrepError::REP_NOT_PROPER_TABLE;
+ for(Uint32 i =0; i < strlen(tableName); i++)
+ if(tableName[i]=='/')
+ noOfSeps++;
+ if(noOfSeps!=2)
+ return GrepError::REP_NOT_PROPER_TABLE;
+ for(Uint32 i=0; i<m_selectedTables.size(); i++) {
+ if(strcmp(tableName, m_selectedTables[i]->tableName)==0) {
+ delete m_selectedTables[i];
+ m_selectedTables.erase(i);
+ return GrepError::NO_ERROR;
+ }
+ }
+ return GrepError::REP_TABLE_NOT_FOUND;
+}
+
+void
+Channel::printTables()
+{
+ if(m_selectedTables.size() == 0)
+ ndbout_c("| ALL TABLES "
+ " |");
+ else {
+ for(Uint32 i=0; i<m_selectedTables.size(); i++)
+ ndbout_c("| %-69s |", m_selectedTables[i]->tableName);
+ }
+}
+
+Vector<struct table *> *
+Channel::getSelectedTables()
+{
+ if(m_selectedTables.size() == 0) return 0;
+ return &m_selectedTables;
+}
+
+/*****************************************************************************
+ * PRINT
+ *****************************************************************************/
+
+void
+Channel::print(Position pos)
+{
+ switch(pos){
+ case PS: ndbout << "PS Rep"; break;
+ case SSReq: ndbout << "Tra-Req"; break;
+ case SS: ndbout << "SS Rep"; break;
+ case AppReq: ndbout << "App-Req"; break;
+ case App: ndbout << "Applied"; break;
+ case DelReq: ndbout << "Del-Req"; break;
+ default: REPABORT("Unknown replication position");
+ }
+}
+
+void
+Channel::print()
+{
+ for (Uint32 i=0; i<m_noOfNodeGroups; i++) {
+ print(i);
+ }
+}
+
+void
+Channel::print(Position pos, Uint32 nodeGrp)
+{
+ print(pos);
+ if (state[nodeGrp][pos].first() == 1 && state[nodeGrp][pos].last() == 0) {
+ ndbout << " EMPTY";
+ } else {
+ ndbout << " [" << state[nodeGrp][pos].first() << "-"
+ << state[nodeGrp][pos].last() << "]";
+ }
+}
+
+static const char*
+channelline =
+"+-------------------------------------------------------------------------+\n"
+;
+
+void
+Channel::getEpochState(Position p,
+ Uint32 nodeGrp,
+ Uint32 * first,
+ Uint32 * last) {
+ if(state[nodeGrp][p].isEmpty()) {
+ *first = 1;
+ *last = 0;
+ return;
+ }
+ *first = state[nodeGrp][p].first();
+ *last = state[nodeGrp][p].last();
+}
+
+
+void
+Channel::print(Uint32 nodeGrp)
+{
+ ndbout << channelline;
+ ndbout_c("| | Meta scan |"
+ " Data scan |");
+ ndbout.print("| ");
+ if (m_metaScanEpochs.isEmpty()) {
+ ndbout.print("| ");
+ } else {
+ ndbout.print("| %10u-%-10u ",
+ m_metaScanEpochs.first(), m_metaScanEpochs.last());
+ }
+ if (m_dataScanEpochs.isEmpty()) {
+ ndbout_c("| |");
+ } else {
+ ndbout_c("| %10u-%-10u |",
+ m_dataScanEpochs.first(), m_dataScanEpochs.last());
+ }
+
+ /* --- */
+
+ ndbout << channelline;
+ ndbout_c("| Source Rep Server | Being Transfered |"
+ " Destination Rep Server |");
+ if (state[nodeGrp][PS].isEmpty()) {
+ ndbout.print("| ");
+ } else {
+ ndbout.print("| %10u-%-10u ",
+ state[nodeGrp][PS].first(), state[nodeGrp][PS].last());
+ }
+ if (state[nodeGrp][SSReq].isEmpty()) {
+ ndbout.print("| ");
+ } else {
+ ndbout.print("| %10u-%-10u ",
+ state[nodeGrp][SSReq].first(), state[nodeGrp][SSReq].last());
+ }
+ if (state[nodeGrp][SS].isEmpty()) {
+ ndbout_c("| |");
+ } else {
+ ndbout_c("| %10u-%-10u |",
+ state[nodeGrp][SS].first(), state[nodeGrp][SS].last());
+ }
+
+ /* --- */
+
+ ndbout << channelline;
+ ndbout_c("| Being Applied | Applied |"
+ " Being Deleted |");
+ if (state[nodeGrp][AppReq].isEmpty()) {
+ ndbout.print("| ");
+ } else {
+ ndbout.print("| %10u-%-10u ", state[nodeGrp][AppReq].first(),
+ state[nodeGrp][AppReq].last());
+ }
+ if (state[nodeGrp][App].isEmpty()) {
+ ndbout.print("| ");
+ } else {
+ ndbout.print("| %10u-%-10u ",
+ state[nodeGrp][App].first(), state[nodeGrp][App].last());
+ }
+ if (state[nodeGrp][DelReq].isEmpty()) {
+ ndbout_c("| |");
+ } else {
+ ndbout_c("| %10u-%-10u |",
+ state[nodeGrp][DelReq].first(), state[nodeGrp][DelReq].last());
+ }
+}
+
+/*****************************************************************************
+ * Private Methods
+ *****************************************************************************/
+
+void
+Channel::invariant()
+{
+ for (Uint32 j=0; j<MAX_NO_OF_NODE_GROUPS; j++)
+ {
+ if (!intervalDisjoint(state[j][SSReq], state[j][SS]))
+ REPABORT("Invariant 1 violated");
+ if (!intervalDisjoint(state[j][AppReq], state[j][App]))
+ REPABORT("Invariant 2 violated");
+ }
+}
diff --git a/ndb/src/rep/state/Channel.hpp b/ndb/src/rep/state/Channel.hpp
new file mode 100644
index 00000000000..cdf4eecca63
--- /dev/null
+++ b/ndb/src/rep/state/Channel.hpp
@@ -0,0 +1,206 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef CHANNEL_HPP
+#define CHANNEL_HPP
+
+#include "Interval.hpp"
+#include <rep/rep_version.hpp>
+#include <Vector.hpp>
+#include <ndb_limits.h>
+#include <GrepError.hpp>
+
+
+/**
+ * Max number of requested epochs from PS
+ */
+#define GREP_SYSTEM_TABLE_MAX_RANGE 20
+
+#define MAX_NO_OF_NODE_GROUPS 32
+
+/**
+ * This table struct is used in m_selectedTables
+ */
+struct table{
+ table(const char * n) {strncpy(tableName, n, MAX_TAB_NAME_SIZE);}
+ char tableName[MAX_TAB_NAME_SIZE];
+};
+
+/**
+ * @class Channel
+ * @brief Represents location of various epochs belonging to a subscription
+ */
+class Channel {
+public:
+ enum StateSub
+ {
+ NO_SUBSCRIPTION_EXISTS,
+
+ CREATING_SUBSCRIPTION_ID,
+ SUBSCRIPTION_ID_CREATED,
+
+ STARTING_SUBSCRIPTION,
+ SUBSCRIPTION_STARTED
+ };
+
+ enum StateRep
+ {
+ CONSISTENT, ///< Consistent database. Grep not running.
+ METALOG_STARTING, ///< Starting. Starting METALOG subscription
+ METALOG_STARTED,
+ METASCAN_STARTING, ///< Starting. Starting METASCAN subscription
+ METASCAN_COMPLETED,
+ DATALOG_STARTING, ///< Starting. Starting DATALOG subscription
+ DATALOG_STARTED,
+ DATASCAN_STARTING, ///< Starting. Starting DATASCAN subscription
+ DATASCAN_COMPLETED,
+ LOG, ///< Started. Cons/Inconsistent. Grep running.
+ ///< All scan records have been applied.
+ STOPPING ///< Channel is stopping
+ };
+
+ /**
+ * Storage "positions" of Epochs
+ */
+ enum Position {
+ PS = 0, ///< Stored on Primary System REP
+ SSReq = 1, ///< Requested for transfer to Standby System
+ SS = 2, ///< Stored on Standby System REP
+ AppReq = 3, ///< Requested to be applied to Standby System
+ App = 4, ///< Has been applied to Standby System
+ DelReq = 5, ///< Has been requested to be deleted on PS REP & SS REP
+ NO_OF_POSITIONS = 6
+ }; //DONT FORGET TO ADD STUFF in position2Name if u add somehting here,
+
+ /***************************************************************************
+ * CONSTRUCTOR / DESTRUCTOR
+ ***************************************************************************/
+ Channel();
+ ~Channel();
+
+ /**
+ * Get and set no of nodegroups that actually exists on PS
+ */
+ void setNoOfNodeGroups(Uint32 n) { m_noOfNodeGroups = n; };
+ Uint32 getNoOfNodeGroups() { return m_noOfNodeGroups; };
+ void getEpochState(Position p,
+ Uint32 nodeGrp,
+ Uint32 * first,
+ Uint32 * last);
+ Uint32 getEpochState(Position p, Uint32 nodegroup);
+ bool m_requestorEnabled;
+ bool m_transferEnabled;
+ bool m_applyEnabled;
+ bool m_deleteEnabled;
+ bool m_autoStartEnabled;
+
+ /***************************************************************************
+ * GETTERS and SETTERS
+ ***************************************************************************/
+ bool requestTransfer(Uint32 nodeGrp, Interval * i);
+ bool requestApply(Uint32 nodeGrp, Uint32 * epoch);
+ bool requestDelete(Uint32 nodeGrp, Interval * i);
+
+ void add(Position pos, Uint32 nodeGrp, const Interval i);
+ void clear(Position pos, Uint32 nodeGrp, const Interval i);
+
+ void setSubId(Uint32 subId) { m_subId=subId; };
+ Uint32 getSubId() { return m_subId; };
+
+ Uint32 getSubKey() { return m_subKey; };
+ void setSubKey(Uint32 subKey) { m_subKey=subKey; };
+
+ bool isSynchable(Uint32 nodeGrp);
+ GrepError::Code addTable(const char * tableName);
+ GrepError::Code removeTable(const char * tableName);
+ void printTables();
+ bool isSelective() {return m_selectedTables.size()>0;};
+ Vector<struct table *> * getSelectedTables();
+
+ void reset();
+
+ StateRep getState() { return m_stateRep; }
+ void setState(StateRep sr) { m_stateRep = sr; }
+
+ StateSub getStateSub() { return m_stateSub; }
+ void setStateSub(StateSub ss) { m_stateSub = ss; }
+
+ Interval getMetaScanEpochs() { return m_metaScanEpochs; }
+ void setMetaScanEpochs(Interval i) { m_metaScanEpochs = i; }
+ Interval getDataScanEpochs() { return m_dataScanEpochs; }
+ void setDataScanEpochs(Interval i) { m_dataScanEpochs = i; }
+
+ GrepError::Code setStopEpochId(Uint32 n);
+ Uint32 getStopEpochId() { return m_stopEpochId; };
+
+ bool isStoppable();
+ bool shouldStop();
+
+ bool subscriptionExists() { return (m_subId != 0 && m_subKey != 0); }
+
+ /***************************************************************************
+ * GETTERS
+ ***************************************************************************/
+ Uint32 getFirst(Position pos, Uint32 nodeGrp) {
+ return state[nodeGrp][pos].first();
+ }
+
+ Uint32 getLast(Position pos, Uint32 nodeGrp) {
+ return state[nodeGrp][pos].last();
+ }
+
+ void getFullyAppliedEpochs(Interval * i);
+
+ /***************************************************************************
+ * PRINT METHODS
+ ***************************************************************************/
+ void print();
+ void print(Position pos);
+ void print(Position pos, Uint32 nodeGrp);
+ void print(Uint32 nodeGrp);
+
+ /***************************************************************************
+ * PUBLIC ATTRIBUTES
+ ***************************************************************************/
+
+private:
+ /***************************************************************************
+ * PRIVATE ATTRIBUTES
+ ***************************************************************************/
+ StateRep m_stateRep; // Replication state
+ StateSub m_stateSub; // Subscription state
+
+ Uint32 m_subId;
+ Uint32 m_subKey;
+
+ Uint32 m_noOfNodeGroups; // Number of node grps in this channel
+ Uint32 m_stopEpochId; // Epoch id to stop subscription
+
+ Interval state[MAX_NO_OF_NODE_GROUPS][NO_OF_POSITIONS];
+
+ Interval m_metaScanEpochs;
+ Interval m_dataScanEpochs;
+
+
+ Vector<struct table *> m_selectedTables;
+ void invariant(); // Abort if channel metadata is inconsistent
+ char * position2Name(Position p);
+public:
+ bool copy(Position from, Position to, Uint32 range,
+ Uint32 * f, Uint32 * l, Uint32 nodeGrp);
+};
+
+#endif
diff --git a/ndb/src/rep/state/Interval.cpp b/ndb/src/rep/state/Interval.cpp
new file mode 100644
index 00000000000..75697fa7548
--- /dev/null
+++ b/ndb/src/rep/state/Interval.cpp
@@ -0,0 +1,169 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "Interval.hpp"
+
+Uint32 max(Uint32 a, Uint32 b) { return a > b ? a : b; }
+Uint32 min(Uint32 a, Uint32 b) { return a < b ? a : b; }
+
+Interval::Interval()
+{
+ set(1, 0); // EmptyInterval
+}
+
+Interval::Interval(Uint32 f, Uint32 l)
+{
+ set(f, l);
+}
+
+bool
+Interval::isEmpty() const
+{
+ return (m_first > m_last) ? true : false;
+}
+
+bool
+Interval::isEqual(Uint32 a, Uint32 b) const
+{
+ return (a==m_first && b==m_last);
+}
+
+bool
+Interval::inInterval(Uint32 a) const
+{
+ return (m_first <= a && a <= m_last);
+}
+
+void
+Interval::set(Uint32 first, Uint32 last)
+{
+ m_first = first;
+ m_last = last;
+ normalize();
+}
+
+void
+Interval::set(const Interval i)
+{
+ m_first = i.first();
+ m_last = i.last();
+ normalize();
+}
+
+void
+Interval::setFirst(Uint32 first)
+{
+ m_first = first;
+}
+
+void
+Interval::setLast(Uint32 last)
+{
+ m_last = last;
+}
+
+void
+Interval::onlyLeft(Uint32 n)
+{
+ if (size() > n) m_last = m_first + n - 1;
+}
+
+void
+Interval::onlyUpToValue(Uint32 n)
+{
+ m_last = min(n, m_last);
+ normalize();
+}
+
+/*****************************************************************************/
+
+void
+Interval::normalize()
+{
+ if (isEmpty()) {
+ m_first = 1;
+ m_last = 0;
+ }
+}
+
+
+/*****************************************************************************/
+
+bool
+intervalAdd(const Interval a, const Interval b, Interval * r)
+{
+ /**
+ * Non-empty disjoint intervals
+ */
+ if (!a.isEmpty() &&
+ !b.isEmpty() &&
+ (a.last() + 1 < b.first() ||
+ b.last() + 1 < a.first()) ) {
+ return false; // Illegal add
+ }
+
+ /**
+ * Interval A empty -> return B
+ */
+ if (a.isEmpty()) {
+ r->set(b);
+ return true;
+ }
+
+ /**
+ * Interval B empty -> return A
+ */
+ if (b.isEmpty()) {
+ r->set(a);
+ return true;
+ }
+
+ r->set(min(a.first(), b.first()),
+ max(a.last(), b.last()));
+ return true;
+}
+
+/**
+ * Subtract the left part of interval 'a' up to last of 'b'.
+ *
+ * @note This is NOT ordinary arithmetic interval minus.
+ * In ordinary arithmetic, [11-25] - [12-15] would be undefined,
+ * but here it is [11-25] - [12-15] = [16-25].
+ */
+void
+intervalLeftMinus(const Interval a, const Interval b, Interval * r)
+{
+ if(b.last() != intervalMax)
+ r->set(max(a.first(), b.last()+1), a.last());
+ else
+ r->set(max(a.first(), intervalMax), a.last());
+}
+
+void
+intervalCut(const Interval a, const Interval b, Interval * r)
+{
+ r->set(max(a.first(), b.first()), min(a.last(), b.last()));
+ r->normalize();
+}
+
+bool
+intervalDisjoint(const Interval a, const Interval b)
+{
+ return (a.isEmpty() ||
+ b.isEmpty() ||
+ a.last() < b.first() ||
+ b.last() < a.first());
+}
diff --git a/ndb/src/rep/state/Interval.hpp b/ndb/src/rep/state/Interval.hpp
new file mode 100644
index 00000000000..935adaf26b1
--- /dev/null
+++ b/ndb/src/rep/state/Interval.hpp
@@ -0,0 +1,107 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef INTERVAL_HPP
+#define INTERVAL_HPP
+
+#include <NdbOut.hpp>
+#include <ndb_types.h>
+
+/**
+ * @class Interval
+ * @brief Represents an interval
+ */
+class Interval {
+public:
+ Interval();
+ Interval(Uint32, Uint32);
+
+ /**
+ * Getters of first and last
+ */
+ inline Uint32 first() const { return m_first; }
+ inline Uint32 last() const { return m_last; }
+
+ /**
+ * Check if interval is empty
+ */
+ bool isEmpty() const;
+ bool isEqual(Uint32 a, Uint32 b) const;
+ bool inInterval(Uint32 a) const;
+
+ /**
+ * Size of interval
+ */
+ Uint32 size() const {
+ return (!isEmpty()) ? m_last - m_first + 1 : 0;
+ }
+
+ /**
+ * Set interval
+ */
+ void set(Uint32 first, Uint32 last);
+ void set(const Interval i);
+
+ void setFirst(Uint32 first);
+ void setLast(Uint32 last);
+
+ /**
+ * Reduce the interval to only the n left elements of the
+ * interval. If the interval is shorter than n, then
+ * interval is not changed.
+ */
+ void onlyLeft(Uint32 n);
+
+ /**
+ * Reduce the interval to have at most the value n
+ * as the last value.
+ * This method can make the interval empty.
+ */
+ void onlyUpToValue(Uint32 n);
+
+ /**
+ * Print
+ */
+ void print() {
+ ndbout << "[" << m_first << "," << m_last << "]";
+ }
+
+ void normalize();
+private:
+ Uint32 m_first;
+ Uint32 m_last;
+};
+
+const Uint32 intervalMin = 0;
+const Uint32 intervalMax = 0xffffffff;
+const Interval emptyInterval(1, 0);
+const Interval universeInterval(intervalMin, intervalMax);
+
+/**
+ * @return true if intervals could be added
+ */
+bool intervalAdd(const Interval a, const Interval b, Interval * c);
+
+void intervalLeftMinus(const Interval a, const Interval b, Interval * c);
+
+void intervalCut(const Interval a, const Interval b, Interval * c);
+
+/**
+ * @return true if intervals are disjoint
+ */
+bool intervalDisjoint(const Interval a, const Interval b);
+
+#endif
diff --git a/ndb/src/rep/state/Makefile b/ndb/src/rep/state/Makefile
new file mode 100644
index 00000000000..3eed69a97dd
--- /dev/null
+++ b/ndb/src/rep/state/Makefile
@@ -0,0 +1,17 @@
+include .defs.mk
+
+TYPE := repserver kernel
+
+ARCHIVE_TARGET := reprequestor
+
+DIR := testRepState \
+ testInterval
+
+SOURCES = RepState.cpp \
+ RepStateEvent.cpp \
+ RepStateRequests.cpp \
+ \
+ Channel.cpp \
+ Interval.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/rep/state/RepState.cpp b/ndb/src/rep/state/RepState.cpp
new file mode 100644
index 00000000000..a34bff25d7f
--- /dev/null
+++ b/ndb/src/rep/state/RepState.cpp
@@ -0,0 +1,865 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "RepState.hpp"
+
+#include <signaldata/SumaImpl.hpp>
+#include <NdbApiSignal.hpp>
+#include <Properties.hpp>
+//#define DBUG_REQUESTOR
+
+#ifdef DBUG_REQUESTOR
+#define DBUG_REQUESTOR_PRINT(X) ndbout_c(X);
+#else
+#define DBUG_REQUESTOR_PRINT(X)
+#endif
+
+/****************************************************************************
+ * Constructor / Destructor / Init
+ ****************************************************************************/
+RepState::RepState()
+{
+ m_connected = UNKNOWN;
+ m_repConnected = UNKNOWN;
+ m_mutex = NdbMutex_Create();
+ m_stopEpoch = 0;
+ m_subIdToRemove = 0;
+ m_subKeyToRemove = 0;
+}
+
+RepState::~RepState()
+{
+ NdbMutex_Destroy(m_mutex);
+}
+
+void
+RepState::setSubscriptionRequests(FuncRequestCreateSubscriptionId f1,
+ FuncRequestCreateSubscription f2,
+ FuncRequestRemoveSubscription f3)
+{
+ m_funcRequestCreateSubscriptionId = f1;
+ m_funcRequestCreateSubscription = f2;
+ m_funcRequestRemoveSubscription = f3;
+}
+
+void
+RepState::setIntervalRequests(FuncRequestTransfer f1,
+ FuncRequestApply f2,
+ FuncRequestDeleteSS f3,
+ FuncRequestDeletePS f4)
+{
+ m_funcRequestTransfer = f1;
+ m_funcRequestApply = f2;
+ m_funcRequestDeleteSS = f3;
+ m_funcRequestDeletePS = f4;
+}
+
+void
+RepState::setStartRequests(FuncRequestStartMetaLog * f5,
+ FuncRequestStartDataLog * f6,
+ FuncRequestStartMetaScan * f7,
+ FuncRequestStartDataScan * f8,
+ FuncRequestEpochInfo * f9)
+{
+ m_funcRequestStartMetaLog = f5;
+ m_funcRequestStartDataLog = f6;
+ m_funcRequestStartMetaScan = f7;
+ m_funcRequestStartDataScan = f8;
+ m_funcRequestEpochInfo = f9;
+}
+
+
+/****************************************************************************
+ * Private Helper functions
+ ****************************************************************************/
+
+void
+RepState::requestTransfer(NdbApiSignal * signal)
+{
+ DBUG_REQUESTOR_PRINT("RepState: Transfer calculations started");
+ for(Uint32 nodeGrp=0; nodeGrp<m_channel.getNoOfNodeGroups(); nodeGrp++) {
+ DBUG_REQUESTOR_PRINT("RepState: Transfer calc for node grp");
+ Interval i;
+ if (m_channel.requestTransfer(nodeGrp, &i)) {
+ m_funcRequestTransfer(m_extSender, signal, nodeGrp, i.first(), i.last());
+ }
+ }
+}
+
+void
+RepState::requestApply(NdbApiSignal * signal)
+{
+ DBUG_REQUESTOR_PRINT("RepState: Apply calculations started");
+ for(Uint32 nodeGrp=0; nodeGrp<m_channel.getNoOfNodeGroups(); nodeGrp++) {
+ DBUG_REQUESTOR_PRINT("RepState: Apply calc for node grp");
+ Uint32 gci;
+ if (m_channel.requestApply(nodeGrp, &gci)) {
+ Uint32 force = (m_channel.getState() == Channel::LOG) ? 0 : 1;
+ m_funcRequestApply(m_applier, signal, nodeGrp, gci, gci, force);
+ }
+ }
+}
+
+void
+RepState::requestDelete(NdbApiSignal * signal)
+{
+ DBUG_REQUESTOR_PRINT("RepState: Delete calculations started");
+ for(Uint32 nodeGrp=0; nodeGrp<m_channel.getNoOfNodeGroups(); nodeGrp++) {
+ DBUG_REQUESTOR_PRINT("RepState: Delete calc for node grp");
+ Interval i;
+ if (m_channel.requestDelete(nodeGrp, &i)){
+ m_funcRequestDeleteSS(m_gciContainer, signal, nodeGrp,
+ i.first(), i.last());
+ m_funcRequestDeletePS(m_extSender, signal, nodeGrp, i.first(), i.last());
+ }
+ }
+}
+
+void
+RepState::requestEpochInfo(NdbApiSignal * signal)
+{
+ DBUG_REQUESTOR_PRINT("RepState: Epoch Info calculations");
+ for(Uint32 nodeGrp=0; nodeGrp<m_channel.getNoOfNodeGroups(); nodeGrp++) {
+ m_funcRequestEpochInfo(m_extSender, signal, nodeGrp);
+ }
+}
+
+/****************************************************************************
+ * Public
+ ****************************************************************************/
+
+GrepError::Code
+RepState::add(Channel::Position s, Uint32 nodeGrp, const Interval i)
+{
+ m_channel.add(s, nodeGrp, i);
+
+ if(s == Channel::PS)
+ {
+ m_connected = CONNECTED;
+ m_connected_counter = 0;
+ }
+
+ Interval fullEpochs;
+ m_channel.getFullyAppliedEpochs(&fullEpochs);
+ if(s == Channel::App &&
+ m_channel.getState() == Channel::DATASCAN_COMPLETED &&
+ fullEpochs.last() >= m_channel.getDataScanEpochs().last() &&
+ fullEpochs.last() >= m_channel.getMetaScanEpochs().last())
+ {
+ RLOG(("[%d-%d] fully applied. Channel state changed to LOG",
+ fullEpochs.first(), fullEpochs.last()));
+ m_channel.setState(Channel::LOG);
+ disableAutoStart();
+ }
+
+ return GrepError::NO_ERROR;
+}
+
+GrepError::Code
+RepState::clear(Channel::Position s, Uint32 nodeGrp, const Interval i)
+{
+ m_channel.clear(s, nodeGrp, i);
+ return GrepError::NO_ERROR;
+}
+
+/****************************************************************************
+ * Execute
+ *
+ * This method should only be called from Requestor!
+ ****************************************************************************/
+
+GrepError::Code
+RepState::protectedExecute()
+{
+ GrepError::Code err;
+
+ NdbMutex_Lock(m_mutex);
+
+ NdbApiSignal* signal = m_extSender->getSignal();
+ if (signal == NULL) {
+ err = GrepError::COULD_NOT_ALLOCATE_MEM_FOR_SIGNAL;
+ } else {
+ err = execute(signal);
+ }
+ NdbMutex_Unlock(m_mutex);
+ return err;
+}
+
+GrepError::Code
+RepState::execute(NdbApiSignal* signal)
+{
+ Uint32 subId = m_channel.getSubId();
+ Uint32 subKey = m_channel.getSubKey();
+
+ if (!m_channel.m_requestorEnabled)
+ return GrepError::NO_ERROR;
+
+ /**
+ * @todo Should have subscriptions in here
+ */
+ requestEpochInfo(signal);
+
+ /**
+ * Update connected counter (Silence time)
+ */
+ m_connected_counter++;
+ if (m_connected_counter > REQUESTOR_EXECUTES_NEEDED_FOR_UNKNOWN_CONNECTION) {
+ m_connected = UNKNOWN;
+ }
+
+ switch (m_channel.getState())
+ {
+ case Channel::CONSISTENT:
+ if (isAutoStartEnabled()) {
+ switch (m_channel.getStateSub())
+ {
+ case Channel::NO_SUBSCRIPTION_EXISTS:
+ m_funcRequestCreateSubscriptionId(m_extSender, signal);
+ m_channel.setStateSub(Channel::CREATING_SUBSCRIPTION_ID);
+ break;
+
+ case Channel::CREATING_SUBSCRIPTION_ID:
+ break;
+
+ case Channel::SUBSCRIPTION_ID_CREATED:
+ if(m_channel.isSelective())
+ m_funcRequestCreateSubscription(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey(),
+ m_channel.getSelectedTables());
+ else
+ m_funcRequestCreateSubscription(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey(),
+ 0);
+ m_channel.setStateSub(Channel::STARTING_SUBSCRIPTION);
+ break;
+
+ case Channel::STARTING_SUBSCRIPTION:
+ break;
+
+ case Channel::SUBSCRIPTION_STARTED:
+ m_funcRequestStartMetaLog(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey());
+ m_channel.setState(Channel::METALOG_STARTING);
+ break;
+ }
+ }
+ break;
+
+ case Channel::METALOG_STARTING:
+ break;
+
+ case Channel::METALOG_STARTED:
+ if (isAutoStartEnabled()) {
+ m_funcRequestStartMetaScan(m_extSender, signal, subId, subKey);
+ m_channel.setState(Channel::METASCAN_STARTING);
+ }
+ break;
+
+ case Channel::METASCAN_STARTING:
+ break;
+
+ case Channel::METASCAN_COMPLETED:
+ if (isAutoStartEnabled()) {
+ m_funcRequestStartDataLog(m_extSender, signal, subId, subKey);
+ m_channel.setState(Channel::DATALOG_STARTING);
+ }
+ break;
+
+ case Channel::DATALOG_STARTING:
+ break;
+
+ case Channel::DATALOG_STARTED:
+ if (isAutoStartEnabled()) {
+ m_funcRequestStartDataScan(m_extSender, signal, subId, subKey);
+ m_channel.setState(Channel::DATASCAN_STARTING);
+ }
+ break;
+
+ case Channel::DATASCAN_STARTING:
+ break;
+
+ case Channel::DATASCAN_COMPLETED:
+ break;
+
+ case Channel::LOG:
+ if (m_channel.shouldStop()) {
+ disableTransfer();
+ m_channel.setState(Channel::STOPPING);
+ }
+ break;
+
+ case Channel::STOPPING:
+ if (m_channel.m_transferEnabled)
+ {
+ REPABORT("Illegal stopping state while transfer is still enabled");
+ }
+ /**
+ * check if channel has a subscription, if not,
+ * check if we have marked a subscription that we want to remove
+ * and remove it. This is used to clean up "dangling subscriptions"
+ * after various crashes
+ */
+ if(!m_channel.subscriptionExists())
+ {
+ if(m_subIdToRemove && m_subKeyToRemove)
+ {
+ m_funcRequestRemoveSubscription(m_extSender, signal,
+ m_subIdToRemove,
+ m_subKeyToRemove);
+ eventSubscriptionDeleted( m_subIdToRemove,
+ m_subKeyToRemove);
+ return GrepError::NO_ERROR;
+ }
+ else {
+ return GrepError::SUBSCRIPTION_ID_NOT_FOUND;
+ }
+ } else {
+ if (m_channel.isStoppable())
+ {
+
+ m_funcRequestRemoveSubscription(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey());
+ eventSubscriptionDeleted(m_channel.getSubId(),
+ m_channel.getSubKey());
+ }
+ else
+ return GrepError::CHANNEL_NOT_STOPPABLE;
+
+ }
+ break;
+
+ default:
+ REPABORT("Illegal replication state");
+ }
+ if (m_channel.m_transferEnabled) requestTransfer(signal);
+ if (m_channel.m_applyEnabled) requestApply(signal);
+ if (m_channel.m_deleteEnabled) requestDelete(signal);
+ return GrepError::NO_ERROR;
+}
+
+/****************************************************************************
+ * Request
+ *
+ * This method should only be called from Main Thread!
+ ****************************************************************************/
+
+GrepError::Code
+RepState::protectedRequest(GrepReq::Request req, Uint32 arg)
+{
+ return protectedRequest(req, arg, 0);
+}
+
+GrepError::Code
+RepState::protectedRequest(GrepReq::Request req, Uint32 arg1, Uint32 arg2)
+{
+ GrepError::Code code;
+ NdbMutex_Lock(m_mutex);
+
+ NdbApiSignal* signal = m_extSender->getSignal();
+ if (signal == NULL) {
+ code = GrepError::COULD_NOT_ALLOCATE_MEM_FOR_SIGNAL;
+ } else {
+ code = request(req, arg1, arg2, signal);
+ }
+
+ NdbMutex_Unlock(m_mutex);
+ return code;
+}
+
+GrepError::Code
+RepState::protectedAddTable(const char * fullTableName)
+{
+ GrepError::Code code;
+ NdbMutex_Lock(m_mutex);
+ code = m_channel.addTable(fullTableName);
+ NdbMutex_Unlock(m_mutex);
+ return code;
+}
+
+GrepError::Code
+RepState::protectedRemoveTable(const char * fullTableName)
+{
+ GrepError::Code code;
+ if(m_channel.getStateSub() != Channel::NO_SUBSCRIPTION_EXISTS)
+ return GrepError::START_ALREADY_IN_PROGRESS;
+ NdbMutex_Lock(m_mutex);
+ code = m_channel.removeTable(fullTableName);
+ NdbMutex_Unlock(m_mutex);
+ return code;
+}
+
+GrepError::Code
+RepState::request(GrepReq::Request request, Uint32 arg1, Uint32 arg2,
+ NdbApiSignal* signal)
+{
+ switch (request)
+ {
+ /*************************************************************************
+ * STATUS etc
+ *************************************************************************/
+
+ case GrepReq::STATUS:
+ printStatus();
+ break;
+
+ case GrepReq::REMOVE_BUFFERS:
+ return GrepError::NOT_YET_IMPLEMENTED;
+
+ /*************************************************************************
+ * START
+ *************************************************************************/
+
+ case GrepReq::CREATE_SUBSCR:
+ if (m_channel.getStateSub() != Channel::NO_SUBSCRIPTION_EXISTS)
+ return GrepError::SUBSCRIPTION_ID_ALREADY_EXIST;
+
+ m_funcRequestCreateSubscriptionId(m_extSender, signal);
+ m_channel.setStateSub(Channel::CREATING_SUBSCRIPTION_ID);
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START_SUBSCR:
+ if (m_channel.getState() == Channel::STOPPING)
+ return GrepError::ILLEGAL_ACTION_WHEN_STOPPING;
+ if (m_channel.getStateSub() != Channel::SUBSCRIPTION_ID_CREATED)
+ return GrepError::SUBSCRIPTION_ID_NOT_FOUND;
+ if(m_channel.isSelective())
+ m_funcRequestCreateSubscription(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey(),
+ m_channel.getSelectedTables());
+ else
+ m_funcRequestCreateSubscription(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey(),
+ 0);
+ m_channel.setStateSub(Channel::STARTING_SUBSCRIPTION);
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START_METALOG:
+ if (m_channel.getState() == Channel::STOPPING)
+ return GrepError::ILLEGAL_ACTION_WHEN_STOPPING;
+ if (m_channel.getStateSub() != Channel::SUBSCRIPTION_STARTED)
+ return GrepError::SUBSCRIPTION_NOT_STARTED;
+ if (m_channel.getState() != Channel::CONSISTENT)
+ return GrepError::START_OF_COMPONENT_IN_WRONG_STATE;
+
+ m_funcRequestStartMetaLog(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey());
+ m_channel.setState(Channel::METALOG_STARTING);
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START_METASCAN:
+ if (m_channel.getState() == Channel::STOPPING)
+ return GrepError::ILLEGAL_ACTION_WHEN_STOPPING;
+ if (m_channel.getStateSub() != Channel::SUBSCRIPTION_STARTED)
+ return GrepError::SUBSCRIPTION_NOT_STARTED;
+ if (m_channel.getState() != Channel::METALOG_STARTED)
+ return GrepError::START_OF_COMPONENT_IN_WRONG_STATE;
+
+ m_funcRequestStartMetaScan(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey());
+ m_channel.setState(Channel::METASCAN_STARTING);
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START_DATALOG:
+ if (m_channel.getState() == Channel::STOPPING)
+ return GrepError::ILLEGAL_ACTION_WHEN_STOPPING;
+ if (m_channel.getStateSub() != Channel::SUBSCRIPTION_STARTED)
+ return GrepError::SUBSCRIPTION_NOT_STARTED;
+ if (m_channel.getState() != Channel::METASCAN_COMPLETED)
+ return GrepError::START_OF_COMPONENT_IN_WRONG_STATE;
+
+ m_funcRequestStartDataLog(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey());
+ m_channel.setState(Channel::DATALOG_STARTING);
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START_DATASCAN:
+ if (m_channel.getState() == Channel::STOPPING)
+ return GrepError::ILLEGAL_ACTION_WHEN_STOPPING;
+ if (m_channel.getStateSub() != Channel::SUBSCRIPTION_STARTED)
+ return GrepError::SUBSCRIPTION_NOT_STARTED;
+ if (m_channel.getState() != Channel::DATALOG_STARTED)
+ return GrepError::START_OF_COMPONENT_IN_WRONG_STATE;
+
+ m_funcRequestStartDataScan(m_extSender, signal,
+ m_channel.getSubId(),
+ m_channel.getSubKey());
+ m_channel.setState(Channel::DATASCAN_STARTING);
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START_REQUESTOR:
+ enable();
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START_TRANSFER:
+ if (m_channel.getState() == Channel::STOPPING)
+ return GrepError::ILLEGAL_ACTION_WHEN_STOPPING;
+ enableTransfer();
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START_APPLY:
+ enableApply();
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START_DELETE:
+ enableDelete();
+ return GrepError::NO_ERROR;
+
+ case GrepReq::START:
+ if (isAutoStartEnabled())
+ return GrepError::START_ALREADY_IN_PROGRESS;
+
+ enableAutoStart();
+ return GrepError::NO_ERROR;
+
+ /*************************************************************************
+ * STOP
+ *************************************************************************/
+
+ case GrepReq::STOP:
+ if (m_channel.getStateSub() == Channel::NO_SUBSCRIPTION_EXISTS)
+ return GrepError::SUBSCRIPTION_NOT_STARTED;
+ if (m_channel.getState() == Channel::STOPPING)
+ return GrepError::ILLEGAL_ACTION_WHEN_STOPPING;
+
+ if (arg1 == 0) {
+ /**
+ * Stop immediately
+ */
+ disableTransfer();
+ m_channel.setState(Channel::STOPPING);
+ m_channel.setStopEpochId(0);
+ return GrepError::NO_ERROR;
+ } else {
+ /**
+ * Set future stop epoch
+ */
+ return m_channel.setStopEpochId(arg1);
+ }
+
+ case GrepReq::STOP_SUBSCR:
+ {
+ if(m_subIdToRemove == 0 && m_subKeyToRemove == 0) {
+ m_subIdToRemove = arg1;
+ m_subKeyToRemove = arg2;
+ } else {
+ return GrepError::ILLEGAL_ACTION_WHEN_STOPPING;
+ }
+
+ if(m_channel.getSubId() != 0 && m_channel.getSubKey() != 0)
+ return GrepError::ILLEGAL_USE_OF_COMMAND;
+ if (m_channel.getState() == Channel::STOPPING)
+ return GrepError::ILLEGAL_ACTION_WHEN_STOPPING;
+ disableTransfer();
+ m_channel.setState(Channel::STOPPING);
+ return GrepError::NO_ERROR;
+ }
+ case GrepReq::STOP_METALOG:
+ case GrepReq::STOP_METASCAN:
+ case GrepReq::STOP_DATALOG:
+ case GrepReq::STOP_DATASCAN:
+ return GrepError::NOT_YET_IMPLEMENTED;
+
+ case GrepReq::STOP_REQUESTOR:
+ disable();
+ return GrepError::NO_ERROR;
+
+ case GrepReq::STOP_TRANSFER:
+ disableTransfer();
+ return GrepError::NO_ERROR;
+
+ case GrepReq::STOP_APPLY:
+ disableApply();
+ return GrepError::NO_ERROR;
+
+ case GrepReq::STOP_DELETE:
+ disableDelete();
+ return GrepError::NO_ERROR;
+
+ default:
+ ndbout_c("RepCommandInterpreter: Illegal request received");
+ return GrepError::NOT_YET_IMPLEMENTED;
+ }
+ return GrepError::NOT_YET_IMPLEMENTED;
+}
+
+/****************************************************************************
+ *
+ ****************************************************************************/
+
+/*
+GrepError::Code
+RepState::slowStop()
+{
+ switch(m_channel.getState())
+ {
+ case Channel::LOG:
+ m_channel.setState(Channel::LOG_SLOW_STOP);
+ return GrepError::NO_ERROR;
+ default:
+ return GrepError::REQUESTOR_ILLEGAL_STATE_FOR_SLOWSTOP;
+ }
+}
+
+GrepError::Code
+RepState::fastStop()
+{
+ switch(m_channel.getState())
+ {
+ case Channel::LOG:
+ m_channel.setState(Channel::LOG_FAST_STOP);
+ return GrepError::NO_ERROR;
+ default:
+ return GrepError::REQUESTOR_ILLEGAL_STATE_FOR_FASTSTOP;
+ }
+}
+*/
+
+/****************************************************************************
+ * Print Status
+ ****************************************************************************/
+
+static const char*
+headerText =
+"+-------------------------------------------------------------------------+\n"
+"| MySQL Replication Server |\n"
+"+-------------------------------------------------------------------------+\n"
+;
+
+static const char*
+channelHeaderText =
+"+-------------------------------------------------------------------------+\n"
+"| Applier Channel 1 Replication Status |\n"
+"+-------------------------------------------------------------------------+\n"
+;
+
+static const char*
+line =
+"+-------------------------------------------------------------------------+\n"
+;
+
+
+Properties *
+RepState::getStatus()
+{
+ Properties * prop = new Properties();
+ if(prop == NULL)
+ return NULL;
+ NdbMutex_Lock(m_mutex);
+
+ prop->put("nodegroups", (int)m_channel.getNoOfNodeGroups());
+// prop->put("epoch_state", m_channel.getEpochState());
+ NdbMutex_Unlock(m_mutex);
+ return prop;
+}
+
+
+Properties * RepState::query(QueryCounter counter, Uint32 replicationId)
+{
+ Properties * prop = new Properties();
+ if(prop == NULL)
+ return NULL;
+ NdbMutex_Lock(m_mutex);
+ if(counter != (Uint32)-1)
+ getEpochState((Channel::Position)counter, prop );
+ prop->put("no_of_nodegroups", m_channel.getNoOfNodeGroups());
+ prop->put("subid", m_channel.getNoOfNodeGroups());
+ prop->put("subkey", m_channel.getSubKey());
+ prop->put("connected_db", m_connected);
+ prop->put("connected_rep", m_repConnected);
+ prop->put("state_sub", (int)m_channel.getStateSub());
+ prop->put("state", (int)m_channel.getState());
+
+ NdbMutex_Unlock(m_mutex);
+ return prop;
+
+}
+
+void
+RepState::getEpochState(Channel::Position pos, Properties * p)
+{
+ char first_buf[20];
+ char last_buf[20];
+ int pos_first = 0, pos_last = 0;
+ Uint32 first = 0, last = 0;
+ for(Uint32 i = 0; i < m_channel.getNoOfNodeGroups() ; i++)
+ {
+ m_channel.getEpochState(pos, i, &first, &last);
+ pos_first += sprintf(first_buf+pos_first,"%d%s",first,",");
+ pos_last += sprintf(last_buf+pos_last,"%d%s",last,",");
+ }
+/**
+ * remove trailing comma
+ */
+ pos_first--;
+ pos_last--;
+ sprintf(first_buf+pos_first,"","");
+ sprintf(last_buf + pos_last,"","");
+
+ p->put("first", first_buf);
+ p->put("last", last_buf);
+
+}
+
+
+void
+RepState::printStatus()
+{
+ /***************************************************************************
+ * Global Status
+ ***************************************************************************/
+ ndbout << headerText;
+ switch (m_connected)
+ {
+ case CONNECTED:
+ ndbout << "| Source: Connected "; break;
+ case DISCONNECTED:
+ ndbout << "| Source: Disconnected "; break;
+ case CONNECTABLE:
+ ndbout << "| Source: Disconnected "; break;
+ default:
+ ndbout << "| Source: Unknown "; break;
+ }
+ switch (m_repConnected)
+ {
+ case CONNECTED:
+ ndbout << "(Rep: Connected) "; break;
+ case DISCONNECTED:
+ ndbout << "(Rep: Disconnected) "; break;
+ case CONNECTABLE:
+ ndbout << "(Rep: Disconnected) "; break;
+ default:
+ ndbout << "(Rep: Unknown) "; break;
+ }
+ ndbout << " |" << endl;
+ ndbout << "| Autostart: " << (isAutoStartEnabled() ? "On " : "Off")
+ << " ";
+ ndbout_c(" Silence time: %10u |", m_connected_counter);
+
+ /***************************************************************************
+ * Channel Status
+ ***************************************************************************/
+ ndbout << channelHeaderText;
+ switch(m_channel.getStateSub()) {
+ case Channel::NO_SUBSCRIPTION_EXISTS:
+ ndbout_c("| Subscription: Non-existing "
+ " |");
+ break;
+ case Channel::CREATING_SUBSCRIPTION_ID:
+ ndbout_c("| Subscription: Non-existing (Id is being created)"
+ " |");
+ break;
+ case Channel::SUBSCRIPTION_ID_CREATED:
+ ndbout_c("| Subscription: %-3d-%-6d in state: Not yet started "
+ " |",
+ m_channel.getSubId(), m_channel.getSubKey());
+ break;
+ case Channel::STARTING_SUBSCRIPTION:
+ ndbout_c("| Subscription: %-3d-%-6d in state: Being started "
+ " |",
+ m_channel.getSubId(), m_channel.getSubKey());
+ break;
+ case Channel::SUBSCRIPTION_STARTED:
+ ndbout_c("| Subscription: %-3d-%-6d in state: Started "
+ " |",
+ m_channel.getSubId(), m_channel.getSubKey());
+ break;
+ default:
+ REPABORT("Illegal subscription state");
+ }
+ ndbout << "| Stop epoch: ";
+ if (m_channel.getStopEpochId() == intervalMax) {
+ ndbout << "No stop defined ";
+ } else {
+ ndbout.print("%-10d ",
+ m_channel.getStopEpochId());
+ }
+ ndbout << " |" << endl;
+
+ ndbout << "| State: ";
+ switch(m_channel.getState())
+ {
+ case Channel::CONSISTENT:
+ ndbout << "Local database is subscription consistent ";
+ break;
+ case Channel::METALOG_STARTING:
+ ndbout << "Starting (Phase 1: Metalog starting) ";
+ break;
+ case Channel::METALOG_STARTED:
+ ndbout << "Starting (Phase 2: Metalog started) ";
+ break;
+ case Channel::METASCAN_STARTING:
+ ndbout << "Starting (Phase 3: Metascan starting) ";
+ break;
+ case Channel::METASCAN_COMPLETED:
+ ndbout << "Starting (Phase 4: Metascan completed) ";
+ break;
+ case Channel::DATALOG_STARTING:
+ ndbout << "Starting (Phase 5: Datalog starting) ";
+ break;
+ case Channel::DATALOG_STARTED:
+ ndbout << "Starting (Phase 6: Datalog started) ";
+ break;
+ case Channel::DATASCAN_STARTING:
+ ndbout << "Starting (Phase 7: Datascan completed) ";
+ break;
+ case Channel::DATASCAN_COMPLETED:
+ ndbout << "Starting (Phase 8: Datascan completed) ";
+ break;
+ case Channel::LOG:
+ ndbout << "Logging ";
+ break;
+ case Channel::STOPPING:
+ ndbout << "Stopping (Stopped when all epochs applied) ";
+ break;
+ }
+ ndbout << " |" << endl;
+
+/* @todo
+ ndbout_c("| Syncable: Yes/Scan/No/Unknown (Not implemented)"
+ " |");
+*/
+ ndbout << "| Requestor: " << (isEnabled() ? "On " : "Off")
+ << " (Transfer: " << (isTransferEnabled() ? "On, " : "Off, ")
+ << "Apply: " << (isApplyEnabled() ? "On, " : "Off, ")
+ << "Delete: " << (isDeleteEnabled() ? "On) " : "Off)")
+ << " |" << endl;
+ ndbout_c("| Tables being replicated using this channel: "
+ " |");
+ m_channel.printTables();
+
+ /**
+ * Print node groups
+ */
+ if (getNoOfNodeGroups() == 0)
+ {
+ ndbout_c("| No node groups are known. "
+ " |");
+ }
+ else
+ {
+ m_channel.print();
+ }
+ ndbout << line;
+}
diff --git a/ndb/src/rep/state/RepState.hpp b/ndb/src/rep/state/RepState.hpp
new file mode 100644
index 00000000000..e88151d5609
--- /dev/null
+++ b/ndb/src/rep/state/RepState.hpp
@@ -0,0 +1,275 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef REP_STATE_HPP
+#define REP_STATE_HPP
+
+#include <GrepError.hpp>
+#include <signaldata/GrepImpl.hpp>
+#include <rep/repapi/repapi.h>
+#include <rep/ExtSender.hpp>
+#include <rep/adapters/AppNDB.hpp>
+
+#include "Channel.hpp"
+#include "Interval.hpp"
+
+#define REQUESTOR_EXECUTES_NEEDED_FOR_UNKNOWN_CONNECTION 5
+
+class NdbApiSignal;
+
+
+/**
+ * @class RepState
+ * @brief The main information about the replication
+ */
+class RepState
+{
+public:
+ RepState();
+ ~RepState();
+ void init(ExtSender * extSender) { m_extSender = extSender; }
+
+ /***************************************************************************
+ * Callback functions
+ *
+ * These are used when RepState wants to do something
+ ***************************************************************************/
+
+ typedef void (FuncRequestCreateSubscriptionId)
+ (void * cbObj, NdbApiSignal* signal);
+
+ typedef void (FuncRequestCreateSubscription)
+ (void * cbObj, NdbApiSignal* signal, Uint32 subId,
+ Uint32 subKey ,
+ Vector <struct table *> * selectedTables);
+
+ typedef void (FuncRequestRemoveSubscription)
+ (void * cbObj, NdbApiSignal* signal, Uint32 subId, Uint32 subKey);
+
+ typedef void (FuncRequestTransfer)
+ (void * cbObj, NdbApiSignal* signal,
+ Uint32 nodeGrp, Uint32 first, Uint32 last);
+
+ typedef void (FuncRequestApply)
+ (void * cbObj, NdbApiSignal* signal,
+ Uint32 nodeGrp, Uint32 first, Uint32 last, Uint32 force);
+
+ typedef void (FuncRequestDeleteSS)
+ (void * cbObj, NdbApiSignal* signal,
+ Uint32 nodeGrp, Uint32 first, Uint32 last);
+
+ typedef void (FuncRequestDeletePS)
+ (void * cbObj, NdbApiSignal* signal,
+ Uint32 nodeGrp, Uint32 first, Uint32 last);
+
+ typedef void (FuncRequestStartMetaLog)
+ (void * cbObj, NdbApiSignal * signal, Uint32 subId, Uint32 subKey);
+
+ typedef void (FuncRequestStartDataLog)
+ (void * cbObj, NdbApiSignal * signal, Uint32 subId, Uint32 subKey);
+
+ typedef void (FuncRequestStartMetaScan)
+ (void * cbObj, NdbApiSignal * signal, Uint32 subId, Uint32 subKey);
+
+ typedef void (FuncRequestStartDataScan)
+ (void * cbObj, NdbApiSignal * signal, Uint32 subId, Uint32 subKey);
+
+ typedef void (FuncRequestEpochInfo)
+ (void * cbObj, NdbApiSignal * signal, Uint32 nodeGrp);
+
+ /***************************************************************************
+ *
+ ***************************************************************************/
+ void setSubscriptionRequests(FuncRequestCreateSubscriptionId f1,
+ FuncRequestCreateSubscription f2,
+ FuncRequestRemoveSubscription f3);
+ void setIntervalRequests(FuncRequestTransfer * f1,
+ FuncRequestApply * f2,
+ FuncRequestDeleteSS * f3,
+ FuncRequestDeletePS * f4);
+ void setStartRequests(FuncRequestStartMetaLog * f5,
+ FuncRequestStartDataLog * f6,
+ FuncRequestStartMetaScan * f7,
+ FuncRequestStartDataScan * f8,
+ FuncRequestEpochInfo * f9);
+
+ /***************************************************************************
+ * Enablings
+ ***************************************************************************/
+ bool isEnabled() { return m_channel.m_requestorEnabled; }
+ bool isTransferEnabled() { return m_channel.m_transferEnabled; }
+ bool isApplyEnabled() { return m_channel.m_applyEnabled; }
+ bool isDeleteEnabled() { return m_channel.m_deleteEnabled; }
+ bool isAutoStartEnabled() { return m_channel.m_autoStartEnabled; }
+
+ void enable() { m_channel.m_requestorEnabled = true; }
+ void enableTransfer() { m_channel.m_transferEnabled = true; }
+ void enableApply() { m_channel.m_applyEnabled = true; }
+ void enableDelete() { m_channel.m_deleteEnabled = true; }
+ void enableAutoStart() { m_channel.m_autoStartEnabled = true; }
+
+ void disable() { m_channel.m_requestorEnabled = false; }
+ void disableTransfer() { m_channel.m_transferEnabled = false; }
+ void disableApply() { m_channel.m_applyEnabled = false;}
+ void disableDelete() { m_channel.m_deleteEnabled = false; }
+ void disableAutoStart() { m_channel.m_autoStartEnabled = false; }
+
+ /***************************************************************************
+ * Node groups
+ ***************************************************************************/
+ void setNoOfNodeGroups(Uint32 n) { m_channel.setNoOfNodeGroups(n); }
+ Uint32 getNoOfNodeGroups() { return m_channel.getNoOfNodeGroups(); }
+
+ /***************************************************************************
+ * Event reporting to RepState
+ *
+ * These are used to update the state of the Requestor when something
+ * has happend.
+ ***************************************************************************/
+ void request(GrepReq::Request request);
+
+ //GrepError::Code createSubscription(Uint32 subId, Uint32 subKey);
+ GrepError::Code protectedExecute();
+ GrepError::Code protectedRequest(GrepReq::Request request, Uint32 arg);
+ GrepError::Code protectedRequest(GrepReq::Request request,
+ Uint32 arg1,
+ Uint32 arg2);
+ GrepError::Code protectedAddTable(const char * fullTableName);
+ GrepError::Code protectedRemoveTable(const char * fullTableName);
+ GrepError::Code add(Channel::Position s, Uint32 nodeGrp, const Interval i);
+ GrepError::Code clear(Channel::Position s, Uint32 nodeGrp, const Interval i);
+
+ void eventSubscriptionDeleted(Uint32 subId, Uint32 subKey);
+
+ void eventMetaLogStarted(NdbApiSignal*, Uint32 subId, Uint32 subKey);
+ void eventDataLogStarted(NdbApiSignal*, Uint32 subId, Uint32 subKey);
+ void eventMetaScanCompleted(NdbApiSignal*, Uint32 subId, Uint32 subKey,
+ Interval epochs);
+ void eventDataScanCompleted(NdbApiSignal*, Uint32 subId, Uint32 subKey,
+ Interval epochs);
+ void eventMetaScanFailed(Uint32 subId, Uint32 subKey, GrepError::Code error);
+ void eventDataScanFailed(Uint32 subId, Uint32 subKey, GrepError::Code error);
+
+ /**
+ * @fn sendInsertConf
+ * @param gci - the gci of the applied GCIBuffer.
+ * @param nodeGrp - the nodeGrp of the applied GCIBuffer.
+ */
+ void eventInsertConf(Uint32 gci, Uint32 nodeGrp);
+
+ /**
+ * @fn sendInsertRef
+ * @param gci - the gci of the applied GCIBuffer.
+ * @param nodeGrp - the nodeGrp of the applied GCIBuffer.
+ * @param tableId - the table of the applied GCIBuffer.
+ */
+ void eventInsertRef(Uint32 gci, Uint32 nodeGrp, Uint32 tableId,
+ GrepError::Code err);
+ void eventCreateTableRef(Uint32 gci,
+ Uint32 tableId,
+ const char * tableName,
+ GrepError::Code err) ;
+
+ void eventSubscriptionIdCreated(Uint32 subId, Uint32 subKey);
+ void eventSubscriptionIdCreateFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error);
+
+ void eventSubscriptionCreated(Uint32 subId, Uint32 subKey);
+ void eventSubscriptionCreateFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error);
+
+ void eventMetaLogStartFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error);
+ void eventDataLogStartFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error);
+
+ void eventNodeConnected(Uint32 nodeId);
+ void eventNodeDisconnected(Uint32 nodeId);
+ void eventNodeConnectable(Uint32 nodeId);
+
+ void printStatus();
+ Properties * getStatus();
+ Properties * query(QueryCounter counter, Uint32 replicationId);
+ Uint32 getSubId() { return m_channel.getSubId(); }
+ Uint32 getSubKey () { return m_channel.getSubKey(); }
+
+ void setApplier(class AppNDB * app) { m_applier = app; }
+ void setGCIContainer(class GCIContainer * c) { m_gciContainer = c; }
+
+ /* @todo should be private */
+ Channel m_channel;
+
+private:
+ /***************************************************************************
+ * PRIVATE ATTRIBUTES
+ ***************************************************************************/
+ ExtSender * m_extSender;
+ AppNDB * m_applier;
+ GCIContainer * m_gciContainer;
+
+ Uint32 m_subIdToRemove;
+ Uint32 m_subKeyToRemove;
+
+
+ enum Connected
+ {
+ UNKNOWN, ///<
+ CONNECTED, ///< Recently received info from (all needed) PS REP
+ DISCONNECTED, ///< Received disconnect info from (some needed) PS REP
+ CONNECTABLE ///< Received disconnect info from (some needed) PS REP
+ };
+ Connected m_connected;
+ Connected m_repConnected;
+ Uint32 m_connected_counter;
+
+ NdbMutex * m_mutex;
+
+ /** @todo Should be channel-specific */
+ Uint32 m_stopEpoch;
+
+ /***************************************************************************
+ * PRIVATE METHODS
+ ***************************************************************************/
+ GrepError::Code execute(NdbApiSignal*);
+ GrepError::Code request(GrepReq::Request req,
+ Uint32 arg1,
+ Uint32 arg2,
+ NdbApiSignal*);
+
+ FuncRequestCreateSubscriptionId * m_funcRequestCreateSubscriptionId;
+ FuncRequestCreateSubscription * m_funcRequestCreateSubscription;
+ FuncRequestRemoveSubscription * m_funcRequestRemoveSubscription;
+
+ FuncRequestTransfer * m_funcRequestTransfer;
+ FuncRequestApply * m_funcRequestApply;
+ FuncRequestDeleteSS * m_funcRequestDeleteSS;
+ FuncRequestDeletePS * m_funcRequestDeletePS;
+
+ FuncRequestStartMetaLog * m_funcRequestStartMetaLog;
+ FuncRequestStartDataLog * m_funcRequestStartDataLog;
+ FuncRequestStartMetaScan * m_funcRequestStartMetaScan;
+ FuncRequestStartDataScan * m_funcRequestStartDataScan;
+ FuncRequestEpochInfo * m_funcRequestEpochInfo;
+
+ void requestTransfer(NdbApiSignal * signal);
+ void requestApply(NdbApiSignal * signal);
+ void requestDelete(NdbApiSignal * signal);
+ void requestEpochInfo(NdbApiSignal * signal);
+ void getEpochState(Channel::Position pos, Properties * p);
+ friend void testRepState();
+};
+
+#endif
diff --git a/ndb/src/rep/state/RepStateEvent.cpp b/ndb/src/rep/state/RepStateEvent.cpp
new file mode 100644
index 00000000000..9be304c8bfa
--- /dev/null
+++ b/ndb/src/rep/state/RepStateEvent.cpp
@@ -0,0 +1,284 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "RepState.hpp"
+
+/****************************************************************************
+ * Public Event Handlers : CREATE SUBSCRIPTION ID
+ ****************************************************************************/
+
+void
+RepState::eventSubscriptionIdCreated(Uint32 subId, Uint32 subKey)
+{
+ if (m_channel.getStateSub() == Channel::CREATING_SUBSCRIPTION_ID)
+ {
+ m_channel.setSubId(subId);
+ m_channel.setSubKey(subKey);
+ m_channel.setStateSub(Channel::SUBSCRIPTION_ID_CREATED);
+ }
+ else
+ {
+ REPABORT("Illegal state for create subscription id conf");
+ }
+}
+
+void
+RepState::eventSubscriptionIdCreateFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error)
+{
+ ndbout_c("\nSubscription id creation failed");
+ ndbout_c("Error %d: %s", error, GrepError::getErrorDesc(error));
+ ndbout_c("Subscription Id: %d, Key: %d", subId, subKey);
+}
+
+/****************************************************************************
+ * Public Event Handlers : CREATE SUBSCRIPTION
+ ****************************************************************************/
+
+void
+RepState::eventSubscriptionCreated(Uint32 subId, Uint32 subKey)
+{
+ if (m_channel.getStateSub() == Channel::STARTING_SUBSCRIPTION)
+ {
+ m_channel.setStateSub(Channel::SUBSCRIPTION_STARTED);
+ }
+ else
+ {
+ REPABORT("Illegal state for create subscription conf");
+ }
+}
+
+void
+RepState::eventSubscriptionCreateFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error)
+{
+ ndbout_c("\nSubscription creation failed");
+ ndbout_c("Error %d: %s", error, GrepError::getErrorDesc(error));
+ ndbout_c("Subscription Id: %d, Key: %d", subId, subKey);
+}
+
+
+/****************************************************************************
+ * Public Event Handlers : META LOG
+ ****************************************************************************/
+
+void
+RepState::eventMetaLogStarted(NdbApiSignal* signal,
+ Uint32 subId, Uint32 subKey)
+{
+ if (m_channel.getState() != Channel::METALOG_STARTING)
+ {
+ RLOG(("WARNING! Metalog started in state %d, should be %d",
+ m_channel.getState(), Channel::METALOG_STARTING));
+ }
+
+ if (!isAutoStartEnabled())
+ {
+ m_channel.setState(Channel::METALOG_STARTED);
+ }
+ else
+ {
+ m_channel.setState(Channel::METALOG_STARTED);
+ m_channel.setState(Channel::METASCAN_STARTING);
+ m_funcRequestStartMetaScan(m_extSender, signal, subId, subKey);
+ }
+}
+
+void
+RepState::eventMetaLogStartFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error)
+{
+ ndbout_c("\nMetalog start failed");
+ ndbout_c("Error %d: %s", error, GrepError::getErrorDesc(error));
+ ndbout_c("Subscription Id: %d, Key: %d", subId, subKey);
+}
+
+/****************************************************************************
+ * Public Event Handlers : META SCAN
+ ****************************************************************************/
+
+void
+RepState::eventMetaScanCompleted(NdbApiSignal* signal,
+ Uint32 subId, Uint32 subKey, Interval epochs)
+{
+ if (m_channel.getState() != Channel::METASCAN_STARTING)
+ {
+ RLOG(("WARNING! Metascan completed in state %d, should be %d",
+ m_channel.getState(), Channel::METASCAN_STARTING));
+ }
+ RLOG(("Metascan completed. Subscription %d-%d, Epochs [%d-%d]",
+ subId, subKey, epochs.first(), epochs.last()));
+
+ m_channel.setState(Channel::METASCAN_COMPLETED);
+
+ if (isAutoStartEnabled())
+ {
+ m_channel.setState(Channel::DATALOG_STARTING);
+ m_funcRequestStartDataLog(m_extSender, signal, subId, subKey);
+ }
+ m_channel.setMetaScanEpochs(epochs);
+}
+
+/****************************************************************************
+ * Public Event Handlers : DATA LOG
+ ****************************************************************************/
+
+void
+RepState::eventDataLogStarted(NdbApiSignal* signal,
+ Uint32 subId, Uint32 subKey)
+{
+ if (m_channel.getState() != Channel::DATALOG_STARTING)
+ {
+ RLOG(("WARNING! Datalog started in state %d, should be %d",
+ m_channel.getState(), Channel::DATALOG_STARTING));
+ }
+
+ m_channel.setState(Channel::DATALOG_STARTED);
+
+ if (isAutoStartEnabled())
+ {
+ m_channel.setState(Channel::DATASCAN_STARTING);
+ m_funcRequestStartDataScan(m_extSender, signal, subId, subKey);
+ }
+}
+
+void
+RepState::eventDataLogStartFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error)
+{
+ ndbout_c("\nDatalog start failed");
+ ndbout_c("Error %d: %s", error, GrepError::getErrorDesc(error));
+ ndbout_c("Subscription Id: %d, Key: %d", subId, subKey);
+}
+
+/****************************************************************************
+ * Public Event Handlers : DATA SCAN
+ ****************************************************************************/
+
+void
+RepState::eventDataScanCompleted(NdbApiSignal* signal,
+ Uint32 subId, Uint32 subKey,
+ Interval epochs)
+{
+ if (m_channel.getState() != Channel::DATASCAN_STARTING)
+ {
+ RLOG(("WARNING! Datascan completed in state %d, should be %d",
+ m_channel.getState(), Channel::DATASCAN_STARTING));
+ }
+ RLOG(("Datascan completed. Subscription %d-%d, Epochs [%d-%d]",
+ subId, subKey, epochs.first(), epochs.last()));
+
+ m_channel.setState(Channel::DATASCAN_COMPLETED);
+ m_channel.setDataScanEpochs(epochs);
+}
+
+/****************************************************************************
+ * Public Event Handlers : FAILURES
+ ****************************************************************************/
+
+void
+RepState::eventMetaScanFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error)
+{
+ ndbout_c("\nMetascan failed");
+ ndbout_c("Error %d: %s", error, GrepError::getErrorDesc(error));
+ ndbout_c("Subscription Id: %d, Key: %d", subId, subKey);
+}
+
+void
+RepState::eventDataScanFailed(Uint32 subId, Uint32 subKey,
+ GrepError::Code error)
+{
+ ndbout_c("\nDatascan failed");
+ ndbout_c("Error %d: %s", error, GrepError::getErrorDesc(error));
+ ndbout_c("Subscription Id: %d, Key: %d", subId, subKey);
+}
+
+/****************************************************************************
+ * Public Event Handlers : APPLY
+ ****************************************************************************/
+
+void
+RepState::eventInsertConf(Uint32 gci, Uint32 nodeGrp)
+{
+ Interval app(gci, gci);
+ add(Channel::App, nodeGrp, app);
+ clear(Channel::AppReq, nodeGrp, app);
+
+#ifdef DEBUG_GREP
+ ndbout_c("RepState: GCI Buffer %d:[%d] applied", nodeGrp, gci);
+#endif
+}
+
+void
+RepState::eventInsertRef(Uint32 gci, Uint32 nodeGrp, Uint32 tableId,
+ GrepError::Code err)
+{
+ ndbout_c("\nTable %d, used in replication, did not exist");
+ RLOG(("ERROR %d:%s. Apply failed (%d[%d] in table %d)",
+ err, GrepError::getErrorDesc(err), nodeGrp, gci, tableId));
+}
+
+
+void
+RepState::eventCreateTableRef(Uint32 gci,
+ Uint32 tableId,
+ const char * tableName,
+ GrepError::Code err)
+{
+ ndbout_c("\nFailed to create table %s with source site table id %d",
+ tableName,
+ tableId);
+
+ RLOG(("ERROR %d:%s. Failed to create table %s with source site table id %d!",
+ err, GrepError::getErrorDesc(err), tableName, tableId));
+}
+
+/****************************************************************************
+ * Public Event Handlers : Connected/Disconnected
+ ****************************************************************************/
+
+void
+RepState::eventNodeConnected(Uint32 nodeId)
+{
+ m_repConnected = CONNECTED;
+}
+
+void
+RepState::eventNodeDisconnected(Uint32 nodeId)
+{
+ m_repConnected = DISCONNECTED;
+}
+
+void
+RepState::eventNodeConnectable(Uint32 nodeId)
+{
+ m_repConnected = CONNECTABLE;
+}
+
+/****************************************************************************
+ * Public Event Handlers : Connected/Disconnected
+ ****************************************************************************/
+
+void
+RepState::eventSubscriptionDeleted(Uint32 subId, Uint32 subKey)
+{
+ m_gciContainer->reset();
+ m_channel.setState(Channel::CONSISTENT);
+ m_channel.reset();
+ m_subIdToRemove = 0;
+ m_subKeyToRemove = 0;
+}
diff --git a/ndb/src/rep/state/RepStateRequests.cpp b/ndb/src/rep/state/RepStateRequests.cpp
new file mode 100644
index 00000000000..02677e141f6
--- /dev/null
+++ b/ndb/src/rep/state/RepStateRequests.cpp
@@ -0,0 +1,294 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "RepState.hpp"
+
+#include <NdbApiSignal.hpp>
+#include <SimpleProperties.hpp>
+#include <UtilBuffer.hpp>
+
+#include <signaldata/GrepImpl.hpp>
+#include <signaldata/RepImpl.hpp>
+#include <signaldata/SumaImpl.hpp>
+
+#include <rep/rep_version.hpp>
+#include "Channel.hpp"
+
+/*****************************************************************************
+ * Helper functions
+ *****************************************************************************/
+
+void
+startSubscription(void * cbObj, NdbApiSignal* signal,
+ SubscriptionData::Part part,
+ Uint32 subId, Uint32 subKey)
+{
+ ExtSender * ext = (ExtSender *) cbObj;
+
+ GrepSubStartReq * req = (GrepSubStartReq *)signal->getDataPtrSend();
+ req->subscriptionId = subId;
+ req->subscriptionKey = subKey;
+ req->part = (Uint32) part;
+ signal->set(0, PSREPBLOCKNO, GSN_GREP_SUB_START_REQ,
+ GrepSubStartReq::SignalLength);
+ ext->sendSignal(signal);
+}
+
+void
+scanSubscription(void * cbObj, NdbApiSignal* signal,
+ SubscriptionData::Part part,
+ Uint32 subId, Uint32 subKey)
+{
+ ExtSender * ext = (ExtSender *) cbObj;
+
+ GrepSubSyncReq * req = (GrepSubSyncReq *)signal->getDataPtrSend();
+ req->subscriptionId = subId;
+ req->subscriptionKey = subKey;
+ req->part = part;
+ signal->set(0, PSREPBLOCKNO, GSN_GREP_SUB_SYNC_REQ,
+ GrepSubSyncReq::SignalLength);
+ ext->sendSignal(signal);
+}
+
+/*****************************************************************************
+ * RepState registered functions
+ *
+ * These registered functions are executed by RepState when
+ * RepState needs to have stuff done.
+ *****************************************************************************/
+
+void
+requestCreateSubscriptionId(void * cbObj, NdbApiSignal* signal)
+{
+ ExtSender * ext = (ExtSender *) cbObj;
+
+ CreateSubscriptionIdReq * req =
+ (CreateSubscriptionIdReq *)signal->getDataPtrSend();
+ req->senderData = ext->getOwnRef();
+ signal->set(0, PSREPBLOCKNO, GSN_GREP_CREATE_SUBID_REQ,
+ CreateSubscriptionIdReq::SignalLength);
+ ext->sendSignal(signal);
+
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Sent request for creation of subscription id to PS");
+#endif
+}
+
+void
+requestCreateSubscription(void * cbObj,
+ NdbApiSignal* signal,
+ Uint32 subId,
+ Uint32 subKey,
+ Vector<struct table *> * selectedTables)
+{
+ ExtSender * ext = (ExtSender *) cbObj;
+
+ GrepSubCreateReq * req = (GrepSubCreateReq *)signal->getDataPtrSend();
+ req->senderRef = ext->getOwnRef();
+ req->subscriptionId = subId;
+ req->subscriptionKey = subKey;
+ if(selectedTables!=0) {
+ UtilBuffer m_buffer;
+ UtilBufferWriter w(m_buffer);
+ LinearSectionPtr tablePtr[3];
+ req->subscriptionType = SubCreateReq::SelectiveTableSnapshot;
+
+ for(Uint32 i=0; i< selectedTables->size(); i++) {
+ w.add(SimpleProperties::StringValue, (*selectedTables)[i]->tableName);
+ }
+
+ tablePtr[0].p = (Uint32*)m_buffer.get_data();
+ tablePtr[0].sz = m_buffer.length() >> 2;
+
+ signal->set(0, PSREPBLOCKNO, GSN_GREP_SUB_CREATE_REQ,
+ GrepSubCreateReq::SignalLength);
+ ext->sendFragmentedSignal(signal, tablePtr, 1);
+ }
+ else {
+ req->subscriptionType = SubCreateReq::DatabaseSnapshot;
+ signal->set(0, PSREPBLOCKNO, GSN_GREP_SUB_CREATE_REQ,
+ GrepSubCreateReq::SignalLength);
+ ext->sendFragmentedSignal(signal, 0, 0);
+ }
+
+
+
+#ifdef DEBUG_GREP_SUBSCRIPTION
+ ndbout_c("Requestor: Sent request for creation of subscription");
+#endif
+}
+
+void
+requestRemoveSubscription(void * cbObj, NdbApiSignal* signal,
+ Uint32 subId, Uint32 subKey)
+{
+ ExtSender * ext = (ExtSender *) cbObj;
+
+ GrepSubRemoveReq * req = (GrepSubRemoveReq *)signal->getDataPtrSend();
+ req->subscriptionId = subId;
+ req->subscriptionKey = subKey;
+ signal->set(0, PSREPBLOCKNO, GSN_GREP_SUB_REMOVE_REQ,
+ GrepSubRemoveReq::SignalLength);
+ ext->sendSignal(signal);
+}
+
+void
+requestTransfer(void * cbObj, NdbApiSignal * signal,
+ Uint32 nodeGrp, Uint32 first, Uint32 last)
+{
+ ExtSender * ext = (ExtSender *) cbObj;
+
+ RepGetGciBufferReq * req = (RepGetGciBufferReq*)signal->getDataPtrSend();
+ req->firstGCI = first;
+ req->lastGCI = last;
+ req->nodeGrp = nodeGrp;
+ req->senderRef = ext->getOwnRef();
+ signal->set(0, PSREPBLOCKNO, GSN_REP_GET_GCIBUFFER_REQ,
+ RepGetGciBufferReq::SignalLength);
+ ext->sendSignal(signal);
+
+#ifdef DEBUG_GREP_TRANSFER
+ ndbout_c("Requestor: Requested PS GCI buffers %d:[%d-%d]",
+ nodeGrp, first, last);
+#endif
+}
+
+void
+requestApply(void * applyObj, NdbApiSignal * signal,
+ Uint32 nodeGrp, Uint32 first, Uint32 last, Uint32 force)
+{
+ AppNDB * applier = (AppNDB *) applyObj;
+
+ if (first != last) {
+ RLOG(("WARNING! Trying to apply range [%d-%d]. This is not implemeted",
+ first, last));
+ }
+ /**
+ * Apply GCIBuffer even if it is empty.
+ */
+ applier->applyBuffer(nodeGrp, first, force);
+ /**
+ * @todo Handle return value from the method above
+ */
+}
+
+void
+requestDeleteSS(void * cbObj, NdbApiSignal * signal,
+ Uint32 nodeGrp, Uint32 firstGCI, Uint32 lastGCI)
+{
+ GCIContainer * container = (GCIContainer *) cbObj;
+
+ RLOG(("Deleting SS:%d:[%d-%d]", nodeGrp, firstGCI, lastGCI));
+
+ if(firstGCI < 0 || lastGCI<=0 || nodeGrp < 0) {
+ REPABORT("Illegal interval or wrong node group");
+ //return GrepError::REP_DELETE_NEGATIVE_EPOCH;
+ }
+
+ /*********************************************
+ * All buffers : Modify to the available ones
+ *********************************************/
+ if(firstGCI==0 && lastGCI==(Uint32)0xFFFF) {
+ container->getAvailableGCIBuffers(nodeGrp, &firstGCI, &lastGCI);
+ }
+
+ if(firstGCI == 0) {
+ Uint32 f, l;
+ container->getAvailableGCIBuffers(nodeGrp, &f, &l);
+ RLOG(("Deleting SS:[%d-%d]", f, l));
+ if(f > firstGCI) firstGCI = f;
+ }
+
+ /**
+ * Delete buffers
+ */
+ for(Uint32 i=firstGCI; i<=lastGCI; i++) {
+ if(!container->destroyGCIBuffer(i, nodeGrp)) {
+ RLOG(("WARNING! Delete non-existing epoch SS:%d:[%d]", nodeGrp, i));
+ }
+ //RLOG(("RepStateRequests: Deleting buffer SS:%d:[%d]", nodeGrp, i));
+ }
+}
+
+void
+requestDeletePS(void * cbObj, NdbApiSignal * signal,
+ Uint32 nodeGrp, Uint32 firstGCI, Uint32 lastGCI)
+{
+ ExtSender * ext = (ExtSender *) cbObj;
+
+ RepClearPSGciBufferReq * psReq =
+ (RepClearPSGciBufferReq*)signal->getDataPtrSend();
+ /**
+ * @todo Should have better senderData /Lars
+ */
+ psReq->senderData = 4711;
+ psReq->senderRef = ext->getOwnRef();
+ psReq->firstGCI = firstGCI;
+ psReq->lastGCI = lastGCI;
+ psReq->nodeGrp = nodeGrp;
+ signal->set(0, PSREPBLOCKNO, GSN_REP_CLEAR_PS_GCIBUFFER_REQ,
+ RepClearPSGciBufferReq::SignalLength);
+ ext->sendSignal(signal);
+
+ RLOG(("Requesting deletion of PS:%d:[%d-%d]", nodeGrp, firstGCI, lastGCI));
+}
+
+/**
+ * Function that requests information from REP PS about stored GCI Buffers
+ */
+void
+requestEpochInfo(void * cbObj, NdbApiSignal* signal, Uint32 nodeGrp)
+{
+ ExtSender * ext = (ExtSender *) cbObj;
+
+ RepGetGciReq * req = (RepGetGciReq *) signal->getDataPtrSend();
+ req->nodeGrp = nodeGrp;
+ signal->set(0, PSREPBLOCKNO, GSN_REP_GET_GCI_REQ,
+ RepGetGciReq::SignalLength);
+ ext->sendSignal(signal);
+}
+
+void
+requestStartMetaLog(void * cbObj, NdbApiSignal * signal,
+ Uint32 subId, Uint32 subKey)
+{
+ RLOG(("Metalog starting. Subscription %d-%d", subId, subKey));
+ startSubscription(cbObj, signal, SubscriptionData::MetaData, subId, subKey);
+}
+
+void
+requestStartDataLog(void * cbObj, NdbApiSignal * signal,
+ Uint32 subId, Uint32 subKey)
+{
+ RLOG(("Datalog starting. Subscription %d-%d", subId, subKey));
+ startSubscription(cbObj, signal, SubscriptionData::TableData, subId, subKey);
+}
+
+void
+requestStartMetaScan(void * cbObj, NdbApiSignal* signal,
+ Uint32 subId, Uint32 subKey)
+{
+ RLOG(("Metascan starting. Subscription %d-%d", subId, subKey));
+ scanSubscription(cbObj, signal, SubscriptionData::MetaData, subId, subKey);
+}
+
+void
+requestStartDataScan(void * cbObj, NdbApiSignal* signal,
+ Uint32 subId, Uint32 subKey)
+{
+ RLOG(("Datascan starting. Subscription %d-%d", subId, subKey));
+ scanSubscription(cbObj, signal, SubscriptionData::TableData, subId, subKey);
+}
diff --git a/ndb/src/rep/state/testInterval/Makefile b/ndb/src/rep/state/testInterval/Makefile
new file mode 100644
index 00000000000..fbb0b48c280
--- /dev/null
+++ b/ndb/src/rep/state/testInterval/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE := kernel ndbapitest
+
+BIN_TARGET := testInterval
+BIN_TARGET_ARCHIVES := portlib general
+
+CCFLAGS_LOC += -I..
+
+SOURCES = testInterval.cpp ../Interval.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/rep/state/testInterval/testInterval.cpp b/ndb/src/rep/state/testInterval/testInterval.cpp
new file mode 100644
index 00000000000..463e4adffb7
--- /dev/null
+++ b/ndb/src/rep/state/testInterval/testInterval.cpp
@@ -0,0 +1,127 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "../Interval.hpp"
+
+#define TEST_REQUIRE(X); if (!(X)) { \
+ ndbout_c("Test failed in line %d", __LINE__); testPassed = false; }
+
+
+int
+main () {
+ bool testPassed = true;
+
+ Interval a, b, c;
+
+ /**
+ * isEmpty
+ */
+ TEST_REQUIRE(a.isEmpty());
+
+ a.set(3,1);
+ TEST_REQUIRE(a.isEmpty());
+
+ /**
+ * isEqual
+ */
+ a.set(1,2);
+ TEST_REQUIRE(a.isEqual(1,2));
+
+ a.set(3,1);
+ TEST_REQUIRE(a.isEqual(1,0)); // The result should be normalized
+
+ /**
+ * intervalAdd -- non-disjoint
+ */
+ a.set(1,3);
+ b.set(3,10);
+ TEST_REQUIRE(intervalAdd(a, b, &c));
+ TEST_REQUIRE(c.isEqual(1,10));
+
+ a.set(3,10);
+ b.set(1,3);
+ TEST_REQUIRE(intervalAdd(a, b, &c));
+ TEST_REQUIRE(c.isEqual(1,10));
+
+ /**
+ * intervalAdd -- consequtive
+ */
+ a.set(1,3);
+ b.set(4,10);
+ TEST_REQUIRE(intervalAdd(a, b, &c));
+ TEST_REQUIRE(c.isEqual(1,10));
+
+ a.set(4,10);
+ b.set(1,3);
+ TEST_REQUIRE(intervalAdd(a, b, &c));
+ TEST_REQUIRE(c.isEqual(1,10));
+
+ /**
+ * intervalAdd -- disjoint
+ */
+ a.set(1,3);
+ b.set(5,10);
+ c.set(4711,4711);
+ TEST_REQUIRE(!intervalAdd(a, b, &c)); // This should not work
+ TEST_REQUIRE(c.isEqual(4711,4711));
+
+ a.set(5,10);
+ b.set(1,3);
+ c.set(4711,4711);
+ TEST_REQUIRE(!intervalAdd(a, b, &c)); // This should not work
+ TEST_REQUIRE(c.isEqual(4711,4711));
+
+ /**
+ * intervalLeftMinus -- non-disjoint
+ */
+ a.set(1,3);
+ b.set(5,10);
+ intervalLeftMinus(a, b, &c);
+ TEST_REQUIRE(c.isEmpty());
+
+ a.set(5,10);
+ b.set(1,3);
+ intervalLeftMinus(a, b, &c);
+ TEST_REQUIRE(c.isEqual(5,10));
+
+ /**
+ * intervalLeftMinus -- consequtive
+ */
+ a.set(1,3);
+ b.set(4,10);
+ intervalLeftMinus(a, b, &c);
+ TEST_REQUIRE(c.isEmpty());
+
+ a.set(4,10);
+ b.set(1,3);
+ intervalLeftMinus(a, b, &c);
+ TEST_REQUIRE(c.isEqual(4,10));
+
+ /**
+ * intervalLeftMinus -- disjoint
+ */
+ a.set(1,3);
+ b.set(5,10);
+ intervalLeftMinus(a, b, &c);
+ TEST_REQUIRE(c.isEmpty());
+
+ a.set(5,10);
+ b.set(1,3);
+ intervalLeftMinus(a, b, &c);
+ TEST_REQUIRE(c.isEqual(5,10));
+
+ ndbout << "Test " << (testPassed ? "passed" : "failed") << "." << endl;
+}
diff --git a/ndb/src/rep/state/testRepState/Makefile b/ndb/src/rep/state/testRepState/Makefile
new file mode 100644
index 00000000000..33c6076eff3
--- /dev/null
+++ b/ndb/src/rep/state/testRepState/Makefile
@@ -0,0 +1,15 @@
+include .defs.mk
+
+TYPE := kernel
+
+BIN_TARGET := testRequestor
+BIN_TARGET_ARCHIVES := portlib general
+
+CCFLAGS_LOC += -I..
+
+SOURCES = testRequestor.cpp \
+ ../Requestor.cpp \
+ ../RepState.cpp \
+ ../Interval.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/rep/state/testRepState/testRequestor.cpp b/ndb/src/rep/state/testRepState/testRequestor.cpp
new file mode 100644
index 00000000000..8989f7098b8
--- /dev/null
+++ b/ndb/src/rep/state/testRepState/testRequestor.cpp
@@ -0,0 +1,166 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "testRequestor.hpp"
+
+#define TEST_REQUIRE(X); if (!(X)) { \
+ ndbout_c("Test failed in line %d", __LINE__); testPassed = false; }
+
+
+struct Result {
+ Uint32 nodeGrp;
+ Uint32 first;
+ Uint32 last;
+ Uint32 force;
+};
+Result result;
+
+/** Callbacks ****************************************************************/
+
+void
+f_transfer(void *, Signal* signal, Uint32 nodeGrp, Uint32 first, Uint32 last)
+{
+ result.nodeGrp = nodeGrp;
+ result.first = first;
+ result.last = last;
+ result.force = 0;
+ ndbout_c("Transfer: %d:[%d-%d] ", nodeGrp, first, last);
+}
+
+void
+f_apply(void *, Signal* signal, Uint32 nodeGrp,
+ Uint32 first, Uint32 last, Uint32 force)
+{
+ result.nodeGrp = nodeGrp;
+ result.first = first;
+ result.last = last;
+ result.force = force;
+ ndbout_c("Apply: %d:[%d-%d] (Force:%d)", nodeGrp, first, last, force);
+}
+
+void
+f_deletePS(void *, Signal* signal, Uint32 nodeGrp, Uint32 first, Uint32 last)
+{
+ result.nodeGrp = nodeGrp;
+ result.first = first;
+ result.last = last;
+ result.force = 0;
+ ndbout_c("DeletePS: %d:[%d-%d] ", nodeGrp, first, last);
+}
+
+void
+f_deleteSS(void *, Signal* signal, Uint32 nodeGrp, Uint32 first, Uint32 last)
+{
+ result.nodeGrp = nodeGrp;
+ result.first = first;
+ result.last = last;
+ result.force = 0;
+ ndbout_c("DeleteSS: %d:[%d-%d] ", nodeGrp, first, last);
+}
+
+void
+requestStartMetaLog(void * cbObj, Signal * signal)
+{
+ ndbout_c("StartMetaLog:");
+}
+
+void
+requestStartDataLog(void * cbObj, Signal * signal)
+{
+ ndbout_c("StartDataLog:");
+}
+
+void
+requestStartMetaScan(void * cbObj, Signal* signal)
+{
+ ndbout_c("StartMetaScan:");
+}
+
+void
+requestStartDataScan(void * cbObj, Signal* signal)
+{
+ ndbout_c("StartDataScan:");
+}
+
+
+/** Compare ****************************************************************/
+
+bool compare(Uint32 nodeGrp, Uint32 first, Uint32 last, Uint32 force)
+{
+ return (result.nodeGrp == nodeGrp && result.first == first &&
+ result.last == last && result.force == force);
+}
+
+
+/** Main *******************************************************************/
+
+void
+testRequestor()
+{
+ Signal * signal;
+ bool testPassed = true;
+
+ Requestor requestor;
+ requestor.setObject(0);
+ requestor.setIntervalRequests(&f_transfer,
+ &f_apply,
+ &f_deletePS,
+ &f_deleteSS);
+ requestor.setStartRequests(&requestStartMetaLog,
+ &requestStartDataLog,
+ &requestStartMetaScan,
+ &requestStartDataScan);
+ requestor.setNoOfNodeGroups(1);
+ requestor.enable();
+ requestor.enableTransfer();
+ requestor.enableDelete();
+ requestor.enableApply();
+ requestor.m_state = Requestor::LOG;
+
+ requestor.printStatus();
+
+ /**
+ * First transfer
+ */
+ Interval i(12,13);
+ requestor.add(RepState::PS, 0, i);
+ requestor.execute(signal);
+ TEST_REQUIRE(compare(0, 12, 13, 0));
+
+ requestor.printStatus();
+
+ /**
+ * State transtion test
+ */
+
+ /**
+ * First apply
+ */
+
+ /**
+ * Test end
+ */
+ if (testPassed) {
+ ndbout << "Test passed!" << endl;
+ } else {
+ ndbout << "Test FAILED!" << endl;
+ }
+}
+
+int
+main () {
+ testRequestor();
+}
diff --git a/ndb/src/rep/state/testRepState/testRequestor.hpp b/ndb/src/rep/state/testRepState/testRequestor.hpp
new file mode 100644
index 00000000000..726b289114d
--- /dev/null
+++ b/ndb/src/rep/state/testRepState/testRequestor.hpp
@@ -0,0 +1,24 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TEST_REQUESTOR_HPP
+#define TEST_REQUESTOR_HPP
+
+#include "../Requestor.hpp"
+
+void testRequestor();
+
+#endif
diff --git a/ndb/src/rep/storage/GCIBuffer.cpp b/ndb/src/rep/storage/GCIBuffer.cpp
new file mode 100644
index 00000000000..5049e47ea66
--- /dev/null
+++ b/ndb/src/rep/storage/GCIBuffer.cpp
@@ -0,0 +1,174 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "GCIBuffer.hpp"
+#include <stdlib.h>
+#include <assert.h>
+
+/*****************************************************************************
+ * Constructor / Destructor
+ *****************************************************************************/
+
+GCIBuffer::GCIBuffer(Uint32 gci, Uint32 id)
+{
+ m_gci = gci;
+ m_id = id;
+ m_complete = false;
+ m_receivedBytes = 0;
+}
+
+GCIBuffer::~GCIBuffer()
+{
+ /**
+ * Loop through all pages and delete them
+ */
+ for(Uint32 i=0; i<m_pageList.size(); i++) {
+ delete m_pageList[i];
+ m_pageList[i] = 0;
+ }
+ m_pageList.clear();
+ // m_pageList = 0;
+}
+
+/*****************************************************************************
+ * Inserts
+ *****************************************************************************/
+
+void
+GCIBuffer::insertLogRecord(Uint32 tableId, Uint32 operation,
+ class LinearSectionPtr ptr[3])
+{
+ GCIPage * p;
+ if(m_pageList.size() == 0) {
+ p = new GCIPage(m_gci);
+ assert(p != NULL);
+ m_pageList.push_back(p);
+ }
+
+ p = m_pageList.back();
+ if (!p->insertLogRecord(tableId, operation, ptr)) {
+ /**
+ * GCIPage is full.
+ */
+ GCIPage * newPage = new GCIPage(m_gci);
+ assert(newPage != NULL);
+ m_pageList.push_back(newPage);
+ bool res = newPage->insertLogRecord(tableId, operation, ptr);
+
+ if(!res) {
+ ndbout << "GCIBuffer: gci : " << m_gci << endl;
+ assert(res);
+ }
+ }
+}
+
+/**
+ * @todo: We must be able to distinguish between Scan meta
+ * data and log meta data.
+ * Currently only scan meta data is considered.
+ */
+void
+GCIBuffer::insertMetaRecord(Uint32 tableId, class LinearSectionPtr ptr[3])
+{
+ GCIPage * p;
+ if(m_pageList.size()==0) {
+ p = new GCIPage(m_gci);
+ assert(p != NULL);
+ m_pageList.push_back(p);
+ }
+
+ p = m_pageList.back();
+
+ if (!p->insertMetaRecord(tableId, ptr)) {
+ /**
+ * Page is full.
+ */
+ GCIPage * newPage = new GCIPage(m_gci);
+ assert(newPage != NULL);
+ m_pageList.push_back(newPage);
+
+ bool res = newPage->insertMetaRecord(tableId, ptr);
+ assert(res);
+ }
+}
+
+void
+GCIBuffer::insertPage(Uint32 gci, char * dataPtr, Uint32 dataBLen)
+{
+ /**
+ * allocate a new GCIPage
+ */
+ GCIPage * page = new GCIPage(gci);
+ assert(page != 0);
+
+ /**
+ * copy data into page
+ */
+ page->copyDataToPage(dataPtr, dataBLen);
+
+ /**
+ * put page on pagelist.
+ */
+ m_pageList.push_back(page);
+
+ /**
+ * Update GCI Buffer received bytes
+ */
+ m_receivedBytes += dataBLen;
+}
+
+
+/*****************************************************************************
+ * Iterator
+ *****************************************************************************/
+
+GCIBuffer::iterator::iterator(const GCIBuffer* gciBuffer)
+{
+ m_gciBuffer = gciBuffer;
+ m_iterator=0;
+
+}
+
+GCIPage *
+GCIBuffer::iterator::first()
+{
+ m_iterator = 0;
+ if(m_gciBuffer->m_pageList.size() == 0) return NULL;
+ return (m_gciBuffer->m_pageList)[m_iterator];
+}
+
+
+GCIPage *
+GCIBuffer::iterator::next()
+{
+ m_iterator++;
+ if(m_gciBuffer->m_pageList.size() == 0) return NULL;
+
+ if((m_iterator<m_gciBuffer->m_pageList.size()))
+ return (m_gciBuffer->m_pageList)[m_iterator];
+ else
+ return NULL;
+}
+
+
+bool
+GCIBuffer::iterator::exists()
+{
+ return (m_iterator < m_gciBuffer->m_pageList.size());
+}
+
+
+
diff --git a/ndb/src/rep/storage/GCIBuffer.hpp b/ndb/src/rep/storage/GCIBuffer.hpp
new file mode 100644
index 00000000000..5a07b149f81
--- /dev/null
+++ b/ndb/src/rep/storage/GCIBuffer.hpp
@@ -0,0 +1,112 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GCI_BUFFER_HPP
+#define GCI_BUFFER_HPP
+
+#include "GCIPage.hpp"
+#include "Vector.hpp"
+#include <TransporterDefinitions.hpp>
+
+#include <signaldata/RepImpl.hpp>
+
+/**
+ * @class GCIBuffer
+ * @brief A GCIBuffer contains pages containing log records for ONE gci.
+ *
+ * @todo Load and save to disk
+ */
+
+class GCIBuffer
+{
+public:
+ GCIBuffer(Uint32 gci, Uint32 id);
+ ~GCIBuffer();
+
+ /**
+ * @fn insertLogRecord
+ * @param tableId Table this will be LogRecord applies to.
+ * @param operation Operation this LogRecord represents
+ * @param ptr Ptr of type LinearSectionPtr that contains the data.
+ * @return A full page or 0, if the insert didn't generate a full page.
+ */
+ void insertLogRecord(Uint32 tableId, Uint32 operation,
+ class LinearSectionPtr ptr[3]);
+
+ void insertMetaRecord(Uint32 tableId, class LinearSectionPtr ptr[3]);
+
+ /**
+ * @fn inserts a page, containing Records into a GCI Buffer.
+ * @param gci - the gci of the page.
+ * @param dataPtr - Pointer originating from Page::m_page.
+ * @param dataBLen - length of dataptr in bytes
+ * @note Page must NOT be deallocated after being inserted!
+ */
+ void insertPage(Uint32 gci, char * dataPtr, Uint32 dataBLen);
+
+ /**
+ * @fn isComplete
+ * @return True if this GCI Buffer is done (gci is completed).
+ */
+ bool isComplete() { return m_complete; };
+ void setComplete() { m_complete = true; };
+
+ /**
+ * @fn getReceivedBytes
+ * @returns the total number of bytes that this buffer has received.
+ */
+ Uint32 getReceivedBytes() const { return m_receivedBytes;} ;
+
+ /**
+ * Iterator for pages
+ */
+ class iterator {
+ public:
+ iterator(const GCIBuffer* gciBuffer);
+ GCIPage * first(); ///< @return First page (or NULL if no page exists)
+ GCIPage * next(); ///< @return Next page (or NULL if no more page exists)
+ bool exists(); ///< @return true if another page exists (for next())
+ private:
+ Uint32 m_iterator;
+ const GCIBuffer * m_gciBuffer;
+ };
+ friend class GCIBuffer::iterator;
+
+ /***************************************************************************
+ * GCI Buffer meta information
+ ***************************************************************************/
+ void setGCI(Uint32 gci) { m_gci = gci; };
+ Uint32 getGCI() { return m_gci; };
+
+ void setId(Uint32 id) { m_id = id; };
+ Uint32 getId() { return m_id; };
+
+ bool m_force; // if true, ignore "execute" errors when
+ // restoring buffer (PUBLIC) during phase
+ // starting.
+private:
+ /***************************************************************************
+ * Private Variables
+ ***************************************************************************/
+ Uint32 m_gci; ///< GCI of this buffer
+ Uint32 m_id; ///< <m_gci, id> names GCIBuffer
+ bool m_complete; ///< GCI complete; buffer contains
+ ///< everything
+ Vector <GCIPage *> m_pageList; ///< Storage for data/log record pages.
+ Uint32 m_receivedBytes; ///< Received bytes in this buffer
+};
+
+#endif
diff --git a/ndb/src/rep/storage/GCIContainer.cpp b/ndb/src/rep/storage/GCIContainer.cpp
new file mode 100644
index 00000000000..c161db0769b
--- /dev/null
+++ b/ndb/src/rep/storage/GCIContainer.cpp
@@ -0,0 +1,272 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "GCIContainer.hpp"
+#include <NdbOut.hpp>
+#include <NdbMem.h>
+#include <new>
+
+#include <rep/rep_version.hpp>
+
+//#define GCICONTAINER_DEBUG
+
+/*****************************************************************************
+ * Constructors / Destructors
+ *****************************************************************************/
+
+GCIContainer::GCIContainer(Uint32 maxNoOfIds)
+{
+ m_maxNoOfIds = maxNoOfIds;
+
+ gciRange = new GCIRange[maxNoOfIds * sizeof(GCIRange)];
+
+ for(Uint32 i = 0; i < maxNoOfIds; i++) {
+ gciRange[i].m_firstGCI = 1; // The empty interval = [1,0]
+ gciRange[i].m_lastGCI = 0;
+ }
+ theMutexPtr = NdbMutex_Create();
+}
+
+GCIContainer::~GCIContainer()
+{
+ for(Uint32 i=0; i < m_bufferList.size(); i++) {
+ delete m_bufferList[i];
+ m_bufferList[i] = 0;
+ }
+
+ m_bufferList=0;
+ delete [] gciRange;
+ NdbMutex_Destroy(theMutexPtr);
+}
+
+/*****************************************************************************
+ * GCIBuffer Create / Destroy
+ *****************************************************************************/
+
+GCIBuffer *
+GCIContainer::createGCIBuffer(Uint32 gci, Uint32 id)
+{
+ GCIBuffer * buf = new GCIBuffer(gci, id);
+ if (buf == NULL) REPABORT("Could not allocate new buffer");
+
+ m_bufferList.push_back(buf, true);
+
+#ifdef GCICONTAINER_DEBUG
+ ndbout_c("GCIContainer: New buffer created (GCI: %d, Id: %d)", gci, id);
+#endif
+ return buf;
+}
+
+/**
+ * Delete all GCI buffers strictly less than "gci"
+ */
+void
+GCIContainer::destroyGCIBuffersBeforeGCI(Uint32 gci, Uint32 id)
+{
+ for(Uint32 i = 0 ; i < m_bufferList.size(); i++) {
+ if(m_bufferList[i]->getGCI() < gci) {
+#ifdef GCICONTAINER_DEBUG
+ ndbout_c("GCIContainer: Destroying buffer (GCI: %d, id: %d)",
+ m_bufferList[i]->getGCI(), id);
+#endif
+ destroyGCIBuffer(i, id);
+ }
+ }
+}
+
+/**
+ * Delete one GCI Buffer
+ */
+bool
+GCIContainer::destroyGCIBuffer(Uint32 gci, Uint32 id)
+{
+ m_bufferList.lock();
+ for(Uint32 i = 0 ; i < m_bufferList.size(); i++) {
+ if((m_bufferList[i]->getGCI() == gci) &&
+ (m_bufferList[i]->getId() == id)) {
+
+ /**
+ * Delete the GCI Buffer
+ */
+ delete m_bufferList[i];
+ m_bufferList[i] = 0;
+
+ /**
+ * Remove from the list of buffers stored in GCIContainer
+ */
+ m_bufferList.erase(i,false);
+ m_bufferList.unlock();
+
+ /**
+ * Set info
+ */
+ NdbMutex_Lock(theMutexPtr);
+ if(gciRange[id].m_firstGCI != gci)
+ RLOG(("WARNING! Buffer %d deleted from [%d-%d]",
+ gci, gciRange[id].m_firstGCI, gciRange[id].m_lastGCI));
+
+ gciRange[id].m_firstGCI++;
+
+ /**
+ * Normalize empty interval to [1,0]
+ */
+ if (gciRange[id].m_firstGCI > gciRange[id].m_lastGCI){
+ gciRange[id].m_firstGCI = 1;
+ gciRange[id].m_lastGCI = 0;
+ }
+ NdbMutex_Unlock(theMutexPtr);
+ return true;
+ }
+ }
+ m_bufferList.unlock();
+ return false;
+}
+
+/*****************************************************************************
+ * GCIBuffer interface
+ *****************************************************************************/
+
+GCIBuffer *
+GCIContainer::getGCIBuffer(Uint32 gci, Uint32 id)
+{
+ GCIBuffer * gciBuffer = 0;
+
+ m_bufferList.lock();
+ for(Uint32 i=0; i < m_bufferList.size(); i++) {
+ gciBuffer = m_bufferList[i];
+ if((gciBuffer->getGCI() == gci) && (gciBuffer->getId() == id)) {
+ m_bufferList.unlock();
+ return gciBuffer;
+ }
+ }
+ m_bufferList.unlock();
+ return 0;
+}
+
+void
+GCIContainer::setCompleted(Uint32 gci, Uint32 id)
+{
+ GCIBuffer * gciBuffer = getGCIBuffer(gci, id);
+ if(gciBuffer == 0) gciBuffer = createGCIBuffer(gci, id);
+
+ gciBuffer->setComplete();
+
+#ifdef GCICONTAINER_DEBUG
+ ndbout_c("GCIContainer: Buffer completely stored in GCIContainer (GCI: %d)",
+ gci);
+#endif
+
+ NdbMutex_Lock(theMutexPtr);
+
+ /**
+ * If this is the first GCI Buffer to be completed
+ * then both first and last must be updated.
+ * Subsequently, only the last value must be updated.
+ */
+ if(gciRange[id].m_firstGCI == 1 && gciRange[id].m_lastGCI == 0) {
+ gciRange[id].m_firstGCI = gci;
+ gciRange[id].m_lastGCI = gci;
+ } else {
+ if (gci != gciRange[id].m_lastGCI + 1) {
+ RLOG(("WARNING! Non-consequtive buffer %u completed [%u-%u])",
+ gci, gciRange[id].m_firstGCI, gciRange[id].m_lastGCI));
+ }
+ gciRange[id].m_lastGCI = gci;
+ }
+ NdbMutex_Unlock(theMutexPtr);
+}
+
+void
+GCIContainer::getAvailableGCIBuffers(Uint32 id, Uint32 * first, Uint32 * last)
+{
+ NdbMutex_Lock(theMutexPtr);
+ *first = gciRange[id].m_firstGCI;
+ *last = gciRange[id].m_lastGCI;
+ NdbMutex_Unlock(theMutexPtr);
+}
+
+/*****************************************************************************
+ * Inserts
+ *****************************************************************************/
+void
+GCIContainer::insertMetaRecord(Uint32 id, Uint32 tableId,
+ class LinearSectionPtr ptr[3], Uint32 gci)
+{
+ /**********************************************************
+ * 1. Find correct GCI Buffer (Doesn't exist? Create one)
+ **********************************************************/
+ GCIBuffer * gciBuffer = getGCIBuffer(gci, id);
+ if(gciBuffer == 0) gciBuffer = createGCIBuffer(gci, id);
+
+ /**********************************
+ * 2. Insert record into GCIBuffer
+ **********************************/
+ gciBuffer->insertMetaRecord(tableId, ptr);
+}
+
+void
+GCIContainer::insertLogRecord(Uint32 id, Uint32 tableId, Uint32 operation,
+ class LinearSectionPtr ptr[3], Uint32 gci)
+{
+ /*********************************************************
+ * 1. Find correct GCI Buffer (doesn't exist? create one)
+ *********************************************************/
+ GCIBuffer * gciBuffer = getGCIBuffer(gci, id);
+ if(gciBuffer == 0) gciBuffer = createGCIBuffer(gci, id);
+ /**********************************
+ * 2. Insert record into GCIBuffer
+ **********************************/
+ gciBuffer->insertLogRecord(tableId, operation, ptr);
+}
+
+void
+GCIContainer::insertPage(Uint32 gci, Uint32 id,
+ char * dataPtr, Uint32 dataBLen)
+{
+ /*********************************************************
+ * 1. Find correct GCI Buffer (doesn't exist? create one)
+ *********************************************************/
+ GCIBuffer * gciBuffer = getGCIBuffer(gci, id);
+ if(gciBuffer == 0) gciBuffer = createGCIBuffer(gci, id);
+
+ /********************************
+ * 2. Insert page into GCIBuffer
+ ********************************/
+ gciBuffer->insertPage(gci, dataPtr, dataBLen);
+}
+
+bool
+GCIContainer::reset()
+{
+ /**
+ * Clear the intervals
+ */
+ for(Uint32 i = 0; i < m_maxNoOfIds; i++) {
+ gciRange[i].m_firstGCI = 1; // The empty interval = [1,0]
+ gciRange[i].m_lastGCI = 0;
+ }
+
+ /**
+ * Destroy ALL gci buffers for ALL ids
+ */
+ for(Uint32 i=0; i < m_bufferList.size(); i++) {
+ delete m_bufferList[i];
+ m_bufferList[i] = 0;
+ }
+ m_bufferList.clear();
+
+ return true;
+}
diff --git a/ndb/src/rep/storage/GCIContainer.hpp b/ndb/src/rep/storage/GCIContainer.hpp
new file mode 100644
index 00000000000..173bb790a57
--- /dev/null
+++ b/ndb/src/rep/storage/GCIContainer.hpp
@@ -0,0 +1,120 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GCI_CONTAINER_HPP
+#define GCI_CONTAINER_HPP
+
+#include <Vector.hpp>
+
+#include "LogRecord.hpp"
+#include "GCIBuffer.hpp"
+
+#include <list>
+#include <iterator>
+
+/**
+ * @class GCIContainer
+ * @brief Responsible for storing LogRecord:s in GCIBuffer:s
+ *
+ * Each GCIBuffer stored in the GCIContainer is named by a pair <GCI, id>.
+ * (On PS REP the id is the nodeId, on SS REP the id is the node group).
+ */
+class GCIContainer {
+public:
+ GCIContainer(Uint32 maxNoOfIds);
+ ~GCIContainer();
+
+ /***************************************************************************
+ * GCIBuffer interface
+ ***************************************************************************/
+ /**
+ * @return GCIBuffer if success, NULL otherwise
+ */
+ GCIBuffer * createGCIBuffer(Uint32 gci, Uint32 id);
+
+ /**
+ * Destroy all buffers with GCI strictly less than gci.
+ */
+ void destroyGCIBuffersBeforeGCI(Uint32 gci, Uint32 id);
+
+ /**
+ * Destroy all buffers with GCI gci.
+ * @return true if buffer was deleted, false if no buffer exists
+ */
+ bool destroyGCIBuffer(Uint32 gci, Uint32 id);
+
+ /**
+ * Fetch buffer
+ * @return GCIBuffer for gci, or NULL if no buffer found
+ */
+ GCIBuffer * getGCIBuffer(Uint32 gci, Uint32 id);
+
+ /**
+ * Set that buffer is completed, i.e. no more records are to be inserted
+ */
+ void setCompleted(Uint32 gci, Uint32 id);
+
+
+ /**
+ * @fn insertPage
+ * @param gci GCI this page belongs to.
+ * @param id Id this page belongs to.
+ * @param dataPtr Pointer originating from Page::m_page
+ * @param dataBLen Length in bytes of data following dataptr.
+ */
+ void insertPage(Uint32 gci, Uint32 id, char * dataPtr, Uint32 dataBLen);
+
+
+ /***************************************************************************
+ * Record interface
+ ***************************************************************************/
+ void insertLogRecord(Uint32 id, Uint32 tableId, Uint32 operation,
+ class LinearSectionPtr ptr[3], Uint32 gci);
+
+ void insertMetaRecord(Uint32 id, Uint32 tableId,
+ class LinearSectionPtr ptr[3], Uint32 gci);
+
+ /**
+ * Get available (complete) GCI Buffers that exists in the container.
+ * first == last means that there is one complete buffer
+ * @param id Id for which to as for available gci buffers.
+ * @param first First complete gci buffer
+ * @param last Last complete gci buffer
+ */
+ void getAvailableGCIBuffers(Uint32 id, Uint32 * first, Uint32 * last);
+
+ /**
+ * Resets the gcicontainer to its original state (initial state and empty)
+ * I.e., same state as when the object was first constructed.
+ * @return true if reset was ok
+ */
+ bool reset();
+
+private:
+ NdbMutex* theMutexPtr;
+ MutexVector <GCIBuffer *> m_bufferList; ///< All GCIBuffers stored
+
+ typedef struct GCIRange {
+ Uint32 m_firstGCI;
+ Uint32 m_lastGCI;
+ };
+
+ Uint32 m_maxNoOfIds;
+
+ GCIRange * gciRange; ///< Array of GCI ranges for each id
+};
+
+#endif
diff --git a/ndb/src/rep/storage/GCIContainerPS.cpp b/ndb/src/rep/storage/GCIContainerPS.cpp
new file mode 100644
index 00000000000..5adb53f965c
--- /dev/null
+++ b/ndb/src/rep/storage/GCIContainerPS.cpp
@@ -0,0 +1,128 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "GCIContainerPS.hpp"
+#include <NdbOut.hpp>
+#include <NdbMem.h>
+#include <new>
+
+GCIContainerPS::GCIContainerPS(Uint32 maxNoOfNodeGrps)
+{
+ m_container = new GCIContainer(maxNoOfNodeGrps);
+ if (!m_container) REPABORT("Could not allocate new GCIContainer");
+}
+
+GCIContainerPS::~GCIContainerPS()
+{
+ delete m_container;
+}
+
+void
+GCIContainerPS::setNodeGroupInfo(NodeGroupInfo * info)
+{
+ m_nodeGroupInfo=info;
+};
+
+void
+GCIContainerPS::createGCIBuffer(Uint32 gci, Uint32 id)
+{
+ m_container->createGCIBuffer(gci, id);
+}
+
+void
+GCIContainerPS::getAvailableGCIBuffers(Uint32 id /*nodegrp */,
+ Uint32 * first, Uint32 * last) {
+
+ Uint32 nodeId = m_nodeGroupInfo->getPrimaryNode(id);
+ if(!nodeId) {
+ *first = 1;
+ *last = 0;
+ return;
+ }
+
+ /**
+ *@todo do smart stuff with this!
+ */
+ m_container->getAvailableGCIBuffers(nodeId, first, last);
+
+}
+
+void
+GCIContainerPS::destroyGCIBuffersBeforeGCI(Uint32 gci)
+{
+ //for each node in every nodeGrp do:
+ NodeGroupInfo::iterator * it;
+ for(Uint32 i=0; i<m_nodeGroupInfo->getNoOfNodeGroups(); i++) {
+ it = new NodeGroupInfo::iterator(i, m_nodeGroupInfo);
+ for(NodeConnectInfo * nci=it->first(); it->exists();nci=it->next()) {
+ m_container->destroyGCIBuffersBeforeGCI(gci, nci->nodeId);
+ }
+ delete it;
+ }
+}
+
+void
+GCIContainerPS::insertLogRecord(Uint32 id, Uint32 tableId, Uint32 operation,
+ class LinearSectionPtr ptr[3], Uint32 gci)
+{
+ m_container->insertLogRecord(id, tableId, operation, ptr, gci);
+}
+
+void
+GCIContainerPS::insertMetaRecord(Uint32 id, Uint32 tableId,
+ class LinearSectionPtr ptr[3], Uint32 gci)
+{
+ m_container->insertMetaRecord(id, tableId, ptr, gci);
+}
+
+void
+GCIContainerPS::setCompleted(Uint32 gci, Uint32 id)
+{
+ m_container->setCompleted(gci, id);
+}
+
+GCIBuffer *
+GCIContainerPS::getGCIBuffer(Uint32 gci, Uint32 id)
+{
+ return m_container->getGCIBuffer(gci, id);
+}
+
+/**
+ * @todo: fix return value
+ */
+bool
+GCIContainerPS::destroyGCIBuffer(Uint32 gci, Uint32 id)
+{
+ //for each node in nodeGrp id do:
+ NodeGroupInfo::iterator * it;
+ it = new NodeGroupInfo::iterator(id, m_nodeGroupInfo);
+ for(NodeConnectInfo * nci=it->first(); it->exists();nci=it->next())
+ {
+ if(!m_container->destroyGCIBuffer(gci, nci->nodeId))
+ {
+ delete it;
+ return false;
+ }
+ }
+ delete it;
+ return true;
+}
+
+bool
+GCIContainerPS::reset()
+{
+ return m_container->reset();
+}
diff --git a/ndb/src/rep/storage/GCIContainerPS.hpp b/ndb/src/rep/storage/GCIContainerPS.hpp
new file mode 100644
index 00000000000..7f5aaac4840
--- /dev/null
+++ b/ndb/src/rep/storage/GCIContainerPS.hpp
@@ -0,0 +1,90 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GCI_CONTAINER_PS_HPP
+#define GCI_CONTAINER_PS_HPP
+
+#include <Vector.hpp>
+#include <TransporterDefinitions.hpp>
+
+#include "NodeGroupInfo.hpp"
+#include <rep/storage/GCIContainer.hpp>
+
+#include <list>
+#include <iterator>
+
+/**
+ * @class GCIContainerPS
+ * @brief Interface to GCIContainer that takes node groups into account
+ */
+class GCIContainerPS
+{
+public:
+ GCIContainerPS(Uint32 maxNoOfNodeGrps);
+ ~GCIContainerPS();
+
+ void setNodeGroupInfo(NodeGroupInfo * info);
+ NodeGroupInfo * getNodeGroupInfo() {return m_nodeGroupInfo;};
+
+ void createGCIBuffer(Uint32 gci, Uint32 id);
+ void getAvailableGCIBuffers(Uint32 id /*nodegrp */,
+ Uint32 * first, Uint32 * last);
+
+ /***************************************************************************
+ * Record interface
+ ***************************************************************************/
+ void insertLogRecord(Uint32 grpId, Uint32 tableId, Uint32 operation,
+ class LinearSectionPtr ptr[3], Uint32 gci);
+
+ void insertMetaRecord(Uint32 grpId, Uint32 tableId,
+ class LinearSectionPtr ptr[3], Uint32 gci);
+
+ /**
+ * Destroy all buffers with GCI strictly less than gci.
+ */
+ void destroyGCIBuffersBeforeGCI(Uint32 gci);
+
+ /**
+ * Set that buffer is completed, i.e. no more records are to be inserted
+ */
+ void setCompleted(Uint32 gci, Uint32 id);
+
+ /**
+ * Fetch buffer
+ * @return GCIBuffer for gci, or NULL if no buffer found
+ */
+ GCIBuffer * getGCIBuffer(Uint32 gci, Uint32 id);
+
+ /**
+ * Destroy all buffers with GCI gci.
+ * @return true if buffer was deleted, false if no buffer exists
+ */
+ bool destroyGCIBuffer(Uint32 gci, Uint32 id);
+
+
+ /**
+ * Resets the gcicontainer to its original state (initial state and empty)
+ * @return true if reset was ok
+ */
+ bool reset();
+
+private:
+ GCIContainer * m_container;
+ NodeGroupInfo * m_nodeGroupInfo;
+};
+
+
+#endif
diff --git a/ndb/src/rep/storage/GCIPage.cpp b/ndb/src/rep/storage/GCIPage.cpp
new file mode 100644
index 00000000000..05ecde2fee1
--- /dev/null
+++ b/ndb/src/rep/storage/GCIPage.cpp
@@ -0,0 +1,165 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "GCIPage.hpp"
+#include "assert.h"
+#include <new>
+
+GCIPage::GCIPage(Uint32 gci)
+{
+ m_first = NULL;
+ m_last = NULL;
+ m_gci = gci;
+ m_full = false;
+ m_currentPagePos=m_page;
+ m_usedBytes = 0;
+}
+
+/*****************************************************************************
+ * Insert
+ *****************************************************************************/
+
+/**
+ * Store a new log record on this page.
+ * @return True if success, false otherwise
+ */
+bool
+GCIPage::insertLogRecord(Uint32 tableId, Uint32 operation,
+ class LinearSectionPtr ptr[3])
+{
+ /**
+ * Calculate size of new logrecord in bytes
+ */
+ assert(m_page!=NULL);
+ Uint32 size = 4*ptr[0].sz + 4*ptr[1].sz + sizeof(LogRecord);
+
+ if(!((m_currentPagePos + size ) < (m_page + m_pageBSize))) {
+ m_full = true;
+ return false; // No free space. GCIBuffer must allocate a new page
+ }
+ LogRecord * lr = new(m_currentPagePos) LogRecord();
+ if (lr==0) REPABORT("Could not allocate new log record");
+
+ lr->recordType = Record::LOG;
+ lr->recordLen = size;
+ lr->operation = operation;
+ lr->tableId = tableId;
+ lr->attributeHeaderWSize = ptr[0].sz;
+ lr->attributeDataWSize = ptr[1].sz;
+
+ m_currentPagePos += sizeof(LogRecord);
+
+ lr->attributeHeader = (Uint32*)m_currentPagePos;
+ memcpy(lr->attributeHeader, ptr[0].p, lr->attributeHeaderWSize * 4);
+
+ m_currentPagePos += lr->attributeHeaderWSize * 4;
+
+ lr->attributeData = (Uint32*)m_currentPagePos;
+ memcpy(lr->attributeData, ptr[1].p, lr->attributeDataWSize * 4);
+
+ m_currentPagePos += lr->attributeDataWSize * 4;
+
+ m_usedBytes+=size;
+ return true;
+}
+
+/**
+ * Store a new log record on this page.
+ * @return True if sucessful, false otherwise.
+ */
+bool
+GCIPage::insertMetaRecord(Uint32 tableId, class LinearSectionPtr ptr[3])
+{
+ /**
+ * Calculate size of new logrecord in bytes
+ */
+ Uint32 size = 4*ptr[0].sz + sizeof(MetaRecord);
+
+ if(!((m_currentPagePos + size ) < (m_page + m_pageBSize))) {
+ m_full = true;
+ return false; // No free space. GCIBuffer must allocate a new page
+ }
+ MetaRecord * mr = new(m_currentPagePos) MetaRecord();
+ if (mr==0) REPABORT("Could not allocate new meta record");
+
+ // mr->operation = operation;
+ mr->recordType = Record::META;
+ mr->recordLen = size;
+
+ mr->tableId = tableId;
+ mr->dataLen = ptr[0].sz;
+
+
+ m_currentPagePos += sizeof(MetaRecord);
+
+ mr->data = (Uint32*)m_currentPagePos;
+ memcpy(mr->data, ptr[0].p, mr->dataLen * 4);
+
+ m_currentPagePos += mr->dataLen * 4;
+
+ m_usedBytes+=size;
+ return true;
+}
+
+/**
+ * copy function
+ */
+void
+GCIPage::copyDataToPage(char * dataPtr, Uint32 dataBLen)
+{
+ assert (dataBLen < m_pageBSize);
+ memcpy(m_page, dataPtr, dataBLen);
+ m_currentPagePos=m_page + dataBLen;
+ m_usedBytes = dataBLen;
+ m_full = true;
+ m_first = (Record * )m_page;
+ dataPtr = 0;
+}
+
+/*****************************************************************************
+ * Iterator
+ *****************************************************************************/
+
+GCIPage::iterator::iterator(const GCIPage* page)
+{
+ m_gciPage = page;
+ m_data = m_gciPage->m_page;
+ m_currentRecord = (Record*)m_data;
+}
+
+Record *
+GCIPage::iterator::first()
+{
+ return m_currentRecord;
+}
+
+Record *
+GCIPage::iterator::next()
+{
+ m_currentRecord = (Record*)
+ ((char*)(m_currentRecord)+ m_currentRecord->recordLen);
+ if((char*)m_currentRecord < (char*)(m_data + m_gciPage->m_usedBytes))
+ return m_currentRecord;
+ else {
+ return 0;
+ }
+}
+
+bool
+GCIPage::iterator::exists()
+{
+ return ((char*)m_currentRecord < (m_data + m_gciPage->m_usedBytes));
+}
diff --git a/ndb/src/rep/storage/GCIPage.hpp b/ndb/src/rep/storage/GCIPage.hpp
new file mode 100644
index 00000000000..50c5ab0cfba
--- /dev/null
+++ b/ndb/src/rep/storage/GCIPage.hpp
@@ -0,0 +1,114 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef GCI_PAGE_HPP
+#define GCI_PAGE_HPP
+
+#include "LogRecord.hpp"
+#include <TransporterDefinitions.hpp>
+
+#include <rep/rep_version.hpp>
+
+/**
+ * @class GCIPage
+ * @brief A GCIPage contains a number of LogRecords for a certain GCI.
+ */
+class GCIPage
+{
+public:
+ GCIPage(Uint32 gci);
+ GCIPage(Uint32 gci, char * dataPtr, Uint32 szBytes);
+
+ /**
+ * @fn insertLogRecord
+ * @param tableId the table this will be LogRecord applies to.
+ * @param operation the operation this LogRecord represents
+ * @param ptr A LinearSectionPtr p'tr that contains the data.
+ * @return PAGE_FULL if the page is full, otherwise "true"
+ */
+ bool insertLogRecord(Uint32 tableId, Uint32 operation,
+ class LinearSectionPtr ptr[3]);
+
+ /**
+ * @fn insertMetaRecord
+ */
+ bool insertMetaRecord(Uint32 tableId, class LinearSectionPtr ptr[3]);
+
+ /**
+ * @fn getFirstRecord
+ * @return First record (or NULL if no record is stored on page)
+ */
+ Record * getFirstRecord() { return m_first; };
+
+ /**
+ * @fn getStorage
+ */
+ Uint32 * getStoragePtr() const {return (Uint32*)m_page;} ;
+ Uint32 getStorageByteSize() const {return m_usedBytes;} ;
+ Uint32 getStorageWordSize() const {return m_usedBytes >> 2;};
+
+ /**
+ * @fn copyDataToPage
+ * @info copy dataPtr to Page
+ * @param dataPtr - data to copy
+ * @param dataBLen - size in bytes to copy.
+ */
+ void copyDataToPage(char * dataPtr, Uint32 szBytes);
+
+ /**
+ * Iterator for records (Not yet used! Maybe should not be used.)
+ */
+ class iterator {
+ public:
+ iterator(const GCIPage* page);
+ Record * first(); ///< @return First record (or NULL if no page exists)
+ Record * next(); ///< @return Next record (or NULL if no more records)
+ bool exists(); ///< @return true if another record exists-for next()
+ private:
+ Record * m_currentRecord;
+ const char * m_data;
+ const GCIPage * m_gciPage;
+ };
+ friend class GCIPage::iterator;
+
+ /**
+ * @fn getGCI
+ * Get the GCI of all log records stored on this page.
+ */
+ Uint32 getGCI() { return m_gci; };
+
+ /**
+ * @fn isFull
+ * @return true if page is full, i.e. is one attempt to add a record
+ * has failed, false otherwise.
+ */
+ bool isFull() { return m_full; };
+
+private:
+ Uint32 m_gci; ///< GCI for this page
+
+ Record * m_first; ///< Pointer to first log record
+ Record * m_last; ///< Pointer to last log record
+
+ bool m_full;
+
+ static const Uint32 m_pageBSize = 8192; ///< Page size in bytes
+ char m_page[m_pageBSize]; ///< Storage for pages
+ char * m_currentPagePos;
+ Uint32 m_usedBytes;
+};
+
+#endif
diff --git a/ndb/src/rep/storage/LogRecord.hpp b/ndb/src/rep/storage/LogRecord.hpp
new file mode 100644
index 00000000000..ba2632e23c7
--- /dev/null
+++ b/ndb/src/rep/storage/LogRecord.hpp
@@ -0,0 +1,82 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef LOG_RECORD_HPP
+#define LOG_RECORD_HPP
+
+#include <ndb_types.h>
+#include <stdlib.h>
+#include <portlib/NdbMem.h>
+
+/**
+ * @class Record
+ * @brief
+ */
+class Record {
+public:
+ enum RecordType { META = 1, LOG = 2 };
+ RecordType recordType;
+ Uint32 recordLen; ///< Size in bytes of entire log record, incl payload
+};
+
+
+/**
+ * @class LogRecord
+ * @brief
+ */
+class LogRecord : public Record {
+public:
+ ~LogRecord() {
+ NdbMem_Free(attributeHeader);
+ NdbMem_Free(attributeData);
+ }
+
+public:
+ Uint32 gci; //0
+ Uint32 operation; //4
+ Uint32 tableId; //8
+
+ Uint32 attributeHeaderWSize; //12
+ Uint32 attributeDataWSize; //16
+ Uint32 * attributeHeader; //20
+ Uint32 * attributeData; //24
+
+ /**
+ * Next pointer
+ */
+};
+
+
+/**
+ * @class MetaRecord
+ * @brief
+ */
+class MetaRecord : public Record {
+public:
+ ~MetaRecord() {
+ NdbMem_Free(data);
+ }
+
+public:
+ Uint32 gci;
+ Uint32 tableId;
+ Uint32 dataLen; //in words of the data (below)
+ Uint32 *data;
+};
+
+
+#endif
+
diff --git a/ndb/src/rep/storage/Makefile b/ndb/src/rep/storage/Makefile
new file mode 100644
index 00000000000..89b3af455e8
--- /dev/null
+++ b/ndb/src/rep/storage/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE := repserver
+
+ARCHIVE_TARGET := repstorage
+
+SOURCES = GCIContainer.cpp \
+ GCIContainerPS.cpp \
+ GCIBuffer.cpp \
+ GCIPage.cpp \
+ NodeGroupInfo.cpp \
+ NodeGroup.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/rep/storage/NodeConnectInfo.hpp b/ndb/src/rep/storage/NodeConnectInfo.hpp
new file mode 100644
index 00000000000..403f92a5999
--- /dev/null
+++ b/ndb/src/rep/storage/NodeConnectInfo.hpp
@@ -0,0 +1,29 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODE_CONNECTINFO_HPP
+#define NODE_CONNECTINFO_HPP
+
+#include <ndb_types.h>
+
+struct NodeConnectInfo {
+ NodeConnectInfo(Uint16 n, bool c): nodeId(n), connected(c) {};
+ Uint32 nodeId;
+ bool connected;
+};
+
+
+#endif
diff --git a/ndb/src/rep/storage/NodeGroup.cpp b/ndb/src/rep/storage/NodeGroup.cpp
new file mode 100644
index 00000000000..33451efb104
--- /dev/null
+++ b/ndb/src/rep/storage/NodeGroup.cpp
@@ -0,0 +1,149 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NodeGroup.hpp"
+#include <NdbOut.hpp>
+
+//#define NODE_GROUP_DEBUG
+
+NodeGroup::NodeGroup(Uint32 nodeGrp) {
+ m_nodeGrp = nodeGrp;
+ m_primaryNode = 0;
+}
+
+NodeGroup::~NodeGroup() {
+ for(Uint32 i=0; i<m_nodeConnectList.size(); i++) {
+ delete m_nodeConnectList[i];
+ m_nodeConnectList.erase(i);
+ }
+}
+
+void
+NodeGroup::addNode(Uint32 nodeId, bool connected) {
+#ifdef NODE_GROUP_DEBUG
+ ndbout_c("NodeGroup: addNode(nodeId=%d, connected=%d), nodegrp=%d",
+ nodeId, connected, m_nodeGrp);
+#endif
+
+ /**
+ * If node already in node group, then do nothing except
+ * setting the connect statusflag for the node (in case it
+ * has changed).
+ */
+ for(Uint32 i=0; i < m_nodeConnectList.size(); i++)
+ if(m_nodeConnectList[i]->nodeId == nodeId) {
+ m_nodeConnectList[i]->connected = connected;
+ return;
+ }
+
+ /**
+ * If node not already in node group, then add node
+ */
+ m_nodeConnectList.push_back(new NodeConnectInfo(nodeId, connected));
+ sort();
+
+#ifdef NODE_GROUP_DEBUG
+ for(Uint32 i=0; i < m_nodeConnectList.size(); i++)
+ ndbout_c("NodeGroup: NodeId=%d", m_nodeConnectList[i]->nodeId);
+#endif
+}
+
+/**
+ * crappy sort
+ */
+void NodeGroup::sort() {
+ NodeConnectInfo * tmp;
+ if(m_nodeConnectList.size()<2)
+ return;
+ for(Uint32 i=0; i < m_nodeConnectList.size()-1; i++) {
+ for(Uint32 j=m_nodeConnectList.size()-1;j>i+1; j--) {
+ if(m_nodeConnectList[j]->nodeId < m_nodeConnectList[j-1]->nodeId) {
+ tmp=m_nodeConnectList[j];
+ m_nodeConnectList[j]=m_nodeConnectList[j-1];
+ m_nodeConnectList[j-1]=tmp;
+ }
+ }
+ }
+}
+
+Uint32
+NodeGroup::getFirstConnectedNode() {
+ for(Uint32 i=0; i<m_nodeConnectList.size(); i++){
+ if(m_nodeConnectList[i]->connected)
+ return m_nodeConnectList[i]->nodeId;
+ }
+ return 0;
+}
+
+Uint32
+NodeGroup::getNodeGrp() {
+ return m_nodeGrp;
+}
+
+Vector <NodeConnectInfo *> *
+NodeGroup::getNodeConnectList(){
+ return &m_nodeConnectList;
+}
+
+void
+NodeGroup::setNodeConnectStatus(Uint32 nodeId, bool connected) {
+ for(Uint32 i=0; i<m_nodeConnectList.size(); i++){
+ if(m_nodeConnectList[i]->nodeId==nodeId) {
+ m_nodeConnectList[i]->connected=connected;
+ break;
+ }
+ }
+}
+
+bool
+NodeGroup::isConnected(Uint32 nodeId) {
+ for(Uint32 i=0; i<m_nodeConnectList.size(); i++){
+ if(m_nodeConnectList[i]->nodeId == nodeId) {
+ return m_nodeConnectList[i]->connected;
+ }
+ }
+ REPABORT1("Check for non-existing node to be connected", nodeId);
+}
+
+
+bool
+NodeGroup::fullyConnected() {
+ for(Uint32 i=0; i<m_nodeConnectList.size(); i++){
+ if(!(m_nodeConnectList[i]->connected))
+ return false;
+ }
+ return true;
+}
+
+bool
+NodeGroup::connectedNodeGrp() {
+ for(Uint32 i=0; i<m_nodeConnectList.size(); i++){
+ if(m_nodeConnectList[i]->connected) {
+ return true;
+ }
+ }
+ return false;
+}
+
+
+bool
+NodeGroup::exists(Uint32 nodeId) {
+ for(Uint32 i=0;i<m_nodeConnectList.size();i++) {
+ if(m_nodeConnectList[i]->nodeId==nodeId)
+ return true;
+ }
+ return false;
+}
diff --git a/ndb/src/rep/storage/NodeGroup.hpp b/ndb/src/rep/storage/NodeGroup.hpp
new file mode 100644
index 00000000000..1f515e02a23
--- /dev/null
+++ b/ndb/src/rep/storage/NodeGroup.hpp
@@ -0,0 +1,109 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODE_GROUP_HPP
+#define NODE_GROUP_HPP
+
+#include "NodeConnectInfo.hpp"
+#include <Vector.hpp>
+#include <ndb_types.h>
+
+#include <rep/rep_version.hpp>
+
+/**
+ * @class NodeGroup
+ * @brief Contains info about all nodes belonging to one node group
+ */
+class NodeGroup {
+public:
+ NodeGroup(Uint32 nodeGrp);
+ ~NodeGroup();
+ /**
+ * Add node to node group
+ * @param nodeId Node id of node to add
+ * @param connected Status of this node (true==connected)
+ */
+ void addNode(Uint32 nodeId, bool connected);
+
+ /**
+ * get first connected node in this node group
+ * @returns nodeId, 0 if there is no connected node...
+ */
+ Uint32 getFirstConnectedNode();
+
+ /**
+ * get the primary node id
+ * @returns nodeId, the primary node id
+ */
+ Uint32 getPrimaryNode() {return m_primaryNode;};
+
+
+ /**
+ * sets a node in this nodegroup as the primary node
+ */
+ void setPrimaryNode(Uint32 nodeId) {m_primaryNode=nodeId;};
+
+
+ /**
+ * get the node group
+ * @returns the nodegroup number (m_nodeGrp)
+ */
+ Uint32 getNodeGrp();
+
+ /**
+ * set the connection status for a particular node
+ * @param nodeId - the nodeId to set the connect status on
+ * @param connected - the status of this node (true==connected)
+ */
+ void setNodeConnectStatus(Uint32 nodeId, bool connected);
+
+ /**
+ * Get the connection status for a particular node
+ * @param nodeId - the nodeId to check the connect status on
+ * @returns true if node is connected, otherwise false
+ */
+ bool isConnected(Uint32 nodeId);
+
+ /**
+ * gives the status of this nodegroup.
+ * @returns true if atleast one node in the node group is connected
+ */
+ bool connectedNodeGrp();
+
+ /**
+ * @returns true if ALL nodes are connected
+ */
+ bool fullyConnected();
+
+ /**
+ *
+ * @returns true if node exists in nodegroup
+ */
+ bool exists(Uint32 nodeId);
+
+ Vector <NodeConnectInfo *> * getNodeConnectList();
+
+private:
+ /**
+ * Sort list (bubble sort)
+ */
+ void sort();
+ Uint32 m_primaryNode;
+ Uint32 m_nodeGrp;
+ Vector<NodeConnectInfo *> m_nodeConnectList;
+};
+
+#endif
diff --git a/ndb/src/rep/storage/NodeGroupInfo.cpp b/ndb/src/rep/storage/NodeGroupInfo.cpp
new file mode 100644
index 00000000000..8c250268997
--- /dev/null
+++ b/ndb/src/rep/storage/NodeGroupInfo.cpp
@@ -0,0 +1,218 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NodeGroupInfo.hpp"
+
+NodeGroupInfo::NodeGroupInfo()
+{
+}
+
+NodeGroupInfo::~NodeGroupInfo()
+{
+ for(Uint32 i=0; i<m_nodeGroupList.size(); i++) {
+ delete m_nodeGroupList[i];
+ }
+ m_nodeGroupList.clear();
+}
+
+
+void
+NodeGroupInfo::setPrimaryNode(Uint32 nodeGrp, Uint32 nodeId) {
+ Uint32 pos;
+ /**
+ * Validation check to find out that the nodegroup really exists.
+ * The list is not sorted, so the index of the nodegroup is returned
+ * in pos.
+ */
+ if(existsNodeGroup(nodeGrp, &pos)) {
+ m_nodeGroupList[pos]->setPrimaryNode(nodeId);
+ } else {
+ /**
+ * could not find node group
+ */
+ RLOG(("Node group not found"));
+ REPABORT("Node group not found");
+ }
+}
+
+Uint32
+NodeGroupInfo::getPrimaryNode(Uint32 nodeGrp) {
+ Uint32 pos;
+ /**
+ * Validation check to find out that the nodegroup really exists.
+ * The list is not sorted, so the index of the nodegroup is returned
+ * in pos.
+ */
+ if(existsNodeGroup(nodeGrp, &pos)) {
+ return m_nodeGroupList[pos]->getPrimaryNode();
+ } else {
+ /**
+ * could not find node group
+ */
+ RLOG(("Node group not found"));
+ REPABORT("Node group not found");
+ }
+}
+
+void
+NodeGroupInfo::addNodeToNodeGrp(Uint32 nodeId, bool connected, Uint32 nodeGrp)
+{
+ Uint32 pos;
+ if(existsNodeGroup(nodeGrp, &pos)) {
+ /**
+ * NG exists -> just add the node
+ */
+ m_nodeGroupList[pos]->addNode(nodeId, connected);
+
+ } else {
+ /**
+ * NG do not exist -> create a new nodeGrp and add the node
+ */
+ m_nodeGroupList.push_back(new NodeGroup(nodeGrp));
+
+ /**
+ * paranoia
+ */
+ if(existsNodeGroup(nodeGrp, &pos)) {
+ m_nodeGroupList[pos]->addNode(nodeId, connected);
+ } else {
+ REPABORT("");
+ }
+ }
+}
+
+Uint32
+NodeGroupInfo::findNodeGroup(Uint32 nodeId)
+{
+ /**
+ * Check for existance in each nodegroup
+ */
+ for(Uint32 i=0; i<m_nodeGroupList.size(); i++) {
+ if(m_nodeGroupList[i]->exists(nodeId)) return i;
+ }
+
+ REPABORT1("No node group known for node", nodeId);
+}
+
+Uint32
+NodeGroupInfo::getFirstConnectedNode(Uint32 nodeGrp)
+{
+ Uint32 pos;
+ /**
+ * Validation check to find out that the nodegroup really exists.
+ * The list is not sorted, so the index of the nodegroup is returned
+ * in pos.
+ */
+ if(existsNodeGroup(nodeGrp, &pos)) {
+ return m_nodeGroupList[pos]->getFirstConnectedNode();
+ } else {
+ /**
+ * could not find node group
+ */
+ REPABORT("");
+ }
+}
+
+bool
+NodeGroupInfo::connectedNodeGrp(Uint32 nodeGrp)
+{
+ return m_nodeGroupList[nodeGrp]->connectedNodeGrp();
+}
+
+bool
+NodeGroupInfo::isConnected(Uint32 nodeId)
+{
+ Uint32 nodeGrp = findNodeGroup(nodeId);
+ return m_nodeGroupList[nodeGrp]->isConnected(nodeId);
+
+}
+
+bool
+NodeGroupInfo::fullyConnected()
+{
+ for(Uint32 i=0; i<m_nodeGroupList.size(); i++) {
+ if(!(m_nodeGroupList[i]->fullyConnected()))
+ return false;
+ }
+ return true;
+}
+
+
+void
+NodeGroupInfo::setConnectStatus(Uint32 nodeId, bool connected)
+{
+ Uint32 nodeGrp = findNodeGroup(nodeId);
+ m_nodeGroupList[nodeGrp]->setNodeConnectStatus(nodeId,connected);
+}
+
+
+bool
+NodeGroupInfo::existsNodeGroup(Uint32 nodeGrp, Uint32 * pos)
+{
+ for(Uint32 i=0; i<m_nodeGroupList.size(); i++) {
+ if(m_nodeGroupList[i]->getNodeGrp()==nodeGrp) {
+ *pos=i;
+ return true;
+ }
+ }
+ return false;
+}
+
+
+/*****************************************************************************
+ * Iterator
+ *****************************************************************************/
+
+NodeGroupInfo::iterator::iterator(Uint32 nodeGrp, NodeGroupInfo * ngi)
+{
+ m_iterator = 0;
+ for(Uint32 i=0; i < ngi->m_nodeGroupList.size(); i++) {
+ if(ngi->m_nodeGroupList[i]->getNodeGrp()==nodeGrp) {
+ m_nodeList = ngi->m_nodeGroupList[i]->getNodeConnectList();
+ return;
+ }
+ }
+ m_nodeList=0;
+}
+
+bool
+NodeGroupInfo::iterator::exists()
+{
+ if(m_nodeList==0) return 0;
+ return (m_iterator < m_nodeList->size());
+}
+
+NodeConnectInfo *
+NodeGroupInfo::iterator::first()
+{
+ m_iterator=0;
+ if(m_nodeList==0) return 0;
+ if(m_nodeList->size() == 0) return 0;
+ return (*m_nodeList)[m_iterator];
+}
+
+NodeConnectInfo *
+NodeGroupInfo::iterator::next()
+{
+ m_iterator++;
+ if(m_nodeList==0) return 0;
+ if(m_nodeList->size() == 0) return 0;
+ if(m_iterator<m_nodeList->size())
+ return (*m_nodeList)[m_iterator];
+ else
+ return 0;
+}
+
diff --git a/ndb/src/rep/storage/NodeGroupInfo.hpp b/ndb/src/rep/storage/NodeGroupInfo.hpp
new file mode 100644
index 00000000000..605ccf76a38
--- /dev/null
+++ b/ndb/src/rep/storage/NodeGroupInfo.hpp
@@ -0,0 +1,147 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NODE_GROUPINFO_HPP
+#define NODE_GROUPINFO_HPP
+
+#include <Vector.hpp>
+#include <NdbStdio.h>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+//#include <NdbSleep.h>
+#include <NdbString.h>
+
+#include "NodeGroup.hpp"
+#include <rep/rep_version.hpp>
+
+/**
+ * @class NodeGroupInfo
+ * @brief Contains info about all node groups and their connectivity status
+ */
+class NodeGroupInfo {
+public:
+ NodeGroupInfo();
+ ~NodeGroupInfo();
+
+ /**
+ * Add a node to a nodegroup together with the status of the node
+ * @param nodeId - the nodeId to add
+ * @param connected - true/false
+ * @param nodeGrp - the nodegroup to add this node to
+ */
+ void addNodeToNodeGrp(Uint32 nodeId, bool connected, Uint32 nodeGrp);
+
+ /**
+ * Get the nodegroup that a node belongs to.
+ * @param nodeId - the nodeId to check wich nodegroup it belongs to
+ * @return the nodegroup
+ */
+ Uint32 findNodeGroup(Uint32 nodeId);
+
+ /**
+ * Get the first connected node in a node group
+ * @param nodegroup - the node group to get the node in.
+ * @return nodeId, 0 if there is no connected node in the nodegroup
+ */
+ Uint32 getFirstConnectedNode(Uint32 nodeGrp);
+
+
+ /**
+ * sets a nodeId in a nodeGroup as the primary node. If the
+ * primary node fails, then a new node in the node group is chosen
+ * @param nodegroup - the node group to get the node in.
+ * @param nodeId, 0 if there is no connected node in the nodegroup
+ */
+ void setPrimaryNode(Uint32 nodeGrp, Uint32 nodeId);
+
+ /**
+ * gets the nodeId in the nodegroup of the primary node.
+ * @param nodegroup - the node group to get the node in.
+ * @return nodeId, 0 if there is no connected node in the nodegroup
+ */
+ Uint32 getPrimaryNode(Uint32 nodeGrp);
+
+
+ /**
+ * Checks if at least one node in the nodegroup is connected.
+ * @param nodeGrp - the nodegrp to check
+ * @return true if >0 nodes are connected in the nodegroup
+ */
+ bool connectedNodeGrp(Uint32 nodeGrp);
+
+ /**
+ * Checks if a node is connected or not
+ * @param nodeId - the nodeId to check connectivity
+ * @return true if node is connected
+ */
+ bool isConnected(Uint32 nodeId);
+
+ /**
+ * Set if a node is connected or not
+ * @param nodeId - the nodeId to set the connect flag fory
+ * @param connected - true if connect false if disconnect
+ */
+ void setConnectStatus(Uint32 nodeId, bool connected);
+
+ /**
+ * Check if all nodes are connected in all nodegroups
+ * @return return true if ALL nodes are connected in ALL nodeGroups
+ */
+ bool fullyConnected();
+
+ /**
+ * Get the number of nodegroups
+ * @return the number of nodegroups.
+ */
+ Uint32 getNoOfNodeGroups() { return m_nodeGroupList.size();};
+
+ /**
+ * @class iterator
+ * The iterator class iterates over a nodegroup, returning nodeIds
+ * in that node group.
+ *
+ * @code
+ * NodeGroupInfo::iterator * it;
+ * for(Uint32 i=0;i < m_nodeGroupInfo->getNoOfNodeGroups();i++) {
+ * it = new NodeGroupInfo::iterator(i,m_nodeGroupInfo);
+ * for(NodeConnectInfo * nci=it->first(); it->exists();nci=it->next())
+ * ndbout_c("Iterating: %d", nci->nodeId);
+ *
+ * }
+ * @end code
+ */
+ class iterator {
+ public:
+ iterator(Uint32 nodeGrp, NodeGroupInfo * ngi);
+ NodeConnectInfo * first(); ///< @return nodeConnectInfo* if exists.
+ ///< (NULL if no more nodes exists)
+ NodeConnectInfo * next(); ///< @return nodeConnectInfo* if exists.
+ ///< (NULL if no more nodes exists)
+ bool exists(); ///< @return true if another nodeId exists (for next())
+ private:
+ Uint32 m_iterator;
+ const Vector<NodeConnectInfo *> * m_nodeList;
+ };
+ friend class NodeGroupInfo::iterator;
+
+private:
+ bool existsNodeGroup(Uint32 nodeGrp, Uint32 * pos);
+
+ Vector<NodeGroup *> m_nodeGroupList;
+};
+
+#endif
diff --git a/ndb/src/rep/transfer/Makefile b/ndb/src/rep/transfer/Makefile
new file mode 100644
index 00000000000..0d8851e287a
--- /dev/null
+++ b/ndb/src/rep/transfer/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapi repserver kernel
+
+ARCHIVE_TARGET := reptransfer
+
+SOURCES = TransPS.cpp \
+ TransSS.cpp \
+ TransSSSubscriptions.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/src/rep/transfer/TransPS.cpp b/ndb/src/rep/transfer/TransPS.cpp
new file mode 100644
index 00000000000..7af53f24415
--- /dev/null
+++ b/ndb/src/rep/transfer/TransPS.cpp
@@ -0,0 +1,550 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <assert.h>
+
+#include "ConfigRetriever.hpp"
+#include <NdbSleep.h>
+
+#include <NdbApiSignal.hpp>
+#include <AttributeHeader.hpp>
+
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <GrepError.hpp>
+#include <SimpleProperties.hpp>
+
+#include "TransPS.hpp"
+#include <rep/storage/NodeGroupInfo.hpp>
+
+/*****************************************************************************
+ * Constructor / Destructor / Init
+ *****************************************************************************/
+TransPS::TransPS(GCIContainerPS* gciContainer)
+{
+ m_repSender = new ExtSender();
+ m_gciContainerPS = gciContainer;
+}
+
+TransPS::~TransPS()
+{
+ delete m_repSender;
+}
+
+void
+TransPS::init(TransporterFacade * tf, const char * connectString)
+{
+ m_signalExecThread = NdbThread_Create(signalExecThread_C,
+ (void **)this,
+ 32768,
+ "TransPS_Service",
+ NDB_THREAD_PRIO_LOW);
+
+ ConfigRetriever configRetriever;
+ // configRetriever.setConnectString(connectString);
+ Properties* config = configRetriever.getConfig("REP", REP_VERSION_ID);
+ if (config == 0) {
+ ndbout << "TransPS: Configuration error: ";
+ const char* erString = configRetriever.getErrorString();
+ if (erString == 0) {
+ erString = "No error specified!";
+ }
+ ndbout << erString << endl;
+ exit(-1);
+ }
+
+ Properties * extConfig;
+ /**
+ * @todo Hardcoded primary system name
+ */
+ if (!config->getCopy("EXTERNAL SYSTEM_External", &extConfig)) {
+ ndbout << "External System \"External\" not found in configuration. "
+ << "Check config.ini." << endl;
+ config->print();
+ exit(-1);
+ }
+
+ m_ownNodeId = configRetriever.getOwnNodeId();
+ extConfig->put("LocalNodeId", m_ownNodeId);
+ extConfig->put("LocalNodeType", "REP");
+ Uint32 noOfConnections;
+ extConfig->get("NoOfConnections", &noOfConnections);
+ /* if (noOfConnections != 1) {
+ ndbout << "TransPS: There are " << noOfConnections << " connections "
+ << "defined in configuration"
+ << endl
+ << " There should be exactly one!" << endl;
+ exit(-1);
+ }
+ */
+ /******************************
+ * Set node id of external REP
+ ******************************/
+ const Properties * connection;
+ const char * extSystem;
+ Uint32 extRepNodeId, tmpOwnNodeId;
+
+ for(Uint32 i=0; i < noOfConnections; i++) {
+ extConfig->get("Connection", i, &connection);
+ if(connection == 0) REPABORT("No connection found");
+
+ if(connection->get("System1", &extSystem)) {
+ connection->get("NodeId1", &extRepNodeId);
+ connection->get("NodeId2", &tmpOwnNodeId);
+ } else {
+ connection->get("System2", &extSystem);
+ connection->get("NodeId1", &tmpOwnNodeId);
+ connection->get("NodeId2", &extRepNodeId);
+ }
+ if(m_ownNodeId == tmpOwnNodeId)
+ break;
+ }
+
+ if(extRepNodeId==0) REPABORT("External replication server not found");
+ if(extSystem==0) REPABORT("External system not found");
+
+ m_ownBlockNo = tf->open(this, execSignal, execNodeStatus);
+ assert(m_ownBlockNo > 0);
+
+ m_ownRef = numberToRef(m_ownBlockNo, m_ownNodeId);
+ assert(m_ownNodeId == tf->ownId());
+
+ ndbout_c("Phase 4 (TransPS): Connection %d to external REP node %d opened",
+ m_ownBlockNo, extRepNodeId);
+
+ m_repSender->setNodeId(extRepNodeId);
+ m_repSender->setOwnRef(m_ownRef);
+ m_repSender->setTransporterFacade(tf);
+}
+
+/*****************************************************************************
+ * Signal Queue Executor
+ *****************************************************************************/
+
+class SigMatch
+{
+public:
+ int gsn;
+ void (TransPS::* function)(NdbApiSignal *signal);
+
+ SigMatch() { gsn = 0; function = NULL; };
+
+ SigMatch(int _gsn, void (TransPS::* _function)(NdbApiSignal *signal)) {
+ gsn = _gsn;
+ function = _function;
+ };
+
+ bool check(NdbApiSignal *signal) {
+ if(signal->readSignalNumber() == gsn) return true;
+ return false;
+ };
+};
+
+void *
+TransPS::signalExecThread_C(void *r)
+{
+ TransPS *repps = (TransPS*)r;
+
+ repps->signalExecThreadRun();
+
+ NdbThread_Exit(0);
+ /* NOTREACHED */
+ return 0;
+}
+
+void
+TransPS::signalExecThreadRun()
+{
+ Vector<SigMatch> sl;
+
+ /**
+ * Signals executed here
+ */
+ sl.push_back(SigMatch(GSN_REP_GET_GCI_REQ,
+ &TransPS::execREP_GET_GCI_REQ));
+ sl.push_back(SigMatch(GSN_REP_GET_GCIBUFFER_REQ,
+ &TransPS::execREP_GET_GCIBUFFER_REQ));
+ sl.push_back(SigMatch(GSN_REP_CLEAR_PS_GCIBUFFER_REQ,
+ &TransPS::execREP_CLEAR_PS_GCIBUFFER_REQ));
+
+ /**
+ * Signals to be forwarded to GREP::PSCoord
+ */
+ sl.push_back(SigMatch(GSN_GREP_SUB_CREATE_REQ, &TransPS::sendSignalGrep));
+
+ /**
+ * Signals to be forwarded to GREP::PSCoord
+ */
+ sl.push_back(SigMatch(GSN_GREP_CREATE_SUBID_REQ, &TransPS::sendSignalGrep));
+ sl.push_back(SigMatch(GSN_GREP_SUB_START_REQ, &TransPS::sendSignalGrep));
+ sl.push_back(SigMatch(GSN_GREP_SUB_SYNC_REQ, &TransPS::sendSignalGrep));
+ sl.push_back(SigMatch(GSN_GREP_SUB_REMOVE_REQ, &TransPS::sendSignalGrep));
+
+ while(1) {
+ SigMatch *handler = NULL;
+ NdbApiSignal *signal = NULL;
+ if(m_signalRecvQueue.waitFor(sl, handler, signal)) {
+#if 0
+ ndbout_c("TransPS: Removed signal from queue (GSN: %d, QSize: %d)",
+ signal->readSignalNumber(), m_signalRecvQueue.size());
+#endif
+ if(handler->function != 0) {
+ (this->*handler->function)(signal);
+ delete signal;
+ signal = 0;
+ } else {
+ REPABORT("Illegal handler for signal");
+ }
+ }
+ }
+}
+
+void
+TransPS::sendSignalRep(NdbApiSignal * s)
+{
+ m_repSender->sendSignal(s);
+}
+
+void
+TransPS::sendSignalGrep(NdbApiSignal * s)
+{
+ m_grepSender->sendSignal(s);
+}
+
+void
+TransPS::sendFragmentedSignalRep(NdbApiSignal * s,
+ LinearSectionPtr ptr[3],
+ Uint32 sections)
+{
+ m_repSender->sendFragmentedSignal(s, ptr, sections);
+}
+
+void
+TransPS::sendFragmentedSignalGrep(NdbApiSignal * s,
+ LinearSectionPtr ptr[3],
+ Uint32 sections)
+{
+ m_grepSender->sendFragmentedSignal(s, ptr, sections);
+}
+
+
+void
+TransPS::execNodeStatus(void* obj, Uint16 nodeId, bool alive, bool nfCompleted)
+{
+// TransPS * thisObj = (TransPS*)obj;
+
+ RLOG(("Node changed state (NodeId %d, Alive %d, nfCompleted %d)",
+ nodeId, alive, nfCompleted));
+
+ if(!alive && !nfCompleted) { }
+
+ if(!alive && nfCompleted) { }
+}
+
+void
+TransPS::execSignal(void* executeObj, NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]){
+
+ TransPS * executor = (TransPS *) executeObj;
+
+ const Uint32 gsn = signal->readSignalNumber();
+ const Uint32 len = signal->getLength();
+
+ NdbApiSignal * s = new NdbApiSignal(executor->m_ownRef);
+ switch(gsn){
+ case GSN_REP_GET_GCI_REQ:
+ case GSN_REP_GET_GCIBUFFER_REQ:
+ case GSN_REP_CLEAR_PS_GCIBUFFER_REQ:
+ s->set(0, SSREPBLOCKNO, gsn, len);
+ memcpy(s->getDataPtrSend(), signal->getDataPtr(), 4 * len);
+ executor->m_signalRecvQueue.receive(s);
+ break;
+ case GSN_GREP_SUB_CREATE_REQ:
+ {
+ if(signal->m_noOfSections > 0) {
+ memcpy(s->getDataPtrSend(), signal->getDataPtr(), 4 * len);
+ s->set(0, GREP, gsn,
+ len);
+ executor->sendFragmentedSignalGrep(s,ptr,1);
+ delete s;
+ } else {
+ s->set(0, GREP, gsn, len);
+ memcpy(s->getDataPtrSend(), signal->getDataPtr(), 4 * len);
+ executor->m_signalRecvQueue.receive(s);
+ }
+ }
+ break;
+ case GSN_GREP_SUB_START_REQ:
+ case GSN_GREP_SUB_SYNC_REQ:
+ case GSN_GREP_SUB_REMOVE_REQ:
+ case GSN_GREP_CREATE_SUBID_REQ:
+ s->set(0, GREP, gsn, len);
+ memcpy(s->getDataPtrSend(), signal->getDataPtr(), 4 * len);
+ executor->m_signalRecvQueue.receive(s);
+ break;
+ default:
+ REPABORT1("Illegal signal received in execSignal", gsn);
+ }
+#if 0
+ ndbout_c("TransPS: Inserted signal into queue (GSN: %d, Len: %d)",
+ signal->readSignalNumber(), len);
+#endif
+}
+
+/*****************************************************************************
+ * Signal Receivers
+ *****************************************************************************/
+
+void
+TransPS::execREP_GET_GCIBUFFER_REQ(NdbApiSignal* signal)
+{
+ RepGetGciBufferReq * req = (RepGetGciBufferReq*)signal->getDataPtr();
+ Uint32 firstGCI = req->firstGCI;
+ Uint32 lastGCI = req->lastGCI;
+ Uint32 nodeGrp = req->nodeGrp;
+
+ RLOG(("Received request for %d:[%d-%d]", nodeGrp, firstGCI, lastGCI));
+
+ NodeGroupInfo * tmp = m_gciContainerPS->getNodeGroupInfo();
+ Uint32 nodeId = tmp->getPrimaryNode(nodeGrp);
+ /**
+ * If there is no connected node in the nodegroup -> abort.
+ * @todo: Handle error when a nodegroup is "dead"
+ */
+ if(!nodeId) {
+ RLOG(("There are no connected nodes in node group %d", nodeGrp));
+ sendREP_GET_GCIBUFFER_REF(signal, firstGCI, lastGCI, nodeGrp,
+ GrepError::REP_NO_CONNECTED_NODES);
+ return;
+ }
+
+ transferPages(firstGCI, lastGCI, nodeId, nodeGrp, signal);
+
+ /**
+ * Done tfxing pages, sending GCIBuffer conf.
+ */
+ Uint32 first, last;
+ m_gciContainerPS->getAvailableGCIBuffers(nodeGrp, &first, &last);
+
+ RepGetGciBufferConf * conf = (RepGetGciBufferConf*)req;
+ conf->senderRef = m_ownRef;
+ conf->firstPSGCI = first; // Buffers found on REP PS (piggy-back info)
+ conf->lastPSGCI = last;
+ conf->firstSSGCI = firstGCI; // Now been transferred to REP SS
+ conf->lastSSGCI = lastGCI;
+ conf->nodeGrp = nodeGrp;
+ signal->set(0, SSREPBLOCKNO, GSN_REP_GET_GCIBUFFER_CONF,
+ RepGetGciBufferConf::SignalLength);
+ sendSignalRep(signal);
+
+ RLOG(("Sent %d:[%d-%d] (Stored PS:%d:[%d-%d])",
+ nodeGrp, firstGCI, lastGCI, nodeGrp, first, last));
+}
+
+void
+TransPS::transferPages(Uint32 firstGCI, Uint32 lastGCI,
+ Uint32 nodeId, Uint32 nodeGrp,
+ NdbApiSignal * signal)
+{
+ /**
+ * Transfer pages in GCI Buffer to SS
+ * When buffer is sent, send accounting information.
+ */
+ RepDataPage * pageData = (RepDataPage*)signal->getDataPtr();
+ LinearSectionPtr ptr[1];
+ GCIPage * page;
+ for(Uint32 i=firstGCI; i<=lastGCI; i++) {
+ Uint32 totalSizeSent = 0;
+ GCIBuffer * buffer = m_gciContainerPS->getGCIBuffer(i, nodeId);
+
+ if(buffer != 0) {
+ GCIBuffer::iterator it(buffer);
+ /**
+ * Send all pages to SS
+ */
+ for (page = it.first(); page != 0; page = it.next()) {
+ ptr[0].p = page->getStoragePtr();
+ ptr[0].sz = page->getStorageWordSize();
+ totalSizeSent += ptr[0].sz;
+ pageData->gci = i;
+ pageData->nodeGrp = nodeGrp;
+ signal->set(0, SSREPBLOCKNO, GSN_REP_DATA_PAGE,
+ RepDataPage::SignalLength);
+ sendFragmentedSignalRep(signal, ptr, 1);
+ }
+
+ /**
+ * Send accounting information to SS
+ */
+ RepGciBufferAccRep * rep = (RepGciBufferAccRep *)pageData;
+ rep->gci = i;
+ rep->nodeGrp = nodeGrp;
+ rep->totalSentBytes = (4 * totalSizeSent); //words to bytes
+ signal->set(0, SSREPBLOCKNO, GSN_REP_GCIBUFFER_ACC_REP,
+ RepGciBufferAccRep::SignalLength);
+ sendSignalRep(signal);
+
+ RLOG(("Sending %d:[%d] (%d bytes) to external REP (nodeId %d)",
+ nodeGrp, i, 4*totalSizeSent, nodeId));
+ }
+ }
+ page = 0;
+}
+
+void
+TransPS::execREP_GET_GCI_REQ(NdbApiSignal* signal)
+{
+ RepGetGciReq * req = (RepGetGciReq*)signal->getDataPtr();
+ Uint32 nodeGrp = req->nodeGrp;
+
+ Uint32 first, last;
+ m_gciContainerPS->getAvailableGCIBuffers(nodeGrp, &first, &last);
+
+ RepGetGciConf * conf = (RepGetGciConf*) req;
+ conf->firstPSGCI = first;
+ conf->lastPSGCI = last;
+ conf->senderRef = m_ownRef;
+ conf->nodeGrp = nodeGrp;
+ signal->set(0, SSREPBLOCKNO, GSN_REP_GET_GCI_CONF,
+ RepGetGciConf::SignalLength);
+ sendSignalRep(signal);
+}
+
+/**
+ * REP_CLEAR_PS_GCIBUFFER_REQ
+ * destroy the GCI buffer in the GCI Container
+ * and send a CONF to Grep::SSCoord
+ */
+void
+TransPS::execREP_CLEAR_PS_GCIBUFFER_REQ(NdbApiSignal * signal)
+{
+ RepClearPSGciBufferReq * const req =
+ (RepClearPSGciBufferReq*)signal->getDataPtr();
+ Uint32 firstGCI = req->firstGCI;
+ Uint32 lastGCI = req->lastGCI;
+ Uint32 nodeGrp = req->nodeGrp;
+
+ assert(firstGCI >= 0 && lastGCI > 0);
+ if(firstGCI<0 && lastGCI <= 0)
+ {
+ RLOG(("WARNING! Illegal delete request ignored"));
+ sendREP_CLEAR_PS_GCIBUFFER_REF(signal, firstGCI, lastGCI,
+ 0, nodeGrp,
+ GrepError::REP_DELETE_NEGATIVE_EPOCH);
+ }
+
+ if(firstGCI==0 && lastGCI==(Uint32)0xFFFF) {
+ m_gciContainerPS->getAvailableGCIBuffers(nodeGrp, &firstGCI, &lastGCI);
+ RLOG(("Deleting PS:[%d-%d]", firstGCI, lastGCI));
+ }
+
+ if(firstGCI == 0) {
+ Uint32 f, l;
+ m_gciContainerPS->getAvailableGCIBuffers(nodeGrp, &f, &l);
+
+ RLOG(("Deleting PS:[%d-%d]", f, l));
+
+ if(f>firstGCI)
+ firstGCI = f;
+ }
+
+ /**
+ * Delete buffer
+ * Abort if we try to destroy a buffer that does not exist
+ * Deleting buffer from every node in the nodegroup
+ */
+ for(Uint32 i=firstGCI; i<=lastGCI; i++) {
+ if(!m_gciContainerPS->destroyGCIBuffer(i, nodeGrp)) {
+ sendREP_CLEAR_PS_GCIBUFFER_REF(signal, firstGCI, lastGCI, i, nodeGrp,
+ GrepError::REP_DELETE_NONEXISTING_EPOCH);
+ return;
+ }
+
+ RLOG(("Deleted PS:%d:[%d]", nodeGrp, i));
+ }
+
+ /**
+ * Send reply to Grep::SSCoord
+ */
+ RepClearPSGciBufferConf * conf = (RepClearPSGciBufferConf*)req;
+ conf->firstGCI = firstGCI;
+ conf->lastGCI = lastGCI;
+ conf->nodeGrp = nodeGrp;
+ signal->set(0, SSREPBLOCKNO, GSN_REP_CLEAR_PS_GCIBUFFER_CONF,
+ RepClearPSGciBufferConf::SignalLength);
+ sendSignalRep(signal);
+}
+
+/*****************************************************************************
+ * Signal Senders
+ *****************************************************************************/
+
+void
+TransPS::sendREP_GET_GCI_REF(NdbApiSignal* signal,
+ Uint32 nodeGrp,
+ Uint32 firstPSGCI, Uint32 lastPSGCI,
+ GrepError::Code err)
+{
+ RepGetGciRef * ref = (RepGetGciRef *)signal->getDataPtrSend();
+ ref->firstPSGCI = firstPSGCI;
+ ref->lastPSGCI = lastPSGCI;
+ ref->firstSSGCI = 0;
+ ref->lastSSGCI = 0;
+ ref->nodeGrp = nodeGrp;
+ ref->err = err;
+ signal->set(0, SSREPBLOCKNO, GSN_REP_GET_GCI_REF,
+ RepGetGciRef::SignalLength);
+ sendSignalRep(signal);
+}
+
+void
+TransPS::sendREP_CLEAR_PS_GCIBUFFER_REF(NdbApiSignal* signal,
+ Uint32 firstGCI, Uint32 lastGCI,
+ Uint32 currentGCI,
+ Uint32 nodeGrp,
+ GrepError::Code err)
+{
+ RepClearPSGciBufferRef * ref =
+ (RepClearPSGciBufferRef *)signal->getDataPtrSend();
+ ref->firstGCI = firstGCI;
+ ref->lastGCI = lastGCI;
+ ref->currentGCI = currentGCI;
+ ref->nodeGrp = nodeGrp;
+ ref->err = err;
+ signal->set(0, SSREPBLOCKNO, GSN_REP_CLEAR_PS_GCIBUFFER_REF,
+ RepClearPSGciBufferRef::SignalLength);
+ sendSignalRep(signal);
+}
+
+void
+TransPS::sendREP_GET_GCIBUFFER_REF(NdbApiSignal* signal,
+ Uint32 firstGCI, Uint32 lastGCI,
+ Uint32 nodeGrp,
+ GrepError::Code err)
+{
+ RepGetGciBufferRef * ref =
+ (RepGetGciBufferRef *)signal->getDataPtrSend();
+ ref->firstPSGCI = firstGCI;
+ ref->lastPSGCI = lastGCI;
+ ref->firstSSGCI = 0;
+ ref->lastSSGCI = 0;
+ ref->nodeGrp = nodeGrp;
+ ref->err = err;
+ signal->set(0, SSREPBLOCKNO, GSN_REP_GET_GCIBUFFER_REF,
+ RepGetGciBufferRef::SignalLength);
+ sendSignalRep(signal);
+}
diff --git a/ndb/src/rep/transfer/TransPS.hpp b/ndb/src/rep/transfer/TransPS.hpp
new file mode 100644
index 00000000000..35823f1eb19
--- /dev/null
+++ b/ndb/src/rep/transfer/TransPS.hpp
@@ -0,0 +1,132 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TransPS_HPP
+#define TransPS_HPP
+
+#include <NdbStdio.h>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbString.h>
+
+#include <TransporterDefinitions.hpp>
+#include <TransporterFacade.hpp>
+#include <ClusterMgr.hpp>
+
+#include <rep/storage/GCIContainerPS.hpp>
+
+#include <signaldata/RepImpl.hpp>
+
+#include <rep/SignalQueue.hpp>
+#include <rep/ExtSender.hpp>
+
+#include <rep/rep_version.hpp>
+
+/**
+ * @class TransPS
+ * @brief Responsible for REP-REP interface in Primary System role
+ */
+class TransPS {
+public:
+ /***************************************************************************
+ * Constructor / Destructor
+ ***************************************************************************/
+ TransPS(GCIContainerPS * gciContainer);
+ ~TransPS();
+
+ void init(TransporterFacade * tf, const char * connectString = NULL);
+
+ /***************************************************************************
+ * Public Methods
+ ***************************************************************************/
+ ExtSender * getRepSender() { return m_repSender; };
+ void setGrepSender(ExtSender * es) { m_grepSender = es; };
+
+private:
+ /***************************************************************************
+ * Private Methods
+ ***************************************************************************/
+ /**
+ * SignalQueue executor thread
+ */
+ void signalExecThreadRun();
+ static void * signalExecThread_C(void *);
+
+ static void execSignal(void* signalSender, NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]);
+
+ static void execNodeStatus(void* signalSender, NodeId,
+ bool alive, bool nfCompleted);
+
+ void sendSignalRep(NdbApiSignal * s);
+ void sendSignalGrep(NdbApiSignal * s);
+
+ void sendFragmentedSignalRep(NdbApiSignal * s, LinearSectionPtr ptr[3],
+ Uint32 sections );
+ void sendFragmentedSignalGrep(NdbApiSignal * s, LinearSectionPtr ptr[3],
+ Uint32 sections );
+
+ /***************************************************************************
+ * Signal executors
+ ***************************************************************************/
+ void execREP_CLEAR_PS_GCIBUFFER_REQ(NdbApiSignal*);
+ void execREP_GET_GCI_REQ(NdbApiSignal*);
+ void execREP_GET_GCIBUFFER_REQ(NdbApiSignal*);
+
+ /***************************************************************************
+ * Ref signal senders
+ ***************************************************************************/
+ void sendREP_GET_GCI_REF(NdbApiSignal* signal, Uint32 nodeGrp,
+ Uint32 firstPSGCI, Uint32 lastPSGCI,
+ GrepError::Code err);
+
+ void sendREP_CLEAR_PS_GCIBUFFER_REF(NdbApiSignal* signal,
+ Uint32 firstGCI, Uint32 lastGCI,
+ Uint32 currentGCI, Uint32 nodeGrp,
+ GrepError::Code err);
+
+ void sendREP_GET_GCIBUFFER_REF(NdbApiSignal* signal,
+ Uint32 firstGCI, Uint32 lastGCI,
+ Uint32 nodeGrp,
+ GrepError::Code err);
+
+ /***************************************************************************
+ * Other Methods
+ ***************************************************************************/
+ void transferPages(Uint32 firstGCI, Uint32 lastGCI, Uint32 id,
+ Uint32 nodeGrp, NdbApiSignal* signal);
+
+ /*************
+ * Variables
+ *************/
+ Uint32 m_ownNodeId; ///< NodeId of this node
+ Uint32 m_ownBlockNo; ///< BlockNo of this "block"
+ BlockReference m_ownRef; ///< Reference to this
+
+ BlockReference m_extRepRef; ///< Node ref of REP at SS
+
+ ExtSender * m_grepSender; ///< Responsible send to GREP
+ ExtSender * m_repSender; ///< Responsible send to REP
+
+ struct NdbThread * m_signalExecThread;
+ class SignalQueue m_signalRecvQueue;
+
+ GCIContainerPS * m_gciContainerPS; ///< Ref to gci container.
+};
+
+#endif
diff --git a/ndb/src/rep/transfer/TransSS.cpp b/ndb/src/rep/transfer/TransSS.cpp
new file mode 100644
index 00000000000..5399bfb4e3f
--- /dev/null
+++ b/ndb/src/rep/transfer/TransSS.cpp
@@ -0,0 +1,650 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <assert.h>
+
+#include "ConfigRetriever.hpp"
+
+#include <NdbApiSignal.hpp>
+#include <AttributeHeader.hpp>
+
+#include <signaldata/RepImpl.hpp>
+#include <signaldata/DictTabInfo.hpp>
+#include <signaldata/GetTabInfo.hpp>
+#include <signaldata/SumaImpl.hpp>
+#include <signaldata/GrepImpl.hpp>
+
+#include <SimpleProperties.hpp>
+#include <rep/rep_version.hpp>
+
+#include "TransSS.hpp"
+
+//#define DEBUG_REP_GET_GCI_CONF
+
+/*****************************************************************************
+ * Constructor / Destructor / Init
+ *****************************************************************************/
+TransSS::TransSS(GCIContainer * gciContainer, RepState * repState)
+{
+ m_repSender = new ExtSender();
+ if (!m_repSender) REPABORT("Could not allocate new ExtSender");
+ m_gciContainer = gciContainer;
+ m_repState = repState;
+}
+
+TransSS::~TransSS()
+{
+ delete m_repSender;
+}
+
+void
+TransSS::init(const char * connectString)
+{
+ m_signalExecThread = NdbThread_Create(signalExecThread_C,
+ (void **)this,
+ 32768,
+ "TransSS_Service",
+ NDB_THREAD_PRIO_LOW);
+ ConfigRetriever configRetriever;
+ configRetriever.setConnectString(connectString);
+
+ Properties* config = configRetriever.getConfig("REP", REP_VERSION_ID);
+ if (config == 0) {
+ ndbout << "Configuration error: ";
+ const char* erString = configRetriever.getErrorString();
+ if (erString == 0) {
+ erString = "No error specified!";
+ }
+ ndbout << erString << endl;
+ exit(-1);
+ }
+ Properties * extConfig;
+
+ /**
+ * @todo Hardcoded standby system name
+ */
+ if (!config->getCopy("EXTERNAL SYSTEM_External", &extConfig)) {
+ ndbout << "External System \"External\" not found in configuration. "
+ << "Check config.ini." << endl;
+ config->print();
+ exit(-1);
+ }
+ m_ownNodeId = configRetriever.getOwnNodeId();
+ extConfig->put("LocalNodeId", m_ownNodeId);
+ extConfig->put("LocalNodeType", "REP");
+ Uint32 noOfConnections;
+ extConfig->get("NoOfConnections", &noOfConnections);
+ /* if (noOfConnections != 1) {
+ ndbout << "TransSS: There are " << noOfConnections << " connections "
+ << "defined in configuration"
+ << endl
+ << " There should be exactly one!" << endl;
+ exit(-1);
+ }*/
+
+ /******************************
+ * Set node id of external REP
+ ******************************/
+ const Properties * connection;
+ const char * extSystem;
+
+ Uint32 extRepNodeId, tmpOwnNodeId;
+
+ for(Uint32 i=0; i < noOfConnections; i++) {
+ extConfig->get("Connection", i, &connection);
+ if(connection == 0) REPABORT("Connection not found");
+
+ if(connection->get("System1", &extSystem)) {
+ connection->get("NodeId1", &extRepNodeId);
+ connection->get("NodeId2", &tmpOwnNodeId);
+ } else {
+ connection->get("System2", &extSystem);
+ connection->get("NodeId1", &tmpOwnNodeId);
+ connection->get("NodeId2", &extRepNodeId);
+ }
+ if(m_ownNodeId == tmpOwnNodeId)
+ break;
+ }
+
+ if(extRepNodeId==0) REPABORT("External replication server not found");
+ if(extSystem==0) REPABORT("External system not found");
+
+ m_transporterFacade = new TransporterFacade();
+ if (!m_transporterFacade->init(extConfig))
+ {
+ ndbout << "TransSS: Failed to initialize transporter facade" << endl;
+ exit(-1);
+ }
+
+ m_ownBlockNo = m_transporterFacade->open(this, execSignal, execNodeStatus);
+ assert(m_ownBlockNo > 0);
+ m_ownRef = numberToRef(m_ownBlockNo, m_ownNodeId);
+ assert(m_ownNodeId == m_transporterFacade->ownId());
+
+ ndbout_c("Phase 2 (TransSS): Connection %d to external REP node %d opened",
+ m_ownBlockNo, extRepNodeId);
+
+ m_repSender->setNodeId(extRepNodeId);
+ m_repSender->setOwnRef(m_ownRef);
+ m_repSender->setTransporterFacade(m_transporterFacade);
+}
+
+/*****************************************************************************
+ * Signal Queue Executor
+ *****************************************************************************/
+
+class SigMatch
+{
+public:
+ int gsn;
+ void (TransSS::* function)(NdbApiSignal *signal);
+
+ SigMatch() { gsn = 0; function = NULL; };
+
+ SigMatch(int _gsn, void (TransSS::* _function)(NdbApiSignal *signal)) {
+ gsn = _gsn;
+ function = _function;
+ };
+
+ bool check(NdbApiSignal *signal) {
+ if(signal->readSignalNumber() == gsn)
+ return true;
+ return false;
+ };
+};
+
+void *
+TransSS::signalExecThread_C(void *r)
+{
+ TransSS *transss = (TransSS*)r;
+
+ transss->signalExecThreadRun();
+ NdbThread_Exit(0);
+ /* NOTREACHED */
+ return 0;
+}
+
+void
+TransSS::signalExecThreadRun()
+{
+ Vector<SigMatch> sl;
+ /**
+ * Signals to be forwarded to TransPS
+ */
+ sl.push_back(SigMatch(GSN_REP_GET_GCI_REQ,
+ &TransSS::sendSignalRep));
+ sl.push_back(SigMatch(GSN_REP_GET_GCIBUFFER_REQ,
+ &TransSS::sendSignalRep));
+ /**
+ * Signals to be executed
+ */
+ sl.push_back(SigMatch(GSN_REP_GCIBUFFER_ACC_REP,
+ &TransSS::execREP_GCIBUFFER_ACC_REP));
+ sl.push_back(SigMatch(GSN_REP_DISCONNECT_REP,
+ &TransSS::execREP_DISCONNECT_REP));
+ sl.push_back(SigMatch(GSN_GREP_SUB_REMOVE_CONF,
+ &TransSS::execGREP_SUB_REMOVE_CONF));
+ sl.push_back(SigMatch(GSN_REP_GET_GCIBUFFER_CONF,
+ &TransSS::execREP_GET_GCIBUFFER_CONF));
+
+ sl.push_back(SigMatch(GSN_REP_CLEAR_PS_GCIBUFFER_CONF,
+ &TransSS::execREP_CLEAR_PS_GCIBUFFER_CONF));
+ sl.push_back(SigMatch(GSN_GREP_SUB_SYNC_CONF,
+ &TransSS::execGREP_SUB_SYNC_CONF));
+ sl.push_back(SigMatch(GSN_GREP_SUB_SYNC_REF,
+ &TransSS::execGREP_SUB_SYNC_REF));
+ sl.push_back(SigMatch(GSN_REP_GET_GCIBUFFER_REF,
+ &TransSS::execREP_GET_GCIBUFFER_REF));
+
+ /**
+ * Signals to be executed : Subscriptions
+ */
+ sl.push_back(SigMatch(GSN_GREP_CREATE_SUBID_CONF,
+ &TransSS::execGREP_CREATE_SUBID_CONF));
+ sl.push_back(SigMatch(GSN_GREP_CREATE_SUBID_REF,
+ &TransSS::execGREP_CREATE_SUBID_REF));
+ sl.push_back(SigMatch(GSN_GREP_SUB_CREATE_CONF,
+ &TransSS::execGREP_SUB_CREATE_CONF));
+ sl.push_back(SigMatch(GSN_GREP_SUB_CREATE_REF,
+ &TransSS::execGREP_SUB_CREATE_REF));
+ sl.push_back(SigMatch(GSN_GREP_SUB_START_CONF,
+ &TransSS::execGREP_SUB_START_CONF));
+ sl.push_back(SigMatch(GSN_GREP_SUB_START_REF,
+ &TransSS::execGREP_SUB_START_REF));
+
+ /**
+ * Signals to be executed and forwarded
+ */
+ sl.push_back(SigMatch(GSN_REP_GET_GCI_CONF,
+ &TransSS::execREP_GET_GCI_CONF));
+
+ /**
+ * Signals to be forwarded
+ */
+ sl.push_back(SigMatch(GSN_GREP_SUB_REMOVE_REF,
+ &TransSS::execGREP_SUB_REMOVE_REF));
+ sl.push_back(SigMatch(GSN_REP_CLEAR_PS_GCIBUFFER_REF,
+ &TransSS::execREP_CLEAR_PS_GCIBUFFER_REF));
+ sl.push_back(SigMatch(GSN_REP_GET_GCI_REF,
+ &TransSS::execREP_GET_GCI_REF));
+
+ while(1) {
+ SigMatch *handler = NULL;
+ NdbApiSignal *signal = NULL;
+ if(m_signalRecvQueue.waitFor(sl, handler, signal))
+ {
+#if 0
+ ndbout_c("TransSS: Removed signal from queue (GSN: %d, QSize: %d)",
+ signal->readSignalNumber(), m_signalRecvQueue.size());
+#endif
+ if(handler->function != 0)
+ {
+ (this->*handler->function)(signal);
+ delete signal;
+ signal = 0;
+ } else {
+ REPABORT("Illegal handler for signal");
+ }
+ }
+ }
+}
+
+void
+TransSS::sendSignalRep(NdbApiSignal * s)
+{
+ m_repSender->sendSignal(s);
+}
+
+void
+TransSS::execNodeStatus(void* obj, Uint16 nodeId,
+ bool alive, bool nfCompleted)
+{
+ TransSS * thisObj = (TransSS*)obj;
+
+ if (alive) {
+ thisObj->m_repState->eventNodeConnected(nodeId);
+
+ } else if (!nfCompleted) {
+ thisObj->m_repState->eventNodeDisconnected(nodeId);
+
+ } else if (nfCompleted) {
+ thisObj->m_repState->eventNodeConnectable(nodeId);
+
+ } else {
+ REPABORT("Illegal state for execNodeStatus");
+ }
+}
+
+void
+TransSS::execSignal(void* executorObj, NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3])
+{
+ TransSS * executor = (TransSS *) executorObj;
+
+ const Uint32 gsn = signal->readSignalNumber();
+ const Uint32 len = signal->getLength();
+
+ NdbApiSignal * s = new NdbApiSignal(executor->m_ownRef);
+ switch (gsn) {
+ case GSN_REP_GET_GCI_REQ:
+ case GSN_REP_GET_GCIBUFFER_REQ:
+ case GSN_REP_GET_GCIBUFFER_CONF:
+ case GSN_GREP_SUB_REMOVE_CONF:
+ case GSN_REP_DISCONNECT_REP:
+ case GSN_REP_GCIBUFFER_ACC_REP:
+ s->set(0, PSREPBLOCKNO, gsn, len);
+ memcpy(s->getDataPtrSend(), signal->getDataPtr(), 4 * len);
+ executor->m_signalRecvQueue.receive(s);
+ break;
+ case GSN_GREP_CREATE_SUBID_CONF:
+ case GSN_GREP_SUB_CREATE_CONF:
+ case GSN_GREP_SUB_START_CONF:
+ case GSN_GREP_SUB_SYNC_CONF:
+ case GSN_REP_GET_GCI_CONF:
+ case GSN_REP_CLEAR_PS_GCIBUFFER_CONF:
+ case GSN_GREP_CREATE_SUBID_REF:
+ case GSN_GREP_SUB_CREATE_REF:
+ case GSN_GREP_SUB_START_REF:
+ case GSN_GREP_SUB_SYNC_REF:
+ case GSN_GREP_SUB_REMOVE_REF:
+ case GSN_REP_GET_GCI_REF:
+ case GSN_REP_GET_GCIBUFFER_REF:
+ case GSN_REP_CLEAR_PS_GCIBUFFER_REF:
+ s->set(0, GREP, gsn, len);
+ memcpy(s->getDataPtrSend(), signal->getDataPtr(), 4 * len);
+ executor->m_signalRecvQueue.receive(s);
+ break;
+ case GSN_REP_DATA_PAGE:
+ executor->execREP_DATA_PAGE(signal, ptr);
+ delete s; s = 0;
+ break;
+ default:
+ REPABORT1("Illegal signal received in execSignal %d", gsn);
+ }
+
+#if 0
+ ndbout_c("TransSS: Inserted signal into queue (GSN: %d, Len: %d)",
+ signal->readSignalNumber(), len);
+#endif
+}
+
+/*****************************************************************************
+ * Signal Executors
+ *****************************************************************************/
+
+void
+TransSS::execREP_DATA_PAGE(NdbApiSignal * signal, LinearSectionPtr ptr[3])
+{
+ RepDataPage * const page = (RepDataPage*)signal->getDataPtr();
+ m_gciContainer->insertPage(page->gci, page->nodeGrp,
+ (char*)(ptr[0].p), 4 * ptr[0].sz);
+}
+
+/**
+ * Recd from TransPS
+ */
+void
+TransSS::execREP_GCIBUFFER_ACC_REP(NdbApiSignal * signal)
+{
+ RepGciBufferAccRep * const rep =
+ (RepGciBufferAccRep * )signal->getDataPtr();
+
+ Uint32 gci = rep->gci;
+ Uint32 nodeGrp = rep->nodeGrp;
+ Uint32 totalSize = rep->totalSentBytes;
+ GCIBuffer * buffer = m_gciContainer->getGCIBuffer(gci, nodeGrp);
+ Uint32 getReceivedBytes = 0;
+ if (buffer != 0)
+ getReceivedBytes = buffer->getReceivedBytes();
+
+ RLOG(("TransSS: Received %d:[%d] (%d of %d bytes)",
+ nodeGrp, gci, getReceivedBytes, totalSize));
+
+ if(getReceivedBytes != totalSize) {
+ REPABORT("Did not receive correct number of bytes");
+ }
+}
+
+/**
+ * Received from primary system
+ */
+void
+TransSS::execREP_GET_GCIBUFFER_CONF(NdbApiSignal * signal)
+{
+ RepGetGciBufferConf * conf = (RepGetGciBufferConf*)signal->getDataPtr();
+ conf->senderRef = m_ownRef;
+ Uint32 first = conf->firstSSGCI;
+ Uint32 last = conf->lastSSGCI;
+ for(Uint32 i = first; i <= last; i++) {
+ m_gciContainer->setCompleted(i, conf->nodeGrp);
+ }
+
+ /**
+ * Buffers @ PS
+ */
+ Interval ps(conf->firstPSGCI, conf->lastPSGCI);
+ m_repState->add(Channel::PS, conf->nodeGrp, ps);
+
+ /**
+ * Buffers @ SS
+ */
+ Uint32 ssfirst, sslast;
+ m_gciContainer->getAvailableGCIBuffers(conf->nodeGrp, &ssfirst, &sslast);
+ Interval ss(ssfirst, sslast);
+ m_repState->clear(Channel::SS, conf->nodeGrp, universeInterval);
+ m_repState->add(Channel::SS, conf->nodeGrp, ss);
+ m_repState->clear(Channel::SSReq, conf->nodeGrp, ss);
+
+ RLOG(("Transfered epochs (PS:%d[%d-%d], SS:%d[%d-%d])",
+ conf->nodeGrp, conf->firstPSGCI, conf->lastPSGCI,
+ conf->nodeGrp, conf->firstSSGCI, conf->lastSSGCI));
+}
+
+/**
+ * Received from primary system
+ */
+void
+TransSS::execGREP_SUB_REMOVE_CONF(NdbApiSignal * signal)
+{
+ GrepSubRemoveConf * conf = (GrepSubRemoveConf* )signal->getDataPtr();
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+
+ /**
+ * @todo Fix this sending
+ */
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionInfo;
+ signal->theData[1] = GrepEvent::GrepSS_SubRemoveConf;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ sendSignal(CMVMI_REF,GSN_EVENT_REP,signal, 4, JBB);
+#endif
+
+ m_repState->eventSubscriptionDeleted(subId, subKey);
+ RLOG(("Subscription deleted (SubId: %d, SubKey: %d)", subId, subKey));
+}
+
+void
+TransSS::execGREP_SUB_REMOVE_REF(NdbApiSignal * signal)
+{
+ GrepSubRemoveRef * ref = (GrepSubRemoveRef* )signal->getDataPtr();
+ Uint32 subId = ref->subscriptionId;
+ Uint32 subKey = ref->subscriptionKey;
+
+ /** @todo: Add repevent for this */
+ RLOG(("TransSS: Warning: Grep sub remove ref (SubId: %d, SubKey: %d)",
+ subId, subKey));
+}
+
+/**
+ * Received from primary system
+ */
+void
+TransSS::execREP_GET_GCI_CONF(NdbApiSignal * signal)
+{
+ RepGetGciConf * conf = (RepGetGciConf*)signal->getDataPtr();
+ Uint32 nodeGrp = conf->nodeGrp;
+ Interval i(conf->firstPSGCI, conf->lastPSGCI);
+ m_repState->add(Channel::PS, nodeGrp, i);
+
+ Uint32 first, last;
+ m_gciContainer->getAvailableGCIBuffers(nodeGrp, &first, &last);
+ Interval j(first, last);
+ m_repState->clear(Channel::SS, nodeGrp, universeInterval);
+ m_repState->add(Channel::SS, nodeGrp, j);
+
+#ifdef DEBUG_REP_GET_GCI_CONF
+ RLOG(("TransSS: Requestor info received "
+ "(PS: %d:[%d-%d], SS: %d:[%d-%d])",
+ conf->nodeGrp, conf->firstPSGCI, conf->lastPSGCI,
+ conf->nodeGrp, conf->firstSSGCI, conf->lastSSGCI));
+#endif
+}
+
+void
+TransSS::execREP_GET_GCI_REF(NdbApiSignal * signal)
+{
+ RepGetGciRef * ref = (RepGetGciRef*)signal->getDataPtr();
+ Uint32 nodeGrp = ref->nodeGrp;
+
+ RLOG(("WARNING! Requestor info request failed (Nodegrp: %d)", nodeGrp));
+}
+
+/**
+ * Recd from GrepPS
+ * This signal means that a DB node has disconnected.
+ * @todo Do we need to know that a DB node disconnected?
+ *
+ * A node has disconnected (REP or PS DB)
+ * @todo let the requestor respond to this event
+ * in a proper way.
+ */
+void
+TransSS::execREP_DISCONNECT_REP(NdbApiSignal * signal)
+{
+ RepDisconnectRep * const rep =
+ (RepDisconnectRep*)signal->getDataPtr();
+
+ //Uint32 nodeId = rep->nodeId;
+ Uint32 nodeType = rep->nodeType;
+
+ if((RepDisconnectRep::NodeType)nodeType == RepDisconnectRep::REP)
+ {
+ m_repState->disable();
+ }
+}
+
+/**
+ * The buffer is now deleted on REP PS. We can now clear it from PS.
+ */
+void
+TransSS::execREP_CLEAR_PS_GCIBUFFER_CONF(NdbApiSignal * signal)
+{
+ RepClearPSGciBufferConf * const conf =
+ (RepClearPSGciBufferConf*)signal->getDataPtr();
+ Uint32 firstGCI = conf->firstGCI;
+ Uint32 lastGCI = conf->lastGCI;
+ Uint32 nodeGrp = conf->nodeGrp;
+ Interval i(firstGCI, lastGCI);
+ m_repState->clear(Channel::PS, nodeGrp, i);
+ m_repState->clear(Channel::DelReq, nodeGrp, i);
+
+ RLOG(("Deleted PS:%d:[%d-%d]", nodeGrp, firstGCI, lastGCI));
+}
+
+/**
+ * Something went wrong when deleting buffer on REP PS
+ */
+void
+TransSS::execREP_CLEAR_PS_GCIBUFFER_REF(NdbApiSignal * signal)
+{
+ RepClearPSGciBufferRef * const ref =
+ (RepClearPSGciBufferRef*)signal->getDataPtr();
+ Uint32 firstGCI = ref->firstGCI;
+ Uint32 lastGCI = ref->lastGCI;
+ Uint32 nodeGrp = ref->nodeGrp;
+
+ RLOG(("WARNING! Could not delete PS:%d:[%d-%d]", nodeGrp, firstGCI, lastGCI));
+}
+
+/*****************************************************************************
+ * Signal Executors : SCAN
+ *****************************************************************************/
+
+/**
+ * Scan has started on PS side... (says PS REP)
+ */
+void
+TransSS::execGREP_SUB_SYNC_CONF(NdbApiSignal* signal)
+{
+ GrepSubSyncConf * const conf = (GrepSubSyncConf * ) signal->getDataPtr();
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ Interval epochs(conf->firstGCI, conf->lastGCI);
+ SubscriptionData::Part part = (SubscriptionData::Part) conf->part;
+
+ switch(part) {
+ case SubscriptionData::MetaData:
+ RLOG(("Metascan completed. Subcription %d-%d, Epochs [%d-%d]",
+ subId, subKey, epochs.first(), epochs.last()));
+ m_repState->eventMetaScanCompleted(signal, subId, subKey, epochs);
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionInfo;
+ signal->theData[1] = GrepEvent::GrepSS_SubSyncMetaConf;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ signal->theData[4] = gci;
+ sendSignal(CMVMI_REF,GSN_EVENT_REP,signal, 5, JBB);
+#endif
+ break;
+ case SubscriptionData::TableData:
+ RLOG(("Datascan completed. Subcription %d-%d, Epochs [%d-%d]",
+ subId, subKey, epochs.first(), epochs.last()));
+ m_repState->eventDataScanCompleted(signal, subId, subKey, epochs);
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionInfo;
+ signal->theData[1] = GrepEvent::GrepSS_SubSyncDataConf;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ signal->theData[4] = gci;
+ sendSignal(CMVMI_REF,GSN_EVENT_REP,signal, 5, JBB);
+#endif
+ break;
+ default:
+ REPABORT3("Wrong subscription part", part, subId, subKey);
+ }
+}
+
+void
+TransSS::execGREP_SUB_SYNC_REF(NdbApiSignal* signal)
+{
+ GrepSubSyncRef * const ref = (GrepSubSyncRef * ) signal->getDataPtr();
+ Uint32 subId = ref->subscriptionId;
+ Uint32 subKey = ref->subscriptionKey;
+ SubscriptionData::Part part = (SubscriptionData::Part) ref->part;
+ GrepError::Code error = (GrepError::Code) ref->err;
+
+ switch(part) {
+ case SubscriptionData::MetaData:
+ m_repState->eventMetaScanFailed(subId, subKey, error);
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionAlert;
+ signal->theData[1] = GrepEvent::GrepSS_SubSyncMetaRef;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ // signal->theData[4] = gci;
+ sendSignal(CMVMI_REF,GSN_EVENT_REP,signal, 5, JBB);
+#endif
+ break;
+ case SubscriptionData::TableData:
+ m_repState->eventDataScanFailed(subId, subKey, error);
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionAlert;
+ signal->theData[1] = GrepEvent::GrepSS_SubSyncDataRef;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ //signal->theData[4] = m_lastScanGCI;
+ sendSignal(CMVMI_REF,GSN_EVENT_REP,signal, 5, JBB);
+#endif
+ break;
+ default:
+ REPABORT3("Wrong subscription part", part, subId, subKey);
+ }
+}
+
+/**
+ * Something went wrong says REP PS
+ */
+void
+TransSS::execREP_GET_GCIBUFFER_REF(NdbApiSignal* signal)
+{
+ RepGetGciBufferRef * const ref = (RepGetGciBufferRef*)signal->getDataPtr();
+ /*
+ Uint32 senderData = ref->senderData;
+ Uint32 senderRef = ref->senderRef;
+ Uint32 firstPSGCI = ref->firstPSGCI;
+ Uint32 lastPSGCI = ref->lastPSGCI;
+ Uint32 firstSSGCI = ref->firstSSGCI;
+ Uint32 lastSSGCI = ref->lastSSGCI;
+ Uint32 currentGCIBuffer = ref->currentGCIBuffer;
+ Uint32 nodeGrp = ref->nodeGrp;
+ */
+ GrepError::Code err = ref->err;
+
+ RLOG(("WARNING! Request to get buffer failed. Error %d:%s",
+ err, GrepError::getErrorDesc(err)));
+}
diff --git a/ndb/src/rep/transfer/TransSS.hpp b/ndb/src/rep/transfer/TransSS.hpp
new file mode 100644
index 00000000000..90f320a079e
--- /dev/null
+++ b/ndb/src/rep/transfer/TransSS.hpp
@@ -0,0 +1,143 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TransSS_HPP
+#define TransSS_HPP
+
+#include <NdbStdio.h>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbString.h>
+
+#include <TransporterDefinitions.hpp>
+#include <TransporterFacade.hpp>
+#include <ClusterMgr.hpp>
+#include <API.hpp>
+
+#include <rep/storage/GCIContainer.hpp>
+
+#include <rep/SignalQueue.hpp>
+#include <rep/ExtSender.hpp>
+
+#include <rep/state/RepState.hpp>
+
+/**
+ * @class TransSS
+ * @brief Responsible for REP-REP interface in Standby System role
+ */
+class TransSS {
+public:
+ /***************************************************************************
+ * Constructor / Destructor / Init
+ ***************************************************************************/
+ TransSS(GCIContainer * gciContainer, RepState * repState);
+ ~TransSS();
+ void init(const char * connectString = NULL);
+
+ /***************************************************************************
+ * Public Methods
+ ***************************************************************************/
+ ExtSender * getRepSender() { return m_repSender; };
+ TransporterFacade * getTransporterFacade() { return m_transporterFacade; };
+
+private:
+ /***************************************************************************
+ * Private Methods
+ ***************************************************************************/
+ void signalExecThreadRun(); ///< SignalQueue executor thread
+ static void * signalExecThread_C(void *);
+
+ static void execSignal(void* executorObj, NdbApiSignal* signal,
+ class LinearSectionPtr ptr[3]);
+ static void execNodeStatus(void* executorObj, NodeId, bool alive,
+ bool nfCompleted);
+
+ void sendSignalRep(NdbApiSignal * s);
+
+ /***************************************************************************
+ * Signal receivers
+ ***************************************************************************/
+ void execREP_GET_GCI_REQ(NdbApiSignal*);
+ void execREP_GET_GCI_CONF(NdbApiSignal*);
+ void execREP_GET_GCI_REF(NdbApiSignal*);
+
+ void execREP_GET_GCIBUFFER_REQ(NdbApiSignal*);
+ void execREP_GET_GCIBUFFER_CONF(NdbApiSignal*);
+ void execREP_GET_GCIBUFFER_REF(NdbApiSignal*);
+
+ void execGREP_SUB_REMOVE_CONF(NdbApiSignal *);
+ void execGREP_SUB_REMOVE_REF(NdbApiSignal *);
+
+ void execREP_INSERT_GCIBUFFER_REQ(NdbApiSignal*);
+ void execREP_INSERT_GCIBUFFER_CONF(NdbApiSignal*);
+ void execREP_INSERT_GCIBUFFER_REF(NdbApiSignal*);
+
+ void execREP_DATA_PAGE(NdbApiSignal* signal, LinearSectionPtr ptr[3]);
+
+ void execREP_GCIBUFFER_ACC_REP(NdbApiSignal*);
+ void execREP_DISCONNECT_REP(NdbApiSignal*);
+
+
+ void execREP_CLEAR_PS_GCIBUFFER_CONF(NdbApiSignal*);
+ void execREP_CLEAR_PS_GCIBUFFER_REF(NdbApiSignal*);
+
+ void execGREP_SUB_SYNC_CONF(NdbApiSignal*);
+ void execGREP_SUB_SYNC_REF(NdbApiSignal*);
+
+ /***************************************************************************
+ * Signal receivers : Subscriptions
+ ***************************************************************************/
+ void execGREP_CREATE_SUBID_CONF(NdbApiSignal*);
+ void execGREP_CREATE_SUBID_REF(NdbApiSignal*);
+ void execGREP_SUB_CREATE_CONF(NdbApiSignal*);
+ void execGREP_SUB_CREATE_REF(NdbApiSignal*);
+ void execGREP_SUB_START_CONF(NdbApiSignal*);
+ void execGREP_SUB_START_REF(NdbApiSignal*);
+
+ /***************************************************************************
+ * Ref signal senders
+ ***************************************************************************/
+
+ void sendREP_GET_GCI_REF(NdbApiSignal* signal, Uint32 nodeGrp,
+ Uint32 firstSSGCI, Uint32 lastSSGCI,
+ GrepError::Code err);
+
+ void sendREP_GET_GCIBUFFER_REF(NdbApiSignal* signal,
+ Uint32 firstGCI, Uint32 lastGCI,
+ Uint32 nodeGrp, GrepError::Code err);
+
+ /***************************************************************************
+ * Private Variables
+ ***************************************************************************/
+ RepState * m_repState;
+
+ struct NdbThread * m_signalExecThread; ///< Signal Queue executor
+ class SignalQueue m_signalRecvQueue;
+
+ ExtSender * m_repSender; ///< Obj responsible send to REP
+
+ Uint32 m_ownNodeId; ///< NodeId of this node
+ Uint32 m_ownBlockNo; ///< BlockNo of this "block"
+ BlockReference m_ownRef; ///< Reference to this
+
+ GCIContainer * m_gciContainer; ///< Ref to gci container.
+
+ TransporterFacade * m_transporterFacade;
+};
+
+#endif
diff --git a/ndb/src/rep/transfer/TransSSSubscriptions.cpp b/ndb/src/rep/transfer/TransSSSubscriptions.cpp
new file mode 100644
index 00000000000..582ba8040a6
--- /dev/null
+++ b/ndb/src/rep/transfer/TransSSSubscriptions.cpp
@@ -0,0 +1,193 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "TransSS.hpp"
+
+#include <signaldata/SumaImpl.hpp>
+#include <GrepError.hpp>
+
+/*****************************************************************************
+ * CREATE SUBSCRIPTION ID
+ *****************************************************************************/
+
+void
+TransSS::execGREP_CREATE_SUBID_CONF(NdbApiSignal* signal)
+{
+ CreateSubscriptionIdConf const * conf =
+ (CreateSubscriptionIdConf *)signal->getDataPtr();
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+
+ /** @todo Fix this */
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionInfo;
+ signal->theData[1] = GrepEvent::GrepSS_CreateSubIdConf;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4 ,JBB);
+#endif
+ m_repState->eventSubscriptionIdCreated(subId, subKey);
+}
+
+void
+TransSS::execGREP_CREATE_SUBID_REF(NdbApiSignal* signal)
+{
+ CreateSubscriptionIdRef const * ref =
+ (CreateSubscriptionIdRef *)signal->getDataPtr();
+ Uint32 subId = ref->subscriptionId;
+ Uint32 subKey = ref->subscriptionKey;
+ GrepError::Code err = (GrepError::Code) ref->err;
+
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionAlert;
+ signal->theData[1] = GrepEvent::GrepSS_CreateSubIdRef;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ signal->theData[4] = err;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 5 ,JBB);
+#endif
+ m_repState->eventSubscriptionIdCreateFailed(subId, subKey, err);
+}
+
+/*****************************************************************************
+ * CREATE SUBSCRIPTION
+ *****************************************************************************/
+
+void
+TransSS::execGREP_SUB_CREATE_CONF(NdbApiSignal* signal)
+{
+ GrepSubCreateConf * const conf = (GrepSubCreateConf *)signal->getDataPtr();
+ Uint32 noOfNodeGroups = conf->noOfNodeGroups;
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+
+ m_repState->setNoOfNodeGroups(noOfNodeGroups);
+
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionInfo;
+ signal->theData[1] = GrepEvent::GrepSS_SubCreateConf;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ signal->theData[4] = noOfNodeGroups;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 5, JBB);
+#endif
+
+ m_repState->eventSubscriptionCreated(subId, subKey);
+}
+
+void
+TransSS::execGREP_SUB_CREATE_REF(NdbApiSignal* signal)
+{
+ GrepSubCreateRef * const ref = (GrepSubCreateRef *)signal->getDataPtr();
+ Uint32 subId = ref->subscriptionId;
+ Uint32 subKey = ref->subscriptionKey;
+ GrepError::Code err = (GrepError::Code)ref->err;
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionAlert;
+ signal->theData[1] = GrepEvent::GrepSS_SubCreateRef;
+ signal->theData[2] = subId;
+ signal->theData[3] = subKey;
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+#endif
+
+ m_repState->eventSubscriptionCreateFailed(subId, subKey, err);
+}
+
+/*****************************************************************************
+ * START SUBSCRIPTION
+ *****************************************************************************/
+
+void
+TransSS::execGREP_SUB_START_CONF(NdbApiSignal* signal)
+{
+ GrepSubStartConf * const conf = (GrepSubStartConf *)signal->getDataPtr();
+ Uint32 subId = conf->subscriptionId;
+ Uint32 subKey = conf->subscriptionKey;
+ SubscriptionData::Part part = (SubscriptionData::Part) conf->part;
+
+ switch(part) {
+ case SubscriptionData::MetaData:
+ RLOG(("Metalog started. Subscription %d-%d", subId, subKey));
+ m_repState->eventMetaLogStarted(signal, subId, subKey);
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionInfo;
+ signal->theData[1] = GrepEvent::GrepSS_SubStartMetaConf;
+ signal->theData[2] = m_requestor.getSubId();
+ signal->theData[3] = m_requestor.getSubKey();
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+#endif
+ break;
+ case SubscriptionData::TableData:
+ RLOG(("Datalog started. Subscription %d-%d", subId, subKey));
+ m_repState->eventDataLogStarted(signal, subId, subKey);
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionInfo;
+ signal->theData[1] = GrepEvent::GrepSS_SubStartDataConf;
+ signal->theData[2] = m_requestor.getSubId();
+ signal->theData[3] = m_requestor.getSubKey();
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+#endif
+ break;
+ default:
+ REPABORT("Illegal type of subscription");
+ }
+}
+
+void
+TransSS::execGREP_SUB_START_REF(NdbApiSignal* signal)
+{
+ GrepSubStartRef * const ref = (GrepSubStartRef *)signal->getDataPtr();
+ Uint32 subId = ref->subscriptionId;
+ Uint32 subKey = ref->subscriptionKey;
+ GrepError::Code err = (GrepError::Code)ref->err;
+ SubscriptionData::Part part = (SubscriptionData::Part) ref->part;
+
+ switch(part) {
+ case SubscriptionData::MetaData:
+ m_repState->eventMetaLogStartFailed(subId, subKey, err);
+#if 1
+ ndbout_c("Requestor: Subscription FAILED to start on Meta Data");
+ ndbout_c("Error code : %d. Error message: %s",
+ err, GrepError::getErrorDesc(err));
+#endif
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionAlert;
+ signal->theData[1] = GrepEvent::GrepSS_SubStartMetaRef;
+ signal->theData[2] = subId; //@todo. manage subscriptions.
+ signal->theData[3] = subKey; //@todo. manage subscriptions.
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+#endif
+ break;
+ case SubscriptionData::TableData:
+ m_repState->eventDataLogStartFailed(subId, subKey, err);
+#if 0
+ signal->theData[0] = EventReport::GrepSubscriptionAlert;
+ signal->theData[1] = GrepEvent::GrepSS_SubStartDataRef;
+ signal->theData[2] = subId; //@todo. manage subscriptions.
+ signal->theData[3] = subKey; //@todo. manage subscriptions.
+ sendSignal(CMVMI_REF, GSN_EVENT_REP, signal, 4, JBB);
+#endif
+#if 1
+ ndbout_c("Requestor: Subscription FAILED to start on Table Data");
+#endif
+ ndbout_c("Error code : %d. Error message: %s",
+ err, GrepError::getErrorDesc(err));
+
+ break;
+ default:
+ REPABORT("Illegal type of subscription");
+ }
+}
diff --git a/ndb/src/scripts/Makefile b/ndb/src/scripts/Makefile
new file mode 100644
index 00000000000..bc8049ac34b
--- /dev/null
+++ b/ndb/src/scripts/Makefile
@@ -0,0 +1,5 @@
+include .defs.mk
+
+DIRS :=
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/Makefile b/ndb/test/Makefile
new file mode 100644
index 00000000000..19472917560
--- /dev/null
+++ b/ndb/test/Makefile
@@ -0,0 +1,19 @@
+include .defs.mk
+
+DIRS := src tools ndbapi run-test
+
+EXTRA_DIRS = newtonapi
+
+ifeq ($(NDB_ARCH), x86_64)
+EXTRA_DIRS =
+endif
+
+DIRS += $(EXTRA_DIRS)
+
+ifneq ($(NDB_ODBC),N)
+DIRS += odbc
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+_bins_ndbapi : _libs_src
diff --git a/ndb/test/include/HugoAsynchTransactions.hpp b/ndb/test/include/HugoAsynchTransactions.hpp
new file mode 100644
index 00000000000..d7e6e8fc187
--- /dev/null
+++ b/ndb/test/include/HugoAsynchTransactions.hpp
@@ -0,0 +1,75 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef HUGO_ASYNCH_TRANSACTIONS_HPP
+#define HUGO_ASYNCH_TRANSACTIONS_HPP
+
+
+#include <NDBT.hpp>
+#include <HugoCalculator.hpp>
+#include <HugoTransactions.hpp>
+
+class HugoAsynchTransactions : private HugoTransactions {
+public:
+ HugoAsynchTransactions(const NdbDictionary::Table&);
+ ~HugoAsynchTransactions();
+ int loadTableAsynch(Ndb*,
+ int records = 0,
+ int batch = 1,
+ int trans = 1,
+ int operations = 1);
+ int pkReadRecordsAsynch(Ndb*,
+ int records = 0,
+ int batch= 1,
+ int trans = 1,
+ int operations = 1);
+ int pkUpdateRecordsAsynch(Ndb*,
+ int records = 0,
+ int batch= 1,
+ int trans = 1,
+ int operations = 1);
+ int pkDelRecordsAsynch(Ndb*,
+ int records = 0,
+ int batch = 1,
+ int trans = 1,
+ int operations = 1);
+ void transactionCompleted();
+
+ long getTransactionsCompleted();
+
+private:
+ enum NDB_OPERATION {NO_INSERT, NO_UPDATE, NO_READ, NO_DELETE};
+
+ void allocTransactions(int trans);
+ void deallocTransactions();
+
+ int executeAsynchOperation(Ndb*,
+ int records,
+ int batch,
+ int trans,
+ int operations,
+ NDB_OPERATION theOperation,
+ ExecType theType = Commit);
+
+ long transactionsCompleted;
+ int numTransactions;
+ NdbConnection** transactions;
+};
+
+
+
+#endif
+
diff --git a/ndb/test/include/HugoCalculator.hpp b/ndb/test/include/HugoCalculator.hpp
new file mode 100644
index 00000000000..b782eb003a3
--- /dev/null
+++ b/ndb/test/include/HugoCalculator.hpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_CALC_HPP
+#define NDBT_CALC_HPP
+
+#include <NDBT_ResultRow.hpp>
+
+/* *************************************************************
+ * HugoCalculator
+ *
+ * Comon class for the Hugo test suite, provides the functions
+ * that is used for calculating values to load in to table and
+ * also knows how to verify a row that's been read from db
+ *
+ * ************************************************************/
+class HugoCalculator {
+public:
+ HugoCalculator(const NdbDictionary::Table& tab);
+ Int32 calcValue(int record, int attrib, int updates) const;
+#if 0
+ U_Int32 calcValue(int record, int attrib, int updates) const;
+ U_Int64 calcValue(int record, int attrib, int updates) const;
+ Int64 calcValue(int record, int attrib, int updates) const;
+ float calcValue(int record, int attrib, int updates) const;
+ double calcValue(int record, int attrib, int updates) const;
+#endif
+ const char* calcValue(int record, int attrib, int updates, char* buf) const;
+
+ int verifyRowValues(NDBT_ResultRow* const pRow) const;
+ int getIdValue(NDBT_ResultRow* const pRow) const;
+ int getUpdatesValue(NDBT_ResultRow* const pRow) const;
+ int isIdCol(int colId) { return m_idCol == colId; };
+ int isUpdateCol(int colId){ return m_updatesCol == colId; };
+private:
+ const NdbDictionary::Table& m_tab;
+ int m_idCol;
+ int m_updatesCol;
+};
+
+
+#endif
+
+
+
+
diff --git a/ndb/test/include/HugoOperations.hpp b/ndb/test/include/HugoOperations.hpp
new file mode 100644
index 00000000000..7295b72b18f
--- /dev/null
+++ b/ndb/test/include/HugoOperations.hpp
@@ -0,0 +1,149 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef HUGO_OPERATIONS_HPP
+#define HUGO_OPERATIONS_HPP
+
+#include <NDBT.hpp>
+#include <HugoCalculator.hpp>
+#include <UtilTransactions.hpp>
+#include <Vector.hpp>
+
+class HugoOperations : public UtilTransactions {
+public:
+ HugoOperations(const NdbDictionary::Table&);
+ ~HugoOperations();
+ int startTransaction(Ndb*);
+ int closeTransaction(Ndb*);
+ NdbConnection* getTransaction();
+ void refresh();
+
+ int pkInsertRecord(Ndb*,
+ int recordNo,
+ int numRecords = 1,
+ int updatesValue = 0);
+
+ int pkReadRecord(Ndb*,
+ int recordNo,
+ bool exclusive = false,
+ int numRecords = 1);
+
+ int pkSimpleReadRecord(Ndb*,
+ int recordNo,
+ int numRecords = 1);
+
+ int pkDirtyReadRecord(Ndb*,
+ int recordNo,
+ int numRecords = 1);
+
+ int pkUpdateRecord(Ndb*,
+ int recordNo,
+ int numRecords = 1,
+ int updatesValue = 0);
+
+ int pkDeleteRecord(Ndb*,
+ int recordNo,
+ int numRecords = 1);
+
+ int scanReadRecords(Ndb* pNdb,
+ Uint32 parallelism = 240, ScanLock lock = SL_Read);
+ int executeScanRead(Ndb*);
+
+ int execute_Commit(Ndb*,
+ AbortOption ao = AbortOnError);
+ int execute_NoCommit(Ndb*,
+ AbortOption ao = AbortOnError);
+ int execute_Rollback(Ndb*);
+
+ int saveCopyOfRecord(int numRecords = 1);
+ int compareRecordToCopy(int numRecords = 1);
+
+ BaseString getRecordStr(int recordNum);
+ int getRecordGci(int recordNum);
+
+ int setValueForAttr(NdbOperation*,
+ int attrId,
+ int rowId,
+ int updateId);
+ int equalForAttr(NdbOperation*,
+ int attrId,
+ int rowId);
+
+ int verifyUpdatesValue(int updatesValue, int _numRows = 0);
+
+ int indexReadRecords(Ndb*, const char * idxName, int recordNo,
+ bool exclusive = false,
+ int records = 1);
+
+ int indexUpdateRecord(Ndb*,
+ const char * idxName,
+ int recordNo,
+ int numRecords = 1,
+ int updatesValue = 0);
+
+protected:
+ void allocRows(int rows);
+ void deallocRows();
+
+ Vector<NDBT_ResultRow*> rows;
+ HugoCalculator calc;
+
+ Vector<BaseString> savedRecords;
+private:
+ NdbConnection* pTrans;
+
+ struct ScanTmp {
+ ScanTmp() {
+ pTrans = 0;
+ m_tmpRow = 0;
+ m_delete = true;
+ m_op = DONE;
+ }
+ ScanTmp(NdbConnection* a, NDBT_ResultRow* b){
+ pTrans = a;
+ m_tmpRow = b;
+ m_delete = true;
+ m_op = DONE;
+ }
+ ScanTmp(const ScanTmp& org){
+ * this = org;
+ }
+ ScanTmp& operator=(const ScanTmp& org){
+ pTrans = org.pTrans;
+ m_tmpRow = org.m_tmpRow;
+ m_delete = org.m_delete;
+ m_op = org.m_op;
+ return * this;
+ }
+
+ ~ScanTmp() {
+ if(m_delete && pTrans)
+ pTrans->close();
+ if(m_delete && m_tmpRow)
+ delete m_tmpRow;
+ }
+
+ NdbConnection * pTrans;
+ NDBT_ResultRow * m_tmpRow;
+ bool m_delete;
+ enum { DONE, READ, UPDATE, DELETE } m_op;
+ };
+ Vector<ScanTmp> m_scans;
+ int run(ScanTmp & tmp);
+
+};
+
+#endif
diff --git a/ndb/test/include/HugoTransactions.hpp b/ndb/test/include/HugoTransactions.hpp
new file mode 100644
index 00000000000..5ff1fef16bc
--- /dev/null
+++ b/ndb/test/include/HugoTransactions.hpp
@@ -0,0 +1,129 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef HUGO_TRANSACTIONS_HPP
+#define HUGO_TRANSACTIONS_HPP
+
+
+#include <NDBT.hpp>
+#include <HugoCalculator.hpp>
+#include <HugoOperations.hpp>
+
+
+class HugoTransactions : public HugoOperations {
+public:
+ HugoTransactions(const NdbDictionary::Table&);
+ ~HugoTransactions();
+ int createEvent(Ndb*);
+ int eventOperation(Ndb*, void* stats,
+ int records);
+ int loadTable(Ndb*,
+ int records,
+ int batch = 512,
+ bool allowConstraintViolation = true,
+ int doSleep = 0);
+ int scanReadRecords(Ndb*,
+ int records,
+ int abort = 0,
+ int parallelism = 1,
+ bool committed = false);
+ int scanReadCommittedRecords(Ndb*,
+ int records,
+ int abort = 0,
+ int parallelism = 1);
+ int pkReadRecords(Ndb*,
+ int records,
+ int batchsize = 1,
+ bool dirty = false);
+
+ int scanUpdateRecords(Ndb*,
+ int records,
+ int abort = 0,
+ int parallelism = 1);
+
+ int scanUpdateRecords1(Ndb*,
+ int records,
+ int abort = 0,
+ int parallelism = 1);
+ int scanUpdateRecords2(Ndb*,
+ int records,
+ int abort = 0,
+ int parallelism = 1);
+ int scanUpdateRecords3(Ndb*,
+ int records,
+ int abort = 0,
+ int parallelism = 1);
+
+ int pkUpdateRecords(Ndb*,
+ int records,
+ int batchsize = 1,
+ int doSleep = 0);
+ int pkInterpretedUpdateRecords(Ndb*,
+ int records,
+ int batchsize = 1);
+ int pkDelRecords(Ndb*,
+ int records = 0,
+ int batch = 1,
+ bool allowConstraintViolation = true,
+ int doSleep = 0);
+ int lockRecords(Ndb*,
+ int records,
+ int percentToLock = 1,
+ int lockTime = 1000);
+ int fillTable(Ndb*,
+ int batch=512);
+
+ /**
+ * Reading using UniqHashIndex with key = pk
+ */
+ int indexReadRecords(Ndb*,
+ const char * idxName,
+ int records,
+ int batchsize = 1);
+
+ int indexUpdateRecords(Ndb*,
+ const char * idxName,
+ int records,
+ int batchsize = 1);
+
+protected:
+ int takeOverAndUpdateRecord(Ndb*,
+ NdbOperation*);
+#if 0
+ int setValueForAttr(NdbOperation*,
+ int attrId,
+ int rowId,
+ int updateId);
+public:
+ int equalForAttr(NdbOperation*,
+ int attrId,
+ int rowId);
+#endif
+
+ int addRowToUpdate(Ndb* pNdb,
+ NdbConnection* pUpdTrans,
+ NdbOperation* pOrgOp);
+
+
+ NDBT_ResultRow row;
+ int m_defaultScanUpdateMethod;
+};
+
+
+
+
+#endif
+
diff --git a/ndb/test/include/NDBT.hpp b/ndb/test/include/NDBT.hpp
new file mode 100644
index 00000000000..657a9cb03b6
--- /dev/null
+++ b/ndb/test/include/NDBT.hpp
@@ -0,0 +1,39 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_TEST_HPP
+#define NDB_TEST_HPP
+
+/**
+ * NdbTest.hpp
+ * This is the main include file to include in test programs
+ * It will include all the other include files in the NDBT-toolkit
+ *
+ */
+
+#include "NDBT_ReturnCodes.h"
+
+#ifdef __cplusplus
+#include "NDBT_Table.hpp"
+#include "NDBT_Tables.hpp"
+#include "NDBT_Error.hpp"
+#include "NDBT_ResultRow.hpp"
+#include "NDBT_Output.hpp"
+
+#endif
+
+
+#endif
diff --git a/ndb/test/include/NDBT_DataSet.hpp b/ndb/test/include/NDBT_DataSet.hpp
new file mode 100644
index 00000000000..1a0122f617c
--- /dev/null
+++ b/ndb/test/include/NDBT_DataSet.hpp
@@ -0,0 +1,140 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_DATA_SET_HPP
+#define NDBT_DATA_SET_HPP
+
+#include "NDBT_Table.hpp"
+#include <NdbApi.hpp>
+
+class NDBT_DataSet;
+
+class NDBT_DataSetFactory {
+public:
+ NDBT_DataSet * createEmpty(const NDBT_Table & table,
+ const char * columns[]);
+
+ NDBT_DataSet * createRandom(const NDBT_DataSet & table,
+ const char * columns[],
+ int rows);
+
+ NDBT_DataSet * createXXX(int noOfDS, const NDBT_DataSet ** datasets);
+};
+
+class NDBT_DataSet {
+ friend class NDBT_DataSetFactory;
+public:
+ /**
+ * Rows in the data set
+ */
+ void setRows(int rows);
+ void addRows(int rows);
+
+ int getNoOfRows() const;
+
+ /**
+ * Columns for a row in the data set
+ */
+ int getNoOfColumns() const;
+ int getNoOfPKs() const;
+
+ const NDBT_Attribute * getColumn(int index);
+ const NDBT_Attribute * getColumn(const char * name);
+
+ /**
+ * Data status in dataset
+ */
+ bool hasPK(int row);
+ bool hasData(int row);
+
+ /**
+ * Do all rows in the dataset have a PK
+ */
+ bool hasPK();
+
+ /**
+ * Do all rows in the dataset has data
+ */
+ bool hasData();
+
+ /**
+ * Getters for data
+ */
+ Uint32 getUInt(int row, int index) const;
+ Uint32 getUInt(int row, const char * attribute) const;
+
+ Int32 getInt(int row, int index) const;
+ Int32 getInt(int row, const char * attribute) const;
+
+ const char * getString(int row, int index) const;
+ const char * getString(int row, const char * attribute) const;
+
+ bool getIsNull(int row, int index) const;
+ bool getIsNull(int row, const char * attribute) const;
+
+ /**
+ * Setters for data
+ */
+ void set(int row, int index, Int32 value);
+ void set(int row, const char * attr, Int32 value);
+
+ void set(int row, int index, Uint32 value);
+ void set(int row, const char * attr, Uint32 value);
+
+ void set(int row, int index, const char * value);
+ void set(int row, const char * attr, const char * value);
+
+ /**
+ * Comparators
+ */
+
+ /**
+ * Is this dataset identical to other dataset
+ *
+ * If either of the datasets have "undefined" rows the answer is false
+ */
+ bool equal(const NDBT_DataSet & other) const;
+
+ /**
+ * Do these dataset have identical PK's
+ *
+ * I.e noOfRows equal
+ *
+ * and for each row there is a corresponding row in the other ds
+ * with the same pk
+ */
+ bool equalPK(const NDBT_DataSet & other) const;
+
+private:
+ NDBT_Attribute * columns;
+
+ Uint32 noOfRows;
+ Uint32 noOfPKs;
+
+ Uint32 * pks;
+ Uint32 * columnSizes;
+
+ char * pkData;
+ char * data;
+
+ char * pk(int row, int pkIndex);
+ char * column(int row, int columnIndex);
+
+ Uint32 * hasPK;
+ Uint32 * hasData;
+};
+
+#endif
diff --git a/ndb/test/include/NDBT_DataSetTransaction.hpp b/ndb/test/include/NDBT_DataSetTransaction.hpp
new file mode 100644
index 00000000000..9f250c566dd
--- /dev/null
+++ b/ndb/test/include/NDBT_DataSetTransaction.hpp
@@ -0,0 +1,162 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_DATA_SET_TRANSACTION_HPP
+#define NDBT_DATA_SET_TRANSACTION_HPP
+
+#include "NDBT_Table.hpp"
+#include <NdbApi.hpp>
+
+class NDBT_DataSet;
+
+/**
+ * This class contains a bunch a methods
+ * that operates on NDB together with a NDBT_DataSet
+ * using <b>one</b> transaction
+ */
+class NDBT_DataSetTransaction {
+public:
+ /**
+ * Store the data into ndb
+ */
+ static void insert(Ndb * theNdbObject,
+ const NDBT_DataSet *,
+ bool rollback = false);
+
+ /**
+ * Read data (using pk) from ndb
+ */
+ static void readByPk(Ndb * theNdbObject,
+ NDBT_DataSet *,
+ bool rollback = false);
+
+ /**
+ * Update data using pk
+ *
+ */
+ static void updateByPk(Ndb * theNdbObject,
+ const NDBT_DataSet *,
+ bool rollback = false);
+
+ /**
+ * Delete
+ */
+ static void deleteByPk(Ndb * theNdbObject,
+ const NDBT_DataSet *,
+ bool rollback = false);
+};
+
+class NDBT_DataSetAsyncTransaction {
+public:
+ enum OperationType {
+ OT_Insert,
+ OT_ReadByPk,
+ OT_UpdateByPk,
+ OT_DeleteByPk
+ };
+
+ /**
+ * A callback for the NDBT_DataSetAsyncTransaction
+ * interface.
+ *
+ * The callback method returns:
+ * - the operation performed
+ * - the data set
+ * - if the transaction was commited or aborted
+ */
+ typedef (* NDBT_DataSetAsyncTransactionCallback)(OperationType,
+ const NDBT_DataSet *,
+ bool commit,
+ void * anyObject);
+ /**
+ * Store the data into ndb
+ */
+ static void insert(Ndb * theNdbObject,
+ const NDBT_DataSet *,
+ bool rollback = false,
+ NDBT_DataSetAsyncTransactionCallback fun,
+ void * anyObject);
+
+ /**
+ * Read data (using pk) from ndb
+ */
+ static void readByPk(Ndb * theNdbObject,
+ NDBT_DataSet *,
+ bool rollback = false,
+ NDBT_DataSetAsyncTransactionCallback fun,
+ void * anyObject);
+
+
+ /**
+ * Update data using pk
+ *
+ */
+ static void updateByPk(Ndb * theNdbObject,
+ const NDBT_DataSet *,
+ bool rollback = false,
+ NDBT_DataSetAsyncTransactionCallback fun,
+ void * anyObject);
+
+
+ /**
+ * Delete
+ */
+ static void deleteByPk(Ndb * theNdbObject,
+ const NDBT_DataSet *,
+ bool rollback = false,
+ NDBT_DataSetAsyncTransactionCallback fun,
+ void * anyObject);
+
+};
+
+class NDBT_DataSetBulkOperation {
+public:
+ /**
+ * Store the data into ndb
+ */
+ static void insert(Ndb * theNdbObject,
+ const NDBT_DataSet *,
+ int parallellTransactions = 1,
+ int operationsPerTransaction = 10);
+
+ /**
+ * Read data (using pk) from ndb
+ */
+ static void readByPk(Ndb * theNdbObject,
+ NDBT_DataSet *,
+ int parallellTransactions = 1,
+ int operationsPerTransaction = 10);
+
+ /**
+ * Update data using pk
+ *
+ */
+ static void updateByPk(Ndb * theNdbObject,
+ const NDBT_DataSet *,
+ int parallellTransactions = 1,
+ int operationsPerTransaction = 10);
+
+ /**
+ * Delete
+ */
+ static void deleteByPk(Ndb * theNdbObject,
+ const NDBT_DataSet *,
+ int parallellTransactions = 1,
+ int operationsPerTransaction = 10);
+};
+
+
+#endif
diff --git a/ndb/test/include/NDBT_Error.hpp b/ndb/test/include/NDBT_Error.hpp
new file mode 100644
index 00000000000..ef107072465
--- /dev/null
+++ b/ndb/test/include/NDBT_Error.hpp
@@ -0,0 +1,97 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_Error_HPP
+#define NDBT_Error_HPP
+
+#include <NdbOut.hpp>
+#include <NdbError.hpp>
+
+/**
+ * NDBT_Error.hpp
+ * This is the main include file about error handling in NDBT test programs
+ *
+ */
+class ErrorData {
+
+public:
+ ErrorData();
+ ~ErrorData();
+
+ /**
+ * Parse cmd line arg
+ *
+ * Return true if successeful
+ */
+ bool parseCmdLineArg(const char ** argv, int & i);
+
+ /**
+ * Print cmd line arguments
+ */
+ void printCmdLineArgs(NdbOut & out = ndbout);
+
+ /**
+ * Print settings
+ */
+ void printSettings(NdbOut & out = ndbout);
+
+ /**
+ * Print error count
+ */
+ void printErrorCounters(NdbOut & out = ndbout) const;
+
+ /**
+ * Reset error counters
+ */
+ void resetErrorCounters();
+
+ /**
+ *
+ */
+ int handleErrorCommon(const NdbError & error);
+
+private:
+ bool key_error;
+ bool temporary_resource_error;
+ bool insufficient_space_error;
+ bool node_recovery_error;
+ bool overload_error;
+ bool timeout_error;
+ bool internal_error;
+ bool user_error;
+ bool application_error;
+
+ Uint32 * errorCountArray;
+};
+
+//
+// ERR prints an NdbError object togheter with a description of where
+// the error occured
+//
+#define ERR_OUT(where, error) \
+ { where << "ERROR: " << error.code << " " \
+ << error.message << endl \
+ << " " << "Status: " << error.status \
+ << ", Classification: " << error.classification << endl\
+ << " " << "File: " << __FILE__ \
+ << " (Line: " << __LINE__ << ")" << endl \
+ ; \
+ }
+
+#define ERR(error) ERR_OUT(g_err, error)
+#define ERR_INFO(error) ERR_OUT(g_info, error)
+
+#endif
diff --git a/ndb/test/include/NDBT_Output.hpp b/ndb/test/include/NDBT_Output.hpp
new file mode 100644
index 00000000000..aaa619ac479
--- /dev/null
+++ b/ndb/test/include/NDBT_Output.hpp
@@ -0,0 +1,30 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_Output_HPP
+#define NDBT_Output_HPP
+
+#include <OutputStream.hpp>
+#include <NdbOut.hpp>
+
+void setOutputLevel(int i);
+
+extern FilteredNdbOut g_err;
+extern FilteredNdbOut g_warning;
+extern FilteredNdbOut g_info;
+extern FilteredNdbOut g_debug;
+
+#endif
diff --git a/ndb/test/include/NDBT_ResultRow.hpp b/ndb/test/include/NDBT_ResultRow.hpp
new file mode 100644
index 00000000000..aa54e892da3
--- /dev/null
+++ b/ndb/test/include/NDBT_ResultRow.hpp
@@ -0,0 +1,55 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_RESULTROW_HPP
+#define NDBT_RESULTROW_HPP
+
+#include <NdbApi.hpp>
+
+class NDBT_ResultRow {
+public:
+ NDBT_ResultRow(const NdbDictionary::Table &tab, char attrib_delimiter='\t');
+ ~NDBT_ResultRow();
+ NdbRecAttr * & attributeStore(int i);
+ const NdbRecAttr * attributeStore(const char* name);
+
+ BaseString c_str();
+
+ NdbOut & header (NdbOut &) const;
+ friend NdbOut & operator << (NdbOut&, const NDBT_ResultRow &);
+
+ /**
+ * Make copy of NDBT_ResultRow
+ */
+ NDBT_ResultRow * clone() const;
+
+private:
+ int cols;
+ char **names;
+ NdbRecAttr **data;
+ char ad[2];
+
+ bool m_ownData;
+ const NdbDictionary::Table & m_table;
+
+ NDBT_ResultRow(const NDBT_ResultRow &);
+ NDBT_ResultRow& operator=(const NDBT_ResultRow &);
+};
+
+
+
+
+#endif
diff --git a/ndb/test/include/NDBT_ReturnCodes.h b/ndb/test/include/NDBT_ReturnCodes.h
new file mode 100644
index 00000000000..0bc71ad8ceb
--- /dev/null
+++ b/ndb/test/include/NDBT_ReturnCodes.h
@@ -0,0 +1,42 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_RETURNCODES_H
+#define NDBT_RETURNCODES_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#define NDBT_OK 0
+#define NDBT_FAILED 1
+#define NDBT_WRONGARGS 2
+#define NDBT_TEMPORARY 3
+/**
+ * NDBT_ProgramExit
+ * This function will print the returncode togheter with a prefix on
+ * the screen and then exit the test program.
+ * Call this function when exiting the main function in your test programs
+ * Returns the return code
+ */
+int NDBT_ProgramExit(int rcode);
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/ndb/test/include/NDBT_Stats.hpp b/ndb/test/include/NDBT_Stats.hpp
new file mode 100644
index 00000000000..15a125dea86
--- /dev/null
+++ b/ndb/test/include/NDBT_Stats.hpp
@@ -0,0 +1,76 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_STATS_HPP
+#define NDBT_STATS_HPP
+
+#include <math.h>
+#include <float.h>
+#include <assert.h>
+
+class NDBT_Stats {
+public:
+ NDBT_Stats() { reset(); }
+
+ void reset() { sum = sum2 = 0.0; max = DBL_MIN; ; min = DBL_MAX; n = 0;}
+
+ void addObservation(double t) {
+ sum+= t;
+ sum2 += (t*t);
+ n++;
+ if(min > t) min = t;
+ if(max < t) max = t;
+ }
+
+ double getMean() const { return sum/n;}
+ double getStddev() const { return sqrt(getVariance()); }
+ double getVariance() const { return (n*sum2 - (sum*sum))/(n*n);}
+ double getMin() const { return min;}
+ double getMax() const { return max;}
+ int getCount() const { return n;}
+
+ NDBT_Stats & operator+=(const NDBT_Stats & c){
+ sum += c.sum;
+ sum2 += c.sum2;
+ n += c.n;
+ if(min > c.min) min = c.min;
+ if(max < c.max) max = c.max;
+ return * this;
+ }
+private:
+ double sum;
+ double sum2;
+ int n;
+ double min, max;
+};
+
+inline
+double
+NDB_SQRT(double x){
+ assert(x >= 0);
+
+ double y = 0;
+ double s = 1;
+ double r = 0;
+ while(y <= x){
+ y += s;
+ s += 2;
+ r += 1;
+ }
+ return r - 1;
+}
+
+#endif
diff --git a/ndb/test/include/NDBT_Table.hpp b/ndb/test/include/NDBT_Table.hpp
new file mode 100644
index 00000000000..a4482fa8084
--- /dev/null
+++ b/ndb/test/include/NDBT_Table.hpp
@@ -0,0 +1,135 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_TABLE_HPP
+#define NDBT_TABLE_HPP
+
+#include <NdbApi.hpp>
+#include <NdbOut.hpp>
+
+#include <string.h>
+
+class NDBT_Attribute : public NdbDictionary::Column {
+ friend class NdbOut& operator <<(class NdbOut&, const NDBT_Attribute &);
+public:
+ NDBT_Attribute(const char* anAttrName,
+ AttrType type,
+ int sz = 4,
+ KeyType key = NoKey,
+ bool nullable = false,
+ StorageAttributeType indexOnly = NormalStorageAttribute,
+ StorageMode _sm = MMBased) :
+ NdbDictionary::Column(anAttrName)
+ {
+ assert(anAttrName != 0);
+
+ setNullable(nullable);
+ setIndexOnlyStorage(indexOnly == IndexStorageAttribute);
+ setPrimaryKey(key != NoKey);
+ setTupleKey(key == TupleId);
+ setLength(1);
+ switch(type){
+ case ::Signed:
+ if(sz == 8)
+ setType(NdbDictionary::Column::Bigint);
+ else if (sz == 4)
+ setType(NdbDictionary::Column::Int);
+ else {
+ setType(NdbDictionary::Column::Int);
+ setLength(sz);
+ }
+ break;
+
+ case ::UnSigned:
+ if(sz == 8)
+ setType(NdbDictionary::Column::Bigunsigned);
+ else if (sz == 4)
+ setType(NdbDictionary::Column::Unsigned);
+ else {
+ setType(NdbDictionary::Column::Unsigned);
+ setLength(sz);
+ }
+ break;
+
+ case ::Float:
+ if(sz == 8)
+ setType(NdbDictionary::Column::Double);
+ else if (sz == 4)
+ setType(NdbDictionary::Column::Float);
+ else{
+ setType(NdbDictionary::Column::Float);
+ setLength(sz);
+ }
+ break;
+
+ case ::String:
+ setType(NdbDictionary::Column::Char);
+ setLength(sz);
+ break;
+
+ case ::NoAttrTypeDef:
+ break;
+ }
+ }
+
+ NDBT_Attribute(const char* _name,
+ Column::Type _type,
+ int _length = 1,
+ bool _pk = false,
+ bool _nullable = false):
+ NdbDictionary::Column(_name)
+ {
+ assert(_name != 0);
+
+ setNullable(_nullable);
+ setPrimaryKey(_pk);
+ setLength(_length);
+ setType(_type);
+ }
+};
+
+class NDBT_Table : public NdbDictionary::Table {
+ /**
+ * Print meta information about table
+ * (information on how it is strored, what the attributes look like etc.)
+ */
+ friend class NdbOut& operator <<(class NdbOut&, const NDBT_Table &);
+public:
+
+ NDBT_Table(const char* name,
+ int noOfAttributes,
+ const NdbDictionary::Column attributes[],
+ bool stored = true)
+ : NdbDictionary::Table(name)
+ {
+ assert(name != 0);
+
+ setStoredTable(stored);
+ for(int i = 0; i<noOfAttributes; i++)
+ addColumn(attributes[i]);
+ }
+
+ static const NdbDictionary::Table * discoverTableFromDb(Ndb* ndb,
+ const char * name);
+};
+
+inline
+const NdbDictionary::Table *
+NDBT_Table::discoverTableFromDb(Ndb* ndb, const char * name){
+ return ndb->getDictionary()->getTable(name);
+}
+
+#endif
diff --git a/ndb/test/include/NDBT_Tables.hpp b/ndb/test/include/NDBT_Tables.hpp
new file mode 100644
index 00000000000..1da9818ee70
--- /dev/null
+++ b/ndb/test/include/NDBT_Tables.hpp
@@ -0,0 +1,47 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_TABLES_HPP
+#define NDBT_TABLES_HPP
+
+
+#include <NDBT.hpp>
+#include <Ndb.hpp>
+#include <NdbDictionary.hpp>
+#include <NDBT_Table.hpp>
+
+class NDBT_Tables {
+public:
+
+ static int createTable(Ndb* pNdb, const char* _name, bool _temp = false);
+ static int createAllTables(Ndb* pNdb, bool _temp, bool existsOK = false);
+ static int createAllTables(Ndb* pNdb);
+
+ static int dropAllTables(Ndb* pNdb);
+
+ static int print(const char * name);
+ static int printAll();
+
+ static const NdbDictionary::Table* getTable(const char* _nam);
+ static const NdbDictionary::Table* getTable(int _num);
+ static int getNumTables();
+
+private:
+ static const NdbDictionary::Table* tableWithPkSize(const char* _nam, Uint32 pkSize);
+};
+#endif
+
+
diff --git a/ndb/test/include/NDBT_Test.hpp b/ndb/test/include/NDBT_Test.hpp
new file mode 100644
index 00000000000..41332bb570c
--- /dev/null
+++ b/ndb/test/include/NDBT_Test.hpp
@@ -0,0 +1,417 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_TEST_HPP
+#define NDBT_TEST_HPP
+
+
+#include "NDBT_ReturnCodes.h"
+#include <Properties.hpp>
+#include <NdbThread.h>
+#include <NdbSleep.h>
+#include <NdbCondition.h>
+#include <NdbTimer.hpp>
+#include <assert.h>
+#include <Vector.hpp>
+#include <NdbDictionary.hpp>
+
+class NDBT_Step;
+class NDBT_TestCase;
+class NDBT_TestSuite;
+class NDBT_TestCaseImpl1;
+
+class NDBT_Context {
+public:
+ NDBT_Context();
+ ~NDBT_Context();
+ const NdbDictionary::Table* getTab();
+ NDBT_TestSuite* getSuite();
+ NDBT_TestCase* getCase();
+
+ // Get arguments
+ int getNumRecords() const;
+ int getNumLoops() const;
+ char * getRemoteMgm() const;
+ // Common place to store state between
+ // steps, for example information from one step to the
+ // verifier about how many records have been inserted
+ Uint32 getProperty(const char*, Uint32 = 0 );
+ const char* getProperty(const char*, const char* );
+ void setProperty(const char*, Uint32);
+ void setProperty(const char*, const char*);
+
+ // Signal that a property value that another
+ // thread might be waiting for has changed
+ void broadcast();
+ // Wait for the signal that a property has changed
+ void wait();
+ void wait_timeout(int msec);
+
+ // Wait until the property has been set to a certain value
+ bool getPropertyWait(const char*, Uint32);
+ const char* getPropertyWait(const char*, const char* );
+
+ // Communicate with other tests
+ void stopTest();
+ bool isTestStopped();
+
+ // Communicate with tests in other API nodes
+ // This is done using a "system" table in the database
+ Uint32 getDbProperty(const char*);
+ bool setDbProperty(const char*, Uint32);
+
+ void setTab(const NdbDictionary::Table*);
+ void setRemoteMgm(char * mgm);
+
+ /**
+ * Get no of steps running/completed
+ */
+ int getNoOfRunningSteps() const ;
+ int getNoOfCompletedSteps() const ;
+private:
+ friend class NDBT_Step;
+ friend class NDBT_TestSuite;
+ friend class NDBT_TestCase;
+ friend class NDBT_TestCaseImpl1;
+
+ void setSuite(NDBT_TestSuite*);
+ void setCase(NDBT_TestCase*);
+ void setNumRecords(int);
+ void setNumLoops(int);
+ const NdbDictionary::Table* tab;
+ NDBT_TestSuite* suite;
+ NDBT_TestCase* testcase;
+ Ndb* ndb;
+ int records;
+ int loops;
+ bool stopped;
+ char * remote_mgm;
+ Properties props;
+ NdbMutex* propertyMutexPtr;
+ NdbCondition* propertyCondPtr;
+};
+
+typedef int (NDBT_TESTFUNC)(NDBT_Context*, NDBT_Step*);
+
+class NDBT_Step {
+public:
+ NDBT_Step(NDBT_TestCase* ptest,
+ const char* pname,
+ NDBT_TESTFUNC* pfunc);
+ int execute(NDBT_Context*);
+ virtual int setUp() = 0;
+ virtual void tearDown() = 0;
+ void setContext(NDBT_Context*);
+ NDBT_Context* getContext();
+ void print();
+ const char* getName() { return name; }
+ int getStepNo() { return step_no; }
+ void setStepNo(int n) { step_no = n; }
+protected:
+ NDBT_Context* m_ctx;
+ const char* name;
+ NDBT_TESTFUNC* func;
+ NDBT_TestCase* testcase;
+ int step_no;
+};
+
+class NDBT_NdbApiStep : public NDBT_Step {
+public:
+ NDBT_NdbApiStep(NDBT_TestCase* ptest,
+ const char* pname,
+ NDBT_TESTFUNC* pfunc);
+ int setUp();
+ void tearDown();
+
+ Ndb* getNdb();
+protected:
+ Ndb* ndb;
+};
+
+class NDBT_ParallelStep : public NDBT_NdbApiStep {
+public:
+ NDBT_ParallelStep(NDBT_TestCase* ptest,
+ const char* pname,
+ NDBT_TESTFUNC* pfunc);
+};
+
+class NDBT_Verifier : public NDBT_NdbApiStep {
+public:
+ NDBT_Verifier(NDBT_TestCase* ptest,
+ const char* name,
+ NDBT_TESTFUNC* func);
+};
+
+class NDBT_Initializer : public NDBT_NdbApiStep {
+public:
+ NDBT_Initializer(NDBT_TestCase* ptest,
+ const char* name,
+ NDBT_TESTFUNC* func);
+};
+
+class NDBT_Finalizer : public NDBT_NdbApiStep {
+public:
+ NDBT_Finalizer(NDBT_TestCase* ptest,
+ const char* name,
+ NDBT_TESTFUNC* func);
+};
+
+
+class NDBT_TestCase {
+public:
+ NDBT_TestCase(NDBT_TestSuite* psuite,
+ const char* name,
+ const char* comment);
+ virtual ~NDBT_TestCase(){};
+ // This is the default executor of a test case
+ // When a test case is executed it will need to be suplied with a number of
+ // different parameters and settings, these are passed to the test in the
+ // NDBT_Context object
+ virtual int execute(NDBT_Context*);
+ void setProperty(const char*, Uint32);
+ void setProperty(const char*, const char*);
+ virtual void print() = 0;
+ virtual void printHTML() = 0;
+
+ const char* getName(){return name;};
+ virtual bool tableExists(NdbDictionary::Table* aTable) = 0;
+ virtual bool isVerify(const NdbDictionary::Table* aTable) = 0;
+
+ virtual void saveTestResult(const NdbDictionary::Table* ptab, int result) = 0;
+ virtual void printTestResult() = 0;
+ void initBeforeTest(){ timer.doReset();};
+
+ /**
+ * Get no of steps running/completed
+ */
+ virtual int getNoOfRunningSteps() const = 0;
+ virtual int getNoOfCompletedSteps() const = 0;
+
+protected:
+ virtual int runInit(NDBT_Context* ctx) = 0;
+ virtual int runSteps(NDBT_Context* ctx) = 0;
+ virtual int runVerifier(NDBT_Context* ctx) = 0;
+ virtual int runFinal(NDBT_Context* ctx) = 0;
+ virtual void addTable(const char* aTableName, bool isVerify=true) = 0;
+
+ void startTimer(NDBT_Context*);
+ void stopTimer(NDBT_Context*);
+ void printTimer(NDBT_Context*);
+
+ const char* name;
+ const char* comment;
+ NDBT_TestSuite* suite;
+ Properties props;
+ NdbTimer timer;
+ bool isVerifyTables;
+};
+
+static const int FAILED_TO_CREATE = 1000;
+static const int FAILED_TO_DISCOVER = 1001;
+
+
+class NDBT_TestCaseResult{
+public:
+ NDBT_TestCaseResult(const char* name, int _result, NDB_TICKS _ticks):
+ m_result(_result){
+ m_name.assign(name);
+ m_ticks = _ticks;
+
+ };
+ const char* getName(){return m_name.c_str(); };
+ int getResult(){return m_result; };
+ const char* getTimeStr(){
+ // Convert to Uint32 in order to be able to print it to screen
+ Uint32 lapTime = (Uint32)m_ticks;
+ Uint32 secTime = lapTime/1000;
+ snprintf(buf, 255, "%d secs (%d ms)", secTime, lapTime);
+ return buf;
+ }
+private:
+ char buf[255];
+ int m_result;
+ BaseString m_name;
+ NDB_TICKS m_ticks;
+};
+
+class NDBT_TestCaseImpl1 : public NDBT_TestCase {
+public:
+ NDBT_TestCaseImpl1(NDBT_TestSuite* psuite,
+ const char* name,
+ const char* comment);
+ virtual ~NDBT_TestCaseImpl1();
+ int addStep(NDBT_Step*);
+ int addVerifier(NDBT_Verifier*);
+ int addInitializer(NDBT_Initializer*);
+ int addFinalizer(NDBT_Finalizer*);
+ void addTable(const char*, bool);
+ bool tableExists(NdbDictionary::Table*);
+ bool isVerify(const NdbDictionary::Table*);
+ void reportStepResult(const NDBT_Step*, int result);
+ // int execute(NDBT_Context* ctx);
+ int runInit(NDBT_Context* ctx);
+ int runSteps(NDBT_Context* ctx);
+ int runVerifier(NDBT_Context* ctx);
+ int runFinal(NDBT_Context* ctx);
+ void print();
+ void printHTML();
+
+ virtual int getNoOfRunningSteps() const;
+ virtual int getNoOfCompletedSteps() const;
+private:
+ static const int NORESULT = 999;
+
+ void saveTestResult(const NdbDictionary::Table* ptab, int result);
+ void printTestResult();
+
+ void startStepInThread(int stepNo, NDBT_Context* ctx);
+ void waitSteps();
+ Vector<NDBT_Step*> steps;
+ Vector<NdbThread*> threads;
+ Vector<int> results;
+ Vector<NDBT_Verifier*> verifiers;
+ Vector<NDBT_Initializer*> initializers;
+ Vector<NDBT_Finalizer*> finalizers;
+ Vector<const NdbDictionary::Table*> testTables;
+ Vector<NDBT_TestCaseResult*> testResults;
+ unsigned numStepsFail;
+ unsigned numStepsOk;
+ unsigned numStepsCompleted;
+ NdbMutex* waitThreadsMutexPtr;
+ NdbCondition* waitThreadsCondPtr;
+};
+
+
+// A NDBT_TestSuite is a collection of TestCases
+// the test suite will know how to execute the test cases
+class NDBT_TestSuite {
+public:
+ NDBT_TestSuite(const char* name);
+ ~NDBT_TestSuite();
+
+ // Default executor of a test suite
+ // supply argc and argv as parameters
+ int execute(int, const char**);
+
+
+ // These function can be used from main in the test program
+ // to control the behaviour of the testsuite
+ void setCreateTable(bool); // Create table before test func is called
+ void setCreateAllTables(bool); // Create all tables before testsuite is executed
+
+ // Prints the testsuite, testcases and teststeps
+ void printExecutionTree();
+ void printExecutionTreeHTML();
+
+ // Prints list of testcases
+ void printCases();
+
+ // Print summary of executed tests
+ void printTestCaseSummary(const char* tcname = NULL);
+
+ /**
+ * Returns current date and time in the format of 2002-12-04 10:00:01
+ */
+ const char* getDate();
+
+ // Returns true if timing info should be printed
+ bool timerIsOn();
+
+
+ int addTest(NDBT_TestCase* pTest);
+private:
+ int executeOne(const char* _tabname, const char* testname = NULL);
+ int executeAll(const char* testname = NULL);
+
+ void execute(Ndb*, const NdbDictionary::Table*, const char* testname = NULL);
+ int report(const char* _tcname = NULL);
+ int reportAllTables(const char* );
+ const char* name;
+ char* remote_mgm;
+ int numTestsOk;
+ int numTestsFail;
+ int numTestsExecuted;
+ Vector<NDBT_TestCase*> tests;
+ NDBT_Context* ctx;
+ int records;
+ int loops;
+ int timer;
+ NdbTimer testSuiteTimer;
+ bool createTable;
+};
+
+
+
+#define NDBT_TESTSUITE(suitname) \
+class C##suitname : public NDBT_TestSuite { \
+public: \
+C##suitname():NDBT_TestSuite(#suitname){ \
+ NDBT_TestCaseImpl1* pt; pt = NULL; \
+ NDBT_Step* pts; pts = NULL; \
+ NDBT_Verifier* ptv; ptv = NULL; \
+ NDBT_Initializer* pti; pti = NULL; \
+ NDBT_Finalizer* ptf; ptf = NULL;
+
+#define TESTCASE(testname, comment) \
+ pt = new NDBT_TestCaseImpl1(this, testname, comment); \
+ addTest(pt);
+
+#define TC_PROPERTY(propname, propval) \
+ pt->setProperty(propname, propval);
+
+#define STEP(stepfunc) \
+ pts = new NDBT_ParallelStep(pt, #stepfunc, stepfunc); \
+ pt->addStep(pts);
+
+// Add a number of equal steps to the testcase
+#define STEPS(stepfunc, num) \
+ for (int i = 0; i < num; i++){ \
+ pts = new NDBT_ParallelStep(pt, #stepfunc, stepfunc); \
+ pt->addStep(pts);\
+ }
+
+#define VERIFIER(stepfunc) \
+ ptv = new NDBT_Verifier(pt, #stepfunc, stepfunc); \
+ pt->addVerifier(ptv);
+
+#define INITIALIZER(stepfunc) \
+ pti = new NDBT_Initializer(pt, #stepfunc, stepfunc); \
+ pt->addInitializer(pti);
+
+#define FINALIZER(stepfunc) \
+ ptf = new NDBT_Finalizer(pt, #stepfunc, stepfunc); \
+ pt->addFinalizer(ptf);
+
+// Test case can be run only on this table(s), can be multiple tables
+// Ex TABLE("T1")
+// TABLE("T3")
+// Means test will only be run on T1 and T3
+#define TABLE(tableName) \
+ pt->addTable(tableName, true);
+
+// Test case can be run on all tables except
+// Ex NOT_TABLE("T10")
+// Means test will be run on all tables execept T10
+#define NOT_TABLE(tableName) \
+ pt->addTable(tableName, false);
+
+#define NDBT_TESTSUITE_END(suitname) \
+ } } ; C##suitname suitname;
+
+// Helper functions for retrieving variables from NDBT_Step
+#define GETNDB(ps) ((NDBT_NdbApiStep*)ps)->getNdb()
+
+#endif
diff --git a/ndb/test/include/NdbBackup.hpp b/ndb/test/include/NdbBackup.hpp
new file mode 100644
index 00000000000..0a372eca7df
--- /dev/null
+++ b/ndb/test/include/NdbBackup.hpp
@@ -0,0 +1,54 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_BACKUP_HPP
+#define NDBT_BACKUP_HPP
+
+#include <mgmapi.h>
+#include <Vector.hpp>
+#include "NdbConfig.hpp"
+#include <NdbRestarter.hpp>
+
+class NdbBackup : public NdbConfig {
+public:
+ NdbBackup(int _own_id, const char* _addr = 0)
+ : NdbConfig(_own_id, _addr) {};
+
+ int start(unsigned & _backup_id);
+ int restore(unsigned _backup_id);
+
+ int NFMaster(NdbRestarter& _restarter);
+ int NFMasterAsSlave(NdbRestarter& _restarter);
+ int NFSlave(NdbRestarter& _restarter);
+ int NF(NdbRestarter& _restarter, int *NFDuringBackup_codes, const int sz, bool onMaster);
+
+ int FailMaster(NdbRestarter& _restarter);
+ int FailMasterAsSlave(NdbRestarter& _restarter);
+ int FailSlave(NdbRestarter& _restarter);
+ int Fail(NdbRestarter& _restarter, int *Fail_codes, const int sz, bool onMaster);
+
+private:
+
+ int execRestore(bool _restore_data,
+ bool _restore_meta,
+ int _node_id,
+ unsigned _backup_id);
+
+ const char * getFileSystemPathForNode(int _node_id);
+
+};
+
+#endif
diff --git a/ndb/test/include/NdbConfig.hpp b/ndb/test/include/NdbConfig.hpp
new file mode 100644
index 00000000000..f13872f4d64
--- /dev/null
+++ b/ndb/test/include/NdbConfig.hpp
@@ -0,0 +1,46 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_CONFIG_HPP
+#define NDBT_CONFIG_HPP
+
+#include <mgmapi.h>
+#include <Vector.hpp>
+#include <NdbRestarter.hpp>
+#include <Properties.hpp>
+
+class NdbConfig : public NdbRestarter{
+public:
+ NdbConfig(int own_id, const char* addr = 0)
+ : NdbRestarter(addr),
+ ownNodeId(own_id) {};
+
+ bool getProperty(unsigned int node_id, const char* type,
+ const char * name, Uint32 * value) const;
+ bool getProperty(unsigned int node_id, const char* type,
+ const char * name, const char ** value) const;
+
+ bool getHostName(unsigned int node_id,
+ const char ** hostname) const;
+protected:
+ bool getPropsForNode(unsigned int node_id,
+ const char* type,
+ const Properties ** props) const;
+
+ int ownNodeId;
+};
+
+#endif
diff --git a/ndb/test/include/NdbGrep.hpp b/ndb/test/include/NdbGrep.hpp
new file mode 100644
index 00000000000..31c49d1e4da
--- /dev/null
+++ b/ndb/test/include/NdbGrep.hpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_GREP_HPP
+#define NDBT_GREP_HPP
+
+#include <mgmapi.h>
+#include <Vector.hpp>
+#include "NdbConfig.hpp"
+#include <NdbRestarter.hpp>
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+class NdbGrep : public NdbConfig {
+public:
+ NdbGrep(int _own_id, const char* _addr = 0)
+ : NdbConfig(_own_id, _addr) {};
+
+ int start();
+ int stop();
+ int query();
+
+
+ int verify(NDBT_Context* ctx);
+
+
+ int NFMaster(NdbRestarter& _restarter);
+ int NFMasterAsSlave(NdbRestarter& _restarter);
+ int NFSlave(NdbRestarter& _restarter);
+ int NF(NdbRestarter& _restarter, int *NFDuringGrep_codes, const int sz, bool onMaster);
+
+ int FailMaster(NdbRestarter& _restarter);
+ int FailMasterAsSlave(NdbRestarter& _restarter);
+ int FailSlave(NdbRestarter& _restarter);
+ int Fail(NdbRestarter& _restarter, int *Fail_codes, const int sz, bool onMaster);
+
+private:
+
+};
+
+#endif
diff --git a/ndb/test/include/NdbRestarter.hpp b/ndb/test/include/NdbRestarter.hpp
new file mode 100644
index 00000000000..cfd5409bb69
--- /dev/null
+++ b/ndb/test/include/NdbRestarter.hpp
@@ -0,0 +1,95 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_RESTARTER_HPP
+#define NDBT_RESTARTER_HPP
+
+#include <mgmapi.h>
+#include <Vector.hpp>
+
+class NdbRestarter {
+public:
+ NdbRestarter(const char* _addr = 0);
+ ~NdbRestarter();
+
+ int getDbNodeId(int _i);
+
+ int restartOneDbNode(int _nodeId,
+ bool initial = false,
+ bool nostart = false,
+ bool abort = false);
+
+ int restartAll(bool initial = false,
+ bool nostart = false,
+ bool abort = false);
+
+ int startAll();
+ int startNodes(int * _nodes, int _num_nodes);
+ int waitClusterStarted(unsigned int _timeout = 120);
+ int waitClusterSingleUser(unsigned int _timeout = 120);
+ int waitClusterStartPhase(int _startphase, unsigned int _timeout = 120);
+ int waitClusterNoStart(unsigned int _timeout = 120);
+ int waitNodesStarted(int * _nodes, int _num_nodes,
+ unsigned int _timeout = 120);
+ int waitNodesStartPhase(int * _nodes, int _num_nodes,
+ int _startphase, unsigned int _timeout = 120);
+ int waitNodesNoStart(int * _nodes, int _num_nodes,
+ unsigned int _timeout = 120);
+
+
+ int getNumDbNodes();
+ int insertErrorInNode(int _nodeId, int error);
+ int insertErrorInAllNodes(int error);
+
+ int enterSingleUserMode(int _nodeId);
+ int exitSingleUserMode();
+
+ int dumpStateOneNode(int _nodeId, int * _args, int _num_args);
+ int dumpStateAllNodes(int * _args, int _num_args);
+
+ int getMasterNodeId();
+ int getRandomNodeOtherNodeGroup(int nodeId, int randomNumber);
+ int getRandomNotMasterNodeId(int randomNumber);
+
+protected:
+
+ int waitClusterState(ndb_mgm_node_status _status,
+ unsigned int _timeout,
+ int _startphase = -1);
+
+ int waitNodesState(int * _nodes, int _num_nodes,
+ ndb_mgm_node_status _status,
+ unsigned int _timeout,
+ int _startphase = -1);
+
+ bool isConnected();
+ int connect();
+ void disconnect();
+ int getStatus();
+
+ Vector<ndb_mgm_node_state> ndbNodes;
+ Vector<ndb_mgm_node_state> mgmNodes;
+ Vector<ndb_mgm_node_state> apiNodes;
+
+ bool connected;
+ const char* addr;
+ const char* host;
+ int port;
+ NdbMgmHandle handle;
+
+};
+
+#endif
diff --git a/ndb/test/include/NdbRestarts.hpp b/ndb/test/include/NdbRestarts.hpp
new file mode 100644
index 00000000000..aabcd7b9975
--- /dev/null
+++ b/ndb/test/include/NdbRestarts.hpp
@@ -0,0 +1,120 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBT_RESTARTS_HPP
+#define NDBT_RESTARTS_HPP
+
+#include <NdbRestarter.hpp>
+#include <NdbTick.h>
+#include <random.h>
+
+/**
+ * This class is used to test Ndb's ability to handle
+ * node- and system-restarts.
+ * For example:
+ * Node restart: Restart one node in the cluster.
+ * System restart: Restart all nodes in the cluster.
+ * Node crash: Crash one node in the middle of execution and bring it up again.
+ * Multiple node crash: Crash multiple nodes with a few seconds or milliseconds delay between.
+ * Initial node restart: Restart one node in the cluster without a filesystem on disk.
+ *
+ * Each restart type is represented by a NdbRestart class and a collection of these are stored
+ * in the NdbRestarts class.
+ *
+ * This class may be used from other programs to execute a particular restart.
+ *
+ */
+
+
+class NdbRestarts {
+public:
+ NdbRestarts(const char* _addr = 0):
+ m_restarter(_addr)
+ {
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ }
+
+ enum NdbRestartType{
+ NODE_RESTART,
+ MULTIPLE_NODE_RESTART,
+ SYSTEM_RESTART
+ };
+
+ struct NdbRestart {
+ typedef int (restartFunc)(NdbRestarter&, const NdbRestart*);
+
+ NdbRestart(const char* _name,
+ NdbRestartType _type,
+ restartFunc* _func,
+ int _requiredNodes,
+ int _arg1 = -1);
+
+ const char * m_name;
+ NdbRestartType m_type;
+ restartFunc* m_restartFunc;
+ int m_numRequiredNodes;
+ int m_arg1;
+
+ };
+
+ int getNumRestarts();
+
+ int executeRestart(int _num, unsigned int _timeout = 120);
+ int executeRestart(const char* _name, unsigned int _timeout = 120);
+
+ void listRestarts();
+ void listRestarts(NdbRestartType _type);
+private:
+ int executeRestart(const NdbRestart*, unsigned int _timeout);
+
+ struct NdbErrorInsert {
+ NdbErrorInsert(const char* _name,
+ int _errorNo);
+
+ const char * m_name;
+ int m_errorNo;
+
+ public:
+ const char* getName();
+ };
+
+ int getNumErrorInserts();
+ const NdbErrorInsert* getError(int _num);
+ const NdbErrorInsert* getRandomError();
+
+ static const NdbErrorInsert m_errors[];
+ static const int m_NoOfErrors;
+
+ const NdbRestart* getRestart(int _num);
+ const NdbRestart* getRestart(const char* _name);
+
+ static const NdbRestart m_restarts[];
+ static const int m_NoOfRestarts;
+
+ NdbRestarter m_restarter;
+};
+
+
+
+
+
+
+
+
+
+
+
+#endif
diff --git a/ndb/test/include/NdbTest.hpp b/ndb/test/include/NdbTest.hpp
new file mode 100644
index 00000000000..a2e612b7ffa
--- /dev/null
+++ b/ndb/test/include/NdbTest.hpp
@@ -0,0 +1,35 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_TEST_HPP
+#define NDB_TEST_HPP
+
+/**
+ * NdbTest.hpp
+ * This is the main include file to include in test programs
+ * It will include all the other include files in the NDBT-toolkit
+ *
+ */
+
+#include "NDBT_ReturnCodes.h"
+
+#ifdef __cplusplus
+#include "NDBT_Table.hpp"
+#include "NDBT_Error.hpp"
+#endif
+
+
+#endif
diff --git a/ndb/test/include/NdbTimer.hpp b/ndb/test/include/NdbTimer.hpp
new file mode 100644
index 00000000000..8d9a088b7b5
--- /dev/null
+++ b/ndb/test/include/NdbTimer.hpp
@@ -0,0 +1,110 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDBTIMER_H
+#define NDBTIMER_H
+
+#include <NdbStdio.h>
+#include <NdbTick.h>
+#include <NdbOut.hpp>
+
+//
+// Class used for measuring time and priting the results
+//
+// Currently measures time in milliseconds
+//
+
+class NdbTimer
+{
+public:
+
+ NdbTimer();
+ ~NdbTimer() {};
+
+ void doStart();
+ void doStop();
+ void doReset();
+ NDB_TICKS elapsedTime();
+ void printTransactionStatistics(const char* text,
+ int numTransactions,
+ int numOperations);
+ void printTestTimer(int numLoops,
+ int numRecords);
+ void printTotalTime(void);
+private:
+ NDB_TICKS startTime;
+ NDB_TICKS stopTime;
+};
+
+inline NdbTimer::NdbTimer(){
+ doReset();
+}
+
+inline void NdbTimer::doReset(void){
+ startTime = 0;
+ stopTime = 0;
+}
+
+inline void NdbTimer::doStart(void){
+ startTime = NdbTick_CurrentMillisecond();
+}
+
+inline void NdbTimer::doStop(void){
+ stopTime = NdbTick_CurrentMillisecond();
+}
+
+inline NDB_TICKS NdbTimer::elapsedTime(void){
+ return (stopTime - startTime);
+}
+
+inline void NdbTimer::printTransactionStatistics(const char* text,
+ int numTransactions,
+ int numOperations){
+
+ // Convert to Uint32 in order to be able to print it to screen
+ Uint32 lapTime = (Uint32)elapsedTime();
+ ndbout_c("%i transactions, %i %s total time = %d ms\nAverage %f ms/transaction, %f ms/%s.\n%f transactions/second, %f %ss/second.\n",
+ numTransactions, numTransactions*numOperations, text, lapTime,
+ ((double)lapTime/numTransactions), ((double)lapTime/(numTransactions*numOperations)), text,
+ 1000.0/((double)lapTime/numOperations), 1000.0/((double)lapTime/(numTransactions*numOperations)), text);
+}
+
+
+
+inline void NdbTimer::printTestTimer(int numLoops,
+ int numRecords){
+ // Convert to Uint32 in order to be able to print it to screen
+ Uint32 lapTime = (Uint32)elapsedTime();
+ ndbout_c("%i loop * %i records, total time = %d ms\nAverage %f ms/loop, %f ms/record.\n%f looop/second, %f records/second.\n",
+ numLoops, numRecords, lapTime,
+ ((double)lapTime/numLoops), ((double)lapTime/(numLoops*numRecords)),
+ 1000.0/((double)lapTime/numLoops), 1000.0/((double)lapTime/(numLoops*numRecords)));
+}
+
+
+inline void NdbTimer::printTotalTime(void){
+ // Convert to Uint32 in order to be able to print it to screen
+ Uint32 lapTime = (Uint32)elapsedTime();
+ Uint32 secTime = lapTime/1000;
+ ndbout_c("Total time : %d seconds (%d ms)\n", secTime, lapTime);
+}
+
+
+
+
+
+
+#endif
diff --git a/ndb/test/include/TestNdbEventOperation.hpp b/ndb/test/include/TestNdbEventOperation.hpp
new file mode 100644
index 00000000000..307b0e0089b
--- /dev/null
+++ b/ndb/test/include/TestNdbEventOperation.hpp
@@ -0,0 +1,24 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+struct EventOperationStats {
+ int n_inserts;
+ int n_deletes;
+ int n_updates;
+ int n_duplicates;
+ int n_consecutive;
+ int n_inconsistent_gcis;
+};
diff --git a/ndb/test/include/UtilTransactions.hpp b/ndb/test/include/UtilTransactions.hpp
new file mode 100644
index 00000000000..b16ab74455e
--- /dev/null
+++ b/ndb/test/include/UtilTransactions.hpp
@@ -0,0 +1,108 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef UTIL_TRANSACTIONS_HPP
+#define UTIL_TRANSACTIONS_HPP
+
+#include <NDBT.hpp>
+
+typedef int (ReadCallBackFn)(NDBT_ResultRow*);
+
+class UtilTransactions {
+public:
+ enum ScanLock {
+ SL_Read = 0,
+ SL_ReadHold = 1,
+ SL_Exclusive = 2
+ };
+
+ UtilTransactions(const NdbDictionary::Table&);
+ UtilTransactions(Ndb* ndb, const char * tableName);
+
+ int clearTable(Ndb*,
+ int records = 0,
+ int parallelism = 240);
+
+ // Delete all records from the table using a scan
+ int clearTable1(Ndb*,
+ int records = 0,
+ int parallelism = 16);
+ // Delete all records from the table using a scan
+ // Using batching
+ int clearTable2(Ndb*,
+ int records = 0,
+ int parallelism = 240);
+
+ int clearTable3(Ndb*,
+ int records = 0,
+ int parallelism = 240);
+
+ int selectCount(Ndb*,
+ int parallelism = 16,
+ int* count_rows = NULL,
+ ScanLock lock = SL_Read,
+ NdbConnection* pTrans = NULL);
+ int scanReadRecords(Ndb*,
+ int parallelism,
+ bool exclusive,
+ int records,
+ int noAttribs,
+ int* attrib_list,
+ ReadCallBackFn* fn = NULL);
+ int verifyIndex(Ndb*,
+ const char* indexName,
+ int parallelism = 240,
+ bool transactional = false);
+
+ int copyTableData(Ndb*,
+ const char* destName);
+
+
+private:
+ static int takeOverAndDeleteRecord(Ndb*,
+ NdbOperation*);
+
+ int addRowToDelete(Ndb* pNdb,
+ NdbConnection* pDelTrans,
+ NdbOperation* pOrgOp);
+
+
+ int addRowToInsert(Ndb* pNdb,
+ NdbConnection* pInsTrans,
+ NDBT_ResultRow & row,
+ const char* insertTabName);
+
+
+ int verifyUniqueIndex(Ndb*,
+ const char* indexName,
+ int parallelism = 240,
+ bool transactional = false);
+
+ int scanAndCompareUniqueIndex(Ndb* pNdb,
+ const char * indexName,
+ int parallelism,
+ bool transactional);
+
+ int readRowFromTableAndIndex(Ndb* pNdb,
+ NdbConnection* pTrans,
+ const char * indexName,
+ NDBT_ResultRow& row );
+protected:
+ int m_defaultClearMethod;
+ const NdbDictionary::Table& tab;
+};
+
+#endif
diff --git a/ndb/test/ndbapi/Makefile b/ndb/test/ndbapi/Makefile
new file mode 100644
index 00000000000..67dbc120e69
--- /dev/null
+++ b/ndb/test/ndbapi/Makefile
@@ -0,0 +1,55 @@
+include .defs.mk
+
+
+ifeq ($(NDB_OS), OSE)
+DIRS = basic flexBench flexAsynch
+else
+DIRS = lmc-bench bank
+BIN_DIRS = \
+ flexAsynch \
+ flexBench \
+ flexHammer \
+ flexScan \
+ flexTT \
+ nodeRecovery \
+ restartgci \
+ create_tab \
+ create_all_tabs \
+ drop_all_tabs \
+ bulk_copy \
+ restarter2 restarter \
+ restarts testScan testNdbApi \
+ testScanInterpreter testIndex \
+ testInterpreter \
+ testOIBasic \
+ testBackup \
+ testBasic \
+ basicAsynch \
+ testNodeRestart \
+ testOperations testTransactions \
+ testSystemRestart \
+ testTimeout \
+ testMgm \
+ testRestartGci \
+ testDataBuffers \
+ testDict \
+ initronja \
+ benchronja \
+ acid \
+ interpreterInTup \
+ telco \
+ indexTest \
+ test_event \
+ indexTest2 \
+ testGrep
+# tables \
+
+ifeq ($(NDB_OS), SOLARIS)
+ifeq ($(NDB_COMPILER), FORTE6)
+ DIRS += flexTimedAsynch
+endif
+endif
+endif
+
+include ${NDB_TOP}/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/acid/Makefile b/ndb/test/ndbapi/acid/Makefile
new file mode 100644
index 00000000000..33dc49fcdea
--- /dev/null
+++ b/ndb/test/ndbapi/acid/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := acid
+
+# Source files of non-templated classes (.C files)
+SOURCES = acid.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/acid/acid.cpp b/ndb/test/ndbapi/acid/acid.cpp
new file mode 100644
index 00000000000..49961531a1c
--- /dev/null
+++ b/ndb/test/ndbapi/acid/acid.cpp
@@ -0,0 +1,561 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <NdbMutex.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbThread.h>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbTest.hpp>
+#include <random.h>
+
+//#define TRACE
+#define DEBUG
+//#define RELEASE
+#define NODE_REC // epaulsa: introduces pointer checks to help 'acid' keep core
+// during node recovery
+
+#ifdef TRACE
+
+#define VerifyMethodInt(c, m) (ReportMethodInt(c->m, c, #c, #m, __FILE__, __LINE__))
+#define VerifyMethodPtr(v, c, m) (v=ReportMethodPtr(c->m, c, #v, #c, #m, __FILE__, __LINE__))
+#define VerifyMethodVoid(c, m) (c->m, ReportMethodVoid(c, #c, #m, __FILE__, __LINE__))
+
+int ReportMethodInt(int iRes, NdbConnection* pNdbConnection, const char* szClass, const char* szMethod, const char* szFile, const int iLine)
+{
+ ndbout << szFile << "(" << iLine << ") : ";
+ ndbout << szClass << "->" << szMethod << " return " << iRes << " : ";
+ ndbout << pNdbConnection->getNdbError();
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbErrorOperation();
+ if(pNdbOperation) {
+ ndbout << " : " << pNdbOperation->getNdbError();
+ }
+ ndbout << " : " << pNdbConnection->getNdbErrorLine();
+ ndbout << endl;
+ return iRes;
+}
+
+template <class C>
+int ReportMethodInt(int iRes, C* pC, const char* szClass, const char* szMethod, const char* szFile, const int iLine)
+{
+ ndbout << szFile << "(" << iLine << ") : ";
+ ndbout << szClass << "->" << szMethod << " return " << iRes << " : ";
+ ndbout << pC->getNdbError();
+ ndbout << endl;
+ return iRes;
+}
+
+template <class R, class C>
+R* ReportMethodPtr(R* pR, C* pC, const char* szVariable, const char* szClass, const char* szMethod, const char* szFile, const int iLine)
+{
+ ndbout << szFile << "(" << iLine << ") : ";
+ ndbout << szVariable << " = " << szClass << "->" << szMethod << " return " << (long)(void*)pR << " : ";
+ ndbout << pC->getNdbError();
+ ndbout << endl;
+ return pR;
+}
+
+template <class C>
+void ReportMethodVoid(C* pC, const char* szClass, const char* szMethod, const char* szFile, const int iLine)
+{
+ ndbout << szFile << "(" << iLine << ") : ";
+ ndbout << szClass << "->" << szMethod << " : ";
+ ndbout << pC->getNdbError();
+ ndbout << endl;
+}
+#endif /* TRACE */
+
+
+#ifdef DEBUG
+
+#define VerifyMethodInt(c, m) (ReportMethodInt(c->m, c, #c, #m, __FILE__, __LINE__))
+#define VerifyMethodPtr(v, c, m) (v=ReportMethodPtr(c->m, c, #v, #c, #m, __FILE__, __LINE__))
+#define VerifyMethodVoid(c, m) (c->m, ReportMethodVoid(c, #c, #m, __FILE__, __LINE__))
+
+int ReportMethodInt(int iRes, NdbConnection* pNdbConnection, const char* szClass, const char* szMethod, const char* szFile, const int iLine)
+{
+ if(iRes<0) {
+ ndbout << szFile << "(" << iLine << ") : ";
+ ndbout << szClass << "->" << szMethod << " return " << iRes << " : ";
+ ndbout << pNdbConnection->getNdbError();
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbErrorOperation();
+ if(pNdbOperation) {
+ ndbout << " : " << pNdbOperation->getNdbError();
+ }
+ ndbout << " : " << pNdbConnection->getNdbErrorLine();
+ ndbout << " : ";
+ ndbout << endl;
+ }
+ return iRes;
+}
+
+template <class C>
+int ReportMethodInt(int iRes, C* pC, const char* szClass, const char* szMethod, const char* szFile, const int iLine)
+{
+ if(iRes<0) {
+ ndbout << szFile << "(" << iLine << ") : ";
+ ndbout << szClass << "->" << szMethod << " return " << iRes << " : ";
+ ndbout << pC->getNdbError();
+ ndbout << endl;
+ }
+ return iRes;
+}
+
+template <class R, class C>
+R* ReportMethodPtr(R* pR, C* pC, const char* szVariable, const char* szClass, const char* szMethod, const char* szFile, const int iLine)
+{
+ if(!pR) {
+ ndbout << szFile << "(" << iLine << ") : ";
+ ndbout << szVariable << " = " << szClass << "->" << szMethod << " return " << " : ";
+ ndbout << pC->getNdbError();
+ ndbout << endl;
+ }
+ return pR;
+}
+
+template <class C>
+void ReportMethodVoid(C* pC, const char* szClass, const char* szMethod, const char* szFile, const int iLine)
+{
+ if(pC->getNdbError().code) {
+ ndbout << szFile << "(" << iLine << ") : ";
+ ndbout << szClass << "->" << szMethod << " : ";
+ ndbout << pC->getNdbError();
+ ndbout << endl;
+ }
+}
+
+
+#endif /* DEBUG */
+
+
+#ifdef RELEASE
+
+#define VerifyMethodInt(c, m) (c->m)
+#define VerifyMethodPtr(v, c, m) (v=(c->m))
+#define VerifyMethodVoid(c, m) (c->m)
+
+int ReportMethodInt(int iRes, NdbConnection* pNdbConnection, const char* szClass, const char* szMethod, const char* szFile, const int iLine)
+{
+ if(iRes<0) {
+ ndbout << szFile << "(" << iLine << ") : ";
+ ndbout << szClass << "->" << szMethod << " return " << iRes << " : ";
+ ndbout << pNdbConnection->getNdbError();
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbErrorOperation();
+ if(pNdbOperation) {
+ ndbout << " : " << pNdbOperation->getNdbError();
+ }
+ ndbout << " : " << pNdbConnection->getNdbErrorLine();
+ ndbout << endl;
+ }
+ return iRes;
+}
+
+#endif /* RELEASE */
+
+// epaulsa =>
+#ifndef NODE_REC
+#define CHK_TR(p)
+#else
+#define CHK_TR(p) if(!p){ \
+ ndbout <<"startTransaction failed, returning now." << endl ; \
+ delete pNdb ; \
+ pNdb = NULL ; \
+ return 0 ; \
+ }
+#endif // NODE_REC
+// <= epaulsa
+
+const char* c_szWarehouse = "WAREHOUSE";
+const char* c_szWarehouseNumber = "W_ID";
+const char* c_szWarehouseSum = "W_SUM";
+const char* c_szWarehouseCount = "W_CNT";
+const char* c_szDistrict = "DISTRICT";
+const char* c_szDistrictWarehouseNumber = "D_W_ID";
+const char* c_szDistrictNumber = "D_ID";
+const char* c_szDistrictSum = "D_SUM";
+const char* c_szDistrictCount = "D_CNT";
+
+Uint32 g_nWarehouseCount = 10;
+Uint32 g_nDistrictPerWarehouse = 10;
+Uint32 g_nThreadCount = 1;
+NdbMutex* g_pNdbMutex = 0;
+
+extern "C" void* NdbThreadFuncInsert(void* pArg)
+{
+ myRandom48Init((long int)NdbTick_CurrentMillisecond());
+ unsigned nSucc = 0;
+ unsigned nFail = 0;
+ Ndb* pNdb = NULL ;
+ pNdb = new Ndb("TEST_DB");
+ VerifyMethodInt(pNdb, init());
+ VerifyMethodInt(pNdb, waitUntilReady());
+
+ while(NdbMutex_Trylock(g_pNdbMutex)) {
+ Uint32 nWarehouse = myRandom48(g_nWarehouseCount);
+ NdbConnection* pNdbConnection = NULL ;
+ VerifyMethodPtr(pNdbConnection, pNdb, startTransaction());
+ CHK_TR(pNdbConnection);
+ NdbOperation* pNdbOperationW = NULL ;
+ VerifyMethodPtr(pNdbOperationW, pNdbConnection, getNdbOperation(c_szWarehouse));
+ VerifyMethodInt(pNdbOperationW, insertTuple());
+ VerifyMethodInt(pNdbOperationW, equal(c_szWarehouseNumber, nWarehouse));
+ VerifyMethodInt(pNdbOperationW, setValue(c_szWarehouseCount, Uint32(1)));
+ Uint32 nWarehouseSum = 0;
+ for(Uint32 nDistrict=0; nDistrict<g_nDistrictPerWarehouse; ++nDistrict) {
+ NdbOperation* pNdbOperationD = NULL ;
+ VerifyMethodPtr(pNdbOperationD, pNdbConnection, getNdbOperation(c_szDistrict));
+ VerifyMethodInt(pNdbOperationD, insertTuple());
+ VerifyMethodInt(pNdbOperationD, equal(c_szDistrictWarehouseNumber, nWarehouse));
+ VerifyMethodInt(pNdbOperationD, equal(c_szDistrictNumber, nDistrict));
+ VerifyMethodInt(pNdbOperationD, setValue(c_szDistrictCount, Uint32(1)));
+ Uint32 nDistrictSum = myRandom48(100);
+ nWarehouseSum += nDistrictSum;
+ VerifyMethodInt(pNdbOperationD, setValue(c_szDistrictSum, nDistrictSum));
+ }
+ VerifyMethodInt(pNdbOperationW, setValue(c_szWarehouseSum, nWarehouseSum));
+ int iExec = pNdbConnection->execute(Commit);
+ int iError = pNdbConnection->getNdbError().code;
+ CommitStatusType cs = pNdbConnection->commitStatus();
+
+ if(iExec<0 && iError!=0 && iError!=266 && iError!=630) {
+ ReportMethodInt(iExec, pNdbConnection, "pNdbConnection", "execute(Commit)", __FILE__, __LINE__);
+ }
+ if(iExec==0) {
+ ++nSucc;
+ } else {
+ ++nFail;
+ }
+ VerifyMethodVoid(pNdb, closeTransaction(pNdbConnection));
+ }
+ ndbout << "insert: " << nSucc << " succeeded, " << nFail << " failed " << endl;
+ NdbMutex_Unlock(g_pNdbMutex);
+ delete pNdb;
+ pNdb = NULL ;
+ return NULL;
+}
+
+
+extern "C" void* NdbThreadFuncUpdate(void* pArg)
+{
+ myRandom48Init((long int)NdbTick_CurrentMillisecond());
+ unsigned nSucc = 0;
+ unsigned nFail = 0;
+ Ndb* pNdb = NULL ;
+ pNdb = new Ndb("TEST_DB");
+ VerifyMethodInt(pNdb, init());
+ VerifyMethodInt(pNdb, waitUntilReady());
+
+ while(NdbMutex_Trylock(g_pNdbMutex)) {
+ Uint32 nWarehouse = myRandom48(g_nWarehouseCount);
+ NdbConnection* pNdbConnection = NULL ;
+ VerifyMethodPtr(pNdbConnection, pNdb, startTransaction());
+ CHK_TR(pNdbConnection) ; // epaulsa
+ NdbOperation* pNdbOperationW = NULL ;
+ VerifyMethodPtr(pNdbOperationW, pNdbConnection, getNdbOperation(c_szWarehouse));
+ VerifyMethodInt(pNdbOperationW, interpretedUpdateTuple());
+ VerifyMethodInt(pNdbOperationW, equal(c_szWarehouseNumber, nWarehouse));
+ VerifyMethodInt(pNdbOperationW, incValue(c_szWarehouseCount, Uint32(1)));
+ Uint32 nWarehouseSum = 0;
+ for(Uint32 nDistrict=0; nDistrict<g_nDistrictPerWarehouse; ++nDistrict) {
+ NdbOperation* pNdbOperationD = NULL ;
+ VerifyMethodPtr(pNdbOperationD, pNdbConnection, getNdbOperation(c_szDistrict));
+ VerifyMethodInt(pNdbOperationD, interpretedUpdateTuple());
+ VerifyMethodInt(pNdbOperationD, equal(c_szDistrictWarehouseNumber, nWarehouse));
+ VerifyMethodInt(pNdbOperationD, equal(c_szDistrictNumber, nDistrict));
+ VerifyMethodInt(pNdbOperationD, incValue(c_szDistrictCount, Uint32(1)));
+ Uint32 nDistrictSum = myRandom48(100);
+ nWarehouseSum += nDistrictSum;
+ VerifyMethodInt(pNdbOperationD, setValue(c_szDistrictSum, nDistrictSum));
+ }
+ VerifyMethodInt(pNdbOperationW, setValue(c_szWarehouseSum, nWarehouseSum));
+ int iExec = pNdbConnection->execute(Commit);
+ int iError = pNdbConnection->getNdbError().code;
+ CommitStatusType cs = pNdbConnection->commitStatus();
+
+ if(iExec<0 && iError!=0 && iError!=266 && iError!=626) {
+ ReportMethodInt(iExec, pNdbConnection, "pNdbConnection", "execute(Commit)", __FILE__, __LINE__);
+ }
+ if(iExec==0) {
+ ++nSucc;
+ } else {
+ ++nFail;
+ }
+ VerifyMethodVoid(pNdb, closeTransaction(pNdbConnection));
+ }
+ ndbout << "update: " << nSucc << " succeeded, " << nFail << " failed " << endl;
+ NdbMutex_Unlock(g_pNdbMutex);
+ delete pNdb;
+ pNdb = NULL ;
+ return NULL;
+}
+
+
+extern "C" void* NdbThreadFuncDelete(void* pArg)
+{
+ myRandom48Init((long int)NdbTick_CurrentMillisecond());
+ unsigned nSucc = 0;
+ unsigned nFail = 0;
+ Ndb* pNdb = NULL ;
+ pNdb = new Ndb("TEST_DB");
+ VerifyMethodInt(pNdb, init());
+ VerifyMethodInt(pNdb, waitUntilReady());
+
+ while(NdbMutex_Trylock(g_pNdbMutex)) {
+ Uint32 nWarehouse = myRandom48(g_nWarehouseCount);
+ NdbConnection* pNdbConnection = NULL ;
+ VerifyMethodPtr(pNdbConnection, pNdb, startTransaction());
+ CHK_TR(pNdbConnection) ; // epaulsa
+ NdbOperation* pNdbOperationW = NULL ;
+ VerifyMethodPtr(pNdbOperationW, pNdbConnection, getNdbOperation(c_szWarehouse));
+ VerifyMethodInt(pNdbOperationW, deleteTuple());
+ VerifyMethodInt(pNdbOperationW, equal(c_szWarehouseNumber, nWarehouse));
+ for(Uint32 nDistrict=0; nDistrict<g_nDistrictPerWarehouse; ++nDistrict) {
+ NdbOperation* pNdbOperationD = NULL ;
+ VerifyMethodPtr(pNdbOperationD, pNdbConnection, getNdbOperation(c_szDistrict));
+ VerifyMethodInt(pNdbOperationD, deleteTuple());
+ VerifyMethodInt(pNdbOperationD, equal(c_szDistrictWarehouseNumber, nWarehouse));
+ VerifyMethodInt(pNdbOperationD, equal(c_szDistrictNumber, nDistrict));
+ }
+ int iExec = pNdbConnection->execute(Commit);
+ int iError = pNdbConnection->getNdbError().code;
+ CommitStatusType cs = pNdbConnection->commitStatus();
+
+ if(iExec<0 && iError!=0 && iError!=266 && iError!=626) {
+ ReportMethodInt(iExec, pNdbConnection, "pNdbConnection", "execute(Commit)", __FILE__, __LINE__);
+ }
+ if(iExec==0) {
+ ++nSucc;
+ } else {
+ ++nFail;
+ }
+ VerifyMethodVoid(pNdb, closeTransaction(pNdbConnection));
+ }
+ ndbout << "delete: " << nSucc << " succeeded, " << nFail << " failed " << endl;
+ NdbMutex_Unlock(g_pNdbMutex);
+ delete pNdb;
+ pNdb = NULL ;
+ return NULL;
+}
+
+
+extern "C" void* NdbThreadFuncRead(void* pArg)
+{
+ myRandom48Init((long int)NdbTick_CurrentMillisecond());
+ unsigned nSucc = 0;
+ unsigned nFail = 0;
+ NdbRecAttr** ppNdbRecAttrDSum = new NdbRecAttr*[g_nDistrictPerWarehouse];
+ NdbRecAttr** ppNdbRecAttrDCnt = new NdbRecAttr*[g_nDistrictPerWarehouse];
+ Ndb* pNdb = NULL ;
+ pNdb = new Ndb("TEST_DB");
+ VerifyMethodInt(pNdb, init());
+ VerifyMethodInt(pNdb, waitUntilReady());
+
+ while(NdbMutex_Trylock(g_pNdbMutex)) {
+ Uint32 nWarehouse = myRandom48(g_nWarehouseCount);
+ NdbConnection* pNdbConnection = NULL ;
+ VerifyMethodPtr(pNdbConnection, pNdb, startTransaction());
+ CHK_TR(pNdbConnection) ; // epaulsa
+ NdbOperation* pNdbOperationW = NULL ;
+ VerifyMethodPtr(pNdbOperationW, pNdbConnection, getNdbOperation(c_szWarehouse));
+ VerifyMethodInt(pNdbOperationW, readTuple());
+ VerifyMethodInt(pNdbOperationW, equal(c_szWarehouseNumber, nWarehouse));
+ NdbRecAttr* pNdbRecAttrWSum;
+ VerifyMethodPtr(pNdbRecAttrWSum, pNdbOperationW, getValue(c_szWarehouseSum, 0));
+ NdbRecAttr* pNdbRecAttrWCnt;
+ VerifyMethodPtr(pNdbRecAttrWCnt, pNdbOperationW, getValue(c_szWarehouseCount, 0));
+ for(Uint32 nDistrict=0; nDistrict<g_nDistrictPerWarehouse; ++nDistrict) {
+ NdbOperation* pNdbOperationD = NULL ;
+ VerifyMethodPtr(pNdbOperationD, pNdbConnection, getNdbOperation(c_szDistrict));
+ VerifyMethodInt(pNdbOperationD, readTuple());
+ VerifyMethodInt(pNdbOperationD, equal(c_szDistrictWarehouseNumber, nWarehouse));
+ VerifyMethodInt(pNdbOperationD, equal(c_szDistrictNumber, nDistrict));
+ VerifyMethodPtr(ppNdbRecAttrDSum[nDistrict], pNdbOperationD, getValue(c_szDistrictSum, 0));
+ VerifyMethodPtr(ppNdbRecAttrDCnt[nDistrict], pNdbOperationD, getValue(c_szDistrictCount, 0));
+ }
+ int iExec = pNdbConnection->execute(Commit);
+ int iError = pNdbConnection->getNdbError().code;
+ CommitStatusType cs = pNdbConnection->commitStatus();
+ if(iExec<0 && iError!=0 && iError!=266 && iError!=626) {
+ ReportMethodInt(iExec, pNdbConnection, "pNdbConnection", "execute(Commit)", __FILE__, __LINE__);
+ }
+ if(iExec==0) {
+ Uint32 nSum = 0;
+ Uint32 nCnt = 0;
+ for(Uint32 nDistrict=0; nDistrict<g_nDistrictPerWarehouse; ++nDistrict) {
+ nSum += ppNdbRecAttrDSum[nDistrict]->u_32_value();
+ nCnt += ppNdbRecAttrDCnt[nDistrict]->u_32_value();
+ }
+ if(nSum!=pNdbRecAttrWSum->u_32_value()
+ || nCnt!=g_nDistrictPerWarehouse*pNdbRecAttrWCnt->u_32_value()) {
+ ndbout << "INCONSISTENT!" << endl;
+ ndbout << "iExec==" << iExec << endl;
+ ndbout << "iError==" << iError << endl;
+ ndbout << endl;
+ ndbout << c_szWarehouseSum << "==" << pNdbRecAttrWSum->u_32_value() << ", ";
+ ndbout << c_szWarehouseCount << "==" << pNdbRecAttrWCnt->u_32_value() << endl;
+ ndbout << "nSum==" << nSum << ", nCnt=" << nCnt << endl;
+ for(Uint32 nDistrict=0; nDistrict<g_nDistrictPerWarehouse; ++nDistrict) {
+ ndbout << c_szDistrictSum << "[" << nDistrict << "]==" << ppNdbRecAttrDSum[nDistrict]->u_32_value() << ", ";
+ ndbout << c_szDistrictCount << "[" << nDistrict << "]==" << ppNdbRecAttrDCnt[nDistrict]->u_32_value() << endl;
+ }
+ VerifyMethodVoid(pNdb, closeTransaction(pNdbConnection));
+ delete pNdb; pNdb = NULL ;
+ delete[] ppNdbRecAttrDSum; ppNdbRecAttrDSum = NULL ;
+ delete[] ppNdbRecAttrDCnt; ppNdbRecAttrDCnt = NULL ;
+ NDBT_ProgramExit(NDBT_FAILED);
+ }
+ ++nSucc;
+ } else {
+ ++nFail;
+ }
+ VerifyMethodVoid(pNdb, closeTransaction(pNdbConnection));
+ }
+ ndbout << "read: " << nSucc << " succeeded, " << nFail << " failed " << endl;
+ NdbMutex_Unlock(g_pNdbMutex);
+ delete pNdb; pNdb = NULL ;
+ delete[] ppNdbRecAttrDSum; ppNdbRecAttrDSum = NULL ;
+ delete[] ppNdbRecAttrDCnt; ppNdbRecAttrDCnt = NULL ;
+ return NULL;
+}
+
+
+NDB_COMMAND(acid, "acid", "acid", "acid", 65535)
+{
+ long nSeconds = 60;
+ int rc = NDBT_OK;
+
+ for(int i=1; i<argc; ++i) {
+ if(argv[i][0]=='-' || argv[i][0]=='/') {
+ switch(argv[i][1]) {
+ case 'w': g_nWarehouseCount=atol(argv[i]+2); break;
+ case 'd': g_nDistrictPerWarehouse=atol(argv[i]+2); break;
+ case 's': nSeconds=atol(argv[i]+2); break;
+ case 't': g_nThreadCount=atol(argv[i]+2); break;
+ default: ndbout << "invalid option" << endl; return 1;
+ }
+ } else {
+ ndbout << "invalid operand" << endl;
+ return 1;
+ }
+ }
+ ndbout << argv[0];
+ ndbout << " -w" << g_nWarehouseCount;
+ ndbout << " -d" << g_nDistrictPerWarehouse;
+ ndbout << " -s" << (int)nSeconds;
+ ndbout << " -t" << g_nThreadCount;
+ ndbout << endl;
+
+ Ndb* pNdb = NULL ;
+ pNdb = new Ndb("TEST_DB");
+ VerifyMethodInt(pNdb, init());
+ VerifyMethodInt(pNdb, waitUntilReady());
+
+ NdbSchemaCon* pNdbSchemaCon = NULL ;
+ VerifyMethodPtr(pNdbSchemaCon, pNdb, startSchemaTransaction());
+ if(!pNdbSchemaCon){
+ ndbout <<"startSchemaTransaction failed, exiting now" << endl ;
+ delete pNdb ;
+ NDBT_ProgramExit(NDBT_FAILED) ;
+ }
+ NdbSchemaOp* pNdbSchemaOp = NULL ;
+ VerifyMethodPtr(pNdbSchemaOp, pNdbSchemaCon, getNdbSchemaOp());
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ VerifyMethodInt(pNdbSchemaOp, createTable(
+ c_szWarehouse,
+ (4+4+4+12)*1.02*g_nWarehouseCount/1024+1,
+ TupleKey,
+ (4+14)*g_nWarehouseCount/8/1024+1,
+ All,
+ 6,
+ 78,
+ 80,
+ 1,
+ false));
+#else
+ VerifyMethodInt(pNdbSchemaOp, createTable(
+ c_szWarehouse,
+ (4+4+4+12)*1.02*g_nWarehouseCount/1024+1,
+ TupleKey,
+ (4+14)*g_nWarehouseCount/8/1024+1));
+#endif
+ VerifyMethodInt(pNdbSchemaOp, createAttribute(c_szWarehouseNumber, TupleKey, 32, 1, UnSigned, MMBased, false));
+ VerifyMethodInt(pNdbSchemaOp, createAttribute(c_szWarehouseSum, NoKey, 32, 1, UnSigned, MMBased, false));
+ VerifyMethodInt(pNdbSchemaOp, createAttribute(c_szWarehouseCount, NoKey, 32, 1, UnSigned, MMBased, false));
+ VerifyMethodInt(pNdbSchemaCon, execute());
+ VerifyMethodVoid(pNdb, closeSchemaTransaction(pNdbSchemaCon));
+
+ VerifyMethodPtr(pNdbSchemaCon, pNdb, startSchemaTransaction());
+ VerifyMethodPtr(pNdbSchemaOp, pNdbSchemaCon, getNdbSchemaOp());
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ VerifyMethodInt(pNdbSchemaOp, createTable(
+ c_szDistrict,
+ (4+4+4+4+12)*1.02*g_nWarehouseCount*g_nDistrictPerWarehouse/1024+1,
+ TupleKey,
+ (4+4+14)*g_nWarehouseCount*g_nDistrictPerWarehouse/8/1024+1,
+ All,
+ 6,
+ 78,
+ 80,
+ 1,
+ false));
+#else
+ VerifyMethodInt(pNdbSchemaOp, createTable(
+ c_szDistrict,
+ (4+4+4+4+12)*1.02*g_nWarehouseCount*g_nDistrictPerWarehouse/1024+1,
+ TupleKey,
+ (4+4+14)*g_nWarehouseCount*g_nDistrictPerWarehouse/8/1024+1));
+
+#endif
+ VerifyMethodInt(pNdbSchemaOp, createAttribute(c_szDistrictWarehouseNumber, TupleKey, 32, 1, UnSigned, MMBased, false));
+ VerifyMethodInt(pNdbSchemaOp, createAttribute(c_szDistrictNumber, TupleKey, 32, 1, UnSigned, MMBased, false));
+ VerifyMethodInt(pNdbSchemaOp, createAttribute(c_szDistrictSum, NoKey, 32, 1, UnSigned, MMBased, false));
+ VerifyMethodInt(pNdbSchemaOp, createAttribute(c_szDistrictCount, NoKey, 32, 1, UnSigned, MMBased, false));
+ VerifyMethodInt(pNdbSchemaCon, execute());
+ VerifyMethodVoid(pNdb, closeSchemaTransaction(pNdbSchemaCon));
+ g_pNdbMutex = NdbMutex_Create();
+ NdbMutex_Lock(g_pNdbMutex);
+
+ NdbThread** ppNdbThread = new NdbThread*[g_nThreadCount*4];
+ for(Uint32 nThread=0; nThread<g_nThreadCount; ++nThread) {
+ ppNdbThread[nThread*4+0] = NdbThread_Create(NdbThreadFuncInsert, 0, 65535, "insert",
+ NDB_THREAD_PRIO_LOW);
+ ppNdbThread[nThread*4+1] = NdbThread_Create(NdbThreadFuncUpdate, 0, 65535, "update",
+ NDB_THREAD_PRIO_LOW);
+ ppNdbThread[nThread*4+2] = NdbThread_Create(NdbThreadFuncDelete, 0, 65535, "delete",
+ NDB_THREAD_PRIO_LOW);
+ ppNdbThread[nThread*4+3] = NdbThread_Create(NdbThreadFuncRead, 0, 65535, "read",
+ NDB_THREAD_PRIO_LOW);
+ }
+
+ NdbSleep_SecSleep(nSeconds);
+ NdbMutex_Unlock(g_pNdbMutex);
+
+ void* pStatus;
+ for(Uint32 nThread=0; nThread<g_nThreadCount; ++nThread) {
+ NdbThread_WaitFor(ppNdbThread[nThread*4+0], &pStatus);
+ NdbThread_WaitFor(ppNdbThread[nThread*4+1], &pStatus);
+ NdbThread_WaitFor(ppNdbThread[nThread*4+2], &pStatus);
+ NdbThread_WaitFor(ppNdbThread[nThread*4+3], &pStatus);
+ }
+
+ NdbMutex_Destroy(g_pNdbMutex);
+ delete[] ppNdbThread;
+ delete pNdb;
+ return NDBT_ProgramExit(rc);
+}
+
diff --git a/ndb/test/ndbapi/acid2/Makefile b/ndb/test/ndbapi/acid2/Makefile
new file mode 100644
index 00000000000..69c9d409b9e
--- /dev/null
+++ b/ndb/test/ndbapi/acid2/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := acid2
+
+# Source files of non-templated classes (.C files)
+SOURCES = acid2.cpp TraceNdbApi.cpp VerifyNdbApi.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/acid2/TraceNdbApi.cpp b/ndb/test/ndbapi/acid2/TraceNdbApi.cpp
new file mode 100644
index 00000000000..bd43b15f2e6
--- /dev/null
+++ b/ndb/test/ndbapi/acid2/TraceNdbApi.cpp
@@ -0,0 +1,543 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbApi.hpp>
+#include <NdbOut.hpp>
+#include <NdbMutex.h>
+
+#include "TraceNdbApi.hpp"
+
+
+int g_nParamTrace;
+NdbMutex* g_pNdbMutexTrace = 0;
+
+
+void TraceBegin(void)
+{
+ if(!g_pNdbMutexTrace)
+ {
+ g_pNdbMutexTrace = NdbMutex_Create();
+ }
+ NdbMutex_Lock(g_pNdbMutexTrace);
+ g_nParamTrace = 0;
+}
+
+void TraceEnd(void)
+{
+ ndbout << endl;
+ g_nParamTrace = 0;
+ NdbMutex_Unlock(g_pNdbMutexTrace);
+}
+
+void TraceMethod(const char* szMethod)
+{
+ ndbout << "->" << szMethod << "(";
+ g_nParamTrace = 0;
+}
+
+void TraceParamComma(void)
+{
+ if(g_nParamTrace)
+ {
+ ndbout << ", ";
+ }
+ ++g_nParamTrace;
+}
+
+void TraceNdb(Ndb* pNdb)
+{
+ TraceParamComma();
+ ndbout << "((Ndb*)" << hex << (Uint32)pNdb << ")";
+}
+
+void TraceNdbSchemaCon(NdbSchemaCon* pNdbSchemaCon)
+{
+ TraceParamComma();
+ ndbout << "((NdbSchemaCon*)" << hex << (Uint32)pNdbSchemaCon << ")";
+}
+
+void TraceNdbSchemaOp(NdbSchemaOp* pNdbSchemaOp)
+{
+ TraceParamComma();
+ ndbout << "((NdbSchemaOp*)" << hex << (Uint32)pNdbSchemaOp << ")";
+}
+
+void TraceNdbConnection(const NdbConnection* pNdbConnection)
+{
+ TraceParamComma();
+ ndbout << "((NdbConnection*)" << hex << (Uint32)pNdbConnection << ")";
+}
+
+void TraceNdbOperation(NdbOperation* pNdbOperation)
+{
+ TraceParamComma();
+ ndbout << "((NdbOperation*)" << hex << (Uint32)pNdbOperation << ")";
+}
+
+void TraceNdbIndexOperation(NdbIndexOperation* pNdbIndexOperation)
+{
+ TraceParamComma();
+ ndbout << "((NdbIndexOperation*)" << hex << (Uint32)pNdbIndexOperation << ")";
+}
+
+void TraceNdbRecAttr(NdbRecAttr* pNdbRecAttr)
+{
+ TraceParamComma();
+ ndbout << "((NdbRecAttr*)" << hex << (Uint32)pNdbRecAttr << ")";
+}
+
+void TraceTable(Table* pTable)
+{
+ TraceParamComma();
+ ndbout << "((Table*)" << hex << (Uint32)pTable << ")";
+}
+
+void TraceString(const char* szParam)
+{
+ TraceParamComma();
+ ndbout << "\"" << szParam << "\"";
+}
+
+void TraceInt(const int i)
+{
+ TraceParamComma();
+ ndbout << "(int)" << dec << i;
+}
+
+void TraceUint32(const Uint32 n)
+{
+ TraceParamComma();
+ ndbout << "(Uint32)" << dec << n;
+}
+
+void TraceKeyType(const KeyType aKeyType)
+{
+ TraceParamComma();
+ switch(aKeyType)
+ {
+ case Undefined: ndbout << "Undefined"; break;
+ case NoKey: ndbout << "NoKey"; break;
+ case TupleKey: ndbout << "TupleKey"; break;
+ case TupleId: ndbout << "TupleId"; break;
+ default: ndbout << "(KeyType)" << aKeyType; break;
+ }
+}
+
+void TraceExecType(const ExecType aExecType)
+{
+ switch(aExecType)
+ {
+ case NoExecTypeDef: ndbout << "NoExecTypeDef"; break;
+ case Prepare: ndbout << "Prepare"; break;
+ case NoCommit: ndbout << "NoCommit"; break;
+ case Commit: ndbout << "Commit"; break;
+ case Rollback: ndbout << "Rollback"; break;
+ default: ndbout << "(ExecType)" << aExecType; break;
+ }
+}
+
+
+void TraceNdbError(const NdbError& err)
+{
+ TraceParamComma();
+ ndbout << "(NdbError)" << err;
+}
+
+
+
+void TraceVoid(void)
+{
+ ndbout << "void";
+}
+
+void TraceReturn(void)
+{
+ ndbout << "); // return ";
+ g_nParamTrace = 0;
+}
+
+
+// TraceNdbSchemaOp
+
+int CTraceNdbSchemaOp::createTable(const char* aTableName)
+{
+ int i = NdbSchemaOp::createTable(aTableName);
+ TraceBegin();
+ TraceNdbSchemaOp(this);
+ TraceMethod("createTable");
+ TraceString(aTableName);
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+int CTraceNdbSchemaOp::createAttribute(const char* aAttrName, KeyType aTupleyKey)
+{
+ int i = NdbSchemaOp::createAttribute(aAttrName, aTupleyKey);
+ TraceBegin();
+ TraceNdbSchemaOp(this);
+ TraceMethod("createAttribute");
+ TraceString(aAttrName);
+ TraceKeyType(aTupleyKey);
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+
+
+// TraceNdbSchemaCon
+
+CTraceNdbSchemaOp* CTraceNdbSchemaCon::getNdbSchemaOp()
+{
+ NdbSchemaOp* pNdbSchemaOp = NdbSchemaCon::getNdbSchemaOp();
+ TraceBegin();
+ TraceNdbSchemaCon(this);
+ TraceMethod("getNdbSchemaOp");
+ TraceReturn();
+ TraceNdbSchemaOp(pNdbSchemaOp);
+ TraceEnd();
+ return (CTraceNdbSchemaOp*)pNdbSchemaOp;
+}
+
+int CTraceNdbSchemaCon::execute()
+{
+ int i = NdbSchemaCon::execute();
+ TraceBegin();
+ TraceNdbSchemaCon(this);
+ TraceMethod("execute");
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+
+
+// TraceNdbRecAttr
+
+Uint32 CTraceNdbRecAttr::u_32_value()
+{
+ Uint32 n = NdbRecAttr::u_32_value();
+ TraceBegin();
+ TraceNdbRecAttr(this);
+ TraceMethod("u_32_value");
+ TraceReturn();
+ TraceUint32(n);
+ TraceEnd();
+ return n;
+}
+
+
+
+// TraceNdbOperation
+
+int CTraceNdbOperation::insertTuple()
+{
+ int i = NdbOperation::insertTuple();
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("insertTuple");
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+int CTraceNdbOperation::updateTuple()
+{
+ int i = NdbOperation::updateTuple();
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("updateTuple");
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+int CTraceNdbOperation::interpretedUpdateTuple()
+{
+ int i = NdbOperation::interpretedUpdateTuple();
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("interpretedUpdateTuple");
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+int CTraceNdbOperation::readTuple()
+{
+ int i = NdbOperation::readTuple();
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("readTuple");
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+
+int CTraceNdbOperation::readTupleExclusive()
+{
+ int i = NdbOperation::readTupleExclusive();
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("readTupleExclusive");
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+
+int CTraceNdbOperation::deleteTuple()
+{
+ int i = NdbOperation::deleteTuple();
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("deleteTuple");
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+int CTraceNdbOperation::equal(const char* anAttrName, Uint32 aValue)
+{
+ int i = NdbOperation::equal(anAttrName, aValue);
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("equal");
+ TraceString(anAttrName);
+ TraceUint32(aValue);
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+int CTraceNdbOperation::setValue(const char* anAttrName, Uint32 aValue)
+{
+ int i = NdbOperation::setValue(anAttrName, aValue);
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("setValue");
+ TraceString(anAttrName);
+ TraceUint32(aValue);
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+int CTraceNdbOperation::incValue(const char* anAttrName, Uint32 aValue)
+{
+ int i = NdbOperation::incValue(anAttrName, aValue);
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("incValue");
+ TraceString(anAttrName);
+ TraceUint32(aValue);
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+CTraceNdbRecAttr* CTraceNdbOperation::getValue(const char* anAttrName)
+{
+ NdbRecAttr* pNdbRecAttr = NdbOperation::getValue(anAttrName);
+ TraceBegin();
+ TraceNdbOperation(this);
+ TraceMethod("getValue");
+ TraceString(anAttrName);
+ TraceReturn();
+ TraceNdbRecAttr(pNdbRecAttr);
+ TraceEnd();
+ return (CTraceNdbRecAttr*)pNdbRecAttr;
+}
+
+
+// TraceNdbIndexOperation
+
+int CTraceNdbIndexOperation::equal(const char* anAttrName, Uint32 aValue)
+{
+ int i = NdbIndexOperation::equal(anAttrName, aValue);
+ TraceBegin();
+ TraceNdbIndexOperation(this);
+ TraceMethod("equal");
+ TraceString(anAttrName);
+ TraceUint32(aValue);
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+
+int CTraceNdbIndexOperation::incValue(const char* anAttrName, Uint32 aValue)
+{
+ int i = NdbIndexOperation::incValue(anAttrName, aValue);
+ TraceBegin();
+ TraceNdbIndexOperation(this);
+ TraceMethod("incValue");
+ TraceString(anAttrName);
+ TraceUint32(aValue);
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+
+CTraceNdbRecAttr* CTraceNdbIndexOperation::getValue(const char* anAttrName)
+{
+ NdbRecAttr* pNdbRecAttr = NdbIndexOperation::getValue(anAttrName);
+ TraceBegin();
+ TraceNdbIndexOperation(this);
+ TraceMethod("getValue");
+ TraceString(anAttrName);
+ TraceReturn();
+ TraceNdbRecAttr(pNdbRecAttr);
+ TraceEnd();
+ return (CTraceNdbRecAttr*)pNdbRecAttr;
+}
+
+
+// TraceNdbConnection
+
+CTraceNdbOperation* CTraceNdbConnection::getNdbOperation(const char* aTableName)
+{
+ NdbOperation* pNdbOperation = NdbConnection::getNdbOperation(aTableName);
+ TraceBegin();
+ TraceNdbConnection(this);
+ TraceMethod("getNdbOperation");
+ TraceString(aTableName);
+ TraceReturn();
+ TraceNdbOperation(pNdbOperation);
+ TraceEnd();
+ return (CTraceNdbOperation*)pNdbOperation;
+}
+
+CTraceNdbIndexOperation* CTraceNdbConnection::getNdbIndexOperation(const char* anIndexName, const char* aTableName)
+{
+ NdbIndexOperation* pNdbIndexOperation = NdbConnection::getNdbIndexOperation(anIndexName, aTableName);
+ TraceBegin();
+ TraceNdbConnection(this);
+ TraceMethod("getNdbIndexOperation");
+ TraceString(anIndexName);
+ TraceString(aTableName);
+ TraceReturn();
+ TraceNdbIndexOperation(pNdbIndexOperation);
+ TraceEnd();
+ return (CTraceNdbIndexOperation*)pNdbIndexOperation;
+}
+
+int CTraceNdbConnection::execute(ExecType aTypeOfExec)
+{
+ int i = NdbConnection::execute(aTypeOfExec);
+ TraceBegin();
+ TraceNdbConnection(this);
+ TraceMethod("execute");
+ TraceExecType(aTypeOfExec);
+ TraceReturn();
+ TraceInt(i);
+ TraceEnd();
+ return i;
+}
+
+const NdbError & CTraceNdbConnection::getNdbError(void) const
+{
+ const NdbError& err = NdbConnection::getNdbError();
+ TraceBegin();
+ TraceNdbConnection(this);
+ TraceMethod("getNdbError");
+ TraceReturn();
+ TraceNdbError(err);
+ TraceEnd();
+ return err;
+}
+
+
+
+// TraceNdb
+
+CTraceNdb::CTraceNdb(const char* aDataBase)
+: Ndb(aDataBase)
+{
+ TraceBegin();
+ TraceNdb(this);
+ TraceMethod("Ndb");
+ TraceString(aDataBase);
+ TraceReturn();
+ TraceVoid();
+ TraceEnd();
+}
+
+CTraceNdbSchemaCon* CTraceNdb::startSchemaTransaction()
+{
+ NdbSchemaCon* pNdbSchemaCon = Ndb::startSchemaTransaction();
+ TraceBegin();
+ TraceNdb(this);
+ TraceMethod("startSchemaTransaction");
+ TraceReturn();
+ TraceNdbSchemaCon(pNdbSchemaCon);
+ TraceEnd();
+ return (CTraceNdbSchemaCon*)pNdbSchemaCon;
+}
+
+void CTraceNdb::closeSchemaTransaction(CTraceNdbSchemaCon* aSchemaCon)
+{
+ Ndb::closeSchemaTransaction(aSchemaCon);
+ TraceBegin();
+ TraceNdb(this);
+ TraceMethod("closeSchemaTransaction");
+ TraceReturn();
+ TraceVoid();
+ TraceEnd();
+}
+
+CTraceNdbConnection* CTraceNdb::startTransaction()
+{
+ NdbConnection* pNdbConnection = Ndb::startTransaction();
+ TraceBegin();
+ TraceNdb(this);
+ TraceMethod("startTransaction");
+ TraceReturn();
+ TraceNdbConnection(pNdbConnection);
+ TraceEnd();
+ return (CTraceNdbConnection*)pNdbConnection;
+}
+
+void CTraceNdb::closeTransaction(CTraceNdbConnection* aConnection)
+{
+ Ndb::closeTransaction(aConnection);
+ TraceBegin();
+ TraceNdb(this);
+ TraceMethod("closeTransaction");
+ TraceNdbConnection(aConnection);
+ TraceReturn();
+ TraceVoid();
+ TraceEnd();
+}
+
+
diff --git a/ndb/test/ndbapi/acid2/TraceNdbApi.hpp b/ndb/test/ndbapi/acid2/TraceNdbApi.hpp
new file mode 100644
index 00000000000..2bd4eab6b70
--- /dev/null
+++ b/ndb/test/ndbapi/acid2/TraceNdbApi.hpp
@@ -0,0 +1,132 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#ifndef TraceNdbApi_hpp
+#define TraceNdbApi_hpp
+
+
+class CTraceNdbSchemaOp : public NdbSchemaOp
+{
+public:
+ int createTable(const char* aTableName);
+ int createAttribute(const char* aAttrName, KeyType aTupleyKey);
+};
+
+
+
+class CTraceNdbSchemaCon : public NdbSchemaCon
+{
+public:
+ CTraceNdbSchemaOp* getNdbSchemaOp();
+ int execute();
+};
+
+
+
+class CTraceNdbRecAttr : public NdbRecAttr
+{
+public:
+ Uint32 u_32_value();
+};
+
+
+class CTraceNdbOperation : public NdbOperation
+{
+public:
+ int insertTuple();
+ int updateTuple();
+ int interpretedUpdateTuple();
+ int readTuple();
+ int readTupleExclusive();
+ int deleteTuple();
+ int equal(const char* anAttrName, Uint32 aValue);
+ int setValue(const char* anAttrName, Uint32 aValue);
+ int incValue(const char* anAttrName, Uint32 aValue);
+ CTraceNdbRecAttr* getValue(const char* anAttrName);
+
+};
+
+
+class CTraceNdbIndexOperation : public NdbIndexOperation
+{
+public:
+ int insertTuple();
+ int updateTuple();
+ int interpretedUpdateTuple();
+ int readTuple();
+ int readTupleExclusive();
+ int deleteTuple();
+ int equal(const char* anAttrName, Uint32 aValue);
+ int setValue(const char* anAttrName, Uint32 aValue);
+ int incValue(const char* anAttrName, Uint32 aValue);
+ CTraceNdbRecAttr* getValue(const char* anAttrName);
+};
+
+
+
+class CTraceNdbConnection : public NdbConnection
+{
+public:
+ CTraceNdbOperation* getNdbOperation(const char* aTableName);
+ CTraceNdbIndexOperation* getNdbIndexOperation(const char* anIndexName, const char* aTableName);
+
+ int execute(ExecType aTypeOfExec);
+
+ int execute_ok(ExecType aTypeOfExec)
+ {
+ return execute(aTypeOfExec);
+ };
+
+ const NdbError & getNdbError(void) const;
+};
+
+
+
+class CTraceNdbDictionary : public NdbDictionary
+{
+public:
+ class CTraceTable : public Table
+ {
+ };
+
+ class CTraceIndex : public Index
+ {
+ };
+
+ class CTraceColumn : public Column
+ {
+ };
+
+ int createTable(const CTraceTable &);
+ int createIndex(const CTraceIndex &);
+};
+
+
+
+class CTraceNdb : public Ndb
+{
+public:
+ CTraceNdb(const char* aDataBase);
+ CTraceNdbSchemaCon* startSchemaTransaction();
+ void closeSchemaTransaction(CTraceNdbSchemaCon* aSchemaCon);
+ CTraceNdbConnection* startTransaction();
+ void closeTransaction(CTraceNdbConnection* aConnection);
+};
+
+
+
+#endif // TraceNdbApi_hpp
diff --git a/ndb/test/ndbapi/acid2/VerifyNdbApi.cpp b/ndb/test/ndbapi/acid2/VerifyNdbApi.cpp
new file mode 100644
index 00000000000..79645827e2c
--- /dev/null
+++ b/ndb/test/ndbapi/acid2/VerifyNdbApi.cpp
@@ -0,0 +1,151 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbApi.hpp>
+#include <NdbOut.hpp>
+#include <NdbMutex.h>
+
+#include "VerifyNdbApi.hpp"
+
+
+NdbMutex* g_pNdbMutexVerify = 0;
+
+
+void VerifyBegin(void)
+{
+ if(!g_pNdbMutexVerify)
+ {
+ g_pNdbMutexVerify = NdbMutex_Create();
+ }
+ NdbMutex_Lock(g_pNdbMutexVerify);
+}
+
+void VerifyEnd(void)
+{
+ NdbMutex_Unlock(g_pNdbMutexVerify);
+}
+
+
+
+void CVerifyNdbSchemaOp::VerifyIntError(const int i, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbSchemaOp::" << szMethod << " returned " << dec << i;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdbSchemaCon::VerifyIntError(const int i, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbSchemaCon::" << szMethod << " returned " << dec << i;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdbSchemaCon::VerifyPtrError(void* p, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbSchemaCon::" << szMethod << " returned " << hex << (Uint32)p;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdbRecAttr::VerifyValueError(const int iNull, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbRecAttr::" << szMethod << " : isNULL() returned " << dec << iNull;
+ ndbout << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdbOperation::VerifyIntError(const int i, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbOperation::" << szMethod << " returned " << dec << i;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdbOperation::VerifyPtrError(void* p, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbOperation::" << szMethod << " returned " << hex << (Uint32)p;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdbIndexOperation::VerifyIntError(const int i, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbIndexOperation::" << szMethod << " returned " << dec << i;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdbIndexOperation::VerifyPtrError(void* p, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbIndexOperation::" << szMethod << " returned " << hex << (Uint32)p;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdbConnection::VerifyIntError(const int i, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbConnection::" << szMethod << " returned " << dec << i;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdbConnection::VerifyPtrError(void* p, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "NdbConnection::" << szMethod << " returned " << hex << (Uint32)p;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdb::VerifyPtrError(void* p, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "Ndb::" << szMethod << " returned " << hex << (Uint32)p;
+ ndbout << " : " << dec << getNdbError().code << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
+void CVerifyNdb::VerifyVoidError(const int iCode, const char* szMethod)
+{
+ VerifyBegin();
+ ndbout << "Ndb::" << szMethod << " : getNdbError().code returned " << dec << iCode;
+ ndbout << " : " << getNdbError().message << endl;
+ VerifyEnd();
+}
+
+
diff --git a/ndb/test/ndbapi/acid2/VerifyNdbApi.hpp b/ndb/test/ndbapi/acid2/VerifyNdbApi.hpp
new file mode 100644
index 00000000000..4a5b8cc8111
--- /dev/null
+++ b/ndb/test/ndbapi/acid2/VerifyNdbApi.hpp
@@ -0,0 +1,466 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#ifndef VerifyNdbApi_hpp
+#define VerifyNdbApi_hpp
+
+
+class CVerifyNdbSchemaOp : public NdbSchemaOp
+{
+public:
+ int createTable(const char* aTableName)
+ {
+ int i = NdbSchemaOp::createTable(aTableName);
+ VerifyInt(i, "createTable");
+ return i;
+ };
+
+ int createAttribute(const char* aAttrName, KeyType aTupleyKey)
+ {
+ int i = NdbSchemaOp::createAttribute(aAttrName, aTupleyKey);
+ VerifyInt(i, "createAttribute");
+ return i;
+ };
+
+private:
+ void VerifyInt(const int i, const char* szMethod)
+ {
+ if(i)
+ {
+ VerifyIntError(i, szMethod);
+ }
+ }
+
+ void VerifyIntError(const int i, const char* szMethod);
+};
+
+
+
+class CVerifyNdbSchemaCon : public NdbSchemaCon
+{
+public:
+ CVerifyNdbSchemaOp* getNdbSchemaOp()
+ {
+ NdbSchemaOp* p = NdbSchemaCon::getNdbSchemaOp();
+ VerifyPtr(p, "getNdbSchemaOp");
+ return (CVerifyNdbSchemaOp*)p;
+ };
+
+ int execute()
+ {
+ int i = NdbSchemaCon::execute();
+ VerifyInt(i, "execute");
+ return i;
+ };
+
+private:
+ void VerifyInt(const int i, const char* szMethod)
+ {
+ if(i)
+ {
+ VerifyIntError(i, szMethod);
+ }
+ }
+
+ void VerifyPtr(void* p, const char* szMethod)
+ {
+ if(!p)
+ {
+ VerifyPtrError(p, szMethod);
+ }
+ }
+
+ void VerifyIntError(const int i, const char* szMethod);
+ void VerifyPtrError(void* p, const char* szMethod);
+};
+
+
+
+class CVerifyNdbRecAttr : public NdbRecAttr
+{
+public:
+ Uint32 u_32_value()
+ {
+ Uint32 n = NdbRecAttr::u_32_value();
+ VerifyValue("u_32_value");
+ return n;
+ };
+
+private:
+ void VerifyValue(const char* szMethod)
+ {
+ int iNull = NdbRecAttr::isNULL();
+ if(iNull)
+ {
+ VerifyValueError(iNull, szMethod);
+ }
+ };
+
+ void VerifyValueError(const int iNull, const char* szMethod);
+};
+
+
+class CVerifyNdbOperation : public NdbOperation
+{
+public:
+ int insertTuple()
+ {
+ int i = NdbOperation::insertTuple();
+ VerifyInt(i, "insertTuple");
+ return i;
+ };
+
+ int updateTuple()
+ {
+ int i = NdbOperation::updateTuple();
+ VerifyInt(i, "updateTuple");
+ return i;
+ };
+
+ int interpretedUpdateTuple()
+ {
+ int i = NdbOperation::interpretedUpdateTuple();
+ VerifyInt(i, "interpretedUpdateTuple");
+ return i;
+ }
+
+ int readTuple()
+ {
+ int i = NdbOperation::readTuple();
+ VerifyInt(i, "readTuple");
+ return i;
+ }
+
+ int readTupleExclusive()
+ {
+ int i = NdbOperation::readTupleExclusive();
+ VerifyInt(i, "readTupleExclusive");
+ return i;
+ }
+
+ int deleteTuple()
+ {
+ int i = NdbOperation::deleteTuple();
+ VerifyInt(i, "deleteTuple");
+ return i;
+ }
+
+ int equal(const char* anAttrName, Uint32 aValue)
+ {
+ int i = NdbOperation::equal(anAttrName, aValue);
+ VerifyInt(i, "equal");
+ return i;
+ }
+
+ int setValue(const char* anAttrName, Uint32 aValue)
+ {
+ int i = NdbOperation::setValue(anAttrName, aValue);
+ VerifyInt(i, "setValue");
+ return i;
+ }
+
+ int incValue(const char* anAttrName, Uint32 aValue)
+ {
+ int i = NdbOperation::incValue(anAttrName, aValue);
+ VerifyInt(i, "incValue");
+ return i;
+ }
+
+ CVerifyNdbRecAttr* getValue(const char* anAttrName)
+ {
+ NdbRecAttr* p = NdbOperation::getValue(anAttrName);
+ VerifyPtr(p, "getValue");
+ return (CVerifyNdbRecAttr*)p;
+ }
+
+
+private:
+ void VerifyInt(const int i, const char* szMethod)
+ {
+ if(i)
+ {
+ VerifyIntError(i, szMethod);
+ }
+ }
+
+ void VerifyPtr(void* p, const char* szMethod)
+ {
+ if(!p)
+ {
+ VerifyPtrError(p, szMethod);
+ }
+ }
+
+ void VerifyIntError(const int i, const char* szMethod);
+ void VerifyPtrError(void* p, const char* szMethod);
+};
+
+
+class CVerifyNdbIndexOperation : public NdbIndexOperation
+{
+public:
+ int insertTuple()
+ {
+ int i = NdbIndexOperation::insertTuple();
+ VerifyInt(i, "insertTuple");
+ return i;
+ };
+
+ int updateTuple()
+ {
+ int i = NdbIndexOperation::updateTuple();
+ VerifyInt(i, "updateTuple");
+ return i;
+ };
+
+ int interpretedUpdateTuple()
+ {
+ int i = NdbIndexOperation::interpretedUpdateTuple();
+ VerifyInt(i, "interpretedUpdateTuple");
+ return i;
+ }
+
+ int readTuple()
+ {
+ int i = NdbIndexOperation::readTuple();
+ VerifyInt(i, "readTuple");
+ return i;
+ }
+
+ int readTupleExclusive()
+ {
+ int i = NdbIndexOperation::readTupleExclusive();
+ VerifyInt(i, "readTupleExclusive");
+ return i;
+ }
+
+ int deleteTuple()
+ {
+ int i = NdbIndexOperation::deleteTuple();
+ VerifyInt(i, "deleteTuple");
+ return i;
+ }
+
+ int equal(const char* anAttrName, Uint32 aValue)
+ {
+ int i = NdbIndexOperation::equal(anAttrName, aValue);
+ VerifyInt(i, "equal");
+ return i;
+ }
+
+ int setValue(const char* anAttrName, Uint32 aValue)
+ {
+ int i = NdbIndexOperation::setValue(anAttrName, aValue);
+ VerifyInt(i, "setValue");
+ return i;
+ }
+
+ int incValue(const char* anAttrName, Uint32 aValue)
+ {
+ int i = NdbIndexOperation::incValue(anAttrName, aValue);
+ VerifyInt(i, "incValue");
+ return i;
+ }
+
+ CVerifyNdbRecAttr* getValue(const char* anAttrName)
+ {
+ NdbRecAttr* p = NdbIndexOperation::getValue(anAttrName);
+ VerifyPtr(p, "getValue");
+ return (CVerifyNdbRecAttr*)p;
+ }
+
+
+private:
+ void VerifyInt(const int i, const char* szMethod)
+ {
+ if(i)
+ {
+ VerifyIntError(i, szMethod);
+ }
+ }
+
+ void VerifyPtr(void* p, const char* szMethod)
+ {
+ if(!p)
+ {
+ VerifyPtrError(p, szMethod);
+ }
+ }
+
+ void VerifyIntError(const int i, const char* szMethod);
+ void VerifyPtrError(void* p, const char* szMethod);
+};
+
+
+class CVerifyNdbConnection : public NdbConnection
+{
+public:
+ CVerifyNdbOperation* getNdbOperation(const char* aTableName)
+ {
+ NdbOperation* p = NdbConnection::getNdbOperation(aTableName);
+ VerifyPtr(p, "getNdbOperation");
+ return (CVerifyNdbOperation*)p;
+ }
+
+ CVerifyNdbIndexOperation* getNdbIndexOperation(const char* anIndexName, const char* aTableName)
+ {
+ NdbIndexOperation* p = NdbConnection::getNdbIndexOperation(anIndexName, aTableName);
+ VerifyPtr(p, "getNdbIndexOperation");
+ return (CVerifyNdbIndexOperation*)p;
+ }
+
+ int execute(ExecType aTypeOfExec)
+ {
+ int i = NdbConnection::execute(aTypeOfExec);
+ VerifyInt(i, "execute");
+ return i;
+ }
+
+ int execute_ok(ExecType aTypeOfExec)
+ {
+ int iExec = NdbConnection::execute(aTypeOfExec);
+ NdbError err = NdbConnection::getNdbError();
+ int iCode = err.code;
+ if(iExec
+ && ((aTypeOfExec==NoCommit && iCode!=0)
+ || (aTypeOfExec==Commit && iCode!=626 && iCode!=630)))
+ {
+ VerifyInt(iExec, "execute");
+ }
+ return iExec;
+ }
+
+
+private:
+ void VerifyInt(const int i, const char* szMethod)
+ {
+ if(i)
+ {
+ VerifyIntError(i, szMethod);
+ }
+ }
+
+ void VerifyPtr(void* p, const char* szMethod)
+ {
+ if(!p)
+ {
+ VerifyPtrError(p, szMethod);
+ }
+ }
+
+ void VerifyIntError(const int i, const char* szMethod);
+ void VerifyPtrError(void* p, const char* szMethod);
+};
+
+
+//class CVerifyTable : public NdbDictionary::Table
+//{
+//public:
+//};
+
+
+class CVerifyNdbDictionary : public NdbDictionary
+{
+public:
+ class CVerifyTable : public Table
+ {
+ public:
+ private:
+ };
+
+ class CVerifyIndex : public Index
+ {
+ public:
+ private:
+ };
+
+ class CVerifyColumn : public Column
+ {
+ public:
+ private:
+ };
+
+ int createTable(const CVerifyTable &);
+ int createIndex(const CVerifyIndex &);
+
+
+private:
+};
+
+
+class CVerifyNdb : public Ndb
+{
+public:
+ CVerifyNdb(const char* aDataBase)
+ : Ndb(aDataBase)
+ {
+ VerifyVoid("Ndb");
+ };
+
+ CVerifyNdbSchemaCon* startSchemaTransaction()
+ {
+ NdbSchemaCon* p = Ndb::startSchemaTransaction();
+ VerifyPtr(p, "startSchemaTransaction");
+ return (CVerifyNdbSchemaCon*)p;
+ };
+
+ void closeSchemaTransaction(CVerifyNdbSchemaCon* aSchemaCon)
+ {
+ Ndb::closeSchemaTransaction(aSchemaCon);
+ VerifyVoid("closeSchemaTransaction");
+ };
+
+ CVerifyNdbConnection* startTransaction()
+ {
+ NdbConnection* p = Ndb::startTransaction();
+ VerifyPtr(p, "startTransaction");
+ return (CVerifyNdbConnection*)p;
+ };
+
+ void closeTransaction(CVerifyNdbConnection* aConnection)
+ {
+ Ndb::closeTransaction(aConnection);
+ VerifyVoid("closeTransaction");
+ };
+
+
+private:
+ void VerifyPtr(void* p, const char* szMethod)
+ {
+ if(!p)
+ {
+ VerifyPtrError(p, szMethod);
+ }
+ }
+
+ void VerifyVoid(const char* szMethod)
+ {
+ NdbError err = Ndb::getNdbError();
+ int iCode = err.code;
+ if(iCode)
+ {
+ VerifyVoidError(iCode, szMethod);
+ }
+ }
+
+ void VerifyPtrError(void* p, const char* szMethod);
+ void VerifyVoidError(const int iCode, const char* szMethod);
+};
+
+
+
+#endif // VerifyNdbApi_hpp
diff --git a/ndb/test/ndbapi/acid2/acid2.cpp b/ndb/test/ndbapi/acid2/acid2.cpp
new file mode 100644
index 00000000000..5835b76453e
--- /dev/null
+++ b/ndb/test/ndbapi/acid2/acid2.cpp
@@ -0,0 +1,693 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbApi.hpp>
+#include <NdbOut.hpp>
+#include <NdbStdio.h>
+#include <NdbThread.h>
+#include <NdbSleep.h>
+#include <NdbMutex.h>
+
+#include "TraceNdbApi.hpp"
+#include "VerifyNdbApi.hpp"
+
+
+#define Ndb CTraceNdb
+#define NdbSchemaCon CTraceNdbSchemaCon
+#define NdbSchemaOp CTraceNdbSchemaOp
+#define NdbConnection CTraceNdbConnection
+#define NdbOperation CTraceNdbOperation
+#define NdbIndexOperation CTraceNdbIndexOperation
+#define NdbRecAttr CTraceNdbRecAttr
+#define Table CTraceTable
+#define Index CTraceIndex
+#define Column CTraceColumn
+#define NdbDictionary CTraceNdbDictionary
+
+/*
+#define Ndb CVerifyNdb
+#define NdbSchemaCon CVerifyNdbSchemaCon
+#define NdbSchemaOp CVerifyNdbSchemaOp
+#define NdbConnection CVerifyNdbConnection
+#define NdbOperation CVerifyNdbOperation
+#define NdbIndexOperation CVerifyNdbIndexOperation
+#define NdbRecAttr CVerifyNdbRecAttr
+#define Table CVerifyTable
+#define Index CVerifyIndex
+#define Column CVerifyColumn
+#define NdbDictionary CVerifyNdbDictionary
+*/
+
+NdbMutex* g_pNdbMutexStop = 0;
+Uint32 g_nPart = 1;
+Uint32 g_nTable = 1;
+Uint32 g_nTuple = 1;
+Uint32 g_nAttribute = 1;
+char* g_szTable = 0;
+char* g_szIndex = 0;
+char* g_szAttribute = 0;
+bool g_bVerify = false;
+bool g_bUseIndex = false;
+
+
+
+#define N 624
+#define M 397
+#define MATRIX_A 0x9908b0df
+#define UPPER_MASK 0x80000000
+#define LOWER_MASK 0x7fffffff
+
+#define TEMPERING_MASK_B 0x9d2c5680
+#define TEMPERING_MASK_C 0xefc60000
+#define TEMPERING_SHIFT_U(y) (y >> 11)
+#define TEMPERING_SHIFT_S(y) (y << 7)
+#define TEMPERING_SHIFT_T(y) (y << 15)
+#define TEMPERING_SHIFT_L(y) (y >> 18)
+
+
+class MT19937
+{
+public:
+ MT19937(void);
+ void sgenrand(unsigned long seed);
+ unsigned long genrand(void);
+
+private:
+ unsigned long mt[N];
+ int mti;
+ unsigned long mag01[2];
+};
+
+
+MT19937::MT19937(void)
+{
+ mti = N+1;
+ mag01[0] = 0x0;
+ mag01[1] = MATRIX_A;
+ sgenrand(4357);
+}
+
+
+void MT19937::sgenrand(unsigned long seed)
+{
+ mt[0]= seed & 0xffffffff;
+ for (mti=1; mti<N; mti++)
+ mt[mti] = (69069 * mt[mti-1]) & 0xffffffff;
+}
+
+
+unsigned long MT19937::genrand(void)
+{
+ unsigned long y;
+ if (mti >= N) {
+ int kk;
+ if (mti == N+1)
+ {
+ sgenrand(4357);
+ }
+ for (kk=0;kk<N-M;kk++) {
+ y = (mt[kk]&UPPER_MASK)|(mt[kk+1]&LOWER_MASK);
+ mt[kk] = mt[kk+M] ^ (y >> 1) ^ mag01[y & 0x1];
+ }
+ for (;kk<N-1;kk++) {
+ y = (mt[kk]&UPPER_MASK)|(mt[kk+1]&LOWER_MASK);
+ mt[kk] = mt[kk+(M-N)] ^ (y >> 1) ^ mag01[y & 0x1];
+ }
+ y = (mt[N-1]&UPPER_MASK)|(mt[0]&LOWER_MASK);
+ mt[N-1] = mt[M-1] ^ (y >> 1) ^ mag01[y & 0x1];
+ mti = 0;
+ }
+ y = mt[mti++];
+ y ^= TEMPERING_SHIFT_U(y);
+ y ^= TEMPERING_SHIFT_S(y) & TEMPERING_MASK_B;
+ y ^= TEMPERING_SHIFT_T(y) & TEMPERING_MASK_C;
+ y ^= TEMPERING_SHIFT_L(y);
+ return y;
+}
+
+
+
+
+
+void CreateTables(Ndb* pNdb)
+{
+ for(Uint32 iTable=0; iTable<g_nTable; ++iTable)
+ {
+ NdbDictionary::Dictionary* pDictionary = pNdb->getDictionary();
+
+ NdbDictionary::Table table;
+ table.setName(g_szTable+iTable*4);
+
+ NdbDictionary::Index index;
+ index.setName(g_szIndex+iTable*4);
+ index.setTable(table.getName());
+ index.setType(NdbDictionary::Index::UniqueHashIndex);
+
+ NdbDictionary::Column columnPK;
+ columnPK.setName("PK");
+ columnPK.setTupleKey(true);
+ table.addColumn(columnPK);
+ index.addIndexColumn(columnPK.getName());
+
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ NdbDictionary::Column columnAttr;
+ columnAttr.setName(g_szAttribute+iAttr*4);
+ columnAttr.setTupleKey(false);
+ table.addColumn(columnAttr);
+ }
+
+ pDictionary->createTable(table);
+ pDictionary->createIndex(index);
+
+ /*
+ NdbSchemaCon* pNdbSchemaCon = pNdb->startSchemaTransaction();
+ NdbSchemaOp* pNdbSchemaOp = pNdbSchemaCon->getNdbSchemaOp();
+ pNdbSchemaOp->createTable(g_szTable+iTable*4);
+ pNdbSchemaOp->createAttribute("PK", TupleKey);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ pNdbSchemaOp->createAttribute(g_szAttribute+iAttr*4, NoKey);
+ }
+
+ pNdbSchemaCon->execute();
+ pNdb->closeSchemaTransaction(pNdbSchemaCon);
+ */
+ }
+}
+
+
+int InsertTransaction(Ndb* pNdb, const Uint32 iPart, const bool bIndex)
+{
+ int iExec = -1;
+ int iCode = -1;
+ NdbConnection* pNdbConnection = pNdb->startTransaction();
+ if(pNdbConnection)
+ {
+ for(Uint32 iTable=0; iTable<g_nTable; ++iTable)
+ {
+ for(Uint32 iTuple=0; iTuple<g_nTuple; ++iTuple)
+ {
+ if(bIndex)
+ {
+ NdbIndexOperation* pNdbIndexOperation = pNdbConnection->getNdbIndexOperation(g_szIndex+iTable*4, g_szTable+iTable*4);
+ pNdbIndexOperation->insertTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbIndexOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ Uint32 nValue = ((iPart*g_nTable+iTable)*g_nTuple+iTuple)*g_nAttribute+iAttr;
+ pNdbIndexOperation->setValue(g_szAttribute+iAttr*4, nValue);
+ }
+ }
+ else
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTable+iTable*4);
+ pNdbOperation->insertTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ Uint32 nValue = ((iPart*g_nTable+iTable)*g_nTuple+iTuple)*g_nAttribute+iAttr;
+ pNdbOperation->setValue(g_szAttribute+iAttr*4, nValue);
+ }
+ }
+ }
+ }
+ iExec = pNdbConnection->execute_ok(Commit);
+ if (iExec == -1)
+ {
+ ndbout << pNdbConnection->getNdbError() << endl;
+ }
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ return 0;
+}
+
+
+int UpdateGetAndSetTransaction(Ndb* pNdb, const Uint32 iPart, const bool bIndex)
+{
+ int iExec = -1;
+ int iCode = -1;
+ NdbRecAttr** ppNdbRecAttr = new NdbRecAttr*[g_nTable*g_nTuple*g_nAttribute];
+ NdbConnection* pNdbConnection = pNdb->startTransaction();
+ if(pNdbConnection)
+ {
+ for(Uint32 iTable=0; iTable<g_nTable; ++iTable)
+ {
+ for(Uint32 iTuple=0; iTuple<g_nTuple; ++iTuple)
+ {
+ if(bIndex)
+ {
+ NdbIndexOperation* pNdbIndexOperation = pNdbConnection->getNdbIndexOperation(g_szIndex+iTable*4, g_szTable+iTable*4);
+ pNdbIndexOperation->readTupleExclusive();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbIndexOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ ppNdbRecAttr[(iTable*g_nTuple+iTuple)*g_nAttribute+iAttr]
+ = pNdbIndexOperation->getValue(g_szAttribute+iAttr*4);
+ }
+ }
+ else
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTable+iTable*4);
+ pNdbOperation->readTupleExclusive();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ ppNdbRecAttr[(iTable*g_nTuple+iTuple)*g_nAttribute+iAttr]
+ = pNdbOperation->getValue(g_szAttribute+iAttr*4);
+ }
+ }
+ }
+ }
+ iExec = pNdbConnection->execute_ok(NoCommit);
+ if( iExec == -1)
+ {
+ ndbout << pNdbConnection->getNdbError() << endl;
+ }
+ }
+ iCode = pNdbConnection->getNdbError().code;
+ if(iExec==0)
+ {
+ for(Uint32 iTable=0; iTable<g_nTable; ++iTable)
+ {
+ for(Uint32 iTuple=0; iTuple<g_nTuple; ++iTuple)
+ {
+ if(bIndex)
+ {
+ NdbIndexOperation* pNdbIndexOperation = pNdbConnection->getNdbIndexOperation(g_szIndex+iTable*4, g_szTable+iTable*4);
+ pNdbIndexOperation->updateTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbIndexOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ NdbRecAttr* pNdbRecAttr
+ = ppNdbRecAttr[(iTable*g_nTuple+iTuple)*g_nAttribute+iAttr];
+ Uint32 nValue = pNdbRecAttr->u_32_value() + 1;
+ pNdbIndexOperation->setValue(g_szAttribute+iAttr*4, nValue);
+ }
+ }
+ else
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTable+iTable*4);
+ pNdbOperation->updateTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ NdbRecAttr* pNdbRecAttr
+ = ppNdbRecAttr[(iTable*g_nTuple+iTuple)*g_nAttribute+iAttr];
+ Uint32 nValue = pNdbRecAttr->u_32_value() + 1;
+ pNdbOperation->setValue(g_szAttribute+iAttr*4, nValue);
+ }
+ }
+ }
+ }
+ iExec = pNdbConnection->execute(Commit);
+ if (iExec == -1)
+ {
+ ndbout << pNdbConnection->getNdbError() << endl;
+ }
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ delete[] ppNdbRecAttr;
+ return 0;
+}
+
+
+int UpdateInterpretedTransaction(Ndb* pNdb, const Uint32 iPart, const bool bIndex)
+{
+ int iExec = -1;
+ int iCode = -1;
+ NdbConnection* pNdbConnection = pNdb->startTransaction();
+ if(pNdbConnection)
+ {
+ for(Uint32 iTable=0; iTable<g_nTable; ++iTable)
+ {
+ for(Uint32 iTuple=0; iTuple<g_nTuple; ++iTuple)
+ {
+ if(bIndex)
+ {
+ NdbIndexOperation* pNdbIndexOperation = pNdbConnection->getNdbIndexOperation(g_szIndex+iTable*4, g_szTable+iTable*4);
+ pNdbIndexOperation->interpretedUpdateTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbIndexOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ pNdbIndexOperation->incValue(g_szAttribute+iAttr*4, (Uint32)1);
+ }
+ }
+ else
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTable+iTable*4);
+ pNdbOperation->interpretedUpdateTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ pNdbOperation->incValue(g_szAttribute+iAttr*4, (Uint32)1);
+ }
+ }
+ }
+ }
+ iExec = pNdbConnection->execute_ok(Commit);
+
+ if (iExec == -1)
+ {
+ ndbout << pNdbConnection->getNdbError() << endl;
+ }
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ return 0;
+}
+
+
+void ReportInconsistency (const Uint32 iPart,
+ const Uint32 iTable,
+ const Uint32 iTuple,
+ const Uint32 iAttr,
+ const Uint32 nValue,
+ const Uint32 nExpected )
+{
+ ndbout << "INCONSISTENCY: ";
+ ndbout << "Part " << iPart;
+ ndbout << ", Table " << iTable;
+ ndbout << ", Tuple " << iTuple;
+ ndbout << ", Attr " << iAttr;
+ ndbout << ", Value " << nValue;
+ ndbout << ", Expected " << nExpected;
+ ndbout << endl;
+}
+
+
+int ReadTransaction(Ndb* pNdb, const Uint32 iPart, const bool bIndex)
+{
+ int iExec = -1;
+ int iCode = -1;
+ NdbRecAttr** ppNdbRecAttr = new NdbRecAttr*[g_nTable*g_nTuple*g_nAttribute];
+ NdbConnection* pNdbConnection = pNdb->startTransaction();
+ if(pNdbConnection)
+ {
+ for(Uint32 iTable=0; iTable<g_nTable; ++iTable)
+ {
+ for(Uint32 iTuple=0; iTuple<g_nTuple; ++iTuple)
+ {
+ if(bIndex)
+ {
+ NdbIndexOperation* pNdbIndexOperation = pNdbConnection->getNdbIndexOperation(g_szIndex+iTable*4, g_szTable+iTable*4);
+ pNdbIndexOperation->readTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbIndexOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ ppNdbRecAttr[(iTable*g_nTuple+iTuple)*g_nAttribute+iAttr]
+ = pNdbIndexOperation->getValue(g_szAttribute+iAttr*4);
+ }
+ }
+ else
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTable+iTable*4);
+ pNdbOperation->readTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbOperation->equal("PK", nPK);
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ ppNdbRecAttr[(iTable*g_nTuple+iTuple)*g_nAttribute+iAttr]
+ = pNdbOperation->getValue(g_szAttribute+iAttr*4);
+ }
+ }
+ }
+ }
+ iExec = pNdbConnection->execute_ok(Commit);
+ if (iExec == -1)
+ {
+ ndbout << pNdbConnection->getNdbError() << endl;
+ }
+ if(iExec==0)
+ {
+ Uint32 nValue0 = ppNdbRecAttr[0]->u_32_value();
+ for(Uint32 iTable=0; iTable<g_nTable; ++iTable)
+ {
+ for(Uint32 iTuple=0; iTuple<g_nTuple; ++iTuple)
+ {
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ Uint32 nValue = ppNdbRecAttr[(iTable*g_nTuple+iTuple)*g_nAttribute+iAttr]->u_32_value();
+ Uint32 nExpected = nValue0 + (iTable*g_nTuple+iTuple)*g_nAttribute+iAttr;
+ if(nValue!=nExpected)
+ {
+ ReportInconsistency(iPart, iTable, iTuple, iAttr, nValue, nExpected);
+ }
+ }
+ }
+ }
+ }
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ return 0;
+}
+
+
+int DeleteTransaction(Ndb* pNdb, const Uint32 iPart, const bool bIndex)
+{
+ int iExec = -1;
+ int iCode = -1;
+ NdbConnection* pNdbConnection = pNdb->startTransaction();
+ if(pNdbConnection)
+ {
+ for(Uint32 iTable=0; iTable<g_nTable; ++iTable)
+ {
+ for(Uint32 iTuple=0; iTuple<g_nTuple; ++iTuple)
+ {
+ if(bIndex)
+ {
+ NdbIndexOperation* pNdbIndexOperation = pNdbConnection->getNdbIndexOperation(g_szIndex+iTable*4, g_szTable+iTable*4);
+ pNdbIndexOperation->deleteTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbIndexOperation->equal("PK", nPK);
+ }
+ else
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTable+iTable*4);
+ pNdbOperation->deleteTuple();
+ Uint32 nPK = iPart*g_nTuple + iTuple;
+ pNdbOperation->equal("PK", nPK);
+ }
+ }
+ }
+ iExec = pNdbConnection->execute_ok(Commit);
+
+ if (iExec == -1)
+ {
+ ndbout << pNdbConnection->getNdbError() << endl;
+ }
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ return 0;
+}
+
+
+extern "C" void* ThreadFunc(void*)
+{
+ Ndb* pNdb = new Ndb("TEST_DB");
+ pNdb->init();
+ pNdb->waitUntilReady();
+
+ MT19937 rndgen;
+ rndgen.sgenrand((unsigned long)pNdb);
+
+ Uint32 nInsertError = 0;
+ Uint32 nInsertCommit = 0;
+ Uint32 nInsertRollback = 0;
+ Uint32 nUpdateGetAndSetError = 0;
+ Uint32 nUpdateGetAndSetCommit = 0;
+ Uint32 nUpdateGetAndSetRollback = 0;
+ Uint32 nReadError = 0;
+ Uint32 nReadCommit = 0;
+ Uint32 nReadRollback = 0;
+ Uint32 nUpdateInterpretedError = 0;
+ Uint32 nUpdateInterpretedCommit = 0;
+ Uint32 nUpdateInterpretedRollback = 0;
+ Uint32 nDeleteError = 0;
+ Uint32 nDeleteCommit = 0;
+ Uint32 nDeleteRollback = 0;
+
+ if (g_bVerify)
+ {
+ for (Uint32 iPart = 0; iPart < g_nPart; iPart++)
+ {
+ switch(ReadTransaction(pNdb, iPart, false))
+ {
+ case -1: ++nReadError; break;
+ case 0: ++nReadCommit; break;
+ case 1: ++nReadRollback; break;
+ }
+ }
+ }
+ else
+ while(NdbMutex_Trylock(g_pNdbMutexStop))
+ {
+ Uint32 iPart = rndgen.genrand() % g_nPart;
+ Uint32 iTrans = rndgen.genrand() % 5;
+ bool bIndex = ((rndgen.genrand() & 1) ? true : false);
+ switch(iTrans)
+ {
+ case 0:
+ switch(InsertTransaction(pNdb, iPart, bIndex))
+ {
+ case -1: ++nInsertError; break;
+ case 0: ++nInsertCommit; break;
+ case 1: ++nInsertRollback; break;
+ }
+ break;
+
+ case 1:
+ switch(UpdateGetAndSetTransaction(pNdb, iPart, bIndex))
+ {
+ case -1: ++nUpdateGetAndSetError; break;
+ case 0: ++nUpdateGetAndSetCommit; break;
+ case 1: ++nUpdateGetAndSetRollback; break;
+ }
+ break;
+
+ case 2:
+ switch(ReadTransaction(pNdb, iPart, bIndex))
+ {
+ case -1: ++nReadError; break;
+ case 0: ++nReadCommit; break;
+ case 1: ++nReadRollback; break;
+ }
+ break;
+
+ case 3:
+ switch(UpdateInterpretedTransaction(pNdb, iPart, bIndex))
+ {
+ case -1: ++nUpdateInterpretedError; break;
+ case 0: ++nUpdateInterpretedCommit; break;
+ case 1: ++nUpdateInterpretedRollback; break;
+ }
+ break;
+
+ case 4:
+ switch(DeleteTransaction(pNdb, iPart, bIndex))
+ {
+ case -1: ++nDeleteError; break;
+ case 0: ++nDeleteCommit; break;
+ case 1: ++nDeleteRollback; break;
+ }
+ break;
+ }
+ }
+
+ ndbout << "I:" << nInsertError << ":" << nInsertCommit << ":" << nInsertRollback;
+ ndbout << " UG:" << nUpdateGetAndSetError << ":" << nUpdateGetAndSetCommit << ":" << nUpdateGetAndSetRollback;
+ ndbout << " R:" << nReadError << ":" << nReadCommit << ":" << nReadRollback;
+ ndbout << " UI:" << nUpdateInterpretedError << ":" << nUpdateInterpretedCommit << ":" << nUpdateInterpretedRollback;
+ ndbout << " D:" << nDeleteError << ":" << nDeleteCommit << ":" << nDeleteRollback << endl;
+ ndbout << endl;
+
+ NdbMutex_Unlock(g_pNdbMutexStop);
+ delete pNdb;
+ return 0;
+}
+
+
+int main(int argc, char* argv[])
+{
+ Uint32 nSeconds = 1;
+ Uint32 nThread = 1;
+
+ for(int iArg=1; iArg<argc; ++iArg)
+ {
+ if(argv[iArg][0]=='-')
+ {
+ switch(argv[iArg][1])
+ {
+ case 'p': g_nPart = atol(argv[iArg]+2); break;
+ case 'b': g_nTable = atol(argv[iArg]+2); break;
+ case 'u': g_nTuple = atol(argv[iArg]+2); break;
+ case 'a': g_nAttribute = atol(argv[iArg]+2); break;
+ case 'v': g_bVerify = true; break;
+ case 't': nThread = atol(argv[iArg]+2); break;
+ case 's': nSeconds = atol(argv[iArg]+2); break;
+ case 'i': g_bUseIndex = true; break;
+ }
+ }
+ }
+ ndbout << argv[0];
+ ndbout << " -p" << g_nPart;
+ ndbout << " -b" << g_nTable;
+ ndbout << " -u" << g_nTuple;
+ ndbout << " -a" << g_nAttribute;
+ if (g_bVerify)
+ ndbout << " -v";
+ ndbout << " -t" << nThread;
+ ndbout << " -s" << nSeconds;
+ ndbout << endl;
+
+ g_szTable = new char[g_nTable*4];
+ for(Uint32 iTable=0; iTable<g_nTable; ++iTable)
+ {
+ sprintf(g_szTable+iTable*4, "T%02d", iTable);
+ }
+
+ g_szIndex = new char[g_nTable*4];
+ for(Uint32 iIndex=0; iIndex<g_nTable; ++iIndex)
+ {
+ sprintf(g_szIndex+iIndex*4, "I%02d", iIndex);
+ }
+
+ g_szAttribute = new char[g_nAttribute*4];
+ for(Uint32 iAttr=0; iAttr<g_nAttribute; ++iAttr)
+ {
+ sprintf(g_szAttribute+iAttr*4, "A%02d", iAttr);
+ }
+
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ pNdb->init();
+ pNdb->waitUntilReady();
+
+ if (!g_bVerify) CreateTables(pNdb);
+ g_pNdbMutexStop = NdbMutex_Create();
+ NdbMutex_Lock(g_pNdbMutexStop);
+
+ NdbThread_SetConcurrencyLevel(nThread+1);
+ NdbThread** ppNdbThread = new NdbThread*[nThread];
+ for(Uint32 iThread=0; iThread<nThread; ++iThread)
+ {
+ ppNdbThread[iThread] = NdbThread_Create(ThreadFunc, 0, 0, "ThreadFunc", NDB_THREAD_PRIO_MEAN);
+ }
+
+ NdbSleep_SecSleep(nSeconds);
+ NdbMutex_Unlock(g_pNdbMutexStop);
+
+ for(Uint32 iThread=0; iThread<nThread; ++iThread)
+ {
+ void* status;
+ NdbThread_WaitFor(ppNdbThread[iThread], &status);
+ }
+
+ NdbMutex_Destroy(g_pNdbMutexStop);
+ g_pNdbMutexStop = 0;
+ delete pNdb;
+}
+
+
diff --git a/ndb/test/ndbapi/bank/Bank.hpp b/ndb/test/ndbapi/bank/Bank.hpp
new file mode 100644
index 00000000000..e6816fd7111
--- /dev/null
+++ b/ndb/test/ndbapi/bank/Bank.hpp
@@ -0,0 +1,142 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef BANK_HPP
+#define BANK_HPP
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+#include <NdbTick.h>
+#include <random.h>
+
+
+class Bank {
+public:
+
+ Bank();
+
+ int createAndLoadBank(bool overWrite);
+ int dropBank();
+
+ int performTransactions(int maxSleepBetweenTrans = 20, int yield=0);
+ int performMakeGLs(int yield=0);
+ int performValidateAllGLs();
+ int performSumAccounts(int maxSleepBetweenSums = 2000, int yield=0);
+ int performIncreaseTime(int maxSleepBetweenDays = 30, int yield=0);
+private:
+
+ int init();
+
+ enum TransactionTypes{
+ WithDrawal = 2000,
+ Deposit = 3000
+ };
+
+ static const int NOT_ENOUGH_FUNDS = 1000;
+ static const int VERIFICATION_FAILED = 1001;
+
+ int performTransaction();
+ int performTransaction(int fromAccountId,
+ int toAccountId,
+ int amount );
+ int performTransactionImpl1(int fromAccountId,
+ int toAccountId,
+ int amount );
+
+ int performValidateGLs(Uint64 age = 20);
+ int performValidateGL(Uint64 GLTime);
+ int performValidatePurged();
+
+ int performMakeGL(int time);
+ int performMakeGLForAccountType(NdbConnection* pTrans,
+ Uint64 time,
+ Uint32 accountTypeId);
+ int sumTransactionsForGL(const Uint64 time,
+ const Uint32 accountType,
+ Uint32& balance,
+ Uint32& withdrawalCount,
+ Uint32& withdrawalSum,
+ Uint32& depositSum,
+ Uint32& depositCount,
+ Uint32& transactionsCount,
+ NdbConnection* pTrans);
+ int getBalanceForAccountType(const Uint32 accountType,
+ Uint32& balance);
+ int getBalanceForGL(const Uint64 glTime,
+ const Uint32 accountType,
+ Uint32 &balance);
+
+ int checkNoTransactionsOlderThan(const Uint32 accountType,
+ const Uint64 oldest);
+ int getOldestPurgedGL(const Uint32 accountType,
+ Uint64 &oldest);
+ int getOldestNotPurgedGL(Uint64 &oldest,
+ Uint32 &accountTypeId,
+ bool &found);
+ int findLastGL(Uint64 &lastTime);
+ int purgeOldGLTransactions(Uint64 currTime, Uint32 age);
+
+ int purgeTransactions(const Uint64 glTime,
+ const Uint32 accountTypeId);
+ int findTransactionsToPurge(const Uint64 glTime,
+ const Uint32 accountType,
+ NdbConnection* pTrans);
+
+
+ int getSumAccounts(Uint32 &sumAccounts,
+ Uint32 &numAccounts);
+ int getNumAccounts();
+ int getNumAccountTypes();
+ int getMaxAmount();
+
+
+ enum SystemValueId {
+ LastTransactionId = 0,
+ CurrentTime = 1
+ };
+
+
+ int readSystemValue(SystemValueId sysValId, Uint64 & value);
+ int increaseSystemValue(SystemValueId sysValId, Uint64 &value);
+ int increaseSystemValue2(SystemValueId sysValId, Uint64 &value);
+ int writeSystemValue(SystemValueId sysValId, Uint64 value);
+ int getNextTransactionId(Uint64 &value);
+ int incCurrTime(Uint64 &value);
+ int getCurrTime(Uint64 &time);
+
+ int createTables();
+ int createTable(const char* tabName);
+
+ int dropTables();
+ int dropTable(const char* tabName);
+
+ int clearTables();
+ int clearTable(const char* tabName);
+
+ int loadGl();
+ int loadAccountType();
+ int loadAccount (int numAccounts);
+ int loadSystemValues();
+
+private:
+
+ Ndb m_ndb;
+ int m_maxAccount;
+ bool m_initialized;
+};
+
+#endif
diff --git a/ndb/test/ndbapi/bank/Makefile b/ndb/test/ndbapi/bank/Makefile
new file mode 100644
index 00000000000..f710f9e6612
--- /dev/null
+++ b/ndb/test/ndbapi/bank/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+DIRS = src bankCreator \
+ bankSumAccounts \
+ bankTransactionMaker \
+ bankValidateAllGLs \
+ bankMakeGL \
+ bankTimer \
+ testBank
+
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bank/bankCreator/Makefile b/ndb/test/ndbapi/bank/bankCreator/Makefile
new file mode 100644
index 00000000000..d40103a8347
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankCreator/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+TYPE = ndbapitest
+BIN_TARGET = bankCreator
+SOURCES = bankCreator.cpp
+BIN_TARGET_LIBS += bank
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bank/bankCreator/bankCreator.cpp b/ndb/test/ndbapi/bank/bankCreator/bankCreator.cpp
new file mode 100644
index 00000000000..ee724236855
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankCreator/bankCreator.cpp
@@ -0,0 +1,55 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+#include "../Bank.hpp"
+
+
+int main(int argc, const char** argv){
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "This program will create and load the tables for bank\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ Bank bank;
+ int overWriteExisting = true;
+ if (bank.createAndLoadBank(overWriteExisting) != NDBT_OK)
+ return NDBT_ProgramExit(NDBT_FAILED);
+ return NDBT_ProgramExit(NDBT_OK);
+
+}
+
+
+
diff --git a/ndb/test/ndbapi/bank/bankMakeGL/Makefile b/ndb/test/ndbapi/bank/bankMakeGL/Makefile
new file mode 100644
index 00000000000..16a092e885c
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankMakeGL/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+TYPE = ndbapitest
+BIN_TARGET = bankMakeGL
+SOURCES = bankMakeGL.cpp
+BIN_TARGET_LIBS += bank
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bank/bankMakeGL/bankMakeGL.cpp b/ndb/test/ndbapi/bank/bankMakeGL/bankMakeGL.cpp
new file mode 100644
index 00000000000..0b6fc9c1f97
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankMakeGL/bankMakeGL.cpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+#include "../Bank.hpp"
+
+
+int main(int argc, const char** argv){
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "This program will make GL records in the bank\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ Bank bank;
+
+ if (bank.performMakeGLs() != 0)
+ return NDBT_ProgramExit(NDBT_FAILED);
+
+ return NDBT_ProgramExit(NDBT_OK);
+
+}
+
+
+
diff --git a/ndb/test/ndbapi/bank/bankSumAccounts/Makefile b/ndb/test/ndbapi/bank/bankSumAccounts/Makefile
new file mode 100644
index 00000000000..34f1cc21bc6
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankSumAccounts/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+TYPE = ndbapitest
+BIN_TARGET = bankSumAccounts
+SOURCES = bankSumAccounts.cpp
+BIN_TARGET_LIBS += bank
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bank/bankSumAccounts/bankSumAccounts.cpp b/ndb/test/ndbapi/bank/bankSumAccounts/bankSumAccounts.cpp
new file mode 100644
index 00000000000..7071de9f63e
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankSumAccounts/bankSumAccounts.cpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+#include "../Bank.hpp"
+
+
+int main(int argc, const char** argv){
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "This program will check the sum of all ACCOUNTS in the bank\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ Bank bank;
+
+ if (bank.performSumAccounts() != 0)
+ return NDBT_ProgramExit(NDBT_FAILED);
+
+ return NDBT_ProgramExit(NDBT_OK);
+
+}
+
+
+
diff --git a/ndb/test/ndbapi/bank/bankTimer/Makefile b/ndb/test/ndbapi/bank/bankTimer/Makefile
new file mode 100644
index 00000000000..a2fcf703723
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankTimer/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+TYPE = ndbapitest
+BIN_TARGET = bankTimer
+SOURCES = bankTimer.cpp
+BIN_TARGET_LIBS += bank
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bank/bankTimer/bankTimer.cpp b/ndb/test/ndbapi/bank/bankTimer/bankTimer.cpp
new file mode 100644
index 00000000000..cfb2c93e4ad
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankTimer/bankTimer.cpp
@@ -0,0 +1,58 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+#include "../Bank.hpp"
+
+
+int main(int argc, const char** argv){
+ int _help = 0;
+ int _wait = 30;
+
+ struct getargs args[] = {
+ { "wait", 'w', arg_integer, &_wait, "Max time to wait between days", "secs" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "This program will increase time in the bank\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ Bank bank;
+
+ if (bank.performIncreaseTime(_wait) != 0)
+ return NDBT_ProgramExit(NDBT_FAILED);
+
+ return NDBT_ProgramExit(NDBT_OK);
+
+}
+
+
+
diff --git a/ndb/test/ndbapi/bank/bankTransactionMaker/Makefile b/ndb/test/ndbapi/bank/bankTransactionMaker/Makefile
new file mode 100644
index 00000000000..2e482898476
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankTransactionMaker/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+TYPE = ndbapitest
+BIN_TARGET = bankTransactionMaker
+SOURCES = bankTransactionMaker.cpp
+BIN_TARGET_LIBS += bank
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bank/bankTransactionMaker/bankTransactionMaker.cpp b/ndb/test/ndbapi/bank/bankTransactionMaker/bankTransactionMaker.cpp
new file mode 100644
index 00000000000..155a35998bb
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankTransactionMaker/bankTransactionMaker.cpp
@@ -0,0 +1,58 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+#include "../Bank.hpp"
+
+
+int main(int argc, const char** argv){
+ int _help = 0;
+ int _wait = 20;
+
+ struct getargs args[] = {
+ { "wait", 'w', arg_integer, &_wait, "Time to wait between transactions", "ms" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "This program will perform transactions in the bank\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ Bank bank;
+
+ if (bank.performTransactions(_wait) != 0)
+ return NDBT_ProgramExit(NDBT_FAILED);
+
+ return NDBT_ProgramExit(NDBT_OK);
+
+}
+
+
+
diff --git a/ndb/test/ndbapi/bank/bankValidateAllGLs/Makefile b/ndb/test/ndbapi/bank/bankValidateAllGLs/Makefile
new file mode 100644
index 00000000000..660b73fb830
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankValidateAllGLs/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+TYPE = ndbapitest
+BIN_TARGET = bankValidateAllGLs
+SOURCES = bankValidateAllGLs.cpp
+BIN_TARGET_LIBS += bank
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bank/bankValidateAllGLs/bankValidateAllGLs.cpp b/ndb/test/ndbapi/bank/bankValidateAllGLs/bankValidateAllGLs.cpp
new file mode 100644
index 00000000000..cc8e2792cbf
--- /dev/null
+++ b/ndb/test/ndbapi/bank/bankValidateAllGLs/bankValidateAllGLs.cpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+#include "../Bank.hpp"
+
+
+int main(int argc, const char** argv){
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "This program will validate all GLs in the bank\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ Bank bank;
+
+ if (bank.performValidateAllGLs() != 0)
+ return NDBT_ProgramExit(NDBT_FAILED);
+
+ return NDBT_ProgramExit(NDBT_OK);
+
+}
+
+
+
diff --git a/ndb/test/ndbapi/bank/src/Bank.cpp b/ndb/test/ndbapi/bank/src/Bank.cpp
new file mode 100644
index 00000000000..11ebf087fd4
--- /dev/null
+++ b/ndb/test/ndbapi/bank/src/Bank.cpp
@@ -0,0 +1,2458 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "../Bank.hpp"
+#include <time.h>
+#include <NdbSleep.h>
+#include <UtilTransactions.hpp>
+
+Bank::Bank():
+ m_ndb("BANK"),
+ m_maxAccount(-1),
+ m_initialized(false)
+{
+
+}
+
+int Bank::init(){
+ if (m_initialized == true)
+ return NDBT_OK;
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+ m_ndb.init();
+ while (m_ndb.waitUntilReady(10) != 0)
+ ndbout << "Waiting for ndb to be ready" << endl;
+
+ if (getNumAccounts() != NDBT_OK)
+ return NDBT_FAILED;
+ return NDBT_OK;
+}
+
+int Bank::performTransactions(int maxSleepBetweenTrans, int yield){
+
+ if (init() != NDBT_OK)
+ return NDBT_FAILED;
+ int transactions = 0;
+
+ while(1){
+
+ while(m_ndb.waitUntilReady(10) != 0)
+ ndbout << "Waiting for ndb to be ready" << endl;
+
+ while(performTransaction() != NDBT_FAILED){
+ transactions++;
+
+ if (maxSleepBetweenTrans > 0){
+ int val = myRandom48(maxSleepBetweenTrans);
+ NdbSleep_MilliSleep(val);
+ }
+
+ if((transactions % 100) == 0)
+ g_info << transactions << endl;
+
+ if (yield != 0 && transactions >= yield)
+ return NDBT_OK;
+ }
+ }
+ return NDBT_FAILED;
+
+}
+
+int Bank::performTransaction(){
+ int result = NDBT_OK;
+
+ if (m_maxAccount <= 0){
+ g_err << "No accounts in bank" << endl;
+ return NDBT_FAILED;
+ }
+
+ int fromAccount = myRandom48(m_maxAccount);
+ int toAccount = myRandom48(m_maxAccount);
+
+ if (fromAccount == toAccount){
+ // Increase toAccount with 1
+ toAccount = (toAccount+1)%m_maxAccount;
+ }
+
+ int maxAmount = getMaxAmount();
+
+ int amount = myRandom48(maxAmount);
+
+ retry_transaction:
+ int res = performTransaction(fromAccount, toAccount, amount);
+ if (res != 0){
+ switch (res){
+ case NDBT_FAILED:
+ g_err << "performTransaction returned NDBT_FAILED" << endl
+ << " fromAccount = " << fromAccount << endl
+ << " toAccount = " << toAccount << endl
+ << " amount = " << amount << endl;
+ result = NDBT_FAILED;
+ break;
+ case NOT_ENOUGH_FUNDS:
+ // ndbout << "performTransaction returned NOT_ENOUGH_FUNDS" << endl;
+ break;
+ case NDBT_TEMPORARY:
+ g_err << "TEMPORARY_ERRROR retrying" << endl;
+ goto retry_transaction;
+ break;
+ default:
+ g_info << "performTransaction returned "<<res << endl;
+ break;
+ }
+ }
+ return result;
+}
+
+/**
+ * Perform a transaction in the bank.
+ * Ie. transfer money from one account to another.
+ *
+ * @param
+ * @return 0 if successful or an error code
+ */
+int Bank::performTransaction(int fromAccountId,
+ int toAccountId,
+ int amount ){
+ /**
+ * 1. Start transaction
+ * 2. Check balance on from account, if there is
+ * not enough funds abort transaction
+ * 3. Update ACCOUNT set balance = balance - amount on
+ * from account
+ * 4. Insert withdrawal in TRANSACTION
+ * 5. Insert deposit in transaction
+ * 6. Update ACCOUNT set balance = balance + amount on
+ * to account
+ * 7. Commit transaction
+ */
+ // g_info << "performTransaction " << fromAccountId
+ // << ", "<<toAccountId<<", "<<amount << endl;
+
+ // Call the first implementation of this trans
+ // In the future we can have several different versions of this trans
+ // and call them randomly
+ return performTransactionImpl1(fromAccountId, toAccountId, amount);
+}
+
+
+int Bank::performTransactionImpl1(int fromAccountId,
+ int toAccountId,
+ int amount ){
+
+ int check;
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if( pTrans == NULL ) {
+ const NdbError err = m_ndb.getNdbError();
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ return NDBT_TEMPORARY;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ /**
+ * Check balance on from account
+ */
+ NdbOperation* pOp = pTrans->getNdbOperation("ACCOUNT");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->readTupleExclusive();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("ACCOUNT_ID", fromAccountId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* balanceFromRec = pOp->getValue("BALANCE");
+ if( balanceFromRec ==NULL ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* fromAccountTypeRec = pOp->getValue("ACCOUNT_TYPE");
+ if( fromAccountTypeRec == NULL ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ m_ndb.closeTransaction(pTrans);
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ return NDBT_TEMPORARY;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ Uint32 balanceFrom = balanceFromRec->u_32_value();
+ // ndbout << "balanceFrom: " << balanceFrom << endl;
+
+ if (((Int64)balanceFrom - amount) < 0){
+ m_ndb.closeTransaction(pTrans);
+ //ndbout << "Not enough funds" << endl;
+ return NOT_ENOUGH_FUNDS;
+ }
+
+ Uint32 fromAccountType = fromAccountTypeRec->u_32_value();
+
+ /**
+ * Read balance on to account
+ */
+ NdbOperation* pOp6 = pTrans->getNdbOperation("ACCOUNT");
+ if (pOp6 == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp6->readTupleExclusive();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp6->equal("ACCOUNT_ID", toAccountId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* balanceToRec = pOp6->getValue("BALANCE");
+ if( balanceToRec == NULL ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* toAccountTypeRec = pOp6->getValue("ACCOUNT_TYPE");
+ if( toAccountTypeRec == NULL ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ m_ndb.closeTransaction(pTrans);
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ return NDBT_TEMPORARY;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ Uint32 balanceTo = balanceToRec->u_32_value();
+ // ndbout << "balanceTo: " << balanceTo << endl;
+ Uint32 toAccountType = toAccountTypeRec->u_32_value();
+
+ // Ok, all clear to do the transaction
+ Uint64 transId;
+ if (getNextTransactionId(transId) != NDBT_OK){
+ return NDBT_FAILED;
+ }
+
+ Uint64 currTime;
+ if (getCurrTime(currTime) != NDBT_OK){
+ return NDBT_FAILED;
+ }
+
+ /**
+ * Update balance on from account
+ */
+ NdbOperation* pOp2 = pTrans->getNdbOperation("ACCOUNT");
+ if (pOp2 == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp2->updateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp2->equal("ACCOUNT_ID", fromAccountId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp2->setValue("BALANCE", balanceFrom - amount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ /**
+ * Update balance on to account
+ */
+ NdbOperation* pOp3 = pTrans->getNdbOperation("ACCOUNT");
+ if (pOp3 == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->updateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->equal("ACCOUNT_ID", toAccountId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->setValue("BALANCE", balanceTo + amount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ /**
+ * Insert withdrawal transaction
+ */
+ NdbOperation* pOp4 = pTrans->getNdbOperation("TRANSACTION");
+ if (pOp4 == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp4->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp4->equal("TRANSACTION_ID", transId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp4->equal("ACCOUNT", fromAccountId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp4->setValue("ACCOUNT_TYPE", fromAccountType);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp4->setValue("OTHER_ACCOUNT", toAccountId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp4->setValue("TRANSACTION_TYPE", WithDrawal);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp4->setValue("TIME", currTime);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp4->setValue("AMOUNT", amount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ /**
+ * Insert deposit transaction
+ */
+ NdbOperation* pOp5 = pTrans->getNdbOperation("TRANSACTION");
+ if (pOp5 == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp5->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp5->equal("TRANSACTION_ID", transId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp5->equal("ACCOUNT", toAccountId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp5->setValue("ACCOUNT_TYPE", toAccountType);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp5->setValue("OTHER_ACCOUNT", fromAccountId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp5->setValue("TRANSACTION_TYPE", Deposit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp5->setValue("TIME", currTime);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp5->setValue("AMOUNT", amount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ m_ndb.closeTransaction(pTrans);
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ return NDBT_TEMPORARY;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_OK;
+}
+
+
+
+
+int Bank::performMakeGLs(int yield){
+ int result;
+ if (init() != NDBT_OK)
+ return NDBT_FAILED;
+
+ int counter, maxCounter;
+ int yieldCounter = 0;
+
+ while (1){
+ // Counters to keep tracck of how many
+ // GLs should be made before performing a validation
+ counter = 0;
+ maxCounter = 50 + myRandom48(100);
+
+ while(m_ndb.waitUntilReady(10) != 0)
+ ndbout << "Waiting for ndb to be ready" << endl;
+
+ /**
+ * Validate GLs and Transactions for previous days
+ *
+ */
+ result = performValidateGLs();
+ if (result != NDBT_OK){
+ if (result == VERIFICATION_FAILED){
+ g_err << "performValidateGLs verification failed" << endl;
+ return NDBT_FAILED;
+ }
+ g_info << "performValidateGLs failed" << endl;
+ continue;
+ }
+
+ result = performValidatePurged();
+ if (result != NDBT_OK){
+ if (result == VERIFICATION_FAILED){
+ g_err << "performValidatePurged verification failed" << endl;
+ return NDBT_FAILED;
+ }
+ g_info << "performValidatePurged failed" << endl;
+ continue;
+ }
+
+ while (1){
+
+ yieldCounter++;
+ if (yield != 0 && yieldCounter >= yield)
+ return NDBT_OK;
+
+ /**
+ * Find last GL time.
+ * ( GL record with highest time value)
+ */
+ Uint64 lastGLTime;
+ if (findLastGL(lastGLTime) != NDBT_OK){
+ g_info << "findLastGL failed" << endl;
+ // Break out of inner while loop
+ break;
+ }
+
+ lastGLTime++;
+
+ /**
+ * If last GL time + 1 is smaller than current time
+ * perform a GL for that time
+ */
+ Uint64 currTime;
+ if (getCurrTime(currTime) != NDBT_OK){
+ g_info << "getCurrTime failed" << endl;
+ // Break out of inner while loop
+ break;
+ }
+ if (lastGLTime < currTime){
+ counter++;
+ if (performMakeGL(lastGLTime) != NDBT_OK){
+ g_info << "performMakeGL failed" << endl;
+ // Break out of inner while loop
+ break;
+ }
+
+ if (counter > maxCounter){
+ // Break out of inner while loop and
+ // validatePreviousGLs
+ g_info << "counter("<<counter<<") > maxCounter("<<maxCounter<<")" << endl;
+ break;
+ }
+
+ } else {
+ ;//ndbout << "It's not time to make GL yet" << endl;
+
+ // ndbout << "Sleeping 1 second" << endl;
+ NdbSleep_SecSleep(1);
+
+ }
+
+ Uint32 age = 3;
+ if (purgeOldGLTransactions(currTime, age) != NDBT_OK){
+ g_info << "purgeOldGLTransactions failed" << endl;
+ // Break out of inner while loop
+ break;
+ }
+
+ }
+ }
+
+ return NDBT_FAILED;
+
+}
+
+int Bank::performValidateAllGLs(){
+ int result;
+ if (init() != NDBT_OK)
+ return NDBT_FAILED;
+
+ while (1){
+
+ while(m_ndb.waitUntilReady(10) != 0)
+ ndbout << "Waiting for ndb to be ready" << endl;
+
+ /**
+ * Validate GLs and Transactions for previous days
+ * Set age so that ALL GL's are validated
+ */
+ int age = 100000;
+ result = performValidateGLs(age);
+ if (result != NDBT_OK){
+ if (result == VERIFICATION_FAILED){
+ g_err << "performValidateGLs verification failed" << endl;
+ return NDBT_FAILED;
+ }
+ g_err << "performValidateGLs failed" << endl;
+ return NDBT_FAILED;
+ }
+
+ /**
+ *
+ *
+ */
+ result = performValidatePurged();
+ if (result != NDBT_OK){
+ if (result == VERIFICATION_FAILED){
+ g_err << "performValidatePurged verification failed" << endl;
+ return NDBT_FAILED;
+ }
+ g_err << "performValidatePurged failed" << endl;
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+ }
+
+ return NDBT_FAILED;
+
+}
+
+int Bank::findLastGL(Uint64 &lastTime){
+
+ int check;
+ /**
+ * SELECT MAX(time) FROM GL
+ */
+ NdbConnection* pScanTrans = m_ndb.startTransaction();
+ if (pScanTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pScanTrans->getNdbOperation("GL");
+ if (pOp == NULL) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanRead(64);
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* timeRec = pOp->getValue("TIME");
+ if( timeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pScanTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ eof = pScanTrans->nextScanResult();
+ lastTime = 0;
+
+ while(eof == 0){
+ rows++;
+ Uint64 t = timeRec->u_32_value();
+
+ if (t > lastTime)
+ lastTime = t;
+
+ eof = pScanTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pScanTrans);
+
+ return NDBT_OK;
+}
+
+
+int Bank::performMakeGL(int time){
+ g_info << "performMakeGL: " << time << endl;
+ /**
+ * Create one GL record for each account type.
+ * All in the same transaction
+ */
+ // Start transaction
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL){
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+ for (int i = 0; i < getNumAccountTypes(); i++){
+
+ if (performMakeGLForAccountType(pTrans, time, i) != NDBT_OK){
+ g_err << "performMakeGLForAccountType returned NDBT_FAILED"<<endl;
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ // Execute transaction
+ if( pTrans->execute(Commit) == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ m_ndb.closeTransaction(pTrans);
+
+ return NDBT_OK;
+}
+
+int Bank::performMakeGLForAccountType(NdbConnection* pTrans,
+ Uint64 glTime,
+ Uint32 accountTypeId){
+ int check;
+
+ Uint32 balance = 0;
+ Uint32 withdrawalCount = 0;
+ Uint32 withdrawalSum = 0;
+ Uint32 depositSum = 0;
+ Uint32 depositCount = 0;
+ Uint32 countTransactions = 0;
+ Uint32 purged = 0;
+
+ // Insert record in GL so that we know
+ // that no one else is performing the same task
+ // Set purged = 0 to indicate that TRANSACTION
+ // records still exist
+ NdbOperation* pOp = pTrans->getNdbOperation("GL");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("TIME", glTime);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("ACCOUNT_TYPE", accountTypeId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->setValue("BALANCE", balance);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->setValue("DEPOSIT_COUNT", depositCount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->setValue("DEPOSIT_SUM", depositSum);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->setValue("WITHDRAWAL_COUNT", withdrawalCount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->setValue("WITHDRAWAL_SUM", withdrawalSum);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->setValue("PURGED", purged);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ ERR(pOp->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Read previous GL record to get old balance
+ NdbOperation* pOp2 = pTrans->getNdbOperation("GL");
+ if (pOp2 == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp2->readTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp2->equal("TIME", glTime-1);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp2->equal("ACCOUNT_TYPE", accountTypeId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* oldBalanceRec = pOp2->getValue("BALANCE");
+ if( oldBalanceRec == NULL ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ ERR(pOp2->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ Uint32 oldBalance = oldBalanceRec->u_32_value();
+ // ndbout << "oldBalance = "<<oldBalance<<endl;
+ balance = oldBalance;
+ // Start a scan transaction to search
+ // for TRANSACTION records with TIME = time
+ // and ACCOUNT_TYPE = accountTypeId
+ // Build sum of all found transactions
+
+ if (sumTransactionsForGL(glTime,
+ accountTypeId,
+ balance,
+ withdrawalCount,
+ withdrawalSum,
+ depositSum,
+ depositCount,
+ countTransactions,
+ pTrans) != NDBT_OK){
+ return NDBT_FAILED;
+ }
+ // ndbout << "sumTransactionsForGL completed" << endl;
+ // ndbout << "balance="<<balance<<endl
+ // << "withdrawalCount="<<withdrawalCount<<endl
+ // << "withdrawalSum="<<withdrawalSum<<endl
+ // << "depositCount="<<depositCount<<endl
+ // << "depositSum="<<depositSum<<endl;
+
+
+
+ NdbOperation* pOp3 = pTrans->getNdbOperation("GL");
+ if (pOp3 == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->updateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->equal("TIME", glTime);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->equal("ACCOUNT_TYPE", accountTypeId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->setValue("BALANCE", balance);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->setValue("DEPOSIT_COUNT", depositCount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->setValue("DEPOSIT_SUM", depositSum);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->setValue("WITHDRAWAL_COUNT", withdrawalCount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->setValue("WITHDRAWAL_SUM", withdrawalSum);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->setValue("PURGED", purged);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Execute transaction
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+
+
+
+int Bank::sumTransactionsForGL(const Uint64 glTime,
+ const Uint32 accountType,
+ Uint32& balance,
+ Uint32& withdrawalCount,
+ Uint32& withdrawalSum,
+ Uint32& depositSum,
+ Uint32& depositCount,
+ Uint32& transactionsCount,
+ NdbConnection* pTrans){
+ int check;
+
+ // g_info << "sumTransactionsForGL: " << glTime << ", " << accountType << endl;
+
+ NdbConnection* pScanTrans = m_ndb.startTransaction();
+ if (pScanTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pScanTrans->getNdbOperation("TRANSACTION");
+ if (pOp == NULL) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanExclusive(64);
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* accountTypeRec = pOp->getValue("ACCOUNT_TYPE");
+ if( accountTypeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* timeRec = pOp->getValue("TIME");
+ if( timeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* transTypeRec = pOp->getValue("TRANSACTION_TYPE");
+ if( transTypeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* amountRec = pOp->getValue("AMOUNT");
+ if( amountRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pScanTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ int rowsFound = 0;
+ eof = pScanTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+ Uint32 a = accountTypeRec->u_32_value();
+ Uint64 t = timeRec->u_64_value();
+
+ if (a == accountType && t == glTime){
+ rowsFound++;
+ // One record found
+ int transType = transTypeRec->u_32_value();
+ int amount = amountRec->u_32_value();
+ if (transType == WithDrawal){
+ withdrawalCount++;
+ withdrawalSum += amount;
+ balance -= amount;
+ } else {
+ assert(transType == Deposit);
+ depositCount++;
+ depositSum += amount;
+ balance += amount;
+ }
+ }
+
+ eof = pScanTrans->nextScanResult();
+
+ if ((rows % 100) == 0){
+ // "refresh" ownner transaction every 100th row
+ if (pTrans->refresh() == -1) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+
+ }
+ if (eof == -1) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pScanTrans);
+ // ndbout << rows << " TRANSACTIONS have been read" << endl;
+ transactionsCount = rowsFound;
+
+ return NDBT_OK;
+
+}
+
+ int Bank::performValidateGLs(Uint64 age){
+
+ Uint64 currTime;
+ if (getCurrTime(currTime) != NDBT_OK){
+ return NDBT_FAILED;
+ }
+ Uint64 glTime = currTime - 1;
+ while((glTime > 0) && ((glTime + age) >= currTime)){
+
+ int result = performValidateGL(glTime);
+ if (result != NDBT_OK){
+ g_err << "performValidateGL failed" << endl;
+ return result;
+ }
+
+ glTime--;
+ }
+
+ return NDBT_OK;
+ }
+
+int Bank::performValidateGL(Uint64 glTime){
+
+ ndbout << "performValidateGL: " << glTime << endl;
+ /**
+ * Rules:
+ * - There should be zero or NoAccountTypes GL records for each glTime
+ * - If purged == 0, then the TRANSACTION table should be checked
+ * to see that there are:
+ * + DEPOSIT_COUNT deposit transactions with account_type == ACCOUNT_TYPE
+ * and TIME == glTime. The sum of these transactions should be
+ * DEPOSIT_SUM
+ * + WITHDRAWAL_COUNT withdrawal transactions with account_type ==
+ * ACCOUNT_TYPE and TIME == glTime. The sum of these transactions
+ * should be WITHDRAWAL_SUM
+ * + BALANCE should be equal to the sum of all transactions plus
+ * the balance of the previous GL record
+ * - If purged == 1 then there should be NO transactions with TIME == glTime
+ * and ACCOUNT_TYPE == account_type
+ *
+ */
+
+ int check;
+ /**
+ * SELECT * FROM GL WHERE account_type = @accountType and time = @time
+ */
+ NdbConnection* pScanTrans = m_ndb.startTransaction();
+ if (pScanTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pScanTrans->getNdbOperation("GL");
+ if (pOp == NULL) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanRead(64);
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* accountTypeRec = pOp->getValue("ACCOUNT_TYPE");
+ if( accountTypeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* timeRec = pOp->getValue("TIME");
+ if( timeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* purgedRec = pOp->getValue("PURGED");
+ if( purgedRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* balanceRec = pOp->getValue("BALANCE");
+ if( balanceRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* depositSumRec = pOp->getValue("DEPOSIT_SUM");
+ if( depositSumRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* depositCountRec = pOp->getValue("DEPOSIT_COUNT");
+ if( depositCountRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* withdrawalSumRec = pOp->getValue("WITHDRAWAL_SUM");
+ if( withdrawalSumRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+ NdbRecAttr* withdrawalCountRec = pOp->getValue("WITHDRAWAL_COUNT");
+ if( withdrawalCountRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pScanTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ int countGlRecords = 0;
+ int result = NDBT_OK;
+ eof = pScanTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+ Uint64 t = timeRec->u_64_value();
+
+ if (t == glTime){
+ countGlRecords++;
+ Uint32 a = accountTypeRec->u_32_value();
+ Uint32 purged = purgedRec->u_32_value();
+ Uint32 wsum = withdrawalSumRec->u_32_value();
+ Uint32 wcount = withdrawalCountRec->u_32_value();
+ Uint32 dsum = depositSumRec->u_32_value();
+ Uint32 dcount = depositCountRec->u_32_value();
+ Uint32 b = balanceRec->u_32_value();
+
+ Uint32 balance = 0;
+ Uint32 withdrawalSum = 0;
+ Uint32 withdrawalCount = 0;
+ Uint32 depositSum = 0;
+ Uint32 depositCount = 0;
+ Uint32 countTransactions = 0;
+ if (purged == 0){
+ // If purged == 0, then the TRANSACTION table should be checked
+ // to see that there are:
+ // + DEPOSIT_COUNT deposit transactions with account_type == ACCOUNT_TYPE
+ // and TIME == glTime. The sum of these transactions should be
+ // DEPOSIT_SUM
+ // + WITHDRAWAL_COUNT withdrawal transactions with account_type ==
+ // ACCOUNT_TYPE and TIME == glTime. The sum of these transactions
+ // should be WITHDRAWAL_SUM
+ // + BALANCE should be equal to the sum of all transactions plus
+ // the balance of the previous GL record
+ if (sumTransactionsForGL(t,
+ a,
+ balance,
+ withdrawalCount,
+ withdrawalSum,
+ depositSum,
+ depositCount,
+ countTransactions,
+ pScanTrans) != NDBT_OK){
+ result = NDBT_FAILED;
+ } else {
+ Uint32 prevBalance = 0;
+ if (getBalanceForGL(t-1, a, prevBalance) != NDBT_OK){
+ result = NDBT_FAILED;
+ } else
+ if (((prevBalance + balance) != b) ||
+ (wsum != withdrawalSum) ||
+ (wcount != withdrawalCount) ||
+ (dsum != depositSum) ||
+ (dcount != depositCount)){
+ g_err << "performValidateGL, sums and counts failed" << endl
+ << "balance : " << balance+prevBalance << "!="<<b<<endl
+ << "with sum : " << withdrawalSum << "!="<<wsum<<endl
+ << "with count: " << withdrawalCount << "!="<<wcount<<endl
+ << "dep sum : " << depositSum << "!="<<dsum<<endl
+ << "dep count : " << depositCount << "!="<<dcount<<endl;
+ result = VERIFICATION_FAILED;
+ }
+ }
+
+ } else {
+ assert(purged == 1);
+ // If purged == 1 then there should be NO transactions with
+ // TIME == glTime and ACCOUNT_TYPE == account_type
+
+ if (sumTransactionsForGL(t,
+ a,
+ balance,
+ withdrawalCount,
+ withdrawalSum,
+ depositSum,
+ depositCount,
+ countTransactions,
+ pScanTrans) != NDBT_OK){
+ result = NDBT_FAILED;
+ } else {
+ if (countTransactions != 0){
+ g_err << "performValidateGL, countTransactions("<<countTransactions<<") != 0" << endl;
+ result = VERIFICATION_FAILED;
+ }
+ }
+ }
+
+ }
+ eof = pScanTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pScanTrans);
+
+ // - There should be zero or NoAccountTypes GL records for each glTime
+ if ((countGlRecords != 0) && (countGlRecords != getNumAccountTypes())){
+ g_err << "performValidateGL: " << endl
+ << "countGlRecords = " << countGlRecords << endl;
+ result = VERIFICATION_FAILED;
+ }
+
+ return result;
+
+
+ }
+
+int Bank::getBalanceForGL(const Uint64 glTime,
+ const Uint32 accountTypeId,
+ Uint32 &balance){
+ int check;
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pTrans->getNdbOperation("GL");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->readTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("TIME", glTime);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("ACCOUNT_TYPE", accountTypeId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* balanceRec = pOp->getValue("BALANCE");
+ if( balanceRec == NULL ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pTrans);
+
+ balance = balanceRec->u_32_value();
+
+ return NDBT_OK;
+}
+
+
+
+int Bank::getOldestPurgedGL(const Uint32 accountType,
+ Uint64 &oldest){
+ int check;
+ /**
+ * SELECT MAX(time) FROM GL WHERE account_type = @accountType and purged=1
+ */
+ NdbConnection* pScanTrans = m_ndb.startTransaction();
+ if (pScanTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pScanTrans->getNdbOperation("GL");
+ if (pOp == NULL) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanRead(64);
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* accountTypeRec = pOp->getValue("ACCOUNT_TYPE");
+ if( accountTypeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* timeRec = pOp->getValue("TIME");
+ if( timeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* purgedRec = pOp->getValue("PURGED");
+ if( purgedRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pScanTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ eof = pScanTrans->nextScanResult();
+ oldest = 0;
+
+ while(eof == 0){
+ rows++;
+ Uint32 a = accountTypeRec->u_32_value();
+ Uint32 p = purgedRec->u_32_value();
+
+ if (a == accountType && p == 1){
+ // One record found
+ Uint64 t = timeRec->u_64_value();
+ if (t > oldest)
+ oldest = t;
+ }
+ eof = pScanTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pScanTrans);
+
+ return NDBT_OK;
+}
+
+int Bank::getOldestNotPurgedGL(Uint64 &oldest,
+ Uint32 &accountTypeId,
+ bool &found){
+ int check;
+ /**
+ * SELECT time, accountTypeId FROM GL
+ * WHERE purged=0 order by time asc
+ */
+ NdbConnection* pScanTrans = m_ndb.startTransaction();
+ if (pScanTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pScanTrans->getNdbOperation("GL");
+ if (pOp == NULL) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanRead(64);
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* accountTypeRec = pOp->getValue("ACCOUNT_TYPE");
+ if( accountTypeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* timeRec = pOp->getValue("TIME");
+ if( timeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* purgedRec = pOp->getValue("PURGED");
+ if( purgedRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pScanTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ eof = pScanTrans->nextScanResult();
+ oldest = (Uint64)-1;
+ found = false;
+
+ while(eof == 0){
+ rows++;
+ Uint32 p = purgedRec->u_32_value();
+ if (p == 0){
+ found = true;
+ // One record found
+ Uint32 a = accountTypeRec->u_32_value();
+ Uint64 t = timeRec->u_64_value();
+ if (t < oldest){
+ oldest = t;
+ accountTypeId = a;
+ }
+ }
+ eof = pScanTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pScanTrans);
+
+ return NDBT_OK;
+}
+
+
+int Bank::checkNoTransactionsOlderThan(const Uint32 accountType,
+ const Uint64 oldest){
+ /**
+ * SELECT COUNT(transaction_id) FROM TRANSACTION
+ * WHERE account_type = @accountType and time <= @oldest
+ *
+ */
+
+ int check;
+ NdbConnection* pScanTrans = m_ndb.startTransaction();
+ if (pScanTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pScanTrans->getNdbOperation("TRANSACTION");
+ if (pOp == NULL) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanRead(64);
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* accountTypeRec = pOp->getValue("ACCOUNT_TYPE");
+ if( accountTypeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* timeRec = pOp->getValue("TIME");
+ if( timeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* transactionIdRec = pOp->getValue("TRANSACTION_ID");
+ if( transactionIdRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pScanTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ int found = 0;
+ eof = pScanTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+ Uint32 a = accountTypeRec->u_32_value();
+ Uint32 t = timeRec->u_32_value();
+
+ if (a == accountType && t <= oldest){
+ // One record found
+ Uint64 ti = transactionIdRec->u_64_value();
+ g_err << "checkNoTransactionsOlderThan found one record" << endl
+ << " t = " << t << endl
+ << " a = " << a << endl
+ << " ti = " << ti << endl;
+ found++;
+ }
+ eof = pScanTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pScanTrans);
+
+ if (found == 0)
+ return NDBT_OK;
+ else
+ return VERIFICATION_FAILED;
+}
+
+
+ int Bank::performValidatePurged(){
+ /**
+ * Make sure there are no TRANSACTIONS older than the oldest
+ * purged GL record
+ *
+ */
+
+ for (int i = 0; i < getNumAccountTypes(); i++){
+ ndbout << "performValidatePurged: " << i << endl;
+ Uint64 oldestGlTime;
+ if (getOldestPurgedGL(i, oldestGlTime) != NDBT_OK){
+ g_err << "getOldestPurgedGL failed" << endl;
+ return NDBT_FAILED;
+ }
+ int result = checkNoTransactionsOlderThan(i, oldestGlTime);
+ if (result != NDBT_OK){
+ g_err << "checkNoTransactionsOlderThan failed" << endl;
+ return result;
+ }
+
+ }
+
+ return NDBT_OK;
+ }
+
+ int Bank::purgeOldGLTransactions(Uint64 currTime, Uint32 age){
+ /**
+ * For each GL record that are older than age and have purged == 0
+ * - delete all TRANSACTIONS belonging to the GL and set purged = 1
+ *
+ *
+ */
+ bool found;
+ int count = 0;
+
+ while(1){
+ count++;
+ if (count > 100)
+ return NDBT_OK;
+
+ // Search for the oldest GL record with purged == 0
+ Uint64 oldestGlTime;
+ Uint32 accountTypeId;
+ if (getOldestNotPurgedGL(oldestGlTime, accountTypeId, found) != NDBT_OK){
+ g_err << "getOldestNotPurgedGL failed" << endl;
+ return NDBT_FAILED;
+ }
+
+
+ if (found == false){
+ // ndbout << "not found" << endl;
+ return NDBT_OK;
+ }
+
+
+// ndbout << "purgeOldGLTransactions" << endl
+// << " oldestGlTime = " << oldestGlTime << endl
+// << " currTime = " << currTime << endl
+// << " age = " << age << endl;
+ // Check if this GL is old enough to be purged
+ if ((currTime < age) || (oldestGlTime > (currTime-age))){
+ // ndbout << "is not old enough" << endl;
+ return NDBT_OK;
+ }
+
+ if (purgeTransactions(oldestGlTime, accountTypeId) != NDBT_OK){
+ g_err << "purgeTransactions failed" << endl;
+ return NDBT_FAILED;
+ }
+ }
+ g_err << "abnormal return" << endl;
+ return NDBT_FAILED;
+ }
+
+
+int Bank::purgeTransactions(const Uint64 glTime,
+ const Uint32 accountTypeId)
+{
+ int check;
+ g_info << "purgeTransactions: " << glTime << ", "<<accountTypeId<<endl;
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL){
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Start by updating the GL record with purged = 1, use NoCommit
+ NdbOperation* pOp = pTrans->getNdbOperation("GL");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->updateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("TIME", glTime);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("ACCOUNT_TYPE", accountTypeId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ Uint32 purged = 1;
+ check = pOp->setValue("PURGED", purged);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Execute transaction
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Find all transactions and take over them for delete
+
+ if(findTransactionsToPurge(glTime,
+ accountTypeId,
+ pTrans) != NDBT_OK){
+ g_err << "findTransactionToPurge failed" << endl;
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_OK;
+}
+
+
+int Bank::findTransactionsToPurge(const Uint64 glTime,
+ const Uint32 accountType,
+ NdbConnection* pTrans){
+ int check;
+
+ NdbConnection* pScanTrans = m_ndb.startTransaction();
+ if (pScanTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pScanTrans->getNdbOperation("TRANSACTION");
+ if (pOp == NULL) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanExclusive(64);
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* timeRec = pOp->getValue("TIME");
+ if( timeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* accountTypeRec = pOp->getValue("ACCOUNT_TYPE");
+ if( accountTypeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pScanTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ int rowsFound = 0;
+ eof = pScanTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+ Uint64 t = timeRec->u_64_value();
+ Uint32 a = accountTypeRec->u_32_value();
+
+ if (a == accountType && t == glTime){
+ rowsFound++;
+ // One record found
+ NdbOperation* pDelOp = pOp->takeOverForDelete(pTrans);
+ if (pDelOp == NULL){
+ ERR(m_ndb.getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ // Execute transaction
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+ }
+ eof = pScanTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pScanTrans);
+ // ndbout << rowsFound << " TRANSACTIONS have been deleted" << endl;
+
+ return NDBT_OK;
+
+}
+
+
+ int Bank::performIncreaseTime(int maxSleepBetweenDays, int yield){
+ if (init() != NDBT_OK)
+ return NDBT_FAILED;
+
+ int yieldCounter = 0;
+
+ while(1){
+
+ while(m_ndb.waitUntilReady(10) != 0)
+ ndbout << "Waiting for ndb to be ready" << endl;
+
+ while(1){
+
+ Uint64 currTime;
+ if (incCurrTime(currTime) != NDBT_OK)
+ break;
+
+ g_info << "Current time is " << currTime << endl;
+ if (maxSleepBetweenDays > 0){
+ int val = myRandom48(maxSleepBetweenDays);
+ NdbSleep_SecSleep(val);
+ }
+
+ yieldCounter++;
+ if (yield != 0 && yieldCounter >= yield)
+ return NDBT_OK;
+
+ }
+ }
+ return NDBT_FAILED;
+ }
+
+
+
+int Bank::readSystemValue(SystemValueId sysValId, Uint64 & value){
+
+ int check;
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL){
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pTrans->getNdbOperation("SYSTEM_VALUES");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->readTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("SYSTEM_VALUES_ID", sysValId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* valueRec = pOp->getValue("VALUE");
+ if( valueRec ==NULL ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ value = valueRec->u_64_value();
+
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_OK;
+
+}
+
+int Bank::writeSystemValue(SystemValueId sysValId, Uint64 value){
+
+ int check;
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL){
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pTrans->getNdbOperation("SYSTEM_VALUES");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("SYSTEM_VALUES_ID", sysValId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->setValue("VALUE", value);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_OK;
+
+}
+
+int Bank::getNextTransactionId(Uint64 &value){
+ return increaseSystemValue2(LastTransactionId, value);
+}
+
+int Bank::incCurrTime(Uint64 &value){
+ return increaseSystemValue(CurrentTime, value);
+}
+
+
+int Bank::increaseSystemValue(SystemValueId sysValId, Uint64 &value){
+ /**
+ * Increase value with one and return
+ * updated value
+ *
+ */
+
+ int check;
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL){
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pTrans->getNdbOperation("SYSTEM_VALUES");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->readTupleExclusive();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("SYSTEM_VALUES_ID", sysValId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* valueRec = pOp->getValue("VALUE");
+ if( valueRec ==NULL ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ value = valueRec->u_64_value();
+ value++;
+
+ NdbOperation* pOp2 = pTrans->getNdbOperation("SYSTEM_VALUES");
+ if (pOp2 == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp2->updateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp2->equal("SYSTEM_VALUES_ID", sysValId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp2->setValue("VALUE", value);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp3 = pTrans->getNdbOperation("SYSTEM_VALUES");
+ if (pOp3 == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->readTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp3->equal("SYSTEM_VALUES_ID", sysValId);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Read new value
+ NdbRecAttr* valueNewRec = pOp3->getValue("VALUE");
+ if( valueNewRec ==NULL ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Check that value updated equals the value we read after the update
+ if (valueNewRec->u_64_value() != value){
+ g_err << "getNextTransactionId: value was not updated" << endl;
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pTrans);
+
+
+ return 0;
+
+}
+
+int Bank::increaseSystemValue2(SystemValueId sysValId, Uint64 &value){
+ /**
+ * Increase value with one and return
+ * updated value
+ * A more optimized version using interpreted update!
+ *
+ */
+
+ int check;
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL){
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pTrans->getNdbOperation("SYSTEM_VALUES");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpretedUpdateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("SYSTEM_VALUES_ID", sysValId );
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint32 valToIncWith = 1;
+ check = pOp->incValue("VALUE", valToIncWith);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* valueRec = pOp->getValue("VALUE");
+ if( valueRec == NULL ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ value = valueRec->u_64_value();
+
+ m_ndb.closeTransaction(pTrans);
+
+ return 0;
+
+}
+
+
+
+int Bank::getCurrTime(Uint64 &time){
+ return readSystemValue(CurrentTime, time);
+}
+
+
+int Bank::performSumAccounts(int maxSleepBetweenSums, int yield){
+ if (init() != NDBT_OK)
+ return NDBT_FAILED;
+
+ int yieldCounter = 0;
+
+ while (1){
+
+ while (m_ndb.waitUntilReady(10) != 0)
+ ndbout << "Waiting for ndb to be ready" << endl;
+
+ Uint32 sumAccounts = 0;
+ Uint32 numAccounts = 0;
+ if (getSumAccounts(sumAccounts, numAccounts) != NDBT_OK){
+ g_err << "getSumAccounts FAILED" << endl;
+ } else {
+
+ g_info << "num="<<numAccounts<<", sum=" << sumAccounts << endl;
+
+ if (sumAccounts != (10000000 + (10000*(numAccounts-1)))){
+ g_err << "performSumAccounts FAILED" << endl
+ << " sumAccounts="<<sumAccounts<<endl
+ << " expected ="<<(10000000 + (10000*(numAccounts-1)))<<endl
+ << " numAccounts="<<numAccounts<<endl;
+ return NDBT_FAILED;
+ }
+
+ if (maxSleepBetweenSums > 0){
+ int val = myRandom48(maxSleepBetweenSums);
+ NdbSleep_MilliSleep(val);
+ }
+ }
+
+ yieldCounter++;
+ if (yield != 0 && yieldCounter >= yield)
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+
+int Bank::getSumAccounts(Uint32 &sumAccounts,
+ Uint32 &numAccounts){
+
+ // SELECT SUM(balance) FROM ACCOUNT
+
+ int check;
+ NdbConnection* pScanTrans = m_ndb.startTransaction();
+ if (pScanTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pScanTrans->getNdbOperation("ACCOUNT");
+ if (pOp == NULL) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanExclusive(64);
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* balanceRec = pOp->getValue("BALANCE");
+ if( balanceRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pScanTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ eof = pScanTrans->nextScanResult();
+
+ while(eof == 0){
+ Uint32 b = balanceRec->u_32_value();
+
+ sumAccounts += b;
+ numAccounts++;
+
+ // ndbout << numAccounts << ": balance =" << b
+ // << ", sum="<< sumAccounts << endl;
+
+ // Take over the operation so that the lock is kept in db
+ NdbOperation* pLockOp = pOp->takeOverForUpdate(pTrans);
+ if (pLockOp == NULL){
+ ERR(m_ndb.getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint32 illegalBalance = 99;
+ check = pLockOp->setValue("BALANCE", illegalBalance);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ // Execute transaction
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ eof = pScanTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // TODO Forget about rolling back, just close pTrans!!
+
+ // Rollback transaction
+ check = pTrans->execute(Rollback);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pScanTrans);
+ m_ndb.closeTransaction(pTrans);
+
+
+ return NDBT_OK;
+
+}
diff --git a/ndb/test/ndbapi/bank/src/BankLoad.cpp b/ndb/test/ndbapi/bank/src/BankLoad.cpp
new file mode 100644
index 00000000000..985a49d5f64
--- /dev/null
+++ b/ndb/test/ndbapi/bank/src/BankLoad.cpp
@@ -0,0 +1,582 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "../Bank.hpp"
+#include <UtilTransactions.hpp>
+
+/**
+ * Default account types
+ *
+ */
+struct AccountTypesStruct {
+ int id;
+ const char* descr;
+};
+const AccountTypesStruct accountTypes[] = {
+ { 0, "KASSA"},
+ { 1, "BANKOMAT"},
+ { 2, "POSTGIRO"},
+ { 3, "LÖNEKONTO"},
+ { 4, "SPARKONTO"}
+};
+
+const int
+accountTypesSize = sizeof(accountTypes)/sizeof(AccountTypesStruct);
+
+
+const char* tableNames[] = {
+ "GL",
+ "ACCOUNT",
+ "SYSTEM_VALUES",
+ "TRANSACTION",
+ "ACCOUNT_TYPE"
+};
+
+const int
+tableNamesSize = sizeof(tableNames)/sizeof(const char*);
+
+
+int Bank::getNumAccountTypes(){
+ return accountTypesSize;
+}
+
+int Bank::createAndLoadBank(bool ovrWrt){
+
+ m_ndb.init();
+ if (m_ndb.waitUntilReady() != 0)
+ return NDBT_FAILED;
+
+ const NdbDictionary::Table* pSysValTab =
+ m_ndb.getDictionary()->getTable("SYSTEM_VALUES");
+ if (pSysValTab != NULL){
+ // The table exists
+ if (ovrWrt == false){
+ ndbout << "Bank already exist and overwrite == false" << endl;
+ return NDBT_FAILED;
+ }
+ }
+
+ if (createTables() != NDBT_OK)
+ return NDBT_FAILED;
+
+ if (clearTables() != NDBT_OK)
+ return NDBT_FAILED;
+
+ if (loadAccountType() != NDBT_OK)
+ return NDBT_FAILED;
+
+ if (loadAccount(10) != NDBT_OK)
+ return NDBT_FAILED;
+
+ if (loadSystemValues() != NDBT_OK)
+ return NDBT_FAILED;
+
+ if (loadGl() != NDBT_OK)
+ return NDBT_FAILED;
+
+ return NDBT_OK;
+
+}
+
+int Bank::dropBank(){
+
+ m_ndb.init();
+ if (m_ndb.waitUntilReady() != 0)
+ return NDBT_FAILED;
+
+ if (dropTables() != NDBT_OK)
+ return NDBT_FAILED;
+
+ return NDBT_OK;
+
+}
+
+int Bank::createTables(){
+ for (int i = 0; i < tableNamesSize; i++){
+ if (createTable(tableNames[i]) != NDBT_OK)
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+int Bank::dropTables(){
+ for (int i = 0; i < tableNamesSize; i++){
+ if (dropTable(tableNames[i]) != NDBT_OK)
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int Bank::clearTables(){
+ for (int i = 0; i < tableNamesSize; i++){
+ if (clearTable(tableNames[i]) != NDBT_OK)
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int Bank::clearTable(const char* tabName){
+ UtilTransactions util(&m_ndb, tabName);
+ if(util.clearTable(&m_ndb, 64) != 0)
+ return NDBT_FAILED;
+ return NDBT_OK;
+}
+
+int Bank::createTable(const char* tabName){
+ ndbout << "createTable " << tabName << endl;
+
+ const NdbDictionary::Table* pTab = NDBT_Tables::getTable(tabName);
+ if (pTab == NULL)
+ return NDBT_FAILED;
+
+ const NdbDictionary::Table* org =
+ m_ndb.getDictionary()->getTable(tabName);
+
+ if (org != 0 && pTab->equal(* org)){
+ return NDBT_OK;
+ }
+
+ if (org != 0){
+ ndbout << "Different table with same name exists" << endl;
+ return NDBT_FAILED;
+ }
+
+ if(m_ndb.getDictionary()->createTable(* pTab) == -1){
+ ndbout << "Failed to create table: " <<
+ m_ndb.getNdbError() << endl;
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+int Bank::dropTable(const char* tabName){
+ const NdbDictionary::Table* org =
+ m_ndb.getDictionary()->getTable(tabName);
+
+ if (org == NULL)
+ return NDBT_OK;
+
+ ndbout << "dropTable " <<tabName<<endl;
+ if (m_ndb.getDictionary()->dropTable(tabName) != 0){
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+
+
+
+
+
+
+
+/**
+ * Load SYSTEM_VALUES table
+ * This table keeps track of system wide settings
+ * For example:
+ * - next transaction id
+ *
+ */
+int Bank::loadSystemValues (){
+int result;
+
+/**
+ * Insert start value for next transaction id
+ *
+ */
+result = writeSystemValue(LastTransactionId, 0);
+
+/**
+ * Insert start value for current time
+ *
+ */
+result = writeSystemValue(CurrentTime, 1);
+
+return result;
+
+}
+
+
+/**
+ * Load GL table
+ *
+ * Insert GL records for time = 0 with balance 0
+ */
+int Bank::loadGl(){
+ g_info << "loadGl" << endl;
+ int check;
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL){
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ for (int i = 0; i < getNumAccountTypes(); i++){
+
+ NdbOperation* pOp = pTrans->getNdbOperation("GL");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint64 time = 0;
+ check = pOp->equal("TIME", time);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("ACCOUNT_TYPE", i);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint32 balance = 0;
+ if (getBalanceForAccountType(i, balance) != NDBT_OK){
+ return NDBT_FAILED;
+ }
+
+ check = pOp->setValue("BALANCE", balance);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint32 depositCount = 0;
+ check = pOp->setValue("DEPOSIT_COUNT", depositCount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint32 depositSum = 0;
+ check = pOp->setValue("DEPOSIT_SUM", depositSum);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint32 withdrawalCount = 0;
+ check = pOp->setValue("WITHDRAWAL_COUNT", withdrawalCount);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint32 withdrawalSum = 0;
+ check = pOp->setValue("WITHDRAWAL_SUM", withdrawalSum);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint32 purged = 1;
+ check = pOp->setValue("PURGED", purged);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ }
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_OK;
+};
+
+
+int Bank::getBalanceForAccountType(const Uint32 accountType,
+ Uint32& balance){
+ int check;
+ g_info << "getBalanceForAccountType: accountType="<<accountType<<endl;
+
+ NdbConnection* pScanTrans = m_ndb.startTransaction();
+ if (pScanTrans == NULL) {
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pScanTrans->getNdbOperation("ACCOUNT");
+ if (pOp == NULL) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanRead(64);
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* accountTypeRec = pOp->getValue("ACCOUNT_TYPE");
+ if( accountTypeRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* balanceRec = pOp->getValue("BALANCE");
+ if( balanceRec ==NULL ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pScanTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ eof = pScanTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+ Uint32 a = accountTypeRec->u_32_value();
+ Uint32 b = balanceRec->u_32_value();
+
+ if (a == accountType){
+ // One record found
+ balance += b;
+ }
+
+ eof = pScanTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ ERR(pScanTrans->getNdbError());
+ m_ndb.closeTransaction(pScanTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pScanTrans);
+ // ndbout << rows << " rows have been read" << endl;
+
+ return NDBT_OK;
+
+}
+
+/**
+ * Load ACCOUNT_TYPE table
+ *
+ *
+ */
+int Bank::loadAccountType(){
+ g_info << "loadAccountType" << endl;
+ int check;
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL){
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ for (int i = 0; i < getNumAccountTypes(); i++){
+
+ NdbOperation* pOp = pTrans->getNdbOperation("ACCOUNT_TYPE");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("ACCOUNT_TYPE_ID", accountTypes[i].id);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->setValue("DESCRIPTION", accountTypes[i].descr);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_OK;
+};
+
+/**
+ * Load ACCOUNT table
+ *
+ *
+ *
+ */
+int Bank::loadAccount (int numAccounts){
+ g_info << "loadAccount" << endl;
+ int check;
+
+ NdbConnection* pTrans = m_ndb.startTransaction();
+ if (pTrans == NULL){
+ ERR(m_ndb.getNdbError());
+ return NDBT_FAILED;
+ }
+
+ for (int i = 0; i < numAccounts; i++){
+
+ NdbOperation* pOp = pTrans->getNdbOperation("ACCOUNT");
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->equal("ACCOUNT_ID", i);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int owner;
+ if (i == 0)
+ owner = 0;
+ else
+ owner = i + 3000;
+ check = pOp->setValue("OWNER", owner);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Load balance so that the bank's account = 0 has 10 millions
+ // and all other accounts have 10000
+ // This set the total balance for the entire bank to
+ // 10000000 + (10000 * numAccounts-1)
+ // Since no money should dissapear from to the bank nor
+ // any money should be added this is a rule that can be checked when
+ // validating the db
+ int balance;
+ if (i == 0){
+ balance = 10000000;
+ } else {
+ balance = 10000;
+ }
+ check = pOp->setValue("BALANCE", balance);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // TODO - This is how to set a value in a 16, 1 attribute, not so nice?
+ // NOTE - its not even possible to set the value 0 in this column
+ // since that is equal to NULL when casting to char*
+ // check = pOp->setValue("ACCOUNT_TYPE", (const char*)(Uint16)(i/accountTypesSize), 2);
+ // NOTE attribute now changed to be a 32 bit
+
+
+ int accountType;
+ if (i == 0)
+ accountType = 0; // KASSA
+ else
+ accountType = ((i%accountTypesSize) == 0 ? 1 : (i%getNumAccountTypes()));
+ check = pOp->setValue("ACCOUNT_TYPE", accountType);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ m_ndb.closeTransaction(pTrans);
+ return NDBT_OK;
+}
+
+
+int Bank::getNumAccounts(){
+ const NdbDictionary::Table* accountTab =
+ m_ndb.getDictionary()->getTable("ACCOUNT");
+ if (accountTab == NULL){
+ g_err << "Table ACCOUNT does not exist" << endl;
+ return NDBT_FAILED;
+ }
+ UtilTransactions util(*accountTab);
+ if(util.selectCount(&m_ndb, 64, &m_maxAccount) != 0)
+ return NDBT_FAILED;
+ return NDBT_OK;
+}
+
+int Bank::getMaxAmount(){
+ return 10000;
+}
diff --git a/ndb/test/ndbapi/bank/src/Makefile b/ndb/test/ndbapi/bank/src/Makefile
new file mode 100644
index 00000000000..e0ab8e0e536
--- /dev/null
+++ b/ndb/test/ndbapi/bank/src/Makefile
@@ -0,0 +1,7 @@
+include .defs.mk
+
+TYPE = ndbapitest
+ARCHIVE_TARGET = bank
+SOURCES = Bank.cpp BankLoad.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bank/testBank/Makefile b/ndb/test/ndbapi/bank/testBank/Makefile
new file mode 100644
index 00000000000..382aaadad7c
--- /dev/null
+++ b/ndb/test/ndbapi/bank/testBank/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testBank
+BIN_TARGET_LIBS += bank
+SOURCES = testBank.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bank/testBank/testBank.cpp b/ndb/test/ndbapi/bank/testBank/testBank.cpp
new file mode 100644
index 00000000000..094ecaa499c
--- /dev/null
+++ b/ndb/test/ndbapi/bank/testBank/testBank.cpp
@@ -0,0 +1,150 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <NdbBackup.hpp>
+
+
+#define CHECK(b) if (!(b)) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ continue; }
+
+
+#include "../Bank.hpp"
+
+int runCreateBank(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int overWriteExisting = true;
+ if (bank.createAndLoadBank(overWriteExisting) != NDBT_OK)
+ return NDBT_FAILED;
+ return NDBT_OK;
+}
+
+int runBankTimer(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int wait = 30; // Max seconds between each "day"
+ int yield = 1; // Loops before bank returns
+
+ while (ctx->isTestStopped() == false) {
+ bank.performIncreaseTime(wait, yield);
+ }
+ return NDBT_OK;
+}
+
+int runBankTransactions(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int wait = 10; // Max ms between each transaction
+ int yield = 100; // Loops before bank returns
+
+ while (ctx->isTestStopped() == false) {
+ bank.performTransactions(wait, yield);
+ }
+ return NDBT_OK;
+}
+
+int runBankGL(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int yield = 20; // Loops before bank returns
+ int result = NDBT_OK;
+
+ while (ctx->isTestStopped() == false) {
+ if (bank.performMakeGLs(yield) != NDBT_OK){
+ ndbout << "bank.performMakeGLs FAILED" << endl;
+ result = NDBT_FAILED;
+ }
+ }
+ return NDBT_OK;
+}
+
+int runBankSum(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int wait = 2000; // Max ms between each sum of accounts
+ int yield = 1; // Loops before bank returns
+ int result = NDBT_OK;
+
+ while (ctx->isTestStopped() == false) {
+ if (bank.performSumAccounts(wait, yield) != NDBT_OK){
+ ndbout << "bank.performSumAccounts FAILED" << endl;
+ result = NDBT_FAILED;
+ }
+ }
+ return result ;
+}
+
+int runDropBank(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ if (bank.dropBank() != NDBT_OK)
+ return NDBT_FAILED;
+ return NDBT_OK;
+}
+
+int runBankController(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int l = 0;
+ int result = NDBT_OK;
+
+ while (l < loops && result != NDBT_FAILED){
+
+ if (pNdb->waitUntilReady() != 0){
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ // Sleep for a while
+ NdbSleep_SecSleep(records);
+
+ l++;
+ }
+
+ if (pNdb->waitUntilReady() != 0){
+ result = NDBT_FAILED;
+ }
+
+ ctx->stopTest();
+
+ return result;
+}
+
+NDBT_TESTSUITE(testBank);
+TESTCASE("Bank",
+ "Run the bank\n"){
+ INITIALIZER(runCreateBank);
+ STEP(runBankTimer);
+ STEP(runBankTransactions);
+ STEP(runBankGL);
+ // TODO STEP(runBankSum);
+ STEP(runBankController);
+ FINALIZER(runDropBank);
+
+}
+NDBT_TESTSUITE_END(testBank);
+
+int main(int argc, const char** argv){
+ // Tables should not be auto created
+ testBank.setCreateTable(false);
+
+ return testBank.execute(argc, argv);
+}
+
+
diff --git a/ndb/test/ndbapi/basicAsynch/Makefile b/ndb/test/ndbapi/basicAsynch/Makefile
new file mode 100755
index 00000000000..802c5e5a2bd
--- /dev/null
+++ b/ndb/test/ndbapi/basicAsynch/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := testBasicAsynch
+
+SOURCES := testBasicAsynch.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/basicAsynch/testBasicAsynch.cpp b/ndb/test/ndbapi/basicAsynch/testBasicAsynch.cpp
new file mode 100755
index 00000000000..a97920e53da
--- /dev/null
+++ b/ndb/test/ndbapi/basicAsynch/testBasicAsynch.cpp
@@ -0,0 +1,186 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NDBT_Test.hpp"
+#include "NDBT_ReturnCodes.h"
+#include "HugoTransactions.hpp"
+#include "HugoAsynchTransactions.hpp"
+#include "UtilTransactions.hpp"
+
+#define GETNDB(ps) ((NDBT_NdbApiStep*)ps)->getNdb()
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int transactions = (records / 100) + 1;
+ int operations = (records / transactions) + 1;
+
+ HugoAsynchTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTableAsynch(GETNDB(step), records, batchSize,
+ transactions, operations) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runInsert(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int transactions = (records / 100) + 1;
+ int operations = (records / transactions) + 1;
+
+ HugoAsynchTransactions hugoTrans(*ctx->getTab());
+ // Insert records, dont allow any
+ // errors(except temporary) while inserting
+ if (hugoTrans.loadTableAsynch(GETNDB(step), records, batchSize,
+ transactions, operations) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runVerifyInsert(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int transactions = (records / 100) + 1;
+ int operations = (records / transactions) + 1;
+
+ HugoAsynchTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.pkDelRecordsAsynch(GETNDB(step), records, batchSize,
+ transactions, operations) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int transactions = (records / 100) + 1;
+ int operations = (records / transactions) + 1;
+
+ HugoAsynchTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.pkDelRecordsAsynch(GETNDB(step), records, batchSize,
+ transactions, operations) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runPkDelete(NDBT_Context* ctx, NDBT_Step* step){
+
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int transactions = (records / 100) + 1;
+ int operations = (records / transactions) + 1;
+
+ int i = 0;
+ HugoAsynchTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ ndbout << i << ": ";
+ if (hugoTrans.pkDelRecordsAsynch(GETNDB(step), records, batchSize,
+ transactions, operations) != 0){
+ return NDBT_FAILED;
+ }
+ // Load table, don't allow any primary key violations
+ if (hugoTrans.loadTableAsynch(GETNDB(step), records, batchSize,
+ transactions, operations) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+
+int runPkRead(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int transactions = (records / 100) + 1;
+ int operations = (records / transactions) + 1;
+
+ int i = 0;
+ HugoAsynchTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ ndbout << i << ": ";
+ if (hugoTrans.pkReadRecordsAsynch(GETNDB(step), records, batchSize,
+ transactions, operations) != NDBT_OK){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runPkUpdate(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int transactions = (records / 100) + 1;
+ int operations = (records / transactions) + 1;
+
+ int i = 0;
+ HugoAsynchTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ ndbout << i << ": ";
+ if (hugoTrans.pkUpdateRecordsAsynch(GETNDB(step), records,
+ batchSize, transactions,
+ operations) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+NDBT_TESTSUITE(testBasicAsynch);
+TESTCASE("PkInsertAsynch",
+ "Verify that we can insert and delete from this table using PK"
+ " NOTE! No errors are allowed!" ){
+ INITIALIZER(runInsert);
+ VERIFIER(runVerifyInsert);
+}
+TESTCASE("PkReadAsynch",
+ "Verify that we can insert, read and delete from this table"
+ " using PK"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkUpdateAsynch",
+ "Verify that we can insert, update and delete from this table"
+ " using PK"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkUpdate);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkDeleteAsynch",
+ "Verify that we can delete from this table using PK"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkDelete);
+ FINALIZER(runClearTable);
+}
+
+NDBT_TESTSUITE_END(testBasicAsynch);
+
+int main(int argc, const char** argv){
+ return testBasicAsynch.execute(argc, argv);
+}
+
diff --git a/ndb/test/ndbapi/bulk_copy/Makefile b/ndb/test/ndbapi/bulk_copy/Makefile
new file mode 100644
index 00000000000..22c05b138b7
--- /dev/null
+++ b/ndb/test/ndbapi/bulk_copy/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := bulk_copy
+
+SOURCES := bulk_copy.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/bulk_copy/bulk_copy.cpp b/ndb/test/ndbapi/bulk_copy/bulk_copy.cpp
new file mode 100644
index 00000000000..f2b28d8b057
--- /dev/null
+++ b/ndb/test/ndbapi/bulk_copy/bulk_copy.cpp
@@ -0,0 +1,276 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbStdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NdbSleep.h>
+#include <NDBT.hpp>
+
+#include <getarg.h>
+
+
+int setValuesFromLine(NdbOperation* pOp,
+ const NdbDictionary::Table* pTab,
+ char* line){
+
+ int check = 0;
+ char* p = line;
+ char* pn;
+ char buf[8000];
+ // Loop through each attribute in this table
+ for (int a = 0; a<pTab->getNoOfColumns(); a++){
+
+ pn = p;
+ while (*pn != ';')
+ pn++;
+
+ memset(buf, 0, sizeof(buf));
+ strncpy(buf, p, pn-p);
+ // ndbout << a << ": " << buf << endl;
+ const NdbDictionary::Column* attr = pTab->getColumn(a);
+ switch (attr->getType()){
+ case NdbDictionary::Column::Unsigned:
+ Int32 sval;
+ if (sscanf(buf, "%d", &sval) == 0)
+ return -2;
+ check = pOp->setValue(a, sval);
+ break;
+
+ case NdbDictionary::Column::Int:
+ Uint32 uval;
+ if (sscanf(buf, "%u", &uval) == 0)
+ return -2;
+ check = pOp->setValue(a, uval);
+ break;
+
+ case NdbDictionary::Column::Char:
+ char buf2[8000];
+ char* p2;
+ memset(buf2, 0, sizeof(buf));
+ p2 = &buf2[0];
+ while(*p != ';'){
+ *p2 = *p;
+ p++;p2++;
+ };
+ *p2 = 0;
+ check = pOp->setValue(a, buf2);
+ break;
+
+ default:
+ check = -2;
+ break;
+ }
+
+ // Move pointer to after next ";"
+ while (*p != ';')
+ p++;
+ p++;
+
+ }
+
+ return check;
+}
+
+
+int insertLine(Ndb* pNdb,
+ const NdbDictionary::Table* pTab,
+ char* line){
+ int check;
+ int retryAttempt = 0;
+ int retryMax = 5;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ while (retryAttempt < retryMax){
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ ERR(pNdb->getNdbError());
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+
+ pOp = pTrans->getNdbOperation(pTab->getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+
+ check = pOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+
+ check = setValuesFromLine(pOp,
+ pTab,
+ line);
+ if (check == -2){
+ pNdb->closeTransaction(pTrans);
+ return -2;
+ }
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+
+
+ // Execute the transaction and insert the record
+ check = pTrans->execute( Commit );
+ if(check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ pNdb->closeTransaction(pTrans);
+
+ switch(err.status){
+ case NdbError::Success:
+ ERR(err);
+ ndbout << "ERROR: NdbError reports success when transcaction failed" << endl;
+ return -1;
+ break;
+
+ case NdbError::TemporaryError:
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ break;
+
+ case NdbError::UnknownResult:
+ ERR(err);
+ return -1;
+ break;
+
+ case NdbError::PermanentError:
+ switch (err.classification){
+ case NdbError::ConstraintViolation:
+ // Tuple already existed, OK in this application, but should be reported
+ ndbout << err.code << " " << err.message << endl;
+ break;
+ default:
+ ERR(err);
+ return -1;
+ break;
+ }
+ break;
+ }
+ }
+ else{
+
+ pNdb->closeTransaction(pTrans);
+ }
+ return 0;
+ }
+ return check;
+}
+
+int insertFile(Ndb* pNdb,
+ const NdbDictionary::Table* pTab,
+ const char* fileName){
+
+ const int MAX_LINE_LEN = 8000;
+ char line[MAX_LINE_LEN];
+ int lineNo = 0;
+
+ FILE* instr = fopen(fileName, "r");
+ if (instr == NULL){
+ ndbout << "Coul'd not open " << fileName << endl;
+ return -1;
+ }
+
+ while(fgets(line, MAX_LINE_LEN, instr)){
+ lineNo++;
+
+ if (line[strlen(line)-1] == '\n') {
+ line[strlen(line)-1] = '\0';
+ }
+
+ int check = insertLine(pNdb, pTab, line);
+ if (check == -2){
+ ndbout << "Wrong format in input data file, line: " << lineNo << endl;
+ fclose(instr);
+ return -1;
+ }
+ if (check == -1){
+ fclose(instr);
+ return -1;
+
+ }
+ }
+
+ fclose(instr);
+ return 0;
+}
+
+
+int main(int argc, const char** argv){
+
+ const char* _tabname = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will bulk copy data from a file to a table in Ndb.\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+ ndbout << "Tablename: " << _tabname << endl;
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ // Connect to Ndb and wait for it to become ready
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table* pTab = MyNdb.getDictionary()->getTable(_tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ char buf[255];
+ snprintf(buf, sizeof(buf), "%s.data", (const char*)_tabname);
+ if (insertFile(&MyNdb, pTab, buf) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+
+}
+
+
+
diff --git a/ndb/test/ndbapi/cello-sessionDb/celloDb.cpp b/ndb/test/ndbapi/cello-sessionDb/celloDb.cpp
new file mode 100644
index 00000000000..ec61e783585
--- /dev/null
+++ b/ndb/test/ndbapi/cello-sessionDb/celloDb.cpp
@@ -0,0 +1,1503 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/* ***************************************************
+ BASIC TEST 1
+ Test basic functions and status of NDB
+
+ Arguments:
+ none
+
+ Returns:
+ 0 - Test passed
+ -1 - Test failed
+ 1 - Invalid arguments
+flexBench
+ * *************************************************** */
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+
+#define MAXATTR 4
+#define MAXTABLES 4
+#define PAGESIZE 8192
+#define OVERHEAD 0.02
+#define NUMBEROFRECORDS 10
+#define PKSIZE 1
+#define ATTRNAMELEN 16
+
+
+static void createTable_IPACCT(Ndb*);
+static void createTable_RPACCT(Ndb*);
+static void createTable_SBMCALL(Ndb*);
+static void createTable_TODACCT(Ndb*);
+
+static void error_handler(const char*);
+static bool error_handler2(const char*, int) ;
+
+static void setAttrNames(void);
+static void setTableNames(void);
+static void create_table(Ndb*);
+static void insert_rows(Ndb*);
+static void update_rows(Ndb*);
+static void delete_rows(Ndb*);
+static void verify_deleted(Ndb*);
+static void read_and_verify_rows(Ndb*);
+
+static void insert_IPACCT(Ndb*, Uint32, Uint32, Uint32, Uint32, Uint32); //3 for Pk, and two data. just to test;
+
+static void read_IPACCT(Ndb* , Uint32 , Uint32 , Uint32 );
+
+static int tAttributeSize;
+static int bTestPassed;
+
+static char tableName[MAXTABLES][ATTRNAMELEN];static char attrName[MAXATTR][ATTRNAMELEN];
+static int attrValue[NUMBEROFRECORDS];
+static int pkValue[NUMBEROFRECORDS];
+static int failed = 0 ;
+#include <NdbOut.hpp>
+
+NDB_COMMAND(celloDb, "celloDb", "celloDb", "celloDb", 65535)
+{
+
+ int tTableId;
+ int i;
+ Ndb MyNdb( "CELLO-SESSION-DB" );
+
+ MyNdb.init();
+
+ // Assume test passed
+ bTestPassed = 0;
+ /*
+ // Initialize global variables
+ for (i = 0; i < NUMBEROFRECORDS; i ++)
+ pkValue[i] = i;
+
+ for (i = 0; i < NUMBEROFRECORDS; i ++)
+ attrValue[i] = i;
+ */
+ tAttributeSize = 1;
+
+ // Wait for Ndb to become ready
+ if (MyNdb.waitUntilReady() == 0) {
+ ndbout << endl << "Cello session db - Starting " << endl;
+ ndbout << "Test basic functions and status of NDB" << endl;
+
+
+
+ createTable_SBMCALL (&MyNdb );
+ createTable_RPACCT (&MyNdb );
+ createTable_TODACCT (&MyNdb );
+ createTable_IPACCT (&MyNdb );
+
+ insert_IPACCT(&MyNdb, 1,2,1,2,2);
+ read_IPACCT(&MyNdb, 1, 2 , 1);
+ /*
+ insert_rows(&MyNdb);
+
+ read_and_verify_rows(&MyNdb);
+
+
+ // Create some new values to use for update
+ for (i = 0; i < NUMBEROFRECORDS; i++)
+ attrValue[i] = NUMBEROFRECORDS-i;
+
+ update_rows(&MyNdb);
+
+ read_and_verify_rows(&MyNdb);
+
+ delete_rows(&MyNdb);
+
+ verify_deleted(&MyNdb);
+ */
+ }
+ else {
+ bTestPassed = -1;
+ }
+
+
+ if (bTestPassed == 0)
+ {
+ // Test passed
+ ndbout << "OK - Test passed" << endl;
+ }
+ else
+ {
+ // Test failed
+ ndbout << "FAIL - Test failed" << endl;
+ exit(-1);
+ }
+ return NULL;
+}
+
+static void
+error_handler(const char* errorText)
+{
+ // Test failed
+ ndbout << endl << "ErrorMessage: " << errorText << endl;
+ bTestPassed = -1;
+}
+
+static void
+createTable_SBMCALL ( Ndb* pMyNdb )
+{
+ /****************************************************************
+ * Create table and attributes.
+ *
+ * create table SBMCALL(
+ * for attribs, see the REQ SPEC for cello session DB
+ * )
+ *
+ ***************************************************************/
+
+ const char* tname = "SBMCALL";
+ Uint32 recordsize = 244; //including 12 byte overhead
+ Uint32 pksize = 8; //size of total prim. key. in bytes. sum of entire composite PK.
+ Uint32 tTableId = pMyNdb->getTable()->openTable(tname);
+
+ if (tTableId == -1) {
+ Uint32 check;
+ Uint32 i;
+ NdbSchemaCon *MySchemaTransaction;
+ NdbSchemaOp *MySchemaOp;
+
+ ndbout << "Creating " << tname << "..." << endl;
+
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+ if( ( MySchemaTransaction == NULL ) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( ( MySchemaOp == NULL ) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Createtable
+
+ Uint32 tablesize=(recordsize*NUMBEROFRECORDS + OVERHEAD * NUMBEROFRECORDS)/1024;
+ Uint32 noPages=(pksize*NUMBEROFRECORDS)/PAGESIZE;
+
+ ndbout << "table size " << tablesize << "for table name " << tname << endl;
+
+ check = MySchemaOp->createTable( tname,
+ tablesize, // Table Size
+ TupleKey, // Key Type
+ noPages, // Nr of Pages
+ All,
+ 6,
+ 78,
+ 80,
+ 1,
+ true
+ );
+
+ if( check == -1 ) {
+ error_handler(MySchemaTransaction->getNdbErrorString());
+ exit(-1);
+ }
+
+
+
+ // Create first column, primary key
+ check = MySchemaOp->createAttribute( "SPBBOARDID",
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create second column, primary key
+ check = MySchemaOp->createAttribute( "CALLID",
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Creat thrid column, RP Session info, byte[16] represented as (String, 16)
+ check = MySchemaOp->createAttribute( "RPSESS",
+ NoKey,
+ 32,
+ 16,
+ String,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+// Creat fourth column, GRE Tunnel info, byte[16] represented as (String, 16)
+ check = MySchemaOp->createAttribute( "GRETUNNEL",
+ NoKey,
+ 32,
+ 16,
+ String,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+// Creat fifth column, PPP Session info, byte[24] represented as (String, 24)
+ check = MySchemaOp->createAttribute( "PPPSESS",
+ NoKey,
+ 32,
+ 24,
+ String,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ if( (MySchemaTransaction->execute() == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ ndbout << "done" << endl;
+
+
+ } //if
+
+ //else table already created , proceed
+}
+
+
+
+static void
+createTable_RPACCT(Ndb*pMyNdb)
+{
+
+ /****************************************************************
+ * Create table and attributes.
+ *
+ * create table RPACCT(
+ * for attribs, see the REQ SPEC for cello session DB
+ * )
+ *
+ ***************************************************************/
+
+ const char* tname = "RPACCT";
+ Uint32 recordsize = 380; //including 12 byte overhead
+ Uint32 pksize = 8; //size of total prim. key. in bytes.
+ Uint32 tTableId = pMyNdb->getTable()->openTable(tname);
+
+ if (tTableId == -1) {
+ Uint32 check;
+ Uint32 i;
+ NdbSchemaCon *MySchemaTransaction;
+ NdbSchemaOp *MySchemaOp;
+
+ ndbout << "Creating " << tname << "..." << endl;
+
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+ if( MySchemaTransaction == NULL )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // Createtable
+
+ Uint32 tablesize=(recordsize*NUMBEROFRECORDS + OVERHEAD * NUMBEROFRECORDS)/1024;
+ Uint32 noPages=(pksize*NUMBEROFRECORDS)/PAGESIZE;
+
+ ndbout << "table size " << tablesize << "for table name " << tname << endl;
+
+ check = MySchemaOp->createTable( tname,
+ tablesize, // Table Size
+ TupleKey, // Key Type
+ noPages // Nr of Pages
+ );
+
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+
+
+ // Create first column, primary key
+ check = MySchemaOp->createAttribute( "SPBBOARDID",
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create second column, primary key
+ check = MySchemaOp->createAttribute( "CALLID",
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Creat thrid column MS ID, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "MSID",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create PDSN FA Address, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "PDSN",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create Serving PCF, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "SPCF",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+ // Create BS ID, 12 byte char, represented as String,12
+ check = MySchemaOp->createAttribute( "BSID",
+ NoKey,
+ 32,
+ 12,
+ String,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+
+
+ // Create User Zone, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "UZ",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create Forward Multiplex, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "FMO",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create Reverse Multiplex, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "RMO",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create Forward Fund rate, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "FFR",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create Reverse Fund rate, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "RFR",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+ // Create Service Option, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "SO",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+
+ // Create Forward Traffic Type, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "FTT",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+// Create Reverse Traffic Type, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "RTT",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+// Create Fund Frame Size, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "FFS",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+// Create Forware Fund RC, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "FFRC",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+ // Create Reverse Fund RC, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "RFRC",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create DCCH Frame Format, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "DCCH",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create Airlink QOS, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "AQOS",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+// Create Bad PPP Frame Count , 4 byte unsigned
+ check = MySchemaOp->createAttribute( "BPPPFC",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+
+// Create Active Time , 4 byte unsigned
+ check = MySchemaOp->createAttribute( "AT",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+// Create Nb Active Transitions , 4 byte unsigned
+ check = MySchemaOp->createAttribute( "NBAT",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+// Create SDB Octet Count In , 4 byte unsigned
+ check = MySchemaOp->createAttribute( "SDBOCI",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+
+// Create Nb SDB In, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "NBSDBI",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+
+// Create Nb SDB Out, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "NBSDBO",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+
+// Create HDLC Bytes received, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "HDLC",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+ if( (MySchemaTransaction->execute() == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ ndbout << "done" << endl;
+
+ } //if
+
+ //else table already created , proceed
+ }
+
+
+static void
+createTable_IPACCT(Ndb* pMyNdb)
+{
+
+
+ /****************************************************************
+ * Create table and attributes.
+ *
+ * create table IPACCT(
+ * for attribs, see the REQ SPEC for cello session DB
+ * )
+ *
+ ***************************************************************/
+
+ const char* tname = "IPACCT";
+ Uint32 recordsize = 70; //including 12 byte overhead
+ Uint32 pksize = 12; //size of total prim. key. in bytes.
+ Uint32 tTableId = pMyNdb->getTable()->openTable(tname);
+
+ if (tTableId == -1) {
+ Uint32 check;
+ Uint32 i;
+ NdbSchemaCon *MySchemaTransaction;
+ NdbSchemaOp *MySchemaOp;
+
+ ndbout << "Creating " << tname << "..." << endl;
+
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+ if( MySchemaTransaction == NULL )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // Createtable
+
+ Uint32 tablesize=(recordsize*NUMBEROFRECORDS + OVERHEAD * NUMBEROFRECORDS)/1024;
+ Uint32 noPages=(pksize*NUMBEROFRECORDS)/PAGESIZE;
+
+ ndbout << "table size " << tablesize << "for table name " << tname << endl;
+
+ check = MySchemaOp->createTable( tname,
+ tablesize, // Table Size
+ TupleKey, // Key Type
+ noPages // Nr of Pages
+ );
+
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+
+
+ // Create first column, primary key
+ check = MySchemaOp->createAttribute( "SPBBOARDID",
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create second column, primary key
+ check = MySchemaOp->createAttribute( "CALLID",
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ // Create third column, primary key
+ check = MySchemaOp->createAttribute( "IPADDR",
+ TupleKey,
+ 32,
+ PKSIZE,
+ String,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+
+// Create Acct session id 4 byte unsigned
+ check = MySchemaOp->createAttribute( "ASID",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+// Create Correlation ID, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "CID",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+// Create MIP HA Address, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "MIPHA",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+
+
+// Create IP technology, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "IPT",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+// Create Compuls Tunnel ID, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "CTID",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+// Create IP QOS, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "IPQOS",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create Data octet count in, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "DOCI",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ // Create Data octet count out, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "DOCO",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ // Create Event time, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "ET",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ // Create In mip sig count, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "IMSC",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+// Create Out mip sig count, 4 byte unsigned
+ check = MySchemaOp->createAttribute( "OMSC",
+ NoKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ if( (MySchemaTransaction->execute() == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ ndbout << "done" << endl;
+
+
+
+ } //if
+
+ //else table already created , proceed
+}
+
+
+static void
+createTable_TODACCT(Ndb* pMyNdb)
+{
+
+
+ /****************************************************************
+ * Create table and attributes.
+ *
+ * create table TODACCT(
+ * for attribs, see the REQ SPEC for cello session DB
+ * )
+ *
+ ***************************************************************/
+
+ const char* tname = "TODACCT";
+ Uint32 recordsize = 92; //including 12 byte overhead
+ Uint32 pksize = 12; //size of total prim. key. in bytes.
+ Uint32 tTableId = pMyNdb->getTable()->openTable(tname);
+
+ if (tTableId == -1) {
+ Uint32 check;
+ Uint32 i;
+ NdbSchemaCon *MySchemaTransaction;
+ NdbSchemaOp *MySchemaOp;
+
+ ndbout << "Creating " << tname << "..." << endl;
+
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+ if( MySchemaTransaction == NULL )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // Createtable
+
+ Uint32 tablesize=(recordsize*NUMBEROFRECORDS + OVERHEAD * NUMBEROFRECORDS)/1024;
+ Uint32 noPages=(pksize*NUMBEROFRECORDS)/PAGESIZE;
+
+ ndbout << "table size " << tablesize << "for table name " << tname << endl;
+
+ check = MySchemaOp->createTable( tname,
+ tablesize, // Table Size
+ TupleKey, // Key Type
+ noPages // Nr of Pages
+ );
+
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+
+
+ // Create first column, primary key
+ check = MySchemaOp->createAttribute( "SPBBOARDID",
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+ // Create second column, primary key
+ check = MySchemaOp->createAttribute( "CALLID",
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ // Create third column, primary key
+ check = MySchemaOp->createAttribute( "IPADDR",
+ TupleKey,
+ 32,
+ PKSIZE,
+ String,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+ // Create third column, primary key
+ check = MySchemaOp->createAttribute( "INDEX",
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+
+// Create Acct session id 4 byte unsigned
+ check = MySchemaOp->createAttribute( "TOD",
+ NoKey,
+ 32,
+ 16,
+ String,
+ MMBased,
+ NullAttribute );
+
+
+ if( (check == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+
+
+ if( (MySchemaTransaction->execute() == -1) && (!error_handler2((const char*)MySchemaTransaction->getNdbErrorString(), MySchemaTransaction->getNdbError())) )
+ exit (-1) ;
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ ndbout << "done" << endl;
+
+ } //if
+
+ //else table already created , proceed
+}
+
+
+
+
+
+
+static void read_IPACCT(Ndb* pMyNdb, Uint32 CALLID, Uint32 SPBBOARDID , Uint32 IPADDR)
+{
+
+
+ int check;
+ int loop_count_ops;
+ int count;
+ int count_attributes;
+ char* value;
+ NdbConnection *MyTransaction;
+ NdbOperation *MyOperation;
+ NdbRecAttr* tTmp;
+
+
+
+ MyTransaction = pMyNdb->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler(pMyNdb->getNdbErrorString());
+
+ MyOperation = MyTransaction->getNdbOperation("IPACCT");
+ if (MyOperation == NULL)
+ error_handler( MyTransaction->getNdbErrorString());
+
+ check = MyOperation->readTuple();
+ if( check == -1 )
+ error_handler( MyTransaction->getNdbErrorString());
+
+ check = MyOperation->equal( "SPBBOARDID",SPBBOARDID );
+ if( check == -1 )
+ error_handler( MyTransaction->getNdbErrorString());
+
+
+ check = MyOperation->equal( "IPADDR","IPADDR" );
+ if( check == -1 )
+ error_handler( MyTransaction->getNdbErrorString());
+
+
+ check = MyOperation->equal( "CALLID",CALLID );
+ if( check == -1 )
+ error_handler( MyTransaction->getNdbErrorString());
+
+
+
+
+ tTmp = MyOperation->getValue("IPQOS", NULL );
+ if( tTmp == NULL )
+ error_handler( MyTransaction->getNdbErrorString());
+ ndbout << " tTmp " << tTmp->isNULL() << endl;
+ MyTransaction->execute(Commit);
+
+ ndbout << " value read " << tTmp->int32_value() << endl;
+
+}
+
+
+
+static void insert_IPACCT(Ndb* pMyNdb, Uint32 CALLID, Uint32 SPBBOARDID , Uint32 IPADDR, Uint32 ASID, Uint32 IPQOS)
+{
+ /****************************************************************
+ * Insert rows
+ *
+ ***************************************************************/
+
+ int check;
+ int loop_count_ops;
+ int count;
+ int i;
+ NdbConnection *MyTransaction;
+ NdbOperation *MyOperation;
+
+ ndbout << "Inserting records..." << flush;
+
+ MyTransaction = pMyNdb->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler(pMyNdb->getNdbErrorString());
+
+ MyOperation = MyTransaction->getNdbOperation("IPACCT");
+ if (MyOperation == NULL)
+ error_handler(MyTransaction->getNdbErrorString());
+
+
+
+ check = MyOperation->insertTuple();
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+
+ ndbout << "insertTuple" << endl;
+
+ check = MyOperation->equal("CALLID",CALLID );
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ ndbout << "equal" << endl;
+
+
+ check = MyOperation->equal("SPBBOARDID",SPBBOARDID );
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ ndbout << "equal" << endl;
+
+
+ check = MyOperation->equal("IPADDR","IPADDR" );
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ ndbout << "equal" << endl;
+
+
+ check = MyOperation->setValue( "IPQOS", IPQOS);
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ ndbout << "Set Value" << endl;
+
+
+
+ check = MyOperation->setValue( "ASID", ASID);
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ ndbout << "Set Value" << endl;
+
+
+ check = MyTransaction->execute( Commit );
+ if(check == -1 ) {
+ ndbout << "error at commit" << endl;
+ error_handler(MyTransaction->getNdbErrorString());
+ }
+ else
+ ;//ndbout << ".";
+
+ pMyNdb->closeTransaction(MyTransaction);
+
+
+
+ ndbout << "OK" << endl;
+
+ return;
+}
+
+static void
+update_rows(Ndb* pMyNdb){
+ /****************************************************************
+ * Update rows in SimpleTable
+ *
+ ***************************************************************/
+
+ int check;
+ int loop_count_ops;
+ int count;
+ int i;
+ NdbConnection *MyTransaction;
+ NdbOperation *MyOperation;
+
+ ndbout << "Updating records..." << flush;
+
+ loop_count_ops = NUMBEROFRECORDS;
+
+ for (count=0 ; count < loop_count_ops ; count++) {
+
+ MyTransaction = pMyNdb->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler( pMyNdb->getNdbErrorString() );
+
+ MyOperation = MyTransaction->getNdbOperation(tableName[0]);
+ if (MyOperation == NULL)
+ error_handler(MyTransaction->getNdbErrorString());
+
+ check = MyOperation->updateTuple();
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+
+ check = MyOperation->equal( attrName[0], (char*)&pkValue[count] );
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+
+ for (i = 1; i < MAXATTR; i++)
+ {
+ check = MyOperation->setValue( attrName[i], (char*)&attrValue[count]);
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ }
+
+ if( MyTransaction->execute( Commit ) == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ else
+ ;//ndbout << ".";
+
+ pMyNdb->closeTransaction(MyTransaction);
+
+ }
+
+ ndbout << "OK" << endl;
+ return;
+
+};
+
+static void
+delete_rows(Ndb* pMyNdb){
+
+ /****************************************************************
+ * Delete rows from SimpleTable
+ *
+ ***************************************************************/
+
+ int check;
+ int loop_count_ops;
+ int count;
+ NdbConnection *MyTransaction;
+ NdbOperation *MyOperation;
+
+ ndbout << "Deleting records..."<< flush;
+
+ loop_count_ops = NUMBEROFRECORDS;
+
+ for (count=0 ; count < loop_count_ops ; count++) {
+
+ MyTransaction = pMyNdb->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler( pMyNdb->getNdbErrorString() );
+
+ MyOperation = MyTransaction->getNdbOperation(tableName[0]);
+ if (MyOperation == NULL)
+ error_handler(MyTransaction->getNdbErrorString());
+
+
+ check = MyOperation->deleteTuple();
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+
+ check = MyOperation->equal( attrName[0], (char*)&pkValue[count] );
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+
+
+ if( MyTransaction->execute( Commit ) == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ else
+ ;// ndbout << ".";
+
+ pMyNdb->closeTransaction(MyTransaction);
+
+ }
+
+ ndbout << "OK" << endl;
+ return;
+
+};
+
+static void
+verify_deleted(Ndb* pMyNdb){
+ int check;
+ int loop_count_ops;
+ int count;
+ NdbConnection *MyTransaction;
+ NdbOperation *MyOperation;
+
+ ndbout << "Verifying deleted records..."<< flush;
+
+ loop_count_ops = NUMBEROFRECORDS;
+
+ for (count=0 ; count < loop_count_ops ; count++)
+ {
+ MyTransaction = pMyNdb->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler(pMyNdb->getNdbErrorString());
+
+ MyOperation = MyTransaction->getNdbOperation(tableName[0]);
+ if (MyOperation == NULL)
+ error_handler( MyTransaction->getNdbErrorString());
+
+ check = MyOperation->readTuple();
+ if( check == -1 )
+ error_handler( MyTransaction->getNdbErrorString());
+
+ check = MyOperation->equal( attrName[0],(char*)&pkValue[count] );
+ if( check == -1 )
+ error_handler( MyTransaction->getNdbErrorString());
+
+ // Exepect to receive an error
+ if( MyTransaction->execute( Commit ) != -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ else
+ {
+ ;//ndbout << ".";
+ }
+
+ pMyNdb->closeTransaction(MyTransaction);
+
+ }
+
+ ndbout << "OK" << endl;
+ return;
+};
+
+static void
+read_and_verify_rows(Ndb* pMyNdb)
+{
+
+ int check;
+ int loop_count_ops;
+ int count;
+ int count_attributes;
+
+ NdbConnection *MyTransaction;
+ NdbOperation *MyOperation;
+ NdbRecAttr* tTmp;
+
+ int readValue[MAXATTR];
+
+ ndbout << "Verifying records..."<< flush;
+
+ loop_count_ops = NUMBEROFRECORDS;
+
+ for (count=0 ; count < loop_count_ops ; count++)
+ {
+ MyTransaction = pMyNdb->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler(pMyNdb->getNdbErrorString());
+
+ MyOperation = MyTransaction->getNdbOperation(tableName[0]);
+ if (MyOperation == NULL)
+ error_handler( MyTransaction->getNdbErrorString());
+
+ check = MyOperation->readTuple();
+ if( check == -1 )
+ error_handler( MyTransaction->getNdbErrorString());
+
+ check = MyOperation->equal( attrName[0],(char*)&pkValue[count] );
+ if( check == -1 )
+ error_handler( MyTransaction->getNdbErrorString());
+
+ for (count_attributes = 1; count_attributes < MAXATTR; count_attributes++)
+ {
+ tTmp = MyOperation->getValue( (char*)attrName[count_attributes], (char*)&readValue[count_attributes] );
+ if( tTmp == NULL )
+ error_handler( MyTransaction->getNdbErrorString());
+ }
+
+ if( MyTransaction->execute( Commit ) == -1 )
+ error_handler(MyTransaction->getNdbErrorString());
+ else
+ {
+ // Check value in db against value in mem
+
+ //ndbout << readValue[1] << " == " << attrValue[count] << endl;
+
+ if ( readValue[1]!=attrValue[count] )
+ error_handler("Verification error!");
+ else
+ if ( readValue[2]!=attrValue[count] )
+ error_handler("Verification error!");
+ else
+ if ( readValue[3]!=attrValue[count] )
+ error_handler("Verification error!");
+ else
+ {
+ ;//ndbout << ".";
+ }
+ }
+ pMyNdb->closeTransaction(MyTransaction);
+
+ }
+
+ ndbout << "OK" << endl;
+ return;
+
+
+
+};
+
+
+static void
+setAttrNames()
+{
+ int i;
+
+ for (i = 0; i < MAXATTR ; i++)
+ {
+ sprintf(&attrName[i][0], "Col%d", i);
+ }
+}
+
+static void
+setTableNames()
+{
+ int i;
+
+ sprintf(&tableName[0][0], "SBMCALL", 0);
+ sprintf(&tableName[1][0], "RPACCT", 0);
+ sprintf(&tableName[2][0], "IPACCT", 0);
+ sprintf(&tableName[3][0], "TODACCT", 0);
+
+}
+
+
+bool error_handler2(const char* error_string, int error_int) {
+ failed++ ;
+ ndbout << error_string << endl ;
+ if ( 4008==error_int || 721==error_int || 266==error_int ){
+ ndbout << endl << "Attempting to recover and continue now..." << endl ;
+ return true ; // return true to retry
+ }
+ return false ; // return false to abort
+}
diff --git a/ndb/test/ndbapi/create_all_tabs/Makefile b/ndb/test/ndbapi/create_all_tabs/Makefile
new file mode 100644
index 00000000000..58309807682
--- /dev/null
+++ b/ndb/test/ndbapi/create_all_tabs/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := create_all_tabs
+
+# Source files of non-templated classes (.C files)
+SOURCES = create_all_tabs.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/create_all_tabs/create_all_tabs.cpp b/ndb/test/ndbapi/create_all_tabs/create_all_tabs.cpp
new file mode 100644
index 00000000000..eaa99e8a79d
--- /dev/null
+++ b/ndb/test/ndbapi/create_all_tabs/create_all_tabs.cpp
@@ -0,0 +1,63 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+
+#include <getarg.h>
+
+
+
+int main(int argc, const char** argv){
+
+ int _temp = false;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "temp", 't', arg_flag, &_temp, "Temporary table", "temp" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "This program will create all standard tables in Ndb.\n"\
+ "The tables is selected from a fixed list of tables\n"\
+ "defined in NDBT_Tables class\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ return NDBT_Tables::createAllTables(&MyNdb, _temp);
+
+ }
+
+
diff --git a/ndb/test/ndbapi/create_tab/Makefile b/ndb/test/ndbapi/create_tab/Makefile
new file mode 100644
index 00000000000..c2ea0b52b15
--- /dev/null
+++ b/ndb/test/ndbapi/create_tab/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := create_tab
+
+# Source files of non-templated classes (.C files)
+SOURCES = create_tab.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/create_tab/create_tab.cpp b/ndb/test/ndbapi/create_tab/create_tab.cpp
new file mode 100644
index 00000000000..2f2911b4ef4
--- /dev/null
+++ b/ndb/test/ndbapi/create_tab/create_tab.cpp
@@ -0,0 +1,107 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+
+#include <getarg.h>
+
+
+
+int main(int argc, const char** argv){
+
+ int _temp = false;
+ int _help = 0;
+ int _all = 0;
+ int _print = 0;
+ const char* _connectstr = NULL;
+
+ struct getargs args[] = {
+ { "all", 'a', arg_flag, &_all, "Create/print all tables" },
+ { "print", 'p', arg_flag, &_print, "Print table(s) instead of creating it"},
+ { "temp", 't', arg_flag, &_temp, "Temporary table", "temp" },
+ { "connstr", 'c', arg_string, &_connectstr, "connect string",
+ "How to connect to NDB"},
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will create one table in Ndb.\n"\
+ "The tables may be selected from a fixed list of tables\n"\
+ "defined in NDBT_Tables class\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help){
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ if(argv[optind] == NULL && !_all){
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ int res = 0;
+ if(_print){
+ /**
+ * Print instead of creating
+ */
+ if(argv[optind] != NULL){
+ for(int i = optind; i<argc; i++)
+ NDBT_Tables::print(argv[i]);
+ } else {
+ NDBT_Tables::printAll();
+ }
+ } else {
+ /**
+ * Creating
+ */
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+ MyNdb.setConnectString(_connectstr);
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ if(_all){
+ res = NDBT_Tables::createAllTables(&MyNdb, _temp);
+ } else {
+ int tmp;
+ for(int i = optind; i<argc; i++){
+ ndbout << "Trying to create " << argv[i] << endl;
+ if((tmp = NDBT_Tables::createTable(&MyNdb, argv[i], _temp)) != 0)
+ res = tmp;
+ }
+ }
+ }
+
+ if(res != 0)
+ return NDBT_ProgramExit(NDBT_FAILED);
+ else
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
+
diff --git a/ndb/test/ndbapi/drop_all_tabs/Makefile b/ndb/test/ndbapi/drop_all_tabs/Makefile
new file mode 100644
index 00000000000..96db0781417
--- /dev/null
+++ b/ndb/test/ndbapi/drop_all_tabs/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := drop_all_tabs
+
+# Source files of non-templated classes (.C files)
+SOURCES = drop_all_tabs.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/drop_all_tabs/drop_all_tabs.cpp b/ndb/test/ndbapi/drop_all_tabs/drop_all_tabs.cpp
new file mode 100644
index 00000000000..8e67493c003
--- /dev/null
+++ b/ndb/test/ndbapi/drop_all_tabs/drop_all_tabs.cpp
@@ -0,0 +1,56 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+
+#include <getarg.h>
+
+int main(int argc, const char** argv){
+
+ int _help = 0;
+ struct getargs args[] = {
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "This program will drop all Ndb standard tables from NDB\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ return NDBT_Tables::dropAllTables(&MyNdb);
+
+ }
+
+
diff --git a/ndb/test/ndbapi/flexAsynch/Makefile b/ndb/test/ndbapi/flexAsynch/Makefile
new file mode 100644
index 00000000000..b7de618a10a
--- /dev/null
+++ b/ndb/test/ndbapi/flexAsynch/Makefile
@@ -0,0 +1,47 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := flexAsynch
+
+# Source files of non-templated classes (.C files)
+SOURCES = flexAsynch.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+# run automated tests
+test:
+ @echo "Results test-scripts flexScan." > report.txt
+ @echo "==============================" >> report.txt
+ @echo >> report.txt
+ @echo "User : \t\t\c" >> report.txt
+ @whoami >> report.txt
+ @echo "Date : \t\t\c" >> report.txt
+ @date >> report.txt
+ @echo >> report.txt
+ @echo "Host : \t\t\c" >> report.txt
+ @uname -n >> report.txt
+ @echo "\nStarting test 1."
+ @echo "\n==============================\n" >> report.txt
+ @test1.sh
+ @echo "\nStarting test 2."
+ @echo "\n==============================\n" >> report.txt
+ @test2.sh
+ @echo "\nStarting test 3."
+ @echo "\n==============================\n" >> report.txt
+ @test3.sh
+
+# run basic tests
+run_test :
+ flexBench
+ flexBench -t 32 -o 5000
+ flexBench -simple
+ flexBench -simple -t 32
+ flexBench -dirty
+ flexBench -dirty -t 32
+ flexBench -write
+ flexBench -write -t 32
+ flexBench -dirty -write
+ flexBench -dirty -write -t 32
+ flexBench -c 16 -t 32 -o 5000
+ flexBench -t 16 -c 30 -o 10000
diff --git a/ndb/test/ndbapi/flexAsynch/flexAsynch.cpp b/ndb/test/ndbapi/flexAsynch/flexAsynch.cpp
new file mode 100644
index 00000000000..3938cd21f78
--- /dev/null
+++ b/ndb/test/ndbapi/flexAsynch/flexAsynch.cpp
@@ -0,0 +1,984 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbApi.hpp"
+#include <NdbMain.h>
+#include <md5_hash.hpp>
+
+#include <NdbThread.h>
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#include <NdbOut.hpp>
+#include <NdbTimer.hpp>
+#include <string.h>
+#include <NdbStdio.h>
+#include <NDBT_Error.hpp>
+
+#include <NdbTest.hpp>
+
+#define MAX_PARTS 4
+#define MAX_SEEK 16
+#define MAXSTRLEN 16
+#define MAXATTR 64
+#define MAXTABLES 64
+#define MAXTHREADS 128
+#define MAXPAR 1024
+#define MAXATTRSIZE 1000
+#define PKSIZE 2
+
+enum StartType {
+ stIdle,
+ stInsert,
+ stRead,
+ stUpdate,
+ stDelete,
+ stStop
+} ;
+
+extern "C" { static void* threadLoop(void*); }
+static void setAttrNames(void);
+static void setTableNames(void);
+static int readArguments(int argc, const char** argv);
+static int createTables(Ndb*);
+static void defineOperation(NdbConnection* aTransObject, StartType aType,
+ Uint32 base, Uint32 aIndex);
+static void execute(StartType aType);
+static bool executeThread(StartType aType, Ndb* aNdbObject, unsigned int);
+static void executeCallback(int result, NdbConnection* NdbObject,
+ void* aObject);
+static bool error_handler(const NdbError & err);
+static Uint32 getKey(Uint32, Uint32) ;
+static void input_error();
+
+
+static int retry_opt = 3 ;
+static int failed = 0 ;
+
+ErrorData * flexAsynchErrorData;
+
+struct ThreadNdb
+{
+ int NoOfOps;
+ int ThreadNo;
+};
+
+static NdbThread* threadLife[MAXTHREADS];
+static int tNodeId;
+static int ThreadReady[MAXTHREADS];
+static StartType ThreadStart[MAXTHREADS];
+static char tableName[MAXTABLES][MAXSTRLEN+1];
+static char attrName[MAXATTR][MAXSTRLEN+1];
+
+// Program Parameters
+static bool tLocal = false;
+static int tLocalPart = 0;
+static int tSendForce = 0;
+static int tNoOfLoops = 1;
+static int tAttributeSize = 1;
+static unsigned int tNoOfThreads = 1;
+static unsigned int tNoOfParallelTrans = 32;
+static unsigned int tNoOfAttributes = 25;
+static unsigned int tNoOfTransactions = 500;
+static unsigned int tNoOfOpsPerTrans = 1;
+static unsigned int tLoadFactor = 80;
+static bool tempTable = false;
+static bool startTransGuess = true;
+
+//Program Flags
+static int theTestFlag = 0;
+static int theSimpleFlag = 0;
+static int theDirtyFlag = 0;
+static int theWriteFlag = 0;
+static int theStdTableNameFlag = 0;
+static int theTableCreateFlag = 0;
+
+#define START_REAL_TIME
+#define STOP_REAL_TIME
+#define START_TIMER { NdbTimer timer; timer.doStart();
+#define STOP_TIMER timer.doStop();
+#define PRINT_TIMER(text, trans, opertrans) timer.printTransactionStatistics(text, trans, opertrans); };
+
+static void
+resetThreads(){
+
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ ThreadReady[i] = 0;
+ ThreadStart[i] = stIdle;
+ }//for
+}
+
+static void
+waitForThreads(void)
+{
+ int cont = 0;
+ do {
+ cont = 0;
+ NdbSleep_MilliSleep(20);
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ if (ThreadReady[i] == 0) {
+ cont = 1;
+ }//if
+ }//for
+ } while (cont == 1);
+}
+
+static void
+tellThreads(StartType what)
+{
+ for (int i = 0; i < tNoOfThreads ; i++)
+ ThreadStart[i] = what;
+}
+
+NDB_COMMAND(flexAsynch, "flexAsynch", "flexAsynch", "flexAsynch", 65535)
+{
+ ThreadNdb* pThreadData;
+ int tLoops=0;
+ int returnValue = NDBT_OK;
+
+ flexAsynchErrorData = new ErrorData;
+ flexAsynchErrorData->resetErrorCounters();
+
+ if (readArguments(argc, argv) != 0){
+ input_error();
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ pThreadData = new ThreadNdb[MAXTHREADS];
+
+ ndbout << endl << "FLEXASYNCH - Starting normal mode" << endl;
+ ndbout << "Perform benchmark of insert, update and delete transactions";
+ ndbout << endl;
+ ndbout << " " << tNoOfThreads << " number of concurrent threads " << endl;
+ ndbout << " " << tNoOfParallelTrans;
+ ndbout << " number of parallel operation per thread " << endl;
+ ndbout << " " << tNoOfTransactions << " transaction(s) per round " << endl;
+ ndbout << " " << tNoOfLoops << " iterations " << endl;
+ ndbout << " " << "Load Factor is " << tLoadFactor << "%" << endl;
+ ndbout << " " << tNoOfAttributes << " attributes per table " << endl;
+ ndbout << " " << tAttributeSize;
+ ndbout << " is the number of 32 bit words per attribute " << endl;
+ if (tempTable == true) {
+ ndbout << " Tables are without logging " << endl;
+ } else {
+ ndbout << " Tables are with logging " << endl;
+ }//if
+ if (startTransGuess == true) {
+ ndbout << " Transactions are executed with hint provided" << endl;
+ } else {
+ ndbout << " Transactions are executed with round robin scheme" << endl;
+ }//if
+ if (tSendForce == 0) {
+ ndbout << " No force send is used, adaptive algorithm used" << endl;
+ } else if (tSendForce == 1) {
+ ndbout << " Force send used" << endl;
+ } else {
+ ndbout << " No force send is used, adaptive algorithm disabled" << endl;
+ }//if
+
+ ndbout << endl;
+
+ NdbThread_SetConcurrencyLevel(2 + tNoOfThreads);
+
+ /* print Setting */
+ flexAsynchErrorData->printSettings(ndbout);
+
+ setAttrNames();
+ setTableNames();
+
+ Ndb * pNdb = new Ndb("TEST_DB");
+ pNdb->init();
+ tNodeId = pNdb->getNodeId();
+
+ ndbout << " NdbAPI node with id = " << pNdb->getNodeId() << endl;
+ ndbout << endl;
+
+ ndbout << "Waiting for ndb to become ready..." <<endl;
+ if (pNdb->waitUntilReady(10000) != 0){
+ ndbout << "NDB is not ready" << endl;
+ ndbout << "Benchmark failed!" << endl;
+ returnValue = NDBT_FAILED;
+ }
+
+ if(returnValue == NDBT_OK){
+ if (createTables(pNdb) != 0){
+ returnValue = NDBT_FAILED;
+ }
+ }
+
+ if(returnValue == NDBT_OK){
+ /****************************************************************
+ * Create NDB objects. *
+ ****************************************************************/
+ resetThreads();
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ pThreadData[i].ThreadNo = i
+;
+ threadLife[i] = NdbThread_Create(threadLoop,
+ (void**)&pThreadData[i],
+ 32768,
+ "flexAsynchThread",
+ NDB_THREAD_PRIO_LOW);
+ }//for
+ ndbout << endl << "All NDB objects and table created" << endl << endl;
+ int noOfTransacts = tNoOfParallelTrans*tNoOfTransactions*tNoOfThreads;
+ /****************************************************************
+ * Execute program. *
+ ****************************************************************/
+
+ for(;;) {
+
+ int loopCount = tLoops + 1 ;
+ ndbout << endl << "Loop # " << loopCount << endl << endl ;
+
+ /****************************************************************
+ * Perform inserts. *
+ ****************************************************************/
+
+ failed = 0 ;
+
+ START_TIMER;
+ execute(stInsert);
+ STOP_TIMER;
+ PRINT_TIMER("insert", noOfTransacts, tNoOfOpsPerTrans);
+
+ if (0 < failed) {
+ int i = retry_opt ;
+ int ci = 1 ;
+ while (0 < failed && 0 < i){
+ ndbout << failed << " of the transactions returned errors!"
+ << endl << endl;
+ ndbout << "Attempting to redo the failed transactions now..."
+ << endl ;
+ ndbout << "Redo attempt " << ci <<" out of " << retry_opt
+ << endl << endl;
+ failed = 0 ;
+ START_TIMER;
+ execute(stInsert);
+ STOP_TIMER;
+ PRINT_TIMER("insert", noOfTransacts, tNoOfOpsPerTrans);
+ i-- ;
+ ci++;
+ }
+ if(0 == failed ){
+ ndbout << endl <<"Redo attempt succeeded" << endl << endl;
+ }else{
+ ndbout << endl <<"Redo attempt failed, moving on now..." << endl
+ << endl;
+ }//if
+ }//if
+
+ /****************************************************************
+ * Perform read. *
+ ****************************************************************/
+
+ failed = 0 ;
+
+ START_TIMER;
+ execute(stRead);
+ STOP_TIMER;
+ PRINT_TIMER("read", noOfTransacts, tNoOfOpsPerTrans);
+
+ if (0 < failed) {
+ int i = retry_opt ;
+ int cr = 1;
+ while (0 < failed && 0 < i){
+ ndbout << failed << " of the transactions returned errors!"<<endl ;
+ ndbout << endl;
+ ndbout <<"Attempting to redo the failed transactions now..." << endl;
+ ndbout << endl;
+ ndbout <<"Redo attempt " << cr <<" out of ";
+ ndbout << retry_opt << endl << endl;
+ failed = 0 ;
+ START_TIMER;
+ execute(stRead);
+ STOP_TIMER;
+ PRINT_TIMER("read", noOfTransacts, tNoOfOpsPerTrans);
+ i-- ;
+ cr++ ;
+ }//while
+ if(0 == failed ) {
+ ndbout << endl <<"Redo attempt succeeded" << endl << endl ;
+ }else{
+ ndbout << endl <<"Redo attempt failed, moving on now..." << endl << endl ;
+ }//if
+ }//if
+
+
+ /****************************************************************
+ * Perform update. *
+ ****************************************************************/
+
+ failed = 0 ;
+
+ START_TIMER;
+ execute(stUpdate);
+ STOP_TIMER;
+ PRINT_TIMER("update", noOfTransacts, tNoOfOpsPerTrans) ;
+
+ if (0 < failed) {
+ int i = retry_opt ;
+ int cu = 1 ;
+ while (0 < failed && 0 < i){
+ ndbout << failed << " of the transactions returned errors!"<<endl ;
+ ndbout << endl;
+ ndbout <<"Attempting to redo the failed transactions now..." << endl;
+ ndbout << endl <<"Redo attempt " << cu <<" out of ";
+ ndbout << retry_opt << endl << endl;
+ failed = 0 ;
+ START_TIMER;
+ execute(stUpdate);
+ STOP_TIMER;
+ PRINT_TIMER("update", noOfTransacts, tNoOfOpsPerTrans);
+ i-- ;
+ cu++ ;
+ }//while
+ if(0 == failed ){
+ ndbout << endl <<"Redo attempt succeeded" << endl << endl;
+ } else {
+ ndbout << endl;
+ ndbout <<"Redo attempt failed, moving on now..." << endl << endl;
+ }//if
+ }//if
+
+ /****************************************************************
+ * Perform read. *
+ ****************************************************************/
+
+ failed = 0 ;
+
+ START_TIMER;
+ execute(stRead);
+ STOP_TIMER;
+ PRINT_TIMER("read", noOfTransacts, tNoOfOpsPerTrans);
+
+ if (0 < failed) {
+ int i = retry_opt ;
+ int cr2 = 1 ;
+ while (0 < failed && 0 < i){
+ ndbout << failed << " of the transactions returned errors!"<<endl ;
+ ndbout << endl;
+ ndbout <<"Attempting to redo the failed transactions now..." << endl;
+ ndbout << endl <<"Redo attempt " << cr2 <<" out of ";
+ ndbout << retry_opt << endl << endl;
+ failed = 0 ;
+ START_TIMER;
+ execute(stRead);
+ STOP_TIMER;
+ PRINT_TIMER("read", noOfTransacts, tNoOfOpsPerTrans);
+ i-- ;
+ cr2++ ;
+ }//while
+ if(0 == failed ){
+ ndbout << endl <<"Redo attempt succeeded" << endl << endl;
+ }else{
+ ndbout << endl;
+ ndbout << "Redo attempt failed, moving on now..." << endl << endl;
+ }//if
+ }//if
+
+
+ /****************************************************************
+ * Perform delete. *
+ ****************************************************************/
+
+ failed = 0 ;
+
+ START_TIMER;
+ execute(stDelete);
+ STOP_TIMER;
+ PRINT_TIMER("delete", noOfTransacts, tNoOfOpsPerTrans);
+
+ if (0 < failed) {
+ int i = retry_opt ;
+ int cd = 1 ;
+ while (0 < failed && 0 < i){
+ ndbout << failed << " of the transactions returned errors!"<< endl ;
+ ndbout << endl;
+ ndbout <<"Attempting to redo the failed transactions now:" << endl ;
+ ndbout << endl <<"Redo attempt " << cd <<" out of ";
+ ndbout << retry_opt << endl << endl;
+ failed = 0 ;
+ START_TIMER;
+ execute(stDelete);
+ STOP_TIMER;
+ PRINT_TIMER("read", noOfTransacts, tNoOfOpsPerTrans);
+ i-- ;
+ cd++ ;
+ }//while
+ if(0 == failed ){
+ ndbout << endl <<"Redo attempt succeeded" << endl << endl ;
+ }else{
+ ndbout << endl;
+ ndbout << "Redo attempt failed, moving on now..." << endl << endl;
+ }//if
+ }//if
+
+ tLoops++;
+ ndbout << "--------------------------------------------------" << endl;
+
+ if(tNoOfLoops != 0){
+ if(tNoOfLoops <= tLoops)
+ break ;
+ }
+ }//for
+
+ execute(stStop);
+ void * tmp;
+ for(int i = 0; i<tNoOfThreads; i++){
+ NdbThread_WaitFor(threadLife[i], &tmp);
+ NdbThread_Destroy(&threadLife[i]);
+ }
+ }
+ delete [] pThreadData;
+ delete pNdb;
+
+ //printing errorCounters
+ flexAsynchErrorData->printErrorCounters(ndbout);
+
+ return NDBT_ProgramExit(returnValue);
+}//main()
+
+
+static void execute(StartType aType)
+{
+ resetThreads();
+ tellThreads(aType);
+ waitForThreads();
+}//execute()
+
+static void*
+threadLoop(void* ThreadData)
+{
+ Ndb* localNdb;
+ StartType tType;
+ ThreadNdb* tabThread = (ThreadNdb*)ThreadData;
+ int threadNo = tabThread->ThreadNo;
+ localNdb = new Ndb("TEST_DB");
+ localNdb->init(1024);
+ localNdb->waitUntilReady(10000);
+ unsigned int threadBase = (threadNo << 16) + tNodeId ;
+
+ for (;;){
+ while (ThreadStart[threadNo] == stIdle) {
+ NdbSleep_MilliSleep(10);
+ }//while
+
+ // Check if signal to exit is received
+ if (ThreadStart[threadNo] == stStop) {
+ break;
+ }//if
+
+ tType = ThreadStart[threadNo];
+ ThreadStart[threadNo] = stIdle;
+ if(!executeThread(tType, localNdb, threadBase)){
+ break;
+ }
+ ThreadReady[threadNo] = 1;
+ }//for
+
+ delete localNdb;
+ ThreadReady[threadNo] = 1;
+
+ NdbThread_Exit(0);
+ return NULL; // Just to keep compiler happy
+}//threadLoop()
+
+static
+bool
+executeThread(StartType aType, Ndb* aNdbObject, unsigned int threadBase) {
+ int i, j, k;
+ NdbConnection* tConArray[1024];
+ unsigned int tBase;
+ unsigned int tBase2;
+
+ for (i = 0; i < tNoOfTransactions; i++) {
+ if (tLocal == false) {
+ tBase = i * tNoOfParallelTrans * tNoOfOpsPerTrans;
+ } else {
+ tBase = i * tNoOfParallelTrans * MAX_SEEK;
+ }//if
+ START_REAL_TIME;
+ for (j = 0; j < tNoOfParallelTrans; j++) {
+ if (tLocal == false) {
+ tBase2 = tBase + (j * tNoOfOpsPerTrans);
+ } else {
+ tBase2 = tBase + (j * MAX_SEEK);
+ tBase2 = getKey(threadBase, tBase2);
+ }//if
+ if (startTransGuess == true) {
+ Uint64 Tkey64;
+ Uint32* Tkey32 = (Uint32*)&Tkey64;
+ Tkey32[0] = threadBase;
+ Tkey32[1] = tBase2;
+ tConArray[j] = aNdbObject->startTransaction((Uint32)0, //Priority
+ (const char*)&Tkey64, //Main PKey
+ (Uint32)4); //Key Length
+ } else {
+ tConArray[j] = aNdbObject->startTransaction();
+ }//if
+ if (tConArray[j] == NULL &&
+ !error_handler(aNdbObject->getNdbError()) ){
+ ndbout << endl << "Unable to recover! Quiting now" << endl ;
+ return false;
+ }//if
+
+ for (k = 0; k < tNoOfOpsPerTrans; k++) {
+ //-------------------------------------------------------
+ // Define the operation, but do not execute it yet.
+ //-------------------------------------------------------
+ defineOperation(tConArray[j], aType, threadBase, (tBase2 + k));
+ }//for
+
+ tConArray[j]->executeAsynchPrepare(Commit, &executeCallback, NULL);
+ }//for
+ STOP_REAL_TIME;
+ //-------------------------------------------------------
+ // Now we have defined a set of operations, it is now time
+ // to execute all of them.
+ //-------------------------------------------------------
+ int Tcomp = aNdbObject->sendPollNdb(3000, 0, 0);
+ while (Tcomp < tNoOfParallelTrans) {
+ int TlocalComp = aNdbObject->pollNdb(3000, 0);
+ Tcomp += TlocalComp;
+ }//while
+ for (j = 0 ; j < tNoOfParallelTrans ; j++) {
+ aNdbObject->closeTransaction(tConArray[j]);
+ }//for
+ }//for
+ return true;
+}//executeThread()
+
+static
+Uint32
+getKey(Uint32 aBase, Uint32 anIndex) {
+ Uint32 Tfound = anIndex;
+ Uint64 Tkey64;
+ Uint32* Tkey32 = (Uint32*)&Tkey64;
+ Tkey32[0] = aBase;
+ Uint32 hash;
+ for (Uint32 i = anIndex; i < (anIndex + MAX_SEEK); i++) {
+ Tkey32[1] = (Uint32)i;
+ hash = md5_hash((Uint64*)&Tkey64, (Uint32)2);
+ hash = (hash >> 6) & (MAX_PARTS - 1);
+ if (hash == tLocalPart) {
+ Tfound = i;
+ break;
+ }//if
+ }//for
+ return Tfound;
+}//getKey()
+
+static void
+executeCallback(int result, NdbConnection* NdbObject, void* aObject)
+{
+ if (result == -1) {
+
+ // Add complete error handling here
+
+ int retCode = flexAsynchErrorData->handleErrorCommon(NdbObject->getNdbError());
+ if (retCode == 1) {
+ if (NdbObject->getNdbError().code != 626 && NdbObject->getNdbError().code != 630){
+ ndbout_c("execute: %s", NdbObject->getNdbError().message);
+ ndbout_c("Error code = %d", NdbObject->getNdbError().code);}
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexAsynch" << endl;
+ } else if (retCode == 3) {
+ /* What can we do here? */
+ ndbout_c("execute: %s", NdbObject->getNdbError().message);
+ }//if(retCode == 3)
+
+ // ndbout << "Error occured in poll:" << endl;
+ // ndbout << NdbObject->getNdbError() << endl;
+ failed++ ;
+ return;
+ }//if
+ return;
+}//executeCallback()
+
+
+
+static void
+defineOperation(NdbConnection* localNdbConnection, StartType aType,
+ Uint32 threadBase, Uint32 aIndex)
+{
+ NdbOperation* localNdbOperation;
+ unsigned int loopCountAttributes = tNoOfAttributes;
+ unsigned int countAttributes;
+ Uint32 attrValue[MAXATTRSIZE];
+
+ //-------------------------------------------------------
+ // Set-up the attribute values for this operation.
+ //-------------------------------------------------------
+ attrValue[0] = threadBase;
+ attrValue[1] = aIndex;
+ for (int k = 2; k < loopCountAttributes; k++) {
+ attrValue[k] = aIndex;
+ }//for
+ localNdbOperation = localNdbConnection->getNdbOperation(tableName[0]);
+ if (localNdbOperation == NULL) {
+ error_handler(localNdbConnection->getNdbError());
+ }//if
+ switch (aType) {
+ case stInsert: { // Insert case
+ if (theWriteFlag == 1 && theDirtyFlag == 1) {
+ localNdbOperation->dirtyWrite();
+ } else if (theWriteFlag == 1) {
+ localNdbOperation->writeTuple();
+ } else {
+ localNdbOperation->insertTuple();
+ }//if
+ break;
+ }//case
+ case stRead: { // Read Case
+ if (theSimpleFlag == 1) {
+ localNdbOperation->simpleRead();
+ } else if (theDirtyFlag == 1) {
+ localNdbOperation->dirtyRead();
+ } else {
+ localNdbOperation->readTuple();
+ }//if
+ break;
+ }//case
+ case stUpdate: { // Update Case
+ if (theWriteFlag == 1 && theDirtyFlag == 1) {
+ localNdbOperation->dirtyWrite();
+ } else if (theWriteFlag == 1) {
+ localNdbOperation->writeTuple();
+ } else if (theDirtyFlag == 1) {
+ localNdbOperation->dirtyUpdate();
+ } else {
+ localNdbOperation->updateTuple();
+ }//if
+ break;
+ }//case
+ case stDelete: { // Delete Case
+ localNdbOperation->deleteTuple();
+ break;
+ }//case
+ default: {
+ error_handler(localNdbOperation->getNdbError());
+ }//default
+ }//switch
+ localNdbOperation->equal((Uint32)0,(char*)&attrValue[0]);
+ switch (aType) {
+ case stInsert: // Insert case
+ case stUpdate: // Update Case
+ {
+ for (countAttributes = 1;
+ countAttributes < loopCountAttributes; countAttributes++) {
+ localNdbOperation->setValue(countAttributes,
+ (char*)&attrValue[0]);
+ }//for
+ break;
+ }//case
+ case stRead: { // Read Case
+ for (countAttributes = 1;
+ countAttributes < loopCountAttributes; countAttributes++) {
+ localNdbOperation->getValue(countAttributes,
+ (char*)&attrValue[0]);
+ }//for
+ break;
+ }//case
+ case stDelete: { // Delete Case
+ break;
+ }//case
+ default: {
+ //goto error_handler; < epaulsa
+ error_handler(localNdbOperation->getNdbError());
+ }//default
+ }//switch
+ return;
+}//defineOperation()
+
+static void setAttrNames()
+{
+ int i;
+
+ for (i = 0; i < MAXATTR ; i++){
+ snprintf(attrName[i], MAXSTRLEN, "COL%d", i);
+ }
+}
+
+
+static void setTableNames()
+{
+ // Note! Uses only uppercase letters in table name's
+ // so that we can look at the tables wits SQL
+ int i;
+ for (i = 0; i < MAXTABLES ; i++){
+ if (theStdTableNameFlag==0){
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d_%d", i,
+ (int)(NdbTick_CurrentMillisecond()/1000));
+ } else {
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d", i);
+ }
+ }
+}
+
+static
+int
+createTables(Ndb* pMyNdb){
+
+ NdbSchemaCon *MySchemaTransaction;
+ NdbSchemaOp *MySchemaOp;
+ int check;
+
+ if (theTableCreateFlag == 0) {
+ for(int i=0; i < 1 ;i++) {
+ ndbout << "Creating " << tableName[i] << "..." << endl;
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+
+ if(MySchemaTransaction == NULL &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if(MySchemaOp == NULL &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ check = MySchemaOp->createTable( tableName[i]
+ ,8 // Table Size
+ ,TupleKey // Key Type
+ ,40 // Nr of Pages
+ ,All
+ ,6
+ ,(tLoadFactor - 5)
+ ,(tLoadFactor)
+ ,1
+ ,!tempTable
+ );
+
+ if (check == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ check = MySchemaOp->createAttribute( (char*)attrName[0],
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if (check == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+ for (int j = 1; j < tNoOfAttributes ; j++){
+ check = MySchemaOp->createAttribute( (char*)attrName[j],
+ NoKey,
+ 32,
+ tAttributeSize,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if (check == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+ }
+
+ if (MySchemaTransaction->execute() == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ }
+ }
+
+ return 0;
+}
+
+static
+bool error_handler(const NdbError & err){
+ ndbout << err << endl ;
+ switch(err.classification){
+ case NdbError::TemporaryResourceError:
+ case NdbError::OverloadError:
+ case NdbError::SchemaError:
+ ndbout << endl << "Attempting to recover and continue now..." << endl ;
+ return true;
+ }
+ return false ; // return false to abort
+}
+static
+bool error_handler(const char* error_string, int error_int) {
+ ndbout << error_string << endl ;
+ if ((4008 == error_int) ||
+ (721 == error_int) ||
+ (266 == error_int)){
+ ndbout << endl << "Attempting to recover and continue now..." << endl ;
+ return true ; // return true to retry
+ }
+ return false ; // return false to abort
+}
+
+static
+int
+readArguments(int argc, const char** argv){
+
+ int i = 1;
+ while (argc > 1){
+ if (strcmp(argv[i], "-t") == 0){
+ tNoOfThreads = atoi(argv[i+1]);
+ if ((tNoOfThreads < 1) || (tNoOfThreads > MAXTHREADS)){
+ ndbout_c("Invalid no of threads");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-p") == 0){
+ tNoOfParallelTrans = atoi(argv[i+1]);
+ if ((tNoOfParallelTrans < 1) || (tNoOfParallelTrans > MAXPAR)){
+ ndbout_c("Invalid no of parallell transactions");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-load_factor") == 0){
+ tLoadFactor = atoi(argv[i+1]);
+ if ((tLoadFactor < 40) || (tLoadFactor > 99)){
+ ndbout_c("Invalid load factor");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-c") == 0) {
+ tNoOfOpsPerTrans = atoi(argv[i+1]);
+ if (tNoOfOpsPerTrans < 1){
+ ndbout_c("Invalid no of operations per transaction");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-o") == 0) {
+ tNoOfTransactions = atoi(argv[i+1]);
+ if (tNoOfTransactions < 1){
+ ndbout_c("Invalid no of transactions");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-a") == 0){
+ tNoOfAttributes = atoi(argv[i+1]);
+ if ((tNoOfAttributes < 2) || (tNoOfAttributes > MAXATTR)){
+ ndbout_c("Invalid no of attributes");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-n") == 0){
+ theStdTableNameFlag = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-l") == 0){
+ tNoOfLoops = atoi(argv[i+1]);
+ if ((tNoOfLoops < 0) || (tNoOfLoops > 100000)){
+ ndbout_c("Invalid no of loops");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-s") == 0){
+ tAttributeSize = atoi(argv[i+1]);
+ if ((tAttributeSize < 1) || (tAttributeSize > MAXATTRSIZE)){
+ ndbout_c("Invalid attributes size");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-local") == 0){
+ tLocalPart = atoi(argv[i+1]);
+ tLocal = true;
+ startTransGuess = true;
+ if ((tLocalPart < 0) || (tLocalPart > MAX_PARTS)){
+ ndbout_c("Invalid local part");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-simple") == 0){
+ theSimpleFlag = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-adaptive") == 0){
+ tSendForce = 0;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-force") == 0){
+ tSendForce = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-non_adaptive") == 0){
+ tSendForce = 2;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-write") == 0){
+ theWriteFlag = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-dirty") == 0){
+ theDirtyFlag = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-test") == 0){
+ theTestFlag = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-no_table_create") == 0){
+ theTableCreateFlag = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-temp") == 0){
+ tempTable = true;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-no_hint") == 0){
+ startTransGuess = false;
+ argc++;
+ i--;
+ } else {
+ return -1;
+ }
+
+ argc -= 2;
+ i = i + 2;
+ }//while
+ if (tLocal == true) {
+ if (tNoOfOpsPerTrans != 1) {
+ ndbout_c("Not valid to have more than one op per trans with local");
+ }//if
+ if (startTransGuess == false) {
+ ndbout_c("Not valid to use no_hint with local");
+ }//if
+ }//if
+ return 0;
+}
+
+static
+void
+input_error(){
+
+ ndbout_c("FLEXASYNCH");
+ ndbout_c(" Perform benchmark of insert, update and delete transactions");
+ ndbout_c("");
+ ndbout_c("Arguments:");
+ ndbout_c(" -t Number of threads to start, default 1");
+ ndbout_c(" -p Number of parallel transactions per thread, default 32");
+ ndbout_c(" -o Number of transactions per loop, default 500");
+ ndbout_c(" -l Number of loops to run, default 1, 0=infinite");
+ ndbout_c(" -load_factor Number Load factor in index in percent (40 -> 99)");
+ ndbout_c(" -a Number of attributes, default 25");
+ ndbout_c(" -c Number of operations per transaction");
+ ndbout_c(" -s Size of each attribute, default 1 ");
+ ndbout_c(" (PK is always of size 1, independent of this value)");
+ ndbout_c(" -simple Use simple read to read from database");
+ ndbout_c(" -dirty Use dirty read to read from database");
+ ndbout_c(" -write Use writeTuple in insert and update");
+ ndbout_c(" -n Use standard table names");
+ ndbout_c(" -no_table_create Don't create tables in db");
+ ndbout_c(" -temp Create table(s) without logging");
+ ndbout_c(" -no_hint Don't give hint on where to execute transaction coordinator");
+ ndbout_c(" -adaptive Use adaptive send algorithm (default)");
+ ndbout_c(" -force Force send when communicating");
+ ndbout_c(" -non_adaptive Send at a 10 millisecond interval");
+ ndbout_c(" -local Number of part, only use keys in one part out of 16");
+}
+
+
+
diff --git a/ndb/test/ndbapi/flexBench/Makefile b/ndb/test/ndbapi/flexBench/Makefile
new file mode 100644
index 00000000000..78d4039258c
--- /dev/null
+++ b/ndb/test/ndbapi/flexBench/Makefile
@@ -0,0 +1,47 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := flexBench
+
+# Source files of non-templated classes (.C files)
+SOURCES = flexBench.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+# run automated tests
+test:
+ @echo "Results test-scripts flexScan." > report.txt
+ @echo "==============================" >> report.txt
+ @echo >> report.txt
+ @echo "User : \t\t\c" >> report.txt
+ @whoami >> report.txt
+ @echo "Date : \t\t\c" >> report.txt
+ @date >> report.txt
+ @echo >> report.txt
+ @echo "Host : \t\t\c" >> report.txt
+ @uname -n >> report.txt
+ @echo "\nStarting test 1."
+ @echo "\n==============================\n" >> report.txt
+ @test1.sh
+ @echo "\nStarting test 2."
+ @echo "\n==============================\n" >> report.txt
+ @test2.sh
+ @echo "\nStarting test 3."
+ @echo "\n==============================\n" >> report.txt
+ @test3.sh
+
+# run basic tests
+run_test :
+ flexBench
+ flexBench -t 32 -o 5000
+ flexBench -simple
+ flexBench -simple -t 32
+ flexBench -dirty
+ flexBench -dirty -t 32
+ flexBench -write
+ flexBench -write -t 32
+ flexBench -dirty -write
+ flexBench -dirty -write -t 32
+ flexBench -c 16 -t 32 -o 5000
+ flexBench -t 16 -c 30 -o 10000
diff --git a/ndb/test/ndbapi/flexBench/flexBench.cpp b/ndb/test/ndbapi/flexBench/flexBench.cpp
new file mode 100644
index 00000000000..17d9be73925
--- /dev/null
+++ b/ndb/test/ndbapi/flexBench/flexBench.cpp
@@ -0,0 +1,1157 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/* ***************************************************
+FLEXBENCH
+Perform benchmark of insert, update and delete transactions
+
+Arguments:
+ -t Number of threads to start, default 1
+ -o Number of operations per loop, default 500
+ -l Number of loops to run, default 1, 0=infinite
+ -a Number of attributes, default 25
+ -c Number of tables, default 1
+ -s Size of each attribute, default 1 (Primary Key is always of size 1,
+ independent of this value)
+ -lkn Number of long primary keys, default 1
+ -lks Size of each long primary key, default 1
+ -simple Use simple read to read from database
+ -dirty Use dirty read to read from database
+ -write Use writeTuple in insert and update
+ -stdtables Use standard table names
+ -no_table_create Don't create tables in db
+ -sleep Sleep a number of seconds before running the test, this
+ can be used so that another flexBench have time to create tables
+ -temp Use tables without logging
+ -verify Verify inserts, updates and deletes
+#ifdef CEBIT_STAT
+ -statserv host:port statistics server to report to
+ -statfreq ops report every ops operations (default 100)
+#endif
+ Returns:
+ 0 - Test passed
+ 1 - Test failed
+ 2 - Invalid arguments
+
+* *************************************************** */
+
+#include "NdbApi.hpp"
+
+#include <string.h>
+#include <assert.h>
+
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbStdio.h>
+#include <NdbTick.h>
+#include <NdbTimer.hpp>
+#include <NdbThread.h>
+
+#include <NdbTest.hpp>
+
+#define MAXSTRLEN 16
+#define MAXATTR 64
+#define MAXTABLES 128
+#define MAXATTRSIZE 1000
+#define MAXNOLONGKEY 16 // Max number of long keys.
+#define MAXLONGKEYTOTALSIZE 1023 // words = 4092 bytes
+
+extern "C" { static void* flexBenchThread(void*); }
+static int readArguments(int argc, const char** argv);
+static int createTables(Ndb*);
+static void sleepBeforeStartingTest(int seconds);
+static void input_error();
+
+enum StartType {
+ stIdle,
+ stInsert,
+ stVerify,
+ stRead,
+ stUpdate,
+ stDelete,
+ stTryDelete,
+ stVerifyDelete,
+ stStop
+};
+
+struct ThreadData
+{
+ int threadNo;
+ NdbThread* threadLife;
+ int threadReady;
+ StartType threadStart;
+ int threadResult;
+};
+
+static int tNodeId = 0 ;
+static char tableName[MAXTABLES][MAXSTRLEN+1];
+static char attrName[MAXATTR][MAXSTRLEN+1];
+static char** longKeyAttrName;
+
+// Program Parameters
+static int tNoOfLoops = 1;
+static int tAttributeSize = 1;
+static unsigned int tNoOfThreads = 1;
+static unsigned int tNoOfTables = 1;
+static unsigned int tNoOfAttributes = 25;
+static unsigned int tNoOfOperations = 500;
+static unsigned int tSleepTime = 0;
+static unsigned int tNoOfLongPK = 1;
+static unsigned int tSizeOfLongPK = 1;
+
+//Program Flags
+static int theSimpleFlag = 0;
+static int theDirtyFlag = 0;
+static int theWriteFlag = 0;
+static int theStdTableNameFlag = 0;
+static int theTableCreateFlag = 0;
+static bool theTempTable = false;
+static bool VerifyFlag = true;
+static bool useLongKeys = false;
+
+
+static ErrorData theErrorData; // Part of flexBench-program
+
+#define START_TIMER { NdbTimer timer; timer.doStart();
+#define STOP_TIMER timer.doStop();
+#define PRINT_TIMER(text, trans, opertrans) timer.printTransactionStatistics(text, trans, opertrans); };
+
+#include <NdbTCP.h>
+
+#ifdef CEBIT_STAT
+#include <NdbMutex.h>
+static bool statEnable = false;
+static char statHost[100];
+static int statFreq = 100;
+static int statPort = 0;
+static int statSock = -1;
+static enum { statError = -1, statClosed, statOpen } statState;
+static NdbMutex statMutex = NDB_MUTEX_INITIALIZER;
+#endif
+
+//-------------------------------------------------------------------
+// Statistical Reporting routines
+//-------------------------------------------------------------------
+#ifdef CEBIT_STAT
+// Experimental client-side statistic for CeBIT
+
+static void
+statReport(enum StartType st, int ops)
+{
+ if (!statEnable)
+ return;
+ if (NdbMutex_Lock(&statMutex) < 0) {
+ if (statState != statError) {
+ ndbout_c("stat: lock mutex failed: %s", strerror(errno));
+ statState = statError;
+ }
+ return;
+ }
+ static int nodeid;
+ // open connection
+ if (statState != statOpen) {
+ char *p = getenv("NDB_NODEID"); // ndbnet sets NDB_NODEID
+ nodeid = p == 0 ? 0 : atoi(p);
+ if ((statSock = socket(PF_INET, SOCK_STREAM, 0)) < 0) {
+ if (statState != statError) {
+ ndbout_c("stat: create socket failed: %s", strerror(errno));
+ statState = statError;
+ }
+ (void)NdbMutex_Unlock(&statMutex);
+ return;
+ }
+ struct sockaddr_in saddr;
+ memset(&saddr, 0, sizeof(saddr));
+ saddr.sin_family = AF_INET;
+ saddr.sin_port = htons(statPort);
+ if (Ndb_getInAddr(&saddr.sin_addr, statHost) < 0) {
+ if (statState != statError) {
+ ndbout_c("stat: host %s not found", statHost);
+ statState = statError;
+ }
+ (void)close(statSock);
+ (void)NdbMutex_Unlock(&statMutex);
+ return;
+ }
+ if (connect(statSock, (struct sockaddr *)&saddr, sizeof(saddr)) < 0) {
+ if (statState != statError) {
+ ndbout_c("stat: connect failed: %s", strerror(errno));
+ statState = statError;
+ }
+ (void)close(statSock);
+ (void)NdbMutex_Unlock(&statMutex);
+ return;
+ }
+ statState = statOpen;
+ ndbout_c("stat: connection to %s:%d opened", statHost, (int)statPort);
+ }
+ const char *text;
+ switch (st) {
+ case stInsert:
+ text = "insert";
+ break;
+ case stVerify:
+ text = "verify";
+ break;
+ case stRead:
+ text = "read";
+ break;
+ case stUpdate:
+ text = "update";
+ break;
+ case stDelete:
+ text = "delete";
+ break;
+ case stVerifyDelete:
+ text = "verifydelete";
+ break;
+ default:
+ text = "unknown";
+ break;
+ }
+ char buf[100];
+ sprintf(buf, "%d %s %d\n", nodeid, text, ops);
+ int len = strlen(buf);
+ // assume SIGPIPE already ignored
+ if (write(statSock, buf, len) != len) {
+ if (statState != statError) {
+ ndbout_c("stat: write failed: %s", strerror(errno));
+ statState = statError;
+ }
+ (void)close(statSock);
+ (void)NdbMutex_Unlock(&statMutex);
+ return;
+ }
+ (void)NdbMutex_Unlock(&statMutex);
+}
+#endif // CEBIT_STAT
+
+static void
+resetThreads(ThreadData* pt){
+ for (unsigned int i = 0; i < tNoOfThreads; i++){
+ pt[i].threadReady = 0;
+ pt[i].threadResult = 0;
+ pt[i].threadStart = stIdle;
+ }
+}
+
+static int
+checkThreadResults(ThreadData* pt){
+ for (unsigned int i = 0; i < tNoOfThreads; i++){
+ if(pt[i].threadResult != 0){
+ ndbout_c("Thread%d reported fatal error %d", i, pt[i].threadResult);
+ return -1;
+ }
+ }
+ return 0;
+}
+
+static
+void
+waitForThreads(ThreadData* pt)
+{
+ int cont = 1;
+ while (cont){
+ NdbSleep_MilliSleep(100);
+ cont = 0;
+ for (unsigned int i = 0; i < tNoOfThreads; i++){
+ if (pt[i].threadReady == 0)
+ cont = 1;
+ }
+ }
+}
+
+static void
+tellThreads(ThreadData* pt, StartType what)
+{
+ for (unsigned int i = 0; i < tNoOfThreads; i++)
+ pt[i].threadStart = what;
+}
+
+NDB_COMMAND(flexBench, "flexBench", "flexBench", "flexbench", 65535)
+{
+ ThreadData* pThreadsData;
+ int tLoops = 0;
+ int returnValue = NDBT_OK;
+
+ if (readArguments(argc, argv) != 0){
+ input_error();
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ if(useLongKeys){
+ longKeyAttrName = (char **) malloc(sizeof(char*) * tNoOfLongPK);
+ for (int i = 0; i < tNoOfLongPK; i++) {
+ longKeyAttrName[i] = (char *) malloc(strlen("KEYATTR ") + 1);
+ memset(longKeyAttrName[i], 0, strlen("KEYATTR ") + 1);
+ sprintf(longKeyAttrName[i], "KEYATTR%i", i);
+ }
+ }
+
+ pThreadsData = new ThreadData[tNoOfThreads];
+
+ ndbout << endl << "FLEXBENCH - Starting normal mode" << endl;
+ ndbout << "Perform benchmark of insert, update and delete transactions"<< endl;
+ ndbout << " " << tNoOfThreads << " thread(s) " << endl;
+ ndbout << " " << tNoOfLoops << " iterations " << endl;
+ ndbout << " " << tNoOfTables << " table(s) and " << 1 << " operation(s) per transaction " <<endl;
+ ndbout << " " << tNoOfAttributes << " attributes per table " << endl;
+ ndbout << " " << tNoOfOperations << " transaction(s) per thread and round " << endl;
+ ndbout << " " << tAttributeSize << " is the number of 32 bit words per attribute "<< endl;
+ ndbout << " " << "Table(s) without logging: " << (Uint32)theTempTable << endl;
+
+ if(useLongKeys)
+ ndbout << " " << "Using long keys with " << tNoOfLongPK << " keys a' " <<
+ tSizeOfLongPK * 4 << " bytes each." << endl;
+
+ ndbout << " " << "Verification is " ;
+ if(VerifyFlag) {
+ ndbout << "enabled" << endl ;
+ }else{
+ ndbout << "disabled" << endl ;
+ }
+ theErrorData.printSettings(ndbout);
+
+ NdbThread_SetConcurrencyLevel(tNoOfThreads + 2);
+
+ Ndb* pNdb;
+ pNdb = new Ndb( "TEST_DB" );
+ pNdb->init();
+
+ tNodeId = pNdb->getNodeId();
+ ndbout << " NdbAPI node with id = " << tNodeId << endl;
+ ndbout << endl;
+
+ ndbout << "Waiting for ndb to become ready..." <<endl;
+ if (pNdb->waitUntilReady(2000) != 0){
+ ndbout << "NDB is not ready" << endl;
+ ndbout << "Benchmark failed!" << endl;
+ returnValue = NDBT_FAILED;
+ }
+
+ if(returnValue == NDBT_OK){
+ if (createTables(pNdb) != 0){
+ returnValue = NDBT_FAILED;
+ }
+ }
+
+ if(returnValue == NDBT_OK){
+
+ sleepBeforeStartingTest(tSleepTime);
+
+ /****************************************************************
+ * Create threads. *
+ ****************************************************************/
+ resetThreads(pThreadsData);
+
+ for (unsigned int i = 0; i < tNoOfThreads; i++){
+ pThreadsData[i].threadNo = i;
+ pThreadsData[i].threadLife = NdbThread_Create(flexBenchThread,
+ (void**)&pThreadsData[i],
+ 32768,
+ "flexBenchThread",
+ NDB_THREAD_PRIO_LOW);
+ }
+
+ waitForThreads(pThreadsData);
+
+ ndbout << endl << "All threads started" << endl << endl;
+
+ /****************************************************************
+ * Execute program. *
+ ****************************************************************/
+
+ for(;;){
+
+ int loopCount = tLoops + 1;
+ ndbout << endl << "Loop # " << loopCount << endl << endl;
+
+ /****************************************************************
+ * Perform inserts. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give insert-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stInsert);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing insert" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("insert", tNoOfOperations*tNoOfThreads, tNoOfTables);
+ /****************************************************************
+ * Verify inserts. *
+ ****************************************************************/
+ if (VerifyFlag) {
+ resetThreads(pThreadsData);
+ ndbout << "Verifying inserts...\t" ;
+ tellThreads(pThreadsData, stVerify);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed while verifying inserts" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }else{
+ ndbout << "\t\tOK" << endl << endl ;
+ }
+ }
+
+ /****************************************************************
+ * Perform read. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give read-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stRead);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing read" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("read", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ /****************************************************************
+ * Perform update. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give insert-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stUpdate);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing update" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("update", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ /****************************************************************
+ * Verify updates. *
+ ****************************************************************/
+ if (VerifyFlag) {
+ resetThreads(pThreadsData);
+ ndbout << "Verifying updates...\t" ;
+ tellThreads(pThreadsData, stVerify);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed while verifying updates" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }else{
+ ndbout << "\t\tOK" << endl << endl ;
+ }
+ }
+
+ /****************************************************************
+ * Perform read. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give insert-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stRead);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing read" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("read", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ /****************************************************************
+ * Perform delete. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give insert-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stDelete);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing delete" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("delete", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ /****************************************************************
+ * Verify deletes. *
+ ****************************************************************/
+ if (VerifyFlag) {
+ resetThreads(pThreadsData);
+ ndbout << "Verifying tuple deletion..." ;
+ tellThreads(pThreadsData, stVerifyDelete);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in verifying deletes" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }else{
+ ndbout << "\t\tOK" << endl << endl ;
+ }
+ }
+
+ ndbout << "--------------------------------------------------" << endl;
+
+ tLoops++;
+
+ if ( 0 != tNoOfLoops && tNoOfLoops <= tLoops )
+ break;
+ theErrorData.printErrorCounters();
+ }
+
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stStop);
+ waitForThreads(pThreadsData);
+
+ void * tmp;
+ for(int i = 0; i<tNoOfThreads; i++){
+ NdbThread_WaitFor(pThreadsData[i].threadLife, &tmp);
+ NdbThread_Destroy(&pThreadsData[i].threadLife);
+ }
+ }
+
+ if (useLongKeys == true) {
+ // Only free these areas if they have been allocated
+ // Otherwise cores will happen
+ for (int i = 0; i < tNoOfLongPK; i++)
+ free(longKeyAttrName[i]);
+ free(longKeyAttrName);
+ } // if
+
+ delete [] pThreadsData;
+ delete pNdb;
+ theErrorData.printErrorCounters();
+ return NDBT_ProgramExit(returnValue);
+}
+////////////////////////////////////////
+
+
+unsigned long get_hash(unsigned long * hash_key, int len)
+{
+ unsigned long hash_value = 147;
+ unsigned h_key;
+ int i;
+ for (i = 0; i < len; i++)
+ {
+ h_key = hash_key[i];
+ hash_value = (hash_value << 5) + hash_value + (h_key & 255);
+ hash_value = (hash_value << 5) + hash_value + ((h_key >> 8) & 255);
+ hash_value = (hash_value << 5) + hash_value + ((h_key >> 16) & 255);
+ hash_value = (hash_value << 5) + hash_value + ((h_key >> 24) & 255);
+ }
+ return hash_value;
+}
+
+// End of warming up phase
+
+
+
+static void* flexBenchThread(void* pArg)
+{
+ ThreadData* pThreadData = (ThreadData*)pArg;
+ unsigned int threadNo, threadBase;
+ Ndb* pNdb = NULL ;
+ NdbConnection *pTrans = NULL ;
+ NdbOperation** pOps = NULL ;
+ StartType tType ;
+ StartType tSaveType ;
+ NdbRecAttr* tTmp = NULL ;
+ int* attrValue = NULL ;
+ int* attrRefValue = NULL ;
+ int check = 0 ;
+ int loopCountOps, loopCountTables, loopCountAttributes;
+ int tAttemptNo = 0;
+ int tRetryAttempts = 20;
+ int tResult = 0;
+ int tSpecialTrans = 0;
+ int nRefLocalOpOffset = 0 ;
+ int nReadBuffSize =
+ tNoOfTables * tNoOfAttributes * sizeof(int) * tAttributeSize ;
+ int nRefBuffSize =
+ tNoOfOperations * tNoOfAttributes * sizeof(int) * tAttributeSize ;
+ unsigned*** longKeyAttrValue;
+
+
+ threadNo = pThreadData->threadNo ;
+
+ attrValue = (int*)malloc(nReadBuffSize) ;
+ attrRefValue = (int*)malloc(nRefBuffSize) ;
+ pOps = (NdbOperation**)malloc(tNoOfTables*sizeof(NdbOperation*)) ;
+ pNdb = new Ndb( "TEST_DB" );
+
+ if(!attrValue || !attrRefValue || !pOps || !pNdb){
+ // Check allocations to make sure we got all the memory we asked for
+ ndbout << "One or more memory allocations failed when starting thread #";
+ ndbout << threadNo << endl ;
+ ndbout << "Thread #" << threadNo << " will now exit" << endl ;
+ tResult = 13 ;
+ free(attrValue) ;
+ free(attrRefValue) ;
+ free(pOps) ;
+ delete pNdb ;
+ NdbThread_Exit(0) ;
+ }
+
+ pNdb->init();
+ pNdb->waitUntilReady();
+
+ // To make sure that two different threads doesn't operate on the same record
+ // Calculate an "unique" number to use as primary key
+ threadBase = (threadNo * 2000000) + (tNodeId * 260000000);
+
+ if(useLongKeys){
+ // Allocate and populate the longkey array.
+ longKeyAttrValue = (unsigned ***) malloc(sizeof(unsigned**) * tNoOfOperations );
+ for (int n = 0; n < tNoOfOperations; n++)
+ longKeyAttrValue[n] = (unsigned **) malloc(sizeof(unsigned*) * tNoOfLongPK );
+ for (int n = 0; n < tNoOfOperations; n++){
+ for (int i = 0; i < tNoOfLongPK ; i++) {
+ longKeyAttrValue[n][i] = (unsigned *) malloc(sizeof(unsigned) * tSizeOfLongPK);
+ memset(longKeyAttrValue[n][i], 0, sizeof(unsigned) * tSizeOfLongPK);
+ for(int j = 0; j < tSizeOfLongPK; j++) {
+ // Repeat the unique value to fill up the long key.
+ longKeyAttrValue[n][i][j] = threadBase + n;
+ }
+ }
+ }
+ }
+
+ int nRefOpOffset = 0 ;
+ //Assign reference attribute values to memory
+ for(int ops = 1 ; ops < tNoOfOperations ; ops++){
+ // Calculate offset value before going into the next loop
+ nRefOpOffset = tAttributeSize*tNoOfAttributes*(ops-1) ;
+ for(int a = 0 ; a < tNoOfAttributes ; a++){
+ *(int*)&attrRefValue[nRefOpOffset + tAttributeSize*a] =
+ (int)(threadBase + ops + a) ;
+ }
+ }
+
+#ifdef CEBIT_STAT
+ // ops not yet reported
+ int statOps = 0;
+#endif
+ for (;;) {
+ pThreadData->threadResult = tResult; // Report error to main thread,
+ // normally tResult is set to 0
+ pThreadData->threadReady = 1;
+
+ while (pThreadData->threadStart == stIdle){
+ NdbSleep_MilliSleep(100);
+ }//while
+
+ // Check if signal to exit is received
+ if (pThreadData->threadStart == stStop){
+ pThreadData->threadReady = 1;
+ // ndbout_c("Thread%d is stopping", threadNo);
+ // In order to stop this thread, the main thread has signaled
+ // stStop, break out of the for loop so that destructors
+ // and the proper exit functions are called
+ break;
+ }//if
+
+ tType = pThreadData->threadStart;
+ tSaveType = tType;
+ pThreadData->threadStart = stIdle;
+
+ // Start transaction, type of transaction
+ // is received in the array ThreadStart
+ loopCountOps = tNoOfOperations;
+ loopCountTables = tNoOfTables;
+ loopCountAttributes = tNoOfAttributes;
+
+ for (int count = 1; count < loopCountOps && tResult == 0;){
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ // This is a fatal error, abort program
+ ndbout << "Could not start transaction in thread" << threadNo;
+ ndbout << endl;
+ ndbout << pNdb->getNdbError() << endl;
+ tResult = 1; // Indicate fatal error
+ break; // Break out of for loop
+ }
+
+ // Calculate the current operation offset in the reference array
+ nRefLocalOpOffset = tAttributeSize*tNoOfAttributes*(count - 1) ;
+
+ for (int countTables = 0;
+ countTables < loopCountTables && tResult == 0;
+ countTables++) {
+
+ pOps[countTables] = pTrans->getNdbOperation(tableName[countTables]);
+ if (pOps[countTables] == NULL) {
+ // This is a fatal error, abort program
+ ndbout << "getNdbOperation: " << pTrans->getNdbError();
+ tResult = 2; // Indicate fatal error
+ break;
+ }//if
+
+ switch (tType) {
+ case stInsert: // Insert case
+ if (theWriteFlag == 1 && theDirtyFlag == 1)
+ pOps[countTables]->dirtyWrite();
+ else if (theWriteFlag == 1)
+ pOps[countTables]->writeTuple();
+ else
+ pOps[countTables]->insertTuple();
+ break;
+ case stRead: // Read Case
+ if (theSimpleFlag == 1)
+ pOps[countTables]->simpleRead();
+ else if (theDirtyFlag == 1)
+ pOps[countTables]->dirtyRead();
+ else
+ pOps[countTables]->readTuple();
+ break;
+ case stUpdate: // Update Case
+ if (theWriteFlag == 1 && theDirtyFlag == 1)
+ pOps[countTables]->dirtyWrite();
+ else if (theWriteFlag == 1)
+ pOps[countTables]->writeTuple();
+ else if (theDirtyFlag == 1)
+ pOps[countTables]->dirtyUpdate();
+ else
+ pOps[countTables]->updateTuple();
+ break;
+ case stDelete: // Delete Case
+ pOps[countTables]->deleteTuple();
+ break;
+ case stVerify:
+ pOps[countTables]->readTuple();
+ break;
+ case stVerifyDelete:
+ pOps[countTables]->readTuple();
+ break;
+ default:
+ assert(false);
+ }//switch
+
+
+ if(useLongKeys){
+ // Loop the equal call so the complete key is send to the kernel.
+ for(int i = 0; i < tNoOfLongPK; i++)
+ pOps[countTables]->equal(longKeyAttrName[i],
+ (char *)longKeyAttrValue[count - 1][i], tSizeOfLongPK*4);
+ }
+ else
+ pOps[countTables]->equal((char*)attrName[0],
+ (char*)&attrRefValue[nRefLocalOpOffset]);
+
+ if (tType == stInsert || tType == stUpdate){
+ for (int ca = 1; ca < loopCountAttributes; ca++){
+ pOps[countTables]->setValue((char*)attrName[ca],
+ (char*)&attrRefValue[nRefLocalOpOffset + tAttributeSize*ca]);
+ }//for
+ } else if (tType == stRead || stVerify == tType) {
+ int nTableOffset = tAttributeSize *
+ loopCountAttributes *
+ countTables ;
+ for (int ca = 1; ca < loopCountAttributes; ca++) {
+ tTmp = pOps[countTables]->getValue((char*)attrName[ca],
+ (char*)&attrValue[nTableOffset + tAttributeSize*ca]);
+ }//for
+ } else if (stVerifyDelete == tType) {
+ if(useLongKeys){
+ int nTableOffset = tAttributeSize *
+ loopCountAttributes *
+ countTables ;
+ tTmp = pOps[countTables]->getValue(longKeyAttrName[0],
+ (char*)&attrValue[nTableOffset]);
+ } else {
+ int nTableOffset = tAttributeSize *
+ loopCountAttributes *
+ countTables ;
+ tTmp = pOps[countTables]->getValue((char*)attrName[0],
+ (char*)&attrValue[nTableOffset]);
+ }
+ }//if
+ }//for Tables loop
+
+ if (tResult != 0)
+ break;
+ check = pTrans->execute(Commit);
+
+ // Decide what kind of error this is
+ if ((tSpecialTrans == 1) &&
+ (check == -1)) {
+ // --------------------------------------------------------------------
+ // A special transaction have been executed, change to check = 0 in
+ // certain situations.
+ // --------------------------------------------------------------------
+ switch (tType) {
+ case stInsert: // Insert case
+ if (630 == pTrans->getNdbError().code ) {
+ check = 0;
+ ndbout << "Insert with 4007 was successful" << endl;
+ }//if
+ break;
+ case stDelete: // Delete Case
+ if (626 == pTrans->getNdbError().code ) {
+ check = 0;
+ ndbout << "Delete with 4007 was successful" << endl;
+ }//if
+ break;
+ default:
+ assert(false);
+ }//switch
+ }//if
+ tSpecialTrans = 0;
+ if (check == -1) {
+ if ((stVerifyDelete == tType) &&
+ (626 == pTrans->getNdbError().code)) {
+ // ----------------------------------------------
+ // It's good news - the deleted tuple is gone,
+ // so reset "check" flag
+ // ----------------------------------------------
+ check = 0 ;
+ } else {
+ int retCode =
+ theErrorData.handleErrorCommon(pTrans->getNdbError());
+ if (retCode == 1) {
+ ndbout_c("execute: %d, %d, %s", count, tType,
+ pTrans->getNdbError().message );
+ ndbout_c("Error code = %d", pTrans->getNdbError().code );
+ tResult = 20;
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexBench" << endl;
+ tResult = 20;
+ } else if (retCode == 3) {
+ // --------------------------------------------------------------------
+ // We are not certain if the transaction was successful or not.
+ // We must reexecute but might very well find that the transaction
+ // actually was updated. Updates and Reads are no problem here. Inserts
+ // will not cause a problem if error code 630 arrives. Deletes will
+ // not cause a problem if 626 arrives.
+ // --------------------------------------------------------------------
+ if ((tType == stInsert) || (tType == stDelete)) {
+ tSpecialTrans = 1;
+ }//if
+ }//if
+ }//if
+ }//if
+ // Check if retries should be made
+ if (check == -1 && tResult == 0) {
+ if (tAttemptNo < tRetryAttempts){
+ tAttemptNo++;
+ } else {
+ // --------------------------------------------------------------------
+ // Too many retries have been made, report error and break out of loop
+ // --------------------------------------------------------------------
+ ndbout << "Thread" << threadNo;
+ ndbout << ": too many errors reported" << endl;
+ tResult = 10;
+ break;
+ }//if
+ }//if
+
+ if (check == 0){
+ // Go to the next record
+ count++;
+ tAttemptNo = 0;
+#ifdef CEBIT_STAT
+ // report successful ops
+ if (statEnable) {
+ statOps += loopCountTables;
+ if (statOps >= statFreq) {
+ statReport(tType, statOps);
+ statOps = 0;
+ }//if
+ }//if
+#endif
+ }//if
+
+ if (stVerify == tType && 0 == check){
+ int nTableOffset = 0 ;
+ for (int a = 1 ; a < loopCountAttributes ; a++){
+ for (int tables = 0 ; tables < loopCountTables ; tables++){
+ nTableOffset = tables*loopCountAttributes*tAttributeSize ;
+ if (*(int*)&attrValue[nTableOffset + tAttributeSize*a] != *(int*)&attrRefValue[nRefLocalOpOffset + tAttributeSize*a]){
+ ndbout << "Error in verify:" << endl ;
+ ndbout << "attrValue[" << nTableOffset + tAttributeSize*a << "] = " << attrValue[a] << endl ;
+ ndbout << "attrRefValue[" << nRefLocalOpOffset + tAttributeSize*a << "]" << attrRefValue[nRefLocalOpOffset + tAttributeSize*a] << endl ;
+ tResult = 11 ;
+ break ;
+ }//if
+ }//for
+ }//for
+ }// if(stVerify ... )
+ pNdb->closeTransaction(pTrans) ;
+ }// operations loop
+#ifdef CEBIT_STAT
+ // report remaining successful ops
+ if (statEnable) {
+ if (statOps > 0) {
+ statReport(tType, statOps);
+ statOps = 0;
+ }//if
+ }//if
+#endif
+ }
+ delete pNdb;
+ free(attrValue) ;
+ free(attrRefValue) ;
+ free(pOps) ;
+
+ if (useLongKeys == true) {
+ // Only free these areas if they have been allocated
+ // Otherwise cores will occur
+ for (int n = 0; n < tNoOfOperations; n++){
+ for (int i = 0; i < tNoOfLongPK; i++) {
+ free(longKeyAttrValue[n][i]);
+ }
+ free(longKeyAttrValue[n]);
+ }
+ free(longKeyAttrValue);
+ } // if
+
+ NdbThread_Exit(0);
+ return NULL; // Just to keep compiler happy
+}
+
+
+static int readArguments(int argc, const char** argv)
+{
+
+ int i = 1;
+ while (argc > 1){
+ if (strcmp(argv[i], "-t") == 0){
+ tNoOfThreads = atoi(argv[i+1]);
+ if ((tNoOfThreads < 1))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-o") == 0){
+ tNoOfOperations = atoi(argv[i+1]);
+ if (tNoOfOperations < 1)
+ return -1;;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-a") == 0){
+ tNoOfAttributes = atoi(argv[i+1]);
+ if ((tNoOfAttributes < 2) || (tNoOfAttributes > MAXATTR))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-lkn") == 0){
+ tNoOfLongPK = atoi(argv[i+1]);
+ useLongKeys = true;
+ if ((tNoOfLongPK < 1) || (tNoOfLongPK > MAXNOLONGKEY) ||
+ (tNoOfLongPK * tSizeOfLongPK) > MAXLONGKEYTOTALSIZE){
+ ndbout << "Argument -lkn is not in the proper range." << endl;
+ return -1;
+ }
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-lks") == 0){
+ tSizeOfLongPK = atoi(argv[i+1]);
+ useLongKeys = true;
+ if ((tSizeOfLongPK < 1) || (tNoOfLongPK * tSizeOfLongPK) > MAXLONGKEYTOTALSIZE){
+ ndbout << "Argument -lks is not in the proper range 1 to " <<
+ MAXLONGKEYTOTALSIZE << endl;
+ return -1;
+ }
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-c") == 0){
+ tNoOfTables = atoi(argv[i+1]);
+ if ((tNoOfTables < 1) || (tNoOfTables > MAXTABLES))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-stdtables") == 0){
+ theStdTableNameFlag = 1;
+ }else if (strcmp(argv[i], "-l") == 0){
+ tNoOfLoops = atoi(argv[i+1]);
+ if ((tNoOfLoops < 0) || (tNoOfLoops > 100000))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-s") == 0){
+ tAttributeSize = atoi(argv[i+1]);
+ if ((tAttributeSize < 1) || (tAttributeSize > MAXATTRSIZE))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-sleep") == 0){
+ tSleepTime = atoi(argv[i+1]);
+ if ((tSleepTime < 1) || (tSleepTime > 3600))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-simple") == 0){
+ theSimpleFlag = 1;
+ }else if (strcmp(argv[i], "-write") == 0){
+ theWriteFlag = 1;
+ }else if (strcmp(argv[i], "-dirty") == 0){
+ theDirtyFlag = 1;
+ }else if (strcmp(argv[i], "-no_table_create") == 0){
+ theTableCreateFlag = 1;
+ }else if (strcmp(argv[i], "-temp") == 0){
+ theTempTable = true;
+ }else if (strcmp(argv[i], "-noverify") == 0){
+ VerifyFlag = false ;
+ }else if (theErrorData.parseCmdLineArg(argv, i) == true){
+ ; //empty, updated in errorArg(..)
+ }else if (strcmp(argv[i], "-verify") == 0){
+ VerifyFlag = true ;
+#ifdef CEBIT_STAT
+ }else if (strcmp(argv[i], "-statserv") == 0){
+ if (! (argc > 2))
+ return -1;
+ const char *p = argv[i+1];
+ const char *q = strrchr(p, ':');
+ if (q == 0)
+ return -1;
+ snprintf(statHost, sizeof(statHost), "%.*s", q-p, p);
+ statPort = atoi(q+1);
+ statEnable = true;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-statfreq") == 0){
+ if (! (argc > 2))
+ return -1;
+ statFreq = atoi(argv[i+1]);
+ if (statFreq < 1)
+ return -1;
+ argc -= 1;
+ i++;
+#endif
+ }else{
+ return -1;
+ }
+ argc -= 1;
+ i++;
+ }
+ return 0;
+}
+
+static void sleepBeforeStartingTest(int seconds){
+ if (seconds > 0){
+ ndbout << "Sleeping(" <<seconds << ")...";
+ NdbSleep_SecSleep(seconds);
+ ndbout << " done!" << endl;
+ }
+}
+
+
+static int
+createTables(Ndb* pMyNdb){
+ for (int i = 0; i < tNoOfAttributes; i++){
+ snprintf(attrName[i], MAXSTRLEN, "COL%d", i);
+ }
+
+ // Note! Uses only uppercase letters in table name's
+ // so that we can look at the tables with SQL
+ for (int i = 0; i < tNoOfTables; i++){
+ if (theStdTableNameFlag == 0){
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d_%d", i,
+ (int)(NdbTick_CurrentMillisecond() / 1000));
+ } else {
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d", i);
+ }
+ }
+
+ for(unsigned i = 0; i < tNoOfTables; i++){
+ ndbout << "Creating " << tableName[i] << "... ";
+
+ NdbDictionary::Table tmpTable(tableName[i]);
+
+ tmpTable.setStoredTable(!theTempTable);
+
+ if(useLongKeys){
+ for(int i = 0; i < tNoOfLongPK; i++) {
+ NdbDictionary::Column col(longKeyAttrName[i]);
+ col.setType(NdbDictionary::Column::Unsigned);
+ col.setLength(tSizeOfLongPK);
+ col.setPrimaryKey(true);
+ tmpTable.addColumn(col);
+ }
+ } else {
+ NdbDictionary::Column col(attrName[0]);
+ col.setType(NdbDictionary::Column::Unsigned);
+ col.setLength(1);
+ col.setPrimaryKey(true);
+ tmpTable.addColumn(col);
+ }
+
+
+ NdbDictionary::Column col;
+ col.setType(NdbDictionary::Column::Unsigned);
+ col.setLength(tAttributeSize);
+ for (unsigned j = 1; j < tNoOfAttributes; j++){
+ col.setName(attrName[j]);
+ tmpTable.addColumn(col);
+ }
+
+ if(pMyNdb->getDictionary()->createTable(tmpTable) == -1){
+ return -1;
+ }
+ ndbout << "done" << endl;
+ }
+
+ return 0;
+}
+
+
+static void input_error(){
+ ndbout << endl << "Invalid argument!" << endl;
+ ndbout << endl << "Arguments:" << endl;
+ ndbout << " -t Number of threads to start, default 1" << endl;
+ ndbout << " -o Number of operations per loop, default 500" << endl;
+ ndbout << " -l Number of loops to run, default 1, 0=infinite" << endl;
+ ndbout << " -a Number of attributes, default 25" << endl;
+ ndbout << " -c Number of tables, default 1" << endl;
+ ndbout << " -s Size of each attribute, default 1 (Primary Key is always of size 1," << endl;
+ ndbout << " independent of this value)" << endl;
+ ndbout << " -lkn Number of long primary keys, default 1" << endl;
+ ndbout << " -lks Size of each long primary key, default 1" << endl;
+
+ ndbout << " -simple Use simple read to read from database" << endl;
+ ndbout << " -dirty Use dirty read to read from database" << endl;
+ ndbout << " -write Use writeTuple in insert and update" << endl;
+ ndbout << " -stdtables Use standard table names" << endl;
+ ndbout << " -no_table_create Don't create tables in db" << endl;
+ ndbout << " -sleep Sleep a number of seconds before running the test, this" << endl;
+ ndbout << " can be used so that another flexBench have time to create tables" << endl;
+ ndbout << " -temp Use tables without logging" << endl;
+ ndbout << " -verify Verify inserts, updates and deletes" << endl ;
+ theErrorData.printCmdLineArgs(ndbout);
+ ndbout << endl <<"Returns:" << endl;
+ ndbout << "\t 0 - Test passed" << endl;
+ ndbout << "\t 1 - Test failed" << endl;
+ ndbout << "\t 2 - Invalid arguments" << endl << endl;
+}
+
+// vim: set sw=2:
diff --git a/ndb/test/ndbapi/flexBench/ndbplot.pl b/ndb/test/ndbapi/flexBench/ndbplot.pl
new file mode 100755
index 00000000000..b16f6d5897d
--- /dev/null
+++ b/ndb/test/ndbapi/flexBench/ndbplot.pl
@@ -0,0 +1,305 @@
+#! /usr/bin/perl
+
+use strict;
+use Getopt::Long;
+use Symbol;
+use Socket;
+
+my $progname = $0;
+$progname =~ s!^.*/|\.pl$!!g;
+my $defaultport = 27127;
+my $defaulttotal = 120;
+my $defaultsample = 5;
+my $defaultrange = 5000;
+
+sub printhelp {
+ print <<END;
+$progname -- plot ndb operation counts in X11 window
+usage: $progname [options]
+--help print this summary and exit
+--debug print lots of debug information
+--port N port number to listen on, default $defaultport
+--total N total time interval shown, default $defaulttotal seconds
+--sample N sample interval, default $defaultsample seconds
+--range N range (max ops per second), default $defaultrange
+--nopct show no percentages in graph titles
+--z "..." add X11/gnuplot options, for example:
+ --z "-bg grey80 -geometry 500x300" --z "-persist"
+END
+ exit(0);
+}
+
+# get options
+use vars qw(
+ $helpflag $debug $serverport $totaltime $sampletime $range $nopct
+ @zopts
+);
+$helpflag = 0;
+$debug = 0;
+$serverport = $defaultport;
+$totaltime = $defaulttotal;
+$sampletime = $defaultsample;
+$range = $defaultrange;
+$nopct = 0;
+@zopts = ();
+GetOptions(
+ 'help' => \$helpflag,
+ 'debug' => \$debug,
+ 'port=i' => \$serverport,
+ 'total=i' => \$totaltime,
+ 'sample=i' => \$sampletime,
+ 'range=i' => \$range,
+ 'nopct' => \$nopct,
+ 'z=s' => \@zopts,
+) or die "try: $progname -h\n";
+$helpflag && printhelp();
+
+# calculate number of data points
+my $samplecnt;
+$samplecnt = int($totaltime / $sampletime) + 1;
+$totaltime = ($samplecnt - 1) * $sampletime;
+warn "total time = $totaltime sec, sample time = $sampletime sec\n";
+
+# open gnuplot
+my $plotfile;
+sub openplot {
+ $plotfile = gensym();
+ if (! open($plotfile, "| gnuplot @zopts")) {
+ die "open plot: $!\n";
+ }
+ my $sav = select($plotfile);
+ $| = 1;
+ select($sav);
+ print $plotfile "clear\n";
+}
+
+# samples
+my @sample; # samples 0..$samplecnt in time order
+my $sampleready = 0; # samples 1..$samplecnt are ready (true/false)
+
+@sample = map({ start => 0 }, 0..$samplecnt);
+
+sub adddata {
+ my($node, $type, $value) = @_;
+ my $now = time;
+ my $s = $sample[0];
+ if ($now - $s->{start} >= $sampletime) {
+ unshift(@sample, {
+ start => $now,
+ total => 0,
+ });
+ $s = $sample[0];
+ pop(@sample); # delete oldest
+ $sampleready = 1;
+ }
+ # if no type then this is just a time tick
+ if ($type) {
+ $s->{$type} += $value;
+ $s->{total} += $value;
+ }
+}
+
+# data file name
+my $datadir;
+if ($ENV{NDB_BASE}) {
+ $datadir = "$ENV{NDB_BASE}/var/plot";
+} else {
+ $datadir = "/var/tmp";
+}
+(-d $datadir || mkdir($datadir, 0777))
+ or die "mkdir $datadir failed: $!\n";
+my $datafile = "$datadir/plot$$.dat";
+warn "writing plot data to $datafile\n";
+
+# refresh the plot
+sub plotsample {
+ my $fh = gensym();
+ if (! open($fh, ">$datafile")) {
+ die "$datafile: $!\n";
+ }
+ # sample 0 is never ready
+ my $currops = "";
+ my $currpct = {};
+ for (my $i = @sample; $i >= 1; $i--) {
+ my $s = $sample[$i];
+ if (! $s->{start}) { # initial empty sample
+ next;
+ }
+ printf $fh "%d", -($i - 1) * $sampletime;
+ printf $fh " %.0f", 1.01 * $s->{"total"} / $sampletime;
+ for my $k (qw(insert update select delete)) {
+ printf $fh " %.0f", $s->{$k} / $sampletime;
+ }
+ printf $fh "\n";
+ if ($i == 1) {
+ $currops = sprintf("%.0f", $s->{"total"} / $sampletime);
+ if (! $nopct && $currops > 0) {
+ $currpct->{"total"} = sprintf("%5s", "");
+ for my $k (qw(insert update select delete)) {
+ $currpct->{$k} = sprintf(" %3.0f%%",
+ 100.0 * $s->{$k} / $s->{"total"});
+ }
+ }
+ }
+ }
+ close($fh);
+ print $plotfile <<END;
+clear
+set title "ops/sec [ $currops ]"
+set xrange [@{[ -($totaltime-1) ]}:0]
+set yrange [0:$range]
+plot \\
+ '$datafile' \\
+ using 1:3 \\
+ title "insert$currpct->{insert}" \\
+ with lines lt 2, \\
+ '$datafile' \\
+ using 1:4 \\
+ title "update$currpct->{update}" \\
+ with lines lt 3, \\
+ '$datafile' \\
+ using 1:5 \\
+ title "select$currpct->{select}" \\
+ with lines lt 4, \\
+ '$datafile' \\
+ using 1:6 \\
+ title "delete$currpct->{delete}" \\
+ with lines lt 5, \\
+ '$datafile' \\
+ using 1:2 \\
+ title "total$currpct->{total}" \\
+ with lines lt 1 lw 2
+END
+}
+
+# set up server socket
+my $sock = gensym();
+if (! socket($sock, PF_INET, SOCK_STREAM, getprotobyname("tcp"))) {
+ die "socket: $!\n";
+}
+if (! setsockopt($sock, SOL_SOCKET, SO_REUSEADDR, pack("l*", 1))) {
+ die "setsockopt: $!\n";
+}
+if (! bind($sock, pack_sockaddr_in($serverport, INADDR_ANY))) {
+ die "bind: $!\n";
+}
+if (! listen($sock, SOMAXCONN)) {
+ die "listen: $!\n";
+}
+
+# bit vectors for select on server socket and clients
+my $readin = '';
+vec($readin, fileno($sock), 1) = 1;
+
+# clients
+my @client = ();
+my $clientid = 0;
+sub addclient {
+ my($conn) = @_;
+ my $c = {
+ conn => $conn,
+ data => "",
+ name => "client " . ++$clientid,
+ };
+ push(@client, $c);
+ vec($readin, fileno($c->{conn}), 1) = 1;
+ if (1 || $debug) {
+ warn "added $c->{name}\n";
+ }
+}
+sub deleteclient {
+ my($c) = @_;
+ @client = grep($_ ne $c, @client);
+ vec($readin, fileno($c->{conn}), 1) = 0;
+ shutdown($c->{conn}, 2);
+ if (1 || $debug) {
+ warn "deleted $c->{name}\n";
+ }
+}
+sub readclient {
+ my($c) = @_;
+ my $data;
+ my $n;
+ eval {
+ local $SIG{ALRM} = sub { die "timeout\n" };
+ alarm(5);
+ $n = sysread($c->{conn}, $data, 512);
+ alarm(0);
+ };
+ if ($@) {
+ chomp($@);
+ warn "$c->{name}: read: $@\n";
+ return undef;
+ }
+ if (!defined($n)) {
+ warn "$c->{name}: read: $!\n";
+ return undef;
+ }
+ $c->{data} .= $data;
+ if ($debug) {
+ warn "$c->{name}: read @{[ length($data) ]} bytes\n";
+ }
+ return $n;
+}
+sub processclient {
+ my($c) = @_;
+ my $i;
+ while (($i = index($c->{data}, "\n")) >= 0) {
+ my $line = substr($c->{data}, 0, $i);
+ $c->{data} = substr($c->{data}, $i+1);
+ my($node, $type, $value) = split(' ', $line);
+ if ($node !~ /^\d+$/) {
+ warn "$c->{name}: $line: bad node id\n";
+ next;
+ }
+ if ($type !~ /^(insert|update|read|delete|verify|verifydelete)$/) {
+ warn "$c->{name}: $line: bad type\n";
+ next;
+ }
+ if ($value !~ /^\d+$/) {
+ warn "$c->{name}: $line: bad value\n";
+ next;
+ }
+ if ($type eq "read") {
+ $type = "select";
+ }
+ adddata($node, $type, $value);
+ }
+}
+
+# main loop
+openplot();
+while (1) {
+ my $readout = '';
+ my $ret = select($readout = $readin, undef, undef, 1.0);
+ if (vec($readout, fileno($sock), 1)) {
+ my $conn = gensym();
+ if (! accept($conn, $sock)) {
+ warn "accept failed: $!\n";
+ } else {
+ addclient($conn);
+ }
+ }
+ for my $c (@client) {
+ if (vec($readout, fileno($c->{conn}), 1)) {
+ my $n = readclient($c);
+ if (! defined($n)) {
+ deleteclient($c);
+ } else {
+ processclient($c);
+ if ($n == 0) { # end of file
+ deleteclient($c);
+ }
+ }
+ }
+ }
+ adddata(); # keep clock ticking
+ if ($sampleready) {
+ if ($debug) {
+ warn "sample ready\n";
+ }
+ plotsample();
+ $sampleready = 0;
+ }
+}
+# vim: set sw=4:
diff --git a/ndb/test/ndbapi/flexHammer/Makefile b/ndb/test/ndbapi/flexHammer/Makefile
new file mode 100644
index 00000000000..c8e436fb7f5
--- /dev/null
+++ b/ndb/test/ndbapi/flexHammer/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := flexHammer
+
+SOURCES := flexHammer.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/flexHammer/README b/ndb/test/ndbapi/flexHammer/README
new file mode 100644
index 00000000000..556582aab96
--- /dev/null
+++ b/ndb/test/ndbapi/flexHammer/README
@@ -0,0 +1,67 @@
+
+Executing flexHammer-tests automatically
+========================================
+
+
+It is possible to execute almost al the flexHammer-tests
+automatically. The procedure contains three steps:
+- increase the number of tabels (flexHammer -c number)
+- increase the number of threads (flexHammer -t number)
+- increase the number of records (flexHammer -r number)
+- increase the number of tabels and threads alternately
+
+Each of these steps are performed by the scripts test1.sh,
+test2.sh, test3.sh and test4.sh. Each test will start Ndb,
+execute the test and close Ndb again in order to execute
+each test in a 'clean' Ndb-environment. So make sure that
+there is no Ndb running when you start the test.
+
+
+1. Setup
+
+To perform the tests automatically, the following issues
+have to be taken care of:
+
+- be sure that you have a directory bin in your home-directory.
+ In this directory, you need to have a link 'runndb' to the
+ ndb executable. You can do this by executing a shell-command like:
+ ln -s ndb/Emulator/Main/ndb runndb
+ The script is not yet so far that it performs checks, so if
+ you forget about this, things will get messy.
+- In this directory you need a Ndb.cfg for a server-configuration.
+
+
+2. Command
+
+I assume you have Ndb and the API compiled or you use the
+'released' version. Compile flexHammer as usual with 'make'.
+Now you can start the tests by typing 'make test'. The
+execution of the test will take a while.
+
+
+3. Results
+
+The scripts will write their results in the file report.txt.
+The scripts will start with a short summary on the test. Then
+it will add 1 line documenting each run of flexHammer that is
+ececuted. Finally, it will print highest 'score'. The file
+report.txt is probably good enough to check in directly as
+testprotocol in ndb/test/docs/testprotocols.
+
+
+4. Log files.
+
+To make it possible to investigate errors, the output from
+the flexScan-run where the error occurred is stored in
+test1.log, test2.log, test3.log or test4.log respectively.
+They are overwritten each time you start 'make test'.
+
+
+HINT
+
+The number of iterations in each test-script is not directly
+limited by the number of attributes or the size of the
+attributes but by the number of tables that you are allowed
+to create. Probably this will be the error that occurs if
+you execute the test. You migh adjust the begin-values and
+the step-size in the individual scripts if you want.
diff --git a/ndb/test/ndbapi/flexHammer/flexHammer.cpp b/ndb/test/ndbapi/flexHammer/flexHammer.cpp
new file mode 100644
index 00000000000..f8a519f021a
--- /dev/null
+++ b/ndb/test/ndbapi/flexHammer/flexHammer.cpp
@@ -0,0 +1,891 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* ***************************************************
+ FLEXHAMMER
+ Hammer ndb with read, insert, update and delete transactions.
+
+ Arguments:
+ -t Number of threads to start, default 1
+ -o Number of operations per hammering-round, default 500
+ -l Number of loops to run, default 1, 0=infinite
+ -a Number of attributes, default 25
+ -c Number of tables, default 1
+ -s Size of each attribute, default 1
+ -simple Use simple read to read from database
+ -dirty Use dirty read to read from database
+ -write Use writeTuple to write to db
+ -r Number of records to Hammer
+ -no_table_create Don't create tables in db
+ -regulate To be able to regulate the load flexHammer produces.
+ -stdtables Use standard table names
+ -sleep Sleep a number of seconds before running the test, this
+ can be used so that another flexProgram have tome to create tables
+
+ Returns:
+ 0 - Test passed
+ -1 - Test failed
+ 1 - Invalid arguments
+
+Revision history:
+ 1.7 020208 epesson: Adapted to use NDBT
+ 1.10 020222 epesson: Finalised handling of thread results
+ 1.11 020222 epesson: Bug in checking results during delete fixed
+
+ * *************************************************** */
+
+#include <NdbApi.hpp>
+
+#include <NdbMain.h>
+#include <NdbThread.h>
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#include <NdbOut.hpp>
+#include <NdbTimer.hpp>
+#include <NdbTick.h>
+#include <NdbTest.hpp>
+#include <string.h>
+#include <assert.h>
+#include <NDBT_Error.hpp>
+
+ErrorData * flexHammerErrorData;
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+#include <outfmt.h>
+#endif
+
+#define MAXSTRLEN 16
+#define MAXATTR 64
+#define MAXTABLES 64
+#define MAXTHREADS 256
+#define MAXATTRSIZE 100
+// Max number of retries if something fails
+#define MaxNoOfAttemptsC 10
+
+enum StartType {
+ stIdle,
+ stHammer,
+ stStop,
+ stLast};
+
+enum MyOpType {
+ otInsert,
+ otRead,
+ otDelete,
+ otUpdate,
+ otLast};
+
+struct ThreadNdb {
+ int threadNo;
+ NdbThread* threadLife;
+ int threadReady;
+ StartType threadStart;
+ int threadResult;};
+
+extern "C" void* flexHammerThread(void*);
+static int setAttrNames(void);
+static int setTableNames(void);
+static int readArguments(int, const char**);
+static int createTables(Ndb*);
+static void sleepBeforeStartingTest(int seconds);
+static int checkThreadResults(ThreadNdb *threadArrayP, char* phase);
+
+//enum OperationType {
+// otInsert,
+// otRead,
+// otUpdate,
+// otDelete,
+// otVerifyDelete,
+// otLast };
+
+enum ReadyType {
+ stReady,
+ stRunning
+} ;
+static int tNoOfThreads;
+static int tNoOfAttributes;
+static int tNoOfTables;
+static int tNoOfBackups;
+static int tAttributeSize;
+static int tNoOfOperations;
+static int tNoOfRecords;
+static int tNoOfLoops;
+static ReadyType ThreadReady[MAXTHREADS];
+static StartType ThreadStart[MAXTHREADS];
+static char tableName[MAXTABLES][MAXSTRLEN];
+static char attrName[MAXATTR][MAXSTRLEN];
+static int theSimpleFlag = 0;
+static int theWriteFlag = 0;
+static int theDirtyFlag = 0;
+static int theTableCreateFlag = 0;
+static int theStandardTableNameFlag = 0;
+static unsigned int tSleepTime = 0;
+
+#define START_TIMER { NdbTimer timer; timer.doStart();
+#define STOP_TIMER timer.doStop();
+#define PRINT_TIMER(text, trans, opertrans) timer.printTransactionStatistics(text, trans, opertrans); };
+
+
+// Initialise thread data
+void
+resetThreads(ThreadNdb *threadArrayP) {
+
+ for (int i = 0; i < tNoOfThreads ; i++)
+ {
+ threadArrayP[i].threadReady = 0;
+ threadArrayP[i].threadResult = 0;
+ threadArrayP[i].threadStart = stIdle;
+ }
+} // resetThreads
+
+void
+waitForThreads(ThreadNdb *threadArrayP)
+{
+ int cont = 1;
+
+ while (cont) {
+ NdbSleep_MilliSleep(100);
+ cont = 0;
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ if (threadArrayP[i].threadReady == 0) {
+ cont = 1;
+ } // if
+ } // for
+ } // while
+} // waitForThreads
+
+void
+tellThreads(ThreadNdb* threadArrayP, const StartType what)
+{
+ for (int i = 0; i < tNoOfThreads ; i++)
+ {
+ threadArrayP[i].threadStart = what;
+ } // for
+} // tellThreads
+
+NDB_COMMAND(flexHammer, "flexHammer", "flexHammer", "flexHammer", 65535)
+//main(int argc, const char** argv)
+{
+ ThreadNdb* pThreads = NULL; // Pointer to thread data array
+ Ndb* pMyNdb = NULL; // Pointer to Ndb object
+ int tLoops = 0;
+ int returnValue = 0;
+ int check = 0;
+
+ flexHammerErrorData = new ErrorData;
+
+ flexHammerErrorData->resetErrorCounters();
+
+ if (readArguments(argc, argv) != 0) {
+ ndbout << "Wrong arguments to flexHammer" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ } // if
+
+ /* print Setting */
+ flexHammerErrorData->printSettings(ndbout);
+
+ check = setAttrNames();
+ if (check == -1) {
+ ndbout << "Couldn't set attribute names" << endl;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ } // if
+ check = setTableNames();
+ if (check == -1) {
+ ndbout << "Couldn't set table names" << endl;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ } // if
+
+ // Create thread data array
+ pThreads = new ThreadNdb[tNoOfThreads];
+ // NdbThread_SetConcurrencyLevel(tNoOfThreads + 2);
+
+ // Create and init Ndb object
+ pMyNdb = new Ndb("TEST_DB");
+ pMyNdb->init();
+
+ // Wait for Ndb to become ready
+ if (pMyNdb->waitUntilReady(10000) != 0) {
+ ndbout << "NDB is not ready" << endl << "Benchmark failed" << endl;
+ returnValue = NDBT_FAILED;
+ }
+
+ else {
+ check = createTables(pMyNdb);
+ if (check != 0) {
+ returnValue = NDBT_FAILED;
+ } // if
+ else {
+ sleepBeforeStartingTest(tSleepTime);
+
+ // Create threads. *
+ resetThreads(pThreads);
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ pThreads[i].threadNo = i;
+ pThreads[i].threadLife = NdbThread_Create(flexHammerThread,
+ (void**)&pThreads[i],
+ 65535,
+ "flexHammerThread",
+ NDB_THREAD_PRIO_LOW);
+ } // for
+
+ // And wait until they are ready
+ waitForThreads(pThreads);
+ if (checkThreadResults(pThreads, "init") != 0) {
+ returnValue = NDBT_FAILED;
+ } // if
+
+
+ if (returnValue == NDBT_OK) {
+ ndbout << endl << "All threads started" << endl << endl;
+
+ for(;;) {
+
+ // Check if it's time to exit program
+ if((tNoOfLoops != 0) && (tNoOfLoops <= tLoops))
+ break;
+
+ // Tell all threads to start hammer
+ ndbout << "Hammering..." << endl;
+
+ resetThreads(pThreads);
+
+ START_TIMER;
+ tellThreads(pThreads, stHammer);
+
+ waitForThreads(pThreads);
+ ndbout << "Threads ready to continue..." << endl;
+ STOP_TIMER;
+
+ // Check here if anything went wrong
+ if (checkThreadResults(pThreads, "hammer") != 0) {
+ ndbout << "Thread(s) failed." << endl;
+ returnValue = NDBT_FAILED;
+ } // if
+
+ PRINT_TIMER("hammer", tNoOfOperations*tNoOfThreads, tNoOfTables*6);
+
+ ndbout << endl;
+
+ tLoops++;
+
+ } // for
+ } // if
+
+ // Signaling threads to stop
+ resetThreads(pThreads);
+ tellThreads(pThreads, stStop);
+
+ // Wait for threads to stop
+ waitForThreads(pThreads);
+
+ ndbout << "----------------------------------------------" << endl << endl;
+ ndbout << "Benchmark completed" << endl;
+ } // else
+ } // else
+ // Clean up
+
+ flexHammerErrorData->printErrorCounters(ndbout);
+
+ // Kill them all!
+ void* tmp;
+ for(int i = 0; i < tNoOfThreads; i++){
+ NdbThread_WaitFor(pThreads[i].threadLife, &tmp);
+ NdbThread_Destroy(&pThreads[i].threadLife);
+ }
+ delete flexHammerErrorData;
+ delete [] pThreads;
+ delete pMyNdb;
+
+ // Exit via NDBT
+ return NDBT_ProgramExit(returnValue);
+
+} //main
+
+extern "C"
+void*
+flexHammerThread(void* pArg)
+{
+ ThreadNdb* pThreadData = (ThreadNdb*)pArg;
+ unsigned int threadNo = pThreadData->threadNo;
+ Ndb* pMyNdb = NULL ;
+ NdbConnection *pMyTransaction = NULL ;
+ // NdbOperation* pMyOperation[MAXTABLES] = {NULL};
+ NdbOperation* pMyOperation[MAXTABLES];
+ int check = 0;
+ int loop_count_ops = 0;
+ int loop_count_tables = 0;
+ int loop_count_attributes = 0;
+ int count_round = 0;
+ int count = 0;
+ int count_tables = 0;
+ int count_attributes = 0;
+ int i = 0;
+ int j = 0;
+ int tThreadResult = 0;
+ MyOpType tMyOpType = otLast;
+ int pkValue = 0;
+ int readValue[MAXATTR][MAXATTRSIZE] = {0};
+ int attrValue[MAXATTRSIZE];
+ NdbRecAttr* tTmp = NULL;
+ int tNoOfAttempts = 0;
+
+ for (i = 0; i < MAXATTRSIZE; i++)
+ attrValue[i] = 0;
+ // Ndb object for each thread
+ pMyNdb = new Ndb( "TEST_DB" );
+ pMyNdb->init();
+ if (pMyNdb->waitUntilReady(10000) != 0) {
+ // Error, NDB is not ready
+ tThreadResult = 99;
+ // Go to idle directly
+ pThreadData->threadStart = stIdle;
+ } // if
+
+ for(;;) {
+ pThreadData->threadResult = tThreadResult;
+ pThreadData->threadReady = 1; // Signalling ready to main
+
+ // If Idle just wait to be stopped from main
+ while (pThreadData->threadStart == stIdle) {
+ NdbSleep_MilliSleep(100);
+ } // while
+
+ // Check if signal to exit is received
+ if (pThreadData->threadStart == stStop) {
+ pThreadData->threadReady = 1;
+ // break out of eternal loop
+ break;
+ } // if
+
+ // Set to Idle to prepare for possible error break
+ pThreadData->threadStart = stIdle;
+
+ // Prepare transaction
+ loop_count_ops = tNoOfOperations;
+ loop_count_tables = tNoOfTables;
+ loop_count_attributes = tNoOfAttributes;
+
+ for (count=0 ; count < loop_count_ops ; count++) {
+
+ //pkValue = (int)(count + thread_base);
+ // This limits the number of records used in this test
+ pkValue = count % tNoOfRecords;
+
+ for (count_round = 0; count_round < 5; ) {
+ switch (count_round) {
+ case 0: // Insert
+ tMyOpType = otInsert;
+ // Increase attrValues
+ for (i=0; i < MAXATTRSIZE; i ++) {
+ attrValue[i]++;
+ }
+ break;
+ case 1:
+ case 3: // Read and verify
+ tMyOpType = otRead;
+ break;
+ case 2: // Update
+ // Increase attrValues
+ for(i=0; i < MAXATTRSIZE; i ++) {
+ attrValue[i]++;
+ }
+ tMyOpType = otUpdate;
+ break;
+ case 4: // Delete
+ tMyOpType = otDelete;
+ break;
+ default:
+ assert(false);
+ break;
+ } // switch
+
+ // Get transaction object
+ pMyTransaction = pMyNdb->startTransaction();
+ if (pMyTransaction == NULL) {
+ // Fatal error
+ tThreadResult = 1;
+ // break out of for count_round loop waiting to be stopped by main
+ break;
+ } // if
+
+ for (count_tables = 0; count_tables < loop_count_tables;
+ count_tables++) {
+ pMyOperation[count_tables] =
+ pMyTransaction->getNdbOperation(tableName[count_tables]);
+ if (pMyOperation[count_tables] == NULL) {
+ //Fatal error
+ tThreadResult = 2;
+ // break out of inner for count_tables loop
+ break;
+ } // if
+
+ switch (tMyOpType) {
+ case otInsert: // Insert case
+ if (theWriteFlag == 1 && theDirtyFlag == 1) {
+ check = pMyOperation[count_tables]->dirtyWrite();
+ } else if (theWriteFlag == 1) {
+ check = pMyOperation[count_tables]->writeTuple();
+ } else {
+ check = pMyOperation[count_tables]->insertTuple();
+ } // if else
+ break;
+ case otRead: // Read Case
+ if (theSimpleFlag == 1) {
+ check = pMyOperation[count_tables]->simpleRead();
+ } else if (theDirtyFlag == 1) {
+ check = pMyOperation[count_tables]->dirtyRead();
+ } else {
+ check = pMyOperation[count_tables]->readTuple();
+ } // if else
+ break;
+ case otUpdate: // Update Case
+ if (theWriteFlag == 1 && theDirtyFlag == 1) {
+ check = pMyOperation[count_tables]->dirtyWrite();
+ } else if (theWriteFlag == 1) {
+ check = pMyOperation[count_tables]->writeTuple();
+ } else if (theDirtyFlag == 1) {
+ check = pMyOperation[count_tables]->dirtyUpdate();
+ } else {
+ check = pMyOperation[count_tables]->updateTuple();
+ } // if else
+ break;
+ case otDelete: // Delete Case
+ check = pMyOperation[count_tables]->deleteTuple();
+ break;
+ default:
+ assert(false);
+ break;
+ } // switch
+ if (check == -1) {
+ // Fatal error
+ tThreadResult = 3;
+ // break out of inner for count_tables loop
+ break;
+ } // if
+
+ check = pMyOperation[count_tables]->equal( (char*)attrName[0],
+ (char*)&pkValue );
+
+ if (check == -1) {
+ // Fatal error
+ tThreadResult = 4;
+ ndbout << "pMyOperation equal failed" << endl;
+ // break out of inner for count_tables loop
+ break;
+ } // if
+
+ check = -1;
+ tTmp = NULL;
+ switch (tMyOpType) {
+ case otInsert: // Insert case
+ case otUpdate: // Update Case
+ for (count_attributes = 1; count_attributes < loop_count_attributes;
+ count_attributes++) {
+ check =
+ pMyOperation[count_tables]->setValue((char*)attrName[count_attributes], (char*)&attrValue[0]);
+ } // for
+ break;
+ case otRead: // Read Case
+ for (count_attributes = 1; count_attributes < loop_count_attributes;
+ count_attributes++) {
+ tTmp = pMyOperation[count_tables]->
+ getValue( (char*)attrName[count_attributes],
+ (char*)&readValue[count_attributes][0] );
+ } // for
+ break;
+ case otDelete: // Delete Case
+ break;
+ default:
+ assert(false);
+ break;
+ } // switch
+ if (check == -1 && tTmp == NULL && tMyOpType != otDelete) {
+ // Fatal error
+ tThreadResult = 5;
+ break;
+ } // if
+ } // for count_tables
+
+ // Only execute if everything is OK
+ if (tThreadResult != 0) {
+ // Close transaction (below)
+ // and continue with next count_round
+ count_round++;
+ tNoOfAttempts = 0;
+ } // if
+ else {
+ check = pMyTransaction->execute(Commit);
+ if (check == -1 ) {
+ const NdbError & err = pMyTransaction->getNdbError();
+
+ // Add complete error handling here
+
+ int retCode = flexHammerErrorData->handleErrorCommon(pMyTransaction->getNdbError());
+ if (retCode == 1) {
+ //if (strcmp(pMyTransaction->getNdbError().message, "Tuple did not exist") != 0 && strcmp(pMyTransaction->getNdbError().message,"Tuple already existed when attempting to insert") != 0) ndbout_c("execute: %s", pMyTransaction->getNdbError().message);
+
+ if (pMyTransaction->getNdbError().code != 626 && pMyTransaction->getNdbError().code != 630){
+ ndbout_c("Error code = %d", pMyTransaction->getNdbError().code);
+ ndbout_c("execute: %s", pMyTransaction->getNdbError().message);}
+
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexHammer" << endl;
+ } else if (retCode == 3) {
+// --------------------------------------------------------------------
+// We are not certain if the transaction was successful or not.
+// We must reexecute but might very well find that the transaction
+// actually was updated. Updates and Reads are no problem here. Inserts
+// will not cause a problem if error code 630 arrives. Deletes will
+// not cause a problem if 626 arrives.
+// --------------------------------------------------------------------
+ /* What can we do here? */
+ ndbout_c("execute: %s", pMyTransaction->getNdbError().message);
+ }//if(retCode == 3)
+ // End of adding complete error handling
+
+ switch( err.classification) {
+ case NdbError::ConstraintViolation: // Tuple already existed
+ count_round++;
+ tNoOfAttempts = 0;
+ break;
+ case NdbError::TimeoutExpired:
+ case NdbError::NodeRecoveryError:
+ case NdbError::TemporaryResourceError:
+ case NdbError::OverloadError:
+ if (tNoOfAttempts <= MaxNoOfAttemptsC) {
+ // Retry
+ tNoOfAttempts++;
+ } else {
+ // Too many retries, continue with next
+ count_round++;
+ tNoOfAttempts = 0;
+ } // else if
+ break;
+ // Fatal, just continue
+ default:
+ count_round++;
+ tNoOfAttempts = 0;
+ break;
+ } // switch
+ } // if
+ else {
+ // Execute commit was OK
+ // This is verifying read values
+ //switch (tMyOpType) {
+ //case otRead: // Read case
+ //for (j = 0; j < tNoOfAttributes; j++) {
+ //for(i = 1; i < tAttributeSize; i++) {
+ //if ( readValue[j][i] != attrValue[i]) {
+ //ndbout << "pkValue = " << pkValue << endl;
+ //ndbout << "readValue != attrValue" << endl;
+ //ndbout << readValue[j][i] << " != " << attrValue[i] << endl;
+ //} // if
+ // } // for
+ //} // for
+ //break;
+ //} // switch
+ count_round++;
+ tNoOfAttempts = 0;
+ } // else if
+ } // else if
+ pMyNdb->closeTransaction(pMyTransaction);
+ } // for count_round
+ } // for count
+ } // for (;;)
+
+ // Clean up
+ delete pMyNdb;
+ pMyNdb = NULL;
+
+ flexHammerErrorData->resetErrorCounters();
+
+ // And exit using NDBT
+ NdbThread_Exit(0);
+
+ return NULL;
+
+} // flexHammerThread
+
+
+int
+readArguments (int argc, const char** argv)
+{
+ int i = 1;
+
+ tNoOfThreads = 5; // Default Value
+ tNoOfOperations = 500; // Default Value
+ tNoOfRecords = 1; // Default Value
+ tNoOfLoops = 1; // Default Value
+ tNoOfAttributes = 25; // Default Value
+ tNoOfTables = 1; // Default Value
+ tNoOfBackups = 0; // Default Value
+ tAttributeSize = 1; // Default Value
+ theTableCreateFlag = 0;
+
+ while (argc > 1) {
+ if (strcmp(argv[i], "-t") == 0) {
+ tNoOfThreads = atoi(argv[i+1]);
+ if ((tNoOfThreads < 1) || (tNoOfThreads > MAXTHREADS))
+ return(1);
+ }
+ else if (strcmp(argv[i], "-o") == 0) {
+ tNoOfOperations = atoi(argv[i+1]);
+ if (tNoOfOperations < 1)
+ return(1);
+ }
+ else if (strcmp(argv[i], "-r") == 0) {
+ tNoOfRecords = atoi(argv[i+1]);
+ if (tNoOfRecords < 1)
+ return(1);
+ }
+ else if (strcmp(argv[i], "-a") == 0) {
+ tNoOfAttributes = atoi(argv[i+1]);
+ if ((tNoOfAttributes < 2) || (tNoOfAttributes > MAXATTR))
+ return(1);
+ }
+ else if (strcmp(argv[i], "-c") == 0) {
+ tNoOfTables = atoi(argv[i+1]);
+ if ((tNoOfTables < 1) || (tNoOfTables > MAXTABLES))
+ return(1);
+ }
+ else if (strcmp(argv[i], "-l") == 0) {
+ tNoOfLoops = atoi(argv[i+1]);
+ if ((tNoOfLoops < 0) || (tNoOfLoops > 100000))
+ return(1);
+ }
+ else if (strcmp(argv[i], "-s") == 0) {
+ tAttributeSize = atoi(argv[i+1]);
+ if ((tAttributeSize < 1) || (tAttributeSize > MAXATTRSIZE))
+ return(1);
+ }
+ else if (strcmp(argv[i], "-sleep") == 0) {
+ tSleepTime = atoi(argv[i+1]);
+ if ((tSleepTime < 1) || (tSleepTime > 3600))
+ exit(-1);
+ }
+ else if (strcmp(argv[i], "-simple") == 0) {
+ theSimpleFlag = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-write") == 0) {
+ theWriteFlag = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-dirty") == 0) {
+ theDirtyFlag = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-no_table_create") == 0) {
+ theTableCreateFlag = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-stdtables") == 0) {
+ theStandardTableNameFlag = 1;
+ argc++;
+ i--;
+ } // if
+ else {
+ return(1);
+ }
+
+ argc -= 2;
+ i = i + 2;
+ } // while
+
+ ndbout << endl << "FLEXHAMMER - Starting normal mode" << endl;
+ ndbout << "Hammer ndb with read, insert, update and delete transactions"<< endl << endl;
+
+ ndbout << " " << tNoOfThreads << " thread(s) " << endl;
+ ndbout << " " << tNoOfLoops << " iterations " << endl;
+ ndbout << " " << tNoOfTables << " table(s) and " << 1 << " operation(s) per transaction " << endl;
+ ndbout << " " << tNoOfRecords << " records to hammer(limit this with the -r option)" << endl;
+ ndbout << " " << tNoOfAttributes << " attributes per table " << endl;
+ ndbout << " " << tNoOfOperations << " transaction(s) per thread and round " << endl;
+ ndbout << " " << tAttributeSize << " is the number of 32 bit words per attribute " << endl << endl;
+ return 0;
+} // readArguments
+
+
+void sleepBeforeStartingTest(int seconds)
+{
+ if (seconds > 0) {
+ ndbout << "Sleeping(" << seconds << ")...";
+ NdbSleep_SecSleep(seconds);
+ ndbout << " done!" << endl;
+ } // if
+} // sleepBeforeStartingTest
+
+static int
+createTables(Ndb* pMyNdb)
+{
+ int i = 0;
+ int j = 0;
+ int check = 0;
+ NdbSchemaCon *MySchemaTransaction = NULL;
+ NdbSchemaOp *MySchemaOp = NULL;
+
+ // Create Table and Attributes.
+ if (theTableCreateFlag == 0) {
+
+ for (i = 0; i < tNoOfTables; i++) {
+
+ ndbout << "Creating " << tableName[i] << "...";
+ // Check if table exists already
+ const void * p = pMyNdb->getDictionary()->getTable(tableName[i]);
+ if (p != 0) {
+ ndbout << " already exists." << endl;
+ // Continue with next table at once
+ continue;
+ } // if
+ ndbout << endl;
+
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+ if (MySchemaTransaction == NULL) {
+ return(-1);
+ } // if
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if (MySchemaOp == NULL) {
+ // Clean up opened schema transaction
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ return(-1);
+ } // if
+
+ // Create tables, rest of parameters are default right now
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ check = MySchemaOp->createTable(tableName[i],
+ 8, // Table Size
+ TupleKey, // Key Type
+ 40, // Nr of Pages
+ All,
+ 6,
+ 78,
+ 80,
+ 1,
+ false);
+
+#else
+ check = MySchemaOp->createTable(tableName[i],
+ 8, // Table Size
+ TupleKey, // Key Type
+ 40); // Nr of Pages
+#endif
+ if (check == -1) {
+ // Clean up opened schema transaction
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ return(-1);
+ } // if
+
+ // Primary key
+ //ndbout << " pk " << (char*)&attrName[0] << "..." << endl;
+ check = MySchemaOp->createAttribute( (char*)attrName[0], TupleKey, 32,
+ 1, UnSigned, MMBased,
+ NotNullAttribute );
+ if (check == -1) {
+ // Clean up opened schema transaction
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ return(-1);
+ } // if
+
+ // Rest of attributes
+ for (j = 1; j < tNoOfAttributes ; j++) {
+ //ndbout << " " << (char*)attrName[j] << "..." << endl;
+ check = MySchemaOp->createAttribute( (char*)attrName[j], NoKey, 32,
+ tAttributeSize, UnSigned, MMBased,
+ NotNullAttribute );
+ if (check == -1) {
+ // Clean up opened schema transaction
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ return(-1);
+ } // if
+ } // for
+
+ // Execute creation
+ check = MySchemaTransaction->execute();
+ if (check == -1) {
+ // Clean up opened schema transaction
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ return(-1);
+ } // if
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ } // for
+ } // if
+
+ return(0);
+
+} // createTables
+
+
+static int setAttrNames()
+{
+ int i = 0;
+ int retVal = 0;
+
+ for (i = 0; i < MAXATTR ; i++) {
+ retVal = snprintf(attrName[i], MAXSTRLEN, "COL%d", i);
+ if (retVal < 0) {
+ // Error in conversion
+ return(-1);
+ } // if
+ } // for
+
+ return (0);
+} // setAttrNames
+
+static int setTableNames()
+{
+ // Note! Uses only uppercase letters in table name's
+ // so that we can look at the tables wits SQL
+ int i = 0;
+ int retVal = 0;
+
+ for (i = 0; i < MAXTABLES ; i++) {
+ if (theStandardTableNameFlag == 0) {
+ retVal = snprintf(tableName[i], MAXSTRLEN, "TAB%d_%d", i,
+ NdbTick_CurrentMillisecond()/1000);
+ } // if
+ else {
+ retVal = snprintf(tableName[i], MAXSTRLEN, "TAB%d", i);
+ } // else
+ if (retVal < 0) {
+ // Error in conversion
+ return(-1);
+ } // if
+ } // for
+
+ return(0);
+} // setTableNames
+
+static int checkThreadResults(ThreadNdb *threadArrayP, char* phase)
+{
+ int i = 0;
+
+ for (i = 0; i < tNoOfThreads; i++) {
+ if (threadArrayP[i].threadResult != 0) {
+ ndbout << "Thread " << i << " reported fatal error "
+ << threadArrayP[i].threadResult << " during " << phase << endl;
+ return(-1);
+ } // if
+ } // for
+
+ return(0);
+}
+
diff --git a/ndb/test/ndbapi/flexScan/Makefile b/ndb/test/ndbapi/flexScan/Makefile
new file mode 100644
index 00000000000..78f9d481063
--- /dev/null
+++ b/ndb/test/ndbapi/flexScan/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := flexScan
+
+SOURCES := flexScan.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/flexScan/README b/ndb/test/ndbapi/flexScan/README
new file mode 100644
index 00000000000..cddbdea5336
--- /dev/null
+++ b/ndb/test/ndbapi/flexScan/README
@@ -0,0 +1,66 @@
+
+Executing flexScan-tests automatically
+======================================
+
+
+It is possible to execute almost al the flexBench-tests
+automatically. The procedure contains three steps:
+- increase the number of attributes (flexScan -a number)
+- increase the size of attributes (flexScan -s number)
+- increase the number of threads (flexScan -t number)
+
+Each of these steps are performed by the scripts test1.sh
+test2.sh and test3.sh. Each test will start Ndb, execute
+the test and close Ndb again in order to execute each test
+in a 'clean' Ndb-environment. So make sure that there is
+no Ndb running when you start the test.
+
+
+1. Setup
+
+To perform the tests automatically, the following issues
+have to be taken care of:
+
+- be sure that you have a directory bin in your home-directory.
+ In this directory, you need to have a link 'runndb' to the
+ ndb executable. You can do this by executing a shell-command like:
+ ln -s ndb/Emulator/Main/ndb runndb
+ The script is not yet so far that it performs checks, so if
+ you forget about this, things will get messy.
+- In this directory you need a Ndb.cfg for a server-configuration.
+
+
+2. Command
+
+I assume you have Ndb and the API compiled or you use the
+'released' version. Compile flexScan as usual with 'make'.
+Now you can start the tests by typing 'make test'. The
+execution of the test will take a while.
+
+
+3. Results
+
+The scripts will write their results in the file report.txt.
+The scripts will start with a short summary on the test. Then
+it will add 1 line documenting each run of flexScan that is
+ececuted. Finally, it will print highest 'score'. The file
+report.txt is probably good enough to check in directly as
+testprotocol in ndb/test/docs/testprotocols.
+
+
+4. Log files.
+
+To make it possible to investigate errors, the output from
+the flexScan-run where the error occurred is stored in
+test1.log, test2.log or test3.log respectively. They are
+overwritten each time you start 'make test'.
+
+
+HINT
+
+The number of iterations in each test-script is not directly
+limited by the number of attributes or the size of the
+attributes but by the number of tables that you are allowed
+to create. Probably this will be the error that occurs if
+you execute the test. You migh adjust the begin-values and
+the step-size in the individual scripts if you want.
diff --git a/ndb/test/ndbapi/flexScan/flexScan.cpp b/ndb/test/ndbapi/flexScan/flexScan.cpp
new file mode 100644
index 00000000000..55163a99fbe
--- /dev/null
+++ b/ndb/test/ndbapi/flexScan/flexScan.cpp
@@ -0,0 +1,1677 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* ***************************************************
+ FLEXSCAN
+ Perform benchmark of:
+ insert
+ read
+ scan read
+ update
+ scan update
+ read
+ scan delete
+ verify delete
+
+ Arguments:
+ -f Location of Ndb.cfg file, default Ndb.cfg
+ -t Number of threads to start, default 1
+ -o Number of operations per loop, default 500 -l Number of loops to run, default 1, 0=infinite
+ -a Number of attributes, default 25
+ -c Number of tables, default 1
+ -s Size of each attribute, default 1
+ -stdtables Use standard table names
+ -no_table_create Don't create tables in db
+ -sleep Sleep a number of seconds before running the test, this
+ can be used so that another flexBench hav etome to create tables
+ -p Parallellism to use 1-32, default:1
+ -abort <number> Test scan abort after a number of tuples
+ -h Print help text
+ -no_scan_update Don't do scan updates
+ -no_scan_delete Don't do scan deletes
+
+ Returns:
+ NDBT_OK - Test passed
+ NDBT_FAILED - Test failed
+
+ Revision history:
+ 1.12 020222 epesson: Rewritten to use NDBT. Major bugs fixed
+
+ * *************************************************** */
+
+#include "NdbApi.hpp"
+
+#include <NdbThread.h>
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#include <NdbOut.hpp>
+#include <NdbTimer.hpp>
+#include <string.h>
+#include <assert.h>
+#include <NdbMain.h>
+#include <NdbTest.hpp>
+#include <NDBT_Error.hpp>
+#include <NdbStdio.h>
+
+#define PKSIZE 1
+#define FOREVER 1
+#define MAXSTRLEN 16
+#define MAXATTR 64
+#define MAXTABLES 64
+#define MAXTHREADS 256
+#define MAXATTRSIZE 64
+
+enum StartType {
+ stIdle,
+ stInsert,
+ stRead,
+ stScanRead,
+ stUpdate,
+ stScanUpdate,
+ stDelete,
+ stVerifyDelete,
+ stScanDelete,
+ stStop,
+ stLast} ;
+
+
+struct ThreadNdb
+{
+ int ThreadNo;
+ NdbThread* threadLife;
+ StartType threadStart;
+ int threadResult;
+ int threadReady;
+};
+
+extern "C" void* flexScanThread(void*);
+static int setAttrNames(void);
+static int setTableNames(void);
+static int createTables(Ndb* pMyNdb);
+static void sleepBeforeStartingTest(int seconds);
+static int readArguments(int argc, const char** argv);
+static void setAttrValues(int* attrValue,
+ int* readValue,
+ int Offset);
+static int insertRows(Ndb* pNdb, int* pkValue, int* attrValue, StartType tType);
+static int readRows(Ndb* pNdb, int* pkValue, int* readValue);
+static int deleteRows(Ndb* pNdb, int* pkValue);
+static int scanReadRows(Ndb* pNdb, int* readValue);
+static int scanUpdateRows(Ndb* pNdb, int* readValue, int* attrValue);
+static int scanDeleteRows(Ndb* pNdb, int* readValue);
+static int verifyDeleteRows(Ndb* pNdb, int* pkValue, int* readValue);
+static void Compare(int* attrValue, int* readValue);
+static void UpdateArray(int *attrValue);
+
+static int tNoOfThreads = 1;
+static int tNoOfAttributes = 25;
+static int tNoOfTables = 1;
+static int tAttributeSize = 1;
+static int tNodeId = 0;
+static int tNoOfOperations = 500;
+static int tNoOfLoops = 1;
+static int tAbortAfter = 0;
+static int tParallellism = 1;
+
+static char tableName[MAXTABLES][MAXSTRLEN];
+static char attrName[MAXATTR][MAXSTRLEN];
+
+static unsigned int tSleepTime = 0;
+
+static int theStdTableNameFlag = 0;
+static int theTableCreateFlag = 0;
+static int theScanAbortTestFlag = 0;
+static int theNoScanUpdateFlag = 0;
+static int theNoScanDeleteFlag = 0;
+
+//flexScanErrorData = new ErrorData;
+ErrorData * flexScanErrorData;
+NdbError * anerror;
+
+//static errorData theErrorData;
+//static unsigned int tErrorCounter[6000];
+
+#define START_TIMER { NdbTimer timer; timer.doStart();
+#define STOP_TIMER timer.doStop();
+#define PRINT_TIMER(text, trans, opertrans) timer.printTransactionStatistics(text, trans, opertrans); };
+
+static void UpdateArray(int *attrValue)
+{
+ int tableCount = 0;
+ int attrCount = 0;
+ int opCount = 0;
+ int sizeCount = 0;
+ int Index = 0;
+ int* pValue = attrValue;
+
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+ for (attrCount = 0; attrCount < tNoOfAttributes-1; attrCount++) {
+ for (opCount = 0; opCount < tNoOfOperations; opCount++) {
+ for (sizeCount = 0; sizeCount < tAttributeSize; sizeCount++) {
+ // Update value in array
+ (*pValue)++;
+ //ndbout << "attrValue[" << tableCount*tNoOfAttributes*tNoOfOperations*tAttributeSize +
+ //attrCount*tNoOfOperations*tAttributeSize + opCount*tAttributeSize + sizeCount <<
+ //"] = " << attrValue[tableCount*tNoOfAttributes*tNoOfOperations*tAttributeSize +
+ //attrCount*tNoOfOperations*tAttributeSize + opCount*tAttributeSize + sizeCount] << endl;
+ // Increment pointer
+ pValue++;
+ } // sizeCount
+ } // for opCount
+ } // for attrCount
+ } // for tableCount
+
+} // Update
+
+static void Compare(int* attrValue, int* readValue)
+{
+ int tableCount = 0;
+ int attrCount = 0;
+ int OpCount = 0;
+ int first = 0;
+ int sizeCount = 0;
+
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+ for (attrCount = 0; attrCount < tNoOfAttributes-1; attrCount++) {
+ for (OpCount = 0; OpCount < tNoOfOperations; OpCount++) {
+ if (memcmp(&(attrValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations +
+ attrCount*tNoOfOperations + OpCount]),
+ &(readValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations +
+ attrCount*tNoOfOperations + OpCount]),
+ tAttributeSize) != 0) {
+ // Values mismatch
+ if (first == 0) {
+ //ndbout << "Read and set values differ for:" << endl;
+ first = 1;
+ ndbout << "Mismatch found.";
+ } // if
+ // Comparision of values after scan update is meaningless right now
+ //ndbout << " table " << tableName[tableCount] <<
+ //" - attr " << attrName[attrCount+1];
+ //for (sizeCount = 0; sizeCount < tAttributeSize; sizeCount++) {
+ //ndbout << ": set " <<
+ //attrValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations*tAttributeSize +
+ //attrCount*tNoOfOperations*tAttributeSize +
+ //tNoOfOperations*tAttributeSize + sizeCount] << " read " <<
+ //readValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations*tAttributeSize +
+ //attrCount*tNoOfOperations*tAttributeSize +
+ //tNoOfOperations*tAttributeSize + sizeCount] << endl;
+ //} // for
+ } // if
+ } // for OpCount
+ } // for attrCount
+ } // for tableCount
+
+ // A final pretty-print
+ if (first == 1) {
+ ndbout << endl;
+ } // if
+} // Compare
+
+static void printInfo()
+{
+ ndbout << endl << "FLEXSCAN - Starting normal mode" << endl;
+ ndbout << "Perform benchmark of insert, update and delete transactions"<< endl;
+ ndbout << " NdbAPI node with id = " << tNodeId << endl;
+ ndbout << " " << tNoOfThreads << " thread(s) " << endl;
+ ndbout << " " << tNoOfLoops << " iterations " << endl;
+ ndbout << " " << tNoOfTables << " table(s) and " << 1 << " operation(s) per transaction "
+ << endl;
+ ndbout << " " << tNoOfAttributes << " attributes per table incl. pk" << endl;
+ ndbout << " " << tNoOfOperations << " transaction(s) per thread and round " << endl;
+ if (theScanAbortTestFlag == 1) {
+ ndbout << " Scan abort test after " << tAbortAfter << " tuples" << endl;
+ } // if
+ ndbout << " " << tParallellism << " parallellism in scans" << endl;
+ ndbout << " " << tAttributeSize << " is the number of 32 bit words per attribute " <<
+ endl << endl;
+
+} // printInfo
+
+static void tellThreads(ThreadNdb *threadArrayP, StartType what)
+{
+ int i = 0;
+
+ for (i = 0; i < tNoOfThreads ; i++)
+ threadArrayP[i].threadStart = what;
+} // tellThreads
+
+static void waitForThreads(ThreadNdb *threadArrayP)
+{
+ int i = 0;
+ int cont = 1;
+
+ while (cont == 1){
+
+ NdbSleep_MilliSleep(10);
+ cont = 0;
+
+ for (i = 0; i < tNoOfThreads ; i++) {
+ if (threadArrayP[i].threadReady == 0) {
+// ndbout << "Main is reporting thread " << i << " not ready" << endl;
+ cont = 1;
+ } // if
+ } // for
+ } // while
+} // waitForThreads
+
+
+static void resetThreads(ThreadNdb *threadArrayP)
+{
+ int i = 0;
+
+ for (i = 0; i < tNoOfThreads ; i++) {
+ threadArrayP[i].threadReady = 0;
+ threadArrayP[i].threadResult = 0;
+ threadArrayP[i].threadStart = stIdle;
+ //ndbout << "threadStart[" << i << "]=" <<
+ //threadArrayP[i].threadStart << endl;
+ } // for
+} // resetThreads
+
+static int checkThreadResults(ThreadNdb *threadArrayP, char *action)
+{
+ int i = 0;
+ int retValue = 0;
+
+ for (i = 0; i < tNoOfThreads; i++) {
+ if (threadArrayP[i].threadResult != 0) {
+ ndbout << "Thread " << i << " reported fatal error "
+ << threadArrayP[i].threadResult << " during " << action << endl;
+ retValue = -1;
+ break;
+ } // if
+ } // for
+
+ return(retValue);
+} // checkThreadResults
+
+NDB_COMMAND(flexScan, "flexScan", "flexScan", "flexScan", 65535)
+{
+ ThreadNdb* pThreads = NULL;
+ Ndb* pMyNdb = NULL;
+ int tLoops = 0;
+ int check = 0;
+ int returnValue = NDBT_OK;
+ int every2ndScanDelete = 0; // Switch between scan delete and normal delete
+
+ flexScanErrorData = new ErrorData;
+
+ flexScanErrorData->resetErrorCounters();
+
+ if (readArguments(argc, argv) != 0) {
+ ndbout << "Wrong arguments to flexScan" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ } // if
+
+ /* print Setting */
+ flexScanErrorData->printSettings(ndbout);
+
+ check = setAttrNames();
+ if (check != 0) {
+ ndbout << "Couldn't set attribute names" << endl;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ } // if
+ check = setTableNames();
+ if (check != 0) {
+ ndbout << "Couldn't set table names" << endl;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ } // if
+
+ pMyNdb = new Ndb ("TEST_DB");
+ pMyNdb->init();
+ tNodeId = pMyNdb->getNodeId();
+
+ printInfo();
+
+ NdbThread_SetConcurrencyLevel(tNoOfThreads + 2);
+ //NdbThread_SetConcurrencyLevel(tNoOfThreads + 8);
+
+ pThreads = new ThreadNdb[tNoOfThreads];
+
+ if (pMyNdb->waitUntilReady(10000) != 0) {
+ ndbout << "NDB is not ready" << endl << "Benchmark failed" << endl;
+ returnValue = NDBT_FAILED;
+ } // if
+
+ else {
+
+ if (createTables(pMyNdb) != 0) {
+ ndbout << "Could not create tables" << endl;
+ returnValue = NDBT_FAILED;
+ } // if
+ else {
+ sleepBeforeStartingTest(tSleepTime);
+
+ resetThreads(pThreads);
+ // Create threads
+ for (int i = 0; i < tNoOfThreads ; i++){
+ pThreads[i].ThreadNo = i;
+ // Ignore the case that thread creation may fail
+ pThreads[i].threadLife = NdbThread_Create(flexScanThread,
+ (void**)&pThreads[i],
+ 327680,
+ "flexScanThread", NDB_THREAD_PRIO_LOW);
+ if (pThreads[i].threadLife == NULL) {
+ ndbout << "Could not create thread " << i << endl;
+ returnValue = NDBT_FAILED;
+ // Use the number of threads that were actually created
+ tNoOfThreads = i;
+ break; // break for loop
+ } // if
+ } // for
+
+ waitForThreads(pThreads);
+ if (checkThreadResults(pThreads, "init") != 0) {
+ returnValue = NDBT_FAILED;
+ } // if
+
+ if (returnValue == NDBT_OK) {
+ ndbout << "All threads started" << endl;
+
+ while (FOREVER) {
+
+ resetThreads(pThreads);
+
+ if ((tNoOfLoops != 0) && (tNoOfLoops <= tLoops)) {
+ break;
+ } // if
+
+ // Insert
+ START_TIMER;
+
+ tellThreads(pThreads, stInsert);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "insert") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("insert", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ resetThreads(pThreads);
+
+ // Read
+ START_TIMER;
+
+ tellThreads(pThreads, stRead);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "read") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("read", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ resetThreads(pThreads);
+
+ // Update
+ START_TIMER;
+
+ tellThreads(pThreads, stUpdate);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "update") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("update", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ resetThreads(pThreads);
+
+ // Scan read
+ START_TIMER;
+
+ tellThreads(pThreads, stScanRead);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "scanread") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("scanread", tNoOfTables*tNoOfThreads, 1);
+
+ resetThreads(pThreads);
+
+ // Update
+ START_TIMER;
+
+ tellThreads(pThreads, stUpdate);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "update") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("update", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ resetThreads(pThreads);
+
+ // Read
+ START_TIMER;
+
+ tellThreads(pThreads, stRead);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "read") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("read", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ resetThreads(pThreads);
+
+ // Only do scan update if told to do so
+ if (theNoScanUpdateFlag == 0) {
+ // Scan update
+ START_TIMER;
+
+ tellThreads(pThreads, stScanUpdate);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "scanupdate") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("scanupdate", tNoOfTables*tNoOfThreads, 1);
+
+ resetThreads(pThreads);
+
+ // Read
+ START_TIMER;
+
+ tellThreads(pThreads, stRead);
+ // tellThreads(pThreads, stScanRead);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "read") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("read", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ resetThreads(pThreads);
+ } // if theNoScanUpdateFlag
+
+ // Shift between delete and scan delete
+ if ((every2ndScanDelete % 2 == 0) || (theNoScanDeleteFlag == 1)){
+ // Delete
+ START_TIMER;
+ tellThreads(pThreads, stDelete);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "delete") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("delete", tNoOfOperations*tNoOfThreads, tNoOfTables);
+ resetThreads(pThreads);
+ } // if
+ else {
+ resetThreads(pThreads); // Scan delete
+ START_TIMER;
+ tellThreads(pThreads, stScanDelete);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "scandelete") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("scandelete", tNoOfTables*tNoOfThreads, 1);
+
+ resetThreads(pThreads);
+ } // else
+ every2ndScanDelete++;
+
+ resetThreads(pThreads); // Verify delete
+ START_TIMER;
+ tellThreads(pThreads, stVerifyDelete);
+ waitForThreads(pThreads);
+
+ STOP_TIMER;
+ if (checkThreadResults(pThreads, "verifydelete") != 0) {
+ returnValue = NDBT_FAILED;
+ break;
+ } // if
+ PRINT_TIMER("verifydelete", tNoOfOperations*tNoOfThreads*tNoOfTables, 1);
+
+ resetThreads(pThreads);
+
+ ndbout << "--------------------------------------------------" << endl;
+ tLoops++;
+
+ } // while
+ } // if
+ } // else
+ } // else
+
+ // Stop threads in a nice way
+ tellThreads(pThreads, stStop);
+ waitForThreads(pThreads);
+
+ // Clean up
+ delete [] pThreads;
+ delete pMyNdb;
+
+ flexScanErrorData->printErrorCounters(ndbout);
+
+ if (returnValue == NDBT_OK) {
+ ndbout << endl << "Benchmark completed successfully" << endl;
+ } // if
+ else {
+ ndbout << endl << "Benchmark failed" << endl;
+ } // else
+
+ // Exit via NDBT
+ return NDBT_ProgramExit(returnValue);;
+} // main
+
+void*
+flexScanThread(void* ThreadData)
+{
+ ThreadNdb* pThreadData = (ThreadNdb*)ThreadData;
+ unsigned int thread_no = pThreadData->ThreadNo;
+ unsigned int thread_base = (thread_no * 2000000) + (tNodeId * 26000);
+ int NrOfScannedRecords = 0;
+ int tThreadResult = 0;
+ Ndb* MyNdb = NULL;
+ NdbConnection *MyTransaction = NULL;
+ NdbOperation* MyOperation[MAXTABLES];
+ int check = 0;
+ StartType tType = stLast;
+ int* pkValue = NULL;
+ int* attrValue = NULL;
+ int* readValue = NULL;
+ int AllocSize = 0;
+ NdbRecAttr* tTmp = NULL;
+ OperationType opType;
+
+ AllocSize = tNoOfTables * (tNoOfAttributes-1) * tNoOfOperations *
+ tAttributeSize * sizeof(int);
+ attrValue = (int*)malloc(AllocSize);
+ readValue = (int*)malloc(AllocSize);
+ pkValue = (int*)malloc(tNoOfOperations * sizeof(int));
+ if ((attrValue == NULL) || (readValue == NULL) || (pkValue == NULL)) {
+ tThreadResult = 98;
+ pThreadData->threadStart = stIdle;
+ } // if
+
+ setAttrValues(attrValue, readValue, thread_base);
+
+ MyNdb = new Ndb( "TEST_DB" );
+ MyNdb->init();
+ if (MyNdb->waitUntilReady(10000) != 0) {
+ tThreadResult = 99;
+ pThreadData->threadStart = stIdle;
+ } // if
+
+ // Set primary key value, same for all tables
+ for (int c = 0; c < tNoOfOperations; c++) {
+ pkValue[c] = (int)(c + thread_base);
+ } // for
+
+ while (FOREVER) {
+ pThreadData->threadResult = tThreadResult;
+ pThreadData->threadReady = 1;
+
+ while (pThreadData->threadStart == stIdle) {
+ NdbSleep_MilliSleep(10);
+ } // while
+
+ // Check if signal to exit is received
+ if (pThreadData->threadStart >= stStop){
+ pThreadData->threadReady = 1;
+ break;
+ } // if
+ tType = pThreadData->threadStart;
+ pThreadData->threadStart = stIdle;
+
+ switch (tType) {
+ case stInsert:
+ check = insertRows(MyNdb, pkValue, attrValue, tType);
+ break;
+ case stRead:
+ check = readRows(MyNdb, pkValue, readValue);
+ Compare(attrValue, readValue);
+ break;
+ case stUpdate:
+ UpdateArray(attrValue);
+ check = insertRows(MyNdb, pkValue, attrValue, tType);
+ break;
+ case stScanRead:
+ //check = readRows(MyNdb, pkValue, readValue);
+ check = scanReadRows(MyNdb, readValue);
+ Compare(attrValue, readValue);
+ break;
+ case stScanUpdate:
+ UpdateArray(attrValue);
+ //tType = stUpdate;
+ //check = insertRows(MyNdb, pkValue, attrValue, tType);
+ check = scanUpdateRows(MyNdb, readValue, attrValue);
+ break;
+ case stDelete:
+ check = deleteRows(MyNdb, pkValue);
+ break;
+ case stScanDelete:
+ check = scanDeleteRows(MyNdb, readValue);
+ break;
+ case stVerifyDelete:
+ check = verifyDeleteRows(MyNdb, pkValue, readValue);
+ break;
+ default:
+ ndbout << "tType is " << tType << endl;
+ assert(false);
+ break;
+ } // switch
+
+ tThreadResult = check;
+
+ if (tThreadResult != 0) {
+ // Check if error is fatak or not
+ } // if
+ else {
+ continue;
+ } // else
+ } // while
+
+ // Clean up
+ delete MyNdb;
+ if (attrValue != NULL) {
+ free(attrValue);
+ } //if
+ if (readValue != NULL) {
+ free(readValue);
+ } // if
+ if (pkValue != NULL) {
+ free(pkValue);
+ } // if
+
+ NdbThread_Exit(0);
+ return NULL;
+
+} // flexScanThread
+
+
+static int setAttrNames()
+{
+ int i = 0;
+ int retVal = 0;
+
+ for (i = 0; i < MAXATTR ; i++) {
+ retVal = snprintf(attrName[i], MAXSTRLEN, "COL%d", i);
+ if (retVal < 0) {
+ return(-1);
+ } // if
+ } // for
+
+ return(0);
+} // setAttrNames
+
+
+static int setTableNames()
+{
+ // Note! Uses only uppercase letters in table name's
+ // so that we can look at the tables with SQL
+ int i = 0;
+ int retVal = 0;
+
+ for (i = 0; i < MAXTABLES ; i++) {
+
+ if (theStdTableNameFlag == 0) {
+ retVal = snprintf(tableName[i], MAXSTRLEN, "TAB%d_%d", i,
+ (int)(NdbTick_CurrentMillisecond() / 1000));
+ } // if
+ else {
+ retVal = snprintf(tableName[i], MAXSTRLEN, "TAB%d", i);
+ } // if else
+
+ if (retVal < 0) {
+ return(-1);
+ } // if
+ } // for
+
+ return(0);
+} // setTableNames
+
+
+// Create Table and Attributes.
+static int createTables(Ndb* pMyNdb)
+{
+
+ NdbSchemaCon *MySchemaTransaction = NULL;
+ NdbSchemaOp *MySchemaOp = NULL;
+ int i = 0;
+ int j = 0;
+ int check = 0;
+
+ if (theTableCreateFlag == 0) {
+
+ i = 0;
+ do {
+ i++;
+ ndbout << endl << "Creating " << tableName[i - 1] << "..." << endl;
+
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+ if( MySchemaTransaction == NULL ) {
+ return (-1);
+ } // if
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL ) {
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ return (-1);
+ } // if
+
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ check = MySchemaOp->createTable(tableName[i - 1],
+ 8, // Table Size
+ TupleKey, // Key Type
+ 40, // Nr of Pages
+ All,
+ 6,
+ 78,
+ 80,
+ 1,
+ false);
+#else
+ check = MySchemaOp->createTable(tableName[i - 1]
+ ,8 // Table Size
+ ,TupleKey // Key Type
+ ,40); // Nr of Pages
+#endif
+ if (check == -1) {
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ return -1;
+ } // if
+
+ check = MySchemaOp->createAttribute( (char*)attrName[0], TupleKey, 32, PKSIZE,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) {
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ return -1;
+ } // if
+
+ for (j = 1; j < tNoOfAttributes ; j++) {
+ check = MySchemaOp->createAttribute( (char*)attrName[j], NoKey, 32, tAttributeSize,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) {
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ return -1;
+ } // if
+ } // for
+
+ if (MySchemaTransaction->execute() == -1) {
+ ndbout << MySchemaTransaction->getNdbError().message << endl;
+ ndbout << "Probably, " << tableName[i - 1] << " already exist" << endl;
+ } // if
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ } while (tNoOfTables > i);
+ }
+
+ return 0;
+} // createTables
+
+static void printUsage()
+{
+ ndbout << "Usage of flexScan:" << endl;
+ ndbout << "-f <path> Location of Ndb.cfg file, default: Ndb.cfg" << endl;
+ ndbout << "-t <int> Number of threads to start, default 1" << endl;
+ ndbout << "-o <int> Number of operations per loop, default 500" << endl;
+ ndbout << "-l <int> Number of loops to run, default 1, 0=infinite" << endl;
+ ndbout << "-a <int> Number of attributes, default 25" << endl;
+ ndbout << "-c <int> Number of tables, default 1" << endl;
+ ndbout << "-s <int> Size of each attribute, default 1" << endl;
+ ndbout << "-stdtables Use standard table names" << endl;
+ ndbout << "-no_table_create Don't create tables in db" << endl;
+ ndbout << "-sleep <int> Sleep a number of seconds before running the test" << endl;
+ ndbout << "-p <int> Parallellism to use 1-32, default:1" << endl;
+ ndbout << "-abort <int> Test scan abort after a number of tuples" << endl;
+ ndbout << "-no_scan_update Don't do scan updates" << endl;
+ ndbout << "-no_scan_delete Don't do scan deletes" << endl;
+ ndbout << "-h Print this text" << endl;
+ // inputErrorArg();
+ flexScanErrorData->printCmdLineArgs(ndbout);
+}
+
+static int readArguments(int argc, const char** argv)
+{
+ int i = 1;
+ int retValue = 0;
+ int printFlag = 0;
+
+ tNoOfThreads = 1; // Set default Value
+ tNoOfTables = 1; // Default Value
+
+ while (argc > 1) {
+ if (strcmp(argv[i], "-t") == 0) {
+ if (argv[i + 1] != NULL) {
+ tNoOfThreads = atoi(argv[i + 1]);
+ if ((tNoOfThreads < 1) || (tNoOfThreads > MAXTHREADS)) {
+ retValue = -1;
+ } // if
+ } // if
+ else {
+ retValue = -1;
+ } // else
+ } // if
+ else if (strcmp(argv[i], "-o") == 0) {
+ if (argv[i + 1] != NULL) {
+ tNoOfOperations = atoi(argv[i + 1]);
+ if (tNoOfOperations < 1) {
+ retValue = -1;
+ } // if
+ } // if
+ else {
+ retValue = -1;
+ } // else
+ } // else if
+ else if (strcmp(argv[i], "-a") == 0) {
+ if (argv[i + 1] != NULL) {
+ tNoOfAttributes = atoi(argv[i + 1]);
+ if ((tNoOfAttributes < 2) || (tNoOfAttributes > MAXATTR)) {
+ retValue = -1;
+ } // if
+ } // if
+ else {
+ retValue = -1;
+ } // else
+ } // else if
+ else if (strcmp(argv[i], "-c") == 0) {
+ if (argv[i + 1] != NULL) {
+ tNoOfTables = atoi(argv[i+1]);
+ if ((tNoOfTables < 1) || (tNoOfTables > MAXTABLES)) {
+ retValue = -1;
+ } // if
+ } // if
+ else {
+ retValue = -1;
+ } // else
+ } // else if
+ else if (strcmp(argv[i], "-l") == 0) {
+ if (argv[i + 1] != NULL) {
+ tNoOfLoops = atoi(argv[i+1]);
+ if ((tNoOfLoops < 0) || (tNoOfLoops > 100000)) {
+ retValue = -1;
+ } // if
+ } // if
+ else {
+ retValue = -1;
+ } // else
+ } // else if
+ else if (strcmp(argv[i], "-s") == 0) {
+ if (argv[i + 1] != NULL) {
+ tAttributeSize = atoi(argv[i+1]);
+ if ((tAttributeSize < 1) || (tAttributeSize > MAXATTRSIZE)) {
+ retValue = -1;
+ } // if
+ } // if
+ else {
+ retValue = -1;
+ } // else
+ } // else if
+ else if (strcmp(argv[i], "-no_table_create") == 0) {
+ theTableCreateFlag = 1;
+ argc++;
+ i--;
+ } // else if
+ else if (strcmp(argv[i], "-stdtables") == 0) {
+ theStdTableNameFlag = 1;
+ argc++;
+ i--;
+ } // else if
+ else if (strcmp(argv[i], "-sleep") == 0) {
+ if (argv[i + 1] != NULL) {
+ tSleepTime = atoi(argv[i+1]);
+ if ((tSleepTime < 1) || (tSleepTime > 3600)) {
+ retValue = -1;
+ } // if
+ } // if
+ else {
+ retValue = -1;
+ } // else
+ } // else if
+ else if (strcmp(argv[i], "-abort") == 0) {
+ // Test scan abort after a number of tuples
+ theScanAbortTestFlag = 1;
+ if (argv[i + 1] != NULL) {
+ tAbortAfter = atoi(argv[i + 1]);
+ } // if
+ else {
+ retValue = -1;
+ } // else
+ } // else if
+ else if (strcmp(argv[i], "-p") == 0) {
+ if (argv[i + 1] != NULL) {
+ tParallellism = atoi(argv[i + 1]);
+ if ((tParallellism < 1) || (tParallellism > 32)) {
+ retValue = -1;
+ } // if
+ } // if
+ else {
+ retValue = -1;
+ } // else
+ } // else if
+ else if (strcmp(argv[i], "-h") == 0) {
+ printFlag = 1;
+ argc++;
+ i--;
+ } // else if
+ else if (strcmp(argv[i], "-no_scan_update") == 0) {
+ theNoScanUpdateFlag = 1;
+ argc++;
+ i--;
+ } // else if
+ else if (strcmp(argv[i], "-no_scan_delete") == 0) {
+ theNoScanDeleteFlag = 1;
+ argc++;
+ i--;
+ } // else if
+ else {
+ retValue = -1;
+ } // else
+
+ argc -= 2;
+ i = i + 2;
+ }
+
+ if ((retValue != 0) || (printFlag == 1)) {
+ printUsage();
+ } // if
+
+ return(retValue);
+
+} // readArguments
+
+static void sleepBeforeStartingTest(int seconds)
+{
+ if (seconds > 0) {
+ ndbout << "Sleeping(" <<seconds << ")...";
+ NdbSleep_SecSleep(seconds);
+ ndbout << " done!" << endl;
+ } // if
+} // sleepBeforeStartingTest
+
+static void setAttrValues(int* attrValue,
+ int* readValue,
+ int Offset)
+{
+ int tableCount = 0;
+ int attrCount = 0;
+ int OpCount = 0;
+ int attrSize = 0;
+ int* pAttr = NULL;
+ int* pRead = NULL;
+
+ // Set attribute values in memory array
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+ for (attrCount = 0; attrCount < tNoOfAttributes-1; attrCount++) {
+ for (OpCount = 0; OpCount < tNoOfOperations; OpCount++) {
+ pAttr = &(attrValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations +
+ attrCount*tNoOfOperations + OpCount]);
+ pRead = &(readValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations +
+ attrCount*tNoOfOperations + OpCount]);
+ for (attrSize = 0; attrSize < tAttributeSize; attrSize++){
+ *pAttr = (int)(Offset + tableCount + attrCount + OpCount + attrSize);
+ //ndbout << "attrValue[" << tableCount*(tNoOfAttributes-1)*tNoOfOperations +
+ //attrCount*tNoOfOperations + OpCount + attrSize << "] = " <<
+ //attrValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations +
+ //attrCount*tNoOfOperations + OpCount + attrSize] << endl;
+ *pRead = 0;
+ pAttr++;
+ pRead++;
+ } // for attrSize
+ } // for OpCount
+ } // for attrCount
+ } // for tableCount
+
+} // setAttrValues
+
+static int insertRows(Ndb* pNdb, // NDB object
+ int* pkValue, // Primary key values
+ int* attrValue, // Attribute values
+ StartType tType)
+{
+ int tResult = 0;
+ int check = 0;
+ int tableCount = 0;
+ int attrCount = 0;
+ NdbConnection* MyTransaction = NULL;
+ NdbOperation* MyOperations[MAXTABLES] = {NULL};
+ int Index = 0;
+ int opCount = 0;
+
+ for (opCount = 0; opCount < tNoOfOperations; opCount++) {
+ MyTransaction = pNdb->startTransaction();
+ if (MyTransaction == NULL) {
+ tResult = 1;
+ } // if
+ else {
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+
+ MyOperations[tableCount] =
+ MyTransaction->getNdbOperation(tableName[tableCount]);
+ if (MyOperations[tableCount] == NULL) {
+ tResult = 2;
+ // Break for tableCount loop
+ break;
+ } // if
+
+ if (tType == stUpdate) {
+ check = MyOperations[tableCount]->updateTuple();
+ } // if
+ else if (tType == stInsert) {
+ check = MyOperations[tableCount]->insertTuple();
+ } // else if
+ else {
+ assert(false);
+ } // else
+
+ if (check == -1) {
+ tResult = 3;
+ break;
+ } // if
+ check = MyOperations[tableCount]->equal((char*)attrName[0],
+ (char*)&(pkValue[opCount]));
+ if (check == -1) {
+ tResult = 7;
+ break;
+ } // if
+
+ for (attrCount = 0; attrCount < tNoOfAttributes - 1; attrCount++) {
+ Index = tableCount * (tNoOfAttributes - 1) * tNoOfOperations * tAttributeSize +
+ attrCount * tNoOfOperations * tAttributeSize + opCount * tAttributeSize;
+ check = MyOperations[tableCount]->
+ setValue((char*)attrName[attrCount + 1],
+ (char*)&(attrValue[Index]));
+ if (check == -1) {
+ tResult = 8;
+ break; // break attrCount loop
+ } // if
+ } // for
+ } // for tableCount
+
+ // Execute transaction with insert one tuple in every table
+ check = MyTransaction->execute(Commit);
+ if (check == -1) {
+ ndbout << MyTransaction->getNdbError().message << endl;
+
+ // Add complete error handling here
+
+ int retCode = flexScanErrorData->handleErrorCommon(MyTransaction->getNdbError());
+ if (retCode == 1) {
+ if (MyTransaction->getNdbError().code != 626 && MyTransaction->getNdbError().code != 630){
+ ndbout_c("execute: %d, %d, %s", opCount, tType, MyTransaction->getNdbError().message);
+ ndbout_c("Error code = %d", MyTransaction->getNdbError().code);}
+ tResult = 20;
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexBench" << endl;
+ tResult = 20;
+ } else if (retCode == 3) {
+// --------------------------------------------------------------------
+// We are not certain if the transaction was successful or not.
+// We must reexecute but might very well find that the transaction
+// actually was updated. Updates and Reads are no problem here. Inserts
+// will not cause a problem if error code 630 arrives. Deletes will
+// not cause a problem if 626 arrives.
+// --------------------------------------------------------------------
+ /* What can we do here? */
+ ndbout_c("execute: %s", MyTransaction->getNdbError().message);
+ }//if(retCode == 3)
+
+ } // if(check == -1)
+
+ pNdb->closeTransaction(MyTransaction);
+ } // else
+ } // for opCount
+
+ return(tResult);
+} // insertRows
+
+static int readRows(Ndb* pNdb,
+ int* pkValue,
+ int* readValue)
+{
+ int tResult = 0;
+ int tableCount = 0;
+ int attrCount = 0;
+ int check = 0;
+ NdbConnection* MyTransaction = NULL;
+ NdbOperation* MyOperations[MAXTABLES] = {NULL};
+ NdbRecAttr* tmp = NULL;
+ int Value = 0;
+ int Index = 0;
+ int opCount = 0;
+
+ for (opCount = 0; opCount < tNoOfOperations; opCount++) {
+ MyTransaction = pNdb->startTransaction();
+ if (MyTransaction == NULL) {
+ tResult = 1;
+ } // if
+ else {
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+
+ MyOperations[tableCount] =
+ MyTransaction->getNdbOperation(tableName[tableCount]);
+ if (MyOperations[tableCount] == NULL) {
+ tResult = 2;
+ // Break for tableCount loop
+ break;
+ } // if
+
+ check = MyOperations[tableCount]->readTuple();
+ if (check == -1) {
+ tResult = 3;
+ break;
+ } // if
+
+ check = MyOperations[tableCount]->
+ equal((char*)attrName[0], (char*)&(pkValue[opCount]));
+ if (check == -1) {
+ tResult = 7;
+ break;
+ } // if
+
+ for (int attrCount = 0; attrCount < tNoOfAttributes - 1; attrCount++) {
+ Index = tableCount * (tNoOfAttributes - 1) * tNoOfOperations * tAttributeSize +
+ attrCount * tNoOfOperations * tAttributeSize + opCount * tAttributeSize;
+ tmp = MyOperations[tableCount]->
+ getValue((char*)attrName[attrCount + 1], (char*)&(readValue[Index]));
+
+ if (tmp == NULL) {
+ tResult = 9;
+ break;
+ } // if
+ } // for attrCount
+ } // for tableCount
+ // Execute transaction reading one tuple in every table
+ check = MyTransaction->execute(Commit);
+ if (check == -1) {
+ ndbout << MyTransaction->getNdbError().message << endl;
+
+ // Add complete error handling here
+
+ int retCode = flexScanErrorData->handleErrorCommon(MyTransaction->getNdbError());
+ if (retCode == 1) {
+ if (MyTransaction->getNdbError().code != 626 && MyTransaction->getNdbError().code != 630){
+ ndbout_c("execute: %d, %s", opCount, MyTransaction ->getNdbError().message );
+ ndbout_c("Error code = %d", MyTransaction->getNdbError().code );}
+ tResult = 20;
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexBench" << endl;
+ tResult = 20;
+ } else if (retCode == 3) {
+// --------------------------------------------------------------------
+// We are not certain if the transaction was successful or not.
+// We must reexecute but might very well find that the transaction
+// actually was updated. Updates and Reads are no problem here. Inserts
+// will not cause a problem if error code 630 arrives. Deletes will
+// not cause a problem if 626 arrives.
+// --------------------------------------------------------------------
+ /* What can we do here? */
+ ndbout_c("execute: %s", MyTransaction ->getNdbError().message );
+ }//if(retCode == 3)
+
+ } // if
+
+ pNdb->closeTransaction(MyTransaction);
+ } // else
+ } // for opCount
+
+ return(tResult);
+} // readRows
+
+static int scanReadRows(Ndb* pNdb, int* readValue)
+{
+ int tResult = 0;
+ int tableCount = 0;
+ int attrCount = 0;
+ int check = 0;
+ int countAbort = 0; // Counts loops until scan abort if requested
+ NdbConnection* MyTransaction = NULL;
+ NdbOperation* MyOperation = NULL;
+ NdbRecAttr* tmp = NULL;
+
+
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+ MyTransaction = pNdb->startTransaction();
+ if (MyTransaction == NULL) {
+ tResult = 1;
+ break;
+ } // if
+ MyOperation = MyTransaction->getNdbOperation(tableName[tableCount]);
+ if (MyOperation == NULL) {
+ tResult = 2;
+ break;
+ } // if
+
+ check = MyOperation->openScanRead(tParallellism);
+ if (check == -1) {
+ tResult = 10;
+ break;
+ } // if
+
+ for (int attrCount = 0; attrCount < tNoOfAttributes-1; attrCount++) {
+ // Get all attributes
+ tmp = MyOperation->
+ getValue((char*)attrName[attrCount+1],
+ (char*)&(readValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations*tAttributeSize +
+ attrCount*tNoOfOperations*tAttributeSize]));
+ if (tmp == NULL) {
+ tResult = 9;
+ break;
+ } // if
+ } // for attrCount
+
+ check = MyTransaction->executeScan();
+ if (check == -1) {
+ tResult = 12;
+ break;
+ } // if
+
+ check = MyTransaction->nextScanResult();
+ while (check == 0) {
+ // Check if scan abort is requested
+ if (theScanAbortTestFlag == 1) {
+ if (countAbort == tAbortAfter) {
+ MyTransaction->stopScan();
+ ndbout << "scanread aborted on request after " << countAbort*tParallellism <<
+ " tuples" << endl;
+ break; // break while loop
+ } // if
+ countAbort++;
+ } // if
+ check = MyTransaction->nextScanResult();
+ } // while
+
+ pNdb->closeTransaction(MyTransaction);
+ } // for tableCount
+
+ return(tResult);
+} // scanReadRows
+
+static int scanUpdateRows(Ndb* pNdb,
+ int* readValue,
+ int* attrValue)
+{
+ int tResult = 0;
+ int tableCount = 0;
+ int attrCount = 0;
+ int check = 0;
+ int opCount = 0;
+ NdbConnection* MyTransaction = NULL;
+ NdbOperation* MyOperation = NULL;
+ NdbConnection* MyTakeOverTrans = NULL;
+ NdbOperation* MyTakeOverOp = NULL;
+ NdbRecAttr* tTmp = NULL;
+
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+ MyTransaction = pNdb->startTransaction();
+ if (MyTransaction == NULL) {
+ tResult = 1;
+ break; // break tableCount for loop
+ } // if
+ MyOperation = MyTransaction->getNdbOperation(tableName[tableCount]);
+ if (MyOperation == NULL) {
+ tResult = 2;
+ break;
+ } // if
+
+ check = MyOperation->openScanExclusive(tParallellism);
+ if (check == -1) {
+ tResult = 11;
+ break;
+ } // if
+
+ MyOperation->interpret_exit_ok();
+ // Fetch all attributes
+ for (int attrCount = 0; attrCount < tNoOfAttributes-1; attrCount++) {
+ tTmp = MyOperation->
+ getValue((char*)attrName[attrCount+1],
+ (char*)&(readValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations*tAttributeSize +
+ attrCount*tNoOfOperations*tAttributeSize]));
+ if (tTmp == NULL) {
+ tResult = 9;
+ break; // break for loop
+ } // if
+ } // for
+ if (tResult != 0) {
+ break; // break while loop also
+ } // if
+
+ check = MyTransaction->executeScan();
+ if (check == -1) {
+ tResult = 12;
+ break;
+ } // if
+ check = MyTransaction->nextScanResult();
+ opCount = 0;
+ while (check == 0) {
+ MyTakeOverTrans = pNdb->startTransaction();
+ MyTakeOverOp = MyOperation->takeOverForUpdate(MyTakeOverTrans);
+ for (attrCount = 0; attrCount < tNoOfAttributes-1; attrCount++) {
+ check = MyTakeOverOp->setValue((char*)attrName[attrCount+1],
+ (char*)&(attrValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations*tAttributeSize +
+ attrCount*tNoOfOperations*tAttributeSize + opCount*tAttributeSize]));
+ } // for
+
+ check = MyTakeOverTrans->execute(Commit);
+ if (check == 0) {
+ check = MyTransaction->nextScanResult();
+ opCount++;
+ } // if
+ else {
+ tResult = 95;
+
+ /* MyTransaction, MyTakeOverTrans, Which one? */
+
+ // Any further error handling?
+ int retCode = flexScanErrorData->handleErrorCommon(MyTakeOverTrans->getNdbError());
+ if (retCode == 1) {
+ if (MyTakeOverTrans->getNdbError().code != 626 && MyTakeOverTrans->getNdbError().code != 630){
+ ndbout_c("execute: %s", MyTakeOverTrans->getNdbError().message);
+ ndbout_c("Error code = %d", MyTakeOverTrans->getNdbError().code);}
+ tResult = 20;
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexBench" << endl;
+ tResult = 20;
+ } else if (retCode == 3) {
+ // --------------------------------------------------------------------
+ // We are not certain if the transaction was successful or not.
+ // We must reexecute but might very well find that the transaction
+ // actually was updated. Updates and Reads are no problem here. Inserts
+ // will not cause a problem if error code 630 arrives. Deletes will
+ // not cause a problem if 626 arrives.
+ // --------------------------------------------------------------------
+ /* What can we do here? */
+ ndbout_c("execute: %s", MyTakeOverTrans->getNdbError().message);
+ }//if(retCode == 3)
+
+ } // else
+ pNdb->closeTransaction(MyTakeOverTrans);
+ } // while
+
+ pNdb->closeTransaction(MyTransaction);
+ } // for
+
+ return(tResult);
+} // scanUpdateRows
+
+static int scanDeleteRows(Ndb* pNdb, int* readValue)
+{
+ int tResult = 0;
+ int tableCount = 0;
+ int attrCount = 0;
+ int check = 0;
+ NdbRecAttr* tTmp = NULL;
+ NdbConnection* MyTransaction = NULL;
+ NdbOperation* MyOperation = NULL;
+ NdbConnection* MyTakeOverTrans = NULL;
+ NdbOperation* MyTakeOverOp = NULL;
+
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+ MyTransaction = pNdb->startTransaction();
+ if (MyTransaction == NULL) {
+ tResult = 1;
+ break; // break tableCount for loop
+ } // if
+
+ MyOperation = MyTransaction->getNdbOperation(tableName[tableCount]);
+ if (MyOperation == NULL) {
+ tResult = 2;
+ break;
+ } // if
+
+ check = MyOperation->openScanExclusive(tParallellism);
+ if (check == -1) {
+ tResult = 11;
+ break;
+ } // if
+
+ MyOperation->interpret_exit_ok();
+ for (int attrCount = 0; attrCount < tNoOfAttributes-1; attrCount++) {
+ tTmp = MyOperation->
+ getValue((char*)attrName[attrCount+1],
+ (char*)&(readValue[tableCount*(tNoOfAttributes-1)*tNoOfOperations +
+ attrCount*tNoOfOperations]));
+ if (tTmp == NULL) {
+ tResult = 9;
+ break;
+ } // if
+ } // for
+
+ check = MyTransaction->executeScan();
+ if (check == -1) {
+ tResult = 12;
+ break;
+ } // if
+ check = MyTransaction->nextScanResult();
+ while (check == 0) {
+ MyTakeOverTrans = pNdb->startTransaction();
+ MyTakeOverOp = MyOperation->takeOverForDelete(MyTakeOverTrans);
+ check = MyTakeOverOp->deleteTuple();
+
+ check = MyTakeOverTrans->execute(Commit);
+
+ //Error handling here
+
+ int retCode =flexScanErrorData->handleErrorCommon(MyTakeOverTrans->getNdbError());
+ if (retCode == 1) {
+ if (MyTakeOverTrans->getNdbError().code != 626 && MyTakeOverTrans->getNdbError().code != 630){
+ ndbout_c("execute: %s", MyTakeOverTrans->getNdbError().message );
+ ndbout_c("Error code = %d", MyTakeOverTrans->getNdbError().code );}
+ tResult = 20;
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexBench" << endl;
+ tResult = 20;
+ } else if (retCode == 3) {
+// --------------------------------------------------------------------
+// We are not certain if the transaction was successful or not.
+// We must reexecute but might very well find that the transaction
+// actually was updated. Updates and Reads are no problem here. Inserts
+// will not cause a problem if error code 630 arrives. Deletes will
+// not cause a problem if 626 arrives.
+// --------------------------------------------------------------------
+ /* What can we do here? */
+ ndbout_c("execute: %s", MyTakeOverTrans->getNdbError().message );
+ }//if(retCode == 3) End of error handling
+
+ pNdb->closeTransaction(MyTakeOverTrans);
+ check = MyTransaction->nextScanResult();
+ } // while
+ pNdb->closeTransaction(MyTransaction);
+ } // for tableCount
+ return(tResult);
+} // scanDeleteRows
+
+static int deleteRows(Ndb* pNdb,
+ int* pkValue)
+{
+ int tResult = 0;
+ NdbConnection* MyTransaction = NULL;
+ int tableCount = 0;
+ int opCount = 0;
+ int check = 0;
+ NdbOperation* MyOperations[MAXTABLES] = {NULL};
+
+ for (opCount = 0; opCount < tNoOfOperations; opCount++) {
+ MyTransaction = pNdb->startTransaction();
+ if (MyTransaction == NULL) {
+ tResult = 1;
+ } // if
+ else {
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+
+ MyOperations[tableCount] =
+ MyTransaction->getNdbOperation(tableName[tableCount]);
+ if (MyOperations[tableCount] == NULL) {
+ tResult = 2;
+ // Break for tableCount loop
+ break;
+ } // if
+
+ check = MyOperations[tableCount]->deleteTuple();
+ if (check == -1) {
+ tResult = 3;
+ break;
+ } // if
+
+ check = MyOperations[tableCount]->
+ equal((char*)attrName[0], (char*)&(pkValue[opCount]));
+ if (check == -1) {
+ tResult = 7;
+ break;
+ } // if
+
+ } // for tableCount
+
+ // Execute transaction deleting one tuple in every table
+ check = MyTransaction->execute(Commit);
+ if (check == -1) {
+ ndbout << MyTransaction->getNdbError().message << endl;
+ // Add complete error handling here
+
+ int retCode = flexScanErrorData->handleErrorCommon(MyTransaction->getNdbError());
+ if (retCode == 1) {
+ if (MyTransaction->getNdbError().code != 626 && MyTransaction->getNdbError().code != 630){
+ ndbout_c("execute: %d, %s", opCount, MyTransaction->getNdbError().message );
+ ndbout_c("Error code = %d", MyTransaction->getNdbError().code );}
+ tResult = 20;
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexBench" << endl;
+ tResult = 20;
+ } else if (retCode == 3) {
+// --------------------------------------------------------------------
+// We are not certain if the transaction was successful or not.
+// We must reexecute but might very well find that the transaction
+// actually was updated. Updates and Reads are no problem here. Inserts
+// will not cause a problem if error code 630 arrives. Deletes will
+// not cause a problem if 626 arrives.
+// --------------------------------------------------------------------
+ /* What can we do here? */
+ ndbout_c("execute: %s", MyTransaction->getNdbError().message );
+ }//if(retCode == 3)
+
+ } // if
+
+ pNdb->closeTransaction(MyTransaction);
+ } // else
+ } // for opCount
+
+ return(tResult);
+
+} // deleteRows
+
+////////////////////////////////////////
+//
+// Name: verifyDeleteRows
+//
+// Purpose: Verifies that all tables are empty by reading every tuple
+// No deletions made here
+//
+// Returns: 'Standard' error codes
+//
+/////////////////////////////////////
+static int verifyDeleteRows(Ndb* pNdb,
+ int* pkValue,
+ int* readValue)
+{
+ int tResult = 0;
+ int tableCount = 0;
+ int attrCount = 0;
+ int check = 0;
+ NdbConnection* MyTransaction = NULL;
+ NdbOperation* MyOperations = NULL;
+ NdbRecAttr* tmp = NULL;
+ int Value = 0;
+ int Index = 0;
+ int opCount = 0;
+
+ for (opCount = 0; opCount < tNoOfOperations; opCount++) {
+ for (tableCount = 0; tableCount < tNoOfTables; tableCount++) {
+ MyTransaction = pNdb->startTransaction();
+ if (MyTransaction == NULL) {
+ tResult = 1;
+ } // if
+ else {
+
+ MyOperations =
+ MyTransaction->getNdbOperation(tableName[tableCount]);
+ if (MyOperations == NULL) {
+ tResult = 2;
+ // Break for tableCount loop
+ break;
+ } // if
+
+ check = MyOperations->readTuple();
+ if (check == -1) {
+ tResult = 3;
+ break;
+ } // if
+
+ check = MyOperations->
+ equal((char*)attrName[0], (char*)&(pkValue[opCount]));
+ if (check == -1) {
+ tResult = 7;
+ break;
+ } // if
+
+ for (int attrCount = 0; attrCount < tNoOfAttributes - 1; attrCount++) {
+ Index = tableCount * (tNoOfAttributes - 1) * tNoOfOperations * tAttributeSize +
+ attrCount * tNoOfOperations * tAttributeSize + opCount * tAttributeSize;
+ tmp = MyOperations->
+ getValue((char*)attrName[attrCount + 1], (char*)&(readValue[Index]));
+
+ if (tmp == NULL) {
+ tResult = 9;
+ break;
+ } // if
+ } // for attrCount
+ // Execute transaction reading one tuple in every table
+ check = MyTransaction->execute(Commit);
+ if ((check == -1) && (MyTransaction->getNdbError().code == 626)){
+ // This is expected because everything should be deleted
+ } // if
+ else if (check == 0) {
+ // We have found a tuple that should have been deleted
+ ndbout << "tuple " << tableName[tableCount] << ":" <<
+ opCount << " was never deleted" << endl;
+ tResult = 97;
+ } // else if
+ else {
+ // Unexpected error
+ ndbout << "Unexpected error during delete" << endl;
+ assert(false);
+ } // else
+
+ pNdb->closeTransaction(MyTransaction);
+
+ } // else
+ } // for tableCount
+ } // for opCount
+
+ return(tResult);
+} // verifyDeleteRows
diff --git a/ndb/test/ndbapi/flexTT/Makefile b/ndb/test/ndbapi/flexTT/Makefile
new file mode 100644
index 00000000000..ba7493d633e
--- /dev/null
+++ b/ndb/test/ndbapi/flexTT/Makefile
@@ -0,0 +1,47 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := flexTT
+
+# Source files of non-templated classes (.C files)
+SOURCES = flexTT.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+# run automated tests
+test:
+ @echo "Results test-scripts flexScan." > report.txt
+ @echo "==============================" >> report.txt
+ @echo >> report.txt
+ @echo "User : \t\t\c" >> report.txt
+ @whoami >> report.txt
+ @echo "Date : \t\t\c" >> report.txt
+ @date >> report.txt
+ @echo >> report.txt
+ @echo "Host : \t\t\c" >> report.txt
+ @uname -n >> report.txt
+ @echo "\nStarting test 1."
+ @echo "\n==============================\n" >> report.txt
+ @test1.sh
+ @echo "\nStarting test 2."
+ @echo "\n==============================\n" >> report.txt
+ @test2.sh
+ @echo "\nStarting test 3."
+ @echo "\n==============================\n" >> report.txt
+ @test3.sh
+
+# run basic tests
+run_test :
+ flexBench
+ flexBench -t 32 -o 5000
+ flexBench -simple
+ flexBench -simple -t 32
+ flexBench -dirty
+ flexBench -dirty -t 32
+ flexBench -write
+ flexBench -write -t 32
+ flexBench -dirty -write
+ flexBench -dirty -write -t 32
+ flexBench -c 16 -t 32 -o 5000
+ flexBench -t 16 -c 30 -o 10000
diff --git a/ndb/test/ndbapi/flexTT/flexTT.cpp b/ndb/test/ndbapi/flexTT/flexTT.cpp
new file mode 100644
index 00000000000..1705f20b706
--- /dev/null
+++ b/ndb/test/ndbapi/flexTT/flexTT.cpp
@@ -0,0 +1,928 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NdbApi.hpp"
+#include <NdbMain.h>
+#include <md5_hash.hpp>
+
+#include <NdbThread.h>
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#include <NdbOut.hpp>
+#include <NdbTimer.hpp>
+#include <string.h>
+#include <NdbStdio.h>
+#include <stdlib.h>
+
+#include <NdbTest.hpp>
+#include <NDBT_Error.hpp>
+
+#define MAX_PARTS 4
+#define MAX_SEEK 16
+#define MAXSTRLEN 16
+#define MAXATTR 64
+#define MAXTABLES 64
+#define MAXTHREADS 128
+#define MAXPAR 1024
+#define MAXATTRSIZE 1000
+#define PKSIZE 1
+
+
+#ifdef NDB_WIN32
+inline long lrand48(void) { return rand(); };
+#endif
+
+
+enum StartType {
+ stIdle,
+ stInsert,
+ stRead,
+ stUpdate,
+ stDelete,
+ stStop
+} ;
+
+struct ThreadNdb
+{
+ int threadNo;
+ Ndb* threadNdb;
+ Uint32 threadBase;
+ Uint32 threadLoopCounter;
+ Uint32 threadNextStart;
+ Uint32 threadStop;
+ Uint32 threadLoopStop;
+ Uint32 threadIncrement;
+ Uint32 threadNoCompleted;
+ bool threadCompleted;
+ StartType threadStartType;
+};
+
+struct TransNdb
+{
+ char transRecord[128];
+ Ndb* transNdb;
+ StartType transStartType;
+ Uint32 vpn_number;
+ Uint32 vpn_identity;
+ Uint32 transErrorCount;
+ NdbOperation* transOperation;
+ ThreadNdb* transThread;
+};
+
+extern "C" { static void* threadLoop(void*); }
+static void setAttrNames(void);
+static void setTableNames(void);
+static int readArguments(int argc, const char** argv);
+static int createTables(Ndb*);
+static bool defineOperation(NdbConnection* aTransObject, TransNdb*,
+ Uint32 vpn_nb, Uint32 vpn_id);
+static bool executeTransaction(TransNdb* transNdbRef);
+static StartType random_choice();
+static void execute(StartType aType);
+static bool executeThread(ThreadNdb*, TransNdb*);
+static void executeCallback(int result, NdbConnection* NdbObject,
+ void* aObject);
+static bool error_handler(const NdbError & err) ;
+static Uint32 getKey(Uint32, Uint32) ;
+static void input_error();
+
+ErrorData * flexTTErrorData;
+
+static NdbThread* threadLife[MAXTHREADS];
+static int tNodeId;
+static int ThreadReady[MAXTHREADS];
+static StartType ThreadStart[MAXTHREADS];
+static char tableName[1][MAXSTRLEN+1];
+static char attrName[5][MAXSTRLEN+1];
+
+// Program Parameters
+static bool tInsert = false;
+static bool tDelete = false;
+static bool tReadUpdate = true;
+static int tUpdateFreq = 20;
+static bool tLocal = false;
+static int tLocalPart = 0;
+static int tMinEvents = 0;
+static int tSendForce = 0;
+static int tNoOfLoops = 1;
+static Uint32 tNoOfThreads = 1;
+static Uint32 tNoOfParallelTrans = 32;
+static Uint32 tNoOfTransactions = 500;
+static Uint32 tLoadFactor = 80;
+static bool tempTable = false;
+static bool startTransGuess = true;
+
+//Program Flags
+static int theSimpleFlag = 0;
+static int theDirtyFlag = 0;
+static int theWriteFlag = 0;
+static int theTableCreateFlag = 1;
+
+#define START_REAL_TIME
+#define STOP_REAL_TIME
+#define START_TIMER { NdbTimer timer; timer.doStart();
+#define STOP_TIMER timer.doStop();
+#define PRINT_TIMER(text, trans, opertrans) timer.printTransactionStatistics(text, trans, opertrans); };
+
+static void
+resetThreads(){
+
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ ThreadReady[i] = 0;
+ ThreadStart[i] = stIdle;
+ }//for
+}
+
+static void
+waitForThreads(void)
+{
+ int cont = 0;
+ do {
+ cont = 0;
+ NdbSleep_MilliSleep(20);
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ if (ThreadReady[i] == 0) {
+ cont = 1;
+ }//if
+ }//for
+ } while (cont == 1);
+}
+
+static void
+tellThreads(StartType what)
+{
+ for (int i = 0; i < tNoOfThreads ; i++)
+ ThreadStart[i] = what;
+}
+
+NDB_COMMAND(flexTT, "flexTT", "flexTT", "flexTT", 65535)
+{
+ ThreadNdb* pThreadData;
+ int returnValue = NDBT_OK;
+
+ flexTTErrorData = new ErrorData;
+ flexTTErrorData->resetErrorCounters();
+
+ if (readArguments(argc, argv) != 0){
+ input_error();
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ pThreadData = new ThreadNdb[MAXTHREADS];
+
+ ndbout << endl << "FLEXTT - Starting normal mode" << endl;
+ ndbout << "Perform TimesTen benchmark" << endl;
+ ndbout << " " << tNoOfThreads << " number of concurrent threads " << endl;
+ ndbout << " " << tNoOfParallelTrans;
+ ndbout << " number of parallel transaction per thread " << endl;
+ ndbout << " " << tNoOfTransactions << " transaction(s) per round " << endl;
+ ndbout << " " << tNoOfLoops << " iterations " << endl;
+ ndbout << " " << "Update Frequency is " << tUpdateFreq << "%" << endl;
+ ndbout << " " << "Load Factor is " << tLoadFactor << "%" << endl;
+ if (tLocal == true) {
+ ndbout << " " << "We only use Local Part = ";
+ ndbout << tLocalPart << endl;
+ }//if
+ if (tempTable == true) {
+ ndbout << " Tables are without logging " << endl;
+ } else {
+ ndbout << " Tables are with logging " << endl;
+ }//if
+ if (startTransGuess == true) {
+ ndbout << " Transactions are executed with hint provided" << endl;
+ } else {
+ ndbout << " Transactions are executed with round robin scheme" << endl;
+ }//if
+ if (tSendForce == 0) {
+ ndbout << " No force send is used, adaptive algorithm used" << endl;
+ } else if (tSendForce == 1) {
+ ndbout << " Force send used" << endl;
+ } else {
+ ndbout << " No force send is used, adaptive algorithm disabled" << endl;
+ }//if
+
+ ndbout << endl;
+
+ /* print Setting */
+ flexTTErrorData->printSettings(ndbout);
+
+ NdbThread_SetConcurrencyLevel(2 + tNoOfThreads);
+
+ setAttrNames();
+ setTableNames();
+
+ Ndb * pNdb = new Ndb("TEST_DB");
+ pNdb->init();
+ tNodeId = pNdb->getNodeId();
+
+ ndbout << " NdbAPI node with id = " << pNdb->getNodeId() << endl;
+ ndbout << endl;
+
+ ndbout << "Waiting for ndb to become ready..." <<endl;
+ if (pNdb->waitUntilReady(2000) != 0){
+ ndbout << "NDB is not ready" << endl;
+ ndbout << "Benchmark failed!" << endl;
+ returnValue = NDBT_FAILED;
+ }
+
+ if(returnValue == NDBT_OK){
+ if (createTables(pNdb) != 0){
+ returnValue = NDBT_FAILED;
+ }
+ }
+
+ if(returnValue == NDBT_OK){
+ /****************************************************************
+ * Create NDB objects. *
+ ****************************************************************/
+ resetThreads();
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ pThreadData[i].threadNo = i;
+ threadLife[i] = NdbThread_Create(threadLoop,
+ (void**)&pThreadData[i],
+ 32768,
+ "flexAsynchThread",
+ NDB_THREAD_PRIO_LOW);
+ }//for
+ ndbout << endl << "All NDB objects and table created" << endl << endl;
+ int noOfTransacts = tNoOfParallelTrans * tNoOfTransactions *
+ tNoOfThreads * tNoOfLoops;
+ /****************************************************************
+ * Execute program. *
+ ****************************************************************/
+ /****************************************************************
+ * Perform inserts. *
+ ****************************************************************/
+
+ if (tInsert == true) {
+ tInsert = false;
+ tReadUpdate = false;
+ START_TIMER;
+ execute(stInsert);
+ STOP_TIMER;
+ PRINT_TIMER("insert", noOfTransacts, 1);
+ }//if
+ /****************************************************************
+ * Perform read + updates. *
+ ****************************************************************/
+
+ if (tReadUpdate == true) {
+ START_TIMER;
+ execute(stRead);
+ STOP_TIMER;
+ PRINT_TIMER("update + read", noOfTransacts, 1);
+ }//if
+ /****************************************************************
+ * Perform delete. *
+ ****************************************************************/
+
+ if (tDelete == true) {
+ tDelete = false;
+ START_TIMER;
+ execute(stDelete);
+ STOP_TIMER;
+ PRINT_TIMER("delete", noOfTransacts, 1);
+ }//if
+ ndbout << "--------------------------------------------------" << endl;
+
+ execute(stStop);
+ void * tmp;
+ for(int i = 0; i<tNoOfThreads; i++){
+ NdbThread_WaitFor(threadLife[i], &tmp);
+ NdbThread_Destroy(&threadLife[i]);
+ }
+ }
+ delete [] pThreadData;
+ delete pNdb;
+
+ //printing errorCounters
+ flexTTErrorData->printErrorCounters(ndbout);
+
+ return NDBT_ProgramExit(returnValue);
+}//main()
+
+
+static void execute(StartType aType)
+{
+ resetThreads();
+ tellThreads(aType);
+ waitForThreads();
+}//execute()
+
+static void*
+threadLoop(void* ThreadData)
+{
+ Ndb* localNdb;
+ ThreadNdb* tabThread = (ThreadNdb*)ThreadData;
+ int loc_threadNo = tabThread->threadNo;
+
+ void * mem = malloc(sizeof(TransNdb)*tNoOfParallelTrans);
+ TransNdb* pTransData = (TransNdb*)mem;
+
+ localNdb = new Ndb("TEST_DB");
+ localNdb->init(1024);
+ localNdb->waitUntilReady();
+
+ if (tLocal == false) {
+ tabThread->threadIncrement = 1;
+ } else {
+ tabThread->threadIncrement = MAX_SEEK;
+ }//if
+ tabThread->threadBase = (loc_threadNo << 16) + tNodeId;
+ tabThread->threadNdb = localNdb;
+ tabThread->threadStop = tNoOfParallelTrans * tNoOfTransactions;
+ tabThread->threadStop *= tabThread->threadIncrement;
+ tabThread->threadLoopStop = tNoOfLoops;
+ Uint32 i, j;
+ for (i = 0; i < tNoOfParallelTrans; i++) {
+ pTransData[i].transNdb = localNdb;
+ pTransData[i].transThread = tabThread;
+ pTransData[i].transOperation = NULL;
+ pTransData[i].transStartType = stIdle;
+ pTransData[i].vpn_number = tabThread->threadBase;
+ pTransData[i].vpn_identity = 0;
+ pTransData[i].transErrorCount = 0;
+ for (j = 0; j < 128; j++) {
+ pTransData[i].transRecord[j] = 0x30;
+ }//for
+ }//for
+
+ for (;;){
+ while (ThreadStart[loc_threadNo] == stIdle) {
+ NdbSleep_MilliSleep(10);
+ }//while
+
+ // Check if signal to exit is received
+ if (ThreadStart[loc_threadNo] == stStop) {
+ break;
+ }//if
+
+ tabThread->threadStartType = ThreadStart[loc_threadNo];
+ tabThread->threadLoopCounter = 0;
+ tabThread->threadCompleted = false;
+ tabThread->threadNoCompleted = 0;
+ tabThread->threadNextStart = 0;
+
+ ThreadStart[loc_threadNo] = stIdle;
+ if(!executeThread(tabThread, pTransData)){
+ break;
+ }
+ ThreadReady[loc_threadNo] = 1;
+ }//for
+
+ free(mem);
+ delete localNdb;
+ ThreadReady[loc_threadNo] = 1;
+
+ NdbThread_Exit(0);
+ return NULL; // Just to keep compiler happy
+}//threadLoop()
+
+static
+bool
+executeThread(ThreadNdb* tabThread, TransNdb* atransDataArrayPtr) {
+ Uint32 i;
+ for (i = 0; i < tNoOfParallelTrans; i++) {
+ TransNdb* transNdbPtr = &atransDataArrayPtr[i];
+ transNdbPtr->vpn_identity = i * tabThread->threadIncrement;
+ transNdbPtr->transStartType = tabThread->threadStartType;
+ if (executeTransaction(transNdbPtr) == false) {
+ return false;
+ }//if
+ }//for
+ tabThread->threadNextStart = tNoOfParallelTrans * tabThread->threadIncrement;
+ do {
+ tabThread->threadNdb->sendPollNdb(3000, tMinEvents, tSendForce);
+ } while (tabThread->threadCompleted == false);
+ return true;
+}//executeThread()
+
+static
+bool executeTransaction(TransNdb* transNdbRef)
+{
+ NdbConnection* MyTrans;
+ ThreadNdb* tabThread = transNdbRef->transThread;
+ Ndb* aNdbObject = transNdbRef->transNdb;
+ Uint32 threadBase = tabThread->threadBase;
+ Uint32 startKey = transNdbRef->vpn_identity;
+ if (tLocal == true) {
+ startKey = getKey(startKey, threadBase);
+ }//if
+ if (startTransGuess == true) {
+ Uint32 tKey[2];
+ tKey[0] = startKey;
+ tKey[1] = threadBase;
+ MyTrans = aNdbObject->startTransaction((Uint32)0, //Priority
+ (const char*)&tKey[0], //Main PKey
+ (Uint32)8); //Key Length
+ } else {
+ MyTrans = aNdbObject->startTransaction();
+ }//if
+ if (MyTrans == NULL) {
+ error_handler(aNdbObject->getNdbError());
+ ndbout << endl << "Unable to recover! Quiting now" << endl ;
+ return false;
+ }//if
+ //-------------------------------------------------------
+ // Define the operation, but do not execute it yet.
+ //-------------------------------------------------------
+ if (!defineOperation(MyTrans, transNdbRef, startKey, threadBase))
+ return false;
+
+ return true;
+}//executeTransaction()
+
+
+static
+Uint32
+getKey(Uint32 aBase, Uint32 aThreadBase) {
+ Uint32 Tfound = aBase;
+ Uint32 hash;
+ Uint64 Tkey64;
+ Uint32* tKey32 = (Uint32*)&Tkey64;
+ tKey32[0] = aThreadBase;
+ for (int i = aBase; i < (aBase + MAX_SEEK); i++) {
+ tKey32[1] = (Uint32)i;
+ hash = md5_hash((Uint64*)&Tkey64, (Uint32)2);
+ hash = (hash >> 6) & (MAX_PARTS - 1);
+ if (hash == tLocalPart) {
+ Tfound = i;
+ break;
+ }//if
+ }//for
+ return Tfound;
+}//getKey()
+
+static void
+executeCallback(int result, NdbConnection* NdbObject, void* aObject)
+{
+ TransNdb* transNdbRef = (TransNdb*)aObject;
+ ThreadNdb* tabThread = transNdbRef->transThread;
+ Ndb* tNdb = transNdbRef->transNdb;
+ Uint32 vpn_id = transNdbRef->vpn_identity;
+ Uint32 vpn_nb = tabThread->threadBase;
+
+ if (result == -1) {
+// Add complete error handling here
+ int retCode = flexTTErrorData->handleErrorCommon(NdbObject->getNdbError());
+ if (retCode == 1) {
+ if (NdbObject->getNdbError().code != 626 &&
+ NdbObject->getNdbError().code != 630) {
+ ndbout_c("execute: %s", NdbObject->getNdbError().message);
+ ndbout_c("Error code = %d", NdbObject->getNdbError().code);
+ }
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexTT" << endl;
+ } else if (retCode == 3) {
+ /* What can we do here? */
+ ndbout_c("execute: %s", NdbObject->getNdbError().message);
+ }//if(retCode == 3)
+ transNdbRef->transErrorCount++;
+ const NdbError & err = NdbObject->getNdbError();
+ switch (err.classification) {
+ case NdbError::NoDataFound:
+ case NdbError::ConstraintViolation:
+ ndbout << "Error with vpn_id = " << vpn_id << " and vpn_nb = ";
+ ndbout << vpn_nb << endl;
+ ndbout << err << endl;
+ goto checkCompleted;
+ case NdbError::OverloadError:
+ NdbSleep_MilliSleep(10);
+ case NdbError::NodeRecoveryError:
+ case NdbError::UnknownResultError:
+ case NdbError::TimeoutExpired:
+ break;
+ default:
+ goto checkCompleted;
+ }//if
+ if ((transNdbRef->transErrorCount > 10) ||
+ (tabThread->threadNoCompleted > 0)) {
+ goto checkCompleted;
+ }//if
+ } else {
+ if (tabThread->threadNoCompleted == 0) {
+ transNdbRef->transErrorCount = 0;
+ transNdbRef->vpn_identity = tabThread->threadNextStart;
+ if (tabThread->threadNextStart == tabThread->threadStop) {
+ tabThread->threadLoopCounter++;
+ transNdbRef->vpn_identity = 0;
+ tabThread->threadNextStart = 0;
+ if (tabThread->threadLoopCounter == tNoOfLoops) {
+ goto checkCompleted;
+ }//if
+ }//if
+ tabThread->threadNextStart += tabThread->threadIncrement;
+ } else {
+ goto checkCompleted;
+ }//if
+ }//if
+ tNdb->closeTransaction(NdbObject);
+ executeTransaction(transNdbRef);
+ return;
+
+checkCompleted:
+ tNdb->closeTransaction(NdbObject);
+ tabThread->threadNoCompleted++;
+ if (tabThread->threadNoCompleted == tNoOfParallelTrans) {
+ tabThread->threadCompleted = true;
+ }//if
+ return;
+}//executeCallback()
+
+static
+StartType
+random_choice()
+{
+//----------------------------------------------------
+// Generate a random key between 0 and tNoOfRecords - 1
+//----------------------------------------------------
+ UintR random_number = lrand48() % 100;
+ if (random_number < tUpdateFreq)
+ return stUpdate;
+ else
+ return stRead;
+}//random_choice()
+
+static bool
+defineOperation(NdbConnection* localNdbConnection, TransNdb* transNdbRef,
+ unsigned int vpn_id, unsigned int vpn_nb)
+{
+ NdbOperation* localNdbOperation;
+ StartType TType = transNdbRef->transStartType;
+
+ //-------------------------------------------------------
+ // Set-up the attribute values for this operation.
+ //-------------------------------------------------------
+ localNdbOperation = localNdbConnection->getNdbOperation(tableName[0]);
+ if (localNdbOperation == NULL) {
+ error_handler(localNdbConnection->getNdbError());
+ return false;
+ }//if
+ switch (TType) {
+ case stInsert: // Insert case
+ if (theWriteFlag == 1 && theDirtyFlag == 1) {
+ localNdbOperation->dirtyWrite();
+ } else if (theWriteFlag == 1) {
+ localNdbOperation->writeTuple();
+ } else {
+ localNdbOperation->insertTuple();
+ }//if
+ break;
+ case stRead: // Read Case
+ TType = random_choice();
+ if (TType == stRead) {
+ if (theSimpleFlag == 1) {
+ localNdbOperation->simpleRead();
+ } else if (theDirtyFlag == 1) {
+ localNdbOperation->dirtyRead();
+ } else {
+ localNdbOperation->readTuple();
+ }//if
+ } else {
+ if (theWriteFlag == 1 && theDirtyFlag == 1) {
+ localNdbOperation->dirtyWrite();
+ } else if (theWriteFlag == 1) {
+ localNdbOperation->writeTuple();
+ } else if (theDirtyFlag == 1) {
+ localNdbOperation->dirtyUpdate();
+ } else {
+ localNdbOperation->updateTuple();
+ }//if
+ }//if
+ break;
+ case stDelete: // Delete Case
+ localNdbOperation->deleteTuple();
+ break;
+ default:
+ error_handler(localNdbOperation->getNdbError());
+ }//switch
+ localNdbOperation->equal((Uint32)0,vpn_id);
+ localNdbOperation->equal((Uint32)1,vpn_nb);
+ char* attrValue = &transNdbRef->transRecord[0];
+ switch (TType) {
+ case stInsert: // Insert case
+ localNdbOperation->setValue((Uint32)2, attrValue);
+ localNdbOperation->setValue((Uint32)3, attrValue);
+ localNdbOperation->setValue((Uint32)4, attrValue);
+ break;
+ case stUpdate: // Update Case
+ localNdbOperation->setValue((Uint32)3, attrValue);
+ break;
+ case stRead: // Read Case
+ localNdbOperation->getValue((Uint32)2, attrValue);
+ localNdbOperation->getValue((Uint32)3, attrValue);
+ localNdbOperation->getValue((Uint32)4, attrValue);
+ break;
+ case stDelete: // Delete Case
+ break;
+ default:
+ error_handler(localNdbOperation->getNdbError());
+ }//switch
+ localNdbConnection->executeAsynchPrepare(Commit, &executeCallback,
+ (void*)transNdbRef);
+ return true;
+}//defineOperation()
+
+
+static void setAttrNames()
+{
+ snprintf(attrName[0], MAXSTRLEN, "VPN_ID");
+ snprintf(attrName[1], MAXSTRLEN, "VPN_NB");
+ snprintf(attrName[2], MAXSTRLEN, "DIRECTORY_NB");
+ snprintf(attrName[3], MAXSTRLEN, "LAST_CALL_PARTY");
+ snprintf(attrName[4], MAXSTRLEN, "DESCR");
+}
+
+
+static void setTableNames()
+{
+ snprintf(tableName[0], MAXSTRLEN, "VPN_USERS");
+}
+
+static
+int
+createTables(Ndb* pMyNdb){
+
+ NdbSchemaCon *MySchemaTransaction;
+ NdbSchemaOp *MySchemaOp;
+ int check;
+
+ if (theTableCreateFlag == 0) {
+ ndbout << "Creating Table: vpn_users " << "..." << endl;
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+
+ if(MySchemaTransaction == NULL &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if(MySchemaOp == NULL &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ check = MySchemaOp->createTable( tableName[0]
+ ,8 // Table Size
+ ,TupleKey // Key Type
+ ,40 // Nr of Pages
+ ,All
+ ,6
+ ,(tLoadFactor - 5)
+ ,tLoadFactor
+ ,1
+ ,!tempTable
+ );
+
+ if (check == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ check = MySchemaOp->createAttribute( (char*)attrName[0],
+ TupleKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if (check == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+ check = MySchemaOp->createAttribute( (char*)attrName[1],
+ TupleKey,
+ 32,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if (check == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+ check = MySchemaOp->createAttribute( (char*)attrName[2],
+ NoKey,
+ 8,
+ 10,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if (check == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ check = MySchemaOp->createAttribute( (char*)attrName[3],
+ NoKey,
+ 8,
+ 10,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if (check == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ check = MySchemaOp->createAttribute( (char*)attrName[4],
+ NoKey,
+ 8,
+ 100,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if (check == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ if (MySchemaTransaction->execute() == -1 &&
+ (!error_handler(MySchemaTransaction->getNdbError())))
+ return -1;
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ }//if
+
+ return 0;
+}
+
+bool error_handler(const NdbError& err){
+ ndbout << err << endl ;
+ switch(err.classification){
+ case NdbError::NodeRecoveryError:
+ case NdbError::SchemaError:
+ case NdbError::TimeoutExpired:
+ ndbout << endl << "Attempting to recover and continue now..." << endl ;
+ return true ; // return true to retry
+ }
+ return false;
+}
+#if 0
+bool error_handler(const char* error_string, int error_int) {
+ ndbout << error_string << endl ;
+ if ((4008 == error_int) ||
+ (677 == error_int) ||
+ (891 == error_int) ||
+ (1221 == error_int) ||
+ (721 == error_int) ||
+ (266 == error_int)) {
+ ndbout << endl << "Attempting to recover and continue now..." << endl ;
+ return true ; // return true to retry
+ }
+ return false ; // return false to abort
+}
+#endif
+
+static
+int
+readArguments(int argc, const char** argv){
+
+ int i = 1;
+ while (argc > 1){
+ if (strcmp(argv[i], "-t") == 0){
+ tNoOfThreads = atoi(argv[i+1]);
+ if ((tNoOfThreads < 1) || (tNoOfThreads > MAXTHREADS)){
+ ndbout_c("Invalid no of threads");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-p") == 0){
+ tNoOfParallelTrans = atoi(argv[i+1]);
+ if ((tNoOfParallelTrans < 1) || (tNoOfParallelTrans > MAXPAR)){
+ ndbout_c("Invalid no of parallell transactions");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-o") == 0) {
+ tNoOfTransactions = atoi(argv[i+1]);
+ if (tNoOfTransactions < 1){
+ ndbout_c("Invalid no of transactions");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-l") == 0){
+ tNoOfLoops = atoi(argv[i+1]);
+ if (tNoOfLoops < 1) {
+ ndbout_c("Invalid no of loops");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-e") == 0){
+ tMinEvents = atoi(argv[i+1]);
+ if ((tMinEvents < 1) || (tMinEvents > tNoOfParallelTrans)) {
+ ndbout_c("Invalid no of loops");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-local") == 0){
+ tLocalPart = atoi(argv[i+1]);
+ tLocal = true;
+ startTransGuess = true;
+ if ((tLocalPart < 0) || (tLocalPart > MAX_PARTS)){
+ ndbout_c("Invalid local part");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-ufreq") == 0){
+ tUpdateFreq = atoi(argv[i+1]);
+ if ((tUpdateFreq < 0) || (tUpdateFreq > 100)){
+ ndbout_c("Invalid Update Frequency");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-load_factor") == 0){
+ tLoadFactor = atoi(argv[i+1]);
+ if ((tLoadFactor < 40) || (tLoadFactor >= 100)){
+ ndbout_c("Invalid LoadFactor");
+ return -1;
+ }
+ } else if (strcmp(argv[i], "-d") == 0){
+ tDelete = true;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-i") == 0){
+ tInsert = true;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-simple") == 0){
+ theSimpleFlag = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-adaptive") == 0){
+ tSendForce = 0;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-force") == 0){
+ tSendForce = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-non_adaptive") == 0){
+ tSendForce = 2;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-write") == 0){
+ theWriteFlag = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-dirty") == 0){
+ theDirtyFlag = 1;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-table_create") == 0){
+ theTableCreateFlag = 0;
+ tInsert = true;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-temp") == 0){
+ tempTable = true;
+ argc++;
+ i--;
+ } else if (strcmp(argv[i], "-no_hint") == 0){
+ startTransGuess = false;
+ argc++;
+ i--;
+ } else {
+ return -1;
+ }
+
+ argc -= 2;
+ i = i + 2;
+ }//while
+ if (tLocal == true) {
+ if (startTransGuess == false) {
+ ndbout_c("Not valid to use no_hint with local");
+ }//if
+ }//if
+ return 0;
+}
+
+static
+void
+input_error(){
+
+ ndbout_c("FLEXTT");
+ ndbout_c(" Perform benchmark of insert, update and delete transactions");
+ ndbout_c("");
+ ndbout_c("Arguments:");
+ ndbout_c(" -t Number of threads to start, default 1");
+ ndbout_c(" -p Number of parallel transactions per thread, default 32");
+ ndbout_c(" -o Number of transactions per loop, default 500");
+ ndbout_c(" -ufreq Number Update Frequency in percent (0 -> 100), rest is read");
+ ndbout_c(" -load_factor Number Fill level in index in percent (40 -> 99)");
+ ndbout_c(" -l Number of loops to run, default 1, 0=infinite");
+ ndbout_c(" -i Start by inserting all records");
+ ndbout_c(" -d End by deleting all records (only one loop)");
+ ndbout_c(" -simple Use simple read to read from database");
+ ndbout_c(" -dirty Use dirty read to read from database");
+ ndbout_c(" -write Use writeTuple in insert and update");
+ ndbout_c(" -n Use standard table names");
+ ndbout_c(" -table_create Create tables in db");
+ ndbout_c(" -temp Create table(s) without logging");
+ ndbout_c(" -no_hint Don't give hint on where to execute transaction coordinator");
+ ndbout_c(" -adaptive Use adaptive send algorithm (default)");
+ ndbout_c(" -force Force send when communicating");
+ ndbout_c(" -non_adaptive Send at a 10 millisecond interval");
+ ndbout_c(" -local Number of part, only use keys in one part out of 16");
+}
diff --git a/ndb/test/ndbapi/flexTimedAsynch/Makefile b/ndb/test/ndbapi/flexTimedAsynch/Makefile
new file mode 100644
index 00000000000..e9995dbd16f
--- /dev/null
+++ b/ndb/test/ndbapi/flexTimedAsynch/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := flexTimedAsynch
+
+# Source files of non-templated classes (.C files)
+SOURCES = flexTimedAsynch.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/flexTimedAsynch/flexTimedAsynch.cpp b/ndb/test/ndbapi/flexTimedAsynch/flexTimedAsynch.cpp
new file mode 100644
index 00000000000..761be53fdd3
--- /dev/null
+++ b/ndb/test/ndbapi/flexTimedAsynch/flexTimedAsynch.cpp
@@ -0,0 +1,852 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* ***************************************************
+ FLEXTIMEDASYNCH
+ Perform benchmark of insert, update and delete transactions.
+
+ Arguments:
+ -t Number of threads to start, i.e., number of parallel loops, default 1
+ -p Number of transactions in a batch, default 32
+ -o Number of batches per loop, default 200
+ -i Time between batch starts, default 0
+ -l Number of loops to run, default 1, 0=infinite
+ -a Number of attributes, default 25
+ -c Number of operations per transaction
+ -s Size of each attribute in 32 bit word, default 1 (Primary Key is always of size 1,
+ independent of this value)
+ -simple Use simple read to read from database
+ -dirty Use dirty read to read from database
+ -write Use writeTuple in insert and update
+ -n Use standard table names
+ -no_table_create Don't create tables in db
+ -temp Use temporary tables, no writing to disk.
+
+ Returns:
+ 0 - Test passed
+ -1 - Test failed
+ 1 - Invalid arguments
+
+ * *************************************************** */
+
+#include "NdbApi.hpp"
+
+#include <NdbThread.h>
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#include <NdbOut.hpp>
+#include <NdbTimer.hpp>
+#include <string.h>
+#include <NdbMain.h>
+#include <NdbTest.hpp>
+
+#include <NDBT_Error.hpp>
+
+#define MAXSTRLEN 16
+#define MAXATTR 64
+#define MAXTABLES 64
+#define MAXTHREADS 256
+#define MAXATTRSIZE 1000
+#define PKSIZE 1
+
+enum StartType { stIdle,
+ stInsert,
+ stRead,
+ stUpdate,
+ stDelete,
+ stStop } ;
+
+ErrorData * flexTimedAsynchErrorData;
+
+struct ThreadNdb
+{
+ int NoOfOps;
+ int ThreadNo;
+ unsigned int threadBase;
+ unsigned int transactionCompleted;
+};
+
+extern "C" void* threadLoop(void*);
+void setAttrNames(void);
+void setTableNames(void);
+void readArguments(int argc, const char** argv);
+void createAttributeSpace();
+void createTables(Ndb*);
+void defineOperation(NdbConnection* aTransObject, StartType aType, unsigned int key, int *);
+void execute(StartType aType);
+void executeThread(StartType aType, Ndb* aNdbObject, ThreadNdb* threadInfo);
+void executeCallback(int result, NdbConnection* NdbObject, void* aObject);
+
+/* epaulsa > *************************************************************/
+bool error_handler(const NdbError &) ; //replaces 'goto' things
+static int failed = 0 ; // lame global variable that keeps track of failed transactions
+ // incremented in executeCallback() and reset in main()
+/************************************************************* < epaulsa */
+
+static NdbThread* threadLife[MAXTHREADS];
+static int tNodeId;
+static int ThreadReady[MAXTHREADS];
+static StartType ThreadStart[MAXTHREADS];
+static char tableName[MAXTABLES][MAXSTRLEN+1];
+static char attrName[MAXATTR][MAXSTRLEN+1];
+static int *getAttrValueTable;
+
+// Program Parameters
+static int tNoOfLoops = 1;
+static int tAttributeSize = 1;
+static unsigned int tNoOfThreads = 1;
+static unsigned int tNoOfTransInBatch = 32;
+static unsigned int tNoOfAttributes = 25;
+static unsigned int tNoOfBatchesInLoop = 200;
+static unsigned int tNoOfOpsPerTrans = 1;
+static unsigned int tTimeBetweenBatches = 0;
+
+//Program Flags
+static int theTestFlag = 0;
+static int theTempFlag = 1;
+static int theSimpleFlag = 0;
+static int theDirtyFlag = 0;
+static int theWriteFlag = 0;
+static int theStdTableNameFlag = 0;
+static int theTableCreateFlag = 0;
+
+#define START_REAL_TIME NdbTimer timer; timer.doStart();
+#define STOP_REAL_TIME timer.doStop();
+
+#define START_TIMER { NdbTimer timer; timer.doStart();
+#define STOP_TIMER timer.doStop();
+#define PRINT_TIMER(text, trans, opertrans) timer.printTransactionStatistics(text, trans, opertrans); };
+
+void
+resetThreads(){
+
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ ThreadReady[i] = 0;
+ ThreadStart[i] = stIdle;
+ }
+}
+
+void
+waitForThreads(void)
+{
+ int cont;
+ do {
+ cont = 0;
+ NdbSleep_MilliSleep(20);
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ if (ThreadReady[i] == 0) {
+ cont = 1;
+ }
+ }
+ } while (cont == 1);
+}
+
+void
+tellThreads(StartType what)
+{
+ for (int i = 0; i < tNoOfThreads ; i++)
+ ThreadStart[i] = what;
+}
+
+void createAttributeSpace(){
+ getAttrValueTable = new int[tAttributeSize*
+ tNoOfThreads *
+ tNoOfAttributes ];
+
+}
+
+void deleteAttributeSpace(){
+ delete [] getAttrValueTable;
+}
+
+NDB_COMMAND(flexTimedAsynch, "flexTimedAsynch", "flexTimedAsynch [-tpoilcas]", "flexTimedAsynch", 65535)
+{
+ ThreadNdb tabThread[MAXTHREADS];
+ int tLoops=0;
+ int returnValue;
+ //NdbOut flexTimedAsynchNdbOut;
+
+ flexTimedAsynchErrorData = new ErrorData;
+ flexTimedAsynchErrorData->resetErrorCounters();
+
+ Ndb* pNdb;
+ pNdb = new Ndb( "TEST_DB" );
+ pNdb->init();
+
+ readArguments(argc, argv);
+
+ createAttributeSpace();
+
+ ndbout << endl << "FLEXTIMEDASYNCH - Starting normal mode" << endl;
+ ndbout << "Perform benchmark of insert, update and delete transactions" << endl << endl;
+
+ if(theTempFlag == 0)
+ ndbout << " " << "Using temporary tables. " << endl;
+
+ // -t, tNoOfThreads
+ ndbout << " " << tNoOfThreads << " number of concurrent threads " << endl;
+ // -c, tNoOfOpsPerTrans
+ ndbout << " " << tNoOfOpsPerTrans << " operations per transaction " << endl;
+ // -p, tNoOfTransInBatch
+ ndbout << " " << tNoOfTransInBatch << " number of transactions in a batch per thread " << endl;
+ // -o, tNoOfBatchesInLoop
+ ndbout << " " << tNoOfBatchesInLoop << " number of batches per loop " << endl;
+ // -i, tTimeBetweenBatches
+ ndbout << " " << tTimeBetweenBatches << " milli seconds at least between batch starts " << endl;
+ // -l, tNoOfLoops
+ ndbout << " " << tNoOfLoops << " loops " << endl;
+ // -a, tNoOfAttributes
+ ndbout << " " << tNoOfAttributes << " attributes per table " << endl;
+ // -s, tAttributeSize
+ ndbout << " " << tAttributeSize << " is the number of 32 bit words per attribute " << endl << endl;
+
+ NdbThread_SetConcurrencyLevel(2 + tNoOfThreads);
+
+ /* print Setting */
+ flexTimedAsynchErrorData->printSettings(ndbout);
+
+ setAttrNames();
+ setTableNames();
+
+ ndbout << "Waiting for ndb to become ready..." <<endl;
+ if (pNdb->waitUntilReady() == 0) {
+ tNodeId = pNdb->getNodeId();
+ ndbout << " NdbAPI node with id = " << tNodeId << endl;
+ createTables(pNdb);
+
+ /****************************************************************
+ * Create NDB objects. *
+ ****************************************************************/
+ resetThreads();
+ for (int i = 0; i < tNoOfThreads ; i++) {
+ tabThread[i].ThreadNo = i;
+
+ threadLife[i] = NdbThread_Create(threadLoop,
+ (void**)&tabThread[i],
+ 32768,
+ "flexTimedAsynchThread",
+ NDB_THREAD_PRIO_LOW);
+ }
+ ndbout << endl << "All NDB objects and table created" << endl << endl;
+ int noOfTransacts = tNoOfTransInBatch*tNoOfBatchesInLoop*tNoOfThreads;
+
+ /****************************************************************
+ * Execute program. *
+ ****************************************************************/
+
+
+ for(;;) {
+
+ int loopCount = tLoops + 1 ;
+ ndbout << endl << "Loop # " << loopCount << endl << endl ;
+
+ /****************************************************************
+ * Perform inserts. *
+ ****************************************************************/
+
+ failed = 0 ;
+
+ START_TIMER;
+ execute(stInsert);
+ STOP_TIMER;
+ PRINT_TIMER("insert", noOfTransacts, tNoOfOpsPerTrans);
+
+ if (0 < failed) {
+ ndbout << failed << " of the transactions returned errors!, moving on now..."<<endl ;
+ }
+
+ /****************************************************************
+ * Perform read. *
+ ****************************************************************/
+
+ failed = 0 ;
+
+ START_TIMER;
+ execute(stRead);
+ STOP_TIMER;
+ PRINT_TIMER("read", noOfTransacts, tNoOfOpsPerTrans);
+
+ if (0 < failed) {
+ ndbout << failed << " of the transactions returned errors!, moving on now..."<<endl ;
+ }
+
+
+
+ /****************************************************************
+ * Perform update. *
+ ***************************************************************/
+
+ failed = 0 ;
+
+ START_TIMER;
+ execute(stUpdate);
+ STOP_TIMER;
+ PRINT_TIMER("update", noOfTransacts, tNoOfOpsPerTrans) ;
+
+ if (0 < failed) {
+ ndbout << failed << " of the transactions returned errors!, moving on now..."<<endl ;
+ }
+
+ /****************************************************************
+ * Perform read after update.
+ ****************************************************************/
+
+ failed = 0 ;
+
+ START_TIMER;
+ execute(stRead);
+ STOP_TIMER;
+ PRINT_TIMER("read", noOfTransacts, tNoOfOpsPerTrans);
+
+ if (0 < failed) {
+ ndbout << failed << " of the transactions returned errors!, moving on now..."<<endl ;
+ }
+
+
+ /****************************************************************
+ * Perform delete. *
+ ****************************************************************/
+
+ failed = 0;
+
+ START_TIMER;
+ execute(stDelete);
+ STOP_TIMER;
+ PRINT_TIMER("delete", noOfTransacts, tNoOfOpsPerTrans);
+
+ if (0 < failed) {
+ ndbout << failed << " of the transactions returned errors!, moving on now..."<<endl ;
+ }
+
+ tLoops++;
+ ndbout << "--------------------------------------------------" << endl;
+
+ if(tNoOfLoops != 0){
+ if(tNoOfLoops <= tLoops)
+ break ;
+ }
+ }
+
+ ndbout << endl << "Benchmark completed!" << endl;
+ returnValue = NDBT_OK;
+
+ execute(stStop);
+ void * tmp;
+ for(int i = 0; i<tNoOfThreads; i++){
+ NdbThread_WaitFor(threadLife[i], &tmp);
+ NdbThread_Destroy(&threadLife[i]);
+ }
+ } else {
+ ndbout << "NDB is not ready" << endl;
+ ndbout << "Benchmark failed!" << endl;
+ returnValue = NDBT_FAILED;
+ }
+
+ deleteAttributeSpace();
+ delete pNdb;
+
+ //printing errorCounters
+ flexTimedAsynchErrorData->printErrorCounters(ndbout);
+
+ return NDBT_ProgramExit(returnValue);
+}//main()
+
+////////////////////////////////////////
+
+void execute(StartType aType)
+{
+ resetThreads();
+ tellThreads(aType);
+ waitForThreads();
+}
+
+void*
+threadLoop(void* ThreadData)
+{
+ // Do work until signaled to stop.
+
+ Ndb* localNdb;
+ StartType tType;
+ ThreadNdb* threadInfo = (ThreadNdb*)ThreadData;
+ int threadNo = threadInfo->ThreadNo;
+ localNdb = new Ndb("TEST_DB");
+ localNdb->init(512);
+ localNdb->waitUntilReady();
+ threadInfo->threadBase = (threadNo * 2000000) + (tNodeId * 260000000);
+
+ for (;;) {
+ while (ThreadStart[threadNo] == stIdle) {
+ NdbSleep_MilliSleep(10);
+ }
+
+ // Check if signal to exit is received
+ if (ThreadStart[threadNo] == stStop) {
+ break;
+ }
+
+ tType = ThreadStart[threadNo];
+ ThreadStart[threadNo] = stIdle;
+ executeThread(tType, localNdb, threadInfo);
+ ThreadReady[threadNo] = 1;
+ }
+
+ delete localNdb;
+ ThreadReady[threadNo] = 1;
+ NdbThread_Exit(0);
+
+ return NULL;
+}
+
+void executeThread(StartType aType, Ndb* aNdbObject, ThreadNdb* threadInfo)
+{
+ // Do all batch job in loop with start specified delay
+ int i, j, k;
+ NdbConnection* tConArray[1024];
+ unsigned int tBase;
+ unsigned int tBase2;
+ int threadId = threadInfo->ThreadNo;
+ int *getValueRowAddress = NULL;
+
+ NdbTimer timer;
+ timer.doStart();
+
+ for (i = 0; i < tNoOfBatchesInLoop; i++) {
+ //tBase = threadBase + (i * tNoOfTransInBatch * tNoOfOpsPerTrans);
+ tBase = threadInfo->threadBase + (i * tNoOfTransInBatch * tNoOfOpsPerTrans);
+ //tCompleted = 0;
+ threadInfo->transactionCompleted = 0;
+
+ for (j = 0; j < tNoOfTransInBatch; j++) {
+ tBase2 = tBase + (j * tNoOfOpsPerTrans);
+ tConArray[j] = aNdbObject->startTransaction();
+ if ( tConArray[j] == NULL && !error_handler(aNdbObject->getNdbError())) {
+ ndbout << endl << "Unable to recover! Quiting now" << endl ;
+ exit (-1) ;
+ return ;
+ }
+
+ for (k = 0; k < tNoOfOpsPerTrans; k++) {
+ //-------------------------------------------------------
+ // Define the operation, but do not execute it yet.
+ //-------------------------------------------------------
+ if(aType == stRead){
+ getValueRowAddress = getAttrValueTable +
+ threadId * tNoOfAttributes * tAttributeSize;
+ }
+ defineOperation(tConArray[j], aType, (tBase2 + k), getValueRowAddress);
+ }
+
+ tConArray[j]->executeAsynchPrepare(Commit, &executeCallback, threadInfo);
+ }
+
+ //-------------------------------------------------------
+ // Now we have defined a set of transactions (= batch), it is now time
+ // to execute all of them.
+ //-------------------------------------------------------
+ aNdbObject->sendPollNdb(3000, 0, 0);
+
+ //while (tCompleted < tNoOfTransInBatch) {
+ while (threadInfo->transactionCompleted < tNoOfTransInBatch) {
+ aNdbObject->pollNdb(3000, 0);
+ ndbout << "threadInfo->transactionCompleted = " <<
+ threadInfo->transactionCompleted << endl;
+ }
+
+ for (j = 0 ; j < tNoOfTransInBatch ; j++) {
+ aNdbObject->closeTransaction(tConArray[j]);
+ }
+
+ // Control the elapsed time since the last batch start.
+ // Wait at least tTimeBetweenBatches milli seconds.
+ timer.doStop();
+ while(timer.elapsedTime() < tTimeBetweenBatches){
+ NdbSleep_MilliSleep(1);
+ timer.doStop();
+ }
+ // Ready to start new batch
+ timer.doStart();
+ }
+ return;
+}
+
+void
+executeCallback(int result, NdbConnection* NdbObject, void* aObject)
+{
+ //tCompleted++;
+ ThreadNdb *threadInfo = (ThreadNdb *)aObject;
+ threadInfo->transactionCompleted++;
+
+ if (result == -1) {
+
+ // Add complete error handling here
+
+ int retCode = flexTimedAsynchErrorData->handleErrorCommon(NdbObject->getNdbError());
+ if (retCode == 1) {
+ if (NdbObject->getNdbError().code != 626 && NdbObject->getNdbError().code != 630){
+ ndbout_c("execute: %s", NdbObject->getNdbError().message);
+ ndbout_c("Error code = %d", NdbObject->getNdbError().code);}
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexAsynch" << endl;
+ } else if (retCode == 3) {
+ /* What can we do here? */
+ ndbout_c("execute: %s", NdbObject->getNdbError().message);
+ }//if(retCode == 3)
+
+ // ndbout << "Error occured in poll:" << NdbObject->getNdbError() <<
+ // " ErrorCode = " << NdbObject->getNdbError() << endl;
+ ndbout << "executeCallback threadInfo->transactionCompleted = " <<
+ threadInfo->transactionCompleted << endl;
+ failed++ ;
+ return;
+ }
+ return;
+}
+
+void
+defineOperation(NdbConnection* localNdbConnection,
+ StartType aType,
+ unsigned int threadBase,
+ int *pRow )
+{
+ NdbOperation* localNdbOperation;
+ unsigned int loopCountAttributes = tNoOfAttributes;
+ unsigned int countAttributes;
+ int attrValue[MAXATTRSIZE];
+
+ //-------------------------------------------------------
+ // Set-up the attribute values for this operation.
+ //-------------------------------------------------------
+ for (int k = 0; k < loopCountAttributes; k++) {
+ *(int *)&attrValue[k] = (int)threadBase;
+ }
+ localNdbOperation = localNdbConnection->getNdbOperation(tableName[0]);
+ if (localNdbOperation == NULL) {
+ error_handler(localNdbOperation->getNdbError()) ;
+ }
+
+ switch (aType) {
+ case stInsert: { // Insert case
+ if (theWriteFlag == 1 && theDirtyFlag == 1) {
+ localNdbOperation->dirtyWrite();
+ } else if (theWriteFlag == 1) {
+ localNdbOperation->writeTuple();
+ } else {
+ localNdbOperation->insertTuple();
+ }
+ break;
+ }
+ case stRead: { // Read Case
+ if (theSimpleFlag == 1) {
+ localNdbOperation->simpleRead();
+ } else if (theDirtyFlag == 1) {
+ localNdbOperation->dirtyRead();
+ } else {
+ localNdbOperation->readTuple();
+ }
+ break;
+ }
+ case stUpdate: { // Update Case
+ if (theWriteFlag == 1 && theDirtyFlag == 1) {
+ localNdbOperation->dirtyWrite();
+ } else if (theWriteFlag == 1) {
+ localNdbOperation->writeTuple();
+ } else if (theDirtyFlag == 1) {
+ localNdbOperation->dirtyUpdate();
+ } else {
+ localNdbOperation->updateTuple();
+ }
+ break;
+ }
+ case stDelete: { // Delete Case
+ localNdbOperation->deleteTuple();
+ break;
+ }
+ default: {
+ error_handler(localNdbOperation->getNdbError());
+ }
+ }
+
+ localNdbOperation->equal((char*)attrName[0],(char*)&attrValue[0]);
+
+ switch (aType) {
+ case stInsert: // Insert case
+ case stUpdate: // Update Case
+ {
+ for (countAttributes = 1; countAttributes < loopCountAttributes; countAttributes++) {
+ localNdbOperation->setValue( (char*)attrName[countAttributes],(char*)&attrValue[0]);
+ }
+ break;
+ }
+ case stRead: { // Read Case
+ for (countAttributes = 1; countAttributes < loopCountAttributes; countAttributes++) {
+ //localNdbOperation->getValue((char*)attrName[countAttributes],(char*)&attrValue[0]);
+ localNdbOperation->getValue((char*)attrName[countAttributes],
+ (char *) (pRow + countAttributes*tAttributeSize));
+ }
+ break;
+ }
+ case stDelete: { // Delete Case
+ break;
+ }
+ default: {
+ error_handler(localNdbOperation->getNdbError());
+ }
+ }
+ return;
+}
+
+void readArguments(int argc, const char** argv)
+{
+ int i = 1;
+ while (argc > 1)
+ {
+ if (strcmp(argv[i], "-t") == 0)
+ {
+ tNoOfThreads = atoi(argv[i+1]);
+ // if ((tNoOfThreads < 1) || (tNoOfThreads > MAXTHREADS))
+ if ((tNoOfThreads < 1) || (tNoOfThreads > MAXTHREADS))
+ exit(-1);
+ }
+ else if (strcmp(argv[i], "-i") == 0)
+ {
+ tTimeBetweenBatches = atoi(argv[i+1]);
+ if (tTimeBetweenBatches < 0)
+ exit(-1);
+ }
+ else if (strcmp(argv[i], "-p") == 0)
+ {
+ tNoOfTransInBatch = atoi(argv[i+1]);
+ //if ((tNoOfTransInBatch < 1) || (tNoOfTransInBatch > MAXTHREADS))
+ if ((tNoOfTransInBatch < 1) || (tNoOfTransInBatch > 10000))
+ exit(-1);
+ }
+ else if (strcmp(argv[i], "-c") == 0)
+ {
+ tNoOfOpsPerTrans = atoi(argv[i+1]);
+ if (tNoOfOpsPerTrans < 1)
+ exit(-1);
+ }
+ else if (strcmp(argv[i], "-o") == 0)
+ {
+ tNoOfBatchesInLoop = atoi(argv[i+1]);
+ if (tNoOfBatchesInLoop < 1)
+ exit(-1);
+ }
+ else if (strcmp(argv[i], "-a") == 0)
+ {
+ tNoOfAttributes = atoi(argv[i+1]);
+ if ((tNoOfAttributes < 2) || (tNoOfAttributes > MAXATTR))
+ exit(-1);
+ }
+ else if (strcmp(argv[i], "-n") == 0)
+ {
+ theStdTableNameFlag = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-l") == 0)
+ {
+ tNoOfLoops = atoi(argv[i+1]);
+ if ((tNoOfLoops < 0) || (tNoOfLoops > 100000))
+ exit(-1);
+ }
+ else if (strcmp(argv[i], "-s") == 0)
+ {
+ tAttributeSize = atoi(argv[i+1]);
+ if ((tAttributeSize < 1) || (tAttributeSize > MAXATTRSIZE))
+ exit(-1);
+ }
+ else if (strcmp(argv[i], "-simple") == 0)
+ {
+ theSimpleFlag = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-write") == 0)
+ {
+ theWriteFlag = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-dirty") == 0)
+ {
+ theDirtyFlag = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-test") == 0)
+ {
+ theTestFlag = 1;
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-temp") == 0)
+ {
+ theTempFlag = 0; // 0 if temporary tables.
+ argc++;
+ i--;
+ }
+ else if (strcmp(argv[i], "-no_table_create") == 0)
+ {
+ theTableCreateFlag = 1;
+ argc++;
+ i--;
+ }
+ else
+ {
+ ndbout << "Arguments: " << endl;
+ ndbout << "-t Number of threads to start, i.e., number of parallel loops, default 1 " << endl;
+ ndbout << "-p Number of transactions in a batch, default 32 " << endl;
+ ndbout << "-o Number of batches per loop, default 200 " << endl;
+ ndbout << "-i Minimum time between batch starts in milli seconds, default 0 " << endl;
+ ndbout << "-l Number of loops to run, default 1, 0=infinite " << endl;
+ ndbout << "-a Number of attributes, default 25 " << endl;
+ ndbout << "-c Number of operations per transaction, default 1 " << endl;
+ ndbout << "-s Size of each attribute in 32 bit word, default 1"
+ "(Primary Key is always of size 1, independent of this value) " << endl;
+ ndbout << "-simple Use simple read to read from database " << endl;
+ ndbout << "-dirty Use dirty read to read from database " << endl;
+ ndbout << "-write Use writeTuple in insert and update " << endl;
+ ndbout << "-n Use standard table names " << endl;
+ ndbout << "-no_table_create Don't create tables in db " << endl;
+ ndbout << "-temp Use temporary tables, no writing to disk. " << endl;
+ exit(-1);
+ }
+
+ argc -= 2;
+ i = i + 2;
+ }
+}
+
+void setAttrNames()
+{
+ int i;
+
+ for (i = 0; i < MAXATTR ; i++)
+ {
+ sprintf(attrName[i], "COL%d", i);
+ }
+}
+
+
+void setTableNames()
+{
+ // Note! Uses only uppercase letters in table name's
+ // so that we can look at the tables with SQL
+ int i;
+ for (i = 0; i < MAXTABLES ; i++)
+ {
+ if (theStdTableNameFlag==1)
+ {
+ sprintf(tableName[i], "TAB%d_%d", tNoOfAttributes,
+ NdbTick_CurrentMillisecond()/1000);
+ } else {
+ sprintf(tableName[i], "TAB%d_%d", tNoOfAttributes, tAttributeSize*4);
+ }
+ }
+}
+
+void createTables(Ndb* pMyNdb)
+{
+
+ NdbSchemaCon *MySchemaTransaction;
+ NdbSchemaOp *MySchemaOp;
+ int check;
+
+ if (theTableCreateFlag == 0)
+ {
+ for(int i=0; i < 1 ;i++)
+ {
+ ndbout << "Creating " << tableName[i] << "..." << endl;
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+
+ if( MySchemaTransaction ==
+ NULL && (!error_handler(MySchemaTransaction->getNdbError())))
+ exit(-1) ;/*goto error_handler; <epaulsa*/
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL
+ && (!error_handler(MySchemaTransaction->getNdbError())))
+ exit(-1) ;
+
+ check = MySchemaOp->createTable( tableName[i],
+ 8, // Table Size
+ TupleKey, // Key Type
+ 40, // Nr of Pages
+ All, // FragmentType
+ 6,
+ 78,
+ 80,
+ 1, // MemoryType
+ theTempFlag // 0 if temporary tables else 1
+ );
+
+ if ( check == -1 && (!error_handler(MySchemaTransaction->getNdbError())))
+ exit(-1) ; /* epaulsa > goto error_handler; < epaulsa */
+
+
+ check = MySchemaOp->createAttribute( (char*)attrName[0],
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if ( check == -1 &&(!error_handler(MySchemaTransaction->getNdbError())))
+ exit(-1) ; /* epaulsa > goto error_handler; < epaulsa */
+
+ for (int j = 1; j < tNoOfAttributes ; j++)
+ {
+ check = MySchemaOp->createAttribute( (char*)attrName[j],
+ NoKey,
+ 32,
+ tAttributeSize,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if ( check == -1
+ && (!error_handler(MySchemaTransaction->getNdbError())))
+ exit(-1) ; /* epaulsa > goto error_handler; < epaulsa */
+ }
+
+ if ( MySchemaTransaction->execute() == -1
+ &&(!error_handler(MySchemaTransaction->getNdbError())))
+ exit(-1) ; /* epaulsa > goto error_handler; < epaulsa */
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+ }
+ }
+
+ return;
+}
+
+bool error_handler(const NdbError & err) {
+ ndbout << err << endl ;
+ if ( 4008==err.code || 721==err.code || 266==err.code ){
+ ndbout << endl << "Attempting to recover and continue now..." << endl ;
+ return true ; // return true to retry
+ }
+ return false ; // return false to abort
+}
+
+
+//*******************************************************************************************
+
+
+
+
+
diff --git a/ndb/test/ndbapi/flex_bench_mysql/Makefile b/ndb/test/ndbapi/flex_bench_mysql/Makefile
new file mode 100644
index 00000000000..d2608526cae
--- /dev/null
+++ b/ndb/test/ndbapi/flex_bench_mysql/Makefile
@@ -0,0 +1,15 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := flex_bench_mysql
+
+# Source files of non-templated classes (.C files)
+SOURCES = flex_bench_mysql.cpp
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/../include)
+BIN_TARGET_LIBS_DIRS += $(NDB_TOP)/../libmysql_r/.libs
+BIN_TARGET_LIBS += z mysqlclient_r
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/flex_bench_mysql/flex_bench_mysql.cpp b/ndb/test/ndbapi/flex_bench_mysql/flex_bench_mysql.cpp
new file mode 100644
index 00000000000..6a00463339b
--- /dev/null
+++ b/ndb/test/ndbapi/flex_bench_mysql/flex_bench_mysql.cpp
@@ -0,0 +1,1753 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/* ***************************************************
+FLEXBENCH
+Perform benchmark of insert, update and delete transactions
+
+Arguments:
+ -t Number of threads to start, default 1
+ -o Number of operations per loop, default 500
+ -l Number of loops to run, default 1, 0=infinite
+ -a Number of attributes, default 25
+ -c Number of tables, default 1
+ -s Size of each attribute, default 1 (Primary Key is always of size 1,
+ independent of this value)
+ -lkn Number of long primary keys, default 1
+ -lks Size of each long primary key, default 1
+ -simple Use simple read to read from database
+ -dirty Use dirty read to read from database
+ -write Use writeTuple in insert and update
+ -stdtables Use standard table names
+ -no_table_create Don't create tables in db
+ -sleep Sleep a number of seconds before running the test, this
+ can be used so that another flexBench have time to create tables
+ -temp Use tables without logging
+ -verify Verify inserts, updates and deletes
+ -use_ndb Use NDB API, otherwise use mysql client
+#ifdef CEBIT_STAT
+ -statserv host:port statistics server to report to
+ -statfreq ops report every ops operations (default 100)
+#endif
+ Returns:
+ 0 - Test passed
+ 1 - Test failed
+ 2 - Invalid arguments
+
+* *************************************************** */
+
+#define USE_MYSQL
+#ifdef USE_MYSQL
+#include <mysql.h>
+#endif
+
+#include "NdbApi.hpp"
+
+#include <string.h>
+#include <assert.h>
+
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbStdio.h>
+#include <NdbTick.h>
+#include <NdbTimer.hpp>
+#include <NdbThread.h>
+#include <NdbAutoPtr.hpp>
+
+#include <NdbTest.hpp>
+
+#define MAXSTRLEN 16
+#define MAXATTR 64
+#define MAXTABLES 128
+#define MAXATTRSIZE 1000
+#define MAXNOLONGKEY 16 // Max number of long keys.
+#define MAXLONGKEYTOTALSIZE 1023 // words = 4092 bytes
+
+extern "C" { static void* flexBenchThread(void*); }
+static int readArguments(int argc, const char** argv);
+#ifdef USE_MYSQL
+static int createTables(MYSQL*);
+static int dropTables(MYSQL*);
+#endif
+static int createTables(Ndb*);
+static void sleepBeforeStartingTest(int seconds);
+static void input_error();
+
+enum StartType {
+ stIdle,
+ stInsert,
+ stVerify,
+ stRead,
+ stUpdate,
+ stDelete,
+ stTryDelete,
+ stVerifyDelete,
+ stStop
+};
+
+struct ThreadData
+{
+ int threadNo;
+ NdbThread* threadLife;
+ int threadReady;
+ StartType threadStart;
+ int threadResult;
+};
+
+static int tNodeId = 0 ;
+static char tableName[MAXTABLES][MAXSTRLEN+1];
+static char attrName[MAXATTR][MAXSTRLEN+1];
+static char** longKeyAttrName;
+
+// Program Parameters
+static int tNoOfLoops = 1;
+static int tAttributeSize = 1;
+static unsigned int tNoOfThreads = 1;
+static unsigned int tNoOfTables = 1;
+static unsigned int tNoOfAttributes = 25;
+static unsigned int tNoOfOperations = 500;
+static unsigned int tSleepTime = 0;
+static unsigned int tNoOfLongPK = 1;
+static unsigned int tSizeOfLongPK = 1;
+static unsigned int t_instances = 1;
+
+//Program Flags
+static int theSimpleFlag = 0;
+static int theDirtyFlag = 0;
+static int theWriteFlag = 0;
+static int theStdTableNameFlag = 0;
+static int theTableCreateFlag = 0;
+static bool theTempTable = false;
+static bool VerifyFlag = true;
+static bool useLongKeys = false;
+static bool verbose = false;
+#ifdef USE_MYSQL
+static bool use_ndb = false;
+static int engine_id = 0;
+static int sockets[16];
+static int n_sockets = 0;
+static char* engine[] =
+ {
+ " ENGINE = NDBCLUSTER ", // use default engine
+ " ENGINE = MEMORY ",
+ " ENGINE = MYISAM ",
+ " ENGINE = INNODB "
+ };
+#else
+static bool use_ndb = true;
+#endif
+
+static ErrorData theErrorData; // Part of flexBench-program
+
+#define START_TIMER { NdbTimer timer; timer.doStart();
+#define STOP_TIMER timer.doStop();
+#define PRINT_TIMER(text, trans, opertrans) timer.printTransactionStatistics(text, trans, opertrans); };
+
+#include <NdbTCP.h>
+
+#ifdef CEBIT_STAT
+#include <NdbMutex.h>
+static bool statEnable = false;
+static char statHost[100];
+static int statFreq = 100;
+static int statPort = 0;
+static int statSock = -1;
+static enum { statError = -1, statClosed, statOpen } statState;
+static NdbMutex statMutex = NDB_MUTEX_INITIALIZER;
+#endif
+
+//-------------------------------------------------------------------
+// Statistical Reporting routines
+//-------------------------------------------------------------------
+#ifdef CEBIT_STAT
+// Experimental client-side statistic for CeBIT
+
+static void
+statReport(enum StartType st, int ops)
+{
+ if (!statEnable)
+ return;
+ if (NdbMutex_Lock(&statMutex) < 0) {
+ if (statState != statError) {
+ ndbout_c("stat: lock mutex failed: %s", strerror(errno));
+ statState = statError;
+ }
+ return;
+ }
+ static int nodeid;
+ // open connection
+ if (statState != statOpen) {
+ char *p = getenv("NDB_NODEID"); // ndbnet sets NDB_NODEID
+ nodeid = p == 0 ? 0 : atoi(p);
+ if ((statSock = socket(PF_INET, SOCK_STREAM, 0)) < 0) {
+ if (statState != statError) {
+ ndbout_c("stat: create socket failed: %s", strerror(errno));
+ statState = statError;
+ }
+ (void)NdbMutex_Unlock(&statMutex);
+ return;
+ }
+ struct sockaddr_in saddr;
+ memset(&saddr, 0, sizeof(saddr));
+ saddr.sin_family = AF_INET;
+ saddr.sin_port = htons(statPort);
+ if (Ndb_getInAddr(&saddr.sin_addr, statHost) < 0) {
+ if (statState != statError) {
+ ndbout_c("stat: host %s not found", statHost);
+ statState = statError;
+ }
+ (void)close(statSock);
+ (void)NdbMutex_Unlock(&statMutex);
+ return;
+ }
+ if (connect(statSock, (struct sockaddr *)&saddr, sizeof(saddr)) < 0) {
+ if (statState != statError) {
+ ndbout_c("stat: connect failed: %s", strerror(errno));
+ statState = statError;
+ }
+ (void)close(statSock);
+ (void)NdbMutex_Unlock(&statMutex);
+ return;
+ }
+ statState = statOpen;
+ ndbout_c("stat: connection to %s:%d opened", statHost, (int)statPort);
+ }
+ const char *text;
+ switch (st) {
+ case stInsert:
+ text = "insert";
+ break;
+ case stVerify:
+ text = "verify";
+ break;
+ case stRead:
+ text = "read";
+ break;
+ case stUpdate:
+ text = "update";
+ break;
+ case stDelete:
+ text = "delete";
+ break;
+ case stVerifyDelete:
+ text = "verifydelete";
+ break;
+ default:
+ text = "unknown";
+ break;
+ }
+ char buf[100];
+ sprintf(buf, "%d %s %d\n", nodeid, text, ops);
+ int len = strlen(buf);
+ // assume SIGPIPE already ignored
+ if (write(statSock, buf, len) != len) {
+ if (statState != statError) {
+ ndbout_c("stat: write failed: %s", strerror(errno));
+ statState = statError;
+ }
+ (void)close(statSock);
+ (void)NdbMutex_Unlock(&statMutex);
+ return;
+ }
+ (void)NdbMutex_Unlock(&statMutex);
+}
+#endif // CEBIT_STAT
+
+static void
+resetThreads(ThreadData* pt){
+ for (unsigned int i = 0; i < tNoOfThreads; i++){
+ pt[i].threadReady = 0;
+ pt[i].threadResult = 0;
+ pt[i].threadStart = stIdle;
+ }
+}
+
+static int
+checkThreadResults(ThreadData* pt){
+ for (unsigned int i = 0; i < tNoOfThreads; i++){
+ if(pt[i].threadResult != 0){
+ ndbout_c("Thread%d reported fatal error %d", i, pt[i].threadResult);
+ return -1;
+ }
+ }
+ return 0;
+}
+
+static
+void
+waitForThreads(ThreadData* pt)
+{
+ int cont = 1;
+ while (cont){
+ NdbSleep_MilliSleep(100);
+ cont = 0;
+ for (unsigned int i = 0; i < tNoOfThreads; i++){
+ if (pt[i].threadReady == 0)
+ cont = 1;
+ }
+ }
+}
+
+static void
+tellThreads(ThreadData* pt, StartType what)
+{
+ for (unsigned int i = 0; i < tNoOfThreads; i++)
+ pt[i].threadStart = what;
+}
+
+NDB_COMMAND(flexBench, "flexBench", "flexBench", "flexbench", 65535)
+{
+ ThreadData* pThreadsData;
+ int tLoops = 0;
+ int returnValue = NDBT_OK;
+ if (readArguments(argc, argv) != 0){
+ input_error();
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ NdbAutoPtr<char> p10;
+ if(useLongKeys){
+ int e1 = sizeof(char*) * tNoOfLongPK;
+ int e2_1 = strlen("KEYATTR ") + 1;
+ int e2 = e2_1 * tNoOfLongPK;
+ char *tmp = (char *) malloc(e1 + e2);
+ p10.reset(tmp);
+ longKeyAttrName = (char **) tmp;
+ tmp += e1;
+ for (Uint32 i = 0; i < tNoOfLongPK; i++) {
+ // longKeyAttrName[i] = (char *) malloc(strlen("KEYATTR ") + 1);
+ longKeyAttrName[i] = tmp;
+ tmp += e2_1;
+ memset(longKeyAttrName[i], 0, e2_1);
+ sprintf(longKeyAttrName[i], "KEYATTR%i", i);
+ }
+ }
+
+ NdbAutoObjArrayPtr<ThreadData>
+ p12( pThreadsData = new ThreadData[tNoOfThreads] );
+
+
+ ndbout << endl << "FLEXBENCH - Starting normal mode" << endl;
+ ndbout << "Perform benchmark of insert, update and delete transactions"<< endl;
+ ndbout << " " << tNoOfThreads << " thread(s) " << endl;
+ ndbout << " " << tNoOfLoops << " iterations " << endl;
+ ndbout << " " << tNoOfTables << " table(s) and " << 1 << " operation(s) per transaction " <<endl;
+ ndbout << " " << tNoOfAttributes << " attributes per table " << endl;
+ ndbout << " " << tNoOfOperations << " transaction(s) per thread and round " << endl;
+ ndbout << " " << tAttributeSize << " is the number of 32 bit words per attribute "<< endl;
+ ndbout << " " << "Table(s) without logging: " << (Uint32)theTempTable << endl;
+
+ if(useLongKeys)
+ ndbout << " " << "Using long keys with " << tNoOfLongPK << " keys a' " <<
+ tSizeOfLongPK * 4 << " bytes each." << endl;
+
+ ndbout << " " << "Verification is " ;
+ if(VerifyFlag) {
+ ndbout << "enabled" << endl ;
+ }else{
+ ndbout << "disabled" << endl ;
+ }
+ if (use_ndb) {
+ ndbout << "Use NDB API with NdbPool in this test case" << endl;
+ ndbout << "Pool size = " << t_instances << endl;
+ } else {
+ ndbout << "Use mysql client with " << engine[engine_id];
+ ndbout << " as engine" << endl;
+ }
+ theErrorData.printSettings(ndbout);
+
+ NdbThread_SetConcurrencyLevel(tNoOfThreads + 2);
+
+#ifdef USE_MYSQL
+ MYSQL mysql;
+ if (!use_ndb) {
+ if ( mysql_thread_safe() == 0 ) {
+ ndbout << "Not thread safe mysql library..." << endl;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ ndbout << "Connecting to MySQL..." <<endl;
+
+ mysql_init(&mysql);
+ {
+ int the_socket = sockets[0];
+ char the_socket_name[1024];
+ sprintf(the_socket_name, "%s%u%s", "/tmp/mysql.",the_socket,".sock");
+ // sprintf(the_socket_name, "%s", "/tmp/mysql.sock");
+ ndbout << the_socket_name << endl;
+ if ( mysql_real_connect(&mysql,
+ "localhost",
+ "root",
+ "",
+ "test",
+ the_socket,
+ the_socket_name,
+ 0) == NULL ) {
+ ndbout << "Connect failed" <<endl;
+ returnValue = NDBT_FAILED;
+ }
+ }
+ if(returnValue == NDBT_OK){
+ mysql_set_server_option(&mysql, MYSQL_OPTION_MULTI_STATEMENTS_ON);
+ if (createTables(&mysql) != 0){
+ returnValue = NDBT_FAILED;
+ }
+ }
+ }
+#endif
+ if (use_ndb) {
+ Uint32 ndb_id = 0;
+ if (!create_instance(t_instances, 1, t_instances)) {
+ ndbout << "Creation of the NdbPool failed" << endl;
+ returnValue = NDBT_FAILED;
+ } else {
+ Ndb* pNdb = get_ndb_object(ndb_id, "test", "def");
+ if (pNdb == NULL) {
+ ndbout << "Failed to get a NDB object" << endl;
+ returnValue = NDBT_FAILED;
+ } else {
+ tNodeId = pNdb->getNodeId();
+ ndbout << " NdbAPI node with id = " << tNodeId << endl;
+ ndbout << endl;
+
+ ndbout << "Waiting for ndb to become ready..." <<endl;
+ if (pNdb->waitUntilReady(2000) != 0){
+ ndbout << "NDB is not ready" << endl;
+ ndbout << "Benchmark failed!" << endl;
+ returnValue = NDBT_FAILED;
+ }
+ if(returnValue == NDBT_OK){
+ if (createTables(pNdb) != 0){
+ returnValue = NDBT_FAILED;
+ }
+ }
+ return_ndb_object(pNdb, ndb_id);
+ }
+ }
+ }
+ if(returnValue == NDBT_OK){
+
+ sleepBeforeStartingTest(tSleepTime);
+
+ /****************************************************************
+ * Create threads. *
+ ****************************************************************/
+ resetThreads(pThreadsData);
+
+ for (unsigned int i = 0; i < tNoOfThreads; i++){
+ pThreadsData[i].threadNo = i;
+ pThreadsData[i].threadLife = NdbThread_Create(flexBenchThread,
+ (void**)&pThreadsData[i],
+ 32768,
+ "flexBenchThread",
+ NDB_THREAD_PRIO_LOW);
+ }
+
+ waitForThreads(pThreadsData);
+
+ ndbout << endl << "All threads started" << endl << endl;
+
+ /****************************************************************
+ * Execute program. *
+ ****************************************************************/
+
+ for(;;){
+
+ int loopCount = tLoops + 1;
+ ndbout << endl << "Loop # " << loopCount << endl << endl;
+
+ /****************************************************************
+ * Perform inserts. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give insert-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stInsert);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing insert" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("insert", tNoOfOperations*tNoOfThreads, tNoOfTables);
+ /****************************************************************
+ * Verify inserts. *
+ ****************************************************************/
+ if (VerifyFlag) {
+ resetThreads(pThreadsData);
+ ndbout << "Verifying inserts...\t" ;
+ tellThreads(pThreadsData, stVerify);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed while verifying inserts" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }else{
+ ndbout << "\t\tOK" << endl << endl ;
+ }
+ }
+
+ /****************************************************************
+ * Perform read. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give read-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stRead);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing read" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("read", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ /****************************************************************
+ * Perform update. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give update-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stUpdate);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing update" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("update", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ /****************************************************************
+ * Verify updates. *
+ ****************************************************************/
+ if (VerifyFlag) {
+ resetThreads(pThreadsData);
+ ndbout << "Verifying updates...\t" ;
+ tellThreads(pThreadsData, stVerify);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed while verifying updates" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }else{
+ ndbout << "\t\tOK" << endl << endl ;
+ }
+ }
+
+ /****************************************************************
+ * Perform read. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give read-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stRead);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing read" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("read", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ /****************************************************************
+ * Perform delete. *
+ ****************************************************************/
+ // Reset and start timer
+ START_TIMER;
+ // Give delete-command to all threads
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stDelete);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in performing delete" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }
+ // stop timer and print results.
+ STOP_TIMER;
+ PRINT_TIMER("delete", tNoOfOperations*tNoOfThreads, tNoOfTables);
+
+ /****************************************************************
+ * Verify deletes. *
+ ****************************************************************/
+ if (VerifyFlag) {
+ resetThreads(pThreadsData);
+ ndbout << "Verifying tuple deletion..." ;
+ tellThreads(pThreadsData, stVerifyDelete);
+ waitForThreads(pThreadsData);
+ if (checkThreadResults(pThreadsData) != 0){
+ ndbout << "Error: Threads failed in verifying deletes" << endl;
+ returnValue = NDBT_FAILED;
+ break;
+ }else{
+ ndbout << "\t\tOK" << endl << endl ;
+ }
+ }
+
+ ndbout << "--------------------------------------------------" << endl;
+
+ tLoops++;
+
+ if ( 0 != tNoOfLoops && tNoOfLoops <= tLoops )
+ break;
+ theErrorData.printErrorCounters();
+ }
+
+ resetThreads(pThreadsData);
+ tellThreads(pThreadsData, stStop);
+ waitForThreads(pThreadsData);
+
+ void * tmp;
+ for(Uint32 i = 0; i<tNoOfThreads; i++){
+ NdbThread_WaitFor(pThreadsData[i].threadLife, &tmp);
+ NdbThread_Destroy(&pThreadsData[i].threadLife);
+ }
+ }
+#ifdef USE_MYSQL
+ if (!use_ndb) {
+ dropTables(&mysql);
+ mysql_close(&mysql);
+ }
+#endif
+ if (use_ndb) {
+ drop_instance();
+ }
+ theErrorData.printErrorCounters();
+ return NDBT_ProgramExit(returnValue);
+}
+////////////////////////////////////////
+
+
+unsigned long get_hash(unsigned long * hash_key, int len)
+{
+ unsigned long hash_value = 147;
+ unsigned h_key;
+ int i;
+ for (i = 0; i < len; i++)
+ {
+ h_key = hash_key[i];
+ hash_value = (hash_value << 5) + hash_value + (h_key & 255);
+ hash_value = (hash_value << 5) + hash_value + ((h_key >> 8) & 255);
+ hash_value = (hash_value << 5) + hash_value + ((h_key >> 16) & 255);
+ hash_value = (hash_value << 5) + hash_value + ((h_key >> 24) & 255);
+ }
+ return hash_value;
+}
+
+// End of warming up phase
+
+
+
+static void* flexBenchThread(void* pArg)
+{
+ ThreadData* pThreadData = (ThreadData*)pArg;
+ unsigned int threadNo, threadBase;
+ Ndb* pNdb = NULL ;
+ Uint32 ndb_id = 0;
+ NdbConnection *pTrans = NULL ;
+ NdbOperation** pOps = NULL ;
+ StartType tType ;
+ StartType tSaveType ;
+ NdbRecAttr* tTmp = NULL ;
+ int* attrValue = NULL ;
+ int* attrRefValue = NULL ;
+ int check = 0 ;
+ int loopCountOps, loopCountTables, loopCountAttributes;
+ int tAttemptNo = 0;
+ int tRetryAttempts = 20;
+ int tResult = 0;
+ int tSpecialTrans = 0;
+ int nRefLocalOpOffset = 0 ;
+ int nReadBuffSize =
+ tNoOfTables * tNoOfAttributes * sizeof(int) * tAttributeSize ;
+ int nRefBuffSize =
+ tNoOfOperations * tNoOfAttributes * sizeof(int) * tAttributeSize ;
+ unsigned*** longKeyAttrValue = NULL;
+
+
+ threadNo = pThreadData->threadNo ;
+
+#ifdef USE_MYSQL
+ MYSQL mysql;
+ int the_socket = sockets[threadNo % n_sockets];
+ char the_socket_name[1024];
+ //sprintf(the_socket_name, "%s", "/tmp/mysql.sock");
+ sprintf(the_socket_name, "%s%u%s", "/tmp/mysql.",the_socket,".sock");
+ if (!use_ndb) {
+ ndbout << the_socket_name << endl;
+ ndbout << "Thread connecting to MySQL... " << endl;
+ mysql_init(&mysql);
+
+ if ( mysql_real_connect(&mysql,
+ "localhost",
+ "root",
+ "",
+ "test",
+ the_socket,
+ the_socket_name,
+ 0) == NULL ) {
+ ndbout << "failed" << endl;
+ NdbThread_Exit(0) ;
+ }
+ ndbout << "ok" << endl;
+
+ int r;
+ if (tNoOfTables > 1)
+ r = mysql_autocommit(&mysql, 0);
+ else
+ r = mysql_autocommit(&mysql, 1);
+
+ if (r) {
+ ndbout << "autocommit on/off failed" << endl;
+ NdbThread_Exit(0) ;
+ }
+ }
+#endif
+
+ NdbAutoPtr<int> p00( attrValue= (int*)malloc(nReadBuffSize) ) ;
+ NdbAutoPtr<int> p01( attrRefValue= (int*)malloc(nRefBuffSize) );
+ if (use_ndb) {
+ pOps = (NdbOperation**)malloc(tNoOfTables*sizeof(NdbOperation*)) ;
+ }
+ NdbAutoPtr<NdbOperation*> p02( pOps );
+
+ if( !attrValue || !attrRefValue ||
+ ( use_ndb && ( !pOps) ) ){
+ // Check allocations to make sure we got all the memory we asked for
+ ndbout << "One or more memory allocations failed when starting thread #";
+ ndbout << threadNo << endl ;
+ ndbout << "Thread #" << threadNo << " will now exit" << endl ;
+ tResult = 13 ;
+ NdbThread_Exit(0) ;
+ }
+
+ if (use_ndb) {
+ pNdb = get_ndb_object(ndb_id, "test", "def");
+ if (pNdb == NULL) {
+ ndbout << "Failed to get an NDB object" << endl;
+ ndbout << "Thread #" << threadNo << " will now exit" << endl ;
+ tResult = 13;
+ NdbThread_Exit(0) ;
+ }
+ pNdb->waitUntilReady();
+ return_ndb_object(pNdb, ndb_id);
+ pNdb = NULL;
+ }
+
+ // To make sure that two different threads doesn't operate on the same record
+ // Calculate an "unique" number to use as primary key
+ threadBase = (threadNo * 2000000) + (tNodeId * 260000000);
+
+ NdbAutoPtr<char> p22;
+ if(useLongKeys){
+ // Allocate and populate the longkey array.
+ int e1 = sizeof(unsigned**) * tNoOfOperations;
+ int e2 = sizeof(unsigned*) * tNoOfLongPK * tNoOfOperations;
+ int e3 = sizeof(unsigned) * tSizeOfLongPK * tNoOfLongPK * tNoOfOperations;
+ char* tmp;
+ p22.reset(tmp = (char*)malloc(e1+e2+e3));
+
+ longKeyAttrValue = (unsigned ***) tmp;
+ tmp += e1;
+ for (Uint32 n = 0; n < tNoOfOperations; n++) {
+ longKeyAttrValue[n] = (unsigned **) tmp;
+ tmp += sizeof(unsigned*) * tNoOfLongPK;
+ }
+
+ for (Uint32 n = 0; n < tNoOfOperations; n++){
+ for (Uint32 i = 0; i < tNoOfLongPK ; i++) {
+ longKeyAttrValue[n][i] = (unsigned *) tmp;
+ tmp += sizeof(unsigned) * tSizeOfLongPK;
+ memset(longKeyAttrValue[n][i], 0, sizeof(unsigned) * tSizeOfLongPK);
+ for(Uint32 j = 0; j < tSizeOfLongPK; j++) {
+ // Repeat the unique value to fill up the long key.
+ longKeyAttrValue[n][i][j] = threadBase + n;
+ }
+ }
+ }
+ }
+
+ int nRefOpOffset = 0 ;
+ //Assign reference attribute values to memory
+ for(Uint32 ops = 1 ; ops < tNoOfOperations ; ops++){
+ // Calculate offset value before going into the next loop
+ nRefOpOffset = tAttributeSize*tNoOfAttributes*(ops-1) ;
+ for(Uint32 a = 0 ; a < tNoOfAttributes ; a++){
+ *(int*)&attrRefValue[nRefOpOffset + tAttributeSize*a] =
+ (int)(threadBase + ops + a) ;
+ }
+ }
+
+#ifdef CEBIT_STAT
+ // ops not yet reported
+ int statOps = 0;
+#endif
+
+#ifdef USE_MYSQL
+ // temporary buffer to store prepared statement text
+ char buf[2048];
+ MYSQL_STMT** prep_read = NULL;
+ MYSQL_STMT** prep_delete = NULL;
+ MYSQL_STMT** prep_update = NULL;
+ MYSQL_STMT** prep_insert = NULL;
+ MYSQL_BIND* bind_delete = NULL;
+ MYSQL_BIND* bind_read = NULL;
+ MYSQL_BIND* bind_update = NULL;
+ MYSQL_BIND* bind_insert = NULL;
+ int* mysql_data = NULL;
+
+ NdbAutoPtr<char> p21;
+
+ if (!use_ndb) {
+ // data array to which prepared statements are bound
+ char* tmp;
+ int e1 = sizeof(int)*tAttributeSize*tNoOfAttributes;
+ int e2 = sizeof(MYSQL_BIND)*tNoOfAttributes;
+ int e3 = sizeof(MYSQL_BIND)*tNoOfAttributes;
+ int e4 = sizeof(MYSQL_BIND)*tNoOfAttributes;
+ int e5 = sizeof(MYSQL_BIND)*1;
+ int e6 = sizeof(MYSQL_STMT*)*tNoOfTables;
+ int e7 = sizeof(MYSQL_STMT*)*tNoOfTables;
+ int e8 = sizeof(MYSQL_STMT*)*tNoOfTables;
+ int e9 = sizeof(MYSQL_STMT*)*tNoOfTables;
+ p21.reset(tmp = (char*)malloc(e1+e2+e3+e4+e5+e6+e7+e8+e9));
+
+ mysql_data = (int*)tmp; tmp += e1;
+ bind_insert = (MYSQL_BIND*)tmp; tmp += e2;
+ bind_update = (MYSQL_BIND*)tmp; tmp += e3;
+ bind_read = (MYSQL_BIND*)tmp; tmp += e4;
+ bind_delete = (MYSQL_BIND*)tmp; tmp += e5;
+ prep_insert = (MYSQL_STMT**)tmp; tmp += e6;
+ prep_update = (MYSQL_STMT**)tmp; tmp += e7;
+ prep_read = (MYSQL_STMT**)tmp; tmp += e8;
+ prep_delete = (MYSQL_STMT**)tmp;
+
+ for (Uint32 ca = 0; ca < tNoOfAttributes; ca++){
+ MYSQL_BIND& bi = bind_insert[ca];
+ bi.buffer_type = MYSQL_TYPE_LONG;
+ bi.buffer = (char*)&mysql_data[ca*tAttributeSize];
+ bi.buffer_length = 0;
+ bi.length = NULL;
+ bi.is_null = NULL;
+ }//for
+
+ for (Uint32 ca = 0; ca < tNoOfAttributes; ca++){
+ MYSQL_BIND& bi = bind_update[ca];
+ bi.buffer_type = MYSQL_TYPE_LONG;
+ if ( ca == tNoOfAttributes-1 ) // the primary key comes last in statement
+ bi.buffer = (char*)&mysql_data[0];
+ else
+ bi.buffer = (char*)&mysql_data[(ca+1)*tAttributeSize];
+ bi.buffer_length = 0;
+ bi.length = NULL;
+ bi.is_null = NULL;
+ }//for
+
+ for (Uint32 ca = 0; ca < tNoOfAttributes; ca++){
+ MYSQL_BIND& bi = bind_read[ca];
+ bi.buffer_type = MYSQL_TYPE_LONG;
+ bi.buffer = (char*)&mysql_data[ca*tAttributeSize];
+ bi.buffer_length = 4;
+ bi.length = NULL;
+ bi.is_null = NULL;
+ }//for
+
+ for (Uint32 ca = 0; ca < 1; ca++){
+ MYSQL_BIND& bi = bind_delete[ca];
+ bi.buffer_type = MYSQL_TYPE_LONG;
+ bi.buffer = (char*)&mysql_data[ca*tAttributeSize];
+ bi.buffer_length = 0;
+ bi.length = NULL;
+ bi.is_null = NULL;
+ }//for
+
+ for (Uint32 i = 0; i < tNoOfTables; i++) {
+ int pos = 0;
+ pos += sprintf(buf+pos, "%s%s%s",
+ "INSERT INTO ",
+ tableName[i],
+ " VALUES(");
+ pos += sprintf(buf+pos, "%s", "?");
+ for (Uint32 j = 1; j < tNoOfAttributes; j++) {
+ pos += sprintf(buf+pos, "%s", ",?");
+ }
+ pos += sprintf(buf+pos, "%s", ")");
+ if (verbose)
+ ndbout << buf << endl;
+ prep_insert[i] = mysql_prepare(&mysql, buf, pos);
+ if (prep_insert[i] == 0) {
+ ndbout << "mysql_prepare: " << mysql_error(&mysql) << endl;
+ NdbThread_Exit(0) ;
+ }
+ if (mysql_bind_param(prep_insert[i], bind_insert)) {
+ ndbout << "mysql_bind_param: " << mysql_error(&mysql) << endl;
+ NdbThread_Exit(0) ;
+ }
+ }
+
+ for (Uint32 i = 0; i < tNoOfTables; i++) {
+ int pos = 0;
+ pos += sprintf(buf+pos, "%s%s%s",
+ "UPDATE ",
+ tableName[i],
+ " SET ");
+ for (Uint32 j = 1; j < tNoOfAttributes; j++) {
+ if (j != 1)
+ pos += sprintf(buf+pos, "%s", ",");
+ pos += sprintf(buf+pos, "%s%s", attrName[j],"=?");
+ }
+ pos += sprintf(buf+pos, "%s%s%s", " WHERE ", attrName[0], "=?");
+
+ if (verbose)
+ ndbout << buf << endl;
+ prep_update[i] = mysql_prepare(&mysql, buf, pos);
+ if (prep_update[i] == 0) {
+ ndbout << "mysql_prepare: " << mysql_error(&mysql) << endl;
+ NdbThread_Exit(0) ;
+ }
+ if (mysql_bind_param(prep_update[i], bind_update)) {
+ ndbout << "mysql_bind_param: " << mysql_error(&mysql) << endl;
+ NdbThread_Exit(0) ;
+ }
+ }
+
+ for (Uint32 i = 0; i < tNoOfTables; i++) {
+ int pos = 0;
+ pos += sprintf(buf+pos, "%s", "SELECT ");
+ for (Uint32 j = 1; j < tNoOfAttributes; j++) {
+ if (j != 1)
+ pos += sprintf(buf+pos, "%s", ",");
+ pos += sprintf(buf+pos, "%s", attrName[j]);
+ }
+ pos += sprintf(buf+pos, "%s%s%s%s%s",
+ " FROM ",
+ tableName[i],
+ " WHERE ",
+ attrName[0],
+ "=?");
+ if (verbose)
+ ndbout << buf << endl;
+ prep_read[i] = mysql_prepare(&mysql, buf, pos);
+ if (prep_read[i] == 0) {
+ ndbout << "mysql_prepare: " << mysql_error(&mysql) << endl;
+ NdbThread_Exit(0) ;
+ }
+ if (mysql_bind_param(prep_read[i], bind_read)) {
+ ndbout << "mysql_bind_param: " << mysql_error(&mysql) << endl;
+ NdbThread_Exit(0) ;
+ }
+ if (mysql_bind_result(prep_read[i], &bind_read[1])) {
+ ndbout << "mysql_bind_result: " << mysql_error(&mysql) << endl;
+ NdbThread_Exit(0) ;
+ }
+ }
+
+ for (Uint32 i = 0; i < tNoOfTables; i++) {
+ int pos = 0;
+ pos += sprintf(buf+pos, "%s%s%s%s%s",
+ "DELETE FROM ",
+ tableName[i],
+ " WHERE ",
+ attrName[0],
+ "=?");
+ if (verbose)
+ ndbout << buf << endl;
+ prep_delete[i] = mysql_prepare(&mysql, buf, pos);
+ if (prep_delete[i] == 0) {
+ ndbout << "mysql_prepare: " << mysql_error(&mysql) << endl;
+ NdbThread_Exit(0) ;
+ }
+ if (mysql_bind_param(prep_delete[i], bind_delete)) {
+ ndbout << "mysql_bind_param: " << mysql_error(&mysql) << endl;
+ NdbThread_Exit(0) ;
+ }
+ }
+ }
+#endif
+
+ for (;;) {
+ pThreadData->threadResult = tResult; // Report error to main thread,
+ // normally tResult is set to 0
+ pThreadData->threadReady = 1;
+
+ while (pThreadData->threadStart == stIdle){
+ NdbSleep_MilliSleep(100);
+ }//while
+
+ // Check if signal to exit is received
+ if (pThreadData->threadStart == stStop){
+ pThreadData->threadReady = 1;
+ // ndbout_c("Thread%d is stopping", threadNo);
+ // In order to stop this thread, the main thread has signaled
+ // stStop, break out of the for loop so that destructors
+ // and the proper exit functions are called
+ break;
+ }//if
+
+ tType = pThreadData->threadStart;
+ tSaveType = tType;
+ pThreadData->threadStart = stIdle;
+
+ // Start transaction, type of transaction
+ // is received in the array ThreadStart
+ loopCountOps = tNoOfOperations;
+ loopCountTables = tNoOfTables;
+ loopCountAttributes = tNoOfAttributes;
+ for (int count = 1; count < loopCountOps && tResult == 0;){
+
+ if (use_ndb) {
+ pNdb = get_ndb_object(ndb_id, "test", "def");
+ if (pNdb == NULL) {
+ ndbout << "Could not get Ndb object in thread" << threadNo;
+ ndbout << endl;
+ tResult = 1; //Indicate fatal error
+ break;
+ }
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ // This is a fatal error, abort program
+ ndbout << "Could not start transaction in thread" << threadNo;
+ ndbout << endl;
+ ndbout << pNdb->getNdbError() << endl;
+ tResult = 1; // Indicate fatal error
+ break; // Break out of for loop
+ }
+ }
+
+ // Calculate the current operation offset in the reference array
+ nRefLocalOpOffset = tAttributeSize*tNoOfAttributes*(count - 1) ;
+ int* tmpAttrRefValue = attrRefValue + nRefLocalOpOffset;
+
+ for (int countTables = 0;
+ countTables < loopCountTables && tResult == 0;
+ countTables++) {
+
+ int nTableOffset = tAttributeSize *
+ loopCountAttributes *
+ countTables ;
+
+ int* tmpAttrValue = attrValue + nTableOffset;
+
+ if (use_ndb) {
+ pOps[countTables] = pTrans->getNdbOperation(tableName[countTables]);
+ if (pOps[countTables] == NULL) {
+ // This is a fatal error, abort program
+ ndbout << "getNdbOperation: " << pTrans->getNdbError();
+ tResult = 2; // Indicate fatal error
+ break;
+ }//if
+
+ switch (tType) {
+ case stInsert: // Insert case
+ if (theWriteFlag == 1 && theDirtyFlag == 1)
+ pOps[countTables]->dirtyWrite();
+ else if (theWriteFlag == 1)
+ pOps[countTables]->writeTuple();
+ else
+ pOps[countTables]->insertTuple();
+ break;
+ case stRead: // Read Case
+ if (theSimpleFlag == 1)
+ pOps[countTables]->simpleRead();
+ else if (theDirtyFlag == 1)
+ pOps[countTables]->dirtyRead();
+ else
+ pOps[countTables]->readTuple();
+ break;
+ case stUpdate: // Update Case
+ if (theWriteFlag == 1 && theDirtyFlag == 1)
+ pOps[countTables]->dirtyWrite();
+ else if (theWriteFlag == 1)
+ pOps[countTables]->writeTuple();
+ else if (theDirtyFlag == 1)
+ pOps[countTables]->dirtyUpdate();
+ else
+ pOps[countTables]->updateTuple();
+ break;
+ case stDelete: // Delete Case
+ pOps[countTables]->deleteTuple();
+ break;
+ case stVerify:
+ pOps[countTables]->readTuple();
+ break;
+ case stVerifyDelete:
+ pOps[countTables]->readTuple();
+ break;
+ default:
+ assert(false);
+ }//switch
+
+ if(useLongKeys){
+ // Loop the equal call so the complete key is send to the kernel.
+ for(Uint32 i = 0; i < tNoOfLongPK; i++)
+ pOps[countTables]->equal(longKeyAttrName[i],
+ (char *)longKeyAttrValue[count - 1][i],
+ tSizeOfLongPK*4);
+ }
+ else
+ pOps[countTables]->equal((char*)attrName[0],
+ (char*)&tmpAttrRefValue[0]);
+
+ if (tType == stInsert) {
+ for (int ca = 1; ca < loopCountAttributes; ca++){
+ pOps[countTables]->setValue((char*)attrName[ca],
+ (char*)&tmpAttrRefValue[tAttributeSize*ca]);
+ }//for
+ } else if (tType == stUpdate) {
+ for (int ca = 1; ca < loopCountAttributes; ca++){
+ int* tmp = (int*)&tmpAttrRefValue[tAttributeSize*ca];
+ if (countTables == 0)
+ (*tmp)++;
+ pOps[countTables]->setValue((char*)attrName[ca],(char*)tmp);
+ }//for
+ } else if (tType == stRead || stVerify == tType) {
+ for (int ca = 1; ca < loopCountAttributes; ca++) {
+ tTmp =
+ pOps[countTables]->getValue((char*)attrName[ca],
+ (char*)&tmpAttrValue[tAttributeSize*ca]);
+ }//for
+ } else if (stVerifyDelete == tType) {
+ if(useLongKeys){
+ tTmp = pOps[countTables]->getValue(longKeyAttrName[0],
+ (char*)&tmpAttrValue[0]);
+ } else {
+ tTmp = pOps[countTables]->getValue((char*)attrName[0],
+ (char*)&tmpAttrValue[0]);
+ }
+ }//if
+ } else { // !use_ndb
+#ifndef USE_MYSQL
+ assert(false);
+#else
+ switch (tType)
+ {
+ case stInsert:
+ for (int ca = 0; ca < loopCountAttributes; ca++){
+ mysql_data[ca] = tmpAttrRefValue[tAttributeSize*ca];
+ }//for
+ if (mysql_execute(prep_insert[countTables])) {
+ ndbout << tableName[countTables];
+ ndbout << " mysql_execute: " << mysql_error(&mysql) << endl;
+ tResult = 1 ;
+ }
+ break;
+ case stUpdate: // Update Case
+ mysql_data[0] = tmpAttrRefValue[0];
+ for (int ca = 1; ca < loopCountAttributes; ca++){
+ int* tmp = (int*)&tmpAttrRefValue[tAttributeSize*ca];
+ if (countTables == 0)
+ (*tmp)++;
+ mysql_data[ca] = *tmp;
+ }//for
+ if (mysql_execute(prep_update[countTables])) {
+ ndbout << tableName[countTables];
+ ndbout << " mysql_execute: " << mysql_error(&mysql) << endl;
+ tResult = 2 ;
+ }
+ break;
+ case stVerify:
+ case stRead: // Read Case
+ mysql_data[0] = tmpAttrRefValue[0];
+ if (mysql_execute(prep_read[countTables])) {
+ ndbout << tableName[countTables];
+ ndbout << " mysql_execute: " << mysql_error(&mysql) << endl;
+ tResult = 3 ;
+ break;
+ }
+ if (mysql_stmt_store_result(prep_read[countTables])) {
+ ndbout << tableName[countTables];
+ ndbout << " mysql_stmt_store_result: "
+ << mysql_error(&mysql) << endl;
+ tResult = 4 ;
+ break;
+ }
+ {
+ int rows= 0;
+ int r;
+ while ( (r= mysql_fetch(prep_read[countTables])) == 0 ){
+ rows++;
+ }
+ if ( r == 1 ) {
+ ndbout << tableName[countTables];
+ ndbout << " mysql_fetch: " << mysql_error(&mysql) << endl;
+ tResult = 5 ;
+ break;
+ }
+ if ( rows != 1 ) {
+ ndbout << tableName[countTables];
+ ndbout << " mysql_fetch: rows = " << rows << endl;
+ tResult = 6 ;
+ break;
+ }
+ }
+ {
+ for (int ca = 1; ca < loopCountAttributes; ca++) {
+ tmpAttrValue[tAttributeSize*ca] = mysql_data[ca];
+ }
+ }
+ break;
+ case stDelete: // Delete Case
+ mysql_data[0] = tmpAttrRefValue[0];
+ if (mysql_execute(prep_delete[countTables])) {
+ ndbout << tableName[countTables];
+ ndbout << " mysql_execute: " << mysql_error(&mysql) << endl;
+ tResult = 7 ;
+ break;
+ }
+ break;
+ case stVerifyDelete:
+ {
+ sprintf(buf, "%s%s%s",
+ "SELECT COUNT(*) FROM ",tableName[countTables],";");
+ if (mysql_query(&mysql, buf)) {
+ ndbout << buf << endl;
+ ndbout << "Error: " << mysql_error(&mysql) << endl;
+ tResult = 8 ;
+ break;
+ }
+ MYSQL_RES *res = mysql_store_result(&mysql);
+ if ( res == NULL ) {
+ ndbout << "mysql_store_result: "
+ << mysql_error(&mysql) << endl
+ << "errno: " << mysql_errno(&mysql) << endl;
+ tResult = 9 ;
+ break;
+ }
+ int num_fields = mysql_num_fields(res);
+ int num_rows = mysql_num_rows(res);
+ if ( num_rows != 1 || num_fields != 1 ) {
+ ndbout << tableName[countTables];
+ ndbout << " mysql_store_result: num_rows = " << num_rows
+ << " num_fields = " << num_fields << endl;
+ tResult = 10 ;
+ break;
+ }
+ MYSQL_ROW row = mysql_fetch_row(res);
+ if ( row == NULL ) {
+ ndbout << "mysql_fetch_row: "
+ << mysql_error(&mysql) << endl;
+ tResult = 11 ;
+ break;
+ }
+ if ( *(char*)row[0] != '0' ) {
+ ndbout << tableName[countTables];
+ ndbout << " mysql_fetch_row: value = "
+ << (char*)(row[0]) << endl;
+ tResult = 12 ;
+ break;
+ }
+ mysql_free_result(res);
+ }
+ break;
+ default:
+ assert(false);
+ }
+#endif
+ }
+ }//for Tables loop
+
+ if (tResult != 0)
+ break;
+
+ if (use_ndb){
+ check = pTrans->execute(Commit);
+ } else {
+#ifdef USE_MYSQL
+ if (tNoOfTables > 1)
+ if (mysql_commit(&mysql)) {
+ ndbout << " mysql_commit: " << mysql_error(&mysql) << endl;
+ tResult = 13;
+ } else
+ check = 0;
+#endif
+ }
+
+ if (use_ndb) {
+ // Decide what kind of error this is
+ if ((tSpecialTrans == 1) &&
+ (check == -1)) {
+// --------------------------------------------------------------------
+// A special transaction have been executed, change to check = 0 in
+// certain situations.
+// --------------------------------------------------------------------
+ switch (tType) {
+ case stInsert: // Insert case
+ if (630 == pTrans->getNdbError().code ) {
+ check = 0;
+ ndbout << "Insert with 4007 was successful" << endl;
+ }//if
+ break;
+ case stDelete: // Delete Case
+ if (626 == pTrans->getNdbError().code ) {
+ check = 0;
+ ndbout << "Delete with 4007 was successful" << endl;
+ }//if
+ break;
+ default:
+ assert(false);
+ }//switch
+ }//if
+ tSpecialTrans = 0;
+ if (check == -1) {
+ if ((stVerifyDelete == tType) &&
+ (626 == pTrans->getNdbError().code)) {
+ // ----------------------------------------------
+ // It's good news - the deleted tuple is gone,
+ // so reset "check" flag
+ // ----------------------------------------------
+ check = 0 ;
+ } else {
+ int retCode =
+ theErrorData.handleErrorCommon(pTrans->getNdbError());
+ if (retCode == 1) {
+ ndbout_c("execute: %d, %d, %s", count, tType,
+ pTrans->getNdbError().message );
+ ndbout_c("Error code = %d", pTrans->getNdbError().code );
+ tResult = 20;
+ } else if (retCode == 2) {
+ ndbout << "4115 should not happen in flexBench" << endl;
+ tResult = 20;
+ } else if (retCode == 3) {
+// --------------------------------------------------------------------
+// We are not certain if the transaction was successful or not.
+// We must reexecute but might very well find that the transaction
+// actually was updated. Updates and Reads are no problem here. Inserts
+// will not cause a problem if error code 630 arrives. Deletes will
+// not cause a problem if 626 arrives.
+// --------------------------------------------------------------------
+ if ((tType == stInsert) || (tType == stDelete)) {
+ tSpecialTrans = 1;
+ }//if
+ }//if
+ }//if
+ }//if
+ // Check if retries should be made
+ if (check == -1 && tResult == 0) {
+ if (tAttemptNo < tRetryAttempts){
+ tAttemptNo++;
+ } else {
+// --------------------------------------------------------------------
+// Too many retries have been made, report error and break out of loop
+// --------------------------------------------------------------------
+ ndbout << "Thread" << threadNo;
+ ndbout << ": too many errors reported" << endl;
+ tResult = 10;
+ break;
+ }//if
+ }//if
+ }
+
+ if (check == 0){
+ // Go to the next record
+ count++;
+ tAttemptNo = 0;
+#ifdef CEBIT_STAT
+ // report successful ops
+ if (statEnable) {
+ statOps += loopCountTables;
+ if (statOps >= statFreq) {
+ statReport(tType, statOps);
+ statOps = 0;
+ }//if
+ }//if
+#endif
+ }//if
+
+ if (stVerify == tType && 0 == check){
+ int nTableOffset = 0 ;
+ for (int a = 1 ; a < loopCountAttributes ; a++){
+ for (int tables = 0 ; tables < loopCountTables ; tables++){
+ nTableOffset = tables*loopCountAttributes*tAttributeSize;
+ int ov =*(int*)&attrValue[nTableOffset + tAttributeSize*a];
+ int nv =*(int*)&tmpAttrRefValue[tAttributeSize*a];
+ if (ov != nv){
+ ndbout << "Error in verify ";
+ ndbout << "pk = " << tmpAttrRefValue[0] << ":" << endl;
+ ndbout << "attrValue[" << nTableOffset + tAttributeSize*a << "] = " << ov << endl ;
+ ndbout << "attrRefValue[" << nRefLocalOpOffset + tAttributeSize*a << "]" << nv << endl ;
+ tResult = 11 ;
+ break ;
+ }//if
+ }//for
+ }//for
+ }// if(stVerify ... )
+ if (use_ndb) {
+ pNdb->closeTransaction(pTrans);
+ return_ndb_object(pNdb, ndb_id);
+ pNdb = NULL;
+ }
+ }// operations loop
+#ifdef CEBIT_STAT
+ // report remaining successful ops
+ if (statEnable) {
+ if (statOps > 0) {
+ statReport(tType, statOps);
+ statOps = 0;
+ }//if
+ }//if
+#endif
+ if (pNdb) {
+ pNdb->closeTransaction(pTrans);
+ return_ndb_object(pNdb, ndb_id);
+ pNdb = NULL;
+ }
+ }
+
+#ifdef USE_MYSQL
+ if (!use_ndb) {
+ mysql_close(&mysql);
+ for (Uint32 i = 0; i < tNoOfTables; i++) {
+ mysql_stmt_close(prep_insert[i]);
+ mysql_stmt_close(prep_update[i]);
+ mysql_stmt_close(prep_delete[i]);
+ mysql_stmt_close(prep_read[i]);
+ }
+ }
+#endif
+ if (use_ndb && pNdb) {
+ ndbout << "I got here " << endl;
+ return_ndb_object(pNdb, ndb_id);
+ }
+ NdbThread_Exit(0);
+ return NULL; // Just to keep compiler happy
+}
+
+
+static int readArguments(int argc, const char** argv)
+{
+
+ int i = 1;
+ while (argc > 1){
+ if (strcmp(argv[i], "-t") == 0){
+ tNoOfThreads = atoi(argv[i+1]);
+ if ((tNoOfThreads < 1))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-o") == 0){
+ tNoOfOperations = atoi(argv[i+1]);
+ if (tNoOfOperations < 1)
+ return -1;;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-a") == 0){
+ tNoOfAttributes = atoi(argv[i+1]);
+ if ((tNoOfAttributes < 2) || (tNoOfAttributes > MAXATTR))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-c") == 0){
+ tNoOfTables = atoi(argv[i+1]);
+ if ((tNoOfTables < 1) || (tNoOfTables > MAXTABLES))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-stdtables") == 0){
+ theStdTableNameFlag = 1;
+ }else if (strcmp(argv[i], "-l") == 0){
+ tNoOfLoops = atoi(argv[i+1]);
+ if ((tNoOfLoops < 0) || (tNoOfLoops > 100000))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-pool_size") == 0){
+ t_instances = atoi(argv[i+1]);
+ if ((t_instances < 1) || (t_instances > 240))
+ return -1;
+ argc -= 1;
+ i++;
+#ifdef USE_MYSQL
+ }else if (strcmp(argv[i], "-engine") == 0){
+ engine_id = atoi(argv[i+1]);
+ if ((engine_id < 0) || (engine_id > 3))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-socket") == 0){
+ sockets[n_sockets] = atoi(argv[i+1]);
+ if (sockets[n_sockets] <= 0)
+ return -1;
+ n_sockets++;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-use_ndb") == 0){
+ use_ndb = true;
+#endif
+ }else if (strcmp(argv[i], "-s") == 0){
+ tAttributeSize = atoi(argv[i+1]);
+ if ((tAttributeSize < 1) || (tAttributeSize > MAXATTRSIZE))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-lkn") == 0){
+ tNoOfLongPK = atoi(argv[i+1]);
+ useLongKeys = true;
+ if ((tNoOfLongPK < 1) || (tNoOfLongPK > MAXNOLONGKEY) ||
+ (tNoOfLongPK * tSizeOfLongPK) > MAXLONGKEYTOTALSIZE){
+ ndbout << "Argument -lkn is not in the proper range." << endl;
+ return -1;
+ }
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-lks") == 0){
+ tSizeOfLongPK = atoi(argv[i+1]);
+ useLongKeys = true;
+ if ((tSizeOfLongPK < 1) || (tNoOfLongPK * tSizeOfLongPK) > MAXLONGKEYTOTALSIZE){
+ ndbout << "Argument -lks is not in the proper range 1 to " <<
+ MAXLONGKEYTOTALSIZE << endl;
+ return -1;
+ }
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-simple") == 0){
+ theSimpleFlag = 1;
+ }else if (strcmp(argv[i], "-write") == 0){
+ theWriteFlag = 1;
+ }else if (strcmp(argv[i], "-dirty") == 0){
+ theDirtyFlag = 1;
+ }else if (strcmp(argv[i], "-sleep") == 0){
+ tSleepTime = atoi(argv[i+1]);
+ if ((tSleepTime < 1) || (tSleepTime > 3600))
+ return -1;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-no_table_create") == 0){
+ theTableCreateFlag = 1;
+ }else if (strcmp(argv[i], "-temp") == 0){
+ theTempTable = true;
+ }else if (strcmp(argv[i], "-noverify") == 0){
+ VerifyFlag = false ;
+ }else if (theErrorData.parseCmdLineArg(argv, i) == true){
+ ; //empty, updated in errorArg(..)
+ }else if (strcmp(argv[i], "-verify") == 0){
+ VerifyFlag = true ;
+#ifdef CEBIT_STAT
+ }else if (strcmp(argv[i], "-statserv") == 0){
+ if (! (argc > 2))
+ return -1;
+ const char *p = argv[i+1];
+ const char *q = strrchr(p, ':');
+ if (q == 0)
+ return -1;
+ snprintf(statHost, sizeof(statHost), "%.*s", q-p, p);
+ statPort = atoi(q+1);
+ statEnable = true;
+ argc -= 1;
+ i++;
+ }else if (strcmp(argv[i], "-statfreq") == 0){
+ if (! (argc > 2))
+ return -1;
+ statFreq = atoi(argv[i+1]);
+ if (statFreq < 1)
+ return -1;
+ argc -= 1;
+ i++;
+#endif
+ }else{
+ return -1;
+ }
+ argc -= 1;
+ i++;
+ }
+#ifdef USE_MYSQL
+ if (n_sockets == 0) {
+ n_sockets = 1;
+ sockets[0] = 3306;
+ }
+#endif
+ return 0;
+}
+
+static void sleepBeforeStartingTest(int seconds){
+ if (seconds > 0){
+ ndbout << "Sleeping(" <<seconds << ")...";
+ NdbSleep_SecSleep(seconds);
+ ndbout << " done!" << endl;
+ }
+}
+
+
+#ifdef USE_MYSQL
+static int
+dropTables(MYSQL* mysqlp){
+ char buf[2048];
+ for(unsigned i = 0; i < tNoOfTables; i++){
+ int pos = 0;
+ ndbout << "Dropping " << tableName[i] << "... ";
+ pos += sprintf(buf+pos, "%s", "DROP TABLE ");
+ pos += sprintf(buf+pos, "%s%s", tableName[i], ";");
+ if (verbose)
+ ndbout << endl << buf << endl;
+ if (mysql_query(mysqlp, buf) != 0){
+ ndbout << "Failed!"<<endl
+ <<mysql_error(mysqlp)<<endl
+ <<buf<<endl;
+ } else
+ ndbout << "OK!" << endl;
+ }
+
+ return 0;
+}
+#endif
+
+#ifdef USE_MYSQL
+static int
+createTables(MYSQL* mysqlp){
+
+ for (Uint32 i = 0; i < tNoOfAttributes; i++){
+ snprintf(attrName[i], MAXSTRLEN, "COL%d", i);
+ }
+
+ // Note! Uses only uppercase letters in table name's
+ // so that we can look at the tables with SQL
+ for (Uint32 i = 0; i < tNoOfTables; i++){
+ if (theStdTableNameFlag == 0){
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d_%d", i,
+ (int)(NdbTick_CurrentMillisecond() / 1000));
+ } else {
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d", i);
+ }
+ }
+
+ char buf[2048];
+ for(unsigned i = 0; i < tNoOfTables; i++){
+ int pos = 0;
+ ndbout << "Creating " << tableName[i] << "... ";
+
+ pos += sprintf(buf+pos, "%s", "CREATE TABLE ");
+ pos += sprintf(buf+pos, "%s%s", tableName[i], " ");
+ if(useLongKeys){
+ for(Uint32 i = 0; i < tNoOfLongPK; i++) {
+ }
+ } else {
+ pos += sprintf(buf+pos, "%s%s%s",
+ "(", attrName[0], " int unsigned primary key");
+ }
+ for (unsigned j = 1; j < tNoOfAttributes; j++)
+ pos += sprintf(buf+pos, "%s%s%s", ",", attrName[j], " int unsigned");
+ pos += sprintf(buf+pos, "%s%s%s", ")", engine[engine_id], ";");
+ if (verbose)
+ ndbout << endl << buf << endl;
+ if (mysql_query(mysqlp, buf) != 0)
+ return -1;
+ ndbout << "done" << endl;
+ }
+ return 0;
+}
+#endif
+
+static int
+createTables(Ndb* pMyNdb){
+
+ for (Uint32 i = 0; i < tNoOfAttributes; i++){
+ snprintf(attrName[i], MAXSTRLEN, "COL%d", i);
+ }
+
+ // Note! Uses only uppercase letters in table name's
+ // so that we can look at the tables with SQL
+ for (Uint32 i = 0; i < tNoOfTables; i++){
+ if (theStdTableNameFlag == 0){
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d_%d", i,
+ (int)(NdbTick_CurrentMillisecond() / 1000));
+ } else {
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d", i);
+ }
+ }
+
+ for(unsigned i = 0; i < tNoOfTables; i++){
+ ndbout << "Creating " << tableName[i] << "... ";
+
+ NdbDictionary::Table tmpTable(tableName[i]);
+
+ tmpTable.setStoredTable(!theTempTable);
+
+ if(useLongKeys){
+ for(Uint32 i = 0; i < tNoOfLongPK; i++) {
+ NdbDictionary::Column col(longKeyAttrName[i]);
+ col.setType(NdbDictionary::Column::Unsigned);
+ col.setLength(tSizeOfLongPK);
+ col.setPrimaryKey(true);
+ tmpTable.addColumn(col);
+ }
+ } else {
+ NdbDictionary::Column col(attrName[0]);
+ col.setType(NdbDictionary::Column::Unsigned);
+ col.setLength(1);
+ col.setPrimaryKey(true);
+ tmpTable.addColumn(col);
+ }
+ NdbDictionary::Column col;
+ col.setType(NdbDictionary::Column::Unsigned);
+ col.setLength(tAttributeSize);
+ for (unsigned j = 1; j < tNoOfAttributes; j++){
+ col.setName(attrName[j]);
+ tmpTable.addColumn(col);
+ }
+ if(pMyNdb->getDictionary()->createTable(tmpTable) == -1){
+ return -1;
+ }
+ ndbout << "done" << endl;
+ }
+
+ return 0;
+}
+
+
+static void input_error(){
+ ndbout << endl << "Invalid argument!" << endl;
+ ndbout << endl << "Arguments:" << endl;
+ ndbout << " -t Number of threads to start, default 1" << endl;
+ ndbout << " -o Number of operations per loop, default 500" << endl;
+ ndbout << " -l Number of loops to run, default 1, 0=infinite" << endl;
+ ndbout << " -a Number of attributes, default 25" << endl;
+ ndbout << " -c Number of tables, default 1" << endl;
+ ndbout << " -s Size of each attribute, default 1 (Primary Key is always of size 1," << endl;
+ ndbout << " independent of this value)" << endl;
+ ndbout << " -lkn Number of long primary keys, default 1" << endl;
+ ndbout << " -lks Size of each long primary key, default 1" << endl;
+
+ ndbout << " -simple Use simple read to read from database" << endl;
+ ndbout << " -dirty Use dirty read to read from database" << endl;
+ ndbout << " -write Use writeTuple in insert and update" << endl;
+ ndbout << " -stdtables Use standard table names" << endl;
+ ndbout << " -no_table_create Don't create tables in db" << endl;
+ ndbout << " -sleep Sleep a number of seconds before running the test, this" << endl;
+ ndbout << " can be used so that another flexBench have time to create tables" << endl;
+ ndbout << " -temp Use tables without logging" << endl;
+ ndbout << " -verify Verify inserts, updates and deletes" << endl ;
+ ndbout << " -use_ndb Use NDB API (otherwise use mysql client)" << endl ;
+ ndbout << " -pool_size Number of Ndb objects in pool" << endl ;
+ theErrorData.printCmdLineArgs(ndbout);
+ ndbout << endl <<"Returns:" << endl;
+ ndbout << "\t 0 - Test passed" << endl;
+ ndbout << "\t 1 - Test failed" << endl;
+ ndbout << "\t 2 - Invalid arguments" << endl << endl;
+}
+
+// vim: set sw=2:
diff --git a/ndb/test/ndbapi/indexTest/Makefile b/ndb/test/ndbapi/indexTest/Makefile
new file mode 100644
index 00000000000..d842e487ee5
--- /dev/null
+++ b/ndb/test/ndbapi/indexTest/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := index
+
+SOURCES := index.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/indexTest/index.cpp b/ndb/test/ndbapi/indexTest/index.cpp
new file mode 100644
index 00000000000..d0eb490e1a0
--- /dev/null
+++ b/ndb/test/ndbapi/indexTest/index.cpp
@@ -0,0 +1,998 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* ***************************************************
+ INDEX TEST 1
+ Test index functionality of NDB
+
+ Arguments:
+ -T create table
+ -L include a long attribute in key or index
+ -2 define primary key with two attributes
+ -c create index
+ -p make index unique (include primary key attribute)
+ -r read using index
+ -u update using index
+ -d delete using index
+ -n<no operations> do n operations (for -I -r -u -d -R -U -D)
+ -o<no parallel operations> (for -I -r -u -d -R -U -D)
+ -m<no indexes>
+
+ Returns:
+ 0 - Test passed
+ -1 - Test failed
+ 1 - Invalid arguments
+ * *************************************************** */
+
+#include <string.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <NdbApi.hpp>
+#include <NdbOut.hpp>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbTest.hpp>
+#include <NDBT_Error.hpp>
+
+#ifndef MIN
+#define MIN(x,y) (((x)<(y))?(x):(y))
+#endif
+
+#define MAX_NO_PARALLEL_OPERATIONS 100
+
+bool testPassed = true;
+
+static void
+error_handler(const NdbError & err)
+{
+ // Test failed
+ ndbout << endl << err << endl;
+ testPassed = false;
+}
+
+static void
+error_handler4(int line, const NdbError & err){
+ ndbout << endl << "Line " << line << endl;
+ // Test failed
+ ndbout << err << endl;
+ testPassed = false;
+}
+
+static char *longName, *sixtysix, *ninetynine, *hundred;
+
+static void createTable(Ndb &myNdb, bool storeInACC, bool twoKey, bool longKey)
+{
+ NdbDictionary::Dictionary* dict = myNdb.getDictionary();
+ NdbDictionary::Table table("PERSON");
+ //NdbDictionary::Column column(); // Bug
+ NdbDictionary::Column column;
+ int res;
+
+ column.setName("NAME");
+ column.setPrimaryKey(true);
+ column.setType(NdbDictionary::Column::Char);
+ column.setLength((longKey)?
+ 1024 // 1KB => long key
+ :12);
+ column.setNullable(false);
+ table.addColumn(column);
+
+ if (twoKey) {
+ column.setName("KEY2");
+ column.setPrimaryKey(true);
+ column.setType(NdbDictionary::Column::Unsigned);
+ column.setLength(1);
+ column.setNullable(false);
+ table.addColumn(column);
+ }
+
+ column.setName("PNUM1");
+ column.setPrimaryKey(false);
+ column.setType(NdbDictionary::Column::Unsigned);
+ column.setLength(1);
+ column.setNullable(false);
+ table.addColumn(column);
+
+ column.setName("PNUM2");
+ column.setPrimaryKey(false);
+ column.setType(NdbDictionary::Column::Unsigned);
+ column.setLength(1);
+ column.setNullable(false);
+ table.addColumn(column);
+
+ column.setName("PNUM3");
+ column.setPrimaryKey(false);
+ column.setType(NdbDictionary::Column::Unsigned);
+ column.setLength(1);
+ column.setNullable(false);
+ table.addColumn(column);
+
+ column.setName("PNUM4");
+ column.setPrimaryKey(false);
+ column.setType(NdbDictionary::Column::Unsigned);
+ column.setLength(1);
+ column.setNullable(false);
+ table.addColumn(column);
+
+ column.setName("AGE");
+ column.setPrimaryKey(false);
+ column.setType(NdbDictionary::Column::Unsigned);
+ column.setLength(1);
+ column.setNullable(false);
+ table.addColumn(column);
+
+ column.setName("STRING_AGE");
+ column.setPrimaryKey(false);
+ column.setType(NdbDictionary::Column::Char);
+ column.setLength(1);
+ column.setLength(256);
+ column.setNullable(false);
+ table.addColumn(column);
+
+ if ((res = dict->createTable(table)) == -1) {
+ error_handler(dict->getNdbError());
+ }
+ else
+ ndbout << "Created table" << ((longKey)?" with long key":"") <<endl;
+}
+
+static void createIndex(Ndb &myNdb, bool includePrimary, unsigned int noOfIndexes)
+{
+ Uint64 before, after;
+ NdbDictionary::Dictionary* dict = myNdb.getDictionary();
+ char indexName[] = "PNUMINDEX0000";
+ int res;
+
+ for(unsigned int indexNum = 0; indexNum < noOfIndexes; indexNum++) {
+ sprintf(indexName, "PNUMINDEX%.4u", indexNum);
+ NdbDictionary::Index index(indexName);
+ index.setTable("PERSON");
+ index.setType(NdbDictionary::Index::UniqueHashIndex);
+ if (includePrimary) {
+ const char* attr_arr[] = {"NAME", "PNUM1", "PNUM3"};
+ index.addIndexColumns(3, attr_arr);
+ }
+ else {
+ const char* attr_arr[] = {"PNUM1", "PNUM3"};
+ index.addIndexColumns(2, attr_arr);
+ }
+ before = NdbTick_CurrentMillisecond();
+ if ((res = dict->createIndex(index)) == -1) {
+ error_handler(dict->getNdbError());
+ }
+ after = NdbTick_CurrentMillisecond();
+ ndbout << "Created index " << indexName << ", " << after - before << " msec" << endl;
+ }
+}
+
+static void insertTable(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool oneTrans, bool twoKey, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbOperation *myOp;
+ char name[] = "Kalle0000000";
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError());
+
+ myOp = myTrans->getNdbOperation("PERSON");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError());
+
+ myOp->insertTuple();
+ sprintf(name, "Kalle%.7i", i);
+ if (longKey)
+ memcpy(longName, name, strlen(name));
+ if (myOp->equal("NAME", (longKey)?longName:name) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (twoKey)
+ if (myOp->equal("KEY2", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("PNUM1", 17) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("PNUM2", 18)) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("PNUM3", 19)) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("PNUM4", 20)) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("AGE", ((i % 2) == 0)?66:99) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("STRING_AGE", ((i % 2) == 0)?sixtysix:ninetynine) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (noOfOperations == 1)
+ printf("Trying to insert person %s\n", name);
+ else
+ printf("Trying to insert %u persons\n", noOfOperations);
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Inserted person %s, %u msec\n", name, (Uint32) after - before);
+ else
+ printf("Inserted %u persons, %u msec\n", noOfOperations, (Uint32) after - before);
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Inserted "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void updateTable(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool oneTrans, bool twoKey, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbOperation *myOp;
+ char name[] = "Kalle0000000";
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError());
+
+ myOp = myTrans->getNdbOperation("PERSON");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError());
+
+ myOp->updateTuple();
+ sprintf(name, "Kalle%.7i", i);
+ if (longKey)
+ memcpy(longName, name, strlen(name));
+ if (myOp->equal("NAME", (longKey)?longName:name) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (twoKey)
+ if (myOp->equal("KEY2", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("PNUM1", 77) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("PNUM2", 88)) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("PNUM4", 99)) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("AGE", 100) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("STRING_AGE", hundred) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (noOfOperations == 1)
+ printf("Trying to update person %s\n", name);
+ else
+ printf("Trying to update %u persons\n", noOfOperations);
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Updated person %s, %u msec\n", name, (Uint32) after - before);
+ else
+ printf("Update %u persons, %u msec\n", noOfOperations, (Uint32) after - before);
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Updated "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void deleteTable(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool oneTrans, bool twoKey, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbOperation *myOp;
+ char name[] = "Kalle0000000";
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError());
+
+ myOp = myTrans->getNdbOperation("PERSON");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError());
+
+ myOp->deleteTuple();
+ sprintf(name, "Kalle%.7i", i);
+ if (longKey)
+ memcpy(longName, name, strlen(name));
+ if (myOp->equal("NAME", (longKey)?longName:name) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (twoKey)
+ if (myOp->equal("KEY2", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (noOfOperations == 1)
+ printf("Trying to delete person %s\n", name);
+ else
+ printf("Trying to delete %u persons\n", noOfOperations);
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Deleted person %s, %u msec\n", name, (Uint32) after - before);
+ else
+ printf("Deleted %u persons, %u msec\n", noOfOperations, (Uint32) after - before);
+
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Deleted "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void readTable(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool oneTrans, bool twoKey, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbOperation *myOp;
+ char name[] = "Kalle0000000";
+ NdbRecAttr* myRecAttrArr[MAX_NO_PARALLEL_OPERATIONS];
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError());
+
+ myOp = myTrans->getNdbOperation("PERSON");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError());
+
+ myOp->readTuple();
+ sprintf(name, "Kalle%.7i", i);
+ if (longKey)
+ memcpy(longName, name, strlen(name));
+ if (myOp->equal("NAME", (longKey)?longName:name) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (twoKey)
+ if (myOp->equal("KEY2", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ myRecAttrArr[j-1] = myOp->getValue("PNUM2", NULL);
+ }
+ if (noOfOperations == 1)
+ printf("Trying to read person %s\n", name);
+ else
+ printf("Trying to read %u persons\n", noOfOperations);
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Read person %s, %u msec\n", name, (Uint32) after - before);
+ else
+ printf("Read %u persons, %u msec\n", noOfOperations, (Uint32) after - before);
+ for(unsigned int j = 0; j<noOfOperations; j++)
+ printf("PNUM2 = %u\n", myRecAttrArr[j]->u_32_value());
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Read "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void readIndex(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool includePrimary, bool oneTrans, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbIndexOperation *myOp;
+ char indexName[] = "PNUMINDEX0000";
+ char name[] = "Kalle0000000";
+ NdbRecAttr* myRecAttrArr[MAX_NO_PARALLEL_OPERATIONS];
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError());
+
+ myOp = myTrans->getNdbIndexOperation(indexName, "PERSON");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError());
+
+ myOp->readTuple();
+ if (includePrimary) {
+ sprintf(name, "Kalle%.7i", i);
+ if (longKey)
+ memcpy(longName, name, strlen(name));
+ if (myOp->equal("NAME", (longKey)?longName:name) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (myOp->equal("PNUM1", 17) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->equal("PNUM3", 19) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ myRecAttrArr[j-1] = myOp->getValue("PNUM2", NULL);
+ }
+ if (noOfOperations == 1)
+ printf("Trying to read person %s\n", name);
+ else
+ printf("Trying to read %u persons\n", noOfOperations);
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Read person %s, %u msec\n", name, (Uint32) after - before);
+ else
+ printf("Read %u persons, %u msec\n", noOfOperations, (Uint32) after - before);
+ for(unsigned int j = 0; j<noOfOperations; j++)
+ printf("PNUM2 = %u\n", myRecAttrArr[j]->u_32_value());
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Read "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void updateIndex(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool includePrimary, bool oneTrans, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbIndexOperation *myOp;
+ char indexName[] = "PNUMINDEX0000";
+ char name[] = "Kalle0000000";
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError());
+
+ myOp = myTrans->getNdbIndexOperation(indexName, "PERSON");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError());
+
+ myOp->updateTuple();
+ if (includePrimary) {
+ sprintf(name, "Kalle%.7i", i);
+ if (longKey)
+ memcpy(longName, name, strlen(name));
+ if (myOp->equal("NAME", (longKey)?longName:name) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (myOp->equal("PNUM1", 17) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->equal("PNUM3", 19) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ // Update index itself, should be possible
+ if (myOp->setValue("PNUM1", 77) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("PNUM2", 88)) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("PNUM4", 99)) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("AGE", 100) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("STRING_AGE", hundred) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (noOfOperations == 1)
+ printf("Trying to update person %s\n", name);
+ else
+ printf("Trying to update %u persons\n", noOfOperations);
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Updated person %s, %u msec\n", name, (Uint32) after - before);
+ else
+ printf("Updated %u persons, %u msec\n", noOfOperations, (Uint32) after - before);
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Updated "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void deleteIndex(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool includePrimary, bool oneTrans, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbIndexOperation *myOp;
+ char indexName[] = "PNUMINDEX0000";
+ char name[] = "Kalle0000000";
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError());
+
+ myOp = myTrans->getNdbIndexOperation(indexName, "PERSON");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError());
+
+ myOp->deleteTuple();
+ if (includePrimary) {
+ sprintf(name, "Kalle%.7i", i);
+ if (longKey)
+ memcpy(longName, name, strlen(name));
+ if (myOp->equal("NAME", (longKey)?longName:name) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (myOp->equal("PNUM1", 17) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->equal("PNUM3", 19) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (noOfOperations == 1)
+ printf("Trying to delete person %s\n", name);
+ else
+ printf("Trying to delete %u persons\n", noOfOperations);
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Deleted person %s, %u msec\n", name, (Uint32) after - before);
+ else
+ printf("Deleted %u persons, %u msec\n", noOfOperations, (Uint32) after - before);
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError());
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Deleted "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void dropIndex(Ndb &myNdb, unsigned int noOfIndexes)
+{
+ for(unsigned int indexNum = 0; indexNum < noOfIndexes; indexNum++) {
+ char indexName[255];
+ sprintf(indexName, "PNUMINDEX%.4u", indexNum);
+ const Uint64 before = NdbTick_CurrentMillisecond();
+ const int retVal = myNdb.getDictionary()->dropIndex(indexName, "PERSON");
+ const Uint64 after = NdbTick_CurrentMillisecond();
+
+ if(retVal == 0){
+ ndbout << "Dropped index " << indexName << ", "
+ << after - before << " msec" << endl;
+ } else {
+ ndbout << "Failed to drop index " << indexName << endl;
+ ndbout << myNdb.getDictionary()->getNdbError() << endl;
+ }
+ }
+}
+
+NDB_COMMAND(indexTest, "indexTest", "indexTest", "indexTest", 65535)
+{
+ bool createTableOp, createIndexOp, dropIndexOp, insertOp, updateOp, deleteOp, readOp, readIndexOp, updateIndexOp, deleteIndexOp, twoKey, longKey;
+ unsigned int noOfTuples = 1;
+ unsigned int noOfOperations = 1;
+ unsigned int noOfIndexes = 1;
+ int i = 1;
+ Ndb myNdb( "TEST_DB" );
+ int check;
+ bool storeInACC = false;
+ bool includePrimary = false;
+ bool oneTransaction = false;
+
+ createTableOp = createIndexOp = dropIndexOp = insertOp = updateOp = deleteOp = readOp = readIndexOp = updateIndexOp = deleteIndexOp = twoKey = longKey = false;
+ // Read arguments
+ if (argc > 1)
+ while (argc > 1)
+ {
+ if (strcmp(argv[i], "-T") == 0)
+ {
+ createTableOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-c") == 0)
+ {
+ createIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-X") == 0)
+ {
+ dropIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-I") == 0)
+ {
+ insertOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-D") == 0)
+ {
+ deleteOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-U") == 0)
+ {
+ updateOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-R") == 0)
+ {
+ readOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-r") == 0)
+ {
+ readIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-u") == 0)
+ {
+ updateIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-d") == 0)
+ {
+ deleteIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-s") == 0)
+ {
+ storeInACC = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-p") == 0)
+ {
+ includePrimary = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-L") == 0)
+ {
+ longKey = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-1") == 0)
+ {
+ oneTransaction = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-2") == 0)
+ {
+ twoKey = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strstr(argv[i], "-n") != 0)
+ {
+ noOfTuples = atoi(argv[i]+2);
+ argc -= 1;
+ i++;
+ }
+ else if (strstr(argv[i], "-o") != 0)
+ {
+ noOfOperations = MIN(MAX_NO_PARALLEL_OPERATIONS, atoi(argv[i]+2));
+ argc -= 1;
+ i++;
+ }
+ else if (strstr(argv[i], "-m") != 0)
+ {
+ noOfIndexes = atoi(argv[i]+2);
+ argc -= 1;
+ i++;
+ }
+ else if (strstr(argv[i], "-h") != 0)
+ {
+ printf("Synopsis:\n");
+ printf("index\n");
+ printf("\t-T create table\n");
+ printf("\t-L include a long attribute in key or index\n");
+ printf("\t-2 define primary key with two attributes\n");
+ printf("\t-c create index\n");
+ printf("\t-p make index unique (include primary key attribute)\n");
+ printf("\t-r read using index\n");
+ printf("\t-u update using index\n");
+ printf("\t-d delete using index\n");
+ printf("\t-n<no operations> do n operations (for -I -r -u -d -R -U -D)\n");
+ printf("\t-o<no parallel operations> (for -I -r -u -d -R -U -D)\n");
+ printf("\t-m<no indexes>\n");
+ argc -= 1;
+ i++;
+ }
+ else {
+ char errStr[256];
+
+ printf(errStr, "Illegal argument: %s", argv[i]);
+ exit(-1);
+ }
+ }
+ else
+ {
+ createTableOp = createIndexOp = dropIndexOp = insertOp = updateOp = deleteOp = true;
+ }
+ if (longKey) {
+ longName = (char *) malloc(1024);
+ for (int i = 0; i < 1023; i++)
+ longName[i] = 'x';
+ longName[1023] = '\0';
+ }
+ sixtysix = (char *) malloc(256);
+ for (int i = 0; i < 255; i++)
+ sixtysix[i] = ' ';
+ sixtysix[255] = '\0';
+ strncpy(sixtysix, "sixtysix", strlen("sixtysix"));
+ ninetynine = (char *) malloc(256);
+ for (int i = 0; i < 255; i++)
+ ninetynine[i] = ' ';
+ ninetynine[255] = '\0';
+ strncpy(ninetynine, "ninetynine", strlen("ninetynine"));
+ hundred = (char *) malloc(256);
+ for (int i = 0; i < 255; i++)
+ hundred[i] = ' ';
+ hundred[255] = '\0';
+ strncpy(hundred, "hundred", strlen("hundred"));
+ myNdb.init();
+ // Wait for Ndb to become ready
+ if (myNdb.waitUntilReady(30) == 0)
+ {
+ if (createTableOp)
+ createTable(myNdb, storeInACC, twoKey, longKey);
+
+ if (createIndexOp)
+ createIndex(myNdb, includePrimary, noOfIndexes);
+
+ if (insertOp)
+ insertTable(myNdb, noOfTuples, noOfOperations, oneTransaction, twoKey, longKey);
+
+ if (updateOp)
+ updateTable(myNdb, noOfTuples, noOfOperations, oneTransaction, twoKey, longKey);
+
+ if (deleteOp)
+ deleteTable(myNdb, noOfTuples, noOfOperations, oneTransaction, twoKey, longKey);
+
+ if (readOp)
+ readTable(myNdb, noOfTuples, noOfOperations, oneTransaction, twoKey, longKey);
+
+ if (readIndexOp)
+ readIndex(myNdb, noOfTuples, noOfOperations, includePrimary, oneTransaction, longKey);
+
+ if (updateIndexOp)
+ updateIndex(myNdb, noOfTuples, noOfOperations, includePrimary, oneTransaction, longKey);
+
+ if (deleteIndexOp)
+ deleteIndex(myNdb, noOfTuples, noOfOperations, includePrimary, oneTransaction, longKey);
+
+ if (dropIndexOp)
+ dropIndex(myNdb, noOfIndexes);
+ }
+
+ if (testPassed)
+ {
+ // Test passed
+ ndbout << "OK - Test passed" << endl;
+ }
+ else
+ {
+ // Test failed
+ ndbout << "FAIL - Test failed" << endl;
+ exit(-1);
+ }
+ return 0;
+}
+
+
diff --git a/ndb/test/ndbapi/indexTest2/Makefile b/ndb/test/ndbapi/indexTest2/Makefile
new file mode 100644
index 00000000000..ad78fd51986
--- /dev/null
+++ b/ndb/test/ndbapi/indexTest2/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := index2
+
+SOURCES := index2.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/indexTest2/index2.cpp b/ndb/test/ndbapi/indexTest2/index2.cpp
new file mode 100644
index 00000000000..5a3674f0bbf
--- /dev/null
+++ b/ndb/test/ndbapi/indexTest2/index2.cpp
@@ -0,0 +1,836 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* ***************************************************
+ INDEX TEST 1
+ Test index functionality of NDB
+
+ Arguments:
+ -T create table
+ -L include a long attribute in key or index
+ -2 define primary key with two attributes
+ -c create index
+ -p make index unique (include primary key attribute)
+ -r read using index
+ -u update using index
+ -d delete using index
+ -n<no operations> do n operations (for -I -r -u -d -R -U -D)
+ -o<no parallel operations> (for -I -r -u -d -R -U -D)
+ -m<no indexes>
+
+ Returns:
+ 0 - Test passed
+ -1 - Test failed
+ 1 - Invalid arguments
+ * *************************************************** */
+
+#include <string.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <NdbApi.hpp>
+#include <NdbOut.hpp>
+#include <NdbTick.h>
+#include <NdbMain.h>
+#include <NdbTest.hpp>
+#include <NDBT_Error.hpp>
+
+#ifndef MIN
+#define MIN(x,y) (((x)<(y))?(x):(y))
+#endif
+
+#define MAX_NO_PARALLEL_OPERATIONS 100
+
+bool testPassed = true;
+
+static void
+error_handler(const char* errorText)
+{
+ // Test failed
+ ndbout << endl << "ErrorMessage: " << errorText << endl;
+ testPassed = false;
+}
+
+static void
+error_handler4(int line, int status, int classif, int errNo, const char* errorText)
+{
+ ndbout << endl << "Line " << line << endl;
+ // Test failed
+ ndbout << "Status " << status << ", Classification " << classif<< ", Error code " << errNo << "\n" << errorText << endl;
+ testPassed = false;
+}
+
+static char *longName, *sixtysix, *ninetynine, *hundred;
+
+static void createTable(Ndb &myNdb, bool storeInACC, bool twoKey, bool longKey)
+{
+ NdbDictionary::Dictionary* dict = myNdb.getDictionary();
+ NdbDictionary::Table table("THE_TABLE");
+ NdbDictionary::Column column;
+ int res;
+
+ column.setName("X");
+ column.setPrimaryKey(true);
+ column.setType(NdbDictionary::Column::Unsigned);
+ column.setLength(1);
+ column.setNullable(false);
+ table.addColumn(column);
+
+ column.setName("Y");
+ column.setPrimaryKey(false);
+ column.setType(NdbDictionary::Column::Unsigned);
+ column.setLength(1);
+ column.setNullable(false);
+ table.addColumn(column);
+
+ if ((res = dict->createTable(table)) == -1) {
+ error_handler(dict->getNdbError().message);
+ }
+ else
+ ndbout << "Created table" << ((longKey)?" with long key":"") <<endl;
+}
+
+static void createIndex(Ndb &myNdb, bool includePrimary, unsigned int noOfIndexes)
+{
+ Uint64 before, after;
+ NdbDictionary::Dictionary* dict = myNdb.getDictionary();
+ char indexName[] = "INDEX0000";
+ int res;
+
+ for(unsigned int indexNum = 0; indexNum < noOfIndexes; indexNum++) {
+ sprintf(indexName, "INDEX%.4u", indexNum);
+ NdbDictionary::Index index(indexName);
+ index.setTable("THE_TABLE");
+ index.setType(NdbDictionary::Index::UniqueHashIndex);
+ if (includePrimary) {
+ const char* attr_arr[] = {"X", "Y"};
+ index.addIndexColumns(2, attr_arr);
+ }
+ else {
+ const char* attr_arr[] = {"Y"};
+ index.addIndexColumns(2, attr_arr);
+ }
+ before = NdbTick_CurrentMillisecond();
+ if ((res = dict->createIndex(index)) == -1) {
+ error_handler(dict->getNdbError().message);
+ }
+ after = NdbTick_CurrentMillisecond();
+ ndbout << "Created index " << indexName << ", " << after - before << " msec"<< endl;
+ }
+}
+
+static void insertTable(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool oneTrans, bool twoKey, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbOperation *myOp;
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError().status, myNdb.getNdbError().classification,
+ myNdb.getNdbError().code, myNdb.getNdbError().message);
+
+ myOp = myTrans->getNdbOperation("THE_TABLE");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+
+ myOp->insertTuple();
+ if (myOp->equal("X", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("Y", i+1) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Inserted 1 tuple, %u msec\n", (Uint32) after - before);
+ else
+ printf("Inserted %u tuples, %u msec\n", noOfOperations, (Uint32) after - before);
+
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Inserted "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void updateTable(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool oneTrans, bool twoKey, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbOperation *myOp;
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError().status, myNdb.getNdbError().classification,
+ myNdb.getNdbError().code, myNdb.getNdbError().message);
+
+ myOp = myTrans->getNdbOperation("THE_TABLE");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+
+ myOp->updateTuple();
+ if (myOp->equal("X", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ if (myOp->setValue("Y", i+2) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Updated 1 tuple, %u msec\n", (Uint32) after - before);
+ else
+ printf("Update %u tuples, %u msec\n", noOfOperations, (Uint32) after - before);
+
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Updated "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void deleteTable(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool oneTrans, bool twoKey, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbOperation *myOp;
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError().status, myNdb.getNdbError().classification,
+ myNdb.getNdbError().code, myNdb.getNdbError().message);
+
+ myOp = myTrans->getNdbOperation("THE_TABLE");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+
+ myOp->deleteTuple();
+ if (myOp->equal("X", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Deleted 1 tuple, %u msec\n", (Uint32) after - before);
+ else
+ printf("Deleted %u tuples, %u msec\n", noOfOperations, (Uint32) after - before);
+
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Deleted "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void readTable(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool oneTrans, bool twoKey, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbOperation *myOp;
+ NdbRecAttr* myRecAttrArr[MAX_NO_PARALLEL_OPERATIONS];
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError().status, myNdb.getNdbError().classification,
+ myNdb.getNdbError().code, myNdb.getNdbError().message);
+
+ myOp = myTrans->getNdbOperation("THE_TABLE");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+
+ myOp->readTuple();
+ if (myOp->equal("X", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ myRecAttrArr[j-1] = myOp->getValue("Y", NULL);
+ }
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Read 1 tuple, %u msec\n", (Uint32) after - before);
+ else
+ printf("Read %u tuples, %u msec\n", noOfOperations, (Uint32) after - before);
+ for(unsigned int j = 0; j<noOfOperations; j++)
+ printf("Y = %u\n", myRecAttrArr[j]->u_32_value());
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Read "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void readIndex(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool includePrimary, bool oneTrans, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbIndexOperation *myOp;
+ char indexName[] = "INDEX0000";
+ NdbRecAttr* myRecAttrArr[MAX_NO_PARALLEL_OPERATIONS];
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError().status, myNdb.getNdbError().classification,
+ myNdb.getNdbError().code, myNdb.getNdbError().message);
+
+ myOp = myTrans->getNdbIndexOperation(indexName, "THE_TABLE");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+
+ myOp->readTuple();
+ if (includePrimary) {
+ if (myOp->equal("X", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (myOp->equal("Y", i+1) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ myRecAttrArr[j-1] = myOp->getValue("Y", NULL);
+ }
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Read 1 tuple, %u msec\n", (Uint32) after - before);
+ else
+ printf("Read %u tuples, %u msec\n", noOfOperations, (Uint32) after - before);
+ for(unsigned int j = 0; j<noOfOperations; j++)
+ printf("Y = %u\n", myRecAttrArr[j]->u_32_value());
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Read "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void updateIndex(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool includePrimary, bool oneTrans, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbIndexOperation *myOp;
+ char indexName[] = "INDEX0000";
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError().status, myNdb.getNdbError().classification,
+ myNdb.getNdbError().code, myNdb.getNdbError().message);
+
+ myOp = myTrans->getNdbIndexOperation(indexName, "THE_TABLE");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+
+ myOp->updateTuple();
+ if (includePrimary) {
+ if (myOp->equal("X", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (myOp->equal("Y", i+1) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ // Update index itself, should be possible
+ if (myOp->setValue("Y", i+2) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Updated 1 tuple, %u msec\n", (Uint32) after - before);
+ else
+ printf("Updated %u tuples, %u msec\n", noOfOperations, (Uint32) after - before);
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Updated "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void deleteIndex(Ndb &myNdb, unsigned int noOfTuples, unsigned int noOfOperations, bool includePrimary, bool oneTrans, bool longKey)
+{
+ Uint64 tbefore, tafter, before, after;
+ NdbConnection *myTrans;
+ NdbIndexOperation *myOp;
+ char indexName[] = "INDEX0000";
+
+ tbefore = NdbTick_CurrentMillisecond();
+ if (oneTrans) myTrans = myNdb.startTransaction();
+ for (unsigned int i = 0; i<noOfTuples; i++) {
+ for(unsigned int j = 1;
+ ((j<=noOfOperations)&&(i<noOfTuples));
+ (++j<=noOfOperations)?i++:i) {
+ if (!oneTrans) myTrans = myNdb.startTransaction();
+ if (myTrans == NULL)
+ error_handler4(__LINE__, myNdb.getNdbError().status, myNdb.getNdbError().classification,
+ myNdb.getNdbError().code, myNdb.getNdbError().message);
+
+ myOp = myTrans->getNdbIndexOperation(indexName, "THE_TABLE");
+ if (myOp == NULL)
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+
+ myOp->deleteTuple();
+ if (includePrimary) {
+ if (myOp->equal("X", i) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ if (myOp->equal("Y", i+1) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ }
+ before = NdbTick_CurrentMillisecond();
+ if (myTrans->execute( (oneTrans) ? NoCommit : Commit ) == -1)
+ {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ myNdb.closeTransaction(myTrans);
+ break;
+ }
+ after = NdbTick_CurrentMillisecond();
+ if (noOfOperations == 1)
+ printf("Deleted 1 tuple, %u msec\n", (Uint32) after - before);
+ else
+ printf("Deleted %u tuples, %u msec\n", noOfOperations, (Uint32) after - before);
+ if (!oneTrans) myNdb.closeTransaction(myTrans);
+ }
+ if (oneTrans) {
+ if (myTrans->execute( Commit ) == -1) {
+ error_handler4(__LINE__, myTrans->getNdbError().status, myTrans->getNdbError().classification,
+ myTrans->getNdbError().code, myTrans->getNdbError().message);
+ }
+ myNdb.closeTransaction(myTrans);
+ }
+ tafter = NdbTick_CurrentMillisecond();
+
+ ndbout << "Deleted "<< noOfTuples << " tuples in " << ((oneTrans) ? 1 : noOfTuples) << " transaction(s), " << tafter - tbefore << " msec" << endl;
+}
+
+static void dropIndex(Ndb &myNdb, unsigned int noOfIndexes)
+{
+ for(unsigned int indexNum = 0; indexNum < noOfIndexes; indexNum++) {
+ char indexName[255];
+ sprintf(indexName, "INDEX%.4u", indexNum);
+ const Uint64 before = NdbTick_CurrentMillisecond();
+ const int retVal = myNdb.getDictionary()->dropIndex(indexName,"THE_TABLE");
+ const Uint64 after = NdbTick_CurrentMillisecond();
+
+ if(retVal == 0){
+ ndbout << "Dropped index " << indexName << ", "
+ << after - before << " msec" << endl;
+ } else {
+ ndbout << "Failed to drop index " << indexName << endl;
+ ndbout << myNdb.getDictionary()->getNdbError() << endl;
+ }
+ }
+}
+
+NDB_COMMAND(indexTest, "indexTest", "indexTest", "indexTest", 65535)
+{
+ bool createTableOp, createIndexOp, dropIndexOp, insertOp, updateOp, deleteOp, readOp, readIndexOp, updateIndexOp, deleteIndexOp, twoKey, longKey;
+ unsigned int noOfTuples = 1;
+ unsigned int noOfOperations = 1;
+ unsigned int noOfIndexes = 1;
+ int i = 1;
+ Ndb myNdb( "TEST_DB" );
+ int check;
+ bool storeInACC = false;
+ bool includePrimary = false;
+ bool oneTransaction = false;
+
+ createTableOp = createIndexOp = dropIndexOp = insertOp = updateOp = deleteOp = readOp = readIndexOp = updateIndexOp = deleteIndexOp = twoKey = longKey = false;
+ // Read arguments
+ if (argc > 1)
+ while (argc > 1)
+ {
+ if (strcmp(argv[i], "-T") == 0)
+ {
+ createTableOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-c") == 0)
+ {
+ createIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-X") == 0)
+ {
+ dropIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-I") == 0)
+ {
+ insertOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-D") == 0)
+ {
+ deleteOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-U") == 0)
+ {
+ updateOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-R") == 0)
+ {
+ readOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-r") == 0)
+ {
+ readIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-u") == 0)
+ {
+ updateIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-d") == 0)
+ {
+ deleteIndexOp = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-s") == 0)
+ {
+ storeInACC = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-p") == 0)
+ {
+ includePrimary = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-L") == 0)
+ {
+ longKey = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-1") == 0)
+ {
+ oneTransaction = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strcmp(argv[i], "-2") == 0)
+ {
+ twoKey = true;
+ argc -= 1;
+ i++;
+ }
+ else if (strstr(argv[i], "-n") != 0)
+ {
+ noOfTuples = atoi(argv[i]+2);
+ argc -= 1;
+ i++;
+ }
+ else if (strstr(argv[i], "-o") != 0)
+ {
+ noOfOperations = MIN(MAX_NO_PARALLEL_OPERATIONS, atoi(argv[i]+2));
+ argc -= 1;
+ i++;
+ }
+ else if (strstr(argv[i], "-m") != 0)
+ {
+ noOfIndexes = atoi(argv[i]+2);
+ argc -= 1;
+ i++;
+ }
+ else if (strstr(argv[i], "-h") != 0)
+ {
+ printf("Synopsis: \
+ index\
+ -T create table\
+ -L include a long attribute in key or index\
+ -2 define primary key with two attributes\
+ -c create index\
+ -p make index unique (include primary key attribute)\
+ -r read using index\
+ -u update using index\
+ -d delete using index\
+ -n<no operations> do n operations (for -I -r -u -d -R -U -D)\
+ -o<no parallel operations> (for -I -r -u -d -R -U -D)\
+ -m<no indexes>\n");
+ argc -= 1;
+ i++;
+ }
+ else {
+ char errStr[256];
+
+ sprintf(errStr, "Illegal argument: %s", argv[i]);
+ error_handler(errStr);
+ exit(-1);
+ }
+ }
+ else
+ {
+ createTableOp = createIndexOp = dropIndexOp = insertOp = updateOp = deleteOp = true;
+ }
+ if (longKey) {
+ longName = (char *) malloc(1024);
+ for (int i = 0; i < 1023; i++)
+ longName[i] = 'x';
+ longName[1023] = '\0';
+ }
+ sixtysix = (char *) malloc(256);
+ for (int i = 0; i < 255; i++)
+ sixtysix[i] = ' ';
+ sixtysix[255] = '\0';
+ strncpy(sixtysix, "sixtysix", strlen("sixtysix"));
+ ninetynine = (char *) malloc(256);
+ for (int i = 0; i < 255; i++)
+ ninetynine[i] = ' ';
+ ninetynine[255] = '\0';
+ strncpy(ninetynine, "ninetynine", strlen("ninetynine"));
+ hundred = (char *) malloc(256);
+ for (int i = 0; i < 255; i++)
+ hundred[i] = ' ';
+ hundred[255] = '\0';
+ strncpy(hundred, "hundred", strlen("hundred"));
+ myNdb.init();
+ // Wait for Ndb to become ready
+ if (myNdb.waitUntilReady(30) == 0)
+ {
+ if (createTableOp)
+ createTable(myNdb, storeInACC, twoKey, longKey);
+
+ if (createIndexOp)
+ createIndex(myNdb, includePrimary, noOfIndexes);
+
+ if (insertOp)
+ insertTable(myNdb, noOfTuples, noOfOperations, oneTransaction, twoKey, longKey);
+
+ if (updateOp)
+ updateTable(myNdb, noOfTuples, noOfOperations, oneTransaction, twoKey, longKey);
+
+ if (deleteOp)
+ deleteTable(myNdb, noOfTuples, noOfOperations, oneTransaction, twoKey, longKey);
+
+ if (readOp)
+ readTable(myNdb, noOfTuples, noOfOperations, oneTransaction, twoKey, longKey);
+
+ if (readIndexOp)
+ readIndex(myNdb, noOfTuples, noOfOperations, includePrimary, oneTransaction, longKey);
+
+ if (updateIndexOp)
+ updateIndex(myNdb, noOfTuples, noOfOperations, includePrimary, oneTransaction, longKey);
+
+ if (deleteIndexOp)
+ deleteIndex(myNdb, noOfTuples, noOfOperations, includePrimary, oneTransaction, longKey);
+
+ if (dropIndexOp)
+ dropIndex(myNdb, noOfIndexes);
+ }
+
+ if (testPassed)
+ {
+ // Test passed
+ ndbout << "OK - Test passed" << endl;
+ }
+ else
+ {
+ // Test failed
+ ndbout << "FAIL - Test failed" << endl;
+ exit(-1);
+ }
+ return NULL;
+}
+
+
diff --git a/ndb/test/ndbapi/interpreterInTup/Makefile b/ndb/test/ndbapi/interpreterInTup/Makefile
new file mode 100644
index 00000000000..074adbf674a
--- /dev/null
+++ b/ndb/test/ndbapi/interpreterInTup/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+
+BIN_TARGET := interpreterInTup
+
+SOURCES := interpreterInTup.cc
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/interpreterInTup/interpreterInTup.cpp b/ndb/test/ndbapi/interpreterInTup/interpreterInTup.cpp
new file mode 100644
index 00000000000..b9d1eca1cc9
--- /dev/null
+++ b/ndb/test/ndbapi/interpreterInTup/interpreterInTup.cpp
@@ -0,0 +1,1527 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* ***************************************************
+ TEST OF INTERPRETER IN TUP
+ Verify that the interpreter in TUP is able to
+ handle and execute all the commands that the
+ NdbApi can isssue
+
+ Arguments:
+
+ operationType 1 openScanRead,
+ 2 openScanExclusive,
+ 3 interpretedUpdateTuple,
+ 4 interpretedDirtyUpdate,
+ 5 interpretedDeleteTuple
+ 6 deleteTuple
+ 7 insertTuple
+ 8 updateTuple
+ 9 writeTuple
+ 10 readTuple
+ 11 readTupleExclusive
+ 12 simpleRead
+ 13 dirtyRead
+ 14 dirtyUpdate
+ 15 dirtyWrite
+
+ tupTest 1 exitMethods
+ 2 incValue
+ 3 subValue
+ 4 readAttr
+ 5 writeAttr
+ 6 loadConst
+ 7 branch
+ 8 branchIfNull
+ 9 addReg
+ 10 subReg
+ 11 subroutineWithBranchLabel
+
+ scanTest Number of the test within each tupTest
+
+* *************************************************** */
+
+#include <NdbStdio.h>
+#include <string.h>
+#include <assert.h>
+#include <NdbOut.hpp>
+#include <NdbThread.h>
+#include <NdbMutex.h>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+
+#define MAXATTR 3
+#define MAXTABLES 12
+#define MAXSTRLEN 8
+#define NUMBEROFRECORDS 1000
+
+typedef enum {
+ FAIL = 0,
+ NO_FAIL,
+ UNDEF
+} TTYPE;
+
+inline void setAttrNames() ;
+inline void setTableNames() ;
+void error_handler(const NdbError & err, TTYPE);
+void create_table(Ndb*);
+void write_rows(Ndb*);
+void update_rows(Ndb*, int, int);
+void delete_rows(Ndb*, int, int);
+void verify_deleted(Ndb*);
+void read_and_verify_rows(Ndb*, bool pre);
+void scan_rows(Ndb*, int, int, int);
+TTYPE t_exitMethods(int, NdbOperation*, int);
+TTYPE t_incValue(int, NdbOperation*);
+TTYPE t_subValue(int, NdbOperation*);
+TTYPE t_readAttr(int, NdbOperation*);
+TTYPE t_writeAttr(int, NdbOperation*);
+TTYPE t_loadConst(int, NdbOperation*, int);
+TTYPE t_branch(int, NdbOperation*);
+TTYPE t_branchIfNull(int, NdbOperation*);
+TTYPE t_addReg(int, NdbOperation*);
+TTYPE t_subReg(int, NdbOperation*);
+TTYPE t_subroutineWithBranchLabel(int, NdbOperation*);
+
+char tableName[MAXTABLES][MAXSTRLEN+1] = {0};
+char attrName[MAXATTR][MAXSTRLEN+1] = {0};
+int attrValue[NUMBEROFRECORDS] = {0};
+int pkValue[NUMBEROFRECORDS] = {0};
+const int tAttributeSize = 1 ;
+const int nRecords = 20 ;
+int bTestPassed = 0;
+
+
+
+int main(int argc, const char** argv) {
+
+ int tTableId = 0;
+ int operationType = 0;
+ int tupTest = 0 ;
+ int scanTest = 0 ;
+ bool loop = 0 ;
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ pNdb->init();
+
+ if (argc != 4 || sscanf(argv[1],"%d", &operationType) != 1) {
+ operationType = 1 ;
+ }
+ if (argc != 4 || sscanf(argv[2],"%d", &tupTest) != 1) {
+ tupTest = 1 ;
+ }
+ if (argc != 4 || sscanf(argv[3],"%d", &scanTest) != 1) {
+ scanTest = 1 ;
+ }
+
+ ndbout << endl
+ << "Test the interpreter in TUP using SimpleTable with\n"
+ << nRecords << " records" << endl << endl ;
+
+ if (pNdb->waitUntilReady(30) != 0) {
+ ndbout << "NDB is not ready" << endl;
+ return -1;
+ }
+
+ // Init the pk and attr values.
+ for (int i = 0; i < NUMBEROFRECORDS; i ++)
+ pkValue[i] = attrValue[i] = i ;
+
+ setAttrNames() ;
+ setTableNames() ;
+
+ const void * p = NDBT_Table::discoverTableFromDb(pNdb, tableName[0]);
+ if (p != 0){
+ create_table(pNdb);
+ }
+
+ write_rows(pNdb);
+
+ ndbout << endl << "Starting interpreter in TUP test." << endl << "Operation type: " ;
+
+ switch(operationType) {
+ case 1:
+ ndbout << "openScanRead" << endl;
+ scan_rows(pNdb, operationType, tupTest, scanTest);
+ break;
+ case 2:
+ ndbout << "openScanExclusive" << endl;
+ scan_rows(pNdb, operationType, tupTest, scanTest);
+ break;
+ case 3:
+ ndbout << "interpretedUpdateTuple" << endl;
+ update_rows(pNdb, tupTest, operationType);
+ break;
+ case 4:
+ ndbout << "interpretedDirtyUpdate" << endl;
+ update_rows(pNdb, tupTest, operationType);
+ break;
+ case 5:
+ ndbout << "interpretedDeleteTuple" << endl;
+ delete_rows(pNdb, tupTest, operationType);
+ break;
+ case 6:
+ ndbout << "deleteTuple" << endl;
+ break;
+ case 7:
+ ndbout << "insertTuple" << endl;
+ break;
+ case 8:
+ ndbout << "updateTuple" << endl;
+ break;
+ case 9:
+ ndbout << "writeTuple" << endl;
+ break;
+ case 10:
+ ndbout << "readTuple" << endl;
+ break;
+ case 11:
+ ndbout << "readTupleExclusive" << endl;
+ break;
+ case 12:
+ ndbout << "simpleRead" << endl;
+ break;
+ case 13:
+ ndbout << "dirtyRead" << endl;
+ break;
+ case 14:
+ ndbout << "dirtyUpdate" << endl;
+ break;
+ case 15:
+ ndbout << "dirtyWrite" << endl;
+ break;
+ default:
+ break ;
+
+ }
+
+// read_and_verify_rows(pNdb, false);
+
+// delete_rows(pNdb, 0, 0) ;
+ delete pNdb ;
+
+ if (bTestPassed == 0) {
+ ndbout << "OK: test passed" << endl;
+ exit(0);
+ }else{
+ ndbout << "FAIL: test failed" << endl;
+ exit(-1);
+ }
+}
+
+void error_handler(const NdbError & err, TTYPE type_expected) {
+
+ ndbout << err << endl ;
+
+ switch (type_expected){
+ case NO_FAIL:
+ bTestPassed = -1 ;
+ break ;
+ case FAIL:
+ ndbout << "OK: error is expected" << endl;
+ break ;
+ case UNDEF:
+ ndbout << "assumed OK: expected result undefined" << endl ;
+ break ;
+ default:
+ break ;
+ }
+}
+
+void create_table(Ndb* pMyNdb) {
+
+ /****************************************************************
+ * Create SimpleTable and Attributes.
+ *
+ * create table SimpleTable1(
+ * col0 int,
+ * col1 int not null,
+ * col2 int not null,
+ * col3 int not null ... 129)
+ *
+ ***************************************************************/
+
+ int check = -1 ;
+ NdbSchemaCon *MySchemaTransaction = NULL ;
+ NdbSchemaOp *MySchemaOp = NULL ;
+
+ ndbout << endl << "Creating " << tableName[0] << " ... " << endl;
+
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+ if(!MySchemaTransaction) error_handler(MySchemaTransaction->getNdbError(), NO_FAIL);
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( !MySchemaOp ) error_handler(MySchemaTransaction->getNdbError(), NO_FAIL);
+ // Create table
+ check = MySchemaOp->createTable( tableName[0],
+ 8, // Table size
+ TupleKey, // Key Type
+ 40 // Nr of Pages
+ );
+
+ if( check == -1 ) error_handler(MySchemaTransaction->getNdbError(), NO_FAIL);
+
+ ndbout << "Creating attributes ... " << flush;
+
+ // Create first column, primary key
+ check = MySchemaOp->createAttribute( attrName[0],
+ TupleKey,
+ 32,
+ 1/*3, tAttributeSize */,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( check == -1 ) error_handler(MySchemaTransaction->getNdbError(), NO_FAIL);
+
+ // create the 2 .. n columns.
+ for ( int i = 1; i < MAXATTR; i++ ){
+ check = MySchemaOp->createAttribute( attrName[i],
+ NoKey,
+ 32,
+ tAttributeSize,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if( check == -1 ) error_handler(MySchemaTransaction->getNdbError(), NO_FAIL);
+ }
+
+ ndbout << "OK" << endl;
+
+ if( MySchemaTransaction->execute() == -1 ) {
+ ndbout << MySchemaTransaction->getNdbError() << endl;
+ }else{
+ ndbout << tableName[0] << " created" << endl;
+ }
+
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+
+ return;
+
+}
+
+
+
+void write_rows (Ndb* pMyNdb) {
+
+ /****************************************************************
+ * Insert rows into SimpleTable
+ *
+ ***************************************************************/
+
+ int check = -1 ;
+ int loop_count_ops = nRecords ;
+ NdbOperation *MyOperation = NULL ;
+ NdbConnection *MyTransaction = NULL ;
+
+ ndbout << endl << "Writing records ..." << flush;
+
+ for (int count=0 ; count < loop_count_ops ; count++){
+ MyTransaction = pMyNdb->startTransaction();
+ if (!MyTransaction) {
+ error_handler(pMyNdb->getNdbError(), NO_FAIL);
+ }//if
+
+ MyOperation = MyTransaction->getNdbOperation(tableName[0]);
+ if (!MyOperation) {
+ error_handler(pMyNdb->getNdbError(), NO_FAIL);
+ }//if
+
+ check = MyOperation->writeTuple();
+ if( check == -1 ) error_handler(MyTransaction->getNdbError(), NO_FAIL);
+
+ check = MyOperation->equal( attrName[0],(char*)&pkValue[count] );
+ if( check == -1 ) error_handler(MyTransaction->getNdbError(), NO_FAIL);
+
+ // Update the columns, index column already ok.
+ for (int i = 1 ; i < MAXATTR; i++){
+ if ((i == 2) && (count > 4)){
+ check = MyOperation->setValue(attrName[i], (char*)&attrValue[count + 1]);
+ }else{
+ check = MyOperation->setValue(attrName[i], (char*)&attrValue[count]);
+ }
+ if( check == -1 ) error_handler(MyTransaction->getNdbError(), NO_FAIL);
+ }
+ check = MyTransaction->execute( Commit );
+ if(check == -1 ) error_handler(MyTransaction->getNdbError(), NO_FAIL);
+
+ pMyNdb->closeTransaction(MyTransaction);
+ }
+ ndbout <<" \tOK" << endl;
+ return;
+}
+
+void verify_deleted(Ndb* pMyNdb) {
+
+ int check = -1 ;
+ int loop_count_ops = nRecords;
+ NdbRecAttr* tTmp;
+ int readValue[MAXATTR];
+ NdbConnection* pMyTransaction = NULL ;
+ NdbOperation* pMyOperation = NULL ;
+
+ ndbout << "Verifying deleted records..."<< flush;
+
+ for (int count=0 ; count < loop_count_ops ; count++){
+
+ NdbConnection* pMyTransaction = pMyNdb->startTransaction();
+ if (!pMyTransaction) error_handler(pMyNdb->getNdbError(), NO_FAIL);
+
+ pMyOperation = pMyTransaction->getNdbOperation(tableName[0]);
+ if (!pMyOperation) error_handler(pMyNdb->getNdbError(), NO_FAIL);
+
+ check = pMyOperation->readTuple();
+ if( check == -1 ) error_handler( pMyTransaction->getNdbError(), NO_FAIL);
+
+ check = pMyOperation->equal( attrName[0],(char*)&pkValue[count] );
+ if( check == -1 ) error_handler( pMyTransaction->getNdbError(), NO_FAIL);
+
+ // Exepect to receive an error
+ if(pMyTransaction->execute(Commit) != -1)
+ if( 626 == pMyTransaction->getNdbError().code){
+ ndbout << pMyTransaction->getNdbError() << endl ;
+ ndbout << "OK" << endl ;
+ }else{
+ error_handler(pMyTransaction->getNdbError(), NO_FAIL) ;
+ }
+
+ pMyNdb->closeTransaction(pMyTransaction);
+ }
+
+ ndbout << "OK" << endl;
+ return;
+};
+
+void read_and_verify_rows(Ndb* pMyNdb, bool pre) {
+
+ int check = -1 ;
+ int loop_count_ops = nRecords;
+ char expectedCOL1[NUMBEROFRECORDS] = {0} ;
+ char expectedCOL2[NUMBEROFRECORDS] = {0} ;
+ NdbConnection *pMyTransaction = NULL ;
+ NdbOperation *MyOp = NULL ;
+ NdbRecAttr* tTmp = NULL ;
+ int readValue[MAXATTR] = {0} ;
+
+ ndbout << "Verifying records...\n"<< endl;
+
+ for (int count=0 ; count < loop_count_ops ; count++){
+
+ pMyTransaction = pMyNdb->startTransaction();
+ if (!pMyTransaction) error_handler(pMyNdb->getNdbError(), NO_FAIL);
+
+ MyOp = pMyTransaction->getNdbOperation(tableName[0]);
+ if (!MyOp) error_handler( pMyTransaction->getNdbError(), NO_FAIL);
+
+
+ check = MyOp->readTuple();
+ if( check == -1 ) error_handler( MyOp->getNdbError(), NO_FAIL);
+
+ check = MyOp->equal( attrName[0],(char*)&pkValue[count] );
+ if( check == -1 ) error_handler( MyOp->getNdbError(), NO_FAIL);
+
+ for (int count_attributes = 1; count_attributes < MAXATTR; count_attributes++){
+
+ tTmp = MyOp->getValue( (char*)attrName[count_attributes], (char*)&readValue[count_attributes] );
+ if(!tTmp) error_handler( MyOp->getNdbError(), NO_FAIL);
+ }
+
+ if( pMyTransaction->execute( Commit ) == -1 ) {
+ error_handler(pMyTransaction->getNdbError(), NO_FAIL);
+ } else {
+ if (pre) {
+ expectedCOL1[count] = readValue[1];
+ expectedCOL2[count] = readValue[2];
+ }
+
+ ndbout << attrName[1] << "\t " << readValue[1] << "\t "
+ << attrName[2] << "\t " << readValue[2] << endl;
+ }
+
+ pMyNdb->closeTransaction(pMyTransaction);
+
+ }
+
+ ndbout << "\nOK\n" << endl;
+
+ return;
+
+};
+
+TTYPE t_exitMethods(int nCalls, NdbOperation * pOp, int opType) {
+
+ ndbout << "Defining exitMethods test " << nCalls << ": " << endl ;;
+ TTYPE ret_val = NO_FAIL ;
+
+ switch(nCalls){
+ case 1: // exit_nok if attr value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 14);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_ok() ;
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break;
+ case 2: // exit_ok if attr value doesn't match
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 14);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_nok() ;
+ pOp->def_label(0);
+ if (opType == 3) {
+ // For update transactions use incValue to update the tuple
+ Uint32 val32 = 5;
+ pOp->incValue("COL2", val32);
+ }
+ pOp->interpret_exit_ok();
+ break ;
+ case 3: // Non-existent value (128): exit_ok if if the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 128);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ pOp->interpret_exit_ok();
+ ret_val = FAIL ;
+ break;
+ case 4: // Non-existent value (128): exit_nok if the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 128);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ ret_val = FAIL ;
+ break;
+ case 5: // exit_nok of the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 2);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break ;
+ case 6: // exit_ok of the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 2);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ if (opType == 3) {
+ // For update transactions use incValue to update the tuple
+ Uint32 val32 = 5;
+ pOp->incValue("COL2", val32);
+ }
+ pOp->interpret_exit_ok();
+ break;
+ case 7: // exit_nok if the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 6);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break;
+ case 8: // exit_ok if the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 6);
+ pOp->branch_ne(1, 2, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ if (opType == 3) {
+ // For update transactions use incValue to update the tuple
+ Uint32 val32 = 5;
+ pOp->incValue("COL2", val32);
+ }
+ pOp->interpret_exit_ok();
+ break ;
+ case 9: // exit_nok if the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 8);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break;
+ case 10: // exit_ok if the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 8);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ if (opType == 3) {
+ // For update transactions use incValue to update the tuple
+ Uint32 val32 = 5;
+ pOp->incValue("COL2", val32);
+ }
+ pOp->interpret_exit_ok();
+ break;
+ case 11: // exit_nok if the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 10);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break;
+ case 12:
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 10);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ if (opType == 3) {
+ // For update transactions use incValue to update the tuple
+ Uint32 val32 = 5;
+ pOp->incValue("COL2", val32);
+ }
+ pOp->interpret_exit_ok();
+ break;
+ case 13:
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 10);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break;
+ case 14: // exit_nok if the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 12);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break;
+ case 15: // exit_ok if the value matches
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, 12);
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ if (opType == 3) {
+ // For update transactions use incValue to update the tuple
+ Uint32 val32 = 5;
+ pOp->incValue("COL2", val32);
+ }
+ pOp->interpret_exit_ok();
+ case 16:
+ pOp->interpret_exit_nok();
+ ret_val = FAIL ;
+ break;
+ case 17:
+ pOp->interpret_exit_ok();
+ break ;
+ case 18:
+ pOp->interpret_exit_nok();
+ ret_val = FAIL ;
+ break ;
+ default:
+ break ;
+ }
+ return ret_val;
+}
+
+TTYPE t_incValue(int nCalls, NdbOperation* pOp) {
+
+ ndbout << "Defining incValue test " << nCalls << ": ";
+ TTYPE ret_val = NO_FAIL;
+
+ Uint32 val32 = 5;
+ Uint64 val64 = 5;
+ Uint32 attr0 = 0;
+ Uint32 attr1 = 1;
+ Uint32 attr20 = 20;
+
+ switch(nCalls) {
+ case 1:
+ pOp->incValue(attrName[1], val32);
+ break;
+ case 2:
+ pOp->incValue(attr1, val32);
+ break;
+ case 3:
+ pOp->incValue(attrName[1], val64);
+ break;
+ case 4:
+ pOp->incValue(attr1, val64);
+ break;
+ case 5:
+ pOp->incValue(attrName[0], val32);
+ ret_val = FAIL ;
+ break;
+ case 6:
+ pOp->incValue(attrName[0], val64);
+ ret_val = FAIL ;
+ break;
+ case 7:
+ pOp->incValue(attr0, val32);
+ ret_val = FAIL ;
+ break;
+ case 8:
+ pOp->incValue(attr0, val64);
+ ret_val = FAIL ;
+ break;
+ case 9:
+ pOp->incValue("COL20", val32);
+ ret_val = FAIL ;
+ break;
+ case 10:
+ pOp->incValue("COL20", val64);
+ ret_val = FAIL ;
+ break;
+ case 11:
+ pOp->incValue(attr20, val32);
+ ret_val = FAIL ;
+ break;
+ case 12:
+ pOp->incValue(attr20, val64);
+ ret_val = FAIL ;
+ break;
+ default:
+ break ;
+ }
+ return ret_val;
+}
+
+TTYPE t_subValue(int nCalls, NdbOperation* pOp) {
+
+ ndbout << "Defining subValue test " << nCalls << ": ";
+
+ Uint32 val32 = 5;
+ Uint64 val64 = 5;
+ Uint32 attr0 = 0;
+ Uint32 attr1 = 1;
+ Uint32 attr20 = 20;
+
+ TTYPE ret_val = NO_FAIL;
+
+ switch(nCalls) {
+ case 1:
+ pOp->subValue("COL2", val32);
+ break;
+ case 2:
+ pOp->subValue(attr1, val32);
+ break;
+ case 3:
+ pOp->subValue("COL0", val32);
+ ret_val = FAIL ;
+ break;
+ case 4:
+ pOp->subValue(attr0, val32);
+ ret_val = FAIL ;
+ break;
+ case 5:
+ pOp->subValue("COL20", val32);
+ ret_val = FAIL ;
+ break;
+ case 6:
+ pOp->subValue(attr20, val32);
+ ret_val = FAIL ;
+ break;
+ case 7:
+ // Missing implementation
+ //pOp->subValue("COL20", val64);
+ ndbout << "Missing implementation" << endl;
+ break;
+ case 8:
+ // Missing implementation
+ //pOp->subValue("COL2", val64);
+ ndbout << "Missing implementation" << endl;
+ break;
+ case 9:
+ // Missing implementation
+ //pOp->subValue("COL0", val64);
+ ndbout << "Missing implementation" << endl;
+ break;
+ case 10:
+ // Missing implementation
+ //pOp->subValue(attr1, val64);
+ ndbout << "Missing implementation" << endl;
+ break;
+ case 11:
+ // Missing implementation
+ //pOp->subValue(attr0, val64);
+ ndbout << "Missing implementation" << endl;
+ break;
+ case 12:
+ // Missing implementation
+ //pOp->subValue(attr20, val64);
+ ndbout << "Missing implementation" << endl;
+ break;
+ default:
+ break ;
+ }
+ return ret_val ;
+}
+
+TTYPE t_readAttr(int nCalls, NdbOperation* pOp) {
+
+ ndbout << "Defining readAttr test " << nCalls << ": ";
+
+ Uint32 attr0 = 0;
+ Uint32 attr1 = 1;
+ Uint32 attr20 = 20;
+ TTYPE ret_val = NO_FAIL;
+
+ switch(nCalls) {
+ case 1:
+ pOp->read_attr("COL1", 1);
+ break;
+ case 2:
+ pOp->read_attr(attr1, 1);
+ ret_val = NO_FAIL ;
+ break;
+ case 3:
+ pOp->read_attr("COL0", 1);
+ ret_val = NO_FAIL ;
+ break;
+ case 4:
+ pOp->read_attr(attr0, 1);
+ ret_val = NO_FAIL ;
+ break;
+ case 5:
+ pOp->read_attr("COL20", 1);
+ ret_val = FAIL ;
+ break;
+ case 6:
+ pOp->read_attr(20, 1);
+ ret_val = FAIL ;
+ break;
+ case 7:
+ pOp->read_attr("COL1", 8);
+ ret_val = FAIL ;
+ break;
+ case 8:
+ pOp->read_attr(attr1, 8);
+ ret_val = FAIL ;
+ break;
+ default:
+ break ;
+ }
+ return ret_val;
+}
+
+TTYPE t_writeAttr(int nCalls, NdbOperation* pOp) {
+
+ ndbout << "Defining writeAttr test " << nCalls << ": ";
+
+ pOp->load_const_u32(1, 5);
+
+ Uint32 attr0 = 0;
+ Uint32 attr1 = 1;
+ Uint32 attr20 = 20;
+ TTYPE ret_val = NO_FAIL ;
+
+ switch(nCalls) {
+ case 1:
+ pOp->write_attr("COL1", 1);
+ break;
+ case 2:
+ pOp->write_attr(attr1, 1);
+ break;
+ case 3:
+ pOp->write_attr("COL0", 1);
+ ret_val = FAIL ;
+ break;
+ case 4:
+ pOp->write_attr(attr0, 1);
+ ret_val = FAIL ;
+ break;
+ case 5:
+ pOp->write_attr("COL20", 1);
+ ret_val = FAIL ;
+ break;
+ case 6:
+ pOp->write_attr(20, 1);
+ ret_val = FAIL ;
+ break;
+ case 7:
+ pOp->write_attr("COL1", 2);
+ ret_val = FAIL ;
+ break;
+ case 8:
+ pOp->write_attr(attr1, 2);
+ ret_val = FAIL ;
+ break;
+ case 9:
+ pOp->write_attr("COL1", 8);
+ ret_val = FAIL ;
+ break;
+ case 10:
+ pOp->write_attr(attr1, 8);
+ ret_val = FAIL ;
+ break;
+ default:
+ break ;
+ }
+ return ret_val ;
+}
+
+TTYPE t_loadConst(int nCalls, NdbOperation* pOp, int opType) {
+
+ ndbout << "Defining loadConst test " << nCalls << " : ";
+ TTYPE ret_val = NO_FAIL ;
+
+ switch(nCalls) {
+ case 1:
+ // Loading null into a valid register
+ pOp->load_const_null(1);
+ break;
+ case 2:
+ // Loading null into an invalid register
+ pOp->load_const_null(8);
+ ret_val = FAIL ;
+ break;
+ case 3:
+ // Test loading a 32-bit value (>65536)
+ pOp->load_const_u32(1, 65600);
+ break;
+ case 4:
+ // Test loading a 16-bit value (<65536)
+ pOp->load_const_u32(1, 65500);
+ break;
+ case 5:
+ // Test by loading to a non-valid register
+ pOp->load_const_u32(8, 2);
+ ret_val = FAIL ;
+ break;
+ case 6:
+ // Test by loading a 64-bit value
+ pOp->load_const_u64(1, 65600);
+ break;
+ case 7:
+ // Test by loading a non-valid register
+ pOp->load_const_u64(8, 2);
+ ret_val = FAIL ;
+ break;
+ case 8:
+ // Test by loading a valid register with -1
+ pOp->load_const_u64(1, -1);
+ ret_val = FAIL ;
+ break;
+
+ default:
+ break ;
+ }
+
+ if (opType == 3 && FAIL != ret_val)
+ pOp->write_attr("COL1", 1);
+ return ret_val;
+}
+
+TTYPE t_branch(int nCalls, NdbOperation* pOp) {
+
+ ndbout << "Defining branch test " << nCalls << ": " ;
+
+ TTYPE ret_val = NO_FAIL ;
+ Uint32 val32=5;
+
+ pOp->read_attr("COL1", 1);
+ pOp->load_const_u32(2, val32);
+
+ switch(nCalls) {
+ case 1:
+ pOp->branch_eq(1, 2, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ pOp->interpret_exit_ok();
+ break;
+ case 2:
+ pOp->branch_eq(2, 1, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ pOp->interpret_exit_ok();
+ break;
+ case 3:
+ pOp->branch_eq(1, 1, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ pOp->interpret_exit_ok();
+ break;
+ default:
+ //ndbout << "t_branch: default case (no test case)" << endl ;
+ //return ret_val = NO_FAIL ;
+ break ;
+ }
+ return ret_val;
+}
+
+TTYPE t_branchIfNull(int nCalls, NdbOperation* pOp) {
+
+ TTYPE ret_val = NO_FAIL;
+ ndbout << "Defining branchIfNull test " << nCalls << ": " << endl ;
+
+ switch(nCalls) {
+ case 1:
+ pOp->load_const_u32(1, 1);
+ pOp->branch_ne_null(1, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ pOp->interpret_exit_ok();
+ break;
+ case 2:
+ pOp->load_const_null(1);
+ pOp->branch_ne_null(1, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break;
+ case 3:
+ pOp->load_const_u32(1, 1);
+ pOp->branch_eq_null(1, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break;
+ case 4:
+ pOp->load_const_null(1);
+ pOp->branch_ne_null(1, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ break;
+ case 5:
+ // Test with a non-initialized register
+ pOp->branch_ne_null(3, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ ret_val = FAIL ;
+ break;
+ case 6:
+ // Test with a non-existing register
+ pOp->branch_ne_null(8, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ ret_val = FAIL ;
+ break;
+ case 7:
+ // Test with a non-initialized register
+ pOp->branch_eq_null(3, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ ret_val = FAIL ;
+ break;
+ case 8:
+ // Test with a non-existing register
+ pOp->branch_ne_null(8, 0);
+ pOp->interpret_exit_ok();
+ pOp->def_label(0);
+ pOp->interpret_exit_nok();
+ ret_val = FAIL ;
+ break;
+ default:
+ break ;
+ }
+
+ return ret_val;
+}
+
+TTYPE t_addReg(int nCalls, NdbOperation* pOp) {
+
+ TTYPE ret_val = NO_FAIL ;
+
+ ndbout << "Defining addReg test " << nCalls << ": ";
+
+ pOp->load_const_u32(1, 65500);
+ pOp->load_const_u32(2, 500);
+ pOp->load_const_u64(3, 65600);
+ pOp->load_const_u64(4, 95600);
+
+ switch(nCalls) {
+ case 1:
+ pOp->add_reg(1, 2, 5);
+ break;
+ case 2:
+ pOp->add_reg(1, 3, 5);
+ break;
+ case 3:
+ pOp->add_reg(3, 1, 5);
+ break;
+ case 4:
+ pOp->add_reg(3, 4, 5);
+ break;
+ case 5:
+ pOp->add_reg(1, 6, 5);
+ break;
+ case 6:
+ pOp->add_reg(6, 1, 5);
+ break;
+ case 7: // illegal register
+ pOp->add_reg(1, 8, 5);
+ ret_val = FAIL ;
+ break;
+ case 8: // another illegal register
+ pOp->add_reg(8, 1, 5);
+ ret_val = FAIL ;
+ break;
+ case 9: // and another one
+ pOp->add_reg(1, 2, 8);
+ ret_val = FAIL ;
+ break;
+ default:
+ break ;
+ }
+ pOp->load_const_u32(7, 65000);
+ pOp->branch_eq(5, 7, 0);
+ pOp->interpret_exit_nok();
+ pOp->def_label(0);
+ pOp->interpret_exit_ok();
+
+ return ret_val ;
+}
+
+TTYPE t_subReg(int nCalls, NdbOperation* pOp) {
+
+ TTYPE ret_val = NO_FAIL ;
+ ndbout << "Defining subReg test: " << nCalls << endl;
+
+ pOp->load_const_u32(1, 65500);
+ pOp->load_const_u32(2, 500);
+ pOp->load_const_u64(3, 65600);
+ pOp->load_const_u64(4, 95600);
+
+ switch(nCalls) {
+ case 1:
+ pOp->sub_reg(1, 2, 5);
+ pOp->load_const_u32(7, 65000);
+ break;
+ case 2:
+ pOp->sub_reg(1, 3, 5);
+ pOp->load_const_u64(7, (Uint64)-100);
+ break;
+ case 3:
+ pOp->sub_reg(3, 1, 5);
+ pOp->load_const_u64(7, (Uint64)100);
+ break;
+ case 4:
+ pOp->sub_reg(3, 4, 5);
+ pOp->load_const_u64(7, (Uint64)-30000);
+ break;
+ case 5:
+ pOp->sub_reg(1, 6, 5);
+ pOp->load_const_u64(7, 0);
+ ret_val = FAIL ;
+ break;
+ case 6:
+ pOp->sub_reg(6, 1, 5);
+ pOp->load_const_u64(7, 0);
+ ret_val = FAIL ;
+ break;
+ case 7:
+ pOp->sub_reg(1, 8, 5);
+ pOp->load_const_u64(7, 0);
+ ret_val = FAIL ;
+ break;
+ case 8:
+ pOp->sub_reg(8, 1, 5);
+ pOp->load_const_u64(7, 0);
+ ret_val = FAIL ;
+ break;
+ case 9:
+ pOp->sub_reg(1, 2, 8);
+ pOp->load_const_u32(7, (Uint32)65000);
+ ret_val = FAIL;
+ break;
+ default:
+ //ndbout << "t_subReg: default case (no test case)" << endl ;
+ //return ret_val = NO_FAIL ;
+ break ;
+ }
+ pOp->branch_eq(5, 7, 0);
+ pOp->interpret_exit_nok() ;
+ pOp->def_label(0);
+ pOp->interpret_exit_ok() ;
+
+ return ret_val;
+}
+
+TTYPE t_subroutineWithBranchLabel(int nCalls, NdbOperation* pOp) {
+
+ TTYPE ret_val = NO_FAIL ;
+ ndbout << "Defining subroutineWithBranchLabel test:" << nCalls << endl;
+
+ pOp->load_const_u32(1, 65500);
+ pOp->load_const_u32(2, 500);
+ pOp->load_const_u64(3, 65600);
+ pOp->load_const_u64(4, 95600);
+ pOp->load_const_u32(7, 65000);
+ pOp->call_sub(0) ;
+
+ switch(nCalls) {
+ case 1:
+ pOp->def_subroutine(0) ;
+ pOp->add_reg(1, 2, 5);
+ break;
+ case 2:
+ pOp->def_subroutine(0) ;
+ pOp->add_reg(1, 3, 5);
+ break;
+ case 3:
+ pOp->def_subroutine(0) ;
+ pOp->add_reg(3, 1, 5);
+ break;
+ case 4:
+ pOp->def_subroutine(0) ;
+ pOp->add_reg(3, 4, 5);
+ break;
+ case 5:
+ pOp->def_subroutine(0) ;
+ pOp->add_reg(1, 6, 5);
+ break;
+ case 6:
+ pOp->def_subroutine(0) ;
+ pOp->add_reg(6, 1, 5);
+ break;
+ case 7: // illegal register
+ pOp->def_subroutine(0) ;
+ pOp->add_reg(1, 8, 5);
+ ret_val = FAIL ;
+ break;
+ case 8: // another illegal register
+ pOp->def_subroutine(0) ;
+ pOp->add_reg(8, 1, 5);
+ ret_val = FAIL ;
+ break;
+ case 9: // and another one
+ pOp->def_subroutine(0) ;
+ pOp->add_reg(1, 2, 8);
+ ret_val = FAIL ;
+ break;
+ case 10: // test subroutine nesting
+ for(int sub = 0; sub < 25 ; ++sub){
+ pOp->call_sub(sub) ;
+ pOp->def_subroutine(sub + 1) ;
+ pOp->interpret_exit_ok() ;
+ pOp->ret_sub() ;
+ }
+ ret_val = FAIL ;
+ default:
+ break ;
+ }
+
+ pOp->branch_label(0) ;
+ pOp->interpret_exit_nok() ;
+ pOp->def_label(0) ;
+ pOp->interpret_exit_ok() ;
+ pOp->ret_sub() ;
+
+ return ret_val ;
+}
+
+
+void scan_rows(Ndb* pMyNdb, int opType, int tupleType, int scanType) {
+
+ int check = -1 ;
+ int loop_count_ops = nRecords ;
+ int eOf = -1 ;
+ int readValue = 0 ;
+ int readValue2 = 0 ;
+ int scanCount = 0 ;
+ NdbRecAttr* tTmp = NULL ;
+ TTYPE fail = NO_FAIL ;
+
+ for (int count=0 ; count < loop_count_ops ; count++) {
+ NdbConnection* MyTransaction = pMyNdb->startTransaction();
+ if (!MyTransaction) error_handler(pMyNdb->getNdbError(), NO_FAIL);
+
+ NdbOperation* MyOperation = MyTransaction->getNdbOperation(tableName[0]);
+ if (!MyOperation) error_handler(pMyNdb->getNdbError(), NO_FAIL);
+
+ if (opType == 1)
+ // Open for scan read, Creates the SCAN_TABREQ and if needed
+ // SCAN_TABINFO signals.
+ check = MyOperation->openScanRead(1);
+ else if (opType == 2)
+ // Open for scan with update of rows.
+ check = MyOperation->openScanExclusive(1);
+
+ // Create ATTRINFO signal(s) for interpreted program used for
+ // defining search criteria and column values that should be returned.
+
+ scanCount = count+1 ;
+
+ switch(tupleType) {
+ case 1:
+ fail = t_exitMethods(scanCount, MyOperation, opType);
+ break;
+ case 2:
+ fail = t_incValue(scanCount, MyOperation);
+ break;
+ case 3:
+ fail = t_subValue(scanCount, MyOperation);
+ break;
+ case 4:
+ fail = t_readAttr(scanCount, MyOperation);
+ break;
+ case 5:
+ fail = t_writeAttr(scanCount, MyOperation);
+ break;
+ case 6:
+ fail = t_loadConst(scanCount, MyOperation, opType);
+ break;
+ case 7:
+ fail = t_branch(scanCount, MyOperation);
+ break;
+ case 8:
+ fail = t_branchIfNull(scanCount, MyOperation);
+ break;
+ case 9:
+ fail = t_addReg(scanCount, MyOperation);
+ break;
+ case 10:
+ fail = t_subReg(scanCount, MyOperation);
+ break;
+ case 11:
+ fail = t_subroutineWithBranchLabel(scanCount, MyOperation);
+ break;
+ default:
+ break ;
+ }
+
+ if(11 != tupleType) MyOperation->getValue(attrName[1], (char*)&readValue);
+
+ // Sends the SCAN_TABREQ, (SCAN_TABINFO) and ATTRINFO signals and then
+ // reads the answer in TRANSID_AI. Confirmation is received through SCAN_TABCONF or
+ // SCAN_TABREF if failure.
+ check = MyTransaction->executeScan();
+ if (check == -1) {
+ //ndbout << endl << "executeScan returned: " << MyTransaction->getNdbError() << endl;
+ error_handler(MyTransaction->getNdbError(), fail) ;
+ pMyNdb->closeTransaction(MyTransaction);
+ }else{
+ // Sends the SCAN_NEXTREQ signal(s) and reads the answer in TRANS_ID signals.
+ // SCAN_TABCONF or SCAN_TABREF is the confirmation.
+ while (eOf = MyTransaction->nextScanResult() == 0) {
+ ndbout << readValue <<"; ";
+ // Here we call takeOverScanOp for update of the tuple.
+ }
+ ndbout << endl ;
+
+ pMyNdb->closeTransaction(MyTransaction);
+ if (eOf == -1) {
+ ndbout << endl << "nextScanResult returned: "<< MyTransaction->getNdbError() << endl;
+ } else {
+ ndbout << "OK" << endl;
+ }
+ }
+ }
+ return;
+
+};
+
+
+void update_rows(Ndb* pMyNdb, int tupleType, int opType) {
+ /****************************************************************
+ * Update rows in SimpleTable
+ * Only updates the first 3 cols.
+ ***************************************************************/
+
+ int check = -1 ;
+ int loop_count_ops = nRecords ;
+ int readValue[MAXATTR] = {0} ;
+ TTYPE ret_val = NO_FAIL ;
+ NdbConnection *MyTransaction = NULL ;
+ NdbOperation *MyOperation = NULL ;
+
+ ndbout << "Updating records ..." << endl << endl;
+
+ for (int count=0 ; count < loop_count_ops ; count++){
+
+ MyTransaction = pMyNdb->startTransaction();
+ if (!MyTransaction) {
+ error_handler(pMyNdb->getNdbError(), NO_FAIL);
+ return;
+ }//if
+
+ MyOperation = MyTransaction->getNdbOperation(tableName[0]);
+ if (MyOperation == NULL) {
+ error_handler(pMyNdb->getNdbError(), NO_FAIL);
+ return;
+ }//if
+
+ // if (operationType == 3)
+ check = MyOperation->interpretedUpdateTuple();
+ // else if (operationType == 4)
+ // check = MyOperation->interpretedDirtyUpdate();
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbError(), NO_FAIL);
+
+ check = MyOperation->equal( attrName[0], (char*)&pkValue[count] );
+ if( check == -1 )
+ error_handler(MyTransaction->getNdbError(), NO_FAIL);
+
+ switch(tupleType) {
+ case 1:
+ ret_val = t_exitMethods(count+1, MyOperation, opType);
+ break;
+ case 2:
+ ret_val = t_incValue(count+1, MyOperation);
+ break;
+ case 3:
+ ret_val = t_subValue(count+1, MyOperation);
+ break;
+ case 4:
+ ret_val = t_readAttr(count+1, MyOperation);
+ break;
+ case 5:
+ ret_val = t_writeAttr(count+1, MyOperation);
+ break;
+ case 6:
+ ret_val = t_loadConst(count+1, MyOperation, opType);
+ break;
+ case 7:
+ ret_val = t_branch(count+1, MyOperation);
+ break;
+ case 8:
+ ret_val = t_branchIfNull(count+1, MyOperation);
+ break;
+ case 9:
+ ret_val = t_addReg(count+1, MyOperation);
+ break;
+ case 10:
+ ret_val = t_subReg(count+1, MyOperation);
+ break;
+ case 11:
+ ret_val = t_subroutineWithBranchLabel(count+1, MyOperation);
+ break;
+ default:
+ break ;
+ }
+
+ MyOperation->getValue("COL1", (char*)&readValue);
+
+ if (MyTransaction->execute( Commit ) == -1 ) {
+ error_handler(MyTransaction->getNdbError(), ret_val);
+ }else if (NO_FAIL == ret_val ) {
+ ndbout << "OK" << endl;
+ } else {
+ bTestPassed = -1;
+ ndbout << "Test passed when expected to fail" << endl;
+ }//if
+ pMyNdb->closeTransaction(MyTransaction);
+
+ }
+
+ ndbout << "Finished updating records" << endl;
+ return;
+};
+
+void delete_rows(Ndb* pMyNdb, int tupleTest, int opType) {
+
+ /****************************************************************
+ * Delete rows from SimpleTable
+ *
+ ***************************************************************/
+
+ int check = 1 ;
+ int loop_count_ops = nRecords ;
+ int readValue[MAXATTR] = {0};
+ NdbConnection *MyTransaction = NULL ;
+ NdbOperation *MyOperation = NULL ;
+ TTYPE ret_val = NO_FAIL ;
+
+ ndbout << "Deleting records ..."<< endl << endl;
+
+ for (int count=0 ; count < loop_count_ops ; count++) {
+
+ MyTransaction = pMyNdb->startTransaction();
+ if (!MyTransaction) error_handler(pMyNdb->getNdbError(), NO_FAIL) ;
+
+ MyOperation = MyTransaction->getNdbOperation(tableName[0]);
+ if (!MyOperation) error_handler(pMyNdb->getNdbError(), NO_FAIL) ;
+
+ check = MyOperation->interpretedDeleteTuple();
+ if( check == -1 ) error_handler(MyTransaction->getNdbError(), NO_FAIL) ;
+
+ check = MyOperation->equal( attrName[0], (char*)&pkValue[count] );
+ if( check == -1 ) error_handler(MyTransaction->getNdbError(), NO_FAIL) ;
+
+ switch(tupleTest) {
+ case 1:
+ ret_val = t_exitMethods(count+1, MyOperation, opType);
+ break;
+ case 2:
+ ret_val = t_incValue(count+1, MyOperation);
+ break;
+ case 3:
+ ret_val = t_subValue(count+1, MyOperation);
+ break;
+ case 4:
+ ret_val = t_readAttr(count+1, MyOperation);
+ break;
+ case 5:
+ ret_val = t_writeAttr(count+1, MyOperation);
+ break;
+ case 6:
+ ret_val = t_loadConst(count+1, MyOperation, opType);
+ break;
+ case 7:
+ ret_val = t_branch(count+1, MyOperation);
+ break;
+ case 8:
+ ret_val = t_branchIfNull(count+1, MyOperation);
+ break;
+ case 9:
+ ret_val = t_addReg(count+1, MyOperation);
+ break;
+ case 10:
+ ret_val = t_subReg(count+1, MyOperation);
+ break;
+ case 11:
+ ret_val = t_subroutineWithBranchLabel(count+1, MyOperation);
+ break;
+ default:
+ break ;
+ }
+
+ if(11 != tupleTest)MyOperation->getValue(attrName[1], (char*)&readValue) ;
+
+ if (MyTransaction->execute( Commit ) == -1 ) {
+ error_handler(MyTransaction->getNdbError(), ret_val);
+ } else if (NO_FAIL == ret_val /*|| UNDEF == ret_val*/ ) {
+ ndbout << "OK" << endl;
+ } else {
+ bTestPassed = -1;
+ ndbout << "Test passed when expected to fail" << endl;
+ }//if
+ ndbout << endl;
+ pMyNdb->closeTransaction(MyTransaction);
+ }
+
+ ndbout << "Finished deleting records" << endl;
+ return;
+
+};
+
+
+inline void setAttrNames(){
+ for (int i = 0; i < MAXATTR; i++){
+ snprintf(attrName[i], MAXSTRLEN, "COL%d", i);
+ }
+}
+
+
+inline void setTableNames(){
+ for (int i = 0; i < MAXTABLES; i++){
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d", i);
+ }
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/Makefile b/ndb/test/ndbapi/lmc-bench/Makefile
new file mode 100644
index 00000000000..af472b1589f
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/Makefile
@@ -0,0 +1,6 @@
+include .defs.mk
+
+DIRS := src async-src script
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/Makefile b/ndb/test/ndbapi/lmc-bench/async-src/Makefile
new file mode 100644
index 00000000000..744d6171139
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+DIRS = \
+ user \
+ generator
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/generator/Makefile b/ndb/test/ndbapi/lmc-bench/async-src/generator/Makefile
new file mode 100644
index 00000000000..c1f84a3ef70
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/generator/Makefile
@@ -0,0 +1,13 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+SOURCES = mainAsyncGenerator.cpp asyncGenerator.cpp
+
+CCFLAGS_LOC := -I../include -I../../include
+
+BIN_TARGET := DbAsyncGenerator
+BIN_TARGET_ARCHIVES := lmc_AsyncUser
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/generator/asyncGenerator.cpp b/ndb/test/ndbapi/lmc-bench/async-src/generator/asyncGenerator.cpp
new file mode 100644
index 00000000000..25eb1830de9
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/generator/asyncGenerator.cpp
@@ -0,0 +1,572 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <stdio.h>
+#include <string.h>
+
+#include "dbGenerator.h"
+#include <NdbApi.hpp>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+static void getRandomSubscriberNumber(SubscriberNumber number);
+static void getRandomServerId(ServerId *serverId);
+static void getRandomChangedBy(ChangedBy changedBy);
+static void getRandomChangedTime(ChangedTime changedTime);
+
+static void clearTransaction(TransactionDefinition *trans);
+static void initGeneratorStatistics(GeneratorStatistics *gen);
+
+static void doOneTransaction(ThreadData * td,
+ int parallellism,
+ int millisSendPoll,
+ int minEventSendPoll,
+ int forceSendPoll);
+static void doTransaction_T1(Ndb * pNDB, ThreadData * td, int async);
+static void doTransaction_T2(Ndb * pNDB, ThreadData * td, int async);
+static void doTransaction_T3(Ndb * pNDB, ThreadData * td, int async);
+static void doTransaction_T4(Ndb * pNDB, ThreadData * td, int async);
+static void doTransaction_T5(Ndb * pNDB, ThreadData * td, int async);
+
+/***************************************************************
+* L O C A L D A T A *
+***************************************************************/
+
+static SequenceValues transactionDefinition[] = {
+ {25, 1},
+ {25, 2},
+ {20, 3},
+ {15, 4},
+ {15, 5},
+ {0, 0}
+};
+
+static SequenceValues rollbackDefinition[] = {
+ {98, 0},
+ {2 , 1},
+ {0, 0}
+};
+
+static int maxsize = 0;
+
+/***************************************************************
+* P U B L I C D A T A *
+***************************************************************/
+
+/***************************************************************
+****************************************************************
+* L O C A L F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+static void getRandomSubscriberNumber(SubscriberNumber number)
+{
+ uint32 tmp;
+ char sbuf[SUBSCRIBER_NUMBER_LENGTH + 1];
+ tmp = myRandom48(NO_OF_SUBSCRIBERS);
+ sprintf(sbuf, "%.*d", SUBSCRIBER_NUMBER_LENGTH, tmp);
+ memcpy(number, sbuf, SUBSCRIBER_NUMBER_LENGTH);
+}
+
+static void getRandomServerId(ServerId *serverId)
+{
+ *serverId = myRandom48(NO_OF_SERVERS);
+}
+
+static void getRandomChangedBy(ChangedBy changedBy)
+{
+ memset(changedBy, myRandom48(26)+'A', CHANGED_BY_LENGTH);
+ changedBy[CHANGED_BY_LENGTH] = 0;
+}
+
+static void getRandomChangedTime(ChangedTime changedTime)
+{
+ memset(changedTime, myRandom48(26)+'A', CHANGED_TIME_LENGTH);
+ changedTime[CHANGED_TIME_LENGTH] = 0;
+}
+
+static void clearTransaction(TransactionDefinition *trans)
+{
+ trans->count = 0;
+ trans->branchExecuted = 0;
+ trans->rollbackExecuted = 0;
+ trans->latencyCounter = myRandom48(127);
+ trans->latency.reset();
+}
+
+static int listFull(SessionList *list)
+{
+ return(list->numberInList == SESSION_LIST_LENGTH);
+}
+
+static int listEmpty(SessionList *list)
+{
+ return(list->numberInList == 0);
+}
+
+static void insertSession(SessionList *list,
+ SubscriberNumber number,
+ ServerId serverId)
+{
+ SessionElement *e;
+ if( listFull(list) ) return;
+
+ e = &list->list[list->writeIndex];
+
+ strcpy(e->subscriberNumber, number);
+ e->serverId = serverId;
+
+ list->writeIndex = (list->writeIndex + 1) % SESSION_LIST_LENGTH;
+ list->numberInList++;
+
+ if( list->numberInList > maxsize )
+ maxsize = list->numberInList;
+}
+
+static SessionElement *getNextSession(SessionList *list)
+{
+ if( listEmpty(list) ) return(0);
+
+ return(&list->list[list->readIndex]);
+}
+
+static void deleteSession(SessionList *list)
+{
+ if( listEmpty(list) ) return;
+
+ list->readIndex = (list->readIndex + 1) % SESSION_LIST_LENGTH;
+ list->numberInList--;
+}
+
+static void initGeneratorStatistics(GeneratorStatistics *gen)
+{
+ int i;
+
+ if( initSequence(&gen->transactionSequence,
+ transactionDefinition) != 0 ) {
+ ndbout_c("could not set the transaction types");
+ exit(0);
+ }
+
+ if( initSequence(&gen->rollbackSequenceT4,
+ rollbackDefinition) != 0 ) {
+ ndbout_c("could not set the rollback sequence");
+ exit(0);
+ }
+
+ if( initSequence(&gen->rollbackSequenceT5,
+ rollbackDefinition) != 0 ) {
+ ndbout_c("could not set the rollback sequence");
+ exit(0);
+ }
+
+ for(i = 0; i < NUM_TRANSACTION_TYPES; i++ )
+ clearTransaction(&gen->transactions[i]);
+
+ gen->totalTransactions = 0;
+
+ gen->activeSessions.numberInList = 0;
+ gen->activeSessions.readIndex = 0;
+ gen->activeSessions.writeIndex = 0;
+}
+
+
+static
+void
+doOneTransaction(ThreadData * td, int p, int millis, int minEvents, int force)
+{
+ int i;
+ unsigned int transactionType;
+ int async = 1;
+ if (p == 1) {
+ async = 0;
+ }//if
+ for(i = 0; i<p; i++){
+ if(td[i].runState == Runnable){
+ transactionType = getNextRandom(&td[i].generator.transactionSequence);
+
+ switch(transactionType) {
+ case 1:
+ doTransaction_T1(td[i].pNDB, &td[i], async);
+ break;
+ case 2:
+ doTransaction_T2(td[i].pNDB, &td[i], async);
+ break;
+ case 3:
+ doTransaction_T3(td[i].pNDB, &td[i], async);
+ break;
+ case 4:
+ doTransaction_T4(td[i].pNDB, &td[i], async);
+ break;
+ case 5:
+ doTransaction_T5(td[i].pNDB, &td[i], async);
+ break;
+ default:
+ ndbout_c("Unknown transaction type: %d", transactionType);
+ }
+ }
+ }
+ if (async == 1) {
+ td[0].pNDB->sendPollNdb(millis, minEvents, force);
+ }//if
+}
+
+static
+void
+doTransaction_T1(Ndb * pNDB, ThreadData * td, int async)
+{
+ /*----------------*/
+ /* Init arguments */
+ /*----------------*/
+ getRandomSubscriberNumber(td->transactionData.number);
+ getRandomChangedBy(td->transactionData.changed_by);
+ snprintf(td->transactionData.changed_time,
+ sizeof(td->transactionData.changed_time),
+ "%ld - %d", td->changedTime++, myRandom48(65536*1024));
+ //getRandomChangedTime(td->transactionData.changed_time);
+ td->transactionData.location = td->transactionData.changed_by[0];
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ td->runState = Running;
+ td->generator.transactions[0].startLatency();
+
+ start_T1(pNDB, td, async);
+}
+
+static
+void
+doTransaction_T2(Ndb * pNDB, ThreadData * td, int async)
+{
+ /*----------------*/
+ /* Init arguments */
+ /*----------------*/
+ getRandomSubscriberNumber(td->transactionData.number);
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ td->runState = Running;
+ td->generator.transactions[1].startLatency();
+
+ start_T2(pNDB, td, async);
+}
+
+static
+void
+doTransaction_T3(Ndb * pNDB, ThreadData * td, int async)
+{
+ SessionElement *se;
+
+ /*----------------*/
+ /* Init arguments */
+ /*----------------*/
+ se = getNextSession(&td->generator.activeSessions);
+ if( se ) {
+ strcpy(td->transactionData.number, se->subscriberNumber);
+ td->transactionData.server_id = se->serverId;
+ td->transactionData.sessionElement = 1;
+ } else {
+ getRandomSubscriberNumber(td->transactionData.number);
+ getRandomServerId(&td->transactionData.server_id);
+ td->transactionData.sessionElement = 0;
+ }
+
+ td->transactionData.server_bit = (1 << td->transactionData.server_id);
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ td->runState = Running;
+ td->generator.transactions[2].startLatency();
+ start_T3(pNDB, td, async);
+}
+
+static
+void
+doTransaction_T4(Ndb * pNDB, ThreadData * td, int async)
+{
+ /*----------------*/
+ /* Init arguments */
+ /*----------------*/
+ getRandomSubscriberNumber(td->transactionData.number);
+ getRandomServerId(&td->transactionData.server_id);
+
+ td->transactionData.server_bit = (1 << td->transactionData.server_id);
+ td->transactionData.do_rollback =
+ getNextRandom(&td->generator.rollbackSequenceT4);
+
+#if 0
+ memset(td->transactionData.session_details,
+ myRandom48(26)+'A', SESSION_DETAILS_LENGTH);
+#endif
+ td->transactionData.session_details[SESSION_DETAILS_LENGTH] = 0;
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ td->runState = Running;
+ td->generator.transactions[3].startLatency();
+ start_T4(pNDB, td, async);
+}
+
+static
+void
+doTransaction_T5(Ndb * pNDB, ThreadData * td, int async)
+{
+ SessionElement * se;
+ se = getNextSession(&td->generator.activeSessions);
+ if( se ) {
+ strcpy(td->transactionData.number, se->subscriberNumber);
+ td->transactionData.server_id = se->serverId;
+ td->transactionData.sessionElement = 1;
+ }
+ else {
+ getRandomSubscriberNumber(td->transactionData.number);
+ getRandomServerId(&td->transactionData.server_id);
+ td->transactionData.sessionElement = 0;
+ }
+
+ td->transactionData.server_bit = (1 << td->transactionData.server_id);
+ td->transactionData.do_rollback
+ = getNextRandom(&td->generator.rollbackSequenceT5);
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ td->runState = Running;
+ td->generator.transactions[4].startLatency();
+ start_T5(pNDB, td, async);
+}
+
+void
+complete_T1(ThreadData * data){
+ data->generator.transactions[0].stopLatency();
+ data->generator.transactions[0].count++;
+
+ data->runState = Runnable;
+ data->generator.totalTransactions++;
+}
+
+void
+complete_T2(ThreadData * data){
+ data->generator.transactions[1].stopLatency();
+ data->generator.transactions[1].count++;
+
+ data->runState = Runnable;
+ data->generator.totalTransactions++;
+}
+
+void
+complete_T3(ThreadData * data){
+
+ data->generator.transactions[2].stopLatency();
+ data->generator.transactions[2].count++;
+
+ if(data->transactionData.branchExecuted)
+ data->generator.transactions[2].branchExecuted++;
+
+ data->runState = Runnable;
+ data->generator.totalTransactions++;
+}
+
+void
+complete_T4(ThreadData * data){
+
+ data->generator.transactions[3].stopLatency();
+ data->generator.transactions[3].count++;
+
+ if(data->transactionData.branchExecuted)
+ data->generator.transactions[3].branchExecuted++;
+ if(data->transactionData.do_rollback)
+ data->generator.transactions[3].rollbackExecuted++;
+
+ if(data->transactionData.branchExecuted &&
+ !data->transactionData.do_rollback){
+ insertSession(&data->generator.activeSessions,
+ data->transactionData.number,
+ data->transactionData.server_id);
+ }
+
+ data->runState = Runnable;
+ data->generator.totalTransactions++;
+
+}
+void
+complete_T5(ThreadData * data){
+
+ data->generator.transactions[4].stopLatency();
+ data->generator.transactions[4].count++;
+
+ if(data->transactionData.branchExecuted)
+ data->generator.transactions[4].branchExecuted++;
+ if(data->transactionData.do_rollback)
+ data->generator.transactions[4].rollbackExecuted++;
+
+ if(data->transactionData.sessionElement &&
+ !data->transactionData.do_rollback){
+ deleteSession(&data->generator.activeSessions);
+ }
+
+ data->runState = Runnable;
+ data->generator.totalTransactions++;
+}
+
+/***************************************************************
+****************************************************************
+* P U B L I C F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+void
+asyncGenerator(ThreadData *data,
+ int parallellism,
+ int millisSendPoll,
+ int minEventSendPoll,
+ int forceSendPoll)
+{
+ ThreadData * startUp;
+
+ GeneratorStatistics *st;
+ double periodStop;
+ double benchTimeStart;
+ double benchTimeEnd;
+ int i, j, done;
+
+ myRandom48Init(data->randomSeed);
+
+ for(i = 0; i<parallellism; i++){
+ initGeneratorStatistics(&data[i].generator);
+ }
+
+ startUp = (ThreadData*)malloc(parallellism * sizeof(ThreadData));
+ memcpy(startUp, data, (parallellism * sizeof(ThreadData)));
+
+ /*----------------*/
+ /* warm up period */
+ /*----------------*/
+ periodStop = userGetTime() + (double)data[0].warmUpSeconds;
+
+ while(userGetTime() < periodStop){
+ doOneTransaction(startUp, parallellism,
+ millisSendPoll, minEventSendPoll, forceSendPoll);
+ }
+
+ ndbout_c("Waiting for startup to finish");
+
+ /**
+ * Wait for all transactions
+ */
+ done = 0;
+ while(!done){
+ done = 1;
+ for(i = 0; i<parallellism; i++){
+ if(startUp[i].runState != Runnable){
+ done = 0;
+ break;
+ }
+ }
+ if(!done){
+ startUp[0].pNDB->sendPollNdb();
+ }
+ }
+ ndbout_c("Benchmark period starts");
+
+ /*-------------------------*/
+ /* normal benchmark period */
+ /*-------------------------*/
+ benchTimeStart = userGetTime();
+
+ periodStop = benchTimeStart + (double)data[0].testSeconds;
+ while(userGetTime() < periodStop)
+ doOneTransaction(data, parallellism,
+ millisSendPoll, minEventSendPoll, forceSendPoll);
+
+ benchTimeEnd = userGetTime();
+
+ ndbout_c("Benchmark period done");
+
+ /**
+ * Wait for all transactions
+ */
+ done = 0;
+ while(!done){
+ done = 1;
+ for(i = 0; i<parallellism; i++){
+ if(data[i].runState != Runnable){
+ done = 0;
+ break;
+ }
+ }
+ if(!done){
+ data[0].pNDB->sendPollNdb();
+ }
+ }
+
+ /*------------------*/
+ /* cool down period */
+ /*------------------*/
+ periodStop = userGetTime() + (double)data[0].coolDownSeconds;
+ while(userGetTime() < periodStop){
+ doOneTransaction(startUp, parallellism,
+ millisSendPoll, minEventSendPoll, forceSendPoll);
+ }
+
+ done = 0;
+ while(!done){
+ done = 1;
+ for(i = 0; i<parallellism; i++){
+ if(startUp[i].runState != Runnable){
+ done = 0;
+ break;
+ }
+ }
+ if(!done){
+ startUp[0].pNDB->sendPollNdb();
+ }
+ }
+
+
+ /*---------------------------------------------------------*/
+ /* add the times for all transaction for inner loop timing */
+ /*---------------------------------------------------------*/
+ for(j = 0; j<parallellism; j++){
+ st = &data[j].generator;
+
+ st->outerLoopTime = benchTimeEnd - benchTimeStart;
+ st->outerTps = getTps(st->totalTransactions, st->outerLoopTime);
+ }
+ /* ndbout_c("maxsize = %d\n",maxsize); */
+
+ free(startUp);
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/generator/mainAsyncGenerator.cpp b/ndb/test/ndbapi/lmc-bench/async-src/generator/mainAsyncGenerator.cpp
new file mode 100644
index 00000000000..d7506c9dd2c
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/generator/mainAsyncGenerator.cpp
@@ -0,0 +1,395 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <NdbUnistd.h>
+
+#include <NdbHost.h>
+#include <NdbSleep.h>
+#include <NdbThread.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbEnv.h>
+#include <NdbTest.hpp>
+
+#include "userInterface.h"
+#include "dbGenerator.h"
+
+static int numProcesses;
+static int numSeconds;
+static int numWarmSeconds;
+static int parallellism;
+static int millisSendPoll;
+static int minEventSendPoll;
+static int forceSendPoll;
+
+static ThreadData *data;
+
+static void usage(const char *prog)
+{
+ const char *progname;
+
+ /*--------------------------------------------*/
+ /* Get the name of the program (without path) */
+ /*--------------------------------------------*/
+ progname = strrchr(prog, '/');
+
+ if (progname == 0)
+ progname = prog;
+ else
+ ++progname;
+
+ ndbout_c(
+ "Usage: %s [-proc <num>] [-warm <num>] [-time <num>] [ -p <num>] "
+ "[-t <num> ] [ -e <num> ] [ -f <num>] \n"
+ " -proc <num> Specifies that <num> is the number of\n"
+ " threads. The default is 1.\n"
+ " -time <num> Specifies that the test will run for <num> sec.\n"
+ " The default is 10 sec\n"
+ " -warm <num> Specifies the warm-up/cooldown period of <num> "
+ "sec.\n"
+ " The default is 10 sec\n"
+ " -p <num> The no of parallell transactions started by "
+ "one thread\n"
+ " -e <num> Minimum no of events before wake up in call to "
+ "sendPoll\n"
+ " Default is 1\n"
+ " -f <num> force parameter to sendPoll\n"
+ " Default is 0\n",
+ progname);
+}
+
+static
+int
+parse_args(int argc, const char **argv)
+{
+ int i;
+
+ numProcesses = 1;
+ numSeconds = 10;
+ numWarmSeconds = 10;
+ parallellism = 1;
+ millisSendPoll = 10000;
+ minEventSendPoll = 1;
+ forceSendPoll = 0;
+
+
+ i = 1;
+ while (i < argc){
+ if (strcmp("-proc",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ return 1;
+ }
+ if (sscanf(argv[i+1], "%d", &numProcesses) == -1 ||
+ numProcesses <= 0 || numProcesses > 127) {
+ ndbout_c("-proc flag requires a positive integer argument [1..127]");
+ return 1;
+ }
+ i += 2;
+ } else if (strcmp("-p", argv[i]) == 0){
+ if(i + 1 >= argc){
+ usage(argv[0]);
+ return 1;
+ }
+ if (sscanf(argv[i+1], "%d", &parallellism) == -1 ||
+ parallellism <= 0){
+ ndbout_c("-p flag requires a positive integer argument");
+ return 1;
+ }
+ i += 2;
+ }
+ else if (strcmp("-time",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ return 1;
+ }
+ if (sscanf(argv[i+1], "%d", &numSeconds) == -1 ||
+ numSeconds < 0) {
+ ndbout_c("-time flag requires a positive integer argument");
+ return 1;
+ }
+ i += 2;
+ }
+ else if (strcmp("-warm",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ return 1;
+ }
+ if (sscanf(argv[i+1], "%d", &numWarmSeconds) == -1 ||
+ numWarmSeconds < 0) {
+ ndbout_c("-warm flag requires a positive integer argument");
+ return 1;
+ }
+ i += 2;
+ }
+ else if (strcmp("-e",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ return 1;
+ }
+ if (sscanf(argv[i+1], "%d", &minEventSendPoll) == -1 ||
+ minEventSendPoll < 0) {
+ ndbout_c("-e flag requires a positive integer argument");
+ return 1;
+ }
+ i += 2;
+ }
+ else if (strcmp("-f",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ usage(argv[0]);
+ return 1;
+ }
+ if (sscanf(argv[i+1], "%d", &forceSendPoll) == -1 ||
+ forceSendPoll < 0) {
+ ndbout_c("-f flag requires a positive integer argument");
+ return 1;
+ }
+ i += 2;
+ }
+ else {
+ return 1;
+ }
+ }
+
+ if(minEventSendPoll > parallellism){
+ ndbout_c("minEventSendPoll(%d) > parallellism(%d)",
+ minEventSendPoll, parallellism);
+ ndbout_c("not very good...");
+ ndbout_c("very bad...");
+ ndbout_c("exiting...");
+ return 1;
+ }
+ return 0;
+}
+
+static
+void
+print_transaction(const char *header,
+ unsigned long totalCount,
+ TransactionDefinition *trans,
+ unsigned int printBranch,
+ unsigned int printRollback)
+{
+ double f;
+
+ ndbout_c(" %s: %d (%.2f%%) "
+ "Latency(ms) avg: %d min: %d max: %d std: %d n: %d",
+ header,
+ trans->count,
+ (double)trans->count / (double)totalCount * 100.0,
+ (int)trans->latency.getMean(),
+ (int)trans->latency.getMin(),
+ (int)trans->latency.getMax(),
+ (int)trans->latency.getStddev(),
+ (int)trans->latency.getCount()
+ );
+
+ if( printBranch ){
+ if( trans->count == 0 )
+ f = 0.0;
+ else
+ f = (double)trans->branchExecuted / (double)trans->count * 100.0;
+ ndbout_c(" Branches Executed: %d (%.2f%%)", trans->branchExecuted, f);
+ }
+
+ if( printRollback ){
+ if( trans->count == 0 )
+ f = 0.0;
+ else
+ f = (double)trans->rollbackExecuted / (double)trans->count * 100.0;
+ ndbout_c(" Rollback Executed: %d (%.2f%%)",trans->rollbackExecuted,f);
+ }
+}
+
+void
+print_stats(const char *title,
+ unsigned int length,
+ unsigned int transactionFlag,
+ GeneratorStatistics *gen,
+ int numProc, int parallellism)
+{
+ int i;
+ char buf[10];
+ char name[MAXHOSTNAMELEN];
+
+ name[0] = 0;
+ NdbHost_GetHostName(name);
+
+ ndbout_c("\n------ %s ------",title);
+ ndbout_c("Length : %d %s",
+ length,
+ transactionFlag ? "Transactions" : "sec");
+ ndbout_c("Processor : %s", name);
+ ndbout_c("Number of Proc: %d",numProc);
+ ndbout_c("Parallellism : %d", parallellism);
+ ndbout_c("\n");
+
+ if( gen->totalTransactions == 0 ) {
+ ndbout_c(" No Transactions for this test");
+ }
+ else {
+ for(i = 0; i < 5; i++) {
+ sprintf(buf, "T%d",i+1);
+ print_transaction(buf,
+ gen->totalTransactions,
+ &gen->transactions[i],
+ i >= 2,
+ i >= 3 );
+ }
+
+ ndbout_c("\n");
+ ndbout_c(" Overall Statistics:");
+ ndbout_c(" Transactions: %d", gen->totalTransactions);
+ ndbout_c(" Outer : %.0f TPS",gen->outerTps);
+ ndbout_c("\n");
+ }
+}
+
+static
+void *
+threadRoutine(void *arg)
+{
+ int i;
+ ThreadData *data = (ThreadData *)arg;
+ Ndb * pNDB;
+
+ pNDB = asyncDbConnect(parallellism);
+ /* NdbSleep_MilliSleep(rand() % 10); */
+
+ for(i = 0; i<parallellism; i++){
+ data[i].pNDB = pNDB;
+ }
+ millisSendPoll = 30000;
+ asyncGenerator(data, parallellism,
+ millisSendPoll, minEventSendPoll, forceSendPoll);
+
+ asyncDbDisconnect(pNDB);
+
+ NdbThread_Exit(0);
+
+ return NULL;
+}
+
+NDB_COMMAND(DbAsyncGenerator, "DbAsyncGenerator",
+ "DbAsyncGenerator", "DbAsyncGenerator", 65535)
+{
+ int i;
+ int j;
+ int k;
+ struct NdbThread* pThread = NULL;
+ GeneratorStatistics stats;
+ GeneratorStatistics *p;
+ char threadName[32];
+ int rc = NDBT_OK;
+ void* tmp = NULL;
+ if(parse_args(argc,argv) != 0){
+ usage(argv[0]);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+
+ ndbout_c("\nStarting Test with %d process(es) for %d %s parallellism %d",
+ numProcesses,
+ numSeconds,
+ "sec",
+ parallellism);
+
+ ndbout_c(" WarmUp/coolDown = %d sec", numWarmSeconds);
+
+ data = (ThreadData*)malloc((numProcesses*parallellism)*sizeof(ThreadData));
+
+ for(i = 0; i < numProcesses; i++) {
+ for(j = 0; j<parallellism; j++){
+ data[i*parallellism+j].warmUpSeconds = numWarmSeconds;
+ data[i*parallellism+j].testSeconds = numSeconds;
+ data[i*parallellism+j].coolDownSeconds = numWarmSeconds;
+ data[i*parallellism+j].randomSeed =
+ NdbTick_CurrentMillisecond()+i+j;
+ data[i*parallellism+j].changedTime = 0;
+ data[i*parallellism+j].runState = Runnable;
+ }
+ sprintf(threadName, "AsyncThread[%d]", i);
+ pThread = NdbThread_Create(threadRoutine,
+ (void**)&data[i*parallellism],
+ 65535,
+ threadName,
+ NDB_THREAD_PRIO_LOW);
+ if(pThread != 0 && pThread != NULL){
+ (&data[i*parallellism])->pThread = pThread;
+ } else {
+ perror("Failed to create thread");
+ rc = NDBT_FAILED;
+ }
+ }
+
+ showTime();
+
+ /*--------------------------------*/
+ /* Wait for all processes to exit */
+ /*--------------------------------*/
+ for(i = 0; i < numProcesses; i++) {
+ NdbThread_WaitFor(data[i*parallellism].pThread, &tmp);
+ NdbThread_Destroy(&data[i*parallellism].pThread);
+ }
+
+ ndbout_c("All threads have finished");
+
+ /*-------------------------------------------*/
+ /* Clear all structures for total statistics */
+ /*-------------------------------------------*/
+ stats.totalTransactions = 0;
+ stats.outerTps = 0.0;
+
+ for(i = 0; i < NUM_TRANSACTION_TYPES; i++ ) {
+ stats.transactions[i].count = 0;
+ stats.transactions[i].branchExecuted = 0;
+ stats.transactions[i].rollbackExecuted = 0;
+ stats.transactions[i].latency.reset();
+ }
+
+ /*--------------------------------*/
+ /* Add the values for all Threads */
+ /*--------------------------------*/
+ for(i = 0; i < numProcesses; i++) {
+ for(k = 0; k<parallellism; k++){
+ p = &data[i*parallellism+k].generator;
+
+ stats.totalTransactions += p->totalTransactions;
+ stats.outerTps += p->outerTps;
+
+ for(j = 0; j < NUM_TRANSACTION_TYPES; j++ ) {
+ stats.transactions[j].count +=
+ p->transactions[j].count;
+ stats.transactions[j].branchExecuted +=
+ p->transactions[j].branchExecuted;
+ stats.transactions[j].rollbackExecuted +=
+ p->transactions[j].rollbackExecuted;
+ stats.transactions[j].latency +=
+ p->transactions[j].latency;
+ }
+ }
+ }
+
+ print_stats("Test Results",
+ numSeconds,
+ 0,
+ &stats,
+ numProcesses,
+ parallellism);
+
+ free(data);
+
+ NDBT_ProgramExit(rc);
+}
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/include/dbGenerator.h b/ndb/test/ndbapi/lmc-bench/async-src/include/dbGenerator.h
new file mode 100644
index 00000000000..2256498e151
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/include/dbGenerator.h
@@ -0,0 +1,63 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBGENERATOR_H
+#define DBGENERATOR_H
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include "testData.h"
+#include "userInterface.h"
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+extern void asyncGenerator(ThreadData *d, int parallellism,
+ int millisSendPoll,
+ int minEventSendPoll,
+ int forceSendPoll);
+
+#ifdef __cplusplus
+}
+#endif
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+
+
+#endif /* DBGENERATOR_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/include/testData.h b/ndb/test/ndbapi/lmc-bench/async-src/include/testData.h
new file mode 100644
index 00000000000..3db85e7342e
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/include/testData.h
@@ -0,0 +1,156 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TESTDATA_H
+#define TESTDATA_H
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+#include <NdbTick.h>
+#include <NdbThread.h>
+#include <NDBT_Stats.hpp>
+#include <random.h>
+#include "testDefinitions.h"
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+#define NUM_TRANSACTION_TYPES 5
+#define SESSION_LIST_LENGTH 1000
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+typedef struct {
+ SubscriberNumber subscriberNumber;
+ ServerId serverId;
+} SessionElement;
+
+typedef struct {
+ SessionElement list[SESSION_LIST_LENGTH];
+ unsigned int readIndex;
+ unsigned int writeIndex;
+ unsigned int numberInList;
+} SessionList;
+
+typedef struct {
+ unsigned int count;
+ unsigned int branchExecuted;
+ unsigned int rollbackExecuted;
+
+ /**
+ * Latency measures
+ */
+ NDB_TICKS startTime;
+ NDBT_Stats latency;
+ unsigned int latencyCounter;
+
+ inline void startLatency(){
+ if((latencyCounter & 127) == 127)
+ startTime = NdbTick_CurrentMillisecond();
+ }
+
+ inline void stopLatency(){
+ if((latencyCounter & 127) == 127){
+ const NDB_TICKS tmp = NdbTick_CurrentMillisecond() - startTime;
+ latency.addObservation(tmp);
+ }
+ latencyCounter++;
+ }
+} TransactionDefinition;
+
+typedef struct {
+ RandomSequence transactionSequence;
+ RandomSequence rollbackSequenceT4;
+ RandomSequence rollbackSequenceT5;
+
+ TransactionDefinition transactions[NUM_TRANSACTION_TYPES];
+
+ unsigned int totalTransactions;
+
+ double outerLoopTime;
+ double outerTps;
+
+ SessionList activeSessions;
+
+} GeneratorStatistics;
+
+typedef enum{
+ Runnable,
+ Running
+} RunState ;
+
+typedef struct {
+ SubscriberNumber number;
+ SubscriberSuffix suffix;
+ SubscriberName name;
+ Location location;
+ ChangedBy changed_by;
+ ChangedTime changed_time;
+ ServerId server_id;
+ ServerBit server_bit;
+ SessionDetails session_details;
+
+ GroupId group_id;
+ ActiveSessions sessions;
+ Permission permission;
+
+ unsigned int do_rollback;
+
+ unsigned int branchExecuted;
+ unsigned int sessionElement;
+} TransactionData ;
+
+typedef struct {
+ struct NdbThread* pThread;
+
+ unsigned long randomSeed;
+ unsigned long changedTime;
+
+ unsigned int warmUpSeconds;
+ unsigned int testSeconds;
+ unsigned int coolDownSeconds;
+
+ GeneratorStatistics generator;
+
+ /**
+ * For async execution
+ */
+ RunState runState;
+ double startTime;
+ TransactionData transactionData;
+ struct Ndb * pNDB;
+} ThreadData;
+
+/***************************************************************
+ * P U B L I C F U N C T I O N S *
+ ***************************************************************/
+
+/***************************************************************
+ * E X T E R N A L D A T A *
+ ***************************************************************/
+
+
+
+#endif /* TESTDATA_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/include/userInterface.h b/ndb/test/ndbapi/lmc-bench/async-src/include/userInterface.h
new file mode 100644
index 00000000000..94bd1e80ab3
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/include/userInterface.h
@@ -0,0 +1,79 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBINTERFACE_H
+#define DBINTERFACE_H
+
+/***************************************************************/
+/* I N C L U D E D F I L E S */
+/***************************************************************/
+
+#include "testDefinitions.h"
+#include "testData.h"
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+/*-----------------------*/
+/* Default Database Name */
+/*-----------------------*/
+#define DEFAULTDB "TestDbClient"
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+typedef struct Ndb Ndb;
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+ extern void showTime();
+ extern double userGetTime(void);
+ extern Ndb *asyncDbConnect(int parallellism);
+ extern void asyncDbDisconnect(Ndb* pNDB);
+
+ extern void start_T1(Ndb * uh, ThreadData * data, int async);
+ extern void start_T2(Ndb * uh, ThreadData * data, int async);
+ extern void start_T3(Ndb * uh, ThreadData * data, int async);
+ extern void start_T4(Ndb * uh, ThreadData * data, int async);
+ extern void start_T5(Ndb * uh, ThreadData * data, int async);
+
+ extern void complete_T1(ThreadData * data);
+ extern void complete_T2(ThreadData * data);
+ extern void complete_T3(ThreadData * data);
+ extern void complete_T4(ThreadData * data);
+ extern void complete_T5(ThreadData * data);
+
+#ifdef __cplusplus
+}
+#endif
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+#endif /* DBINTERFACE_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/user/Makefile b/ndb/test/ndbapi/lmc-bench/async-src/user/Makefile
new file mode 100644
index 00000000000..c0b532a8359
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/user/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+ARCHIVE_TARGET := lmc_AsyncUser
+
+SOURCES := userInterface.C ndb_async2.C
+
+CCFLAGS_LOC = -I../include -I../../include
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/user/macros.h b/ndb/test/ndbapi/lmc-bench/async-src/user/macros.h
new file mode 100644
index 00000000000..c049cdbad33
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/user/macros.h
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MACROS_H
+#define MACROS_H
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <NdbOut.hpp>
+
+#define ERROR(x) {ndbout_c((x));}
+#define ERROR1(x,y) {ndbout_c((x), (y));}
+#define ERROR2(x,y,z) {ndbout_c((x), (y), (z));}
+#define ERROR3(x,y,z,u) {ndbout_c((x), (y), (z), (u));}
+#define ERROR4(x,y,z,u,w) {ndbout_c((x), (y), (z), (u), (w));}
+
+#define INIT_RANDOM(x) srand48((x))
+#define UI_RANDOM(x) ((unsigned int)(lrand48()%(x)))
+
+#define ASSERT(cond, message) \
+ { if(!(cond)) { ERROR(message); exit(-1); }}
+
+#ifdef DEBUG_ON
+#define DEBUG(x) {ndbout_c((x));}
+#define DEBUG1(x,y) {ndbout_c((x), (y));}
+#define DEBUG2(x,y,z) {ndbout_c((x), (y), (z));}
+#define DEBUG3(x,y,z,u) {ndbout_c((x), (y), (z), (u));}
+#define DEBUG4(x,y,z,u,w) {ndbout_c((x), (y), (z), (u), (w));}
+#define DEBUG5(x,y,z,u,w, v) {ndbout_c((x), (y), (z), (u), (w), (v));}
+#else
+#define DEBUG(x)
+#define DEBUG1(x,y)
+#define DEBUG2(x,y,z)
+#define DEBUG3(x,y,z,u)
+#define DEBUG4(x,y,z,u,w)
+#define DEBUG5(x,y,z,u,w, v)
+#endif
+
+#endif
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/user/ndb_async1.cpp b/ndb/test/ndbapi/lmc-bench/async-src/user/ndb_async1.cpp
new file mode 100644
index 00000000000..2a84f6b2aca
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/user/ndb_async1.cpp
@@ -0,0 +1,647 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#define DEBUG_ON
+
+#include "userInterface.h"
+
+#include "macros.h"
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+
+#include <NdbApi.hpp>
+
+inline
+NdbConnection *
+startTransaction(Ndb * pNDB,
+ ServerId inServerId,
+ const SubscriberNumber inNumber){
+
+ const int keyDataLenBytes = sizeof(ServerId)+SUBSCRIBER_NUMBER_LENGTH;
+ const int keyDataLen_64Words = keyDataLenBytes >> 3;
+
+ Uint64 keyDataBuf[keyDataLen_64Words+1]; // The "+1" is for rounding...
+
+ char * keyDataBuf_charP = (char *)&keyDataBuf[0];
+ Uint32 * keyDataBuf_wo32P = (Uint32 *)&keyDataBuf[0];
+
+ // Server Id comes first
+ keyDataBuf_wo32P[0] = inServerId;
+ // Then subscriber number
+ memcpy(&keyDataBuf_charP[sizeof(ServerId)], inNumber,
+ SUBSCRIBER_NUMBER_LENGTH);
+
+ return pNDB->startTransaction(0, keyDataBuf_charP, keyDataLenBytes);
+}
+
+void T1_Callback(int result, NdbConnection * pCon, void * threadData);
+void T2_Callback(int result, NdbConnection * pCon, void * threadData);
+void T3_Callback_1(int result, NdbConnection * pCon, void * threadData);
+void T3_Callback_2(int result, NdbConnection * pCon, void * threadData);
+void T3_Callback_3(int result, NdbConnection * pCon, void * threadData);
+void T4_Callback_1(int result, NdbConnection * pCon, void * threadData);
+void T4_Callback_2(int result, NdbConnection * pCon, void * threadData);
+void T4_Callback_3(int result, NdbConnection * pCon, void * threadData);
+void T5_Callback_1(int result, NdbConnection * pCon, void * threadData);
+void T5_Callback_2(int result, NdbConnection * pCon, void * threadData);
+void T5_Callback_3(int result, NdbConnection * pCon, void * threadData);
+
+/**
+ * Transaction 1 - T1
+ *
+ * Update location and changed by/time on a subscriber
+ *
+ * Input:
+ * SubscriberNumber,
+ * Location,
+ * ChangedBy,
+ * ChangedTime
+ *
+ * Output:
+ */
+void
+start_T1(Ndb * pNDB, ThreadData * td){
+
+ DEBUG2("T1(%.*s): - Starting\n", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number);
+
+ int check;
+ NdbConnection * pCON = pNDB->startTransaction();
+ if (pCON != NULL) {
+ NdbOperation *MyOp = pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ if (MyOp != NULL) {
+ MyOp->updateTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->setValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->setValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->setValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ pCON->executeAsynchPrepare( Commit , T1_Callback, td);
+ } else {
+ CHECK_NULL(MyOp, "T1: getNdbOperation", pCON);
+ }//if
+ } else {
+ error_handler("T1-1: startTranscation",
+ pNDB->getNdbErrorString(),
+ pNDB->getNdbError());
+ }//if
+}
+
+void
+T1_Callback(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+
+ DEBUG2("T1(%.*s): - Completing\n", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number);
+
+ CHECK_MINUS_ONE(result, "T1: Commit",
+ pCON);
+ td->pNDB->closeTransaction(pCON);
+ complete_T1(td);
+}
+
+/**
+ * Transaction 2 - T2
+ *
+ * Read from Subscriber:
+ *
+ * Input:
+ * SubscriberNumber
+ *
+ * Output:
+ * Location
+ * Changed by
+ * Changed Timestamp
+ * Name
+ */
+void
+start_T2(Ndb * pNDB, ThreadData * td){
+
+ DEBUG3("T2(%.*s, %p): - Starting\n", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.location);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * pCON = pNDB->startTransaction();
+ if (pCON == NULL)
+ error_handler("T2-1: startTransaction",
+ pNDB->getNdbErrorString(),
+ pNDB->getNdbError());
+
+ NdbOperation *MyOp= pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOp, "T2: getNdbOperation",
+ pCON);
+
+ MyOp->readTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ MyOp->getValue(IND_SUBSCRIBER_NAME,
+ td->transactionData.name);
+ pCON->executeAsynchPrepare( Commit, T2_Callback, td );
+}
+
+void
+T2_Callback(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ DEBUG3("T2(%.*s, %p): - Completing\n", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.location);
+
+ CHECK_MINUS_ONE(result, "T2: Commit", pCON);
+ td->pNDB->closeTransaction(pCON);
+ complete_T2(td);
+}
+
+/**
+ * Transaction 3 - T3
+ *
+ * Read session details
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ *
+ * Output:
+ * BranchExecuted
+ * SessionDetails
+ * ChangedBy
+ * ChangedTime
+ * Location
+ */
+void
+start_T3(Ndb * pNDB, ThreadData * td){
+
+ DEBUG3("T3(%.*s, %.2d): - Starting\n", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * pCON = startTransaction(pNDB,
+ td->transactionData.server_id,
+ td->transactionData.number);
+ if (pCON == NULL)
+ error_handler("T3-1: startTranscation",
+ pNDB->getNdbErrorString(),
+ pNDB->getNdbError());
+
+ NdbOperation *MyOp= pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOp, "T3-1: getNdbOperation",
+ pCON);
+
+ MyOp->readTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ MyOp->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&td->transactionData.group_id);
+ MyOp->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&td->transactionData.sessions);
+ pCON->executeAsynchPrepare( NoCommit , T3_Callback_1, td);
+}
+
+void
+T3_Callback_1(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ DEBUG3("T3(%.*s, %.2d): - Callback 1\n", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ CHECK_MINUS_ONE(result, "T3-1: NoCommit", pCON);
+
+ NdbOperation * MyOp = pCON->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOp, "T3-2: getNdbOperation",
+ pCON);
+
+ MyOp->readTuple();
+ MyOp->equal(IND_GROUP_ID,
+ (char*)&td->transactionData.group_id);
+ MyOp->getValue(IND_GROUP_ALLOW_READ,
+ (char *)&td->transactionData.permission);
+ pCON->executeAsynchPrepare( NoCommit, T3_Callback_2, td );
+}
+
+void
+T3_Callback_2(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+
+ CHECK_MINUS_ONE(result, "T3-2: NoCommit", pCON);
+
+ Uint32 permission = td->transactionData.permission;
+ Uint32 sessions = td->transactionData.sessions;
+ Uint32 server_bit = td->transactionData.server_bit;
+
+ if(((permission & server_bit) == server_bit) &&
+ ((sessions & server_bit) == server_bit)){
+
+ memcpy(td->transactionData.suffix,
+ &td->transactionData.number
+ [SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH],
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+ DEBUG5("T3(%.*s, %.2d): - Callback 2 - reading(%.*s)\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH,
+ td->transactionData.suffix);
+
+ /* Operation 3 */
+ NdbOperation * MyOp = pCON->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOp, "T3-3: getNdbOperation",
+ pCON);
+
+ MyOp->simpleRead();
+ MyOp->equal(IND_SESSION_SUBSCRIBER,
+ (char*)td->transactionData.number);
+ MyOp->equal(IND_SESSION_SERVER,
+ (char*)&td->transactionData.server_id);
+ MyOp->getValue(IND_SESSION_DATA,
+ (char *)td->transactionData.session_details);
+
+ /* Operation 4 */
+ MyOp = pCON->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOp, "T3-4: getNdbOperation",
+ pCON);
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SERVER_ID,
+ (char*)&td->transactionData.server_id);
+ MyOp->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)td->transactionData.suffix);
+ MyOp->incValue(IND_SERVER_READS, (uint32)1);
+ td->transactionData.branchExecuted = 1;
+ } else {
+ DEBUG3("T3(%.*s, %.2d): - Callback 2 - no read\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+ td->transactionData.branchExecuted = 0;
+ }
+ pCON->executeAsynchPrepare( Commit, T3_Callback_3, td );
+}
+
+void
+T3_Callback_3(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ DEBUG3("T3(%.*s, %.2d): - Completing\n", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ CHECK_MINUS_ONE(result, "T3-3: Commit", pCON);
+
+ td->pNDB->closeTransaction(pCON);
+ complete_T3(td);
+}
+
+/**
+ * Transaction 4 - T4
+ *
+ * Create session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * SessionDetails,
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+void
+start_T4(Ndb * pNDB, ThreadData * td){
+
+ DEBUG3("T4(%.*s, %.2d): - Starting\n", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * pCON = startTransaction(pNDB,
+ td->transactionData.server_id,
+ td->transactionData.number);
+ if (pCON == NULL)
+ error_handler("T4-1: startTranscation",
+ pNDB->getNdbErrorString(),
+ pNDB->getNdbError());
+
+ NdbOperation *MyOp= pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOp, "T4-1: getNdbOperation",
+ pCON);
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ MyOp->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&td->transactionData.group_id);
+ MyOp->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&td->transactionData.sessions);
+ MyOp->incValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)td->transactionData.server_bit);
+ pCON->executeAsynchPrepare( NoCommit , T4_Callback_1, td);
+}
+
+void
+T4_Callback_1(int result, NdbConnection * pCON, void * threadData){
+ CHECK_MINUS_ONE(result, "T4-1: NoCommit", pCON);
+ ThreadData * td = (ThreadData *)threadData;
+
+ DEBUG3("T4(%.*s, %.2d): - Callback 1\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+
+ NdbOperation * MyOp = pCON->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOp, "T4-2: getNdbOperation",
+ pCON);
+
+ MyOp->readTuple();
+ MyOp->equal(IND_GROUP_ID,
+ (char*)&td->transactionData.group_id);
+ MyOp->getValue(IND_GROUP_ALLOW_INSERT,
+ (char *)&td->transactionData.permission);
+ pCON->executeAsynchPrepare( NoCommit , T4_Callback_2, td);
+}
+
+void
+T4_Callback_2(int result, NdbConnection * pCON, void * threadData){
+ CHECK_MINUS_ONE(result, "T4-2: NoCommit", pCON);
+ ThreadData * td = (ThreadData *)threadData;
+
+ Uint32 permission = td->transactionData.permission;
+ Uint32 sessions = td->transactionData.sessions;
+ Uint32 server_bit = td->transactionData.server_bit;
+
+ if(((permission & server_bit) == server_bit) &&
+ ((sessions & server_bit) == 0)){
+
+ memcpy(td->transactionData.suffix,
+ &td->transactionData.number
+ [SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH],
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+
+ DEBUG5("T4(%.*s, %.2d): - Callback 2 - inserting(%.*s)\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH,
+ td->transactionData.suffix);
+
+ /* Operation 3 */
+
+ NdbOperation * MyOp = pCON->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOp, "T4-3: getNdbOperation",
+ pCON);
+
+ MyOp->insertTuple();
+ MyOp->equal(IND_SESSION_SUBSCRIBER,
+ (char*)td->transactionData.number);
+ MyOp->equal(IND_SESSION_SERVER,
+ (char*)&td->transactionData.server_id);
+ MyOp->setValue(SESSION_DATA,
+ (char *)td->transactionData.session_details);
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOp = pCON->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOp, "T4-5: getNdbOperation",
+ pCON);
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SERVER_ID,
+ (char*)&td->transactionData.server_id);
+ MyOp->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)td->transactionData.suffix);
+ MyOp->incValue(IND_SERVER_INSERTS, (uint32)1);
+ td->transactionData.branchExecuted = 1;
+ } else {
+ td->transactionData.branchExecuted = 0;
+ DEBUG5("T4(%.*s, %.2d): - Callback 2 - %s %s\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ ((permission & server_bit) ?
+ "permission - " : "no permission - "),
+ ((sessions & server_bit) ?
+ "in session - " : "no in session - "));
+ }
+
+ if(!td->transactionData.do_rollback && td->transactionData.branchExecuted){
+ pCON->executeAsynchPrepare(Commit, T4_Callback_3, td);
+ } else {
+ pCON->executeAsynchPrepare(Rollback, T4_Callback_3, td);
+ }
+}
+
+void
+T4_Callback_3(int result, NdbConnection * pCON, void * threadData){
+ CHECK_MINUS_ONE(result, "T4-3: Commit", pCON);
+ ThreadData * td = (ThreadData *)threadData;
+
+ DEBUG3("T4(%.*s, %.2d): - Completing\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ td->pNDB->closeTransaction(pCON);
+ complete_T4(td);
+}
+
+/**
+ * Transaction 5 - T5
+ *
+ * Delete session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+void
+start_T5(Ndb * pNDB, ThreadData * td){
+
+ DEBUG3("T5(%.*s, %.2d): - Starting\n", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * pCON = pNDB->startTransaction();
+ if (pCON == NULL)
+ error_handler("T5-1: startTranscation",
+ pNDB->getNdbErrorString(),
+ pNDB->getNdbError());
+
+ NdbOperation * MyOp= pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOp, "T5-1: getNdbOperation",
+ pCON);
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ MyOp->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&td->transactionData.group_id);
+ MyOp->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&td->transactionData.sessions);
+ MyOp->subValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)td->transactionData.server_bit);
+ pCON->executeAsynchPrepare( NoCommit, T5_Callback_1, td );
+}
+
+void
+T5_Callback_1(int result, NdbConnection * pCON, void * threadData){
+ CHECK_MINUS_ONE(result, "T5-1: NoCommit", pCON);
+ ThreadData * td = (ThreadData *)threadData;
+
+ DEBUG3("T5(%.*s, %.2d): - Callback 1\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ NdbOperation * MyOp = pCON->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOp, "T5-2: getNdbOperation",
+ pCON);
+
+ MyOp->readTuple();
+ MyOp->equal(IND_GROUP_ID,
+ (char*)&td->transactionData.group_id);
+ MyOp->getValue(IND_GROUP_ALLOW_DELETE,
+ (char *)&td->transactionData.permission);
+ pCON->executeAsynchPrepare( NoCommit, T5_Callback_2, td );
+}
+
+void
+T5_Callback_2(int result, NdbConnection * pCON, void * threadData){
+ CHECK_MINUS_ONE(result, "T5-2: NoCommit", pCON);
+ ThreadData * td = (ThreadData *)threadData;
+
+ Uint32 permission = td->transactionData.permission;
+ Uint32 sessions = td->transactionData.sessions;
+ Uint32 server_bit = td->transactionData.server_bit;
+
+ if(((permission & server_bit) == server_bit) &&
+ ((sessions & server_bit) == server_bit)){
+
+ memcpy(td->transactionData.suffix,
+ &td->transactionData.number
+ [SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH],
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+
+ DEBUG5("T5(%.*s, %.2d): - Callback 2 - deleting(%.*s)\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH,
+ td->transactionData.suffix);
+
+ /* Operation 3 */
+ NdbOperation * MyOp = pCON->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOp, "T5-3: getNdbOperation",
+ pCON);
+
+ MyOp->deleteTuple();
+ MyOp->equal(IND_SESSION_SUBSCRIBER,
+ (char*)td->transactionData.number);
+ MyOp->equal(IND_SESSION_SERVER,
+ (char*)&td->transactionData.server_id);
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOp = pCON->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOp, "T5-5: getNdbOperation",
+ pCON);
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SERVER_ID,
+ (char*)&td->transactionData.server_id);
+ MyOp->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)td->transactionData.suffix);
+ MyOp->incValue(IND_SERVER_DELETES, (uint32)1);
+ td->transactionData.branchExecuted = 1;
+ } else {
+ td->transactionData.branchExecuted = 0;
+
+ DEBUG5("T5(%.*s, %.2d): - Callback 2 - no delete - %s %s\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ ((permission & server_bit) ?
+ "permission - " : "no permission - "),
+ ((sessions & server_bit) ?
+ "in session - " : "no in session - "));
+ }
+
+ if(!td->transactionData.do_rollback && td->transactionData.branchExecuted){
+ pCON->executeAsynchPrepare(Commit, T5_Callback_3, td);
+ } else {
+ pCON->executeAsynchPrepare(Rollback, T5_Callback_3, td);
+ }
+}
+
+void
+T5_Callback_3(int result, NdbConnection * pCON, void * threadData){
+ CHECK_MINUS_ONE(result, "T5-3: Commit", pCON);
+ ThreadData * td = (ThreadData *)threadData;
+
+ DEBUG3("T5(%.*s, %.2d): - Completing\n",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ td->pNDB->closeTransaction(pCON);
+ complete_T5(td);
+}
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/user/ndb_async2.cpp b/ndb/test/ndbapi/lmc-bench/async-src/user/ndb_async2.cpp
new file mode 100644
index 00000000000..0c1d138defb
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/user/ndb_async2.cpp
@@ -0,0 +1,754 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#define DEBUG_ON
+
+#include <string.h>
+#include "userInterface.h"
+
+#include "macros.h"
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+#include <NdbSleep.h>
+
+#include <NdbApi.hpp>
+
+void T1_Callback(int result, NdbConnection * pCon, void * threadData);
+void T2_Callback(int result, NdbConnection * pCon, void * threadData);
+void T3_Callback_1(int result, NdbConnection * pCon, void * threadData);
+void T3_Callback_2(int result, NdbConnection * pCon, void * threadData);
+void T3_Callback_3(int result, NdbConnection * pCon, void * threadData);
+void T4_Callback_1(int result, NdbConnection * pCon, void * threadData);
+void T4_Callback_2(int result, NdbConnection * pCon, void * threadData);
+void T4_Callback_3(int result, NdbConnection * pCon, void * threadData);
+void T5_Callback_1(int result, NdbConnection * pCon, void * threadData);
+void T5_Callback_2(int result, NdbConnection * pCon, void * threadData);
+void T5_Callback_3(int result, NdbConnection * pCon, void * threadData);
+
+static int stat_async = 0;
+
+/**
+ * Transaction 1 - T1
+ *
+ * Update location and changed by/time on a subscriber
+ *
+ * Input:
+ * SubscriberNumber,
+ * Location,
+ * ChangedBy,
+ * ChangedTime
+ *
+ * Output:
+ */
+
+#define SFX_START (SUBSCRIBER_NUMBER_LENGTH - SUBSCRIBER_NUMBER_SUFFIX_LENGTH)
+
+inline
+NdbConnection *
+startTransaction(Ndb * pNDB, ThreadData * td){
+ return pNDB->startTransactionDGroup (0,
+ &td->transactionData.number[SFX_START],
+ 1);
+}
+
+void
+start_T1(Ndb * pNDB, ThreadData * td, int async){
+
+ DEBUG2("T1(%.*s): - Starting", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number);
+
+ NdbConnection * pCON = 0;
+ while((pCON = startTransaction(pNDB, td)) == 0){
+ CHECK_ALLOWED_ERROR("T1: startTransaction", td, pNDB->getNdbError());
+ NdbSleep_MilliSleep(10);
+ }
+
+ NdbOperation *MyOp = pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ if (MyOp != NULL) {
+ MyOp->updateTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->setValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->setValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->setValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ if (async == 1) {
+ pCON->executeAsynchPrepare( Commit , T1_Callback, td);
+ } else {
+ int result = pCON->execute(Commit);
+ T1_Callback(result, pCON, (void*)td);
+ return;
+ }//if
+ } else {
+ CHECK_NULL(MyOp, "T1: getNdbOperation", td, pCON->getNdbError());
+ }//if
+}
+
+void
+T1_Callback(int result, NdbConnection * pCON, void * threadData) {
+ ThreadData * td = (ThreadData *)threadData;
+
+ DEBUG2("T1(%.*s): - Completing", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number);
+
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T1: Commit", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T1(td->pNDB, td, stat_async);
+ return;
+ }//if
+ td->pNDB->closeTransaction(pCON);
+ complete_T1(td);
+}
+
+/**
+ * Transaction 2 - T2
+ *
+ * Read from Subscriber:
+ *
+ * Input:
+ * SubscriberNumber
+ *
+ * Output:
+ * Location
+ * Changed by
+ * Changed Timestamp
+ * Name
+ */
+void
+start_T2(Ndb * pNDB, ThreadData * td, int async){
+
+ DEBUG3("T2(%.*s, %d): - Starting", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.location);
+
+ NdbConnection * pCON = 0;
+
+ while((pCON = startTransaction(pNDB, td)) == 0){
+ CHECK_ALLOWED_ERROR("T2-1: startTransaction", td, pNDB->getNdbError());
+ NdbSleep_MilliSleep(10);
+ }
+
+ NdbOperation *MyOp= pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOp, "T2: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->readTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ MyOp->getValue(IND_SUBSCRIBER_NAME,
+ td->transactionData.name);
+ if (async == 1) {
+ pCON->executeAsynchPrepare( Commit , T2_Callback, td);
+ } else {
+ int result = pCON->execute(Commit);
+ T2_Callback(result, pCON, (void*)td);
+ return;
+ }//if
+}
+
+void
+T2_Callback(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ DEBUG3("T2(%.*s, %d): - Completing", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.location);
+
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T2: Commit", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T2(td->pNDB, td, stat_async);
+ return;
+ }//if
+ td->pNDB->closeTransaction(pCON);
+ complete_T2(td);
+}
+
+/**
+ * Transaction 3 - T3
+ *
+ * Read session details
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ *
+ * Output:
+ * BranchExecuted
+ * SessionDetails
+ * ChangedBy
+ * ChangedTime
+ * Location
+ */
+void
+start_T3(Ndb * pNDB, ThreadData * td, int async){
+
+ DEBUG3("T3(%.*s, %.2d): - Starting", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ NdbConnection * pCON = 0;
+
+ while((pCON = startTransaction(pNDB, td)) == 0){
+ CHECK_ALLOWED_ERROR("T3-1: startTransaction", td, pNDB->getNdbError());
+ NdbSleep_MilliSleep(10);
+ }
+
+ NdbOperation *MyOp= pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOp, "T3-1: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->readTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ MyOp->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&td->transactionData.group_id);
+ MyOp->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&td->transactionData.sessions);
+ stat_async = async;
+ if (async == 1) {
+ pCON->executeAsynchPrepare( NoCommit , T3_Callback_1, td);
+ } else {
+ int result = pCON->execute( NoCommit );
+ T3_Callback_1(result, pCON, (void*)td);
+ return;
+ }//if
+}
+
+void
+T3_Callback_1(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ DEBUG3("T3(%.*s, %.2d): - Callback 1", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T3-1: execute", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T3(td->pNDB, td, stat_async);
+ return;
+ }//if
+
+ NdbOperation * MyOp = pCON->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOp, "T3-2: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->readTuple();
+ MyOp->equal(IND_GROUP_ID,
+ (char*)&td->transactionData.group_id);
+ MyOp->getValue(IND_GROUP_ALLOW_READ,
+ (char *)&td->transactionData.permission);
+ if (stat_async == 1) {
+ pCON->executeAsynchPrepare( NoCommit , T3_Callback_2, td);
+ } else {
+ int result = pCON->execute( NoCommit );
+ T3_Callback_2(result, pCON, (void*)td);
+ return;
+ }//if
+}
+
+void
+T3_Callback_2(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T3-2: execute", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T3(td->pNDB, td, stat_async);
+ return;
+ }//if
+
+ Uint32 permission = td->transactionData.permission;
+ Uint32 sessions = td->transactionData.sessions;
+ Uint32 server_bit = td->transactionData.server_bit;
+
+ if(((permission & server_bit) == server_bit) &&
+ ((sessions & server_bit) == server_bit)){
+
+ memcpy(td->transactionData.suffix,
+ &td->transactionData.number[SFX_START],
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+ DEBUG5("T3(%.*s, %.2d): - Callback 2 - reading(%.*s)",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH,
+ td->transactionData.suffix);
+
+ /* Operation 3 */
+ NdbOperation * MyOp = pCON->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOp, "T3-3: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->simpleRead();
+ MyOp->equal(IND_SESSION_SUBSCRIBER,
+ (char*)td->transactionData.number);
+ MyOp->equal(IND_SESSION_SERVER,
+ (char*)&td->transactionData.server_id);
+ MyOp->getValue(IND_SESSION_DATA,
+ (char *)td->transactionData.session_details);
+
+ /* Operation 4 */
+ MyOp = pCON->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOp, "T3-4: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SERVER_ID,
+ (char*)&td->transactionData.server_id);
+ MyOp->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)td->transactionData.suffix);
+ MyOp->incValue(IND_SERVER_READS, (uint32)1);
+ td->transactionData.branchExecuted = 1;
+ } else {
+ DEBUG3("T3(%.*s, %.2d): - Callback 2 - no read",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+ td->transactionData.branchExecuted = 0;
+ }
+ if (stat_async == 1) {
+ pCON->executeAsynchPrepare( Commit , T3_Callback_3, td);
+ } else {
+ int result = pCON->execute( Commit );
+ T3_Callback_3(result, pCON, (void*)td);
+ return;
+ }//if
+}
+
+void
+T3_Callback_3(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ DEBUG3("T3(%.*s, %.2d): - Completing", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T3-3: Commit", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T3(td->pNDB, td, stat_async);
+ return;
+ }//if
+ td->pNDB->closeTransaction(pCON);
+ complete_T3(td);
+}
+
+/**
+ * Transaction 4 - T4
+ *
+ * Create session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * SessionDetails,
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+void
+start_T4(Ndb * pNDB, ThreadData * td, int async){
+
+ DEBUG3("T4(%.*s, %.2d): - Starting", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ NdbConnection * pCON = 0;
+ while((pCON = startTransaction(pNDB, td)) == 0){
+ CHECK_ALLOWED_ERROR("T4-1: startTransaction", td, pNDB->getNdbError());
+ NdbSleep_MilliSleep(10);
+ }
+
+ NdbOperation *MyOp= pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOp, "T4-1: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ MyOp->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&td->transactionData.group_id);
+ MyOp->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&td->transactionData.sessions);
+ MyOp->incValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)td->transactionData.server_bit);
+ stat_async = async;
+ if (async == 1) {
+ pCON->executeAsynchPrepare( NoCommit , T4_Callback_1, td);
+ } else {
+ int result = pCON->execute( NoCommit );
+ T4_Callback_1(result, pCON, (void*)td);
+ return;
+ }//if
+}
+
+void
+T4_Callback_1(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T4-1: execute", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T4(td->pNDB, td, stat_async);
+ return;
+ }//if
+
+ DEBUG3("T4(%.*s, %.2d): - Callback 1",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+
+ NdbOperation * MyOp = pCON->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOp, "T4-2: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->readTuple();
+ MyOp->equal(IND_GROUP_ID,
+ (char*)&td->transactionData.group_id);
+ MyOp->getValue(IND_GROUP_ALLOW_INSERT,
+ (char *)&td->transactionData.permission);
+ if (stat_async == 1) {
+ pCON->executeAsynchPrepare( NoCommit , T4_Callback_2, td);
+ } else {
+ int result = pCON->execute( NoCommit );
+ T4_Callback_2(result, pCON, (void*)td);
+ return;
+ }//if
+}
+
+void
+T4_Callback_2(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T4-2: execute", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T4(td->pNDB, td, stat_async);
+ return;
+ }//if
+
+ Uint32 permission = td->transactionData.permission;
+ Uint32 sessions = td->transactionData.sessions;
+ Uint32 server_bit = td->transactionData.server_bit;
+
+ if(((permission & server_bit) == server_bit) &&
+ ((sessions & server_bit) == 0)){
+
+ memcpy(td->transactionData.suffix,
+ &td->transactionData.number[SFX_START],
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+
+ DEBUG5("T4(%.*s, %.2d): - Callback 2 - inserting(%.*s)",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH,
+ td->transactionData.suffix);
+
+ /* Operation 3 */
+
+ NdbOperation * MyOp = pCON->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOp, "T4-3: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->insertTuple();
+ MyOp->equal(IND_SESSION_SUBSCRIBER,
+ (char*)td->transactionData.number);
+ MyOp->equal(IND_SESSION_SERVER,
+ (char*)&td->transactionData.server_id);
+ MyOp->setValue(SESSION_DATA,
+ (char *)td->transactionData.session_details);
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOp = pCON->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOp, "T4-5: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SERVER_ID,
+ (char*)&td->transactionData.server_id);
+ MyOp->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)td->transactionData.suffix);
+ MyOp->incValue(IND_SERVER_INSERTS, (uint32)1);
+ td->transactionData.branchExecuted = 1;
+ } else {
+ td->transactionData.branchExecuted = 0;
+ DEBUG5("T4(%.*s, %.2d): - Callback 2 - %s %s",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ ((permission & server_bit) ?
+ "permission - " : "no permission - "),
+ ((sessions & server_bit) ?
+ "in session - " : "no in session - "));
+ }
+
+ if(!td->transactionData.do_rollback && td->transactionData.branchExecuted){
+ if (stat_async == 1) {
+ pCON->executeAsynchPrepare( Commit , T4_Callback_3, td);
+ } else {
+ int result = pCON->execute( Commit );
+ T4_Callback_3(result, pCON, (void*)td);
+ return;
+ }//if
+ } else {
+ if (stat_async == 1) {
+ pCON->executeAsynchPrepare( Rollback , T4_Callback_3, td);
+ } else {
+ int result = pCON->execute( Rollback );
+ T4_Callback_3(result, pCON, (void*)td);
+ return;
+ }//if
+ }
+}
+
+void
+T4_Callback_3(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T4-3: Commit", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T4(td->pNDB, td, stat_async);
+ return;
+ }//if
+
+ DEBUG3("T4(%.*s, %.2d): - Completing",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ td->pNDB->closeTransaction(pCON);
+ complete_T4(td);
+}
+
+/**
+ * Transaction 5 - T5
+ *
+ * Delete session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+void
+start_T5(Ndb * pNDB, ThreadData * td, int async){
+
+ DEBUG3("T5(%.*s, %.2d): - Starting", SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ NdbConnection * pCON = 0;
+ while((pCON = startTransaction(pNDB, td)) == 0){
+ CHECK_ALLOWED_ERROR("T5-1: startTransaction", td, pNDB->getNdbError());
+ NdbSleep_MilliSleep(10);
+ }
+
+ NdbOperation * MyOp= pCON->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOp, "T5-1: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SUBSCRIBER_NUMBER,
+ td->transactionData.number);
+ MyOp->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&td->transactionData.location);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ td->transactionData.changed_by);
+ MyOp->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ td->transactionData.changed_time);
+ MyOp->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&td->transactionData.group_id);
+ MyOp->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&td->transactionData.sessions);
+ MyOp->subValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)td->transactionData.server_bit);
+ stat_async = async;
+ if (async == 1) {
+ pCON->executeAsynchPrepare( NoCommit , T5_Callback_1, td);
+ } else {
+ int result = pCON->execute( NoCommit );
+ T5_Callback_1(result, pCON, (void*)td);
+ return;
+ }//if
+}
+
+void
+T5_Callback_1(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T5-1: execute", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T5(td->pNDB, td, stat_async);
+ return;
+ }//if
+
+ DEBUG3("T5(%.*s, %.2d): - Callback 1",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ NdbOperation * MyOp = pCON->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOp, "T5-2: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->readTuple();
+ MyOp->equal(IND_GROUP_ID,
+ (char*)&td->transactionData.group_id);
+ MyOp->getValue(IND_GROUP_ALLOW_DELETE,
+ (char *)&td->transactionData.permission);
+ if (stat_async == 1) {
+ pCON->executeAsynchPrepare( NoCommit , T5_Callback_2, td);
+ } else {
+ int result = pCON->execute( NoCommit );
+ T5_Callback_2(result, pCON, (void*)td);
+ return;
+ }//if
+}
+
+void
+T5_Callback_2(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T5-2: execute", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T5(td->pNDB, td, stat_async);
+ return;
+ }//if
+
+ Uint32 permission = td->transactionData.permission;
+ Uint32 sessions = td->transactionData.sessions;
+ Uint32 server_bit = td->transactionData.server_bit;
+
+ if(((permission & server_bit) == server_bit) &&
+ ((sessions & server_bit) == server_bit)){
+
+ memcpy(td->transactionData.suffix,
+ &td->transactionData.number[SFX_START],
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+
+ DEBUG5("T5(%.*s, %.2d): - Callback 2 - deleting(%.*s)",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH,
+ td->transactionData.suffix);
+
+ /* Operation 3 */
+ NdbOperation * MyOp = pCON->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOp, "T5-3: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->deleteTuple();
+ MyOp->equal(IND_SESSION_SUBSCRIBER,
+ (char*)td->transactionData.number);
+ MyOp->equal(IND_SESSION_SERVER,
+ (char*)&td->transactionData.server_id);
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOp = pCON->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOp, "T5-5: getNdbOperation", td,
+ pCON->getNdbError());
+
+ MyOp->interpretedUpdateTuple();
+ MyOp->equal(IND_SERVER_ID,
+ (char*)&td->transactionData.server_id);
+ MyOp->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)td->transactionData.suffix);
+ MyOp->incValue(IND_SERVER_DELETES, (uint32)1);
+ td->transactionData.branchExecuted = 1;
+ } else {
+ td->transactionData.branchExecuted = 0;
+
+ DEBUG5("T5(%.*s, %.2d): - Callback 2 - no delete - %s %s",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id,
+ ((permission & server_bit) ?
+ "permission - " : "no permission - "),
+ ((sessions & server_bit) ?
+ "in session - " : "no in session - "));
+ }
+
+ if(!td->transactionData.do_rollback && td->transactionData.branchExecuted){
+ if (stat_async == 1) {
+ pCON->executeAsynchPrepare( Commit , T5_Callback_3, td);
+ } else {
+ int result = pCON->execute( Commit );
+ T5_Callback_3(result, pCON, (void*)td);
+ return;
+ }//if
+ } else {
+ if (stat_async == 1) {
+ pCON->executeAsynchPrepare( Rollback , T5_Callback_3, td);
+ } else {
+ int result = pCON->execute( Rollback );
+ T5_Callback_3(result, pCON, (void*)td);
+ return;
+ }//if
+ }
+}
+
+void
+T5_Callback_3(int result, NdbConnection * pCON, void * threadData){
+ ThreadData * td = (ThreadData *)threadData;
+ if (result == -1) {
+ CHECK_ALLOWED_ERROR("T5-3: Commit", td, pCON->getNdbError());
+ td->pNDB->closeTransaction(pCON);
+ start_T5(td->pNDB, td, stat_async);
+ return;
+ }//if
+
+ DEBUG3("T5(%.*s, %.2d): - Completing",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number,
+ td->transactionData.server_id);
+
+ td->pNDB->closeTransaction(pCON);
+ complete_T5(td);
+}
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/user/ndb_error.hpp b/ndb/test/ndbapi/lmc-bench/async-src/user/ndb_error.hpp
new file mode 100644
index 00000000000..91a061c7cf4
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/user/ndb_error.hpp
@@ -0,0 +1,63 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_ERROR_H
+#define NDB_ERROR_H
+
+#include <stdio.h>
+#include <NdbOut.hpp>
+#include "userInterface.h"
+#include <NdbError.hpp>
+
+inline
+void
+CHECK_ALLOWED_ERROR(const char * str,
+ const ThreadData * td,
+ const struct NdbError & error){
+
+ char buf[100];
+ snprintf(buf, sizeof(buf), "subscriber = %.*s ",
+ SUBSCRIBER_NUMBER_LENGTH,
+ td->transactionData.number);
+ ndbout << str << " " << error << endl
+ << buf;
+ showTime();
+
+ switch(error.classification) {
+ case NdbError::TimeoutExpired:
+ case NdbError::OverloadError:
+ case NdbError::TemporaryResourceError:
+ case NdbError::NodeRecoveryError:
+ break;
+ default:
+ if(error.status != NdbError::TemporaryError)
+ exit(-1);
+ }
+}
+
+inline
+void
+CHECK_NULL(void * null,
+ const char * str,
+ const ThreadData * td,
+ const struct NdbError & err){
+ if(null == 0){
+ CHECK_ALLOWED_ERROR(str, td, err);
+ exit(-1);
+ }
+}
+
+#endif
diff --git a/ndb/test/ndbapi/lmc-bench/async-src/user/userInterface.cpp b/ndb/test/ndbapi/lmc-bench/async-src/user/userInterface.cpp
new file mode 100644
index 00000000000..ece82628ba7
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/async-src/user/userInterface.cpp
@@ -0,0 +1,120 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <sys/types.h>
+#include <time.h>
+
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+#include "userInterface.h"
+#include <NdbMutex.h>
+#include <NdbThread.h>
+#include <NdbTick.h>
+#include <NdbApi.hpp>
+#include <NdbOut.hpp>
+#include <NdbStdio.h>
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+#ifndef NDB_WIN32
+#include <unistd.h>
+#endif
+
+
+static NdbMutex* startupMutex = NdbMutex_Create();
+
+Ndb*
+asyncDbConnect(int parallellism){
+ NdbMutex_Lock(startupMutex);
+ Ndb * pNDB = new Ndb("");
+
+ pNDB->init(parallellism + 1);
+
+ while(pNDB->waitUntilReady() != 0){
+ }
+
+ NdbMutex_Unlock(startupMutex);
+
+ return pNDB;
+}
+
+void
+asyncDbDisconnect(Ndb* pNDB)
+{
+ delete pNDB;
+}
+
+double
+userGetTime(void)
+{
+ static bool initialized = false;
+ static NDB_TICKS initSecs = 0;
+ static Uint32 initMicros = 0;
+ double timeValue = 0;
+
+ if ( !initialized ) {
+ initialized = true;
+ NdbTick_CurrentMicrosecond(&initSecs, &initMicros);
+ timeValue = 0.0;
+ } else {
+ NDB_TICKS secs = 0;
+ Uint32 micros = 0;
+
+ NdbTick_CurrentMicrosecond(&secs, &micros);
+ double s = (double)secs - (double)initSecs;
+ double us = (double)micros - (double)initMicros;
+
+ timeValue = s + (us / 1000000.0);
+ }
+ return timeValue;
+}
+
+void showTime()
+{
+ char buf[128];
+ struct tm* tm_now;
+ time_t now;
+ now = ::time((time_t*)NULL);
+ tm_now = ::gmtime(&now);
+
+ ::snprintf(buf, 128,
+ "%d-%.2d-%.2d %.2d:%.2d:%.2d",
+ tm_now->tm_year + 1900,
+ tm_now->tm_mon,
+ tm_now->tm_mday,
+ tm_now->tm_hour,
+ tm_now->tm_min,
+ tm_now->tm_sec);
+
+ ndbout_c("Time: %s", buf);
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/bin/.empty b/ndb/test/ndbapi/lmc-bench/bin/.empty
new file mode 100644
index 00000000000..e69de29bb2d
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/bin/.empty
diff --git a/ndb/test/ndbapi/lmc-bench/include/ndb_schema.hpp b/ndb/test/ndbapi/lmc-bench/include/ndb_schema.hpp
new file mode 100644
index 00000000000..af08bc2eecd
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/include/ndb_schema.hpp
@@ -0,0 +1,78 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_SCHEMA_H
+#define NDB_SCHEMA_H
+
+#include "testDefinitions.h"
+
+#define SUBSCRIBER_TABLE "SUBSCRIBER"
+#define SUBSCRIBER_NUMBER "NUMBER"
+#define SUBSCRIBER_LOCATION "LOCATION"
+#define SUBSCRIBER_NAME "NAME"
+#define SUBSCRIBER_GROUP "GROUP_ID"
+#define SUBSCRIBER_SESSIONS "SESSIONS"
+#define SUBSCRIBER_CHANGED_BY "CHANGED_BY"
+#define SUBSCRIBER_CHANGED_TIME "CHANGED_TIME"
+
+#define SERVER_TABLE "SERVER"
+#define SERVER_ID "SERVER_ID"
+#define SERVER_SUBSCRIBER_SUFFIX "SUFFIX"
+#define SERVER_NAME "NAME"
+#define SERVER_READS "NO_OF_READ"
+#define SERVER_INSERTS "NO_OF_INSERT"
+#define SERVER_DELETES "NO_OF_DELETE"
+
+#define GROUP_TABLE "GROUP"
+#define GROUP_ID "GROUP_ID"
+#define GROUP_NAME "GROUP_NAME"
+#define GROUP_ALLOW_READ "ALLOW_READ"
+#define GROUP_ALLOW_INSERT "ALLOW_INSERT"
+#define GROUP_ALLOW_DELETE "ALLOW_DELETE"
+
+#define SESSION_TABLE "SESSION"
+#define SESSION_SERVER "SERVER_ID"
+#define SESSION_SUBSCRIBER "NUMBER"
+#define SESSION_DATA "DATA"
+
+/** Numbers */
+
+#define IND_SUBSCRIBER_NUMBER (unsigned)0
+#define IND_SUBSCRIBER_NAME (unsigned)1
+#define IND_SUBSCRIBER_GROUP (unsigned)2
+#define IND_SUBSCRIBER_LOCATION (unsigned)3
+#define IND_SUBSCRIBER_SESSIONS (unsigned)4
+#define IND_SUBSCRIBER_CHANGED_BY (unsigned)5
+#define IND_SUBSCRIBER_CHANGED_TIME (unsigned)6
+
+#define IND_SERVER_SUBSCRIBER_SUFFIX (unsigned)0
+#define IND_SERVER_ID (unsigned)1
+#define IND_SERVER_NAME (unsigned)2
+#define IND_SERVER_READS (unsigned)3
+#define IND_SERVER_INSERTS (unsigned)4
+#define IND_SERVER_DELETES (unsigned)5
+
+#define IND_GROUP_ID (unsigned)0
+#define IND_GROUP_NAME (unsigned)1
+#define IND_GROUP_ALLOW_READ (unsigned)2
+#define IND_GROUP_ALLOW_INSERT (unsigned)3
+#define IND_GROUP_ALLOW_DELETE (unsigned)4
+
+#define IND_SESSION_SUBSCRIBER (unsigned)0
+#define IND_SESSION_SERVER (unsigned)1
+#define IND_SESSION_DATA (unsigned)2
+
+#endif
diff --git a/ndb/test/ndbapi/lmc-bench/include/testDefinitions.h b/ndb/test/ndbapi/lmc-bench/include/testDefinitions.h
new file mode 100644
index 00000000000..c6ad11016b2
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/include/testDefinitions.h
@@ -0,0 +1,96 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TESTDEFINITIONS_H
+#define TESTDEFINITIONS_H
+
+/***************************************************************/
+/* I N C L U D E D F I L E S */
+/***************************************************************/
+
+#include <ndb_types.h>
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+typedef Uint32 uint32;
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+#define OP_PER_TRANS 200
+#define NO_OF_SUBSCRIBERS 500000
+#define NO_OF_GROUPS 100
+#define NO_OF_SERVERS 20
+
+#define SUBSCRIBER_NUMBER_LENGTH 12
+#define SUBSCRIBER_NUMBER_SUFFIX_LENGTH 2
+
+#define SUBSCRIBER_NAME_LENGTH 32
+#define CHANGED_BY_LENGTH 32
+#define CHANGED_TIME_LENGTH 32
+#define SESSION_DETAILS_LENGTH 2000
+#define SERVER_NAME_LENGTH 32
+#define GROUP_NAME_LENGTH 32
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+#define PADDING 4
+
+typedef char SubscriberNumber[SUBSCRIBER_NUMBER_LENGTH];
+typedef char SubscriberSuffix[SUBSCRIBER_NUMBER_SUFFIX_LENGTH + 2];
+typedef char SubscriberName[SUBSCRIBER_NAME_LENGTH];
+typedef char ServerName[SERVER_NAME_LENGTH];
+typedef char GroupName[GROUP_NAME_LENGTH];
+typedef char ChangedBy[CHANGED_BY_LENGTH];
+typedef char ChangedTime[CHANGED_TIME_LENGTH];
+typedef char SessionDetails[SESSION_DETAILS_LENGTH];
+typedef uint32 ServerId;
+typedef uint32 ServerBit;
+typedef uint32 GroupId;
+typedef uint32 Location;
+typedef uint32 Permission;
+
+typedef uint32 Counter;
+typedef uint32 ActiveSessions;
+typedef unsigned int BranchExecuted;
+typedef unsigned int DoRollback;
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+#ifdef __cplusplus
+}
+#endif
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+
+
+#endif /* TESTDEFINITIONS_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/lib/.empty b/ndb/test/ndbapi/lmc-bench/lib/.empty
new file mode 100644
index 00000000000..e69de29bb2d
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/lib/.empty
diff --git a/ndb/test/ndbapi/lmc-bench/script/Makefile b/ndb/test/ndbapi/lmc-bench/script/Makefile
new file mode 100644
index 00000000000..240b5957573
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/script/Makefile
@@ -0,0 +1,5 @@
+include .defs.mk
+
+SOURCES.sh := async-lmc-bench.sh async-lmc-bench-l.sh async-lmc-bench-p10.sh async-lmc-bench-l-p10.sh
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench-l-p10.sh b/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench-l-p10.sh
new file mode 100755
index 00000000000..1ce3969f9fb
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench-l-p10.sh
@@ -0,0 +1,14 @@
+#!/bin/sh
+
+DbCreate -l
+ret=$?
+if [ $ret -ne 0 ]
+then
+ echo "DbCreate failed"
+ exit $ret
+fi
+
+DbAsyncGenerator -time 300 -p 10 $*
+ret=$?
+exit $ret
+
diff --git a/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench-l.sh b/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench-l.sh
new file mode 100755
index 00000000000..a5de71395c4
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench-l.sh
@@ -0,0 +1,14 @@
+#!/bin/sh
+
+DbCreate -l
+ret=$?
+if [ $ret -ne 0 ]
+then
+ echo "DbCreate failed"
+ exit $ret
+fi
+
+DbAsyncGenerator -time 300 $*
+ret=$?
+exit $ret
+
diff --git a/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench-p10.sh b/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench-p10.sh
new file mode 100755
index 00000000000..92c853cdd86
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench-p10.sh
@@ -0,0 +1,14 @@
+#!/bin/sh
+
+DbCreate
+ret=$?
+if [ $ret -ne 0 ]
+then
+ echo "DbCreate failed"
+ exit $ret
+fi
+
+DbAsyncGenerator -time 300 -p 10 $*
+ret=$?
+exit $ret
+
diff --git a/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench.sh b/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench.sh
new file mode 100755
index 00000000000..da8e9d9bf42
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/script/async-lmc-bench.sh
@@ -0,0 +1,14 @@
+#!/bin/sh
+
+DbCreate
+ret=$?
+if [ $ret -ne 0 ]
+then
+ echo "DbCreate failed"
+ exit $ret
+fi
+
+DbAsyncGenerator -time 300 $*
+ret=$?
+exit $ret
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/Makefile b/ndb/test/ndbapi/lmc-bench/src/Makefile
new file mode 100644
index 00000000000..ae7fac9c49b
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+DIRS = \
+ user \
+ populator \
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/README b/ndb/test/ndbapi/lmc-bench/src/README
new file mode 100644
index 00000000000..e81c8ba0051
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/README
@@ -0,0 +1,8 @@
+
+Note that you have to use gnumake to build
+
+On ndbs05:
+use 'gmake' instead of 'make'
+
+On hfXXX:
+do 'module add make'
diff --git a/ndb/test/ndbapi/lmc-bench/src/generator/Makefile b/ndb/test/ndbapi/lmc-bench/src/generator/Makefile
new file mode 100644
index 00000000000..143d9ba655e
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/generator/Makefile
@@ -0,0 +1,17 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+SOURCES = mainGenerator.c dbGenerator.c
+
+CCFLAGS_LOC := -I../include -I../../include
+
+OBJECTS = \
+ mainGenerator.o\
+ dbGenerator.o
+
+BIN_TARGET := DbGenerator
+BIN_TARGET_ARCHIVES := lmc_User
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/generator/dbGenerator.c b/ndb/test/ndbapi/lmc-bench/src/generator/dbGenerator.c
new file mode 100644
index 00000000000..eedcd914d85
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/generator/dbGenerator.c
@@ -0,0 +1,544 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <stdio.h>
+#include <string.h>
+#include "dbGenerator.h"
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+static void getRandomSubscriberNumber(SubscriberNumber number);
+static void getRandomServerId(ServerId *serverId);
+static void getRandomChangedBy(ChangedBy changedBy);
+static void getRandomChangedTime(ChangedTime changedTime);
+
+static void clearTransaction(TransactionDefinition *trans);
+static void initGeneratorStatistics(GeneratorStatistics *gen);
+
+static void doOneTransaction(UserHandle *uh, GeneratorStatistics *gen);
+static void doTransaction_T1(UserHandle *uh, GeneratorStatistics *gen);
+static void doTransaction_T2(UserHandle *uh, GeneratorStatistics *gen);
+static void doTransaction_T3(UserHandle *uh, GeneratorStatistics *gen);
+static void doTransaction_T4(UserHandle *uh, GeneratorStatistics *gen);
+static void doTransaction_T5(UserHandle *uh, GeneratorStatistics *gen);
+
+/***************************************************************
+* L O C A L D A T A *
+***************************************************************/
+
+static SequenceValues transactionDefinition[] = {
+ {25, 1},
+ {25, 2},
+ {20, 3},
+ {15, 4},
+ {15, 5},
+ {0, 0}
+};
+
+static SequenceValues rollbackDefinition[] = {
+ {98, 0},
+ {2 , 1},
+ {0, 0}
+};
+
+static int maxsize = 0;
+
+/***************************************************************
+* P U B L I C D A T A *
+***************************************************************/
+
+/***************************************************************
+****************************************************************
+* L O C A L F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+static void getRandomSubscriberNumber(SubscriberNumber number)
+{
+ uint32 tmp;
+ char sbuf[SUBSCRIBER_NUMBER_LENGTH + 1];
+ tmp = myRandom48(NO_OF_SUBSCRIBERS);
+ sprintf(sbuf, "%.*d", SUBSCRIBER_NUMBER_LENGTH, tmp);
+ memcpy(number, sbuf, SUBSCRIBER_NUMBER_LENGTH);
+}
+
+static void getRandomServerId(ServerId *serverId)
+{
+ *serverId = myRandom48(NO_OF_SERVERS);
+}
+
+static void getRandomChangedBy(ChangedBy changedBy)
+{
+ memset(changedBy, myRandom48(26)+'A', CHANGED_BY_LENGTH);
+ changedBy[CHANGED_BY_LENGTH] = 0;
+}
+
+static void getRandomChangedTime(ChangedTime changedTime)
+{
+ memset(changedTime, myRandom48(26)+'A', CHANGED_TIME_LENGTH);
+ changedTime[CHANGED_TIME_LENGTH] = 0;
+}
+
+static void clearTransaction(TransactionDefinition *trans)
+{
+ trans->benchTime = 0.0;
+ trans->count = 0;
+ trans->branchExecuted = 0;
+ trans->rollbackExecuted = 0;
+}
+
+static int listFull(SessionList *list)
+{
+ return(list->numberInList == SESSION_LIST_LENGTH);
+}
+
+static int listEmpty(SessionList *list)
+{
+ return(list->numberInList == 0);
+}
+
+static void insertSession(SessionList *list,
+ SubscriberNumber number,
+ ServerId serverId)
+{
+ SessionElement *e;
+ if( listFull(list) ) return;
+
+ e = &list->list[list->writeIndex];
+
+ strcpy(e->subscriberNumber, number);
+ e->serverId = serverId;
+
+ list->writeIndex = (list->writeIndex + 1) % SESSION_LIST_LENGTH;
+ list->numberInList++;
+
+if( list->numberInList > maxsize )
+maxsize = list->numberInList;
+}
+
+static SessionElement *getNextSession(SessionList *list)
+{
+ if( listEmpty(list) ) return(0);
+
+ return(&list->list[list->readIndex]);
+}
+
+static void deleteSession(SessionList *list)
+{
+ if( listEmpty(list) ) return;
+
+ list->readIndex = (list->readIndex + 1) % SESSION_LIST_LENGTH;
+ list->numberInList--;
+}
+
+static void initGeneratorStatistics(GeneratorStatistics *gen)
+{
+ int i;
+
+ if( initSequence(&gen->transactionSequence,
+ transactionDefinition) != 0 ) {
+ printf("could not set the transaction types\n");
+ exit(0);
+ }
+
+ if( initSequence(&gen->rollbackSequenceT4,
+ rollbackDefinition) != 0 ) {
+ printf("could not set the rollback sequence\n");
+ exit(0);
+ }
+
+ if( initSequence(&gen->rollbackSequenceT5,
+ rollbackDefinition) != 0 ) {
+ printf("could not set the rollback sequence\n");
+ exit(0);
+ }
+
+ for(i = 0; i < NUM_TRANSACTION_TYPES; i++ )
+ clearTransaction(&gen->transactions[i]);
+
+ gen->totalTransactions = 0;
+
+ gen->activeSessions.numberInList = 0;
+ gen->activeSessions.readIndex = 0;
+ gen->activeSessions.writeIndex = 0;
+}
+
+
+static void doOneTransaction(UserHandle *uh, GeneratorStatistics *gen)
+{
+ unsigned int transactionType;
+
+ transactionType = getNextRandom(&gen->transactionSequence);
+
+ switch(transactionType) {
+ case 1:
+ doTransaction_T1(uh, gen);
+ break;
+ case 2:
+ doTransaction_T2(uh, gen);
+ break;
+ case 3:
+ doTransaction_T3(uh, gen);
+ break;
+ case 4:
+ doTransaction_T4(uh, gen);
+ break;
+ case 5:
+ doTransaction_T5(uh, gen);
+ break;
+ default:
+ printf("Unknown transaction type: %d\n", transactionType);
+ }
+
+ gen->totalTransactions++;
+}
+
+static void doTransaction_T1(UserHandle *uh, GeneratorStatistics *gen)
+{
+ SubscriberNumber number;
+ Location new_location;
+ ChangedBy changed_by;
+ ChangedTime changed_time;
+
+ double start_time;
+ double end_time;
+ double transaction_time;
+
+ unsigned int tid = 0;
+
+ /*----------------*/
+ /* Init arguments */
+ /*----------------*/
+ getRandomSubscriberNumber(number);
+ getRandomChangedBy(changed_by);
+ getRandomChangedTime(changed_time);
+ new_location = changed_by[0];
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ start_time = userGetTimeSync();
+ userTransaction_T1(uh,
+ number,
+ new_location,
+ changed_by,
+ changed_time);
+ end_time = userGetTimeSync();
+
+ /*-------------------*/
+ /* Update Statistics */
+ /*-------------------*/
+ transaction_time = end_time - start_time;
+ gen->transactions[tid].benchTime += transaction_time;
+ gen->transactions[tid].count++;
+}
+
+static void doTransaction_T2(UserHandle *uh, GeneratorStatistics *gen)
+{
+ SubscriberNumber number;
+ Location new_location;
+ ChangedBy changed_by;
+ ChangedTime changed_time;
+ SubscriberName subscriberName;
+
+ double start_time;
+ double end_time;
+ double transaction_time;
+
+ unsigned int tid = 1;
+
+ /*----------------*/
+ /* Init arguments */
+ /*----------------*/
+ getRandomSubscriberNumber(number);
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ start_time = userGetTimeSync();
+ userTransaction_T2(uh,
+ number,
+ &new_location,
+ changed_by,
+ changed_time,
+ subscriberName);
+ end_time = userGetTimeSync();
+
+ /*-------------------*/
+ /* Update Statistics */
+ /*-------------------*/
+ transaction_time = end_time - start_time;
+ gen->transactions[tid].benchTime += transaction_time;
+ gen->transactions[tid].count++;
+}
+
+static void doTransaction_T3(UserHandle *uh, GeneratorStatistics *gen)
+{
+ SubscriberNumber number;
+ ServerId serverId;
+ ServerBit serverBit;
+ SessionDetails sessionDetails;
+ unsigned int branchExecuted;
+ SessionElement *se;
+
+ double start_time;
+ double end_time;
+ double transaction_time;
+
+ unsigned int tid = 2;
+
+ /*----------------*/
+ /* Init arguments */
+ /*----------------*/
+ se = getNextSession(&gen->activeSessions);
+ if( se ) {
+ strcpy(number, se->subscriberNumber);
+ serverId = se->serverId;
+ }
+ else {
+ getRandomSubscriberNumber(number);
+ getRandomServerId(&serverId);
+ }
+
+ serverBit = 1 << serverId;
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ start_time = userGetTimeSync();
+ userTransaction_T3(uh,
+ number,
+ serverId,
+ serverBit,
+ sessionDetails,
+ &branchExecuted);
+ end_time = userGetTimeSync();
+
+ /*-------------------*/
+ /* Update Statistics */
+ /*-------------------*/
+ transaction_time = end_time - start_time;
+ gen->transactions[tid].benchTime += transaction_time;
+ gen->transactions[tid].count++;
+
+ if(branchExecuted)
+ gen->transactions[tid].branchExecuted++;
+}
+
+static void doTransaction_T4(UserHandle *uh, GeneratorStatistics *gen)
+{
+ SubscriberNumber number;
+ ServerId serverId;
+ ServerBit serverBit;
+ SessionDetails sessionDetails;
+ unsigned int branchExecuted;
+ unsigned int rollback;
+
+ double start_time;
+ double end_time;
+ double transaction_time;
+
+ unsigned int tid = 3;
+
+ /*----------------*/
+ /* Init arguments */
+ /*----------------*/
+ getRandomSubscriberNumber(number);
+ getRandomServerId(&serverId);
+
+ serverBit = 1 << serverId;
+ rollback = getNextRandom(&gen->rollbackSequenceT4);
+
+ memset(sessionDetails, myRandom48(26)+'A', SESSION_DETAILS_LENGTH);
+ sessionDetails[SESSION_DETAILS_LENGTH] = 0;
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ start_time = userGetTimeSync();
+ userTransaction_T4(uh,
+ number,
+ serverId,
+ serverBit,
+ sessionDetails,
+ rollback,
+ &branchExecuted);
+ end_time = userGetTimeSync();
+
+ /*-------------------*/
+ /* Update Statistics */
+ /*-------------------*/
+ transaction_time = end_time - start_time;
+ gen->transactions[tid].benchTime += transaction_time;
+ gen->transactions[tid].count++;
+
+ if(branchExecuted)
+ gen->transactions[tid].branchExecuted++;
+ if(rollback)
+ gen->transactions[tid].rollbackExecuted++;
+
+ if( branchExecuted && !rollback ) {
+ insertSession(&gen->activeSessions, number, serverId);
+ }
+}
+
+static void doTransaction_T5(UserHandle *uh, GeneratorStatistics *gen)
+{
+ SubscriberNumber number;
+ ServerId serverId;
+ ServerBit serverBit;
+ unsigned int branchExecuted;
+ unsigned int rollback;
+ SessionElement *se;
+
+ double start_time;
+ double end_time;
+ double transaction_time;
+
+ unsigned int tid = 4;
+
+ /*----------------*/
+ /* Init arguments */
+ /*----------------*/
+ se = getNextSession(&gen->activeSessions);
+ if( se ) {
+ strcpy(number, se->subscriberNumber);
+ serverId = se->serverId;
+ }
+ else {
+ getRandomSubscriberNumber(number);
+ getRandomServerId(&serverId);
+ }
+
+ serverBit = 1 << serverId;
+ rollback = getNextRandom(&gen->rollbackSequenceT5);
+
+ /*-----------------*/
+ /* Run transaction */
+ /*-----------------*/
+ start_time = userGetTimeSync();
+ userTransaction_T5(uh,
+ number,
+ serverId,
+ serverBit,
+ rollback,
+ &branchExecuted);
+ end_time = userGetTimeSync();
+
+ /*-------------------*/
+ /* Update Statistics */
+ /*-------------------*/
+ transaction_time = end_time - start_time;
+ gen->transactions[tid].benchTime += transaction_time;
+ gen->transactions[tid].count++;
+
+ if(branchExecuted)
+ gen->transactions[tid].branchExecuted++;
+ if(rollback)
+ gen->transactions[tid].rollbackExecuted++;
+
+ if( se && !rollback) {
+ deleteSession(&gen->activeSessions);
+ }
+}
+
+/***************************************************************
+****************************************************************
+* P U B L I C F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+
+void dbGenerator(UserHandle *uh, ThreadData *data)
+{
+ GeneratorStatistics rg_warmUp;
+ GeneratorStatistics rg_coolDown;
+ GeneratorStatistics *st;
+ double periodStop;
+ double benchTimeStart;
+ double benchTimeEnd;
+ int i;
+
+ myRandom48Init(data->randomSeed);
+
+ initGeneratorStatistics(&rg_warmUp);
+ initGeneratorStatistics(&data->generator);
+ initGeneratorStatistics(&rg_coolDown);
+
+ /*----------------*/
+ /* warm up period */
+ /*----------------*/
+ periodStop = userGetTimeSync() + (double)data->warmUpSeconds;
+ while(userGetTimeSync() < periodStop){
+ doOneTransaction(uh, &rg_warmUp);
+ }
+
+ /*-------------------------*/
+ /* normal benchmark period */
+ /*-------------------------*/
+ benchTimeStart = userGetTimeSync();
+
+ if( data->numTransactions > 0 ) {
+ for(i = 0; i < data->numTransactions; i++)
+ doOneTransaction(uh, &data->generator);
+ }
+ else {
+ periodStop = benchTimeStart + (double)data->testSeconds;
+ while(userGetTimeSync() < periodStop)
+ doOneTransaction(uh, &data->generator);
+ }
+
+ benchTimeEnd = userGetTimeSync();
+
+ /*------------------*/
+ /* cool down period */
+ /*------------------*/
+ periodStop = benchTimeEnd + data->coolDownSeconds;
+ while(userGetTimeSync() < periodStop){
+ doOneTransaction(uh, &rg_coolDown);
+ }
+
+ /*---------------------------------------------------------*/
+ /* add the times for all transaction for inner loop timing */
+ /*---------------------------------------------------------*/
+ st = &data->generator;
+ st->innerLoopTime = 0.0;
+ for(i = 0 ; i < NUM_TRANSACTION_TYPES; i++) {
+ st->innerLoopTime += st->transactions[i].benchTime;
+ st->transactions[i].tps = getTps(st->transactions[i].count,
+ st->transactions[i].benchTime);
+ }
+
+ st->outerLoopTime = benchTimeEnd - benchTimeStart;
+ st->outerTps = getTps(st->totalTransactions, st->outerLoopTime);
+ st->innerTps = getTps(st->totalTransactions, st->innerLoopTime);
+
+ /* printf("maxsize = %d\n",maxsize); */
+}
diff --git a/ndb/test/ndbapi/lmc-bench/src/generator/dbGenerator.h b/ndb/test/ndbapi/lmc-bench/src/generator/dbGenerator.h
new file mode 100644
index 00000000000..824688b6cf9
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/generator/dbGenerator.h
@@ -0,0 +1,61 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBGENERATOR_H
+#define DBGENERATOR_H
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include "testData.h"
+#include "userInterface.h"
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+extern void dbGenerator(UserHandle *uh, ThreadData *data);
+
+
+#ifdef __cplusplus
+}
+#endif
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+
+
+#endif /* DBGENERATOR_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/generator/mainGenerator.c b/ndb/test/ndbapi/lmc-bench/src/generator/mainGenerator.c
new file mode 100644
index 00000000000..6ddf0a47775
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/generator/mainGenerator.c
@@ -0,0 +1,327 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+#include <pthread.h>
+
+#include <NdbHost.h>
+#include <NdbSleep.h>
+#include <NdbThread.h>
+#include <NdbMain.h>
+
+#include "userInterface.h"
+#include "dbGenerator.h"
+
+
+static int numProcesses;
+static int numTransactions;
+static int numSeconds;
+static int numWarmSeconds;
+static char *testDbName;
+
+static ThreadData data[100];
+
+typedef struct {
+ pthread_t threadId;
+ int waitSeconds;
+ int toExit;
+}CheckpointData;
+
+static void usage(char *prog)
+{
+ char *progname;
+
+ /*--------------------------------------------*/
+ /* Get the name of the program (without path) */
+ /*--------------------------------------------*/
+ progname = strrchr(prog, '/');
+
+ if (progname == 0)
+ progname = prog;
+ else
+ ++progname;
+
+ fprintf(stderr,
+ "Usage: %s [-db <name>] [-proc <num>] [-transactions <num>] [-time <num>]\n"
+ " -db <name> Specifies the database name\n"
+ " default = '%s'\n"
+ " -proc <num> Specifies that <num> is the number of\n"
+ " concurrent processes. The default is 1.\n"
+ " -transactions <num> Specifies that <num> transactions will be\n"
+ " performed. The default is to do a specific time interval\n"
+ " -time <num> Specifies that the test will run for <num> sec.\n"
+ " The default is 10 sec\n"
+ " -warm <num> Specifies the warm-up/cooldown period of <num> sec.\n"
+ " The default is 10 sec\n",
+ progname, DEFAULTDB);
+ exit(1);
+}
+
+static void parse_args(int argc,char **argv)
+{
+ int i;
+
+ testDbName = DEFAULTDB;
+ numProcesses = 1;
+ numTransactions = 0;
+ numSeconds = 10;
+ numWarmSeconds = 10;
+
+ i = 1;
+ while (i < argc){
+ if (strcmp("-db",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ usage(argv[0]);
+ exit(1);
+ }
+ testDbName = argv[i + 1];
+ i += 2;
+ }
+ else if (strcmp("-proc",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ usage(argv[0]);
+ exit(1);
+ }
+ if (sscanf(argv[i+1], "%d", &numProcesses) == -1 ||
+ numProcesses <= 0 || numProcesses > 99) {
+ fprintf(stderr, "-proc flag requires a positive integer argument [1..99]\n");
+ usage(argv[0]);
+ exit(1);
+ }
+ i += 2;
+ }
+ else if (strcmp("-transactions",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ usage(argv[0]);
+ exit(1);
+ }
+ if (sscanf(argv[i+1], "%d", &numTransactions) == -1 ||
+ numTransactions < 0) {
+ fprintf(stderr, "-transactions flag requires a positive integer argument\n");
+ usage(argv[0]);
+ exit(1);
+ }
+ i += 2;
+ }
+ else if (strcmp("-time",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ usage(argv[0]);
+ exit(1);
+ }
+ if (sscanf(argv[i+1], "%d", &numSeconds) == -1 ||
+ numSeconds < 0) {
+ fprintf(stderr, "-time flag requires a positive integer argument\n");
+ usage(argv[0]);
+ exit(1);
+ }
+ i += 2;
+ }
+ else if (strcmp("-warm",argv[i]) == 0) {
+ if (i + 1 >= argc) {
+ usage(argv[0]);
+ exit(1);
+ }
+ if (sscanf(argv[i+1], "%d", &numWarmSeconds) == -1 ||
+ numWarmSeconds < 0) {
+ fprintf(stderr, "-warm flag requires a positive integer argument\n");
+ usage(argv[0]);
+ exit(1);
+ }
+ i += 2;
+ }
+ else
+ usage(argv[0]);
+ }
+}
+
+static void print_transaction(const char *header,
+ unsigned long totalCount,
+ TransactionDefinition *trans,
+ unsigned int printBranch,
+ unsigned int printRollback)
+{
+ double f;
+
+ printf(" %s: %d (%.2f%%) Time: %.4f sec TPS = %.0f\n",
+ header,
+ trans->count,
+ (double)trans->count / (double)totalCount * 100.0,
+ trans->benchTime,
+ trans->tps);
+
+ if( printBranch ){
+ if( trans->count == 0 )
+ f = 0.0;
+ else
+ f = (double)trans->branchExecuted / (double)trans->count * 100.0;
+ printf(" Branches Executed: %d (%.2f%%)\n", trans->branchExecuted, f);
+ }
+
+ if( printRollback ){
+ if( trans->count == 0 )
+ f = 0.0;
+ else
+ f = (double)trans->rollbackExecuted / (double)trans->count * 100.0;
+ printf(" Rollback Executed: %d (%.2f%%)\n", trans->rollbackExecuted, f);
+ }
+}
+
+void print_stats_sync(const char *title,
+ unsigned int length,
+ unsigned int transactionFlag,
+ GeneratorStatistics *gen,
+ int numProc)
+{
+ int i;
+ char buf[10];
+ char name[100];
+
+ name[0] = 0;
+ NdbHost_GetHostName(name);
+
+ printf("\n------ %s ------\n",title);
+ printf("Length : %d %s\n",
+ length,
+ transactionFlag ? "Transactions" : "sec");
+ printf("Processor : %s\n", name);
+ printf("Number of Proc: %d\n",numProc);
+ printf("\n");
+
+ if( gen->totalTransactions == 0 ) {
+ printf(" No Transactions for this test\n");
+ }
+ else {
+ for(i = 0; i < 5; i++) {
+ sprintf(buf, "T%d",i+1);
+ print_transaction(buf,
+ gen->totalTransactions,
+ &gen->transactions[i],
+ i >= 2,
+ i >= 3 );
+ }
+
+ printf("\n");
+ printf(" Overall Statistics:\n");
+ printf(" Transactions: %d\n", gen->totalTransactions);
+ printf(" Inner : %.0f TPS\n",gen->innerTps);
+ printf(" Outer : %.0f TPS\n",gen->outerTps);
+ printf("\n");
+ }
+}
+
+static void *threadRoutine(void *arg)
+{
+ UserHandle *uh;
+ ThreadData *data = (ThreadData *)arg;
+
+ uh = userDbConnect(0, testDbName);
+ NdbSleep_MilliSleep(data->threadId);
+ dbGenerator(uh,data);
+ userDbDisconnect(uh);
+
+ pthread_exit(0);
+ return(0);
+}
+
+NDB_COMMAND(DbGenerator, "DbGenerator", "DbGenerator", "DbGenerator", 16384)
+{
+ int i;
+ int j;
+ GeneratorStatistics stats;
+ GeneratorStatistics *p;
+ CheckpointData cd;
+
+ parse_args(argc,argv);
+
+ printf("\nStarting Test with %d process(es) for %d %s\n",
+ numProcesses,
+ numTransactions ? numTransactions : numSeconds,
+ numTransactions ? "Transactions" : "sec");
+ printf(" WarmUp/coolDown = %d sec\n", numWarmSeconds);
+
+ /*
+ cd.waitSeconds = 300;
+ cd.toExit = 0;
+ pthread_create(&cd.threadId, 0, checkpointRoutine, &cd);
+ */
+
+ for(i = 0; i < numProcesses; i++) {
+ data[i].warmUpSeconds = numWarmSeconds;
+ data[i].testSeconds = numSeconds;
+ data[i].coolDownSeconds = numWarmSeconds;
+ data[i].numTransactions = numTransactions;
+ data[i].randomSeed = time(0)+i;
+ j = pthread_create(&data[i].threadId, 0, threadRoutine, &data[i]);
+ if(j != 0){
+ perror("Failed to create thread");
+ }
+ }
+
+ /*--------------------------------*/
+ /* Wait for all processes to exit */
+ /*--------------------------------*/
+ for(i = 0; i < numProcesses; i++)
+ pthread_join(data[i].threadId, 0);
+
+ printf("All threads have finished\n");
+
+ cd.toExit = 1;
+
+ /*-------------------------------------------*/
+ /* Clear all structures for total statistics */
+ /*-------------------------------------------*/
+ stats.totalTransactions = 0;
+ stats.outerTps = 0.0;
+ stats.innerTps = 0.0;
+
+ for(i = 0; i < NUM_TRANSACTION_TYPES; i++ ) {
+ stats.transactions[i].benchTime = 0.0;
+ stats.transactions[i].count = 0;
+ stats.transactions[i].tps = 0.0;
+ stats.transactions[i].branchExecuted = 0;
+ stats.transactions[i].rollbackExecuted = 0;
+ }
+
+ /*--------------------------------*/
+ /* Add the values for all Threads */
+ /*--------------------------------*/
+ for(i = 0; i < numProcesses; i++) {
+ p = &data[i].generator;
+
+ stats.totalTransactions += p->totalTransactions;
+ stats.outerTps += p->outerTps;
+ stats.innerTps += p->innerTps;
+
+ for(j = 0; j < NUM_TRANSACTION_TYPES; j++ ) {
+ stats.transactions[j].benchTime += p->transactions[j].benchTime;
+ stats.transactions[j].count += p->transactions[j].count;
+ stats.transactions[j].tps += p->transactions[j].tps;
+ stats.transactions[j].branchExecuted += p->transactions[j].branchExecuted;
+ stats.transactions[j].rollbackExecuted += p->transactions[j].rollbackExecuted;
+ }
+ }
+
+ print_stats_sync("Test Results",
+ numTransactions ? numTransactions : numSeconds,
+ numTransactions ? 1 : 0,
+ &stats,
+ numProcesses);
+
+ return(0);
+}
diff --git a/ndb/test/ndbapi/lmc-bench/src/include/testData.h b/ndb/test/ndbapi/lmc-bench/src/include/testData.h
new file mode 100644
index 00000000000..863c230502b
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/include/testData.h
@@ -0,0 +1,103 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef TESTDATA_H
+#define TESTDATA_H
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include "testDefinitions.h"
+#include <random.h>
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+#define NUM_TRANSACTION_TYPES 5
+#define SESSION_LIST_LENGTH 1000
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+typedef struct {
+ SubscriberNumber subscriberNumber;
+ ServerId serverId;
+} SessionElement;
+
+typedef struct {
+ SessionElement list[SESSION_LIST_LENGTH];
+ unsigned int readIndex;
+ unsigned int writeIndex;
+ unsigned int numberInList;
+} SessionList;
+
+typedef struct {
+ double benchTime;
+ unsigned int count;
+ double tps;
+ unsigned int branchExecuted;
+ unsigned int rollbackExecuted;
+}TransactionDefinition;
+
+typedef struct {
+ RandomSequence transactionSequence;
+ RandomSequence rollbackSequenceT4;
+ RandomSequence rollbackSequenceT5;
+
+ TransactionDefinition transactions[NUM_TRANSACTION_TYPES];
+
+ unsigned int totalTransactions;
+
+ double innerLoopTime;
+ double innerTps;
+
+ double outerLoopTime;
+ double outerTps;
+
+ SessionList activeSessions;
+} GeneratorStatistics;
+
+typedef struct {
+ unsigned long threadId;
+ unsigned long randomSeed;
+
+ unsigned int warmUpSeconds;
+ unsigned int testSeconds;
+ unsigned int coolDownSeconds;
+ unsigned int numTransactions;
+
+ GeneratorStatistics generator;
+}ThreadData;
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+
+
+#endif /* TESTDATA_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/include/userInterface.h b/ndb/test/ndbapi/lmc-bench/src/include/userInterface.h
new file mode 100644
index 00000000000..b70ded87756
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/include/userInterface.h
@@ -0,0 +1,128 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBINTERFACE_H
+#define DBINTERFACE_H
+
+/***************************************************************/
+/* I N C L U D E D F I L E S */
+/***************************************************************/
+
+#include "testDefinitions.h"
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+/*-----------------------*/
+/* Default Database Name */
+/*-----------------------*/
+#define DEFAULTDB "TestDbClient"
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+typedef struct {
+ struct Ndb * pNDB;
+ struct NdbConnection * pCurrTrans;
+} UserHandle;
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+extern double userGetTimeSync(void);
+
+extern void userCheckpoint(UserHandle *uh);
+
+extern UserHandle *userDbConnect(uint32 createDb, char *dbName);
+extern void userDbDisconnect(UserHandle *uh);
+
+extern int userDbInsertServer(UserHandle *uh,
+ ServerId serverId,
+ SubscriberSuffix suffix,
+ ServerName name);
+
+extern int userDbInsertSubscriber(UserHandle *uh,
+ SubscriberNumber number,
+ uint32 groupId,
+ SubscriberName name);
+
+extern int userDbInsertGroup(UserHandle *uh,
+ GroupId groupId,
+ GroupName name,
+ Permission allowRead,
+ Permission allowInsert,
+ Permission allowDelete);
+
+extern int userDbCommit(UserHandle *uh);
+extern int userDbRollback(UserHandle *uh);
+
+extern void userTransaction_T1(UserHandle *uh,
+ SubscriberNumber number,
+ Location new_location,
+ ChangedBy changed_by,
+ ChangedTime changed_time);
+
+extern void userTransaction_T2(UserHandle *uh,
+ SubscriberNumber number,
+ Location *new_location,
+ ChangedBy changed_by,
+ ChangedTime changed_time,
+ SubscriberName subscriberName);
+
+extern void userTransaction_T3(UserHandle *uh,
+ SubscriberNumber number,
+ ServerId server_id,
+ ServerBit server_bit,
+ SessionDetails session_details,
+ unsigned int *branch_executed);
+
+extern void userTransaction_T4(UserHandle *uh,
+ SubscriberNumber number,
+ ServerId server_id,
+ ServerBit server_bit,
+ SessionDetails session_details,
+ unsigned int do_rollback,
+ unsigned int *branch_executed);
+
+extern void userTransaction_T5(UserHandle *uh,
+ SubscriberNumber number,
+ ServerId server_id,
+ ServerBit server_bit,
+ unsigned int do_rollback,
+ unsigned int *branch_executed);
+
+
+#ifdef __cplusplus
+}
+#endif
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+#endif /* DBINTERFACE_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/makevars.linux b/ndb/test/ndbapi/lmc-bench/src/makevars.linux
new file mode 100644
index 00000000000..a933669cfe7
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/makevars.linux
@@ -0,0 +1,6 @@
+PROJECT_TOP = /home/lmcritr/ecurlmc/users/lmcritr/dbBenchmark
+
+CFLAGS = -Wall -Wstrict-prototypes -O2 -I/opt/TimesTen4.1/32/include/ -I../include
+LDFLAGS = -L/opt/TimesTen4.1/32/lib -Wl,-rpath,/opt/TimesTen4.1/32/lib
+LIBS = -ltten -ldl
+LIBSCS = -lttclient -ldl
diff --git a/ndb/test/ndbapi/lmc-bench/src/makevars.sparc b/ndb/test/ndbapi/lmc-bench/src/makevars.sparc
new file mode 100644
index 00000000000..57ab8bf982f
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/makevars.sparc
@@ -0,0 +1,15 @@
+
+include $(UAS_TOP)/Defs.mk
+
+LINK.CC = CC
+CC := /opt/as/forte6/SUNWspro/bin/cc
+export CC
+
+NDB_LIB = -L$(UAS_TOP)/API -lNDB_API \
+ -L$(UAS_OSPACE_LOC)/lib -lospace \
+ -lrt
+
+CFLAGS = -xO3 -I../include -mt
+LDFLAGS = $(NDB_LIB) -lpthread
+LIBS =
+LIBSCS =
diff --git a/ndb/test/ndbapi/lmc-bench/src/populator/Makefile b/ndb/test/ndbapi/lmc-bench/src/populator/Makefile
new file mode 100644
index 00000000000..2107c948843
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/populator/Makefile
@@ -0,0 +1,15 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := DbCreate
+BIN_TARGET_ARCHIVES := lmc_User
+
+CCFLAGS_LOC:= -I../include -I../../include
+
+SOURCES := \
+ mainPopulate.c\
+ dbPopulate.c
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/populator/dbPopulate.c b/ndb/test/ndbapi/lmc-bench/src/populator/dbPopulate.c
new file mode 100644
index 00000000000..9f8629ec1f0
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/populator/dbPopulate.c
@@ -0,0 +1,246 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include "userInterface.h"
+
+#include "dbPopulate.h"
+#include <NdbOut.hpp>
+#include <random.h>
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+static void getRandomSubscriberData(int subscriberNo,
+ SubscriberNumber number,
+ SubscriberName name);
+
+static void populate(char *title,
+ int count,
+ void (*func)(UserHandle*,int),
+ UserHandle *uh);
+
+static void populateServers(UserHandle *uh, int count);
+static void populateSubscribers(UserHandle *uh, int count);
+static void populateGroups(UserHandle *uh, int count);
+
+/***************************************************************
+* L O C A L D A T A *
+***************************************************************/
+
+static SequenceValues permissionsDefinition[] = {
+ {90, 1},
+ {10, 0},
+ {0, 0}
+};
+
+/***************************************************************
+* P U B L I C D A T A *
+***************************************************************/
+
+
+/***************************************************************
+****************************************************************
+* L O C A L F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+static void getRandomSubscriberData(int subscriberNo,
+ SubscriberNumber number,
+ SubscriberName name)
+{
+ char sbuf[SUBSCRIBER_NUMBER_LENGTH + 1];
+ sprintf(sbuf, "%.*d", SUBSCRIBER_NUMBER_LENGTH, subscriberNo);
+ memcpy(number, sbuf, SUBSCRIBER_NUMBER_LENGTH);
+
+ memset(name, myRandom48(26)+'A', SUBSCRIBER_NAME_LENGTH);
+}
+
+static void populate(char *title,
+ int count,
+ void (*func)(UserHandle*, int),
+ UserHandle *uh)
+{
+ ndbout_c("Populating %d '%s' ... ",count, title);
+ /* fflush(stdout); */
+ func(uh,count);
+ ndbout_c("done");
+}
+
+static void populateServers(UserHandle *uh, int count)
+{
+ int i, j;
+ int len;
+ char tmp[80];
+ int suffix_length = 1;
+ ServerName serverName;
+ SubscriberSuffix suffix;
+
+ int commitCount = 0;
+
+ for(i = 0; i < SUBSCRIBER_NUMBER_SUFFIX_LENGTH; i++)
+ suffix_length *= 10;
+
+ for(i = 0; i < count; i++) {
+ sprintf(tmp, "-Server %d-", i);
+
+ len = strlen(tmp);
+ for(j = 0; j < SERVER_NAME_LENGTH; j++){
+ serverName[j] = tmp[j % len];
+ }
+ /* serverName[j] = 0; not null-terminated */
+
+ for(j = 0; j < suffix_length; j++){
+ char sbuf[SUBSCRIBER_NUMBER_SUFFIX_LENGTH + 1];
+ sprintf(sbuf, "%.*d", SUBSCRIBER_NUMBER_SUFFIX_LENGTH, j);
+ memcpy(suffix, sbuf, SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+ userDbInsertServer(uh, i, suffix, serverName);
+ commitCount ++;
+ if((commitCount % OP_PER_TRANS) == 0)
+ userDbCommit(uh);
+ }
+ }
+ if((commitCount % OP_PER_TRANS) != 0)
+ userDbCommit(uh);
+}
+
+static void populateSubscribers(UserHandle *uh, int count)
+{
+ SubscriberNumber number;
+ SubscriberName name;
+ int i, j, k;
+ int res;
+
+ SequenceValues values[NO_OF_GROUPS+1];
+ RandomSequence seq;
+
+ for(i = 0; i < NO_OF_GROUPS; i++) {
+ values[i].length = 1;
+ values[i].value = i;
+ }
+
+ values[i].length = 0;
+ values[i].value = 0;
+
+ if( initSequence(&seq, values) != 0 ) {
+ ndbout_c("could not set the sequence of random groups");
+ exit(0);
+ }
+
+#define RETRIES 25
+
+ for(i = 0; i < count; i+= OP_PER_TRANS) {
+ for(j = 0; j<RETRIES; j++){
+ for(k = 0; k<OP_PER_TRANS && i+k < count; k++){
+ getRandomSubscriberData(i+k, number, name);
+ userDbInsertSubscriber(uh, number, getNextRandom(&seq), name);
+ }
+ res = userDbCommit(uh);
+ if(res == 0)
+ break;
+ if(res != 1){
+ ndbout_c("Terminating");
+ exit(0);
+ }
+ }
+ if(j == RETRIES){
+ ndbout_c("Terminating");
+ exit(0);
+ }
+ }
+}
+
+static void populateGroups(UserHandle *uh, int count)
+{
+ int i;
+ int j;
+ int len;
+ RandomSequence seq;
+ Permission allow[NO_OF_GROUPS];
+ ServerBit serverBit;
+ GroupName groupName;
+ char tmp[80];
+ int commitCount = 0;
+
+ if( initSequence(&seq, permissionsDefinition) != 0 ) {
+ ndbout_c("could not set the sequence of random permissions");
+ exit(0);
+ }
+
+ for(i = 0; i < NO_OF_GROUPS; i++)
+ allow[i] = 0;
+
+ for(i = 0; i < NO_OF_SERVERS; i++) {
+ serverBit = 1 << i;
+
+ for(j = 0; j < NO_OF_GROUPS; j++ ) {
+ if( getNextRandom(&seq) )
+ allow[j] |= serverBit;
+ }
+ }
+
+ for(i = 0; i < NO_OF_GROUPS; i++) {
+ sprintf(tmp, "-Group %d-", i);
+
+ len = strlen(tmp);
+
+ for(j = 0; j < GROUP_NAME_LENGTH; j++) {
+ groupName[j] = tmp[j % len];
+ }
+ /* groupName[j] = 0; not null-terminated */
+
+ userDbInsertGroup(uh,
+ i,
+ groupName,
+ allow[i],
+ allow[i],
+ allow[i]);
+ commitCount ++;
+ if((commitCount % OP_PER_TRANS) == 0)
+ userDbCommit(uh);
+ }
+ if((commitCount % OP_PER_TRANS) != 0)
+ userDbCommit(uh);
+}
+
+/***************************************************************
+****************************************************************
+* P U B L I C F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+void dbPopulate(UserHandle *uh)
+{
+ populate("servers", NO_OF_SERVERS, populateServers, uh);
+ populate("subscribers", NO_OF_SUBSCRIBERS, populateSubscribers, uh);
+ populate("groups", NO_OF_GROUPS, populateGroups, uh);
+}
diff --git a/ndb/test/ndbapi/lmc-bench/src/populator/dbPopulate.h b/ndb/test/ndbapi/lmc-bench/src/populator/dbPopulate.h
new file mode 100644
index 00000000000..1916720e141
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/populator/dbPopulate.h
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef DBPOPULATE_H
+#define DBPOPULATE_H
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include "userInterface.h"
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+extern void dbPopulate(UserHandle *uh);
+
+#ifdef __cplusplus
+}
+#endif
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+
+
+#endif /* DBPOPULATE_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/populator/mainPopulate.c b/ndb/test/ndbapi/lmc-bench/src/populator/mainPopulate.c
new file mode 100644
index 00000000000..9dde902d006
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/populator/mainPopulate.c
@@ -0,0 +1,78 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#include "userInterface.h"
+#include "dbPopulate.h"
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <random.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+int useTableLogging;
+int useIndexTables;
+#ifdef __cplusplus
+}
+#endif
+
+
+static
+void usage(const char *prog)
+{
+
+ ndbout_c(
+ "Usage: %s [-l]\n"
+ " -l Use logging and checkpointing on tables\n",
+ " -i Use index tables\n",
+ prog);
+
+ exit(1);
+}
+
+NDB_COMMAND(DbCreate, "DbCreate", "DbCreate", "DbCreate", 16384)
+{
+ int i;
+ UserHandle *uh;
+
+ useTableLogging = useIndexTables = 0;
+
+ for(i = 1; i<argc; i++){
+ if(strcmp(argv[i], "-l") == 0){
+ useTableLogging = 1;
+ } else if(strcmp(argv[i], "-i") == 0){
+ useIndexTables = 1;
+ } else {
+ usage(argv[0]);
+ return 0;
+ }
+ }
+
+ ndbout_c("Using %s tables and %s key storage",
+ useTableLogging ? "logging" : "temporary",
+ useIndexTables ? "index" : "normal");
+
+ myRandom48Init(0x3e6f);
+
+ uh = userDbConnect(1, 0);
+ dbPopulate(uh);
+ userDbDisconnect(uh);
+ return(0);
+}
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/Makefile b/ndb/test/ndbapi/lmc-bench/src/user/Makefile
new file mode 100644
index 00000000000..9bf229ac84c
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+ARCHIVE_TARGET := lmc_User
+
+SOURCES := userInterface.C
+
+CCFLAGS_LOC = -I../include -I../../include
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/localDbPrepare.c b/ndb/test/ndbapi/lmc-bench/src/user/localDbPrepare.c
new file mode 100644
index 00000000000..ca8a64ab59c
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/localDbPrepare.c
@@ -0,0 +1,648 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <stdio.h>
+
+#include "userInterface.h"
+#include "userHandle.h"
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+
+/***************************************************************
+* L O C A L D A T A *
+***************************************************************/
+
+/*----------------*/
+/* Transaction T1 */
+/*----------------*/
+static char *update_subscriber_stmnt = "update subscriber set \
+location = ?,changedBy = ?, changedTime = ? where subscriberNumber = ?";
+
+/*----------------*/
+/* Transaction T2 */
+/*----------------*/
+static char *read_subscriber_stmnt = "select subscriberName,location,\
+changedBy,changedTime from subscriber where subscriberNumber = ? for update";
+
+/*----------------*/
+/* Transaction T3 */
+/*----------------*/
+static char *read_subscriber_session_stmnt = "select activeSessions,groupId,\
+changedBy,changedTime from subscriber where subscriberNumber = ? for update";
+
+static char *read_group_allowRead_stmnt = "select allowRead from userGroup \
+where groupId = ?";
+static char *read_group_allowInsert_stmnt = "select allowInsert from userGroup \
+where groupId = ?";
+static char *read_group_allowDelete_stmnt = "select allowDelete from userGroup \
+where groupId = ?";
+
+static char *read_session_details_stmnt = "select sessionData from userSession \
+where subscriberNumber = ? and serverId = ? for update";
+
+static char *update_noOfRead_stmnt = "update server \
+set noOfRead = noOfRead + 1 where serverId = ? and subscriberSuffix = ?";
+static char *update_noOfInsert_stmnt = "update server \
+set noOfInsert = noOfInsert + 1 where serverId = ? and subscriberSuffix = ?";
+static char *update_noOfDelete_stmnt = "update server \
+set noOfDelete = noOfDelete + 1 where serverId = ? and subscriberSuffix = ?";
+
+static char *insert_session_stmnt = "insert into userSession values (?,?,?)";
+
+static char *delete_session_stmnt = "delete from userSession \
+where subscriberNumber = ? and serverId = ?";
+
+static char *update_subscriber_session_stmnt = "update subscriber set \
+activeSessions = ? where subscriberNumber = ?";
+
+/***************************************************************
+* P U B L I C D A T A *
+***************************************************************/
+
+
+/***************************************************************
+****************************************************************
+* L O C A L F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+extern void handle_error(SQLHDBC hdbc,
+ SQLHENV henv,
+ SQLHSTMT hstmt,
+ SQLRETURN rc,
+ char *filename,
+ int lineno);
+
+/***************************************************************
+****************************************************************
+* P U B L I C F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+int localDbPrepare(UserHandle *uh)
+{
+ SQLRETURN rc;
+
+ if(!uh) return(-1);
+
+ /*-----------------------------*/
+ /* Update Subscriber Statement */
+ /*-----------------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->updateSubscriber.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate insert group statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->updateSubscriber.stmt,(SQLCHAR *) update_subscriber_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+/*
+handle_error(uh->hdbc, uh->henv, uh->updateSubscriber.stmt, rc, __FILE__, __LINE__);
+*/
+ printf("Unable to prepare update subscriber statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateSubscriber.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->updateSubscriber.values.location,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare update subscriber statement param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateSubscriber.stmt,
+ 2,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ CHANGED_BY_LENGTH+1,0,
+ uh->updateSubscriber.values.changedBy,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare update subscriber statement param 2\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateSubscriber.stmt,
+ 3,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ CHANGED_TIME_LENGTH+1,0,
+ uh->updateSubscriber.values.changedTime,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare update subscriber statement param 3\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateSubscriber.stmt,
+ 4,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_LENGTH+1,0,
+ uh->updateSubscriber.values.number,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare update subscriber statement param 3\n");
+ return(-1);
+ }
+
+ /*---------------------------*/
+ /* Read Subscriber Statement */
+ /*---------------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->readSubscriber.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate read subscriber statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->readSubscriber.stmt,(SQLCHAR *) read_subscriber_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read subscriber statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->readSubscriber.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_LENGTH+1,0,
+ uh->readSubscriber.values.number,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read subscriber statement param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readSubscriber.stmt, 1,
+ SQL_C_CHAR,
+ uh->readSubscriber.values.name, SUBSCRIBER_NAME_LENGTH+1,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 1 to read subscriber statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readSubscriber.stmt, 2,
+ SQL_C_DEFAULT,
+ &uh->readSubscriber.values.location, 1,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 2 to read subscriber statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readSubscriber.stmt, 3,
+ SQL_C_CHAR,
+ uh->readSubscriber.values.changedBy, CHANGED_BY_LENGTH+1,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 3 to read subscriber statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readSubscriber.stmt, 4,
+ SQL_C_CHAR,
+ uh->readSubscriber.values.changedTime, CHANGED_TIME_LENGTH+1,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 4 to read subscriber statement\n");
+ return(-1);
+ }
+
+ /*------------------------------------*/
+ /* Read Subscriber Sessions Statement */
+ /*------------------------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->readSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate read subscriber session statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->readSubscriberSession.stmt,(SQLCHAR *) read_subscriber_session_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read subscriber sessions statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->readSubscriberSession.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_LENGTH+1,0,
+ uh->readSubscriberSession.values.number,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read subscriber statement param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readSubscriberSession.stmt, 1,
+ SQL_C_DEFAULT,
+ &uh->readSubscriberSession.values.activeSessions, 0,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 1 to read subscriber sessions statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readSubscriberSession.stmt, 2,
+ SQL_C_DEFAULT,
+ &uh->readSubscriberSession.values.groupId, 0,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 2 to read subscriber sessions statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readSubscriberSession.stmt, 3,
+ SQL_C_CHAR,
+ uh->readSubscriberSession.values.changedBy, CHANGED_BY_LENGTH+1,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 3 to read subscriber sessions statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readSubscriberSession.stmt, 4,
+ SQL_C_CHAR,
+ uh->readSubscriberSession.values.changedTime, CHANGED_TIME_LENGTH+1,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 4 to read subscriber sessions statement\n");
+ return(-1);
+ }
+
+ /*--------------------------------*/
+ /* Read Group AllowRead Statement */
+ /*--------------------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->readGroupAllowRead.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate read subscriber session statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->readGroupAllowRead.stmt,(SQLCHAR *) read_group_allowRead_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read group allow read statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->readGroupAllowRead.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->readGroupAllowRead.values.groupId,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read allow read statement param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readGroupAllowRead.stmt, 1,
+ SQL_C_DEFAULT,
+ &uh->readGroupAllowRead.values.allowRead, 0,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 1 to read group allow read statement\n");
+ return(-1);
+ }
+
+ /*----------------------------------*/
+ /* Read Group AllowInsert Statement */
+ /*----------------------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->readGroupAllowInsert.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate read subscriber session statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->readGroupAllowInsert.stmt,(SQLCHAR *) read_group_allowInsert_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read group allow read statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->readGroupAllowInsert.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->readGroupAllowInsert.values.groupId,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read allow read statement param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readGroupAllowInsert.stmt, 1,
+ SQL_C_DEFAULT,
+ &uh->readGroupAllowInsert.values.allowInsert, 0,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 1 to read group allow read statement\n");
+ return(-1);
+ }
+
+ /*----------------------------------*/
+ /* Read Group AllowDelete Statement */
+ /*----------------------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->readGroupAllowDelete.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate read subscriber session statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->readGroupAllowDelete.stmt,(SQLCHAR *) read_group_allowDelete_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read group allow read statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->readGroupAllowDelete.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->readGroupAllowDelete.values.groupId,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read allow read statement param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readGroupAllowDelete.stmt, 1,
+ SQL_C_DEFAULT,
+ &uh->readGroupAllowDelete.values.allowDelete, 0,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 1 to read group allow read statement\n");
+ return(-1);
+ }
+
+ /*----------------------*/
+ /* read session details */
+ /*----------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->readSessionDetails.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate read session details statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->readSessionDetails.stmt,(SQLCHAR *) read_session_details_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read session details statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->readSessionDetails.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_LENGTH+1,0,
+ uh->readSessionDetails.values.number,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read sessions param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->readSessionDetails.stmt,
+ 2,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->readSessionDetails.values.serverId,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read sessions param 2\n");
+ return(-1);
+ }
+
+ rc = SQLBindCol(uh->readSessionDetails.stmt, 1,
+ SQL_C_CHAR,
+ uh->readSessionDetails.values.details, SESSION_DETAILS_LENGTH+1,
+ NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to bind column 1 to read group allow read statement\n");
+ return(-1);
+ }
+
+ /*-------------------*/
+ /* Update no of Read */
+ /*-------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->updateServerNoOfRead.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate update noOfRead statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->updateServerNoOfRead.stmt,(SQLCHAR *) update_noOfRead_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare update noOfRead statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateServerNoOfRead.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->updateServerNoOfRead.values.serverId,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read noOfRead param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateServerNoOfRead.stmt,
+ 2,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH+1,0,
+ uh->updateServerNoOfRead.values.suffix,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read noOfRead param 2\n");
+ return(-1);
+ }
+
+ /*----------------*/
+ /* Insert Session */
+ /*----------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->insertSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate update noOfRead statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->insertSession.stmt,(SQLCHAR *) insert_session_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare insert session statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->insertSession.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_LENGTH+1,0,
+ uh->insertSession.values.number,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read sessions param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->insertSession.stmt,
+ 2,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->insertSession.values.serverId,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read noOfRead param 2\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->insertSession.stmt,
+ 3,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SESSION_DETAILS_LENGTH+1,0,
+ uh->insertSession.values.details,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read sessions param 1\n");
+ return(-1);
+ }
+
+ /*----------------------------*/
+ /* Update subscriber sessions */
+ /*----------------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->updateSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate update noOfRead statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->updateSubscriberSession.stmt,(SQLCHAR *) update_subscriber_session_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare update subscriber session statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateSubscriberSession.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->updateSubscriberSession.values.activeSessions,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read noOfRead param 2\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateSubscriberSession.stmt,
+ 2,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_LENGTH+1,0,
+ uh->updateSubscriberSession.values.number,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read sessions param 1\n");
+ return(-1);
+ }
+
+ /*---------------------*/
+ /* Update no of Insert */
+ /*---------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->updateServerNoOfInsert.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate update noOfRead statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->updateServerNoOfInsert.stmt,(SQLCHAR *) update_noOfInsert_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare update noOfRead statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateServerNoOfInsert.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->updateServerNoOfInsert.values.serverId,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read noOfRead param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateServerNoOfInsert.stmt,
+ 2,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH+1,0,
+ uh->updateServerNoOfInsert.values.suffix,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read noOfRead param 2\n");
+ return(-1);
+ }
+
+ /*----------------*/
+ /* Delete Session */
+ /*----------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->deleteSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate update noOfRead statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->deleteSession.stmt,(SQLCHAR *) delete_session_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare insert session statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->deleteSession.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_LENGTH+1,0,
+ uh->deleteSession.values.number,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read sessions param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->deleteSession.stmt,
+ 2,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->deleteSession.values.serverId,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read noOfRead param 2\n");
+ return(-1);
+ }
+
+ /*---------------------*/
+ /* Update no of Delete */
+ /*---------------------*/
+ rc = SQLAllocStmt(uh->hdbc, &uh->updateServerNoOfDelete.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate update noOfRead statement\n");
+ return(-1);
+ }
+
+ rc = SQLPrepare(uh->updateServerNoOfDelete.stmt,(SQLCHAR *) update_noOfDelete_stmnt, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare update noOfRead statement\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateServerNoOfDelete.stmt,
+ 1,SQL_PARAM_INPUT,SQL_C_DEFAULT,SQL_INTEGER,
+ 0,0,
+ &uh->updateServerNoOfDelete.values.serverId,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read noOfRead param 1\n");
+ return(-1);
+ }
+
+ rc = SQLBindParameter(uh->updateServerNoOfDelete.stmt,
+ 2,SQL_PARAM_INPUT,SQL_C_CHAR,SQL_CHAR,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH+1,0,
+ uh->updateServerNoOfInsert.values.suffix,0,NULL);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to prepare read noOfRead param 2\n");
+ return(-1);
+ }
+
+ /*-------------------------------*/
+ /* Commit all prepare statements */
+ /*-------------------------------*/
+ rc = SQLTransact(uh->henv, uh->hdbc, SQL_COMMIT);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to commit all prepare insert statement\n");
+ return(-1);
+ }
+
+ return(0);
+}
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/macros.h b/ndb/test/ndbapi/lmc-bench/src/user/macros.h
new file mode 100644
index 00000000000..beb4352c269
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/macros.h
@@ -0,0 +1,52 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef MACROS_H
+#define MACROS_H
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <NdbOut.hpp>
+
+#define ERROR(x) {ndbout_c((x)); }
+#define ERROR1(x,y) {ndbout_c((x), (y)); }
+#define ERROR2(x,y,z) {ndbout_c((x), (y), (z)); }
+#define ERROR3(x,y,z,u) {ndbout_c((x), (y), (z), (u)); }
+#define ERROR4(x,y,z,u,w) {ndbout_c((x), (y), (z), (u), (w)); }
+
+#define INIT_RANDOM(x) srand48((x))
+#define UI_RANDOM(x) ((unsigned int)(lrand48()%(x)))
+
+#define ASSERT(cond, message) \
+ { if(!(cond)) { ERROR(message); exit(-1); }}
+
+#ifdef DEBUG_ON
+#define DEBUG(x) {ndbout_c((x)); }
+#define DEBUG1(x,y) {ndbout_c((x), (y)); }
+#define DEBUG2(x,y,z) {ndbout_c((x), (y), (z)); }
+#define DEBUG3(x,y,z,u) {ndbout_c((x), (y), (z), (u)); }
+#define DEBUG4(x,y,z,u,w) {ndbout_c((x), (y), (z), (u), (w)); }
+#define DEBUG5(x,y,z,u,w, v) {ndbout_c((x), (y), (z), (u), (w), (v)); }
+#else
+#define DEBUG(x)
+#define DEBUG1(x,y)
+#define DEBUG2(x,y,z)
+#define DEBUG3(x,y,z,u)
+#define DEBUG4(x,y,z,u,w)
+#define DEBUG5(x,y,z,u,w, v)
+#endif
+
+#endif
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/ndb_error.hpp b/ndb/test/ndbapi/lmc-bench/src/user/ndb_error.hpp
new file mode 100644
index 00000000000..5f792342ed9
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/ndb_error.hpp
@@ -0,0 +1,32 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef NDB_ERROR_H
+#define NDB_ERROR_H
+
+#include <stdio.h>
+#include <NdbOut.hpp>
+
+#define error_handler(x,y, z) { \
+ ndbout << x << " " << y << endl; \
+ exit(-1); }
+
+#define CHECK_NULL(x,y, z) if(x == 0) \
+ error_handler(y,(z->getNdbError()), 0)
+#define CHECK_MINUS_ONE(x, y, z) if(x == -1) \
+ error_handler(y,(z->getNdbError()), 0)
+
+#endif
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_populate.cpp b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_populate.cpp
new file mode 100644
index 00000000000..6a35bccd064
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_populate.cpp
@@ -0,0 +1,165 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+extern "C" {
+#include "user_populate.h"
+}
+
+#include <stdio.h>
+#include <NdbApi.hpp>
+
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+
+int
+insert_subscriber(void * obj,
+ SubscriberNumber number,
+ SubscriberName name,
+ GroupId groupId,
+ Location l,
+ ActiveSessions activeSessions,
+ ChangedBy changedBy,
+ ChangedTime changedTime){
+ Ndb * pNDB = (Ndb *)obj;
+ int check;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "getNdbOperation", MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "insertTuple", MyTransaction);
+
+ check = MyOperation->equal(SUBSCRIBER_NUMBER, number);
+ CHECK_MINUS_ONE(check, "equal", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_NAME, name);
+ CHECK_MINUS_ONE(check, "setValue name", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_GROUP, (char*)&groupId);
+ CHECK_MINUS_ONE(check, "setValue group", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_LOCATION, (char*)&l);
+ CHECK_MINUS_ONE(check, "setValue location", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_SESSIONS, (char*)&activeSessions);
+ CHECK_MINUS_ONE(check, "setValue sessions", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_CHANGED_BY, changedBy);
+ CHECK_MINUS_ONE(check, "setValue changedBy", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_CHANGED_TIME, changedTime);
+ CHECK_MINUS_ONE(check, "setValue changedTime", MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "commit", MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+ return 0;
+}
+
+int
+insert_server(void * obj,
+ ServerId serverId,
+ SubscriberSuffix suffix,
+ ServerName name,
+ Counter noOfRead,
+ Counter noOfInsert,
+ Counter noOfDelete){
+ Ndb * pNDB = (Ndb *)obj;
+ int check;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "getNdbOperation", MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "insert tuple", MyTransaction);
+
+ check = MyOperation->equal(SERVER_ID, (char*)&serverId);
+ CHECK_MINUS_ONE(check, "setValue id", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_SUBSCRIBER_SUFFIX, suffix);
+ CHECK_MINUS_ONE(check, "setValue suffix", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_NAME, name);
+ CHECK_MINUS_ONE(check, "setValue name", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_READS, (char*)&noOfRead);
+ CHECK_MINUS_ONE(check, "setValue reads", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_INSERTS, (char*)&noOfInsert);
+ CHECK_MINUS_ONE(check, "setValue inserts", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_DELETES, (char*)&noOfDelete);
+ CHECK_MINUS_ONE(check, "setValue deletes", MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "commit", MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+ return 0;
+}
+
+int
+insert_group(void * obj,
+ GroupId groupId,
+ GroupName name,
+ Permission allowRead,
+ Permission allowInsert,
+ Permission allowDelete){
+ Ndb * pNDB = (Ndb *)obj;
+ int check;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "getNdbOperation", MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "insertTuple", MyTransaction);
+
+ check = MyOperation->equal(GROUP_ID, (char*)&groupId);
+ CHECK_MINUS_ONE(check, "equal", MyTransaction);
+
+ check = MyOperation->setValue(GROUP_NAME, name);
+ CHECK_MINUS_ONE(check, "setValue name", MyTransaction);
+
+ check = MyOperation->setValue(GROUP_ALLOW_READ, (char*)&allowRead);
+ CHECK_MINUS_ONE(check, "setValue allowRead", MyTransaction);
+
+ check = MyOperation->setValue(GROUP_ALLOW_INSERT, (char*)&allowInsert);
+ CHECK_MINUS_ONE(check, "setValue allowInsert", MyTransaction);
+
+ check = MyOperation->setValue(GROUP_ALLOW_DELETE, (char*)&allowDelete);
+ CHECK_MINUS_ONE(check, "setValue allowDelete", MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "commit", MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+ return 0;
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction.cpp b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction.cpp
new file mode 100644
index 00000000000..182f1f99586
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction.cpp
@@ -0,0 +1,825 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#define DEBUG_ON
+
+extern "C" {
+#include "user_transaction.h"
+};
+
+#include "macros.h"
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+
+#include <time.h>
+#include <NdbApi.hpp>
+
+/**
+ * Transaction 1 - T1
+ *
+ * Update location and changed by/time on a subscriber
+ *
+ * Input:
+ * SubscriberNumber,
+ * Location,
+ * ChangedBy,
+ * ChangedTime
+ *
+ * Output:
+ */
+int
+T1(void * obj,
+ const SubscriberNumber number,
+ const Location new_location,
+ const ChangedBy changed_by,
+ const ChangedTime changed_time,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T1: getNdbOperation", MyTransaction);
+
+ check = MyOperation->updateTuple();
+ CHECK_MINUS_ONE(check, "T1: updateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T1: equal subscriber",
+ MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_LOCATION,
+ (char *)&new_location);
+ CHECK_MINUS_ONE(check, "T1: setValue location",
+ MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_by",
+ MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_time",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T1: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 2 - T2
+ *
+ * Read from Subscriber:
+ *
+ * Input:
+ * SubscriberNumber
+ *
+ * Output:
+ * Location
+ * Changed by
+ * Changed Timestamp
+ * Name
+ */
+int
+T2(void * obj,
+ const SubscriberNumber number,
+ Location * readLocation,
+ ChangedBy changed_by,
+ ChangedTime changed_time,
+ SubscriberName subscriberName,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T2: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T2: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_LOCATION,
+ (char *)readLocation);
+ CHECK_NULL(check2, "T2: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_NULL(check2, "T2: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_NULL(check2, "T2: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_NAME,
+ subscriberName);
+ CHECK_NULL(check2, "T2: getValue name",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T2: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 3 - T3
+ *
+ * Read session details
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ *
+ * Output:
+ * BranchExecuted
+ * SessionDetails
+ * ChangedBy
+ * ChangedTime
+ * Location
+ */
+int
+T3(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ SessionDetails outSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T3-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T3-1: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T3-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T3-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T3-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T3-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T3-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T3-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T3-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T3-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(GROUP_ALLOW_READ,
+ (char *)&permission);
+ CHECK_NULL(check2, "T3-2: getValue allow_read",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-2: NoCommit",
+ MyTransaction);
+
+ DEBUG3("T3(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("reading - ");
+
+ /* Operation 3 */
+
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T3-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-3: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T3-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-3: equal server id",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SESSION_DATA,
+ (char *)outSessionDetails);
+ CHECK_NULL(check2, "T3-3: getValue session details",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-3: NoCommit",
+ MyTransaction);
+
+ /* Operation 4 */
+
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T3-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T3-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-4: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T3-4: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(SERVER_READS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T3-4: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-4: NoCommit",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ (* outBranchExecuted) = 0;
+ }
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T3: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 4 - T4
+ *
+ * Create session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * SessionDetails,
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T4(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ const SessionDetails inSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T4-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ DEBUG3("T4(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ NdbOperation * MyOperation = 0;
+
+ MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T4-1: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTupleExclusive();
+ CHECK_MINUS_ONE(check, "T4-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T4-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T4-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T4-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T4-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T4-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T4-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T4-2: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T4-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T4-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(GROUP_ALLOW_INSERT,
+ (char *)&permission);
+ CHECK_NULL(check2, "T4-2: getValue allow_insert",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-2: NoCommit",
+ MyTransaction);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == 0)){
+
+ DEBUG("inserting - ");
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T4-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "T4-3: insertTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T4-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-3: equal server id",
+ MyTransaction);
+
+ check = MyOperation->setValue(SESSION_DATA,
+ (char *)inSessionDetails);
+ CHECK_MINUS_ONE(check, "T4-3: setValue session details",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-3: NoCommit",
+ MyTransaction);
+
+ /* Operation 4 */
+ MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T4-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SUBSCRIBER_NUMBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T4-4: equal number",
+ MyTransaction);
+
+ check = MyOperation->incValue(SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T4-4: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-4: NoCommit",
+ MyTransaction);
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T4-5: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T4-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(SERVER_INSERTS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T4-5: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-5: NoCommit",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T4: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T4:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 5 - T5
+ *
+ * Delete session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T5(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+ NdbConnection * MyTransaction = 0;
+ NdbOperation * MyOperation = 0;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T5-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T5-1: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTupleExclusive();
+ CHECK_MINUS_ONE(check, "T5-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T5-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T5-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T5-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T5-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T5-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T5-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T5-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T5-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T5-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(GROUP_ALLOW_DELETE,
+ (char *)&permission);
+ CHECK_NULL(check2, "T5-2: getValue allow_delete",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-2: NoCommit",
+ MyTransaction);
+
+ DEBUG3("T5(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("deleting - ");
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T5-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->deleteTuple();
+ CHECK_MINUS_ONE(check, "T5-3: deleteTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T5-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-3: equal server id",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-3: NoCommit",
+ MyTransaction);
+
+ /* Operation 4 */
+ MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T5-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SUBSCRIBER_NUMBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T5-4: equal number",
+ MyTransaction);
+
+ check = MyOperation->subValue(SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T5-4: dec value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-4: NoCommit",
+ MyTransaction);
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T5-5: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T5-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(SERVER_DELETES, (uint32)1);
+ CHECK_MINUS_ONE(check, "T5-5: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-5: NoCommit",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T5: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T5:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction2.cpp b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction2.cpp
new file mode 100644
index 00000000000..df3c7a7989e
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction2.cpp
@@ -0,0 +1,825 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#define DEBUG_ON
+
+extern "C" {
+#include "user_transaction.h"
+};
+
+#include "macros.h"
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+
+#include <time.h>
+#include <NdbApi.hpp>
+
+/**
+ * Transaction 1 - T1
+ *
+ * Update location and changed by/time on a subscriber
+ *
+ * Input:
+ * SubscriberNumber,
+ * Location,
+ * ChangedBy,
+ * ChangedTime
+ *
+ * Output:
+ */
+int
+T1(void * obj,
+ const SubscriberNumber number,
+ const Location new_location,
+ const ChangedBy changed_by,
+ const ChangedTime changed_time,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T1: getNdbOperation", MyTransaction);
+
+ check = MyOperation->updateTuple();
+ CHECK_MINUS_ONE(check, "T1: updateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T1: equal subscriber",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&new_location);
+ CHECK_MINUS_ONE(check, "T1: setValue location",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_by",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_time",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T1: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 2 - T2
+ *
+ * Read from Subscriber:
+ *
+ * Input:
+ * SubscriberNumber
+ *
+ * Output:
+ * Location
+ * Changed by
+ * Changed Timestamp
+ * Name
+ */
+int
+T2(void * obj,
+ const SubscriberNumber number,
+ Location * readLocation,
+ ChangedBy changed_by,
+ ChangedTime changed_time,
+ SubscriberName subscriberName,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T2: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T2: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)readLocation);
+ CHECK_NULL(check2, "T2: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_NULL(check2, "T2: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_NULL(check2, "T2: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_NAME,
+ subscriberName);
+ CHECK_NULL(check2, "T2: getValue name",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T2: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 3 - T3
+ *
+ * Read session details
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ *
+ * Output:
+ * BranchExecuted
+ * SessionDetails
+ * ChangedBy
+ * ChangedTime
+ * Location
+ */
+int
+T3(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ SessionDetails outSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T3-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T3-1: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T3-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T3-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T3-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T3-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T3-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T3-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T3-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T3-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_READ,
+ (char *)&permission);
+ CHECK_NULL(check2, "T3-2: getValue allow_read",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-2: NoCommit",
+ MyTransaction);
+
+ DEBUG3("T3(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("reading - ");
+
+ /* Operation 3 */
+
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T3-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-3: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T3-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-3: equal server id",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SESSION_DATA,
+ (char *)outSessionDetails);
+ CHECK_NULL(check2, "T3-3: getValue session details",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-3: NoCommit",
+ MyTransaction);
+
+ /* Operation 4 */
+
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T3-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T3-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-4: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T3-4: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_READS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T3-4: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-4: NoCommit",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ (* outBranchExecuted) = 0;
+ }
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T3: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 4 - T4
+ *
+ * Create session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * SessionDetails,
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T4(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ const SessionDetails inSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T4-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ DEBUG3("T4(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ NdbOperation * MyOperation = 0;
+
+ MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T4-1: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTupleExclusive();
+ CHECK_MINUS_ONE(check, "T4-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T4-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T4-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T4-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T4-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T4-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T4-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T4-2: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T4-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T4-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_INSERT,
+ (char *)&permission);
+ CHECK_NULL(check2, "T4-2: getValue allow_insert",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-2: NoCommit",
+ MyTransaction);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == 0)){
+
+ DEBUG("inserting - ");
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T4-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "T4-3: insertTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T4-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-3: equal server id",
+ MyTransaction);
+
+ check = MyOperation->setValue(SESSION_DATA,
+ (char *)inSessionDetails);
+ CHECK_MINUS_ONE(check, "T4-3: setValue session details",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-3: NoCommit",
+ MyTransaction);
+
+ /* Operation 4 */
+ MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T4-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T4-4: equal number",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T4-4: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-4: NoCommit",
+ MyTransaction);
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T4-5: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T4-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_INSERTS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T4-5: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-5: NoCommit",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T4: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T4:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 5 - T5
+ *
+ * Delete session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T5(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+ NdbConnection * MyTransaction = 0;
+ NdbOperation * MyOperation = 0;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T5-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T5-1: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTupleExclusive();
+ CHECK_MINUS_ONE(check, "T5-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T5-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T5-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T5-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T5-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T5-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T5-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T5-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T5-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T5-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_DELETE,
+ (char *)&permission);
+ CHECK_NULL(check2, "T5-2: getValue allow_delete",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-2: NoCommit",
+ MyTransaction);
+
+ DEBUG3("T5(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("deleting - ");
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T5-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->deleteTuple();
+ CHECK_MINUS_ONE(check, "T5-3: deleteTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T5-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-3: equal server id",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-3: NoCommit",
+ MyTransaction);
+
+ /* Operation 4 */
+ MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T5-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T5-4: equal number",
+ MyTransaction);
+
+ check = MyOperation->subValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T5-4: dec value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-4: NoCommit",
+ MyTransaction);
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T5-5: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T5-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_DELETES, (uint32)1);
+ CHECK_MINUS_ONE(check, "T5-5: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-5: NoCommit",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T5: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T5:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction3.cpp b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction3.cpp
new file mode 100644
index 00000000000..d2c92ecd424
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction3.cpp
@@ -0,0 +1,793 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#define DEBUG_ON
+
+extern "C" {
+#include "user_transaction.h"
+};
+
+#include "macros.h"
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+
+#include <time.h>
+#include <NdbApi.hpp>
+
+/**
+ * Transaction 1 - T1
+ *
+ * Update location and changed by/time on a subscriber
+ *
+ * Input:
+ * SubscriberNumber,
+ * Location,
+ * ChangedBy,
+ * ChangedTime
+ *
+ * Output:
+ */
+int
+T1(void * obj,
+ const SubscriberNumber number,
+ const Location new_location,
+ const ChangedBy changed_by,
+ const ChangedTime changed_time,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T1: getNdbOperation", MyTransaction);
+
+ check = MyOperation->updateTuple();
+ CHECK_MINUS_ONE(check, "T1: updateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T1: equal subscriber",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&new_location);
+ CHECK_MINUS_ONE(check, "T1: setValue location",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_by",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_time",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T1: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 2 - T2
+ *
+ * Read from Subscriber:
+ *
+ * Input:
+ * SubscriberNumber
+ *
+ * Output:
+ * Location
+ * Changed by
+ * Changed Timestamp
+ * Name
+ */
+int
+T2(void * obj,
+ const SubscriberNumber number,
+ Location * readLocation,
+ ChangedBy changed_by,
+ ChangedTime changed_time,
+ SubscriberName subscriberName,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T2: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T2: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)readLocation);
+ CHECK_NULL(check2, "T2: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_NULL(check2, "T2: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_NULL(check2, "T2: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_NAME,
+ subscriberName);
+ CHECK_NULL(check2, "T2: getValue name",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T2: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 3 - T3
+ *
+ * Read session details
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ *
+ * Output:
+ * BranchExecuted
+ * SessionDetails
+ * ChangedBy
+ * ChangedTime
+ * Location
+ */
+int
+T3(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ SessionDetails outSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T3-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T3-1: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T3-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T3-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T3-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T3-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T3-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T3-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T3-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T3-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_READ,
+ (char *)&permission);
+ CHECK_NULL(check2, "T3-2: getValue allow_read",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-2: NoCommit",
+ MyTransaction);
+
+ DEBUG3("T3(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("reading - ");
+
+ /* Operation 3 */
+
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T3-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-3: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T3-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-3: equal server id",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SESSION_DATA,
+ (char *)outSessionDetails);
+ CHECK_NULL(check2, "T3-3: getValue session details",
+ MyTransaction);
+
+ /* Operation 4 */
+
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T3-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T3-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-4: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T3-4: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_READS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T3-4: inc value",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ (* outBranchExecuted) = 0;
+ }
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T3: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 4 - T4
+ *
+ * Create session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * SessionDetails,
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T4(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ const SessionDetails inSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T4-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ DEBUG3("T4(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ NdbOperation * MyOperation = 0;
+
+ MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T4-1: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTupleExclusive();
+ CHECK_MINUS_ONE(check, "T4-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T4-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T4-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T4-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T4-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T4-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T4-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T4-2: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T4-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T4-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_INSERT,
+ (char *)&permission);
+ CHECK_NULL(check2, "T4-2: getValue allow_insert",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-2: NoCommit",
+ MyTransaction);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == 0)){
+
+ DEBUG("inserting - ");
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T4-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "T4-3: insertTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T4-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-3: equal server id",
+ MyTransaction);
+
+ check = MyOperation->setValue(SESSION_DATA,
+ (char *)inSessionDetails);
+ CHECK_MINUS_ONE(check, "T4-3: setValue session details",
+ MyTransaction);
+
+ /* Operation 4 */
+ MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T4-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T4-4: equal number",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T4-4: inc value",
+ MyTransaction);
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T4-5: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T4-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_INSERTS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T4-5: inc value",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T4: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T4:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 5 - T5
+ *
+ * Delete session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T5(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+ NdbConnection * MyTransaction = 0;
+ NdbOperation * MyOperation = 0;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T5-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T5-1: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTupleExclusive();
+ CHECK_MINUS_ONE(check, "T5-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T5-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T5-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T5-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T5-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T5-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T5-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T5-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T5-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T5-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_DELETE,
+ (char *)&permission);
+ CHECK_NULL(check2, "T5-2: getValue allow_delete",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-2: NoCommit",
+ MyTransaction);
+
+ DEBUG3("T5(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("deleting - ");
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T5-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->deleteTuple();
+ CHECK_MINUS_ONE(check, "T5-3: deleteTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T5-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-3: equal server id",
+ MyTransaction);
+
+ /* Operation 4 */
+ MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T5-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T5-4: equal number",
+ MyTransaction);
+
+ check = MyOperation->subValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T5-4: dec value",
+ MyTransaction);
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T5-5: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T5-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_DELETES, (uint32)1);
+ CHECK_MINUS_ONE(check, "T5-5: inc value",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T5: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T5:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction4.cpp b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction4.cpp
new file mode 100644
index 00000000000..e652c7bfed8
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction4.cpp
@@ -0,0 +1,770 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#define DEBUG_ON
+
+extern "C" {
+#include "user_transaction.h"
+};
+
+#include "macros.h"
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+
+#include <time.h>
+#include <NdbApi.hpp>
+
+/**
+ * Transaction 1 - T1
+ *
+ * Update location and changed by/time on a subscriber
+ *
+ * Input:
+ * SubscriberNumber,
+ * Location,
+ * ChangedBy,
+ * ChangedTime
+ *
+ * Output:
+ */
+int
+T1(void * obj,
+ const SubscriberNumber number,
+ const Location new_location,
+ const ChangedBy changed_by,
+ const ChangedTime changed_time,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ DEBUG2("T1(%.*s):\n", SUBSCRIBER_NUMBER_LENGTH, number);
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T1-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T1: getNdbOperation", MyTransaction);
+
+ check = MyOperation->updateTuple();
+ CHECK_MINUS_ONE(check, "T1: updateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T1: equal subscriber",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&new_location);
+ CHECK_MINUS_ONE(check, "T1: setValue location",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_by",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_time",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T1: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 2 - T2
+ *
+ * Read from Subscriber:
+ *
+ * Input:
+ * SubscriberNumber
+ *
+ * Output:
+ * Location
+ * Changed by
+ * Changed Timestamp
+ * Name
+ */
+int
+T2(void * obj,
+ const SubscriberNumber number,
+ Location * readLocation,
+ ChangedBy changed_by,
+ ChangedTime changed_time,
+ SubscriberName subscriberName,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ DEBUG2("T2(%.*s):\n", SUBSCRIBER_NUMBER_LENGTH, number);
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T2-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T2: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)readLocation);
+ CHECK_NULL(check2, "T2: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_NULL(check2, "T2: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_NULL(check2, "T2: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_NAME,
+ subscriberName);
+ CHECK_NULL(check2, "T2: getValue name",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T2: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 3 - T3
+ *
+ * Read session details
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ *
+ * Output:
+ * BranchExecuted
+ * SessionDetails
+ * ChangedBy
+ * ChangedTime
+ * Location
+ */
+int
+T3(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ SessionDetails outSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ DEBUG3("T3(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T3-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T3-1: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T3-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T3-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T3-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T3-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T3-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T3-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T3-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T3-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_READ,
+ (char *)&permission);
+ CHECK_NULL(check2, "T3-2: getValue allow_read",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-2: NoCommit",
+ MyTransaction);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("reading - ");
+
+ /* Operation 3 */
+
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T3-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-3: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T3-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-3: equal server id",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SESSION_DATA,
+ (char *)outSessionDetails);
+ CHECK_NULL(check2, "T3-3: getValue session details",
+ MyTransaction);
+
+ /* Operation 4 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T3-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T3-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-4: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T3-4: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_READS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T3-4: inc value",
+ MyTransaction);
+ (* outBranchExecuted) = 1;
+ } else {
+ (* outBranchExecuted) = 0;
+ }
+ DEBUG("commit...");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T3: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ DEBUG("done\n");
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 4 - T4
+ *
+ * Create session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * SessionDetails,
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T4(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ const SessionDetails inSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ DEBUG3("T4(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T4-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T4-1: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T4-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T4-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T4-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T4-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T4-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T4-1: getValue sessions",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T4-4: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T4-2: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T4-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T4-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_INSERT,
+ (char *)&permission);
+ CHECK_NULL(check2, "T4-2: getValue allow_insert",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-2: NoCommit",
+ MyTransaction);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == 0)){
+
+ DEBUG("inserting - ");
+
+ /* Operation 3 */
+
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T4-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "T4-3: insertTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T4-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-3: equal server id",
+ MyTransaction);
+
+ check = MyOperation->setValue(SESSION_DATA,
+ (char *)inSessionDetails);
+ CHECK_MINUS_ONE(check, "T4-3: setValue session details",
+ MyTransaction);
+
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T4-5: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T4-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_INSERTS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T4-5: inc value",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback && (* outBranchExecuted)){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T4: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T4:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 5 - T5
+ *
+ * Delete session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T5(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ DEBUG3("T5(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ Ndb * pNDB = (Ndb *) obj;
+ NdbConnection * MyTransaction = 0;
+ NdbOperation * MyOperation = 0;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T5-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T5-1: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T5-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T5-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T5-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T5-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T5-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T5-1: getValue sessions",
+ MyTransaction);
+
+ check = MyOperation->subValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T5-4: dec value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T5-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T5-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T5-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_DELETE,
+ (char *)&permission);
+ CHECK_NULL(check2, "T5-2: getValue allow_delete",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-2: NoCommit",
+ MyTransaction);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("deleting - ");
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T5-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->deleteTuple();
+ CHECK_MINUS_ONE(check, "T5-3: deleteTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T5-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-3: equal server id",
+ MyTransaction);
+
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T5-5: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T5-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_DELETES, (uint32)1);
+ CHECK_MINUS_ONE(check, "T5-5: inc value",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback && (* outBranchExecuted)){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T5: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T5:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction5.cpp b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction5.cpp
new file mode 100644
index 00000000000..86580008d10
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction5.cpp
@@ -0,0 +1,769 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#define DEBUG_ON
+
+extern "C" {
+#include "user_transaction.h"
+};
+
+#include "macros.h"
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+
+#include <time.h>
+#include <NdbApi.hpp>
+
+/**
+ * Transaction 1 - T1
+ *
+ * Update location and changed by/time on a subscriber
+ *
+ * Input:
+ * SubscriberNumber,
+ * Location,
+ * ChangedBy,
+ * ChangedTime
+ *
+ * Output:
+ */
+int
+T1(void * obj,
+ const SubscriberNumber number,
+ const Location new_location,
+ const ChangedBy changed_by,
+ const ChangedTime changed_time,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ DEBUG2("T1(%.*s):\n", SUBSCRIBER_NUMBER_LENGTH, number);
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T1-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T1: getNdbOperation", MyTransaction);
+
+ check = MyOperation->updateTuple();
+ CHECK_MINUS_ONE(check, "T1: updateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T1: equal subscriber",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&new_location);
+ CHECK_MINUS_ONE(check, "T1: setValue location",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_by",
+ MyTransaction);
+
+ check = MyOperation->setValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_MINUS_ONE(check, "T1: setValue changed_time",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T1: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 2 - T2
+ *
+ * Read from Subscriber:
+ *
+ * Input:
+ * SubscriberNumber
+ *
+ * Output:
+ * Location
+ * Changed by
+ * Changed Timestamp
+ * Name
+ */
+int
+T2(void * obj,
+ const SubscriberNumber number,
+ Location * readLocation,
+ ChangedBy changed_by,
+ ChangedTime changed_time,
+ SubscriberName subscriberName,
+ BenchmarkTime * transaction_time){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ DEBUG2("T2(%.*s):\n", SUBSCRIBER_NUMBER_LENGTH, number);
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T2-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ CHECK_MINUS_ONE(check, "T2: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)readLocation);
+ CHECK_NULL(check2, "T2: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ CHECK_NULL(check2, "T2: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ CHECK_NULL(check2, "T2: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_NAME,
+ subscriberName);
+ CHECK_NULL(check2, "T2: getValue name",
+ MyTransaction);
+
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T2: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(transaction_time);
+ time_diff(transaction_time, &start);
+ return 0;
+}
+
+/**
+ * Transaction 3 - T3
+ *
+ * Read session details
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ *
+ * Output:
+ * BranchExecuted
+ * SessionDetails
+ * ChangedBy
+ * ChangedTime
+ * Location
+ */
+int
+T3(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ SessionDetails outSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T3-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T3-1: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T3-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T3-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T3-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T3-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T3-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T3-1: getValue sessions",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T3-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T3-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T3-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_READ,
+ (char *)&permission);
+ CHECK_NULL(check2, "T3-2: getValue allow_read",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-2: NoCommit",
+ MyTransaction);
+
+ DEBUG3("T3(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("reading - ");
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T3-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->simpleRead();
+ CHECK_MINUS_ONE(check, "T3-3: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T3-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-3: equal server id",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SESSION_DATA,
+ (char *)outSessionDetails);
+ CHECK_NULL(check2, "T3-3: getValue session details",
+ MyTransaction);
+
+ /* Operation 4 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T3-4: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T3-4: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T3-4: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T3-4: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_READS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T3-4: inc value",
+ MyTransaction);
+ (* outBranchExecuted) = 1;
+ } else {
+ (* outBranchExecuted) = 0;
+ }
+ DEBUG("commit...");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T3: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ DEBUG("done\n");
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 4 - T4
+ *
+ * Create session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * SessionDetails,
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T4(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ const SessionDetails inSessionDetails,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T4-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T4-1: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T4-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T4-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T4-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T4-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T4-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T4-1: getValue sessions",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T4-4: inc value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T4-2: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T4-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T4-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_INSERT,
+ (char *)&permission);
+ CHECK_NULL(check2, "T4-2: getValue allow_insert",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-2: NoCommit",
+ MyTransaction);
+
+ DEBUG3("T4(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == 0)){
+
+ DEBUG("inserting - ");
+
+ /* Operation 3 */
+
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T4-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "T4-3: insertTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T4-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-3: equal server id",
+ MyTransaction);
+
+ check = MyOperation->setValue(SESSION_DATA,
+ (char *)inSessionDetails);
+ CHECK_MINUS_ONE(check, "T4-3: setValue session details",
+ MyTransaction);
+
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T4-5: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T4-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T4-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T4-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_INSERTS, (uint32)1);
+ CHECK_MINUS_ONE(check, "T4-5: inc value",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback && (* outBranchExecuted)){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T4: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T4:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
+
+/**
+ * Transaction 5 - T5
+ *
+ * Delete session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+int
+T5(void * obj,
+ const SubscriberNumber inNumber,
+ const SubscriberSuffix inSuffix,
+ const ServerId inServerId,
+ const ServerBit inServerBit,
+ ChangedBy outChangedBy,
+ ChangedTime outChangedTime,
+ Location * outLocation,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted,
+ BenchmarkTime * outTransactionTime){
+
+ Ndb * pNDB = (Ndb *) obj;
+ NdbConnection * MyTransaction = 0;
+ NdbOperation * MyOperation = 0;
+
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+
+ BenchmarkTime start;
+ get_time(&start);
+
+ int check;
+ NdbRecAttr * check2;
+
+ MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T5-1: startTranscation", pNDB->getNdbErrorString(), 0);
+
+ MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T5-1: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-1: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ CHECK_MINUS_ONE(check, "T5-1: equal subscriber",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)outLocation);
+ CHECK_NULL(check2, "T5-1: getValue location",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ CHECK_NULL(check2, "T5-1: getValue changed_by",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ CHECK_NULL(check2, "T5-1: getValue changed_time",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ CHECK_NULL(check2, "T5-1: getValue group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ CHECK_NULL(check2, "T5-1: getValue sessions",
+ MyTransaction);
+
+ check = MyOperation->subValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ CHECK_MINUS_ONE(check, "T5-4: dec value",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T5-2: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->readTuple();
+ CHECK_MINUS_ONE(check, "T5-2: readTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ CHECK_MINUS_ONE(check, "T5-2: equal group",
+ MyTransaction);
+
+ check2 = MyOperation->getValue(IND_GROUP_ALLOW_DELETE,
+ (char *)&permission);
+ CHECK_NULL(check2, "T5-2: getValue allow_delete",
+ MyTransaction);
+
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-2: NoCommit",
+ MyTransaction);
+
+ DEBUG3("T5(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ DEBUG("deleting - ");
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T5-3: getNdbOperation",
+ MyTransaction);
+
+ check = MyOperation->deleteTuple();
+ CHECK_MINUS_ONE(check, "T5-3: deleteTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ CHECK_MINUS_ONE(check, "T5-3: equal number",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-3: equal server id",
+ MyTransaction);
+
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T5-5: getNdbOperation",
+ MyTransaction);
+
+
+ check = MyOperation->interpretedUpdateTuple();
+ CHECK_MINUS_ONE(check, "T5-5: interpretedUpdateTuple",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ CHECK_MINUS_ONE(check, "T5-5: equal serverId",
+ MyTransaction);
+
+ check = MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ CHECK_MINUS_ONE(check, "T5-5: equal suffix",
+ MyTransaction);
+
+ check = MyOperation->incValue(IND_SERVER_DELETES, (uint32)1);
+ CHECK_MINUS_ONE(check, "T5-5: inc value",
+ MyTransaction);
+
+ (* outBranchExecuted) = 1;
+ } else {
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ (* outBranchExecuted) = 0;
+ }
+
+ if(!inDoRollback && (* outBranchExecuted)){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T5: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T5:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+
+ get_time(outTransactionTime);
+ time_diff(outTransactionTime, &start);
+ return 0;
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction6.cpp b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction6.cpp
new file mode 100644
index 00000000000..262f38e9ffb
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/ndb_user_transaction6.cpp
@@ -0,0 +1,561 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+//#define DEBUG_ON
+
+#include <string.h>
+#include "userHandle.h"
+#include "userInterface.h"
+
+#include "macros.h"
+#include "ndb_schema.hpp"
+#include "ndb_error.hpp"
+
+#include <NdbApi.hpp>
+
+
+void
+userCheckpoint(UserHandle *uh){
+}
+
+inline
+NdbConnection *
+startTransaction(Ndb * pNDB, ServerId inServerId, const SubscriberNumber inNumber){
+
+ const int keyDataLenBytes = sizeof(ServerId)+SUBSCRIBER_NUMBER_LENGTH;
+ const int keyDataLen_64Words = keyDataLenBytes >> 3;
+
+ Uint64 keyDataBuf[keyDataLen_64Words+1]; // The "+1" is for rounding...
+
+ char * keyDataBuf_charP = (char *)&keyDataBuf[0];
+ Uint32 * keyDataBuf_wo32P = (Uint32 *)&keyDataBuf[0];
+
+ // Server Id comes first
+ keyDataBuf_wo32P[0] = inServerId;
+ // Then subscriber number
+ memcpy(&keyDataBuf_charP[sizeof(ServerId)], inNumber, SUBSCRIBER_NUMBER_LENGTH);
+
+ return pNDB->startTransaction(0, keyDataBuf_charP, keyDataLenBytes);
+}
+
+/**
+ * Transaction 1 - T1
+ *
+ * Update location and changed by/time on a subscriber
+ *
+ * Input:
+ * SubscriberNumber,
+ * Location,
+ * ChangedBy,
+ * ChangedTime
+ *
+ * Output:
+ */
+void
+userTransaction_T1(UserHandle * uh,
+ SubscriberNumber number,
+ Location new_location,
+ ChangedBy changed_by,
+ ChangedTime changed_time){
+ Ndb * pNDB = uh->pNDB;
+
+ DEBUG2("T1(%.*s):\n", SUBSCRIBER_NUMBER_LENGTH, number);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction != NULL) {
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ if (MyOperation != NULL) {
+ MyOperation->updateTuple();
+ MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ MyOperation->setValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&new_location);
+ MyOperation->setValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ MyOperation->setValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ check = MyTransaction->execute( Commit );
+ if (check != -1) {
+ pNDB->closeTransaction(MyTransaction);
+ return;
+ } else {
+ CHECK_MINUS_ONE(check, "T1: Commit",
+ MyTransaction);
+ }//if
+ } else {
+ CHECK_NULL(MyOperation, "T1: getNdbOperation", MyTransaction);
+ }//if
+ } else {
+ error_handler("T1-1: startTranscation", pNDB->getNdbErrorString(), pNDB->getNdbError());
+ }//if
+}
+
+/**
+ * Transaction 2 - T2
+ *
+ * Read from Subscriber:
+ *
+ * Input:
+ * SubscriberNumber
+ *
+ * Output:
+ * Location
+ * Changed by
+ * Changed Timestamp
+ * Name
+ */
+void
+userTransaction_T2(UserHandle * uh,
+ SubscriberNumber number,
+ Location * readLocation,
+ ChangedBy changed_by,
+ ChangedTime changed_time,
+ SubscriberName subscriberName){
+ Ndb * pNDB = uh->pNDB;
+
+ DEBUG2("T2(%.*s):\n", SUBSCRIBER_NUMBER_LENGTH, number);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T2-1: startTransaction", pNDB->getNdbErrorString(), pNDB->getNdbError());
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T2: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->readTuple();
+ MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ number);
+ MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)readLocation);
+ MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ changed_by);
+ MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ changed_time);
+ MyOperation->getValue(IND_SUBSCRIBER_NAME,
+ subscriberName);
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T2: Commit",
+ MyTransaction);
+ pNDB->closeTransaction(MyTransaction);
+}
+
+/**
+ * Transaction 3 - T3
+ *
+ * Read session details
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ *
+ * Output:
+ * BranchExecuted
+ * SessionDetails
+ * ChangedBy
+ * ChangedTime
+ * Location
+ */
+void
+userTransaction_T3(UserHandle * uh,
+ SubscriberNumber inNumber,
+ ServerId inServerId,
+ ServerBit inServerBit,
+ SessionDetails outSessionDetails,
+ BranchExecuted * outBranchExecuted){
+ Ndb * pNDB = uh->pNDB;
+
+ char outChangedBy [sizeof(ChangedBy) +(4-(sizeof(ChangedBy) & 3))];
+ char outChangedTime [sizeof(ChangedTime)+(4-(sizeof(ChangedTime) & 3))];
+ Location outLocation;
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+ SubscriberSuffix inSuffix;
+
+ DEBUG3("T3(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = startTransaction(pNDB, inServerId, inNumber);
+ if (MyTransaction == NULL)
+ error_handler("T3-1: startTranscation", pNDB->getNdbErrorString(), pNDB->getNdbError());
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T3-1: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->readTuple();
+ MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&outLocation);
+ MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-1: NoCommit",
+ MyTransaction);
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T3-2: getNdbOperation",
+ MyTransaction);
+
+
+ MyOperation->readTuple();
+ MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ MyOperation->getValue(IND_GROUP_ALLOW_READ,
+ (char *)&permission);
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T3-2: NoCommit",
+ MyTransaction);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ memcpy(inSuffix,
+ &inNumber[SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH], SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+ DEBUG2("reading(%.*s) - ", SUBSCRIBER_NUMBER_SUFFIX_LENGTH, inSuffix);
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T3-3: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->simpleRead();
+
+ MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ MyOperation->getValue(IND_SESSION_DATA,
+ (char *)outSessionDetails);
+ /* Operation 4 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T3-4: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->interpretedUpdateTuple();
+ MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ MyOperation->incValue(IND_SERVER_READS, (uint32)1);
+ (* outBranchExecuted) = 1;
+ } else {
+ (* outBranchExecuted) = 0;
+ }
+ DEBUG("commit...");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T3: Commit",
+ MyTransaction);
+
+ pNDB->closeTransaction(MyTransaction);
+
+ DEBUG("done\n");
+}
+
+
+/**
+ * Transaction 4 - T4
+ *
+ * Create session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * SessionDetails,
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+void
+userTransaction_T4(UserHandle * uh,
+ SubscriberNumber inNumber,
+ ServerId inServerId,
+ ServerBit inServerBit,
+ SessionDetails inSessionDetails,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted){
+
+ Ndb * pNDB = uh->pNDB;
+
+ char outChangedBy [sizeof(ChangedBy) +(4-(sizeof(ChangedBy) & 3))];
+ char outChangedTime [sizeof(ChangedTime)+(4-(sizeof(ChangedTime) & 3))];
+ Location outLocation;
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+ SubscriberSuffix inSuffix;
+
+ DEBUG3("T4(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ int check;
+ NdbRecAttr * check2;
+
+ NdbConnection * MyTransaction = startTransaction(pNDB, inServerId, inNumber);
+ if (MyTransaction == NULL)
+ error_handler("T4-1: startTranscation", pNDB->getNdbErrorString(), pNDB->getNdbError());
+
+ NdbOperation *MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T4-1: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->interpretedUpdateTuple();
+ MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&outLocation);
+ MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ outChangedBy);
+ MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ outChangedTime);
+ MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ MyOperation->incValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ check = MyTransaction->execute( NoCommit );
+
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T4-2: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->readTuple();
+ MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ MyOperation->getValue(IND_GROUP_ALLOW_INSERT,
+ (char *)&permission);
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T4-2: NoCommit",
+ MyTransaction);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == 0)){
+
+ memcpy(inSuffix,
+ &inNumber[SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH], SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+
+ DEBUG2("inserting(%.*s) - ", SUBSCRIBER_NUMBER_SUFFIX_LENGTH, inSuffix);
+
+ /* Operation 3 */
+
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T4-3: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->insertTuple();
+ MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ MyOperation->setValue(SESSION_DATA,
+ (char *)inSessionDetails);
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T4-5: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->interpretedUpdateTuple();
+ MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ MyOperation->incValue(IND_SERVER_INSERTS, (uint32)1);
+ (* outBranchExecuted) = 1;
+ } else {
+ (* outBranchExecuted) = 0;
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ }
+
+ if(!inDoRollback && (* outBranchExecuted)){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T4: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T4:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+}
+
+
+/**
+ * Transaction 5 - T5
+ *
+ * Delete session
+ *
+ * Input:
+ * SubscriberNumber
+ * ServerId
+ * ServerBit
+ * DoRollback
+ * Output:
+ * ChangedBy
+ * ChangedTime
+ * Location
+ * BranchExecuted
+ */
+void
+userTransaction_T5(UserHandle * uh,
+ SubscriberNumber inNumber,
+ ServerId inServerId,
+ ServerBit inServerBit,
+ DoRollback inDoRollback,
+ BranchExecuted * outBranchExecuted){
+ Ndb * pNDB = uh->pNDB;
+
+ DEBUG3("T5(%.*s, %.2d): ", SUBSCRIBER_NUMBER_LENGTH, inNumber, inServerId);
+
+ NdbConnection * MyTransaction = 0;
+ NdbOperation * MyOperation = 0;
+
+ char outChangedBy [sizeof(ChangedBy) +(4-(sizeof(ChangedBy) & 3))];
+ char outChangedTime [sizeof(ChangedTime)+(4-(sizeof(ChangedTime) & 3))];
+ Location outLocation;
+ GroupId groupId;
+ ActiveSessions sessions;
+ Permission permission;
+ SubscriberSuffix inSuffix;
+
+ int check;
+ NdbRecAttr * check2;
+
+ MyTransaction = pNDB->startTransaction();
+ if (MyTransaction == NULL)
+ error_handler("T5-1: startTranscation", pNDB->getNdbErrorString(), pNDB->getNdbError());
+
+ MyOperation= MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "T5-1: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->interpretedUpdateTuple();
+ MyOperation->equal(IND_SUBSCRIBER_NUMBER,
+ inNumber);
+ MyOperation->getValue(IND_SUBSCRIBER_LOCATION,
+ (char *)&outLocation);
+ MyOperation->getValue(IND_SUBSCRIBER_CHANGED_BY,
+ &outChangedBy[0]);
+ MyOperation->getValue(IND_SUBSCRIBER_CHANGED_TIME,
+ &outChangedTime[0]);
+ MyOperation->getValue(IND_SUBSCRIBER_GROUP,
+ (char *)&groupId);
+ MyOperation->getValue(IND_SUBSCRIBER_SESSIONS,
+ (char *)&sessions);
+ MyOperation->subValue(IND_SUBSCRIBER_SESSIONS,
+ (uint32)inServerBit);
+ MyTransaction->execute( NoCommit );
+ /* Operation 2 */
+
+ MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "T5-2: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->readTuple();
+ MyOperation->equal(IND_GROUP_ID,
+ (char*)&groupId);
+ MyOperation->getValue(IND_GROUP_ALLOW_DELETE,
+ (char *)&permission);
+ check = MyTransaction->execute( NoCommit );
+ CHECK_MINUS_ONE(check, "T5-2: NoCommit",
+ MyTransaction);
+
+ if(((permission & inServerBit) == inServerBit) &&
+ ((sessions & inServerBit) == inServerBit)){
+
+ memcpy(inSuffix,
+ &inNumber[SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH], SUBSCRIBER_NUMBER_SUFFIX_LENGTH);
+
+ DEBUG2("deleting(%.*s) - ", SUBSCRIBER_NUMBER_SUFFIX_LENGTH, inSuffix);
+
+ /* Operation 3 */
+ MyOperation = MyTransaction->getNdbOperation(SESSION_TABLE);
+ CHECK_NULL(MyOperation, "T5-3: getNdbOperation",
+ MyTransaction);
+
+ MyOperation->deleteTuple();
+ MyOperation->equal(IND_SESSION_SUBSCRIBER,
+ (char*)inNumber);
+ MyOperation->equal(IND_SESSION_SERVER,
+ (char*)&inServerId);
+ /* Operation 4 */
+
+ /* Operation 5 */
+ MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "T5-5: getNdbOperation",
+ MyTransaction);
+
+
+ MyOperation->interpretedUpdateTuple();
+ MyOperation->equal(IND_SERVER_ID,
+ (char*)&inServerId);
+ MyOperation->equal(IND_SERVER_SUBSCRIBER_SUFFIX,
+ (char*)inSuffix);
+ MyOperation->incValue(IND_SERVER_DELETES, (uint32)1);
+ (* outBranchExecuted) = 1;
+ } else {
+ (* outBranchExecuted) = 0;
+ DEBUG1("%s", ((permission & inServerBit) ? "permission - " : "no permission - "));
+ DEBUG1("%s", ((sessions & inServerBit) ? "in session - " : "no in session - "));
+ }
+
+ if(!inDoRollback && (* outBranchExecuted)){
+ DEBUG("commit\n");
+ check = MyTransaction->execute( Commit );
+ CHECK_MINUS_ONE(check, "T5: Commit",
+ MyTransaction);
+ } else {
+ DEBUG("rollback\n");
+ check = MyTransaction->execute(Rollback);
+ CHECK_MINUS_ONE(check, "T5:Rollback",
+ MyTransaction);
+
+ }
+
+ pNDB->closeTransaction(MyTransaction);
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/old/Makefile b/ndb/test/ndbapi/lmc-bench/src/user/old/Makefile
new file mode 100644
index 00000000000..9b1247d44af
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/old/Makefile
@@ -0,0 +1,10 @@
+include ../makevars.$(ARCH)
+
+LIBRARY = ../../lib/libUser.so
+
+OBJECTS = \
+ $(LIBRARY)(localDbPrepare.o)\
+ $(LIBRARY)(userInterface.o)\
+ $(LIBRARY)(userTransaction.o)
+
+$(LIBRARY): $(OBJECTS)
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/old/userHandle.h b/ndb/test/ndbapi/lmc-bench/src/user/old/userHandle.h
new file mode 100644
index 00000000000..1de468d4dad
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/old/userHandle.h
@@ -0,0 +1,190 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef USERHANDLE_H
+#define USERHANDLE_H
+
+/***************************************************************/
+/* I N C L U D E D F I L E S */
+/***************************************************************/
+
+#include "sql.h"
+#include "sqlext.h"
+#include "sqltypes.h"
+
+#include "testDefinitions.h"
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+struct userHandle{
+ SQLHENV henv;
+ SQLHDBC hdbc;
+ SQLHSTMT stmt;
+
+ /*----------------*/
+ /* Transaction T1 */
+ /*----------------*/
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ SubscriberNumber number;
+ Location location;
+ ChangedBy changedBy;
+ ChangedTime changedTime;
+ }values;
+ }updateSubscriber;
+
+ /*----------------*/
+ /* Transaction T2 */
+ /*----------------*/
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ SubscriberNumber number;
+ SubscriberName name;
+ Location location;
+ ChangedBy changedBy;
+ ChangedTime changedTime;
+ }values;
+ }readSubscriber;
+
+ /*----------------*/
+ /* Transaction T3 */
+ /*----------------*/
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ GroupId groupId;
+ Permission allowRead;
+ }values;
+ }readGroupAllowRead;
+
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ SubscriberNumber number;
+ ServerId serverId;
+ SessionDetails details;
+ }values;
+ }readSessionDetails;
+
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ ServerId serverId;
+ SubscriberSuffix suffix;
+ }values;
+ }updateServerNoOfRead;
+
+ /*----------------*/
+ /* Transaction T4 */
+ /*----------------*/
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ GroupId groupId;
+ Permission allowInsert;
+ }values;
+ }readGroupAllowInsert;
+
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ SubscriberNumber number;
+ ServerId serverId;
+ SessionDetails details;
+ }values;
+ }insertSession;
+
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ ServerId serverId;
+ SubscriberSuffix suffix;
+ }values;
+ }updateServerNoOfInsert;
+
+ /*----------------*/
+ /* Transaction T5 */
+ /*----------------*/
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ GroupId groupId;
+ Permission allowDelete;
+ }values;
+ }readGroupAllowDelete;
+
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ SubscriberNumber number;
+ ServerId serverId;
+ }values;
+ }deleteSession;
+
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ ServerId serverId;
+ SubscriberSuffix suffix;
+ }values;
+ }updateServerNoOfDelete;
+
+ /*--------------------------*/
+ /* Transaction T3 + T4 + T5 */
+ /*--------------------------*/
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ SubscriberNumber number;
+ uint32 activeSessions;
+ GroupId groupId;
+ ChangedBy changedBy;
+ ChangedTime changedTime;
+ }values;
+ }readSubscriberSession;
+
+ struct {
+ SQLHSTMT stmt;
+ struct {
+ SubscriberNumber number;
+ uint32 activeSessions;
+ }values;
+ }updateSubscriberSession;
+};
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+
+#endif /* USERHANDLE_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/old/userInterface.c b/ndb/test/ndbapi/lmc-bench/src/user/old/userInterface.c
new file mode 100644
index 00000000000..c68f287f5dd
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/old/userInterface.c
@@ -0,0 +1,456 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <sys/types.h>
+#include <sys/time.h>
+
+#include "userInterface.h"
+#include "userHandle.h"
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+extern int localDbPrepare(UserHandle *uh);
+
+static int dbCreate(UserHandle *uh);
+
+/***************************************************************
+* L O C A L D A T A *
+***************************************************************/
+
+static char *create_subscriber_table =
+"CREATE TABLE subscriber(\
+subscriberNumber CHAR(12) NOT NULL primary key,\
+subscriberName CHAR(32) NOT NULL,\
+groupId INT NOT NULL,\
+location INT NOT NULL,\
+activeSessions INT NOT NULL,\
+changedBy CHAR(32) NOT NULL,\
+changedTime CHAR(32) NOT NULL)";
+
+static char *create_group_table =
+"CREATE TABLE userGroup(\
+groupId INT NOT NULL primary key,\
+groupName CHAR(32) NOT NULL,\
+allowRead INT NOT NULL,\
+allowInsert INT NOT NULL,\
+allowDelete INT NOT NULL)";
+
+static char *create_server_table = "CREATE TABLE server(\
+serverId INT NOT NULL,\
+subscriberSuffix CHAR(2) NOT NULL,\
+serverName CHAR(32) NOT NULL,\
+noOfRead INT NOT NULL,\
+noOfInsert INT NOT NULL,\
+noOfDelete INT NOT NULL,\
+PRIMARY KEY(serverId,subscriberSuffix))";
+
+static char *create_session_table =
+"CREATE TABLE userSession(\
+subscriberNumber CHAR(12) NOT NULL,\
+serverId INT NOT NULL,\
+sessionData CHAR(2000) NOT NULL,\
+PRIMARY KEY(subscriberNumber,serverId))";
+
+/***************************************************************
+* P U B L I C D A T A *
+***************************************************************/
+
+
+/***************************************************************
+****************************************************************
+* L O C A L F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+/***************************************************************
+****************************************************************
+* P U B L I C F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+/*-----------------------------------*/
+/* Time related Functions */
+/* */
+/* Returns a double value in seconds */
+/*-----------------------------------*/
+double userGetTime(void)
+{
+ static int initialized = 0;
+ static struct timeval initTime;
+ double timeValue;
+
+ if( !initialized ) {
+ initialized = 1;
+ gettimeofday(&initTime, 0);
+ timeValue = 0.0;
+ }
+ else {
+ struct timeval tv;
+ double s;
+ double us;
+
+ gettimeofday(&tv, 0);
+ s = (double)tv.tv_sec - (double)initTime.tv_sec;
+ us = (double)tv.tv_usec - (double)initTime.tv_usec;
+
+ timeValue = s + (us / 1000000.0);
+ }
+
+ return(timeValue);
+}
+
+
+void handle_error(SQLHDBC hdbc,
+ SQLHENV henv,
+ SQLHSTMT hstmt,
+ SQLRETURN rc,
+ char *filename,
+ int lineno)
+{
+#define MSG_LNG 512
+
+ int isError = 0;
+ SQLRETURN ret = SQL_SUCCESS;
+ SQLCHAR szSqlState[MSG_LNG]; /* SQL state string */
+ SQLCHAR szErrorMsg[MSG_LNG]; /* Error msg text buffer pointer */
+ SQLINTEGER pfNativeError; /* Native error code */
+ SQLSMALLINT pcbErrorMsg; /* Error msg text Available bytes */
+
+ if ( rc == SQL_SUCCESS || rc == SQL_NO_DATA_FOUND )
+ return;
+ else if ( rc == SQL_INVALID_HANDLE ) {
+ printf("ERROR in %s, line %d: invalid handle\n",
+ filename, lineno);
+ isError = 1;
+ }
+ else if ( rc == SQL_SUCCESS_WITH_INFO ) {
+ printf("WARNING in %s, line %d\n",
+ filename, lineno);
+ isError = 0;
+ }
+ else if ( rc == SQL_ERROR ) {
+ printf("ERROR in %s, line %d\n",
+ filename, lineno);
+ isError = 1;
+ }
+
+ fflush(stdout);
+
+ while ( ret == SQL_SUCCESS || ret == SQL_SUCCESS_WITH_INFO ) {
+ ret = SQLError(henv, hdbc, hstmt, szSqlState, &pfNativeError, szErrorMsg,
+ MSG_LNG, &pcbErrorMsg);
+
+ switch (ret) {
+ case SQL_SUCCESS:
+ case SQL_SUCCESS_WITH_INFO:
+ printf("%s\n*** ODBC Error/Warning = %s, "
+ "Additional Error/Warning = %d\n",
+ szErrorMsg, szSqlState, pfNativeError);
+
+ if(ret == SQL_SUCCESS_WITH_INFO)
+ printf("(Note: error message was truncated.\n");
+ break;
+
+ case SQL_INVALID_HANDLE:
+ printf("Call to SQLError failed with return code of "
+ "SQL_INVALID_HANDLE.\n");
+ break;
+
+ case SQL_ERROR:
+ printf("Call to SQLError failed with return code of SQL_ERROR.\n");
+ break;
+
+ case SQL_NO_DATA_FOUND:
+ break;
+
+ default:
+ printf("Call to SQLError failed with return code of %d.\n", ret);
+ }
+ }
+
+ if ( isError )
+ exit(1);
+}
+
+static int dbCreate(UserHandle *uh)
+{
+ SQLRETURN rc;
+ SQLHSTMT creatstmt;
+
+ if(!uh) return(-1);
+
+ rc = SQLAllocStmt(uh->hdbc, &creatstmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate create statement\n");
+ return(-1);
+ }
+
+ rc = SQLExecDirect(creatstmt,(SQLCHAR *)create_subscriber_table, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to create subscriber table\n");
+ return(-1);
+ }
+
+ rc = SQLExecDirect(creatstmt,(SQLCHAR *)create_group_table, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to create group table\n");
+ return(-1);
+ }
+
+ rc = SQLExecDirect(creatstmt,(SQLCHAR *)create_server_table, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to create server table\n");
+ return(-1);
+ }
+
+ rc = SQLExecDirect(creatstmt,(SQLCHAR *)create_session_table, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to create session table\n");
+ return(-1);
+ }
+
+ rc = SQLTransact(uh->henv, uh->hdbc, SQL_COMMIT);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to commit all create table\n");
+ return(-1);
+ }
+
+ rc = SQLFreeStmt(creatstmt, SQL_DROP);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to free create statement\n");
+ return(-1);
+ }
+
+ return(0);
+}
+
+UserHandle *userDbConnect(uint32 createDb, char *dbName)
+{
+ char connStrIn[512]; /* ODBC Connection String */
+ char connStrOut[2048];
+ SQLRETURN rc;
+ UserHandle *uh;
+
+ /*--------------------------*/
+ /* Build the Connect string */
+ /*--------------------------*/
+ sprintf(connStrIn,
+ "AutoCreate=%d;OverWrite=%d;DSN=%s",
+ createDb ? 1 : 0,
+ createDb ? 1 : 0,
+ dbName);
+
+ uh = calloc(1, sizeof(UserHandle));
+ if( !uh ) {
+ printf("Unable to allocate memory for Handle\n");
+ return(0);
+ }
+
+ /*---------------------------------*/
+ /* Allocate the Environment Handle */
+ /*---------------------------------*/
+ rc = SQLAllocEnv(&uh->henv);
+
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate Environment Handle\n");
+ return(0);
+ }
+
+ /*--------------------------------*/
+ /* Allocate the DB Connect Handle */
+ /*--------------------------------*/
+ rc = SQLAllocConnect(uh->henv, &uh->hdbc);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate a connection handle\n");
+ return(0);
+ }
+
+ /*-------------------------*/
+ /* Connect to the Database */
+ /*-------------------------*/
+ rc = SQLDriverConnect(uh->hdbc, NULL,
+ (SQLCHAR *)connStrIn, SQL_NTS,
+ (SQLCHAR *)connStrOut, sizeof (connStrOut),
+ NULL, SQL_DRIVER_NOPROMPT);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+handle_error(uh->hdbc, uh->henv, NULL, rc, __FILE__, __LINE__);
+ printf("Unable to connect to database server\n");
+ return(0);
+ }
+
+ rc = SQLSetConnectOption(uh->hdbc, SQL_AUTOCOMMIT, SQL_AUTOCOMMIT_OFF);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to set connection option\n");
+ return(0);
+ }
+
+ rc = SQLAllocStmt(uh->hdbc, &uh->stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to allocate immediate statement\n");
+ return(0);
+ }
+
+ if( createDb )
+ dbCreate(uh);
+
+ if( localDbPrepare(uh) < 0 )
+ return(0);
+
+ return(uh);
+}
+
+void userDbDisconnect(UserHandle *uh)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ rc = SQLDisconnect(uh->hdbc);
+
+ SQLFreeConnect(uh->hdbc);
+ SQLFreeEnv(uh->henv);
+ free(uh);
+}
+
+int userDbInsertServer(UserHandle *uh,
+ ServerId serverId,
+ SubscriberSuffix suffix,
+ ServerName name)
+{
+ SQLRETURN rc;
+ char buf[1000];
+
+ if(!uh) return(-1);
+
+ sprintf(buf, "insert into server values (%d,'%.*s','%s',0,0,0)",
+ serverId,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH, suffix,
+ name);
+
+ rc = SQLExecDirect(uh->stmt, (unsigned char *)buf, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to execute insert server\n");
+ return(-1);
+ }
+
+ return( userDbCommit(uh) );
+}
+
+int userDbInsertSubscriber(UserHandle *uh,
+ SubscriberNumber number,
+ uint32 groupId,
+ SubscriberName name)
+{
+ SQLRETURN rc;
+ char buf[1000];
+
+ if(!uh) return(-1);
+
+ sprintf(buf, "insert into subscriber values ('%s','%s',%d,0,0,'','')",
+ number,
+ name,
+ groupId);
+
+ rc = SQLExecDirect(uh->stmt, (unsigned char*)buf, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to execute insert subscriber\n");
+ return(-1);
+ }
+
+ return( userDbCommit(uh) );
+}
+
+int userDbInsertGroup(UserHandle *uh,
+ GroupId groupId,
+ GroupName name,
+ Permission allowRead,
+ Permission allowInsert,
+ Permission allowDelete)
+{
+ SQLRETURN rc;
+ char buf[1000];
+
+ if(!uh) return(-1);
+
+ sprintf(buf, "insert into usergroup values (%d,'%s',%d,%d,%d)",
+ groupId,
+ name,
+ allowRead,
+ allowInsert,
+ allowDelete);
+
+ rc = SQLExecDirect(uh->stmt, (unsigned char*)buf, SQL_NTS);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to execute insert group\n");
+ return(-1);
+ }
+
+ return( userDbCommit(uh) );
+}
+
+int userDbCommit(UserHandle *uh)
+{
+ SQLRETURN rc;
+ if(!uh) return(-1);
+
+ rc = SQLTransact(uh->henv, uh->hdbc, SQL_COMMIT);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+handle_error(uh->hdbc, uh->henv, 0, rc, __FILE__, __LINE__);
+ printf("Unable to commit Transaction\n");
+ return(-1);
+ }
+
+ return(0);
+}
+
+int userDbRollback(UserHandle *uh)
+{
+ SQLRETURN rc;
+ if(!uh) return(-1);
+
+ rc = SQLTransact(uh->henv, uh->hdbc, SQL_ROLLBACK);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("Unable to rollback Transaction\n");
+ return(-1);
+ }
+
+ return(0);
+}
+
+void userCheckpoint(UserHandle *uh)
+{
+ SQLRETURN rc;
+ if(!uh) return;
+
+ rc = SQLExecDirect(uh->stmt, (SQLCHAR *)"call ttCheckpointFuzzy", SQL_NTS);
+ userDbCommit(uh);
+}
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/old/userTransaction.c b/ndb/test/ndbapi/lmc-bench/src/user/old/userTransaction.c
new file mode 100644
index 00000000000..17069b0a042
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/old/userTransaction.c
@@ -0,0 +1,474 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <stdio.h>
+#include <string.h>
+#include <time.h>
+
+#include "sql.h"
+#include "sqlext.h"
+
+
+#include "userInterface.h"
+#include "userHandle.h"
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+static int readSubscriberSessions(UserHandle *uh,
+ SubscriberNumber number,
+ char *transactionType);
+
+/***************************************************************
+* L O C A L D A T A *
+***************************************************************/
+
+extern void handle_error(SQLHDBC hdbc,
+ SQLHENV henv,
+ SQLHSTMT hstmt,
+ SQLRETURN rc,
+ char *filename,
+ int lineno);
+
+/***************************************************************
+* P U B L I C D A T A *
+***************************************************************/
+
+
+/***************************************************************
+****************************************************************
+* L O C A L F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+static int readSubscriberSessions(UserHandle *uh,
+ SubscriberNumber number,
+ char *transactionType)
+{
+ SQLRETURN rc;
+
+ /*-----------------------------------------------------*/
+ /* SELECT activeSessions,groupId,changedBy,changedTime */
+ /* FROM SUBSCRIBER */
+ /* WHERE subscriberNumber=x; */
+ /*-----------------------------------------------------*/
+ strcpy(uh->readSubscriberSession.values.number,number);
+
+ rc = SQLExecute(uh->readSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("%s %s\n",
+ transactionType,
+ "Unable to execute read subscriber session");
+ return(-1);
+ }
+
+ rc = SQLFetch(uh->readSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("%s %s\n",
+ transactionType,
+ "Unable to fetch read subscriber session");
+ return(-1);
+ }
+
+ return(0);
+}
+
+/***************************************************************
+****************************************************************
+* P U B L I C F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+void userTransaction_T1(UserHandle *uh,
+ SubscriberNumber number,
+ Location new_location,
+ ChangedBy changed_by,
+ ChangedTime changed_time)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ /*---------------------------------------------*/
+ /* Update the subscriber information */
+ /* */
+ /* UPDATE SUBSCRIBER */
+ /* SET location=x, changedBy=x, changedTime=x */
+ /* WHERE subscriberNumber=x; */
+ /*---------------------------------------------*/
+ strcpy(uh->updateSubscriber.values.number, number);
+ uh->updateSubscriber.values.location = new_location;
+ strcpy(uh->updateSubscriber.values.changedBy, changed_by);
+ strcpy(uh->updateSubscriber.values.changedTime, changed_time);
+
+ rc = SQLExecute(uh->updateSubscriber.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T1 Unable to execute update subscriber\n");
+ return;
+ }
+
+ userDbCommit(uh);
+}
+
+void userTransaction_T2(UserHandle *uh,
+ SubscriberNumber number,
+ Location *new_location,
+ ChangedBy changed_by,
+ ChangedTime changed_time,
+ SubscriberName subscriberName)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ /*------------------------------------------------------*/
+ /* Read the information from the subscriber table */
+ /* */
+ /* SELECT location,subscriberName,changedBy,changedTime */
+ /* FROM SUBSCRIBER */
+ /* WHERE subscriberNumber=x; */
+ /*------------------------------------------------------*/
+ strcpy(uh->readSubscriber.values.number,number);
+
+ rc = SQLExecute(uh->readSubscriber.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T2 Unable to execute read subscriber\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readSubscriber.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T2 Unable to fetch read subscriber\n");
+ return;
+ }
+
+ userDbCommit(uh);
+
+ strcpy(subscriberName, uh->readSubscriber.values.name);
+ *new_location = uh->readSubscriber.values.location;
+ strcpy(changed_by, uh->readSubscriber.values.changedBy);
+ strcpy(changed_time, uh->readSubscriber.values.changedTime);
+}
+
+void userTransaction_T3(UserHandle *uh,
+ SubscriberNumber number,
+ ServerId server_id,
+ ServerBit server_bit,
+ SessionDetails session_details,
+ unsigned int *branch_executed)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ *branch_executed = 0;
+
+ /*--------------------------------------*/
+ /* Read active sessions from subscriber */
+ /*--------------------------------------*/
+ if( readSubscriberSessions(uh, number, "T3") < 0 )
+ return;
+
+ /*-----------------------------------------------*/
+ /* Read the 'read' Permissions for the userGroup */
+ /* */
+ /* SELECT allowRead */
+ /* FROM USERGROUP */
+ /* WHERE groupId=x */
+ /*-----------------------------------------------*/
+ uh->readGroupAllowRead.values.groupId = uh->readSubscriberSession.values.groupId;
+
+ rc = SQLExecute(uh->readGroupAllowRead.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to execute read group allow read\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readGroupAllowRead.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to fetch read group allow read\n");
+ return;
+ }
+
+ if( uh->readGroupAllowRead.values.allowRead & server_bit &&
+ uh->readSubscriberSession.values.activeSessions & server_bit ) {
+
+ /*----------------------------------------------------*/
+ /* Read the sessionDetails from the userSession table */
+ /* */
+ /* SELECT sessionData */
+ /* FROM userSession */
+ /* WHERE subscriberNumber=x, serverId=x */
+ /*----------------------------------------------------*/
+ strcpy(uh->readSessionDetails.values.number,number);
+ uh->readSessionDetails.values.serverId = server_id;
+
+ rc = SQLExecute(uh->readSessionDetails.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to execute read session details\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readSessionDetails.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to fetch read session details\n");
+ return;
+ }
+
+ strcpy(session_details, uh->readSessionDetails.values.details);
+
+ /*----------------------------------------*/
+ /* Increment noOfRead field in the server */
+ /* */
+ /* UPDATE server */
+ /* SET noOfRead=noOfRead+1 */
+ /* WHERE serverId=x,subscriberSuffix=x */
+ /*----------------------------------------*/
+ uh->updateServerNoOfRead.values.serverId = server_id;
+ strcpy(uh->updateServerNoOfRead.values.suffix,
+ &number[SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH]);
+
+ rc = SQLExecute(uh->updateServerNoOfRead.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to execute read no of read\n");
+ return;
+ }
+
+ *branch_executed = 1;
+ }
+
+ userDbCommit(uh);
+}
+
+void userTransaction_T4(UserHandle *uh,
+ SubscriberNumber number,
+ ServerId server_id,
+ ServerBit server_bit,
+ SessionDetails session_details,
+ unsigned int do_rollback,
+ unsigned int *branch_executed)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ *branch_executed = 0;
+
+ /*--------------------------------------*/
+ /* Read active sessions from subscriber */
+ /*--------------------------------------*/
+ if( readSubscriberSessions(uh, number, "T4") < 0 )
+ return;
+
+ /*-------------------------------------------------*/
+ /* Read the 'insert' Permissions for the userGroup */
+ /* */
+ /* SELECT allowInsert */
+ /* FROM USERGROUP */
+ /* WHERE groupId=x */
+ /*-------------------------------------------------*/
+ uh->readGroupAllowInsert.values.groupId = uh->readSubscriberSession.values.groupId;
+
+ rc = SQLExecute(uh->readGroupAllowInsert.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T4 Unable to execute read group allow insert\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readGroupAllowInsert.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T4 Unable to fetch read group allow insert\n");
+ return;
+ }
+
+ if( uh->readGroupAllowInsert.values.allowInsert & server_bit &&
+ !(uh->readSubscriberSession.values.activeSessions & server_bit) ) {
+
+ /*---------------------------------------------*/
+ /* Insert the session to the userSession table */
+ /* */
+ /* INSERT INTO userSession */
+ /* VALUES (x,x,x) */
+ /*---------------------------------------------*/
+ strcpy(uh->insertSession.values.number, number);
+ uh->insertSession.values.serverId = server_id;
+ strcpy(uh->insertSession.values.details, session_details);
+
+ rc = SQLExecute(uh->insertSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+handle_error(uh->hdbc, uh->henv, uh->insertSession.stmt, rc, __FILE__, __LINE__);
+ printf("T4 Unable to execute insert session \n");
+ return;
+ }
+
+ /*----------------------------------------*/
+ /* Update subscriber activeSessions field */
+ /* */
+ /* UPDATE subscriber */
+ /* SET activeSessions=x */
+ /* WHERE subscriberNumber=x */
+ /*----------------------------------------*/
+ strcpy(uh->updateSubscriberSession.values.number, number);
+ uh->updateSubscriberSession.values.activeSessions =
+ uh->readSubscriberSession.values.activeSessions | server_bit;
+
+ rc = SQLExecute(uh->updateSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T4 Unable to execute update session \n");
+ return;
+ }
+
+ /*------------------------------------------*/
+ /* Increment noOfInsert field in the server */
+ /* */
+ /* UPDATE server */
+ /* SET noOfInsert=noOfInsert+1 */
+ /* WHERE serverId=x,subscriberSuffix=x */
+ /*------------------------------------------*/
+ uh->updateServerNoOfInsert.values.serverId = server_id;
+ strcpy(uh->updateServerNoOfInsert.values.suffix,
+ &number[SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH]);
+
+ rc = SQLExecute(uh->updateServerNoOfInsert.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T4 Unable to execute update no of read\n");
+ return;
+ }
+
+ *branch_executed = 1;
+ }
+
+ if(do_rollback)
+ userDbRollback(uh);
+ else
+ userDbCommit(uh);
+}
+
+void userTransaction_T5(UserHandle *uh,
+ SubscriberNumber number,
+ ServerId server_id,
+ ServerBit server_bit,
+ unsigned int do_rollback,
+ unsigned int *branch_executed)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ *branch_executed = 0;
+
+ /*--------------------------------------*/
+ /* Read active sessions from subscriber */
+ /*--------------------------------------*/
+ if( readSubscriberSessions(uh, number, "T5") < 0 )
+ return;
+
+ /*-------------------------------------------------*/
+ /* Read the 'delete' Permissions for the userGroup */
+ /* */
+ /* SELECT allowDelete */
+ /* FROM USERGROUP */
+ /* WHERE groupId=x */
+ /*-------------------------------------------------*/
+ uh->readGroupAllowDelete.values.groupId = uh->readSubscriberSession.values.groupId;
+
+ rc = SQLExecute(uh->readGroupAllowDelete.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to execute read group allow delete\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readGroupAllowDelete.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to fetch read group allow delete\n");
+ return;
+ }
+
+ if( uh->readGroupAllowDelete.values.allowDelete & server_bit &&
+ uh->readSubscriberSession.values.activeSessions & server_bit ) {
+
+ /*-----------------------------------------------*/
+ /* Delete the session from the userSession table */
+ /* */
+ /* DELETE FROM userSession */
+ /* WHERE subscriberNumber=x,serverId=x */
+ /*-----------------------------------------------*/
+ strcpy(uh->deleteSession.values.number,number);
+ uh->deleteSession.values.serverId = server_id;
+
+ rc = SQLExecute(uh->deleteSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to execute delete session\n");
+ return;
+ }
+
+ /*----------------------------------------*/
+ /* Update subscriber activeSessions field */
+ /* */
+ /* UPDATE subscriber */
+ /* SET activeSessions=x */
+ /* WHERE subscriberNumber=x */
+ /*----------------------------------------*/
+ strcpy(uh->updateSubscriberSession.values.number, number);
+ uh->updateSubscriberSession.values.activeSessions =
+ uh->readSubscriberSession.values.activeSessions & ~server_bit;
+
+ rc = SQLExecute(uh->updateSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to execute update subscriber session \n");
+ return;
+ }
+
+ /*------------------------------------------*/
+ /* Increment noOfDelete field in the server */
+ /* */
+ /* UPDATE server */
+ /* SET noOfDelete=noOfDelete+1 */
+ /* WHERE serverId=x,subscriberSuffix=x */
+ /*------------------------------------------*/
+ uh->updateServerNoOfDelete.values.serverId = server_id;
+ strcpy(uh->updateServerNoOfDelete.values.suffix,
+ &number[SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH]);
+
+ rc = SQLExecute(uh->updateServerNoOfDelete.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to execute update no of delete\n");
+ return;
+ }
+
+ *branch_executed = 1;
+ }
+
+ if(do_rollback)
+ userDbRollback(uh);
+ else
+ userDbCommit(uh);
+}
+
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/userHandle.h b/ndb/test/ndbapi/lmc-bench/src/user/userHandle.h
new file mode 100644
index 00000000000..6da76fc2bff
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/userHandle.h
@@ -0,0 +1,51 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef USERHANDLE_H
+#define USERHANDLE_H
+
+/***************************************************************/
+/* I N C L U D E D F I L E S */
+/***************************************************************/
+
+#include <NdbApi.hpp>
+#include "testDefinitions.h"
+
+/***************************************************************
+* M A C R O S *
+***************************************************************/
+
+/***************************************************************/
+/* C O N S T A N T S */
+/***************************************************************/
+
+/***************************************************************
+* D A T A S T R U C T U R E S *
+***************************************************************/
+
+typedef Ndb userHandle;
+
+/***************************************************************
+* P U B L I C F U N C T I O N S *
+***************************************************************/
+
+/***************************************************************
+* E X T E R N A L D A T A *
+***************************************************************/
+
+
+#endif /* USERHANDLE_H */
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/userInterface.cpp b/ndb/test/ndbapi/lmc-bench/src/user/userInterface.cpp
new file mode 100644
index 00000000000..fc3f6955a47
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/userInterface.cpp
@@ -0,0 +1,742 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <sys/types.h>
+#ifndef NDB_WIN32
+#include <sys/time.h>
+#endif
+
+#include "ndb_error.hpp"
+#include "userHandle.h"
+#include "userInterface.h"
+#include <NdbThread.h>
+#include <NdbTick.h>
+#include <NdbMutex.h>
+#include <NdbSleep.h>
+#include "ndb_schema.hpp"
+#include <NDBT.hpp>
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+extern int localDbPrepare(UserHandle *uh);
+
+static int dbCreate(UserHandle *uh);
+
+/***************************************************************
+* L O C A L D A T A *
+***************************************************************/
+
+/***************************************************************
+* P U B L I C D A T A *
+***************************************************************/
+
+
+/***************************************************************
+****************************************************************
+* L O C A L F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+/***************************************************************
+****************************************************************
+* P U B L I C F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+/*-----------------------------------*/
+/* Time related Functions */
+/* */
+/* Returns a double value in seconds */
+/*-----------------------------------*/
+double userGetTimeSync(void)
+{
+ static int initialized = 0;
+ static NDB_TICKS initSecs = 0;
+ static Uint32 initMicros = 0;
+ double timeValue = 0;
+
+ if ( !initialized ) {
+ initialized = 1;
+ NdbTick_CurrentMicrosecond(&initSecs, &initMicros);
+ timeValue = 0.0;
+ } else {
+ NDB_TICKS secs = 0;
+ Uint32 micros = 0;
+
+ NdbTick_CurrentMicrosecond(&secs, &micros);
+
+ double s = (double)secs - (double)initSecs;
+ double us = (double)secs - (double)initMicros;
+
+ timeValue = s + (us / 1000000.0);
+ }
+
+ return timeValue;
+}
+
+// 0 - OK
+// 1 - Retry transaction
+// 2 - Permanent
+int
+userDbCommit(UserHandle *uh){
+ if(uh->pCurrTrans != 0){
+ int check = uh->pCurrTrans->execute( Commit );
+ NdbError err = uh->pCurrTrans->getNdbError();
+ uh->pNDB->closeTransaction(uh->pCurrTrans);
+ uh->pCurrTrans = 0;
+
+ if(err.status != NdbError::Success)
+ ndbout << err << endl;
+
+ if(err.status == NdbError::TemporaryError &&
+ err.classification == NdbError::OverloadError){
+ NdbSleep_SecSleep(3);
+ }
+
+ return err.status;
+ }
+ return 2;
+}
+
+/**
+ * TRUE - Normal table
+ * FALSE - Table w.o. checkpoing and logging
+ */
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+extern int useTableLogging;
+extern int useIndexTables;
+#ifdef __cplusplus
+}
+#endif
+
+
+int
+create_table_server(Ndb * pNDB){
+
+ int check;
+
+ NdbSchemaCon * MySchemaTransaction = pNDB->startSchemaTransaction();
+ if( MySchemaTransaction == NULL )
+ error_handler("startSchemaTransaction", pNDB->getNdbError(), 0);
+
+ NdbSchemaOp * MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL )
+ error_handler("getNdbSchemaOp", MySchemaTransaction->getNdbError(), 0);
+
+ // Create table
+ check = MySchemaOp->createTable( SERVER_TABLE,
+ 8, // Table size
+ TupleKey, // Key Type
+ 1 // Nr of Pages
+ ,DistributionGroup,
+ 6,
+ 78,
+ 80,
+ 1,
+ useTableLogging
+ );
+ if( check == -1 )
+ error_handler("createTable", MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute
+ ( SERVER_SUBSCRIBER_SUFFIX,
+ TupleKey,
+ sizeof(char) << 3,
+ SUBSCRIBER_NUMBER_SUFFIX_LENGTH,
+ String,
+ MMBased,
+ NotNullAttribute,
+ NormalStorageAttribute,
+ 0,
+ 1,
+ 16);
+ if( check == -1 )
+ error_handler("createAttribute (subscriber suffix)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ // Create first column, primary key
+ check = MySchemaOp->createAttribute( SERVER_ID,
+ TupleKey,
+ sizeof(ServerId) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (serverid)",
+ MySchemaTransaction->getNdbError(), 0);
+
+
+ check = MySchemaOp->createAttribute( SERVER_NAME,
+ NoKey,
+ sizeof(char) << 3,
+ SERVER_NAME_LENGTH,
+ String,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (server name)",
+ MySchemaTransaction->getNdbError(), 0);
+
+
+ check = MySchemaOp->createAttribute( SERVER_READS,
+ NoKey,
+ sizeof(Counter) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (server reads)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute( SERVER_INSERTS,
+ NoKey,
+ sizeof(Counter) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (server inserts)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute( SERVER_DELETES,
+ NoKey,
+ sizeof(Counter) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (server deletes)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ if( MySchemaTransaction->execute() == -1 ) {
+ error_handler("schemaTransaction->execute()",
+ MySchemaTransaction->getNdbError(), 0);
+ }
+ pNDB->closeSchemaTransaction(MySchemaTransaction);
+ return 0;
+}
+
+int
+create_table_group(Ndb * pNDB){
+ int check;
+
+ NdbSchemaCon * MySchemaTransaction = pNDB->startSchemaTransaction();
+ if( MySchemaTransaction == NULL )
+ error_handler("startSchemaTransaction", pNDB->getNdbError(), 0);
+
+ NdbSchemaOp * MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL )
+ error_handler("getNdbSchemaOp", MySchemaTransaction->getNdbError(), 0);
+
+ // Create table
+ check = MySchemaOp->createTable( GROUP_TABLE,
+ 8, // Table size
+ TupleKey, // Key Type
+ 1 // Nr of Pages
+ ,All,
+ 6,
+ 78,
+ 80,
+ 1,
+ useTableLogging
+ );
+
+ if( check == -1 )
+ error_handler("createTable", MySchemaTransaction->getNdbError(), 0);
+
+ // Create first column, primary key
+ check = MySchemaOp->createAttribute( GROUP_ID,
+ TupleKey,
+ sizeof(GroupId) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (group id)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute( GROUP_NAME,
+ NoKey,
+ sizeof(char) << 3,
+ GROUP_NAME_LENGTH,
+ String,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (group name)",
+ MySchemaTransaction->getNdbError(), 0);
+
+
+ check = MySchemaOp->createAttribute( GROUP_ALLOW_READ,
+ NoKey,
+ sizeof(Permission) << 3,
+ 1,
+ String,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (group read)",
+ MySchemaTransaction->getNdbError(), 0);
+
+
+ check = MySchemaOp->createAttribute( GROUP_ALLOW_INSERT,
+ NoKey,
+ sizeof(Permission) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (group insert)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute( GROUP_ALLOW_DELETE,
+ NoKey,
+ sizeof(Permission) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (group delete)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ if( MySchemaTransaction->execute() == -1 ) {
+ error_handler("schemaTransaction->execute()",
+ MySchemaTransaction->getNdbError(), 0);
+ }
+ pNDB->closeSchemaTransaction(MySchemaTransaction);
+ return 0;
+}
+
+int
+create_table_subscriber(Ndb * pNDB){
+ int check;
+ NdbSchemaCon * MySchemaTransaction = pNDB->startSchemaTransaction();
+ if( MySchemaTransaction == NULL )
+ error_handler("startSchemaTransaction", pNDB->getNdbError(), 0);
+
+ NdbSchemaOp * MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL )
+ error_handler("getNdbSchemaOp", MySchemaTransaction->getNdbError(), 0);
+
+ // Create table
+ check = MySchemaOp->createTable( SUBSCRIBER_TABLE,
+ 8, // Table size
+ TupleKey, // Key Type
+ 1 // Nr of Pages
+ ,DistributionGroup,
+ 6,
+ 78,
+ 80,
+ 1,
+ useTableLogging
+ );
+ if( check == -1 )
+ error_handler("createTable", MySchemaTransaction->getNdbError(), 0);
+
+ // Create first column, primary key
+ check = MySchemaOp->createAttribute
+ ( SUBSCRIBER_NUMBER,
+ TupleKey,
+ sizeof(char) << 3,
+ SUBSCRIBER_NUMBER_LENGTH,
+ String,
+ MMBased,
+ NotNullAttribute,
+ (useIndexTables ? IndexStorageAttribute : NormalStorageAttribute),
+ 0,
+ 1,
+ 16);
+ if( check == -1 )
+ error_handler("createAttribute (subscriber number)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute( SUBSCRIBER_NAME,
+ NoKey,
+ sizeof(char) << 3,
+ SUBSCRIBER_NAME_LENGTH,
+ String,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (subscriber name)",
+ MySchemaTransaction->getNdbError(), 0);
+
+
+ check = MySchemaOp->createAttribute( SUBSCRIBER_GROUP,
+ NoKey,
+ sizeof(GroupId) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (subscriber_group)",
+ MySchemaTransaction->getNdbError(), 0);
+
+
+ check = MySchemaOp->createAttribute( SUBSCRIBER_LOCATION,
+ NoKey,
+ sizeof(Location) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (server reads)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute( SUBSCRIBER_SESSIONS,
+ NoKey,
+ sizeof(ActiveSessions) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (subscriber_sessions)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute( SUBSCRIBER_CHANGED_BY,
+ NoKey,
+ sizeof(char) << 3,
+ CHANGED_BY_LENGTH,
+ String,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (subscriber_changed_by)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute( SUBSCRIBER_CHANGED_TIME,
+ NoKey,
+ sizeof(char) << 3,
+ CHANGED_TIME_LENGTH,
+ String,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (subscriber_changed_time)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ if( MySchemaTransaction->execute() == -1 ) {
+ error_handler("schemaTransaction->execute()",
+ MySchemaTransaction->getNdbError(), 0);
+ }
+ pNDB->closeSchemaTransaction(MySchemaTransaction);
+ return 0;
+}
+
+int
+create_table_session(Ndb * pNDB){
+ int check;
+ NdbSchemaCon * MySchemaTransaction = pNDB->startSchemaTransaction();
+ if( MySchemaTransaction == NULL )
+ error_handler("startSchemaTransaction", pNDB->getNdbError(), 0);
+
+ NdbSchemaOp * MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL )
+ error_handler("getNdbSchemaOp",
+ MySchemaTransaction->getNdbError(), 0);
+
+ // Create table
+ check = MySchemaOp->createTable( SESSION_TABLE,
+ 8, // Table size
+ TupleKey, // Key Type
+ 1 // Nr of Pages
+ ,DistributionGroup,
+ 6,
+ 78,
+ 80,
+ 1,
+ useTableLogging
+ );
+ if( check == -1 )
+ error_handler("createTable", MySchemaTransaction->getNdbError(), 0);
+
+ check = MySchemaOp->createAttribute( SESSION_SUBSCRIBER,
+ TupleKey,
+ sizeof(char) << 3,
+ SUBSCRIBER_NUMBER_LENGTH,
+ String,
+ MMBased,
+ NotNullAttribute,
+ NormalStorageAttribute,
+ 0,
+ 1,
+ 16);
+ if( check == -1 )
+ error_handler("createAttribute (session_subscriber)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ // Create first column, primary key
+ check = MySchemaOp->createAttribute( SESSION_SERVER,
+ TupleKey,
+ sizeof(ServerId) << 3,
+ 1,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (session_server)",
+ MySchemaTransaction->getNdbError(), 0);
+
+
+ check = MySchemaOp->createAttribute( SESSION_DATA,
+ NoKey,
+ sizeof(char) << 3,
+ SESSION_DETAILS_LENGTH,
+ String,
+ MMBased,
+ NotNullAttribute );
+ if( check == -1 )
+ error_handler("createAttribute (session_data)",
+ MySchemaTransaction->getNdbError(), 0);
+
+ if( MySchemaTransaction->execute() == -1 ) {
+ error_handler("schemaTransaction->execute()",
+ MySchemaTransaction->getNdbError(), 0);
+ }
+ pNDB->closeSchemaTransaction(MySchemaTransaction);
+ return 0;
+}
+
+void
+create_table(const char * name, int (* function)(Ndb * pNDB), Ndb* pNDB){
+ printf("creating table %s...", name);
+ if(pNDB->getDictionary()->getTable(name) != 0){
+ printf(" it already exists\n");
+ return;
+ } else {
+ printf("\n");
+ }
+ function(pNDB);
+ printf("creating table %s... done\n", name);
+}
+
+static int dbCreate(Ndb * pNDB)
+{
+ create_table(SUBSCRIBER_TABLE, create_table_subscriber, pNDB);
+ create_table(GROUP_TABLE , create_table_group, pNDB);
+ create_table(SESSION_TABLE , create_table_session, pNDB);
+ create_table(SERVER_TABLE , create_table_server, pNDB);
+ return 0;
+}
+
+#ifndef NDB_WIN32
+#include <unistd.h>
+#endif
+
+static NdbMutex* startupMutex = NdbMutex_Create();
+
+UserHandle*
+userDbConnect(uint32 createDb, char *dbName)
+{
+ NdbMutex_Lock(startupMutex);
+
+ Ndb * pNDB = new Ndb("");
+
+ //printf("Initializing...\n");
+ pNDB->init();
+
+ //printf("Waiting...");
+ while(pNDB->waitUntilReady() != 0){
+ //printf("...");
+ }
+ // printf("done\n");
+
+ if( createDb )
+ dbCreate(pNDB);
+
+
+ UserHandle * uh = new UserHandle;
+ uh->pNDB = pNDB;
+ uh->pCurrTrans = 0;
+
+ NdbMutex_Unlock(startupMutex);
+
+ return uh;
+}
+
+void userDbDisconnect(UserHandle *uh)
+{
+ delete uh;
+}
+
+int userDbInsertServer(UserHandle *uh,
+ ServerId serverId,
+ SubscriberSuffix suffix,
+ ServerName name)
+{
+ int check;
+
+ uint32 noOfRead = 0;
+ uint32 noOfInsert = 0;
+ uint32 noOfDelete = 0;
+
+ NdbConnection * MyTransaction = 0;
+ if(uh->pCurrTrans != 0){
+ MyTransaction = uh->pCurrTrans;
+ } else {
+ uh->pCurrTrans = MyTransaction = uh->pNDB->startTransaction();
+ }
+ if (MyTransaction == NULL)
+ error_handler("startTranscation", uh->pNDB->getNdbError(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SERVER_TABLE);
+ CHECK_NULL(MyOperation, "getNdbOperation", MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "insert tuple", MyTransaction);
+
+ check = MyOperation->equal(SERVER_ID, (char*)&serverId);
+ CHECK_MINUS_ONE(check, "setValue id", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_SUBSCRIBER_SUFFIX, suffix);
+ CHECK_MINUS_ONE(check, "setValue suffix", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_NAME, name);
+ CHECK_MINUS_ONE(check, "setValue name", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_READS, (char*)&noOfRead);
+ CHECK_MINUS_ONE(check, "setValue reads", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_INSERTS, (char*)&noOfInsert);
+ CHECK_MINUS_ONE(check, "setValue inserts", MyTransaction);
+
+ check = MyOperation->setValue(SERVER_DELETES, (char*)&noOfDelete);
+ CHECK_MINUS_ONE(check, "setValue deletes", MyTransaction);
+
+ return 0;
+}
+
+int userDbInsertSubscriber(UserHandle *uh,
+ SubscriberNumber number,
+ uint32 groupId,
+ SubscriberName name)
+{
+ int check;
+ uint32 activeSessions = 0;
+ Location l = 0;
+ ChangedBy changedBy; snprintf(changedBy, sizeof(changedBy), "ChangedBy");
+ ChangedTime changedTime; snprintf(changedTime, sizeof(changedTime), "ChangedTime");
+
+ NdbConnection * MyTransaction = 0;
+ if(uh->pCurrTrans != 0){
+ MyTransaction = uh->pCurrTrans;
+ } else {
+ uh->pCurrTrans = MyTransaction = uh->pNDB->startTransaction();
+ }
+ if (MyTransaction == NULL)
+ error_handler("startTranscation", uh->pNDB->getNdbError(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(SUBSCRIBER_TABLE);
+ CHECK_NULL(MyOperation, "getNdbOperation", MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "insertTuple", MyTransaction);
+
+ check = MyOperation->equal(SUBSCRIBER_NUMBER, number);
+ CHECK_MINUS_ONE(check, "equal", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_NAME, name);
+ CHECK_MINUS_ONE(check, "setValue name", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_GROUP, (char*)&groupId);
+ CHECK_MINUS_ONE(check, "setValue group", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_LOCATION, (char*)&l);
+ CHECK_MINUS_ONE(check, "setValue location", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_SESSIONS, (char*)&activeSessions);
+ CHECK_MINUS_ONE(check, "setValue sessions", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_CHANGED_BY, changedBy);
+ CHECK_MINUS_ONE(check, "setValue changedBy", MyTransaction);
+
+ check = MyOperation->setValue(SUBSCRIBER_CHANGED_TIME, changedTime);
+ CHECK_MINUS_ONE(check, "setValue changedTime", MyTransaction);
+
+ return 0;
+}
+
+int userDbInsertGroup(UserHandle *uh,
+ GroupId groupId,
+ GroupName name,
+ Permission allowRead,
+ Permission allowInsert,
+ Permission allowDelete)
+{
+ int check;
+
+ NdbConnection * MyTransaction = 0;
+ if(uh->pCurrTrans != 0){
+ MyTransaction = uh->pCurrTrans;
+ } else {
+ uh->pCurrTrans = MyTransaction = uh->pNDB->startTransaction();
+ }
+ if (MyTransaction == NULL)
+ error_handler("startTranscation", uh->pNDB->getNdbError(), 0);
+
+ NdbOperation *MyOperation = MyTransaction->getNdbOperation(GROUP_TABLE);
+ CHECK_NULL(MyOperation, "getNdbOperation", MyTransaction);
+
+ check = MyOperation->insertTuple();
+ CHECK_MINUS_ONE(check, "insertTuple", MyTransaction);
+
+ check = MyOperation->equal(GROUP_ID, (char*)&groupId);
+ CHECK_MINUS_ONE(check, "equal", MyTransaction);
+
+ check = MyOperation->setValue(GROUP_NAME, name);
+ CHECK_MINUS_ONE(check, "setValue name", MyTransaction);
+
+ check = MyOperation->setValue(GROUP_ALLOW_READ, (char*)&allowRead);
+ CHECK_MINUS_ONE(check, "setValue allowRead", MyTransaction);
+
+ check = MyOperation->setValue(GROUP_ALLOW_INSERT, (char*)&allowInsert);
+ CHECK_MINUS_ONE(check, "setValue allowInsert", MyTransaction);
+
+ check = MyOperation->setValue(GROUP_ALLOW_DELETE, (char*)&allowDelete);
+ CHECK_MINUS_ONE(check, "setValue allowDelete", MyTransaction);
+
+ return 0;
+}
+
diff --git a/ndb/test/ndbapi/lmc-bench/src/user/userTransaction.c b/ndb/test/ndbapi/lmc-bench/src/user/userTransaction.c
new file mode 100644
index 00000000000..17069b0a042
--- /dev/null
+++ b/ndb/test/ndbapi/lmc-bench/src/user/userTransaction.c
@@ -0,0 +1,474 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/***************************************************************
+* I N C L U D E D F I L E S *
+***************************************************************/
+
+#include <stdio.h>
+#include <string.h>
+#include <time.h>
+
+#include "sql.h"
+#include "sqlext.h"
+
+
+#include "userInterface.h"
+#include "userHandle.h"
+
+/***************************************************************
+* L O C A L C O N S T A N T S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L D A T A S T R U C T U R E S *
+***************************************************************/
+
+/***************************************************************
+* L O C A L F U N C T I O N S *
+***************************************************************/
+
+static int readSubscriberSessions(UserHandle *uh,
+ SubscriberNumber number,
+ char *transactionType);
+
+/***************************************************************
+* L O C A L D A T A *
+***************************************************************/
+
+extern void handle_error(SQLHDBC hdbc,
+ SQLHENV henv,
+ SQLHSTMT hstmt,
+ SQLRETURN rc,
+ char *filename,
+ int lineno);
+
+/***************************************************************
+* P U B L I C D A T A *
+***************************************************************/
+
+
+/***************************************************************
+****************************************************************
+* L O C A L F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+static int readSubscriberSessions(UserHandle *uh,
+ SubscriberNumber number,
+ char *transactionType)
+{
+ SQLRETURN rc;
+
+ /*-----------------------------------------------------*/
+ /* SELECT activeSessions,groupId,changedBy,changedTime */
+ /* FROM SUBSCRIBER */
+ /* WHERE subscriberNumber=x; */
+ /*-----------------------------------------------------*/
+ strcpy(uh->readSubscriberSession.values.number,number);
+
+ rc = SQLExecute(uh->readSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("%s %s\n",
+ transactionType,
+ "Unable to execute read subscriber session");
+ return(-1);
+ }
+
+ rc = SQLFetch(uh->readSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("%s %s\n",
+ transactionType,
+ "Unable to fetch read subscriber session");
+ return(-1);
+ }
+
+ return(0);
+}
+
+/***************************************************************
+****************************************************************
+* P U B L I C F U N C T I O N S C O D E S E C T I O N *
+****************************************************************
+***************************************************************/
+
+void userTransaction_T1(UserHandle *uh,
+ SubscriberNumber number,
+ Location new_location,
+ ChangedBy changed_by,
+ ChangedTime changed_time)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ /*---------------------------------------------*/
+ /* Update the subscriber information */
+ /* */
+ /* UPDATE SUBSCRIBER */
+ /* SET location=x, changedBy=x, changedTime=x */
+ /* WHERE subscriberNumber=x; */
+ /*---------------------------------------------*/
+ strcpy(uh->updateSubscriber.values.number, number);
+ uh->updateSubscriber.values.location = new_location;
+ strcpy(uh->updateSubscriber.values.changedBy, changed_by);
+ strcpy(uh->updateSubscriber.values.changedTime, changed_time);
+
+ rc = SQLExecute(uh->updateSubscriber.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T1 Unable to execute update subscriber\n");
+ return;
+ }
+
+ userDbCommit(uh);
+}
+
+void userTransaction_T2(UserHandle *uh,
+ SubscriberNumber number,
+ Location *new_location,
+ ChangedBy changed_by,
+ ChangedTime changed_time,
+ SubscriberName subscriberName)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ /*------------------------------------------------------*/
+ /* Read the information from the subscriber table */
+ /* */
+ /* SELECT location,subscriberName,changedBy,changedTime */
+ /* FROM SUBSCRIBER */
+ /* WHERE subscriberNumber=x; */
+ /*------------------------------------------------------*/
+ strcpy(uh->readSubscriber.values.number,number);
+
+ rc = SQLExecute(uh->readSubscriber.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T2 Unable to execute read subscriber\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readSubscriber.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T2 Unable to fetch read subscriber\n");
+ return;
+ }
+
+ userDbCommit(uh);
+
+ strcpy(subscriberName, uh->readSubscriber.values.name);
+ *new_location = uh->readSubscriber.values.location;
+ strcpy(changed_by, uh->readSubscriber.values.changedBy);
+ strcpy(changed_time, uh->readSubscriber.values.changedTime);
+}
+
+void userTransaction_T3(UserHandle *uh,
+ SubscriberNumber number,
+ ServerId server_id,
+ ServerBit server_bit,
+ SessionDetails session_details,
+ unsigned int *branch_executed)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ *branch_executed = 0;
+
+ /*--------------------------------------*/
+ /* Read active sessions from subscriber */
+ /*--------------------------------------*/
+ if( readSubscriberSessions(uh, number, "T3") < 0 )
+ return;
+
+ /*-----------------------------------------------*/
+ /* Read the 'read' Permissions for the userGroup */
+ /* */
+ /* SELECT allowRead */
+ /* FROM USERGROUP */
+ /* WHERE groupId=x */
+ /*-----------------------------------------------*/
+ uh->readGroupAllowRead.values.groupId = uh->readSubscriberSession.values.groupId;
+
+ rc = SQLExecute(uh->readGroupAllowRead.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to execute read group allow read\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readGroupAllowRead.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to fetch read group allow read\n");
+ return;
+ }
+
+ if( uh->readGroupAllowRead.values.allowRead & server_bit &&
+ uh->readSubscriberSession.values.activeSessions & server_bit ) {
+
+ /*----------------------------------------------------*/
+ /* Read the sessionDetails from the userSession table */
+ /* */
+ /* SELECT sessionData */
+ /* FROM userSession */
+ /* WHERE subscriberNumber=x, serverId=x */
+ /*----------------------------------------------------*/
+ strcpy(uh->readSessionDetails.values.number,number);
+ uh->readSessionDetails.values.serverId = server_id;
+
+ rc = SQLExecute(uh->readSessionDetails.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to execute read session details\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readSessionDetails.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to fetch read session details\n");
+ return;
+ }
+
+ strcpy(session_details, uh->readSessionDetails.values.details);
+
+ /*----------------------------------------*/
+ /* Increment noOfRead field in the server */
+ /* */
+ /* UPDATE server */
+ /* SET noOfRead=noOfRead+1 */
+ /* WHERE serverId=x,subscriberSuffix=x */
+ /*----------------------------------------*/
+ uh->updateServerNoOfRead.values.serverId = server_id;
+ strcpy(uh->updateServerNoOfRead.values.suffix,
+ &number[SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH]);
+
+ rc = SQLExecute(uh->updateServerNoOfRead.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T3 Unable to execute read no of read\n");
+ return;
+ }
+
+ *branch_executed = 1;
+ }
+
+ userDbCommit(uh);
+}
+
+void userTransaction_T4(UserHandle *uh,
+ SubscriberNumber number,
+ ServerId server_id,
+ ServerBit server_bit,
+ SessionDetails session_details,
+ unsigned int do_rollback,
+ unsigned int *branch_executed)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ *branch_executed = 0;
+
+ /*--------------------------------------*/
+ /* Read active sessions from subscriber */
+ /*--------------------------------------*/
+ if( readSubscriberSessions(uh, number, "T4") < 0 )
+ return;
+
+ /*-------------------------------------------------*/
+ /* Read the 'insert' Permissions for the userGroup */
+ /* */
+ /* SELECT allowInsert */
+ /* FROM USERGROUP */
+ /* WHERE groupId=x */
+ /*-------------------------------------------------*/
+ uh->readGroupAllowInsert.values.groupId = uh->readSubscriberSession.values.groupId;
+
+ rc = SQLExecute(uh->readGroupAllowInsert.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T4 Unable to execute read group allow insert\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readGroupAllowInsert.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T4 Unable to fetch read group allow insert\n");
+ return;
+ }
+
+ if( uh->readGroupAllowInsert.values.allowInsert & server_bit &&
+ !(uh->readSubscriberSession.values.activeSessions & server_bit) ) {
+
+ /*---------------------------------------------*/
+ /* Insert the session to the userSession table */
+ /* */
+ /* INSERT INTO userSession */
+ /* VALUES (x,x,x) */
+ /*---------------------------------------------*/
+ strcpy(uh->insertSession.values.number, number);
+ uh->insertSession.values.serverId = server_id;
+ strcpy(uh->insertSession.values.details, session_details);
+
+ rc = SQLExecute(uh->insertSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+handle_error(uh->hdbc, uh->henv, uh->insertSession.stmt, rc, __FILE__, __LINE__);
+ printf("T4 Unable to execute insert session \n");
+ return;
+ }
+
+ /*----------------------------------------*/
+ /* Update subscriber activeSessions field */
+ /* */
+ /* UPDATE subscriber */
+ /* SET activeSessions=x */
+ /* WHERE subscriberNumber=x */
+ /*----------------------------------------*/
+ strcpy(uh->updateSubscriberSession.values.number, number);
+ uh->updateSubscriberSession.values.activeSessions =
+ uh->readSubscriberSession.values.activeSessions | server_bit;
+
+ rc = SQLExecute(uh->updateSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T4 Unable to execute update session \n");
+ return;
+ }
+
+ /*------------------------------------------*/
+ /* Increment noOfInsert field in the server */
+ /* */
+ /* UPDATE server */
+ /* SET noOfInsert=noOfInsert+1 */
+ /* WHERE serverId=x,subscriberSuffix=x */
+ /*------------------------------------------*/
+ uh->updateServerNoOfInsert.values.serverId = server_id;
+ strcpy(uh->updateServerNoOfInsert.values.suffix,
+ &number[SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH]);
+
+ rc = SQLExecute(uh->updateServerNoOfInsert.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T4 Unable to execute update no of read\n");
+ return;
+ }
+
+ *branch_executed = 1;
+ }
+
+ if(do_rollback)
+ userDbRollback(uh);
+ else
+ userDbCommit(uh);
+}
+
+void userTransaction_T5(UserHandle *uh,
+ SubscriberNumber number,
+ ServerId server_id,
+ ServerBit server_bit,
+ unsigned int do_rollback,
+ unsigned int *branch_executed)
+{
+ SQLRETURN rc;
+
+ if(!uh) return;
+
+ *branch_executed = 0;
+
+ /*--------------------------------------*/
+ /* Read active sessions from subscriber */
+ /*--------------------------------------*/
+ if( readSubscriberSessions(uh, number, "T5") < 0 )
+ return;
+
+ /*-------------------------------------------------*/
+ /* Read the 'delete' Permissions for the userGroup */
+ /* */
+ /* SELECT allowDelete */
+ /* FROM USERGROUP */
+ /* WHERE groupId=x */
+ /*-------------------------------------------------*/
+ uh->readGroupAllowDelete.values.groupId = uh->readSubscriberSession.values.groupId;
+
+ rc = SQLExecute(uh->readGroupAllowDelete.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to execute read group allow delete\n");
+ return;
+ }
+
+ rc = SQLFetch(uh->readGroupAllowDelete.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to fetch read group allow delete\n");
+ return;
+ }
+
+ if( uh->readGroupAllowDelete.values.allowDelete & server_bit &&
+ uh->readSubscriberSession.values.activeSessions & server_bit ) {
+
+ /*-----------------------------------------------*/
+ /* Delete the session from the userSession table */
+ /* */
+ /* DELETE FROM userSession */
+ /* WHERE subscriberNumber=x,serverId=x */
+ /*-----------------------------------------------*/
+ strcpy(uh->deleteSession.values.number,number);
+ uh->deleteSession.values.serverId = server_id;
+
+ rc = SQLExecute(uh->deleteSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to execute delete session\n");
+ return;
+ }
+
+ /*----------------------------------------*/
+ /* Update subscriber activeSessions field */
+ /* */
+ /* UPDATE subscriber */
+ /* SET activeSessions=x */
+ /* WHERE subscriberNumber=x */
+ /*----------------------------------------*/
+ strcpy(uh->updateSubscriberSession.values.number, number);
+ uh->updateSubscriberSession.values.activeSessions =
+ uh->readSubscriberSession.values.activeSessions & ~server_bit;
+
+ rc = SQLExecute(uh->updateSubscriberSession.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to execute update subscriber session \n");
+ return;
+ }
+
+ /*------------------------------------------*/
+ /* Increment noOfDelete field in the server */
+ /* */
+ /* UPDATE server */
+ /* SET noOfDelete=noOfDelete+1 */
+ /* WHERE serverId=x,subscriberSuffix=x */
+ /*------------------------------------------*/
+ uh->updateServerNoOfDelete.values.serverId = server_id;
+ strcpy(uh->updateServerNoOfDelete.values.suffix,
+ &number[SUBSCRIBER_NUMBER_LENGTH-SUBSCRIBER_NUMBER_SUFFIX_LENGTH]);
+
+ rc = SQLExecute(uh->updateServerNoOfDelete.stmt);
+ if( rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ printf("T5 Unable to execute update no of delete\n");
+ return;
+ }
+
+ *branch_executed = 1;
+ }
+
+ if(do_rollback)
+ userDbRollback(uh);
+ else
+ userDbCommit(uh);
+}
+
+
diff --git a/ndb/test/ndbapi/restarter/Makefile b/ndb/test/ndbapi/restarter/Makefile
new file mode 100644
index 00000000000..041fbfd82ba
--- /dev/null
+++ b/ndb/test/ndbapi/restarter/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := restarter
+
+# Source files of non-templated classes (.C files)
+SOURCES = restarter.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/restarter/restarter.cpp b/ndb/test/ndbapi/restarter/restarter.cpp
new file mode 100644
index 00000000000..ad3507df98a
--- /dev/null
+++ b/ndb/test/ndbapi/restarter/restarter.cpp
@@ -0,0 +1,131 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "mgmapi.h"
+#include <string.h>
+#include <NdbMain.h>
+#include <OutputStream.hpp>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <NdbRestarter.hpp>
+#include <NdbRestarts.hpp>
+#include <NDBT.hpp>
+#include <assert.h>
+#include <NdbStdio.h>
+
+int main(int argc, const char** argv){
+
+ const char* _hostName = NULL;
+ int _loops = 10;
+ int _wait = 15;
+ int _help = 0;
+ int _error_insert = 0;
+ int _initial = 0;
+ int _master = 0;
+ int _maxwait = 120;
+ int _multiple = 0;
+
+ struct getargs args[] = {
+ { "seconds", 's', arg_integer, &_wait,
+ "Seconds to wait between each restart(0=random)", "secs" },
+ { "max seconds", 'm', arg_integer, &_maxwait,
+ "Max seconds to wait between each restart. Default is 120 seconds",
+ "msecs" },
+ { "loops", 'l', arg_integer, &_loops,
+ "Number of loops(0=forever)", "loops"},
+ { "initial", 'i', arg_flag, &_initial, "Initial node restart"},
+ { "error-insert", 'e', arg_flag, &_error_insert, "Use error insert"},
+ { "master", 'm', arg_flag, &_master,
+ "Restart the master"},
+ { "multiple", 'x', arg_flag, &_multiple,
+ "Multiple random node restarts. OBS! Even and odd node Ids must be separated into each node group"},
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "hostname:port\n"\
+ "This program will connect to the mgmsrv of a NDB cluster.\n"\
+ "It will then wait for all nodes to be started, then restart node(s)\n"\
+ "and wait for all to restart inbetween. It will do this \n"\
+ "loop number of times\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _hostName = argv[optind];
+
+
+ NdbRestarts restarts(_hostName);
+ NdbRestarter restarter(_hostName);
+
+ const char* restartName = "";
+ if (_multiple){
+ if (_master){
+ restartName = "TwoMasterNodeFailure";
+ }
+ else {
+ // Restart 50 percent of nodes
+ restartName = "FiftyPercentFail";
+ }
+ }
+ else if (_master){
+ restartName = "RestartMasterNodeError";
+ }else {
+ if (_error_insert)
+ restartName = "RestartRandomNodeError";
+ else if (_initial)
+ restartName = "RestartRandomNodeInitial";
+ else
+ restartName = "RestartRandomNode";
+ }
+
+ ndbout << "Performing " << restartName << endl;
+
+ int result = NDBT_OK;
+ int l = 0;
+ while (_loops == 0 || l<_loops){
+
+ g_info << "Waiting for cluster to start" << endl;
+ while (restarter.waitClusterStarted(1) != 0){
+ //g_warning << "Ndb failed to start in 2 minutes" << endl;
+ }
+
+ int seconds = _wait;
+ if(seconds==0) {
+ // Create random value, default 120 secs
+ seconds = (rand() % _maxwait) + 1;
+ }
+ g_info << "Waiting for " << seconds << "(" << _maxwait
+ << ") secs " << endl;
+ NdbSleep_SecSleep(seconds);
+
+ g_info << l << ": Restarting node(s) " << endl;
+
+ if (restarts.executeRestart(restartName) != 0){
+ result = NDBT_FAILED;
+ break;
+ }
+
+ l++;
+ }
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/test/ndbapi/restarter2/Makefile b/ndb/test/ndbapi/restarter2/Makefile
new file mode 100644
index 00000000000..ba33a2e21dc
--- /dev/null
+++ b/ndb/test/ndbapi/restarter2/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := restarter2
+
+# Source files of non-templated classes (.C files)
+SOURCES = restarter2.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/restarter2/restarter2.cpp b/ndb/test/ndbapi/restarter2/restarter2.cpp
new file mode 100644
index 00000000000..71eaf1a9b0f
--- /dev/null
+++ b/ndb/test/ndbapi/restarter2/restarter2.cpp
@@ -0,0 +1,118 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <string.h>
+#include <NdbMain.h>
+#include <OutputStream.hpp>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <NdbRestarter.hpp>
+#include <NDBT.hpp>
+#include <assert.h>
+#include <NdbStdio.h>
+
+int main(int argc, const char** argv){
+
+ const char* _hostName = NULL;
+ int _loops = 10;
+ int _wait = 15;
+ int _help = 0;
+#if 0
+ int _crash = 0;
+ int _abort = 0;
+#endif
+
+ struct getargs args[] = {
+ { "seconds", 's', arg_integer, &_wait, "Seconds to wait between each restart(0=random)", "secs" },
+ { "loops", 'l', arg_integer, &_loops, "Number of loops", "loops 0=forever"},
+#if 0
+ // Not yet!
+ { "abort", 'a', arg_flag, &_abort, "Restart abort"},
+ { "crash", 'c', arg_flag, &_crash, "Crash instead of restart"},
+#endif
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "hostname:port\n"\
+ "This program will connect to the mgmsrv of a NDB cluster.\n"\
+ "It will wait for all nodes to be started, then restart all nodes\n"\
+ "into nostart state. Then after a random delay it will tell all nodes\n"\
+ "to start. It will do this loop number of times\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _hostName = argv[optind];
+
+ NdbRestarter restarter(_hostName);
+#if 0
+ if(_abort && _crash){
+ g_err << "You can't specify both abort and crash" << endl;
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ if(_abort){
+ restarter.setRestartType(NdbRestarter::AbortRestart);
+ }
+ if(_crash){
+ restarter.setRestartType(NdbRestarter::Crash);
+ }
+#endif
+
+ int l = 0;
+ while (_loops == 0 || l<_loops){
+ g_info << "Waiting for cluster to start" << endl;
+ while(restarter.waitClusterStarted(120) != 0){
+ g_warning << "Ndb failed to start in 2 minutes" << endl;
+ }
+
+ int seconds = _wait;
+ if(seconds==0)
+ seconds = (rand() % 120) + 1; // Create random value max 120 secs
+ g_info << "Waiting for "<<seconds<<" secs" << endl;
+ NdbSleep_SecSleep(seconds);
+
+ g_info << l << ": restarting all nodes with nostart" << endl;
+ const bool b = (restarter.restartAll(false, true, false) == 0);
+ assert(b);
+
+ g_info << "Waiting for cluster to enter nostart" << endl;
+ while(restarter.waitClusterNoStart(120) != 0){
+ g_warning << "Ndb failed to enter no start in 2 minutes" << endl;
+ }
+
+ seconds = _wait;
+ if(seconds==0)
+ seconds = (rand() % 120) + 1; // Create random value max 120 secs
+ g_info << "Waiting for " <<seconds<<" secs" << endl;
+ NdbSleep_SecSleep(seconds);
+
+ g_info << l << ": Telling all nodes to start" << endl;
+ const bool b2 = (restarter.startAll() == 0);
+ assert(b2);
+
+ l++;
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/test/ndbapi/restarts/Makefile b/ndb/test/ndbapi/restarts/Makefile
new file mode 100644
index 00000000000..9f14b81fae5
--- /dev/null
+++ b/ndb/test/ndbapi/restarts/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := restarts
+
+# Source files of non-templated classes (.C files)
+SOURCES = restarts.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/restarts/restarts.cpp b/ndb/test/ndbapi/restarts/restarts.cpp
new file mode 100644
index 00000000000..2f9bab3b233
--- /dev/null
+++ b/ndb/test/ndbapi/restarts/restarts.cpp
@@ -0,0 +1,117 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "mgmapi.h"
+#include <string.h>
+#include <NdbMain.h>
+#include <OutputStream.hpp>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <NdbRestarts.hpp>
+#include <NDBT.hpp>
+#include <assert.h>
+#include <NdbStdio.h>
+
+int main(int argc, const char** argv){
+
+ const char* _restartName = NULL;
+ int _loops = 1;
+ int _wait = -1;
+ int _maxwait = 120;
+ int _help = 0;
+ int _list = 0;
+ int _random = 0;
+ int _timeout = 0;
+ int _all = 0;
+
+ struct getargs args[] = {
+ { "seconds", 's', arg_integer, &_wait,
+ "Seconds to wait between each restart(0=random)", "secs" },
+ { "max seconds", 'm', arg_integer, &_maxwait,
+ "Max seconds to wait between each restart. Default is 120 seconds", "msecs" },
+ { "loops", 'l', arg_integer, &_loops, "Number of loops(0=forever)", "loops"},
+ { "timeout", 't', arg_integer, &_timeout, "Timeout waiting for nodes to start", "seconds"},
+ { "random", 'r', arg_flag, &_random, "Select restart case randomly",
+ ""},
+ { "all", 'a', arg_flag, &_all, "Run all restarts",
+ ""},
+ { "list-restarts", '\0', arg_flag, &_list, "List available restarts", ""},
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "testname\n" \
+ "This program will perform node restart, \n"\
+ "multiple node restart or system-restart.\n"\
+ "Use --list-restarts to see whats available\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ if (_list){
+ NdbRestarts restarts;
+ restarts.listRestarts();
+ return NDBT_ProgramExit(NDBT_OK);
+ }
+
+ if ((argv[optind] == NULL) && (_random == 0)){
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _restartName = argv[optind];
+
+ NdbRestarts restarts;
+
+ int res = NDBT_OK;
+ int l = 0;
+ while (_loops == 0 || l < _loops){
+
+ if (_all) {
+ // Execute all restarts, set loops to numRestarts
+ // so that ecvery restart is executed once
+ _loops = restarts.getNumRestarts();
+ res = restarts.executeRestart(l, _timeout);
+ } else if (_random) {
+ int num = rand() % restarts.getNumRestarts();
+ res = restarts.executeRestart(num, _timeout);
+ } else {
+ res = restarts.executeRestart(_restartName, _timeout);
+ }
+ if (res != NDBT_OK)
+ break;
+
+ if (_wait >= 0){
+ int seconds = _wait;
+ if(seconds==0) {
+ // Create random value, default 120 secs
+ seconds = (rand() % _maxwait) + 1;
+ }
+ g_info << "Waiting for " << seconds << "(" << _maxwait
+ << ") secs " << endl;
+ NdbSleep_SecSleep(seconds);
+ }
+
+ l++;
+ }
+ return NDBT_ProgramExit(res);
+}
diff --git a/ndb/test/ndbapi/ronja/benchronja/Makefile b/ndb/test/ndbapi/ronja/benchronja/Makefile
new file mode 100644
index 00000000000..f0521c3ba77
--- /dev/null
+++ b/ndb/test/ndbapi/ronja/benchronja/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+
+BIN_TARGET := benchronja
+
+SOURCES := benchronja.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/ronja/benchronja/benchronja.cpp b/ndb/test/ndbapi/ronja/benchronja/benchronja.cpp
new file mode 100644
index 00000000000..71fa286a21b
--- /dev/null
+++ b/ndb/test/ndbapi/ronja/benchronja/benchronja.cpp
@@ -0,0 +1,1205 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/* ***************************************************
+ NODEREC
+ Perform benchmark of insert, update and delete transactions
+
+ Arguments:
+ -t Number of threads to start, default 1
+ -o Number of loops per thread, default 100000
+
+
+ * *************************************************** */
+
+#include <NdbApi.hpp>
+#include <NdbTest.hpp>
+#include <NdbOut.hpp>
+#include <NdbThread.h>
+#include <NdbSleep.h>
+#include <NdbMain.h>
+#include <NdbTimer.hpp>
+#include <NdbTick.h>
+#include <random.h>
+
+#include <stdlib.h>
+#include <string.h>
+#include <stdio.h>
+#include <math.h>
+
+#define MAX_TIMERS 4
+#define MAXSTRLEN 16
+#define MAXATTR 64
+#define MAXTABLES 64
+#define MAXTHREADS 256
+#define MAXATTRSIZE 8000
+#define START_TIMER NdbTimer timer; timer.doStart();
+#define STOP_TIMER timer.doStop();
+#define START_TIMER_TOP NdbTimer timer_top; timer_top.doStart();
+#define STOP_TIMER_TOP timer_top.doStop();
+
+void* ThreadExec(void*);
+struct ThreadNdb
+{
+ int NoOfOps;
+ int ThreadNo;
+ Ndb* NdbRef;
+};
+
+static NdbThread* threadLife[MAXTHREADS];
+static unsigned int tNoOfThreads;
+static unsigned int tNoOfOpsPerExecute;
+static unsigned int tNoOfRecords;
+static unsigned int tNoOfOperations;
+static int ThreadReady[MAXTHREADS];
+static int ThreadStart[MAXTHREADS];
+
+NDB_COMMAND(benchronja, "benchronja", "benchronja", "benchronja", 65535){
+
+ ThreadNdb tabThread[MAXTHREADS];
+ int i = 0 ;
+ int cont = 0 ;
+ Ndb* pMyNdb = NULL ; //( "TEST_DB" );
+ int tmp = 0 ;
+ int nTest = 0 ;
+ char inp[100] ;
+
+ tNoOfThreads = 1; // Default Value
+ tNoOfOpsPerExecute = 1; // Default Value
+ tNoOfOperations = 100000; // Default Value
+ tNoOfRecords = 500 ; // Default Value <epaulsa: changed from original 500,000 to match 'initronja's' default
+ i = 1;
+ while (argc > 1)
+ {
+ if (strcmp(argv[i], "-t") == 0){
+ tNoOfThreads = atoi(argv[i+1]);
+ if ((tNoOfThreads < 1) || (tNoOfThreads > MAXTHREADS)) goto error_input;
+ }else if (strcmp(argv[i], "-o") == 0){
+ tNoOfOperations = atoi(argv[i+1]);
+ if (tNoOfOperations < 1) goto error_input;
+ }else if (strcmp(argv[i], "-r") == 0){
+ tNoOfRecords = atoi(argv[i+1]);
+ if ((tNoOfRecords < 1) || (tNoOfRecords > 1000000000)) goto error_input;
+ }else if (strcmp(argv[i], "-p") == 0){
+ nTest = atoi(argv[i+1]) ;
+ if (0 > nTest || 18 < nTest) goto error_input ;
+ }else if (strcmp(argv[i], "-c") == 0){
+ tNoOfOpsPerExecute = atoi(argv[i+1]);
+ if ((tNoOfOpsPerExecute < 1) || (tNoOfOpsPerExecute > 1024)) goto error_input;
+ }else{
+ goto error_input;
+ }
+ argc -= 2;
+ i = i + 2;
+ }
+
+ ndbout << "Initialisation started. " << endl;
+ pMyNdb = new Ndb("TEST_DB") ;
+ pMyNdb->init();
+ ndbout << "Initialisation completed. " << endl;
+
+ ndbout << endl << "Execute Ronja Benchmark" << endl;
+ ndbout << " NdbAPI node with id = " << pMyNdb->getNodeId() << endl;
+ ndbout << " " << tNoOfThreads << " thread(s) " << endl;
+ ndbout << " " << tNoOfOperations << " transaction(s) per thread and round " << endl;
+
+ if (pMyNdb->waitUntilReady(120) != 0) {
+ ndbout << "Benchmark failed - NDB is not ready" << endl;
+ delete pMyNdb ;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }//if
+
+ NdbThread_SetConcurrencyLevel(2 + tNoOfThreads);
+
+ for (i = 0; i < tNoOfThreads ; i++) {
+ ThreadReady[i] = 0;
+ ThreadStart[i] = 0;
+ }//for
+
+ for (i = 0; i < tNoOfThreads ; i++) {
+ tabThread[i].ThreadNo = i;
+ tabThread[i].NdbRef = NULL;
+ tabThread[i].NoOfOps = tNoOfOperations;
+ threadLife[i] = NdbThread_Create(ThreadExec,
+ (void**)&tabThread[i],
+ 32768,
+ "RonjaThread",
+ NDB_THREAD_PRIO_LOW);
+ }//for
+
+ cont = 1;
+ while (cont) {
+ NdbSleep_MilliSleep(10);
+ cont = 0;
+ for (i = 0; i < tNoOfThreads ; i++)
+ if (!ThreadReady[i]) cont = 1;
+ }//while
+
+ ndbout << "All threads started" << endl;
+
+ if(!nTest){
+
+ for (;;){
+
+ inp[0] = 0;
+ ndbout << endl << "What to do next:" << endl;
+ ndbout << "1 \t=> Perform lookups in short table" << endl;
+ ndbout << "2 \t=> Perform lookups in long table" << endl;
+ ndbout << "3 \t=> Perform updates in short table" << endl;
+ ndbout << "4 \t=> Perform updates in long table" << endl;
+ ndbout << "5 \t=> Perform 50% lookups/50% updates in short table" << endl;
+ ndbout << "6 \t=> Perform 50% lookups/50% updates in long table" << endl;
+ ndbout << "7 \t=> Perform 80% lookups/20% updates in short table" << endl;
+ ndbout << "8 \t=> Perform 80% lookups/20% updates in long table" << endl;
+ ndbout << "9 \t=> Perform 25% lookups short/25% lookups long/25% updates short/25% updates long" << endl;
+ ndbout << "10\t=> Test bug with replicated interpreted updates, short table" << endl;
+ ndbout << "11\t=> Test interpreter functions, short table" << endl;
+ ndbout << "12\t=> Test bug with replicated interpreted updates, long table" << endl;
+ ndbout << "13\t=> Test interpreter functions, long table" << endl;
+ ndbout << "14\t=> Perform lookups in short table, no guess of TC" << endl;
+ ndbout << "15\t=> Perform lookups in long table, no guess of TC" << endl;
+ ndbout << "16\t=> Perform updates in short table, no guess of TC" << endl;
+ ndbout << "17\t=> Perform updates in long table, no guess of TC" << endl;
+ ndbout << "18\t=> Multi record updates of transactions" << endl;
+ ndbout << "All other responses will exit" << endl;
+ ndbout << "_____________________________" << endl << endl ;
+
+ int inp_i = 0;
+ do {
+ inp[inp_i] = (char) fgetc(stdin);
+ if (inp[inp_i] == '\n' || inp[inp_i] == EOF) {
+ inp[inp_i] ='\0';
+ break;
+ }
+ inp_i++;
+
+ } while (inp[inp_i - 1] != '\n' && inp[inp_i - 1] != EOF);
+
+ tmp = atoi(inp);
+
+ if ((tmp > 18) || (tmp <= 0)) break;
+
+ ndbout << "Starting test " << tmp << "..." << endl;
+
+ for (i = 0; i < tNoOfThreads ; i++){ ThreadStart[i] = tmp; }
+
+ cont = 1;
+ while (cont) {
+ NdbSleep_MilliSleep(10);
+ cont = 0;
+ for (i = 0; i < tNoOfThreads ; i++){
+ if (!ThreadReady[i]) cont = 1;
+ }
+ }//while
+ }//for(;;)
+
+ }else{
+
+ if(19 == nTest){
+ ndbout << "Executing all 18 available tests..." << endl << endl;
+ for (int count = 1; count < nTest; count++){
+ ndbout << "Test " << count << endl ;
+ ndbout << "------" << endl << endl ;
+ for (i = 0; i < tNoOfThreads ; i++) { ThreadStart[i] = count ; }
+ cont = 1;
+ while (cont) {
+ NdbSleep_MilliSleep(10);
+ cont = 0;
+ for (i = 0; i < tNoOfThreads ; i++){
+ if (!ThreadReady[i]) cont = 1;
+ }
+ }
+ }//for
+ }else{
+ ndbout << endl << "Executing test " << nTest << endl << endl;
+ for (i = 0; i < tNoOfThreads ; i++) { ThreadStart[i] = nTest ; }
+ cont = 1;
+ while (cont) {
+ NdbSleep_MilliSleep(10);
+ cont = 0;
+ for (i = 0; i < tNoOfThreads ; i++){
+ if (!ThreadReady[i]) cont = 1;
+ }
+ }
+ }//if(18 == nTest)
+ } //if(!nTest)
+
+ ndbout << "--------------------------------------------------" << endl;
+
+ for (i = 0; i < tNoOfThreads ; i++) ThreadReady[i] = 0;
+ // Signaling threads to stop
+ for (i = 0; i < tNoOfThreads ; i++) ThreadStart[i] = 999;
+
+ // Wait for threads to stop
+ cont = 1;
+ do {
+ NdbSleep_MilliSleep(1);
+ cont = 0;
+ for (i = 0; i < tNoOfThreads ; i++){
+ if (ThreadReady[i] == 0) cont = 1;
+ }
+ } while (cont == 1);
+
+ delete pMyNdb ;
+ ndbout << endl << "Ronja Benchmark completed" << endl;
+ return NDBT_ProgramExit(NDBT_OK) ;
+
+error_input:
+ ndbout << endl << " Ivalid parameter(s)" << endl;
+ ndbout << " Usage: benchronja [-t threads][-r rec] [-o ops] [-c ops_per_exec] [-p test], where:" << endl;
+ ndbout << " threads - the number of threads to start; default: 1" << endl;
+ ndbout << " rec - the number of records in the tables; default: 500" << endl;
+ ndbout << " ops - the number of operations per transaction; default: 100000" << endl;
+ ndbout << " ops_per_exec - the number of operations per execution; default: 1" << endl ;
+ ndbout << " test - the number of test to execute; 19 executes all available tests; default: 0"<< endl ;
+ ndbout << " which enters a loop expecting manual input of test number to execute." << endl << endl ;
+ delete pMyNdb ;
+ return NDBT_ProgramExit(NDBT_WRONGARGS) ;
+
+ }
+////////////////////////////////////////
+
+void commitTrans(Ndb* aNdb, NdbConnection* aCon)
+{
+ int ret = aCon->execute(Commit);
+ assert (ret != -1);
+ aNdb->closeTransaction(aCon);
+}
+
+void rollbackTrans(Ndb* aNdb, NdbConnection* aCon)
+{
+ int ret = aCon->execute(Rollback);
+ assert (ret != -1);
+ aNdb->closeTransaction(aCon);
+}
+
+void updateNoCommit(NdbConnection* aCon, Uint32* flip, unsigned int key)
+{
+ NdbOperation* theOperation;
+
+ *flip = *flip + 1;
+ theOperation = aCon->getNdbOperation("SHORT_REC");
+ theOperation->updateTuple();
+ theOperation->equal((Uint32)0, key);
+ theOperation->setValue((Uint32)1, (char*)flip);
+ int ret = aCon->execute(NoCommit);
+ assert (ret != -1);
+}
+
+void updateNoCommitFail(NdbConnection* aCon, unsigned int key)
+{
+ NdbOperation* theOperation;
+
+ Uint32 flip = 0;
+ theOperation = aCon->getNdbOperation("SHORT_REC");
+ theOperation->updateTuple();
+ theOperation->equal((Uint32)0, key);
+ theOperation->setValue((Uint32)1, (char*)flip);
+ int ret = aCon->execute(NoCommit);
+ assert (ret == -1);
+}
+
+void deleteNoCommit(NdbConnection* aCon, Uint32* flip, unsigned int key)
+{
+ NdbOperation* theOperation;
+
+ *flip = 0;
+ theOperation = aCon->getNdbOperation("SHORT_REC");
+ theOperation->deleteTuple();
+ theOperation->equal((Uint32)0, key);
+ int ret = aCon->execute(NoCommit);
+ assert (ret != -1);
+}
+
+void insertNoCommit(NdbConnection* aCon, Uint32* flip, unsigned int key)
+{
+ NdbOperation* theOperation;
+ Uint32 placeholder[100];
+
+ *flip = *flip + 1;
+ theOperation = aCon->getNdbOperation("SHORT_REC");
+ theOperation->insertTuple();
+ theOperation->equal((Uint32)0, key);
+ theOperation->setValue((Uint32)1, (char*)flip);
+ theOperation->setValue((Uint32)2, (char*)&placeholder[0]);
+ theOperation->setValue((Uint32)3, (char*)&placeholder[0]);
+ int ret = aCon->execute(NoCommit);
+ assert (ret != -1);
+}
+
+void writeNoCommit(NdbConnection* aCon, Uint32* flip, unsigned int key)
+{
+ NdbOperation* theOperation;
+ Uint32 placeholder[100];
+
+ *flip = *flip + 1;
+ theOperation = aCon->getNdbOperation("SHORT_REC");
+ theOperation->writeTuple();
+ theOperation->equal((Uint32)0, key);
+ theOperation->setValue((Uint32)1, (char*)flip);
+ theOperation->setValue((Uint32)2, (char*)&placeholder[0]);
+ theOperation->setValue((Uint32)3, (char*)&placeholder[0]);
+ int ret = aCon->execute(NoCommit);
+ assert (ret != -1);
+}
+
+void readNoCommit(NdbConnection* aCon, Uint32* flip, Uint32 key, int expected_ret)
+{
+ NdbOperation* theOperation;
+ Uint32 readFlip;
+
+ theOperation = aCon->getNdbOperation("SHORT_REC");
+ theOperation->readTuple();
+ theOperation->equal((Uint32)0, key);
+ theOperation->getValue((Uint32)1, (char*)&readFlip);
+ int ret = aCon->execute(NoCommit);
+ assert (ret == expected_ret);
+ if (ret == 0)
+ assert (*flip == readFlip);
+}
+
+void readDirtyNoCommit(NdbConnection* aCon, Uint32* flip, Uint32 key, int expected_ret)
+{
+ NdbOperation* theOperation;
+ Uint32 readFlip;
+
+ theOperation = aCon->getNdbOperation("SHORT_REC");
+ theOperation->committedRead();
+ theOperation->equal((Uint32)0, key);
+ theOperation->getValue((Uint32)1, (char*)&readFlip);
+ int ret = aCon->execute(NoCommit);
+ assert (ret == expected_ret);
+ if (ret == 0)
+ assert (*flip == readFlip);
+}
+
+void readVerify(Ndb* aNdb, Uint32* flip, Uint32 key, int expected_ret)
+{
+ NdbConnection* theTransaction;
+ theTransaction = aNdb->startTransaction();
+ readNoCommit(theTransaction, flip, key, expected_ret);
+ commitTrans(aNdb, theTransaction);
+}
+
+void readDirty(Ndb* aNdb, Uint32* flip, Uint32 key, int expected_ret)
+{
+ NdbOperation* theOperation;
+ NdbConnection* theTransaction;
+ Uint32 readFlip;
+
+ theTransaction = aNdb->startTransaction();
+ theOperation = theTransaction->getNdbOperation("SHORT_REC");
+ theOperation->committedRead();
+ theOperation->equal((Uint32)0, key);
+ theOperation->getValue((Uint32)1, (char*)&readFlip);
+ int ret = theTransaction->execute(Commit);
+ assert (ret == expected_ret);
+ if (ret == 0)
+ assert (*flip == readFlip);
+ aNdb->closeTransaction(theTransaction);
+}
+
+int multiRecordTest(Ndb* aNdb, unsigned int key)
+{
+ NdbConnection* theTransaction;
+ Uint32 flip = 0;
+ Uint32 save_flip;
+ ndbout << "0" << endl;
+
+ theTransaction = aNdb->startTransaction();
+
+ updateNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, 0);
+
+ updateNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, 0);
+
+ commitTrans(aNdb, theTransaction);
+
+ ndbout << "1 " << endl;
+
+ readVerify(aNdb, &flip, key, 0);
+ readDirty(aNdb, &flip, key, 0);
+ save_flip = flip;
+ ndbout << "1.1 " << endl;
+
+ theTransaction = aNdb->startTransaction();
+
+ deleteNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, -1);
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+ readDirtyNoCommit(theTransaction, &flip, key, -1);
+ ndbout << "1.2 " << endl;
+
+ insertNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, 0);
+ readDirtyNoCommit(theTransaction, &flip, key, 0);
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+ ndbout << "1.3 " << endl;
+
+ updateNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, 0);
+ readDirtyNoCommit(theTransaction, &flip, key, 0);
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+ ndbout << "1.4 " << endl;
+
+ commitTrans(aNdb, theTransaction);
+
+ ndbout << "2 " << endl;
+
+ readDirty(aNdb, &flip, key, 0); // COMMITTED READ!!!
+ readVerify(aNdb, &flip, key, 0);
+
+ save_flip = flip;
+ theTransaction = aNdb->startTransaction();
+
+ deleteNoCommit(theTransaction, &flip, key);
+
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+ readDirtyNoCommit(theTransaction, &flip, key, -1); // COMMITTED READ!!!
+ readNoCommit(theTransaction, &flip, key, -1);
+
+ insertNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, 0);
+
+ updateNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, 0);
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+ readDirtyNoCommit(theTransaction, &flip, key, 0); // COMMITTED READ!!!
+
+ deleteNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, -1);
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+ readDirtyNoCommit(theTransaction, &flip, key, -1);
+
+ rollbackTrans(aNdb, theTransaction);
+
+ ndbout << "3 " << endl;
+
+ flip = save_flip;
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+ readVerify(aNdb, &flip, key, 0);
+
+ theTransaction = aNdb->startTransaction();
+
+ updateNoCommit(theTransaction, &flip, key);
+
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+ readDirtyNoCommit(theTransaction, &flip, key, 0);
+ readNoCommit(theTransaction, &flip, key, 0);
+
+ deleteNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, -1);
+ readDirtyNoCommit(theTransaction, &flip, key, -1);
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+
+ insertNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, 0);
+ readDirtyNoCommit(theTransaction, &flip, key, 0);
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+
+ updateNoCommit(theTransaction, &flip, key);
+
+ readNoCommit(theTransaction, &flip, key, 0);
+ readDirtyNoCommit(theTransaction, &flip, key, 0);
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+
+ deleteNoCommit(theTransaction, &flip, key);
+
+ readDirty(aNdb, &save_flip, key, 0); // COMMITTED READ!!!
+ readNoCommit(theTransaction, &flip, key, -1);
+ readDirtyNoCommit(theTransaction, &flip, key, -1);
+
+ commitTrans(aNdb, theTransaction);
+
+ ndbout << "4 " << endl;
+
+ readVerify(aNdb, &flip, key, -1);
+
+ theTransaction = aNdb->startTransaction();
+
+ insertNoCommit(theTransaction, &flip, key);
+
+ readDirty(aNdb, &save_flip, key, -1); // COMMITTED READ!!!
+ readNoCommit(theTransaction, &flip, key, 0);
+ readDirtyNoCommit(theTransaction, &flip, key, 0);
+
+ deleteNoCommit(theTransaction, &flip, key);
+
+ readDirty(aNdb, &save_flip, key, -1); // COMMITTED READ!!!
+ readNoCommit(theTransaction, &flip, key, -1);
+ readDirtyNoCommit(theTransaction, &flip, key, -1);
+
+ insertNoCommit(theTransaction, &flip, key);
+
+ readDirty(aNdb, &save_flip, key, -1); // COMMITTED READ!!!
+ readNoCommit(theTransaction, &flip, key, 0);
+ readDirtyNoCommit(theTransaction, &flip, key, 0);
+
+ updateNoCommit(theTransaction, &flip, key);
+
+ readDirty(aNdb, &save_flip, key, -1); // COMMITTED READ!!!
+ readNoCommit(theTransaction, &flip, key, 0);
+ readDirtyNoCommit(theTransaction, &flip, key, 0);
+
+ deleteNoCommit(theTransaction, &flip, key);
+
+ readDirty(aNdb, &save_flip, key, -1); // COMMITTED READ!!!
+ readNoCommit(theTransaction, &flip, key, -1);
+ readDirtyNoCommit(theTransaction, &flip, key, -1);
+
+ commitTrans(aNdb, theTransaction);
+
+ ndbout << "5 " << endl;
+
+ readDirty(aNdb, &flip, key, -1); // COMMITTED READ!!!
+ readVerify(aNdb, &flip, key, -1);
+
+ theTransaction = aNdb->startTransaction();
+
+ insertNoCommit(theTransaction, &flip, key);
+
+ readDirty(aNdb, &flip, key, -1); // COMMITTED READ!!!
+ readDirtyNoCommit(theTransaction, &flip, key, 0); // COMMITTED READ!!!
+
+ commitTrans(aNdb, theTransaction);
+ readDirty(aNdb, &flip, key, 0); // COMMITTED READ!!!
+
+ ndbout << "6 " << endl;
+
+ theTransaction = aNdb->startTransaction();
+
+ deleteNoCommit(theTransaction, &flip, key);
+ updateNoCommitFail(theTransaction, key);
+ rollbackTrans(aNdb, theTransaction);
+ return 0;
+}
+
+int lookup(Ndb* aNdb, unsigned int key, unsigned int long_short, int guess){
+
+ int placeholder[500];
+ unsigned int flip, count;
+ int ret_value, i;
+ NdbConnection* theTransaction;
+ NdbOperation* theOperation;
+ if ( !aNdb ) return -1 ;
+
+ if (guess != 0)
+ theTransaction = aNdb->startTransaction((Uint32)0, (const char*)&key, (Uint32)4);
+ else
+ theTransaction = aNdb->startTransaction();
+
+ for (i = 0; i < tNoOfOpsPerExecute; i++) {
+ if (long_short == 0)
+ theOperation = theTransaction->getNdbOperation("SHORT_REC");
+ else
+ theOperation = theTransaction->getNdbOperation("LONG_REC");
+ if (theOperation == NULL) {
+ ndbout << "Table missing" << endl;
+ aNdb->closeTransaction(theTransaction) ;
+ return -1;
+ }//if
+ theOperation->simpleRead();
+ theOperation->equal((Uint32)0, key);
+ theOperation->getValue((Uint32)1, (char*)&flip);
+ theOperation->getValue((Uint32)2, (char*)&count);
+ if (theOperation->getValue((Uint32)3, (char*)&placeholder[0]) == NULL) {
+ ndbout << "Error in definition phase = " << theTransaction->getNdbError() << endl;
+ aNdb->closeTransaction(theTransaction);
+ return -1;
+ }//if
+ }//for
+ ret_value = theTransaction->execute(Commit);
+ if (ret_value == -1)
+ ndbout << "Error in lookup:" << theTransaction->getNdbError() << endl;
+ aNdb->closeTransaction(theTransaction);
+ return ret_value;
+}//lookup()
+
+int update(Ndb* aNdb, unsigned int key, unsigned int long_short, int guess)
+{
+ int placeholder[500];
+ int ret_value, i;
+ unsigned int flip, count;
+ NdbConnection* theTransaction;
+ NdbOperation* theOperation;
+
+ if ( !aNdb ) return -1 ;
+
+ if (guess != 0)
+ theTransaction = aNdb->startTransaction((Uint32)0, (const char*)&key, (Uint32)4);
+ else
+ theTransaction = aNdb->startTransaction();
+
+ for (i = 0; i < tNoOfOpsPerExecute; i++) {
+ if (long_short == 0)
+ theOperation = theTransaction->getNdbOperation("SHORT_REC"); // Use table SHORT_REC
+ else
+ theOperation = theTransaction->getNdbOperation("LONG_REC"); // Use table LONG_REC
+ if (theOperation == NULL) {
+ ndbout << "Table missing" << endl;
+ aNdb->closeTransaction(theTransaction) ;
+ delete aNdb ;
+ return -1;
+ }//if
+ theOperation->interpretedUpdateTuple(); // Send interpreted program to NDB kernel
+ theOperation->equal((Uint32)0, key); // Search key
+ theOperation->getValue((Uint32)1, (char*)&flip); // Read value of flip
+ theOperation->getValue((Uint32)2, (char*)&count); // Read value of count
+ theOperation->getValue((Uint32)3, (char*)&placeholder[0]); // Read value of placeholder
+ theOperation->load_const_u32((Uint32)1, (Uint32)0); // Load register 1 with 0
+ theOperation->read_attr((Uint32)1, (Uint32)2); // Read Flip value into register 2
+ theOperation->branch_eq((Uint32)1, (Uint32)2, (Uint32)0); // If Flip (register 2) == 0 (register 1) goto label 0
+ theOperation->branch_label((Uint32)1); // Goto label 1
+ theOperation->def_label((Uint32)0); // Define label 0
+ theOperation->load_const_u32((Uint32)1, (Uint32)1); // Load register 1 with 1
+ theOperation->def_label((Uint32)1); // Define label 0
+ theOperation->write_attr((Uint32)1, (Uint32)1); // Write 1 (register 1) into Flip
+ ret_value = theOperation->incValue((Uint32)2, (Uint32)1); // Increment Count by 1
+ if (ret_value == -1) {
+ ndbout << "Error in definition phase " << endl;
+ aNdb->closeTransaction(theTransaction);
+ return ret_value;
+ }//if
+ }//for
+ ret_value = theTransaction->execute(Commit); // Perform the actual read and update
+ if (ret_value == -1) {
+ ndbout << "Error in update:" << theTransaction->getNdbError() << endl;
+ aNdb->closeTransaction(theTransaction); // < epaulsa
+ return ret_value ;
+ }//if
+ aNdb->closeTransaction(theTransaction);
+ return ret_value;
+}//update()
+
+int update_bug(Ndb* aNdb, unsigned int key, unsigned int long_short)
+{
+ int placeholder[500];
+ int ret_value, i;
+ unsigned int flip, count;
+ NdbConnection* theTransaction;
+ NdbOperation* theOperation;
+
+ if ( !aNdb ) return -1 ;
+
+ theTransaction = aNdb->startTransaction();
+ for (i = 0; i < tNoOfOpsPerExecute; i++) {
+ if (long_short == 0)
+ theOperation = theTransaction->getNdbOperation("SHORT_REC"); // Use table SHORT_REC
+ else
+ theOperation = theTransaction->getNdbOperation("LONG_REC"); // Use table LONG_REC
+ if (theOperation == NULL) {
+ ndbout << "Table missing" << endl;
+ aNdb->closeTransaction(theTransaction) ;
+ return -1;
+ }//if
+ theOperation->interpretedUpdateTuple(); // Send interpreted program to NDB kernel
+ theOperation->equal((Uint32)0, key); // Search key
+ theOperation->getValue((Uint32)1, (char*)&flip); // Read value of flip
+ theOperation->getValue((Uint32)2, (char*)&count); // Read value of count
+ theOperation->getValue((Uint32)3, (char*)&placeholder[0]); // Read value of placeholder
+ theOperation->load_const_u32((Uint32)1, (Uint32)0); // Load register 1 with 0
+ theOperation->read_attr((Uint32)1, (Uint32)2); // Read Flip value into register 2
+ theOperation->branch_eq((Uint32)1, (Uint32)2, (Uint32)0); // If Flip (register 2) == 0 (register 1) goto label 0
+ theOperation->branch_label((Uint32)1); // Goto label 1
+ theOperation->def_label((Uint32)0); // Define label 0
+ theOperation->load_const_u32((Uint32)1, (Uint32)1); // Load register 1 with 1
+ theOperation->def_label((Uint32)1); // Define label 0
+ theOperation->write_attr((Uint32)1, (Uint32)1); // Write 1 (register 1) into Flip
+ ret_value = theOperation->incValue((Uint32)2, (Uint32)1); // Increment Count by 1
+ if (ret_value == -1) {
+ ndbout << "Error in definition phase " << endl;
+ aNdb->closeTransaction(theTransaction);
+ return ret_value;
+ }//if
+ }//for
+ ret_value = theTransaction->execute(NoCommit); // Perform the actual read and update
+ if (ret_value == -1) {
+ ndbout << "Error in update:" << theTransaction->getNdbError() << endl;
+ aNdb->closeTransaction(theTransaction);
+ return ret_value ;
+ }//if
+ aNdb->closeTransaction(theTransaction);
+ return ret_value;
+}//update_bug()
+
+int update_interpreter_test(Ndb* aNdb, unsigned int key, unsigned int long_short)
+{
+ int placeholder[500];
+ int ret_value, i;
+ unsigned int flip, count;
+ NdbConnection* theTransaction;
+ NdbOperation* theOperation;
+ Uint32 Tlabel = 0;
+
+ if ( !aNdb ) return -1 ;
+
+//------------------------------------------------------------------------------
+// Start the transaction and get a unique transaction id
+//------------------------------------------------------------------------------
+ theTransaction = aNdb->startTransaction();
+ for (i = 0; i < tNoOfOpsPerExecute; i++) {
+//------------------------------------------------------------------------------
+// Get the proper table object and load schema information if not already
+// present.
+//------------------------------------------------------------------------------
+ if (long_short == 0)
+ theOperation = theTransaction->getNdbOperation("SHORT_REC"); // Use table SHORT_REC
+ else
+ theOperation = theTransaction->getNdbOperation("LONG_REC"); // Use table LONG_REC
+ if (theOperation == NULL) {
+ ndbout << "Table missing" << endl;
+ aNdb->closeTransaction(theTransaction) ;
+ return -1;
+ }//if
+//------------------------------------------------------------------------------
+// Define the operation type and the tuple key (primary key in this case).
+//------------------------------------------------------------------------------
+ theOperation->interpretedUpdateTuple(); // Send interpreted program to NDB kernel
+ theOperation->equal((Uint32)0, key); // Search key
+
+//------------------------------------------------------------------------------
+// Perform initial read of attributes before updating them
+//------------------------------------------------------------------------------
+ theOperation->getValue((Uint32)1, (char*)&flip); // Read value of flip
+ theOperation->getValue((Uint32)2, (char*)&count); // Read value of count
+ theOperation->getValue((Uint32)3, (char*)&placeholder[0]); // Read value of placeholder
+
+//------------------------------------------------------------------------------
+// Test that the various branch operations can handle things correctly.
+// Test first 2 + 3 = 5 with 32 bit registers
+// Next test the same with 32 bit + 64 bit = 64
+//------------------------------------------------------------------------------
+ theOperation->load_const_u32((Uint32)4, (Uint32)0); // Load register 4 with 0
+
+ theOperation->load_const_u32((Uint32)0, (Uint32)0);
+ theOperation->load_const_u32((Uint32)1, (Uint32)3);
+ theOperation->load_const_u32((Uint32)2, (Uint32)5);
+ theOperation->load_const_u32((Uint32)3, (Uint32)1);
+ theOperation->def_label(Tlabel++);
+ theOperation->def_label(Tlabel++);
+ theOperation->sub_reg((Uint32)2, (Uint32)3, (Uint32)2);
+ theOperation->branch_ne((Uint32)2, (Uint32)0, (Uint32)0);
+ theOperation->load_const_u32((Uint32)2, (Uint32)5);
+ theOperation->sub_reg((Uint32)1, (Uint32)3, (Uint32)1);
+ theOperation->branch_ne((Uint32)1, (Uint32)0, (Uint32)1);
+
+ theOperation->load_const_u32((Uint32)1, (Uint32)2); // Load register 1 with 2
+ theOperation->load_const_u32((Uint32)2, (Uint32)3); // Load register 2 with 3
+ theOperation->add_reg((Uint32)1, (Uint32)2, (Uint32)1); // 2+3 = 5 into reg 1
+ theOperation->load_const_u32((Uint32)2, (Uint32)5); // Load register 2 with 5
+
+ theOperation->def_label(Tlabel++);
+
+ theOperation->branch_eq((Uint32)1, (Uint32)2, Tlabel);
+ theOperation->interpret_exit_nok((Uint32)6001);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->branch_ne((Uint32)1, (Uint32)2, Tlabel);
+ theOperation->branch_label(Tlabel + 1);
+ theOperation->def_label(Tlabel++);
+ theOperation->interpret_exit_nok((Uint32)6002);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->branch_lt((Uint32)1, (Uint32)2, Tlabel);
+ theOperation->branch_label(Tlabel + 1);
+ theOperation->def_label(Tlabel++);
+ theOperation->interpret_exit_nok((Uint32)6003);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->branch_gt((Uint32)1, (Uint32)2, Tlabel);
+ theOperation->branch_label(Tlabel + 1);
+ theOperation->def_label(Tlabel++);
+ theOperation->interpret_exit_nok((Uint32)6005);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->branch_eq_null((Uint32)1, Tlabel);
+ theOperation->branch_label(Tlabel + 1);
+ theOperation->def_label(Tlabel++);
+ theOperation->interpret_exit_nok((Uint32)6006);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->branch_ne_null((Uint32)1,Tlabel);
+ theOperation->interpret_exit_nok((Uint32)6007);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->branch_ge((Uint32)1, (Uint32)2, Tlabel);
+ theOperation->interpret_exit_nok((Uint32)6008);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->branch_eq_null((Uint32)6,Tlabel);
+ theOperation->interpret_exit_nok((Uint32)6009);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->branch_ne_null((Uint32)6, Tlabel);
+ theOperation->branch_label(Tlabel + 1);
+ theOperation->def_label(Tlabel++);
+ theOperation->interpret_exit_nok((Uint32)6010);
+
+ theOperation->def_label(Tlabel++);
+
+ theOperation->load_const_u32((Uint32)5, (Uint32)1);
+ theOperation->add_reg((Uint32)4, (Uint32)5, (Uint32)4);
+
+ theOperation->load_const_u32((Uint32)5, (Uint32)1);
+ theOperation->branch_eq((Uint32)4, (Uint32)5, Tlabel);
+
+
+ theOperation->load_const_u32((Uint32)5, (Uint32)2);
+ theOperation->branch_eq((Uint32)4, (Uint32)5, (Tlabel + 1));
+
+ theOperation->load_const_u32((Uint32)5, (Uint32)3);
+ theOperation->branch_eq((Uint32)4, (Uint32)5, (Tlabel + 2));
+
+ theOperation->load_const_u32((Uint32)5, (Uint32)4);
+ theOperation->branch_eq((Uint32)4, (Uint32)5, (Tlabel + 3));
+
+ theOperation->branch_label(Tlabel + 4);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->load_const_u32((Uint32)1, (Uint32)200000);
+ theOperation->load_const_u32((Uint32)2, (Uint32)300000);
+ theOperation->add_reg((Uint32)1, (Uint32)2, (Uint32)1);
+ theOperation->load_const_u32((Uint32)2, (Uint32)500000);
+ theOperation->branch_label((Uint32)2);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->load_const_u32((Uint32)1, (Uint32)200000);
+ theOperation->load_const_u32((Uint32)2, (Uint32)300000);
+ theOperation->add_reg((Uint32)1, (Uint32)2, (Uint32)1);
+ theOperation->load_const_u32((Uint32)2, (Uint32)500000);
+ theOperation->branch_label((Uint32)2);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->load_const_u32((Uint32)1, (Uint32)2);
+ Uint64 x = 0;
+ theOperation->load_const_u64((Uint32)2, (Uint64)(x - 1));
+ theOperation->add_reg((Uint32)1, (Uint32)2, (Uint32)1);
+ theOperation->load_const_u32((Uint32)2, (Uint32)1);
+ theOperation->branch_label((Uint32)2);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->load_const_u32((Uint32)1, (Uint32)2);
+ theOperation->load_const_u64((Uint32)2, (Uint64)(x - 1));
+ theOperation->add_reg((Uint32)1, (Uint32)2, (Uint32)1);
+ theOperation->load_const_u64((Uint32)2, (Uint64)1);
+ theOperation->branch_label((Uint32)2);
+
+ theOperation->def_label(Tlabel++);
+ theOperation->read_attr((Uint32)1, (Uint32)2);
+ theOperation->branch_eq((Uint32)1, (Uint32)2, Tlabel);
+ theOperation->load_const_u32((Uint32)1, (Uint32)0);
+ theOperation->branch_label(Tlabel + 1);
+ theOperation->def_label(Tlabel++);
+ theOperation->load_const_u32((Uint32)1, (Uint32)1);
+ theOperation->def_label(Tlabel++);
+ theOperation->write_attr((Uint32)1, (Uint32)1);
+ ret_value = theOperation->incValue((Uint32)2, (Uint32)1);
+ if (ret_value == -1) {
+ ndbout << "Error in definition phase " << endl;
+ ndbout << "Error = " << theOperation->getNdbError() << " on line = " << theOperation->getNdbErrorLine() << endl;
+ aNdb->closeTransaction(theTransaction);
+ return ret_value;
+ }//if
+ }//for
+//------------------------------------------------------------------------------
+//------------------------------------------------------------------------------
+ ret_value = theTransaction->execute(Commit); // Perform the actual read and update
+ if (ret_value == -1) {
+ ndbout << "Error in update:" << theTransaction->getNdbError() << endl;
+ aNdb->closeTransaction(theTransaction); // < epaulsa
+ return ret_value ;
+ }//if
+//------------------------------------------------------------------------------
+//------------------------------------------------------------------------------
+ aNdb->closeTransaction(theTransaction);
+ return ret_value;
+}//update_interpreter_test()
+
+void* ThreadExec(void* ThreadData){
+
+ ThreadNdb* tabThread = (ThreadNdb*)ThreadData;
+ Ndb* pMyNdb = NULL ;
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+ int Tsuccess = 0 ;
+ int check = 0 ;
+ int loop_count_ops = 0;
+ int count, i, Ti;
+ int tType = 0 ;
+ int remType = 0 ;
+ unsigned int thread_no = 0 ;
+ unsigned long total_milliseconds;
+ unsigned int key = 0 ;
+ unsigned int prob = 0 ;
+ unsigned long transaction_time = 0 ;
+ unsigned long transaction_max_time = 0 ;
+ unsigned long min_time, max_time[MAX_TIMERS];
+ double mean_time, mean_square_time, std_time;
+
+ thread_no = tabThread->ThreadNo;
+ pMyNdb = tabThread->NdbRef;
+ if (!pMyNdb) {
+ pMyNdb = new Ndb( "TEST_DB" );
+ pMyNdb->init();
+ }//if
+
+ for (;;){
+
+ min_time = 0xFFFFFFFF;
+ //for (Ti = 0; Ti < MAX_TIMERS ; Ti++) max_time[Ti] = 0;
+ memset(&max_time, 0, sizeof max_time) ;
+ mean_time = 0;
+ mean_square_time = 0;
+ ThreadReady[thread_no] = 1;
+
+ while (!ThreadStart[thread_no]){
+ NdbSleep_MilliSleep(1);
+ }
+
+ // Check if signal to exit is received
+ if (ThreadStart[thread_no] == 999){
+ delete pMyNdb;
+ pMyNdb = NULL ;
+ ThreadReady[thread_no] = 1;
+ NdbThread_Exit(0) ;
+ return 0 ;
+ }//if
+
+ tType = ThreadStart[thread_no];
+ remType = tType;
+ ThreadStart[thread_no] = 0;
+ ThreadReady[thread_no] = 0 ;
+
+ // Start transaction, type of transaction
+ // is received in the array ThreadStart
+ loop_count_ops = tNoOfOperations;
+
+ START_TIMER_TOP
+ for (count=0 ; count < loop_count_ops ; count++) {
+
+ Tsuccess = 0;
+//----------------------------------------------------
+// Generate a random key between 0 and tNoOfRecords - 1
+//----------------------------------------------------
+ key = myRandom48(tNoOfRecords);
+//----------------------------------------------------
+// Start time measurement of transaction.
+//----------------------------------------------------
+ START_TIMER
+ //do {
+ switch (remType){
+ case 1:
+//----------------------------------------------------
+// Only lookups in short record table
+//----------------------------------------------------
+ Tsuccess = lookup(pMyNdb, key, 0, 1);
+ break;
+
+ case 2:
+//----------------------------------------------------
+// Only lookups in long record table
+//----------------------------------------------------
+ Tsuccess = lookup(pMyNdb, key, 1, 1);
+ break;
+ case 3:
+//----------------------------------------------------
+// Only updates in short record table
+//----------------------------------------------------
+ Tsuccess = update(pMyNdb, key, 0, 1);
+ break;
+ case 4:
+//----------------------------------------------------
+// Only updates in long record table
+//----------------------------------------------------
+ Tsuccess = update(pMyNdb, key, 1, 1);
+ break;
+ case 5:
+//----------------------------------------------------
+// 50% read/50 % update in short record table
+//----------------------------------------------------
+ prob = myRandom48(100);
+ if (prob < 50)
+ Tsuccess = update(pMyNdb, key, 0, 1);
+ else
+ Tsuccess = lookup(pMyNdb, key, 0, 1);
+ break;
+ case 6:
+//----------------------------------------------------
+// 50% read/50 % update in long record table
+//----------------------------------------------------
+ prob = myRandom48(100);
+ if (prob < 50)
+ Tsuccess = update(pMyNdb, key, 1, 1);
+ else
+ Tsuccess = lookup(pMyNdb, key, 1, 1);
+ break;
+ case 7:
+//----------------------------------------------------
+// 80 read/20 % update in short record table
+//----------------------------------------------------
+ prob = myRandom48(100);
+ if (prob < 20)
+ Tsuccess = update(pMyNdb, key, 0, 1);
+ else
+ Tsuccess = lookup(pMyNdb, key, 0, 1);
+ break;
+ case 8:
+//----------------------------------------------------
+// 80 read/20 % update in long record table
+//----------------------------------------------------
+ prob = myRandom48(100);
+ if (prob < 20)
+ Tsuccess = update(pMyNdb, key, 1, 1);
+ else
+ Tsuccess = lookup(pMyNdb, key, 1, 1);
+ break;
+ case 9:
+//----------------------------------------------------
+// 25 read short/25 % read long/25 % update short/25 % update long
+//----------------------------------------------------
+ prob = myRandom48(100);
+ if (prob < 25)
+ Tsuccess = update(pMyNdb, key, 0, 1);
+ else if (prob < 50)
+ Tsuccess = update(pMyNdb, key, 1, 1);
+ else if (prob < 75)
+ Tsuccess = lookup(pMyNdb, key, 0, 1);
+ else
+ Tsuccess = lookup(pMyNdb, key, 1, 1);
+ break;
+ case 10:
+//----------------------------------------------------
+// Test bug with replicated interpreted update, short table
+//----------------------------------------------------
+ Tsuccess = update_bug(pMyNdb, key, 0);
+ break;
+ case 11:
+//----------------------------------------------------
+// Test interpreter functions, short table
+//----------------------------------------------------
+ Tsuccess = update_interpreter_test(pMyNdb, key, 0);
+ break;
+ case 12:
+//----------------------------------------------------
+// Test bug with replicated interpreted update, long table
+//----------------------------------------------------
+ Tsuccess = update_bug(pMyNdb, key, 1);
+ break;
+ case 13:
+//----------------------------------------------------
+// Test interpreter functions, long table
+//----------------------------------------------------
+ Tsuccess = update_interpreter_test(pMyNdb, key, 1);
+ break;
+ case 14:
+//----------------------------------------------------
+// Only lookups in short record table
+//----------------------------------------------------
+ Tsuccess = lookup(pMyNdb, key, 0, 0);
+ break;
+ case 15:
+//----------------------------------------------------
+// Only lookups in long record table
+//----------------------------------------------------
+ Tsuccess = lookup(pMyNdb, key, 1, 0);
+ break;
+ case 16:
+//----------------------------------------------------
+// Only updates in short record table
+//----------------------------------------------------
+ Tsuccess = update(pMyNdb, key, 0, 0);
+ break;
+ case 17:
+//----------------------------------------------------
+// Only updates in long record table
+//----------------------------------------------------
+ Tsuccess = update(pMyNdb, key, 1, 0);
+ break;
+ case 18:
+ Tsuccess = multiRecordTest(pMyNdb, key);
+ break;
+ default:
+ break;
+ }//switch
+ //} while (0);//
+ if(-1 == Tsuccess) {
+ NDBT_ProgramExit(NDBT_FAILED);
+ exit(-1);
+ } // for
+//----------------------------------------------------
+// Stop time measurement of transaction.
+//----------------------------------------------------
+ STOP_TIMER
+ transaction_time = (unsigned long)timer.elapsedTime() ;//stopTimer(&theStartTime);
+//----------------------------------------------------
+// Perform calculations of time measurements.
+//----------------------------------------------------
+ transaction_max_time = transaction_time;
+ for (Ti = 0; Ti < MAX_TIMERS; Ti++) {
+ if (transaction_max_time > max_time[Ti]) {
+ Uint32 tmp = max_time[Ti];
+ max_time[Ti] = transaction_max_time;
+ transaction_max_time = tmp;
+ }//if
+ }//if
+ if (transaction_time < min_time) min_time = transaction_time;
+ mean_time = (double)transaction_time + mean_time;
+ mean_square_time = (double)(transaction_time * transaction_time) + mean_square_time;
+ }//for
+//----------------------------------------------------
+// Calculate mean and standard deviation
+//----------------------------------------------------
+ STOP_TIMER_TOP
+ total_milliseconds = (unsigned long)timer_top.elapsedTime() ;//stopTimer(&total_time);
+ mean_time = mean_time / loop_count_ops;
+ mean_square_time = mean_square_time / loop_count_ops;
+ std_time = sqrt(mean_square_time - (mean_time * mean_time));
+//----------------------------------------------------
+// Report statistics
+//----------------------------------------------------
+ ndbout << "Thread = " << thread_no << " reporting:" << endl ;
+ ndbout << "------------------------------" << endl ;
+ ndbout << "Total time is " << (unsigned int)(total_milliseconds /1000);
+ ndbout << " seconds and " << (unsigned int)(total_milliseconds % 1000);
+ ndbout << " milliseconds" << endl;
+ ndbout << "Minimum time = " << (unsigned int)min_time << " milliseconds" << endl;
+ for (Ti = 0; Ti < MAX_TIMERS; Ti++) {
+ ndbout << "Maximum timer " << Ti << " = " << (unsigned int)max_time[Ti] << " milliseconds" << endl;
+ ndbout << "Mean time = " << (unsigned int)mean_time << " milliseconds" << endl;
+ ndbout << "Standard deviation on time = " << (unsigned int)std_time;
+ ndbout << " milliseconds" << endl << endl ;
+ }//for
+ ndbout << endl ;
+
+ } // for(;;)
+
+ delete pMyNdb ;
+ NdbThread_Exit(0) ;
+ return 0 ; // Compiler is happy now
+}
+
diff --git a/ndb/test/ndbapi/ronja/initronja/Makefile b/ndb/test/ndbapi/ronja/initronja/Makefile
new file mode 100644
index 00000000000..dd66dd813d1
--- /dev/null
+++ b/ndb/test/ndbapi/ronja/initronja/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := initronja
+
+SOURCES := initronja.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/ronja/initronja/initronja.cpp b/ndb/test/ndbapi/ronja/initronja/initronja.cpp
new file mode 100644
index 00000000000..f3f4d9628e2
--- /dev/null
+++ b/ndb/test/ndbapi/ronja/initronja/initronja.cpp
@@ -0,0 +1,349 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+/* ***************************************************
+ INITRONJA
+ Initialise benchmark for Ronja Database
+ * *************************************************** */
+
+#include "NdbApi.hpp"
+#include <NdbOut.hpp>
+#include <NdbMain.h>
+#include <NdbTest.hpp>
+#include <string.h>
+
+#define MAXSTRLEN 16
+#define MAXATTR 64
+#define MAXTABLES 64
+#define MAXTHREADS 256
+#define MAXATTRSIZE 8000
+
+static unsigned int tNoOfRecords;
+static unsigned int tNoOfLoops;
+static unsigned int tNoOfTables;
+static int tAttributeSize;
+static int tNodeId;
+static unsigned int tValue;
+static unsigned int tNoOfOperations;
+static char tableName[MAXTABLES][MAXSTRLEN];
+static char attrName[MAXATTR][MAXSTRLEN];
+
+inline int InsertRecords(Ndb*, int) ;
+
+NDB_COMMAND(initronja, "initronja", "initronja", "initronja", 65535){
+
+ Ndb* pNdb = NULL ;
+ NdbSchemaCon *MySchemaTransaction = NULL ;
+ NdbSchemaOp *MySchemaOp = NULL ;
+
+
+ int check, status, i, j, cont ;
+ check = status = i = j = cont = 0 ;
+ tNoOfRecords = 500 ;
+ tNoOfLoops = tNoOfRecords / 10;
+
+ i = 1;
+ while (argc > 1){
+
+ if (strcmp(argv[i], "-r") == 0){
+ if( NULL == argv[i+1] ) goto error_input ;
+ tNoOfRecords = atoi(argv[i+1]);
+ tNoOfRecords = tNoOfRecords - (tNoOfRecords % 10);
+ tNoOfLoops = tNoOfRecords / 10;
+ if ((tNoOfRecords < 1) || (tNoOfRecords > 1000000000)) goto error_input;
+ }else{
+ goto error_input;
+ }
+
+ argc -= 2;
+ i = i + 2; //
+ }
+
+ pNdb = new Ndb( "TEST_DB" ) ;
+ ndbout << "Initialisation started. " << endl;
+ pNdb->init();
+ ndbout << "Initialisation completed. " << endl;
+
+ tNodeId = pNdb->getNodeId();
+ ndbout << endl << "Initial loading of Ronja Database" << endl;
+ ndbout << " NdbAPI node with id = " << tNodeId << endl;
+
+ if (pNdb->waitUntilReady(30) != 0) {
+ ndbout << "Benchmark failed - NDB is not ready" << endl;
+ delete pNdb ;
+ return NDBT_ProgramExit(NDBT_FAILED) ;
+ }//if
+
+ ndbout << endl << "Creating the table SHORT_REC" << "..." << endl;
+
+ MySchemaTransaction = pNdb->startSchemaTransaction();
+ if(!MySchemaTransaction) goto error_handler;
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if(!MySchemaOp) goto error_handler;
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ check = MySchemaOp->createTable( "SHORT_REC"
+ ,8 // Table Size
+ ,TupleKey // Key Type
+ ,40 // Nr of Pages
+ ,All
+ ,6
+ ,78
+ ,80
+ ,1
+ ,false);
+#else
+ check = MySchemaOp->createTable( "SHORT_REC"
+ ,8 // Table Size
+ ,TupleKey // Key Type
+ ,40 // Nr of Pages
+ );
+#endif
+ if (check == -1) goto error_handler;
+
+ ndbout << "Key attribute..." ;
+ check = MySchemaOp->createAttribute( (char*)"Key", TupleKey, 32, 1,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) goto error_handler;
+ ndbout << "\t\tOK" << endl ;
+
+ ndbout << "Flip attribute..." ;
+ check = MySchemaOp->createAttribute("Flip", NoKey, 32, 1,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) goto error_handler;
+ ndbout << "\t\tOK" << endl ;
+
+ ndbout << "Count attribute..." ;
+ check = MySchemaOp->createAttribute("Count", NoKey, 32, 1,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) goto error_handler;
+ ndbout << "\t\tOK" << endl ;
+
+ ndbout << "Placeholder attribute..." ;
+ check = MySchemaOp->createAttribute("Placeholder", NoKey, 8, 90,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) goto error_handler;
+ ndbout << "\tOK" << endl ;
+
+ if (MySchemaTransaction->execute() == -1) {
+ if(721 == MySchemaOp->getNdbError().code){
+ ndbout << "Table SHORT_REC already exists" << endl ;
+ }else{
+ ndbout << MySchemaTransaction->getNdbError() << endl;
+ }
+ }else{
+ ndbout << "SHORT_REC created " << endl;
+ }// if
+
+ pNdb->closeSchemaTransaction(MySchemaTransaction);
+
+ ndbout << endl << "Creating the table LONG_REC..." << endl;
+
+ MySchemaTransaction = pNdb->startSchemaTransaction();
+ if(!MySchemaTransaction) goto error_handler;
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if(!MySchemaOp) goto error_handler;
+#if defined NDB_OSE || defined NDB_SOFTOSE
+ check = MySchemaOp->createTable( "LONG_REC"
+ ,8 // Table Size
+ ,TupleKey // Key Type
+ ,40 // Nr of Pages
+ ,All
+ ,6
+ ,78
+ ,80
+ ,1
+ ,false);
+#else
+ check = MySchemaOp->createTable( "LONG_REC"
+ ,8 // Table Size
+ ,TupleKey // Key Type
+ ,40 // Nr of Pages
+ );
+#endif
+
+ if (check == -1) goto error_handler;
+
+ ndbout << "Key attribute..." ;
+ check = MySchemaOp->createAttribute( (char*)"Key", TupleKey, 32, 1,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) goto error_handler;
+ ndbout << "\t\tOK" << endl ;
+
+ ndbout << "Flip attribute..." ;
+ check = MySchemaOp->createAttribute("Flip", NoKey, 32, 1,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) goto error_handler;
+ ndbout << "\t\tOK" << endl ;
+
+ ndbout << "Count attribute..." ;
+ check = MySchemaOp->createAttribute("Count", NoKey, 32, 1,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) goto error_handler;
+ ndbout << "\t\tOK" << endl ;
+
+ ndbout << "Placeholder attribute..." ;
+ check = MySchemaOp->createAttribute("Placeholder", NoKey, 8, 1014,
+ UnSigned, MMBased, NotNullAttribute );
+ if (check == -1) goto error_handler;
+ ndbout << "\tOK" << endl ;
+
+ if (MySchemaTransaction->execute() == -1) {
+ if(721 == MySchemaOp->getNdbError().code){
+ ndbout << "Table LONG_REC already exists" << endl ;
+ }else{
+ ndbout << MySchemaTransaction->getNdbError() << endl;
+ }
+ }else{
+ ndbout << "LONG_REC created" << endl;
+ }// if
+
+ pNdb->closeSchemaTransaction(MySchemaTransaction);
+
+
+ check = InsertRecords(pNdb, tNoOfRecords);
+
+ delete pNdb ;
+
+ if(-1 == check){
+ ndbout << endl << "Initial loading of Ronja Database failed" << endl;
+ return NDBT_ProgramExit(NDBT_FAILED) ;
+ }else{
+ ndbout << endl << "Initial loading of Ronja Database completed" << endl;
+ return NDBT_ProgramExit(NDBT_OK) ;
+ }
+
+
+
+
+
+error_handler:
+ ndbout << "SchemaTransaction returned error:" ;
+ ndbout << MySchemaTransaction->getNdbError() << endl;
+ pNdb->closeSchemaTransaction(MySchemaTransaction);
+ delete pNdb ;
+ NDBT_ProgramExit(NDBT_FAILED) ;
+ exit(-1);
+
+error_input:
+ ndbout << endl << " Ivalid parameter(s)" << endl;
+ ndbout << " Usage: initronja [-r n] , where 'n' is the number of records to be inserted" << endl;
+ ndbout << " If omitted, 500 records will be created by default" << endl;
+ ndbout << " Note: use this number in combination with '-r' argument when running 'benchronja'" << endl << endl;
+ NDBT_ProgramExit(NDBT_WRONGARGS) ;
+ exit(1);
+}
+////////////////////////////////////////
+
+inline int InsertRecords(Ndb* pNdb, int nNoRecords){
+
+ NdbConnection *MyTransaction = NULL ;
+ NdbOperation* MyOperation[10];
+
+ int Tsuccess = 0 ;
+ int loop_count_ops = 2 * tNoOfLoops;
+ int loop_count_tables = 10;
+ int loop_count_attributes = 0 ;
+ int check = 0;
+ int count = 0 ;
+ int count_tables = 0;
+ int count_attributes = 0 ;
+ int i = 0 ;
+ int tType = 0 ;
+ unsigned int attrValue[1000];
+ unsigned int setAttrValue = 0;
+ unsigned int keyValue[3];
+
+ for (i = 0; i < 1000; i ++) attrValue[i] = 1;
+
+ for (count=0 ; count < loop_count_ops ; count++){
+ if ((((count / 100)* 100) == count) && (count != 0)){
+ ndbout << "1000 records inserted again, " << (count/100) << "000 records now inserted" << endl;
+ }
+
+ MyTransaction = pNdb->startTransaction();
+ if(!MyTransaction){
+ ndbout << "startTransaction: " << pNdb->getNdbError();
+ ndbout << " count = " << count << endl;
+ return -1 ;
+ }
+
+ for (count_tables = 0; count_tables < loop_count_tables; count_tables++) {
+ if (count < tNoOfLoops) {
+ keyValue[0] = count*10 + count_tables ;
+ MyOperation[count_tables] = MyTransaction->getNdbOperation("SHORT_REC") ;
+ }else{
+ keyValue[0] = (count - tNoOfLoops)*10 + count_tables;
+ MyOperation[count_tables] = MyTransaction->getNdbOperation("LONG_REC");
+ }//if
+
+ if (!MyOperation[count_tables]) goto error_handler1;
+
+ check = MyOperation[count_tables]->insertTuple();
+ if (check == -1) goto error_handler2;
+
+ check = MyOperation[count_tables]->equal("Key",(char*)&keyValue[0]);
+ if (check == -1) goto error_handler4;
+
+ check = MyOperation[count_tables]->setValue("Flip",(char*)&setAttrValue);
+ if (check == -1) goto error_handler5;
+
+ check = MyOperation[count_tables]->setValue("Count",(char*)&setAttrValue);
+ if (check == -1) goto error_handler5;
+
+ check = MyOperation[count_tables]->setValue("Placeholder",(char*)&attrValue[0]);
+ if (check == -1) goto error_handler5;
+ }//for
+
+ if (MyTransaction->execute( Commit ) == -1){
+ ndbout << MyTransaction->getNdbError()<< endl ;
+ ndbout << "count = " << count << endl;
+ }//if
+
+ pNdb->closeTransaction(MyTransaction) ;
+ }//for
+ return 0;
+
+error_handler1:
+ ndbout << "Error occured in getNdbOperation " << endl;
+ ndbout << MyTransaction->getNdbError() << endl;
+ pNdb->closeTransaction(MyTransaction);
+ return -1 ;
+
+error_handler2:
+ ndbout << "Error occured in defining operation " << endl;
+ ndbout << MyOperation[count_tables]->getNdbError() << endl;
+ pNdb->closeTransaction(MyTransaction);
+ return -1 ;
+
+error_handler3:
+ pNdb->closeTransaction(MyTransaction);
+ return -1 ;
+
+error_handler4:
+ ndbout << "Error occured in equal " << endl;
+ ndbout << MyOperation[count_tables]->getNdbError() << endl;
+ pNdb->closeTransaction(MyTransaction);
+ return -1 ;
+
+error_handler5:
+ ndbout << "Error occured in get/setValue " << endl;
+ ndbout << MyOperation[count_tables]->getNdbError() << endl;
+ pNdb->closeTransaction(MyTransaction);
+ return -1 ;
+
+}
diff --git a/ndb/test/ndbapi/telco/InsertRecs.cpp b/ndb/test/ndbapi/telco/InsertRecs.cpp
new file mode 100644
index 00000000000..f42786d666d
--- /dev/null
+++ b/ndb/test/ndbapi/telco/InsertRecs.cpp
@@ -0,0 +1,571 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+// InsertRecs.cpp : Defines the entry point for the console application.
+//
+
+
+#include <NdbApi.hpp>
+#include <windows.h>
+#include <tchar.h>
+
+
+// data for CALL_CONTEXT and GROUP_RESOURCE
+static TCHAR STATUS_DATA[]=_T("000102030405060708090A0B0C0D0E0F000102030405060708090A0B0C0D0E0F")
+ _T("101112131415161718191A1B1C1D1E1F000102030405060708090A0B0C0D0E0F")
+ _T("202122232425262728292A2B2C2D2E2F000102030405060708090A0B0C0D0E0F")
+ _T("303132333435363738393A3B3C3D3E3F000102030405060708090A0B0C0D0E0F")
+ _T("404142434445464748494A4B4C4D4E4F000102030405060708090A0B0C0D0E0F")
+ _T("505152535455565758595A5B5C5D5E5F000102030405060708090A0B0C0D0E0F")
+ _T("606162636465666768696A6B6C6D6E6F000102030405060708090A0B0C0D0E0F")
+ _T("707172737475767778797A7B7C7D7E7F000102030405060708090A0B0C0D0E0F")
+ _T("808182838485868788898A8B8C8D8E8F000102030405060708090A0B0C0D0E0F")
+ _T("909192939495969798999A9B9C9D9E9F000102030405060708090A0B0C0D0E0F")
+ _T("10010110210310410510610710810910A000102030405060708090A0B0C0D0EF")
+ _T("10B10C10D10E10F110111112113114115000102030405060708090A0B0C0D0EF")
+ _T("11611711811911A11B11C11D11E11F120000102030405060708090A0B0C0D0EF")
+ _T("12112212312412512612712812912A12B000102030405060708090A0B0C0D0EF")
+ _T("12C12D12E12F130131132134135136137000102030405060708090A0B0C0D0EF")
+ _T("13813913A13B13C13D13E13F140141142000102030405060708090A0B0C0D0EF")
+ _T("14314414514614714814914A14B14C14D000102030405060708090A0B0C0D0EF")
+ _T("14E14F150151152153154155156157158000102030405060708090A0B0C0D0EF")
+ _T("15915A15B15C15D15E15F160161162163000102030405060708090A0B0C0D0EF")
+ _T("16416516616716816916A16B16C16D16E000102030405060708090A0B0C0D0EF")
+ _T("16F170171172173174175176177178179000102030405060708090A0B0C0D0EF")
+ _T("17A17B17C17D17E17F180181182183184000102030405060708090A0B0C0D0EF")
+ _T("18518618718818918A18B18C18D18E18F000102030405060708090A0B0C0D0EF")
+ _T("19019119219319419519619719819919A000102030405060708090A0B0C0D0EF")
+ _T("19B19C19D19E19F200201202203204205000102030405060708090A0B0C0D0EF")
+ _T("20620720820920A20B20C20D20F210211000102030405060708090A0B0C0D0EF")
+ _T("21221321421521621721821921A21B21C000102030405060708090A0B0C0D0EF")
+ _T("21D21E21F220221222223224225226227000102030405060708090A0B0C0D0EF")
+ _T("22822922A22B22C22D22E22F230231232000102030405060708090A0B0C0D0EF")
+ _T("23323423523623723823923A23B23C23D000102030405060708090A0B0C0D0EF")
+ _T("23E23F240241242243244245246247248000102030405060708090A0B0C0D0EF")
+ _T("24924A24B24C24D24E24F250251252253000102030405060708090A0B0C0D0EF")
+ _T("101112131415161718191A1B1C1D1E1F000102030405060708090A0B0C0D0E0F")
+ _T("202122232425262728292A2B2C2D2E2F000102030405060708090A0B0C0D0E0F")
+ _T("303132333435363738393A3B3C3D3E3F000102030405060708090A0B0C0D0E0F")
+ _T("404142434445464748494A4B4C4D4E4F000102030405060708090A0B0C0D0E0F")
+ _T("505152535455565758595A5B5C5D5E5F000102030405060708090A0B0C0D0E0F")
+ _T("606162636465666768696A6B6C6D6E6F000102030405060708090A0B0C0D0E0F")
+ _T("707172737475767778797A7B7C7D7E7F000102030405060708090A0B0C0D0E0F")
+ _T("808182838485868788898A8B8C8D8E8F000102030405060708090A0B0C0D0E0F")
+ _T("909192939495969798999A9B9C9D9E9F000102030405060708090A0B0C0D0E0F")
+ _T("10010110210310410510610710810910A000102030405060708090A0B0C0D0EF")
+ _T("10B10C10D10E10F110111112113114115000102030405060708090A0B0C0D0EF")
+ _T("11611711811911A11B11C11D11E11F120000102030405060708090A0B0C0D0EF")
+ _T("12112212312412512612712812912A12B000102030405060708090A0B0C0D0EF")
+ _T("12C12D12E12F130131132134135136137000102030405060708090A0B0C0D0EF")
+ _T("13813913A13B13C13D13E13F140141142000102030405060708090A0B0C0D0EF")
+ _T("14314414514614714814914A14B14C14D000102030405060708090A0B0C0D0EF")
+ _T("14E14F150151152153154155156157158000102030405060708090A0B0C0D0EF")
+ _T("15915A15B15C15D15E15F160161162163000102030405060708090A0B0C0D0EF")
+ _T("16416516616716816916A16B16C16D16E000102030405060708090A0B0C0D0EF")
+ _T("16F170171172173174175176177178179000102030405060708090A0B0C0D0EF")
+ _T("17A17B17C17D17E17F180181182183184000102030405060708090A0B0C0D0EF")
+ _T("18518618718818918A18B18C18D18E18F000102030405060708090A0B0C0D0EF")
+ _T("19019119219319419519619719819919A000102030405060708090A0B0C0D0EF")
+ _T("19B19C19D19E19F200201202203204205000102030405060708090A0B0C0D0EF")
+ _T("20620720820920A20B20C20D20F210211000102030405060708090A0B0C0D0EF")
+ _T("21221321421521621721821921A21B21C000102030405060708090A0B0C0D0EF")
+ _T("21D21E21F220221222223224225226227000102030405060708090A0B0C0D0EF")
+ _T("22822922A22B22C22D22E22F230231232000102030405060708090A0B0C0D0EF")
+ _T("23323423523623723823923A23B23C23D000102030405060708090A0B0C0D0EF")
+ _T("2366890FE1438751097E7F6325DC0E6326F")
+ _T("25425525625725825925A25B25C25D25E25F000102030405060708090A0B0C0F");
+// Thread function for Call Context Inserts
+
+struct _ParamStruct
+{
+ HANDLE hShutdownEvent;
+ int nStartingRecordNum;
+ long* pnNumCallsProcessed;
+};
+
+HANDLE hShutdownEvent = 0;
+
+BOOL WINAPI ConsoleCtrlHandler(DWORD dwCtrlType)
+{
+ if(CTRL_C_EVENT == dwCtrlType)
+ {
+ SetEvent(hShutdownEvent);
+ return TRUE;
+ }
+ return FALSE;
+}
+
+
+
+
+DWORD WINAPI RuntimeCallContext(LPVOID lpParam)
+{
+ long nNumCallsProcessed = 0;
+
+ struct _ParamStruct* pData = (struct _ParamStruct*)lpParam;
+ int nStartingRecordID = pData->nStartingRecordNum;
+
+ Ndb* pNdb;
+ NdbConnection* pNdbConnection;
+ NdbOperation* pNdbOperation;
+ NdbRecAttr* pNdbRecAttrContextData;
+
+ char pchContextData[4008];
+
+ LARGE_INTEGER freq;
+ LARGE_INTEGER liStartTime, liEndTime;
+
+ pNdb = new Ndb("TEST_DB");
+ if(!pNdb)
+ {
+ printf("new Ndb failed\n");
+ return 0;
+ }
+
+ try
+ {
+ if(pNdb->init(1)
+ || pNdb->waitUntilReady())
+ {
+ throw pNdb;
+ }
+
+ while(WaitForSingleObject(pData->hShutdownEvent,0) != WAIT_OBJECT_0)
+ {
+ nStartingRecordID++;
+
+ bool bTimeLatency = (nStartingRecordID == 100) ? TRUE : FALSE;
+
+ if (bTimeLatency)
+ {
+ BOOL bSuccess = QueryPerformanceFrequency(&freq);
+ if (!bSuccess)
+ printf("Error retrieving frequency: %d\n", GetLastError());
+
+ }
+
+ for (int i=0; i < 20; i++)
+ {
+ switch(i)
+ {
+ case 3:
+ case 6:
+ case 9:
+ case 11:
+ case 12:
+ case 15:
+ case 18: // Query Record
+ if (bTimeLatency)
+ QueryPerformanceCounter(&liStartTime);
+
+ pNdbConnection = pNdb->startTransaction((Uint32)0, (const char*)&nStartingRecordID, (Uint32)4);
+ if(!pNdbConnection)
+ {
+ throw pNdb;
+ }
+ pNdbOperation = pNdbConnection->getNdbOperation(_T("CallContext"));
+ if(!pNdbOperation)
+ {
+ throw pNdbConnection;
+ }
+ if(pNdbOperation->readTuple()
+ || pNdbOperation->equal(_T("ContextId"), nStartingRecordID))
+ {
+ throw pNdbOperation;
+ }
+ pNdbRecAttrContextData = pNdbOperation->getValue(_T("ContextData"), pchContextData);
+ if(!pNdbRecAttrContextData)
+ {
+ throw pNdbOperation;
+ }
+ if(pNdbConnection->execute(Commit))
+ {
+ throw pNdbConnection;
+ }
+ pNdb->closeTransaction(pNdbConnection);
+
+ if (bTimeLatency)
+ {
+ QueryPerformanceCounter(&liEndTime);
+ printf("Read = %d msec.\n", (liEndTime.QuadPart - liStartTime.QuadPart) / (freq.QuadPart/1000));
+ }
+ break;
+
+ case 19: // Delete Record
+ if (bTimeLatency)
+ QueryPerformanceCounter(&liStartTime);
+
+ pNdbConnection = pNdb->startTransaction((Uint32)0, (const char*)&nStartingRecordID, (Uint32)4);
+ if(!pNdbConnection)
+ {
+ throw pNdb;
+ }
+ pNdbOperation = pNdbConnection->getNdbOperation(_T("CallContext"));
+ if(!pNdbOperation)
+ {
+ throw pNdbConnection;
+ }
+ if(pNdbOperation->deleteTuple()
+ || pNdbOperation->equal(_T("ContextId"), nStartingRecordID))
+ {
+ throw pNdbOperation;
+ }
+ if(pNdbConnection->execute(Commit))
+ {
+ throw pNdbConnection;
+ }
+ pNdb->closeTransaction(pNdbConnection);
+
+ if (bTimeLatency)
+ {
+ QueryPerformanceCounter(&liEndTime);
+ printf("Delete = %d msec.\n", (liEndTime.QuadPart - liStartTime.QuadPart) / (freq.QuadPart/1000));
+ }
+ break;
+
+ case 0: // Insert Record
+ if (bTimeLatency)
+ QueryPerformanceCounter(&liStartTime);
+
+ pNdbConnection = pNdb->startTransaction((Uint32)0, (const char*)&nStartingRecordID, (Uint32)4);
+ if(!pNdbConnection)
+ {
+ throw pNdb;
+ }
+ pNdbOperation = pNdbConnection->getNdbOperation(_T("CallContext"));
+ if(!pNdbOperation)
+ {
+ throw pNdbConnection;
+ }
+ if(pNdbOperation->insertTuple()
+ || pNdbOperation->equal(_T("ContextId"), nStartingRecordID)
+ || pNdbOperation->setValue(_T("Version"), Int32(1))
+ || pNdbOperation->setValue(_T("LockFlag"), Int32(1))
+ || pNdbOperation->setValue(_T("LockTime"), Int32(1))
+ || pNdbOperation->setValue(_T("LockTimeUSec"), Int32(1))
+ || pNdbOperation->setValue(_T("ContextData"), STATUS_DATA, sizeof(STATUS_DATA)))
+ {
+ throw pNdbOperation;
+ }
+ if(pNdbConnection->execute(Commit))
+ {
+ throw pNdbConnection;
+ }
+ pNdb->closeTransaction(pNdbConnection);
+
+ if (bTimeLatency)
+ {
+ QueryPerformanceCounter(&liEndTime);
+ printf("Insert = %d msec.\n", (liEndTime.QuadPart - liStartTime.QuadPart) / (freq.QuadPart/1000));
+ }
+ break;
+
+ default: // Update Record
+ if (bTimeLatency)
+ QueryPerformanceCounter(&liStartTime);
+
+ pNdbConnection = pNdb->startTransaction((Uint32)0, (const char*)&nStartingRecordID, (Uint32)4);
+ if(!pNdbConnection)
+ {
+ throw pNdb;
+ }
+ pNdbOperation = pNdbConnection->getNdbOperation(_T("CallContext"));
+ if(!pNdbOperation)
+ {
+ throw pNdbConnection;
+ }
+ if(pNdbOperation->updateTuple())
+ {
+ throw pNdbOperation;
+ }
+ if(pNdbOperation->equal(_T("ContextId"), nStartingRecordID)
+ || pNdbOperation->setValue(_T("ContextData"), STATUS_DATA, sizeof(STATUS_DATA)))
+ {
+ throw pNdbOperation;
+ }
+ if(pNdbConnection->execute(Commit))
+ {
+ throw pNdbConnection;
+ }
+ pNdb->closeTransaction(pNdbConnection);
+
+ if (bTimeLatency)
+ {
+ QueryPerformanceCounter(&liEndTime);
+ printf("Update = %d msec.\n", (liEndTime.QuadPart - liStartTime.QuadPart) / (freq.QuadPart/1000));
+ }
+
+ break;
+ }
+ }
+
+ nNumCallsProcessed++;
+
+ InterlockedIncrement(pData->pnNumCallsProcessed);
+ }
+
+ delete pNdb;
+ }
+ catch(Ndb* pNdb)
+ {
+ printf("%d: \n\t%s\n\t%s\n",
+ pNdb->getNdbError(),
+ pNdb->getNdbErrorString(),
+ "Ndb");
+ delete pNdb;
+ }
+ catch(NdbConnection* pNdbConnection)
+ {
+ printf("%d: \n\t%s\n\t%s\n",
+ pNdbConnection->getNdbError(),
+ pNdbConnection->getNdbErrorString(),
+ "NdbConnection");
+ pNdb->closeTransaction(pNdbConnection);
+ delete pNdb;
+ }
+ catch(NdbOperation* pNdbOperation)
+ {
+ printf("%d: \n\t%s\n\t%s\n",
+ pNdbOperation->getNdbError(),
+ pNdbOperation->getNdbErrorString(),
+ "NdbOperation");
+ pNdb->closeTransaction(pNdbConnection);
+ delete pNdb;
+ }
+
+ return 0;
+}
+
+
+void Initialize(Ndb* pNdb, long nInsert, bool bStoredTable)
+{
+ NdbSchemaCon* pNdbSchemaCon;
+ NdbSchemaOp* pNdbSchemaOp;
+ NdbConnection* pNdbConnection;
+ NdbOperation* pNdbOperation;
+
+ try
+ {
+ _tprintf(_T("Create CallContext table\n"));
+
+ pNdbSchemaCon = pNdb->startSchemaTransaction();
+ if(!pNdbSchemaCon)
+ {
+ throw pNdb;
+ }
+ pNdbSchemaOp = pNdbSchemaCon->getNdbSchemaOp();
+ if(!pNdbSchemaOp)
+ {
+ throw pNdbSchemaCon;
+ }
+ if(pNdbSchemaOp->createTable(_T("CallContext"), 8, TupleKey, 2, All, 6, 78, 80, 1, bStoredTable)
+ || pNdbSchemaOp->createAttribute(_T("ContextId"), TupleKey, 32, 1, Signed)
+ || pNdbSchemaOp->createAttribute(_T("Version"), NoKey, 32, 1, Signed)
+ || pNdbSchemaOp->createAttribute(_T("LockFlag"), NoKey, 32, 1, Signed)
+ || pNdbSchemaOp->createAttribute(_T("LockTime"), NoKey, 32, 1, Signed)
+ || pNdbSchemaOp->createAttribute(_T("LockTimeUSec"), NoKey, 32, 1, Signed)
+ || pNdbSchemaOp->createAttribute(_T("ContextData"), NoKey, 8, 4004, String))
+ {
+ throw pNdbSchemaOp;
+ }
+ if(pNdbSchemaCon->execute())
+ {
+ throw pNdbSchemaCon;
+ }
+ pNdb->closeSchemaTransaction(pNdbSchemaCon);
+
+ _tprintf(_T("Insert %d tuples in the CallContext table\n"), nInsert);
+ for(long i=0; i<nInsert; ++i)
+ {
+ long iContextId = -i;
+ pNdbConnection = pNdb->startTransaction((Uint32)0, (const char*)&iContextId, (Uint32)4);
+ if(!pNdbConnection)
+ {
+ throw pNdb;
+ }
+ pNdbOperation = pNdbConnection->getNdbOperation(_T("CallContext"));
+ if(!pNdbOperation)
+ {
+ throw pNdbConnection;
+ }
+ if(pNdbOperation->insertTuple()
+ || pNdbOperation->equal(_T("ContextId"), iContextId)
+ || pNdbOperation->setValue(_T("Version"), Int32(1))
+ || pNdbOperation->setValue(_T("LockFlag"), Int32(1))
+ || pNdbOperation->setValue(_T("LockTime"), Int32(1))
+ || pNdbOperation->setValue(_T("LockTimeUSec"), Int32(1))
+ || pNdbOperation->setValue(_T("ContextData"), STATUS_DATA, sizeof(STATUS_DATA)))
+ {
+ throw pNdbOperation;
+ }
+ if(pNdbConnection->execute(Commit))
+ {
+ throw pNdbConnection;
+ }
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ _tprintf(_T("initialisation done\n"));
+ }
+ catch(Ndb* pNdb)
+ {
+ printf("%d: \n\t%s\n\t%s\n",
+ pNdb->getNdbError(),
+ pNdb->getNdbErrorString(),
+ "Ndb");
+ delete pNdb;
+ }
+ catch(NdbConnection* pNdbConnection)
+ {
+ printf("%d: \n\t%s\n\t%s\n",
+ pNdbConnection->getNdbError(),
+ pNdbConnection->getNdbErrorString(),
+ "NdbConnection");
+ pNdb->closeTransaction(pNdbConnection);
+ delete pNdb;
+ }
+ catch(NdbOperation* pNdbOperation)
+ {
+ printf("%d: \n\t%s\n\t%s\n",
+ pNdbOperation->getNdbError(),
+ pNdbOperation->getNdbErrorString(),
+ "NdbOperation");
+ pNdb->closeTransaction(pNdbConnection);
+ delete pNdb;
+ }
+ catch(NdbSchemaCon* pNdbSchemaCon)
+ {
+ printf("%d: \n\t%s\n\t%s\n",
+ pNdbSchemaCon->getNdbError(),
+ pNdbSchemaCon->getNdbErrorString(),
+ "pNdbSchemaCon");
+ pNdb->closeSchemaTransaction(pNdbSchemaCon);
+ delete pNdb;
+ }
+ catch(NdbSchemaOp* pNdbSchemaOp)
+ {
+ printf("%d: \n\t%s\n\t%s\n",
+ pNdbSchemaOp->getNdbError(),
+ pNdbSchemaOp->getNdbErrorString(),
+ "pNdbSchemaOp");
+ pNdb->closeTransaction(pNdbConnection);
+ delete pNdb;
+ }
+}
+
+
+int _tmain(int argc, _TCHAR* argv[])
+{
+ long nNumThreads=4;
+ long nSeed = 0;
+ long nInsert = 0;
+ bool bStoredTable = true;
+ if(lstrcmp(argv[1],_T("/?")) == 0)
+ {
+ _tprintf(_T("InsertRecs [No.Of Threads] [Record Seed No.] [Init no. of rec.] [Stored?]\n"));
+ return 0;
+ }
+
+ if(argc > 1)
+ nNumThreads = _ttol(argv[1]);
+ else
+ nNumThreads = 4;
+ if (argc > 2)
+ nSeed = _ttol(argv[2]);
+ _tprintf(_T("Num of Threads = %d, Seed = %d"), nNumThreads, nSeed);
+
+ if(argc>3)
+ nInsert = _ttol(argv[3]);
+ if(argc>4)
+ bStoredTable = (_ttol(argv[4])!=0);
+
+ long nNumCallsProcessed = 0;
+
+ SetConsoleCtrlHandler(ConsoleCtrlHandler,true);
+ hShutdownEvent = CreateEvent(NULL,TRUE,FALSE,NULL);
+
+ // initiate windows sockets
+ WORD wVersionRequested;
+ WSADATA wsaData;
+ int err;
+ wVersionRequested = MAKEWORD( 2, 2 );
+ err = WSAStartup( wVersionRequested, &wsaData );
+ if ( err != 0 ) {
+ _tprintf(_T("could not find a usable WinSock DLL\n"));
+ return 0;
+ }
+ if ( LOBYTE( wsaData.wVersion ) != 2
+ || HIBYTE( wsaData.wVersion ) != 2 )
+ {
+ _tprintf(_T("could not find a usable WinSock DLL\n"));
+ WSACleanup();
+ return 0;
+ }
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if(!pNdb)
+ {
+ _tprintf(_T("could not construct ndb\n"));
+ return 0;
+ }
+ if(pNdb->init(1)
+ || pNdb->waitUntilReady())
+ {
+ _tprintf(_T("could not initialize ndb\n"));
+ return 0;
+ }
+
+ if(nInsert>0)
+ {
+ Initialize(pNdb, nInsert, bStoredTable);
+ }
+
+ if(nNumThreads>0)
+ {
+ _tprintf(_T("creating %d threads\n"), nNumThreads);
+ DWORD dwStartTime = GetTickCount();
+
+ DWORD dwThreadID = 0;
+ HANDLE hThreads[50];
+
+ struct _ParamStruct params[50];
+
+ for(int ij=0;ij<nNumThreads;ij++) {
+ params[ij].hShutdownEvent = hShutdownEvent;
+ params[ij].nStartingRecordNum = (ij*5000) + nSeed;
+ params[ij].pnNumCallsProcessed = &nNumCallsProcessed;
+ }
+
+ for(ij=0;ij<nNumThreads;ij++) {
+ hThreads[ij] = CreateThread(NULL,NULL,RuntimeCallContext,&params[ij],0,&dwThreadID);
+ }
+
+ //Wait for the threads to finish
+ WaitForMultipleObjects(nNumThreads,hThreads,TRUE,INFINITE);
+ DWORD dwEndTime = GetTickCount();
+
+ //Print time taken
+ _tprintf(_T("Time Taken for %d Calls is %ld msec (= %ld calls/sec\n"),
+ nNumCallsProcessed,dwEndTime-dwStartTime, (1000*nNumCallsProcessed/(dwEndTime-dwStartTime)));
+ }
+
+ delete pNdb;
+ WSACleanup();
+ CloseHandle(hShutdownEvent);
+
+ return 0;
+}
+
+
diff --git a/ndb/test/ndbapi/telco/Makefile b/ndb/test/ndbapi/telco/Makefile
new file mode 100644
index 00000000000..8f82c714119
--- /dev/null
+++ b/ndb/test/ndbapi/telco/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := telco
+
+# Source files of non-templated classes (.C files)
+SOURCES = msa.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/telco/adoInsertRecs.cpp b/ndb/test/ndbapi/telco/adoInsertRecs.cpp
new file mode 100644
index 00000000000..0bc67ef641b
--- /dev/null
+++ b/ndb/test/ndbapi/telco/adoInsertRecs.cpp
@@ -0,0 +1,363 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+// InsertRecs.cpp : Defines the entry point for the console application.
+//
+
+
+#include "stdafx.h"
+#import "C:\Program Files\Common Files\System\ADO\msado15.dll" \
+ no_namespace rename("EOF", "EndOfFile")
+
+
+// data for CALL_CONTEXT and GROUP_RESOURCE
+static TCHAR STATUS_DATA[]=_T("000102030405060708090A0B0C0D0E0F000102030405060708090A0B0C0D0E0F")
+ _T("101112131415161718191A1B1C1D1E1F000102030405060708090A0B0C0D0E0F")
+ _T("202122232425262728292A2B2C2D2E2F000102030405060708090A0B0C0D0E0F")
+ _T("303132333435363738393A3B3C3D3E3F000102030405060708090A0B0C0D0E0F")
+ _T("404142434445464748494A4B4C4D4E4F000102030405060708090A0B0C0D0E0F")
+ _T("505152535455565758595A5B5C5D5E5F000102030405060708090A0B0C0D0E0F")
+ _T("606162636465666768696A6B6C6D6E6F000102030405060708090A0B0C0D0E0F")
+ _T("707172737475767778797A7B7C7D7E7F000102030405060708090A0B0C0D0E0F")
+ _T("808182838485868788898A8B8C8D8E8F000102030405060708090A0B0C0D0E0F")
+ _T("909192939495969798999A9B9C9D9E9F000102030405060708090A0B0C0D0E0F")
+ _T("10010110210310410510610710810910A000102030405060708090A0B0C0D0EF")
+ _T("10B10C10D10E10F110111112113114115000102030405060708090A0B0C0D0EF")
+ _T("11611711811911A11B11C11D11E11F120000102030405060708090A0B0C0D0EF")
+ _T("12112212312412512612712812912A12B000102030405060708090A0B0C0D0EF")
+ _T("12C12D12E12F130131132134135136137000102030405060708090A0B0C0D0EF")
+ _T("13813913A13B13C13D13E13F140141142000102030405060708090A0B0C0D0EF")
+ _T("14314414514614714814914A14B14C14D000102030405060708090A0B0C0D0EF")
+ _T("14E14F150151152153154155156157158000102030405060708090A0B0C0D0EF")
+ _T("15915A15B15C15D15E15F160161162163000102030405060708090A0B0C0D0EF")
+ _T("16416516616716816916A16B16C16D16E000102030405060708090A0B0C0D0EF")
+ _T("16F170171172173174175176177178179000102030405060708090A0B0C0D0EF")
+ _T("17A17B17C17D17E17F180181182183184000102030405060708090A0B0C0D0EF")
+ _T("18518618718818918A18B18C18D18E18F000102030405060708090A0B0C0D0EF")
+ _T("19019119219319419519619719819919A000102030405060708090A0B0C0D0EF")
+ _T("19B19C19D19E19F200201202203204205000102030405060708090A0B0C0D0EF")
+ _T("20620720820920A20B20C20D20F210211000102030405060708090A0B0C0D0EF")
+ _T("21221321421521621721821921A21B21C000102030405060708090A0B0C0D0EF")
+ _T("21D21E21F220221222223224225226227000102030405060708090A0B0C0D0EF")
+ _T("22822922A22B22C22D22E22F230231232000102030405060708090A0B0C0D0EF")
+ _T("23323423523623723823923A23B23C23D000102030405060708090A0B0C0D0EF")
+ _T("23E23F240241242243244245246247248000102030405060708090A0B0C0D0EF")
+ _T("24924A24B24C24D24E24F250251252253000102030405060708090A0B0C0D0EF")
+ _T("101112131415161718191A1B1C1D1E1F000102030405060708090A0B0C0D0E0F")
+ _T("202122232425262728292A2B2C2D2E2F000102030405060708090A0B0C0D0E0F")
+ _T("303132333435363738393A3B3C3D3E3F000102030405060708090A0B0C0D0E0F")
+ _T("404142434445464748494A4B4C4D4E4F000102030405060708090A0B0C0D0E0F")
+ _T("505152535455565758595A5B5C5D5E5F000102030405060708090A0B0C0D0E0F")
+ _T("606162636465666768696A6B6C6D6E6F000102030405060708090A0B0C0D0E0F")
+ _T("707172737475767778797A7B7C7D7E7F000102030405060708090A0B0C0D0E0F")
+ _T("808182838485868788898A8B8C8D8E8F000102030405060708090A0B0C0D0E0F")
+ _T("909192939495969798999A9B9C9D9E9F000102030405060708090A0B0C0D0E0F")
+ _T("10010110210310410510610710810910A000102030405060708090A0B0C0D0EF")
+ _T("10B10C10D10E10F110111112113114115000102030405060708090A0B0C0D0EF")
+ _T("11611711811911A11B11C11D11E11F120000102030405060708090A0B0C0D0EF")
+ _T("12112212312412512612712812912A12B000102030405060708090A0B0C0D0EF")
+ _T("12C12D12E12F130131132134135136137000102030405060708090A0B0C0D0EF")
+ _T("13813913A13B13C13D13E13F140141142000102030405060708090A0B0C0D0EF")
+ _T("14314414514614714814914A14B14C14D000102030405060708090A0B0C0D0EF")
+ _T("14E14F150151152153154155156157158000102030405060708090A0B0C0D0EF")
+ _T("15915A15B15C15D15E15F160161162163000102030405060708090A0B0C0D0EF")
+ _T("16416516616716816916A16B16C16D16E000102030405060708090A0B0C0D0EF")
+ _T("16F170171172173174175176177178179000102030405060708090A0B0C0D0EF")
+ _T("17A17B17C17D17E17F180181182183184000102030405060708090A0B0C0D0EF")
+ _T("18518618718818918A18B18C18D18E18F000102030405060708090A0B0C0D0EF")
+ _T("19019119219319419519619719819919A000102030405060708090A0B0C0D0EF")
+ _T("19B19C19D19E19F200201202203204205000102030405060708090A0B0C0D0EF")
+ _T("20620720820920A20B20C20D20F210211000102030405060708090A0B0C0D0EF")
+ _T("21221321421521621721821921A21B21C000102030405060708090A0B0C0D0EF")
+ _T("21D21E21F220221222223224225226227000102030405060708090A0B0C0D0EF")
+ _T("22822922A22B22C22D22E22F230231232000102030405060708090A0B0C0D0EF")
+ _T("23323423523623723823923A23B23C23D000102030405060708090A0B0C0D0EF")
+ _T("2366890FE1438751097E7F6325DC0E6326F")
+ _T("25425525625725825925A25B25C25D25E25F000102030405060708090A0B0C0F");
+// Thread function for Call Context Inserts
+
+struct _ParamStruct
+{
+ HANDLE hShutdownEvent;
+ int nStartingRecordNum;
+ long* pnNumCallsProcessed;
+};
+
+HANDLE hShutdownEvent = 0;
+
+BOOL WINAPI ConsoleCtrlHandler(DWORD dwCtrlType)
+{
+ if(CTRL_C_EVENT == dwCtrlType)
+ {
+ SetEvent(hShutdownEvent);
+ return TRUE;
+ }
+ return FALSE;
+}
+
+DWORD WINAPI RuntimeCallContext(LPVOID lpParam)
+{
+ long nNumCallsProcessed = 0;
+
+ struct _ParamStruct* pData = (struct _ParamStruct*)lpParam;
+ int nStartingRecordID = pData->nStartingRecordNum;
+
+ HRESULT hr = CoInitialize(NULL);
+ if(FAILED(hr))
+ {
+ printf("Error Initializing COM Library\n");
+ return (int)hr;
+ }
+
+ _ConnectionPtr cn = NULL;
+ _CommandPtr cmdUpdate = NULL, cmdInsert = NULL, cmdDelete = NULL, cmdSelect = NULL;
+ _RecordsetPtr rs = NULL;
+ _ParameterPtr paramContextID = NULL;
+ _ParameterPtr paramVersion = NULL;
+ _ParameterPtr paramLockFlag = NULL;
+ _ParameterPtr ttparamLockFlag = NULL;
+ _ParameterPtr paramLockTime = NULL;
+ _ParameterPtr paramLockTimeUSec = NULL;
+ _ParameterPtr paramContextData = NULL;
+ _variant_t vtVersion;
+ _variant_t vtLockFlag;
+ _variant_t vtLockTime;
+ _variant_t vtLockTimeUSec;
+ _variant_t vtContextData;
+ // Initialize Values
+ vtVersion = CALL_CONTEXT_VERSION;
+ vtLockFlag = CALL_CONTEXT_LOCK_FLAG;
+ vtLockTime = CALL_CONTEXT_LOCK_TIME;
+ vtLockTimeUSec = CALL_CONTEXT_LOCK_TIME_USEC;
+ vtContextData = STATUS_DATA;
+
+ LARGE_INTEGER freq;
+
+ DWORD dwStartTime, dwEndTime;
+ LARGE_INTEGER liStartTime, liEndTime;
+
+ try
+ {
+ cn.CreateInstance(__uuidof(Connection));
+ cn->ConnectionString = _T("DSN=TTTelcoCS;");
+ cn->Open(_T(""),_T(""),_T(""),adConnectUnspecified);
+
+ cmdUpdate.CreateInstance(__uuidof(Command));
+ cmdInsert.CreateInstance(__uuidof(Command));
+ cmdDelete.CreateInstance(__uuidof(Command));
+ cmdSelect.CreateInstance(__uuidof(Command));
+
+ TCHAR tszInsert[10000], tszUpdate[10000];
+ memset(tszInsert, 0, sizeof(tszInsert));
+ memset(tszUpdate, 0, sizeof(tszUpdate));
+ strcpy(tszInsert, "INSERT INTO dbo.CallContext(ContextId,Version,LockFlag,LockTime,LockTimeUSec,ContextData) VALUES(?,?,?,?,?,'");
+ strcat(tszInsert, STATUS_DATA);
+ strcat(tszInsert, "')");
+
+ cmdInsert->CommandText= tszInsert;
+ cmdInsert->ActiveConnection = cn;
+ cmdInsert->Prepared = TRUE;
+
+
+ strcpy(tszUpdate, "UPDATE dbo.CallContext SET ContextData = '");
+ strcat(tszUpdate, STATUS_DATA);
+ strcat(tszUpdate, "' WHERE ContextId = ?");
+ cmdUpdate->CommandText= tszUpdate;
+ cmdUpdate->ActiveConnection = cn;
+ cmdUpdate->Prepared = TRUE;
+
+ cmdDelete->CommandText=_T("DELETE FROM dbo.CallContext WHERE ContextId = ?");
+ cmdDelete->ActiveConnection = cn;
+ cmdDelete->Prepared = TRUE;
+
+ cmdSelect->CommandText=_T("SELECT ContextData FROM dbo.CallContext WHERE ContextId = ?");
+ cmdSelect->ActiveConnection = cn;
+ cmdSelect->Prepared = TRUE;
+
+
+ //Create params
+ paramContextID = cmdInsert->CreateParameter(_T("ContextID"),adInteger,adParamInput,sizeof(int),nStartingRecordID);
+ paramVersion = cmdInsert->CreateParameter(_T("Version"),adInteger,adParamInput,sizeof(int),1);//vtVersion);
+ paramLockFlag = cmdInsert->CreateParameter(_T("LockFlag"),adInteger,adParamInput,sizeof(int),1);//vtLockFlag);
+ ttparamLockFlag = cmdUpdate->CreateParameter(_T("LockFlag"),adInteger,adParamInput,sizeof(int),1);//vtLockFlag);
+ paramLockTime = cmdInsert->CreateParameter(_T("LockTime"),adInteger,adParamInput,sizeof(int),1);//vtLockTime);
+ paramLockTimeUSec = cmdInsert->CreateParameter(_T("LockTimeUSec"),adInteger,adParamInput,sizeof(int),1);//vtLockTimeUSec);
+ paramContextData = cmdInsert->CreateParameter(_T("ContextData"), adBSTR, adParamInput, SysStringByteLen(vtContextData.bstrVal), vtContextData);
+ //paramContextData->put_Value(vtContextData);
+
+
+
+ //Append params
+ cmdInsert->Parameters->Append(paramContextID);
+ cmdInsert->Parameters->Append(paramVersion);
+ cmdInsert->Parameters->Append(paramLockFlag);
+ cmdInsert->Parameters->Append(paramLockTime);
+ cmdInsert->Parameters->Append(paramLockTimeUSec);
+ //cmdInsert->Parameters->Append(paramContextData);
+
+
+ cmdUpdate->Parameters->Append(paramContextID);
+ //cmdUpdate->Parameters->Append(paramContextID);
+
+ cmdSelect->Parameters->Append(paramContextID);
+
+ cmdDelete->Parameters->Append(paramContextID);
+
+ while(WaitForSingleObject(pData->hShutdownEvent,0) != WAIT_OBJECT_0)
+ {
+ paramContextID->Value = nStartingRecordID++;
+
+ bool bTimeLatency = (nStartingRecordID == 100) ? TRUE : FALSE;
+
+ if (bTimeLatency)
+ {
+ BOOL bSuccess = QueryPerformanceFrequency(&freq);
+ if (!bSuccess)
+ printf("Error retrieving frequency: %d\n", GetLastError());
+
+ }
+
+
+
+ for (int i=0; i < 20; i++)
+ {
+ switch(i)
+ {
+ case 3:
+ case 6:
+ case 9:
+ case 11:
+ case 12:
+ case 15:
+ case 18: // Query Record
+ if (bTimeLatency)
+ QueryPerformanceCounter(&liStartTime);
+
+ cmdSelect->Execute(NULL, NULL, -1);
+ if (bTimeLatency)
+ {
+ QueryPerformanceCounter(&liEndTime);
+ printf("Read = %d msec.\n", (liEndTime.QuadPart - liStartTime.QuadPart) / (freq.QuadPart/1000));
+ }
+ break;
+ case 19: // Delete Record
+ if (bTimeLatency)
+ QueryPerformanceCounter(&liStartTime);
+ cmdDelete->Execute(NULL,NULL,adExecuteNoRecords);
+ if (bTimeLatency)
+ {
+ QueryPerformanceCounter(&liEndTime);
+ printf("Delete = %d msec.\n", (liEndTime.QuadPart - liStartTime.QuadPart) / (freq.QuadPart/1000));
+ }
+ break;
+ case 0: // Insert Record
+ if (bTimeLatency)
+ QueryPerformanceCounter(&liStartTime);
+ cmdInsert->Execute(NULL,NULL,adExecuteNoRecords);
+ if (bTimeLatency)
+ {
+ QueryPerformanceCounter(&liEndTime);
+ printf("Insert = %d msec.\n", (liEndTime.QuadPart - liStartTime.QuadPart) / (freq.QuadPart/1000));
+ }
+ break;
+ default: // Update Record
+ if (bTimeLatency)
+ QueryPerformanceCounter(&liStartTime);
+ cmdUpdate->Execute(NULL,NULL,adExecuteNoRecords);
+ if (bTimeLatency)
+ {
+ QueryPerformanceCounter(&liEndTime);
+ printf("Update = %d msec.\n", (liEndTime.QuadPart - liStartTime.QuadPart) / (freq.QuadPart/1000));
+ }
+
+ break;
+ }
+ }
+
+ nNumCallsProcessed++;
+
+ InterlockedIncrement(pData->pnNumCallsProcessed);
+ }
+
+ cn->Close();
+ }
+ catch(_com_error &e)
+ {
+ printf("%d: \n\t%s\n\t%s\n",
+ e.Error(),
+ e.ErrorMessage(),
+ e.Source());
+
+ }
+
+ return 0;
+}
+
+
+int _tmain(int argc, _TCHAR* argv[])
+{
+ long nNumThreads=4;
+ long nSeed = 0;
+ if(lstrcmp(argv[1],_T("/?")) == 0)
+ {
+ _tprintf(_T("InsertRecs [No.Of Threads] [Record Seed No.]\n"));
+ return 0;
+ }
+
+ if(argc > 1)
+ nNumThreads = _ttol(argv[1]);
+ else
+ nNumThreads = 4;
+ if (argc > 2)
+ nSeed = _ttol(argv[2]);
+ _tprintf(_T("Num of Threads = %d, Seed = %d"), nNumThreads, nSeed);
+
+ long nNumCallsProcessed = 0;
+
+ SetConsoleCtrlHandler(ConsoleCtrlHandler,true);
+ hShutdownEvent = CreateEvent(NULL,TRUE,FALSE,NULL);
+
+ DWORD dwStartTime = GetTickCount();
+
+ DWORD dwThreadID = 0;
+ HANDLE hThreads[50];
+
+ struct _ParamStruct params[50];
+
+
+ for(int ij=0;ij<nNumThreads;ij++) {
+ params[ij].hShutdownEvent = hShutdownEvent;
+ params[ij].nStartingRecordNum = (ij*5000) + nSeed;
+ params[ij].pnNumCallsProcessed = &nNumCallsProcessed;
+ }
+
+
+ for(int ij=0;ij<nNumThreads;ij++) {
+ hThreads[ij] = CreateThread(NULL,NULL,RuntimeCallContext,&params[ij],0,&dwThreadID);
+ }
+
+ //Wait for the threads to finish
+ WaitForMultipleObjects(nNumThreads,hThreads,TRUE,INFINITE);
+ DWORD dwEndTime = GetTickCount();
+
+ CloseHandle(hShutdownEvent);
+
+ //Print time taken
+ _tprintf(_T("Time Taken for %d Calls is %ld msec (= %ld calls/sec\n"),
+ nNumCallsProcessed,dwEndTime-dwStartTime, (1000*nNumCallsProcessed/(dwEndTime-dwStartTime)));
+ return 0;
+
+}
diff --git a/ndb/test/ndbapi/telco/msa.cpp b/ndb/test/ndbapi/telco/msa.cpp
new file mode 100644
index 00000000000..f074733dce4
--- /dev/null
+++ b/ndb/test/ndbapi/telco/msa.cpp
@@ -0,0 +1,1206 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <assert.h>
+#include <math.h>
+
+#include <NdbApi.hpp>
+#include <NdbCondition.h>
+#include <NdbMutex.h>
+#include <NdbSleep.h>
+#include <NdbStdio.h>
+#include <NdbThread.h>
+#include <NdbTick.h>
+#include <NdbString.h>
+
+const char* const c_szDatabaseName = "TEST_DB";
+
+const char* const c_szTableNameStored = "CCStored";
+const char* const c_szTableNameTemp = "CCTemp";
+
+const char* const c_szContextId = "ContextId";
+const char* const c_szVersion = "Version";
+const char* const c_szLockFlag = "LockFlag";
+const char* const c_szLockTime = "LockTime";
+const char* const c_szLockTimeUSec = "LockTimeUSec";
+const char* const c_szContextData = "ContextData";
+
+const char* g_szTableName = c_szTableNameStored;
+
+
+#ifdef NDB_WIN32
+HANDLE hShutdownEvent = 0;
+#else
+#include <signal.h>
+bool bShutdownEvent = false;
+#endif
+long g_nMaxContextIdPerThread = 5000;
+long g_nNumThreads = 0;
+long g_nMaxCallsPerSecond = 0;
+long g_nMaxRetry = 50;
+bool g_bWriteTuple = false;
+bool g_bInsertInitial = false;
+bool g_bVerifyInitial = false;
+
+NdbMutex* g_pNdbMutexPrintf = 0;
+NdbMutex* g_pNdbMutexIncrement = 0;
+long g_nNumCallsProcessed = 0;
+NDB_TICKS g_tStartTime = 0;
+NDB_TICKS g_tEndTime = 0;
+
+long g_nNumberOfInitialInsert = 0;
+long g_nNumberOfInitialVerify = 0;
+
+const long c_nMaxMillisecForAllCall = 5000;
+long* g_plCountMillisecForCall = 0;
+const long c_nMaxMillisecForAllTrans = 5000;
+long* g_plCountMillisecForTrans = 0;
+bool g_bReport = false;
+bool g_bReportPlus = false;
+
+
+// data for CALL_CONTEXT and GROUP_RESOURCE
+static char STATUS_DATA[]=
+"000102030405060708090A0B0C0D0E0F000102030405060708090A0B0C0D0E0F"
+"101112131415161718191A1B1C1D1E1F000102030405060708090A0B0C0D0E0F"
+"202122232425262728292A2B2C2D2E2F000102030405060708090A0B0C0D0E0F"
+"303132333435363738393A3B3C3D3E3F000102030405060708090A0B0C0D0E0F"
+"404142434445464748494A4B4C4D4E4F000102030405060708090A0B0C0D0E0F"
+"505152535455565758595A5B5C5D5E5F000102030405060708090A0B0C0D0E0F"
+"606162636465666768696A6B6C6D6E6F000102030405060708090A0B0C0D0E0F"
+"707172737475767778797A7B7C7D7E7F000102030405060708090A0B0C0D0E0F"
+"808182838485868788898A8B8C8D8E8F000102030405060708090A0B0C0D0E0F"
+"909192939495969798999A9B9C9D9E9F000102030405060708090A0B0C0D0E0F"
+"10010110210310410510610710810910A000102030405060708090A0B0C0D0EF"
+"10B10C10D10E10F110111112113114115000102030405060708090A0B0C0D0EF"
+"11611711811911A11B11C11D11E11F120000102030405060708090A0B0C0D0EF"
+"12112212312412512612712812912A12B000102030405060708090A0B0C0D0EF"
+"12C12D12E12F130131132134135136137000102030405060708090A0B0C0D0EF"
+"13813913A13B13C13D13E13F140141142000102030405060708090A0B0C0D0EF"
+"14314414514614714814914A14B14C14D000102030405060708090A0B0C0D0EF"
+"14E14F150151152153154155156157158000102030405060708090A0B0C0D0EF"
+"15915A15B15C15D15E15F160161162163000102030405060708090A0B0C0D0EF"
+"16416516616716816916A16B16C16D16E000102030405060708090A0B0C0D0EF"
+"16F170171172173174175176177178179000102030405060708090A0B0C0D0EF"
+"17A17B17C17D17E17F180181182183184000102030405060708090A0B0C0D0EF"
+"18518618718818918A18B18C18D18E18F000102030405060708090A0B0C0D0EF"
+"19019119219319419519619719819919A000102030405060708090A0B0C0D0EF"
+"19B19C19D19E19F200201202203204205000102030405060708090A0B0C0D0EF"
+"20620720820920A20B20C20D20F210211000102030405060708090A0B0C0D0EF"
+"21221321421521621721821921A21B21C000102030405060708090A0B0C0D0EF"
+"21D21E21F220221222223224225226227000102030405060708090A0B0C0D0EF"
+"22822922A22B22C22D22E22F230231232000102030405060708090A0B0C0D0EF"
+"23323423523623723823923A23B23C23D000102030405060708090A0B0C0D0EF"
+"23E23F240241242243244245246247248000102030405060708090A0B0C0D0EF"
+"24924A24B24C24D24E24F250251252253000102030405060708090A0B0C0D0EF"
+"101112131415161718191A1B1C1D1E1F000102030405060708090A0B0C0D0E0F"
+"202122232425262728292A2B2C2D2E2F000102030405060708090A0B0C0D0E0F"
+"303132333435363738393A3B3C3D3E3F000102030405060708090A0B0C0D0E0F"
+"404142434445464748494A4B4C4D4E4F000102030405060708090A0B0C0D0E0F"
+"505152535455565758595A5B5C5D5E5F000102030405060708090A0B0C0D0E0F"
+"606162636465666768696A6B6C6D6E6F000102030405060708090A0B0C0D0E0F"
+"707172737475767778797A7B7C7D7E7F000102030405060708090A0B0C0D0E0F"
+"808182838485868788898A8B8C8D8E8F000102030405060708090A0B0C0D0E0F"
+"909192939495969798999A9B9C9D9E9F000102030405060708090A0B0C0D0E0F"
+"10010110210310410510610710810910A000102030405060708090A0B0C0D0EF"
+"10B10C10D10E10F110111112113114115000102030405060708090A0B0C0D0EF"
+"11611711811911A11B11C11D11E11F120000102030405060708090A0B0C0D0EF"
+"12112212312412512612712812912A12B000102030405060708090A0B0C0D0EF"
+"12C12D12E12F130131132134135136137000102030405060708090A0B0C0D0EF"
+"13813913A13B13C13D13E13F140141142000102030405060708090A0B0C0D0EF"
+"14314414514614714814914A14B14C14D000102030405060708090A0B0C0D0EF"
+"14E14F150151152153154155156157158000102030405060708090A0B0C0D0EF"
+"15915A15B15C15D15E15F160161162163000102030405060708090A0B0C0D0EF"
+"16416516616716816916A16B16C16D16E000102030405060708090A0B0C0D0EF"
+"16F170171172173174175176177178179000102030405060708090A0B0C0D0EF"
+"17A17B17C17D17E17F180181182183184000102030405060708090A0B0C0D0EF"
+"18518618718818918A18B18C18D18E18F000102030405060708090A0B0C0D0EF"
+"19019119219319419519619719819919A000102030405060708090A0B0C0D0EF"
+"19B19C19D19E19F200201202203204205000102030405060708090A0B0C0D0EF"
+"20620720820920A20B20C20D20F210211000102030405060708090A0B0C0D0EF"
+"21221321421521621721821921A21B21C000102030405060708090A0B0C0D0EF"
+"21D21E21F220221222223224225226227000102030405060708090A0B0C0D0EF"
+"22822922A22B22C22D22E22F230231232000102030405060708090A0B0C0D0EF"
+"23323423523623723823923A23B23C23D000102030405060708090A0B0C0D0EF"
+"2366890FE1438751097E7F6325DC0E6326F"
+"25425525625725825925A25B25C25D25E25F000102030405060708090A0B0C0F";
+
+long g_nStatusDataSize = sizeof(STATUS_DATA);
+
+
+// Thread function for Call Context Inserts
+
+
+#ifdef NDB_WIN32
+
+BOOL WINAPI ConsoleCtrlHandler(DWORD dwCtrlType)
+{
+ if(CTRL_C_EVENT == dwCtrlType)
+ {
+ SetEvent(hShutdownEvent);
+ return TRUE;
+ }
+ return FALSE;
+}
+
+#else
+
+void CtrlCHandler(int)
+{
+ bShutdownEvent = true;
+}
+
+#endif
+
+
+
+void ReportNdbError(const char* szMsg, const NdbError& err)
+{
+ NdbMutex_Lock(g_pNdbMutexPrintf);
+ printf("%s: %d: %s\n", szMsg, err.code, (err.message ? err.message : ""));
+ NdbMutex_Unlock(g_pNdbMutexPrintf);
+}
+
+
+void
+ReportCallsPerSecond(long nNumCallsProcessed,
+ NDB_TICKS tStartTime,
+ NDB_TICKS tEndTime)
+{
+ NDB_TICKS tElapsed = tEndTime - tStartTime;
+ long lCallsPerSec;
+ if(tElapsed>0)
+ lCallsPerSec = (long)((1000*nNumCallsProcessed)/tElapsed);
+ else
+ lCallsPerSec = 0;
+
+ NdbMutex_Lock(g_pNdbMutexPrintf);
+ printf("Time Taken for %ld Calls is %ld msec (= %ld calls/sec)\n",
+ nNumCallsProcessed, (long)tElapsed, lCallsPerSec);
+ NdbMutex_Unlock(g_pNdbMutexPrintf);
+}
+
+
+#ifndef NDB_WIN32
+void InterlockedIncrement(long* lp) // expensive
+{
+ NdbMutex_Lock(g_pNdbMutexIncrement);
+ (*lp)++;
+ NdbMutex_Unlock(g_pNdbMutexIncrement);
+}
+#endif
+
+
+void InterlockedIncrementAndReport(void)
+{
+ NdbMutex_Lock(g_pNdbMutexIncrement);
+ ++g_nNumCallsProcessed;
+ if((g_nNumCallsProcessed%1000)==0)
+ {
+ g_tEndTime = NdbTick_CurrentMillisecond();
+ if(g_tStartTime)
+ ReportCallsPerSecond(1000, g_tStartTime, g_tEndTime);
+
+ g_tStartTime = g_tEndTime;
+ }
+ NdbMutex_Unlock(g_pNdbMutexIncrement);
+}
+
+
+void SleepOneCall(void)
+{
+ int iMillisecToSleep;
+ if(g_nMaxCallsPerSecond>0)
+ iMillisecToSleep = (1000*g_nNumThreads)/g_nMaxCallsPerSecond;
+ else
+ iMillisecToSleep = 50;
+
+ if(iMillisecToSleep>0)
+ NdbSleep_MilliSleep(iMillisecToSleep);
+
+}
+
+
+
+int QueryTransaction(Ndb* pNdb,
+ long iContextId,
+ long* piVersion,
+ long* piLockFlag,
+ long* piLockTime,
+ long* piLockTimeUSec,
+ char* pchContextData,
+ NdbError& err)
+{
+ int iRes = -1;
+ NdbConnection* pNdbConnection = pNdb->startTransaction(0, (const char*)&iContextId, 4);
+ if(pNdbConnection)
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTableName);
+ if(pNdbOperation)
+ {
+ NdbRecAttr* pNdbRecAttrVersion;
+ NdbRecAttr* pNdbRecAttrLockFlag;
+ NdbRecAttr* pNdbRecAttrLockTime;
+ NdbRecAttr* pNdbRecAttrLockTimeUSec;
+ NdbRecAttr* pNdbRecAttrContextData;
+ if(!pNdbOperation->readTuple()
+ && !pNdbOperation->equal(c_szContextId, (Int32)iContextId)
+ && (pNdbRecAttrVersion=pNdbOperation->getValue(c_szVersion, (char*)piVersion))
+ && (pNdbRecAttrLockFlag=pNdbOperation->getValue(c_szLockFlag, (char*)piLockFlag))
+ && (pNdbRecAttrLockTime=pNdbOperation->getValue(c_szLockTime, (char*)piLockTime))
+ && (pNdbRecAttrLockTimeUSec=pNdbOperation->getValue(c_szLockTimeUSec, (char*)piLockTimeUSec))
+ && (pNdbRecAttrContextData=pNdbOperation->getValue(c_szContextData, pchContextData)))
+ {
+ if(!pNdbConnection->execute(Commit))
+ iRes = 0;
+ else
+ err = pNdbConnection->getNdbError();
+ }
+ else
+ err = pNdbOperation->getNdbError();
+ }
+ else
+ err = pNdbConnection->getNdbError();
+
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ else
+ err = pNdb->getNdbError();
+
+ return iRes;
+}
+
+
+int RetryQueryTransaction(Ndb* pNdb,
+ long iContextId,
+ long* piVersion,
+ long* piLockFlag,
+ long* piLockTime,
+ long* piLockTimeUSec,
+ char* pchContextData,
+ NdbError& err,
+ int& nRetry)
+{
+ int iRes = -1;
+ nRetry = 0;
+ bool bRetry = true;
+ while(bRetry && nRetry<g_nMaxRetry)
+ {
+ if(!QueryTransaction(pNdb, iContextId, piVersion, piLockFlag,
+ piLockTime, piLockTimeUSec, pchContextData, err))
+ {
+ iRes = 0;
+ bRetry = false;
+ }
+ else
+ {
+ switch(err.status)
+ {
+ case NdbError::TemporaryError:
+ case NdbError::UnknownResult:
+ SleepOneCall();
+ ++nRetry;
+ break;
+
+ case NdbError::PermanentError:
+ default:
+ bRetry = false;
+ break;
+ }
+ }
+ }
+ return iRes;
+}
+
+
+int DeleteTransaction(Ndb* pNdb, long iContextId, NdbError& err)
+{
+ int iRes = -1;
+ NdbConnection* pNdbConnection = pNdb->startTransaction(0, (const char*)&iContextId, 4);
+ if(pNdbConnection)
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTableName);
+ if(pNdbOperation)
+ {
+ if(!pNdbOperation->deleteTuple()
+ && !pNdbOperation->equal(c_szContextId, (Int32)iContextId))
+ {
+ if(pNdbConnection->execute(Commit) == 0)
+ iRes = 0;
+ else
+ err = pNdbConnection->getNdbError();
+ }
+ else
+ err = pNdbOperation->getNdbError();
+ }
+ else
+ err = pNdbConnection->getNdbError();
+
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ else
+ err = pNdb->getNdbError();
+
+ return iRes;
+}
+
+
+
+int RetryDeleteTransaction(Ndb* pNdb, long iContextId, NdbError& err, int& nRetry)
+{
+ int iRes = -1;
+ nRetry = 0;
+ bool bRetry = true;
+ bool bUnknown = false;
+ while(bRetry && nRetry<g_nMaxRetry)
+ {
+ if(!DeleteTransaction(pNdb, iContextId, err))
+ {
+ iRes = 0;
+ bRetry = false;
+ }
+ else
+ {
+ switch(err.status)
+ {
+ case NdbError::UnknownResult:
+ bUnknown = true;
+ ++nRetry;
+ break;
+
+ case NdbError::TemporaryError:
+ bUnknown = false;
+ SleepOneCall();
+ ++nRetry;
+ break;
+
+ case NdbError::PermanentError:
+ if(err.code==626 && bUnknown)
+ iRes = 0;
+ bRetry = false;
+ break;
+
+ default:
+ bRetry = false;
+ break;
+ }
+ }
+ }
+ return iRes;
+}
+
+
+
+int InsertTransaction(Ndb* pNdb,
+ long iContextID,
+ long iVersion,
+ long iLockFlag,
+ long iLockTime,
+ long iLockTimeUSec,
+ const char* pchContextData,
+ NdbError& err)
+{
+ int iRes = -1;
+ NdbConnection* pNdbConnection = pNdb->startTransaction(0, (const char*)&iContextID, 4);
+ if(pNdbConnection)
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTableName);
+ if(pNdbOperation)
+ {
+ if(!(g_bWriteTuple ? pNdbOperation->writeTuple() : pNdbOperation->insertTuple())
+ && !pNdbOperation->equal(c_szContextId, (Int32)iContextID)
+ && !pNdbOperation->setValue(c_szVersion, (Int32)iVersion)
+ && !pNdbOperation->setValue(c_szLockFlag, (Int32)iLockFlag)
+ && !pNdbOperation->setValue(c_szLockTime, (Int32)iLockTime)
+ && !pNdbOperation->setValue(c_szLockTimeUSec, (Int32)iLockTimeUSec)
+ && !pNdbOperation->setValue(c_szContextData, pchContextData, g_nStatusDataSize))
+ {
+ if(!pNdbConnection->execute(Commit))
+ iRes = 0;
+ else
+ err = pNdbConnection->getNdbError();
+ }
+ else
+ err = pNdbOperation->getNdbError();
+ }
+ else
+ err = pNdbConnection->getNdbError();
+
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ else
+ err = pNdb->getNdbError();
+
+ return iRes;
+}
+
+
+
+int RetryInsertTransaction(Ndb* pNdb,
+ long iContextId,
+ long iVersion,
+ long iLockFlag,
+ long iLockTime,
+ long iLockTimeUSec,
+ const char* pchContextData,
+ NdbError& err, int& nRetry)
+{
+ int iRes = -1;
+ nRetry = 0;
+ bool bRetry = true;
+ bool bUnknown = false;
+ while(bRetry && nRetry<g_nMaxRetry)
+ {
+ if(!InsertTransaction(pNdb, iContextId, iVersion, iLockFlag,
+ iLockTime, iLockTimeUSec, pchContextData, err))
+ {
+ iRes = 0;
+ bRetry = false;
+ }
+ else
+ {
+ switch(err.status)
+ {
+ case NdbError::UnknownResult:
+ bUnknown = true;
+ ++nRetry;
+ break;
+
+ case NdbError::TemporaryError:
+ bUnknown = false;
+ SleepOneCall();
+ ++nRetry;
+ break;
+
+ case NdbError::PermanentError:
+ if(err.code==630 && bUnknown)
+ iRes = 0;
+ bRetry = false;
+ break;
+
+ default:
+ bRetry = false;
+ break;
+ }
+ }
+ }
+ return iRes;
+}
+
+
+int UpdateTransaction(Ndb* pNdb, long iContextId, NdbError& err)
+{
+ int iRes = -1;
+ NdbConnection* pNdbConnection = pNdb->startTransaction(0, (const char*)&iContextId, 4);
+ if(pNdbConnection)
+ {
+ NdbOperation* pNdbOperation = pNdbConnection->getNdbOperation(g_szTableName);
+ if(pNdbOperation)
+ {
+ if(!pNdbOperation->updateTuple()
+ && !pNdbOperation->equal(c_szContextId, (Int32)iContextId)
+ && !pNdbOperation->setValue(c_szContextData, STATUS_DATA, g_nStatusDataSize))
+ {
+ if(!pNdbConnection->execute(Commit))
+ iRes = 0;
+ else
+ err = pNdbConnection->getNdbError();
+ }
+ else
+ err = pNdbOperation->getNdbError();
+ }
+ else
+ err = pNdbConnection->getNdbError();
+
+ pNdb->closeTransaction(pNdbConnection);
+ }
+ else
+ err = pNdb->getNdbError();
+
+ return iRes;
+}
+
+
+int RetryUpdateTransaction(Ndb* pNdb, long iContextId, NdbError& err, int& nRetry)
+{
+ int iRes = -1;
+ nRetry = 0;
+ bool bRetry = true;
+ while(bRetry && nRetry<g_nMaxRetry)
+ {
+ if(!UpdateTransaction(pNdb, iContextId, err))
+ {
+ iRes = 0;
+ bRetry = false;
+ }
+ else
+ {
+ switch(err.status)
+ {
+ case NdbError::TemporaryError:
+ case NdbError::UnknownResult:
+ SleepOneCall();
+ ++nRetry;
+ break;
+
+ case NdbError::PermanentError:
+ default:
+ bRetry = false;
+ break;
+ }
+ }
+ }
+ return iRes;
+}
+
+
+
+int InsertInitialRecords(Ndb* pNdb, long nInsert, long nSeed)
+{
+ int iRes = -1;
+ char szMsg[100];
+ for(long i=0; i<nInsert; ++i)
+ {
+ int iContextID = i+nSeed;
+ int nRetry = 0;
+ NdbError err;
+ memset(&err, 0, sizeof(err));
+ NDB_TICKS tStartTrans = NdbTick_CurrentMillisecond();
+ iRes = RetryInsertTransaction(pNdb, iContextID, nSeed, iContextID,
+ (long)(tStartTrans/1000), (long)((tStartTrans%1000)*1000),
+ STATUS_DATA, err, nRetry);
+ NDB_TICKS tEndTrans = NdbTick_CurrentMillisecond();
+ long lMillisecForThisTrans = (long)(tEndTrans-tStartTrans);
+ if(nRetry>0)
+ {
+ sprintf(szMsg, "insert retried %d times, time %ld msec.",
+ nRetry, lMillisecForThisTrans);
+ ReportNdbError(szMsg, err);
+ }
+ if(iRes)
+ {
+ ReportNdbError("Insert initial record failed", err);
+ return iRes;
+ }
+ InterlockedIncrement(&g_nNumberOfInitialInsert);
+ }
+ return iRes;
+}
+
+
+
+int VerifyInitialRecords(Ndb* pNdb, long nVerify, long nSeed)
+{
+ int iRes = -1;
+ char* pchContextData = new char[g_nStatusDataSize];
+ char szMsg[100];
+ long iPrevLockTime = -1;
+ long iPrevLockTimeUSec = -1;
+ for(long i=0; i<nVerify; ++i)
+ {
+ int iContextID = i+nSeed;
+ long iVersion = 0;
+ long iLockFlag = 0;
+ long iLockTime = 0;
+ long iLockTimeUSec = 0;
+ int nRetry = 0;
+ NdbError err;
+ memset(&err, 0, sizeof(err));
+ NDB_TICKS tStartTrans = NdbTick_CurrentMillisecond();
+ iRes = RetryQueryTransaction(pNdb, iContextID, &iVersion, &iLockFlag,
+ &iLockTime, &iLockTimeUSec, pchContextData, err, nRetry);
+ NDB_TICKS tEndTrans = NdbTick_CurrentMillisecond();
+ long lMillisecForThisTrans = (long)(tEndTrans-tStartTrans);
+ if(nRetry>0)
+ {
+ sprintf(szMsg, "verify retried %d times, time %ld msec.",
+ nRetry, lMillisecForThisTrans);
+ ReportNdbError(szMsg, err);
+ }
+ if(iRes)
+ {
+ ReportNdbError("Read initial record failed", err);
+ delete[] pchContextData;
+ return iRes;
+ }
+ if(memcmp(pchContextData, STATUS_DATA, g_nStatusDataSize))
+ {
+ sprintf(szMsg, "wrong context data in tuple %d", iContextID);
+ ReportNdbError(szMsg, err);
+ delete[] pchContextData;
+ return -1;
+ }
+ if(iVersion!=nSeed
+ || iLockFlag!=iContextID
+ || iLockTime<iPrevLockTime
+ || (iLockTime==iPrevLockTime && iLockTimeUSec<iPrevLockTimeUSec))
+ {
+ sprintf(szMsg, "wrong call data in tuple %d", iContextID);
+ ReportNdbError(szMsg, err);
+ delete[] pchContextData;
+ return -1;
+ }
+ iPrevLockTime = iLockTime;
+ iPrevLockTimeUSec = iLockTimeUSec;
+ InterlockedIncrement(&g_nNumberOfInitialVerify);
+ }
+ delete[] pchContextData;
+ return iRes;
+}
+
+
+
+
+
+void* RuntimeCallContext(void* lpParam)
+{
+ long nNumCallsProcessed = 0;
+ int nStartingRecordID = *(int*)lpParam;
+
+ Ndb* pNdb;
+ char* pchContextData = new char[g_nStatusDataSize];
+ char szMsg[100];
+
+ int iRes;
+ const char* szOp;
+ long iVersion;
+ long iLockFlag;
+ long iLockTime;
+ long iLockTimeUSec;
+
+ pNdb = new Ndb("TEST_DB");
+ if(!pNdb)
+ {
+ NdbMutex_Lock(g_pNdbMutexPrintf);
+ printf("new Ndb failed\n");
+ NdbMutex_Unlock(g_pNdbMutexPrintf);
+ delete[] pchContextData;
+ return 0;
+ }
+
+ if(pNdb->init(1) || pNdb->waitUntilReady())
+ {
+ ReportNdbError("init of Ndb failed", pNdb->getNdbError());
+ delete pNdb;
+ delete[] pchContextData;
+ return 0;
+ }
+
+ if(g_bInsertInitial)
+ {
+ if(InsertInitialRecords(pNdb, g_nMaxContextIdPerThread, -nStartingRecordID-g_nMaxContextIdPerThread))
+ {
+ delete pNdb;
+ delete[] pchContextData;
+ return 0;
+ }
+ }
+
+ if(g_bVerifyInitial)
+ {
+ NdbError err;
+ memset(&err, 0, sizeof(err));
+ if(VerifyInitialRecords(pNdb, g_nMaxContextIdPerThread, -nStartingRecordID-g_nMaxContextIdPerThread))
+ {
+ delete pNdb;
+ delete[] pchContextData;
+ return 0;
+ }
+ }
+ if(g_bInsertInitial || g_bVerifyInitial)
+ {
+ delete[] pchContextData;
+ return 0;
+ }
+
+ long nContextID = nStartingRecordID;
+#ifdef NDB_WIN32
+ while(WaitForSingleObject(hShutdownEvent,0) != WAIT_OBJECT_0)
+#else
+ while(!bShutdownEvent)
+#endif
+ {
+ ++nContextID;
+ nContextID %= g_nMaxContextIdPerThread;
+ nContextID += nStartingRecordID;
+
+ bool bTimeLatency = (nContextID==100);
+
+ NDB_TICKS tStartCall = NdbTick_CurrentMillisecond();
+ for (int i=0; i < 20; i++)
+ {
+ int nRetry = 0;
+ NdbError err;
+ memset(&err, 0, sizeof(err));
+ NDB_TICKS tStartTrans = NdbTick_CurrentMillisecond();
+ switch(i)
+ {
+ case 3:
+ case 6:
+ case 9:
+ case 11:
+ case 12:
+ case 15:
+ case 18: // Query Record
+ szOp = "Read";
+ iRes = RetryQueryTransaction(pNdb, nContextID, &iVersion, &iLockFlag,
+ &iLockTime, &iLockTimeUSec, pchContextData, err, nRetry);
+ break;
+
+ case 19: // Delete Record
+ szOp = "Delete";
+ iRes = RetryDeleteTransaction(pNdb, nContextID, err, nRetry);
+ break;
+
+ case 0: // Insert Record
+ szOp = "Insert";
+ iRes = RetryInsertTransaction(pNdb, nContextID, 1, 1, 1, 1, STATUS_DATA, err, nRetry);
+ break;
+
+ default: // Update Record
+ szOp = "Update";
+ iRes = RetryUpdateTransaction(pNdb, nContextID, err, nRetry);
+ break;
+ }
+ NDB_TICKS tEndTrans = NdbTick_CurrentMillisecond();
+ long lMillisecForThisTrans = (long)(tEndTrans-tStartTrans);
+
+ if(g_bReport)
+ {
+ assert(lMillisecForThisTrans>=0 && lMillisecForThisTrans<c_nMaxMillisecForAllTrans);
+ InterlockedIncrement(g_plCountMillisecForTrans+lMillisecForThisTrans);
+ }
+
+ if(nRetry>0)
+ {
+ sprintf(szMsg, "%s retried %d times, time %ld msec.",
+ szOp, nRetry, lMillisecForThisTrans);
+ ReportNdbError(szMsg, err);
+ }
+ else if(bTimeLatency)
+ {
+ NdbMutex_Lock(g_pNdbMutexPrintf);
+ printf("%s = %ld msec.\n", szOp, lMillisecForThisTrans);
+ NdbMutex_Unlock(g_pNdbMutexPrintf);
+ }
+
+ if(iRes)
+ {
+ sprintf(szMsg, "%s failed after %ld calls, terminating thread",
+ szOp, nNumCallsProcessed);
+ ReportNdbError(szMsg, err);
+ delete pNdb;
+ delete[] pchContextData;
+ return 0;
+ }
+ }
+ NDB_TICKS tEndCall = NdbTick_CurrentMillisecond();
+ long lMillisecForThisCall = (long)(tEndCall-tStartCall);
+
+ if(g_bReport)
+ {
+ assert(lMillisecForThisCall>=0 && lMillisecForThisCall<c_nMaxMillisecForAllCall);
+ InterlockedIncrement(g_plCountMillisecForCall+lMillisecForThisCall);
+ }
+
+ if(bTimeLatency)
+ {
+ NdbMutex_Lock(g_pNdbMutexPrintf);
+ printf("Total time for call is %ld msec.\n", (long)lMillisecForThisCall);
+ NdbMutex_Unlock(g_pNdbMutexPrintf);
+ }
+
+ nNumCallsProcessed++;
+ InterlockedIncrementAndReport();
+ if(g_nMaxCallsPerSecond>0)
+ {
+ int iMillisecToSleep = (1000*g_nNumThreads)/g_nMaxCallsPerSecond;
+ iMillisecToSleep -= lMillisecForThisCall;
+ if(iMillisecToSleep>0)
+ {
+ NdbSleep_MilliSleep(iMillisecToSleep);
+ }
+ }
+ }
+
+ NdbMutex_Lock(g_pNdbMutexPrintf);
+ printf("Terminating thread after %ld calls\n", nNumCallsProcessed);
+ NdbMutex_Unlock(g_pNdbMutexPrintf);
+
+ delete pNdb;
+ delete[] pchContextData;
+ return 0;
+}
+
+
+int CreateCallContextTable(Ndb* pNdb, const char* szTableName, bool bStored)
+{
+ int iRes = -1;
+ NdbError err;
+ memset(&err, 0, sizeof(err));
+
+ NdbSchemaCon* pNdbSchemaCon = pNdb->startSchemaTransaction();
+ if(pNdbSchemaCon)
+ {
+ NdbSchemaOp* pNdbSchemaOp = pNdbSchemaCon->getNdbSchemaOp();
+ if(pNdbSchemaOp)
+ {
+ if(!pNdbSchemaOp->createTable(szTableName, 8, TupleKey, 2,
+ All, 6, 78, 80, 1, bStored)
+ && !pNdbSchemaOp->createAttribute(c_szContextId, TupleKey, 32, 1, Signed)
+ && !pNdbSchemaOp->createAttribute(c_szVersion, NoKey, 32, 1, Signed)
+ && !pNdbSchemaOp->createAttribute(c_szLockFlag, NoKey, 32, 1, Signed)
+ && !pNdbSchemaOp->createAttribute(c_szLockTime, NoKey, 32, 1, Signed)
+ && !pNdbSchemaOp->createAttribute(c_szLockTimeUSec, NoKey, 32, 1, Signed)
+ && !pNdbSchemaOp->createAttribute(c_szContextData, NoKey, 8, g_nStatusDataSize, String))
+ {
+ if(!pNdbSchemaCon->execute())
+ iRes = 0;
+ else
+ err = pNdbSchemaCon->getNdbError();
+ }
+ else
+ err = pNdbSchemaOp->getNdbError();
+ }
+ else
+ err = pNdbSchemaCon->getNdbError();
+ pNdb->closeSchemaTransaction(pNdbSchemaCon);
+ }
+ else
+ err = pNdb->getNdbError();
+
+ if(iRes)
+ {
+ ReportNdbError("create call context table failed", err);
+ }
+ return iRes;
+}
+
+
+
+void ReportResponseTimeStatistics(const char* szStat, long* plCount, const long lSize)
+{
+ long lCount = 0;
+ Int64 llSum = 0;
+ Int64 llSum2 = 0;
+ long lMin = -1;
+ long lMax = -1;
+
+ for(long l=0; l<lSize; ++l)
+ {
+ if(plCount[l]>0)
+ {
+ lCount += plCount[l];
+ llSum += (Int64)l*(Int64)plCount[l];
+ llSum2 += (Int64)l*(Int64)l*(Int64)plCount[l];
+ if(lMin==-1 || l<lMin)
+ {
+ lMin = l;
+ }
+ if(lMax==-1 || l>lMax)
+ {
+ lMax = l;
+ }
+ }
+ }
+
+ long lAvg = long(llSum/lCount);
+ double dblVar = ((double)lCount*(double)llSum2 - (double)llSum*(double)llSum)/((double)lCount*(double)(lCount-1));
+ long lStd = long(sqrt(dblVar));
+
+ long lMed = -1;
+ long l95 = -1;
+ long lSel = -1;
+ for(long l=lMin; l<=lMax; ++l)
+ {
+ if(plCount[l]>0)
+ {
+ lSel += plCount[l];
+ if(lMed==-1 && lSel>=(lCount/2))
+ {
+ lMed = l;
+ }
+ if(l95==-1 && lSel>=((lCount*95)/100))
+ {
+ l95 = l;
+ }
+ if(g_bReportPlus)
+ {
+ printf("%ld\t%ld\n", l, plCount[l]);
+ }
+ }
+ }
+
+ printf("%s: Count=%ld, Min=%ld, Max=%ld, Avg=%ld, Std=%ld, Med=%ld, 95%%=%ld\n",
+ szStat, lCount, lMin, lMax, lAvg, lStd, lMed, l95);
+}
+
+
+
+void ShowHelp(const char* szCmd)
+{
+ printf("%s -t<threads> [-s<seed>] [-b<batch>] [-c<maxcps>] [-m<size>] [-d] [-i] [-v] [-f] [-w] [-r[+]]\n", szCmd);
+ printf("%s -?\n", szCmd);
+ puts("-d\t\tcreate the table");
+ puts("-i\t\tinsert initial records");
+ puts("-v\t\tverify initial records");
+ puts("-t<threads>\tnumber of threads making calls");
+ puts("-s<seed>\toffset for primary key");
+ puts("-b<batch>\tbatch size per thread");
+ puts("-c<maxcps>\tmax number of calls per second for this process");
+ puts("-m<size>\tsize of context data");
+ puts("-f\t\tno checkpointing and no logging");
+ puts("-w\t\tuse writeTuple instead of insertTuple");
+ puts("-r\t\treport response time statistics");
+ puts("-r+\t\treport response time distribution");
+ puts("-?\t\thelp");
+}
+
+
+int main(int argc, char* argv[])
+{
+ int iRes = -1;
+ g_nNumThreads = 0;
+ g_nMaxCallsPerSecond = 0;
+ long nSeed = 0;
+ bool bStoredTable = true;
+ bool bCreateTable = false;
+ g_bWriteTuple = false;
+ g_bReport = false;
+ g_bReportPlus = false;
+
+ for(int i=1; i<argc; ++i)
+ {
+ if(argv[i][0]=='-' || argv[i][0]=='/')
+ {
+ switch(argv[i][1])
+ {
+ case 't':
+ g_nNumThreads = atol(argv[i]+2);
+ break;
+ case 's':
+ nSeed = atol(argv[i]+2);
+ break;
+ case 'b':
+ g_nMaxContextIdPerThread = atol(argv[i]+2);
+ break;
+ case 'm':
+ g_nStatusDataSize = atol(argv[i]+2);
+ if(g_nStatusDataSize>sizeof(STATUS_DATA))
+ {
+ g_nStatusDataSize = sizeof(STATUS_DATA);
+ }
+ break;
+ case 'i':
+ g_bInsertInitial = true;
+ break;
+ case 'v':
+ g_bVerifyInitial = true;
+ break;
+ case 'd':
+ bCreateTable = true;
+ break;
+ case 'f':
+ bStoredTable = false;
+ break;
+ case 'w':
+ g_bWriteTuple = true;
+ break;
+ case 'r':
+ g_bReport = true;
+ if(argv[i][2]=='+')
+ {
+ g_bReportPlus = true;
+ }
+ break;
+ case 'c':
+ g_nMaxCallsPerSecond = atol(argv[i]+2);
+ break;
+ case '?':
+ default:
+ ShowHelp(argv[0]);
+ return -1;
+ }
+ }
+ else
+ {
+ ShowHelp(argv[0]);
+ return -1;
+ }
+ }
+ if(bCreateTable)
+ puts("-d\tcreate the table");
+ if(g_bInsertInitial)
+ printf("-i\tinsert initial records\n");
+ if(g_bVerifyInitial)
+ printf("-v\tverify initial records\n");
+ if(g_nNumThreads>0)
+ printf("-t%ld\tnumber of threads making calls\n", g_nNumThreads);
+ if(g_nNumThreads>0)
+ {
+ printf("-s%ld\toffset for primary key\n", nSeed);
+ printf("-b%ld\tbatch size per thread\n", g_nMaxContextIdPerThread);
+ }
+ if(g_nMaxCallsPerSecond>0)
+ printf("-c%ld\tmax number of calls per second for this process\n", g_nMaxCallsPerSecond);
+ if(!bStoredTable)
+ puts("-f\tno checkpointing and no logging to disk");
+ if(g_bWriteTuple)
+ puts("-w\tuse writeTuple instead of insertTuple");
+ if(g_bReport)
+ puts("-r\treport response time statistics");
+ if(g_bReportPlus)
+ puts("-r+\treport response time distribution");
+
+ if(!bCreateTable && g_nNumThreads<=0)
+ {
+ ShowHelp(argv[0]);
+ return -1;
+ }
+ printf("-m%ld\tsize of context data\n", g_nStatusDataSize);
+
+ g_szTableName = (bStoredTable ? c_szTableNameStored : c_szTableNameTemp);
+
+#ifdef NDB_WIN32
+ SetConsoleCtrlHandler(ConsoleCtrlHandler, true);
+#else
+ signal(SIGINT, CtrlCHandler);
+#endif
+
+ if(g_bReport)
+ {
+ g_plCountMillisecForCall = new long[c_nMaxMillisecForAllCall];
+ memset(g_plCountMillisecForCall, 0, c_nMaxMillisecForAllCall*sizeof(long));
+ g_plCountMillisecForTrans = new long[c_nMaxMillisecForAllTrans];
+ memset(g_plCountMillisecForTrans, 0, c_nMaxMillisecForAllTrans*sizeof(long));
+ }
+
+ g_pNdbMutexIncrement = NdbMutex_Create();
+ g_pNdbMutexPrintf = NdbMutex_Create();
+#ifdef NDB_WIN32
+ hShutdownEvent = CreateEvent(NULL,TRUE,FALSE,NULL);
+#endif
+
+ Ndb* pNdb = new Ndb(c_szDatabaseName);
+ if(!pNdb)
+ {
+ printf("could not construct ndb\n");
+ return 1;
+ }
+
+ if(pNdb->init(1) || pNdb->waitUntilReady())
+ {
+ ReportNdbError("could not initialize ndb\n", pNdb->getNdbError());
+ delete pNdb;
+ return 2;
+ }
+
+ if(bCreateTable)
+ {
+ printf("Create CallContext table\n");
+ if (bStoredTable)
+ {
+ if (CreateCallContextTable(pNdb, c_szTableNameStored, true))
+ {
+ printf("Create table failed\n");
+ delete pNdb;
+ return 3;
+ }
+ }
+ else
+ {
+ if (CreateCallContextTable(pNdb, c_szTableNameTemp, false))
+ {
+ printf("Create table failed\n");
+ delete pNdb;
+ return 3;
+ }
+ }
+ }
+
+ if(g_nNumThreads>0)
+ {
+ printf("creating %d threads\n", (int)g_nNumThreads);
+ if(g_bInsertInitial)
+ {
+ printf("each thread will insert %ld initial records, total %ld inserts\n",
+ g_nMaxContextIdPerThread, g_nNumThreads*g_nMaxContextIdPerThread);
+ }
+ if(g_bVerifyInitial)
+ {
+ printf("each thread will verify %ld initial records, total %ld reads\n",
+ g_nMaxContextIdPerThread, g_nNumThreads*g_nMaxContextIdPerThread);
+ }
+
+ g_nNumberOfInitialInsert = 0;
+ g_nNumberOfInitialVerify = 0;
+
+ NDB_TICKS tStartTime = NdbTick_CurrentMillisecond();
+ NdbThread* pThreads[256];
+ int pnStartingRecordNum[256];
+ int ij;
+ for(ij=0;ij<g_nNumThreads;ij++)
+ {
+ pnStartingRecordNum[ij] = (ij*g_nMaxContextIdPerThread) + nSeed;
+ }
+
+ for(ij=0;ij<g_nNumThreads;ij++)
+ {
+ pThreads[ij] = NdbThread_Create(RuntimeCallContext,
+ (void**)(pnStartingRecordNum+ij),
+ 0, "RuntimeCallContext", NDB_THREAD_PRIO_LOW);
+ }
+
+ //Wait for the threads to finish
+ for(ij=0;ij<g_nNumThreads;ij++)
+ {
+ void* status;
+ NdbThread_WaitFor(pThreads[ij], &status);
+ }
+ NDB_TICKS tEndTime = NdbTick_CurrentMillisecond();
+
+ //Print time taken
+ printf("Time Taken for %ld Calls is %ld msec (= %ld calls/sec)\n",
+ g_nNumCallsProcessed,
+ (long)(tEndTime-tStartTime),
+ (long)((1000*g_nNumCallsProcessed)/(tEndTime-tStartTime)));
+
+ if(g_bInsertInitial)
+ printf("successfully inserted %ld tuples\n", g_nNumberOfInitialInsert);
+ if(g_bVerifyInitial)
+ printf("successfully verified %ld tuples\n", g_nNumberOfInitialVerify);
+ }
+
+ delete pNdb;
+
+#ifdef NDB_WIN32
+ CloseHandle(hShutdownEvent);
+#endif
+ NdbMutex_Destroy(g_pNdbMutexIncrement);
+ NdbMutex_Destroy(g_pNdbMutexPrintf);
+
+ if(g_bReport)
+ {
+ ReportResponseTimeStatistics("Calls", g_plCountMillisecForCall, c_nMaxMillisecForAllCall);
+ ReportResponseTimeStatistics("Transactions", g_plCountMillisecForTrans, c_nMaxMillisecForAllTrans);
+
+ delete[] g_plCountMillisecForCall;
+ delete[] g_plCountMillisecForTrans;
+ }
+
+ return 0;
+}
+
diff --git a/ndb/test/ndbapi/telco/readme b/ndb/test/ndbapi/telco/readme
new file mode 100644
index 00000000000..627b4256eef
--- /dev/null
+++ b/ndb/test/ndbapi/telco/readme
@@ -0,0 +1,9 @@
+
+adoInsertRecs.cpp - the original evaluation program
+
+InsertRecs.cpp - replaced ado with ndb api, still windows only
+
+msa.cpp - removed windows and exceptions, portable
+
+
+
diff --git a/ndb/test/ndbapi/testBackup/Makefile b/ndb/test/ndbapi/testBackup/Makefile
new file mode 100644
index 00000000000..ce0e404803c
--- /dev/null
+++ b/ndb/test/ndbapi/testBackup/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testBackup
+BIN_TARGET_LIBS += bank
+SOURCES = testBackup.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testBackup/testBackup.cpp b/ndb/test/ndbapi/testBackup/testBackup.cpp
new file mode 100644
index 00000000000..f9ae7ffcbbc
--- /dev/null
+++ b/ndb/test/ndbapi/testBackup/testBackup.cpp
@@ -0,0 +1,476 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbBackup.hpp>
+
+
+#define CHECK(b) if (!(b)) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ continue; }
+
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+bool testMaster = true;
+bool testSlave = false;
+
+int setMaster(NDBT_Context* ctx, NDBT_Step* step){
+ testMaster = true;
+ testSlave = false;
+ return NDBT_OK;
+}
+int setMasterAsSlave(NDBT_Context* ctx, NDBT_Step* step){
+ testMaster = true;
+ testSlave = true;
+ return NDBT_OK;
+}
+int setSlave(NDBT_Context* ctx, NDBT_Step* step){
+ testMaster = false;
+ testSlave = true;
+ return NDBT_OK;
+}
+
+int runAbort(NDBT_Context* ctx, NDBT_Step* step){
+ NdbBackup backup(GETNDB(step)->getNodeId()+1);
+
+ NdbRestarter restarter;
+
+ if (restarter.getNumDbNodes() < 2){
+ ctx->stopTest();
+ return NDBT_OK;
+ }
+
+ if(restarter.waitClusterStarted(60) != 0){
+ g_err << "Cluster failed to start" << endl;
+ return NDBT_FAILED;
+ }
+
+ if (testMaster) {
+ if (testSlave) {
+ if (backup.NFMasterAsSlave(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ } else {
+ if (backup.NFMaster(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ }
+ } else {
+ if (backup.NFSlave(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ }
+
+ return NDBT_OK;
+}
+
+int runFail(NDBT_Context* ctx, NDBT_Step* step){
+ NdbBackup backup(GETNDB(step)->getNodeId()+1);
+
+ NdbRestarter restarter;
+
+ if (restarter.getNumDbNodes() < 2){
+ ctx->stopTest();
+ return NDBT_OK;
+ }
+
+ if(restarter.waitClusterStarted(60) != 0){
+ g_err << "Cluster failed to start" << endl;
+ return NDBT_FAILED;
+ }
+
+ if (testMaster) {
+ if (testSlave) {
+ if (backup.FailMasterAsSlave(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ } else {
+ if (backup.FailMaster(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ }
+ } else {
+ if (backup.FailSlave(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ }
+
+ return NDBT_OK;
+}
+
+int runBackupOne(NDBT_Context* ctx, NDBT_Step* step){
+ NdbBackup backup(GETNDB(step)->getNodeId()+1);
+ unsigned backupId = 0;
+
+ if (backup.start(backupId) == -1){
+ return NDBT_FAILED;
+ }
+ ndbout << "Started backup " << backupId << endl;
+ ctx->setProperty("BackupId", backupId);
+
+ return NDBT_OK;
+}
+
+int runRestartInitial(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+
+ Ndb* pNdb = GETNDB(step);
+
+ const NdbDictionary::Table *tab = ctx->getTab();
+ pNdb->getDictionary()->dropTable(tab->getName());
+
+ if (restarter.restartAll(true) != 0)
+ return NDBT_FAILED;
+
+ return NDBT_OK;
+}
+
+int runRestoreOne(NDBT_Context* ctx, NDBT_Step* step){
+ NdbBackup backup(GETNDB(step)->getNodeId()+1);
+ unsigned backupId = ctx->getProperty("BackupId");
+
+ ndbout << "Restoring backup " << backupId << endl;
+
+ if (backup.restore(backupId) == -1){
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+int runVerifyOne(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int count = 0;
+
+ ndbout << *(const NDBT_Table*)ctx->getTab() << endl;
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ do{
+
+ // Check that there are as many records as we expected
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+
+ g_err << "count = " << count;
+ g_err << " records = " << records;
+ g_err << endl;
+
+ CHECK(count == records);
+
+ // Read and verify every record
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+
+ } while (false);
+
+ return result;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+#include "../bank/Bank.hpp"
+
+int runCreateBank(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int overWriteExisting = true;
+ if (bank.createAndLoadBank(overWriteExisting) != NDBT_OK)
+ return NDBT_FAILED;
+ return NDBT_OK;
+}
+
+int runBankTimer(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int wait = 30; // Max seconds between each "day"
+ int yield = 1; // Loops before bank returns
+
+ while (ctx->isTestStopped() == false) {
+ bank.performIncreaseTime(wait, yield);
+ }
+ return NDBT_OK;
+}
+
+int runBankTransactions(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int wait = 10; // Max ms between each transaction
+ int yield = 100; // Loops before bank returns
+
+ while (ctx->isTestStopped() == false) {
+ bank.performTransactions(wait, yield);
+ }
+ return NDBT_OK;
+}
+
+int runBankGL(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int yield = 20; // Loops before bank returns
+ int result = NDBT_OK;
+
+ while (ctx->isTestStopped() == false) {
+ if (bank.performMakeGLs(yield) != NDBT_OK){
+ ndbout << "bank.performMakeGLs FAILED" << endl;
+ result = NDBT_FAILED;
+ }
+ }
+ return NDBT_OK;
+}
+
+int runBankSum(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int wait = 2000; // Max ms between each sum of accounts
+ int yield = 1; // Loops before bank returns
+ int result = NDBT_OK;
+
+ while (ctx->isTestStopped() == false) {
+ if (bank.performSumAccounts(wait, yield) != NDBT_OK){
+ ndbout << "bank.performSumAccounts FAILED" << endl;
+ result = NDBT_FAILED;
+ }
+ }
+ return result ;
+}
+
+int runDropBank(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ if (bank.dropBank() != NDBT_OK)
+ return NDBT_FAILED;
+ return NDBT_OK;
+}
+
+int runBackupBank(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int l = 0;
+ int maxSleep = 30; // Max seconds between each backup
+ Ndb* pNdb = GETNDB(step);
+ NdbBackup backup(GETNDB(step)->getNodeId()+1);
+ unsigned minBackupId = ~0;
+ unsigned maxBackupId = 0;
+ unsigned backupId = 0;
+ int result = NDBT_OK;
+
+ while (l < loops && result != NDBT_FAILED){
+
+ if (pNdb->waitUntilReady() != 0){
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ // Sleep for a while
+ NdbSleep_SecSleep(maxSleep);
+
+ // Perform backup
+ if (backup.start(backupId) != 0){
+ ndbout << "backup.start failed" << endl;
+ result = NDBT_FAILED;
+ continue;
+ }
+ ndbout << "Started backup " << backupId << endl;
+
+ // Remember min and max backupid
+ if (backupId < minBackupId)
+ minBackupId = backupId;
+
+ if (backupId > maxBackupId)
+ maxBackupId = backupId;
+
+ ndbout << " maxBackupId = " << maxBackupId
+ << ", minBackupId = " << minBackupId << endl;
+ ctx->setProperty("MinBackupId", minBackupId);
+ ctx->setProperty("MaxBackupId", maxBackupId);
+
+ l++;
+ }
+
+ ctx->stopTest();
+
+ return result;
+}
+
+int runRestoreBankAndVerify(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+ NdbBackup backup(GETNDB(step)->getNodeId()+1);
+ unsigned minBackupId = ctx->getProperty("MinBackupId");
+ unsigned maxBackupId = ctx->getProperty("MaxBackupId");
+ unsigned backupId = minBackupId;
+ int result = NDBT_OK;
+ int errSumAccounts = 0;
+ int errValidateGL = 0;
+
+ ndbout << " maxBackupId = " << maxBackupId << endl;
+ ndbout << " minBackupId = " << minBackupId << endl;
+
+ while (backupId <= maxBackupId){
+
+ // TEMPORARY FIX
+ // To erase all tables from cache(s)
+ // To be removed, maybe replaced by ndb.invalidate();
+ {
+ Bank bank;
+
+ if (bank.dropBank() != NDBT_OK){
+ result = NDBT_FAILED;
+ break;
+ }
+ }
+ // END TEMPORARY FIX
+
+ ndbout << "Performing initial restart" << endl;
+ if (restarter.restartAll(true) != 0)
+ return NDBT_FAILED;
+
+ if (restarter.waitClusterStarted() != 0)
+ return NDBT_FAILED;
+
+ ndbout << "Restoring backup " << backupId << endl;
+ if (backup.restore(backupId) == -1){
+ return NDBT_FAILED;
+ }
+ ndbout << "Backup " << backupId << " restored" << endl;
+
+ // Let bank verify
+ Bank bank;
+
+ int wait = 0;
+ int yield = 1;
+ if (bank.performSumAccounts(wait, yield) != 0){
+ ndbout << "bank.performSumAccounts FAILED" << endl;
+ ndbout << " backupId = " << backupId << endl << endl;
+ result = NDBT_FAILED;
+ errSumAccounts++;
+ }
+
+ if (bank.performValidateAllGLs() != 0){
+ ndbout << "bank.performValidateAllGLs FAILED" << endl;
+ ndbout << " backupId = " << backupId << endl << endl;
+ result = NDBT_FAILED;
+ errValidateGL++;
+ }
+
+ backupId++;
+ }
+
+ if (result != NDBT_OK){
+ ndbout << "Verification of backup failed" << endl
+ << " errValidateGL="<<errValidateGL<<endl
+ << " errSumAccounts="<<errSumAccounts<<endl << endl;
+ }
+
+ return result;
+}
+
+NDBT_TESTSUITE(testBackup);
+TESTCASE("BackupOne",
+ "Test that backup and restore works on one table \n"
+ "1. Load table\n"
+ "2. Backup\n"
+ "3. Restart -i\n"
+ "4. Restore\n"
+ "5. Verify count and content of table\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runBackupOne);
+ INITIALIZER(runRestartInitial);
+ INITIALIZER(runRestoreOne);
+ VERIFIER(runVerifyOne);
+ FINALIZER(runClearTable);
+
+}
+TESTCASE("BackupBank",
+ "Test that backup and restore works during transaction load\n"
+ " by backing up the bank"
+ "1. Create bank\n"
+ "2a. Start bank and let it run\n"
+ "2b. Perform loop number of backups of the bank\n"
+ " when backups are finished tell bank to close\n"
+ "3. Restart ndb -i and reload each backup\n"
+ " let bank verify that the backup is consistent\n"
+ "4. Drop bank\n"){
+ INITIALIZER(runCreateBank);
+ STEP(runBankTimer);
+ STEP(runBankTransactions);
+ STEP(runBankGL);
+ // TODO STEP(runBankSum);
+ STEP(runBackupBank);
+ VERIFIER(runRestoreBankAndVerify);
+ // FINALIZER(runDropBank);
+
+}
+TESTCASE("NFMaster",
+ "Test that backup behaves during node failiure\n"){
+ INITIALIZER(setMaster);
+ STEP(runAbort);
+
+}
+TESTCASE("NFMasterAsSlave",
+ "Test that backup behaves during node failiure\n"){
+ INITIALIZER(setMasterAsSlave);
+ STEP(runAbort);
+
+}
+TESTCASE("NFSlave",
+ "Test that backup behaves during node failiure\n"){
+ INITIALIZER(setSlave);
+ STEP(runAbort);
+
+}
+TESTCASE("FailMaster",
+ "Test that backup behaves during node failiure\n"){
+ INITIALIZER(setMaster);
+ STEP(runFail);
+
+}
+TESTCASE("FailMasterAsSlave",
+ "Test that backup behaves during node failiure\n"){
+ INITIALIZER(setMasterAsSlave);
+ STEP(runFail);
+
+}
+TESTCASE("FailSlave",
+ "Test that backup behaves during node failiure\n"){
+ INITIALIZER(setSlave);
+ STEP(runFail);
+
+}
+NDBT_TESTSUITE_END(testBackup);
+
+int main(int argc, const char** argv){
+ return testBackup.execute(argc, argv);
+}
+
+
diff --git a/ndb/test/ndbapi/testBasic/Makefile b/ndb/test/ndbapi/testBasic/Makefile
new file mode 100644
index 00000000000..755b19939cb
--- /dev/null
+++ b/ndb/test/ndbapi/testBasic/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := testBasic
+
+SOURCES := testBasic.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testBasic/testBasic.cpp b/ndb/test/ndbapi/testBasic/testBasic.cpp
new file mode 100644
index 00000000000..64dfe492c2c
--- /dev/null
+++ b/ndb/test/ndbapi/testBasic/testBasic.cpp
@@ -0,0 +1,1265 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT_Test.hpp>
+#include <NDBT_ReturnCodes.h>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+
+#define GETNDB(ps) ((NDBT_NdbApiStep*)ps)->getNdb()
+
+
+/**
+ * TODO
+ * dirtyWrite, write, dirtyUpdate
+ * delete should be visible to same transaction
+ *
+ */
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runInsert(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ // Insert records, dont allow any
+ // errors(except temporary) while inserting
+ if (hugoTrans.loadTable(GETNDB(step), records, 1, false) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runInsertTwice(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ // Insert records, expect primary key violation 630
+ if (hugoTrans.loadTable(GETNDB(step), records, 1, false) != 630){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runVerifyInsert(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.pkDelRecords(GETNDB(step), records, 1, false) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runInsertUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ g_info << endl;
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ g_info << endl;
+ return NDBT_OK;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.pkDelRecords(GETNDB(step), records, batchSize) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runPkDelete(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (hugoTrans.pkDelRecords(GETNDB(step), records) != 0){
+ g_info << endl;
+ return NDBT_FAILED;
+ }
+ // Load table, don't allow any primary key violations
+ if (hugoTrans.loadTable(GETNDB(step), records, 512, false) != 0){
+ g_info << endl;
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ g_info << endl;
+ return NDBT_OK;
+}
+
+
+int runPkRead(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (hugoTrans.pkReadRecords(GETNDB(step), records, batchSize) != NDBT_OK){
+ g_info << endl;
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ g_info << endl;
+ return NDBT_OK;
+}
+
+int runPkDirtyRead(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int i = 0;
+ bool dirty = true;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (hugoTrans.pkReadRecords(GETNDB(step), records,
+ batchSize, dirty) != NDBT_OK){
+ g_info << endl;
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ g_info << endl;
+ return NDBT_OK;
+}
+
+int runPkReadUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (hugoTrans.pkReadRecords(GETNDB(step), records, batchSize) != 0){
+ g_info << endl;
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ g_info << endl;
+ return NDBT_OK;
+}
+
+int runPkUpdate(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << "|- " << i << ": ";
+ if (hugoTrans.pkUpdateRecords(GETNDB(step), records, batchSize) != 0){
+ g_info << endl;
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ g_info << endl;
+ return NDBT_OK;
+}
+
+int runPkUpdateUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped()) {
+ g_info << i << ": ";
+ if (hugoTrans.pkUpdateRecords(GETNDB(step), records, batchSize) != 0){
+ g_info << endl;
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ g_info << endl;
+ return NDBT_OK;
+}
+
+int runLocker(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ if (hugoTrans.lockRecords(GETNDB(step), records, 10, 500) != 0){
+ result = NDBT_FAILED;
+ }
+ ctx->stopTest();
+
+ return result;
+}
+
+int
+runInsertOne(NDBT_Context* ctx, NDBT_Step* step){
+
+ if(ctx->getProperty("InsertCommitted", (Uint32)0) != 0){
+ abort();
+ }
+
+ while(ctx->getProperty("Read1Performed", (Uint32)0) == 0){
+ NdbSleep_MilliSleep(20);
+ }
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ if (hugoTrans.loadTable(GETNDB(step), 1, 1) != 0){
+ return NDBT_FAILED;
+ }
+
+ ctx->setProperty("InsertCommitted", 1);
+
+ NdbSleep_SecSleep(2);
+
+ return NDBT_OK;
+}
+
+static
+int
+readOneNoCommit(Ndb* pNdb, NdbConnection* pTrans,
+ const NdbDictionary::Table* tab,NDBT_ResultRow * row){
+
+ NdbOperation * pOp = pTrans->getNdbOperation(tab->getName());
+ if (pOp == NULL){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ HugoTransactions tmp(*tab);
+
+ int check = pOp->readTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab->getNoOfColumns(); a++){
+ if (tab->getColumn(a)->getPrimaryKey() == true){
+ if(tmp.equalForAttr(pOp, a, 0) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab->getNoOfColumns(); a++){
+ if((row->attributeStore(a) =
+ pOp->getValue(tab->getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ ERR(err);
+ return err.code;
+ }
+ return NDBT_OK;
+}
+
+int
+runReadOne(NDBT_Context* ctx, NDBT_Step* step){
+
+ Ndb* pNdb = GETNDB(step);
+ const NdbDictionary::Table* tab = ctx->getTab();
+ NDBT_ResultRow row1(*tab);
+ NDBT_ResultRow row2(*tab);
+
+ if(ctx->getProperty("Read1Performed", (Uint32)0) != 0){
+ abort();
+ }
+
+ if(ctx->getProperty("InsertCommitted", (Uint32)0) != 0){
+ abort();
+ }
+
+ NdbConnection * pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ abort();
+ }
+
+ // Read a record with NoCommit
+ // Since the record isn't inserted yet it wil return 626
+ const int res1 = readOneNoCommit(pNdb, pTrans, tab, &row1);
+ g_info << "|- res1 = " << res1 << endl;
+
+ ctx->setProperty("Read1Performed", 1);
+
+ while(ctx->getProperty("InsertCommitted", (Uint32)0) == 0 &&
+ !ctx->isTestStopped()){
+ g_info << "|- Waiting for insert" << endl;
+ NdbSleep_MilliSleep(20);
+ }
+
+ if(ctx->isTestStopped()){
+ abort();
+ }
+
+ // Now the record should have been inserted
+ // Read it once again in the same transaction
+ // Should also reutrn 626 if reads are consistent
+
+ // NOTE! Currently it's not possible to start a new operation
+ // on a transaction that has returned an error code
+ // This is wat fail in this test
+ // MASV 20030624
+ const int res2 = readOneNoCommit(pNdb, pTrans, tab, &row2);
+
+ pTrans->execute(Commit);
+ pNdb->closeTransaction(pTrans);
+ g_info << "|- res2 = " << res2 << endl;
+
+ if (res2 == 626 && res1 == res2)
+ return NDBT_OK;
+ else
+ return NDBT_FAILED;
+}
+
+int runFillTable(NDBT_Context* ctx, NDBT_Step* step){
+ int batch = 512; //4096;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.fillTable(GETNDB(step), batch ) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runClearTable2(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records, 240) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+#define CHECK(b) if (!(b)) { \
+ ndbout << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ break; }
+
+int runNoCommitSleep(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+ int sleepTime = 100; // ms
+ for (int i = 2; i < 8; i++){
+
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ ndbout << i <<": Sleeping for " << sleepTime << " ms" << endl;
+ NdbSleep_MilliSleep(sleepTime);
+
+ // Dont care about result of these ops
+ hugoOps.pkReadRecord(pNdb, 1, true);
+ hugoOps.closeTransaction(pNdb);
+
+ sleepTime = sleepTime *i;
+ }
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runCommit626(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Commit transaction
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 626);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Commit transaction
+ // Multiple operations
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 2, true) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 3, true) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 626);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runCommit630(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Commit transaction
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, 1) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 630);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runCommit_TryCommit626(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Commit transaction, TryCommit
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb, TryCommit) == 626);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Commit transaction, TryCommit
+ // Several operations in one transaction
+ // The insert is OK
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 2, true) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 3, true) == 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, 1) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 4, true) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb, TryCommit) == 626);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runCommit_TryCommit630(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Commit transaction, TryCommit
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, 1) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb, TryCommit) == 630);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runCommit_CommitAsMuchAsPossible626(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Commit transaction, CommitAsMuchAsPossible
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb, CommitAsMuchAsPossible) == 626);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Commit transaction, CommitAsMuchAsPossible
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 2, true) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 3, true) == 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, 1) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 4, true) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb, CommitAsMuchAsPossible) == 626);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runCommit_CommitAsMuchAsPossible630(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Commit transaction, CommitAsMuchAsPossible
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, 1) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb, CommitAsMuchAsPossible) == 630);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runNoCommit626(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // No commit transaction, readTuple
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, false) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 626);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // No commit transaction, readTupleExcluive
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 626);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runNoCommit630(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // No commit transaction
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, 1) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 630);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runNoCommitRollback626(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // No commit transaction, rollback
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 626);
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // No commit transaction, rollback
+ // Multiple operations
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 2, true) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 3, true) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 4, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 626);
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runNoCommitRollback630(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // No commit transaction, rollback
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, 1) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 630);
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+
+int runNoCommitAndClose(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Read
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ for (int i = 0; i < 10; i++)
+ CHECK(hugoOps.pkReadRecord(pNdb, i, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Update
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ for (int i = 0; i < 10; i++)
+ CHECK(hugoOps.pkUpdateRecord(pNdb, i) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Delete
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ for (int i = 0; i < 10; i++)
+ CHECK(hugoOps.pkDeleteRecord(pNdb, i) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Try to insert, record should already exist
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ for (int i = 0; i < 10; i++)
+ CHECK(hugoOps.pkInsertRecord(pNdb, i) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 630);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+
+
+int runCheckRollbackDelete(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+
+ // Read value and save it for later
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ CHECK(hugoOps.saveCopyOfRecord() == NDBT_OK);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Delete record 5
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkDeleteRecord(pNdb, 5) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ // Check record is deleted
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 626);
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Check record is not deleted
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Check record is back to original value
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ CHECK(hugoOps.compareRecordToCopy() == NDBT_OK);
+
+
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runCheckRollbackUpdate(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+ int numRecords = 5;
+ do{
+
+ // Read value and save it for later
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, false, numRecords) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ CHECK(hugoOps.verifyUpdatesValue(0) == NDBT_OK); // Update value 0
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Update record 5
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkUpdateRecord(pNdb, 1, numRecords, 5) == 0);// Updates value 5
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ // Check record is updated
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true, numRecords) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ CHECK(hugoOps.verifyUpdatesValue(5) == NDBT_OK); // Updates value 5
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Check record is back to original value
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 1, true, numRecords) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ CHECK(hugoOps.verifyUpdatesValue(0) == NDBT_OK); // Updates value 0
+
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runCheckRollbackDeleteMultiple(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Read value and save it for later
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, false, 10) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ CHECK(hugoOps.verifyUpdatesValue(0) == NDBT_OK);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ Uint32 updatesValue = 0;
+
+ for(Uint32 i = 0; i<1; i++){
+ // Read record 5 - 10
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ for(Uint32 j = 0; j<10; j++){
+ // Update record 5 - 10
+ updatesValue++;
+ CHECK(hugoOps.pkUpdateRecord(pNdb, 5, 10, updatesValue) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ CHECK(hugoOps.verifyUpdatesValue(updatesValue) == 0);
+ }
+
+ for(Uint32 j = 0; j<10; j++){
+ // Delete record 5 - 10 times
+ CHECK(hugoOps.pkDeleteRecord(pNdb, 5, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+#if 0
+ // Check records are deleted
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 626);
+#endif
+
+ updatesValue++;
+ CHECK(hugoOps.pkInsertRecord(pNdb, 5, 10, updatesValue) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ CHECK(hugoOps.verifyUpdatesValue(updatesValue) == 0);
+ }
+
+ CHECK(hugoOps.pkDeleteRecord(pNdb, 5, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ // Check records are deleted
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 626);
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+ }
+
+ // Check records are not deleted
+ // after rollback
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true, 10) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ CHECK(hugoOps.verifyUpdatesValue(0) == NDBT_OK);
+
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+
+int runCheckImplicitRollbackDelete(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Read record 5
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Update record 5
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkUpdateRecord(pNdb, 5) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Delete record 5
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkDeleteRecord(pNdb, 5) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Check record is not deleted
+ // Close transaction should have rollbacked
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runCheckCommitDelete(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Read 10 records
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ // Update 10 records
+ CHECK(hugoOps.pkUpdateRecord(pNdb, 5, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ // Delete 10 records
+ CHECK(hugoOps.pkDeleteRecord(pNdb, 5, 10) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Check record's are deleted
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true, 10) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 626);
+
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runRollbackNothing(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Delete record 5 - 15
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkDeleteRecord(pNdb, 5, 10) == 0);
+ // Rollback
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ // Check records are not deleted
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, 5, true, 10) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+
+ }while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int runMassiveRollback(NDBT_Context* ctx, NDBT_Step* step){
+
+ NdbRestarter restarter;
+ const int records = 4 * restarter.getNumDbNodes();
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ const Uint32 OPS_PER_TRANS = 256;
+ const Uint32 OPS_TOTAL = 4096;
+
+ for(int row = 0; row < records; row++){
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ for(int i = 0; i<OPS_TOTAL; i += OPS_PER_TRANS){
+ for(int j = 0; j<OPS_PER_TRANS; j++){
+ CHECK(hugoOps.pkUpdateRecord(pNdb, row, 1, i) == 0);
+ }
+ g_info << "Performed " << (i+OPS_PER_TRANS) << " updates on row: " << row
+ << endl;
+ if(result != NDBT_OK){
+ break;
+ }
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+ }
+ if(result != NDBT_OK){
+ break;
+ }
+ g_info << "executeRollback" << endl;
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+ }
+
+ hugoOps.closeTransaction(pNdb);
+ return result;
+}
+
+int
+runMassiveRollback2(NDBT_Context* ctx, NDBT_Step* step){
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), 1) != 0){
+ return NDBT_FAILED;
+ }
+
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ const Uint32 OPS_TOTAL = 4096;
+ const Uint32 LOOPS = 10;
+
+ for(int loop = 0; loop<LOOPS; loop++){
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ for(int i = 0; i<OPS_TOTAL-1; i ++){
+ if((i & 1) == 0){
+ CHECK(hugoOps.pkUpdateRecord(pNdb, 0, 1, loop) == 0);
+ } else {
+ CHECK(hugoOps.pkUpdateRecord(pNdb, 1, 1, loop) == 0);
+ }
+ }
+ CHECK(hugoOps.execute_Commit(pNdb) == 626);
+ CHECK(hugoOps.execute_Rollback(pNdb) == 0);
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+ }
+
+ hugoOps.closeTransaction(pNdb);
+ return result;
+}
+
+
+NDBT_TESTSUITE(testBasic);
+TESTCASE("PkInsert",
+ "Verify that we can insert and delete from this table using PK"
+ "NOTE! No errors are allowed!" ){
+ INITIALIZER(runInsert);
+ VERIFIER(runVerifyInsert);
+}
+TESTCASE("PkRead",
+ "Verify that we can insert, read and delete from this table using PK"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkDirtyRead",
+ "Verify that we can insert, dirty read and delete from this table using PK"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkDirtyRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkUpdate",
+ "Verify that we can insert, update and delete from this table using PK"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkUpdate);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkDelete",
+ "Verify that we can delete from this table using PK"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkDelete);
+ FINALIZER(runClearTable);
+}
+TESTCASE("UpdateAndRead",
+ "Verify that we can read and update at the same time"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkRead);
+ STEP(runPkRead);
+ STEP(runPkRead);
+ STEP(runPkUpdate);
+ STEP(runPkUpdate);
+ STEP(runPkUpdate);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkReadAndLocker",
+ "Verify that we can read although there are "\
+ " a number of 1 second locks in the table"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkReadUntilStopped);
+ STEP(runLocker);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkReadAndLocker2",
+ "Verify that we can read and update although there are "\
+ " a number of 1 second locks in the table"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runLocker);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkReadUpdateAndLocker",
+ "Verify that we can read and update although there are "\
+ " a number of 1 second locks in the table"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkUpdateUntilStopped);
+ STEP(runPkUpdateUntilStopped);
+ STEP(runLocker);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ReadWithLocksAndInserts",
+ "TR457: This test is added to verify that an insert of a records "\
+ "that is already in the database does not delete the record"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runLocker);
+ STEP(runInsertUntilStopped);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ReadConsistency",
+ "Check that a read within a transaction returns the " \
+ "same result no matter"){
+ STEP(runInsertOne);
+ STEP(runReadOne);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("PkInsertTwice",
+ "Verify that we can't insert an already inserted record."
+ "Error should be returned" ){
+ INITIALIZER(runLoadTable);
+ STEP(runInsertTwice);
+ FINALIZER(runClearTable);
+}
+TESTCASE("Fill",
+ "Verify what happens when we fill the db" ){
+ INITIALIZER(runFillTable);
+ INITIALIZER(runPkRead);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("NoCommitSleep",
+ "Verify what happens when a NoCommit transaction is aborted by "
+ "NDB because the application is sleeping" ){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runNoCommitSleep);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("Commit626",
+ "Verify what happens when a Commit transaction is aborted by "
+ "NDB because the record does no exist" ){
+ INITIALIZER(runClearTable2);
+ INITIALIZER(runCommit626);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("CommitTry626",
+ "Verify what happens when a Commit(TryCommit) \n"
+ "transaction is aborted by "
+ "NDB because the record does no exist" ){
+ INITIALIZER(runClearTable2);
+ INITIALIZER(runCommit_TryCommit626);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("CommitAsMuch626",
+ "Verify what happens when a Commit(CommitAsMuchAsPossible) \n"
+ "transaction is aborted by\n"
+ "NDB because the record does no exist" ){
+ INITIALIZER(runClearTable2);
+ INITIALIZER(runCommit_CommitAsMuchAsPossible626);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("NoCommit626",
+ "Verify what happens when a NoCommit transaction is aborted by "
+ "NDB because the record does no exist" ){
+ INITIALIZER(runClearTable2);
+ INITIALIZER(runNoCommit626);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("NoCommitRollback626",
+ "Verify what happens when a NoCommit transaction is aborted by "
+ "NDB because the record does no exist and then we try to rollback\n"
+ "the transaction" ){
+ INITIALIZER(runClearTable2);
+ INITIALIZER(runNoCommitRollback626);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("Commit630",
+ "Verify what happens when a Commit transaction is aborted by "
+ "NDB because the record already exist" ){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCommit630);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("CommitTry630",
+ "Verify what happens when a Commit(TryCommit) \n"
+ "transaction is aborted by "
+ "NDB because the record already exist" ){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCommit_TryCommit630);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("CommitAsMuch630",
+ "Verify what happens when a Commit(CommitAsMuchAsPossible) \n"
+ "transaction is aborted by\n"
+ "NDB because the record already exist" ){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCommit_CommitAsMuchAsPossible630);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("NoCommit630",
+ "Verify what happens when a NoCommit transaction is aborted by "
+ "NDB because the record already exist" ){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runNoCommit630);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("NoCommitRollback630",
+ "Verify what happens when a NoCommit transaction is aborted by "
+ "NDB because the record already exist and then we try to rollback\n"
+ "the transaction" ){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runNoCommitRollback630);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("NoCommitAndClose",
+ "Verify what happens when a NoCommit transaction is closed "
+ "without rolling back the transaction " ){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runNoCommitAndClose);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("RollbackDelete",
+ "Test rollback of a no committed delete"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCheckRollbackDelete);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("RollbackUpdate",
+ "Test rollback of a no committed update"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCheckRollbackUpdate);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("RollbackDeleteMultiple",
+ "Test rollback of 10 non committed delete"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCheckRollbackDeleteMultiple);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("ImplicitRollbackDelete",
+ "Test close transaction after a no commited delete\n"
+ "this would give an implicit rollback of the delete\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCheckImplicitRollbackDelete);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("CommitDelete",
+ "Test close transaction after a no commited delete\n"
+ "this would give an implicit rollback of the delete\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCheckCommitDelete);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("RollbackNothing",
+ "Test rollback of nothing"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runRollbackNothing);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("MassiveRollback",
+ "Test rollback of 4096 operations"){
+ INITIALIZER(runClearTable2);
+ INITIALIZER(runMassiveRollback);
+ FINALIZER(runClearTable2);
+}
+TESTCASE("MassiveRollback2",
+ "Test rollback of 4096 operations"){
+ INITIALIZER(runClearTable2);
+ INITIALIZER(runMassiveRollback2);
+ FINALIZER(runClearTable2);
+}
+NDBT_TESTSUITE_END(testBasic);
+
+int main(int argc, const char** argv){
+ return testBasic.execute(argc, argv);
+}
+
+
+
diff --git a/ndb/test/ndbapi/testBlobs/Makefile b/ndb/test/ndbapi/testBlobs/Makefile
new file mode 100644
index 00000000000..cc5bb629c17
--- /dev/null
+++ b/ndb/test/ndbapi/testBlobs/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testBlobs
+
+SOURCES = testBlobs.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
+CCFLAGS_LOC += -I$(NDB_TOP)/include/kernel
diff --git a/ndb/test/ndbapi/testBlobs/testBlobs.cpp b/ndb/test/ndbapi/testBlobs/testBlobs.cpp
new file mode 100644
index 00000000000..b8fe51dc1e4
--- /dev/null
+++ b/ndb/test/ndbapi/testBlobs/testBlobs.cpp
@@ -0,0 +1,200 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*
+ * testBlobs
+ */
+
+#include <new>
+#include <stdlib.h>
+#include <stdarg.h>
+#include <assert.h>
+#include <NdbUnistd.h>
+#include <NdbString.h>
+#include <NdbStdio.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbThread.h>
+#include <NdbMutex.h>
+#include <NdbCondition.h>
+#include <NdbTest.hpp>
+#include <NdbTick.h>
+#include <ndb_limits.h>
+
+struct Opt {
+ bool m_core;
+ const char* m_table;
+ Opt() :
+ m_core(false),
+ m_table("TB1")
+ {
+ }
+};
+
+static Opt opt;
+
+static void printusage()
+{
+ Opt d;
+ ndbout
+ << "usage: testBlobs [options]" << endl
+ << "-core dump core on error - default " << d.m_core << endl
+ ;
+}
+
+static Ndb* myNdb = 0;
+static NdbDictionary::Dictionary* myDic = 0;
+static NdbConnection* myCon = 0;
+static NdbOperation* myOp = 0;
+static NdbBlob* myBlob = 0;
+
+static void
+fatal(const char* fmt, ...)
+{
+ va_list ap;
+ char buf[200];
+ va_start(ap, fmt);
+ vsnprintf(buf, sizeof(buf), fmt, ap);
+ va_end(ap);
+ ndbout << "fatal: " << buf << endl;
+ if (myNdb != 0 && myNdb->getNdbError().code != 0)
+ ndbout << "ndb - " << myNdb->getNdbError() << endl;
+ if (myDic != 0 && myDic->getNdbError().code != 0)
+ ndbout << "dic - " << myDic->getNdbError() << endl;
+ if (opt.m_core)
+ abort();
+ NDBT_ProgramExit(NDBT_FAILED);
+ exit(1);
+}
+
+static void
+dropBlobsTable()
+{
+ NdbDictionary::Table tab(NDB_BLOB_TABLE_NAME);
+ if (myDic->dropTable(tab) == -1)
+ if (myDic->getNdbError().code != 709)
+ fatal("dropTable");
+}
+
+static void
+createBlobsTable()
+{
+ NdbDictionary::Table tab(NDB_BLOB_TABLE_NAME);
+ // col 0
+ NdbDictionary::Column col0("BLOBID");
+ col0.setPrimaryKey(true);
+ col0.setType(NdbDictionary::Column::Bigunsigned);
+ tab.addColumn(col0);
+ // col 1
+ NdbDictionary::Column col1("DATA");
+ col1.setPrimaryKey(false);
+ col1.setType(NdbDictionary::Column::Binary);
+ col1.setLength(NDB_BLOB_PIECE_SIZE);
+ tab.addColumn(col1);
+ // create
+ if (myDic->createTable(tab) == -1)
+ fatal("createTable");
+}
+
+static void
+dropTable()
+{
+ NdbDictionary::Table tab(opt.m_table);
+ if (myDic->dropTable(tab) == -1)
+ if (myDic->getNdbError().code != 709)
+ fatal("dropTable");
+}
+
+static void
+createTable()
+{
+ NdbDictionary::Table tab(opt.m_table);
+ // col 0
+ NdbDictionary::Column col0("A");
+ col0.setPrimaryKey(true);
+ col0.setType(NdbDictionary::Column::Unsigned);
+ tab.addColumn(col0);
+ // col 1
+ NdbDictionary::Column col1("B");
+ col1.setPrimaryKey(false);
+ col1.setType(NdbDictionary::Column::Blob);
+ tab.addColumn(col1);
+ // create
+ if (myDic->createTable(tab) == -1)
+ fatal("createTable");
+}
+
+static void
+insertData(Uint32 key)
+{
+}
+
+static void
+insertTuples()
+{
+ for (Uint32 key = 0; key <= 99; key++) {
+ if ((myCon = myNdb->startTransaction()) == 0)
+ fatal("startTransaction");
+ if ((myOp = myCon->getNdbOperation(opt.m_table)) == 0)
+ fatal("getNdbOperation");
+ if (myOp->insertTuple() == -1)
+ fatal("insertTuple");
+ if (myOp->setValue((unsigned)0, key) == -1)
+ fatal("setValue %u", (unsigned)key);
+ if ((myBlob = myOp->setBlob(1)) == 0)
+ fatal("setBlob");
+ if (myCon->execute(NoCommit) == -1)
+ fatal("execute NoCommit");
+ insertData(key);
+ if (myCon->execute(Commit) == -1)
+ fatal("execute Commit");
+ myNdb->closeTransaction(myCon);
+ myOp = 0;
+ myBlob = 0;
+ myCon = 0;
+ }
+}
+
+static void
+testMain()
+{
+ myNdb = new Ndb("TEST_DB");
+ if (myNdb->init() != 0)
+ fatal("init");
+ if (myNdb->waitUntilReady() < 0)
+ fatal("waitUntilReady");
+ myDic = myNdb->getDictionary();
+ dropBlobsTable();
+ createBlobsTable(); // until moved to Ndbcntr
+ dropTable();
+ createTable();
+ insertTuples();
+}
+
+NDB_COMMAND(testOdbcDriver, "testBlobs", "testBlobs", "testBlobs", 65535)
+{
+ while (++argv, --argc > 0) {
+ const char* arg = argv[0];
+ if (strcmp(arg, "-core") == 0) {
+ opt.m_core = true;
+ continue;
+ }
+ }
+ testMain();
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
+// vim: set sw=4:
diff --git a/ndb/test/ndbapi/testDataBuffers/Makefile b/ndb/test/ndbapi/testDataBuffers/Makefile
new file mode 100644
index 00000000000..181fbc829d4
--- /dev/null
+++ b/ndb/test/ndbapi/testDataBuffers/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testDataBuffers
+
+SOURCES = testDataBuffers.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testDataBuffers/testDataBuffers.cpp b/ndb/test/ndbapi/testDataBuffers/testDataBuffers.cpp
new file mode 100644
index 00000000000..fd6570fad0a
--- /dev/null
+++ b/ndb/test/ndbapi/testDataBuffers/testDataBuffers.cpp
@@ -0,0 +1,618 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*
+ * testDataBuffers
+ *
+ * Test getValue() of byte arrays:
+ * - using application buffers of different alignments and sizes
+ * - using NdbApi allocated small (<32) and big (>=32) buffers
+ *
+ * Verifies fixes to tickets 189 and 206.
+ *
+ * Options: see printusage() below.
+ *
+ * Creates tables TB00 to TB15
+ */
+
+#include <stdlib.h>
+#include <string.h>
+#include <stdarg.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NdbTest.hpp>
+#include <NdbStdio.h>
+
+// limits
+static int const MaxAttr = 64;
+static int const MaxOper = 1000;
+static int const MaxSize = 10000;
+static int const MaxOff = 64; // max offset to add to data buffer
+static int const MaxData = MaxSize + MaxOff + 100;
+
+// options
+static int attrcnt = 25;
+static int existok = 0;
+static bool kontinue = false;
+static int loopcnt = 1;
+static int opercnt = 100; // also does this many scans
+static int randomizer = 171317;
+static int sizelim = 500;
+static int xverbose = 0;
+
+static void printusage() {
+ ndbout
+ << "usage: testDataBuffers options [default/max]"
+ << endl
+ << "NOTE: too large combinations result in NDB error"
+ << endl
+ << "-a N number of attributes (including the key) [25/64]"
+ << endl
+ << "-e no error if table exists (assumed to have same structure)"
+ << endl
+ << "-k on error continue with next test case"
+ << endl
+ << "-l N number of loops to run, 0 means infinite [1]"
+ << endl
+ << "-o N number of operations (rows in each table) [100/1000]"
+ << endl
+ << "-r N source of randomness (big number (prime)) [171317]"
+ << endl
+ << "-s N array size limit (rounded up in some tests) [500/10000]"
+ << endl
+ << "-x extremely verbose"
+ << endl
+ << "Tables: TB00 .. TB15"
+ << endl
+ ;
+}
+
+static Ndb* ndb = 0;
+static NdbSchemaCon* tcon = 0;
+static NdbSchemaOp* top = 0;
+static NdbConnection* con = 0;
+static NdbOperation* op = 0;
+
+static int
+ndberror(char const* fmt, ...)
+{
+ va_list ap;
+ char buf[200];
+ va_start(ap, fmt);
+ vsnprintf(buf, sizeof(buf), fmt, ap);
+ va_end(ap);
+ ndbout << buf << " --" << endl;
+ if (ndb)
+ ndbout << "ndb : " << ndb->getNdbError() << endl;
+ if (tcon)
+ ndbout << "tcon: " << tcon->getNdbError() << endl;
+ if (top)
+ ndbout << "top : " << top->getNdbError() << endl;
+ if (con)
+ ndbout << "con : " << con->getNdbError() << endl;
+ if (op)
+ ndbout << "op : " << op->getNdbError() << endl;
+ return -1;
+}
+
+static int
+chkerror(char const* fmt, ...)
+{
+ va_list ap;
+ char buf[200];
+ va_start(ap, fmt);
+ vsnprintf(buf, sizeof(buf), fmt, ap);
+ va_end(ap);
+ ndbout << "*** check failed: " << buf << " ***" << endl;
+ return -1;
+}
+
+// alignment of addresses and data sizes
+
+static bool isAligned(unsigned x)
+{
+ return ((x & 3) == 0);
+}
+static bool isAligned(char* p)
+{
+ return isAligned(unsigned(p));
+}
+static unsigned toAligned(unsigned x)
+{
+ while (! isAligned(x))
+ x++;
+ return x;
+}
+static char* toAligned(char* p)
+{
+ while (! isAligned(p))
+ p++;
+ return p;
+}
+
+// byte value for key k column i byte j
+static int byteVal(int k, int i, int j)
+{
+ return '0' + (k + i + j) % 10;
+}
+
+// tables
+
+static char tab[20] = "";
+
+static struct col {
+ char aAttrName[20];
+ AttrType aAttrType;
+ int aAttrSize;
+ int aArraySize;
+ KeyType aTupleKey;
+ bool nullable;
+ NdbRecAttr* aRa;
+ char* buf;
+ int bufsiz;
+ char data[MaxData];
+} ccol[MaxAttr];
+
+static int key = 0;
+
+// independent test bits
+static bool alignAddr; // align our buffer addresses to 4x
+static bool alignSize; // align data sizes to 4x
+static bool useBuf; // use our buffers for output
+static bool noRandom; // do not randomize sizes and offsets
+static int testbits = 4;
+
+static int
+makeSize(int i)
+{
+ int n;
+ if (noRandom)
+ n = i;
+ else
+ n = i * randomizer;
+ n %= sizelim;
+ if (n <= 0)
+ n = 1;
+ if (alignSize)
+ n = toAligned(n);
+ return n;
+}
+
+static int
+makeOff(int k)
+{
+ int n;
+ if (alignAddr)
+ n = 0;
+ else if (noRandom)
+ n = k;
+ else
+ n = k * randomizer;
+ n %= MaxOff;
+ if (n < 0)
+ n = -n;
+ return n;
+}
+
+static int
+testcase(int flag)
+{
+ ndbout << "--- case " << flag << " ---" << endl;
+ sprintf(tab, "TB%02d", flag);
+
+ alignAddr = ! (flag & 1);
+ ndbout << (alignAddr ? "align addresses" : "mis-align addresses") << endl;
+ alignSize = ! (flag & 2);
+ ndbout << (alignSize ? "align data sizes" : "mis-align data sizes") << endl;
+ useBuf = ! (flag & 4);
+ ndbout << (useBuf ? "use our buffers" : "use ndbapi buffers") << endl;
+ noRandom = ! (flag & 8);
+ ndbout << (noRandom ? "simple sizes" : "randomize sizes") << endl;
+
+ int smax = 0, stot = 0;
+ if (xverbose)
+ ndbout << "- define table " << tab << endl;
+ for (int i = 0; i < attrcnt; i++) {
+ col& c = ccol[i];
+ memset(&c, 0, sizeof(c));
+ sprintf(c.aAttrName, "C%d", i);
+ if (i == 0) {
+ c.aAttrType = UnSigned;
+ c.aAttrSize = 32;
+ c.aArraySize = 1;
+ c.aTupleKey = TupleKey;
+ c.nullable = false;
+ } else {
+ c.aAttrType = String;
+ c.aAttrSize = 8;
+ c.aArraySize = makeSize(i);
+ if (smax < c.aArraySize)
+ smax = c.aArraySize;
+ stot += c.aArraySize;
+ c.aTupleKey = NoKey;
+ c.nullable = true;
+ if (xverbose)
+ ndbout << "-- column " << i << " size=" << c.aArraySize << endl;
+ }
+ c.buf = toAligned(c.data);
+ c.bufsiz = sizeof(c.data) - (c.buf - c.data);
+ }
+ ndbout << "tab=" << tab << " cols=" << attrcnt
+ << " size max=" << smax << " tot=" << stot << endl;
+
+ ndb = new Ndb("TEST_DB");
+ if (ndb->init() != 0)
+ return ndberror("init");
+ if (ndb->waitUntilReady(30) < 0)
+ return ndberror("waitUntilReady");
+
+ if ((tcon = ndb->startSchemaTransaction()) == 0)
+ return ndberror("startSchemaTransaction");
+ if ((top = tcon->getNdbSchemaOp()) == 0)
+ return ndberror("getNdbSchemaOp");
+ if (top->createTable(tab) < 0)
+ return ndberror("createTable");
+ for (int i = 0; i < attrcnt; i++) {
+ col& c = ccol[i];
+ if (top->createAttribute(
+ c.aAttrName,
+ c.aTupleKey,
+ c.aAttrSize,
+ c.aArraySize,
+ c.aAttrType,
+ MMBased,
+ c.nullable
+ ) < 0)
+ return ndberror("createAttribute col=%d", i);
+ }
+ if (tcon->execute() < 0) {
+ if (! (tcon->getNdbError().code == 721 && existok))
+ return ndberror("execute");
+ ndbout << "using " << tab << endl;
+ } else {
+ ndbout << "created " << tab << endl;
+ }
+ top = 0;
+ tcon = 0;
+
+ if (xverbose)
+ ndbout << "- delete" << endl;
+ int delcnt = 0;
+ for (key = 0; key < opercnt; key++) {
+ if ((con = ndb->startTransaction()) == 0)
+ return ndberror("startTransaction key=%d", key);
+ if ((op = con->getNdbOperation(tab)) == 0)
+ return ndberror("getNdbOperation key=%d", key);
+ if (op->deleteTuple() < 0)
+ return ndberror("deleteTuple key=%d", key);
+ for (int i = 0; i < attrcnt; i++) {
+ col& c = ccol[i];
+ if (i == 0) {
+ if (op->equal(c.aAttrName, (char*)&key, sizeof(key)) < 0)
+ return ndberror("equal key=%d", key);
+ } else {
+ }
+ }
+ if (con->execute(Commit) < 0) {
+ if (con->getNdbError().code != 626)
+ return ndberror("execute key=%d", key);
+ } else {
+ delcnt++;
+ }
+ ndb->closeTransaction(con);
+ }
+ con = 0;
+ op = 0;
+ ndbout << "deleted " << delcnt << endl;
+
+ if (xverbose)
+ ndbout << "- insert" << endl;
+ for (key = 0; key < opercnt; key++) {
+ int off = makeOff(key);
+ if ((con = ndb->startTransaction()) == 0)
+ return ndberror("startTransaction key=%d", key);
+ if ((op = con->getNdbOperation(tab)) == 0)
+ return ndberror("getNdbOperation key=%d", key);
+ if (op->insertTuple() < 0)
+ return ndberror("insertTuple key=%d", key);
+ for (int i = 0; i < attrcnt; i++) {
+ col& c = ccol[i];
+ if (i == 0) {
+ if (op->equal(c.aAttrName, (char*)&key, sizeof(key)) < 0)
+ return ndberror("equal key=%d", key);
+ } else {
+ memset(c.buf, 'A', c.bufsiz);
+ for (int j = 0; j < c.aArraySize; j++)
+ c.buf[j + off] = byteVal(key, i, j);
+ if (op->setValue(c.aAttrName, c.buf + off, c.aArraySize) < 0)
+ return ndberror("setValue key=%d col=%d", key, i);
+ }
+ }
+ if (con->execute(Commit) < 0)
+ return ndberror("execute key=%d", key);
+ ndb->closeTransaction(con);
+ }
+ con = 0;
+ op = 0;
+ ndbout << "inserted " << key << endl;
+
+ if (xverbose)
+ ndbout << "- select" << endl;
+ for (key = 0; key < opercnt; key++) {
+ int off = makeOff(key);
+ if (xverbose)
+ ndbout << "-- key " << key << " off=" << off << endl;
+ if ((con = ndb->startTransaction()) == 0)
+ return ndberror("startTransaction key=%d", key);
+ if ((op = con->getNdbOperation(tab)) == 0)
+ return ndberror("getNdbOperation key=%d", key);
+ if (op->readTuple() < 0)
+ return ndberror("readTuple key=%d", key);
+ for (int i = 0; i < attrcnt; i++) {
+ col& c = ccol[i];
+ if (i == 0) {
+ if (op->equal(c.aAttrName, (char*)&key, sizeof(key)) < 0)
+ return ndberror("equal key=%d", key);
+ } else {
+ if (xverbose) {
+ char tmp[20];
+ if (useBuf)
+ sprintf(tmp, "0x%x", int(c.buf + off));
+ else
+ strcpy(tmp, "ndbapi");
+ ndbout << "--- column " << i << " addr=" << tmp << endl;
+ }
+ memset(c.buf, 'B', c.bufsiz);
+ if (useBuf) {
+ if (op->getValue(c.aAttrName, c.buf + off) < 0)
+ return ndberror("getValue key=%d col=%d", key, i);
+ } else {
+ if ((c.aRa = op->getValue(c.aAttrName)) == 0)
+ return ndberror("getValue key=%d col=%d", key, i);
+ }
+ }
+ }
+ if (con->execute(Commit) != 0)
+ return ndberror("execute key=%d", key);
+ for (int i = 0; i < attrcnt; i++) {
+ col& c = ccol[i];
+ if (i == 0) {
+ } else if (useBuf) {
+ for (int j = 0; j < off; j++) {
+ if (c.buf[j] != 'B') {
+ return chkerror("mismatch before key=%d col=%d pos=%d ok=%02x bad=%02x",
+ key, i, j, 'B', c.buf[j]);
+ }
+ }
+ for (int j = 0; j < c.aArraySize; j++) {
+ if (c.buf[j + off] != byteVal(key, i, j)) {
+ return chkerror("mismatch key=%d col=%d pos=%d ok=%02x bad=%02x",
+ key, i, j, byteVal(key, i, j), c.buf[j]);
+ }
+ }
+ for (int j = c.aArraySize + off; j < c.bufsiz; j++) {
+ if (c.buf[j] != 'B') {
+ return chkerror("mismatch after key=%d col=%d pos=%d ok=%02x bad=%02x",
+ key, i, j, 'B', c.buf[j]);
+ }
+ }
+ } else {
+ char* buf = c.aRa->aRef();
+ if (buf == 0)
+ return ndberror("null aRef key=%d col%d", key, i);
+ for (int j = 0; j < c.aArraySize; j++) {
+ if (buf[j] != byteVal(key, i, j)) {
+ return chkerror("mismatch key=%d col=%d pos=%d ok=%02x bad=%02x",
+ key, i, j, byteVal(key, i, j), buf[j]);
+ }
+ }
+ }
+ }
+ ndb->closeTransaction(con);
+ }
+ con = 0;
+ op = 0;
+ ndbout << "selected " << key << endl;
+
+ if (xverbose)
+ ndbout << "- scan" << endl;
+ char found[MaxOper];
+ for (int k = 0; k < opercnt; k++)
+ found[k] = 0;
+ for (key = 0; key < opercnt; key++) {
+ int off = makeOff(key);
+ if (xverbose)
+ ndbout << "-- key " << key << " off=" << off << endl;
+ int newkey = 0;
+ if ((con = ndb->startTransaction()) == 0)
+ return ndberror("startTransaction key=%d", key);
+ if ((op = con->getNdbOperation(tab)) == 0)
+ return ndberror("getNdbOperation key=%d", key);
+ if (op->openScanRead(1) < 0)
+ return ndberror("openScanRead key=%d", key);
+ {
+ col& c = ccol[0];
+ if (op->load_const_u32(1, key) < 0)
+ return ndberror("load_const_u32");
+ if (op->read_attr(c.aAttrName, 2) < 0)
+ return ndberror("read_attr");
+ if (op->branch_eq(1, 2, 0) < 0)
+ return ndberror("branch_eq");
+ if (op->interpret_exit_nok() < 0)
+ return ndberror("interpret_exit_nok");
+ if (op->def_label(0) < 0)
+ return ndberror("def_label");
+ if (op->interpret_exit_ok() < 0)
+ return ndberror("interpret_exit_ok");
+ }
+ for (int i = 0; i < attrcnt; i++) {
+ col& c = ccol[i];
+ if (i == 0) {
+ if (op->getValue(c.aAttrName, (char*)&newkey) < 0)
+ return ndberror("getValue key=%d col=%d", key, i);
+ } else {
+ if (xverbose) {
+ char tmp[20];
+ if (useBuf)
+ sprintf(tmp, "0x%x", int(c.buf + off));
+ else
+ strcpy(tmp, "ndbapi");
+ ndbout << "--- column " << i << " addr=" << tmp << endl;
+ }
+ memset(c.buf, 'C', c.bufsiz);
+ if (useBuf) {
+ if (op->getValue(c.aAttrName, c.buf + off) < 0)
+ return ndberror("getValue key=%d col=%d", key, i);
+ } else {
+ if ((c.aRa = op->getValue(c.aAttrName)) == 0)
+ return ndberror("getValue key=%d col=%d", key, i);
+ }
+ }
+ }
+ if (con->executeScan() < 0)
+ return ndberror("executeScan key=%d", key);
+ int ret, cnt = 0;
+ while ((ret = con->nextScanResult()) == 0) {
+ if (key != newkey)
+ return ndberror("unexpected key=%d newkey=%d", key, newkey);
+ for (int i = 1; i < attrcnt; i++) {
+ col& c = ccol[i];
+ if (useBuf) {
+ for (int j = 0; j < off; j++) {
+ if (c.buf[j] != 'C') {
+ return chkerror("mismatch before key=%d col=%d pos=%d ok=%02x bad=%02x",
+ key, i, j, 'C', c.buf[j]);
+ }
+ }
+ for (int j = 0; j < c.aArraySize; j++) {
+ if (c.buf[j + off] != byteVal(key, i, j)) {
+ return chkerror("mismatch key=%d col=%d pos=%d ok=%02x bad=%02x",
+ key, i, j, byteVal(key, i, j), c.buf[j]);
+ }
+ }
+ for (int j = c.aArraySize + off; j < c.bufsiz; j++) {
+ if (c.buf[j] != 'C') {
+ return chkerror("mismatch after key=%d col=%d pos=%d ok=%02x bad=%02x",
+ key, i, j, 'C', c.buf[j]);
+ }
+ }
+ } else {
+ char* buf = c.aRa->aRef();
+ if (buf == 0)
+ return ndberror("null aRef key=%d col%d", key, i);
+ for (int j = 0; j < c.aArraySize; j++) {
+ if (buf[j] != byteVal(key, i, j)) {
+ return chkerror("mismatch key=%d col=%d pos=%d ok=%02x bad=%02x",
+ key, i, j, byteVal(key, i, j), buf[j]);
+ }
+ }
+ }
+ }
+ cnt++;
+ }
+ if (ret < 0)
+ return ndberror("nextScanResult key=%d", key);
+ if (cnt != 1)
+ return ndberror("scan key=%d found %d", key, cnt);
+ found[key] = 1;
+ ndb->closeTransaction(con);
+ }
+ con = 0;
+ op = 0;
+ for (int k = 0; k < opercnt; k++)
+ if (! found[k])
+ return ndberror("key %d not found", k);
+ ndbout << "scanned " << key << endl;
+
+ ndb = 0;
+ ndbout << "done" << endl;
+ return 0;
+}
+
+NDB_COMMAND(testDataBuffers, "testDataBuffers", "testDataBuffers", "testDataBuffers", 65535)
+{
+ while (++argv, --argc > 0) {
+ char const* p = argv[0];
+ if (*p++ != '-' || strlen(p) != 1)
+ goto wrongargs;
+ switch (*p) {
+ case 'a':
+ if (++argv, --argc > 0) {
+ attrcnt = atoi(argv[0]);
+ if (1 <= attrcnt && attrcnt <= MaxAttr)
+ break;
+ }
+ goto wrongargs;
+ case 'e':
+ existok = 1;
+ break;
+ case 'k':
+ kontinue = true;
+ break;
+ case 'l':
+ if (++argv, --argc > 0) {
+ loopcnt = atoi(argv[0]);
+ if (0 <= loopcnt)
+ break;
+ }
+ goto wrongargs;
+ case 'o':
+ if (++argv, --argc > 0) {
+ opercnt = atoi(argv[0]);
+ if (0 <= opercnt && opercnt <= MaxOper)
+ break;
+ }
+ goto wrongargs;
+ case 'r':
+ if (++argv, --argc > 0) {
+ randomizer = atoi(argv[0]);
+ if (1 <= randomizer)
+ break;
+ }
+ goto wrongargs;
+ case 's':
+ if (++argv, --argc > 0) {
+ sizelim = atoi(argv[0]);
+ if (1 <= sizelim && sizelim <= MaxSize)
+ break;
+ }
+ goto wrongargs;
+ case 'x':
+ xverbose = 1;
+ break;
+ default:
+ wrongargs:
+ printusage();
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ }
+ unsigned ok = true;
+ for (int i = 1; 0 == loopcnt || i <= loopcnt; i++) {
+ ndbout << "=== loop " << i << " ===" << endl;
+ for (int flag = 0; flag < (1<<testbits); flag++) {
+ if (testcase(flag) < 0) {
+ ok = false;
+ if (! kontinue)
+ goto out;
+ }
+ }
+ }
+out:
+ return NDBT_ProgramExit(ok ? NDBT_OK : NDBT_FAILED);
+}
+
+// vim: set sw=4:
diff --git a/ndb/test/ndbapi/testDict/Makefile b/ndb/test/ndbapi/testDict/Makefile
new file mode 100644
index 00000000000..75d493c3424
--- /dev/null
+++ b/ndb/test/ndbapi/testDict/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testDict
+
+SOURCES = testDict.cpp
+
+CFLAGS_testDict.cpp := -I$(call fixpath,$(NDB_TOP)/include/kernel)
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testDict/testDict.cpp b/ndb/test/ndbapi/testDict/testDict.cpp
new file mode 100644
index 00000000000..40c508efddd
--- /dev/null
+++ b/ndb/test/ndbapi/testDict/testDict.cpp
@@ -0,0 +1,1578 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <Vector.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+#include <../../include/kernel/ndb_limits.h>
+#include <random.h>
+#include <NdbAutoPtr.hpp>
+
+#define CHECK(b) if (!(b)) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ break; }
+
+#define CHECK2(b, c) if (!(b)) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << ": " << c << endl; \
+ result = NDBT_FAILED; \
+ goto end; }
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(pNdb, records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+int runCreateInvalidTables(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+
+ char failTabName[256];
+
+ for (int i = 0; i < 10; i++){
+ snprintf(failTabName, 256, "F%d", i);
+
+ const NdbDictionary::Table* pFailTab = NDBT_Tables::getTable(failTabName);
+ if (pFailTab != NULL){
+ ndbout << "|- " << failTabName << endl;
+
+ // Try to create table in db
+ if (pFailTab->createTableInDb(pNdb) == 0){
+ ndbout << failTabName << " created, this was not expected"<< endl;
+ result = NDBT_FAILED;
+ }
+
+ // Verify that table is not in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, failTabName) ;
+ if (pTab2 != NULL){
+ ndbout << failTabName << " was found in DB, this was not expected"<< endl;
+ result = NDBT_FAILED;
+ if (pFailTab->equal(*pTab2) == true){
+ ndbout << "It was equal" << endl;
+ } else {
+ ndbout << "It was not equal" << endl;
+ }
+ int records = 1000;
+ HugoTransactions hugoTrans(*pTab2);
+ if (hugoTrans.loadTable(pNdb, records) != 0){
+ ndbout << "It can NOT be loaded" << endl;
+ } else{
+ ndbout << "It can be loaded" << endl;
+
+ UtilTransactions utilTrans(*pTab2);
+ if (utilTrans.clearTable(pNdb, records, 64) != 0){
+ ndbout << "It can NOT be cleared" << endl;
+ } else{
+ ndbout << "It can be cleared" << endl;
+ }
+ }
+
+ if (pNdb->getDictionary()->dropTable(pTab2->getName()) == -1){
+ ndbout << "It can NOT be dropped" << endl;
+ } else {
+ ndbout << "It can be dropped" << endl;
+ }
+ }
+ }
+ }
+ return result;
+}
+
+int runCreateTheTable(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ // Try to create table in db
+ if (pTab->createTableInDb(pNdb) != 0){
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+ ctx->setTab(pTab2);
+
+ return NDBT_OK;
+}
+
+int runCreateTableWhenDbIsFull(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ const char* tabName = "TRANSACTION"; //Use a util table
+
+ const NdbDictionary::Table* pTab = NDBT_Tables::getTable(tabName);
+ if (pTab != NULL){
+ ndbout << "|- " << tabName << endl;
+
+ // Verify that table is not in db
+ if (NDBT_Table::discoverTableFromDb(pNdb, tabName) != NULL){
+ ndbout << tabName << " was found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+
+ // Try to create table in db
+ if (pTab->createTableInDb(pNdb) == 0){
+ result = NDBT_FAILED;
+ }
+
+ // Verify that table is in db
+ if (NDBT_Table::discoverTableFromDb(pNdb, tabName) != NULL){
+ ndbout << tabName << " was found in DB"<< endl;
+ result = NDBT_FAILED;
+ }
+ }
+
+ return result;
+}
+
+int runDropTableWhenDbIsFull(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ const char* tabName = "TRANSACTION"; //Use a util table
+
+ const NdbDictionary::Table* pTab = NDBT_Table::discoverTableFromDb(pNdb, tabName);
+ if (pTab != NULL){
+ ndbout << "|- TRANSACTION" << endl;
+
+ // Try to drop table in db
+ if (pNdb->getDictionary()->dropTable(pTab->getName()) == -1){
+ result = NDBT_FAILED;
+ }
+
+ // Verify that table is not in db
+ if (NDBT_Table::discoverTableFromDb(pNdb, tabName) != NULL){
+ ndbout << tabName << " was found in DB"<< endl;
+ result = NDBT_FAILED;
+ }
+ }
+
+ return result;
+
+}
+
+
+int runCreateAndDrop(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int loops = ctx->getNumLoops();
+ int i = 0;
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ ndbout << "|- " << pTab->getName() << endl;
+
+ while (i < loops){
+
+ ndbout << i << ": ";
+ // Try to create table in db
+ if (pTab->createTableInDb(pNdb) != 0){
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+
+ if (pNdb->getDictionary()->dropTable(pTab2->getName())){
+ ndbout << "Failed to drop "<<pTab2->getName()<<" in db" << endl;
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is not in db
+ const NdbDictionary::Table* pTab3 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab3 != NULL){
+ ndbout << pTab3->getName() << " was found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+
+ return NDBT_OK;
+}
+
+int runCreateAndDropWithData(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int i = 0;
+
+ NdbRestarter restarter;
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ if(restarter.dumpStateAllNodes(&val, 1) != 0){
+ int result;
+ do { CHECK(0); } while (0);
+ g_err << "Unable to change timebetween LCP" << endl;
+ return NDBT_FAILED;
+ }
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ ndbout << "|- " << pTab->getName() << endl;
+
+ while (i < loops){
+ ndbout << i << ": ";
+ // Try to create table in db
+ if (pTab->createTableInDb(pNdb) != 0){
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+
+ HugoTransactions hugoTrans(*pTab2);
+ if (hugoTrans.loadTable(pNdb, records) != 0){
+ return NDBT_FAILED;
+ }
+
+ int count = 0;
+ UtilTransactions utilTrans(*pTab2);
+ if (utilTrans.selectCount(pNdb, 64, &count) != 0){
+ return NDBT_FAILED;
+ }
+ if (count != records){
+ ndbout << count <<" != "<<records << endl;
+ return NDBT_FAILED;
+ }
+
+ if (pNdb->getDictionary()->dropTable(pTab2->getName()) != 0){
+ ndbout << "Failed to drop "<<pTab2->getName()<<" in db" << endl;
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is not in db
+ const NdbDictionary::Table* pTab3 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab3 != NULL){
+ ndbout << pTab3->getName() << " was found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+
+
+ i++;
+ }
+
+ return NDBT_OK;
+}
+
+int runFillTable(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.fillTable(pNdb) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable(pNdb, records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runCreateAndDropDuring(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ int i = 0;
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ ndbout << "|- " << pTab->getName() << endl;
+
+ while (i < loops && result == NDBT_OK){
+ ndbout << i << ": " << endl;
+ // Try to create table in db
+
+ Ndb* pNdb = GETNDB(step);
+ g_debug << "Creating table" << endl;
+
+ if (pTab->createTableInDb(pNdb) != 0){
+ g_err << "createTableInDb failed" << endl;
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ g_debug << "Verifying creation of table" << endl;
+
+ // Verify that table is in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ g_err << pTab->getName() << " was not found in DB"<< endl;
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ NdbSleep_MilliSleep(3000);
+
+ g_debug << "Dropping table" << endl;
+
+
+ if (pNdb->getDictionary()->dropTable(pTab2->getName()) != 0){
+ g_err << "Failed to drop "<<pTab2->getName()<<" in db" << endl;
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ g_debug << "Verifying dropping of table" << endl;
+
+ // Verify that table is not in db
+ const NdbDictionary::Table* pTab3 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab3 != NULL){
+ g_err << pTab3->getName() << " was found in DB"<< endl;
+ result = NDBT_FAILED;
+ continue;
+ }
+ i++;
+ }
+ ctx->stopTest();
+
+ return result;
+}
+
+
+int runUseTableUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ while (ctx->isTestStopped() == false) {
+ // g_info << i++ << ": ";
+
+
+ // Delete and recreate Ndb object
+ // Otherwise you always get Invalid Schema Version
+ // It would be a nice feature to remove this two lines
+ //step->tearDown();
+ //step->setUp();
+
+ Ndb* pNdb = GETNDB(step);
+
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL)
+ continue;
+
+ int res;
+ HugoTransactions hugoTrans(*pTab2);
+ if ((res = hugoTrans.loadTable(pNdb, records)) != 0){
+ NdbError err = pNdb->getNdbError(res);
+ if(err.classification == NdbError::SchemaError){
+ pNdb->getDictionary()->invalidateTable(pTab->getName());
+ }
+ continue;
+ }
+
+ UtilTransactions utilTrans(*pTab2);
+ if ((res = utilTrans.clearTable(pNdb, records)) != 0){
+ NdbError err = pNdb->getNdbError(res);
+ if(err.classification == NdbError::SchemaError){
+ pNdb->getDictionary()->invalidateTable(pTab->getName());
+ }
+ continue;
+ }
+ }
+ g_info << endl;
+ return NDBT_OK;
+}
+
+
+int runCreateMaxTables(NDBT_Context* ctx, NDBT_Step* step){
+ int failures = 0;
+ char tabName[256];
+ int numTables = ctx->getProperty("tables", 1000);
+ Ndb* pNdb = GETNDB(step);
+
+ for (int i = 0; i < numTables && failures < 5; i++){
+ snprintf(tabName, 256, "MAXTAB%d", i);
+
+ if (pNdb->waitUntilReady(30) != 0){
+ // Db is not ready, return with failure
+ return NDBT_FAILED;
+ }
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ ndbout << "|- " << tabName << endl;
+
+ // Set new name for T1
+ NdbDictionary::Table newTab(* pTab);
+ newTab.setName(tabName);
+
+ // Try to create table in db
+ if (newTab.createTableInDb(pNdb) != 0){
+ ndbout << tabName << " coult not be created"<< endl;
+ failures++;
+ continue;
+ }
+
+ // Verify that table exists in db
+ const NdbDictionary::Table* pTab3 =
+ NDBT_Table::discoverTableFromDb(pNdb, tabName) ;
+ if (pTab3 == NULL){
+ ndbout << tabName << " was not found in DB"<< endl;
+ failures++;
+ continue;
+ }
+
+ if (pTab->equal(*pTab3) == false){
+ ndbout << "It was not equal" << endl;
+ failures++;
+ }
+
+ int records = 1000;
+ HugoTransactions hugoTrans(*pTab3);
+ if (hugoTrans.loadTable(pNdb, records) != 0){
+ ndbout << "It can NOT be loaded" << endl;
+ } else{
+ ndbout << "It can be loaded" << endl;
+
+ UtilTransactions utilTrans(*pTab3);
+ if (utilTrans.clearTable(pNdb, records, 64) != 0){
+ ndbout << "It can NOT be cleared" << endl;
+ } else{
+ ndbout << "It can be cleared" << endl;
+ }
+ }
+
+ }
+ if (pNdb->waitUntilReady(30) != 0){
+ // Db is not ready, return with failure
+ return NDBT_FAILED;
+ }
+ // HURRAAA!
+ return NDBT_OK;
+}
+
+int runDropMaxTables(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ char tabName[256];
+ int numTables = ctx->getProperty("tables", 1000);
+ Ndb* pNdb = GETNDB(step);
+
+ for (int i = 0; i < numTables; i++){
+ snprintf(tabName, 256, "MAXTAB%d", i);
+
+ if (pNdb->waitUntilReady(30) != 0){
+ // Db is not ready, return with failure
+ return NDBT_FAILED;
+ }
+
+ // Verify that table exists in db
+ const NdbDictionary::Table* pTab3 =
+ NDBT_Table::discoverTableFromDb(pNdb, tabName) ;
+ if (pTab3 == NULL){
+ ndbout << tabName << " was not found in DB"<< endl;
+ continue;
+ }
+
+
+ // Try to drop table in db
+ if (pNdb->getDictionary()->dropTable(pTab3->getName()) != 0){
+ ndbout << tabName << " coult not be dropped"<< endl;
+ result = NDBT_FAILED;
+ }
+
+ }
+ return result;
+}
+
+int runTestFragmentTypes(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int fragTtype = ctx->getProperty("FragmentType");
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ NdbRestarter restarter;
+
+ // enum FragmentType {
+ // Unknown = 0,
+ // Single = 1, ///< Only one fragment
+ // All = 2, ///< Default value. One fragment per node group
+ // AllLarge = 3 ///< Sixten fragments per node group.
+ // };
+
+ if (pNdb->waitUntilReady(30) != 0){
+ // Db is not ready, return with failure
+ return NDBT_FAILED;
+ }
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ NdbDictionary::Table newTab(* pTab);
+ // Set fragment type for table
+ newTab.setFragmentType((NdbDictionary::Object::FragmentType)fragTtype);
+
+ // Try to create table in db
+ if (newTab.createTableInDb(pNdb) != 0){
+ ndbout << newTab.getName() << " could not be created"
+ << ", fragmentType = "<<fragTtype <<endl;
+ return NDBT_FAILED;
+ }
+
+ // Verify that table exists in db
+ const NdbDictionary::Table* pTab3 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName()) ;
+ if (pTab3 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+
+ }
+
+ if (pTab3->getFragmentType() != fragTtype){
+ ndbout << pTab->getName() << " fragmentType error "<< endl;
+ result = NDBT_FAILED;
+ goto drop_the_tab;
+ }
+
+ if (pTab->equal(*pTab3) == false){
+ ndbout << "It was not equal" << endl;
+ result = NDBT_FAILED;
+ goto drop_the_tab;
+ }
+
+ do {
+
+ HugoTransactions hugoTrans(*pTab3);
+ UtilTransactions utilTrans(*pTab3);
+ int count;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(hugoTrans.scanUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+
+ // restart all
+ ndbout << "Restarting cluster" << endl;
+ CHECK(restarter.restartAll() == 0);
+ int timeout = 120;
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ // Verify content
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records, 64) == 0);
+
+ } while(false);
+
+ drop_the_tab:
+
+ // Try to drop table in db
+ if (pNdb->getDictionary()->dropTable(pTab3->getName()) != 0){
+ ndbout << pTab3->getName() << " could not be dropped"<< endl;
+ result = NDBT_FAILED;
+ }
+
+ return result;
+}
+
+
+int runTestTemporaryTables(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ Ndb* pNdb = GETNDB(step);
+ int i = 0;
+ NdbRestarter restarter;
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ ndbout << "|- " << pTab->getName() << endl;
+
+ NdbDictionary::Table newTab(* pTab);
+ // Set table as temporary
+ newTab.setStoredTable(false);
+
+ // Try to create table in db
+ if (newTab.createTableInDb(pNdb) != 0){
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+
+ if (pTab2->getStoredTable() != false){
+ ndbout << pTab->getName() << " was not temporary in DB"<< endl;
+ result = NDBT_FAILED;
+ goto drop_the_tab;
+ }
+
+
+ while (i < loops && result == NDBT_OK){
+ ndbout << i << ": ";
+
+ HugoTransactions hugoTrans(*pTab2);
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+
+ int count = 0;
+ UtilTransactions utilTrans(*pTab2);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+
+ // restart all
+ ndbout << "Restarting cluster" << endl;
+ CHECK(restarter.restartAll() == 0);
+ int timeout = 120;
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+
+ i++;
+ }
+
+ drop_the_tab:
+
+
+ if (pNdb->getDictionary()->dropTable(pTab2->getName()) != 0){
+ ndbout << "Failed to drop "<<pTab2->getName()<<" in db" << endl;
+ result = NDBT_FAILED;
+ }
+
+ // Verify that table is not in db
+ const NdbDictionary::Table* pTab3 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab3 != NULL){
+ ndbout << pTab3->getName() << " was found in DB"<< endl;
+ result = NDBT_FAILED;
+ }
+
+ return result;
+}
+
+int runPkSizes(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ char tabName[256];
+ int minPkSize = 1;
+ ndbout << "minPkSize=" <<minPkSize<<endl;
+ int maxPkSize = MAX_KEY_SIZE_IN_WORDS * 4;
+ ndbout << "maxPkSize=" <<maxPkSize<<endl;
+ Ndb* pNdb = GETNDB(step);
+ int numRecords = ctx->getNumRecords();
+
+ for (int i = minPkSize; i < maxPkSize; i++){
+ snprintf(tabName, 256, "TPK_%d", i);
+
+ int records = numRecords;
+ int max = ~0;
+ // Limit num records for small PKs
+ if (i == 1)
+ max = 99;
+ if (i == 2)
+ max = 999;
+ if (i == 3)
+ max = 9999;
+ if (records > max)
+ records = max;
+ ndbout << "records =" << records << endl;
+
+ if (pNdb->waitUntilReady(30) != 0){
+ // Db is not ready, return with failure
+ return NDBT_FAILED;
+ }
+
+ ndbout << "|- " << tabName << endl;
+
+ if (NDBT_Tables::createTable(pNdb, tabName) != 0){
+ ndbout << tabName << " could not be created"<< endl;
+ return NDBT_FAILED;
+ }
+
+ // Verify that table exists in db
+ const NdbDictionary::Table* pTab3 =
+ NDBT_Table::discoverTableFromDb(pNdb, tabName) ;
+ if (pTab3 == NULL){
+ g_err << tabName << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+
+ // ndbout << *pTab3 << endl;
+
+ if (pTab3->equal(*NDBT_Tables::getTable(tabName)) == false){
+ g_err << "It was not equal" << endl;
+ return NDBT_FAILED;
+ }
+
+ do {
+ // Do it all
+ HugoTransactions hugoTrans(*pTab3);
+ UtilTransactions utilTrans(*pTab3);
+ int count;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(hugoTrans.scanUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+
+#if 0
+ // Fill table
+ CHECK(hugoTrans.fillTable(pNdb) == 0);
+ CHECK(utilTrans.clearTable2(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+#endif
+ } while(false);
+
+ // Drop table
+ if (pNdb->getDictionary()->dropTable(pTab3->getName()) != 0){
+ ndbout << "Failed to drop "<<pTab3->getName()<<" in db" << endl;
+ return NDBT_FAILED;
+ }
+ }
+ return result;
+}
+
+int runStoreFrm(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+
+ for (int l = 0; l < loops && result == NDBT_OK ; l++){
+
+ Uint32 dataLen = (Uint32)myRandom48(MAX_FRM_DATA_SIZE);
+ // size_t dataLen = 10;
+ unsigned char data[MAX_FRM_DATA_SIZE];
+
+ char start = l + 248;
+ for(Uint32 i = 0; i < dataLen; i++){
+ data[i] = start;
+ start++;
+ }
+#if 0
+ ndbout << "dataLen="<<dataLen<<endl;
+ for (Uint32 i = 0; i < dataLen; i++){
+ unsigned char c = data[i];
+ ndbout << hex << c << ", ";
+ }
+ ndbout << endl;
+#endif
+
+ NdbDictionary::Table newTab(* pTab);
+ void* pData = &data;
+ newTab.setFrm(pData, dataLen);
+
+ // Try to create table in db
+ if (newTab.createTableInDb(pNdb) != 0){
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ // Verify that table is in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ g_err << pTab->getName() << " was not found in DB"<< endl;
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ const void* pData2 = pTab2->getFrmData();
+ Uint32 resultLen = pTab2->getFrmLength();
+ if (dataLen != resultLen){
+ g_err << "Length of data failure" << endl
+ << " expected = " << dataLen << endl
+ << " got = " << resultLen << endl;
+ result = NDBT_FAILED;
+ }
+
+ // Verfiy the frm data
+ if (memcmp(pData, pData2, resultLen) != 0){
+ g_err << "Wrong data recieved" << endl;
+ for (size_t i = 0; i < dataLen; i++){
+ unsigned char c = ((unsigned char*)pData2)[i];
+ g_err << hex << c << ", ";
+ }
+ g_err << endl;
+ result = NDBT_FAILED;
+ }
+
+ if (pNdb->getDictionary()->dropTable(pTab2->getName()) != 0){
+ g_err << "It can NOT be dropped" << endl;
+ result = NDBT_FAILED;
+ }
+ }
+
+ return result;
+}
+
+int runStoreFrmError(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+
+ for (int l = 0; l < loops && result == NDBT_OK ; l++){
+
+ const Uint32 dataLen = MAX_FRM_DATA_SIZE + 10;
+ unsigned char data[dataLen];
+
+ char start = l + 248;
+ for(Uint32 i = 0; i < dataLen; i++){
+ data[i] = start;
+ start++;
+ }
+#if 0
+ ndbout << "dataLen="<<dataLen<<endl;
+ for (Uint32 i = 0; i < dataLen; i++){
+ unsigned char c = data[i];
+ ndbout << hex << c << ", ";
+ }
+ ndbout << endl;
+#endif
+
+ NdbDictionary::Table newTab(* pTab);
+
+ void* pData = &data;
+ newTab.setFrm(pData, dataLen);
+
+ // Try to create table in db
+ if (newTab.createTableInDb(pNdb) == 0){
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 != NULL){
+ g_err << pTab->getName() << " was found in DB"<< endl;
+ result = NDBT_FAILED;
+ if (pNdb->getDictionary()->dropTable(pTab2->getName()) != 0){
+ g_err << "It can NOT be dropped" << endl;
+ result = NDBT_FAILED;
+ }
+
+ continue;
+ }
+
+ }
+
+ return result;
+}
+
+int verifyTablesAreEqual(const NdbDictionary::Table* pTab, const NdbDictionary::Table* pTab2){
+ // Verify that getPrimaryKey only returned true for primary keys
+ for (int i = 0; i < pTab2->getNoOfColumns(); i++){
+ const NdbDictionary::Column* col = pTab->getColumn(i);
+ const NdbDictionary::Column* col2 = pTab2->getColumn(i);
+ if (col->getPrimaryKey() != col2->getPrimaryKey()){
+ g_err << "col->getPrimaryKey() != col2->getPrimaryKey()" << endl;
+ return NDBT_FAILED;
+ }
+ }
+
+ if (!pTab->equal(*pTab2)){
+ g_err << "equal failed" << endl;
+ g_info << *pTab;
+ g_info << *pTab2;
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runGetPrimaryKey(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ ndbout << "|- " << pTab->getName() << endl;
+ g_info << *pTab;
+ // Try to create table in db
+ if (pTab->createTableInDb(pNdb) != 0){
+ return NDBT_FAILED;
+ }
+
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+
+ int result = NDBT_OK;
+ if (verifyTablesAreEqual(pTab, pTab2) != NDBT_OK)
+ result = NDBT_FAILED;
+
+
+#if 0
+ // Create an index on the table and see what
+ // the function returns now
+ char name[200];
+ sprintf(name, "%s_X007", pTab->getName());
+ NDBT_Index* pInd = new NDBT_Index(name);
+ pInd->setTable(pTab->getName());
+ pInd->setType(NdbDictionary::Index::UniqueHashIndex);
+ // pInd->setLogging(false);
+ for (int i = 0; i < 2; i++){
+ const NDBT_Attribute* pAttr = pTab->getAttribute(i);
+ pInd->addAttribute(*pAttr);
+ }
+ g_info << "Create index:" << endl << *pInd;
+ if (pInd->createIndexInDb(pNdb, false) != 0){
+ result = NDBT_FAILED;
+ }
+ delete pInd;
+
+ const NdbDictionary::Table* pTab3 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab3 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+
+ if (verifyTablesAreEqual(pTab, pTab3) != NDBT_OK)
+ result = NDBT_FAILED;
+ if (verifyTablesAreEqual(pTab2, pTab3) != NDBT_OK)
+ result = NDBT_FAILED;
+#endif
+
+#if 0
+ if (pTab2->getDictionary()->dropTable(pNdb) != 0){
+ ndbout << "Failed to drop "<<pTab2->getName()<<" in db" << endl;
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is not in db
+ const NdbDictionary::Table* pTab4 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab4 != NULL){
+ ndbout << pTab4->getName() << " was found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+#endif
+
+ return result;
+}
+
+int
+NF_codes[] = {
+ 14000
+ ,14001
+ //,14002
+};
+
+int
+runNF1(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+ if(restarter.getNumDbNodes() < 2)
+ return NDBT_OK;
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+ Ndb* pNdb = GETNDB(step);
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ NdbDictionary::Dictionary* dict = pNdb->getDictionary();
+ dict->dropTable(pTab->getName());
+
+ int result = NDBT_OK;
+
+ /**
+ * Need to run LCP at high rate otherwise
+ * packed replicas become "to many"
+ */
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ if(restarter.dumpStateAllNodes(&val, 1) != 0){
+ do { CHECK(0); } while(0);
+ g_err << "Failed to set LCP to min value" << endl;
+ return NDBT_FAILED;
+ }
+
+ const int loops = ctx->getNumLoops();
+ for (int l = 0; l < loops && result == NDBT_OK ; l++){
+ const int sz = sizeof(NF_codes)/sizeof(NF_codes[0]);
+ for(int i = 0; i<sz; i++){
+ int rand = myRandom48(restarter.getNumDbNodes());
+ int nodeId = restarter.getRandomNotMasterNodeId(rand);
+ int error = NF_codes[i];
+
+ g_info << "NF1: node = " << nodeId << " error code = " << error << endl;
+
+ int val2[] = { DumpStateOrd::CmvmiSetRestartOnErrorInsert, 3};
+
+ CHECK2(restarter.dumpStateOneNode(nodeId, val2, 2) == 0,
+ "failed to set RestartOnErrorInsert");
+
+ CHECK2(restarter.insertErrorInNode(nodeId, error) == 0,
+ "failed to set error insert");
+
+ CHECK2(dict->createTable(* pTab) == 0,
+ "failed to create table");
+
+ CHECK2(restarter.waitNodesNoStart(&nodeId, 1) == 0,
+ "waitNodesNoStart failed");
+
+ if(myRandom48(100) > 50){
+ CHECK2(restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ CHECK2(restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+
+ CHECK2(dict->dropTable(pTab->getName()) == 0,
+ "drop table failed");
+ } else {
+ CHECK2(dict->dropTable(pTab->getName()) == 0,
+ "drop table failed");
+
+ CHECK2(restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ CHECK2(restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+ }
+
+ CHECK2(restarter.dumpStateOneNode(nodeId, &val, 1) == 0,
+ "Failed to set LCP to min value");
+ }
+ }
+ end:
+ dict->dropTable(pTab->getName());
+
+ return result;
+}
+
+#define APIERROR(error) \
+ { g_err << "Error in " << __FILE__ << ", line:" << __LINE__ << ", code:" \
+ << error.code << ", msg: " << error.message << "." << endl; \
+ }
+
+int
+runCreateAutoincrementTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ Uint32 startvalues[5] = {256-2, 0, 256*256-2, ~0, 256*256*256-2};
+
+ int ret = NDBT_OK;
+
+ for (int jj = 0; jj < 5 && ret == NDBT_OK; jj++) {
+ char tabname[] = "AUTOINCTAB";
+ Uint32 startvalue = startvalues[jj];
+
+ NdbDictionary::Table myTable;
+ NdbDictionary::Column myColumn;
+
+ Ndb* myNdb = GETNDB(step);
+ NdbDictionary::Dictionary* myDict = myNdb->getDictionary();
+
+
+ if (myDict->getTable(tabname) != NULL) {
+ g_err << "NDB already has example table: " << tabname << endl;
+ APIERROR(myNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ myTable.setName(tabname);
+
+ myColumn.setName("ATTR1");
+ myColumn.setPrimaryKey(true);
+ myColumn.setType(NdbDictionary::Column::Unsigned);
+ myColumn.setLength(1);
+ myColumn.setNullable(false);
+ myColumn.setAutoIncrement(true);
+ if (startvalue != ~0) // check that default value starts with 1
+ myColumn.setAutoIncrementInitialValue(startvalue);
+ myTable.addColumn(myColumn);
+
+ if (myDict->createTable(myTable) == -1) {
+ g_err << "Failed to create table " << tabname << endl;
+ APIERROR(myNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+
+
+ if (startvalue == ~0) // check that default value starts with 1
+ startvalue = 1;
+
+ for (int i = 0; i < 16; i++) {
+
+ Uint64 value = myNdb->getAutoIncrementValue(tabname, 1);
+
+ if (value != (startvalue+i)) {
+ g_err << "value = " << value << " expected " << startvalue+i << endl;;
+ APIERROR(myNdb->getNdbError());
+ // ret = NDBT_FAILED;
+ // break;
+ }
+ }
+
+ if (myDict->dropTable(tabname) == -1) {
+ g_err << "Failed to drop table " << tabname << endl;
+ APIERROR(myNdb->getNdbError());
+ ret = NDBT_FAILED;
+ }
+ }
+
+ return ret;
+}
+
+int
+runTableRename(NDBT_Context* ctx, NDBT_Step* step){
+
+ int result = NDBT_OK;
+
+ Ndb* pNdb = GETNDB(step);
+ NdbDictionary::Dictionary* dict = pNdb->getDictionary();
+ int records = ctx->getNumRecords();
+ const int loops = ctx->getNumLoops();
+
+ ndbout << "|- " << ctx->getTab()->getName() << endl;
+
+ for (int l = 0; l < loops && result == NDBT_OK ; l++){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ // Try to create table in db
+ if (pTab->createTableInDb(pNdb) != 0){
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+ ctx->setTab(pTab2);
+
+ // Load table
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(pNdb, records) != 0){
+ return NDBT_FAILED;
+ }
+
+ // Rename table
+ BaseString pTabName(pTab->getName());
+ BaseString pTabNewName(pTabName);
+ pTabNewName.append("xx");
+
+ const NdbDictionary::Table * oldTable = dict->getTable(pTabName.c_str());
+ if (oldTable) {
+ NdbDictionary::Table newTable = dict->getTableForAlteration(pTabName.c_str());
+ newTable.setName(pTabNewName.c_str());
+ CHECK2(dict->alterTable(newTable) == 0,
+ "TableRename failed");
+ }
+ else {
+ result = NDBT_FAILED;
+ }
+
+ // Verify table contents
+ NdbDictionary::Table pNewTab(pTabNewName.c_str());
+
+ UtilTransactions utilTrans(pNewTab);
+ if (utilTrans.clearTable(pNdb, records) != 0){
+ continue;
+ }
+
+ // Drop table
+ dict->dropTable(pNewTab.getName());
+ }
+ end:
+
+ return result;
+}
+
+int
+runTableRenameNF(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+ if(restarter.getNumDbNodes() < 2)
+ return NDBT_OK;
+
+ int result = NDBT_OK;
+
+ Ndb* pNdb = GETNDB(step);
+ NdbDictionary::Dictionary* dict = pNdb->getDictionary();
+ int records = ctx->getNumRecords();
+ const int loops = ctx->getNumLoops();
+
+ ndbout << "|- " << ctx->getTab()->getName() << endl;
+
+ for (int l = 0; l < loops && result == NDBT_OK ; l++){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ // Try to create table in db
+ if (pTab->createTableInDb(pNdb) != 0){
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+ ctx->setTab(pTab2);
+
+ // Load table
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(pNdb, records) != 0){
+ return NDBT_FAILED;
+ }
+
+ BaseString pTabName(pTab->getName());
+ BaseString pTabNewName(pTabName);
+ pTabNewName.append("xx");
+
+ const NdbDictionary::Table * oldTable = dict->getTable(pTabName.c_str());
+ if (oldTable) {
+ NdbDictionary::Table newTable = dict->getTableForAlteration(pTabName.c_str());
+ newTable.setName(pTabNewName.c_str());
+ CHECK2(dict->alterTable(newTable) == 0,
+ "TableRename failed");
+ }
+ else {
+ result = NDBT_FAILED;
+ }
+
+ // Restart one node at a time
+
+ /**
+ * Need to run LCP at high rate otherwise
+ * packed replicas become "to many"
+ */
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ if(restarter.dumpStateAllNodes(&val, 1) != 0){
+ do { CHECK(0); } while(0);
+ g_err << "Failed to set LCP to min value" << endl;
+ return NDBT_FAILED;
+ }
+
+ const int numNodes = restarter.getNumDbNodes();
+ for(int i = 0; i<numNodes; i++){
+ int nodeId = restarter.getDbNodeId(i);
+ int error = NF_codes[i];
+
+ g_info << "NF1: node = " << nodeId << " error code = " << error << endl;
+
+ CHECK2(restarter.restartOneDbNode(nodeId) == 0,
+ "failed to set restartOneDbNode");
+
+ CHECK2(restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+
+ }
+
+ // Verify table contents
+ NdbDictionary::Table pNewTab(pTabNewName.c_str());
+
+ UtilTransactions utilTrans(pNewTab);
+ if (utilTrans.clearTable(pNdb, records) != 0){
+ continue;
+ }
+
+ // Drop table
+ dict->dropTable(pTabNewName.c_str());
+ }
+ end:
+ return result;
+}
+
+int
+runTableRenameSR(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+ if(restarter.getNumDbNodes() < 2)
+ return NDBT_OK;
+
+ int result = NDBT_OK;
+
+ Ndb* pNdb = GETNDB(step);
+ NdbDictionary::Dictionary* dict = pNdb->getDictionary();
+ int records = ctx->getNumRecords();
+ const int loops = ctx->getNumLoops();
+
+ ndbout << "|- " << ctx->getTab()->getName() << endl;
+
+ for (int l = 0; l < loops && result == NDBT_OK ; l++){
+ // Rename table
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ // Try to create table in db
+ if (pTab->createTableInDb(pNdb) != 0){
+ return NDBT_FAILED;
+ }
+
+ // Verify that table is in db
+ const NdbDictionary::Table* pTab2 =
+ NDBT_Table::discoverTableFromDb(pNdb, pTab->getName());
+ if (pTab2 == NULL){
+ ndbout << pTab->getName() << " was not found in DB"<< endl;
+ return NDBT_FAILED;
+ }
+ ctx->setTab(pTab2);
+
+ // Load table
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(pNdb, records) != 0){
+ return NDBT_FAILED;
+ }
+
+ BaseString pTabName(pTab->getName());
+ BaseString pTabNewName(pTabName);
+ pTabNewName.append("xx");
+
+ const NdbDictionary::Table * oldTable = dict->getTable(pTabName.c_str());
+ if (oldTable) {
+ NdbDictionary::Table newTable = dict->getTableForAlteration(pTabName.c_str());
+ newTable.setName(pTabNewName.c_str());
+ CHECK2(dict->alterTable(newTable) == 0,
+ "TableRename failed");
+ }
+ else {
+ result = NDBT_FAILED;
+ }
+
+ // Restart cluster
+
+ /**
+ * Need to run LCP at high rate otherwise
+ * packed replicas become "to many"
+ */
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ if(restarter.dumpStateAllNodes(&val, 1) != 0){
+ do { CHECK(0); } while(0);
+ g_err << "Failed to set LCP to min value" << endl;
+ return NDBT_FAILED;
+ }
+
+ CHECK2(restarter.restartAll() == 0,
+ "failed to set restartOneDbNode");
+
+ CHECK2(restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+
+ // Verify table contents
+ NdbDictionary::Table pNewTab(pTabNewName.c_str());
+
+ UtilTransactions utilTrans(pNewTab);
+ if (utilTrans.clearTable(pNdb, records) != 0){
+ continue;
+ }
+
+ // Drop table
+ dict->dropTable(pTabNewName.c_str());
+ }
+ end:
+ return result;
+}
+
+static void
+f(const NdbDictionary::Column * col){
+ if(col == 0){
+ abort();
+ }
+}
+
+int
+runTestDictionaryPerf(NDBT_Context* ctx, NDBT_Step* step){
+ Vector<char*> cols;
+ Vector<const NdbDictionary::Table*> tabs;
+
+ Ndb* pNdb = GETNDB(step);
+
+ const Uint32 count = NDBT_Tables::getNumTables();
+ for (int i=0; i < count; i++){
+ const NdbDictionary::Table * tab = NDBT_Tables::getTable(i);
+ pNdb->getDictionary()->createTable(* tab);
+
+ const NdbDictionary::Table * tab2 = pNdb->getDictionary()->getTable(tab->getName());
+
+ for(size_t j = 0; j<tab->getNoOfColumns(); j++){
+ cols.push_back((char*)tab2);
+ cols.push_back(strdup(tab->getColumn(j)->getName()));
+ }
+ }
+
+ const Uint32 times = 10000000;
+
+ ndbout_c("%d tables and %d columns",
+ NDBT_Tables::getNumTables(), cols.size()/2);
+
+ char ** tcols = cols.getBase();
+
+ srand(time(0));
+ Uint32 size = cols.size() / 2;
+ char ** columns = &cols[0];
+ Uint64 start = NdbTick_CurrentMillisecond();
+ for(int i = 0; i<times; i++){
+ int j = 2 * (rand() % size);
+ const NdbDictionary::Table* tab = (const NdbDictionary::Table*)tcols[j];
+ const char * col = tcols[j+1];
+ const NdbDictionary::Column* column = tab->getColumn(col);
+ f(column);
+ }
+ Uint64 stop = NdbTick_CurrentMillisecond();
+ stop -= start;
+
+ Uint64 per = stop;
+ per *= 1000;
+ per /= times;
+
+ ndbout_c("%d random getColumn(name) in %Ld ms -> %d us/get",
+ times, stop, per);
+
+ return NDBT_OK;
+}
+
+NDBT_TESTSUITE(testDict);
+TESTCASE("CreateAndDrop",
+ "Try to create and drop the table loop number of times\n"){
+ INITIALIZER(runCreateAndDrop);
+}
+TESTCASE("CreateAndDropWithData",
+ "Try to create and drop the table when it's filled with data\n"
+ "do this loop number of times\n"){
+ INITIALIZER(runCreateAndDropWithData);
+}
+TESTCASE("CreateAndDropDuring",
+ "Try to create and drop the table when other thread is using it\n"
+ "do this loop number of times\n"){
+ STEP(runCreateAndDropDuring);
+ STEP(runUseTableUntilStopped);
+}
+TESTCASE("CreateInvalidTables",
+ "Try to create the invalid tables we have defined\n"){
+ INITIALIZER(runCreateInvalidTables);
+}
+TESTCASE("CreateTableWhenDbIsFull",
+ "Try to create a new table when db already is full\n"){
+ INITIALIZER(runCreateTheTable);
+ INITIALIZER(runFillTable);
+ INITIALIZER(runCreateTableWhenDbIsFull);
+ INITIALIZER(runDropTableWhenDbIsFull);
+ FINALIZER(runClearTable);
+}
+TESTCASE("FragmentTypeSingle",
+ "Create the table with fragment type Single\n"){
+ TC_PROPERTY("FragmentType", 1);
+ INITIALIZER(runTestFragmentTypes);
+}
+TESTCASE("FragmentTypeAll",
+ "Create the table with fragment type All\n"){
+ TC_PROPERTY("FragmentType", 2);
+ INITIALIZER(runTestFragmentTypes);
+}
+TESTCASE("FragmentTypeAllLarge",
+ "Create the table with fragment type AllLarge\n"){
+ TC_PROPERTY("FragmentType", 3);
+ INITIALIZER(runTestFragmentTypes);
+}
+TESTCASE("TemporaryTables",
+ "Create the table as temporary and make sure it doesn't\n"
+ "contain any data when system is restarted\n"){
+ INITIALIZER(runTestTemporaryTables);
+}
+TESTCASE("CreateMaxTables",
+ "Create tables until db says that it can't create any more\n"){
+ TC_PROPERTY("tables", 1000);
+ INITIALIZER(runCreateMaxTables);
+ FINALIZER(runDropMaxTables);
+}
+TESTCASE("PkSizes",
+ "Create tables with all different primary key sizes.\n"\
+ "Test all data operations insert, update, delete etc.\n"\
+ "Drop table."){
+ INITIALIZER(runPkSizes);
+}
+TESTCASE("StoreFrm",
+ "Test that a frm file can be properly stored as part of the\n"
+ "data in Dict."){
+ INITIALIZER(runStoreFrm);
+}
+TESTCASE("GetPrimaryKey",
+ "Test the function NdbDictionary::Column::getPrimaryKey\n"
+ "It should return true only if the column is part of \n"
+ "the primary key in the table"){
+ INITIALIZER(runGetPrimaryKey);
+}
+TESTCASE("StoreFrmError",
+ "Test that a frm file with too long length can't be stored."){
+ INITIALIZER(runStoreFrmError);
+}
+TESTCASE("NF1",
+ "Test that create table can handle NF (not master)"){
+ INITIALIZER(runNF1);
+}
+TESTCASE("TableRename",
+ "Test basic table rename"){
+ INITIALIZER(runTableRename);
+}
+TESTCASE("TableRenameNF",
+ "Test that table rename can handle node failure"){
+ INITIALIZER(runTableRenameNF);
+}
+TESTCASE("TableRenameSR",
+ "Test that table rename can handle system restart"){
+ INITIALIZER(runTableRenameSR);
+}
+TESTCASE("DictionaryPerf",
+ ""){
+ INITIALIZER(runTestDictionaryPerf);
+}
+NDBT_TESTSUITE_END(testDict);
+
+int main(int argc, const char** argv){
+ // Tables should not be auto created
+ testDict.setCreateTable(false);
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ return testDict.execute(argc, argv);
+}
+
+
diff --git a/ndb/test/ndbapi/testGrep/Makefile b/ndb/test/ndbapi/testGrep/Makefile
new file mode 100644
index 00000000000..34fdd7113d0
--- /dev/null
+++ b/ndb/test/ndbapi/testGrep/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE = ndbapitest
+DIRS = verify
+BIN_TARGET = testGrep
+BIN_TARGET_LIBS += bank
+SOURCES = testGrep.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/ndbapi/testGrep/testGrep.cpp b/ndb/test/ndbapi/testGrep/testGrep.cpp
new file mode 100644
index 00000000000..4b870f6f9a9
--- /dev/null
+++ b/ndb/test/ndbapi/testGrep/testGrep.cpp
@@ -0,0 +1,540 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbGrep.hpp>
+
+
+#define CHECK(b) if (!(b)) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ continue; }
+
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runPkUpdate(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << "|- " << i << ": ";
+ if (hugoTrans.pkUpdateRecords(GETNDB(step), records, batchSize) != 0){
+ g_info << endl;
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ g_info << endl;
+ return NDBT_OK;
+}
+
+int runRestartInitial(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+
+ Ndb* pNdb = GETNDB(step);
+
+ const NdbDictionary::Table *tab = ctx->getTab();
+ pNdb->getDictionary()->dropTable(tab->getName());
+
+ if (restarter.restartAll(true) != 0)
+ return NDBT_FAILED;
+
+ return NDBT_OK;
+}
+
+int runRestarter(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ NdbRestarter restarter;
+ int i = 0;
+ int lastId = 0;
+
+ if (restarter.getNumDbNodes() < 2){
+ ctx->stopTest();
+ return NDBT_OK;
+ }
+
+ if(restarter.waitClusterStarted(60) != 0){
+ g_err << "Cluster failed to start" << endl;
+ return NDBT_FAILED;
+ }
+
+ loops *= restarter.getNumDbNodes();
+ while(i<loops && result != NDBT_FAILED && !ctx->isTestStopped()){
+
+ int id = lastId % restarter.getNumDbNodes();
+ int nodeId = restarter.getDbNodeId(id);
+ ndbout << "Restart node " << nodeId << endl;
+ if(restarter.restartOneDbNode(nodeId) != 0){
+ g_err << "Failed to restartNextDbNode" << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+
+ if(restarter.waitClusterStarted(60) != 0){
+ g_err << "Cluster failed to start" << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+
+ NdbSleep_SecSleep(1);
+
+ lastId++;
+ i++;
+ }
+
+ ctx->stopTest();
+
+ return result;
+}
+
+int runCheckAllNodesStarted(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+
+ if(restarter.waitClusterStarted(1) != 0){
+ g_err << "All nodes was not started " << endl;
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+
+bool testMaster = true;
+bool testSlave = false;
+
+int setMaster(NDBT_Context* ctx, NDBT_Step* step){
+ testMaster = true;
+ testSlave = false;
+ return NDBT_OK;
+}
+int setMasterAsSlave(NDBT_Context* ctx, NDBT_Step* step){
+ testMaster = true;
+ testSlave = true;
+ return NDBT_OK;
+}
+int setSlave(NDBT_Context* ctx, NDBT_Step* step){
+ testMaster = false;
+ testSlave = true;
+ return NDBT_OK;
+}
+
+int runAbort(NDBT_Context* ctx, NDBT_Step* step){
+
+
+ NdbGrep grep(GETNDB(step)->getNodeId()+1);
+ NdbRestarter restarter;
+
+ if (restarter.getNumDbNodes() < 2){
+ ctx->stopTest();
+ return NDBT_OK;
+ }
+
+ if(restarter.waitClusterStarted(60) != 0){
+ g_err << "Cluster failed to start" << endl;
+ return NDBT_FAILED;
+ }
+
+ if (testMaster) {
+ if (testSlave) {
+ if (grep.NFMasterAsSlave(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ } else {
+ if (grep.NFMaster(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ }
+ } else {
+ if (grep.NFSlave(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ }
+
+ return NDBT_OK;
+}
+
+int runFail(NDBT_Context* ctx, NDBT_Step* step){
+ NdbGrep grep(GETNDB(step)->getNodeId()+1);
+
+ NdbRestarter restarter;
+
+ if (restarter.getNumDbNodes() < 2){
+ ctx->stopTest();
+ return NDBT_OK;
+ }
+
+ if(restarter.waitClusterStarted(60) != 0){
+ g_err << "Cluster failed to start" << endl;
+ return NDBT_FAILED;
+ }
+
+ if (testMaster) {
+ if (testSlave) {
+ if (grep.FailMasterAsSlave(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ } else {
+ if (grep.FailMaster(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ }
+ } else {
+ if (grep.FailSlave(restarter) == -1){
+ return NDBT_FAILED;
+ }
+ }
+
+ return NDBT_OK;
+}
+
+int runGrepBasic(NDBT_Context* ctx, NDBT_Step* step){
+ NdbGrep grep(GETNDB(step)->getNodeId()+1);
+ unsigned grepId = 0;
+
+ if (grep.start() == -1){
+ return NDBT_FAILED;
+ }
+ ndbout << "Started grep " << grepId << endl;
+ ctx->setProperty("GrepId", grepId);
+
+ return NDBT_OK;
+}
+
+
+
+
+int runVerifyBasic(NDBT_Context* ctx, NDBT_Step* step){
+ NdbGrep grep(GETNDB(step)->getNodeId()+1, ctx->getRemoteMgm());
+ ndbout_c("no of nodes %d" ,grep.getNumDbNodes());
+ int result;
+ if ((result = grep.verify(ctx)) == -1){
+ return NDBT_FAILED;
+ }
+ return result;
+}
+
+
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+#include "../bank/Bank.hpp"
+
+int runCreateBank(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int overWriteExisting = true;
+ if (bank.createAndLoadBank(overWriteExisting) != NDBT_OK)
+ return NDBT_FAILED;
+ return NDBT_OK;
+}
+
+int runBankTimer(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int wait = 30; // Max seconds between each "day"
+ int yield = 1; // Loops before bank returns
+
+ while (ctx->isTestStopped() == false) {
+ bank.performIncreaseTime(wait, yield);
+ }
+ return NDBT_OK;
+}
+
+int runBankTransactions(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int wait = 10; // Max ms between each transaction
+ int yield = 100; // Loops before bank returns
+
+ while (ctx->isTestStopped() == false) {
+ bank.performTransactions(wait, yield);
+ }
+ return NDBT_OK;
+}
+
+int runBankGL(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int yield = 20; // Loops before bank returns
+ int result = NDBT_OK;
+
+ while (ctx->isTestStopped() == false) {
+ if (bank.performMakeGLs(yield) != NDBT_OK){
+ ndbout << "bank.performMakeGLs FAILED" << endl;
+ result = NDBT_FAILED;
+ }
+ }
+ return NDBT_OK;
+}
+
+int runBankSum(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ int wait = 2000; // Max ms between each sum of accounts
+ int yield = 1; // Loops before bank returns
+ int result = NDBT_OK;
+
+ while (ctx->isTestStopped() == false) {
+ if (bank.performSumAccounts(wait, yield) != NDBT_OK){
+ ndbout << "bank.performSumAccounts FAILED" << endl;
+ result = NDBT_FAILED;
+ }
+ }
+ return result ;
+}
+
+int runDropBank(NDBT_Context* ctx, NDBT_Step* step){
+ Bank bank;
+ if (bank.dropBank() != NDBT_OK)
+ return NDBT_FAILED;
+ return NDBT_OK;
+}
+
+int runGrepBank(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int l = 0;
+ int maxSleep = 30; // Max seconds between each grep
+ Ndb* pNdb = GETNDB(step);
+ NdbGrep grep(GETNDB(step)->getNodeId()+1);
+ unsigned minGrepId = ~0;
+ unsigned maxGrepId = 0;
+ unsigned grepId = 0;
+ int result = NDBT_OK;
+
+ while (l < loops && result != NDBT_FAILED){
+
+ if (pNdb->waitUntilReady() != 0){
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ // Sleep for a while
+ NdbSleep_SecSleep(maxSleep);
+
+ // Perform grep
+ if (grep.start() != 0){
+ ndbout << "grep.start failed" << endl;
+ result = NDBT_FAILED;
+ continue;
+ }
+ ndbout << "Started grep " << grepId << endl;
+
+ // Remember min and max grepid
+ if (grepId < minGrepId)
+ minGrepId = grepId;
+
+ if (grepId > maxGrepId)
+ maxGrepId = grepId;
+
+ ndbout << " maxGrepId = " << maxGrepId
+ << ", minGrepId = " << minGrepId << endl;
+ ctx->setProperty("MinGrepId", minGrepId);
+ ctx->setProperty("MaxGrepId", maxGrepId);
+
+ l++;
+ }
+
+ ctx->stopTest();
+
+ return result;
+}
+/*
+int runRestoreBankAndVerify(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+ NdbGrep grep(GETNDB(step)->getNodeId()+1);
+ unsigned minGrepId = ctx->getProperty("MinGrepId");
+ unsigned maxGrepId = ctx->getProperty("MaxGrepId");
+ unsigned grepId = minGrepId;
+ int result = NDBT_OK;
+ int errSumAccounts = 0;
+ int errValidateGL = 0;
+
+ ndbout << " maxGrepId = " << maxGrepId << endl;
+ ndbout << " minGrepId = " << minGrepId << endl;
+
+ while (grepId <= maxGrepId){
+
+ // TEMPORARY FIX
+ // To erase all tables from cache(s)
+ // To be removed, maybe replaced by ndb.invalidate();
+ {
+ Bank bank;
+
+ if (bank.dropBank() != NDBT_OK){
+ result = NDBT_FAILED;
+ break;
+ }
+ }
+ // END TEMPORARY FIX
+
+ ndbout << "Performing initial restart" << endl;
+ if (restarter.restartAll(true) != 0)
+ return NDBT_FAILED;
+
+ if (restarter.waitClusterStarted() != 0)
+ return NDBT_FAILED;
+
+ ndbout << "Restoring grep " << grepId << endl;
+ if (grep.restore(grepId) == -1){
+ return NDBT_FAILED;
+ }
+ ndbout << "Grep " << grepId << " restored" << endl;
+
+ // Let bank verify
+ Bank bank;
+
+ int wait = 0;
+ int yield = 1;
+ if (bank.performSumAccounts(wait, yield) != 0){
+ ndbout << "bank.performSumAccounts FAILED" << endl;
+ ndbout << " grepId = " << grepId << endl << endl;
+ result = NDBT_FAILED;
+ errSumAccounts++;
+ }
+
+ if (bank.performValidateAllGLs() != 0){
+ ndbout << "bank.performValidateAllGLs FAILED" << endl;
+ ndbout << " grepId = " << grepId << endl << endl;
+ result = NDBT_FAILED;
+ errValidateGL++;
+ }
+
+ grepId++;
+ }
+
+ if (result != NDBT_OK){
+ ndbout << "Verification of grep failed" << endl
+ << " errValidateGL="<<errValidateGL<<endl
+ << " errSumAccounts="<<errSumAccounts<<endl << endl;
+ }
+
+ return result;
+}
+*/
+NDBT_TESTSUITE(testGrep);
+TESTCASE("GrepBasic",
+ "Test that Global Replication works on one table \n"
+ "1. Load table\n"
+ "2. Grep\n"
+ "3. Restart -i\n"
+ "4. Restore\n"
+ "5. Verify count and content of table\n"){
+ INITIALIZER(runLoadTable);
+ VERIFIER(runVerifyBasic);
+ FINALIZER(runClearTable);
+
+}
+
+TESTCASE("GrepNodeRestart",
+ "Test that Global Replication works on one table \n"
+ "1. Load table\n"
+ "2. Grep\n"
+ "3. Restart -i\n"
+ "4. Restore\n"
+ "5. Verify count and content of table\n"){
+ INITIALIZER(runLoadTable);
+ STEP(runPkUpdate);
+ STEP(runRestarter);
+ VERIFIER(runVerifyBasic);
+ FINALIZER(runClearTable);
+}
+
+
+
+
+TESTCASE("GrepBank",
+ "Test that grep and restore works during transaction load\n"
+ " by backing up the bank"
+ "1. Create bank\n"
+ "2a. Start bank and let it run\n"
+ "2b. Perform loop number of greps of the bank\n"
+ " when greps are finished tell bank to close\n"
+ "3. Restart ndb -i and reload each grep\n"
+ " let bank verify that the grep is consistent\n"
+ "4. Drop bank\n"){
+ INITIALIZER(runCreateBank);
+ STEP(runBankTimer);
+ STEP(runBankTransactions);
+ STEP(runBankGL);
+ // TODO STEP(runBankSum);
+ STEP(runGrepBank);
+ // VERIFIER(runRestoreBankAndVerify);
+ // FINALIZER(runDropBank);
+
+}
+TESTCASE("NFMaster",
+ "Test that grep behaves during node failiure\n"){
+ INITIALIZER(setMaster);
+ STEP(runAbort);
+
+}
+TESTCASE("NFMasterAsSlave",
+ "Test that grep behaves during node failiure\n"){
+ INITIALIZER(setMasterAsSlave);
+ STEP(runAbort);
+
+}
+TESTCASE("NFSlave",
+ "Test that grep behaves during node failiure\n"){
+ INITIALIZER(setSlave);
+ STEP(runAbort);
+
+}
+TESTCASE("FailMaster",
+ "Test that grep behaves during node failiure\n"){
+ INITIALIZER(setMaster);
+ STEP(runFail);
+
+}
+TESTCASE("FailMasterAsSlave",
+ "Test that grep behaves during node failiure\n"){
+ INITIALIZER(setMasterAsSlave);
+ STEP(runFail);
+
+}
+TESTCASE("FailSlave",
+ "Test that grep behaves during node failiure\n"){
+ INITIALIZER(setSlave);
+ STEP(runFail);
+
+}
+NDBT_TESTSUITE_END(testGrep);
+
+int main(int argc, const char** argv){
+ return testGrep.execute(argc, argv);
+}
+
+
diff --git a/ndb/test/ndbapi/testGrep/verify/Makefile b/ndb/test/ndbapi/testGrep/verify/Makefile
new file mode 100644
index 00000000000..4e6182de6b2
--- /dev/null
+++ b/ndb/test/ndbapi/testGrep/verify/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testGrepVerify
+BIN_TARGET_LIBS += bank
+SOURCES = testGrepVerify.cpp
+
+CFLAGS_testGrepVerify.cpp += -I$(call fixpath,$(NDB_TOP)/include/kernel) -I$(call fixpath,$(NDB_TOP)/include/mgmcommon)
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testGrep/verify/testGrepVerify.cpp b/ndb/test/ndbapi/testGrep/verify/testGrepVerify.cpp
new file mode 100644
index 00000000000..056aa9bf173
--- /dev/null
+++ b/ndb/test/ndbapi/testGrep/verify/testGrepVerify.cpp
@@ -0,0 +1,122 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "mgmapi.h"
+#include <string.h>
+#include <NdbMain.h>
+#include <OutputStream.hpp>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <ConfigRetriever.hpp>
+#include <ndb_version.h>
+#include <assert.h>
+#include <NdbStdio.h>
+
+
+#define CHECK(b) if (!(b)) { \
+ g_err << "ERR: "<< "getStep" \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ continue; }
+
+int main(int argc, const char** argv){
+
+
+ const char * connectString = NULL;
+ const char * table = NULL;
+ int records = 0;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "connectString", 'c', arg_string, &connectString,
+ "ConnectString", "nodeid=<api id>;host=<hostname:port>" },
+ { "tableName", 't', arg_string, &table,
+ "table", "Table" },
+ { "records", 'r', arg_integer, &records, "Number of records", "recs"},
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "hostname:port\n"\
+ "This program will connect to the mgmsrv of a NDB cluster.\n"\
+ "It will then wait for all nodes to be started, then restart node(s)\n"\
+ "and wait for all to restart inbetween. It will do this \n"\
+ "loop number of times\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ ndbout_c("table %s connectStirng %s", table, connectString);
+ if(connectString == 0)
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ if(table == 0)
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+
+ Ndb::useFullyQualifiedNames(false);
+
+ Ndb * m_ndb = new Ndb("");
+ m_ndb->setConnectString(connectString);
+ Ndb::useFullyQualifiedNames(false);
+ /**
+ * @todo Set proper max no of transactions?? needed?? Default 12??
+ */
+ m_ndb->init(2048);
+ Ndb::useFullyQualifiedNames(false);
+ if (m_ndb->waitUntilReady() != 0){
+ ndbout_c("NDB Cluster not ready for connections");
+ }
+
+ int count = 0;
+ int result = NDBT_OK;
+
+
+ const NdbDictionary::Table * tab = NDBT_Table::discoverTableFromDb( m_ndb, table);
+// ndbout << *tab << endl;
+
+ UtilTransactions utilTrans(*tab);
+ HugoTransactions hugoTrans(*tab);
+
+ do{
+
+ // Check that there are as many records as we expected
+ CHECK(utilTrans.selectCount(m_ndb, 64, &count) == 0);
+
+ g_err << "count = " << count;
+ g_err << " records = " << records;
+ g_err << endl;
+
+ CHECK(count == records);
+
+ // Read and verify every record
+ CHECK(hugoTrans.pkReadRecords(m_ndb, records) == 0);
+
+ } while (false);
+
+
+ return NDBT_ProgramExit(result);
+
+}
diff --git a/ndb/test/ndbapi/testIndex/Makefile b/ndb/test/ndbapi/testIndex/Makefile
new file mode 100644
index 00000000000..e5cd4542c9c
--- /dev/null
+++ b/ndb/test/ndbapi/testIndex/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testIndex
+
+SOURCES = testIndex.cpp
+
+CFLAGS_testIndex.cpp := -I$(call fixpath,$(NDB_TOP)/include/kernel)
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testIndex/testIndex.cpp b/ndb/test/ndbapi/testIndex/testIndex.cpp
new file mode 100644
index 00000000000..a39c4ac49d5
--- /dev/null
+++ b/ndb/test/ndbapi/testIndex/testIndex.cpp
@@ -0,0 +1,1494 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <NdbRestarts.hpp>
+#include <Vector.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+
+#define CHECK(b) if (!(b)) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+}
+
+
+struct Attrib {
+ bool indexCreated;
+ int numAttribs;
+ int attribs[1024];
+ Attrib(){
+ numAttribs = 0;
+ indexCreated = false;
+ }
+};
+class AttribList {
+public:
+ AttribList(){};
+ ~AttribList(){
+ for(size_t i = 0; i < attriblist.size(); i++){
+ delete attriblist[i];
+ }
+ };
+ void buildAttribList(const NdbDictionary::Table* pTab);
+ Vector<Attrib*> attriblist;
+};
+
+void AttribList::buildAttribList(const NdbDictionary::Table* pTab){
+ attriblist.clear();
+
+ Attrib* attr;
+ // Build attrib definitions that describes which attributes to build index
+ // Try to build strange combinations, not just "all" or all PK's
+
+ for(int i = 1; i <= pTab->getNoOfColumns(); i++){
+ attr = new Attrib;
+ attr->numAttribs = i;
+ for(int a = 0; a<i; a++)
+ attr->attribs[a] = a;
+ attriblist.push_back(attr);
+ }
+ int b = 0;
+ for(int i = pTab->getNoOfColumns()-1; i > 0; i--){
+ attr = new Attrib;
+ attr->numAttribs = i;
+ b++;
+ for(int a = 0; a<i; a++)
+ attr->attribs[a] = a+b;
+ attriblist.push_back(attr);
+ }
+ for(int i = pTab->getNoOfColumns(); i > 0; i--){
+ attr = new Attrib;
+ attr->numAttribs = pTab->getNoOfColumns() - i;
+ for(int a = 0; a<pTab->getNoOfColumns() - i; a++)
+ attr->attribs[a] = pTab->getNoOfColumns()-a-1;
+ attriblist.push_back(attr);
+ }
+ for(int i = 1; i < pTab->getNoOfColumns(); i++){
+ attr = new Attrib;
+ attr->numAttribs = pTab->getNoOfColumns() - i;
+ for(int a = 0; a<pTab->getNoOfColumns() - i; a++)
+ attr->attribs[a] = pTab->getNoOfColumns()-a-1;
+ attriblist.push_back(attr);
+ }
+ for(int i = 1; i < pTab->getNoOfColumns(); i++){
+ attr = new Attrib;
+ attr->numAttribs = 2;
+ for(int a = 0; a<2; a++){
+ attr->attribs[a] = i%pTab->getNoOfColumns();
+ }
+ attriblist.push_back(attr);
+ }
+
+ // Last
+ attr = new Attrib;
+ attr->numAttribs = 1;
+ attr->attribs[0] = pTab->getNoOfColumns()-1;
+ attriblist.push_back(attr);
+
+ // Last and first
+ attr = new Attrib;
+ attr->numAttribs = 2;
+ attr->attribs[0] = pTab->getNoOfColumns()-1;
+ attr->attribs[1] = 0;
+ attriblist.push_back(attr);
+
+ // First and last
+ attr = new Attrib;
+ attr->numAttribs = 2;
+ attr->attribs[0] = 0;
+ attr->attribs[1] = pTab->getNoOfColumns()-1;
+ attriblist.push_back(attr);
+
+#if 0
+ for(size_t i = 0; i < attriblist.size(); i++){
+
+ ndbout << attriblist[i]->numAttribs << ": " ;
+ for(int a = 0; a < attriblist[i]->numAttribs; a++)
+ ndbout << attriblist[i]->attribs[a] << ", ";
+ ndbout << endl;
+ }
+#endif
+
+}
+
+char idxName[255];
+char pkIdxName[255];
+
+static const int SKIP_INDEX = 99;
+
+int create_index(NDBT_Context* ctx, int indxNum,
+ const NdbDictionary::Table* pTab,
+ Ndb* pNdb, Attrib* attr, bool logged){
+ bool orderedIndex = ctx->getProperty("OrderedIndex", (unsigned)0);
+ int result = NDBT_OK;
+
+ HugoCalculator calc(*pTab);
+
+ if (attr->numAttribs == 1 &&
+ calc.isUpdateCol(attr->attribs[0]) == true){
+ // Don't create index for the Hugo update column
+ // since it's not unique
+ return SKIP_INDEX;
+ }
+
+ // Create index
+ snprintf(idxName, 255, "IDC%d", indxNum);
+ if (orderedIndex)
+ ndbout << "Creating " << ((logged)?"logged ": "temporary ") << "ordered index "<<idxName << " (";
+ else
+ ndbout << "Creating " << ((logged)?"logged ": "temporary ") << "unique index "<<idxName << " (";
+ ndbout << flush;
+ NdbDictionary::Index pIdx(idxName);
+ pIdx.setTable(pTab->getName());
+ if (orderedIndex)
+ pIdx.setType(NdbDictionary::Index::OrderedIndex);
+ else
+ pIdx.setType(NdbDictionary::Index::UniqueHashIndex);
+ for (int c = 0; c< attr->numAttribs; c++){
+ int attrNo = attr->attribs[c];
+ pIdx.addIndexColumn(pTab->getColumn(attrNo)->getName());
+ ndbout << pTab->getColumn(attrNo)->getName()<<" ";
+ }
+
+ pIdx.setStoredIndex(logged);
+ ndbout << ") ";
+ if (pNdb->getDictionary()->createIndex(pIdx) != 0){
+ attr->indexCreated = false;
+ ndbout << "FAILED!" << endl;
+ const NdbError err = pNdb->getDictionary()->getNdbError();
+ ERR(err);
+ if(err.classification == NdbError::ApplicationError)
+ return SKIP_INDEX;
+
+ return NDBT_FAILED;
+ } else {
+ ndbout << "OK!" << endl;
+ attr->indexCreated = true;
+ }
+ return result;
+}
+
+
+int drop_index(int indxNum, Ndb* pNdb,
+ const NdbDictionary::Table* pTab, Attrib* attr){
+ int result = NDBT_OK;
+
+ if (attr->indexCreated == false)
+ return NDBT_OK;
+
+ snprintf(idxName, 255, "IDC%d", indxNum);
+
+ // Drop index
+ ndbout << "Dropping index "<<idxName<<"(" << pTab->getName() << ") ";
+ if (pNdb->getDictionary()->dropIndex(idxName, pTab->getName()) != 0){
+ ndbout << "FAILED!" << endl;
+ ERR(pNdb->getDictionary()->getNdbError());
+ result = NDBT_FAILED;
+ } else {
+ ndbout << "OK!" << endl;
+ }
+ return result;
+}
+
+int runCreateIndexes(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int l = 0;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ // NOTE If we need to test creating both logged and non logged indexes
+ // this should be divided into two testcases
+ // The paramater logged should then be specified
+ // as a TC_PROPERTY. ex TC_PROPERTY("LoggedIndexes", 1);
+ // and read into the test step like
+ bool logged = ctx->getProperty("LoggedIndexes", 1);
+
+ AttribList attrList;
+ attrList.buildAttribList(pTab);
+
+
+ while (l < loops && result == NDBT_OK){
+
+ for (unsigned int i = 0; i < attrList.attriblist.size(); i++){
+
+ // Try to create index
+ if (create_index(ctx, i, pTab, pNdb, attrList.attriblist[i], logged) == NDBT_FAILED)
+ result = NDBT_FAILED;
+ }
+
+ // Now drop all indexes that where created
+ for (unsigned int i = 0; i < attrList.attriblist.size(); i++){
+
+ // Try to drop index
+ if (drop_index(i, pNdb, pTab, attrList.attriblist[i]) != NDBT_OK)
+ result = NDBT_FAILED;
+ }
+
+ l++;
+ }
+
+ return result;
+}
+
+int createRandomIndex(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+ bool logged = ctx->getProperty("LoggedIndexes", 1);
+
+ AttribList attrList;
+ attrList.buildAttribList(pTab);
+
+ int retries = 10;
+ while(retries > 0){
+ const Uint32 i = rand() % attrList.attriblist.size();
+ int res = create_index(ctx, i, pTab, pNdb, attrList.attriblist[i],
+ logged);
+ if (res == SKIP_INDEX)
+ continue;
+
+ if (res == NDBT_FAILED){
+ retries--;
+ continue;
+ }
+
+ ctx->setProperty("createRandomIndex", i);
+ // Now drop all indexes that where created
+
+ return NDBT_OK;
+ }
+
+ return NDBT_FAILED;
+}
+
+int createRandomIndex_Drop(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+
+ Uint32 i = ctx->getProperty("createRandomIndex");
+
+ snprintf(idxName, 255, "IDC%d", i);
+
+ // Drop index
+ ndbout << "Dropping index " << idxName << " ";
+ if (pNdb->getDictionary()->dropIndex(idxName,
+ ctx->getTab()->getName()) != 0){
+ ndbout << "FAILED!" << endl;
+ ERR(pNdb->getDictionary()->getNdbError());
+ return NDBT_FAILED;
+ } else {
+ ndbout << "OK!" << endl;
+ }
+
+ return NDBT_OK;
+}
+
+int createPkIndex(NDBT_Context* ctx, NDBT_Step* step){
+ bool orderedIndex = ctx->getProperty("OrderedIndex", (unsigned)0);
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+
+ bool logged = ctx->getProperty("LoggedIndexes", 1);
+
+ // Create index
+ snprintf(pkIdxName, 255, "IDC_PK_%s", pTab->getName());
+ if (orderedIndex)
+ ndbout << "Creating " << ((logged)?"logged ": "temporary ") << "ordered index "
+ << pkIdxName << " (";
+ else
+ ndbout << "Creating " << ((logged)?"logged ": "temporary ") << "unique index "
+ << pkIdxName << " (";
+
+ NdbDictionary::Index pIdx(pkIdxName);
+ pIdx.setTable(pTab->getName());
+ if (orderedIndex)
+ pIdx.setType(NdbDictionary::Index::OrderedIndex);
+ else
+ pIdx.setType(NdbDictionary::Index::UniqueHashIndex);
+ for (int c = 0; c< pTab->getNoOfColumns(); c++){
+ const NdbDictionary::Column * col = pTab->getColumn(c);
+ if(col->getPrimaryKey()){
+ pIdx.addIndexColumn(col->getName());
+ ndbout << col->getName() <<" ";
+ }
+ }
+
+ pIdx.setStoredIndex(logged);
+ ndbout << ") ";
+ if (pNdb->getDictionary()->createIndex(pIdx) != 0){
+ ndbout << "FAILED!" << endl;
+ const NdbError err = pNdb->getDictionary()->getNdbError();
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ ndbout << "OK!" << endl;
+ return NDBT_OK;
+}
+
+int createPkIndex_Drop(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+
+ // Drop index
+ ndbout << "Dropping index " << pkIdxName << " ";
+ if (pNdb->getDictionary()->dropIndex(pkIdxName,
+ pTab->getName()) != 0){
+ ndbout << "FAILED!" << endl;
+ ERR(pNdb->getDictionary()->getNdbError());
+ return NDBT_FAILED;
+ } else {
+ ndbout << "OK!" << endl;
+ }
+
+ return NDBT_OK;
+}
+
+int
+runVerifyIndex(NDBT_Context* ctx, NDBT_Step* step){
+ // Verify that data in index match
+ // table data
+ Ndb* pNdb = GETNDB(step);
+ UtilTransactions utilTrans(*ctx->getTab());
+ const int batchSize = ctx->getProperty("BatchSize", 16);
+ const int parallelism = batchSize > 240 ? 240 : batchSize;
+
+ do {
+ if (utilTrans.verifyIndex(pNdb, idxName, parallelism, true) != 0){
+ g_err << "Inconsistent index" << endl;
+ return NDBT_FAILED;
+ }
+ } while(ctx->isTestStopped() == false);
+ return NDBT_OK;
+}
+
+int
+runTransactions1(NDBT_Context* ctx, NDBT_Step* step){
+ // Verify that data in index match
+ // table data
+ Ndb* pNdb = GETNDB(step);
+ HugoTransactions hugoTrans(*ctx->getTab());
+ const int batchSize = ctx->getProperty("BatchSize", 50);
+
+ int rows = ctx->getNumRecords();
+ while (ctx->isTestStopped() == false) {
+ if (hugoTrans.pkUpdateRecords(pNdb, rows, batchSize) != 0){
+ g_err << "Updated table failed" << endl;
+ return NDBT_FAILED;
+ }
+ if (hugoTrans.scanUpdateRecords(pNdb, rows, batchSize) != 0){
+ g_err << "Updated table failed" << endl;
+ return NDBT_FAILED;
+ }
+ }
+ return NDBT_OK;
+}
+
+int
+runTransactions2(NDBT_Context* ctx, NDBT_Step* step){
+ // Verify that data in index match
+ // table data
+ Ndb* pNdb = GETNDB(step);
+ HugoTransactions hugoTrans(*ctx->getTab());
+ const int batchSize = ctx->getProperty("BatchSize", 50);
+
+ int rows = ctx->getNumRecords();
+ while (ctx->isTestStopped() == false) {
+#if 1
+ if (hugoTrans.indexReadRecords(pNdb, pkIdxName, rows, batchSize) != 0){
+ g_err << "Index read failed" << endl;
+ return NDBT_FAILED;
+ }
+#endif
+
+ if(ctx->isTestStopped())
+ break;
+#if 1
+ if (hugoTrans.indexUpdateRecords(pNdb, pkIdxName, rows, batchSize) != 0){
+ g_err << "Index update failed" << endl;
+ return NDBT_FAILED;
+ }
+#endif
+ }
+ return NDBT_OK;
+}
+
+int
+runTransactions3(NDBT_Context* ctx, NDBT_Step* step){
+ // Verify that data in index match
+ // table data
+ Ndb* pNdb = GETNDB(step);
+ HugoTransactions hugoTrans(*ctx->getTab());
+ UtilTransactions utilTrans(*ctx->getTab());
+ const int batchSize = ctx->getProperty("BatchSize", 32);
+ const int parallel = batchSize > 240 ? 240 : batchSize;
+
+ int rows = ctx->getNumRecords();
+ while (ctx->isTestStopped() == false) {
+ if(hugoTrans.loadTable(pNdb, rows, batchSize, false) != 0){
+ g_err << "Load table failed" << endl;
+ return NDBT_FAILED;
+ }
+ if(ctx->isTestStopped())
+ break;
+
+ if (hugoTrans.pkUpdateRecords(pNdb, rows, batchSize) != 0){
+ g_err << "Updated table failed" << endl;
+ return NDBT_FAILED;
+ }
+
+ if(ctx->isTestStopped())
+ break;
+
+ if (hugoTrans.indexReadRecords(pNdb, pkIdxName, rows, batchSize) != 0){
+ g_err << "Index read failed" << endl;
+ return NDBT_FAILED;
+ }
+
+ if(ctx->isTestStopped())
+ break;
+
+ if (hugoTrans.indexUpdateRecords(pNdb, pkIdxName, rows, batchSize) != 0){
+ g_err << "Index update failed" << endl;
+ return NDBT_FAILED;
+ }
+
+ if(ctx->isTestStopped())
+ break;
+
+ if (hugoTrans.scanUpdateRecords(pNdb, rows, 5, parallel) != 0){
+ g_err << "Scan updated table failed" << endl;
+ return NDBT_FAILED;
+ }
+
+ if(ctx->isTestStopped())
+ break;
+
+ if(utilTrans.clearTable(pNdb, rows, parallel) != 0){
+ g_err << "Clear table failed" << endl;
+ return NDBT_FAILED;
+ }
+ if(ctx->isTestStopped())
+ break;
+
+ int count = -1;
+ if(utilTrans.selectCount(pNdb, 64, &count) != 0 || count != 0)
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runRestarts(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ NDBT_TestCase* pCase = ctx->getCase();
+ NdbRestarts restarts;
+ int i = 0;
+ int timeout = 240;
+
+ while(i<loops && result != NDBT_FAILED && !ctx->isTestStopped()){
+ if(restarts.executeRestart("RestartRandomNodeAbort", timeout) != 0){
+ g_err << "Failed to executeRestart(" <<pCase->getName() <<")" << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+ i++;
+ }
+ ctx->stopTest();
+ return result;
+}
+
+int runCreateLoadDropIndex(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int l = 0;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int parallelism = batchSize > 240? 240: batchSize;
+ ndbout << "batchSize="<<batchSize<<endl;
+ bool logged = ctx->getProperty("LoggedIndexes", 1);
+
+ HugoTransactions hugoTrans(*pTab);
+ UtilTransactions utilTrans(*pTab);
+ AttribList attrList;
+ attrList.buildAttribList(pTab);
+
+ for (unsigned int i = 0; i < attrList.attriblist.size(); i++){
+
+ while (l < loops && result == NDBT_OK){
+
+ if ((l % 2) == 0){
+ // Create index first and then load
+
+ // Try to create index
+ if (create_index(ctx, i, pTab, pNdb, attrList.attriblist[i], logged) == NDBT_FAILED){
+ result = NDBT_FAILED;
+ }
+
+ // Load the table with data
+ ndbout << "Loading data after" << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records, batchSize) == 0);
+
+
+ } else {
+ // Load table then create index
+
+ // Load the table with data
+ ndbout << "Loading data before" << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records, batchSize) == 0);
+
+ // Try to create index
+ if (create_index(ctx, i, pTab, pNdb, attrList.attriblist[i], logged) == NDBT_FAILED)
+ result = NDBT_FAILED;
+
+ }
+
+ // Verify that data in index match
+ // table data
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+
+ // Do it all...
+ ndbout <<"Doing it all"<<endl;
+ int count;
+ ndbout << " pkUpdateRecords" << endl;
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records, batchSize) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records, batchSize) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+ ndbout << " pkDelRecords half" << endl;
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2, batchSize) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+ ndbout << " scanUpdateRecords" << endl;
+ CHECK(hugoTrans.scanUpdateRecords(pNdb, records/2, parallelism) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+ ndbout << " clearTable" << endl;
+ CHECK(utilTrans.clearTable(pNdb, records/2, parallelism) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+ ndbout << " loadTable" << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records, batchSize) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+ ndbout << " loadTable again" << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records, batchSize) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+
+
+ if ((l % 2) == 0){
+ // Drop index first and then clear
+
+ // Try to create index
+ if (drop_index(i, pNdb, pTab, attrList.attriblist[i]) != NDBT_OK){
+ result = NDBT_FAILED;
+ }
+
+ // Clear table
+ ndbout << "Clearing table after" << endl;
+ CHECK(hugoTrans.clearTable(pNdb, records, parallelism) == 0);
+
+
+ } else {
+ // Clear table then drop index
+
+ //Clear table
+ ndbout << "Clearing table before" << endl;
+ CHECK(hugoTrans.clearTable(pNdb, records, parallelism) == 0);
+
+ // Try to drop index
+ if (drop_index(i, pNdb, pTab, attrList.attriblist[i]) != NDBT_OK)
+ result = NDBT_FAILED;
+ }
+
+ ndbout << " Done!" << endl;
+ l++;
+ }
+
+ // Make sure index is dropped
+ drop_index(i, pNdb, pTab, attrList.attriblist[i]);
+
+ }
+
+ return result;
+}
+
+int runInsertDelete(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ int parallelism = batchSize > 240? 240: batchSize;
+ ndbout << "batchSize="<<batchSize<<endl;
+ bool logged = ctx->getProperty("LoggedIndexes", 1);
+
+ HugoTransactions hugoTrans(*pTab);
+ UtilTransactions utilTrans(*pTab);
+
+ AttribList attrList;
+ attrList.buildAttribList(pTab);
+
+ for (unsigned int i = 0; i < attrList.attriblist.size(); i++){
+
+ Attrib* attr = attrList.attriblist[i];
+ // Create index
+ if (create_index(ctx, i, pTab, pNdb, attr, logged) == NDBT_OK){
+ int l = 1;
+ while (l <= loops && result == NDBT_OK){
+
+ CHECK(hugoTrans.loadTable(pNdb, records, batchSize) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records, parallelism) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, parallelism) == 0);
+ l++;
+ }
+
+ // Drop index
+ if (drop_index(i, pNdb, pTab, attr) != NDBT_OK)
+ result = NDBT_FAILED;
+ }
+ }
+
+ return result;
+}
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ int batchSize = ctx->getProperty("BatchSize", 1);
+ if(hugoTrans.loadTable(GETNDB(step), records, batchSize) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runSystemRestart1(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int timeout = 300;
+ Uint32 loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int count;
+ NdbRestarter restarter;
+ Uint32 i = 1;
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+ const char * name = ctx->getTab()->getName();
+ while(i<=loops && result != NDBT_FAILED){
+
+ ndbout << "Loop " << i << "/"<< loops <<" started" << endl;
+ /*
+ 1. Load data
+ 2. Restart cluster and verify records
+ 3. Update records
+ 4. Restart cluster and verify records
+ 5. Delete half of the records
+ 6. Restart cluster and verify records
+ 7. Delete all records
+ 8. Restart cluster and verify records
+ 9. Insert, update, delete records
+ 10. Restart cluster and verify records
+ 11. Insert, update, delete records
+ 12. Restart cluster with error insert 5020 and verify records
+ */
+ ndbout << "Loading records..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records, 1) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+
+ ndbout << "Restarting cluster" << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+
+ ndbout << "Updating records..." << endl;
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+
+ ndbout << "Deleting 50% of records..." << endl;
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.scanReadRecords(pNdb, records/2, 0, 64) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+
+ ndbout << "Deleting all records..." << endl;
+ CHECK(utilTrans.clearTable(pNdb, records/2) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+
+ ndbout << "Doing it all..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records, 1) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(hugoTrans.scanUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records, 1) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records, 1) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+
+ ndbout << "Doing it all..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records, 1) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+ CHECK(hugoTrans.scanUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.verifyIndex(pNdb, idxName, 16, false) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records, 1) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+
+ ndbout << "Restarting cluster with error insert 5020..." << endl;
+ CHECK(restarter.restartAll(false, true) == 0);
+ CHECK(restarter.waitClusterNoStart(timeout) == 0);
+ CHECK(restarter.insertErrorInAllNodes(5020) == 0);
+ CHECK(restarter.startAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ i++;
+ }
+
+ ctx->stopTest();
+ ndbout << "runSystemRestart1 finished" << endl;
+
+ return result;
+}
+
+#define CHECK2(b, t) if(!b){ g_err << __LINE__ << ": " << t << endl; break;}
+
+int
+runMixed1(NDBT_Context* ctx, NDBT_Step* step){
+ // Verify that data in index match
+ // table data
+ Ndb* pNdb = GETNDB(step);
+ HugoOperations hugoOps(*ctx->getTab());
+
+
+ do {
+ // TC1
+ g_err << "pkRead, indexRead, Commit" << endl;
+ CHECK2(hugoOps.startTransaction(pNdb) == 0, "startTransaction");
+ CHECK2(hugoOps.indexReadRecords(pNdb, pkIdxName, 0) == 0, "indexReadRecords");
+ CHECK2(hugoOps.pkReadRecord(pNdb, 0) == 0, "pkReadRecord");
+ CHECK2(hugoOps.execute_Commit(pNdb) == 0, "executeCommit");
+ CHECK2(hugoOps.closeTransaction(pNdb) == 0, "closeTransaction");
+
+ // TC1
+ g_err << "pkRead, indexRead, Commit" << endl;
+ CHECK2(hugoOps.startTransaction(pNdb) == 0, "startTransaction");
+ CHECK2(hugoOps.pkReadRecord(pNdb, 0) == 0, "pkReadRecord");
+ CHECK2(hugoOps.indexReadRecords(pNdb, pkIdxName, 0) == 0, "indexReadRecords");
+ CHECK2(hugoOps.execute_Commit(pNdb) == 0, "executeCommit");
+ CHECK2(hugoOps.closeTransaction(pNdb) == 0, "closeTransaction");
+
+
+ // TC2
+ g_err << "pkRead, indexRead, NoCommit, Commit" << endl;
+ CHECK2(hugoOps.startTransaction(pNdb) == 0, "startTransaction");
+ CHECK2(hugoOps.pkReadRecord(pNdb, 0) == 0, "pkReadRecord");
+ CHECK2(hugoOps.indexReadRecords(pNdb, pkIdxName, 0) == 0,
+ "indexReadRecords");
+ CHECK2(hugoOps.execute_NoCommit(pNdb) == 0, "executeNoCommit");
+ CHECK2(hugoOps.execute_Commit(pNdb) == 0, "executeCommit");
+ CHECK2(hugoOps.closeTransaction(pNdb) == 0, "closeTransaction");
+
+ // TC3
+ g_err << "pkRead, pkRead, Commit" << endl;
+ CHECK2(hugoOps.startTransaction(pNdb) == 0, "startTransaction ");
+ CHECK2(hugoOps.pkReadRecord(pNdb, 0) == 0, "pkReadRecords ");
+ CHECK2(hugoOps.pkReadRecord(pNdb, 0) == 0, "pkReadRecords ");
+ CHECK2(hugoOps.execute_Commit(pNdb) == 0, "executeCommit");
+ CHECK2(hugoOps.closeTransaction(pNdb) == 0, "closeTransaction ");
+
+ // TC4
+ g_err << "indexRead, indexRead, Commit" << endl;
+
+ CHECK2(hugoOps.startTransaction(pNdb) == 0, "startTransaction ");
+ CHECK2(hugoOps.indexReadRecords(pNdb, pkIdxName, 0) == 0, "indexReadRecords");
+ CHECK2(hugoOps.indexReadRecords(pNdb, pkIdxName, 0) == 0, "indexReadRecords");
+ CHECK2(hugoOps.execute_Commit(pNdb) == 0, "executeCommit");
+
+ CHECK2(hugoOps.closeTransaction(pNdb) == 0, "closeTransaction ");
+
+ return NDBT_OK;
+ } while(false);
+
+
+ hugoOps.closeTransaction(pNdb);
+ return NDBT_FAILED;
+}
+
+int
+runBuildDuring(NDBT_Context* ctx, NDBT_Step* step){
+ // Verify that data in index match
+ // table data
+ const int Threads = ctx->getProperty("Threads", (Uint32)0);
+ const int loops = ctx->getNumLoops();
+
+ for(int i = 0; i<loops; i++){
+#if 1
+ if(createPkIndex(ctx, step) != NDBT_OK){
+ g_err << "Failed to create index" << endl;
+ return NDBT_FAILED;
+ }
+#endif
+
+ if(ctx->isTestStopped())
+ break;
+
+#if 1
+ if(createRandomIndex(ctx, step) != NDBT_OK){
+ g_err << "Failed to create index" << endl;
+ return NDBT_FAILED;
+ }
+#endif
+
+ if(ctx->isTestStopped())
+ break;
+
+ ctx->setProperty("pause", 1);
+ int count = 0;
+ for(int j = 0; count < Threads && !ctx->isTestStopped();
+ j = (j+1) % Threads){
+ char buf[255];
+ sprintf(buf, "Thread%d_paused", j);
+ int tmp = ctx->getProperty(buf, (Uint32)0);
+ count += tmp;
+ }
+
+ if(ctx->isTestStopped())
+ break;
+
+#if 1
+ if(createPkIndex_Drop(ctx, step) != NDBT_OK){
+ g_err << "Failed to drop index" << endl;
+ return NDBT_FAILED;
+ }
+#endif
+
+ if(ctx->isTestStopped())
+ break;
+
+#if 1
+ if(createRandomIndex_Drop(ctx, step) != NDBT_OK){
+ g_err << "Failed to drop index" << endl;
+ return NDBT_FAILED;
+ }
+#endif
+
+ ctx->setProperty("pause", (Uint32)0);
+ NdbSleep_SecSleep(2);
+ }
+
+ ctx->stopTest();
+ return NDBT_OK;
+}
+
+static NdbLockable g_lock;
+static int threadCounter = 0;
+
+void
+wait_paused(NDBT_Context* ctx, int id){
+ if(ctx->getProperty("pause", (Uint32)0) == 1){
+ char buf[255];
+ sprintf(buf, "Thread%d_paused", id);
+ ctx->setProperty(buf, 1);
+ while(!ctx->isTestStopped() && ctx->getProperty("pause", (Uint32)0) == 1){
+ NdbSleep_MilliSleep(250);
+ }
+ ctx->setProperty(buf, (Uint32)0);
+ }
+}
+
+int
+runTransactions4(NDBT_Context* ctx, NDBT_Step* step){
+
+ g_lock.lock();
+ const int ThreadId = threadCounter++;
+ g_lock.unlock();
+
+ // Verify that data in index match
+ // table data
+ Ndb* pNdb = GETNDB(step);
+ HugoTransactions hugoTrans(*ctx->getTab());
+ UtilTransactions utilTrans(*ctx->getTab());
+ const int batchSize = ctx->getProperty("BatchSize", 32);
+ const int parallel = batchSize > 240 ? 240 : batchSize;
+
+ int rows = ctx->getNumRecords();
+ while (ctx->isTestStopped() == false) {
+ if(hugoTrans.loadTable(pNdb, rows, batchSize, false) != 0){
+ g_err << "Load table failed" << endl;
+ return NDBT_FAILED;
+ }
+
+ wait_paused(ctx, ThreadId);
+
+ if(ctx->isTestStopped())
+ break;
+
+ if (hugoTrans.pkUpdateRecords(pNdb, rows, batchSize) != 0){
+ g_err << "Updated table failed" << endl;
+ return NDBT_FAILED;
+ }
+
+ wait_paused(ctx, ThreadId);
+
+ if(ctx->isTestStopped())
+ break;
+
+ if (hugoTrans.scanUpdateRecords(pNdb, rows, 5, parallel) != 0){
+ g_err << "Scan updated table failed" << endl;
+ return NDBT_FAILED;
+ }
+
+ wait_paused(ctx, ThreadId);
+
+ if(ctx->isTestStopped())
+ break;
+
+ if(utilTrans.clearTable(pNdb, rows, parallel) != 0){
+ g_err << "Clear table failed" << endl;
+ return NDBT_FAILED;
+ }
+ }
+ return NDBT_OK;
+}
+
+int
+runUniqueNullTransactions(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+
+ bool logged = ctx->getProperty("LoggedIndexes", 1);
+ bool orderedIndex = ctx->getProperty("OrderedIndex", (unsigned)0);
+ NdbConnection * pTrans = 0;
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ // Create index
+ char nullIndex[255];
+ snprintf(nullIndex, 255, "IDC_PK_%s_NULL", pTab->getName());
+ if (orderedIndex)
+ ndbout << "Creating " << ((logged)?"logged ": "temporary ") << "ordered index "
+ << pkIdxName << " (";
+ else
+ ndbout << "Creating " << ((logged)?"logged ": "temporary ") << "unique index "
+ << pkIdxName << " (";
+
+ NdbDictionary::Index pIdx(pkIdxName);
+ pIdx.setTable(pTab->getName());
+ if (orderedIndex)
+ pIdx.setType(NdbDictionary::Index::OrderedIndex);
+ else
+ pIdx.setType(NdbDictionary::Index::UniqueHashIndex);
+ pIdx.setStoredIndex(logged);
+
+ for (int c = 0; c< pTab->getNoOfColumns(); c++){
+ const NdbDictionary::Column * col = pTab->getColumn(c);
+ if(col->getPrimaryKey()){
+ pIdx.addIndexColumn(col->getName());
+ ndbout << col->getName() <<" ";
+ }
+ }
+
+ int colId = -1;
+ for (int c = 0; c< pTab->getNoOfColumns(); c++){
+ const NdbDictionary::Column * col = pTab->getColumn(c);
+ if(col->getNullable()){
+ pIdx.addIndexColumn(col->getName());
+ ndbout << col->getName() <<" ";
+ colId = c;
+ break;
+ }
+ }
+ ndbout << ") ";
+
+ if(colId == -1){
+ ndbout << endl << "No nullable column found -> NDBT_FAILED" << endl;
+ return NDBT_FAILED;
+ }
+
+ if (pNdb->getDictionary()->createIndex(pIdx) != 0){
+ ndbout << "FAILED!" << endl;
+ const NdbError err = pNdb->getDictionary()->getNdbError();
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ int result = NDBT_OK;
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ const int batchSize = ctx->getProperty("BatchSize", 50);
+ int loops = ctx->getNumLoops();
+ int rows = ctx->getNumRecords();
+ while (loops-- > 0 && ctx->isTestStopped() == false) {
+ if (hugoTrans.pkUpdateRecords(pNdb, rows, batchSize) != 0){
+ g_err << "Updated table failed" << endl;
+ result = NDBT_FAILED;
+ goto done;
+ }
+ }
+
+ if(ctx->isTestStopped()){
+ goto done;
+ }
+
+ ctx->stopTest();
+ while(ctx->getNoOfRunningSteps() > 1){
+ NdbSleep_MilliSleep(100);
+ }
+
+ result = NDBT_FAILED;
+ pTrans = pNdb->startTransaction();
+ NdbScanOperation * sOp;
+ NdbOperation * uOp;
+ NdbResultSet * rs;
+ int eof;
+ if(!pTrans) goto done;
+ sOp = pTrans->getNdbScanOperation(pTab->getName());
+ if(!sOp) goto done;
+ rs = sOp->readTuples(240, NdbScanOperation::LM_Exclusive);
+ if(!rs) goto done;
+ if(pTrans->execute(NoCommit) == -1) goto done;
+ while((eof = rs->nextResult(true)) == 0){
+ do {
+ NdbOperation * uOp = rs->updateTuple();
+ if(uOp == 0) goto done;
+ uOp->setValue(colId, 0);
+ } while((eof = rs->nextResult(false)) == 0);
+ eof = pTrans->execute(Commit);
+ if(eof == -1) goto done;
+ }
+
+ done:
+ if(pTrans) pNdb->closeTransaction(pTrans);
+ pNdb->getDictionary()->dropIndex(nullIndex, pTab->getName());
+ return result;
+}
+
+int runLQHKEYREF(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops() * 100;
+ NdbRestarter restarter;
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+#if 0
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ if(restarter.dumpStateAllNodes(&val, 1) != 0){
+ g_err << "Failed to dump DihMinTimeBetweenLCP" << endl;
+ return NDBT_FAILED;
+ }
+#endif
+
+ for(int i = 0; i<loops && !ctx->isTestStopped(); i++){
+ int randomId = myRandom48(restarter.getNumDbNodes());
+ int nodeId = restarter.getDbNodeId(randomId);
+
+ const Uint32 error = 5031 + (i % 3);
+
+ if(restarter.insertErrorInNode(nodeId, error) != 0){
+ g_err << "Failed to error insert( " << error << ") in node "
+ << nodeId << endl;
+ return NDBT_FAILED;
+ }
+ }
+
+ ctx->stopTest();
+ return NDBT_OK;
+}
+
+NDBT_TESTSUITE(testIndex);
+TESTCASE("CreateAll",
+ "Test that we can create all various indexes on each table\n"
+ "Then drop the indexes\n"){
+ INITIALIZER(runCreateIndexes);
+}
+TESTCASE("CreateAll_O",
+ "Test that we can create all various indexes on each table\n"
+ "Then drop the indexes\n"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runCreateIndexes);
+}
+TESTCASE("InsertDeleteGentle",
+ "Create one index, then perform insert and delete in the table\n"
+ "loop number of times. Use batch size 1."){
+ TC_PROPERTY("BatchSize", 1);
+ INITIALIZER(runInsertDelete);
+ FINALIZER(runClearTable);
+}
+TESTCASE("InsertDeleteGentle_O",
+ "Create one index, then perform insert and delete in the table\n"
+ "loop number of times. Use batch size 1."){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ TC_PROPERTY("BatchSize", 1);
+ INITIALIZER(runInsertDelete);
+ FINALIZER(runClearTable);
+}
+TESTCASE("InsertDelete",
+ "Create one index, then perform insert and delete in the table\n"
+ "loop number of times. Use batchsize 512 to stress db more"){
+ TC_PROPERTY("BatchSize", 512);
+ INITIALIZER(runInsertDelete);
+ FINALIZER(runClearTable);
+
+}
+TESTCASE("InsertDelete_O",
+ "Create one index, then perform insert and delete in the table\n"
+ "loop number of times. Use batchsize 512 to stress db more"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ TC_PROPERTY("BatchSize", 512);
+ INITIALIZER(runInsertDelete);
+ FINALIZER(runClearTable);
+
+}
+TESTCASE("CreateLoadDropGentle",
+ "Try to create, drop and load various indexes \n"
+ "on table loop number of times.Usa batch size 1.\n"){
+ TC_PROPERTY("BatchSize", 1);
+ INITIALIZER(runCreateLoadDropIndex);
+}
+TESTCASE("CreateLoadDropGentle_O",
+ "Try to create, drop and load various indexes \n"
+ "on table loop number of times.Usa batch size 1.\n"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ TC_PROPERTY("BatchSize", 1);
+ INITIALIZER(runCreateLoadDropIndex);
+}
+TESTCASE("CreateLoadDrop",
+ "Try to create, drop and load various indexes \n"
+ "on table loop number of times. Use batchsize 512 to stress db more\n"){
+ TC_PROPERTY("BatchSize", 512);
+ INITIALIZER(runCreateLoadDropIndex);
+}
+TESTCASE("CreateLoadDrop_O",
+ "Try to create, drop and load various indexes \n"
+ "on table loop number of times. Use batchsize 512 to stress db more\n"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ TC_PROPERTY("BatchSize", 512);
+ INITIALIZER(runCreateLoadDropIndex);
+}
+TESTCASE("NFNR1",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ STEP(runTransactions1);
+ STEP(runTransactions1);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NFNR1_O",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ STEP(runTransactions1);
+ STEP(runTransactions1);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NFNR2",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ STEP(runTransactions2);
+ STEP(runTransactions2);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NFNR2_O",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ STEP(runTransactions2);
+ STEP(runTransactions2);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NFNR3",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ STEP(runRestarts);
+ STEP(runTransactions3);
+ STEP(runVerifyIndex);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NFNR3_O",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ STEP(runRestarts);
+ STEP(runTransactions3);
+ STEP(runVerifyIndex);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NFNR4",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ STEP(runTransactions1);
+ STEP(runTransactions1);
+ STEP(runTransactions2);
+ STEP(runTransactions2);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NFNR4_O",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ STEP(runTransactions1);
+ STEP(runTransactions1);
+ STEP(runTransactions2);
+ STEP(runTransactions2);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NFNR5",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ TC_PROPERTY("BatchSize", (unsigned)1);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runLQHKEYREF);
+ STEP(runTransactions1);
+ STEP(runTransactions1);
+ STEP(runTransactions2);
+ STEP(runTransactions2);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NFNR5_O",
+ "Test that indexes are correctly maintained during node fail and node restart"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ TC_PROPERTY("BatchSize", (unsigned)1);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runLQHKEYREF);
+ STEP(runTransactions1);
+ STEP(runTransactions1);
+ STEP(runTransactions2);
+ STEP(runTransactions2);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("SR1",
+ "Test that indexes are correctly maintained during SR"){
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ STEP(runSystemRestart1);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("MixedTransaction",
+ "Test mixing of index and normal operations"){
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createPkIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runMixed1);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("SR1_O",
+ "Test that indexes are correctly maintained during SR"){
+ TC_PROPERTY("OrderedIndex", 1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ STEP(runSystemRestart1);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(runClearTable);
+}
+TESTCASE("BuildDuring",
+ "Test that index build when running transactions work"){
+ TC_PROPERTY("OrderedIndex", (unsigned)0);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ TC_PROPERTY("Threads", 2); // # runTransactions4
+ INITIALIZER(runClearTable);
+ STEP(runBuildDuring);
+ STEP(runTransactions4);
+ STEP(runTransactions4);
+ FINALIZER(runClearTable);
+}
+TESTCASE("BuildDuring_O",
+ "Test that index build when running transactions work"){
+ TC_PROPERTY("OrderedIndex", (unsigned)1);
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ TC_PROPERTY("Threads", 2); // # runTransactions4
+ INITIALIZER(runClearTable);
+ STEP(runBuildDuring);
+ STEP(runTransactions4);
+ STEP(runTransactions4);
+ FINALIZER(runClearTable);
+}
+TESTCASE("UniqueNull",
+ "Test that unique indexes and nulls"){
+ TC_PROPERTY("LoggedIndexes", (unsigned)0);
+ INITIALIZER(runClearTable);
+ INITIALIZER(createRandomIndex);
+ INITIALIZER(createPkIndex);
+ INITIALIZER(runLoadTable);
+ STEP(runTransactions1);
+ STEP(runTransactions2);
+ STEP(runUniqueNullTransactions);
+ FINALIZER(runVerifyIndex);
+ FINALIZER(createRandomIndex_Drop);
+ FINALIZER(createPkIndex_Drop);
+ FINALIZER(runClearTable);
+}
+NDBT_TESTSUITE_END(testIndex);
+
+int main(int argc, const char** argv){
+ return testIndex.execute(argc, argv);
+}
+
+
diff --git a/ndb/test/ndbapi/testInterpreter/Makefile b/ndb/test/ndbapi/testInterpreter/Makefile
new file mode 100644
index 00000000000..e84287a1b16
--- /dev/null
+++ b/ndb/test/ndbapi/testInterpreter/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testInterpreter
+
+SOURCES = testInterpreter.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testInterpreter/testInterpreter.cpp b/ndb/test/ndbapi/testInterpreter/testInterpreter.cpp
new file mode 100644
index 00000000000..9c584d6f581
--- /dev/null
+++ b/ndb/test/ndbapi/testInterpreter/testInterpreter.cpp
@@ -0,0 +1,231 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <NdbRestarts.hpp>
+#include <Vector.hpp>
+#include <random.h>
+#include <NdbTick.h>
+
+
+#define CHECK(b) if (!(b)) { \
+ ndbout << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ continue; }
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int batchSize = ctx->getProperty("BatchSize", 1);
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.pkDelRecords(GETNDB(step), records, batchSize) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runTestIncValue64(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ // NDBT_Table* pTab = ctx->getTab();
+ //Ndb* pNdb = GETNDB(step);
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.pkInterpretedUpdateRecords(GETNDB(step),
+ records) != 0){
+ return NDBT_FAILED;
+ }
+
+ // Verify the update
+ if (hugoTrans.pkReadRecords(GETNDB(step),
+ records) != 0){
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+
+}
+
+int runTestIncValue32(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ const NdbDictionary::Table * pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+
+
+ NdbConnection* pTrans = pNdb->startTransaction();
+ if (pTrans == NULL){
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pTrans->getNdbOperation(pTab->getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int check = pOp->interpretedUpdateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+
+ // Primary keys
+ Uint32 pkVal = 1;
+ check = pOp->equal("KOL1", pkVal );
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Attributes
+
+ // Update column
+ Uint32 valToIncWith = 1;
+ check = pOp->incValue("KOL2", valToIncWith);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbRecAttr* valueRec = pOp->getValue("KOL2");
+ if( valueRec == NULL ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ Uint32 value = valueRec->u_32_value();
+
+ pNdb->closeTransaction(pTrans);
+
+
+ return NDBT_OK;
+}
+
+
+NDBT_TESTSUITE(testInterpreter);
+TESTCASE("IncValue32",
+ "Test incValue for 32 bit integer\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runTestIncValue32);
+ FINALIZER(runClearTable);
+}
+TESTCASE("IncValue64",
+ "Test incValue for 64 bit integer\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runTestIncValue64);
+ FINALIZER(runClearTable);
+}
+#if 0
+TESTCASE("MaxTransactions",
+ "Start transactions until no more can be created\n"){
+ INITIALIZER(runTestMaxTransaction);
+}
+TESTCASE("MaxOperations",
+ "Get operations until no more can be created\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runTestMaxOperations);
+ FINALIZER(runClearTable);
+}
+TESTCASE("MaxGetValue",
+ "Call getValue loads of time\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runTestGetValue);
+ FINALIZER(runClearTable);
+}
+TESTCASE("MaxEqual",
+ "Call equal loads of time\n"){
+ INITIALIZER(runTestEqual);
+}
+TESTCASE("DeleteNdb",
+ "Make sure that a deleted Ndb object is properly deleted\n"
+ "and removed from transporter\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runTestDeleteNdb);
+ FINALIZER(runClearTable);
+}
+TESTCASE("WaitUntilReady",
+ "Make sure you get an error message when calling waitUntilReady\n"
+ "without an init'ed Ndb\n"){
+ INITIALIZER(runTestWaitUntilReady);
+}
+TESTCASE("GetOperationNoTab",
+ "Call getNdbOperation on a table that does not exist\n"){
+ INITIALIZER(runGetNdbOperationNoTab);
+}
+TESTCASE("MissingOperation",
+ "Missing operation request(insertTuple) should give an error code\n"){
+ INITIALIZER(runMissingOperation);
+}
+TESTCASE("GetValueInUpdate",
+ "Test that it's not possible to perform getValue in an update\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runGetValueInUpdate);
+ FINALIZER(runClearTable);
+}
+TESTCASE("UpdateWithoutKeys",
+ "Test that it's not possible to perform update without setting\n"
+ "PKs"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runUpdateWithoutKeys);
+ FINALIZER(runClearTable);
+}
+TESTCASE("UpdateWithoutValues",
+ "Test that it's not possible to perform update without setValues\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runUpdateWithoutValues);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NdbErrorOperation",
+ "Test that NdbErrorOperation is properly set"){
+ INITIALIZER(runCheckGetNdbErrorOperation);
+}
+#endif
+NDBT_TESTSUITE_END(testInterpreter);
+
+int main(int argc, const char** argv){
+ // TABLE("T1");
+ return testInterpreter.execute(argc, argv);
+}
+
+
diff --git a/ndb/test/ndbapi/testMgm/Makefile b/ndb/test/ndbapi/testMgm/Makefile
new file mode 100644
index 00000000000..be50d3dae7e
--- /dev/null
+++ b/ndb/test/ndbapi/testMgm/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testMgm
+
+SOURCES = testMgm.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testMgm/testMgm.cpp b/ndb/test/ndbapi/testMgm/testMgm.cpp
new file mode 100644
index 00000000000..54ca64c6e81
--- /dev/null
+++ b/ndb/test/ndbapi/testMgm/testMgm.cpp
@@ -0,0 +1,185 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <Vector.hpp>
+#include <random.h>
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+int create_index_on_pk(Ndb* pNdb, const char* tabName){
+ int result = NDBT_OK;
+
+ const NdbDictionary::Table * tab = NDBT_Table::discoverTableFromDb(pNdb,
+ tabName);
+
+ // Create index
+ const char* idxName = "IDX_ON_PK";
+ ndbout << "Create: " <<idxName << "( ";
+ NdbDictionary::Index pIdx(idxName);
+ pIdx.setTable(tabName);
+ pIdx.setType(NdbDictionary::Index::UniqueHashIndex);
+ for (int c = 0; c< tab->getNoOfPrimaryKeys(); c++){
+ pIdx.addIndexColumn(tab->getPrimaryKey(c));
+ ndbout << tab->getPrimaryKey(c)<<" ";
+ }
+
+ ndbout << ") ";
+ if (pNdb->getDictionary()->createIndex(pIdx) != 0){
+ ndbout << "FAILED!" << endl;
+ const NdbError err = pNdb->getDictionary()->getNdbError();
+ ERR(err);
+ result = NDBT_FAILED;
+ } else {
+ ndbout << "OK!" << endl;
+ }
+ delete tab;
+ return result;
+}
+
+int drop_index_on_pk(Ndb* pNdb, const char* tabName){
+ int result = NDBT_OK;
+ const char* idxName = "IDX_ON_PK";
+ ndbout << "Drop: " << idxName;
+ if (pNdb->getDictionary()->dropIndex(idxName, tabName) != 0){
+ ndbout << "FAILED!" << endl;
+ const NdbError err = pNdb->getDictionary()->getNdbError();
+ ERR(err);
+ result = NDBT_FAILED;
+ } else {
+ ndbout << "OK!" << endl;
+ }
+ return result;
+}
+
+
+#define CHECK(b) if (!(b)) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ continue; }
+
+int runTestSingleUserMode(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ Ndb* pNdb = GETNDB(step);
+ NdbRestarter restarter;
+ char tabName[255];
+ strncpy(tabName, ctx->getTab()->getName(), 255);
+ ndbout << "tabName="<<tabName<<endl;
+
+ int i = 0;
+ int count;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ UtilTransactions utilTrans(*ctx->getTab());
+ while (i<loops && result == NDBT_OK) {
+ g_info << i << ": ";
+ int timeout = 120;
+ // Test that the single user mode api can do everything
+ CHECK(restarter.enterSingleUserMode(pNdb->getNodeId()) == 0);
+ CHECK(restarter.waitClusterSingleUser(timeout) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records, 128) == 0);
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(hugoTrans.scanReadRecords(pNdb, records/2, 0, 64) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+ CHECK(utilTrans.clearTable(pNdb, records/2) == 0);
+ CHECK(restarter.exitSingleUserMode() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+
+ // Test create index in single user mode
+ CHECK(restarter.enterSingleUserMode(pNdb->getNodeId()) == 0);
+ CHECK(restarter.waitClusterSingleUser(timeout) == 0);
+ CHECK(create_index_on_pk(pNdb, tabName) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records, 128) == 0);
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(drop_index_on_pk(pNdb, tabName) == 0);
+ CHECK(restarter.exitSingleUserMode() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+
+ // Test recreate index in single user mode
+ CHECK(create_index_on_pk(pNdb, tabName) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records, 128) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(restarter.enterSingleUserMode(pNdb->getNodeId()) == 0);
+ CHECK(restarter.waitClusterSingleUser(timeout) == 0);
+ CHECK(drop_index_on_pk(pNdb, tabName) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(create_index_on_pk(pNdb, tabName) == 0);
+ CHECK(restarter.exitSingleUserMode() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(drop_index_on_pk(pNdb, tabName) == 0);
+
+ CHECK(utilTrans.clearTable(GETNDB(step), records) == 0);
+
+ ndbout << "Restarting cluster" << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ i++;
+
+ }
+ return result;
+}
+
+
+
+NDBT_TESTSUITE(testMgm);
+TESTCASE("SingleUserMode",
+ "Test single user mode"){
+ INITIALIZER(runTestSingleUserMode);
+ FINALIZER(runClearTable);
+}
+NDBT_TESTSUITE_END(testMgm);
+
+int main(int argc, const char** argv){
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ return testMgm.execute(argc, argv);
+}
+
diff --git a/ndb/test/ndbapi/testNdbApi/Makefile b/ndb/test/ndbapi/testNdbApi/Makefile
new file mode 100644
index 00000000000..3bb3cba427e
--- /dev/null
+++ b/ndb/test/ndbapi/testNdbApi/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testNdbApi
+
+SOURCES = testNdbApi.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testNdbApi/testNdbApi.cpp b/ndb/test/ndbapi/testNdbApi/testNdbApi.cpp
new file mode 100644
index 00000000000..c0e262f590f
--- /dev/null
+++ b/ndb/test/ndbapi/testNdbApi/testNdbApi.cpp
@@ -0,0 +1,1013 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <NdbRestarts.hpp>
+#include <Vector.hpp>
+#include <random.h>
+#include <NdbTick.h>
+#include <AttrType.hpp>
+
+
+#define CHECK(b) if (!(b)) { \
+ ndbout << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ continue; }
+
+#define CHECKE(b) if (!(b)) { \
+ errors++; \
+ ndbout << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ continue; }
+
+
+int runTestMaxNdb(NDBT_Context* ctx, NDBT_Step* step){
+ Uint32 loops = ctx->getNumLoops();
+ Uint32 l = 0;
+ int oldi = 0;
+ int result = NDBT_OK;
+
+ while (l < loops && result == NDBT_OK){
+ ndbout_c("loop %d", l + 1);
+ int errors = 0;
+ int maxErrors = 5;
+
+ Vector<Ndb*> ndbVector;
+ int i = 0;
+ int init = 0;
+ do {
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ errors++;
+ continue;
+
+ }
+ i++;
+
+ ndbVector.push_back(pNdb);
+
+ if (pNdb->init()){
+ ERR(pNdb->getNdbError());
+ errors++;
+ continue;
+ }
+
+ init++;
+
+ } while (errors == 0);
+
+ ndbout << i << " ndb objects created" << endl;
+
+ if (l > 0 && i != oldi && init != MAX_NO_THREADS){
+ ndbout << l << ": not as manyNdb objects created" << endl
+ << i << " != " << oldi << endl;
+ result = NDBT_FAILED;
+ }
+
+ oldi = i;
+
+
+ for(size_t i = 0; i < ndbVector.size(); i++){
+ delete ndbVector[i];
+ if(((i+1) % 250) == 0){
+ ndbout << "Deleted " << (Uint64) i << " ndb objects " << endl;
+ }
+ }
+ ndbVector.clear();
+
+ l++;
+ }
+
+ return result;
+}
+
+int runTestMaxTransaction(NDBT_Context* ctx, NDBT_Step* step){
+ Uint32 loops = ctx->getNumLoops();
+ Uint32 l = 0;
+ int oldi = 0;
+ int result = NDBT_OK;
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init(2048)){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ while (l < loops && result == NDBT_OK){
+ int errors = 0;
+ int maxErrors = 5;
+
+ Vector<NdbConnection*> conVector;
+
+
+ int i = 0;
+ do {
+
+ NdbConnection* pCon;
+
+ int type = i%4;
+ switch (type){
+ case 0:
+ pCon = pNdb->startTransaction();
+ break;
+ case 1:
+ pCon = pNdb->startTransaction(2,
+ "DATA",
+ 4);
+ break;
+ case 2:
+ pCon = pNdb->startTransactionDGroup(1,
+ "TEST",
+ 0);
+ break;
+ case 3:
+ pCon = pNdb->startTransactionDGroup(2,
+ "TEST",
+ 1);
+ break;
+
+ default:
+ abort();
+ }
+
+ if (pCon == NULL){
+ ERR(pNdb->getNdbError());
+ errors++;
+ continue;
+ }
+
+ conVector.push_back(pCon);
+
+ i++;
+ } while (errors < maxErrors);
+
+ ndbout << i << " connections created" << endl;
+
+ if (l > 0 && i != oldi){
+ ndbout << l << ": not as many transactions created" << endl
+ << i << " != " << oldi << endl;
+ result = NDBT_FAILED;
+ }
+
+ oldi = i;
+
+
+ for(size_t i = 0; i < conVector.size(); i++){
+ pNdb->closeTransaction(conVector[i]);
+ }
+ conVector.clear();
+ l++;
+
+ }
+
+ // BONUS Test closeTransaction with null trans
+ pNdb->closeTransaction(NULL);
+
+ delete pNdb;
+
+
+ return result;
+}
+
+int runTestMaxOperations(NDBT_Context* ctx, NDBT_Step* step){
+ Uint32 l = 1;
+ int result = NDBT_OK;
+ int maxOpsLimit = 1;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init(2048)){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ HugoOperations hugoOps(*pTab);
+
+ bool endTest = false;
+ while (!endTest && result == NDBT_OK){
+ int errors = 0;
+ int maxErrors = 5;
+
+ maxOpsLimit = l*1000;
+
+ if (hugoOps.startTransaction(pNdb) != NDBT_OK){
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ int i = 0;
+ while (errors < maxErrors){
+
+ if(hugoOps.pkReadRecord(pNdb,1, false, 1) != NDBT_OK){
+ errors++;
+ continue;
+ }
+
+ i++;
+
+ if (i >= maxOpsLimit){
+ errors = maxErrors;
+ }
+
+ }
+
+ ndbout << i << " operations used" << endl;
+
+ int execResult = hugoOps.execute_Commit(pNdb);
+ switch(execResult){
+ case NDBT_OK:
+ break;
+ case 233: // Out of operation records in transaction coordinator
+ // OK - end test
+ endTest = true;
+ break;
+ default:
+ result = NDBT_FAILED;
+ break;
+ }
+
+ hugoOps.closeTransaction(pNdb);
+
+ l++;
+
+ }
+
+ delete pNdb;
+
+ return result;
+}
+
+int runTestGetValue(NDBT_Context* ctx, NDBT_Step* step){
+
+ int result = NDBT_OK;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init(2048)){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ HugoOperations hugoOps(*pTab);
+
+ for (int m = 1; m < 100; m++){
+ int errors = 0;
+ int maxErrors = 5;
+
+ NdbConnection* pCon = pNdb->startTransaction();
+ if (pCon == NULL){
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pCon->getNdbOperation(pTab->getName());
+ if (pOp == NULL){
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ if (pOp->readTuple() != 0){
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ for(int a = 0; a<pTab->getNoOfColumns(); a++){
+ if (pTab->getColumn(a)->getPrimaryKey() == true){
+ if(hugoOps.equalForAttr(pOp, a, 1) != 0){
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ int i = 0;
+ int maxLimit = 1000*m;
+ do {
+
+ if (pOp->getValue(pTab->getColumn(1)->getName()) == NULL) {
+ const NdbError err = pCon->getNdbError();
+ ERR(err);
+ if (err.code == 0)
+ result = NDBT_FAILED;
+ errors++;
+ continue;
+ }
+
+ i++;
+
+ } while (errors < maxErrors && i < maxLimit);
+
+ ndbout << i << " getValues called" << endl;
+
+
+ if (pCon->execute(Commit) != 0){
+ const NdbError err = pCon->getNdbError();
+ switch(err.code){
+ case 880: // TUP - Read too much
+ case 823: // TUP - Too much AI
+ case 4257: // NDBAPI - Too much AI
+ // OK errors
+ ERR(pCon->getNdbError());
+ break;
+ default:
+ ERR(pCon->getNdbError());
+ ndbout << "Illegal error" << endl;
+ result= NDBT_FAILED;
+ break;
+ }
+ }
+
+ pNdb->closeTransaction(pCon);
+
+ }// m
+
+
+ delete pNdb;
+
+ return result;
+}
+
+int runTestEqual(NDBT_Context* ctx, NDBT_Step* step){
+ Uint32 loops = ctx->getNumLoops();
+ Uint32 l = 0;
+ int result = NDBT_OK;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init(2048)){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ HugoOperations hugoOps(*pTab);
+
+ while (l < loops){
+ for(int m = 1; m < 10; m++){
+ int errors = 0;
+ int maxErrors = 5;
+
+ NdbConnection* pCon = pNdb->startTransaction();
+ if (pCon == NULL){
+ ndbout << "Could not start transaction" << endl;
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pCon->getNdbOperation(pTab->getName());
+ if (pOp == NULL){
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ if (pOp->readTuple() != 0){
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ int i = 0;
+ int maxLimit = 1000*m;
+ do {
+
+ if ((l%2)!=0){
+ // Forward
+ for(int a = 0; a<pTab->getNoOfColumns(); a++){
+ if (pTab->getColumn(a)->getPrimaryKey() == true){
+ if(hugoOps.equalForAttr(pOp, a, 1) != 0){
+ const NdbError err = pCon->getNdbError();
+ ERR(err);
+ if (err.code == 0)
+ result = NDBT_FAILED;
+ errors++;
+ }
+ }
+ }
+ } else {
+ // Backward
+ for(int a = pTab->getNoOfColumns()-1; a>=0; a--){
+ if (pTab->getColumn(a)->getPrimaryKey() == true){
+ if(hugoOps.equalForAttr(pOp, a, 1) != 0){
+ const NdbError err = pCon->getNdbError();
+ ERR(err);
+ if (err.code == 0)
+ result = NDBT_FAILED;
+ errors++;
+ }
+ }
+ }
+ }
+
+ i++;
+
+ } while (errors < maxErrors && i < maxLimit);
+
+ if (pOp->getValue(pTab->getColumn(1)->getName()) == NULL) {
+ const NdbError err = pCon->getNdbError();
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ if (err.code == 4225) {
+ return NDBT_OK;
+ } else {
+ return NDBT_FAILED;
+ }//if
+ }
+
+ ndbout << i << " equal called" << endl;
+
+
+ int check = pCon->execute(Commit);
+ if (check != 0){
+ ERR(pCon->getNdbError());
+ }
+
+ pNdb->closeTransaction(pCon);
+
+ }// m
+ l++;
+
+ }// l
+
+ delete pNdb;
+ return result;
+}
+
+int runTestDeleteNdb(NDBT_Context* ctx, NDBT_Step* step){
+ Uint32 loops = ctx->getNumLoops();
+ Uint32 l = 0;
+ int result = NDBT_OK;
+ NdbRestarts restarts;
+ Vector<Ndb*> ndbVector;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ HugoTransactions hugoTrans(*pTab);
+ int records = ctx->getNumRecords();
+
+ while (l < loops && result == NDBT_OK){
+
+ // Create 5 ndb objects
+ for( int i = 0; i < 5; i++){
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ result = NDBT_FAILED;
+ goto end_test;
+ }
+ ndbVector.push_back(pNdb);
+
+ if (pNdb->init()){
+ ERR(pNdb->getNdbError());
+ result = NDBT_FAILED;
+ goto end_test;
+ }
+ if (pNdb->waitUntilReady() != 0){
+ ERR(pNdb->getNdbError());
+ result = NDBT_FAILED;
+ goto end_test;
+ }
+ if (hugoTrans.pkReadRecords(pNdb, records) != 0){
+ result = NDBT_FAILED;
+ goto end_test;
+ }
+ }
+
+ if ((l % 2) == 0){
+ // Restart random node
+ ndbout << "Restart random node " << endl;
+ if(restarts.executeRestart("RestartRandomNodeAbort", 120) != 0){
+ g_err << "Failed to executeRestart(RestartRandomNode)"<<endl;
+ result = NDBT_FAILED;
+ goto end_test;
+ }
+ } else {
+ // Restart all nodes
+ ndbout << "Restart all nodes " << endl;
+ if(restarts.executeRestart("RestartAllNodesAbort", 120) != 0){
+ g_err << "Failed to executeRestart(RestartAllNodes)"<<endl;
+ result = NDBT_FAILED;
+ goto end_test;
+ }
+ }
+
+ // Delete the ndb objects
+ for(size_t i = 0; i < ndbVector.size(); i++)
+ delete ndbVector[i];
+ ndbVector.clear();
+ l++;
+ }
+
+
+ end_test:
+
+ for(size_t i = 0; i < ndbVector.size(); i++)
+ delete ndbVector[i];
+ ndbVector.clear();
+
+ return result;
+}
+
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runTestWaitUntilReady(NDBT_Context* ctx, NDBT_Step* step){
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+
+ // Forget about calling pNdb->init();
+
+ if (pNdb->waitUntilReady() == 0){
+ ndbout << "waitUntilReady returned OK" << endl;
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+ const NdbError err = pNdb->getNdbError();
+ delete pNdb;
+
+ ERR(err);
+ if (err.code != 4256)
+ return NDBT_FAILED;
+
+ return NDBT_OK;
+}
+
+int runGetNdbOperationNoTab(NDBT_Context* ctx, NDBT_Step* step){
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init()){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbConnection* pCon = pNdb->startTransaction();
+ if (pCon == NULL){
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ // Call getNdbOperation on an unknown table
+ NdbOperation* pOp = pCon->getNdbOperation("HUPP76");
+ if (pOp == NULL){
+ NdbError err = pCon->getNdbError();
+ ERR(err);
+ if (err.code == 0){
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+ }
+
+ pNdb->closeTransaction(pCon);
+
+ delete pNdb;
+
+ return NDBT_OK;
+}
+
+int runMissingOperation(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init()){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbConnection* pCon = pNdb->startTransaction();
+ if (pCon == NULL){
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pCon->getNdbOperation(pTab->getName());
+ if (pOp == NULL){
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ // Forget about calling pOp->insertTuple();
+
+ // Call getValue should not work
+ if (pOp->getValue(pTab->getColumn(1)->getName()) == NULL) {
+ const NdbError err = pCon->getNdbError();
+ ERR(err);
+ if (err.code == 0){
+ ndbout << "hupp" << endl;
+ result = NDBT_FAILED;
+ }
+ } else {
+ ndbout << "hupp2" << endl;
+ result = NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+
+ return result;
+}
+
+int runGetValueInUpdate(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init()){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbConnection* pCon = pNdb->startTransaction();
+ if (pCon == NULL){
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pCon->getNdbOperation(pTab->getName());
+ if (pOp == NULL){
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ if (pOp->updateTuple() != 0){
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ // Call getValue should not work
+ if (pOp->getValue(pTab->getColumn(1)->getName()) == NULL) {
+ // It didn't work
+ const NdbError err = pCon->getNdbError();
+ ERR(err);
+ if (err.code == 0){
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+ } else {
+ // It worked, not good!
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ int check = pCon->execute(Commit);
+ if (check != 0){
+ ERR(pCon->getNdbError());
+ }
+
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+
+ return NDBT_OK;
+}
+
+int runUpdateWithoutValues(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ HugoOperations hugoOps(*pTab);
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init()){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbConnection* pCon = pNdb->startTransaction();
+ if (pCon == NULL){
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pCon->getNdbOperation(pTab->getName());
+ if (pOp == NULL){
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ if (pOp->updateTuple() != 0){
+ pNdb->closeTransaction(pCon);
+ ERR(pOp->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ for(int a = 0; a<pTab->getNoOfColumns(); a++){
+ if (pTab->getColumn(a)->getPrimaryKey() == true){
+ if(hugoOps.equalForAttr(pOp, a, 1) != 0){
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Dont' call any setValues
+
+ // Execute should not work
+ int check = pCon->execute(Commit);
+ if (check == 0){
+ ndbout << "execute worked" << endl;
+ result = NDBT_FAILED;
+ } else {
+ ERR(pCon->getNdbError());
+ }
+
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+
+ return result;
+}
+
+int runUpdateWithoutKeys(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init()){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbConnection* pCon = pNdb->startTransaction();
+ if (pCon == NULL){
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pCon->getNdbOperation(pTab->getName());
+ if (pOp == NULL){
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ if (pOp->updateTuple() != 0){
+ pNdb->closeTransaction(pCon);
+ ERR(pOp->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ // Dont' call any equal or setValues
+
+ // Execute should not work
+ int check = pCon->execute(Commit);
+ if (check == 0){
+ ndbout << "execute worked" << endl;
+ result = NDBT_FAILED;
+ } else {
+ ERR(pCon->getNdbError());
+ }
+
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+
+ return result;
+}
+
+int runCheckGetNdbErrorOperation(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+
+ Ndb* pNdb = new Ndb("TEST_DB");
+ if (pNdb == NULL){
+ ndbout << "pNdb == NULL" << endl;
+ return NDBT_FAILED;
+ }
+ if (pNdb->init(2048)){
+ ERR(pNdb->getNdbError());
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ HugoOperations hugoOps(*pTab);
+
+
+ NdbConnection* pCon = pNdb->startTransaction();
+ if (pCon == NULL){
+ ndbout << "Could not start transaction" << endl;
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pCon->getNdbOperation(pTab->getName());
+ if (pOp == NULL){
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ delete pNdb;
+ return NDBT_FAILED;
+ }
+
+ // Dont call readTuple here
+ // That's the error!
+
+ for(int a = 0; a<pTab->getNoOfColumns(); a++){
+ if (pTab->getColumn(a)->getPrimaryKey() == true){
+ if(hugoOps.equalForAttr(pOp, a, 1) != 0){
+ // An error has occured, check that
+ // it's possible to get the NdbErrorOperation
+ const NdbError err = pCon->getNdbError();
+ ERR(err);
+ if (err.code == 0)
+ result = NDBT_FAILED;
+
+ NdbOperation* pOp2 = pCon->getNdbErrorOperation();
+ if (pOp2 == NULL)
+ result = NDBT_FAILED;
+ else {
+ const NdbError err2 = pOp2->getNdbError();
+ ERR(err2);
+ if (err.code == 0)
+ result = NDBT_FAILED;
+ }
+ }
+ }
+ }
+
+ pNdb->closeTransaction(pCon);
+
+ delete pNdb;
+ return result;
+}
+
+
+NDBT_TESTSUITE(testNdbApi);
+TESTCASE("MaxNdb",
+ "Create Ndb objects until no more can be created\n"){
+ INITIALIZER(runTestMaxNdb);
+}
+TESTCASE("MaxTransactions",
+ "Start transactions until no more can be created\n"){
+ INITIALIZER(runTestMaxTransaction);
+}
+TESTCASE("MaxOperations",
+ "Get operations until no more can be created\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runTestMaxOperations);
+ FINALIZER(runClearTable);
+}
+TESTCASE("MaxGetValue",
+ "Call getValue loads of time\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runTestGetValue);
+ FINALIZER(runClearTable);
+}
+TESTCASE("MaxEqual",
+ "Call equal loads of time\n"){
+ INITIALIZER(runTestEqual);
+}
+TESTCASE("DeleteNdb",
+ "Make sure that a deleted Ndb object is properly deleted\n"
+ "and removed from transporter\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runTestDeleteNdb);
+ FINALIZER(runClearTable);
+}
+TESTCASE("WaitUntilReady",
+ "Make sure you get an error message when calling waitUntilReady\n"
+ "without an init'ed Ndb\n"){
+ INITIALIZER(runTestWaitUntilReady);
+}
+TESTCASE("GetOperationNoTab",
+ "Call getNdbOperation on a table that does not exist\n"){
+ INITIALIZER(runGetNdbOperationNoTab);
+}
+TESTCASE("MissingOperation",
+ "Missing operation request(insertTuple) should give an error code\n"){
+ INITIALIZER(runMissingOperation);
+}
+TESTCASE("GetValueInUpdate",
+ "Test that it's not possible to perform getValue in an update\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runGetValueInUpdate);
+ FINALIZER(runClearTable);
+}
+TESTCASE("UpdateWithoutKeys",
+ "Test that it's not possible to perform update without setting\n"
+ "PKs"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runUpdateWithoutKeys);
+ FINALIZER(runClearTable);
+}
+TESTCASE("UpdateWithoutValues",
+ "Test that it's not possible to perform update without setValues\n"){
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runUpdateWithoutValues);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NdbErrorOperation",
+ "Test that NdbErrorOperation is properly set"){
+ INITIALIZER(runCheckGetNdbErrorOperation);
+}
+NDBT_TESTSUITE_END(testNdbApi);
+
+int main(int argc, const char** argv){
+ // TABLE("T1");
+ return testNdbApi.execute(argc, argv);
+}
+
+
diff --git a/ndb/test/ndbapi/testNodeRestart/Makefile b/ndb/test/ndbapi/testNodeRestart/Makefile
new file mode 100644
index 00000000000..8c13ab3beb4
--- /dev/null
+++ b/ndb/test/ndbapi/testNodeRestart/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testNodeRestart
+
+SOURCES = testNodeRestart.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testNodeRestart/testNodeRestart.cpp b/ndb/test/ndbapi/testNodeRestart/testNodeRestart.cpp
new file mode 100644
index 00000000000..fd591f04c69
--- /dev/null
+++ b/ndb/test/ndbapi/testNodeRestart/testNodeRestart.cpp
@@ -0,0 +1,449 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <NdbRestarts.hpp>
+#include <Vector.hpp>
+
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runFillTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.fillTable(GETNDB(step)) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runInsertUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int records = ctx->getNumRecords();
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return result;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runClearTableUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int i = 0;
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (utilTrans.clearTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runScanReadUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int records = ctx->getNumRecords();
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (hugoTrans.scanReadRecords(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return result;
+}
+
+int runPkReadUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int records = ctx->getNumRecords();
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (hugoTrans.pkReadRecords(GETNDB(step), records, 128) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return result;
+}
+
+int runPkUpdateUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int records = ctx->getNumRecords();
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (hugoTrans.pkUpdateRecords(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return result;
+}
+
+int runScanUpdateUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int records = ctx->getNumRecords();
+ int parallelism = ctx->getProperty("Parallelism", 1);
+ int abort = ctx->getProperty("AbortProb", (Uint32)0);
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (hugoTrans.scanUpdateRecords(GETNDB(step), records, abort,
+ parallelism) == NDBT_FAILED){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return result;
+}
+
+int runScanReadVerify(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ if (hugoTrans.scanReadRecords(GETNDB(step), records, 0, 64) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runRestarter(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ NdbRestarter restarter;
+ int i = 0;
+ int lastId = 0;
+
+ if (restarter.getNumDbNodes() < 2){
+ ctx->stopTest();
+ return NDBT_OK;
+ }
+
+ if(restarter.waitClusterStarted(60) != 0){
+ g_err << "Cluster failed to start" << endl;
+ return NDBT_FAILED;
+ }
+
+ loops *= restarter.getNumDbNodes();
+ while(i<loops && result != NDBT_FAILED && !ctx->isTestStopped()){
+
+ int id = lastId % restarter.getNumDbNodes();
+ int nodeId = restarter.getDbNodeId(id);
+ ndbout << "Restart node " << nodeId << endl;
+ if(restarter.restartOneDbNode(nodeId) != 0){
+ g_err << "Failed to restartNextDbNode" << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+
+ if(restarter.waitClusterStarted(60) != 0){
+ g_err << "Cluster failed to start" << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+
+ NdbSleep_SecSleep(1);
+
+ lastId++;
+ i++;
+ }
+
+ ctx->stopTest();
+
+ return result;
+}
+
+int runCheckAllNodesStarted(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+
+ if(restarter.waitClusterStarted(1) != 0){
+ g_err << "All nodes was not started " << endl;
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+
+
+int runRestarts(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ NDBT_TestCase* pCase = ctx->getCase();
+ NdbRestarts restarts;
+ int i = 0;
+ int timeout = 240;
+
+ while(i<loops && result != NDBT_FAILED && !ctx->isTestStopped()){
+
+ if(restarts.executeRestart(pCase->getName(), timeout) != 0){
+ g_err << "Failed to executeRestart(" <<pCase->getName() <<")" << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+ i++;
+ }
+ return result;
+}
+
+NDBT_TESTSUITE(testNodeRestart);
+TESTCASE("NoLoad",
+ "Test that one node at a time can be stopped and then restarted "\
+ "when there are no load on the system. Do this loop number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarter);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkRead",
+ "Test that one node at a time can be stopped and then restarted "\
+ "perform pk read while restarting. Do this loop number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarter);
+ STEP(runPkReadUntilStopped);
+ FINALIZER(runClearTable);
+}
+TESTCASE("PkReadPkUpdate",
+ "Test that one node at a time can be stopped and then restarted "\
+ "perform pk read and pk update while restarting. Do this loop number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarter);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkUpdateUntilStopped);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ReadUpdateScan",
+ "Test that one node at a time can be stopped and then restarted "\
+ "perform pk read, pk update and scan reads while restarting. Do this loop number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarter);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkUpdateUntilStopped);
+ STEP(runScanReadUntilStopped);
+ STEP(runScanUpdateUntilStopped);
+ FINALIZER(runClearTable);
+}
+TESTCASE("Terror",
+ "Test that one node at a time can be stopped and then restarted "\
+ "perform all kind of transactions while restarting. Do this loop number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarter);
+ STEP(runPkReadUntilStopped);
+ STEP(runPkUpdateUntilStopped);
+ STEP(runScanReadUntilStopped);
+ STEP(runScanUpdateUntilStopped);
+ STEP(runInsertUntilStopped);
+ STEP(runClearTableUntilStopped);
+ FINALIZER(runClearTable);
+}
+TESTCASE("FullDb",
+ "Test that one node at a time can be stopped and then restarted "\
+ "when db is full. Do this loop number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runFillTable);
+ STEP(runRestarter);
+ FINALIZER(runClearTable);
+}
+TESTCASE("RestartRandomNode",
+ "Test that we can execute the restart RestartRandomNode loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("RestartRandomNodeError",
+ "Test that we can execute the restart RestartRandomNodeError loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("RestartRandomNodeInitial",
+ "Test that we can execute the restart RestartRandomNodeInitial loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("RestartNFDuringNR",
+ "Test that we can execute the restart RestartNFDuringNR loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("RestartMasterNodeError",
+ "Test that we can execute the restart RestartMasterNodeError loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+
+TESTCASE("TwoNodeFailure",
+ "Test that we can execute the restart TwoNodeFailure\n"\
+ "(which is a multiple node failure restart) loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("TwoMasterNodeFailure",
+ "Test that we can execute the restart TwoMasterNodeFailure\n"\
+ "(which is a multiple node failure restart) loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("FiftyPercentFail",
+ "Test that we can execute the restart FiftyPercentFail\n"\
+ "(which is a multiple node failure restart) loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("RestartAllNodes",
+ "Test that we can execute the restart RestartAllNodes\n"\
+ "(which is a system restart) loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("RestartAllNodesAbort",
+ "Test that we can execute the restart RestartAllNodesAbort\n"\
+ "(which is a system restart) loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("RestartAllNodesError9999",
+ "Test that we can execute the restart RestartAllNodesError9999\n"\
+ "(which is a system restart) loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("FiftyPercentStopAndWait",
+ "Test that we can execute the restart FiftyPercentStopAndWait\n"\
+ "(which is a system restart) loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("RestartNodeDuringLCP",
+ "Test that we can execute the restart RestartRandomNode loop\n"\
+ "number of times"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+TESTCASE("StopOnError",
+ "Test StopOnError. A node that has StopOnError set to false "\
+ "should restart automatically when an error occurs"){
+ INITIALIZER(runCheckAllNodesStarted);
+ INITIALIZER(runLoadTable);
+ STEP(runRestarts);
+ FINALIZER(runScanReadVerify);
+ FINALIZER(runClearTable);
+}
+NDBT_TESTSUITE_END(testNodeRestart);
+
+int main(int argc, const char** argv){
+#if 0
+ // It might be interesting to have longer defaults for num
+ // loops in this test
+ // Just performing 100 node restarts would not be enough?
+ // We can have initialisers in the NDBT_Testcase class like
+ // this...
+ testNodeRestart.setDefaultLoops(1000);
+#endif
+ return testNodeRestart.execute(argc, argv);
+}
+
diff --git a/ndb/test/ndbapi/testOIBasic/Makefile b/ndb/test/ndbapi/testOIBasic/Makefile
new file mode 100644
index 00000000000..1bbbcf1d17e
--- /dev/null
+++ b/ndb/test/ndbapi/testOIBasic/Makefile
@@ -0,0 +1,13 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testOIBasic
+
+SOURCES = testOIBasic.cpp
+
+ifeq ($(NDB_COMPILER),GCC)
+CCFLAGS_WARNINGS += -Wno-unused -Wformat
+endif
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testOIBasic/testOIBasic.cpp b/ndb/test/ndbapi/testOIBasic/testOIBasic.cpp
new file mode 100644
index 00000000000..68fa6ec5474
--- /dev/null
+++ b/ndb/test/ndbapi/testOIBasic/testOIBasic.cpp
@@ -0,0 +1,2772 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*
+ * testOIBasic - ordered index test
+ */
+
+#include <stdlib.h>
+#include <unistd.h>
+#include <string.h>
+#include <stdarg.h>
+#include <assert.h>
+#include <new>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NdbTest.hpp>
+#include <NdbStdio.h>
+#include <NdbMutex.h>
+#include <NdbCondition.h>
+#include <NdbThread.h>
+#include <NdbTick.h>
+
+// options
+
+struct Opt {
+ // common options
+ const char* m_case;
+ bool m_core;
+ bool m_dups;
+ NdbDictionary::Object::FragmentType m_fragtype;
+ const char* m_index;
+ unsigned m_loop;
+ unsigned m_rows;
+ unsigned m_scanrd;
+ unsigned m_scanex;
+ unsigned m_seed;
+ unsigned m_subloop;
+ const char* m_table;
+ unsigned m_threads;
+ unsigned m_v;
+ Opt() :
+ m_case(0),
+ m_core(false),
+ m_dups(false),
+ m_fragtype(NdbDictionary::Object::FragUndefined),
+ m_index(0),
+ m_loop(1),
+ m_rows(1000),
+ m_scanrd(240),
+ m_scanex(240),
+ m_seed(1),
+ m_subloop(4),
+ m_table(0),
+ m_threads(4),
+ m_v(1) {
+ }
+};
+
+static Opt g_opt;
+
+static void printcases();
+static void printtables();
+
+static void
+printhelp()
+{
+ Opt d;
+ ndbout
+ << "usage: testOIbasic [options]" << endl
+ << " -case abc only given test cases (letters a-z)" << endl
+ << " -core core dump on error [" << d.m_core << "]" << endl
+ << " -dups allow duplicate tuples from index scan [" << d.m_dups << "]" << endl
+ << " -fragtype T fragment type single/small/medium/large" << endl
+ << " -index xyz only given index numbers (digits 1-9)" << endl
+ << " -loop N loop count full suite forever=0 [" << d.m_loop << "]" << endl
+ << " -rows N rows per thread [" << d.m_rows << "]" << endl
+ << " -scanrd N scan read parallelism [" << d.m_scanrd << "]" << endl
+ << " -scanex N scan exclusive parallelism [" << d.m_scanex << "]" << endl
+ << " -seed N srandom seed [" << d.m_seed << "]" << endl
+ << " -subloop N subtest loop count [" << d.m_subloop << "]" << endl
+ << " -table xyz only given table numbers (digits 1-9)" << endl
+ << " -threads N number of threads [" << d.m_threads << "]" << endl
+ << " -vN verbosity [" << d.m_v << "]" << endl
+ << " -h or -help print this help text" << endl
+ ;
+ printcases();
+ printtables();
+}
+
+// log and error macros
+
+static NdbMutex ndbout_mutex = NDB_MUTEX_INITIALIZER;
+
+static unsigned getthrno();
+
+static const char*
+getthrstr()
+{
+ static char buf[20];
+ unsigned n = getthrno();
+ if (n == (unsigned)-1)
+ strcpy(buf, "");
+ else {
+ unsigned m =
+ g_opt.m_threads < 10 ? 1 :
+ g_opt.m_threads < 100 ? 2 : 3;
+ sprintf(buf, "[%0*u] ", m, n);
+ }
+ return buf;
+}
+
+#define LLN(n, s) \
+ do { \
+ if ((n) > g_opt.m_v) break; \
+ NdbMutex_Lock(&ndbout_mutex); \
+ ndbout << getthrstr() << s << endl; \
+ NdbMutex_Unlock(&ndbout_mutex); \
+ } while(0)
+
+#define LL0(s) LLN(0, s)
+#define LL1(s) LLN(1, s)
+#define LL2(s) LLN(2, s)
+#define LL3(s) LLN(3, s)
+#define LL4(s) LLN(4, s)
+#define LL5(s) LLN(5, s)
+
+// following check a condition and return -1 on failure
+
+#undef CHK // simple check
+#undef CHKTRY // execute action (try-catch) on failure
+#undef CHKMSG // print extra message on failure
+#undef CHKCON // print NDB API errors on failure
+
+#define CHK(x) CHKTRY(x, ;)
+
+#define CHKTRY(x, act) \
+ do { \
+ if (x) break; \
+ LL0("line " << __LINE__ << ": " << #x << " failed"); \
+ if (g_opt.m_core) abort(); \
+ act; \
+ return -1; \
+ } while (0)
+
+#define CHKMSG(x, msg) \
+ do { \
+ if (x) break; \
+ LL0("line " << __LINE__ << ": " << #x << " failed: " << msg); \
+ if (g_opt.m_core) abort(); \
+ return -1; \
+ } while (0)
+
+#define CHKCON(x, con) \
+ do { \
+ if (x) break; \
+ LL0("line " << __LINE__ << ": " << #x << " failed"); \
+ (con).printerror(ndbout); \
+ if (g_opt.m_core) abort(); \
+ return -1; \
+ } while (0)
+
+// method parameters base class
+
+class Thr;
+class Con;
+class Tab;
+class Set;
+
+struct Par : public Opt {
+ unsigned m_no;
+ Con* m_con;
+ Con& con() const { assert(m_con != 0); return *m_con; }
+ const Tab* m_tab;
+ const Tab& tab() const { assert(m_tab != 0); return *m_tab; }
+ Set* m_set;
+ Set& set() const { assert(m_set != 0); return *m_set; }
+ unsigned m_totrows;
+ unsigned m_batch;
+ // value calculation
+ unsigned m_pctnull;
+ unsigned m_range;
+ unsigned m_pctrange;
+ // do verify after read
+ bool m_verify;
+ // timer location
+ Par(const Opt& opt) :
+ Opt(opt),
+ m_no(0),
+ m_con(0),
+ m_tab(0),
+ m_set(0),
+ m_totrows(m_threads * m_rows),
+ m_batch(32),
+ m_pctnull(10),
+ m_range(m_rows),
+ m_pctrange(0),
+ m_verify(false) {
+ }
+};
+
+static bool
+usetable(unsigned i)
+{
+ return g_opt.m_table == 0 || strchr(g_opt.m_table, '1' + i) != 0;
+}
+
+static bool
+useindex(unsigned i)
+{
+ return g_opt.m_index == 0 || strchr(g_opt.m_index, '1' + i) != 0;
+}
+
+static unsigned
+thrrow(Par par, unsigned j)
+{
+ return par.m_threads * j + par.m_no;
+}
+
+static bool
+isthrrow(Par par, unsigned i)
+{
+ return i % par.m_threads == par.m_no;
+}
+
+// timer
+
+struct Tmr {
+ void clr();
+ void on();
+ void off(unsigned cnt = 0);
+ const char* time();
+ const char* over(const Tmr& t1);
+ NDB_TICKS m_on;
+ unsigned m_ms;
+ unsigned m_cnt;
+ char m_time[100];
+ char m_over[100];
+ Tmr() { clr(); }
+};
+
+void
+Tmr::clr()
+{
+ m_on = m_ms = m_cnt = m_time[0] = m_over[0] = 0;
+}
+
+void
+Tmr::on()
+{
+ assert(m_on == 0);
+ m_on = NdbTick_CurrentMillisecond();
+}
+
+void
+Tmr::off(unsigned cnt)
+{
+ NDB_TICKS off = NdbTick_CurrentMillisecond();
+ assert(m_on != 0 && off >= m_on);
+ m_ms += off - m_on;
+ m_cnt += cnt;
+ m_on = 0;
+}
+
+const char*
+Tmr::time()
+{
+ if (m_cnt == 0) {
+ sprintf(m_time, "%u ms", m_ms);
+ } else {
+ sprintf(m_time, "%u ms per %u ( %u ms per 1000 )", m_ms, m_cnt, (1000 * m_ms) / m_cnt);
+ }
+ return m_time;
+}
+
+const char*
+Tmr::over(const Tmr& t1)
+{
+ if (0 < t1.m_ms && t1.m_ms < m_ms) {
+ sprintf(m_over, "%u pct", (100 * (m_ms - t1.m_ms)) / t1.m_ms);
+ } else {
+ sprintf(m_over, "[cannot measure]");
+ }
+ return m_over;
+}
+
+// tables and indexes
+
+// Col - table column
+
+struct Col {
+ unsigned m_num;
+ const char* m_name;
+ bool m_pk;
+ NdbDictionary::Column::Type m_type;
+ unsigned m_length;
+ bool m_nullable;
+ void verify(const void* addr) const;
+};
+
+void
+Col::verify(const void* addr) const
+{
+ switch (m_type) {
+ case NdbDictionary::Column::Unsigned:
+ break;
+ case NdbDictionary::Column::Varchar:
+ {
+ const unsigned char* p = (const unsigned char*)addr;
+ unsigned n = (p[0] << 8) | p[1];
+ assert(n <= m_length);
+ for (unsigned i = 0; i < n; i++) {
+ assert(p[2 + i] != 0);
+ }
+ for (unsigned i = n; i < m_length; i++) {
+ assert(p[2 + i] == 0);
+ }
+ }
+ break;
+ default:
+ assert(false);
+ break;
+ }
+}
+
+static NdbOut&
+operator<<(NdbOut& out, const Col& col)
+{
+ out << "col " << col.m_num;
+ out << " " << col.m_name;
+ switch (col.m_type) {
+ case NdbDictionary::Column::Unsigned:
+ out << " unsigned";
+ break;
+ case NdbDictionary::Column::Varchar:
+ out << " varchar(" << col.m_length << ")";
+ break;
+ default:
+ out << "type" << (int)col.m_type;
+ assert(false);
+ break;
+ }
+ out << (col.m_pk ? " pk" : "");
+ out << (col.m_nullable ? " nullable" : "");
+ return out;
+}
+
+// ICol - index column
+
+struct ICol {
+ unsigned m_num;
+ struct Col m_col;
+};
+
+// ITab - index
+
+struct ITab {
+ const char* m_name;
+ unsigned m_icols;
+ const ICol* m_icol;
+};
+
+static NdbOut&
+operator<<(NdbOut& out, const ITab& itab)
+{
+ out << "itab " << itab.m_name << " " << itab.m_icols;
+ for (unsigned k = 0; k < itab.m_icols; k++) {
+ out << endl;
+ out << "icol " << k << " " << itab.m_icol[k].m_col;
+ }
+ return out;
+}
+
+// Tab - table
+
+struct Tab {
+ const char* m_name;
+ unsigned m_cols;
+ const Col* m_col;
+ unsigned m_itabs;
+ const ITab* m_itab;
+};
+
+static NdbOut&
+operator<<(NdbOut& out, const Tab& tab)
+{
+ out << "tab " << tab.m_name << " " << tab.m_cols;
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ out << endl;
+ out << tab.m_col[k];
+ }
+ for (unsigned i = 0; i < tab.m_itabs; i++) {
+ if (! useindex(i))
+ continue;
+ out << endl;
+ out << tab.m_itab[i];
+ }
+ return out;
+}
+
+// tt1 + tt1x1 tt1x2 tt1x3 tt1x4
+
+static const Col
+tt1col[] = {
+ { 0, "A", 1, NdbDictionary::Column::Unsigned, 1, 0 },
+ { 1, "B", 0, NdbDictionary::Column::Unsigned, 1, 1 },
+ { 2, "C", 0, NdbDictionary::Column::Unsigned, 1, 1 },
+ { 3, "D", 0, NdbDictionary::Column::Unsigned, 1, 1 },
+ { 4, "E", 0, NdbDictionary::Column::Unsigned, 1, 1 }
+};
+
+static const ICol
+tt1x1col[] = {
+ { 0, tt1col[1] }
+};
+
+static const ICol
+tt1x2col[] = {
+ { 0, tt1col[1] },
+ { 1, tt1col[2] }
+};
+
+static const ICol
+tt1x3col[] = {
+ { 0, tt1col[3] },
+ { 1, tt1col[2] },
+ { 2, tt1col[1] }
+};
+
+static const ICol
+tt1x4col[] = {
+ { 0, tt1col[1] },
+ { 1, tt1col[4] },
+ { 2, tt1col[2] },
+ { 3, tt1col[3] }
+};
+
+static const ITab
+tt1x1 = {
+ "TT1X1", 1, tt1x1col
+};
+
+static const ITab
+tt1x2 = {
+ "TT1X2", 2, tt1x2col
+};
+
+static const ITab
+tt1x3 = {
+ "TT1X3", 3, tt1x3col
+};
+
+static const ITab
+tt1x4 = {
+ "TT1X4", 4, tt1x4col
+};
+
+static const ITab
+tt1itab[] = {
+ tt1x1,
+ tt1x2,
+ tt1x3,
+ tt1x4
+};
+
+static const Tab
+tt1 = {
+ "TT1", 5, tt1col, 4, tt1itab
+};
+
+// tt2 + tt2x1 tt2x2 tt2x3
+
+static const Col
+tt2col[] = {
+ { 0, "A", 1, NdbDictionary::Column::Unsigned, 1, 0 },
+ { 1, "B", 0, NdbDictionary::Column::Unsigned, 1, 1 },
+ { 2, "C", 0, NdbDictionary::Column::Varchar, 20, 1 },
+ { 3, "D", 0, NdbDictionary::Column::Varchar, 5, 1 },
+ { 4, "E", 0, NdbDictionary::Column::Varchar, 5, 1 }
+};
+
+static const ICol
+tt2x1col[] = {
+ { 0, tt2col[1] },
+ { 1, tt2col[2] }
+};
+
+static const ICol
+tt2x2col[] = {
+ { 0, tt2col[2] },
+ { 1, tt2col[1] }
+};
+
+static const ICol
+tt2x3col[] = {
+ { 0, tt2col[3] },
+ { 1, tt2col[4] }
+};
+
+static const ITab
+tt2x1 = {
+ "TT2X1", 2, tt2x1col
+};
+
+static const ITab
+tt2x2 = {
+ "TT2X2", 2, tt2x2col
+};
+
+static const ITab
+tt2x3 = {
+ "TT2X3", 2, tt2x3col
+};
+
+static const ITab
+tt2itab[] = {
+ tt2x1,
+ tt2x2,
+ tt2x3
+};
+
+static const Tab
+tt2 = {
+ "TT2", 5, tt2col, 3, tt2itab
+};
+
+// all tables
+
+static const Tab
+tablist[] = {
+ tt1,
+ tt2
+};
+
+static const unsigned
+tabcount = sizeof(tablist) / sizeof(tablist[0]);
+
+// connections
+
+struct Con {
+ Ndb* m_ndb;
+ NdbDictionary::Dictionary* m_dic;
+ NdbConnection* m_tx;
+ NdbOperation* m_op;
+ NdbConnection* m_scantx;
+ NdbOperation* m_scanop;
+ enum ScanMode { ScanNo = 0, Committed, Latest, Exclusive };
+ ScanMode m_scanmode;
+ enum ErrType { ErrNone = 0, ErrDeadlock, ErrOther };
+ ErrType m_errtype;
+ Con() :
+ m_ndb(0), m_dic(0), m_tx(0), m_op(0),
+ m_scantx(0), m_scanop(0), m_scanmode(ScanNo), m_errtype(ErrNone) {}
+ int connect();
+ void disconnect();
+ int startTransaction();
+ int startBuddyTransaction(const Con& con);
+ int getNdbOperation(const Tab& tab);
+ int getNdbOperation(const ITab& itab, const Tab& tab);
+ int equal(int num, const char* addr);
+ int getValue(int num, NdbRecAttr*& rec);
+ int setValue(int num, const char* addr);
+ int setBound(int num, int type, const void* value);
+ int execute(ExecType t);
+ int openScanRead(unsigned parallelism);
+ int openScanExclusive(unsigned parallelism);
+ int executeScan();
+ int nextScanResult();
+ int takeOverForUpdate(Con& scan);
+ int takeOverForDelete(Con& scan);
+ void closeTransaction();
+ void printerror(NdbOut& out);
+ // flush dict cache
+ int bugger() {
+ //disconnect();
+ //CHK(connect() == 0);
+ return 0;
+ }
+};
+
+int
+Con::connect()
+{
+ assert(m_ndb == 0);
+ m_ndb = new Ndb("TEST_DB");
+ CHKCON(m_ndb->init() == 0, *this);
+ CHKCON(m_ndb->waitUntilReady(30) == 0, *this);
+ m_dic = m_ndb->getDictionary();
+ m_tx = 0, m_op = 0;
+ return 0;
+}
+
+void
+Con::disconnect()
+{
+ delete m_ndb;
+ m_ndb = 0, m_dic = 0, m_tx = 0, m_op = 0;
+}
+
+int
+Con::startTransaction()
+{
+ assert(m_ndb != 0 && m_tx == 0);
+ CHKCON((m_tx = m_ndb->startTransaction()) != 0, *this);
+ return 0;
+}
+
+int
+Con::startBuddyTransaction(const Con& con)
+{
+ assert(m_ndb != 0 && m_tx == 0 && con.m_ndb == m_ndb && con.m_tx != 0);
+ CHKCON((m_tx = m_ndb->hupp(con.m_tx)) != 0, *this);
+ return 0;
+}
+
+int
+Con::getNdbOperation(const Tab& tab)
+{
+ assert(m_tx != 0);
+ CHKCON((m_op = m_tx->getNdbOperation(tab.m_name)) != 0, *this);
+ return 0;
+}
+
+int
+Con::getNdbOperation(const ITab& itab, const Tab& tab)
+{
+ CHKCON((m_op = m_tx->getNdbOperation(itab.m_name, tab.m_name)) != 0, *this);
+ return 0;
+}
+
+int
+Con::equal(int num, const char* addr)
+{
+ assert(m_tx != 0 && m_op != 0);
+ CHKCON(m_op->equal(num, addr) == 0, *this);
+ return 0;
+}
+
+int
+Con::getValue(int num, NdbRecAttr*& rec)
+{
+ assert(m_tx != 0 && m_op != 0);
+ CHKCON((rec = m_op->getValue(num, 0)) != 0, *this);
+ return 0;
+}
+
+int
+Con::setValue(int num, const char* addr)
+{
+ assert(m_tx != 0 && m_op != 0);
+ CHKCON(m_op->setValue(num, addr) == 0, *this);
+ return 0;
+}
+
+int
+Con::setBound(int num, int type, const void* value)
+{
+ assert(m_tx != 0 && m_op != 0);
+ CHKCON(m_op->setBound(num, type, value) == 0, *this);
+ return 0;
+}
+
+int
+Con::execute(ExecType t)
+{
+ assert(m_tx != 0);
+ CHKCON(m_tx->execute(t) == 0, *this);
+ return 0;
+}
+
+int
+Con::openScanRead(unsigned parallelism)
+{
+ assert(m_tx != 0 && m_op != 0);
+ CHKCON(m_op->openScanRead(parallelism) == 0, *this);
+ return 0;
+}
+
+int
+Con::openScanExclusive(unsigned parallelism)
+{
+ assert(m_tx != 0 && m_op != 0);
+ CHKCON(m_op->openScanExclusive(parallelism) == 0, *this);
+ return 0;
+}
+
+int
+Con::executeScan()
+{
+ CHKCON(m_tx->executeScan() == 0, *this);
+ return 0;
+}
+
+int
+Con::nextScanResult()
+{
+ int ret;
+ CHKCON((ret = m_tx->nextScanResult()) != -1, *this);
+ assert(ret == 0 || ret == 1);
+ return ret;
+}
+
+int
+Con::takeOverForUpdate(Con& scan)
+{
+ assert(m_tx != 0 && scan.m_op != 0);
+ CHKCON((m_op = scan.m_op->takeOverForUpdate(m_tx)) != 0, scan);
+ return 0;
+}
+
+int
+Con::takeOverForDelete(Con& scan)
+{
+ assert(m_tx != 0 && scan.m_op != 0);
+ CHKCON((m_op = scan.m_op->takeOverForUpdate(m_tx)) != 0, scan);
+ return 0;
+}
+
+void
+Con::closeTransaction()
+{
+ assert(m_ndb != 0 && m_tx != 0);
+ m_ndb->closeTransaction(m_tx);
+ m_tx = 0, m_op = 0;
+}
+
+void
+Con::printerror(NdbOut& out)
+{
+ m_errtype = ErrOther;
+ unsigned any = 0;
+ int code;
+ if (m_ndb) {
+ if ((code = m_ndb->getNdbError().code) != 0) {
+ LL0(++any << " ndb: error " << m_ndb->getNdbError());
+ }
+ if (m_dic && (code = m_dic->getNdbError().code) != 0) {
+ LL0(++any << " dic: error " << m_dic->getNdbError());
+ }
+ if (m_tx) {
+ if ((code = m_tx->getNdbError().code) != 0) {
+ LL0(++any << " con: error " << m_tx->getNdbError());
+ if (code == 266 || code == 274 || code == 296 || code == 297)
+ m_errtype = ErrDeadlock;
+ }
+ if (m_op && m_op->getNdbError().code != 0) {
+ LL0(++any << " op : error " << m_op->getNdbError());
+ }
+ }
+ }
+ if (! any) {
+ LL0("failed but no NDB error code");
+ }
+}
+
+// dictionary operations
+
+static int
+invalidateindex(Par par, const ITab& itab)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ con.m_dic->invalidateIndex(itab.m_name, tab.m_name);
+ return 0;
+}
+
+static int
+invalidateindex(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ for (unsigned i = 0; i < tab.m_itabs; i++) {
+ if (! useindex(i))
+ continue;
+ const ITab& itab = tab.m_itab[i];
+ invalidateindex(par, itab);
+ }
+ return 0;
+}
+
+static int
+invalidatetable(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ invalidateindex(par);
+ con.m_dic->invalidateTable(tab.m_name);
+ return 0;
+}
+
+static int
+droptable(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ if (con.m_dic->getTable(tab.m_name) == 0) {
+ // how to check for error
+ LL4("no table " << tab.m_name);
+ } else {
+ LL3("drop table " << tab.m_name);
+ CHKCON(con.m_dic->dropTable(tab.m_name) == 0, con);
+ }
+ return 0;
+}
+
+static int
+createtable(Par par)
+{
+ Con& con = par.con();
+ CHK(con.bugger() == 0);
+ const Tab& tab = par.tab();
+ LL3("create table " << tab.m_name);
+ LL4(tab);
+ NdbDictionary::Table t(tab.m_name);
+ if (par.m_fragtype != NdbDictionary::Object::FragUndefined) {
+ t.setFragmentType(par.m_fragtype);
+ }
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ const Col& col = tab.m_col[k];
+ NdbDictionary::Column c(col.m_name);
+ c.setPrimaryKey(col.m_pk);
+ c.setType(col.m_type);
+ c.setLength(col.m_length);
+ c.setNullable(col.m_nullable);
+ t.addColumn(c);
+ }
+ CHKCON(con.m_dic->createTable(t) == 0, con);
+ return 0;
+}
+
+static int
+dropindex(Par par, const ITab& itab)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ if (con.m_dic->getIndex(itab.m_name, tab.m_name) == 0) {
+ // how to check for error
+ LL4("no index " << itab.m_name);
+ } else {
+ LL3("drop index " << itab.m_name);
+ CHKCON(con.m_dic->dropIndex(itab.m_name, tab.m_name) == 0, con);
+ }
+ return 0;
+}
+
+static int
+dropindex(Par par)
+{
+ const Tab& tab = par.tab();
+ for (unsigned i = 0; i < tab.m_itabs; i++) {
+ if (! useindex(i))
+ continue;
+ const ITab& itab = tab.m_itab[i];
+ CHK(dropindex(par, itab) == 0);
+ }
+ return 0;
+}
+
+static int
+createindex(Par par, const ITab& itab)
+{
+ Con& con = par.con();
+ CHK(con.bugger() == 0);
+ const Tab& tab = par.tab();
+ LL3("create index " << itab.m_name);
+ LL4(itab);
+ NdbDictionary::Index x(itab.m_name);
+ x.setTable(tab.m_name);
+ x.setType(NdbDictionary::Index::OrderedIndex);
+ x.setLogging(false);
+ for (unsigned k = 0; k < itab.m_icols; k++) {
+ const Col& col = itab.m_icol[k].m_col;
+ x.addColumnName(col.m_name);
+ }
+ CHKCON(con.m_dic->createIndex(x) == 0, con);
+ return 0;
+}
+
+static int
+createindex(Par par)
+{
+ const Tab& tab = par.tab();
+ for (unsigned i = 0; i < tab.m_itabs; i++) {
+ if (! useindex(i))
+ continue;
+ const ITab& itab = tab.m_itab[i];
+ CHK(createindex(par, itab) == 0);
+ }
+ return 0;
+}
+
+// data sets
+
+static unsigned
+urandom(unsigned n)
+{
+ if (n == 0)
+ return 0;
+ unsigned i = random() % n;
+ return i;
+}
+
+static int
+irandom(unsigned n)
+{
+ if (n == 0)
+ return 0;
+ int i = random() % n;
+ if (random() & 0x1)
+ i = -i;
+ return i;
+}
+
+// Val - typed column value
+
+struct Val {
+ const Col& m_col;
+ union {
+ Uint32 m_uint32;
+ char* m_varchar;
+ };
+ Val(const Col& col);
+ ~Val();
+ void copy(const Val& val2);
+ void copy(const void* addr);
+ const void* dataaddr() const;
+ bool m_null;
+ int setval(Par par) const;
+ void calc(Par par, unsigned i);
+ int verify(const Val& val2) const;
+ int cmp(const Val& val2) const;
+private:
+ Val& operator=(const Val& val2);
+};
+
+static NdbOut&
+operator<<(NdbOut& out, const Val& val);
+
+Val::Val(const Col& col) :
+ m_col(col)
+{
+ switch (col.m_type) {
+ case NdbDictionary::Column::Unsigned:
+ break;
+ case NdbDictionary::Column::Varchar:
+ m_varchar = new char [2 + col.m_length];
+ break;
+ default:
+ assert(false);
+ break;
+ }
+}
+
+Val::~Val()
+{
+ const Col& col = m_col;
+ switch (col.m_type) {
+ case NdbDictionary::Column::Unsigned:
+ break;
+ case NdbDictionary::Column::Varchar:
+ delete [] m_varchar;
+ break;
+ default:
+ assert(false);
+ break;
+ }
+}
+
+void
+Val::copy(const Val& val2)
+{
+ const Col& col = m_col;
+ const Col& col2 = val2.m_col;
+ assert(col.m_type == col2.m_type && col.m_length == col2.m_length);
+ if (val2.m_null) {
+ m_null = true;
+ return;
+ }
+ copy(val2.dataaddr());
+}
+
+void
+Val::copy(const void* addr)
+{
+ const Col& col = m_col;
+ switch (col.m_type) {
+ case NdbDictionary::Column::Unsigned:
+ m_uint32 = *(const Uint32*)addr;
+ break;
+ case NdbDictionary::Column::Varchar:
+ memcpy(m_varchar, addr, 2 + col.m_length);
+ break;
+ default:
+ assert(false);
+ break;
+ }
+ m_null = false;
+}
+
+const void*
+Val::dataaddr() const
+{
+ const Col& col = m_col;
+ switch (col.m_type) {
+ case NdbDictionary::Column::Unsigned:
+ return &m_uint32;
+ case NdbDictionary::Column::Varchar:
+ return m_varchar;
+ default:
+ break;
+ }
+ assert(false);
+ return 0;
+}
+
+int
+Val::setval(Par par) const
+{
+ Con& con = par.con();
+ const Col& col = m_col;
+ const char* addr = (const char*)dataaddr();
+ if (m_null)
+ addr = 0;
+ if (col.m_pk)
+ CHK(con.equal(col.m_num, addr) == 0);
+ else
+ CHK(con.setValue(col.m_num, addr) == 0);
+ LL5("setval [" << m_col << "] " << *this);
+ return 0;
+}
+
+void
+Val::calc(Par par, unsigned i)
+{
+ const Col& col = m_col;
+ m_null = false;
+ if (col.m_pk) {
+ m_uint32 = i;
+ return;
+ }
+ if (col.m_nullable && urandom(100) < par.m_pctnull) {
+ m_null = true;
+ return;
+ }
+ unsigned v = par.m_range + irandom((par.m_pctrange * par.m_range) / 100);
+ switch (col.m_type) {
+ case NdbDictionary::Column::Unsigned:
+ m_uint32 = v;
+ break;
+ case NdbDictionary::Column::Varchar:
+ {
+ unsigned n = 0;
+ while (n < col.m_length) {
+ if (urandom(1 + col.m_length) == 0) {
+ // nice distribution on lengths
+ break;
+ }
+ m_varchar[2 + n++] = 'a' + urandom((par.m_pctrange * 10) / 100);
+ }
+ m_varchar[0] = (n >> 8);
+ m_varchar[1] = (n & 0xff);
+ while (n < col.m_length) {
+ m_varchar[2 + n++] = 0;
+ }
+ }
+ break;
+ default:
+ assert(false);
+ break;
+ }
+ // verify format
+ col.verify(dataaddr());
+}
+
+int
+Val::verify(const Val& val2) const
+{
+ CHK(cmp(val2) == 0);
+ return 0;
+}
+
+int
+Val::cmp(const Val& val2) const
+{
+ const Col& col = m_col;
+ const Col& col2 = val2.m_col;
+ assert(col.m_type == col2.m_type && col.m_length == col2.m_length);
+ if (m_null || val2.m_null) {
+ if (! m_null)
+ return -1;
+ if (! val2.m_null)
+ return +1;
+ return 0;
+ }
+ // verify data formats
+ col.verify(dataaddr());
+ col.verify(val2.dataaddr());
+ // compare
+ switch (col.m_type) {
+ case NdbDictionary::Column::Unsigned:
+ if (m_uint32 < val2.m_uint32)
+ return -1;
+ if (m_uint32 > val2.m_uint32)
+ return +1;
+ return 0;
+ case NdbDictionary::Column::Varchar:
+ return memcmp(&m_varchar[2], &val2.m_varchar[2], col.m_length);
+ default:
+ break;
+ }
+ assert(false);
+ return 0;
+}
+
+static NdbOut&
+operator<<(NdbOut& out, const Val& val)
+{
+ const Col& col = val.m_col;
+ if (val.m_null) {
+ out << "NULL";
+ return out;
+ }
+ switch (col.m_type) {
+ case NdbDictionary::Column::Unsigned:
+ out << val.m_uint32;
+ break;
+ case NdbDictionary::Column::Varchar:
+ {
+ char buf[8000];
+ unsigned n = (val.m_varchar[0] << 8) | val.m_varchar[1];
+ assert(n <= col.m_length);
+ sprintf(buf, "'%.*s'[%d]", n, &val.m_varchar[2], n);
+ out << buf;
+ }
+ break;
+ default:
+ out << "type" << col.m_type;
+ assert(false);
+ break;
+ }
+ return out;
+}
+
+// Row - table tuple
+
+struct Row {
+ const Tab& m_tab;
+ Val** m_val;
+ bool m_exist;
+ Row(const Tab& tab);
+ ~Row();
+ void copy(const Row& row2);
+ void calc(Par par, unsigned i);
+ int verify(const Row& row2) const;
+ int insrow(Par par);
+ int updrow(Par par);
+ int delrow(Par par);
+ int selrow(Par par);
+ int setrow(Par par);
+ int cmp(const Row& row2) const;
+private:
+ Row& operator=(const Row& row2);
+};
+
+Row::Row(const Tab& tab) :
+ m_tab(tab)
+{
+ m_val = new Val* [tab.m_cols];
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ const Col& col = tab.m_col[k];
+ m_val[k] = new Val(col);
+ }
+ m_exist = false;
+}
+
+Row::~Row()
+{
+ const Tab& tab = m_tab;
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ delete m_val[k];
+ }
+ delete [] m_val;
+}
+
+void
+Row::copy(const Row& row2)
+{
+ const Tab& tab = m_tab;
+ assert(&tab == &row2.m_tab);
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ Val& val = *m_val[k];
+ const Val& val2 = *row2.m_val[k];
+ val.copy(val2);
+ }
+}
+
+void
+Row::calc(Par par, unsigned i)
+{
+ const Tab& tab = m_tab;
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ Val& val = *m_val[k];
+ val.calc(par, i);
+ }
+}
+
+int
+Row::verify(const Row& row2) const
+{
+ const Tab& tab = m_tab;
+ assert(&tab == &row2.m_tab);
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ const Val& val = *m_val[k];
+ const Val& val2 = *row2.m_val[k];
+ CHK(val.verify(val2) == 0);
+ }
+ return 0;
+}
+
+int
+Row::insrow(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = m_tab;
+ assert(! m_exist);
+ CHK(con.getNdbOperation(tab) == 0);
+ CHKCON(con.m_op->insertTuple() == 0, con);
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ const Val& val = *m_val[k];
+ CHK(val.setval(par) == 0);
+ }
+ m_exist = true;
+ return 0;
+}
+
+int
+Row::updrow(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = m_tab;
+ assert(m_exist);
+ CHK(con.getNdbOperation(tab) == 0);
+ CHKCON(con.m_op->updateTuple() == 0, con);
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ const Val& val = *m_val[k];
+ CHK(val.setval(par) == 0);
+ }
+ return 0;
+}
+
+int
+Row::delrow(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = m_tab;
+ assert(m_exist);
+ CHK(con.getNdbOperation(m_tab) == 0);
+ CHKCON(con.m_op->deleteTuple() == 0, con);
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ const Val& val = *m_val[k];
+ const Col& col = val.m_col;
+ if (col.m_pk)
+ CHK(val.setval(par) == 0);
+ }
+ m_exist = false;
+ return 0;
+}
+
+int
+Row::selrow(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = m_tab;
+ CHK(con.getNdbOperation(m_tab) == 0);
+ CHKCON(con.m_op->readTuple() == 0, con);
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ const Val& val = *m_val[k];
+ const Col& col = val.m_col;
+ if (col.m_pk)
+ CHK(val.setval(par) == 0);
+ }
+ m_exist = false;
+ return 0;
+}
+
+int
+Row::setrow(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = m_tab;
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ const Val& val = *m_val[k];
+ const Col& col = val.m_col;
+ if (! col.m_pk)
+ CHK(val.setval(par) == 0);
+ }
+ return 0;
+}
+
+int
+Row::cmp(const Row& row2) const
+{
+ const Tab& tab = m_tab;
+ assert(&tab == &row2.m_tab);
+ int c = 0;
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ const Val& val = *m_val[k];
+ const Val& val2 = *row2.m_val[k];
+ if ((c = val.cmp(val2)) != 0)
+ break;
+ }
+ return c;
+}
+
+static NdbOut&
+operator<<(NdbOut& out, const Row& row)
+{
+ const Tab& tab = row.m_tab;
+ for (unsigned i = 0; i < tab.m_cols; i++) {
+ if (i > 0)
+ out << " ";
+ out << *row.m_val[i];
+ }
+ return out;
+}
+
+// Set - set of table tuples
+
+struct Set {
+ const Tab& m_tab;
+ unsigned m_rows;
+ unsigned m_count;
+ Row** m_row;
+ Row** m_saverow;
+ Row* m_keyrow;
+ NdbRecAttr** m_rec;
+ Set(const Tab& tab, unsigned rows);
+ ~Set();
+ // row methods
+ bool exist(unsigned i) const;
+ void calc(Par par, unsigned i);
+ int insrow(Par par, unsigned i);
+ int updrow(Par par, unsigned i);
+ int delrow(Par par, unsigned i);
+ int selrow(Par par, unsigned i);
+ int setrow(Par par, unsigned i);
+ int getval(Par par);
+ int getkey(Par par, unsigned* i);
+ int putval(unsigned i, bool force);
+ // set methods
+ int verify(const Set& set2) const;
+ void savepoint();
+ void commit();
+ void rollback();
+ // locking (not perfect since ops may complete in different order)
+ NdbMutex* m_mutex;
+ void lock() {
+ NdbMutex_Lock(m_mutex);
+ }
+ void unlock() {
+ NdbMutex_Unlock(m_mutex);
+ }
+private:
+ Set& operator=(const Set& set2);
+};
+
+Set::Set(const Tab& tab, unsigned rows) :
+ m_tab(tab)
+{
+ m_rows = rows;
+ m_count = 0;
+ m_row = new Row* [m_rows];
+ for (unsigned i = 0; i < m_rows; i++) {
+ m_row[i] = 0;
+ }
+ m_saverow = 0;
+ m_keyrow = new Row(tab);
+ m_rec = new NdbRecAttr* [tab.m_cols];
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ m_rec[k] = 0;
+ }
+ m_mutex = NdbMutex_Create();
+ assert(m_mutex != 0);
+}
+
+Set::~Set()
+{
+ for (unsigned i = 0; i < m_rows; i++) {
+ delete m_row[i];
+ if (m_saverow != 0)
+ delete m_saverow[i];
+ }
+ delete [] m_row;
+ delete [] m_saverow;
+ delete m_keyrow;
+ delete [] m_rec;
+ NdbMutex_Destroy(m_mutex);
+}
+
+bool
+Set::exist(unsigned i) const
+{
+ assert(i < m_rows);
+ return m_row[i] != 0 && m_row[i]->m_exist;
+}
+
+void
+Set::calc(Par par, unsigned i)
+{
+ const Tab& tab = m_tab;
+ if (m_row[i] == 0)
+ m_row[i] = new Row(tab);
+ Row& row = *m_row[i];
+ // value generation parameters
+ par.m_pctnull = 10;
+ par.m_pctrange = 40;
+ row.calc(par, i);
+}
+
+int
+Set::insrow(Par par, unsigned i)
+{
+ assert(m_row[i] != 0 && m_count < m_rows);
+ CHK(m_row[i]->insrow(par) == 0);
+ m_count++;
+ return 0;
+}
+
+int
+Set::updrow(Par par, unsigned i)
+{
+ assert(m_row[i] != 0);
+ CHK(m_row[i]->updrow(par) == 0);
+ return 0;
+}
+
+int
+Set::delrow(Par par, unsigned i)
+{
+ assert(m_row[i] != 0 && m_count != 0);
+ CHK(m_row[i]->delrow(par) == 0);
+ m_count--;
+ return 0;
+}
+
+int
+Set::selrow(Par par, unsigned i)
+{
+ Con& con = par.con();
+ m_keyrow->calc(par, i);
+ CHK(m_keyrow->selrow(par) == 0);
+ CHK(getval(par) == 0);
+ return 0;
+}
+
+int
+Set::setrow(Par par, unsigned i)
+{
+ Con& con = par.con();
+ assert(m_row[i] != 0);
+ CHK(m_row[i]->setrow(par) == 0);
+ return 0;
+}
+
+int
+Set::getval(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = m_tab;
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ CHK(con.getValue(k, m_rec[k]) == 0);
+ }
+ return 0;
+}
+
+int
+Set::getkey(Par par, unsigned* i)
+{
+ assert(m_rec[0] != 0);
+ const char* aRef0 = m_rec[0]->aRef();
+ Uint32 key = *(const Uint32*)aRef0;
+ CHKMSG(key < m_rows, "key=" << key << " rows=" << m_rows);
+ *i = key;
+ return 0;
+}
+
+int
+Set::putval(unsigned i, bool force)
+{
+ const Tab& tab = m_tab;
+ if (m_row[i] == 0)
+ m_row[i] = new Row(tab);
+ Row& row = *m_row[i];
+ CHK(! row.m_exist || force);
+ for (unsigned k = 0; k < tab.m_cols; k++) {
+ Val& val = *row.m_val[k];
+ NdbRecAttr* rec = m_rec[k];
+ assert(rec != 0);
+ if (rec->isNULL()) {
+ val.m_null = true;
+ continue;
+ }
+ const char* aRef = m_rec[k]->aRef();
+ val.copy(aRef);
+ val.m_null = false;
+ }
+ if (! row.m_exist) {
+ row.m_exist = true;
+ m_count++;
+ }
+ return 0;
+}
+
+int
+Set::verify(const Set& set2) const
+{
+ const Tab& tab = m_tab;
+ assert(&tab == &set2.m_tab && m_rows == set2.m_rows);
+ CHKMSG(m_count == set2.m_count, "set=" << m_count << " set2=" << set2.m_count);
+ for (unsigned i = 0; i < m_rows; i++) {
+ CHK(exist(i) == set2.exist(i));
+ if (! exist(i))
+ continue;
+ Row& row = *m_row[i];
+ Row& row2 = *set2.m_row[i];
+ CHK(row.verify(row2) == 0);
+ }
+ return 0;
+}
+
+void
+Set::savepoint()
+{
+ const Tab& tab = m_tab;
+ assert(m_saverow == 0);
+ m_saverow = new Row* [m_rows];
+ for (unsigned i = 0; i < m_rows; i++) {
+ if (m_row[i] == 0)
+ m_saverow[i] = 0;
+ else {
+ m_saverow[i] = new Row(tab);
+ m_saverow[i]->copy(*m_row[i]);
+ }
+ }
+}
+
+void
+Set::commit()
+{
+ delete [] m_saverow;
+ m_saverow = 0;
+}
+
+void
+Set::rollback()
+{
+ assert(m_saverow != 0);
+ m_row = m_saverow;
+ m_saverow = 0;
+}
+
+static NdbOut&
+operator<<(NdbOut& out, const Set& set)
+{
+ for (unsigned i = 0; i < set.m_rows; i++) {
+ const Row& row = *set.m_row[i];
+ if (i > 0)
+ out << endl;
+ out << row;
+ }
+ return out;
+}
+
+// BVal - range scan bound
+
+struct BVal : public Val {
+ const ICol& m_icol;
+ int m_type;
+ BVal(const ICol& icol);
+ int setbnd(Par par) const;
+};
+
+BVal::BVal(const ICol& icol) :
+ Val(icol.m_col),
+ m_icol(icol)
+{
+}
+
+int
+BVal::setbnd(Par par) const
+{
+ Con& con = par.con();
+ const char* addr = (const char*)dataaddr();
+ assert(! m_null);
+ const ICol& icol = m_icol;
+ CHK(con.setBound(icol.m_num, m_type, addr) == 0);
+ return 0;
+}
+
+static NdbOut&
+operator<<(NdbOut& out, const BVal& bval)
+{
+ const ICol& icol = bval.m_icol;
+ const Col& col = icol.m_col;
+ const Val& val = bval;
+ out << "type " << bval.m_type;
+ out << " icol " << icol.m_num;
+ out << " col " << col.m_name << "(" << col.m_num << ")";
+ out << " value " << val;
+ return out;
+}
+
+// BSet - set of bounds
+
+struct BSet {
+ const Tab& m_tab;
+ const ITab& m_itab;
+ unsigned m_alloc;
+ unsigned m_bvals;
+ BVal** m_bval;
+ BSet(const Tab& tab, const ITab& itab, unsigned rows);
+ void calc(Par par);
+ int setbnd(Par par) const;
+ void filter(const Set& set, Set& set2) const;
+};
+
+BSet::BSet(const Tab& tab, const ITab& itab, unsigned rows) :
+ m_tab(tab),
+ m_itab(itab),
+ m_alloc(2 * itab.m_icols),
+ m_bvals(0)
+{
+ m_bval = new BVal* [m_alloc];
+}
+
+void
+BSet::calc(Par par)
+{
+ const ITab& itab = m_itab;
+ for (unsigned k = 0; k < itab.m_icols; k++) {
+ const ICol& icol = itab.m_icol[k];
+ const Col& col = icol.m_col;
+ for (unsigned i = 0; i <= 1; i++) {
+ if (urandom(10) == 0)
+ return;
+ assert(m_bvals < m_alloc);
+ BVal& bval = *new BVal(icol);
+ m_bval[m_bvals++] = &bval;
+ bval.m_null = false;
+ // equality bound only on i==0
+ unsigned sel = urandom(5 - i);
+ if (sel < 2)
+ bval.m_type = 0 | (1 << i);
+ else if (sel < 4)
+ bval.m_type = 1 | (1 << i);
+ else
+ bval.m_type = 4;
+ if (k + 1 < itab.m_icols)
+ bval.m_type = 4;
+ // value generation parammeters
+ par.m_pctnull = 0;
+ par.m_pctrange = 50; // bit higher
+ do {
+ bval.calc(par, 0);
+ if (i == 1) {
+ assert(m_bvals >= 2);
+ const BVal& bv1 = *m_bval[m_bvals - 2];
+ const BVal& bv2 = *m_bval[m_bvals - 1];
+ if (bv1.cmp(bv2) > 0 && urandom(100) != 0)
+ continue;
+ }
+ } while (0);
+ // equality bound only once
+ if (bval.m_type == 4)
+ break;
+ }
+ }
+}
+
+int
+BSet::setbnd(Par par) const
+{
+ for (unsigned j = 0; j < m_bvals; j++) {
+ const BVal& bval = *m_bval[j];
+ CHK(bval.setbnd(par) == 0);
+ }
+ return 0;
+}
+
+void
+BSet::filter(const Set& set, Set& set2) const
+{
+ const Tab& tab = m_tab;
+ const ITab& itab = m_itab;
+ assert(&tab == &set2.m_tab && set.m_rows == set2.m_rows);
+ assert(set2.m_count == 0);
+ for (unsigned i = 0; i < set.m_rows; i++) {
+ if (! set.exist(i))
+ continue;
+ const Row& row = *set.m_row[i];
+ bool ok1 = false;
+ for (unsigned k = 0; k < itab.m_icols; k++) {
+ const ICol& icol = itab.m_icol[k];
+ const Col& col = icol.m_col;
+ const Val& val = *row.m_val[col.m_num];
+ if (! val.m_null) {
+ ok1 = true;
+ break;
+ }
+ }
+ if (! ok1)
+ continue;
+ bool ok2 = true;
+ for (unsigned j = 0; j < m_bvals; j++) {
+ const BVal& bval = *m_bval[j];
+ const ICol& icol = bval.m_icol;
+ const Col& col = icol.m_col;
+ const Val& val = *row.m_val[col.m_num];
+ int ret = bval.cmp(val);
+ if (bval.m_type == 0)
+ ok2 = (ret <= 0);
+ else if (bval.m_type == 1)
+ ok2 = (ret < 0);
+ else if (bval.m_type == 2)
+ ok2 = (ret >= 0);
+ else if (bval.m_type == 3)
+ ok2 = (ret > 0);
+ else if (bval.m_type == 4)
+ ok2 = (ret == 0);
+ else {
+ assert(false);
+ }
+ if (! ok2)
+ break;
+ }
+ if (! ok2)
+ continue;
+ if (set2.m_row[i] == 0)
+ set2.m_row[i] = new Row(tab);
+ Row& row2 = *set2.m_row[i];
+ assert(! row2.m_exist);
+ row2.copy(row);
+ row2.m_exist = true;
+ set2.m_count++;
+ }
+}
+
+static NdbOut&
+operator<<(NdbOut& out, const BSet& bset)
+{
+ out << "bounds=" << bset.m_bvals;
+ for (unsigned j = 0; j < bset.m_bvals; j++) {
+ out << endl;
+ const BVal& bval = *bset.m_bval[j];
+ out << "bound " << j << ": " << bval;
+ }
+ return out;
+}
+
+// pk operations
+
+static int
+pkinsert(Par par)
+{
+ Con& con = par.con();
+ Set& set = par.set();
+ LL3("pkinsert");
+ CHK(con.startTransaction() == 0);
+ unsigned n = 0;
+ for (unsigned j = 0; j < par.m_rows; j++) {
+ unsigned i = thrrow(par, j);
+ set.lock();
+ if (set.exist(i)) {
+ set.unlock();
+ continue;
+ }
+ set.calc(par, i);
+ LL4("pkinsert " << i << ": " << *set.m_row[i]);
+ CHKTRY(set.insrow(par, i) == 0, set.unlock());
+ set.unlock();
+ if (++n == par.m_batch) {
+ CHK(con.execute(Commit) == 0);
+ con.closeTransaction();
+ CHK(con.startTransaction() == 0);
+ n = 0;
+ }
+ }
+ if (n != 0) {
+ CHK(con.execute(Commit) == 0);
+ n = 0;
+ }
+ con.closeTransaction();
+ return 0;
+};
+
+static int
+pkupdate(Par par)
+{
+ Con& con = par.con();
+ Set& set = par.set();
+ LL3("pkupdate");
+ CHK(con.startTransaction() == 0);
+ unsigned n = 0;
+ for (unsigned j = 0; j < par.m_rows; j++) {
+ unsigned i = thrrow(par, j);
+ set.lock();
+ if (! set.exist(i)) {
+ set.unlock();
+ continue;
+ }
+ set.calc(par, i);
+ LL4("pkupdate " << i << ": " << *set.m_row[i]);
+ CHKTRY(set.updrow(par, i) == 0, set.unlock());
+ set.unlock();
+ if (++n == par.m_batch) {
+ CHK(con.execute(Commit) == 0);
+ con.closeTransaction();
+ CHK(con.startTransaction() == 0);
+ n = 0;
+ }
+ }
+ if (n != 0) {
+ CHK(con.execute(Commit) == 0);
+ n = 0;
+ }
+ con.closeTransaction();
+ return 0;
+};
+
+static int
+pkdelete(Par par)
+{
+ Con& con = par.con();
+ Set& set = par.set();
+ LL3("pkdelete");
+ CHK(con.startTransaction() == 0);
+ unsigned n = 0;
+ for (unsigned j = 0; j < par.m_rows; j++) {
+ unsigned i = thrrow(par, j);
+ set.lock();
+ if (! set.exist(i)) {
+ set.unlock();
+ continue;
+ }
+ LL4("pkdelete " << i << ": " << *set.m_row[i]);
+ CHKTRY(set.delrow(par, i) == 0, set.unlock());
+ set.unlock();
+ if (++n == par.m_batch) {
+ CHK(con.execute(Commit) == 0);
+ con.closeTransaction();
+ CHK(con.startTransaction() == 0);
+ n = 0;
+ }
+ }
+ if (n != 0) {
+ CHK(con.execute(Commit) == 0);
+ n = 0;
+ }
+ con.closeTransaction();
+ return 0;
+};
+
+static int
+pkread(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ const Set& set = par.set();
+ LL3((par.m_verify ? "pkverify " : "pkread ") << tab.m_name);
+ // expected
+ const Set& set1 = set;
+ Set set2(tab, set.m_rows);
+ for (unsigned i = 0; i < set.m_rows; i++) {
+ if (! set.exist(i))
+ continue;
+ CHK(con.startTransaction() == 0);
+ CHK(set2.selrow(par, i) == 0);
+ CHK(con.execute(Commit) == 0);
+ unsigned i2 = (unsigned)-1;
+ CHK(set2.getkey(par, &i2) == 0 && i == i2);
+ CHK(set2.putval(i, false) == 0);
+ LL4("row " << set2.m_count << ": " << *set2.m_row[i]);
+ con.closeTransaction();
+ }
+ if (par.m_verify)
+ CHK(set1.verify(set2) == 0);
+ return 0;
+}
+
+// scan read
+
+static int
+scanreadtable(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ const Set& set = par.set();
+ // expected
+ const Set& set1 = set;
+ LL3((par.m_verify ? "scanverify " : "scanread ") << tab.m_name);
+ Set set2(tab, set.m_rows);
+ CHK(con.startTransaction() == 0);
+ CHK(con.getNdbOperation(tab) == 0);
+ CHK(con.openScanRead(par.m_scanrd) == 0);
+ set2.getval(par);
+ CHK(con.executeScan() == 0);
+ while (1) {
+ int ret;
+ CHK((ret = con.nextScanResult()) == 0 || ret == 1);
+ if (ret == 1)
+ break;
+ unsigned i = (unsigned)-1;
+ CHK(set2.getkey(par, &i) == 0);
+ CHK(set2.putval(i, false) == 0);
+ LL4("row " << set2.m_count << ": " << *set2.m_row[i]);
+ }
+ con.closeTransaction();
+ if (par.m_verify)
+ CHK(set1.verify(set2) == 0);
+ return 0;
+}
+
+static int
+scanreadindex(Par par, const ITab& itab, const BSet& bset)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ const Set& set = par.set();
+ // expected
+ Set set1(tab, set.m_rows);
+ bset.filter(set, set1);
+ LL3((par.m_verify ? "scanverify " : "scanread ") << itab.m_name << " bounds=" << bset.m_bvals);
+ LL4(bset);
+ Set set2(tab, set.m_rows);
+ CHK(con.startTransaction() == 0);
+ CHK(con.getNdbOperation(itab, tab) == 0);
+ CHK(con.openScanRead(par.m_scanrd) == 0);
+ CHK(bset.setbnd(par) == 0);
+ set2.getval(par);
+ CHK(con.executeScan() == 0);
+ while (1) {
+ int ret;
+ CHK((ret = con.nextScanResult()) == 0 || ret == 1);
+ if (ret == 1)
+ break;
+ unsigned i = (unsigned)-1;
+ CHK(set2.getkey(par, &i) == 0);
+ LL4("key " << i);
+ CHK(set2.putval(i, par.m_dups) == 0);
+ LL4("row " << set2.m_count << ": " << *set2.m_row[i]);
+ }
+ con.closeTransaction();
+ if (par.m_verify)
+ CHK(set1.verify(set2) == 0);
+ return 0;
+}
+
+static int
+scanreadindex(Par par, const ITab& itab)
+{
+ const Tab& tab = par.tab();
+ for (unsigned i = 0; i < par.m_subloop; i++) {
+ BSet bset(tab, itab, par.m_rows);
+ bset.calc(par);
+ CHK(scanreadindex(par, itab, bset) == 0);
+ }
+ return 0;
+}
+
+static int
+scanreadindex(Par par)
+{
+ const Tab& tab = par.tab();
+ for (unsigned i = 0; i < tab.m_itabs; i++) {
+ if (! useindex(i))
+ continue;
+ const ITab& itab = tab.m_itab[i];
+ CHK(scanreadindex(par, itab) == 0);
+ }
+ return 0;
+}
+
+static int
+scanreadall(Par par)
+{
+ if (par.m_no < 11)
+ CHK(scanreadtable(par) == 0);
+ CHK(scanreadindex(par) == 0);
+ return 0;
+}
+
+// scan update
+
+static int
+scanupdatetable(Par par)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ Set& set = par.set();
+ LL3("scan update " << tab.m_name);
+ Set set2(tab, set.m_rows);
+ CHK(con.startTransaction() == 0);
+ CHK(con.getNdbOperation(tab) == 0);
+ CHK(con.openScanExclusive(par.m_scanex) == 0);
+ set2.getval(par);
+ CHK(con.executeScan() == 0);
+ unsigned count = 0;
+ // updating trans
+ Con con2;
+ con2.m_ndb = con.m_ndb;
+ CHK(con2.startBuddyTransaction(con) == 0);
+ while (1) {
+ int ret;
+ CHK((ret = con.nextScanResult()) == 0 || ret == 1);
+ if (ret == 1)
+ break;
+ unsigned i = (unsigned)-1;
+ CHK(set2.getkey(par, &i) == 0);
+ LL4("key " << i);
+ CHK(set2.putval(i, false) == 0);
+ CHK(con2.takeOverForUpdate(con) == 0);
+ Par par2 = par;
+ par2.m_con = &con2;
+ set.lock();
+ set.calc(par, i);
+ LL4("scan update " << tab.m_name << ": " << *set.m_row[i]);
+ CHKTRY(set.setrow(par2, i) == 0, set.unlock());
+ set.unlock();
+ CHK(con2.execute(NoCommit) == 0);
+ count++;
+ }
+ CHK(con2.execute(Commit) == 0);
+ con2.closeTransaction();
+ LL3("scan update " << tab.m_name << " rows updated=" << count);
+ con.closeTransaction();
+ return 0;
+}
+
+static int
+scanupdateindex(Par par, const ITab& itab, const BSet& bset)
+{
+ Con& con = par.con();
+ const Tab& tab = par.tab();
+ Set& set = par.set();
+ LL3("scan update " << itab.m_name);
+ Set set2(tab, set.m_rows);
+ CHK(con.startTransaction() == 0);
+ CHK(con.getNdbOperation(itab, tab) == 0);
+ CHK(con.openScanExclusive(par.m_scanex) == 0);
+ CHK(bset.setbnd(par) == 0);
+ set2.getval(par);
+ CHK(con.executeScan() == 0);
+ unsigned count = 0;
+ // updating trans
+ Con con2;
+ con2.m_ndb = con.m_ndb;
+ CHK(con2.startBuddyTransaction(con) == 0);
+ while (1) {
+ int ret;
+ CHK((ret = con.nextScanResult()) == 0 || ret == 1);
+ if (ret == 1)
+ break;
+ unsigned i = (unsigned)-1;
+ CHK(set2.getkey(par, &i) == 0);
+ LL4("key " << i);
+ CHK(set2.putval(i, par.m_dups) == 0);
+ // avoid deadlock for now
+ //if (! isthrrow(par, i))
+ //continue;
+ CHK(con2.takeOverForUpdate(con) == 0);
+ Par par2 = par;
+ par2.m_con = &con2;
+ set.lock();
+ set.calc(par, i);
+ LL4("scan update " << itab.m_name << ": " << *set.m_row[i]);
+ CHKTRY(set.setrow(par2, i) == 0, set.unlock());
+ set.unlock();
+ CHK(con2.execute(NoCommit) == 0);
+ count++;
+ }
+ CHK(con2.execute(Commit) == 0);
+ con2.closeTransaction();
+ LL3("scan update " << itab.m_name << " rows updated=" << count);
+ con.closeTransaction();
+ return 0;
+}
+
+static int
+scanupdateindex(Par par, const ITab& itab)
+{
+ const Tab& tab = par.tab();
+ for (unsigned i = 0; i < par.m_subloop; i++) {
+ BSet bset(tab, itab, par.m_rows);
+ bset.calc(par);
+ CHK(scanupdateindex(par, itab, bset) == 0);
+ }
+ return 0;
+}
+
+static int
+scanupdateindex(Par par)
+{
+ const Tab& tab = par.tab();
+ for (unsigned i = 0; i < tab.m_itabs; i++) {
+ if (! useindex(i))
+ continue;
+ const ITab& itab = tab.m_itab[i];
+ CHK(scanupdateindex(par, itab) == 0);
+ }
+ return 0;
+}
+
+static int
+scanupdateall(Par par)
+{
+ CHK(scanupdatetable(par) == 0);
+ CHK(scanupdateindex(par) == 0);
+ return 0;
+}
+
+// medium level routines
+
+static bool
+ignoreverifyerror(Par par)
+{
+ Con& con = par.con();
+ bool b = par.m_threads > 1;
+ if (b) {
+ LL1("ignore verify error");
+ if (con.m_tx != 0)
+ con.closeTransaction();
+ return true;
+ }
+ return b;
+}
+
+static int
+readverify(Par par)
+{
+ par.m_verify = true;
+ CHK(pkread(par) == 0 || ignoreverifyerror(par));
+ CHK(scanreadall(par) == 0 || ignoreverifyerror(par));
+ return 0;
+}
+
+static bool
+ignoredeadlock(Par par)
+{
+ Con& con = par.con();
+ if (con.m_errtype == Con::ErrDeadlock) {
+ LL1("ignore deadlock");
+ con.closeTransaction();
+ return true;
+ }
+ return false;
+}
+
+static int
+pkupdatescanread(Par par)
+{
+ par.m_dups = true;
+ unsigned sel = urandom(10);
+ if (sel < 5) {
+ CHK(pkupdate(par) == 0);
+ } else if (sel < 6) {
+ par.m_verify = false;
+ CHK(scanreadtable(par) == 0);
+ } else {
+ par.m_verify = false;
+ CHK(scanreadindex(par) == 0);
+ }
+ return 0;
+}
+
+static int
+mixedoperations(Par par)
+{
+ par.m_dups = true;
+ unsigned sel = urandom(10);
+ if (sel < 2) {
+ CHK(pkdelete(par) == 0 || ignoredeadlock(par));
+ } else if (sel < 4) {
+ CHK(pkupdate(par) == 0 || ignoredeadlock(par));
+ } else if (sel < 6) {
+ CHK(scanupdatetable(par) == 0 || ignoredeadlock(par));
+ } else {
+ CHK(scanupdateindex(par) == 0 || ignoredeadlock(par));
+ }
+ return 0;
+}
+
+static int
+pkupdateindexbuild(Par par)
+{
+ if (par.m_no == 0) {
+ CHK(createindex(par) == 0);
+ CHK(invalidateindex(par) == 0);
+ } else {
+ CHK(pkupdate(par) == 0);
+ }
+ return 0;
+}
+
+// threads
+
+typedef int (*TFunc)(Par par);
+enum TMode { ST = 1, MT = 2 };
+
+extern "C" { static void* runthread(void* arg); }
+
+struct Thr {
+ enum State { Wait, Start, Stop, Stopped, Exit };
+ State m_state;
+ Par m_par;
+ Uint64 m_id;
+ NdbThread* m_thread;
+ NdbMutex* m_mutex;
+ NdbCondition* m_cond;
+ TFunc m_func;
+ int m_ret;
+ void* m_status;
+ Thr(Par par, unsigned n);
+ ~Thr();
+ int run();
+ void start();
+ void stop();
+ void stopped();
+ void exit();
+ //
+ void lock() {
+ NdbMutex_Lock(m_mutex);
+ }
+ void unlock() {
+ NdbMutex_Unlock(m_mutex);
+ }
+ void wait() {
+ NdbCondition_Wait(m_cond, m_mutex);
+ }
+ void signal() {
+ NdbCondition_Signal(m_cond);
+ }
+ void join() {
+ NdbThread_WaitFor(m_thread, &m_status);
+ m_thread = 0;
+ }
+};
+
+Thr::Thr(Par par, unsigned n) :
+ m_state(Wait),
+ m_par(par),
+ m_id(0),
+ m_thread(0),
+ m_mutex(0),
+ m_cond(0),
+ m_func(0),
+ m_ret(0),
+ m_status(0)
+{
+ m_par.m_no = n;
+ char buf[10];
+ sprintf(buf, "thr%03u", par.m_no);
+ const char* name = strcpy(new char[10], buf);
+ // mutex
+ m_mutex = NdbMutex_Create();
+ m_cond = NdbCondition_Create();
+ assert(m_mutex != 0 && m_cond != 0);
+ // run
+ const unsigned stacksize = 256 * 1024;
+ const NDB_THREAD_PRIO prio = NDB_THREAD_PRIO_LOW;
+ m_thread = NdbThread_Create(runthread, (void**)this, stacksize, name, prio);
+}
+
+Thr::~Thr()
+{
+ if (m_thread != 0) {
+ NdbThread_Destroy(&m_thread);
+ m_thread = 0;
+ }
+ if (m_cond != 0) {
+ NdbCondition_Destroy(m_cond);
+ m_cond = 0;
+ }
+ if (m_mutex != 0) {
+ NdbMutex_Destroy(m_mutex);
+ m_mutex = 0;
+ }
+}
+
+static void*
+runthread(void* arg)
+{
+ Thr& thr = *(Thr*)arg;
+ thr.m_id = (Uint64)pthread_self();
+ if (thr.run() < 0) {
+ LL1("exit on error");
+ } else {
+ LL4("exit ok");
+ }
+ return 0;
+}
+
+int
+Thr::run()
+{
+ LL4("run");
+ Con con;
+ CHK(con.connect() == 0);
+ m_par.m_con = &con;
+ LL4("connected");
+ while (1) {
+ lock();
+ while (m_state != Start && m_state != Exit) {
+ LL4("wait");
+ wait();
+ }
+ if (m_state == Exit) {
+ LL4("exit");
+ unlock();
+ break;
+ }
+ LL4("start");
+ CHK(con.bugger() == 0);
+ assert(m_state == Start);
+ m_ret = (*m_func)(m_par);
+ m_state = Stopped;
+ LL4("stop");
+ signal();
+ unlock();
+ CHK(m_ret == 0);
+ }
+ con.disconnect();
+ return 0;
+}
+
+void
+Thr::start()
+{
+ lock();
+ m_state = Start;
+ signal();
+ unlock();
+}
+
+void
+Thr::stop()
+{
+ lock();
+ m_state = Stop;
+ signal();
+ unlock();
+}
+
+void
+Thr::stopped()
+{
+ lock();
+ while (m_state != Stopped)
+ wait();
+ m_state = Wait;
+ unlock();
+}
+
+void
+Thr::exit()
+{
+ lock();
+ m_state = Exit;
+ signal();
+ unlock();
+}
+
+// test run
+
+static Thr** g_thrlist = 0;
+
+static unsigned
+getthrno()
+{
+ if (g_thrlist != 0) {
+ Uint64 id = (Uint64)pthread_self();
+ for (unsigned n = 0; n < g_opt.m_threads; n++) {
+ if (g_thrlist[n] != 0) {
+ const Thr& thr = *g_thrlist[n];
+ if (thr.m_id == id)
+ return thr.m_par.m_no;
+ }
+ }
+ }
+ return (unsigned)-1;
+}
+
+static int
+runstep(Par par, const char* fname, TFunc func, unsigned mode)
+{
+ LL2(fname);
+ const int threads = (mode & ST ? 1 : par.m_threads);
+ for (int n = 0; n < threads; n++) {
+ LL4("start " << n);
+ Thr& thr = *g_thrlist[n];
+ thr.m_par.m_tab = par.m_tab;
+ thr.m_par.m_set = par.m_set;
+ thr.m_func = func;
+ thr.start();
+ }
+ unsigned errs = 0;
+ for (int n = threads - 1; n >= 0; n--) {
+ LL4("stop " << n);
+ Thr& thr = *g_thrlist[n];
+ thr.stopped();
+ if (thr.m_ret != 0)
+ errs++;
+ }
+ CHK(errs == 0);
+ return 0;
+}
+
+#define RUNSTEP(par, func, mode) CHK(runstep(par, #func, func, mode) == 0)
+
+static int
+tbuild(Par par)
+{
+ RUNSTEP(par, droptable, ST);
+ RUNSTEP(par, createtable, ST);
+ RUNSTEP(par, invalidatetable, MT);
+ for (unsigned i = 0; i < par.m_subloop; i++) {
+ if (i % 2 == 0) {
+ RUNSTEP(par, createindex, ST);
+ RUNSTEP(par, invalidateindex, MT);
+ RUNSTEP(par, pkinsert, MT);
+ } else {
+ RUNSTEP(par, pkinsert, MT);
+ RUNSTEP(par, createindex, ST);
+ RUNSTEP(par, invalidateindex, MT);
+ }
+ RUNSTEP(par, readverify, MT);
+ RUNSTEP(par, pkdelete, MT);
+ RUNSTEP(par, readverify, MT);
+ RUNSTEP(par, dropindex, ST);
+ }
+ return 0;
+}
+
+static int
+tpkops(Par par)
+{
+ RUNSTEP(par, droptable, ST);
+ RUNSTEP(par, createtable, ST);
+ RUNSTEP(par, invalidatetable, MT);
+ RUNSTEP(par, pkinsert, MT);
+ RUNSTEP(par, createindex, ST);
+ RUNSTEP(par, invalidateindex, MT);
+ RUNSTEP(par, readverify, MT);
+ for (unsigned i = 0; i < par.m_subloop; i++) {
+ RUNSTEP(par, pkupdatescanread, MT);
+ RUNSTEP(par, readverify, MT);
+ }
+ RUNSTEP(par, pkdelete, MT);
+ RUNSTEP(par, readverify, MT);
+ return 0;
+}
+
+static int
+tmixedops(Par par)
+{
+ RUNSTEP(par, droptable, ST);
+ RUNSTEP(par, createtable, ST);
+ RUNSTEP(par, invalidatetable, MT);
+ RUNSTEP(par, pkinsert, MT);
+ RUNSTEP(par, createindex, ST);
+ RUNSTEP(par, invalidateindex, MT);
+ RUNSTEP(par, readverify, MT);
+ for (unsigned i = 0; i < par.m_subloop; i++) {
+ RUNSTEP(par, mixedoperations, MT);
+ RUNSTEP(par, readverify, MT);
+ }
+ return 0;
+}
+
+static int
+tbusybuild(Par par)
+{
+ RUNSTEP(par, droptable, ST);
+ RUNSTEP(par, createtable, ST);
+ RUNSTEP(par, invalidatetable, MT);
+ RUNSTEP(par, pkinsert, MT);
+ for (unsigned i = 0; i < par.m_subloop; i++) {
+ RUNSTEP(par, pkupdateindexbuild, MT);
+ RUNSTEP(par, readverify, MT);
+ RUNSTEP(par, dropindex, ST);
+ }
+ return 0;
+}
+
+static int
+ttiming(Par par)
+{
+ Tmr t0, t1, t2;
+ RUNSTEP(par, droptable, ST);
+ RUNSTEP(par, createtable, ST);
+ RUNSTEP(par, invalidatetable, MT);
+ for (unsigned i = 0; i < par.m_subloop; i++) {
+ RUNSTEP(par, pkinsert, MT);
+ t1.on();
+ RUNSTEP(par, pkupdate, MT);
+ t1.off(par.m_totrows);
+ t0.on();
+ RUNSTEP(par, createindex, ST);
+ RUNSTEP(par, invalidateindex, MT);
+ t0.off(par.m_totrows);
+ t2.on();
+ RUNSTEP(par, pkupdate, MT);
+ t2.off(par.m_totrows);
+ RUNSTEP(par, dropindex, ST);
+ }
+ LL1("build index - " << t0.time());
+ LL1("update - " << t1.time());
+ LL1("update indexed - " << t2.time());
+ LL1("overhead - " << t2.over(t1));
+ return 0;
+}
+
+static int
+tdrop(Par par)
+{
+ RUNSTEP(par, droptable, ST);
+ return 0;
+}
+
+struct TCase {
+ const char* m_name;
+ TFunc m_func;
+ const char* m_desc;
+ TCase(const char* name, TFunc func, const char* desc) :
+ m_name(name),
+ m_func(func),
+ m_desc(desc) {
+ }
+};
+
+static const TCase
+tcaselist[] = {
+ TCase("a", tbuild, "index build"),
+ TCase("b", tpkops, "pk operations and scan reads"),
+ TCase("c", tmixedops, "pk operations and scan operations"),
+ TCase("d", tbusybuild, "pk operations and index build"),
+ TCase("t", ttiming, "time index build and maintenance"),
+ TCase("z", tdrop, "drop test tables")
+};
+
+static const unsigned
+tcasecount = sizeof(tcaselist) / sizeof(tcaselist[0]);
+
+static void
+printcases()
+{
+ ndbout << "test cases:" << endl;
+ for (unsigned i = 0; i < tcasecount; i++) {
+ const TCase& tcase = tcaselist[i];
+ ndbout << " " << tcase.m_name << " - " << tcase.m_desc << endl;
+ }
+}
+
+static void
+printtables()
+{
+ ndbout << "tables and indexes:" << endl;
+ for (unsigned j = 0; j < tabcount; j++) {
+ const Tab& tab = tablist[j];
+ ndbout << " " << tab.m_name;
+ for (unsigned i = 0; i < tab.m_itabs; i++) {
+ const ITab& itab = tab.m_itab[i];
+ ndbout << " " << itab.m_name;
+ }
+ ndbout << endl;
+ }
+}
+
+static int
+runtest(Par par)
+{
+ LL1("start");
+ srandom(par.m_seed);
+ Con con;
+ CHK(con.connect() == 0);
+ par.m_con = &con;
+ g_thrlist = new Thr* [par.m_threads];
+ for (unsigned n = 0; n < par.m_threads; n++) {
+ g_thrlist[n] = 0;
+ }
+ for (unsigned n = 0; n < par.m_threads; n++) {
+ g_thrlist[n] = new Thr(par, n);
+ Thr& thr = *g_thrlist[n];
+ assert(thr.m_thread != 0);
+ }
+ for (unsigned l = 0; par.m_loop == 0 || l < par.m_loop; l++) {
+ LL1("loop " << l);
+ for (unsigned i = 0; i < tcasecount; i++) {
+ const TCase& tcase = tcaselist[i];
+ if (par.m_case != 0 && strchr(par.m_case, tcase.m_name[0]) == 0)
+ continue;
+ LL1("case " << tcase.m_name << " - " << tcase.m_desc);
+ for (unsigned j = 0; j < tabcount; j++) {
+ if (! usetable(j))
+ continue;
+ const Tab& tab = tablist[j];
+ par.m_tab = &tab;
+ Set set(tab, par.m_totrows);
+ par.m_set = &set;
+ LL1("table " << tab.m_name);
+ CHK(tcase.m_func(par) == 0);
+ }
+ }
+ }
+ for (unsigned n = 0; n < par.m_threads; n++) {
+ Thr& thr = *g_thrlist[n];
+ thr.exit();
+ }
+ for (unsigned n = 0; n < par.m_threads; n++) {
+ Thr& thr = *g_thrlist[n];
+ thr.join();
+ delete &thr;
+ }
+ delete [] g_thrlist;
+ g_thrlist = 0;
+ con.disconnect();
+ LL1("done");
+ return 0;
+}
+
+NDB_COMMAND(testOIBasic, "testOIBasic", "testOIBasic", "testOIBasic", 65535)
+{
+ while (++argv, --argc > 0) {
+ const char* arg = argv[0];
+ if (*arg != '-') {
+ ndbout << "testOIBasic: unknown argument " << arg;
+ goto usage;
+ }
+ if (strcmp(arg, "-case") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_case = strdup(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-core") == 0) {
+ g_opt.m_core = true;
+ continue;
+ }
+ if (strcmp(arg, "-dups") == 0) {
+ g_opt.m_dups = true;
+ continue;
+ }
+ if (strcmp(arg, "-fragtype") == 0) {
+ if (++argv, --argc > 0) {
+ if (strcmp(argv[0], "single") == 0) {
+ g_opt.m_fragtype = NdbDictionary::Object::FragSingle;
+ continue;
+ }
+ if (strcmp(argv[0], "small") == 0) {
+ g_opt.m_fragtype = NdbDictionary::Object::FragAllSmall;
+ continue;
+ }
+ if (strcmp(argv[0], "medium") == 0) {
+ g_opt.m_fragtype = NdbDictionary::Object::FragAllMedium;
+ continue;
+ }
+ if (strcmp(argv[0], "large") == 0) {
+ g_opt.m_fragtype = NdbDictionary::Object::FragAllLarge;
+ continue;
+ }
+ }
+ }
+ if (strcmp(arg, "-index") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_index = strdup(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-loop") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_loop = atoi(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-rows") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_rows = atoi(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-scanrd") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_scanrd = atoi(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-scanex") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_scanex = atoi(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-seed") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_seed = atoi(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-subloop") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_subloop = atoi(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-table") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_table = strdup(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-threads") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_threads = atoi(argv[0]);
+ continue;
+ }
+ }
+ if (strcmp(arg, "-v") == 0) {
+ if (++argv, --argc > 0) {
+ g_opt.m_v = atoi(argv[0]);
+ continue;
+ }
+ }
+ if (strncmp(arg, "-v", 2) == 0 && isdigit(arg[2])) {
+ g_opt.m_v = atoi(&arg[2]);
+ continue;
+ }
+ if (strcmp(arg, "-h") == 0 || strcmp(arg, "-help") == 0) {
+ printhelp();
+ goto wrongargs;
+ }
+ ndbout << "testOIBasic: unknown option " << arg;
+ goto usage;
+ }
+ {
+ Par par(g_opt);
+ if (runtest(par) < 0)
+ goto failed;
+ }
+ // always exit with NDBT code
+ok:
+ return NDBT_ProgramExit(NDBT_OK);
+failed:
+ return NDBT_ProgramExit(NDBT_FAILED);
+usage:
+ ndbout << " (use -h for help)" << endl;
+wrongargs:
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+}
+
+// vim: set sw=2 et:
diff --git a/ndb/test/ndbapi/testOIBasic/times.txt b/ndb/test/ndbapi/testOIBasic/times.txt
new file mode 100644
index 00000000000..641e9ddb4bf
--- /dev/null
+++ b/ndb/test/ndbapi/testOIBasic/times.txt
@@ -0,0 +1,8 @@
+one db-node
+testOIBasic -case t -table 1 -index 1 -fragtype small -threads 10 -rows 5000 -subloop 1
+------------------------------------------------------------
+040331
+build index - 5769 ms per 50000 ( 115 ms per 1000 )
+update - 5962 ms per 50000 ( 119 ms per 1000 )
+update indexed - 14851 ms per 50000 ( 297 ms per 1000 )
+overhead - 149 pct
diff --git a/ndb/test/ndbapi/testOperations/Makefile b/ndb/test/ndbapi/testOperations/Makefile
new file mode 100644
index 00000000000..25546ade639
--- /dev/null
+++ b/ndb/test/ndbapi/testOperations/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := testOperations
+
+SOURCES := testOperations.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testOperations/testOperations.cpp b/ndb/test/ndbapi/testOperations/testOperations.cpp
new file mode 100644
index 00000000000..bb58e69e898
--- /dev/null
+++ b/ndb/test/ndbapi/testOperations/testOperations.cpp
@@ -0,0 +1,271 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NDBT_Test.hpp"
+#include "NDBT_ReturnCodes.h"
+#include "HugoTransactions.hpp"
+#include "UtilTransactions.hpp"
+
+struct OperationTestCase {
+ const char * name;
+ bool preCond; // start transaction | insert | commit
+
+ // start transaction
+ const char * op1;
+ const int res1;
+ const int val1;
+
+ // no commit
+
+ const char * op2;
+ const int res2;
+ const int val2;
+ // Commit
+
+ // start transaction
+ // op3 = READ
+ const int res3;
+ const int val3;
+ // commit transaction
+};
+
+OperationTestCase matrix[] = {
+ { "ReadRead", true, "READ", 0, 0, "READ", 0, 0, 0, 0 },
+ { "ReadReadEx", true, "READ", 0, 0, "READ-EX", 0, 0, 0, 0 },
+ { "ReadSimpleRead", true, "READ", 0, 0, "S-READ", 0, 0, 0, 0 },
+ { "ReadDirtyRead", true, "READ", 0, 0, "D-READ", 0, 0, 0, 0 },
+ { "ReadInsert", true, "READ", 0, 0, "INSERT", 630, 1, 0, 0 },
+ { "ReadUpdate", true, "READ", 0, 0, "UPDATE", 0, 1, 0, 1 },
+ { "ReadDelete", true, "READ", 0, 0, "DELETE", 0, 0, 626, 0 },
+
+ { "FReadRead", false, "READ", 626, 0, "READ", 626, 0, 626, 0 },
+ { "FReadReadEx", false, "READ", 626, 0, "READ-EX", 626, 0, 626, 0 },
+ { "FReadSimpleRead", false, "READ", 626, 0, "S-READ", 626, 0, 626, 0 },
+ { "FReadDirtyRead", false, "READ", 626, 0, "D-READ", 626, 0, 626, 0 },
+ { "FReadInsert", false, "READ", 626, 0, "INSERT", 0, 1, 0, 1 },
+ { "FReadUpdate", false, "READ", 626, 0, "UPDATE", 626, 0, 626, 0 },
+ { "FReadDelete", false, "READ", 626, 0, "DELETE", 626, 0, 626, 0 },
+
+ { "ReadExRead", true, "READ-EX", 0, 0, "READ", 0, 0, 0, 0 },
+ { "ReadExReadEx", true, "READ-EX", 0, 0, "READ-EX", 0, 0, 0, 0 },
+ { "ReadExSimpleRead", true, "READ-EX", 0, 0, "S-READ", 0, 0, 0, 0 },
+ { "ReadExDirtyRead", true, "READ-EX", 0, 0, "D-READ", 0, 0, 0, 0 },
+ { "ReadExInsert", true, "READ-EX", 0, 0, "INSERT", 630, 1, 0, 0 },
+ { "ReadExUpdate", true, "READ-EX", 0, 0, "UPDATE", 0, 1, 0, 1 },
+ { "ReadExDelete", true, "READ-EX", 0, 0, "DELETE", 0, 0, 626, 0 },
+
+ { "InsertRead", false, "INSERT", 0, 0, "READ", 0, 0, 0, 0 },
+ { "InsertReadEx", false, "INSERT", 0, 0, "READ-EX", 0, 0, 0, 0 },
+ { "InsertSimpleRead",false, "INSERT", 0, 0, "S-READ", 0, 0, 0, 0 },
+ { "InsertDirtyRead", false, "INSERT", 0, 0, "D-READ", 0, 0, 0, 0 },
+ { "InsertInsert", false, "INSERT", 0, 0, "INSERT", 630, 0, 626, 0 },
+ { "InsertUpdate", false, "INSERT", 0, 0, "UPDATE", 0, 1, 0, 1 },
+ { "InsertDelete", false, "INSERT", 0, 0, "DELETE", 0, 0, 626, 0 },
+
+ { "UpdateRead", true, "UPDATE", 0, 1, "READ", 0, 1, 0, 1 },
+ { "UpdateReadEx", true, "UPDATE", 0, 1, "READ-EX", 0, 1, 0, 1 },
+ { "UpdateSimpleRead", true, "UPDATE", 0, 1, "S-READ", 0, 1, 0, 1 },
+ { "UpdateDirtyRead", true, "UPDATE", 0, 1, "D-READ", 0, 1, 0, 1 },
+ { "UpdateInsert", true, "UPDATE", 0, 1, "INSERT", 630, 0, 0, 0 },
+ { "UpdateUpdate", true, "UPDATE", 0, 1, "UPDATE", 0, 2, 0, 2 },
+ { "UpdateDelete", true, "UPDATE", 0, 1, "DELETE", 0, 0, 626, 0 },
+
+ { "DeleteRead", true, "DELETE", 0, 0, "READ", 626, 0, 0, 0 },
+ { "DeleteReadEx", true, "DELETE", 0, 0, "READ-EX", 626, 0, 0, 0 },
+ { "DeleteSimpleRead", true, "DELETE", 0, 0, "S-READ", 626, 0, 0, 0 },
+ { "DeleteDirtyRead", true, "DELETE", 0, 0, "D-READ", 626, 0, 0, 0 },
+ { "DeleteInsert", true, "DELETE", 0, 0, "INSERT", 0, 1, 0, 1 },
+ { "DeleteUpdate", true, "DELETE", 0, 0, "UPDATE", 626, 1, 0, 0 },
+ { "DeleteDelete", true, "DELETE", 0, 0, "DELETE", 626, 0, 0, 0 }
+};
+
+#define CHECK(b) if (!(b)) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ break; }
+
+int
+runOp(HugoOperations & hugoOps,
+ Ndb * pNdb,
+ const char * op,
+ int value){
+
+#define C2(x, y) { int r = (x); int s = (y); if(r != s) {\
+ g_err << "ERR: failed on line " << __LINE__ << ": " \
+ << r << " != " << s << endl; \
+ return NDBT_FAILED; }}
+
+ if(strcmp(op, "READ") == 0){
+ C2(hugoOps.pkReadRecord(pNdb, 1, false, 1), 0);
+ } else if(strcmp(op, "READ-EX") == 0){
+ C2(hugoOps.pkReadRecord(pNdb, 1, true, 1), 0);
+ } else if(strcmp(op, "S-READ") == 0){
+ C2(hugoOps.pkSimpleReadRecord(pNdb, 1, 1), 0);
+ } else if(strcmp(op, "D-READ") == 0){
+ C2(hugoOps.pkDirtyReadRecord(pNdb, 1, 1), 0);
+ } else if(strcmp(op, "INSERT") == 0){
+ C2(hugoOps.pkInsertRecord(pNdb, 1, 1, value), 0);
+ } else if(strcmp(op, "UPDATE") == 0){
+ C2(hugoOps.pkUpdateRecord(pNdb, 1, 1, value), 0);
+ } else if(strcmp(op, "DELETE") == 0){
+ C2(hugoOps.pkDeleteRecord(pNdb, 1, 1), 0);
+ } else {
+ g_err << __FILE__ << " - " << __LINE__
+ << ": Unknown operation" << op << endl;
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+int
+checkVal(HugoOperations & hugoOps,
+ const char * op,
+ int value,
+ int result){
+ if(result != 0)
+ return NDBT_OK;
+
+ if(strcmp(op, "READ") == 0){
+ } else if(strcmp(op, "READ-EX") == 0){
+ } else if(strcmp(op, "S-READ") == 0){
+ } else if(strcmp(op, "D-READ") == 0){
+ } else {
+ return NDBT_OK;
+ }
+
+ return hugoOps.verifyUpdatesValue(value);
+}
+
+int
+runTwoOperations(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ const char * op1 = ctx->getProperty("op1", "NONE");
+ const int val1 = ctx->getProperty("val1", ~0);
+ const int res1 = ctx->getProperty("res1", ~0);
+ const char * op2 = ctx->getProperty("op2", "NONE");
+ const int res2 = ctx->getProperty("res2", ~0);
+ const int val2 = ctx->getProperty("val2", ~0);
+
+ const int res3 = ctx->getProperty("res3", ~0);
+ const int val3 = ctx->getProperty("val3", ~0);
+
+ do {
+ // Insert, read
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(runOp(hugoOps, pNdb, op1, val1) == 0);
+ AbortOption oa = (res1 == 0) ? AbortOnError : IgnoreError;
+ CHECK(hugoOps.execute_NoCommit(pNdb, oa) == res1);
+ CHECK(checkVal(hugoOps, op1, val1, res1) == 0);
+
+ ndbout_c("-- running op 2");
+
+ CHECK(runOp(hugoOps, pNdb, op2, val2) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == res2);
+ CHECK(checkVal(hugoOps, op2, val2, res2) == 0);
+
+ } while(false);
+ hugoOps.closeTransaction(pNdb);
+
+ if(result != NDBT_OK)
+ return result;
+
+ do {
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(runOp(hugoOps, pNdb, "READ", 0) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == res3);
+ CHECK(checkVal(hugoOps, "READ", val3, res3) == 0);
+ } while(false);
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int
+runInsertRecord(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Insert, insert
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, 1) == 0);
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+
+ } while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int
+runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records, 240) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int
+main(int argc, const char** argv){
+
+ NDBT_TestSuite ts("testOperations");
+ for(Uint32 i = 0; i<sizeof(matrix)/sizeof(matrix[0]); i++){
+ NDBT_TestCaseImpl1 *pt = new NDBT_TestCaseImpl1(&ts, matrix[i].name, "");
+
+ pt->addInitializer(new NDBT_Initializer(pt,
+ "runClearTable",
+ runClearTable));
+
+ if(matrix[i].preCond){
+ pt->addInitializer(new NDBT_Initializer(pt,
+ "runInsertRecord",
+ runInsertRecord));
+ }
+
+ pt->setProperty("op1", matrix[i].op1);
+ pt->setProperty("res1", matrix[i].res1);
+ pt->setProperty("val1", matrix[i].val1);
+
+ pt->setProperty("op2", matrix[i].op2);
+ pt->setProperty("res2", matrix[i].res2);
+ pt->setProperty("val2", matrix[i].val2);
+
+ pt->setProperty("res3", matrix[i].res3);
+ pt->setProperty("val3", matrix[i].val3);
+
+ pt->addStep(new NDBT_ParallelStep(pt,
+ matrix[i].name,
+ runTwoOperations));
+ pt->addFinalizer(new NDBT_Finalizer(pt,
+ "runClearTable",
+ runClearTable));
+
+ ts.addTest(pt);
+ }
+
+ return ts.execute(argc, argv);
+}
+
diff --git a/ndb/test/ndbapi/testOrderedIndex/Makefile b/ndb/test/ndbapi/testOrderedIndex/Makefile
new file mode 100644
index 00000000000..d8899a37895
--- /dev/null
+++ b/ndb/test/ndbapi/testOrderedIndex/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testOrderedIndex
+
+SOURCES = testOrderedIndex.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testOrderedIndex/testOrderedIndex.cpp b/ndb/test/ndbapi/testOrderedIndex/testOrderedIndex.cpp
new file mode 100644
index 00000000000..51cc53c9975
--- /dev/null
+++ b/ndb/test/ndbapi/testOrderedIndex/testOrderedIndex.cpp
@@ -0,0 +1,224 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <Vector.hpp>
+#include <ndbapi_limits.h>
+
+const unsigned MaxTableAttrs = NDB_MAX_ATTRIBUTES_IN_TABLE;
+const unsigned MaxIndexAttrs = NDB_MAX_ATTRIBUTES_IN_INDEX;
+const unsigned MaxIndexes = 20;
+
+static unsigned
+urandom(unsigned n)
+{
+ unsigned i = random();
+ return i % n;
+}
+
+static int
+runDropIndex(NDBT_Context* ctx, NDBT_Step* step)
+{
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+ NdbDictionary::Dictionary* pDic = pNdb->getDictionary();
+ NdbDictionary::Dictionary::List list;
+ if (pDic->listIndexes(list, pTab->getName()) != 0) {
+ g_err << pTab->getName() << ": listIndexes failed" << endl;
+ ERR(pDic->getNdbError());
+ return NDBT_FAILED;
+ }
+ for (unsigned i = 0; i < list.count; i++) {
+ NDBT_Index* pInd = new NDBT_Index(list.elements[i].name);
+ pInd->setTable(pTab->getName());
+ g_info << "Drop index:" << endl << *pInd;
+ if (pInd->dropIndexInDb(pNdb) != 0) {
+ return NDBT_FAILED;
+ }
+ }
+ return NDBT_OK;
+}
+
+static Uint32 workaround[1000];
+
+static void
+setTableProperty(NDBT_Context* ctx, NDBT_Table* pTab, const char* name, Uint32 num)
+{
+ char key[200];
+ sprintf(key, "%s-%s", name, pTab->getName());
+ //ctx->setProperty(key, num);
+ workaround[pTab->getTableId()] = num;
+}
+
+static Uint32
+getTableProperty(NDBT_Context* ctx, NDBT_Table* pTab, const char* name)
+{
+ char key[200];
+ sprintf(key, "%s-%s", name, pTab->getName());
+ //Uint32 num = ctx->getProperty(key, (Uint32)-1);
+ Uint32 num = workaround[pTab->getTableId()];
+ assert(num != (Uint32)-1);
+ return num;
+}
+
+static int
+runCreateIndex(NDBT_Context* ctx, NDBT_Step* step)
+{
+ srandom(1);
+ NDBT_Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+ unsigned numTabAttrs = pTab->getNumAttributes();
+ unsigned numIndex = 0;
+ while (numIndex < MaxIndexes) {
+ if (numIndex != 0 && urandom(10) == 0)
+ break;
+ char buf[200];
+ sprintf(buf, "%s_X%03d", pTab->getName(), numIndex);
+ NDBT_Index* pInd = new NDBT_Index(buf);
+ pInd->setTable(pTab->getName());
+ pInd->setType(NdbDictionary::Index::OrderedIndex);
+ pInd->setLogging(false);
+ unsigned numAttrs = 0;
+ while (numAttrs < MaxIndexAttrs) {
+ if (numAttrs != 0 && urandom(5) == 0)
+ break;
+ unsigned i = urandom(numTabAttrs);
+ const NDBT_Attribute* pAttr = pTab->getAttribute(i);
+ bool found = false;
+ for (unsigned j = 0; j < numAttrs; j++) {
+ if (strcmp(pAttr->getName(), pInd->getAttribute(j)->getName()) == 0) {
+ found = true;
+ break;
+ }
+ }
+ if (found)
+ continue;
+ pInd->addAttribute(*pAttr);
+ numAttrs++;
+ }
+ g_info << "Create index:" << endl << *pInd;
+ if (pInd->createIndexInDb(pNdb, false) != 0)
+ continue;
+ numIndex++;
+ }
+ setTableProperty(ctx, pTab, "numIndex", numIndex);
+ g_info << "Created " << numIndex << " indexes on " << pTab->getName() << endl;
+ return NDBT_OK;
+}
+
+static int
+runInsertUpdate(NDBT_Context* ctx, NDBT_Step* step)
+{
+ NDBT_Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+ int ret;
+ g_info << "Insert: " << pTab->getName() << endl;
+ HugoTransactions hugoTrans(*pTab);
+ ret = hugoTrans.loadTable(pNdb, ctx->getNumRecords(), 100);
+ if (ret != 0) {
+ g_err << "ERR: " << step->getName() << "failed" << endl;
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+static int
+runFullScan(NDBT_Context* ctx, NDBT_Step* step)
+{
+ NDBT_Table* pTab = ctx->getTab();
+ Ndb* pNdb = GETNDB(step);
+ unsigned cntIndex = getTableProperty(ctx, pTab, "numIndex");
+ for (unsigned numIndex = 0; numIndex < cntIndex; numIndex++) {
+ char buf[200];
+ sprintf(buf, "%s_X%03d", pTab->getName(), numIndex);
+ NDBT_Index* pInd = NDBT_Index::discoverIndexFromDb(pNdb, buf, pTab->getName());
+ assert(pInd != 0);
+ g_info << "Scan index:" << pInd->getName() << endl << *pInd;
+ NdbConnection* pCon = pNdb->startTransaction();
+ if (pCon == 0) {
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+ NdbOperation* pOp = pCon->getNdbOperation(pInd->getName(),
+ pTab->getName());
+ if (pOp == 0) {
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ return NDBT_FAILED;
+ }
+ if (pOp->openScanRead() != 0) {
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ return NDBT_FAILED;
+ }
+ if (pCon->executeScan() != 0) {
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ return NDBT_FAILED;
+ }
+ unsigned rows = 0;
+ while (1) {
+ int ret = pCon->nextScanResult();
+ if (ret == 0) {
+ rows++;
+ } else if (ret == 1) {
+ break;
+ } else {
+ ERR(pCon->getNdbError());
+ pNdb->closeTransaction(pCon);
+ return NDBT_FAILED;
+ }
+ }
+ pNdb->closeTransaction(pCon);
+ g_info << "Scanned " << rows << " rows" << endl;
+ }
+ return NDBT_OK;
+}
+
+NDBT_TESTSUITE(testOrderedIndex);
+TESTCASE(
+ "DropIndex",
+ "Drop any old indexes") {
+ INITIALIZER(runDropIndex);
+}
+TESTCASE(
+ "CreateIndex",
+ "Create ordered indexes") {
+ INITIALIZER(runCreateIndex);
+}
+TESTCASE(
+ "InsertUpdate",
+ "Run inserts and updates") {
+ INITIALIZER(runInsertUpdate);
+}
+TESTCASE(
+ "FullScan",
+ "Full scan on each ordered index") {
+ INITIALIZER(runFullScan);
+}
+NDBT_TESTSUITE_END(testOrderedIndex);
+
+int
+main(int argc, const char** argv)
+{
+ return testOrderedIndex.execute(argc, argv);
+}
+
+// vim: set sw=2:
diff --git a/ndb/test/ndbapi/testRestartGci/Makefile b/ndb/test/ndbapi/testRestartGci/Makefile
new file mode 100644
index 00000000000..24f449b747d
--- /dev/null
+++ b/ndb/test/ndbapi/testRestartGci/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := testRestartGci
+
+SOURCES := testRestartGci.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testRestartGci/testRestartGci.cpp b/ndb/test/ndbapi/testRestartGci/testRestartGci.cpp
new file mode 100644
index 00000000000..1e36368ba62
--- /dev/null
+++ b/ndb/test/ndbapi/testRestartGci/testRestartGci.cpp
@@ -0,0 +1,218 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NDBT_Test.hpp"
+#include "NDBT_ReturnCodes.h"
+#include "HugoTransactions.hpp"
+#include "UtilTransactions.hpp"
+#include "NdbRestarter.hpp"
+
+
+/**
+ * Global vector to keep track of
+ * records stored in db
+ */
+
+struct SavedRecord {
+ int m_gci;
+ BaseString m_str;
+ SavedRecord(int _gci, BaseString _str){
+ m_gci = _gci;
+ m_str.assign(_str);
+ }
+ SavedRecord(){
+ m_gci = 0;
+ m_str = "";
+ };
+};
+Vector<SavedRecord> savedRecords;
+
+
+#define CHECK(b) if (!(b)) { \
+ ndbout << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ break; }
+
+int runInsertRememberGci(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int records = ctx->getNumRecords();
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+ int i = 0;
+
+ while(ctx->isTestStopped() == false && i < records){
+ // Insert record and read it in same transaction
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, i) == 0);
+ if (hugoOps.execute_NoCommit(pNdb) != 0){
+ ndbout << "Could not insert record " << i << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+ CHECK(hugoOps.pkReadRecord(pNdb, i, false) == 0);
+ if (hugoOps.execute_Commit(pNdb) != 0){
+ ndbout << "Did not find record in DB " << i << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+ savedRecords.push_back(SavedRecord(hugoOps.getRecordGci(0),
+ hugoOps.getRecordStr(0)));
+
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+ i++;
+ };
+
+ return result;
+}
+
+int runRestartGciControl(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ Ndb* pNdb = GETNDB(step);
+ UtilTransactions utilTrans(*ctx->getTab());
+ NdbRestarter restarter;
+
+ // Wait until we have enough records in db
+ int count = 0;
+ while (count < records){
+ if (utilTrans.selectCount(pNdb, 64, &count) != 0){
+ ctx->stopTest();
+ return NDBT_FAILED;
+ }
+ }
+
+ // Restart cluster with abort
+ if (restarter.restartAll(false, false, true) != 0){
+ ctx->stopTest();
+ return NDBT_FAILED;
+ }
+
+ // Stop the other thread
+ ctx->stopTest();
+
+ if (restarter.waitClusterStarted(300) != 0){
+ return NDBT_FAILED;
+ }
+
+ if (pNdb->waitUntilReady() != 0){
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+int runVerifyInserts(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ Ndb* pNdb = GETNDB(step);
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoOperations hugoOps(*ctx->getTab());
+ NdbRestarter restarter;
+
+ int restartGCI = pNdb->NdbTamper(ReadRestartGCI, 0);
+
+ ndbout << "restartGCI = " << restartGCI << endl;
+ int count = 0;
+ if (utilTrans.selectCount(pNdb, 64, &count) != 0){
+ return NDBT_FAILED;
+ }
+
+ // RULE1: The vector with saved records should have exactly as many
+ // records with lower or same gci as there are in DB
+ int recordsWithLowerOrSameGci = 0;
+ for (unsigned i = 0; i < savedRecords.size(); i++){
+ if (savedRecords[i].m_gci <= restartGCI)
+ recordsWithLowerOrSameGci++;
+ }
+ if (recordsWithLowerOrSameGci != count){
+ ndbout << "ERR: Wrong number of expected records" << endl;
+ result = NDBT_FAILED;
+ }
+
+
+ // RULE2: The records found in db should have same or lower
+ // gci as in the vector
+ for (unsigned i = 0; i < savedRecords.size(); i++){
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, i, false) == 0);
+ if (hugoOps.execute_Commit(pNdb) != 0){
+ // Record was not found in db'
+
+ // Check record gci
+ if (savedRecords[i].m_gci <= restartGCI){
+ ndbout << "ERR: Record "<<i<<" should have existed" << endl;
+ result = NDBT_FAILED;
+ }
+ } else {
+ // Record was found in db
+ BaseString str = hugoOps.getRecordStr(0);
+ // Check record string
+ if (!(savedRecords[i].m_str == str)){
+ ndbout << "ERR: Record "<<i<<" str did not match "<< endl;
+ result = NDBT_FAILED;
+ }
+ // Check record gci
+ if (savedRecords[i].m_gci > restartGCI){
+ ndbout << "ERR: Record "<<i<<" should not have existed" << endl;
+ result = NDBT_FAILED;
+ }
+ }
+
+ CHECK(hugoOps.closeTransaction(pNdb) == 0);
+ }
+
+
+ ndbout << "There are " << count << " records in db" << endl;
+ ndbout << "There are " << savedRecords.size()
+ << " records in vector" << endl;
+
+ ndbout << "There are " << recordsWithLowerOrSameGci
+ << " records with lower or same gci than " << restartGCI << endl;
+
+ return result;
+}
+
+int runClearGlobals(NDBT_Context* ctx, NDBT_Step* step){
+ savedRecords.clear();
+ return NDBT_OK;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records, 240) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+NDBT_TESTSUITE(testRestartGci);
+TESTCASE("InsertRestartGci",
+ "Verify that only expected records are still in NDB\n"
+ "after a restart" ){
+ INITIALIZER(runClearTable);
+ INITIALIZER(runClearGlobals);
+ STEP(runInsertRememberGci);
+ STEP(runRestartGciControl);
+ VERIFIER(runVerifyInserts);
+ FINALIZER(runClearTable);
+}
+NDBT_TESTSUITE_END(testRestartGci);
+
+int main(int argc, const char** argv){
+ return testRestartGci.execute(argc, argv);
+}
diff --git a/ndb/test/ndbapi/testScan/Makefile b/ndb/test/ndbapi/testScan/Makefile
new file mode 100644
index 00000000000..fe48f5bc926
--- /dev/null
+++ b/ndb/test/ndbapi/testScan/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testScan
+
+SOURCES = testScan.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testScan/ScanFunctions.hpp b/ndb/test/ndbapi/testScan/ScanFunctions.hpp
new file mode 100644
index 00000000000..36d01909861
--- /dev/null
+++ b/ndb/test/ndbapi/testScan/ScanFunctions.hpp
@@ -0,0 +1,392 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+
+
+
+struct Attrib {
+ int numAttribs;
+ int attribs[1024];
+};
+class AttribList {
+public:
+ AttribList(){};
+ ~AttribList(){
+ for(size_t i = 0; i < attriblist.size(); i++){
+ delete attriblist[i];
+ }
+ };
+ void buildAttribList(const NdbDictionary::Table* pTab);
+ Vector<Attrib*> attriblist;
+};
+
+
+// Functions that help out in testing that we may call
+// scan functions in wrong order etc
+// and receive a proper errormessage
+class ScanFunctions {
+public:
+ ScanFunctions(const NdbDictionary::Table& _tab) : tab(_tab){
+ }
+ enum ActionType {
+ CloseWithoutStop,
+ NextScanWhenNoMore,
+ ExecuteScanWithOutOpenScan,
+ OnlyOneScanPerTrans,
+ OnlyOneOpBeforeOpenScan,
+ OnlyOpenScanOnce,
+ OnlyOneOpInScanTrans,
+ CheckInactivityTimeOut,
+ CheckInactivityBeforeClose ,
+ NoCloseTransaction,
+ EqualAfterOpenScan
+ };
+
+
+ int scanReadFunctions(Ndb* pNdb,
+ int records,
+ int parallelism,
+ ActionType action,
+ bool exclusive);
+private:
+ const NdbDictionary::Table& tab;
+};
+
+
+inline
+int
+ScanFunctions::scanReadFunctions(Ndb* pNdb,
+ int records,
+ int parallelism,
+ ActionType action,
+ bool exclusive){
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int sleepTime = 10;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ while (true){
+ if (retryAttempt >= retryMax){
+ g_err << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ // Execute the scan without defining a scan operation
+ if(action != ExecuteScanWithOutOpenScan){
+
+ if (action == OnlyOneOpBeforeOpenScan){
+ // There can only be one operation defined when calling openScan
+ NdbOperation* pOp3;
+ pOp3 = pTrans->getNdbOperation(tab.getName());
+ if (pOp3 == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if (exclusive == true)
+ check = pOp->openScanExclusive(parallelism);
+ else
+ check = pOp->openScanRead(parallelism);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+
+ if (action == OnlyOneScanPerTrans){
+ // There can only be one operation in a scan transaction
+ NdbOperation* pOp4;
+ pOp4 = pTrans->getNdbOperation(tab.getName());
+ if (pOp4 == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ if (action == OnlyOpenScanOnce){
+ // Call openScan one more time when it's already defined
+ check = pOp->openScanRead(parallelism);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ if (action == OnlyOneOpInScanTrans){
+ // Try to add another op to this scanTransaction
+ NdbOperation* pOp2;
+ pOp2 = pTrans->getNdbOperation(tab.getName());
+ if (pOp2 == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+
+ if (action==EqualAfterOpenScan){
+ check = pOp->equal(tab.getColumn(0)->getName(), 10);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if(pOp->getValue(tab.getColumn(a)->getName()) == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+
+ int abortCount = records / 10;
+ bool abortTrans = (action==CloseWithoutStop);
+ int eof;
+ int rows = 0;
+ eof = pTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+
+ if (abortCount == rows && abortTrans == true){
+ g_info << "Scan is aborted after "<<abortCount<<" rows" << endl;
+
+ if (action != CloseWithoutStop){
+ // Test that we can closeTrans without stopScan
+ check = pTrans->stopScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+
+ pNdb->closeTransaction(pTrans);
+ return NDBT_OK;
+ }
+
+ if(action == CheckInactivityTimeOut){
+ if ((rows % (records / 10)) == 0){
+ // Sleep for a long time before calling nextScanResult
+ if (sleepTime > 1)
+ sleepTime--;
+ g_info << "Sleeping "<<sleepTime<<" secs " << endl;
+ NdbSleep_SecSleep(sleepTime);
+ }
+ }
+
+ eof = pTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+
+ // Be cruel, call nextScanResult after error
+ for(int i=0; i<10; i++){
+ eof =pTrans->nextScanResult();
+ if(eof == 0){
+ g_err << "nextScanResult returned eof = " << eof << endl
+ << " That is an error when there are no more records" << endl;
+ return NDBT_FAILED;
+ }
+ }
+ // Be cruel end
+
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ g_info << "Starting over" << endl;
+
+ // If test is CheckInactivityTimeOut
+ // error 296 is expected
+ if ((action == CheckInactivityTimeOut) &&
+ (err.code == 296))
+ return NDBT_OK;
+
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if (action == NextScanWhenNoMore){
+ g_info << "Calling nextScanresult when there are no more records" << endl;
+ for(int i=0; i<10; i++){
+ eof =pTrans->nextScanResult();
+ if(eof == 0){
+ g_err << "nextScanResult returned eof = " << eof << endl
+ << " That is an error when there are no more records" << endl;
+ return NDBT_FAILED;
+ }
+ }
+
+ }
+ if(action ==CheckInactivityBeforeClose){
+ // Sleep for a long time before calling close
+ g_info << "NdbSleep_SecSleep(5) before close transaction" << endl;
+ NdbSleep_SecSleep(5);
+ }
+ if(action == NoCloseTransaction)
+ g_info << "Forgetting to close transaction" << endl;
+ else
+ pNdb->closeTransaction(pTrans);
+
+ g_info << rows << " rows have been read" << endl;
+ if (records != 0 && rows != records){
+ g_err << "Check expected number of records failed" << endl
+ << " expected=" << records <<", " << endl
+ << " read=" << rows << endl;
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+
+
+}
+
+void AttribList::buildAttribList(const NdbDictionary::Table* pTab){
+ attriblist.clear();
+
+ Attrib* attr;
+ // Build attrib definitions that describes which attributes to read
+ // Try to build strange combinations, not just "all" or all PK's
+
+ // Scan without reading any attributes
+ attr = new Attrib;
+ attr->numAttribs = 0;
+ attriblist.push_back(attr);
+
+ for(int i = 1; i < pTab->getNoOfColumns(); i++){
+ attr = new Attrib;
+ attr->numAttribs = i;
+ for(int a = 0; a<i; a++)
+ attr->attribs[a] = a;
+ attriblist.push_back(attr);
+ }
+ for(int i = pTab->getNoOfColumns()-1; i > 0; i--){
+ attr = new Attrib;
+ attr->numAttribs = i;
+ for(int a = 0; a<i; a++)
+ attr->attribs[a] = a;
+ attriblist.push_back(attr);
+ }
+ for(int i = pTab->getNoOfColumns(); i > 0; i--){
+ attr = new Attrib;
+ attr->numAttribs = pTab->getNoOfColumns() - i;
+ for(int a = 0; a<pTab->getNoOfColumns() - i; a++)
+ attr->attribs[a] = pTab->getNoOfColumns()-a-1;
+ attriblist.push_back(attr);
+ }
+ for(int i = 1; i < pTab->getNoOfColumns(); i++){
+ attr = new Attrib;
+ attr->numAttribs = pTab->getNoOfColumns() - i;
+ for(int a = 0; a<pTab->getNoOfColumns() - i; a++)
+ attr->attribs[a] = pTab->getNoOfColumns()-a-1;
+ attriblist.push_back(attr);
+ }
+ for(int i = 1; i < pTab->getNoOfColumns(); i++){
+ attr = new Attrib;
+ attr->numAttribs = 2;
+ for(int a = 0; a<2; a++){
+ attr->attribs[a] = i%pTab->getNoOfColumns();
+ }
+ attriblist.push_back(attr);
+ }
+
+ // Last
+ attr = new Attrib;
+ attr->numAttribs = 1;
+ attr->attribs[0] = pTab->getNoOfColumns()-1;
+ attriblist.push_back(attr);
+
+ // Last and first
+ attr = new Attrib;
+ attr->numAttribs = 2;
+ attr->attribs[0] = pTab->getNoOfColumns()-1;
+ attr->attribs[1] = 0;
+ attriblist.push_back(attr);
+
+ // First and last
+ attr = new Attrib;
+ attr->numAttribs = 2;
+ attr->attribs[0] = 0;
+ attr->attribs[1] = pTab->getNoOfColumns()-1;
+ attriblist.push_back(attr);
+
+#if 1
+ for(size_t i = 0; i < attriblist.size(); i++){
+
+ g_info << attriblist[i]->numAttribs << ": " ;
+ for(int a = 0; a < attriblist[i]->numAttribs; a++)
+ g_info << attriblist[i]->attribs[a] << ", ";
+ g_info << endl;
+ }
+#endif
+
+}
diff --git a/ndb/test/ndbapi/testScan/testScan.cpp b/ndb/test/ndbapi/testScan/testScan.cpp
new file mode 100644
index 00000000000..c48b41ee0b9
--- /dev/null
+++ b/ndb/test/ndbapi/testScan/testScan.cpp
@@ -0,0 +1,1293 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <Vector.hpp>
+#include "ScanFunctions.hpp"
+#include <random.h>
+
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+int runCreateAllTables(NDBT_Context* ctx, NDBT_Step* step){
+
+ return NDBT_Tables::createAllTables(GETNDB(step), false, true);
+}
+
+int runDropAllTablesExceptTestTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ for (int i=0; i < NDBT_Tables::getNumTables(); i++){
+
+ const NdbDictionary::Table* tab = NDBT_Tables::getTable(i);
+ if (tab == NULL){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ // Don't drop test table
+ if (strcmp(tab->getName(), ctx->getTab()->getName()) == 0){
+ continue;
+ }
+
+ int res = GETNDB(step)->getDictionary()->dropTable(tab->getName());
+ if(res != -1){
+ return NDBT_FAILED;
+ }
+ }
+ return NDBT_OK;
+}
+
+
+int runLoadAllTables(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ for (int i=0; i < NDBT_Tables::getNumTables(); i++){
+
+ const NdbDictionary::Table* tab = NDBT_Tables::getTable(i);
+ if (tab == NULL){
+ return NDBT_FAILED;
+ }
+ HugoTransactions hugoTrans(*tab);
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ }
+ return NDBT_OK;
+}
+
+int runScanReadRandomTable(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int parallelism = ctx->getProperty("Parallelism", 240);
+ int abort = ctx->getProperty("AbortProb");
+
+ int i = 0;
+ while (i<loops) {
+
+ int tabNum = myRandom48(NDBT_Tables::getNumTables());
+ const NdbDictionary::Table* tab = NDBT_Tables::getTable(tabNum);
+ if (tab == NULL){
+ g_info << "tab == NULL" << endl;
+ return NDBT_FAILED;
+ }
+
+ g_info << "Scan reading from table " << tab->getName() << endl;
+ HugoTransactions hugoTrans(*tab);
+
+ g_info << i << ": ";
+ if (hugoTrans.scanReadRecords(GETNDB(step), records, abort, parallelism) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runInsertUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (hugoTrans.loadTable(GETNDB(step), records, 1) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runInsertDelete(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int records = ctx->getNumRecords();
+ int loops = ctx->getNumLoops();
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ UtilTransactions utilTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (hugoTrans.loadTable(GETNDB(step), records, 1) != 0){
+ result = NDBT_FAILED;
+ break;
+ }
+ if (utilTrans.clearTable(GETNDB(step), records) != 0){
+ result = NDBT_FAILED;
+ break;
+ }
+ i++;
+ }
+
+ ctx->stopTest();
+
+ return result;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runScanDelete(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+
+ int i = 0;
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (utilTrans.clearTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ // Load table, don't allow any primary key violations
+ if (hugoTrans.loadTable(GETNDB(step), records, 512, false) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runScanDelete2(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+
+ int i = 0;
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ while (i<loops) {
+ g_info << i << ": ";
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ // Load table, don't allow any primary key violations
+ if (hugoTrans.loadTable(GETNDB(step), records, 512, false) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runVerifyTable(NDBT_Context* ctx, NDBT_Step* step){
+ return NDBT_OK;
+}
+
+int runScanRead(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int parallelism = ctx->getProperty("Parallelism", 240);
+ int abort = ctx->getProperty("AbortProb");
+
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (hugoTrans.scanReadRecords(GETNDB(step), records, abort, parallelism) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runScanReadCommitted(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int parallelism = ctx->getProperty("Parallelism", 240);
+ int abort = ctx->getProperty("AbortProb");
+
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (hugoTrans.scanReadCommittedRecords(GETNDB(step), records,
+ abort, parallelism) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runScanReadError(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int parallelism = 240; // Max parallelism
+ int error = ctx->getProperty("ErrorCode");
+ NdbRestarter restarter;
+
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+
+ ndbout << "insertErrorInAllNodes("<<error<<")"<<endl;
+ if (restarter.insertErrorInAllNodes(error) != 0){
+ ndbout << "Could not insert error in all nodes "<<endl;
+ return NDBT_FAILED;
+ }
+
+ if (hugoTrans.scanReadRecords(GETNDB(step), records, 0, parallelism) != 0){
+ result = NDBT_FAILED;
+ }
+ i++;
+ }
+
+ restarter.insertErrorInAllNodes(0);
+ return result;
+}
+
+int runScanReadErrorOneNode(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int parallelism = 240; // Max parallelism
+ int error = ctx->getProperty("ErrorCode");
+ NdbRestarter restarter;
+ int lastId = 0;
+
+ if (restarter.getNumDbNodes() < 2){
+ ctx->stopTest();
+ return NDBT_OK;
+ }
+
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops && result == NDBT_OK) {
+ g_info << i << ": ";
+
+ int nodeId = restarter.getDbNodeId(lastId);
+ lastId = (lastId + 1) % restarter.getNumDbNodes();
+ ndbout << "insertErrorInNode("<<nodeId<<", "<<error<<")"<<endl;
+ if (restarter.insertErrorInNode(nodeId, error) != 0){
+ ndbout << "Could not insert error in node="<<nodeId<<endl;
+ return NDBT_FAILED;
+ }
+
+ for (int j=0; j<10; j++){
+ if (hugoTrans.scanReadRecords(GETNDB(step),
+ records, 0, parallelism) != 0)
+ result = NDBT_FAILED;
+ }
+
+
+ if(restarter.waitClusterStarted(120) != 0){
+ g_err << "Cluster failed to restart" << endl;
+ result = NDBT_FAILED;
+ }
+ restarter.insertErrorInAllNodes(0);
+
+ i++;
+ }
+ restarter.insertErrorInAllNodes(0);
+ return result;
+}
+
+int runRestartAll(NDBT_Context* ctx, NDBT_Step* step){
+
+ NdbRestarter restarter;
+
+ if (restarter.restartAll() != 0){
+ ndbout << "Could not restart all nodes"<<endl;
+ return NDBT_FAILED;
+ }
+
+ if (restarter.waitClusterStarted(120) != 0){
+ ndbout << "Could not restarted" << endl;
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+static int RANDOM_PARALLELISM = 9999;
+
+int runScanReadUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int i = 0;
+
+ int parallelism = ctx->getProperty("Parallelism", 240);
+ int para = parallelism;
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ if (parallelism == RANDOM_PARALLELISM)
+ para = myRandom48(239)+1;
+
+ g_info << i << ": ";
+ if (hugoTrans.scanReadRecords(GETNDB(step), records, 0, para) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runScanReadUntilStoppedNoCount(NDBT_Context* ctx, NDBT_Step* step){
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ g_info << i << ": ";
+ if (hugoTrans.scanReadRecords(GETNDB(step), 0) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+
+int runPkRead(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (hugoTrans.pkReadRecords(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runScanUpdate(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int parallelism = ctx->getProperty("Parallelism", 1);
+ int abort = ctx->getProperty("AbortProb");
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+
+ if (hugoTrans.scanUpdateRecords(GETNDB(step), records, abort, parallelism) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runScanUpdateUntilStopped(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int i = 0;
+
+ int parallelism = ctx->getProperty("Parallelism", 240);
+ int para = parallelism;
+
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (ctx->isTestStopped() == false) {
+ if (parallelism == RANDOM_PARALLELISM)
+ para = myRandom48(239)+1;
+
+ g_info << i << ": ";
+ if (hugoTrans.scanUpdateRecords(GETNDB(step), records, 0, para) == NDBT_FAILED){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+
+int runScanUpdate2(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int parallelism = ctx->getProperty("Parallelism", 240);
+ int abort = ctx->getProperty("AbortProb");
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (hugoTrans.scanUpdateRecords2(GETNDB(step), records, abort, parallelism) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runLocker(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ if (hugoTrans.lockRecords(GETNDB(step), records, 5, 500) != 0){
+ result = NDBT_FAILED;
+ }
+ ctx->stopTest();
+
+ return result;
+}
+
+int runRestarter(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ NdbRestarter restarter;
+ int i = 0;
+ int lastId = 0;
+ int timeout = 240;
+
+ if (restarter.getNumDbNodes() < 2){
+ ctx->stopTest();
+ return NDBT_OK;
+ }
+ while(i<loops && result != NDBT_FAILED){
+ if(restarter.waitClusterStarted(timeout) != 0){
+ g_err << "Cluster failed to start 1" << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+ NdbSleep_SecSleep(10);
+
+ int nodeId = restarter.getDbNodeId(lastId);
+ lastId = (lastId + 1) % restarter.getNumDbNodes();
+ if(restarter.restartOneDbNode(nodeId) != 0){
+ g_err << "Failed to restartNextDbNode" << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+ i++;
+ }
+ if(restarter.waitClusterStarted(timeout) != 0){
+ g_err << "Cluster failed to start 2" << endl;
+ result = NDBT_FAILED;
+ }
+
+ ctx->stopTest();
+
+ return result;
+}
+
+int runRestarter9999(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ NdbRestarter restarter;
+ int i = 0;
+ int lastId = 0;
+
+ if (restarter.getNumDbNodes() < 2){
+ ctx->stopTest();
+ return NDBT_OK;
+ }
+ while(i<loops && result != NDBT_FAILED){
+ if(restarter.waitClusterStarted(120) != 0){
+ g_err << "Cluster failed to start" << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+ NdbSleep_SecSleep(10);
+
+ int nodeId = restarter.getDbNodeId(lastId);
+ lastId = (lastId + 1) % restarter.getNumDbNodes();
+ if(restarter.insertErrorInNode(nodeId, 9999) != 0){
+ g_err << "Failed to insertErrorInNode="<<nodeId << endl;
+ result = NDBT_FAILED;
+ break;
+ }
+ NdbSleep_SecSleep(10);
+ i++;
+ }
+ if(restarter.waitClusterStarted(120) != 0){
+ g_err << "Cluster failed to start" << endl;
+ result = NDBT_FAILED;
+ }
+
+ ctx->stopTest();
+
+ return result;
+}
+
+
+int runCheckGetValue(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int parallelism = ctx->getProperty("Parallelism", 1);
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+ AttribList alist;
+ alist.buildAttribList(pTab);
+ UtilTransactions utilTrans(*pTab);
+ for(size_t i = 0; i < alist.attriblist.size(); i++){
+ g_info << (unsigned)i << endl;
+ if(utilTrans.scanReadRecords(GETNDB(step),
+ parallelism,
+ false,
+ records,
+ alist.attriblist[i]->numAttribs,
+ alist.attriblist[i]->attribs) != 0){
+ numFailed++;
+ }
+ if(utilTrans.scanReadRecords(GETNDB(step),
+ parallelism,
+ true,
+ records,
+ alist.attriblist[i]->numAttribs,
+ alist.attriblist[i]->attribs) != 0){
+ numFailed++;
+ }
+ }
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+}
+
+int runCloseWithoutStop(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+ ScanFunctions scanF(*pTab);
+ // Iterate over all possible parallelism valuse
+ for (int p = 1; p<240; p++){
+ g_info << p << " CloseWithoutStop openScan" << endl;
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ p,
+ ScanFunctions::CloseWithoutStop,
+ false) != 0){
+ numFailed++;
+ }
+ g_info << p << " CloseWithoutStop openScanExclusive" << endl;
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ p,
+ ScanFunctions::CloseWithoutStop,
+ true) != 0){
+ numFailed++;
+ }
+ }
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+}
+
+int runNextScanWhenNoMore(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+ ScanFunctions scanF(*pTab);
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::NextScanWhenNoMore,
+ false) != 0){
+ numFailed++;
+ }
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::NextScanWhenNoMore,
+ true) != 0){
+ numFailed++;
+ }
+
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+}
+
+int runEqualAfterOpenScan(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+ ScanFunctions scanF(*pTab);
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::EqualAfterOpenScan,
+ false) == NDBT_OK){
+ numFailed++;
+ }
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::EqualAfterOpenScan,
+ true) == NDBT_OK){
+ numFailed++;
+ }
+
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+}
+
+int runOnlyOpenScanOnce(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+ ScanFunctions scanF(*pTab);
+ g_info << "OnlyOpenScanOnce openScanRead" << endl;
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::OnlyOpenScanOnce,
+ false) == 0){
+ numFailed++;
+ }
+ g_info << "OnlyOpenScanOnce openScanExclusive" << endl;
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::OnlyOpenScanOnce,
+ true) == 0){
+ numFailed++;
+ }
+
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+}
+
+int runOnlyOneOpInScanTrans(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+
+ ScanFunctions scanF(*pTab);
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::OnlyOneOpInScanTrans,
+ false) == 0){
+ numFailed++;
+ }
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::OnlyOneOpInScanTrans,
+ true) == 0){
+ numFailed++;
+ }
+
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+
+}
+
+int runExecuteScanWithoutOpenScan(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+ ScanFunctions scanF(*pTab);
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 1,
+ ScanFunctions::ExecuteScanWithOutOpenScan,
+ false) == 0){
+ numFailed++;
+ }
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+}
+
+
+
+int runOnlyOneOpBeforeOpenScan(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+
+ ScanFunctions scanF(*pTab);
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::OnlyOneOpBeforeOpenScan,
+ false) == 0){
+ numFailed++;
+ }
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::OnlyOneOpBeforeOpenScan,
+ true) == 0){
+ numFailed++;
+ }
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+
+}
+int runOnlyOneScanPerTrans(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+
+ ScanFunctions scanF(*pTab);
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::OnlyOneScanPerTrans,
+ false) == 0){
+ numFailed++;
+ }
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::OnlyOneScanPerTrans,
+ true) == 0){
+ numFailed++;
+ }
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+
+}
+
+int runNoCloseTransaction(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+
+ ScanFunctions scanF(*pTab);
+ int l = 0;
+ while(l < loops){
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::NoCloseTransaction,
+ false) != 0){
+ numFailed++;
+ }
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 6,
+ ScanFunctions::NoCloseTransaction,
+ true) != 0){
+ numFailed++;
+ }
+ l++;
+ }
+
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+
+}
+
+int runCheckInactivityTimeOut(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+
+ ScanFunctions scanF(*pTab);
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 1,
+ ScanFunctions::CheckInactivityTimeOut,
+ false) != NDBT_OK){
+ numFailed++;
+ }
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 240,
+ ScanFunctions::CheckInactivityTimeOut,
+ true) != NDBT_OK){
+ numFailed++;
+ }
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+
+}
+
+int runCheckInactivityBeforeClose(NDBT_Context* ctx, NDBT_Step* step){
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ int records = ctx->getNumRecords();
+ int numFailed = 0;
+
+ ScanFunctions scanF(*pTab);
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 16,
+ ScanFunctions::CheckInactivityBeforeClose,
+ false) != 0){
+ numFailed++;
+ }
+ if (scanF.scanReadFunctions(GETNDB(step),
+ records,
+ 240,
+ ScanFunctions::CheckInactivityBeforeClose,
+ true) != 0){
+ numFailed++;
+ }
+
+ if(numFailed > 0)
+ return NDBT_FAILED;
+ else
+ return NDBT_OK;
+
+}
+
+
+
+NDBT_TESTSUITE(testScan);
+TESTCASE("ScanRead",
+ "Verify scan requirement: It should be possible "\
+ "to read all records in a table without knowing their "\
+ "primary key."){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 1);
+ STEP(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanRead16",
+ "Verify scan requirement: It should be possible to scan read "\
+ "with parallelism, test with parallelism 16"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 16);
+ STEP(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanRead240",
+ "Verify scan requirement: It should be possible to scan read with "\
+ "parallelism, test with parallelism 240(240 would automatically be "\
+ "downgraded to the maximum parallelism value for the current config)"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 240);
+ STEP(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadCommitted240",
+ "Verify scan requirement: It should be possible to scan read committed with "\
+ "parallelism, test with parallelism 240(240 would automatically be "\
+ "downgraded to the maximum parallelism value for the current config)"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 240);
+ STEP(runScanReadCommitted);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanUpdate",
+ "Verify scan requirement: It should be possible "\
+ "to update all records in a table without knowing their"\
+ " primary key."){
+ INITIALIZER(runLoadTable);
+ STEP(runScanUpdate);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanUpdate2",
+ "Verify scan requirement: It should be possible "\
+ "to update all records in a table without knowing their"\
+ " primary key. Do this efficently by calling nextScanResult(false) "\
+ "in order to update the records already fetched to the api in one batch."){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 240);
+ STEP(runScanUpdate2);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanDelete",
+ "Verify scan requirement: It should be possible "\
+ "to delete all records in a table without knowing their"\
+ " primary key."){
+ INITIALIZER(runLoadTable);
+ STEP(runScanDelete);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanDelete2",
+ "Verify scan requirement: It should be possible "\
+ "to delete all records in a table without knowing their"\
+ " primary key. Do this efficently by calling nextScanResult(false) "\
+ "in order to delete the records already fetched to the api in one batch."){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 240);
+ STEP(runScanDelete2);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanUpdateAndScanRead",
+ "Verify scan requirement: It should be possible to run "\
+ "scan read and scan update at the same time"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 16);
+ STEP(runScanRead);
+ STEP(runScanUpdate);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadAndLocker",
+ "Verify scan requirement: The locks are not kept throughout "\
+ "the entire scan operation. This means that a scan does not "\
+ "lock the entire table, only the records it's currently "\
+ "operating on. This will test how scan performs when there are "\
+ " a number of 1 second locks in the table"){
+ INITIALIZER(runLoadTable);
+ STEP(runScanReadUntilStopped);
+ STEP(runLocker);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadAndPkRead",
+ "Verify scan requirement: The locks are not kept throughout "\
+ "the entire scan operation. This means that a scan does not "\
+ "lock the entire table, only the records it's currently "\
+ "operating on. This will test how scan performs when there are "\
+ " a pk reads "){
+ INITIALIZER(runLoadTable);
+ STEPS(runScanRead, 2);
+ STEPS(runPkRead, 2);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanRead488",
+ "Verify scan requirement: It's only possible to have 11 concurrent "\
+ "scans per fragment running in Ndb kernel at the same time. "\
+ "When this limit is exceeded the scan will be aborted with errorcode "\
+ "488."){
+ INITIALIZER(runLoadTable);
+ STEPS(runScanRead, 15);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanRead40",
+ "Verify scan requirement: Scan with 40 simultaneous threads"){
+ INITIALIZER(runLoadTable);
+ STEPS(runScanRead, 40);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanRead100",
+ "Verify scan requirement: Scan with 100 simultaneous threads"){
+ INITIALIZER(runLoadTable);
+ STEPS(runScanRead, 100);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanRead40RandomTable",
+ "Verify scan requirement: Scan with 40 simultaneous threads. "\
+ "Use random table for the scan"){
+ INITIALIZER(runCreateAllTables);
+ INITIALIZER(runLoadAllTables);
+ STEPS(runScanReadRandomTable, 40);
+ FINALIZER(runDropAllTablesExceptTestTable);
+}
+TESTCASE("ScanRead100RandomTable",
+ "Verify scan requirement: Scan with 100 simultaneous threads. "\
+ "Use random table for the scan"){
+ INITIALIZER(runCreateAllTables);
+ INITIALIZER(runLoadAllTables);
+ STEPS(runScanReadRandomTable, 100);
+ FINALIZER(runDropAllTablesExceptTestTable);
+}
+TESTCASE("ScanReadRandomPrepare",
+ "Create and load tables for ScanRead40RandomNoTableCreate."){
+ INITIALIZER(runCreateAllTables);
+ INITIALIZER(runLoadAllTables);
+}
+TESTCASE("ScanRead40RandomNoTableCreate",
+ "Verify scan requirement: Scan with 40 simultaneous threads. "\
+ "Use random table for the scan. Dont create or load the tables."){
+ STEPS(runScanReadRandomTable, 40);
+}
+TESTCASE("ScanRead100RandomNoTableCreate",
+ "Verify scan requirement: Scan with 100 simultaneous threads. "\
+ "Use random table for the scan. Dont create or load the tables."){
+ STEPS(runScanReadRandomTable, 100);
+}
+TESTCASE("ScanWithLocksAndInserts",
+ "TR457: This test is added to verify that an insert of a records "\
+ "that is already in the database does not delete the record"){
+ INITIALIZER(runLoadTable);
+ STEPS(runScanReadUntilStopped, 2);
+ STEP(runLocker);
+ STEP(runInsertUntilStopped);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadAbort",
+ "Scan requirement: A scan may be aborted by the application "\
+ "at any time. This can be performed even if there are more "\
+ "tuples to scan."){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("AbortProb", 90);
+ STEPS(runScanRead, 3);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadAbort15",
+ "Scan requirement: A scan may be aborted by the application "\
+ "at any time. This can be performed even if there are more "\
+ "tuples to scan. Use parallelism 15"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 15);
+ TC_PROPERTY("AbortProb", 90);
+ STEPS(runScanRead, 3);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadAbort240",
+ "Scan requirement: A scan may be aborted by the application "\
+ "at any time. This can be performed even if there are more "\
+ "tuples to scan. Use parallelism 240(it will be downgraded to max para for this config)"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 240);
+ TC_PROPERTY("AbortProb", 90);
+ STEPS(runScanRead, 3);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanUpdateAbort16",
+ "Scan requirement: A scan may be aborted by the application "\
+ "at any time. This can be performed even if there are more "\
+ "tuples to scan. Use parallelism 16"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 16);
+ TC_PROPERTY("AbortProb", 90);
+ STEPS(runScanUpdate, 3);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanUpdateAbort240",
+ "Scan requirement: A scan may be aborted by the application "\
+ "at any time. This can be performed even if there are more "\
+ "tuples to scan. Use parallelism 240(it will be downgraded to max para for this config)"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 240);
+ TC_PROPERTY("AbortProb", 90);
+ STEPS(runScanUpdate, 3);
+ FINALIZER(runClearTable);
+}
+TESTCASE("CheckGetValue",
+ "Check that we can call getValue to read attributes"\
+ "Especially interesting to see if we can read only the"\
+ " first, last or any two attributes from the table"){
+ INITIALIZER(runLoadTable);
+ STEP(runCheckGetValue);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("CloseWithoutStop",
+ "Check that we can close the scanning transaction without calling "\
+ "stopScan"){
+ INITIALIZER(runLoadTable);
+ STEP(runCloseWithoutStop);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NextScanWhenNoMore",
+ "Check that we can call nextScanResult when there are no more "\
+ "records, and that it returns a valid value"){
+ INITIALIZER(runLoadTable);
+ STEP(runNextScanWhenNoMore);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("EqualAfterOpenScan",
+ "Check that we can't call equal after openScan"){
+ STEP(runEqualAfterOpenScan);
+}
+TESTCASE("ExecuteScanWithoutOpenScan",
+ "Check that we can't call executeScan without defining a scan "\
+ "with openScan"){
+ INITIALIZER(runLoadTable);
+ STEP(runExecuteScanWithoutOpenScan);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("OnlyOpenScanOnce",
+ "Check that we may only call openScan once in the same trans"){
+ INITIALIZER(runLoadTable);
+ STEP(runOnlyOpenScanOnce);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("OnlyOneOpInScanTrans",
+ "Check that we can have only one operation in a scan trans"){
+ INITIALIZER(runLoadTable);
+ STEP(runOnlyOneOpInScanTrans);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("OnlyOneOpBeforeOpenScan",
+ "Check that we can have only one operation in a trans defined "\
+ "when calling openScan "){
+ INITIALIZER(runLoadTable);
+ STEP(runOnlyOneOpBeforeOpenScan);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("OnlyOneScanPerTrans",
+ "Check that we can have only one scan operation in a trans"){
+ INITIALIZER(runLoadTable);
+ STEP(runOnlyOneScanPerTrans);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("NoCloseTransaction",
+ "Check behaviour when close transaction is not called "){
+ INITIALIZER(runLoadTable);
+ STEP(runNoCloseTransaction);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("CheckInactivityTimeOut",
+ "Check behaviour when the api sleeps for a long time before continuing scan "){
+ INITIALIZER(runLoadTable);
+ STEP(runCheckInactivityTimeOut);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("CheckInactivityBeforeClose",
+ "Check behaviour when the api sleeps for a long time before calling close scan "){
+ INITIALIZER(runLoadTable);
+ STEP(runCheckInactivityBeforeClose);
+ VERIFIER(runScanRead);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadError5021",
+ "Scan and insert error 5021, one node is expected to crash"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("ErrorCode", 5021);
+ STEP(runScanReadErrorOneNode);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadError5022",
+ "Scan and insert error 5022, one node is expected to crash"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("ErrorCode", 5022);
+ TC_PROPERTY("NodeNumber", 2);
+ STEP(runScanReadErrorOneNode);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadError5023",
+ "Scan and insert error 5023"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("ErrorCode", 5023);
+ STEP(runScanReadError);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadError5024",
+ "Scan and insert error 5024"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("ErrorCode", 5024);
+ STEP(runScanReadError);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadError5025",
+ "Scan and insert error 5025"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("ErrorCode", 5025);
+ STEP(runScanReadError);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadError5030",
+ "Scan and insert error 5030."\
+ "Drop all SCAN_NEXTREQ signals in LQH until the node is "\
+ "shutdown with SYSTEM_ERROR because of scan fragment timeout"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("ErrorCode", 5030);
+ STEP(runScanReadErrorOneNode);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanReadRestart",
+ "Scan requirement:A scan should be able to start and "\
+ "complete during node recovery and when one or more nodes "\
+ "in the cluster is down.Use random parallelism "){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", RANDOM_PARALLELISM); // Random
+ STEP(runScanReadUntilStopped);
+ STEP(runRestarter);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanUpdateRestart",
+ "Scan requirement:A scan should be able to start and "\
+ "complete during node recovery and when one or more nodes "\
+ "in the cluster is down. Use random parallelism"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", RANDOM_PARALLELISM); // Random
+ STEP(runScanUpdateUntilStopped);
+ STEP(runRestarter);
+ FINALIZER(runClearTable);
+}
+#if 0
+TESTCASE("ScanReadRestart9999",
+ "Scan requirement:A scan should be able to start and "\
+ "complete during node recovery and when one or more nodes "\
+ "in the cluster is down. Use parallelism 240."\
+ "Restart using error insert 9999"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 240);
+ STEP(runScanReadUntilStopped);
+ STEP(runRestarter9999);
+ FINALIZER(runClearTable);
+}
+TESTCASE("ScanUpdateRestart9999",
+ "Scan requirement:A scan should be able to start and "\
+ "complete during node recovery and when one or more nodes "\
+ "in the cluster is down. Use parallelism 240."\
+ "Restart using error insert 9999"){
+ INITIALIZER(runLoadTable);
+ TC_PROPERTY("Parallelism", 240);
+ STEP(runScanReadUntilStopped);
+ STEP(runScanUpdateUntilStopped);
+ STEP(runRestarter9999);
+ FINALIZER(runClearTable);
+}
+#endif
+TESTCASE("InsertDelete",
+ "Load and delete all while scan updating and scan reading\n"\
+ "Alexander Lukas special"){
+ INITIALIZER(runClearTable);
+ STEP(runScanReadUntilStoppedNoCount);
+ STEP(runScanUpdateUntilStopped);
+ STEP(runInsertDelete);
+ FINALIZER(runClearTable);
+}
+TESTCASE("CheckAfterTerror",
+ "Check that we can still scan read after this terror of NdbApi"){
+ INITIALIZER(runLoadTable);
+ STEPS(runScanRead, 5);
+ FINALIZER(runClearTable);
+}
+NDBT_TESTSUITE_END(testScan);
+
+int main(int argc, const char** argv){
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ return testScan.execute(argc, argv);
+}
+
diff --git a/ndb/test/ndbapi/testScanInterpreter/Makefile b/ndb/test/ndbapi/testScanInterpreter/Makefile
new file mode 100644
index 00000000000..c7d96494148
--- /dev/null
+++ b/ndb/test/ndbapi/testScanInterpreter/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testScanInterpreter
+
+SOURCES = testScanInterpreter.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testScanInterpreter/ScanFilter.hpp b/ndb/test/ndbapi/testScanInterpreter/ScanFilter.hpp
new file mode 100644
index 00000000000..09786756798
--- /dev/null
+++ b/ndb/test/ndbapi/testScanInterpreter/ScanFilter.hpp
@@ -0,0 +1,131 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SCAN_FILTER_HPP
+#define SCAN_FILTER_HPP
+
+class ScanFilter {
+public:
+#if 0
+ /**
+ * Create a scan filter for table tab
+ * colNo - column to filter on
+ * val - val to use when selecting valu to filter on
+ *
+ */
+ ScanFilter(const NDBT_Table& tab,
+ int colNo,
+ int val);
+#endif
+ ScanFilter(int records = 1000){};
+ virtual int filterOp(NdbOperation*) = 0;
+ virtual int verifyRecord(NDBT_ResultRow&) = 0;
+private:
+
+ // const NDBT_Table& tab;
+};
+
+class LessThanFilter : public ScanFilter {
+public:
+ LessThanFilter(int records){ compare_value = records / 100; };
+private:
+ Uint32 compare_value;
+ int filterOp(NdbOperation* pOp);
+ int verifyRecord(NDBT_ResultRow&);
+};
+
+class EqualFilter : public ScanFilter {
+ static const Uint32 compare_value = 100;
+ int filterOp(NdbOperation* pOp);
+ int verifyRecord(NDBT_ResultRow&);
+};
+
+class NoFilter : public ScanFilter {
+ int filterOp(NdbOperation* pOp);
+ int verifyRecord(NDBT_ResultRow&);
+};
+
+
+int LessThanFilter::filterOp(NdbOperation* pOp){
+
+ if (pOp->load_const_u32(1, compare_value) != 0)
+ return NDBT_FAILED;
+
+ if (pOp->read_attr("KOL2", 2) != 0)
+ return NDBT_FAILED;
+
+ if (pOp->branch_lt(1, 2, 0) != 0)
+ return NDBT_FAILED;
+
+ if (pOp->interpret_exit_nok() != 0)
+ return NDBT_FAILED;
+
+ if (pOp->def_label(0) != 0)
+ return NDBT_FAILED;
+
+ if (pOp->interpret_exit_ok() != 0)
+ return NDBT_FAILED;
+
+ return NDBT_OK;
+}
+
+int LessThanFilter::verifyRecord(NDBT_ResultRow& row){
+ NdbRecAttr* rec = row.attributeStore(1);
+ if (rec->u_32_value() < compare_value)
+ return NDBT_OK;
+ return NDBT_FAILED;
+}
+
+int EqualFilter::filterOp(NdbOperation* pOp){
+
+ if (pOp->load_const_u32(1, compare_value) != 0)
+ return NDBT_FAILED;
+
+ if (pOp->read_attr("KOL2", 2) != 0)
+ return NDBT_FAILED;
+
+ if (pOp->branch_eq(1, 2, 0) != 0)
+ return NDBT_FAILED;
+
+ if (pOp->interpret_exit_nok() != 0)
+ return NDBT_FAILED;
+
+ if (pOp->def_label(0) != 0)
+ return NDBT_FAILED;
+
+ if (pOp->interpret_exit_ok() != 0)
+ return NDBT_FAILED;
+
+ return NDBT_OK;
+}
+
+int EqualFilter::verifyRecord(NDBT_ResultRow& row){
+ NdbRecAttr* rec = row.attributeStore(1);
+ if (rec->u_32_value() == compare_value)
+ return NDBT_OK;
+ return NDBT_FAILED;
+}
+
+int NoFilter::filterOp(NdbOperation* pOp){
+ return NDBT_OK;
+}
+
+int NoFilter::verifyRecord(NDBT_ResultRow& row){
+ // Check if this record should be in the result set or not
+ return NDBT_OK;
+}
+
+#endif
diff --git a/ndb/test/ndbapi/testScanInterpreter/ScanInterpretTest.hpp b/ndb/test/ndbapi/testScanInterpreter/ScanInterpretTest.hpp
new file mode 100644
index 00000000000..3862de34111
--- /dev/null
+++ b/ndb/test/ndbapi/testScanInterpreter/ScanInterpretTest.hpp
@@ -0,0 +1,528 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef SCAN_INTERPRET_TEST_HPP
+#define SCAN_INTERPRET_TEST_HPP
+
+#include "ScanFilter.hpp"
+
+class ScanInterpretTest {
+public:
+ ScanInterpretTest(const NdbDictionary::Table& _tab,
+ const NdbDictionary::Table& _restab) :
+ tab(_tab),
+ restab(_restab),
+ row(_tab){
+ }
+
+ int scanRead(Ndb*,
+ int records,
+ int parallelism,
+ ScanFilter& filter);
+ int scanReadVerify(Ndb*,
+ int records,
+ int parallelism,
+ ScanFilter& filter);
+
+ int addRowToInsert(Ndb* pNdb,
+ NdbConnection* pInsTrans);
+ int addRowToCheckTrans(Ndb* pNdb,
+ NdbConnection* pCheckTrans);
+private:
+ const NdbDictionary::Table& tab;
+ const NdbDictionary::Table& restab;
+ NDBT_ResultRow row;
+
+};
+
+
+inline
+int
+ScanInterpretTest::addRowToInsert(Ndb* pNdb,
+ NdbConnection* pInsTrans){
+
+ NdbOperation* pOp =
+ pInsTrans->getNdbOperation(restab.getName());
+ if (pOp == NULL) {
+ ERR(pInsTrans->getNdbError());
+ pNdb->closeTransaction(pInsTrans);
+ return NDBT_FAILED;
+ }
+
+ if( pOp->insertTuple() == -1 ) {
+ ERR(pInsTrans->getNdbError());
+ pNdb->closeTransaction(pInsTrans);
+ return NDBT_FAILED;
+ }
+
+ // Copy all attribute to the new operation
+ for (int a = 0; a<restab.getNoOfColumns(); a++){
+ const NdbDictionary::Column* attr = tab.getColumn(a);
+ NdbRecAttr* reca = row.attributeStore(a);
+ int check = -1;
+ switch (attr->getType()){
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Binary:
+ case NdbDictionary::Column::Varbinary:{
+ check = pOp->setValue( attr->getName(),
+ reca->aRef());
+ break;
+ }
+ case NdbDictionary::Column::Int:{
+ check = pOp->setValue( attr->getName(),
+ reca->int32_value());
+ }
+ break;
+ case NdbDictionary::Column::Bigint:{
+ check = pOp->setValue( attr->getName(),
+ reca->int64_value());
+ }
+ break;
+ case NdbDictionary::Column::Unsigned:{
+ check = pOp->setValue( attr->getName(),
+ reca->u_32_value());
+ }
+ break;
+ case NdbDictionary::Column::Bigunsigned:{
+ check = pOp->setValue( attr->getName(),
+ reca->u_64_value());
+ }
+ break;
+ case NdbDictionary::Column::Float:
+ check = pOp->setValue( attr->getName(),
+ reca->float_value());
+
+ break;
+ default:
+ check = -1;
+ break;
+ }
+ if(check != 0){
+ ERR(pInsTrans->getNdbError());
+ pNdb->closeTransaction(pInsTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ return NDBT_OK;
+}
+
+inline
+int
+ScanInterpretTest::addRowToCheckTrans(Ndb* pNdb,
+ NdbConnection* pCheckTrans){
+
+ NdbOperation* pOp =
+ pCheckTrans->getNdbOperation(restab.getName());
+ if (pOp == NULL) {
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ if(pOp->readTuple() != 0) {
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Copy pk attribute's to the new operation
+ for (int a = 0; a<restab.getNoOfColumns(); a++){
+ const NdbDictionary::Column* attr = restab.getColumn(a);
+ if (attr->getPrimaryKey() == true){
+ NdbRecAttr* reca = row.attributeStore(a);
+ int check = -1;
+ switch (attr->getType()){
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Binary:
+ case NdbDictionary::Column::Varbinary:{
+ check = pOp->equal( attr->getName(),
+ reca->aRef());
+ break;
+ }
+ case NdbDictionary::Column::Int:{
+ check = pOp->equal( attr->getName(),
+ reca->int32_value());
+ }
+ break;
+ case NdbDictionary::Column::Bigint:{
+ check = pOp->equal( attr->getName(),
+ reca->int64_value());
+ }
+ break;
+ case NdbDictionary::Column::Unsigned:{
+ check = pOp->equal( attr->getName(),
+ reca->u_32_value());
+ }
+ break;
+ case NdbDictionary::Column::Bigunsigned:{
+ check = pOp->equal( attr->getName(),
+ reca->u_64_value());
+ }
+ break;
+ default:
+ check = -1;
+ break;
+ }
+ if(check != 0){
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ return NDBT_OK;
+}
+
+inline
+int
+ScanInterpretTest::scanRead(Ndb* pNdb,
+ int records,
+ int parallelism,
+ ScanFilter& filter){
+ int retryAttempt = 0;
+ int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ ndbout << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanRead(parallelism);
+ //check = pOp->openScanExclusive(parallelism);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if (filter.filterOp(pOp) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Read all attributes
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((row.attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ NdbConnection* pInsTrans;
+
+ while((eof = pTrans->nextScanResult(true)) == 0){
+ pInsTrans = pNdb->startTransaction();
+ if (pInsTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+ ERR(err);
+ return NDBT_FAILED;
+ }
+ do {
+ rows++;
+ if (addRowToInsert(pNdb, pInsTrans) != 0){
+ pNdb->closeTransaction(pTrans);
+ pNdb->closeTransaction(pInsTrans);
+ return NDBT_FAILED;
+ }
+ } while((eof = pTrans->nextScanResult(false)) == 0);
+
+ check = pInsTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pInsTrans->getNdbError();
+ ERR(err);
+ pNdb->closeTransaction(pInsTrans);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ pNdb->closeTransaction(pInsTrans);
+
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ g_info << rows << " rows have been scanned" << endl;
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+inline
+int
+ScanInterpretTest::scanReadVerify(Ndb* pNdb,
+ int records,
+ int parallelism,
+ ScanFilter& filter){
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ ndbout << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) { if (pOp->getValue("KOL2") == 0){
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+
+
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanRead(parallelism);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if (check == -1) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+
+ // Read all attributes
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((row.attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ int rowsNoExist = 0;
+ int rowsExist = 0;
+ int existingRecordsNotFound = 0;
+ int nonExistingRecordsFound = 0;
+
+
+ NdbConnection* pExistTrans;
+ NdbConnection* pNoExistTrans;
+
+ while((eof = pTrans->nextScanResult(true)) == 0){
+ pExistTrans = pNdb->startTransaction();
+ if (pExistTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+ ERR(err);
+ return NDBT_FAILED;
+ }
+ pNoExistTrans = pNdb->startTransaction();
+ if (pNoExistTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+ ERR(err);
+ return NDBT_FAILED;
+ }
+ do {
+ rows++;
+ if (filter.verifyRecord(row) == NDBT_OK){
+ rowsExist++;
+ if (addRowToCheckTrans(pNdb, pExistTrans) != 0){
+ pNdb->closeTransaction(pTrans);
+ pNdb->closeTransaction(pExistTrans);
+ pNdb->closeTransaction(pNoExistTrans);
+ return NDBT_FAILED;
+ }
+ }else{
+ rowsNoExist++;
+ if (addRowToCheckTrans(pNdb, pNoExistTrans) != 0){
+ pNdb->closeTransaction(pTrans);
+ pNdb->closeTransaction(pExistTrans);
+ pNdb->closeTransaction(pNoExistTrans);
+ return NDBT_FAILED;
+ }
+ }
+ } while((eof = pTrans->nextScanResult(false)) == 0);
+
+
+ // Execute the transaction containing reads of
+ // all the records that should be in the result table
+ check = pExistTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pExistTrans->getNdbError();
+ ERR(err);
+ if (err.code != 626){
+ pNdb->closeTransaction(pExistTrans);
+ pNdb->closeTransaction(pNoExistTrans);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }else{
+ // Some of the records expected to be found wasn't
+ // there
+ existingRecordsNotFound = 1;
+ }
+ }
+ pNdb->closeTransaction(pExistTrans);
+
+ // Execute the transaction containing reads of
+ // all the records that should NOT be in the result table
+ check = pNoExistTrans->execute(Commit, CommitAsMuchAsPossible);
+ if( check == -1 ) {
+ const NdbError err = pNoExistTrans->getNdbError();
+ // The transactions error code should be zero
+ if (err.code != 626){
+ ERR(err);
+ pNdb->closeTransaction(pNoExistTrans);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ // Loop through the no existing transaction and check that no
+ // operations where successful
+ const NdbOperation* pOp2 = NULL;
+ while ((pOp2 = pNoExistTrans->getNextCompletedOperation(pOp2)) != NULL){
+ const NdbError err = pOp2->getNdbError();
+ if (err.code != 626){
+ ndbout << "err.code = " << err.code<< endl;
+ nonExistingRecordsFound = 1;
+ }
+ }
+ }
+
+ pNdb->closeTransaction(pNoExistTrans);
+
+
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int testResult = NDBT_OK;
+ int rowsResult = 0;
+ UtilTransactions utilTrans(restab);
+ if (utilTrans.selectCount(pNdb,
+ 240,
+ &rowsResult) != 0){
+ return NDBT_FAILED;
+ }
+ if (existingRecordsNotFound == 1){
+ ndbout << "!!! Expected records not found" << endl;
+ testResult = NDBT_FAILED;
+ }
+ if (nonExistingRecordsFound == 1){
+ ndbout << "!!! Unxpected records found" << endl;
+ testResult = NDBT_FAILED;
+ }
+ ndbout << rows << " rows scanned("
+ << rowsExist << " found, " << rowsResult<<" expected)" << endl;
+ if (rowsResult != rowsExist){
+ ndbout << "!!! Number of rows in result table different from expected" << endl;
+ testResult = NDBT_FAILED;
+ }
+
+ return testResult;
+ }
+ return NDBT_FAILED;
+}
+
+#endif
diff --git a/ndb/test/ndbapi/testScanInterpreter/testScanInterpreter.cpp b/ndb/test/ndbapi/testScanInterpreter/testScanInterpreter.cpp
new file mode 100644
index 00000000000..18fd98bdbb3
--- /dev/null
+++ b/ndb/test/ndbapi/testScanInterpreter/testScanInterpreter.cpp
@@ -0,0 +1,280 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NDBT_Test.hpp"
+#include "NDBT_ReturnCodes.h"
+#include "HugoTransactions.hpp"
+#include "UtilTransactions.hpp"
+#include "NdbRestarter.hpp"
+#include <Vector.hpp>
+#include "ScanFilter.hpp"
+#include "ScanInterpretTest.hpp"
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runClearResTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ const NdbDictionary::Table* pResTab =
+ GETNDB(step)->getDictionary()->getTable(ctx->getProperty("ResultTabName", "NULL"));
+
+ UtilTransactions utilTrans(*pResTab);
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runScanRead(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int parallelism = ctx->getProperty("Parallelism", 1);
+
+ int i = 0;
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while (i<loops) {
+ g_info << i << ": ";
+ if (hugoTrans.scanReadRecords(GETNDB(step), records, 0, parallelism) != 0){
+ return NDBT_FAILED;
+ }
+ i++;
+ }
+ return NDBT_OK;
+}
+
+int runScanReadResTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ int parallelism = ctx->getProperty("Parallelism", 1);
+ const NdbDictionary::Table* pResTab =
+ NDBT_Table::discoverTableFromDb(GETNDB(step),
+ ctx->getProperty("ResultTabName", "NULL"));
+
+ HugoTransactions hugoTrans(*pResTab);
+ if (hugoTrans.scanReadRecords(GETNDB(step), records, 0, parallelism) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runCreateResultTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ char newTabName[256];
+ snprintf(newTabName, 256, "%s_RES", pTab->getName());
+ ctx->setProperty("ResultTabName", newTabName);
+
+ NdbDictionary::Table resTab(* pTab);
+ resTab.setName(newTabName);
+
+ if (GETNDB(step)->getDictionary()->createTable(resTab) != 0){
+ g_err << newTabName << " creation failed!"<< endl;
+ return NDBT_FAILED;
+ }else{
+ g_info << newTabName << " created!"<< endl;
+ return NDBT_OK;
+ }
+}
+
+int scanWithFilter(NDBT_Context* ctx, NDBT_Step* step, ScanFilter& filt){
+ int records = ctx->getNumRecords();
+ const char* resTabName = ctx->getProperty("ResultTabName", "NULL");
+ if (strcmp(resTabName, "NULL") == 0)
+ return NDBT_FAILED;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ const NdbDictionary::Table* pResTab = NDBT_Table::discoverTableFromDb(GETNDB(step), resTabName);
+ if (pResTab == NULL)
+ return NDBT_FAILED;
+
+ ScanInterpretTest interpretTest(*pTab, *pResTab);
+ if (interpretTest.scanRead(GETNDB(step),
+ records,
+ 16,
+ filt) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+int runScanLessThan(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ LessThanFilter filt(records);
+ return scanWithFilter(ctx, step, filt);
+}
+int runScanEqual(NDBT_Context* ctx, NDBT_Step* step){
+ EqualFilter filt;
+ return scanWithFilter(ctx, step, filt);
+}
+
+int scanVerifyWithFilter(NDBT_Context* ctx, NDBT_Step* step, ScanFilter& filt){
+ int records = ctx->getNumRecords();
+ const char* resTabName = ctx->getProperty("ResultTabName", "NULL");
+ if (strcmp(resTabName, "NULL") == 0)
+ return NDBT_FAILED;
+ const NdbDictionary::Table* pTab = ctx->getTab();
+ const NdbDictionary::Table* pResTab = NDBT_Table::discoverTableFromDb(GETNDB(step), resTabName);
+ if (pResTab == NULL)
+ return NDBT_FAILED;
+
+ ScanInterpretTest interpretTest(*pTab, *pResTab);
+ if (interpretTest.scanReadVerify(GETNDB(step),
+ records,
+ 16,
+ filt) != NDBT_OK){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+int runScanLessThanVerify(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+ LessThanFilter filt(records);
+ return scanVerifyWithFilter(ctx, step, filt);
+}
+int runScanEqualVerify(NDBT_Context* ctx, NDBT_Step* step){
+ EqualFilter filt;
+ return scanVerifyWithFilter(ctx, step, filt);
+}
+
+int runScanEqualLoop(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int l = 0;
+ EqualFilter filt;
+ while(l < loops){
+ if (scanWithFilter(ctx, step, filt) != NDBT_OK)
+ return NDBT_FAILED;
+ if (runClearResTable(ctx, step) != NDBT_OK)
+ return NDBT_FAILED;
+ l++;
+ }
+ return NDBT_OK;
+}
+
+
+int runScanEqualVerifyLoop(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int l = 0;
+ EqualFilter filt;
+ while(l < loops){
+ if (scanWithFilter(ctx, step, filt) != NDBT_OK)
+ return NDBT_FAILED;
+ if (scanVerifyWithFilter(ctx, step, filt) != NDBT_OK)
+ return NDBT_FAILED;
+ if (runClearResTable(ctx, step) != NDBT_OK)
+ return NDBT_FAILED;
+ l++;
+ }
+ return NDBT_OK;
+}
+
+int runScanLessThanLoop(NDBT_Context* ctx, NDBT_Step* step){
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int l = 0;
+ LessThanFilter filt(records);
+ while(l < loops){
+ if (scanWithFilter(ctx, step, filt) != NDBT_OK)
+ return NDBT_FAILED;
+ if (runClearResTable(ctx, step) != NDBT_OK)
+ return NDBT_FAILED;
+ l++;
+ }
+ return NDBT_OK;
+}
+
+NDBT_TESTSUITE(testScanInterpreter);
+TESTCASE("ScanLessThan",
+ "Read all records in table TX with attrX less "\
+ "than a value and store the resultset in TX_RES."\
+ "Then compare records in TX_RES with records in TX."){
+ TABLE("T1");
+ TABLE("T2");
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCreateResultTable);
+ STEP(runScanLessThan);
+ VERIFIER(runScanLessThanVerify);
+ FINALIZER(runClearTable);
+ FINALIZER(runClearResTable);
+}
+TESTCASE("ScanEqual",
+ "Read all records in table TX with attrX equal "\
+ "to a value and store the resultset in TX_RES."\
+ "Then compare records in TX_RES with records in TX."){
+ TABLE("T1");
+ TABLE("T2");
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCreateResultTable);
+ STEP(runScanEqual);
+ VERIFIER(runScanEqualVerify);
+ FINALIZER(runClearTable);
+ FINALIZER(runClearResTable);
+}
+TESTCASE("ScanEqualLoop",
+ "Scan all records in TX equal to a value."\
+ "Do this loop number of times"){
+ TABLE("T1");
+ TABLE("T2");
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCreateResultTable);
+ STEP(runScanEqualLoop);
+ FINALIZER(runClearTable);
+ FINALIZER(runClearResTable);
+}
+TESTCASE("ScanEqualVerifyLoop",
+ "Scan all records in TX equal to a value."\
+ "Verify record in TX_RES table"\
+ "Do this loop number of times"){
+ TABLE("T1");
+ TABLE("T2");
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCreateResultTable);
+ STEP(runScanEqualVerifyLoop);
+ FINALIZER(runClearTable);
+ FINALIZER(runClearResTable);
+}
+TESTCASE("ScanLessThanLoop",
+ "Scan all records in TX less than a value."\
+ "Do this loop number of times"){
+ TABLE("T1");
+ TABLE("T2");
+ INITIALIZER(runLoadTable);
+ INITIALIZER(runCreateResultTable);
+ STEP(runScanLessThanLoop);
+ FINALIZER(runClearTable);
+ FINALIZER(runClearResTable);
+}
+NDBT_TESTSUITE_END(testScanInterpreter);
+
+int main(int argc, const char** argv){
+ return testScanInterpreter.execute(argc, argv);
+}
+
+
+
diff --git a/ndb/test/ndbapi/testSystemRestart/Makefile b/ndb/test/ndbapi/testSystemRestart/Makefile
new file mode 100644
index 00000000000..7a306eb313d
--- /dev/null
+++ b/ndb/test/ndbapi/testSystemRestart/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testSystemRestart
+
+SOURCES = testSystemRestart.cpp
+
+CFLAGS_testSystemRestart.cpp := -I$(call fixpath,$(NDB_TOP)/include/kernel)
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testSystemRestart/testSystemRestart.cpp b/ndb/test/ndbapi/testSystemRestart/testSystemRestart.cpp
new file mode 100644
index 00000000000..1b8a35487cb
--- /dev/null
+++ b/ndb/test/ndbapi/testSystemRestart/testSystemRestart.cpp
@@ -0,0 +1,942 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <Vector.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+#define CHECK(b) if (!(b)) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ continue; }
+
+int runSystemRestart1(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int timeout = 300;
+ Uint32 loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int count;
+ NdbRestarter restarter;
+ Uint32 i = 1;
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while(i<=loops && result != NDBT_FAILED){
+
+ ndbout << "Loop " << i << "/"<< loops <<" started" << endl;
+ /*
+ 1. Load data
+ 2. Restart cluster and verify records
+ 3. Update records
+ 4. Restart cluster and verify records
+ 5. Delete half of the records
+ 6. Restart cluster and verify records
+ 7. Delete all records
+ 8. Restart cluster and verify records
+ 9. Insert, update, delete records
+ 10. Restart cluster and verify records
+ 11. Insert, update, delete records
+ 12. Restart cluster with error insert 5020 and verify records
+ */
+ ndbout << "Loading records..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+
+ ndbout << "Restarting cluster" << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+
+ ndbout << "Updating records..." << endl;
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+
+ ndbout << "Deleting 50% of records..." << endl;
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.scanReadRecords(pNdb, records/2, 0, 64) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+
+ ndbout << "Deleting all records..." << endl;
+ CHECK(utilTrans.clearTable(pNdb, records/2) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+
+ ndbout << "Doing it all..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(hugoTrans.scanUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+
+ ndbout << "Doing it all..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(hugoTrans.scanUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+
+ ndbout << "Restarting cluster with error insert 5020..." << endl;
+ CHECK(restarter.restartAll(false, true) == 0);
+ CHECK(restarter.waitClusterNoStart(timeout) == 0);
+ CHECK(restarter.insertErrorInAllNodes(5020) == 0);
+ CHECK(restarter.startAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ i++;
+ }
+
+ ndbout << "runSystemRestart1 finished" << endl;
+
+ return result;
+}
+
+int runSystemRestart2(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+/// int timeout = 300;
+ int timeout = 120;
+ Uint32 loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int count;
+ NdbRestarter restarter;
+ Uint32 i = 1;
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while(i<=loops && result != NDBT_FAILED && !ctx->isTestStopped()){
+
+ ndbout << "Loop " << i << "/"<< loops <<" started" << endl;
+ /* Use error 7070 to set time between LCP to it's min value
+ 1. Load data
+ 2. Restart cluster and verify records
+ 3. Update records
+ 4. Restart cluster and verify records
+ 5. Delete half of the records
+ 6. Restart cluster and verify records
+ 7. Delete all records
+ 8. Restart cluster and verify records
+ 9. Insert, update, delete records
+ 10. Restart cluster and verify records
+ */
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+
+ ndbout << "Loading records..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+
+ ndbout << "Restarting cluster" << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+
+ ndbout << "Updating records..." << endl;
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+
+ ndbout << "Deleting 50% of records..." << endl;
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.scanReadRecords(pNdb, records/2, 0, 64) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+
+ ndbout << "Deleting all records..." << endl;
+ CHECK(utilTrans.clearTable(pNdb, records/2) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+
+ ndbout << "Doing it all..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(hugoTrans.scanUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+
+ i++;
+ }
+
+ ndbout << "runSystemRestart2 finished" << endl;
+
+ return result;
+}
+
+int runSystemRestartTestUndoLog(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int timeout = 300;
+ Uint32 loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int count;
+ NdbRestarter restarter;
+ Uint32 i = 1;
+
+ int dump7080[2];
+ dump7080[0] = 7080;
+ dump7080[1] = ctx->getTab()->getTableId();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while(i<=loops && result != NDBT_FAILED){
+
+ ndbout << "Loop " << i << "/"<< loops <<" started" << endl;
+ /*
+ 1. Start LCP, turn on undologging but delay write of datapages.
+ 2. Insert, update, delete records
+ 3. Complete writing of data pages and finish LCP.
+ 4. Restart cluster and verify records
+ */
+ // Use dump state 7080 to delay writing of datapages
+ // for the current table
+ ndbout << "Dump state: "<<dump7080[0]<<", "<<dump7080[1]<<endl;
+ CHECK(restarter.dumpStateAllNodes(dump7080, 2) == 0);
+ NdbSleep_SecSleep(10);
+
+ ndbout << "Doing it all..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+ CHECK(hugoTrans.scanUpdateRecords(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+
+ // Reset error and let LCP continue
+ CHECK(restarter.insertErrorInAllNodes(0) == 0);
+ NdbSleep_SecSleep(60);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+
+ // Use dump state 7080 to delay writing of datapages
+ // for the current table
+ ndbout << "Dump state: "<<dump7080[0]<<", "<<dump7080[1]<<endl;
+ CHECK(restarter.dumpStateAllNodes(dump7080, 2) == 0);
+ NdbSleep_SecSleep(10);
+
+ ndbout << "Doing it all, delete 50%..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+
+ // Reset error and let LCP continue
+ CHECK(restarter.insertErrorInAllNodes(0) == 0);
+ NdbSleep_SecSleep(20);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.scanReadRecords(pNdb, records/2, 0, 64) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+ CHECK(utilTrans.clearTable(pNdb, records) == 0);
+
+ i++;
+ }
+
+ ndbout << "runSystemRestartTestUndoLog finished" << endl;
+
+ return result;
+}
+
+int runSystemRestartTestFullDb(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int timeout = 300;
+ Uint32 loops = ctx->getNumLoops();
+ int count1, count2;
+ NdbRestarter restarter;
+ Uint32 i = 1;
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+ while(i<=loops && result != NDBT_FAILED){
+
+ ndbout << "Loop " << i << "/"<< loops <<" started" << endl;
+ /*
+ 1. Load data until db reports it's full
+ 2. Restart cluster and verify records
+ */
+ ndbout << "Filling up table..." << endl;
+ CHECK(hugoTrans.fillTable(pNdb) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count1) == 0);
+ ndbout << "Db is full. Table has "<<count1 <<" records."<< endl;
+
+ ndbout << "Restarting cluster" << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(hugoTrans.scanReadRecords(pNdb, count1) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count2) == 0);
+ CHECK(count1 == count2);
+
+ ndbout << "Deleting all records..." << endl;
+ CHECK(utilTrans.clearTable2(pNdb, count1) == 0);
+
+ ndbout << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count1) == 0);
+ CHECK(count1 == 0);
+
+ i++;
+ }
+
+ ndbout << "runSystemRestartTestFullDb finished" << endl;
+
+ return result;
+}
+
+int runSystemRestart3(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int timeout = 300;
+ Uint32 loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int count;
+ NdbRestarter restarter;
+ Uint32 i = 1;
+
+ const Uint32 nodeCount = restarter.getNumDbNodes();
+ if(nodeCount < 2){
+ g_info << "SR3 - Needs atleast 2 nodes to test" << endl;
+ return NDBT_OK;
+ }
+
+ Vector<int> nodeIds;
+ for(Uint32 i = 0; i<nodeCount; i++)
+ nodeIds.push_back(restarter.getDbNodeId(i));
+
+ Uint32 currentRestartNodeIndex = 0;
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ while(i<=loops && result != NDBT_FAILED){
+
+ g_info << "Loop " << i << "/"<< loops <<" started" << endl;
+ /**
+ * 1. Load data
+ * 2. Restart 1 node -nostart
+ * 3. Update records
+ * 4. Restart cluster and verify records
+ * 5. Restart 1 node -nostart
+ * 6. Delete half of the records
+ * 7. Restart cluster and verify records
+ * 8. Restart 1 node -nostart
+ * 9. Delete all records
+ * 10. Restart cluster and verify records
+ */
+ g_info << "Loading records..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+
+ /*** 1 ***/
+ g_info << "1 - Stopping one node" << endl;
+ CHECK(restarter.restartOneDbNode(nodeIds[currentRestartNodeIndex],
+ false,
+ true,
+ false) == 0);
+ currentRestartNodeIndex = (currentRestartNodeIndex + 1 ) % nodeCount;
+
+ g_info << "Updating records..." << endl;
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+
+ g_info << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ g_info << "Verifying records..." << endl;
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+
+ g_info << "2 - Stopping one node" << endl;
+ CHECK(restarter.restartOneDbNode(nodeIds[currentRestartNodeIndex],
+ false,
+ true,
+ false) == 0);
+ currentRestartNodeIndex = (currentRestartNodeIndex + 1 ) % nodeCount;
+
+ g_info << "Deleting 50% of records..." << endl;
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+
+ g_info << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ g_info << "Verifying records..." << endl;
+ CHECK(hugoTrans.scanReadRecords(pNdb, records/2, 0, 64) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+
+ g_info << "3 - Stopping one node" << endl;
+ CHECK(restarter.restartOneDbNode(nodeIds[currentRestartNodeIndex],
+ false,
+ true,
+ false) == 0);
+ currentRestartNodeIndex = (currentRestartNodeIndex + 1 ) % nodeCount;
+ g_info << "Deleting all records..." << endl;
+ CHECK(utilTrans.clearTable(pNdb, records/2) == 0);
+
+ g_info << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+
+ i++;
+ }
+
+ g_info << "runSystemRestart3 finished" << endl;
+
+ return result;
+}
+
+int runSystemRestart4(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int timeout = 300;
+ Uint32 loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int count;
+ NdbRestarter restarter;
+ Uint32 i = 1;
+
+ const Uint32 nodeCount = restarter.getNumDbNodes();
+ if(nodeCount < 2){
+ g_info << "SR4 - Needs atleast 2 nodes to test" << endl;
+ return NDBT_OK;
+ }
+
+ Vector<int> nodeIds;
+ for(Uint32 i = 0; i<nodeCount; i++)
+ nodeIds.push_back(restarter.getDbNodeId(i));
+
+ Uint32 currentRestartNodeIndex = 0;
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ if(restarter.dumpStateAllNodes(&val, 1) != 0){
+ g_err << "ERR: "<< step->getName()
+ << " failed on line " << __LINE__ << endl;
+ return NDBT_FAILED;
+ }
+ }
+
+ while(i<=loops && result != NDBT_FAILED){
+
+ g_info << "Loop " << i << "/"<< loops <<" started" << endl;
+ /**
+ * 1. Load data
+ * 2. Restart 1 node -nostart
+ * 3. Update records
+ * 4. Restart cluster and verify records
+ * 5. Restart 1 node -nostart
+ * 6. Delete half of the records
+ * 7. Restart cluster and verify records
+ * 8. Restart 1 node -nostart
+ * 9. Delete all records
+ * 10. Restart cluster and verify records
+ */
+ g_info << "Loading records..." << endl;
+ CHECK(hugoTrans.loadTable(pNdb, records) == 0);
+
+ /*** 1 ***/
+ g_info << "1 - Stopping one node" << endl;
+ CHECK(restarter.restartOneDbNode(nodeIds[currentRestartNodeIndex],
+ false,
+ true,
+ false) == 0);
+ currentRestartNodeIndex = (currentRestartNodeIndex + 1 ) % nodeCount;
+
+ g_info << "Updating records..." << endl;
+ CHECK(hugoTrans.pkUpdateRecords(pNdb, records) == 0);
+
+ g_info << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ g_info << "Verifying records..." << endl;
+ CHECK(hugoTrans.pkReadRecords(pNdb, records) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == records);
+
+ g_info << "2 - Stopping one node" << endl;
+ CHECK(restarter.restartOneDbNode(nodeIds[currentRestartNodeIndex],
+ false,
+ true,
+ false) == 0);
+ currentRestartNodeIndex = (currentRestartNodeIndex + 1 ) % nodeCount;
+
+ g_info << "Deleting 50% of records..." << endl;
+ CHECK(hugoTrans.pkDelRecords(pNdb, records/2) == 0);
+
+ g_info << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ g_info << "Verifying records..." << endl;
+ CHECK(hugoTrans.scanReadRecords(pNdb, records/2, 0, 64) == 0);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == (records/2));
+
+ g_info << "3 - Stopping one node" << endl;
+ CHECK(restarter.restartOneDbNode(nodeIds[currentRestartNodeIndex],
+ false,
+ true,
+ false) == 0);
+ currentRestartNodeIndex = (currentRestartNodeIndex + 1 ) % nodeCount;
+ g_info << "Deleting all records..." << endl;
+ CHECK(utilTrans.clearTable(pNdb, records/2) == 0);
+
+ g_info << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll() == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ CHECK(count == 0);
+
+ i++;
+ }
+
+ g_info << "runSystemRestart4 finished" << endl;
+
+ return result;
+}
+
+int runSystemRestart5(NDBT_Context* ctx, NDBT_Step* step){
+ Ndb* pNdb = GETNDB(step);
+ int result = NDBT_OK;
+ int timeout = 300;
+ Uint32 loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ int count;
+ NdbRestarter restarter;
+ Uint32 i = 1;
+
+ const Uint32 nodeCount = restarter.getNumDbNodes();
+ if(nodeCount < 2){
+ g_info << "SR5 - Needs atleast 2 nodes to test" << endl;
+ return NDBT_OK;
+ }
+
+ Vector<int> nodeIds;
+ for(Uint32 i = 0; i<nodeCount; i++)
+ nodeIds.push_back(restarter.getDbNodeId(i));
+
+ Uint32 currentRestartNodeIndex = 0;
+ UtilTransactions utilTrans(*ctx->getTab());
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ if(restarter.dumpStateAllNodes(&val, 1) != 0){
+ g_err << "ERR: "<< step->getName()
+ << " failed on line " << __LINE__ << endl;
+ return NDBT_FAILED;
+ }
+ }
+
+ while(i<=loops && result != NDBT_FAILED){
+
+ g_info << "Loop " << i << "/"<< loops <<" started" << endl;
+ /**
+ * 1. Load data
+ * 2. Restart 1 node -nostart
+ * 3. Update records
+ * 4. Restart cluster and verify records
+ * 5. Restart 1 node -nostart
+ * 6. Delete half of the records
+ * 7. Restart cluster and verify records
+ * 8. Restart 1 node -nostart
+ * 9. Delete all records
+ * 10. Restart cluster and verify records
+ */
+ g_info << "Loading records..." << endl;
+ hugoTrans.loadTable(pNdb, records);
+
+ /*** 1 ***/
+ g_info << "1 - Stopping one node" << endl;
+ CHECK(restarter.restartOneDbNode(nodeIds[currentRestartNodeIndex],
+ false,
+ true,
+ false) == 0);
+ currentRestartNodeIndex = (currentRestartNodeIndex + 1 ) % nodeCount;
+
+ g_info << "Updating records..." << endl;
+ hugoTrans.pkUpdateRecords(pNdb, records);
+
+ g_info << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll(false, false, true) == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ g_info << "Verifying records..." << endl;
+ hugoTrans.pkReadRecords(pNdb, records);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ //CHECK(count == records);
+
+ g_info << "2 - Stopping one node" << endl;
+ CHECK(restarter.restartOneDbNode(nodeIds[currentRestartNodeIndex],
+ false,
+ true,
+ false) == 0);
+ currentRestartNodeIndex = (currentRestartNodeIndex + 1 ) % nodeCount;
+
+ g_info << "Deleting 50% of records..." << endl;
+ hugoTrans.pkDelRecords(pNdb, records/2);
+
+ g_info << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll(false, false, true) == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ g_info << "Verifying records..." << endl;
+ hugoTrans.scanReadRecords(pNdb, records/2, 0, 64);
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ //CHECK(count == (records/2));
+
+ g_info << "3 - Stopping one node" << endl;
+ CHECK(restarter.restartOneDbNode(nodeIds[currentRestartNodeIndex],
+ false,
+ true,
+ false) == 0);
+ currentRestartNodeIndex = (currentRestartNodeIndex + 1 ) % nodeCount;
+ g_info << "Deleting all records..." << endl;
+ utilTrans.clearTable(pNdb, records/2);
+
+ g_info << "Restarting cluster..." << endl;
+ CHECK(restarter.restartAll(false, false, true) == 0);
+ CHECK(restarter.waitClusterStarted(timeout) == 0);
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(restarter.dumpStateAllNodes(&val, 1) == 0);
+ }
+ CHECK(pNdb->waitUntilReady(timeout) == 0);
+
+ ndbout << "Verifying records..." << endl;
+ CHECK(utilTrans.selectCount(pNdb, 64, &count) == 0);
+ //CHECK(count == 0);
+
+ CHECK(utilTrans.clearTable(pNdb) == 0);
+ i++;
+ }
+
+ g_info << "runSystemRestart5 finished" << endl;
+
+ return result;
+}
+
+int runWaitStarted(NDBT_Context* ctx, NDBT_Step* step){
+
+ NdbRestarter restarter;
+ restarter.waitClusterStarted(300);
+
+ NdbSleep_SecSleep(3);
+ return NDBT_OK;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+NDBT_TESTSUITE(testSystemRestart);
+TESTCASE("SR1",
+ "Basic system restart test. Focus on testing restart from REDO log.\n"
+ "NOTE! Time between lcp's and gcp's should be left at default, \n"
+ "so that Ndb uses the Redo log when restarting\n"
+ "1. Load records\n"
+ "2. Restart cluster and verify records \n"
+ "3. Update records\n"
+ "4. Restart cluster and verify records \n"
+ "5. Delete half of the records \n"
+ "6. Restart cluster and verify records \n"
+ "7. Delete all records \n"
+ "8. Restart cluster and verify records \n"
+ "9. Insert, update, delete records \n"
+ "10. Restart cluster and verify records\n"
+ "11. Insert, update, delete records \n"
+ "12. Restart cluster with error insert 5020 and verify records\n"){
+ INITIALIZER(runWaitStarted);
+ STEP(runSystemRestart1);
+ FINALIZER(runClearTable);
+}
+TESTCASE("SR2",
+ "Basic system restart test. Focus on testing restart from LCP\n"
+ "NOTE! Time between lcp's is automatically set to it's min value\n"
+ "so that Ndb uses LCP's when restarting.\n"
+ "1. Load records\n"
+ "2. Restart cluster and verify records \n"
+ "3. Update records\n"
+ "4. Restart cluster and verify records \n"
+ "5. Delete half of the records \n"
+ "6. Restart cluster and verify records \n"
+ "7. Delete all records \n"
+ "8. Restart cluster and verify records \n"
+ "9. Insert, update, delete records \n"
+ "10. Restart cluster and verify records\n"){
+ INITIALIZER(runWaitStarted);
+ STEP(runSystemRestart2);
+ FINALIZER(runClearTable);
+}
+TESTCASE("SR_UNDO",
+ "System restart test. Focus on testing of undologging\n"
+ "in DBACC and DBTUP.\n"
+ "This is done by starting a LCP, turn on undologging \n"
+ "but don't start writing the datapages. This will force all\n"
+ "operations to be written into the undolog.\n"
+ "Then write datapages and complete LCP.\n"
+ "Restart the system\n"){
+ INITIALIZER(runWaitStarted);
+ STEP(runSystemRestartTestUndoLog);
+ FINALIZER(runClearTable);
+}
+TESTCASE("SR_FULLDB",
+ "System restart test. Test to restart when DB is full.\n"){
+ INITIALIZER(runWaitStarted);
+ STEP(runSystemRestartTestFullDb);
+ FINALIZER(runClearTable);
+}
+TESTCASE("SR3",
+ "System restart test. Focus on testing restart from with\n"
+ "not all nodes alive when system went down\n"
+ "* 1. Load data\n"
+ "* 2. Restart 1 node -nostart\n"
+ "* 3. Update records\n"
+ "* 4. Restart cluster and verify records\n"
+ "* 5. Restart 1 node -nostart\n"
+ "* 6. Delete half of the records\n"
+ "* 7. Restart cluster and verify records\n"
+ "* 8. Restart 1 node -nostart\n"
+ "* 9. Delete all records\n"
+ "* 10. Restart cluster and verify records\n"){
+ INITIALIZER(runWaitStarted);
+ STEP(runSystemRestart3);
+ FINALIZER(runClearTable);
+}
+TESTCASE("SR4",
+ "System restart test. Focus on testing restart from with\n"
+ "not all nodes alive when system went down but running LCP at\n"
+ "high speed so that sometimes a TO is required to start cluster\n"
+ "* 1. Load data\n"
+ "* 2. Restart 1 node -nostart\n"
+ "* 3. Update records\n"
+ "* 4. Restart cluster and verify records\n"
+ "* 5. Restart 1 node -nostart\n"
+ "* 6. Delete half of the records\n"
+ "* 7. Restart cluster and verify records\n"
+ "* 8. Restart 1 node -nostart\n"
+ "* 9. Delete all records\n"
+ "* 10. Restart cluster and verify records\n"){
+ INITIALIZER(runWaitStarted);
+ STEP(runSystemRestart4);
+ FINALIZER(runClearTable);
+}
+TESTCASE("SR5",
+ "As SR4 but making restart aborts\n"
+ "* 1. Load data\n"
+ "* 2. Restart 1 node -nostart\n"
+ "* 3. Update records\n"
+ "* 4. Restart cluster and verify records\n"
+ "* 5. Restart 1 node -nostart\n"
+ "* 6. Delete half of the records\n"
+ "* 7. Restart cluster and verify records\n"
+ "* 8. Restart 1 node -nostart\n"
+ "* 9. Delete all records\n"
+ "* 10. Restart cluster and verify records\n"){
+ INITIALIZER(runWaitStarted);
+ STEP(runSystemRestart5);
+ FINALIZER(runClearTable);
+}
+NDBT_TESTSUITE_END(testSystemRestart);
+
+int main(int argc, const char** argv){
+ return testSystemRestart.execute(argc, argv);
+}
+
+
diff --git a/ndb/test/ndbapi/testTimeout/Makefile b/ndb/test/ndbapi/testTimeout/Makefile
new file mode 100644
index 00000000000..01a9df9887f
--- /dev/null
+++ b/ndb/test/ndbapi/testTimeout/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = testTimeout
+
+SOURCES = testTimeout.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testTimeout/testTimeout.cpp b/ndb/test/ndbapi/testTimeout/testTimeout.cpp
new file mode 100644
index 00000000000..de1d2cfc40b
--- /dev/null
+++ b/ndb/test/ndbapi/testTimeout/testTimeout.cpp
@@ -0,0 +1,261 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Test.hpp>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <random.h>
+#include <NdbConfig.hpp>
+
+
+int runLoadTable(NDBT_Context* ctx, NDBT_Step* step){
+
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+ if (hugoTrans.loadTable(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+
+#define CHECK(b) if (!(b)) { \
+ ndbout << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl; \
+ result = NDBT_FAILED; \
+ break; }
+
+int runTimeoutTrans(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ NdbConfig conf(GETNDB(step)->getNodeId()+1);
+ unsigned int nodeId = conf.getMasterNodeId();
+ int stepNo = step->getStepNo();
+ Uint32 timeoutVal;
+ if (!conf.getProperty(nodeId,
+ "DB",
+ "TransactionInactiveTimeout",
+ &timeoutVal)){
+ return NDBT_FAILED;
+ }
+ int minSleep = (int)(timeoutVal * 1.5);
+ int maxSleep = timeoutVal * 2;
+ ndbout << "TransactionInactiveTimeout="<<timeoutVal
+ << ", minSleep="<<minSleep
+ << ", maxSleep="<<maxSleep<<endl;
+
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ for (int l = 0; l < loops && result == NDBT_OK; l++){
+
+ do{
+ // Commit transaction
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, stepNo, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ int sleep = minSleep + myRandom48(maxSleep-minSleep);
+ ndbout << "Sleeping for " << sleep << " milliseconds" << endl;
+ NdbSleep_MilliSleep(sleep);
+
+ // Expect that transaction has timed-out
+ CHECK(hugoOps.execute_Commit(pNdb) == 237);
+
+ } while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ }
+
+ return result;
+}
+
+int runDontTimeoutTrans(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ NdbConfig conf(GETNDB(step)->getNodeId()+1);
+ unsigned int nodeId = conf.getMasterNodeId();
+ int stepNo = step->getStepNo();
+ Uint32 timeoutVal;
+ if (!conf.getProperty(nodeId,
+ "DB",
+ "TransactionInactiveTimeout",
+ &timeoutVal)){
+ return NDBT_FAILED;
+ }
+ int maxSleep = (int)(timeoutVal * 0.5);
+ ndbout << "TransactionInactiveTimeout="<<timeoutVal
+ << ", maxSleep="<<maxSleep<<endl;
+
+
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ for (int l = 0; l < loops && result == NDBT_OK; l++){
+
+ do{
+ // Commit transaction
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ CHECK(hugoOps.pkReadRecord(pNdb, stepNo, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ int sleep = myRandom48(maxSleep);
+ ndbout << "Sleeping for " << sleep << " milliseconds" << endl;
+ NdbSleep_MilliSleep(sleep);
+
+ // Expect that transaction has NOT timed-out
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+
+ } while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+
+ }
+
+
+ return result;
+}
+
+int runBuddyTransNoTimeout(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ NdbConfig conf(GETNDB(step)->getNodeId()+1);
+ unsigned int nodeId = conf.getMasterNodeId();
+ int stepNo = step->getStepNo();
+ Uint32 timeoutVal;
+ if (!conf.getProperty(nodeId,
+ "DB",
+ "TransactionInactiveTimeout",
+ &timeoutVal)){
+ return NDBT_FAILED;
+ }
+ int maxSleep = (int)(timeoutVal * 0.3);
+ ndbout << "TransactionInactiveTimeout="<<timeoutVal
+ << ", maxSleep="<<maxSleep<<endl;
+
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ for (int l = 1; l < loops && result == NDBT_OK; l++){
+
+ do{
+ // Start an insert trans
+ CHECK(hugoOps.startTransaction(pNdb) == 0);
+ int recordNo = records + (stepNo*loops) + l;
+ CHECK(hugoOps.pkInsertRecord(pNdb, recordNo, true) == 0);
+ CHECK(hugoOps.execute_NoCommit(pNdb) == 0);
+
+ for (int i = 0; i < 10; i++){
+ // Perform buddy scan reads
+ CHECK(hugoOps.scanReadRecords(pNdb) == 0);
+ CHECK(hugoOps.executeScanRead(pNdb) == 0);
+
+ int sleep = myRandom48(maxSleep);
+ ndbout << "Sleeping for " << sleep << " milliseconds" << endl;
+ NdbSleep_MilliSleep(sleep);
+ }
+
+ // Expect that transaction has NOT timed-out
+ CHECK(hugoOps.execute_Commit(pNdb) == 0);
+
+ } while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+
+ }
+
+
+ return result;
+}
+
+NDBT_TESTSUITE(testTimeout);
+TESTCASE("DontTimeoutTransaction",
+ "Test that the transaction does not timeout "\
+ "if we sleep during the transaction. Use a sleep "\
+ "value which is smaller than TransactionInactiveTimeout"){
+ INITIALIZER(runLoadTable);
+ STEPS(runDontTimeoutTrans, 1);
+ FINALIZER(runClearTable);
+}
+TESTCASE("DontTimeoutTransaction5",
+ "Test that the transaction does not timeout "\
+ "if we sleep during the transaction. Use a sleep "\
+ "value which is smaller than TransactionInactiveTimeout" \
+ "Five simultaneous threads"){
+ INITIALIZER(runLoadTable);
+ STEPS(runDontTimeoutTrans, 5);
+ FINALIZER(runClearTable);
+}
+TESTCASE("TimeoutTransaction",
+ "Test that the transaction does timeout "\
+ "if we sleep during the transaction. Use a sleep "\
+ "value which is larger than TransactionInactiveTimeout"){
+ INITIALIZER(runLoadTable);
+ STEPS(runTimeoutTrans, 1);
+ FINALIZER(runClearTable);
+}
+TESTCASE("TimeoutTransaction5",
+ "Test that the transaction does timeout " \
+ "if we sleep during the transaction. Use a sleep " \
+ "value which is larger than TransactionInactiveTimeout" \
+ "Five simultaneous threads"){
+ INITIALIZER(runLoadTable);
+ STEPS(runTimeoutTrans, 5);
+ FINALIZER(runClearTable);
+}
+TESTCASE("BuddyTransNoTimeout",
+ "Start a transaction and perform an insert with NoCommit. " \
+ "Start a buddy transaction wich performs long running scans " \
+ "and sleeps. " \
+ "The total sleep time is longer than TransactionInactiveTimeout" \
+ "Commit the first transaction, it should not have timed out."){
+ INITIALIZER(runLoadTable);
+ STEPS(runBuddyTransNoTimeout, 1);
+ FINALIZER(runClearTable);
+}
+TESTCASE("BuddyTransNoTimeout5",
+ "Start a transaction and perform an insert with NoCommit. " \
+ "Start a buddy transaction wich performs long running scans " \
+ "and sleeps. " \
+ "The total sleep time is longer than TransactionInactiveTimeout" \
+ "Commit the first transaction, it should not have timed out." \
+ "Five simultaneous threads"){
+ INITIALIZER(runLoadTable);
+ STEPS(runBuddyTransNoTimeout, 5);
+ FINALIZER(runClearTable);
+}
+NDBT_TESTSUITE_END(testTimeout);
+
+int main(int argc, const char** argv){
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ return testTimeout.execute(argc, argv);
+}
+
diff --git a/ndb/test/ndbapi/testTransactions/Makefile b/ndb/test/ndbapi/testTransactions/Makefile
new file mode 100644
index 00000000000..0279a526923
--- /dev/null
+++ b/ndb/test/ndbapi/testTransactions/Makefile
@@ -0,0 +1,10 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := testTransactions
+
+SOURCES := testTransactions.cpp
+CFLAGS_testTransactions.cpp := -I$(call fixpath,$(NDB_TOP)/include/kernel)
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/testTransactions/testTransactions.cpp b/ndb/test/ndbapi/testTransactions/testTransactions.cpp
new file mode 100644
index 00000000000..9ce928f8736
--- /dev/null
+++ b/ndb/test/ndbapi/testTransactions/testTransactions.cpp
@@ -0,0 +1,411 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT_Test.hpp>
+#include <NDBT_ReturnCodes.h>
+#include <HugoTransactions.hpp>
+#include <UtilTransactions.hpp>
+#include <NdbRestarter.hpp>
+#include <signaldata/DumpStateOrd.hpp>
+
+struct OperationTestCase {
+ const char * name;
+ bool preCond; // start transaction | insert | commit
+
+ // start transaction 1
+ const char * op1;
+ const int val1;
+
+ // no commit
+
+ // start transaction 2
+ const char * op2;
+ const int res2;
+ const int val2;
+ // no commit
+
+ // commit transaction 1
+ // commit transaction 2
+
+ // start transaction
+ // op3 = READ
+ const int res3;
+ const int val3;
+ // commit transaction
+};
+
+#define X -1
+
+OperationTestCase matrix[] = {
+ { "ReadRead", true, "READ", 1, "READ", 0, 1, 0, 1 },
+ { "ReadReadEx", true, "READ", 1, "READ-EX", 266, X, 0, 1 },
+ { "ReadSimpleRead", true, "READ", 1, "S-READ", 0, 1, 0, 1 },
+ { "ReadDirtyRead", true, "READ", 1, "D-READ", 0, 1, 0, 1 },
+ { "ReadInsert", true, "READ", 1, "INSERT", 266, X, 0, 1 },
+ { "ReadUpdate", true, "READ", 1, "UPDATE", 266, X, 0, 1 },
+ { "ReadDelete", true, "READ", 1, "DELETE", 266, X, 0, 1 },
+ { "ReadScan", true, "READ", 1, "SCAN", 0, 1, 0, 1 },
+ { "ReadScanHl", true, "READ", 1, "SCAN-HL", 0, 1, 0, 1 },
+ { "ReadScanEx", true, "READ", 1, "SCAN-EX", 274, X, 0, 1 },
+#if 0
+ { "ReadScanUp", true, "READ", 1, "SCAN-UP", 266, X, 0, 1 },
+ { "ReadScanDe", true, "READ", 1, "SCAN-DE", 266, X, 0, 1 },
+#endif
+
+ { "ScanRead", true, "SCAN", 1, "READ", 0, 1, 0, 1 },
+ { "ScanReadEx", true, "SCAN", 1, "READ-EX", 0, 1, 0, 1 },
+ { "ScanSimpleRead", true, "SCAN", 1, "S-READ", 0, 1, 0, 1 },
+ { "ScanDirtyRead", true, "SCAN", 1, "D-READ", 0, 1, 0, 1 },
+ { "ScanInsert", true, "SCAN", 1, "INSERT", 630, X, 0, 1 },
+ { "ScanUpdate", true, "SCAN", 1, "UPDATE", 0, 2, 0, 2 },
+ { "ScanDelete", true, "SCAN", 1, "DELETE", 0, X, 626, X },
+ { "ScanScan", true, "SCAN", 1, "SCAN", 0, 1, 0, 1 },
+ { "ScanScanHl", true, "SCAN", 1, "SCAN-HL", 0, 1, 0, 1 },
+ { "ScanScanEx", true, "SCAN", 1, "SCAN-EX", 0, 1, 0, 1 },
+#if 0
+ { "ScanScanUp", true, "SCAN", 1, "SCAN-UP", 266, X, 0, 1 },
+ { "ScanScanDe", true, "SCAN", 1, "SCAN-DE", 266, X, 0, 1 },
+#endif
+
+ { "ScanHlRead", true, "SCAN-HL",1, "READ", 0, 1, 0, 1 },
+ { "ScanHlReadEx", true, "SCAN-HL",1, "READ-EX", 266, 1, 0, 1 },
+ { "ScanHlSimpleRead", true, "SCAN-HL",1, "S-READ", 0, 1, 0, 1 },
+ { "ScanHlDirtyRead", true, "SCAN-HL",1, "D-READ", 0, 1, 0, 1 },
+ { "ScanHlInsert", true, "SCAN-HL",1, "INSERT", 266, X, 0, 1 },
+ { "ScanHlUpdate", true, "SCAN-HL",1, "UPDATE", 266, 2, 0, 1 },
+ { "ScanHlDelete", true, "SCAN-HL",1, "DELETE", 266, X, 0, 1 },
+ { "ScanHlScan", true, "SCAN-HL",1, "SCAN", 0, 1, 0, 1 },
+ { "ScanHlScanHl", true, "SCAN-HL",1, "SCAN-HL", 0, 1, 0, 1 },
+ { "ScanHlScanEx", true, "SCAN-HL",1, "SCAN-EX", 274, X, 0, 1 },
+#if 0
+ { "ScanHlScanUp", true, "SCAN-HL",1, "SCAN-UP", 266, X, 0, 1 },
+ { "ScanHlScanDe", true, "SCAN-HL",1, "SCAN-DE", 266, X, 0, 1 },
+#endif
+
+ { "ScanExRead", true, "SCAN-EX",1, "READ", 266, 1, 0, 1 },
+ { "ScanExReadEx", true, "SCAN-EX",1, "READ-EX", 266, 1, 0, 1 },
+ { "ScanExSimpleRead", true, "SCAN-EX",1, "S-READ", 266, 1, 0, 1 },
+ { "ScanExDirtyRead", true, "SCAN-EX",1, "D-READ", 0, 1, 0, 1 },
+ { "ScanExInsert", true, "SCAN-EX",1, "INSERT", 266, X, 0, 1 },
+ { "ScanExUpdate", true, "SCAN-EX",1, "UPDATE", 266, 2, 0, 1 },
+ { "ScanExDelete", true, "SCAN-EX",1, "DELETE", 266, X, 0, 1 },
+ { "ScanExScan", true, "SCAN-EX",1, "SCAN", 274, X, 0, 1 },
+ { "ScanExScanHl", true, "SCAN-EX",1, "SCAN-HL", 274, X, 0, 1 },
+ { "ScanExScanEx", true, "SCAN-EX",1, "SCAN-EX", 274, X, 0, 1 },
+#if 0
+ { "ScanExScanUp", true, "SCAN-EX",1, "SCAN-UP", 266, X, 0, 1 },
+ { "ScanExScanDe", true, "SCAN-EX",1, "SCAN-DE", 266, X, 0, 1 },
+#endif
+
+ { "ReadExRead", true, "READ-EX",1, "READ", 266, X, 0, 1 },
+ { "ReadExReadEx", true, "READ-EX",1, "READ-EX", 266, X, 0, 1 },
+ { "ReadExSimpleRead", true, "READ-EX",1, "S-READ", 266, X, 0, 1 },
+ { "ReadExDirtyRead", true, "READ-EX",1, "D-READ", 0, 1, 0, 1 },
+ { "ReadExInsert", true, "READ-EX",1, "INSERT", 266, X, 0, 1 },
+ { "ReadExUpdate", true, "READ-EX",1, "UPDATE", 266, X, 0, 1 },
+ { "ReadExDelete", true, "READ-EX",1, "DELETE", 266, X, 0, 1 },
+ { "ReadExScan", true, "READ-EX",1, "SCAN", 274, 1, 0, 1 },
+ { "ReadExScanHl", true, "READ-EX",1, "SCAN-HL", 274, 1, 0, 1 },
+ { "ReadExScanEx", true, "READ-EX",1, "SCAN-EX", 274, X, 0, 1 },
+#if 0
+ { "ReadExScanUp", true, "READ-EX",1, "SCAN-UP", 266, X, 0, 1 },
+ { "ReadExScanDe", true, "READ-EX",1, "SCAN-DE", 266, X, 0, 1 },
+#endif
+
+ { "InsertRead", false, "INSERT", 1, "READ", 266, X, 0, 1 },
+ { "InsertReadEx", false, "INSERT", 1, "READ-EX", 266, X, 0, 1 },
+ { "InsertSimpleRead",false, "INSERT", 1, "S-READ", 266, X, 0, 1 },
+ { "InsertDirtyRead", false, "INSERT", 1, "D-READ", 626, X, 0, 1 },
+ { "InsertInsert", false, "INSERT", 1, "INSERT", 266, X, 0, 1 },
+ { "InsertUpdate", false, "INSERT", 1, "UPDATE", 266, X, 0, 1 },
+ { "InsertDelete", false, "INSERT", 1, "DELETE", 266, X, 0, 1 },
+ { "InsertScan", false, "INSERT", 1, "SCAN", 274, X, 0, 1 },
+ { "InsertScanHl", false, "INSERT", 1, "SCAN-HL", 274, X, 0, 1 },
+ { "InsertScanEx", false, "INSERT", 1, "SCAN-EX", 274, X, 0, 1 },
+#if 0
+ { "InsertScanUp", false, "INSERT", 1, "SCAN-UP", 266, X, 0, 1 },
+ { "InsertScanDe", false, "INSERT", 1, "SCAN-DE", 266, X, 0, 1 },
+#endif
+
+ { "UpdateRead", true, "UPDATE", 2, "READ", 266, X, 0, 2 },
+ { "UpdateReadEx", true, "UPDATE", 2, "READ-EX", 266, X, 0, 2 },
+ { "UpdateSimpleRead", true, "UPDATE", 2, "S-READ", 266, X, 0, 2 },
+ { "UpdateDirtyRead", true, "UPDATE", 2, "D-READ", 0, 1, 0, 2 },
+ { "UpdateInsert", true, "UPDATE", 2, "INSERT", 266, X, 0, 2 },
+ { "UpdateUpdate", true, "UPDATE", 2, "UPDATE", 266, X, 0, 2 },
+ { "UpdateDelete", true, "UPDATE", 2, "DELETE", 266, X, 0, 2 },
+ { "UpdateScan", true, "UPDATE", 2, "SCAN", 274, X, 0, 2 },
+ { "UpdateScanHl", true, "UPDATE", 2, "SCAN-HL", 274, X, 0, 2 },
+ { "UpdateScanEx", true, "UPDATE", 2, "SCAN-EX", 274, X, 0, 2 },
+#if 0
+ { "UpdateScanUp", true, "UPDATE", 2, "SCAN-UP", 266, X, 0, 2 },
+ { "UpdateScanDe", true, "UPDATE", 2, "SCAN-DE", 266, X, 0, 2 },
+#endif
+
+ { "DeleteRead", true, "DELETE", X, "READ", 266, X, 626, X },
+ { "DeleteReadEx", true, "DELETE", X, "READ-EX", 266, X, 626, X },
+ { "DeleteSimpleRead", true, "DELETE", X, "S-READ", 266, X, 626, X },
+ { "DeleteDirtyRead", true, "DELETE", X, "D-READ", 0, 1, 626, X },
+ { "DeleteInsert", true, "DELETE", X, "INSERT", 266, X, 626, X },
+ { "DeleteUpdate", true, "DELETE", X, "UPDATE", 266, X, 626, X },
+ { "DeleteDelete", true, "DELETE", X, "DELETE", 266, X, 626, X },
+ { "DeleteScan", true, "DELETE", X, "SCAN", 274, X, 626, X },
+ { "DeleteScanHl", true, "DELETE", X, "SCAN-HL", 274, X, 626, X },
+ { "DeleteScanEx", true, "DELETE", X, "SCAN-EX", 274, X, 626, X },
+#if 0
+ { "DeleteScanUp", true, "DELETE", X, "SCAN-UP", 266, X, 626, X },
+ { "DeleteScanDe", true, "DELETE", X, "SCAN-DE", 266, X, 626, X }
+#endif
+
+};
+
+#define CHECK(a, b) { int x = a; int y = b; if (x != y) { \
+ g_err << "ERR: "<< step->getName() \
+ << " failed on line " << __LINE__ << endl << " " \
+ << x << " != " << y << endl;\
+ result = NDBT_FAILED; \
+ break; } }
+
+int
+runOp(HugoOperations & hugoOps,
+ Ndb * pNdb,
+ const char * op,
+ int value){
+
+#define C2(x) if(!(x)) {\
+ g_err << "ERR: failed on line " << __LINE__ << endl; \
+ return NDBT_FAILED; }
+
+ if(strcmp(op, "READ") == 0){
+ C2(hugoOps.pkReadRecord(pNdb, 1, false, 1) == 0);
+ } else if(strcmp(op, "READ-EX") == 0){
+ C2(hugoOps.pkReadRecord(pNdb, 1, true, 1) == 0);
+ } else if(strcmp(op, "S-READ") == 0){
+ C2(hugoOps.pkSimpleReadRecord(pNdb, 1, 1) == 0);
+ } else if(strcmp(op, "D-READ") == 0){
+ C2(hugoOps.pkDirtyReadRecord(pNdb, 1, 1) == 0);
+ } else if(strcmp(op, "INSERT") == 0){
+ C2(hugoOps.pkInsertRecord(pNdb, 1, 1, value) == 0);
+ } else if(strcmp(op, "UPDATE") == 0){
+ C2(hugoOps.pkUpdateRecord(pNdb, 1, 1, value) == 0);
+ } else if(strcmp(op, "DELETE") == 0){
+ C2(hugoOps.pkDeleteRecord(pNdb, 1, 1) == 0);
+ } else if(strcmp(op, "SCAN") == 0){
+ C2(hugoOps.scanReadRecords(pNdb) == 0);
+ } else if(strcmp(op, "SCAN-HL") == 0){
+ C2(hugoOps.scanReadRecords(pNdb, 240, HugoOperations::SL_ReadHold) == 0);
+ } else if(strcmp(op, "SCAN-EX") == 0){
+ C2(hugoOps.scanReadRecords(pNdb, 240, HugoOperations::SL_Exclusive) == 0);
+ } else {
+ g_err << __FILE__ << " - " << __LINE__
+ << ": Unknown operation" << op << endl;
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+int
+checkVal(HugoOperations & hugoOps,
+ const char * op,
+ int value,
+ int result){
+
+ if(result != 0)
+ return NDBT_OK;
+
+ if(strcmp(op, "READ") == 0){
+ } else if(strcmp(op, "READ-EX") == 0){
+ } else if(strcmp(op, "S-READ") == 0){
+ } else if(strcmp(op, "D-READ") == 0){
+ } else if(strcmp(op, "SCAN") == 0){
+ } else if(strcmp(op, "SCAN-HL") == 0){
+ } else if(strcmp(op, "SCAN-EX") == 0){
+ } else {
+ return NDBT_OK;
+ }
+
+ return hugoOps.verifyUpdatesValue(value);
+}
+
+#define TIMEOUT 100
+
+int
+setTransactionTimeout(NDBT_Context* ctx, NDBT_Step* step){
+ NdbRestarter restarter;
+
+ int val[] =
+ { DumpStateOrd::TcSetTransactionTimeout, TIMEOUT };
+ if(restarter.dumpStateAllNodes(val, 2) != 0){
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+}
+
+int
+runTwoTrans1(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations T1(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ const char * op1 = ctx->getProperty("op1", "NONE");
+ int val1 = ctx->getProperty("val1", ~0);
+
+ do {
+ // Insert, read
+ CHECK(T1.startTransaction(pNdb), 0);
+ CHECK(runOp(T1, pNdb, op1, val1), 0);
+ CHECK(T1.execute_NoCommit(pNdb), 0);
+ CHECK(checkVal(T1, op1, val1, 0), 0);
+
+ ctx->setProperty("T1-1-Complete", 1);
+ while(ctx->getProperty("T2-Complete", (Uint32)0) == 0){
+ T1.refresh();
+ NdbSleep_MilliSleep(10);
+ }
+
+ CHECK(T1.execute_Commit(pNdb), 0);
+
+ } while(false);
+ T1.closeTransaction(pNdb);
+
+ if(result != NDBT_OK)
+ return result;
+
+ const int res3 = ctx->getProperty("res3", ~0);
+ const int val3 = ctx->getProperty("val3", ~0);
+
+ do {
+ CHECK(T1.startTransaction(pNdb), 0);
+ CHECK(runOp(T1, pNdb, "READ", 0), 0);
+ CHECK(T1.execute_Commit(pNdb), res3);
+ CHECK(checkVal(T1, "READ", val3, res3), 0);
+ } while(false);
+ T1.closeTransaction(pNdb);
+
+ return result;
+}
+
+int
+runTwoTrans2(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations T2(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ const char * op2 = ctx->getProperty("op2", "NONE");
+ const int res2 = ctx->getProperty("res2", ~0);
+ const int val2 = ctx->getProperty("val2", ~0);
+
+ while(ctx->getProperty("T1-1-Complete", (Uint32)0) == 0 &&
+ !ctx->isTestStopped()){
+ NdbSleep_MilliSleep(100);
+ }
+
+ if(!ctx->isTestStopped()){
+ do {
+ CHECK(T2.startTransaction(pNdb), 0);
+ CHECK(runOp(T2, pNdb, op2, val2), 0);
+ CHECK(T2.execute_NoCommit(pNdb), res2);
+ CHECK(checkVal(T2, op2, val2, res2), 0);
+ if(res2 == 0){
+ CHECK(T2.execute_Commit(pNdb), res2);
+ }
+ } while(false);
+ T2.closeTransaction(pNdb);
+ }
+
+ ctx->setProperty("T2-Complete", 1);
+
+ return result;
+}
+
+int
+runInsertRecord(NDBT_Context* ctx, NDBT_Step* step){
+ int result = NDBT_OK;
+ HugoOperations hugoOps(*ctx->getTab());
+ Ndb* pNdb = GETNDB(step);
+
+ do{
+ // Insert, insert
+ CHECK(hugoOps.startTransaction(pNdb), 0);
+ CHECK(hugoOps.pkInsertRecord(pNdb, 1, 1, 1), 0);
+ CHECK(hugoOps.execute_Commit(pNdb), 0);
+ } while(false);
+
+ hugoOps.closeTransaction(pNdb);
+
+ return result;
+}
+
+int
+runClearTable(NDBT_Context* ctx, NDBT_Step* step){
+ int records = ctx->getNumRecords();
+
+ UtilTransactions utilTrans(*ctx->getTab());
+ if (utilTrans.clearTable2(GETNDB(step), records, 240) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int
+main(int argc, const char** argv){
+
+ NDBT_TestSuite ts("testOperations");
+ for(Uint32 i = 0; i<sizeof(matrix)/sizeof(matrix[0]); i++){
+ NDBT_TestCaseImpl1 *pt = new NDBT_TestCaseImpl1(&ts, matrix[i].name, "");
+
+ pt->addInitializer(new NDBT_Initializer(pt,
+ "runClearTable",
+ runClearTable));
+
+ if(matrix[i].preCond){
+ pt->addInitializer(new NDBT_Initializer(pt,
+ "runInsertRecord",
+ runInsertRecord));
+ }
+
+ pt->addInitializer(new NDBT_Initializer(pt,
+ "setTransactionTimeout",
+ setTransactionTimeout));
+
+ pt->setProperty("op1", matrix[i].op1);
+ pt->setProperty("val1", matrix[i].val1);
+
+ pt->setProperty("op2", matrix[i].op2);
+ pt->setProperty("res2", matrix[i].res2);
+ pt->setProperty("val2", matrix[i].val2);
+
+ pt->setProperty("res3", matrix[i].res3);
+ pt->setProperty("val3", matrix[i].val3);
+
+ pt->addStep(new NDBT_ParallelStep(pt,
+ matrix[i].name,
+ runTwoTrans1));
+ pt->addStep(new NDBT_ParallelStep(pt,
+ matrix[i].name,
+ runTwoTrans2));
+ pt->addFinalizer(new NDBT_Finalizer(pt,
+ "runClearTable",
+ runClearTable));
+
+ ts.addTest(pt);
+ }
+
+ return ts.execute(argc, argv);
+}
+
diff --git a/ndb/test/ndbapi/test_event/Makefile b/ndb/test/ndbapi/test_event/Makefile
new file mode 100644
index 00000000000..6299fa47845
--- /dev/null
+++ b/ndb/test/ndbapi/test_event/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := test_event
+
+SOURCES := test_event.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/test_event/test_event.cpp b/ndb/test/ndbapi/test_event/test_event.cpp
new file mode 100644
index 00000000000..40fc1c6defa
--- /dev/null
+++ b/ndb/test/ndbapi/test_event/test_event.cpp
@@ -0,0 +1,142 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NDBT_Test.hpp"
+#include "NDBT_ReturnCodes.h"
+#include "HugoTransactions.hpp"
+#include "UtilTransactions.hpp"
+#include "TestNdbEventOperation.hpp"
+
+#define GETNDB(ps) ((NDBT_NdbApiStep*)ps)->getNdb()
+
+int runCreateEvent(NDBT_Context* ctx, NDBT_Step* step)
+{
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ if (hugoTrans.createEvent(GETNDB(step)) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int theThreadIdCounter = 0;
+
+int runEventOperation(NDBT_Context* ctx, NDBT_Step* step)
+{
+ int tId = theThreadIdCounter++;
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ EventOperationStats stats;
+
+ g_info << "***** Id " << tId << endl;
+
+ // sleep(tId);
+
+ if (hugoTrans.eventOperation(GETNDB(step), (void*)&stats, 3*records) != 0){
+ return NDBT_FAILED;
+ }
+
+ int ret;
+ if (stats.n_inserts == records &&
+ stats.n_deletes == records &&
+ stats.n_updates == records &&
+ stats.n_consecutive == 3 &&
+ stats.n_duplicates == 0)
+ ret = NDBT_OK;
+ else
+ ret = NDBT_FAILED;
+
+ if (ret == NDBT_FAILED) {
+ ndbout << "n_inserts = " << stats.n_inserts << endl;
+ ndbout << "n_deletes = " << stats.n_deletes << endl;
+ ndbout << "n_updates = " << stats.n_updates << endl;
+ ndbout << "n_consecutive = " << stats.n_consecutive << endl;
+ ndbout << "n_duplicates = " << stats.n_duplicates << endl;
+ ndbout << "n_inconsistent_gcis = " << stats.n_inconsistent_gcis << endl;
+ }
+
+ return ret;
+}
+
+int runEventLoad(NDBT_Context* ctx, NDBT_Step* step)
+{
+ int loops = ctx->getNumLoops();
+ int records = ctx->getNumRecords();
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ sleep(5);
+ sleep(theThreadIdCounter);
+
+ if (hugoTrans.loadTable(GETNDB(step), records, 1, true, loops) != 0){
+ return NDBT_FAILED;
+ }
+ if (hugoTrans.pkUpdateRecords(GETNDB(step), records, 1, loops) != 0){
+ return NDBT_FAILED;
+ }
+ if (hugoTrans.pkDelRecords(GETNDB(step), records, 1, true, loops) != 0){
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int runDropEvent(NDBT_Context* ctx, NDBT_Step* step)
+{
+ HugoTransactions hugoTrans(*ctx->getTab());
+
+ theThreadIdCounter = 0;
+ // if (hugoTrans.createEvent(GETNDB(step)) != 0){
+ // return NDBT_FAILED;
+ // }
+ return NDBT_OK;
+}
+
+// INITIALIZER(runInsert);
+// STEP(runPkRead);
+// VERIFIER(runVerifyInsert);
+// FINALIZER(runClearTable);
+
+NDBT_TESTSUITE(test_event);
+TESTCASE("BasicEventOperation",
+ "Verify that we can listen to Events"
+ "NOTE! No errors are allowed!" ){
+ INITIALIZER(runCreateEvent);
+ STEP(runEventOperation);
+ STEP(runEventOperation);
+ STEP(runEventOperation);
+ STEP(runEventOperation);
+ STEP(runEventLoad);
+ FINALIZER(runDropEvent);
+}
+NDBT_TESTSUITE_END(test_event);
+
+#if 0
+NDBT_TESTSUITE(test_event);
+TESTCASE("ParallellEventOperation",
+ "Verify that we can listen to Events in Parallell"
+ "NOTE! No errors are allowed!" ){
+ INITIALIZER(runCreateAllEvent);
+ STEP(runEventOperation);
+ FINALIZER(runDropEvent);
+}
+NDBT_TESTSUITE_END(test_event);
+#endif
+
+int main(int argc, const char** argv){
+ return test_event.execute(argc, argv);
+}
+
diff --git a/ndb/test/ndbapi/vw_test/Makefile b/ndb/test/ndbapi/vw_test/Makefile
new file mode 100644
index 00000000000..144873dcc69
--- /dev/null
+++ b/ndb/test/ndbapi/vw_test/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := vw_test
+BIN_TARGET_LIBS := orafunctr decode cirk inifunc
+BIN_TARGET_LIBS_DIRS := /home/ndb/junk/vw/ndb/lib
+
+SOURCES := cdrserver.C
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/ndbapi/vw_test/bcd.h b/ndb/test/ndbapi/vw_test/bcd.h
new file mode 100644
index 00000000000..ce1309693c8
--- /dev/null
+++ b/ndb/test/ndbapi/vw_test/bcd.h
@@ -0,0 +1,27 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <math.h>
+#include <stdio.h>
+
+struct bcdtab {
+ char tab[3];
+};
+
+int dec2hex(int dec,int last);
+int bcd_code (char *bcd_in,char *bcd_out);
+int bcd_decode (int bcd_len,char *bcd_in,char *bcd_out);
+int bcd_decode2 (int bcd_len,char *bcd_in,char *bcd_out);
diff --git a/ndb/test/ndbapi/vw_test/cdrserver.cpp b/ndb/test/ndbapi/vw_test/cdrserver.cpp
new file mode 100644
index 00000000000..3c3f32e8886
--- /dev/null
+++ b/ndb/test/ndbapi/vw_test/cdrserver.cpp
@@ -0,0 +1,1633 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* **************************************************************** */
+/* */
+/* S E R V . T C P */
+/* * This is an example program that demonstrates the use of */
+/* stream sockets as an IPC mechanism. This contains the server, */
+/* and is intended to operate in conjunction with the client */
+/* program found in client.tcp. Together, these two programs */
+/* demonstrate many of the features of sockets, as well as good */
+/* conventions for using these features. */
+/* * This program provides a service called "example". In order for*/
+/* it to function, an entry for it needs to exist in the */
+/* ./etc/services file. The port address for this service can be */
+/* any port number that is likely to be unused, such as 22375, */
+/* for example. The host on which the client will be running */
+/* must also have the same entry (same port number) in its */
+/* ./etc/services file. */
+/* **************************************************************** */
+/******** NDB INCLUDE ******/
+#include <NdbApi.hpp>
+/***************************/
+/*#include <sys/shm.h>*/
+#include <unistd.h>
+#include <stdlib.h>
+#include <strings.h>
+#include <pthread.h>
+#include <sys/sem.h>
+#include <sys/types.h>
+#include <sys/shm.h>
+#include <sys/socket.h>
+#include <errno.h>
+#include <netinet/in.h>
+#include <signal.h>
+#include <stdio.h>
+#include <netdb.h>
+#include <time.h>
+#include <synch.h>
+#include <sched.h>
+#include <limits.h>
+#include <math.h>
+
+extern "C" {
+#include "utv.h"
+#include "vcdrfunc.h"
+#include "bcd.h"
+}
+
+#ifndef TESTLEV
+#define TESTLEV
+#endif
+//#define DEBUG
+//#define MYDEBUG
+//#define SETDBG
+
+//#define ops_before_exe 64
+#define MAXOPSEXEC 1024
+
+/* Used in nanosleep */
+/**** NDB ********/
+static int bTestPassed;
+void create_table(Ndb* pMyNdb);
+void error_handler(const char* errorText);
+/*****************/
+static struct timespec tmspec1;
+static int server(long int);
+
+/* Function for initiating the cdr-area and make it clean for ongoing calls */
+
+static int s; /* connected socket descriptor */
+static int ls; /* listen socket descriptor */
+
+static struct hostent *hp; /* pointer to host info for remote host */
+static struct servent *sp; /* pointer to service information */
+
+struct linger linger; /* allow a lingering, graceful close; */
+ /* used when setting SO_LINGER */
+
+static struct sockaddr_in myaddr_in; /* for local socket address */
+static struct sockaddr_in peeraddr_in; /* for peer socket address */
+
+static FILE *fi; /* Log output */
+static char temp[600]="";
+
+static int ops_before_exe = 1; /* Number of operations per execute, default is 1,
+ but it can be changed with the -o parameter. */
+
+/*----------------------------------------------------------------------
+
+ M A I N
+ * This routine starts the server. It forks, leaving the child
+ to do all the work, so it does not have to be run in the
+ background. It sets up the listen socket, and for each incoming
+ connection, it forks a child process to process the data. It
+ will loop forever, until killed by a signal.
+
+ ----------------------------------------------------------------------*/
+
+/****** NDB *******/
+static char *tableName = "VWTABLE";
+/******************/
+
+#include <iostream>
+using namespace std;
+
+int main(int argc, const char** argv)
+{
+ /******** NDB ***********/
+ /*
+ Ndb MyNdb( "TEST_DB" );
+ int tTableId;
+ */
+ /************************/
+ char tmpbuf[400];
+ /* Loop and status variables */
+ int i,j,found;
+
+ /* Used by the server */
+ int addrlen;
+
+ /* return code used with functions */
+ int rc;
+
+ i = 1;
+ while (argc > 1)
+ {
+ if (strcmp(argv[i], "-o") == 0)
+ {
+ ops_before_exe = atoi(argv[i+1]);
+ if ((ops_before_exe < 1) || (ops_before_exe > MAXOPSEXEC))
+ {
+ cout << "Number of operations per execute must be at least 1, and at most " << MAXOPSEXEC << endl;
+ exit(1);
+ }
+
+ }
+ else
+ {
+ cout << "Invalid parameter!" << endl << "Look in cdrserver.C for more info." << endl;
+ exit(1);
+ }
+
+ argc -= 2;
+ i = i + 2;
+ }
+
+
+ /* Setup log handling */
+ logname(temp,"Cdrserver","Mother","");
+ puts(temp);
+ fi=fopen(temp,"w");
+ if (fi == NULL)
+ {
+ perror(argv[0]);
+ exit(EXIT_FAILURE);
+ }
+ m2log(fi,"Initiation of program");
+
+ /***** NDB ******/
+ /*
+ MyNdb.init();
+ if (MyNdb.waitUntilReady(30) != 0)
+ {
+ puts("Not ready");
+ exit(-1);
+ }
+ tTableId = MyNdb.getTable()->openTable(tableName);
+ if (tTableId == -1)
+ {
+ printf("%d: Creating table",getpid());
+ create_table(&MyNdb);
+ }
+ else printf("%d: Table already create",getpid());
+ */
+
+ /****************/
+
+ /* clear out address structures */
+ memset ((char *)&myaddr_in, 0, sizeof(struct sockaddr_in));
+ memset ((char *)&peeraddr_in, 0, sizeof(struct sockaddr_in));
+
+ m2log(fi,"Socket setup starting");
+
+ /* Set up address structure for the listen socket. */
+ myaddr_in.sin_family = AF_INET;
+
+ /* The server should listen on the wildcard address, */
+ /* rather than its own internet address. This is */
+ /* generally good practice for servers, because on */
+ /* systems which are connected to more than one */
+ /* network at once will be able to have one server */
+ /* listening on all networks at once. Even when the */
+ /* host is connected to only one network, this is good */
+ /* practice, because it makes the server program more */
+ /* portable. */
+
+ myaddr_in.sin_addr.s_addr = INADDR_ANY;
+ /* Find the information for the "cdrserver" server */
+ /* in order to get the needed port number. */
+
+ sp = getservbyname ("cdrserver", "tcp");
+ if (sp == NULL) {
+ m2log(fi,"Service cdrserver not found in /etc/services");
+ m2log(fi,"Terminating.");
+ exit(EXIT_FAILURE);
+ }
+
+ myaddr_in.sin_port = sp->s_port;
+
+ /* Create the listen socket.i */
+
+ ls = socket (AF_INET, SOCK_STREAM, 0);
+ if (ls == -1) {
+ m2log(fi,"Unable to create socket");
+ m2log(fi,"Terminating.");
+ exit(EXIT_FAILURE);
+ }
+ printf("Socket created\n");
+ printf("Wait..........\n");
+ /* Bind the listen address to the socket. */
+ if (bind(ls,(struct sockaddr*)&myaddr_in, sizeof(struct sockaddr_in)) == -1) {
+ m2log(fi,"Unable to bind address");
+ m2log(fi,"Terminating.");
+ exit(EXIT_FAILURE);
+ }
+
+ /* Initiate the listen on the socket so remote users */
+ /* can connect. The listen backlog is set to 5, which */
+ /* is the largest currently supported. */
+
+ if (listen(ls, 5) == -1) {
+ m2log(fi,"Unable to listen on socket");
+ m2log(fi,"Terminating.");
+ exit(EXIT_FAILURE);
+ }
+
+ /* Now, all the initialization of the server is */
+ /* complete, and any user errors will have already */
+ /* been detected. Now we can fork the daemon and */
+ /* return to the user. We need to do a setpgrp */
+ /* so that the daemon will no longer be associated */
+ /* with the user's control terminal. This is done */
+ /* before the fork, so that the child will not be */
+ /* a process group leader. Otherwise, if the child */
+ /* were to open a terminal, it would become associated */
+ /* with that terminal as its control terminal. It is */
+ /* always best for the parent to do the setpgrp. */
+
+ m2log(fi,"Socket setup completed");
+ m2log(fi,"Start server");
+
+ setpgrp();
+
+ /* Initiate the tmspec struct for use with nanosleep() */
+ tmspec1.tv_sec = 0;
+ tmspec1.tv_nsec = 1;
+
+ printf("Waiting for client to connect.........\n");
+ printf("Done\n");
+ switch (fork()) {
+ case -1: /* Unable to fork, for some reason. */
+ m2log(fi,"Failed to start server");
+ m2log(fi,"Terminating.");
+ fclose(fi);
+ perror(argv[0]);
+ fprintf(stderr, "%s: unable to fork daemon\n", argv[0]);
+ exit(EXIT_FAILURE);
+
+ break;
+ case 0: /* The child process (daemon) comes here. */
+ m2log(fi,"Server started");
+
+ /* Close stdin and stderr so that they will not */
+ /* be kept open. Stdout is assumed to have been */
+ /* redirected to some logging file, or /dev/null. */
+ /* From now on, the daemon will not report any */
+ /* error messages. This daemon will loop forever, */
+ /* waiting for connections and forking a child */
+ /* server to handle each one. */
+
+ close((int)stdin);
+ close((int)stderr);
+ /* Set SIGCLD to SIG_IGN, in order to prevent */
+ /* the accumulation of zombies as each child */
+ /* terminates. This means the daemon does not */
+ /* have to make wait calls to clean them up. */
+
+ signal(SIGCLD, SIG_IGN);
+ for(EVER) {
+ if ((checkchangelog(fi,temp))==0)
+ m2log(fi,"Waiting for connection");
+ /* Note that addrlen is passed as a pointer */
+ /* so that the accept call can return the */
+ /* size of the returned address. */
+
+ addrlen = sizeof(struct sockaddr_in);
+
+ /* This call will block until a new */
+ /* connection arrives. Then, it will */
+ /* return the address of the connecting */
+ /* peer, and a new socket descriptor, s, */
+ /* for that connection. */
+
+ s = accept(ls,(struct sockaddr*) &peeraddr_in, &addrlen);
+ #ifdef MYDEBUG
+ puts("accepted");
+ #endif
+ if ((checkchangelog(fi,temp))==0)
+ m2log(fi,"Connection attempt from a client");
+ if ((checkchangelog(fi,temp))==0)
+ m2log(fi,"Start communication server");
+
+ if ( s == -1) exit(EXIT_FAILURE);
+ switch (fork()) {
+ case -1: /* Can't fork, just exit. */
+ if ((checkchangelog(fi,temp))==0)
+ m2log(fi,"Start communication server failed.");
+ exit(EXIT_FAILURE);
+ break;
+ case 0: /* Child process comes here. */
+
+ /* Get clients adress and save it in the info area */
+ /* Keep track of how many times the client connects to the server */
+ printf("Connect attempt from client %u\n",peeraddr_in.sin_addr.s_addr);
+ server(peeraddr_in.sin_addr.s_addr);
+ exit(EXIT_FAILURE);
+ break;
+ default: /* Daemon process comes here. */
+ /* The daemon needs to remember */
+ /* to close the new accept socket */
+ /* after forking the child. This */
+ /* prevents the daemon from running */
+ /* out of file descriptor space. It */
+ /* also means that when the server */
+ /* closes the socket, that it will */
+ /* allow the socket to be destroyed */
+ /* since it will be the last close. */
+ close(s);
+ break;
+ }
+ }
+ default: /* Parent process comes here. */
+ exit(EXIT_FAILURE);
+ }
+ return EXIT_SUCCESS;
+}
+
+/*----------------------------------------------------------------------
+
+ S E R V E R
+ * This is the actual server routine that the daemon forks to
+ handle each individual connection. Its purpose is to receive
+ the request packets from the remote client, process them,
+ and return the results to the client. It will also write some
+ logging information to stdout.
+
+ ----------------------------------------------------------------------*/
+
+server(long int servernum)
+{
+ /******** NDB ***********/
+ Ndb MyNdb( "TEST_DB" );
+ int tTableId;
+ NdbConnection *MyTransaction;
+ NdbOperation *MyOperation;
+ int check;
+ int c1 = 0;
+ int c2 = 0;
+ int c3 = 0;
+ int c4 = 0;
+ int act_index = 0;
+ /************************/
+ register unsigned int reqcnt; /* keeps count of number of requests */
+ register unsigned int i; /* Loop counters */
+ register int x;
+ register short done; /* Loop variable */
+ short int found;
+
+ /* The server index number */
+ int thisServer;
+
+ /* Variables used to keep track of some statistics */
+ time_t ourtime;
+ time_t tmptime;
+ int tmpvalue;
+ long int tmptransfer;
+ long int transfer;
+ int ops = 0;
+
+ /* Variables used by the server */
+ char buf[400]; /* This example uses 10 byte messages. */
+ char *inet_ntoa();
+ char *hostname; /* points to the remote host's name string */
+ int len;
+ int rcvbuf_size;
+
+ long ctid;
+
+ unsigned char uc;
+
+ /* Variables used by the logging facilitiy */
+ char msg[600];
+ char crap[600];
+ char lognamn[600];
+
+ FILE *log;
+
+ /* scheduling parameter for pthread */
+ struct sched_param param1,param2,param3;
+
+ /* Header information */
+ /* cdrtype not used */
+ /*short cdrtype; */ /* 1 CDR Typ */
+ short cdrlen; /* 2 CDR recored length in bytes excluding CDR type */
+ short cdrsubtype; /* 1 CDR subtype */
+ unsigned int cdrid; /* 8 CDR unique number of each call */
+ unsigned int cdrtime; /* 4 CDR Time in seconds */
+ short cdrmillisec; /* 2 CDR Milliseconds */
+ short cdrstatus; /* 1 CDR For future use */
+ short cdrequipeid; /* 1 CDR Equipment id */
+ int cdrreserved1; /* 4 CDR For future use */
+
+ /* Defined or calculated for each record */
+ int cdrrestlen; /* Unprocessed data left in record in bytes */
+
+ /* Gemensamma datatyper */
+ unsigned short parmtype_prev; /* 1 Parameter type */
+ unsigned short parmtype; /* 1 Parameter type */
+ unsigned short parmlen; /* 1 Parameter type */
+
+ int rc; /* return code for functions */
+
+ /* Attribute object used with threads */
+ pthread_attr_t attr1;
+ pthread_attr_t attr2;
+ pthread_attr_t attr3;
+ struct cdr_record *tmpcdrptr,*ftest;
+ void *dat;
+
+ int error_from_client = 0;
+
+ /* Konstanter */
+ const int headerlen = 24; /* Length of header record */
+
+ parmtype_prev = 99;
+ reqcnt = 0;
+
+ /* Close the listen socket inherited from the daemon. */
+ close(ls);
+
+ printf("Use the readinfo program to get information about server status\n\n");
+
+ if((checkchangelog(fi,temp))==0)
+ c2log(fi,"Communication server started");
+
+ /* Look up the host information for the remote host */
+ /* that we have connected with. Its internet address */
+ /* was returned by the accept call, in the main */
+ /* daemon loop above. */
+
+ hp=gethostbyaddr((char *) &peeraddr_in.sin_addr,sizeof(struct in_addr),peeraddr_in.sin_family);
+
+ if (hp == NULL) {
+ /* The information is unavailable for the remote */
+ /* host. Just format its internet address to be */
+ /* printed out in the logging information. The */
+ /* address will be shown in "internet dot format". */
+
+ /*
+ hostname = inet_ntoa(peeraddr_in.sin_addr);
+ */
+ sprintf(hostname,"Test");
+ logname(lognamn,"Cdrserver","Child",hostname);
+ }
+ else {
+ hostname = hp->h_name; /* point to host's name */
+ logname(lognamn,"Cdrserver","Child",hostname);
+ }
+
+ log=fopen(lognamn,"w");
+ if (log == NULL)
+ {
+ perror(hostname);
+ exit(EXIT_FAILURE);
+ }
+ n2log(log,"Setup in progress");
+ /* Log a startup message. */
+
+ /* The port number must be converted first to host byte */
+ /* order before printing. On most hosts, this is not */
+ /* necessary, but the ntohs() call is included here so */
+ /* that this program could easily be ported to a host */
+ /* that does require it. */
+
+ snprintf(msg,sizeof(msg),"Startup from %s port %u",hostname,ntohs(peeraddr_in.sin_port));
+ if ((checkchangelog(fi,temp))==0)
+ c2log(fi,msg);
+ n2log(log,msg);
+ snprintf(msg,sizeof(msg),"For further information, see log(%s)",lognamn);
+ if ((checkchangelog(fi,temp))==0)
+ c2log(fi,msg);
+
+ /* Set the socket for a lingering, graceful close. */
+ /* This will cause a final close of this socket to wait until */
+ /* all * data sent on it has been received by the remote host. */
+
+ linger.l_onoff =1;
+ linger.l_linger =0;
+ if (setsockopt(s, SOL_SOCKET, SO_LINGER,(const char*)&linger,sizeof(linger)) == -1) {
+ snprintf(msg,sizeof(msg),"Setting SO_LINGER, l_onoff=%d, l_linger=%d",linger.l_onoff,linger.l_linger);
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ goto errout;
+ }
+
+ /* Set the socket for a lingering, graceful close. */
+ /* This will cause a final close of this socket to wait until all * data sent */
+ /* on it has been received by the remote host. */
+
+ rcvbuf_size=64*1024;
+
+ if (setsockopt(s, SOL_SOCKET, SO_RCVBUF,(const char*) &rcvbuf_size,sizeof(rcvbuf_size)) == -1) {
+ snprintf(msg,sizeof(msg),"Setting SO_RCVBUF = %d",rcvbuf_size);
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ goto errout;
+ }
+
+ /* Set nodelay on socket */
+ n2log(log,"Port setup complete");
+
+ /* Go into a loop, receiving requests from the remote */
+ /* client. After the client has sent the last request, */
+ /* it will do a shutdown for sending, which will cause */
+ /* an end-of-file condition to appear on this end of the */
+ /* connection. After all of the client's requests have */
+ /* been received, the next recv call will return zero */
+ /* bytes, signalling an end-of-file condition. This is */
+ /* how the server will know that no more requests will */
+ /* follow, and the loop will be exited. */
+
+ n2log(log,"Setup completed");
+
+ /* Fetch the process id for the server */
+
+ /* Inititate the variables used for counting transfer rates and rec/sec */
+ tmpvalue = 0;
+ tmptime = 0;
+ tmptransfer = 0;
+ transfer = 0;
+
+ printf("Client %s connected\nStarting to process the data\n\n",hostname);
+
+ tmpcdrptr = (struct cdr_record*)malloc(sizeof(struct cdr_record));
+
+ /***** NDB ******/
+ MyNdb.init();
+ if (MyNdb.waitUntilReady(30) != 0)
+ {
+ puts("Not ready");
+ exit(-1);
+ }
+ tTableId = MyNdb.getTable()->openTable(tableName);
+ if (tTableId == -1)
+ {
+ printf("%d: Creating table",getpid());
+ create_table(&MyNdb);
+ }
+ else printf("%d: Table already created",getpid());
+
+ /****************/
+
+ while (len = recv(s,buf,headerlen,MSG_WAITALL)) {
+ if (len == -1) {
+ snprintf(msg,sizeof(msg),"Error from recv");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ goto errout; /* error from recv */
+ }
+
+ /* The reason this while loop exists is that there */
+ /* is a remote possibility of the above recv returning */
+ /* less than 10 bytes. This is because a recv returns */
+ /* as soon as there is some data, and will not wait for */
+ /* all of the requested data to arrive. Since 10 bytes */
+ /* is relatively small compared to the allowed TCP */
+ /* packet sizes, a partial receive is unlikely. If */
+ /* this example had used 2048 bytes requests instead, */
+ /* a partial receive would be far more likely. */
+ /* This loop will keep receiving until all 10 bytes */
+ /* have been received, thus guaranteeing that the */
+ /* next recv at the top of the loop will start at */
+ /* the begining of the next request. */
+
+ for (;len < headerlen;) {
+ x = recv(s,buf,(headerlen-len),0);
+ if (x == -1) {
+ snprintf(msg,sizeof(msg),"Error from recv");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ goto errout; /* error from recv */
+ }
+ len=len+x;
+ }
+
+ if (ops == 0) {
+ MyTransaction = MyNdb.startTransaction();
+ if (MyTransaction == NULL)
+ error_handler(MyNdb.getNdbErrorString());
+ }//if
+
+ MyOperation = MyTransaction->getNdbOperation(tableName);
+ if (MyOperation == NULL)
+ error_handler(MyTransaction->getNdbErrorString());
+ /*------------------------------------------------------*/
+ /* Parse header of CDR records */
+ /*------------------------------------------------------*/
+
+ /*------------------------------------------------------*/
+ /* 1. Type of cdr */
+ /*------------------------------------------------------*/
+ /* Not used for the moment
+ cdrtype=(char)buf[0];
+ */
+ /*------------------------------------------------------*/
+ /* 2. Total length of CDR */
+ /*------------------------------------------------------*/
+ swab(buf+1,buf+1,2);
+ memcpy(&cdrlen,buf+1,2);
+ /*------------------------------------------------------*/
+ /* 3. Partial type of CDR */
+ /*------------------------------------------------------*/
+ cdrsubtype=(char)buf[3];
+ switch (cdrsubtype)
+ {
+ case 0:
+ c1++;
+ tmpcdrptr->CallAttemptState = 1;
+ check = MyOperation->insertTuple();
+ break;
+ case 1:
+ c2++;
+ tmpcdrptr->CallAttemptState = 2;
+ check = MyOperation->updateTuple();
+ break;
+ case 2:
+ c3++;
+ tmpcdrptr->CallAttemptState = 3;
+ check = MyOperation->deleteTuple();
+ break;
+ case 3:
+ c4++;
+ tmpcdrptr->CallAttemptState = 4;
+ check = MyOperation->deleteTuple();
+ break;
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ }
+ /*cdrsubtype=(cdrsubtype << 24) >> 24;*/
+ /*------------------------------------------------------*/
+ /* 4. ID number */
+ /*------------------------------------------------------*/
+ /*swab(buf+4,buf+4,4);*/ /* ABCD -> BADC */
+ /*
+ swab(buf+4,buf+4,4);
+ swab(buf+5,buf+5,2);
+ swab(buf+6,buf+6,2);
+ swab(buf+4,buf+4,2);
+ swab(buf+5,buf+5,2);
+ */
+ memcpy(&cdrid,buf+4,4);
+ tmpcdrptr->CallIdentificationNumber = cdrid;
+ #ifdef SETDBG
+ puts("CIN");
+ #endif
+ check = MyOperation->equal("CIN",(char*)&cdrid);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ #ifdef SETDBG
+ puts("CAS");
+ #endif
+
+ if (cdrsubtype < 2)
+ {
+ check = MyOperation->setValue("CAS",(char*)&cdrsubtype);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ }
+ /*------------------------------------------------------*/
+ /* 5. Time stamp */
+ /*------------------------------------------------------*/
+ swab(buf+12,buf+12,4);
+ swab(buf+13,buf+13,2);
+ swab(buf+14,buf+14,2);
+ swab(buf+12,buf+12,2);
+ swab(buf+13,buf+13,2);
+ memcpy(&cdrtime,buf+12,4);
+ switch (cdrsubtype)
+ {
+ case 0:
+ #ifdef SETDBG
+ puts("START_TIME");
+ #endif
+ check = MyOperation->setValue("START_TIME",(char*)&cdrtime);
+ break;
+ case 1:
+ #ifdef SETDBG
+ puts("Start1");
+ #endif
+ check = MyOperation->setValue("StartOfCharge",(char*)&cdrtime);
+ break;
+ case 2:
+ #ifdef SETDBG
+ puts("Start2");
+ #endif
+ /*
+ check = MyOperation->setValue("StopOfCharge",(char*)&cdrtime);
+ */
+ check = 0;
+ break;
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ }
+ /*------------------------------------------------------*/
+ /* 6. Milliseconds */
+ /*------------------------------------------------------*/
+ /* Not used by application
+ swab(buf+16,buf+16,2);
+ memcpy(&cdrmillisec,buf+16,2);
+ */
+ /*------------------------------------------------------*/
+ /* 7. CDR status reserverd for future use */
+ /*------------------------------------------------------*/
+ /* Not used by application
+ memcpy(&cdrstatus,buf+18,1);
+ */
+ /*------------------------------------------------------*/
+ /* 8. CDR equipe id, number of sending equipement */
+ /*------------------------------------------------------*/
+ /* Not used by application
+ memcpy(&cdrequipeid,buf+19,1);
+ */
+ /*cdrequipeid=(cdrequipeid << 24) >> 24;*/
+ /*------------------------------------------------------*/
+ /* 9. CDR reserverd for furter use */
+ /*------------------------------------------------------*/
+ /* Not used by applikation
+ swab(buf+20,buf+20,4);
+ swab(buf+21,buf+21,2);
+ swab(buf+22,buf+22,2);
+ swab(buf+20,buf+20,2);
+ swab(buf+21,buf+21,2);
+ memcpy(&cdrreserved1,buf+20,4);
+ */
+ /*------------------------------------------------------*/
+ /* calculate length of datapart in record */
+ /* Formula recordlength-headerlen-1 */
+ /*------------------------------------------------------*/
+ cdrrestlen=cdrlen-(headerlen-1);
+ /*------------------------------------------------------*/
+ /* Finished with header */
+ /*------------------------------------------------------*/
+ /* Read remaining cdr data into buffer for furter */
+ /* handling. */
+ /*------------------------------------------------------*/
+ len = recv(s,buf,cdrrestlen,MSG_WAITALL);
+ if (len == -1) {
+ snprintf(msg,sizeof(msg),"Error from recv");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ goto errout; /* error from recv */
+ }
+ for (;len<cdrrestlen;) {
+ x = recv(s,buf,len-cdrrestlen,0);
+ if (x == -1) {
+ snprintf(msg,sizeof(msg),"Error from recv");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ goto errout; /* error from recv */
+ }
+ len=len+x;
+ }
+ done=FALSE;
+
+ /* Count the transfer/sec */
+ tmptransfer += cdrlen;
+ if (cdrsubtype > 1)
+ {
+ #ifdef SETDBG
+ puts("Going to execute");
+ #endif
+ ops++;
+ if (ops == ops_before_exe) {
+ ops = 0;
+ check = MyTransaction->execute(Commit, CommitAsMuchAsPossible);
+ if ((check == -1) && (MyTransaction->getNdbError() != 0))
+ error_handler(MyTransaction->getNdbErrorString());
+ MyNdb.closeTransaction(MyTransaction);
+ #ifdef SETDBG
+ puts("Transaction closed");
+ #endif
+ }//if
+ reqcnt++;
+ continue;
+ }
+ for (x=0;x<=cdrrestlen && !done && cdrrestlen > 1;) {
+ uc=buf[x];
+ parmtype=uc;
+ /*parmtype=(parmtype << 24) >> 24;*/ /* Modified in sun worked in hp */
+
+ parmlen = buf[x+1];
+ /*parmlen =(parmlen << 24) >> 24;*/
+ x+=2;
+
+ switch (parmtype) {
+ case 4: /* Called party number */
+ bcd_decode2(parmlen,&buf[x],crap);
+ tmpcdrptr->BSubscriberNumberLength = (char)parmlen;
+ strcpy(tmpcdrptr->BSubscriberNumber,crap);
+ tmpcdrptr->BSubscriberNumber[parmlen] = '\0';
+ x=x+(parmlen/2);
+ if (parmlen % 2) x++;
+ tmpcdrptr->USED_FIELDS |= B_BSubscriberNumber;
+ #ifdef SETDBG
+ puts("BNumber");
+ #endif
+ check = MyOperation->setValue("BNumber",(char*)&tmpcdrptr->BSubscriberNumber);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 9: /* Calling Partys cataegory */
+ if (parmlen != 1) printf("ERROR: Calling partys category has wrong length %d\n",parmlen);
+ else tmpcdrptr->ACategory=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_ACategory;
+ #ifdef SETDBG
+ puts("ACategory");
+ #endif
+ check = MyOperation->setValue("ACategory",(char*)&tmpcdrptr->ACategory);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 10: /* Calling Party Number */
+ bcd_decode2(parmlen,&buf[x],crap);
+ tmpcdrptr->ASubscriberNumberLength = (char)parmlen;
+ strcpy(tmpcdrptr->ASubscriberNumber,crap);
+ tmpcdrptr->ASubscriberNumber[parmlen] = '\0';
+ x=x+(parmlen/2);
+ if (parmlen % 2) x++;
+ tmpcdrptr->USED_FIELDS |= B_ASubscriberNumber;
+ #ifdef SETDBG
+ puts("ANumber");
+ #endif
+ check = MyOperation->setValue("ANumber",(char*)&tmpcdrptr->ASubscriberNumber);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 11: /* Redirecting number */
+ bcd_decode2(parmlen,&buf[x],crap);
+ strcpy(tmpcdrptr->RedirectingNumber,crap);
+ x=x+(parmlen/2);
+ if (parmlen % 2) x++;
+ tmpcdrptr->USED_FIELDS |= B_RedirectingNumber;
+ #ifdef SETDBG
+ puts("RNumber");
+ #endif
+ check = MyOperation->setValue("RNumber",(char*)&tmpcdrptr->RedirectingNumber);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 17: /* Called partys category */
+ if (parmlen != 1) printf("ERROR: Called partys category has wrong length %d\n",parmlen);
+ else tmpcdrptr->EndOfSelectionInformation=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_EndOfSelectionInformation;
+ #ifdef SETDBG
+ puts("EndOfSelInf");
+ #endif
+ check = MyOperation->setValue("EndOfSelInf",(char*)&tmpcdrptr->EndOfSelectionInformation);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 18: /* Release reason */
+ if (parmlen != 1) printf("ERROR: Release reason has wrong length %d\n",parmlen);
+ else tmpcdrptr->CauseCode=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_CauseCode;
+ #ifdef SETDBG
+ puts("CauseCode");
+ #endif
+ check = MyOperation->setValue("CauseCode",(char*)&tmpcdrptr->CauseCode);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 19: /* Redirection information */
+ switch (parmlen) {
+ case 1:
+ tmpcdrptr->ReroutingIndicator= (char)buf[x];
+ tmpcdrptr->USED_FIELDS |= B_ReroutingIndicator;
+ break;
+ case 2:
+ swab(buf+x,buf+x,2);
+ tmpcdrptr->ReroutingIndicator= buf[x];
+ tmpcdrptr->USED_FIELDS |= B_ReroutingIndicator;
+ break;
+ default :
+ snprintf(msg,sizeof(msg),"ERROR: Redirection information has wrong length %d\n",parmlen);
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ break;
+ #ifdef SETDBG
+ puts("RI");
+ #endif
+ check = MyOperation->setValue("RI",(char*)&tmpcdrptr->ReroutingIndicator);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ }
+ x+=parmlen;
+ break;
+ case 32: /* User to user information */
+ if (parmlen != 1) printf("ERROR: User to User information has wrong length %d\n",parmlen);
+ else tmpcdrptr->UserToUserInformation=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_UserToUserInformation;
+ #ifdef SETDBG
+ puts("UserToUserInf");
+ #endif
+ check = MyOperation->setValue("UserToUserInf",(char*)&tmpcdrptr->UserToUserInformation);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 40: /* Original called number */
+ bcd_decode2(parmlen,&buf[x],crap);
+ strcpy(tmpcdrptr->OriginalCalledNumber,crap);
+ x=x+(parmlen/2);
+ if (parmlen % 2) x++;
+ tmpcdrptr->USED_FIELDS |= B_OriginalCalledNumber;
+ #ifdef SETDBG
+ puts("ONumber");
+ #endif
+ check = MyOperation->setValue("ONumber",(char*)&tmpcdrptr->OriginalCalledNumber);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 42: /* User to user indicator */
+ if (parmlen != 1) printf("ERROR: User to User indicator has wrong length %d\n",parmlen);
+ else tmpcdrptr->UserToUserIndicatior=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_UserToUserIndicatior;
+ #ifdef SETDBG
+ puts("UserToUserInd");
+ #endif
+ check = MyOperation->setValue("UserToUserInd",(char*)&tmpcdrptr->UserToUserIndicatior);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 63: /* Location number */
+ bcd_decode2(parmlen,&buf[x],crap);
+ strcpy(tmpcdrptr->LocationCode,crap);
+ x=x+(parmlen/2);
+ if (parmlen % 2) x++;
+ tmpcdrptr->USED_FIELDS |= B_LocationCode;
+ #ifdef SETDBG
+ puts("LocationCode");
+ #endif
+ check = MyOperation->setValue("LocationCode",(char*)&tmpcdrptr->LocationCode);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 240: /* Calling Partys cataegory */
+ if (parmlen != 1) printf("ERROR: Calling partys category has wrong length %d\n",parmlen);
+ else tmpcdrptr->NetworkIndicator=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_NetworkIndicator;
+ #ifdef SETDBG
+ puts("NIndicator");
+ #endif
+ check = MyOperation->setValue("NIndicator",(char*)&tmpcdrptr->NetworkIndicator);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 241: /* Calling Partys cataegory */
+ if (parmlen != 1) printf("ERROR: Calling partys category has wrong length %d\n",parmlen);
+ else tmpcdrptr->TonASubscriberNumber=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_TonASubscriberNumber;
+ #ifdef SETDBG
+ puts("TonANumber");
+ #endif
+ check = MyOperation->setValue("TonANumber",(char*)&tmpcdrptr->TonASubscriberNumber);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 242: /* Calling Partys cataegory */
+ if (parmlen != 1) printf("ERROR: Calling partys category has wrong length %d\n",parmlen);
+ else tmpcdrptr->TonBSubscriberNumber=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_TonBSubscriberNumber;
+ #ifdef SETDBG
+ puts("TonBNumber");
+ #endif
+ check = MyOperation->setValue("TonBNumber",(char*)&tmpcdrptr->TonBSubscriberNumber);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 243: /* Calling Partys cataegory */
+ if (parmlen != 1) printf("ERROR: Calling partys category has wrong length %d\n",parmlen);
+ else tmpcdrptr->TonRedirectingNumber=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_TonRedirectingNumber;
+ #ifdef SETDBG
+ puts("TonRNumber");
+ #endif
+ check = MyOperation->setValue("TonRNumber",(char*)&tmpcdrptr->TonRedirectingNumber);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 244: /* Calling Partys cataegory */
+ if (parmlen != 1) printf("ERROR: Calling partys category has wrong length %d\n",parmlen);
+ else tmpcdrptr->TonOriginalCalledNumber=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_TonOriginalCalledNumber;
+ #ifdef SETDBG
+ puts("TonONumber");
+ #endif
+ check = MyOperation->setValue("TonONumber",(char*)&tmpcdrptr->TonOriginalCalledNumber);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 245: /* Calling Partys cataegory */
+ if (parmlen != 1) printf("ERROR: Calling partys category has wrong length %d\n",parmlen);
+ else tmpcdrptr->TonLocationCode=(char)buf[x];
+ x+=parmlen;
+ tmpcdrptr->USED_FIELDS |= B_TonLocationCode;
+ #ifdef SETDBG
+ puts("TonLocationCode");
+ #endif
+ check = MyOperation->setValue("TonLocationCode",(char*)&tmpcdrptr->TonLocationCode);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 252: /* RINParameter Parameter */
+ switch (parmlen) {
+ case 1:
+ tmpcdrptr->RINParameter=buf[x];
+ tmpcdrptr->USED_FIELDS |= B_RINParameter;
+ break;
+ case 2:
+ swab(buf+x,buf+x,2);
+ tmpcdrptr->RINParameter = buf[x] << 8;
+ tmpcdrptr->USED_FIELDS |= B_RINParameter;
+ break;
+ default :
+ snprintf(msg,sizeof(msg),"ERROR: Rin parameter has wrong length %d\n",parmlen);
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ break;
+ }
+ x+=parmlen;
+ #ifdef SETDBG
+ puts("RINParameter");
+ #endif
+ check = MyOperation->setValue("RINParameter",(char*)&tmpcdrptr->RINParameter);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 253: /* OriginatingPointCode */
+ switch (parmlen) {
+ case 2:
+ swab(buf+x,buf+x,2);
+ memcpy(&tmpcdrptr->OriginatingPointCode,(buf+x),2);
+ tmpcdrptr->USED_FIELDS |= B_OriginatingPointCode;
+ break;
+ case 3:
+ swab(buf+x,buf+x,2);
+ swab(buf+(x+1),buf+(x+1),2);
+ swab(buf+x,buf+x,2);
+ memcpy(&tmpcdrptr->OriginatingPointCode,(buf+x),3);
+ tmpcdrptr->USED_FIELDS |= B_OriginatingPointCode;
+ break;
+ default :
+ snprintf(msg,sizeof(msg),"ERROR: OriginatingPointCode parameter has wrong length %d\n",parmlen);
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ break;
+ }
+ x+=parmlen;
+ #ifdef SETDBG
+ puts("OPC");
+ #endif
+ check = MyOperation->setValue("OPC",(char*)&tmpcdrptr->OriginatingPointCode);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 254: /* DestinationPointCode */
+ switch (parmlen) {
+ case 2:
+ swab(buf+x,buf+x,2);
+ memcpy(&tmpcdrptr->DestinationPointCode,(buf+x),2);
+ /*
+ tmpcdrptr->DestinationPointCode = buf[x] << 8;
+ */
+ tmpcdrptr->USED_FIELDS |= B_DestinationPointCode;
+ break;
+ case 3:
+ swab(buf+x,buf+x,2);
+ swab(buf+(x+1),buf+(x+1),2);
+ swab(buf+x,buf+x,2);
+ memcpy(&tmpcdrptr->DestinationPointCode,(buf+x),3);
+ tmpcdrptr->USED_FIELDS |= B_DestinationPointCode;
+ break;
+ default :
+ snprintf(msg,sizeof(msg),"ERROR: DestinationPointCode parameter has wrong length %d\n",parmlen);
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ break;
+ }
+ x+=parmlen;
+ #ifdef SETDBG
+ puts("DPC");
+ #endif
+ check = MyOperation->setValue("DPC",(char*)&tmpcdrptr->DestinationPointCode);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ case 255: /* CircuitIdentificationCode */
+ swab(buf+x,buf+x,2);
+ memcpy(&tmpcdrptr->CircuitIdentificationCode,(buf+x),2);
+ tmpcdrptr->USED_FIELDS |= B_CircuitIdentificationCode;
+ x+=parmlen;
+ #ifdef SETDBG
+ puts("CIC");
+ #endif
+ check = MyOperation->setValue("CIC",(char*)&tmpcdrptr->CircuitIdentificationCode);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+ default:
+ printf("ERROR: Undefined parmtype %d , previous %d, length %d\n",parmtype,parmtype_prev,parmlen);
+ snprintf(msg,sizeof(msg),"ERROR: Undefined parmtype %d , previous %d, length %d\n",parmtype,parmtype_prev,parmlen);
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ if (parmlen == 0) {
+ x++;
+ }
+ x+=parmlen;
+ break;
+ }
+ parmtype_prev=parmtype;
+ if ((cdrrestlen-x) == 1) {
+ done=TRUE;
+ }
+ }
+ time(&ourtime);
+ if (ourtime != tmptime)
+ {
+ transfer = tmptransfer;
+ tmptransfer = 0;
+ if (++act_index == 30)
+ {
+ act_index = 0;
+ printf("Transfer=%d\n",transfer);
+ printf("Total operations=%d\n",reqcnt);
+ printf("CAS1=%d\n",c1/30);
+ printf("CAS2=%d\n",c2/30);
+ printf("CAS3=%d\n",c3/30);
+ c1=0;
+ c2=0;
+ c3=0;
+ }
+ tmptime = ourtime;
+ }
+ switch (cdrsubtype) {
+ case 0:
+ tmpcdrptr->ClientId = servernum;
+ #ifdef SETDBG
+ puts("ClientId");
+ #endif
+ check = MyOperation->setValue("ClientId",(char*)&tmpcdrptr->ClientId);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ tmpcdrptr->OurSTART_TIME = ourtime;
+ #ifdef SETDBG
+ puts("OurSTART_TIME");
+ #endif
+ check = MyOperation->setValue("OurSTART_TIME",(char*)&tmpcdrptr->OurSTART_TIME);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ tmpcdrptr->USED_FIELDS |= B_START_TIME;
+ #ifdef SETDBG
+ puts("USED_FIELDS");
+ #endif
+ check = MyOperation->setValue("USED_FIELDS",(char*)&tmpcdrptr->USED_FIELDS);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+
+ case 1:
+ tmpcdrptr->OurTimeForStartOfCharge = ourtime;
+ #ifdef SETDBG
+ puts("OurStartOfCharge");
+ #endif
+ check = MyOperation->setValue("OurStartOfCharge",(char*)&tmpcdrptr->OurTimeForStartOfCharge);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ tmpcdrptr->USED_FIELDS |= B_TimeForStartOfCharge;
+ #ifdef SETDBG
+ puts("USED_FIELDS");
+ #endif
+ check = MyOperation->setValue("USED_FIELDS",(char*)&tmpcdrptr->USED_FIELDS);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+
+ case 2:
+ tmpcdrptr->OurTimeForStopOfCharge = ourtime;
+ #ifdef SETDBG
+ puts("OurStopOfCharge");
+ #endif
+ check = MyOperation->setValue("OurStopOfCharge",(char*)&tmpcdrptr->OurTimeForStopOfCharge);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ tmpcdrptr->USED_FIELDS |= B_TimeForStopOfCharge;
+ #ifdef SETDBG
+ puts("USED_FIELDS");
+ #endif
+ check = MyOperation->setValue("USED_FIELDS",(char*)&tmpcdrptr->USED_FIELDS);
+ if (check == -1)
+ error_handler(MyTransaction->getNdbErrorString());
+ break;
+
+ case 3:
+ tmpcdrptr->CallAttemptState = 4;
+ break;
+ default:
+ snprintf(msg,sizeof(msg),"cdrtype %d unknown",cdrsubtype);
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ goto errout;
+ break;
+ }
+ ops++;
+ if (ops == ops_before_exe) {
+ ops = 0;
+ #ifdef SETDBG
+ puts("Going to execute");
+ #endif
+ check = MyTransaction->execute(Commit, CommitAsMuchAsPossible);
+ if ((check == -1) && (MyTransaction->getNdbError() != 0))
+ error_handler(MyTransaction->getNdbErrorString());
+ MyNdb.closeTransaction(MyTransaction);
+ #ifdef SETDBG
+ puts("Transaction closed");
+ #endif
+
+ #ifdef SETDBG
+ puts("New transaction initiated");
+ #endif
+ }//if
+ /* Increment the request count. */
+ reqcnt++;
+
+ /* Send a response back to the client. */
+
+ /* if (send(s, buf, 10, 0) != 10) goto errout; */
+ }
+
+ /* The loop has terminated, because there are no */
+ /* more requests to be serviced. As mentioned above, */
+ /* this close will block until all of the sent replies */
+ /* have been received by the remote host. The reason */
+ /* for lingering on the close is so that the server will */
+ /* have a better idea of when the remote has picked up */
+ /* all of the data. This will allow the start and finish */
+ /* times printed in the log file to reflect more accurately */
+ /* the length of time this connection was */
+ /* The port number must be converted first to host byte */
+ /* order before printing. On most hosts, this is not */
+ /* necessary, but the ntohs() call is included here so */
+ /* that this program could easily be ported to a host */
+ /* that does require it. */
+
+ snprintf(msg,sizeof(msg),"Completed %s port %u, %d requests",hostname,ntohs(peeraddr_in.sin_port), reqcnt);
+ if ((checkchangelog(fi,temp))==0)
+ c2log(fi,msg);
+ error_from_client = 1;
+ snprintf(msg,sizeof(msg),"Communicate with threads");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ snprintf(msg,sizeof(msg),"Waiting for threads to return from work");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ snprintf(msg,sizeof(msg),"Closing down");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ close(s);
+ fclose(log);
+ return EXIT_SUCCESS;
+
+errout:
+ snprintf(msg,sizeof(msg),"Connection with %s aborted on error\n", hostname);
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ if ((checkchangelog(fi,temp))==0)
+ c2log(fi,msg);
+ error_from_client = 1;
+ snprintf(msg,sizeof(msg),"Communicate with threads");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ snprintf(msg,sizeof(msg),"Waiting for threads to return from work");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ snprintf(msg,sizeof(msg),"Closing down");
+ if ((checkchangelog(log,lognamn))==0)
+ n2log(log,msg);
+ close(s);
+ fclose(log);
+ return EXIT_FAILURE;
+}
+
+void
+create_table(Ndb* pMyNdb)
+{
+
+ /****************************************************************
+ * Create table and attributes.
+ *
+ * create table basictab1(
+ * col1 int,
+ * col2 int not null,
+ * col3 int not null,
+ * col4 int not null
+ * )
+ *
+ ***************************************************************/
+
+ int check;
+ int i;
+ NdbSchemaCon *MySchemaTransaction;
+ NdbSchemaOp *MySchemaOp;
+ int tAttributeSize;
+
+ tAttributeSize = 1;
+
+ cout << "Creating " << tableName << "..." << endl;
+
+ MySchemaTransaction = pMyNdb->startSchemaTransaction();
+ if( MySchemaTransaction == NULL )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ if( MySchemaOp == NULL )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // Createtable
+ check = MySchemaOp->createTable( tableName,
+ 8, // Table Size
+ TupleKey, // Key Type
+ 40 // Nr of Pages
+ );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // CallIdentificationNumber Create first column, primary key
+ check = MySchemaOp->createAttribute( "CIN",
+ TupleKey,
+ 32,
+ tAttributeSize,
+ UnSigned, MMBased,
+ NotNullAttribute
+ );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+
+ // USED_FIELDS Create attributes
+ check = MySchemaOp->createAttribute( "USED_FIELDS", NoKey, 32,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // ClientId Create attributes
+ check = MySchemaOp->createAttribute( "ClientId", NoKey, 32,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // START_TIME Create attributes
+ check = MySchemaOp->createAttribute( "START_TIME", NoKey, 32,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // OurSTART_TIME Create attributes
+ check = MySchemaOp->createAttribute( "OurSTART_TIME", NoKey, 32,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // TimeForStartOfCharge Create attributes
+ check = MySchemaOp->createAttribute( "StartOfCharge", NoKey, 32,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // TimeForStopOfCharge Create attributes
+ check = MySchemaOp->createAttribute( "StopOfCharge", NoKey, 32,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // OurTimeForStartOfCharge Create attributes
+ check = MySchemaOp->createAttribute( "OurStartOfCharge", NoKey, 32,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // OurTimeForStopOfCharge Create attributes
+ check = MySchemaOp->createAttribute( "OurStopOfCharge", NoKey, 32,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // DestinationPointCode Create attributes
+ check = MySchemaOp->createAttribute( "DPC", NoKey, 16,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // OriginatingPointCode Create attributes
+ check = MySchemaOp->createAttribute( "OPC", NoKey, 16,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // CircuitIdentificationCode Create attributes
+ check = MySchemaOp->createAttribute( "CIC", NoKey, 16,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // ReroutingIndicator Create attributes
+ check = MySchemaOp->createAttribute( "RI", NoKey, 16,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // RINParameter Create attributes
+ check = MySchemaOp->createAttribute( "RINParameter", NoKey, 16,
+ tAttributeSize, UnSigned, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // NetworkIndicator Create attributes
+ check = MySchemaOp->createAttribute( "NIndicator", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // CallAttemptState Create attributes
+ check = MySchemaOp->createAttribute( "CAS", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // ACategory Create attributes
+ check = MySchemaOp->createAttribute( "ACategory", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // EndOfSelectionInformation Create attributes
+ check = MySchemaOp->createAttribute( "EndOfSelInf", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // UserToUserInformation Create attributes
+ check = MySchemaOp->createAttribute( "UserToUserInf", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // UserToUserIndicator Create attributes
+ check = MySchemaOp->createAttribute( "UserToUserInd", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // CauseCode Create attributes
+ check = MySchemaOp->createAttribute( "CauseCode", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // ASubscriberNumber attributes
+ check = MySchemaOp->createAttribute( "ANumber", NoKey, 8,
+ ASubscriberNumber_SIZE, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // ASubscriberNumberLenght attributes
+ check = MySchemaOp->createAttribute( "ANumberLength", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // TonASubscriberNumber attributes
+ check = MySchemaOp->createAttribute( "TonANumber", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // BSubscriberNumber attributes
+ check = MySchemaOp->createAttribute( "BNumber", NoKey, 8,
+ BSubscriberNumber_SIZE, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // BSubscriberNumberLength attributes
+ check = MySchemaOp->createAttribute( "BNumberLength", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // TonBSubscriberNumber attributes
+ check = MySchemaOp->createAttribute( "TonBNumber", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // RedirectingNumber attributes
+ check = MySchemaOp->createAttribute( "RNumber", NoKey, 8,
+ ASubscriberNumber_SIZE, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // TonRedirectingNumber attributes
+ check = MySchemaOp->createAttribute( "TonRNumber", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // OriginalCalledNumber attributes
+ check = MySchemaOp->createAttribute( "ONumber", NoKey, 8,
+ ASubscriberNumber_SIZE, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // TonOriginalCalledNumber attributes
+ check = MySchemaOp->createAttribute( "TonONumber", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // LocationCode attributes
+ check = MySchemaOp->createAttribute( "LocationCode", NoKey, 8,
+ ASubscriberNumber_SIZE, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ // TonLocationCode attributes
+ check = MySchemaOp->createAttribute( "TonLocationCode", NoKey, 8,
+ tAttributeSize, Signed, MMBased,
+ NullAttribute );
+ if( check == -1 )
+ error_handler(MySchemaTransaction->getNdbErrorString());
+
+ if( MySchemaTransaction->execute() == -1 ) {
+ cout << tableName << " already exist" << endl;
+ cout << "Message: " << MySchemaTransaction->getNdbErrorString() << endl;
+ }
+ else
+ {
+ cout << tableName << " created" << endl;
+ }
+ pMyNdb->closeSchemaTransaction(MySchemaTransaction);
+
+ return;
+}
+
+void
+error_handler(const char* errorText)
+{
+ // Test failed
+ cout << endl << "ErrorMessage: " << errorText << endl;
+ bTestPassed = -1;
+}
diff --git a/ndb/test/ndbapi/vw_test/script/client_start b/ndb/test/ndbapi/vw_test/script/client_start
new file mode 100644
index 00000000000..2965be6fbb5
--- /dev/null
+++ b/ndb/test/ndbapi/vw_test/script/client_start
@@ -0,0 +1,10 @@
+# Argument to the client program is:
+# 1. ip-adress to the server
+# 2. location to the raw cdr-file
+# 3. nanoseconds (0-1000) between writing the buffer to the port
+# 4. how many writes to the buffer before the sleep command should accur
+# Argument 3 and 4 controlls the flow of the raw cdr-file to the cdrserver
+
+cd $VCDRPATH/bin
+# ./client stat181.xxx.com /ext06/data/indata_fraud1/port2file.data.-2089540139 1000 0 &
+./client xxx.xxx.xxx.xxx.xxx /export2/home/ndb/vw/data/port2file_data_-2089540139 0 100000 &
diff --git a/ndb/test/ndbapi/vw_test/size.cpp b/ndb/test/ndbapi/vw_test/size.cpp
new file mode 100644
index 00000000000..397cc02f4f6
--- /dev/null
+++ b/ndb/test/ndbapi/vw_test/size.cpp
@@ -0,0 +1,27 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include "utv.h"
+
+int main(void)
+{
+ printf("cdrstruct=%d\n",sizeof(struct cdr_record));
+ printf("long int=%d\n",sizeof(long int));
+ printf("int=%d\n",sizeof(int));
+ printf("short int=%d\n",sizeof(short int));
+ return 0;
+}
diff --git a/ndb/test/ndbapi/vw_test/utv.h b/ndb/test/ndbapi/vw_test/utv.h
new file mode 100644
index 00000000000..6f378e5595b
--- /dev/null
+++ b/ndb/test/ndbapi/vw_test/utv.h
@@ -0,0 +1,161 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <semaphore.h>
+#include <thread.h>
+#include <limits.h>
+
+#define TESTLEV
+
+#define ASubscriberNumber_SIZE 16
+#define BSubscriberNumber_SIZE 29
+#define TRUE 1
+#define FALSE 0
+#define WRITE_LIMIT 100000
+#define EVER ;;
+#define CONNINFO "/"
+#define FILE_MODE (S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH)
+
+#define BIT_1 0x1
+#define BIT_2 0x2
+#define BIT_3 0x4
+#define BIT_4 0x8
+#define BIT_5 0x10
+#define BIT_6 0x20
+#define BIT_7 0x40
+#define BIT_8 0x80
+
+/*------------------------------------------------------*/
+/* record defines structure over an alarm thresholds */
+/* CallAttemptState Beskriver status på samtal */
+/* 0 - Subscriber is calling */
+/* 1 - Called part answer call */
+/* 2 - Release of call */
+/* 3-255 reserved for furter use */
+/* USED_FILEDS Indicates active fields within call */
+/* bit 1 - START_TIME */
+/* 2 - TimeForStartOfCharge */
+/* 3 - TimeForStopOfCharge */
+/* 4 - ReroutingIndicator */
+/* 5 - RINParameter */
+/* 6 - ACategory */
+/* 7 - EndOfSelectionInformation */
+/* 8 - UserToUserIndicatior */
+/* 9 - UserToUserInformation */
+/* 10 - CauseCode */
+/* 11 - ASubscriberNumber */
+/* 12 - BSubscriberNumber */
+/* 13 - RedirectingNumber */
+/* 14 - OriginalCalledNumber */
+/* 15 - LocationCode */
+/* 16 - OriginatingPointCode */
+/* 17 - DestinationPointCode */
+/* 18 - CircuitIdentificationCode */
+/* 19 - NetworkIndicator */
+/*------------------------------------------------------*/
+
+struct cdr_record
+{
+ unsigned int USED_FIELDS;
+ unsigned long ClientId;
+ unsigned int CallIdentificationNumber;
+ unsigned int START_TIME;
+ unsigned int OurSTART_TIME;
+ unsigned int TimeForStartOfCharge;
+ unsigned int TimeForStopOfCharge;
+ time_t OurTimeForStartOfCharge;
+ time_t OurTimeForStopOfCharge;
+ unsigned short DestinationPointCode;
+ unsigned short CircuitIdentificationCode;
+ unsigned short OriginatingPointCode;
+ unsigned short ReroutingIndicator;
+ unsigned short RINParameter;
+ char NetworkIndicator;
+ char CallAttemptState;
+ char ACategory;
+ char EndOfSelectionInformation;
+ char UserToUserInformation;
+ char UserToUserIndicatior;
+ char CauseCode;
+ char ASubscriberNumber[ASubscriberNumber_SIZE];
+ char ASubscriberNumberLength;
+ char TonASubscriberNumber;
+ char BSubscriberNumber[BSubscriberNumber_SIZE];
+ char BSubscriberNumberLength;
+ char TonBSubscriberNumber;
+ char RedirectingNumber[16];
+ char TonRedirectingNumber;
+ char OriginalCalledNumber[16];
+ char TonOriginalCalledNumber;
+ char LocationCode[16];
+ char TonLocationCode;
+};
+
+/*------------------------------------------------------*/
+/* Define switches for each tag */
+/*------------------------------------------------------*/
+
+#define B_START_TIME 0x1
+#define B_TimeForStartOfCharge 0x2
+#define B_TimeForStopOfCharge 0x4
+#define B_ReroutingIndicator 0x8
+#define B_RINParameter 0x10
+#define B_ACategory 0x20
+#define B_EndOfSelectionInformation 0x40
+#define B_UserToUserIndicatior 0x80
+#define B_UserToUserInformation 0x100
+#define B_CauseCode 0x200
+#define B_ASubscriberNumber 0x400
+#define B_BSubscriberNumber 0x800
+#define B_RedirectingNumber 0x1000
+#define B_OriginalCalledNumber 0x2000
+#define B_LocationCode 0x4000
+#define B_OriginatingPointCode 0x8000
+#define B_DestinationPointCode 0x10000
+#define B_CircuitIdentificationCode 0x20000
+
+#define B_NetworkIndicator 0x40000
+#define B_TonASubscriberNumber 0x80000
+#define B_TonBSubscriberNumber 0x100000
+#define B_TonRedirectingNumber 0x200000
+#define B_TonOriginalCalledNumber 0x4000000
+#define B_TonLocationCode 0x8000000
+
+#define K_START_TIME 0xFF01
+#define K_TimeForStartOfCharge 0xFF02
+#define K_TimeForStopOfCharge 0xFF03
+#define K_ReroutingIndicator 0x13
+#define K_RINParameter 0xFC
+#define K_ACategory 0x09
+#define K_EndOfSelectionInformation 0x11
+#define K_UserToUserIndicatior 0x2A
+#define K_UserToUserInformation 0x20
+#define K_CauseCode 0x12
+#define K_ASubscriberNumber 0x0A
+#define K_BSubscriberNumber 0x04
+#define K_RedirectingNumber 0x0B
+#define K_OriginalCalledNumber 0x28
+#define K_LocationCode 0x3F
+#define K_OriginatingPointCode 0xFD
+#define K_DestinationPointCode 0xFE
+#define K_CircuitIdentificationCode 0xFF
+
+#define K_NetworkIndicator 0xF0
+#define K_TonASubscriberNumber 0xF1
+#define K_TonBSubscriberNumber 0xF2
+#define K_TonRedirectingNumber 0xF3
+#define K_TonOriginalCalledNumber 0xF4
+#define K_TonLocationCode 0xF5
diff --git a/ndb/test/ndbapi/vw_test/vcdrfunc.h b/ndb/test/ndbapi/vw_test/vcdrfunc.h
new file mode 100644
index 00000000000..3c5444d733b
--- /dev/null
+++ b/ndb/test/ndbapi/vw_test/vcdrfunc.h
@@ -0,0 +1,55 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/********************************************************/
+/* Common functions */
+/* unix_ps checks if a process is running with a */
+/* name and pid rc 0=not running */
+/* 1=Running */
+/* logname create a log filename */
+/* Parm */
+/* 1 lvl1 name */
+/* 2 lvl2 name */
+/* 3 lvl3 name */
+/* m2log Skriv log rader som moder */
+/* Parm */
+/* 1 pointer to filehandler */
+/* 2 Log text max 600 tecken */
+/* c2log Skriv log rader som barn */
+/* Parm */
+/* 1 pointer to filehandler */
+/* 2 Log text max 600 tecken */
+/* n2log Skriv log rader utan relation */
+/* Parm */
+/* 1 pointer to filehandler */
+/* 2 Log text max 600 tecken */
+/********************************************************/
+
+int n2log(FILE *fi,char *text);
+int m2log(FILE *fi,char *text);
+int c2log(FILE *fi,char *text);
+int checkchangelog(FILE* fp,char *filename);
+void logname(char *filename, char *lvl1, char *lvl2, char *lvl3);
+void logname_unique_day(char *filename, char *lvl1, char *lvl2, char *lvl3);
+int unix_ps(char *proc_name,char *pid);
+/*
+int unix_ps2(char *proc_name,char *pid);
+*/
+int unix_ps3(char *proc_name);
+int replacetoken(const char* instring,char token,char replace);
+int CompAsciiNum(char *, char *);
+int CompareIt(char *,char *);
+int CompCdrNum(const void *,const void *,void *);
diff --git a/ndb/test/ndbnet/test.run b/ndb/test/ndbnet/test.run
new file mode 100644
index 00000000000..30042488c92
--- /dev/null
+++ b/ndb/test/ndbnet/test.run
@@ -0,0 +1,3 @@
+#
+
+system("printenv|sort");
diff --git a/ndb/test/ndbnet/testError.run b/ndb/test/ndbnet/testError.run
new file mode 100644
index 00000000000..3cce489a3da
--- /dev/null
+++ b/ndb/test/ndbnet/testError.run
@@ -0,0 +1,266 @@
+#
+# file : test/ndbnet/testError.run
+# usage: perl testError.run
+#
+# you need to have $NDB_TOP/lib/perl5 on search path $PERL5LIB
+# or else write perl -I$NDB_TOP/lib/perl5 test1.run
+#
+# The database is specified by the $NDB_DATABASE environment variable
+#
+# method names and argument style will change slightly.
+#
+#
+
+use strict;
+use NDB::Run;
+
+my $env = NDB::Run->getenv;
+my $log = $env->getlog;
+$log->setpart(time => 1, line => 0);
+$log->setprio("info");
+
+my $database = $ENV{NDB_DATABASE};
+my $api_cmd = $ENV{API_CMD};
+$log->put("start test database=$database");
+$env->init or $log->push("init failed")->fatal;
+
+my $db = $env->getdb($database) or $log->push->fatal;
+my $mgm = $db->getnode(1) or $log->push->fatal;
+my $api = $db->getnode(4) or $log->push->fatal;
+
+my @dbnode = (); # array of db nodes indexed 2..3
+for my $i (2..3) {
+ $dbnode[$i] = $db->getnode($i) or $log->push->fatal;
+}
+
+# list of db nodes and errors to insert
+my @errors = ( # array of array refs
+ [ 2, 9998 ],
+ [ 2, 37017 ],
+ [ 2, 37018 ],
+ [ 2, 37017 ],
+ [ 2, 37018 ],
+ [ 2, 37017 ],
+ [ 2, 37018 ],
+ [ 2, 37017 ],
+ [ 2, 37018 ],
+ [ 2, 37017 ],
+ [ 2, 37018 ],
+ [ 2, 37017 ],
+ [ 2, 9998 ],
+ [ 3, 9998 ],
+ [ 3, 9998 ],
+ [ 2, 9998 ],
+ [ 3, 9998 ],
+ [ 3, 9998 ],
+ [ 2, 38002 ],
+ [ 2, 38002 ],
+ [ 2, 8002 ],
+ [ 3, 8029 ],
+ [ 2, 8030 ],
+ [ 2, 8031 ],
+ [ 3, 8020 ],
+ [ 2, 8021 ],
+ [ 3, 8022 ],
+ [ 2, 8023 ],
+ [ 3, 8025 ],
+ [ 2, 8027 ],
+ [ 2, 38002 ],
+ [ 3, 38029 ],
+ [ 2, 38030 ],
+ [ 2, 38031 ],
+ [ 3, 38020 ],
+ [ 2, 38021 ],
+ [ 3, 38022 ],
+ [ 2, 38023 ],
+ [ 3, 38025 ],
+ [ 2, 38027 ],
+ [ 2, 48002 ],
+ [ 3, 48029 ],
+ [ 2, 48030 ],
+ [ 2, 48031 ],
+ [ 3, 48020 ],
+ [ 2, 48021 ],
+ [ 3, 48022 ],
+ [ 2, 48023 ],
+ [ 3, 48025 ],
+ [ 2, 48027 ],
+ [ 2, 9999 ],
+ [ 3, 9999 ],
+ [ 3, 9999 ],
+ [ 2, 9998 ],
+ [ 3, 9998 ],
+ [ 3, 9998 ],
+ [ 2, 9998 ],
+ [ 3, 9998 ],
+ [ 3, 9998 ],
+ [ 3, 37000 ],
+ [ 2, 37001 ],
+ [ 2, 37002 ],
+ [ 2, 37003 ],
+ [ 2, 47005 ],
+ [ 2, 47006 ],
+ [ 2, 47007 ],
+ [ 2, 47008 ],
+ [ 2, 45000 ],
+ [ 2, 37005 ],
+ [ 2, 37006 ],
+ [ 2, 37007 ],
+ [ 2, 37008 ],
+ [ 2, 35000 ],
+ [ 2, 37009 ],
+ [ 2, 37010 ],
+ [ 2, 37013 ],
+ [ 2, 37014 ],
+ [ 2, 37015 ],
+ [ 2, 37016 ],
+ [ 2, 37017 ],
+ [ 2, 37018 ],
+ [ 2, 37019 ],
+ [ 2, 47020 ],
+ [ 2, 37020 ],
+ [ 2, 48000 ],
+ [ 2, 38000 ],
+ [ 2, 48001 ],
+ [ 2, 38001 ],
+ [ 2, 45001 ],
+ [ 2, 35001 ],
+);
+
+$db->kill;
+$db->start({init_rm=>1}) or $log->push->fatal;
+sleep 10;
+
+# should be option (or default) in $db->start
+sub wait_until_started {
+ my $local_cnt = 60;
+ while (--$local_cnt > 0) {
+ sleep 2;
+ my $ret = $mgm->write("all status", { wait => 3 });
+ $ret or $log->fatal;
+ my $output = $ret->{output};
+ if ($output =~ /\bstarted\b(.|\n)*started\b/i) {
+ $log->put("*** db is started ***")->info;
+ return;
+ }
+ if ($output =~ /\bno.contact\b(.|\n)*no.contact\b/i) {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** db is dead ***")->fatal;
+ }
+ }
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** node recovery failed ***")->fatal;
+}
+
+sub getdbstatus {
+ my $ret = $mgm->write("all status", { wait => 3 });
+ $ret or return undef;
+ my $output = $ret->{output};
+ my @status = (); # indexed 2..3
+ for my $i (2..3) {
+ my $s = "Unknown";
+ if ($output =~ /Node\s*$i\s*:\s*(\w+)/i) {
+ $s = $1;
+ }#if
+ $status[$i] = lc $s;
+ }#for
+ return \@status;
+}
+
+# count elapsed time
+my $lasttime;
+sub settime { $lasttime = time };
+sub gettime { return time - $lasttime };
+
+wait_until_started();
+$api->start({run=>$api_cmd});
+sleep 10;
+
+# loop over error inserts
+for my $e (@errors) {
+for my $loop (1..7) {
+ my $i = $e->[0]; # db node number 2..3
+ my $c = $e->[1]; # error code
+ my $dead_node_id = $i;
+ my $dbnode = $dbnode[$i];
+ my $two = 2;
+ my $three = 3;
+ my $kill_no = 9998;
+
+ $log->put("insert error $c")->push($dbnode)->notice;
+
+ # insert error
+ if ($c eq $kill_no) {
+ $dbnode->kill
+ or $log->put("Kill 1 failed")->fatal;
+ } else {
+ $mgm->write("$i error $c")
+ or $log->put("insert error fault")->fatal;
+ }#if
+
+ # after a few seconds check that node is dead
+ settime();
+ loop: {
+ gettime() <= 300
+ or $log->put("db node $i refuses to die")->fatal;
+ my $status = getdbstatus()
+ or $log->put("getdbstatus error")->fatal;
+
+ if (($status->[$two] eq 'no') && ($status->[$three] eq 'no')) {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** db is dead ***")->fatal;
+ }#if
+ if ($c < 10000) {
+ if ($status->[$i] ne 'no') { # ...contact
+ sleep 2;
+ redo loop;
+ }#if
+ $dead_node_id = $i;
+ } else {
+ if (($status->[$two] eq 'no') &&
+ ($status->[$three] eq 'started')) {
+ $dead_node_id = 2;
+ } else {
+ if (($status->[$three] eq 'no') &&
+ ($status->[$two] eq 'started')) {
+ $dead_node_id = 3;
+ } else {
+ sleep 2;
+ redo loop;
+ }#if
+ }#if
+ }#if
+ }#loop
+
+ my $dead_node = $dbnode[$dead_node_id];
+ # have to even check the process is gone
+ sleep 5;
+ if ($dead_node->stat ne "down") {
+ $log->put("ndb did not die, kill it")->push($dead_node)->warn;
+ $dead_node->kill
+ or $log->put("Kill 2 failed")->fatal;
+ }#if
+
+ $log->put("node $dead_node_id is dead")->notice;
+
+ # start the failed node
+ $dead_node->start
+ or $log->put("Start ndb node failed")->fatal;
+
+ wait_until_started();
+ $log->put("node $dead_node_id is up again")->notice;
+
+ # check test pgm is running
+ my $stat = $api->stat
+ or $log->put("api->stat failed")->fatal;
+ if ($stat ne "up") {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $db->kill;
+ $log->put("flexBench has crashed")->fatal;
+ }#if
+}#for
+}#for
+print "NDBT_ProgramExit: 0 - test ready\n";
+$db->kill;
+
+# vim: set sw=4:
diff --git a/ndb/test/ndbnet/testMNF.run b/ndb/test/ndbnet/testMNF.run
new file mode 100644
index 00000000000..df226cd3359
--- /dev/null
+++ b/ndb/test/ndbnet/testMNF.run
@@ -0,0 +1,277 @@
+#
+# file : test/ndbnet/testError.run
+# usage: perl testError.run
+#
+# you need to have $NDB_TOP/lib/perl5 on search path $PERL5LIB
+# or else write perl -I$NDB_TOP/lib/perl5 test1.run
+#
+# The database is specified by the $NDB_DATABASE environment variable
+#
+# method names and argument style will change slightly.
+#
+#
+
+use strict;
+use NDB::Run;
+
+my $env = NDB::Run->getenv;
+my $log = $env->getlog;
+$log->setpart(time => 1, line => 0);
+$log->setprio("info");
+
+my $database = $ENV{NDB_DATABASE};
+my $api_cmd = $ENV{API_CMD};
+my $api_cmd2 = $ENV{API_CMD2};
+$log->put("start test database=$database");
+$env->init or $log->push("init failed")->fatal;
+
+my $db = $env->getdb($database) or $log->push->fatal;
+my $mgm = $db->getnode(1) or $log->push->fatal;
+my $api = $db->getnode(6) or $log->push->fatal;
+my $api2 = $db->getnode(7) or $log->push->fatal;
+
+my @dbnode = (); # array of db nodes indexed 2..5
+for my $i (2..5) {
+ $dbnode[$i] = $db->getnode($i) or $log->push->fatal;
+}
+
+# list of db nodes and errors to insert
+my @errors = ( # array of array refs
+ [ 3, 4, 8030, 8031 ],
+ [ 2, 5, 8027, 8030 ],
+ [ 2, 4, 5000, 5000 ],
+ [ 2, 5, 5000, 5000 ],
+ [ 3, 4, 5000, 7008 ],
+ [ 3, 5, 5000, 7008 ],
+ [ 2, 4, 5000, 7007 ],
+ [ 3, 4, 5000, 7007 ],
+ [ 3, 5, 5000, 7006 ],
+ [ 2, 5, 5000, 7006 ],
+ [ 2, 4, 5000, 7005 ],
+ [ 3, 5, 5000, 7005 ],
+ [ 3, 4, 7005, 7005 ],
+ [ 2, 5, 7005, 7005 ],
+ [ 3, 4, 7005, 7007 ],
+ [ 3, 5, 7005, 7007 ],
+ [ 2, 5, 7005, 7008 ],
+ [ 2, 4, 7005, 7008 ],
+ [ 3, 5, 7006, 7006 ],
+ [ 3, 4, 7006, 7006 ],
+ [ 2, 4, 7006, 7007 ],
+ [ 2, 5, 7006, 7007 ],
+ [ 2, 5, 7006, 7008 ],
+ [ 2, 4, 7006, 7008 ],
+ [ 3, 4, 7007, 7007 ],
+ [ 3, 5, 7007, 7007 ],
+ [ 3, 5, 7007, 7008 ],
+ [ 3, 4, 7007, 7008 ],
+ [ 2, 4, 7008, 7008 ],
+ [ 2, 5, 7008, 7008 ],
+ [ 2, 5, 7008, 7008 ],
+ [ 2, 5, 7008, 7008 ],
+ [ 2, 5, 8000, 8000 ],
+ [ 2, 4, 8000, 8000 ],
+ [ 3, 5, 8000, 8001 ],
+ [ 3, 4, 8000, 8001 ],
+ [ 2, 5, 8000, 5001 ],
+ [ 3, 5, 8000, 5001 ],
+ [ 3, 4, 8001, 5001 ],
+ [ 2, 4, 8001, 5001 ],
+ [ 2, 5, 8002, 8029 ],
+ [ 3, 4, 8030, 8031 ],
+ [ 3, 5, 8020, 8021 ],
+ [ 2, 4, 8022, 9999 ],
+ [ 2, 4, 8023, 8025 ],
+ [ 2, 5, 8027, 8030 ],
+ [ 3, 4, 8002, 8002 ],
+ [ 3, 5, 8029, 8030 ],
+ [ 3, 5, 8031, 8031 ],
+ [ 3, 4, 8020, 8020 ],
+ [ 2, 4, 8021, 8021 ],
+ [ 2, 5, 8022, 8022 ],
+ [ 3, 4, 8023, 8023 ],
+ [ 3, 5, 8025, 8025 ],
+ [ 2, 4, 8027, 8027 ],
+ [ 2, 5, 8027, 8027 ],
+ [ 3, 4, 8023, 9999 ],
+ [ 3, 5, 8025, 9998 ],
+ [ 2, 4, 8027, 9999 ],
+ [ 2, 5, 8027, 9998 ],
+ [ 3, 4, 8000, 9999 ],
+ [ 3, 5, 8001, 9998 ],
+ [ 2, 4, 5001, 9999 ],
+ [ 2, 5, 5000, 9999 ],
+ [ 3, 4, 7005, 9999 ],
+ [ 3, 5, 7006, 9999 ],
+ [ 2, 4, 7007, 9999 ],
+ [ 2, 5, 7008, 9999 ],
+ [ 2, 4, 9998, 9998 ],
+ [ 3, 5, 9998, 9998 ],
+ [ 2, 4, 9998, 9998 ],
+ [ 2, 4, 9998, 9998 ],
+ [ 3, 5, 9998, 9998 ],
+ [ 2, 4, 9999, 9999 ],
+ [ 2, 4, 9998, 9998 ],
+ [ 3, 5, 9999, 9999 ],
+ [ 2, 4, 9999, 9999 ],
+ [ 2, 4, 9999, 9999 ],
+ [ 3, 5, 9999, 9999 ],
+);
+
+$db->kill;
+$db->start({init_rm=>1}) or $log->push->fatal;
+sleep 10;
+
+# should be option (or default) in $db->start
+sub wait_until_started {
+ my $local_cnt = 60;
+ while (--$local_cnt > 0) {
+ sleep 5;
+ my $ret = $mgm->write("all status", { wait => 3 });
+ $ret or $log->fatal;
+ my $output = $ret->{output};
+ if ($output =~ /((.|\n)*\bstarted\b(.|\n)*){4}/i) {
+ $log->put("*** db is started ***")->info;
+ return;
+ }
+ if ($output =~ /((.|\n)*\bno.contact\b(.|\n)*){4}/i) {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** db is dead ***")->fatal;
+ }
+ }
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** node recovery failed ***")->fatal;
+}
+
+sub getdbstatus {
+ my $ret = $mgm->write("all status", { wait => 3 });
+ $ret or return undef;
+ my $output = $ret->{output};
+ my @status = (); # indexed 2..5
+ for my $i (2..5) {
+ my $s = "Unknown";
+ if ($output =~ /Node\s*$i\s*:\s*(\w+)/i) {
+ $s = $1;
+ }#if
+ $status[$i] = lc $s;
+ }#for
+ return \@status;
+}
+
+# count elapsed time
+my $lasttime;
+sub settime { $lasttime = time };
+sub gettime { return time - $lasttime };
+
+wait_until_started();
+$api->start({run=>$api_cmd});
+sleep 15;
+$api2->start({run=>$api_cmd2});
+sleep 15;
+
+# loop over error inserts
+for my $x (0..1) {
+for my $e (@errors) {
+for my $z (0..3) {
+ my $i1 = $e->[0]; # db node number 2..5
+ my $i2 = $e->[1]; # db node number 2..5
+ my $c1 = $e->[2]; # error code
+ my $c2 = $e->[3]; # error code
+ my $dbnode1 = $dbnode[$i1];
+ my $dbnode2 = $dbnode[$i2];
+ my $kill_no = 9998;
+ my @survivor = ();
+ my $survivor_count = 0;
+
+ $log->put("insert error $c1 on node $i1")->push($dbnode1)->notice;
+ $log->put("and insert error $c2 on node $i2")->push($dbnode2)->notice;
+
+ for my $node (2..5) {
+ if (($node ne $i1) && ($node ne $i2)) {
+ $survivor[$survivor_count] = $node;
+ $survivor_count++;
+ }#if
+ }#for
+ # insert error
+ if ($c1 eq $kill_no) {
+ $dbnode1->kill
+ or $log->put("Kill 1 failed")->fatal;
+ } else {
+ $mgm->write("$i1 error $c1")
+ or $log->put("insert error 1 fault")->fatal;
+ }#if
+ if ($c2 eq $kill_no) {
+ $dbnode2->kill
+ or $log->put("Kill 2 failed")->fatal;
+ } else {
+ $mgm->write("$i2 error $c2")
+ or $log->put("insert error 2 fault")->fatal;
+ }#if
+
+ # after a few seconds check that node is dead
+ settime();
+ loop: {
+ gettime() <= 300
+ or $log->put("db node $i1 or $i2 refuses to die")->fatal;
+ my $status = getdbstatus()
+ or $log->put("getdbstatus error")->fatal;
+
+ if (($status->[$survivor[0]] eq 'no') ||
+ ($status->[$survivor[1]] eq 'no')) {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $db->kill;
+ $log->put("*** db is dead ***")->fatal;
+ }#if
+ if (($status->[$i1] ne 'no') || ($status->[$i2] ne 'no')) { # ...contact
+ sleep 2;
+ redo loop;
+ }#if
+ }#loop
+
+ # have to even check the process is gone
+ sleep 5;
+ if ($dbnode1->stat ne "down") {
+ $log->put("ndb did not die, kill it")->push($dbnode1)->warn;
+ $dbnode1->kill
+ or $log->put("Kill 3 failed")->fatal;
+ }#if
+ if ($dbnode2->stat ne "down") {
+ $log->put("ndb did not die, kill it")->push($dbnode2)->warn;
+ $dbnode2->kill
+ or $log->put("Kill 4 failed")->fatal;
+ }#if
+
+ $log->put("node $i1 and node $i2 is dead")->notice;
+
+ # start the failed nodes
+ $dbnode1->start
+ or $log->put("Start ndb node 1 failed")->fatal;
+ $dbnode2->start
+ or $log->put("Start ndb node 2 failed")->fatal;
+ wait_until_started();
+ $log->put("node $i1 and node $i2 is up again")->notice;
+
+ # check test pgm is running
+ my $stat = $api->stat
+ or $log->put("api->stat failed")->fatal;
+ if ($stat ne "up") {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ sleep 15;
+ $db->kill;
+ $log->put("flexBench has crashed")->fatal;
+ }#if
+ my $stat = $api2->stat
+ or $log->put("api2->stat failed")->fatal;
+ if ($stat ne "up") {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ sleep 15;
+ $db->kill;
+ $log->put("flexBench2 has crashed")->fatal;
+ }#if
+}#for
+}#for
+}#for
+print "NDBT_ProgramExit: 0 - test ready\n";
+$db->kill;
+
+# vim: set sw=4:
diff --git a/ndb/test/ndbnet/testNR.run b/ndb/test/ndbnet/testNR.run
new file mode 100644
index 00000000000..01a3d76266d
--- /dev/null
+++ b/ndb/test/ndbnet/testNR.run
@@ -0,0 +1,60 @@
+#
+
+use strict;
+use NDB::Run;
+
+my $env = NDB::Run->getenv;
+my $log = $env->getlog;
+$log->setpart(time => 1, line => 0);
+$log->setprio("info");
+
+my $database = $ENV{NDB_DATABASE};
+$log->put("start test database=$database");
+$env->init or $log->push("init failed")->fatal;
+
+my $db = $env->getdb($database) or $log->push->fatal;
+my $mgm = $db->getnode(1) or $log->push->fatal;
+my $db1 = $db->getnode(2) or $log->push->fatal;
+my $db2 = $db->getnode(3) or $log->push->fatal;
+my $api = $db->getnode(4) or $log->push->fatal;
+
+$db->kill;
+$db->start({init_rm=>1}) or $db->kill, $log->push->fatal;
+sleep 10;
+
+# should be option (or default) in $db->start
+sub wait_until_started {
+ my $local_cnt = 100;
+ while (--$local_cnt > 0) {
+ sleep 2;
+ my $ret = $mgm->write("all status", { wait => 2 });
+ $ret or $db->kill, $log->fatal;
+ my $output = $ret->{output};
+ if ($output =~ /\bstarted\b(.|\n)*started\b/i) {
+ $log->put("*** db is started ***")->info;
+ return;
+ }
+ if ($output =~ /\bno.contact\b(.|\n)*no.contact\b/i) {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $db->kill, $log->put("*** db is dead ***")->fatal;
+ }
+ }
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $db->kill, $log->put("*** node recovery failed ***")->fatal;
+}
+
+my $cnt = 0;
+wait_until_started();
+$api->start({run=>"flexBench -t 32 -l 800"});
+while (1) {
+ wait_until_started();
+ sleep 2;
+ my $dbx = (++$cnt % 2 == 1 ? $db1 : $db2);
+ $dbx->kill or $db->kill, $log->fatal;
+ sleep 2;
+ $dbx->start; # start the node
+}
+
+$db->kill;
+
+# vim: set sw=4:
diff --git a/ndb/test/ndbnet/testNR1.run b/ndb/test/ndbnet/testNR1.run
new file mode 100644
index 00000000000..8819a92c8ca
--- /dev/null
+++ b/ndb/test/ndbnet/testNR1.run
@@ -0,0 +1,61 @@
+# Node recovery killing 1 node out of 4 at the time and waiting for recover
+
+use strict;
+use NDB::Run;
+
+my $env = NDB::Run->getenv;
+my $log = $env->getlog;
+$log->setpart(time => 1, line => 0);
+$log->setprio("info");
+
+my $database = $ENV{NDB_DATABASE};
+$log->put("start test database=$database");
+$env->init or $log->push("init failed")->fatal;
+
+my $db = $env->getdb($database) or $log->push->fatal;
+my $mgm = $db->getnode(1) or $log->push->fatal;
+my $db1 = $db->getnode(2) or $log->push->fatal;
+my $api = $db->getnode(6) or $log->push->fatal;
+
+$db->kill;
+$db->start({init_rm=>1}) or $log->push->fatal;
+sleep 10;
+
+# should be option (or default) in $db->start
+sub wait_until_started {
+ my $local_cnt = 100;
+ while (--$local_cnt > 0) {
+ sleep 2;
+ my $ret = $mgm->write("all status", { wait => 2 });
+ $ret or $log->fatal;
+ my $output = $ret->{output};
+ if ($output =~ /((.|\n)*\bstarted\b(.|\n)*){1}/i) {
+# if all 4 nodes started
+ $log->put("*** db is started ***")->info;
+ return;
+ }
+ if ($output =~ /((.|\n)*\bno.contact\b(.|\n)*){1}/i) {
+#if all 4 nodes no contact
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** db is dead ***")->fatal;
+ }
+ }
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** node recovery failed ***")->fatal;
+}
+
+my $cnt = 0;
+wait_until_started();
+$api->start({run=>"flexBench -t 32 -l 800"});
+while (1) {
+ wait_until_started();
+ sleep 10;
+ $db1->kill or $log->fatal;
+ sleep 10;
+ $db1->start; # start the node
+ sleep 10;
+}
+
+$db->kill;
+
+# vim: set sw=4:
diff --git a/ndb/test/ndbnet/testNR4.run b/ndb/test/ndbnet/testNR4.run
new file mode 100644
index 00000000000..f7a5eef3494
--- /dev/null
+++ b/ndb/test/ndbnet/testNR4.run
@@ -0,0 +1,77 @@
+# Node recovery killing 1 node out of 4 at the time and waiting for recover
+
+use strict;
+use NDB::Run;
+
+my $env = NDB::Run->getenv;
+my $log = $env->getlog;
+$log->setpart(time => 1, line => 0);
+$log->setprio("info");
+
+my $database = $ENV{NDB_DATABASE};
+$log->put("start test database=$database");
+$env->init or $log->push("init failed")->fatal;
+
+my $db = $env->getdb($database) or $log->push->fatal;
+my $mgm = $db->getnode(1) or $log->push->fatal;
+my $db1 = $db->getnode(2) or $log->push->fatal;
+my $db2 = $db->getnode(3) or $log->push->fatal;
+my $db3 = $db->getnode(4) or $log->push->fatal;
+my $db4 = $db->getnode(5) or $log->push->fatal;
+my $api = $db->getnode(6) or $log->push->fatal;
+
+$db->kill;
+$db->start({init_rm=>1}) or $log->push->fatal;
+sleep 10;
+
+# should be option (or default) in $db->start
+sub wait_until_started {
+ my $local_cnt = 100;
+ while (--$local_cnt > 0) {
+ sleep 2;
+ my $ret = $mgm->write("all status", { wait => 2 });
+ $ret or $log->fatal;
+ my $output = $ret->{output};
+ if ($output =~ /((.|\n)*\bstarted\b(.|\n)*){4}/i) {
+# if all 4 nodes started
+ $log->put("*** db is started ***")->info;
+ return;
+ }
+ if ($output =~ /((.|\n)*\bno.contact\b(.|\n)*){4}/i) {
+#if all 4 nodes no contact
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** db is dead ***")->fatal;
+ }
+ }
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** node recovery failed ***")->fatal;
+}
+
+my $cnt = 0;
+wait_until_started();
+$api->start({run=>"flexBench -t 32 -l 800"});
+while (1) {
+ wait_until_started();
+ sleep 10;
+ my $id = (++$cnt % 4);
+ my $dbx = 0;
+ if ($id eq 0) {
+ $dbx = $db1;
+ }
+ elsif ($id eq 1) {
+ $dbx = $db2;
+ }
+ elsif ($id eq 2) {
+ $dbx = $db3;
+ }
+ else {
+ $dbx = $db4;
+ }
+ $dbx->kill or $log->fatal;
+ sleep 10;
+ $dbx->start; # start the node
+}
+
+$db->kill;
+
+# vim: set sw=4:
diff --git a/ndb/test/ndbnet/testSRhang.run b/ndb/test/ndbnet/testSRhang.run
new file mode 100644
index 00000000000..8cb65a75ded
--- /dev/null
+++ b/ndb/test/ndbnet/testSRhang.run
@@ -0,0 +1,50 @@
+#
+
+use strict;
+use NDB::Run;
+
+my $env = NDB::Run->getenv;
+my $log = $env->getlog;
+$log->setpart(time => 1, line => 0);
+$log->setprio("info");
+
+my $database = $ENV{NDB_DATABASE};
+$log->put("start test database=$database");
+$env->init or $log->push("init failed")->fatal;
+
+my $db = $env->getdb($database) or $log->push->fatal;
+my $mgm = $db->getnode(1) or $log->push->fatal;
+
+# should be option (or default) in $db->start
+sub wait_until_started {
+ my $local_cnt = 200;
+ while (--$local_cnt > 0) {
+ sleep 2;
+ if ($local_cnt < 150) {
+ my $x = $mgm->write("all dump 1", { wait => 2 });
+ }#if
+ my $ret = $mgm->write("all status", { wait => 2 });
+ $ret or $log->fatal;
+ my $output = $ret->{output};
+ if ($output =~ /\bstarted\b(.|\n)*started\b/i) {
+ $log->put("*** db is started ***")->info;
+ return;
+ }#if
+ if ($output =~ /\bno.contact\b(.|\n)*no.contact\b/i) {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $db->kill;
+ $log->put("*** db is dead ***")->fatal;
+ }#if
+ }#while
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $db->kill;
+ $log->put("*** initial start failed ***")->fatal;
+}#sub wait_until_started
+while (1) {
+ $db->kill;
+ $db->start({init_rm=>1}) or $log->push->fatal;
+ sleep 10;
+ wait_until_started();
+}#while
+
+# vim: set sw=4:
diff --git a/ndb/test/ndbnet/testTR295.run b/ndb/test/ndbnet/testTR295.run
new file mode 100644
index 00000000000..ce4250b60ae
--- /dev/null
+++ b/ndb/test/ndbnet/testTR295.run
@@ -0,0 +1,75 @@
+# testing TR295, kill non-master when recovering in phase 4
+
+use strict;
+use NDB::Run;
+
+my $env = NDB::Run->getenv;
+my $log = $env->getlog;
+$log->setpart(time => 1, line => 0);
+$log->setprio("info");
+
+my $database = $ENV{NDB_DATABASE};
+$log->put("start test database=$database");
+$env->init or $log->push("init failed")->fatal;
+
+my $db = $env->getdb($database) or $log->push->fatal;
+my $mgm = $db->getnode(1) or $log->push->fatal;
+my $db1 = $db->getnode(2) or $log->push->fatal;
+my $db2 = $db->getnode(3) or $log->push->fatal;
+
+$db->kill;
+$db->start({init_rm=>1}) or $log->push->fatal;
+sleep 10;
+
+# should be option (or default) in $db->start
+sub wait_until_started {
+ my $local_cnt = 100;
+ while (--$local_cnt > 0) {
+ sleep 2;
+ my $ret = $mgm->write("all status", { wait => 2 });
+ $ret or $log->fatal;
+ my $output = $ret->{output};
+ if ($output =~ /\bstarted\b(.|\n)*started\b/i) {
+ $log->put("*** db is started ***")->info;
+ return;
+ }
+ if ($output =~ /\bno.contact\b(.|\n)*no.contact\b/i) {
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** db is dead ***")->fatal;
+ # Maybe try a system restart?
+ }
+ }
+ print "NDBT_ProgramExit: 1 - test error\n";
+ $log->put("*** node recovery failed ***")->fatal;
+}
+sub wait_until_phase4_and_kill {
+# kill db2 as quick as possible when it has reached phase4
+my $cnt = 1000;
+ while (--$cnt > 0) {
+ my $ret = $mgm->write("3 status", {wait => 0});
+ #$ret or log->fatal;
+ my $output = $ret->{output};
+ if ($output =~ /\bphase 4\b/i) {
+ $db2->kill or log->fatal;
+ return;
+ }
+ }
+ print "NDBT_ProgramExit: 1 -test error\n";
+ $log->put("*** node restart failed after 1000 loops ***")->fatal;
+}
+
+my $cnt = 0;
+wait_until_started();
+while (1) {
+ wait_until_started();
+ sleep 2;
+ $db2->kill or $log->fatal;
+ $db2->start; # start the node
+ wait_until_phase4_and_kill();
+ sleep 10;
+ $db2->start;
+}
+
+$db->kill;
+
+# vim: set sw=4:
diff --git a/ndb/test/newtonapi/Makefile b/ndb/test/newtonapi/Makefile
new file mode 100644
index 00000000000..e3eabd26c64
--- /dev/null
+++ b/ndb/test/newtonapi/Makefile
@@ -0,0 +1,8 @@
+include .defs.mk
+
+DIRS := \
+ basic_test \
+ perf_test
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/newtonapi/basic_test/Makefile b/ndb/test/newtonapi/basic_test/Makefile
new file mode 100644
index 00000000000..d7eaf984b12
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/Makefile
@@ -0,0 +1,25 @@
+include .defs.mk
+
+TYPE := util
+
+PIC_ARCHIVE := Y
+ARCHIVE_TARGET := newtonbasictestcommon
+
+A_LIB := Y
+SO_LIB := Y
+PIC_LIB := Y
+
+LIB_TARGET := NEWTON_BASICTEST_COMMON
+LIB_TARGET_ARCHIVES := $(ARCHIVE_TARGET) NEWTON_API
+
+
+SOURCES = common.cpp
+
+DIRS := basic \
+ ptr_binding \
+ bulk_read
+
+CCFLAGS_LOC := -I$(call fixpath,$(NDB_TOP)/include/util) -I$(call fixpath,$(NDB_TOP)/include/newtonapi)
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/newtonapi/basic_test/basic/Makefile b/ndb/test/newtonapi/basic_test/basic/Makefile
new file mode 100644
index 00000000000..7e2945d2e5f
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/basic/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := newton_basic
+BIN_TARGET_LIBS :=
+BIN_TARGET_ARCHIVES := NEWTON_BASICTEST_COMMON NEWTON_API
+SOURCES := basic.cpp
+
+CCFLAGS_LOC := -I.. -I$(call fixpath,$(NDB_TOP)/include/util) -I$(call fixpath,$(NDB_TOP)/include/newtonapi) -I$(call fixpath,$(NDB_TOP)/include/portlib)
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/test/newtonapi/basic_test/basic/basic.cpp b/ndb/test/newtonapi/basic_test/basic/basic.cpp
new file mode 100644
index 00000000000..90f5bf14acf
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/basic/basic.cpp
@@ -0,0 +1,322 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+extern "C" {
+#include <dba.h>
+}
+
+#include "common.hpp"
+
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbMain.h>
+#include <assert.h>
+
+static const
+DBA_ColumnDesc_t EmpColDesc[] = {
+ { "emp_no", DBA_INT, PCN_SIZE_OF( Employee, EmpNo ), PCN_TRUE },
+ { "first_name", DBA_CHAR, PCN_SIZE_OF( Employee, FirstName ), PCN_FALSE },
+ { "last_name", DBA_CHAR, PCN_SIZE_OF( Employee, LastName ), PCN_FALSE }
+};
+
+static const
+DBA_ColumnDesc_t AddColDesc[] = {
+ { "emp_no", DBA_INT, PCN_SIZE_OF( Address, EmpNo ), PCN_TRUE },
+ { "street_name", DBA_CHAR, PCN_SIZE_OF( Address, StreetName ), PCN_FALSE},
+ { "street_no", DBA_INT, PCN_SIZE_OF( Address, StreetNo ), PCN_FALSE},
+ { "city", DBA_CHAR, PCN_SIZE_OF( Address, City ), PCN_FALSE}
+} ;
+
+static const
+DBA_ColumnBinding_t EmpBindings[] = {
+ DBA_BINDING( "emp_no", DBA_INT, Employee, EmpNo ),
+ DBA_BINDING( "last_name", DBA_CHAR, Employee, LastName ),
+ DBA_BINDING( "first_name", DBA_CHAR, Employee, FirstName)
+};
+
+static const
+DBA_ColumnBinding_t AddBindings[] = {
+ DBA_BINDING( "emp_no", DBA_INT, Address, EmpNo ),
+ DBA_BINDING( "street_name", DBA_CHAR, Address, StreetName ),
+ DBA_BINDING( "street_no", DBA_INT, Address, StreetNo ),
+ DBA_BINDING( "city", DBA_CHAR, Address, City )
+};
+
+static DBA_Binding_t * EmpB;
+static DBA_Binding_t * AddB;
+
+static const int Rows = 6;
+
+static
+Employee_t EMP_TABLE_DATA[] = {
+ { 1242, "Joe", "Dalton" },
+ { 123, "Lucky", "Luke" },
+ { 456, "Averell", "Dalton" },
+ { 8976, "Gaston", "Lagaffe" },
+ { 1122, "Jolly", "Jumper" },
+ { 3211, "Leffe", "Pagrotsky" }
+};
+
+static
+Employee_t EMP_TABLE_DATA_READ[] = {
+ { 1242, "", "" },
+ { 123, "", "" },
+ { 456, "", "" },
+ { 8976, "", "" },
+ { 1122, "", "" },
+ { 3211, "", "" }
+};
+
+static
+Address_t ADD_TABLE_DATA[] = {
+ { 1242, "Lonesome Street", 12, "Crime Town" },
+ { 123, "Pistol Road", 13, "Fort Mount" },
+ { 456, "Banking Blv.", 43, "Las Vegas" },
+ { 8976, "ChancylleZee", 54, "Paris" },
+ { 1122, "Lucky", 111, "Wild West" },
+ { 3211, "Parlament St.", 11, "Stockholm" }
+};
+
+static
+Address_t ADD_TABLE_DATA_READ[] = {
+ { 1242, "", 0, "" },
+ { 123, "", 0, "" },
+ { 456, "", 0, "" },
+ { 8976, "", 0, "" },
+ { 1122, "", 0, "" },
+ { 3211, "", 0, "" }
+};
+
+static const char EMP_TABLE[] = "employees";
+static const char ADD_TABLE[] = "addresses";
+
+static const int EmpNbCol = 3;
+static const int AddNbCol = 4;
+
+static
+void
+DbCreate(void){
+
+ ndbout << "Opening database" << endl;
+ require( DBA_Open() == DBA_NO_ERROR );
+
+ ndbout << "Creating tables" << endl;
+ require( DBA_CreateTable( EMP_TABLE, EmpNbCol, EmpColDesc ) == DBA_NO_ERROR );
+ require( DBA_CreateTable( ADD_TABLE, AddNbCol, AddColDesc ) == DBA_NO_ERROR );
+
+ ndbout << "Checking for table existance" << endl;
+ require( DBA_TableExists( EMP_TABLE ) );
+ require( DBA_TableExists( ADD_TABLE ) );
+}
+
+static
+void
+CreateBindings(void){
+ ndbout << "Creating bindings" << endl;
+
+ EmpB = DBA_CreateBinding(EMP_TABLE,
+ EmpNbCol,
+ EmpBindings,
+ sizeof(Employee_t) );
+ require(EmpB != 0);
+
+ AddB = DBA_CreateBinding(ADD_TABLE,
+ AddNbCol,
+ AddBindings,
+ sizeof(Address_t) );
+ require(AddB != 0);
+}
+
+extern "C" {
+ static void insertCallback( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void deleteCallback( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void updateCallback( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void readCallback ( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void writeCallback ( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+}
+
+static
+void BasicArray(){
+ ndbout << "Testing basic array operations" << endl;
+
+ // Basic insert
+ DBA_ArrayInsertRows(EmpB, EMP_TABLE_DATA, Rows-2, insertCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ArrayReadRows(EmpB, EMP_TABLE_DATA_READ, Rows-2, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+
+ // Basic update
+ AlterRows(EMP_TABLE_DATA, Rows-2);
+ DBA_ArrayUpdateRows(EmpB, EMP_TABLE_DATA, Rows-2, updateCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ArrayReadRows(EmpB, EMP_TABLE_DATA_READ, Rows-2, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+
+ // Basic write
+ AlterRows(EMP_TABLE_DATA, Rows);
+ DBA_ArrayWriteRows(EmpB, EMP_TABLE_DATA, Rows, writeCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ArrayReadRows(EmpB, EMP_TABLE_DATA_READ, Rows, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows, EMP_TABLE_DATA_READ);
+
+ // Basic delete
+ DBA_ArrayDeleteRows(EmpB, EMP_TABLE_DATA, Rows, deleteCallback);
+ NdbSleep_SecSleep(1);
+}
+
+static
+void Multi(){
+ ndbout << "Testing multi operations" << endl;
+
+ const int R2 = Rows + Rows;
+
+ DBA_Binding_t * Bindings[R2];
+ void * DATA[R2];
+ void * DATA_READ[R2];
+ for(int i = 0; i<Rows; i++){
+ Bindings[2*i] = EmpB;
+ Bindings[2*i+1] = AddB;
+
+ DATA[2*i] = &EMP_TABLE_DATA[i];
+ DATA[2*i+1] = &ADD_TABLE_DATA[i];
+
+ DATA_READ[2*i] = &EMP_TABLE_DATA_READ[i];
+ DATA_READ[2*i+1] = &ADD_TABLE_DATA_READ[i];
+ }
+
+ // Basic insert
+ DBA_MultiInsertRow(Bindings, DATA, R2-4, insertCallback);
+ NdbSleep_SecSleep(1);
+ DBA_MultiReadRow (Bindings, DATA_READ, R2-4, readCallback);
+ NdbSleep_SecSleep(1);
+
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows-2, ADD_TABLE_DATA_READ);
+
+ // Basic update
+ AlterRows(EMP_TABLE_DATA, Rows-2);
+ AlterRows(ADD_TABLE_DATA, Rows-2);
+ DBA_MultiUpdateRow(Bindings, DATA, R2-4, updateCallback);
+ NdbSleep_SecSleep(1);
+ DBA_MultiReadRow (Bindings, DATA_READ, R2-4, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows-2, ADD_TABLE_DATA_READ);
+
+ // Basic write
+ AlterRows(EMP_TABLE_DATA, Rows);
+ AlterRows(ADD_TABLE_DATA, Rows);
+ DBA_MultiWriteRow(Bindings, DATA, R2, writeCallback);
+ NdbSleep_SecSleep(1);
+ DBA_MultiReadRow (Bindings, DATA_READ, R2, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows, ADD_TABLE_DATA_READ);
+
+ // Basic delete
+ DBA_MultiDeleteRow(Bindings, DATA, R2, deleteCallback);
+ NdbSleep_SecSleep(1);
+}
+
+static
+void BasicPtr(){
+ ndbout << "Testing array of pointer operations" << endl;
+ Employee_t * EmpData[Rows];
+ Employee_t * EmpDataRead[Rows];
+ for(int i = 0; i<Rows; i++){
+ EmpData[i] = &EMP_TABLE_DATA[i];
+ EmpDataRead[i] = &EMP_TABLE_DATA_READ[i];
+ }
+
+ void * const * EMP_TABLE_DATA2 = (void * const *)EmpData;
+ void * const * EMP_TABLE_DATA_READ2 = (void * const *)EmpDataRead;
+
+ // Basic insert
+ DBA_InsertRows(EmpB, EMP_TABLE_DATA2, Rows-2, insertCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ReadRows (EmpB, EMP_TABLE_DATA_READ2, Rows-2, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+
+ // Basic update
+ AlterRows(EMP_TABLE_DATA, Rows-2);
+ DBA_UpdateRows(EmpB, EMP_TABLE_DATA2, Rows-2, updateCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ReadRows (EmpB, EMP_TABLE_DATA_READ2, Rows-2, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+
+ // Basic write
+ AlterRows (EMP_TABLE_DATA, Rows);
+ DBA_WriteRows(EmpB, EMP_TABLE_DATA2, Rows, writeCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ReadRows (EmpB, EMP_TABLE_DATA_READ2, Rows, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows, EMP_TABLE_DATA_READ);
+
+ // Basic delete
+ DBA_DeleteRows(EmpB, EMP_TABLE_DATA2, Rows, deleteCallback);
+ NdbSleep_SecSleep(1);
+}
+
+/*---------------------------------------------------------------------------*/
+NDB_COMMAND(newton_basic, "newton_basic",
+ "newton_basic", "newton_basic", 65535){
+
+ DbCreate();
+ CreateBindings();
+
+ BasicArray();
+ BasicPtr();
+ Multi();
+
+ DBA_Close();
+
+ return 0;
+}
+
+
+
+/**
+ * callbackStatusCheck checks whether or not the operation succeeded
+ */
+void
+callbackStatusCheck( DBA_Error_t status, const char* operation) {
+ ndbout_c("%s: %d", operation, status);
+}
+
+void insertCallback( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "insert");
+}
+void deleteCallback( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "delete");
+}
+void updateCallback( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "update");
+}
+void readCallback ( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "read");
+}
+void writeCallback ( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "write");
+}
+
diff --git a/ndb/test/newtonapi/basic_test/bulk_read/Makefile b/ndb/test/newtonapi/basic_test/bulk_read/Makefile
new file mode 100644
index 00000000000..c45bbad7957
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/bulk_read/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := newton_br
+BIN_TARGET_LIBS :=
+BIN_TARGET_ARCHIVES := NEWTON_BASICTEST_COMMON NEWTON_API
+SOURCES := br_test.cpp
+
+CCFLAGS_LOC := -I.. -I$(call fixpath,$(NDB_TOP)/include/util) -I$(call fixpath,$(NDB_TOP)/include/newtonapi) -I$(call fixpath,$(NDB_TOP)/include/portlib)
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/test/newtonapi/basic_test/bulk_read/br_test.cpp b/ndb/test/newtonapi/basic_test/bulk_read/br_test.cpp
new file mode 100644
index 00000000000..fb26abc65aa
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/bulk_read/br_test.cpp
@@ -0,0 +1,263 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+extern "C" {
+#include <dba.h>
+}
+
+#include "common.hpp"
+
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbMain.h>
+#include <assert.h>
+
+static const
+DBA_ColumnDesc_t EmpColDesc[] = {
+ { "emp_no", DBA_INT, PCN_SIZE_OF( Employee, EmpNo ), PCN_TRUE },
+ { "first_name", DBA_CHAR, PCN_SIZE_OF( Employee, FirstName ), PCN_FALSE },
+ { "last_name", DBA_CHAR, PCN_SIZE_OF( Employee, LastName ), PCN_FALSE }
+};
+
+static const
+DBA_ColumnDesc_t AddColDesc[] = {
+ { "emp_no", DBA_INT, PCN_SIZE_OF( Address, EmpNo ), PCN_TRUE },
+ { "street_name", DBA_CHAR, PCN_SIZE_OF( Address, StreetName ), PCN_FALSE},
+ { "street_no", DBA_INT, PCN_SIZE_OF( Address, StreetNo ), PCN_FALSE},
+ { "city", DBA_CHAR, PCN_SIZE_OF( Address, City ), PCN_FALSE}
+} ;
+
+static const
+DBA_ColumnBinding_t EmpBindings[] = {
+ DBA_BINDING( "emp_no", DBA_INT, Employee, EmpNo ),
+ DBA_BINDING( "last_name", DBA_CHAR, Employee, LastName ),
+ DBA_BINDING( "first_name", DBA_CHAR, Employee, FirstName)
+};
+
+static const
+DBA_ColumnBinding_t AddBindings[] = {
+ DBA_BINDING( "emp_no", DBA_INT, Address, EmpNo ),
+ DBA_BINDING( "street_name", DBA_CHAR, Address, StreetName ),
+ DBA_BINDING( "street_no", DBA_INT, Address, StreetNo ),
+ DBA_BINDING( "city", DBA_CHAR, Address, City )
+};
+
+static DBA_Binding_t * EmpB;
+static DBA_Binding_t * AddB;
+
+static const int Rows = 6;
+
+static
+Employee_t EMP_TABLE_DATA[] = {
+ { 1242, "Joe", "Dalton" },
+ { 123, "Lucky", "Luke" },
+ { 456, "Averell", "Dalton" },
+ { 8976, "Gaston", "Lagaffe" },
+ { 1122, "Jolly", "Jumper" },
+ { 3211, "Leffe", "Pagrotsky" }
+};
+
+static
+Employee_t EMP_TABLE_DATA_READ[] = {
+ { 1242, "", "" },
+ { 123, "", "" },
+ { 456, "", "" },
+ { 8976, "", "" },
+ { 1122, "", "" },
+ { 3211, "", "" }
+};
+
+static
+Address_t ADD_TABLE_DATA[] = {
+ { 1242, "Lonesome Street", 12, "Crime Town" },
+ { 123, "Pistol Road", 13, "Fort Mount" },
+ { 456, "Banking Blv.", 43, "Las Vegas" },
+ { 8976, "ChancylleZee", 54, "Paris" },
+ { 1122, "Lucky", 111, "Wild West" },
+ { 3211, "Parlament St.", 11, "Stockholm" }
+};
+
+static
+Address_t ADD_TABLE_DATA_READ[] = {
+ { 1242, "", 0, "" },
+ { 123, "", 0, "" },
+ { 456, "", 0, "" },
+ { 8976, "", 0, "" },
+ { 1122, "", 0, "" },
+ { 3211, "", 0, "" }
+};
+
+static const char EMP_TABLE[] = "employees";
+static const char ADD_TABLE[] = "addresses";
+
+static const int EmpNbCol = 3;
+static const int AddNbCol = 4;
+
+static
+void
+DbCreate(void){
+
+ ndbout << "Opening database" << endl;
+ require( DBA_Open() == DBA_NO_ERROR );
+
+ ndbout << "Creating tables" << endl;
+ require( DBA_CreateTable( EMP_TABLE, EmpNbCol, EmpColDesc ) == DBA_NO_ERROR );
+ require( DBA_CreateTable( ADD_TABLE, AddNbCol, AddColDesc ) == DBA_NO_ERROR );
+
+ ndbout << "Checking for table existance" << endl;
+ require( DBA_TableExists( EMP_TABLE ) );
+ require( DBA_TableExists( ADD_TABLE ) );
+}
+
+static
+void
+CreateBindings(void){
+ ndbout << "Creating bindings" << endl;
+
+ EmpB = DBA_CreateBinding(EMP_TABLE,
+ EmpNbCol,
+ EmpBindings,
+ sizeof(Employee_t) );
+ require(EmpB != 0);
+
+ AddB = DBA_CreateBinding(ADD_TABLE,
+ AddNbCol,
+ AddBindings,
+ sizeof(Address_t) );
+ require(AddB != 0);
+}
+
+int
+CountRows(DBA_BulkReadResultSet_t * rs, int count){
+ int res = 0;
+ for(int i = 0; i<count; i++)
+ if(rs[i].RowFoundIndicator)
+ res++;
+ return res;
+}
+
+extern "C" {
+ static void insertCallback( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void deleteCallback( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void updateCallback( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void readCallback ( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void writeCallback ( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+}
+
+static
+void Multi(){
+ ndbout << "Testing multi operations" << endl;
+
+ DBA_ArrayInsertRows(EmpB, EMP_TABLE_DATA, Rows-2, insertCallback);
+ DBA_ArrayInsertRows(AddB, ADD_TABLE_DATA, Rows-2, insertCallback);
+ NdbSleep_SecSleep(1);
+
+ const int R2 = Rows + Rows;
+
+ DBA_Binding_t * Bindings[2];
+ DBA_BulkReadResultSet_t DataRead[R2];
+
+ Bindings[0] = EmpB;
+ Bindings[1] = AddB;
+
+ for(int i = 0; i<Rows; i++)
+ DataRead[i].DataPtr = &EMP_TABLE_DATA_READ[i];
+
+ for(int i = 0; i<Rows; i++)
+ DataRead[i+Rows].DataPtr = &ADD_TABLE_DATA_READ[i];
+
+ NdbSleep_SecSleep(1);
+
+ DBA_BulkMultiReadRows(Bindings, DataRead, 2, Rows, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows-2, ADD_TABLE_DATA_READ);
+
+ require(CountRows(DataRead, R2) == (R2-4));
+
+ // Basic delete
+ DBA_ArrayDeleteRows(EmpB, EMP_TABLE_DATA, Rows-2, deleteCallback);
+ DBA_ArrayDeleteRows(AddB, ADD_TABLE_DATA, Rows-2, deleteCallback);
+ NdbSleep_SecSleep(1);
+}
+
+static
+void BasicPtr(){
+ ndbout << "Testing array of pointer operations" << endl;
+
+ // Basic insert
+ DBA_ArrayInsertRows(EmpB, EMP_TABLE_DATA, Rows-2, insertCallback);
+ NdbSleep_SecSleep(1);
+
+ DBA_BulkReadResultSet_t EmpDataRead[Rows];
+ for(int i = 0; i<Rows; i++){
+ EmpDataRead[i].DataPtr = &EMP_TABLE_DATA_READ[i];
+ }
+
+ DBA_BulkReadRows(EmpB, EmpDataRead, Rows, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+ require(CountRows(EmpDataRead, Rows) == (Rows-2));
+
+ // Basic delete
+ DBA_ArrayDeleteRows(EmpB, EMP_TABLE_DATA, Rows-2, deleteCallback);
+ NdbSleep_SecSleep(1);
+}
+
+/*---------------------------------------------------------------------------*/
+NDB_COMMAND(newton_br, "newton_br",
+ "newton_br", "newton_br", 65535){
+
+ DbCreate();
+ CreateBindings();
+
+ BasicPtr();
+ Multi();
+
+ DBA_Close();
+
+ return 0;
+}
+
+
+
+/**
+ * callbackStatusCheck checks whether or not the operation succeeded
+ */
+void
+callbackStatusCheck( DBA_Error_t status, const char* operation) {
+ ndbout_c("%s: %d", operation, status);
+}
+
+void insertCallback( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "insert");
+}
+void deleteCallback( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "delete");
+}
+void updateCallback( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "update");
+}
+void readCallback ( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "read");
+}
+void writeCallback ( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "write");
+}
+
diff --git a/ndb/test/newtonapi/basic_test/common.cpp b/ndb/test/newtonapi/basic_test/common.cpp
new file mode 100644
index 00000000000..6fd63730832
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/common.cpp
@@ -0,0 +1,137 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "common.hpp"
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <ctype.h>
+
+NdbOut &
+operator << (NdbOut & out, const Employee_t & emp){
+ out << emp.EmpNo << " \"" << emp.FirstName << "\" \""
+ << emp.LastName << "\"";
+ return out;
+}
+
+bool
+operator==(const Employee_t & e1, const Employee_t & e2){
+ if(e1.EmpNo != e2.EmpNo)
+ return false;
+ if(strcmp(e1.FirstName, e2.FirstName) != 0)
+ return false;
+ return strcmp(e1.LastName, e2.LastName) == 0;
+}
+
+void
+Alter(Employee_t & emp){
+ static int updown = 0;
+ if(updown == 0){
+ for(int i = 0; i<strlen(emp.FirstName); i++)
+ toupper(emp.FirstName[i]);
+
+ for(int i = 0; i<strlen(emp.LastName); i++)
+ toupper(emp.LastName[i]);
+ } else {
+ for(int i = 0; i<strlen(emp.FirstName); i++)
+ tolower(emp.FirstName[i]);
+
+ for(int i = 0; i<strlen(emp.LastName); i++)
+ tolower(emp.LastName[i]);
+ }
+ updown = 1 - updown;
+}
+
+void
+CompareRows(Employee_t * data1,
+ int rows,
+ Employee_t * data2){
+ for(int i = 0; i<rows; i++){
+ if(!(data1[i] == data2[i])){
+ ndbout << data1[i] << endl
+ << data2[i] << endl;
+ }
+ }
+}
+
+void
+AlterRows(Employee_t * data1, int rows){
+ for(int i = 0; i<rows; i++){
+ Alter(data1[i]);
+ }
+}
+
+inline
+NdbOut &
+operator << (NdbOut & out, const Address_t & adr){
+ out << adr.EmpNo << " \"" << adr.StreetName << "\" "
+ << adr.StreetNo << " \"" << adr.City << "\"";
+ return out;
+}
+
+inline
+bool
+operator==(const Address_t & a1, const Address_t & a2){
+ if(a1.EmpNo != a2.EmpNo)
+ return false;
+ if(a1.StreetNo != a2.StreetNo)
+ return false;
+ if(strcmp(a1.StreetName, a2.StreetName) != 0)
+ return false;
+ return strcmp(a1.City, a2.City) == 0;
+}
+
+inline
+void
+Alter(Address_t & emp){
+ static int updown = 0;
+ if(updown == 0){
+ for(int i = 0; i<strlen(emp.StreetName); i++)
+ toupper(emp.StreetName[i]);
+
+ for(int i = 0; i<strlen(emp.City); i++)
+ toupper(emp.City[i]);
+ } else {
+ for(int i = 0; i<strlen(emp.StreetName); i++)
+ tolower(emp.StreetName[i]);
+
+ for(int i = 0; i<strlen(emp.City); i++)
+ tolower(emp.City[i]);
+ }
+ emp.StreetNo *= emp.EmpNo;
+ updown = 1 - updown;
+}
+
+void
+CompareRows(Address_t * data1,
+ int rows,
+ Address_t * data2){
+ for(int i = 0; i<rows; i++){
+ if(!(data1[i] == data2[i])){
+ ndbout << data1[i] << endl
+ << data2[i] << endl;
+ }
+ }
+}
+
+void
+AlterRows(Address_t * data1, int rows){
+ for(int i = 0; i<rows; i++){
+ Alter(data1[i]);
+ }
+}
+
diff --git a/ndb/test/newtonapi/basic_test/common.hpp b/ndb/test/newtonapi/basic_test/common.hpp
new file mode 100644
index 00000000000..e081df723ef
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/common.hpp
@@ -0,0 +1,65 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef COMMON_H
+#define COMMON_H
+
+extern "C" {
+#include <dba.h>
+}
+
+#include <NdbOut.hpp>
+#include <stdlib.h>
+
+typedef struct Employee {
+ UInt32_t EmpNo;
+ char FirstName[24];
+ char LastName[24];
+
+ struct Address * EmployeeAddress;
+} Employee_t;
+
+typedef struct Address {
+ UInt32_t EmpNo;
+ char StreetName[24];
+ UInt32_t StreetNo;
+ char City[12];
+} Address_t;
+
+/**
+ * Employee functions
+ */
+NdbOut & operator << (NdbOut & out, const Employee_t & emp);
+bool operator==(const Employee_t & e1, const Employee_t & e2);
+void Alter(Employee_t & emp);
+void CompareRows(Employee_t * data1, int rows, Employee_t * data2);
+void AlterRows(Employee_t * data1, int rows);
+
+/**
+ * Address functions
+ */
+NdbOut & operator << (NdbOut & out, const Address_t & adr);
+bool operator==(const Address_t & a1, const Address_t & a2);
+void Alter(Address_t & emp);
+void CompareRows(Address_t * data1, int rows, Address_t * data2);
+void AlterRows(Address_t * data1, int rows);
+
+inline void require(bool test){
+ if(!test)
+ abort();
+}
+
+#endif
diff --git a/ndb/test/newtonapi/basic_test/ptr_binding/Makefile b/ndb/test/newtonapi/basic_test/ptr_binding/Makefile
new file mode 100644
index 00000000000..95e87d47e62
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/ptr_binding/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := newton_pb
+BIN_TARGET_LIBS :=
+BIN_TARGET_ARCHIVES := NEWTON_BASICTEST_COMMON NEWTON_API
+SOURCES := ptr_binding_test.cpp
+
+CCFLAGS_LOC := -I.. -I$(call fixpath,$(NDB_TOP)/include/util) -I$(call fixpath,$(NDB_TOP)/include/newtonapi) -I$(call fixpath,$(NDB_TOP)/include/portlib)
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/test/newtonapi/basic_test/ptr_binding/ptr_binding_test.cpp b/ndb/test/newtonapi/basic_test/ptr_binding/ptr_binding_test.cpp
new file mode 100644
index 00000000000..fc6dfc40372
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/ptr_binding/ptr_binding_test.cpp
@@ -0,0 +1,265 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+extern "C" {
+#include <dba.h>
+}
+
+#include "common.hpp"
+
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbMain.h>
+#include <assert.h>
+
+static const
+DBA_ColumnDesc_t ColDesc[] = {
+ { "emp_no", DBA_INT, PCN_SIZE_OF( Employee, EmpNo ), PCN_TRUE },
+ { "first_name", DBA_CHAR, PCN_SIZE_OF( Employee, FirstName ), PCN_FALSE },
+ { "last_name", DBA_CHAR, PCN_SIZE_OF( Employee, LastName ), PCN_FALSE },
+ { "street_name",DBA_CHAR, PCN_SIZE_OF( Address, StreetName ), PCN_FALSE },
+ { "street_no", DBA_INT, PCN_SIZE_OF( Address, StreetNo ), PCN_FALSE },
+ { "city", DBA_CHAR, PCN_SIZE_OF( Address, City ), PCN_FALSE }
+};
+static const int NbCol = 6;
+
+static const
+DBA_ColumnBinding_t AddBindings[] = {
+ //DBA_BINDING( "emp_no", DBA_INT, Address, EmpNo ),
+ DBA_BINDING( "street_name", DBA_CHAR, Address, StreetName ),
+ DBA_BINDING( "street_no", DBA_INT, Address, StreetNo ),
+ DBA_BINDING( "city", DBA_CHAR, Address, City )
+};
+
+static const
+int AddBindingRows = sizeof(AddBindings)/sizeof(DBA_ColumnBinding_t);
+
+static const
+DBA_ColumnBinding_t EmpBindings[] = {
+ DBA_BINDING( "emp_no", DBA_INT, Employee, EmpNo ),
+ DBA_BINDING( "last_name", DBA_CHAR, Employee, LastName ),
+ DBA_BINDING( "first_name", DBA_CHAR, Employee, FirstName),
+ DBA_BINDING_PTR(Employee, EmployeeAddress, AddBindings, AddBindingRows)
+};
+static const
+int EmpBindingRows = sizeof(EmpBindings)/sizeof(DBA_ColumnBinding_t);
+
+static DBA_Binding_t * Bind;
+
+static const int Rows = 6;
+
+static
+Address_t ADD_TABLE_DATA[] = {
+ { 1242, "Lonesome Street", 12, "Crime Town" },
+ { 123, "Pistol Road", 13, "Fort Mount" },
+ { 456, "Banking Blv.", 43, "Las Vegas" },
+ { 8976, "ChancylleZee", 54, "Paris" },
+ { 1122, "Lucky", 111, "Wild West" },
+ { 3211, "Parlament St.", 11, "Stockholm" }
+};
+
+static
+Address_t ADD_TABLE_DATA_READ[] = {
+ { 1242, "", 0, "" },
+ { 123, "", 0, "" },
+ { 456, "", 0, "" },
+ { 8976, "", 0, "" },
+ { 1122, "", 0, "" },
+ { 3211, "", 0, "" }
+};
+
+static
+Employee_t EMP_TABLE_DATA[] = {
+ { 1242, "Joe", "Dalton" , &ADD_TABLE_DATA[0] },
+ { 123, "Lucky", "Luke" , &ADD_TABLE_DATA[1] },
+ { 456, "Averell", "Dalton" , &ADD_TABLE_DATA[2] },
+ { 8976, "Gaston", "Lagaffe", &ADD_TABLE_DATA[3] },
+ { 1122, "Jolly", "Jumper" , &ADD_TABLE_DATA[4] },
+ { 3211, "Leffe", "Pagrotsky", &ADD_TABLE_DATA[5] },
+};
+
+static
+Employee_t EMP_TABLE_DATA_READ[] = {
+ { 1242, "", "", &ADD_TABLE_DATA_READ[0] },
+ { 123, "", "", &ADD_TABLE_DATA_READ[1] },
+ { 456, "", "", &ADD_TABLE_DATA_READ[2] },
+ { 8976, "", "", &ADD_TABLE_DATA_READ[3] },
+ { 1122, "", "", &ADD_TABLE_DATA_READ[4] },
+ { 3211, "", "", &ADD_TABLE_DATA_READ[5] }
+};
+
+static const char TABLE[] = "employee_address";
+
+static
+void
+DbCreate(void){
+
+ ndbout << "Opening database" << endl;
+ require( DBA_Open() == DBA_NO_ERROR );
+
+ ndbout << "Creating tables" << endl;
+ require( DBA_CreateTable( TABLE, NbCol, ColDesc ) == DBA_NO_ERROR );
+
+ ndbout << "Checking for table existance" << endl;
+ require( DBA_TableExists( TABLE ) );
+}
+
+static
+void
+CreateBindings(void){
+ ndbout << "Creating bindings" << endl;
+
+ Bind = DBA_CreateBinding(TABLE,
+ EmpBindingRows,
+ EmpBindings,
+ sizeof(Employee_t) );
+
+ require(Bind != 0);
+}
+
+extern "C" {
+ static void insertCallback( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void deleteCallback( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void updateCallback( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void readCallback ( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+ static void writeCallback ( DBA_ReqId_t, DBA_Error_t, DBA_ErrorCode_t );
+}
+
+static
+void BasicArray(){
+ ndbout << "Testing basic array operations" << endl;
+
+ // Basic insert
+ DBA_ArrayInsertRows(Bind, EMP_TABLE_DATA, Rows-2, insertCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ArrayReadRows (Bind, EMP_TABLE_DATA_READ, Rows-2, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows-2, ADD_TABLE_DATA_READ);
+
+ // Basic update
+ AlterRows (EMP_TABLE_DATA, Rows-2);
+ AlterRows (ADD_TABLE_DATA, Rows-2);
+ DBA_ArrayUpdateRows(Bind, EMP_TABLE_DATA, Rows-2, updateCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ArrayReadRows (Bind, EMP_TABLE_DATA_READ, Rows-2, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows-2, ADD_TABLE_DATA_READ);
+
+ // Basic write
+ AlterRows (EMP_TABLE_DATA, Rows);
+ AlterRows (ADD_TABLE_DATA, Rows);
+ DBA_ArrayWriteRows(Bind, EMP_TABLE_DATA, Rows, writeCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ArrayReadRows (Bind, EMP_TABLE_DATA_READ, Rows, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows, ADD_TABLE_DATA_READ);
+
+ // Basic delete
+ DBA_ArrayDeleteRows(Bind, EMP_TABLE_DATA, Rows, deleteCallback);
+ NdbSleep_SecSleep(1);
+}
+
+static
+void BasicPtr(){
+ ndbout << "Testing array of pointer operations" << endl;
+ Employee_t * EmpData[Rows];
+ Employee_t * EmpDataRead[Rows];
+ for(int i = 0; i<Rows; i++){
+ EmpData[i] = &EMP_TABLE_DATA[i];
+ EmpDataRead[i] = &EMP_TABLE_DATA_READ[i];
+ }
+
+ void * const * EMP_TABLE_DATA2 = (void * const *)EmpData;
+ void * const * EMP_TABLE_DATA_READ2 = (void * const *)EmpDataRead;
+
+ // Basic insert
+ DBA_InsertRows(Bind, EMP_TABLE_DATA2, Rows-2, insertCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ReadRows (Bind, EMP_TABLE_DATA_READ2, Rows-2, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows-2, ADD_TABLE_DATA_READ);
+
+ // Basic update
+ AlterRows (ADD_TABLE_DATA, Rows-2);
+ AlterRows (EMP_TABLE_DATA, Rows-2);
+ DBA_UpdateRows(Bind, EMP_TABLE_DATA2, Rows-2, updateCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ReadRows (Bind, EMP_TABLE_DATA_READ2, Rows-2, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows-2, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows-2, ADD_TABLE_DATA_READ);
+
+ // Basic write
+ AlterRows (ADD_TABLE_DATA, Rows);
+ AlterRows (EMP_TABLE_DATA, Rows);
+ DBA_WriteRows(Bind, EMP_TABLE_DATA2, Rows, writeCallback);
+ NdbSleep_SecSleep(1);
+ DBA_ReadRows (Bind, EMP_TABLE_DATA_READ2, Rows, readCallback);
+ NdbSleep_SecSleep(1);
+ CompareRows(EMP_TABLE_DATA, Rows, EMP_TABLE_DATA_READ);
+ CompareRows(ADD_TABLE_DATA, Rows, ADD_TABLE_DATA_READ);
+
+ // Basic delete
+ DBA_DeleteRows(Bind, EMP_TABLE_DATA2, Rows, deleteCallback);
+ NdbSleep_SecSleep(1);
+}
+
+/*---------------------------------------------------------------------------*/
+NDB_COMMAND(newton_pb, "newton_pb",
+ "newton_pb", "newton_pb", 65535){
+
+ DbCreate();
+ CreateBindings();
+
+ BasicArray();
+ BasicPtr();
+
+ DBA_Close();
+
+ return 0;
+}
+
+/**
+ * callbackStatusCheck checks whether or not the operation succeeded
+ */
+void
+callbackStatusCheck( DBA_Error_t status, const char* operation) {
+ ndbout_c("%s: %d", operation, status);
+}
+
+void insertCallback( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "insert");
+}
+void deleteCallback( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "delete");
+}
+void updateCallback( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "update");
+}
+void readCallback ( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "read");
+}
+void writeCallback ( DBA_ReqId_t, DBA_Error_t s, DBA_ErrorCode_t ){
+ callbackStatusCheck(s, "write");
+}
+
diff --git a/ndb/test/newtonapi/basic_test/too_basic.cpp b/ndb/test/newtonapi/basic_test/too_basic.cpp
new file mode 100644
index 00000000000..9099f0d9154
--- /dev/null
+++ b/ndb/test/newtonapi/basic_test/too_basic.cpp
@@ -0,0 +1,109 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+
+/****** THIS LINE IS 80 CHARACTERS WIDE - DO *NOT* EXCEED 80 CHARACTERS! ****/
+
+//#include <cfg/cfg_db.h>
+//#include <init/init_start_restart.h>
+//#include "pcn_types.h"
+//#include <testing/testing.h>
+
+extern "C" {
+#include <cfg_db.h>
+}
+
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+
+
+typedef struct Employee {
+
+ UInt32_t EmpNo;
+ char FirstName[22];
+ char LastName[22];
+
+} Employee_t;
+#define CHECK_DB_CALL( Call ) \
+ CheckDbCall( Call, #Call, __FILE__, __LINE__ )
+
+
+
+/* --- Exported functions --- */
+
+/*---------------------------------------------------------------------------*/
+int main() {
+
+
+ char EMP_TABLE_NAME[] = "employees";
+
+ Employee_t t;
+
+ CFG_DbColumnDesc_t EmpColDesc[] = {
+ { "first_name", CFG_DB_CHAR, PCN_SIZE_OF( Employee, FirstName ),
+ PCN_FALSE },
+ { "emp_no", CFG_DB_INT, PCN_SIZE_OF( Employee, EmpNo ), PCN_TRUE },
+ { "last_name", CFG_DB_CHAR, PCN_SIZE_OF( Employee, LastName ),
+ PCN_FALSE },
+ };
+
+ int EmpNbCol = 3;
+
+
+
+ CFG_DbColumnBinding_t ColBindings[] = {
+ CFG_DB_BINDING( "last_name", CFG_DB_CHAR, Employee, LastName ),
+ CFG_DB_BINDING( "emp_no", CFG_DB_INT, Employee, EmpNo ),
+ CFG_DB_BINDING( "first_name", CFG_DB_CHAR, Employee, FirstName)
+ };
+
+
+ Employee_t EMP_TABLE_DATA[] = {
+ { 1242, "Joe", "Dalton" },
+ { 123, "Lucky", "Luke" },
+ { 456, "Averell", "Dalton" },
+ { 8976, "Gaston", "Lagaffe" }
+ };
+
+
+ char* DbName;
+
+ DbName = NULL;
+
+
+ // On Linux: will destroy the table to start from a clean slate.
+
+ CFG_DbDestroy();
+ CFG_DbOpen( &DbName ) ;
+ CFG_DbCreateTable( EMP_TABLE_NAME,
+ EmpNbCol, EmpColDesc );
+
+ CFG_DbTableExists( EMP_TABLE_NAME );
+
+ //#ifndef CELLO_PLATFORM
+ //CHECK_DB_CALL( CFG_DbDumpSchema( stdout ) );
+ //#endif
+
+ CFG_DbClose();
+ // INIT_StopSystem();
+
+}
+
+
diff --git a/ndb/test/newtonapi/perf_test/Makefile b/ndb/test/newtonapi/perf_test/Makefile
new file mode 100644
index 00000000000..2be004d4277
--- /dev/null
+++ b/ndb/test/newtonapi/perf_test/Makefile
@@ -0,0 +1,14 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := newton_perf
+BIN_TARGET_LIBS :=
+BIN_TARGET_ARCHIVES := NEWTON_API
+SOURCES := perf.cpp
+
+CCFLAGS_LOC := -I$(call fixpath,$(NDB_TOP)/include/util) -I$(call fixpath,$(NDB_TOP)/include/newtonapi) -I$(call fixpath,$(NDB_TOP)/include/portlib)
+
+include $(NDB_TOP)/Epilogue.mk
+
+
diff --git a/ndb/test/newtonapi/perf_test/perf.cpp b/ndb/test/newtonapi/perf_test/perf.cpp
new file mode 100644
index 00000000000..81d4cc5fd08
--- /dev/null
+++ b/ndb/test/newtonapi/perf_test/perf.cpp
@@ -0,0 +1,646 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+extern "C" {
+#include <dba.h>
+}
+
+#include <stdlib.h>
+#include <string.h>
+#include <assert.h>
+
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbTimer.hpp>
+#include <NDBT_Stats.hpp>
+#include <NDBT_ReturnCodes.h>
+#include <NdbMain.h>
+#include <time.h>
+
+static const int NP_Insert = 0;
+static const int NP_Update = 1;
+static const int NP_WriteUpdate = 2;
+static const int NP_WriteInsert = 3;
+static const int NP_Delete = 4;
+static const int NP_BulkRead = 5;
+static const int NP_MAX = 5;
+
+static const char * Operations[] = {
+ "Insert ",
+ "Update ",
+ "WriteUpd",
+ "WriteIns",
+ "Delete ",
+ "BulkRead"
+};
+
+/**
+ * Configuration variables
+ */
+static int NoOfTransactions = 10000;
+static int ParallellTransactions = 1000;
+static int OperationsPerTransaction = 10;
+static int NoOfColumns = 20;
+static int BytesPerInsert = 300;
+static int BytesPerUpdate = 200;
+static int LoopCount = 10;
+
+/**
+ * Global variables
+ */
+static char TableName[255];
+static DBA_ColumnDesc_t * ColumnDescriptions;
+static DBA_ColumnBinding_t * InsertBindings;
+static DBA_ColumnBinding_t * UpdateBindings; static int UpdateBindingColumns;
+static DBA_ColumnBinding_t * DeleteBindings;
+
+static char * TestData;
+static DBA_Binding_t * InsertB;
+static DBA_Binding_t * UpdateB;
+static DBA_Binding_t * DeleteB;
+
+/**
+ * Function prototypes
+ */
+static void sequence(int loops);
+
+inline void * getPtr(int rowNo) { return TestData+rowNo*BytesPerInsert;}
+inline void setPK(int rowNo, int pk){ * (int *)getPtr(rowNo) = pk; }
+
+static void SetupTestData();
+static void CleanupTestData();
+
+static bool CreateTable();
+static bool CleanTable();
+static bool CreateBindings();
+
+static void usage();
+
+static
+void
+usage(){
+ int ForceSend, Interval;
+ DBA_GetParameter(0, &Interval);
+ DBA_GetParameter(3, &ForceSend);
+
+ ndbout << "newtonPerf" << endl
+ << " -n Transactions per loop and operation ("
+ << NoOfTransactions << ")" << endl
+ << " -p parallell transactions (" << ParallellTransactions << ")"
+ << endl
+ << " -o operations per transaction (" << OperationsPerTransaction
+ << ")" << endl
+ << " -a no of columns (" << NoOfColumns << ")" << endl
+ << " -b Table size in bytes (" << BytesPerInsert << ")" << endl
+ << " -u Bytes per update (" << BytesPerUpdate << ")" << endl
+ << " -l Loop count (" << LoopCount << ")" << endl
+ << " -i Interval (" << Interval << "ms)" << endl
+ << " -f Force send algorithm (" << ForceSend << ")" << endl
+ << " -h Help" << endl;
+
+}
+
+static
+bool
+parseArgs(int argc, const char **argv){
+ bool a = false, b = false, u = false;
+
+ for(int i = 1; i<argc; i++){
+ if(argv[i][0] != '-'){
+ ndbout << "Invalid argument: " << argv[i] << endl;
+ return false;
+ }
+
+ if(argv[i][1] == 'h')
+ return false;
+
+ if(i == argc-1){
+ ndbout << "Expecting argument to " << argv[i] << endl;
+ return false;
+ }
+
+ switch(argv[i][1]){
+ case 'n':
+ NoOfTransactions = atoi(argv[i+1]);
+ break;
+ case 'p':
+ ParallellTransactions = atoi(argv[i+1]);
+ break;
+ case 'o':
+ OperationsPerTransaction = atoi(argv[i+1]);
+ break;
+ case 'a':
+ NoOfColumns = atoi(argv[i+1]);
+ a = true;
+ break;
+ case 'b':
+ BytesPerInsert = atoi(argv[i+1]);
+ b = true;
+ break;
+ case 'u':
+ BytesPerUpdate = atoi(argv[i+1]);
+ u = true;
+ break;
+ case 'l':
+ LoopCount = atoi(argv[i+1]);
+ break;
+ case 'f':
+ {
+ const int val = atoi(argv[i+1]);
+ if(DBA_SetParameter(3, val) != DBA_NO_ERROR){
+ ndbout << "Invalid force send algorithm: "
+ << DBA_GetLatestErrorMsg()
+ << "(" << DBA_GetLatestError() << ")" << endl;
+ return false;
+ }
+ }
+ break;
+ case 'i':
+ {
+ const int val = atoi(argv[i+1]);
+ if(DBA_SetParameter(0, val) != DBA_NO_ERROR){
+ ndbout << "Invalid NBP interval: "
+ << DBA_GetLatestErrorMsg()
+ << "(" << DBA_GetLatestError() << ")" << endl;
+ return false;
+ }
+ }
+ break;
+ default:
+ ndbout << "Invalid option: " << argv[i] << endl;
+ return false;
+ }
+ i++;
+ }
+ if(a && !b) BytesPerInsert = 15 * NoOfColumns;
+ if(!a && b) NoOfColumns = ((BytesPerInsert + 14) / 15)+1;
+
+ if(!u)
+ BytesPerUpdate = (2 * BytesPerInsert) / 3;
+
+ bool t = true;
+ if(NoOfColumns < 2) t = false;
+ if(BytesPerInsert < 8) t = false;
+ if(BytesPerUpdate < 8) t = false;
+
+ if(!t){
+ ndbout << "Invalid arguments combination of -a -b -u not working out"
+ << endl;
+ return false;
+ }
+ return true;
+}
+
+NDB_COMMAND(newton_perf, "newton_perf",
+ "newton_perf", "newton_perf", 65535){
+
+ if(!parseArgs(argc, argv)){
+ usage();
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ ndbout << "-----------" << endl;
+ usage();
+ ndbout << endl;
+
+ SetupTestData();
+
+ DBA_Open();
+
+ if(!CreateTable()){
+ DBA_Close();
+ CleanupTestData();
+ return 0;
+ }
+
+ if(!CreateBindings()){
+ DBA_Close();
+ CleanupTestData();
+ return 0;
+ }
+
+ CleanTable();
+
+ sequence(LoopCount);
+
+ DBA_Close();
+ CleanupTestData();
+
+ DBA_DestroyBinding(InsertB);
+ DBA_DestroyBinding(UpdateB);
+ DBA_DestroyBinding(DeleteB);
+}
+
+static
+void
+ErrorMsg(const char * s){
+ ndbout << s
+ << ": " << DBA_GetLatestError() << "-" << DBA_GetLatestErrorMsg()
+ << ", " << DBA_GetLatestNdbError()
+ << endl;
+}
+
+static
+int
+m4(int i){
+ const int j = i - (i & 3);
+ return j;
+}
+
+static
+void
+SetupTestData(){
+ ndbout << "Creating testdata" << endl;
+
+ ColumnDescriptions = new DBA_ColumnDesc_t[NoOfColumns];
+ InsertBindings = new DBA_ColumnBinding_t[NoOfColumns];
+
+ const int sz = m4((BytesPerInsert - ((NoOfColumns+1)/2)*4)/(NoOfColumns/2));
+ int sum = 0;
+ UpdateBindingColumns = 0;
+ for(int i = 0; i<NoOfColumns; i++){
+ char tmp[16];
+ if((i % 2) == 0){
+ sprintf(tmp, "I%d", i);
+ ColumnDescriptions[i].DataType = DBA_INT;
+ ColumnDescriptions[i].Size = 4;
+ sum += 4;
+ } else {
+ sprintf(tmp, "S%d", i);
+ ColumnDescriptions[i].DataType = DBA_CHAR;
+ ColumnDescriptions[i].Size = sz;
+ sum += sz;
+ }
+ ColumnDescriptions[i].IsKey = 0;
+ ColumnDescriptions[i].Name = strdup(tmp);
+
+ InsertBindings[i].Name = strdup(tmp);
+ InsertBindings[i].DataType = ColumnDescriptions[i].DataType;
+ InsertBindings[i].Size = ColumnDescriptions[i].Size;
+ InsertBindings[i].Offset = sum - ColumnDescriptions[i].Size;
+ InsertBindings[i].Ptr = 0;
+
+ if(sum <= BytesPerUpdate)
+ UpdateBindingColumns++;
+ }
+ if(UpdateBindingColumns == 1)
+ UpdateBindingColumns++;
+
+ ColumnDescriptions[0].IsKey = 1;
+
+ assert(sum <= BytesPerInsert);
+ sprintf(TableName, "NEWTON_%d_%d", sum, NoOfColumns);
+
+ UpdateBindings = new DBA_ColumnBinding_t[UpdateBindingColumns];
+ memcpy(UpdateBindings, InsertBindings,
+ UpdateBindingColumns*sizeof(DBA_ColumnBinding_t));
+
+ DeleteBindings = new DBA_ColumnBinding_t[1];
+ memcpy(DeleteBindings, InsertBindings,
+ 1*sizeof(DBA_ColumnBinding_t));
+
+ TestData = (char *)malloc(NoOfTransactions *
+ OperationsPerTransaction * BytesPerInsert);
+
+ assert(TestData != 0);
+ for(int i = 0; i<NoOfTransactions; i++)
+ for(int j = 0; j<OperationsPerTransaction; j++){
+ const int pk = i * OperationsPerTransaction + j;
+ setPK(pk, pk);
+ }
+}
+
+static
+void
+CleanupTestData(){
+ free(TestData);
+ for(int i = 0; i<NoOfColumns; i++){
+ free((char*)ColumnDescriptions[i].Name);
+ free((char*)InsertBindings[i].Name);
+ }
+ delete [] ColumnDescriptions;
+ delete [] InsertBindings;
+ delete [] UpdateBindings;
+ delete [] DeleteBindings;
+}
+
+
+static bool CleanReturnValue = true;
+static int CleanCallbacks = 0;
+static int CleanRows = 0;
+
+extern "C"
+void
+CleanCallback(DBA_ReqId_t reqId, DBA_Error_t error, DBA_ErrorCode_t ec){
+ CleanCallbacks++;
+ if(error == DBA_NO_ERROR)
+ CleanRows++;
+}
+
+static
+bool
+CleanTable(){
+ ndbout << "Cleaning table..." << flush;
+ CleanReturnValue = true;
+ CleanCallbacks = 0;
+ CleanRows = 0;
+ for(int i = 0; i<NoOfTransactions * OperationsPerTransaction; i++){
+ DBA_ArrayDeleteRows(DeleteB,
+ getPtr(i), 1,
+ CleanCallback);
+ while((i-CleanCallbacks)>ParallellTransactions)
+ NdbSleep_MilliSleep(100);
+ }
+ while(CleanCallbacks != (NoOfTransactions * OperationsPerTransaction))
+ NdbSleep_SecSleep(1);
+
+ ndbout << CleanRows << " rows deleted" << endl;
+
+ return CleanReturnValue;
+}
+
+static
+bool
+CreateBindings(){
+ ndbout << "Creating bindings" << endl;
+ InsertB = UpdateB = DeleteB = 0;
+
+ InsertB = DBA_CreateBinding(TableName, NoOfColumns,
+ InsertBindings, BytesPerInsert);
+ if(InsertB == 0){
+ ErrorMsg("Failed to create insert bindings");
+ return false;
+ }
+
+ UpdateB = DBA_CreateBinding(TableName, UpdateBindingColumns,
+ UpdateBindings, BytesPerInsert);
+ if(UpdateB == 0){
+ ErrorMsg("Failed to create update bindings");
+ DBA_DestroyBinding(InsertB);
+ return false;
+ }
+
+ DeleteB = DBA_CreateBinding(TableName, 1,
+ DeleteBindings, BytesPerInsert);
+ if(DeleteB == 0){
+ ErrorMsg("Failed to create delete bindings");
+ DBA_DestroyBinding(InsertB);
+ DBA_DestroyBinding(UpdateB);
+ return false;
+ }
+ return true;
+}
+
+static
+bool
+CreateTable(){
+ ndbout << "Creating " << TableName << endl;
+ return DBA_CreateTable( TableName,
+ NoOfColumns,
+ ColumnDescriptions ) == DBA_NO_ERROR;
+}
+
+/**
+ *
+ */
+static NdbTimer SequenceTimer;
+
+static int CurrentOp = NP_Insert;
+static int SequenceSent = 0;
+static int SequenceRecv = 0;
+static NDBT_Stats SequenceStats[NP_MAX][4];
+static NDBT_Stats SequenceLatency[NP_MAX];
+
+static int HashMax;
+static DBA_ReqId_t * ReqHash; // ReqId - Latency/Row
+static int * ReqHashPos; // (row in StartTime)
+
+static int SequenceLatencyPos;
+static NDB_TICKS * StartTime;
+
+static
+inline
+int
+computeHashMax(int elements){
+ HashMax = 1;
+ while(HashMax < elements)
+ HashMax *= 2;
+
+ if(HashMax < 1024)
+ HashMax = 1024;
+
+ return HashMax;
+}
+
+static
+inline
+int
+hash(DBA_ReqId_t request){
+ int r = (request >> 2) & (HashMax-1);
+ return r;
+}
+
+static
+inline
+void
+addRequest(DBA_ReqId_t request, int pos){
+
+ int i = hash(request);
+
+ while(ReqHash[i] != 0)
+ i = ((i + 1) & (HashMax-1));
+
+ ReqHash[i] = request;
+ ReqHashPos[i] = pos;
+}
+
+static
+inline
+int
+getRequest(DBA_ReqId_t request){
+
+ int i = hash(request);
+
+ while(ReqHash[i] != request)
+ i = ((i + 1) & (HashMax-1));
+
+ ReqHash[i] = 0;
+
+ return ReqHashPos[i];
+}
+
+extern "C"
+void
+SequenceCallback(DBA_ReqId_t reqId, DBA_Error_t error, DBA_ErrorCode_t ec){
+ int p = getRequest(reqId) - 1;
+
+ if(error != DBA_NO_ERROR){
+ ndbout << "p = " << p << endl;
+ ndbout << "DBA_GetErrorMsg(" << error << ") = "
+ << DBA_GetErrorMsg(error) << endl;
+ ndbout << "DBA_GetNdbErrorMsg(" << ec << ") = "
+ << DBA_GetNdbErrorMsg(ec) << endl;
+
+ assert(error == DBA_NO_ERROR);
+ }
+
+ SequenceRecv++;
+ if(SequenceRecv == NoOfTransactions){
+ SequenceTimer.doStop();
+ }
+
+ if((p & 127) == 127){
+ NDB_TICKS t = NdbTick_CurrentMillisecond() - StartTime[p];
+ SequenceLatency[CurrentOp].addObservation(t);
+ }
+}
+
+typedef DBA_ReqId_t (* DBA_ArrayFunction)( const DBA_Binding_t* pBindings,
+ const void * pData,
+ int NbRows,
+ DBA_AsyncCallbackFn_t CbFunc );
+
+inline
+int
+min(int a, int b){
+ return a > b ? b : a;
+}
+
+static
+void
+SequenceOp(DBA_ArrayFunction func, const DBA_Binding_t* pBindings, int op){
+ SequenceSent = 0;
+ SequenceRecv = 0;
+ SequenceLatencyPos = 1;
+ CurrentOp = op;
+
+ SequenceTimer.doStart();
+ for(int i = 0; i<NoOfTransactions; ){
+ const int l1 = ParallellTransactions - (SequenceSent - SequenceRecv);
+ const int l2 = min(NoOfTransactions - i, l1);
+ for(int j = 0; j<l2; j++){
+ const DBA_ReqId_t r = func(pBindings,
+ getPtr(i*OperationsPerTransaction),
+ OperationsPerTransaction,
+ SequenceCallback);
+ assert(r != 0);
+ SequenceSent++;
+ addRequest(r, i + 1);
+ i++;
+
+ if((SequenceSent & 127) == 127){
+ NDB_TICKS t = NdbTick_CurrentMillisecond();
+ StartTime[i] = t;
+ }
+ }
+ if(l2 == 0)
+ NdbSleep_MilliSleep(10);
+ }
+
+ while(SequenceRecv != SequenceSent)
+ NdbSleep_SecSleep(1);
+
+ ndbout << "Performed " << NoOfTransactions << " " << Operations[op]
+ << " in ";
+
+ double p = NoOfTransactions * 1000;
+ double t = SequenceTimer.elapsedTime();
+ double o = p * OperationsPerTransaction;
+
+ p /= t;
+ o /= t;
+
+ int _p = p;
+ int _o = o;
+
+ double b = 0;
+
+ switch(op){
+ case NP_Insert:
+ case NP_WriteInsert:
+ case NP_WriteUpdate:
+ case NP_BulkRead:
+ b = BytesPerInsert;
+ break;
+ case NP_Update:
+ b = BytesPerUpdate;
+ break;
+ case NP_Delete:
+ b = 4;
+ break;
+ default:
+ b = 0;
+ }
+ b *= NoOfTransactions * OperationsPerTransaction;
+ b /= t;
+ int _b = b;
+
+ SequenceStats[op][0].addObservation(t);
+ SequenceStats[op][1].addObservation(p);
+ SequenceStats[op][2].addObservation(o);
+ SequenceStats[op][3].addObservation(b);
+
+ int t2 = SequenceStats[op][0].getMean();
+ int p2 = SequenceStats[op][1].getMean();
+ int o2 = SequenceStats[op][2].getMean();
+ int b2 = SequenceStats[op][3].getMean();
+
+ ndbout << SequenceTimer.elapsedTime() << "(" << t2 << ")ms";
+ ndbout << " -> " << _p << "(" << p2 << ") T/s - " << _o
+ << "(" << o2 << ") O/s - " << _b << "(" << b2 << ") Kb/s" << endl;
+
+ ndbout << " Latency (ms) Avg: " << (int)SequenceLatency[op].getMean()
+ << " min: " << (int)SequenceLatency[op].getMin()
+ << " max: " << (int)SequenceLatency[op].getMax()
+ << " stddev: " << (int)SequenceLatency[op].getStddev()
+ << " n: " << SequenceLatency[op].getCount() << endl;
+}
+
+/**
+ * Sequence
+ */
+static
+void
+sequence(int loops){
+ computeHashMax(ParallellTransactions);
+ ReqHash = new DBA_ReqId_t[HashMax];
+ ReqHashPos = new int[HashMax];
+ StartTime = new NDB_TICKS[NoOfTransactions];
+
+ for(int i = 0; i<NP_MAX; i++){
+ SequenceLatency[i].reset();
+ for(int j = 0; j<4; j++)
+ SequenceStats[i][j].reset();
+ }
+ for(int i = 0; i<loops; i++){
+ ndbout << "Loop #" << (i+1) << endl;
+ SequenceOp(DBA_ArrayInsertRows, InsertB, NP_Insert);
+
+ // BulkRead
+
+ SequenceOp(DBA_ArrayUpdateRows, UpdateB, NP_Update);
+ SequenceOp(DBA_ArrayWriteRows, InsertB, NP_WriteUpdate);
+ SequenceOp(DBA_ArrayDeleteRows, DeleteB, NP_Delete);
+ SequenceOp(DBA_ArrayWriteRows, InsertB, NP_WriteInsert);
+ SequenceOp(DBA_ArrayDeleteRows, DeleteB, NP_Delete);
+ ndbout << "-------------------" << endl << endl;
+ }
+
+ delete [] ReqHash;
+ delete [] ReqHashPos;
+ delete [] StartTime;
+}
diff --git a/ndb/test/odbc/Makefile b/ndb/test/odbc/Makefile
new file mode 100644
index 00000000000..eab61ebd359
--- /dev/null
+++ b/ndb/test/odbc/Makefile
@@ -0,0 +1,13 @@
+include .defs.mk
+
+DIRS += driver
+
+ifneq ($(findstring odbc, $(wildcard /usr/lib/libodbc.so)),)
+DIRS += dm-unixodbc
+endif
+
+ifneq ($(findstring $(NDB_OS), SOLARIS),)
+DIRS += dm-iodbc
+endif
+
+include ${NDB_TOP}/Epilogue.mk
diff --git a/ndb/test/odbc/SQL99_test/Makefile b/ndb/test/odbc/SQL99_test/Makefile
new file mode 100644
index 00000000000..3ac06016670
--- /dev/null
+++ b/ndb/test/odbc/SQL99_test/Makefile
@@ -0,0 +1,26 @@
+include .defs.mk
+
+TYPE = odbcdriver
+
+BIN_TARGET = SQL99_test
+
+SOURCES = SQL99_test.cpp
+
+CCFLAGS_LOC += -I/usr/local/include \
+ -I$(NDB_TOP)/test/include \
+ -I$(NDB_TOP)/include \
+ -I$(NDB_TOP)/src/client/odbc/common
+
+
+CCFLAGS_WARNINGS += -Wno-unused
+
+LIBS_SPEC += \
+ -lNDBT \
+ -lodbc \
+ -lodbcinst \
+ -lportlib
+
+
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/odbc/SQL99_test/SQL99_test.cpp b/ndb/test/odbc/SQL99_test/SQL99_test.cpp
new file mode 100644
index 00000000000..eda9ff33834
--- /dev/null
+++ b/ndb/test/odbc/SQL99_test/SQL99_test.cpp
@@ -0,0 +1,2138 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+// ODBC.cpp : Defines the entry point for the console application.
+//
+
+#include "SQL99_test.h"
+#include <iostream> // Loose later
+
+using namespace std; //
+
+#define MAXCOL 64
+#define DEFCOL 4
+
+#define MAXROW 64
+#define DEFROW 8
+
+#define MAXTHREADS 24
+#define DEFTHREADS 2
+
+#define MAXTABLES 16
+#define DEFTABLES 2
+
+#define MAXLOOPS 100000
+#define DEFLOOPS 4
+
+#define UPDATE_VALUE 7
+
+#define PKSIZE 2
+
+
+static int nNoOfThreads = 1 ;
+static int nNoOfCol = 4 ;
+static int nNoOfRows = 2 ;
+static int nNoOfLoops = 0 ;
+static int nNoOfTables = 2 ;
+static int nAPI = 0 ;
+static int tAttributeSize = sizeof(char) ;
+static attr_type AttributeType = T_CHAR ;
+static int nAggregate = 0 ;
+static int nArithmetic = 0 ;
+static int nPrint = 0 ;
+static int nColList = 0 ;
+static char szColNames[MAXCOL*MAX_COL_NAME] = { 0 } ;
+int createTables(char* szTableName, int nTables) ;
+
+
+/*************************************************
+Function: main - the entry point
+*************************************************/
+int main(int argc, char* argv[]){
+
+ int nRetrunValue = NDBT_FAILED ;
+ SQLRETURN rc = SQL_ERROR ;
+ double dResultA = 0 ;
+ double dResultB = 0 ;
+ double dInput = 0 ;
+ int x = 0, y = 0 ;
+ int* pIntRefBuffer = NULL ;
+ float* pFloatRefBuffer = NULL ;
+ double* pDoubleRefBuffer = NULL ;
+ char* pCharRefBuffer = NULL ;
+ char szColBuffer[MAX_COL_NAME] = { 0 } ;
+
+
+ ParseArguments(argc, (const char**)argv) ;
+
+ PARAMS* pparams = (PARAMS*)malloc(sizeof(PARAMS)*nNoOfThreads) ;
+ memset(pparams, 0, (sizeof(PARAMS)*nNoOfThreads)) ;
+
+ char* szTableNames = (char*)malloc(sizeof(char)*nNoOfTables*MAX_TABLE_NAME) ;
+ memset(szTableNames, 0, sizeof(char)*nNoOfTables*MAX_TABLE_NAME) ;
+
+ UintPtr pThreadHandles[MAXTHREADS] = { NULL } ;
+
+ AssignTableNames(szTableNames, nNoOfTables) ;
+
+ if(nAPI){
+ if(0 != createTables(szTableNames, nNoOfTables)){
+ printf("Failed to create tables through NDB API; quitting...\n") ;
+ NDBT_ProgramExit(NDBT_FAILED) ;
+ return NDBT_FAILED ;
+ }
+ }else{
+
+ //CreateDemoTables(szTableNames, nNoOfTables, DROP) ;
+ rc = CreateDemoTables(szTableNames, nNoOfTables, CREATE) ;
+ if(!(SQL_SUCCESS == rc || SQL_SUCCESS_WITH_INFO == rc)){
+ printf("Failed to create tables, quiting now.\n") ;
+ NDBT_ProgramExit(NDBT_FAILED) ;
+ return NDBT_FAILED ;
+ }
+ }
+
+ // Store column names in the buffer for use in some stmts
+ int k = 0 ;
+ for(;;){
+ memset((char*)szColBuffer, 0, strlen(szColBuffer)) ;
+ sprintf((char*)szColBuffer, "COL%d", k) ;
+ strcat((char*)szColNames, (char*)szColBuffer) ;
+ ++k ;
+ if( k == nNoOfCol ){
+ break ;
+ }
+ strcat((char*)szColNames, ", ") ;
+ } // for
+
+
+ switch(AttributeType){
+ case T_INT:
+ pIntRefBuffer = (int*)malloc(sizeof(int)*nNoOfRows*nNoOfCol*nNoOfThreads) ;
+ memset(pIntRefBuffer, 0, sizeof(int)*nNoOfRows*nNoOfCol*nNoOfThreads) ;
+ AssignRefNumValues(pIntRefBuffer, T_INT, nPrint) ;
+ StartThreads(pparams, (void*)pIntRefBuffer, nNoOfTables, szTableNames, AttributeType, pThreadHandles) ;
+ break ;
+ case T_FLOAT:
+ pFloatRefBuffer = (float*)malloc(sizeof(float)*nNoOfRows*nNoOfCol*nNoOfThreads) ;
+ memset(pFloatRefBuffer, 0, sizeof(float)*nNoOfRows*nNoOfCol*nNoOfThreads) ;
+ AssignRefNumValues(pFloatRefBuffer, T_FLOAT, nPrint) ;
+ StartThreads(pparams, (void*)pFloatRefBuffer, nNoOfTables, szTableNames, AttributeType, pThreadHandles) ;
+ break ;
+/* case T_DOUBLE:
+ pDoubleRefBuffer = (double*)malloc(sizeof(double)*nNoOfRows*nNoOfCol*nNoOfThreads) ;
+ memset(pDoubleRefBuffer, 0, sizeof(double)*nNoOfRows*nNoOfCol*nNoOfThreads) ;
+ AssignRefNumValues(pDoubleRefBuffer, T_DOUBLE, 0) ;
+ StartThreads(pparams, (void*)pDoubleRefBuffer, nNoOfTables, szTableNames, AttributeType, pThreadHandles) ;
+ break ;
+*/
+ case T_CHAR:
+ pCharRefBuffer = (char*)malloc(sizeof(char)*nNoOfRows*nNoOfCol*nNoOfThreads*MAX_CHAR_ATTR_LEN) ;
+ memset(pCharRefBuffer, 0, sizeof(char)*nNoOfRows*nNoOfCol*nNoOfThreads*MAX_CHAR_ATTR_LEN) ;
+ AssignRefCharValues(pCharRefBuffer, nPrint ) ;
+ StartThreads(pparams, (void*)pCharRefBuffer, nNoOfTables, szTableNames, AttributeType, pThreadHandles) ;
+ break ;
+ default:
+ break ;
+ }
+
+ NdbThread_SetConcurrencyLevel(nNoOfThreads + 2) ;
+
+
+ printf("\nPerforming inserts...") ;
+ SetThreadOperationType(pparams, T_INSERT) ;
+ if(0 < WaitForThreads(pparams)){
+ printf("\t\t%d thread(s) failed\n") ;
+ }else{
+ printf("\t\tdone\n") ;
+ }
+ printf("----------------------\n\n") ;
+ PrintAll(szTableNames, nNoOfTables, AttributeType) ;
+
+ printf("\nVerifying inserts...") ;
+ SetThreadOperationType(pparams, T_READ_VERIFY) ;
+ if(0 < WaitForThreads(pparams)){
+ printf("\t\t%d thread(s) failed\n") ;
+ }else{
+ printf("\t\tdone\n") ;
+ }
+ printf("----------------------\n\n") ;
+
+ printf("\nPerforming updates...") ;
+ SetThreadOperationType(pparams, T_UPDATE) ;
+ if(0 < WaitForThreads(pparams)){
+ printf("\t\t%d thread(s) failed\n") ;
+ }else{
+ printf("\t\tdone\n") ;
+ }
+ printf("----------------------\n\n") ;
+ //PrintAll(szTableNames, nNoOfTables, AttributeType) ;
+
+ printf("\nVerifying updates...") ;
+ SetThreadOperationType(pparams, T_READ_VERIFY) ;
+ if(0 < WaitForThreads(pparams)){
+ printf("\t\t%d thread(s) failed\n") ;
+ }else{
+ printf("\t\tdone\n") ;
+ }
+ printf("----------------------\n\n") ;
+
+ printf("\nPerforming reads...") ;
+ SetThreadOperationType(pparams, T_READ) ;
+ if(0 < WaitForThreads(pparams)){
+ printf("\t\t%d thread(s) failed\n") ;
+ }else{
+ printf("\t\tdone\n") ;
+ }
+ printf("----------------------\n\n") ;
+ PrintAll(szTableNames, nNoOfTables, AttributeType) ;
+
+
+ if(T_CHAR != AttributeType && nAggregate){
+ printf("\nTesting aggregate functions for each table\n\n") ;
+ printf("FN\tCOLUMN\tVALUE\t\t\tTOTAL ROWS WHERE\n\t\t\t\t\tVALUE(S) > VALUE\n--------------------------------------------------------\n\n") ;
+
+ for(y = 0 ; y < nNoOfTables ; ++y){
+ for(x = 0; x < nNoOfCol ; ++x){
+ dResultA = dResultB = 0 ;
+ AggregateFn(FN_MIN, (char*)(szTableNames + MAX_TABLE_NAME*y), x, NULL, &dResultA, AttributeType) ;
+ AggregateFn(FN_COUNT, (char*)(szTableNames + MAX_TABLE_NAME*y) , x, &dResultA, &dResultB, AttributeType) ;
+ ATTR_TYPE_SWITCH_AGR("MIN", x, dResultA, dResultB, AttributeType) ;
+ }
+ }
+
+ for(y = 0; y < nNoOfTables ; ++y){
+ for(x = 0; x < nNoOfCol ; ++x){
+ dResultA = dResultB = 0 ;
+ AggregateFn(FN_MAX, (char*)(szTableNames + MAX_TABLE_NAME*y), x, NULL, &dResultA, AttributeType) ;
+ AggregateFn(FN_COUNT, (char*)(szTableNames + MAX_TABLE_NAME*y), x, &dResultA, &dResultB, AttributeType) ;
+ ATTR_TYPE_SWITCH_AGR("MAX", x, dResultA, dResultB, AttributeType) ;
+ }
+ }
+
+ for(y = 0 ; y < nNoOfTables ; ++y){
+ for(x = 0; x < nNoOfCol ; ++x){
+ dResultA = dResultB = 0 ;
+ AggregateFn(FN_AVG, (char*)(szTableNames + MAX_TABLE_NAME*y), x, NULL, &dResultA, AttributeType) ;
+ AggregateFn(FN_COUNT, (char*)(szTableNames + MAX_TABLE_NAME*y), x, &dResultA, &dResultB, AttributeType) ;
+ ATTR_TYPE_SWITCH_AGR("AVG", x, dResultA, dResultB, AttributeType)
+ }
+ }
+
+ printf("--------------------------------------------------------\n\n") ;
+ }
+
+ if(T_CHAR != AttributeType && nArithmetic){
+
+ float nVal = (rand() % 10) /1.82342 ;
+
+ for(int h = 0 ; h < nNoOfTables ; ++h){
+
+ printf("\nTesting arithmetic operators\nfor each column in %s:\n----------------------\n", (char*)(szTableNames + MAX_TABLE_NAME*sizeof(char)*h) ) ;
+
+ printf("\nOperator [ * ]... \t\t") ;
+ ArithOp((char*)(szTableNames + MAX_TABLE_NAME*sizeof(char)*h), nNoOfCol, &nVal, AttributeType, MULTI) ;
+ printf("done\n") ;
+
+ printf("\nOperator [ / ]... \t\t") ;
+ ArithOp((char*)(szTableNames + MAX_TABLE_NAME*sizeof(char)*h), nNoOfCol, &nVal, AttributeType, DIVIDE) ;
+ printf("done\n") ;
+
+ printf("\nOperator [ + ]... \t\t") ;
+ ArithOp((char*)(szTableNames + MAX_TABLE_NAME*sizeof(char)*h), nNoOfCol, &nVal, AttributeType, PLUS) ;
+ printf("done\n") ;
+
+ printf("\nOperator [ - ]... \t\t") ;
+ ArithOp((char*)(szTableNames + MAX_TABLE_NAME*sizeof(char)*h), nNoOfCol, &nVal, AttributeType, MINUS) ;
+ printf("done\n\n") ;
+ /*
+ printf("\nOperator [ % ]... \t\t") ;
+ ArithOp((char*)szTableNames, nNoOfCol, &nVal, AttributeType, MODULO) ;
+ printf("done\n\n") ;
+ */
+ }
+ }
+/*
+ printf("\nPerforming deletes...") ;
+ SetThreadOperationType(pparams, T_DELETE) ;
+ if(0 < WaitForThreads(pparams)){
+ printf("\t\t%d thread(s) failed\n") ;
+ }else{
+ printf("\t\tdone\n") ;
+ }
+ printf("----------------------\n\n") ;
+
+ printf("\nVerifying deletes...") ;
+ SetThreadOperationType(pparams, T_DELETE_VERIFY) ;
+ if(0 < WaitForThreads(pparams)){
+ printf("\t\t%d thread(s) failed\n") ;
+ }else{
+ printf("\t\tdone\n") ;
+ }
+ printf("----------------------\n\n") ;
+*/
+ StopThreads(pparams, pThreadHandles) ;
+
+ //PrintAll(szTableNames, nNoOfTables, AttributeType) ;
+
+ //CreateDemoTables(szTableNames, nNoOfTables, DROP) ;
+
+ free((void*)szTableNames) ;
+ free((void*)pparams) ;
+ free((void*)pIntRefBuffer) ;
+ free((void*)pFloatRefBuffer) ;
+ free((void*)pDoubleRefBuffer) ;
+ free((void*)pCharRefBuffer) ;
+
+ return 0;
+}
+
+
+
+/**************************************************
+Function: ParseArguments
+***************************************************/
+void ParseArguments(int argc, const char** argv){
+
+ int i = 1;
+
+ while (argc > 1){
+
+ if (strcmp(argv[i], "-t") == 0)
+ {
+ nNoOfThreads = atoi(argv[i+1]);
+ if ((nNoOfThreads < 1) || (nNoOfThreads > MAXTHREADS))
+ nNoOfThreads = DEFTHREADS ;
+ }
+ else if (strcmp(argv[i], "-c") == 0)
+ {
+ nNoOfCol = atoi(argv[i+1]);
+ if ((nNoOfCol < 2) || (nNoOfCol > MAXCOL))
+ nNoOfCol = DEFCOL ;
+ }
+ else if (strcmp(argv[i], "-l") == 0)
+ {
+ nNoOfLoops = atoi(argv[i+1]);
+ if ((nNoOfLoops < 0) || (nNoOfLoops > MAXLOOPS))
+ nNoOfLoops = DEFLOOPS ;
+ }
+ else if (strcmp(argv[i], "-r") == 0)
+ {
+ nNoOfRows = atoi(argv[i+1]);;
+ if ((nNoOfRows < 0) || (nNoOfRows > MAXROW))
+ nNoOfRows = DEFROW ;
+ }
+ else if (strcmp(argv[i], "-m") == 0)
+ {
+ nArithmetic = 1 ;
+ argc++ ;
+ i-- ;
+ }
+ else if (strcmp(argv[i], "-g") == 0)
+ {
+ nAggregate = 1 ;
+ argc++ ;
+ i-- ;
+ }
+ else if (strcmp(argv[i], "-n") == 0)
+ {
+ nAPI = 1 ;
+ argc++ ;
+ i-- ;
+ }
+ else if (strcmp(argv[i], "-v") == 0)
+ {
+ nPrint = 1 ;
+ argc++ ;
+ i-- ;
+ }
+ else if (strcmp(argv[i], "-a") == 0)
+ {
+ if(strcmp(argv[i+1], "int") == 0){
+ AttributeType = T_INT ;
+ tAttributeSize = 32 ;
+ }else if(strcmp(argv[i+1], "float") == 0){
+ AttributeType = T_FLOAT ;
+ tAttributeSize = 64 ;
+ }else if(strcmp(argv[i+1], "char") == 0){
+ AttributeType = T_CHAR ;
+ }
+ }
+ else
+ {
+ cout << "Arguments:\n";
+ cout << "-n Create tables using NDB API (vs ODBC by default)" << endl;
+ cout << "-t Number of threads; maximum 24, default 2\n" << endl;
+ cout << "-c Number of columns per table; maximum 64, default 4\n" << endl;
+ cout << "-r Number of rows; maximum 64, default 8\n" << endl;
+ cout << "-a Type of attribute to use: int, double or char; default int " << endl;
+ cout << "-g Test aggregate functions" << endl;
+ cout << "-m Test arithmetic operators" << endl;
+ cout << "-v Print executed statements" << endl;
+ exit(-1);
+ }
+
+ argc -= 2 ;
+ i = i + 2 ;
+ }
+
+char *szAttrType[MAX_STR_LEN] = { 0 } ;
+switch(AttributeType){
+ case T_INT:
+ strcpy((char*)szAttrType, "Integer") ;
+ break ;
+ case T_FLOAT:
+ strcpy((char*)szAttrType, "Float") ;
+ break ;
+/* case T_DOUBLE:
+ strcpy((char*)szAttrType, "Double") ;
+ break ;
+*/
+ case T_CHAR:
+ strcpy((char*)szAttrType, "Character") ;
+ break ;
+ default:
+ strcpy((char*)szAttrType, "Not defined") ;
+ break ;
+ }
+
+
+printf("\n\nCurrent parameters: %d thread(s), %d tables, %d rows, %d colums, attribute type: %s\n\n", nNoOfThreads, nNoOfTables, nNoOfRows, nNoOfCol, szAttrType) ;
+ }
+
+
+
+
+/*************************************************
+Function: ThreadFnInt - thread function
+for int attributes
+*************************************************/
+void* ThreadFnInt(void* pparams){
+
+ SQLRETURN retcode = SQL_ERROR ;
+ SQLCHAR szStmtBuffer[MAX_SQL_STMT] = { 0 } ;
+ SQLCHAR szValueBuffer[MAX_VALUE_LEN] = { 0 } ;
+ SQLCHAR szAuxBuffer[MAX_STR_LEN] = { 0 } ;
+ SQLCHAR szColBuffer[MAX_COL_NAME] = { 0 } ;
+ SQLINTEGER cbInt = 0 ;
+ ODBC_HANDLES stHandles ;
+ memset(&stHandles, 0, sizeof(ODBC_HANDLES)) ;
+
+ int r = 0, j = 0 ;
+ //Get thread parameters
+ PARAMS* p = (PARAMS*)pparams ;
+ int* pRef = (int*)p->pThreadRef ;
+
+ int* pBindBuffer = (int*)malloc(sizeof(int)*nNoOfCol) ;
+
+ //printf("Thread #%d\n", p->nThreadID) ;
+
+ retcode = GetHandles(&stHandles, GET, 0) ;
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode ) {
+ p->report_status = S_STARTED ;
+ }else{
+ printf("Thread #%d failed to allocate handles, exiting now.\n", p->nThreadID) ;
+ free((void*)pBindBuffer) ;
+ p->nError = 1 ;
+ p->report_status = S_EXIT ;
+ return 0 ;
+ }
+
+ //p->report_status = S_STARTED ;
+
+ //Main thread loop
+ for(;;){
+
+ while(S_IDLE == p->thread_status){
+ NdbSleep_MilliSleep(1) ;
+ }
+
+ if(S_STOP == p->thread_status) {
+ break ;
+ }else{
+ p->thread_status = S_BUSY ;
+ }
+
+ switch(p->op_type){
+
+
+ /************************************** T_INSERT case **************************************/
+ case T_INSERT:
+
+ for(r = 0 ; r < nNoOfRows ; ++r){
+
+ if(!nColList){
+ sprintf((char*)szStmtBuffer, "INSERT INTO %s VALUES(", p->szTableName) ;
+ }else{
+ sprintf((char*)szStmtBuffer, "INSERT INTO %s (%s) VALUES(", p->szTableName, szColNames) ;
+ }
+
+ //sprintf((char*)szStmtBuffer, "INSERT INTO %s VALUES(", p->szTableName) ;
+
+ for(j = 0 ;;){
+ sprintf((char*)szValueBuffer,"%d", pRef[nNoOfCol*r + j]) ;
+ strncat((char*)szStmtBuffer, (char*)szValueBuffer, strlen((char*)szValueBuffer)) ;
+ ++j ;
+ if(nNoOfCol == j) break ;
+ strcat((char*)szStmtBuffer, ", ") ;
+ }
+ strcat((char*)szStmtBuffer, ")") ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS) ;
+ if(SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ }else{
+ p->nError = 1 ;
+ printf("INSERT in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ }
+ break ;
+
+
+ /************************************** T_READ case **************************************/
+ case T_READ:
+
+ for(r = 0 ; r < nNoOfRows ; r++){
+
+ sprintf((char*)szStmtBuffer, "SELECT * FROM %s WHERE COL0 = %d", p->szTableName, pRef[nNoOfCol*r]) ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ ODBC_FN(SQLExecDirect(stHandles.hstmt, (SQLCHAR*)szStmtBuffer, SQL_NTS), retcode) ;
+
+ for(j = 0 ; j < nNoOfCol ; ++j){
+ ODBC_FN(SQLBindCol(stHandles.hstmt, (j+1), SQL_C_SLONG, (void*)&pBindBuffer[j], sizeof(SQLINTEGER), &cbInt), retcode) ;
+ }
+
+ for (;;) {
+ retcode = SQLFetch(stHandles.hstmt);
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ for(int k = 0 ; k < nNoOfCol ; ++k){
+ if(p->nVerifyFlag){
+ if(pBindBuffer[k] != pRef[nNoOfCol*r + k])
+ printf("Expected: %d Actual: %d\n", pBindBuffer[k], pRef[nNoOfCol*r + k]) ;
+ }
+ }
+ }else if(SQL_NO_DATA == retcode){
+ break ;
+ }else{
+ p->nError = 1 ;
+ printf("READ in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ //printf("\n") ;
+ }
+ SQLCloseCursor(stHandles.hstmt) ;
+ }
+ break ;
+
+
+ /************************************** T_UPDATE case **************************************/
+ case T_UPDATE:
+ for(r = 0 ; r < nNoOfRows ; ++r){
+
+ sprintf((char*)szStmtBuffer, "UPDATE %s SET ", p->szTableName) ;
+ for(j = 1 ;;){
+ pRef[nNoOfCol*r + j] = pRef[nNoOfCol*r + j] + UPDATE_VALUE ;
+ sprintf((char*)szColBuffer,"COL%d = %d", j, pRef[nNoOfCol*r + j]) ;
+ strncat((char*)szStmtBuffer, (char*)szColBuffer, strlen((char*)szColBuffer)) ;
+ ++j ;
+ if(nNoOfCol == j) break ;
+ strcat((char*)szStmtBuffer, ", ") ;
+ }
+ sprintf((char*)szAuxBuffer, " WHERE COL0 = %d ;", pRef[nNoOfCol*r]) ;
+ strcat((char*)szStmtBuffer, (char*)szAuxBuffer);
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS) ;
+
+ if(SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ }else{
+ p->nError = 1 ;
+ printf("UPDATE in thread %d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ }
+ break ;
+
+
+ /************************************** T_DELETE case **************************************/
+ case T_DELETE:
+ for(r = 0 ; r < nNoOfRows ; ++r){
+ sprintf((char*)szStmtBuffer, "DELETE * FROM %s WHERE COL0 = %d", p->szTableName, pRef[nNoOfCol*r]) ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS) ;
+ if(SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ }else if( 1 == p->nVerifyFlag && SQL_NO_DATA != retcode){
+ p->nError = 1 ;
+ printf("\nVerification failed: the row found\n") ;
+ }else{
+ p->nError = 1 ;
+ printf("INSERT in thread %d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+
+ }
+ break ;
+
+
+ /************************************** default case **************************************/
+ default:
+ break ;
+ }//switch
+p->thread_status = S_IDLE ;
+ } //for
+
+free((void*)pBindBuffer) ;
+GetHandles(&stHandles, FREE, 0) ;
+p->thread_status = S_EXIT ;
+return 0 ;
+ };
+
+
+
+/*************************************************
+Function: ThreadFnFloat - thread function
+for float attributes
+*************************************************/
+void* ThreadFnFloat(void* pparams){
+
+ SQLRETURN retcode = SQL_ERROR ;
+ SQLCHAR szStmtBuffer[MAX_SQL_STMT] = { 0 } ;
+ SQLCHAR szValueBuffer[MAX_VALUE_LEN] = { 0 } ;
+ SQLCHAR szAuxBuffer[MAX_STR_LEN] = { 0 } ;
+ SQLCHAR szColBuffer[MAX_COL_NAME] = { 0 } ;
+ SQLINTEGER cbFloat = 0 ;
+ ODBC_HANDLES stHandles ;
+ memset(&stHandles, 0, sizeof(ODBC_HANDLES)) ;
+
+ int r = 0, j = 0 ;
+ //Get thread parameters
+ PARAMS* p = (PARAMS*)pparams ;
+
+ float* pRef = (float*)p->pThreadRef ;
+ float* pBindBuffer = (float*)malloc(sizeof(float)*nNoOfCol) ;
+
+ //printf("Thread #%d\n", p->nThreadID) ;
+
+ retcode = GetHandles(&stHandles, GET, 0) ;
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode ) {
+ p->report_status = S_STARTED ;
+ }else{
+ printf("Thread #%d failed to allocate handles, exiting now.\n", p->nThreadID) ;
+ free((void*)pBindBuffer) ;
+ p->nError = 1 ;
+ p->report_status = S_EXIT ;
+ return 0 ;
+ }
+
+ //p->report_status = S_STARTED ;
+
+ //Main thread loop
+ for(;;){
+
+ while(S_IDLE == p->thread_status){
+ NdbSleep_MilliSleep(1) ;
+ }
+
+ if(S_STOP == p->thread_status) {
+ break ;
+ }else{
+ p->thread_status = S_BUSY ;
+ }
+
+ switch(p->op_type){
+
+
+ /************************************** T_INSERT case **************************************/
+ case T_INSERT:
+
+ for(r = 0 ; r < nNoOfRows ; ++r){
+
+ if(!nColList){
+ sprintf((char*)szStmtBuffer, "INSERT INTO %s VALUES(", p->szTableName) ;
+ }else{
+ sprintf((char*)szStmtBuffer, "INSERT INTO %s (%s) VALUES(", p->szTableName, szColNames) ;
+ }
+
+ //sprintf((char*)szStmtBuffer, "INSERT INTO %s VALUES(", p->szTableName) ;
+
+ for(j = 0 ;;){
+ sprintf((char*)szValueBuffer,"%f", pRef[nNoOfCol*r + j]) ;
+ strncat((char*)szStmtBuffer, (char*)szValueBuffer, strlen((const char*)szValueBuffer)) ;
+ ++j ;
+ if(nNoOfCol == j) break ;
+ strcat((char*)szStmtBuffer, ", ") ;
+ }
+ strcat((char*)szStmtBuffer, ")") ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS) ;
+ if(SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ }else{
+ p->nError = 1 ;
+ printf("INSERT in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ }
+ break ;
+
+
+ /************************************** T_READ case **************************************/
+ case T_READ:
+
+ for(r = 0 ; r < nNoOfRows ; ++r){
+
+ sprintf((char*)szStmtBuffer, "SELECT * FROM %s WHERE COL0 = %f", p->szTableName, pRef[nNoOfCol*r]) ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ ODBC_FN(SQLExecDirect(stHandles.hstmt, (SQLCHAR*)szStmtBuffer, SQL_NTS), retcode) ;
+
+ for(j = 0 ; j < nNoOfCol ; ++j){
+ ODBC_FN(SQLBindCol(stHandles.hstmt, (j+1), SQL_C_FLOAT, (void*)&pBindBuffer[j], sizeof(SQLFLOAT), &cbFloat), retcode) ;
+ }
+
+ for (;;) {
+ retcode = SQLFetch(stHandles.hstmt);
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ for(int k = 0 ; k < nNoOfCol ; ++k){
+ if(p->nVerifyFlag){
+ if(abs(pBindBuffer[k] - pRef[nNoOfCol*r + k]) > FLTDEV )
+ printf("Expected: %f Actual: %f\n", pBindBuffer[k], pRef[nNoOfCol*r + k]) ;
+ }
+ }
+ }else if(SQL_NO_DATA == retcode){
+ break ;
+ }else{
+ p->nError = 1 ;
+ printf("READ in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ //printf("\n") ;
+ }
+ SQLCloseCursor(stHandles.hstmt) ;
+ }
+ break ;
+
+
+ /************************************** T_UPDATE case **************************************/
+ case T_UPDATE:
+ for(r = 0 ; r < nNoOfRows ; ++r){
+
+ sprintf((char*)szStmtBuffer, "UPDATE %s SET ", p->szTableName) ;
+ for(j = 1 ;;){
+ pRef[nNoOfCol*r + j] = pRef[nNoOfCol*r + j] + UPDATE_VALUE ;
+ sprintf((char*)szColBuffer,"COL%d = %f", j, pRef[nNoOfCol*r + j]) ;
+ strncat((char*)szStmtBuffer, (char*)szColBuffer, strlen((char*)szColBuffer)) ;
+ ++j ;
+ if(nNoOfCol == j) break ;
+ strcat((char*)szStmtBuffer, ", ") ;
+ }
+ sprintf((char*)szAuxBuffer, " WHERE COL0 = %f ;", pRef[nNoOfCol*r]) ;
+ strcat((char*)szStmtBuffer, (char*)szAuxBuffer);
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS) ;
+ if(SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ }else{
+ p->nError = 1 ;
+ printf("UPDATE in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ }
+ break ;
+
+
+ /************************************** T_DELETE case **************************************/
+ case T_DELETE:
+ for(r = 0 ; r < nNoOfRows ; ++r){
+ sprintf((char*)szStmtBuffer, "DELETE * FROM %s WHERE COL0 = %f", p->szTableName, pRef[nNoOfCol*r]) ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, (SQLCHAR*)szStmtBuffer, SQL_NTS) ;
+ if(SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ }else if( 1 == p->nVerifyFlag && SQL_NO_DATA != retcode){
+ p->nError = 1 ;
+ printf("\nVerification failed: still row exists\n") ;
+ }else{
+ p->nError = 1 ;
+ printf("DELETE in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ }
+ break ;
+
+
+ /************************************** default case **************************************/
+ default:
+ break ;
+ }//switch
+p->thread_status = S_IDLE ;
+ } //for
+
+free((void*)pBindBuffer) ;
+GetHandles(&stHandles, FREE, 0) ;
+p->thread_status = S_EXIT ;
+return 0 ;
+ };
+
+
+
+/*************************************************
+Function: ThreadFnDouble - thread function
+for double attributes
+*************************************************/
+/*
+void* ThreadFnDouble(void* pparams){
+
+ SQLRETURN retcode = SQL_ERROR ;
+ SQLCHAR szStmtBuffer[MAX_SQL_STMT] = { 0 } ;
+ SQLCHAR szValueBuffer[MAX_VALUE_LEN] = { 0 } ;
+ SQLCHAR szAuxBuffer[MAX_STR_LEN] = { 0 } ;
+ SQLCHAR szColBuffer[MAX_COL_NAME] = { 0 } ;
+ SQLINTEGER cbDouble = 0 ;
+ ODBC_HANDLES stHandles ;
+ memset(&stHandles, 0, sizeof(ODBC_HANDLES)) ;
+ int r = 0, j = 0 ;
+
+ //Get thread parameters
+ PARAMS* p = (PARAMS*)pparams ;
+ double* pRef = (double*)p->pThreadRef ;
+
+ double* pBindBuffer = (double*)malloc(sizeof(double)*nNoOfCol) ;
+
+ //printf("Thread #%d\n", p->nThreadID) ;
+
+ retcode = GetHandles(&stHandles, GET, 0) ;
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode ) {
+ p->report_status = S_STARTED ;
+ }else{
+ printf("Thread #%d failed to allocate handles, exiting now.\n", p->nThreadID) ;
+ free((void*)pBindBuffer) ;
+ p->report_status = S_EXIT ;
+ return 0 ;
+ }
+ //p->report_status = S_STARTED ;
+
+ //Main thread loop
+ for(;;){
+
+ while(S_IDLE == p->thread_status){
+ NdbSleep_MilliSleep(1) ;
+ }
+
+ if(S_STOP == p->thread_status) {
+ break ;
+ }else{
+ p->thread_status = S_BUSY ;
+ }
+
+ switch(p->op_type){
+
+
+ /************************************** T_INSERT case **************************************/
+ /* case T_INSERT:
+
+ for(r = 0 ; r < nNoOfRows ; ++r){
+ sprintf((char*)szStmtBuffer, "INSERT INTO %s VALUES(", p->szTableName) ;
+ for(j = 0 ;;){
+ sprintf((char*)szValueBuffer,"%.9f", pRef[nNoOfCol*r + j]) ;
+ strncat((char*)szStmtBuffer, (char*)szValueBuffer, strlen((const char*)szValueBuffer)) ;
+ ++j ;
+ if(nNoOfCol == j) break ;
+ strcat((char*)szStmtBuffer, ", ") ;
+ }
+ strcat((char*)szStmtBuffer, ")") ;
+ ODBC_FN(SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS), retcode) ;
+ }
+
+ break ;
+
+
+ /************************************** T_READ case **************************************/
+ /* case T_READ:
+
+ for(r = 0 ; r < nNoOfRows ; ++r){
+ sprintf((char*)szStmtBuffer, "SELECT * FROM %s WHERE COL0 = %.9f", p->szTableName, pRef[nNoOfCol*r]) ;
+ ODBC_FN(SQLExecDirect(stHandles.hstmt, (SQLCHAR*)szStmtBuffer, SQL_NTS), retcode) ;
+ for(j = 0 ; j < nNoOfCol ; ++j){
+ ODBC_FN(SQLBindCol(stHandles.hstmt, (j+1), SQL_C_DOUBLE, (void*)&pBindBuffer[j], sizeof(SQLDOUBLE), &cbDouble), retcode) ;
+ }
+ for (;;) {
+ retcode = SQLFetch(stHandles.hstmt);
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ for(int k = 0 ; k < nNoOfCol ; ++k){
+ if(p->nVerifyFlag){
+ if(abs(pBindBuffer[k] - pRef[nNoOfCol*r + k]) > DBLDEV)
+ printf("Expected: %.9f Actual: %.9f\n", pBindBuffer[k], pRef[nNoOfCol*r + k]) ;
+ }
+ }
+ }else if(SQL_NO_DATA == retcode){
+ break ;
+ }else{
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ //printf("\n") ;
+ }
+ SQLCloseCursor(stHandles.hstmt) ;
+ }
+ break ;
+
+
+ /************************************** T_UPDATE case **************************************/
+ /* case T_UPDATE:
+ for(r = 0 ; r < nNoOfRows ; ++r){
+
+ sprintf((char*)szStmtBuffer, "UPDATE %s SET ", p->szTableName) ;
+ for(j = 1 ;;){
+ pRef[nNoOfCol*r + j] = pRef[nNoOfCol*r + j] + UPDATE_VALUE ;
+ sprintf((char*)szColBuffer,"COL%d = %.9f", j, pRef[nNoOfCol*r + j]) ;
+ strncat((char*)szStmtBuffer, (char*)szColBuffer, strlen((char*)szColBuffer)) ;
+ ++j ;
+ if(nNoOfCol == j) break ;
+ strcat((char*)szStmtBuffer, ", ") ;
+ }
+ sprintf((char*)szAuxBuffer, " WHERE COL0 = %.9f ;", pRef[nNoOfCol*r]) ;
+ strcat((char*)szStmtBuffer, (char*)szAuxBuffer);
+ ODBC_FN(SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS), retcode) ;
+ }
+ break ;
+
+
+ /************************************** T_DELETE case **************************************/
+ /* case T_DELETE:
+ for(r = 0 ; r < nNoOfRows ; ++r){
+ sprintf((char*)szStmtBuffer, "DELETE FROM %s WHERE COL0 = %.9f", p->szTableName, pRef[nNoOfCol*r]) ;
+ retcode = SQLExecDirect(stHandles.hstmt, (SQLCHAR*)szStmtBuffer, SQL_NTS) ;
+ if( 1 == p->nVerifyFlag && SQL_NO_DATA != retcode ){
+ printf("\nVerification failed: still row exists\n") ;
+ }
+ }
+ break ;
+
+
+ /************************************** default case **************************************/
+ /* default:
+ break ;
+ }//switch
+p->thread_status = S_IDLE ;
+ } //for
+
+free((void*)pBindBuffer) ;
+GetHandles(&stHandles, FREE, 0) ;
+p->thread_status = S_EXIT ;
+return 0 ;
+ };
+
+
+
+/*************************************************
+Function: ThreadFnChar - thread function
+for character attributes
+*************************************************/
+void* ThreadFnChar(void* pparams){
+
+ SQLRETURN retcode = SQL_ERROR ;
+ SQLCHAR szStmtBuffer[MAX_SQL_STMT] = { 0 } ;
+ SQLCHAR szValueBuffer[MAX_VALUE_LEN] = { 0 } ;
+ SQLCHAR szAuxBuffer[MAX_STR_LEN] = { 0 } ;
+ SQLCHAR szColBuffer[MAX_COL_NAME] = { 0 } ;
+ SQLINTEGER cbChar = 0 ;
+ ODBC_HANDLES stHandles ;
+ memset(&stHandles, 0, sizeof(ODBC_HANDLES)) ;
+ int r = 0, j = 0 ;
+
+ //Get thread parameters
+ PARAMS* p = (PARAMS*)pparams ;
+ char* pRef = (char*)p->pThreadRef ;
+ char* pBindBuffer = (char*)malloc(sizeof(char)*nNoOfCol*MAX_CHAR_ATTR_LEN) ;
+
+ //printf("Thread #%d\n", p->nThreadID) ;
+
+ retcode = GetHandles(&stHandles, GET, 0) ;
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode ) {
+ p->report_status = S_STARTED ;
+ }else{
+ printf("Thread #%d failed to allocate handles, retcode = %d, exiting now.\n", p->nThreadID, retcode) ;
+ p->nError = 1 ;
+ free((void*)pBindBuffer) ;
+ p->report_status = S_EXIT ;
+ return 0 ;
+ }
+
+ //Main thread loop
+ for(;;){
+
+ while(S_IDLE == p->thread_status){
+ NdbSleep_MilliSleep(1) ;
+ }
+
+ if(S_STOP == p->thread_status) {
+ break ;
+ }else{
+ p->thread_status = S_BUSY ;
+ }
+
+ switch(p->op_type){
+
+
+ /************************************** T_INSERT case **************************************/
+ case T_INSERT:
+
+ for(r = 0 ; r < nNoOfRows ; ++r){
+ memset(szStmtBuffer, 0, strlen(szStmtBuffer)) ;
+ if(!nColList){
+ sprintf((char*)szStmtBuffer, "INSERT INTO %s VALUES(", p->szTableName) ;
+ }else{
+ sprintf((char*)szStmtBuffer, "INSERT INTO %s (%s) VALUES(", p->szTableName, szColNames) ;
+ }
+
+ for(j = 0 ;;){
+ sprintf((char*)szValueBuffer,"'%s'", (char*)(pRef + nNoOfCol*r*MAX_CHAR_ATTR_LEN*sizeof(char) + j*MAX_CHAR_ATTR_LEN*sizeof(char))) ;
+ strncat((char*)szStmtBuffer, (char*)szValueBuffer, strlen((const char*)szValueBuffer)) ;
+ ++j ;
+ if(nNoOfCol == j) break ;
+ strcat((char*)szStmtBuffer, ", ") ;
+ }
+ strcat((char*)szStmtBuffer, ")") ;
+
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS) ;
+ if(SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ }else{
+ p->nError = 1 ;
+ printf("INSERT in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ }
+
+ break ;
+
+
+ /************************************** T_READ case **************************************/
+ case T_READ:
+
+ for(r = 0 ; r < nNoOfRows ; ++r){
+ sprintf((char*)szStmtBuffer, "SELECT * FROM %s WHERE COL0 = '%s'", p->szTableName, (char*)(pRef + nNoOfCol*r*MAX_CHAR_ATTR_LEN*sizeof(char))) ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ ODBC_FN(SQLExecDirect(stHandles.hstmt, (SQLCHAR*)szStmtBuffer, SQL_NTS), retcode) ;
+ for(j = 0 ; j < nNoOfCol ; ++j){
+ ODBC_FN(SQLBindCol(stHandles.hstmt, (j+1), SQL_C_CHAR, (void*)(pBindBuffer+j*MAX_CHAR_ATTR_LEN*sizeof(char)), MAX_CHAR_ATTR_LEN, &cbChar), retcode) ;
+ }
+ for (;;) {
+ retcode = SQLFetch(stHandles.hstmt);
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ for(int k = 0 ; k < nNoOfCol ; ++k){
+ if(p->nVerifyFlag){
+ if(!strcmp((char*)(pBindBuffer + k*MAX_CHAR_ATTR_LEN*sizeof(char)), (char*)(pRef + nNoOfCol*r*MAX_CHAR_ATTR_LEN*sizeof(char) + k*MAX_CHAR_ATTR_LEN*sizeof(char))))
+ printf("Expected: %s Actual: %s\n", (char*)(pBindBuffer + k*MAX_CHAR_ATTR_LEN*sizeof(char)), (char*)(pRef + nNoOfCol*r*MAX_CHAR_ATTR_LEN*sizeof(char) + k*MAX_CHAR_ATTR_LEN*sizeof(char))) ;
+ }
+ }
+ }else if(SQL_NO_DATA == retcode){
+ break ;
+ }else{
+ p->nError = 1 ;
+ printf("READ in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ //printf("\n") ;
+ }
+ SQLCloseCursor(stHandles.hstmt) ;
+ }
+ break ;
+
+
+ /************************************** T_UPDATE case **************************************/
+ case T_UPDATE:
+ for(r = 0 ; r < nNoOfRows ; ++r){
+ sprintf((char*)szStmtBuffer, "UPDATE %s SET ", p->szTableName) ;
+ for(j = 1 ;;){
+ swab((char*)(pRef + nNoOfCol*r*MAX_CHAR_ATTR_LEN*sizeof(char) + j*MAX_CHAR_ATTR_LEN*sizeof(char)), (char*)szColBuffer, MAX_CHAR_ATTR_LEN*sizeof(char)) ;
+ memcpy((void*)(pRef + nNoOfCol*r*MAX_CHAR_ATTR_LEN*sizeof(char) + j*MAX_CHAR_ATTR_LEN*sizeof(char)), (void*)szColBuffer, MAX_CHAR_ATTR_LEN*sizeof(char)) ;
+ sprintf((char*)szColBuffer,"COL%d = '%s'", j, (char*)(pRef + nNoOfCol*r*MAX_CHAR_ATTR_LEN*sizeof(char) + j*MAX_CHAR_ATTR_LEN*sizeof(char))) ;
+ strncat((char*)szStmtBuffer, (char*)szColBuffer, strlen((char*)szColBuffer)) ;
+ ++j ;
+ if(nNoOfCol == j) break ;
+ strcat((char*)szStmtBuffer, ", ") ;
+ }
+ sprintf( (char*)szAuxBuffer, " WHERE COL0 = '%s';", (char*)(pRef + nNoOfCol*r*MAX_CHAR_ATTR_LEN*sizeof(char)) ) ;
+ strcat((char*)szStmtBuffer, (char*)szAuxBuffer) ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS) ;
+ if(SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ }else{
+ p->nError = 1 ;
+ printf("UPDATE in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ }
+ break ;
+
+
+ /************************************** T_DELETE case **************************************/
+ case T_DELETE:
+ for(r = 0 ; r < nNoOfRows ; ++r){
+ sprintf((char*)szStmtBuffer, "DELETE FROM %s WHERE COL0 = '%s\'", p->szTableName, (char*)(pRef + nNoOfCol*r*MAX_CHAR_ATTR_LEN*sizeof(char))) ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS) ;
+ if(SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ }else if(1 == p->nVerifyFlag && SQL_NO_DATA != retcode){
+ p->nError = 1 ;
+ printf("\nVerification failed: still row exists\n") ;
+ }else{
+ p->nError = 1 ;
+ printf("INSERT in thread #%d failed\n", p->nThreadID) ;
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+ }
+ break ;
+
+
+ /************************************** default case **************************************/
+ default:
+ break ;
+ }//switch
+ p->thread_status = S_IDLE ;
+ } //for
+
+ free((void*)pBindBuffer) ;
+ GetHandles(&stHandles, FREE, 0) ;
+ p->thread_status = S_EXIT ;
+ return 0 ;
+};
+
+
+
+/*************************************************
+Function: CreateDemoTable
+*************************************************/
+SQLRETURN CreateDemoTables(char* szTableName, int nTables, table_opt op){
+
+ SQLRETURN retcode = SQL_ERROR ;
+ SQLCHAR szStmtBuffer[MAX_SQL_STMT] = { 0 } ;
+ SQLCHAR szColBuffer[MAX_COL_NAME] = { 0 } ;
+ SQLCHAR szAuxBuffer[32] = { 0 } ;
+ ODBC_HANDLES stHandles ;
+ memset(&stHandles, 0, sizeof(ODBC_HANDLES)) ;
+ int c = 0 ;
+
+ GetHandles(&stHandles, GET, 0) ;
+
+ if(CREATE == op){
+
+ for(c = 0; c < nTables ; ++c){
+ sprintf((char*)szStmtBuffer, "CREATE TABLE %s (", (char*)(szTableName+MAX_TABLE_NAME*c)) ;
+ int j = 0 ;
+ for(;;){
+ sprintf((char*)szColBuffer, "COL%d ", j) ;
+ strcat((char*)szStmtBuffer, (char*)szColBuffer) ;
+ ++j ;
+
+ switch(AttributeType){
+ case T_INT:
+ strcat((char*)szStmtBuffer, "INTEGER") ;
+ break ;
+ case T_FLOAT:
+ strcat((char*)szStmtBuffer, "FLOAT") ;
+ break ;
+
+/* case T_DOUBLE:
+ strcat((char*)szStmtBuffer, "DOUBLE") ;
+ break ;
+*/
+ case T_CHAR:
+ sprintf((char*)szAuxBuffer, "CHAR(%d)", MAX_CHAR_ATTR_LEN) ;
+ strcat((char*)szStmtBuffer, (char*)szAuxBuffer) ;
+ break ;
+ default:
+ break ;
+ }
+
+ if(nNoOfCol <= j){
+ strcat((char*)szStmtBuffer, ")") ;
+ break ;
+ }
+ strcat((char*)szStmtBuffer, ", ") ;
+ } //for(;;)
+
+
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ ODBC_FN(SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS), retcode) ;
+ if(SQL_SUCCESS != retcode) HandleError(stHandles.hstmt , SQL_HANDLE_STMT) ;
+
+
+ }// for()
+
+ }else{
+
+ for(c = 0 ; c < nTables ; ++c){
+ sprintf((char*)szStmtBuffer, "DROP TABLE %s ", (char*)(szTableName + MAX_TABLE_NAME*c)) ;
+ //ODBC_FN(SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS), retcode) ;
+ if(nPrint) printf("\n> %s\n", szStmtBuffer) ;
+ retcode = SQLExecDirect(stHandles.hstmt, szStmtBuffer, SQL_NTS) ;
+ }
+
+ }
+
+ GetHandles(&stHandles, FREE, 0) ;
+
+ return retcode ;
+}
+
+
+
+/*************************************************
+Function: AssignTableNames()
+*************************************************/
+inline void AssignTableNames(char* szBuffer, int nTables){
+ for(int c = 0 ; c < nTables ; ++c){
+ sprintf((char*)(szBuffer + MAX_TABLE_NAME*sizeof(char)*c), "TAB%d", c) ;
+ }
+return ;
+ }
+
+
+
+
+/*************************************************
+Function: StartThreads()
+*************************************************/
+
+inline void StartThreads(PARAMS* p, void* pRef, int nTables, char* szTables, attr_type attrType, UintPtr* pHandles) {
+
+ int* pInt = NULL ;
+ float* pFloat = NULL ;
+ double* pDouble = NULL ;
+ char* pChar = NULL ;
+ UintPtr pTmpThread = NULL ;
+
+ bool bFlap = 1 ;
+ for(int f = 0 ; f < nNoOfThreads ; ++f){
+ p[f].nThreadID = f ;
+ p[f].nError = 0 ;
+ p[f].thread_status = S_IDLE ;
+ p[f].op_type = T_WAIT ;
+ if(bFlap){
+ strncpy((char*)p[f].szTableName, (char*)szTables, MAX_TABLE_NAME) ;
+ }else{
+ strncpy((char*)p[f].szTableName, (char*)(szTables + MAX_TABLE_NAME*sizeof(char)), MAX_TABLE_NAME) ;
+ }
+ bFlap = !bFlap ;
+ //pTmpThread = pHandles[ ;
+
+ switch(attrType){
+ case T_INT:
+ pInt = (int*)pRef ;
+ p[f].pThreadRef = (void*)&pInt[nNoOfRows*nNoOfCol*f] ;
+ pHandles[f] = (UintPtr)NdbThread_Create(ThreadFnInt, (void**)&p[f], 32768, "SQL99_test", NDB_THREAD_PRIO_MEAN) ;
+ break ;
+ case T_FLOAT:
+ pFloat = (float*)pRef ;
+ p[f].pThreadRef = (void*)&pFloat[nNoOfRows*nNoOfCol*f] ;
+ pHandles[f] = (UintPtr)NdbThread_Create(ThreadFnFloat, (void**)&p[f], 32768, "SQL99_test", NDB_THREAD_PRIO_MEAN) ;
+ break ;
+ /*
+ case T_DOUBLE:
+ pDouble = (double*)pRef ;
+ p[f].pThreadRef = (void*)&pDouble[nNoOfRows*nNoOfCol*f] ;
+ pHandles[f] = (UintPtr)NdbThread_Create(ThreadFnDouble, (void**)&p[f], 32768, "SQL99_test", NDB_THREAD_PRIO_MEAN) ;
+ break ;
+ */
+ case T_CHAR:
+ pChar = (char*)pRef ;
+ p[f].pThreadRef = (void*)&pChar[nNoOfRows*nNoOfCol*f*MAX_CHAR_ATTR_LEN] ;
+ pHandles[f] = (UintPtr)NdbThread_Create(ThreadFnChar, (void**)&p[f], 32768, "SQL99_test", NDB_THREAD_PRIO_MEAN) ;
+ default:
+ break ;
+ }
+ while(!(S_STARTED != p[f].report_status || S_EXIT != p[f].report_status)){
+ NdbSleep_MilliSleep(1) ;
+ }
+ }
+ return ;
+}
+
+
+
+/*************************************************
+Function: SetThreadOperationType()
+*************************************************/
+inline void SetThreadOperationType(PARAMS* p, type op){
+
+ for(int e = 0 ; e < nNoOfThreads ; ++e){
+ p[e].nVerifyFlag = 0 ;
+ if(T_READ_VERIFY == op){
+ p[e].nVerifyFlag = 1 ;
+ p[e].op_type = T_READ ;
+ }else if(T_DELETE_VERIFY == op){
+ p[e].nVerifyFlag = 1 ;
+ p[e].op_type = T_DELETE ;
+ }else{
+ p[e].op_type = op ;
+ }
+ p[e].thread_status = S_GET_BUSY ;
+ }
+return ;
+ }
+
+
+
+/*************************************************
+Function: WaitForThreads()
+*************************************************/
+inline int WaitForThreads(PARAMS* p) {
+
+ int ret_value = 0 ;
+ for(int w = 0 ; w < nNoOfThreads ; ++w){
+ while(!(S_IDLE != p[w].thread_status || S_EXIT != p[w].report_status)) {
+ NdbSleep_MilliSleep(1) ;
+ }
+ ret_value += p[w].nError ;
+ }
+ return ret_value ;
+}
+
+
+
+/*************************************************
+Function: StopThreads()
+*************************************************/
+inline void StopThreads(PARAMS* p, UintPtr* pHandles) {
+
+ for(int k = 0 ; k < nNoOfThreads ; ++k){
+ while(!(S_IDLE != p[k].thread_status || S_EXIT != p[k].report_status)){
+ NdbSleep_MilliSleep(1) ;
+ }
+ p[k].thread_status = S_STOP ;
+ while(!(S_EXIT != p[k].thread_status || S_EXIT != p[k].report_status)){
+ NdbSleep_MilliSleep(1) ;
+ }
+ NdbThread_Destroy((NdbThread**)&pHandles[k]) ;
+ }
+
+ return ;
+}
+
+
+
+/*************************************************
+Function: PrintAll()
+*************************************************/
+inline void PrintAll(char* szTableName, int nTables, attr_type attrType){
+
+ SQLRETURN retcode = SQL_ERROR ;
+ SQLCHAR* szStmt[MAX_SQL_STMT] = { 0 } ;
+ ODBC_HANDLES stHandles ;
+ memset(&stHandles, 0, sizeof(ODBC_HANDLES)) ;
+ double* pDoubleBuffer = NULL ;
+ char* pCharBuffer = NULL ;
+
+ if(T_CHAR != attrType){
+ pDoubleBuffer = (double*)malloc(sizeof(double)*nNoOfCol) ;
+ }else{
+ pCharBuffer = (char*)malloc(sizeof(char)*nNoOfCol*MAX_CHAR_ATTR_LEN) ;
+ }
+
+ SQLINTEGER cbLen = 0 ;
+
+ GetHandles(&stHandles, GET, 0) ;
+
+ for(int c = 0 ; c < nTables ; ++c){
+
+ int nCol = 0, nRows = 0 ;
+
+ printf("Table: \"%s\":\n------------------\n", (char*)(szTableName + MAX_TABLE_NAME*c*sizeof(char))) ;
+
+ sprintf((char*)szStmt, "SELECT * FROM %s", (char*)(szTableName + MAX_TABLE_NAME*c*sizeof(char))) ;
+ if(nPrint) printf("\n> %s\n", szStmt) ;
+ ODBC_FN(SQLExecDirect(stHandles.hstmt, (SQLCHAR*)szStmt, SQL_NTS), retcode) ;
+
+ for(int i = 0 ; i < nNoOfCol ; ++i){
+
+ if(T_CHAR != attrType){
+ ODBC_FN(SQLBindCol(stHandles.hstmt, (i+1), SQL_C_DOUBLE, (void*)&pDoubleBuffer[i], sizeof(SQLDOUBLE), &cbLen), retcode) ;
+ }else{
+ ODBC_FN(SQLBindCol(stHandles.hstmt, (i+1), SQL_C_CHAR, (void*)(pCharBuffer + i*MAX_CHAR_ATTR_LEN*sizeof(char)), MAX_CHAR_ATTR_LEN*sizeof(char), &cbLen), retcode) ;
+ }
+ nCol++ ;
+
+ }
+
+ int k = 0 ; //out of the <for> loop
+ for (;;) {
+
+ retcode = SQLFetch(stHandles.hstmt);
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode ){
+ for(k = 0 ; k < nNoOfCol ; ++k){
+ if(T_CHAR != attrType){
+ ATTR_TYPE_SWITCH_T(pDoubleBuffer[k], AttributeType) ;
+ }else{
+ printf("%s\t", (char*)(pCharBuffer + k*MAX_CHAR_ATTR_LEN)) ;
+ }
+ }
+ }else if(SQL_NO_DATA == retcode){
+ if(0 == k){
+ printf("<empty>\n") ;
+ break ;
+ }else{
+ break ;
+ }
+ }else{
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+
+ ++nRows ;
+ printf("\n") ;
+ }
+
+ SQLCloseCursor(stHandles.hstmt) ;
+
+ printf("------------------\n") ;
+ printf("Rows: %d Columns: %d\n\n", nRows, nCol) ;
+
+ }
+
+ free((void*)pDoubleBuffer) ;
+ free((void*)pCharBuffer) ;
+ GetHandles(&stHandles, FREE, 0) ;
+
+ return ;
+}
+
+
+
+/*************************************************
+Function: AssignRefCharValues()
+*************************************************/
+void AssignRefCharValues(char* pRef, bool bVerbose) {
+
+ int count = 0, rows = 0, nThreadOffset = 0, nRowOffset = 0 ;
+ char szStrBuffer[MAX_CHAR_ATTR_LEN] = { 0 } ;
+ int char_count = sizeof(szANSI)/sizeof(char) ;
+
+ for(int c = 0 ; c < nNoOfThreads ; ++c){
+ nThreadOffset = nNoOfRows*nNoOfCol*c*MAX_CHAR_ATTR_LEN*sizeof(char) ;
+ for(int d = 0 ; d < nNoOfRows ; ++d){
+ nRowOffset = nNoOfCol*d*MAX_CHAR_ATTR_LEN*sizeof(char) ; ++rows ;
+ for(int i = 0 ; i < nNoOfCol ; ++i){
+ for(int j = 0 ; j < (MAX_CHAR_ATTR_LEN - 2) ; ++j){
+ int h = (char)(rand() % (char_count-1)) ;
+ szStrBuffer[j] = szANSI[h] ;
+ }
+ szStrBuffer[MAX_CHAR_ATTR_LEN - 1] = '\0' ;
+
+ strcpy((char*)(pRef + nThreadOffset + nRowOffset + i*MAX_CHAR_ATTR_LEN*sizeof(char)), (char*)szStrBuffer) ;
+ count++ ;
+ if(bVerbose){
+ printf(" %s ", (char*)(pRef + nThreadOffset + nRowOffset + i*MAX_CHAR_ATTR_LEN*sizeof(char))) ;
+ }
+ }
+ if(bVerbose) {
+ printf("\n") ;
+ NdbSleep_MilliSleep(10) ;
+ }
+ }
+ }
+
+if(bVerbose){
+ printf("_____________________") ;
+ printf("\nRows: %d Values: %d\n\n", rows, count) ;
+ }
+
+return ;
+ }
+
+
+/*
+
+
+sprintf((char*)szStmtBuffer, "INSERT INTO %s VALUES(", p->szTableName) ;
+for(j = 0 ;;){
+strcat((char*)szStmtBuffer, "?") ;
+++j ;
+if(nNoOfCol == j) break ;
+strcat((char*)szStmtBuffer, ", ") ;
+}
+strcat((char*)szStmtBuffer, ")") ;
+
+ODBC_FN(SQLPrepare(stHandles.hstmt, szStmtBuffer, SQL_NTS), retcode) ;
+
+for(j = 0 ; j < nNoOfCol ; ++j){
+ODBC_FN(SQLBindParameter(stHandles.hstmt, (j+1), SQL_PARAM_INPUT, SQL_C_FLOAT, SQL_FLOAT, 0, 0, (void*)&pBindBuffer[j], 0, &cbFloat), retcode) ;
+HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+}
+
+for(r = 0 ; r < nNoOfRows ; ++r){
+for(j = 0 ; j < nNoOfCol ; ++j){
+pBindBuffer[j] = pRef[nNoOfCol*r + j] ;
+}
+ODBC_FN(SQLExecute(stHandles.hstmt), retcode) ;
+HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+}
+
+*/
+
+
+
+
+/*************************************************
+Function: HandleError
+*************************************************/
+
+void HandleError(void* handle, SQLSMALLINT HandleType){
+
+ SQLCHAR szError[MAX_STR_LEN], szSqlState[32] ;
+ SQLINTEGER nError = 0 ;
+ SQLSMALLINT nHandleType = HandleType ;
+ SQLSMALLINT nLength = 0 ;
+ SQLHANDLE SQLHandle = handle ;
+ SQLGetDiagRec(nHandleType, SQLHandle, 1, szSqlState, &nError, szError, 128, &nLength) ;
+ printf("Error: %s\nSqlState: %s\n", szError, szSqlState) ;
+
+ return ;
+ }
+
+
+
+/*************************************************
+Function: ReportError
+*************************************************/
+
+void ReportError(char* szFn, char* szBuffer, char* szFile, int iLine){
+
+ printf("%s %s\nFile: %s\nLine: %d\n", szFn, szBuffer, szFile, iLine) ;
+
+ return ;
+}
+
+
+
+/*************************************************
+Function: GetHandles()
+*************************************************/
+
+SQLRETURN GetHandles(ODBC_HANDLES* pHandles, handle_op op, bool bDriverInfo){
+
+ SQLRETURN retcode = SQL_ERROR ;
+
+ if(GET == op){
+
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &pHandles->henv);
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode ) {
+ retcode = SQLSetEnvAttr(pHandles->henv, SQL_ATTR_ODBC_VERSION, (void*)SQL_OV_ODBC2, 0);
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode) {
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC, pHandles->henv, &pHandles->hdbc);
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode) {
+
+ //SQLSetConnectAttr(pHandles->hdbc, SQL_LOGIN_TIMEOUT, (void*)5, 0);
+
+ retcode = SQLConnect(pHandles->hdbc, (SQLCHAR*)"", SQL_NTS, (SQLCHAR*)"", SQL_NTS, (SQLCHAR*)"", SQL_NTS ) ;
+
+ SQL_SUCCESS == SQLSetConnectAttr(pHandles->hdbc, SQL_ATTR_AUTOCOMMIT, (void*)SQL_AUTOCOMMIT_ON, 0) ;
+ //printf("AUTOCOMMIT is on\n") ;
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode) {
+ // retcode still holds the value returned by SQLConnect
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT, pHandles->hdbc, &pHandles->hstmt) ;
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode) {
+ if(bDriverInfo) GetDriverAndSourceInfo(pHandles->hdbc) ;
+ // printf("All handles allocated OK\n", retcode);
+ }else{ // SQLAllocHandle()
+ REPORTERROR((char*)"SQLAllocHandle()", (char*)"failed") ;
+ HandleError(pHandles->hdbc, SQL_HANDLE_DBC) ;
+ ODBC_FN(SQLDisconnect(pHandles->hdbc), retcode) ;
+ ODBC_FN(SQLFreeHandle(SQL_HANDLE_DBC, pHandles->hdbc), retcode) ;
+ ODBC_FN(SQLFreeHandle(SQL_HANDLE_ENV, pHandles->henv), retcode) ;
+ retcode = SQL_ERROR ;
+ }
+ }else{ // SQLConnect()
+ REPORTERROR((char*)"SQLConnect()", (char*)"failed" ) ;
+ HandleError(pHandles->hdbc, SQL_HANDLE_DBC) ;
+ ODBC_FN(SQLFreeHandle(SQL_HANDLE_DBC, pHandles->hdbc), retcode) ;
+ ODBC_FN(SQLFreeHandle(SQL_HANDLE_ENV, pHandles->henv), retcode) ;
+ retcode = SQL_ERROR ;
+ }
+ }else{ // SQLAllocHandle()
+ REPORTERROR((char*)"SQLAllocHandle()", "failed" ) ;
+ HandleError(pHandles->hdbc, SQL_HANDLE_DBC) ;
+ ODBC_FN(SQLFreeHandle(SQL_HANDLE_ENV, pHandles->henv), retcode) ;
+ retcode = SQL_ERROR ;
+ }
+ }else{ // SQLSetEnvAttr()
+ REPORTERROR((char*)"SQLSetEnvAttr()", "failed" ) ;
+ HandleError(pHandles->henv, SQL_HANDLE_ENV) ;
+ ODBC_FN(SQLFreeHandle(SQL_HANDLE_ENV, pHandles->henv), retcode) ;
+ retcode = SQL_ERROR ;
+ }
+ }else{ // SQLAllocHandle()
+ REPORTERROR((char*)"SQLAllocHandle()", "failed" ) ;
+ HandleError(pHandles->henv, SQL_HANDLE_ENV) ;
+ retcode = SQL_ERROR ;
+ }
+ }else{
+ ODBC_FN(SQLFreeHandle(SQL_HANDLE_STMT, pHandles->hstmt), retcode) ;
+ ODBC_FN(SQLDisconnect(pHandles->hdbc), retcode) ;
+ ODBC_FN(SQLFreeHandle(SQL_HANDLE_DBC, pHandles->hdbc), retcode) ;
+ ODBC_FN(SQLFreeHandle(SQL_HANDLE_ENV, pHandles->henv), retcode) ;
+ }
+
+ return retcode ;
+}
+
+
+
+/*************************************************
+Function: AggretateFn():
+<aggr_fn fn> - name of the aggregate function to use
+<char* szTableName> - name of the table
+<int nCol> - number of the column
+<double* pdIn> - pointer to double containing the value to be used in a call to COUNT; used only by this function
+<double* pdOut> - pointer to double that will recieve the result
+<attr_type attrType> - type of the attribute
+*************************************************/
+SQLRETURN AggregateFn(aggr_fn fn, char* szTableName, int nCol, double* pdIn, double* pdOut, attr_type attrType){
+
+ SQLRETURN retcode = SQL_ERROR ;
+ SQLCHAR* szStmt[MAX_SQL_STMT] = { 0 } ;
+ SQLCHAR szValueBuffer[MAX_VALUE_LEN] = { 0 } ;
+ SQLCHAR szAuxBuffer[MAX_STR_LEN] = { 0 } ;
+ SQLCHAR szColBuffer[MAX_COL_NAME] = { 0 } ;
+ ODBC_HANDLES stHandles ;
+ memset(&stHandles, 0, sizeof(ODBC_HANDLES)) ;
+ SQLINTEGER cbDouble = 0 ;
+
+ GetHandles(&stHandles, GET, 0) ;
+
+ switch(fn){
+ case FN_COUNT:
+ switch(attrType){
+ case T_INT:
+ sprintf((char*)szStmt, "SELECT COUNT(*) FROM %s WHERE COL%d > %d", szTableName, nCol, (int)*pdIn) ;
+ break ;
+ case T_FLOAT:
+ sprintf((char*)szStmt, "SELECT COUNT(*) FROM %s WHERE COL%d > %f", szTableName, nCol, (float)*pdIn) ;
+ break ;
+/* case T_DOUBLE:
+ sprintf((char*)szStmt, "SELECT COUNT(*) FROM %s WHERE COL%d > %.15f", szTableName, nCol, *pdIn) ;
+ break ;
+*/
+ default:
+ break ;
+ }
+ break ;
+ case FN_SUM:
+ sprintf((char*)szStmt, "SELECT SUM(COL%d) FROM %s", nCol, szTableName) ;
+ break ;
+ case FN_AVG:
+ sprintf((char*)szStmt, "SELECT AVG(COL%d) FROM %s", nCol, szTableName) ;
+ break ;
+ case FN_MAX:
+ sprintf((char*)szStmt, "SELECT MAX(COL%d) FROM %s", nCol, szTableName) ;
+ break ;
+ case FN_MIN:
+ sprintf((char*)szStmt, "SELECT MIN(COL%d) FROM %s", nCol, szTableName) ;
+ break ;
+ case FN_VARIANCE: // not implemented
+ //sprintf((char*)szStmt, "SELECT VARIANCE(COL%d) FROM %s;", nCol, szTableName) ;
+ break ;
+ case FN_STDDEV: // not implemented
+ //sprintf((char*)szStmt, "SELECT STDDEV(COL%d) FROM %s;", nCol, szTableName) ;
+ break ;
+ default:
+ break ;
+ }
+//printf("%s\n", szStmt) ;
+
+retcode = SQLExecDirect(stHandles.hstmt, (SQLCHAR*)szStmt, SQL_NTS) ;
+if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode ){
+ retcode = SQLBindCol(stHandles.hstmt, 1, SQL_C_DOUBLE, (void*)pdOut, sizeof(SQLDOUBLE), &cbDouble) ;
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode ){
+ retcode = SQLFetch(stHandles.hstmt) ;
+ }
+ }
+
+if(SQL_SUCCESS != retcode && SQL_SUCCESS_WITH_INFO != retcode){
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+
+SQLCloseCursor(stHandles.hstmt) ;
+
+GetHandles(&stHandles, FREE, 0) ;
+
+return retcode ;
+
+ };
+
+
+
+/*************************************************
+Function: GetDriverAndSourceInfo()
+*************************************************/
+SQLRETURN GetDriverAndSourceInfo(SQLHDBC hdbc){
+
+ SQLRETURN retcode = SQL_ERROR ;
+
+ SQLCHAR buffer[255] ;
+ SQLUSMALLINT snValue = 0 ;
+ SQLSMALLINT outlen = 0 ;
+
+ printf( "-------------------------------------------\n" ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_DATA_SOURCE_NAME, buffer, 255, &outlen ) ;
+
+ printf( "Connected to Server: %s\n", buffer ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_DATABASE_NAME, buffer, 255, &outlen ) ;
+ printf( " Database name: %s\n", buffer ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_SERVER_NAME, buffer, 255, &outlen ) ;
+ printf( " Instance name: %s\n", buffer ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_DBMS_NAME, buffer, 255, &outlen ) ;
+ printf( " DBMS name: %s\n", buffer ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_DBMS_VER, buffer, 255, &outlen ) ;
+ printf( " DBMS version: %s\n", buffer ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_ODBC_VER, buffer, 255, &outlen ) ;
+ printf( " ODBC version: %s\n", buffer ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_DRIVER_NAME, buffer, 255, &outlen ) ;
+ printf( " Driver name: %s\n", buffer ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_DRIVER_VER, buffer, 255, &outlen ) ;
+ printf( " Driver version: %s\n", buffer ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_MAX_DRIVER_CONNECTIONS, &snValue, sizeof(SQLSMALLINT), &outlen ) ;
+ printf( " Max connections: %d\n", snValue ) ;
+
+ retcode = SQLGetInfo( hdbc, SQL_CURSOR_COMMIT_BEHAVIOR, &snValue, sizeof(SQLSMALLINT), &outlen ) ;
+ printf( "Autocommit behavior:") ;
+
+ switch(snValue){
+ case SQL_CB_DELETE:
+ printf(" SQL_CB_DELETE\n") ;
+ break ;
+ case SQL_CB_CLOSE:
+ printf(" SQL_CB_CLOSE\n") ;
+ break ;
+ case SQL_CB_PRESERVE:
+ printf(" SQL_CB_PRESERVE\n") ;
+ break ;
+ default:
+ printf(" undefined\n") ;
+ break ;
+ }
+
+ printf( "-------------------------------------------\n" ) ;
+
+ return retcode ;
+
+}
+
+
+
+/*************************************************
+Function: ArithOp()
+*************************************************/
+
+int ArithOp(char* szTable, int nTotalCols, float* pValue, attr_type attrType, arth_op op){
+
+ SQLRETURN retcode = SQL_ERROR ;
+ int nVerRet = -1 ;
+ SQLCHAR szStmt[MAX_SQL_STMT] = { 0 } ;
+ SQLCHAR szEndBuffer[MAX_STR_LEN] = { 0 } ;
+ SQLCHAR szColBuffer[MAX_COL_NAME] = { 0 } ;
+ SQLCHAR szAuxBuffer[MAX_STR_LEN] = { 0 } ;
+ ODBC_HANDLES stHandles ;
+ memset(&stHandles, 0, sizeof(ODBC_HANDLES)) ;
+
+ void* pBuffer = NULL ;
+ SQLINTEGER BindInt = 0, IntResult = 0, RefIntResult = 0 ;
+ SQLFLOAT BindFloat = 0, FloatResult = 0, RefFloatResult = 0 ;
+ SQLDOUBLE BindDouble = 0, DoubleResult = 0, RefDoubleResult = 0 ;
+ SQLINTEGER cbSize = 0 ;
+ SQLINTEGER cbLen = 0 ;
+ SQLSMALLINT cbTarget = 0 ;
+
+ GetHandles(&stHandles, GET, 0) ;
+
+ for(int c = 0 ; c < nTotalCols ; ++c){
+
+ sprintf((char*)szStmt, "SELECT COL%d, (COL%d", c, c) ;
+ switch(op){
+ case MINUS:
+ strcat((char*)szStmt, " - ") ;
+ break ;
+ case PLUS:
+ strcat((char*)szStmt, " + ") ;
+ break ;
+ case MULTI:
+ strcat((char*)szStmt, " * ") ;
+ break ;
+ case DIVIDE:
+ strcat((char*)szStmt, " / ") ;
+ break ;
+ case MODULO:
+ //strcat((char*)szStmt, " % ") ; Not implemented
+ GetHandles(&stHandles, FREE, 0) ;
+ return -1 ; //Close handles and return
+ break ;
+ default:
+ break ;
+ }
+
+ sprintf((char*)(szAuxBuffer),"%.9f) ", *((float*)(pValue))) ;
+ strcat((char*)szStmt, (char*)szAuxBuffer) ;
+ sprintf((char*)szEndBuffer, "FROM %s", szTable) ;
+ strcat((char*)szStmt, (char*)szEndBuffer) ;
+
+ ODBC_FN(SQLExecDirect(stHandles.hstmt, (SQLCHAR*)szStmt, SQL_NTS), retcode) ;
+ if(retcode == SQL_ERROR){
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ printf("\n%s\n", szStmt) ;
+ }
+
+ SQLSMALLINT cbNameLen = 0, cbSqlType = 0, cbNullable = 0, cbColScale = 0 ;
+ SQLINTEGER cbColSize = 0 ;
+ SQLDescribeCol(stHandles.hstmt, 2, szColBuffer, MAX_COL_NAME-1, &cbNameLen, &cbSqlType, (unsigned long*)&cbColSize, &cbColScale, &cbNullable) ;
+
+ switch(cbSqlType){
+ case SQL_NUMERIC:
+ pBuffer = &IntResult ;
+ cbSize = sizeof(SQLINTEGER) ;
+ cbTarget = SQL_C_ULONG ;
+ case SQL_INTEGER:
+ pBuffer = &IntResult ;
+ cbSize = sizeof(SQLINTEGER) ;
+ cbTarget = SQL_C_LONG ;
+ break ;
+ case SQL_FLOAT:
+ pBuffer = &FloatResult ;
+ cbSize = sizeof(SQLFLOAT) ;
+ cbTarget = SQL_C_FLOAT ;
+ break ;
+ case SQL_DOUBLE:
+ pBuffer = &DoubleResult ;
+ cbSize = sizeof(SQLDOUBLE) ;
+ cbTarget = SQL_C_DOUBLE ;
+ break ;
+ default:
+ printf("\nUndefined result type: %d\n", cbSqlType) ;
+ break ;
+ }
+
+ switch(attrType){
+ case T_INT:
+ ODBC_FN(SQLBindCol(stHandles.hstmt, 1, SQL_C_SLONG, (void*)&BindInt, sizeof(SQLINTEGER), &cbLen), retcode) ;
+ break ;
+ case T_FLOAT:
+ ODBC_FN(SQLBindCol(stHandles.hstmt, 1, SQL_C_FLOAT, (void*)&BindFloat, sizeof(SQLFLOAT), &cbLen), retcode) ;
+ break ;
+ /* case T_DOUBLE:
+ ODBC_FN(SQLBindCol(stHandles.hstmt, 1, SQL_C_DOUBLE, (void*)&BindDouble, sizeof(SQLDOUBLE), &cbLen), retcode) ;
+ break ;
+ */
+ default:
+ break ;
+ }
+
+ ODBC_FN(SQLBindCol(stHandles.hstmt, 2, cbTarget, pBuffer, cbSize, &cbLen), retcode) ;
+
+ retcode = SQLFetch(stHandles.hstmt) ;
+
+ if (SQL_SUCCESS == retcode || SQL_SUCCESS_WITH_INFO == retcode){
+ switch(attrType){
+ case T_INT:
+ switch(cbSqlType){
+ case SQL_INTEGER:
+ nVerRet = VerifyArthOp((int*)&BindInt, pValue, (int*)pBuffer, op) ;
+ break ;
+ case SQL_FLOAT:
+ nVerRet = VerifyArthOp((int*)&BindInt, pValue, (float*)pBuffer, op) ;
+ break ;
+ case SQL_DOUBLE:
+ nVerRet = VerifyArthOp((int*)&BindInt, pValue, (double*)pBuffer, op) ;
+ break ;
+ case SQL_NUMERIC:
+ nVerRet = VerifyArthOp((int*)&BindInt, pValue, (int*)pBuffer, op) ;
+ break ;
+ default:
+ break ;
+ }
+ break ;
+
+ case T_FLOAT:
+ switch(cbSqlType){
+ case SQL_INTEGER:
+ nVerRet = VerifyArthOp((float*)&BindFloat, pValue, (int*)pBuffer, op) ;
+ break ;
+ case SQL_FLOAT:
+ nVerRet = VerifyArthOp((float*)&BindFloat, pValue, (float*)pBuffer, op) ;
+ break ;
+ case SQL_DOUBLE:
+ nVerRet = VerifyArthOp((float*)&BindFloat, pValue, (double*)pBuffer, op) ;
+ break ;
+ default:
+ break ;
+ }
+ break ;
+ /* case T_DOUBLE:
+ switch(cbSqlType){
+ case SQL_INTEGER:
+ nVerRet = VerifyArthOp((double*)&BindDouble, pValue, (int*)pBuffer, op) ;
+ break ;
+ case SQL_FLOAT:
+ nVerRet = VerifyArthOp((double*)&BindDouble, pValue, (float*)pBuffer, op) ;
+ break ;
+ case SQL_DOUBLE:
+ nVerRet = VerifyArthOp((double*)&BindDouble, pValue, (double*)pBuffer, op) ;
+ break ;
+ default:
+ break ;
+ }
+ break ;
+ */
+ default:
+ break ;
+ }
+ if(-1 == nVerRet){
+ printf("\nVerification failed.\n") ;
+ return nVerRet ;
+ }else if(SQL_NO_DATA == retcode){
+ break ;
+ }
+ }else{
+
+ HandleError(stHandles.hstmt, SQL_HANDLE_STMT) ;
+ }
+
+ SQLCloseCursor(stHandles.hstmt) ;
+ }
+
+ GetHandles(&stHandles, FREE, 0) ;
+
+ return nVerRet ;
+}
+
+
+
+
+/*************************************************
+Function: Join()
+*************************************************/
+SQLRETURN Join(char* szTable, int nTables, int nCol, join_type joinType){
+
+ SQLRETURN retcode = SQL_ERROR ;
+ SQLCHAR szStmt[MAX_SQL_STMT] = { 0 } ;
+ SQLCHAR szEndBuffer[MAX_STR_LEN] = { 0 } ;
+ SQLCHAR szColBuffer[MAX_COL_NAME] = { 0 } ;
+ SQLCHAR szAuxBuffer[MAX_STR_LEN] = { 0 } ;
+
+ ODBC_HANDLES stHandles ;
+ memset(&stHandles, 0, sizeof(ODBC_HANDLES)) ;
+
+ int c = 0, t = 0 ;
+
+ GetHandles(&stHandles, GET, 0) ;
+
+ for(c = 0 ; c < nCol ; ++c) {
+
+ switch(joinType){
+ case ITSELF:
+ sprintf((char*)szStmt, "SELECT * FROM %s, %s", (char*)szTable, (char*)szTable) ;
+ break ;
+ case EQUI:
+ break ;
+ case NON_EQUI:
+ break ;
+ case INNER:
+ break ;
+ case OUTTER:
+ break ;
+ default:
+ break ;
+ }
+ }
+
+GetHandles(&stHandles, FREE, 0) ;
+
+return retcode ;
+
+}
+
+
+
+SQLRETURN GetResults(SQLHSTMT){
+
+ SQLRETURN retcode = SQL_ERROR ;
+
+ return retcode ;
+}
+
+/*
+
+int createTables(char* szTableName, int nTables){
+
+ for (int i = 0; i < nNoOfCol; i++){
+ snprintf(attrName[i], MAXSTRLEN, "COL%d", i) ;
+ }
+
+ for (int i = 0; i < nTables; i++){
+ snprintf(tableName[i], MAXSTRLEN, "TAB%d", i) ;
+ }
+
+ for(unsigned i = 0; i < nTables; i++){
+
+ ndbout << "Creating " << szTableName[i] << "... " ;
+
+ NDBT_Table tmpTable(szTableName[i]) ;
+
+ tmpTable.setStoredTable(!theTempTable) ;
+
+ tmpTable.addAttribute(NDBT_Attribute(attrName[0],
+ UnSigned,
+ 4, // 4 Bytes
+ TupleKey));
+ }
+
+
+ for (int j = 1 ; j < nNoOfCol ; j++)
+ tmpTable.addAttribute(NDBT_Attribute(attrName[j], UnSigned, 4*tAttributeSize)) ;
+
+ if(tmpTable.createTableInDb(pMyNdb) == -1){
+ return -1 ;
+ }
+
+ ndbout << "done" << endl ;
+
+ return 0;
+}
+*/
+
+/*************************************************
+Function: createTables()
+Uses NDB API to create tables for the tests
+*************************************************/
+
+int createTables(char* szTableName, int nTables){
+
+ Ndb * pNdb = new Ndb("TEST_DB") ;
+ pNdb->init();
+
+ ndbout << "Waiting for ndb to become ready..." <<endl;
+ if (pNdb->waitUntilReady(10000) != 0){
+ ndbout << "NDB is not ready" << endl;
+ ndbout << "Benchmark failed!" << endl;
+ delete pNdb ;
+ return -1 ;
+ }
+
+ NdbSchemaCon *MySchemaTransaction = NULL ;
+ NdbSchemaOp *MySchemaOp = NULL ;
+ int check = -1 ;
+ char szColNameBuffer[MAX_COL_NAME] = { 0 } ;
+ int tLoadFactor = 80 ;
+
+ for(int i=0 ; i < nTables ; ++i) {
+
+ ndbout << "Creating " << (char*)(szTableName+MAX_TABLE_NAME*i) << "..." << endl ;
+
+ MySchemaTransaction = pNdb->startSchemaTransaction() ;
+ //printf("MySchemaTransaction - OK\n") ;
+ if(MySchemaTransaction == NULL){
+ printf("MySchemaTransaction is NULL\n") ;
+ delete pNdb ;
+ return -1 ;
+ }
+
+ MySchemaOp = MySchemaTransaction->getNdbSchemaOp();
+ //printf("MySchemaTransaction->getNdb... - OK\n") ;
+ if(MySchemaOp == NULL){
+ printf("MySchemaOp is NULL\n") ;
+ delete pNdb ;
+ return -1 ;
+ }
+
+ check = MySchemaOp->createTable( (const char*)(szTableName+MAX_TABLE_NAME*i)
+ ,8 // Table Size
+ ,TupleKey // Key Type
+ ,40 // Nr of Pages
+ ,All
+ ,6
+ ,(tLoadFactor - 5)
+ ,(tLoadFactor)
+ ,1
+ ,0
+ );
+
+ if (check == -1){
+ printf("MySchemaOp->createTable failed\n") ;
+ delete pNdb ;
+ return -1 ;
+ }
+
+ snprintf(szColNameBuffer, MAX_COL_NAME, "COL%d", 0) ;
+ check = MySchemaOp->createAttribute( szColNameBuffer,
+ TupleKey,
+ 32,
+ PKSIZE,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if (check == -1){
+ printf("MySchemaOp->createAttribute() #1 failed\n") ;
+ delete pNdb ;
+ return -1 ;
+ }
+
+ for (int j = 1; j < nNoOfCol ; j++){
+ snprintf(szColNameBuffer, MAX_COL_NAME, "COL%d", j) ;
+ check = MySchemaOp->createAttribute(szColNameBuffer,
+ NoKey,
+ 32,
+ tAttributeSize,
+ UnSigned,
+ MMBased,
+ NotNullAttribute );
+
+ if (check == -1){
+ printf("MySchemaOp->createAttribute() #2 failed\n") ;
+ delete pNdb ;
+ return -1;
+ }
+ }
+
+ if (MySchemaTransaction->execute() == -1){
+ printf("MySchemaTransaction->execute() failed\n") ;
+ printf("%s\n", MySchemaTransaction->getNdbError().message) ;
+ return -1 ;
+ delete pNdb ;
+ }
+
+ pNdb->closeSchemaTransaction(MySchemaTransaction);
+ }
+
+ return 0;
+}
+
+
+
diff --git a/ndb/test/odbc/SQL99_test/SQL99_test.h b/ndb/test/odbc/SQL99_test/SQL99_test.h
new file mode 100644
index 00000000000..1c49f4a9a51
--- /dev/null
+++ b/ndb/test/odbc/SQL99_test/SQL99_test.h
@@ -0,0 +1,261 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <ndb_types.h>
+#include <NdbThread.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NDBT.hpp>
+#include <sqlext.h>
+#include <stdio.h>
+//#include <stdlib.h>
+#include <unistd.h>
+//#include <windows.h>
+//#include <process.h>
+
+#define MAX_STR_LEN 128
+#define MAX_TABLE_NAME 32
+#define MAX_COL_NAME 32
+#define MAX_SQL_STMT 2048
+#define MAX_VALUE_LEN 32
+#define MAX_CHAR_ATTR_LEN 24
+#define NUM_COL_ARITHM 2
+#define FLTDEV 0.0001
+//#define DBLDEV 0.000000001
+
+#define REPORTERROR(fn, str) ReportError(fn, str, __FILE__, __LINE__)
+#define REPORT(str) printf((str))
+
+#define ATTR_TYPE_SWITCH(buffer, ptr, attr) switch(attr){ \
+ case T_INT:\
+ sprintf((char*)(buffer),"%d", (int)(ptr)) ;\
+ break ;\
+ case T_FLOAT:\
+ sprintf((char*)(buffer),"%f", (float)(ptr)) ;\
+ break ;\
+ default:\
+ break ;\
+ }
+
+#define ATTR_TYPE_SWITCH_T(value, attr) switch(attr){ \
+ case T_INT:\
+ printf("%d \t", (int)(value)) ;\
+ break ;\
+ case T_FLOAT:\
+ printf("%f \t", (float)(value)) ;\
+ break ;\
+ default:\
+ break ;\
+ }
+
+#define ATTR_TYPE_SWITCH_AGR(str, value_A, value_B, value_C, attr) switch(attr){ \
+ case T_INT:\
+ printf("%s\t%d %d\t\t\t%d\n\n", str, value_A, (int)value_B, (int)value_C) ; break ;\
+ case T_FLOAT:\
+ printf("%s\t%d %f\t\t\t%d\n\n", str, value_A, value_B, (int)value_C) ; break ;\
+ default:\
+ break ;\
+ }
+
+
+#define ODBC_FN(fn, rc) rc = ((((fn)))) ; if(SQL_SUCCESS == rc || SQL_SUCCESS_WITH_INFO == rc){;}else ReportError("ODBC function", "failed in ", __FILE__, __LINE__)
+
+
+typedef enum attr_type_tag {
+ T_INT,
+ T_FLOAT,
+// T_DOUBLE,
+ T_CHAR
+} attr_type ;
+
+typedef enum aggr_fn_tag {
+ FN_COUNT,
+ FN_SUM,
+ FN_AVG,
+ FN_MAX,
+ FN_MIN,
+ FN_VARIANCE,
+ FN_STDDEV
+} aggr_fn ;
+
+typedef enum join_type_tag {
+ ITSELF,
+ EQUI,
+ NON_EQUI,
+ INNER,
+ OUTTER
+} join_type ;
+
+typedef enum arth_op_tag {
+ MINUS,
+ PLUS,
+ MULTI,
+ DIVIDE,
+ MODULO
+} arth_op ;
+
+typedef struct ODBC_HANDLES_tag{
+ SQLHENV henv ;
+ SQLHDBC hdbc ;
+ SQLHSTMT hstmt ;
+} ODBC_HANDLES ;
+
+typedef enum handle_op_tag{
+ GET,
+ FREE
+} handle_op ;
+
+typedef enum test_case_tag {
+ NUMERIC_DATA_TYPES,
+ CHAR_DATA_TYPES,
+ IDENTIFIERS,
+ BASIC_QUERY,
+ PREDICATE_SEARCH,
+ DATA_MANIPULATION,
+ NULL_SUPPORT,
+ BASIC_CONSTRAINTS,
+ TRANSACTION,
+ SET_FUNCTIONS,
+ BASIC_SCHEMA,
+ JOINED_TABLE,
+ ALL
+} test_case ;
+
+typedef enum status_tag{
+ S_STOP,
+ S_IDLE,
+ S_STARTED,
+ S_GET_BUSY,
+ S_BUSY,
+ S_EXIT
+} status ;
+
+typedef enum type_tag {
+ T_INSERT,
+ T_READ,
+ T_UPDATE,
+ T_DELETE,
+ T_READ_VERIFY,
+ T_DELETE_VERIFY,
+ T_WAIT
+} type ;
+
+typedef struct PARAMS_tag {
+ int nThreadID ;
+ int nError ;
+ int nVerifyFlag ;
+ status thread_status ;
+ status report_status ;
+ type op_type ;
+ void* pThreadRef ;
+ char szTableName[MAX_TABLE_NAME] ;
+} PARAMS ;
+
+typedef enum table_opt_tag {
+ CREATE,
+ DROP
+} table_opt ;
+
+static char szANSI[] ="0123456789ABCEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz" ;
+
+void ReportError(char* szFn, char* szBuffer, char* szFile, int iLine) ;
+void HandleError(void*, SQLSMALLINT) ;
+SQLRETURN GetHandles(ODBC_HANDLES*, handle_op, bool) ;
+SQLRETURN AggregateFn(aggr_fn, char*, int, double*, double*, attr_type) ;
+SQLRETURN GetDriverAndSourceInfo(SQLHDBC) ;
+SQLRETURN Join(char*, join_type) ;
+SQLRETURN GetResults(SQLHSTMT) ;
+int ArithOp(char*, int, float*, attr_type, arth_op) ;
+void ParseArguments(int argc, const char** argv) ;
+void* ThreadFnInt(void*) ;
+void* ThreadFnFloat(void*) ;
+//void* ThreadFnDouble(void*) ;
+void* ThreadFnChar(void*) ;
+inline void AssignTableNames(char* szBuffer, int nTables) ;
+SQLRETURN CreateDemoTables(char*, int, table_opt) ;
+inline void StartThreads(PARAMS*, void*, int, char*, attr_type, UintPtr*) ;
+inline void SetThreadOperationType(PARAMS*, type) ;
+inline int WaitForThreads(PARAMS*) ;
+inline void StopThreads(PARAMS*, UintPtr*) ;
+inline void PrintAll(char* szTableName, int, attr_type) ;
+void AssignRefCharValues(char*, bool) ;
+
+template <class T, class V>
+int VerifyArthOp(V* tValue, float* tOperand, T* tRes, arth_op op){
+
+ int nResult = 0 ;
+ int nValue = 0, nOperand = 0 ;
+
+ switch(op){
+ case MINUS:
+ if(FLTDEV < abs((*tValue - *tOperand) - *tRes))
+ nResult = -1 ;
+ break ;
+ case PLUS:
+ if(FLTDEV < abs((*tValue + *tOperand) - *tRes))
+ nResult = -1 ;
+ break ;
+ case MULTI:
+ if(FLTDEV < abs((*tValue * *tOperand) - *tRes))
+ nResult = -1 ;
+ break ;
+ case DIVIDE:
+ if(FLTDEV < abs((*tValue / *tOperand) - *tRes))
+ nResult = -1 ;
+ break ;
+ case MODULO:
+ nValue = *tValue ;
+ nOperand = *tOperand ;
+ if(*tRes != (nValue % nOperand))
+ nResult = -1 ;
+ break ;
+ }
+
+ return nResult ;
+}
+
+template <class P> void AssignRefNumValues(P* pRef, attr_type attrType, bool bVerbose) {
+
+ int count = 0, rows = 0, nThreadOffset = 0, nRowOffset = 0 ;
+ P* p = (P*)pRef ;
+
+ float fRandomBase = (rand()*rand()) % 100;
+ for(int c = 0 ; c < nNoOfThreads ; ++c){
+ nThreadOffset = nNoOfRows*nNoOfCol*c ;
+ for(int d = 0 ; d < nNoOfRows ; ++d){
+ nRowOffset = nNoOfCol*d ; ++rows ;
+ for(int i = 0 ; i < nNoOfCol ; ++i){
+ (p[nThreadOffset + nRowOffset + i]) = (fRandomBase*(c+1) + (d+3)*7 + i)/1.1034093201 ;
+ ++count ;
+ if(bVerbose){
+ ATTR_TYPE_SWITCH_T(p[nThreadOffset + nRowOffset + i], AttributeType) ;
+ }
+ }
+ if(bVerbose) { printf("\n") ; NdbSleep_MilliSleep(10) ;
+ }
+ }
+ }
+
+ if(bVerbose){
+ printf("_____________________") ;
+ printf("\nRows: %d Values: %d\n\n", rows, count) ;
+ }
+
+ return ;
+}
+
+
diff --git a/ndb/test/odbc/client/Makefile b/ndb/test/odbc/client/Makefile
new file mode 100644
index 00000000000..4b962f5b65a
--- /dev/null
+++ b/ndb/test/odbc/client/Makefile
@@ -0,0 +1,95 @@
+include .defs.mk
+
+TYPE := odbcclient
+#TYPE := odbcdriver
+
+BIN_TARGET := testOdbcClient
+#BIN_TARGET := testodbc2
+
+
+# Source files of non-templated classes (.C files)
+SOURCES = main.cpp \
+ SQLFetchTest.cpp \
+ SQLDisconnectTest.cpp \
+ SQLTablesTest.cpp \
+ SQLGetInfoTest.cpp \
+ SQLGetTypeInfoTest.cpp \
+ SQLGetFunctionsTest.cpp \
+ SQLGetDataTest.cpp \
+ SQLCancelTest.cpp \
+ SQLTransactTest.cpp \
+ SQLGetCursorNameTest.cpp \
+ SQLSetCursorNameTest.cpp \
+ SQLRowCountTest.cpp \
+ SQLNumResultColsTest.cpp \
+ SQLDescribeColTest.cpp \
+ SQLExecDirectTest.cpp \
+ SQLColAttributeTest.cpp \
+ SQLColAttributeTest1.cpp \
+ SQLColAttributeTest2.cpp \
+ SQLColAttributeTest3.cpp \
+ SQLBindColTest.cpp \
+ SQLDriverConnectTest.cpp \
+ SQLPrepareTest.cpp \
+ SQLGetDiagRecSimpleTest.cpp \
+ SQLConnectTest.cpp
+
+XSOURCES = testodbc2.cpp
+XSOURCES = \
+ main.cpp \
+ SQLDriverConnectTest.cpp \
+ SQLPrepareTest.cpp \
+ SQLMoreResultsTest.cpp \
+ SQLGetStmtAttrTest.cpp \
+ SQLGetEnvAttrTest.cpp \
+ SQLGetConnectAttrTest.cpp \
+ SQLExecuteTest.cpp \
+ SQLExecDirectTest.cpp \
+ SQLDisconnectTest.cpp \
+ SQLCloseCursorTest.cpp \
+ SQLCancelTest.cpp \
+ SQLBindColTest.cpp \
+ SQLDescribeColTest.cpp \
+ SQLGetTypeInfoTest.cpp \
+ SQLGetFunctionsTest.cpp \
+ SQLNumResultColsTest.cpp \
+ SQLSetDescFieldTest.cpp \
+ SQLGetDescRecTest.cpp \
+ SQLEndTranTest.cpp \
+ SQLGetInfoTest.cpp \
+ SQLConnectTest.cpp \
+ SQLAllocHandleTest.cpp \
+ SQLAllocEnvTest.cpp \
+ SQLRowCountTest.cpp \
+ SQLFetchScrollTest.cpp \
+ SQLFetchTest.cpp \
+ SQLGetDescFieldTest.cpp \
+ SQLSetDescRecTest.cpp \
+ SQLFreeHandleTest.cpp
+
+ifeq ($(TYPE),odbcdriver)
+LIBS_SPEC += \
+ -lodbcdriver_pic \
+ -lodbchandles_pic \
+ -lodbccodegen_pic \
+ -lodbccompiler_pic \
+ -lodbcexecutor_pic \
+ -lodbccommon_pic \
+ -lodbcdictionary_pic \
+ -lNDBT \
+ -lportlib
+endif
+
+ifeq ($(TYPE),odbcclient)
+LIBS_SPEC += \
+ -lportlib \
+ -lNDBT
+endif
+
+CCFLAGS_LOC += -I/usr/local/include \
+ -I$(NDB_TOP)/include/ndbapi \
+ -I$(NDB_TOP)/test/include
+
+include $(NDB_TOP)/Epilogue.mk
+#LIBS_LOC += -L/usr/local/opt/iODBC/lib
+#LIBS_SPEC = -liodbc -lNDBT -lportlib
diff --git a/ndb/test/odbc/client/NDBT_ALLOCHANDLE.cpp b/ndb/test/odbc/client/NDBT_ALLOCHANDLE.cpp
new file mode 100644
index 00000000000..336f4a46554
--- /dev/null
+++ b/ndb/test/odbc/client/NDBT_ALLOCHANDLE.cpp
@@ -0,0 +1,53 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "common.h"
+#include <NdbTest.hpp>
+#include <NdbMain.h>
+
+SQLRETURN SQLHENV_check, SQLHENV_FREE_check;
+
+int NDBT_ALLOCHANDLE()
+{
+ /*****************************HENV Handle*****************************/
+
+ SQLHENV henv;
+ SQLHENV_check = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+
+ if (SQLHENV_check == -1) {
+ return(-1);
+ //return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ if (SQLHENV_check == 0) {
+ return 0;
+ }
+
+ SQLHENV_FREE_check = SQLFreeHandle(SQL_HANDLE_ENV, henv);
+
+ if (SQLHENV_FREE_check == -1) {
+ // Deallocate any allocated memory, if it exists
+ return(-1);
+ //return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ if (SQLHENV_FREE_check == 0) {
+ return 0;
+ }
+
+ return 0;
+}
+
diff --git a/ndb/test/odbc/client/NDBT_ALLOCHANDLE_HDBC.cpp b/ndb/test/odbc/client/NDBT_ALLOCHANDLE_HDBC.cpp
new file mode 100644
index 00000000000..8477a71edbf
--- /dev/null
+++ b/ndb/test/odbc/client/NDBT_ALLOCHANDLE_HDBC.cpp
@@ -0,0 +1,59 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "common.h"
+#include <NdbTest.hpp>
+#include <NdbMain.h>
+
+SQLRETURN SQLHENVFREE_check, SQLHDBC_check;
+
+
+// NDB_COMMAND(SQLTest1, ......., 65535)
+int NDBT_ALLOCHANDLE_HDBC()
+{
+
+ SQLHENV henv;
+ SQLHDBC hdbc;
+
+ /*****************************HDBC Handle*****************************/
+ SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+
+ SQLHDBC_check = SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);
+
+ if (SQLHDBC_check == -1) {
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ if (SQLHDBC_check == 0) {
+ return 0;
+ }
+
+ SQLHENVFREE_check = SQLFreeHandle(SQL_HANDLE_ENV, henv);
+
+ if (SQLHENVFREE_check == -1) {
+ // Deallocate any allocated memory, if it exists
+ return(-1);
+ //return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ if (SQLHENVFREE_check == 0) {
+ return 0;
+ }
+}
+
+
+
+
diff --git a/ndb/test/odbc/client/NDBT_SQLConnect.cpp b/ndb/test/odbc/client/NDBT_SQLConnect.cpp
new file mode 100644
index 00000000000..da97ffebea4
--- /dev/null
+++ b/ndb/test/odbc/client/NDBT_SQLConnect.cpp
@@ -0,0 +1,82 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "common.h"
+#include <NdbTest.hpp>
+#include <NdbMain.h>
+
+SQLRETURN retcode, SQLSTATEs;
+SQLHENV henv;
+SQLHDBC hdbc;
+
+void NDBT_Connect_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int NDBT_SQLConnect()
+{
+
+ /*****************************SQLConnect AutoTest*****************************/
+
+ // Allocate An Environment Handle
+ SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+
+ // This part does not include in sqlcli.h, it is only in ODBC
+ // Set the ODBC application Version to 3.x
+ // SQLSetEnvattr(henv, SQL_ATTR_ODBC_VERSION, (SQLPOINTER) SQL_OV_ODBC3, SQL_IS_UINTERGER);
+
+ // Allocate A Connection Handle
+ SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);
+
+ // Connect to NDB
+ retcode = SQLConnect(hdbc,
+ (SQLCHAR*) "Sales",
+ 5,
+ (SQLCHAR*) "JohnS",
+ 5,
+ (SQLCHAR*) "Sesame",
+ 6);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+ else
+ { if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ NDBT_Connect_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+ // Free the Connection Handle
+ SQLFreeHandle(SQL_HANDLE_DBC, hdbc);
+
+ // Free the Environment Handle
+ SQLFreeHandle(SQL_HANDLE_ENV, henv);
+
+ return 0;
+}
+
+
+void NDBT_Connect_DisplayError(SQLSMALLINT HandleType, SQLHDBC InputHandle)
+{
+ SQLRETURN Sqlstate;
+ int i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
diff --git a/ndb/test/odbc/client/NDBT_SQLPrepare.cpp b/ndb/test/odbc/client/NDBT_SQLPrepare.cpp
new file mode 100644
index 00000000000..4aaff6a7df9
--- /dev/null
+++ b/ndb/test/odbc/client/NDBT_SQLPrepare.cpp
@@ -0,0 +1,109 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <sqlext.h>
+#include <stdio.h>
+
+#include <NdbTest.hpp>
+#include <NdbMain.h>
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN SQLPrepare_retcode, SQLAllocHandl_retcode, SQLSTATEs;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void NDBT_SQLPrepare_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+ // Execute a statement to retrieve rows from the Customers table. We can
+ // create the table and inside rows into NDB by invoking SQLExecute() or
+ // another program called TestDirectSQL
+
+int NDBT_SQLPrepare()
+{
+ // Allocate An Environment Handle
+ SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+
+ // Allocate A Connection Handle
+ SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);
+
+ // Allocate A Connection Handle
+ SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);
+
+ // Connecte to database
+ SQLConnect(hdbc, (SQLCHAR*) "Sales", 5, (SQLCHAR*) "JohnS", 5, (SQLCHAR*) "Sesame", 6);
+
+ // Allocate A Statement Handle
+ SQLAllocHandl_retcode = SQLAllocHandle(SQL_HANDLE_STMT, hdbc, &hstmt);
+
+ /* We can change the SQL statement in the SQLPrepare() function according to the requirement of Johnny. */
+ /* The order of the SQL statement could be CREATE, INSERT, UPDATE, SELECT, DELETE or another special SQL */
+
+ if (SQLAllocHandl_retcode == SQL_SUCCESS){
+ SQLPrepare_retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", 56);
+
+ if (SQLPrepare_retcode == SQL_INVALID_HANDLE)
+ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE and SQL_SUCCESS still appeared. Please check programm" << endl;
+
+ if (SQLPrepare_retcode == SQL_ERROR || SQLPrepare_retcode == SQL_SUCCESS_WITH_INFO)
+ NDBT_SQLPrepare_DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ SQLExecute(hstmt);
+
+ SQLFreeHandle(SQL_HANDLE_STMT, hstmt);
+ }
+
+ // Disconnect from the database before free Connection Handle and Environment Handle
+ SQLDisconnect(hdbc);
+
+ // Free the Connection Handle
+ SQLFreeHandle(SQL_HANDLE_DBC, hdbc);
+
+ // Free the Environment Handle
+ SQLFreeHandle(SQL_HANDLE_ENV, henv);
+
+ return 0;
+
+ }
+
+
+void NDBT_SQLPrepare_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLAllocEnvTest.cpp b/ndb/test/odbc/client/SQLAllocEnvTest.cpp
new file mode 100644
index 00000000000..ce50c4b7ccd
--- /dev/null
+++ b/ndb/test/odbc/client/SQLAllocEnvTest.cpp
@@ -0,0 +1,115 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "common.h"
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHENV henv;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+
+void sqlallocenv_deal_with_HENV(SQLSMALLINT HandleType, SQLHENV InputHandle);
+void sqlallocenv_deal_with_HDBC(SQLSMALLINT HandleType, SQLHDBC InputHandle);
+
+void DisplayError_HDBC_free(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDBC InputHandle);
+void DisplayError_HENV_free(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHENV InputHandle);
+
+
+int SQLAllocEnvTest()
+{
+
+/* Environment test for SQLAllocEnv() */
+ndbout << "Environment test for SQLAllocEnv()" << endl;
+//SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+
+sqlallocenv_deal_with_HENV(SQL_HANDLE_ENV, henv);
+
+//SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+//SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);
+sqlallocenv_deal_with_HDBC(SQL_HANDLE_DBC, hdbc);
+
+return 0;
+
+}
+
+void sqlallocenv_deal_with_HDBC(SQLSMALLINT HandleType, SQLHDBC InputHandle)
+{
+ SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+ retcode = SQLAllocHandle(HandleType, henv, &InputHandle);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+ ndbout << "the InputHandle is SQLHDBC:" << InputHandle << endl;
+ ndbout << "retcode = " << retcode << endl;
+
+ /* ***
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ DisplayError_HDBC_free(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ *** */
+}
+
+void sqlallocenv_deal_with_HENV(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ retcode = SQLAllocEnv(&InputHandle);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+ ndbout << "the InputHandle is SQLHENV:" << InputHandle << endl;
+ ndbout << "retcode = " << retcode << endl;
+ /*
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ DisplayError_HENV_free(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ */
+ }
+
+
+void DisplayError_HENV_free(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+void DisplayError_HDBC_free(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDBC InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
diff --git a/ndb/test/odbc/client/SQLAllocHandleTest.cpp b/ndb/test/odbc/client/SQLAllocHandleTest.cpp
new file mode 100644
index 00000000000..0c51e2e46b7
--- /dev/null
+++ b/ndb/test/odbc/client/SQLAllocHandleTest.cpp
@@ -0,0 +1,314 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "common.h"
+
+using namespace std;
+
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+long strangehandle;
+
+void handle_deal_with_HSTMT(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+void handle_deal_with_HENV(SQLSMALLINT HandleType, SQLHENV InputHandle);
+void handle_deal_with_HDESC(SQLSMALLINT HandleType, SQLHDESC InputHandle);
+void handle_deal_with_HDBC(SQLSMALLINT HandleType, SQLHDBC InputHandle);
+//void handle_deal_with_int(SQLSMALLINT HandleType, long InputHandle);
+
+void DisplayError_HDBC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDBC InputHandle);
+void DisplayError_HSTMT(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+void DisplayError_HENV(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHENV InputHandle);
+void DisplayError_HDESC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDESC InputHandle);
+//void DisplayError_int(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, int InputHandle);
+
+int SQLAllocHandleTest()
+{
+
+strangehandle = 6;
+
+/*Allocate environment handle */
+
+//retcode = SQLFreeHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE);
+
+/* ENV */
+ndbout << endl;
+ndbout << "The HandleType: Allocate Environment handle" << endl;
+ndbout << endl;
+
+handle_deal_with_HENV(SQL_HANDLE_ENV, SQL_NULL_HANDLE);
+
+handle_deal_with_HENV(SQL_HANDLE_ENV, henv);
+
+handle_deal_with_HDBC(SQL_HANDLE_ENV, hdbc);
+
+handle_deal_with_HSTMT(SQL_HANDLE_ENV, hstmt);
+
+handle_deal_with_HDESC(SQL_HANDLE_ENV, hdesc);
+
+//handle_deal_with_int(SQL_HANDLE_ENV, strangehandle);
+
+/* DBC */
+ndbout << endl;
+ndbout << "The HandleType: Allocate Connection handle" << endl;
+ndbout << endl;
+
+handle_deal_with_HDBC(SQL_HANDLE_DBC, SQL_NULL_HANDLE);
+
+handle_deal_with_HENV(SQL_HANDLE_DBC, henv);
+
+handle_deal_with_HDBC(SQL_HANDLE_DBC, hdbc);
+
+handle_deal_with_HSTMT(SQL_HANDLE_DBC, hstmt);
+
+handle_deal_with_HDESC(SQL_HANDLE_DBC, hdesc);
+
+//handle_deal_with_int(SQL_HANDLE_DBC, strangehandle);
+
+/* STMT */
+ndbout << endl;
+ndbout << "The HandleType: Allocate Statement handle" << endl;
+ndbout << endl;
+
+handle_deal_with_HSTMT(SQL_HANDLE_STMT, SQL_NULL_HANDLE);
+
+handle_deal_with_HENV(SQL_HANDLE_STMT, henv);
+
+handle_deal_with_HDBC(SQL_HANDLE_STMT, hdbc);
+
+handle_deal_with_HSTMT(SQL_HANDLE_STMT, hstmt);
+
+handle_deal_with_HDESC(SQL_HANDLE_STMT, hdesc);
+
+//handle_deal_with_int(SQL_HANDLE_STMT, strangehandle);
+
+
+/* DESC */
+ndbout << endl;
+ndbout << "The HandType: Allocate Descriptor handle" << endl;
+ndbout << endl;
+
+handle_deal_with_HDESC(SQL_HANDLE_DESC, SQL_NULL_HANDLE);
+
+handle_deal_with_HENV(SQL_HANDLE_DESC, henv);
+
+handle_deal_with_HDBC(SQL_HANDLE_DESC, hdbc);
+
+handle_deal_with_HSTMT(SQL_HANDLE_DESC, hstmt);
+
+handle_deal_with_HDESC(SQL_HANDLE_DESC, hdesc);
+
+//handle_deal_with_int(SQL_HANDLE_DESC, strangehandle);
+
+
+/* strangehandle */
+ndbout << endl;
+ndbout << "The HandType: strangehandle" << endl;
+ndbout << endl;
+
+//handle_deal_with_int(strangehandle, SQL_NULL_HANDLE);
+
+handle_deal_with_HENV(strangehandle, henv);
+
+handle_deal_with_HDBC(strangehandle, hdbc);
+
+handle_deal_with_HSTMT(strangehandle, hstmt);
+
+handle_deal_with_HDESC(strangehandle, hdesc);
+
+// handle_deal_with_int(strangehandle, strangehandle);
+
+return 0;
+
+}
+
+void handle_deal_with_HDBC(SQLSMALLINT HandleType, SQLHDBC InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLAllocHandle(HandleType, InputHandle, &hdbc);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+
+ ndbout << "the InputHandle is SQLHDBC:" << InputHandle << endl;
+
+ ndbout << "return &hdbc: " << (long)&hdbc << endl;
+ ndbout << "the retcode state is:" << retcode << endl;
+ ndbout << endl;
+
+ /*
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ DisplayError_HDBC(Sqlstate, HandleType, InputHandle);
+ i ++;
+ }
+ }
+ */
+ }
+
+
+void handle_deal_with_HSTMT(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLAllocHandle(HandleType, InputHandle, &hstmt);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+
+ ndbout << "the InputHandle is SQLHSTMT:" << InputHandle << endl;
+
+ ndbout << "return &hstmt: " << (long)&hstmt << endl;
+ ndbout << "the output retcode is:" << retcode << endl;
+ ndbout << endl;
+ /*
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ DisplayError_HSTMT(Sqlstate, HandleType, InputHandle);
+ i ++;
+ }
+ }
+ */
+ }
+
+void handle_deal_with_HENV(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLAllocHandle(HandleType, InputHandle, &henv);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+
+ ndbout << "the InputHandle is SQLHENV:" << InputHandle << endl;
+
+ ndbout << "return &henv: " << (long)&henv << endl;
+ ndbout << "the output retcode is:" << retcode << endl;
+ ndbout << endl;
+ /*
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ DisplayError_HENV(Sqlstate, HandleType, InputHandle);
+ i ++;
+ }
+ }
+ */
+ }
+
+void handle_deal_with_HDESC(SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLAllocHandle(HandleType, InputHandle, &hdesc);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+
+ ndbout << "the InputHandle is SQLHDESC:" << InputHandle << endl;
+
+ ndbout << "return &hdesc: " << (long)&hdesc << endl;
+ ndbout << "the output retcode is:" << retcode << endl;
+ ndbout << endl;
+ /*
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ DisplayError_HDESC(Sqlstate, HandleType, InputHandle);
+ i ++;
+ }
+ }
+ */
+ }
+
+
+//void handle_deal_with_int(SQLSMALLINT HandleType, long InputHandle)
+//{
+// SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+// retcode = SQLAllocHandle(HandleType, InputHandle, &InputHandle);
+//
+// ndbout << "the HandleType is: " << HandleType << endl;
+//
+// ndbout << "the InputHandle is stranghandle:" << InputHandle << endl;
+//
+// ndbout << "return &InputHandle: " << (long)&InputHandle << endl;
+// ndbout << "the output retcode is:" << retcode << endl;
+// ndbout << endl;
+// /*
+// if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+// i = 1;
+// while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+// Sqlstate, &NativeError, Msg, sizeof(Msg),
+// &MsgLen)) != SQL_NO_DATA) {
+//
+// DisplayError_int(Sqlstate, HandleType, InputHandle);
+//
+// i ++;
+// }
+// }
+// */
+// }
+
+
+void DisplayError_HENV(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+
+void DisplayError_HDBC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDBC InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+void DisplayError_HSTMT(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+void DisplayError_HDESC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+//void DisplayError_int(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, int InputHandle)
+//{
+// ndbout << "the HandleType is:" << HandleType << endl;
+// ndbout << "the InputHandle is :" << InputHandle << endl;
+// ndbout << "the output state is:" << (char *)Sqlstate << endl;
+//}
diff --git a/ndb/test/odbc/client/SQLAllocHandleTest_bf.cpp b/ndb/test/odbc/client/SQLAllocHandleTest_bf.cpp
new file mode 100644
index 00000000000..7786675243a
--- /dev/null
+++ b/ndb/test/odbc/client/SQLAllocHandleTest_bf.cpp
@@ -0,0 +1,259 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include sqlcli.h;
+#include stdio.h;
+
+#define SQL_MAX_MESSAGE_LENGTH 200;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR SqlState[6], SQLStmt[100], Msg[SQL_MAX_MESSAGE_LENGTH];
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+
+struct handle_set
+{
+SQLHDBC hdbc_varible;
+SQLHSTMT hstmt_varible;
+SQLHENV henv_varible;
+SQLHDESC hdesc_varible;
+INTEGER strangehandle;
+}
+
+static int
+check(
+ SQLSMALLINT HandleType,
+ SQLHANDLE inputhandle,
+ SQLHANDLE *outputhandle,
+ SQLRETURN wantret,
+ char *wantSqlstate)
+{
+ SQLRETURN ret;
+ SQLCHAR Sqlstate[20];
+
+ ret = SQLAllocHandle(handletype, inputhandle, outputhandle);
+ if (ret != wantret) {
+ // report error
+ return -1;
+ }
+ if (ret == SQL_INVALID_HANDLE) {
+ // cannot get diag
+ return 0;
+ }
+ // TODO
+ ret = SQLGetDiagRec(HandleType, InputHandle, 1, Sqlstate, &NativeError, Msg, sizeof(Msg), &MsgLen);
+ if (strcmp(Sqlstate, wantSqlstate) != 0) {
+ // report error;
+ return -1;
+ }
+ return 0;
+}
+
+int
+Test_SQLAllocHandle()
+{
+ SQLRETURN ret;
+ SQLHENV henv;
+ SQLDBC dbc;
+ int i;
+
+ // env
+ check(SQL_HANDLE_ENV, SQL_NULL_HANDLE, 0, SQL_ERROR, "HY009");
+ for (i = 0; i < 1000; i++) {
+ if (i != SQL_NULL_HANDLE)
+ check(SQL_HANDLE_ENV, i, &henv, SQL_INVALID_HANDLE, 0);
+ }
+ if (check(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv, SQL_SUCCESS, "00000") < 0)
+ return -1;
+
+ // dbc
+ check(SQL_HANDLE_DBC, henv, 0, SQL_ERROR, "HY009");
+ for (i = 0; i < 1000; i++) {
+ if (i != henv)
+ check(SQL_HANDLE_DBC, i, &dbc, SQL_INVALID_HANDLE, 0);
+ }
+ if (check(SQL_HANDLE_DBC, henv, &dbc, SQL_SUCCESS, "00000") < 0)
+ return -1;
+
+ //??
+ check(SQL_HANDLE_ENV, dbc, 0, SQL_ERROR, "HY092");
+
+ // TODO
+ // stmt
+
+ return 0;
+}
+
+
+handle_set handlevalue;
+
+handlevalue.hdbc_varible = hdbc;
+handlevalue.hstmt_varible = hstmt;
+handlevalue.henv_varible = henv;
+handlevalue.hdesc_varible = hdesc;
+handlevalue.stranghandle = 67;
+
+ /*Allocate environment handle */
+//retcode = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+
+while (int j = 0; j++; j < 6) {
+ if ( j = 0 )
+ handle_deal_with(SQL_HANDLE_ENV, SQL_NULL_HANDLE, );
+
+ else if ( j = 1 )
+ handle_deal_with(SQL_HANDLE_ENV, handlevalue.henv_varible, );
+
+ else if ( j = 2 )
+ handle_deal_with(SQL_HANDLE_ENV, handlevalue.hdbc_varible, );
+
+ else if ( j = 3 )
+ handle_deal_with(SQL_HANDLE_ENV, handlevalue.hstmt_varible, );
+
+ else if ( j = 4 )
+ handle_deal_with(SQL_HANDLE_ENV, handlevalue.hdesc_varible, );
+
+ else
+ handle_deal_with(SQL_HANDLE_ENV, handlevalue.stranghandle, );
+
+ }
+
+ while (int j = 0; j++; j < 6) {
+ if ( j = 0 )
+ handle_deal_with(SQL_HANDLE_DBC, SQL_NULL_HANDLE, );
+
+ else if ( j = 1 )
+ handle_deal_with(SQL_HANDLE_DBC, handlevalue.henv_varible, );
+
+ else if ( j = 2 )
+ handle_deal_with(SQL_HANDLE_DBC, handlevalue.hdbc_varible, );
+
+ else if ( j = 3 )
+ handle_deal_with(SQL_HANDLE_DBC, handlevalue.hstmt_varible, );
+
+ else if ( j = 4 )
+ handle_deal_with(SQL_HANDLE_DBC, handlevalue.hdesc_varible, );
+
+ else
+ handle_deal_with(SQL_HANDLE_DBC, handlevalue.stranghandle, );
+
+ }
+
+
+ while (int j = 0; j++; j < 6) {
+ if ( j = 0 )
+ handle_deal_with(SQL_HANDLE_STMT, SQL_NULL_HANDLE, );
+
+ else if ( j = 1 )
+ handle_deal_with(SQL_HANDLE_STMT, handlevalue.henv_varible, );
+
+ else if ( j = 2 )
+ handle_deal_with(SQL_HANDLE_STMT, handlevalue.hdbc_varible, );
+
+ else if ( j = 3 )
+ handle_deal_with(SQL_HANDLE_STMT, handlevalue.hstmt_varible, );
+
+ else if ( j = 4 )
+ handle_deal_with(SQL_HANDLE_STMT, handlevalue.hdesc_varible, );
+
+ else
+ handle_deal_with(SQL_HANDLE_STMT, handlevalue.stranghandle, );
+
+ }
+
+
+
+ while (int j = 0; j++; j < 6) {
+ if ( j = 0 )
+ handle_deal_with(SQL_HANDLE_DESC, SQL_NULL_HANDLE, );
+
+ else if ( j = 1 )
+ handle_deal_with(SQL_HANDLE_DESC, handlevalue.henv_varible, );
+
+ else if ( j = 2 )
+ handle_deal_with(SQL_HANDLE_DESC, handlevalue.hdbc_varible, );
+
+ else if ( j = 3 )
+ handle_deal_with(SQL_HANDLE_DESC, handlevalue.hstmt_varible, );
+
+ else if ( j = 4 )
+ handle_deal_with(SQL_HANDLE_DESC, handlevalue.hdesc_varible, );
+
+ else
+ handle_deal_with(SQL_HANDLE_DESC, handlevalue.stranghandle, );
+
+ }
+
+ while (int j = 0; j++; j < 6) {
+ if ( j = 0 )
+ handle_deal_with(handlevalue.stranghandle, SQL_NULL_HANDLE, );
+
+ else if ( j = 1 )
+ handle_deal_with(handlevalue.stranghandle, handlevalue.henv_varible, );
+
+ else if ( j = 2 )
+ handle_deal_with(handlevalue.stranghandle, handlevalue.hdbc_varible, );
+
+ else if ( j = 3 )
+ handle_deal_with(handlevalue.stranghandle, handlevalue.hstmt_varible, );
+
+ else if ( j = 4 )
+ handle_deal_with(handlevalue.stranghandle handlevalue.hdesc_varible, );
+
+ else
+ handle_deal_with(handlevalue.stranghandle, handlevalue.stranghandle, );
+
+ }
+
+
+}
+
+
+void DisplayError(SQLCHAR SqlState[6], string SQLSTATE, string flag, SQLSMALLINT HandleType, SQLHANDLE InputHandle)
+{
+cout << "the operation is: " << flag << endl;
+cout << "the HandleType is:" << HandleType << endl;
+cout << "the InputHandle is :"<< InputHandle <<endl;
+cout << "the correct state is:" << SQLSTATE << endl;
+cout << "the output state is:" << Sqlstate << endl;
+}
+
+}
+
+
+void handle_deal_with(SQLSMALLINT HandleType, SQLHANDLE InputHandle, string SQLSTATE)
+{
+ retcode = SQLAllocHandle(HandleType, InputHandle, OutputHandlePtr);
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATE) {
+
+ if (SQLSTATE = Sqlstate )
+ DisplayError(SqlState, SQLSTATE, 'OK');
+
+ else
+ DisplayError(SqlState, SQLSTATE, 'failure');
+
+ i ++;
+ }
+ }
+ }
diff --git a/ndb/test/odbc/client/SQLBindColTest.cpp b/ndb/test/odbc/client/SQLBindColTest.cpp
new file mode 100644
index 00000000000..e2cd4ce73d1
--- /dev/null
+++ b/ndb/test/odbc/client/SQLBindColTest.cpp
@@ -0,0 +1,537 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLBindColTest.cpp
+ */
+#include <common.hpp>
+using namespace std;
+
+#define BindCol_NAME_LEN 10
+#define BindCol_PHONE_LEN 10
+#define BindCol_ADDRESS_LEN 10
+#define BindCol_Price_LEN 10
+#define BindCol_Weight_LEN 10
+#define BindCol_Tax_LEN 10
+
+#define BindCol_SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+//SQLHDBC BindCol_hdbc;
+//SQLHSTMT BindCol_hstmt;
+//SQLHENV BindCol_henv;
+//SQLHDESC BindCol_hdesc;
+//SQLRETURN BCret;
+
+//SQLCHAR BindCol_Name[BindCol_NAME_LEN], BindCol_Phone[BindCol_PHONE_LEN];
+//SQLCHAR BindCol_Address[BindCol_ADDRESS_LEN];
+//SQLINTEGER NativeError;
+//unsigned long BindCol_CustID;
+
+void BindCol_DisplayError(SQLSMALLINT BindCol_HandleType,
+ SQLHSTMT BindCol_InputHandle);
+
+/**
+ * Test setting column to bind
+ * for a column in a result
+ *
+ * -# Bind columns 1
+ * -# Bind columns 2
+ * -# Bind columns 3
+ * -# Bind columns 4
+ * -# Bind columns 5
+ * -# Bind columns 6
+ * -# Bind columns 7
+ * @return Zero, if test succeeded
+ */
+
+int SQLBindColTest()
+{
+
+ SQLHDBC BindCol_hdbc;
+ SQLHSTMT BindCol_hstmt;
+ SQLHENV BindCol_henv;
+ SQLHDESC BindCol_hdesc;
+
+ SQLCHAR SQLStmt1 [240];
+ SQLCHAR SQLStmt2 [240];
+ SQLCHAR SQLStmt3 [120];
+
+ SQLRETURN BCret;
+
+ unsigned long BindCol_CustID;
+ SQLCHAR BindCol_Name[BindCol_NAME_LEN];
+ short BindCol_Account;
+ unsigned short BindCol_Phone;
+ long BindCol_Price;
+ float BindCol_Weight;
+ double BindCol_Tax;
+
+ ndbout << endl << "Start SQLBindCol Testing" << endl;
+
+ //*******************************************************************
+ //** hstmt
+ //** Execute a statement to retrieve rows from the Customers table **
+ //** We can create the table and insert rows into Customers **
+ //*******************************************************************
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ BCret = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &BindCol_henv);
+
+if (BCret == SQL_SUCCESS || BCret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ BCret = SQLSetEnvAttr(BindCol_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+if (BCret == SQL_SUCCESS || BCret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+//**********************************
+//** Allocate A Connection Handle **
+//**********************************
+
+ BCret = SQLAllocHandle(SQL_HANDLE_DBC,
+ BindCol_henv,
+ &BindCol_hdbc);
+
+ if (BCret == SQL_SUCCESS || BCret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ BCret = SQLConnect(BindCol_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (BCret == SQL_SUCCESS || BCret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ BCret = SQLAllocHandle(SQL_HANDLE_STMT,
+ BindCol_hdbc,
+ &BindCol_hstmt);
+ if(BCret == SQL_SUCCESS || BCret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ /* Primary key is Integer and Char */
+ strcpy((char *) SQLStmt1, "CREATE TABLE Customer1(CustID Integer, Name Char(12), Account Char(12), Phone Char(12), Price Char(6), Weight Char(6), Tax Char(6), Primary Key(CustID, Name))");
+
+ strcpy((char *) SQLStmt2, "INSERT INTO Customer1 (CustID, Name, Account, Phone, Price, Weight, Tax) VALUES(588, 'peter','6808','7190890', '5.68', '1.58', '0.88')");
+
+ strcpy((char *) SQLStmt3, "SELECT * FROM Customer1");
+
+ //************************************************
+ //** Prepare and Execute CREATE TABLE statement **
+ //************************************************
+ ndbout << endl << "Prepare and Execute CREATE TABLE statement ......" << endl;
+ ndbout << ">>>>" << (char*)SQLStmt1 << "<<<<" << endl;
+ BCret = SQLExecDirect(BindCol_hstmt,
+ SQLStmt1,
+ SQL_NTS);
+ if (BCret == SQL_SUCCESS)
+ ndbout << "Prepare and Execute CREATE TABLE statement OK!"
+ << endl<< endl;
+
+ if (BCret == SQL_ERROR || BCret == SQL_SUCCESS_WITH_INFO)
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+
+ if (BCret == -2)
+ {
+ ndbout << "BCret = SQLExexDirect()=" << BCret << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+
+ //*******************************************************
+ //** Prepare and Execute INSERT statement with prepare **
+ //*******************************************************
+ ndbout << "Prepare and Execute INSERT statement ......" << endl;
+ ndbout << ">>>>" << (char*)SQLStmt2 << "<<<<" << endl;
+ BCret = SQLExecDirect(BindCol_hstmt,
+ SQLStmt2,
+ SQL_NTS);
+
+ if (BCret == SQL_SUCCESS)
+ ndbout << "Prepare and Execute INSERT statement OK!"
+ << endl << endl;
+
+ if (BCret == SQL_ERROR || BCret == SQL_SUCCESS_WITH_INFO)
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+
+ if (BCret == -2)
+ {
+ ndbout << "BCret = SQLExexDirect()=" << BCret << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+
+ //******************************************
+ //** Prepare and EXECUTE SELECT statement **
+ //******************************************
+ ndbout << "Prepare and Execute SELECT statement ......" << endl;
+ ndbout << ">>>>" << (char*)SQLStmt3 << "<<<<" << endl;
+ BCret = SQLExecDirect(BindCol_hstmt,
+ SQLStmt3,
+ SQL_NTS);
+
+ if (BCret == SQL_SUCCESS)
+ ndbout << "Prepare and Execute SELECT statement OK!"
+ << endl << endl;
+
+ if (BCret == SQL_ERROR || BCret == SQL_SUCCESS_WITH_INFO)
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+
+ if (BCret == -2)
+ {
+ ndbout << "BCret = SQLExexDirect()=" << BCret << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+
+ //*******************************
+ //** Execute SELECT statement **
+ //*******************************
+ // BCret = SQLExecute(BindCol_hstmt);
+ // if (BCret == SQL_ERROR || BCret == SQL_SUCCESS_WITH_INFO)
+ // {
+ // ndbout << "BCret = " << BCret << endl;
+ // BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ // }
+ // else
+ // {
+
+ if (BCret == SQL_SUCCESS)
+ ndbout << "Execute INSERT statement OK!" << endl;
+
+ //*********************
+ //** Test1 **
+ //** Bind columns 1 **
+ //*********************
+
+ BCret =SQLBindCol(BindCol_hstmt,
+ 1,
+ SQL_C_ULONG,
+ &BindCol_CustID,
+ sizeof(BindCol_CustID),
+ NULL);
+
+ if (BCret == SQL_SUCCESS)
+ {
+ ndbout << endl << "Bind col 1 OK!" << endl;
+ }
+ else if (BCret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Bind Col 1 OK but with INFO" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+ else if (BCret == SQL_ERROR)
+ {
+ ndbout << "Bind Col 1 Failed!" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ return NDBT_FAILED;
+ }
+ else
+ ndbout << endl;
+
+ //*********************
+ //** Test2 **
+ //** Bind columns 2 **
+ //*********************
+
+ BCret =SQLBindCol(BindCol_hstmt,
+ 2,
+ SQL_C_CHAR,
+ &BindCol_Name,
+ BindCol_NAME_LEN,
+ NULL);
+
+ if (BCret == SQL_SUCCESS)
+ {
+ ndbout << "Bind col 2 OK!" << endl;
+ }
+ else if (BCret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Bind Col 2 OK but with INFO" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+ else if (BCret == SQL_ERROR)
+ {
+ ndbout << "Bind Col 2 Failed!" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ return NDBT_FAILED;
+ }
+ else
+ ndbout << endl;
+
+ //*********************
+ //** Test3 **
+ //** Bind columns 3 **
+ //*********************
+
+ BCret = SQLBindCol(BindCol_hstmt,
+ 3,
+ SQL_C_USHORT,
+ &BindCol_Account,
+ sizeof(BindCol_Account),
+ NULL);
+
+ if (BCret == SQL_ERROR)
+ {
+ ndbout << "Bind Col 3 Failed!" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ return NDBT_FAILED;
+ }
+ else if (BCret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Bind Col 3 OK but with INFO" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+ else if (BCret == SQL_SUCCESS)
+ {
+ ndbout << "Bind col 3 OK!" << endl;
+ }
+ else
+ ndbout << endl;
+
+ //*********************
+ //** Test4 **
+ //** Bind columns 4 **
+ //*********************
+
+ BCret = SQLBindCol(BindCol_hstmt,
+ 4,
+ SQL_C_USHORT,
+ &BindCol_Phone,
+ sizeof(BindCol_Phone),
+ NULL);
+
+ if (BCret == SQL_ERROR)
+ {
+ ndbout << "Bind Col 4 Failed!" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ return NDBT_FAILED;
+ }
+ else if (BCret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Bind Col 4 OK but with INFO" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+ else if (BCret == SQL_SUCCESS)
+ {
+ ndbout << "Bind col 4 OK!" << endl;
+ }
+ else
+ ndbout << endl;
+
+ //*********************
+ //** Test5 **
+ //** Bind columns 5 **
+ //*********************
+
+ BCret = SQLBindCol(BindCol_hstmt,
+ 5,
+ SQL_C_SLONG,
+ &BindCol_Price,
+ sizeof(BindCol_Price),
+ NULL);
+
+ if (BCret == SQL_ERROR)
+ {
+ ndbout << "Bind Col 5 Failed!" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ return NDBT_FAILED;
+ }
+ else if (BCret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Bind Col 5 OK but with INFO" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+ else if (BCret == SQL_SUCCESS)
+ {
+ ndbout << "Bind col 5 OK!" << endl;
+ }
+ else
+ ndbout << endl;
+
+ //*********************
+ //** Test6 **
+ //** Bind columns 6 **
+ //*********************
+
+ BCret = SQLBindCol(BindCol_hstmt,
+ 6,
+ SQL_C_FLOAT,
+ &BindCol_Weight,
+ sizeof(BindCol_Weight),
+ NULL);
+
+ if (BCret == SQL_ERROR)
+ {
+ ndbout << "Bind Col 6 Failed!" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ return NDBT_FAILED;
+ }
+ else if (BCret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Bind Col 6 OK but with INFO" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+ else if (BCret == SQL_SUCCESS)
+ {
+ ndbout << "Bind col 6 OK!" << endl;
+ }
+ else
+ ndbout << endl;
+
+ //*********************
+ //** Test7 **
+ //** Bind columns 7 **
+ //*********************
+
+ BCret = SQLBindCol(BindCol_hstmt,
+ 7,
+ SQL_C_DOUBLE,
+ &BindCol_Tax,
+ sizeof(BindCol_Tax),
+ NULL);
+
+ if (BCret == SQL_ERROR)
+ {
+ ndbout << "Bind Col 7 Failed!" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ return NDBT_FAILED;
+ }
+ else if (BCret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Bind Col 7 OK but with INFO" << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ }
+ else if (BCret == SQL_SUCCESS)
+ {
+ ndbout << "Bind col 7 OK!" << endl;
+ }
+ else
+ ndbout << endl;
+
+ //}
+
+//*****************************************
+//* Fetch and print each row of data. On **
+//* an error, display a message and exit **
+//*****************************************
+
+BCret = SQLFetch(BindCol_hstmt);
+
+ ndbout << endl << "BCret = SQLFetch(BindCol_hstmt) = "
+ << BCret << endl;
+
+if (BCret == SQL_ERROR)
+{
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+ return NDBT_FAILED;
+}
+else if (BCret == SQL_SUCCESS_WITH_INFO)
+{
+ ndbout << "CustID = " << (int)BindCol_CustID << endl;
+ ndbout << "Name = " << (char *)BindCol_Name << endl;
+ ndbout << "Account = " << (int)BindCol_Account << endl;
+ ndbout << "Phone = " << (int)BindCol_Phone << endl;
+ ndbout << "Price = " << (int)BindCol_Price << endl;
+ ndbout << "Weight = " << (int)BindCol_Weight << endl;
+ ndbout << "Tax = " << (int)BindCol_Tax << endl;
+ BindCol_DisplayError(SQL_HANDLE_STMT, BindCol_hstmt);
+}
+else
+{
+ ndbout << "CustID = " << (int)BindCol_CustID << endl;
+ ndbout << "Name = " << (char *)BindCol_Name << endl;
+ ndbout << "Account = " << (int)BindCol_Account << endl;
+ ndbout << "Phone = " << (int)BindCol_Phone << endl;
+ ndbout << "Price = " << (int)BindCol_Price << endl;
+ ndbout << "Weight = " << (int)BindCol_Weight << endl;
+ ndbout << "Tax = " << (int)BindCol_Tax << endl;
+}
+
+// *********************************
+// ** Disconnect and Free Handles **
+// *********************************
+SQLDisconnect(BindCol_hdbc);
+SQLFreeHandle(SQL_HANDLE_STMT, BindCol_hstmt);
+SQLFreeHandle(SQL_HANDLE_DBC, BindCol_hdbc);
+SQLFreeHandle(SQL_HANDLE_ENV, BindCol_henv);
+
+return NDBT_OK;
+
+}
+
+void BindCol_DisplayError(SQLSMALLINT BindCol_HandleType,
+ SQLHSTMT BindCol_InputHandle)
+{
+ SQLSMALLINT BindCol_i = 1;
+ SQLRETURN BindCol__SQLSTATEs;
+ SQLCHAR BindCol_Sqlstate[5];
+ SQLCHAR BindCol_Msg[BindCol_SQL_MAXIMUM_MESSAGE_LENGTH];
+ SQLSMALLINT BindCol_MsgLen;
+ SQLINTEGER NativeError;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((BindCol__SQLSTATEs = SQLGetDiagRec(BindCol_HandleType,
+ BindCol_InputHandle,
+ BindCol_i,
+ BindCol_Sqlstate,
+ &NativeError,
+ BindCol_Msg,
+ sizeof(BindCol_Msg),
+ &BindCol_MsgLen)
+ ) != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << BindCol_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)BindCol_InputHandle << endl;
+ ndbout << "the BindCol_Msg is: " << (char *) BindCol_Msg << endl;
+ ndbout << "the output state is:" << (char *)BindCol_Sqlstate << endl;
+
+ BindCol_i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
diff --git a/ndb/test/odbc/client/SQLBindParameterTest.cpp b/ndb/test/odbc/client/SQLBindParameterTest.cpp
new file mode 100644
index 00000000000..2ffd2892064
--- /dev/null
+++ b/ndb/test/odbc/client/SQLBindParameterTest.cpp
@@ -0,0 +1,219 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <sqlcli.h>
+#include <stdio.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+
+SQLHSTMT hstmt;
+
+SQLSMALLINT sOrderID;
+SQLSMALLINT sCustID;
+DATE_STRUCT dsOpenDate;
+SQLCHAR szSalesPerson[SALES_PERSON_LEN];
+
+SQLCHAR szStatus[STATUS_LEN],Sqlstate[5], Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLINTEGER cbOrderID = 0, cbCustID = 0, cbOpenDate = 0, cbSalesPerson = SQL_NTS, cbStatus = SQL_NTS, NativeError;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLSMALLINT i, MsgLen;
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int SQLBindParameterTest ()
+{
+
+ /* hstmt */
+ //** Execute a statement to retrieve rows from the Customers table.
+ //** We can create the table and inside rows in
+ //** NDB by another program TestDirectSQL.
+ //** In this test program(SQLBindParameterTest),we only have three rows in
+ //** table ORDERS
+
+ //************************
+ //** Define a statement **
+ //************************
+ strcpy( (char *) SQLStmt,
+ "INSERT INTO Customers (CUSTID, Name, Address, Phone) VALUES (2, 'paul, 'Alzato', '468719989');
+
+/* Prepare the SQL statement with parameter markers. */
+retcode = SQLPrepare(hstmt, SQLStmt, SQL_NTS);
+
+/* Specify data types and buffers for OrderID, CustID, OpenDate, SalesPerson, */
+/* Status parameter data. */
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ /* ParameterNumber is less than 1 */
+retcode = SQLBindParameter(hstmt,
+ 0,
+ SQL_PARAM_INPUT,
+ SQL_C_SSHORT,
+ SQL_INTEGER,
+ 0,
+ 0,
+ &sOrderID,
+ 0,
+ &cbOrderID);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ /* InputOutputMode is not one of the code values in Table 11 */
+retcode = SQLBindParameter(hstmt,
+ 1,
+ 3,
+ SQL_C_SSHORT,
+ SQL_INTEGER,
+ 0,
+ 0,
+ &sOrderID,
+ 0,
+ &cbOrderID);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ /* ParameterType is not one of the code values in Table 37 */
+retcode = SQLBindParameter(hstmt,
+ 1,
+ 3,
+ SQL_C_SSHORT,
+ 114,
+ 0,
+ 0,
+ &sOrderID,
+ 0,
+ &cbOrderID);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DisplayError(SQL_HANDLE_STMT, hstmt);
+
+SQLBindParameter(hstmt,
+ 1,
+ SQL_PARAM_INPUT,
+ SQL_C_SSHORT,
+ SQL_INTEGER,
+ 0,
+ 0,
+ &sOrderID,
+ 0,
+ &cbOrderID);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DisplayError(SQL_HANDLE_STMT, hstmt);
+
+
+SQLBindParameter(hstmt,
+ 2,
+ SQL_PARAM_INPUT,
+ SQL_C_SSHORT,
+ SQL_INTEGER,
+ 0,
+ 0,
+ &sCustID,
+ 0,
+ &cbCustID);
+
+SQLBindParameter(hstmt,
+ 3,
+ SQL_PARAM_INPUT,
+ SQL_C_TYPE_DATE,
+ SQL_TYPE_DATE,
+ 0,
+ 0,
+ &dsOpenDate,
+ 0,
+ &cbOpenDate);
+
+SQLBindParameter(hstmt,
+ 4,
+ SQL_PARAM_INPUT,
+ SQL_C_CHAR,
+ SQL_CHAR,
+ SALES_PERSON_LEN,
+ 0,
+ szSalesPerson,
+ 0,
+ &cbSalesPerson);
+
+SQLBindParameter(hstmt,
+ 5,
+ SQL_PARAM_INPUT,
+ SQL_C_CHAR,
+ SQL_CHAR,
+ STATUS_LEN,
+ 0,
+ szStatus,
+ 0,
+ &cbStatus);
+
+/*
+
+/* Specify first row of parameter data. */
+sOrderID = 1001;
+sCustID = 298;
+dsOpenDate.year = 1996;
+dsOpenDate.month = 3;
+dsOpenDate.day = 8;
+strcpy(szSalesPerson, "Johnson");
+strcpy(szStatus, "Closed");
+
+/* Execute statement with first row. */
+retcode = SQLExecute(hstmt);
+
+/* Specify second row of parameter data. */
+sOrderID = 1002;
+sCustID = 501;
+dsOpenDate.year = 1996;
+dsOpenDate.month = 3;
+dsOpenDate.day = 9;
+strcpy(szSalesPerson, "Bailey");
+strcpy(szStatus, "Open");
+
+/* Execute statement with second row. */
+retcode = SQLExecute(hstmt);
+
+*/
+
+}
+
+ return 0;
+
+ }
+
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLCancelTest.cpp b/ndb/test/odbc/client/SQLCancelTest.cpp
new file mode 100644
index 00000000000..904ffab6979
--- /dev/null
+++ b/ndb/test/odbc/client/SQLCancelTest.cpp
@@ -0,0 +1,254 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLCancelTest.cpp
+ */
+
+#include <common.hpp>
+#define Cancel_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC CC_hdbc;
+SQLHSTMT CC_hstmt;
+SQLHENV CC_henv;
+SQLHDESC CC_hdesc;
+
+void Cancel_DisplayError(SQLSMALLINT Cancel_HandleType,
+ SQLHSTMT Cancel_InputHandle);
+/**
+ * Test to terminate SQL statement precessing
+ *
+ * Tests:
+ * -# normal case test with correct hstmt handle
+ * -# SQL_STILL_EXECUTING case test with hstmt handle
+ * -# abnormal case test with incorrect hdbc, henv and hdesc handle
+ * @return Zero, if test succeeded
+ */
+
+int SQLCancelTest()
+{
+
+ SQLRETURN retcode;
+ SQLCHAR SQLStmt [120];
+
+ ndbout << endl << "Start SQLCancel Testing" << endl;
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &CC_henv);
+
+ if(retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(CC_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC, CC_henv, &CC_hdbc);
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(CC_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ ndbout << "Failure to Connect DB!" << endl;
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT, CC_hdbc, &CC_hstmt);
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ strcpy((char *) SQLStmt,
+ "select * FROM Customers");
+
+ //*************************
+ //** Prepare a statement **
+ //*************************
+
+ retcode = SQLPrepare(CC_hstmt,
+ SQLStmt,
+ SQL_NTS);
+
+ //***********************
+ //** Execute statement **
+ //***********************
+
+ retcode = SQLExecute(CC_hstmt);
+
+ //************************************************
+ //** Test 1 **
+ //** Input correct hstmt handle for normal test **
+ //************************************************
+
+ retcode = SQLCancel(CC_hstmt);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ {
+ ndbout << "Test 1" << endl;
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but SQL_INVALID_HANDLE"
+ << "still appeared. Please check program" << endl;
+ }
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Test 1" << endl;
+ Cancel_DisplayError(SQL_HANDLE_STMT, CC_hstmt);
+ }
+ //************************************************
+ //** Test 2 **
+ //** SQL_STILL_EXECUTING is not defined **
+ //************************************************
+
+ if (retcode == SQL_STILL_EXECUTING)
+ {
+ ndbout << "Test 2" << endl;
+ ndbout << "The function is still processing." << endl;
+ }
+
+ if (retcode == SQL_ERROR)
+ {
+ ndbout << "Test 2" << endl;
+ ndbout << "The Asynchronous processing was successfully canceled!"
+ << endl;
+ }
+ //*********************************
+ //** Test 3 **
+ //** Input incorrect henv handle **
+ //*********************************
+
+ retcode = SQLCancel(CC_henv);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ {
+ ndbout << "Test 3" << endl;
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but SQL_SUCCESS_WITH_INFO"
+ << " still appeared. Please check program" << endl;
+ Cancel_DisplayError(SQL_HANDLE_ENV, CC_henv);
+ }
+
+ //*********************************
+ //** Test 4 **
+ //** Input incorrect hdbc handle **
+ //*********************************
+
+ retcode = SQLCancel(CC_hdbc);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ {
+ ndbout << "Test 4" << endl;
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but SQL_SUCCESS_WITH_INFO"
+ << "still appeared. Please check programm" << endl;
+ Cancel_DisplayError(SQL_HANDLE_DBC, CC_hdbc);
+ }
+
+ //**********************************
+ //** Test 5 **
+ //** Input incorrect handle hdesc **
+ //**********************************
+
+ retcode = SQLCancel(CC_hdesc);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ {
+ ndbout << endl
+ << "Handle Type is SQL_HANDLE_DESC, but SQL_SUCCESS_WITH_INFO"
+ << "still appeared. Please check program" << endl;
+ ndbout << "Test 5" << endl;
+ Cancel_DisplayError(SQL_HANDLE_DESC, CC_hdesc);
+ }
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(CC_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, CC_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, CC_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, CC_henv);
+
+ return NDBT_OK;
+
+ }
+
+void Cancel_DisplayError(SQLSMALLINT Cancel_HandleType,
+ SQLHSTMT Cancel_InputHandle)
+{
+ SQLCHAR Sqlstate[5];
+ SQLRETURN SQLSTATEs;
+ SQLINTEGER NativeError;
+ SQLSMALLINT i, MsgLen;
+ SQLCHAR Msg[Cancel_MESSAGE_LENGTH];
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(Cancel_HandleType,
+ Cancel_InputHandle,
+ i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << Cancel_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)Cancel_InputHandle << endl;
+ ndbout << "the Msg is: " << (char *) Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLCloseCursorTest.cpp b/ndb/test/odbc/client/SQLCloseCursorTest.cpp
new file mode 100644
index 00000000000..35f125df59d
--- /dev/null
+++ b/ndb/test/odbc/client/SQLCloseCursorTest.cpp
@@ -0,0 +1,92 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void CloseCursor_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int SQLCloseCursorTest()
+{
+ /* "If there is no open cursor associated with S, then an exception is raised: invalid cursor state" How to test this case */
+
+ /* hstmt */
+ retcode = SQLCloseCursor(hstmt);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ CloseCursor_DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ /* henv */
+ retcode = SQLCloseCursor(henv);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_SUCCESS_WITH_INFO still appeared. Please check programm" << endl;
+ // CloseCursor_DisplayError(SQL_HANDLE_ENV, henv);
+
+ /* hdbc */
+ retcode = SQLCloseCursor(hdbc);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_SUCCESS_WITH_INFO still appeared. Please check programm" << endl;
+ // CloseCursor_DisplayError(SQL_HANDLE_DBC, hdbc);
+
+ /* hdesc */
+ retcode = SQLCloseCursor(hdesc);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ ndbout << "Handle Type is SQL_HANDLE_DESC, but string SQL_SUCCESS_WITH_INFO still appeared. Please check programm" << endl;
+ // CloseCursor_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+ return 0;
+
+ }
+
+
+void CloseCursor_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLColAttributeTest.cpp b/ndb/test/odbc/client/SQLColAttributeTest.cpp
new file mode 100644
index 00000000000..4c067c21d7d
--- /dev/null
+++ b/ndb/test/odbc/client/SQLColAttributeTest.cpp
@@ -0,0 +1,328 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * @file SQLColAttributeTest.cpp
+ */
+
+#include <common.hpp>
+using namespace std;
+
+#define MAXIMUM_MESSAGE_LENGTH_Test 200
+#define BufferLengthTest 156
+
+SQLHSTMT ColAtt_hstmt;
+SQLHSTMT ColAtt_hdbc;
+SQLHENV ColAtt_henv;
+SQLHDESC ColAtt_hdesc;
+
+SQLCHAR CharacterAttributePtr;
+SQLINTEGER NumericAttributePtr;
+SQLSMALLINT StringLengthPtr;
+
+SQLRETURN ColAtt_ret;
+
+void ColAtt_DisplayError(SQLSMALLINT ColAtt_HandleType,
+ SQLHSTMT ColAtt_InputHandle);
+
+/**
+ * Test returning descriptor information
+ *
+ * Tests:
+ * -# Call SQLColAttribute, without preceeding SQLPrepare
+ * -# ???
+ *
+ * @return Zero, if test succeeded
+ */
+int SQLColAttributeTest()
+{
+ ndbout << endl << "Start SQLColAttribute Testing" << endl;
+
+ SQLCHAR SQLStmt [120];
+
+ /********************************************************************
+ ** Test 1: **
+ ** **
+ ** Checks to execute SQLColAttribute, when there is no **
+ ** prepared or executed statement associated with StatementHandle **
+ ** **
+ ** Intended result: SQL_ERROR ??? **
+ ********************************************************************/
+ ColAtt_ret = SQLColAttribute(ColAtt_hstmt,
+ 1,
+ SQL_DESC_AUTO_UNIQUE_VALUE,
+ &CharacterAttributePtr,
+ BufferLengthTest,
+ &StringLengthPtr,
+ &NumericAttributePtr);
+
+ if (ColAtt_ret == SQL_ERROR)
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_ret << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayError(SQL_HANDLE_STMT, ColAtt_hstmt);
+ }
+ else if (ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_ret << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayError(SQL_HANDLE_STMT, ColAtt_hstmt);
+ }
+ else if (ColAtt_ret == SQL_SUCCESS)
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_ret << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayError(SQL_HANDLE_STMT, ColAtt_hstmt);
+ }
+ else if (ColAtt_ret == -2)
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_ret << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayError(SQL_HANDLE_STMT, ColAtt_hstmt);
+ }
+ else
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_ret << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayError(SQL_HANDLE_STMT, ColAtt_hstmt);
+ }
+
+ //*******************************************************************
+ //** Test 2: **
+ //** **
+ //** hstmt **
+ //** Execute a statement to retrieve rows from the Customers table **
+ //** We can create the table and insert rows into Mysql **
+ //** **
+ //** Intended result: ??? **
+ //*******************************************************************
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ ColAtt_ret = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &ColAtt_henv);
+
+ if (ColAtt_ret == SQL_SUCCESS || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 2.x **
+ //*********************************************
+ ColAtt_ret = SQLSetEnvAttr(ColAtt_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC2,
+ SQL_IS_UINTEGER);
+
+ if (ColAtt_ret == SQL_SUCCESS || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 2.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ ColAtt_ret = SQLAllocHandle(SQL_HANDLE_DBC,
+ ColAtt_henv,
+ &ColAtt_hdbc);
+
+ if (ColAtt_ret == SQL_SUCCESS || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ ColAtt_ret = SQLConnect(ColAtt_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (ColAtt_ret == SQL_SUCCESS || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ ColAtt_ret = SQLAllocHandle(SQL_HANDLE_STMT,
+ ColAtt_hdbc,
+ &ColAtt_hstmt);
+ if(ColAtt_ret == SQL_SUCCESS || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ /*
+ strcpy((char *) SQLStmt,
+ "DELETE FROM Customers WHERE CustID = 6");
+ */
+
+ strcpy((char *) SQLStmt,
+ "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES (6, 'Jan', 'LM vag 8', '969696')");
+
+ /*
+ strcpy((char *) SQLStmt,
+ "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES (?, ?, ?, ?)");
+ */
+
+ //********************************
+ //** Prepare SQL statement **
+ //********************************
+ ColAtt_ret = SQLPrepare(ColAtt_hstmt,
+ SQLStmt,
+ SQL_NTS);
+
+ if (ColAtt_ret == SQL_SUCCESS || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ {
+ //**************************************************************
+ //** FieldIdentifer is not one of the code valuess in Table 20,
+ //** "Codes used for descriptor fields"
+ //**************************************************************
+ ColAtt_ret = SQLColAttribute(ColAtt_hstmt,
+ 2,
+ 9999,
+ &CharacterAttributePtr,
+ BufferLengthTest,
+ &StringLengthPtr,
+ &NumericAttributePtr);
+
+ if (ColAtt_ret == SQL_ERROR || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << endl << "FieldIdentifer is not one of the" << endl;
+ ndbout << "code valuess in Table 20, Codes used for" << endl;
+ ndbout << "descriptor fields" <<endl;
+ ColAtt_DisplayError(SQL_HANDLE_STMT, ColAtt_hstmt);
+ }
+
+ //****************************************************************
+ //** Let TYPE is 'ITEM' in Table 20, ColumnNumber is less than one
+ //****************************************************************
+ ColAtt_ret = SQLColAttribute(ColAtt_hstmt,
+ -1,
+ SQL_DESC_BASE_COLUMN_NAME,
+ &CharacterAttributePtr,
+ BufferLengthTest,
+ &StringLengthPtr,
+ &NumericAttributePtr);
+
+ if (ColAtt_ret == SQL_ERROR || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Let TYPE is 'ITEM' in Table 20,ColumnNumber"
+ << "is less than one" << endl;
+ ColAtt_DisplayError(SQL_HANDLE_STMT, ColAtt_hstmt);
+ }
+
+ //*********************************************************
+ //** Let TYPE is 'ITEM' in Table 20, FieldIdentifer is zero
+ //*********************************************************
+ ColAtt_ret = SQLColAttribute(ColAtt_hstmt,
+ 1018,
+ 0,
+ &CharacterAttributePtr,
+ BufferLengthTest,
+ &StringLengthPtr,
+ &NumericAttributePtr);
+
+ if (ColAtt_ret == SQL_ERROR || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Let TYPE is 'ITEM' in Table 20, FieldIdentifer"
+ << " is zero" <<endl;
+ ColAtt_DisplayError(SQL_HANDLE_STMT, ColAtt_hstmt);
+ }
+
+ //**********************************************************
+ //** Let TYPE is 'ITEM' in Table 20, ColumnNumber is greater
+ //** than TOP_LEVEL_COUNT(1044)
+ //*********************************************************
+ ColAtt_ret = SQLColAttribute(ColAtt_hstmt,
+ 1045,
+ SQL_DESC_BASE_COLUMN_NAME,
+ &CharacterAttributePtr,
+ BufferLengthTest,
+ &StringLengthPtr,
+ &NumericAttributePtr);
+
+ if (ColAtt_ret == SQL_ERROR || ColAtt_ret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Let TYPE is 'ITEM' in Table 20, ColumnNumber" << endl
+ << "is greater than TOP_LEVEL_COUNT(1044)" << endl;
+ ColAtt_DisplayError(SQL_HANDLE_STMT, ColAtt_hstmt);
+ }
+
+ }
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(ColAtt_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, ColAtt_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, ColAtt_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, ColAtt_henv);
+
+ return NDBT_OK;
+}
+
+void ColAtt_DisplayError(SQLSMALLINT ColAtt_HandleType,
+ SQLHSTMT ColAtt_InputHandle)
+{
+ SQLSMALLINT ColAtt_i = 1;
+ SQLRETURN ColAtt_SQLSTATEs;
+ SQLCHAR ColAtt_Sqlstate[5];
+ SQLCHAR ColAtt_Msg[MAXIMUM_MESSAGE_LENGTH_Test];
+ SQLSMALLINT ColAtt_MsgLen;
+ SQLINTEGER ColAtt_NativeError;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((ColAtt_SQLSTATEs = SQLGetDiagRec(ColAtt_HandleType,
+ ColAtt_InputHandle,
+ ColAtt_i,
+ ColAtt_Sqlstate,
+ &ColAtt_NativeError,
+ ColAtt_Msg,
+ sizeof(ColAtt_Msg),
+ &ColAtt_MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << ColAtt_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)ColAtt_InputHandle << endl;
+ ndbout << "the ColAtt_Msg is: " << (char *) ColAtt_Msg << endl;
+ ndbout << "the output state is:" << (char *)ColAtt_Sqlstate << endl;
+
+ ColAtt_i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLColAttributeTest1.cpp b/ndb/test/odbc/client/SQLColAttributeTest1.cpp
new file mode 100644
index 00000000000..322a21eefc1
--- /dev/null
+++ b/ndb/test/odbc/client/SQLColAttributeTest1.cpp
@@ -0,0 +1,143 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**
+ * @file SQLColAttributeTest1.cpp
+ */
+
+#include <common.hpp>
+using namespace std;
+
+#define MAXIMUM_MESSAGE_LENGTH_Test1 200
+#define BufferLenghTest1 156
+
+SQLHSTMT ColAtt_hstmtTest1;
+SQLHSTMT ColAtt_hdbcTest1;
+SQLHENV ColAtt_henvTest1;
+SQLHDESC ColAtt_hdescTest1;
+
+SQLCHAR CharacterAttributePtrTest1;
+SQLINTEGER NumericAttributePtrTest1;
+SQLSMALLINT StringLengthPtrTest1;
+
+SQLRETURN ColAtt_retTest1;
+
+void ColAtt_DisplayErrorTest1(SQLSMALLINT ColAtt_HandleType,
+ SQLHSTMT ColAtt_InputHandle);
+
+/**
+ * Test returning descriptor information
+ *
+ * Tests:
+ * -# Execute SQLColAttribute without prepared or executed statement
+ *
+ * @return Zero, if test succeeded
+ */
+int SQLColAttributeTest1()
+{
+ ndbout << endl << "Start SQLColAttribute Testing1" << endl;
+ /********************************************************************
+ ** Test : **
+ ** **
+ ** Checks to execute SQLColAttribute, when there is no **
+ ** prepared or executed statement associated with StatementHandle **
+ ** **
+ ** Intended result:CLI-specific condition-function sequence error **
+ ********************************************************************/
+ ColAtt_retTest1 = SQLColAttribute(ColAtt_hstmtTest1,
+ 1,
+ SQL_DESC_AUTO_UNIQUE_VALUE,
+ &CharacterAttributePtrTest1,
+ BufferLenghTest1,
+ &StringLengthPtrTest1,
+ &NumericAttributePtrTest1);
+
+ if (ColAtt_retTest1 == SQL_ERROR)
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_retTest1 << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayErrorTest1(SQL_HANDLE_STMT, ColAtt_hstmtTest1);
+ }
+ else if (ColAtt_retTest1 == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_retTest1 << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayErrorTest1(SQL_HANDLE_STMT, ColAtt_hstmtTest1);
+ }
+ else if (ColAtt_retTest1 == SQL_SUCCESS)
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_retTest1 << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayErrorTest1(SQL_HANDLE_STMT, ColAtt_hstmtTest1);
+ }
+ else if (ColAtt_retTest1 == -2)
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_retTest1 << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayErrorTest1(SQL_HANDLE_STMT, ColAtt_hstmtTest1);
+ }
+ else
+ {
+ ndbout << "ColAtt_ret = " << ColAtt_retTest1 << endl;
+ ndbout << endl << "There is no prepared or executed" << endl
+ << " statement associated with StatementHandle" << endl;
+ ColAtt_DisplayErrorTest1(SQL_HANDLE_STMT, ColAtt_hstmtTest1);
+ }
+
+ return NDBT_OK;
+}
+
+void ColAtt_DisplayErrorTest1(SQLSMALLINT ColAtt_HandleType,
+ SQLHSTMT ColAtt_InputHandle)
+{
+ SQLSMALLINT ColAtt_i = 1;
+ SQLRETURN ColAtt_SQLSTATEs;
+ SQLCHAR ColAtt_Sqlstate[5];
+ SQLCHAR ColAtt_Msg[MAXIMUM_MESSAGE_LENGTH_Test1];
+ SQLSMALLINT ColAtt_MsgLen;
+ SQLINTEGER ColAtt_NativeError;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((ColAtt_SQLSTATEs = SQLGetDiagRec(ColAtt_HandleType,
+ ColAtt_InputHandle,
+ ColAtt_i,
+ ColAtt_Sqlstate,
+ &ColAtt_NativeError,
+ ColAtt_Msg,
+ sizeof(ColAtt_Msg),
+ &ColAtt_MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << ColAtt_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)ColAtt_InputHandle << endl;
+ ndbout << "the ColAtt_Msg is: " << (char *) ColAtt_Msg << endl;
+ ndbout << "the output state is:" << (char *)ColAtt_Sqlstate << endl;
+
+ ColAtt_i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLColAttributeTest2.cpp b/ndb/test/odbc/client/SQLColAttributeTest2.cpp
new file mode 100644
index 00000000000..18cffae76c1
--- /dev/null
+++ b/ndb/test/odbc/client/SQLColAttributeTest2.cpp
@@ -0,0 +1,277 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLColAttributeTest2.cpp
+ */
+
+#include <common.hpp>
+using namespace std;
+
+#define MAXIMUM_MESSAGE_LENGTH_Test2 200
+#define BufferLengthTest2 156
+
+SQLHSTMT ColAtt_hstmtTest2;
+SQLHSTMT ColAtt_hdbcTest2;
+SQLHENV ColAtt_henvTest2;
+SQLHDESC ColAtt_hdescTest2;
+
+SQLCHAR CharacterAttributePtrTest2;
+SQLINTEGER NumericAttributePtrTest2;
+SQLSMALLINT StringLengthPtrTest2;
+
+SQLRETURN ColAtt_retTest2;
+
+void ColAtt_DisplayErrorTest2(SQLSMALLINT ColAttTest2_HandleType,
+ SQLHSTMT ColAttTest2_InputHandle);
+
+/**
+ * Test returning descriptor information
+ *
+ * Test:
+ * -# Call SQLColAttribute without preceeding SQLExecute
+ * -# Let TYPE is 'ITEM' in Table 20, FieldIdentifer is zero
+ * -# Let TYPE is 'ITEM' in Table 20, ColumnNumber is less than one
+ * -# FieldIdentifer is not one of the code valuess in Table 20
+ * -# Let TYPE is 'ITEM' in Table 20, ColumnNumber is greater than 1044
+ *
+ * @return Zero, if test succeeded
+ */
+int SQLColAttributeTest2()
+{
+ ndbout << endl << "Start SQLColAttribute Testing2" << endl;
+
+ SQLCHAR SQLStmt [120];
+
+ //*******************************************************************
+ //** Test **
+ //** **
+ //** hstmt **
+ //** Prepare a statement without executing the statement **
+ //** **
+ //** Intended result: table Customer should not have new row **
+ //*******************************************************************
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ ColAtt_retTest2 = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &ColAtt_henvTest2);
+
+ if (ColAtt_retTest2 == SQL_SUCCESS || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ ColAtt_retTest2 = SQLSetEnvAttr(ColAtt_henvTest2,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (ColAtt_retTest2 == SQL_SUCCESS || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 2.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ ColAtt_retTest2 = SQLAllocHandle(SQL_HANDLE_DBC,
+ ColAtt_henvTest2,
+ &ColAtt_hdbcTest2);
+
+ if (ColAtt_retTest2 == SQL_SUCCESS || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ ColAtt_retTest2 = SQLConnect(ColAtt_hdbcTest2,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (ColAtt_retTest2 == SQL_SUCCESS || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ ColAtt_retTest2 = SQLAllocHandle(SQL_HANDLE_STMT,
+ ColAtt_hdbcTest2,
+ &ColAtt_hstmtTest2);
+ if(ColAtt_retTest2 == SQL_SUCCESS || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ /*
+ strcpy((char *) SQLStmt,
+ "DELETE FROM Customers WHERE CustID = 6");
+ */
+
+ strcpy((char *) SQLStmt,
+ "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES (6, 'Jan', 'LM vag 8', '969696')");
+
+ /*
+ strcpy((char *) SQLStmt,
+ "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES (?, ?, ?, ?)");
+ */
+
+ //********************************
+ //** Prepare SQL statement **
+ //********************************
+ ColAtt_retTest2 = SQLPrepare(ColAtt_hstmtTest2,
+ SQLStmt,
+ SQL_NTS);
+
+ if (ColAtt_retTest2 == SQL_SUCCESS || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ {
+ //**************************************************************
+ //** FieldIdentifer is not one of the code valuess in Table 20,
+ //** "Codes used for descriptor fields"
+ //**************************************************************
+ ColAtt_retTest2 = SQLColAttribute(ColAtt_hstmtTest2,
+ 2,
+ 9999,
+ &CharacterAttributePtrTest2,
+ BufferLengthTest2,
+ &StringLengthPtrTest2,
+ &NumericAttributePtrTest2);
+
+ if (ColAtt_retTest2 == SQL_ERROR || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << endl << "FieldIdentifer is not one of the" << endl;
+ ndbout << "code valuess in Table 20, Codes used for" << endl;
+ ndbout << "descriptor fields" <<endl;
+ ColAtt_DisplayErrorTest2(SQL_HANDLE_STMT, ColAtt_hstmtTest2);
+ }
+
+ //****************************************************************
+ //** Let TYPE is 'ITEM' in Table 20, ColumnNumber is less than one
+ //****************************************************************
+ ColAtt_retTest2 = SQLColAttribute(ColAtt_hstmtTest2,
+ -1,
+ SQL_DESC_BASE_COLUMN_NAME,
+ &CharacterAttributePtrTest2,
+ BufferLengthTest2,
+ &StringLengthPtrTest2,
+ &NumericAttributePtrTest2);
+
+ if (ColAtt_retTest2 == SQL_ERROR || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Let TYPE is 'ITEM' in Table 20,ColumnNumber"
+ << "is less than one" << endl;
+ ColAtt_DisplayErrorTest2(SQL_HANDLE_STMT, ColAtt_hstmtTest2);
+ }
+
+ //*********************************************************
+ //** Let TYPE is 'ITEM' in Table 20, FieldIdentifer is zero
+ //*********************************************************
+ ColAtt_retTest2 = SQLColAttribute(ColAtt_hstmtTest2,
+ 1018,
+ 0,
+ &CharacterAttributePtrTest2,
+ BufferLengthTest2,
+ &StringLengthPtrTest2,
+ &NumericAttributePtrTest2);
+
+ if (ColAtt_retTest2 == SQL_ERROR || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Let TYPE is 'ITEM' in Table 20, FieldIdentifer"
+ << " is zero" <<endl;
+ ColAtt_DisplayErrorTest2(SQL_HANDLE_STMT, ColAtt_hstmtTest2);
+ }
+
+ //**********************************************************
+ //** Let TYPE is 'ITEM' in Table 20, ColumnNumber is greater
+ //** than TOP_LEVEL_COUNT(1044)
+ //*********************************************************
+ ColAtt_retTest2 = SQLColAttribute(ColAtt_hstmtTest2,
+ 1045,
+ SQL_DESC_BASE_COLUMN_NAME,
+ &CharacterAttributePtrTest2,
+ BufferLengthTest2,
+ &StringLengthPtrTest2,
+ &NumericAttributePtrTest2);
+
+ if (ColAtt_retTest2 == SQL_ERROR || ColAtt_retTest2 == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Let TYPE is 'ITEM' in Table 20, ColumnNumber" << endl
+ << "is greater than TOP_LEVEL_COUNT(1044)" << endl;
+ ColAtt_DisplayErrorTest2(SQL_HANDLE_STMT, ColAtt_hstmtTest2);
+ }
+
+ }
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(ColAtt_hdbcTest2);
+ SQLFreeHandle(SQL_HANDLE_STMT, ColAtt_hstmtTest2);
+ SQLFreeHandle(SQL_HANDLE_DBC, ColAtt_hdbcTest2);
+ SQLFreeHandle(SQL_HANDLE_ENV, ColAtt_henvTest2);
+
+ return NDBT_OK;
+}
+
+
+void ColAtt_DisplayErrorTest2(SQLSMALLINT ColAttTest2_HandleType,
+ SQLHSTMT ColAttTest2_InputHandle)
+{
+ SQLSMALLINT ColAtt_i = 1;
+ SQLRETURN ColAtt_SQLSTATEs;
+ SQLCHAR ColAtt_Sqlstate[5];
+ SQLCHAR ColAtt_Msg[MAXIMUM_MESSAGE_LENGTH_Test2];
+ SQLSMALLINT ColAtt_MsgLen;
+ SQLINTEGER ColAtt_NativeError;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((ColAtt_SQLSTATEs = SQLGetDiagRec(ColAttTest2_HandleType,
+ ColAttTest2_InputHandle,
+ ColAtt_i,
+ ColAtt_Sqlstate,
+ &ColAtt_NativeError,
+ ColAtt_Msg,
+ sizeof(ColAtt_Msg),
+ &ColAtt_MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << ColAttTest2_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)ColAttTest2_InputHandle << endl;
+ ndbout << "the ColAtt_Msg is: " << (char *) ColAtt_Msg << endl;
+ ndbout << "the output state is:" << (char *)ColAtt_Sqlstate << endl;
+
+ ColAtt_i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
diff --git a/ndb/test/odbc/client/SQLColAttributeTest3.cpp b/ndb/test/odbc/client/SQLColAttributeTest3.cpp
new file mode 100644
index 00000000000..f8817565711
--- /dev/null
+++ b/ndb/test/odbc/client/SQLColAttributeTest3.cpp
@@ -0,0 +1,275 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLColAttributeTest3.cpp
+ */
+
+#include <common.hpp>
+using namespace std;
+
+#define MAXIMUM_MESSAGE_LENGTH_Test3 200
+#define BufferLengthTest3 156
+
+SQLHSTMT ColAtt_hstmtTest3;
+SQLHSTMT ColAtt_hdbcTest3;
+SQLHENV ColAtt_henvTest3;
+SQLHDESC ColAtt_hdescTest3;
+
+SQLCHAR TypeName[18];
+SQLSMALLINT TypeNameLen;
+
+SQLRETURN ColAtt_retTest3;
+
+void ColAtt_DisplayErrorTest3(SQLSMALLINT ColAttTest3_HandleType,
+ SQLHSTMT ColAttTest3_InputHandle);
+
+/**
+ * Test returning descriptor information
+ *
+ * Test:
+ * -# Print out column name without executing statement
+ *
+ * @return Zero, if test succeeded
+ */
+
+int SQLColAttributeTest3()
+{
+ ndbout << endl << "Start SQLColAttribute Testing3" << endl;
+
+ SQLCHAR SQLStmt [120];
+
+ //********************************************************************
+ //** Test 3: **
+ //** **
+ //** Prepare a statement without executing the statement **
+ //** We want to print out the Type Name of each column in the table **
+ //** Customers **
+ //** **
+ //** Intended result: Only display column name, but there is no new **
+ //** row in table Customers **
+ //********************************************************************
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ ColAtt_retTest3 = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &ColAtt_henvTest3);
+
+ if (ColAtt_retTest3 == SQL_SUCCESS || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ ColAtt_retTest3 = SQLSetEnvAttr(ColAtt_henvTest3,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (ColAtt_retTest3 == SQL_SUCCESS || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ ColAtt_retTest3 = SQLAllocHandle(SQL_HANDLE_DBC,
+ ColAtt_henvTest3,
+ &ColAtt_hdbcTest3);
+
+ if (ColAtt_retTest3 == SQL_SUCCESS || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ ColAtt_retTest3 = SQLConnect(ColAtt_hdbcTest3,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (ColAtt_retTest3 == SQL_SUCCESS || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ ColAtt_retTest3 = SQLAllocHandle(SQL_HANDLE_STMT,
+ ColAtt_hdbcTest3,
+ &ColAtt_hstmtTest3);
+ if(ColAtt_retTest3 == SQL_SUCCESS || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ /*
+ strcpy((char *) SQLStmt,
+ "DELETE FROM Customers WHERE CustID = 6");
+ */
+
+ strcpy((char *) SQLStmt,
+ "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES (6, 'Jan', 'LM vag 8', '969696')");
+
+ /*
+ strcpy((char *) SQLStmt,
+ "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES (?, ?, ?, ?)");
+ */
+
+ //*****************************
+ //** Prepare SQL statement **
+ //*****************************
+ ColAtt_retTest3 = SQLPrepare(ColAtt_hstmtTest3,
+ SQLStmt,
+ SQL_NTS);
+
+ if (ColAtt_retTest3 == SQL_SUCCESS || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ {
+ //************************************
+ //** Display the name of column one **
+ //************************************
+ ColAtt_retTest3 = SQLColAttribute(ColAtt_hstmtTest3,
+ 1,
+ SQL_COLUMN_TYPE_NAME,
+ TypeName,
+ sizeof(TypeName),
+ &TypeNameLen,
+ NULL);
+
+ if (ColAtt_retTest3 == SQL_ERROR || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << endl << "ColAtt_retTest3 = " << ColAtt_retTest3 << endl;
+ ndbout << endl << "Name of column 1 is:"
+ << (char *)TypeName <<endl;
+ ColAtt_DisplayErrorTest3(SQL_HANDLE_STMT, ColAtt_hstmtTest3);
+ }
+
+ //************************************
+ //** Display the name of column two **
+ //************************************
+ ColAtt_retTest3 = SQLColAttribute(ColAtt_hstmtTest3,
+ 2,
+ SQL_DESC_BASE_COLUMN_NAME,
+ TypeName,
+ sizeof(TypeName),
+ &TypeNameLen,
+ NULL);
+
+ if (ColAtt_retTest3 == SQL_ERROR || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << endl << "ColAtt_retTest3 = " << ColAtt_retTest3 << endl;
+ ndbout << endl << "Name of column 2 is:"
+ << (char *)TypeName <<endl;
+ ColAtt_DisplayErrorTest3(SQL_HANDLE_STMT, ColAtt_hstmtTest3);
+ }
+
+ //***************************************
+ //** Display the name of column three **
+ //***************************************
+ ColAtt_retTest3 = SQLColAttribute(ColAtt_hstmtTest3,
+ 3,
+ SQL_DESC_BASE_COLUMN_NAME,
+ TypeName,
+ sizeof(TypeName),
+ &TypeNameLen,
+ NULL);
+
+ if (ColAtt_retTest3 == SQL_ERROR || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "ColAtt_retTest3 = " << ColAtt_retTest3 << endl;
+ ndbout << endl << "Name of column 3 is:"
+ << (char *)TypeName <<endl;
+ ColAtt_DisplayErrorTest3(SQL_HANDLE_STMT, ColAtt_hstmtTest3);
+ }
+
+ //**************************************
+ //** Display the name of column four **
+ //**************************************
+ ColAtt_retTest3 = SQLColAttribute(ColAtt_hstmtTest3,
+ 4,
+ SQL_DESC_BASE_COLUMN_NAME,
+ TypeName,
+ sizeof(TypeName),
+ &TypeNameLen,
+ NULL);
+
+ if (ColAtt_retTest3 == SQL_ERROR || ColAtt_retTest3 == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "ColAtt_retTest3 = " << ColAtt_retTest3 << endl;
+ ndbout << endl << "Name of column 4 is:"
+ << (char *)TypeName <<endl;
+ ColAtt_DisplayErrorTest3(SQL_HANDLE_STMT, ColAtt_hstmtTest3);
+ }
+
+ }
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(ColAtt_hdbcTest3);
+ SQLFreeHandle(SQL_HANDLE_STMT, ColAtt_hstmtTest3);
+ SQLFreeHandle(SQL_HANDLE_DBC, ColAtt_hdbcTest3);
+ SQLFreeHandle(SQL_HANDLE_ENV, ColAtt_henvTest3);
+
+ return NDBT_OK;
+}
+
+void ColAtt_DisplayErrorTest3(SQLSMALLINT ColAttTest3_HandleType,
+ SQLHSTMT ColAttTest3_InputHandle)
+{
+ SQLSMALLINT ColAtt_i = 1;
+ SQLRETURN ColAtt_SQLSTATEs;
+ SQLCHAR ColAtt_Sqlstate[5];
+ SQLCHAR ColAtt_Msg[MAXIMUM_MESSAGE_LENGTH_Test3];
+ SQLSMALLINT ColAtt_MsgLen;
+ SQLINTEGER ColAtt_NativeError;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((ColAtt_SQLSTATEs = SQLGetDiagRec(ColAttTest3_HandleType,
+ ColAttTest3_InputHandle,
+ ColAtt_i,
+ ColAtt_Sqlstate,
+ &ColAtt_NativeError,
+ ColAtt_Msg,
+ sizeof(ColAtt_Msg),
+ &ColAtt_MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << ColAttTest3_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)ColAttTest3_InputHandle << endl;
+ ndbout << "the ColAtt_Msg is: " << (char *) ColAtt_Msg << endl;
+ ndbout << "the output state is:" << (char *)ColAtt_Sqlstate << endl;
+
+ ColAtt_i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
diff --git a/ndb/test/odbc/client/SQLConnectTest.cpp b/ndb/test/odbc/client/SQLConnectTest.cpp
new file mode 100644
index 00000000000..552fc8640fe
--- /dev/null
+++ b/ndb/test/odbc/client/SQLConnectTest.cpp
@@ -0,0 +1,165 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLConnectTest.cpp
+ */
+#include <common.hpp>
+using namespace std;
+
+SQLHDBC conn_hdbc;
+SQLHSTMT conn_hstmt;
+SQLHENV conn_henv;
+SQLHDESC conn_hdesc;
+SQLRETURN conn_retcode;
+
+#define conn_SQL_MAXIMUM_MESSAGE_LENGTH 200
+SQLCHAR conn_Sqlstate[5];
+
+SQLINTEGER conn_NativeError;
+SQLSMALLINT conn_MsgLen;
+SQLCHAR conn_Msg[conn_SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void SQLConnectTest_DisplayError_HDBC(SQLSMALLINT conn_HandleType,
+ SQLHDBC conn_InputHandle);
+
+/**
+ * -# Test to make a connection to an ODBC data source
+ *
+ * @return Zero, if test succeeded
+ */
+int SQLConnectTest()
+{
+ ndbout << endl << "Start SQLConnect Testing" << endl;
+
+ // ************************************
+ // ** Allocate an environment handle **
+ // ************************************
+ conn_retcode = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &conn_henv);
+ //conn_retcode = SQLAllocEnv(&conn_henv);
+ if(conn_retcode == SQL_SUCCESS || conn_retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Allocated an environment handle!" << endl;
+ }
+ else
+ {
+ ndbout << "Failed to allocate environment handle!" << endl;
+ return NDBT_FAILED;
+ }
+
+ // *********************************************
+ // ** Set the ODBC application Version to 3.x **
+ // *********************************************
+ conn_retcode = SQLSetEnvAttr(conn_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+ if (conn_retcode == SQL_SUCCESS || conn_retcode == SQL_SUCCESS_WITH_INFO) {
+ ndbout << "Set ODBC application version to 3.x" << endl;
+ } else {
+ ndbout << "Failed to set application version!" << endl;
+ return NDBT_FAILED;
+ }
+
+ // **********************************
+ // ** Allocate a connection handle **
+ // **********************************
+ conn_retcode = SQLAllocHandle(SQL_HANDLE_DBC, conn_henv, &conn_hdbc);
+ // retcode = SQLAllocConnect(conn_henv, &conn_hdbc);
+ if (conn_retcode == SQL_SUCCESS || conn_retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Allocated a connection handle!" << endl;
+ }
+ else
+ {
+ ndbout << "Failed to allocate connection handle!" << endl;
+ return NDBT_FAILED;
+ }
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ conn_retcode = SQLConnect(conn_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+ ndbout << "conn_retcode = " << conn_retcode << endl;
+ if (conn_retcode == SQL_SUCCESS)
+ {
+ ndbout << "Connected to DB!" << endl;
+ }
+ else if (conn_retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Connected to DB, but SQL_SUCCESS_WITH_INFO!" << endl;
+ SQLConnectTest_DisplayError_HDBC(SQL_HANDLE_DBC, conn_hdbc);
+ }
+ else if (conn_retcode == SQL_INVALID_HANDLE)
+ {
+ ndbout << "SQL_INVALID_HANDLE appeared. Please check program." << endl;
+ return NDBT_FAILED;
+ }
+ else if (conn_retcode == SQL_ERROR)
+ {
+ ndbout << "Failed to connect!" << endl;
+ SQLConnectTest_DisplayError_HDBC(SQL_HANDLE_DBC, conn_hdbc);
+ return NDBT_FAILED;
+ }
+ else
+ ;
+
+ // ******************
+ // ** Free Handles **
+ // ******************
+ SQLDisconnect(conn_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, conn_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, conn_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, conn_henv);
+ return NDBT_OK;
+}
+
+
+void SQLConnectTest_DisplayError_HDBC(SQLSMALLINT conn_HandleType,
+ SQLHDBC conn_InputHandle) {
+ SQLSMALLINT conn_i = 1;
+ SQLRETURN conn_SQLSTATE;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((conn_SQLSTATE = SQLGetDiagRec(conn_HandleType,
+ conn_InputHandle,
+ conn_i,
+ conn_Sqlstate,
+ &conn_NativeError,
+ conn_Msg,
+ sizeof(conn_Msg),
+ &conn_MsgLen)
+ ) != SQL_NO_DATA)
+ {
+ ndbout << "SQLSTATE = " << conn_SQLSTATE << endl;
+ ndbout << "the HandleType is: " << conn_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)conn_InputHandle << endl;
+ ndbout << "the conn_Msg is: " << (char *) conn_Msg << endl;
+ ndbout << "the output state is:" << (char *)conn_Sqlstate << endl;
+
+ conn_i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
diff --git a/ndb/test/odbc/client/SQLCopyDescTest.cpp b/ndb/test/odbc/client/SQLCopyDescTest.cpp
new file mode 100644
index 00000000000..4a3742f97ae
--- /dev/null
+++ b/ndb/test/odbc/client/SQLCopyDescTest.cpp
@@ -0,0 +1,140 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <sqlcli.h>
+#include <stdio.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+
+#define ROWS 100
+#define DESC_LEN 50
+
+
+// Template for a row
+typedef struct {
+ SQLINTEGER sPartID;
+ SQLINTEGER cbPartID;
+ SQLUCHAR szDescription[DESC_LENGTH];
+ SQLINTEGER cbDescription;
+ REAL sPrice;
+ SQLINTEGER cbPrice;
+} PartsSource;
+
+PartsSource rget[ROWS]; // rowset buffer
+SQLUSMALLINT sts_ptr[ROWS]; // status pointer
+SQLHSTMT hstmt0, hstmt1;
+SQLHDESC hArd0, hIrd0, hApd1, hIpd1;
+
+
+SQLHSTMT hstmt;
+SQLHDESC hdesc;
+
+SQLSMALLINT RecNumber;
+SQLCHAR szSalesPerson[SALES_PERSON_LEN];
+
+SQLCHAR Sqlstate[5], Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLINTEGER NativeError;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLINTEGER ValuePtr1;
+SQLCHAR ValuePtr2;
+SQLSMALLINT ValuePtr3;
+
+
+SQLINTEGER StringLengthPtr;
+
+SQLSMALLINT i, MsgLen;
+
+void DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle);
+
+int SQLCopyDescTest ()
+{
+
+
+ // We can create the table and insert rows in NDB by program TestDirectSQL.
+ // In this test program(SQLGetCopyRecTest),we only have three rows in table ORDERS
+
+
+// ARD and IRD of hstmt0
+SQLGetStmtAttr(hstmt0, SQL_ATTR_APP_ROW_DESC, &hArd0, 0, NULL);
+SQLGetStmtAttr(hstmt0, SQL_ATTR_IMP_ROW_DESC, &hIrd0, 0, NULL);
+
+// APD and IPD of hstmt1
+SQLGetStmtAttr(hstmt1, SQL_ATTR_APP_PARAM_DESC, &hApd1, 0, NULL);
+SQLGetStmtAttr(hstmt1, SQL_ATTR_IMP_PARAM_DESC, &hIpd1, 0, NULL);
+
+// Use row-wise binding on hstmt0 to fetch rows
+SQLSetStmtAttr(hstmt0, SQL_ATTR_ROW_BIND_TYPE, (SQLPOINTER) sizeof(PartsSource), 0);
+
+// Set rowset size for hstmt0
+SQLSetStmtAttr(hstmt0, SQL_ATTR_ROW_ARRAY_SIZE, (SQLPOINTER) ROWS, 0);
+
+// Execute a select statement
+SQLExecDirect(hstmt0, "SELECT PARTID, DESCRIPTION, PRICE FROM PARTS ORDER BY 3, 1, 2"",
+ SQL_NTS);
+
+// Bind
+SQLBindCol(hstmt0, 1, SQL_C_SLONG, rget[0].sPartID, 0,
+ &rget[0].cbPartID);
+SQLBindCol(hstmt0, 2, SQL_C_CHAR, &rget[0].szDescription, DESC_LEN,
+ &rget[0].cbDescription);
+SQLBindCol(hstmt0, 3, SQL_C_FLOAT, rget[0].sPrice,
+ 0, &rget[0].cbPrice);
+
+ // Perform parameter bindings on hstmt1.
+ /* If SourceDeschandle does not identify an allocated CLI descriptor area */
+ retcode1 = SQLCopyDesc(hArd0, hApd1);
+ retcode2 = SQLCopyDesc(hIrd0, hIpd1);
+
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DisplayError(SQL_HANDLE_DESC, hdesc);
+
+
+ /* If TargetDeschandle does not identify an allocated CLI descriptor area */
+ retcode = SQLCopyDesc(hdesc, );
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DisplayError(SQL_HANDLE_DESC, hdesc);
+
+
+ return 0;
+
+ }
+
+
+void DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLDescribeColTest.cpp b/ndb/test/odbc/client/SQLDescribeColTest.cpp
new file mode 100644
index 00000000000..9f55c6a1cfe
--- /dev/null
+++ b/ndb/test/odbc/client/SQLDescribeColTest.cpp
@@ -0,0 +1,260 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLDescribeColTest.cpp
+ */
+#include <common.hpp>
+
+using namespace std;
+
+#define DC_Column_NAME_LEN 50
+#define DC_MESSAGE_LENGTH 200
+
+SQLHSTMT DC_hstmt;
+SQLHDBC DC_hdbc;
+SQLHENV DC_henv;
+SQLHDESC DC_hdesc;
+
+void DescribeCol_DisplayError(SQLSMALLINT DC_HandleType,
+ SQLHSTMT DC_InputHandle);
+
+/**
+ * Test to retrieve basic result data set metadata information
+ * (specifically, column name, SQL data type, column size, decimal
+ * precision, and nullability) for a specified column in a result
+ * data set
+ * -# No prepared or executed statement when executing
+ * -# ColumnNumber is less than 1
+ * -# ColumnNumber is greater than the value of the TOP_LEVEL_COUNT field of IRD
+ * @return Zero, if test succeeded
+ */
+
+int SQLDescribeColTest()
+{
+ SQLCHAR SQLStmt [120];
+ SQLRETURN retcode;
+ SQLCHAR ColumnName[DC_Column_NAME_LEN];
+ SQLSMALLINT NameLength, DataTypePtr, DecimalDigitsPtr, NullablePtr;
+ SQLUINTEGER ColumnSizePtr;
+
+ ndbout << "Start SQLDescribeCol Test " << endl;
+ //******************************************************************
+ //** Test1 **
+ //** There is no prepared or executed statement associated with **
+ //** StatementHandle **
+ //******************************************************************
+
+ retcode = SQLDescribeCol(DC_hstmt,
+ (SQLUSMALLINT)1,
+ ColumnName,
+ sizeof(ColumnName),
+ &NameLength,
+ &DataTypePtr,
+ &ColumnSizePtr,
+ &DecimalDigitsPtr,
+ &NullablePtr);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DescribeCol_DisplayError(SQL_HANDLE_STMT, DC_hstmt);
+
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &DC_henv);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(DC_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ DC_henv,
+ &DC_hdbc);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(DC_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT,
+ DC_hdbc,
+ &DC_hstmt);
+ if(retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ strcpy((char *) SQLStmt,
+ "SELECT * FROM Customers");
+
+ //***********************************************
+ //** Prepare and Execute the SQL statement **
+ //***********************************************
+
+ retcode = SQLExecDirect(DC_hstmt,
+ SQLStmt,
+ SQL_NTS);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ //*********************************
+ //** ColumnNumber is from 1 to 4 **
+ //*********************************
+ ndbout << endl << "ColumnNumber is from 1 to 4" << endl;
+
+ for (int ii = 1; ii <= 4; ii++)
+ {
+ retcode = SQLDescribeCol(DC_hstmt,
+ ii,
+ ColumnName,
+ sizeof(ColumnName),
+ &NameLength,
+ &DataTypePtr,
+ &ColumnSizePtr,
+ &DecimalDigitsPtr,
+ &NullablePtr);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Column Name = " << (char *)ColumnName << endl;
+
+ }
+
+ //*********************************
+ //** Test2 **
+ //** ColumnNumber is less than 1 **
+ //*********************************
+
+ retcode = SQLDescribeCol(DC_hstmt,
+ (SQLUSMALLINT)-1,
+ ColumnName,
+ sizeof(ColumnName),
+ &NameLength,
+ &DataTypePtr,
+ &ColumnSizePtr,
+ &DecimalDigitsPtr,
+ &NullablePtr);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << endl << "ColumnNumber is less than 1" << endl;
+ DescribeCol_DisplayError(SQL_HANDLE_STMT, DC_hstmt);
+
+ //*********************************************************************
+ //** Test3 **
+ //** ColumnNumber is greater than N(the value of the TOP_LEVEL_COUNT **
+ //** field of IRD) **
+ //*********************************************************************
+
+ ndbout << endl <<"ColumnNumber is greater than N(the value"
+ << "of the TOP_LEVEL_COUNTfield of IRD)" << endl;
+
+ retcode = SQLDescribeCol(DC_hstmt,
+ (SQLUSMALLINT)1045,
+ ColumnName,
+ sizeof(ColumnName),
+ &NameLength,
+ &DataTypePtr,
+ &ColumnSizePtr,
+ &DecimalDigitsPtr,
+ &NullablePtr);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DescribeCol_DisplayError(SQL_HANDLE_STMT, DC_hstmt);
+
+}
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(DC_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, DC_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, DC_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, DC_henv);
+
+ return NDBT_OK;
+}
+
+void DescribeCol_DisplayError(SQLSMALLINT DC_HandleType,
+ SQLHSTMT DC_InputHandle)
+{
+ SQLCHAR Sqlstate[5], Msg[DC_MESSAGE_LENGTH];
+ SQLINTEGER NativeError;
+ SQLSMALLINT DC_i, MsgLen;
+ SQLRETURN SQLSTATEs;
+
+ DC_i = 1;
+
+ while ((SQLSTATEs = SQLGetDiagRec(DC_HandleType,
+ DC_InputHandle,
+ DC_i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << DC_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)DC_InputHandle << endl;
+ ndbout << "the return message is:" << (char *)Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ DC_i ++;
+ break;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLDisconnectTest.cpp b/ndb/test/odbc/client/SQLDisconnectTest.cpp
new file mode 100644
index 00000000000..823b446ab84
--- /dev/null
+++ b/ndb/test/odbc/client/SQLDisconnectTest.cpp
@@ -0,0 +1,155 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLDisconnectTest.cpp
+ */
+
+#include <common.hpp>
+#define disc_SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC disc_hdbc;
+SQLHSTMT disc_hstmt;
+SQLHENV disc_henv;
+SQLHDESC disc_hdesc;
+
+void Disconnect_DisplayError_HDBC(SQLSMALLINT disc_HandleType,
+ SQLHDBC disc_InputHandle);
+/**
+ * Test to close the data source connection associated with
+ * a specific connection handle
+ *
+ * -# Normal case testing
+ * @return Zero, if test succeeded
+ */
+
+int SQLDisconnectTest()
+{
+ SQLRETURN disc_retcode;
+ ndbout << endl << "Start SQLDisconnect Testing" << endl;
+
+ // ************************************
+ // ** Allocate an environment handle **
+ // ************************************
+ disc_retcode = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &disc_henv);
+
+ if(disc_retcode == SQL_SUCCESS || disc_retcode == SQL_SUCCESS_WITH_INFO) {
+ ndbout << "Allocated an environment handle!" << endl;
+ } else {
+ ndbout << "Failed to allocate environment handle!" << endl;
+ return NDBT_FAILED;
+ }
+
+ // *********************************************
+ // ** Set the ODBC application Version to 3.x **
+ // *********************************************
+ disc_retcode = SQLSetEnvAttr(disc_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+ if (disc_retcode == SQL_SUCCESS || disc_retcode == SQL_SUCCESS_WITH_INFO) {
+ ndbout << "Set ODBC application version to 3.x" << endl;
+ } else {
+ ndbout << "Failed to set application version!" << endl;
+ return NDBT_FAILED;
+ }
+
+ // **********************************
+ // ** Allocate a connection handle **
+ // **********************************
+ disc_retcode = SQLAllocHandle(SQL_HANDLE_DBC, disc_henv, &disc_hdbc);
+
+ if (disc_retcode == SQL_SUCCESS || disc_retcode == SQL_SUCCESS_WITH_INFO) {
+ ndbout << "Allocated a connection handle!" << endl;
+ } else {
+ ndbout << "Failed to allocate connection handle!" << endl;
+ return NDBT_FAILED;
+ }
+
+ // *******************
+ // ** connect to DB **
+ // *******************
+ disc_retcode = SQLConnect(disc_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ // **********************
+ // ** Disconnect to DB **
+ // **********************
+ disc_retcode = SQLDisconnect(disc_hdbc);
+
+ if (disc_retcode == SQL_INVALID_HANDLE)
+{
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE"
+ << " still appeared. Please check program" << endl;
+ Disconnect_DisplayError_HDBC(SQL_HANDLE_DBC, disc_hdbc);
+}
+
+ if (disc_retcode == SQL_ERROR || disc_retcode == SQL_SUCCESS_WITH_INFO)
+{
+ ndbout << "disconnect retcode = " << disc_retcode << endl;
+ Disconnect_DisplayError_HDBC(SQL_HANDLE_DBC, disc_hdbc);
+}
+ // ******************
+ // ** Free Handles **
+ // ******************
+ SQLFreeHandle(SQL_HANDLE_STMT, disc_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, disc_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, disc_henv);
+
+ return NDBT_OK;
+
+ }
+
+void Disconnect_DisplayError_HDBC(SQLSMALLINT disc_HandleType,
+ SQLHDBC disc_InputHandle)
+{
+ SQLCHAR disc_Msg[disc_SQL_MAXIMUM_MESSAGE_LENGTH];
+ SQLSMALLINT disc_i, disc_MsgLen;
+ SQLINTEGER disc_NativeError;
+ SQLRETURN disc_SQLSTATEs;
+ disc_i = 1;
+ SQLCHAR disc_Sqlstate[5];
+
+ while ((disc_SQLSTATEs = SQLGetDiagRec(disc_HandleType,
+ disc_InputHandle,
+ disc_i,
+ disc_Sqlstate,
+ &disc_NativeError,
+ disc_Msg,
+ sizeof(disc_Msg),
+ &disc_MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << disc_HandleType << endl;
+ ndbout << "the InputHandle is :" <<(long)disc_InputHandle << endl;
+ ndbout << "the output state is:" << (char *)disc_Sqlstate << endl;
+
+ disc_i ++;
+ break;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLDriverConnectTest.cpp b/ndb/test/odbc/client/SQLDriverConnectTest.cpp
new file mode 100644
index 00000000000..fc3b1d10f91
--- /dev/null
+++ b/ndb/test/odbc/client/SQLDriverConnectTest.cpp
@@ -0,0 +1,96 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.hpp>
+#include <string.h>
+
+using namespace std;
+
+SQLHDBC driconn_hdbc;
+SQLHSTMT driconn_hstmt;
+SQLHENV driconn_henv;
+SQLHDESC driconn_hdesc;
+SQLRETURN driconn_retcode, driconn_SQLSTATEs;
+
+#define driconn_SQL_MAXIMUM_MESSAGE_LENGTH 200
+SQLCHAR driconn_Sqlstate[5];
+
+SQLINTEGER driconn_NativeError;
+SQLSMALLINT driconn_i, driconn_MsgLen;
+SQLCHAR driconn_Msg[driconn_SQL_MAXIMUM_MESSAGE_LENGTH], driconn_ConnectIn[30];
+
+void SQLDriverConnectTest_DisplayError_HDBC(SQLSMALLINT driconn_HandleType, SQLHDBC driconn_InputHandle);
+
+int SQLDriverConnectTest()
+{
+ ndbout << endl << "Start SQLDriverConnect Testing" << endl;
+ // Allocate An Environment Handle
+ driconn_retcode = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &driconn_henv);
+
+ // Set the ODBC application Version to 3.x
+ driconn_retcode = SQLSetEnvAttr(driconn_henv, SQL_ATTR_ODBC_VERSION, (SQLPOINTER) SQL_OV_ODBC3, SQL_IS_UINTEGER);
+
+ if (driconn_retcode == SQL_SUCCESS || driconn_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x" << endl;
+
+ // Allocate A Connection Handle
+ driconn_retcode = SQLAllocHandle(SQL_HANDLE_DBC, driconn_henv, &driconn_hdbc);
+
+ if (driconn_retcode == SQL_SUCCESS || driconn_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocation A Connection Handle" << endl;
+
+ // Build A Connection String
+ strcpy((char*) driconn_ConnectIn, "DSN=ndb;UID=x;PWD=y");
+
+ // Connect to NDB
+ driconn_retcode = SQLDriverConnect(driconn_hdbc, NULL, driconn_ConnectIn, SQL_NTS, NULL, 0, NULL, SQL_DRIVER_NOPROMPT);
+ ndbout << "retcode = " << driconn_retcode << endl;
+ ndbout << "Before pringing out information about connection, we print out retcode = " << driconn_retcode << endl;
+
+ if (driconn_retcode == SQL_SUCCESS || driconn_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to NDB" << endl;
+
+ if (driconn_retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ else
+ { if (driconn_retcode == SQL_ERROR || driconn_retcode == SQL_SUCCESS_WITH_INFO)
+ SQLDriverConnectTest_DisplayError_HDBC(SQL_HANDLE_DBC, driconn_hdbc);}
+
+ // Free the Connection Handle
+ SQLFreeHandle(SQL_HANDLE_DBC, driconn_hdbc);
+
+ // Free the Environment Handle
+ SQLFreeHandle(SQL_HANDLE_ENV, driconn_henv);
+
+ return 0;
+ }
+
+void SQLDriverConnectTest_DisplayError_HDBC(SQLSMALLINT driconn_HandleType, SQLHDBC driconn_InputHandle)
+{
+ driconn_i = 1;
+ while ((driconn_SQLSTATEs = SQLGetDiagRec(driconn_HandleType, driconn_InputHandle, driconn_i,
+ driconn_Sqlstate, &driconn_NativeError, driconn_Msg, sizeof(driconn_Msg),
+ &driconn_MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << driconn_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)driconn_InputHandle << endl;
+ ndbout << "the output state is:" << (char *)driconn_Sqlstate << endl;
+
+ driconn_i ++;
+ }
+
+}
diff --git a/ndb/test/odbc/client/SQLEndTranTest.cpp b/ndb/test/odbc/client/SQLEndTranTest.cpp
new file mode 100644
index 00000000000..06c497954fd
--- /dev/null
+++ b/ndb/test/odbc/client/SQLEndTranTest.cpp
@@ -0,0 +1,108 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+
+using namespace std;
+
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLINTEGER strangehandle;
+SQLRETURN retcode, retcodeprepare, SQLSTATEs;
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLSMALLINT Not_In_Table13;
+
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+
+void SQLEndTran_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int SQLEndTranTest()
+{
+
+ strangehandle = 67;
+ /* hstmt */
+ // Execute a statement to retrieve rows from the Customers table. We can create the table and
+ // inside rows into NDB by program TestDirectSQL
+ // retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", 56);
+
+ retcodeprepare = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcodeprepare == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ retcode = SQLExecute(hstmt);
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+
+ /* HandleType is not in Table 13 */
+ Not_In_Table13 = 67;
+ SQLSTATEs = SQLEndTran(Not_In_Table13, (void*)strangehandle , SQL_COMMIT);
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(67, 67, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:67" << endl;
+ ndbout << "the InputHandle is :67" << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+ }
+
+ /* HandleType is STATEMENT HANDLE, if the value of Handle does not identity an allocated SQL_statement */
+ SQLSTATEs = SQLEndTran(SQL_HANDLE_STMT, hdbc, SQL_COMMIT);
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLEndTran_DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ /* The value of CompletionType is not in Table 14 */
+ SQLSTATEs = SQLEndTran(SQL_HANDLE_STMT, hstmt, 8888);
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLEndTran_DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ }
+
+ }
+ return 0;
+
+ }
+
+
+void SQLEndTran_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLErrorTest.cpp b/ndb/test/odbc/client/SQLErrorTest.cpp
new file mode 100644
index 00000000000..5220e7b5eed
--- /dev/null
+++ b/ndb/test/odbc/client/SQLErrorTest.cpp
@@ -0,0 +1,107 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#if 0
+
+#include <NdbOut.hpp>
+#include <sqlcli.h>
+#include <stdio.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR Sqlstate[5];
+
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+SQLCHAR szName[NAME_LEN], szPhone[PHONE_LEN];
+SQLINTEGER sCustID, cbName, cbCustID, cbPhone, NativeError;
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int SQLBindColTest ()
+{
+
+ /* hstmt */
+ // Execute a statement to retrieve rows from the Customers table. We can create the table and inside rows in
+ // NDB by another program TestDirectSQL. In this test program(SQLBindColTest),we only have three rows in
+ // table CUSTOMERS
+
+retcode = SQLExecDirect(hstmt, (SQLCHAR*)"SELECT CUSTID, NAME, PHONE FROM CUSTOMERS ORDER BY 2, 1, 3", SQL_NTS);
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+SQLBindCol(hstmt, 0, SQL_C_ULONG, &sCustID, 0, &cbCustID);
+while (TRUE) {
+retcode = SQLFetch(hstmt);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ }
+
+
+SQLBindCol(hstmt, 4, SQL_C_ULONG, &sCustID, 0, &cbCustID);
+while (TRUE) {
+retcode = SQLFetch(hstmt);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ }
+
+/* Bind columns 1, 2, and 3 */
+SQLBindCol(hstmt, 1, SQL_C_ULONG, &sCustID, 0, &cbCustID);
+SQLBindCol(hstmt, 2, SQL_C_CHAR, szName, NAME_LEN, &cbName);
+SQLBindCol(hstmt, 3, SQL_C_CHAR, szPhone, PHONE_LEN, &cbPhone);
+/* Fetch and print each row of data. On */
+/* an error, display a message and exit. */
+while (TRUE) {
+retcode = SQLFetch(hstmt);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ }
+ }
+
+ return 0;
+
+ }
+
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+#endif
diff --git a/ndb/test/odbc/client/SQLExecDirectTest.cpp b/ndb/test/odbc/client/SQLExecDirectTest.cpp
new file mode 100644
index 00000000000..b9b4e770412
--- /dev/null
+++ b/ndb/test/odbc/client/SQLExecDirectTest.cpp
@@ -0,0 +1,353 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLExecDirectTest.cpp
+ */
+#include <common.hpp>
+#define EXD_MESSAGE_LENGTH 200
+#define EXD_NAME_LEN 10
+#define EXD_PHONE_LEN 10
+#define EXD_ADDRESS_LEN 10
+using namespace std;
+
+SQLHDBC EXD_hdbc;
+SQLHSTMT EXD_hstmt;
+SQLHENV EXD_henv;
+SQLHDESC EXD_hdesc;
+SQLRETURN EXD_ret, SQLSTATEs;
+
+void ExecDirect_DisplayError(SQLSMALLINT EXD_HandleType,
+ SQLHSTMT EXD_InputHandle);
+
+int EXD_Display_Result(SQLHSTMT EXDR_InputHandle);
+
+/**
+ * Test to execute a prepared ststement
+ *
+ * -# Normal case: Prepare and Execute a prepared statement
+ * -# Prepare and Execute an empty statement
+ * -# Prepare and Execute a statement with wrong henv handle
+ * -# Prepare and Execute a statement with wrong hdbc handle
+ * -# Prepare and Execute a statement with wrong hdesc handle
+ * @return Zero, if test succeeded
+ */
+
+int SQLExecDirectTest()
+{
+ ndbout << endl << "Start ExecDirect Testing" << endl;
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ EXD_ret = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &EXD_henv);
+
+if (EXD_ret == SQL_SUCCESS || EXD_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ EXD_ret = SQLSetEnvAttr(EXD_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+if (EXD_ret == SQL_SUCCESS || EXD_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ EXD_ret = SQLAllocHandle(SQL_HANDLE_DBC,
+ EXD_henv,
+ &EXD_hdbc);
+
+if (EXD_ret == SQL_SUCCESS || EXD_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ EXD_ret = SQLConnect(EXD_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+if (EXD_ret == SQL_SUCCESS || EXD_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ EXD_ret = SQLAllocHandle(SQL_HANDLE_STMT,
+ EXD_hdbc,
+ &EXD_hstmt);
+if(EXD_ret == SQL_SUCCESS || EXD_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //**********************************************
+ //** Test1 **
+ //** Prepare and Execute a prepared statement **
+ //**********************************************
+ EXD_ret = SQLExecDirect(EXD_hstmt,
+ (SQLCHAR*)"SELECT * FROM Customers",
+ SQL_NTS);
+
+ if (EXD_ret == SQL_INVALID_HANDLE)
+ {
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but SQL_INVALID_HANDLE" << endl;
+ ndbout << "still appeared. Please check program" << endl;
+ }
+
+ if (EXD_ret == SQL_ERROR || EXD_ret == SQL_SUCCESS_WITH_INFO)
+ ExecDirect_DisplayError(SQL_HANDLE_STMT, EXD_hstmt);
+
+ //*************************
+ //** Display the results **
+ //*************************
+
+ EXD_Display_Result(EXD_hstmt);
+
+ //*******************************************
+ //** Test2 **
+ //** Prepare and Execute an empty statement**
+ //** in order to see what will happen **
+ //*******************************************
+ EXD_ret = SQLExecDirect(EXD_hstmt,
+ (SQLCHAR*)" ",
+ SQL_NTS);
+
+ if (EXD_ret == SQL_ERROR || EXD_ret == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Prepare and Execute an empty statement," << endl;
+ ndbout << "The following case happened!" << endl;
+ ExecDirect_DisplayError(SQL_HANDLE_STMT, EXD_hstmt);
+ }
+
+ //***************************************************************
+ //** Test3 **
+ //** Prepare and Execute a statement with wrong henv handle **
+ //** in order to see what will happen **
+ //***************************************************************
+ EXD_ret = SQLExecDirect(EXD_henv,
+ (SQLCHAR*)"SELECT * FROM Customers",
+ SQL_NTS);
+
+ if (EXD_ret == SQL_SUCCESS_WITH_INFO || EXD_ret == SQL_SUCCESS)
+ { ndbout << "Handle Type is SQL_HANDLE_HENV, but SQL_INVALID_HANDLE" << endl;
+ ndbout << "still appeared. Please check programm" << endl;
+ ExecDirect_DisplayError(SQL_HANDLE_ENV, EXD_henv);
+ }
+
+ //******************************************************************
+ //** Test4 **
+ //** Prepare and Execute a statement with wrong hdbc handle **
+ //** in order to see what will happen **
+ //******************************************************************
+
+ EXD_ret = SQLExecDirect(EXD_hdbc,
+ (SQLCHAR*)"SELECT * FROM Customers",
+ SQL_NTS);
+
+ if (EXD_ret == SQL_SUCCESS_WITH_INFO || EXD_ret == SQL_SUCCESS)
+ ExecDirect_DisplayError(SQL_HANDLE_DBC, EXD_hdbc);
+
+ //*******************************************************************
+ //** Test5 **
+ //** Prepare and Execute a statement with wrong hdesc handle **
+ //** in order to see what will happen **
+ //*******************************************************************
+
+ EXD_ret = SQLExecDirect(EXD_hdesc,
+ (SQLCHAR*)"SELECT * FROM Customers",
+ SQL_NTS);
+
+ if (EXD_ret == SQL_SUCCESS_WITH_INFO || EXD_ret == SQL_SUCCESS)
+ {
+ ndbout << "Handle Type is SQL_HANDLE_DESC, but SQL_SUCCESS_WITH_INFO" <<endl;
+ ndbout << "appeared. Please check program" << endl;
+ ExecDirect_DisplayError(SQL_HANDLE_DESC, EXD_hdesc);
+ }
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(EXD_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, EXD_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, EXD_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, EXD_henv);
+
+ return NDBT_OK;
+
+ }
+
+
+void ExecDirect_DisplayError(SQLSMALLINT EXD_HandleType,
+ SQLHSTMT EXD_InputHandle)
+{
+ SQLCHAR EXD_Sqlstate[5];
+ SQLINTEGER EXD_NativeError;
+ SQLSMALLINT EXD_i, EXD_MsgLen;
+ SQLCHAR EXD_Msg[EXD_MESSAGE_LENGTH];
+ SQLRETURN SQLSTATEs;
+ EXD_i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(EXD_HandleType,
+ EXD_InputHandle,
+ EXD_i,
+ EXD_Sqlstate,
+ &EXD_NativeError,
+ EXD_Msg,
+ sizeof(EXD_Msg),
+ &EXD_MsgLen))
+ != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << EXD_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)EXD_InputHandle << endl;
+ ndbout << "the ColAtt_Msg is: " << (char *) EXD_Msg << endl;
+ ndbout << "the output state is:" << (char *)EXD_Sqlstate << endl;
+
+ EXD_i ++;
+ // break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+int EXD_Display_Result(SQLHSTMT EXDR_InputHandle)
+{
+ SQLRETURN EXD_retcode;
+ unsigned long EXD_CustID;
+ SQLCHAR EXD_Name[EXD_NAME_LEN], EXD_Phone[EXD_PHONE_LEN];
+ SQLCHAR EXD_Address[EXD_ADDRESS_LEN];
+
+ //*********************
+ //** Bind columns 1 **
+ //*********************
+ EXD_retcode =SQLBindCol(EXDR_InputHandle,
+ 1,
+ SQL_C_ULONG,
+ &EXD_CustID,
+ sizeof(EXD_CustID),
+ NULL);
+ if (EXD_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLBindCol, SQL_ERROR happened!" << endl;
+ ExecDirect_DisplayError(SQL_HANDLE_STMT, EXDR_InputHandle);
+ return NDBT_FAILED;
+ }
+ //*********************
+ //** Bind columns 2 **
+ //*********************
+
+ EXD_retcode =SQLBindCol(EXDR_InputHandle,
+ 2,
+ SQL_C_CHAR,
+ &EXD_Name,
+ EXD_NAME_LEN,
+ NULL);
+ if (EXD_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLBindCol, SQL_ERROR happened!" << endl;
+ ExecDirect_DisplayError(SQL_HANDLE_STMT, EXDR_InputHandle);
+ return NDBT_FAILED;
+ }
+
+ //*********************
+ //** Bind columns 3 **
+ //*********************
+
+ EXD_retcode = SQLBindCol(EXDR_InputHandle,
+ 3,
+ SQL_C_CHAR,
+ &EXD_Address,
+ EXD_ADDRESS_LEN,
+ NULL);
+
+ if (EXD_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLBindCol, SQL_ERROR happened!" << endl;
+ ExecDirect_DisplayError(SQL_HANDLE_STMT, EXDR_InputHandle);
+ return NDBT_FAILED;
+ }
+
+ //*********************
+ //** Bind columns 4 **
+ //*********************
+
+ EXD_retcode = SQLBindCol(EXDR_InputHandle,
+ 4,
+ SQL_C_CHAR,
+ &EXD_Phone,
+ EXD_PHONE_LEN,
+ NULL);
+
+ if (EXD_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLBindCol, SQL_ERROR happened!" << endl;
+ ExecDirect_DisplayError(SQL_HANDLE_STMT, EXDR_InputHandle);
+ return NDBT_FAILED;
+ }
+
+ //*****************************************
+ //* Fetch and print each row of data. On **
+ //* an error, display a message and exit **
+ //*****************************************
+
+ if (EXD_retcode != SQL_ERROR)
+ EXD_retcode = SQLFetch(EXDR_InputHandle);
+
+ ndbout << endl << "EXD_retcode = SQLFetch(EXDR_InputHandle) = "
+ << EXD_retcode << endl;
+
+ if (EXD_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLFetch, SQL_ERROR happened!" << endl;
+ ExecDirect_DisplayError(SQL_HANDLE_STMT, EXDR_InputHandle);
+ return NDBT_FAILED;
+ }
+ else if (EXD_retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "CustID = " << (int)EXD_CustID << endl;
+ ndbout << "Name = " << (char *)EXD_Name << endl;
+ ndbout << "Address = " << (char *)EXD_Address << endl;
+ ndbout << "Phone = " << (char *)EXD_Phone << endl;
+ ExecDirect_DisplayError(SQL_HANDLE_STMT, EXDR_InputHandle);
+ }
+ else
+ {
+ ndbout << "CustID = " << (int)EXD_CustID << endl;
+ ndbout << "Name = " << (char *)EXD_Name << endl;
+ ndbout << "Address = " << (char *)EXD_Address << endl;
+ ndbout << "Phone = " << (char *)EXD_Phone << endl;
+ }
+ return 0;
+}
diff --git a/ndb/test/odbc/client/SQLExecuteTest.cpp b/ndb/test/odbc/client/SQLExecuteTest.cpp
new file mode 100644
index 00000000000..5f6bdb5d4bf
--- /dev/null
+++ b/ndb/test/odbc/client/SQLExecuteTest.cpp
@@ -0,0 +1,122 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLExecuteTest.cpp
+ */
+
+#include <common.hpp>
+#define ESQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC Ehdbc;
+SQLHSTMT Ehstmt;
+SQLHENV Ehenv;
+SQLHDESC Ehdesc;
+
+void Execute_DisplayError(SQLSMALLINT EHandleType,
+ SQLHSTMT EInputHandle);
+
+/**
+ * Test to execute a SQL statement in a data result set
+ *
+ * Tests:
+ * -# Test1 There is no executed statement
+ * @return Zero, if test succeeded
+ */
+int SQLExecuteTest()
+{
+
+ SQLRETURN retcode;
+ /* hstmt */
+ retcode = SQLExecute(Ehstmt);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but SQL_INVALID_HANDLE" << endl;
+ ndbout << "still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ Execute_DisplayError(SQL_HANDLE_STMT, Ehstmt);
+
+ /* henv */
+ retcode = SQLExecute(Ehenv);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but SQL_SUCCESS_WITH_INFO"
+ << "still appeared. Please check programm" << endl;
+ // Execute_DisplayError(SQL_HANDLE_ENV, Ehenv);
+
+ /* hdbc */
+ retcode = SQLExecute(Ehdbc);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but SQL_SUCCESS_WITH_INFO"
+ <<"still appeared. Please check programm" << endl;
+ // Execute_DisplayError(SQL_HANDLE_DBC, Ehdbc);
+
+ /* hdesc */
+ retcode = SQLExecute(Ehdesc);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ ndbout << "Handle Type is SQL_HANDLE_DESC, but SQL_SUCCESS_WITH_INFO"
+ << "still appeared. Please check programm" << endl;
+ // Execute_DisplayError(SQL_HANDLE_DESC, Ehdesc);
+
+ return NDBT_OK;
+
+ }
+
+
+void Execute_DisplayError(SQLSMALLINT EHandleType,
+ SQLHSTMT EInputHandle)
+{
+ SQLCHAR Sqlstate[5];
+
+ SQLINTEGER NativeError;
+ SQLSMALLINT i, MsgLen;
+ SQLCHAR Msg[ESQL_MAXIMUM_MESSAGE_LENGTH];
+ SQLRETURN SQLSTATEs;
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(EHandleType,
+ EInputHandle,
+ i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << EHandleType << endl;
+ ndbout << "the InputHandle is :" << EInputHandle << endl;
+ ndbout << "the Msg is :" << (char *)Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ break;
+
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLFetchScrollTest.cpp b/ndb/test/odbc/client/SQLFetchScrollTest.cpp
new file mode 100644
index 00000000000..4a11ccd143e
--- /dev/null
+++ b/ndb/test/odbc/client/SQLFetchScrollTest.cpp
@@ -0,0 +1,82 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <stdio.h>
+#include <sqlext.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+SQLHSTMT hstmt;
+SQLSMALLINT RecNumber;
+SQLCHAR szSalesPerson[SALES_PERSON_LEN];
+
+SQLCHAR Sqlstate[5], Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLINTEGER NativeError;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLINTEGER ValuePtr1;
+SQLCHAR ValuePtr2;
+SQLSMALLINT ValuePtr3;
+SQLSMALLINT i, MsgLen;
+
+void SFCT_DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle);
+
+int SQLFetchScrollTest ()
+{
+
+ // FetchScroll a statement to retrieve rows from the Customers table. We can
+ // create the table and insert rows in NDB by program TestDirectSQL
+
+ /* There is no executed statement associated with the allocated SQL-statement identified by StatementHandle */
+retcode = SQLFetchScroll(hstmt, SQL_FETCH_NEXT, 1);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SFCT_DisplayError(SQL_HANDLE_DESC, hstmt);
+
+ /* FetchOrientation is not one of the code values in Table24 */
+retcode = SQLFetchScroll(hstmt, SQL_FETCH_NEXT, 8);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SFCT_DisplayError(SQL_HANDLE_DESC, hstmt);
+
+ return 0;
+
+ }
+
+
+void SFCT_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLFetchTest.cpp b/ndb/test/odbc/client/SQLFetchTest.cpp
new file mode 100644
index 00000000000..bd62fcb2f04
--- /dev/null
+++ b/ndb/test/odbc/client/SQLFetchTest.cpp
@@ -0,0 +1,438 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLFetchTest.cpp
+ */
+
+#include <common.hpp>
+#define F_MESSAGE_LENGTH 200
+using namespace std;
+
+#define F_NAME_LEN 20
+#define F_PHONE_LEN 20
+#define F_ADDRESS_LEN 20
+
+SQLHSTMT F_hstmt;
+SQLHDESC F_hdbc;
+SQLHENV F_henv;
+SQLHDESC F_hdesc;
+
+void SQLFetchTest_DisplayError(SQLSMALLINT F_HandleType,
+ SQLHDESC F_InputHandle);
+
+/**
+ * Test to advance a cursor to the next row of data in a data result set
+ * and to retrieve data from any bound columns that exist for that row
+ * into their associated application variables
+ *
+ * Tests:
+ * _# Test1 Execute statements and display the results
+ * -# Test2 There is no executed statement
+ * @return Zero, if test succeeded
+ */
+int SQLFetchTest()
+{
+ SQLRETURN retcode;
+ SQLCHAR SQLStmt[120];
+ SQLCHAR SQLStmt1[120];
+ SQLCHAR SQLStmt2[120];
+ SQLCHAR SQLStmt3[120];
+ SQLCHAR SQLStmt4[120];
+
+ SQLCHAR F_CustID[20];
+ SQLCHAR F_Name[F_NAME_LEN], F_Phone[F_PHONE_LEN];
+ SQLCHAR F_Address[F_ADDRESS_LEN];
+
+ ndbout << "Start SQLFetch Testing!" << endl;
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &F_henv);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(F_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ F_henv,
+ &F_hdbc);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(F_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT,
+ F_hdbc,
+ &F_hstmt);
+if(retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ /* *** CustID is Integer *** */
+ strcpy((char *) SQLStmt1, "CREATE TABLE Customers (CustID Integer, Name Char(12), Address Char(12), Phone Char(12), Primary Key(CustID, Name))");
+
+ /* *** the primary key is alone *** */
+// strcpy((char *) SQLStmt1, "CREATE TABLE Customers (CustID Integer, Name Char(12), Address Char(12), Phone Char(12), Primary Key(CustID))");
+
+ strcpy((char *) SQLStmt2, "INSERT INTO Customers (CustID, Name, Address,Phone) VALUES(188, 'peter','LM Vag8','7190890')");
+
+ /* *** CustID is Float *** */
+// strcpy((char *) SQLStmt1,
+// "CREATE TABLE Customers (CustID float, Name Char(12), Address Char(12), Phone Char(12), Primary Key(CustID))");
+// strcpy((char *) SQLStmt2, "INSERT INTO Customers (CustID, Name, Address,Phone) VALUES(1.1516, 'peter','LM Vag8','7190890')");
+
+ /* *** CustID is Char *** */
+ // strcpy((char *) SQLStmt1, "CREATE TABLE Customers (CustID char(6), Name Char(12), Address Char(12), Phone Char(12), Primary Key(CustID))");
+
+ // strcpy((char *) SQLStmt2, "INSERT INTO Customers (CustID, Name, Address,Phone) VALUES('000001', 'peter','LM Vag8','7190890')");
+
+ /* The UPDATE statements */
+ // strcpy((char *) SQLStmt3, "UPDATE Customers SET Phone = '98998' WHERE CustID = 1.1516");
+
+ // strcpy((char *) SQLStmt3, "UPDATE Customers SET Phone = '98998' WHERE CustID = '000001'");
+
+ // strcpy((char *) SQLStmt3, "UPDATE Customers SET Phone = '98998' WHERE CustID = 188");
+
+ strcpy((char *) SQLStmt3, "UPDATE Customers SET Phone = '98998' WHERE CustID = 188 AND Name = 'peter'");
+
+ // DELETE statements
+
+ // DELETE all records
+ // strcpy((char *) SQLStmt4, "DELETE FROM Customers");
+
+ // DELETE One record
+ // strcpy((char *) SQLStmt4, "DELETE FROM Customers WHERE CustID = 1.1516");
+ // strcpy((char *) SQLStmt4, "DELETE FROM Customers WHERE CustID = '000001'");
+ // strcpy((char *) SQLStmt4, "DELETE FROM Customers WHERE CustID = 188 AND Name = 'peter'");
+ // strcpy((char *) SQLStmt4, "DELETE FROM Customers WHERE CustID = 188");
+
+ strcpy((char *) SQLStmt4, "DELETE FROM Customers WHERE CustID = 188 AND Name = 'peter'");
+
+ //SELECT statements
+ strcpy((char *) SQLStmt, "SELECT * FROM Customers");
+
+ //********************************
+ //** Prepare CREATE statements **
+ //********************************
+
+ ndbout << ">>>>" << (char*)SQLStmt1 << "<<<<" << endl;
+ retcode = SQLPrepare(F_hstmt,
+ SQLStmt1,
+ SQL_NTS);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+
+ //******************************************************************
+ //** There is no executed statement associated with the allocated **
+ //** SQL-statement identified by StatementHandle **
+ //******************************************************************
+
+ //This function is correct after testing. We don't test again.
+ /*
+ retcode = SQLFetch(F_hstmt);
+ ndbout << endl << "retcode = SQLFetch(F_hstmt) = " << retcode << endl;
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "There is no executed statement associated with" << endl;
+ ndbout << "the allocated SQL-statement" << endl;
+ SQLFetchTest_DisplayError(SQL_HANDLE_DESC, F_hstmt);
+
+ }
+ */
+
+ //*******************************
+ //** Execute CREATE statement **
+ //*******************************
+
+ retcode = SQLExecute(F_hstmt);
+
+ if (retcode == 0)
+ ndbout << endl << "Execute CREATE TABLE Statement OK!" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+
+ //********************************
+ //** Prepare INSERT statements **
+ //********************************
+
+ ndbout << ">>>>" << (char*)SQLStmt2 << "<<<<" << endl;
+ retcode = SQLPrepare(F_hstmt,
+ SQLStmt2,
+ SQL_NTS);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+
+
+ //******************************
+ //** Execute INSERT statement **
+ //******************************
+ retcode = SQLExecute(F_hstmt);
+
+ if (retcode == 0)
+ ndbout << endl <<"Execute INSERT Statement OK!" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+
+ //********************************
+ //** Prepare UPDATE statements **
+ //********************************
+
+ ndbout << ">>>>" << (char*)SQLStmt3 << "<<<<" << endl;
+ retcode = SQLPrepare(F_hstmt,
+ SQLStmt3,
+ SQL_NTS);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+
+ //******************************
+ //** Execute UPDATE statement **
+ //******************************
+ retcode = SQLExecute(F_hstmt);
+
+ if (retcode == 0)
+ ndbout << endl <<"Execute UPDATE Statement OK!" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+
+ //********************************
+ //** Prepare DELETE statements **
+ //********************************
+ ndbout << ">>>>" << (char*)SQLStmt4 << "<<<<" << endl;
+ retcode = SQLPrepare(F_hstmt,
+ SQLStmt4,
+ SQL_NTS);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << endl << "Preparing DELETE Statement failure!" << endl;
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+ }
+
+ //******************************
+ //** Execute DELETE statement **
+ //******************************
+
+ retcode = SQLExecute(F_hstmt);
+
+ if (retcode == 0)
+ ndbout << endl <<"Execute DELETE Statement OK!" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "DELETE Statement executing failure!" << endl;
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+ }
+ //********************************
+ //** Prepare SELECT statements **
+ //********************************
+ ndbout << ">>>>" << (char*)SQLStmt << "<<<<" << endl;
+ retcode = SQLPrepare(F_hstmt,
+ SQLStmt,
+ SQL_NTS);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+
+ /*
+ //******************************
+ //** Execute SELECT statement **
+ //******************************
+
+ retcode = SQLExecute(F_hstmt);
+
+ if (retcode == 0)
+ ndbout << endl <<"Execute SELECT Statement OK!" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+ */
+
+ //********************
+ //** Bind columns **
+ //********************
+
+ retcode =SQLBindCol(F_hstmt,
+ 1,
+ SQL_C_CHAR,
+ F_CustID,
+ sizeof(F_CustID),
+ NULL);
+ ndbout << endl << "Bind Col1 retcode = " << retcode << " OK!" << endl;
+
+ retcode =SQLBindCol(F_hstmt,
+ 2,
+ SQL_C_CHAR,
+ F_Name,
+ F_NAME_LEN,
+ NULL);
+
+ ndbout << "Bind Col2 retcode = " << retcode << " OK!" << endl;
+
+ retcode = SQLBindCol(F_hstmt,
+ 3,
+ SQL_C_CHAR,
+ F_Address,
+ F_ADDRESS_LEN,
+ NULL);
+
+ ndbout << "Bind Col3 retcode = " << retcode << " OK!" << endl;
+
+ retcode = SQLBindCol(F_hstmt,
+ 4,
+ SQL_C_CHAR,
+ F_Phone,
+ F_PHONE_LEN,
+ NULL);
+
+ ndbout << "Bind Col4 retcode = " << retcode << " OK!" << endl;
+
+ //******************************
+ //** Execute SELECT statement **
+ //******************************
+
+ retcode = SQLExecute(F_hstmt);
+
+ if (retcode == 0)
+ ndbout << endl <<"Execute SELECT Statement OK!" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+
+ //***************
+ //* Fetch data **
+ //***************
+ ndbout << endl <<"Executing Fetch SELECT Statement ......" << endl;
+
+ retcode = SQLFetch(F_hstmt);
+
+ if (retcode == 100)
+ ndbout << endl <<"Execute Fetch SELECT Statement, But No DATA!" << endl;
+
+ if (retcode == 0)
+ ndbout << endl <<"Execute Fetch SELECT Statement OK!" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLFetchTest_DisplayError(SQL_HANDLE_STMT, F_hstmt);
+
+ //*******************
+ //* Display result **
+ //*******************
+ ndbout << endl << "The results is : " << endl;
+ ndbout << "CustID = " << (char *)F_CustID << endl;
+ ndbout << "Name = " << (char *)F_Name << endl;
+ ndbout << "Address = " << (char *)F_Address << endl;
+ ndbout << "Phone = " << (char *)F_Phone << endl;
+
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(F_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, F_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, F_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, F_henv);
+
+ return NDBT_OK;
+
+ }
+
+
+void SQLFetchTest_DisplayError(SQLSMALLINT F_HandleType,
+ SQLHSTMT F_InputHandle)
+{
+ SQLCHAR Sqlstate[50], Msg[F_MESSAGE_LENGTH];
+ SQLRETURN SQLSTATEs;
+ SQLINTEGER NativeError;
+ SQLSMALLINT i, MsgLen;
+ Msg[0] = 0;
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(F_HandleType,
+ F_InputHandle,
+ i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+{
+
+ ndbout << "the HandleType is:" << F_HandleType << endl;
+ ndbout << "the InputHandle is :" <<(long)F_InputHandle << endl;
+ ndbout << "the Msg is :" << (char *)Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ break;
+}
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLFreeHandleTest.cpp b/ndb/test/odbc/client/SQLFreeHandleTest.cpp
new file mode 100644
index 00000000000..3a7241dbe68
--- /dev/null
+++ b/ndb/test/odbc/client/SQLFreeHandleTest.cpp
@@ -0,0 +1,195 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "common.h"
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+int strangehandle;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+
+
+void freehandle_deal_with_HSTMT(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+void freehandle_deal_with_HENV(SQLSMALLINT HandleType, SQLHENV InputHandle);
+void freehandle_deal_with_HDESC(SQLSMALLINT HandleType, SQLHDESC InputHandle);
+void freehandle_deal_with_HDBC(SQLSMALLINT HandleType, SQLHDBC InputHandle);
+
+void freehandle_DisplayError_HDBC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDBC InputHandle);
+void freehandle_DisplayError_HSTMT(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+void freehandle_DisplayError_HENV(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHENV InputHandle);
+void freehandle_DisplayError_HDESC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDESC InputHandle);
+
+int SQLFreeHandleTest ()
+{
+
+strangehandle = 67;
+
+/* ENV */
+ndbout << "Environment Handle" << endl;
+SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+freehandle_deal_with_HENV(SQL_HANDLE_ENV, henv);
+
+/* DBC */
+ndbout << "Connection Handle" << endl;
+SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
+SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);
+freehandle_deal_with_HDBC(SQL_HANDLE_DBC, hdbc);
+
+/* STMT */
+ndbout << "Statement Handle" << endl;
+SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);
+SQLAllocHandle(SQL_HANDLE_STMT, hdbc, &hstmt);
+freehandle_deal_with_HSTMT(SQL_HANDLE_STMT, hstmt);
+
+/* DESC */
+ndbout << "Descriptor Handle" << endl;
+SQLAllocHandle(SQL_HANDLE_DESC, hdbc, &hdesc);
+freehandle_deal_with_HDESC(SQL_HANDLE_DESC, hdesc);
+
+return 0;
+
+}
+
+
+void freehandle_deal_with_HDBC(SQLSMALLINT HandleType, SQLHDBC InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLFreeHandle(HandleType, InputHandle);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+ ndbout << "the InputHandle is SQLHDBC:" << InputHandle << endl;
+ ndbout << "retcode = " << retcode << endl;
+ /*
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ DisplayError_HDBC_free(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ */
+ }
+
+
+void freehandle_deal_with_HSTMT(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLFreeHandle(HandleType, InputHandle);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+ ndbout << "the InputHandle is SQLHSTMT:" << InputHandle << endl;
+ ndbout << "retcode = " << retcode << endl;
+ /*
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ DisplayError_HSTMT_free(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ */
+ }
+
+void freehandle_deal_with_HENV(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLFreeHandle(HandleType, InputHandle);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+ ndbout << "the InputHandle is SQLHENV:" << InputHandle << endl;
+ ndbout << "retcode = " << retcode << endl;
+ /*
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ DisplayError_HENV_free(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ */
+ }
+
+void freehandle_deal_with_HDESC(SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLFreeHandle(HandleType, InputHandle);
+
+ ndbout << "the HandleType is : " << HandleType << endl;
+ ndbout << "the InputHandle is SQLHDESC:" << InputHandle << endl;
+ ndbout << "retcode = " << retcode << endl;
+ /*
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ DisplayError_HDESC_free(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ */
+ }
+
+
+void freehandle_DisplayError_HENV(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+void freehandle_DisplayError_HDBC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDBC InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+void freehandle_DisplayError_HSTMT(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+void freehandle_DisplayError_HDESC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
diff --git a/ndb/test/odbc/client/SQLFreeStmtTest.cpp b/ndb/test/odbc/client/SQLFreeStmtTest.cpp
new file mode 100644
index 00000000000..e636b3063de
--- /dev/null
+++ b/ndb/test/odbc/client/SQLFreeStmtTest.cpp
@@ -0,0 +1,182 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+int strange_handle;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+
+struct handle_set
+{
+SQLHDBC hdbc_varible;
+SQLHSTMT hstmt_varible;
+SQLHENV henv_varible;
+SQLHDESC hdesc_varible;
+int strangehandle;
+};
+handle_set handlevalue;
+
+void handle_deal_with_HSTMT(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+void handle_deal_with_HENV(SQLSMALLINT HandleType, SQLHENV InputHandle);
+void handle_deal_with_HDESC(SQLSMALLINT HandleType, SQLHDESC InputHandle);
+void handle_deal_with_HDBC(SQLSMALLINT HandleType, SQLHDBC InputHandle);
+
+void DisplayError_HDBC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDBC InputHandle);
+void DisplayError_HSTMT(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+void DisplayError_HENV(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHENV InputHandle);
+void DisplayError_HDESC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDESC InputHandle);
+
+int SQLFreeStmtTest ()
+{
+
+handlevalue.hdbc_varible = hdbc;
+handlevalue.hstmt_varible = hstmt;
+handlevalue.henv_varible = henv;
+handlevalue.hdesc_varible = hdesc;
+handlevalue.strangehandle = 67;
+
+
+/* ENV */
+handle_deal_with_HENV(SQL_HANDLE_ENV, SQL_NULL_HANDLE);
+
+/* DBC */
+handle_deal_with_HDBC(SQL_HANDLE_DBC, SQL_NULL_HANDLE);
+
+/* STMT */
+handle_deal_with_HSTMT(SQL_HANDLE_STMT, SQL_NULL_HANDLE);
+
+/* DESC */
+handle_deal_with_HDESC(SQL_HANDLE_DESC, SQL_NULL_HANDLE);
+
+return 0;
+
+}
+
+
+void handle_deal_with_HDBC(SQLSMALLINT HandleType, SQLHDBC InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLFreeHandle(HandleType, InputHandle);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ DisplayError_HDBC(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ }
+
+
+void handle_deal_with_HSTMT(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLFreeHandle(HandleType, InputHandle);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ DisplayError_HSTMT(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ }
+
+void handle_deal_with_HENV(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLFreeHandle(HandleType, InputHandle);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ DisplayError_HENV(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ }
+
+void handle_deal_with_HDESC(SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+ retcode = SQLFreeHandle(HandleType, InputHandle);
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ DisplayError_HDESC(Sqlstate, HandleType, InputHandle);
+
+ i ++;
+ }
+ }
+ }
+
+
+void DisplayError_HENV(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+
+void DisplayError_HDBC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDBC InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+void DisplayError_HSTMT(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+void DisplayError_HDESC(SQLCHAR Sqlstate[6], SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+}
+
+
diff --git a/ndb/test/odbc/client/SQLGetConnectAttrTest.cpp b/ndb/test/odbc/client/SQLGetConnectAttrTest.cpp
new file mode 100644
index 00000000000..8d5a5c0dbbb
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetConnectAttrTest.cpp
@@ -0,0 +1,131 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLPOINTER ValuePtr;
+SQLINTEGER GetConnectAttr_StringLengthPtr;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void GetConnectAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle);
+
+int SQLGetConnectAttrTest()
+{
+ /* SQL/CLI attributes */
+ // char PtrValue1[2] = {'SQL_TRUE', 'SQL_FALSE'};
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetConnectAttr(hdbc, SQL_ATTR_AUTO_IPD, ValuePtr, 36, &GetConnectAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc); // }
+
+ /* ODBC attributes */
+ /*
+ char PtrValue1[3] = {'SQL_MODE_READ_ONLY', 'SQL_MODE_READ_WRITE'};
+ for (i=0; i < 3; i++) {
+ retcode = SQLGetConnectAttr(hdbc, SQL_ATTR_ACCESS_MODE, (void*)PtrValue1[i], sizeof(PtrValue1[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+
+ char PtrValue2[2] = {'SQL_ASYNC_ENABLE_OFF', 'SQL_ASYNC_ENABLE_ON'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetConnectAttr(hdbc, SQL_ATTR_ASYNC_ENABLE, (void*)PtrValue2[i], sizeof(PtrValue2[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+
+ char PtrValue4[2] = {'SQL_AUTOCOMMIT_OFF', 'SQL_AUTOCOMMIT_ON'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLGetConnectAttr(hdbc, SQL_ATTR_AUTOCOMMIT, (void*)PtrValue4[i], sizeof(PtrValue4[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+ char PtrValue5[2] = {'SQL_CD_TRUE', 'SQL_CD_FALSE'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLGetConnectAttr(hdbc, SQL_ATTR_CONNECTION_DEAD, (void*)PtrValue4[i], sizeof(PtrValue5[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+
+ char PtrValue5[2] = {'SQL_CD_TRUE', 'SQL_CD_FALSE'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLGetConnectAttr(hdbc, SQL_ATTR_CONNECTION_TIMEOUT, (void*)PtrValue4[i], sizeof(PtrValue5[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+ */
+
+ return 0;
+
+ }
+
+
+void GetConnectAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetCursorNameTest.cpp b/ndb/test/odbc/client/SQLGetCursorNameTest.cpp
new file mode 100644
index 00000000000..1e3ed9f557e
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetCursorNameTest.cpp
@@ -0,0 +1,221 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLGetCursorNameTest.cpp
+ */
+
+#include <common.hpp>
+using namespace std;
+
+#define GCN_MESSAGE_LENGTH 50
+
+SQLHSTMT GCN_hstmt;
+SQLHDESC GCN_hdesc;
+SQLHENV GCN_henv;
+SQLHDBC GCN_hdbc;
+
+void GCN_DisplayError(SQLSMALLINT GCN_HandleType,
+ SQLHDESC GCN_InputHandle);
+
+/**
+ * Test to assign a user-defined name to a cursor that is
+ * associated with an active SQL statement handle
+ *
+ * Tests:
+ * -# if there is no user-defined cursor name, then try to
+ * get user-definedcursor name
+ * -# get cursor name in normal case
+ *
+ * @return Zero, if test succeeded
+ */
+
+int SQLGetCursorNameTest()
+{
+ SQLRETURN retcode;
+ SQLCHAR SQLStmt [120];
+ SQLCHAR CursorName [80];
+ SQLSMALLINT CNameSize;
+
+ ndbout << endl << "Start SQLGetCursorName Testing" << endl;
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &GCN_henv);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(GCN_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ GCN_henv,
+ &GCN_hdbc);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(GCN_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT,
+ GCN_hdbc,
+ &GCN_hstmt);
+
+ if(retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ strcpy((char *) SQLStmt,
+ "SELECT * FROM Customers WHERE Address = 'LM Vag 8'");
+
+ //*************************
+ //** Prepare a statement **
+ //*************************
+
+ retcode = SQLPrepare(GCN_hstmt,
+ SQLStmt,
+ SQL_NTS);
+
+ //*************************************************************************
+ //** if there is no user-defined cursor name, try to get the cursor name **
+ //*************************************************************************
+ retcode = SQLGetCursorName(GCN_hstmt,
+ CursorName,
+ sizeof(CursorName),
+ &CNameSize);
+
+ if (retcode != SQL_SUCCESS)
+ {
+ ndbout << endl << "retcode =" << retcode << endl;
+ GCN_DisplayError(SQL_HANDLE_STMT, GCN_hstmt);
+ }
+ else
+ ndbout << endl << "The cursor name is : " << (char *) CursorName << endl;
+
+ //*************************
+ //** Set the cursor name **
+ //*************************
+ retcode = SQLSetCursorName(GCN_hstmt,
+ (char *)"Customer_CURSOR",
+ SQL_NTS);
+
+ //***************************
+ //** Execute the statement **
+ //***************************
+ retcode = SQLExecute(GCN_hstmt);
+
+ //**********************************************
+ //** retrieve and display the new cursor name **
+ //**********************************************
+ retcode = SQLGetCursorName(GCN_hstmt,
+ CursorName,
+ sizeof(CursorName),
+ &CNameSize);
+
+ if (retcode != SQL_SUCCESS)
+ {
+ ndbout << endl << "retcode =" << retcode << endl;
+ GCN_DisplayError(SQL_HANDLE_STMT, GCN_hstmt);
+ }
+ else
+ ndbout << endl << "The cursor name is : " << (char *) CursorName << endl;
+
+ //****************
+ // Free Handles **
+ //****************
+ SQLDisconnect(GCN_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, GCN_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, GCN_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, GCN_henv);
+
+ return NDBT_OK;
+
+ }
+
+
+void GCN_DisplayError(SQLSMALLINT GCN_HandleType, SQLHDESC GCN_InputHandle)
+{
+
+ SQLINTEGER NativeError;
+ SQLCHAR Sqlstate[5], Msg[GCN_MESSAGE_LENGTH];
+ SQLRETURN SQLSTATEs;
+ SQLSMALLINT i, MsgLen;
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(GCN_HandleType,
+ GCN_InputHandle, i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << GCN_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)GCN_InputHandle << endl;
+ ndbout << "the Msg is: " << (char *) Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetDataTest.cpp b/ndb/test/odbc/client/SQLGetDataTest.cpp
new file mode 100644
index 00000000000..9d958c6c953
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetDataTest.cpp
@@ -0,0 +1,358 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLGetDataTest.cpp
+ */
+
+#include <common.hpp>
+using namespace std;
+
+#define GD_MESSAGE_LENGTH 200
+
+SQLHSTMT GD_hstmt;
+SQLHENV GD_henv;
+SQLHDBC GD_hdbc;
+SQLHDESC GD_hdesc;
+
+void GetData_DisplayError(SQLSMALLINT GD_HandleType, SQLHSTMT GD_InputHandle);
+
+/**
+ * Test to retrieve data for a single unbound column
+ * in the current row of a result data set
+ *
+ * Tests:
+ * -# Test1 There is no fetched rowset associated with S
+ * -# Test2 column number is less than zero
+ * -# Test3 fetched rowset is empty
+ * @return Zero, if test succeeded
+ */
+
+int SQLGetDataTest()
+{
+ SQLRETURN retcode;
+ SQLCHAR ColumnName;
+ SQLINTEGER CustID;
+ // SQLCHAR Name, Address, Phone;
+ SQLCHAR SQLStmt [120];
+ SQLCHAR SQLStmt1 [120];
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &GD_henv);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(GD_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.X!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ GD_henv,
+ &GD_hdbc);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(GD_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT,
+ GD_hdbc,
+ &GD_hstmt);
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //*****************************
+ //** Define SELECT statement **
+ //*****************************
+
+ strcpy((char *) SQLStmt, "SELECT * FROM Customers");
+
+
+ //***********************************
+ //** Prepare SELECT SQL statement **
+ //***********************************
+
+ retcode = SQLPrepare(GD_hstmt,
+ SQLStmt,
+ SQL_NTS);
+ ndbout << endl << "Preparing SELECT, retcode = SQLprepare()= "
+ << retcode << endl;
+
+ //*********************************
+ //** Execute prepared statement **
+ //*********************************
+
+ // if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ //{
+
+ retcode = SQLExecute(GD_hstmt);
+
+ ndbout << "Exexuting SELECT, retcode = SQLExecute()= "
+ << retcode << endl;
+
+ // if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ // {
+
+ //*****************************************************************
+ //** Test1 **
+ //** There is no fetched rowset associated with S(SQL-statement) **
+ //*****************************************************************
+
+ retcode = SQLGetData(GD_hstmt,
+ 1,
+ SQL_C_SLONG,
+ &CustID,
+ sizeof(CustID),
+ NULL);
+ ndbout << "retcode = SQLGetData()= " << retcode << endl;
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << endl << "Test 1:" << endl;
+ ndbout << "There is no fetched rowset associated with SQL"
+ << " statement. But system reported SUCCESS or"
+ << " SUCCESS_WITH_INFO. Please check the function!" << endl;
+ GetData_DisplayError(SQL_HANDLE_STMT, GD_hstmt);
+ }
+ else if (retcode == SQL_ERROR)
+ {
+ ndbout << endl << "Test 1:" << endl;
+ ndbout << "There is no fetched rowset associated with SQL"
+ << " statement. The system reported ERROR "
+ << " The function is OK!" << endl;
+ }
+ else
+ ndbout << endl;
+
+ //*******************************
+ //** Fetch Data from database **
+ //*******************************
+
+ retcode = SQLFetch(GD_hstmt);
+
+ ndbout << endl
+ << "Fetching after Executing SELECT, retcode = SQLFetch()= "
+ << retcode << endl;
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+
+ //**************************************
+ //** Test2 **
+ //** column number is less than zero **
+ //**************************************
+
+ retcode = SQLGetData(GD_hstmt,
+ 0,
+ SQL_C_ULONG,
+ &CustID,
+ sizeof(CustID),
+ NULL);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Test 2:" <<"Column number is less than zero"
+ << " The system reported SUCCESS or SUCCESS_WITH_INFO."
+ << " Check the function, please!" <<endl;
+ GetData_DisplayError(SQL_HANDLE_STMT, GD_hstmt);
+ }
+ else if (retcode == SQL_ERROR)
+ {
+ ndbout << "Test 2:" << "Column number is less than zero."
+ << " The system reported SQL_ERROR."
+ << " The function is OK!" << endl;
+ }
+ else
+ ndbout << endl;
+ }
+ // }
+
+ // }
+
+ //*****************************
+ //** Define DELETE statement **
+ //*****************************
+
+ // strcpy((char *) SQLStmt1, "DELETE FROM Customers");
+ strcpy((char *) SQLStmt1, "DELETE FROM Customers WHERE CustID = 568 AND Name = 'Hans Peter'");
+
+ //***********************************
+ //** Prepare DELETE SQL statement **
+ //***********************************
+
+ retcode = SQLPrepare(GD_hstmt,
+ SQLStmt1,
+ SQL_NTS);
+ ndbout << endl << "Preparing DELETE, retcode = SQLPrepare()= "
+ << retcode << endl;
+
+ //****************************************
+ //** Execute prepared DELETE statement **
+ //****************************************
+
+ // if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ // {
+
+ retcode = SQLExecute(GD_hstmt);
+
+ ndbout << "Executing DELETE, retcode = SQLExecute()= "
+ << retcode << endl;
+
+ // if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ // {
+
+ retcode = SQLFetch(GD_hstmt);
+
+ ndbout << "Fetching after Executing DELETE, retcode = SQLExecute()= "
+ << retcode << endl;
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+
+ //******************************************************
+ //** Test3 **
+ //** If the fetched rowset associated with **
+ //** Statement is empty, condition is raised: NO DATA **
+ //** We can delete all rows in table Customers for **
+ //** this case **
+ //******************************************************
+
+ retcode = SQLGetData(GD_hstmt,
+ 1,
+ SQL_C_ULONG,
+ &CustID,
+ sizeof(CustID),
+ NULL);
+
+ if (retcode == SQL_ERROR)
+ {
+ ndbout << "Test 3:" << endl;
+ ndbout << "The fetched rowset associated"
+ << "with Statementhandle is empty. The system"
+ << " reported SQL_ERROR. Check the function!" << endl;
+ GetData_DisplayError(SQL_HANDLE_STMT, GD_hstmt);
+ }
+ else if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Test 3:" << endl;
+ ndbout << "The fetched rowset associated"
+ << "with Statementhandle is empty. The system"
+ << " reported SUCCESS. Check the function!" << endl;
+ GetData_DisplayError(SQL_HANDLE_STMT, GD_hstmt);
+ }
+ else if (retcode == 100)
+ {
+ ndbout << "Test 3:" << endl;
+ ndbout << "The fetched rowset associated"
+ << "with Statementhandle is empty. The system"
+ << " reported SQL_NO_DATA. The function is OK!" << endl;
+ }
+ }
+ else if (retcode == SQL_ERROR)
+ {
+ ndbout << "Test 3 falied!" << endl;
+ GetData_DisplayError(SQL_HANDLE_STMT, GD_hstmt);
+ }
+ else
+ ndbout << " " << endl;
+
+ // }
+ // }
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(GD_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, GD_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, GD_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, GD_henv);
+
+ return NDBT_OK;
+}
+
+void GetData_DisplayError(SQLSMALLINT GD_HandleType, SQLHSTMT GD_InputHandle)
+{
+
+ SQLSMALLINT i, MsgLen;
+ SQLRETURN SQLSTATEs;
+ SQLCHAR Sqlstate[5], Msg[GD_MESSAGE_LENGTH];
+ SQLINTEGER NativeError;
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(GD_HandleType,
+ GD_InputHandle,
+ i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << GD_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)GD_InputHandle << endl;
+ ndbout << "Phone = " << (char *)Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetDescFieldTest.cpp b/ndb/test/odbc/client/SQLGetDescFieldTest.cpp
new file mode 100644
index 00000000000..b789ed75378
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetDescFieldTest.cpp
@@ -0,0 +1,113 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <sqlext.h>
+#include <stdio.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+SQLHSTMT hstmt;
+SQLHDESC hdesc;
+
+SQLSMALLINT RecNumber;
+SQLCHAR szSalesPerson[SALES_PERSON_LEN];
+
+SQLCHAR Sqlstate[5], Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLINTEGER NativeError;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLINTEGER ValuePtr1;
+SQLCHAR ValuePtr2;
+SQLSMALLINT ValuePtr3;
+
+
+
+SQLINTEGER StringLengthPtr;
+
+SQLSMALLINT i, MsgLen;
+
+void SQLGetDescFieldTest_DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle);
+
+int SQLGetDescFieldTest()
+{
+
+ /* If MBR is 'PS' and there is no prepared or execute statement associated with S*/
+retcode = SQLGetDescField(hdesc, 1, SQL_DESC_ARRAY_SIZE, &ValuePtr1, 128, &StringLengthPtr);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLGetDescFieldTest_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+ /* hstmt */
+ // SQLPrepare a statement to select rows from the ORDERS Table. We can create the
+ // table and inside rows in NDB by another program TestDirectSQL. In this test
+ // program(SQLGetDescRecTest),we only have three rows in table ORDERS
+
+/* Prepare the SQL statement with parameter markers. */
+retcode = SQLPrepare(hstmt, (SQLCHAR *)"SELECT ORDERID, CUSTID, OPENDATE, SALESPERSON, STATUS FROM ORDERS)", SQL_NTS);
+
+/* SELECT OrderID, CustID, OpenDate, SalesPerson from Table ORDERS */
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+
+ /* If FI(FieldIdentifer) is not one of the code values in Table 20 */
+retcode = SQLGetDescField(hdesc, 1, 9999, &ValuePtr1, 128, &StringLengthPtr);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLGetDescFieldTest_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+
+ /* RecoderNumber is less than 1 */
+retcode = SQLGetDescField(hdesc, -1, SQL_DESC_ARRAY_SIZE, &ValuePtr1, 128, &StringLengthPtr);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLGetDescFieldTest_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+ /* RecoderNumber is greater than N, N be the value of the COUNT field of D, D be the allocated CLI */
+ /* descriptor area identified by DescriptorHandle */
+retcode = SQLGetDescField(hdesc, 4, SQL_DESC_ARRAY_SIZE, &ValuePtr1, 128, &StringLengthPtr);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLGetDescFieldTest_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+}
+
+ return 0;
+
+ }
+
+
+void SQLGetDescFieldTest_DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetDescRecTest.cpp b/ndb/test/odbc/client/SQLGetDescRecTest.cpp
new file mode 100644
index 00000000000..5944f393a71
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetDescRecTest.cpp
@@ -0,0 +1,95 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+SQLHSTMT hstmt;
+SQLHDESC hdesc;
+
+SQLSMALLINT RecNumber;
+SQLCHAR szSalesPerson[SALES_PERSON_LEN];
+
+SQLCHAR Sqlstate[5], Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLINTEGER NativeError;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR Name;
+SQLINTEGER LengthPtr;
+SQLSMALLINT SGDR_StringLengthPtr, TypePtr, SubTypePtr, PrecisionPtr, ScalePtr, NullablePtr;
+
+SQLSMALLINT i, MsgLen;
+
+void SQLGetDescRec_DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle);
+
+int SQLGetDescRecTest()
+{
+
+ /* hstmt */
+ // SQLPrepare a statement to select rows from the ORDERS Table. We can create the table and inside rows in
+ // NDB by another program TestDirectSQL. In this test program(SQLGetDescRecTest),we only have three rows in
+ // table ORDERS
+
+/* Prepare the SQL statement with parameter markers. */
+retcode = SQLPrepare(hstmt, (SQLCHAR *)"SELECT ORDERID, CUSTID, OPENDATE, SALESPERSON, STATUS FROM ORDERS)", SQL_NTS);
+
+/* SELECT OrderID, CustID, OpenDate, SalesPerson from Table ORDERS */
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ /* RecoderNumber is less than 1 */
+retcode = SQLGetDescRec(hdesc, -1, &Name, sizeof(Name), &SGDR_StringLengthPtr, &TypePtr, &SubTypePtr, &LengthPtr, &PrecisionPtr, &ScalePtr, &NullablePtr);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLGetDescRec_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+ /* RecoderNumber is greater than N, N be the value of the COUNT field of D, D be the allocated CLI */
+ /* descriptor area identified by DescriptorHandle */
+retcode = SQLGetDescRec(hdesc, 4, &Name, sizeof(Name), &SGDR_StringLengthPtr, &TypePtr, &SubTypePtr, &LengthPtr, &PrecisionPtr, &ScalePtr, &NullablePtr);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLGetDescRec_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+}
+
+ return 0;
+
+ }
+
+
+void SQLGetDescRec_DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetDiagFieldTest.cpp b/ndb/test/odbc/client/SQLGetDiagFieldTest.cpp
new file mode 100644
index 00000000000..ef9bc3eb3fc
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetDiagFieldTest.cpp
@@ -0,0 +1,236 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <sqlcli.h>
+#include <stdio.h>
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLINTEGER strangehandle;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+SQLSMALLINT StringLengthPtr;
+SQLINTEGER DiagInfoPtr1;
+SQLCHAR DiagInfoPtr2;
+SQLRETURN DiagInfoPtr3;
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int SQLGetDiagFieldTest ()
+{
+
+ strangehandle = 67;
+ /* hstmt */
+ // Execute a statement to retrieve rows from the Customers table. We can create the table and inside rows in
+ // NDB by another program TestDirectSQL
+ // const SQLCHAR *StatementText = "SELECT CustID, Name, Address, Phone FROM Customers";
+ // retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", 56);
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+while ((SQLSTATEs = SQLGetDiagField(67, 67, i, SQL_DIAG_CURSOR_ROW_COUNT, &DiagInfoPtr1, 128, &StringLengthPtr)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+
+ /* HandleType indicates ENVIRNMENT HANDLE and Handle does not identify an allocated SQL_environment */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+while ((SQLSTATEs = SQLGetDiagField(SQL_HANDLE_ENV, hdbc, i, SQL_DIAG_CURSOR_ROW_COUNT, &DiagInfoPtr1, 128, &StringLengthPtr)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+
+ /* HandleType indicates CONNECTION HANDLE and Handle does not identify an allocated SQL_connection */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+while ((SQLSTATEs = SQLGetDiagField(SQL_HANDLE_DBC, henv, i, SQL_DIAG_CURSOR_ROW_COUNT, &DiagInfoPtr1, 128, &StringLengthPtr)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+ /* HandleType indicates STATEMENT HANDLE and Handle does not identify an allocated SQL_statement */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+while ((SQLSTATEs = SQLGetDiagField(SQL_HANDLE_STMT, hdbc, i, SQL_DIAG_CURSOR_ROW_COUNT, &DiagInfoPtr1, 128, &StringLengthPtr)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+ /* HandleType indicates DESCRIPTOR HANDLE and Handle does not identify an allocated SQL_descriptor */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+while ((SQLSTATEs = SQLGetDiagField(SQL_HANDLE_DESC, hdbc, i, SQL_DIAG_CURSOR_ROW_COUNT, &DiagInfoPtr1, 128, &StringLengthPtr)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+ /* DiagIdentifer is not one of the code values in Table12 */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+
+where ((SQLSTATEs = SQLGetDiagField(SQL_HANDLE_DBC, hdbc, -1, 99, &DiagInfoPtr1, 128, &StringLengthPtr)) != SQL_NO_DATA) {
+
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+ /* If TYPE is 'STATUS' and RN is greater than N */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+
+where ((SQLSTATEs = SQLGetDiagField(SQL_HANDLE_DBC, hdbc, 9999, 8, &DiagInfoPtr2, 128, &StringLengthPtr)) != SQL_NO_DATA) {
+
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+
+ /* If TYPE is 'STATUS' and RN is less than 1 */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+
+ where ((SQLSTATEs = SQLGetDiagField(SQL_HANDLE_DBC, hdbc, -1, 8, &DiagInfoPtr2, 128, &StringLengthPtr)) != SQL_NO_DATA) {
+
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+ /* If DI indicates ROW_COUNT and R is neither Execute nor ExecDirect, then an exception condition is raised */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+
+where ((SQLSTATEs = SQLGetDiagField(SQL_HANDLE_DBC, hdbc, i, SQL_DIAG_ROW_COUNT, &DiagInfoPtr1, 128, &StringLengthPtr)) != SQL_NO_DATA) {
+
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+
+ return 0;
+
+ }
+
+
+
+
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetDiagRecSimpleTest.cpp b/ndb/test/odbc/client/SQLGetDiagRecSimpleTest.cpp
new file mode 100644
index 00000000000..8fa4a2b3dbb
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetDiagRecSimpleTest.cpp
@@ -0,0 +1,167 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLGetDiagRecSimpleTest.cpp
+ */
+#include <common.hpp>
+#include <string.h>
+
+using namespace std;
+
+SQLHDBC GDS_hdbc;
+SQLHSTMT GDS_hstmt;
+SQLHENV GDS_henv;
+SQLHDESC GDS_hdesc;
+SQLRETURN GDS_retcode, GDS_RETURN;
+
+#define GDS_SQL_MAXIMUM_MESSAGE_LENGTH 255
+SQLCHAR GDS_Sqlstate[5];
+
+SQLINTEGER GDS_NativeError;
+SQLSMALLINT GDS_i = 1, GDS_MsgLen;
+SQLCHAR GDS_Msg[GDS_SQL_MAXIMUM_MESSAGE_LENGTH], GDS_ConnectIn[30];
+
+/**
+ * Test SQLGetDiagRec return value
+ *
+ * -#Simply test Msg when return is SQL_NO_DATA
+ * -#Simply test Msg when return is SQL_SUCCESS
+ * -#Simply test Msg when return is SQL_SUCCESS_WITH_INFO
+ * -#Simply test Msg when return is SQL_INVALID_HANDLE
+ * -#Simply test Msg when return is SQL_ERROR
+ *
+ * @return Zero, if test succeeded
+ */
+
+int SQLGetDiagRecSimpleTest()
+{
+ ndbout << endl << "Start SQLGetDiagRec Simple Testing" << endl;
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+
+ GDS_retcode = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &GDS_henv);
+ if (GDS_retcode == SQL_SUCCESS || GDS_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated An Environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+
+ GDS_retcode = SQLSetEnvAttr(GDS_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (GDS_retcode == SQL_SUCCESS || GDS_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ GDS_retcode = SQLAllocHandle(SQL_HANDLE_DBC, GDS_henv, &GDS_hdbc);
+
+ if (GDS_retcode == SQL_SUCCESS || GDS_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated A Connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+
+ GDS_retcode = SQLConnect(GDS_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (GDS_retcode == SQL_SUCCESS || GDS_retcode == SQL_SUCCESS_WITH_INFO){
+ ndbout << "Success connection to DB!" << endl;
+ ndbout << "GDS_retcode = " << GDS_retcode << endl;
+ ndbout << "SQL_SUCCESS = " << SQL_SUCCESS << endl;
+ ndbout << "SQL_SUCCESS_WITH_INFO = " << SQL_SUCCESS_WITH_INFO << endl;}
+
+ ndbout << endl;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ if (GDS_retcode != SQL_SUCCESS || GDS_retcode != SQL_SUCCESS_WITH_INFO){
+ ndbout << "GDS_retcode = " << GDS_retcode << endl;
+ ndbout << "SQL_SUCCESS = " << SQL_SUCCESS << endl;
+ ndbout << "SQL_SUCCESS_WITH_INFO = " << SQL_SUCCESS_WITH_INFO << endl;
+
+ GDS_RETURN = SQLGetDiagRec(SQL_HANDLE_DBC,
+ GDS_hdbc,
+ GDS_i,
+ GDS_Sqlstate,
+ &GDS_NativeError,
+ GDS_Msg,
+ sizeof(GDS_Msg),
+ &GDS_MsgLen);
+
+ if (GDS_RETURN == SQL_NO_DATA){
+ ndbout << "GDS_SQLSTATES = SQL_NO_DATA" << endl;
+ ndbout << "the HandleType is:" << SQL_HANDLE_DBC << endl;
+ ndbout << "the Handle is :" << (long)GDS_hdbc << endl;
+ ndbout << "the GDS_Msg is :" << (char *)GDS_Msg << endl;
+ ndbout << "the sqlstate is:" << (char *)GDS_Sqlstate << endl;}
+
+ else if (GDS_RETURN == SQL_SUCCESS){
+ ndbout << "GDS_SQLSTATES = SQL_SUCCESS" << endl;
+ ndbout << "the HandleType is:" << SQL_HANDLE_DBC << endl;
+ ndbout << "the Handle is :" << (long)GDS_hdbc << endl;
+ ndbout << "the GDS_Msg is :" << (char *)GDS_Msg << endl;
+ ndbout << "the sqlstate is:" << (char *)GDS_Sqlstate << endl;}
+
+ else if (GDS_RETURN == SQL_SUCCESS_WITH_INFO){
+ ndbout << "GDS_SQLSTATES = SQL_SUCCESS_WITH_INFO" << endl;
+ ndbout << "the HandleType is:" << SQL_HANDLE_DBC << endl;
+ ndbout << "the Handle is :" << (long)GDS_hdbc << endl;
+ ndbout << "the GDS_Msg is :" << (char *)GDS_Msg << endl;
+ ndbout << "the sqlstate is:" << (char *)GDS_Sqlstate << endl;}
+
+ else if (GDS_RETURN == SQL_INVALID_HANDLE){
+ ndbout << "GDS_SQLSTATES = SQL_INVALID_HANDLE" << endl;
+ ndbout << "the HandleType is:" << SQL_HANDLE_DBC << endl;
+ ndbout << "the Handle is :" << (long)GDS_hdbc << endl;
+ ndbout << "the GDS_Msg is :" << (char *)GDS_Msg << endl;
+ ndbout << "the sqlstate is:" << (char *)GDS_Sqlstate << endl;}
+
+ else{
+ ndbout << "GDS_RETURN = SQL_ERROR" << endl;
+ ndbout << "the HandleType is:" << SQL_HANDLE_DBC << endl;
+ ndbout << "the Handle is :" << (long)GDS_hdbc << endl;
+ ndbout << "the GDS_Msg is :" << (char *)GDS_Msg << endl;
+ ndbout << "the sqlstate is:" << (char *)GDS_Sqlstate << endl;
+ }
+ }
+ ndbout << "-------------------------------------------------" << endl;
+
+ //******************
+ //** Free Handles **
+ //******************
+ SQLDisconnect(GDS_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, GDS_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, GDS_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, GDS_henv);
+ return NDBT_OK;
+ }
+
diff --git a/ndb/test/odbc/client/SQLGetDiagRecTest.cpp b/ndb/test/odbc/client/SQLGetDiagRecTest.cpp
new file mode 100644
index 00000000000..27c78edaa4d
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetDiagRecTest.cpp
@@ -0,0 +1,207 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <sqlcli.h>
+#include <stdio.h>
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLINTEGER strangehandle;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int SQLGetDiagRecTest()
+{
+
+ strangehandle = 67;
+ /* hstmt */
+ // Execute a statement to retrieve rows from the Customers table. We can create the table and inside rows in
+ // NDB by another program TestDirectSQL
+ // const SQLCHAR *StatementText = "SELECT CustID, Name, Address, Phone FROM Customers";
+
+ // retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", 56);
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(67, 67, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+
+ /* HandleType indicates ENVIRNMENT HANDLE and Handle does not identify an allocated SQL_environment */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(SQL_HANDLE_ENV, hdbc, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+
+ /* HandleType indicates CONNECTION HANDLE and Handle does not identify an allocated SQL_connection */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(SQL_HANDLE_DBC, henv, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+ /* HandleType indicates STATEMENT HANDLE and Handle does not identify an allocated SQL_statement */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(SQL_HANDLE_STMT, hdbc, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+ /* HandleType indicates DESCRIPTOR HANDLE and Handle does not identify an allocated SQL_descriptor */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(SQL_HANDLE_DESC, hdbc, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+
+ /* RecordNumber is less than one */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+
+ where ((SQLSTATEs = SQLGetDiagRec(SQL_HANDLE_DBC, hdbc, -1,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+ /* RecordNumber is greater than N */
+
+ retcode = SQLPrepare(hstmt, (SQLCHAR*)"SELECT CustID, Name, Address, Phone FROM Customers", SQL_NTS);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS) {
+
+ where ((SQLSTATEs = SQLGetDiagRec(SQL_HANDLE_DBC, hdbc, 9999,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ if (SQLSTATEs == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+ }
+ i ++;
+ }
+ }
+
+
+ return 0;
+
+ }
+
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetEnvAttrTest.cpp b/ndb/test/odbc/client/SQLGetEnvAttrTest.cpp
new file mode 100644
index 00000000000..efc8117d6d2
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetEnvAttrTest.cpp
@@ -0,0 +1,110 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLPOINTER ValuePtr;
+SQLINTEGER GetEnvAttr_StringLengthPtr;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void GetEnvAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle);
+
+int SQLGetEnvAttrTest()
+{
+ /* ODBC attributres */
+
+ /*
+ // char PtrValue1[3] = {'SQL_CP_OFF', 'SQL_CP_ONE_DRIVER', 'SQL_CP_ONE_PER_HENV'};
+ // for (i=0; i < 3; i++) {
+ retcode = SQLGetEnvAttr(henv, SQL_ATTR_CONNECTION_POOLING, ValuePtr, 36, &GetEnvAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetEnvAttr_DisplayError(SQL_HANDLE_ENV, henv); // }
+
+
+ // char PtrValue2[2] = {'SQL_CP_STRICT_MATCH', 'SQL_CP_RELAXED_MATCH'};
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetEnvAttr(henv, SQL_ATTR_CP_MATCH, ValuePtr, 36, &GetEnvAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetEnvAttr_DisplayError(SQL_HANDLE_ENV, henv); // }
+
+
+ // char PtrValue3[2] = {'SQL_OV_ODBC3', 'SQL_OV_ODBC2'};
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetEnvAttr(henv, SQL_ATTR_ODBC_VERSION, ValuePtr, 36, &GetEnvAttr_StringLengthPtr );
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetEnvAttr_DisplayError(SQL_HANDLE_ENV, henv); // }
+
+ */
+
+ // char PtrValue4[2] = {'SQL_TRUE', 'SQL_FALSE'};
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetEnvAttr(henv, SQL_ATTR_OUTPUT_NTS, ValuePtr, 36, &GetEnvAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetEnvAttr_DisplayError(SQL_HANDLE_ENV, henv); // }
+
+ return 0;
+
+ }
+
+
+void GetEnvAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetFunctionsTest.cpp b/ndb/test/odbc/client/SQLGetFunctionsTest.cpp
new file mode 100644
index 00000000000..c6feb8ec033
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetFunctionsTest.cpp
@@ -0,0 +1,284 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLGetFunctionsTest.cpp
+ */
+
+
+#include <common.hpp>
+#define GF_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC GF_hdbc;
+SQLHSTMT GF_hstmt;
+SQLHENV GF_henv;
+
+void SQLGetFunctions_DisplayError(SQLSMALLINT GF_HandleType,
+ SQLHDBC GF_InputHandle);
+
+/**
+ * Test whether a specific ODBC API function is supported by
+ * the driver an application is currently connected to.
+ *
+ * In this test program, we can change ODBC function values in order to
+ * know different which fuction is supported by ODBC drivers
+ * Tests:
+ * -# Test1 There is no established SQL-connection
+ * -# Test2 ConnectionHandle does not identify an allocated SQL-connection
+ * -# Test3 The value of FunctionId is not in table 27
+ * -# Test4 Normal case test
+ * @return Zero, if test succeeded
+ */
+
+int SQLGetFunctionsTest()
+{
+ SQLUSMALLINT TableExists, Supported;
+ SQLCHAR SQLStmt [120];
+ SQLRETURN retcode;
+
+ ndbout << endl << "Start SQLGetFunctions Testing" << endl;
+
+ //**********************************************************
+ //** Test 1 **
+ //** If there is no established SQL-connection associated **
+ //** with allocated SQL-connection **
+ //**********************************************************
+
+ retcode = SQLGetFunctions(GF_hdbc, SQL_API_SQLTABLES, &TableExists);
+ if (retcode == -2)
+{
+ ndbout << endl << "Test 1" << endl;
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << endl << "There is no established SQL-connection" << endl;
+ ndbout << "associated with allocated SQL_connection" << endl;
+ SQLGetFunctions_DisplayError(SQL_HANDLE_STMT, GF_hstmt);
+}
+ else if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+{
+ ndbout << endl << "Test 1" << endl;
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << endl << "There is no established SQL-connection" << endl;
+ ndbout << "associated with allocated SQL_connection" << endl;
+ SQLGetFunctions_DisplayError(SQL_HANDLE_STMT, GF_hstmt);
+}
+ else
+{
+ ndbout << endl << "Test 1" << endl;
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << endl << "There is no established SQL-connection" << endl;
+ ndbout << "associated with allocated SQL_connection" << endl;
+ SQLGetFunctions_DisplayError(SQL_HANDLE_STMT, GF_hstmt);
+}
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &GF_henv);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(GF_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.X!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ GF_henv,
+ &GF_hdbc);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(GF_hdbc,
+ (SQLCHAR*) connectString(),
+ SQL_NTS,
+ (SQLCHAR*) "",
+ SQL_NTS,
+ (SQLCHAR*) "",
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+{
+
+ //*************************************************************
+ //** Test 2 **
+ //** If ConnectionHandle does not identify an allocated **
+ //** SQL-connection, then an exception condition is raised **
+ //*************************************************************
+ retcode = SQLGetFunctions(GF_hdbc, SQL_API_SQLTABLES, &TableExists);
+ if (retcode == -2)
+{
+ ndbout << endl << "Test 2" << endl;
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << "If ConnectionHandle does not identify an allocated" << endl;
+ ndbout << "SQL-connection,an exception condition is raised" << endl;
+ SQLGetFunctions_DisplayError(SQL_HANDLE_STMT, GF_hstmt);
+}
+ else if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+{
+ ndbout << endl << "Test 2" << endl;
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << "If ConnectionHandle does not identify an allocated" << endl;
+ ndbout << "SQL-connection,an exception condition is raised" << endl;
+ SQLGetFunctions_DisplayError(SQL_HANDLE_STMT, GF_hstmt);
+}
+ else
+{
+ ndbout << endl << "Test 2 :" << endl;
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << "If ConnectionHandle does not identify an allocated" << endl;
+ ndbout << "SQL-connection,an exception condition is raised" << endl;
+ SQLGetFunctions_DisplayError(SQL_HANDLE_STMT, GF_hstmt);
+}
+
+ //*************************************************************
+ //** Test 3 **
+ //** If the value of FunctionId is not in table 27, "Codes **
+ //** used to identify SQL/CLI routines" **
+ //*************************************************************
+
+ retcode = SQLGetFunctions(GF_hdbc, 88888, &TableExists);
+ ndbout<< "TableExists = " << TableExists << endl;
+ if (retcode == -2)
+{
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << "Test 3 : The value of FunctionId is not in table 27" << endl;
+ SQLGetFunctions_DisplayError(SQL_HANDLE_STMT, GF_hstmt);
+}
+ else if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+{
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << "Test 3 : The value of FunctionId is not in table 27" << endl;
+ SQLGetFunctions_DisplayError(SQL_HANDLE_STMT, GF_hstmt);
+}
+ else
+{
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << "Test 3 : The value of FunctionId is not in table 27" << endl;
+ SQLGetFunctions_DisplayError(SQL_HANDLE_STMT, GF_hstmt);
+}
+
+ //******************
+ //** Test 4 **
+ //** Normal case **
+ //******************
+ ndbout << "Test 4:" << endl;
+ retcode = SQLGetFunctions(GF_hdbc, SQL_API_SQLBROWSECONNECT, &Supported);
+ ndbout << "retcode = " << retcode << endl;
+ if (Supported == TRUE)
+{
+ ndbout << "Supported = " << Supported << endl;
+ ndbout << "SQLBrowseConnect is supported by the current data source"
+ << endl;
+}
+ else
+{
+ ndbout << "Supported = " << Supported << endl;
+ ndbout << "SQLBrowseConnect isn't supported by the current data source"
+ << endl;
+}
+
+
+ //******************
+ //** Test 5 **
+ //** Normal case **
+ //******************
+ ndbout << endl << "Test 5:" << endl;
+ retcode = SQLGetFunctions(GF_hdbc, SQL_API_SQLFETCH, &Supported);
+ ndbout << "retcode = " << retcode << endl;
+ if (Supported == TRUE)
+{
+ ndbout << "Supported = " << Supported << endl;
+ ndbout << "SQLFETCH is supported by the current data source" << endl;
+}
+ else
+{
+ ndbout << "Supported = " << Supported << endl;
+ ndbout << "SQLFETCH isn't supported by the current data source" << endl;
+}
+
+}
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(GF_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, GF_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, GF_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, GF_henv);
+
+ return NDBT_OK;
+
+}
+
+
+void SQLGetFunctions_DisplayError(SQLSMALLINT GF_HandleType,
+ SQLHDBC GF_InputHandle)
+{
+ SQLRETURN SQLSTATEs;
+ SQLCHAR Sqlstate[50];
+ SQLINTEGER NativeError;
+ SQLSMALLINT i, MsgLen;
+ SQLCHAR Msg[GF_MESSAGE_LENGTH];
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ Msg[0] = 0;
+ while ((SQLSTATEs = SQLGetDiagRec(GF_HandleType,
+ GF_InputHandle,
+ i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << GF_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)GF_InputHandle << endl;
+ ndbout << "the Msg is :" << (char *) Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ Msg[0] = 0;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetInfoTest.cpp b/ndb/test/odbc/client/SQLGetInfoTest.cpp
new file mode 100644
index 00000000000..95f7562dafe
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetInfoTest.cpp
@@ -0,0 +1,215 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLGetInfoTest.cpp
+ */
+
+#include <common.hpp>
+
+using namespace std;
+
+SQLHDBC GI_hdbc;
+SQLHSTMT GI_hstmt;
+SQLHENV GI_henv;
+
+#define GI_MESSAGE_LENGTH 200
+
+SQLCHAR Msg[GI_MESSAGE_LENGTH];
+
+void SQLGetInfoTest_DisplayError(SQLSMALLINT GI_HandleType,
+ SQLHDBC GI_InputHandle);
+
+/**
+ * Test to retrieve general information about the driver and
+ * the data source an application is currently connected to.
+ *
+ * Tests:
+ * -# Test The value of FunctionId is not in table 27
+ * @return Zero, if test succeeded
+ */
+
+int SQLGetInfoTest()
+{
+ SQLRETURN retcode;
+ SQLINTEGER InfoValuePtr;
+ SQLSMALLINT SLPStringLengthPtr;
+
+ ndbout << endl << "Start SQLGetInfo Testing" << endl;
+
+ //******************************************************
+ //** The value of FunctionId is not in Table 27, then **
+ //** an exception condition is raised **
+ //******************************************************
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &GI_henv);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(GI_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ GI_henv,
+ &GI_hdbc);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(GI_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+
+ // **********************
+ // ** GET INFO FROM DB **
+ // *********************
+
+ retcode = SQLGetInfo(GI_hdbc,
+ SQL_DATABASE_NAME,
+ &InfoValuePtr,
+ sizeof(InfoValuePtr),
+ &SLPStringLengthPtr);
+
+ if (retcode == SQL_SUCCESS)
+ ndbout << endl << "Database Name:" << InfoValuePtr << endl;
+ else
+ {
+ ndbout << endl << "retcode = SQLGetInfo() = " << retcode <<endl;
+ SQLGetInfoTest_DisplayError(SQL_HANDLE_STMT, GI_hstmt);
+ }
+
+ retcode = SQLGetInfo(GI_hdbc,
+ SQL_DRIVER_NAME,
+ &InfoValuePtr,
+ sizeof(InfoValuePtr),
+ &SLPStringLengthPtr);
+
+ if (retcode == SQL_SUCCESS)
+ ndbout << endl << "Driver Name:" << InfoValuePtr << endl;
+ else
+ {
+ ndbout << endl << "retcode = SQLGetInfo() = " << retcode <<endl;
+ SQLGetInfoTest_DisplayError(SQL_HANDLE_STMT, GI_hstmt);
+ }
+
+ // **************************
+ // ** INPUT WRONG InfoType **
+ // **************************
+ retcode = SQLGetInfo(GI_hdbc,
+ 8888,
+ &InfoValuePtr,
+ sizeof(InfoValuePtr),
+ &SLPStringLengthPtr);
+ if (retcode == -2)
+ {
+ ndbout << endl <<"retcode = " << retcode << endl;
+ ndbout << "System reported -2. Please check your test programme"
+ << " about the connectionhandle." << endl;
+ SQLGetInfoTest_DisplayError(SQL_HANDLE_STMT, GI_hstmt);
+ }
+ else if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << endl << "retcode = " << retcode << endl;
+ ndbout << "The information of InfoType is not in Table 28,"
+ << " but SQLGetInfo() executeed succeddfully."
+ << " Check the function!" <<endl;
+ SQLGetInfoTest_DisplayError(SQL_HANDLE_STMT, GI_hstmt);
+ }
+ else if (retcode == SQL_ERROR)
+ {
+ ndbout << endl << "retcode = " << retcode << endl;
+ ndbout << "Input a wrong InfoType. The system found the"
+ << " information of InfoType is not in Table 28."
+ << " Test successful!" << endl;
+ }
+ else
+ ndbout << endl;
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(GI_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, GI_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, GI_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, GI_henv);
+
+ return NDBT_OK;
+
+ }
+
+
+void SQLGetInfoTest_DisplayError(SQLSMALLINT GI_HandleType,
+ SQLHDBC GI_InputHandle)
+{
+ SQLRETURN SQLSTATEs;
+ SQLINTEGER NativeError;
+ SQLCHAR Sqlstate[50];
+ SQLSMALLINT i, MsgLen;
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+
+ while ((SQLSTATEs = SQLGetDiagRec(GI_HandleType,
+ GI_InputHandle,
+ i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+{
+
+ ndbout << "the GI_HandleType is:" << GI_HandleType << endl;
+ ndbout << "the GI_InputHandle is :" << (long)GI_InputHandle << endl;
+ ndbout << "the Msg is :" << (char *) Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetStmtAttrTest.cpp b/ndb/test/odbc/client/SQLGetStmtAttrTest.cpp
new file mode 100644
index 00000000000..2052af60ee0
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetStmtAttrTest.cpp
@@ -0,0 +1,155 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLPOINTER ValuePtr;
+SQLINTEGER GetStmtAttr_StringLengthPtr;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void GetStmtAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle);
+
+int SQLGetStmtAttrTest()
+{
+ /* SQL/CLI attributes */
+ /* SQL_ATTR_APP_PARAM_DESC */
+ // char PtrValue1[1] = {'SQL_NULL_DESC'};
+ // for (i=0; i < 1; i++) {
+ retcode = SQLGetStmtAttr(hstmt, SQL_ATTR_APP_PARAM_DESC, ValuePtr, 36, &GetStmtAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);//}
+
+ /* SQL_ATTR_APP_ROW_DESC */
+ // char PtrValue2[1] = {'SQL_NULL_DESC'}; /* ? */
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetStmtAttr(hstmt, SQL_ATTR_APP_ROW_DESC, ValuePtr, 36, &GetStmtAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);//}
+
+ /* SQL_ATTR_CURSOR_SCROLLABLE */
+ // char PtrValue3[2] = {'SQL_NONSCROLLABLE', 'SQL_SCROLLABLE'}; /* ? */
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetStmtAttr(hstmt, SQL_ATTR_CURSOR_SCROLLABLE, ValuePtr, 36, &GetStmtAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);//}
+
+ /* SQL_ATTR_CURSOR_SENSITIVITY */
+ // char PtrValue4[3] = {'SQL_UNSPECIFIED', 'SQL_INSENSITIVE', 'SQL_SENSITIVE'}; /* ? */
+ // for (i=0; i < 3; i++) {
+ retcode = SQLGetStmtAttr(hstmt, SQL_ATTR_CURSOR_SENSITIVITY, ValuePtr, 36, &GetStmtAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);//}
+
+ /* SQL_ATTR_METADATA_ID */
+ // char PtrValue5[2] = {'SQL_TRUE', 'SQL_FALSE'}; /* ? */
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetStmtAttr(hstmt, SQL_ATTR_METADATA_ID, ValuePtr, 36, &GetStmtAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);//}
+
+ /* SQL_ATTR_IMP_ROW_DESC */
+ // char PtrValue6[2] = {'TRUE', 'FALSE'}; /* ? */
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetStmtAttr(hstmt, SQL_ATTR_IMP_ROW_DESC, ValuePtr, 36, &GetStmtAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);//}
+
+
+ /* SQL_ATTR_IMP_PARAM_DESC */
+ // char PtrValue6[2] = {'TRUE', 'FALSE'}; /* ? */
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetStmtAttr(hstmt, SQL_ATTR_IMP_PARAM_DESC, ValuePtr, 36, &GetStmtAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);//}
+
+
+ /* SQL_ATTR_METADATA_ID */
+ // char PtrValue6[2] = {'TRUE', 'FALSE'}; /* ? */
+ // for (i=0; i < 2; i++) {
+ retcode = SQLGetStmtAttr(hstmt, SQL_ATTR_METADATA_ID, ValuePtr, 36, &GetStmtAttr_StringLengthPtr);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ GetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);//}
+
+ return 0;
+
+
+ }
+
+
+void GetStmtAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLGetTypeInfoTest.cpp b/ndb/test/odbc/client/SQLGetTypeInfoTest.cpp
new file mode 100644
index 00000000000..5925d1cc1ae
--- /dev/null
+++ b/ndb/test/odbc/client/SQLGetTypeInfoTest.cpp
@@ -0,0 +1,202 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLGetTypeInfoTest.cpp
+ */
+
+#include <common.hpp>
+#define GT_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC GT_hdbc;
+SQLHSTMT GT_hstmt;
+SQLHENV GT_henv;
+
+void SQLGetTypeInfoTest_DisplayError(SQLSMALLINT GT_HandleType,
+ SQLHDBC GT_InputHandle);
+
+/**
+ * Test to retrieve general information about the data types
+ * supported by the data source an application is currently connected to.
+ *
+ * Tests:
+ * -# Test The value of FunctionId is not in table 37
+ * @return Zero, if test succeeded
+ */
+int SQLGetTypeInfoTest()
+{
+ SQLRETURN retcode;
+ SQLSMALLINT ColumnSize;
+ unsigned long TypeName;
+ // SQLCHAR TypeName[128];
+
+ ndbout << endl << "Start SQLGetTypeInfo Testing" << endl;
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &GT_henv);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(GT_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.X!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ GT_henv,
+ &GT_hdbc);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(GT_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT,
+ GT_hdbc,
+ &GT_hstmt);
+
+
+ //***********************************************
+ //** Get DataType From the Current Application **
+ //***********************************************
+ retcode = SQLGetTypeInfo(GT_hstmt, SQL_CHAR);
+ ndbout << "retcode =SQLGetTypeInfo()= " << retcode << endl;
+ if (retcode == SQL_SUCCESS)
+ {
+ retcode =SQLBindCol(GT_hstmt,
+ 2,
+ SQL_C_ULONG,
+ TypeName,
+ sizeof(TypeName),
+ NULL);
+ ndbout << "retcode = SQLBindCol()= " << retcode << endl;
+
+ // retcode =SQLBindCol(GT_hstmt,
+ // 1,
+ // SQL_C_DEFAULT,
+ // ColumnSize,
+ // sizeof(ColumnSize),
+ // NULL);
+
+ retcode = SQLFetch(GT_hstmt);
+ ndbout << "retcode = SQLFETCH()=" << retcode << endl;
+ ndbout << "DataType = " << TypeName << endl;
+
+ }
+
+ //*******************************************************
+ //** If the Value of DataType is not in Table 37, then **
+ //** an exception condition is raised **
+ //*******************************************************
+ retcode = SQLGetTypeInfo(GT_hstmt, 8888888);
+ if (retcode == -2)
+ {
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << "The value of DataType is not in table 37" << endl;
+ SQLGetTypeInfoTest_DisplayError(SQL_HANDLE_STMT, GT_hstmt);
+ }
+ else if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << endl << "The value of DataType is not in Table 37" << endl;
+ SQLGetTypeInfoTest_DisplayError(SQL_HANDLE_STMT, GT_hstmt);
+ }
+ else
+ {
+ ndbout << "retcode = " << retcode << endl;
+ ndbout << endl << "The value of DataType is not in Table 37" << endl;
+ SQLGetTypeInfoTest_DisplayError(SQL_HANDLE_STMT, GT_hstmt);
+ }
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(GT_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, GT_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, GT_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, GT_henv);
+
+ return NDBT_OK;
+
+ }
+
+void SQLGetTypeInfoTest_DisplayError(SQLSMALLINT GT_HandleType,
+ SQLHDBC GT_InputHandle)
+{
+ SQLINTEGER NativeError;
+ SQLSMALLINT i, MsgLen;
+ SQLCHAR Msg[GT_MESSAGE_LENGTH];
+ SQLRETURN SQLSTATEs;
+ SQLCHAR Sqlstate[50];
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(GT_HandleType,
+ GT_InputHandle,
+ i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+{
+
+ ndbout << "the HandleType is:" << GT_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)GT_InputHandle << endl;
+ ndbout << "the Msg is :" << (char *) Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLMoreResultsTest.cpp b/ndb/test/odbc/client/SQLMoreResultsTest.cpp
new file mode 100644
index 00000000000..cba8b0dc53e
--- /dev/null
+++ b/ndb/test/odbc/client/SQLMoreResultsTest.cpp
@@ -0,0 +1,91 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void SQLMoreResults_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int SQLMoreResultsTest()
+{
+
+ /* hstmt */
+ retcode = SQLMoreResults(hstmt);
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLMoreResults_DisplayError(SQL_HANDLE_STMT, hstmt);
+
+ /* henv */
+ retcode = SQLMoreResults(henv);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_SUCCESS_WITH_INFO still appeared. Please check programm" << endl;
+ // SQLMoreResults_DisplayError(SQL_HANDLE_ENV, henv);
+
+ /* hdbc */
+ retcode = SQLMoreResults(hdbc);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_SUCCESS_WITH_INFO still appeared. Please check programm" << endl;
+ // SQLMoreResults_DisplayError(SQL_HANDLE_DBC, hdbc);
+
+ /* hdesc */
+ retcode = SQLMoreResults(hdesc);
+
+ if (retcode == SQL_SUCCESS_WITH_INFO || retcode == SQL_SUCCESS)
+ ndbout << "Handle Type is SQL_HANDLE_DESC, but string SQL_SUCCESS_WITH_INFO still appeared. Please check programm" << endl;
+ // SQLMoreResults_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+ return 0;
+
+ }
+
+
+void SQLMoreResults_DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLNumResultColsTest.cpp b/ndb/test/odbc/client/SQLNumResultColsTest.cpp
new file mode 100644
index 00000000000..8f0c1dba94c
--- /dev/null
+++ b/ndb/test/odbc/client/SQLNumResultColsTest.cpp
@@ -0,0 +1,202 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLNumResultColsTest.cpp
+ */
+#include <common.hpp>
+using namespace std;
+
+#define NRC_MESSAGE_LENGTH 200
+
+SQLHSTMT NRC_hstmt;
+SQLHSTMT NRC_hdbc;
+SQLHENV NRC_henv;
+SQLHDESC NRC_hdesc;
+
+void SQLNumResultColsTest_DisplayError(SQLSMALLINT NRC_HandleType,
+ SQLHSTMT NRC_InputHandle);
+
+/**
+ * Test returning descriptor information
+ *
+ * Tests:
+ * -# Testing how many columns exist in the result data set
+ *
+ * @return Zero, if test succeeded
+ */
+
+int SQLNumResultColsTest()
+{
+ SQLRETURN retcode;
+ SQLSMALLINT NumColumns;
+ SQLCHAR SQLStmt[NRC_MESSAGE_LENGTH];
+
+ ndbout << endl << "Start SQLNumResultCols Testing" << endl << endl;
+
+ //**************************************************************
+ //** If there is no prepared or executed statement associated **
+ //** with SQL-statement **
+ //**************************************************************
+
+ retcode = SQLNumResultCols(NRC_hstmt, &NumColumns);
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+
+ SQLNumResultColsTest_DisplayError(SQL_HANDLE_STMT, NRC_hstmt);
+ }
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &NRC_henv);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(NRC_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ NRC_henv,
+ &NRC_hdbc);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(NRC_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT,
+ NRC_hdbc,
+ &NRC_hstmt);
+ if(retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ strcpy((char *) SQLStmt, "SELECT * FROM Customers");
+
+ // strcpy((char *) SQLStmt,
+ // "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES (7, 'pet', 'LM vag 8', '88888')");
+
+ //*******************************************
+ //** Prepare and Execute the SQL statement **
+ //*******************************************
+
+ retcode = SQLExecDirect(NRC_hstmt,
+ SQLStmt,
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ //*****************************************************
+ //** Only general error test. It is not in test rule **
+ //*****************************************************
+
+ retcode = SQLNumResultCols(NRC_hstmt, &NumColumns);
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << endl << "Number of columns in the result data set" << endl;
+ ndbout << NumColumns << endl;
+ }
+ else
+ SQLNumResultColsTest_DisplayError(SQL_HANDLE_STMT, NRC_hstmt);
+ }
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(NRC_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, NRC_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, NRC_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, NRC_henv);
+
+ return NDBT_OK;
+
+}
+
+void SQLNumResultColsTest_DisplayError(SQLSMALLINT NRC_HandleType,
+ SQLHSTMT NRC_InputHandle)
+{
+ SQLRETURN SQLSTATEs;
+ SQLINTEGER NativeError;
+ SQLSMALLINT i, MsgLen;
+ SQLCHAR Msg[NRC_MESSAGE_LENGTH],Sqlstate[5];
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(NRC_HandleType,
+ NRC_InputHandle,
+ i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << NRC_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)NRC_InputHandle << endl;
+ ndbout << "the Msg is: " << (char *)Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLParamDataTest.cpp b/ndb/test/odbc/client/SQLParamDataTest.cpp
new file mode 100644
index 00000000000..92d491dfaf5
--- /dev/null
+++ b/ndb/test/odbc/client/SQLParamDataTest.cpp
@@ -0,0 +1,105 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <sqlcli.h>
+#include <stdio.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+
+#define NAME_LEN 50
+#define PHONE_LEN 50
+
+SQLCHAR szName[NAME_LEN], szPhone[PHONE_LEN];
+SQLINTEGER sCustID, cbName, cbAge, cbBirthday;
+
+SQLHSTMT hstmt;
+SQLHENV henv;
+
+SQLCHAR szSalesPerson[SALES_PERSON_LEN];
+
+SQLCHAR szStatus[STATUS_LEN], Sqlstate[5], Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLINTEGER cbOrderID = 0, cbCustID = 0, cbOpenDate = 0, cbSalesPerson = SQL_NTS, cbStatus = SQL_NTS, NativeError;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLSMALLINT sOrderID;
+
+SQLSMALLINT i, MsgLen;
+
+SQLCHAR ColumnName;
+SQLSMALLINT TargetValuePtr;
+SQLINTEGER StrLen_or_IndPtr;
+SQLPOINTER ValuePtrPtr;
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int SQLParamDataTest()
+{
+
+ /* hstmt */
+ // We can create the table ORDERS and insert rows into ORDERS
+ // NDB by program TestDirectSQL. In this test program, We only have three rows in table ORDERS
+
+/* Prepare the SQL statement with parameter markers. */
+retcode = SQLPrepare(hstmt, (SQLCHAR *)"SELECT ORDERID, CUSTID, OPENDATE, SALESPERSON, STATUS FROM ORDERS", SQL_NTS);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+ retcode = SQLExecute(hstmt);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+ retcode = SQLBindParameter(hstmt, 1, SQL_PARAM_INPUT, SQL_C_SSHORT, SQL_INTEGER, 16, 0, &sOrderID, 16, &cbOrderID);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+while (retcode == SQL_NEED_DATA) {
+ retcode = SQLParamData(hstmt, &ValuePtrPtr);
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO){
+ DisplayError(SQL_HANDLE_STMT, hstmt);
+ }
+ }
+
+
+ }
+}
+}
+ return 0;
+
+ }
+
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLPrepareTest.cpp b/ndb/test/odbc/client/SQLPrepareTest.cpp
new file mode 100644
index 00000000000..2ebbc224b85
--- /dev/null
+++ b/ndb/test/odbc/client/SQLPrepareTest.cpp
@@ -0,0 +1,285 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLprepareTest.cpp
+ */
+#include <common.hpp>
+#define pare_SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC pare_hdbc;
+SQLHSTMT pare_hstmt;
+SQLHENV pare_henv;
+SQLHDESC pare_hdesc;
+SQLRETURN pare_retcode, pare_SQLSTATEs;
+
+SQLCHAR pare_Sqlstate[5];
+
+SQLINTEGER pare_NativeError;
+SQLSMALLINT pare_i, pare_MsgLen;
+SQLCHAR pare_Msg[pare_SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void Prepare_DisplayError(SQLSMALLINT pare_HandleType,
+ SQLHSTMT pare_InputHandle);
+
+/**
+ * Test to prepare a statement with different handles
+ *
+ * -# Input correct hstmt handle
+ * -# Input incorrect henv handle
+ * -# Input incorrect hdbc handle
+ * -# Input incorrect handle hdesc
+ *
+ * @return Zero, if test succeeded
+ */
+int SQLPrepareTest()
+{
+ SQLCHAR SQLStmt [120];
+ ndbout << endl << "Start SQLPrepare Testing" << endl;
+ ndbout << endl << "Test 1" << endl;
+ //*********************************
+ //** Test1 **
+ //** Input correct hstmt handle **
+ //*********************************
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ pare_retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &pare_henv);
+
+ if(pare_retcode == SQL_SUCCESS || pare_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ pare_retcode = SQLSetEnvAttr(pare_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (pare_retcode == SQL_SUCCESS || pare_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+ pare_retcode = SQLAllocHandle(SQL_HANDLE_DBC, pare_henv, &pare_hdbc);
+ if (pare_retcode == SQL_SUCCESS || pare_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ pare_retcode = SQLConnect(pare_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (pare_retcode == SQL_SUCCESS || pare_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ ndbout << "Failure to Connect DB!" << endl;
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+ pare_retcode = SQLAllocHandle(SQL_HANDLE_STMT, pare_hdbc, &pare_hstmt);
+ if (pare_retcode == SQL_SUCCESS || pare_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ strcpy( (char *) SQLStmt, "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES(2, 'Hans Peter', 'LM Vag8', '468719000')");
+
+ pare_retcode = SQLPrepare(pare_hstmt,
+ SQLStmt,
+ SQL_NTS);
+
+ if (pare_retcode == SQL_INVALID_HANDLE)
+ {
+ ndbout << "pare_retcode = " << pare_retcode << endl;
+ ndbout << "HandleType is SQL_HANDLE_STMT, but SQL_INVALID_HANDLE"
+ << endl;
+ ndbout << "appeared. Please check program!" << endl;
+ }
+ else if (pare_retcode == SQL_ERROR || pare_retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ Prepare_DisplayError(SQL_HANDLE_STMT, pare_hstmt);
+ }
+ else
+ {
+ //***********************
+ //** Execute statement **
+ //***********************
+ pare_retcode = SQLExecute(pare_hstmt);
+ if (pare_retcode != SQL_SUCCESS)
+ {
+ ndbout << "pare_retcode = " << pare_retcode << endl;
+ Prepare_DisplayError(SQL_HANDLE_STMT, pare_hstmt);
+ }
+ else
+ ndbout << endl << "Test 1:Input correct HSTMT handle. OK!" << endl;
+ }
+
+ //*********************************
+ //** Test2 **
+ //** Input incorrect henv handle **
+ //*********************************
+
+ strcpy( (char *) SQLStmt, "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES(3, 'Hans', 'LM8', '51888')");
+
+ pare_retcode = SQLPrepare(pare_henv,
+ SQLStmt,
+ SQL_NTS);
+
+ ndbout << endl << "Test 2" << endl;
+ if (pare_retcode == SQL_SUCCESS_WITH_INFO || pare_retcode == SQL_SUCCESS)
+ {
+ FAILURE("Wrong SQL_HANDLE_HENV, but success returned. Check it!");
+ }
+ else if (pare_retcode == SQL_INVALID_HANDLE)
+ {
+ ndbout << "Wrong SQL_HANDLE_HENV input and -2 appeared. OK!" << endl ;
+ }
+ else
+ ;
+ /*
+ {
+ ndbout << "Input wrong SQL_HANDLE_ENV, but SQL_SUCCESS_W_I" << endl;
+ ndbout << "and SQL_SUCCESS appeared. Please check program!" << endl;
+ return NDBT_FAILED;
+ }
+ */
+
+ //*********************************
+ //** Test3 **
+ //** Input incorrect hdbc handle **
+ //*********************************
+
+ strcpy( (char *) SQLStmt, "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES(4, 'HP', 'VÄG8', '90888')");
+
+ pare_retcode = SQLPrepare(pare_hdbc,
+ SQLStmt,
+ SQL_NTS);
+
+ ndbout << endl << "Test 3" << endl;
+ if (pare_retcode == SQL_SUCCESS_WITH_INFO || pare_retcode == SQL_SUCCESS)
+ {
+ FAILURE("Wrong SQL_HANDLE_HDBC, but success returned. Check it!");
+ }
+ else if (pare_retcode == SQL_INVALID_HANDLE)
+ {
+ ndbout << "Wrong SQL_HANDLE_HDBC input and -2 appeared. OK!" << endl ;
+ }
+ else
+ ;
+
+ /*
+ {
+ ndbout << "Input wrong statement handle SQL_HANDLE_DBC" << endl;
+ ndbout << "but SQL_SUCCESS_WITH_INFO" << endl;
+ ndbout << "and SQL_SUCCESS still appeared. Please check program" << endl;
+ return NDBT_FAILED;
+ }
+
+ */
+ //**********************************
+ //** Test4 **
+ //** Input incorrect handle hdesc **
+ //**********************************
+
+ strcpy( (char *) SQLStmt, "INSERT INTO Customers (CustID, Name, Address, Phone) VALUES(5, 'Richard', 'VÄG8', '56888')");
+
+ pare_retcode = SQLPrepare(pare_hdesc,
+ SQLStmt,
+ SQL_NTS);
+
+ ndbout << endl << "Test 4" << endl;
+ if (pare_retcode == SQL_SUCCESS_WITH_INFO || pare_retcode == SQL_SUCCESS)
+ {
+ FAILURE("Wrong SQL_HANDLE_DESC, but success returned");
+ }
+ else if (pare_retcode == SQL_INVALID_HANDLE)
+ {
+ ndbout << "Wrong SQL_HANDLE_DESC input and -2 appeared. OK!" << endl ;
+ }
+ else
+ ndbout << endl;
+
+ /*
+ {
+ ndbout << "TEST FAILURE: Input wrong SQL_HANDLE_DESC, "
+ << "but SQL_SUCCESS_WITH_INFO or SQL_SUCCESS was returned."
+ << endl;
+ return NDBT_FAILED;
+ }
+ */
+
+ //****************
+ // Free Handles **
+ //****************
+ SQLDisconnect(pare_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, pare_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, pare_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, pare_henv);
+
+ return NDBT_OK;
+
+}
+
+void Prepare_DisplayError(SQLSMALLINT pare_HandleType,
+ SQLHSTMT pare_InputHandle)
+{
+ SQLSMALLINT pare_i = 1;
+ SQLRETURN pare_SQLSTATEs;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((pare_SQLSTATEs = SQLGetDiagRec(pare_HandleType,
+ pare_InputHandle,
+ pare_i,
+ pare_Sqlstate,
+ &pare_NativeError,
+ pare_Msg,
+ sizeof(pare_Msg),
+ &pare_MsgLen)
+ ) != SQL_NO_DATA)
+ {
+ ndbout << "SQLSTATE = " << pare_SQLSTATEs << endl;
+ ndbout << "the HandleType is:" << pare_HandleType << endl;
+ ndbout << "the Handle is :" << (long)pare_InputHandle << endl;
+ ndbout << "the conn_Msg is: " << (char *) pare_Msg << endl;
+ ndbout << "the output state is:" << (char *)pare_Sqlstate << endl;
+
+ pare_i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLPutDataTest.cpp b/ndb/test/odbc/client/SQLPutDataTest.cpp
new file mode 100644
index 00000000000..38a8458fec4
--- /dev/null
+++ b/ndb/test/odbc/client/SQLPutDataTest.cpp
@@ -0,0 +1,108 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <sqlcli.h>
+#include <stdio.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+
+#define NAME_LEN 50
+#define PHONE_LEN 50
+
+SQLCHAR szName[NAME_LEN], szPhone[PHONE_LEN];
+SQLINTEGER sCustID, cbName, cbAge, cbBirthday;
+
+SQLHSTMT hstmt;
+SQLHENV henv;
+
+SQLCHAR szSalesPerson[SALES_PERSON_LEN];
+
+SQLCHAR szStatus[STATUS_LEN], Sqlstate[5], Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLINTEGER cbOrderID = 0, cbCustID = 0, cbOpenDate = 0, cbSalesPerson = SQL_NTS, cbStatus = SQL_NTS, NativeError;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLSMALLINT sOrderID;
+
+SQLSMALLINT i, MsgLen;
+
+SQLCHAR ColumnName;
+SQLSMALLINT TargetValuePtr;
+SQLINTEGER StrLen_or_Ind, DataPtr;
+SQLPOINTER ValuePtrPtr;
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle);
+
+int SQLPutDataTest()
+{
+
+ /* hstmt */
+ // We can create the table ORDERS and insert rows into ORDERS
+ // NDB by program TestDirectSQL. In this test program, We only have three rows in table ORDERS
+
+/* Prepare the SQL statement with parameter markers. */
+retcode = SQLPrepare(hstmt, (SQLCHAR *)"SELECT ORDERID, CUSTID, OPENDATE, SALESPERSON, STATUS FROM ORDERS", SQL_NTS);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+ retcode = SQLExecute(hstmt);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+ retcode = SQLBindParameter(hstmt, 1, SQL_PARAM_INPUT, SQL_C_SSHORT, SQL_INTEGER, 16, 0, &sOrderID, 16, &cbOrderID);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+while (retcode == SQL_NEED_DATA) {
+ retcode = SQLParamData(hstmt, &ValuePtrPtr);
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO){
+ retcode = SQLPutData(hstmt, &DataPtr, StrLen_or_Ind);
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO){
+ DisplayError(SQL_HANDLE_STMT, hstmt);
+ }
+ }
+ }
+
+
+ }
+}
+}
+ return 0;
+
+ }
+
+
+void DisplayError(SQLSMALLINT HandleType, SQLHSTMT InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLRowCountTest.cpp b/ndb/test/odbc/client/SQLRowCountTest.cpp
new file mode 100644
index 00000000000..f298017c519
--- /dev/null
+++ b/ndb/test/odbc/client/SQLRowCountTest.cpp
@@ -0,0 +1,203 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLRowCountTest.cpp
+ */
+
+#include <common.hpp>
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+#define RC_MESSAGE_LENGTH 200
+
+SQLHSTMT RC_hstmt;
+SQLHDBC RC_hdbc;
+SQLHENV RC_henv;
+SQLHDESC RC_hdesc;
+
+void SQLRowCountTest_DisplayError(SQLSMALLINT RC_HandleType,
+ SQLHSTMT RC_InputHandle);
+
+/**
+ * Test to obtain a count of the number of rows
+ * in a table
+ *
+ * -# Call SQLRowCount without executed statement
+ * -# Call SQLRowCount with normal case
+ *
+ * @return Zero, if test succeeded
+ */
+
+int SQLRowCountTest()
+{
+ SQLRETURN retcode;
+ unsigned long RowCount;
+ SQLCHAR SQLStmt [120];
+
+ ndbout << endl << "Start SQLRowCount Testing" << endl;
+
+ //************************************************************************
+ //* If there is no executed statement, an execption condotion is raised **
+ //************************************************************************
+
+ retcode = SQLRowCount(RC_hstmt, &RowCount);
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ {
+
+ SQLRowCountTest_DisplayError(SQL_HANDLE_STMT, RC_hstmt);
+ }
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &RC_henv);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(RC_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ RC_henv,
+ &RC_hdbc);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(RC_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT,
+ RC_hdbc,
+ &RC_hstmt);
+ if(retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+ strcpy((char *) SQLStmt, "INSERT INTO Customers (CustID, Name, Address,Phone) VALUES(588, 'HeYong','LM888','919888')");
+
+ //*******************************
+ //* Prepare the SQL statement **
+ //*******************************
+
+ retcode = SQLPrepare(RC_hstmt,
+ SQLStmt,
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ //******************************
+ //* Execute the SQL statement **
+ //******************************
+ retcode = SQLExecute(RC_hstmt);
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ //***************
+ // Normal test **
+ //***************
+ retcode = SQLRowCount(RC_hstmt, &RowCount);
+ if (retcode == SQL_ERROR )
+ SQLRowCountTest_DisplayError(SQL_HANDLE_STMT, RC_hstmt);
+ else
+ ndbout << endl << "Number of the rows in the table Customers: "
+ << (int)RowCount << endl;
+ }
+ }
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(RC_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, RC_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, RC_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, RC_henv);
+
+ return NDBT_OK;
+
+}
+
+void SQLRowCountTest_DisplayError(SQLSMALLINT RC_HandleType,
+ SQLHSTMT RC_InputHandle)
+{
+ SQLRETURN SQLSTATEs;
+ SQLSMALLINT i, MsgLen;
+ SQLCHAR Sqlstate[5], Msg[RC_MESSAGE_LENGTH];
+ SQLINTEGER NativeError;
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(RC_HandleType,
+ RC_InputHandle,
+ i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+{
+
+ ndbout << "the HandleType is:" << RC_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)RC_InputHandle << endl;
+ ndbout << "the Msg:" << (char *)Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+}
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLSetConnectAttrTest.cpp b/ndb/test/odbc/client/SQLSetConnectAttrTest.cpp
new file mode 100644
index 00000000000..c41ef885521
--- /dev/null
+++ b/ndb/test/odbc/client/SQLSetConnectAttrTest.cpp
@@ -0,0 +1,131 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLPOINTER ValuePtr;
+//SQLINTEGER StringLength;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void SetConnectAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle);
+
+int SQLSetConnectAttrTest ()
+{
+ /* SQL/CLI attributes */
+ char PtrValue1[2] = {'SQL_TRUE', 'SQL_FALSE'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetConnectAttr(hdbc, SQL_ATTR_AUTO_IPD, (void*)PtrValue1[i], sizeof(PtrValue1[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+ /* ODBC attributes */
+ /*
+ char PtrValue1[3] = {'SQL_MODE_READ_ONLY', 'SQL_MODE_READ_WRITE'};
+ for (i=0; i < 3; i++) {
+ retcode = SQLSetConnectAttr(hdbc, SQL_ATTR_ACCESS_MODE, (void*)PtrValue1[i], sizeof(PtrValue1[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+
+ char PtrValue2[2] = {'SQL_ASYNC_ENABLE_OFF', 'SQL_ASYNC_ENABLE_ON'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetConnectAttr(hdbc, SQL_ATTR_ASYNC_ENABLE, (void*)PtrValue2[i], sizeof(PtrValue2[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+
+ char PtrValue4[2] = {'SQL_AUTOCOMMIT_OFF', 'SQL_AUTOCOMMIT_ON'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetConnectAttr(hdbc, SQL_ATTR_AUTOCOMMIT, (void*)PtrValue4[i], sizeof(PtrValue4[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+ char PtrValue5[2] = {'SQL_CD_TRUE', 'SQL_CD_FALSE'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetConnectAttr(hdbc, SQL_ATTR_CONNECTION_DEAD, (void*)PtrValue4[i], sizeof(PtrValue5[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+
+ char PtrValue5[2] = {'SQL_CD_TRUE', 'SQL_CD_FALSE'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetConnectAttr(hdbc, SQL_ATTR_CONNECTION_TIMEOUT, (void*)PtrValue4[i], sizeof(PtrValue5[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_DBC, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetConnectAttr_DisplayError(SQL_HANDLE_DBC, hdbc);}
+
+ */
+
+ return 0;
+
+ }
+
+
+void SetConnectAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLSetCursorNameTest.cpp b/ndb/test/odbc/client/SQLSetCursorNameTest.cpp
new file mode 100644
index 00000000000..b35cf9fefc2
--- /dev/null
+++ b/ndb/test/odbc/client/SQLSetCursorNameTest.cpp
@@ -0,0 +1,215 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLSetCursorNameTest.cpp
+ */
+#include <common.hpp>
+using namespace std;
+
+#define SCN_MESSAGE_LENGTH 50
+
+SQLHSTMT SCN_hstmt;
+SQLHDESC SCN_hdesc;
+SQLHENV SCN_henv;
+SQLHDBC SCN_hdbc;
+
+void SCN_DisplayError(SQLSMALLINT SCN_HandleType,
+ SQLHDESC SCN_InputHandle);
+
+/**
+ * Test to assign a user-defined name to a cursor that is
+ * associated with an active SQL statement handle
+ *
+ * Tests:
+ * -# set user-defined cursor name to zero
+ * -# set user-defined cursor name in normal case
+ *
+ * @return Zero, if test succeeded
+ */
+
+int SQLSetCursorNameTest()
+{
+ SQLRETURN retcode;
+ SQLCHAR SQLStmt [120];
+ SQLCHAR CursorName [80];
+ SQLSMALLINT CNameSize;
+
+ ndbout << endl << "Start SQLSetCursorName Testing" << endl;
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &SCN_henv);
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ retcode = SQLSetEnvAttr(SCN_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ SCN_henv,
+ &SCN_hdbc);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ retcode = SQLConnect(SCN_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ retcode = SQLAllocHandle(SQL_HANDLE_STMT,
+ SCN_hdbc,
+ &SCN_hstmt);
+
+ if(retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //************************
+ //** Define a statement **
+ //************************
+
+ strcpy((char *) SQLStmt,
+ "SELECT * FROM Customers WHERE Address = 'LM Vag 8'");
+
+ //*************************
+ //** Prepare a statement **
+ //*************************
+
+ retcode = SQLPrepare(SCN_hstmt,
+ SQLStmt,
+ SQL_NTS);
+
+ //**********************************
+ //** Set the cursor name with zero**
+ //**********************************
+ retcode = SQLSetCursorName(SCN_hstmt,
+ (char *)"",
+ SQL_NTS);
+
+ if (retcode != SQL_SUCCESS)
+ {
+ ndbout << endl << "retcode =" << retcode << endl;
+ SCN_DisplayError(SQL_HANDLE_STMT, SCN_hstmt);
+ }
+
+ //*************************
+ //** Set the cursor name **
+ //*************************
+ retcode = SQLSetCursorName(SCN_hstmt,
+ (char *)"Customer_CURSOR",
+ SQL_NTS);
+
+ if (retcode != SQL_SUCCESS)
+ {
+ ndbout << endl << "retcode =" << retcode << endl;
+ SCN_DisplayError(SQL_HANDLE_STMT, SCN_hstmt);
+ }
+ //***************************
+ //** Execute the statement **
+ //***************************
+ retcode = SQLExecute(SCN_hstmt);
+
+ //**********************************************
+ //** retrieve and display the new cursor name **
+ //**********************************************
+ retcode = SQLGetCursorName(SCN_hstmt,
+ CursorName,
+ sizeof(CursorName),
+ &CNameSize);
+
+ ndbout << endl << "The cursor name is : " << (char *) CursorName << endl;
+
+ //****************
+ // Free Handles **
+ //****************
+ SQLDisconnect(SCN_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, SCN_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, SCN_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, SCN_henv);
+
+ return NDBT_OK;
+
+ }
+
+
+void SCN_DisplayError(SQLSMALLINT SCN_HandleType, SQLHDESC SCN_InputHandle)
+{
+
+ SQLINTEGER NativeError;
+ SQLCHAR Sqlstate[5], Msg[SCN_MESSAGE_LENGTH];
+ SQLRETURN SQLSTATEs;
+ SQLSMALLINT i, MsgLen;
+ i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(SCN_HandleType,
+ SCN_InputHandle, i,
+ Sqlstate,
+ &NativeError,
+ Msg,
+ sizeof(Msg),
+ &MsgLen))
+ != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << SCN_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)SCN_InputHandle << endl;
+ ndbout << "the Msg is: " << (char *) Msg << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLSetDescFieldTest.cpp b/ndb/test/odbc/client/SQLSetDescFieldTest.cpp
new file mode 100644
index 00000000000..798622e0f75
--- /dev/null
+++ b/ndb/test/odbc/client/SQLSetDescFieldTest.cpp
@@ -0,0 +1,100 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+SQLHSTMT hstmt;
+SQLHDESC hdesc;
+
+SQLSMALLINT RecNumber;
+SQLCHAR szSalesPerson[SALES_PERSON_LEN];
+
+SQLCHAR Sqlstate[5], Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLINTEGER NativeError;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLINTEGER ValuePtr1;
+SQLCHAR ValuePtr2;
+SQLSMALLINT ValuePtr3;
+
+SQLSMALLINT i, MsgLen;
+
+void SQLSetDescFieldTest_DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle);
+
+int SQLSetDescFieldTest()
+{
+
+ /* hstmt */
+ // SQLPrepare a statement to select rows from the ORDERS Table. We can create the table and inside rows in
+ // NDB by another program TestDirectSQL. In this test program(SQLGetDescRecTest),we only have three rows in
+ // table ORDERS
+
+/* Prepare the SQL statement with parameter markers. */
+retcode = SQLPrepare(hstmt, (SQLCHAR *)"SELECT ORDERID, CUSTID, OPENDATE, SALESPERSON, STATUS FROM ORDERS)", SQL_NTS);
+
+/* SELECT OrderID, CustID, OpenDate, SalesPerson from Table ORDERS */
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+ retcode = SQLExecute(hstmt);
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ /* If FI(FieldIdentifer) is not one of the code values in Table 20 */
+retcode = SQLSetDescField(hdesc, 1, 9999, &ValuePtr1, 128);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLSetDescFieldTest_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+
+ /* RecoderNumber is less than 1 and the value of the Type column in the Table is ITEM */
+retcode = SQLSetDescField(hdesc, -1, SQL_DESC_LENGTH, &ValuePtr1, 128);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLSetDescFieldTest_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+
+}
+
+}
+
+ return 0;
+
+ }
+
+
+void SQLSetDescFieldTest_DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLSetDescRecTest.cpp b/ndb/test/odbc/client/SQLSetDescRecTest.cpp
new file mode 100644
index 00000000000..d97af576cb0
--- /dev/null
+++ b/ndb/test/odbc/client/SQLSetDescRecTest.cpp
@@ -0,0 +1,99 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <sqlext.h>
+#include <stdio.h>
+
+using namespace std;
+
+#define NAME_LEN 50
+#define PHONE_LEN 10
+#define SALES_PERSON_LEN 10
+#define STATUS_LEN 6
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+SQLHSTMT hstmt;
+SQLHDESC hdesc;
+
+SQLSMALLINT RecNumber;
+SQLCHAR szSalesPerson[SALES_PERSON_LEN];
+
+SQLCHAR Sqlstate[5], Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+SQLINTEGER NativeError;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLCHAR Name;
+SQLINTEGER LengthPtr;
+
+SQLSMALLINT i, MsgLen;
+
+SQLINTEGER StringLengthPtr, IndicatorPtr;
+SQLPOINTER DataPtr;
+
+void SQLSetDescRecTest_DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle);
+
+int SQLSetDescRecTest()
+{
+
+ /* hstmt */
+ // SQLPrepare a statement to select rows from the ORDERS Table. We can create the table and inside rows in
+ // NDB by program TestDirectSQL. In this test program(SQLSetDescRecTest),we only have three rows in
+ // table ORDERS
+
+/* Prepare the SQL statement with parameter markers. */
+retcode = SQLPrepare(hstmt, (SQLCHAR *)"SELECT ORDERID, CUSTID, OPENDATE, SALESPERSON, STATUS FROM ORDERS)", SQL_NTS);
+
+/* SELECT OrderID, CustID, OpenDate, SalesPerson from Table ORDERS */
+
+if (retcode == SQL_SUCCESS || retcode == SQL_SUCCESS_WITH_INFO) {
+
+ /* RecoderNumber is less than 1 */
+retcode = SQLSetDescRec(hdesc, -1, 1002, 1007, 1013, 1005, 1006, (void *)DataPtr, &StringLengthPtr, &IndicatorPtr);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLSetDescRecTest_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+ /* RecoderNumber is greater than N, N be the value of the COUNT field of D, D be the allocated CLI */
+ /* descriptor area identified by DescriptorHandle */
+retcode = SQLSetDescRec(hdesc, 4, 1002, 1007, 1013, 1005, 1006, (void *)DataPtr, &StringLengthPtr, &IndicatorPtr);
+if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SQLSetDescRecTest_DisplayError(SQL_HANDLE_DESC, hdesc);
+
+}
+
+ return 0;
+
+ }
+
+
+void SQLSetDescRecTest_DisplayError(SQLSMALLINT HandleType, SQLHDESC InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLSetEnvAttrTest.cpp b/ndb/test/odbc/client/SQLSetEnvAttrTest.cpp
new file mode 100644
index 00000000000..16ae5671ca3
--- /dev/null
+++ b/ndb/test/odbc/client/SQLSetEnvAttrTest.cpp
@@ -0,0 +1,108 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLPOINTER ValuePtr;
+SQLINTEGER SetEnvAttr_StringLength;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void SetEnvAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle);
+
+int SQLSetEnvAttrTest()
+{
+ /* ODBC attributes */
+ /*
+ char PtrValue1[3] = {'SQL_CP_OFF', 'SQL_CP_ONE_DRIVER', 'SQL_CP_ONE_PER_HENV'};
+ for (i=0; i < 3; i++) {
+ retcode = SQLSetEnvAttr(henv, SQL_ATTR_CONNECTION_POOLING, (void*)PtrValue1[i], sizeof(PtrValue1[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetEnvAttr_DisplayError(SQL_HANDLE_ENV, henv);}
+
+
+ char PtrValue2[2] = {'SQL_CP_STRICT_MATCH', 'SQL_CP_RELAXED_MATCH'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetEnvAttr(henv, SQL_ATTR_CP_MATCH, (void*)PtrValue2[i], sizeof(PtrValue2[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetEnvAttr_DisplayError(SQL_HANDLE_ENV, henv);}
+
+
+ char PtrValue3[2] = {'SQL_OV_ODBC3', 'SQL_OV_ODBC2'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetEnvAttr(henv, SQL_ATTR_ODBC_VERSION, (void*)PtrValue3[i], sizeof(PtrValue3[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetEnvAttr_DisplayError(SQL_HANDLE_ENV, henv);}
+ */
+
+ char PtrValue4[2] = {'SQL_TRUE', 'SQL_FALSE'};
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetEnvAttr(henv, SQL_ATTR_OUTPUT_NTS, (void*)PtrValue4[i], sizeof(PtrValue4[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_ENV, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetEnvAttr_DisplayError(SQL_HANDLE_ENV, henv);}
+
+ return 0;
+
+ }
+
+
+void SetEnvAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLSetStmtAttrTest.cpp b/ndb/test/odbc/client/SQLSetStmtAttrTest.cpp
new file mode 100644
index 00000000000..646f82cd306
--- /dev/null
+++ b/ndb/test/odbc/client/SQLSetStmtAttrTest.cpp
@@ -0,0 +1,108 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <common.h>
+#define SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+using namespace std;
+
+SQLHDBC hdbc;
+SQLHSTMT hstmt;
+SQLHENV henv;
+SQLHDESC hdesc;
+SQLRETURN retcode, SQLSTATEs;
+
+SQLPOINTER ValuePtr;
+//SQLINTEGER StringLength;
+
+SQLCHAR Sqlstate[5];
+
+SQLINTEGER NativeError;
+SQLSMALLINT i, MsgLen;
+SQLCHAR Msg[SQL_MAXIMUM_MESSAGE_LENGTH];
+
+void SetStmtAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle);
+
+int SQLSetStmtAttrTest()
+{
+ /* SQL/CLI attributes */
+ /* SQL_ATTR_APP_PARAM_DESC */
+ char PtrValue1[13] = {'SQL_NULL_DESC'};
+ for (i=0; i < 1; i++) {
+ retcode = SQLSetStmtAttr(hstmt, SQL_ATTR_APP_PARAM_DESC, (void*)PtrValue1[i], sizeof(PtrValue1[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);}
+
+ /* SQL_ATTR_APP_ROW_DESC */
+ char PtrValue2[1] = {'SQL_NULL_DESC'}; /* ? */
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetStmtAttr(hstmt, SQL_ATTR_APP_ROW_DESC, (void*)PtrValue2[i], sizeof(PtrValue2[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);}
+
+ /* SQL_ATTR_CURSOR_SCROLLABLE */
+ char PtrValue3[2] = {'SQL_NONSCROLLABLE', 'SQL_SCROLLABLE'}; /* ? */
+ for (i=0; i < 2; i++) {
+ retcode = SQLSetStmtAttr(hstmt, SQL_ATTR_CURSOR_SCROLLABLE, (void*)PtrValue3[i], sizeof(PtrValue3[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);}
+
+ /* SQL_ATTR_CURSOR_SENSITIVITY */
+ char PtrValue4[3] = {'SQL_UNSPECIFIED', 'SQL_INSENSITIVE', 'SQL_SENSITIVE'}; /* ? */
+ for (i=0; i < 3; i++) {
+ retcode = SQLSetStmtAttr(hstmt, SQL_ATTR_CURSOR_SENSITIVITY, (void*)PtrValue4[i], sizeof(PtrValue4[i]));
+
+ if (retcode == SQL_INVALID_HANDLE)
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but string SQL_INVALID_HANDLE still appeared. Please check programm" << endl;
+
+ if (retcode == SQL_ERROR || retcode == SQL_SUCCESS_WITH_INFO)
+ SetStmtAttr_DisplayError(SQL_HANDLE_STMT, hstmt);}
+
+ return 0;
+
+ }
+
+
+void SetStmtAttr_DisplayError(SQLSMALLINT HandleType, SQLHENV InputHandle)
+{
+ i = 1;
+ while ((SQLSTATEs = SQLGetDiagRec(HandleType, InputHandle, i,
+ Sqlstate, &NativeError, Msg, sizeof(Msg),
+ &MsgLen)) != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << HandleType << endl;
+ ndbout << "the InputHandle is :" << InputHandle << endl;
+ ndbout << "the output state is:" << (char *)Sqlstate << endl;
+
+ i ++;
+ }
+
+}
+
+
+
diff --git a/ndb/test/odbc/client/SQLTablesTest.cpp b/ndb/test/odbc/client/SQLTablesTest.cpp
new file mode 100644
index 00000000000..735efd81e9c
--- /dev/null
+++ b/ndb/test/odbc/client/SQLTablesTest.cpp
@@ -0,0 +1,227 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLTablesTest.cpp
+ */
+#include <common.hpp>
+using namespace std;
+
+#define Tables_NAME_LEN 12
+#define Tables_PHONE_LEN 12
+#define Tables_ADDRESS_LEN 12
+#define Tables_SQL_MAXIMUM_MESSAGE_LENGTH 200
+
+SQLHDBC Tables_hdbc;
+SQLHSTMT Tables_hstmt;
+SQLHENV Tables_henv;
+SQLHDESC Tables_hdesc;
+
+void Tables_DisplayError(SQLSMALLINT Tables_HandleType,
+ SQLHSTMT Tables_InputHandle);
+
+/**
+ * Test to retrieve a list of table names stored in aspecified
+ * data source's system
+ *
+ * -# Normal case test: print out the table name in the data result set
+ * @return Zero, if test succeeded
+ */
+
+int SQLTablesTest()
+{
+ SQLRETURN Tables_retcode;
+ SQLCHAR Tables_Name[Tables_NAME_LEN], Tables_Phone[Tables_PHONE_LEN];
+ SQLCHAR Tables_Address[Tables_ADDRESS_LEN];
+ SQLINTEGER Tables_CustID;
+
+ ndbout << endl << "Start SQLTables Testing" << endl;
+
+ //*******************************************************************
+ //** hstmt
+ //** Execute a statement to retrieve rows from the Customers table **
+ //** We can create the table and insert rows into Customers **
+ //*******************************************************************
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ Tables_retcode = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &Tables_henv);
+
+if (Tables_retcode == SQL_SUCCESS || Tables_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ Tables_retcode = SQLSetEnvAttr(Tables_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+if (Tables_retcode == SQL_SUCCESS || Tables_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ Tables_retcode = SQLAllocHandle(SQL_HANDLE_DBC,
+ Tables_henv,
+ &Tables_hdbc);
+
+if (Tables_retcode == SQL_SUCCESS || Tables_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ Tables_retcode = SQLConnect(Tables_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+if (Tables_retcode == SQL_SUCCESS || Tables_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ Tables_retcode = SQLAllocHandle(SQL_HANDLE_STMT,
+ Tables_hdbc,
+ &Tables_hstmt);
+
+if (Tables_retcode == SQL_SUCCESS || Tables_retcode == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //**************************************************************
+ //** Retrieve information about the tables in the data source **
+ //**************************************************************
+ Tables_retcode = SQLTables(Tables_hstmt,
+ NULL,
+ 0,
+ NULL,
+ 0,
+ (SQLCHAR *)"%",
+ 128,
+ (SQLCHAR *)"TABLES",
+ 128);
+
+ ndbout <<"Tables_retcode = SQLTables() =" << Tables_retcode;
+
+ if (Tables_retcode == SQL_ERROR)
+ Tables_DisplayError(SQL_HANDLE_STMT, Tables_hstmt);
+ //*******************************************
+ //** Bind columns 3 in the result data set **
+ //*******************************************
+
+ Tables_retcode = SQLBindCol(Tables_hstmt,
+ 3,
+ SQL_C_CHAR,
+ &Tables_Name,
+ Tables_NAME_LEN,
+ NULL);
+
+ ndbout <<"Tables_retcode = SQLBindCol() =" << Tables_retcode;
+
+ //**********************************************
+ //* Fetch and print out data in the result On **
+ //* an error, display a message and exit **
+ //**********************************************
+
+ Tables_retcode = SQLFetch(Tables_hstmt);
+
+
+ ndbout <<"Tables_retcode = SQLFetch() =" << Tables_retcode;
+
+ ndbout << endl << "Tables_retcode = SQLFetch(Tables_hstmt) = "
+ << Tables_retcode << endl;
+
+ if (Tables_retcode == SQL_ERROR)
+ {
+ Tables_DisplayError(SQL_HANDLE_STMT, Tables_hstmt);
+ return NDBT_FAILED;
+ }
+ else if (Tables_retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "Table Name = " << (char *)Tables_Name << endl;
+ Tables_DisplayError(SQL_HANDLE_STMT, Tables_hstmt);
+ }
+ else if (Tables_retcode == SQL_NO_DATA)
+ Tables_DisplayError(SQL_HANDLE_STMT, Tables_hstmt);
+ else
+ {
+ ndbout << "TableName = " << (char *)Tables_Name << endl;
+ Tables_DisplayError(SQL_HANDLE_STMT, Tables_hstmt);
+ }
+
+ // *********************************
+ // ** Disconnect and Free Handles **
+ // *********************************
+ SQLDisconnect(Tables_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, Tables_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, Tables_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, Tables_henv);
+
+ return NDBT_OK;
+}
+
+void Tables_DisplayError(SQLSMALLINT Tables_HandleType,
+ SQLHSTMT Tables_InputHandle)
+{
+ SQLINTEGER NativeError;
+ SQLSMALLINT Tables_i = 1;
+ SQLRETURN Tables__SQLSTATEs;
+ SQLCHAR Tables_Sqlstate[5];
+ SQLCHAR Tables_Msg[Tables_SQL_MAXIMUM_MESSAGE_LENGTH];
+ SQLSMALLINT Tables_MsgLen;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((Tables__SQLSTATEs = SQLGetDiagRec(Tables_HandleType,
+ Tables_InputHandle,
+ Tables_i,
+ Tables_Sqlstate,
+ &NativeError,
+ Tables_Msg,
+ sizeof(Tables_Msg),
+ &Tables_MsgLen)
+ ) != SQL_NO_DATA)
+ {
+
+ ndbout << "the HandleType is:" << Tables_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)Tables_InputHandle << endl;
+ ndbout << "the Tables_Msg is: " << (char *) Tables_Msg << endl;
+ ndbout << "the output state is:" << (char *)Tables_Sqlstate << endl;
+
+ Tables_i ++;
+ break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
diff --git a/ndb/test/odbc/client/SQLTransactTest.cpp b/ndb/test/odbc/client/SQLTransactTest.cpp
new file mode 100644
index 00000000000..e9abe42129d
--- /dev/null
+++ b/ndb/test/odbc/client/SQLTransactTest.cpp
@@ -0,0 +1,305 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file SQLTransactTest.cpp
+ */
+#include <common.hpp>
+#define STR_MESSAGE_LENGTH 200
+#define STR_NAME_LEN 20
+#define STR_PHONE_LEN 20
+#define STR_ADDRESS_LEN 20
+using namespace std;
+
+SQLHDBC STR_hdbc;
+SQLHSTMT STR_hstmt;
+SQLHENV STR_henv;
+SQLHDESC STR_hdesc;
+
+void Transact_DisplayError(SQLSMALLINT STR_HandleType,
+ SQLHSTMT STR_InputHandle);
+
+int STR_Display_Result(SQLHSTMT EXDR_InputHandle);
+
+/**
+ * Test:
+ * -#Test to request a commit or a rollback operation for
+ * all active transactions associated with a specific
+ * environment or connection handle
+ *
+ * @return Zero, if test succeeded
+ */
+
+int SQLTransactTest()
+{
+ SQLRETURN STR_ret;
+
+ ndbout << endl << "Start SQLTransact Testing" << endl;
+
+ //************************************
+ //** Allocate An Environment Handle **
+ //************************************
+ STR_ret = SQLAllocHandle(SQL_HANDLE_ENV,
+ SQL_NULL_HANDLE,
+ &STR_henv);
+
+ if (STR_ret == SQL_SUCCESS || STR_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated an environment Handle!" << endl;
+
+ //*********************************************
+ //** Set the ODBC application Version to 3.x **
+ //*********************************************
+ STR_ret = SQLSetEnvAttr(STR_henv,
+ SQL_ATTR_ODBC_VERSION,
+ (SQLPOINTER) SQL_OV_ODBC3,
+ SQL_IS_UINTEGER);
+
+ if (STR_ret == SQL_SUCCESS || STR_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Set the ODBC application Version to 3.x!" << endl;
+
+ //**********************************
+ //** Allocate A Connection Handle **
+ //**********************************
+
+ STR_ret = SQLAllocHandle(SQL_HANDLE_DBC,
+ STR_henv,
+ &STR_hdbc);
+
+ if (STR_ret == SQL_SUCCESS || STR_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a connection Handle!" << endl;
+
+ // *******************
+ // ** Connect to DB **
+ // *******************
+ STR_ret = SQLConnect(STR_hdbc,
+ (SQLCHAR *) connectString(),
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS,
+ (SQLCHAR *) "",
+ SQL_NTS);
+
+ if (STR_ret == SQL_SUCCESS || STR_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Connected to DB : OK!" << endl;
+ else
+ {
+ ndbout << "Failure to Connect DB!" << endl;
+ return NDBT_FAILED;
+ }
+
+ //*******************************
+ //** Allocate statement handle **
+ //*******************************
+
+ STR_ret = SQLAllocHandle(SQL_HANDLE_STMT,
+ STR_hdbc,
+ &STR_hstmt);
+ if(STR_ret == SQL_SUCCESS || STR_ret == SQL_SUCCESS_WITH_INFO)
+ ndbout << "Allocated a statement handle!" << endl;
+
+ //********************************
+ //** Turn Manual-Commit Mode On **
+ //********************************
+ STR_ret = SQLSetConnectOption(STR_hdbc,
+ SQL_AUTOCOMMIT,
+ (UDWORD) SQL_AUTOCOMMIT_OFF);
+
+ //**********************************************
+ //** Prepare and Execute a prepared statement **
+ //**********************************************
+ STR_ret = SQLExecDirect(STR_hstmt,
+ (SQLCHAR*)"SELECT * FROM Customers",
+ SQL_NTS);
+
+ if (STR_ret == SQL_INVALID_HANDLE)
+ {
+ ndbout << "Handle Type is SQL_HANDLE_STMT, but SQL_INVALID_HANDLE"
+ << endl;
+ ndbout << "still appeared. Please check program" << endl;
+ }
+
+ if (STR_ret == SQL_ERROR || STR_ret == SQL_SUCCESS_WITH_INFO)
+ Transact_DisplayError(SQL_HANDLE_STMT, STR_hstmt);
+
+ //*************************
+ //** Display the results **
+ //*************************
+
+ STR_Display_Result(STR_hstmt);
+
+ //****************************
+ //** Commit the transaction **
+ //****************************
+ STR_ret = SQLTransact(STR_henv,
+ STR_hdbc,
+ SQL_COMMIT);
+
+ //****************
+ // Free Handles **
+ //****************
+ SQLDisconnect(STR_hdbc);
+ SQLFreeHandle(SQL_HANDLE_STMT, STR_hstmt);
+ SQLFreeHandle(SQL_HANDLE_DBC, STR_hdbc);
+ SQLFreeHandle(SQL_HANDLE_ENV, STR_henv);
+
+ return NDBT_OK;
+
+ }
+
+void Transact_DisplayError(SQLSMALLINT STR_HandleType,
+ SQLHSTMT STR_InputHandle)
+{
+ SQLCHAR STR_Sqlstate[5];
+ SQLINTEGER STR_NativeError;
+ SQLSMALLINT STR_i, STR_MsgLen;
+ SQLCHAR STR_Msg[STR_MESSAGE_LENGTH];
+ SQLRETURN SQLSTATEs;
+ STR_i = 1;
+
+ ndbout << "-------------------------------------------------" << endl;
+ ndbout << "Error diagnostics:" << endl;
+
+ while ((SQLSTATEs = SQLGetDiagRec(STR_HandleType,
+ STR_InputHandle,
+ STR_i,
+ STR_Sqlstate,
+ &STR_NativeError,
+ STR_Msg,
+ sizeof(STR_Msg),
+ &STR_MsgLen))
+ != SQL_NO_DATA) {
+
+ ndbout << "the HandleType is:" << STR_HandleType << endl;
+ ndbout << "the InputHandle is :" << (long)STR_InputHandle << endl;
+ ndbout << "the STR_Msg is: " << (char *) STR_Msg << endl;
+ ndbout << "the output state is:" << (char *)STR_Sqlstate << endl;
+
+ STR_i ++;
+ // break;
+ }
+ ndbout << "-------------------------------------------------" << endl;
+}
+
+int STR_Display_Result(SQLHSTMT STR_InputHandle)
+{
+ SQLRETURN STR_retcode;
+ unsigned long STR_CustID;
+ SQLCHAR STR_Name[STR_NAME_LEN], STR_Phone[STR_PHONE_LEN];
+ SQLCHAR STR_Address[STR_ADDRESS_LEN];
+
+ //*********************
+ //** Bind columns 1 **
+ //*********************
+ STR_retcode =SQLBindCol(STR_InputHandle,
+ 1,
+ SQL_C_ULONG,
+ &STR_CustID,
+ sizeof(STR_CustID),
+ NULL);
+ if (STR_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLBindCol, SQL_ERROR happened!" << endl;
+ Transact_DisplayError(SQL_HANDLE_STMT, STR_InputHandle);
+ return NDBT_FAILED;
+ }
+
+ //*********************
+ //** Bind columns 2 **
+ //*********************
+
+ STR_retcode =SQLBindCol(STR_InputHandle,
+ 2,
+ SQL_C_CHAR,
+ &STR_Name,
+ STR_NAME_LEN,
+ NULL);
+ if (STR_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLBindCol, SQL_ERROR happened!" << endl;
+ Transact_DisplayError(SQL_HANDLE_STMT, STR_InputHandle);
+ return NDBT_FAILED;
+ }
+
+ //*********************
+ //** Bind columns 3 **
+ //*********************
+
+ STR_retcode = SQLBindCol(STR_InputHandle,
+ 3,
+ SQL_C_CHAR,
+ &STR_Address,
+ STR_ADDRESS_LEN,
+ NULL);
+
+ if (STR_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLBindCol, SQL_ERROR happened!" << endl;
+ Transact_DisplayError(SQL_HANDLE_STMT, STR_InputHandle);
+ return NDBT_FAILED;
+ }
+
+ //*********************
+ //** Bind columns 4 **
+ //*********************
+
+ STR_retcode = SQLBindCol(STR_InputHandle,
+ 4,
+ SQL_C_CHAR,
+ &STR_Phone,
+ STR_PHONE_LEN,
+ NULL);
+
+ if (STR_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLBindCol, SQL_ERROR happened!" << endl;
+ Transact_DisplayError(SQL_HANDLE_STMT, STR_InputHandle);
+ return NDBT_FAILED;
+ }
+
+ //*****************************************
+ //* Fetch and print each row of data. On **
+ //* an error, display a message and exit **
+ //*****************************************
+
+ if (STR_retcode != SQL_ERROR)
+ STR_retcode = SQLFetch(STR_InputHandle);
+
+ ndbout << endl << "STR_retcode = SQLFetch(STR_InputHandle) = "
+ << STR_retcode << endl;
+
+ if (STR_retcode == SQL_ERROR)
+ {
+ ndbout << "Executing SQLFetch, SQL_ERROR happened!" << endl;
+ Transact_DisplayError(SQL_HANDLE_STMT, STR_InputHandle);
+ return NDBT_FAILED;
+ }
+ else if (STR_retcode == SQL_SUCCESS_WITH_INFO)
+ {
+ ndbout << "CustID = " << (int)STR_CustID << endl;
+ ndbout << "Name = " << (char *)STR_Name << endl;
+ ndbout << "Address = " << (char *)STR_Address << endl;
+ ndbout << "Phone = " << (char *)STR_Phone << endl;
+ Transact_DisplayError(SQL_HANDLE_STMT, STR_InputHandle);
+ }
+ else
+ {
+ ndbout << "CustID = " << (int)STR_CustID << endl;
+ ndbout << "Name = " << (char *)STR_Name << endl;
+ ndbout << "Address = " << (char *)STR_Address << endl;
+ ndbout << "Phone = " << (char *)STR_Phone << endl;
+ }
+ return 0;
+}
diff --git a/ndb/test/odbc/client/common.hpp b/ndb/test/odbc/client/common.hpp
new file mode 100644
index 00000000000..236decf1b95
--- /dev/null
+++ b/ndb/test/odbc/client/common.hpp
@@ -0,0 +1,81 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <NDBT.hpp>
+#include <sqlext.h>
+#include <stdio.h>
+#include <string.h>
+#define FAILURE(msg) { ndbout << "TEST FAILURE: " << msg << endl \
+ << "-- File: " << __FILE__ << ", Line: " << __LINE__ << endl; \
+ return NDBT_FAILED; }
+
+char* connectString();
+
+int SQLFetchTest();
+int SQLDisconnectTest();
+int SQLTablesTest();
+int SQLBindColTest();
+int SQLGetInfoTest();
+int SQLGetTypeInfoTest();
+int SQLGetDataTest();
+int SQLGetFunctionsTest();
+int SQLColAttributeTest();
+int SQLColAttributeTest1();
+int SQLColAttributeTest2();
+int SQLColAttributeTest3();
+int SQLGetDiagRecSimpleTest();
+int SQLDriverConnectTest();
+int SQLAllocEnvTest();
+int SQLFreeHandleTest();
+int SQLFetchScrollTest();
+int SQLFetchTest();
+int SQLGetDescRecTest();
+int SQLSetDescFieldTest();
+int SQLGetDescFieldTest();
+int SQLSetDescRecTest();
+int SQLSetCursorNameTest();
+int SQLGetCursorNameTest();
+int SQLRowCountTest();
+int SQLGetInfoTest();
+int SQLTransactTest();
+int SQLEndTranTest();
+int SQLNumResultColsTest();
+int SQLGetTypeInfoTest();
+int SQLGetFunctionsTest();
+int SQLDescribeColTest();
+int SQLAllocHandleTest();
+int SQLCancelTest();
+int SQLCloseCursorTest();
+int SQLConnectTest();
+int SQLDisconnectTest();
+int SQLExecDirectTest();
+int SQLExecuteTest();
+int SQLFreeHandleTest();
+int SQLFreeStmtTest();
+int SQLGetConnectAttrTest();
+int SQLGetEnvAttrTest();
+int SQLGetStmtAttrTest();
+int SQLMoreResultsTest();
+int SQLPrepareTest();
+int SQLSetConnectAttrTest();
+int SQLSetEnvAttrTest();
+int SQLSetStmtAttrTest();
+
+// int NDBT_ALLOCHANDLE();
+// int NDBT_ALLOCHANDLE_HDBC();
+// int NDBT_SQLPrepare();
+// int NDBT_SQLConnect();
diff --git a/ndb/test/odbc/client/main.cpp b/ndb/test/odbc/client/main.cpp
new file mode 100644
index 00000000000..b202b6de111
--- /dev/null
+++ b/ndb/test/odbc/client/main.cpp
@@ -0,0 +1,158 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+ /**
+ * @file main.cpp
+ */
+
+#include <common.hpp>
+
+/**
+ * main ODBC Tests
+ *
+ * Tests main ODBC functions.
+ */
+
+#include <common.hpp>
+
+int check = NDBT_OK;
+static char* myConnectString;
+
+char* connectString()
+{
+ return myConnectString;
+}
+
+int main(int argc, char** argv)
+{
+
+ if (argc != 3) {
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ myConnectString = argv[1];
+
+ if ( strcmp(argv[2], "help") == 0 )
+ {
+ ndbout << "Number of Testing Program " << " Name of Testing Program" << endl;
+ ndbout << " 1 SQLGetDataTest()" << endl;
+ ndbout << " 2 SQLTablesTest()" << endl;
+ ndbout << " 3 SQLGetFunctionsTest()" << endl;
+ ndbout << " 4 SQLGetInfoTest()" << endl;
+ ndbout << " 5 SQLGetTypeInfoTest()" << endl;
+ ndbout << " 6 SQLDisconnectTest()" << endl;
+ ndbout << " 7 SQLFetchTest()" << endl;
+ ndbout << " 8 SQLRowCountTest()" << endl;
+ ndbout << " 9 SQLGetCursorNameTest()" << endl;
+ ndbout << " 10 SQLCancelTest()" << endl;
+ ndbout << " 11 SQLTransactTest()" << endl;
+ ndbout << " 12 SQLSetCursorNameTest()" << endl;
+ ndbout << " 13 SQLNumResultColsTest()" << endl;
+ ndbout << " 14 SQLDescribeColTest()" << endl;
+ ndbout << " 15 SQLExecDirectTest()" << endl;
+ ndbout << " 16 SQLColAttributeTest3()" << endl;
+ ndbout << " 17 SQLColAttributeTest2()" << endl;
+ ndbout << " 18 SQLColAttributeTest1()" << endl;
+ ndbout << " 19 SQLColAttributeTest()" << endl;
+ ndbout << " 20 SQLBindColTest()" << endl;
+ ndbout << " 21 SQLGetDiagRecSimpleTest()" << endl;
+ ndbout << " 22 SQLConnectTest()" << endl;
+ ndbout << " 23 SQLPrepareTest()" << endl;
+ }
+ else
+ {
+
+ ndbout << endl << "Executing Files Name = " << argv[0] << endl;
+ ndbout << "The Number of testing program = " << argv[2] << endl;
+
+ int i = atoi(argv[2]);
+ switch (i) {
+ case 1:
+ if (check == NDBT_OK) check = SQLGetDataTest();
+ break;
+ case 2:
+ if (check == NDBT_OK) check = SQLTablesTest();
+ break;
+ case 3:
+ if (check == NDBT_OK) check = SQLGetFunctionsTest();
+ break;
+ case 4:
+ if (check == NDBT_OK) check = SQLGetInfoTest();
+ break;
+ case 5:
+ if (check == NDBT_OK) check = SQLGetTypeInfoTest();
+ break;
+ case 6:
+ if (check == NDBT_OK) check = SQLDisconnectTest();
+ break;
+ case 7:
+ if (check == NDBT_OK) check = SQLFetchTest();
+ break;
+ case 8:
+ if (check == NDBT_OK) check = SQLRowCountTest();
+ break;
+ case 9:
+ if (check == NDBT_OK) check = SQLGetCursorNameTest();
+ break;
+ case 10:
+ if (check == NDBT_OK) check = SQLCancelTest();
+ break;
+ case 11:
+ if (check == NDBT_OK) check = SQLTransactTest();
+ break;
+ case 12:
+ if (check == NDBT_OK) check = SQLSetCursorNameTest();
+ break;
+ case 13:
+ if (check == NDBT_OK) check = SQLNumResultColsTest();
+ break;
+ case 14:
+ if (check == NDBT_OK) check = SQLDescribeColTest();
+ break;
+ case 15:
+ if (check == NDBT_OK) check = SQLExecDirectTest();
+ break;
+ case 16:
+ if (check == NDBT_OK) check = SQLColAttributeTest3();
+ break;
+ case 17:
+ if (check == NDBT_OK) check = SQLColAttributeTest2();
+ break;
+ case 18:
+ if (check == NDBT_OK) check = SQLColAttributeTest1();
+ break;
+ case 19:
+ if (check == NDBT_OK) check = SQLColAttributeTest();
+ break;
+ case 20:
+ if (check == NDBT_OK) check = SQLBindColTest();
+ break;
+ case 21:
+ if (check == NDBT_OK) check = SQLGetDiagRecSimpleTest();
+ break;
+ case 22:
+ if (check == NDBT_OK) check = SQLConnectTest();
+ break;
+ case 23:
+ if (check == NDBT_OK) check = SQLPrepareTest();
+ break;
+ }
+ }
+
+ return NDBT_ProgramExit(check);
+}
+
+
+
diff --git a/ndb/test/odbc/dm-iodbc/Makefile b/ndb/test/odbc/dm-iodbc/Makefile
new file mode 100644
index 00000000000..ad0f0d39f5f
--- /dev/null
+++ b/ndb/test/odbc/dm-iodbc/Makefile
@@ -0,0 +1,38 @@
+include .defs.mk
+
+TYPE = *
+
+BIN_TARGET = testOdbcDMi
+
+SOURCES = testOdbcDMi.cpp
+
+CCFLAGS_LOC += \
+ -I$(NDB_TOP)/include \
+ -I$(NDB_TOP)/include/ndbapi \
+ -I$(NDB_TOP)/include/portlib \
+ -I$(NDB_TOP)/include/util \
+ -I$(NDB_TOP)/test/include
+
+CCFLAGS_WARNINGS += -Wno-unused -Wno-sign-compare
+
+CCFLAGS_TOP += -DHAVE_LONG_LONG -DiODBC
+
+BIN_TARGET_LIBS = NDBT general portlib
+
+ifeq ($(NDB_OS),SOLARIS)
+CCFLAGS_TOP += -DDMALLOC
+LIBS_LOC += -L/usr/local/opt/iODBC/lib
+LIBS_LOC += -R/usr/local/opt/iODBC/lib
+BIN_TARGET_LIBS += iodbc
+BIN_TARGET_LIBS += dmallocthcxx
+endif
+
+ifeq ($(NDB_OS),LINUX)
+BIN_TARGET_LIBS_DIRS += /usr/local/opt/iODBC/lib
+BIN_TARGET_LIBS += iodbc
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+testOdbcDMi.cpp:
+ ln -s ../driver/testOdbcDriver.cpp $@
diff --git a/ndb/test/odbc/dm-unixodbc/Makefile b/ndb/test/odbc/dm-unixodbc/Makefile
new file mode 100644
index 00000000000..50d8e3b5e05
--- /dev/null
+++ b/ndb/test/odbc/dm-unixodbc/Makefile
@@ -0,0 +1,39 @@
+include .defs.mk
+
+TYPE = *
+
+BIN_TARGET = testOdbcDMu
+
+SOURCES = testOdbcDMu.cpp
+
+CCFLAGS_LOC += \
+ -I$(NDB_TOP)/include \
+ -I$(NDB_TOP)/include/ndbapi \
+ -I$(NDB_TOP)/include/portlib \
+ -I$(NDB_TOP)/include/util \
+ -I$(NDB_TOP)/test/include
+
+CCFLAGS_WARNINGS += -Wno-unused -Wno-sign-compare
+
+CCFLAGS_TOP += -DHAVE_LONG_LONG -DunixODBC
+
+BIN_TARGET_LIBS = NDBT general portlib
+
+ifeq ($(NDB_OS),SOLARIS)
+CCFLAGS_TOP += -DDMALLOC
+LIBS_LOC += -L/usr/local/lib
+BIN_TARGET_LIBS += odbc odbcinst
+BIN_TARGET_LIBS += dmallocthcxx
+endif
+
+ifeq ($(NDB_OS),LINUX)
+BIN_TARGET_LIBS += odbc odbcinst
+BIN_TARGET_LIBS_DIRS += .
+dummy := $(shell [ ! -f /usr/lib/libodbc.so ] || ln -sf /usr/lib/libodbc.so.1 libodbc.so)
+dummy := $(shell [ ! -f /usr/lib/libodbcinst.so ] || ln -sf /usr/lib/libodbcinst.so.1 libodbcinst.so)
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+testOdbcDMu.cpp:
+ ln -s ../driver/testOdbcDriver.cpp $@
diff --git a/ndb/test/odbc/driver/Makefile b/ndb/test/odbc/driver/Makefile
new file mode 100644
index 00000000000..5cf83d73106
--- /dev/null
+++ b/ndb/test/odbc/driver/Makefile
@@ -0,0 +1,30 @@
+include .defs.mk
+
+TYPE = *
+
+BIN_TARGET = testOdbcDriver
+
+SOURCES = testOdbcDriver.cpp
+
+CCFLAGS_LOC += \
+ -I$(NDB_TOP)/include \
+ -I$(NDB_TOP)/include/ndbapi \
+ -I$(NDB_TOP)/include/portlib \
+ -I$(NDB_TOP)/include/util \
+ -I$(NDB_TOP)/test/include \
+ -I/usr/local/include
+
+CCFLAGS_WARNINGS += -Wno-unused -Wno-sign-compare -Wformat
+
+CCFLAGS_TOP += -DHAVE_LONG_LONG -DndbODBC
+
+BIN_TARGET_LIBS = NDBT NDB_ODBC
+
+ifeq ($(NDB_OS),SOLARIS)
+BIN_TARGET_LIBS += dmallocthcxx
+CCFLAGS_TOP += -DDMALLOC
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+$(BIN_DIR)$(BIN_TARGET): Makefile
diff --git a/ndb/test/odbc/driver/testOdbcDriver.cpp b/ndb/test/odbc/driver/testOdbcDriver.cpp
new file mode 100644
index 00000000000..9731c00eeaf
--- /dev/null
+++ b/ndb/test/odbc/driver/testOdbcDriver.cpp
@@ -0,0 +1,4969 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* Copyright (C) 2003 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
+
+ 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 */
+
+/*
+ * testOdbcDriver
+ *
+ * Test of ODBC and SQL using a fixed set of tables.
+ */
+
+#include <new>
+#include <NdbUnistd.h>
+#include <stdlib.h>
+#include <NdbString.h>
+#include <stdarg.h>
+#include <NdbStdio.h>
+#include <ndb_version.h>
+#include <kernel/ndb_limits.h>
+#include <Bitmask.hpp>
+#include <kernel/AttributeList.hpp>
+#ifdef ndbODBC
+#include <NdbApi.hpp>
+#endif
+#include <math.h>
+#include <sqlext.h>
+#include <assert.h>
+
+#undef BOOL
+
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbThread.h>
+#include <NdbMutex.h>
+#include <NdbCondition.h>
+#include <NdbTick.h>
+#include <NdbSleep.h>
+
+#ifdef ndbODBC
+#include <NdbTest.hpp>
+#else
+#define NDBT_OK 0
+#define NDBT_FAILED 1
+#define NDBT_WRONGARGS 2
+static int
+NDBT_ProgramExit(int rcode)
+{
+ const char* rtext = "Unknown";
+ switch (rcode) {
+ case NDBT_OK:
+ rtext = "OK";
+ break;
+ case NDBT_FAILED:
+ rtext = "Failed";
+ break;
+ case NDBT_WRONGARGS:
+ rtext = "Wrong arguments";
+ break;
+ };
+ ndbout_c("\nNDBT_ProgramExit: %d - %s\n", rcode, rtext);
+ return rcode;
+}
+#endif
+
+#ifdef DMALLOC
+#include <dmalloc.h>
+#endif
+
+#define arraySize(x) (sizeof(x)/sizeof(x[0]))
+
+#define SQL_ATTR_NDB_TUPLES_FETCHED 66601
+
+// options
+
+#define MAX_THR 128 // max threads
+
+struct Opt {
+ const char* m_name[100];
+ unsigned m_namecnt;
+ bool m_core;
+ unsigned m_depth;
+ const char* m_dsn;
+ unsigned m_errs;
+ const char* m_fragtype;
+ unsigned m_frob;
+ const char* m_home;
+ unsigned m_loop;
+ bool m_nogetd;
+ bool m_noputd;
+ bool m_nosort;
+ unsigned m_scale;
+ bool m_serial;
+ const char* m_skip[100];
+ unsigned m_skipcnt;
+ unsigned m_subloop;
+ const char* m_table;
+ unsigned m_threads;
+ unsigned m_trace;
+ unsigned m_v;
+ Opt() :
+ m_namecnt(0),
+ m_core(false),
+ m_depth(5),
+ m_dsn("NDB"),
+ m_errs(0),
+ m_fragtype(0),
+ m_frob(0),
+ m_home(0),
+ m_loop(1),
+ m_nogetd(false),
+ m_noputd(false),
+ m_nosort(false),
+ m_scale(100),
+ m_serial(false),
+ m_skipcnt(0),
+ m_subloop(1),
+ m_table(0),
+ m_threads(1),
+ m_trace(0),
+ m_v(1) {
+ for (unsigned i = 0; i < arraySize(m_name); i++)
+ m_name[i] = 0;
+ for (unsigned i = 0; i < arraySize(m_skip); i++)
+ m_skip[i] = 0;
+ }
+};
+
+static Opt opt;
+
+static void listCases();
+static void listTables();
+static void printusage()
+{
+ Opt d;
+ ndbout
+ << "usage: testOdbcDriver [options]" << endl
+ << "-case name run only named tests (substring match - can be repeated)" << endl
+ << "-core dump core on failure" << endl
+ << "-depth N join depth - default " << d.m_depth << endl
+ << "-dsn string data source name - default " << d.m_dsn << endl
+ << "-errs N allow N errors before quitting - default " << d.m_errs << endl
+ << "-fragtype t fragment type single/small/medium/large" << d.m_errs << endl
+ << "-frob X case-dependent tweak (number)" << endl
+ << "-home dir set NDB_HOME (contains Ndb.cfg)" << endl
+ << "-loop N loop N times (0 = forever) - default " << d.m_loop << endl
+ << "-nogetd do not use SQLGetData - default " << d.m_nogetd << endl
+ << "-noputd do not use SQLPutData - default " << d.m_noputd << endl
+ << "-nosort no order-by in verify scan (checks non-Pk values only)" << endl
+ << "-scale N row count etc - default " << d.m_scale << endl
+ << "-serial run multi-threaded test cases one at a time" << endl
+ << "-skip name skip named tests (substring match - can be repeated)" << endl
+ << "-subloop N loop count per case (same threads) - default " << d.m_subloop << endl
+ << "-table T do only table T (table name on built-in list)" << endl
+ << "-threads N number of threads (max " << MAX_THR << ") - default " << d.m_threads << endl
+ << "-trace N trace in NDB ODBC driver - default " << d.m_trace << endl
+ << "-v N verbosity - default " << d.m_v << endl
+ ;
+ listCases();
+ listTables();
+}
+
+static void
+fatal(const char* fmt, ...)
+{
+ va_list ap;
+ char buf[200];
+ va_start(ap, fmt);
+ vsnprintf(buf, sizeof(buf), fmt, ap);
+ va_end(ap);
+ ndbout << buf << endl;
+ if (opt.m_errs != 0) {
+ opt.m_errs--;
+ return;
+ }
+ if (opt.m_core)
+ abort();
+ NDBT_ProgramExit(NDBT_FAILED);
+ exit(1);
+}
+
+static void
+cleanprint(const char* s, unsigned n)
+{
+ for (unsigned i = 0; i < n; i++) {
+ char b[10];
+ if (0x20 < s[i] && s[i] <= 0x7e)
+ sprintf(b, "%c", s[i]);
+ else
+ sprintf(b, "\\%02x", (unsigned)s[i]);
+ ndbout << b;
+ }
+}
+
+// global mutex
+static NdbMutex my_mutex = NDB_MUTEX_INITIALIZER;
+static void lock_mutex() { NdbMutex_Lock(&my_mutex); }
+static void unlock_mutex() { NdbMutex_Unlock(&my_mutex); }
+
+// semaphore zeroed before each call to a test routine
+static unsigned my_sema = 0;
+
+// print mutex
+static NdbMutex out_mutex = NDB_MUTEX_INITIALIZER;
+static NdbOut& lock(NdbOut& out) { NdbMutex_Lock(&out_mutex); return out; }
+static NdbOut& unlock(NdbOut& out) { NdbMutex_Unlock(&out_mutex); return out; }
+
+static unsigned
+urandom(unsigned n)
+{
+ assert(n != 0);
+ unsigned i = random();
+ return i % n;
+}
+
+// test cases
+
+struct Test;
+
+struct Case {
+ enum Mode {
+ Single = 1, // single thread
+ Serial = 2, // all threads but one at a time
+ Thread = 3 // all threads in parallel
+ };
+ const char* m_name;
+ void (*m_func)(Test& test);
+ Mode m_mode;
+ unsigned m_stuff;
+ const char* m_desc;
+ Case(const char* name, void (*func)(Test& test), Mode mode, unsigned stuff, const char* desc) :
+ m_name(name),
+ m_func(func),
+ m_mode(mode),
+ m_stuff(stuff),
+ m_desc(desc) {
+ }
+ const char* modename() const {
+ const char* s = "?";
+ if (m_mode == Case::Single)
+ return "Single";
+ if (m_mode == Case::Serial)
+ return "Serial";
+ if (m_mode == Case::Thread)
+ return "Thread";
+ return "?";
+ }
+ bool matchcase() const {
+ if (opt.m_namecnt == 0)
+ return ! skipcase();
+ for (unsigned i = 0; i < opt.m_namecnt; i++) {
+ if (strstr(m_name, opt.m_name[i]) != 0)
+ return ! skipcase();
+ }
+ return false;
+ }
+private:
+ bool skipcase() const {
+ for (unsigned i = 0; i < opt.m_skipcnt; i++) {
+ if (strstr(m_name, opt.m_skip[i]) != 0)
+ return true;
+ }
+ return false;
+ }
+};
+
+// calculate values
+
+struct Calc {
+ enum { m_mul = 1000000 };
+ unsigned m_no;
+ unsigned m_base;
+ unsigned m_salt; // modifies non-PK values
+ bool m_const; // base non-PK values on PK of row 0
+ Calc(unsigned no) :
+ m_no(no),
+ m_salt(0),
+ m_const(false) {
+ m_base = m_no * m_mul;
+ }
+ void calcPk(unsigned rownum, char* v, unsigned n) const {
+ char b[10];
+ sprintf(b, "%08x", m_base + rownum);
+ for (unsigned i = 0; i < n; i++) {
+ char c = i < n - 1 ? b[i % 8] : 0;
+ v[i] = c;
+ }
+ }
+ void calcPk(unsigned rownum, long* v) const {
+ *v = m_base + rownum;
+ }
+ void hashPk(unsigned* hash, const char* v, unsigned n) const {
+ for (unsigned i = 0; i < n; i++) {
+ *hash ^= (v[i] << i);
+ }
+ }
+ void hashPk(unsigned* hash, long v) const {
+ *hash ^= v;
+ }
+ void calcNk(unsigned hash, char* v, unsigned n, SQLINTEGER* ind, bool null) const {
+ unsigned m = hash % n;
+ for (unsigned i = 0; i < n; i++) {
+ char c = i < m ? 'a' + (hash + i) % ('z' - 'a' + 1) : i < n - 1 ? ' ' : 0;
+ v[i] = c;
+ }
+ *ind = null && hash % 9 == 0 ? SQL_NULL_DATA : SQL_NTS;
+ }
+ void calcNk(unsigned hash, long* v, SQLINTEGER* ind, bool null) const {
+ *v = long(hash);
+ *ind = null && hash % 7 == 0 ? SQL_NULL_DATA : 0;
+ }
+ void calcNk(unsigned hash, double* v, SQLINTEGER* ind, bool null) const {
+ *v = long(hash) / 1000.0;
+ *ind = null && hash % 5 == 0 ? SQL_NULL_DATA : 0;
+ }
+ bool verify(const char* v1, SQLINTEGER ind1, const char* v2, SQLINTEGER ind2, unsigned n) {
+ if (ind1 == SQL_NULL_DATA && ind2 == SQL_NULL_DATA)
+ return true;
+ if (ind1 != SQL_NULL_DATA && ind2 != SQL_NULL_DATA)
+ if (memcmp(v1, v2, n) == 0)
+ return true;
+ if (ind1 == SQL_NULL_DATA)
+ v1 = "NULL";
+ if (ind2 == SQL_NULL_DATA)
+ v2 = "NULL";
+ ndbout << "verify failed: got ";
+ if (ind1 == SQL_NULL_DATA)
+ ndbout << "NULL";
+ else
+ cleanprint(v1, n);
+ ndbout << " != ";
+ if (ind2 == SQL_NULL_DATA)
+ ndbout << "NULL";
+ else
+ cleanprint(v2, n);
+ ndbout << endl;
+ return false;
+ }
+ bool verify(long v1, SQLINTEGER ind1, long v2, SQLINTEGER ind2) {
+ char buf1[40], buf2[40];
+ if (ind1 == SQL_NULL_DATA && ind2 == SQL_NULL_DATA)
+ return true;
+ if (ind1 != SQL_NULL_DATA && ind2 != SQL_NULL_DATA)
+ if (v1 == v2)
+ return true;
+ if (ind1 == SQL_NULL_DATA)
+ strcpy(buf1, "NULL");
+ else
+ sprintf(buf1, "%ld", v1);
+ if (ind2 == SQL_NULL_DATA)
+ strcpy(buf2, "NULL");
+ else
+ sprintf(buf2, "%ld", v2);
+ ndbout << "verify failed: got " << buf1 << " != " << buf2 << endl;
+ return false;
+ }
+ bool verify(double v1, SQLINTEGER ind1, double v2, SQLINTEGER ind2) {
+ char buf1[40], buf2[40];
+ if (ind1 == SQL_NULL_DATA && ind2 == SQL_NULL_DATA)
+ return true;
+ if (ind1 != SQL_NULL_DATA && ind2 != SQL_NULL_DATA)
+ if (fabs(v1 - v2) < 1) // XXX
+ return true;
+ if (ind1 == SQL_NULL_DATA)
+ strcpy(buf1, "NULL");
+ else
+ sprintf(buf1, "%.10f", v1);
+ if (ind2 == SQL_NULL_DATA)
+ strcpy(buf2, "NULL");
+ else
+ sprintf(buf2, "%.10f", v2);
+ ndbout << "verify failed: got " << buf1 << " != " << buf2 << endl;
+ return false;
+ }
+};
+
+#if defined(NDB_SOLARIS) || defined(NDB_LINUX) || defined(NDB_MACOSX)
+#define HAVE_SBRK
+#else
+#undef HAVE_SBRK
+#endif
+
+struct Timer {
+ Timer() :
+ m_cnt(0),
+ m_calls(0),
+ m_on(0),
+ m_msec(0)
+#ifndef NDB_WIN32
+ ,
+ m_brk(0),
+ m_incr(0)
+#endif
+ {
+ }
+ void timerOn() {
+ m_cnt = 0;
+ m_calls = 0;
+ m_on = NdbTick_CurrentMillisecond();
+#ifdef HAVE_SBRK
+ m_brk = (int)sbrk(0);
+#endif
+ }
+ void timerOff() {
+ m_msec = NdbTick_CurrentMillisecond() - m_on;
+ if (m_msec <= 0)
+ m_msec = 1;
+#ifdef HAVE_SBRK
+ m_incr = (int)sbrk(0) - m_brk;
+ if (m_incr < 0)
+ m_incr = 0;
+#endif
+ }
+ void timerCnt(unsigned cnt) {
+ m_cnt += cnt;
+ }
+ void timerCnt(const Timer& timer) {
+ m_cnt += timer.m_cnt;
+ m_calls += timer.m_calls;
+ }
+ friend NdbOut& operator<<(NdbOut& out, const Timer& timer) {
+ out << timer.m_cnt << " ( " << 1000 * timer.m_cnt / timer.m_msec << "/sec )";
+#ifdef HAVE_SBRK
+ out << " - " << timer.m_incr << " sbrk";
+ if (opt.m_namecnt != 0) { // per case meaningless if many cases
+ if (timer.m_cnt > 0)
+ out << " ( " << timer.m_incr / timer.m_cnt << "/cnt )";
+ }
+#endif
+ out << " - " << timer.m_calls << " calls";
+ return out;
+ }
+protected:
+ unsigned m_cnt; // count rows or whatever
+ unsigned m_calls; // count ODBC function calls
+ NDB_TICKS m_on;
+ unsigned m_msec;
+#ifdef HAVE_SBRK
+ int m_brk;
+ int m_incr;
+#endif
+};
+
+#define MAX_MESSAGE 500
+#define MAX_DIAG 20
+
+struct Diag {
+ char m_state[5+1];
+ SQLINTEGER m_native;
+ char m_message[MAX_MESSAGE];
+ unsigned m_flag; // temp use
+ Diag() {
+ strcpy(m_state, "00000");
+ m_native = 0;
+ memset(m_message, 0, sizeof(m_message));
+ m_flag = 0;
+ }
+ const char* text() {
+ snprintf(m_buf, sizeof(m_buf), "%s %d '%s'", m_state, (int)m_native, m_message);
+ return m_buf;
+ }
+ void getDiag(SQLSMALLINT type, SQLHANDLE handle, unsigned k, unsigned count) {
+ int ret;
+ SQLSMALLINT length = -1;
+ memset(m_message, 0, MAX_MESSAGE);
+ ret = SQLGetDiagRec(type, handle, k, (SQLCHAR*)m_state, &m_native, (SQLCHAR*)m_message, MAX_MESSAGE, &length);
+ if (k <= count && ret != SQL_SUCCESS)
+ fatal("SQLGetDiagRec %d of %d: return %d != SQL_SUCCESS", k, count, (int)ret);
+ if (k <= count && strlen(m_message) != length)
+ fatal("SQLGetDiagRec %d of %d: message length %d != %d", k, count, strlen(m_message), length);
+ if (k > count && ret != SQL_NO_DATA)
+ fatal("SQLGetDiagRec %d of %d: return %d != SQL_NO_DATA", k, count, (int)ret);
+ m_flag = 0;
+ }
+private:
+ char m_buf[MAX_MESSAGE];
+};
+
+struct Diags {
+ Diag m_diag[MAX_DIAG];
+ SQLINTEGER m_diagCount;
+ SQLINTEGER m_rowCount;
+ SQLINTEGER m_functionCode;
+ void getDiags(SQLSMALLINT type, SQLHANDLE handle) {
+ int ret;
+ m_diagCount = -1;
+ ret = SQLGetDiagField(type, handle, 0, SQL_DIAG_NUMBER, &m_diagCount, SQL_IS_INTEGER, 0);
+ if (ret == SQL_INVALID_HANDLE)
+ return;
+ if (ret != SQL_SUCCESS)
+ fatal("SQLGetDiagField: return %d != SQL_SUCCESS", (int)ret);
+ if (m_diagCount < 0 || m_diagCount > MAX_DIAG)
+ fatal("SQLGetDiagField: count %d", (int)m_diagCount);
+ for (unsigned k = 0; k < MAX_DIAG; k++) {
+ m_diag[k].getDiag(type, handle, k + 1, m_diagCount);
+ if (k == m_diagCount + 1)
+ break;
+ }
+ m_rowCount = -1;
+ m_functionCode = SQL_DIAG_UNKNOWN_STATEMENT;
+ if (type == SQL_HANDLE_STMT) {
+ ret = SQLGetDiagField(type, handle, 0, SQL_DIAG_ROW_COUNT, &m_rowCount, SQL_IS_INTEGER, 0);
+#ifndef iODBC
+ if (ret != SQL_SUCCESS)
+ fatal("SQLGetDiagField: return %d != SQL_SUCCESS", (int)ret);
+#endif
+ ret = SQLGetDiagField(type, handle, 0, SQL_DIAG_DYNAMIC_FUNCTION_CODE, &m_functionCode, SQL_IS_INTEGER, 0);
+ }
+ }
+ void showDiags() {
+ for (unsigned k = 0; 0 <= m_diagCount && k < m_diagCount; k++) {
+ Diag& diag = m_diag[k];
+ ndbout << "diag " << k + 1;
+ ndbout << (diag.m_flag ? " [*]" : " [ ]");
+ ndbout << " " << diag.text() << endl;
+ if (k > 10)
+ abort();
+ }
+ }
+};
+
+struct Exp {
+ int m_ret;
+ const char* m_state;
+ SQLINTEGER m_native;
+ Exp() : m_ret(SQL_SUCCESS), m_state(""), m_native(0) {}
+ Exp(int ret, const char* state) : m_ret(ret), m_state(state) {}
+};
+
+struct Test : Calc, Timer, Diags {
+ Test(unsigned no, unsigned loop) :
+ Calc(no),
+ m_loop(loop),
+ m_stuff(0),
+ m_perf(false),
+ ccp(0) {
+ exp(SQL_SUCCESS, 0, 0, true);
+ }
+ unsigned m_loop; // current loop
+ Exp m_expList[20]; // expected results
+ unsigned m_expCount;
+ int m_ret; // actual return code
+ int m_stuff; // the stuff of abuse
+ bool m_perf; // check no diags on success
+ const Case* ccp; // current case
+ void exp(int ret, const char* state, SQLINTEGER native, bool reset) {
+ if (reset)
+ m_expCount = 0;
+ unsigned i = m_expCount++;
+ assert(i < arraySize(m_expList) - 1);
+ m_expList[i].m_ret = ret;
+ m_expList[i].m_state = state == 0 ? "" : state;
+ m_expList[i].m_native = native;
+ }
+ void runCase(const Case& cc) {
+ ccp = &cc;
+ if (opt.m_v >= 3)
+ ndbout << cc.m_name << ": start" << endl;
+ m_rowCount = -1;
+ NDB_TICKS m_ms1 = NdbTick_CurrentMillisecond();
+ m_salt = m_loop | (16 << cc.m_stuff);
+ m_const = cc.m_stuff == 0;
+ m_stuff = cc.m_stuff;
+ (*cc.m_func)(*this);
+ NDB_TICKS m_ms2 = NdbTick_CurrentMillisecond();
+ }
+ void run(SQLSMALLINT type, SQLHANDLE handle, int line, int ret) {
+ m_calls++;
+ m_ret = ret;
+ if (m_perf && (m_ret == SQL_SUCCESS))
+ return;
+ m_diagCount = 0;
+ if (handle != SQL_NULL_HANDLE)
+ getDiags(type, handle);
+ if (m_diagCount <= 0 && (ret != SQL_SUCCESS && ret != SQL_INVALID_HANDLE && ret != SQL_NEED_DATA && ret != SQL_NO_DATA)) {
+ fatal("%s: thr %d line %d: ret=%d but no diag records", ccp->m_name, m_no, line, ret);
+ }
+ for (unsigned k = 0; 0 <= m_diagCount && k < m_diagCount; k++) {
+ Diag& diag = m_diag[k];
+ bool match = false;
+ for (unsigned i = 0; i < m_expCount; i++) {
+ if (strcmp(diag.m_state, m_expList[i].m_state) == 0 && (diag.m_native % 10000 == m_expList[i].m_native % 10000 || m_expList[i].m_native == -1)) {
+ match = true;
+ diag.m_flag = 0;
+ continue;
+ }
+ diag.m_flag = 1; // mark unexpected
+ }
+ if (! match) {
+ showDiags();
+ fatal("%s: thr %d line %d: unexpected diag [*] ret=%d cnt=%d", ccp->m_name, m_no, line, (int)ret, (int)m_diagCount);
+ }
+ }
+ bool match = false;
+ for (unsigned i = 0; i < m_expCount; i++) {
+ if (ret == m_expList[i].m_ret) {
+ match = true;
+ break;
+ }
+ }
+ if (! match) {
+ showDiags();
+ fatal("%s: thr %d line %d: ret=%d not expected", ccp->m_name, m_no, line, ret);
+ }
+ // reset expected to success
+ exp(SQL_SUCCESS, 0, 0, true);
+ }
+ void chk(SQLSMALLINT type, SQLHANDLE handle, int line, bool match, const char* fmt, ...) {
+ if (match)
+ return;
+ va_list ap;
+ va_start(ap, fmt);
+ char buf[500];
+ vsnprintf(buf, sizeof(buf), fmt, ap);
+ va_end(ap);
+ fatal("%s: thr %d line %d: check failed - %s", ccp->m_name, m_no, line, buf);
+ }
+};
+
+#define HNull 0, SQL_NULL_HANDLE, __LINE__
+#define HEnv(h) SQL_HANDLE_ENV, h, __LINE__
+#define HDbc(h) SQL_HANDLE_DBC, h, __LINE__
+#define HStmt(h) SQL_HANDLE_STMT, h, __LINE__
+#define HDesc(h) SQL_HANDLE_DESC, h, __LINE__
+
+// string support
+
+#define MAX_SQL 20000
+
+static void
+scopy(char*& ptr, const char* fmt, ...)
+{
+ va_list ap;
+ va_start(ap, fmt);
+ vsprintf(ptr, fmt, ap);
+ va_end(ap);
+ ptr += strlen(ptr);
+}
+
+static bool
+blankeq(const char* s1, const char* s2, bool caseSensitive = false)
+{
+ unsigned n1 = strlen(s1);
+ unsigned n2 = strlen(s2);
+ unsigned i = 0;
+ char c1 = 0;
+ char c2 = 0;
+ while (i < n1 || i < n2) {
+ c1 = i < n1 ? s1[i] : 0x20;
+ if (! caseSensitive && 'a' <= c1 && c1 <= 'z')
+ c1 -= 'a' - 'A';
+ c2 = i < n2 ? s2[i] : 0x20;
+ if (! caseSensitive && 'a' <= c2 && c2 <= 'z')
+ c2 -= 'a' - 'A';
+ if (c1 != c2)
+ break;
+ i++;
+ }
+ return c1 == c2;
+}
+
+// columns and tables
+
+struct Col {
+ enum Type {
+ Char = SQL_CHAR,
+ Varchar = SQL_VARCHAR,
+ Int = SQL_INTEGER,
+ Bigint = SQL_BIGINT,
+ Real = SQL_REAL,
+ Double = SQL_DOUBLE
+ };
+ enum CType {
+ CChar = SQL_C_CHAR,
+ CLong = SQL_C_SLONG,
+ CDouble = SQL_C_DOUBLE
+ };
+ enum Cons {
+ Null, // nullable
+ NotNull, // not nullable
+ Pk // part of primary key
+ };
+ const char* m_name;
+ Type m_type;
+ unsigned m_length;
+ Cons m_cons;
+ CType m_ctype;
+ Col() :
+ m_type((Type)999) {
+ }
+ Col(const char* name, Type type, unsigned length, Cons cons, CType ctype) :
+ m_name(name),
+ m_type(type),
+ m_length(length),
+ m_cons(cons),
+ m_ctype(ctype) {
+ }
+ unsigned size() const {
+ switch (m_type) {
+ case Char:
+ case Varchar:
+ return m_length;
+ case Int:
+ return 4;
+ case Bigint:
+ return 8;
+ case Real:
+ return 4;
+ case Double:
+ return 8;
+ }
+ assert(false);
+ return 0;
+ }
+ unsigned csize() const { // size as char plus terminating null
+ switch (m_ctype) {
+ case CChar:
+ return m_length + 1;
+ case CLong:
+ return 12;
+ case CDouble:
+ return 24;
+ }
+ assert(false);
+ return 0;
+ }
+ void typespec(char*& ptr) const {
+ switch (m_type) {
+ case Char:
+ scopy(ptr, "char(%d)", m_length);
+ return;
+ case Varchar:
+ scopy(ptr, "varchar(%d)", m_length);
+ return;
+ case Int:
+ scopy(ptr, "int");
+ return;
+ case Bigint:
+ scopy(ptr, "bigint");
+ return;
+ case Real:
+ scopy(ptr, "real");
+ return;
+ case Double:
+ scopy(ptr, "float");
+ return;
+ }
+ assert(false);
+ }
+ SQLSMALLINT type() const {
+ return (SQLSMALLINT)m_type;
+ }
+ SQLSMALLINT ctype() const {
+ return (SQLSMALLINT)m_ctype;
+ }
+ void create(char*& ptr, bool pk) const {
+ scopy(ptr, "%s", m_name);
+ scopy(ptr, " ");
+ typespec(ptr);
+ if (m_cons == Pk && pk) {
+ scopy(ptr, " primary key");
+ }
+ if (m_cons == NotNull) {
+ scopy(ptr, " not null");
+ }
+ }
+};
+
+static Col ColUndef;
+
+struct Tab {
+ const char* m_name;
+ const Col* m_colList;
+ unsigned m_colCount;
+ unsigned m_pkCount;
+ unsigned* m_pkIndex;
+ unsigned m_nkCount;
+ unsigned* m_nkIndex;
+ char m_upperName[20];
+ Tab(const char* name, const Col* colList, unsigned colCount) :
+ m_name(name),
+ m_colList(colList),
+ m_colCount(colCount) {
+ m_pkCount = 0;
+ m_nkCount = 0;
+ for (unsigned i = 0; i < m_colCount; i++) {
+ const Col& col = m_colList[i];
+ if (col.m_cons == Col::Pk)
+ m_pkCount++;
+ else
+ m_nkCount++;
+ }
+ m_pkIndex = new unsigned[m_pkCount];
+ m_nkIndex = new unsigned[m_nkCount];
+ unsigned pk = 0;
+ unsigned nk = 0;
+ for (unsigned i = 0; i < m_colCount; i++) {
+ const Col& col = m_colList[i];
+ if (col.m_cons == Col::Pk)
+ m_pkIndex[pk++] = i;
+ else
+ m_nkIndex[nk++] = i;
+ }
+ assert(pk == m_pkCount && nk == m_nkCount);
+ strcpy(m_upperName, m_name);
+ for (char* p = m_upperName; *p != 0; p++) {
+ if ('a' <= *p && *p <= 'z')
+ *p -= 'a' - 'A';
+ }
+ }
+ ~Tab() {
+ delete[] m_pkIndex;
+ delete[] m_nkIndex;
+ }
+ void drop(char*& ptr) const {
+ scopy(ptr, "drop table %s", m_name);
+ }
+ void create(char*& ptr) const {
+ scopy(ptr, "create table %s (", m_name);
+ for (unsigned i = 0; i < m_colCount; i++) {
+ if (i > 0)
+ scopy(ptr, ", ");
+ const Col& col = m_colList[i];
+ col.create(ptr, m_pkCount == 1);
+ }
+ if (m_pkCount != 1) {
+ scopy(ptr, ", primary key (");
+ for (unsigned i = 0; i < m_pkCount; i++) {
+ const Col& col = m_colList[m_pkIndex[i]];
+ if (i > 0)
+ scopy(ptr, ", ");
+ scopy(ptr, "%s", col.m_name);
+ }
+ scopy(ptr, ")");
+ }
+ scopy(ptr, ")");
+ }
+ void wherePk(char*& ptr) const {
+ scopy(ptr, " where");
+ for (unsigned i = 0; i < m_pkCount; i++) {
+ const Col& col = m_colList[m_pkIndex[i]];
+ if (i > 0)
+ scopy(ptr, " and");
+ scopy(ptr, " %s = ?", col.m_name);
+ }
+ }
+ void whereRange(char*& ptr) const {
+ scopy(ptr, " where");
+ for (unsigned i = 0; i < m_pkCount; i++) {
+ const Col& col = m_colList[m_pkIndex[i]];
+ if (i > 0)
+ scopy(ptr, " and");
+ scopy(ptr, " ? <= %s", col.m_name);
+ scopy(ptr, " and ");
+ scopy(ptr, "%s < ?", col.m_name);
+ }
+ }
+ void orderPk(char*& ptr) const {
+ scopy(ptr, " order by");
+ for (unsigned i = 0; i < m_pkCount; i++) {
+ const Col& col = m_colList[m_pkIndex[i]];
+ if (i > 0)
+ scopy(ptr, ", ");
+ else
+ scopy(ptr, " ");
+ scopy(ptr, "%s", col.m_name);
+ }
+ }
+ void selectPk(char*& ptr) const {
+ scopy(ptr, "select * from %s", m_name);
+ wherePk(ptr);
+ }
+ void selectAll(char*& ptr) const {
+ scopy(ptr, "select * from %s", m_name);
+ }
+ void selectRange(char*& ptr, bool sort) const {
+ selectAll(ptr);
+ whereRange(ptr);
+ if (sort)
+ orderPk(ptr);
+ }
+ void selectCount(char*& ptr) const {
+ scopy(ptr, "select count(*) from %s", m_name);
+ }
+ void insertAll(char*& ptr) const {
+ scopy(ptr, "insert into %s values (", m_name);
+ for (unsigned i = 0; i < m_colCount; i++) {
+ if (i > 0)
+ scopy(ptr, ", ");
+ scopy(ptr, "?");
+ }
+ scopy(ptr, ")");
+ }
+ void updatePk(char*& ptr) const {
+ scopy(ptr, "update %s set", m_name);
+ for (unsigned i = 0; i < m_nkCount; i++) {
+ const Col& col = m_colList[m_nkIndex[i]];
+ if (i > 0)
+ scopy(ptr, ", ");
+ else
+ scopy(ptr, " ");
+ scopy(ptr, "%s = ?", col.m_name);
+ }
+ wherePk(ptr);
+ }
+ void updateRange(char*& ptr) const {
+ scopy(ptr, "update %s set", m_name);
+ for (unsigned i = 0; i < m_nkCount; i++) {
+ const Col& col = m_colList[m_nkIndex[i]];
+ if (i > 0)
+ scopy(ptr, ", ");
+ else
+ scopy(ptr, " ");
+ scopy(ptr, "%s = ?", col.m_name); // XXX constant for now
+ }
+ whereRange(ptr);
+ }
+ void deleteAll(char*& ptr) const {
+ scopy(ptr, "delete from %s", m_name);
+ }
+ void deletePk(char*& ptr) const {
+ scopy(ptr, "delete from %s", m_name);
+ wherePk(ptr);
+ }
+ void deleteRange(char*& ptr) const {
+ scopy(ptr, "delete from %s", m_name);
+ whereRange(ptr);
+ }
+ // simple
+ void insertDirect(char*& ptr, unsigned n) const {
+ scopy(ptr, "insert into %s values (", m_name);
+ for (unsigned i = 0; i < m_colCount; i++) {
+ const Col& col = m_colList[i];
+ if (i > 0)
+ scopy(ptr, ", ");
+ if (col.m_type == Col::Char || col.m_type == Col::Varchar) {
+ scopy(ptr, "'");
+ for (unsigned i = 0; i <= n % col.m_length; i++)
+ scopy(ptr, "%c", 'a' + (n + i) % 26);
+ scopy(ptr, "'");
+ } else if (col.m_type == Col::Int || col.m_type == Col::Bigint) {
+ scopy(ptr, "%u", n);
+ } else if (col.m_type == Col::Real || col.m_type == Col::Double) {
+ scopy(ptr, "%.3f", n * 0.001);
+ } else {
+ assert(false);
+ }
+ }
+ scopy(ptr, ")");
+ }
+ void whereDirect(char*& ptr, unsigned n) const {
+ scopy(ptr, " where");
+ for (unsigned i = 0; i < m_pkCount; i++) {
+ const Col& col = m_colList[m_pkIndex[i]];
+ if (i > 0)
+ scopy(ptr, ", ");
+ else
+ scopy(ptr, " ");
+ scopy(ptr, "%s = ", col.m_name);
+ if (col.m_type == Col::Char || col.m_type == Col::Varchar) {
+ scopy(ptr, "'");
+ for (unsigned i = 0; i <= n % col.m_length; i++)
+ scopy(ptr, "%c", 'a' + (n + i) % 26);
+ scopy(ptr, "'");
+ } else if (col.m_type == Col::Int || col.m_type == Col::Bigint) {
+ scopy(ptr, "%u", n);
+ } else {
+ assert(false);
+ }
+ }
+ }
+ void countDirect(char*& ptr, unsigned n) const {
+ scopy(ptr, "select count(*) from %s", m_name);
+ whereDirect(ptr, n);
+ }
+ void deleteDirect(char*& ptr, unsigned n) const {
+ scopy(ptr, "delete from %s", m_name);
+ whereDirect(ptr, n);
+ }
+ // joins
+ void selectCart(char*& ptr, unsigned cnt) const {
+ scopy(ptr, "select count(*) from");
+ for (unsigned j = 0; j < cnt; j++) {
+ if (j > 0)
+ scopy(ptr, ",");
+ scopy(ptr, " %s", m_name);
+ scopy(ptr, " t%u", j);
+ }
+ }
+ void selectJoin(char*& ptr, unsigned cnt) const {
+ scopy(ptr, "select * from");
+ for (unsigned j = 0; j < cnt; j++) {
+ if (j > 0)
+ scopy(ptr, ",");
+ scopy(ptr, " %s", m_name);
+ scopy(ptr, " t%u", j);
+ }
+ for (unsigned i = 0; i < m_pkCount; i++) {
+ const Col& col = m_colList[m_pkIndex[i]];
+ for (unsigned j = 0; j < cnt - 1; j++) {
+ if (i == 0 && j == 0)
+ scopy(ptr, " where");
+ else
+ scopy(ptr, " and");
+ scopy(ptr, " t%u.%s = t%u.%s", j, col.m_name, j + 1, col.m_name);
+ }
+ }
+ }
+ // check if selected on command line
+ bool optok() const {
+ return opt.m_table == 0 || strcasecmp(m_name, opt.m_table) == 0;
+ }
+};
+
+// the test tables
+
+static Col col0[] = {
+ Col( "a", Col::Bigint, 0, Col::Pk, Col::CLong ),
+ Col( "b", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c", Col::Char, 4, Col::NotNull, Col::CChar ),
+ Col( "d", Col::Double, 0, Col::Null, Col::CDouble ),
+ Col( "e", Col::Char, 40, Col::Null, Col::CChar ),
+ Col( "f", Col::Char, 10, Col::Null, Col::CChar )
+};
+
+static Col col1[] = {
+ Col( "c0", Col::Int, 0, Col::Pk, Col::CLong ),
+ Col( "c1", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c2", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c3", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c4", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c5", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c6", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c7", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c8", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c9", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c10", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c11", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c12", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c13", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c14", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c15", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c16", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c17", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c18", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c19", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c20", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c21", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c22", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c23", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c24", Col::Int, 0, Col::NotNull, Col::CLong ),
+ Col( "c25", Col::Int, 0, Col::NotNull, Col::CLong )
+};
+
+static Col col2[] = {
+ Col( "a", Col::Int, 0, Col::Pk, Col::CLong ),
+ Col( "c", Col::Char, 8000, Col::NotNull, Col::CChar )
+};
+
+static Col col3[] = {
+ Col( "a", Col::Int, 0, Col::Pk, Col::CLong ),
+ Col( "c1", Col::Varchar, 1, Col::Null, Col::CChar ),
+ Col( "c2", Col::Varchar, 2, Col::Null, Col::CChar ),
+ Col( "c3", Col::Varchar, 3, Col::Null, Col::CChar ),
+ Col( "c4", Col::Varchar, 4, Col::Null, Col::CChar ),
+ Col( "c5", Col::Varchar, 10, Col::Null, Col::CChar ),
+ Col( "c6", Col::Varchar, 40, Col::Null, Col::CChar ),
+ Col( "c7", Col::Varchar, 255, Col::Null, Col::CChar ),
+ Col( "c8", Col::Varchar, 4000, Col::Null, Col::CChar )
+};
+
+static Col col4[] = {
+ Col( "a", Col::Char, 8, Col::Pk, Col::CChar ),
+ Col( "b", Col::Char, 8, Col::NotNull, Col::CChar ),
+};
+
+static Tab tabList[] = {
+#define colList(x) x, arraySize(x)
+ Tab( "tt00", colList(col0) ),
+ Tab( "tt01", colList(col1) ), // fläskbench special
+ Tab( "tt02", colList(col2) ),
+ Tab( "tt03", colList(col3) ),
+ Tab( "tt04", colList(col4) )
+#undef colList
+};
+
+static const unsigned tabCount = arraySize(tabList);
+static const unsigned maxColCount = 100; // per table - keep up to date
+
+static bool
+findTable()
+{
+ for (unsigned i = 0; i < tabCount; i++) {
+ const Tab& tab = tabList[i];
+ if (tab.optok())
+ return true;
+ }
+ return false;
+}
+
+static void
+listTables()
+{
+ ndbout << "tables:" << endl;
+ for (unsigned i = 0; i < tabCount; i++) {
+ const Tab& tab = tabList[i];
+ if (i > 0)
+ ndbout << " ";
+ ndbout << tab.m_name;
+ }
+ ndbout << endl;
+}
+
+// data fields and rows
+
+struct Fld {
+ const Col& m_col;
+ union {
+ char* m_char;
+ long m_long;
+ double m_double;
+ };
+ SQLINTEGER m_ind;
+ SQLINTEGER m_need; // constant
+ Fld() :
+ m_col(ColUndef),
+ m_need(0) {
+ }
+ Fld(const Col& col) :
+ m_col(col),
+ m_need(SQL_LEN_DATA_AT_EXEC(0)) {
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ m_char = new char[m_col.csize()];
+ memset(m_char, 0, m_col.csize());
+ break;
+ case Col::CLong:
+ m_long = 0;
+ break;
+ case Col::CDouble:
+ m_double = 0.0;
+ break;
+ }
+ m_ind = -1;
+ }
+ ~Fld() {
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ delete[] m_char;
+ break;
+ case Col::CLong:
+ break;
+ case Col::CDouble:
+ break;
+ }
+ }
+ void zero() {
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ memset(m_char, 0x1f, m_col.csize());
+ break;
+ case Col::CLong:
+ m_long = 0x1f1f1f1f;
+ break;
+ case Col::CDouble:
+ m_double = 1111111.1111111;
+ break;
+ }
+ m_ind = -1;
+ }
+ // copy values from another field
+ void copy(const Fld& fld) {
+ assert(&m_col == &fld.m_col);
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ memcpy(m_char, fld.m_char, m_col.csize());
+ break;
+ case Col::CLong:
+ m_long = fld.m_long;
+ break;
+ case Col::CDouble:
+ m_double = fld.m_double;
+ break;
+ default:
+ assert(false);
+ break;
+ }
+ m_ind = fld.m_ind;
+ }
+ SQLPOINTER caddr() {
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ return (SQLPOINTER)m_char;
+ case Col::CLong:
+ return (SQLPOINTER)&m_long;
+ case Col::CDouble:
+ return (SQLPOINTER)&m_double;
+ }
+ assert(false);
+ return 0;
+ }
+ SQLINTEGER* ind() {
+ return &m_ind;
+ }
+ SQLINTEGER* need() {
+ m_need = SQL_LEN_DATA_AT_EXEC(0);
+ return &m_need;
+ }
+ void calcPk(const Test& test, unsigned rownum) {
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ test.calcPk(rownum, m_char, m_col.csize());
+ m_ind = SQL_NTS;
+ return;
+ case Col::CLong:
+ test.calcPk(rownum, &m_long);
+ m_ind = 0;
+ return;
+ case Col::CDouble:
+ assert(false);
+ return;
+ }
+ assert(false);
+ }
+ void hashPk(const Test& test, unsigned* hash) const {
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ test.hashPk(hash, m_char, m_col.csize());
+ return;
+ case Col::CLong:
+ test.hashPk(hash, m_long);
+ return;
+ case Col::CDouble:
+ assert(false);
+ return;
+ }
+ assert(false);
+ }
+ void calcNk(const Test& test, unsigned hash) {
+ bool null = m_col.m_cons == Col::Null;
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ test.calcNk(hash, m_char, m_col.csize(), &m_ind, null);
+ return;
+ case Col::CLong:
+ test.calcNk(hash, &m_long, &m_ind, null);
+ return;
+ case Col::CDouble:
+ test.calcNk(hash, &m_double, &m_ind, null);
+ return;
+ }
+ assert(false);
+ }
+ bool verify(Test& test, const Fld& fld) {
+ assert(&m_col == &fld.m_col);
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ return test.verify(m_char, m_ind, fld.m_char, fld.m_ind, m_col.csize());
+ case Col::CLong:
+ return test.verify(m_long, m_ind, fld.m_long, fld.m_ind);
+ case Col::CDouble:
+ return test.verify(m_double, m_ind, fld.m_double, fld.m_ind);
+ }
+ assert(false);
+ }
+ // debug
+ void print() const {
+ if (m_ind == SQL_NULL_DATA)
+ ndbout << "NULL";
+ else {
+ switch (m_col.m_ctype) {
+ case Col::CChar:
+ ndbout << m_char;
+ break;
+ case Col::CLong:
+ ndbout << (int)m_long;
+ break;
+ case Col::CDouble:
+ ndbout << m_double;
+ break;
+ }
+ }
+ }
+};
+
+struct Row {
+ const Tab& m_tab;
+ Fld* m_fldList;
+ Row(const Tab& tab) :
+ m_tab(tab) {
+ m_fldList = new Fld[m_tab.m_colCount];
+ for (unsigned i = 0; i < m_tab.m_colCount; i++) {
+ const Col& col = m_tab.m_colList[i];
+ void* place = &m_fldList[i];
+ new (place) Fld(col);
+ }
+ }
+ ~Row() {
+ delete[] m_fldList;
+ }
+ // copy values from another row
+ void copy(const Row& row) {
+ assert(&m_tab == &row.m_tab);
+ for (unsigned i = 0; i < m_tab.m_colCount; i++) {
+ Fld& fld = m_fldList[i];
+ fld.copy(row.m_fldList[i]);
+ }
+ }
+ // primary key value is determined by row number
+ void calcPk(Test& test, unsigned rownum) {
+ for (unsigned i = 0; i < m_tab.m_pkCount; i++) {
+ Fld& fld = m_fldList[m_tab.m_pkIndex[i]];
+ fld.calcPk(test, rownum);
+ }
+ }
+ // other fields are determined by primary key value
+ void calcNk(Test& test) {
+ unsigned hash = test.m_salt;
+ for (unsigned i = 0; i < m_tab.m_pkCount; i++) {
+ Fld& fld = m_fldList[m_tab.m_pkIndex[i]];
+ fld.hashPk(test, &hash);
+ }
+ for (unsigned i = 0; i < m_tab.m_colCount; i++) {
+ const Col& col = m_tab.m_colList[i];
+ if (col.m_cons == Col::Pk)
+ continue;
+ Fld& fld = m_fldList[i];
+ fld.calcNk(test, hash);
+ }
+ }
+ // verify against another row
+ bool verifyPk(Test& test, const Row& row) const {
+ assert(&m_tab == &row.m_tab);
+ for (unsigned i = 0; i < m_tab.m_pkCount; i++) {
+ Fld& fld = m_fldList[m_tab.m_pkIndex[i]];
+ if (! fld.verify(test, row.m_fldList[m_tab.m_pkIndex[i]])) {
+ ndbout << "verify failed: tab=" << m_tab.m_name << " col=" << fld.m_col.m_name << endl;
+ return false;
+ }
+ }
+ return true;
+ }
+ bool verifyNk(Test& test, const Row& row) const {
+ assert(&m_tab == &row.m_tab);
+ for (unsigned i = 0; i < m_tab.m_nkCount; i++) {
+ Fld& fld = m_fldList[m_tab.m_nkIndex[i]];
+ if (! fld.verify(test, row.m_fldList[m_tab.m_nkIndex[i]])) {
+ ndbout << "verify failed: tab=" << m_tab.m_name << " col=" << fld.m_col.m_name << endl;
+ return false;
+ }
+ }
+ return true;
+ }
+ bool verify(Test& test, const Row& row) const {
+ return verifyPk(test, row) && verifyNk(test, row);
+ }
+ // debug
+ void print() const {
+ ndbout << "row";
+ for (unsigned i = 0; i < m_tab.m_colCount; i++) {
+ ndbout << " " << i << "=";
+ Fld& fld = m_fldList[i];
+ fld.print();
+ }
+ ndbout << endl;
+ }
+};
+
+// set ODBC version - required
+
+static void
+setVersion(Test& test, SQLHANDLE hEnv)
+{
+ test.run(HEnv(hEnv), SQLSetEnvAttr(hEnv, SQL_ATTR_ODBC_VERSION, (SQLPOINTER)SQL_OV_ODBC3, SQL_IS_INTEGER));
+}
+
+// set autocommit
+
+static void
+setAutocommit(Test& test, SQLHANDLE hDbc, bool on)
+{
+ SQLUINTEGER value = on ? SQL_AUTOCOMMIT_ON : SQL_AUTOCOMMIT_OFF;
+ test.run(HDbc(hDbc), SQLSetConnectAttr(hDbc, SQL_ATTR_AUTOCOMMIT, (SQLPOINTER)value, SQL_IS_UINTEGER));
+ SQLUINTEGER value2 = (SQLUINTEGER)-1;
+ test.run(HDbc(hDbc), SQLGetConnectAttr(hDbc, SQL_ATTR_AUTOCOMMIT, (SQLPOINTER)&value2, SQL_IS_UINTEGER, 0));
+ test.chk(HDbc(hDbc), value2 == value, "got %u != %u", (unsigned)value2, (unsigned)value);
+}
+
+// subroutines - migrate simple common routines here
+
+static void
+allocEnv(Test& test, SQLHANDLE& hEnv)
+{
+ test.run(HNull, SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &hEnv));
+ setVersion(test, hEnv);
+}
+
+static void
+allocDbc(Test& test, SQLHANDLE hEnv, SQLHANDLE& hDbc)
+{
+ test.run(HEnv(hEnv), SQLAllocHandle(SQL_HANDLE_DBC, hEnv, &hDbc));
+}
+
+static void
+allocConn(Test& test, SQLHANDLE hEnv, SQLHANDLE& hDbc)
+{
+ allocDbc(test, hEnv, hDbc);
+#ifdef unixODBC
+ test.exp(SQL_SUCCESS_WITH_INFO, "IM003", 0, false); // unicode??
+ test.exp(SQL_SUCCESS_WITH_INFO, "01000", 0, false); // version??
+#endif
+ test.run(HDbc(hDbc), SQLConnect(hDbc, (SQLCHAR*)opt.m_dsn, SQL_NTS, (SQLCHAR*)"user", SQL_NTS, (SQLCHAR*)"pass", SQL_NTS));
+}
+
+static void
+allocStmt(Test& test, SQLHANDLE hDbc, SQLHANDLE& hStmt)
+{
+ test.run(HDbc(hDbc), SQLAllocHandle(SQL_HANDLE_STMT, hDbc, &hStmt));
+}
+
+static void
+allocAll(Test& test, SQLHANDLE& hEnv, SQLHANDLE& hDbc, SQLHANDLE& hStmt)
+{
+ allocEnv(test, hEnv);
+ allocConn(test, hEnv, hDbc);
+ allocStmt(test, hDbc, hStmt);
+}
+
+static void
+allocAll(Test& test, SQLHANDLE& hEnv, SQLHANDLE& hDbc, SQLHANDLE* hStmtList, unsigned nStmt)
+{
+ allocEnv(test, hEnv);
+ allocConn(test, hEnv, hDbc);
+ for (unsigned i = 0; i < nStmt; i++)
+ allocStmt(test, hDbc, hStmtList[i]);
+}
+
+static void
+freeEnv(Test& test, SQLHANDLE hEnv)
+{
+ test.run(HNull, SQLFreeHandle(SQL_HANDLE_ENV, hEnv));
+}
+
+static void
+freeDbc(Test& test, SQLHANDLE hEnv, SQLHANDLE hDbc)
+{
+ test.run(HEnv(hEnv), SQLFreeHandle(SQL_HANDLE_DBC, hDbc));
+}
+
+static void
+freeConn(Test& test, SQLHANDLE hEnv, SQLHANDLE hDbc)
+{
+ test.run(HDbc(hDbc), SQLDisconnect(hDbc));
+ test.run(HEnv(hEnv), SQLFreeHandle(SQL_HANDLE_DBC, hDbc));
+}
+
+static void
+freeStmt(Test& test, SQLHANDLE hDbc, SQLHANDLE hStmt)
+{
+ test.run(HDbc(hDbc), SQLFreeHandle(SQL_HANDLE_STMT, hStmt));
+}
+
+static void
+freeAll(Test& test, SQLHANDLE hEnv, SQLHANDLE hDbc, SQLHANDLE hStmt)
+{
+ freeStmt(test, hDbc, hStmt);
+ freeConn(test, hEnv, hDbc);
+ freeEnv(test, hEnv);
+}
+
+static void
+freeAll(Test& test, SQLHANDLE hEnv, SQLHANDLE hDbc, SQLHANDLE* hStmtList, unsigned nStmt)
+{
+ for (unsigned i = 0; i < nStmt; i++)
+ freeStmt(test, hDbc, hStmtList[i]);
+ freeConn(test, hEnv, hDbc);
+ freeEnv(test, hEnv);
+}
+
+#define chkTuplesFetched(/*Test&*/ _test, /*SQLHANDLE*/ _hStmt, /*SQLUINTEGER*/ _countExp) \
+do { \
+ SQLUINTEGER _count = (SQLUINTEGER)-1; \
+ getTuplesFetched(_test, _hStmt, &_count); \
+ test.chk(HStmt(_hStmt), _count == _countExp, "tuples: got %ld != %ld", (long)_count, (long)_countExp); \
+} while (0)
+
+static void
+getTuplesFetched(Test& test, SQLHANDLE hStmt, SQLUINTEGER* count)
+{
+ *count = (SQLUINTEGER)-1;
+ test.run(HStmt(hStmt), SQLGetStmtAttr(hStmt, SQL_ATTR_NDB_TUPLES_FETCHED, count, SQL_IS_POINTER, 0));
+}
+
+static void
+selectCount(Test& test, SQLHANDLE hStmt, const char* sql, long* count)
+{
+ if (opt.m_v >= 3)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ SQLINTEGER ind;
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1, SQL_C_SLONG, count, 0, &ind));
+ ind = -1;
+ *count = -1;
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ unsigned k = 0;
+ while (1) {
+ if (k == 1)
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (k == 1)
+ break;
+ k++;
+ }
+ test.chk(HStmt(hStmt), ind == sizeof(long), "got %d != %d", (int)ind, (int)sizeof(long));
+ test.chk(HStmt(hStmt), *count >= 0, "got %ld < 0", *count);
+ chkTuplesFetched(test, hStmt, *count);
+#ifndef iODBC
+ //
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+#else
+ test.run(HStmt(hStmt), SQLFreeStmt(hStmt, SQL_CLOSE));
+#endif
+}
+
+static void
+selectCount(Test& test, SQLHANDLE hStmt, const Tab& tab, long* count)
+{
+ static char sql[MAX_SQL], *sqlptr; // XXX static or core
+ tab.selectCount(sqlptr = sql);
+ selectCount(test, hStmt, sql, count);
+}
+
+static void
+verifyCount(Test& test, SQLHANDLE hStmt, const Tab& tab, long countExp)
+{
+ long count = -1;
+ selectCount(test, hStmt, tab, &count);
+ test.chk(HStmt(hStmt), count == countExp, "got %ld != %ld", count, countExp);
+}
+
+#define chkRowCount(/*Test&*/ _test, /*SQLHANDLE*/ _hStmt, /*SQLINTEGER*/ _countExp) \
+do { \
+ SQLINTEGER _count = -1; \
+ getRowCount(_test, _hStmt, &_count); \
+ test.chk(HStmt(_hStmt), _count == _countExp, "rowcount: got %ld != %ld", (long)_count, (long)_countExp); \
+} while (0)
+
+static void
+getRowCount(Test& test, SQLHANDLE hStmt, SQLINTEGER* count)
+{
+ *count = -1;
+ test.run(HStmt(hStmt), SQLRowCount(hStmt, count));
+}
+
+// handle allocation
+
+static void
+testAlloc(Test& test)
+{
+ const unsigned n1 = (opt.m_scale >> 8) & 0xf; // default 500 = 0x1f4
+ const unsigned n2 = (opt.m_scale >> 4) & 0xf;
+ const unsigned n3 = (opt.m_scale >> 0) & 0xf;
+ const unsigned count = n1 + n1 * n2 + n1 * n2 * n3;
+ SQLHANDLE hEnvList[0xf];
+ SQLHANDLE hDbcList[0xf][0xf];
+ SQLHANDLE hStmtList[0xf][0xf][0xf];
+ for (unsigned i1 = 0; i1 < n1; i1++) {
+ SQLHANDLE& hEnv = hEnvList[i1];
+ test.run(HNull, SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &hEnv));
+ test.run(HEnv(hEnv), SQLSetEnvAttr(hEnv, SQL_ATTR_ODBC_VERSION, (SQLPOINTER)SQL_OV_ODBC3, SQL_IS_INTEGER));
+ for (unsigned i2 = 0; i2 < n2; i2++) {
+ SQLHANDLE& hDbc = hDbcList[i1][i2];
+ test.run(HEnv(hEnv), SQLAllocHandle(SQL_HANDLE_DBC, hEnv, &hDbc));
+#ifdef unixODBC
+ test.exp(SQL_SUCCESS_WITH_INFO, "IM003", 0, false); // unicode??
+ test.exp(SQL_SUCCESS_WITH_INFO, "01000", 0, false); // version??
+#endif
+ test.run(HDbc(hDbc), SQLConnect(hDbc, (SQLCHAR*)opt.m_dsn, SQL_NTS, (SQLCHAR*)"user", SQL_NTS, (SQLCHAR*)"pass", SQL_NTS));
+ // some attributes
+ test.exp(SQL_ERROR, "HY092", -1, true); // read-only attribute
+ test.run(HDbc(hDbc), SQLSetConnectAttr(hDbc, SQL_ATTR_AUTO_IPD, (SQLPOINTER)SQL_TRUE, SQL_IS_UINTEGER));
+ test.exp(SQL_ERROR, "HYC00", -1, true); // not supported
+ test.run(HDbc(hDbc), SQLSetConnectAttr(hDbc, SQL_ATTR_TXN_ISOLATION, (SQLPOINTER)SQL_TXN_SERIALIZABLE, SQL_IS_UINTEGER));
+ test.run(HDbc(hDbc), SQLSetConnectAttr(hDbc, SQL_ATTR_CURRENT_CATALOG, (SQLPOINTER)"DEFAULT", strlen("DEFAULT")));
+ for (unsigned i3 = 0; i3 < n3; i3++) {
+ SQLHANDLE& hStmt = hStmtList[i1][i2][i3];
+ test.run(HDbc(hDbc), SQLAllocHandle(SQL_HANDLE_STMT, hDbc, &hStmt));
+ SQLHANDLE ipd0, ipd1;
+ SQLHANDLE ird0, ird1;
+ SQLHANDLE apd0, apd1, apd2;
+ SQLHANDLE ard0, ard1, ard2;
+ // get
+ ipd0 = ird0 = apd0 = ard0 = 0;
+ test.run(HStmt(hStmt), SQLGetStmtAttr(hStmt, SQL_ATTR_IMP_PARAM_DESC, &ipd0, SQL_IS_POINTER, 0));
+ test.run(HStmt(hStmt), SQLGetStmtAttr(hStmt, SQL_ATTR_IMP_ROW_DESC, &ird0, SQL_IS_POINTER, 0));
+ test.run(HStmt(hStmt), SQLGetStmtAttr(hStmt, SQL_ATTR_APP_PARAM_DESC, &apd0, SQL_IS_POINTER, 0));
+ test.run(HStmt(hStmt), SQLGetStmtAttr(hStmt, SQL_ATTR_APP_ROW_DESC, &ard0, SQL_IS_POINTER, 0));
+#ifndef unixODBC
+ test.chk(HStmt(hStmt), ipd0 != 0, "got 0");
+ test.chk(HStmt(hStmt), ird0 != 0, "got 0");
+ test.chk(HStmt(hStmt), apd0 != 0, "got 0");
+ test.chk(HStmt(hStmt), ard0 != 0, "got 0");
+#endif
+ // alloc
+ ipd1 = ird1 = apd1 = ard1 = 0;
+ test.run(HDbc(hDbc), SQLAllocHandle(SQL_HANDLE_DESC, hDbc, &ipd1));
+ test.run(HDbc(hDbc), SQLAllocHandle(SQL_HANDLE_DESC, hDbc, &ird1));
+ test.run(HDbc(hDbc), SQLAllocHandle(SQL_HANDLE_DESC, hDbc, &apd1));
+ test.run(HDbc(hDbc), SQLAllocHandle(SQL_HANDLE_DESC, hDbc, &ard1));
+ test.chk(HDbc(hDbc), ipd1 != 0 && ird1 != 0 && apd1 != 0 && ard1 != 0, "got null");
+ // set
+ test.exp(SQL_ERROR, "HY092", -1, true); // read-only attribute
+ test.run(HStmt(hStmt), SQLSetStmtAttr(hStmt, SQL_ATTR_IMP_PARAM_DESC, ipd1, SQL_IS_POINTER));
+ test.exp(SQL_ERROR, "HY092", -1, true); // read-only attribute
+ test.run(HStmt(hStmt), SQLSetStmtAttr(hStmt, SQL_ATTR_IMP_ROW_DESC, ird1, SQL_IS_POINTER));
+ test.run(HStmt(hStmt), SQLSetStmtAttr(hStmt, SQL_ATTR_APP_PARAM_DESC, apd1, SQL_IS_POINTER));
+ test.run(HStmt(hStmt), SQLSetStmtAttr(hStmt, SQL_ATTR_APP_ROW_DESC, ard1, SQL_IS_POINTER));
+ // get
+
+ apd2 = ard2 = 0;
+ test.run(HStmt(hStmt), SQLGetStmtAttr(hStmt, SQL_ATTR_APP_PARAM_DESC, &apd2, SQL_IS_POINTER, 0));
+ test.run(HStmt(hStmt), SQLGetStmtAttr(hStmt, SQL_ATTR_APP_ROW_DESC, &ard2, SQL_IS_POINTER, 0));
+ test.chk(HStmt(hStmt), apd2 == apd1, "got %x != %x", (unsigned)apd2, (unsigned)apd1);
+ test.chk(HStmt(hStmt), ard2 == ard1, "got %x != %x", (unsigned)ard2, (unsigned)ard1);
+ // free
+ test.run(HDbc(hDbc), SQLFreeHandle(SQL_HANDLE_DESC, ipd1));
+ test.run(HDbc(hDbc), SQLFreeHandle(SQL_HANDLE_DESC, ird1));
+ test.run(HDbc(hDbc), SQLFreeHandle(SQL_HANDLE_DESC, apd1));
+ test.run(HDbc(hDbc), SQLFreeHandle(SQL_HANDLE_DESC, ard1));
+ }
+ }
+ }
+ test.timerCnt(count);
+ if (opt.m_v >= 3)
+ ndbout << "allocated " << count << endl;
+ for (unsigned i1 = 0; i1 < n1; i1++) {
+ SQLHANDLE& hEnv = hEnvList[i1];
+ for (unsigned i2 = 0; i2 < n2; i2++) {
+ SQLHANDLE& hDbc = hDbcList[i1][i2];
+ if (i2 % 2 == 0) {
+ for (unsigned i3 = 0; i3 < n3; i3++) {
+ SQLHANDLE& hStmt = hStmtList[i1][i2][i3];
+ test.run(HDbc(hDbc), SQLFreeHandle(SQL_HANDLE_STMT, hStmt));
+ }
+ } else {
+ // cleaned up by SQLDisconnect
+ }
+ test.run(HDbc(hDbc), SQLDisconnect(hDbc));
+ test.run(HEnv(hEnv), SQLFreeHandle(SQL_HANDLE_DBC, hDbc));
+ }
+ test.run(HNull, SQLFreeHandle(SQL_HANDLE_ENV, hEnv));
+ }
+ test.timerCnt(count);
+ if (opt.m_v >= 3)
+ ndbout << "freed " << count << endl;
+}
+
+// create tables
+
+static void
+testCreate(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ // drop
+ tab.drop(sqlptr = sql);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ test.exp(SQL_ERROR, "IM000", 2040709, false);
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ if (test.m_ret == SQL_SUCCESS)
+ test.chk(HStmt(hStmt), test.m_functionCode == SQL_DIAG_DROP_TABLE, "got %d != %d", test.m_functionCode, SQL_DIAG_DROP_TABLE);
+ if (test.m_ret == SQL_SUCCESS && opt.m_v >= 2)
+ ndbout << "table " << tab.m_name << " dropped" << endl;
+ if (test.m_ret != SQL_SUCCESS && opt.m_v >= 2)
+ ndbout << "table " << tab.m_name << " does not exist" << endl;
+ test.timerCnt(1);
+ // create
+ tab.create(sqlptr = sql);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ test.exp(SQL_ERROR, "IM000", 2040721, false);
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ if (test.m_ret == SQL_SUCCESS)
+ test.chk(HStmt(hStmt), test.m_functionCode == SQL_DIAG_CREATE_TABLE, "got %d != %d", test.m_functionCode, SQL_DIAG_CREATE_TABLE);
+ if (test.m_ret == SQL_SUCCESS && opt.m_v >= 2)
+ ndbout << "table " << tab.m_name << " created" << endl;
+ if (test.m_ret != SQL_SUCCESS && opt.m_v >= 2)
+ ndbout << "table " << tab.m_name << " already exists" << endl;
+ test.timerCnt(1);
+ }
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+// prepare without execute
+
+static void
+testPrepare(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned cnt = opt.m_depth; cnt <= opt.m_depth; cnt++) {
+ for (unsigned i = 0; i < tabCount; i++) {
+ Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ tab.selectJoin(sqlptr = sql, cnt);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ SQLSMALLINT colCount = -1;
+ SQLSMALLINT colExp = cnt * tab.m_colCount;
+ test.run(HStmt(hStmt), SQLNumResultCols(hStmt, &colCount));
+ test.chk(HStmt(hStmt), colCount == colExp, "got %d != %d", (int)colCount, (int)colExp);
+ test.timerCnt(1);
+ }
+ }
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+// catalog functions
+
+static void
+testCatalog(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ odbc_typeinfo: {
+ long type[] = {
+ SQL_CHAR, SQL_VARCHAR, SQL_SMALLINT, SQL_INTEGER, SQL_BIGINT, SQL_REAL, SQL_DOUBLE
+ };
+ unsigned rows[] = {
+ 1, 1, 2, 2, 2, 1, 1 // 2 for signed and unsigned
+ };
+ for (unsigned i = 0; i < arraySize(type); i++) {
+ test.run(HStmt(hStmt), SQLGetTypeInfo(hStmt, type[i]));
+ long dataType = 0;
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 2, SQL_C_SLONG, &dataType, 0, 0));
+ unsigned k = 0;
+ while (1) {
+ if (k == rows[i])
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (k == rows[i])
+ break;
+ test.chk(HStmt(hStmt), dataType == type[i], "got %ld != %ld", dataType, type[i]);
+ test.timerCnt(1);
+ k++;
+ }
+#ifndef iODBC
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+#else
+ freeStmt(test, hDbc, hStmt);
+ allocStmt(test, hDbc, hStmt);
+#endif
+ }
+ if (opt.m_v >= 2)
+ ndbout << "found " << (UintPtr)arraySize(type) << " data types" << endl;
+ test.run(HStmt(hStmt), SQLFreeStmt(hStmt, SQL_UNBIND));
+ }
+ odbc_tables: {
+ unsigned found[tabCount];
+ for (unsigned i = 0; i < tabCount; i++)
+ found[i] = 0;
+ test.run(HStmt(hStmt), SQLTables(hStmt, (SQLCHAR*)0, 0, (SQLCHAR*)0, 0, (SQLCHAR*)0, 0, (SQLCHAR*)0, 0));
+ char tableName[200] = "";
+ char tableType[200] = "";
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 3, SQL_C_CHAR, tableName, sizeof(tableName), 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 4, SQL_C_CHAR, tableType, sizeof(tableType), 0));
+ unsigned cnt = 0;
+ while (1) {
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (test.m_ret == SQL_NO_DATA)
+ break;
+ test.timerCnt(1);
+ cnt++;
+ if (! blankeq(tableType, "TABLE"))
+ continue;
+ for (unsigned i = 0; i < tabCount; i++) {
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ if (! blankeq(tab.m_name, tableName))
+ continue;
+ test.chk(HStmt(hStmt), found[i] == 0, "duplicate table %s", tab.m_name);
+ found[i]++;
+ }
+ }
+#ifndef iODBC
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+#else
+ freeStmt(test, hDbc, hStmt);
+ allocStmt(test, hDbc, hStmt);
+#endif
+ for (unsigned i = 0; i < tabCount; i++) {
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ test.chk(HStmt(hStmt), found[i] == 1, "table %s not found", tab.m_name);
+ }
+ if (opt.m_v >= 2)
+ ndbout << "found " << cnt << " tables" << endl;
+ test.run(HStmt(hStmt), SQLFreeStmt(hStmt, SQL_UNBIND));
+ }
+ odbc_columns: {
+ unsigned found[tabCount][maxColCount];
+ for (unsigned i = 0; i < tabCount; i++) {
+ for (unsigned j = 0; j < maxColCount; j++)
+ found[i][j] = 0;
+ }
+ test.run(HStmt(hStmt), SQLColumns(hStmt, (SQLCHAR*)0, 0, (SQLCHAR*)0, 0, (SQLCHAR*)0, 0, (SQLCHAR*)0, 0));
+ char tableName[200] = "";
+ char columnName[200] = "";
+ long dataType = 0;
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 3, SQL_C_CHAR, tableName, sizeof(tableName), 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 4, SQL_C_CHAR, columnName, sizeof(columnName), 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 5, SQL_C_SLONG, &dataType, 0, 0));
+ unsigned cnt = 0;
+ while (1) {
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (test.m_ret == SQL_NO_DATA)
+ break;
+ test.timerCnt(1);
+ cnt++;
+ for (unsigned i = 0; i < tabCount; i++) {
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ if (! blankeq(tab.m_name, tableName))
+ continue;
+ bool columnFound = false;
+ for (unsigned j = 0; j < tab.m_colCount; j++) {
+ const Col& col = tab.m_colList[j];
+ if (! blankeq(col.m_name, columnName))
+ continue;
+ test.chk(HStmt(hStmt), found[i][j] == 0, "duplicate column %s.%s", tableName, columnName);
+ found[i][j]++;
+ columnFound = true;
+ }
+ test.chk(HStmt(hStmt), columnFound, "unknown column %s.%s", tableName, columnName);
+ }
+ }
+#ifndef iODBC
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+#else
+ freeStmt(test, hDbc, hStmt);
+ allocStmt(test, hDbc, hStmt);
+#endif
+ for (unsigned i = 0; i < tabCount; i++) {
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ for (unsigned j = 0; j < tab.m_colCount; j++) {
+ const Col& col = tab.m_colList[j];
+ test.chk(HStmt(hStmt), found[i][j] == 1, "column %s.%s not found", tab.m_name, col.m_name);
+ }
+ }
+ if (opt.m_v >= 2)
+ ndbout << "found " << cnt << " columns" << endl;
+ test.run(HStmt(hStmt), SQLFreeStmt(hStmt, SQL_UNBIND));
+ }
+ odbc_primarykeys: {
+ // table patterns are no allowed
+ for (unsigned i = 0; i < tabCount; i++) {
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ char tmp[200]; // p.i.t.a
+ strcpy(tmp, tab.m_name);
+ for (char* a = tmp; *a != 0; a++) {
+ if ('a' <= *a && *a <= 'z')
+ *a -= 'a' - 'A';
+ }
+ test.run(HStmt(hStmt), SQLPrimaryKeys(hStmt, (SQLCHAR*)0, 0, (SQLCHAR*)0, 0, (SQLCHAR*)tmp, SQL_NTS));
+ char tableName[200] = "";
+ char columnName[200] = "";
+ long keySeq = -1;
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 3, SQL_C_CHAR, tableName, sizeof(tableName), 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 4, SQL_C_CHAR, columnName, sizeof(columnName), 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 5, SQL_C_SLONG, &keySeq, 0, 0));
+ unsigned cnt = 0;
+ while (1) {
+ if (cnt == tab.m_pkCount)
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (test.m_ret == SQL_NO_DATA)
+ break;
+ test.chk(HStmt(hStmt), keySeq == 1 + cnt, "got %ld != %u", keySeq, 1 + cnt);
+ const Col& col = tab.m_colList[tab.m_pkIndex[keySeq - 1]];
+ test.chk(HStmt(hStmt), blankeq(columnName, col.m_name), "got %s != %s", columnName, col.m_name);
+ test.timerCnt(1);
+ cnt++;
+ }
+#ifndef iODBC
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+#else
+ freeStmt(test, hDbc, hStmt);
+ allocStmt(test, hDbc, hStmt);
+#endif
+ }
+ test.run(HStmt(hStmt), SQLFreeStmt(hStmt, SQL_UNBIND));
+ }
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+// insert
+
+static void
+testInsert(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ // prepare
+ tab.insertAll(sqlptr = sql);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ SQLSMALLINT parCount = -1;
+ test.run(HStmt(hStmt), SQLNumParams(hStmt, &parCount));
+ test.chk(HStmt(hStmt), parCount == tab.m_colCount, "got %d != %d", (int)parCount, (int)tab.m_colCount);
+ // bind parameters
+ Row row(tab);
+ for (unsigned j = 0; j < tab.m_colCount; j++) {
+ Fld& fld = row.m_fldList[j];
+ const Col& col = fld.m_col;
+ // every other at-exec
+ SQLPOINTER caddr;
+ SQLINTEGER* ind;
+ if (opt.m_noputd || j % 2 == 0) {
+ caddr = fld.caddr();
+ ind = fld.ind();
+ } else {
+ caddr = (SQLPOINTER)j;
+ ind = fld.need();
+ }
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, caddr, col.csize(), ind));
+ }
+ // bind columns (none)
+ SQLSMALLINT colCount = -1;
+ test.run(HStmt(hStmt), SQLNumResultCols(hStmt, &colCount));
+ test.chk(HStmt(hStmt), colCount == 0, "got %d != 0", (int)colCount);
+ // execute
+ for (unsigned k = 0; k < opt.m_scale; k++) {
+ if (k % 5 == 0) {
+ // rebind
+ unsigned j = 0;
+ Fld& fld = row.m_fldList[j];
+ const Col& col = fld.m_col;
+ // every other at-exec
+ SQLPOINTER caddr;
+ SQLINTEGER* ind;
+ if (opt.m_noputd || j % 2 == 0) {
+ caddr = fld.caddr();
+ ind = fld.ind();
+ } else {
+ caddr = (SQLPOINTER)j;
+ ind = fld.need();
+ }
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, caddr, col.csize(), ind));
+ }
+ row.calcPk(test, k);
+ row.calcNk(test);
+ unsigned needData = opt.m_noputd ? 0 : tab.m_colCount / 2;
+ if (needData)
+ test.exp(SQL_NEED_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.chk(HStmt(hStmt), test.m_functionCode == SQL_DIAG_INSERT, "got %d != %d", test.m_functionCode, SQL_DIAG_INSERT);
+ if (needData) {
+ while (1) {
+ SQLPOINTER jPtr = (SQLPOINTER)999;
+ if (needData)
+ test.exp(SQL_NEED_DATA, 0, 0, true);
+ // completes SQLExecute on success
+ test.run(HStmt(hStmt), SQLParamData(hStmt, &jPtr));
+ if (! needData)
+ break;
+ unsigned j = (unsigned)jPtr;
+ test.chk(HStmt(hStmt), j < tab.m_colCount && j % 2 != 0, "got %u 0x%x", j, j);
+ Fld& fld = row.m_fldList[j];
+ const Col& col = fld.m_col;
+ SQLSMALLINT ctype = col.ctype();
+ if (k % 2 == 0 || ctype != Col::CChar)
+ test.run(HStmt(hStmt), SQLPutData(hStmt, fld.caddr(), *fld.ind()));
+ else {
+ // put in pieces
+ unsigned size = col.csize() - 1; // omit null terminator
+ char* caddr = (char*)(fld.caddr());
+ unsigned off = 0;
+ while (off < size) {
+ unsigned m = size / 7; // bytes to put
+ if (m == 0)
+ m = 1;
+ if (m > size - off)
+ m = size - off;
+ bool putNull = (*fld.ind() == SQL_NULL_DATA);
+ // no null terminator
+ SQLINTEGER len = putNull ? SQL_NULL_DATA : (int)m;
+ test.run(HStmt(hStmt), SQLPutData(hStmt, caddr + off, len));
+ if (putNull)
+ break;
+ off += m;
+ }
+ }
+ needData--;
+ }
+ }
+ chkRowCount(test, hStmt, 1);
+ chkTuplesFetched(test, hStmt, 0);
+ }
+ test.timerCnt(opt.m_scale);
+ if (opt.m_v >= 3)
+ ndbout << "inserted " << opt.m_scale << " into " << tab.m_name << endl;
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+// count
+
+static void
+testCount(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ long count = -1;
+ selectCount(test, hStmt, tab, &count);
+ test.chk(HStmt(hStmt), count == opt.m_scale * opt.m_threads, "got %ld != %u", count, opt.m_scale * opt.m_threads);
+ test.timerCnt(count);
+ if (opt.m_v >= 3)
+ ndbout << "counted " << (int)count << " rows in " << tab.m_name << endl;
+ }
+ // scan all at same time
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ tab.selectAll(sqlptr = sql);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ unsigned k = 0;
+ while (1) {
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ if (k == opt.m_scale * opt.m_threads)
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (k != opt.m_scale * opt.m_threads) {
+ chkTuplesFetched(test, hStmt, k + 1);
+ test.timerCnt(1);
+ } else {
+ chkTuplesFetched(test, hStmt, k);
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLMoreResults(hStmt));
+ }
+ }
+ if (k == opt.m_scale * opt.m_threads)
+ break;
+ k++;
+ }
+ if (opt.m_v >= 3)
+ ndbout << "scanned " << opt.m_scale << " rows from each table" << endl;
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+// update
+
+static void
+testUpdatePk(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ // prepare
+ tab.updatePk(sqlptr = sql);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // bind parameters
+ Row row(tab);
+ SQLSMALLINT parCount = -1;
+ test.run(HStmt(hStmt), SQLNumParams(hStmt, &parCount));
+ test.chk(HStmt(hStmt), parCount == tab.m_colCount, "got %d != %d", (int)parCount, (int)tab.m_colCount);
+ for (unsigned j = 0; j < tab.m_nkCount; j++) {
+ Fld& fld = row.m_fldList[tab.m_nkIndex[j]];
+ const Col& col = fld.m_col;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fld.caddr(), col.csize(), fld.ind()));
+ }
+ for (unsigned j = 0; j < tab.m_pkCount; j++) {
+ Fld& fld = row.m_fldList[tab.m_pkIndex[j]];
+ const Col& col = fld.m_col;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + tab.m_nkCount + j, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fld.caddr(), col.csize(), fld.ind()));
+ }
+ // bind columns (none)
+ SQLSMALLINT colCount = -1;
+ test.run(HStmt(hStmt), SQLNumResultCols(hStmt, &colCount));
+ test.chk(HStmt(hStmt), colCount == 0, "got %d != 0", (int)colCount);
+ // execute
+ for (unsigned k = 0; k < opt.m_scale; k++) {
+ if (k % 5 == 0) {
+ unsigned j = 0;
+ Fld& fld = row.m_fldList[tab.m_nkIndex[j]];
+ const Col& col = fld.m_col;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fld.caddr(), col.csize(), fld.ind()));
+ }
+ row.calcPk(test, k);
+ row.calcNk(test);
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.chk(HStmt(hStmt), test.m_functionCode == SQL_DIAG_UPDATE_WHERE, "got %d != %d", test.m_functionCode, SQL_DIAG_UPDATE_WHERE);
+ chkRowCount(test, hStmt, 1);
+ // direct update, no read has been necessary
+ chkTuplesFetched(test, hStmt, 0);
+ }
+ test.timerCnt(opt.m_scale);
+ if (opt.m_v >= 3)
+ ndbout << "updated " << opt.m_scale << " in " << tab.m_name << endl;
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+static void
+testUpdateScan(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ // prepare
+ tab.updateRange(sqlptr = sql);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // bind parameters
+ Row row(tab); // for set clause
+ Row rowlo(tab); // for pk ranges
+ Row rowhi(tab);
+ SQLSMALLINT parCount = -1;
+ test.run(HStmt(hStmt), SQLNumParams(hStmt, &parCount));
+ test.chk(HStmt(hStmt), parCount == tab.m_nkCount + 2 * tab.m_pkCount, "got %d != %d", (int)parCount, (int)tab.m_nkCount + 2 * (int)tab.m_pkCount);
+ for (unsigned j = 0; j < tab.m_nkCount; j++) {
+ const Col& col = tab.m_colList[tab.m_nkIndex[j]];
+ Fld& fld = row.m_fldList[tab.m_nkIndex[j]];
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fld.caddr(), col.csize(), fld.ind()));
+ }
+ bool canInterp = true;
+ for (unsigned j = 0; j < tab.m_pkCount; j++) {
+ const Col& col = tab.m_colList[tab.m_pkIndex[j]];
+ Fld& fldlo = rowlo.m_fldList[tab.m_pkIndex[j]];
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + tab.m_nkCount + 2 * j + 0, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fldlo.caddr(), col.csize(), fldlo.ind()));
+ Fld& fldhi = rowhi.m_fldList[tab.m_pkIndex[j]];
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + tab.m_nkCount + 2 * j + 1, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fldhi.caddr(), col.csize(), fldhi.ind()));
+ if (col.m_type != Col::Char)
+ canInterp = false; // XXX no unsigned yet
+ }
+ // execute
+ row.calcPk(test, 0);
+ row.calcNk(test);
+ rowlo.calcPk(test, 0);
+ rowhi.calcPk(test, test.m_mul); // sucks
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.chk(HStmt(hStmt), test.m_functionCode == SQL_DIAG_UPDATE_WHERE, "got %d != %d", test.m_functionCode, SQL_DIAG_UPDATE_WHERE);
+ chkRowCount(test, hStmt, opt.m_scale);
+ chkTuplesFetched(test, hStmt, canInterp ? opt.m_scale : opt.m_scale * opt.m_threads);
+ test.timerCnt(opt.m_scale);
+ if (opt.m_v >= 3)
+ ndbout << "updated " << opt.m_scale << " in " << tab.m_name << endl;
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+// verify
+
+static void
+testVerifyPk(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ // prepare
+ tab.selectPk(sqlptr = sql);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // use same row for input and output
+ Row row(tab);
+ // bind parameters
+ SQLSMALLINT parCount = -1;
+ test.run(HStmt(hStmt), SQLNumParams(hStmt, &parCount));
+ test.chk(HStmt(hStmt), parCount == tab.m_pkCount, "got %d != %d", (int)parCount, (int)tab.m_pkCount);
+ for (unsigned j = 0; j < tab.m_pkCount; j++) {
+ Fld& fld = row.m_fldList[tab.m_pkIndex[j]];
+ const Col& col = fld.m_col;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fld.caddr(), col.csize(), fld.ind()));
+ }
+ // bind columns
+ SQLSMALLINT colCount = -1;
+ test.run(HStmt(hStmt), SQLNumResultCols(hStmt, &colCount));
+ test.chk(HStmt(hStmt), colCount == tab.m_colCount, "got %d != %d", (int)colCount, (int)tab.m_colCount);
+ for (unsigned j = 0; j < tab.m_colCount; j++) {
+ Fld& fld = row.m_fldList[j];
+ const Col& col = fld.m_col;
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1 + j, col.ctype(), fld.caddr(), col.csize(), fld.ind()));
+ }
+ // row for SQLGetData
+ Row rowGet(tab);
+ // reference row
+ Row rowRef(tab);
+ // execute
+ for (unsigned k = 0; k < opt.m_scale; k++) {
+ if (k % 5 == 0) {
+ // rebind
+ unsigned j = 0;
+ Fld& fld = row.m_fldList[tab.m_pkIndex[j]];
+ const Col& col = fld.m_col;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fld.caddr(), col.csize(), fld.ind()));
+ }
+ row.calcPk(test, k);
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.chk(HStmt(hStmt), test.m_functionCode == SQL_DIAG_SELECT_CURSOR, "got %d != %d", test.m_functionCode, SQL_DIAG_SELECT_CURSOR);
+ // fetch
+ for (unsigned k2 = 0; ; k2++) {
+ if (k2 == 1)
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ chkTuplesFetched(test, hStmt, 1);
+ if (k2 == 1)
+ break;
+ rowRef.calcPk(test, k);
+ test.chk(HStmt(hStmt), row.verifyPk(test, rowRef), "verify row=%d", k);
+ if (test.m_const)
+ rowRef.calcPk(test, 0);
+ rowRef.calcNk(test);
+ test.chk(HStmt(hStmt), row.verifyNk(test, rowRef), "verify row=%d", k);
+ // SQLGetData is supported independent of SQLBindCol
+ if (opt.m_nogetd)
+ continue;
+ for (unsigned j = 0; j < tab.m_colCount; j++) {
+ Fld& fld = rowGet.m_fldList[j];
+ fld.zero();
+ const Col& col = fld.m_col;
+ // test both variants
+ SQLSMALLINT ctype = k % 2 == 0 ? col.ctype() : SQL_ARD_TYPE;
+ if (ctype != Col::CChar)
+ test.run(HStmt(hStmt), SQLGetData(hStmt, 1 + j, ctype, fld.caddr(), col.csize(), fld.ind()));
+ else {
+ // get in pieces
+ unsigned size = col.csize() - 1; // omit null terminator
+ char* caddr = (char*)(fld.caddr());
+ unsigned off = 0;
+ while (off < size) {
+ unsigned m = size / 3; // bytes to get
+ if (m == 0)
+ m = 1;
+ if (m > size - off)
+ m = size - off;
+ bool getNull = (rowRef.m_fldList[j].m_ind == SQL_NULL_DATA);
+ if (off + m < size && ! getNull)
+ test.exp(SQL_SUCCESS_WITH_INFO, "01004", -1, true);
+ // include null terminator in buffer size
+ test.run(HStmt(hStmt), SQLGetData(hStmt, 1 + j, ctype, caddr + off, m + 1, fld.ind()));
+ int ind = *fld.ind();
+ if (getNull) {
+ test.chk(HStmt(hStmt), ind == SQL_NULL_DATA, "got %d", ind);
+ break;
+ }
+ test.chk(HStmt(hStmt), ind == size - off, "got %d != %u", ind, size - off);
+ off += m;
+ }
+ }
+ }
+ rowRef.calcPk(test, k);
+ test.chk(HStmt(hStmt), rowGet.verifyPk(test, rowRef), "verify row=%d", k);
+ if (test.m_const)
+ rowRef.calcPk(test, 0);
+ rowRef.calcNk(test);
+ test.chk(HStmt(hStmt), rowGet.verifyNk(test, rowRef), "verify row=%d", k);
+ // SQLGetData again
+ for (unsigned j = 0; j < tab.m_colCount; j++) {
+ Fld& fld = rowGet.m_fldList[j];
+ const Col& col = fld.m_col;
+ // test both variants
+ SQLSMALLINT ctype = k % 2 == 0 ? col.ctype() : SQL_ARD_TYPE;
+ // expect no more data
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLGetData(hStmt, 1 + j, ctype, fld.caddr(), col.csize(), fld.ind()));
+ }
+ }
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ }
+ test.timerCnt(opt.m_scale);
+ if (opt.m_v >= 3)
+ ndbout << "verified " << opt.m_scale << " from " << tab.m_name << endl;
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+static void
+testVerifyScan(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ // prepare
+ tab.selectRange(sqlptr = sql, ! opt.m_nosort);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // bind parameters
+ Row rowlo(tab); // use available PK fields..
+ Row rowhi(tab); // since we have no other way for now
+ SQLSMALLINT parCount = -1;
+ test.run(HStmt(hStmt), SQLNumParams(hStmt, &parCount));
+ test.chk(HStmt(hStmt), parCount == 2 * tab.m_pkCount, "got %d != %d", (int)parCount, 2 * (int)tab.m_pkCount);
+ for (unsigned j = 0; j < tab.m_pkCount; j++) {
+ const Col& col = tab.m_colList[tab.m_pkIndex[j]];
+ Fld& fldlo = rowlo.m_fldList[tab.m_pkIndex[j]];
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + 2 * j + 0, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fldlo.caddr(), col.csize(), fldlo.ind()));
+ Fld& fldhi = rowhi.m_fldList[tab.m_pkIndex[j]];
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + 2 * j + 1, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fldhi.caddr(), col.csize(), fldhi.ind()));
+ }
+ // bind columns
+ Row row(tab);
+ SQLSMALLINT colCount = -1;
+ test.run(HStmt(hStmt), SQLNumResultCols(hStmt, &colCount));
+ test.chk(HStmt(hStmt), colCount == tab.m_colCount, "got %d != %d", (int)colCount, (int)tab.m_colCount);
+ for (unsigned j = 0; j < tab.m_colCount; j++) {
+ Fld& fld = row.m_fldList[j];
+ const Col& col = fld.m_col;
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1 + j, col.ctype(), fld.caddr(), col.csize(), fld.ind()));
+ }
+ // execute
+ rowlo.calcPk(test, 0);
+ rowhi.calcPk(test, test.m_mul); // sucks
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.chk(HStmt(hStmt), test.m_functionCode == SQL_DIAG_SELECT_CURSOR, "got %d != %d", test.m_functionCode, SQL_DIAG_SELECT_CURSOR);
+ // reference row
+ Row rowRef(tab);
+ // fetch
+ unsigned k = 0;
+ SQLUINTEGER rowCount1 = (SQLUINTEGER)-1;
+ test.run(HStmt(hStmt), SQLSetStmtAttr(hStmt, SQL_ATTR_ROWS_FETCHED_PTR, &rowCount1, SQL_IS_POINTER));
+ while (1) {
+ unsigned countExp;
+ if (k == opt.m_scale) {
+ countExp = k;
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ } else {
+ countExp = k + 1;
+ }
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ // let me count the ways..
+ chkRowCount(test, hStmt, countExp);
+ test.chk(HStmt(hStmt), rowCount1 == countExp, "got %lu != %u", rowCount1, countExp);
+ SQLUINTEGER rowCount2 = (SQLUINTEGER)-1;
+ test.run(HStmt(hStmt), SQLGetStmtAttr(hStmt, SQL_ATTR_ROW_NUMBER, &rowCount2, SQL_IS_POINTER, 0));
+ test.chk(HStmt(hStmt), rowCount2 == countExp, "got %lu != %u", rowCount2, countExp);
+ if (k == opt.m_scale)
+ break;
+ if (! opt.m_nosort) {
+ // expecting k-th row
+ rowRef.calcPk(test, k);
+ test.chk(HStmt(hStmt), row.verifyPk(test, rowRef), "verify row=%d", k);
+ if (test.m_const)
+ rowRef.calcPk(test, 0);
+ rowRef.calcNk(test);
+ test.chk(HStmt(hStmt), row.verifyNk(test, rowRef), "verify row=%d", k);
+ } else {
+ // expecting random row
+ rowRef.copy(row);
+ test.chk(HStmt(hStmt), row.verifyNk(test, rowRef), "verify row=%d", k);
+ }
+ k++;
+ }
+ test.timerCnt(opt.m_scale);
+ if (opt.m_v >= 3)
+ ndbout << "verified " << opt.m_scale << " from " << tab.m_name << endl;
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+// self-join (scan followed by pk lookups)
+
+static void
+testJoin(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned cnt = opt.m_depth; cnt <= opt.m_depth; cnt++) {
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ tab.selectJoin(sqlptr = sql, cnt);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ unsigned k = 0;
+ while (1) {
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ if (k == opt.m_scale * opt.m_threads)
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (k == opt.m_scale * opt.m_threads) {
+ chkTuplesFetched(test, hStmt, k * opt.m_depth);
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ } else {
+ chkTuplesFetched(test, hStmt, (k + 1) * opt.m_depth);
+ test.timerCnt(1);
+ }
+ }
+ if (k == opt.m_scale * opt.m_threads)
+ break;
+ k++;
+ }
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+// cartesian join (multiple nested scans)
+
+static void
+testCart(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned cnt = 2; cnt <= 2; cnt++) {
+ unsigned rows = 1;
+ //for (unsigned k = 0; k < opt.m_depth; k++) {
+ //rows *= opt.m_scale * opt.m_threads;
+ //}
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ tab.selectCart(sqlptr = sql, cnt);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ unsigned k = 0;
+ while (1) {
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ if (k == rows)
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (k == rows) {
+ //chkTuplesFetched(test, hStmt, k);
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ } else {
+ //chkTuplesFetched(test, hStmt, k + 1);
+ test.timerCnt(1);
+ }
+ }
+ if (k == rows)
+ break;
+ k++;
+ }
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+// delete
+
+static void
+testDeleteAll(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ long count0 = -1;
+ selectCount(test, hStmt, tab, &count0);
+ tab.deleteAll(sqlptr = sql);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ if (count0 == 0)
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+#ifndef iODBC
+ test.chk(HStmt(hStmt), test.m_functionCode == SQL_DIAG_DELETE_WHERE, "got %d != %d", test.m_functionCode, SQL_DIAG_DELETE_WHERE);
+#endif
+ SQLINTEGER rowCount = -1;
+ getRowCount(test, hStmt, &rowCount);
+ test.timerCnt(rowCount);
+ test.chk(HStmt(hStmt), rowCount == count0, "got %d != %ld", (int)rowCount, count0);
+ chkTuplesFetched(test, hStmt, rowCount);
+ if (opt.m_v >= 3)
+ ndbout << "deleted " << (int)rowCount << " from " << tab.m_name << endl;
+ long count = -1;
+ selectCount(test, hStmt, tab, &count);
+ test.chk(HStmt(hStmt), count == 0, "got %ld != 0", count);
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+static void
+testDeletePk(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ // prepare
+ tab.deletePk(sqlptr = sql);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // bind parameters
+ Row row(tab);
+ SQLSMALLINT parCount = -1;
+ test.run(HStmt(hStmt), SQLNumParams(hStmt, &parCount));
+ test.chk(HStmt(hStmt), parCount == tab.m_pkCount, "got %d != %d", (int)parCount, (int)tab.m_colCount);
+ for (unsigned j = 0; j < tab.m_pkCount; j++) {
+ Fld& fld = row.m_fldList[tab.m_pkIndex[j]];
+ const Col& col = fld.m_col;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, col.ctype(), col.type(), col.size(), 0, fld.caddr(), col.csize(), fld.ind()));
+ }
+ // bind columns (none)
+ SQLSMALLINT colCount = -1;
+ test.run(HStmt(hStmt), SQLNumResultCols(hStmt, &colCount));
+ test.chk(HStmt(hStmt), colCount == 0, "got %d != 0", (int)colCount);
+ // execute
+ for (unsigned k = 0; k < opt.m_scale; k++) {
+ row.calcPk(test, k);
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.chk(HStmt(hStmt), test.m_functionCode == SQL_DIAG_DELETE_WHERE, "got %d != %d", test.m_functionCode, SQL_DIAG_DELETE_WHERE);
+ chkRowCount(test, hStmt, 1);
+ // direct delete, no fetch required
+ chkTuplesFetched(test, hStmt, 0);
+ }
+ test.timerCnt(opt.m_scale);
+ if (opt.m_v >= 3)
+ ndbout << "updated " << opt.m_scale << " in " << tab.m_name << endl;
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+static void
+testTrans(Test& test)
+{
+#ifdef unixODBC
+ if (opt.m_v >= 1)
+ ndbout << "unixODBC does not support transactions - test skipped" << endl;
+#else
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ // delete all
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ tab.deleteAll(sqlptr = sql);
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ SQLINTEGER rowCount = -1;
+ getRowCount(test, hStmt, &rowCount);
+ if (opt.m_v >= 3)
+ ndbout << "deleted " << (int)rowCount << " from " << tab.m_name << endl;
+ }
+ setAutocommit(test, hDbc, false);
+ if (opt.m_v >= 2)
+ ndbout << "set autocommit OFF" << endl;
+ for (int commit = 0; commit < opt.m_scale; commit += 1) {
+ bool rollback = (commit % 2 == 0);
+ // XXX delete with no data leaves trans in error state for 2nd table
+ if (commit > 0 && rollback) { // previous case was commit
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ tab.deleteDirect(sqlptr = sql, 0);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ test.run(HDbc(hDbc), SQLEndTran(SQL_HANDLE_DBC, hDbc, SQL_COMMIT));
+ }
+ // insert
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ tab.insertDirect(sqlptr = sql, 0);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ if (opt.m_v >= 2)
+ ndbout << tab.m_name << ": inserted 1 row" << endl;
+ }
+ // count them via pk
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ tab.countDirect(sqlptr = sql, 0);
+ long count = -1;
+ long countExp = 1;
+ selectCount(test, hStmt, sql, &count);
+ test.chk(HStmt(hStmt), count == countExp, "got %ld != %ld", count, countExp);
+ }
+ // count them via scan
+ for (unsigned i = 0; i < tabCount; i++) {
+ // XXX hupp no work
+ break;
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ long count = -1;
+ long countExp = 1;
+ selectCount(test, hStmt, tab, &count);
+ test.chk(HStmt(hStmt), count == countExp, "got %ld != %ld", count, countExp);
+ }
+ // rollback or commit
+ if (rollback) {
+ if (opt.m_v >= 2)
+ ndbout << "end trans ROLLBACK" << endl;
+ test.run(HDbc(hDbc), SQLEndTran(SQL_HANDLE_DBC, hDbc, SQL_ROLLBACK));
+ } else {
+ if (opt.m_v >= 2)
+ ndbout << "end trans COMMIT" << endl;
+ test.run(HDbc(hDbc), SQLEndTran(SQL_HANDLE_DBC, hDbc, SQL_COMMIT));
+ }
+ // count them via pk again
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ tab.countDirect(sqlptr = sql, 0);
+ long count = -1;
+ long countExp = rollback ? 0 : 1;
+ selectCount(test, hStmt, sql, &count);
+ test.chk(HStmt(hStmt), count == countExp, "got %ld != %ld", count, countExp);
+ }
+ // count them via scan again
+ for (unsigned i = 0; i < tabCount; i++) {
+ // XXX hupp no work
+ break;
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ long count = -1;
+ long countExp = rollback ? 0 : 1;
+ selectCount(test, hStmt, tab, &count);
+ test.chk(HStmt(hStmt), count == countExp, "got %ld != %ld", count, countExp);
+ }
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+#endif
+}
+
+static void
+testConcur(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmtList[tabCount];
+ allocAll(test, hEnv, hDbc, hStmtList, tabCount);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ SQLHANDLE& hStmt = hStmtList[i];
+ const Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ // delete all
+ tab.deleteAll(sqlptr = sql);
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // insert some
+ unsigned rowcount = 10;
+ for (unsigned n = 0; n < rowcount; n++) {
+ tab.insertDirect(sqlptr = sql, n);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ verifyCount(test, hStmt, tab, rowcount);
+ // start query scan followed by pk lookups
+ tab.selectJoin(sqlptr = sql, 2);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // start fetch
+ unsigned k = 0;
+ while (1) {
+ if (k > 0)
+ test.exp(SQL_ERROR, "24000", -1, true); // commit closed cursor
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (k > 0)
+ break;
+ // delete some random row
+ tab.deleteDirect(sqlptr = sql, k);
+ // try using same statement
+ test.exp(SQL_ERROR, "24000", -1, true); // cursor is open
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // try using different statement
+ SQLHANDLE hStmt2;
+ allocStmt(test, hDbc, hStmt2);
+ test.run(HStmt(hStmt2), SQLExecDirect(hStmt2, (SQLCHAR*)sql, SQL_NTS));
+ k++;
+ }
+ test.exp(SQL_ERROR, "24000", -1, true); // cursor is not open
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ test.timerCnt(rowcount);
+ }
+ freeAll(test, hEnv, hDbc, hStmtList, tabCount);
+}
+
+static void
+testReadcom(Test& test)
+{
+ testDeleteAll(test);
+ testInsert(test);
+ const unsigned nc = 3;
+ SQLHANDLE hEnv[nc], hDbc[nc], hStmt[nc];
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned j = 0; j < nc; j++)
+ allocAll(test, hEnv[j], hDbc[j], hStmt[j]);
+ for (unsigned i = 0; i < tabCount; i++) {
+ Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ long count;
+ // check count
+ count = -1;
+ selectCount(test, hStmt[0], tab, &count);
+ test.chk(HStmt(hStmt[0]), count == opt.m_scale, "got %d != %d", (int)count, (int)opt.m_scale);
+ // scan delete uncommitted with handle 0
+ setAutocommit(test, hDbc[0], false);
+ tab.deleteAll(sqlptr = sql);
+ if (opt.m_scale == 0)
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt[0]), SQLExecDirect(hStmt[0], (SQLCHAR*)sql, SQL_NTS));
+ // scan via other tx should not hang and see all rows
+ for (unsigned j = 0; j < nc; j++) {
+ count = -1;
+ int want = j == 0 ? 0 : opt.m_scale;
+ selectCount(test, hStmt[j], tab, &count);
+ test.chk(HStmt(hStmt[j]), count == want, "tx %u: got %d != %d", j, (int)count, want);
+ if (opt.m_v >= 2)
+ ndbout << "tx " << j << " ok !" << endl;
+ }
+ // setting autocommit on commits the delete
+ setAutocommit(test, hDbc[0], true);
+ // check count
+ count = -1;
+ selectCount(test, hStmt[0], tab, &count);
+ test.chk(HStmt(hStmt[0]), count == 0, "got %d != 0", (int)count);
+ }
+ for (unsigned j = 0; j < nc; j++)
+ freeAll(test, hEnv[j], hDbc[j], hStmt[j]);
+}
+
+static void
+testPerf(Test& test)
+{
+ if (test.m_stuff == 0) {
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ char sql[MAX_SQL], *sqlptr;
+ for (unsigned i = 0; i < tabCount; i++) {
+ Tab& tab = tabList[i];
+ if (! tab.optok())
+ continue;
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ tab.deleteAll(sqlptr = sql);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ long count0 = -1;
+ // XXX triggers SEGV somewhere
+ //selectCount(test, hStmt, tab, &count0);
+ //test.chk(HStmt(hStmt), count0 == 0, "got %d != 0", (int)count0);
+ }
+ freeAll(test, hEnv, hDbc, hStmt);
+ return;
+ }
+ assert(test.m_stuff == 1 || test.m_stuff == 2);
+ bool ndbapi = (test.m_stuff == 1);
+ tt01: {
+ const unsigned OFF = 1000000;
+ const unsigned N = 25;
+ Tab& tab = tabList[1];
+ if (! tab.optok())
+ goto out;
+ if (ndbapi) {
+#ifndef ndbODBC
+ if (opt.m_v >= 1)
+ ndbout << "running via DM - test skipped" << endl;
+#else
+ Ndb* ndb = new Ndb("TEST_DB");
+ ndb->init();
+ if (ndb->waitUntilReady() != 0) {
+ ndbout << ndb->getNdbError() << endl;
+ fatal("waitUntilReady");
+ }
+ Uint32 val[1+N];
+ // insert
+ for (unsigned k = 1; k <= opt.m_scale; k++) {
+ NdbConnection* con = ndb->startTransaction();
+ if (con == 0) {
+ ndbout << ndb->getNdbError() << endl;
+ fatal("startTransaction");
+ }
+ NdbOperation* op = con->getNdbOperation(tab.m_upperName);
+ if (op == 0) {
+ ndbout << con->getNdbError() << endl;
+ fatal("getNdbOperation");
+ }
+ if (op->insertTuple() == -1) {
+ ndbout << op->getNdbError() << endl;
+ fatal("insertTuple");
+ }
+ for (unsigned j = 0; j <= N; j++) {
+ val[j] = (j == 0 ? k + test.m_no * OFF : k * j);
+ if (j == 0) {
+ if (op->equal(j, val[j]) == -1) {
+ ndbout << op->getNdbError() << endl;
+ fatal("equal");
+ }
+ } else {
+ if (op->setValue(j, val[j]) == -1) {
+ ndbout << op->getNdbError() << endl;
+ fatal("setValue");
+ }
+ }
+ }
+ if (con->execute(Commit) == -1) {
+ ndbout << con->getNdbError() << endl;
+ fatal("execute");
+ }
+ ndb->closeTransaction(con);
+ }
+ test.timerCnt(opt.m_scale);
+ // select PK
+ for (unsigned k = 1; k <= opt.m_scale; k++) {
+ NdbConnection* con = ndb->startTransaction();
+ if (con == 0) {
+ ndbout << ndb->getNdbError() << endl;
+ fatal("startTransaction");
+ }
+ NdbOperation* op = con->getNdbOperation(tab.m_upperName);
+ if (op == 0) {
+ ndbout << con->getNdbError() << endl;
+ fatal("getNdbOperation");
+ }
+ if (op->readTuple() == -1) {
+ ndbout << op->getNdbError() << endl;
+ fatal("insertTuple");
+ }
+ for (unsigned j = 0; j <= N; j++) {
+ val[j] = (j == 0 ? k + test.m_no * OFF : 0);
+ if (j == 0) {
+ if (op->equal(j, val[j]) == -1) {
+ ndbout << op->getNdbError() << endl;
+ fatal("equal");
+ }
+ } else {
+ if (op->getValue(j, (char*)&val[j]) == 0) {
+ ndbout << op->getNdbError() << endl;
+ fatal("getValue");
+ }
+ }
+ }
+ if (con->execute(Commit) == -1) {
+ ndbout << con->getNdbError() << endl;
+ fatal("execute");
+ }
+ for (unsigned j = 1; j <= N; j++) {
+ assert(val[j] == k * j);
+ }
+ ndb->closeTransaction(con);
+ }
+ test.timerCnt(opt.m_scale);
+ // delete PK
+ for (unsigned k = 1; k <= opt.m_scale; k++) {
+ NdbConnection* con = ndb->startTransaction();
+ if (con == 0) {
+ ndbout << ndb->getNdbError() << endl;
+ fatal("startTransaction");
+ }
+ NdbOperation* op = con->getNdbOperation(tab.m_upperName);
+ if (op == 0) {
+ ndbout << con->getNdbError() << endl;
+ fatal("getNdbOperation");
+ }
+ if (op->deleteTuple() == -1) {
+ ndbout << op->getNdbError() << endl;
+ fatal("deleteTuple");
+ }
+ unsigned j = 0;
+ val[j] = k + test.m_no * OFF;
+ if (op->equal(j, val[j]) == -1) {
+ ndbout << op->getNdbError() << endl;
+ fatal("equal");
+ }
+ if (con->execute(Commit) == -1) {
+ ndbout << con->getNdbError() << endl;
+ fatal("execute");
+ }
+ ndb->closeTransaction(con);
+ }
+ test.timerCnt(opt.m_scale);
+ delete ndb;
+#endif
+ } else {
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ long val[1+N];
+ char sql[MAX_SQL], *sqlptr;
+ // insert
+ tab.insertAll(sqlptr = sql);
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ for (unsigned j = 0; j <= N; j++) {
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_INTEGER, 0, 0, &val[j], 0, 0));
+ }
+ test.m_perf = true;
+ for (unsigned k = 1; k <= opt.m_scale; k++) {
+ for (unsigned j = 0; j <= N; j++) {
+ val[j] = (j == 0 ? k + test.m_no * OFF : k * j);
+ }
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ }
+ test.m_perf = false;
+ test.timerCnt(opt.m_scale);
+ // select PK
+ tab.selectPk(sqlptr = sql);
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ for (unsigned j = 0; j <= N; j++) {
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1 + j, SQL_C_SLONG, &val[j], 0, 0));
+ }
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + N + 1, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_INTEGER, 0, 0, &val[0], 0, 0));
+ test.m_perf = true;
+ for (unsigned k = 1; k <= opt.m_scale; k++) {
+ val[0] = k + test.m_no * OFF;
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ for (unsigned j = 1; j <= N; j++) {
+ assert(val[j] == k * j);
+ }
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ }
+ test.m_perf = false;
+ test.timerCnt(opt.m_scale);
+ // delete PK
+ tab.deletePk(sqlptr = sql);
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ unsigned j = 0;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1 + j, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_INTEGER, 0, 0, &val[j], 0, 0));
+ test.m_perf = true;
+ for (unsigned k = 1; k <= opt.m_scale; k++) {
+ val[j] = k + test.m_no * OFF;
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ }
+ test.m_perf = false;
+ test.timerCnt(opt.m_scale);
+ freeAll(test, hEnv, hDbc, hStmt);
+ }
+ out:
+ ;
+ }
+}
+
+struct Sql {
+ const char* m_sql;
+ int m_functionCode;
+ int m_rowCount;
+ int m_tuplesFetched;
+ long m_lastValue;
+ unsigned long m_bindValue;
+ int m_ret;
+ const char* m_state;
+ SQLINTEGER m_native;
+ bool m_reset;
+ // run this function instead
+ typedef void (*TestFunc)(Test& test);
+ TestFunc m_testFunc;
+ Sql() :
+ m_sql(0) {
+ }
+ Sql(const char* do_cmd) :
+ m_sql(do_cmd) {
+ }
+ Sql(const char* sql, int functionCode, int rowCount, int tuplesFetched, long lastValue, long bindValue) :
+ m_sql(sql),
+ m_functionCode(functionCode),
+ m_rowCount(rowCount),
+ m_tuplesFetched(tuplesFetched),
+ m_lastValue(lastValue),
+ m_bindValue(bindValue),
+ m_ret(SQL_SUCCESS),
+ m_state(0),
+ m_native(0),
+ m_reset(true),
+ m_testFunc(0) {
+ }
+ // the 4 numbers after SQL_DIAG... rowCount tuplesFetched lastValue bindValue
+ Sql(const char* sql, int functionCode, int rowCount, int tuplesFetched, long lastValue, long bindValue, int ret, const char* state, SQLINTEGER native, bool reset) :
+ m_sql(sql),
+ m_functionCode(functionCode),
+ m_rowCount(rowCount),
+ m_tuplesFetched(tuplesFetched),
+ m_lastValue(lastValue),
+ m_bindValue(bindValue),
+ m_ret(ret),
+ m_state(state),
+ m_native(native),
+ m_reset(reset),
+ m_testFunc(0) {
+ }
+ Sql(const char* text, TestFunc testFunc) :
+ m_sql(text),
+ m_testFunc(testFunc) {
+ }
+ static const char* set_autocommit_on() {
+ return "set autocommit on";
+ }
+ static const char* set_autocommit_off() {
+ return "set autocommit off";
+ }
+ static const char* do_commit() {
+ return "commit";
+ }
+ static const char* do_rollback() {
+ return "rollback";
+ }
+};
+
+// 90
+
+static const Sql
+miscSql90[] = {
+ Sql("select * from dual",
+ SQL_DIAG_SELECT_CURSOR, 1, 0, -1, -1),
+ Sql("drop table tt90a",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table tt90a (a int, b int, c int, primary key(b, c)) storage(large) logging",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ Sql()
+};
+
+// 91
+
+static const Sql
+miscSql91[] = {
+ Sql("drop table tt91a",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table tt91a (a bigint unsigned primary key, b bigint unsigned not null, c varchar(10))",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ Sql("insert into tt91a values (1, 111, 'aaa')",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ // fails
+ Sql("insert into tt91a values (2, null, 'ccc')",
+ SQL_DIAG_INSERT, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2014203, true),
+ Sql("update tt91a set b = 222 where a = 2",
+ SQL_DIAG_UPDATE_WHERE, 0, 0, -1, -1,
+ SQL_NO_DATA, 0, 0, true),
+ // two more
+ Sql("insert into tt91a values (2, 222, 'ccc')",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ Sql("insert into tt91a values (3, 333, 'bbb')",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ // direct update
+ Sql("update tt91a set b = 112 where a = 1",
+ SQL_DIAG_UPDATE_WHERE, 1, 0, -1, -1),
+ Sql("update tt91a set b = 113 where a = 1 and b > 111",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ // update and delete with interpreted scan
+ Sql("update tt91a set b = 114 where b < 114",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ Sql("delete from tt91a where b < 115",
+ SQL_DIAG_DELETE_WHERE, 1, 1, -1, -1),
+ Sql("insert into tt91a values (1, 111, 'aaa')",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ // check rows: 1,111,aaa + 2,222,ccc + 3,333,bbb
+ Sql("select * from tt91a order by c",
+ SQL_DIAG_SELECT_CURSOR, 3, 3, 2, -1),
+ Sql("select * from tt91a order by c desc",
+ SQL_DIAG_SELECT_CURSOR, 3, 3, 1, -1),
+ Sql("select * from tt91a where a = 2",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, -1, -1),
+ Sql("select * from tt91a where a + b = 224",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, -1, -1),
+ Sql("select * from tt91a where a = 4",
+ SQL_DIAG_SELECT_CURSOR, 0, 0, -1, -1),
+ Sql("select b-a from tt91a order by a-b",
+ SQL_DIAG_SELECT_CURSOR, 3, 3, 110, -1),
+ Sql("select sum(a+b) from tt91a",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 672, -1),
+ Sql("select x.b, y.b, z.b from tt91a x, tt91a y, tt91a z where x.b <= y.b and y.b < z.b order by x.b",
+ SQL_DIAG_SELECT_CURSOR, 4, 13, 222, -1),
+ Sql("select x.b, y.b, z.b from tt91a x, tt91a y, tt91a z where x.b + y.b = z.b order by x.b",
+ SQL_DIAG_SELECT_CURSOR, 3, 15, 222, -1),
+ // tmp index
+ Sql("create unique hash index xx91a on tt91a(b)",
+ SQL_DIAG_CREATE_INDEX, -1, -1, -1, -1),
+ Sql("select x.b, y.b, z.b from tt91a x, tt91a y, tt91a z where x.b + y.b = z.b order by x.b",
+ SQL_DIAG_SELECT_CURSOR, 3, 15, 222, -1),
+ Sql("drop index xx91a on tt91a",
+ SQL_DIAG_DROP_INDEX, -1, -1, -1, -1),
+ // add some duplicates
+ Sql("insert into tt91a values (4, 222, 'ccc')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("insert into tt91a values (5, 333, 'bbb')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("insert into tt91a values (6, 333, 'bbb')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ // check rows: 1,111,aaa + 2 * 2,222,ccc + 3 * 3,333,bbb
+ Sql("select count(*) from tt91a",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 6, -1),
+ Sql("select a+b from tt91a where (b = 111 or b = 222 ) and (b = 222 or b = 333) and a > 1 and a < 3",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 224, -1),
+ Sql("select sum(a) from tt91a having min(a) = 1 and max(a) = 6",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 21, -1),
+ Sql("select sum(a) from tt91a where a = 2 or a = 4 having min(a) = 2 and max(a) = 4",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 6, -1),
+ Sql("select sum(a) from tt91a having min(a) = 1 and max(a) = 5",
+ SQL_DIAG_SELECT_CURSOR, 0, -1, -1, -1),
+ Sql("select sum(a), b from tt91a group by b order by b",
+ SQL_DIAG_SELECT_CURSOR, 3, -1, 14, -1),
+ Sql("select sum(a), b, c from tt91a group by b, c order by c",
+ SQL_DIAG_SELECT_CURSOR, 3, -1, 6, -1),
+ Sql("select b, sum(a) from tt91a group by b having b = 37 * sum(a)",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 222, -1),
+ // simple varchar vs interpreter test
+ Sql("select count(*) from tt91a where c = 'ccc'",
+ SQL_DIAG_SELECT_CURSOR, 1, 2, 2, -1),
+ Sql("select count(*) from tt91a where c like '%b%'",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 3, -1),
+ // interpreter limits (crashes in api on v211)
+#if NDB_VERSION_MAJOR >= 3
+ Sql("select count(*) from tt91a where a in (99,98,97,96,95,94,93,92,91,90,89,88,87,86,85,84,83,82,81,80,79,78,77,76,75,74,73,72,71,70,69,68,67,66,65,64,63,62,61,60,59,58,57,56,55,54,53,52,51,50,49,48,47,46,45,44,43,42,41,40,39,38,37,36,35,34,33,32,31,30,29,28,27,26,25,24,23,22,21,20,19,18,17,16,15,14,13,12,11,10,9,8,7,6,5,4,3,2,99,98,97,96,95,94,93,92,91,90,89,88,87,86,85,84,83,82,81,80,79,78,77,76,75,74,73,72,71,70,69,68,67,66,65,64,63,62,61,60,59,58,57,56,55,54,53,52,51,50,49,48,47,46,45,44,43,42,41,40,39,38,37,36,35,34,33,32,31,30,29,28,27,26,25,24,23,22,21,20,19,18,17,16,15,14,13,12,11,10,9,8,7,6,5,4,3,2,99,98,97,96,95,94,93,92,91,90,89,88,87,86,85,84,83,82,81,80,79,78,77,76,75,74,73,72,71,70,69,68,67,66,65,64,63,62,61,60,59,58,57,56,55,54,53,52,51,50,49,48,47,46,45,44,43,42,41,40,39,38,37,36,35,34,33,32,31,30,29,28,27,26,25,24,23,22,21,20,19,18,17,16,15,14,13,12,11,10,9,8,7,6,5,4,3,2,99,98,97,96,95,94,93,92,91,90,89,88,87,86,85,84,83,82,81,80,79,78,77,76,75,74,73,72,71,70,69,68,67,66,65,64,63,62,61,60,59,58,57,56,55,54,53,52,51,50,49,48,47,46,45,44,43,42,41,40,39,38,37,36,35,34,33,32,31,30,29,28,27,26,25,24,23,22,21,20,19,18,17,16,15,14,13,12,11,10,9,8,7,6,5,4,3,2,99,98,97,96,95,94,93,92,91,90,89,88,87,86,85,84,83,82,81,80,79,78,77,76,75,74,73,72,71,70,69,68,67,66,65,64,63,62,61,60,59,58,57,56,55,54,53,52,51,50,49,48,47,46,45,44,43,42,41,40,39,38,37,36,35,34,33,32,31,30,29,28,27,26,25,24,23,22,21,20,19,18,17,16,15,14,13,12,11,10,9,8,7,6,5,4,3,2,99,98,97,96,95,94,93,92,91,90,89,88,87,86,85,84,83,82,81,80,79,78,77,76,75,74,73,72,71,70,69,68,67,66,65,64,63,62,61,60,59,58,57,56,55,54,53,52,51,50,49,48,47,46,45,44,43,42,41,40,39,38,37,36,35,34,33,32,31,30,29,28,27,26,25,24,23,22,21,20,19,18,17,16,15,14,13,12,11,10,9,8,7,6,5,4,3,2)",
+ SQL_DIAG_SELECT_CURSOR, 1, 5, 5, -1),
+ Sql("select count(*) from tt91a where c in ('xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','bbb','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy','zzzzz','xxxxx','yyyyy')",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 3, -1),
+#endif
+ // distinct
+ Sql("select distinct b from tt91a order by b",
+ SQL_DIAG_SELECT_CURSOR, 3, -1, 333, -1),
+ // some illegal groupings
+ Sql("select a from tt91a group by b",
+ -1, -1, -1, -1, -1,
+ SQL_ERROR, "IM000", -1, -1),
+ Sql("select sum(a) from tt91a group by b having a = 2",
+ -1, -1, -1, -1, -1,
+ SQL_ERROR, "IM000", -1, -1),
+ Sql("select sum(a) from tt91a group by b order by a",
+ -1, -1, -1, -1, -1,
+ SQL_ERROR, "IM000", -1, -1),
+ // string functions
+ Sql("insert into tt91a (c, b, a) values ('abcdef', 999, 9)",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("select count(*) from tt91a where left(c, 2) = 'ab' and substr(c, 3, 2) = 'cd' and right(c, 2) = 'ef'",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 1, -1),
+ // nulls
+ Sql("update tt91a set c = null where a > 8",
+ SQL_DIAG_UPDATE_WHERE, 1, -1, -1, -1),
+ Sql("select a from tt91a where c is null and b is not null order by a",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 9, -1),
+ Sql("select a from tt91a where not (c is not null or b is null) order by a",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 9, -1),
+ // null value guard in interpreter
+ Sql("select count(*) from tt91a where c < 'x' or c > 'x' or c != 'x' or c = 'x'",
+ SQL_DIAG_SELECT_CURSOR, 1, 6, 6, -1),
+ Sql("delete from tt91a where c is null",
+ SQL_DIAG_DELETE_WHERE, 1, -1, -1, -1),
+ // indexes
+ Sql("update tt91a set b = a + 5",
+ SQL_DIAG_UPDATE_WHERE, 6, 6, -1, -1),
+ Sql("create unique hash index xx91a on tt91a(b)",
+ SQL_DIAG_CREATE_INDEX, -1, -1, -1, -1),
+ // scan y primary key x
+ Sql("select x.b from tt91a x, tt91a y where x.a = y.b + 0",
+ SQL_DIAG_SELECT_CURSOR, 1, 7, 11, -1),
+ // scan x index y
+ Sql("select x.b from tt91a x, tt91a y where x.a + 0 = y.b",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 11, -1),
+ // scan x scan y
+ Sql("select x.b from tt91a x, tt91a y where x.a + 0 = y.b + 0",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 11, -1),
+ // dml ops
+ Sql("delete from tt91a where b = 11 and a > 999",
+ SQL_DIAG_DELETE_WHERE, 0, 1, -1, -1,
+ SQL_NO_DATA, 0, 0, true),
+ Sql("delete from tt91a where b = 11",
+ SQL_DIAG_DELETE_WHERE, 1, 0, -1, -1),
+ Sql("delete from tt91a where b = 11",
+ SQL_DIAG_DELETE_WHERE, 0, 0, -1, -1,
+ SQL_NO_DATA, 0, 0, true),
+ Sql("update tt91a set b = 10*10 where b = 10",
+ SQL_DIAG_UPDATE_WHERE, 1, 0, -1, -1),
+ Sql("update tt91a set b = 10 where b = 10*10",
+ SQL_DIAG_UPDATE_WHERE, 1, 0, -1, -1),
+ Sql("update tt91a set b = 10*10 where b = 10 and b >= 10",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ Sql("update tt91a set b = 10 where b = 10*10 and b >= 10*10",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ // char vs varchar
+ Sql("drop table tt91b",
+ SQL_DIAG_DROP_TABLE, -1, -1, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table tt91b (a int primary key, b char(5), c varchar(5))",
+ SQL_DIAG_CREATE_TABLE, -1, -1, -1, -1),
+ Sql("insert into tt91b values (1, 'abc', 'abc')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("insert into tt91b values (2, 'xyz', 'xyz')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("insert into tt91b values (3, 'xyz', 'xyz ')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ // char = char strips blanks
+ Sql("select count(*) from tt91b x where (x.b = 'abc') or x.a = x.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 1, -1),
+ Sql("select count(*) from tt91b x where (x.b = 'abc')",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql("select count(*) from tt91b x where (x.b = 'abc ') or x.a = x.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 1, -1),
+ Sql("select count(*) from tt91b x where (x.b = 'abc ')",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ // varchar = char
+ Sql("select count(*) from tt91b x where (x.c = 'abc') or x.a = x.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 1, -1),
+ Sql("select count(*) from tt91b x where (x.c = 'abc')",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql("select count(*) from tt91b x where (x.c = 'abc ') or x.a = x.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 0, -1),
+ Sql("select count(*) from tt91b x where (x.c = 'abc ')",
+ SQL_DIAG_SELECT_CURSOR, 1, 0, 0, -1),
+ // char = varchar
+ Sql("select count(*) from tt91b x, tt91b y where (x.b = y.c) or x.a = x.a+1 or y.a = y.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 2, -1),
+ Sql("select count(*) from tt91b x, tt91b y where (x.b = y.c)",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 2, -1),
+ // varchar = varchar
+ Sql("select count(*) from tt91b x, tt91b y where (x.c = y.c) or x.a = x.a+1 or y.a = y.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 3, -1),
+ Sql("select count(*) from tt91b x, tt91b y where (x.c = y.c)",
+ SQL_DIAG_SELECT_CURSOR, 1, -1, 3, -1),
+ // less
+ Sql("select 10 * x.a + y.a from tt91b x, tt91b y where (x.b < y.b) or x.a = x.a+1 or y.a = y.a+1 order by x.a, y.a",
+ SQL_DIAG_SELECT_CURSOR, 2, -1, 13, -1),
+ Sql("select 10 * x.a + y.a from tt91b x, tt91b y where (x.b < y.b) order by x.a, y.a",
+ SQL_DIAG_SELECT_CURSOR, 2, -1, 13, -1),
+ Sql("select 10 * x.a + y.a from tt91b x, tt91b y where (x.c < y.c) or x.a = x.a+1 or y.a = y.a+1 order by x.a, y.a",
+ SQL_DIAG_SELECT_CURSOR, 3, -1, 23, -1),
+ Sql("select 10 * x.a + y.a from tt91b x, tt91b y where (x.c < y.c) order by x.a, y.a",
+ SQL_DIAG_SELECT_CURSOR, 3, -1, 23, -1),
+ // like
+ Sql("select count(*) from tt91b x where (x.b like 'a%') or x.a = x.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 1, -1),
+ Sql("select count(*) from tt91b x where (x.b like 'a%')",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql("select count(*) from tt91b x where (x.b like 'x%z') or x.a = x.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 0, -1),
+ Sql("select count(*) from tt91b x where (x.b like 'x%z')",
+ SQL_DIAG_SELECT_CURSOR, 1, 0, 0, -1),
+ Sql("select count(*) from tt91b x where (x.a+0 = 2 and x.c like 'x%z') or x.a = x.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 1, -1),
+ Sql("select count(*) from tt91b x where (x.a+0 = 2 and x.c like 'x%z')",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql("select count(*) from tt91b x where (x.a+0 = 3 and x.c like 'x%z ') or x.a = x.a+1",
+ SQL_DIAG_SELECT_CURSOR, 1, 3, 1, -1),
+ Sql("select count(*) from tt91b x where (x.a+0 = 3 and x.c like 'x%z ')",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql()
+};
+
+// 92
+
+static void
+testMisc92a(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ char sql[MAX_SQL];
+ char tname[20];
+ sprintf(tname, "tt92%c", 0140 + test.m_no);
+ if (test.m_loop == 1) {
+ lock_mutex();
+ sprintf(sql, "drop table %s", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.exp(SQL_ERROR, "IM000", 2040709, false);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ sprintf(sql, "create table %s (a int unsigned primary key, b int unsigned not null)", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ unlock_mutex();
+ } else {
+ sprintf(sql, "delete from %s", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ test.run(HStmt(hStmt), SQLEndTran(SQL_HANDLE_DBC, hDbc, SQL_COMMIT));
+ }
+ for (int on = true; on >= false; on--) {
+ if (opt.m_v >= 2)
+ ndbout << "set autocommit " << (on ? "ON" : "OFF") << endl;
+ setAutocommit(test, hDbc, on);
+ // insert rows
+ if (opt.m_v >= 2)
+ ndbout << "SQL: insert into " << tname << " ..." << opt.m_scale << endl;
+ for (unsigned k = 0; k < opt.m_scale; k++) {
+ sprintf(sql, "insert into %s values (%u, %u)", tname, k, 10 * k);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ // commit always
+ test.run(HStmt(hStmt), SQLEndTran(SQL_HANDLE_DBC, hDbc, SQL_COMMIT));
+ // scan delete
+ sprintf(sql, "delete from %s", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // rollback or commit
+ test.run(HStmt(hStmt), SQLEndTran(SQL_HANDLE_DBC, hDbc, on ? SQL_COMMIT : SQL_ROLLBACK));
+ // count
+ long count = -1;
+ sprintf(sql, "select count(*) from %s", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ selectCount(test, hStmt, sql, &count);
+ test.chk(HStmt(hStmt), count == on ? 0 : opt.m_scale, "%s: got %d != %d", tname, (int)count, (int)opt.m_scale);
+ }
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+static const Sql
+miscSql92[] = {
+ // create in C func
+ Sql("testMisc92a", testMisc92a),
+ Sql()
+};
+
+// 93
+
+static void
+testMisc93a(Test& test)
+{
+ SQLHANDLE hEnv[2], hDbc[2], hStmt[2];
+ allocAll(test, hEnv[0], hDbc[0], hStmt[0]);
+ allocAll(test, hEnv[1], hDbc[1], hStmt[1]);
+ char sql[MAX_SQL];
+ // select via primary key
+ setAutocommit(test, hDbc[0], false);
+ sprintf(sql, "select c1 from tt93a where c0 = 1");
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt[0]), SQLExecDirect(hStmt[0], (SQLCHAR*)sql, SQL_NTS));
+ // update via another trans must time out
+ sprintf(sql, "update tt93a set c1 = 'b' where c0 = 1");
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt[1]), SQLExecDirect(hStmt[1], (SQLCHAR*)sql, SQL_NTS));
+ freeAll(test, hEnv[0], hDbc[0], hStmt[0]);
+ freeAll(test, hEnv[1], hDbc[1], hStmt[1]);
+}
+
+static const Sql
+miscSql93[] = {
+ // create in C func
+ Sql("drop table tt93a",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table tt93a (c0 int primary key, c1 char(10))",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ Sql("insert into tt93a values(1, 'a')",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ Sql("testMisc93a", testMisc93a),
+ Sql()
+};
+
+// 95
+
+static const Sql
+miscSql95[] = {
+ Sql("drop table tt95a",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table tt95a (a int not null, b char(10) not null, c int not null, d char(10), primary key(a, b)) storage(small)",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ // ordered index create and drop
+ Sql("create index xx95a on tt95a (c, d) nologging",
+ SQL_DIAG_CREATE_INDEX, -1, -1, -1, -1),
+ Sql("drop index xx95a on tt95a",
+ SQL_DIAG_DROP_INDEX, -1, -1, -1, -1),
+ Sql("create index xx95a on tt95a (c) nologging",
+ SQL_DIAG_CREATE_INDEX, -1, -1, -1, -1),
+ Sql("insert into tt95a values(1, 'a', 10, 'b')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("insert into tt95a values(2, 'a', 20, 'b')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("insert into tt95a values(3, 'a', 30, 'b')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("select a from tt95a where c = 20",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 2, -1),
+ Sql("delete from tt95a where c = 10",
+ SQL_DIAG_DELETE_WHERE, 1, 1, -1, -1),
+ Sql("update tt95a set c = 300 where c = 30",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ Sql("delete from tt95a where c = 300",
+ SQL_DIAG_DELETE_WHERE, 1, 1, -1, -1),
+ Sql("delete from tt95a",
+ SQL_DIAG_DELETE_WHERE, 1, 1, -1, -1),
+ // simple insert and rollback
+ Sql("-- simple insert and rollback"),
+ Sql(Sql::set_autocommit_off()),
+ Sql("insert into tt95a values(1, 'a', 10, 'b')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("select count(*) from tt95a",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql("select count(*) from tt95a where c = 10",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql(Sql::do_rollback()),
+ Sql(Sql::set_autocommit_on()),
+ Sql("select count(*) from tt95a",
+ SQL_DIAG_SELECT_CURSOR, 1, 0, 0, -1),
+ // simple update and rollback
+ Sql("-- simple update and rollback"),
+ Sql("insert into tt95a values(1, 'a', 10, 'b')",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql(Sql::set_autocommit_off()),
+ Sql("update tt95a set c = 20 where c = 10",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ Sql("select count(*) from tt95a",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql("select count(*) from tt95a where c = 20",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql(Sql::do_rollback()),
+ Sql(Sql::set_autocommit_on()),
+ Sql("select count(*) from tt95a where c = 10",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ // simple delete and rollback
+ Sql("-- simple delete and rollback"),
+ Sql(Sql::set_autocommit_off()),
+ Sql("delete from tt95a where c = 10",
+ SQL_DIAG_DELETE_WHERE, 1, 1, -1, -1),
+ Sql("select count(*) from tt95a",
+ SQL_DIAG_SELECT_CURSOR, 0, 0, 0, -1),
+ Sql("select count(*) from tt95a where c = 10",
+ SQL_DIAG_SELECT_CURSOR, 0, 0, 0, -1),
+ Sql(Sql::do_rollback()),
+ Sql(Sql::set_autocommit_on()),
+ Sql("select count(*) from tt95a where c = 10",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ // multiple update
+ Sql("-- multiple update and rollback"),
+ Sql(Sql::set_autocommit_off()),
+ Sql("update tt95a set c = 20 where c = 10",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ Sql("select count(*) from tt95a where c = 20",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql("update tt95a set c = 30 where c = 20",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ Sql("select count(*) from tt95a where c = 30",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql("update tt95a set c = 40 where c = 30",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ Sql("select count(*) from tt95a where c = 40",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql("update tt95a set c = 50 where c = 40",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ Sql("select count(*) from tt95a where c = 50",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ Sql(Sql::do_rollback()),
+ Sql(Sql::set_autocommit_on()),
+ Sql("select count(*) from tt95a where c = 10",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1, -1),
+ // another variant which found no tuple via index (aligment issue)
+ Sql("drop table tt95b",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table tt95b (a int primary key, b char(10) not null, c int not null)",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ Sql("create index xx95b on tt95b (b, c) nologging",
+ SQL_DIAG_CREATE_INDEX, -1, -1, -1, -1),
+ Sql("insert into tt95b values(0,'0123456789',1)",
+ SQL_DIAG_INSERT, 1, -1, -1, -1),
+ Sql("select a from tt95b where b='0123456789'",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 0, -1),
+ // update index key to different value
+ Sql("update tt95b set b = '9876543210' where b = '0123456789'",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+ // same value goes nuts...
+ Sql("update tt95b set b = '9876543210'",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+#if 0
+ // ...if done via index key (variant of halloween problem)
+ Sql("update tt95b set b = '9876543210' where b = '9876543210'",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, -1),
+#endif
+ Sql()
+};
+
+// 96
+
+static void
+testMisc96a(Test& test)
+{
+ // single thread
+ if (test.m_no != 1)
+ return;
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ char sql[MAX_SQL], *sqlptr;
+ char tname[20];
+ strcpy(tname, "tt96a");
+ // drop table
+ scopy(sqlptr = sql, "drop table %s", tname);
+ test.exp(SQL_ERROR, "IM000", 2040709, false);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // create table with many attributes
+ unsigned attrs = 1 + opt.m_scale;
+ if (attrs > MAX_ATTRIBUTES_IN_TABLE)
+ attrs = MAX_ATTRIBUTES_IN_TABLE;
+ if (attrs > 64)
+ attrs = 64;
+ scopy(sqlptr = sql, "create table %s (c0 int primary key", tname);
+ for (unsigned j = 1; j < attrs; j++) {
+ if (j % 2 == 0)
+ scopy(sqlptr, ", c%d int unsigned not null", j);
+ else
+ scopy(sqlptr, ", c%d char(10) not null", j);
+ }
+ scopy(sqlptr, ")");
+ if (opt.m_fragtype != 0)
+ scopy(sqlptr, " storage(%s)", opt.m_fragtype);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // create or drop indexes
+ const unsigned seed = 1000037 * test.m_loop + 1000039 * opt.m_scale;
+ srandom(seed);
+ const unsigned imax = opt.m_scale < 20 ? opt.m_scale : 20;
+ AttributeMask* imasks = new AttributeMask[imax];
+ unsigned ccnt = 0;
+ unsigned dcnt = 0;
+ for (unsigned n = 0; n < imax; n++)
+ imasks[n].clear();
+ while (ccnt + dcnt < opt.m_scale) {
+ char iname[20];
+ unsigned n = urandom(imax);
+ sprintf(iname, "xx96a%02d", n);
+ AttributeMask& imask = imasks[n];
+ unsigned sel = urandom(10);
+ if (imask.isclear()) {
+ // create one
+ unsigned ncol = 0;
+ unsigned cols[MAX_ATTRIBUTES_IN_INDEX];
+ unsigned cnum = urandom(attrs);
+ cols[ncol++] = cnum;
+ while (ncol < MAX_ATTRIBUTES_IN_INDEX) {
+ unsigned sel2 = urandom(10);
+ if (sel2 < 2)
+ break;
+ unsigned cnum2 = urandom(attrs);
+ if (sel2 < 9 && cnum2 == 0)
+ continue;
+ unsigned j;
+ for (j = 0; j < ncol; j++) {
+ if (cols[j] == cnum2)
+ break;
+ }
+ if (j == ncol)
+ cols[ncol++] = cnum2;
+ }
+ if (sel < 3) {
+ scopy(sqlptr = sql, "create unique hash index %s on %s (", iname, tname);
+ for (unsigned j = 0; j < ncol; j++)
+ scopy(sqlptr, "%sc%d", j == 0 ? "" : ", ", cols[j]);
+ scopy(sqlptr, ")");
+ } else {
+ scopy(sqlptr = sql, "create index %s on %s (", iname, tname);
+ for (unsigned j = 0; j < ncol; j++)
+ scopy(sqlptr, "%sc%d", j == 0 ? "" : ", ", cols[j]);
+ scopy(sqlptr, ") nologging");
+ }
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ for (unsigned j = 0; j < ncol; j++)
+ imask.set(cols[j]);
+ ccnt++;
+ } else if (sel < 5 && ccnt > dcnt + 1) {
+ scopy(sqlptr = sql, "drop index %s on %s", iname, tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ imask.clear();
+ dcnt++;
+ }
+ }
+ // insert unique data
+ unsigned rows = opt.m_scale;
+ unsigned* uval = new unsigned[rows];
+ for (unsigned i = 0; i < rows; i++) {
+ uval[i] = urandom(4);
+ scopy(sqlptr = sql, "insert into %s values(", tname);
+ for (unsigned j = 0; j < attrs; j++) {
+ if (j != 0)
+ scopy(sqlptr, ",");
+ unsigned v = (i << 10) | (j << 2) | uval[i];
+ if (j == 0)
+ scopy(sqlptr, "%u", i);
+ else if (j % 2 == 0)
+ scopy(sqlptr, "%u", v);
+ else
+ scopy(sqlptr, "'%010u'", v);
+ }
+ scopy(sqlptr, ")");
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ // update each row via random index
+ for (unsigned i = 0; i < rows; i++) {
+ unsigned uold = uval[i];
+ uval[i] = 3 - uval[i];
+ AttributeMask imask;
+ do {
+ unsigned j = urandom(imax);
+ imask = imasks[j];
+ } while (imask.isclear());
+ scopy(sqlptr = sql, "update %s set", tname);
+ for (unsigned j = 1; j < attrs; j++) {
+ if (j != 1)
+ scopy(sqlptr, ",");
+ /*
+ * Equality update is just barely doable before savepoints
+ * provided we change value of keys in every index.
+ */
+ unsigned v = (i << 10) | (j << 2) | uval[i];
+ if (j == 0)
+ ;
+ else if (j % 2 == 0)
+ scopy(sqlptr, " c%d=%u", j, v);
+ else
+ scopy(sqlptr, " c%d='%010u'", j, v);
+ }
+ scopy(sqlptr, " where 1=1");
+ while (! imask.isclear()) {
+ unsigned j = urandom(attrs);
+ if (imask.get(j)) {
+ unsigned v = (i << 10) | (j << 2) | uold;
+ scopy(sqlptr, " and c%d=", j);
+ if (j == 0)
+ scopy(sqlptr, "%u", i);
+ else if (j % 2 == 0)
+ scopy(sqlptr, "%u", v);
+ else
+ scopy(sqlptr, "'%010u'", v);
+ imask.clear(j);
+ }
+ }
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ chkRowCount(test, hStmt, 1);
+ }
+ // delete all
+ scopy(sqlptr = sql, "delete from %s", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ //
+ if (opt.m_v >= 2)
+ ndbout << tname << ": creates " << ccnt << " drops " << dcnt << endl;
+ delete [] imasks;
+ delete [] uval;
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+static const Sql
+miscSql96[] = {
+ Sql("testMisc96a", testMisc96a),
+ Sql()
+};
+
+// 97
+
+static void
+testMisc97a(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ const char* tname = "TT97A";
+ const char* iname = "XX97A";
+ char sql[MAX_SQL];
+ // create in some thread
+ lock_mutex();
+ if (my_sema == 0) {
+ if (opt.m_v >= 1)
+ ndbout << "thread " << test.m_no << " does setup" << endl;
+ sprintf(sql, "drop table %s", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL[" << test.m_no << "]: " << sql << endl;
+ test.exp(SQL_ERROR, "IM000", 2040709, false);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // a-pk b-index c-counter
+ sprintf(sql, "create table %s (a int primary key, b int, c int) storage(small)", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL[" << test.m_no << "]: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ for (unsigned i = 0; i < opt.m_scale; i++) {
+ sprintf(sql, "insert into %s values (%d, %d, %d)", tname, i, 10 * i, 0);
+ if (opt.m_v >= 3)
+ ndbout << "SQL[" << test.m_no << "]: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ sprintf(sql, "create index %s on %s (b) nologging", iname, tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL[" << test.m_no << "]: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ my_sema = 1;
+ }
+ unlock_mutex();
+ assert(my_sema == 1);
+ // parallel run - default rotating pk, ts, is
+ // frob: low 3 hex digits give alt sequence e.g. 0x311 = pk, pk, is
+ // frob: 4-th hex digit non-zero says use NDB API e.g. 0x1000
+ unsigned typelist[3] = { 1, 2, 3 };
+ for (unsigned i = 0; i < 3; i++) {
+ unsigned t = (opt.m_frob >> (i * 4)) & 0xf;
+ if (t != 0)
+ typelist[i] = t;
+ }
+ unsigned type = typelist[(test.m_no - 1) % 3];
+ if ((opt.m_frob & 0xf000) == 0) {
+ for (unsigned i = 0; i < opt.m_scale; i++) {
+ if (type == 1) {
+ // pk update
+ sprintf(sql, "update %s set c = c + 1 where a = %d", tname, i % opt.m_scale);
+ if (opt.m_v >= 3)
+ ndbout << lock << "SQL[" << test.m_no << "]: " << sql << endl << unlock;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ if (type == 2) {
+ // table scan update
+ sprintf(sql, "update %s set c = c + 1 where b + 0 = %d", tname, 10 * i);
+ if (opt.m_v >= 3)
+ ndbout << lock << "SQL[" << test.m_no << "]: " << sql << endl << unlock;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ if (type == 3) {
+ // index scan update
+ sprintf(sql, "update %s set c = c + 1 where b = %d", tname, 10 * i);
+ if (opt.m_v >= 3)
+ ndbout << lock << "SQL[" << test.m_no << "]: " << sql << endl << unlock;
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ }
+ }
+ } else {
+#ifdef ndbODBC
+#define CHK(o, x) do { if (! (x)) { fatal("line %d: %d %s", __LINE__, o->getNdbError().code, o->getNdbError().message); } } while (0)
+ Ndb* ndb = new Ndb("TEST_DB");
+ ndb->init();
+ CHK(ndb, ndb->waitUntilReady() == 0);
+ Int32 a, b, c;
+ for (unsigned i = 0; i < opt.m_scale; i++) {
+ if (type == 1) {
+ // pk update with exclusive read
+ NdbConnection* con;
+ NdbOperation* op;
+ CHK(ndb, (con = ndb->startTransaction()) != 0);
+ a = i;
+ c = -1;
+ CHK(con, (op = con->getNdbOperation(tname)) != 0);
+ CHK(op, op->readTupleExclusive() == 0);
+ CHK(op, op->equal((unsigned)0, (char*)&a, 0) == 0);
+ CHK(op, op->getValue(2, (char*)&c) != 0);
+ CHK(con, con->execute(NoCommit) == 0);
+ c = c + 1;
+ CHK(con, (op = con->getNdbOperation(tname)) != 0);
+ CHK(op, op->updateTuple() == 0);
+ CHK(op, op->equal((unsigned)0, (char*)&a, 0) == 0);
+ CHK(op, op->setValue(2, (char*)&c) == 0);
+ CHK(con, con->execute(Commit) == 0);
+ ndb->closeTransaction(con);
+ if (opt.m_v >= 3)
+ ndbout << lock << "thr " << test.m_no << " pk a=" << i << " c=" << c << endl << unlock;
+ }
+ if (type == 2) {
+ // table scan update
+ NdbConnection* con;
+ NdbOperation* op;
+ CHK(ndb, (con = ndb->startTransaction()) != 0);
+ CHK(con, (op = con->getNdbOperation(tname)) != 0);
+ CHK(con, op->openScanExclusive(240) == 0);
+ CHK(op, op->getValue((unsigned)0, (char*)&a) != 0);
+ CHK(op, op->getValue(2, (char*)&c) != 0);
+ CHK(con, con->executeScan() == 0);
+ unsigned rows = 0;
+ unsigned updates = 0;
+ while (1) {
+ int ret;
+ a = -1;
+ c = -1;
+ CHK(con, (ret = con->nextScanResult()) == 0 || ret == 1);
+ if (ret == 1)
+ break;
+ rows++;
+ if (a == i) {
+ NdbConnection* con2;
+ NdbOperation* op2;
+ CHK(ndb, (con2 = ndb->startTransaction()) != 0);
+ CHK(op, (op2 = op->takeOverForUpdate(con2)) != 0);
+ c = c + 1;
+ CHK(op2, op2->setValue(2, (char*)&c) == 0);
+ CHK(con2, con2->execute(Commit) == 0);
+ ndb->closeTransaction(con2);
+ updates++;
+ if (opt.m_v >= 3)
+ ndbout << lock << "thr " << test.m_no << " ts rows=" << rows << " a=" << i << " c=" << c << endl << unlock;
+ // test stop scan too
+ CHK(con, con->stopScan() == 0);
+ break;
+ }
+ }
+ ndb->closeTransaction(con);
+ test.chk(HStmt(hStmt), updates == 1, "got %u != 1", updates);
+ }
+ if (type == 3) {
+ // index scan update
+ NdbConnection* con;
+ NdbOperation* op;
+ CHK(ndb, (con = ndb->startTransaction()) != 0);
+ CHK(con, (op = con->getNdbOperation(iname, tname)) != 0);
+ CHK(con, op->openScanExclusive(240) == 0);
+ b = 10 * i;
+ CHK(con, op->setBound((unsigned)0, 4, &b, sizeof(b)) == 0);
+ CHK(op, op->getValue((unsigned)0, (char*)&a) != 0);
+ CHK(op, op->getValue(2, (char*)&c) != 0);
+ CHK(con, con->executeScan() == 0);
+ unsigned rows = 0;
+ unsigned updates = 0;
+ while (1) {
+ int ret;
+ a = -1;
+ c = -1;
+ CHK(con, (ret = con->nextScanResult()) == 0 || ret == 1);
+ if (ret == 1)
+ break;
+ rows++;
+ if (a == i) {
+ NdbConnection* con2;
+ NdbOperation* op2;
+ CHK(ndb, (con2 = ndb->startTransaction()) != 0);
+ CHK(op, (op2 = op->takeOverForUpdate(con2)) != 0);
+ c = c + 1;
+ CHK(op2, op2->setValue(2, (char*)&c) == 0);
+ CHK(con2, con2->execute(Commit) == 0);
+ ndb->closeTransaction(con2);
+ updates++;
+ if (opt.m_v >= 3)
+ ndbout << lock << "thr " << test.m_no << " is rows=" << rows << " a=" << i << " c=" << c << endl << unlock;
+ // test stop scan too
+ CHK(con, con->stopScan() == 0);
+ break;
+ }
+ }
+ ndb->closeTransaction(con);
+ test.chk(HStmt(hStmt), rows == 1, "got %u != 1", rows);
+ test.chk(HStmt(hStmt), updates == 1, "got %u != 1", updates);
+ }
+ }
+ delete ndb;
+#undef CHK
+#endif
+ }
+ // verify result
+ lock_mutex();
+ if (++my_sema == 1 + opt.m_threads) {
+ if (opt.m_v >= 1)
+ ndbout << "thread " << test.m_no << " does verification" << endl;
+ sprintf(sql, "select * from %s order by a", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL[" << test.m_no << "]: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ long a, b, c;
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1, SQL_C_SLONG, &a, 0, 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 2, SQL_C_SLONG, &b, 0, 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 3, SQL_C_SLONG, &c, 0, 0));
+ for (unsigned i = 0; i < opt.m_scale; i++) {
+ a = b = c = -1;
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ test.chk(HStmt(hStmt), a == i, "a: got %ld != %u", a, i);
+ test.chk(HStmt(hStmt), b == 10 * i, "b: got %ld != %u", b, 10 * i);
+ test.chk(HStmt(hStmt), c == opt.m_threads, "c: got %ld != %u", c, opt.m_threads);
+ if (opt.m_v >= 4)
+ ndbout << "verified " << i << endl;
+ }
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (opt.m_v >= 2)
+ ndbout << "thr " << test.m_no << " verified " << opt.m_scale << " rows" << endl;
+ my_sema = 0;
+ }
+ unlock_mutex();
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+static const Sql
+miscSql97[] = {
+ Sql("testMisc97a", testMisc97a),
+ Sql()
+};
+
+// 99
+
+static void
+testMisc99a(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ // bad
+ const char* sqlInsertBad = "insert into tt99a values(?, ?, ?, ?, ?)";
+ test.exp(SQL_ERROR, "21S01", -1, true);
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sqlInsertBad, SQL_NTS));
+ // good
+ const char* sqlInsert = "insert into tt99a (col1, col2, col3, col4, col5) values(?, ?, ?, ?, ?)";
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sqlInsert, SQL_NTS));
+ unsigned long value;
+ for (unsigned i = 1; i <= 5; i++) {
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, i, SQL_PARAM_INPUT, SQL_C_ULONG, SQL_INTEGER, 0, 0, &value, 0, 0));
+ }
+ const unsigned long base = 1000000000;
+ const unsigned long scale = 10;
+ for (value = base; value < base + scale; value++) {
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ }
+ // bug1: re-analyze of converted expression...
+ const char* sqlSelect = "select col5 from tt99a where col2 + 0 = ?";
+ unsigned long output;
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1, SQL_C_ULONG, &output, 0, 0));
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sqlSelect, SQL_NTS));
+ // bug2: previous bind must survive a new SQLPrepare
+ if (0) {
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1, SQL_PARAM_INPUT, SQL_C_ULONG, SQL_INTEGER, 0, 0, &value, 0, 0));
+ }
+ for (value = base; value < base + scale; value++) {
+ if (value > base + 4) {
+ // bug1: ...when IPD changed by JDBC
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1, SQL_PARAM_INPUT, SQL_C_ULONG, SQL_INTEGER, 0, 0, &value, 0, 0));
+ }
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ output = (unsigned long)-1;
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ test.chk(HStmt(hStmt), output == value, "got %lu != %lu", output, value);
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ test.timerCnt(1);
+ }
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+static void
+testMisc99c(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ const char* sql = "select b from tt99c where a = ?";
+ const unsigned long c1 = 2100000000U;
+ const unsigned long c2 = 4100000000U;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ unsigned long aval, bval;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1, SQL_PARAM_INPUT, SQL_C_ULONG, SQL_INTEGER, 0, 0, &aval, 0, 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1, SQL_C_ULONG, &bval, 0, 0));
+ // uno
+ for (unsigned i = 0; i < opt.m_scale; i++) {
+ aval = c1;
+ bval = (unsigned long)-1;
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << " [?=" << (Uint64)aval << "]" << endl;
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ test.chk(HStmt(hStmt), bval == c2, "got %lu != %lu", bval, c2);
+ //test.exp(SQL_NO_DATA, 0, 0, true);
+ //test.run(HStmt(hStmt), SQLFetch(hStmt));
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ }
+ // dos
+ for (unsigned i = 0; i < opt.m_scale; i++) {
+ break; // XXX not yet, hangs in NDB ?!?
+ aval = c2;
+ bval = (unsigned long)-1;
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << " [?=" << (Uint64)aval << "]" << endl;
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ test.chk(HStmt(hStmt), bval == c1, "got %lu != %lu", bval, c2);
+ //test.exp(SQL_NO_DATA, 0, 0, true);
+ //test.run(HStmt(hStmt), SQLFetch(hStmt));
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ }
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+static void
+testMisc99d(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ const char* tname = "TT99D";
+ char sql[MAX_SQL];
+ sprintf(sql, "drop table %s", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.exp(SQL_ERROR, "IM000", 2040709, false);
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ sprintf(sql, "create table %s (a bigint unsigned, b bigint, primary key (a))", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLExecDirect(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ sprintf(sql, "insert into %s values (?, ?)", tname);
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql << endl;
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ // XXX replace by 100 when signed vs unsigned resolved
+ const unsigned num = 78;
+ SQLUBIGINT aval;
+ SQLBIGINT bval;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1, SQL_PARAM_INPUT, SQL_C_UBIGINT, SQL_BIGINT, 0, 0, &aval, 0, 0));
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 2, SQL_PARAM_INPUT, SQL_C_SBIGINT, SQL_BIGINT, 0, 0, &bval, 0, 0));
+ for (SQLBIGINT i = 0; i < num; i++) {
+ if (opt.m_v >= 3)
+ ndbout << "insert " << i << endl;
+ aval = i * i * i * i * i * i * i * i * i * i; // 10
+ bval = -aval;
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ }
+ sprintf(sql, "select a, b from tt99d where a = ?");
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql, SQL_NTS));
+ SQLUBIGINT kval;
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1, SQL_PARAM_INPUT, SQL_C_UBIGINT, SQL_BIGINT, 0, 0, &kval, 0, 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1, SQL_C_UBIGINT, &aval, 0, 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 2, SQL_C_SBIGINT, &bval, 0, 0));
+ for (SQLBIGINT i = 0; i < num; i++) {
+ kval = i * i * i * i * i * i * i * i * i * i; // 10
+ if (opt.m_v >= 3)
+ ndbout << "fetch " << i << " key " << kval << endl;
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ aval = bval = 0;
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ test.chk(HStmt(hStmt), aval == kval && bval == -kval, "got %llu, %lld != %llu, %lld", aval, bval, kval, -kval);
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ }
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+static void
+testMiscC2(Test& test)
+{
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+#if 0
+ {
+ char POP[255];
+ char PORT[255];
+ char ACCESSNODE[255];
+
+ const char* sqlSelect = "select PORT from AAA where POP=? and ACCESSNODE=?";
+
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sqlSelect, SQL_NTS));
+
+ for (int j=0; j<5; j++) {
+ printf("Loop %u\n", j);
+ printf("LINE %u\n", __LINE__);
+
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_VARCHAR, 255, 0, POP, 255, 0));
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 2, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_VARCHAR, 255, 0, ACCESSNODE, 255, 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1, SQL_C_CHAR, PORT, 255, 0));
+
+ sprintf(POP, "a");
+ sprintf(ACCESSNODE, "b");
+
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ printf("got %s\n", PORT);
+ printf("LINE %u\n", __LINE__);
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ printf("LINE %u\n", __LINE__);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ printf("LINE %u\n", __LINE__);
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ printf("LINE %u\n", __LINE__);
+ }
+ }
+ return;
+#endif
+
+ char POP[255];
+ char PORT[255];
+ char ACCESSNODE[255];
+ unsigned long VLAN = 0;
+ unsigned long SNMP_INDEX = 0;
+ unsigned long PORT_STATE = 0;
+ unsigned long STATIC_PORT = 0;
+ unsigned long COMMENT = 0;
+
+ const char* sqlSelect = "select PORT, PORT_STATE from PORTS where POP=? and ACCESSNODE=?";
+
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sqlSelect, SQL_NTS));
+
+ for (int j=0; j<5; j++) {
+ printf("Loop %u\n", j);
+ printf("LINE %u\n", __LINE__);
+
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_VARCHAR, 255, 0, POP, 255, 0));
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 2, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_VARCHAR, 255, 0, ACCESSNODE, 255, 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1, SQL_C_CHAR, PORT, 255, 0));
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 2, SQL_C_ULONG, &PORT_STATE, 0, 0));
+
+ sprintf(POP, "row%u.i%u.bredband.com", 2, 3);
+ sprintf(ACCESSNODE, "as%u", 2);
+
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ for (int i=0; i < 3; i++) {
+ PORT_STATE=0;
+ sprintf(PORT, "XXXXXXXXXXXXXXXXXXXXX");
+
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ printf("got %s %lu\n", PORT, PORT_STATE);
+ // test.chk(HStmt(hStmt), false, "got %s != %s", "xxx", PORT);
+ }
+ printf("LINE %u\n", __LINE__);
+ test.exp(SQL_NO_DATA, 0, 0, true);
+ printf("LINE %u\n", __LINE__);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ printf("LINE %u\n", __LINE__);
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ printf("LINE %u\n", __LINE__);
+ }
+}
+
+static const Sql
+miscSqlC2[] = {
+ Sql("drop table PORTS",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table PORTS (POP varchar(200) not null, ACCESSNODE varchar(200) not null, PORT varchar(200) not null, VLAN int unsigned, SNMP_INDEX int unsigned, PORT_STATE int unsigned, STATIC_PORT int unsigned, COMMENT int unsigned, primary key (POP,ACCESSNODE,PORT))",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ Sql("create index xxPORTS on PORTS (POP, ACCESSNODE) nologging",
+ SQL_DIAG_CREATE_INDEX, -1, -1, -1, -1),
+ Sql("insert into PORTS values ('row2.i3.bredband.com','as2','Fa0/0',0,1,2,3,4)",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ Sql("insert into PORTS values ('row2.i3.bredband.com','as2','Fa0/1',1,2,3,4,5)",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ Sql("insert into PORTS values ('row2.i3.bredband.com','as2','Fa0/2',2,3,4,5,6)",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ Sql("select PORT, PORT_STATE from PORTS where POP='row2.i3.bredband.com' and ACCESSNODE='as2'",
+ SQL_DIAG_SELECT_CURSOR, 3, 3, -1, -1),
+
+ Sql("drop table AAA",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table AAA (POP varchar(200), ACCESSNODE varchar(200) not null, PORT varchar(200) not null, primary key (POP,ACCESSNODE,PORT))",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ Sql("create index xxAAA on AAA (POP, ACCESSNODE) nologging",
+ SQL_DIAG_CREATE_INDEX, -1, -1, -1, -1),
+ Sql("insert into AAA values ('a','b','A')",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+
+ Sql("testMiscC2", testMiscC2),
+ Sql()
+};
+
+/*
+> SELECT PORT, PORT_STATE FROM PORTS where pop=? and accessnode=?
+> SELECT VLAN, SNMP_INDEX, PORT_STATE, STATIC_PORT, COMMENT FROM PORTS WHERE POP=? AND ACCESSNODE=? AND PORT=?
+> select count(*) from ports
+> select snmp_index from ports where pop='row2.i3.bredband.com' and accessnode='as2' and port='Fa0/2'
+
+> SELECT MAC, MAC_EXPIRE, IP, IP_EXPIRE, HOSTNAME, DETECTED, STATUS, STATIC_DNS, BLOCKED, NUM_REQUESTS, ACCESSTYPE, OS_TYPE, GATE_WAY, DIRTY_FLAG, LOCKED_IP FROM CLIENTS WHERE PORT=? AND ACCESSNODE=? AND POP=?
+> SELECT SERVICES.ACCESSTYPE, SERVICES.NUM_IP, SERVICES.TEXPIRE, SERVICES.CUSTOMER_ID, SERVICES.LEASED_NUM_IP, SERVICES.PROVIDER, SERVICES.LOCKED_IP, SERVICES.STATIC_DNS, SERVICES.SUSPENDED_SERVICE FROM SERVICES , ACCESSTYPES WHERE SERVICES.PORT = ? AND SERVICES.ACCESSNODE = ? AND SERVICES.POP = ? AND SERVICES.ACCESSTYPE=ACCESSTYPES.ACCESSTYPE
+*/
+
+static const Sql
+miscSql99[] = {
+ Sql("drop table tt99a",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table tt99a (col1 int unsigned primary key, col2 int unsigned, col3 int unsigned, col4 int unsigned, col5 int unsigned, col6 varchar(7) default 'abc123')",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ // inserts 10 rows, all same, start value 1000000000
+ Sql("testMisc99a", testMisc99a),
+ // interpreted scan plus bind parameter
+ Sql("select col1 from tt99a where col2 = ?",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1000000004, 1000000004),
+ Sql("select col1 from tt99a where col2 = 1000000000 + ?",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1000000004, 4),
+ Sql("select col1 from tt99a where col2 = ? + 1000000000",
+ SQL_DIAG_SELECT_CURSOR, 1, 1, 1000000004, 4),
+ // same not interpreted, tuple count 10
+ Sql("select col1 from tt99a where col2 + 0 = 1000000000 + ?",
+ SQL_DIAG_SELECT_CURSOR, 1, 10, 1000000004, 4),
+ // varchar variations
+ Sql("select count(*) from tt99a where col6 = 'abc123'",
+ SQL_DIAG_SELECT_CURSOR, 1, 10, 10, -1),
+ Sql("select count(*) from tt99a where left(col6, ?) = 'abc1'",
+ SQL_DIAG_SELECT_CURSOR, 1, 10, 10, 4),
+ Sql("select count(*) from tt99a where left(col6, ?) = 'abc1'",
+ SQL_DIAG_SELECT_CURSOR, 1, 10, 0, 3),
+ // tpc-b inspired, wrong optimization to direct update
+ Sql("drop table tt99b",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table tt99b(a int primary key, b int not null, c double precision)",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ Sql("insert into tt99b values(1, 10, 100.0)",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ Sql("insert into tt99b values(9, 90, 900.0)",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ Sql("create unique hash index tt99y on tt99b (b)",
+ SQL_DIAG_CREATE_INDEX, -1, 0, -1, -1),
+ // first scan update..
+ Sql("update tt99b set c = c + ? where a+0 = 1",
+ SQL_DIAG_UPDATE_WHERE, 1, 2, -1, 10),
+ Sql("update tt99b set c = c + ? where b+0 = 10",
+ SQL_DIAG_UPDATE_WHERE, 1, 2, -1, 10),
+ // then optimized..
+ Sql("update tt99b set c = c + ? where a = 1",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, 10),
+ Sql("update tt99b set c = c + ? where b = 10",
+ SQL_DIAG_UPDATE_WHERE, 1, 1, -1, 10),
+ // verify..
+ Sql("select count(*) from tt99b where 100-1 < c and c < 140-1",
+ SQL_DIAG_SELECT_CURSOR, 1, 2, 0, -1),
+ Sql("select count(*) from tt99b where 140-.001 < c and c < 140+.001",
+ SQL_DIAG_SELECT_CURSOR, 1, 2, 1, -1),
+ // unsigned test
+ Sql("drop table tt99c",
+ SQL_DIAG_DROP_TABLE, -1, 0, -1, -1,
+ SQL_ERROR, "IM000", 2040709, false),
+ Sql("create table tt99c(a int unsigned primary key, b int unsigned)",
+ SQL_DIAG_CREATE_TABLE, -1, 0, -1, -1),
+ Sql("insert into tt99c values(2100000000, 4100000000)",
+ SQL_DIAG_INSERT, 1, 0, -1, -1),
+ Sql("insert into tt99c (a, b) select b, a from tt99c",
+ SQL_DIAG_INSERT, 1, 1, -1, -1),
+ Sql("testMisc99c", testMisc99c),
+ // new external type SQL_C_[SU]BIGINT
+ Sql("testMisc99d", testMisc99d),
+ Sql()
+};
+
+static const struct { const Sql* sql; int minscale; }
+miscSql[11] = {
+ { miscSql90, 0 },
+ { miscSql91, 0 },
+ { miscSql92, 0 },
+ { miscSql93, 0 },
+ { 0, 0 }, // 94
+ { miscSql95, 0 },
+ { miscSql96, 0 },
+ { miscSql97, 0 },
+ { 0, 0 }, // 98
+ { miscSql99, 0 },
+ { miscSqlC2, 0 }
+};
+
+static void
+testSql(Test& test)
+{
+ const unsigned salt = test.m_stuff; // mess
+ if (opt.m_scale < miscSql[salt].minscale) {
+ if (opt.m_v >= 1)
+ ndbout << "skip - requires scale >= " << miscSql[salt].minscale << endl;
+ return;
+ }
+ assert(0 <= salt && salt < 11 && miscSql[salt].sql != 0);
+ SQLHANDLE hEnv, hDbc, hStmt;
+ allocAll(test, hEnv, hDbc, hStmt);
+ for (unsigned i = 0; ; i++) {
+ const Sql& sql = miscSql[salt].sql[i];
+ if (sql.m_sql == 0)
+ break;
+ if (opt.m_v >= 2)
+ ndbout << "SQL: " << sql.m_sql << endl;
+ if (sql.m_testFunc != 0) {
+ (*sql.m_testFunc)(test);
+ continue;
+ }
+ if (strncmp(sql.m_sql, "--", 2) == 0) {
+ continue;
+ }
+ if (strcmp(sql.m_sql, Sql::set_autocommit_on()) == 0) {
+ setAutocommit(test, hDbc, true);
+ continue;
+ }
+ if (strcmp(sql.m_sql, Sql::set_autocommit_off()) == 0) {
+ setAutocommit(test, hDbc, false);
+ continue;
+ }
+ if (strcmp(sql.m_sql, Sql::do_commit()) == 0) {
+ test.run(HDbc(hDbc), SQLEndTran(SQL_HANDLE_DBC, hDbc, SQL_COMMIT));
+ continue;
+ }
+ if (strcmp(sql.m_sql, Sql::do_rollback()) == 0) {
+ test.run(HDbc(hDbc), SQLEndTran(SQL_HANDLE_DBC, hDbc, SQL_ROLLBACK));
+ continue;
+ }
+ if (opt.m_v >= 3) {
+ ndbout << "expect:";
+ ndbout << " ret=" << sql.m_ret;
+ ndbout << " rows=" << sql.m_rowCount;
+ ndbout << " tuples=" << sql.m_tuplesFetched;
+ ndbout << endl;
+ }
+ test.run(HStmt(hStmt), SQLFreeStmt(hStmt, SQL_UNBIND));
+ test.run(HStmt(hStmt), SQLFreeStmt(hStmt, SQL_RESET_PARAMS));
+ // prep
+ test.exp(sql.m_ret, sql.m_state, sql.m_native, false);
+ test.run(HStmt(hStmt), SQLPrepare(hStmt, (SQLCHAR*)sql.m_sql, SQL_NTS));
+ if (test.m_ret != SQL_SUCCESS)
+ continue;
+ // bind between prep and exec like JDBC
+ unsigned long bindValue = sql.m_bindValue;
+ for (int k = 0; k <= 1; k++) {
+ if (bindValue != -1) {
+ assert(strchr(sql.m_sql, '?') != 0);
+ test.run(HStmt(hStmt), SQLBindParameter(hStmt, 1, SQL_PARAM_INPUT, SQL_C_ULONG, SQL_INTEGER, 0, 0, &bindValue, 0, 0));
+ }
+ if (k == 0) {
+ if (bindValue != -1) {
+ test.run(HStmt(hStmt), SQLFreeStmt(hStmt, SQL_RESET_PARAMS));
+ // exec with unbound parameter
+ test.exp(SQL_ERROR, "HY010", -1, true);
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.chk(HStmt(hStmt), test.m_functionCode == sql.m_functionCode || sql.m_functionCode == -1, "func: got %d != %d", (int)test.m_functionCode, (int)sql.m_functionCode);
+ }
+ } else {
+ // exec
+ test.exp(sql.m_ret, sql.m_state, sql.m_native, sql.m_reset);
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ test.chk(HStmt(hStmt), test.m_functionCode == sql.m_functionCode || sql.m_functionCode == -1, "func: got %d != %d", (int)test.m_functionCode, (int)sql.m_functionCode);
+ }
+ }
+ if (sql.m_rowCount != -1) {
+ if (sql.m_functionCode == SQL_DIAG_SELECT_CURSOR) {
+ long lastValue;
+ if (sql.m_lastValue != -1)
+ test.run(HStmt(hStmt), SQLBindCol(hStmt, 1, SQL_C_SLONG, &lastValue, 0, 0));
+ unsigned k = 0;
+ do {
+ int rowCount = 0;
+ lastValue = -1;
+ while (1) {
+ test.exp(SQL_NO_DATA, 0, 0, false);
+ test.run(HStmt(hStmt), SQLFetch(hStmt));
+ if (test.m_ret == SQL_NO_DATA)
+ break;
+ rowCount++;
+ }
+ test.chk(HStmt(hStmt), rowCount == sql.m_rowCount, "rowCount: got %d != %d", (int)rowCount, (int)sql.m_rowCount);
+ if (sql.m_tuplesFetched != -1)
+ chkTuplesFetched(test, hStmt, sql.m_tuplesFetched);
+ if (rowCount > 0 && sql.m_lastValue != -1)
+ test.chk(HStmt(hStmt), lastValue == sql.m_lastValue, "lastValue: got %ld != %ld", (long)lastValue, (long)sql.m_lastValue);
+ test.run(HStmt(hStmt), SQLCloseCursor(hStmt));
+ if (++k >= opt.m_scale)
+ break;
+ test.run(HStmt(hStmt), SQLExecute(hStmt));
+ } while (1);
+ test.timerCnt(opt.m_scale);
+ } else {
+ assert(sql.m_lastValue == -1);
+ chkRowCount(test, hStmt, sql.m_rowCount);
+ if (sql.m_tuplesFetched != -1)
+ chkTuplesFetched(test, hStmt, sql.m_tuplesFetched);
+ test.timerCnt(1);
+ }
+ }
+ }
+ freeAll(test, hEnv, hDbc, hStmt);
+}
+
+// name, function, runmode, salt (0=const or n/a), description
+static const Case caseList[] = {
+ Case( "00alloc", testAlloc, Case::Thread, 0, "allocate handles" ),
+ Case( "01create", testCreate, Case::Single, 0, "create tables for the test" ),
+ Case( "02prepare", testPrepare, Case::Thread, 0, "prepare without execute" ),
+ Case( "03catalog", testCatalog, Case::Thread, 0, "catalog functions" ),
+ Case( "10insert", testInsert, Case::Thread, 1, "insert computed rows" ),
+ Case( "11delall", testDeleteAll, Case::Single, 0, "delete all rows via scan" ),
+ Case( "12insert", testInsert, Case::Thread, 1, "insert computed rows again" ),
+ Case( "13count", testCount, Case::Single, 0, "count rows" ),
+ Case( "14verpk", testVerifyPk, Case::Thread, 1, "verify via primary key" ),
+ Case( "15verscan", testVerifyScan, Case::Serial, 1, "verify via range scans" ),
+ Case( "16join", testJoin, Case::Single, 0, "multiple self-join" ),
+ Case( "17cart", testCart, Case::Single, 0, "cartesian join" ),
+ Case( "20updpk", testUpdatePk, Case::Thread, 2, "update via primary key" ),
+ Case( "21verpk", testVerifyPk, Case::Thread, 2, "verify via primary key" ),
+ Case( "22verscan", testVerifyScan, Case::Serial, 2, "verify via range scans" ),
+ Case( "23updscan", testUpdateScan, Case::Serial, 0, "update via scan" ),
+ Case( "24verpk", testVerifyPk, Case::Thread, 0, "verify via primary key" ),
+ Case( "25verscan", testVerifyScan, Case::Serial, 0, "verify via range scans" ),
+ Case( "26delpk", testDeletePk, Case::Thread, 0, "delete via primary key" ),
+ Case( "30trans", testTrans, Case::Single, 3, "rollback and commit" ),
+ Case( "31concur", testConcur, Case::Single, 0, "commit across open cursor" ),
+ Case( "32readcom", testReadcom, Case::Single, 0, "read committed" ),
+ Case( "40perf", testPerf, Case::Single, 0, "perf test prepare" ),
+ Case( "41perf", testPerf, Case::Thread, 1, "perf test NDB API" ),
+ Case( "42perf", testPerf, Case::Thread, 2, "perf test NDB ODBC" ),
+ Case( "90sql", testSql, Case::Single, 0, "misc SQL: metadata" ),
+ Case( "91sql", testSql, Case::Single, 1, "misc SQL: misc" ),
+ Case( "92sql", testSql, Case::Thread, 2, "misc SQL: scan rollback" ),
+ Case( "93sql", testSql, Case::Single, 3, "misc SQL: locking" ),
+ Case( "95sql", testSql, Case::Single, 5, "misc SQL: indexes (simple)" ),
+ Case( "96sql", testSql, Case::Single, 6, "misc SQL: indexes" ),
+ Case( "97sql", testSql, Case::Thread, 7, "misc SQL: indexes" ),
+ Case( "99sql", testSql, Case::Single, 9, "misc SQL: bug du jour" ),
+ Case( "C2", testSql, Case::Single, 10, "misc SQL: C2" )
+};
+
+static const unsigned caseCount = arraySize(caseList);
+
+static bool
+findCase(const char* name)
+{
+ for (unsigned i = 0; i < caseCount; i++) {
+ const Case& cc = caseList[i];
+ if (strstr(cc.m_name, name) != 0)
+ return true;
+ }
+ return false;
+}
+
+static void
+listCases()
+{
+ ndbout << "cases:" << endl;
+ unsigned m = 0;
+ for (unsigned i = 0; i < caseCount; i++) {
+ const Case& cc = caseList[i];
+ if (m < strlen(cc.m_name))
+ m = strlen(cc.m_name);
+ }
+ for (unsigned i = 0; i < caseCount; i++) {
+ const Case& cc = caseList[i];
+ char buf[200];
+ sprintf(buf, "%-*s [%-6s] %s", m, cc.m_name, cc.modename(), cc.m_desc);
+ ndbout << buf << endl;
+ }
+}
+
+// threads
+
+extern "C" { static void* testThr(void* arg); }
+
+struct Thr {
+ enum State {
+ Wait = 1, // wait for test case
+ Run = 2, // run the test case
+ Done = 3, // done with the case
+ Exit = 4 // exit thread
+ };
+ unsigned m_no; // thread number 1 .. max
+ NdbThread* m_thr; // thread id etc
+ const Case* m_case; // current case
+ State m_state; // condition variable
+ NdbMutex* m_mutex; // condition guard
+ NdbCondition* m_cond;
+ void* m_status; // exit status (not used)
+ Test m_test; // test runner
+ Thr(unsigned no, unsigned loop) :
+ m_no(no),
+ m_thr(0),
+ m_case(0),
+ m_state(Wait),
+ m_mutex(NdbMutex_Create()),
+ m_cond(NdbCondition_Create()),
+ m_status(0),
+ m_test(no, loop) {
+ }
+ ~Thr() {
+ destroy();
+ NdbCondition_Destroy(m_cond);
+ NdbMutex_Destroy(m_mutex);
+ }
+ void create() {
+ assert(m_thr == 0);
+ m_thr = NdbThread_Create(testThr, (void**)this, 64*1024, "test", NDB_THREAD_PRIO_LOW);
+ }
+ void destroy() {
+ if (m_thr != 0)
+ NdbThread_Destroy(&m_thr);
+ m_thr = 0;
+ }
+ void lock() {
+ NdbMutex_Lock(m_mutex);
+ }
+ void unlock() {
+ NdbMutex_Unlock(m_mutex);
+ }
+ void wait() {
+ NdbCondition_Wait(m_cond, m_mutex);
+ }
+ void signal() {
+ NdbCondition_Signal(m_cond);
+ }
+ void join() {
+ NdbThread_WaitFor(m_thr, &m_status);
+ m_thr = 0;
+ }
+ // called from main
+ void mainStart(const Case& cc) {
+ lock();
+ m_case = &cc;
+ m_state = Run;
+ signal();
+ unlock();
+ }
+ void mainStop() {
+ lock();
+ while (m_state != Done) {
+ if (opt.m_v >= 4)
+ ndbout << ::lock << "thr " << m_no << " [main] wait state=" << m_state << endl << ::unlock;
+ wait();
+ }
+ if (opt.m_v >= 4)
+ ndbout << ::lock << "thr " << m_no << " [main] done" << endl << ::unlock;
+ m_state = Wait;
+ unlock();
+ }
+ // run in thread
+ void testSelf() {
+ while (1) {
+ lock();
+ while (m_state != Run && m_state != Exit) {
+ if (opt.m_v >= 4)
+ ndbout << ::lock << "thr " << m_no << " [self] wait state=" << m_state << endl << ::unlock;
+ wait();
+ }
+ if (m_state == Run) {
+ if (opt.m_v >= 4)
+ ndbout << ::lock << "thr " << m_no << " [self] run" << endl << ::unlock;
+ assert(m_case != 0);
+ m_test.timerOn();
+ m_test.runCase(*m_case);
+ m_test.timerOff();
+ m_state = Done;
+ if (opt.m_v >= 4)
+ ndbout << ::lock << "thr " << m_no << " [self] done" << endl << ::unlock;
+ signal();
+ unlock();
+ } else if (m_state == Exit) {
+ unlock();
+ break;
+ } else {
+ assert(false);
+ }
+ }
+ if (opt.m_v >= 4)
+ ndbout << ::lock << "thr " << m_no << " [self] exit" << endl << ::unlock;
+ }
+};
+
+static void*
+testThr(void* arg)
+{
+ Thr& thr = *(Thr*)arg;
+ thr.testSelf();
+ return 0;
+}
+
+#ifdef DMALLOC
+extern "C" {
+
+static int malloc_bytes = 0;
+static int free_bytes = 0;
+
+static void
+malloc_track(const char *file, const unsigned int line, const int func_id, const DMALLOC_SIZE byte_size, const DMALLOC_SIZE alignment, const DMALLOC_PNT old_addr, const DMALLOC_PNT new_addr)
+{
+ if (func_id == DMALLOC_FUNC_MALLOC) {
+ malloc_bytes += byte_size;
+ return;
+ }
+ if (func_id == DMALLOC_FUNC_FREE) {
+ DMALLOC_SIZE size = 0;
+ dmalloc_examine(old_addr, &size, 0, 0, 0);
+ free_bytes += size;
+ // XXX useless - byte_size and size are 0
+ return;
+ }
+}
+
+}
+#endif /* DMALLOC */
+
+static void
+testMain()
+{
+#ifndef NDB_LINUX /* valgrind-1.0.3 does not support */
+ NdbThread_SetConcurrencyLevel(opt.m_threads + 2);
+#endif
+#ifdef DMALLOC
+ dmalloc_track(malloc_track);
+#endif
+ Test test(0, 0);
+#ifdef ndbODBC
+ Ndb* ndb = 0;
+ if (1) { // pre-alloc one Ndb object
+ ndb = new Ndb("TEST_DB");
+ ndb->init();
+ if (ndb->waitUntilReady() != 0) {
+ ndbout << ndb->getNdbError() << endl;
+ fatal("waitUntilReady");
+ }
+ }
+#endif
+ for (unsigned loop = 1; opt.m_loop == 0 || loop <= opt.m_loop; loop++) {
+ if (opt.m_v >= 2)
+ ndbout << "loop " << loop << endl;
+ // create new set of threads in each loop
+ Thr** thrList = new Thr* [1 + opt.m_threads];
+ for (unsigned n = 1; n <= opt.m_threads; n++) {
+ Thr& thr = *(thrList[n] = new Thr(n, loop));
+ thr.create();
+ if (opt.m_v >= 4)
+ ndbout << "thr " << n << " [main] created" << endl;
+ }
+#ifdef DMALLOC
+ malloc_bytes = free_bytes = 0;
+#endif
+ for (unsigned i = 0; i < caseCount; i++) {
+ const Case& cc = caseList[i];
+ if (! cc.matchcase())
+ continue;
+ if (opt.m_v >= 2)
+ ndbout << "RUN: " << cc.m_name << " - " << cc.m_desc << endl;
+ test.timerOn();
+ for (unsigned subloop = 1; subloop <= opt.m_subloop; subloop++) {
+ my_sema = 0;
+ if (opt.m_v >= 3)
+ ndbout << "subloop " << subloop << endl;
+ if (cc.m_mode == Case::Single) {
+ Thr& thr = *thrList[1];
+ thr.mainStart(cc);
+ thr.mainStop();
+ test.timerCnt(thr.m_test);
+ } else if (cc.m_mode == Case::Serial) {
+ for (unsigned n = 1; n <= opt.m_threads; n++) {
+ Thr& thr = *thrList[n];
+ thr.mainStart(cc);
+ thr.mainStop();
+ test.timerCnt(thr.m_test);
+ }
+ } else if (cc.m_mode == Case::Thread) {
+ for (unsigned n = 1; n <= opt.m_threads; n++) {
+ Thr& thr = *thrList[n];
+ thr.mainStart(cc);
+ }
+ for (unsigned n = 1; n <= opt.m_threads; n++) {
+ Thr& thr = *thrList[n];
+ thr.mainStop();
+ test.timerCnt(thr.m_test);
+ }
+ } else {
+ assert(false);
+ }
+ }
+ test.timerOff();
+ if (opt.m_v >= 1)
+ ndbout << cc.m_name << " total " << test << endl;
+ }
+#ifdef DMALLOC
+ if (opt.m_v >= 9) // XXX useless now
+ ndbout << "malloc " << malloc_bytes << " free " << free_bytes << " lost " << malloc_bytes - free_bytes << endl;
+#endif
+ // tell threads to exit
+ for (unsigned n = 1; n <= opt.m_threads; n++) {
+ Thr& thr = *thrList[n];
+ thr.lock();
+ thr.m_state = Thr::Exit;
+ thr.signal();
+ thr.unlock();
+ if (opt.m_v >= 4)
+ ndbout << "thr " << n << " [main] told to exit" << endl;
+ }
+ for (unsigned n = 1; n <= opt.m_threads; n++) {
+ Thr& thr = *thrList[n];
+ thr.join();
+ if (opt.m_v >= 4)
+ ndbout << "thr " << n << " [main] joined" << endl;
+ delete &thr;
+ }
+ delete[] thrList;
+ }
+#ifdef ndbODBC
+ delete ndb;
+#endif
+}
+
+static bool
+str2num(const char* arg, const char* str, unsigned* num, unsigned lo = 0, unsigned hi = 0)
+{
+ char* end = 0;
+ long n = strtol(str, &end, 0);
+ if (end == 0 || *end != 0 || n < 0) {
+ ndbout << arg << " " << str << " is invalid number" << endl;
+ return false;
+ }
+ if (lo != 0 && n < lo) {
+ ndbout << arg << " " << str << " is too small min = " << lo << endl;
+ return false;
+ }
+ if (hi != 0 && n > hi) {
+ ndbout << arg << " " << str << " is too large max = " << hi << endl;
+ return false;
+ }
+ *num = n;
+ return true;
+}
+
+NDB_COMMAND(testOdbcDriver, "testOdbcDriver", "testOdbcDriver", "testOdbcDriver", 65535)
+{
+ while (++argv, --argc > 0) {
+ const char* arg = argv[0];
+ if (strcmp(arg, "-case") == 0) {
+ if (++argv, --argc > 0) {
+ assert(opt.m_namecnt < arraySize(opt.m_name));
+ opt.m_name[opt.m_namecnt++] = argv[0];
+ if (findCase(argv[0]))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-core") == 0) {
+ opt.m_core = true;
+ continue;
+ }
+ if (strcmp(arg, "-depth") == 0) {
+ if (++argv, --argc > 0) {
+ if (str2num(arg, argv[0], &opt.m_depth))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-dsn") == 0) {
+ if (++argv, --argc > 0) {
+ opt.m_dsn = argv[0];
+ continue;
+ }
+ }
+ if (strcmp(arg, "-frob") == 0) {
+ if (++argv, --argc > 0) {
+ if (str2num(arg, argv[0], &opt.m_frob))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-errs") == 0) {
+ if (++argv, --argc > 0) {
+ if (str2num(arg, argv[0], &opt.m_errs))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-fragtype") == 0) {
+ if (++argv, --argc > 0) {
+ opt.m_fragtype = argv[0];
+ continue;
+ }
+ }
+ if (strcmp(arg, "-home") == 0) {
+ if (++argv, --argc > 0) {
+ opt.m_home = argv[0];
+ continue;
+ }
+ }
+ if (strcmp(arg, "-loop") == 0) {
+ if (++argv, --argc > 0) {
+ if (str2num(arg, argv[0], &opt.m_loop))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-nogetd") == 0) {
+ opt.m_nogetd = true;
+ continue;
+ }
+ if (strcmp(arg, "-noputd") == 0) {
+ opt.m_noputd = true;
+ continue;
+ }
+ if (strcmp(arg, "-nosort") == 0) {
+ opt.m_nosort = true;
+ continue;
+ }
+ if (strcmp(arg, "-scale") == 0) {
+ if (++argv, --argc > 0) {
+ if (str2num(arg, argv[0], &opt.m_scale))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-serial") == 0) {
+ opt.m_serial = true;
+ continue;
+ }
+ if (strcmp(arg, "-skip") == 0) {
+ if (++argv, --argc > 0) {
+ assert(opt.m_skipcnt < arraySize(opt.m_skip));
+ opt.m_skip[opt.m_skipcnt++] = argv[0];
+ if (findCase(argv[0]))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-subloop") == 0) {
+ if (++argv, --argc > 0) {
+ if (str2num(arg, argv[0], &opt.m_subloop))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-table") == 0) {
+ if (++argv, --argc > 0) {
+ opt.m_table = argv[0];
+ if (findTable())
+ continue;
+ }
+ }
+ if (strcmp(arg, "-threads") == 0) {
+ if (++argv, --argc > 0) {
+ if (str2num(arg, argv[0], &opt.m_threads, 1, MAX_THR))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-trace") == 0) {
+ if (++argv, --argc > 0) {
+ if (str2num(arg, argv[0], &opt.m_trace))
+ continue;
+ }
+ }
+ if (strcmp(arg, "-v") == 0) {
+ if (++argv, --argc > 0) {
+ if (str2num(arg, argv[0], &opt.m_v))
+ continue;
+ }
+ }
+ if (strncmp(arg, "-v", 2) == 0 && isdigit(arg[2])) {
+ if (str2num(arg, &arg[2], &opt.m_v))
+ continue;
+ }
+ printusage();
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ homeEnv: {
+ static char env[1000];
+ if (opt.m_home != 0) {
+ sprintf(env, "NDB_HOME=%s", opt.m_home);
+ putenv(env);
+ }
+ }
+ traceEnv: {
+ static char env[40];
+ sprintf(env, "NDB_ODBC_TRACE=%u", opt.m_trace);
+ putenv(env);
+ }
+ debugEnv: {
+ static char env[40];
+ sprintf(env, "NDB_ODBC_DEBUG=%d", 1);
+ putenv(env);
+ }
+ testMain();
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
+// vim: set sw=4:
diff --git a/ndb/test/odbc/test_compiler/Makefile b/ndb/test/odbc/test_compiler/Makefile
new file mode 100644
index 00000000000..34819f21171
--- /dev/null
+++ b/ndb/test/odbc/test_compiler/Makefile
@@ -0,0 +1,21 @@
+include .defs.mk
+
+TYPE = odbcdriver
+
+BIN_TARGET = test_compiler
+
+SOURCES = test_compiler.cpp
+
+CCFLAGS_LOC += \
+ -I$(NDB_TOP)/src/client/odbc/common \
+ -I$(NDB_TOP)/src/client/odbc/dictionary \
+ -I$(NDB_TOP)/src/client/odbc/compiler
+
+CCFLAGS_WARNINGS += -Wno-unused
+
+LIBS_SPEC += \
+ -lodbccompiler_pic
+
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/odbc/test_compiler/test_compiler.cpp b/ndb/test/odbc/test_compiler/test_compiler.cpp
new file mode 100644
index 00000000000..042e9e6d4bf
--- /dev/null
+++ b/ndb/test/odbc/test_compiler/test_compiler.cpp
@@ -0,0 +1,233 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/**********************************************************************************
+ test_compiler.cpp
+ Tests the code tree generated
+ by the compiler
+***********************************************************************************/
+#include <stdio.h>
+#include <memory.h>
+#include "SQL_compiler.hpp"
+#include "SQL_code_tree.hpp"
+
+typedef struct stSTMT_REF_tag {
+ char* szTestStmt ;
+ SQL_compiler* pC ;
+ SQL_code_tree* pRefTree ;
+ int nFlag ; /* indicate if the struct haa a code tree and compiler and should be processed */
+} stSTMT_REF ;
+
+int compare_trees(SQL_code_tree* pCompilerOutput, SQL_code_tree* pReference) ;
+
+/* Assign statements to szTestStmt and NULL to pTestRef */
+
+static stSTMT_REF stTestRef[] = {
+/* 0 */ {"create table foo (pk integer primary key, a integer, b varchar(20), check (a is not null))", NULL, NULL, 0},
+/* 1 */ {"insert into foo (pk, a, b) values (1, 10, 'ett')", NULL, NULL, 0},
+/* 2 */ {"insert into foo values (2, 20)", NULL, NULL, 0},
+/* 3 */ {"delete from foo", NULL, NULL, 1},
+/* 4 */ {"delete from foo where pk=5", NULL, NULL, 0},
+/* 5 */ {"delete from foo where a<10 or b='test'", NULL, NULL, 0},
+/* 6 */ {"update foo set a=100, b=null", NULL, NULL, 0},
+/* 7 */ {"update foo set a=0 where pk=1", NULL, NULL, 0},
+/* 8 */ {"update foo set a=a+pk where b is null", NULL, NULL, 0},
+/* 9 */ {"select * from foo", NULL, NULL, 0},
+/* 10 */ {"select pk, a, b from foo where pk=1", NULL, NULL, 0},
+/* 11 */ {"select * from foo order by a", NULL, NULL, 0},
+/* 12 */ {"select * from foo A, foo B where A.pk=B.a and A.a<2*B.a", NULL, NULL, 0}
+} ;
+
+
+int main(int argc, char* argv[]){
+
+ int retcode = 0 ;
+ int nTests = sizeof(stTestRef)/sizeof(stSTMT_REF) ;
+
+ for(int c = 0 ; c < nTests ; c++) {
+ if(stTestRef[c].nFlag){
+ stTestRef[c].pC = new SQL_compiler() ;
+ stTestRef[c].pRefTree = new SQL_code_tree() ;
+ }
+ }
+
+ /* Create reference code trees */
+
+ /*
+ Statement: 0 "create table foo (pk integer primary key, a integer, b varchar(20), check (a is not null))"
+ */
+
+
+ /*
+ Statement: 1
+ */
+
+
+
+ /*
+ Statement: 2
+ */
+
+
+
+ /*
+ Statement: 3 "delete from foo"
+ */
+
+ stTestRef[3].pRefTree->shift('N') ;
+ stTestRef[3].pRefTree->shift('D') ;
+ stTestRef[3].pRefTree->shift('B') ;
+ stTestRef[3].pRefTree->reduce(0x2050400e, 3) ;
+ stTestRef[3].pRefTree->shift('F') ;
+ stTestRef[3].pRefTree->shift('O') ;
+ stTestRef[3].pRefTree->shift('O') ;
+ stTestRef[3].pRefTree->reduce(0x20502003, 3) ;
+ stTestRef[3].pRefTree->reduce(0x2050400f, 1) ;
+ stTestRef[3].pRefTree->reduce(0x20504007, 2) ;
+ stTestRef[3].pRefTree->reduce(0x21407003, 1) ;
+ stTestRef[3].pRefTree->shift(0x205021ca) ;
+ stTestRef[3].pRefTree->reduce(0x20630001, 1) ;
+ stTestRef[3].pRefTree->reduce(0x20815001, 1) ;
+ stTestRef[3].pRefTree->shift(0x21407002) ;
+ stTestRef[3].pRefTree->reduce(0x21407004, 3) ;
+ stTestRef[3].pRefTree->shift(0x21407002) ;
+ stTestRef[3].pRefTree->reduce(0x21407005, 1) ;
+ stTestRef[3].pRefTree->shift(0x21414001) ;
+ stTestRef[3].pRefTree->shift(0x21414002) ;
+ stTestRef[3].pRefTree->reduce(0x21407001, 4) ;
+ stTestRef[3].pRefTree->reduce(0x51506004, 1) ;
+ stTestRef[3].pRefTree->reduce(0x51506003, 1) ;
+
+ /*
+ Statement: 4
+ */
+
+
+
+ /*
+ Statement: 5
+ */
+
+
+
+
+ /*
+ Statement: 6
+ */
+
+
+
+
+ /*
+ Statement: 7
+ */
+
+
+
+ /*
+ Statement: 8
+ */
+
+
+
+ /*
+ Statement: 9
+ */
+
+
+
+ /*
+ Statement: 10
+ */
+
+
+
+ /*
+ Statement: 11
+ */
+
+
+
+ /*
+ Statement: 12
+ */
+
+
+
+ for(int i = 0 ; i < nTests ; i++){
+ /* Check to see if the statement has an associated code tree and compiler */
+ if(stTestRef[i].nFlag){
+ stTestRef[i].pC->prepare( stTestRef[i].szTestStmt, strlen(stTestRef[i].szTestStmt)) ;
+ if( 0 != compare_trees(&stTestRef[i].pC->m_code_tree, stTestRef[i].pRefTree) ){
+ printf("\nCompiler generated tree for statement #%d: \"%s\"\ndeviates from its reference\n", i, stTestRef[i].szTestStmt) ;
+ retcode = -1 ;
+ break ;
+ }else{
+ printf("\nTrees for statement #%d: \"%s\" match nicely -- OK\n", i, stTestRef[i].szTestStmt) ;
+ retcode = 0 ;
+ }
+ }
+ }
+
+ for(int d = 0 ; d < nTests ; d++) {
+ if(stTestRef[d].nFlag){
+ delete stTestRef[d].pC ;
+ delete stTestRef[d].pRefTree ;
+ }
+ }
+
+ return retcode ;
+
+}
+
+
+
+
+int compare_trees(SQL_code_tree* pCompilerOutput, SQL_code_tree* pReference){
+
+ int nTop = -1 ;
+
+ if(pCompilerOutput->top()== pReference->top()){
+
+ nTop = pReference->top() ;
+
+ }else{
+ printf("\npCompilerOutput->top() = %d;\tpReference->top() = %d\n", pCompilerOutput->top(), pReference->top()) ;
+ return -1 ;
+ }
+
+ pCompilerOutput->beginPostfix() ;
+ pReference->beginPostfix() ;
+
+ for(int r = 0 ; r < nTop ; r++){
+ if(pCompilerOutput->symbol() != pReference->symbol()){
+
+ printf("Deviation found in position %d\n", r) ;
+ printf("pCompilerOutput->symbol() = 0x%X;\tpReference->symbol() = 0x%X\n", pCompilerOutput->symbol(), pReference->symbol()) ;
+ return -1 ;
+
+ }else{
+
+ pCompilerOutput->nextPostfix() ;
+ pReference->nextPostfix() ;
+
+ }
+ }
+
+ return 0 ;
+
+}
+
diff --git a/ndb/test/odbc/tpcb/Makefile b/ndb/test/odbc/tpcb/Makefile
new file mode 100644
index 00000000000..8ab429c8ea1
--- /dev/null
+++ b/ndb/test/odbc/tpcb/Makefile
@@ -0,0 +1,30 @@
+include .defs.mk
+
+TYPE = *
+
+BIN_TARGET = tpcb
+
+SOURCES = ttTime.c tpcb.cpp
+
+CCFLAGS_LOC += \
+ -I$(NDB_TOP)/include \
+ -I$(NDB_TOP)/include/ndbapi \
+ -I$(NDB_TOP)/include/portlib \
+ -I$(NDB_TOP)/include/util \
+ -I$(NDB_TOP)/test/include \
+ -I/usr/local/include
+
+CCFLAGS_WARNINGS += -Wno-unused -Wno-sign-compare -Wformat
+
+CCFLAGS_TOP += -DndbODBC
+
+BIN_TARGET_LIBS = NDB_API_pic NDB_ODBC_pic NDBT
+
+ifeq ($(NDB_OS),SOLARIS)
+BIN_TARGET_LIBS += dmallocthcxx
+CCFLAGS_TOP += -DDMALLOC
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+$(BIN_DIR)$(BIN_TARGET): Makefile
diff --git a/ndb/test/odbc/tpcb/Makefile_mysql b/ndb/test/odbc/tpcb/Makefile_mysql
new file mode 100644
index 00000000000..4e1b9a25fe2
--- /dev/null
+++ b/ndb/test/odbc/tpcb/Makefile_mysql
@@ -0,0 +1,33 @@
+include $(NDB_TOP)/Defs.mk
+
+TYPE = odbcclient
+
+BIN_TARGET = tpcb
+
+SOURCES = ttTime.c tpcb.cpp
+
+CCFLAGS_LOC += -I/usr/local/include \
+ -I$(NDB_TOP)/test/include \
+ -I$(NDB_TOP)/include \
+ -I$(NDB_TOP)/include/util \
+ -I$(NDB_TOP)/src/client/odbc/common
+
+
+
+
+#CCFLAGS_WARNINGS += -Wno-unused
+
+LIBS_LOC += -L/usr/local/lib
+BIN_TARGET_LIBS_DIRS += /usr/local/lib
+BIN_TARGET_LIBS += odbc odbcinst
+
+#LIBS_SPEC += -pg
+# -lNDBT \
+# -lodbc \
+# -lodbcinst \
+# -lportlib
+
+
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/odbc/tpcb/Makefile_ndb b/ndb/test/odbc/tpcb/Makefile_ndb
new file mode 100644
index 00000000000..85960413ef0
--- /dev/null
+++ b/ndb/test/odbc/tpcb/Makefile_ndb
@@ -0,0 +1,30 @@
+include $(NDB_TOP)/Defs.mk
+
+TYPE = *
+
+BIN_TARGET = tpcb
+
+SOURCES = ttTime.c tpcb.cpp
+
+CCFLAGS_LOC += \
+ -I$(NDB_TOP)/include \
+ -I$(NDB_TOP)/include/ndbapi \
+ -I$(NDB_TOP)/include/portlib \
+ -I$(NDB_TOP)/include/util \
+ -I$(NDB_TOP)/test/include \
+ -I/usr/local/include
+
+CCFLAGS_WARNINGS += -Wno-unused -Wno-sign-compare -Wformat
+
+CCFLAGS_TOP += -DndbODBC
+
+BIN_TARGET_LIBS = NDB_API_pic NDB_ODBC_pic NDBT
+
+ifeq ($(NDB_OS),SOLARIS)
+BIN_TARGET_LIBS += dmallocthcxx
+CCFLAGS_TOP += -DDMALLOC
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
+$(BIN_DIR)$(BIN_TARGET): Makefile
diff --git a/ndb/test/odbc/tpcb/readme.txt b/ndb/test/odbc/tpcb/readme.txt
new file mode 100644
index 00000000000..008cafb9d2f
--- /dev/null
+++ b/ndb/test/odbc/tpcb/readme.txt
@@ -0,0 +1,15 @@
+'tpcb' requires an .odbc.ini file in
+/etc/
+or in
+/home/user/
+
+The .odbc.ini file must contain a DSN entry called ndb:
+
+#--------- .odbc.ini example --------------------
+
+[ndb]
+Driver = /path_to_installation/lib/libNDB_ODBC.so
+
+#--------- End of example -----------------------
+
+
diff --git a/ndb/test/odbc/tpcb/timesten.h b/ndb/test/odbc/tpcb/timesten.h
new file mode 100644
index 00000000000..45579f9d277
--- /dev/null
+++ b/ndb/test/odbc/tpcb/timesten.h
@@ -0,0 +1,188 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*
+ * $Revision: 1.1 $
+ * (c) Copyright 1997-2003, TimesTen, Inc.
+ * All rights reserved.
+ */
+
+#ifndef TIMESTEN_H_INCLUDED
+#define TIMESTEN_H_INCLUDED
+
+#ifdef _WIN32
+#include <windows.h>
+#endif
+
+#include <sql.h>
+#include <sqltypes.h>
+#include <sqlext.h>
+/*
+ * TimesTen extension to application data types; only usable
+ * when application directly linked to the TimesTen driver.
+ */
+#define SQL_C_ADDR 100
+
+#ifndef SQL_C_SBIGINT
+#if (ODBCVER < 0x0300)
+#define SQL_C_SBIGINT (SQL_BIGINT+SQL_SIGNED_OFFSET)
+#define SQL_C_UBIGINT (SQL_BIGINT+SQL_UNSIGNED_OFFSET)
+#endif
+#endif
+
+#define SQL_C_BIGINT SQL_C_SBIGINT
+
+#if (ODBCVER < 0x0300)
+#ifdef _WIN32
+typedef __int64 SQLBIGINT;
+/* On Unix platforms SQLBIGINT is defined in odbcinclude directory*/
+#endif
+#endif
+
+#define BIGINT SQLBIGINT
+
+#ifdef _WIN32
+#define UBIGINT unsigned __int64
+#else
+#define UBIGINT unsigned long long
+#endif
+
+
+#define SQL_WCHAR (-8)
+#define SQL_WVARCHAR (-9)
+#define SQL_WLONGVARCHAR (-10)
+#define SQL_C_WCHAR SQL_WCHAR
+
+/* SQLGetInfo() InfoTypes */
+#define SQL_CONVERT_WCHAR 122
+#define SQL_CONVERT_WLONGVARCHAR 125
+#define SQL_CONVERT_WVARCHAR 126
+
+/* TimesTen specific SQLGetInfo types */
+#define TT_REPLICATION_INVALID (SQL_INFO_DRIVER_START + 2000)
+
+/* SQLGetInfo() return value bitmasks */
+#ifndef SQL_CVT_WCHAR
+/*
+** These definitions differ from Microsoft in that they are not
+** specified as long (e.g. 0x00200000L), hence they are protected
+** by the ifndef above.
+*/
+#define SQL_CVT_WCHAR 0x00200000
+#define SQL_CVT_WLONGVARCHAR 0x00400000
+#define SQL_CVT_WVARCHAR 0x00800000
+#endif
+
+/*
+** The Microsoft Driver Manager SQLBindParameter() will not pass SQL_WCHAR
+** through. Use this hack to get around it.
+*/
+#define SQL_WCHAR_DM_SQLBINDPARAMETER_BYPASS -888
+
+/* This is an extension to ODBC's isolation levels. It reflects an
+ * earlier implementation of read-committed that released locks on
+ * next fetch, rather than releasing locks before returning value to
+ * application. */
+#define SQL_TXN_CURSOR_STABILITY 0x00001000
+#define SQL_TXN_NOBLOCK_DELETE 0x00002000
+
+/* TimesTen-specific connection option */
+#define TT_PREFETCH_CLOSE 10001
+#define TT_PREFETCH_CLOSE_OFF 0
+#define TT_PREFETCH_CLOSE_ON 1
+
+/* Adding a new sql connection option */
+#define TT_PREFETCH_COUNT 10003
+#define TT_PREFETCH_COUNT_MAX 128
+
+/*
+ * Platform specific data types for integers that scale
+ * with pointer size
+ */
+
+#ifdef _IA64_
+typedef signed __int64 tt_ptrint;
+typedef unsigned __int64 tt_uptrint;
+#else
+#ifdef _WIN32
+typedef signed long tt_ptrint;
+typedef unsigned long tt_uptrint;
+#else
+typedef signed long tt_ptrint;
+typedef unsigned long tt_uptrint;
+#endif
+#endif
+
+#ifdef _WIN32
+typedef signed __int64 tt_int8;
+typedef unsigned __int64 tt_uint8;
+#else
+typedef signed long long tt_int8;
+typedef unsigned long long tt_uint8;
+#endif
+
+/* printf formats for pointer-sized integers */
+#ifdef _IA64_ /* 64-bit NT */
+#define PTRINT_FMT "I64d"
+#define UPTRINT_FMT "I64u"
+#define xPTRINT_FMT "I64x"
+#define XPTRINT_FMT "I64X"
+#else
+#ifdef _WIN32 /* 32-bit NT */
+#define PTRINT_FMT "ld"
+#define UPTRINT_FMT "lu"
+#define xPTRINT_FMT "lx"
+#define XPTRINT_FMT "lX"
+#else /* 32 and 64-bit UNIX */
+#define PTRINT_FMT "ld"
+#define UPTRINT_FMT "lu"
+#define xPTRINT_FMT "lx"
+#define XPTRINT_FMT "lX"
+#endif
+#endif
+
+/* printf formats for 8-byte integers */
+#ifndef INT8_FMT_DEFINED
+#ifdef _WIN32 /* 32 and 64-bit NT */
+#define INT8_FMT "I64d"
+#define UINT8_FMT "I64u"
+#define xINT8_FMT "I64x"
+#define XINT8_FMT "I64X"
+#else /* 32 and 64-bit UNIX */
+#define INT8_FMT "lld"
+#define UINT8_FMT "llu"
+#define xINT8_FMT "llx"
+#define XINT8_FMT "llX"
+#endif
+#define INT8_FMT_DEFINED 1
+#endif
+
+/* The following types are defined in the newer odbc include files
+ from Microsoft
+*/
+#if defined (_WIN32) && !defined (_IA64_)
+#ifndef SQLROWSETSIZE
+#define SQLROWSETSIZE SQLUINTEGER
+#define SQLLEN SQLINTEGER
+#define SQLROWOFFSET SQLINTEGER
+#define SQLROWCOUNT SQLUINTEGER
+#define SQLULEN SQLUINTEGER
+#define SQLSETPOSIROW SQLUSMALLINT
+#endif
+#endif
+
+
+#endif
diff --git a/ndb/test/odbc/tpcb/tpcb.cpp b/ndb/test/odbc/tpcb/tpcb.cpp
new file mode 100644
index 00000000000..60d746e7844
--- /dev/null
+++ b/ndb/test/odbc/tpcb/tpcb.cpp
@@ -0,0 +1,1415 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+static const volatile char cvsid[] = "$Id: tpcb.cpp,v 1.4 2003/09/26 09:04:34 johan Exp $";
+/*
+ * $Revision: 1.4 $
+ * (c) Copyright 1996-2003, TimesTen, Inc.
+ * All rights reserved.
+ */
+
+/* This source is best displayed with a tabstop of 4 */
+
+#define NDB
+
+//#define MYSQL
+
+#ifdef WIN32
+#include <windows.h>
+#include "ttRand.h"
+#else
+#if !defined NDB && !defined MYSQL
+#include <sqlunix.h>
+#endif
+#include <unistd.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#ifdef SB_P_OS_CHORUS
+#include "ttRand.h"
+#endif
+#endif
+
+#include <math.h>
+#include <time.h>
+#include <sql.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <ctype.h>
+
+#if !defined NDB && !defined MYSQL
+#include "ttTime.h"
+#include "utils.h"
+#include "tt_version.h"
+#include "timesten.h"
+#endif
+
+#if defined NDB || defined MYSQL
+#include <NdbOut.hpp>
+#include <string.h>
+
+#include <sqlext.h>
+#include <sql.h>
+extern "C" {
+#include "ttTime.h"
+#include "timesten.h"
+ void ttGetWallClockTime(ttWallClockTime* timeP);
+ void ttCalcElapsedWallClockTime(ttWallClockTime* beforeP,
+ ttWallClockTime* afterP,
+ double* nmillisecondsP);
+ void ttGetThreadTimes(ttThreadTimes * endRes);
+ void ttCalcElapsedThreadTimes(ttThreadTimes* startRes,
+ ttThreadTimes * endRes,
+ double * kernel,
+ double * user);
+}
+
+#define app_exit exit
+#define status_msg0 ndbout_c
+#define status_msg1 ndbout_c
+#define status_msg2 ndbout_c
+#define err_msg0 ndbout_c
+#define err_msg1 ndbout_c
+#define err_msg3 ndbout_c
+#define out_msg0 ndbout_c
+#define out_msg1 ndbout_c
+#define out_msg3 ndbout_c
+#define CONN_STR_LEN 255
+#define DBMS_TIMESTEN 1
+#define DBMS_MSSQL 2
+#define DBMS_UNKNOWN 3
+#define ABORT_DISCONNECT_EXIT 1
+#define NO_EXIT 0
+#define ERROR_EXIT 1
+#define DISCONNECT_EXIT 2
+#endif
+
+#define VERBOSE_NOMSGS 0
+/* this value is used for results (and err msgs) only */
+#define VERBOSE_RESULTS 1
+/* this value is the default for the cmdline demo */
+#define VERBOSE_DFLT 2
+#define VERBOSE_ALL 3
+
+#ifdef MYSQL
+#define DSNNAME "DSN=myodbc3"
+#elif defined NDB
+#define DSNNAME "DSN=ndb"
+#else
+#define DSNNAME "DSN="
+#endif
+
+/* number of branches, tellers, and accounts */
+
+#define NumBranches 1
+#define TellersPerBranch 10
+#define AccountsPerBranch 10000
+
+/* number of transactions to execute */
+
+#define NumXacts 25000
+
+/* starting seed value for the random number generator */
+
+#define SeedVal 84773
+
+/* for MS SQL, the drop, create and use database statements */
+
+#define DatabaseDropStmt "drop database tpcbDB;"
+#ifdef MYSQL
+#define DatabaseCreateStmt "create database tpcbDB;"
+#else
+#define DatabaseCreateStmt "create database tpcbDB ON DEFAULT = %d;"
+#endif
+#define DatabaseUseStmt "use tpcbDB;"
+
+/*
+ * Specifications of table columns.
+ * Fillers of 80, 80, 84, and 24 bytes, respectively, are used
+ * to ensure that rows are the width required by the benchmark.
+ *
+ * Note: The TimesTen and MS SQL CREATE TABLE statements for the
+ * accounts, tellers and branches tables are different.
+ *
+ */
+
+#define TuplesPerPage 256
+
+
+#ifdef MYSQL
+
+#define AccountCrTblStmt "create table accounts \
+(number integer not null primary key, \
+branchnum integer not null, \
+balance float not null, \
+filler char(80));"
+
+#define TellerCrTblStmt "create table tellers \
+(number integer not null primary key, \
+branchnum integer not null, \
+balance float not null, \
+filler char(80));"
+
+#define BranchCrTblStmt "create table branches \
+(number integer not null primary key, \
+balance float not null, \
+filler char(84));"
+
+#endif
+
+
+#ifdef NDB
+#define AccountCrTblStmt "create table accounts \
+(number integer not null primary key, \
+branchnum integer not null, \
+balance float not null, \
+filler char(80)) nologging"
+
+#define TellerCrTblStmt "create table tellers \
+(number integer not null primary key, \
+branchnum integer not null, \
+balance float not null, \
+filler char(80)) nologging"
+
+#define BranchCrTblStmt "create table branches \
+(number integer not null primary key, \
+balance float not null, \
+filler char(84)) nologging"
+#endif
+
+#ifdef NDB
+
+#define HistoryCrTblStmt "create table History \
+(tellernum integer not null, \
+branchnum integer not null, \
+accountnum integer not null, \
+delta float not null, \
+createtime integer not null, \
+filler char(24), \
+primary key (tellernum, branchnum, accountnum, delta, createtime)) nologging"
+
+#else
+
+#ifdef MYSQL
+
+#define HistoryCrTblStmt "create table History \
+(tellernum integer not null, \
+branchnum integer not null, \
+accountnum integer not null, \
+delta float(53) not null, \
+createtime integer not null, \
+filler char(24))"
+#endif
+
+#define HistoryCrTblStmt "create table History \
+(tellernum integer not null, \
+branchnum integer not null, \
+accountnum integer not null, \
+delta float(53) not null, \
+createtime integer not null, \
+filler char(24));"
+#endif
+
+#define TTAccountCrTblStmt "create table accounts \
+(number integer not null primary key, \
+branchnum integer not null, \
+balance float(53) not null, \
+filler char(80)) unique hash on (number) pages = %" PTRINT_FMT ";"
+
+#define TTTellerCrTblStmt "create table tellers \
+(number integer not null primary key, \
+branchnum integer not null, \
+balance float(53) not null, \
+filler char(80)) unique hash on (number) pages = %" PTRINT_FMT ";"
+
+#define TTBranchCrTblStmt "create table branches \
+(number integer not null primary key, \
+balance float(53) not null, \
+filler char(84)) unique hash on (number) pages = %" PTRINT_FMT ";"
+
+
+/* Insertion statements used to populate the tables */
+
+#define NumInsStmts 3
+char* insStmt[NumInsStmts] = {
+ "insert into branches values (?, 0.0, NULL)",
+ "insert into tellers values (?, ?, 0.0, NULL)",
+ "insert into accounts values (?, ?, 0.0, NULL)"
+};
+
+/* Transaction statements used to update the tables */
+
+#define NumXactStmts 5
+
+#ifdef NDB
+char* tpcbXactStmt[NumXactStmts] = {
+ "update accounts \
+set balance = balance + ? \
+where number = ?",
+
+ "select balance \
+from accounts \
+where number = ?",
+
+ "update tellers \
+set balance = balance + ? \
+where number = ?",
+
+ "update branches \
+set balance = balance + ? \
+where number = ?",
+
+ "insert into History(tellernum, branchnum, \
+accountnum, delta, createtime, filler) \
+values (?, ?, ?, ?, ?, NULL)"
+};
+
+#else
+char* tpcbXactStmt[NumXactStmts] = {
+ "update accounts \
+set balance = balance + ? \
+where number = ?;",
+
+ "select balance \
+from accounts \
+where number = ?;",
+
+ "update tellers \
+set balance = balance + ? \
+where number = ?;",
+
+ "update branches \
+set balance = balance + ? \
+where number = ?;",
+
+ "insert into History \
+values (?, ?, ?, ?, ?, NULL);"
+};
+
+
+#endif
+
+/* Global parameters and flags (typically set by parse_args()) */
+
+int tabFlag = 0; /* Default is NOT tab output mode */
+char szConnStrIn[CONN_STR_LEN]; /* ODBC Connection String */
+int printXactTimes = 0; /* Transaction statistics
+ * gathering flag */
+char statFile[FILENAME_MAX]; /* Transaction statistics filename */
+int scaleFactor = 2; /* Default TPS scale factor */
+int numBranchTups; /* Number of branches */
+int numTellerTups; /* Number of tellers */
+int numAccountTups; /* Number of accounts */
+int numNonLocalAccountTups; /* Number of local accounts */
+int numXacts = NumXacts; /* Default number of transactions */
+int verbose = VERBOSE_DFLT; /* Verbose level */
+FILE *statusfp; /* File for status messages */
+
+
+
+int DBMSType; /* DBMS type (DBMS_TIMESTEN, DBMS_MSSQL...) */
+
+
+SQLHENV henv; /* Environment handle */
+
+
+
+
+
+
+
+void handle_errors( SQLHDBC hdbc, SQLHSTMT hstmt, int errcode, int action, char * msg,
+ char * file, int line) {
+
+ if (errcode == SQL_SUCCESS)
+ return;
+
+ if(errcode == SQL_ERROR) {
+ int ret;
+ long diagCount=0;
+ short length=0;
+ SQLCHAR state[10] = "";
+ SQLCHAR message[200] = "";
+ long native = 0;
+ if(hstmt != 0) {
+ ret = SQLGetDiagField(SQL_HANDLE_STMT, hstmt, 0, SQL_DIAG_NUMBER, &diagCount, SQL_IS_INTEGER, 0);
+
+ for(long i = 0; i < diagCount; i++) {
+ ret = SQLGetDiagRec(SQL_HANDLE_STMT, hstmt, i, (SQLCHAR*)state, &native, (SQLCHAR*)message, 200, &length);
+ ndbout_c("GetDiagRec: Message : %s ", message);
+ }
+ }
+ }
+
+ if(errcode != SQL_SUCCESS) {
+ ndbout_c("Message: %s", msg);
+ switch(errcode) {
+ case SQL_SUCCESS_WITH_INFO:
+ ndbout_c("SQL_SUCCESS_WITH_INFO");
+ break;
+ case SQL_STILL_EXECUTING:
+ ndbout_c("SQL_STILL_EXECUTING");
+ break;
+ case SQL_ERROR:
+ ndbout_c("SQL_ERROR");
+ break;
+ case SQL_INVALID_HANDLE:
+ ndbout_c("SQL_INVALID_HANDLE");
+ break;
+ default:
+ ndbout_c("Some other error");
+ }
+ exit(1);
+ }
+
+
+}
+
+
+
+
+
+/*********************************************************************
+ * FUNCTION: usage
+ *
+ * DESCRIPTION: This function prints a usage message describing
+ * the command line options of the program.
+ *
+ * PARAMETERS: char* prog full program path name
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+static void usage(char *prog)
+{
+ char *progname;
+
+ /* Get the name of the program (sans path). */
+
+#ifdef WIN32
+ progname = strrchr(prog, '\\');
+#else
+ progname = strrchr(prog, '/');
+#endif
+ if (progname == 0)
+ progname = prog;
+ else
+ ++progname;
+
+ /* Print the usage message */
+
+ fprintf(stderr,
+ "Usage:\t%s [-h] [-help] [-V] [-connStr <string>] [-v <level>]\n"
+ "\t\t[-xact <xacts>] [-scale <scale>] [-tabs] [-s <statfile>]\n\n"
+ " -h Prints this message and exits.\n"
+ " -help Same as -h.\n"
+ " -V Prints version number and exits.\n"
+ " -connStr <string> Specifies an ODBC connection string to replace the\n"
+ " default DSN for the program. The default is\n"
+ " \"DSN=TpcbData<version>;OverWrite=1\".\n"
+ " -v <level> Verbose level\n"
+ " 0 = errors only\n"
+ " 1 = results only\n"
+ " 2 = results and some status messages (default)\n"
+ " 3 = all messages\n"
+ " -xact <xacts> Specifies the number of transactions to be run\n"
+ " The default is 25000 transactions.\n"
+ " -scale <scale> Specifies a scale factor which determines the\n"
+ " number of branches (scale), tellers (scale x 10),\n"
+ " accounts (scale x 10000) and non-local accounts\n"
+ " ((scale-1) x 10000. The default scale factor is 2.\n"
+ " -tabs Specifies that the output be a tab-separated\n"
+ " format suitable for import into a spreadsheet.\n"
+ " Results only go to stdout; status and other\n"
+ " messages go to stderr.\n"
+ " -s <statfile> Prints individual transaction times to <statfile>.\n",
+ progname);
+}
+
+/*********************************************************************
+ *
+ * FUNCTION: parse_args
+ *
+ * DESCRIPTION: This function parses the command line arguments
+ * passed to main(), setting the appropriate global
+ * variables and issuing a usage message for
+ * invalid arguments.
+ *
+ * PARAMETERS: int argc # of arguments from main()
+ * char *argv[] arguments from main()
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+void
+parse_args(int argc, char *argv[])
+{
+ int i = 1;
+
+ *szConnStrIn = 0;
+
+ while (i < argc) {
+
+ if ( !strcmp(argv[i], "-h") || !strcmp(argv[i], "-help") ) {
+ usage(argv[0]);
+ app_exit(0);
+ }
+ /*
+ if (!strcmp(argv[i], "-V")) {
+ printf("%s\n", TTVERSION_STRING);
+ app_exit(0);
+ }
+ */
+ if (strcmp(argv[i], "-s") == 0) {
+ if (argc < i+2 ) {
+ usage(argv[0]);
+ app_exit(1);
+ }
+ if (sscanf(argv[i+1], "%s", statFile) == 0) {
+ usage(argv[0]);
+ app_exit(1);
+ }
+ printXactTimes = 1;
+ i += 2;
+ }
+ else if (!strcmp(argv[i], "-connStr")) {
+ if (argc < i+2 ) {
+ usage(argv[0]);
+ app_exit(1);
+ }
+ strcpy(szConnStrIn, argv[i+1]);
+ i += 2;
+ continue;
+ }
+ else if (strcmp("-v", argv[i]) == 0) {
+ if (argc < i+2 ) {
+ usage(argv[0]);
+ app_exit(1);
+ }
+ if (sscanf(argv[i+1], "%d", &verbose) == -1 ||
+ verbose < 0 || verbose > 3) {
+ fprintf(stderr, "-v flag requires an integer parameter (0-3)\n");
+ usage(argv[0]);
+ app_exit(1);
+ }
+ i += 2;
+ }
+ else if (strcmp("-xact",argv[i]) == 0) {
+ if (argc < i+2 ) {
+ usage(argv[0]);
+ app_exit(1);
+ }
+
+ if (sscanf(argv[i+1], "%" PTRINT_FMT, &numXacts) == -1 || numXacts < 0) {
+ fprintf(stderr, "-xact flag requires a non-negative integer argument\n");
+ usage(argv[0]);
+ app_exit(1);
+ }
+
+ i += 2;
+ }
+ else if (strcmp("-scale",argv[i]) == 0) {
+ if (argc < i+2 ) {
+ usage(argv[0]);
+ app_exit(1);
+ }
+ if (sscanf(argv[i+1], "%d", &scaleFactor) == -1 || scaleFactor < 1) {
+ fprintf(stderr, "-scale flag requires an integer argument >= 1\n");
+ usage(argv[0]);
+ app_exit(1);
+ }
+ /* Calculate tuple sizes */
+ numBranchTups = NumBranches * scaleFactor;
+ numTellerTups = TellersPerBranch * scaleFactor;
+ numAccountTups = AccountsPerBranch * scaleFactor;
+ numNonLocalAccountTups = AccountsPerBranch * (scaleFactor-1);
+ i += 2;
+ }
+ else if (strcmp("-tabs",argv[i]) == 0) {
+ tabFlag = 1;
+ statusfp = stderr;
+ i += 1;
+ }
+ else {
+ usage(argv[0]);
+ app_exit(1);
+ }
+ }
+}
+
+/*********************************************************************
+ *
+ * FUNCTION: doImmed
+ *
+ * DESCRIPTION: This function executes and frees the specified
+ * statement. It is used as a direct means to
+ * create the tables used by this benchmark,
+ *
+ * PARAMETERS: SQLHDBC hdbc SQL Connection handle
+ * SQLHSTMT hs SQL Statement handle
+ * char* cmd SQL Statement text
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+void
+doImmed(SQLHDBC hdbc, SQLHSTMT hs, char* cmd)
+{
+ SQLRETURN rc;
+
+ /* Execute the command */
+
+ rc = SQLExecDirect(hs, (SQLCHAR *) cmd, SQL_NTS);
+ handle_errors(hdbc, hs, rc, ABORT_DISCONNECT_EXIT,
+ "Error executing statement", __FILE__, __LINE__);
+
+ /* Close associated cursor and drop pending results */
+
+ rc = SQLFreeStmt(hs, SQL_CLOSE);
+ handle_errors(hdbc, hs, rc, ABORT_DISCONNECT_EXIT,
+ "closing statement handle",
+ __FILE__, __LINE__);
+
+}
+
+
+/*********************************************************************
+ *
+ * FUNCTION: main
+ *
+ * DESCRIPTION: This is the main function of the tpcb benchmark.
+ * It connects to an ODBC data source, creates and
+ * populates tables, updates the tables in a user-
+ * specified number of transactions and reports on
+ * on the transaction times.
+ *
+ * PARAMETERS: int argc # of command line arguments
+ * char *argv[] command line arguments
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+int
+main(int argc, char *argv[])
+{
+
+ /* variables used for setting up the tables */
+
+ char cmdStr[1024];
+ char errstr[4096];
+
+ /* variables used during transactions */
+
+ int accountNum;
+ int tellerNum;
+ int branchNum;
+ int timeStamp;
+ double delta;
+ unsigned int lrand;
+ unsigned short *srands, localLimit;
+ int lp64;
+
+ /* variables used for timing and statistics */
+
+ int warmup;
+ double kernel, user, real;
+ ttThreadTimes startRes, endRes;
+ ttWallClockTime startT, endT;
+ ttWallClockTime** rtStart;
+ ttWallClockTime** rtEnd;
+ double** resTime;
+ double maxTime, totTime;
+ int i;
+ int j;
+ int numLocalXacts=0, numRemoteXacts=0;
+
+ /* variables for ODBC */
+
+ SQLHDBC hdbc;
+ SQLHSTMT hstmt;
+ SQLHSTMT txstmt[NumXactStmts];
+ SQLRETURN rc;
+ char DBMSName[32];
+ char DBMSVersion[32];
+ int databaseSize;
+
+ int fThreadTime = 1;
+
+#ifdef WIN32
+ OSVERSIONINFO sysInfo;
+
+ sysInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
+ GetVersionEx (&sysInfo);
+
+ /* GetThreadTimes is not supported on 95/98. Hence,
+ we do not support Resource/User/System times */
+ if (sysInfo.dwPlatformId != VER_PLATFORM_WIN32_NT)
+ fThreadTime = 0;
+#endif
+#if defined(TTCLIENTSERVER) && defined(__hpux) && !defined(__LP64__)
+ /* HP requires this for C main programs that call aC++ shared libs */
+ _main();
+#endif /* hpux32 */
+
+ /* Set up default signal handlers */
+
+#ifndef NDB
+ /* StopRequestClear();
+ if (HandleSignals() != 0) {
+ err_msg0("Unable to set signal handlers\n");
+ return 1;
+ }
+ */
+#endif
+ /* set IO mode for demo */
+ /* set_io_mode(); */
+
+ /* initialize the file for status messages */
+ statusfp = stdout;
+
+ /* set variable for 8-byte longs */
+ lp64 = (sizeof(lrand) == 8);
+
+ /* set the default tuple sizes */
+
+ numBranchTups = NumBranches * scaleFactor;
+ numTellerTups = TellersPerBranch * scaleFactor;
+ numAccountTups = AccountsPerBranch * scaleFactor;
+ numNonLocalAccountTups = AccountsPerBranch * (scaleFactor-1);
+
+ /* parse the command arguments */
+ parse_args(argc, argv);
+
+ /* allocate the transaction-based variables */
+
+ rtStart = (ttWallClockTime**) malloc(numXacts * sizeof(ttWallClockTime*));
+ if (!rtStart) {
+ err_msg0("Cannot allocate the transaction timing structures");
+ app_exit(1);
+ }
+ for (i = 0; i < numXacts; i++) {
+ rtStart[i] = (ttWallClockTime*) malloc(sizeof(ttWallClockTime));
+ if (!rtStart[i]) {
+ err_msg0("Cannot allocate the transaction timing structures");
+ app_exit(1);
+ }
+ }
+
+ rtEnd = (ttWallClockTime**) malloc(numXacts * sizeof(ttWallClockTime*));
+ if (!rtEnd) {
+ err_msg0("Cannot allocate the transaction timing structures");
+ app_exit(1);
+ }
+ for (i = 0; i < numXacts; i++) {
+ rtEnd[i] = (ttWallClockTime*) malloc(sizeof(ttWallClockTime));
+ if (!rtEnd[i]) {
+ err_msg0("Cannot allocate the transaction timing structures");
+ app_exit(1);
+ }
+ }
+
+ resTime = (double**) malloc(numXacts * sizeof(double*));
+ if (!resTime) {
+ err_msg0("Cannot allocate the transaction timing structures");
+ app_exit(1);
+ }
+ for (i = 0; i < numXacts; i++) {
+ resTime[i] = (double*) malloc(sizeof(double));
+ if (!resTime[i]) {
+ err_msg0("Cannot allocate the transaction timing structures");
+ app_exit(1);
+ }
+ }
+
+ /* ODBC initialization */
+
+ rc = SQLAllocEnv(&henv);
+ if (rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO) {
+ /* error occurred -- don't bother calling handle_errors, since handle
+ * is not valid so SQLError won't work */
+ err_msg3("ERROR in %s, line %d: %s\n",
+ __FILE__, __LINE__, "allocating an environment handle");
+ app_exit(1);
+ }
+ SQLSetEnvAttr(henv, SQL_ATTR_ODBC_VERSION, (SQLPOINTER)SQL_OV_ODBC3, SQL_IS_INTEGER);
+
+ /* call this in case of warning */
+ handle_errors(NULL, NULL, rc, NO_EXIT,
+ "allocating execution environment",
+ __FILE__, __LINE__);
+
+ rc = SQLAllocConnect(henv, &hdbc);
+ handle_errors(NULL, NULL, rc, ERROR_EXIT,
+ "allocating connection handle",
+ __FILE__, __LINE__);
+
+ /* Connect to data store */
+
+ status_msg0("Connecting to the data source...\n");
+
+ /* Set up the connection options if not specified on the command line
+ * (default to TimesTen settings).
+ */
+
+ if ( !*szConnStrIn ) {
+ /* Running the benchmark with a scale factor creates (scale) branches,
+ * (scale x 10) tellers, (scale x 10000) accounts and ((scale-1) x 10000)
+ * non-local accounts. The size of the table rows are branches (141)
+ * tellers (141) and accounts (141). Therefore the data size requirements
+ * of this benchmark is:
+ * size ~= 141 * ((scale * 20011) - 10000) (bytes)
+ *
+ * Multiply data size by 20% to account for additional DB overhead (e.g.
+ * indexes), and round up the nearest 10Mb for safety.
+ */
+
+ int est_size = (int) (3.6 * scaleFactor + 10.0);
+ est_size = est_size - (est_size % 10);
+
+ sprintf(szConnStrIn,"OverWrite=1;PermSize=%d;%s",
+ est_size, DSNNAME);
+ status_msg0("Connecting to the data source... %s \n", szConnStrIn);
+ }
+
+ rc = SQLDriverConnect(hdbc, NULL, (SQLCHAR *) szConnStrIn, SQL_NTS,
+ NULL, 0, NULL,
+ SQL_DRIVER_NOPROMPT);
+
+ status_msg0("Connected to the data source...\n");
+ sprintf(errstr, "connecting to driver (connect string %s)\n",
+ szConnStrIn);
+ handle_errors(hdbc, NULL, rc, ERROR_EXIT,
+ errstr, __FILE__, __LINE__);
+
+ /* Turn auto-commit off */
+
+ rc = SQLSetConnectOption(hdbc, SQL_AUTOCOMMIT, SQL_AUTOCOMMIT_OFF);
+ handle_errors(hdbc, NULL, rc, DISCONNECT_EXIT,
+ "switching off the AUTO_COMMIT option",
+ __FILE__, __LINE__);
+
+ /* Allocate a statement handle */
+
+ rc = SQLAllocStmt(hdbc, &hstmt);
+ handle_errors(hdbc, NULL, rc, DISCONNECT_EXIT,
+ "allocating a statement handle",
+ __FILE__, __LINE__);
+
+ /* (Implicit) Transaction begin */
+
+ /* Determine the DBMS Type*/
+
+ DBMSName[0] = '\0';
+ rc = SQLGetInfo(hdbc, SQL_DBMS_NAME, (PTR) &DBMSName,
+ sizeof(DBMSName), NULL);
+ rc = SQLGetInfo(hdbc, SQL_DRIVER_VER, (PTR) &DBMSVersion,
+ sizeof(DBMSVersion), NULL);
+
+ if (strcmp(DBMSName, "TimesTen") == 0)
+ DBMSType = DBMS_TIMESTEN;
+ else if (strcmp(DBMSName, "Microsoft SQL Server") == 0)
+ DBMSType = DBMS_MSSQL;
+ else DBMSType = DBMS_UNKNOWN;
+
+ /* if not TimesTen: delete (if it exists), create & use the new database */
+
+ if (DBMSType != DBMS_TIMESTEN) {
+ status_msg0("Deleting the database...\n");
+ rc = SQLExecDirect(hstmt, (SQLCHAR *) DatabaseDropStmt, SQL_NTS);
+
+ /* estimate database size, size = data space + log space
+ * data space = (#tuples)/(tuples per page) * 2K bytes/page
+ * tuples per page = useable page size / row size (no index) = 2016/(96+2)
+ * log space = #transactions * average log size for the program transaction mix
+ * database size is in MB
+ */
+
+ databaseSize = (int) ceil((((numBranchTups + numTellerTups + numAccountTups)/
+ (2016/98)) * 2048 + (numXacts * 600)) / 1000000.0);
+
+ status_msg1("Creating the database (%dMB)...\n", databaseSize);
+#ifndef NDB
+ sprintf(cmdStr, DatabaseCreateStmt, databaseSize);
+ doImmed(hdbc, hstmt, cmdStr);
+ strcpy(cmdStr, DatabaseUseStmt);
+ doImmed(hdbc, hstmt, cmdStr);
+#endif
+ }
+
+ status_msg2("Connected to '%s' version '%s'...\n", DBMSName, DBMSVersion);
+
+ /* create branches table */
+ status_msg0("Creating tasddbles...\n");
+#ifndef NDB
+ if (DBMSType == DBMS_TIMESTEN)
+ sprintf(cmdStr, TTBranchCrTblStmt, numBranchTups/TuplesPerPage + 1);
+ else
+#endif
+ sprintf(cmdStr, BranchCrTblStmt);
+ doImmed(hdbc, hstmt, cmdStr);
+
+ /* create tellers table */
+#ifndef NDB
+ if (DBMSType == DBMS_TIMESTEN)
+ sprintf(cmdStr, TTTellerCrTblStmt, numTellerTups/TuplesPerPage + 1);
+
+ else
+#endif
+ sprintf(cmdStr, TellerCrTblStmt);
+ doImmed(hdbc, hstmt, cmdStr);
+
+ /* create accounts table */
+#ifndef NDB
+ if (DBMSType == DBMS_TIMESTEN)
+ sprintf(cmdStr, TTAccountCrTblStmt, numAccountTups/TuplesPerPage + 1);
+ else
+#endif
+ sprintf(cmdStr, AccountCrTblStmt);
+ doImmed(hdbc, hstmt, cmdStr);
+
+ /* create History table */
+
+ doImmed(hdbc, hstmt, HistoryCrTblStmt);
+
+ /* lock the database during population */
+#ifndef NDB
+ if ( DBMSType == DBMS_TIMESTEN ) {
+ rc = SQLExecDirect(hstmt, (SQLCHAR *)"call ttlocklevel('DS')", SQL_NTS);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "specifying dbs lock usage",
+ __FILE__, __LINE__);
+ /* make sure dbs lock take effect in next transaction */
+ rc = SQLTransact(henv,hdbc,SQL_COMMIT);
+ if ( rc != SQL_SUCCESS) {
+ handle_errors(hdbc, SQL_NULL_HSTMT, rc, ERROR_EXIT,
+ "committing transaction",
+ __FILE__, __LINE__);
+ }
+ }
+#endif
+ /* populate branches table */
+
+
+ rc = SQLPrepare(hstmt, (SQLCHAR *) insStmt[0], SQL_NTS);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "preparing statement",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(hstmt, 1, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_INTEGER, 10, 0, &branchNum, sizeof branchNum, NULL);
+
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+
+ status_msg1("Populating branches table (%" PTRINT_FMT " rows)...\n",
+ numBranchTups);
+
+
+ for (i=0; i<numBranchTups; i++) {
+ branchNum = i;
+ rc = SQLExecute(hstmt);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "Error executing statement",
+ __FILE__, __LINE__);
+ }
+
+ /* Reset all bind-parameters for the statement handle. */
+ rc = SQLFreeStmt(hstmt, SQL_RESET_PARAMS);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "resetting parms on statement handle",
+ __FILE__, __LINE__);
+
+ /* populate tellers table */
+
+ status_msg1("Populating tellers table (%" PTRINT_FMT " rows)...\n",
+ numTellerTups);
+
+
+ rc = SQLPrepare(hstmt, (SQLCHAR *) insStmt[1], SQL_NTS);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "preparing statement",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(hstmt, 1, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_INTEGER,
+ 10, 0, &tellerNum, sizeof tellerNum, NULL);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(hstmt, 2, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_INTEGER,
+ 10, 0, &branchNum, sizeof branchNum, NULL);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ for (i=0; i<numTellerTups; i++) {
+ tellerNum = i;
+ branchNum = i/TellersPerBranch;
+ rc = SQLExecute(hstmt);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "Error executing statement",
+ __FILE__, __LINE__);
+ }
+
+ /* Reset all bind-parameters for the statement handle. */
+
+ rc = SQLFreeStmt(hstmt, SQL_RESET_PARAMS);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "resetting parms on statement handle",
+ __FILE__, __LINE__);
+
+ /* populate accounts table */
+
+ status_msg1("Populating accounts table (%" PTRINT_FMT " rows)...\n",
+ numAccountTups);
+
+ rc = SQLPrepare(hstmt, (SQLCHAR *) insStmt[2], SQL_NTS);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "preparing statement",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(hstmt, 1, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_INTEGER,
+ 10, 0, &accountNum, sizeof accountNum, NULL);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(hstmt, 2, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_INTEGER,
+ 10, 0, &branchNum, sizeof branchNum, NULL);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ for (i=0; i<numAccountTups; i++) {
+ accountNum = i;
+ branchNum = i/AccountsPerBranch;
+ rc = SQLExecute(hstmt);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "Error executing statement",
+ __FILE__, __LINE__);
+ }
+ status_msg0("Commit...\n");
+ rc = SQLTransact(henv, hdbc, SQL_COMMIT);
+ status_msg0("Commit done...\n");
+ handle_errors(hdbc, NULL, rc, ERROR_EXIT,
+ "committing transaction",
+ __FILE__, __LINE__);
+
+ /* compile SQL statements of transaction */
+
+ status_msg0("Compiling statements of transaction...\n");
+ for (i=0; i<NumXactStmts; i++) {
+#ifndef NDB
+ rc = SQLAllocStmt(hdbc, &txstmt[i]);
+ handle_errors(hdbc, NULL, rc, ABORT_DISCONNECT_EXIT,
+ "allocating a statement handle",
+ __FILE__, __LINE__);
+#else
+ rc = SQLAllocHandle(SQL_HANDLE_STMT, hdbc, &txstmt[i]);
+ handle_errors(hdbc, NULL, rc, ABORT_DISCONNECT_EXIT,
+ "allocating a statement handle",
+ __FILE__, __LINE__);
+
+#endif
+
+ rc = SQLPrepare(txstmt[i], (SQLCHAR *) tpcbXactStmt[i], SQL_NTS);
+ handle_errors(hdbc, txstmt[i], rc, ABORT_DISCONNECT_EXIT,
+ "preparing statement",
+ __FILE__, __LINE__);
+ }
+
+ /* unuse dbs lock */
+#ifndef NDB
+ if ( DBMSType == DBMS_TIMESTEN ) {
+ rc = SQLExecDirect(hstmt, (SQLCHAR *)"call ttlocklevel('Row')", SQL_NTS);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "specifying row lock usage",
+ __FILE__, __LINE__);
+ }
+#endif
+
+ /* commit transaction */
+
+ rc = SQLTransact(henv, hdbc, SQL_COMMIT);
+ handle_errors(hdbc, NULL, rc, ERROR_EXIT,
+ "committing transaction",
+ __FILE__, __LINE__);
+
+
+ /* Initialize random seed and timers */
+
+ srand48(SeedVal);
+ localLimit = (unsigned short)((1<<16) * 0.85);
+
+ /* Initialize parameter lists for each of the transactions */
+
+ rc = SQLBindParameter(txstmt[0], 1, SQL_PARAM_INPUT, SQL_C_DOUBLE,
+ SQL_DOUBLE, 15, 0, &delta, sizeof delta, NULL);
+ handle_errors(hdbc, txstmt[0], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[0], 2, SQL_PARAM_INPUT, SQL_C_SLONG,
+ SQL_INTEGER, 10, 0, &accountNum, sizeof accountNum,
+ NULL);
+ handle_errors(hdbc, txstmt[0], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[1], 1, SQL_PARAM_INPUT, SQL_C_SLONG,
+ SQL_INTEGER, 10, 0, &accountNum, sizeof accountNum,
+ NULL);
+ handle_errors(hdbc, txstmt[1], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[2], 1, SQL_PARAM_INPUT, SQL_C_DOUBLE,
+ SQL_DOUBLE, 15, 0, &delta, sizeof delta, NULL);
+ handle_errors(hdbc, txstmt[2], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[2], 2, SQL_PARAM_INPUT, SQL_C_SLONG,
+ SQL_INTEGER, 10, 0, &tellerNum, sizeof tellerNum,
+ NULL);
+ handle_errors(hdbc, txstmt[2], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[3], 1, SQL_PARAM_INPUT, SQL_C_DOUBLE,
+ SQL_DOUBLE, 15, 0, &delta, sizeof delta, NULL);
+ handle_errors(hdbc, txstmt[3], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[3], 2, SQL_PARAM_INPUT, SQL_C_SLONG,
+ SQL_INTEGER, 10, 0, &branchNum, sizeof branchNum,
+ NULL);
+ handle_errors(hdbc, txstmt[3], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[4], 1, SQL_PARAM_INPUT, SQL_C_SLONG,
+ SQL_INTEGER, 10, 0, &tellerNum, sizeof tellerNum,
+ NULL);
+ handle_errors(hdbc, txstmt[4], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[4], 2, SQL_PARAM_INPUT, SQL_C_SLONG,
+ SQL_INTEGER, 10, 0, &branchNum, sizeof branchNum,
+ NULL);
+ handle_errors(hdbc, txstmt[4], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[4], 3, SQL_PARAM_INPUT, SQL_C_SLONG,
+ SQL_INTEGER, 10, 0, &accountNum, sizeof accountNum,
+ NULL);
+ handle_errors(hdbc, txstmt[4], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[4], 4, SQL_PARAM_INPUT, SQL_C_DOUBLE,
+ SQL_DOUBLE, 15, 0, &delta, sizeof delta, NULL);
+ handle_errors(hdbc, txstmt[4], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ rc = SQLBindParameter(txstmt[4], 5, SQL_PARAM_INPUT, SQL_C_SLONG,
+ SQL_INTEGER, 10, 0, &timeStamp, sizeof timeStamp,
+ NULL);
+ handle_errors(hdbc, txstmt[4], rc, ABORT_DISCONNECT_EXIT,
+ "binding parameter",
+ __FILE__, __LINE__);
+
+ /* Execute transaction loop.
+ * Do it twice, once briefly as a warm-up. */
+
+
+
+ for (warmup = 1; warmup >= 0; warmup--) {
+
+ int max_i = (warmup ? numXacts/10 : numXacts);
+
+ /* Execute tpcb transaction max_i times.*/
+
+ if (warmup) {
+ status_msg1("\nWarming up with %d tpcb transactions...\n", max_i);
+ }
+ else {
+ status_msg1("Executing and timing %d tpcb transactions...\n", max_i);
+ }
+
+ ttGetWallClockTime(&startT);
+ ttGetThreadTimes(&startRes);
+
+ for (i = 0; i < max_i; i++) {
+
+ lrand = lrand48();
+ srands = (unsigned short *)(&lrand);
+ if (lp64) srands += 2; /* skip high half -- all zero */
+
+ /* randomly choose a teller */
+
+ tellerNum = srands[0] % numTellerTups;
+
+ /* compute branch */
+
+ branchNum = (tellerNum / TellersPerBranch);
+
+ /* randomly choose an account */
+
+ if (srands[1] < localLimit || numBranchTups == 1) {
+
+ /* choose account local to selected branch */
+
+ accountNum = branchNum * AccountsPerBranch +
+ (lrand48() % AccountsPerBranch);
+
+ ++numLocalXacts;
+
+ }
+ else {
+ /* choose account not local to selected branch */
+
+ /* first select account in range [0,numNonLocalAccountTups) */
+
+ accountNum = lrand48() % numNonLocalAccountTups;
+
+ /* if branch number of selected account is at least as big
+ * as local branch number, then increment account number
+ * by AccountsPerBranch to skip over local accounts
+ */
+
+ if ((accountNum/AccountsPerBranch) >= branchNum)
+ accountNum += AccountsPerBranch;
+
+ ++numRemoteXacts;
+ }
+
+ /* select delta amount, -999,999 to +999,999 */
+
+ delta = ((lrand48() % 1999999) - 999999);
+
+
+ /* begin timing the "residence time" */
+
+ ttGetWallClockTime(rtStart[i]);
+
+ for ( j = 0; j < NumXactStmts - 2; j++) {
+ rc = SQLExecute(txstmt[j]);
+ handle_errors(hdbc, txstmt[j], rc, ABORT_DISCONNECT_EXIT,
+ "Error executing statement1",
+ __FILE__, __LINE__);
+
+ /* Close the handle after the SELECT statement
+ * (txstmt[1]) for non TimesTen DBMS' */
+
+ if ((DBMSType != DBMS_TIMESTEN) && (j == 1)) {
+ SQLFreeStmt(txstmt[1], SQL_CLOSE);
+ }
+
+
+ }
+
+ /* note that time must be taken within the */
+ timeStamp = time(NULL);
+
+ rc = SQLExecute(txstmt[NumXactStmts - 1]);
+ handle_errors(hdbc, txstmt[NumXactStmts - 1], rc,
+ ABORT_DISCONNECT_EXIT, "Error executing statement2",
+ __FILE__, __LINE__);
+
+ rc = SQLTransact(henv, hdbc, SQL_COMMIT);
+ handle_errors(hdbc, NULL, rc, ERROR_EXIT,
+ "Error committing transaction",
+ __FILE__, __LINE__);
+
+ ttGetWallClockTime(rtEnd[i]);
+
+ } /* end fortransaction loop */
+
+
+
+
+ ttGetThreadTimes(&endRes);
+ ttGetWallClockTime(&endT);
+ ttCalcElapsedThreadTimes(&startRes, &endRes, &kernel, &user);
+ ttCalcElapsedWallClockTime(&startT, &endT, &real);
+
+ if (warmup) {
+ if (!tabFlag) {
+ if (verbose) {
+ if (fThreadTime) {
+ out_msg0(" time user system\n");
+
+ out_msg3("Warmup time (sec): %12.3f %12.3f %12.3f\n\n",
+ real/1000.0, user, kernel);
+ } else {
+ out_msg1("Warmup time (sec): %12.3f\n\n", real/1000.0);
+ }
+ }
+ } else {
+ if (verbose) {
+ if (fThreadTime) {
+ out_msg0("\ttime\tuser\tsystem\n");
+
+ out_msg3("Warmup time (sec):\t%12.3f\t%12.3f\t%12.3f\n",
+ real/1000.0, user, kernel);
+ } else {
+ out_msg1("Warmup time (sec):\t%12.3f\n", real/1000.0);
+ }
+ }
+ }
+ }
+ }
+
+ status_msg0("\nExecution completed...\n");
+
+ /* Compute and report timing statistics */
+
+ maxTime = 0.0;
+ totTime = 0.0;
+
+ for (i = 0; i < numXacts; i++) {
+ ttCalcElapsedWallClockTime(rtStart[i], rtEnd[i], resTime[i]);
+ totTime += *(resTime[i]);
+
+ if (*(resTime[i]) > maxTime) maxTime = *(resTime[i]);
+ }
+
+ if (!tabFlag) {
+ if (verbose) {
+ if (fThreadTime) {
+ out_msg0(" time user system\n");
+ out_msg3("Total time (sec): %12.3f %12.3f %12.3f\n",
+ real/1000.0, user, kernel);
+ } else {
+ out_msg1("Total time (sec): %12.3f\n", real/1000.0);
+ }
+ }
+
+ if (verbose)
+ out_msg1("\nAverage transaction time (msec):%12.3f\n",
+ totTime/numXacts);
+ if (verbose)
+ out_msg1("Maximum transaction time (msec):%12.3f\n", maxTime);
+ if (verbose)
+ out_msg1("\nLocal transactions: %7" PTRINT_FMT "\n", numLocalXacts);
+ if (verbose)
+ out_msg1("Remote transactions: %7" PTRINT_FMT "\n", numRemoteXacts);
+
+ } else {
+ if (verbose) {
+ if (fThreadTime) {
+ out_msg0("\ttime\tuser\tsystem\n");
+ out_msg3("Total time (sec):\t%12.3f\t%12.3f\t%12.3f\n",
+ real/1000.0, user, kernel);
+ } else {
+ out_msg1("Total time (sec):\t%12.3f\n", real/1000.0);
+ }
+ }
+
+ if (verbose)
+ out_msg1("\nAverage transaction time (msec):\t%12.3f\n",
+ totTime/numXacts);
+ if (verbose)
+ out_msg1("Maximum transaction time (msec):\t%12.3f\n", maxTime);
+ if (verbose)
+ out_msg1("Local transactions:\t%7" PTRINT_FMT "\n", numLocalXacts);
+
+ if (verbose)
+ out_msg1("Remote transactions:\t%7" PTRINT_FMT "\n", numRemoteXacts);
+
+
+
+ }
+
+ /* If the statfile option is selected, print each transaction's time */
+
+ if (printXactTimes) {
+ FILE * fp;
+ if ( (fp = fopen (statFile, "w")) == NULL ) {
+ err_msg1("Unable to open stat file %s for writing\n\n", statFile);
+ } else {
+ for (int i = 0; i < numXacts; i++)
+ fprintf(fp,"%6d: %12.3f\n", i, *(resTime[i]));
+ fclose(fp);
+ }
+ }
+
+ /* Disconnect and return */
+
+ rc = SQLFreeStmt(hstmt, SQL_DROP);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "dropping the statement handle",
+ __FILE__, __LINE__);
+
+ for (int i=0; i<NumXactStmts; i++) {
+ rc = SQLFreeStmt(txstmt[i], SQL_DROP);
+ handle_errors(hdbc, hstmt, rc, ABORT_DISCONNECT_EXIT,
+ "dropping the statement handle",
+ __FILE__, __LINE__);
+ }
+
+ if (verbose >= VERBOSE_DFLT)
+ status_msg0("Disconnecting from the data source...\n");
+
+ rc = SQLDisconnect(hdbc);
+ handle_errors(hdbc, NULL, rc, ERROR_EXIT,
+ "disconnecting",
+ __FILE__, __LINE__);
+
+ rc = SQLFreeConnect(hdbc);
+ handle_errors(hdbc, NULL, rc, ERROR_EXIT,
+ "freeing connection handle",
+ __FILE__, __LINE__);
+
+ rc = SQLFreeEnv(henv);
+ handle_errors(NULL, NULL, rc, ERROR_EXIT,
+ "freeing environment handle",
+ __FILE__, __LINE__);
+
+ app_exit(0);
+ return 0;
+}
+
+
+
+
+
+/* Emacs variable settings */
+/* Local Variables: */
+/* tab-width:8 */
+/* indent-tabs-mode:nil */
+/* c-basic-offset:2 */
+/* End: */
+
+
+
diff --git a/ndb/test/odbc/tpcb/ttTime.c b/ndb/test/odbc/tpcb/ttTime.c
new file mode 100644
index 00000000000..8f10b0c6b91
--- /dev/null
+++ b/ndb/test/odbc/tpcb/ttTime.c
@@ -0,0 +1,366 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+static const volatile char cvsid[] = "$Id: ttTime.c,v 1.1 2003/09/23 12:43:46 johan Exp $";
+/*
+ * $Revision: 1.1 $
+ * (c) Copyright 1996-2003, TimesTen, Inc.
+ * All rights reserved.
+ *
+ */
+
+
+/* Contains functions for performing elapsed-time calculations
+ in a portable manner */
+
+#include "ttTime.h"
+
+#ifdef WIN32
+
+#include <stdio.h>
+#include <mapiutil.h>
+
+/*------------*/
+/* NT VERSION */
+/*------------*/
+
+/*********************************************************************
+ *
+ * FUNCTION: ttGetThreadTimes
+ *
+ * DESCRIPTION: This function sets the supplied parameter's
+ * user and kernel time for the current thread.
+ *
+ * PARAMETERS: ttThreadTimes* timesP thread time structure
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+void
+ttGetThreadTimes(ttThreadTimes* timesP)
+{
+ BOOL rc;
+ HANDLE curThread;
+ FILETIME creationTime;
+ FILETIME exitTime;
+ FILETIME kTime;
+ FILETIME uTime;
+
+ memset (&kTime, 0, sizeof (FILETIME));
+ memset (&uTime, 0, sizeof (FILETIME));
+
+ curThread = GetCurrentThread();
+ rc = GetThreadTimes(curThread,
+ &creationTime,
+ &exitTime,
+ &kTime,
+ &uTime);
+
+ timesP->kernelTime = kTime;
+ timesP->userTime = uTime;
+
+}
+
+/*********************************************************************
+ *
+ * FUNCTION: ttCalcElapsedThreadTimes
+ *
+ * DESCRIPTION: This function calculates the user and kernel
+ * time deltas.
+ *
+ * PARAMETERS: ttThreadTimes* beforeP beginning timestamp (IN)
+ * ttThreadTimes* afterP ending timestamp (IN)
+ * double* kernelDeltaP kernel time delta (OUT)
+ * double* userDeltaP user time delta (OUT)
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+void
+ttCalcElapsedThreadTimes(ttThreadTimes* beforeP,
+ ttThreadTimes* afterP,
+ double* kernelDeltaP,
+ double* userDeltaP)
+{
+ static const double secPerHi = (double) 4.294967296; /* 2**32 * 10**-9 */
+ FILETIME *before, *after;
+
+ before = &beforeP->kernelTime;
+ after = &afterP->kernelTime;
+ *kernelDeltaP = (double) ((after->dwHighDateTime - before->dwHighDateTime) * secPerHi
+ + (after->dwLowDateTime - before->dwLowDateTime) * 100e-9);
+ before = &beforeP->userTime;
+ after = &afterP->userTime;
+ *userDeltaP = (double) ((after->dwHighDateTime - before->dwHighDateTime) * secPerHi
+ + (after->dwLowDateTime - before->dwLowDateTime) * 100e-9);
+}
+
+/*********************************************************************
+ *
+ * FUNCTION: ttGetWallClockTime
+ *
+ * DESCRIPTION: This function gets the current wall-clock time.
+ *
+ * PARAMETERS: ttWallClockTime* timeP tms time structure (OUT)
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+void
+ttGetWallClockTime(ttWallClockTime* timeP)
+{
+ LARGE_INTEGER frequency;
+ if ( QueryPerformanceFrequency(&frequency) ) {
+ QueryPerformanceCounter(&(timeP->time64));
+ }
+ else {
+ _ftime(&(timeP->notSoLargeTime));
+ }
+}
+
+/*********************************************************************
+ *
+ * FUNCTION: ttCalcElapsedWallClockTime
+ *
+ * DESCRIPTION: This function calculates the elapsed wall-clock
+ * time in msec.
+ *
+ * PARAMETERS: ttWallClockTime* beforeP starting timestamp
+ * ttWallClockTime* afterP ending timestamp
+ * double* nmillisecondsP elapsed time (OUT)
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+void
+ttCalcElapsedWallClockTime(ttWallClockTime* beforeP,
+ ttWallClockTime* afterP,
+ double* nmillisecondsP)
+{
+ LARGE_INTEGER frequency;
+
+ if ( QueryPerformanceFrequency(&frequency) ) {
+ *nmillisecondsP = 1000 * ((double) (afterP->time64.QuadPart
+ - beforeP->time64.QuadPart))
+ / frequency.QuadPart;
+
+ }
+ else {
+ double start;
+ double end;
+
+ start = (double) beforeP->notSoLargeTime.time * 1000. +
+ (double) beforeP->notSoLargeTime.millitm;
+ end = (double) afterP->notSoLargeTime.time * 1000. +
+ (double) afterP->notSoLargeTime.millitm;
+
+ *nmillisecondsP = (double) (end - start);
+ }
+}
+
+#elif defined (RTSYS_VXWORKS)
+
+/*-----------------*/
+/* VxWorks VERSION */
+/*-----------------*/
+
+/*
+ * The TimeBase registers have a period of 60ns, i.e.
+ * 0.00000006 or (6e-8) seconds.
+ */
+#define TIMER_MSEC_PER_CYC (6e-5)
+
+void
+ttGetWallClockTime(ttWallClockTime* timeP)
+{
+ vxTimeBaseGet(&timeP->sep.upper32, &timeP->sep.lower32);
+}
+
+
+void
+ttCalcElapsedWallClockTime(ttWallClockTime* beforeP,
+ ttWallClockTime* afterP,
+ double* nmillisecondsP)
+{
+ *nmillisecondsP = (double)(afterP->val - beforeP->val) * TIMER_MSEC_PER_CYC;
+}
+
+
+#else
+
+/*--------------*/
+/* UNIX VERSION */
+/*--------------*/
+
+#include <unistd.h>
+
+/*********************************************************************
+ *
+ * FUNCTION: ttGetThreadTimes
+ *
+ * DESCRIPTION: This function sets the supplied parameter's
+ * tms structure.
+ *
+ * PARAMETERS: ttThreadTimes* timesP tms time structure
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+#ifdef SB_P_OS_CHORUS
+void ttGetThreadTimes(ttThreadTimes* timesP)
+{
+ KnCap actorCap;
+
+ if (acap (agetId(), &actorCap) == -1) {
+ timesP->ins.tmSec = 0;
+ timesP->ins.tmNSec = 0;
+ timesP->ext.tmSec = 0;
+ timesP->ext.tmNSec = 0;
+ }
+ else {
+ (void) threadTimes (&actorCap, K_ALLACTORTHREADS,
+ &timesP->ins, &timesP->ext);
+ }
+}
+#else
+void ttGetThreadTimes(ttThreadTimes* timesP)
+{
+ (void) times(timesP);
+}
+#endif
+
+/*********************************************************************
+ *
+ * FUNCTION: ttCalcElapsedThreadTimes
+ *
+ * DESCRIPTION: This function calculates the user and kernel
+ * time deltas.
+ *
+ * PARAMETERS: ttThreadTimes* beforeP beginning timestamp (IN)
+ * ttThreadTimes* afterP ending timestamp (IN)
+ * double* kernelDeltaP kernel time delta (OUT)
+ * double* userDeltaP user time delta (OUT)
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+#ifdef SB_P_OS_CHORUS
+void
+ttCalcElapsedThreadTimes(ttThreadTimes* beforeP,
+ ttThreadTimes* afterP,
+ double* kernelDeltaP,
+ double* userDeltaP)
+{
+ double kernelBefore;
+ double kernelAfter;
+ double userBefore;
+ double userAfter;
+
+ kernelBefore = (beforeP->ext.tmSec) + (beforeP->ext.tmNSec / 1e9);
+ kernelAfter = (afterP->ext.tmSec) + (afterP->ext.tmNSec / 1e9);
+ *kernelDeltaP = kernelAfter - kernelBefore;
+
+ userBefore = (beforeP->ins.tmSec) + (beforeP->ins.tmNSec / 1e9);
+ userAfter = (afterP->ins.tmSec) + (afterP->ins.tmNSec / 1e9);
+ *userDeltaP = userAfter - userBefore;
+
+}
+#else
+void
+ttCalcElapsedThreadTimes(ttThreadTimes* beforeP,
+ ttThreadTimes* afterP,
+ double* kernelDeltaP,
+ double* userDeltaP)
+{
+ double ticks = (double)sysconf(_SC_CLK_TCK);
+
+ *kernelDeltaP = (afterP->tms_stime - beforeP->tms_stime) / ticks;
+ *userDeltaP = (afterP->tms_utime - beforeP->tms_utime) / ticks;
+}
+#endif
+
+/*********************************************************************
+ *
+ * FUNCTION: ttGetWallClockTime
+ *
+ * DESCRIPTION: This function gets the current wall-clock time.
+ *
+ * PARAMETERS: ttWallClockTime* timeP tms time structure (OUT)
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+void
+ttGetWallClockTime(ttWallClockTime* timeP)
+{
+ gettimeofday(timeP, NULL);
+}
+
+/*********************************************************************
+ *
+ * FUNCTION: ttCalcElapsedWallClockTime
+ *
+ * DESCRIPTION: This function calculates the elapsed wall-clock
+ * time is msec.
+ *
+ * PARAMETERS: ttWallClockTime* beforeP starting timestamp
+ * ttWallClockTime* afterP ending timestamp
+ * double* nmillisecondsP elapsed time (OUT)
+ *
+ * RETURNS: void
+ *
+ * NOTES: NONE
+ *
+ *********************************************************************/
+
+void
+ttCalcElapsedWallClockTime(ttWallClockTime* beforeP,
+ ttWallClockTime* afterP,
+ double* nmillisP)
+{
+ *nmillisP = (afterP->tv_sec - beforeP->tv_sec)*1000.0 +
+ (afterP->tv_usec - beforeP->tv_usec)/1000.0;
+}
+
+#endif
+
+/* Emacs variable settings */
+/* Local Variables: */
+/* tab-width:8 */
+/* indent-tabs-mode:nil */
+/* c-basic-offset:2 */
+/* End: */
diff --git a/ndb/test/odbc/tpcb/ttTime.h b/ndb/test/odbc/tpcb/ttTime.h
new file mode 100644
index 00000000000..f78b71667fe
--- /dev/null
+++ b/ndb/test/odbc/tpcb/ttTime.h
@@ -0,0 +1,125 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*
+ * $Revision: 1.1 $
+ * (c) Copyright 1996-2003, TimesTen, Inc.
+ * All rights reserved.
+ *
+ */
+
+#ifndef __TT_TIME
+#define __TT_TIME
+
+
+#ifdef WIN32
+
+#include <windows.h>
+#include <sys/types.h>
+#include <sys/timeb.h>
+
+typedef struct {
+ FILETIME kernelTime;
+ FILETIME userTime;
+} ttThreadTimes;
+
+
+typedef union {
+ LARGE_INTEGER time64;
+ struct _timeb notSoLargeTime;
+} ttWallClockTime;
+
+#elif defined(RTSYS_VXWORKS)
+
+#define srand48(x) sb_srand48((x))
+#define drand48() sb_drand48()
+
+#ifdef SB_P_OS_VXPPC
+/* For little-endian switch the lower, upper fields */
+typedef union {
+ struct {
+ unsigned int upper32;
+ unsigned int lower32;
+ } sep;
+ long long val;
+} ttWallClockTime;
+
+/*
+ * This is a VxWorks private function to read the PPC's 64 bit Time Base
+ * Register. This is the assembler dump of this function.
+ 001126e4 7cad42e6 mftb r5, TBU
+ 001126e8 7ccc42e6 mftb r6, TBL
+ 001126ec 7ced42e6 mftb r7, TBU
+ 001126f0 7c053800 cmp crf0, 0, r5, r7
+ 001126f4 4082fff0 bc 0x4, 0x2, vxTimeBaseGet
+ 001126f8 90a30000 stw r5, 0x0(r3)
+ 001126fc 90c40000 stw r6, 0x0(r4)
+ 00112700 4e800020 blr
+ * This is a fine grained timer with a period of 60ns.
+ */
+void vxTimeBaseGet(unsigned int* pUpper32, unsigned int* pLower32);
+#endif /* SB_P_OS_VXPPC */
+
+#elif defined(SB_P_OS_CHORUS)
+#include <sys/types.h>
+#include <sys/times.h>
+#include <sys/time.h>
+
+#include <vtimer/chVtimer.h>
+
+struct chrTimes {
+ KnTimeVal ins;
+ KnTimeVal ext;
+};
+typedef struct chrTimes ttThreadTimes;
+
+typedef struct timeval ttWallClockTime;
+
+#else
+/* UNIX version */
+
+#include <sys/times.h>
+#include <sys/time.h>
+
+typedef struct tms ttThreadTimes;
+
+typedef struct timeval ttWallClockTime;
+
+#endif /* NT, VxWorks, Chorus, Unix */
+
+
+#ifndef RTSYS_VXWORKS
+void ttGetThreadTimes(ttThreadTimes* timesP);
+void ttCalcElapsedThreadTimes(ttThreadTimes* beforeP, ttThreadTimes* afterP,
+ double* kernelDeltaP, double* userDeltaP);
+#endif /* ! VXWORKS */
+void ttGetWallClockTime(ttWallClockTime* timeP);
+void ttCalcElapsedWallClockTime(ttWallClockTime* beforeP,
+ ttWallClockTime* afterP,
+ double* nmillisecondsP);
+
+
+
+
+
+#endif /* __TT_TIME */
+
+/* Emacs variable settings */
+/* Local Variables: */
+/* tab-width:8 */
+/* indent-tabs-mode:nil */
+/* c-basic-offset:2 */
+/* End: */
diff --git a/ndb/test/run-test/Makefile b/ndb/test/run-test/Makefile
new file mode 100644
index 00000000000..6b4689b2dbb
--- /dev/null
+++ b/ndb/test/run-test/Makefile
@@ -0,0 +1,22 @@
+include .defs.mk
+
+TYPE := util
+
+BIN_TARGET := atrt
+BIN_TARGET_LIBS := mgmapi
+
+SOURCES = main.cpp
+SCRIPTS = atrt-analyze-result.sh atrt-gather-result.sh atrt-setup.sh \
+ atrt-clear-result.sh make-config.sh
+
+OBJECTS_LOC = $(call fixpath,$(NDB_TOP)/src/mgmclient/CpcClient.o)
+
+CFLAGS_main.cpp := -I$(call fixpath,$(NDB_TOP)/src/mgmclient)
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/include/mgmapi)
+
+include $(NDB_TOP)/Epilogue.mk
+
+_bins::
+ -rm -f $(SCRIPTS:%=$(NDB_TOP)/bin/%)
+ cp $(SCRIPTS) $(NDB_TOP)/bin
+
diff --git a/ndb/test/run-test/README.ATRT b/ndb/test/run-test/README.ATRT
new file mode 100644
index 00000000000..7fe04ccdac4
--- /dev/null
+++ b/ndb/test/run-test/README.ATRT
@@ -0,0 +1,34 @@
+
+!-- install ndb_cpcd
+!-- many steps? future RPM
+
+!-- deploy binaries and libraries to hosts, rsync
+% export DEPLOY_DST="mc05:/space/tomas/keso"
+% ssh mc05 mkdir /space/tomas/keso
+% export RSYNC_RSH=ssh
+% make
+
+% mkdir -p /tmp/atrt-run-2-node
+% cd /tmp/atrt-run-2-node
+% cat > d.txt
+baseport: 9321
+basedir: /space/tomas/keso
+mgm: localhost
+ndb: localhost localhost
+api: localhost
+% cat > default.txt
+[DB DEFAULT]
+NoOfReplicas: 2
+% bin/make-config.sh -m d.txt -t default.txt -d .
+
+% atrt -v -v
+test_event -r 5 T1
+
+
+!-- check output
+% tail -f /space/tomas/keso/run/4.ndb_api/log.out
+
+
+!-- check processes
+% export NDB_CPCC_HOSTS="mc05"
+% ndb_cpcc
diff --git a/ndb/test/run-test/atrt-analyze-result.sh b/ndb/test/run-test/atrt-analyze-result.sh
new file mode 100755
index 00000000000..0fa46e918ef
--- /dev/null
+++ b/ndb/test/run-test/atrt-analyze-result.sh
@@ -0,0 +1,12 @@
+#!/bin/sh
+
+f=`find result -name 'log.out' | xargs grep "NDBT_ProgramExit: " | grep -c "Failed"`
+o=`find result -name 'log.out' | xargs grep "NDBT_ProgramExit: " | grep -c "OK"`
+
+if [ $o -gt 0 -a $f -eq 0 ]
+then
+ exit 0
+fi
+
+exit 1
+
diff --git a/ndb/test/run-test/atrt-clear-result.sh b/ndb/test/run-test/atrt-clear-result.sh
new file mode 100755
index 00000000000..57d3d43d247
--- /dev/null
+++ b/ndb/test/run-test/atrt-clear-result.sh
@@ -0,0 +1,4 @@
+#!/bin/sh
+
+set -e
+rm -rf result
diff --git a/ndb/test/run-test/atrt-gather-result.sh b/ndb/test/run-test/atrt-gather-result.sh
new file mode 100755
index 00000000000..93d4ae428d0
--- /dev/null
+++ b/ndb/test/run-test/atrt-gather-result.sh
@@ -0,0 +1,16 @@
+#!/bin/sh
+
+set -e
+
+mkdir -p result
+cd result
+rm -rf *
+
+while [ $# -gt 0 ]
+do
+ rsync -a "$1" .
+ shift
+done
+
+
+
diff --git a/ndb/test/run-test/atrt-setup.sh b/ndb/test/run-test/atrt-setup.sh
new file mode 100755
index 00000000000..aff5d4119dc
--- /dev/null
+++ b/ndb/test/run-test/atrt-setup.sh
@@ -0,0 +1,6 @@
+#!/bin/sh
+
+set -e
+
+ssh $1 mkdir -p $3
+rsync -a --delete --force --ignore-errors $2 $1:$3
diff --git a/ndb/test/run-test/main.cpp b/ndb/test/run-test/main.cpp
new file mode 100644
index 00000000000..1ce9124431c
--- /dev/null
+++ b/ndb/test/run-test/main.cpp
@@ -0,0 +1,942 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <stdio.h>
+#include <getarg.h>
+#include <BaseString.hpp>
+#include <Parser.hpp>
+#include <NdbOut.hpp>
+#include <Properties.hpp>
+#include <NdbAutoPtr.hpp>
+
+#include "run-test.hpp"
+#include <SysLogHandler.hpp>
+#include <FileLogHandler.hpp>
+
+#include <mgmapi.h>
+#include "CpcClient.hpp"
+
+/**
+ psuedo code for run-test.bin
+
+ define autotest_wrapper process at each host
+ start ndb-processes
+
+ for each testcase
+ do
+ start mysqld processes
+ start replication processes
+ start test programs
+
+ wait until test program finished or max time passed
+
+ stop test program
+ stop replication processes
+ stop mysqld processes
+
+ write report data-file
+ if test failed and ! last test
+ restart ndb processes
+
+ drop all tables created by test
+ done
+
+ stop ndb processes
+ undefined wrapper processes
+*/
+
+/** Global variables */
+static const char progname[] = "ndb_atrt";
+static const char * g_gather_progname = "atrt-gather-result.sh";
+static const char * g_analyze_progname = "atrt-analyze-result.sh";
+static const char * g_clear_progname = "atrt-clear-result.sh";
+static const char * g_setup_progname = "atrt-setup.sh";
+
+static const char * g_setup_path = 0;
+static const char * g_process_config_filename = "d.txt";
+static const char * g_log_filename = 0;
+static const char * g_test_case_filename = 0;
+static const char * g_report_filename = 0;
+
+static const char * g_default_user = 0;
+static const char * g_default_base_dir = 0;
+static int g_default_base_port = 0;
+
+static int g_report = 0;
+static int g_verbosity = 0;
+static FILE * g_report_file = 0;
+static FILE * g_test_case_file = stdin;
+
+Logger g_logger;
+atrt_config g_config;
+
+static int g_mode_bench = 0;
+static int g_mode_regression = 0;
+static int g_mode_interactive = 0;
+static int g_mode = 0;
+
+static
+struct getargs args[] = {
+ { "process-config", 0, arg_string, &g_process_config_filename, 0, 0 },
+ { "setup-path", 0, arg_string, &g_setup_path, 0, 0 },
+ { 0, 'v', arg_counter, &g_verbosity, 0, 0 },
+ { "log-file", 0, arg_string, &g_log_filename, 0, 0 },
+ { "testcase-file", 'f', arg_string, &g_test_case_filename, 0, 0 },
+ { 0, 'R', arg_flag, &g_report, 0, 0 },
+ { "report-file", 0, arg_string, &g_report_filename, 0, 0 },
+ { "interactive", 'i', arg_flag, &g_mode_interactive, 0, 0 },
+ { "regression", 'r', arg_flag, &g_mode_regression, 0, 0 },
+ { "bench", 'b', arg_flag, &g_mode_bench, 0, 0 },
+};
+
+const int arg_count = 10;
+
+int
+main(int argc, const char ** argv){
+
+ bool restart = true;
+ int lineno = 1;
+ int test_no = 1;
+
+ const int p_ndb = atrt_process::NDB_MGM | atrt_process::NDB_DB;
+ const int p_servers = atrt_process::MYSQL_SERVER | atrt_process::NDB_REP;
+ const int p_clients = atrt_process::MYSQL_CLIENT | atrt_process::NDB_API;
+
+ g_logger.setCategory(progname);
+ g_logger.enable(Logger::LL_ALL);
+ g_logger.createConsoleHandler();
+
+ if(!parse_args(argc, argv))
+ goto end;
+
+ g_logger.info("Starting...");
+ if(!setup_config(g_config))
+ goto end;
+
+ g_logger.info("Connecting to hosts");
+ if(!connect_hosts(g_config))
+ goto end;
+
+ if(!setup_hosts(g_config))
+ goto end;
+
+ if(!start_processes(g_config, atrt_process::NDB_MGM))
+ goto end;
+
+ if(!connect_ndb_mgm(g_config)){
+ goto end;
+ }
+
+ /**
+ * Main loop
+ */
+ while(!feof(g_test_case_file)){
+ /**
+ * Do we need to restart ndb
+ */
+ if(restart){
+ g_logger.info("(Re)starting ndb processes");
+ if(!stop_processes(g_config, atrt_process::NDB_DB))
+ goto end;
+
+ if(!wait_ndb(g_config, NDB_MGM_NODE_STATUS_NO_CONTACT))
+ goto end;
+
+ if(!start_processes(g_config, atrt_process::NDB_DB))
+ goto end;
+
+ if(!wait_ndb(g_config, NDB_MGM_NODE_STATUS_STARTED))
+ goto end;
+
+ g_logger.info("Ndb start completed");
+ }
+
+ const int start_line = lineno;
+ atrt_testcase test_case;
+ if(!read_test_case(g_test_case_file, test_case, lineno))
+ goto end;
+
+ g_logger.info("#%d - %s %s",
+ test_no,
+ test_case.m_command.c_str(), test_case.m_args.c_str());
+
+ // Assign processes to programs
+ if(!setup_test_case(g_config, test_case))
+ goto end;
+
+ if(!start_processes(g_config, p_servers))
+ goto end;
+
+ if(!start_processes(g_config, p_clients))
+ goto end;
+
+ int result = 0;
+
+ const time_t start = time(0);
+ time_t now = start;
+ do {
+ if(!update_status(g_config, atrt_process::ALL))
+ goto end;
+
+ int count = 0;
+
+ if((count = is_running(g_config, p_ndb)) != 2){
+ result = ERR_NDB_FAILED;
+ break;
+ }
+
+ if((count = is_running(g_config, p_servers)) != 2){
+ result = ERR_SERVERS_FAILED;
+ break;
+ }
+
+ if((count = is_running(g_config, p_clients)) == 0){
+ break;
+ }
+
+ now = time(0);
+ if(now > (start + test_case.m_max_time)){
+ result = ERR_MAX_TIME_ELAPSED;
+ break;
+ }
+ sleep(1);
+ } while(true);
+
+ const time_t elapsed = time(0) - start;
+
+ if(!stop_processes(g_config, p_clients))
+ goto end;
+
+ if(!stop_processes(g_config, p_servers))
+ goto end;
+
+ if(!gather_result(g_config, &result))
+ goto end;
+
+ g_logger.info("#%d %s(%d)",
+ test_no,
+ (result == 0 ? "OK" : "FAILED"), result);
+
+ if(g_report_file != 0){
+ fprintf(g_report_file, "%s %s ; %d ; %d ; %d\n",
+ test_case.m_command.c_str(),
+ test_case.m_args.c_str(),
+ test_no, result, elapsed);
+ fflush(g_report_file);
+ }
+
+ if(g_mode_bench || (g_mode_regression && result)){
+ BaseString tmp;
+ tmp.assfmt("result.%d", test_no);
+ if(rename("result", tmp.c_str()) != 0){
+ g_logger.critical("Failed to rename %s as %s",
+ "result", tmp.c_str());
+ goto end;
+ }
+ }
+
+ if(g_mode_interactive && result){
+ g_logger.info
+ ("Encountered failed test in interactive mode - terminating");
+ break;
+ }
+
+ if(result != 0){
+ restart = true;
+ } else {
+ restart = false;
+ }
+ test_no++;
+ }
+
+ end:
+ if(g_report_file != 0){
+ fclose(g_report_file);
+ g_report_file = 0;
+ }
+
+ if(g_test_case_file != 0 && g_test_case_file != stdin){
+ fclose(g_test_case_file);
+ g_test_case_file = 0;
+ }
+
+ stop_processes(g_config, atrt_process::ALL);
+ return 0;
+}
+
+bool
+parse_args(int argc, const char** argv){
+ int optind = 0;
+ if(getarg(args, arg_count, argc, argv, &optind)) {
+ arg_printusage(args, arg_count, progname, "");
+ return false;
+ }
+
+ if(g_log_filename != 0){
+ g_logger.removeConsoleHandler();
+ g_logger.addHandler(new FileLogHandler(g_log_filename));
+ }
+
+ {
+ int tmp = Logger::LL_WARNING - g_verbosity;
+ tmp = (tmp < Logger::LL_DEBUG ? Logger::LL_DEBUG : tmp);
+ g_logger.disable(Logger::LL_ALL);
+ g_logger.enable((Logger::LoggerLevel)tmp, Logger::LL_ALERT);
+ }
+
+
+
+ if(!g_process_config_filename){
+ g_logger.critical("Process config not specified!");
+ return false;
+ }
+
+ if(!g_setup_path){
+ char buf[1024];
+ if(getcwd(buf, sizeof(buf))){
+ g_setup_path = strdup(buf);
+ g_logger.info("Setup path not specified, using %s", buf);
+ } else {
+ g_logger.critical("Setup path not specified!\n");
+ return false;
+ }
+ }
+
+ if(g_report & !g_report_filename){
+ g_report_filename = "report.txt";
+ }
+
+ if(g_report_filename){
+ g_report_file = fopen(g_report_filename, "w");
+ if(g_report_file == 0){
+ g_logger.critical("Unable to create report file: %s", g_report_filename);
+ return false;
+ }
+ }
+
+ if(g_test_case_filename){
+ g_test_case_file = fopen(g_test_case_filename, "r");
+ if(g_test_case_file == 0){
+ g_logger.critical("Unable to open file: %s", g_test_case_filename);
+ return false;
+ }
+ }
+
+ int sum = g_mode_interactive + g_mode_regression + g_mode_bench;
+ if(sum == 0){
+ g_mode_interactive = 1;
+ }
+
+ if(sum > 1){
+ g_logger.critical
+ ("Only one of bench/regression/interactive can be specified");
+ return false;
+ }
+
+ g_default_user = strdup(getenv("USER"));
+
+ return true;
+}
+
+
+static
+atrt_host *
+find(const BaseString& host, Vector<atrt_host> & hosts){
+ for(size_t i = 0; i<hosts.size(); i++){
+ if(hosts[i].m_hostname == host){
+ return &hosts[i];
+ }
+ }
+ return 0;
+}
+
+bool
+setup_config(atrt_config& config){
+
+ FILE * f = fopen(g_process_config_filename, "r");
+ if(!f){
+ g_logger.critical("Failed to open process config file: %s",
+ g_process_config_filename);
+ return false;
+ }
+ bool result = true;
+
+ int lineno = 0;
+ char buf[2048];
+ while(fgets(buf, 2048, f)){
+ lineno++;
+
+ BaseString tmp(buf);
+ tmp.trim(" \t\n\r");
+
+ if(tmp.length() == 0 || tmp == "" || tmp.c_str()[0] == '#')
+ continue;
+
+ Vector<BaseString> split1;
+ if(tmp.split(split1, ":", 2) != 2){
+ g_logger.warning("Invalid line %d in %s - ignoring",
+ lineno, g_process_config_filename);
+ continue;
+ }
+
+ if(split1[0].trim() == "basedir"){
+ g_default_base_dir = strdup(split1[1].trim().c_str());
+ continue;
+ }
+
+ if(split1[0].trim() == "baseport"){
+ g_default_base_port = atoi(split1[1].trim().c_str());
+ continue;
+ }
+
+ if(split1[0].trim() == "user"){
+ g_default_user = strdup(split1[1].trim().c_str());
+ continue;
+ }
+
+ Vector<BaseString> hosts;
+ if(split1[1].trim().split(hosts) <= 0){
+ g_logger.warning("Invalid line %d in %s - ignoring",
+ lineno, g_process_config_filename);
+ }
+
+ // 1 - Check hosts
+ for(size_t i = 0; i<hosts.size(); i++){
+ Vector<BaseString> tmp;
+ hosts[i].split(tmp, ":");
+ BaseString hostname = tmp[0].trim();
+ BaseString base_dir;
+ if(tmp.size() >= 2)
+ base_dir = tmp[1];
+ else if(g_default_base_dir == 0){
+ g_logger.critical("Basedir not specified...");
+ return false;
+ }
+
+ atrt_host * host_ptr;
+ if((host_ptr = find(hostname, config.m_hosts)) == 0){
+ atrt_host host;
+ host.m_index = config.m_hosts.size();
+ host.m_cpcd = new SimpleCpcClient(hostname.c_str(), 1234);
+ host.m_base_dir = (base_dir.empty() ? g_default_base_dir : base_dir);
+ host.m_user = g_default_user;
+ host.m_hostname = hostname.c_str();
+ config.m_hosts.push_back(host);
+ } else {
+ if(!base_dir.empty() && (base_dir == host_ptr->m_base_dir)){
+ g_logger.critical("Inconsistent base dir definition for host %s"
+ ", \"%s\" != \"%s\"", hostname.c_str(),
+ base_dir.c_str(), host_ptr->m_base_dir.c_str());
+ return false;
+ }
+ }
+ }
+
+ for(size_t i = 0; i<hosts.size(); i++){
+ BaseString & tmp = hosts[i];
+ atrt_host * host = find(tmp, config.m_hosts);
+
+ const int index = config.m_processes.size() + 1;
+ atrt_process proc;
+ proc.m_index = index;
+ proc.m_host = host;
+ proc.m_proc.m_id = -1;
+ proc.m_proc.m_type = "temporary";
+ proc.m_proc.m_owner = "atrt";
+ proc.m_proc.m_group = "group";
+ proc.m_proc.m_cwd.assign(host->m_base_dir).append("/run/");
+ proc.m_proc.m_env.assign("LD_LIBRARY_PATH=").append(host->m_base_dir).append("/lib");
+ proc.m_proc.m_stdout = "log.out";
+ proc.m_proc.m_stderr = "2>&1";
+ proc.m_proc.m_runas = proc.m_host->m_user;
+ proc.m_proc.m_ulimit = "c:unlimited";
+ proc.m_hostname = proc.m_host->m_hostname;
+ proc.m_ndb_mgm_port = g_default_base_port;
+ if(split1[0] == "mgm"){
+ proc.m_type = atrt_process::NDB_MGM;
+ proc.m_proc.m_name.assfmt("%d-%s", index, "ndb_mgm");
+ proc.m_proc.m_path.assign(host->m_base_dir).append("/bin/mgmtsrvr");
+ proc.m_proc.m_args = "-n -c initconfig.txt";
+ proc.m_proc.m_cwd.appfmt("%d.ndb_mgm", index);
+ } else if(split1[0] == "ndb"){
+ proc.m_type = atrt_process::NDB_DB;
+ proc.m_proc.m_name.assfmt("%d-%s", index, "ndb_db");
+ proc.m_proc.m_path.assign(host->m_base_dir).append("/bin/ndb");
+ proc.m_proc.m_args = "-i -n";
+ proc.m_proc.m_cwd.appfmt("%d.ndb_db", index);
+ } else if(split1[0] == "api"){
+ proc.m_type = atrt_process::NDB_API;
+ proc.m_proc.m_name.assfmt("%d-%s", index, "ndb_api");
+ proc.m_proc.m_path = "";
+ proc.m_proc.m_args = "";
+ proc.m_proc.m_cwd.appfmt("%d.ndb_api", index);
+ } else {
+ g_logger.critical("%s:%d: Unhandled process type: %s",
+ g_process_config_filename, lineno,
+ split1[0].c_str());
+ result = false;
+ goto end;
+ }
+ config.m_processes.push_back(proc);
+ }
+ }
+
+ end:
+ fclose(f);
+ return result;
+}
+
+bool
+connect_hosts(atrt_config& config){
+ for(size_t i = 0; i<config.m_hosts.size(); i++){
+ if(config.m_hosts[i].m_cpcd->connect() != 0){
+ g_logger.error("Unable to connect to cpc %s:%d",
+ config.m_hosts[i].m_cpcd->getHost(),
+ config.m_hosts[i].m_cpcd->getPort());
+ return false;
+ }
+ g_logger.debug("Connected to %s:%d",
+ config.m_hosts[i].m_cpcd->getHost(),
+ config.m_hosts[i].m_cpcd->getPort());
+ }
+
+ return true;
+}
+
+bool
+connect_ndb_mgm(atrt_process & proc){
+ NdbMgmHandle handle = ndb_mgm_create_handle();
+ if(handle == 0){
+ g_logger.critical("Unable to create mgm handle");
+ return false;
+ }
+ BaseString tmp = proc.m_hostname;
+ tmp.appfmt(":%d", proc.m_ndb_mgm_port);
+ time_t start = time(0);
+ const time_t max_connect_time = 30;
+ do {
+ if(ndb_mgm_connect(handle, tmp.c_str()) != -1){
+ proc.m_ndb_mgm_handle = handle;
+ return true;
+ }
+ sleep(1);
+ } while(time(0) < (start + max_connect_time));
+ g_logger.critical("Unable to connect to ndb mgm %s", tmp.c_str());
+ return false;
+}
+
+bool
+connect_ndb_mgm(atrt_config& config){
+ for(size_t i = 0; i<config.m_processes.size(); i++){
+ atrt_process & proc = config.m_processes[i];
+ if((proc.m_type & atrt_process::NDB_MGM) != 0){
+ if(!connect_ndb_mgm(proc)){
+ return false;
+ }
+ }
+ }
+
+ return true;
+}
+
+static int remap(int i){
+ if(i == NDB_MGM_NODE_STATUS_NO_CONTACT) return NDB_MGM_NODE_STATUS_UNKNOWN;
+ if(i == NDB_MGM_NODE_STATUS_UNKNOWN) return NDB_MGM_NODE_STATUS_NO_CONTACT;
+ return i;
+}
+
+bool
+wait_ndb(atrt_config& config, int goal){
+
+ goal = remap(goal);
+
+
+ /**
+ * Get mgm handle for cluster
+ */
+ NdbMgmHandle handle = 0;
+ for(size_t i = 0; i<config.m_processes.size(); i++){
+ atrt_process & proc = config.m_processes[i];
+ if((proc.m_type & atrt_process::NDB_MGM) != 0){
+ handle = proc.m_ndb_mgm_handle;
+ break;
+ }
+ }
+ if(handle == 0){
+ g_logger.critical("Unable to find mgm handle");
+ return false;
+ }
+
+ if(goal == NDB_MGM_NODE_STATUS_STARTED){
+ /**
+ * 1) wait NOT_STARTED
+ * 2) send start
+ * 3) wait STARTED
+ */
+ if(!wait_ndb(config, NDB_MGM_NODE_STATUS_NOT_STARTED))
+ return false;
+
+ ndb_mgm_start(handle, 0, 0);
+ }
+
+ struct ndb_mgm_cluster_state * state;
+
+ time_t now = time(0);
+ time_t end = now + 360;
+ int min = remap(NDB_MGM_NODE_STATUS_NO_CONTACT);
+ int min2 = goal;
+
+ while(now < end){
+ /**
+ * 1) retreive current state
+ */
+ state = ndb_mgm_get_status(handle);
+ if(state == 0){
+ g_logger.critical("Unable to poll db state");
+ return false;
+ }
+ NdbAutoPtr<void> tmp(state);
+
+ min2 = goal;
+ for(int i = 0; i<state->no_of_nodes; i++){
+ if(state->node_states[i].node_type == NDB_MGM_NODE_TYPE_NDB){
+ const int s = remap(state->node_states[i].node_status);
+ min2 = (min2 < s ? min2 : s );
+
+ if(s < remap(NDB_MGM_NODE_STATUS_NO_CONTACT) ||
+ s > NDB_MGM_NODE_STATUS_STARTED){
+ g_logger.critical("Strange DB status during start: %d %d", i, min2);
+ return false;
+ }
+ }
+ }
+
+ if(min2 < min){
+ g_logger.critical("wait ndb failed %d %d %d", min, min2, goal);
+ return false;
+ }
+
+ if(min2 == goal){
+ return true;
+ break;
+ }
+
+ min = min2;
+ now = time(0);
+ }
+
+ g_logger.critical("wait ndb timed out %d %d %d", min, min2, goal);
+
+ return false;
+}
+
+bool
+start_process(atrt_process & proc){
+ if(proc.m_proc.m_id != -1){
+ g_logger.critical("starting already started process: %d", proc.m_index);
+ return false;
+ }
+
+ BaseString path = proc.m_proc.m_cwd.substr(proc.m_host->m_base_dir.length()+BaseString("/run").length());
+
+ BaseString tmp = g_setup_progname;
+ tmp.appfmt(" %s %s/%s/ %s",
+ proc.m_host->m_hostname.c_str(),
+ g_setup_path,
+ path.c_str(),
+ proc.m_proc.m_cwd.c_str());
+
+ const int r1 = system(tmp.c_str());
+ if(r1 != 0){
+ g_logger.critical("Failed to setup process");
+ return false;
+ }
+
+ {
+ Properties reply;
+ if(proc.m_host->m_cpcd->define_process(proc.m_proc, reply) != 0){
+ BaseString msg;
+ reply.get("errormessage", msg);
+ g_logger.error("Unable to define process: %s", msg.c_str());
+ return false;
+ }
+ }
+ {
+ Properties reply;
+ if(proc.m_host->m_cpcd->start_process(proc.m_proc.m_id, reply) != 0){
+ BaseString msg;
+ reply.get("errormessage", msg);
+ g_logger.error("Unable to start process: %s", msg.c_str());
+ return false;
+ }
+ }
+ return true;
+}
+
+bool
+start_processes(atrt_config& config, int types){
+ for(size_t i = 0; i<config.m_processes.size(); i++){
+ atrt_process & proc = config.m_processes[i];
+ if((types & proc.m_type) != 0){
+ if(!start_process(proc)){
+ return false;
+ }
+ }
+ }
+ return true;
+}
+
+bool
+stop_process(atrt_process & proc){
+ if(proc.m_proc.m_id == -1){
+ return true;
+ }
+
+ {
+ Properties reply;
+ if(proc.m_host->m_cpcd->stop_process(proc.m_proc.m_id, reply) != 0){
+ Uint32 status;
+ reply.get("status", &status);
+ if(status != 4){
+ BaseString msg;
+ reply.get("errormessage", msg);
+ g_logger.error("Unable to stop process: %s(%d)", msg.c_str(), status);
+ return false;
+ }
+ }
+ }
+ {
+ Properties reply;
+ if(proc.m_host->m_cpcd->undefine_process(proc.m_proc.m_id, reply) != 0){
+ BaseString msg;
+ reply.get("errormessage", msg);
+ g_logger.error("Unable to undefine process: %s", msg.c_str());
+ return false;
+ }
+ proc.m_proc.m_id = -1;
+ }
+ return true;
+}
+
+bool
+stop_processes(atrt_config& config, int types){
+ for(size_t i = 0; i<config.m_processes.size(); i++){
+ atrt_process & proc = config.m_processes[i];
+ if((types & proc.m_type) != 0){
+ if(!stop_process(proc)){
+ return false;
+ }
+ }
+ }
+ return true;
+}
+
+bool
+update_status(atrt_config& config, int){
+
+ Vector<Vector<SimpleCpcClient::Process> > m_procs;
+
+ Vector<SimpleCpcClient::Process> dummy;
+ m_procs.fill(config.m_hosts.size(), dummy);
+ for(size_t i = 0; i<config.m_hosts.size(); i++){
+ Properties p;
+ config.m_hosts[i].m_cpcd->list_processes(m_procs[i], p);
+ }
+
+ for(size_t i = 0; i<config.m_processes.size(); i++){
+ atrt_process & proc = config.m_processes[i];
+ Vector<SimpleCpcClient::Process> & h_procs = m_procs[proc.m_host->m_index];
+ bool found = false;
+ for(size_t j = 0; j<h_procs.size(); j++){
+ if(proc.m_proc.m_id == h_procs[j].m_id){
+ found = true;
+ proc.m_proc.m_status = h_procs[j].m_status;
+ break;
+ }
+ }
+ if(!found){
+ g_logger.error("update_status: not found");
+ return false;
+ }
+ }
+ return true;
+}
+
+int
+is_running(atrt_config& config, int types){
+ int found = 0, running = 0;
+ for(size_t i = 0; i<config.m_processes.size(); i++){
+ atrt_process & proc = config.m_processes[i];
+ if((types & proc.m_type) != 0){
+ found++;
+ if(proc.m_proc.m_status == "running")
+ running++;
+ }
+ }
+
+ if(found == running)
+ return 2;
+ if(running == 0)
+ return 0;
+ return 1;
+}
+
+
+int
+insert(const char * pair, Properties & p){
+ BaseString tmp(pair);
+
+ tmp.trim(" \t\n\r");
+
+ Vector<BaseString> split;
+ tmp.split(split, ":=", 2);
+
+ if(split.size() != 2)
+ return -1;
+
+ p.put(split[0].trim().c_str(), split[1].trim().c_str());
+
+ return 0;
+}
+
+bool
+read_test_case(FILE * file, atrt_testcase& tc, int& line){
+
+ Properties p;
+ int elements = 0;
+ char buf[1024];
+ while(!feof(file)){
+ if(!fgets(buf, 1024, file))
+ break;
+
+ line++;
+ BaseString tmp = buf;
+
+ if(tmp.length() > 0 && tmp.c_str()[0] == '#')
+ continue;
+
+ if(insert(tmp.c_str(), p) != 0)
+ break;
+
+ elements++;
+ }
+
+ if(elements == 0){
+ if(file == stdin){
+ BaseString tmp(buf);
+ tmp.trim(" \t\n\r");
+ Vector<BaseString> split;
+ tmp.split(split, " ", 2);
+ tc.m_command = split[0];
+ if(split.size() == 2)
+ tc.m_args = split[1];
+ else
+ tc.m_args = "";
+ tc.m_max_time = 60000;
+ return true;
+ }
+ return false;
+ }
+
+ if(!p.get("cmd", tc.m_command)){
+ g_logger.critical("Invalid test file: cmd is missing near line: %d", line);
+ return false;
+ }
+
+ if(!p.get("args", tc.m_args))
+ tc.m_args = "";
+
+ const char * mt = 0;
+ if(!p.get("max-time", &mt))
+ tc.m_max_time = 60000;
+ else
+ tc.m_max_time = atoi(mt);
+
+ return true;
+}
+
+bool
+setup_test_case(atrt_config& config, const atrt_testcase& tc){
+ const int r1 = system(g_clear_progname);
+ if(r1 != 0){
+ g_logger.critical("Failed to clear result");
+ return false;
+ }
+
+ for(size_t i = 0; i<config.m_processes.size(); i++){
+ atrt_process & proc = config.m_processes[i];
+ if(proc.m_type == atrt_process::NDB_API){
+ proc.m_proc.m_path.assign(proc.m_host->m_base_dir).append("/bin/").append(tc.m_command);
+ proc.m_proc.m_args.assign(tc.m_args);
+ return true;
+ }
+ }
+
+ return false;
+}
+
+bool
+gather_result(atrt_config& config, int * result){
+ BaseString tmp = g_gather_progname;
+ for(size_t i = 0; i<config.m_processes.size(); i++){
+ atrt_process & proc = config.m_processes[i];
+ tmp.appfmt(" %s:%s",
+ proc.m_hostname.c_str(),
+ proc.m_proc.m_cwd.c_str());
+ }
+
+ const int r1 = system(tmp.c_str());
+ if(r1 != 0){
+ g_logger.critical("Failed to gather result");
+ return false;
+ }
+
+ const int r2 = system(g_analyze_progname);
+
+ if(r2 == -1 || r2 == (127 << 8)){
+ g_logger.critical("Failed to analyze results");
+ return false;
+ }
+
+ * result = r2 ;
+ return true;
+}
+
+bool
+setup_hosts(atrt_config& config){
+ const int r1 = system(g_clear_progname);
+ if(r1 != 0){
+ g_logger.critical("Failed to clear result");
+ return false;
+ }
+
+ for(size_t i = 0; i<config.m_hosts.size(); i++){
+ BaseString tmp = g_setup_progname;
+ tmp.appfmt(" %s %s/ %s/run",
+ config.m_hosts[i].m_hostname.c_str(),
+ g_setup_path,
+ config.m_hosts[i].m_base_dir.c_str());
+
+ const int r1 = system(tmp.c_str());
+ if(r1 != 0){
+ g_logger.critical("Failed to setup %s",
+ config.m_hosts[i].m_hostname.c_str());
+ return false;
+ }
+ }
+ return true;
+}
diff --git a/ndb/test/run-test/make-config.sh b/ndb/test/run-test/make-config.sh
new file mode 100755
index 00000000000..5394b0654d4
--- /dev/null
+++ b/ndb/test/run-test/make-config.sh
@@ -0,0 +1,465 @@
+#!/bin/sh
+# NAME
+# make-config.sh - Makes a config file for mgm server
+#
+# SYNOPSIS
+# make-config.sh [ -t <template> ] [-s] [ -m <machine conf> [ -d <directory> ]
+#
+# DESCRIPTION
+#
+# OPTIONS
+#
+# EXAMPLES
+#
+#
+# ENVIRONMENT
+# NDB_PROJ_HOME Home dir for ndb
+#
+# FILES
+# $NDB_PROJ_HOME/lib/funcs.sh general shell script functions
+#
+#
+# SEE ALSO
+#
+# DIAGNOSTICTS
+#
+# VERSION
+# 1.0
+# 1.1 021112 epesson: Adapted for new mgmt server in NDB 2.00
+#
+# AUTHOR
+# Jonas Oreland
+#
+# CHANGES
+# also generate ndbnet config
+#
+
+progname=`basename $0`
+synopsis="make-config.sh [ -t template ] [ -m <machine conf> ] [ -d <dst directory> ][-s] [<mgm host>]"
+
+#: ${NDB_PROJ_HOME:?} # If undefined, exit with error message
+
+#: ${NDB_LOCAL_BUILD_OPTIONS:=--} # If undef, set to --. Keeps getopts happy.
+ # You may have to experiment a bit
+ # to get quoting right (if you need it).
+
+
+#. $NDB_PROJ_HOME/lib/funcs.sh # Load some good stuff
+trace() {
+ echo $* 1>&2
+}
+syndie() {
+ trace $*
+ exit 1
+}
+
+# defaults for options related variables
+#
+
+mgm_nodes=0
+ndb_nodes=0
+api_nodes=0
+uniq_id=$$.$$
+own_host=`hostname`
+dst_dir=""
+template=/dev/null
+machines=/dev/null
+verbose=yes
+
+# used if error when parsing the options environment variable
+#
+env_opterr="options environment variable: <<$options>>"
+
+# Option parsing, for the options variable as well as the command line.
+#
+# We want to be able to set options in an environment variable,
+# as well as on the command line. In order not to have to repeat
+# the same getopts information twice, we loop two times over the
+# getopts while loop. The first time, we process options from
+# the options environment variable, the second time we process
+# options from the command line.
+#
+# The things to change are the actual options and what they do.
+#
+add_node(){
+ no=$1; shift
+ type=$1; shift
+ echo $* | awk 'BEGIN{FS=":";}{h=$1; if(h=="localhost") h="'$own_host'";
+ printf("%s_%d_host=%s\n", "'$type'", "'$no'", h);
+ if(NF>1 && $2!="") printf("%s_%d_port=%d\n",
+ "'$type'", "'$no'", $2);
+ if(NF>2 && $3!="") printf("%s_%d_dir=%s\n",
+ "'$type'", "'$no'", $3);
+ }'
+}
+
+
+add_mgm_node(){
+ mgm_nodes=`cat /tmp/mgm_nodes.$uniq_id | grep "_host=" | wc -l`
+ mgm_nodes=`expr $mgm_nodes + 1`
+ while [ $# -gt 0 ]
+ do
+ add_node ${mgm_nodes} mgm_node $1 >> /tmp/mgm_nodes.$uniq_id
+ shift
+ mgm_nodes=`expr $mgm_nodes + 1`
+ done
+}
+
+add_ndb_node(){
+ ndb_nodes=`cat /tmp/ndb_nodes.$uniq_id | grep "_host=" | wc -l`
+ ndb_nodes=`expr $ndb_nodes + 1`
+ while [ $# -gt 0 ]
+ do
+ add_node ${ndb_nodes} ndb_node $1 >> /tmp/ndb_nodes.$uniq_id
+ shift
+ ndb_nodes=`expr $ndb_nodes + 1`
+ done
+}
+
+add_api_node(){
+ api_nodes=`cat /tmp/api_nodes.$uniq_id | grep "_host=" |wc -l`
+ api_nodes=`expr $api_nodes + 1`
+ while [ $# -gt 0 ]
+ do
+ add_node ${api_nodes} api_node $1 >> /tmp/api_nodes.$uniq_id
+ shift
+ api_nodes=`expr $api_nodes + 1`
+ done
+}
+
+rm -rf /tmp/mgm_nodes.$uniq_id ; touch /tmp/mgm_nodes.$uniq_id
+rm -rf /tmp/ndb_nodes.$uniq_id ; touch /tmp/ndb_nodes.$uniq_id
+rm -rf /tmp/api_nodes.$uniq_id ; touch /tmp/api_nodes.$uniq_id
+
+for optstring in "$options" "" # 1. options variable 2. cmd line
+do
+
+ while getopts d:m:t:n:o:a:b:p:s i $optstring # optstring empty => no arg => cmd line
+ do
+ case $i in
+
+ q) verbose="";; # echo important things
+ t) template=$OPTARG;; # Template
+ d) dst_dir=$OPTARG;; # Destination directory
+ m) machines=$OPTARG;; # Machine configuration
+ s) mgm_start=yes;; # Make mgm start script
+ \?) syndie $env_opterr;; # print synopsis and exit
+
+ esac
+ done
+
+ [ -n "$optstring" ] && OPTIND=1 # Reset for round 2, cmdline options
+
+ env_opterr= # Round 2 should not use the value
+
+done
+shift `expr $OPTIND - 1`
+
+if [ -z "$dst_dir" ]
+then
+ verbose=
+fi
+
+skip(){
+ no=$1; shift
+ shift $no
+ echo $*
+}
+
+# --- option parsing done ---
+grep "^ndb: " $machines | while read node
+do
+ node=`skip 1 $node`
+ add_ndb_node $node
+done
+
+grep "^api: " $machines | while read node
+do
+ node=`skip 1 $node`
+ add_api_node $node
+done
+
+grep "^mgm: " $machines | while read node
+do
+ node=`skip 1 $node`
+ add_mgm_node $node
+done
+
+tmp=`grep "^baseport: " $machines | tail -1 | cut -d ":" -f 2`
+if [ "$tmp" ]
+then
+ baseport=`echo $tmp`
+else
+ syndie "Unable to find baseport"
+fi
+
+trim(){
+ echo $*
+}
+tmp=`grep "^basedir: " $machines | tail -1 | cut -d ":" -f 2`
+if [ "$tmp" ]
+then
+ basedir=`trim $tmp`
+fi
+
+# -- Load enviroment --
+ndb_nodes=`cat /tmp/ndb_nodes.$uniq_id | grep "_host=" | wc -l`
+api_nodes=`cat /tmp/api_nodes.$uniq_id | grep "_host=" | wc -l`
+mgm_nodes=`cat /tmp/mgm_nodes.$uniq_id | grep "_host=" | wc -l`
+. /tmp/ndb_nodes.$uniq_id
+. /tmp/api_nodes.$uniq_id
+. /tmp/mgm_nodes.$uniq_id
+rm -f /tmp/ndb_nodes.$uniq_id /tmp/api_nodes.$uniq_id /tmp/mgm_nodes.$uniq_id
+
+# -- Verify
+trace "Verifying arguments"
+
+if [ ! -r $template ]
+then
+ syndie "Unable to read template file: $template"
+fi
+
+if [ $ndb_nodes -le 0 ]
+then
+ syndie "No ndb nodes specified"
+fi
+
+if [ $api_nodes -le 0 ]
+then
+ syndie "No api nodes specified"
+fi
+
+if [ $mgm_nodes -gt 1 ]
+then
+ syndie "More than one mgm node specified"
+fi
+
+if [ $mgm_nodes -eq 0 ]
+then
+ trace "No managment server specified using `hostname`"
+ mgm_nodes=1
+ mgm_node_1=`hostname`
+fi
+
+if [ -n "$dst_dir" ]
+then
+ mkdir -p $dst_dir
+ if [ ! -d $dst_dir ]
+ then
+ syndie "Unable to create dst dir: $dst_dir"
+ fi
+ DST=/tmp/$uniq_id
+fi
+
+# --- option verifying done ---
+
+# Find uniq computers
+i=1
+while [ $i -le $mgm_nodes ]
+do
+ echo `eval echo "\$"mgm_node_${i}_host` >> /tmp/hosts.$uniq_id
+ i=`expr $i + 1`
+done
+
+i=1
+while [ $i -le $ndb_nodes ]
+do
+ echo `eval echo "\$"ndb_node_${i}_host` >> /tmp/hosts.$uniq_id
+ i=`expr $i + 1`
+done
+
+i=1
+while [ $i -le $api_nodes ]
+do
+ echo `eval echo "\$"api_node_${i}_host` >> /tmp/hosts.$uniq_id
+ i=`expr $i + 1`
+done
+
+sort -u -o /tmp/hosts.$uniq_id /tmp/hosts.$uniq_id
+
+get_computer_id(){
+ grep -w -n $1 /tmp/hosts.$uniq_id | cut -d ":" -f 1
+}
+
+get_mgm_computer_id(){
+ a=`eval echo "\$"mgm_node_${1}_host`
+ get_computer_id $a
+}
+
+get_ndb_computer_id(){
+ a=`eval echo "\$"ndb_node_${1}_host`
+ get_computer_id $a
+}
+
+get_api_computer_id(){
+ a=`eval echo "\$"api_node_${1}_host`
+ get_computer_id $a
+}
+
+# -- Write config files --
+
+mgm_port=$baseport
+
+(
+ i=1
+ #echo "COMPUTERS"
+ cat /tmp/hosts.$uniq_id | while read host
+ do
+ echo "[COMPUTER]"
+ echo "Id: $i"
+ echo "ByteOrder: Big"
+ echo "HostName: $host"
+ echo
+ i=`expr $i + 1`
+ done
+
+ node_id=1
+ echo
+
+ # Mgm process
+ echo
+ echo "[MGM]"
+ echo "Id: $node_id"
+ echo "ExecuteOnComputer: `get_mgm_computer_id 1`"
+ echo "PortNumber: $mgm_port"
+ node_id=`expr $node_id + 1`
+
+ # Ndb processes
+ i=1
+ ndb_nodes=`trim $ndb_nodes`
+ while [ $i -le $ndb_nodes ]
+ do
+ echo
+ echo "[DB]"
+ echo "Id: $node_id"
+ echo "ExecuteOnComputer: `get_ndb_computer_id $i`"
+ echo "FileSystemPath: $basedir/run/node-${node_id}-fs"
+ i=`expr $i + 1`
+ node_id=`expr $node_id + 1`
+ done
+
+ # API processes
+ i=1
+ while [ $i -le $api_nodes ]
+ do
+ echo
+ echo "[API]"
+ echo "Id: $node_id"
+ echo "ExecuteOnComputer: `get_api_computer_id $i`"
+ i=`expr $i + 1`
+ node_id=`expr $node_id + 1`
+ done
+
+ # Connections
+ current_port=`expr $mgm_port + 1`
+ echo
+
+ # Connect Mgm with all ndb-nodes
+ i=1
+ while [ $i -le $ndb_nodes ]
+ do
+ echo
+ echo "[TCP]"
+ echo "NodeId1: 1"
+ echo "NodeId2: `expr $i + 1`"
+ echo "PortNumber: $current_port"
+ i=`expr $i + 1`
+ current_port=`expr $current_port + 1`
+ done
+
+ # Connect All ndb processes with all ndb processes
+ i=1
+ while [ $i -le $ndb_nodes ]
+ do
+ j=`expr $i + 1`
+ while [ $j -le $ndb_nodes ]
+ do
+ echo
+ echo "[TCP]"
+ echo "NodeId1: `expr $i + 1`"
+ echo "NodeId2: `expr $j + 1`"
+ echo "PortNumber: $current_port"
+ j=`expr $j + 1`
+ current_port=`expr $current_port + 1`
+ done
+ i=`expr $i + 1`
+ done
+
+ # Connect all ndb-nodes with all api nodes
+ i=1
+ while [ $i -le $ndb_nodes ]
+ do
+ j=1
+ while [ $j -le $api_nodes ]
+ do
+ echo
+ echo "[TCP]"
+ echo "NodeId1: `expr $i + 1`"
+ echo "NodeId2: `expr $j + $ndb_nodes + 1`"
+ echo "PortNumber: $current_port"
+ j=`expr $j + 1`
+ current_port=`expr $current_port + 1`
+ done
+ i=`expr $i + 1`
+ done
+ echo
+) > $DST
+
+trace "Init config file done"
+
+if [ -z "$dst_dir" ]
+then
+ cat $DST
+ rm -f $DST
+ rm -f /tmp/hosts.$uniq_id
+ exit 0
+fi
+
+###
+# Create Ndb.cfg files
+
+# nodeid=2;host=localhost:2200
+
+# Mgm node
+mkcfg(){
+ mkdir -p $dst_dir/${2}.ndb_${1}
+ (
+ echo "OwnProcessId $2"
+ echo "host://${mgm_node_1_host}:${mgm_port}"
+ ) > $dst_dir/${2}.ndb_${1}/Ndb.cfg
+ if [ $1 = "db" ]
+ then
+ mkdir $dst_dir/node-${2}-fs
+ fi
+}
+
+mkcfg mgm 1
+cat $DST > $dst_dir/1.ndb_mgm/initconfig.txt
+
+trace "Creating Ndb.cfg for ndb nodes"
+
+current_node=2
+i=1
+while [ $i -le $ndb_nodes ]
+do
+ mkcfg db ${current_node}
+ i=`expr $i + 1`
+ current_node=`expr $current_node + 1`
+done
+
+trace "Creating Ndb.cfg for api nodes"
+
+i=1
+while [ $i -le $api_nodes ]
+do
+ mkcfg api ${current_node}
+ i=`expr $i + 1`
+ current_node=`expr $current_node + 1`
+done
+
+rm -f $DST
+rm -f /tmp/hosts.$uniq_id
+
+
+exit 0
+# vim: set sw=4:
diff --git a/ndb/test/run-test/make-html-reports.sh b/ndb/test/run-test/make-html-reports.sh
new file mode 100755
index 00000000000..079650a729f
--- /dev/null
+++ b/ndb/test/run-test/make-html-reports.sh
@@ -0,0 +1,437 @@
+#!/bin/sh
+# NAME
+# make-html-reports.sh
+#
+# SYNOPSIS
+# make-html-reports.sh [-q] [ -R <YYYY-MM-DD> ] [ -s <src dir> ] [ -d <dst dir> ] [ -c <conf dir> ]
+#
+# DESCRIPTION
+#
+# OPTIONS
+#
+# EXAMPLES
+#
+#
+# ENVIRONMENT
+# NDB_PROJ_HOME Home dir for ndb
+#
+# FILES
+# $NDB_PROJ_HOME/lib/funcs.sh general shell script functions
+#
+#
+# SEE ALSO
+#
+# DIAGNOSTICTS
+#
+# VERSION
+# 1.0
+#
+# AUTHOR
+# Jonas Oreland
+#
+
+progname=`basename $0`
+synopsis="make-html-reports.sh [ -R <YYYY-MM-DD> ] [ -s <src dir> ] [ -d <dst dir> ] [ -c <conf dir> ]"
+
+: ${NDB_PROJ_HOME:?} # If undefined, exit with error message
+
+: ${NDB_LOCAL_BUILD_OPTIONS:=--} # If undef, set to --. Keeps getopts happy.
+ # You may have to experiment a bit
+ # to get quoting right (if you need it).
+
+
+. $NDB_PROJ_HOME/lib/funcs.sh # Load some good stuff
+
+# defaults for options related variables
+#
+
+
+src_dir=`pwd`
+dst_dir=`pwd`
+conf_dir=`pwd`
+report_date=`date '+%Y-%m-%d'`
+uniq_id=$$.$$
+verbose=yes
+
+# used if error when parsing the options environment variable
+#
+env_opterr="options environment variable: <<$options>>"
+
+# Option parsing, for the options variable as well as the command line.
+#
+# We want to be able to set options in an environment variable,
+# as well as on the command line. In order not to have to repeat
+# the same getopts information twice, we loop two times over the
+# getopts while loop. The first time, we process options from
+# the options environment variable, the second time we process
+# options from the command line.
+#
+# The things to change are the actual options and what they do.
+#
+#
+
+for optstring in "$options" "" # 1. options variable 2. cmd line
+do
+
+ while getopts q:s:R:d:c: i $optstring # optstring empty => no arg => cmd line
+ do
+ case $i in
+
+ q) verbose="";; # echo important things
+ d) dst_dir=$OPTARG;; # Destination directory
+ s) src_dir=$OPTARG;; # Destination directory
+ c) conf_dir=$OPTARG;; #
+ R) report_date=$OPTARG;; #
+ \?) syndie $env_opterr;; # print synopsis and exit
+
+ esac
+ done
+
+ [ -n "$optstring" ] && OPTIND=1 # Reset for round 2, cmdline options
+
+ env_opterr= # Round 2 should not use the value
+
+done
+shift `expr $OPTIND - 1`
+
+src_dir=`abspath $src_dir`
+dst_dir=`abspath $dst_dir`
+conf_dir=`abspath $conf_dir`
+
+###
+#
+# General html functions
+header(){
+ cat <<EOF
+<html><head><title>$*</title></head>
+<body>
+EOF
+}
+
+footer(){
+ cat <<EOF
+</body></html>
+EOF
+}
+
+heading(){
+ h=$1; shift
+ cat <<EOF
+<h$h>$*</h$h>
+EOF
+}
+
+table(){
+ echo "<table $*>"
+}
+
+end_table(){
+ echo "</table>"
+}
+
+row(){
+ echo "<tr>"
+}
+
+end_row(){
+ echo "</tr>"
+}
+
+c_column(){
+ cat <<EOF
+<td valign=center align=center>$*</td>
+EOF
+}
+
+bold(){
+ cat <<EOF
+<b>$*</b>
+EOF
+}
+column(){
+ cat <<EOF
+<td valign=center align=left>$*</td>
+EOF
+}
+
+para(){
+ cat <<EOF
+<p></p>
+EOF
+}
+
+hr(){
+ cat <<EOF
+<hr>
+EOF
+}
+
+# --- option parsing done ---
+
+# -- Verify
+trace "Verifying arguments"
+summary_file=$src_dir/reports/summary.$report_date
+
+if [ ! -r $summary_file ]
+then
+ syndie "Invalid src directory or report date: $summary_file not found"
+fi
+
+if [ ! -d $conf_dir/configurations ]
+then
+ syndie "Invalid src directory: $conf_dir/configurations not found"
+fi
+
+if [ ! -d $conf_dir/testcases ]
+then
+ syndie "Invalid src directory: $conf_dir/testcases not found"
+fi
+
+if [ ! -d $dst_dir ]
+then
+ syndie "Invalid dst dir..."
+fi
+
+# --- option verifying done ---
+
+trace "src_dir: $src_dir"
+trace "dst_dir: $dst_dir"
+trace "conf_dir: $conf_dir"
+trace "report date: $report_date"
+
+###
+config_spec(){
+ cat <<EOF
+<a href=#$1>$1</a>
+EOF
+}
+
+config_spec_include(){
+ # Print the $1 file to the file we are generating
+ cat <<EOF
+<a name=$1><pre>
+EOF
+ if [ -r $conf_dir/configurations/$1 ]
+ then
+ cat -E $conf_dir/configurations/$1 | sed 's/\$/<BR>/g'
+ else
+ cat <<EOF
+ Config spec $1 not found
+EOF
+ fi
+cat <<EOF
+</pre></a>
+EOF
+}
+
+time_spec(){
+ # $1 - secs
+ _ts_tmp=$1
+
+ _ts_s=`expr $_ts_tmp % 60`
+ _ts_tmp=`expr $_ts_tmp / 60`
+
+ _ts_m=`expr $_ts_tmp % 60`
+ _ts_tmp=`expr $_ts_tmp / 60`
+
+ _ts_h=$_ts_tmp
+
+ if [ $_ts_h -gt 0 ]
+ then
+ ret="${_ts_h}h"
+ fi
+
+ [ $_ts_m -gt 0 ] || [ $_ts_h -gt 0 ] && ret="$ret${_ts_m}m"
+
+ ret="$ret${_ts_s}s"
+ echo $ret
+}
+
+log_spec(){
+ _ff_=$src_dir/log/$report_date/$1.$2/test.$3.out
+ if [ -r $_ff_ ] && [ -s $_ff_ ]
+ then
+ _f2_=$dst_dir/log.$report_date.$1.$2.$3.out.gz
+ if [ -r $_f2_ ]
+ then
+ rm $_f2_
+ fi
+ cp $_ff_ $dst_dir/log.$report_date.$1.$2.$3.out
+ gzip $dst_dir/log.$report_date.$1.$2.$3.out
+ rm -f $dst_dir/log.$report_date.$1.$2.$3.out
+ echo "<a href=log.$report_date.$1.$2.$3.out.gz>Log file</a>"
+ else
+ echo "-"
+ fi
+}
+
+err_spec(){
+ _ff_=$src_dir/log/$report_date/$1.$2/test.$3.err.tar
+ if [ -r $_ff_ ] && [ -s $_ff_ ]
+ then
+ cp $_ff_ $dst_dir/err.$report_date.$1.$2.$3.err.tar
+ gzip $dst_dir/err.$report_date.$1.$2.$3.err.tar
+ rm -f $dst_dir/err.$report_date.$1.$2.$3.err.tar
+ echo "<a href=err.$report_date.$1.$2.$3.err.tar.gz>Error tarball</a>"
+ else
+ echo "-"
+ fi
+}
+
+command_spec(){
+ echo $* | sed 's/;/<BR>/g'
+}
+
+### Main
+
+html_summary_file=$dst_dir/summary.$report_date.html
+
+trace "Creating summary"
+(
+ eval `grep "TOTAL" $summary_file | awk -F";" '{ printf("test_file=\"%s\"; elapsed=\"%s\"; started=\"%s\"; stopped=\"%s\"", $2, $3, $4, $5); }'`
+
+ header "Autotest summary $report_date"
+ heading 1 "Autotest summary $report_date"
+ table
+ row ; column `bold test file: `; column $test_file ; end_row
+ row ; column `bold Started:` ; column "$started "; end_row
+ row ; column `bold Stopped:` ; column "$stopped "; end_row
+ row ; column `bold Elapsed:` ; column "`time_spec $elapsed secs`" ; end_row
+ end_table
+ hr
+
+ table "border=1"
+ row
+ c_column `bold Report`
+ c_column `bold Tag`
+ c_column `bold Version`
+ c_column `bold Distr-Config`
+ c_column `bold Db-Config`
+ c_column `bold Type`
+ c_column `bold Test file`
+ c_column `bold Make`
+ c_column `bold Config`
+ c_column `bold Test time`
+ c_column `bold Passed`
+ c_column `bold Failed`
+ end_row
+
+ grep -v "^#" $summary_file | grep -v TOTAL | sed 's/;/ /g' | \
+ while read tag version config template type test_file make_res make_time conf_res conf_time test_time passed failed
+ do
+ row
+ if [ -r $src_dir/reports/report.$tag.$version.$config.$template.$type.$test_file.$report_date ]
+ then
+ column "<a href=\"report.$tag.$version.$config.$template.$type.$test_file.$report_date.html\">report</a>"
+ else
+ column "-"
+ fi
+
+ column $tag
+ column $version
+ column $config
+ column $template
+ column $type
+ column $test_file
+ column "$make_res(`time_spec $make_time`)"
+ column "$conf_res(`time_spec $conf_time`)"
+ c_column "`time_spec $test_time`"
+ c_column `bold $passed`
+ c_column `bold $failed`
+ end_row
+ done
+ end_table
+
+ footer
+) > $html_summary_file
+
+for i in $src_dir/reports/report.*.$report_date
+do
+ f=`basename $i`
+ trace "Creating report: $f"
+ eval `echo $f | awk -F"." '{printf("tag=%s;version=%s;config=%s;template=%s;type=%s;test_file=%s", $2, $3, $4, $5, $6, $7);}'`
+
+ (
+ header "Autotest report $report_date"
+ heading 1 "Autotest report $report_date"
+ table #"border=1"
+ row ; column `bold Tag:`; column $tag ; end_row
+ row ; column `bold Version:` ; column $version ; end_row
+ row ; column `bold Configuration:` ; column `config_spec $config`; end_row
+ row ; column `bold Template:` ; column `config_spec $template`; end_row
+ row ; column `bold Type:` ; column $type ; end_row
+ row ; column `bold Test file:` ; column $test_file; end_row
+ end_table
+ hr
+
+ table "border=1"
+ row
+ c_column `bold Test case`
+ c_column `bold Result`
+ c_column `bold Test time`
+ c_column `bold Logfile`
+ c_column `bold Error tarfile`
+ end_row
+
+ grep -v "^#" $i | sed 's/;/ /g' | \
+ while read test_no test_res test_time cmd
+ do
+ row
+ column "`command_spec $cmd`"
+ case "$test_res" in
+ 0)
+ column "PASSED";;
+ 1001)
+ column "API error";;
+ 1002)
+ column "Max time expired";;
+ 1003)
+ column "Mgm port busy";;
+ *)
+ column "Unknown: $test_res";;
+ esac
+
+ column "`time_spec $test_time`"
+
+ column "`log_spec $tag $version $test_no`"
+ column "`err_spec $tag $version $test_no`"
+ end_row
+ done
+ end_table
+
+ # Last on page we include spec
+ # of used machines and template for config
+ # for future reference
+ hr
+ table "border=1"
+ row; column `bold Configuration:` $config; end_row
+ row; column `config_spec_include $config`; end_row
+ end_table
+ hr
+ table "border=1"
+ row; column `bold Template:` $template; end_row
+ row; column `config_spec_include $template`; end_row
+ end_table
+
+ footer
+
+ ) > $dst_dir/$f.html
+done
+
+# Re creating index
+trace "Recreating index"
+(
+ header "Autotest super-duper index"
+ heading 1 "<center>Autotest super-duper index</center>"
+ hr
+ for i in `ls $dst_dir/summary.*.html | sort -r -n`
+ do
+ f=`basename $i`
+ cat <<EOF
+<p><a href=$f>$f</a></p>
+EOF
+ done
+ footer
+) > $dst_dir/index.html
+
+exit 0
diff --git a/ndb/test/run-test/make-index.sh b/ndb/test/run-test/make-index.sh
new file mode 100755
index 00000000000..944f0df790b
--- /dev/null
+++ b/ndb/test/run-test/make-index.sh
@@ -0,0 +1,242 @@
+#!/bin/sh
+# NAME
+# make-index.sh
+#
+# SYNOPSIS
+# make-index.sh [ -d <dir> ]
+#
+# DESCRIPTION
+#
+# OPTIONS
+#
+# EXAMPLES
+#
+#
+# ENVIRONMENT
+# NDB_PROJ_HOME Home dir for ndb
+#
+# FILES
+# $NDB_PROJ_HOME/lib/funcs.sh general shell script functions
+#
+#
+# SEE ALSO
+#
+# DIAGNOSTICTS
+#
+# VERSION
+# 1.0
+#
+# AUTHOR
+# Jonas Oreland
+#
+
+progname=`basename $0`
+synopsis="make-index.sh [ -d <dir> ]"
+
+: ${NDB_PROJ_HOME:?} # If undefined, exit with error message
+
+: ${NDB_LOCAL_BUILD_OPTIONS:=--} # If undef, set to --. Keeps getopts happy.
+ # You may have to experiment a bit
+ # to get quoting right (if you need it).
+
+
+. $NDB_PROJ_HOME/lib/funcs.sh # Load some good stuff
+
+# defaults for options related variables
+#
+
+dst_dir=/home/autotest/html
+report_date=`date '+%Y-%m-%d'`
+uniq_id=$$.$$
+verbose=yes
+
+# used if error when parsing the options environment variable
+#
+env_opterr="options environment variable: <<$options>>"
+
+# Option parsing, for the options variable as well as the command line.
+#
+# We want to be able to set options in an environment variable,
+# as well as on the command line. In order not to have to repeat
+# the same getopts information twice, we loop two times over the
+# getopts while loop. The first time, we process options from
+# the options environment variable, the second time we process
+# options from the command line.
+#
+# The things to change are the actual options and what they do.
+#
+#
+
+for optstring in "$options" "" # 1. options variable 2. cmd line
+do
+
+ while getopts q:s:R:d: i $optstring # optstring empty => no arg => cmd line
+ do
+ case $i in
+
+ q) verbose="";; # echo important things
+ d) dst_dir=$OPTARG;; # Destination directory
+ \?) syndie $env_opterr;; # print synopsis and exit
+
+ esac
+ done
+
+ [ -n "$optstring" ] && OPTIND=1 # Reset for round 2, cmdline options
+
+ env_opterr= # Round 2 should not use the value
+
+done
+shift `expr $OPTIND - 1`
+
+dst_dir=`abspath $dst_dir`
+
+###
+#
+# General html functions
+header(){
+ cat <<EOF
+<html><head><title>$*</title></head>
+<body>
+EOF
+}
+
+footer(){
+ cat <<EOF
+</body></html>
+EOF
+}
+
+heading(){
+ h=$1; shift
+ cat <<EOF
+<h$h>$*</h$h>
+EOF
+}
+
+table(){
+ echo "<table $*>"
+}
+
+end_table(){
+ echo "</table>"
+}
+
+row(){
+ echo "<tr>"
+}
+
+end_row(){
+ echo "</tr>"
+}
+
+c_column(){
+ cat <<EOF
+<td valign=center align=center>$*</td>
+EOF
+}
+
+bold(){
+ cat <<EOF
+<b>$*</b>
+EOF
+}
+column(){
+ cat <<EOF
+<td valign=center align=left>$*</td>
+EOF
+}
+
+para(){
+ cat <<EOF
+<p></p>
+EOF
+}
+
+hr(){
+ cat <<EOF
+<hr>
+EOF
+}
+
+inc_summary() {
+ grep -v 'html>' $2 | grep -v body | sed 's/href="/href="'$1'\//g'
+}
+
+# --- option parsing done ---
+
+
+
+# -- Verify
+trace "Verifying arguments"
+
+# --- option verifying done ---
+
+### Main
+
+# Re creating index
+trace "Creating index"
+(
+ header "Autotest super-duper index"
+ heading 1 "<center>Autotest super-duper index</center>"
+ cat -E README.autotest | sed 's/\$/<BR>/g'
+ echo "<br>"
+ echo "Current <a href="crontab.current">crontab</a> installed on mc01 running [" `uname -a` "]"
+ hr
+
+ dirs=`find $dst_dir -name 'summary.*.html' -type f -maxdepth 2 -exec dirname {} \; | sort -u`
+
+ dates=`find $dst_dir -name 'summary.*.html' -type f -maxdepth 2 -exec basename {} \; | sed 's/summary\.\(.*\)\.html/\1/g' | sort -u | sort -r`
+
+ echo "<p align=center>"
+
+#inline 5 latest reports
+ r_count=5
+ for d in $dates
+ do
+ for o in $dirs
+ do
+ o=`basename $o`
+ if [ -r $dst_dir/$o/summary.$d.html ]
+ then
+ inc_summary $o $dst_dir/$o/summary.$d.html
+ hr
+
+ r_count=`expr $r_count - 1`
+ if [ $r_count -eq 0 ]
+ then
+ break 2
+ fi
+ fi
+ done
+ done
+
+ table "border=1"
+ row
+ for i in $dirs
+ do
+ i=`basename $i`
+ column `bold $i`
+ done
+ end_row
+
+
+ for d in $dates
+ do
+ row
+ for o in $dirs
+ do
+ o=`basename $o`
+ if [ -r $dst_dir/$o/summary.$d.html ]
+ then
+ column "<a href=$o/summary.$d.html>$d</a>"
+ else
+ column ""
+ fi
+ done
+ end_row
+ done
+ end_table
+ footer
+) > $dst_dir/index.html
+
+exit 0
diff --git a/ndb/test/run-test/run-test.hpp b/ndb/test/run-test/run-test.hpp
new file mode 100644
index 00000000000..8387f8638ed
--- /dev/null
+++ b/ndb/test/run-test/run-test.hpp
@@ -0,0 +1,92 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#ifndef atrt_config_hpp
+#define atrt_config_hpp
+
+#include <getarg.h>
+#include <Vector.hpp>
+#include <BaseString.hpp>
+#include <Logger.hpp>
+#include <mgmapi.h>
+#include <CpcClient.hpp>
+
+enum ErrorCodes {
+ ERR_OK = 0,
+ ERR_NDB_FAILED = 101,
+ ERR_SERVERS_FAILED = 102,
+ ERR_MAX_TIME_ELAPSED = 103
+};
+
+struct atrt_host {
+ size_t m_index;
+ BaseString m_user;
+ BaseString m_base_dir;
+ BaseString m_hostname;
+ SimpleCpcClient * m_cpcd;
+};
+
+struct atrt_process {
+ size_t m_index;
+ BaseString m_hostname;
+ struct atrt_host * m_host;
+
+ enum Type {
+ ALL = 255,
+ NDB_DB = 1,
+ NDB_API = 2,
+ NDB_MGM = 4,
+ NDB_REP = 8,
+ MYSQL_SERVER = 16,
+ MYSQL_CLIENT = 32
+ } m_type;
+
+ SimpleCpcClient::Process m_proc;
+ short m_ndb_mgm_port;
+ NdbMgmHandle m_ndb_mgm_handle; // if type == ndb_mgm
+};
+
+struct atrt_config {
+ BaseString m_key;
+ Vector<atrt_host> m_hosts;
+ Vector<atrt_process> m_processes;
+};
+
+struct atrt_testcase {
+ time_t m_max_time;
+ BaseString m_command;
+ BaseString m_args;
+};
+
+extern Logger g_logger;
+
+bool parse_args(int argc, const char** argv);
+bool setup_config(atrt_config&);
+bool connect_hosts(atrt_config&);
+bool connect_ndb_mgm(atrt_config&);
+bool wait_ndb(atrt_config&, int ndb_mgm_node_status);
+bool start_processes(atrt_config&, int);
+bool stop_processes(atrt_config&, int);
+bool update_status(atrt_config&, int);
+int is_running(atrt_config&, int);
+bool gather_result(atrt_config&, int * result);
+
+bool read_test_case(FILE *, atrt_testcase&, int& line);
+bool setup_test_case(atrt_config&, const atrt_testcase&);
+
+bool setup_hosts(atrt_config&);
+
+#endif
diff --git a/ndb/test/src/HugoAsynchTransactions.cpp b/ndb/test/src/HugoAsynchTransactions.cpp
new file mode 100644
index 00000000000..d045032d455
--- /dev/null
+++ b/ndb/test/src/HugoAsynchTransactions.cpp
@@ -0,0 +1,491 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbSleep.h>
+#include <HugoAsynchTransactions.hpp>
+
+HugoAsynchTransactions::HugoAsynchTransactions(const NdbDictionary::Table& _tab):
+ HugoTransactions(_tab),
+ transactionsCompleted(0),
+ numTransactions(0),
+ transactions(NULL){
+}
+
+HugoAsynchTransactions::~HugoAsynchTransactions(){
+ deallocTransactions();
+}
+
+void asynchCallback(int result, NdbConnection* pTrans,
+ void* anObject) {
+ HugoAsynchTransactions* pHugo = (HugoAsynchTransactions*) anObject;
+
+ pHugo->transactionCompleted();
+
+ if (result == -1) {
+ const NdbError err = pTrans->getNdbError();
+ switch(err.status) {
+ case NdbError::Success:
+ ERR(err);
+ g_info << "ERROR: NdbError reports success when transcaction failed"
+ << endl;
+ break;
+
+ case NdbError::TemporaryError:
+ ERR(err);
+ break;
+
+#if 0
+ case 626: // Tuple did not exist
+ g_info << (unsigned int)pHugo->getTransactionsCompleted() << ": "
+ << err.code << " " << err.message << endl;
+ break;
+#endif
+
+ case NdbError::UnknownResult:
+ ERR(err);
+ break;
+
+ case NdbError::PermanentError:
+ switch (err.classification) {
+ case NdbError::ConstraintViolation:
+ // Tuple already existed, OK in this application,
+ // but should be reported
+ g_info << (unsigned int)pHugo->getTransactionsCompleted()
+ << ": " << err.code << " " << err.message << endl;
+ break;
+ default:
+ ERR(err);
+ break;
+ }
+ break;
+ }
+ } else {// if (result == -1)
+ /*
+ ndbout << (unsigned int)pHugo->getTransactionsCompleted() << " completed"
+ << endl;
+ */
+ }
+}
+
+int
+HugoAsynchTransactions::loadTableAsynch(Ndb* pNdb,
+ int records,
+ int batch,
+ int trans,
+ int operations){
+
+ int result = executeAsynchOperation(pNdb, records, batch, trans, operations,
+ NO_INSERT);
+ g_info << (unsigned int)transactionsCompleted * operations
+ << "|- inserted..." << endl;
+
+ return result;
+}
+
+void
+HugoAsynchTransactions::transactionCompleted() {
+ transactionsCompleted++;
+}
+
+long
+HugoAsynchTransactions::getTransactionsCompleted() {
+ return transactionsCompleted;
+}
+
+int
+HugoAsynchTransactions::pkDelRecordsAsynch(Ndb* pNdb,
+ int records,
+ int batch,
+ int trans,
+ int operations) {
+
+ g_info << "|- Deleting records asynchronous..." << endl;
+
+ int result = executeAsynchOperation(pNdb, records, batch, trans,
+ operations,
+ NO_DELETE);
+ g_info << "|- " << (unsigned int)transactionsCompleted * operations
+ << " deleted..." << endl;
+
+ return result;
+}
+
+int
+HugoAsynchTransactions::pkReadRecordsAsynch(Ndb* pNdb,
+ int records,
+ int batch,
+ int trans,
+ int operations) {
+
+ g_info << "|- Reading records asynchronous..." << endl;
+
+ allocRows(trans*operations);
+ int result = executeAsynchOperation(pNdb, records, batch, trans, operations,
+ NO_READ);
+
+ g_info << "|- " << (unsigned int)transactionsCompleted * operations
+ << " read..."
+ << endl;
+
+ deallocRows();
+
+ return result;
+}
+
+int
+HugoAsynchTransactions::pkUpdateRecordsAsynch(Ndb* pNdb,
+ int records,
+ int batch,
+ int trans,
+ int operations) {
+
+ g_info << "|- Updating records asynchronous..." << endl;
+
+ int check = 0;
+ int cTrans = 0;
+ int cReadRecords = 0;
+ int cReadIndex = 0;
+ int cRecords = 0;
+ int cIndex = 0;
+
+ transactionsCompleted = 0;
+
+ allocRows(trans*operations);
+ allocTransactions(trans);
+
+ for (int i = 0; i < batch; i++) { // For each batch
+ while (cRecords < records*batch) {
+ cTrans = 0;
+ cReadIndex = 0;
+ for (int t = 0; t < trans; t++) { // For each transaction
+ transactions[t] = pNdb->startTransaction();
+ if (transactions[t] == NULL) {
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+ for (int k = 0; k < operations; k++) { // For each operation
+ NdbOperation* pOp = transactions[t]->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+
+ // Read
+ // Define primary keys
+ check = pOp->readTupleExclusive();
+ for (int a = 0; a < tab.getNoOfColumns(); a++) {
+ if (tab.getColumn(a)->getPrimaryKey() == true) {
+ if (equalForAttr(pOp, a, cReadRecords) != 0){
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ // Define attributes to read
+ for (int a = 0; a < tab.getNoOfColumns(); a++) {
+ if ((rows[cReadIndex]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+ }
+ cReadIndex++;
+ cReadRecords++;
+
+ } // For each operation
+
+ // Let's prepare...
+ transactions[t]->executeAsynchPrepare(NoCommit, &asynchCallback,
+ this);
+ cTrans++;
+
+ if (cReadRecords >= records) {
+ // No more transactions needed
+ break;
+ }
+ } // For each transaction
+
+ // Wait for all outstanding transactions
+ pNdb->sendPollNdb(3000, 0, 0);
+
+ // Verify the data!
+ for (int r = 0; r < trans*operations; r++) {
+ if (calc.verifyRowValues(rows[r]) != 0) {
+ g_info << "|- Verify failed..." << endl;
+ // Close all transactions
+ for (int t = 0; t < cTrans; t++) {
+ pNdb->closeTransaction(transactions[t]);
+ }
+ return NDBT_FAILED;
+ }
+ }
+
+ // Update
+ cTrans = 0;
+ cIndex = 0;
+ for (int t = 0; t < trans; t++) { // For each transaction
+ for (int k = 0; k < operations; k++) { // For each operation
+ NdbOperation* pOp = transactions[t]->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+
+ int updates = calc.getUpdatesValue(rows[cIndex]) + 1;
+
+ check = pOp->updateTuple();
+ if (check == -1) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+
+ // Set search condition for the record
+ for (int a = 0; a < tab.getNoOfColumns(); a++) {
+ if (tab.getColumn(a)->getPrimaryKey() == true) {
+ if (equalForAttr(pOp, a, cRecords) != 0) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Update the record
+ for (int a = 0; a < tab.getNoOfColumns(); a++) {
+ if (tab.getColumn(a)->getPrimaryKey() == false) {
+ if (setValueForAttr(pOp, a, cRecords, updates) != 0) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ cIndex++;
+ cRecords++;
+
+ } // For each operation
+
+ // Let's prepare...
+ transactions[t]->executeAsynchPrepare(Commit, &asynchCallback,
+ this);
+ cTrans++;
+
+ if (cRecords >= records) {
+ // No more transactions needed
+ break;
+ }
+ } // For each transaction
+
+ // Wait for all outstanding transactions
+ pNdb->sendPollNdb(3000, 0, 0);
+
+ // Close all transactions
+ for (int t = 0; t < cTrans; t++) {
+ pNdb->closeTransaction(transactions[t]);
+ }
+
+ } // while (cRecords < records*batch)
+
+ } // For each batch
+
+ deallocTransactions();
+ deallocRows();
+
+ g_info << "|- " << ((unsigned int)transactionsCompleted * operations)/2
+ << " updated..." << endl;
+ return NDBT_OK;
+}
+
+void
+HugoAsynchTransactions::allocTransactions(int trans) {
+ if (transactions != NULL) {
+ deallocTransactions();
+ }
+ numTransactions = trans;
+ transactions = new NdbConnection*[numTransactions];
+}
+
+void
+HugoAsynchTransactions::deallocTransactions() {
+ if (transactions != NULL){
+ delete[] transactions;
+ }
+ transactions = NULL;
+}
+
+int
+HugoAsynchTransactions::executeAsynchOperation(Ndb* pNdb,
+ int records,
+ int batch,
+ int trans,
+ int operations,
+ NDB_OPERATION theOperation,
+ ExecType theType) {
+
+ int check = 0;
+ // int retryAttempt = 0; // Not used at the moment
+ // int retryMax = 5; // Not used at the moment
+ int cTrans = 0;
+ int cRecords = 0;
+ int cIndex = 0;
+
+ transactionsCompleted = 0;
+ allocTransactions(trans);
+
+ for (int i = 0; i < batch; i++) { // For each batch
+ while (cRecords < records*batch) {
+ cTrans = 0;
+ cIndex = 0;
+ for (int t = 0; t < trans; t++) { // For each transaction
+ transactions[t] = pNdb->startTransaction();
+ if (transactions[t] == NULL) {
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+ for (int k = 0; k < operations; k++) { // For each operation
+ NdbOperation* pOp = transactions[t]->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+
+ switch (theOperation) {
+ case NO_INSERT:
+ // Insert
+ check = pOp->insertTuple();
+ if (check == -1) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+
+ // Set a calculated value for each attribute in this table
+ for (int a = 0; a < tab.getNoOfColumns(); a++) {
+ if (setValueForAttr(pOp, a, cRecords, 0 ) != 0) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+ } // For each attribute
+ break;
+ case NO_UPDATE:
+ // This is a special case and is handled in the calling client...
+ break;
+ break;
+ case NO_READ:
+ // Define primary keys
+ check = pOp->readTuple();
+ for (int a = 0; a < tab.getNoOfColumns(); a++) {
+ if (tab.getColumn(a)->getPrimaryKey() == true) {
+ if (equalForAttr(pOp, a, cRecords) != 0){
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ // Define attributes to read
+ for (int a = 0; a < tab.getNoOfColumns(); a++) {
+ if ((rows[cIndex]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+ }
+ break;
+ case NO_DELETE:
+ // Delete
+ check = pOp->deleteTuple();
+ if (check == -1) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for (int a = 0; a < tab.getNoOfColumns(); a++) {
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if (equalForAttr(pOp, a, cRecords) != 0) {
+ ERR(transactions[t]->getNdbError());
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ break;
+ default:
+ // Should not happen...
+ pNdb->closeTransaction(transactions[t]);
+ return NDBT_FAILED;
+ }
+
+ cIndex++;
+ cRecords++;
+
+ } // For each operation
+
+ // Let's prepare...
+ transactions[t]->executeAsynchPrepare(theType, &asynchCallback,
+ this);
+ cTrans++;
+
+ if (cRecords >= records) {
+ // No more transactions needed
+ break;
+ }
+ } // For each transaction
+
+ // Wait for all outstanding transactions
+ pNdb->sendPollNdb(3000, 0, 0);
+
+ // ugly... it's starts to resemble flexXXX ...:(
+ switch (theOperation) {
+ case NO_READ:
+ // Verify the data!
+ for (int r = 0; r < trans*operations; r++) {
+ if (calc.verifyRowValues(rows[r]) != 0) {
+ g_info << "|- Verify failed..." << endl;
+ // Close all transactions
+ for (int t = 0; t < cTrans; t++) {
+ pNdb->closeTransaction(transactions[t]);
+ }
+ return NDBT_FAILED;
+ }
+ }
+ break;
+ }
+
+ // Close all transactions
+ for (int t = 0; t < cTrans; t++) {
+ pNdb->closeTransaction(transactions[t]);
+ }
+
+ } // while (cRecords < records*batch)
+
+ } // For each batch
+
+ deallocTransactions();
+
+ return NDBT_OK;
+
+}
diff --git a/ndb/test/src/HugoCalculator.cpp b/ndb/test/src/HugoCalculator.cpp
new file mode 100644
index 00000000000..9e2ba9f143e
--- /dev/null
+++ b/ndb/test/src/HugoCalculator.cpp
@@ -0,0 +1,243 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HugoCalculator.hpp"
+#include <assert.h>
+#include <NdbStdio.h>
+#include <NDBT.hpp>
+
+/* *************************************************************
+ * HugoCalculator
+ *
+ * Comon class for the Hugo test suite, provides the functions
+ * that is used for calculating values to load in to table and
+ * also knows how to verify a row that's been read from db
+ *
+ * ************************************************************/
+HugoCalculator::HugoCalculator(const NdbDictionary::Table& tab) : m_tab(tab) {
+
+ // The "id" column of this table is found in the first integer column
+ for (int i=0; i<m_tab.getNoOfColumns(); i++){
+ const NdbDictionary::Column* attr = m_tab.getColumn(i);
+ if (attr->getType() == NdbDictionary::Column::Unsigned){
+ m_idCol = i;
+ break;
+ }
+ }
+
+ // The "number of updates" column for this table is found in the last column
+ for (int i=m_tab.getNoOfColumns()-1; i>=0; i--){
+ const NdbDictionary::Column* attr = m_tab.getColumn(i);
+ if (attr->getType() == NdbDictionary::Column::Unsigned){
+ m_updatesCol = i;
+ break;
+ }
+ }
+#if 0
+ ndbout << "idCol = " << m_idCol << endl;
+ ndbout << "updatesCol = " << m_updatesCol << endl;
+#endif
+ // Check that idCol is not conflicting with updatesCol
+ assert(m_idCol != m_updatesCol && m_idCol != -1 && m_updatesCol != -1);
+};
+
+Int32
+HugoCalculator::calcValue(int record,
+ int attrib,
+ int updates) const {
+ const NdbDictionary::Column* attr = m_tab.getColumn(attrib);
+ // If this is the "id" column
+ if (attrib == m_idCol)
+ return record;
+
+ // If this is the update column
+ if (attrib == m_updatesCol)
+ return updates;
+
+
+ Int32 val;
+ if (attr->getPrimaryKey())
+ val = record + attrib;
+ else
+ val = record + attrib + updates;
+ return val;
+};
+#if 0
+HugoCalculator::U_Int32 calcValue(int record, int attrib, int updates) const;
+HugoCalculator::U_Int64 calcValue(int record, int attrib, int updates) const;
+HugoCalculator::Int64 calcValue(int record, int attrib, int updates) const;
+HugoCalculator::float calcValue(int record, int attrib, int updates) const;
+HugoCalculator::double calcValue(int record, int attrib, int updates) const;
+#endif
+const char*
+HugoCalculator::calcValue(int record,
+ int attrib,
+ int updates,
+ char* buf) const {
+ const char a[26] = {"UAWBORCTDPEFQGNYHISJMKXLZ"};
+ const NdbDictionary::Column* attr = m_tab.getColumn(attrib);
+ int val = calcValue(record, attrib, updates);
+
+ int len;
+ if (attr->getPrimaryKey()){
+ // Create a string where val is printed as chars in the beginning
+ // of the string, then fill with other chars
+ // The string length is set to the same size as the attribute
+ len = attr->getLength();
+ snprintf(buf, len, "%d", val);
+ for(int i=strlen(buf); i < len; i++)
+ buf[i] = a[((val^i)%25)];
+ } else{
+
+ // Fill buf with some pattern so that we can detect
+ // anomalies in the area that we don't fill with chars
+ for (int i = 0; i<attr->getLength(); i++)
+ buf[i] = ((i+2) % 255);
+
+ // Calculate length of the string to create. We want the string
+ // length to be varied between max and min of this attribute.
+
+ len = val % (attr->getLength() + 1);
+ // If len == 0 return NULL if this is a nullable attribute
+ if (len == 0){
+ if(attr->getNullable() == true)
+ return NULL;
+ else
+ len++;
+ }
+ for(int i=0; i < len; i++)
+ buf[i] = a[((val^i)%25)];
+ buf[len] = 0;
+ }
+ return buf;
+};
+
+int
+HugoCalculator::verifyRowValues(NDBT_ResultRow* const pRow) const{
+ int id, updates;
+
+ id = pRow->attributeStore(m_idCol)->u_32_value();
+ updates = pRow->attributeStore(m_updatesCol)->u_32_value();
+
+ // Check the values of each column
+ for (int i = 0; i<m_tab.getNoOfColumns(); i++){
+ if (i != m_updatesCol && id != m_idCol) {
+
+ const NdbDictionary::Column* attr = m_tab.getColumn(i);
+ switch (attr->getType()){
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Binary:
+ case NdbDictionary::Column::Varbinary:{
+ int result = 0;
+ char* buf = new char[attr->getLength()+1];
+ const char* res = calcValue(id, i, updates, buf);
+ if (res == NULL){
+ if (!pRow->attributeStore(i)->isNULL()){
+ g_err << "|- NULL ERROR: expected a NULL but the column was not null" << endl;
+ g_err << "|- The row: \"" << (*pRow) << "\"" << endl;
+ result = -1;
+ }
+ } else{
+ if (memcmp(res, pRow->attributeStore(i)->aRef(), pRow->attributeStore(i)->arraySize()) != 0){
+ // if (memcmp(res, pRow->attributeStore(i)->aRef(), pRow->attributeStore(i)->getLength()) != 0){
+ g_err << "arraySize(): "
+ << pRow->attributeStore(i)->arraySize()
+ << ", NdbDict::Column::getLength(): " << attr->getLength()
+ << endl;
+ const char* buf2 = pRow->attributeStore(i)->aRef();
+ for (int j = 0; j < pRow->attributeStore(i)->arraySize(); j++)
+ {
+ g_err << j << ":" << buf[j] << "[" << buf2[j] << "]";
+ if (buf[j] != buf2[j])
+ {
+ g_err << "==>Match failed!";
+ }
+ g_err << endl;
+ }
+ g_err << endl;
+ g_err << "|- Invalid data found in attribute " << i << ": \""
+ << pRow->attributeStore(i)->aRef()
+ << "\" != \"" << res << "\"" << endl
+ << "Length of expected=" << (unsigned)strlen(res) << endl
+ << "Lenght of read="
+ << (unsigned)strlen(pRow->attributeStore(i)->aRef()) << endl;
+ g_err << "|- The row: \"" << (* pRow) << "\"" << endl;
+ result = -1;
+ }
+ }
+ delete []buf;
+ if (result != 0)
+ return result;
+ }
+ break;
+ case NdbDictionary::Column::Int:
+ case NdbDictionary::Column::Unsigned:{
+ Int32 cval = calcValue(id, i, updates);
+ Int32 val = pRow->attributeStore(i)->int32_value();
+ if (val != cval){
+ g_err << "|- Invalid data found: \"" << val << "\" != \""
+ << cval << "\"" << endl;
+ g_err << "|- The row: \"" << (* pRow) << "\"" << endl;
+ return -1;
+ }
+ break;
+ }
+ case NdbDictionary::Column::Bigint:
+ case NdbDictionary::Column::Bigunsigned:{
+ Uint64 cval = calcValue(id, i, updates);
+ Uint64 val = pRow->attributeStore(i)->u_64_value();
+ if (val != cval){
+ g_err << "|- Invalid data found: \"" << val << "\" != \""
+ << cval << "\""
+ << endl;
+ g_err << "|- The row: \"" << (* pRow) << "\"" << endl;
+ return -1;
+ }
+ }
+ break;
+ case NdbDictionary::Column::Float:{
+ float cval = calcValue(id, i, updates);
+ float val = pRow->attributeStore(i)->float_value();
+ if (val != cval){
+ g_err << "|- Invalid data found: \"" << val << "\" != \""
+ << cval << "\"" << endl;
+ g_err << "|- The row: \"" << (* pRow) << "\"" << endl;
+ return -1;
+ }
+ }
+ break;
+ case NdbDictionary::Column::Undefined:
+ default:
+ assert(false);
+ break;
+ }
+
+ }
+ }
+ return 0;
+}
+
+int
+HugoCalculator::getIdValue(NDBT_ResultRow* const pRow) const {
+ return pRow->attributeStore(m_idCol)->u_32_value();
+}
+
+int
+HugoCalculator::getUpdatesValue(NDBT_ResultRow* const pRow) const {
+ return pRow->attributeStore(m_updatesCol)->u_32_value();
+}
+
diff --git a/ndb/test/src/HugoOperations.cpp b/ndb/test/src/HugoOperations.cpp
new file mode 100644
index 00000000000..edcec460ba0
--- /dev/null
+++ b/ndb/test/src/HugoOperations.cpp
@@ -0,0 +1,793 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <HugoOperations.hpp>
+
+
+int HugoOperations::startTransaction(Ndb* pNdb){
+
+ if (pTrans != NULL){
+ ndbout << "HugoOperations::startTransaction, pTrans != NULL" << endl;
+ return NDBT_FAILED;
+ }
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+ ERR(err);
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int HugoOperations::closeTransaction(Ndb* pNdb){
+
+ if (pTrans != NULL){
+ pNdb->closeTransaction(pTrans);
+ pTrans = NULL;
+ }
+ pTrans = NULL;
+
+ return NDBT_OK;
+}
+
+NdbConnection* HugoOperations::getTransaction(){
+ return pTrans;
+}
+
+int HugoOperations::pkReadRecord(Ndb* pNdb,
+ int recordNo,
+ bool exclusive,
+ int numRecords){
+
+ allocRows(numRecords);
+ int check;
+ for(int r=0; r < numRecords; r++){
+ NdbOperation* pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ if (exclusive == true)
+ check = pOp->readTupleExclusive();
+ else
+ check = pOp->readTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+recordNo) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((rows[r]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+ return NDBT_OK;
+}
+
+int HugoOperations::pkDirtyReadRecord(Ndb* pNdb,
+ int recordNo,
+ int numRecords){
+
+ allocRows(numRecords);
+ int check;
+ for(int r=0; r < numRecords; r++){
+ NdbOperation* pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->dirtyRead();
+
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+recordNo) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((rows[r]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+ return NDBT_OK;
+}
+
+int HugoOperations::pkSimpleReadRecord(Ndb* pNdb,
+ int recordNo,
+ int numRecords){
+
+ allocRows(numRecords);
+ int check;
+ for(int r=0; r < numRecords; r++){
+ NdbOperation* pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->simpleRead();
+
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+recordNo) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((rows[r]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+ return NDBT_OK;
+}
+
+int HugoOperations::pkUpdateRecord(Ndb* pNdb,
+ int recordNo,
+ int numRecords,
+ int updatesValue){
+
+ allocRows(numRecords);
+ int check;
+ for(int r=0; r < numRecords; r++){
+ NdbOperation* pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->updateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+recordNo) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to update
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == false){
+ if(setValueForAttr(pOp, a, recordNo+r, updatesValue ) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+ }
+ return NDBT_OK;
+}
+
+int HugoOperations::pkInsertRecord(Ndb* pNdb,
+ int recordNo,
+ int numRecords,
+ int updatesValue){
+
+ int check;
+ for(int r=0; r < numRecords; r++){
+ NdbOperation* pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+recordNo) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to update
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == false){
+ if(setValueForAttr(pOp, a, recordNo+r, updatesValue ) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+ }
+ return NDBT_OK;
+}
+
+int HugoOperations::pkDeleteRecord(Ndb* pNdb,
+ int recordNo,
+ int numRecords){
+
+ int check;
+ for(int r=0; r < numRecords; r++){
+ NdbOperation* pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->deleteTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+recordNo) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+ }
+ return NDBT_OK;
+}
+
+int HugoOperations::scanReadRecords(Ndb* pNdb,
+ Uint32 parallelism, ScanLock lock){
+
+ NdbConnection * pCon = pNdb->hupp(pTrans);
+ NDBT_ResultRow * m_tmpRow = new NDBT_ResultRow(tab);
+ ScanTmp tmp(pCon, m_tmpRow);
+ tmp.m_op = ScanTmp::READ;
+
+ NdbOperation* pOp = pCon->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pCon->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ int check = 0;
+ switch(lock){
+ case SL_ReadHold:
+ check = pOp->openScanReadHoldLock(parallelism);
+ break;
+ case SL_Exclusive:
+ check = pOp->openScanExclusive(parallelism);
+ break;
+ case SL_Read:
+ default:
+ check = pOp->openScanRead(parallelism);
+ }
+
+ if( check == -1 ) {
+ ERR(pCon->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pCon->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((m_tmpRow->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pCon->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+
+ check = tmp.pTrans->executeScan();
+ if( check == -1 ) {
+ NdbError err = tmp.pTrans->getNdbError();
+ ERR(err);
+ return err.code;
+ }
+
+ tmp.m_delete = false;
+ m_scans.push_back(tmp);
+
+ return 0;
+}
+
+int HugoOperations::executeScanRead(Ndb* pNdb){
+
+ int check = 0;
+ for(Uint32 i = 0; i<m_scans.size(); i++){
+ ScanTmp & tmp = m_scans[i];
+ check = run(tmp);
+ if(check != 0){
+ return check;
+ }
+ }
+ while(m_scans.size() > 0){
+ ScanTmp & tmp = m_scans[m_scans.size() - 1];
+ if(tmp.m_op != ScanTmp::DONE)
+ abort();
+
+ tmp.pTrans->close();
+ delete tmp.m_tmpRow;
+ m_scans.erase(m_scans.size() - 1);
+ }
+ if(check != 0){
+ return check;
+ }
+
+ return NDBT_OK;
+}
+
+int HugoOperations::execute_Commit(Ndb* pNdb,
+ AbortOption eao){
+
+ int check = 0;
+ while(m_scans.size() > 0){
+ ScanTmp & tmp = m_scans[m_scans.size() - 1];
+ if(tmp.m_op != ScanTmp::DONE)
+ abort();
+
+ tmp.pTrans->close();
+ delete tmp.m_tmpRow;
+ m_scans.erase(m_scans.size() - 1);
+ }
+ if(check != 0){
+ return check;
+ }
+
+ check = pTrans->execute(Commit, eao);
+
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ ERR(err);
+ NdbOperation* pOp = pTrans->getNdbErrorOperation();
+ if (pOp != NULL){
+ const NdbError err2 = pOp->getNdbError();
+ ERR(err2);
+ }
+ if (err.code == 0)
+ return NDBT_FAILED;
+ return err.code;
+ }
+ return NDBT_OK;
+}
+
+int
+HugoOperations::run(ScanTmp & tmp){
+ int count = 0;
+ if(tmp.m_op == ScanTmp::DONE)
+ abort();
+
+ int eof = tmp.pTrans->nextScanResult(true) ;
+ while(eof == 0){
+ count++;
+ switch(tmp.m_op){
+ case ScanTmp::READ:
+ case ScanTmp::UPDATE:
+ case ScanTmp::DELETE:
+ break;
+ case ScanTmp::DONE:
+ abort();
+ }
+ rows.push_back(tmp.m_tmpRow->clone());
+ eof = tmp.pTrans->nextScanResult(false);
+ }
+
+ tmp.m_op = ScanTmp::DONE;
+ if (eof == -1) {
+ deallocRows();
+ NdbError err = tmp.pTrans->getNdbError();
+ ERR(err);
+ return err.code;
+ }
+
+ if(count == 0)
+ return 626;
+
+ return 0;
+}
+
+int HugoOperations::execute_NoCommit(Ndb* pNdb, AbortOption eao){
+
+ int check;
+ for(Uint32 i = 0; i<m_scans.size(); i++){
+ ScanTmp & tmp = m_scans[i];
+ check = run(tmp);
+ if(check != 0){
+ return check;
+ }
+ }
+
+ check = pTrans->execute(NoCommit, eao);
+
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ ERR(err);
+ NdbOperation* pOp;
+ while ((pOp = pTrans->getNdbErrorOperation()) != NULL){
+ const NdbError err2 = pOp->getNdbError();
+ ERR(err2);
+ }
+ if (err.code == 0)
+ return NDBT_FAILED;
+ return err.code;
+ }
+ return NDBT_OK;
+}
+
+int HugoOperations::execute_Rollback(Ndb* pNdb){
+ int check;
+ check = pTrans->execute(Rollback);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ ERR(err);
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+HugoOperations::HugoOperations(const NdbDictionary::Table& _tab):
+ UtilTransactions(_tab),
+ calc(_tab),
+ pTrans(NULL){
+
+}
+
+HugoOperations::~HugoOperations(){
+ deallocRows();
+}
+
+
+int HugoOperations::equalForAttr(NdbOperation* pOp,
+ int attrId,
+ int rowId){
+ int check = 0;
+ const NdbDictionary::Column* attr = tab.getColumn(attrId);
+ if (attr->getPrimaryKey() == false){
+ g_info << "Can't call equalForAttr on non PK attribute" << endl;
+ return NDBT_FAILED;
+ }
+
+ switch (attr->getType()){
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Binary:
+ case NdbDictionary::Column::Varbinary:{
+ char buf[8000];
+ memset(buf, 0, sizeof(buf));
+ check = pOp->equal( attr->getName(), calc.calcValue(rowId, attrId, 0, buf));
+ break;
+ }
+ case NdbDictionary::Column::Int:
+ check = pOp->equal( attr->getName(), (Int32)calc.calcValue(rowId, attrId, 0));
+ break;
+ case NdbDictionary::Column::Unsigned:
+ check = pOp->equal( attr->getName(), (Uint32)calc.calcValue(rowId, attrId, 0));
+ break;
+ case NdbDictionary::Column::Bigint:
+ check = pOp->equal( attr->getName(), (Int64)calc.calcValue(rowId, attrId, 0));
+ break;
+ case NdbDictionary::Column::Bigunsigned:
+ check = pOp->equal( attr->getName(), (Uint64)calc.calcValue(rowId, attrId, 0));
+ break;
+ case NdbDictionary::Column::Float:
+ g_info << "Float not allowed as PK value" << endl;
+ check = -1;
+ break;
+
+ default:
+ g_info << "default" << endl;
+ check = -1;
+ break;
+ }
+ return check;
+}
+
+int HugoOperations::setValueForAttr(NdbOperation* pOp,
+ int attrId,
+ int rowId,
+ int updateId){
+ int check = 0;
+ const NdbDictionary::Column* attr = tab.getColumn(attrId);
+
+ if (attr->getTupleKey()){
+ // Don't set values for TupleId PKs
+ return check;
+ }
+
+ switch (attr->getType()){
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Binary:
+ case NdbDictionary::Column::Varbinary:{
+ char buf[8000];
+ check = pOp->setValue( attr->getName(),
+ calc.calcValue(rowId, attrId, updateId, buf));
+ break;
+ }
+ case NdbDictionary::Column::Int:{
+ Int32 val = calc.calcValue(rowId, attrId, updateId);
+ check = pOp->setValue( attr->getName(), val);
+ }
+ break;
+ case NdbDictionary::Column::Bigint:{
+ Int64 val = calc.calcValue(rowId, attrId, updateId);
+ check = pOp->setValue( attr->getName(),
+ val);
+ }
+ break;
+ case NdbDictionary::Column::Unsigned:{
+ Uint32 val = calc.calcValue(rowId, attrId, updateId);
+ check = pOp->setValue( attr->getName(), val);
+ }
+ break;
+ case NdbDictionary::Column::Bigunsigned:{
+ Uint64 val = calc.calcValue(rowId, attrId, updateId);
+ check = pOp->setValue( attr->getName(),
+ val);
+ }
+ break;
+ case NdbDictionary::Column::Float:
+ check = pOp->setValue( attr->getName(),
+ (float)calc.calcValue(rowId, attrId, updateId));
+ break;
+ default:
+ check = -1;
+ break;
+ }
+ return check;
+}
+
+int
+HugoOperations::verifyUpdatesValue(int updatesValue, int _numRows){
+ _numRows = (_numRows == 0 ? rows.size() : _numRows);
+
+ int result = NDBT_OK;
+
+ for(int i = 0; i<_numRows; i++){
+ if(calc.verifyRowValues(rows[i]) != NDBT_OK){
+ g_err << "Inconsistent row"
+ << endl << "\t" << rows[i]->c_str().c_str() << endl;
+ result = NDBT_FAILED;
+ continue;
+ }
+
+ if(calc.getUpdatesValue(rows[i]) != updatesValue){
+ result = NDBT_FAILED;
+ g_err << "Invalid updates value for row " << i << endl
+ << " updatesValue: " << updatesValue << endl
+ << " calc.getUpdatesValue: " << calc.getUpdatesValue(rows[i]) << endl
+ << rows[i]->c_str().c_str() << endl;
+ continue;
+ }
+ }
+
+ if(_numRows == 0){
+ g_err << "No rows -> Invalid updates value" << endl;
+ return NDBT_FAILED;
+ }
+
+ return result;
+}
+
+void HugoOperations::allocRows(int _numRows){
+ deallocRows();
+
+ if(_numRows <= 0){
+ g_info << "Illegal value for num rows : " << _numRows << endl;
+ abort();
+ }
+
+ for(int b=0; b<_numRows; b++){
+ rows.push_back(new NDBT_ResultRow(tab));
+ }
+}
+
+void HugoOperations::deallocRows(){
+ while(rows.size() > 0){
+ delete rows.back();
+ rows.erase(rows.size() - 1);
+ }
+}
+
+int HugoOperations::saveCopyOfRecord(int numRecords ){
+
+ if (numRecords > rows.size())
+ return NDBT_FAILED;
+
+ for (int i = 0; i < numRecords; i++){
+ savedRecords.push_back(rows[i]->c_str());
+ }
+ return NDBT_OK;
+}
+
+BaseString HugoOperations::getRecordStr(int recordNum){
+ if (recordNum > rows.size())
+ return NULL;
+ return rows[recordNum]->c_str();
+}
+
+int HugoOperations::getRecordGci(int recordNum){
+ return pTrans->getGCI();
+}
+
+
+int HugoOperations::compareRecordToCopy(int numRecords ){
+ if (numRecords > rows.size())
+ return NDBT_FAILED;
+ if ((unsigned)numRecords > savedRecords.size())
+ return NDBT_FAILED;
+
+ int result = NDBT_OK;
+ for (int i = 0; i < numRecords; i++){
+ BaseString str = rows[i]->c_str();
+ ndbout << "row["<<i<<"]: " << str << endl;
+ ndbout << "sav["<<i<<"]: " << savedRecords[i] << endl;
+ if (savedRecords[i] == str){
+ ;
+ } else {
+ result = NDBT_FAILED;
+ }
+ }
+ return result;
+}
+
+void
+HugoOperations::refresh() {
+ NdbConnection* t = getTransaction();
+ if(t)
+ t->refresh();
+ for(Uint32 i = 0; i<m_scans.size(); i++){
+ if(m_scans[i].pTrans)
+ m_scans[i].pTrans->refresh();
+ }
+}
+
+int HugoOperations::indexReadRecords(Ndb*, const char * idxName, int recordNo,
+ bool exclusive,
+ int numRecords){
+
+ allocRows(numRecords);
+ int check;
+ for(int r=0; r < numRecords; r++){
+ NdbOperation* pOp = pTrans->getNdbIndexOperation(idxName, tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ if (exclusive == true)
+ check = pOp->readTupleExclusive();
+ else
+ check = pOp->readTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+recordNo) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((rows[r]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+ return NDBT_OK;
+}
+
+int
+HugoOperations::indexUpdateRecord(Ndb*,
+ const char * idxName,
+ int recordNo,
+ int numRecords,
+ int updatesValue){
+
+ allocRows(numRecords);
+ int check;
+ for(int r=0; r < numRecords; r++){
+ NdbOperation* pOp = pTrans->getNdbIndexOperation(idxName, tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pOp->updateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+recordNo) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to update
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == false){
+ if(setValueForAttr(pOp, a, recordNo+r, updatesValue ) != 0){
+ ERR(pTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+ }
+ }
+ return NDBT_OK;
+}
diff --git a/ndb/test/src/HugoTransactions.cpp b/ndb/test/src/HugoTransactions.cpp
new file mode 100644
index 00000000000..b1c55fcc780
--- /dev/null
+++ b/ndb/test/src/HugoTransactions.cpp
@@ -0,0 +1,2404 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "HugoTransactions.hpp"
+#include <NdbSleep.h>
+
+
+HugoTransactions::HugoTransactions(const NdbDictionary::Table& _tab):
+ HugoOperations(_tab),
+ row(_tab){
+
+ m_defaultScanUpdateMethod = 3;
+}
+
+HugoTransactions::~HugoTransactions(){
+ deallocRows();
+}
+
+
+int HugoTransactions::scanReadCommittedRecords(Ndb* pNdb,
+ int records,
+ int abortPercent,
+ int parallelism){
+ return scanReadRecords(pNdb, records, abortPercent, parallelism, true);
+}
+
+int
+HugoTransactions::scanReadRecords(Ndb* pNdb,
+ int records,
+ int abortPercent,
+ int parallelism,
+ bool committed){
+
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_err << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if (committed == true)
+ check = pOp->openScanReadCommitted(parallelism);
+ else
+ check = pOp->openScanRead(parallelism);
+
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((row.attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Abort after 1-100 or 1-records rows
+ int ranVal = rand();
+ int abortCount = ranVal % (records == 0 ? 100 : records);
+ bool abortTrans = false;
+ if (abort > 0){
+ // Abort if abortCount is less then abortPercent
+ if (abortCount < abortPercent)
+ abortTrans = true;
+ }
+
+ int eof;
+ int rows = 0;
+ eof = pTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+ if (calc.verifyRowValues(&row) != 0){
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if (abortCount == rows && abortTrans == true){
+ ndbout << "Scan is aborted" << endl;
+ g_info << "Scan is aborted" << endl;
+ check = pTrans->stopScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+ return NDBT_OK;
+ }
+
+ eof = pTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR_INFO(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ switch (err.code){
+ case 488:
+ case 245:
+ case 490:
+ // Too many active scans, no limit on number of retry attempts
+ break;
+ default:
+ retryAttempt++;
+ }
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ g_info << rows << " rows have been read" << endl;
+ if (records != 0 && rows != records){
+ g_err << "Check expected number of records failed" << endl
+ << " expected=" << records <<", " << endl
+ << " read=" << rows << endl;
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+
+#define RESTART_SCAN 99
+
+// Take over one record from pOrgOp and update it
+int
+HugoTransactions::takeOverAndUpdateRecord(Ndb* pNdb,
+ NdbOperation* pOrgOp){
+ int retryAttempt = 0;
+ const int retryMax = 10;
+ int check;
+ NdbConnection *pUpdTrans;
+ NdbOperation *pUpdOp;
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pUpdTrans = pNdb->startTransaction();
+ if (pUpdTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ if ((pUpdOp = pOrgOp->takeOverForUpdate(pUpdTrans)) == NULL){
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ int updates = calc.getUpdatesValue(&row) + 1;
+ int id = calc.getIdValue(&row);
+
+ // Set a calculated value for each non-PK attribute in this table
+ for (int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == false){
+ if(setValueForAttr(pUpdOp, a, id, updates ) != 0){
+ ERR(pUpdTrans->getNdbError());
+ pNdb->closeTransaction(pUpdTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ check = pUpdTrans->execute( Commit );
+ if(check == -1 ) {
+ const NdbError err = pUpdTrans->getNdbError();
+ pNdb->closeTransaction(pUpdTrans);
+
+ ERR(err);
+ if(err.code == 499 || err.code == 250){
+ return RESTART_SCAN;
+ }
+
+ switch(err.status){
+ case NdbError::Success:
+ g_info << "ERROR: NdbError reports success when transcaction failed"
+ << endl;
+ return NDBT_FAILED;
+ break;
+
+ case NdbError::TemporaryError:
+ NdbSleep_MilliSleep(50+50*retryAttempt);
+ retryAttempt++;
+ continue;
+ break;
+
+ case NdbError::UnknownResult:
+ return NDBT_FAILED;
+ break;
+
+ default:
+ case NdbError::PermanentError:
+ switch (err.code){
+ case 499:
+ case 250:
+ return NDBT_TEMPORARY;
+
+ default:
+ return NDBT_FAILED;
+ break;
+ }
+ break;
+ }
+ }
+ else{
+ pNdb->closeTransaction(pUpdTrans);
+ }
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+int
+HugoTransactions::scanUpdateRecords(Ndb* pNdb,
+ int records,
+ int abortPercent,
+ int parallelism){
+ if(m_defaultScanUpdateMethod == 1){
+ return scanUpdateRecords1(pNdb, records, abortPercent, parallelism);
+ } else if(m_defaultScanUpdateMethod == 2){
+ return scanUpdateRecords2(pNdb, records, abortPercent, parallelism);
+ } else {
+ return scanUpdateRecords3(pNdb, records, abortPercent, parallelism);
+ }
+}
+
+// Scan all records exclusive and update
+// them one by one
+int
+HugoTransactions::scanUpdateRecords1(Ndb* pNdb,
+ int records,
+ int abortPercent,
+ int parallelism){
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanExclusive(parallelism);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Read all attributes from this table
+ for(int a=0; a<tab.getNoOfColumns(); a++){
+ if((row.attributeStore(a) = pOp->getValue(tab.getColumn(a)->getName())) == NULL){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Abort after 1-100 or 1-records rows
+ int ranVal = rand();
+ int abortCount = ranVal % (records == 0 ? 100 : records);
+ bool abortTrans = false;
+ if (abort > 0){
+ // Abort if abortCount is less then abortPercent
+ if (abortCount < abortPercent)
+ abortTrans = true;
+ }
+
+
+ int eof;
+ int rows = 0;
+
+ eof = pTrans->nextScanResult();
+ while(eof == 0){
+ rows++;
+
+ if (abortCount == rows && abortTrans == true){
+ g_info << "Scan is aborted" << endl;
+ // This scan should be aborted
+ check = pTrans->stopScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+ return NDBT_OK;
+ }
+ int res = takeOverAndUpdateRecord(pNdb, pOp);
+ if(res == RESTART_SCAN){
+ eof = -2;
+ continue;
+ }
+ if (res != 0){
+ pNdb->closeTransaction(pTrans);
+ return res;
+ }
+
+ eof = pTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ switch (err.code){
+ case 488:
+ case 245:
+ case 490:
+ // Too many active scans, no limit on number of retry attempts
+ break;
+ default:
+ retryAttempt++;
+ }
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if(eof == -2){
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ g_info << rows << " rows have been updated" << endl;
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+
+// Scan all records exclusive and update
+// them batched by asking nextScanResult to
+// give us all cached records before fetching new
+// records from db
+int
+HugoTransactions::scanUpdateRecords2(Ndb* pNdb,
+ int records,
+ int abortPercent,
+ int parallelism){
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanExclusive(parallelism);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Read all attributes from this table
+ for(int a=0; a<tab.getNoOfColumns(); a++){
+ if((row.attributeStore(a) = pOp->getValue(tab.getColumn(a)->getName())) == NULL){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Abort after 1-100 or 1-records rows
+ int ranVal = rand();
+ int abortCount = ranVal % (records == 0 ? 100 : records);
+ bool abortTrans = false;
+ if (abort > 0){
+ // Abort if abortCount is less then abortPercent
+ if (abortCount < abortPercent)
+ abortTrans = true;
+ }
+
+ int eof;
+ int rows = 0;
+ NdbConnection* pUpTrans;
+
+ while((eof = pTrans->nextScanResult(true)) == 0){
+ pUpTrans = pNdb->startTransaction();
+ if (pUpTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+ do {
+ rows++;
+ if (addRowToUpdate(pNdb, pUpTrans, pOp) != 0){
+ pNdb->closeTransaction(pUpTrans);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ } while((eof = pTrans->nextScanResult(false)) == 0);
+
+ if (abortCount == rows && abortTrans == true){
+ g_info << "Scan is aborted" << endl;
+ // This scan should be aborted
+ check = pTrans->stopScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ pNdb->closeTransaction(pUpTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+ pNdb->closeTransaction(pUpTrans);
+ return NDBT_OK;
+ }
+
+ check = pUpTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pUpTrans->getNdbError();
+ ERR(err);
+ pNdb->closeTransaction(pUpTrans);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ pNdb->closeTransaction(pUpTrans);
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ g_info << rows << " rows have been updated" << endl;
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+int
+HugoTransactions::addRowToUpdate(Ndb* pNdb,
+ NdbConnection* pUpdTrans,
+ NdbOperation* pOrgOp){
+
+ int updates = calc.getUpdatesValue(&row) + 1;
+ int r = calc.getIdValue(&row);
+
+ NdbOperation* pUpdOp = pOrgOp->takeOverForUpdate(pUpdTrans);
+ if (pUpdOp == NULL){
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == false){
+ if(setValueForAttr(pUpdOp, a, r, updates ) != 0){
+ ERR(pUpdTrans->getNdbError());
+ pNdb->closeTransaction(pUpdTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ return NDBT_OK;
+}
+
+
+int
+HugoTransactions::scanUpdateRecords3(Ndb* pNdb,
+ int records,
+ int abortPercent,
+ int parallelism){
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbScanOperation *pOp;
+
+
+ while (true){
+ restart:
+ if (retryAttempt++ >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+ ERR(err);
+ if (err.status == NdbError::TemporaryError){
+ NdbSleep_MilliSleep(50);
+ continue;
+ }
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbScanOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ NdbResultSet *rs = pOp->readTuplesExclusive(parallelism);
+ if( rs == 0 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Read all attributes from this table
+ for(int a=0; a<tab.getNoOfColumns(); a++){
+ if((row.attributeStore(a) = pOp->getValue(tab.getColumn(a)->getName())) == NULL){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ if (err.status == NdbError::TemporaryError){
+ NdbSleep_MilliSleep(50);
+ continue;
+ }
+ return NDBT_FAILED;
+ }
+
+ // Abort after 1-100 or 1-records rows
+ int ranVal = rand();
+ int abortCount = ranVal % (records == 0 ? 100 : records);
+ bool abortTrans = false;
+ if (abort > 0){
+ // Abort if abortCount is less then abortPercent
+ if (abortCount < abortPercent)
+ abortTrans = true;
+ }
+
+ int rows = 0;
+ while((check = rs->nextResult(true)) == 0){
+ do {
+ rows++;
+ NdbOperation* pUp = rs->updateTuple();
+ if(pUp == 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ const int updates = calc.getUpdatesValue(&row) + 1;
+ const int r = calc.getIdValue(&row);
+
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == false){
+ if(setValueForAttr(pUp, a, r, updates ) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ if (rows == abortCount && abortTrans == true){
+ g_info << "Scan is aborted" << endl;
+ // This scan should be aborted
+ pNdb->closeTransaction(pTrans);
+ return NDBT_OK;
+ }
+ } while((check = rs->nextResult(false)) == 0);
+
+ if(check != -1){
+ check = pTrans->execute(Commit);
+ pTrans->releaseCompletedOperations();
+ }
+
+ const NdbError err = pTrans->getNdbError();
+ if( check == -1 ) {
+ pNdb->closeTransaction(pTrans);
+ ERR(err);
+ if (err.status == NdbError::TemporaryError){
+ NdbSleep_MilliSleep(50);
+ goto restart;
+ }
+ return NDBT_FAILED;
+ }
+ }
+ pNdb->closeTransaction(pTrans);
+
+ g_info << rows << " rows have been updated" << endl;
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+int
+HugoTransactions::loadTable(Ndb* pNdb,
+ int records,
+ int batch,
+ bool allowConstraintViolation,
+ int doSleep){
+ int check;
+ int retryAttempt = 0;
+ int retryMax = 5;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ const int org = batch;
+ const int cols = tab.getNoOfColumns();
+ const int brow = tab.getRowSizeInBytes();
+ const int bytes = 12 + brow + 4 * cols;
+ batch = (batch * 256); // -> 512 -> 65536k per commit
+ batch = batch/bytes; //
+ batch = batch == 0 ? 1 : batch;
+
+ if(batch != org){
+ g_info << "batch = " << org << " rowsize = " << bytes
+ << " -> rows/commit = " << batch << endl;
+ }
+
+ g_info << "|- Inserting records..." << endl;
+ for (int c=0 ; c<records ; ){
+
+ if (retryAttempt >= retryMax){
+ g_info << "Record " << c << " could not be inserted, has retried "
+ << retryAttempt << " times " << endl;
+ // Reset retry counters and continue with next record
+ retryAttempt = 0;
+ c++;
+ }
+ if (doSleep > 0)
+ NdbSleep_MilliSleep(doSleep);
+
+ pTrans = pNdb->startTransaction();
+
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ for(int b = 0; b < batch && c+b<records; b++){
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Set a calculated value for each attribute in this table
+ for (int a = 0; a<tab.getNoOfColumns(); a++){
+ if(setValueForAttr(pOp, a, c+b, 0 ) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Execute the transaction and insert the record
+ check = pTrans->execute( Commit );
+ if(check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ pNdb->closeTransaction(pTrans);
+
+ switch(err.status){
+ case NdbError::Success:
+ ERR(err);
+ g_info << "ERROR: NdbError reports success when transcaction failed"
+ << endl;
+ return NDBT_FAILED;
+ break;
+
+ case NdbError::TemporaryError:
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ break;
+
+ case NdbError::UnknownResult:
+ ERR(err);
+ return NDBT_FAILED;
+ break;
+
+ case NdbError::PermanentError:
+ if (allowConstraintViolation == true){
+ switch (err.classification){
+ case NdbError::ConstraintViolation:
+ // Tuple already existed, OK but should be reported
+ g_info << c << ": " << err.code << " " << err.message << endl;
+ c++;
+ continue;
+ break;
+ default:
+ break;
+ }
+ }
+ ERR(err);
+ return err.code;
+ break;
+ }
+ }
+ else{
+ pNdb->closeTransaction(pTrans);
+ }
+
+ // Step to next record
+ c = c+batch;
+ retryAttempt = 0;
+ }
+ return NDBT_OK;
+}
+
+int
+HugoTransactions::fillTable(Ndb* pNdb,
+ int batch){
+ int check;
+ int retryAttempt = 0;
+ int retryMax = 5;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ g_info << "|- Inserting records..." << endl;
+ for (int c=0 ; ; ){
+
+ if (retryAttempt >= retryMax){
+ g_info << "Record " << c << " could not be inserted, has retried "
+ << retryAttempt << " times " << endl;
+ // Reset retry counters and continue with next record
+ retryAttempt = 0;
+ c++;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ for(int b = 0; b < batch; b++){
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Set a calculated value for each attribute in this table
+ for (int a = 0; a<tab.getNoOfColumns(); a++){
+ if(setValueForAttr(pOp, a, c+b, 0 ) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Execute the transaction and insert the record
+ check = pTrans->execute( Commit, CommitAsMuchAsPossible );
+ if(check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ pNdb->closeTransaction(pTrans);
+
+ switch(err.status){
+ case NdbError::Success:
+ ERR(err);
+ g_info << "ERROR: NdbError reports success when transcaction failed"
+ << endl;
+ return NDBT_FAILED;
+ break;
+
+ case NdbError::TemporaryError:
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ break;
+
+ case NdbError::UnknownResult:
+ ERR(err);
+ return NDBT_FAILED;
+ break;
+
+ case NdbError::PermanentError:
+ // if (allowConstraintViolation == true){
+ // switch (err.classification){
+ // case NdbError::ConstraintViolation:
+ // // Tuple already existed, OK but should be reported
+ // g_info << c << ": " << err.code << " " << err.message << endl;
+ // c++;
+ // continue;
+ // break;
+ // default:
+ // break;es
+ // }
+ // }
+
+ // Check if this is the "db full" error
+ if (err.classification==NdbError::InsufficientSpace){
+ ERR(err);
+ return NDBT_OK;
+ }
+
+ if (err.classification == NdbError::ConstraintViolation){
+ ERR(err);
+ break;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ break;
+ }
+ }
+ else{
+ pNdb->closeTransaction(pTrans);
+ }
+
+ // Step to next record
+ c = c+batch;
+ retryAttempt = 0;
+ }
+ return NDBT_OK;
+}
+
+int
+HugoTransactions::createEvent(Ndb* pNdb){
+
+ char eventName[1024];
+ sprintf(eventName,"%s_EVENT",tab.getName());
+
+ NdbDictionary::Dictionary *myDict = pNdb->getDictionary();
+
+ if (!myDict) {
+ printf("Event Creation failedDictionary not found");
+ return NDBT_FAILED;
+ }
+
+ NdbDictionary::Event myEvent(eventName);
+ myEvent.setTable(tab.getName());
+ myEvent.addTableEvent(NdbDictionary::Event::TE_ALL);
+ // myEvent.addTableEvent(NdbDictionary::Event::TE_INSERT);
+ // myEvent.addTableEvent(NdbDictionary::Event::TE_UPDATE);
+ // myEvent.addTableEvent(NdbDictionary::Event::TE_DELETE);
+
+ // const NdbDictionary::Table *_table = myDict->getTable(tab.getName());
+ for(int a = 0; a < tab.getNoOfColumns(); a++){
+ // myEvent.addEventColumn(_table->getColumn(a)->getName());
+ myEvent.addEventColumn(a);
+ }
+
+ int res = myDict->createEvent(myEvent); // Add event to database
+
+ if (res == 0)
+ myEvent.print();
+ else {
+ g_info << "Event creation failed\n";
+ g_info << "trying drop Event, maybe event exists\n";
+ res = myDict->dropEvent(eventName);
+ if (res) {
+ g_err << "failed to drop event\n";
+ return NDBT_FAILED;
+ }
+ // try again
+ res = myDict->createEvent(myEvent); // Add event to database
+ if (res) {
+ g_err << "failed to create event\n";
+ return NDBT_FAILED;
+ }
+ }
+
+ return NDBT_OK;
+}
+
+#include <NdbEventOperation.hpp>
+#include "TestNdbEventOperation.hpp"
+#include <NdbAutoPtr.hpp>
+
+struct receivedEvent {
+ Uint32 pk;
+ Uint32 count;
+ Uint32 event;
+};
+
+int XXXXX = 0;
+int
+HugoTransactions::eventOperation(Ndb* pNdb, void* pstats,
+ int records) {
+ int myXXXXX = XXXXX++;
+
+ const char function[] = "HugoTransactions::eventOperation: ";
+ struct receivedEvent* recInsertEvent;
+ NdbAutoObjArrayPtr<struct receivedEvent>
+ p00( recInsertEvent = new struct receivedEvent[3*records] );
+ struct receivedEvent* recUpdateEvent = &recInsertEvent[records];
+ struct receivedEvent* recDeleteEvent = &recInsertEvent[2*records];
+
+ EventOperationStats &stats = *(EventOperationStats*)pstats;
+
+ stats.n_inserts = 0;
+ stats.n_deletes = 0;
+ stats.n_updates = 0;
+ stats.n_consecutive = 0;
+ stats.n_duplicates = 0;
+ stats.n_inconsistent_gcis = 0;
+
+ for (int i = 0; i < records; i++) {
+ recInsertEvent[i].pk = 0xFFFFFFFF;
+ recInsertEvent[i].count = 0;
+ recInsertEvent[i].event = 0xFFFFFFFF;
+
+ recUpdateEvent[i].pk = 0xFFFFFFFF;
+ recUpdateEvent[i].count = 0;
+ recUpdateEvent[i].event = 0xFFFFFFFF;
+
+ recDeleteEvent[i].pk = 0xFFFFFFFF;
+ recDeleteEvent[i].count = 0;
+ recDeleteEvent[i].event = 0xFFFFFFFF;
+ }
+
+ NdbDictionary::Dictionary *myDict = pNdb->getDictionary();
+
+ if (!myDict) {
+ g_err << function << "Event Creation failedDictionary not found\n";
+ return NDBT_FAILED;
+ }
+
+ int r = 0;
+ NdbEventOperation *pOp;
+
+ char eventName[1024];
+ sprintf(eventName,"%s_EVENT",tab.getName());
+ int noEventColumnName = tab.getNoOfColumns();
+
+ g_info << function << "create EventOperation\n";
+ pOp = pNdb->createEventOperation(eventName, 100);
+ if ( pOp == NULL ) {
+ g_err << function << "Event operation creation failed\n";
+ return NDBT_FAILED;
+ }
+
+ g_info << function << "get values\n";
+ NdbRecAttr* recAttr[1024];
+ NdbRecAttr* recAttrPre[1024];
+
+ const NdbDictionary::Table *_table = myDict->getTable(tab.getName());
+
+ for (int a = 0; a < noEventColumnName; a++) {
+ recAttr[a] = pOp->getValue(_table->getColumn(a)->getName());
+ recAttrPre[a] = pOp->getPreValue(_table->getColumn(a)->getName());
+ }
+
+ // set up the callbacks
+ g_info << function << "execute\n";
+ if (pOp->execute()) { // This starts changes to "start flowing"
+ g_err << function << "operation execution failed\n";
+ return NDBT_FAILED;
+ }
+
+ g_info << function << "ok\n";
+
+ int count = 0;
+ Uint32 last_inconsitant_gci = 0xEFFFFFF0;
+
+ while (r < records){
+ //printf("now waiting for event...\n");
+ int res = pNdb->pollEvents(1000); // wait for event or 1000 ms
+
+ if (res > 0) {
+ //printf("got data! %d\n", r);
+ int overrun;
+ while (pOp->next(&overrun) > 0) {
+ r++;
+ r += overrun;
+ count++;
+
+ Uint32 gci = pOp->getGCI();
+ Uint32 pk = recAttr[0]->u_32_value();
+
+ if (!pOp->isConsistent()) {
+ if (last_inconsitant_gci != gci) {
+ last_inconsitant_gci = gci;
+ stats.n_inconsistent_gcis++;
+ }
+ g_warning << "A node failure has occured and events might be missing\n";
+ }
+ g_info << function << "GCI " << gci << ": " << count;
+ struct receivedEvent* recEvent;
+ switch (pOp->getEventType()) {
+ case NdbDictionary::Event::TE_INSERT:
+ stats.n_inserts++;
+ g_info << " INSERT: ";
+ recEvent = recInsertEvent;
+ break;
+ case NdbDictionary::Event::TE_DELETE:
+ stats.n_deletes++;
+ g_info << " DELETE: ";
+ recEvent = recDeleteEvent;
+ break;
+ case NdbDictionary::Event::TE_UPDATE:
+ stats.n_updates++;
+ g_info << " UPDATE: ";
+ recEvent = recUpdateEvent;
+ break;
+ }
+
+ if (pk < records) {
+ recEvent[pk].pk = pk;
+ recEvent[pk].count++;
+ }
+
+ g_info << "overrun " << overrun << " pk " << pk;
+ for (int i = 1; i < noEventColumnName; i++) {
+ if (recAttr[i]->isNULL() >= 0) { // we have a value
+ g_info << " post[" << i << "]=";
+ if (recAttr[i]->isNULL() == 0) // we have a non-null value
+ g_info << recAttr[i]->u_32_value();
+ else // we have a null value
+ g_info << "NULL";
+ }
+ if (recAttrPre[i]->isNULL() >= 0) { // we have a value
+ g_info << " pre[" << i << "]=";
+ if (recAttrPre[i]->isNULL() == 0) // we have a non-null value
+ g_info << recAttrPre[i]->u_32_value();
+ else // we have a null value
+ g_info << "NULL";
+ }
+ }
+ g_info << endl;
+ }
+ } else
+ ;//printf("timed out\n");
+ }
+
+ // sleep ((XXXXX-myXXXXX)*2);
+
+ g_info << myXXXXX << "dropping event operation" << endl;
+
+ int res = pNdb->dropEventOperation(pOp);
+ if (res != 0) {
+ g_err << "operation execution failed\n";
+ return NDBT_FAILED;
+ }
+
+ g_info << myXXXXX << " ok" << endl;
+
+ if (stats.n_inserts > 0) {
+ stats.n_consecutive++;
+ }
+ if (stats.n_deletes > 0) {
+ stats.n_consecutive++;
+ }
+ if (stats.n_updates > 0) {
+ stats.n_consecutive++;
+ }
+ for (Uint32 i = 0; i < records/3; i++) {
+ if (recInsertEvent[i].pk != i) {
+ stats.n_consecutive ++;
+ ndbout << "missing insert pk " << i << endl;
+ } else if (recInsertEvent[i].count > 1) {
+ ndbout << "duplicates insert pk " << i
+ << " count " << recInsertEvent[i].count << endl;
+ stats.n_duplicates += recInsertEvent[i].count-1;
+ }
+ if (recUpdateEvent[i].pk != i) {
+ stats.n_consecutive ++;
+ ndbout << "missing update pk " << i << endl;
+ } else if (recUpdateEvent[i].count > 1) {
+ ndbout << "duplicates update pk " << i
+ << " count " << recUpdateEvent[i].count << endl;
+ stats.n_duplicates += recUpdateEvent[i].count-1;
+ }
+ if (recDeleteEvent[i].pk != i) {
+ stats.n_consecutive ++;
+ ndbout << "missing delete pk " << i << endl;
+ } else if (recDeleteEvent[i].count > 1) {
+ ndbout << "duplicates delete pk " << i
+ << " count " << recDeleteEvent[i].count << endl;
+ stats.n_duplicates += recDeleteEvent[i].count-1;
+ }
+ }
+
+ return NDBT_OK;
+}
+
+int
+HugoTransactions::pkReadRecords(Ndb* pNdb,
+ int records,
+ int batchsize,
+ bool dirty){
+ int reads = 0;
+ int r = 0;
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ if (batchsize == 0) {
+ g_info << "ERROR: Argument batchsize == 0 in pkReadRecords(). Not allowed." << endl;
+ return NDBT_FAILED;
+ }
+
+ allocRows(batchsize);
+
+ while (r < records){
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ for(int b=0; (b<batchsize) && (r+b < records); b++){
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if (dirty == true){
+ check = pOp->dirtyRead();
+ } else {
+ check = pOp->readTuple();
+ }
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+b) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((rows[b]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ switch(err.code){
+ case 626: // Tuple did not exist
+ g_info << r << ": " << err.code << " " << err.message << endl;
+ r++;
+ break;
+
+ default:
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ } else{
+ for (int b=0; (b<batchsize) && (r+b<records); b++){
+ if (calc.verifyRowValues(rows[b]) != 0){
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ reads++;
+ r++;
+ }
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ }
+ deallocRows();
+ g_info << reads << " records read" << endl;
+ return NDBT_OK;
+}
+
+
+
+int
+HugoTransactions::pkUpdateRecords(Ndb* pNdb,
+ int records,
+ int batch,
+ int doSleep){
+ int updated = 0;
+ int r = 0;
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ allocRows(batch);
+
+ g_info << "|- Updating records..." << endl;
+ while (r < records){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ if (doSleep > 0)
+ NdbSleep_MilliSleep(doSleep);
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ for(int b = 0; b<batch && (r+b) < records; b++){
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->readTupleExclusive();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+b) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((rows[b]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ for(int b = 0; b<batch && (b+r)<records; b++){
+ if (calc.verifyRowValues(rows[b]) != 0){
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int updates = calc.getUpdatesValue(rows[b]) + 1;
+
+ NdbOperation* pUpdOp;
+ pUpdOp = pTrans->getNdbOperation(tab.getName());
+ if (pUpdOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pUpdOp->updateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pUpdOp, a, r+b) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == false){
+ if(setValueForAttr(pUpdOp, a, r+b, updates ) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ ndbout << "r = " << r << endl;
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ else{
+ updated += batch;
+ }
+
+
+ pNdb->closeTransaction(pTrans);
+
+ r += batch; // Read next record
+ }
+
+ deallocRows();
+ g_info << "|- " << updated << " records updated" << endl;
+ return NDBT_OK;
+}
+
+int
+HugoTransactions::pkInterpretedUpdateRecords(Ndb* pNdb,
+ int records,
+ int batch){
+ int updated = 0;
+ int r = 0;
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+
+ while (r < records){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ NdbOperation* pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->readTupleExclusive();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Read update value
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (calc.isUpdateCol(a) == true){
+ if((row.attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ check = pTrans->execute(NoCommit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int updates = calc.getUpdatesValue(&row) + 1;
+
+ NdbOperation* pUpdOp;
+ pUpdOp = pTrans->getNdbOperation(tab.getName());
+ if (pUpdOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pUpdOp->interpretedUpdateTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // PKs
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pUpdOp, a, r) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Update col
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if ((tab.getColumn(a)->getPrimaryKey() == false) &&
+ (calc.isUpdateCol(a) == true)){
+
+ // TODO switch for 32/64 bit
+ const NdbDictionary::Column* attr = tab.getColumn(a);
+ Uint32 valToIncWith = 1;
+ check = pUpdOp->incValue(attr->getName(), valToIncWith);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Remaining attributes
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if ((tab.getColumn(a)->getPrimaryKey() == false) &&
+ (calc.isUpdateCol(a) == false)){
+ if(setValueForAttr(pUpdOp, a, r, updates ) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ ndbout << "r = " << r << endl;
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ else{
+ updated++;
+ }
+
+
+ pNdb->closeTransaction(pTrans);
+
+ r++; // Read next record
+
+ }
+
+ g_info << "|- " << updated << " records updated" << endl;
+ return NDBT_OK;
+}
+
+int
+HugoTransactions::pkDelRecords(Ndb* pNdb,
+ int records,
+ int batch,
+ bool allowConstraintViolation,
+ int doSleep){
+ // TODO Batch is not implemented
+ int deleted = 0;
+ int r = 0;
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ g_info << "|- Deleting records..." << endl;
+ while (r < records){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ if (doSleep > 0)
+ NdbSleep_MilliSleep(doSleep);
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->deleteTuple();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ check = pTrans->execute(Commit);
+ if( check == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ switch(err.status){
+ case NdbError::TemporaryError:
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ break;
+
+ case NdbError::PermanentError:
+ if (allowConstraintViolation == true){
+ switch (err.classification){
+ case NdbError::ConstraintViolation:
+ // Tuple did not exist, OK but should be reported
+ g_info << r << ": " << err.code << " " << err.message << endl;
+ continue;
+ break;
+ default:
+ break;
+ }
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ break;
+
+ default:
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ else {
+ deleted++;
+ }
+ pNdb->closeTransaction(pTrans);
+
+ r++; // Read next record
+
+ }
+
+ g_info << "|- " << deleted << " records deleted" << endl;
+ return NDBT_OK;
+}
+
+
+int
+HugoTransactions::lockRecords(Ndb* pNdb,
+ int records,
+ int percentToLock,
+ int lockTime){
+ // Place a lock on percentToLock% of the records in the Db
+ // Keep the locks for lockTime ms, commit operation
+ // and lock som other records
+ int r = 0;
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ // Calculate how many records to lock in each batch
+ if (percentToLock <= 0)
+ percentToLock = 1;
+ double percentVal = (double)percentToLock / 100;
+ int lockBatch = (int)(records * percentVal);
+ if (lockBatch <= 0)
+ lockBatch = 1;
+
+ allocRows(lockBatch);
+
+ while (r < records){
+ g_info << "|- Locking " << lockBatch << " records..." << endl;
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ for(int b = 0; (b<lockBatch) && (r+b < records); b++){
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->readTupleExclusive();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+b) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((rows[b]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ // NoCommit lockTime times with 100 millis interval
+ int sleepInterval = 50;
+ int lockCount = lockTime / sleepInterval;
+ int commitCount = 0;
+ do {
+ check = pTrans->execute(NoCommit);
+ if( check == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ for (int b=0; (b<lockBatch) && (r+b<records); b++){
+ if (calc.verifyRowValues(rows[b]) != 0){
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ commitCount++;
+ NdbSleep_MilliSleep(sleepInterval);
+ } while (commitCount < lockCount);
+
+ // Really commit the trans, puuh!
+ check = pTrans->execute(Commit);
+ if( check == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ else{
+ for (int b=0; (b<lockBatch) && (r<records); b++){
+ if (calc.verifyRowValues(rows[b]) != 0){
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ r++; // Read next record
+ }
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+
+ }
+ deallocRows();
+ g_info << "|- Record locking completed" << endl;
+ return NDBT_OK;
+}
+
+int
+HugoTransactions::indexReadRecords(Ndb* pNdb,
+ const char * idxName,
+ int records,
+ int batchsize){
+ int reads = 0;
+ int r = 0;
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+ NdbScanOperation *sOp;
+ NdbResultSet * rs;
+
+ const NdbDictionary::Index* pIndex
+ = pNdb->getDictionary()->getIndex(idxName, tab.getName());
+
+ const bool ordered = (pIndex->getType()==NdbDictionary::Index::OrderedIndex);
+
+ if (batchsize == 0) {
+ g_info << "ERROR: Argument batchsize == 0 in indexReadRecords(). "
+ << "Not allowed." << endl;
+ return NDBT_FAILED;
+ }
+
+ if (ordered) {
+ batchsize = 1;
+ }
+
+ allocRows(batchsize);
+
+ while (r < records){
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ for(int b=0; (b<batchsize) && (r+b < records); b++){
+ if(!ordered){
+ pOp = pTrans->getNdbIndexOperation(idxName, tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ check = pOp->readTuple();
+ } else {
+ pOp = sOp = pTrans->getNdbScanOperation(idxName, tab.getName());
+ if (sOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = 0;
+ rs = sOp->readTuples();
+ }
+
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+b) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((rows[b]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ check = pTrans->execute(Commit);
+ check = (check == -1 ? -1 : !ordered ? check : rs->nextResult(true));
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ switch(err.code){
+ case 626: // Tuple did not exist
+ g_info << r << ": " << err.code << " " << err.message << endl;
+ r++;
+ break;
+
+ default:
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ } else{
+ for (int b=0; (b<batchsize) && (r+b<records); b++){
+ if (calc.verifyRowValues(rows[b]) != 0){
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ reads++;
+ r++;
+ }
+ if(ordered && rs->nextResult(true) == 0){
+ ndbout << "Error when comparing records "
+ << " - index op next_result to many" << endl;
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ pNdb->closeTransaction(pTrans);
+ }
+ deallocRows();
+ g_info << reads << " records read" << endl;
+ return NDBT_OK;
+}
+
+
+
+int
+HugoTransactions::indexUpdateRecords(Ndb* pNdb,
+ const char * idxName,
+ int records,
+ int batchsize){
+
+ int updated = 0;
+ int r = 0;
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+ NdbScanOperation * sOp;
+ NdbResultSet * rs;
+
+ const NdbDictionary::Index* pIndex
+ = pNdb->getDictionary()->getIndex(idxName, tab.getName());
+
+ const bool ordered = (pIndex->getType()==NdbDictionary::Index::OrderedIndex);
+ if (ordered){
+ batchsize = 1;
+ }
+
+ allocRows(batchsize);
+
+ while (r < records){
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ for(int b = 0; b<batchsize && (b+r)<records; b++){
+ if(!ordered){
+ pOp = pTrans->getNdbIndexOperation(idxName, tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->readTupleExclusive();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ } else {
+ pOp = sOp = pTrans->getNdbScanOperation(idxName, tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = 0;
+ rs = sOp->readTuplesExclusive();
+ }
+
+ // Define primary keys
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pOp, a, r+b) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ // Define attributes to read
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((rows[b]->attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+
+ check = pTrans->execute(NoCommit);
+ check = (check == -1 ? -1 : !ordered ? check : rs->nextResult(true));
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+
+ if (err.status == NdbError::TemporaryError){
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ return NDBT_FAILED;
+ }
+
+ if(ordered && check != 0){
+ g_err << "Row: " << r << " not found!!" << endl;
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ for(int b = 0; b<batchsize && (b+r)<records; b++){
+ if (calc.verifyRowValues(rows[b]) != 0){
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int updates = calc.getUpdatesValue(rows[b]) + 1;
+
+ NdbOperation* pUpdOp;
+ if(!ordered){
+ pUpdOp = pTrans->getNdbIndexOperation(idxName, tab.getName());
+ check = (pUpdOp == 0 ? -1 : pUpdOp->updateTuple());
+ } else {
+ pUpdOp = rs->updateTuple();
+ }
+
+ if (pUpdOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if(!ordered){
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == true){
+ if(equalForAttr(pUpdOp, a, r+b) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ }
+
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey() == false){
+ if(setValueForAttr(pUpdOp, a, r+b, updates ) != 0){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ }
+
+ check = pTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+
+ if (err.status == NdbError::TemporaryError){
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ndbout << "r = " << r << endl;
+ return NDBT_FAILED;
+ } else {
+ updated += batchsize;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ r+= batchsize; // Read next record
+ }
+
+ g_info << "|- " << updated << " records updated" << endl;
+ return NDBT_OK;
+}
+
+
diff --git a/ndb/test/src/Makefile b/ndb/test/src/Makefile
new file mode 100644
index 00000000000..2b634bcd3cd
--- /dev/null
+++ b/ndb/test/src/Makefile
@@ -0,0 +1,31 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+ARCHIVE_TARGET := NDBT
+
+SOURCES = NDBT_ReturnCodes.cpp \
+ NDBT_Error.cpp NDBT_Tables.cpp NDBT_ResultRow.cpp \
+ NDBT_Test.cpp HugoCalculator.cpp \
+ HugoOperations.cpp HugoTransactions.cpp \
+ HugoAsynchTransactions.cpp UtilTransactions.cpp \
+ NdbRestarter.cpp NdbRestarts.cpp NDBT_Output.cpp \
+ NdbBackup.cpp NdbConfig.cpp NdbGrep.cpp NDBT_Table.cpp
+
+SOURCES.c =
+
+CFLAGS_NdbRestarter.cpp := -I$(call fixpath,$(NDB_TOP)/src/common/mgmcommon)
+CFLAGS_NdbBackup.cpp := -I$(call fixpath,$(NDB_TOP)/include/mgmcommon)
+CFLAGS_NdbConfig.cpp := -I$(call fixpath,$(NDB_TOP)/include/mgmcommon)
+CFLAGS_NdbRestarts.cpp := -I$(call fixpath,$(NDB_TOP)/include/kernel)
+CFLAGS_NdbBackup.cpp += -I$(call fixpath,$(NDB_TOP)/include/kernel)
+CFLAGS_NdbGrep.cpp += -I$(call fixpath,$(NDB_TOP)/include/kernel) -I$(call fixpath,$(NDB_TOP)/include/mgmcommon)
+
+include $(NDB_TOP)/Epilogue.mk
+
+
+
+
+
+
+
diff --git a/ndb/test/src/NDBT_Error.cpp b/ndb/test/src/NDBT_Error.cpp
new file mode 100644
index 00000000000..92c0b2e5c1f
--- /dev/null
+++ b/ndb/test/src/NDBT_Error.cpp
@@ -0,0 +1,285 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/* NDBT_Error.cpp */
+/* This program deals with error handling */
+
+#include <string.h>
+#include <assert.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbStdio.h>
+#include <NdbTest.hpp>
+#include <NDBT_Error.hpp>
+#include <NdbSleep.h>
+
+
+ErrorData::ErrorData()
+{
+ errorCountArray = new Uint32[6000];
+ resetErrorCounters();
+
+ key_error = false;
+ temporary_resource_error = true;
+ insufficient_space_error = false;
+ node_recovery_error = true;
+ overload_error = true;
+ timeout_error = true;
+ internal_error = true;
+ user_error = true;
+ application_error = false;
+}
+
+ErrorData::~ErrorData()
+{
+ delete [] errorCountArray;
+}
+
+
+//-------------------------------------------------------------------
+// Error Handling routines
+//-------------------------------------------------------------------
+
+int ErrorData::handleErrorCommon(const NdbError & error)
+{
+ int retValue = 1;
+ if (error.code > 6000) {
+ if (user_error == true) {
+ retValue = 0;
+ }//if
+ return retValue;
+ }//if
+ errorCountArray[error.code]++;
+ switch(error.classification){
+ case NdbError::NoDataFound:
+ case NdbError::ConstraintViolation:
+ if (key_error == true) {
+ retValue = 0;
+ }//if
+ break;
+ case NdbError::TemporaryResourceError:
+ if (temporary_resource_error == true) {
+ retValue = 0;
+ }//if
+ break;
+ case NdbError::InsufficientSpace:
+ if (insufficient_space_error == true) {
+ retValue = 0;
+ }//if
+ break;
+ case NdbError::NodeRecoveryError:
+ if (node_recovery_error == true) {
+ retValue = 0;
+ }//if
+ break;
+
+ case NdbError::UnknownResultError:
+ if(error.code == 4012){
+ retValue = 0;
+ }
+ if(error.code == 4115){
+ retValue = 2;
+ }
+ if(error.code == 4007 && node_recovery_error == true){
+ retValue = 3;
+ }
+ break;
+ case NdbError::OverloadError:
+ if (overload_error == true) {
+ NdbSleep_MilliSleep(50);
+ retValue = 0;
+ }//if
+ break;
+ case NdbError::TimeoutExpired:
+ if (timeout_error == true) {
+ retValue = 0;
+ }//if
+ break;
+ case NdbError::InternalError:
+ if (internal_error == true) {
+ retValue = 0;
+ }//if
+ break;
+ case NdbError::ApplicationError:
+ if (application_error == true) {
+ retValue = 0;
+ }//if
+ break;
+ case NdbError::UserDefinedError:
+ if (user_error == true) {
+ retValue = 0;
+ }//if
+ break;
+ default:
+ break;
+ }//switch
+ if(error.status == NdbError::TemporaryError)
+ retValue = 0;
+
+ return retValue;
+}//handleErrorCommon()
+
+
+void ErrorData::printErrorCounters(NdbOut & out) const
+{
+ int localLoop;
+ for (localLoop = 0; localLoop < 6000; localLoop++) {
+ int errCount = (int)errorCountArray[localLoop];
+ if (errCount > 0) {
+ out << "NDBT: ErrorCode = " << localLoop << " occurred ";
+ out << errCount << " times" << endl;
+ }//if
+ }//for
+}//printErrorCounters()
+
+
+void ErrorData::printSettings(NdbOut & out)
+{
+ out << "Key Errors are ";
+ if (key_error == false) {
+ out << "disallowed" << endl;
+ } else {
+ out << "allowed" << endl;
+ }//if
+ out << "Temporary Resource Errors are ";
+ if (temporary_resource_error == false) {
+ out << "disallowed" << endl;
+ } else {
+ out << "allowed" << endl;
+ }//if
+ if (internal_error == true) {
+ out << "Insufficient Space Errors are ";
+ }
+ if (insufficient_space_error == false) {
+ out << "disallowed" << endl;
+ } else {
+ out << "allowed" << endl;
+ }//if
+ out << "Node Recovery Errors are ";
+ if (node_recovery_error == false) {
+ out << "disallowed" << endl;
+ } else {
+ out << "allowed" << endl;
+ }//if
+ out << "Overload Errors are ";
+ if (overload_error == false) {
+ out << "disallowed" << endl;
+ } else {
+ out << "allowed" << endl;
+ }//if
+ out << "Timeout Errors are ";
+ if (timeout_error == false) {
+ out << "disallowed" << endl;
+ } else {
+ out << "allowed" << endl;
+ }//if
+ out << "Internal NDB Errors are ";
+ if (internal_error == false) {
+ out << "disallowed" << endl;
+ } else {
+ out << "allowed" << endl;
+ }//if
+ out << "User logic reported Errors are ";
+ if (user_error == false) {
+ out << "disallowed" << endl;
+ } else {
+ out << "allowed" << endl;
+ }//if
+ out << "Application Errors are ";
+ if (application_error == false) {
+ out << "disallowed" << endl;
+ } else {
+ out << "allowed" << endl;
+ }//if
+}//printSettings
+
+
+void ErrorData::printCmdLineArgs(NdbOut & out)
+{
+ out << " -key_err Allow key errors" << endl;
+ out << " -no_key_err Disallow key errors (default)" << endl;
+ out << " -temp_res_err Allow temporary resource errors (default)";
+ out << endl;
+ out << " -no_temp_res_err Disallow temporary resource errors" << endl;
+ out << " -ins_space_err Allow insufficient space errors" << endl;
+ out << " -no_ins_space_err Disallow insufficient space errors (default)";
+ out << endl;
+ out << " -noderec_err Allow Node Recovery errors (default)" << endl;
+ out << " -no_noderec_err Disallow Node Recovery errors" << endl;
+ out << " -overload_err Allow Overload errors (default)" << endl;
+ out << " -no_overload_err Disallow Overload errors" << endl;
+ out << " -timeout_err Allow Time-out errors (default)" << endl;
+ out << " -no_timeout_err Disallow Time-out errors" << endl;
+ out << " -internal_err Allow Internal NDB errors" << endl;
+ out << " -no_internal_err Disallow Internal NDB errors (default)";
+ out << " -user_err Allow user logic reported errors (default)";
+ out << endl;
+ out << " -no_user_err Disallow user logic reported errors";
+ out << endl;
+
+}//printCmdLineArgs()
+
+
+bool ErrorData::parseCmdLineArg(const char** argv, int & i)
+{
+ bool ret_Value = true;
+ if (strcmp(argv[i], "-key_err") == 0){
+ key_error = true;
+ } else if (strcmp(argv[i], "-no_key_err") == 0){
+ key_error = false;
+ } else if (strcmp(argv[i], "-temp_res_err") == 0){
+ temporary_resource_error = true;
+ } else if (strcmp(argv[i], "-no_temp_res_err") == 0){
+ temporary_resource_error = false;
+ } else if (strcmp(argv[i], "-ins_space_err") == 0){
+ insufficient_space_error = true;
+ } else if (strcmp(argv[i], "-no_ins_space_err") == 0){
+ insufficient_space_error = false;
+ } else if (strcmp(argv[i], "-noderec_err") == 0){
+ node_recovery_error = true;
+ } else if (strcmp(argv[i], "-no_noderec_err") == 0){
+ node_recovery_error = false;
+ } else if (strcmp(argv[i], "-overload_err") == 0){
+ overload_error = true;
+ } else if (strcmp(argv[i], "-no_overload_err") == 0){
+ overload_error = false;
+ } else if (strcmp(argv[i], "-timeout_err") == 0){
+ timeout_error = true;
+ } else if (strcmp(argv[i], "-no_timeout_err") == 0){
+ timeout_error = false;
+ } else if (strcmp(argv[i], "-internal_err") == 0){
+ internal_error = true;
+ } else if (strcmp(argv[i], "-no_internal_err") == 0){
+ internal_error = false;
+ } else if (strcmp(argv[i], "-user_err") == 0){
+ user_error = true;
+ } else if (strcmp(argv[i], "-no_user_err") == 0){
+ user_error = false;
+ } else {
+ ret_Value = false;
+ }//if
+ return ret_Value;
+}//bool parseCmdline
+
+void ErrorData::resetErrorCounters()
+{
+ for (int i = 0; i < 6000; i++){
+ errorCountArray[i] = 0 ;
+ }
+}
+
+
+
diff --git a/ndb/test/src/NDBT_Output.cpp b/ndb/test/src/NDBT_Output.cpp
new file mode 100644
index 00000000000..633d71991d0
--- /dev/null
+++ b/ndb/test/src/NDBT_Output.cpp
@@ -0,0 +1,36 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "NDBT_Output.hpp"
+
+FileOutputStream gerr_fileoutputstream(stderr);
+FileOutputStream gwarning_fileoutputstream(stderr);
+FileOutputStream ginfo_fileoutputstream(stdout);
+FileOutputStream gdebug_fileoutputstream(stdout);
+
+FilteredNdbOut g_err(gerr_fileoutputstream, 0, 2);
+FilteredNdbOut g_warning(gwarning_fileoutputstream, 1, 2);
+FilteredNdbOut g_info(ginfo_fileoutputstream, 2, 2);
+FilteredNdbOut g_debug(gdebug_fileoutputstream, 3, 2);
+
+void
+setOutputLevel(int i){
+ g_err.setLevel(i);
+ g_warning.setLevel(i);
+ g_info.setLevel(i);
+ g_debug.setLevel(i);
+}
diff --git a/ndb/test/src/NDBT_ResultRow.cpp b/ndb/test/src/NDBT_ResultRow.cpp
new file mode 100644
index 00000000000..098e4cfb796
--- /dev/null
+++ b/ndb/test/src/NDBT_ResultRow.cpp
@@ -0,0 +1,199 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NDBT_ResultRow.hpp"
+#include <assert.h>
+#include <stdio.h>
+#include <string.h>
+#include <NdbOut.hpp>
+
+NDBT_ResultRow::NDBT_ResultRow(const NdbDictionary::Table& tab,
+ char attrib_delimiter)
+ : m_table(tab)
+{
+ assert(tab.getObjectStatus() == NdbDictionary::Object::Retrieved);
+
+ cols = tab.getNoOfColumns();
+ names = new char * [cols];
+ data = new NdbRecAttr * [cols];
+
+ for(int i = 0; i<cols; i++){
+ names[i] = new char[255];
+ strcpy(names[i], tab.getColumn(i)->getName());
+ }
+
+ ad[0] = attrib_delimiter;
+ ad[1] = 0;
+ m_ownData = false;
+}
+
+NDBT_ResultRow::~NDBT_ResultRow(){
+ for(int i = 0; i<cols; i++){
+ delete [] names[i];
+ }
+ delete [] names;
+
+ if(m_ownData){
+ for(int i = 0; i<cols; i++)
+ delete data[i];
+ }
+ delete [] data;
+}
+
+NdbRecAttr* &
+NDBT_ResultRow::attributeStore(int i){
+
+ return data[i];
+}
+
+
+const
+NdbRecAttr *
+NDBT_ResultRow::attributeStore(const char* name){
+ for(int i = 0; i<cols; i++){
+ if (strcmp(names[i], name) == 0)
+ return data[i];
+ }
+ assert(false);
+}
+
+NdbOut &
+NDBT_ResultRow::header (NdbOut & out) const {
+ for(int i = 0; i<cols; i++){
+ out << names[i];
+ if (i < cols-1)
+ out << ad;
+ }
+ return out;
+}
+
+BaseString NDBT_ResultRow::c_str() {
+
+ BaseString str;
+
+ char buf[10];
+ for(int i = 0; i<cols; i++){
+ if(data[i]->isNULL()){
+ sprintf(buf, "NULL");
+ str.append(buf);
+ }else{
+ Uint32* p = (Uint32*)data[i]->aRef();
+ Uint32 sizeInBytes = data[i]->attrSize() * data[i]->arraySize();
+ for (Uint32 j = 0; j < sizeInBytes; j+=(sizeof(Uint32))){
+ str.append("H'");
+ sprintf(buf, "%.8x", *p);
+ p++;
+ str.append(buf);
+ if ((j + sizeof(Uint32)) < sizeInBytes)
+ str.append(", ");
+ }
+ }
+ str.append("\n");
+ }
+ str.append("*");
+
+ //ndbout << "NDBT_ResultRow::c_str() = " << str.c_str() << endl;
+
+ return str;
+}
+
+NdbOut &
+operator << (NdbOut& ndbout, const NDBT_ResultRow & res) {
+ for(int i = 0; i<res.cols; i++){
+ if(res.data[i]->isNULL())
+ ndbout << "NULL";
+ else{
+ const int size = res.data[i]->attrSize();
+ const int aSize = res.data[i]->arraySize();
+ switch(res.data[i]->attrType()){
+ case UnSigned:
+ switch(size){
+ case 8:
+ ndbout << res.data[i]->u_64_value();
+ break;
+ case 4:
+ ndbout << res.data[i]->u_32_value();
+ break;
+ case 2:
+ ndbout << res.data[i]->u_short_value();
+ break;
+ case 1:
+ ndbout << (unsigned) res.data[i]->u_char_value();
+ break;
+ default:
+ ndbout << "Unknown size";
+ }
+ break;
+
+ case Signed:
+ switch(size){
+ case 8:
+ ndbout << res.data[i]->int64_value();
+ break;
+ case 4:
+ ndbout << res.data[i]->int32_value();
+ break;
+ case 2:
+ ndbout << res.data[i]->short_value();
+ break;
+ case 1:
+ ndbout << (int) res.data[i]->char_value();
+ break;
+ default:
+ ndbout << "Unknown size";
+ }
+ break;
+
+ case String:
+ {
+ char * buf = new char[aSize+1];
+ memcpy(buf, res.data[i]->aRef(), aSize);
+ buf[aSize] = 0;
+ ndbout << buf;
+ delete [] buf;
+ // Null terminate string
+ //res.data[i][res.sizes[i]] = 0;
+ //ndbout << res.data[i];
+ }
+ break;
+
+ case Float:
+ ndbout_c("%f", res.data[i]->float_value());
+ break;
+
+ default:
+ ndbout << "Unknown(" << res.data[i]->attrType() << ")";
+ break;
+ }
+ }
+ if (i < res.cols-1)
+ ndbout << res.ad;
+ }
+
+ return ndbout;
+}
+
+NDBT_ResultRow *
+NDBT_ResultRow::clone () const {
+
+ NDBT_ResultRow * row = new NDBT_ResultRow(m_table, ad[0]);
+ row->m_ownData = true;
+ for(Uint32 i = 0; i<m_table.getNoOfColumns(); i++){
+ row->data[i] = data[i]->clone();
+ }
+
+ return row;
+}
diff --git a/ndb/test/src/NDBT_ReturnCodes.cpp b/ndb/test/src/NDBT_ReturnCodes.cpp
new file mode 100644
index 00000000000..542547c7a48
--- /dev/null
+++ b/ndb/test/src/NDBT_ReturnCodes.cpp
@@ -0,0 +1,50 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NDBT_ReturnCodes.h"
+
+/* Ndb include files */
+#include <NdbOut.hpp>
+
+/* System include files */
+#include <stdlib.h>
+
+const char* rcodeToChar(int rcode){
+ switch (rcode){
+ case NDBT_OK:
+ return "OK";
+ break;
+ case NDBT_FAILED:
+ return "Failed";
+ break;
+ case NDBT_WRONGARGS:
+ return "Wrong arguments";
+ break;
+ case NDBT_TEMPORARY:
+ return "Temporary error";
+ break;
+
+ default:
+ return "Unknown";
+ break;
+ }
+}
+
+int NDBT_ProgramExit(int rcode){
+ ndbout_c("\nNDBT_ProgramExit: %d - %s\n", rcode, rcodeToChar(rcode));
+ // exit(rcode);
+ return rcode;
+}
diff --git a/ndb/test/src/NDBT_Table.cpp b/ndb/test/src/NDBT_Table.cpp
new file mode 100644
index 00000000000..2bd2c265f10
--- /dev/null
+++ b/ndb/test/src/NDBT_Table.cpp
@@ -0,0 +1,158 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NDBT_Table.hpp"
+#include <NdbString.h>
+#include <assert.h>
+#include <NdbTimer.hpp>
+#include <NDBT.hpp>
+
+
+class NdbOut&
+operator <<(class NdbOut& ndbout, const NDBT_Attribute & attr){
+
+ NdbDictionary::Column::Type type = attr.getType();
+ bool key = attr.getPrimaryKey();
+ bool null = attr.getNullable();
+
+ ndbout << attr.getName() << "\t";
+ char tmp[100];
+ if(attr.getLength() != 1)
+ snprintf(tmp, 100," [%d]", attr.getLength());
+ else
+ tmp[0] = 0;
+
+ switch(type){
+ case NdbDictionary::Column::Tinyint:
+ ndbout << "Tinyint" << tmp;
+ break;
+ case NdbDictionary::Column::Tinyunsigned:
+ ndbout << "Tinyunsigned" << tmp;
+ break;
+ case NdbDictionary::Column::Smallint:
+ ndbout << "Smallint" << tmp;
+ break;
+ case NdbDictionary::Column::Smallunsigned:
+ ndbout << "Smallunsigned" << tmp;
+ break;
+ case NdbDictionary::Column::Mediumint:
+ ndbout << "Mediumint" << tmp;
+ break;
+ case NdbDictionary::Column::Mediumunsigned:
+ ndbout << "Mediumunsigned" << tmp;
+ break;
+ case NdbDictionary::Column::Int:
+ ndbout << "Int" << tmp;
+ break;
+ case NdbDictionary::Column::Unsigned:
+ ndbout << "Unsigned" << tmp;
+ break;
+ case NdbDictionary::Column::Bigint:
+ ndbout << "Bigint" << tmp;
+ break;
+ case NdbDictionary::Column::Bigunsigned:
+ ndbout << "Bigunsigned" << tmp;
+ break;
+ case NdbDictionary::Column::Float:
+ ndbout << "Float" << tmp;
+ break;
+ case NdbDictionary::Column::Double:
+ ndbout << "Double" << tmp;
+ break;
+ case NdbDictionary::Column::Decimal:
+ ndbout << "Decimal("
+ << attr.getScale() << ", " << attr.getPrecision() << ")"
+ << tmp;
+ break;
+ case NdbDictionary::Column::Char:
+ ndbout << "Char(" << attr.getLength() << ")";
+ break;
+ case NdbDictionary::Column::Varchar:
+ ndbout << "Varchar(" << attr.getLength() << ")";
+ break;
+ case NdbDictionary::Column::Binary:
+ ndbout << "Binary(" << attr.getLength() << ")";
+ break;
+ case NdbDictionary::Column::Varbinary:
+ ndbout << "Varbinary(" << attr.getLength() << ")";
+ break;
+ case NdbDictionary::Column::Datetime:
+ ndbout << "Datetime" << tmp;
+ break;
+ case NdbDictionary::Column::Timespec:
+ ndbout << "Timespec" << tmp;
+ break;
+ case NdbDictionary::Column::Blob:
+ ndbout << "Blob" << tmp;
+ break;
+ case NdbDictionary::Column::Undefined:
+ ndbout << "Undefined" << tmp;
+ break;
+ default:
+ ndbout << "Unknown(" << type << ")";
+ }
+
+ ndbout << "\t";
+ if(null){
+ ndbout << "NULL";
+ } else {
+ ndbout << "NOT NULL";
+ }
+ ndbout << "\t";
+
+ if(key)
+ ndbout << "\tprimary key";
+
+ return ndbout;
+}
+
+class NdbOut&
+operator <<(class NdbOut& ndbout, const NDBT_Table & tab)
+{
+ ndbout << "-- " << tab.getName() << " --" << endl;
+
+ ndbout << "Version: " << tab.getObjectVersion() << endl;
+ ndbout << "Fragment type: " << tab.getFragmentType() << endl;
+ ndbout << "K Value: " << tab.getKValue()<< endl;
+ ndbout << "Min load factor: " << tab.getMinLoadFactor()<< endl;
+ ndbout << "Max load factor: " << tab.getMaxLoadFactor()<< endl;
+ ndbout << "Temporary table: " << (tab.getStoredTable() ? "no" : "yes") << endl;
+ ndbout << "Number of attributes: " << tab.getNoOfColumns() << endl;
+ ndbout << "Number of primary keys: " << tab.getNoOfPrimaryKeys() << endl;
+ //<< ((tab.getTupleKey() == TupleId) ? " tupleid" : "") <<endl;
+ ndbout << "TableStatus: ";
+ switch(tab.getObjectStatus()){
+ case NdbDictionary::Object::New:
+ ndbout << "New" << endl;
+ break;
+ case NdbDictionary::Object::Changed:
+ ndbout << "Changed" << endl;
+ break;
+ case NdbDictionary::Object::Retrieved:
+ ndbout << "Retrieved" << endl;
+ break;
+ default:
+ ndbout << "Unknown(" << tab.getObjectStatus() << ")" << endl;
+ }
+
+ ndbout << "-- Attributes -- " << endl;
+ int noOfAttributes = tab.getNoOfColumns();
+ for(int i = 0; i<noOfAttributes; i++){
+ ndbout << (* (const NDBT_Attribute*)tab.getColumn(i)) << endl;
+ }
+
+ return ndbout;
+}
diff --git a/ndb/test/src/NDBT_Tables.cpp b/ndb/test/src/NDBT_Tables.cpp
new file mode 100644
index 00000000000..c7d4f458165
--- /dev/null
+++ b/ndb/test/src/NDBT_Tables.cpp
@@ -0,0 +1,842 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NDBT.hpp>
+#include <NDBT_Table.hpp>
+#include <NDBT_Tables.hpp>
+
+#include <assert.h>
+/* ******************************************************* */
+// Define Ndb standard tables
+//
+// USE ONLY UPPERLETTERS IN TAB AND COLUMN NAMES
+/* ******************************************************* */
+
+/*
+ * These are our "official" test tables
+ *
+ */
+/* T1 */
+static
+const
+NDBT_Attribute T1Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned),
+};
+static
+const
+NDBT_Table T1("T1", sizeof(T1Attribs)/sizeof(NDBT_Attribute), T1Attribs);
+
+/* T2 */
+static
+const
+NDBT_Attribute T2Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Bigunsigned, 1, true),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned,
+ 1, false, true), // Nullable
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned)
+};
+static
+const
+NDBT_Table T2("T2", sizeof(T2Attribs)/sizeof(NDBT_Attribute), T2Attribs);
+
+/* T3 */
+static
+const
+NDBT_Attribute T3Attribs[] = {
+ NDBT_Attribute("ID", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("PERSNR", NdbDictionary::Column::Char, 10),
+ NDBT_Attribute("NAME", NdbDictionary::Column::Char, 25),
+ NDBT_Attribute("ADRESS", NdbDictionary::Column::Char, 50),
+ NDBT_Attribute("ADRESS2", NdbDictionary::Column::Char,
+ 30, false, true), // Nullable
+ NDBT_Attribute("FÖDELSEÅR", NdbDictionary::Column::Unsigned)
+};
+static
+const
+NDBT_Table T3("T3", sizeof(T3Attribs)/sizeof(NDBT_Attribute), T3Attribs);
+
+/* T4 */
+static
+const
+NDBT_Attribute T4Attribs[] = {
+ NDBT_Attribute("REGNR", NdbDictionary::Column::Char, 6, true),
+ NDBT_Attribute("YEAR", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("OWNER", NdbDictionary::Column::Char, 25),
+ NDBT_Attribute("ADRESS", NdbDictionary::Column::Char, 50),
+ NDBT_Attribute("ADRESS2", NdbDictionary::Column::Char,
+ 30, false, true), // Nullable
+ NDBT_Attribute("OWNERID", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("CHECKDATE", NdbDictionary::Column::Unsigned)
+};
+static
+const
+NDBT_Table T4("T4", sizeof(T4Attribs)/sizeof(NDBT_Attribute), T4Attribs);
+
+/* T5 */
+static
+const
+NDBT_Attribute T5Attribs[] = {
+ NDBT_Attribute("OWNERID", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("REGNR", NdbDictionary::Column::Char, 6, true),
+ NDBT_Attribute("CREATEDDATE", NdbDictionary::Column::Unsigned)
+};
+static
+const
+NDBT_Table T5("T5", sizeof(T5Attribs)/sizeof(NDBT_Attribute), T5Attribs);
+
+/* T6 */
+static
+const
+NDBT_Attribute T6Attribs[] = {
+ NDBT_Attribute("PK1", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("ATTR1", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR2", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR3", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR4", NdbDictionary::Column::Char,
+ 47, false, true),// Nullable
+ NDBT_Attribute("ATTR5", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR6", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR7", NdbDictionary::Column::Char,
+ 48, false, true),// Nullable
+ NDBT_Attribute("ATTR8", NdbDictionary::Column::Char,
+ 50, false, true), // Nullable
+ NDBT_Attribute("ATTR9", NdbDictionary::Column::Int),
+ NDBT_Attribute("ATTR10", NdbDictionary::Column::Float),
+ NDBT_Attribute("ATTR11", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR12", NdbDictionary::Column::Char, 49),
+ NDBT_Attribute("ATTR13", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR14", NdbDictionary::Column::Char, 50),
+ NDBT_Attribute("ATTR15", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR16", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR17", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR18", NdbDictionary::Column::Char, 257),
+ NDBT_Attribute("ATTR19", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR20", NdbDictionary::Column::Unsigned),
+};
+static
+const
+NDBT_Table T6("T6", sizeof(T6Attribs)/sizeof(NDBT_Attribute), T6Attribs);
+
+/* T7 */
+static
+const
+NDBT_Attribute T7Attribs[] = {
+ NDBT_Attribute("PK1", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("PK2", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("PK3", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("PK4", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("ATTR1", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR2", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR3", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR4", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR5", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR6", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR7", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR8", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR9", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR10", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR11", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR12", NdbDictionary::Column::Char, 259),
+ NDBT_Attribute("ATTR13", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR14", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR15", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR16", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR17", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR18", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR19", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ATTR20", NdbDictionary::Column::Unsigned),
+};
+static
+const
+NDBT_Table T7("T7", sizeof(T7Attribs)/sizeof(NDBT_Attribute), T7Attribs);
+
+/* T8 */
+static
+const
+NDBT_Attribute T8Attribs[] = {
+ NDBT_Attribute("PERSON_ID", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("NAME", NdbDictionary::Column::Char, 257),
+ NDBT_Attribute("ADRESS", NdbDictionary::Column::Char, 513),
+ NDBT_Attribute("POSTADRESS", NdbDictionary::Column::Char, 1173),
+ NDBT_Attribute("VALUE", NdbDictionary::Column::Unsigned),
+
+};
+static
+const
+NDBT_Table T8("T8", sizeof(T8Attribs)/sizeof(NDBT_Attribute), T8Attribs);
+
+/* T9 */
+static
+const
+NDBT_Attribute T9Attribs[] = {
+ NDBT_Attribute("KF_SKAPAD", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("PLATS_ID", NdbDictionary::Column::Char, 2, true),
+ NDBT_Attribute("TNR_SKAPAD", NdbDictionary::Column::Char, 12, true),
+ NDBT_Attribute("DELG_MOT", NdbDictionary::Column::Char, 1, true),
+ NDBT_Attribute("VALUE", NdbDictionary::Column::Unsigned),
+};
+static
+const
+NDBT_Table T9("T9", sizeof(T9Attribs)/sizeof(NDBT_Attribute), T9Attribs);
+
+/* T10 - Long key table */
+static
+const
+NDBT_Attribute T10Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Char, 256, true),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Char, 257),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned),
+};
+static
+const
+NDBT_Table T10("T10", sizeof(T10Attribs)/sizeof(NDBT_Attribute), T10Attribs);
+
+
+/* T11 - Primary key is not first attribute */
+static
+const
+NDBT_Attribute T11Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Char, 111),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Char, 113)
+};
+
+static
+const
+NDBT_Table T11("T11", sizeof(T11Attribs)/sizeof(NDBT_Attribute), T11Attribs);
+
+/* T12 - 16 primary keys */
+static
+const
+NDBT_Attribute T12Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL6", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL7", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL8", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL9", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL10", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL11", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL12", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL13", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL14", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL15", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL16", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL20", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL30", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL40", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL50", NdbDictionary::Column::Unsigned)
+};
+
+static
+const
+NDBT_Table T12("T12", sizeof(T12Attribs)/sizeof(NDBT_Attribute), T12Attribs);
+
+/* T13 - Long key table */
+static
+const
+NDBT_Attribute T13Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Char, 257, true),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Char, 259, true),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Char, 113, true),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL6", NdbDictionary::Column::Unsigned),
+};
+static
+const
+NDBT_Table T13("T13", sizeof(T13Attribs)/sizeof(NDBT_Attribute), T13Attribs);
+
+/* T14 - 5 primary keys */
+static
+const
+NDBT_Attribute T14Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Char, 4, true),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL20", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL30", NdbDictionary::Column::Int),
+ NDBT_Attribute("KOL40", NdbDictionary::Column::Float),
+ NDBT_Attribute("KOL50", NdbDictionary::Column::Char, 200)
+};
+
+
+static
+const
+NDBT_Table T14("T14", sizeof(T14Attribs)/sizeof(NDBT_Attribute), T14Attribs);
+
+/*
+ C2 DHCP TABLES, MAYBE THESE SHOULD BE MOVED TO THE UTIL_TABLES?
+*/
+static
+const
+NDBT_Attribute C2_PORTS_Attribs[] = {
+ NDBT_Attribute("ID", NdbDictionary::Column::Unsigned, true),
+ NDBT_Attribute("PORT", NdbDictionary::Column::Char, 16, true),
+ NDBT_Attribute("ACCESSNODE", NdbDictionary::Column::Char, 16, true),
+ NDBT_Attribute("POP", NdbDictionary::Column::Char, 64, true),
+ NDBT_Attribute("VLAN", NdbDictionary::Column::Char, 16),
+ NDBT_Attribute("COMMENT", NdbDictionary::Column::Char, 128),
+ NDBT_Attribute("SNMPINDEX", NdbDictionary::Column::Int),
+ NDBT_Attribute("PORTSTATE", NdbDictionary::Column::Int),
+ NDBT_Attribute("UPDATES", NdbDictionary::Column::Unsigned)
+};
+
+static
+NDBT_Table C2_PORTS("C2_PORTS", sizeof(C2_PORTS_Attribs)/sizeof(NDBT_Attribute), C2_PORTS_Attribs);
+
+static
+const
+NDBT_Attribute C2_SERVICES_Attribs[] = {
+ NDBT_Attribute("ID", NdbDictionary::Column::Unsigned, true),
+ NDBT_Attribute("PORT", NdbDictionary::Column::Char, 16, true),
+ NDBT_Attribute("ACCESSNODE", NdbDictionary::Column::Char, 16, true),
+ NDBT_Attribute("POP", NdbDictionary::Column::Char, 64, true),
+ NDBT_Attribute("ACCESSTYPE", NdbDictionary::Column::Int, true),
+ NDBT_Attribute("CUSTOMER_ID", NdbDictionary::Column::Int),
+ NDBT_Attribute("PROVIDER", NdbDictionary::Column::Int),
+ NDBT_Attribute("TEXPIRE", NdbDictionary::Column::Int),
+ NDBT_Attribute("NUM_IP", NdbDictionary::Column::Int),
+ NDBT_Attribute("LEASED_NUM_IP", NdbDictionary::Column::Int),
+ NDBT_Attribute("LOCKED_IP", NdbDictionary::Column::Int),
+ NDBT_Attribute("STATIC_DNS", NdbDictionary::Column::Int),
+ NDBT_Attribute("SUSPENDED_SERVICES", NdbDictionary::Column::Int),
+ NDBT_Attribute("UPDATES", NdbDictionary::Column::Unsigned)
+};
+
+static
+NDBT_Table C2_SERVICES("C2_SERVICES", sizeof(C2_SERVICES_Attribs)/sizeof(NDBT_Attribute), C2_SERVICES_Attribs);
+
+static
+const
+NDBT_Attribute C2_CLIENTS_Attribs[] = {
+ NDBT_Attribute("ID", NdbDictionary::Column::Unsigned, true),
+ NDBT_Attribute("PORT", NdbDictionary::Column::Char, 16), // SI2
+ NDBT_Attribute("ACCESSNODE", NdbDictionary::Column::Char, 16), // SI2
+ NDBT_Attribute("POP", NdbDictionary::Column::Char, 64), // SI2
+ NDBT_Attribute("MAC", NdbDictionary::Column::Char, 12, true),
+ NDBT_Attribute("MAC_EXPIRE", NdbDictionary::Column::Int, 1),
+ NDBT_Attribute("IIP", NdbDictionary::Column::Int), // SI1
+ NDBT_Attribute("P_EXPIRE", NdbDictionary::Column::Int),
+ NDBT_Attribute("HOSTNAME", NdbDictionary::Column::Char, 32),
+ NDBT_Attribute("DETECTED", NdbDictionary::Column::Int),
+ NDBT_Attribute("STATUS", NdbDictionary::Column::Int),
+ NDBT_Attribute("NUM_REQUESTS", NdbDictionary::Column::Int),
+ NDBT_Attribute("ACCESSTYPE", NdbDictionary::Column::Int),
+ NDBT_Attribute("OS_TYPE", NdbDictionary::Column::Int),
+ NDBT_Attribute("GW", NdbDictionary::Column::Int),
+ NDBT_Attribute("UPDATES", NdbDictionary::Column::Unsigned)
+};
+
+static
+NDBT_Table C2_CLIENTS("C2_CLIENTS", sizeof(C2_CLIENTS_Attribs)/sizeof(NDBT_Attribute), C2_CLIENTS_Attribs);
+
+// Define array with pointer to all tables
+static
+const
+NDBT_Table *test_tables[]=
+{
+ &T1,
+ &T2,
+ &T3,
+ &T4,
+ &T5,
+ &T6,
+ &T7,
+ &T8,
+ &T9,
+ &T10,
+ &T11,
+ &T12,
+ &T13,
+ &T14,
+ &C2_PORTS,
+ &C2_SERVICES,
+ &C2_CLIENTS
+
+};
+
+static
+const
+int numTestTables = sizeof(test_tables)/sizeof(NDBT_Table*);
+
+
+/**
+ * Define tables we should not be able to create
+ */
+
+/* F1
+ *
+ * Error: PK and column with same name
+ */
+static
+const
+NDBT_Attribute F1Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Unsigned)
+};
+
+static
+const
+NDBT_Table F1("F1", sizeof(F1Attribs)/sizeof(NDBT_Attribute), F1Attribs);
+
+/* F2
+ *
+ * Error: Two columns with same name
+ */
+static
+const
+NDBT_Attribute F2Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned)
+};
+
+static
+const
+NDBT_Table F2("F2", sizeof(F2Attribs)/sizeof(NDBT_Attribute), F2Attribs);
+
+/* F3
+ *
+ * Error: Too many primary keys defined, 16 is max?
+ */
+static
+const
+NDBT_Attribute F3Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL6", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL7", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL8", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL9", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL10", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL11", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL12", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL13", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL14", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL15", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL16", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL17", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("KOL20", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL30", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL40", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL50", NdbDictionary::Column::Unsigned)
+};
+
+static
+const
+NDBT_Table F3("F3", sizeof(F3Attribs)/sizeof(NDBT_Attribute), F3Attribs);
+
+/* F4
+ *
+ * Error: Too long key
+ */
+static
+const
+NDBT_Attribute F4Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Unsigned, 9999999, true),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned)
+};
+
+static
+const
+NDBT_Table F4("F4", sizeof(F4Attribs)/sizeof(NDBT_Attribute), F4Attribs);
+
+/* F5
+ *
+ * Error: Too long attr name
+ */
+static
+const
+NDBT_Attribute F5Attribs[] = {
+ NDBT_Attribute("KOL1WITHVERRYLONGNAME_ISITTOLONG", NdbDictionary::Column::Unsigned, true),
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned)
+};
+
+static
+const
+NDBT_Table F5("F5", sizeof(F5Attribs)/sizeof(NDBT_Attribute), F5Attribs);
+
+/* F6
+ *
+ * Error: Zero length of pk attribute
+ */
+static
+const
+NDBT_Attribute F6Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Char, 0, true, false),
+ NDBT_Attribute("KOL2", NdbDictionary::Column::Char, 256),
+};
+
+static
+const
+NDBT_Table F6("F6", sizeof(F6Attribs)/sizeof(NDBT_Attribute), F6Attribs);
+
+/* F7
+ *
+ * Error: Table without primary key
+ */
+static
+const
+NDBT_Attribute F7Attribs[] = {
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Unsigned)
+};
+
+NDBT_Table F7("F7", sizeof(F7Attribs)/sizeof(NDBT_Attribute), F7Attribs);
+
+/* F8
+ *
+ * Error: Table without nullable primary key
+ */
+static
+const
+NDBT_Attribute F8Attribs[] = {
+ NDBT_Attribute("KOL3", NdbDictionary::Column::Int, 1, true, true),
+ NDBT_Attribute("KOL4", NdbDictionary::Column::Int),
+ NDBT_Attribute("KOL5", NdbDictionary::Column::Int)
+};
+
+NDBT_Table F8("F8", sizeof(F8Attribs)/sizeof(NDBT_Attribute), F8Attribs);
+
+
+/* F15 - 2-node crash in v20x */
+static
+const
+NDBT_Attribute F15Attribs[] = {
+ NDBT_Attribute("KOL1", NdbDictionary::Column::Char, 40, true)
+};
+static
+const
+NDBT_Table F15("F15", sizeof(F15Attribs)/sizeof(NDBT_Attribute), F15Attribs);
+
+// Define array with pointer to tables that we should not be able to create
+static
+const
+NDBT_Table *fail_tables[]=
+{
+ &F1,
+ &F2,
+ &F3,
+ &F4,
+ &F5,
+ &F6,
+ &F7,
+ &F8,
+ &F15
+};
+
+static
+const
+int numFailTables = sizeof(fail_tables)/sizeof(NDBT_Table*);
+
+
+/**
+ * Define util tables that we may create
+ */
+
+
+/* GL
+ * General ledger table for bank application
+ */
+static
+const
+NDBT_Attribute GL_Attribs[] = {
+ NDBT_Attribute("TIME", NdbDictionary::Column::Bigunsigned, 1, true),
+ NDBT_Attribute("ACCOUNT_TYPE", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("BALANCE", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("DEPOSIT_COUNT", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("DEPOSIT_SUM", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("WITHDRAWAL_COUNT", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("WITHDRAWAL_SUM", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("PURGED", NdbDictionary::Column::Unsigned)
+};
+
+static
+NDBT_Table GL("GL", sizeof(GL_Attribs)/sizeof(NDBT_Attribute), GL_Attribs);
+
+/* ACCOUNT
+ * Account table for bank application
+ */
+static
+const
+NDBT_Attribute ACCOUNT_Attribs[] = {
+ NDBT_Attribute("ACCOUNT_ID", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("OWNER", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("BALANCE", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("ACCOUNT_TYPE", NdbDictionary::Column::Unsigned),
+};
+
+static
+NDBT_Table ACCOUNT("ACCOUNT", sizeof(ACCOUNT_Attribs)/sizeof(NDBT_Attribute), ACCOUNT_Attribs);
+
+/* TRANSACTION
+ * Transaction table for bank application
+ */
+static
+const
+NDBT_Attribute TRANSACTION_Attribs[] = {
+ NDBT_Attribute("TRANSACTION_ID", NdbDictionary::Column::Bigunsigned, 1, true),
+ NDBT_Attribute("ACCOUNT", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("ACCOUNT_TYPE", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("OTHER_ACCOUNT", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("TRANSACTION_TYPE", NdbDictionary::Column::Unsigned),
+ NDBT_Attribute("TIME", NdbDictionary::Column::Bigunsigned),
+ NDBT_Attribute("AMOUNT", NdbDictionary::Column::Unsigned),
+};
+
+static
+NDBT_Table TRANSACTION("TRANSACTION", sizeof(TRANSACTION_Attribs)/sizeof(NDBT_Attribute), TRANSACTION_Attribs);
+
+/* SYSTEM_VALUES
+ * System values table for bank application
+ */
+static
+const
+NDBT_Attribute SYSTEM_VALUES_Attribs[] = {
+ NDBT_Attribute("SYSTEM_VALUES_ID", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("VALUE", NdbDictionary::Column::Bigunsigned)
+};
+
+static
+NDBT_Table SYSTEM_VALUES("SYSTEM_VALUES", sizeof(SYSTEM_VALUES_Attribs)/sizeof(NDBT_Attribute), SYSTEM_VALUES_Attribs);
+
+/* ACCOUNT_TYPES
+ * Account types table for bank application
+ */
+static
+const
+NDBT_Attribute ACCOUNT_TYPES_Attribs[] = {
+ NDBT_Attribute("ACCOUNT_TYPE_ID", NdbDictionary::Column::Unsigned, 1, true),
+ NDBT_Attribute("DESCRIPTION", NdbDictionary::Column::Char, 64)
+};
+
+static
+NDBT_Table ACCOUNT_TYPES("ACCOUNT_TYPE", sizeof(ACCOUNT_TYPES_Attribs)/sizeof(NDBT_Attribute), ACCOUNT_TYPES_Attribs);
+
+
+// Define array with pointer to util tables
+static
+const
+NDBT_Table *util_tables[]=
+{
+ &GL,
+ &ACCOUNT,
+ &TRANSACTION,
+ &SYSTEM_VALUES,
+ &ACCOUNT_TYPES
+};
+
+static
+const
+int numUtilTables = sizeof(util_tables)/sizeof(NDBT_Table*);
+
+
+const
+NdbDictionary::Table*
+NDBT_Tables::getTable(const char* _nam){
+ // Search tables list to find a table
+ NDBT_Table* tab = NULL;
+ for (int i=0; i<numTestTables; i++){
+ if (strcmp(test_tables[i]->getName(), _nam) == 0){
+ return test_tables[i];
+ }
+ }
+ for (int i=0; i<numFailTables; i++){
+ if (strcmp(fail_tables[i]->getName(), _nam) == 0){
+ return fail_tables[i];
+ }
+ }
+ for (int i=0; i<numUtilTables; i++){
+ if (strcmp(util_tables[i]->getName(), _nam) == 0){
+ return util_tables[i];
+ }
+ }
+ // TPK_no tables
+ // Dynamcially create table vith primary key size
+ // set to no
+ // Useful for testing key sizes 1 - max
+ int pkSizeOfTable;
+ if(sscanf(_nam, "TPK_%d", &pkSizeOfTable) == 1){
+ return tableWithPkSize(_nam, pkSizeOfTable);
+ }
+ return tab;
+}
+
+const NdbDictionary::Table*
+NDBT_Tables::tableWithPkSize(const char* _nam, Uint32 pkSize){
+ NdbDictionary::Table* tab = new NdbDictionary::Table(_nam);
+
+ // Add one PK of the desired length
+ tab->addColumn(NDBT_Attribute("PK1",
+ NdbDictionary::Column::Char,
+ pkSize,
+ true));
+
+ // Add 4 attributes
+ tab->addColumn(NDBT_Attribute("ATTR1",
+ NdbDictionary::Column::Char,
+ 21));
+
+ tab->addColumn(NDBT_Attribute("ATTR2",
+ NdbDictionary::Column::Char,
+ 124));
+
+ tab->addColumn(NDBT_Attribute("ATTR3",
+ NdbDictionary::Column::Unsigned));
+
+ tab->addColumn(NDBT_Attribute("ATTR4",
+ NdbDictionary::Column::Unsigned));
+
+ return tab;
+}
+
+const NdbDictionary::Table*
+NDBT_Tables::getTable(int _num){
+ // Get table at pos _num
+ assert(_num < numTestTables);
+ return test_tables[_num];
+}
+
+int
+NDBT_Tables::getNumTables(){
+ return numTestTables;
+}
+
+int
+NDBT_Tables::createAllTables(Ndb* pNdb, bool _temp, bool existsOk){
+
+ for (int i=0; i < NDBT_Tables::getNumTables(); i++){
+
+ const NdbDictionary::Table* tab = NDBT_Tables::getTable(i);
+ if (tab == NULL){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ // Set temporary table
+ NdbDictionary::Table tmpTab(* tab);
+ tmpTab.setStoredTable(_temp? 0 : 1);
+
+ int r = pNdb->getDictionary()->createTable(tmpTab);
+ if(r == -1){
+ if (existsOk && pNdb->getNdbError().code == 721)
+ ;
+ else {
+ return NDBT_FAILED;
+ }
+ }
+ }
+ return NDBT_OK;
+}
+
+int
+NDBT_Tables::createAllTables(Ndb* pNdb){
+ return createAllTables(pNdb, false);
+}
+
+int
+NDBT_Tables::createTable(Ndb* pNdb, const char* _name, bool _temp){
+
+ const NdbDictionary::Table* tab = NDBT_Tables::getTable(_name);
+ if (tab == NULL){
+ ndbout << "Could not create table " << _name
+ << ", it doesn't exist in list of tables "\
+ "that NDBT_Tables can create!" << endl;
+ return NDBT_WRONGARGS;
+ }
+
+ NdbDictionary::Table tmpTab(* tab);
+ tmpTab.setStoredTable(_temp ? 0 : 1);
+
+ int r = pNdb->getDictionary()->createTable(tmpTab);
+ return r;
+}
+
+int
+NDBT_Tables::dropAllTables(Ndb* pNdb){
+
+ for (int i=0; i < NDBT_Tables::getNumTables(); i++){
+
+ const NdbDictionary::Table* tab = NDBT_Tables::getTable(i);
+ if (tab == NULL){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ if(pNdb->getDictionary()->dropTable(tab->getName()) == -1){
+ return NDBT_FAILED;
+ }
+ }
+ return NDBT_OK;
+}
+
+
+int
+NDBT_Tables::print(const char * _name){
+
+ const NdbDictionary::Table* tab = NDBT_Tables::getTable(_name);
+ if (tab == NULL){
+ ndbout << "Could not print table " << _name
+ << ", it doesn't exist in list of tables "
+ << "that NDBT_Tables can create!" << endl;
+ return NDBT_WRONGARGS;
+ }
+ ndbout << (* tab) << endl;
+ return NDBT_OK;
+}
+
+int
+NDBT_Tables::printAll(){
+
+ for (int i=0; i < getNumTables(); i++){
+
+ const NdbDictionary::Table* tab = getTable(i);
+ if (tab == NULL){
+ abort();
+ }
+ ndbout << (* tab) << endl;
+ }
+
+ return NDBT_OK;
+}
diff --git a/ndb/test/src/NDBT_Test.cpp b/ndb/test/src/NDBT_Test.cpp
new file mode 100644
index 00000000000..e56e699af76
--- /dev/null
+++ b/ndb/test/src/NDBT_Test.cpp
@@ -0,0 +1,1119 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NDBT.hpp"
+#include "NDBT_Test.hpp"
+
+#include <PortDefs.h>
+
+#include <getarg.h>
+#include <time.h>
+
+// No verbose outxput
+
+NDBT_Context::NDBT_Context(){
+ tab = NULL;
+ suite = NULL;
+ testcase = NULL;
+ ndb = NULL;
+ records = 1;
+ loops = 1;
+ stopped = false;
+ remote_mgm ="";
+ propertyMutexPtr = NdbMutex_Create();
+ propertyCondPtr = NdbCondition_Create();
+}
+
+
+char * NDBT_Context::getRemoteMgm() const {
+ return remote_mgm;
+}
+void NDBT_Context::setRemoteMgm(char * mgm) {
+ remote_mgm = strdup(mgm);
+}
+
+
+NDBT_Context::~NDBT_Context(){
+ NdbCondition_Destroy(propertyCondPtr);
+ NdbMutex_Destroy(propertyMutexPtr);
+}
+
+const NdbDictionary::Table* NDBT_Context::getTab(){
+ assert(tab != NULL);
+ return tab;
+}
+
+NDBT_TestSuite* NDBT_Context::getSuite(){
+ assert(suite != NULL);
+ return suite;
+}
+
+NDBT_TestCase* NDBT_Context::getCase(){
+ assert(testcase != NULL);
+ return testcase;
+}
+
+int NDBT_Context::getNumRecords() const{
+ return records;
+}
+
+int NDBT_Context::getNumLoops() const{
+ return loops;
+}
+
+int NDBT_Context::getNoOfRunningSteps() const {
+ return testcase->getNoOfRunningSteps();
+
+}
+int NDBT_Context::getNoOfCompletedSteps() const {
+ return testcase->getNoOfCompletedSteps();
+}
+
+
+Uint32 NDBT_Context::getProperty(const char* _name, Uint32 _default){
+ Uint32 val;
+ NdbMutex_Lock(propertyMutexPtr);
+ if(!props.get(_name, &val))
+ val = _default;
+ NdbMutex_Unlock(propertyMutexPtr);
+ return val;
+}
+
+bool NDBT_Context::getPropertyWait(const char* _name, Uint32 _waitVal){
+ bool result;
+ NdbMutex_Lock(propertyMutexPtr);
+ Uint32 val =! _waitVal;
+
+ while((!props.get(_name, &val) || (props.get(_name, &val) && val != _waitVal)) &&
+ !stopped)
+ NdbCondition_Wait(propertyCondPtr,
+ propertyMutexPtr);
+ result = (val == _waitVal);
+ NdbMutex_Unlock(propertyMutexPtr);
+ return stopped;
+}
+
+const char* NDBT_Context::getProperty(const char* _name, const char* _default){
+ const char* val;
+ NdbMutex_Lock(propertyMutexPtr);
+ if(!props.get(_name, &val))
+ val = _default;
+ NdbMutex_Unlock(propertyMutexPtr);
+ return val;
+}
+
+const char* NDBT_Context::getPropertyWait(const char* _name, const char* _waitVal){
+ const char* val;
+ NdbMutex_Lock(propertyMutexPtr);
+ while(!props.get(_name, &val) && (strcmp(val, _waitVal)==0))
+ NdbCondition_Wait(propertyCondPtr,
+ propertyMutexPtr);
+
+ NdbMutex_Unlock(propertyMutexPtr);
+ return val;
+}
+
+void NDBT_Context::setProperty(const char* _name, Uint32 _val){
+ NdbMutex_Lock(propertyMutexPtr);
+ const bool b = props.put(_name, _val, true);
+ assert(b == true);
+ NdbMutex_Unlock(propertyMutexPtr);
+}
+
+void NDBT_Context::setProperty(const char* _name, const char* _val){
+ NdbMutex_Lock(propertyMutexPtr);
+ const bool b = props.put(_name, _val);
+ assert(b == true);
+ NdbMutex_Unlock(propertyMutexPtr);
+}
+
+void NDBT_Context::stopTest(){
+ NdbMutex_Lock(propertyMutexPtr);
+ g_info << "|- stopTest called" << endl;
+ stopped = true;
+ NdbCondition_Broadcast(propertyCondPtr);
+ NdbMutex_Unlock(propertyMutexPtr);
+}
+
+bool NDBT_Context::isTestStopped(){
+ NdbMutex_Lock(propertyMutexPtr);
+ bool val = stopped;
+ NdbMutex_Unlock(propertyMutexPtr);
+ return val;
+}
+
+void NDBT_Context::wait(){
+ NdbMutex_Lock(propertyMutexPtr);
+ NdbCondition_Wait(propertyCondPtr,
+ propertyMutexPtr);
+ NdbMutex_Unlock(propertyMutexPtr);
+}
+
+void NDBT_Context::wait_timeout(int msec){
+ NdbMutex_Lock(propertyMutexPtr);
+ NdbCondition_WaitTimeout(propertyCondPtr,
+ propertyMutexPtr,
+ msec);
+ NdbMutex_Unlock(propertyMutexPtr);
+}
+
+void NDBT_Context::broadcast(){
+ NdbMutex_Lock(propertyMutexPtr);
+ NdbCondition_Broadcast(propertyCondPtr);
+ NdbMutex_Unlock(propertyMutexPtr);
+}
+
+Uint32 NDBT_Context::getDbProperty(const char*){
+ abort();
+ return 0;
+}
+
+bool NDBT_Context::setDbProperty(const char*, Uint32){
+ abort();
+ return true;
+}
+
+void NDBT_Context::setTab(const NdbDictionary::Table* ptab){
+ assert(ptab != NULL);
+ tab = ptab;
+}
+
+void NDBT_Context::setSuite(NDBT_TestSuite* psuite){
+ assert(psuite != NULL);
+ suite = psuite;
+}
+
+void NDBT_Context::setCase(NDBT_TestCase* pcase){
+ assert(pcase != NULL);
+ testcase = pcase;
+}
+
+void NDBT_Context::setNumRecords(int _records){
+ records = _records;
+
+}
+
+void NDBT_Context::setNumLoops(int _loops){
+ loops = _loops;
+}
+
+NDBT_Step::NDBT_Step(NDBT_TestCase* ptest, const char* pname,
+ NDBT_TESTFUNC* pfunc): name(pname){
+ assert(pfunc != NULL);
+ func = pfunc;
+ testcase = ptest;
+ step_no = -1;
+}
+
+int NDBT_Step::execute(NDBT_Context* ctx) {
+ assert(ctx != NULL);
+
+ int result;
+
+ g_info << " |- " << name << " started [" << ctx->suite->getDate() << "]"
+ << endl;
+
+ result = setUp();
+ if (result != NDBT_OK){
+ return result;
+ }
+
+ result = func(ctx, this);
+
+ if (result != NDBT_OK) {
+ g_err << " |- " << name << " FAILED [" << ctx->suite->getDate()
+ << "]" << endl;
+ }
+ else {
+ g_info << " |- " << name << " PASSED [" << ctx->suite->getDate() << "]"
+ << endl;
+ }
+
+ tearDown();
+
+ return result;
+}
+
+void NDBT_Step::setContext(NDBT_Context* pctx){
+ assert(pctx != NULL);
+ m_ctx = pctx;
+}
+
+NDBT_Context* NDBT_Step::getContext(){
+ assert(m_ctx != NULL);
+ return m_ctx;
+}
+
+NDBT_NdbApiStep::NDBT_NdbApiStep(NDBT_TestCase* ptest,
+ const char* pname,
+ NDBT_TESTFUNC* pfunc)
+ : NDBT_Step(ptest, pname, pfunc),
+ ndb(NULL) {
+}
+
+
+int
+NDBT_NdbApiStep::setUp(){
+ ndb = new Ndb( "TEST_DB" );
+ ndb->init(1024);
+
+ int result = ndb->waitUntilReady(300); // 5 minutes
+ if (result != 0){
+ g_err << name << ": Ndb was not ready" << endl;
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+void
+NDBT_NdbApiStep::tearDown(){
+ delete ndb;
+ ndb = NULL;
+}
+
+Ndb* NDBT_NdbApiStep::getNdb(){
+ assert(ndb != NULL);
+ return ndb;
+}
+
+
+NDBT_ParallelStep::NDBT_ParallelStep(NDBT_TestCase* ptest,
+ const char* pname,
+ NDBT_TESTFUNC* pfunc)
+ : NDBT_NdbApiStep(ptest, pname, pfunc) {
+}
+NDBT_Verifier::NDBT_Verifier(NDBT_TestCase* ptest,
+ const char* pname,
+ NDBT_TESTFUNC* pfunc)
+ : NDBT_NdbApiStep(ptest, pname, pfunc) {
+}
+NDBT_Initializer::NDBT_Initializer(NDBT_TestCase* ptest,
+ const char* pname,
+ NDBT_TESTFUNC* pfunc)
+ : NDBT_NdbApiStep(ptest, pname, pfunc) {
+}
+NDBT_Finalizer::NDBT_Finalizer(NDBT_TestCase* ptest,
+ const char* pname,
+ NDBT_TESTFUNC* pfunc)
+ : NDBT_NdbApiStep(ptest, pname, pfunc) {
+}
+
+NDBT_TestCase::NDBT_TestCase(NDBT_TestSuite* psuite,
+ const char* pname,
+ const char* pcomment) :
+ name(pname) ,
+ comment(pcomment),
+ suite(psuite){
+ assert(suite != NULL);
+}
+
+
+NDBT_TestCaseImpl1::NDBT_TestCaseImpl1(NDBT_TestSuite* psuite,
+ const char* pname,
+ const char* pcomment) :
+ NDBT_TestCase(psuite, pname, pcomment){
+
+ numStepsOk = 0;
+ numStepsFail = 0;
+ numStepsCompleted = 0;
+ waitThreadsMutexPtr = NdbMutex_Create();
+ waitThreadsCondPtr = NdbCondition_Create();
+}
+
+NDBT_TestCaseImpl1::~NDBT_TestCaseImpl1(){
+ NdbCondition_Destroy(waitThreadsCondPtr);
+ NdbMutex_Destroy(waitThreadsMutexPtr);
+
+ for(size_t i = 0; i < initializers.size(); i++)
+ delete initializers[i];
+ initializers.clear();
+ for(size_t i = 0; i < verifiers.size(); i++)
+ delete verifiers[i];
+ verifiers.clear();
+ for(size_t i = 0; i < finalizers.size(); i++)
+ delete finalizers[i];
+ finalizers.clear();
+ for(size_t i = 0; i < steps.size(); i++)
+ delete steps[i];
+ steps.clear();
+ results.clear();
+ for(size_t i = 0; i < testTables.size(); i++)
+ delete testTables[i];
+ testTables.clear();
+ for(size_t i = 0; i < testResults.size(); i++)
+ delete testResults[i];
+ testResults.clear();
+
+}
+
+int NDBT_TestCaseImpl1::addStep(NDBT_Step* pStep){
+ assert(pStep != NULL);
+ steps.push_back(pStep);
+ pStep->setStepNo(steps.size());
+ int res = NORESULT;
+ results.push_back(res);
+ return 0;
+}
+
+int NDBT_TestCaseImpl1::addVerifier(NDBT_Verifier* pVerifier){
+ assert(pVerifier != NULL);
+ verifiers.push_back(pVerifier);
+ return 0;
+}
+
+int NDBT_TestCaseImpl1::addInitializer(NDBT_Initializer* pInitializer){
+ assert(pInitializer != NULL);
+ initializers.push_back(pInitializer);
+ return 0;
+}
+
+int NDBT_TestCaseImpl1::addFinalizer(NDBT_Finalizer* pFinalizer){
+ assert(pFinalizer != NULL);
+ finalizers.push_back(pFinalizer);
+ return 0;
+}
+
+void NDBT_TestCaseImpl1::addTable(const char* tableName, bool isVerify) {
+ assert(tableName != NULL);
+ const NdbDictionary::Table* pTable = NDBT_Tables::getTable(tableName);
+ assert(pTable != NULL);
+ testTables.push_back(pTable);
+ isVerifyTables = isVerify;
+}
+
+bool NDBT_TestCaseImpl1::tableExists(NdbDictionary::Table* aTable) {
+ for (unsigned i = 0; i < testTables.size(); i++) {
+ if (strcasecmp(testTables[i]->getName(), aTable->getName()) == 0) {
+ return true;
+ }
+ }
+ return false;
+}
+
+bool NDBT_TestCaseImpl1::isVerify(const NdbDictionary::Table* aTable) {
+ if (testTables.size() > 0) {
+ int found = false;
+ // OK, we either exclude or include this table in the actual test
+ for (unsigned i = 0; i < testTables.size(); i++) {
+ if (strcasecmp(testTables[i]->getName(), aTable->getName()) == 0) {
+ // Found one!
+ if (isVerifyTables) {
+ // Found one to test
+ found = true;
+ } else {
+ // Skip this one!
+ found = false;
+ }
+ }
+ } // for
+ return found;
+ } else {
+ // No included or excluded test tables, i.e., all tables should be
+ // tested
+ return true;
+ }
+ return true;
+}
+
+void NDBT_TestCase::setProperty(const char* _name, Uint32 _val){
+ const bool b = props.put(_name, _val);
+ assert(b == true);
+}
+
+void NDBT_TestCase::setProperty(const char* _name, const char* _val){
+ const bool b = props.put(_name, _val);
+ assert(b == true);
+}
+
+
+void *
+runStep(void * s){
+ assert(s != NULL);
+ NDBT_Step* pStep = (NDBT_Step*)s;
+ NDBT_Context* ctx = pStep->getContext();
+ assert(ctx != NULL);
+ // Execute function
+ int res = pStep->execute(ctx);
+ if(res != NDBT_OK){
+ ctx->stopTest();
+ }
+ // Report
+ NDBT_TestCaseImpl1* pCase = (NDBT_TestCaseImpl1*)ctx->getCase();
+ assert(pCase != NULL);
+ pCase->reportStepResult(pStep, res);
+ return NULL;
+}
+
+extern "C"
+void *
+runStep_C(void * s)
+{
+ runStep(s);
+ NdbThread_Exit(0);
+ return NULL;
+}
+
+
+void NDBT_TestCaseImpl1::startStepInThread(int stepNo, NDBT_Context* ctx){
+ NDBT_Step* pStep = steps[stepNo];
+ pStep->setContext(ctx);
+ char buf[16];
+ snprintf(buf, sizeof(buf), "step_%d", stepNo);
+ NdbThread* pThread = NdbThread_Create(runStep_C,
+ (void**)pStep,
+ 65535,
+ buf,
+ NDB_THREAD_PRIO_LOW);
+ threads.push_back(pThread);
+}
+
+void NDBT_TestCaseImpl1::waitSteps(){
+ NdbMutex_Lock(waitThreadsMutexPtr);
+ while(numStepsCompleted != steps.size())
+ NdbCondition_Wait(waitThreadsCondPtr,
+ waitThreadsMutexPtr);
+
+ unsigned completedSteps = 0;
+ for(unsigned i=0; i<steps.size(); i++){
+ if (results[i] != NORESULT){
+ completedSteps++;
+ if (results[i] == NDBT_OK)
+ numStepsOk++;
+ else
+ numStepsFail++;
+ }
+ }
+ assert(completedSteps == steps.size());
+ assert(completedSteps == numStepsCompleted);
+
+ NdbMutex_Unlock(waitThreadsMutexPtr);
+ void *status;
+ for(unsigned i=0; i<steps.size();i++){
+ NdbThread_WaitFor(threads[i], &status);
+ NdbThread_Destroy(&threads[i]);
+ }
+}
+
+
+int
+NDBT_TestCaseImpl1::getNoOfRunningSteps() const {
+ return steps.size() - getNoOfCompletedSteps();
+}
+
+int
+NDBT_TestCaseImpl1::getNoOfCompletedSteps() const {
+ return numStepsCompleted;
+}
+
+void NDBT_TestCaseImpl1::reportStepResult(const NDBT_Step* pStep, int result){
+ NdbMutex_Lock(waitThreadsMutexPtr);
+ assert(pStep != NULL);
+ for (unsigned i = 0; i < steps.size(); i++){
+ if(steps[i] != NULL && steps[i] == pStep){
+ results[i] = result;
+ numStepsCompleted++;
+ }
+ }
+ if(numStepsCompleted == steps.size()){
+ NdbCondition_Signal(waitThreadsCondPtr);
+ }
+ NdbMutex_Unlock(waitThreadsMutexPtr);
+}
+
+
+int NDBT_TestCase::execute(NDBT_Context* ctx){
+ int res;
+
+ ndbout << "- " << name << " started [" << ctx->suite->getDate()
+ << "]" << endl;
+
+ ctx->setCase(this);
+
+ // Copy test case properties to ctx
+ Properties::Iterator it(&props);
+ for(const char * key = it.first(); key != 0; key = it.next()){
+
+ PropertiesType pt;
+ const bool b = props.getTypeOf(key, &pt);
+ assert(b == true);
+ switch(pt){
+ case PropertiesType_Uint32:{
+ Uint32 val;
+ props.get(key, &val);
+ ctx->setProperty(key, val);
+ break;
+ }
+ case PropertiesType_char:{
+ const char * val;
+ props.get(key, &val);
+ ctx->setProperty(key, val);
+ break;
+ }
+ default:
+ abort();
+ }
+ }
+
+ // start timer so that we get a time even if
+ // test case consist only of initializer
+ startTimer(ctx);
+
+ if ((res = runInit(ctx)) == NDBT_OK){
+ // If initialiser is ok, run steps
+
+ res = runSteps(ctx);
+ if (res == NDBT_OK){
+ // If steps is ok, run verifier
+ res = runVerifier(ctx);
+ }
+
+ }
+
+ stopTimer(ctx);
+ printTimer(ctx);
+
+ // Always run finalizer to clean up db
+ runFinal(ctx);
+
+ if (res == NDBT_OK) {
+ ndbout << "- " << name << " PASSED [" << ctx->suite->getDate() << "]"
+ << endl;
+ }
+ else {
+ ndbout << "- " << name << " FAILED [" << ctx->suite->getDate() << "]"
+ << endl;
+ }
+ return res;
+};
+
+
+void NDBT_TestCase::startTimer(NDBT_Context* ctx){
+ timer.doStart();
+}
+
+void NDBT_TestCase::stopTimer(NDBT_Context* ctx){
+ timer.doStop();
+}
+
+void NDBT_TestCase::printTimer(NDBT_Context* ctx){
+ if (suite->timerIsOn()){
+ g_info << endl;
+ timer.printTestTimer(ctx->getNumLoops(), ctx->getNumRecords());
+ }
+}
+
+int NDBT_TestCaseImpl1::runInit(NDBT_Context* ctx){
+ int res = NDBT_OK;
+ for (unsigned i = 0; i < initializers.size(); i++){
+ initializers[i]->setContext(ctx);
+ res = initializers[i]->execute(ctx);
+ if (res != NDBT_OK)
+ break;
+ }
+ return res;
+}
+
+int NDBT_TestCaseImpl1::runSteps(NDBT_Context* ctx){
+ int res = NDBT_OK;
+
+ // Reset variables
+ numStepsOk = 0;
+ numStepsFail = 0;
+ numStepsCompleted = 0;
+
+ for (unsigned i = 0; i < steps.size(); i++)
+ startStepInThread(i, ctx);
+ waitSteps();
+
+ for(unsigned i = 0; i < steps.size(); i++)
+ if (results[i] != NDBT_OK)
+ res = NDBT_FAILED;
+ return res;
+}
+
+int NDBT_TestCaseImpl1::runVerifier(NDBT_Context* ctx){
+ int res = NDBT_OK;
+ for (unsigned i = 0; i < verifiers.size(); i++){
+ verifiers[i]->setContext(ctx);
+ res = verifiers[i]->execute(ctx);
+ if (res != NDBT_OK)
+ break;
+ }
+ return res;
+}
+
+int NDBT_TestCaseImpl1::runFinal(NDBT_Context* ctx){
+ int res = NDBT_OK;
+ for (unsigned i = 0; i < finalizers.size(); i++){
+ finalizers[i]->setContext(ctx);
+ res = finalizers[i]->execute(ctx);
+ if (res != NDBT_OK)
+ break;
+ }
+ return res;
+}
+
+
+void NDBT_TestCaseImpl1::saveTestResult(const NdbDictionary::Table* ptab,
+ int result){
+ testResults.push_back(new NDBT_TestCaseResult(ptab->getName(),
+ result,
+ timer.elapsedTime()));
+}
+
+void NDBT_TestCaseImpl1::printTestResult(){
+
+ char buf[255];
+ ndbout << name<<endl;
+
+ for (unsigned i = 0; i < testResults.size(); i++){
+ NDBT_TestCaseResult* tcr = testResults[i];
+ const char* res;
+ if (tcr->getResult() == NDBT_OK)
+ res = "OK";
+ else if (tcr->getResult() == NDBT_FAILED)
+ res = "FAIL";
+ else if (tcr->getResult() == FAILED_TO_CREATE)
+ res = "FAILED TO CREATE TABLE";
+ else if (tcr->getResult() == FAILED_TO_DISCOVER)
+ res = "FAILED TO DISCOVER TABLE";
+ snprintf(buf, 255," %-10s %-5s %-20s", tcr->getName(), res, tcr->getTimeStr());
+ ndbout << buf<<endl;
+ }
+}
+
+
+
+
+
+NDBT_TestSuite::NDBT_TestSuite(const char* pname):name(pname){
+ numTestsOk = 0;
+ numTestsFail = 0;
+ numTestsExecuted = 0;
+ records = 0;
+ loops = 0;
+ createTable = true;
+}
+
+
+NDBT_TestSuite::~NDBT_TestSuite(){
+ for(unsigned i=0; i<tests.size(); i++){
+ delete tests[i];
+ }
+ tests.clear();
+}
+
+void NDBT_TestSuite::setCreateTable(bool _flag){
+ createTable = _flag;
+}
+
+bool NDBT_TestSuite::timerIsOn(){
+ return (timer != 0);
+}
+
+int NDBT_TestSuite::addTest(NDBT_TestCase* pTest){
+ assert(pTest != NULL);
+ tests.push_back(pTest);
+ return 0;
+}
+
+int NDBT_TestSuite::executeAll(const char* _testname){
+
+ if(tests.size() == 0)
+ return NDBT_FAILED;
+ Ndb ndb("TEST_DB");
+ ndb.init(1024);
+
+ int result = ndb.waitUntilReady(300); // 5 minutes
+ if (result != 0){
+ g_err << name <<": Ndb was not ready" << endl;
+ return NDBT_FAILED;
+ }
+
+ ndbout << name << " started [" << getDate() << "]" << endl;
+
+ testSuiteTimer.doStart();
+
+ for (int t=0; t < NDBT_Tables::getNumTables(); t++){
+ const NdbDictionary::Table* ptab = NDBT_Tables::getTable(t);
+ ndbout << "|- " << ptab->getName() << endl;
+ execute(&ndb, ptab, _testname);
+ }
+ testSuiteTimer.doStop();
+ return reportAllTables(_testname);
+}
+
+int
+NDBT_TestSuite::executeOne(const char* _tabname, const char* _testname){
+
+ if(tests.size() == 0)
+ return NDBT_FAILED;
+ Ndb ndb("TEST_DB");
+ ndb.init(1024);
+
+ int result = ndb.waitUntilReady(300); // 5 minutes
+ if (result != 0){
+ g_err << name <<": Ndb was not ready" << endl;
+ return NDBT_FAILED;
+ }
+
+ ndbout << name << " started [" << getDate() << "]" << endl;
+
+ const NdbDictionary::Table* ptab = NDBT_Tables::getTable(_tabname);
+ if (ptab == NULL)
+ return NDBT_FAILED;
+
+ ndbout << "|- " << ptab->getName() << endl;
+
+ execute(&ndb, ptab, _testname);
+
+ if (numTestsFail > 0){
+ return NDBT_FAILED;
+ }else{
+ return NDBT_OK;
+ }
+}
+
+void NDBT_TestSuite::execute(Ndb* ndb, const NdbDictionary::Table* pTab,
+ const char* _testname){
+ int result;
+
+
+ for (unsigned t = 0; t < tests.size(); t++){
+
+ if (_testname != NULL &&
+ strcasecmp(tests[t]->getName(), _testname) != 0)
+ continue;
+
+ if (tests[t]->isVerify(pTab) == false) {
+ continue;
+ }
+
+ tests[t]->initBeforeTest();
+
+ NdbDictionary::Dictionary* pDict = ndb->getDictionary();
+ const NdbDictionary::Table* pTab2 = pDict->getTable(pTab->getName());
+ if (createTable == true){
+
+ if (pTab2 != 0 && !pTab->equal(* pTab2)){
+ numTestsFail++;
+ numTestsExecuted++;
+ g_err << "ERROR0: Failed to create table " << pTab->getName() << endl;
+ tests[t]->saveTestResult(pTab, FAILED_TO_CREATE);
+ continue;
+ }
+
+ if(pTab2 == 0 && pDict->createTable(* pTab) != 0){
+ numTestsFail++;
+ numTestsExecuted++;
+ g_err << "ERROR1: Failed to create table " << pTab->getName() << endl;
+ tests[t]->saveTestResult(pTab, FAILED_TO_CREATE);
+ continue;
+ }
+ pTab2 = pDict->getTable(pTab->getName());
+ }
+
+ ctx = new NDBT_Context();
+ ctx->setTab(pTab2);
+ ctx->setNumRecords(records);
+ ctx->setNumLoops(loops);
+ if(remote_mgm != NULL)
+ ctx->setRemoteMgm(remote_mgm);
+ ctx->setSuite(this);
+
+ result = tests[t]->execute(ctx);
+ tests[t]->saveTestResult(pTab, result);
+ if (result != NDBT_OK)
+ numTestsFail++;
+ else
+ numTestsOk++;
+ numTestsExecuted++;
+ delete ctx;
+ }
+}
+
+
+
+
+int
+NDBT_TestSuite::report(const char* _tcname){
+ int result;
+ ndbout << "Completed " << name << " [" << getDate() << "]" << endl;
+ printTestCaseSummary(_tcname);
+ ndbout << numTestsExecuted << " test(s) executed" << endl;
+ ndbout << numTestsOk << " test(s) OK"
+ << endl;
+ if(numTestsFail > 0)
+ ndbout << numTestsFail << " test(s) failed"
+ << endl;
+ testSuiteTimer.printTotalTime();
+ if (numTestsFail > 0 || numTestsExecuted == 0){
+ result = NDBT_FAILED;
+ }else{
+ result = NDBT_OK;
+ }
+ return result;
+}
+
+void NDBT_TestSuite::printTestCaseSummary(const char* _tcname){
+ ndbout << "= SUMMARY OF TEST EXECUTION ==============" << endl;
+ for (unsigned t = 0; t < tests.size(); t++){
+ if (_tcname != NULL &&
+ strcasecmp(tests[t]->getName(), _tcname) != 0)
+ continue;
+
+ tests[t]->printTestResult();
+ }
+ ndbout << "==========================================" << endl;
+}
+
+int NDBT_TestSuite::reportAllTables(const char* _testname){
+ int result;
+ ndbout << "Completed running test [" << getDate() << "]" << endl;
+ const int totalNumTests = numTestsExecuted;
+ printTestCaseSummary(_testname);
+ ndbout << numTestsExecuted<< " test(s) executed" << endl;
+ ndbout << numTestsOk << " test(s) OK("
+ <<(int)(((float)numTestsOk/totalNumTests)*100.0) <<"%)"
+ << endl;
+ if(numTestsFail > 0)
+ ndbout << numTestsFail << " test(s) failed("
+ <<(int)(((float)numTestsFail/totalNumTests)*100.0) <<"%)"
+ << endl;
+ testSuiteTimer.printTotalTime();
+ if (numTestsExecuted > 0){
+ if (numTestsFail > 0){
+ result = NDBT_FAILED;
+ }else{
+ result = NDBT_OK;
+ }
+ } else {
+ result = NDBT_FAILED;
+ }
+ return result;
+}
+
+int NDBT_TestSuite::execute(int argc, const char** argv){
+ int res = NDBT_FAILED;
+ /* Arguments:
+ Run only a subset of tests
+ -n testname Which test to run
+ Recommendations to test functions:
+ --records Number of records to use(default: 10000)
+ --loops Number of loops to execute in the test(default: 100)
+
+ Other parameters should:
+ * be calculated from the above two parameters
+ * be divided into different test cases, ex. one testcase runs
+ with FragmentType = Single and another perfoms the same
+ test with FragmentType = Large
+ * let the test case iterate over all/subset of appropriate parameters
+ ex. iterate over FragmentType = Single to FragmentType = AllLarge
+
+ Remeber that the intention is that it should be _easy_ to run
+ a complete test suite without any greater knowledge of what
+ should be tested ie. keep arguments at a minimum
+ */
+ int _records = 1000;
+ int _loops = 5;
+ int _timer = 0;
+ char * _remote_mgm =NULL;
+ char* _testname = NULL;
+ const char* _tabname = NULL;
+ int _print = false;
+ int _print_html = false;
+
+ int _print_cases = false;
+ int _verbose = false;
+
+ struct getargs args[] = {
+ { "print", '\0', arg_flag, &_print, "Print execution tree", "" },
+ { "print_html", '\0', arg_flag, &_print_html, "Print execution tree in html table format", "" },
+ { "print_cases", '\0', arg_flag, &_print_cases, "Print list of test cases", "" },
+ { "records", 'r', arg_integer, &_records, "Number of records", "records" },
+ { "loops", 'l', arg_integer, &_loops, "Number of loops", "loops" },
+ { "testname", 'n', arg_string, &_testname, "Name of test to run", "testname" },
+ { "remote_mgm", 'm', arg_string, &_remote_mgm,
+ "host:port to mgmsrv of remote cluster", "host:port" },
+ { "timer", 't', arg_flag, &_timer, "Print execution time", "time" },
+ { "verbose", 'v', arg_flag, &_verbose, "Print verbose status", "verbose" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+
+ if(getarg(args, num_args, argc, argv, &optind)) {
+ arg_printusage(args, num_args, argv[0], "tabname1 tabname2 ... tabnameN\n");
+ return NDBT_WRONGARGS;
+ }
+ // Check if table name is supplied
+ if (argv[optind] != NULL)
+ _tabname = argv[optind];
+
+ if (_print == true){
+ printExecutionTree();
+ return 0;
+ }
+
+ if (_print_html == true){
+ printExecutionTreeHTML();
+ return 0;
+ }
+
+ if (_print_cases == true){
+ printCases();
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ if (_verbose)
+ setOutputLevel(2); // Show g_info
+ else
+ setOutputLevel(0); // Show only g_err ?
+
+ remote_mgm = _remote_mgm;
+ records = _records;
+ loops = _loops;
+ timer = _timer;
+
+ if(optind == argc){
+ // No table specified
+ res = executeAll(_testname);
+ } else {
+ testSuiteTimer.doStart();
+ for(int i = optind; i<argc; i++){
+ executeOne(argv[i], _testname);
+ }
+ testSuiteTimer.doStop();
+ res = report(_testname);
+ }
+
+ return NDBT_ProgramExit(res);
+}
+
+
+
+void NDBT_TestSuite::printExecutionTree(){
+ ndbout << "Testsuite: " << name << endl;
+ for (unsigned t = 0; t < tests.size(); t++){
+ tests[t]->print();
+ ndbout << endl;
+ }
+}
+
+void NDBT_TestSuite::printExecutionTreeHTML(){
+ ndbout << "<tr>" << endl;
+ ndbout << "<td><h3>" << name << "</h3></td>" << endl;
+ ndbout << "</tr>" << endl;
+ for (unsigned t = 0; t < tests.size(); t++){
+ tests[t]->printHTML();
+ ndbout << endl;
+ }
+
+}
+
+void NDBT_TestSuite::printCases(){
+ ndbout << "# Testsuite: " << name << endl;
+ ndbout << "# Number of tests: " << tests.size() << endl;
+ for (unsigned t = 0; t < tests.size(); t++){
+ ndbout << name << " -n " << tests[t]->getName() << endl;
+ }
+}
+
+const char* NDBT_TestSuite::getDate(){
+ static char theTime[128];
+ struct tm* tm_now;
+ time_t now;
+ now = time((time_t*)NULL);
+#ifdef NDB_WIN32
+ tm_now = localtime(&now);
+#else
+ tm_now = gmtime(&now);
+#endif
+
+ snprintf(theTime, 128,
+ "%d-%.2d-%.2d %.2d:%.2d:%.2d",
+ tm_now->tm_year + 1900,
+ tm_now->tm_mon + 1,
+ tm_now->tm_mday,
+ tm_now->tm_hour,
+ tm_now->tm_min,
+ tm_now->tm_sec);
+
+ return theTime;
+}
+
+void NDBT_TestCaseImpl1::printHTML(){
+
+ ndbout << "<tr><td>&nbsp;</td>" << endl;
+ ndbout << "<td name=tc>" << endl << name << "</td><td width=70%>"
+ << comment << "</td></tr>" << endl;
+}
+
+void NDBT_TestCaseImpl1::print(){
+ ndbout << "Test case: " << name << endl;
+ ndbout << "Description: "<< comment << endl;
+
+ ndbout << "Parameters: " << endl;
+
+ Properties::Iterator it(&props);
+ for(const char * key = it.first(); key != 0; key = it.next()){
+ PropertiesType pt;
+ const bool b = props.getTypeOf(key, &pt);
+ assert(b == true);
+ switch(pt){
+ case PropertiesType_Uint32:{
+ Uint32 val;
+ props.get(key, &val);
+ ndbout << " " << key << ": " << val << endl;
+ break;
+ }
+ case PropertiesType_char:{
+ const char * val;
+ props.get(key, &val);
+ ndbout << " " << key << ": " << val << endl;
+ break;
+ }
+ default:
+ abort();
+ }
+ }
+
+ for(unsigned i=0; i<initializers.size(); i++){
+ ndbout << "Initializers[" << i << "]: " << endl;
+ initializers[i]->print();
+ }
+ for(unsigned i=0; i<steps.size(); i++){
+ ndbout << "Step[" << i << "]: " << endl;
+ steps[i]->print();
+ }
+ for(unsigned i=0; i<verifiers.size(); i++){
+ ndbout << "Verifier[" << i << "]: " << endl;
+ verifiers[i]->print();
+ }
+ for(unsigned i=0; i<finalizers.size(); i++){
+ ndbout << "Finalizer[" << i << "]: " << endl;
+ finalizers[i]->print();
+ }
+
+}
+
+void NDBT_Step::print(){
+ ndbout << " "<< name << endl;
+
+}
+
+
+
+
diff --git a/ndb/test/src/NdbBackup.cpp b/ndb/test/src/NdbBackup.cpp
new file mode 100644
index 00000000000..689aae64c81
--- /dev/null
+++ b/ndb/test/src/NdbBackup.cpp
@@ -0,0 +1,452 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/DumpStateOrd.hpp>
+#include <NdbBackup.hpp>
+#include <NdbOut.hpp>
+#include <NDBT_Output.hpp>
+#include <assert.h>
+#include <NdbConfig.h>
+#include <ConfigRetriever.hpp>
+#include <ndb_version.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <random.h>
+#include <NdbTick.h>
+
+#define CHECK(b, m) { int _xx = b; if (!(_xx)) { \
+ ndbout << "ERR: "<< m \
+ << " " << "File: " << __FILE__ \
+ << " (Line: " << __LINE__ << ")" << "- " << _xx << endl; \
+ return NDBT_FAILED; } }
+
+
+int
+NdbBackup::start(unsigned int & _backup_id){
+
+
+ if (!isConnected())
+ return -1;
+
+ ndb_mgm_reply reply;
+ reply.return_code = 0;
+
+ if (ndb_mgm_start_backup(handle,
+ &_backup_id,
+ &reply) == -1) {
+ g_err << "Could not start backup " << endl;
+ g_err << "Error: " << reply.message << endl;
+ return -1;
+ }
+
+ if(reply.return_code != 0){
+ g_err << "PLEASE CHECK CODE NdbBackup.cpp line=" << __LINE__ << endl;
+ g_err << "Error: " << reply.message << endl;
+ return reply.return_code;
+ }
+ return 0;
+}
+
+
+const char *
+NdbBackup::getFileSystemPathForNode(int _node_id){
+
+ /**
+ * Fetch configuration from management server
+ */
+ char buf[255];
+ ConfigRetriever cr;
+
+
+ Properties * p = cr.getConfig(host,
+ port,
+ _node_id,
+ NDB_VERSION);
+ if(p == 0){
+ const char * s = cr.getErrorString();
+ if(s == 0)
+ s = "No error given!";
+
+ ndbout << "Could not fetch configuration" << endl;
+ ndbout << s << endl;
+ return NULL;
+ }
+
+ /**
+ * Setup cluster configuration data
+ */
+ const Properties * db = 0;
+ if (!p->get("Node", _node_id, &db)) {
+ ndbout << "Invalid configuration fetched, DB missing" << endl;
+ return NULL;
+ }
+ const char * type;
+ if(!(db->get("Type", &type) && strcmp(type, "DB") == 0)){
+ ndbout <<"Invalid configuration fetched, I'm wrong type of node" << endl;
+ return NULL;
+ }
+
+ const char * path;
+ if (!db->get("FileSystemPath", &path)){
+ ndbout << "FileSystemPath not found" << endl;
+ return NULL;
+ }
+
+ return path;
+
+}
+
+int
+NdbBackup::execRestore(bool _restore_data,
+ bool _restore_meta,
+ int _node_id,
+ unsigned _backup_id){
+ const int buf_len = 1000;
+ char buf[buf_len];
+
+ const char* path = getFileSystemPathForNode(_node_id);
+ if (path == NULL)
+ return -1;
+
+ const char *host;
+ if (!getHostName(_node_id, &host)){
+ return -1;
+ }
+
+ /*
+ * Copy backup files to local dir
+ */
+
+ snprintf(buf, buf_len,
+ "scp %s:%s/BACKUP/BACKUP-%d/* .",
+ host, path,
+ _backup_id);
+
+ ndbout << "buf: "<< buf <<endl;
+ int res = system(buf);
+
+ ndbout << "res: " << res << endl;
+
+#if 0
+ snprintf(buf, 255, "restore -c \"nodeid=%d;host=%s\" -n %d -b %d %s %s %s/BACKUP/BACKUP-%d",
+ ownNodeId,
+ addr,
+ _node_id,
+ _backup_id,
+ _restore_data?"-r":"",
+ _restore_meta?"-m":"",
+ path,
+ _backup_id);
+
+#endif
+
+ snprintf(buf, 255, "restore -c \"nodeid=%d;host=%s\" -n %d -b %d %s %s .",
+ ownNodeId,
+ addr,
+ _node_id,
+ _backup_id,
+ _restore_data?"-r":"",
+ _restore_meta?"-m":"");
+
+ // path,
+ // _backup_id);
+
+
+ ndbout << "buf: "<< buf <<endl;
+ res = system(buf);
+
+ ndbout << "res: " << res << endl;
+
+ return res;
+
+}
+
+int
+NdbBackup::restore(unsigned _backup_id){
+
+ if (!isConnected())
+ return -1;
+
+ if (getStatus() != 0)
+ return -1;
+
+ int res;
+ if ( ndbNodes.size() == 1) {
+ // restore metadata and data in one call
+ res = execRestore(true, true, ndbNodes[0].node_id, _backup_id);
+ } else {
+ assert(ndbNodes.size() > 1);
+
+ // restore metadata first
+ res = execRestore(false, true, ndbNodes[0].node_id, _backup_id);
+
+
+ // Restore data once for each node
+ for(size_t i = 0; i < ndbNodes.size(); i++){
+ res = execRestore(true, false, ndbNodes[i].node_id, _backup_id);
+ }
+ }
+
+ return 0;
+}
+
+// Master failure
+int
+NFDuringBackupM_codes[] = {
+ 10003,
+ 10004,
+ 10005,
+ 10007,
+ 10008,
+ 10009,
+ 10010,
+ 10012,
+ 10013
+};
+
+// Slave failure
+int
+NFDuringBackupS_codes[] = {
+ 10014,
+ 10015,
+ 10016,
+ 10017,
+ 10018,
+ 10020
+};
+
+// Master takeover etc...
+int
+NFDuringBackupSL_codes[] = {
+ 10001,
+ 10002,
+ 10021
+};
+
+int
+NdbBackup::NFMaster(NdbRestarter& _restarter){
+ const int sz = sizeof(NFDuringBackupM_codes)/sizeof(NFDuringBackupM_codes[0]);
+ return NF(_restarter, NFDuringBackupM_codes, sz, true);
+}
+
+int
+NdbBackup::NFMasterAsSlave(NdbRestarter& _restarter){
+ const int sz = sizeof(NFDuringBackupS_codes)/sizeof(NFDuringBackupS_codes[0]);
+ return NF(_restarter, NFDuringBackupS_codes, sz, true);
+}
+
+int
+NdbBackup::NFSlave(NdbRestarter& _restarter){
+ const int sz = sizeof(NFDuringBackupS_codes)/sizeof(NFDuringBackupS_codes[0]);
+ return NF(_restarter, NFDuringBackupS_codes, sz, false);
+}
+
+int
+NdbBackup::NF(NdbRestarter& _restarter, int *NFDuringBackup_codes, const int sz, bool onMaster){
+ {
+ int nodeId = _restarter.getMasterNodeId();
+
+ CHECK(_restarter.restartOneDbNode(nodeId, false, true, true) == 0,
+ "Could not restart node "<< nodeId);
+
+ CHECK(_restarter.waitNodesNoStart(&nodeId, 1) == 0,
+ "waitNodesNoStart failed");
+
+ CHECK(_restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ NdbSleep_SecSleep(10);
+ }
+
+ CHECK(_restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+
+ int nNodes = _restarter.getNumDbNodes();
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+ for(int i = 0; i<sz; i++){
+
+ int error = NFDuringBackup_codes[i];
+ unsigned int backupId;
+
+ const int masterNodeId = _restarter.getMasterNodeId();
+ CHECK(masterNodeId > 0, "getMasterNodeId failed");
+ int nodeId;
+
+ nodeId = masterNodeId;
+ if (!onMaster) {
+ int randomId;
+ while (nodeId == masterNodeId) {
+ randomId = myRandom48(nNodes);
+ nodeId = _restarter.getDbNodeId(randomId);
+ }
+ }
+
+ g_err << "NdbBackup::NF node = " << nodeId
+ << " error code = " << error << " masterNodeId = "
+ << masterNodeId << endl;
+
+
+ int val = DumpStateOrd::CmvmiSetRestartOnErrorInsert;
+ CHECK(_restarter.dumpStateOneNode(nodeId, &val, 1) == 0,
+ "failed to set RestartOnErrorInsert");
+ CHECK(_restarter.insertErrorInNode(nodeId, error) == 0,
+ "failed to set error insert");
+
+ g_info << "error inserted" << endl;
+
+ g_info << "starting backup" << endl;
+ int r = start(backupId);
+ g_info << "r = " << r
+ << " (which should fail) started with id = " << backupId << endl;
+ if (r == 0) {
+ g_err << "Backup should have failed on error_insertion " << error << endl
+ << "Master = " << masterNodeId << "Node = " << nodeId << endl;
+ }
+
+ CHECK(_restarter.waitNodesNoStart(&nodeId, 1) == 0,
+ "waitNodesNoStart failed");
+
+ g_info << "number of nodes running " << _restarter.getNumDbNodes() << endl;
+
+ if (_restarter.getNumDbNodes() != nNodes) {
+ g_err << "Failure: cluster not up" << endl;
+ return NDBT_FAILED;
+ }
+
+ NdbSleep_SecSleep(1);
+
+ g_info << "starting new backup" << endl;
+ CHECK(start(backupId) == 0,
+ "failed to start backup");
+ g_info << "(which should succeed) started with id = " << backupId << endl;
+
+ g_info << "starting node" << endl;
+ CHECK(_restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ CHECK(_restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+ g_info << "node started" << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 10099) == 0,
+ "failed to set error insert");
+ }
+
+ return NDBT_OK;
+};
+
+int
+FailS_codes[] = {
+ 10023,
+ 10024,
+ 10025,
+ 10026,
+ 10027,
+ 10028,
+ 10029,
+ 10030,
+ 10031
+};
+
+int
+FailM_codes[] = {
+ 10023,
+ 10024,
+ 10025,
+ 10026,
+ 10027,
+ 10028,
+ 10029,
+ 10030,
+ 10031
+};
+
+int
+NdbBackup::FailMaster(NdbRestarter& _restarter){
+ const int sz = sizeof(FailM_codes)/sizeof(FailM_codes[0]);
+ return Fail(_restarter, FailM_codes, sz, true);
+}
+
+int
+NdbBackup::FailMasterAsSlave(NdbRestarter& _restarter){
+ const int sz = sizeof(FailS_codes)/sizeof(FailS_codes[0]);
+ return Fail(_restarter, FailS_codes, sz, true);
+}
+
+int
+NdbBackup::FailSlave(NdbRestarter& _restarter){
+ const int sz = sizeof(FailS_codes)/sizeof(FailS_codes[0]);
+ return Fail(_restarter, FailS_codes, sz, false);
+}
+
+int
+NdbBackup::Fail(NdbRestarter& _restarter, int *Fail_codes, const int sz, bool onMaster){
+
+ CHECK(_restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+
+ int nNodes = _restarter.getNumDbNodes();
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+ for(int i = 0; i<sz; i++){
+ int error = Fail_codes[i];
+ unsigned int backupId;
+
+ const int masterNodeId = _restarter.getMasterNodeId();
+ CHECK(masterNodeId > 0, "getMasterNodeId failed");
+ int nodeId;
+
+ nodeId = masterNodeId;
+ if (!onMaster) {
+ int randomId;
+ while (nodeId == masterNodeId) {
+ randomId = myRandom48(nNodes);
+ nodeId = _restarter.getDbNodeId(randomId);
+ }
+ }
+
+ g_err << "NdbBackup::Fail node = " << nodeId
+ << " error code = " << error << " masterNodeId = "
+ << masterNodeId << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, error) == 0,
+ "failed to set error insert");
+
+ g_info << "error inserted" << endl;
+ g_info << "waiting some before starting backup" << endl;
+
+ g_info << "starting backup" << endl;
+ int r = start(backupId);
+ g_info << "r = " << r
+ << " (which should fail) started with id = " << backupId << endl;
+ if (r == 0) {
+ g_err << "Backup should have failed on error_insertion " << error << endl
+ << "Master = " << masterNodeId << "Node = " << nodeId << endl;
+ }
+
+ CHECK(_restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 10099) == 0,
+ "failed to set error insert");
+ }
+
+ return NDBT_OK;
+}
+
diff --git a/ndb/test/src/NdbConfig.cpp b/ndb/test/src/NdbConfig.cpp
new file mode 100644
index 00000000000..d61b74cf62b
--- /dev/null
+++ b/ndb/test/src/NdbConfig.cpp
@@ -0,0 +1,163 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "NdbConfig.hpp"
+#include <NdbOut.hpp>
+#include <NDBT_Output.hpp>
+#include <assert.h>
+#include <NdbConfig.h>
+#include <ConfigRetriever.hpp>
+#include <ndb_version.h>
+
+
+
+bool
+NdbConfig::getPropsForNode(unsigned int node_id,
+ const char* type,
+ const Properties ** props) const {
+
+ /**
+ * Fetch configuration from management server
+ */
+ char buf[255];
+ ConfigRetriever cr;
+
+
+ Properties * p = cr.getConfig(host,
+ port,
+ node_id,
+ NDB_VERSION);
+ if(p == 0){
+ const char * s = cr.getErrorString();
+ if(s == 0)
+ s = "No error given!";
+
+ ndbout << "Could not fetch configuration" << endl;
+ ndbout << s << endl;
+ return false;
+ }
+
+ /**
+ * Setup cluster configuration data
+ */
+ if (!p->get("Node", node_id, props)) {
+ ndbout << "Invalid configuration fetched no info for nodeId = "
+ << node_id << endl;
+ return false;
+ }
+ const char * str;
+ if(!((*props)->get("Type", &str) && strcmp(str, type) == 0)){
+ ndbout <<"Invalid configuration fetched, type != " << type << endl;
+ return false;
+ }
+
+ return true;
+}
+
+bool
+NdbConfig::getProperty(unsigned int node_id,
+ const char* type,
+ const char* name,
+ const char ** value) const {
+ const Properties * db = 0;
+
+ if(!getPropsForNode(node_id, type, &db)){
+ return false;
+ }
+
+ if (!db->get(name, value)){
+ ndbout << name << " not found" << endl;
+ return false;
+ }
+
+ return true;
+}
+
+bool
+NdbConfig::getProperty(unsigned int node_id,
+ const char* type,
+ const char* name,
+ Uint32 * value) const {
+ const Properties * db = 0;
+
+ if(!getPropsForNode(node_id, type, &db)){
+ return false;
+ }
+
+ if (!db->get(name, value)){
+ ndbout << name << " not found" << endl;
+ return false;
+ }
+
+ return true;
+}
+
+
+bool
+NdbConfig::getHostName(unsigned int node_id,
+ const char ** hostname) const {
+ /**
+ * Fetch configuration from management server
+ */
+ char buf[255];
+ ConfigRetriever cr;
+
+
+ Properties * p = cr.getConfig(host,
+ port,
+ node_id,
+ NDB_VERSION);
+ if(p == 0){
+ const char * s = cr.getErrorString();
+ if(s == 0)
+ s = "No error given!";
+
+ ndbout << "Could not fetch configuration" << endl;
+ ndbout << s << endl;
+ return false;
+ }
+
+ /**
+ * Setup cluster configuration data
+ */
+ const Properties * node_props;
+ if (!p->get("Node", node_id, &node_props)) {
+ ndbout << "Invalid configuration fetched no info for node = "
+ << node_id << endl;
+ return false;
+ }
+ const char* computer_id_str;
+ if (!node_props->get("ExecuteOnComputer", &computer_id_str)){
+ ndbout << "ExecuteOnComputer not found" << endl;
+ return false;
+ }
+
+
+ const Properties * comp_props;
+ if (!p->get("Computer", atoi(computer_id_str), &comp_props)) {
+ ndbout << "Invalid configuration fetched no info for computer = "
+ << node_id << endl;
+ return false;
+ }
+ if (!comp_props->get("HostName", hostname)){
+ ndbout << "HostName not found" << endl;
+ return false;
+ }
+
+
+ return true;
+}
+
diff --git a/ndb/test/src/NdbGrep.cpp b/ndb/test/src/NdbGrep.cpp
new file mode 100644
index 00000000000..747c62d5bc6
--- /dev/null
+++ b/ndb/test/src/NdbGrep.cpp
@@ -0,0 +1,334 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <signaldata/DumpStateOrd.hpp>
+#include <NdbGrep.hpp>
+#include <NdbOut.hpp>
+#include <NDBT_Output.hpp>
+#include <assert.h>
+#include <NdbConfig.h>
+#include <ConfigRetriever.hpp>
+#include <ndb_version.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <random.h>
+#include <NdbTick.h>
+
+#define CHECK(b, m) { int _xx = b; if (!(_xx)) { \
+ ndbout << "ERR: "<< m \
+ << " " << "File: " << __FILE__ \
+ << " (Line: " << __LINE__ << ")" << "- " << _xx << endl; \
+ return NDBT_FAILED; } }
+
+
+int
+NdbGrep::start(){
+
+ return 1;
+}
+
+int
+NdbGrep::stop(){
+
+ return 1;
+}
+
+
+int
+NdbGrep::query(){
+
+ return 1;
+}
+
+
+int
+NdbGrep::verify(NDBT_Context * ctx){
+
+ if (!isConnected())
+ return -1;
+
+ char cheat_table[255];
+ snprintf(cheat_table, 255, "TEST_DB/def/%s",ctx->getTab()->getName());
+
+ char buf[255];
+ snprintf(buf, 255, "testGrepVerify -c \"nodeid=%d;host=%s\" -t %s -r %d",
+ 4, //cheat. Hardcoded nodeid....
+ ctx->getRemoteMgm(),
+ cheat_table,
+ ctx->getNumRecords());
+
+
+ ndbout << "buf: "<< buf <<endl;
+ int res = system(buf);
+
+ ndbout << "res: " << res << endl;
+
+ return res;
+
+
+
+
+}
+
+
+// Master failure
+int
+NFDuringGrepM_codes[] = {
+ 10003,
+ 10004,
+ 10005,
+ 10007,
+ 10008,
+ 10009,
+ 10010,
+ 10012,
+ 10013
+};
+
+// Slave failure
+int
+NFDuringGrepS_codes[] = {
+ 10014,
+ 10015,
+ 10016,
+ 10017,
+ 10018,
+ 10020
+};
+
+// Master takeover etc...
+int
+NFDuringGrepSL_codes[] = {
+ 10001,
+ 10002,
+ 10021
+};
+
+int
+NdbGrep::NFMaster(NdbRestarter& _restarter){
+ const int sz = sizeof(NFDuringGrepM_codes)/sizeof(NFDuringGrepM_codes[0]);
+ return NF(_restarter, NFDuringGrepM_codes, sz, true);
+}
+
+int
+NdbGrep::NFMasterAsSlave(NdbRestarter& _restarter){
+ const int sz = sizeof(NFDuringGrepS_codes)/sizeof(NFDuringGrepS_codes[0]);
+ return NF(_restarter, NFDuringGrepS_codes, sz, true);
+}
+
+int
+NdbGrep::NFSlave(NdbRestarter& _restarter){
+ const int sz = sizeof(NFDuringGrepS_codes)/sizeof(NFDuringGrepS_codes[0]);
+ return NF(_restarter, NFDuringGrepS_codes, sz, false);
+}
+
+int
+NdbGrep::NF(NdbRestarter& _restarter, int *NFDuringGrep_codes, const int sz, bool onMaster){
+ {
+ int nodeId = _restarter.getMasterNodeId();
+
+ CHECK(_restarter.restartOneDbNode(nodeId, false, true, true) == 0,
+ "Could not restart node "<< nodeId);
+
+ CHECK(_restarter.waitNodesNoStart(&nodeId, 1) == 0,
+ "waitNodesNoStart failed");
+
+ CHECK(_restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ NdbSleep_SecSleep(10);
+ }
+
+ CHECK(_restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+
+ int nNodes = _restarter.getNumDbNodes();
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+ for(int i = 0; i<sz; i++){
+
+ int error = NFDuringGrep_codes[i];
+ unsigned int backupId;
+
+ const int masterNodeId = _restarter.getMasterNodeId();
+ CHECK(masterNodeId > 0, "getMasterNodeId failed");
+ int nodeId;
+
+ nodeId = masterNodeId;
+ if (!onMaster) {
+ int randomId;
+ while (nodeId == masterNodeId) {
+ randomId = myRandom48(nNodes);
+ nodeId = _restarter.getDbNodeId(randomId);
+ }
+ }
+
+ g_err << "NdbGrep::NF node = " << nodeId
+ << " error code = " << error << " masterNodeId = "
+ << masterNodeId << endl;
+
+
+ int val = DumpStateOrd::CmvmiSetRestartOnErrorInsert;
+ CHECK(_restarter.dumpStateOneNode(nodeId, &val, 1) == 0,
+ "failed to set RestartOnErrorInsert");
+ CHECK(_restarter.insertErrorInNode(nodeId, error) == 0,
+ "failed to set error insert");
+
+ g_info << "error inserted" << endl;
+
+ g_info << "starting backup" << endl;
+ int r = start();
+ g_info << "r = " << r
+ << " (which should fail) started with id = " << backupId << endl;
+ if (r == 0) {
+ g_err << "Grep should have failed on error_insertion " << error << endl
+ << "Master = " << masterNodeId << "Node = " << nodeId << endl;
+ }
+
+ CHECK(_restarter.waitNodesNoStart(&nodeId, 1) == 0,
+ "waitNodesNoStart failed");
+
+ g_info << "number of nodes running " << _restarter.getNumDbNodes() << endl;
+
+ if (_restarter.getNumDbNodes() != nNodes) {
+ g_err << "Failure: cluster not up" << endl;
+ return NDBT_FAILED;
+ }
+
+ NdbSleep_SecSleep(1);
+
+ g_info << "starting new backup" << endl;
+ CHECK(start() == 0,
+ "failed to start backup");
+ g_info << "(which should succeed) started with id = " << backupId << endl;
+
+ g_info << "starting node" << endl;
+ CHECK(_restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ CHECK(_restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+ g_info << "node started" << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 10099) == 0,
+ "failed to set error insert");
+ }
+
+ return NDBT_OK;
+}
+
+int
+FailS_codes[] = {
+ 10023,
+ 10024,
+ 10025,
+ 10026,
+ 10027,
+ 10028,
+ 10029,
+ 10030,
+ 10031
+};
+
+int
+FailM_codes[] = {
+ 10023,
+ 10024,
+ 10025,
+ 10026,
+ 10027,
+ 10028,
+ 10029,
+ 10030,
+ 10031
+};
+
+int
+NdbGrep::FailMaster(NdbRestarter& _restarter){
+ const int sz = sizeof(FailM_codes)/sizeof(FailM_codes[0]);
+ return Fail(_restarter, FailM_codes, sz, true);
+}
+
+int
+NdbGrep::FailMasterAsSlave(NdbRestarter& _restarter){
+ const int sz = sizeof(FailS_codes)/sizeof(FailS_codes[0]);
+ return Fail(_restarter, FailS_codes, sz, true);
+}
+
+int
+NdbGrep::FailSlave(NdbRestarter& _restarter){
+ const int sz = sizeof(FailS_codes)/sizeof(FailS_codes[0]);
+ return Fail(_restarter, FailS_codes, sz, false);
+}
+
+int
+NdbGrep::Fail(NdbRestarter& _restarter, int *Fail_codes, const int sz, bool onMaster){
+
+ CHECK(_restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+
+ int nNodes = _restarter.getNumDbNodes();
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+ for(int i = 0; i<sz; i++){
+ int error = Fail_codes[i];
+ unsigned int backupId;
+
+ const int masterNodeId = _restarter.getMasterNodeId();
+ CHECK(masterNodeId > 0, "getMasterNodeId failed");
+ int nodeId;
+
+ nodeId = masterNodeId;
+ if (!onMaster) {
+ int randomId;
+ while (nodeId == masterNodeId) {
+ randomId = myRandom48(nNodes);
+ nodeId = _restarter.getDbNodeId(randomId);
+ }
+ }
+
+ g_err << "NdbGrep::Fail node = " << nodeId
+ << " error code = " << error << " masterNodeId = "
+ << masterNodeId << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, error) == 0,
+ "failed to set error insert");
+
+ g_info << "error inserted" << endl;
+ g_info << "waiting some before starting backup" << endl;
+
+ g_info << "starting backup" << endl;
+ int r = start();
+ g_info << "r = " << r
+ << " (which should fail) started with id = " << backupId << endl;
+ if (r == 0) {
+ g_err << "Grep should have failed on error_insertion " << error << endl
+ << "Master = " << masterNodeId << "Node = " << nodeId << endl;
+ }
+
+ CHECK(_restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 10099) == 0,
+ "failed to set error insert");
+ }
+
+ return NDBT_OK;
+}
+
+
diff --git a/ndb/test/src/NdbRestarter.cpp b/ndb/test/src/NdbRestarter.cpp
new file mode 100644
index 00000000000..1df214572c7
--- /dev/null
+++ b/ndb/test/src/NdbRestarter.cpp
@@ -0,0 +1,664 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbRestarter.hpp>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <NdbTick.h>
+#include <LocalConfig.hpp>
+#include <mgmapi_debug.h>
+#include <NDBT_Output.hpp>
+#include <random.h>
+#include <kernel/ndb_limits.h>
+#include <ndb_version.h>
+#include <assert.h>
+
+#define MGMERR(h) \
+ ndbout << "latest_error="<<ndb_mgm_get_latest_error(h) \
+ << ", line="<<ndb_mgm_get_latest_error_line(h) \
+ << endl;
+
+
+NdbRestarter::NdbRestarter(const char* _addr):
+ connected(false),
+ addr(_addr),
+ host(NULL),
+ port(-1),
+ handle(NULL)
+{
+ if (addr == NULL){
+ LocalConfig lcfg;
+ if(!lcfg.init()){
+ lcfg.printError();
+ lcfg.printUsage();
+ g_err << "NdbRestarter - Error parsing local config file" << endl;
+ return;
+ }
+
+ if (lcfg.items == 0){
+ g_err << "NdbRestarter - No management servers configured in local config file" << endl;
+ return;
+ }
+
+ for (int i = 0; i<lcfg.items; i++){
+ MgmtSrvrId * m = lcfg.ids[i];
+
+ switch(m->type){
+ case MgmId_TCP:
+ char buf[255];
+ snprintf(buf, 255, "%s:%d", m->data.tcp.remoteHost, m->data.tcp.port);
+ addr = strdup(buf);
+ host = strdup(m->data.tcp.remoteHost);
+ port = m->data.tcp.port;
+ break;
+ case MgmId_File:
+ break;
+ default:
+ break;
+ }
+ if (addr != NULL)
+ break;
+ }
+ }
+
+}
+
+NdbRestarter::~NdbRestarter(){
+ disconnect();
+}
+
+int NdbRestarter::getDbNodeId(int _i){
+ if (!isConnected())
+ return -1;
+
+ if (getStatus() != 0)
+ return -1;
+
+ for(size_t i = 0; i < ndbNodes.size(); i++){
+ if (i == (unsigned)_i){
+ return ndbNodes[i].node_id;
+ }
+ }
+ return -1;
+}
+
+
+int
+NdbRestarter::restartOneDbNode(int _nodeId,
+ bool inital,
+ bool nostart,
+ bool abort){
+ if (!isConnected())
+ return -1;
+
+ int ret = 0;
+
+ if ((ret = ndb_mgm_restart2(handle, 1, &_nodeId,
+ inital, nostart, abort)) <= 0) {
+ /**
+ * ndb_mgm_restart2 returned error, one reason could
+ * be that the node have not stopped fast enough!
+ * Check status of the node to see if it's on the
+ * way down. If that's the case ignore the error
+ */
+
+ if (getStatus() != 0)
+ return -1;
+
+ g_info << "ndb_mgm_restart2 returned with error, checking node state" << endl;
+
+ for(size_t i = 0; i < ndbNodes.size(); i++){
+ if(ndbNodes[i].node_id == _nodeId){
+ g_info <<_nodeId<<": status="<<ndbNodes[i].node_status<<endl;
+ /* Node found check state */
+ switch(ndbNodes[i].node_status){
+ case NDB_MGM_NODE_STATUS_RESTARTING:
+ case NDB_MGM_NODE_STATUS_SHUTTING_DOWN:
+ return 0;
+ default:
+ break;
+ }
+ }
+ }
+
+ MGMERR(handle);
+ g_err << "Could not stop node with id = "<< _nodeId << endl;
+ return -1;
+ }
+
+ return 0;
+}
+
+int
+NdbRestarter::getMasterNodeId(){
+ if (!isConnected())
+ return -1;
+
+ if (getStatus() != 0)
+ return -1;
+
+ int min = 0;
+ int node = -1;
+ for(size_t i = 0; i < ndbNodes.size(); i++){
+ if(min == 0 || ndbNodes[i].dynamic_id < min){
+ min = ndbNodes[i].dynamic_id;
+ node = ndbNodes[i].node_id;
+ }
+ }
+
+ return node;
+}
+
+int
+NdbRestarter::getRandomNotMasterNodeId(int rand){
+ int master = getMasterNodeId();
+ if(master == -1)
+ return -1;
+
+ int counter = 0;
+ rand = rand % ndbNodes.size();
+ while(counter++ < ndbNodes.size() && ndbNodes[rand].node_id == master)
+ rand = (rand + 1) % ndbNodes.size();
+
+ if(ndbNodes[rand].node_id != master)
+ return ndbNodes[rand].node_id;
+ return -1;
+}
+
+int
+NdbRestarter::getRandomNodeOtherNodeGroup(int nodeId, int rand){
+ if (!isConnected())
+ return -1;
+
+ if (getStatus() != 0)
+ return -1;
+
+ int node_group = -1;
+ for(size_t i = 0; i < ndbNodes.size(); i++){
+ if(ndbNodes[i].node_id == nodeId){
+ node_group = ndbNodes[i].node_group;
+ break;
+ }
+ }
+ if(node_group == -1){
+ return -1;
+ }
+
+ int counter = 0;
+ rand = rand % ndbNodes.size();
+ while(counter++ < ndbNodes.size() && ndbNodes[rand].node_group == node_group)
+ rand = (rand + 1) % ndbNodes.size();
+
+ if(ndbNodes[rand].node_group != node_group)
+ return ndbNodes[rand].node_id;
+
+ return -1;
+}
+
+int
+NdbRestarter::waitClusterStarted(unsigned int _timeout){
+ return waitClusterState(NDB_MGM_NODE_STATUS_STARTED, _timeout);
+}
+
+int
+NdbRestarter::waitClusterStartPhase(int _startphase, unsigned int _timeout){
+ return waitClusterState(NDB_MGM_NODE_STATUS_STARTING, _timeout, _startphase);
+}
+
+int
+NdbRestarter::waitClusterSingleUser(unsigned int _timeout){
+ return waitClusterState(NDB_MGM_NODE_STATUS_SINGLEUSER, _timeout);
+}
+
+int
+NdbRestarter::waitClusterNoStart(unsigned int _timeout){
+ return waitClusterState(NDB_MGM_NODE_STATUS_NOT_STARTED, _timeout);
+}
+
+int
+NdbRestarter::waitClusterState(ndb_mgm_node_status _status,
+ unsigned int _timeout,
+ int _startphase){
+
+ int nodes[MAX_NDB_NODES];
+ int numNodes = 0;
+
+ if (getStatus() != 0)
+ return -1;
+
+ // Collect all nodes into nodes
+ for (size_t i = 0; i < ndbNodes.size(); i++){
+ nodes[i] = ndbNodes[i].node_id;
+ numNodes++;
+ }
+
+ return waitNodesState(nodes, numNodes, _status, _timeout, _startphase);
+}
+
+
+int
+NdbRestarter::waitNodesState(int * _nodes, int _num_nodes,
+ ndb_mgm_node_status _status,
+ unsigned int _timeout,
+ int _startphase){
+
+ if (!isConnected()){
+ g_err << "!isConnected"<<endl;
+ return -1;
+ }
+
+ unsigned int attempts = 0;
+ unsigned int resetAttempts = 0;
+ const unsigned int MAX_RESET_ATTEMPTS = 10;
+ bool allInState = false;
+ while (allInState == false){
+ if (_timeout > 0 && attempts > _timeout){
+ /**
+ * Timeout has expired waiting for the nodes to enter
+ * the state we want
+ */
+ bool waitMore = false;
+ /**
+ * Make special check if we are waiting for
+ * cluster to become started
+ */
+ if(_status == NDB_MGM_NODE_STATUS_STARTED){
+ waitMore = true;
+ /**
+ * First check if any node is not starting
+ * then it's no idea to wait anymore
+ */
+ for (size_t n = 0; n < ndbNodes.size(); n++){
+ if (ndbNodes[n].node_status != NDB_MGM_NODE_STATUS_STARTED &&
+ ndbNodes[n].node_status != NDB_MGM_NODE_STATUS_STARTING)
+ waitMore = false;
+
+ }
+ }
+
+ if (!waitMore || resetAttempts > MAX_RESET_ATTEMPTS){
+ g_err << "waitNodeState("
+ << ndb_mgm_get_node_status_string(_status)
+ <<", "<<_startphase<<")"
+ << " timeout after " << attempts <<" attemps" << endl;
+ return -1;
+ }
+
+ g_err << "waitNodeState("
+ << ndb_mgm_get_node_status_string(_status)
+ <<", "<<_startphase<<")"
+ << " resetting number of attempts "
+ << resetAttempts << endl;
+ attempts = 0;
+ resetAttempts++;
+
+ }
+
+ allInState = true;
+ if (getStatus() != 0){
+ g_err << "getStatus != 0" << endl;
+ return -1;
+ }
+
+ // ndbout << "waitNodeState; _num_nodes = " << _num_nodes << endl;
+ // for (int i = 0; i < _num_nodes; i++)
+ // ndbout << " node["<<i<<"] =" <<_nodes[i] << endl;
+
+ for (int i = 0; i < _num_nodes; i++){
+ ndb_mgm_node_state* ndbNode = NULL;
+ for (size_t n = 0; n < ndbNodes.size(); n++){
+ if (ndbNodes[n].node_id == _nodes[i])
+ ndbNode = &ndbNodes[n];
+ }
+
+ if(ndbNode == NULL){
+ allInState = false;
+ continue;
+ }
+
+ g_info << "State node " << ndbNode->node_id << " "
+ << ndb_mgm_get_node_status_string(ndbNode->node_status)<< endl;
+
+ assert(ndbNode != NULL);
+
+ if(_status == NDB_MGM_NODE_STATUS_STARTING &&
+ ((ndbNode->node_status == NDB_MGM_NODE_STATUS_STARTING &&
+ ndbNode->start_phase >= _startphase) ||
+ (ndbNode->node_status == NDB_MGM_NODE_STATUS_STARTED)))
+ continue;
+
+ if (_status == NDB_MGM_NODE_STATUS_STARTING){
+ g_info << "status = "
+ << ndb_mgm_get_node_status_string(ndbNode->node_status)
+ <<", start_phase="<<ndbNode->start_phase<<endl;
+ if (ndbNode->node_status != _status) {
+ if (ndbNode->node_status < _status)
+ allInState = false;
+ else
+ g_info << "node_status(" << ndbNode->node_status
+ <<") != _status("<<_status<<")"<<endl;
+ } else if (ndbNode->start_phase < _startphase)
+ allInState = false;
+ } else {
+ if (ndbNode->node_status != _status)
+ allInState = false;
+ }
+ }
+ g_info << "Waiting for cluster enter state"
+ << ndb_mgm_get_node_status_string(_status)<< endl;
+ NdbSleep_SecSleep(1);
+ attempts++;
+ }
+ return 0;
+}
+
+int NdbRestarter::waitNodesStarted(int * _nodes, int _num_nodes,
+ unsigned int _timeout){
+ return waitNodesState(_nodes, _num_nodes,
+ NDB_MGM_NODE_STATUS_STARTED, _timeout);
+}
+
+int NdbRestarter::waitNodesStartPhase(int * _nodes, int _num_nodes,
+ int _startphase, unsigned int _timeout){
+ return waitNodesState(_nodes, _num_nodes,
+ NDB_MGM_NODE_STATUS_STARTING, _timeout,
+ _startphase);
+}
+
+int NdbRestarter::waitNodesNoStart(int * _nodes, int _num_nodes,
+ unsigned int _timeout){
+ return waitNodesState(_nodes, _num_nodes,
+ NDB_MGM_NODE_STATUS_NOT_STARTED, _timeout);
+}
+
+bool
+NdbRestarter::isConnected(){
+ if (connected == true)
+ return true;
+ return connect() == 0;
+}
+
+int
+NdbRestarter::connect(){
+ handle = ndb_mgm_create_handle();
+ if (handle == NULL){
+ g_err << "handle == NULL" << endl;
+ return -1;
+ }
+ g_info << "Connecting to mgmsrv at " << addr << endl;
+ if (ndb_mgm_connect(handle, addr) == -1) {
+ MGMERR(handle);
+ g_err << "Connection to " << addr << " failed" << endl;
+ return -1;
+ }
+
+ connected = true;
+ return 0;
+}
+
+void
+NdbRestarter::disconnect(){
+ if (handle != NULL){
+ ndb_mgm_disconnect(handle);
+ ndb_mgm_destroy_handle(&handle);
+ }
+ connected = false;
+}
+
+int
+NdbRestarter::getStatus(){
+ int retries = 0;
+ struct ndb_mgm_cluster_state * status;
+ struct ndb_mgm_node_state * node;
+
+ ndbNodes.clear();
+ mgmNodes.clear();
+ apiNodes.clear();
+
+ if (!isConnected())
+ return -1;
+
+ while(retries < 10){
+ status = ndb_mgm_get_status(handle);
+ if (status == NULL){
+ ndbout << "status==NULL, retries="<<retries<<endl;
+ MGMERR(handle);
+ retries++;
+ continue;
+ }
+ for (int i = 0; i < status->no_of_nodes; i++){
+ node = &status->node_states[i];
+ switch(node->node_type){
+ case NDB_MGM_NODE_TYPE_NDB:
+ ndbNodes.push_back(*node);
+ break;
+ case NDB_MGM_NODE_TYPE_MGM:
+ mgmNodes.push_back(*node);
+ break;
+ case NDB_MGM_NODE_TYPE_API:
+ apiNodes.push_back(*node);
+ break;
+ default:
+ if(node->node_status == NDB_MGM_NODE_STATUS_UNKNOWN ||
+ node->node_status == NDB_MGM_NODE_STATUS_NO_CONTACT){
+ retries++;
+ ndbNodes.clear();
+ mgmNodes.clear();
+ apiNodes.clear();
+ free(status);
+ status = NULL;
+ i = status->no_of_nodes;
+
+ ndbout << "kalle"<< endl;
+ break;
+ }
+ abort();
+ break;
+ }
+ }
+ if(status == 0){
+ ndbout << "status == 0" << endl;
+ continue;
+ }
+ free(status);
+ return 0;
+ }
+
+ g_err << "getStatus failed" << endl;
+ return -1;
+}
+
+
+int NdbRestarter::getNumDbNodes(){
+ if (!isConnected())
+ return -1;
+
+ if (getStatus() != 0)
+ return -1;
+
+ return ndbNodes.size();
+}
+
+int NdbRestarter::restartAll(bool initial,
+ bool nostart,
+ bool abort){
+
+ if (!isConnected())
+ return -1;
+
+ if (ndb_mgm_restart2(handle, 0, NULL, initial, 1, abort) == -1) {
+ MGMERR(handle);
+ g_err << "Could not restart(stop) all nodes " << endl;
+ // return -1; Continue anyway - Magnus
+ }
+
+ if (waitClusterNoStart(60) != 0){
+ g_err << "Cluster didnt enter STATUS_NOT_STARTED within 60s" << endl;
+ return -1;
+ }
+
+ if(nostart){
+ g_debug << "restartAll: nostart == true" << endl;
+ return 0;
+ }
+
+ if (ndb_mgm_start(handle, 0, NULL) == -1) {
+ MGMERR(handle);
+ g_err << "Could not restart(start) all nodes " << endl;
+ return -1;
+ }
+
+ return 0;
+}
+
+int NdbRestarter::startAll(){
+ if (!isConnected())
+ return -1;
+
+ if (ndb_mgm_start(handle, 0, NULL) == -1) {
+ MGMERR(handle);
+ g_err << "Could not start all nodes " << endl;
+ return -1;
+ }
+
+ return 0;
+
+}
+
+int NdbRestarter::startNodes(int * nodes, int num_nodes){
+ if (!isConnected())
+ return -1;
+
+ if (ndb_mgm_start(handle, num_nodes, nodes) != num_nodes) {
+ MGMERR(handle);
+ g_err << "Could not start all nodes " << endl;
+ return -1;
+ }
+
+ return 0;
+}
+
+int NdbRestarter::insertErrorInNode(int _nodeId, int _error){
+ if (!isConnected())
+ return -1;
+
+ ndb_mgm_reply reply;
+ reply.return_code = 0;
+
+ if (ndb_mgm_insert_error(handle, _nodeId, _error, &reply) == -1){
+ MGMERR(handle);
+ g_err << "Could not insert error in node with id = "<< _nodeId << endl;
+ }
+ if(reply.return_code != 0){
+ g_err << "Error: " << reply.message << endl;
+ }
+ return 0;
+}
+
+int NdbRestarter::insertErrorInAllNodes(int _error){
+ if (!isConnected())
+ return -1;
+
+ if (getStatus() != 0)
+ return -1;
+
+ int result = 0;
+
+ for(size_t i = 0; i < ndbNodes.size(); i++){
+ g_debug << "inserting error in node " << ndbNodes[i].node_id << endl;
+ if (insertErrorInNode(ndbNodes[i].node_id, _error) == -1)
+ result = -1;
+ }
+ return result;
+
+}
+
+
+
+int NdbRestarter::dumpStateOneNode(int _nodeId, int * _args, int _num_args){
+ if (!isConnected())
+ return -1;
+
+ ndb_mgm_reply reply;
+ reply.return_code = 0;
+
+ if (ndb_mgm_dump_state(handle, _nodeId, _args, _num_args, &reply) == -1){
+ MGMERR(handle);
+ g_err << "Could not dump state in node with id = "<< _nodeId << endl;
+ }
+
+ if(reply.return_code != 0){
+ g_err << "Error: " << reply.message << endl;
+ }
+ return reply.return_code;
+}
+
+int NdbRestarter::dumpStateAllNodes(int * _args, int _num_args){
+ if (!isConnected())
+ return -1;
+
+ if (getStatus() != 0)
+ return -1;
+
+ int result = 0;
+
+ for(size_t i = 0; i < ndbNodes.size(); i++){
+ g_debug << "dumping state in node " << ndbNodes[i].node_id << endl;
+ if (dumpStateOneNode(ndbNodes[i].node_id, _args, _num_args) == -1)
+ result = -1;
+ }
+ return result;
+
+}
+
+
+int NdbRestarter::enterSingleUserMode(int _nodeId){
+ if (!isConnected())
+ return -1;
+
+ ndb_mgm_reply reply;
+ reply.return_code = 0;
+
+ if (ndb_mgm_enter_single_user(handle, _nodeId, &reply) == -1){
+ MGMERR(handle);
+ g_err << "Could not enter single user mode api node = "<< _nodeId << endl;
+ }
+
+ if(reply.return_code != 0){
+ g_err << "Error: " << reply.message << endl;
+ }
+
+ return reply.return_code;
+}
+
+
+int NdbRestarter::exitSingleUserMode(){
+ if (!isConnected())
+ return -1;
+
+ ndb_mgm_reply reply;
+ reply.return_code = 0;
+
+ if (ndb_mgm_exit_single_user(handle, &reply) == -1){
+ MGMERR(handle);
+ g_err << "Could not exit single user mode " << endl;
+ }
+
+ if(reply.return_code != 0){
+ g_err << "Error: " << reply.message << endl;
+ }
+ return reply.return_code;
+}
diff --git a/ndb/test/src/NdbRestarts.cpp b/ndb/test/src/NdbRestarts.cpp
new file mode 100644
index 00000000000..f6a85d69fc2
--- /dev/null
+++ b/ndb/test/src/NdbRestarts.cpp
@@ -0,0 +1,875 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbRestarts.hpp>
+#include <NDBT.hpp>
+#include <string.h>
+#include <NdbSleep.h>
+#include <kernel/ndb_limits.h>
+#include <signaldata/DumpStateOrd.hpp>
+#include <NdbEnv.h>
+
+
+int restartRandomNodeGraceful(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int restartRandomNodeAbort(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int restartRandomNodeError(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int restartRandomNodeInitial(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int restartNFDuringNR(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int restartMasterNodeError(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int twoNodeFailure(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int fiftyPercentFail(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int twoMasterNodeFailure(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int restartAllNodesGracfeul(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int restartAllNodesAbort(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int restartAllNodesError9999(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int fiftyPercentStopAndWait(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int restartNodeDuringLCP(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart);
+int stopOnError(NdbRestarter&, const NdbRestarts::NdbRestart*);
+int getRandomNodeId(NdbRestarter& _restarter);
+
+/**
+ * Define list of restarts
+ * - name of restart
+ * - function perfoming the restart
+ * - required number of nodes
+ * - ...
+ * - arg1, used depending of restart
+ * - arg2, used depending of restart
+ */
+
+const NdbRestarts::NdbRestart NdbRestarts::m_restarts[] = {
+ /*********************************************************
+ *
+ * NODE RESTARTS with 1 node restarted
+ *
+ *********************************************************/
+ /**
+ * Restart a randomly selected node
+ * with graceful shutdown
+ */
+ NdbRestart("RestartRandomNode",
+ NODE_RESTART,
+ restartRandomNodeGraceful,
+ 2),
+ /**
+ * Restart a randomly selected node
+ * with immediate(abort) shutdown
+ */
+ NdbRestart("RestartRandomNodeAbort",
+ NODE_RESTART,
+ restartRandomNodeAbort,
+ 2),
+ /**
+ * Restart a randomly selected node
+ * with error insert
+ *
+ */
+ NdbRestart("RestartRandomNodeError",
+ NODE_RESTART,
+ restartRandomNodeError,
+ 2),
+ /**
+ * Restart the master node
+ * with error insert
+ */
+ NdbRestart("RestartMasterNodeError",
+ NODE_RESTART,
+ restartMasterNodeError,
+ 2),
+ /**
+ * Restart a randomly selected node without fileystem
+ *
+ */
+ NdbRestart("RestartRandomNodeInitial",
+ NODE_RESTART,
+ restartRandomNodeInitial,
+ 2),
+ /**
+ * Restart a randomly selected node and then
+ * crash it while restarting
+ *
+ */
+ NdbRestart("RestartNFDuringNR",
+ NODE_RESTART,
+ restartNFDuringNR,
+ 2),
+
+ /**
+ * Set StopOnError and crash the node by sending
+ * SYSTEM_ERROR to it
+ *
+ */
+ NdbRestart("StopOnError",
+ NODE_RESTART,
+ stopOnError,
+ 1),
+
+ /*********************************************************
+ *
+ * MULTIPLE NODE RESTARTS with more than 1 node
+ *
+ *********************************************************/
+ /**
+ * 2 nodes restart, select nodes to restart randomly and restart
+ * with a small random delay between restarts
+ */
+ NdbRestart("TwoNodeFailure",
+ MULTIPLE_NODE_RESTART,
+ twoNodeFailure,
+ 4),
+ /**
+ * 2 nodes restart, select master nodes and restart with
+ * a small random delay between restarts
+ */
+
+ NdbRestart("TwoMasterNodeFailure",
+ MULTIPLE_NODE_RESTART,
+ twoMasterNodeFailure,
+ 4),
+
+ NdbRestart("FiftyPercentFail",
+ MULTIPLE_NODE_RESTART,
+ fiftyPercentFail,
+ 2),
+
+ /*********************************************************
+ *
+ * SYSTEM RESTARTS
+ *
+ *********************************************************/
+ /**
+ * Restart all nodes with graceful shutdown
+ *
+ */
+
+ NdbRestart("RestartAllNodes",
+ SYSTEM_RESTART,
+ restartAllNodesGracfeul,
+ 1),
+ /**
+ * Restart all nodes immediately without
+ * graful shutdown
+ */
+ NdbRestart("RestartAllNodesAbort",
+ SYSTEM_RESTART,
+ restartAllNodesAbort,
+ 1),
+ /**
+ * Restart all nodes with error insert 9999
+ * TODO! We can later add more errors like 9998, 9997 etc.
+ */
+ NdbRestart("RestartAllNodesError9999",
+ SYSTEM_RESTART,
+ restartAllNodesError9999,
+ 1),
+ /**
+ * Stop 50% of all nodes with error insert 9999
+ * Wait for a random number of minutes
+ * Stop the rest of the nodes and then start all again
+ */
+ NdbRestart("FiftyPercentStopAndWait",
+ SYSTEM_RESTART,
+ fiftyPercentStopAndWait,
+ 2),
+ /**
+ * Restart a master node during LCP with error inserts.
+ */
+ NdbRestart("RestartNodeDuringLCP",
+ NODE_RESTART,
+ restartNodeDuringLCP,
+ 2),
+};
+
+const int NdbRestarts::m_NoOfRestarts = sizeof(m_restarts) / sizeof(NdbRestart);
+
+
+const NdbRestarts::NdbErrorInsert NdbRestarts::m_errors[] = {
+ NdbErrorInsert("Error9999", 9999)
+};
+
+const int NdbRestarts::m_NoOfErrors = sizeof(m_errors) / sizeof(NdbErrorInsert);
+
+NdbRestarts::NdbRestart::NdbRestart(const char* _name,
+ NdbRestartType _type,
+ restartFunc* _func,
+ int _requiredNodes,
+ int _arg1){
+ m_name = _name;
+ m_type = _type;
+ m_restartFunc = _func;
+ m_numRequiredNodes = _requiredNodes;
+ // m_arg1 = arg1;
+};
+
+
+int NdbRestarts::getNumRestarts(){
+ return m_NoOfRestarts;
+}
+
+const NdbRestarts::NdbRestart* NdbRestarts::getRestart(int _num){
+ if (_num >= m_NoOfRestarts)
+ return NULL;
+
+ return &m_restarts[_num];
+}
+
+const NdbRestarts::NdbRestart* NdbRestarts::getRestart(const char* _name){
+ for(int i = 0; i < m_NoOfRestarts; i++){
+ if (strcmp(m_restarts[i].m_name, _name) == 0){
+ return &m_restarts[i];
+ }
+ }
+ g_err << "The restart \""<< _name << "\" not found in NdbRestarts" << endl;
+ return NULL;
+}
+
+
+int NdbRestarts::executeRestart(const NdbRestarts::NdbRestart* _restart,
+ unsigned int _timeout){
+ // Check that there are enough nodes in the cluster
+ // for this test
+ NdbRestarter restarter;
+ if (_restart->m_numRequiredNodes > restarter.getNumDbNodes()){
+ g_err << "This test requires " << _restart->m_numRequiredNodes << " nodes "
+ << "there are only "<< restarter.getNumDbNodes() <<" nodes in cluster"
+ << endl;
+ return NDBT_OK;
+ }
+ if (restarter.waitClusterStarted(120) != 0){
+ // If cluster is not started when we shall peform restart
+ // the restart can not be executed and the test fails
+ return NDBT_FAILED;
+ }
+
+ int res = _restart->m_restartFunc(restarter, _restart);
+
+ // Sleep a little waiting for nodes to react to command
+ NdbSleep_SecSleep(2);
+
+ if (_timeout == 0){
+ // If timeout == 0 wait for ever
+ while(restarter.waitClusterStarted(60) != 0)
+ g_err << "Cluster is not started after restart. Waiting 60s more..."
+ << endl;
+ } else {
+ if (restarter.waitClusterStarted(_timeout) != 0){
+ g_err<<"Cluster failed to start" << endl;
+ res = NDBT_FAILED;
+ }
+ }
+
+ return res;
+}
+
+int NdbRestarts::executeRestart(int _num,
+ unsigned int _timeout){
+ const NdbRestarts::NdbRestart* r = getRestart(_num);
+ if (r == NULL)
+ return NDBT_FAILED;
+
+ int res = executeRestart(r, _timeout);
+ return res;
+}
+
+int NdbRestarts::executeRestart(const char* _name,
+ unsigned int _timeout){
+ const NdbRestarts::NdbRestart* r = getRestart(_name);
+ if (r == NULL)
+ return NDBT_FAILED;
+
+ int res = executeRestart(r, _timeout);
+ return res;
+}
+
+void NdbRestarts::listRestarts(NdbRestartType _type){
+ for(int i = 0; i < m_NoOfRestarts; i++){
+ if (m_restarts[i].m_type == _type)
+ ndbout << " " << m_restarts[i].m_name << ", min "
+ << m_restarts[i].m_numRequiredNodes
+ << " nodes"<< endl;
+ }
+}
+
+void NdbRestarts::listRestarts(){
+ ndbout << "NODE RESTARTS" << endl;
+ listRestarts(NODE_RESTART);
+ ndbout << "MULTIPLE NODE RESTARTS" << endl;
+ listRestarts(MULTIPLE_NODE_RESTART);
+ ndbout << "SYSTEM RESTARTS" << endl;
+ listRestarts(SYSTEM_RESTART);
+}
+
+NdbRestarts::NdbErrorInsert::NdbErrorInsert(const char* _name,
+ int _errorNo){
+
+ m_name = _name;
+ m_errorNo = _errorNo;
+}
+
+int NdbRestarts::getNumErrorInserts(){
+ return m_NoOfErrors;
+}
+
+const NdbRestarts::NdbErrorInsert* NdbRestarts::getError(int _num){
+ if (_num >= m_NoOfErrors)
+ return NULL;
+ return &m_errors[_num];
+}
+
+const NdbRestarts::NdbErrorInsert* NdbRestarts::getRandomError(){
+ int randomId = myRandom48(m_NoOfErrors);
+ return &m_errors[randomId];
+}
+
+
+
+/**
+ *
+ * IMPLEMENTATION OF THE DIFFERENT RESTARTS
+ * Each function should perform it's action
+ * and the returned NDBT_OK or NDBT_FAILED
+ *
+ */
+
+
+#define CHECK(b, m) { int _xx = b; if (!(_xx)) { \
+ ndbout << "ERR: "<< m \
+ << " " << "File: " << __FILE__ \
+ << " (Line: " << __LINE__ << ")" << "- " << _xx << endl; \
+ return NDBT_FAILED; } }
+
+
+
+int restartRandomNodeGraceful(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ int randomId = myRandom48(_restarter.getNumDbNodes());
+ int nodeId = _restarter.getDbNodeId(randomId);
+
+ g_info << _restart->m_name << ": node = "<<nodeId << endl;
+
+ CHECK(_restarter.restartOneDbNode(nodeId) == 0,
+ "Could not restart node "<<nodeId);
+
+ return NDBT_OK;
+};
+
+int restartRandomNodeAbort(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ int randomId = myRandom48(_restarter.getNumDbNodes());
+ int nodeId = _restarter.getDbNodeId(randomId);
+
+ g_info << _restart->m_name << ": node = "<<nodeId << endl;
+
+ CHECK(_restarter.restartOneDbNode(nodeId, false, false, true) == 0,
+ "Could not restart node "<<nodeId);
+
+ return NDBT_OK;
+};
+
+int restartRandomNodeError(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ int randomId = myRandom48(_restarter.getNumDbNodes());
+ int nodeId = _restarter.getDbNodeId(randomId);
+
+ ndbout << _restart->m_name << ": node = "<<nodeId << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 9999) == 0,
+ "Could not restart node "<<nodeId);
+
+ return NDBT_OK;
+};
+
+int restartMasterNodeError(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ int nodeId = _restarter.getDbNodeId(0);
+
+ g_info << _restart->m_name << ": node = "<<nodeId << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 39999) == 0,
+ "Could not restart node "<<nodeId);
+
+ return NDBT_OK;
+};
+
+int restartRandomNodeInitial(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ int randomId = myRandom48(_restarter.getNumDbNodes());
+ int nodeId = _restarter.getDbNodeId(randomId);
+
+ g_info << _restart->m_name << ": node = "<<nodeId << endl;
+
+ CHECK(_restarter.restartOneDbNode(nodeId, true) == 0,
+ "Could not restart node "<<nodeId);
+
+ return NDBT_OK;
+};
+
+int twoNodeFailure(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ int randomId = myRandom48(_restarter.getNumDbNodes());
+ int nodeId = _restarter.getDbNodeId(randomId);
+ g_info << _restart->m_name << ": node = "<< nodeId << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 9999) == 0,
+ "Could not restart node "<< nodeId);
+
+ // Create random value, max 10 secs
+ int max = 10;
+ int seconds = (myRandom48(max)) + 1;
+ g_info << "Waiting for " << seconds << "(" << max
+ << ") secs " << endl;
+ NdbSleep_SecSleep(seconds);
+
+ randomId = (random() % _restarter.getNumDbNodes());
+ nodeId = _restarter.getDbNodeId(randomId);
+ g_info << _restart->m_name << ": node = "<< nodeId << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 9999) == 0,
+ "Could not restart node "<< nodeId);
+
+ return NDBT_OK;
+};
+
+int twoMasterNodeFailure(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ int nodeId = _restarter.getDbNodeId(0);
+ g_info << _restart->m_name << ": node = "<< nodeId << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 39999) == 0,
+ "Could not restart node "<< nodeId);
+
+ // Create random value, max 10 secs
+ int max = 10;
+ int seconds = (myRandom48(max)) + 1;
+ g_info << "Waiting for " << seconds << "(" << max
+ << ") secs " << endl;
+ NdbSleep_SecSleep(seconds);
+
+ nodeId = _restarter.getDbNodeId(0);
+ g_info << _restart->m_name << ": node = "<< nodeId << endl;
+
+ CHECK(_restarter.insertErrorInNode(nodeId, 39999) == 0,
+ "Could not restart node "<< nodeId);
+
+ return NDBT_OK;
+};
+
+int get50PercentOfNodes(NdbRestarter& restarter,
+ int * _nodes){
+ // For now simply return all nodes with even node id
+ // TODO Check nodegroup and return one node from each
+
+ int num50Percent = restarter.getNumDbNodes() / 2;
+ assert(num50Percent <= MAX_NDB_NODES);
+
+ // Calculate which nodes to stop, select all even nodes
+ for (int i = 0; i < num50Percent; i++){
+ _nodes[i] = restarter.getDbNodeId(i*2);
+ }
+ return num50Percent;
+}
+
+int fiftyPercentFail(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+
+ int nodes[MAX_NDB_NODES];
+
+ int numNodes = get50PercentOfNodes(_restarter, nodes);
+
+ // Stop the nodes, with nostart and abort
+ for (int i = 0; i < numNodes; i++){
+ g_info << "Stopping node "<< nodes[i] << endl;
+ int res = _restarter.restartOneDbNode(nodes[i], false, true, true);
+ CHECK(res == 0, "Could not stop node: "<< nodes[i]);
+ }
+
+ CHECK(_restarter.waitNodesNoStart(nodes, numNodes) == 0,
+ "waitNodesNoStart");
+
+ // Order all nodes to start
+ ndbout << "Starting all nodes" << endl;
+ CHECK(_restarter.startAll() == 0,
+ "Could not start all nodes");
+
+ return NDBT_OK;
+};
+
+
+int restartAllNodesGracfeul(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ g_info << _restart->m_name << endl;
+
+ // Restart graceful
+ CHECK(_restarter.restartAll() == 0,
+ "Could not restart all nodes");
+
+ return NDBT_OK;
+
+};
+
+int restartAllNodesAbort(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ g_info << _restart->m_name << endl;
+
+ // Restart abort
+ CHECK(_restarter.restartAll(false, false, true) == 0,
+ "Could not restart all nodes");
+
+ return NDBT_OK;
+};
+
+int restartAllNodesError9999(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ g_info << _restart->m_name << endl;
+
+ // Restart with error insert
+ CHECK(_restarter.insertErrorInAllNodes(9999) == 0,
+ "Could not restart all nodes ");
+
+ return NDBT_OK;
+};
+
+int fiftyPercentStopAndWait(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ int nodes[MAX_NDB_NODES];
+ int numNodes = get50PercentOfNodes(_restarter, nodes);
+
+ // Stop the nodes, with nostart and abort
+ for (int i = 0; i < numNodes; i++){
+ g_info << "Stopping node "<<nodes[i] << endl;
+ int res = _restarter.restartOneDbNode(nodes[i], false, true, true);
+ CHECK(res == 0, "Could not stop node: "<< nodes[i]);
+ }
+
+ CHECK(_restarter.waitNodesNoStart(nodes, numNodes) == 0,
+ "waitNodesNoStart");
+
+ // Create random value, max 120 secs
+ int max = 120;
+ int seconds = (myRandom48(max)) + 1;
+ g_info << "Waiting for " << seconds << "(" << max
+ << ") secs " << endl;
+ NdbSleep_SecSleep(seconds);
+
+
+ // Restart graceful
+ CHECK(_restarter.restartAll() == 0,
+ "Could not restart all nodes");
+
+ g_info << _restart->m_name << endl;
+
+ return NDBT_OK;
+};
+
+int
+NFDuringNR_codes[] = {
+ 7121,
+ 5027,
+ 7172,
+ 6000,
+ 6001,
+ 6002,
+ 7171,
+ 7130,
+ 7133,
+ 7138,
+ 7154,
+ 7144,
+ 5026,
+ 7139,
+ 7132,
+
+ //LCP
+ 8000,
+ 8001,
+ 5010,
+ 7022,
+ 7024,
+ 7016,
+ 7017,
+ 5002
+};
+
+int restartNFDuringNR(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+ const int sz = sizeof(NFDuringNR_codes)/sizeof(NFDuringNR_codes[0]);
+ for(int i = 0; i<sz; i++){
+ int randomId = myRandom48(_restarter.getNumDbNodes());
+ int nodeId = _restarter.getDbNodeId(randomId);
+ int error = NFDuringNR_codes[i];
+
+ g_info << _restart->m_name << ": node = " << nodeId
+ << " error code = " << error << endl;
+
+ CHECK(_restarter.restartOneDbNode(nodeId, false, true, true) == 0,
+ "Could not restart node "<< nodeId);
+
+ CHECK(_restarter.waitNodesNoStart(&nodeId, 1) == 0,
+ "waitNodesNoStart failed");
+
+ int val = DumpStateOrd::CmvmiSetRestartOnErrorInsert;
+ CHECK(_restarter.dumpStateOneNode(nodeId, &val, 1) == 0,
+ "failed to set RestartOnErrorInsert");
+
+ CHECK(_restarter.insertErrorInNode(nodeId, error) == 0,
+ "failed to set error insert");
+
+ CHECK(_restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ NdbSleep_SecSleep(3);
+
+ //CHECK(_restarter.waitNodesNoStart(&nodeId, 1) == 0,
+ // "waitNodesNoStart failed");
+ _restarter.waitNodesNoStart(&nodeId, 1);
+
+ CHECK(_restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ CHECK(_restarter.waitNodesStarted(&nodeId, 1) == 0,
+ "waitNodesStarted failed");
+ }
+
+ return NDBT_OK;
+
+ if(_restarter.getNumDbNodes() < 4)
+ return NDBT_OK;
+
+ char buf[256];
+ if(NdbEnv_GetEnv("USER", buf, 256) == 0 || strcmp(buf, "ejonore") != 0)
+ return NDBT_OK;
+
+ for(int i = 0; i<sz; i++){
+ const int randomId = myRandom48(_restarter.getNumDbNodes());
+ int nodeId = _restarter.getDbNodeId(randomId);
+ const int error = NFDuringNR_codes[i];
+
+ const int masterNodeId = _restarter.getMasterNodeId();
+ CHECK(masterNodeId > 0, "getMasterNodeId failed");
+ int crashNodeId = 0;
+ do {
+ int rand = myRandom48(1000);
+ crashNodeId = _restarter.getRandomNodeOtherNodeGroup(nodeId, rand);
+ } while(crashNodeId == masterNodeId);
+
+ CHECK(crashNodeId > 0, "getMasterNodeId failed");
+
+ g_info << _restart->m_name << " restarting node = " << nodeId
+ << " error code = " << error
+ << " crash node = " << crashNodeId << endl;
+
+ CHECK(_restarter.restartOneDbNode(nodeId, false, true, true) == 0,
+ "Could not restart node "<< nodeId);
+
+ CHECK(_restarter.waitNodesNoStart(&nodeId, 1) == 0,
+ "waitNodesNoStart failed");
+
+ int val = DumpStateOrd::CmvmiSetRestartOnErrorInsert;
+ CHECK(_restarter.dumpStateOneNode(crashNodeId, &val, 2) == 0,
+ "failed to set RestartOnErrorInsert");
+
+ CHECK(_restarter.insertErrorInNode(crashNodeId, error) == 0,
+ "failed to set error insert");
+
+ CHECK(_restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ CHECK(_restarter.waitClusterStarted() == 0,
+ "waitClusterStarted failed");
+ }
+
+ return NDBT_OK;
+};
+
+int
+NRDuringLCP_Master_codes[] = {
+ 7009, // Insert system error in master when local checkpoint is idle.
+ 7010, // Insert system error in master when local checkpoint is in the
+ // state clcpStatus = CALCULATE_KEEP_GCI.
+ 7013, // Insert system error in master when local checkpoint is in the
+ // state clcpStatus = COPY_GCI before sending COPY_GCIREQ.
+ 7014, // Insert system error in master when local checkpoint is in the
+ // state clcpStatus = TC_CLOPSIZE before sending TC_CLOPSIZEREQ.
+ 7015, // Insert system error in master when local checkpoint is in the
+ // state clcpStatus = START_LCP_ROUND before sending START_LCP_ROUND.
+ 7019, // Insert system error in master when local checkpoint is in the
+ // state clcpStatus = IDLE before sending CONTINUEB(ZCHECK_TC_COUNTER).
+ 7075, // Master. Don't send any LCP_FRAG_ORD(last=true)
+ // And crash when all have "not" been sent
+ 7021, // Crash in master when receiving START_LCP_REQ
+ 7023, // Crash in master when sending START_LCP_CONF
+ 7025, // Crash in master when receiving LCP_FRAG_REP
+ 7026, // Crash in master when changing state to LCP_TAB_COMPLETED
+ 7027 // Crash in master when changing state to LCP_TAB_SAVED
+};
+
+int
+NRDuringLCP_NonMaster_codes[] = {
+ 7020, // Insert system error in local checkpoint participant at reception
+ // of COPY_GCIREQ.
+ 8000, // Crash particpant when receiving TCGETOPSIZEREQ
+ 8001, // Crash particpant when receiving TC_CLOPSIZEREQ
+ 5010, // Crash any when receiving LCP_FRAGORD
+ 7022, // Crash in !master when receiving START_LCP_REQ
+ 7024, // Crash in !master when sending START_LCP_CONF
+ 7016, // Crash in !master when receiving LCP_FRAG_REP
+ 7017, // Crash in !master when changing state to LCP_TAB_COMPLETED
+ 7018 // Crash in !master when changing state to LCP_TAB_SAVED
+};
+
+int restartNodeDuringLCP(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart) {
+
+ // Master
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(_restarter.dumpStateAllNodes(&val, 1) == 0,
+ "Failed to set LCP to min value"); // Set LCP to min val
+ int sz = sizeof(NRDuringLCP_Master_codes)/
+ sizeof(NRDuringLCP_Master_codes[0]);
+ for(int i = 0; i<sz; i++) {
+
+ int error = NRDuringLCP_Master_codes[i];
+ int masterNodeId = _restarter.getMasterNodeId();
+
+ CHECK(masterNodeId > 0, "getMasterNodeId failed");
+
+ ndbout << _restart->m_name << " restarting master node = " << masterNodeId
+ << " error code = " << error << endl;
+
+ {
+ int val = DumpStateOrd::CmvmiSetRestartOnErrorInsert;
+ CHECK(_restarter.dumpStateAllNodes(&val, 1) == 0,
+ "failed to set RestartOnErrorInsert");
+ }
+
+ CHECK(_restarter.insertErrorInNode(masterNodeId, error) == 0,
+ "failed to set error insert");
+
+ CHECK(_restarter.waitNodesNoStart(&masterNodeId, 1, 300) == 0,
+ "failed to wait no start");
+
+ CHECK(_restarter.startNodes(&masterNodeId, 1) == 0,
+ "failed to start node");
+
+ CHECK(_restarter.waitClusterStarted(300) == 0,
+ "waitClusterStarted failed");
+
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(_restarter.dumpStateOneNode(masterNodeId, &val, 1) == 0,
+ "failed to set error insert");
+ }
+ }
+
+ // NON-Master
+ sz = sizeof(NRDuringLCP_NonMaster_codes)/
+ sizeof(NRDuringLCP_NonMaster_codes[0]);
+ for(int i = 0; i<sz; i++) {
+
+ int error = NRDuringLCP_NonMaster_codes[i];
+ int nodeId = getRandomNodeId(_restarter);
+ int masterNodeId = _restarter.getMasterNodeId();
+ CHECK(masterNodeId > 0, "getMasterNodeId failed");
+
+ while (nodeId == masterNodeId) {
+ nodeId = getRandomNodeId(_restarter);
+ }
+
+ ndbout << _restart->m_name << " restarting non-master node = " << nodeId
+ << " error code = " << error << endl;
+
+ int val = DumpStateOrd::CmvmiSetRestartOnErrorInsert;
+ CHECK(_restarter.dumpStateAllNodes(&val, 1) == 0,
+ "failed to set RestartOnErrorInsert");
+
+ CHECK(_restarter.insertErrorInNode(nodeId, error) == 0,
+ "failed to set error insert");
+
+ CHECK(_restarter.waitNodesNoStart(&nodeId, 1, 300) == 0,
+ "failed to wait no start");
+
+ CHECK(_restarter.startNodes(&nodeId, 1) == 0,
+ "failed to start node");
+
+ CHECK(_restarter.waitClusterStarted(300) == 0,
+ "waitClusterStarted failed");
+
+ {
+ int val = DumpStateOrd::DihMinTimeBetweenLCP;
+ CHECK(_restarter.dumpStateOneNode(nodeId, &val, 1) == 0,
+ "failed to set error insert");
+ }
+ }
+
+ return NDBT_OK;
+}
+
+int stopOnError(NdbRestarter& _restarter,
+ const NdbRestarts::NdbRestart* _restart){
+
+ myRandom48Init(NdbTick_CurrentMillisecond());
+
+ int randomId = myRandom48(_restarter.getNumDbNodes());
+ int nodeId = _restarter.getDbNodeId(randomId);
+
+ do {
+ g_info << _restart->m_name << ": node = " << nodeId
+ << endl;
+
+ CHECK(_restarter.waitClusterStarted(300) == 0,
+ "waitClusterStarted failed");
+
+ int val = DumpStateOrd::NdbcntrTestStopOnError;
+ CHECK(_restarter.dumpStateOneNode(nodeId, &val, 1) == 0,
+ "failed to set NdbcntrTestStopOnError");
+
+ NdbSleep_SecSleep(3);
+
+ CHECK(_restarter.waitClusterStarted(300) == 0,
+ "waitClusterStarted failed");
+ } while (false);
+
+ return NDBT_OK;
+};
+
+int getRandomNodeId(NdbRestarter& _restarter) {
+ myRandom48Init(NdbTick_CurrentMillisecond());
+ int randomId = myRandom48(_restarter.getNumDbNodes());
+ int nodeId = _restarter.getDbNodeId(randomId);
+
+ return nodeId;
+}
diff --git a/ndb/test/src/UtilTransactions.cpp b/ndb/test/src/UtilTransactions.cpp
new file mode 100644
index 00000000000..927c0b99bc3
--- /dev/null
+++ b/ndb/test/src/UtilTransactions.cpp
@@ -0,0 +1,1393 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include "UtilTransactions.hpp"
+#include <NdbSleep.h>
+#include <NdbScanFilter.hpp>
+
+#define VERBOSE 0
+
+UtilTransactions::UtilTransactions(const NdbDictionary::Table& _tab):
+ tab(_tab){
+ m_defaultClearMethod = 3;
+}
+
+UtilTransactions::UtilTransactions(Ndb* ndb, const char * name) :
+ tab(* ndb->getDictionary()->getTable(name)){
+ m_defaultClearMethod = 3;
+}
+
+#define RESTART_SCAN 99
+
+#define RETURN_FAIL(err) return (err.code != 0 ? err.code : NDBT_FAILED)
+
+int
+UtilTransactions::clearTable(Ndb* pNdb,
+ int records,
+ int parallelism){
+ if(m_defaultClearMethod == 1){
+ return clearTable1(pNdb, records, parallelism);
+ } else if(m_defaultClearMethod == 2){
+ return clearTable2(pNdb, records, parallelism);
+ } else {
+ return clearTable3(pNdb, records, parallelism);
+ }
+}
+
+int
+UtilTransactions::clearTable1(Ndb* pNdb,
+ int records,
+ int parallelism){
+ // Scan all records exclusive and delete
+ // them one by one
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: Has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ RETURN_FAIL(err);
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ NdbError err = pNdb->getNdbError();
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ RETURN_FAIL(err);
+ }
+
+ check = pOp->openScanExclusive(parallelism);
+ if( check == -1 ) {
+ NdbError err = pNdb->getNdbError();
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ RETURN_FAIL(err);
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ NdbError err = pNdb->getNdbError();
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ RETURN_FAIL(err);
+ }
+#if 0
+ // It's not necessary to read and PK's
+ // Information about the PK's are sent in
+ // KEYINFO20 signals anyway and used by takeOverScan
+
+ // Read the primary keys from this table
+ for(int a=0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey()){
+ if(pOp->getValue(tab.getColumn(a)->getName()) == NULL){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ RETURN_FAIL(err);
+ }
+ }
+ }
+#endif
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ RETURN_FAIL(err);
+ }
+
+ int eof;
+ int rows = 0;
+
+ eof = pTrans->nextScanResult();
+ while(eof == 0){
+ rows++;
+
+ int res = takeOverAndDeleteRecord(pNdb, pOp);
+ if(res == RESTART_SCAN){
+ eof = -2;
+ continue;
+ }
+
+ if (res != 0){
+ NdbError err = pNdb->getNdbError(res);
+ pNdb->closeTransaction(pTrans);
+ RETURN_FAIL(err);
+ }
+
+ eof = pTrans->nextScanResult();
+ }
+
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ // If error = 488 there should be no limit on number of retry attempts
+ if (err.code != 488)
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ RETURN_FAIL(err);
+ }
+
+ if(eof == -2){
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ g_info << rows << " deleted" << endl;
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+int
+UtilTransactions::clearTable2(Ndb* pNdb,
+ int records,
+ int parallelism){
+ // Scan all records exclusive and delete
+ // them one by one
+ int retryAttempt = 0;
+ const int retryMax = 10;
+ int deletedRows = 0;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanExclusive(parallelism);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+#if 0
+ // It's not necessary to read any PK's
+ // Information about the PK's are sent in
+ // KEYINFO20 signals anyway and used by takeOverScan
+
+ // Read the primary keys from this table
+ for(int a=0; a<tab.getNoOfColumns(); a++){
+ if (tab.getColumn(a)->getPrimaryKey()){
+ if(pOp->getValue(tab.getColumn(a)->getName()) == NULL){
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+ }
+ }
+#endif
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ NdbConnection* pDelTrans;
+
+ while((eof = pTrans->nextScanResult(true)) == 0){
+ pDelTrans = pNdb->startTransaction();
+ if (pDelTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+#if 0
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+#endif
+ ERR(err);
+ pNdb->closeTransaction(pDelTrans);
+ return NDBT_FAILED;
+ }
+ do {
+ deletedRows++;
+ if (addRowToDelete(pNdb, pDelTrans, pOp) != 0){
+ pNdb->closeTransaction(pDelTrans);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ } while((eof = pTrans->nextScanResult(false)) == 0);
+
+ check = pDelTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pDelTrans->getNdbError();
+ ERR(err);
+ pNdb->closeTransaction(pDelTrans);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ pNdb->closeTransaction(pDelTrans);
+
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ // If error = 488 there should be no limit on number of retry attempts
+ if (err.code != 488)
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ g_info << deletedRows << " rows deleted" << endl;
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+int
+UtilTransactions::clearTable3(Ndb* pNdb,
+ int records,
+ int parallelism){
+ // Scan all records exclusive and delete
+ // them one by one
+ int retryAttempt = 0;
+ const int retryMax = 10;
+ int deletedRows = 0;
+ int check;
+ NdbConnection *pTrans;
+ NdbScanOperation *pOp;
+ NdbError err;
+
+ while (true){
+ restart:
+ if (retryAttempt++ >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ err = pNdb->getNdbError();
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ continue;
+ }
+ goto failed;
+ }
+
+ pOp = pTrans->getNdbScanOperation(tab.getName());
+ if (pOp == NULL) {
+ goto failed;
+ }
+
+ NdbResultSet * rs = pOp->readTuplesExclusive(parallelism);
+ if( rs == 0 ) {
+ goto failed;
+ }
+
+ if(pTrans->execute(NoCommit) != 0){
+ err = pTrans->getNdbError();
+ if(err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ continue;
+ }
+ goto failed;
+ }
+
+ while((check = rs->nextResult(true)) == 0){
+ do {
+ if (rs->deleteTuple() != 0){
+ goto failed;
+ }
+ deletedRows++;
+ } while((check = rs->nextResult(false)) == 0);
+
+ if(check != -1){
+ check = pTrans->execute(Commit);
+ pTrans->releaseCompletedOperations();
+ }
+
+ err = pTrans->getNdbError();
+ if(check == -1){
+ if(err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ goto restart;
+ }
+ goto failed;
+ }
+ }
+ pNdb->closeTransaction(pTrans);
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+
+ failed:
+ if(pTrans != 0) pNdb->closeTransaction(pTrans);
+ ERR(err);
+ return (err.code != 0 ? err.code : NDBT_FAILED);
+}
+
+int
+UtilTransactions::copyTableData(Ndb* pNdb,
+ const char* destName){
+ // Scan all records and copy
+ // them to destName table
+ int retryAttempt = 0;
+ const int retryMax = 10;
+ int insertedRows = 0;
+ int parallelism = 240;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+ NDBT_ResultRow row(tab);
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->openScanRead(parallelism);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Read all attributes
+ for (int a = 0; a < tab.getNoOfColumns(); a++){
+ if ((row.attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ NdbConnection* pInsTrans;
+
+ while((eof = pTrans->nextScanResult(true)) == 0){
+ pInsTrans = pNdb->startTransaction();
+ if (pInsTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+ ERR(err);
+ pNdb->closeTransaction(pInsTrans);
+ return NDBT_FAILED;
+ }
+ do {
+ insertedRows++;
+ if (addRowToInsert(pNdb, pInsTrans, row, destName) != 0){
+ pNdb->closeTransaction(pInsTrans);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ } while((eof = pTrans->nextScanResult(false)) == 0);
+
+ check = pInsTrans->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pInsTrans->getNdbError();
+ ERR(err);
+ pNdb->closeTransaction(pInsTrans);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ pNdb->closeTransaction(pInsTrans);
+
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ // If error = 488 there should be no limit on number of retry attempts
+ if (err.code != 488)
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ g_info << insertedRows << " rows copied" << endl;
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+int
+UtilTransactions::addRowToDelete(Ndb* pNdb,
+ NdbConnection* pDelTrans,
+ NdbOperation* pOrgOp){
+
+ NdbOperation* pDelOp = pOrgOp->takeOverForDelete(pDelTrans);
+ if (pDelOp == NULL){
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+ return NDBT_OK;
+}
+
+int
+UtilTransactions::addRowToInsert(Ndb* pNdb,
+ NdbConnection* pInsTrans,
+ NDBT_ResultRow & row,
+ const char *insertTabName){
+
+ int check;
+ NdbOperation* pInsOp;
+
+ pInsOp = pInsTrans->getNdbOperation(insertTabName);
+ if (pInsOp == NULL) {
+ ERR(pInsTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ check = pInsOp->insertTuple();
+ if( check == -1 ) {
+ ERR(pInsTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+
+ // Set all attributes
+ for (int a = 0; a < tab.getNoOfColumns(); a++){
+ NdbRecAttr* r = row.attributeStore(a);
+ int sz = r->attrSize() * r->arraySize();
+ if (pInsOp->setValue(tab.getColumn(a)->getName(),
+ r->aRef(),
+ sz) != 0) {
+ ERR(pInsTrans->getNdbError());
+ return NDBT_FAILED;
+ }
+ }
+
+ return NDBT_OK;
+}
+
+// Take over one record from pOrgOp and delete it
+int
+UtilTransactions::takeOverAndDeleteRecord(Ndb* pNdb,
+ NdbOperation* pOrgOp){
+
+ int retryAttempt = 0;
+ const int retryMax = 10;
+ int check;
+ NdbConnection *pDelTrans;
+ NdbOperation *pDelOp;
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pDelTrans = pNdb->startTransaction();
+ if (pDelTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ if ((pDelOp = pOrgOp->takeOverForDelete(pDelTrans)) == NULL){
+ ERR(pNdb->getNdbError());
+ return NDBT_FAILED;
+ }
+
+#if 0
+ // It should not be necessary to call deleteTuple HERE!!!
+ check = pDelOp->deleteTuple();
+ if( check == -1 ) {
+ ERR(pDelTrans->getNdbError());
+ pNdb->closeTransaction(pDelTrans);
+ return NDBT_FAILED;
+ }
+#endif
+
+ check = pDelTrans->execute( Commit );
+ if(check == -1 ) {
+ const NdbError err = pDelTrans->getNdbError();
+ pNdb->closeTransaction(pDelTrans);
+
+ ERR(err);
+ if(err.code == 250 || err.code == 499)
+ return RESTART_SCAN;
+
+ switch(err.status){
+ case NdbError::Success:
+ g_info << "ERROR: NdbError reports success when transcaction failed"
+ << endl;
+ RETURN_FAIL(err);
+ break;
+
+ case NdbError::TemporaryError:
+ NdbSleep_MilliSleep(50+50*retryAttempt);
+ retryAttempt++;
+ continue;
+ break;
+
+ case NdbError::UnknownResult:
+ RETURN_FAIL(err);
+ break;
+
+ default:
+ case NdbError::PermanentError:
+ switch (err.classification){
+ default:
+ RETURN_FAIL(err);
+ break;
+ }
+ break;
+ }
+ }
+ else{
+ pNdb->closeTransaction(pDelTrans);
+ }
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+
+
+
+int
+UtilTransactions::scanReadRecords(Ndb* pNdb,
+ int parallelism,
+ bool exclusive,
+ int records,
+ int noAttribs,
+ int *attrib_list,
+ ReadCallBackFn* fn){
+
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+ NDBT_ResultRow row(tab);
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if (exclusive == true)
+ check = pOp->openScanExclusive(parallelism);
+ else
+ check = pOp->openScanRead(parallelism);
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Call getValue for all the attributes supplied in attrib_list
+ // ************************************************
+ for (int a = 0; a < noAttribs; a++){
+ if (attrib_list[a] < tab.getNoOfColumns()){
+ g_info << "getValue(" << attrib_list[a] << ")" << endl;
+ if ((row.attributeStore(attrib_list[a]) =
+ pOp->getValue(tab.getColumn(attrib_list[a])->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+ }
+ // *************************************************
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ eof = pTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+
+ // Call callback for each record returned
+ if(fn != NULL)
+ fn(&row);
+ eof = pTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+ g_info << rows << " rows have been read" << endl;
+ if (records != 0 && rows != records){
+ g_info << "Check expected number of records failed" << endl
+ << " expected=" << records <<", " << endl
+ << " read=" << rows << endl;
+ return NDBT_FAILED;
+ }
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+int
+UtilTransactions::selectCount(Ndb* pNdb,
+ int parallelism,
+ int* count_rows,
+ ScanLock lock,
+ NdbConnection* pBuddyTrans){
+
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->hupp(pBuddyTrans);
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ switch(lock){
+ case SL_ReadHold:
+ check = pOp->openScanReadHoldLock(parallelism);
+ break;
+ case SL_Exclusive:
+ check = pOp->openScanExclusive(parallelism);
+ break;
+ case SL_Read:
+ default:
+ check = pOp->openScanRead(parallelism);
+ }
+
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if(0){
+ NdbScanFilter sf(pOp);
+ sf.begin(NdbScanFilter::OR);
+ sf.eq(2, (Uint32)30);
+ sf.end();
+ } else {
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ eof = pTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+ eof = pTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ if (count_rows != NULL){
+ *count_rows = rows;
+ }
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+
+
+int
+UtilTransactions::verifyIndex(Ndb* pNdb,
+ const char* indexName,
+ int parallelism,
+ bool transactional){
+
+
+ const NdbDictionary::Index* pIndex
+ = pNdb->getDictionary()->getIndex(indexName, tab.getName());
+ if (pIndex == 0){
+ ndbout << " Index " << indexName << " does not exist!" << endl;
+ return NDBT_FAILED;
+ }
+
+ switch (pIndex->getType()){
+ case UniqueHashIndex:
+ case OrderedIndex:
+ return verifyUniqueIndex(pNdb, indexName, parallelism, transactional);
+ break;
+ default:
+ ndbout << "Unknown index type" << endl;
+ break;
+ }
+
+ return NDBT_FAILED;
+}
+
+int
+UtilTransactions::verifyUniqueIndex(Ndb* pNdb,
+ const char* indexName,
+ int parallelism,
+ bool transactional){
+
+ /**
+ * Scan all rows in TABLE and for each found row make one read in
+ * TABLE and one using INDEX_TABLE. Then compare the two returned
+ * rows. They should be equal!
+ *
+ */
+
+ if (scanAndCompareUniqueIndex(pNdb,
+ indexName,
+ parallelism,
+ transactional) != NDBT_OK){
+ return NDBT_FAILED;
+ }
+
+
+ return NDBT_OK;
+
+}
+
+
+int
+UtilTransactions::scanAndCompareUniqueIndex(Ndb* pNdb,
+ const char * indexName,
+ int parallelism,
+ bool transactional){
+
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+ NDBT_ResultRow row(tab);
+
+ parallelism = 1;
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return NDBT_FAILED;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return NDBT_FAILED;
+ }
+
+ pOp = pTrans->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ if(transactional){
+ check = pOp->openScanReadHoldLock(parallelism);
+ } else {
+ check = pOp->openScanRead(parallelism);
+ }
+
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ // Read all attributes
+ for (int a = 0; a < tab.getNoOfColumns(); a++){
+ if ((row.attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+ }
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ int eof;
+ int rows = 0;
+ eof = pTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+
+ // ndbout << row.c_str().c_str() << endl;
+
+
+ if (readRowFromTableAndIndex(pNdb,
+ pTrans,
+ indexName,
+ row) != NDBT_OK){
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+
+
+
+ eof = pTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ rows--;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return NDBT_FAILED;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ return NDBT_OK;
+ }
+ return NDBT_FAILED;
+}
+int
+UtilTransactions::readRowFromTableAndIndex(Ndb* pNdb,
+ NdbConnection* scanTrans,
+ const char * indexName,
+ NDBT_ResultRow& row ){
+ const NdbDictionary::Index* pIndex
+ = pNdb->getDictionary()->getIndex(indexName, tab.getName());
+
+ if (pIndex == 0){
+ ndbout << " Index " << indexName << " does not exist!" << endl;
+ return NDBT_FAILED;
+ }
+
+ NdbDictionary::Index::Type indexType= pIndex->getType();
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans1=NULL;
+ NdbResultSet *cursor= NULL;
+ NdbOperation *pOp;
+
+ int return_code= NDBT_FAILED;
+
+ // Allocate place to store the result
+ NDBT_ResultRow tabRow(tab);
+ NDBT_ResultRow indexRow(tab);
+
+
+ while (true){
+ if(retryAttempt)
+ ndbout_c("retryAttempt %d", retryAttempt);
+ if (retryAttempt >= retryMax){
+ g_info << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ goto close_all;
+ }
+
+ pTrans1 = pNdb->hupp(scanTrans); //startTransaction();
+ if (pTrans1 == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+
+ if(err.code == 0){
+ return_code = NDBT_OK;
+ goto close_all;
+ }
+ ERR(err);
+ goto close_all;
+ }
+
+ /**
+ * Read the record from TABLE
+ */
+ pOp = pTrans1->getNdbOperation(tab.getName());
+ if (pOp == NULL) {
+ ERR(pTrans1->getNdbError());
+ goto close_all;
+ }
+
+ check = pOp->readTuple();
+ if( check == -1 ) {
+ ERR(pTrans1->getNdbError());
+ pNdb->closeTransaction(pTrans1);
+ goto close_all;
+ }
+
+ // Define primary keys
+#if VERBOSE
+ printf("PK: ");
+#endif
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ const NdbDictionary::Column* attr = tab.getColumn(a);
+ if (attr->getPrimaryKey() == true){
+ if (pOp->equal(attr->getName(), row.attributeStore(a)->aRef()) != 0){
+ ERR(pTrans1->getNdbError());
+ goto close_all;
+ }
+#if VERBOSE
+ printf("%s = %d: ", attr->getName(), row.attributeStore(a)->aRef());
+#endif
+ }
+ }
+#if VERBOSE
+ printf("\n");
+#endif
+ // Read all attributes
+#if VERBOSE
+ printf("Reading %u attributes: ", tab.getNoOfColumns());
+#endif
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ if((tabRow.attributeStore(a) =
+ pOp->getValue(tab.getColumn(a)->getName())) == 0) {
+ ERR(pTrans1->getNdbError());
+ goto close_all;
+ }
+#if VERBOSE
+ printf("%s ", tab.getColumn(a)->getName());
+#endif
+ }
+#if VERBOSE
+ printf("\n");
+#endif
+
+ /**
+ * Read the record from INDEX_TABLE
+ */
+ NdbIndexOperation* pIndexOp= NULL;
+ NdbScanOperation *pScanOp= NULL;
+ {
+ void* pOpCheck= NULL;
+ if (indexType == NdbDictionary::Index::UniqueHashIndex) {
+ pOpCheck= pIndexOp= pTrans1->getNdbIndexOperation(indexName, tab.getName());
+ } else {
+ pOpCheck= pScanOp= pTrans1->getNdbScanOperation(indexName, tab.getName());
+ }
+
+ if (pOpCheck == NULL) {
+ ERR(pTrans1->getNdbError());
+ goto close_all;
+ }
+ }
+
+ {
+ bool not_ok;
+ if (pIndexOp) {
+ not_ok = pIndexOp->readTuple() == -1;
+ } else {
+ not_ok = (cursor= pScanOp->readTuples()) == 0;
+ }
+
+ if( not_ok ) {
+ ERR(pTrans1->getNdbError());
+ goto close_all;
+ }
+ }
+
+ // Define primary keys for index
+#if VERBOSE
+ printf("SI: ");
+#endif
+ for(int a = 0; a<(int)pIndex->getNoOfColumns(); a++){
+ const NdbDictionary::Column * col = pIndex->getColumn(a);
+
+ int r;
+ if (pIndexOp)
+ r = pIndexOp->equal(col->getName(), row.attributeStore(col->getName())->aRef());
+ else {
+ // setBound not possible for null attributes
+ if ( !row.attributeStore(col->getName())->isNULL() ) {
+ r = pScanOp->setBound(col->getName(),
+ NdbOperation::BoundEQ,
+ row.attributeStore(col->getName())->aRef());
+ }
+ }
+ if (r != 0){
+ ERR(pTrans1->getNdbError());
+ goto close_all;
+ }
+#if VERBOSE
+ printf("%s = %d: ", col->getName(), row.attributeStore(a)->aRef());
+#endif
+ }
+#if VERBOSE
+ printf("\n");
+#endif
+
+ // Read all attributes
+#if VERBOSE
+ printf("Reading %u attributes: ", tab.getNoOfColumns());
+#endif
+ for(int a = 0; a<tab.getNoOfColumns(); a++){
+ void* pCheck;
+
+ if (pIndexOp)
+ pCheck= indexRow.attributeStore(a)=
+ pIndexOp->getValue(tab.getColumn(a)->getName());
+ else
+ pCheck= indexRow.attributeStore(a)=
+ pScanOp->getValue(tab.getColumn(a)->getName());
+
+ if(pCheck == NULL) {
+ ERR(pTrans1->getNdbError());
+ goto close_all;
+ }
+#if VERBOSE
+ printf("%s ", tab.getColumn(a)->getName());
+#endif
+ }
+#if VERBOSE
+ printf("\n");
+#endif
+
+ check = pTrans1->execute(Commit);
+ if( check == -1 ) {
+ const NdbError err = pTrans1->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ ERR(err);
+ pNdb->closeTransaction(pTrans1);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ndbout << "Error when comparing records - normal op" << endl;
+ ERR(err);
+ ndbout << "row: " << row.c_str().c_str() << endl;
+ goto close_all;
+ }
+
+ /**
+ * Compare the two rows
+ */
+ if (pScanOp) {
+ if (cursor->nextResult() != 0){
+ const NdbError err = pTrans1->getNdbError();
+ ERR(err);
+ ndbout << "Error when comparing records - index op next_result missing" << endl;
+ ndbout << "row: " << row.c_str().c_str() << endl;
+ goto close_all;
+ }
+ }
+ if (!(tabRow.c_str() == indexRow.c_str())){
+ ndbout << "Error when comapring records" << endl;
+ ndbout << " tabRow: \n" << tabRow.c_str().c_str() << endl;
+ ndbout << " indexRow: \n" << indexRow.c_str().c_str() << endl;
+ goto close_all;
+ }
+ if (pScanOp) {
+ if (cursor->nextResult() == 0){
+ ndbout << "Error when comparing records - index op next_result to many" << endl;
+ ndbout << "row: " << row.c_str().c_str() << endl;
+ goto close_all;
+ }
+ }
+ return_code= NDBT_OK;
+ goto close_all;
+ }
+
+ close_all:
+ if (cursor)
+ cursor->close();
+ if (pTrans1)
+ pNdb->closeTransaction(pTrans1);
+
+ return return_code;
+}
diff --git a/ndb/test/tools/Makefile b/ndb/test/tools/Makefile
new file mode 100644
index 00000000000..b8e90ae207f
--- /dev/null
+++ b/ndb/test/tools/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+DIRS := hugoCalculator hugoFill hugoLoad hugoLockRecords \
+ hugoPkDelete hugoPkRead hugoPkReadRecord hugoPkUpdate \
+ hugoScanRead hugoScanUpdate restart waiter
+
+include $(NDB_TOP)/Epilogue.mk
+
+_bins_ndbapi : _libs_src
diff --git a/ndb/test/tools/hugoCalculator/Makefile b/ndb/test/tools/hugoCalculator/Makefile
new file mode 100644
index 00000000000..a29deeaacd3
--- /dev/null
+++ b/ndb/test/tools/hugoCalculator/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoCalculator
+
+# Source files of non-templated classes (.C files)
+SOURCES = hugoCalculator.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/tools/hugoCalculator/hugoCalculator.cpp b/ndb/test/tools/hugoCalculator/hugoCalculator.cpp
new file mode 100644
index 00000000000..4cb801d3d73
--- /dev/null
+++ b/ndb/test/tools/hugoCalculator/hugoCalculator.cpp
@@ -0,0 +1,68 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <stdlib.h>
+#include <NdbSleep.h>
+#include <NDBT_Tables.hpp>
+#include <getarg.h>
+#include <NDBT.hpp>
+#include <HugoCalculator.hpp>
+
+//extern NdbOut g_info;
+
+int main(int argc, const char** argv)
+{
+ int _row = 0;
+ int _column = 0;
+ int _updates = 0;
+ const char* _tableName = NULL;
+
+ struct getargs args[] = {
+ { "row", 'r', arg_integer, &_row, "The row number", "row" },
+ { "column", 'c', arg_integer, &_column, "The column id", "column" },
+ { "updates", 'u', arg_integer, &_updates, "# of updates", "updates" }
+ };
+
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+
+ if(getarg(args, num_args, argc, argv, &optind) || argv[optind] == NULL) {
+ arg_printusage(args, num_args, argv[0], "table name\n");
+ return NDBT_WRONGARGS;
+ }
+ // Check if table name is supplied
+ if (argv[optind] != NULL)
+ _tableName = argv[optind];
+
+
+ const NdbDictionary::Table* table = NDBT_Tables::getTable(_tableName);
+ const NdbDictionary::Column * attribute = table->getColumn(_column);
+
+ g_info << "Table " << _tableName << endl
+ << "Row: " << _row << ", "
+ << "Column(" << attribute->getName() << ")"
+ << "[" << attribute->getType() << "]"
+ << ", Updates: " << _updates
+ << endl;
+
+ HugoCalculator calc(*table);
+ char buf[8000];
+ g_info << "Value: " << calc.calcValue(_row, _column, _updates, buf)
+ << endl;
+
+ return 0;
+}
diff --git a/ndb/test/tools/hugoFill/Makefile b/ndb/test/tools/hugoFill/Makefile
new file mode 100644
index 00000000000..3da745810b6
--- /dev/null
+++ b/ndb/test/tools/hugoFill/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoFill
+
+# Source files of non-templated classes (.C files)
+SOURCES = hugoFill.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/tools/hugoFill/hugoFill.cpp b/ndb/test/tools/hugoFill/hugoFill.cpp
new file mode 100644
index 00000000000..748623cb253
--- /dev/null
+++ b/ndb/test/tools/hugoFill/hugoFill.cpp
@@ -0,0 +1,78 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NdbSleep.h>
+#include <NDBT.hpp>
+#include <HugoTransactions.hpp>
+#include <getarg.h>
+
+
+int main(int argc, const char** argv){
+
+ int _records = 0;
+ const char* _tabname = NULL;
+ int _help = 0;
+ int _batch = 512;
+
+ struct getargs args[] = {
+ { "batch", 'b', arg_integer, &_batch, "Number of operations in each transaction", "batch" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will load one table in Ndb with calculated data \n"\
+ "until the database is full. \n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ // Connect to Ndb and wait for it to become ready
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table* pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ HugoTransactions hugoTrans(*pTab);
+ if (hugoTrans.fillTable(&MyNdb,
+ _batch) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/test/tools/hugoLoad/Makefile b/ndb/test/tools/hugoLoad/Makefile
new file mode 100644
index 00000000000..7c5756d0d41
--- /dev/null
+++ b/ndb/test/tools/hugoLoad/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoLoad
+
+# Source files of non-templated classes (.C files)
+SOURCES = hugoLoad.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/tools/hugoLoad/hugoLoad.cpp b/ndb/test/tools/hugoLoad/hugoLoad.cpp
new file mode 100644
index 00000000000..be7f878d106
--- /dev/null
+++ b/ndb/test/tools/hugoLoad/hugoLoad.cpp
@@ -0,0 +1,82 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NdbSleep.h>
+#include <NDBT.hpp>
+#include <HugoTransactions.hpp>
+#include <getarg.h>
+
+
+int main(int argc, const char** argv){
+
+ int _records = 0;
+ const char* _tabname = NULL;
+ int _help = 0;
+ int _batch = 512;
+
+ struct getargs args[] = {
+ { "records", 'r', arg_integer, &_records, "Number of records", "recs" },
+ { "batch", 'b', arg_integer, &_batch, "Number of operations in each transaction", "batch" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will load one table in Ndb with calculated data. \n"\
+ "This means that it is possible to check the validity of the data \n"\
+ "at a later time. The last column in each table is used as an update \n"\
+ "counter, it's initialised to zero and should be incremented for each \n"\
+ "update of the record. \n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _records == 0 || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ // Connect to Ndb and wait for it to become ready
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table* pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ HugoTransactions hugoTrans(*pTab);
+ if (hugoTrans.loadTable(&MyNdb,
+ _records,
+ _batch) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/test/tools/hugoLockRecords/Makefile b/ndb/test/tools/hugoLockRecords/Makefile
new file mode 100644
index 00000000000..3235750cbf8
--- /dev/null
+++ b/ndb/test/tools/hugoLockRecords/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoLockRecords
+
+SOURCES := hugoLockRecords.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/tools/hugoLockRecords/hugoLockRecords.cpp b/ndb/test/tools/hugoLockRecords/hugoLockRecords.cpp
new file mode 100644
index 00000000000..90c08649ec2
--- /dev/null
+++ b/ndb/test/tools/hugoLockRecords/hugoLockRecords.cpp
@@ -0,0 +1,90 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <HugoTransactions.hpp>
+
+int main(int argc, const char** argv){
+
+ int _records = 0;
+ int _loops = 1;
+ int _percentVal = 1;
+ int _lockTime = 1000;
+ const char* _tabname = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "loops", 'l', arg_integer, &_loops, "number of times to run this program(0=infinite loop)", "loops" },
+ { "records", 'r', arg_integer, &_records, "Number of records", "recs" },
+ { "locktime", 't', arg_integer, &_lockTime, "Time in ms to hold lock(default=1000)", "ms" },
+ { "percent", 'p', arg_integer, &_percentVal, "Percent of records to lock(default=1%)", "%" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will lock p% of the records in the table for x milliseconds\n"\
+ "then it will lock the next 1% and continue to do so until it has locked \n"\
+ "all records in the table\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _records == 0 || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table * pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ HugoTransactions hugoTrans(*pTab);
+ int i = 0;
+ while (i<_loops || _loops==0) {
+ ndbout << i << ": ";
+ if (hugoTrans.lockRecords(&MyNdb, _records, _percentVal, _lockTime) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+ i++;
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
diff --git a/ndb/test/tools/hugoPkDelete/Makefile b/ndb/test/tools/hugoPkDelete/Makefile
new file mode 100644
index 00000000000..e6d53611c54
--- /dev/null
+++ b/ndb/test/tools/hugoPkDelete/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoPkDelete
+
+SOURCES := hugoPkDel.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/tools/hugoPkDelete/hugoPkDel.cpp b/ndb/test/tools/hugoPkDelete/hugoPkDel.cpp
new file mode 100644
index 00000000000..f77dc21bd0b
--- /dev/null
+++ b/ndb/test/tools/hugoPkDelete/hugoPkDel.cpp
@@ -0,0 +1,86 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <HugoTransactions.hpp>
+
+int main(int argc, const char** argv){
+
+ int _records = 0;
+ int _loops = 1;
+ int _batch = 0;
+ const char* _tabname = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "loops", 'l', arg_integer, &_loops, "number of times to run this program(0=infinite loop)", "loops" },
+ // { "batch", 'b', arg_integer, &_batch, "batch value", "batch" },
+ { "records", 'r', arg_integer, &_records, "Number of records", "records" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will delete all records in a table using PK \n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _records == 0 || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table * pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ HugoTransactions hugoTrans(*pTab);
+ int i = 0;
+ while (i<_loops || _loops==0) {
+ ndbout << i << ": ";
+ if (hugoTrans.pkDelRecords(&MyNdb, _records) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+ i++;
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
diff --git a/ndb/test/tools/hugoPkRead/Makefile b/ndb/test/tools/hugoPkRead/Makefile
new file mode 100644
index 00000000000..03580dc0d18
--- /dev/null
+++ b/ndb/test/tools/hugoPkRead/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoPkRead
+
+SOURCES := hugoPkRead.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/tools/hugoPkRead/hugoPkRead.cpp b/ndb/test/tools/hugoPkRead/hugoPkRead.cpp
new file mode 100644
index 00000000000..2e9c2c35260
--- /dev/null
+++ b/ndb/test/tools/hugoPkRead/hugoPkRead.cpp
@@ -0,0 +1,91 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <HugoTransactions.hpp>
+
+
+int main(int argc, const char** argv){
+
+ int _records = 0;
+ int _loops = 1;
+ int _abort = 0;
+ int _batch = 1;
+ const char* _tabname = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "aborts", 'a', arg_integer, &_abort, "percent of transactions that are aborted", "abort%" },
+ { "loops", 'l', arg_integer, &_loops, "number of times to run this program(0=infinite loop)", "loops" },
+ { "batch", 'b', arg_integer, &_batch, "batch value(not 0)", "batch" },
+ { "records", 'r', arg_integer, &_records, "Number of records", "records" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will read 'r' records from one table in Ndb. \n"\
+ "It will verify every column read by calculating the expected value.\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _records == 0 || _batch == 0 || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table * pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ HugoTransactions hugoTrans(*pTab);
+ int i = 0;
+ while (i<_loops || _loops==0) {
+ ndbout << i << ": ";
+ if (hugoTrans.pkReadRecords(&MyNdb, _records, _batch) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+ i++;
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
diff --git a/ndb/test/tools/hugoPkReadRecord/Makefile b/ndb/test/tools/hugoPkReadRecord/Makefile
new file mode 100644
index 00000000000..158a79a5666
--- /dev/null
+++ b/ndb/test/tools/hugoPkReadRecord/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoPkReadRecord
+
+# Source files of non-templated classes (.C files)
+SOURCES = hugoPkReadRecord.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/tools/hugoPkReadRecord/hugoPkReadRecord.cpp b/ndb/test/tools/hugoPkReadRecord/hugoPkReadRecord.cpp
new file mode 100644
index 00000000000..6335c391bc3
--- /dev/null
+++ b/ndb/test/tools/hugoPkReadRecord/hugoPkReadRecord.cpp
@@ -0,0 +1,193 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <NdbOut.hpp>
+#include <stdlib.h>
+#include <NdbSleep.h>
+#include <NDBT_Tables.hpp>
+#include <getarg.h>
+#include <NDBT.hpp>
+#include <Ndb.hpp>
+#include <NdbDictionary.hpp>
+
+//extern NdbOut g_info;
+
+int main(int argc, const char** argv)
+{
+ int _row = 0;
+ int _hex = 0;
+ int _primaryKey = 0;
+ const char* _tableName = NULL;
+
+ struct getargs args[] = {
+ { "row", 'r',
+ arg_integer, &_row, "The row number", "row" },
+ { "primarykey", 'p',
+ arg_integer, &_primaryKey, "The primary key", "primarykey" },
+ { "hex", 'h',
+ arg_flag, &_hex, "Print hex", "hex" }
+ };
+
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+
+ if(getarg(args, num_args, argc, argv, &optind) || argv[optind] == NULL) {
+ arg_printusage(args, num_args, argv[0], "table name\n");
+ return NDBT_WRONGARGS;
+ }
+ // Check if table name is supplied
+ if (argv[optind] != NULL)
+ _tableName = argv[optind];
+
+
+ const NdbDictionary::Table* table = NDBT_Tables::getTable(_tableName);
+ // const NDBT_Attribute* attribute = table->getAttribute(_column);
+
+ g_info << "Table " << _tableName << endl
+ << "Row: " << _row << ", PrimaryKey: " << _primaryKey
+ << endl;
+
+ Ndb* ndb = new Ndb("TEST_DB");
+ if (ndb->init() == 0 && ndb->waitUntilReady(30) == 0)
+ {
+ NdbConnection* conn = ndb->startTransaction();
+ if (conn == NULL)
+ {
+ g_info << "ERROR: " << ndb->getNdbError() << endl;
+ delete ndb;
+ return -1;
+ }
+ NdbOperation* op = conn->getNdbOperation(_tableName);
+ if (op == NULL)
+ {
+ g_info << "ERROR: " << conn->getNdbError() << endl;
+ delete ndb;
+ return -1;
+ }
+ op->readTuple();
+ NdbRecAttr** data = new NdbRecAttr*[table->getNoOfColumns()];
+ for (int i = 0; i < table->getNoOfColumns(); i++)
+ {
+ const NdbDictionary::Column* c = table->getColumn(i);
+ if (c->getPrimaryKey())
+ {
+ op->equal(c->getName(), _primaryKey);
+ data[i] = op->getValue(c->getName(), NULL);
+ }
+ else
+ {
+ data[i] = op->getValue(c->getName(), NULL);
+ }
+ }
+
+ if (conn->execute(Commit) == 0)
+ {
+ // Print column names
+ for (int i = 0; i < table->getNoOfColumns(); i++)
+ {
+ const NdbDictionary::Column* c = table->getColumn(i);
+
+ g_info
+ << c->getName()
+ << "[" << c->getType() << "] ";
+ }
+ g_info << endl;
+
+ if (_hex)
+ {
+ g_info << hex;
+ }
+ for (int i = 0; i < table->getNoOfColumns(); i++)
+ {
+ NdbRecAttr* a = data[i];
+ switch(a->getType())
+ {
+ case NdbDictionary::Column::Char:
+ case NdbDictionary::Column::Varchar:
+ case NdbDictionary::Column::Binary:
+ case NdbDictionary::Column::Varbinary:
+ {
+ if (_hex)
+ {
+ char* b = a->aRef();
+ for (int j = 0; j < a->arraySize(); j++)
+ {
+ //ndbout_c("%x", b[j]);
+ g_info << hex << b[j] << "[" << dec << j << "]";
+ }
+ }
+ else
+ {
+ g_info << "\""
+ << a->aRef() << "\"";
+ }
+ g_info << " ";
+ }
+ break;
+
+ case NdbDictionary::Column::Int:
+ case NdbDictionary::Column::Unsigned:
+ {
+ g_info << a->int32_value() << " ";
+ }
+ break;
+
+ case NdbDictionary::Column::Bigint:
+ case NdbDictionary::Column::Bigunsigned:
+ {
+ g_info << a->int64_value() << " ";
+ }
+ break;
+
+ case NdbDictionary::Column::Float:
+ {
+ g_info << a->float_value() << " ";
+ }
+ break;
+
+ case NdbDictionary::Column::Undefined:
+ default:
+ {
+ g_info << "Undefined!!! ";
+ }
+ break;
+
+ } // case
+ g_info << " ";
+ } // for
+ g_info << endl;
+ } // if (conn
+ else
+ {
+ g_info << "Failed to commit read transaction... "
+ << conn->getNdbError()
+ << ", commitStatus = " << conn->commitStatus()
+ << endl;
+ }
+
+ delete[] data;
+
+ ndb->closeTransaction(conn);
+ } // if (ndb.init
+ else
+ {
+ g_info << "ERROR: Unable to connect to NDB, "
+ << ndb->getNdbError() << endl;
+ }
+ delete ndb;
+
+ return 0;
+}
diff --git a/ndb/test/tools/hugoPkUpdate/Makefile b/ndb/test/tools/hugoPkUpdate/Makefile
new file mode 100644
index 00000000000..48795b62206
--- /dev/null
+++ b/ndb/test/tools/hugoPkUpdate/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoPkUpdate
+
+SOURCES := hugoPkUpd.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/tools/hugoPkUpdate/hugoPkUpd.cpp b/ndb/test/tools/hugoPkUpdate/hugoPkUpd.cpp
new file mode 100644
index 00000000000..141d01e3aee
--- /dev/null
+++ b/ndb/test/tools/hugoPkUpdate/hugoPkUpd.cpp
@@ -0,0 +1,88 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <HugoTransactions.hpp>
+
+int main(int argc, const char** argv){
+
+ int _records = 0;
+ int _loops = 1;
+ int _abort = 0;
+ int _batch = 0;
+ const char* _tabname = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "aborts", 'a', arg_integer, &_abort, "percent of transactions that are aborted", "abort%" },
+ { "loops", 'l', arg_integer, &_loops, "number of times to run this program(0=infinite loop)", "loops" },
+ // { "batch", 'b', arg_integer, &_batch, "batch value", "batch" },
+ { "records", 'r', arg_integer, &_records, "Number of records", "records" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will update all records in a table using PK\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _records == 0 || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table * pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ HugoTransactions hugoTrans(*pTab);
+ int i = 0;
+ while (i<_loops || _loops==0) {
+ ndbout << "loop " << i << ": ";
+ if (hugoTrans.pkUpdateRecords(&MyNdb,
+ _records) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+ i++;
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/test/tools/hugoScanRead/Makefile b/ndb/test/tools/hugoScanRead/Makefile
new file mode 100644
index 00000000000..b88377c299e
--- /dev/null
+++ b/ndb/test/tools/hugoScanRead/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoScanRead
+
+SOURCES := hugoScanRead.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/tools/hugoScanRead/hugoScanRead.cpp b/ndb/test/tools/hugoScanRead/hugoScanRead.cpp
new file mode 100644
index 00000000000..2376280d004
--- /dev/null
+++ b/ndb/test/tools/hugoScanRead/hugoScanRead.cpp
@@ -0,0 +1,90 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <HugoTransactions.hpp>
+
+int main(int argc, const char** argv){
+
+ int _records = 0;
+ int _loops = 1;
+ int _abort = 0;
+ int _parallelism = 1;
+ const char* _tabname = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "aborts", 'a', arg_integer, &_abort, "percent of transactions that are aborted", "abort%" },
+ { "loops", 'l', arg_integer, &_loops, "number of times to run this program(0=infinite loop)", "loops" },
+ { "parallelism", 'p', arg_integer, &_parallelism, "parallelism(1-240)", "para" },
+ { "records", 'r', arg_integer, &_records, "Number of records", "recs" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ " tabname\n"\
+ "This program will scan read all records in one table in Ndb.\n"\
+ "It will verify every column read by calculating the expected value.\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || argv[optind] == NULL || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table * pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ HugoTransactions hugoTrans(*pTab);
+ int i = 0;
+ while (i<_loops || _loops==0) {
+ ndbout << i << ": ";
+ if(hugoTrans.scanReadRecords(&MyNdb,
+ 0,
+ _abort,
+ _parallelism) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+ i++;
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/test/tools/hugoScanUpdate/Makefile b/ndb/test/tools/hugoScanUpdate/Makefile
new file mode 100644
index 00000000000..ec0e07bfd84
--- /dev/null
+++ b/ndb/test/tools/hugoScanUpdate/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := hugoScanUpdate
+
+SOURCES := hugoScanUpd.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/test/tools/hugoScanUpdate/hugoScanUpd.cpp b/ndb/test/tools/hugoScanUpdate/hugoScanUpd.cpp
new file mode 100644
index 00000000000..56cd3b8c969
--- /dev/null
+++ b/ndb/test/tools/hugoScanUpdate/hugoScanUpd.cpp
@@ -0,0 +1,100 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <HugoTransactions.hpp>
+
+int main(int argc, const char** argv){
+
+ int _records = 0;
+ int _loops = 1;
+ int _parallelism = 1;
+ int _ver2 = 0;
+ const char* _tabname = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "loops", 'l', arg_integer, &_loops, "number of times to run this program(0=infinite loop)", "loops" },
+ { "parallelism", 'p', arg_integer, &_parallelism, "parallelism(1-240)", "para" },
+ { "records", 'r', arg_integer, &_records, "Number of records", "recs" },
+ { "ver2", '2', arg_flag, &_ver2, "Use version 2 of scanUpdateRecords", "" },
+ { "ver2", '1', arg_negative_flag, &_ver2, "Use version 1 of scanUpdateRecords (default)", "" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will scan update all records in one table in Ndb\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table * pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ HugoTransactions hugoTrans(*pTab);
+ int i = 0;
+ int res = NDBT_FAILED;
+ while (i<_loops || _loops==0) {
+ ndbout << i << ": ";
+ if (_ver2 == 0){
+ res = hugoTrans.scanUpdateRecords(&MyNdb,
+ _records,
+ 0,
+ _parallelism);
+ } else{
+ res = hugoTrans.scanUpdateRecords2(&MyNdb,
+ _records,
+ 0,
+ _parallelism);
+ }
+ if (res != NDBT_OK ){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+ i++;
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/test/tools/restart/Makefile b/ndb/test/tools/restart/Makefile
new file mode 100644
index 00000000000..05d9e98c5bc
--- /dev/null
+++ b/ndb/test/tools/restart/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := restart
+
+# Source files of non-templated classes (.C files)
+SOURCES = restart.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/tools/restart/restart.cpp b/ndb/test/tools/restart/restart.cpp
new file mode 100644
index 00000000000..f391aecabe1
--- /dev/null
+++ b/ndb/test/tools/restart/restart.cpp
@@ -0,0 +1,84 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <string.h>
+#include <NdbMain.h>
+#include <OutputStream.hpp>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+#include <NdbRestarter.hpp>
+#include <NDBT.hpp>
+#include <assert.h>
+#include <NdbStdio.h>
+
+int main(int argc, const char** argv){
+
+ const char* _hostName = NULL;
+ int _initial = 0;
+ int _help = 0;
+ int _wait = 1;
+
+
+ struct getargs args[] = {
+ { "initial", 'i', arg_flag, &_initial, "Do initial restart"},
+ { "wait", '\0', arg_negative_flag, &_wait, "Wait until restarted(default=true)"},
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "hostname:port\n"\
+ "This program will connect to the mgmsrv of a NDB cluster\n"\
+ " and restart the cluster. \n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _hostName = argv[optind];
+
+ NdbRestarter restarter(_hostName);
+ setOutputLevel(1); // Show only g_err
+ int result = NDBT_OK;
+ if (_initial){
+ ndbout << "Restarting cluster with initial restart" << endl;
+ if (restarter.restartAll(true, false, false) != 0)
+ result = NDBT_FAILED;
+ } else {
+ ndbout << "Restarting cluster " << endl;
+ if (restarter.restartAll() != 0)
+ result = NDBT_FAILED;
+ }
+ if (result == NDBT_FAILED){
+ g_err << "Failed to restart cluster" << endl;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ if (_wait == 1){
+ ndbout << "Waiting for cluster to start" << endl;
+ if ( restarter.waitClusterStarted(120) != 0){
+ ndbout << "Failed waiting for restart of cluster" << endl;
+ result = NDBT_FAILED;
+ }
+ }
+ ndbout << "Cluster restarted" << endl;
+
+ return NDBT_ProgramExit(result);
+}
diff --git a/ndb/test/tools/waiter/Makefile b/ndb/test/tools/waiter/Makefile
new file mode 100644
index 00000000000..da2c9daff00
--- /dev/null
+++ b/ndb/test/tools/waiter/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := waiter
+
+# Source files of non-templated classes (.C files)
+SOURCES = waiter.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/test/tools/waiter/waiter.cpp b/ndb/test/tools/waiter/waiter.cpp
new file mode 100644
index 00000000000..14803fec71d
--- /dev/null
+++ b/ndb/test/tools/waiter/waiter.cpp
@@ -0,0 +1,57 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include "mgmapi.h"
+#include <string.h>
+#include <NdbMain.h>
+#include <NdbOut.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+
+
+#include <NdbRestarter.hpp>
+#include <NDBT.hpp>
+#include <assert.h>
+
+int main(int argc, const char** argv){
+
+ const char* _hostName = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "hostname:port\n"\
+ "This program will connect to the mgmsrv of a NDB cluster.\n"\
+ "It will then wait for all nodes to be started\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _hostName = argv[optind];
+
+ NdbRestarter restarter(_hostName);
+
+ if (restarter.waitClusterStarted() != 0)
+ return NDBT_ProgramExit(NDBT_FAILED);
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/tools/Makefile b/ndb/tools/Makefile
new file mode 100644
index 00000000000..40f440e2e7d
--- /dev/null
+++ b/ndb/tools/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+BIN_DIRS = init_rm select_all select_count desc list_tables \
+ drop_tab delete_all copy_tab \
+ create_index drop_index verify_index cpcc
+
+ifneq ($(NDB_ODBC),N)
+BIN_DIRS += ndbsql
+endif
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/tools/clean-links.sh b/ndb/tools/clean-links.sh
new file mode 100755
index 00000000000..01820f30616
--- /dev/null
+++ b/ndb/tools/clean-links.sh
@@ -0,0 +1,21 @@
+#! /bin/sh
+
+# 1 - Dir
+# 2 - Link dst
+
+if [ $# -lt 1 ]
+then
+ exit 0
+fi
+
+files=`find $1 -type l -maxdepth 1`
+res=$?
+if [ $res -ne 0 ] || [ "$files" = "" ]
+then
+ exit 0
+fi
+
+rm -f $files
+
+
+
diff --git a/ndb/tools/copy_tab/Makefile b/ndb/tools/copy_tab/Makefile
new file mode 100644
index 00000000000..4ad33a26652
--- /dev/null
+++ b/ndb/tools/copy_tab/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := copy_tab
+
+SOURCES := copy_tab.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/tools/copy_tab/copy_tab.cpp b/ndb/tools/copy_tab/copy_tab.cpp
new file mode 100644
index 00000000000..32cfb0b35ff
--- /dev/null
+++ b/ndb/tools/copy_tab/copy_tab.cpp
@@ -0,0 +1,99 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+#include "UtilTransactions.hpp"
+
+#include <getarg.h>
+
+int main(int argc, const char** argv){
+
+ const char* _tabname = NULL;
+ const char* _to_tabname = NULL;
+ const char* _dbname = "TEST_DB";
+ const char* _connectstr = NULL;
+ int _copy_data = true;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "database", 'd', arg_string, &_dbname, "dbname",
+ "Name of database table is in"},
+ { "connstr", 'c', arg_string, &_connectstr, "connect string",
+ "How to connect to NDB"},
+ { "copy-data", '\0', arg_negative_flag, &_copy_data, "Don't copy data to new table",
+ "How to connect to NDB"},
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "srctab desttab\n"\
+ "This program will copy one table in Ndb\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || argv[optind + 1] == NULL || _help){
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+ _to_tabname = argv[optind+1];
+
+ if (_connectstr)
+ Ndb::setConnectString(_connectstr);
+ Ndb MyNdb(_dbname);
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ ndbout << "Copying table " << _tabname << " to " << _to_tabname << "...";
+ const NdbDictionary::Table* ptab = MyNdb.getDictionary()->getTable(_tabname);
+ if (ptab){
+ NdbDictionary::Table tab2(*ptab);
+ tab2.setName(_to_tabname);
+ if (MyNdb.getDictionary()->createTable(tab2) != 0){
+ ndbout << endl << MyNdb.getDictionary()->getNdbError() << endl;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+ } else {
+ ndbout << endl << MyNdb.getDictionary()->getNdbError() << endl;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+ ndbout << "OK" << endl;
+ if (_copy_data){
+ ndbout << "Copying data..."<<endl;
+ const NdbDictionary::Table * tab3 =
+ NDBT_Table::discoverTableFromDb(&MyNdb,
+ _tabname);
+ // if (!tab3)
+
+ UtilTransactions util(*tab3);
+
+ if(util.copyTableData(&MyNdb,
+ _to_tabname) != NDBT_OK){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+ ndbout << "OK" << endl;
+ }
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/tools/cpcc/Makefile b/ndb/tools/cpcc/Makefile
new file mode 100644
index 00000000000..78f8c61e464
--- /dev/null
+++ b/ndb/tools/cpcc/Makefile
@@ -0,0 +1,12 @@
+include .defs.mk
+
+TYPE = util
+
+BIN_TARGET = ndb_cpcc
+
+SOURCES = cpcc.cpp
+OBJECTS_LOC = $(call fixpath,$(NDB_TOP)/src/mgmclient/CpcClient.o)
+
+CFLAGS_cpcc.cpp := -I$(call fixpath,$(NDB_TOP)/src/mgmclient)
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/tools/cpcc/cpcc.cpp b/ndb/tools/cpcc/cpcc.cpp
new file mode 100644
index 00000000000..5a826f250c0
--- /dev/null
+++ b/ndb/tools/cpcc/cpcc.cpp
@@ -0,0 +1,349 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <stdarg.h>
+#include <getarg.h>
+#include "CpcClient.hpp"
+#include <NdbEnv.h>
+
+#define DEFAULT_PORT 1234
+#define ENV_HOSTS "NDB_CPCC_HOSTS"
+
+struct settings {
+ int m_longl;
+ short m_port;
+} g_settings = { 0 , DEFAULT_PORT };
+
+Vector<SimpleCpcClient*> g_hosts;
+int connect(Vector<SimpleCpcClient*>&);
+
+class Expression {
+public:
+ virtual bool evaluate(SimpleCpcClient*, const SimpleCpcClient::Process &)= 0;
+};
+
+int for_each(Vector<SimpleCpcClient*>& list, Expression &);
+int start_stop(const char * cmd, Vector<SimpleCpcClient*>& list,
+ Vector<Vector<Uint32> >& procs);
+
+class True : public Expression {
+public:
+ virtual bool evaluate(SimpleCpcClient*, const SimpleCpcClient::Process & p){
+ return true;
+ }
+};
+
+class FieldEQ : public Expression {
+ BaseString m_field;
+ BaseString m_value;
+public:
+ FieldEQ(const BaseString & field, const BaseString & value){
+ m_field = field;
+ m_value = value;
+ }
+ virtual ~FieldEQ(){}
+
+ virtual bool evaluate(SimpleCpcClient*, const SimpleCpcClient::Process & p){
+ BaseString v;
+ if(m_field == "name") v = p.m_name;
+
+ if(m_field == "type") v = p.m_type;
+ if(m_field == "status") v = p.m_status;
+ if(m_field == "owner") v = p.m_owner;
+ if(m_field == "group") v = p.m_group;
+ if(m_field == "path") v = p.m_path;
+ if(m_field == "args") v = p.m_args;
+ if(m_field == "env") v = p.m_env;
+ if(m_field == "cwd") v = p.m_cwd;
+
+ if(m_field == "stdin") v = p.m_stdin;
+ if(m_field == "stdout") v = p.m_stdout;
+ if(m_field == "stderr") v = p.m_stderr;
+
+ return v == m_value;
+ }
+};
+
+class Match : public Expression {
+ Expression & m_cond;
+ Expression & m_apply;
+public:
+ Match(Expression& condition, Expression & rule)
+ : m_cond(condition), m_apply(rule) {
+ }
+ virtual ~Match(){}
+
+ virtual bool evaluate(SimpleCpcClient* c,const SimpleCpcClient::Process & p){
+ if(m_cond.evaluate(c, p))
+ return m_apply.evaluate(c, p);
+ return false;
+ }
+};
+
+class Operate : public Expression {
+ const char * cmd;
+ SimpleCpcClient * host;
+ settings & sets;
+public:
+ Operate(const char * c, settings & s) : sets(s) {
+ cmd = c;
+ host = 0;
+ }
+
+ virtual bool evaluate(SimpleCpcClient*, const SimpleCpcClient::Process & p);
+};
+
+class ProcEQ : public Expression {
+ SimpleCpcClient * host;
+ Uint32 id;
+public:
+ ProcEQ(SimpleCpcClient* h, Uint32 i){
+ host = h; id = i;
+ }
+
+ virtual bool evaluate(SimpleCpcClient* c,const SimpleCpcClient::Process & p){
+ return p.m_id == id && c == host;
+ }
+};
+
+class OrExpr : public Expression {
+ Expression * m_rule;
+ Vector<Expression *> m_cond;
+ bool on_empty;
+public:
+ OrExpr(Expression * rule, bool onEmp = true){
+ m_rule = rule;
+ on_empty = onEmp;
+ }
+
+ virtual ~OrExpr(){}
+
+ virtual bool evaluate(SimpleCpcClient* c, const SimpleCpcClient::Process & p){
+ bool run = on_empty;
+ for(size_t i = 0; i<m_cond.size(); i++){
+ if(m_cond[i]->evaluate(c, p)){
+ run = true;
+ break;
+ }
+ }
+ if(run)
+ return m_rule->evaluate(c, p);
+ return false;
+ }
+
+ void push_back(Expression * expr){
+ m_cond.push_back(expr);
+ }
+};
+
+void
+add_host(Vector<SimpleCpcClient*> & hosts, BaseString tmp){
+ Vector<BaseString> split;
+ tmp.split(split, ":");
+
+ short port = g_settings.m_port;
+ if(split.size() > 1)
+ port = atoi(split[1].c_str());
+
+ hosts.push_back(new SimpleCpcClient(split[0].c_str(), port));
+}
+
+void
+add_hosts(Vector<SimpleCpcClient*> & hosts, BaseString list){
+ Vector<BaseString> split;
+ list.split(split);
+ for(size_t i = 0; i<split.size(); i++){
+ add_host(hosts, split[i]);
+ }
+}
+
+int
+main(int argc, const char** argv){
+ int help = 0;
+ const char *cmd=0, *name=0, *group=0, *owner=0;
+ int list = 0, start = 0, stop = 0, rm = 0;
+ struct getargs args[] = {
+ { "cmd", 'c', arg_string, &cmd, "command", "command to run (default ls)" }
+ ,{ "name", 'n', arg_string, &name,
+ "apply command for all processes with name" }
+ ,{ "group", 'g', arg_string, &group,
+ "apply command for all processes in group" }
+ ,{ "owner", 'g', arg_string, &owner,
+ "apply command for all processes with owner" }
+ ,{ "long", 'l', arg_flag, &g_settings.m_longl, "long", "long listing"}
+ ,{ "usage", '?', arg_flag, &help, "Print help", "" }
+ ,{ "ls", 0, arg_flag, &list, "-c list", "list process(es)" }
+ ,{ "start", 0, arg_flag, &start, "-c start", "start process(es)" }
+ ,{ "stop", 0, arg_flag, &stop, "-c stop", "stop process(es)" }
+ ,{ "rm", 0, arg_flag, &rm, "-c rm", "undefine process(es)" }
+ };
+ const int num_args = 10;
+
+ int optind = 0;
+ char desc[] = "[host:[port]]\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return 1;
+ }
+
+ if(list + start + stop + rm > 1){
+ ndbout_c("Can only specify one command");
+ arg_printusage(args, num_args, argv[0], desc);
+ return 1;
+ }
+
+ if(list) cmd = "list";
+ if(start) cmd = "start";
+ if(stop) cmd = "stop";
+ if(rm) cmd = "rm";
+ if(!cmd) cmd = "list";
+
+ Expression * m_expr = 0;
+
+ for(int i = optind; i<argc; i++){
+ add_host(g_hosts, argv[i]);
+ }
+
+ OrExpr * orE = new OrExpr(new Operate(cmd, g_settings), true);
+ m_expr = orE;
+ for(int i = optind; i<argc; i++){
+ BaseString tmp(argv[i]);
+ Vector<BaseString> split;
+ tmp.split(split, ":");
+
+ if(split.size() > 2){
+ Uint32 id = atoi(split[2].c_str());
+ orE->push_back(new ProcEQ(g_hosts[i-optind], id));
+ }
+ }
+
+ if(g_hosts.size() == 0){
+ char buf[1024];
+ if(NdbEnv_GetEnv(ENV_HOSTS, buf, sizeof(buf))){
+ add_hosts(g_hosts, BaseString(buf));
+ }
+ }
+
+ if(g_hosts.size() == 0){
+ g_hosts.push_back(new SimpleCpcClient("localhost", g_settings.m_port));
+ }
+
+ if(group != 0){
+ Expression * tmp = new FieldEQ("group", group);
+ m_expr = new Match(* tmp, * m_expr);
+ }
+
+ if(name != 0){
+ Expression * tmp = new FieldEQ("name", name);
+ m_expr = new Match(* tmp, * m_expr);
+ }
+
+ if(owner != 0){
+ Expression * tmp = new FieldEQ("owner", owner);
+ m_expr = new Match(* tmp, * m_expr);
+ }
+
+ connect(g_hosts);
+ for_each(g_hosts, * m_expr);
+
+ return 0;
+}
+
+int
+connect(Vector<SimpleCpcClient*>& list){
+ for(size_t i = 0; i<list.size(); i++){
+ if(list[i]->connect() != 0){
+ ndbout_c("Failed to connect to %s:%d",
+ list[i]->getHost(), list[i]->getPort());
+ delete list[i]; list[i] = 0;
+ }
+ }
+ return 0;
+}
+
+int
+for_each(Vector<SimpleCpcClient*>& list, Expression & expr){
+ for(size_t i = 0; i<list.size(); i++){
+ if(list[i] == 0)
+ continue;
+ Properties p;
+ Vector<SimpleCpcClient::Process> procs;
+ if(list[i]->list_processes(procs, p) != 0){
+ ndbout << "Failed to list processes on "
+ << list[i]->getHost() << ":" << list[i]->getPort() << endl;
+ }
+ for(size_t j = 0; j<procs.size(); j++)
+ expr.evaluate(list[i], procs[j]);
+ }
+ return 0;
+}
+
+bool
+Operate::evaluate(SimpleCpcClient* c, const SimpleCpcClient::Process & pp){
+ Uint32 id = pp.m_id;
+ Properties p;
+ int res;
+
+ if(strcasecmp(cmd, "start") == 0)
+ res = c->start_process(id, p);
+ else if(strcasecmp(cmd, "stop") == 0)
+ res = c->stop_process(id, p);
+ else if(strcasecmp(cmd, "rm") == 0)
+ res = c->undefine_process(id, p);
+ else if(strcasecmp(cmd, "list") == 0){
+ if(!sets.m_longl){
+ if(host != c){
+ ndbout_c("--- %s:%d", c->getHost(), c->getPort());
+ host = c;
+ }
+ }
+
+ char s = 0;
+ const char * status = pp.m_status.c_str();
+ if(strcmp(status, "stopped") == 0) s = '-';
+ if(strcmp(status, "starting") == 0) s = 's';
+ if(strcmp(status, "running") == 0) s = 'r';
+ if(strcmp(status, "stopping") == 0) s = 'k';
+ if(s == 0) s = '?';
+
+ if(!sets.m_longl){
+ ndbout_c("%c%c\t%d\t%s\t%s\t%s(%s)",
+ s,
+ pp.m_type.c_str()[0], id, pp.m_owner.c_str(),
+ pp.m_group.c_str(), pp.m_name.c_str(), pp.m_path.c_str());
+ } else {
+ ndbout_c("%c%c %s:%d:%d %s %s %s(%s)",
+ s, pp.m_type.c_str()[0], c->getHost(), c->getPort(),
+ id, pp.m_owner.c_str(), pp.m_group.c_str(),
+ pp.m_name.c_str(), pp.m_path.c_str());
+ }
+ return true;
+ }
+
+ if(res != 0){
+ BaseString msg;
+ p.get("errormessage", msg);
+ ndbout_c("Failed to %s %d on %s:%d - %s",
+ cmd, id,
+ c->getHost(), c->getPort(), msg.c_str());
+ return false;
+ }
+
+ return true;
+}
+
diff --git a/ndb/tools/create_index/Makefile b/ndb/tools/create_index/Makefile
new file mode 100644
index 00000000000..38f2df970c4
--- /dev/null
+++ b/ndb/tools/create_index/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := create_index
+
+# Source files of non-templated classes (.C files)
+SOURCES = create_index.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/tools/create_index/create_index.cpp b/ndb/tools/create_index/create_index.cpp
new file mode 100644
index 00000000000..32da39a5208
--- /dev/null
+++ b/ndb/tools/create_index/create_index.cpp
@@ -0,0 +1,95 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+
+#include <getarg.h>
+
+
+
+int
+main(int argc, const char** argv){
+
+ const char* _dbname = "TEST_DB";
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "database", 'd', arg_string, &_dbname, "dbname",
+ "Name of database table is in"},
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "<tabname>+\n"\
+ "This program will create one unique hash index named ind_<tabname> "
+ " for each table. The index will contain all columns in the table";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help ||
+ argv[optind] == NULL){
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+
+ Ndb MyNdb(_dbname);
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ NdbDictionary::Dictionary * dict = MyNdb.getDictionary();
+
+ for(int i = optind; i<argc; i++){
+ const NdbDictionary::Table * tab = dict->getTable(argv[i]);
+ if(tab == 0){
+ g_err << "Unknown table: " << argv[i] << endl;
+ continue;
+ }
+
+ if(tab->getNoOfColumns() > 16){
+ g_err << "Table " << argv[i] << " has more than 16 columns" << endl;
+ }
+
+ NdbDictionary::Index ind;
+ char buf[512];
+ sprintf(buf, "IND_%s", argv[i]);
+ ind.setName(buf);
+ ind.setTable(argv[i]);
+ ind.setType(NdbDictionary::Index::UniqueHashIndex);
+ for(int c = 0; c<tab->getNoOfColumns(); c++)
+ ind.addIndexColumn(tab->getColumn(c)->getName());
+
+ ndbout << "creating index " << buf << " on table " << argv[i] << "...";
+ const int res = dict->createIndex(ind);
+ if(res != 0)
+ ndbout << endl << dict->getNdbError() << endl;
+ else
+ ndbout << "OK" << endl;
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
+
diff --git a/ndb/tools/delete_all/Makefile b/ndb/tools/delete_all/Makefile
new file mode 100644
index 00000000000..1cae240eb8f
--- /dev/null
+++ b/ndb/tools/delete_all/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := delete_all
+
+SOURCES := delete_all.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/tools/delete_all/delete_all.cpp b/ndb/tools/delete_all/delete_all.cpp
new file mode 100644
index 00000000000..e78ad4a2e1e
--- /dev/null
+++ b/ndb/tools/delete_all/delete_all.cpp
@@ -0,0 +1,93 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NdbSleep.h>
+#include <NDBT.hpp>
+
+#include <getarg.h>
+
+#include <UtilTransactions.hpp>
+
+int main(int argc, const char** argv){
+
+ const char* _tabname = NULL;
+ const char* _dbname = "TEST_DB";
+ int _help = 0;
+ int _ver2 = 1;
+
+ struct getargs args[] = {
+ { "usage", '?', arg_flag, &_help, "Print help", "" },
+ { "ver2", '2', arg_flag, &_ver2, "Use version 2 of clearTable (default)", "" },
+ { "ver2", '1', arg_negative_flag, &_ver2, "Use version 1 of clearTable", "" },
+ { "database", 'd', arg_string, &_dbname, "dbname",
+ "Name of database table is in"}
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will delete all records in the specified table using scan delete.\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb(_dbname);
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ // Connect to Ndb and wait for it to become ready
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ int res = NDBT_OK;
+ for(int i = optind; i<argc; i++){
+ const NdbDictionary::Table * pTab = NDBT_Table::discoverTableFromDb(&MyNdb, argv[i]);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ ndbout << "Deleting all from " << argv[i] << "...";
+ UtilTransactions utilTrans(*pTab);
+ int tmp = NDBT_OK;
+ if (_ver2 == 0){
+ if(utilTrans.clearTable(&MyNdb) == NDBT_FAILED)
+ tmp = NDBT_FAILED;
+ } else {
+ if(utilTrans.clearTable3(&MyNdb) == NDBT_FAILED)
+ tmp = NDBT_FAILED;
+ }
+ if(tmp == NDBT_FAILED){
+ res = tmp;
+ ndbout << "FAILED" << endl;
+ }
+ }
+ return NDBT_ProgramExit(res);
+}
+
diff --git a/ndb/tools/desc/Makefile b/ndb/tools/desc/Makefile
new file mode 100644
index 00000000000..614984cfd35
--- /dev/null
+++ b/ndb/tools/desc/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := desc
+
+SOURCES := desc.C
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/tools/desc/desc.cpp b/ndb/tools/desc/desc.cpp
new file mode 100644
index 00000000000..7481190614c
--- /dev/null
+++ b/ndb/tools/desc/desc.cpp
@@ -0,0 +1,78 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <getarg.h>
+#include <NDBT.hpp>
+#include <NdbApi.hpp>
+
+
+int main(int argc, const char** argv){
+ const char* _tabname = NULL;
+ const char* _dbname = "TEST_DB";
+ int _frm = 0;
+ int _unqualified = 0;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "unqualified", 'u', arg_integer, &_unqualified, "unqualified",
+ "Use unqualified table names"},
+ { "database", 'd', arg_string, &_dbname, "dbname",
+ "Name of database table is in"},
+ { "frm-data", 'f', arg_flag, &_frm, "Show frm data for table", "" } ,
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program list all properties of table(s) in NDB Cluster.\n"\
+ " ex: desc T1 T2 T4\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL ||_help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ Ndb* pMyNdb;
+ pMyNdb = new Ndb(_dbname);
+ pMyNdb->useFullyQualifiedNames(!_unqualified);
+ pMyNdb->init();
+
+ ndbout << "Waiting...";
+ while (pMyNdb->waitUntilReady() != 0) {
+ ndbout << "...";
+ }
+ ndbout << endl;
+
+ NdbDictionary::Dictionary * dict = pMyNdb->getDictionary();
+ for (int i = optind; i < argc; i++) {
+ NDBT_Table* pTab = (NDBT_Table*)dict->getTable(argv[i]);
+ if (pTab != 0) {
+ ndbout << (* pTab) << endl;
+ if (_frm){
+ ndbout << "getFrmLength: "<< endl
+ << pTab->getFrmLength() << endl;
+ }
+ } else {
+ ndbout << argv[i] << ": " << dict->getNdbError() << endl;
+ }
+ }
+
+ delete pMyNdb;
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/tools/drop_index/Makefile b/ndb/tools/drop_index/Makefile
new file mode 100644
index 00000000000..969bee51064
--- /dev/null
+++ b/ndb/tools/drop_index/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := drop_index
+
+# Source files of non-templated classes (.C files)
+SOURCES = drop_index.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/tools/drop_index/drop_index.cpp b/ndb/tools/drop_index/drop_index.cpp
new file mode 100644
index 00000000000..146f01113b2
--- /dev/null
+++ b/ndb/tools/drop_index/drop_index.cpp
@@ -0,0 +1,76 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+
+#include <getarg.h>
+
+int main(int argc, const char** argv){
+
+ const char* _tabname = NULL;
+ const char* _dbname = "TEST_DB";
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "database", 'd', arg_string, &_dbname, "dbname",
+ "Name of database table is in"},
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "<indexname>+\n"\
+ "This program will drop index(es) in Ndb\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _help){
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb(_dbname);
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ int res = 0;
+ for(int i = optind; i<argc; i++){
+ ndbout << "Dropping index " << argv[i] << "...";
+ int tmp;
+ if((tmp = MyNdb.getDictionary()->dropIndex(argv[i], 0)) != 0){
+ ndbout << endl << MyNdb.getDictionary()->getNdbError() << endl;
+ res = tmp;
+ } else {
+ ndbout << "OK" << endl;
+ }
+ }
+
+ if(res != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/tools/drop_tab/Makefile b/ndb/tools/drop_tab/Makefile
new file mode 100644
index 00000000000..d7b21fe982c
--- /dev/null
+++ b/ndb/tools/drop_tab/Makefile
@@ -0,0 +1,11 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := drop_tab
+
+# Source files of non-templated classes (.C files)
+SOURCES = drop_tab.cpp
+
+include $(NDB_TOP)/Epilogue.mk
+
diff --git a/ndb/tools/drop_tab/drop_tab.cpp b/ndb/tools/drop_tab/drop_tab.cpp
new file mode 100644
index 00000000000..5946ada5956
--- /dev/null
+++ b/ndb/tools/drop_tab/drop_tab.cpp
@@ -0,0 +1,80 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdio.h>
+
+#include <NdbOut.hpp>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+
+#include <getarg.h>
+
+int main(int argc, const char** argv){
+
+ const char* _tabname = NULL;
+ const char* _dbname = "TEST_DB";
+ const char* _connectstr = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "database", 'd', arg_string, &_dbname, "dbname",
+ "Name of database table is in"},
+ { "connstr", 'c', arg_string, &_connectstr, "connect string",
+ "How to connect to NDB"},
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program will drop one table in Ndb\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _help){
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ if (_connectstr)
+ Ndb::setConnectString(_connectstr);
+ Ndb MyNdb(_dbname);
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ int res = 0;
+ for(int i = optind; i<argc; i++){
+ ndbout << "Dropping table " << argv[i] << "...";
+ int tmp;
+ if((tmp = MyNdb.getDictionary()->dropTable(argv[i])) != 0){
+ ndbout << endl << MyNdb.getDictionary()->getNdbError() << endl;
+ res = tmp;
+ } else {
+ ndbout << "OK" << endl;
+ }
+ }
+
+ if(res != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
diff --git a/ndb/tools/init_rm/init_rm.c b/ndb/tools/init_rm/init_rm.c
new file mode 100644
index 00000000000..7a6265d9f77
--- /dev/null
+++ b/ndb/tools/init_rm/init_rm.c
@@ -0,0 +1,69 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <sys/types.h>
+#include <dirent.h>
+#include <string.h>
+#include <stdio.h>
+#include <unistd.h>
+
+#include <NdbMain.h>
+#include <NdbEnv.h>
+
+void rmdir_recurs(const char* path){
+ DIR* dirp;
+ struct dirent * dp;
+ char buf[255];
+
+ dirp = opendir(path);
+ while ((dp = readdir(dirp)) != NULL){
+ if ((strcmp(".", dp->d_name) != 0) && (strcmp("..", dp->d_name) != 0)) {
+ sprintf(buf, "%s/%s", path, dp->d_name);
+ // printf(" %s\n", buf);
+ if (remove(buf) == 0){
+ printf("."); //printf("Removed: %s\n", buf); // The file was removed
+ } else {
+ // The file was not removed, try to remove it as a directory
+ if(rmdir(buf) == 0){
+ ; //printf("Removed dir: %s\n", buf); // The dir was removed
+ } else {
+ // The directory was not removed, call this function again recursively
+ ; //printf("Call rm_dir: %s\n", buf);
+ rmdir_recurs(buf);
+ rmdir(buf);
+ }
+ }
+ }
+ }
+ closedir(dirp);
+}
+
+NDB_COMMAND(init_rm, "init_rm", "init_rm [path to dir]", "Removes all files and dirs below [path to dir], default = /d/ndb/fs. WARNING can remove a lot of useful files!", 4096){
+
+ if(argc == 2){
+ printf("Removing all files and dirs in %s\n", argv[1]);
+ rmdir_recurs(argv[1]);
+ } else if(argc == 1){
+ printf("Removing all files and dirs in /d/ndb/fs\n");
+ rmdir_recurs("/d/ndb/fs");
+ }
+ printf("\n");
+
+ return 0;
+
+
+}
+
diff --git a/ndb/tools/list_tables/Makefile b/ndb/tools/list_tables/Makefile
new file mode 100644
index 00000000000..b60f161ee68
--- /dev/null
+++ b/ndb/tools/list_tables/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE = ndbapitest
+
+BIN_TARGET = list_tables
+
+SOURCES = listTables.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/tools/list_tables/listTables.cpp b/ndb/tools/list_tables/listTables.cpp
new file mode 100644
index 00000000000..c14808050c6
--- /dev/null
+++ b/ndb/tools/list_tables/listTables.cpp
@@ -0,0 +1,194 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*
+ * list_tables
+ *
+ * List objects(tables, triggers, etc.) in NDB Cluster
+ *
+ */
+
+#include <stdarg.h>
+#include <getarg.h>
+#include <NdbApi.hpp>
+#include <NDBT.hpp>
+
+
+static Ndb* ndb = 0;
+static NdbDictionary::Dictionary* dic = 0;
+
+static void
+fatal(char const* fmt, ...)
+{
+ va_list ap;
+ char buf[500];
+ va_start(ap, fmt);
+ vsnprintf(buf, sizeof(buf), fmt, ap);
+ va_end(ap);
+ ndbout << buf;
+ if (ndb)
+ ndbout << " - " << ndb->getNdbError();
+ ndbout << endl;
+ NDBT_ProgramExit(NDBT_FAILED);
+ exit(1);
+}
+
+static void
+list(const char * tabname,
+ NdbDictionary::Object::Type type)
+{
+ NdbDictionary::Dictionary::List list;
+ if (tabname == 0) {
+ if (dic->listObjects(list, type) == -1)
+ fatal("listObjects");
+ } else {
+ if (dic->listIndexes(list, tabname) == -1)
+ fatal("listIndexes");
+ }
+ if (Ndb::usingFullyQualifiedNames())
+ ndbout_c("%-5s %-20s %-8s %-7s %-12s %-8s %s", "id", "type", "state", "logging", "database", "schema", "name");
+ else
+ ndbout_c("%-5s %-20s %-8s %-7s %s", "id", "type", "state", "logging", "name");
+ for (unsigned i = 0; i < list.count; i++) {
+ NdbDictionary::Dictionary::List::Element& elt = list.elements[i];
+ char type[100];
+ bool isTable = false;
+ switch (elt.type) {
+ case NdbDictionary::Object::SystemTable:
+ strcpy(type, "SystemTable");
+ isTable = true;
+ break;
+ case NdbDictionary::Object::UserTable:
+ strcpy(type, "UserTable");
+ isTable = true;
+ break;
+ case NdbDictionary::Object::UniqueHashIndex:
+ strcpy(type, "UniqueHashIndex");
+ isTable = true;
+ break;
+ case NdbDictionary::Object::OrderedIndex:
+ strcpy(type, "OrderedIndex");
+ isTable = true;
+ break;
+ case NdbDictionary::Object::HashIndexTrigger:
+ strcpy(type, "HashIndexTrigger");
+ break;
+ case NdbDictionary::Object::IndexTrigger:
+ strcpy(type, "IndexTrigger");
+ break;
+ case NdbDictionary::Object::SubscriptionTrigger:
+ strcpy(type, "SubscriptionTrigger");
+ break;
+ case NdbDictionary::Object::ReadOnlyConstraint:
+ strcpy(type, "ReadOnlyConstraint");
+ break;
+ default:
+ sprintf(type, "%d", (int)elt.type);
+ break;
+ }
+ char state[100];
+ switch (elt.state) {
+ case NdbDictionary::Object::StateOffline:
+ strcpy(state, "Offline");
+ break;
+ case NdbDictionary::Object::StateBuilding:
+ strcpy(state, "Building");
+ break;
+ case NdbDictionary::Object::StateDropping:
+ strcpy(state, "Dropping");
+ break;
+ case NdbDictionary::Object::StateOnline:
+ strcpy(state, "Online");
+ break;
+ case NdbDictionary::Object::StateBroken:
+ strcpy(state, "Broken");
+ break;
+ default:
+ sprintf(state, "%d", (int)elt.state);
+ break;
+ }
+ char store[100];
+ if (! isTable)
+ strcpy(store, "-");
+ else {
+ switch (elt.store) {
+ case NdbDictionary::Object::StoreTemporary:
+ strcpy(store, "No");
+ break;
+ case NdbDictionary::Object::StorePermanent:
+ strcpy(store, "Yes");
+ break;
+ default:
+ sprintf(state, "%d", (int)elt.store);
+ break;
+ }
+ }
+ if (Ndb::usingFullyQualifiedNames())
+ ndbout_c("%-5d %-20s %-8s %-7s %-12s %-8s %s", elt.id, type, state, store, (elt.database)?elt.database:"", (elt.schema)?elt.schema:"", elt.name);
+ else
+ ndbout_c("%-5d %-20s %-8s %-7s %s", elt.id, type, state, store, elt.name);
+ }
+}
+
+int main(int argc, const char** argv){
+ int _loops = 1;
+ const char* _tabname = NULL;
+ const char* _dbname = "TEST_DB";
+ int _unqualified = 0;
+ int _type = 0;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "loops", 'l', arg_integer, &_loops, "loops",
+ "Number of times to run(default = 1)" },
+ { "unqualified", 'u', arg_flag, &_unqualified, "unqualified",
+ "Use unqualified table names"},
+ { "database", 'd', arg_string, &_dbname, "dbname",
+ "Name of database table is in"},
+ { "type", 't', arg_integer, &_type, "type",
+ "Type of objects to show, see NdbDictionary.hpp for numbers(default = 0)" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program list all system objects in NDB Cluster.\n"\
+ "Type of objects to display can be limited with -t option\n"\
+ " ex: list_tables -t 2 would show all UserTables\n"\
+ "To show all indexes for a table write table name as final argument\n"\
+ " ex: list_tables T1\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ ndb = new Ndb(_dbname);
+ ndb->useFullyQualifiedNames(!_unqualified);
+ if (ndb->init() != 0)
+ fatal("init");
+ if (ndb->waitUntilReady(30) < 0)
+ fatal("waitUntilReady");
+ dic = ndb->getDictionary();
+ for (int i = 0; _loops == 0 || i < _loops; i++) {
+ list(_tabname, (NdbDictionary::Object::Type)_type);
+ }
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
+// vim: set sw=4:
diff --git a/ndb/tools/make-errors.pl b/ndb/tools/make-errors.pl
new file mode 100644
index 00000000000..65819209a89
--- /dev/null
+++ b/ndb/tools/make-errors.pl
@@ -0,0 +1,181 @@
+#! /usr/local/bin/perl
+
+use strict;
+use Getopt::Long;
+use XML::Parser;
+
+(my $progname = $0) =~ s!^.*/!!;
+
+sub usage {
+ my $errstr = "@_";
+ while (chomp($errstr)) {}
+ print <<END;
+$progname: $errstr
+$progname -- read codes.xml and write codes.hpp and codes.cpp
+usage: $progname [options] codes.xml
+-c check xml file only
+-d check xml file and show diff against old hpp and cpp
+END
+ exit(1);
+}
+
+my $opts = {};
+opts: {
+ local $SIG{__WARN__} = \&usage;
+ GetOptions($opts, qw(c d));
+}
+@ARGV == 1 or usage("one filename argument expected");
+my $filexml = shift;
+$filexml =~ /^(.*)\.xml$/ or usage("$filexml does not end in .xml");
+my $filehpp = "$1.hpp";
+my $filecpp = "$1.cpp";
+
+my $temphpp = "$filehpp-new";
+my $tempcpp = "$filecpp-new";
+unlink $temphpp, $tempcpp;
+open(HPP, ">$temphpp") or die "$temphpp: $!\n";
+open(CPP, ">$tempcpp") or die "$tempcpp: $!\n";
+
+my $i2 = " " x 2;
+my $i4 = " " x 4;
+my $lb = "{";
+my $rb = "}";
+
+sub disclaimer {
+ my $filename = shift;
+ return <<END;
+/*
+ * $filename -- DO NOT EDIT !!
+ *
+ * To create a new version (both *.hpp and *.cpp):
+ *
+ * 1) edit $filexml
+ * 2) perl tools/$progname $filexml
+ * 3) check all files (*.xml *.hpp *.cpp) into CVS
+ *
+ * On RedHat linux requires perl-XML-Parser package.
+ */
+END
+}
+
+my $classname = $filehpp;
+$classname =~ s!^.*/!!;
+$classname =~ s/\.hpp$//;
+
+sub handle_init {
+ my($parser) = @_;
+ my $guard = $filehpp;
+ $guard =~ s!^.*/!!;
+ $guard =~ s!([a-z])([A-Z])!${1}_${2}!g;
+ $guard =~ s!\.!_!g;
+ $guard = uc($guard);
+ print HPP "#ifndef $guard\n#define $guard\n\n";
+ print HPP disclaimer($filehpp), "\n";
+ print HPP "class $classname $lb\n";
+ print HPP "${i2}enum Value $lb\n";
+ print CPP disclaimer($filecpp), "\n";
+ print CPP "/* included in Ndberror.cpp */\n\n";
+}
+
+my %classhash = (
+ ApplicationError => 1,
+ NoDataFound => 1,
+ ConstraintViolation => 1,
+ SchemaError => 1,
+ UserDefinedError => 1,
+ InsufficientSpace => 1,
+ TemporaryResourceError => 1,
+ NodeRecoveryError => 1,
+ OverloadError => 1,
+ TimeoutExpired => 1,
+ UnknownResultError => 1,
+ InternalError => 1,
+ FunctionNotImplemented => 1,
+ UnknownErrorCode => 1,
+ NodeShutdown => 1,
+);
+
+my $section = undef;
+my %codehash = ();
+my %namehash = ();
+
+sub handle_start {
+ my($parser, $tag, %attr) = @_;
+ if ($tag eq 'Error') {
+ return;
+ }
+ if ($tag eq 'Section') {
+ $section = $attr{name};
+ $section =~ /^\w+$/ or
+ $parser->xpcroak("invalid or missing section name");
+ return;
+ }
+ if ($tag eq 'Code') {
+ print HPP ",\n" if %codehash;
+ print CPP ",\n" if %codehash;
+ my $name = $attr{name};
+ my $class = $attr{class};
+ my $code = $attr{code};
+ my $message = $attr{message};
+ $name =~ /^\w+$/ or
+ $parser->xpcroak("invalid or missing error name '$name'");
+ $namehash{$name}++ and
+ $parser->xpcroak("duplicate error name '$name'");
+ $classhash{$class} or
+ $parser->xpcroak("invalid or missing error class '$class'");
+ $code =~ /^\d+$/ or
+ $parser->xpcroak("invalid or missing error code '$code'");
+ $codehash{$code}++ and
+ $parser->xpcroak("duplicate error code '$code'");
+ $message =~ /\S/ or
+ $parser->xpcroak("invalid or missing error message '$message'");
+ $message =~ s/^\s+|\s+$//g;
+ my $enum = "${section}_${name}";
+ print HPP "${i4}$enum = $code";
+ print CPP "${i2}$lb ${classname}::$enum,\n";
+ print CPP "${i4}NdbError::$class,\n";
+ print CPP "${i4}\"$message\"\n";
+ print CPP "${i2}$rb";
+ return;
+ }
+ $parser->xpcroak("unknown tag $tag");
+}
+
+sub handle_end {
+ my($parser, $tag) = @_;
+}
+
+sub handle_final {
+ print HPP "\n" if %codehash;
+ print HPP "${i2}$rb;\n";
+ print HPP "$rb;\n\n#endif\n";
+ print CPP ",\n" if 1;
+ return 1;
+}
+
+my $parser = new XML::Parser(
+ ParseParamEnt => 1,
+ Handlers => {
+ Init => \&handle_init,
+ Start => \&handle_start,
+ End => \&handle_end,
+ Final => \&handle_final,
+ },
+ ErrorContext => 0,
+);
+eval {
+ $parser->parsefile($filexml);
+};
+if ($@) {
+ my $errstr = join("\n", grep(m!\S! && ! m!^\s*at\s!, split(/\n/, $@)));
+ die "$filexml:\n$errstr\n";
+}
+
+close(HPP);
+close(CPP);
+rename($temphpp, $filehpp);
+rename($tempcpp, $filecpp);
+
+1;
+
+# vim:set sw=4:
diff --git a/ndb/tools/make-links.sh b/ndb/tools/make-links.sh
new file mode 100755
index 00000000000..e0c4f55986e
--- /dev/null
+++ b/ndb/tools/make-links.sh
@@ -0,0 +1,20 @@
+#! /bin/sh
+
+# 1 - Link top src
+# 2 - Link dst
+
+if [ $# -lt 2 ]
+then
+ exit 0
+fi
+
+name=`basename $2`
+files=`find $1/$name -type f -name '*.h*'`
+
+for i in $files
+do
+ ln -s $i $2/`basename $i`
+done
+
+
+
diff --git a/ndb/tools/ndbnet/Makefile.PL b/ndb/tools/ndbnet/Makefile.PL
new file mode 100644
index 00000000000..4b27a17de15
--- /dev/null
+++ b/ndb/tools/ndbnet/Makefile.PL
@@ -0,0 +1,158 @@
+# -*- perl -*-
+
+use strict;
+use Config;
+use ExtUtils::MakeMaker qw(WriteMakefile);
+use Test::Harness;
+
+require 5.005;
+
+my $base;
+if ($base ||= $ENV{NDB_BASE}) {
+ warn "Using NDB_BASE=$base\n";
+}
+$base or die "FATAL: need env.variable NDB_BASE\n";
+
+my $top;
+if ($top ||= $ENV{NDB_TOP}) {
+ warn "Using NDB_TOP=$top\n";
+}
+$top or die "FATAL: need env.variable NDB_TOP\n";
+
+my @scripts = qw(ndbnet.pl ndbnetd.pl);
+
+for my $f (@scripts) {
+ my $p = $f;
+ $p =~ s/\.pl$//;
+ unlink("$p.sh");
+ open(G, ">$p.sh") or die "$p.sh: $!";
+ if ($Config{osname} ne 'MSWin32') {
+ print G <<END;
+#! /bin/sh
+
+# installed in \$NDB_BASE
+# selects which $p to run (normally from latest release)
+# created in source directory by "make install-base"
+
+NDB_BASE=$base
+export NDB_BASE
+
+PATH=\$NDB_BASE/bin:\$PATH
+export PATH
+
+LD_LIBRARY_PATH=\$NDB_BASE/lib:\$LD_LIBRARY_PATH
+export LD_LIBRARY_PATH
+
+PERL5LIB=\$NDB_BASE/lib/perl5:\$PERL5LIB
+export PERL5LIB
+
+NDB_TOP=$top
+export NDB_TOP
+
+PATH=\$NDB_TOP/bin:\$PATH
+export PATH
+
+LD_LIBRARY_PATH=\$NDB_TOP/lib:\$LD_LIBRARY_PATH
+export LD_LIBRARY_PATH
+
+PERL5LIB=\$NDB_TOP/lib/perl5:\$PERL5LIB
+export PERL5LIB
+
+exec perl \$NDB_TOP/lib/perl5/$p.pl "\$@"
+END
+ } else {
+ print G <<END;
+rem installed in \$NDB_BASE
+rem selects which $p to run (normally from latest release)
+rem created in source directory by "make install-base"
+
+set NDB_BASE=$base
+set PATH=%NDB_BASE%\\bin;%PATH%
+set PERL5LIB=%NDB_BASE%\\lib\\perl5;%PERL5LIB%
+set NDB_TOP=$top
+set PATH=%NDB_TOP%\\bin;%PATH%
+set PERL5LIB=%NDB_TOP%\\lib\\perl5;%PERL5LIB%
+perl %NDB_TOP%\\lib\\perl5\\$p.pl %1 %2 %3 %4 %5 %6 %7 %8 %9
+END
+ }
+ close G;
+}
+
+unshift(@INC, 'lib');
+$main::onlymodules = 1;
+require lib::NDB::Util;
+require lib::NDB::Net;
+require lib::NDB::Run;
+
+my @modules = (
+ q(NDB::Util),
+ @NDB::Util::modules,
+ q(NDB::Net),
+ @NDB::Net::modules,
+ q(NDB::Run),
+ @NDB::Run::modules,
+);
+
+my @modulepaths = map { s!::!/!g; s!$!.pm!; $_ } @modules;
+
+my %pm = ();
+for my $pl (@scripts) {
+ $pm{"$pl"} = "\$(INST_LIBDIR)/$pl";
+}
+for my $pm (@modulepaths) {
+ $pm{"lib/$pm"} = "\$(INST_LIBDIR)/$pm";
+}
+
+WriteMakefile(
+ NAME=> 'NDB',
+ PM=> \%pm,
+ EXE_FILES=> [ qw(ndbrun) ],
+# install
+ PREFIX=> $top,
+ LIB=> "$top/lib/perl5",
+);
+
+sub MY::postamble {
+ my $mk = "";
+ $mk .= "\n" . <<END;
+# NDB make targets
+libs: all install
+bins:
+links:
+depend:
+clean_dep:
+#clean:
+cleanall:
+tidy:
+#distclean:
+check:
+ perl -Ilib -cw -e "use NDB::Util"
+ perl -Ilib -cw -e "use NDB::Net"
+ perl -Ilib -cw -e "use NDB::Run"
+ perl -Ilib -cw ndbnetd.pl
+ perl -Ilib -cw ndbnet.pl
+END
+ if ($Config{osname} ne 'MSWin32') {
+ $mk .= "\n" . <<END;
+# install startup scripts to \$NDB_BASE
+install-base:
+ test "\$\$NDB_BASE"
+ mkdir -p \$\$NDB_BASE/bin
+ rm -f \$\$NDB_BASE/bin/ndbnet
+ cp -p ndbnet.sh \$\$NDB_BASE/bin/ndbnet
+ chmod +x \$\$NDB_BASE/bin/ndbnet
+ rm -f \$\$NDB_BASE/bin/ndbnetd
+ cp -p ndbnetd.sh \$\$NDB_BASE/bin/ndbnetd
+ chmod +x \$\$NDB_BASE/bin/ndbnetd
+END
+ } else {
+ $mk .= "\n" . <<END;
+install-base:
+ copy ndbnet.sh $base\\bin\\ndbnet.bat
+ copy ndbnetd.sh $base\\bin\\ndbnetd.bat
+END
+ }
+ return $mk;
+}
+
+1;
diff --git a/ndb/tools/ndbnet/lib/NDB/Net.pm b/ndb/tools/ndbnet/lib/NDB/Net.pm
new file mode 100644
index 00000000000..3b7b16bb3cf
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net.pm
@@ -0,0 +1,42 @@
+package NDB::Net;
+
+use strict;
+use Carp;
+require Exporter;
+
+use NDB::Util;
+
+use vars qw(@ISA @EXPORT @EXPORT_OK);
+@ISA = qw(Exporter);
+
+use vars qw(@modules);
+@modules = qw(
+ NDB::Net::Base
+ NDB::Net::Client
+ NDB::Net::Command
+ NDB::Net::Config
+ NDB::Net::Database
+ NDB::Net::Env
+ NDB::Net::Node
+ NDB::Net::NodeApi
+ NDB::Net::NodeDb
+ NDB::Net::NodeMgmt
+ NDB::Net::Server
+ NDB::Net::ServerINET
+ NDB::Net::ServerUNIX
+);
+
+return 1 if $main::onlymodules;
+
+for my $module (@modules) {
+ eval "require $module";
+ $@ and confess "$module $@";
+}
+
+for my $module (@modules) {
+ eval "$module->initmodule";
+ $@ and confess "$module $@";
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/Base.pm b/ndb/tools/ndbnet/lib/NDB/Net/Base.pm
new file mode 100644
index 00000000000..900446138e8
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/Base.pm
@@ -0,0 +1,12 @@
+package NDB::Net::Base;
+
+use strict;
+use Carp;
+
+require NDB::Util::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::Base);
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/Client.pm b/ndb/tools/ndbnet/lib/NDB/Net/Client.pm
new file mode 100644
index 00000000000..d34a18d63af
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/Client.pm
@@ -0,0 +1,252 @@
+package NDB::Net::Client;
+
+use strict;
+use Carp;
+use POSIX();
+use Socket;
+
+require NDB::Net::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+my %clientcache = ();
+my $clientid = 0;
+
+NDB::Net::Client->attributes(
+ id => sub { /^\d+$/ },
+ addtime => sub { /^\d+$/ },
+ state => sub { /^(new|input|cmd)$/ },
+ socket => sub { ref && $_->isa('NDB::Util::Socket') },
+ serversocket => sub { ref && $_->isa('NDB::Util::Socket') },
+ serverlock => sub { ref && $_->isa('NDB::Util::Lock') },
+ event => sub { ref && $_->isa('NDB::Util::Event') },
+ context => sub { defined },
+ cmd => sub { ref && $_->isa('NDB::Net::Command') },
+);
+
+sub desc {
+ my $client = shift;
+ my $id = $client->getid;
+ my $fileno = fileno($client->getsocket->getfh);
+ return "client $id fd=$fileno";
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $client = $class->SUPER::new(%attr);
+ $client->setid(++$clientid)
+ or $log->push, return undef;
+ $client->setaddtime(time)
+ or $log->push, return undef;
+ $client->setstate(q(new))
+ or $log->push, return undef;
+ $client->setsocket($attr{socket})
+ or $log->push, return undef;
+ $client->setserversocket($attr{serversocket})
+ or $log->push, return undef;
+ $client->setserverlock($attr{serverlock})
+ or $log->push, return undef;
+ $client->setevent($attr{event})
+ or $log->push, return undef;
+ $client->setcontext($attr{context})
+ or $log->push, return undef;
+ $log->put("add")->push($client)->info;
+ $clientcache{$client->getid} = $client;
+ return $client;
+}
+
+sub listall {
+ my $class = shift;
+ my $list = [];
+ for my $id (sort { $a <=> $b } keys %clientcache) {
+ my $client = $clientcache{$id};
+ push(@$list, $client);
+ }
+ return $list;
+}
+
+sub exists {
+ my $client = shift;
+ return exists($clientcache{$client->getid});
+}
+
+sub delete {
+ my $client = shift;
+ $log->put("delete")->push($client)->info;
+ $client->getevent->clear($client->getsocket, 'r');
+ $client->getsocket->close;
+ delete $clientcache{$client->getid} or confess 'oops';
+}
+
+sub deleteother {
+ my $thisclient = shift;
+ for my $id (sort { $a <=> $b } keys %clientcache) {
+ my $client = $clientcache{$id};
+ if ($client ne $thisclient) {
+ $client->delete;
+ }
+ }
+}
+
+sub deleteall {
+ my $class = shift;
+ for my $id (sort { $a <=> $b } keys %clientcache) {
+ my $client = $clientcache{$id};
+ $client->delete;
+ }
+}
+
+# processing
+
+sub processnew {
+ my $client = shift;
+ @_ == 0 or confess 0+@_;
+ $log->put("process new")->push($client)->debug;
+ $client->getevent->set($client->getsocket, 'r');
+ $log->attachuser(io => $client->getsocket);
+ $client->setstate(q(input))
+ or $log->push, return undef;
+ return 1;
+}
+
+sub processinput {
+ my $client = shift;
+ @_ == 0 or confess 0+@_;
+ $log->put("process input")->push($client)->debug;
+ my $line = $client->getsocket->readline;
+ if (! defined($line)) {
+ $log->push;
+ return undef;
+ }
+ if (length($line) == 0) {
+ if ($client->getsocket->getreadend) {
+ $log->put("no command")->push($client);
+ return undef;
+ }
+ $log->put("wait for input")->push($client)->debug;
+ return 1;
+ }
+ $log->put("got line: $line")->push($client)->info;
+ $client->getevent->clear($client->getsocket, 'r');
+ my $cmd = NDB::Net::Command->new(line => $line)
+ or $log->push, return undef;
+ $log->put("command received")->push($cmd)->push($client)->debug;
+ $client->setcmd($cmd)
+ or $log->push, return undef;
+ $client->setstate(q(cmd))
+ or $log->push, return undef;
+ return 1;
+}
+
+sub processcmd {
+ my $client = shift;
+ @_ == 0 or confess 0+@_;
+ $log->put("process cmd")->push($client)->debug;
+ my $cmd = $client->getcmd;
+ my $context = $client->getcontext;
+ my $name_fg = "cmd_" . $cmd->getname . "_fg";
+ my $name_bg = "cmd_" . $cmd->getname . "_bg";
+ my $fg = $context->can($name_fg);
+ my $bg = $context->can($name_bg);
+ unless ($fg || $bg) {
+ $log->put("%s: unimplemented", $cmd->getname);
+ return undef;
+ }
+ my $ret;
+ if ($fg) {
+ $log->put($name_fg)->push($cmd)->push($client)->info;
+ if (! ref($context)) {
+ $ret = &$fg($cmd);
+ }
+ else {
+ $ret = &$fg($context, $cmd);
+ }
+ defined($ret)
+ or $log->push, return undef;
+ if (! $bg) {
+ $log->push($name_fg)->putvalue($ret)->user;
+ return 1;
+ }
+ }
+ if ($bg) {
+ $log->put($name_bg)->push($cmd)->push($client)->info;
+ my $pid = fork;
+ if (! defined($pid)) {
+ $log->put("fork failed: $!");
+ return undef;
+ }
+ if ($pid == 0) {
+ $client->getserversocket->close;
+ $client->getserverlock->close;
+ $client->deleteother;
+ if (! ref($context)) {
+ $ret = &$bg($cmd);
+ }
+ else {
+ $ret = &$bg($context, $cmd);
+ }
+ if (! $ret) {
+ $log->push($client)->error;
+ $log->push($name_bg)->putvalue(undef)->user;
+ exit(1);
+ }
+ $log->push($name_bg)->putvalue($ret)->user;
+ exit(0);
+ }
+ }
+ return 1;
+}
+
+sub process {
+ my $client = shift;
+ @_ == 0 or confess 0+@_;
+ try: {
+ if ($client->getstate eq q(new)) {
+ $client->processnew
+ or $log->push, last try;
+ }
+ if ($client->getstate eq q(input)) {
+ $client->processinput
+ or $log->push, last try;
+ }
+ if ($client->getstate eq q(cmd)) {
+ $client->processcmd
+ or $log->push, last try;
+ $log->detachuser;
+ $client->delete;
+ return 1;
+ }
+ return 1;
+ }
+ $log->push($client)->error;
+ $log->putvalue(undef)->user;
+ $log->detachuser;
+ $client->delete;
+ return undef;
+}
+
+sub processall {
+ my $class = shift;
+ @_ == 0 or confess 0+@_;
+ my $list = $class->listall;
+ for my $client (@$list) {
+ $client->process;
+ }
+ while ((my $pid = waitpid(-1, &POSIX::WNOHANG)) > 0) {
+ $log->put("harvested pid=$pid")->info;
+ }
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/Command.pm b/ndb/tools/ndbnet/lib/NDB/Net/Command.pm
new file mode 100644
index 00000000000..30145d09fa9
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/Command.pm
@@ -0,0 +1,641 @@
+package NDB::Net::Command;
+
+use strict;
+use Carp;
+use Getopt::Long;
+use Text::ParseWords ();
+use Text::Tabs ();
+
+require NDB::Net::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+my($cmdtab, $aliastab);
+
+NDB::Net::Command->attributes(
+ name => sub { /^\s*\w+\b/ },
+ argv => sub { ref eq 'ARRAY' },
+ optspec => sub { ref eq 'ARRAY' },
+ argspec => sub { /^\d+$/ || ref eq 'CODE' },
+ short => sub { defined && ! ref },
+ help => sub { defined && ! ref },
+ opts => sub { ref eq 'HASH' },
+ args => sub { ref eq 'ARRAY' },
+);
+
+sub desc {
+ my $cmd = shift;
+ return "command " . $cmd->getname("?");
+};
+
+sub processname {
+ my $cmd = shift;
+ @_ == 0 or confess 0+@_;
+ my $cmdargv = $cmd->getargv;
+ my $name = shift(@$cmdargv);
+ my %seen = ();
+ while ((my $entry) = grep($name eq $_->{name}, @$aliastab)) {
+ $seen{$name}++ && last;
+ unshift(@$cmdargv, split(' ', $entry->{value}));
+ $name = shift(@$cmdargv);
+ }
+ if ((my $entry) = grep($_->{name} eq $name, @$cmdtab)) {
+ $cmd->setname($entry->{name})
+ or $log->push, return undef;
+ $cmd->setoptspec($entry->{optspec})
+ or $log->push, return undef;
+ $cmd->setargspec($entry->{argspec})
+ or $log->push, return undef;
+ }
+ else {
+ $log->put("$name: undefined")->push($cmd);
+ return undef;
+ }
+ return 1;
+}
+
+sub getopttype {
+ my $cmd = shift;
+ my($key) = @_;
+ if (grep(/^$key$/, @{$cmd->getoptspec})) {
+ return 1;
+ }
+ if (grep(/^$key=/, @{$cmd->getoptspec})) {
+ return 2;
+ }
+ return undef;
+}
+
+sub processargv {
+ my $cmd = shift;
+ @_ == 0 or confess 0+@_;
+ my $cmdargv = $cmd->getargv;
+ my @newargv = ();
+ while (@$cmdargv) {
+ my $v = shift(@$cmdargv);
+ if (! defined($v)) {
+ next;
+ }
+ if (ref($v) eq 'ARRAY') {
+ unshift(@$cmdargv, @$v); # push back
+ next;
+ }
+ if (ref($v) eq 'HASH') {
+ for my $k (sort keys %$v) {
+ if ($cmd->getopttype($k) == 1) {
+ push(@newargv, "--$k");
+ next;
+ }
+ if ($cmd->getopttype($k) == 2) {
+ push(@newargv, "--$k", $v->{$k});
+ next;
+ }
+ $log->put("$k: undefined option")->push($cmd);
+ return undef;
+ }
+ next;
+ }
+ if (ref($v)) {
+ confess 'oops';
+ }
+ push(@newargv, $v);
+ }
+ push(@$cmdargv, @newargv);
+ return 1;
+}
+
+sub processopts {
+ my $cmd = shift;
+ @_ == 0 or confess 0+@_;
+ my $cmdargv = $cmd->getargv;
+ local(@ARGV) = @$cmdargv;
+ try: {
+ local $SIG{__WARN__} = sub {
+ my $errstr = "@_";
+ while (chomp($errstr)) {}
+ $log->put($errstr)->push($cmd);
+ };
+ $cmd->setopts({})
+ or $log->push, return undef;
+ Getopt::Long::Configure(qw(
+ default no_getopt_compat no_ignore_case
+ ));
+ GetOptions($cmd->getopts, @{$cmd->getoptspec})
+ or return undef;
+ }
+ $cmd->setargs([ @ARGV ])
+ or $log->push, return undef;
+ return 1;
+}
+
+sub processargs {
+ my $cmd = shift;
+ @_ == 0 or confess 0+@_;
+ my $cmdargs = $cmd->getargs;
+ if ($cmd->getargspec =~ /^\d+$/) {
+ if (@$cmdargs != $cmd->getargspec) {
+ $log->put("invalid arg count %d != %d",
+ scalar(@$cmdargs), $cmd->getargspec)->push($cmd);
+ return undef;
+ }
+ }
+ if (ref($cmd->getargspec) eq 'CODE') {
+ local $_ = scalar(@$cmdargs);
+ if (! &{$cmd->getargspec}()) {
+ $log->put("invalid arg count %d",
+ scalar(@$cmdargs))->push($cmd);
+ return undef;
+ }
+ }
+ return 1;
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my %attr = @_;
+ my $cmd = $class->SUPER::new(%attr);
+ my $cmdargv = [];
+ $cmd->setargv($cmdargv)
+ or $log->push, return undef;
+ my $line = $attr{line};
+ my $argv = $attr{argv};
+ defined($line) != defined($argv) # exactly one
+ or confess 'oops';
+ if (defined($line)) {
+ ! ref($line) or confess 'oops';
+ push(@$cmdargv, Text::ParseWords::shellwords($line));
+ }
+ if (defined($argv)) {
+ ref($argv) eq 'ARRAY' or confess 'oops';
+ push(@$cmdargv, @$argv);
+ }
+ if (! @$cmdargv) {
+ $log->put("empty command");
+ return undef;
+ }
+ $cmd->processname
+ or $log->push, return undef;
+ $cmd->processargv
+ or $log->push, return undef;
+ $cmd->processopts
+ or $log->push, return undef;
+ $cmd->processargs
+ or $log->push, return undef;
+ return $cmd;
+}
+
+sub getline {
+ my $cmd = shift;
+ @_ == 0 or confess 0+@_;
+ my @text = ($cmd->getname);
+ for my $k (sort keys %{$cmd->getopts}) {
+ if ($cmd->getopttype($k) == 1) {
+ push(@text, "--$k");
+ next;
+ }
+ if ($cmd->getopttype($k) == 2) {
+ push(@text, "--$k", quotemeta($cmd->getopts->{$k}));
+ next;
+ }
+ confess 'oops';
+ }
+ for my $s (@{$cmd->getargs}) {
+ push(@text, quotemeta($s));
+ }
+ return "@text";
+}
+
+sub setopt {
+ my $cmd = shift;
+ my($key, $value) = @_;
+ if ($cmd->getopttype($key) == 1) {
+ @_ == 1 or confess 0+@_;
+ $cmd->getopts->{$key} = 1;
+ }
+ elsif ($cmd->getopttype($key) == 2) {
+ @_ == 2 or confess 0+@_;
+ $cmd->getopts->{$key} = $value;
+ }
+ else {
+ confess 'oops';
+ }
+}
+
+sub getopt {
+ my $cmd = shift;
+ @_ == 1 or confess 0+@_;
+ my($key) = @_;
+ $cmd->getopttype($key) or confess 'oops';
+ return $cmd->getopts->{$key};
+}
+
+sub setarg {
+ my $cmd = shift;
+ @_ == 2 or confess 0+@_;
+ my($idx, $value) = @_;
+ $cmd->getargs->[$idx] = $value;
+}
+
+sub getarg {
+ my $cmd = shift;
+ @_ == 1 or confess 0+@_;
+ my($idx) = @_;
+ return $cmd->getargs->[$idx];
+}
+
+sub getarglist {
+ my $cmd = shift;
+ @_ == 1 or confess 0+@_;
+ my($idx) = @_;
+ my @args = @{$cmd->getargs};
+ @args = @args[$idx..$#args];
+ return \@args;
+}
+
+sub helptext {
+ my $cmd = shift;
+ @_ <= 1 or confess 0+@_;
+ my $name = $cmd->getargs->[0];
+ my $text = "";
+ my $indent = " "x4;
+ if (defined($name)) {
+ for my $entry (@$aliastab) {
+ if ($entry->{name} eq $name) {
+ $text .= "alias $name=\"$entry->{value}\"\n";
+ ($name) = split(' ', $entry->{value});
+ last;
+ }
+ }
+ }
+ else {
+ $text .= "COMMANDS\n";
+ }
+ for my $entry (@$cmdtab) {
+ if (defined($name)) {
+ if ($entry->{name} eq $name) {
+ $text .= uc($name) . "\n";
+ for my $t (split(/\n/, $entry->{help})) {
+ $text .= $indent;
+ $text .= Text::Tabs::expand($t) . "\n";
+ }
+ last;
+ }
+ }
+ else {
+ $text .= $indent;
+ $text .= sprintf("%-16s%s\n", $entry->{name}, $entry->{short});
+ }
+ }
+ if (! $text) {
+ $log->put("$name: undefined");
+ return undef;
+ }
+ return $text;
+}
+
+sub aliastext {
+ my $cmd = shift;
+ @_ == 0 or confess 0+@_;
+ my $text = "";
+ my $indent = " "x4;
+ $text .= "ALIASES\n";
+ for my $entry (@$aliastab) {
+ $text .= $indent;
+ $text .= sprintf("%-16s%s\n", $entry->{name}, $entry->{value});
+ }
+ return $text;
+}
+
+# commands
+# name command name (unique)
+# optspec option spec in Getopt::Long style
+# argspec arg count (number or sub)
+# short one line summary
+# help long help text
+# opts options HASH (after parse)
+# args arguments ARRAY (after parse)
+
+$cmdtab = [
+ {
+ name => "help",
+ optspec => [ qw() ],
+ argspec => sub { $_[0] <= 1 },
+ short => "print help (try: h h)",
+ help => <<END,
+help [name]
+name command name or alias
+
+Print help summary or longer help text for one command.
+
+General:
+
+Options can be placed anywhere on command line and can be abbreviated.
+Example: "start db11 -i" instead of "start --init_rm db11".
+
+Several commands have internal option --local which makes current server
+do the work, instead of passing it to other servers. This option should
+not be used explicitly, except for testing.
+END
+ },
+ {
+ name => "alias",
+ optspec => [ qw() ],
+ argspec => 0,
+ short => "list aliases",
+ help => <<END,
+alias
+
+List built-in aliases. New ones cannot be defined (yet).
+END
+ },
+ {
+ name => "quit",
+ optspec => [ qw() ],
+ argspec => 0,
+ short => "exit ndbnet",
+ help => <<END,
+quit
+
+Exit ndbnet client.
+END
+ },
+ {
+ name => "server",
+ optspec => [ qw(all direct pass parallel script=s local) ],
+ argspec => sub { $_ >= 1 },
+ short => "net server commands",
+ help => <<END,
+server action id... [options]
+action start restart stop ping
+id net server id from net config
+--all do all servers listed in net config
+--direct do not use a server
+--pass pass current ndb environment to remote command
+--parallel run in parallel when possible
+--script path remote script instead of "ndbnetd"
+--local for internal use by servers
+
+Each host needs one net server (ndbnetd). It should be started
+from latest ndb installation, for example at system boot time.
+A "server ping" is used to check that all servers are up (option
+--all is added if no server ids are given).
+
+Other actions are mainly for testing. A "server start" tries to
+start servers via "ssh". This does not work if "ssh" is not allowed
+or if the remote command does not get right environment.
+
+Option --direct makes this ndbnet client do the work. It is assumed
+for "server start" and it requires that a local net config exists.
+Option --pass is useful in a homogeneous (NFS) environment.
+
+There are aliases "startserver" for "server start", etc.
+END
+ },
+ {
+ name => "start",
+ optspec => [ qw(init_rm nostart stop kill config old home=s clean proxy=s) ],
+ argspec => 1,
+ short => "start database",
+ help => <<END,
+start dbname [options]
+dbname database name
+--init_rm destroy existing database files on each node
+--nostart for DB nodes only do "ndb -n"
+--stop do "stop dbname" first
+--kill do "kill dbname" first
+--config create run config but start no processes
+--old use existing config files
+--home dir override home (product dir) from config
+--clean passed to startnode
+--proxy list generate proxy ports (read the source)
+
+Start a database as follows:
+
+- start mgmt servers on all mgmt nodes
+- start ndb processes on all db nodes
+- send "all start" to first mgmt server (redundant)
+- start processes on all api nodes (if runtype!="manual")
+
+Older database versions (v1.0) are started similarly except that there
+are no management servers.
+
+The --proxy option is used for testing network problems.
+END
+ },
+ {
+ name => "startnode",
+ optspec => [ qw(init_rm nostart config old run=s home=s local clean proxy=s) ],
+ argspec => 2,
+ short => "start database node",
+ help => <<END,
+startnode dbname nodeid [options]
+dbname database name
+nodeid node number
+--init_rm destroy existing database files (if db node)
+--nostart if DB node only do "ndb -n"
+--config create run config but start no processes
+--old use existing config files
+--run cmd run this shell command, default from config file
+--home dir override home (product dir) from config
+--local node must be local to this ndbnet server
+--clean remove old node dir first
+--proxy list processed by mgmt nodes, see "start" command
+
+Start the process on one database node. The node can be of any type
+(mgmt/db/api). If already running, does nothing.
+
+The --run option specifies a simple shell command (not pipeline etc).
+Defaults:
+
+- mgmt node => mgmtsrvr -p port -l Ndb.cfg -i config.txt -c config.bin
+ where port comes from ndbnet.xml
+- db node => ndb
+- api node => based on ndbnet config, default empty
+
+The node server exits when the command exits (unless runtype is set to
+auto). Command exit status is not available.
+
+Used internally by db "start" command.
+END
+ },
+ {
+ name => "stop",
+ optspec => [ qw() ],
+ argspec => 1,
+ short => "stop database",
+ help => <<END,
+stop dbname [options]
+dbname database name
+
+Stop a database as follows (see also "stopnode" command):
+
+- send SIGTERM to api processes, wait for them to exit
+- send "all stop" command to first mgmt server
+- wait for db processes to exit
+- send "quit" to mgmt servers, wait for them to exit
+END
+ },
+ {
+ name => "stopnode",
+ optspec => [ qw(local) ],
+ argspec => 2,
+ short => "stop process on one node",
+ help => <<END,
+stopnode dbname nodeid [options]
+dbname database name
+nodeid node number
+--local node must be local to this server
+
+Stop process on one database node. Action depends on node type:
+
+- api node: send SIGTERM to the process, wait for it to exit
+- db node: no action, wait for the ndb process to exit
+- mgmt node: send "quit" command to mgmt server, wait for it to exit
+
+Used internally by db "stop" command.
+END
+ },
+ {
+ name => "kill",
+ optspec => [ qw() ],
+ argspec => 1,
+ short => "kill processes on all nodes",
+ help => <<END,
+kill dbname [options]
+dbname database name
+
+Send SIGKILL to processes on all nodes and wait for them to exit.
+END
+ },
+ {
+ name => "killnode",
+ optspec => [ qw(local) ],
+ argspec => 2,
+ short => "kill process on one node",
+ help => <<END,
+killnode dbname nodeid [options]
+dbname database name
+nodeid node number
+--local node must be local to this server
+
+Send SIGKILL to the process on the node and wait for it to exit.
+
+Used internally by db "kill" command.
+END
+ },
+ {
+ name => "statnode",
+ optspec => [ qw(local) ],
+ argspec => 2,
+ short => "get node run status (internal)",
+ help => <<END,
+statnode dbname nodeid [options]
+dbname database name
+nodeid node number
+--local node must be local to this server
+
+Get node run status (up/down) as a process. Used internally
+and may not produce any output in ndbnet command.
+END
+ },
+ {
+ name => "list",
+ optspec => [ qw(quick short) ],
+ argspec => sub { 1 },
+ short => "list databases",
+ help => <<END,
+list [dbname] [options]
+dbname database name, default is to list all
+--quick only output config, do not query status
+--short do list nodes
+
+List databases and nodes. Internally returns a data structure
+of process and mgmt server status values for each node. Externally
+(in ndbnet command) this is formatted as a listing.
+END
+ },
+ {
+ name => "writenode",
+ optspec => [ qw(wait=i local) ],
+ argspec => 3,
+ short => "write line of text to the process on a node",
+ help => <<END,
+writenode dbname nodeid "some text"
+dbname database name
+nodeid node number
+"some text" arbitrary text (quote if spaces)
+--wait n wait n seconds for any response
+--local node must be local to this server
+
+Write the text and a newline to the standard input of the process
+running on the node. If wait > 0 is specified, prints whatever
+the process wrote to stdout/stderr during that time.
+
+Used internally by "start" and other commands.
+END
+ },
+];
+
+# aliases
+# name alias
+# value expansion
+
+$aliastab = [
+ {
+ name => "h",
+ value => "help",
+ },
+ {
+ name => "q",
+ value => "quit",
+ },
+ {
+ name => "EOF",
+ value => "quit",
+ },
+ {
+ name => "startserver",
+ value => "server start",
+ },
+ {
+ name => "ss",
+ value => "server start",
+ },
+ {
+ name => "restartserver",
+ value => "server restart",
+ },
+ {
+ name => "rss",
+ value => "server restart",
+ },
+ {
+ name => "stopserver",
+ value => "server stop",
+ },
+ {
+ name => "pingserver",
+ value => "server ping",
+ },
+ {
+ name => "ps",
+ value => "server ping",
+ },
+ {
+ name => "l",
+ value => "list",
+ },
+];
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/Config.pm b/ndb/tools/ndbnet/lib/NDB/Net/Config.pm
new file mode 100644
index 00000000000..4c5db3cd3f5
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/Config.pm
@@ -0,0 +1,235 @@
+package NDB::Net::Config;
+
+use strict;
+use Carp;
+use Symbol;
+use Socket;
+use Errno;
+use XML::Parser;
+
+require NDB::Net::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Net::Config->attributes(
+ file => sub { /^\S+$/ },
+ loadtime => sub { /^\d+$/ },
+);
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $netcfg = $class->SUPER::new(%attr);
+ $netcfg->setfile($attr{file})
+ or $log->put, return undef;
+ return $netcfg;
+}
+
+sub desc {
+ my $netcfg = shift;
+ return $netcfg->getfile;
+}
+
+use vars qw(@context);
+
+sub handle_start {
+ my($parser, $tag, %attr) = @_;
+ my $p = $context[-1];
+ my $q = {};
+ $p->{$tag} ||= [];
+ push(@{$p->{$tag}}, $q);
+ for my $k (keys %attr) {
+ $q->{$k} = $attr{$k};
+ }
+ push(@context, $q);
+ return 1;
+}
+
+sub handle_end {
+ my($parser, $tag, %attr) = @_;
+ pop(@context);
+ return 1;
+}
+
+sub load {
+ my $netcfg = shift;
+ my $file = $netcfg->getfile;
+ my @s;
+ while (1) {
+ if (@s = stat($file)) {
+ last;
+ }
+ $log->put("$file: stat failed: $!");
+ if (! $!{ESTALE}) {
+ return undef;
+ }
+ $log->put("(retry)")->info;
+ sleep 1;
+ }
+ if ($s[9] <= $netcfg->getloadtime(0)) {
+ return 1;
+ }
+ my $fh = gensym();
+ if (! open($fh, "<$file")) {
+ $log->put("$file: open for read failed: $!");
+ return undef;
+ }
+ my $text = "";
+ my $line;
+ while (defined($line = <$fh>)) {
+ $text .= $line;
+ }
+ close($fh);
+ my $parser = XML::Parser->new(
+ ParseParamEnt => 1,
+ Handlers => {
+ Start => \&handle_start,
+ End => \&handle_end,
+ },
+ );
+ delete $netcfg->{config};
+ local @context = ($netcfg);
+ $parser->parse($text);
+ $netcfg->{text} = $text;
+ $netcfg->{config} = $netcfg->{config}[0];
+ $netcfg->setloadtime(time)
+ or $log->push, return undef;
+ NDB::Net::Server->deleteall;
+ NDB::Net::Database->deleteall;
+ NDB::Net::Node->deleteall;
+ return 1;
+}
+
+sub getservers {
+ my $netcfg = shift;
+ @_ == 0 or confess 0+@_;
+ my $servers = [];
+ my $slist = $netcfg->{config}{server} || [];
+ for my $s (@$slist) {
+ my $server;
+ $server = NDB::Net::ServerINET->get($s->{id});
+ if (! $server) {
+ $server = NDB::Net::ServerINET->new(%$s);
+ if (! $server) {
+ $log->push($netcfg)->warn;
+ next;
+ }
+ }
+ push(@$servers, $server);
+ }
+ return $servers;
+}
+
+sub getdatabases {
+ my $netcfg = shift;
+ @_ == 0 or confess 0+@_;
+ my $databases = [];
+ my $dlist = $netcfg->{config}{database} || [];
+ for my $d (@$dlist) {
+ if ($d->{isproto} eq "y") {
+ next;
+ }
+ if ($d->{name} !~ /^\w(\w|-)*$/) {
+ $log->put("$d->{name}: invalid db name")->push($netcfg)->warn;
+ next;
+ }
+ my $db = $netcfg->getdatabase($d->{name});
+ if (! $db) {
+ $log->push->warn;
+ next;
+ }
+ push(@$databases, $db);
+ }
+ return $databases;
+}
+
+sub getdatabase {
+ my $netcfg = shift;
+ @_ == 1 or confess 0+@_;
+ my($name) = @_;
+ $netcfg->getservers or return undef; # cache them
+ my $default = $netcfg->{config}{default}[0] || {};
+ my $db;
+ my $dlist = $netcfg->{config}{database} || [];
+ my $nlist;
+ for my $d (@$dlist) {
+ ($d->{name} ne $name) && next;
+ if ($d->{isproto} eq "y") {
+ next;
+ }
+ my %attr = (%$default, %$d);
+ $db = NDB::Net::Database->new(%attr);
+ if (! $db) {
+ $log->push($netcfg);
+ return undef;
+ }
+ if ($d->{proto}) {
+ if ($d->{isproto} eq "y") {
+ $log->put("$name: prototypes cannot be recursive");
+ return undef;
+ }
+ for my $d2 (@$dlist) {
+ ($d2->{name} ne $d->{proto}) && next;
+ if ($d2->{isproto} ne "y") {
+ $log->put("$name: $d2->{name} is not a prototype");
+ return undef;
+ }
+ if (! $d->{node}) {
+ $d->{node} = $d2->{node};
+ }
+ last;
+ }
+ }
+ $nlist = $d->{node} || [];
+ last;
+ }
+ if (! $db) {
+ $log->put("$name: no such db")->push($netcfg);
+ return undef;
+ }
+ if (! @$nlist) {
+ $log->put("$name: empty node list")->push($netcfg);
+ return undef;
+ }
+ for my $n (@$nlist) {
+ my $node;
+ try: {
+ my $server = NDB::Net::Server->get($n->{server})
+ or last try;
+ my %attr = (%$n, db => $db, server => $server);
+ my $type = $attr{type};
+ if ($type eq 'db') {
+ $node = NDB::Net::NodeDb->new(%attr)
+ or last try;
+ }
+ if ($type eq 'mgmt') {
+ $node = NDB::Net::NodeMgmt->new(%attr)
+ or last try;
+ }
+ if ($type eq 'api') {
+ $node = NDB::Net::NodeApi->new(%attr)
+ or last try;
+ }
+ $log->put("bad node type '$type'");
+ }
+ if (! $node) {
+ $log->push($netcfg);
+ $db->delete;
+ return undef;
+ }
+ }
+ return $db;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/Database.pm b/ndb/tools/ndbnet/lib/NDB/Net/Database.pm
new file mode 100644
index 00000000000..7ea15be0650
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/Database.pm
@@ -0,0 +1,321 @@
+package NDB::Net::Database;
+
+use strict;
+use Carp;
+use Symbol;
+
+require NDB::Net::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+my %dbcache = ();
+
+NDB::Net::Database->attributes(
+ name => sub { s/^\s+|\s+$//g; /^\S+$/ && ! m!/! },
+ comment => sub { defined },
+ version => sub { /^\d+(\.\d+)*$/ },
+ base => sub { $^O eq 'MSWin32' || m!^/\S+$! },
+ home => sub { $^O eq 'MSWin32' || m!^/\S+$! },
+ nodeport => sub { $_ > 0 },
+);
+
+sub desc {
+ my $db = shift;
+ return $db->getname;
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $db = $class->SUPER::new(%attr);
+ $db->setname($attr{name})
+ or $log->push, return undef;
+ if ($dbcache{$db->getname}) {
+ $log->put("duplicate db")->push($db);
+ return undef;
+ }
+ $db->setcomment($attr{comment});
+ $db->setversion($attr{version})
+ or $log->push, return undef;
+ if (defined($attr{base})) {
+ $db->setbase($attr{base})
+ or $log->push, return undef;
+ }
+ if (defined($attr{home})) {
+ if ($^O ne 'MSWin32' && $attr{home} !~ m!^/! && $db->hasbase) {
+ $attr{home} = $db->getbase . "/$attr{home}";
+ }
+ $db->sethome($attr{home})
+ or $log->push, return undef;
+ }
+ if (defined($attr{nodeport})) {
+ $db->setnodeport($attr{nodeport})
+ or $log->push, return undef;
+ }
+ if ($^O eq 'MSWin32' && ! $db->hasnodeport) {
+ $log->put("nodeport required on windows")->push($db), return undef;
+ }
+ $db->{nodehash} = {};
+ $dbcache{$db->getname} = $db;
+ return $db;
+}
+
+sub delete {
+ my $db = shift;
+ my $nodelist = $db->getnodelist('all');
+ for my $node (@$nodelist) {
+ $node->delete;
+ }
+ delete $dbcache{$db->getname};
+}
+
+sub deleteall {
+ my $class = shift;
+ for my $name (sort keys %dbcache) {
+ my $db = $dbcache{$name};
+ $db->delete;
+ }
+}
+
+# assume numerical dot separated version numbers like 1.1.2
+sub cmpversion {
+ my $db = shift;
+ my $version = shift;
+ my @x = split(/\./, $db->getversion);
+ my @y = split(/\./, $version);
+ while (@x || @y) {
+ return -1 if $x[0] < $y[0];
+ return +1 if $x[0] > $y[0];
+ shift(@x);
+ shift(@y);
+ }
+ return 0;
+}
+
+# nodes
+
+sub addnode {
+ my $db = shift;
+ @_ == 1 or confess 0+@_;
+ my($node) = @_;
+ unless (ref($node) && $node->isa('NDB::Net::Node')) {
+ confess 'oops';
+ }
+ my $id = $node->getid;
+ if ($db->{nodehash}{$id}) {
+ $log->put("$id: duplicate node id")->push($db);
+ return undef;
+ }
+ $db->{nodehash}{$id} = $node;
+ return 1;
+}
+
+sub getnode {
+ my $db = shift;
+ @_ == 1 or confess 0+@_;
+ my($id) = @_;
+ $id += 0;
+ my $node = $db->{nodehash}{$id};
+ if (! $node) {
+ $log->put("$id: no such node id")->push($db);
+ return undef;
+ }
+ return $node;
+}
+
+sub getnodelist {
+ my $db = shift;
+ @_ == 1 or confess 0+@_;
+ my($type) = @_;
+ $type =~ /^(all|mgmt|db|api)$/ or confess 'oops';
+ my @nodes = ();
+ for my $id (sort { $a <=> $b } keys %{$db->{nodehash}}) {
+ my $node = $db->{nodehash}{$id};
+ if ($type eq 'all' or $type eq $node->gettype) {
+ push(@nodes, $node);
+ }
+ }
+ return \@nodes;
+}
+
+# start /stop
+
+sub start {
+ my $db = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ if ($opts->{stop} || $opts->{kill}) {
+ my $method = $opts->{stop} ? "stop" : "kill";
+ my %opts = ();
+ $db->$method(\%opts)
+ or $log->push, return undef;
+ }
+ $log->put("start")->push($db)->info;
+ my $nodesmgmt = $db->getnodelist('mgmt');
+ my $nodesdb = $db->getnodelist('db');
+ my $nodesapi = $db->getnodelist('api');
+ my $ret;
+ try: {
+ my %startopts = ();
+ for my $k (qw(local init_rm nostart config old home clean proxy)) {
+ $startopts{$k} = $opts->{$k} if defined($opts->{$k});
+ }
+ my %writeopts = ();
+ for my $k (qw(local)) {
+ $writeopts{$k} = $opts->{$k} if defined($opts->{$k});
+ }
+ if ($db->cmpversion("1.0") > 0) {
+ for my $node (@$nodesmgmt) {
+ $node->start(\%startopts) or last try;
+ }
+ for my $node (@$nodesdb) {
+ $node->start(\%startopts) or last try;
+ }
+ if (! $opts->{config}) {
+ for my $node (@$nodesmgmt) { # probably redundant
+ $node->write(\%writeopts, "all start") or last try;
+ last;
+ }
+ }
+ }
+ else {
+ for my $node (@$nodesdb) {
+ $node->start(\%startopts) or last try;
+ }
+ if (! $opts->{config}) {
+ for my $node (@$nodesdb) { # probably redundant
+ $node->write(\%writeopts, "start") or last try;
+ }
+ }
+ }
+ for my $node (@$nodesapi) {
+ my %apiopts = %startopts;
+ if ($node->getruntype eq 'manual') {
+ $apiopts{config} = 1;
+ }
+ $node->start(\%apiopts) or last try;
+ }
+ $ret = 1;
+ }
+ if (! $ret) {
+ $log->push("start failed")->push($db);
+ return undef;
+ }
+ my $msg = ! $opts->{config} ? "start done" : "config created";
+ $log->put($msg)->push($db)->user;
+ return 1;
+}
+
+sub stop {
+ my $db = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ $log->put("stop")->push($db)->info;
+ my $nodesmgmt = $db->getnodelist('mgmt');
+ my $nodesdb = $db->getnodelist('db');
+ my $nodesapi = $db->getnodelist('api');
+ my $ret;
+ try: {
+ for my $node (@$nodesapi) {
+ $node->stop($opts) or last try;
+ }
+ if ($db->cmpversion("1.0") > 0) {
+ for my $node (@$nodesmgmt) {
+ $node->write($opts, "all stop") or last try;
+ last;
+ }
+ for my $node (@$nodesdb) {
+ $node->stop($opts) or last try;
+ }
+ for my $node (@$nodesmgmt) {
+ $node->stop($opts) or last try;
+ }
+ }
+ else {
+ for my $node (@$nodesdb) {
+ $node->write($opts, "stop") or last try;
+ }
+ for my $node (@$nodesdb) {
+ $node->stop($opts) or last try;
+ }
+ }
+ $ret = 1;
+ }
+ if (! $ret) {
+ $log->push("stop failed")->push($db);
+ return undef;
+ }
+ $log->put("stop done")->push($db)->user;
+ return 1;
+}
+
+sub kill {
+ my $db = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ $log->put("kill")->push($db)->info;
+ my $nodesmgmt = $db->getnodelist('mgmt');
+ my $nodesdb = $db->getnodelist('db');
+ my $nodesapi = $db->getnodelist('api');
+ my $ret = 1;
+ try: {
+ for my $node (@$nodesapi) {
+ $node->kill($opts) || ($ret = undef);
+ }
+ for my $node (@$nodesdb) {
+ $node->kill($opts) || ($ret = undef);
+ }
+ for my $node (@$nodesmgmt) {
+ $node->kill($opts) || ($ret = undef);
+ }
+ }
+ if (! $ret) {
+ $log->push("kill failed")->push($db);
+ return undef;
+ }
+ $log->put("kill done")->push($db)->user;
+ return 1;
+}
+
+sub list {
+ my $db = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ my $dbsts = {};
+ $dbsts->{comment} = $db->getcomment("");
+ $dbsts->{home} = $db->gethome;
+ $log->put("status")->push($db)->info;
+ my $mgmsts;
+ for my $node (@{$db->getnodelist('mgmt')}) {
+ $mgmsts = $node->get_status or
+ $log->push->error;
+ last;
+ }
+ $mgmsts ||= {};
+ for my $node (@{$db->getnodelist('all')}) {
+ my $id = $node->getid;
+ my $nodests = $dbsts->{node}{$id} ||= {};
+ my $stat = $node->stat($opts) or
+ $log->push->error;
+ $nodests->{id} = $id;
+ $nodests->{type} = $node->gettype;
+ $nodests->{comment} = $node->getcomment("");
+ $nodests->{host} = $node->getserver->gethost;
+ $nodests->{run} = $stat || "error";
+ $nodests->{status} = $mgmsts->{node}{$id};
+ }
+ return $dbsts;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/Env.pm b/ndb/tools/ndbnet/lib/NDB/Net/Env.pm
new file mode 100644
index 00000000000..d79e72f2bb3
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/Env.pm
@@ -0,0 +1,94 @@
+package NDB::Net::Env;
+
+use strict;
+use File::Spec;
+use Carp;
+
+require NDB::Net::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Base);
+
+# environment variables
+#
+# NDB_TOP source dir or installation dir
+# NDB_BASE base dir not tied to any release or database
+# NDB_NETCFG ndbnet config file, default $NDB_BASE/etc/ndbnet.xml
+#
+# ndbnet explicitly unsets NDB_TOP and NDB_HOME because they are
+# specific to each database or database node
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Net::Env->attributes(
+ base => sub { /^\S+$/ },
+ netcfg => sub { /^\S+$/ },
+ hostname => sub { /^\S+$/ },
+);
+
+my $instance;
+
+sub desc {
+ my $netenv = shift;
+ return "net environment";;
+}
+
+sub instance {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ if ($instance) {
+ return $instance;
+ }
+ for my $var (qw(NDB_TOP NDB_HOME)) {
+ my $top = delete $ENV{$var};
+ if (defined($top)) {
+ if ($^O ne 'MSWin32') {
+ $ENV{PATH} =~ s!(^|:)$top/bin($|:)!$1$2!g;
+ $ENV{LD_LIBRARY_PATH} =~ s!(^|:)$top/lib($|:)!$1$2!g;
+ $ENV{PERL5LIB} =~ s!(^|:)$top/lib/perl5($|:)!$1$2!g;
+ }
+ }
+ }
+ my $netenv = $class->SUPER::new(%attr);
+ for my $base ($attr{base}, $ENV{NDB_BASE}) {
+ if (defined($base)) {
+ $netenv->setbase($base)
+ or $log->push, return undef;
+ }
+ }
+ for my $netcfg ($attr{netcfg}, $ENV{NDB_NETCFG}) {
+ if (defined($netcfg)) {
+ $netenv->setnetcfg($netcfg)
+ or $log->push, return undef;
+ }
+ }
+ if ($netenv->hasbase && ! $netenv->hasnetcfg) {
+ $netenv->setnetcfg(File::Spec->catfile($netenv->getbase, "etc", "ndbnet.xml"))
+ or $log->push, return undef;
+ }
+ my $uname;
+ if ($^O ne 'MSWin32') {
+ chomp($uname = `uname -n`);
+ } else {
+ chomp($uname = `hostname`);
+ }
+ my($hostname) = gethostbyname($uname);
+ if (! defined($hostname)) {
+ $uname =~ s!\..*$!!;
+ ($hostname) = gethostbyname($uname);
+ }
+ $netenv->sethostname($hostname)
+ or $log->push, return undef;
+ $instance = $netenv;
+ return $instance;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/Node.pm b/ndb/tools/ndbnet/lib/NDB/Net/Node.pm
new file mode 100644
index 00000000000..f41bf51168d
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/Node.pm
@@ -0,0 +1,747 @@
+package NDB::Net::Node;
+
+use strict;
+use Carp;
+use Symbol;
+use Socket;
+use IPC::Open3;
+use POSIX();
+use Errno;
+use File::Spec;
+
+require NDB::Net::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+my %nodecache = ();
+
+NDB::Net::Node->attributes(
+ db => sub { ref && $_->isa('NDB::Net::Database') },
+ comment => sub { defined },
+ id => sub { s/^\s+|\s+$//g; s/^0+(\d+)$/$1/; /^\d+$/ && $_ > 0 },
+ type => sub { s/^\s+|\s+$//g; /^(mgmt|db|api)$/ },
+ server => sub { ref && $_->isa('NDB::Net::Server') },
+ base => sub { File::Spec->file_name_is_absolute($_) },
+ home => sub { File::Spec->file_name_is_absolute($_) },
+ state => sub { /^(new|run|stop)$/ },
+ run => sub { defined },
+ runenv => sub { defined },
+ runtype => sub { m!(auto|once|manual)$! },
+ lockpid => sub { $_ != 0 },
+ iow => sub { ref && $_->isa('NDB::Util::IO') },
+ ior => sub { ref && $_->isa('NDB::Util::IO') },
+ pid => sub { $_ > 1 },
+ event => sub { ref && $_->isa('NDB::Util::Event') },
+);
+
+sub desc {
+ my $node = shift;
+ my $dbname = $node->getdb->getname;
+ my $id = $node->getid;
+ my $type = $node->gettype;
+ return "$dbname.$id-$type";
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $node = $class->SUPER::new(%attr);
+ $node->setdb($attr{db})
+ or $log->push, return undef;
+ $node->setid($attr{id})
+ or $log->push, return undef;
+ if ($nodecache{$node->getdb->getname,$node->getid}) {
+ $log->put("duplicate node")->push($node);
+ return undef;
+ }
+ $node->setcomment($attr{comment});
+ $node->settype($attr{type})
+ or $log->push, return undef;
+ if ($node->getdb->cmpversion("1.0") <= 0 && $node->gettype eq 'mgmt') {
+ $log->put("no mgmt nodes in db version <= 1.0")->push($node);
+ return undef;
+ }
+ $node->setserver($attr{server})
+ or $log->push, return undef;
+ for my $base ($attr{base}, $node->getdb->getbase(undef)) {
+ if (defined($base)) {
+ $node->setbase($base)
+ or $log->push, return undef;
+ }
+ }
+ for my $home ($attr{home}, $node->getdb->gethome(undef)) {
+ if (defined($home)) {
+ if ($^O ne 'MSWin32' && $home !~ m!^/! && $node->hasbase) {
+ $home = $node->getbase . "/$home";
+ }
+ $node->sethome($home)
+ or $log->push, return undef;
+ }
+ }
+ if (! $node->hashome) {
+ $log->put("home not defined")->push($node);
+ return undef;
+ }
+ $node->setstate('new')
+ or $log->push, return undef;
+ if (defined($attr{run})) {
+ $node->setrun($attr{run})
+ or $log->push, return undef;
+ }
+ if (defined($attr{runenv})) {
+ $node->setrunenv($attr{runenv})
+ or $log->push, return undef;
+ }
+ if (defined($attr{runtype})) {
+ $node->setruntype($attr{runtype})
+ or $log->push, return undef;
+ }
+ if (! $node->hasruntype) {
+ my $runtype = "manual";
+ $runtype = "once"
+ if $node->gettype =~ /^(mgmt|db)$/ || $node->hasrun;
+ $node->setruntype($runtype)
+ or $log->push, return undef;
+ }
+ if (! $node->getdb->addnode($node)) {
+ $log->push;
+ return undef;
+ }
+ $nodecache{$node->getdb->getname,$node->getid} = $node;
+ return $node;
+}
+
+sub delete {
+ my $node = shift;
+ delete $nodecache{$node->getdb->getname,$node->getid} or
+ confess 'oops';
+}
+
+sub deleteall {
+ my $class = shift;
+ for my $k (sort keys %nodecache) {
+ my $node = $nodecache{$k};
+ $node->delete;
+ }
+}
+
+# node startup
+
+sub getconfdir {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $netenv = NDB::Net::Env->instance;
+ my $name = File::Spec->catfile($netenv->getbase, "etc");
+ my $dir = NDB::Util::Dir->new(path => $name);
+ return $dir;
+}
+
+sub getdbdir {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $netenv = NDB::Net::Env->instance;
+ my $name = File::Spec->catfile($netenv->getbase, "db", $node->getdb->getname);
+ my $dir = NDB::Util::Dir->new(path => $name);
+ return $dir;
+}
+
+sub getnodedir {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = sprintf("%s-%s", $node->getid, $node->gettype);
+ my $dir = $node->getdbdir->getdir($name);
+ return $dir;
+}
+
+sub getrundir {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = sprintf("run");
+ my $dir = $node->getdbdir->getdir($name);
+ return $dir;
+}
+
+sub getlogdir {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = sprintf("log");
+ my $dir = $node->getdbdir->getdir($name);
+ return $dir;
+}
+
+sub getlock {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = sprintf("%s-%s.pid", $node->getid, $node->gettype);
+ my $lock = $node->getrundir->getfile($name)->getlock;
+ return $lock;
+}
+
+sub getsocketfile {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = sprintf("%s-%s.socket", $node->getid, $node->gettype);
+ my $file = $node->getrundir->getfile($name);
+ return $file;
+}
+
+sub getlogfile {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = sprintf("%s-%s.log", $node->getid, $node->gettype);
+ my $file = $node->getlogdir->getfile($name);
+ return $file;
+}
+
+sub getshellfile {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = sprintf("run.sh");
+ my $file = $node->getnodedir->getfile($name);
+ return $file;
+}
+
+sub getlocalcfg {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = "Ndb.cfg";
+ my $file = $node->getnodedir->getfile($name);
+ return $file;
+}
+
+sub writelocalcfg {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $db = $node->getdb;
+ my $file = $node->getlocalcfg;
+ $file->mkdir or $log->push, return undef;
+ if ($db->cmpversion("1.0") <= 0) {
+ my $section = "";
+ my $edit = sub {
+ chomp;
+ if (/^\s*\[\s*(\S+)\s*\]/) {
+ $section = uc($1);
+ }
+ if ($section eq 'OWN_HOST') {
+ if (/^\s*ThisHostId\b/i) {
+ $_ = "ThisHostId " . $node->getid;
+ }
+ }
+ if ($section eq 'CM') {
+ if (/^\s*ThisNodeId\b/i) {
+ $_ = "ThisNodeId " . $node->getid;
+ }
+ }
+ if (0 and $section eq 'PROCESS_ID') {
+ if (/^\s*Host(\d+)\s+(\S+)(.*)/) {
+ my $id2 = $1;
+ my $host2 = $2;
+ my $rest2 = $3;
+ my $node2 = $db->getnode($id2)
+ or $log->push, return undef;
+ $_ = "Host$id2 ";
+ $_ .= $node2->getserver->getcanon;
+ $_ .= " $rest2";
+ }
+ }
+ $_ .= "\n";
+ return 1;
+ };
+ $node->getinifile->copyedit($file, $edit)
+ or $log->push, return undef;
+ }
+ else {
+ my @text = ();
+ push(@text, sprintf("OwnProcessId %s", $node->getid));
+ my $nodesmgmt = $db->getnodelist('mgmt');
+ for my $mnode (@$nodesmgmt) {
+ my $host = $mnode->getserver->getcanon;
+ my $port = $mnode->getport;
+ push(@text, "$host $port");
+ }
+ $file->putlines(\@text) or $log->push, return undef;
+ }
+ return 1;
+}
+
+sub getinifile {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = sprintf("%s.ini", $node->getdb->getname);
+ my $file = $node->getconfdir->getfile($name);
+ return $file;
+}
+
+sub getbincfg {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = sprintf("config.bin");
+ my $file = $node->getnodedir->getfile($name);
+ return $file;
+}
+
+sub getenvdefs {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my $opts = shift;
+ my $home = $opts->{home} || $node->gethome;
+ my $netenv = NDB::Net::Env->instance;
+ if (! File::Spec->file_name_is_absolute($home)) {
+ $netenv->hasbase
+ or $log->put("no base and home=$home not absolute"), return undef;
+ $home = File::Spec->catfile($netenv->getbase, $home);
+ }
+ (-d $home)
+ or $log->put("$home: no such directory"), return undef;
+ my $defs;
+ if ($^O ne 'MSWin32') {
+ $defs = <<END;
+# @{[ $node->desc ]} @{[ $node->getcomment("") ]}
+# @{[ $node->getserver->desc ]} @{[ $node->getserver->getcanon ]}
+#
+debugger=\$1
+#
+NDB_TOP=$home
+export NDB_TOP
+PATH=\$NDB_TOP/bin:\$PATH
+export PATH
+LD_LIBRARY_PATH=\$NDB_TOP/lib:\$LD_LIBRARY_PATH
+export LD_LIBRARY_PATH
+PERL5LIB=\$NDB_TOP/lib/perl5:\$PERL5LIB
+export PERL5LIB
+NDB_NODEID=@{[ $node->getid ]}
+export NDB_NODEID
+NDB_NODETYPE=@{[ $node->gettype ]}
+export NDB_NODETYPE
+ulimit -Sc unlimited
+END
+ if ($node->hasrunenv) {
+ $defs .= <<END;
+#
+cd @{[ $node->getnodedir->getpath ]} || exit 1
+@{[ $node->getrunenv ]}
+END
+ }
+ $defs .= <<END;
+#
+unset NDB_HOME # current NdbConfig.c would look here
+#
+END
+ } else {
+ $defs = <<END;
+rem @{[ $node->desc ]} @{[ $node->getcomment("") ]}
+rem @{[ $node->getserver->desc ]} @{[ $node->getserver->getcanon ]}
+rem
+set NDB_TOP=$home
+set PATH=%NDB_TOP%\\bin;%PATH%
+set PERL5LIB=%NDB_TOP%\\lib\\perl5;%PERL5LIB%
+set NDB_NODEID=@{[ $node->getid ]}
+set NDB_NODETYPE=@{[ $node->gettype ]}
+END
+ if ($node->hasrunenv) {
+ $defs .= <<END;
+rem
+@{[ $node->getrunenv ]}
+END
+ }
+ $defs .= <<END;
+rem
+rem current NdbConfig.c would look here
+set NDB_HOME=
+rem
+END
+ }
+ chomp($defs);
+ return $defs;
+}
+
+sub startlocal {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ $log->put("start local")->push($node)->info;
+ my $lock = $node->getlock;
+ $lock->mkdir or $log->push, return undef;
+ anon: {
+ my $ret = $lock->test;
+ defined($ret) or $log->push, return undef;
+ if ($ret) {
+ $log->put("already running under serverpid=%s",
+ $lock->getpid)->push($node)->user;
+ return 1;
+ }
+ $lock->set or $log->push, return undef;
+ }
+ if ($opts->{clean}) {
+ $node->getnodedir->rmdir(1);
+ $node->getlogfile->unlink;
+ }
+ if (! $opts->{old}) {
+ $node->writelocalcfg or $log->push, return undef;
+ $node->handleprepare($opts) or $log->push, return undef;
+ }
+ anon: {
+ $lock->close;
+ if ($opts->{config}) {
+ return 1;
+ }
+ my $file = $node->getlogfile;
+ $file->mkdir or $log->push, return undef;
+ my $pid = fork();
+ defined($pid) or $log->put("fork failed: $!"), return undef;
+ if ($pid) {
+ exit(0);
+ }
+ $lock->set or $log->push->fatal;
+ $node->setlockpid($$) or $log->push->fatal;
+ if ($^O ne 'MSWin32') {
+ POSIX::setsid() or $log->put("setsid failed: $!")->fatal;
+ }
+ $log->setfile($file->getpath) or $log->push->fatal;
+ }
+ my $socket;
+ anon: {
+ my $file = $node->getsocketfile;
+ $file->mkdir or $log->push($node)->fatal;
+ unlink($file->getpath);
+ if ($^O ne 'MSWin32') {
+ $socket = NDB::Util::SocketUNIX->new
+ or $log->push($node)->fatal;
+ } else {
+ $socket = NDB::Util::SocketINET->new
+ or $log->push($node)->fatal;
+ }
+ $socket->setopt(SOL_SOCKET, SO_REUSEADDR, 1)
+ or $log->push($node)->fatal;
+ if ($^O ne 'MSWin32') {
+ $socket->bind($file->getpath)
+ or $log->push($node)->fatal;
+ } else {
+ $socket->bind($node->getdb->getnodeport + $node->getid)
+ or $log->push($node)->fatal;
+ }
+ $socket->listen
+ or $log->push($node)->fatal;
+ }
+ START: {
+ my $w = gensym();
+ my $r = gensym();
+ my @arg = ('/bin/sh', $node->getshellfile->getpath);
+ my $pid = open3($w, $r, undef, @arg);
+ $node->setiow(NDB::Util::IO->new(fh => $w))
+ or $log->push->fatal;
+ $node->setior(NDB::Util::IO->new(fh => $r))
+ or $log->push->fatal;
+ $node->setpid($pid)
+ or $log->push->fatal;
+ }
+ $node->setstate('run')
+ or $log->push($node)->fatal;
+ $log->put("started host=%s pid=%s",
+ $node->getserver->gethost, $node->getpid)->push($node)->user;
+ $log->push("started")->push($node)->putvalue(1)->user;
+ $log->detachuser;
+ NDB::Net::Client->deleteall;
+ my $event = NDB::Util::Event->new;
+ $event->set($socket, 'r');
+ $event->set($node->getior, 'r');
+ loop: {
+ try: {
+ my $n = $event->poll(10);
+ if (! defined($n)) {
+ $log->push->error;
+ sleep 1;
+ last try;
+ }
+ if (! $n) {
+ $log->push->debug;
+ last try;
+ }
+ if ($node->hasior && $event->test($node->getior, 'r')) {
+ my $data = $node->getior->read;
+ if (! defined($data)) {
+ $log->push->fatal;
+ }
+ if (length($data) > 0) {
+ $node->handleoutput($opts, $data);
+ }
+ if ($node->getior->getreadend) {
+ $log->put("input closed")->warn;
+ $event->clear($node->getior, 'r');
+ $node->getior->close;
+ $node->delior;
+ $node->handleeof($opts);
+ last loop;
+ }
+ }
+ if (! $event->test($socket, 'r')) {
+ last try;
+ }
+ my $csocket = $socket->accept(10);
+ if (! defined($csocket)) {
+ $log->push->error;
+ last try;
+ }
+ if (! $csocket) {
+ $log->push->warn;
+ last try;
+ }
+ my $client = NDB::Net::Client->new(
+ socket => $csocket,
+ serversocket => $socket,
+ serverlock => $lock,
+ event => $event,
+ context => $node,
+ );
+ $client or $log->push->fatal;
+ }
+ NDB::Net::Client->processall;
+ redo loop;
+ }
+ if ($node->getruntype eq "auto") {
+ if ($node->getstate eq "run") {
+ $log->put("restart in 5 seconds...")->info;
+ sleep 5;
+ goto START;
+ }
+ $log->put("stopping, skip restart")->info;
+ }
+ $lock->close;
+ $node->getsocketfile->unlink;
+ while (wait() != -1) {}
+ $log->put("exit")->push->info;
+ exit(0);
+}
+
+# handlers can be overridden in subclass
+
+sub handleprepare { confess 'oops'; }
+
+sub handleoutput {
+ my $node = shift;
+ @_ == 2 or confess 0+@_;
+ my($opts, $data) = @_;
+ $data =~ s/\015//g;
+ $data = $node->{savedata} . $data;
+ while ((my $i = index($data, "\n")) >= 0) {
+ my $line = substr($data, 0, $i);
+ $data = substr($data, $i+1);
+ $log->put($line)->info;
+ if ($opts->{user} && $line !~ /^\s*$/) {
+ $log->put($line)->user;
+ }
+ }
+ $node->{savedata} = $data;
+ if (1 && length $node->{savedata}) { # XXX partial line
+ my $line = $node->{savedata};
+ $log->put($line)->info;
+ if ($opts->{user} && $line !~ /^\s*$/) {
+ $log->put($line)->user;
+ }
+ $node->{savedata} = "";
+ }
+}
+
+sub handleeof {
+}
+
+# command subs can be overridden by subclass
+
+sub waitforexit {
+ my $node = shift;
+ my $lock = $node->getlock;
+ my $lockpid = $node->getlockpid;
+ my $n1 = 0;
+ my $n2 = 10;
+ while (1) {
+ my $ret = $lock->test;
+ defined($ret) or $log->push, return undef;
+ if (! $ret) {
+ $log->put("exit done")->push($node)->user;
+ last;
+ }
+ if ($lockpid != $lock->getpid) {
+ $log->put("restarted: lock pid changed %s->%s",
+ $lockpid, $lock->getpid)->push($node);
+ return undef;
+ }
+ if (++$n1 >= $n2) {
+ $n2 *= 2;
+ $log->put("wait for exit")->push($node)->user;
+ }
+ select(undef, undef, undef, 0.1);
+ }
+ return 1;
+}
+
+sub cmd_stopnode_bg {
+ my($node, $cmd) = @_;
+ return $node->waitforexit;
+}
+
+sub cmd_killnode_fg {
+ my($node, $cmd) = @_;
+ my $pid = $node->getpid;
+ $log->put("kill -9 $pid")->push($node)->user;
+ kill(9, $pid);
+ $node->setstate('stop')
+ or $log->push($node), return undef;
+ return 1;
+}
+
+sub cmd_killnode_bg {
+ my($node, $cmd) = @_;
+ return $node->waitforexit;
+}
+
+sub cmd_statnode_bg {
+ my($node, $cmd) = @_;
+ return "up";
+}
+
+sub cmd_writenode_fg {
+ my($node, $cmd) = @_;
+ my $text = $cmd->getarg(2);
+ while(chomp($text)) {};
+ $log->put("write: $text")->push($node)->user;
+ $node->getiow->write("$text\n");
+ my $output = "";
+ if ((my $num = $cmd->getopt("wait")) > 0) {
+ my $lim = time + $num;
+ $node->getior->settimeout(1);
+ loop: {
+ my $data = $node->getior->read;
+ if (length($data) > 0) {
+ $node->handleoutput({user => 1}, $data);
+ $output .= $data;
+ }
+ redo loop if time < $lim;
+ }
+ $node->getior->settimeout(0);
+ }
+ return { output => $output };
+}
+
+# commands
+
+sub doremote {
+ my $node = shift;
+ my($cmdname, $opts, @args) = @_;
+ my $server = $node->getserver;
+ $log->put("$cmdname remote")->push($server)->push($node)->info;
+ my $argv = [
+ $cmdname, q(--local),
+ $opts, $node->getdb->getname, $node->getid, @args ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $server->request($cmd)
+ or $log->push, return undef;
+ return $ret;
+}
+
+sub dolocal {
+ my $node = shift;
+ my($cmdname, $opts, @args) = @_;
+ $log->put("$cmdname local")->push($node)->info;
+ if (! $node->getserver->islocal) {
+ $log->put("not local")->push($node->getserver)->push($node);
+ return undef;
+ }
+ if ($cmdname eq "startnode") {
+ return $node->startlocal($opts);
+ }
+ my $lock = $node->getlock;
+ anon: {
+ my $ret = $lock->test;
+ defined($ret) or $log->push, return undef;
+ if (! $ret) {
+ if ($cmdname eq "statnode") {
+ return "down";
+ }
+ $log->put("not running")->push($node)->user;
+ return $cmdname eq "writenode" ? undef : 1;
+ }
+ }
+ my $server;
+ anon: {
+ my $path = $node->getsocketfile->getpath;
+ if (! -e $path) {
+ $log->put("$path: no socket")->push($node);
+ return undef;
+ }
+ if ($^O ne 'MSWin32') {
+ $server = NDB::Net::ServerUNIX->new(id => 0, path => $path)
+ or $log->push, return undef;
+ } else {
+ $server = NDB::Net::ServerINET->new(id => 0, host => $node->getserver->getcanon, port => $node->getdb->getnodeport + $node->getid)
+ or $log->push, return undef;
+ }
+ }
+ my $argv = [
+ $cmdname,
+ $opts, $node->getdb->getname, $node->getid, @args ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $server->request($cmd)
+ or $log->push, return undef;
+ $log->put("$cmdname done")->push($node)->info;
+ return $ret;
+}
+
+sub start {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ $log->put("start")->push($node)->info;
+ my $do = $opts->{local} ? "dolocal" : "doremote";
+ return $node->$do("startnode", $opts);
+}
+
+sub stop {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ $log->put("stop")->push($node)->info;
+ my $do = $opts->{local} ? "dolocal" : "doremote";
+ return $node->$do("stopnode", $opts);
+}
+
+sub kill {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ $log->put("kill")->push($node)->info;
+ my $do = $opts->{local} ? "dolocal" : "doremote";
+ return $node->$do("killnode", $opts);
+}
+
+sub stat {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ $log->put("stat")->push($node)->info;
+ my $do = $opts->{local} ? "dolocal" : "doremote";
+ return $node->$do("statnode", $opts);
+}
+
+sub write {
+ my $node = shift;
+ @_ == 2 or confess 0+@_;
+ my($opts, $text) = @_;
+ $log->put("write: $text")->push($node)->info;
+ my $do = $opts->{local} ? "dolocal" : "doremote";
+ return $node->$do("writenode", $opts, $text);
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/NodeApi.pm b/ndb/tools/ndbnet/lib/NDB/Net/NodeApi.pm
new file mode 100644
index 00000000000..08f5f85577d
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/NodeApi.pm
@@ -0,0 +1,84 @@
+package NDB::Net::NodeApi;
+
+use strict;
+use Carp;
+use Symbol;
+
+require NDB::Net::Node;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Node);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Net::NodeApi->attributes();
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $node = $class->SUPER::new(%attr, type => 'api')
+ or $log->push, return undef;
+ return 1;
+}
+
+# run methods
+
+sub handleprepare {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ my $netenv = NDB::Net::Env->instance;
+ my $envdefs = $node->getenvdefs($opts);
+ defined($envdefs) or return undef;
+ my $nodedir = $node->getnodedir;
+ my $shellfile = $node->getshellfile;
+ my $run;
+ if ($node->hasrun) {
+ $run = $node->getrun;
+ }
+ if (defined($opts->{run})) {
+ $run = $opts->{run};
+ }
+ if (defined($run)) {
+ $log->put("run: $run")->push($node)->user;
+ }
+ if ($^O ne 'MSWin32') {
+ $shellfile->puttext(<<END) or $log->push, return undef;
+$envdefs
+cd @{[ $nodedir->getpath ]} || exit 1
+set -x
+exec \$DEBUGGER $run
+END
+ } else {
+ $shellfile->puttext(<<END) or $log->push, return undef;
+$envdefs
+cd @{[ $nodedir->getpath ]}
+call $run
+END
+ }
+ return 1;
+}
+
+sub cmd_stopnode_fg {
+ my($node, $cmd) = @_;
+ my $pid = $node->getpid;
+ unless ($pid > 1) {
+ $log->put("bad pid=$pid")->push($node);
+ return undef;
+ }
+ $log->put("kill -15 $pid")->push($node)->user;
+ kill(15, $pid);
+ $node->setstate('stop')
+ or log->push($node), return undef;
+ return 1;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/NodeDb.pm b/ndb/tools/ndbnet/lib/NDB/Net/NodeDb.pm
new file mode 100644
index 00000000000..88a35ba4f8d
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/NodeDb.pm
@@ -0,0 +1,116 @@
+package NDB::Net::NodeDb;
+
+use strict;
+use Carp;
+use Symbol;
+
+require NDB::Net::Node;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Node);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Net::NodeDb->attributes(
+ fsdir => sub { s/^\s+|\s+$//g; /^\S+$/ },
+);
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $node = $class->SUPER::new(%attr, type => 'db')
+ or $log->push, return undef;
+ $node->setfsdir($attr{fsdir})
+ or $log->push, return undef;
+ return 1;
+}
+
+# run methods
+
+sub handleprepare {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ my $netenv = NDB::Net::Env->instance;
+ my $envdefs = $node->getenvdefs($opts);
+ defined($envdefs) or return undef;
+ my $nodedir = $node->getnodedir;
+ my $shellfile = $node->getshellfile;
+ my $fsdir = NDB::Util::Dir->new(
+ path => sprintf("%s/%s/%s-%s.fs",
+ $node->getfsdir, $node->getdb->getname, $node->getid, $node->gettype));
+ $fsdir->mkdir or $log->push, return undef;
+ my $init_rm;
+ my $run;
+ if ($^O ne 'MSWin32') {
+ $init_rm = "# no -i";
+ if ($opts->{init_rm}) {
+ $init_rm = 'rm -f $NDB_FILESYSTEM/*/DBDIH/P0.sysfile';
+ }
+ $run = "\$NDB_TOP/bin/ndb";
+ } else {
+ $init_rm = "rem no -i";
+ if ($opts->{init_rm}) {
+ $init_rm =
+ 'del/f %NDB_FILESYSTEM%\D1\DBDIH\P0.sysfile' . "\n" .
+ 'del/f %NDB_FILESYSTEM%\D2\DBDIH\P0.sysfile';
+ }
+ $run = "ndb";
+ }
+ if ($node->getdb->cmpversion("1.0") <= 0) {
+ $run .= " -s";
+ }
+ if ($opts->{nostart}) {
+ $run .= " -n";
+ }
+ if ($node->hasrun) {
+ $run = $node->getrun;
+ }
+ if (defined($opts->{run})) {
+ $run = $opts->{run};
+ }
+ $log->put("run: $run")->push($node)->user;
+ if ($^O ne 'MSWin32') {
+ $shellfile->puttext(<<END) or $log->push, return undef;
+$envdefs
+NDB_FILESYSTEM=@{[ $fsdir->getpath ]}
+export NDB_FILESYSTEM
+# v1.0 compat
+UAS_FILESYSTEM=\$NDB_FILESYSTEM
+export UAS_FILESYSTEM
+mkdir -p \$NDB_FILESYSTEM
+$init_rm
+cd @{[ $nodedir->getpath ]} || exit 1
+exec \$debugger $run
+END
+ } else {
+ $shellfile->puttext(<<END) or $log->push, return undef;
+$envdefs
+set NDB_FILESYSTEM=@{[ $fsdir->getpath ]}
+rem v1.0 compat
+set UAS_FILESYSTEM=%NDB_FILESYSTEM%
+mkdir %NDB_FILESYSTEM%
+$init_rm
+cd @{[ $nodedir->getpath ]}
+call $run
+END
+ }
+ return 1;
+}
+
+sub cmd_stopnode_fg {
+ my($node, $cmd) = @_;
+ $node->setstate('stop')
+ or log->push($node), return undef;
+ return 1;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/NodeMgmt.pm b/ndb/tools/ndbnet/lib/NDB/Net/NodeMgmt.pm
new file mode 100644
index 00000000000..1056e3df623
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/NodeMgmt.pm
@@ -0,0 +1,318 @@
+package NDB::Net::NodeMgmt;
+
+use strict;
+use Carp;
+use Symbol;
+
+require NDB::Net::Node;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Node);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Net::NodeMgmt->attributes(
+ port => sub { s/^\s+|\s+$//g; /^\d+$/ },
+);
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $node = $class->SUPER::new(%attr, type => 'mgmt')
+ or $log->push, return undef;
+ $node->setport($attr{port})
+ or $log->push, return undef;
+ return 1;
+}
+
+# socket parser methods
+
+sub socketcommand {
+ my $node = shift;
+ my $socket;
+ $socket = NDB::Util::SocketINET->new or
+ $log->push($node), return undef;
+ $socket->settimeout(10);
+ $socket->connect($node->getserver->getcanon, $node->getport) or
+ $log->push($node), return undef;
+ $socket->write("GET STATUS\r\nBYE\r\n") or
+ $log->push($node), return undef;
+ my $out = "";
+ my $data;
+ while ($data = $socket->read) {
+ $out .= $data;
+ }
+ $socket->close;
+ $out =~ s/\015//g;
+ return $out;
+}
+
+sub get_status {
+ my $node = shift;
+ my $out = $node->socketcommand or
+ $log->push, return undef;
+ my @out = split(/\n/, $out);
+ $out[0] =~ /^get\s+status\s+(\d+)/i or
+ $log->put("bad line 0: $out[0]"), return undef;
+ my $cnt = $1;
+ my $ret = {};
+ for (my $i = 1; $i <= $cnt; $i++) {
+ $out[$i] =~ /^$i\s+(.*)/ or
+ $log->put("bad line $i: $out[$i]"), return undef;
+ my $text = $1;
+ $text =~ s/^\s+|\s+$//g;
+ if ($text =~ /^ndb\s+(no_contact)\s+(\d+)$/i) {
+ $text = lc "$1";
+ } elsif ($text =~ /^ndb\s+(starting)\s+(\d+)$/i) {
+ $text = lc "$1/$2";
+ } elsif ($text =~ /^ndb\s+(started)\s+(\d+)$/i) {
+ $text = lc "$1";
+ } elsif ($text =~ /^ndb\s+(shutting_down)\s+(\d+)$/i) {
+ $text = lc "$1";
+ } elsif ($text =~ /^ndb\s+(restarting)\s+(\d+)$/i) {
+ $text = lc "$1";
+ } elsif ($text =~ /^ndb\s+(unknown)\s+(\d+)$/i) {
+ $text = lc "$1";
+ }
+ $ret->{node}{$i} = $text;
+ }
+ return $ret;
+}
+
+# run methods
+
+sub getautoinifile {
+ my $node = shift;
+ @_ == 0 or confess 0+@_;
+ my $name = "config.txt";
+ my $file = $node->getnodedir->getfile($name);
+ return $file;
+}
+
+sub writeautoinifile {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ my $db = $node->getdb;
+ my $nodelist = $db->getnodelist('all');
+ my $computers = {};
+ for my $n (@$nodelist) {
+ $computers->{$n->getserver->getid} ||= {
+ id => $n->getserver->getid,
+ hostname => $n->getserver->getcanon,
+ };
+ }
+ my $section = ""; # e.g. PROCESSES
+ my $auto;
+ my $edit = sub {
+ chomp;
+ s/^\s+|\s+$//g;
+ if (/^(\w+)$/) {
+ $section = uc($1);
+ }
+ elsif (/^\@loop$/i) {
+ $_ = "#$_";
+ if ($auto) {
+ $log->put("nested \@loop");
+ return undef;
+ }
+ $auto = {};
+ }
+ elsif (/^\@base\s+(\S+)\s*$/) {
+ my $arg = $1;
+ $_ = "#$_";
+ if (! $auto) {
+ $log->put("unexpected \@base");
+ return undef;
+ }
+ if ($arg !~ /^\d+$/) {
+ $log->put("non-numerical \@base");
+ return undef;
+ }
+ $auto->{base} = $arg;
+ }
+ elsif (/^\@end$/i) {
+ $_ = "#$_";
+ if (! $auto) {
+ $log->put("unmatched \@end");
+ return undef;
+ }
+ if ($section eq 'COMPUTERS') {
+ for my $id (sort { $a <=> $b } keys %$computers) {
+ my $computer = $computers->{$id};
+ $_ .= "\n";
+ $_ .= "\nId: " . $computer->{id};
+ $_ .= "\nHostName: " . $computer->{hostname};
+ if ($auto->{list}) {
+ $_ .= "\n#defaults";
+ for my $s (@{$auto->{list}}) {
+ $_ .= "\n$s";
+ }
+ }
+ }
+ }
+ elsif ($section eq 'PROCESSES') {
+ for my $n (@$nodelist) {
+ if ($auto->{type} && $n->gettype ne lc($auto->{type})) {
+ next;
+ }
+ $_ .= "\n";
+ $_ .= "\nType: " . uc($n->gettype);
+ $_ .= "\nId: " . $n->getid;
+ $_ .= "\nExecuteOnComputer: " . $n->getserver->getid;
+ if ($auto->{list}) {
+ $_ .= "\n#defaults";
+ for my $s (@{$auto->{list}}) {
+ $_ .= "\n$s";
+ }
+ }
+ }
+ }
+ elsif ($section eq 'CONNECTIONS') {
+ if (! $auto->{type}) {
+ $log->put("cannot generate CONNECTIONS without type");
+ return undef;
+ }
+ if (! defined($auto->{base})) {
+ $log->put("need \@base for CONNECTIONS");
+ return undef;
+ }
+ my $key = $auto->{base};
+ for (my $i1 = 0; $i1 <= $#$nodelist; $i1++) {
+ for (my $i2 = $i1+1; $i2 <= $#$nodelist; $i2++) {
+ my $n1 = $nodelist->[$i1];
+ my $n2 = $nodelist->[$i2];
+ if ($n1->gettype ne 'db' && $n2->gettype ne 'db') {
+ next;
+ }
+ $_ .= "\n";
+ $_ .= "\nType: $auto->{type}";
+ $_ .= "\nProcessId1: " . $n1->getid;
+ $_ .= "\nProcessId2: " . $n2->getid;
+ $key++;
+ if ($auto->{type} eq 'TCP') {
+ $_ .= "\nPortNumber: $key";
+ if (my $list = $opts->{proxy}) {
+ my $id1 = $n1->getid;
+ my $id2 = $n2->getid;
+ if ($list =~ /\b$id1\b.*-.*\b$id2\b/) {
+ $key++;
+ $_ .= "\nProxy: $key";
+ } elsif ($list =~ /\b$id2\b.*-.*\b$id1\b/) {
+ $key++;
+ $_ .= "\nProxy: $key";
+ }
+ }
+ }
+ elsif ($auto->{type} eq 'SHM') {
+ $_ .= "\nShmKey: $key";
+ }
+ else {
+ $log->put("cannot handle CONNECTIONS type $auto->{type}");
+ return undef;
+ }
+ if ($auto->{list}) {
+ $_ .= "\n#defaults";
+ for my $s (@{$auto->{list}}) {
+ $_ .= "\n$s";
+ }
+ }
+ }
+ }
+ }
+ else {
+ $log->put("found \@end in unknown section '$section'");
+ return undef;
+ }
+ undef $auto;
+ }
+ elsif (/^$/) {
+ }
+ elsif ($auto) {
+ if (/^Type:\s*(\w+)$/i) {
+ $auto->{type} = uc($1);
+ }
+ else {
+ $auto->{list} ||= [];
+ push(@{$auto->{list}}, $_);
+ }
+ $_ = "";
+ return 1; # no output
+ }
+ $_ .= "\n";
+ return 1;
+ };
+ $node->getautoinifile->mkdir
+ or $log->push, return undef;
+ $node->getinifile->copyedit($node->getautoinifile, $edit)
+ or $log->push, return undef;
+ return 1;
+}
+
+sub handleprepare {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($opts) = @_;
+ my $envdefs = $node->getenvdefs($opts);
+ defined($envdefs) or return undef;
+ my $nodedir = $node->getnodedir;
+ my $shellfile = $node->getshellfile;
+ my $port = $node->getport;
+ my $lpath = $node->getlocalcfg->getbasename;
+ $node->writeautoinifile($opts)
+ or $log->push, return undef;
+ my $ipath = $node->getautoinifile->getbasename;
+ $node->getbincfg->mkdir or $log->push, return undef;
+ my $cpath = $node->getbincfg->getbasename;
+ my $run;
+ if ($^O ne 'MSWin32') {
+ $run = "\$NDB_TOP/bin/mgmtsrvr";
+ } else {
+ $run = "mgmtsrvr";
+ }
+ my $statport = $port + 1;
+ $run .= " -l $lpath -c $ipath";
+ if ($node->hasrun) {
+ $run = $node->getrun;
+ }
+ if (defined($opts->{run})) {
+ $run = $opts->{run};
+ }
+ $log->put("run: $run")->push($node)->user;
+ if ($^O ne 'MSWin32') {
+ $shellfile->puttext(<<END) or $log->push, return undef;
+$envdefs
+cd @{[ $nodedir->getpath ]} || exit 1
+set -x
+exec \$DEBUGGER $run
+END
+ } else {
+ $shellfile->puttext(<<END) or $log->push, return undef;
+$envdefs
+cd @{[ $nodedir->getpath ]}
+call $run
+END
+ }
+ return 1;
+}
+
+sub cmd_stopnode_fg {
+ my $node = shift;
+ @_ == 1 or confess 0+@_;
+ my($cmd) = @_;
+ $log->put("write: quit")->push($node)->user;
+ $node->getiow->write("quit\n");
+ $node->setstate('stop')
+ or log->push($node), return undef;
+ return 1;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/Server.pm b/ndb/tools/ndbnet/lib/NDB/Net/Server.pm
new file mode 100644
index 00000000000..5d2118f0ffe
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/Server.pm
@@ -0,0 +1,149 @@
+package NDB::Net::Server;
+
+use strict;
+use Carp;
+use Socket;
+
+require NDB::Net::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+my %servercache = ();
+
+NDB::Net::Server->attributes(
+ id => sub { s/^\s+|\s+$//g; m/^\S+$/ && ! m!/! },
+ domain => sub { $_ == PF_UNIX || $_ == PF_INET },
+);
+
+sub desc {
+ my $server = shift;
+ my $id = $server->getid;
+ return "server $id";
+}
+
+sub add {
+ my $server = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ if ($servercache{$server->getid}) {
+ $log->put("duplicate server")->push($server);
+ return undef;
+ }
+ $servercache{$server->getid} = $server;
+ return 1;
+}
+
+sub get {
+ my $class = shift;
+ @_ == 1 or confess 0+@_;
+ my($id) = @_;
+ $id =~ s/^\s+|\s+$//g;
+ my $server = $servercache{$id};
+ if (! $server) {
+ $log->put("$id: undefined server");
+ return undef;
+ }
+ $log->put("found")->push($server)->debug;
+ return $server;
+}
+
+sub delete {
+ my $server = shift;
+ delete $servercache{$server->getid};
+}
+
+sub deleteall {
+ my $class = shift;
+ for my $id (sort keys %servercache) {
+ my $server = $servercache{$id};
+ $server->delete;
+ }
+}
+
+# local server is this server process
+
+my $localserver;
+
+sub setlocal {
+ my $server = shift;
+ @_ == 0 or confess 0+@_;
+ $localserver = $server;
+}
+
+sub islocal {
+ my $server = shift;
+ @_ == 0 or confess 0+@_;
+ return $localserver eq $server;
+}
+
+# client side
+
+sub testconnect {
+ my $server = shift;
+ @_ == 0 or confess 0+@_;
+ my $socket = $server->connect or
+ $log->push($server), return undef;
+ $socket->close;
+ return 1;
+}
+
+sub request {
+ my $server = shift;
+ @_ == 1 or confess 0+@_;
+ my($cmd) = @_;
+ unless (ref($cmd) && $cmd->isa('NDB::Net::Command')) {
+ confess 'oops';
+ }
+ my $socket = $server->connect
+ or $log->push($server), return undef;
+ anon: {
+ my $line = $cmd->getline;
+ my $n = $socket->write("$line\n");
+ defined($n) && $n == length("$line\n")
+ or $log->push($server), return undef;
+ shutdown($socket->{fh}, 1);
+ }
+ my $value;
+ try: {
+ my $last;
+ loop: {
+ my $line = $socket->readline;
+ defined($line)
+ or $log->push($server), last try;
+ if ($socket->getreadend) {
+ last loop;
+ }
+ while (chomp($line)) {}
+ $log->put($line)->user
+ unless $log->hasvalue($line);
+ $last = $line;
+ redo loop;
+ }
+ if (! $log->hasvalue($last)) {
+ $log->put("missing return value in \"$last\"")->push($server);
+ last try;
+ }
+ $value = $log->getvalue($last);
+ defined($value)
+ or $log->push, last try;
+ $value = $value->[0];
+ if (! defined($value)) {
+ $log->put("failed")->push($cmd);
+ last try;
+ }
+ }
+ $socket->close;
+ return $value;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/ServerINET.pm b/ndb/tools/ndbnet/lib/NDB/Net/ServerINET.pm
new file mode 100644
index 00000000000..a065c186855
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/ServerINET.pm
@@ -0,0 +1,116 @@
+package NDB::Net::ServerINET;
+
+use strict;
+use Carp;
+use Socket;
+
+require NDB::Net::Server;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Server);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Net::ServerINET->attributes(
+ host => sub { s/^\s+|\s+$//g; /^\S+$/ },
+ port => sub { s/^\s+|\s+$//g; /^\d+$/ },
+ canon => sub { s/^\s+|\s+$//g; /^\S+$/ },
+ aliases => sub { ref($_) eq 'ARRAY' },
+);
+
+
+sub desc {
+ my $server = shift;
+ my $id = $server->getid;
+ my $host = $server->gethost;
+ return "server $id at $host";
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $server = $class->SUPER::new(%attr);
+ $server->setid($attr{id})
+ or $log->push, return undef;
+ $server->setdomain(PF_INET)
+ or $log->push, return undef;
+ $server->sethost($attr{host})
+ or $log->push, return undef;
+ $server->setport($attr{port})
+ or $log->push, return undef;
+ my($canon, $aliases) = gethostbyname($server->gethost);
+ if (! defined($canon)) {
+ $log->put("%s: unknown host", $server->gethost);
+ return undef;
+ }
+ $server->setcanon($canon)
+ or $log->push, return undef;
+ $server->setaliases([ split(' ', $aliases) ])
+ or $log->push, return undef;
+ $server->add or
+ $log->push, return undef;
+ $log->put("added")->push($server)->debug;
+ return $server;
+}
+
+# find matching servers
+
+sub match {
+ my $class = shift;
+ @_ == 3 or confess 0+@_;
+ my($host, $port, $servers) = @_;
+ if (! defined($port) && $host =~ /:/) {
+ ($host, $port) = split(/:/, $host, 2);
+ }
+ $host =~ s/^\s+|\s+$//g;
+ my($canon) = gethostbyname($host);
+ unless (defined($canon)) {
+ $log->put("$host: unknown host");
+ return undef;
+ }
+ my $hostport = $host;
+ if (defined($port)) {
+ $port =~ s/^\s+|\s+$//g;
+ $port =~ /\d+$/
+ or $log->put("$port: non-numeric port"), return undef;
+ $hostport .= ":$port";
+ }
+ my @server = ();
+ for my $s (@$servers) {
+ ($s->getdomain == PF_INET) || next;
+ ($s->getcanon eq $canon) || next;
+ ($port && $s->getport != $port) && next;
+ push(@server, $s);
+ }
+ if (! @server) {
+ $log->put("$hostport: no server found");
+ }
+ if (@server > 1) {
+ $log->put("$hostport: multiple servers at ports ",
+ join(' ', map($_->getport, @server)));
+ }
+ return \@server;
+}
+
+# client side
+
+sub connect {
+ my $server = shift;
+ @_ == 0 or confess 0+@_;
+ my $socket;
+ $socket = NDB::Util::SocketINET->new or
+ $log->push, return undef;
+ $socket->connect($server->gethost, $server->getport) or
+ $log->push, return undef;
+ return $socket;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Net/ServerUNIX.pm b/ndb/tools/ndbnet/lib/NDB/Net/ServerUNIX.pm
new file mode 100644
index 00000000000..b3fa245d5ee
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Net/ServerUNIX.pm
@@ -0,0 +1,54 @@
+package NDB::Net::ServerUNIX;
+
+use strict;
+use Carp;
+use Socket;
+
+require NDB::Net::Server;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Net::Server);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Net::ServerUNIX->attributes(
+ path => sub { s/^\s+|\s+$//g; /^\S+$/ },
+);
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $server = $class->SUPER::new(%attr);
+ $server->setid($attr{id})
+ or $log->push, return undef;
+ $server->setdomain(PF_UNIX)
+ or $log->push, return undef;
+ $server->setpath($attr{path})
+ or $log->push, return undef;
+ $server->add or
+ $log->push, return undef;
+ return $server;
+}
+
+# client side
+
+sub connect {
+ my $server = shift;
+ @_ == 0 or confess 0+@_;
+ my $socket;
+ $socket = NDB::Util::SocketUNIX->new or
+ $log->push, return undef;
+ $socket->connect($server->getpath) or
+ $log->push, return undef;
+ return $socket;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Run.pm b/ndb/tools/ndbnet/lib/NDB/Run.pm
new file mode 100644
index 00000000000..a8cabde544c
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Run.pm
@@ -0,0 +1,40 @@
+package NDB::Run;
+
+use strict;
+use Carp;
+require Exporter;
+
+use NDB::Net;
+
+use vars qw(@ISA @EXPORT @EXPORT_OK);
+@ISA = qw(Exporter);
+
+use vars qw(@modules);
+@modules = qw(
+ NDB::Run::Base
+ NDB::Run::Database
+ NDB::Run::Env
+ NDB::Run::Node
+);
+
+return 1 if $main::onlymodules;
+
+for my $module (@modules) {
+ eval "require $module";
+ $@ and confess "$module $@";
+}
+
+for my $module (@modules) {
+ eval "$module->initmodule";
+ $@ and confess "$module $@";
+}
+
+# methods
+
+sub getenv {
+ my $class = shift;
+ return NDB::Run::Env->new(@_);
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Run/Base.pm b/ndb/tools/ndbnet/lib/NDB/Run/Base.pm
new file mode 100644
index 00000000000..4769f2c4441
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Run/Base.pm
@@ -0,0 +1,12 @@
+package NDB::Run::Base;
+
+use strict;
+use Carp;
+
+require NDB::Util::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::Base);
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Run/Database.pm b/ndb/tools/ndbnet/lib/NDB/Run/Database.pm
new file mode 100644
index 00000000000..9a12ddb20b3
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Run/Database.pm
@@ -0,0 +1,89 @@
+package NDB::Run::Database;
+
+use strict;
+use Carp;
+
+require NDB::Run::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Run::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Run::Database->attributes(
+ name => sub { s/^\s+|\s+$//g; /^\S+$/ && ! m!/! },
+ env => sub { ref && $_->isa('NDB::Run::Env') },
+);
+
+sub desc {
+ my $db = shift;
+ return $db->getname;
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $db = $class->SUPER::new(%attr);
+ $db->setname($attr{name})
+ or $log->push, return undef;
+ $db->setenv($attr{env})
+ or $log->push, return undef;
+ return $db;
+}
+
+sub getnode {
+ my $db = shift;
+ @_ == 1 or croak q(usage: $node = $db->getnode($id));
+ my($id) = @_;
+ my $node = NDB::Run::Node->new(db => $db, id => $id)
+ or $log->push, return undef;
+ return $node;
+}
+
+# commands
+
+sub start {
+ my $db = shift;
+ my $opts = shift;
+ my $argv = [ 'start', $db->getname, $opts ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $db->getenv->docmd($cmd);
+ defined($ret)
+ or $log->push, return undef;
+ return $ret;
+}
+
+sub stop {
+ my $db = shift;
+ my $opts = shift;
+ my $argv = [ 'stop', $db->getname, $opts ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $db->getenv->docmd($cmd);
+ defined($ret)
+ or $log->push, return undef;
+ return $ret;
+}
+
+sub kill {
+ my $db = shift;
+ my $opts = shift;
+ my $argv = [ 'kill', $db->getname, $opts ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $db->getenv->docmd($cmd);
+ defined($ret)
+ or $log->push, return undef;
+ return $ret;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Run/Env.pm b/ndb/tools/ndbnet/lib/NDB/Run/Env.pm
new file mode 100644
index 00000000000..e851a82636b
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Run/Env.pm
@@ -0,0 +1,84 @@
+package NDB::Run::Env;
+
+use strict;
+use Carp;
+
+require NDB::Run::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Run::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Run::Env->attributes(
+ server => sub { ref && $_->isa('NDB::Net::Server') },
+);
+
+sub desc {
+ "env";
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $env = $class->SUPER::new(%attr);
+ return $env;
+}
+
+sub getdb {
+ my $env = shift;
+ @_ == 1 or croak q(usage: $db = $env->getdb($name));
+ my($name) = @_;
+ my $db = NDB::Run::Database->new(env => $env, name => $name)
+ or $log->push, return undef;
+ return $db;
+}
+
+# commands
+
+sub init {
+ my $env = shift;
+ my $netenv = NDB::Net::Env->instance;
+ my $netcfg = NDB::Net::Config->new(file => $netenv->getnetcfg)
+ or $log->push, return undef;
+ $netcfg->load
+ or $log->push, return undef;
+ my $servers = $netcfg->getservers
+ or $log->push, return undef;
+ my $server;
+ for my $s (@$servers) {
+ if (! $s->testconnect) {
+ $log->push->warn;
+ next;
+ }
+ $server = $s;
+ last;
+ }
+ if (! $server) {
+ $log->put("no available server")->push($netcfg);
+ return undef;
+ }
+ $env->setserver($server)
+ or $log->push, return undef;
+ $log->put("selected")->push($server)->info;
+ return 1;
+}
+
+sub docmd {
+ my $env = shift;
+ my $cmd = shift;
+ my $ret = $env->getserver->request($cmd);
+ defined($ret)
+ or $log->push, return undef;
+ return $ret;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Run/Node.pm b/ndb/tools/ndbnet/lib/NDB/Run/Node.pm
new file mode 100644
index 00000000000..e657021b229
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Run/Node.pm
@@ -0,0 +1,114 @@
+package NDB::Run::Node;
+
+use strict;
+use Carp;
+
+require NDB::Run::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Run::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Run::Node->attributes(
+ env => sub { ref && $_->isa('NDB::Run::Env') },
+ db => sub { ref && $_->isa('NDB::Run::Database') },
+ dbname => sub { s/^\s+|\s+$//g; /^\S+$/ && ! m!/! },
+ id => sub { s/^\s+|\s+$//g; s/^0+(\d+)$/$1/; /^\d+$/ && $_ > 0 },
+ type => sub { s/^\s+|\s+$//g; /^(mgmt|db|api)$/ },
+);
+
+sub desc {
+ my $node = shift;
+ my $dbname = $node->getdb->getname;
+ my $id = $node->getid;
+ my $type = "?"; # $node->gettype;
+ return "$dbname.$id-$type";
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $node = $class->SUPER::new(%attr);
+ $node->setdb($attr{db})
+ or $log->push, return undef;
+ $node->setenv($node->getdb->getenv)
+ or $log->push, return undef;
+ $node->setdbname($node->getdb->getname)
+ or $log->push, return undef;
+ $node->setid($attr{id})
+ or $log->push, return undef;
+# $node->settype($attr{type})
+# or $log->push, return undef;
+ return $node;
+}
+
+# commands
+
+sub start {
+ my $node = shift;
+ my $opts = shift;
+ my $argv = [ 'startnode', $node->getdb->getname, $node->getid, $opts ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $node->getenv->docmd($cmd)
+ or $log->push, return undef;
+ return $ret;
+}
+
+sub stop {
+ my $node = shift;
+ my $opts = shift;
+ my $argv = [ 'stopnode', $node->getdb->getname, $node->getid, $opts ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $node->getenv->docmd($cmd)
+ or $log->push, return undef;
+ return $ret;
+}
+
+sub kill {
+ my $node = shift;
+ my $opts = shift;
+ my $argv = [ 'killnode', $node->getdb->getname, $node->getid, $opts ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $node->getenv->docmd($cmd)
+ or $log->push, return undef;
+ return $ret;
+}
+
+sub stat {
+ my $node = shift;
+ my $opts = shift;
+ my $argv = [ 'statnode', $node->getdb->getname, $node->getid, $opts ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $node->getenv->docmd($cmd)
+ or $log->push, return undef;
+ return $ret;
+}
+
+sub write {
+ my $node = shift;
+ my $text = shift;
+ my $opts = shift;
+ my $argv = [ 'writenode', $node->getdb->getname, $node->getid, $text, $opts ];
+ my $cmd = NDB::Net::Command->new(argv => $argv)
+ or $log->push, return undef;
+ my $ret = $node->getenv->docmd($cmd)
+ or $log->push, return undef;
+ ref($ret) eq 'HASH' && defined($ret->{output})
+ or confess 'oops';
+ return $ret;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util.pm b/ndb/tools/ndbnet/lib/NDB/Util.pm
new file mode 100644
index 00000000000..d5db35cbf13
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util.pm
@@ -0,0 +1,37 @@
+package NDB::Util;
+
+use strict;
+use Carp;
+require Exporter;
+
+use vars qw(@ISA @EXPORT @EXPORT_OK);
+@ISA = qw(Exporter);
+
+use vars qw(@modules);
+@modules = qw(
+ NDB::Util::Base
+ NDB::Util::Dir
+ NDB::Util::Event
+ NDB::Util::File
+ NDB::Util::IO
+ NDB::Util::Lock
+ NDB::Util::Log
+ NDB::Util::Socket
+ NDB::Util::SocketINET
+ NDB::Util::SocketUNIX
+);
+
+return 1 if $main::onlymodules;
+
+for my $module (@modules) {
+ eval "require $module";
+ $@ and confess "$module $@";
+}
+
+for my $module (@modules) {
+ eval "$module->initmodule";
+ $@ and confess "$module $@";
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/Base.pm b/ndb/tools/ndbnet/lib/NDB/Util/Base.pm
new file mode 100644
index 00000000000..20df78a3b9b
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/Base.pm
@@ -0,0 +1,113 @@
+package NDB::Util::Base;
+
+use strict;
+use Carp;
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+sub new {
+ my $class = shift;
+ my $this = bless {}, $class;
+ return $this;
+}
+
+sub getlog {
+ my $this = shift;
+ return NDB::Util::Log->instance;
+}
+
+# clone an object
+# extra attributes override or delete (if value is undef)
+sub clone {
+ my $this = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $that = bless {}, ref($this);
+ for my $attr (sort keys %$this) {
+ if (! exists($attr{$attr})) {
+ my $get = "get$attr";
+ $attr{$attr} = $this->$get();
+ }
+ }
+ for my $attr (sort keys %attr) {
+ if (defined($attr{$attr})) {
+ my $set = "set$attr";
+ $that->$set($attr{$attr});
+ }
+ }
+ return $that;
+}
+
+# methods for member variables:
+# - set returns 1 on success and undef on undefined or invalid value
+# - get aborts unless value exists or a default (maybe undef) is given
+# - has tests existence of value
+# - del deletes the value and returns it (maybe undef)
+
+sub attributes {
+ @_ % 2 == 1 or confess 0+@_;
+ my $class = shift;
+ my @attr = @_;
+ while (@attr) {
+ my $attr = shift @attr;
+ my $filter = shift @attr;
+ $attr =~ /^\w+$/ or confess $attr;
+ ref($filter) eq 'CODE' or confess $attr;
+ my $set = sub {
+ @_ == 2 or confess "set$attr: arg count: @_";
+ my $this = shift;
+ my $value = shift;
+ if (! defined($value)) {
+ $log->put("set$attr: undefined value")->push($this);
+ return undef;
+ }
+ local $_ = $value;
+ if (! &$filter($this)) {
+ $log->put("set$attr: invalid value: $value")->push($this);
+ return undef;
+ }
+ $value = $_;
+ if (! defined($value)) {
+ confess "set$attr: changed to undef";
+ }
+ $this->{$attr} = $value;
+ return 1;
+ };
+ my $get = sub {
+ @_ == 1 || @_ == 2 or confess "get$attr: arg count: @_";
+ my $this = shift;
+ my $value = $this->{$attr};
+ if (! defined($value)) {
+ @_ == 0 and confess "get$attr: no value";
+ $value = shift;
+ }
+ return $value;
+ };
+ my $has = sub {
+ @_ == 1 or confess "has$attr: arg count: @_";
+ my $this = shift;
+ my $value = $this->{$attr};
+ return defined($value);
+ };
+ my $del = sub {
+ @_ == 1 or confess "del$attr: arg count: @_";
+ my $this = shift;
+ my $value = delete $this->{$attr};
+ return $value;
+ };
+ no strict 'refs';
+ *{"${class}::set$attr"} = $set;
+ *{"${class}::get$attr"} = $get;
+ *{"${class}::has$attr"} = $has;
+ *{"${class}::del$attr"} = $del;
+ }
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/Dir.pm b/ndb/tools/ndbnet/lib/NDB/Util/Dir.pm
new file mode 100644
index 00000000000..90609b971c7
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/Dir.pm
@@ -0,0 +1,170 @@
+package NDB::Util::Dir;
+
+use strict;
+use Carp;
+use Symbol;
+use Errno;
+use File::Basename;
+
+require NDB::Util::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Util::Dir->attributes(
+ path => sub { length > 0 },
+);
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $dir = $class->SUPER::new(%attr);
+ $dir->setpath($attr{path})
+ or $log->push, return undef;
+ return $dir;
+}
+
+sub desc {
+ my $dir = shift;
+ return $dir->getpath;
+}
+
+sub getparent {
+ my $dir = shift;
+ @_ == 0 or confess 0+@_;
+ my $ppath = dirname($dir->getpath);
+ my $pdir = NDB::Util::Dir->new(path => $ppath);
+ return $pdir;
+}
+
+sub getdir {
+ my $dir = shift;
+ @_ == 1 or confess 0+@_;
+ my($name) = @_;
+ my $dirpath = $dir->getpath;
+ my $path = $dirpath eq '.' ? $name : File::Spec->catfile($dirpath, $name);
+ my $entry = NDB::Util::Dir->new(path => $path);
+ return $entry;
+}
+
+sub getfile {
+ my $dir = shift;
+ @_ == 1 or confess 0+@_;
+ my($name) = @_;
+ my $dirpath = $dir->getpath;
+ my $path = $dirpath eq '.' ? $name : File::Spec->catfile($dirpath, $name);
+ my $entry = NDB::Util::File->new(path => $path);
+ return $entry;
+}
+
+# list
+
+sub listdirs {
+ my $dir = shift;
+ @_ == 0 or confess 0+@_;
+ my @list = ();
+ my $dirpath = $dir->getpath;
+ my $dh = gensym();
+ if (! opendir($dh, $dirpath)) {
+ $log->put("opendir failed: $!")->push($dir);
+ return undef;
+ }
+ while (defined(my $name = readdir($dh))) {
+ if ($name eq '.' || $name eq '..') {
+ next;
+ }
+ my $path = $dirpath eq '.' ? $name : "$dirpath/$name";
+ if (! -l $path && -d $path) {
+ my $dir2 = NDB::Util::Dir->new(path => $path)
+ or $log->push, return undef;
+ push(@list, $dir2);
+ }
+ }
+ close($dh);
+ return \@list;
+}
+
+sub listfiles {
+ my $dir = shift;
+ @_ == 0 or confess 0+@_;
+ my @list = ();
+ my $dirpath = $dir->getpath;
+ my $dh = gensym();
+ if (! opendir($dh, $dirpath)) {
+ $log->put("opendir failed: $!")->push($dir);
+ return undef;
+ }
+ while (defined(my $name = readdir($dh))) {
+ if ($name eq '.' || $name eq '..') {
+ next;
+ }
+ my $path = $dirpath eq '.' ? $name : "$dirpath/$name";
+ if (! -d $path && -e $path) {
+ my $file2 = NDB::Util::File->new(path => $path)
+ or $log->push, return undef;
+ push(@list, $file2);
+ }
+ }
+ close($dh);
+ return \@list;
+}
+
+# create / remove
+
+sub mkdir {
+ my $dir = shift;
+ @_ == 0 or confess 0+@_;
+ if (! -d $dir->getpath) {
+ my $pdir = $dir->getparent;
+ if (length($pdir->getpath) >= length($dir->getpath)) {
+ $log->put("mkdir looping")->push($dir);
+ return undef;
+ }
+ $pdir->mkdir or return undef;
+ if (! mkdir($dir->getpath, 0777)) {
+ my $errstr = "$!";
+ if (-d $dir->getpath) {
+ return 1;
+ }
+ $log->put("mkdir failed: $errstr")->push($dir);
+ return undef;
+ }
+ }
+ return 1;
+}
+
+sub rmdir {
+ my $dir = shift;
+ my $keep = shift; # keep top level
+ $log->put("remove")->push($dir)->info;
+ my $list;
+ $list = $dir->listdirs or $log->push, return undef;
+ for my $d (@$list) {
+ $d->rmdir or $log->push, return undef;
+ }
+ $list = $dir->listfiles or $log->push, return undef;
+ for my $f (@$list) {
+ $f->unlink or $log->push, return undef;
+ }
+ if (! $keep && ! rmdir($dir->getpath)) {
+ my $errstr = "$!";
+ if (! -e $dir->getpath) {
+ return 1;
+ }
+ $log->put("rmdir failed: $errstr")->push($dir);
+ return undef;
+ }
+ return 1;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/Event.pm b/ndb/tools/ndbnet/lib/NDB/Util/Event.pm
new file mode 100644
index 00000000000..a3ad32cd7fb
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/Event.pm
@@ -0,0 +1,103 @@
+package NDB::Util::Event;
+
+use strict;
+use Carp;
+use Errno;
+
+require NDB::Util::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Util::Event->attributes();
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $event = $class->SUPER::new(%attr);
+ return $event;
+}
+
+# set and test bits
+
+sub check {
+ my $event = shift;
+ my($file, $type) = @_;
+ my $fileno;
+ if (ref($file) eq 'GLOB') {
+ $fileno = fileno($file);
+ }
+ elsif (ref($file)) {
+ $file->can("getfh") or confess 'oops';
+ $fileno = fileno($file->getfh);
+ }
+ else {
+ $fileno = $file;
+ }
+ defined($fileno) or confess 'oops';
+ $fileno =~ s/^\s+|\s+$//g;
+ $fileno =~ /^\d+$/ or confess 'oops';
+ $type =~ /^[rwe]$/ or confess 'oops';
+ return ($fileno, $type);
+}
+
+sub set {
+ my $event = shift;
+ @_ == 2 or confess 0+@_;
+ my($fileno, $type) = $event->check(@_);
+ vec($event->{"i_$type"}, $fileno, 1) = 1;
+}
+
+sub clear {
+ my $event = shift;
+ @_ == 2 or confess 0+@_;
+ my($fileno, $type) = $event->check(@_);
+ vec($event->{"i_$type"}, $fileno, 1) = 0;
+}
+
+sub test {
+ my $event = shift;
+ @_ == 2 or confess 0+@_;
+ my($fileno, $type) = $event->check(@_);
+ return vec($event->{"o_$type"}, $fileno, 1);
+}
+
+# poll
+
+sub poll {
+ my $event = shift;
+ @_ <= 1 or confess 'oops';
+ my $timeout = shift;
+ if (defined($timeout)) {
+ $timeout =~ /^\d+$/ or confess 'oops';
+ }
+ $event->{o_r} = $event->{i_r};
+ $event->{o_w} = $event->{i_w};
+ $event->{o_e} = $event->{i_e};
+ my $n;
+ $n = select($event->{o_r}, $event->{o_w}, $event->{o_e}, $timeout);
+ if ($n < 0 || ! defined($n)) {
+ if ($! == Errno::EINTR) {
+ $log->put("select interrupted");
+ return 0;
+ }
+ $log->put("select failed: $!");
+ return undef;
+ }
+ if (! $n) {
+ $log->put("select timed out");
+ }
+ return $n;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/File.pm b/ndb/tools/ndbnet/lib/NDB/Util/File.pm
new file mode 100644
index 00000000000..4b3cb38191c
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/File.pm
@@ -0,0 +1,163 @@
+package NDB::Util::File;
+
+use strict;
+use Carp;
+use Symbol;
+use Errno;
+use File::Basename;
+
+require NDB::Util::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Util::File->attributes(
+ path => sub { length > 0 },
+);
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $file = $class->SUPER::new(%attr);
+ $file->setpath($attr{path})
+ or $log->push, return undef;
+ return $file;
+}
+
+sub desc {
+ my $file = shift;
+ return $file->getpath;
+}
+
+sub getdir {
+ my $file = shift;
+ @_ == 0 or confess 0+@_;
+ my $dirpath = dirname($file->getpath);
+ my $dir = NDB::Util::Dir->new(path => $dirpath);
+ return $dir;
+}
+
+sub getlock {
+ my $file = shift;
+ @_ == 0 or confess 0+@_;
+ my $lock = NDB::Util::Lock->new(path => $file->getpath);
+ return $lock;
+}
+
+sub getbasename {
+ my $file = shift;
+ @_ == 0 or confess 0+@_;
+ return basename($file->getpath);
+}
+
+# make dir, unlink
+
+sub mkdir {
+ my $file = shift;
+ @_ == 0 or confess 0+@_;
+ return $file->getdir->mkdir;
+}
+
+sub unlink {
+ my $file = shift;
+ @_ == 0 or confess 0+@_;
+ $log->put("remove")->push($file)->debug;
+ if (-e $file->getpath) {
+ if (! unlink($file->getpath)) {
+ my $errstr = "$!";
+ if (! -e $file->getpath) {
+ return 1;
+ }
+ $log->put("unlink failed: $errstr")->push($file);
+ return undef;
+ }
+ }
+ return 1;
+}
+
+# read /write
+
+sub open {
+ my $file = shift;
+ @_ == 1 or confess 0+@_;
+ my($mode) = @_;
+ my $fh = gensym();
+ if (! open($fh, $mode.$file->getpath)) {
+ $log->put("open$mode failed")->push($file);
+ return undef;
+ }
+ my $io = NDB::Util::IO->new;
+ $io->setfh($fh)
+ or $log->push, return undef;
+ return $io;
+}
+
+sub puttext {
+ my $file = shift;
+ @_ == 1 or confess 0+@_;
+ my($text) = @_;
+ ref($text) and confess 'oops';
+ $file->mkdir
+ or $log->push, return undef;
+ $file->unlink
+ or $log->push, return undef;
+ my $io = $file->open(">")
+ or $log->push, return undef;
+ if (! $io->write($text)) {
+ $log->push($file);
+ $io->close;
+ return undef;
+ }
+ if (! $io->close) {
+ $log->push($file);
+ return undef;
+ }
+ return 1;
+}
+
+sub putlines {
+ my $file = shift;
+ @_ == 1 or confess 0+@_;
+ my($lines) = @_;
+ ref($lines) eq 'ARRAY' or confess 'oops';
+ my $text = join("\n", @$lines) . "\n";
+ $file->puttext($text) or $log->push, return undef;
+ return 1;
+}
+
+sub copyedit {
+ my $file1 = shift;
+ @_ == 2 or confess 0+@_;
+ my($file2, $edit) = @_;
+ my $io1 = $file1->open("<")
+ or $log->push, return undef;
+ my $io2 = $file2->open(">")
+ or $log->push, return undef;
+ local $_;
+ my $fh1 = $io1->getfh;
+ my $fh2 = $io2->getfh;
+ my $line = 0;
+ while (defined($_ = <$fh1>)) {
+ $line++;
+ if (! &$edit()) {
+ $log->push("line $line")->push($file1);
+ return undef;
+ }
+ print $fh2 $_;
+ }
+ $io1->close;
+ $io2->close;
+ return 1;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/IO.pm b/ndb/tools/ndbnet/lib/NDB/Util/IO.pm
new file mode 100644
index 00000000000..34f4d0a150d
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/IO.pm
@@ -0,0 +1,213 @@
+package NDB::Util::IO;
+
+use strict;
+use Carp;
+
+require NDB::Util::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::Base);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Util::IO->attributes(
+ readbuf => sub { defined },
+ readend => sub { defined },
+ writebuf => sub { defined },
+ writeend => sub { defined },
+ iosize => sub { $_ > 0 },
+ timeout => sub { /^\d+$/ },
+ fh => sub { ref($_) eq 'GLOB' && defined(fileno($_)) },
+);
+
+sub desc {
+ my $io = shift;
+ my $fileno = $io->hasfh ? fileno($io->getfh) : -1;
+ return "fd=$fileno";
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $io = $class->SUPER::new(%attr);
+ $io->setreadbuf("")
+ or $log->push, return undef;
+ $io->setreadend(0)
+ or $log->push, return undef;
+ $io->setwritebuf("")
+ or $log->push, return undef;
+ $io->setwriteend(0)
+ or $log->push, return undef;
+ $io->setiosize(1024)
+ or $log->push, return undef;
+ $io->settimeout(0)
+ or $log->push, return undef;
+ if (defined($attr{fh})) {
+ $io->setfh($attr{fh})
+ or $log->push, return undef;
+ }
+ return $io;
+}
+
+# input / output
+
+sub read {
+ my $io = shift;
+ @_ == 0 or confess 0+@_;
+ if ($io->getreadend) {
+ return "";
+ }
+ my $size = $io->getiosize;
+ my $timeout = $io->hastimeout ? $io->gettimeout : 0;
+ my $fh = $io->getfh;
+ my $n;
+ my $data;
+ eval {
+ if ($^O ne 'MSWin32' && $timeout > 0) {
+ local $SIG{ALRM} = sub { die("timed out\n") };
+ alarm($timeout);
+ $n = sysread($fh, $data, $size);
+ alarm(0);
+ }
+ else {
+ $n = sysread($fh, $data, $size);
+ }
+ };
+ if ($@) {
+ $log->put("read error: $@")->push($io);
+ return undef;
+ }
+ if (! defined($n)) {
+ $log->put("read failed: $!")->push($io);
+ return undef;
+ }
+ if ($n == 0) {
+ $io->setreadend(1)
+ or $log->push, return undef;
+ $log->put("read EOF")->push($io)->debug;
+ return "";
+ }
+ (my $show = $data) =~ s!\n!\\n!g;
+ $log->put("read: $show")->push($io)->debug;
+ return $data;
+}
+
+sub readbuf {
+ my $io = shift;
+ @_ == 0 or confess 0+@_;
+ my $data = $io->read;
+ defined($data) or
+ $log->push, return undef;
+ if (length($data) == 0) {
+ return 0;
+ }
+ $io->setreadbuf($io->getreadbuf . $data)
+ or $log->push, return undef;
+ return 1;
+}
+
+sub readupto {
+ my $io = shift;
+ @_ == 1 or confess 0+@_;
+ my($code) = @_;
+ ref($code) eq 'CODE' or confess 'oops';
+ my $k = &$code($io->getreadbuf);
+ if (! defined($k)) {
+ $log->push($io);
+ return undef;
+ }
+ if ($k == 0) {
+ my $n = $io->readbuf;
+ defined($n) or
+ $log->push, return undef;
+ if ($n == 0) {
+ if ($io->getreadbuf eq "") {
+ return "";
+ }
+ $log->put("incomplete input: %s", $io->getreadbuf)->push($io);
+ return undef;
+ }
+ $k = &$code($io->getreadbuf);
+ if (! defined($k)) {
+ $log->push($io);
+ return undef;
+ }
+ if ($k == 0) {
+ return "";
+ }
+ }
+ my $head = substr($io->getreadbuf, 0, $k);
+ my $tail = substr($io->getreadbuf, $k);
+ $io->setreadbuf($tail)
+ or $log->push, return undef;
+ return $head;
+}
+
+sub readline {
+ my $io = shift;
+ @_ == 0 or confess 0+@_;
+ my $code = sub {
+ my $i = index($_[0], "\n");
+ return $i < 0 ? 0 : $i + 1;
+ };
+ return $io->readupto($code);
+}
+
+sub write {
+ my $io = shift;
+ @_ == 1 or confess 0+@_;
+ my($data) = @_;
+ my $timeout = $io->hastimeout ? $io->gettimeout : 0;
+ my $fh = $io->getfh;
+ (my $show = $data) =~ s!\n!\\n!g;
+ $log->put("write: $show")->push($io)->debug;
+ my $n;
+ my $size = length($data);
+ eval {
+ local $SIG{PIPE} = sub { die("broken pipe\n") };
+ if ($^O ne 'MSWin32' && $timeout > 0) {
+ local $SIG{ALRM} = sub { die("timed out\n") };
+ alarm($timeout);
+ $n = syswrite($fh, $data, $size);
+ alarm(0);
+ }
+ else {
+ $n = syswrite($fh, $data, $size);
+ }
+ };
+ if ($@) {
+ $log->put("write error: $@")->push($io);
+ return undef;
+ }
+ if (! defined($n)) {
+ $log->put("write failed: $!")->push($io);
+ return undef;
+ }
+ if ($n > $size) {
+ $log->put("impossible write: $n > $size")->push($io);
+ return undef;
+ }
+ if ($n != $size) { # need not be error
+ $log->put("short write: $n < $size")->push($io);
+ }
+ return $n;
+}
+
+sub close {
+ my $io = shift;
+ @_ == 0 or confess 0+@_;
+ if (! close($io->delfh)) {
+ $log->put("close failed: $!")->push($io);
+ return undef;
+ }
+ return 1;
+}
+
+1;
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/Lock.pm b/ndb/tools/ndbnet/lib/NDB/Util/Lock.pm
new file mode 100644
index 00000000000..b515e633059
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/Lock.pm
@@ -0,0 +1,136 @@
+package NDB::Util::Lock;
+
+use strict;
+use Carp;
+use Symbol;
+use Fcntl qw(:flock);
+use Errno;
+use File::Basename;
+
+require NDB::Util::File;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::File);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Util::Lock->attributes(
+ pid => sub { $_ != 0 },
+);
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $lock = $class->SUPER::new(%attr);
+ return $lock;
+}
+
+sub desc {
+ my $lock = shift;
+ return $lock->getpath;
+}
+
+# test / set
+
+sub test {
+ my $lock = shift;
+ @_ == 0 or confess 0+@_;
+ my $fh = gensym();
+ if (! open($fh, "+<$lock->{path}")) {
+ if ($! != Errno::ENOENT) {
+ $log->put("$lock->{path}: open failed: $!");
+ return undef;
+ }
+ return 0; # file does not exist
+ }
+ if (flock($fh, LOCK_EX|LOCK_NB)) {
+ close($fh);
+ return 0; # file was not locked
+ }
+ if ($^O eq 'MSWin32') {
+ close($fh);
+ if (! open($fh, "<$lock->{path}x")) {
+ $log->put("$lock->{path}x: open failed: $!");
+ return undef;
+ }
+ }
+ my $pid = <$fh>;
+ close($fh);
+ ($pid) = split(' ', $pid);
+ if ($pid+0 == 0) {
+ $log->put("$lock->{path}: locked but pid='$pid' is zero");
+ return undef;
+ }
+ $lock->{pid} = $pid;
+ return 1; # file was locked
+}
+
+sub set {
+ my $lock = shift;
+ @_ == 0 or confess 0+@_;
+ my $fh = gensym();
+ if (! open($fh, "+<$lock->{path}")) {
+ if ($! != Errno::ENOENT) {
+ $log->put("$lock->{path}: open failed: $!");
+ return undef;
+ }
+ close($fh);
+ if (! open($fh, ">$lock->{path}")) {
+ $log->put("$lock->{path}: create failed: $!");
+ return undef;
+ }
+ }
+ if (! flock($fh, LOCK_EX|LOCK_NB)) {
+ $log->put("$lock->{path}: flock failed: $!");
+ close($fh);
+ return 0; # file was probably locked
+ }
+ my $line = "$$\n";
+ if ($^O eq 'MSWin32') {
+ my $gh = gensym();
+ if (! open($gh, ">$lock->{path}x")) {
+ $log->put("$lock->{path}x: open for write failed: $!");
+ close($fh);
+ return undef;
+ }
+ if (! syswrite($gh, $line)) {
+ close($fh);
+ close($gh);
+ $log->put("$lock->{path}x: write failed: $!");
+ return undef;
+ }
+ close($gh);
+ } else {
+ if (! truncate($fh, 0)) {
+ close($fh);
+ $log->put("$lock->{path}: truncate failed: $!");
+ return undef;
+ }
+ if (! syswrite($fh, $line)) {
+ close($fh);
+ $log->put("$lock->{path}: write failed: $!");
+ return undef;
+ }
+ }
+ $lock->{fh} = $fh;
+ return 1; # file is now locked by us
+}
+
+sub close {
+ my $lock = shift;
+ @_ == 0 or confess 0+@_;
+ my $fh = delete $lock->{fh};
+ if ($fh) {
+ close($fh);
+ }
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/Log.pm b/ndb/tools/ndbnet/lib/NDB/Util/Log.pm
new file mode 100644
index 00000000000..44b39df84e6
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/Log.pm
@@ -0,0 +1,367 @@
+package NDB::Util::Log;
+
+use strict;
+use Carp;
+use Symbol;
+use Data::Dumper ();
+
+require NDB::Util::Base;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::Base);
+
+# constructors
+
+my $instance = undef;
+my %attached = ();
+
+my %priolevel = qw(user 0 fatal 1 error 2 warn 3 notice 4 info 5 debug 6);
+my %partlist = qw(time 1 pid 2 prio 3 text 4 line 5);
+
+NDB::Util::Log->attributes(
+ prio => sub { defined($priolevel{$_}) },
+ parts => sub { ref eq 'HASH' },
+ stack => sub { ref eq 'ARRAY' },
+ io => sub { ref && $_->isa('NDB::Util::IO') },
+ active => sub { defined },
+ censor => sub { ref eq 'ARRAY' },
+);
+
+sub setpart {
+ my $log = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ while (@_) {
+ my $part = shift;
+ my $onoff = shift;
+ $partlist{$part} or confess 'oops';
+ $log->getparts->{$part} = $onoff;
+ }
+}
+
+sub getpart {
+ my $log = shift;
+ @_ == 1 or confess 0+@_;
+ my($part) = @_;
+ $partlist{$part} or confess 'oops';
+ return $log->getparts->{$part};
+}
+
+sub instance {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ if (! $instance) {
+ $instance = $class->SUPER::new(%attr);
+ $instance->setprio(q(info));
+ $instance->setparts({ text => 1 });
+ $instance->setstack([]);
+ $instance->setcensor([]);
+ my $io = NDB::Util::IO->new(fh => \*STDERR, %attr)
+ or confess 'oops';
+ $instance->setio($io);
+ }
+ return $instance;
+}
+
+# attached logs are written in parallel to main log
+# user log is a special server-to-client log
+
+sub attach {
+ my $log = shift;
+ @_ % 2 == 1 or confess 0+@_;
+ my($key, %attr) = @_;
+ $attached{$key} and confess 'oops';
+ my $alog = $attached{$key} = $log->clone(%attr);
+ return $alog;
+}
+
+sub detach {
+ my $log = shift;
+ @_ == 1 or confess 0+@_;
+ my($key) = @_;
+ $attached{$key} or return undef;
+ my $alog = delete $attached{$key};
+ return $alog;
+}
+
+sub attachuser {
+ my $log = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ %attr = (
+ prio => q(user),
+ parts => { text => 1 },
+ censor => [ qw(NDB::Net::Client NDB::Util::IO) ],
+ %attr);
+ my $alog = $log->attach(q(user), %attr);
+ return $alog;
+}
+
+sub detachuser {
+ my $log = shift;
+ @_ == 0 or confess 0+@_;
+ my $alog = $log->detach(q(user));
+ return $alog;
+}
+
+# input / output
+
+sub setfile {
+ my $log = shift;
+ @_ == 1 or confess 0+@_;
+ my $file = shift;
+ if (! open(STDOUT, ">>$file")) {
+ $log->put("$file: open for append failed: $!");
+ return undef;
+ }
+ select(STDOUT);
+ $| = 1;
+ open(STDERR, ">&STDOUT");
+ select(STDERR);
+ $| = 1;
+ return 1;
+}
+
+sub close {
+ my $log = shift;
+ $log->getio->close;
+}
+
+sub closeall {
+ my $class = shift;
+ for my $key (sort keys %attached) {
+ my $log = $attached{$key};
+ $log->close;
+ }
+ $instance->close;
+}
+
+# private
+
+sub entry {
+ my $log = shift;
+ my($clear, $file, $line, @args) = @_;
+ $file =~ s!^.*\bNDB/!!;
+ $file =~ s!^.*/bin/([^/]+)$!$1!;
+ my $text = undef;
+ if (@args) {
+ $text = shift(@args);
+ if (! ref($text)) {
+ if (@args) {
+ $text = sprintf($text, @args);
+ }
+ while (chomp($text)) {}
+ }
+ }
+ if ($clear) {
+ $#{$log->getstack} = -1;
+ }
+ push(@{$log->getstack}, {
+ line => "$file($line)",
+ text => $text,
+ });
+}
+
+sub matchlevel {
+ my $log = shift;
+ my $msgprio = shift;
+ my $logprio = $log->getprio;
+ my $msglevel = $priolevel{$msgprio};
+ my $loglevel = $priolevel{$logprio};
+ defined($msglevel) && defined($loglevel)
+ or confess 'oops';
+ if ($msglevel == 0 && $loglevel == 0) {
+ return $msgprio eq $logprio;
+ }
+ if ($msglevel == 0 && $loglevel != 0) {
+ return $loglevel >= $priolevel{q(info)};
+ }
+ if ($msglevel != 0 && $loglevel == 0) {
+ return $msglevel <= $priolevel{q(notice)};
+ }
+ if ($msglevel != 0 && $loglevel != 0) {
+ return $msglevel <= $loglevel;
+ }
+ confess 'oops';
+}
+
+sub print {
+ my $log = shift;
+ @_ == 2 or confess 0+@_;
+ my($prio, $tmpstack) = @_;
+ if ($log->hasactive) { # avoid recursion
+ return;
+ }
+ if (! $log->matchlevel($prio)) {
+ return;
+ }
+ $log->setactive(1);
+ my @text = ();
+ if ($log->getpart(q(time))) {
+ my @t = localtime(time);
+ push(@text, sprintf("%02d-%02d/%02d:%02d:%02d",
+ 1+$t[4], $t[3], $t[2], $t[1], $t[0]));
+ }
+ if ($log->getpart(q(pid))) {
+ push(@text, "[$$]");
+ }
+ if ($log->getpart(q(prio)) &&
+ (0 == $priolevel{$prio} || $priolevel{$prio} <= $priolevel{notice}))
+ {
+ push(@text, "[$prio]");
+ }
+ if ($log->getpart(q(text))) {
+ my @stack = @$tmpstack;
+ while (@stack) {
+ my $s = pop(@stack);
+ my $text = $s->{text};
+ if (ref($text)) {
+ if (grep($text->isa($_), @{$log->getcensor})) {
+ next;
+ }
+ $text = $text->desc;
+ }
+ push(@text, $text) if length($text) > 0;
+ }
+ }
+ if ($log->getpart(q(line)) &&
+ (0 < $priolevel{$prio} && $priolevel{$prio} <= $priolevel{warn}))
+ {
+ push(@text, "at");
+ my @stack = @$tmpstack;
+ while (@stack) {
+ my $s = shift(@stack);
+ defined($s->{line}) or confess 'oops';
+ if ($text[-1] ne $s->{line}) {
+ push(@text, $s->{line});
+ }
+ }
+ }
+ $log->getio->write("@text\n");
+ $log->delactive;
+}
+
+sub printall {
+ my $log = shift;
+ @_ == 1 or confess 0+@_;
+ my($prio) = @_;
+ my $logstack = $log->getstack;
+ if (! @$logstack) {
+ $log->put("[missing log message]");
+ }
+ my @tmpstack = ();
+ while (@$logstack) {
+ push(@tmpstack, shift(@$logstack));
+ }
+ for my $key (sort keys %attached) {
+ my $alog = $attached{$key};
+ $alog->print($prio, \@tmpstack);
+ }
+ $instance->print($prio, \@tmpstack);
+}
+
+# public
+
+sub push {
+ my $log = shift;
+ my(@args) = @_;
+ my($pkg, $file, $line) = caller;
+ $log->entry(0, $file, $line, @args);
+ return $log;
+}
+
+sub put {
+ my $log = shift;
+ my(@args) = @_;
+ my($pkg, $file, $line) = caller;
+ $log->entry(1, $file, $line, @args);
+ return $log;
+}
+
+sub fatal {
+ my $log = shift;
+ @_ == 0 or confess 0+@_;
+ $log->printall(q(fatal));
+ exit(1);
+}
+
+sub error {
+ my $log = shift;
+ @_ == 0 or confess 0+@_;
+ $log->printall(q(error));
+ return $log;
+}
+
+sub warn {
+ my $log = shift;
+ @_ == 0 or confess 0+@_;
+ $log->printall(q(warn));
+ return $log;
+}
+
+sub notice {
+ my $log = shift;
+ @_ == 0 or confess 0+@_;
+ $log->printall(q(notice));
+ return $log;
+}
+
+sub info {
+ my $log = shift;
+ @_ == 0 or confess 0+@_;
+ $log->printall(q(info));
+ return $log;
+}
+
+sub debug {
+ my $log = shift;
+ @_ == 0 or confess 0+@_;
+ $log->printall(q(debug));
+ return $log;
+}
+
+sub user {
+ my $log = shift;
+ @_ == 0 or confess 0+@_;
+ $log->printall(q(user));
+ return $log;
+}
+
+# return values from server to client
+
+sub putvalue {
+ my $log = shift;
+ @_ == 1 or confess 0+@_;
+ my($value) = @_;
+ my $d = Data::Dumper->new([$value], [qw($value)]);
+ $d->Indent(0);
+ $d->Useqq(1);
+ my $dump = $d->Dump;
+ $dump =~ /^\s*\$value\s*=\s*(.*);\s*$/ or confess $dump;
+ $log->push("[value $1]");
+}
+
+sub hasvalue {
+ my $log = shift;
+ @_ == 1 or confess 0+@_;
+ my($line) = @_;
+ return $line =~ /\[value\s+(.*)\]/;
+}
+
+sub getvalue {
+ my $log = shift;
+ @_ == 1 or confess 0+@_;
+ my($line) = @_;
+ $line =~ /\[value\s+(.*)\]/ or confess $line;
+ my $expr = $1;
+ my($value);
+ eval "\$value = $expr";
+ if ($@) {
+ $log->put("$line: eval error: $@");
+ return undef;
+ }
+ return [$value];
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/Socket.pm b/ndb/tools/ndbnet/lib/NDB/Util/Socket.pm
new file mode 100644
index 00000000000..00e8b6eca51
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/Socket.pm
@@ -0,0 +1,158 @@
+package NDB::Util::Socket;
+
+use strict;
+use Carp;
+use Symbol;
+use Socket;
+use Errno;
+
+require NDB::Util::IO;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::IO);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Util::Socket->attributes(
+ domain => sub { $_ == PF_INET || $_ == PF_UNIX },
+ type => sub { $_ == SOCK_STREAM },
+ proto => sub { /^(0|tcp)$/ },
+);
+
+sub desc {
+ my $socket = shift;
+ return $socket->SUPER::desc;
+}
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $socket = $class->SUPER::new(%attr);
+ $socket->setdomain($attr{domain})
+ or $log->push, return undef;
+ $socket->settype($attr{type})
+ or $log->push, return undef;
+ $socket->setproto($attr{proto})
+ or $log->push, return undef;
+ my $nproto;
+ if ($socket->getproto =~ /^\d+/) {
+ $nproto = $socket->getproto;
+ }
+ else {
+ $nproto = getprotobyname($socket->getproto);
+ unless (defined($nproto)) {
+ $log->put("%s: getprotobyname failed", $socket->getproto);
+ return undef;
+ }
+ }
+ my $fh = gensym();
+ if (! socket($fh, $socket->getdomain, $socket->gettype, $nproto)) {
+ $log->put("create socket failed: $!");
+ return undef;
+ }
+ $socket->setfh($fh)
+ or $log->push, return undef;
+ return $socket;
+}
+
+sub setopt {
+ my $socket = shift;
+ @_ >= 2 or confess 'oops';
+ my $level = shift;
+ my $optname = shift;
+ my $optval = @_ ? pack("l*", @_) : undef;
+ my $fh = $socket->getfh;
+ if (! setsockopt($fh, $level, $optname, $optval)) {
+ $log->put("setsockopt failed: $!")->push($socket);
+ return undef;
+ }
+ return 1;
+}
+
+sub connect {
+ my $socket = shift;
+ @_ == 1 or confess 0+@_;
+ my($paddr) = @_;
+ my $fh = $socket->getfh;
+ if (! connect($fh, $paddr)) {
+ $log->put("connect failed: $!")->push($socket);
+ return undef;
+ }
+ $log->put("connect done")->push($socket)->debug;
+ return 1;
+}
+
+sub bind {
+ my $socket = shift;
+ @_ == 1 or confess 0+@_;
+ my($paddr) = @_;
+ my $fh = $socket->getfh;
+ if (! bind($fh, $paddr)) {
+ $log->put("bind failed: $!")->push($socket);
+ return undef;
+ }
+ return 1;
+}
+
+sub listen {
+ my $socket = shift;
+ @_ == 0 or confess 0+@_;
+ my $fh = $socket->getfh;
+ if (! listen($fh, SOMAXCONN)) {
+ $log->put("listen failed: $!")->push($socket);
+ return undef;
+ }
+ return 1;
+}
+
+sub accept {
+ my $socket = shift;
+ @_ == 1 or confess 0+@_;
+ my($timeout) = @_;
+ $timeout =~ /^\d+$/ or confess 'oops';
+ my $fh = $socket->getfh;
+ my $gh = gensym();
+ my $paddr;
+ eval {
+ if ($^O ne 'MSWin32' && $timeout > 0) {
+ local $SIG{ALRM} = sub { die("timed out\n") };
+ alarm($timeout);
+ $paddr = accept($gh, $fh);
+ alarm(0);
+ }
+ else {
+ $paddr = accept($gh, $fh);
+ }
+ };
+ if ($@) {
+ $log->put("accept failed: $@")->push($socket);
+ return undef;
+ }
+ if (! $paddr) {
+ my $errno = 0+$!;
+ if ($errno == Errno::EINTR) {
+ $log->put("accept interrupted")->push($socket);
+ return 0;
+ }
+ $log->put("accept failed: $!")->push($socket);
+ return undef;
+ }
+ my $csocket = $socket->clone(fh => $gh);
+ $csocket->acceptaddr($paddr);
+ return $csocket;
+}
+
+sub DESTROY {
+ my $socket = shift;
+ $socket->close;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/SocketINET.pm b/ndb/tools/ndbnet/lib/NDB/Util/SocketINET.pm
new file mode 100644
index 00000000000..faaa568a08e
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/SocketINET.pm
@@ -0,0 +1,86 @@
+package NDB::Util::SocketINET;
+
+use strict;
+use Carp;
+use Symbol;
+use Socket;
+use Errno;
+
+require NDB::Util::Socket;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::Socket);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Util::SocketINET->attributes(
+ host => sub { /^\S+$/ },
+ port => sub { /^\d+$/ },
+);
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $socket = $class->SUPER::new(%attr,
+ domain => PF_INET, type => SOCK_STREAM, proto => 'tcp')
+ or $log->push, return undef;
+ return $socket;
+}
+
+sub connect {
+ my $socket = shift;
+ @_ == 2 or confess 0+@_;
+ my($host, $port) = @_;
+ $port =~ /^\d+$/ or confess 'oops';
+ my $iaddr = inet_aton($host);
+ if (! $iaddr) {
+ $log->put("host $host not found")->push($socket);
+ return undef;
+ }
+ my $paddr = pack_sockaddr_in($port, $iaddr);
+ $socket->SUPER::connect($paddr)
+ or $log->push, return undef;
+ $socket->sethost($host)
+ or $log->push, return undef;
+ $socket->setport($port)
+ or $log->push, return undef;
+ return 1;
+}
+
+sub bind {
+ my $socket = shift;
+ @_ == 1 or confess 0+@_;
+ my($port) = @_;
+ $port =~ /^\d+$/ or confess 'oops';
+ my $paddr = pack_sockaddr_in($port, INADDR_ANY);
+ $socket->SUPER::bind($paddr)
+ or $log->push, return undef;
+ $socket->setport($port)
+ or $log->push, return undef;
+ return 1;
+}
+
+sub acceptaddr {
+ my $csocket = shift;
+ @_ == 1 or confess 0+@_;
+ my($paddr) = @_;
+ my($port, $iaddr) = unpack_sockaddr_in($paddr);
+ my $host = gethostbyaddr($iaddr, AF_INET);
+ $csocket->sethost($host)
+ or $log->push, return undef;
+ $csocket->setport($port)
+ or $log->push, return undef;
+ $log->put("accept: host=%s port=%d",
+ $csocket->gethost, $csocket->getport)->push($csocket)->debug;
+ return 1;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/lib/NDB/Util/SocketUNIX.pm b/ndb/tools/ndbnet/lib/NDB/Util/SocketUNIX.pm
new file mode 100644
index 00000000000..9c6b3115f6a
--- /dev/null
+++ b/ndb/tools/ndbnet/lib/NDB/Util/SocketUNIX.pm
@@ -0,0 +1,76 @@
+package NDB::Util::SocketUNIX;
+
+use strict;
+use Carp;
+use Symbol;
+use Socket;
+use Errno;
+
+require NDB::Util::Socket;
+
+use vars qw(@ISA);
+@ISA = qw(NDB::Util::Socket);
+
+# constructors
+
+my $log;
+
+sub initmodule {
+ $log = NDB::Util::Log->instance;
+}
+
+NDB::Util::SocketUNIX->attributes(
+ path => sub { /^\S+$/ },
+);
+
+sub new {
+ my $class = shift;
+ @_ % 2 == 0 or confess 0+@_;
+ my(%attr) = @_;
+ my $socket = $class->SUPER::new(%attr,
+ domain => PF_UNIX, type => SOCK_STREAM, proto => 0)
+ or $log->push, return undef;
+ return $socket;
+}
+
+sub connect {
+ my $socket = shift;
+ @_ == 1 or confess 0+@_;
+ my($path) = @_;
+ $path =~ /^\S+$/ or confess 'oops';
+ my $paddr = pack_sockaddr_un($path);
+ $socket->SUPER::connect($paddr)
+ or $log->push, return undef;
+ $socket->setpath($path)
+ or $log->push, return undef;
+ return 1;
+}
+
+sub bind {
+ my $socket = shift;
+ @_ == 1 or confess 0+@_;
+ my($path) = @_;
+ $path =~ /^\S+$/ or confess 'oops';
+ my $paddr = pack_sockaddr_un($path);
+ $socket->SUPER::bind($paddr)
+ or $log->push, return undef;
+ $socket->setpath($path)
+ or $log->push, return undef;
+ return 1;
+}
+
+sub acceptaddr {
+ my $csocket = shift;
+ @_ == 1 or confess 0+@_;
+ my($paddr) = @_;
+ return 1; # crash
+ my $path = unpack_sockaddr_un($paddr);
+ $csocket->setpath($path)
+ or $log->push, return undef;
+ $log->put("%s accept: path=%s",
+ $csocket->getpath)->push($csocket)->debug;
+ return 1;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/ndbnet.pl b/ndb/tools/ndbnet/ndbnet.pl
new file mode 100644
index 00000000000..5f6648da46d
--- /dev/null
+++ b/ndb/tools/ndbnet/ndbnet.pl
@@ -0,0 +1,339 @@
+#! /usr/local/bin/perl
+
+use strict;
+use POSIX();
+use Socket;
+use Getopt::Long;
+use File::Basename;
+use Term::ReadLine;
+
+use NDB::Net;
+
+select(STDOUT);
+$| = 1;
+
+# get options and environment
+
+my $log = NDB::Util::Log->instance;
+$log->setpart();
+
+sub printhelp {
+ print <<END;
+ndbnet -- ndbnet client
+usage: ndbnet [options] [command...]
+--help print this text and exit
+--base dir ndb installation, default \$NDB_BASE
+--netcfg file net config, default \$NDB_BASE/etc/ndbnet.xml
+--server id ndbnetd server id, or host:port if no config
+--noterm no prompting and no input editing
+--log prio debug/info/notice/warn/error/fatal, default info
+command... command (by default becomes interactive)
+END
+ exit(0);
+}
+
+my $progopts = {};
+my @progargv;
+
+anon: {
+ local $SIG{__WARN__} = sub {
+ my $errstr = "@_";
+ while (chomp($errstr)) {}
+ $log->put("$errstr (try --help)")->fatal;
+ };
+ Getopt::Long::Configure(qw(
+ default no_getopt_compat no_ignore_case require_order
+ ));
+ GetOptions($progopts, qw(
+ help base=s netcfg=s server=s noterm log=s
+ ));
+}
+
+$progopts->{help} && printhelp();
+if (defined(my $prio = $progopts->{log})) {
+ $log->setprio($prio);
+}
+@progargv = @ARGV;
+
+my $netenv = NDB::Net::Env->instance(
+ base => $progopts->{base},
+ netcfg => $progopts->{netcfg},
+);
+$netenv or $log->fatal;
+
+# get servers from command line or from net config
+
+my @servers = ();
+my $netcfg;
+if ($netenv->hasnetcfg) {
+ $netcfg = NDB::Net::Config->new(file => $netenv->getnetcfg);
+}
+
+if (defined(my $id = $progopts->{server})) {
+ if ($id !~ /:/) {
+ $netcfg or $log->put("need net config to find server $id")->fatal;
+ $netcfg->load or $log->push->fatal;
+ $netcfg->getservers or $log->push->fatal;
+ my $s = NDB::Net::Server->get($id) or $log->fatal;
+ push(@servers, $s);
+ } else {
+ my($host, $port) = split(/:/, $id, 2);
+ my $s = NDB::Net::ServerINET->new(id => "?", host => $host, port => $port)
+ or $log->fatal;
+ push(@servers, $s);
+ }
+} else {
+ $netcfg or $log->put("need net config to find servers")->fatal;
+ $netcfg->load or $log->push->fatal;
+ my $list = $netcfg->getservers or $log->fatal;
+ @servers= @$list;
+ @servers or $log->put("no servers")->push($netcfg)->fatal;
+}
+
+# server commands
+
+my $server;
+sub doserver {
+ my($cmd) = @_;
+ my $ret;
+ my $found;
+ for my $s (@servers) {
+ if (! $s->testconnect) {
+ $log->warn;
+ next;
+ }
+ $found = 1;
+ if ($server ne $s) {
+ $server = $s;
+ $log->put("selected")->push($server)->debug;
+ }
+ $ret = $server->request($cmd);
+ last;
+ }
+ if (! $found) {
+ $log->put("no available server");
+ return undef;
+ }
+ my %seen = ();
+ @servers = grep(! $seen{$_}++, $server, @servers);
+ defined($ret) or $log->push, return undef;
+ return $ret;
+}
+
+# local commands
+
+sub cmd_help {
+ my($cmd) = @_;
+ my $text = $cmd->helptext;
+ defined($text) or return undef;
+ while(chomp($text)) {}
+ print $text, "\n";
+ return 1;
+}
+
+sub cmd_alias {
+ my($cmd) = @_;
+ my $text = $cmd->aliastext;
+ while(chomp($text)) {}
+ print $text, "\n";
+}
+
+sub cmd_quit {
+ my($cmd) = @_;
+ $log->put("bye-bye")->info;
+ exit(0);
+}
+
+sub cmd_server {
+ my($cmd) = @_;
+ my $action = $cmd->getarg(0);
+ if ($action !~ /^(start|restart|stop|ping)$/) {
+ $log->put("$action: undefined action");
+ return undef;
+ }
+ if ($action eq 'start') {
+ $cmd->setopt('direct')
+ or $log->push, return undef;
+ }
+ if ($action eq 'ping' && ! @{$cmd->getarglist(1)}) {
+ $cmd->setopt('all')
+ or $log->push, return undef;
+ }
+ if (! $cmd->getopt('direct')) {
+ return doserver($cmd);
+ }
+ $netcfg->load
+ or return undef;
+ my $servers = $netcfg->getservers
+ or return undef;
+ my $list;
+ if ($cmd->getopt('all')) {
+ $list = $servers;
+ }
+ else {
+ $list = [];
+ for my $id (@{$cmd->getarglist(1)}) {
+ if (my $s = NDB::Net::ServerINET->get($id)) {
+ push(@$list, $s);
+ next;
+ }
+ if (my $s = NDB::Net::ServerINET->match($id, undef, $servers)) {
+ if (@$s) {
+ push(@$list, @$s);
+ next;
+ }
+ }
+ $log->push;
+ return undef;
+ }
+ }
+ if (! @$list) {
+ $log->put("no servers specified, use --all for all")->info;
+ return 1;
+ }
+ for my $s (@$list) {
+ if ($action eq 'ping') {
+ if ($s->testconnect) {
+ $log->put("is alive")->push($s);
+ }
+ $log->info;
+ next;
+ }
+ if ($action eq 'start') {
+ if ($s->testconnect) {
+ $log->put("already running")->push($s)->info;
+ next;
+ }
+ }
+ my $script = $cmd->getopt('script') || "ndbnetd";
+ my @cmd = ($script);
+ if ($action eq 'restart') {
+ push(@cmd, "--restart");
+ }
+ if ($action eq 'stop') {
+ push(@cmd, "--stop");
+ }
+ if ($cmd->getopt('pass')) {
+ my $base = $netenv->getbase;
+ $cmd[0] = "$base/bin/$cmd[0]";
+ }
+ if ($cmd->getopt('parallel')) {
+ my $pid = fork;
+ defined($pid) or
+ $log->push("fork failed: $!"), return undef;
+ $pid > 0 && next;
+ }
+ $log->put("$action via ssh")->push($s->getcanon)->push($s)->info;
+ $log->put("run: @cmd")->push($s)->debug;
+ system 'ssh', '-n', $s->getcanon, "@cmd";
+ if ($cmd->getopt('parallel')) {
+ exit(0);
+ }
+ }
+ if ($cmd->getopt('parallel')) {
+ while ((my $pid = waitpid(-1, &POSIX::WNOHANG)) > 0) {
+ ;
+ }
+ }
+ return 1;
+}
+
+sub cmd_list {
+ my($cmd) = @_;
+ my $ret = doserver($cmd) or
+ $log->push, return undef;
+ my @out = ();
+ my @o = qw(NAME NODES PROCESS STATUS COMMENT);
+ push(@out, [ @o ]);
+ for my $name (sort keys %$ret) {
+ $#o = -1;
+ $o[0] = $name;
+ my $dbsts = $ret->{$name};
+ my @tmp = sort { $a->{id} <=> $b->{id} } values %{$dbsts->{node}};
+ my @nodesmgmt = grep($_->{type} eq 'mgmt', @tmp);
+ my @nodesdb = grep($_->{type} eq 'db', @tmp);
+ my @nodesapi = grep($_->{type} eq 'api', @tmp);
+ my @nodes = (@nodesmgmt, @nodesdb, @nodesapi);
+ $o[1] = sprintf("%d/%d/%d", 0+@nodesmgmt, 0+@nodesdb, 0+@nodesapi);
+ $o[2] = "-";
+ $o[3] = "-";
+ $o[4] = $dbsts->{comment};
+ $o[4] .= " - " if length $o[4];
+ $o[4] .= basename($dbsts->{home});
+ push(@out, [ @o ]);
+ for my $nodests (@nodes) {
+ $#o = -1;
+ $o[0] = $nodests->{id} . "-" . $nodests->{type};
+ $o[1] = $nodests->{host};
+ $o[1] =~ s/\..*//;
+ $o[2] = $nodests->{run};
+ $o[3] = $nodests->{status} || "-";
+ $o[4] = $nodests->{comment} || "-";
+ push(@out, [ @o ]);
+ }
+ }
+ my @len = ( 8, 8, 8, 8 );
+ for my $o (@out) {
+ for my $i (0..$#len) {
+ $len[$i] = length($o->[$i]) if $len[$i] < length($o->[$i]);
+ }
+ }
+ for my $o (@out) {
+ my @t = ();
+ for my $i (0..$#{$out[0]}) {
+ my $f = $len[$i] ? "%-$len[$i].$len[$i]s" : "%s";
+ push(@t, sprintf($f, $o->[$i]));
+ }
+ print "@t\n";
+ }
+ return 1;
+}
+
+# main program
+
+sub docmd {
+ my(@args) = @_;
+ my $cmd = NDB::Net::Command->new(@args)
+ or return undef;
+ my $name = $cmd->getname;
+ my $doit;
+ {
+ no strict 'refs';
+ $doit = *{"cmd_$name"};
+ }
+ if (! defined(&$doit)) {
+ $doit = \&doserver;
+ }
+ my $ret = &$doit($cmd);
+ defined($ret) or $log->push, return undef;
+ return $ret;
+}
+
+if (@progargv) {
+ docmd(argv => \@progargv) or $log->push->fatal;
+ exit(0);
+}
+
+my $term;
+if ((-t STDIN) && (-t STDOUT) && ! $progopts->{noterm}) {
+ $term = Term::ReadLine->new("ndbnet");
+ $term->ornaments(0);
+}
+
+print "type 'h' for help\n" if $term;
+while (1) {
+ my($line);
+ while (! $line) {
+ $line = $term ? $term->readline("> ") : <STDIN>;
+ if (! defined($line)) {
+ print("\n") if $term;
+ $line = 'EOF';
+ }
+ }
+ my $ret = docmd(line => $line);
+ $ret or $log->error;
+ ($line eq 'EOF') && last;
+}
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/ndbnetd.pl b/ndb/tools/ndbnet/ndbnetd.pl
new file mode 100644
index 00000000000..95fa5322abc
--- /dev/null
+++ b/ndb/tools/ndbnet/ndbnetd.pl
@@ -0,0 +1,400 @@
+#! /usr/local/bin/perl
+
+use strict;
+use POSIX();
+use Socket;
+use Getopt::Long;
+use File::Basename;
+use File::Spec;
+
+use NDB::Net;
+
+# save argv for restart via client
+my @origargv = @ARGV;
+
+# get options and environment
+
+my $log = NDB::Util::Log->instance;
+
+sub printhelp {
+ print <<END;
+ndbnetd -- ndbnet daemon
+usage: ndbnetd [options]
+--help print this text and exit
+--base dir ndb installation, default \$NDB_BASE
+--netcfg file net config, default \$NDB_BASE/etc/ndbnet.xml
+--port num port number (if more than 1 server on this host)
+--stop kill any existing server
+--restart kill any existing server and start a new one
+--fg run in foreground (test option)
+--log prio debug/info/notice/warn/error/fatal, default info
+END
+ exit(0);
+}
+
+my $progopts = {};
+anon: {
+ local $SIG{__WARN__} = sub {
+ my $errstr = "@_";
+ while (chomp($errstr)) {}
+ $log->put("$errstr (try --help)")->fatal;
+ };
+ Getopt::Long::Configure(qw(
+ default no_getopt_compat no_ignore_case no_require_order
+ ));
+ GetOptions($progopts, qw(
+ help base=s netcfg=s port=i stop restart fg log=s
+ ));
+}
+$progopts->{help} && printhelp();
+if (defined(my $prio = $progopts->{log})) {
+ $log->setprio($prio);
+}
+@ARGV and $log->put("extra args on command line")->fatal;
+
+my $netenv = NDB::Net::Env->instance(
+ base => $progopts->{base},
+ netcfg => $progopts->{netcfg},
+);
+$netenv or $log->fatal;
+$netenv->hasbase or $log->put("need NDB_BASE")->fatal;
+
+# load net config and find our entry
+
+my $netcfg = NDB::Net::Config->new(file => $netenv->getnetcfg)
+ or $log->push->fatal;
+my $server;
+
+sub loadnetcfg {
+ $netcfg->load or $log->push->fatal;
+ my $servers = $netcfg->getservers or $log->fatal;
+ my $host = $netenv->gethostname;
+ my $port = $progopts->{port} || 0;
+ my $list = NDB::Net::ServerINET->match($host, $port, $servers)
+ or $log->push->fatal;
+ @$list == 1
+ or $log->push->fatal;
+ $server = $list->[0];
+ $server->setlocal;
+}
+loadnetcfg();
+$log->put("this server")->push($server)->debug;
+
+# check if server already running
+
+my $lock;
+anon: {
+ my $dir = NDB::Util::Dir->new(path => File::Spec->catfile($netenv->getbase, "run"));
+ $dir->mkdir or $log->fatal;
+ my $name = sprintf("ndbnet%s.pid", $server->getid);
+ $lock = $dir->getfile($name)->getlock;
+ my $ret;
+ $ret = $lock->test;
+ defined($ret) or $log->fatal;
+ if ($ret) {
+ if ($progopts->{stop} || $progopts->{restart}) {
+ $log->put("stopping server %s pid=%s", $netenv->gethostname, $lock->getpid)->info;
+ if ($^O ne 'MSWin32') {
+ kill -15, $lock->getpid;
+ } else {
+ kill 15, $lock->getpid;
+ }
+ while (1) {
+ sleep 1;
+ $ret = $lock->test;
+ defined($ret) or $log->fatal;
+ if ($ret) {
+ if (! kill(0, $lock->getpid) && $! == Errno::ESRCH) {
+ $log->put("locked but gone (linux bug?)")->info;
+ $lock->unlink;
+ $ret = 0;
+ }
+ }
+ if (! $ret) {
+ if ($progopts->{stop}) {
+ $log->put("stopped")->info;
+ exit(0);
+ }
+ $log->put("restarting server %s", $netenv->gethostname)->info;
+ last;
+ }
+ }
+ }
+ else {
+ $log->put("already running pid=%s", $lock->getpid)->fatal;
+ }
+ }
+ else {
+ if ($progopts->{stop}) {
+ $log->put("not running")->info;
+ exit(0);
+ }
+ }
+ $lock->set or $log->fatal;
+}
+
+# become daemon, re-obtain the lock, direct log to file
+
+anon: {
+ $log->setpart(time => 1, pid => 1, prio => 1, line => 1);
+ $progopts->{fg} && last anon;
+ $lock->close;
+ my $dir = NDB::Util::Dir->new(path => $netenv->getbase . "/log");
+ $dir->mkdir or $log->fatal;
+ my $pid = fork();
+ defined($pid) or $log->put("fork failed: $!")->fatal;
+ if ($pid) {
+ exit(0);
+ }
+ $lock->set or $log->fatal;
+ if ($^O ne 'MSWin32') {
+ POSIX::setsid() or $log->put("setsid failed: $!")->fatal;
+ }
+ open(STDIN, "</dev/null");
+ my $name = sprintf("ndbnet%s.log", $server->getid);
+ $log->setfile($dir->getfile($name)->getpath) or $log->fatal;
+}
+$log->put("ndbnetd started pid=$$ port=%s", $server->getport)->info;
+
+# create server socket and event
+
+my $socket = NDB::Util::SocketINET->new or $log->fatal;
+my $event = NDB::Util::Event->new;
+
+# commands
+
+sub cmd_server_fg {
+ my($cmd) = @_;
+ my $action = $cmd->getarg(0);
+ if (! $cmd->getopt('local')) {
+ return 1;
+ }
+ if ($action eq 'restart') {
+ my $prog = $netenv->getbase . "/bin/ndbnetd";
+ my @argv = @origargv;
+ if (! grep(/^--restart$/, @argv)) {
+ push(@argv, "--restart");
+ }
+ unshift(@argv, basename($prog));
+ $lock->close;
+ $socket->close;
+ $log->put("restart: @argv")->push($server)->user;
+ $log->put("server restart")->putvalue(1)->user;
+ exec $prog @argv;
+ die "restart failed: $!";
+ }
+ if ($action eq 'stop') {
+ $log->put("stop by request")->push($server)->user;
+ $log->put("server stop")->putvalue(1)->user;
+ exit(0);
+ }
+ if ($action eq 'ping') {
+ return 1;
+ }
+ $log->put("$action: unimplemented");
+ return undef;
+}
+
+sub cmd_server_bg {
+ my($cmd) = @_;
+ loadnetcfg() or return undef;
+ my $action = $cmd->getarg(0);
+ if (! $cmd->getopt('local')) {
+ $cmd->setopt('local')
+ or $log->push, return undef;
+ my $servers = $netcfg->getservers or $log->fatal;
+ my $list;
+ if ($cmd->getopt('all')) {
+ $list = $servers;
+ }
+ else {
+ $list = [];
+ for my $id (@{$cmd->getarglist(1)}) {
+ if (my $s = NDB::Net::ServerINET->get($id)) {
+ push(@$list, $s);
+ next;
+ }
+ if (my $s = NDB::Net::ServerINET->match($id, undef, $servers)) {
+ if (@$s) {
+ push(@$list, @$s);
+ next;
+ }
+ }
+ $log->push;
+ return undef;
+ }
+ }
+ my $fail = 0;
+ for my $s (@$list) {
+ if (! $s->request($cmd)) {
+ $log->push->user;
+ $fail++;
+ }
+ }
+ if ($fail) {
+ $log->put("failed %d/%d", $fail, scalar(@$list));
+ return undef;
+ }
+ return 1;
+ }
+ if ($action eq 'restart') {
+ return 1;
+ }
+ if ($action eq 'stop') {
+ return 1;
+ }
+ if ($action eq 'ping') {
+ $log->put("is alive")->push($server)->user;
+ return 1;
+ }
+ $log->put("$action: unimplemented");
+ return undef;
+}
+
+sub cmd_start_bg {
+ my($cmd) = @_;
+ loadnetcfg() or return undef;
+ my $db = $netcfg->getdatabase($cmd->getarg(0)) or return undef;
+ $db->start($cmd->getopts) or return undef;
+ return 1;
+}
+
+sub cmd_startnode_bg {
+ my($cmd) = @_;
+ loadnetcfg() or return undef;
+ my $db = $netcfg->getdatabase($cmd->getarg(0)) or return undef;
+ my $node = $db->getnode($cmd->getarg(1)) or return undef;
+ $node->start($cmd->getopts) or return undef;
+ return 1;
+}
+
+sub cmd_stop_bg {
+ my($cmd) = @_;
+ my $db = $netcfg->getdatabase($cmd->getarg(0)) or return undef;
+ $db->stop($cmd->getopts) or return undef;
+ return 1;
+}
+
+sub cmd_stopnode_bg {
+ my($cmd) = @_;
+ my $db = $netcfg->getdatabase($cmd->getarg(0)) or return undef;
+ my $node = $db->getnode($cmd->getarg(1)) or return undef;
+ $node->stop($cmd->getopts) or return undef;
+ return 1;
+}
+
+sub cmd_kill_bg {
+ my($cmd) = @_;
+ my $db = $netcfg->getdatabase($cmd->getarg(0)) or return undef;
+ $db->kill($cmd->getopts) or return undef;
+ return 1;
+}
+
+sub cmd_killnode_bg {
+ my($cmd) = @_;
+ my $db = $netcfg->getdatabase($cmd->getarg(0)) or return undef;
+ my $node = $db->getnode($cmd->getarg(1)) or return undef;
+ $node->kill($cmd->getopts) or return undef;
+ return 1;
+}
+
+sub cmd_statnode_bg {
+ my($cmd) = @_;
+ my $db = $netcfg->getdatabase($cmd->getarg(0)) or return undef;
+ my $node = $db->getnode($cmd->getarg(1)) or return undef;
+ my $ret = $node->stat($cmd->getopts) or return undef;
+ return $ret;
+}
+
+sub cmd_list_bg {
+ my($cmd) = @_;
+ loadnetcfg() or return undef;
+ my $dblist;
+ if ($cmd->getarg(0)) {
+ my $db = $netcfg->getdatabase($cmd->getarg(0)) or return undef;
+ $dblist = [ $db ];
+ } else {
+ $dblist = $netcfg->getdatabases or return undef;
+ }
+ my $ret = {};
+ for my $db (@$dblist) {
+ my $status = $db->list($cmd->getopts) || "error";
+ $ret->{$db->getname} = $status;
+ }
+ return $ret;
+}
+
+sub cmd_writenode_bg {
+ my($cmd) = @_;
+ my $db = $netcfg->getdatabase($cmd->getarg(0)) or return undef;
+ my $node = $db->getnode($cmd->getarg(1)) or return undef;
+ my $ret = $node->write($cmd->getopts, $cmd->getarg(2)) or return undef;
+ return $ret;
+}
+
+# main program
+
+sub checkchild {
+ while ((my $pid = waitpid(-1, &POSIX::WNOHANG)) > 0) {
+ $log->put("harvested pid=$pid")->info;
+ }
+}
+
+my $gotterm = 0;
+$SIG{INT} = sub { $gotterm = 1 };
+$SIG{TERM} = sub { $gotterm = 1 };
+
+$socket->setopt(SOL_SOCKET, SO_REUSEADDR, 1) or $log->fatal;
+$socket->bind($server->getport) or $log->fatal;
+$socket->listen or $log->fatal;
+$event->set($socket, 'r');
+
+loop: {
+ try: {
+ my $n = $event->poll(10);
+ if ($gotterm) {
+ $log->put("terminate on signal")->info;
+ last try;
+ }
+ if (! defined($n)) {
+ $log->error;
+ sleep 1;
+ last try;
+ }
+ if (! $n) {
+ $log->debug;
+ last try;
+ }
+ if (! $event->test($socket, 'r')) {
+ last try;
+ }
+ my $csocket = $socket->accept(10);
+ if (! defined($csocket)) {
+ $log->error;
+ last try;
+ }
+ if (! $csocket) {
+ $log->warn;
+ last try;
+ }
+ my $client = NDB::Net::Client->new(
+ socket => $csocket,
+ serversocket => $socket,
+ serverlock => $lock,
+ event => $event,
+ context => 'main',
+ );
+ $client or $log->fatal;
+ }
+ loadnetcfg() or $log->fatal;
+ NDB::Net::Client->processall;
+ if ($gotterm) {
+ last loop;
+ }
+ redo loop;
+}
+
+$log->put("ndbnetd done")->info;
+
+1;
+# vim:set sw=4:
diff --git a/ndb/tools/ndbnet/ndbrun b/ndb/tools/ndbnet/ndbrun
new file mode 100644
index 00000000000..99121276d99
--- /dev/null
+++ b/ndb/tools/ndbnet/ndbrun
@@ -0,0 +1,33 @@
+#! /bin/sh
+
+# just for autotest for now
+
+case $# in
+1) script=$1 ;;
+*) echo "usage: $0 script"; exit 1 ;;
+esac
+
+case $NDB_TOP in
+/*) ;;
+*) echo "$0: NDB_TOP not defined" >&2; exit 1 ;;
+esac
+
+case $script in
+/*) ;;
+*) for d in $NDB_TOP $NDB_TOP/test $NDB_TOP/test/ndbnet; do
+ if [ -f $d/$script ]; then
+ script=$d/$script
+ break
+ fi
+ done ;;
+esac
+
+if [ ! -f $script ]; then
+ echo "$0: $script: script not found" >&2; exit 1
+fi
+
+PERL5LIB=$NDB_TOP/lib/perl5:$PERL5LIB; export PERL5LIB
+
+perl -cw $script || exit 1
+perl $script
+exit $?
diff --git a/ndb/tools/ndbsql/Makefile b/ndb/tools/ndbsql/Makefile
new file mode 100644
index 00000000000..81ca87b0414
--- /dev/null
+++ b/ndb/tools/ndbsql/Makefile
@@ -0,0 +1,44 @@
+include .defs.mk
+
+TYPE := util
+
+BIN_TARGET := ndbsql
+
+#
+# If BIN_TARGET_LIBS include NDB_ODBC then the ODBC lib is
+# linked into the program and the user does not need to
+# set up any ODBC stuff to make it work.
+#
+# If you want to use this program together with some
+# other DBMS (e.g. MySQL or Oracle), then comment the line below.
+#
+BIN_TARGET_LIBS = NDB_ODBC
+
+#BIN_TARGET_ARCHIVES := mgmapi NDB_API
+
+ifneq ($(USE_EDITLINE), N)
+BIN_TARGET_ARCHIVES += editline
+#DIRS := mkconfig
+endif
+
+BIN_FLAGS += $(TERMCAP_LIB)
+
+#ifneq ($(USE_TERMCAP), N)
+#LDFLAGS_LOC = -ltermcap
+#endif
+
+
+# Source files of non-templated classes (.cpp files)
+SOURCES = \
+ ndbsql.cpp
+
+CCFLAGS_LOC += -I$(call fixpath,$(NDB_TOP)/src/ndbapi) \
+ -I$(call fixpath,$(NDB_TOP)/include/mgmapi) \
+ -I$(call fixpath,$(NDB_TOP)/include/util) \
+ -I$(call fixpath,$(NDB_TOP)/src/common/mgmcommon)
+
+
+include $(NDB_TOP)/Epilogue.mk
+
+_bins_mkconfig : $(NDB_TOP)/bin/$(BIN_TARGET)
+
diff --git a/ndb/tools/ndbsql/ndbsql.cpp b/ndb/tools/ndbsql/ndbsql.cpp
new file mode 100644
index 00000000000..ce73a972f47
--- /dev/null
+++ b/ndb/tools/ndbsql/ndbsql.cpp
@@ -0,0 +1,947 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+/*******************************************************************************
+ * NDB Cluster NDB SQL -- A simple SQL Command-line Interface
+ *
+ ******************************************************************************/
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <string.h>
+#ifdef NDB_MACOSX
+#include <stdlib.h>
+#else
+#include <malloc.h>
+#endif
+#include <errno.h>
+#include <editline/editline.h>
+#include <NdbOut.hpp>
+#include <ctype.h>
+#include <wctype.h>
+
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: Readline and string handling
+ * ------------------------------------------------------------------------
+ **************************************************************************/
+#define MAXBUF 2048
+static char* s_readBuf;
+static int s_bufSize = MAXBUF;
+
+static char*
+readSQL_File(FILE* inputFile)
+{
+ int c;
+ int i = 0;
+ if (feof(inputFile))
+ return 0;
+ while ((c = getc(inputFile)) != EOF) {
+ if (i == s_bufSize-1) {
+ s_bufSize *= 2;
+ s_readBuf = (char*)realloc(s_readBuf, s_bufSize);
+ }
+ s_readBuf[i] = c;
+ if (c == '\n')
+ break;
+ i++;
+ }
+ s_readBuf[i] = 0;
+ return s_readBuf;
+}
+
+static char*
+readline_gets(const char* prompt, bool batchMode, FILE* inputFile)
+{
+ static char *line_read = (char *)NULL;
+
+ // Disable the default file-name completion action of TAB
+ // rl_bind_key ('\t', rl_insert);
+
+ if (batchMode)
+ /* Read one line from a file. */
+ line_read = readSQL_File(inputFile);
+ else
+ /* Get a line from the user. */
+ line_read = readline(prompt);
+
+ /* If the line has any text in it, save it in the history. */
+ if (!batchMode)
+ if (line_read && *line_read) add_history(line_read);
+
+ return (line_read);
+}
+
+#ifdef NDB_WIN32
+extern "C"
+{
+ char* readline(const char* prompt)
+ {
+ fputs(prompt, stdout);
+ return fgets(s_readBuf, MAXBUF, stdin);
+ }
+ void add_history(char*)
+ {
+ }
+}
+#endif
+
+bool emptyString(const char* s) {
+ if (s == NULL) {
+ return true;
+ }
+
+ for (unsigned int i = 0; i < strlen(s); ++i) {
+ if (! isspace(s[i])) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: ODBC Handling
+ * ------------------------------------------------------------------------
+ **************************************************************************/
+
+#include <sqlext.h>
+#include <stdio.h>
+#include <string.h>
+#ifdef NDB_MACOSX
+#include <stdlib.h>
+#else
+#include <malloc.h>
+#endif
+/**
+ * In the case where the user types a SELECT statement,
+ * the function fetches and displays all rows of the result set.
+ *
+ * This example illustrates the use of GetDiagField to identify the
+ * type of SQL statement executed and, for SQL statements where the
+ * row count is defined on all implementations, the use of GetDiagField
+ * to obtain the row count.
+ */
+#define MAXCOLS 100
+#define max(a,b) ((a)>(b)?(a):(b))
+
+#define MAX_MESSAGE 500
+
+void getDiag(SQLSMALLINT type, SQLHANDLE handle, unsigned k, unsigned count)
+{
+ char message[MAX_MESSAGE];
+ char state[6];
+ SQLINTEGER native;
+
+ SQLSMALLINT length = -1;
+ memset(message, 0, MAX_MESSAGE);
+ int ret = SQLGetDiagRec(type, handle, k, (SQLCHAR*)state,
+ &native, (SQLCHAR*)message, MAX_MESSAGE, &length);
+ if (ret == SQL_NO_DATA) {
+ ndbout << "No error diagnostics available" << endl;
+ return;
+ }
+ ndbout << message << endl;
+
+ if (k <= count && ret != SQL_SUCCESS)
+ ndbout_c("SQLGetDiagRec %d of %d: return %d != SQL_SUCCESS",
+ k, count, (int)ret);
+ if (k <= count && (SQLSMALLINT) strlen(message) != length)
+ ndbout_c("SQLGetDiagRec %d of %d: message length %d != %d",
+ k, count, strlen(message), length);
+ if (k > count && ret != SQL_NO_DATA)
+ ndbout_c("SQLGetDiagRec %d of %d: return %d != SQL_NO_DATA",
+ k, count, (int)ret);
+}
+
+int print_err(SQLSMALLINT handletype, SQLHDBC hdbc) {
+ getDiag(handletype, hdbc, 1, 1);
+
+ return -1;
+}
+
+
+/***************************************************************
+ * The following functions are given for completeness, but are
+ * not relevant for understanding the database processing
+ * nature of CLI
+ ***************************************************************/
+#define MAX_NUM_PRECISION 15
+/*#define max length of char string representation of no. as:
+= max(precision) + leading sign +E +expsign + max exp length
+= 15 +1 +1 +1 +2
+= 15 +5
+*/
+#define MAX_NUM_STRING_SIZE (MAX_NUM_PRECISION + 5)
+
+int build_indicator_message(SQLCHAR *errmsg, SQLPOINTER *data,
+ SQLINTEGER collen, SQLINTEGER *outlen,
+ SQLSMALLINT colnum) {
+ if (*outlen == SQL_NULL_DATA) {
+ (void)strcpy((char *)data, "NULL");
+ *outlen=4;
+ } else {
+ sprintf((char *)errmsg+strlen((char *)errmsg),
+ "%ld chars truncated, col %d\n", *outlen-collen+1,
+ colnum);
+ *outlen=255;
+ }
+ return 0;
+}
+
+
+SQLINTEGER display_length(SQLSMALLINT coltype, SQLINTEGER collen,
+ SQLCHAR *colname) {
+ switch (coltype) {
+ case SQL_VARCHAR:
+ case SQL_CHAR:
+ //case SQL_BLOB:
+ //case SQL_CLOB:
+ case SQL_BIT:
+ //case SQL_REF:
+ //case SQL_BIT_VARYING:
+ return(max(collen,(SQLINTEGER) strlen((char *)colname))+1);
+ case SQL_FLOAT:
+ case SQL_DOUBLE:
+ case SQL_NUMERIC:
+ case SQL_REAL:
+ case SQL_DECIMAL:
+ return(max(MAX_NUM_STRING_SIZE,strlen((char *)colname))+1);
+ case SQL_TYPE_DATE:
+ case SQL_TYPE_TIME:
+ //case SQL_TYPE_TIME_WITH_TIMEZONE:
+ case SQL_TYPE_TIMESTAMP:
+ //case SQL_TYPE_TIMESTAMP_WITH_TIMEZONE:
+ case SQL_INTERVAL_YEAR:
+ case SQL_INTERVAL_MONTH:
+ case SQL_INTERVAL_DAY:
+ case SQL_INTERVAL_HOUR:
+ case SQL_INTERVAL_MINUTE:
+ case SQL_INTERVAL_SECOND:
+ case SQL_INTERVAL_YEAR_TO_MONTH:
+ case SQL_INTERVAL_DAY_TO_HOUR:
+ case SQL_INTERVAL_DAY_TO_MINUTE:
+ case SQL_INTERVAL_DAY_TO_SECOND:
+ case SQL_INTERVAL_HOUR_TO_MINUTE:
+ case SQL_INTERVAL_HOUR_TO_SECOND:
+ case SQL_INTERVAL_MINUTE_TO_SECOND:
+ return(max(collen,(SQLINTEGER) strlen((char *)colname))+1);
+ case SQL_INTEGER:
+ //case SQL_BLOB_LOCATOR:
+ //case SQL_CLOB_LOCATOR:
+ //case SQL_UDT_LOCATOR:
+ //case SQL_ARRAY_LOCATOR:
+ return(max(11,strlen((char *)colname))+1);
+ case SQL_BIGINT:
+ return(max(21,strlen((char *)colname))+1);
+ case SQL_SMALLINT:
+ return(max(5,strlen((char *)colname))+1);
+ default:
+ (void)printf("Unknown datatype, %d\n", coltype);
+ return(0);
+ }
+}
+
+struct Con {
+ const char* dsn;
+ SQLHENV henv;
+ SQLHDBC hdbc;
+ Con(const char* _dsn) :
+ dsn(_dsn), henv(SQL_NULL_HANDLE), hdbc(SQL_NULL_HANDLE) {}
+};
+
+static int
+do_connect(Con& con)
+{
+ int ret;
+
+ // allocate an environment handle
+ ret = SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &con.henv);
+ if (ret != SQL_SUCCESS)
+ return -1;
+
+ // set odbc version (required)
+ ret = SQLSetEnvAttr(con.henv, SQL_ATTR_ODBC_VERSION, (SQLPOINTER)SQL_OV_ODBC3, 0);
+ if (ret != SQL_SUCCESS)
+ return -1;
+
+ // allocate a connection handle
+ ret = SQLAllocHandle(SQL_HANDLE_DBC, con.henv, &con.hdbc);
+ if (ret != SQL_SUCCESS)
+ return -1;
+
+ // connect to database
+ SQLCHAR szConnStrOut[256];
+ SQLSMALLINT cbConnStrOut;
+ ret = SQLDriverConnect(con.hdbc, 0, (SQLCHAR*)con.dsn, SQL_NTS,
+ szConnStrOut, sizeof(szConnStrOut), &cbConnStrOut, SQL_DRIVER_COMPLETE);
+ if (ret != SQL_SUCCESS) {
+ ndbout << "Connection failure: Could not connect to database" << endl;
+ print_err(SQL_HANDLE_DBC, con.hdbc);
+ return -1;
+ }
+
+ return 0;
+}
+
+static int
+do_disconnect(Con& con)
+{
+ // disconnect from database
+ SQLDisconnect(con.hdbc);
+
+ // free connection handle
+ SQLFreeHandle(SQL_HANDLE_DBC, con.hdbc);
+ con.hdbc = SQL_NULL_HANDLE;
+
+ // free environment handle
+ SQLFreeHandle(SQL_HANDLE_ENV, con.henv);
+ con.henv = SQL_NULL_HANDLE;
+
+ return 0;
+}
+
+static int
+get_autocommit(Con& con)
+{
+ int ret;
+ SQLUINTEGER v;
+ ret = SQLGetConnectAttr(con.hdbc, SQL_ATTR_AUTOCOMMIT, (SQLPOINTER)&v, SQL_IS_UINTEGER, 0);
+ if (ret != SQL_SUCCESS) {
+ ndbout << "Get autocommit failed" << endl;
+ print_err(SQL_HANDLE_DBC, con.hdbc);
+ return -1;
+ }
+ return v;
+}
+
+static int
+set_autocommit(Con& con, SQLUINTEGER v)
+{
+ int ret;
+ ret = SQLSetConnectAttr(con.hdbc, SQL_ATTR_AUTOCOMMIT, (SQLPOINTER)v, SQL_IS_UINTEGER);
+ if (ret != SQL_SUCCESS) {
+ ndbout << "Set autocommit failed" << endl;
+ print_err(SQL_HANDLE_DBC, con.hdbc);
+ return -1;
+ }
+ return 0;
+}
+
+static int
+do_commit(Con& con)
+{
+ int ret = SQLEndTran(SQL_HANDLE_DBC, con.hdbc, SQL_COMMIT);
+ if (ret != SQL_SUCCESS) {
+ ndbout << "Commit failed" << endl;
+ print_err(SQL_HANDLE_DBC, con.hdbc);
+ return -1;
+ }
+ return 0;
+}
+
+static int
+do_rollback(Con& con)
+{
+ int ret = SQLEndTran(SQL_HANDLE_DBC, con.hdbc, SQL_ROLLBACK);
+ if (ret != SQL_SUCCESS) {
+ ndbout << "Rollback failed" << endl;
+ print_err(SQL_HANDLE_DBC, con.hdbc);
+ return -1;
+ }
+ return 0;
+}
+
+static int
+do_stmt(Con& con, const char *sqlstr)
+{
+ SQLHSTMT hstmt;
+ SQLCHAR errmsg[256];
+ SQLCHAR colname[32];
+ SQLSMALLINT coltype;
+ SQLSMALLINT colnamelen;
+ SQLSMALLINT nullable;
+ SQLUINTEGER collen[MAXCOLS];
+ SQLSMALLINT scale;
+ SQLINTEGER outlen[MAXCOLS];
+ SQLCHAR *data[MAXCOLS];
+ SQLSMALLINT nresultcols = 0;
+ SQLINTEGER rowcount;
+ SQLINTEGER stmttype;
+ SQLRETURN rc;
+
+ /* allocate a statement handle */
+ SQLAllocHandle(SQL_HANDLE_STMT, con.hdbc, &hstmt);
+
+ /* execute the SQL statement */
+ rc = SQLExecDirect(hstmt, (SQLCHAR*)sqlstr, SQL_NTS);
+ if (rc == SQL_ERROR) {
+ ndbout << "Operation failed" << endl;
+ print_err(SQL_HANDLE_STMT, hstmt);
+ return -1;
+ }
+ if (rc != SQL_SUCCESS && rc != SQL_SUCCESS_WITH_INFO && rc != SQL_NO_DATA_FOUND) {
+ ndbout << "Operation returned unknown code " << rc << endl;
+ return -1;
+ }
+
+ /* see what kind of statement it was */
+ SQLGetDiagField(SQL_HANDLE_STMT, hstmt, 0,
+ SQL_DIAG_DYNAMIC_FUNCTION_CODE,
+ (SQLPOINTER)&stmttype, SQL_IS_INTEGER, (SQLSMALLINT *)NULL);
+
+ switch (stmttype) {
+ /* SELECT statement */
+ case SQL_DIAG_SELECT_CURSOR:
+ /* determine number of result columns */
+ SQLNumResultCols(hstmt, &nresultcols);
+
+ /***********************
+ * Display column names
+ ***********************/
+ /* Print vertical divider */
+ printf("|");
+ for (int i=0; i<nresultcols; i++) {
+ SQLDescribeCol(hstmt, i+1, colname, sizeof(colname),
+ &colnamelen, &coltype, &collen[i], &scale, &nullable);
+ collen[i] = display_length(coltype, collen[i], colname);
+ for (SQLUINTEGER j=0; j<collen[i]; j++) printf("-");
+ printf("--+");
+ }
+ printf("\n");
+
+ printf("|");
+ for (int i=0; i<nresultcols; i++) {
+ SQLDescribeCol(hstmt, i+1, colname, sizeof(colname),
+ &colnamelen, &coltype, &collen[i], &scale, &nullable);
+
+ /* assume there is a display_length function which
+ computes correct length given the data type */
+ collen[i] = display_length(coltype, collen[i], colname);
+ (void)printf(" %*.*s |", (int)collen[i], (int)collen[i], (char *)colname);
+
+ /* allocate memory to bind column */
+ data[i] = (SQLCHAR *) malloc(collen[i]);
+ if (data[i] == NULL) {
+ ndbout << "Failed to allocate malloc memory in NDB SQL program"
+ << endl;
+ exit(-1);
+ }
+
+ /* bind columns to program vars, converting all types to CHAR */
+ SQLBindCol(hstmt, i+1, SQL_C_CHAR, data[i], collen[i], &outlen[i]);
+ }
+ printf("\n");
+
+ /* Print vertical divider */
+ printf("|");
+ for (int i=0; i<nresultcols; i++) {
+ SQLDescribeCol(hstmt, i+1, colname, sizeof(colname),
+ &colnamelen, &coltype, &collen[i], &scale, &nullable);
+ collen[i] = display_length(coltype, collen[i], colname);
+ for (SQLUINTEGER j=0; j<collen[i]; j++) printf("-");
+ printf("--+");
+ }
+ printf("\n");
+
+ /**********************
+ * Display result rows
+ **********************/
+ {
+ int no_of_rows_fetched=0;
+ while (1) {
+ rc=SQLFetch(hstmt);
+ errmsg[0] = '\0';
+ if (rc == SQL_ERROR) {
+ print_err(SQL_HANDLE_STMT, hstmt);
+ break;
+ }
+ if (rc == SQL_NO_DATA) break;
+ if (rc == SQL_SUCCESS) {
+ printf("|");
+ for (int i=0; i<nresultcols; i++) {
+ if (outlen[i] == SQL_NULL_DATA
+ || outlen[i] >= (SQLINTEGER) collen[i])
+ build_indicator_message(errmsg,
+ (SQLPOINTER *)data[i], collen[i],
+ &outlen[i], i);
+ (void)printf(" %*.*s |", (int)collen[i], (int)collen[i],
+ (char *)data[i]);
+ }
+ /* print any truncation messages */
+ (void)printf("\n%s", (char *)errmsg);
+ } else if (rc == SQL_SUCCESS_WITH_INFO) {
+ printf("|");
+ for (int i=0; i<nresultcols; i++) {
+ if (outlen[i] == SQL_NULL_DATA
+ || outlen[i] >= (SQLINTEGER) collen[i])
+ build_indicator_message(errmsg,
+ (SQLPOINTER *)data[i], collen[i],
+ &outlen[i], i);
+ (void)printf(" %*.*s |", (int)collen[i], (int)collen[i],
+ (char *)data[i]);
+ } /* for all columns in this row */
+ /* print any truncation messages */
+ (void)printf("\n%s", (char *)errmsg);
+ }
+ no_of_rows_fetched++;
+ } /* while rows to fetch */
+ /* Print vertical divider */
+ printf("|");
+ for (int i=0; i<nresultcols; i++) {
+ SQLDescribeCol(hstmt, i+1, colname, sizeof(colname),
+ &colnamelen, &coltype, &collen[i], &scale, &nullable);
+ collen[i] = display_length(coltype, collen[i], colname);
+ for (SQLUINTEGER j=0; j<collen[i]; j++) printf("-");
+ printf("--+");
+ }
+ printf("\n");
+ ndbout << no_of_rows_fetched << " rows fetched" << endl;
+ }
+ SQLCloseCursor(hstmt);
+ break;
+ /* searched DELETE, INSERT or searched UPDATE statement */
+ case SQL_DIAG_DELETE_WHERE:
+ case SQL_DIAG_INSERT:
+ case SQL_DIAG_UPDATE_WHERE:
+ /* check rowcount */
+ SQLRowCount(hstmt, (SQLINTEGER*)&rowcount);
+ ndbout << (int)rowcount << " rows affected" << endl;
+ break;
+ /* other statements */
+ case SQL_DIAG_ALTER_TABLE:
+ case SQL_DIAG_CREATE_TABLE:
+ case SQL_DIAG_CREATE_VIEW:
+ case SQL_DIAG_DROP_TABLE:
+ case SQL_DIAG_DROP_VIEW:
+ case SQL_DIAG_CREATE_INDEX:
+ case SQL_DIAG_DROP_INDEX:
+ case SQL_DIAG_DYNAMIC_DELETE_CURSOR:
+ case SQL_DIAG_DYNAMIC_UPDATE_CURSOR:
+ case SQL_DIAG_GRANT:
+ case SQL_DIAG_REVOKE:
+ ndbout << "Operation successful" << endl;
+ break;
+ /* implementation-defined statement */
+ default:
+ (void)printf("Unknown Statement type=%ld\n", stmttype);
+ break;
+ }
+
+ /* free data buffers */
+ for (int i=0; i<nresultcols; i++) {
+ (void)free(data[i]);
+ }
+
+ SQLFreeHandle(SQL_HANDLE_STMT, hstmt); // free statement handle
+ return(0);
+}
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: Help
+ * ------------------------------------------------------------------------
+ **************************************************************************/
+
+void print_help() {
+ ndbout << "Commands:" << endl
+ << "set Print currect settings" << endl
+ << "set trace N Set NDB ODBC trace level to N (0-5)" << endl
+ << "set autocommit on Commit each statement (default)" << endl
+ << "set autocommit off Use explicit commit/rollback - may time out!" << endl
+ << "commit Commit changes to database" << endl
+ << "rollback Rollback (undo) any changes" << endl
+ << "whenever sqlerror Define action: exit or continue (default)" << endl
+ << endl
+ << "help Print this help" << endl
+ << "help create Print create table examples" << endl
+ << "help insert Print insert examples" << endl
+ << "help select Print select examples" << endl
+ << "help delete Print delete examples" << endl
+ << "help update Print update examples" << endl
+ << "help virtual Print help on NDB ODBC virtual tables" << endl
+ << "list tables Lists all table names" << endl
+ << endl
+ << "All other commands are sent to the NDB ODBC SQL executor"
+ << endl << endl;
+}
+
+void print_help_create() {
+ ndbout << "Create Table Examples" << endl << endl
+ << "create table t ( a integer not null, b char(20) not null," << endl
+ << " c float, primary key(a, b) )" << endl
+ << "create table t ( ndb$tid bigint unsigned primary key," << endl
+ << " b char(20) not null, c float )" << endl
+ << "create table t ( a int auto_increment primary key," << endl
+ << " b char(20) not null, c float )" << endl
+ << "create table t ( a int primary key," << endl
+ << " b int default 100 )" << endl
+ << endl
+ << "For more information read NDB Cluster ODBC Manual."
+ << endl;
+}
+
+void print_help_insert() {
+ ndbout << "Insert Examples" << endl << endl
+ << "insert into t(a, c) values (123, 'abc')" << endl
+ << "insert into t1(a, c) select a + 10 * b, c from t2" << endl
+ << "insert into t values(null, 'abc', 1.23)" << endl
+ << "insert into t(b, c) values('abc', 1.23)" << endl
+ << endl
+ << "For more information read NDB Cluster ODBC Manual."
+ << endl;
+}
+
+void print_help_select() {
+ ndbout << "Select Examples" << endl << endl
+ << "select a + b * c from t where a <= b + c and (b > c or c > 10)"
+ << endl
+ << "select a.x, b.y, c.z from t1 a, t2 b, t2 c where a.x + b.y < c.z"
+ << endl
+ << "select * from t1, t2 where a1 > 5 order by b1 + b2, c1 desc"
+ << endl
+ << "select count(*), max(a), 1 + sum(b) + avg(c * d) from t" << endl
+ << "select * from t where a < 10 or b > 10" << endl
+ << "select * from t where pk = 5 and b > 10" << endl
+ << "select * from t1, t2, t3 where t1.pk = t2.x and t2.pk = t3.y"
+ << endl << endl
+ << "For more information read NDB Cluster ODBC Manual."
+ << endl;
+}
+
+void print_help_update() {
+ ndbout << "Update and Delete Examples" << endl << endl
+ << "update t set a = b + 5, c = d where c > 10" << endl
+ << "update t set a = b + 5, c = d where pk = 5 and c > 10" << endl
+ << "update t set a = 5, c = 7 where pk = 5" << endl
+ << "delete from t where c > 10" << endl
+ << "delete from t where pk = 5 and c > 10" << endl
+ << "delete from t where pk = 5" << endl
+ << endl
+ << "For more information read NDB Cluster ODBC Manual."
+ << endl;
+}
+
+void print_help_virtual() {
+ ndbout << "Virtual tables" << endl << endl
+ << "* DUAL"
+ << " a 1-row table - example: select SYSDATE from DUAL" << endl
+ << "* ODBC$TYPEINFO" << endl
+ << " corresponds to SQLGetTypeInfo" << endl
+ << "* ODBC$TABLES" << endl
+ << " corresponds to SQLTables (ordered by NDB table id)" << endl
+ << "* ODBC$COLUMNS" << endl
+ << " corresponds to SQLColumns (ordered by NDB table id)" << endl
+ << "* ODBC$PRIMARYKEYS" << endl
+ << " corresponds to SQLPrimaryKeys (ordered by NDB table id)" << endl
+ << endl
+ << "For more information read NDB Cluster ODBC Manual."
+ << endl;
+}
+
+/**************************************************************************
+ * ------------------------------------------------------------------------
+ * MODULE: Main
+ * ------------------------------------------------------------------------
+ **************************************************************************/
+
+int main(int argc, const char** argv)
+{
+ const char* usage = "Usage: ndbsql [-h] [-d dsn] [-f file] [stmt]\n-h help\n-d <database name or connect string>\n-f <file name> batch mode\nstmt single SQL statement\n";
+ const char* dsn = "TEST_DB";
+ bool helpFlg = false, batchMode = false;
+ const char* fileName = 0;
+ FILE* inputFile = stdin;
+ const char* singleStmt = 0;
+
+ s_readBuf = (char*)malloc(s_bufSize);
+ while (++argv, --argc > 0) {
+ const char* arg = argv[0];
+ if (arg[0] != '-')
+ break;
+ if (strcmp(arg, "-d") == 0) {
+ if (++argv, --argc > 0) {
+ dsn = argv[0];
+ continue;
+ }
+ }
+ if (strcmp(arg, "-h") == 0) {
+ helpFlg = true;
+ continue;
+ }
+ if (strcmp(arg, "-f") == 0) {
+ if (++argv, --argc > 0) {
+ fileName = argv[0];
+ continue;
+ }
+ }
+ ndbout << usage;
+ return 1;
+ }
+ if (helpFlg) {
+ ndbout << usage << "\n";
+ print_help();
+ return 0;
+ }
+ if (fileName != 0) {
+ if (argc > 0) {
+ ndbout << usage;
+ return 1;
+ }
+ if ((inputFile = fopen(fileName, "r")) == 0) {
+ ndbout << "Could not read file " << fileName << ": " << strerror(errno) << endl;
+ return 1;
+ }
+ batchMode = true;
+ }
+ if (argc > 0) {
+ singleStmt = argv[0];
+ batchMode = true;
+ }
+ if (! batchMode)
+ ndbout << "NDB Cluster NDB SQL -- A simple SQL Command-line Interface\n\n";
+
+ Con con(dsn);
+ if (do_connect(con) < 0)
+ return 1;
+ if (! batchMode)
+ ndbout << "Terminate SQL statements with a semi-colon ';'\n";
+
+ char* line = 0;
+ char* line2 = 0;
+ char* line3 = 0;
+ unsigned lineno = 0;
+ bool has_semi;
+ bool exit_on_error = false;
+ int exit_code = 0;
+ while (1) {
+ free(line);
+ line = 0;
+ lineno = 0;
+
+more_lines:
+ free(line2);
+ free(line3);
+ line2 = line3 = 0;
+ lineno++;
+ has_semi = false;
+ char prompt[20];
+ if (lineno == 1)
+ strcpy(prompt, "SQL> ");
+ else
+ sprintf(prompt, "%4d ", lineno);
+ if (singleStmt != 0) {
+ line = strdup(singleStmt);
+ int n = strlen(line);
+ while (n > 0 && isspace(line[n - 1])) {
+ line[--n] = 0;
+ }
+ if (n > 0 && line[n - 1] == ';')
+ line[n - 1] = 0;
+ has_semi = true; // regardless
+ } else {
+ const char *line1 = readline_gets(prompt, batchMode, inputFile);
+ if (line1 != 0) {
+ if (line == 0)
+ line = strdup(line1);
+ else {
+ line = (char*)realloc(line, strlen(line) + 1 + strlen(line1) + 1);
+ strcat(line, "\n");
+ strcat(line, line1);
+ }
+ if (batchMode)
+ ndbout << prompt << line1 << endl;
+ } else {
+ if (! batchMode)
+ ndbout << endl;
+ if (line != 0)
+ ndbout << "Ignored unterminated SQL statement" << endl;
+ break;
+ }
+ }
+
+ line2 = (char*)malloc(strlen(line) + 1);
+ {
+ char* p = line2;
+ char* q = line;
+ bool str = false;
+ while (*q != 0) {
+ if (*q == '\'') {
+ str = !str;
+ *p++ = *q++;
+ } else if (!str && *q == '-' && *(q + 1) == '-') {
+ while (*q != 0 && *q != '\n')
+ q++;
+ } else
+ *p++ = *q++;
+ }
+ *p = 0;
+ int n = strlen(line2);
+ while (n > 0 && isspace(line2[n - 1]))
+ line2[--n] = 0;
+ if (n > 0 && line2[n - 1] == ';') {
+ line2[--n] = 0;
+ has_semi = true;
+ }
+ }
+ line3 = strdup(line2);
+ char* tok[10];
+ int ntok = 0;
+ tok[ntok] = strtok(line3, " ");
+ while (tok[ntok] != 0) {
+ ntok++;
+ if (ntok == 10)
+ break;
+ tok[ntok] = strtok(0, " ");
+ }
+ if (ntok == 0)
+ continue;
+
+ if (!strcasecmp(tok[0], "help") || !strcmp(tok[0], "?")) {
+ if (ntok != 2)
+ print_help();
+ else if (!strcasecmp(tok[1], "create"))
+ print_help_create();
+ else if (!strcasecmp(tok[1], "insert"))
+ print_help_insert();
+ else if (strcasecmp(tok[1], "select"))
+ print_help_select();
+ else if (!strcasecmp(tok[1], "delete"))
+ print_help_update();
+ else if (!strcasecmp(tok[1], "update"))
+ print_help_update();
+ else if (!strcasecmp(tok[1], "virtual"))
+ print_help_virtual();
+ else
+ print_help();
+ continue;
+ }
+
+ if (!strcasecmp(tok[0], "list")) {
+ if (ntok == 2 && !strcasecmp(tok[1], "tables")) {
+ free(line2);
+ line2 = strdup("SELECT TABLE_NAME FROM ODBC$TABLES");
+ has_semi = true;
+ } else {
+ ndbout << "Invalid list option - try help" << endl;
+ continue;
+ }
+ }
+
+ if (ntok == 1 && !strcasecmp(tok[0], "quit"))
+ break;
+ if (ntok == 1 && !strcasecmp(tok[0], "exit"))
+ break;
+ if (ntok == 1 && !strcasecmp(tok[0], "bye"))
+ break;
+
+ if (!strcasecmp(tok[0], "set")) {
+ if (ntok == 1) {
+ char* p;
+ p = getenv("NDB_ODBC_TRACE");
+ ndbout << "Trace level is " << (p ? atoi(p) : 0) << endl;
+ int ret = get_autocommit(con);
+ if (ret != -1)
+ ndbout << "Autocommit is " << (ret == SQL_AUTOCOMMIT_ON ? "on" : "off") << endl;
+ } else if (ntok == 3 && !strcasecmp(tok[1], "trace")) {
+ static char env[40];
+ int n = tok[2] ? atoi(tok[2]) : 0;
+ sprintf(env, "NDB_ODBC_TRACE=%d", n);
+ putenv(env);
+ ndbout << "Trace level set to " << n << endl;
+ } else if (ntok == 3 && !strcasecmp(tok[1], "autocommit")) {
+ if (tok[2] && !strcasecmp(tok[2], "on")) {
+ int ret = set_autocommit(con, SQL_AUTOCOMMIT_ON);
+ if (ret != -1)
+ ndbout << "Autocommit set to ON" << endl;
+ } else if (tok[2] && !strcasecmp(tok[2], "off")) {
+ int ret = set_autocommit(con, SQL_AUTOCOMMIT_OFF);
+ if (ret != -1)
+ ndbout << "Autocommit set to OFF - transaction may time out" << endl;
+ } else {
+ ndbout << "Invalid autocommit option - try help" << endl;
+ }
+ } else {
+ ndbout << "Invalid set command - try help" << endl;
+ }
+ continue;
+ }
+
+ if (ntok >= 2 &&
+ !strcasecmp(tok[0], "whenever") && !strcasecmp(tok[1], "sqlerror")) {
+ if (ntok == 3 && !strcasecmp(tok[2], "exit"))
+ exit_on_error = true;
+ else if (ntok == 3 && !strcasecmp(tok[2], "continue"))
+ exit_on_error = false;
+ else {
+ ndbout << "Invalid whenever clause - try help" << endl;
+ }
+ continue;
+ }
+
+ if (!strcasecmp(tok[0], "commit")) {
+ if (ntok == 1) {
+ if (do_commit(con) != -1)
+ ndbout << "Commit done" << endl;
+ else {
+ exit_code = 1;
+ if (exit_on_error) {
+ ndbout << "Exit on error" << endl;
+ break;
+ }
+ }
+ } else {
+ ndbout << "Invalid commit command - try help" << endl;
+ }
+ continue;
+ }
+
+ if (!strcasecmp(tok[0], "rollback")) {
+ if (ntok == 1) {
+ if (do_rollback(con) != -1)
+ ndbout << "Rollback done" << endl;
+ else {
+ exit_code = 1;
+ if (exit_on_error) {
+ ndbout << "Exit on error" << endl;
+ break;
+ }
+ }
+ } else {
+ ndbout << "Invalid commit command - try help" << endl;
+ }
+ continue;
+ }
+
+ if (! has_semi)
+ goto more_lines;
+ if (do_stmt(con, line2) != 0) {
+ exit_code = 1;
+ if (exit_on_error) {
+ ndbout << "Exit on error" << endl;
+ break;
+ }
+ }
+ if (singleStmt)
+ break;
+ }
+ do_disconnect(con);
+ return exit_code;
+}
+
+// vim: set sw=2 et:
diff --git a/ndb/tools/rgrep b/ndb/tools/rgrep
new file mode 100755
index 00000000000..212b068639d
--- /dev/null
+++ b/ndb/tools/rgrep
@@ -0,0 +1,194 @@
+#!/vobs/wds/swt/bin/perl
+
+die "Usage: rgrep [-iredblLn] regexp filepat ...\n rgrep -h for help\n"
+ if $#ARGV < $[;
+
+# Written by Piet van Oostrum <piet@cs.ruu.nl>
+# This is really free software
+
+# Mats Lidell added support for gzip.
+# Mats Lidell added support for skipping line numbers.
+
+$nextopt = 1;
+$igncase = '';
+$regpat = 0;
+$links = 0;
+$error = 0;
+$skipbin = 1;
+$havenl = 1;
+$debug = 0;
+
+do { $regexp = shift (@ARGV); } while &checkopt ($regexp);
+$icreg = $igncase;
+$igncase = '';
+
+eval 'sub grep_file {
+ while (<F>) {
+ $ln++;
+ if (/$regexp/o' . $icreg .') {
+ print "$file:$ln:" if $havenl;
+ print "$_";
+ print "\n" if substr($_, -1, 1) ne "\n";
+ }
+ }
+}';
+
+for (@ARGV) {
+ if (! &checkopt ($_)) {
+ if ($igncase || $regpat || /[?*[]/ || ! -e) {
+ if ($regpat) {
+ s/#/\\#/g;
+ $_ = "#$_#";
+ } else { # translate File pattern into regexp
+ $re = '#($|/)'; $save = $_;
+ while (/[[*?+()|.^$#]/) {
+ $re .= $`;
+ $c = $&;
+ $_ = $';
+ if ($c eq '*') { $c = '[^/]*'; }
+ elsif ($c eq '?') { $c = '[^/]'; }
+ elsif ($c eq '[') {
+ if (/.\]/) { $c = "[$`$&"; $_ = $'; }
+ else {
+ $error++;
+ printf stderr "Illegal filepattern %s\n", $save;
+ }
+ } else { $c = "\\$c"; }
+ $re .= $c;
+ }
+ $_ = "$re$_\$#$igncase";
+ }
+ print "filepat: $_\n" if $debug;
+ push (@filepat, $_);
+ }
+ else { push (@files, $_); print "file: $_\n" if $debug; }
+ }
+}
+
+exit 1 if $errors ;
+
+if ($#filepat < $[) {
+ eval "sub in_pat {1;}" ;
+}
+else {
+ $subtxt = 'sub in_pat { local ($f) = @_;';
+ $or = "";
+ for (@filepat) {
+ $subtxt .= $or . '$f =~ m' . $_;
+ $or = " || ";
+ }
+ $subtxt .= ';};1';
+
+ if (! eval $subtxt) {
+ print $@;
+ exit 1;
+ }
+}
+
+@files = (".") if $#files < $[;
+
+for $file (@files) {
+ &do_grep ($file);
+}
+
+sub do_grep {
+ local ($file) = @_;
+ local (*F, $ln, $f, $g, @dirfiles);
+ if (-f $file) {
+ if (open (F, $file)) {
+ if (-B F) { # binary file -- may be compressed/compacted/gziped
+ if (($cx1 = getc(F)) eq "\377" && (getc(F) eq "\037")) {
+ open (F, "uncompact < $file|");
+ if ($skipbin && -B F) { close (F); return; }
+ }
+ elsif ($cx1 eq "\037" && (($cx2 = getc(F)) eq "\235")) {
+ open (F, "uncompress < $file|");
+ if ($skipbin && -B F) { close (F); return; }
+ }
+ elsif ($cx1 eq "\037" && $cx2 eq "\213") {
+ open (F, "gzip -d < $file|");
+ if ($skipbin && -B F) { close (F); return; }
+ }
+ elsif ($skipbin) {
+ close (F); return;
+ }
+ }
+ print "Reading $file\n" if $debug;
+ &grep_file;
+ } else {
+ print stderr "Cannot open $file\n";
+ }
+ }
+ elsif (-d $file) {
+ print "Entering $file\n" if $debug;
+ if (opendir (F, $file)) {
+ @dirfiles = readdir (F);
+ closedir (F);
+ for $f (@dirfiles) {
+ next if ($f eq '.' || $f eq '..');
+ $g = "$file/$f";
+ next if (-l $g && ($links < 1 || $links == 1 && -d $g));
+ if (-f $g && &in_pat ($g) || -d _) {
+ &do_grep ($g);
+ }
+ }
+ } else {
+ print stderr "Can't open $file\n";
+ }
+ }
+}
+
+sub checkopt {
+ local ($_) = $_[0];
+ if (/^-/ && $nextopt) {
+ $nextopt = 1;
+ @opt = split (/-*/,$_); shift (@opt);
+ for $opt (@opt) {
+ if ($opt eq 'i') { $igncase = 'i'; }
+ elsif ($opt eq 'd') { $debug = 1; }
+ elsif ($opt eq 'l') { $links = 1; }
+ elsif ($opt eq 'L') { $links = 2; }
+ elsif ($opt eq 'b') { $skipbin = 0; }
+ elsif ($opt eq 'r') { $regpat = 1; }
+ elsif ($opt eq 'e') { $nextopt = 0; }
+ elsif ($opt eq 'n') { $havenl = 0; }
+ elsif ($opt eq 'h' || $opt eq 'H') { & help; }
+ else { $error++; printf stderr "Unknown option -%s\n", $opt; }
+ }
+ return 1;
+ }
+ $nextopt = 1;
+ return 0;
+}
+
+sub help {
+ print <<'HELP'; exit 0;
+Usage: rgrep [-iredblL] regexp filepat ...
+ regexp = perl regular expression to search
+ filepat ... = a list of files and directories to be searched or
+ file patterns to match filenames.
+ filepat will be interpreted as file or directory name if it exists
+ as such, and does not contain the metacharacters [ ] ? or *. After
+ the options -i and -r all filepats will be considered patterns.
+ rgrep will search all files in any of the directories given (and its
+ subdirectories) that match any of the filepats, except binary files.
+ Compressed files will be searched in uncompressed form.
+ Note: filepats may contain / contrary to find usage.
+ -b Don't skip binary files.
+ -i Ignore case, either in the regexp or in filename matching (depending
+ on the location). Before the regexp only applies to the regexp,
+ otherwise to the filepats following it.
+ -r The following filepats are treated as real perl regexps rather than
+ shell style filename patterns. In this case / is not a special
+ character, i.e. it is matched by . and matching is not anchored (you
+ must supply ^ and $ yourself). E.g. a.b matches the file /xa/by/zz.
+ -l Do follow symbolic links only for files (default is do not follow).
+ -L Do follow symbolic links for files and directories.
+ -e Do not interpret following argument as option. Useful if regexp or
+ filepat starts with a -.
+ -d Debugging: Give a lot of output on what happens.
+ -n Don't precede each line by its relative line number in the file.
+ -h print this message and exit.
+Piet van Oostrum <piet@cs.ruu.nl>
+HELP
+}
diff --git a/ndb/tools/select_all/Makefile b/ndb/tools/select_all/Makefile
new file mode 100644
index 00000000000..e14e411b3a5
--- /dev/null
+++ b/ndb/tools/select_all/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := select_all
+
+SOURCES := select_all.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/tools/select_all/select_all.cpp b/ndb/tools/select_all/select_all.cpp
new file mode 100644
index 00000000000..32e9d1c6872
--- /dev/null
+++ b/ndb/tools/select_all/select_all.cpp
@@ -0,0 +1,286 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+#include <NdbScanFilter.hpp>
+
+
+int scanReadRecords(Ndb*,
+ const NdbDictionary::Table*,
+ int parallel,
+ int lockType,
+ bool headers,
+ bool useHexFormat,
+ char delim);
+
+int main(int argc, const char** argv){
+ int _parallelism = 240;
+ const char* _delimiter = "\t";
+ int _header = true;
+ int _useHexFormat = false;
+ const char* _tabname = NULL;
+ const char* _dbname = "TEST_DB";
+ int _help = 0;
+ int _lock = 0;
+
+ struct getargs args[] = {
+ { "database", 'd', arg_string, &_dbname, "dbname",
+ "Name of database table is in"},
+ { "parallelism", 'p', arg_integer, &_parallelism, "parallelism",
+ "parallelism" },
+ { "header", 'h', arg_flag, &_header, "Print header", "header" },
+ { "useHexFormat", 'x', arg_flag, &_useHexFormat,
+ "Output numbers in hexadecimal format", "useHexFormat" },
+ { "delimiter", 'd', arg_string, &_delimiter, "Column delimiter",
+ "delimiter" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" },
+ { "lock", 'l', arg_integer, &_lock,
+ "Read(0), Read-hold(1), Exclusive(2)", "lock"}
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname\n"\
+ "This program reads all records from one table in NDB Cluster\n"\
+ "and print them to stdout. This is performed using a scan read.\n"\
+ "(It only print error messages if it encounters a permanent error.)\n"\
+ "It can also be used to dump the content of a table to file \n"\
+ " ex: select_all --no-header --delimiter=';' T4 > T4.data\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+
+ // Connect to Ndb
+ Ndb MyNdb(_dbname);
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ // Connect to Ndb and wait for it to become ready
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table* pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ if (scanReadRecords(&MyNdb,
+ pTab,
+ _parallelism,
+ _lock,
+ _header,
+ _useHexFormat,
+ (char)*_delimiter) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+
+}
+
+int scanReadRecords(Ndb* pNdb,
+ const NdbDictionary::Table* pTab,
+ int parallel,
+ int _lock,
+ bool headers,
+ bool useHexFormat,
+ char delimiter){
+
+ int retryAttempt = 0;
+ const int retryMax = 100;
+ int check;
+ NdbConnection *pTrans;
+ NdbOperation *pOp;
+
+ NDBT_ResultRow * row = new NDBT_ResultRow(*pTab, delimiter);
+
+ while (true){
+
+ if (retryAttempt >= retryMax){
+ ndbout << "ERROR: has retried this operation " << retryAttempt
+ << " times, failing!" << endl;
+ return -1;
+ }
+
+ pTrans = pNdb->startTransaction();
+ if (pTrans == NULL) {
+ const NdbError err = pNdb->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ return -1;
+ }
+
+ pOp = pTrans->getNdbOperation(pTab->getName());
+ if (pOp == NULL) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+
+ switch(_lock){
+ case 1:
+ check = pOp->openScanReadHoldLock(parallel);
+ break;
+ case 2:
+ check = pOp->openScanExclusive(parallel);
+ break;
+ default:
+ check = pOp->openScanRead(parallel);
+ }
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+
+ if(0){
+ NdbScanFilter sf(pOp);
+#if 0
+ sf.begin(NdbScanFilter::AND);
+ sf.le(0, (Uint32)10);
+
+ sf.end();
+#elif 0
+ sf.begin(NdbScanFilter::OR);
+ sf.begin(NdbScanFilter::AND);
+ sf.ge(0, (Uint32)10);
+ sf.lt(0, (Uint32)20);
+ sf.end();
+ sf.begin(NdbScanFilter::AND);
+ sf.ge(0, (Uint32)30);
+ sf.lt(0, (Uint32)40);
+ sf.end();
+ sf.end();
+#elif 1
+ sf.begin(NdbScanFilter::AND);
+ sf.begin(NdbScanFilter::OR);
+ sf.begin(NdbScanFilter::AND);
+ sf.ge(0, (Uint32)10);
+ sf.lt(0, (Uint32)20);
+ sf.end();
+ sf.begin(NdbScanFilter::AND);
+ sf.ge(0, (Uint32)30);
+ sf.lt(0, (Uint32)40);
+ sf.end();
+ sf.end();
+ sf.begin(NdbScanFilter::OR);
+ sf.begin(NdbScanFilter::AND);
+ sf.ge(0, (Uint32)0);
+ sf.lt(0, (Uint32)50);
+ sf.end();
+ sf.begin(NdbScanFilter::AND);
+ sf.ge(0, (Uint32)100);
+ sf.lt(0, (Uint32)200);
+ sf.end();
+ sf.end();
+ sf.end();
+#endif
+ } else {
+ check = pOp->interpret_exit_ok();
+ if( check == -1 ) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+ }
+
+ for(int a = 0; a<pTab->getNoOfColumns(); a++){
+ if((row->attributeStore(a) =
+ pOp->getValue(pTab->getColumn(a)->getName())) == 0) {
+ ERR(pTrans->getNdbError());
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+ }
+
+ check = pTrans->executeScan();
+ if( check == -1 ) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+
+ if (headers)
+ row->header(ndbout) << endl;
+
+ int eof;
+ int rows = 0;
+ eof = pTrans->nextScanResult();
+
+ while(eof == 0){
+ rows++;
+
+ if (useHexFormat) {
+ ndbout.setHexFormat(1) << (*row) << endl;
+ } else {
+ ndbout << (*row) << endl;
+ }
+
+ eof = pTrans->nextScanResult();
+ }
+ if (eof == -1) {
+ const NdbError err = pTrans->getNdbError();
+
+ if (err.status == NdbError::TemporaryError){
+ pNdb->closeTransaction(pTrans);
+ NdbSleep_MilliSleep(50);
+ retryAttempt++;
+ continue;
+ }
+ ERR(err);
+ pNdb->closeTransaction(pTrans);
+ return -1;
+ }
+
+ pNdb->closeTransaction(pTrans);
+
+ ndbout << rows << " rows returned" << endl;
+
+ return 0;
+ }
+ return -1;
+}
diff --git a/ndb/tools/select_count/Makefile b/ndb/tools/select_count/Makefile
new file mode 100644
index 00000000000..35a53c6b046
--- /dev/null
+++ b/ndb/tools/select_count/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := select_count
+
+SOURCES := select_count.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/tools/select_count/select_count.cpp b/ndb/tools/select_count/select_count.cpp
new file mode 100644
index 00000000000..4d281b9bdd5
--- /dev/null
+++ b/ndb/tools/select_count/select_count.cpp
@@ -0,0 +1,90 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+#include <UtilTransactions.hpp>
+
+
+int main(int argc, const char** argv){
+ const char* _dbname = "TEST_DB";
+ int _parallelism = 240;
+ int _help = 0;
+ int _lock = 0;
+
+ struct getargs args[] = {
+ { "database", 'd', arg_string, &_dbname, "dbname",
+ "Name of database table is in"},
+ { "parallelism", 's', arg_integer, &_parallelism, "parallelism", "parallelism" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" },
+ { "lock", 'l', arg_integer, &_lock,
+ "Read(0), Read-hold(1), Exclusive(2)", "lock"}
+
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname1 ... tabnameN\n"\
+ "This program will count the number of records in tables\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+
+ // Connect to Ndb
+ Ndb MyNdb(_dbname);
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ // Connect to Ndb and wait for it to become ready
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ for(int i = optind; i<argc; i++){
+ // Check if table exists in db
+ const NdbDictionary::Table * pTab = NDBT_Table::discoverTableFromDb(&MyNdb, argv[i]);
+ if(pTab == NULL){
+ ndbout << " Table " << argv[i] << " does not exist!" << endl;
+ continue;
+ }
+
+ int rows = 0;
+ UtilTransactions utilTrans(*pTab);
+ if (utilTrans.selectCount(&MyNdb, _parallelism, &rows,
+ (UtilTransactions::ScanLock)_lock) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ ndbout << rows << " records in table " << argv[i] << endl;
+ }
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
+
+
diff --git a/ndb/tools/src/counterviewer/CounterViewer.java b/ndb/tools/src/counterviewer/CounterViewer.java
new file mode 100644
index 00000000000..317c1c75e28
--- /dev/null
+++ b/ndb/tools/src/counterviewer/CounterViewer.java
@@ -0,0 +1,725 @@
+
+import java.awt.*;
+import java.awt.event.*;
+import java.util.*;
+import java.io.*;
+import java.net.*;
+import javax.swing.*;
+
+class Node extends Observable {
+ public final static int UNDEFINED = -1;
+ public final static int NDB_NODE = 0;
+ public final static int MGM_NODE = 1;
+ public final static int API_NODE = 2;
+
+ public int getNodeType() { return m_nodeType;}
+ public static int getNodeType(String str) {
+ if(str.equals("NDB"))
+ return NDB_NODE;
+ if(str.equals("API"))
+ return API_NODE;
+ if(str.equals("MGM"))
+ return MGM_NODE;
+ return UNDEFINED;
+ }
+
+ protected int m_nodeType;
+}
+
+class NdbNode extends Node {
+ public NdbNode(){
+ m_nodeType = NDB_NODE;
+ }
+
+ public Counters getCounters() { return counters; }
+
+ public void setCounters(Counters _c) {
+
+ if(_c == null){
+ counters = null;
+ setChanged();
+ notifyObservers();
+ return;
+ }
+
+ int old_tps = 0;
+ int old_ops = 0;
+ int old_aps = 0;
+ int diff = 5;
+ if(counters != null){
+ old_tps = counters.tps;
+ old_ops = counters.ops;
+ old_aps = counters.aps;
+ diff = 5; //_c.epochsecs - counters.epochsecs;
+ }
+
+ switch(_c.type){
+ case Counters.TRANSACTIONS:
+ _c.tps = (_c.transactions -_c.aborts)/ diff;
+ _c.aps = _c.aborts / diff;
+ _c.ops = old_ops;
+ break;
+ case Counters.OPERATIONS:
+ _c.tps = old_tps;
+ _c.aps = old_aps;
+ _c.ops = _c.operations / diff;
+ break;
+ }
+
+ counters = _c;
+ setChanged();
+ notifyObservers();
+ }
+
+ public int getNodeState(){
+ return nodeState;
+ }
+
+ public static int getNodeState(String state){
+ if(state.equals("NOT_STARTED") ||
+ state.equals("NO_CONTACT"))
+ return 0;
+ return 1;
+ }
+
+ public void setState(int nodeState){
+ this.nodeState = nodeState;
+ if(nodeState == 0)
+ counters = null;
+ }
+
+ private int nodeState;
+ private Counters counters;
+}
+
+class MgmNode extends Node { public MgmNode(){ m_nodeType = MGM_NODE; } }
+class ApiNode extends Node { public ApiNode(){ m_nodeType = API_NODE; } }
+
+class Counters {
+
+ public static final int TRANSACTIONS = 0;
+ public static final int OPERATIONS = 1;
+
+ public Counters(){
+ transactions = operations = -1;
+ }
+
+ public int type;
+ public int transactions;
+ public int operations;
+ public int aborts;
+ public int tps;
+ public int ops;
+ public int aps;
+ public int epochsecs;
+
+ public String toString(){
+ return "[Counters"+
+ " transactions="+transactions+
+ " operations="+operations+
+ " tps="+tps+
+ " ops="+ops+
+ " ]";
+ }
+}
+
+class NdbCluster extends Observable {
+
+ NdbCluster(int node_types[], int num_nodes, int maxNodeId){
+
+ nodes = new Node[maxNodeId+1];
+ maxCounters = new Counters();
+
+ for(int i = 0; i<maxNodeId; i++)
+ nodes[i] = null;
+
+ for(int i = 1; i<num_nodes; i++)
+ switch(node_types[i]){
+ case Node.NDB_NODE:
+ nodes[i] = new NdbNode();
+ break;
+ case Node.API_NODE:
+ nodes[i] = new ApiNode();
+ break;
+ case Node.MGM_NODE:
+ nodes[i] = new MgmNode();
+ default:
+ }
+ }
+
+ public int getNoOfNdbNodes(){
+ if(nodes == null)
+ return 0;
+ int retVal = 0;
+ for(int i = 1; i<nodes.length; i++)
+ if(getNodeType(i) == Node.NDB_NODE)
+ retVal++;
+ return retVal;
+ }
+
+ public int getNodeType(int nodeId){
+ if(nodes == null || nodeId > nodes.length || nodes[nodeId] == null)
+ return Node.UNDEFINED;
+ return nodes[nodeId].getNodeType();
+ }
+
+ public Counters getCounters(int nodeId){
+ if(nodes == null || nodeId > nodes.length || nodes[nodeId] == null ||
+ nodes[nodeId].getNodeType() != Node.NDB_NODE)
+ return null;
+ return ((NdbNode)nodes[nodeId]).getCounters();
+ }
+
+ public void setCounters(int nodeId, Counters _c){
+ if(nodes == null || nodeId > nodes.length || nodes[nodeId] == null)
+ return;
+ ((NdbNode)nodes[nodeId]).setCounters(_c);
+
+ int maxSum = 0;
+ for(int i = 1; i<nodes.length; i++){
+ Counters c = getCounters(i);
+ if(c != null){
+ int sum = c.tps + c.ops + c.aps;
+ if(sum > maxSum){
+ maxCounters = c;
+ maxSum = sum;
+ }
+ }
+ }
+ setChanged();
+ notifyObservers();
+ }
+
+ public void setState(int nodeId, int nodeType, int nodeState){
+ if(nodes == null || nodeId > nodes.length || nodes[nodeId] == null ||
+ nodes[nodeId].getNodeType() != nodeType)
+ return;
+
+ if(nodeType != Node.NDB_NODE)
+ return;
+
+ ((NdbNode)nodes[nodeId]).setState(nodeState);
+ }
+
+ public void setNoConnection(){
+ for(int i = 1; i<nodes.length; i++){
+ Counters c = getCounters(i);
+ if(c != null){
+ setCounters(i, null);
+ }
+ }
+ }
+
+ public Counters getMaxCounters(){
+ return maxCounters;
+ }
+
+ private Node nodes[];
+ private Counters maxCounters;
+}
+
+class CountersPanel extends JPanel implements Observer
+{
+ public CountersPanel(Dimension dim, NdbCluster _cluster, int maxInRow)
+ {
+ cluster = _cluster;
+ cluster.addObserver(this);
+ maxRow = maxInRow;
+ reSize(dim);
+ }
+
+ private void showCounters(Graphics g, int node, int x, int y, boolean p)
+ {
+ Counters counters = cluster.getCounters(node);
+
+ if (counters == null || p){
+ // Mark processor as not available
+ g.setColor(Color.black);
+ g.fillRect(x, y, width, height);
+ } else {
+ int red = (counters.aps * height) / scale;
+ int green = (counters.tps * height) / scale;
+ int yellow = (counters.ops * height) / scale;
+
+ System.out.println("tps="+counters.tps+" ops="+counters.ops+" scale="+scale+" green="+green+" yellow="+yellow);
+
+ g.setColor(Color.white);
+ g.fillRect(x, y, width, height);
+ if (yellow + green + red <= height){ // Max 100% load
+ int yellow_scaled = yellow;
+ int green_scaled = green;
+ int red_scaled = red;
+ if (red_scaled > 0){
+ g.setColor(Color.red);
+ g.fillRect(x,
+ height + y - red_scaled,
+ width,
+ red_scaled);
+ }
+ if (green_scaled > 0){
+ g.setColor(Color.green);
+ g.fillRect(x,
+ height + y - red_scaled - green_scaled,
+ width,
+ green_scaled);
+ }
+ if (yellow_scaled > 0){
+ g.setColor(Color.yellow);
+ g.fillRect(x,
+ height + y - red_scaled - green_scaled - yellow_scaled,
+ width,
+ yellow_scaled);
+ }
+ }
+ // Draw box
+ g.setColor(Color.black);
+ g.drawRect(x, y, width, height);
+
+ float f = ((float)height)/((float)(lines+1));
+
+ for(int i = 0; i<lines; i++){
+ int ytmp = (int)(y+height-(i+1)*f);
+ g.drawLine(x, ytmp, x+width, ytmp);
+ }
+ }
+ }
+
+ public void paintComponent(Graphics g)
+ {
+ super.paintComponent(g);
+
+ Counters maxCounters = cluster.getMaxCounters();
+ final int sum = maxCounters.tps+maxCounters.ops+maxCounters.aps;
+ boolean skipDraw = false;
+ if(sum == 0){
+ skipDraw = true;
+ } else {
+ lines = (sum / 1000) + 1;
+ scale = (lines+1) * 1000;
+ }
+
+ int nodeId = 0;
+ int nodeNo = 0;
+ final int noOfNdbNodes = cluster.getNoOfNdbNodes();
+ for(int row = 0; row <= noOfNdbNodes / maxRow; row++) {
+ int end = Math.min(noOfNdbNodes, (row + 1) * maxRow);
+ for (int pos = 0; (nodeNo < noOfNdbNodes) && (pos < maxRow); pos++){
+ while(cluster.getNodeType(nodeId) != Node.NDB_NODE)
+ nodeId++;
+ showCounters(g,
+ nodeId,
+ xindent + (xgap + width) * pos,
+ yindent + row * (height + ygap),
+ skipDraw);
+ nodeNo++;
+ nodeId++;
+ }
+ }
+ }
+
+ public void setWidth(int maxInRow)
+ {
+ maxRow = maxInRow;
+ }
+
+ public void reSize(Dimension dim)
+ {
+ final int noOfNdbNodes = cluster.getNoOfNdbNodes();
+
+ // System.out.println(dim);
+ int noRows = (int) Math.ceil((double) noOfNdbNodes / (double) maxRow);
+ xgap = (noOfNdbNodes > 1) ? Math.max(2, dim.width / 50) : 0;
+ ygap = (noOfNdbNodes > 1) ? Math.max(2, dim.height / 20) : 0;
+ xindent = 10;
+ yindent = 10;
+ int heightOfScroll = 20;
+ Insets insets = getInsets();
+ width = (dim.width - (insets.left + insets.right) - 2*xindent + xgap)/maxRow - xgap;
+ height = (dim.height - (insets.top + insets.bottom) - 2*yindent + ygap - heightOfScroll)/noRows - ygap;
+ }
+
+
+ public void update(Observable o, Object arg){
+ repaint();
+ }
+
+ private int width, height, maxRow, xgap, ygap, xindent, yindent;
+ private int scale;
+ private int lines;
+ private NdbCluster cluster;
+}
+
+class CountersFrame extends JFrame
+ implements ComponentListener, AdjustmentListener
+{
+
+ public CountersFrame(NdbCluster cluster)
+ {
+ setTitle("CountersViewer");
+
+ final int noOfNdbNodes = cluster.getNoOfNdbNodes();
+
+ processorWidth = Math.min(noOfNdbNodes, 10);
+ setSize(Math.max(50, processorWidth*20),
+ Math.max(100, 200*noOfNdbNodes/processorWidth));
+ JPanel p = new JPanel();
+ addComponentListener(this);
+ p.addComponentListener(this);
+ getContentPane().add(p, "Center");
+ myPanel = new CountersPanel(getContentPane().getSize(),
+ cluster,
+ processorWidth);
+ getContentPane().add(myPanel, "Center");
+ JPanel labelAndScroll = new JPanel();
+ labelAndScroll.setLayout(new GridLayout (1, 2));
+ myWidthLabel = new JLabel("Width: " + processorWidth);
+ labelAndScroll.add(myWidthLabel);
+ myWidthScroll = new JScrollBar(Adjustable.HORIZONTAL,
+ Math.min(noOfNdbNodes, 10), 0, 1,
+ noOfNdbNodes);
+ myWidthScroll.addAdjustmentListener(this);
+ labelAndScroll.add(myWidthScroll);
+ if (noOfNdbNodes > 1)
+ getContentPane().add(labelAndScroll, "South");
+ }
+
+ public void componentHidden(ComponentEvent e) {
+ }
+
+ public void componentMoved(ComponentEvent e) {
+ }
+
+ public void componentResized(ComponentEvent e) {
+ myPanel.reSize(getContentPane().getSize());
+ repaint();
+ }
+
+ public void componentShown(ComponentEvent e) {
+ }
+
+ public void adjustmentValueChanged(AdjustmentEvent evt)
+ {
+ myPanel.setWidth(myWidthScroll.getValue());
+ myPanel.reSize(getContentPane().getSize());
+ myWidthLabel.setText("Width: " + myWidthScroll.getValue());
+ repaint();
+ }
+
+ private JScrollBar myWidthScroll;
+ private JLabel myWidthLabel;
+ private CountersPanel myPanel;
+ private int processorWidth = 10;
+}
+
+class CountersConnection {
+
+ public CountersConnection(String host, int port){
+ this.host = host;
+ this.port = port;
+ }
+
+ public boolean connect(){
+ if(br == null){
+ try {
+ InetAddress target = InetAddress.getByName(host);
+ sock = new Socket(target, port);
+
+ br = new BufferedReader(new InputStreamReader
+ (sock.getInputStream()));
+ } catch (Exception e){
+ System.out.println("connect: " + e);
+ }
+ }
+
+ if (br == null)
+ return false;
+ return true;
+ }
+
+ public void disconnect(){
+ try {
+ sock.close();
+ } catch (Exception e){
+ System.out.println("disconnect: " + e);
+ }
+ sock = null;
+ br = null;
+ }
+
+ public boolean readCounters(NdbCluster cluster) {
+ if(!connect()){
+ cluster.setNoConnection();
+ return false;
+ }
+ String str = null;
+
+ try {
+ str = br.readLine();
+ } catch (Exception e){
+ System.out.println("readLine: " + e);
+ }
+ if(str == null){
+ disconnect();
+ return false;
+ }
+ StringTokenizer st = new StringTokenizer(str, " ");
+
+ int nodeId = 0;
+ Counters c = new Counters();
+
+ while(st.hasMoreTokens()){
+ String tmp = st.nextToken();
+ int ind = 0;
+ if(tmp.startsWith("nodeid=")){
+ nodeId = Integer.valueOf(tmp.substring(7)).intValue();
+ } else if(tmp.startsWith("trans=")){
+ c.transactions = Integer.valueOf(tmp.substring(6)).intValue();
+ c.type = Counters.TRANSACTIONS;
+ } else if(tmp.startsWith("abort=")){
+ c.aborts = Integer.valueOf(tmp.substring(6)).intValue();
+ c.type = Counters.TRANSACTIONS;
+ } else if(tmp.startsWith("epochsecs=")){
+ c.epochsecs = Integer.valueOf(tmp.substring(11)).intValue();
+ } else if(tmp.startsWith("operations=")){
+ c.operations = Integer.valueOf(tmp.substring(11)).intValue();
+ c.type = Counters.OPERATIONS;
+ }
+ }
+
+ if(nodeId != 0)
+ cluster.setCounters(nodeId, c);
+
+ return true;
+ }
+
+ private Socket sock;
+ private BufferedReader br;
+ private String host;
+ private int port;
+}
+
+class MgmConnection {
+
+ public MgmConnection(String host, int port){
+ this.host = host;
+ this.port = port;
+ }
+
+ public NdbCluster getClusterInfo(){
+ NdbCluster cluster = null;
+ if(!connect())
+ return cluster;
+
+ out.println("get info cluster");
+ String str = null;
+ try {
+ str = br.readLine();
+ if(str.startsWith("GET INFO 0")){
+ StringTokenizer st = new StringTokenizer
+ (str.substring(11));
+ int nodes[] = new int[255];
+
+ int type = Node.UNDEFINED;
+ int num_nodes = 0;
+ int maxNodeId = 0;
+ while(st.hasMoreTokens()){
+ String tmp = st.nextToken();
+ final int t = Node.getNodeType(tmp);
+ if(t != Node.UNDEFINED)
+ type = t;
+
+ int nodeId = 0;
+ try {
+ nodeId = Integer.parseInt(tmp);
+ } catch (Exception e) {}
+ if(nodeId != 0){
+ num_nodes ++;
+ nodes[nodeId] = type;
+ if(nodeId > maxNodeId)
+ maxNodeId = nodeId;
+ }
+ }
+ cluster = new NdbCluster(nodes, num_nodes,
+ maxNodeId);
+ }
+
+ } catch(Exception e){
+ System.out.println("readLine: "+e);
+ }
+ return cluster;
+ }
+
+ public boolean connect(){
+ if(br == null || out == null){
+ try {
+ InetAddress target = InetAddress.getByName(host);
+ sock = new Socket(target, port);
+
+ br = new BufferedReader(new InputStreamReader
+ (sock.getInputStream()));
+ out = new PrintWriter(sock.getOutputStream(), true);
+ } catch (Exception e){
+ System.out.println("connect: " + e);
+ }
+ }
+
+ if (br == null || out == null)
+ return false;
+ return true;
+ }
+
+ public void disconnect(){
+ try {
+ sock.close();
+ } catch (Exception e){
+ System.out.println("disconnect: " + e);
+ }
+ sock = null;
+ br = null;
+ out = null;
+ }
+
+ public boolean readStatus(NdbCluster cluster){
+
+ if(!connect()){
+ cluster.setNoConnection();
+ return false;
+ }
+
+ String str = null;
+ try {
+ out.println("get status");
+ str = br.readLine();
+ } catch (Exception e){
+ System.out.println("readLine: " + e);
+ }
+ if(str == null){
+ disconnect();
+ return false;
+ }
+
+ if(!str.startsWith("GET STATUS")){
+ disconnect();
+ return false;
+ }
+
+ int nodes = 0;
+ try {
+ nodes = Integer.parseInt(str.substring(11));
+ } catch(Exception e){
+ System.out.println("parseInt "+e);
+ }
+ if(nodes == 0){
+ disconnect();
+ return false;
+ }
+
+ try {
+ for(; nodes > 0 ; nodes --){
+ str = br.readLine();
+ StringTokenizer st = new StringTokenizer(str);
+
+ String s_nodeId = st.nextToken();
+ final int nodeId = Integer.parseInt(s_nodeId);
+
+ String s_type = st.nextToken();
+ String s_state = st.nextToken();
+ String s_phase = st.nextToken();
+ int type = Node.getNodeType(s_type);
+ int state = NdbNode.getNodeState(s_state);
+
+ cluster.setState(nodeId, type, state);
+ }
+ } catch (Exception e){
+ disconnect();
+ return false;
+ }
+
+ return true;
+ }
+
+ public int getStatisticsPort(){
+ if(!connect())
+ return -1;
+
+ String str = null;
+ try {
+ out.println("stat port");
+ str = br.readLine();
+ } catch (Exception e){
+ System.out.println("readLine: " + e);
+ }
+ if(str == null){
+ disconnect();
+ return -1;
+ }
+
+ if(!str.startsWith("STAT PORT 0")){
+ disconnect();
+ return -1;
+ }
+
+ try {
+ return Integer.parseInt(str.substring(12));
+ } catch (Exception e){
+ System.out.println("parseInt "+e);
+ }
+ return -1;
+ }
+
+ private Socket sock;
+ private BufferedReader br;
+ private PrintWriter out;
+ private String host;
+ private int port;
+}
+
+class CounterViewer {
+
+ public static void usage(){
+ System.out.println("java CounterViewer <mgm host> <mgm port>");
+ }
+
+ public static void main(String args[]){
+ try {
+ String host = args[0];
+ int port = Integer.parseInt(args[1]);
+ new CounterViewer(host, port).run();
+ return;
+ } catch (Exception e){
+ }
+ usage();
+ }
+
+ MgmConnection mgmConnection;
+ CountersConnection countersConnection;
+
+ NdbCluster cluster;
+ boolean ok;
+
+ CounterViewer(String host, int port){
+ ok = false;
+
+ mgmConnection = new MgmConnection(host, port);
+ int statPort = mgmConnection.getStatisticsPort();
+ if(statPort < 0)
+ return;
+
+ countersConnection = new CountersConnection(host, statPort);
+ cluster = mgmConnection.getClusterInfo();
+
+ CountersFrame f = new CountersFrame(cluster);
+ f.setSize (300, 200);
+ f.show();
+
+ ok = true;
+ }
+
+ void run(){
+ while(ok){
+ mgmConnection.readStatus(cluster);
+ countersConnection.readCounters(cluster);
+ }
+ }
+}
+
diff --git a/ndb/tools/transproxy/Makefile b/ndb/tools/transproxy/Makefile
new file mode 100644
index 00000000000..d6a76ed2e3d
--- /dev/null
+++ b/ndb/tools/transproxy/Makefile
@@ -0,0 +1,29 @@
+include .defs.mk
+
+TYPE =
+
+BIN_TARGET = transproxy
+
+SOURCES = transproxy.cpp
+
+CCFLAGS_LOC +=\
+ -I$(NDB_TOP)/include/kernel \
+ -I$(NDB_TOP)/include/mgmcommon \
+ -I$(NDB_TOP)/src/common/mgmcommon \
+ -I$(NDB_TOP)/src/mgmsrv
+
+LIBS_LOC +=\
+ -L$(NDB_TOP)/lib
+
+LIBS_SPEC +=\
+ $(NDB_TOP)/src/mgmsrv/InitConfigFileParser.o \
+ $(NDB_TOP)/src/mgmsrv/Config.o \
+ $(NDB_TOP)/src/mgmsrv/Container.o \
+ $(NDB_TOP)/src/mgmsrv/Str.o \
+ $(NDB_TOP)/src/mgmsrv/convertStrToInt.o \
+ -lNDB_API \
+ -leventlogger \
+ -llogger \
+ -lportlib
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/tools/transproxy/transproxy.cpp b/ndb/tools/transproxy/transproxy.cpp
new file mode 100644
index 00000000000..4c1308e63e7
--- /dev/null
+++ b/ndb/tools/transproxy/transproxy.cpp
@@ -0,0 +1,369 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+#include <stdlib.h>
+#include <unistd.h>
+#include <stdarg.h>
+#include <string.h>
+#include <stdio.h>
+#include <errno.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+
+#include <NdbTCP.h>
+#include <NdbOut.hpp>
+#include <NdbThread.h>
+#include <NdbSleep.h>
+#include <Properties.hpp>
+#include <LocalConfig.hpp>
+#include <Config.hpp>
+#include <InitConfigFileParser.hpp>
+#include <IPCConfig.hpp>
+
+static void
+fatal(char const* fmt, ...)
+{
+ va_list ap;
+ char buf[200];
+ va_start(ap, fmt);
+ vsnprintf(buf, sizeof(buf), fmt, ap);
+ va_end(ap);
+ ndbout << "FATAL: " << buf << endl;
+ sleep(1);
+ exit(1);
+}
+
+static void
+debug(char const* fmt, ...)
+{
+ va_list ap;
+ char buf[200];
+ va_start(ap, fmt);
+ vsnprintf(buf, sizeof(buf), fmt, ap);
+ va_end(ap);
+ ndbout << buf << endl;
+}
+
+// node
+struct Node {
+ enum Type { MGM = 1, DB = 2, API = 3 };
+ Type type;
+ unsigned id; // node id
+ static Node* list;
+ static unsigned count;
+ static Node* find(unsigned n) {
+ for (unsigned i = 0; i < count; i++) {
+ if (list[i].id == n)
+ return &list[i];
+ }
+ return 0;
+ }
+};
+
+unsigned Node::count = 0;
+Node* Node::list = 0;
+
+struct Copy {
+ int rfd; // read from
+ int wfd; // write to
+ unsigned char* buf;
+ unsigned bufsiz;
+ NdbThread* thread;
+ void run();
+ char info[20];
+};
+
+// connection between nodes 0-server side 1-client side
+// we are client to 0 and server to 1
+struct Conn {
+ Node* node[2]; // the nodes
+ unsigned port; // server port
+ unsigned proxy; // proxy port
+ static unsigned count;
+ static unsigned proxycount;
+ static Conn* list;
+ NdbThread* thread; // thread handling this connection
+ void run(); // run the connection
+ int sockfd[2]; // socket 0-on server side 1-client side
+ void conn0(); // connect to side 0
+ void conn1(); // connect to side 0
+ char info[20];
+ Copy copy[2]; // 0-to-1 and 1-to-0
+};
+
+unsigned Conn::count = 0;
+unsigned Conn::proxycount = 0;
+Conn* Conn::list = 0;
+
+// global data
+static char* hostname = 0;
+static struct sockaddr_in hostaddr;
+static char* localcfgfile = 0;
+static char* initcfgfile = 0;
+static unsigned ownnodeid = 0;
+
+static void
+properr(const Properties* props, const char* name, int i = -1)
+{
+ if (i < 0) {
+ fatal("get %s failed: errno = %d", name, props->getPropertiesErrno());
+ } else {
+ fatal("get %s_%d failed: errno = %d", name, i, props->getPropertiesErrno());
+ }
+}
+
+// read config and load it into our structs
+static void
+getcfg()
+{
+ LocalConfig lcfg;
+ if (! lcfg.read(localcfgfile)) {
+ fatal("read %s failed", localcfgfile);
+ }
+ ownnodeid = lcfg._ownNodeId;
+ debug("ownnodeid = %d", ownnodeid);
+ InitConfigFileParser pars(initcfgfile);
+ Config icfg;
+ if (! pars.getConfig(icfg)) {
+ fatal("parse %s failed", initcfgfile);
+ }
+ Properties* ccfg = icfg.getConfig(ownnodeid);
+ if (ccfg == 0) {
+ const char* err = "unknown error";
+ fatal("getConfig: %s", err);
+ }
+ ccfg->put("NodeId", ownnodeid);
+ ccfg->put("NodeType", "MGM");
+ if (! ccfg->get("NoOfNodes", &Node::count)) {
+ properr(ccfg, "NoOfNodes", -1);
+ }
+ debug("Node::count = %d", Node::count);
+ Node::list = new Node[Node::count];
+ for (unsigned i = 0; i < Node::count; i++) {
+ Node& node = Node::list[i];
+ const Properties* nodecfg;
+ if (! ccfg->get("Node", 1+i, &nodecfg)) {
+ properr(ccfg, "Node", 1+i);
+ }
+ const char* type;
+ if (! nodecfg->get("Type", &type)) {
+ properr(nodecfg, "Type");
+ }
+ if (strcmp(type, "MGM") == 0) {
+ node.type = Node::MGM;
+ } else if (strcmp(type, "DB") == 0) {
+ node.type = Node::DB;
+ } else if (strcmp(type, "API") == 0) {
+ node.type = Node::API;
+ } else {
+ fatal("prop %s_%d bad Type = %s", "Node", 1+i, type);
+ }
+ if (! nodecfg->get("NodeId", &node.id)) {
+ properr(nodecfg, "NodeId");
+ }
+ debug("node id=%d type=%d", node.id, node.type);
+ }
+ IPCConfig ipccfg(ccfg);
+ if (ipccfg.init() != 0) {
+ fatal("ipccfg init failed");
+ }
+ if (! ccfg->get("NoOfConnections", &Conn::count)) {
+ properr(ccfg, "NoOfConnections");
+ }
+ debug("Conn::count = %d", Conn::count);
+ Conn::list = new Conn[Conn::count];
+ for (unsigned i = 0; i < Conn::count; i++) {
+ Conn& conn = Conn::list[i];
+ const Properties* conncfg;
+ if (! ccfg->get("Connection", i, &conncfg)) {
+ properr(ccfg, "Connection", i);
+ }
+ unsigned n;
+ if (! conncfg->get("NodeId1", &n)) {
+ properr(conncfg, "NodeId1");
+ }
+ if ((conn.node[0] = Node::find(n)) == 0) {
+ fatal("node %d not found", n);
+ }
+ if (! conncfg->get("NodeId2", &n)) {
+ properr(conncfg, "NodeId2");
+ }
+ if ((conn.node[1] = Node::find(n)) == 0) {
+ fatal("node %d not found", n);
+ }
+ if (! conncfg->get("PortNumber", &conn.port)) {
+ properr(conncfg, "PortNumber");
+ }
+ conn.proxy = 0;
+ const char* proxy;
+ if (conncfg->get("Proxy", &proxy)) {
+ conn.proxy = atoi(proxy);
+ if (conn.proxy > 0) {
+ Conn::proxycount++;
+ }
+ }
+ sprintf(conn.info, "conn %d-%d", conn.node[0]->id, conn.node[1]->id);
+ }
+}
+
+void
+Conn::conn0()
+{
+ int fd;
+ while (1) {
+ if ((fd = socket(PF_INET, SOCK_STREAM, 0)) == -1) {
+ fatal("%s: create client socket failed: %s", info, strerror(errno));
+ }
+ struct sockaddr_in servaddr;
+ memset(&servaddr, 0, sizeof(servaddr));
+ servaddr.sin_family = AF_INET;
+ servaddr.sin_port = htons(port);
+ servaddr.sin_addr = hostaddr.sin_addr;
+#if 0 // coredump
+ if (Ndb_getInAddr(&servaddr.sin_addr, hostname) != 0) {
+ fatal("%s: hostname %s lookup failed", info, hostname);
+ }
+#endif
+ if (connect(fd, (struct sockaddr*)&servaddr, sizeof(servaddr)) == 0)
+ break;
+ if (errno != ECONNREFUSED) {
+ fatal("%s: connect failed: %s", info, strerror(errno));
+ }
+ close(fd);
+ NdbSleep_MilliSleep(100);
+ }
+ sockfd[0] = fd;
+ debug("%s: side 0 connected", info);
+}
+
+void
+Conn::conn1()
+{
+ int servfd;
+ if ((servfd = socket(PF_INET, SOCK_STREAM, 0)) == -1) {
+ fatal("%s: create server socket failed: %s", info, strerror(errno));
+ }
+ struct sockaddr_in servaddr;
+ memset(&servaddr, 0, sizeof(servaddr));
+ servaddr.sin_family = AF_INET;
+ servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
+ servaddr.sin_port = htons(proxy);
+ const int on = 1;
+ setsockopt(servfd, SOL_SOCKET, SO_REUSEADDR, (const char*)&on, sizeof(on));
+ if (bind(servfd, (struct sockaddr*) &servaddr, sizeof(servaddr)) == -1) {
+ fatal("%s: bind %d failed: %s", info, proxy, strerror(errno));
+ }
+ if (listen(servfd, 1) == -1) {
+ fatal("%s: listen %d failed: %s", info, proxy, strerror(errno));
+ }
+ int fd;
+ if ((fd = accept(servfd, 0, 0)) == -1) {
+ fatal("%s: accept failed: %s", info, strerror(errno));
+ }
+ sockfd[1] = fd;
+ close(servfd);
+ debug("%s: side 1 connected", info);
+}
+
+void
+Copy::run()
+{
+ debug("%s: start", info);
+ int n, m;
+ while (1) {
+ n = read(rfd, buf, sizeof(buf));
+ if (n < 0)
+ fatal("read error: %s", strerror(errno));
+ m = write(wfd, buf, n);
+ if (m != n)
+ fatal("write error: %s", strerror(errno));
+ }
+ debug("%s: stop", info);
+}
+
+extern "C" void*
+copyrun_C(void* copy)
+{
+ ((Copy*) copy)->run();
+ NdbThread_Exit(0);
+ return 0;
+}
+
+void
+Conn::run()
+{
+ debug("%s: start", info);
+ conn1();
+ conn0();
+ const unsigned siz = 32 * 1024;
+ for (int i = 0; i < 2; i++) {
+ Copy& copy = this->copy[i];
+ copy.rfd = sockfd[i];
+ copy.wfd = sockfd[1-i];
+ copy.buf = new unsigned char[siz];
+ copy.bufsiz = siz;
+ sprintf(copy.info, "copy %d-%d", this->node[i]->id, this->node[1-i]->id);
+ copy.thread = NdbThread_Create(copyrun_C, (void**)&copy,
+ 8192, "copyrun", NDB_THREAD_PRIO_LOW);
+ if (copy.thread == 0) {
+ fatal("%s: create thread %d failed errno=%d", i, errno);
+ }
+ }
+ debug("%s: stop", info);
+}
+
+extern "C" void*
+connrun_C(void* conn)
+{
+ ((Conn*) conn)->run();
+ NdbThread_Exit(0);
+ return 0;
+}
+
+static void
+start()
+{
+ NdbThread_SetConcurrencyLevel(3 * Conn::proxycount + 2);
+ for (unsigned i = 0; i < Conn::count; i++) {
+ Conn& conn = Conn::list[i];
+ if (! conn.proxy)
+ continue;
+ conn.thread = NdbThread_Create(connrun_C, (void**)&conn,
+ 8192, "connrun", NDB_THREAD_PRIO_LOW);
+ if (conn.thread == 0) {
+ fatal("create thread %d failed errno=%d", i, errno);
+ }
+ }
+ sleep(3600);
+}
+
+int
+main(int av, char** ac)
+{
+ debug("start");
+ hostname = "ndb-srv7";
+ if (Ndb_getInAddr(&hostaddr.sin_addr, hostname) != 0) {
+ fatal("hostname %s lookup failed", hostname);
+ }
+ localcfgfile = "Ndb.cfg";
+ initcfgfile = "config.txt";
+ getcfg();
+ start();
+ debug("done");
+ return 0;
+}
+
+// vim: set sw=4 noet:
diff --git a/ndb/tools/verify_index/Makefile b/ndb/tools/verify_index/Makefile
new file mode 100644
index 00000000000..f6b31e4dc8e
--- /dev/null
+++ b/ndb/tools/verify_index/Makefile
@@ -0,0 +1,9 @@
+include .defs.mk
+
+TYPE := ndbapitest
+
+BIN_TARGET := verify_index
+
+SOURCES := verify_index.cpp
+
+include $(NDB_TOP)/Epilogue.mk
diff --git a/ndb/tools/verify_index/verify_index.cpp b/ndb/tools/verify_index/verify_index.cpp
new file mode 100644
index 00000000000..324bb11cfe4
--- /dev/null
+++ b/ndb/tools/verify_index/verify_index.cpp
@@ -0,0 +1,86 @@
+/* Copyright (C) 2003 MySQL AB
+
+ 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 */
+
+
+#include <NdbStdio.h>
+#include <stdlib.h>
+#include <NdbOut.hpp>
+
+#include <NdbApi.hpp>
+#include <NdbMain.h>
+#include <NDBT.hpp>
+#include <NdbSleep.h>
+#include <getarg.h>
+#include <UtilTransactions.hpp>
+
+
+int main(int argc, const char** argv){
+ int _parallelism = 240;
+ const char* _tabname = NULL;
+ const char* _indexname = NULL;
+ int _help = 0;
+
+ struct getargs args[] = {
+ { "parallelism", 's', arg_integer, &_parallelism, "parallelism", "parallelism" },
+ { "usage", '?', arg_flag, &_help, "Print help", "" }
+ };
+ int num_args = sizeof(args) / sizeof(args[0]);
+ int optind = 0;
+ char desc[] =
+ "tabname indexname\n"\
+ "This program will verify the index [indexname] and compare it to data\n"
+ "in table [tablename]\n";
+
+ if(getarg(args, num_args, argc, argv, &optind) ||
+ argv[optind] == NULL || argv[optind+1] == NULL || _help) {
+ arg_printusage(args, num_args, argv[0], desc);
+ return NDBT_ProgramExit(NDBT_WRONGARGS);
+ }
+ _tabname = argv[optind];
+ _indexname = argv[optind+1];
+
+ // Connect to Ndb
+ Ndb MyNdb( "TEST_DB" );
+
+ if(MyNdb.init() != 0){
+ ERR(MyNdb.getNdbError());
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ // Connect to Ndb and wait for it to become ready
+ while(MyNdb.waitUntilReady() != 0)
+ ndbout << "Waiting for ndb to become ready..." << endl;
+
+ // Check if table exists in db
+ const NdbDictionary::Table * pTab = NDBT_Table::discoverTableFromDb(&MyNdb, _tabname);
+ if(pTab == NULL){
+ ndbout << " Table " << _tabname << " does not exist!" << endl;
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ int rows = 0;
+ UtilTransactions utilTrans(*pTab);
+ if (utilTrans.verifyIndex(&MyNdb,
+ _indexname,
+ _parallelism) != 0){
+ return NDBT_ProgramExit(NDBT_FAILED);
+ }
+
+ return NDBT_ProgramExit(NDBT_OK);
+}
+
+
+